Plume/src/routes/user.rs

170 lines
5.5 KiB
Rust
Raw Normal View History

2018-04-22 18:13:12 +00:00
use rocket::request::Form;
use rocket::response::Redirect;
2018-04-24 12:31:02 +00:00
use rocket_contrib::Template;
2018-05-01 11:48:19 +00:00
use serde_json;
2018-04-22 18:13:12 +00:00
2018-05-04 13:13:55 +00:00
use activity_pub::{activity, activity_pub, ActivityPub, context};
2018-04-24 09:21:39 +00:00
use activity_pub::actor::Actor;
2018-05-01 14:00:29 +00:00
use activity_pub::inbox::Inbox;
use activity_pub::outbox::{broadcast, Outbox};
2018-04-22 18:13:12 +00:00
use db_conn::DbConn;
2018-05-01 19:57:30 +00:00
use models::follows::*;
2018-04-22 18:13:12 +00:00
use models::instance::Instance;
use models::posts::Post;
2018-04-24 09:21:39 +00:00
use models::users::*;
2018-04-22 18:13:12 +00:00
2018-04-23 09:52:44 +00:00
#[get("/me")]
2018-05-09 19:09:52 +00:00
fn me(user: User) -> Redirect {
Redirect::to(format!("/@/{}", user.username).as_ref())
2018-04-23 09:52:44 +00:00
}
2018-04-22 18:13:12 +00:00
2018-04-23 15:09:05 +00:00
#[get("/@/<name>", rank = 2)]
2018-05-10 20:31:52 +00:00
fn details(name: String, conn: DbConn, account: Option<User>) -> Template {
2018-05-01 11:48:19 +00:00
let user = User::find_by_fqn(&*conn, name).unwrap();
let recents = Post::get_recents_for_author(&*conn, &user, 5);
let user_id = user.id.clone();
let n_followers = user.get_followers(&*conn).len();
2018-05-01 11:48:19 +00:00
Template::render("users/details", json!({
2018-05-10 20:31:52 +00:00
"user": serde_json::to_value(user).unwrap(),
"account": account,
"recents": recents.into_iter().map(|p| {
json!({
"post": p,
"author": p.get_authors(&*conn)[0],
2018-05-12 17:59:38 +00:00
"url": format!("/~/{}/{}/", p.get_blog(&*conn).actor_id, p.slug),
"date": p.creation_date.timestamp()
})
}).collect::<Vec<serde_json::Value>>(),
"is_self": account.map(|a| a.id == user_id).unwrap_or(false),
"n_followers": n_followers
2018-05-01 11:48:19 +00:00
}))
2018-04-22 18:13:12 +00:00
}
2018-05-01 19:57:30 +00:00
#[get("/@/<name>/follow")]
fn follow(name: String, conn: DbConn, user: User) -> Redirect {
let target = User::find_by_fqn(&*conn, name.clone()).unwrap();
Follow::insert(&*conn, NewFollow {
follower_id: user.id,
following_id: target.id
});
broadcast(&*conn, &user, activity::Follow::new(&user, &target, &*conn), vec![target]);
2018-05-01 19:57:30 +00:00
Redirect::to(format!("/@/{}", name).as_ref())
}
#[get("/@/<name>/followers", rank = 2)]
fn followers(name: String, conn: DbConn, account: Option<User>) -> Template {
let user = User::find_by_fqn(&*conn, name.clone()).unwrap();
let user_id = user.id.clone();
Template::render("users/followers", json!({
"user": serde_json::to_value(user.clone()).unwrap(),
"followers": user.get_followers(&*conn).into_iter().map(|f| {
let fqn = f.get_fqn(&*conn);
let mut json = serde_json::to_value(f).unwrap();
json["fqn"] = serde_json::Value::String(fqn);
json
}).collect::<Vec<serde_json::Value>>(),
"account": account,
"is_self": account.map(|a| a.id == user_id).unwrap_or(false)
}))
}
2018-04-23 15:09:05 +00:00
#[get("/@/<name>", format = "application/activity+json", rank = 1)]
fn activity_details(name: String, conn: DbConn) -> ActivityPub {
2018-05-01 11:48:19 +00:00
let user = User::find_local(&*conn, name).unwrap();
2018-04-24 12:31:02 +00:00
user.as_activity_pub(&*conn)
2018-04-23 15:09:05 +00:00
}
2018-04-22 18:13:12 +00:00
#[get("/users/new")]
2018-05-10 20:31:52 +00:00
fn new(user: Option<User>) -> Template {
Template::render("users/new", json!({
"account": user
}))
2018-04-22 18:13:12 +00:00
}
2018-05-12 15:30:14 +00:00
#[get("/@/<name>/edit")]
fn edit(name: String, user: User) -> Option<Template> {
if user.username == name && !name.contains("@") {
Some(Template::render("users/edit", json!({
"account": user
})))
} else {
None
}
}
#[derive(FromForm)]
struct UpdateUserForm {
display_name: Option<String>,
email: Option<String>,
summary: Option<String>,
}
#[put("/@/<_name>/edit", data = "<data>")]
fn update(_name: String, conn: DbConn, user: User, data: Form<UpdateUserForm>) -> Redirect {
user.update(&*conn,
data.get().display_name.clone().unwrap_or(user.display_name.to_string()).to_string(),
data.get().email.clone().unwrap_or(user.email.clone().unwrap()).to_string(),
data.get().summary.clone().unwrap_or(user.summary.to_string())
);
Redirect::to("/me")
}
2018-04-22 18:13:12 +00:00
#[derive(FromForm)]
struct NewUserForm {
username: String,
email: String,
password: String,
password_confirmation: String
}
#[post("/users/new", data = "<data>")]
fn create(conn: DbConn, data: Form<NewUserForm>) -> Redirect {
let inst = Instance::get_local(&*conn).unwrap();
let form = data.get();
if form.password == form.password_confirmation {
User::insert(&*conn, NewUser::new_local(
form.username.to_string(),
form.username.to_string(),
!inst.has_admin(&*conn),
String::from(""),
form.email.to_string(),
User::hash_pass(form.password.to_string()),
inst.id
)).update_boxes(&*conn);
2018-04-22 18:13:12 +00:00
}
Redirect::to(format!("/@/{}", data.get().username).as_str())
}
2018-04-29 18:01:42 +00:00
#[get("/@/<name>/outbox")]
fn outbox(name: String, conn: DbConn) -> Outbox {
2018-05-01 11:48:19 +00:00
let user = User::find_local(&*conn, name).unwrap();
2018-04-29 18:01:42 +00:00
user.outbox(&*conn)
}
2018-05-01 14:00:29 +00:00
#[post("/@/<name>/inbox", data = "<data>")]
fn inbox(name: String, conn: DbConn, data: String) -> String {
let user = User::find_local(&*conn, name).unwrap();
let act: serde_json::Value = serde_json::from_str(&data[..]).unwrap();
user.received(&*conn, act);
String::from("")
}
2018-05-04 13:13:55 +00:00
#[get("/@/<name>/followers", format = "application/activity+json")]
fn ap_followers(name: String, conn: DbConn) -> ActivityPub {
2018-05-04 13:13:55 +00:00
let user = User::find_local(&*conn, name).unwrap();
let followers = user.get_followers(&*conn).into_iter().map(|f| f.compute_id(&*conn)).collect::<Vec<String>>();
let json = json!({
"@context": context(),
"id": user.compute_box(&*conn, "followers"),
"type": "OrderedCollection",
"totalItems": followers.len(),
"orderedItems": followers
});
activity_pub(json)
}