Plume/src/routes/comments.rs

92 lines
3.4 KiB
Rust
Raw Normal View History

use activitypub::object::Note;
2018-06-04 21:57:03 +02:00
use rocket::{
2018-06-24 18:58:57 +02:00
request::LenientForm,
response::Redirect
2018-06-04 21:57:03 +02:00
};
use rocket_i18n::I18n;
2018-06-29 14:56:00 +02:00
use validator::Validate;
use template_utils::Ructe;
2018-05-10 11:44:57 +02:00
use plume_common::{utils, activity_pub::{broadcast, ApRequest, ActivityStream}};
use plume_models::{
blogs::Blog,
2018-05-19 09:39:59 +02:00
comments::*,
db_conn::DbConn,
mentions::Mention,
2018-05-19 09:39:59 +02:00
posts::Post,
safe_string::SafeString,
tags::Tag,
2018-05-19 09:39:59 +02:00
users::User
};
use Worker;
2018-05-10 11:44:57 +02:00
#[derive(Default, FromForm, Debug, Validate, Serialize)]
pub struct NewCommentForm {
pub responding_to: Option<i32>,
#[validate(length(min = "1", message = "Your comment can't be empty"))]
pub content: String,
pub warning: String,
2018-05-10 11:44:57 +02:00
}
#[post("/~/<blog_name>/<slug>/comment", data = "<form>")]
pub fn create(blog_name: String, slug: String, form: LenientForm<NewCommentForm>, user: User, conn: DbConn, worker: Worker, intl: I18n)
-> Result<Redirect, Option<Ructe>> {
let blog = Blog::find_by_fqn(&*conn, &blog_name).ok_or(None)?;
let post = Post::find_by_slug(&*conn, &slug, blog.id).ok_or(None)?;
2018-07-06 11:51:19 +02:00
form.validate()
.map(|_| {
2018-10-20 16:38:16 +02:00
let (html, mentions, _hashtags) = utils::md_to_html(form.content.as_ref());
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()
}).update_ap_url(&*conn);
let new_comment = comm.create_activity(&*conn);
2018-05-19 00:04:30 +02:00
// save mentions
for ment in mentions {
Mention::from_activity(&*conn, &Mention::build_activity(&*conn, &ment), post.id, true, true);
}
// federate
let dest = User::one_by_instance(&*conn);
2018-07-26 17:51:41 +02:00
let user_clone = user.clone();
worker.execute(move || broadcast(&user_clone, new_comment, dest));
2018-09-09 13:37:20 +02:00
Redirect::to(uri!(super::posts::details: blog = blog_name, slug = slug))
2018-07-06 11:51:19 +02:00
})
.map_err(|errors| {
// TODO: de-duplicate this code
let comments = Comment::list_by_post(&*conn, post.id);
2018-05-10 17:36:32 +02:00
let previous = form.responding_to.map(|r| Comment::get(&*conn, r)
.expect("posts::details_reponse: Error retrieving previous comment"));
Some(render!(posts::details(
&(&*conn, &intl.catalog, Some(user.clone())),
post.clone(),
blog,
&*form,
errors,
Tag::for_post(&*conn, post.id),
comments.into_iter().filter(|c| c.in_response_to_id.is_none()).collect::<Vec<Comment>>(),
previous,
post.get_likes(&*conn).len(),
post.get_reshares(&*conn).len(),
user.has_liked(&*conn, &post),
user.has_reshared(&*conn, &post),
user.is_following(&*conn, post.get_authors(&*conn)[0].id),
post.get_authors(&*conn)[0].clone()
)))
})
2018-05-10 11:44:57 +02:00
}
#[get("/~/<_blog>/<_slug>/comment/<id>")]
pub fn activity_pub(_blog: String, _slug: String, id: i32, _ap: ApRequest, conn: DbConn) -> Option<ActivityStream<Note>> {
Comment::get(&*conn, id).map(|c| ActivityStream::new(c.to_activity(&*conn)))
}