Plume/src/routes/blogs.rs

167 lines
5.5 KiB
Rust
Raw Normal View History

2018-06-21 23:07:04 +02:00
use activitypub::collection::OrderedCollection;
2018-09-01 22:08:26 +02:00
use atom_syndication::{Entry, FeedBuilder};
2018-05-19 09:39:59 +02:00
use rocket::{
http::ContentType,
2018-06-24 18:58:57 +02:00
request::LenientForm,
response::{Redirect, Flash, content::Content}
2018-05-19 09:39:59 +02:00
};
2018-04-24 14:31:02 +02:00
use rocket_contrib::Template;
use serde_json;
use std::{collections::HashMap, borrow::Cow};
2018-07-06 11:51:19 +02:00
use validator::{Validate, ValidationError, ValidationErrors};
2018-04-23 12:54:37 +02:00
use plume_common::activity_pub::{ActivityStream, ApRequest};
use plume_common::utils;
use plume_models::{
2018-05-19 09:39:59 +02:00
blog_authors::*,
blogs::*,
db_conn::DbConn,
2018-05-19 09:39:59 +02:00
instance::Instance,
posts::Post,
users::User
};
use routes::Page;
use Searcher;
2018-04-23 12:54:37 +02:00
#[get("/~/<name>?<page>", rank = 2)]
fn paginated_details(name: String, conn: DbConn, user: Option<User>, page: Page) -> Template {
may_fail!(user.map(|u| u.to_json(&*conn)), Blog::find_by_fqn(&*conn, &name), "Requested blog couldn't be found", |blog| {
let posts = Post::blog_page(&*conn, &blog, page.limits());
let articles = Post::get_for_blog(&*conn, &blog);
let authors = &blog.list_authors(&*conn);
Template::render("blogs/details", json!({
2018-07-26 19:10:50 +02:00
"blog": &blog.to_json(&*conn),
2018-09-03 15:59:02 +02:00
"account": user.clone().map(|u| u.to_json(&*conn)),
"is_author": user.map(|x| x.is_author_in(&*conn, &blog)),
"posts": posts.into_iter().map(|p| p.to_json(&*conn)).collect::<Vec<serde_json::Value>>(),
"authors": authors.into_iter().map(|u| u.to_json(&*conn)).collect::<Vec<serde_json::Value>>(),
"n_authors": authors.len(),
2018-07-25 14:29:34 +02:00
"n_articles": articles.len(),
"page": page.page,
"n_pages": Page::total(articles.len() as i32)
}))
})
2018-04-23 12:54:37 +02:00
}
#[get("/~/<name>", rank = 3)]
fn details(name: String, conn: DbConn, user: Option<User>) -> Template {
paginated_details(name, conn, user, Page::first())
}
#[get("/~/<name>", rank = 1)]
fn activity_details(name: String, conn: DbConn, _ap: ApRequest) -> Option<ActivityStream<CustomGroup>> {
let blog = Blog::find_local(&*conn, &name)?;
Some(ActivityStream::new(blog.to_activity(&*conn)))
2018-04-23 17:09:05 +02:00
}
2018-04-23 12:54:37 +02:00
#[get("/blogs/new")]
2018-09-03 15:59:02 +02:00
fn new(user: User, conn: DbConn) -> Template {
2018-05-10 22:31:52 +02:00
Template::render("blogs/new", json!({
2018-09-03 15:59:02 +02:00
"account": user.to_json(&*conn),
"errors": null,
"form": null
2018-05-10 22:31:52 +02:00
}))
2018-04-23 12:54:37 +02:00
}
2018-06-04 21:57:03 +02:00
#[get("/blogs/new", rank = 2)]
fn new_auth() -> Flash<Redirect>{
utils::requires_login(
"You need to be logged in order to create a new blog",
uri!(new)
)
2018-06-04 21:57:03 +02:00
}
#[derive(FromForm, Validate, Serialize)]
2018-04-23 12:54:37 +02:00
struct NewBlogForm {
#[validate(custom(function = "valid_slug", message = "Invalid name"))]
2018-04-23 12:54:37 +02:00
pub title: String
}
2018-06-29 14:56:00 +02:00
fn valid_slug(title: &str) -> Result<(), ValidationError> {
let slug = utils::make_actor_id(title);
if slug.is_empty() {
2018-06-29 14:56:00 +02:00
Err(ValidationError::new("empty_slug"))
} else {
Ok(())
}
}
2018-04-23 12:54:37 +02:00
#[post("/blogs/new", data = "<data>")]
2018-07-06 11:51:19 +02:00
fn create(conn: DbConn, data: LenientForm<NewBlogForm>, user: User) -> Result<Redirect, Template> {
2018-04-23 12:54:37 +02:00
let form = data.get();
let slug = utils::make_actor_id(&form.title);
2018-07-06 11:51:19 +02:00
let mut errors = match form.validate() {
Ok(_) => ValidationErrors::new(),
Err(e) => e
};
if Blog::find_local(&*conn, &slug).is_some() {
errors.add("title", ValidationError {
code: Cow::from("existing_slug"),
message: Some(Cow::from("A blog with the same name already exists.")),
params: HashMap::new()
});
2018-07-06 11:51:19 +02:00
}
if errors.is_empty() {
let blog = Blog::insert(&*conn, NewBlog::new_local(
2018-07-06 11:51:19 +02:00
slug.clone(),
form.title.to_string(),
String::from(""),
Instance::local_id(&*conn)
));
blog.update_boxes(&*conn);
2018-04-23 15:22:07 +02:00
BlogAuthor::insert(&*conn, NewBlogAuthor {
blog_id: blog.id,
author_id: user.id,
is_owner: true
});
2018-07-06 11:51:19 +02:00
Ok(Redirect::to(uri!(details: name = slug.clone())))
} else {
println!("{:?}", errors);
2018-07-06 11:51:19 +02:00
Err(Template::render("blogs/new", json!({
2018-09-03 15:59:02 +02:00
"account": user.to_json(&*conn),
"errors": errors.inner(),
"form": form
2018-07-06 11:51:19 +02:00
})))
}
2018-04-23 12:54:37 +02:00
}
2018-04-29 19:49:56 +02:00
2018-10-20 15:03:59 +02:00
#[post("/~/<name>/delete")]
fn delete(conn: DbConn, name: String, user: Option<User>, searcher: Searcher) -> Result<Redirect, Option<Template>>{
let blog = Blog::find_local(&*conn, &name).ok_or(None)?;
if user.map(|u| u.is_author_in(&*conn, &blog)).unwrap_or(false) {
blog.delete(&conn, &searcher);
2018-10-20 15:03:59 +02:00
Ok(Redirect::to(uri!(super::instance::index)))
} else {
Err(Some(Template::render("errors/403", json!({// TODO actually return 403 error code
"error_message": "You are not allowed to delete this blog."
}))))
}
}
2018-04-29 19:49:56 +02:00
#[get("/~/<name>/outbox")]
fn outbox(name: String, conn: DbConn) -> Option<ActivityStream<OrderedCollection>> {
let blog = Blog::find_local(&*conn, &name)?;
Some(blog.outbox(&*conn))
2018-04-29 19:49:56 +02:00
}
2018-09-01 22:08:26 +02:00
#[get("/~/<name>/atom.xml")]
fn atom_feed(name: String, conn: DbConn) -> Option<Content<String>> {
let blog = Blog::find_by_fqn(&*conn, &name)?;
2018-09-01 22:08:26 +02:00
let feed = FeedBuilder::default()
.title(blog.title.clone())
.id(Instance::get_local(&*conn).expect("blogs::atom_feed: local instance not found error")
.compute_box("~", &name, "atom.xml"))
2018-09-01 22:08:26 +02:00
.entries(Post::get_recents_for_blog(&*conn, &blog, 15)
.into_iter()
.map(|p| super::post_to_atom(p, &*conn))
.collect::<Vec<Entry>>())
.build()
.expect("blogs::atom_feed: feed creation error");
Some(Content(ContentType::new("application", "atom+xml"), feed.to_string()))
2018-09-01 22:08:26 +02:00
}