use crate::template_utils::Ructe; use activitypub::object::Note; use rocket::{ request::LenientForm, response::{Flash, Redirect}, }; use validator::Validate; use std::time::Duration; use crate::routes::errors::ErrorPage; use crate::template_utils::IntoContext; use plume_common::{ activity_pub::{broadcast, ActivityStream, ApRequest}, utils, }; use plume_models::{ blogs::Blog, comments::*, db_conn::DbConn, inbox::inbox, instance::Instance, medias::Media, mentions::Mention, posts::Post, safe_string::SafeString, tags::Tag, users::User, Error, PlumeRocket, CONFIG, }; #[derive(Default, FromForm, Debug, Validate)] pub struct NewCommentForm { pub responding_to: Option, #[validate(length(min = 1, message = "Your comment can't be empty"))] pub content: String, pub warning: String, } #[post("/~///comment", data = "
")] pub fn create( blog_name: String, slug: String, form: LenientForm, user: User, conn: DbConn, rockets: PlumeRocket, ) -> Result, Ructe> { let blog = Blog::find_by_fqn(&conn, &blog_name).expect("comments::create: blog error"); let post = Post::find_by_slug(&conn, &slug, blog.id).expect("comments::create: post error"); form.validate() .map(|_| { let (html, mentions, _hashtags) = utils::md_to_html( form.content.as_ref(), Some( &Instance::get_local() .expect("comments::create: local instance error") .public_domain, ), true, Some(Media::get_media_processor(&conn, vec![&user])), ); let comm = Comment::insert( &conn, NewComment { content: SafeString::new(html.as_ref()), in_response_to_id: form.responding_to, post_id: post.id, author_id: user.id, ap_url: None, sensitive: !form.warning.is_empty(), spoiler_text: form.warning.clone(), public_visibility: true, }, ) .expect("comments::create: insert error"); let new_comment = comm .create_activity(&conn) .expect("comments::create: activity error"); // save mentions for ment in mentions { Mention::from_activity( &conn, &Mention::build_activity(&conn, &ment) .expect("comments::create: build mention error"), comm.id, false, true, ) .expect("comments::create: mention save error"); } comm.notify(&conn).expect("comments::create: notify error"); // federate let dest = User::one_by_instance(&conn).expect("comments::create: dest error"); let user_clone = user.clone(); rockets.worker.execute(move || { broadcast(&user_clone, new_comment, dest, CONFIG.proxy().cloned()) }); Flash::success( Redirect::to(uri!( super::posts::details: blog = blog_name, slug = slug, responding_to = _ )), i18n!(&rockets.intl.catalog, "Your comment has been posted."), ) }) .map_err(|errors| { // TODO: de-duplicate this code let comments = CommentTree::from_post(&conn, &post, Some(&user)) .expect("comments::create: comments error"); let previous = form.responding_to.and_then(|r| Comment::get(&conn, r).ok()); render!(posts::details( &(&conn, &rockets).to_context(), post.clone(), blog, &*form, errors, Tag::for_post(&conn, post.id).expect("comments::create: tags error"), comments, previous, post.count_likes(&conn) .expect("comments::create: count likes error"), post.count_reshares(&conn) .expect("comments::create: count reshares error"), user.has_liked(&conn, &post) .expect("comments::create: liked error"), user.has_reshared(&conn, &post) .expect("comments::create: reshared error"), user.is_following( &*conn, post.get_authors(&conn) .expect("comments::create: authors error")[0] .id ) .expect("comments::create: following error"), post.get_authors(&conn) .expect("comments::create: authors error")[0] .clone() )) }) } #[post("/~///comment//delete")] pub fn delete( blog: String, slug: String, id: i32, user: User, conn: DbConn, rockets: PlumeRocket, ) -> Result, ErrorPage> { if let Ok(comment) = Comment::get(&conn, id) { if comment.author_id == user.id { let dest = User::one_by_instance(&conn)?; let delete_activity = comment.build_delete(&conn)?; inbox( &conn, serde_json::to_value(&delete_activity).map_err(Error::from)?, )?; let user_c = user.clone(); rockets.worker.execute(move || { broadcast(&user_c, delete_activity, dest, CONFIG.proxy().cloned()) }); rockets .worker .execute_after(Duration::from_secs(10 * 60), move || { user.rotate_keypair(&conn) .expect("Failed to rotate keypair"); }); } } Ok(Flash::success( Redirect::to(uri!( super::posts::details: blog = blog, slug = slug, responding_to = _ )), i18n!(&rockets.intl.catalog, "Your comment has been deleted."), )) } #[get("/~/<_blog>/<_slug>/comment/")] pub fn activity_pub( _blog: String, _slug: String, id: i32, _ap: ApRequest, conn: DbConn, ) -> Option> { Comment::get(&conn, id) .and_then(|c| c.to_activity(&conn)) .ok() .map(ActivityStream::new) }