Plume/src/routes/posts.rs

402 lines
16 KiB
Rust
Raw Normal View History

use activitypub::object::Article;
use chrono::Utc;
use heck::{CamelCase, KebabCase};
use rocket::{State, request::LenientForm};
2018-06-04 21:57:03 +02:00
use rocket::response::{Redirect, Flash};
2018-04-23 16:25:39 +02:00
use rocket_contrib::Template;
2018-05-10 11:44:57 +02:00
use serde_json;
use std::{collections::{HashMap, HashSet}, borrow::Cow};
2018-07-06 11:51:19 +02:00
use validator::{Validate, ValidationError, ValidationErrors};
use workerpool::{Pool, thunk::*};
2018-04-23 16:25:39 +02:00
use plume_common::activity_pub::{broadcast, ActivityStream, ApRequest, inbox::Deletable};
use plume_common::utils;
use plume_models::{
2018-05-19 09:39:59 +02:00
blogs::*,
db_conn::DbConn,
2018-05-19 09:39:59 +02:00
comments::Comment,
instance::Instance,
2018-06-20 22:58:11 +02:00
mentions::Mention,
2018-05-19 09:39:59 +02:00
post_authors::*,
posts::*,
safe_string::SafeString,
tags::*,
2018-05-19 09:39:59 +02:00
users::User
};
#[derive(FromForm)]
struct CommentQuery {
responding_to: Option<i32>
}
2018-04-23 16:25:39 +02:00
// See: https://github.com/SergioBenitez/Rocket/pull/454
#[get("/~/<blog>/<slug>", rank = 4)]
2018-05-10 22:31:52 +02:00
fn details(blog: String, slug: String, conn: DbConn, user: Option<User>) -> Template {
details_response(blog, slug, conn, user, None)
}
#[get("/~/<blog>/<slug>?<query>")]
fn details_response(blog: String, slug: String, conn: DbConn, user: Option<User>, query: Option<CommentQuery>) -> Template {
2018-09-03 15:59:02 +02:00
may_fail!(user.map(|u| u.to_json(&*conn)), Blog::find_by_fqn(&*conn, blog), "Couldn't find this blog", |blog| {
may_fail!(user.map(|u| u.to_json(&*conn)), Post::find_by_slug(&*conn, slug, blog.id), "Couldn't find this post", |post| {
if post.published || post.get_authors(&*conn).into_iter().any(|a| a.id == user.clone().map(|u| u.id).unwrap_or(0)) {
let comments = Comment::list_by_post(&*conn, post.id);
let comms = comments.clone();
Template::render("posts/details", json!({
"author": post.get_authors(&*conn)[0].to_json(&*conn),
"article": post.to_json(&*conn),
"blog": blog.to_json(&*conn),
"comments": &comments.into_iter().filter_map(|c| if c.in_response_to_id.is_none() {
Some(c.to_json(&*conn, &comms))
} else {
None
}).collect::<Vec<serde_json::Value>>(),
"n_likes": post.get_likes(&*conn).len(),
"has_liked": user.clone().map(|u| u.has_liked(&*conn, &post)).unwrap_or(false),
"n_reshares": post.get_reshares(&*conn).len(),
"has_reshared": user.clone().map(|u| u.has_reshared(&*conn, &post)).unwrap_or(false),
"account": &user.clone().map(|u| u.to_json(&*conn)),
"date": &post.creation_date.timestamp(),
"previous": query.and_then(|q| q.responding_to.map(|r| Comment::get(&*conn, r)
.expect("posts::details_reponse: Error retrieving previous comment").to_json(&*conn, &vec![]))),
"user_fqn": user.clone().map(|u| u.get_fqn(&*conn)).unwrap_or(String::new()),
"is_author": user.clone().map(|u| post.get_authors(&*conn).into_iter().any(|a| u.id == a.id)).unwrap_or(false),
"is_following": user.map(|u| u.is_following(&*conn, post.get_authors(&*conn)[0].id)).unwrap_or(false)
}))
} else {
Template::render("errors/403", json!({
"error_message": "This post isn't published yet."
}))
}
})
})
2018-04-23 16:25:39 +02:00
}
#[get("/~/<blog>/<slug>", rank = 3)]
fn activity_details(blog: String, slug: String, conn: DbConn, _ap: ApRequest) -> Result<ActivityStream<Article>, Option<String>> {
let blog = Blog::find_by_fqn(&*conn, blog).ok_or(None)?;
let post = Post::find_by_slug(&*conn, slug, blog.id).ok_or(None)?;
if post.published {
Ok(ActivityStream::new(post.into_activity(&*conn)))
} else {
Err(Some(String::from("Not published yet.")))
}
2018-04-23 16:25:39 +02:00
}
2018-06-04 21:57:03 +02:00
#[get("/~/<blog>/new", rank = 2)]
fn new_auth(blog: String) -> Flash<Redirect> {
utils::requires_login(
"You need to be logged in order to write a new post",
uri!(new: blog = blog).into()
)
2018-04-23 16:25:39 +02:00
}
#[get("/~/<blog>/new", rank = 1)]
fn new(blog: String, user: User, conn: DbConn) -> Option<Template> {
let b = Blog::find_by_fqn(&*conn, blog.to_string())?;
if !user.is_author_in(&*conn, b.clone()) {
Some(Template::render("errors/403", json!({// TODO actually return 403 error code
"error_message": "You are not author in this blog."
})))
} else {
Some(Template::render("posts/new", json!({
2018-09-03 15:59:02 +02:00
"account": user.to_json(&*conn),
"instance": Instance::get_local(&*conn),
2018-09-06 23:39:22 +02:00
"editing": false,
"errors": null,
"form": null,
"is_draft": true,
})))
}
}
2018-09-06 23:39:22 +02:00
#[get("/~/<blog>/<slug>/edit")]
fn edit(blog: String, slug: String, user: User, conn: DbConn) -> Option<Template> {
let b = Blog::find_by_fqn(&*conn, blog.to_string())?;
let post = Post::find_by_slug(&*conn, slug, b.id)?;
2018-09-06 23:39:22 +02:00
if !user.is_author_in(&*conn, b) {
Some(Template::render("errors/403", json!({// TODO actually return 403 error code
2018-09-06 23:39:22 +02:00
"error_message": "You are not author in this blog."
})))
2018-09-06 23:39:22 +02:00
} else {
let source = if post.source.len() > 0 {
post.source
} else {
post.content.get().clone() // fallback to HTML if the markdown was not stored
};
Some(Template::render("posts/new", json!({
2018-09-06 23:39:22 +02:00
"account": user.to_json(&*conn),
"instance": Instance::get_local(&*conn),
"editing": true,
"errors": null,
"form": NewPostForm {
title: post.title.clone(),
subtitle: post.subtitle.clone(),
content: source,
2018-09-06 23:39:22 +02:00
tags: Tag::for_post(&*conn, post.id)
.into_iter()
.filter_map(|t| if !t.is_hashtag {Some(t.tag)} else {None})
2018-09-06 23:39:22 +02:00
.collect::<Vec<String>>()
.join(", "),
license: post.license.clone(),
draft: true,
},
"is_draft": !post.published
})))
2018-09-06 23:39:22 +02:00
}
}
#[post("/~/<blog>/<slug>/edit", data = "<data>")]
fn update(blog: String, slug: String, user: User, conn: DbConn, data: LenientForm<NewPostForm>, worker: State<Pool<ThunkWorker<()>>>)
-> Result<Redirect, Option<Template>> {
let b = Blog::find_by_fqn(&*conn, blog.to_string()).ok_or(None)?;
let mut post = Post::find_by_slug(&*conn, slug.clone(), b.id).ok_or(None)?;
2018-09-06 23:39:22 +02:00
let form = data.get();
let new_slug = if !post.published {
form.title.to_string().to_kebab_case()
} else {
post.slug
};
2018-09-06 23:39:22 +02:00
let mut errors = match form.validate() {
Ok(_) => ValidationErrors::new(),
Err(e) => e
};
2018-09-07 19:51:53 +02:00
if new_slug != slug {
if let Some(_) = Post::find_by_slug(&*conn, new_slug.clone(), b.id) {
2018-09-07 19:51:53 +02:00
errors.add("title", ValidationError {
code: Cow::from("existing_slug"),
message: Some(Cow::from("A post with the same title already exists.")),
params: HashMap::new()
});
}
2018-09-06 23:39:22 +02:00
}
if errors.is_empty() {
if !user.is_author_in(&*conn, b) {
2018-09-06 23:39:22 +02:00
// actually it's not "Ok"…
Ok(Redirect::to(uri!(super::blogs::details: name = blog)))
} else {
let (content, mentions, hashtags) = utils::md_to_html(form.content.to_string().as_ref());
2018-09-06 23:39:22 +02:00
let license = if form.license.len() > 0 {
form.license.to_string()
} else {
Instance::get_local(&*conn).map(|i| i.default_license).unwrap_or(String::from("CC-BY-SA"))
2018-09-06 23:39:22 +02:00
};
// update publication date if when this article is no longer a draft
let newly_published = if !post.published && !form.draft {
post.published = true;
post.creation_date = Utc::now().naive_utc();
true
} else {
false
};
2018-09-06 23:39:22 +02:00
post.slug = new_slug.clone();
post.title = form.title.clone();
post.subtitle = form.subtitle.clone();
post.content = SafeString::new(&content);
post.source = form.content.clone();
post.license = license;
post.update(&*conn);
let post = post.update_ap_url(&*conn);
if post.published {
2018-10-28 10:05:02 +01:00
post.update_mentions(&conn, mentions.into_iter().map(|m| Mention::build_activity(&conn, m)).collect());
2018-09-06 23:39:22 +02:00
}
2018-10-20 14:05:41 +02:00
let old_tags = Tag::for_post(&*conn, post.id).into_iter().collect::<Vec<_>>();
let old_non_hashtags = old_tags.iter().filter_map(|tag| if !tag.is_hashtag {Some(tag.tag.clone())} else {None}).collect();
let old_hashtags = old_tags.iter().filter_map(|tag| if tag.is_hashtag {Some(tag.tag.clone())} else {None}).collect();
let tags = form.tags.split(",").map(|t| t.trim().to_camel_case()).filter(|t| t.len() > 0).collect::<HashSet<_>>();
for tag in tags.difference(&old_non_hashtags) {
Tag::insert(&*conn, NewTag {
tag: tag.clone(),
is_hashtag: false,
post_id: post.id
});
2018-10-20 14:05:41 +02:00
}
for ot in old_tags.iter() {
if !tags.contains(&ot.tag) && !ot.is_hashtag {
ot.delete(&conn);
}
}
let hashtags = hashtags.into_iter().map(|h| h.to_camel_case()).collect::<HashSet<_>>();
for hashtag in hashtags.difference(&old_hashtags) {
Tag::insert(&*conn, NewTag {
tag: hashtag.clone(),
is_hashtag: true,
post_id: post.id,
});
}
2018-10-20 14:05:41 +02:00
for ot in old_tags {
if !hashtags.contains(&ot.tag) && ot.is_hashtag {
2018-10-20 14:05:41 +02:00
ot.delete(&conn);
}
2018-09-06 23:39:22 +02:00
}
if post.published {
if newly_published {
let act = post.create_activity(&conn);
let dest = User::one_by_instance(&*conn);
worker.execute(Thunk::of(move || broadcast(&user, act, dest)));
} else {
let act = post.update_activity(&*conn);
let dest = User::one_by_instance(&*conn);
worker.execute(Thunk::of(move || broadcast(&user, act, dest)));
}
}
2018-09-06 23:39:22 +02:00
2018-09-07 19:51:53 +02:00
Ok(Redirect::to(uri!(details: blog = blog, slug = new_slug)))
2018-09-06 23:39:22 +02:00
}
} else {
Err(Some(Template::render("posts/new", json!({
2018-09-06 23:39:22 +02:00
"account": user.to_json(&*conn),
"instance": Instance::get_local(&*conn),
2018-09-07 19:51:53 +02:00
"editing": true,
2018-09-06 23:39:22 +02:00
"errors": errors.inner(),
"form": form,
"is_draft": form.draft,
}))))
2018-09-06 23:39:22 +02:00
}
}
#[derive(FromForm, Validate, Serialize)]
2018-04-23 16:25:39 +02:00
struct NewPostForm {
#[validate(custom(function = "valid_slug", message = "Invalid title"))]
2018-04-23 16:25:39 +02:00
pub title: String,
2018-09-04 13:26:13 +02:00
pub subtitle: String,
2018-04-23 16:25:39 +02:00
pub content: String,
pub tags: String,
pub license: String,
pub draft: bool,
2018-04-23 16:25:39 +02:00
}
2018-06-29 14:56:00 +02:00
fn valid_slug(title: &str) -> Result<(), ValidationError> {
let slug = title.to_string().to_kebab_case();
if slug.len() == 0 {
Err(ValidationError::new("empty_slug"))
2018-07-06 11:51:19 +02:00
} else if slug == "new" {
Err(ValidationError::new("invalid_slug"))
2018-06-29 14:56:00 +02:00
} else {
Ok(())
}
}
2018-04-23 16:25:39 +02:00
#[post("/~/<blog_name>/new", data = "<data>")]
fn create(blog_name: String, data: LenientForm<NewPostForm>, user: User, conn: DbConn, worker: State<Pool<ThunkWorker<()>>>) -> Result<Redirect, Option<Template>> {
let blog = Blog::find_by_fqn(&*conn, blog_name.to_string()).ok_or(None)?;
2018-04-23 16:25:39 +02:00
let form = data.get();
let slug = form.title.to_string().to_kebab_case();
2018-09-03 15:59:02 +02:00
2018-07-06 11:51:19 +02:00
let mut errors = match form.validate() {
Ok(_) => ValidationErrors::new(),
Err(e) => e
};
if let Some(_) = Post::find_by_slug(&*conn, slug.clone(), blog.id) {
errors.add("title", ValidationError {
code: Cow::from("existing_slug"),
message: Some(Cow::from("A post with the same title already exists.")),
params: HashMap::new()
});
2018-07-06 11:51:19 +02:00
}
2018-05-24 12:42:45 +02:00
2018-07-06 11:51:19 +02:00
if errors.is_empty() {
if !user.is_author_in(&*conn, blog.clone()) {
// actually it's not "Ok"…
Ok(Redirect::to(uri!(super::blogs::details: name = blog_name)))
} else {
let (content, mentions, hashtags) = utils::md_to_html(form.content.to_string().as_ref());
2018-05-24 12:42:45 +02:00
let post = Post::insert(&*conn, NewPost {
blog_id: blog.id,
slug: slug.to_string(),
title: form.title.to_string(),
content: SafeString::new(&content),
published: !form.draft,
license: if form.license.len() > 0 {
form.license.to_string()
} else {
Instance::get_local(&*conn).map(|i| i.default_license).unwrap_or(String::from("CC-BY-SA"))
},
ap_url: "".to_string(),
2018-09-04 13:26:13 +02:00
creation_date: None,
subtitle: form.subtitle.clone(),
source: form.content.clone(),
});
2018-06-22 17:17:53 +02:00
let post = post.update_ap_url(&*conn);
PostAuthor::insert(&*conn, NewPostAuthor {
post_id: post.id,
author_id: user.id
});
2018-05-01 17:51:49 +02:00
let tags = form.tags.split(",").map(|t| t.trim().to_camel_case()).filter(|t| t.len() > 0).collect::<HashSet<_>>();
for tag in tags {
Tag::insert(&*conn, NewTag {
tag: tag,
is_hashtag: false,
post_id: post.id
});
}
for hashtag in hashtags {
Tag::insert(&*conn, NewTag {
tag: hashtag.to_camel_case(),
is_hashtag: true,
post_id: post.id
});
}
if post.published {
for m in mentions.into_iter() {
Mention::from_activity(&*conn, Mention::build_activity(&*conn, m), post.id, true, true);
}
let act = post.create_activity(&*conn);
let dest = User::one_by_instance(&*conn);
worker.execute(Thunk::of(move || broadcast(&user, act, dest)));
}
2018-05-01 17:51:49 +02:00
2018-07-06 11:51:19 +02:00
Ok(Redirect::to(uri!(details: blog = blog_name, slug = slug)))
}
2018-07-06 11:51:19 +02:00
} else {
Err(Some(Template::render("posts/new", json!({
2018-09-03 15:59:02 +02:00
"account": user.to_json(&*conn),
"instance": Instance::get_local(&*conn),
2018-09-06 23:39:22 +02:00
"editing": false,
"errors": errors.inner(),
"form": form,
"is_draft": form.draft
}))))
}
2018-04-23 16:25:39 +02:00
}
#[post("/~/<blog_name>/<slug>/delete")]
fn delete(blog_name: String, slug: String, conn: DbConn, user: User, worker: State<Pool<ThunkWorker<()>>>) -> Redirect {
let post = Blog::find_by_fqn(&*conn, blog_name.clone())
.and_then(|blog| Post::find_by_slug(&*conn, slug.clone(), blog.id));
if let Some(post) = post {
if !post.get_authors(&*conn).into_iter().any(|a| a.id == user.id) {
Redirect::to(uri!(details: blog = blog_name.clone(), slug = slug.clone()))
} else {
let dest = User::one_by_instance(&*conn);
let delete_activity = post.delete(&*conn);
worker.execute(Thunk::of(move || broadcast(&user, delete_activity, dest)));
Redirect::to(uri!(super::blogs::details: name = blog_name))
}
} else {
Redirect::to(uri!(super::blogs::details: name = blog_name))
}
}