2018-11-24 12:44:17 +01:00
|
|
|
use activitypub::{activity::Create, collection::OrderedCollection, object::Article};
|
2018-09-01 22:08:26 +02:00
|
|
|
use atom_syndication::{Entry, FeedBuilder};
|
2018-07-26 17:51:41 +02:00
|
|
|
use rocket::{
|
2018-11-24 12:44:17 +01:00
|
|
|
http::{ContentType, Cookies},
|
2018-07-26 17:51:41 +02:00
|
|
|
request::LenientForm,
|
2018-11-24 12:44:17 +01:00
|
|
|
response::{status, Content, Flash, Redirect},
|
2018-06-07 10:39:22 +02:00
|
|
|
};
|
2018-04-24 14:31:02 +02:00
|
|
|
use rocket_contrib::Template;
|
2018-05-01 13:48:19 +02:00
|
|
|
use serde_json;
|
2018-06-29 14:56:00 +02:00
|
|
|
use validator::{Validate, ValidationError};
|
2018-09-03 20:53:20 +02:00
|
|
|
use workerpool::thunk::*;
|
2018-04-22 20:13:12 +02:00
|
|
|
|
2018-11-24 12:44:17 +01:00
|
|
|
use inbox::Inbox;
|
2018-06-23 18:36:11 +02:00
|
|
|
use plume_common::activity_pub::{
|
2018-11-24 12:44:17 +01:00
|
|
|
broadcast,
|
|
|
|
inbox::{Deletable, FromActivity, Notify},
|
|
|
|
sign::{verify_http_headers, Signable},
|
|
|
|
ActivityStream, ApRequest, Id, IntoId,
|
2018-05-19 09:39:59 +02:00
|
|
|
};
|
2018-06-23 18:36:11 +02:00
|
|
|
use plume_common::utils;
|
|
|
|
use plume_models::{
|
2018-11-24 12:44:17 +01:00
|
|
|
blogs::Blog, db_conn::DbConn, follows, headers::Headers, instance::Instance, posts::Post,
|
|
|
|
reshares::Reshare, users::*,
|
2018-05-19 09:39:59 +02:00
|
|
|
};
|
2018-07-25 15:50:29 +02:00
|
|
|
use routes::Page;
|
2018-09-03 20:53:20 +02:00
|
|
|
use Worker;
|
2018-04-22 20:13:12 +02:00
|
|
|
|
2018-04-23 11:52:44 +02:00
|
|
|
#[get("/me")]
|
2018-06-18 19:28:28 +02:00
|
|
|
fn me(user: Option<User>) -> Result<Redirect, Flash<Redirect>> {
|
2018-06-07 10:39:22 +02:00
|
|
|
match user {
|
2018-06-19 23:20:27 +02:00
|
|
|
Some(user) => Ok(Redirect::to(uri!(details: name = user.username))),
|
2018-11-26 10:21:52 +01:00
|
|
|
None => Err(utils::requires_login("", uri!(me))),
|
2018-06-07 10:39:22 +02:00
|
|
|
}
|
2018-04-23 11:52:44 +02:00
|
|
|
}
|
2018-04-22 20:13:12 +02:00
|
|
|
|
2018-04-23 17:09:05 +02:00
|
|
|
#[get("/@/<name>", rank = 2)]
|
2018-11-24 12:44:17 +01:00
|
|
|
fn details(
|
|
|
|
name: String,
|
|
|
|
conn: DbConn,
|
|
|
|
account: Option<User>,
|
|
|
|
worker: Worker,
|
|
|
|
fecth_articles_conn: DbConn,
|
|
|
|
fecth_followers_conn: DbConn,
|
|
|
|
update_conn: DbConn,
|
|
|
|
) -> Template {
|
|
|
|
may_fail!(
|
|
|
|
account.map(|a| a.to_json(&*conn)),
|
2018-11-26 10:21:52 +01:00
|
|
|
User::find_by_fqn(&*conn, &name),
|
2018-11-24 12:44:17 +01:00
|
|
|
"Couldn't find requested user",
|
|
|
|
|user| {
|
|
|
|
let recents = Post::get_recents_for_author(&*conn, &user, 6);
|
|
|
|
let reshares = Reshare::get_recents_for_author(&*conn, &user, 6);
|
2018-11-26 10:21:52 +01:00
|
|
|
let user_id = user.id;
|
2018-11-24 12:44:17 +01:00
|
|
|
let n_followers = user.get_followers(&*conn).len();
|
|
|
|
|
|
|
|
if !user.get_instance(&*conn).local {
|
|
|
|
// Fetch new articles
|
|
|
|
let user_clone = user.clone();
|
|
|
|
worker.execute(Thunk::of(move || {
|
|
|
|
for create_act in user_clone.fetch_outbox::<Create>() {
|
|
|
|
match create_act.create_props.object_object::<Article>() {
|
|
|
|
Ok(article) => {
|
|
|
|
Post::from_activity(
|
|
|
|
&*fecth_articles_conn,
|
|
|
|
article,
|
|
|
|
user_clone.clone().into_id(),
|
|
|
|
);
|
|
|
|
println!("Fetched article from remote user");
|
|
|
|
}
|
|
|
|
Err(e) => {
|
|
|
|
println!("Error while fetching articles in background: {:?}", e)
|
|
|
|
}
|
2018-07-26 22:59:41 +02:00
|
|
|
}
|
2018-07-26 22:23:53 +02:00
|
|
|
}
|
2018-11-24 12:44:17 +01:00
|
|
|
}));
|
2018-09-03 20:53:20 +02:00
|
|
|
|
2018-11-24 12:44:17 +01:00
|
|
|
// Fetch followers
|
|
|
|
let user_clone = user.clone();
|
2018-09-03 20:53:20 +02:00
|
|
|
worker.execute(Thunk::of(move || {
|
2018-11-24 12:44:17 +01:00
|
|
|
for user_id in user_clone.fetch_followers_ids() {
|
|
|
|
let follower =
|
2018-11-26 10:21:52 +01:00
|
|
|
User::find_by_ap_url(&*fecth_followers_conn, &user_id)
|
2018-11-24 12:44:17 +01:00
|
|
|
.unwrap_or_else(|| {
|
2018-11-26 10:21:52 +01:00
|
|
|
User::fetch_from_url(&*fecth_followers_conn, &user_id)
|
2018-11-24 12:44:17 +01:00
|
|
|
.expect("user::details: Couldn't fetch follower")
|
|
|
|
});
|
|
|
|
follows::Follow::insert(
|
|
|
|
&*fecth_followers_conn,
|
|
|
|
follows::NewFollow {
|
|
|
|
follower_id: follower.id,
|
|
|
|
following_id: user_clone.id,
|
|
|
|
ap_url: format!("{}/follow/{}", follower.ap_url, user_clone.ap_url),
|
|
|
|
},
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}));
|
|
|
|
|
|
|
|
// Update profile information if needed
|
|
|
|
let user_clone = user.clone();
|
|
|
|
if user.needs_update() {
|
|
|
|
worker.execute(Thunk::of(move || {
|
|
|
|
user_clone.refetch(&*update_conn);
|
|
|
|
}))
|
|
|
|
}
|
2018-09-03 20:53:20 +02:00
|
|
|
}
|
2018-07-26 22:23:53 +02:00
|
|
|
|
2018-11-24 12:44:17 +01:00
|
|
|
Template::render(
|
|
|
|
"users/details",
|
|
|
|
json!({
|
2018-07-20 17:59:16 +02:00
|
|
|
"user": user.to_json(&*conn),
|
2018-06-18 19:28:28 +02:00
|
|
|
"instance_url": user.get_instance(&*conn).public_domain,
|
|
|
|
"is_remote": user.instance_id != Instance::local_id(&*conn),
|
|
|
|
"follows": account.clone().map(|x| x.is_following(&*conn, user.id)).unwrap_or(false),
|
2018-09-03 15:59:02 +02:00
|
|
|
"account": account.clone().map(|a| a.to_json(&*conn)),
|
2018-06-18 19:28:28 +02:00
|
|
|
"recents": recents.into_iter().map(|p| p.to_json(&*conn)).collect::<Vec<serde_json::Value>>(),
|
|
|
|
"reshares": reshares.into_iter().map(|r| r.get_post(&*conn).unwrap().to_json(&*conn)).collect::<Vec<serde_json::Value>>(),
|
|
|
|
"is_self": account.map(|a| a.id == user_id).unwrap_or(false),
|
|
|
|
"n_followers": n_followers
|
2018-11-24 12:44:17 +01:00
|
|
|
}),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
)
|
2018-04-22 20:13:12 +02:00
|
|
|
}
|
|
|
|
|
2018-06-10 19:55:08 +02:00
|
|
|
#[get("/dashboard")]
|
|
|
|
fn dashboard(user: User, conn: DbConn) -> Template {
|
2018-11-24 12:44:17 +01:00
|
|
|
let blogs = Blog::find_for_author(&*conn, &user);
|
|
|
|
Template::render(
|
|
|
|
"users/dashboard",
|
|
|
|
json!({
|
2018-09-03 15:59:02 +02:00
|
|
|
"account": user.to_json(&*conn),
|
2018-09-10 21:06:00 +02:00
|
|
|
"blogs": blogs,
|
|
|
|
"drafts": Post::drafts_by_author(&*conn, &user).into_iter().map(|a| a.to_json(&*conn)).collect::<Vec<serde_json::Value>>(),
|
2018-11-24 12:44:17 +01:00
|
|
|
}),
|
|
|
|
)
|
2018-06-10 19:55:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#[get("/dashboard", rank = 2)]
|
|
|
|
fn dashboard_auth() -> Flash<Redirect> {
|
2018-09-08 01:11:27 +02:00
|
|
|
utils::requires_login(
|
|
|
|
"You need to be logged in order to access your dashboard",
|
2018-11-26 10:21:52 +01:00
|
|
|
uri!(dashboard),
|
2018-09-08 01:11:27 +02:00
|
|
|
)
|
2018-06-10 19:55:08 +02:00
|
|
|
}
|
|
|
|
|
2018-09-19 19:13:07 +02:00
|
|
|
#[post("/@/<name>/follow")]
|
2018-10-20 11:04:20 +02:00
|
|
|
fn follow(name: String, conn: DbConn, user: User, worker: Worker) -> Option<Redirect> {
|
2018-11-26 10:21:52 +01:00
|
|
|
let target = User::find_by_fqn(&*conn, &name)?;
|
2018-09-04 12:37:58 +02:00
|
|
|
if let Some(follow) = follows::Follow::find(&*conn, user.id, target.id) {
|
|
|
|
let delete_act = follow.delete(&*conn);
|
2018-11-24 12:44:17 +01:00
|
|
|
worker.execute(Thunk::of(move || {
|
|
|
|
broadcast(&user, delete_act, vec![target])
|
|
|
|
}));
|
2018-09-04 12:37:58 +02:00
|
|
|
} else {
|
2018-11-24 12:44:17 +01:00
|
|
|
let f = follows::Follow::insert(
|
|
|
|
&*conn,
|
|
|
|
follows::NewFollow {
|
|
|
|
follower_id: user.id,
|
|
|
|
following_id: target.id,
|
|
|
|
ap_url: format!("{}/follow/{}", user.ap_url, target.ap_url),
|
|
|
|
},
|
|
|
|
);
|
2018-09-04 12:37:58 +02:00
|
|
|
f.notify(&*conn);
|
2018-06-20 23:51:47 +02:00
|
|
|
|
2018-11-26 10:21:52 +01:00
|
|
|
let act = f.to_activity(&*conn);
|
2018-09-04 12:37:58 +02:00
|
|
|
worker.execute(Thunk::of(move || broadcast(&user, act, vec![target])));
|
|
|
|
}
|
2018-10-20 11:04:20 +02:00
|
|
|
Some(Redirect::to(uri!(details: name = name)))
|
2018-05-01 21:57:30 +02:00
|
|
|
}
|
|
|
|
|
2018-09-19 19:13:07 +02:00
|
|
|
#[post("/@/<name>/follow", rank = 2)]
|
2018-06-07 10:39:22 +02:00
|
|
|
fn follow_auth(name: String) -> Flash<Redirect> {
|
2018-09-08 01:11:27 +02:00
|
|
|
utils::requires_login(
|
|
|
|
"You need to be logged in order to follow someone",
|
2018-11-26 10:21:52 +01:00
|
|
|
uri!(follow: name = name),
|
2018-09-08 01:11:27 +02:00
|
|
|
)
|
2018-06-07 10:39:22 +02:00
|
|
|
}
|
|
|
|
|
2018-07-25 15:50:29 +02:00
|
|
|
#[get("/@/<name>/followers?<page>")]
|
|
|
|
fn followers_paginated(name: String, conn: DbConn, account: Option<User>, page: Page) -> Template {
|
2018-11-24 12:44:17 +01:00
|
|
|
may_fail!(
|
|
|
|
account.map(|a| a.to_json(&*conn)),
|
2018-11-26 10:21:52 +01:00
|
|
|
User::find_by_fqn(&*conn, &name),
|
2018-11-24 12:44:17 +01:00
|
|
|
"Couldn't find requested user",
|
|
|
|
|user| {
|
2018-11-26 10:21:52 +01:00
|
|
|
let user_id = user.id;
|
2018-11-24 12:44:17 +01:00
|
|
|
let followers_count = user.get_followers(&*conn).len();
|
|
|
|
|
|
|
|
Template::render(
|
|
|
|
"users/followers",
|
|
|
|
json!({
|
2018-07-20 17:59:16 +02:00
|
|
|
"user": user.to_json(&*conn),
|
2018-06-18 19:28:28 +02:00
|
|
|
"instance_url": user.get_instance(&*conn).public_domain,
|
|
|
|
"is_remote": user.instance_id != Instance::local_id(&*conn),
|
|
|
|
"follows": account.clone().map(|x| x.is_following(&*conn, user.id)).unwrap_or(false),
|
2018-07-25 15:50:29 +02:00
|
|
|
"followers": user.get_followers_page(&*conn, page.limits()).into_iter().map(|f| f.to_json(&*conn)).collect::<Vec<serde_json::Value>>(),
|
2018-09-03 15:59:02 +02:00
|
|
|
"account": account.clone().map(|a| a.to_json(&*conn)),
|
2018-06-18 19:28:28 +02:00
|
|
|
"is_self": account.map(|a| a.id == user_id).unwrap_or(false),
|
2018-07-25 15:50:29 +02:00
|
|
|
"n_followers": followers_count,
|
|
|
|
"page": page.page,
|
|
|
|
"n_pages": Page::total(followers_count as i32)
|
2018-11-24 12:44:17 +01:00
|
|
|
}),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
)
|
2018-05-13 13:53:58 +02:00
|
|
|
}
|
|
|
|
|
2018-07-25 15:50:29 +02:00
|
|
|
#[get("/@/<name>/followers", rank = 2)]
|
|
|
|
fn followers(name: String, conn: DbConn, account: Option<User>) -> Template {
|
|
|
|
followers_paginated(name, conn, account, Page::first())
|
|
|
|
}
|
|
|
|
|
2018-07-11 17:30:01 +02:00
|
|
|
#[get("/@/<name>", rank = 1)]
|
2018-11-24 12:44:17 +01:00
|
|
|
fn activity_details(
|
|
|
|
name: String,
|
|
|
|
conn: DbConn,
|
|
|
|
_ap: ApRequest,
|
|
|
|
) -> Option<ActivityStream<CustomPerson>> {
|
2018-11-26 10:21:52 +01:00
|
|
|
let user = User::find_local(&*conn, &name)?;
|
|
|
|
Some(ActivityStream::new(user.to_activity(&*conn)))
|
2018-04-23 17:09:05 +02:00
|
|
|
}
|
|
|
|
|
2018-04-22 20:13:12 +02:00
|
|
|
#[get("/users/new")]
|
2018-09-03 15:59:02 +02:00
|
|
|
fn new(user: Option<User>, conn: DbConn) -> Template {
|
2018-11-24 12:44:17 +01:00
|
|
|
Template::render(
|
|
|
|
"users/new",
|
|
|
|
json!({
|
2018-09-03 18:05:45 +02:00
|
|
|
"enabled": Instance::get_local(&*conn).map(|i| i.open_registrations).unwrap_or(true),
|
2018-09-03 15:59:02 +02:00
|
|
|
"account": user.map(|u| u.to_json(&*conn)),
|
2018-07-06 19:29:36 +02:00
|
|
|
"errors": null,
|
|
|
|
"form": null
|
2018-11-24 12:44:17 +01:00
|
|
|
}),
|
|
|
|
)
|
2018-04-22 20:13:12 +02:00
|
|
|
}
|
|
|
|
|
2018-05-12 17:30:14 +02:00
|
|
|
#[get("/@/<name>/edit")]
|
2018-09-03 15:59:02 +02:00
|
|
|
fn edit(name: String, user: User, conn: DbConn) -> Option<Template> {
|
2018-11-26 10:21:52 +01:00
|
|
|
if user.username == name && !name.contains('@') {
|
2018-11-24 12:44:17 +01:00
|
|
|
Some(Template::render(
|
|
|
|
"users/edit",
|
|
|
|
json!({
|
2018-09-03 15:59:02 +02:00
|
|
|
"account": user.to_json(&*conn)
|
2018-11-24 12:44:17 +01:00
|
|
|
}),
|
|
|
|
))
|
2018-05-12 17:30:14 +02:00
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-07 10:39:22 +02:00
|
|
|
#[get("/@/<name>/edit", rank = 2)]
|
|
|
|
fn edit_auth(name: String) -> Flash<Redirect> {
|
2018-09-08 01:11:27 +02:00
|
|
|
utils::requires_login(
|
|
|
|
"You need to be logged in order to edit your profile",
|
2018-11-26 10:21:52 +01:00
|
|
|
uri!(edit: name = name),
|
2018-09-08 01:11:27 +02:00
|
|
|
)
|
2018-06-07 10:39:22 +02:00
|
|
|
}
|
|
|
|
|
2018-05-12 17:30:14 +02:00
|
|
|
#[derive(FromForm)]
|
|
|
|
struct UpdateUserForm {
|
|
|
|
display_name: Option<String>,
|
|
|
|
email: Option<String>,
|
|
|
|
summary: Option<String>,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[put("/@/<_name>/edit", data = "<data>")]
|
2018-06-24 18:58:57 +02:00
|
|
|
fn update(_name: String, conn: DbConn, user: User, data: LenientForm<UpdateUserForm>) -> Redirect {
|
2018-11-24 12:44:17 +01:00
|
|
|
user.update(
|
|
|
|
&*conn,
|
|
|
|
data.get()
|
|
|
|
.display_name
|
|
|
|
.clone()
|
2018-11-26 10:21:52 +01:00
|
|
|
.unwrap_or_else(|| user.display_name.to_string())
|
2018-11-24 12:44:17 +01:00
|
|
|
.to_string(),
|
|
|
|
data.get()
|
|
|
|
.email
|
|
|
|
.clone()
|
2018-11-26 10:21:52 +01:00
|
|
|
.unwrap_or_else(|| user.email.clone().unwrap())
|
2018-11-24 12:44:17 +01:00
|
|
|
.to_string(),
|
|
|
|
data.get()
|
|
|
|
.summary
|
|
|
|
.clone()
|
2018-11-26 10:21:52 +01:00
|
|
|
.unwrap_or_else(|| user.summary.to_string()),
|
2018-05-12 17:30:14 +02:00
|
|
|
);
|
2018-06-19 23:20:27 +02:00
|
|
|
Redirect::to(uri!(me))
|
2018-05-12 17:30:14 +02:00
|
|
|
}
|
|
|
|
|
2018-09-19 19:13:07 +02:00
|
|
|
#[post("/@/<name>/delete")]
|
2018-10-20 11:04:20 +02:00
|
|
|
fn delete(name: String, conn: DbConn, user: User, mut cookies: Cookies) -> Option<Redirect> {
|
2018-11-26 10:21:52 +01:00
|
|
|
let account = User::find_by_fqn(&*conn, &name)?;
|
2018-09-09 21:49:24 +02:00
|
|
|
if user.id == account.id {
|
|
|
|
account.delete(&*conn);
|
|
|
|
|
2018-11-26 10:21:52 +01:00
|
|
|
if let Some(cookie) = cookies.get_private(AUTH_COOKIE) {
|
|
|
|
cookies.remove_private(cookie);
|
|
|
|
}
|
2018-09-09 21:49:24 +02:00
|
|
|
|
2018-10-20 11:04:20 +02:00
|
|
|
Some(Redirect::to(uri!(super::instance::index)))
|
2018-09-09 21:49:24 +02:00
|
|
|
} else {
|
2018-10-20 11:04:20 +02:00
|
|
|
Some(Redirect::to(uri!(edit: name = name)))
|
2018-09-09 21:49:24 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-06 19:29:36 +02:00
|
|
|
#[derive(FromForm, Serialize, Validate)]
|
2018-11-24 12:44:17 +01:00
|
|
|
#[validate(
|
|
|
|
schema(
|
|
|
|
function = "passwords_match",
|
|
|
|
skip_on_field_errors = "false",
|
|
|
|
message = "Passwords are not matching"
|
|
|
|
)
|
|
|
|
)]
|
2018-04-22 20:13:12 +02:00
|
|
|
struct NewUserForm {
|
2018-12-02 12:43:03 +01:00
|
|
|
#[validate(length(min = "1", message = "Username can't be empty"),
|
|
|
|
custom( function = "validate_username", message = "User name is not allowed to contain any of < > & @ ' or \""))]
|
2018-04-22 20:13:12 +02:00
|
|
|
username: String,
|
2018-07-06 19:29:36 +02:00
|
|
|
#[validate(email(message = "Invalid email"))]
|
2018-04-22 20:13:12 +02:00
|
|
|
email: String,
|
2018-11-24 12:44:17 +01:00
|
|
|
#[validate(
|
|
|
|
length(
|
|
|
|
min = "8",
|
|
|
|
message = "Password should be at least 8 characters long"
|
|
|
|
)
|
|
|
|
)]
|
2018-04-22 20:13:12 +02:00
|
|
|
password: String,
|
2018-11-24 12:44:17 +01:00
|
|
|
#[validate(
|
|
|
|
length(
|
|
|
|
min = "8",
|
|
|
|
message = "Password should be at least 8 characters long"
|
|
|
|
)
|
|
|
|
)]
|
|
|
|
password_confirmation: String,
|
2018-04-22 20:13:12 +02:00
|
|
|
}
|
|
|
|
|
2018-06-29 14:56:00 +02:00
|
|
|
fn passwords_match(form: &NewUserForm) -> Result<(), ValidationError> {
|
|
|
|
if form.password != form.password_confirmation {
|
|
|
|
Err(ValidationError::new("password_match"))
|
|
|
|
} else {
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-02 12:43:03 +01:00
|
|
|
fn validate_username(username: &str) -> Result<(), ValidationError> {
|
|
|
|
if username.contains(&['<', '>', '&', '@', '\'', '"'][..]) {
|
|
|
|
Err(ValidationError::new("username_illegal_char"))
|
|
|
|
} else {
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-22 20:13:12 +02:00
|
|
|
#[post("/users/new", data = "<data>")]
|
2018-07-06 11:51:19 +02:00
|
|
|
fn create(conn: DbConn, data: LenientForm<NewUserForm>) -> Result<Redirect, Template> {
|
2018-11-24 12:44:17 +01:00
|
|
|
if !Instance::get_local(&*conn)
|
|
|
|
.map(|i| i.open_registrations)
|
|
|
|
.unwrap_or(true)
|
|
|
|
{
|
2018-09-03 18:05:45 +02:00
|
|
|
return Ok(Redirect::to(uri!(new))); // Actually, it is an error
|
|
|
|
}
|
|
|
|
|
2018-04-22 20:13:12 +02:00
|
|
|
let form = data.get();
|
2018-07-06 11:51:19 +02:00
|
|
|
form.validate()
|
|
|
|
.map(|_| {
|
2018-11-24 12:44:17 +01:00
|
|
|
NewUser::new_local(
|
2018-07-06 11:51:19 +02:00
|
|
|
&*conn,
|
|
|
|
form.username.to_string(),
|
|
|
|
form.username.to_string(),
|
|
|
|
false,
|
2018-11-26 10:21:52 +01:00
|
|
|
"",
|
2018-07-06 11:51:19 +02:00
|
|
|
form.email.to_string(),
|
2018-11-26 10:21:52 +01:00
|
|
|
User::hash_pass(&form.password),
|
2018-07-06 11:51:19 +02:00
|
|
|
).update_boxes(&*conn);
|
|
|
|
Redirect::to(uri!(super::session::new))
|
|
|
|
})
|
2018-11-24 12:44:17 +01:00
|
|
|
.map_err(|e| {
|
|
|
|
Template::render(
|
|
|
|
"users/new",
|
|
|
|
json!({
|
2018-09-07 21:13:14 +02:00
|
|
|
"enabled": Instance::get_local(&*conn).map(|i| i.open_registrations).unwrap_or(true),
|
2018-07-06 19:29:36 +02:00
|
|
|
"errors": e.inner(),
|
|
|
|
"form": form
|
2018-11-24 12:44:17 +01:00
|
|
|
}),
|
|
|
|
)
|
|
|
|
})
|
2018-04-22 20:13:12 +02:00
|
|
|
}
|
2018-04-29 20:01:42 +02:00
|
|
|
|
|
|
|
#[get("/@/<name>/outbox")]
|
2018-10-20 11:04:20 +02:00
|
|
|
fn outbox(name: String, conn: DbConn) -> Option<ActivityStream<OrderedCollection>> {
|
2018-11-26 10:21:52 +01:00
|
|
|
let user = User::find_local(&*conn, &name)?;
|
2018-10-20 11:04:20 +02:00
|
|
|
Some(user.outbox(&*conn))
|
2018-04-29 20:01:42 +02:00
|
|
|
}
|
2018-05-01 16:00:29 +02:00
|
|
|
|
|
|
|
#[post("/@/<name>/inbox", data = "<data>")]
|
2018-11-24 12:44:17 +01:00
|
|
|
fn inbox(
|
|
|
|
name: String,
|
|
|
|
conn: DbConn,
|
|
|
|
data: String,
|
|
|
|
headers: Headers,
|
|
|
|
) -> Result<String, Option<status::BadRequest<&'static str>>> {
|
2018-11-26 10:21:52 +01:00
|
|
|
let user = User::find_local(&*conn, &name).ok_or(None)?;
|
2018-11-24 12:44:17 +01:00
|
|
|
let act: serde_json::Value =
|
2018-11-26 10:21:52 +01:00
|
|
|
serde_json::from_str(&data).expect("user::inbox: deserialization error");
|
2018-09-08 23:05:48 +02:00
|
|
|
|
|
|
|
let activity = act.clone();
|
2018-11-24 12:44:17 +01:00
|
|
|
let actor_id = activity["actor"]
|
|
|
|
.as_str()
|
|
|
|
.or_else(|| activity["actor"]["id"].as_str())
|
|
|
|
.ok_or(Some(status::BadRequest(Some(
|
|
|
|
"Missing actor id for activity",
|
|
|
|
))))?;
|
2018-10-03 20:48:25 +02:00
|
|
|
|
2018-11-26 10:21:52 +01:00
|
|
|
let actor = User::from_url(&conn, actor_id).expect("user::inbox: user error");
|
|
|
|
if !verify_http_headers(&actor, &headers.0, &data).is_secure()
|
2018-11-24 12:44:17 +01:00
|
|
|
&& !act.clone().verify(&actor)
|
|
|
|
{
|
|
|
|
println!(
|
|
|
|
"Rejected invalid activity supposedly from {}, with headers {:?}",
|
|
|
|
actor.username, headers.0
|
|
|
|
);
|
2018-10-20 11:04:20 +02:00
|
|
|
return Err(Some(status::BadRequest(Some("Invalid signature"))));
|
2018-10-03 20:48:25 +02:00
|
|
|
}
|
|
|
|
|
2018-11-26 10:21:52 +01:00
|
|
|
if Instance::is_blocked(&*conn, actor_id) {
|
2018-10-20 11:04:20 +02:00
|
|
|
return Ok(String::new());
|
2018-09-08 23:05:48 +02:00
|
|
|
}
|
2018-10-20 11:04:20 +02:00
|
|
|
Ok(match user.received(&*conn, act) {
|
2018-06-21 18:00:37 +02:00
|
|
|
Ok(_) => String::new(),
|
|
|
|
Err(e) => {
|
2018-09-08 01:11:27 +02:00
|
|
|
println!("User inbox error: {}\n{}", e.as_fail(), e.backtrace());
|
|
|
|
format!("Error: {}", e.as_fail())
|
2018-06-21 18:00:37 +02:00
|
|
|
}
|
2018-10-20 11:04:20 +02:00
|
|
|
})
|
2018-05-01 16:00:29 +02:00
|
|
|
}
|
2018-05-04 15:13:55 +02:00
|
|
|
|
2018-07-11 17:30:01 +02:00
|
|
|
#[get("/@/<name>/followers")]
|
2018-11-24 12:44:17 +01:00
|
|
|
fn ap_followers(
|
|
|
|
name: String,
|
|
|
|
conn: DbConn,
|
|
|
|
_ap: ApRequest,
|
|
|
|
) -> Option<ActivityStream<OrderedCollection>> {
|
2018-11-26 10:21:52 +01:00
|
|
|
let user = User::find_local(&*conn, &name)?;
|
2018-11-24 12:44:17 +01:00
|
|
|
let followers = user
|
|
|
|
.get_followers(&*conn)
|
|
|
|
.into_iter()
|
|
|
|
.map(|f| Id::new(f.ap_url))
|
|
|
|
.collect::<Vec<Id>>();
|
2018-06-21 19:09:18 +02:00
|
|
|
|
|
|
|
let mut coll = OrderedCollection::default();
|
2018-11-24 12:44:17 +01:00
|
|
|
coll.object_props
|
|
|
|
.set_id_string(user.followers_endpoint)
|
|
|
|
.expect("user::ap_followers: id error");
|
|
|
|
coll.collection_props
|
|
|
|
.set_total_items_u64(followers.len() as u64)
|
|
|
|
.expect("user::ap_followers: totalItems error");
|
|
|
|
coll.collection_props
|
|
|
|
.set_items_link_vec(followers)
|
|
|
|
.expect("user::ap_followers items error");
|
2018-10-20 11:04:20 +02:00
|
|
|
Some(ActivityStream::new(coll))
|
2018-05-04 15:13:55 +02:00
|
|
|
}
|
2018-09-01 22:08:26 +02:00
|
|
|
|
|
|
|
#[get("/@/<name>/atom.xml")]
|
2018-10-20 11:04:20 +02:00
|
|
|
fn atom_feed(name: String, conn: DbConn) -> Option<Content<String>> {
|
2018-11-26 10:21:52 +01:00
|
|
|
let author = User::find_by_fqn(&*conn, &name)?;
|
2018-09-01 22:08:26 +02:00
|
|
|
let feed = FeedBuilder::default()
|
|
|
|
.title(author.display_name.clone())
|
2018-11-24 12:44:17 +01:00
|
|
|
.id(Instance::get_local(&*conn)
|
|
|
|
.unwrap()
|
2018-11-26 10:21:52 +01:00
|
|
|
.compute_box("~", &name, "atom.xml"))
|
2018-11-24 12:44:17 +01:00
|
|
|
.entries(
|
|
|
|
Post::get_recents_for_author(&*conn, &author, 15)
|
|
|
|
.into_iter()
|
|
|
|
.map(|p| super::post_to_atom(p, &*conn))
|
|
|
|
.collect::<Vec<Entry>>(),
|
|
|
|
)
|
2018-09-01 22:08:26 +02:00
|
|
|
.build()
|
2018-10-20 11:04:20 +02:00
|
|
|
.expect("user::atom_feed: Error building Atom feed");
|
2018-11-24 12:44:17 +01:00
|
|
|
Some(Content(
|
|
|
|
ContentType::new("application", "atom+xml"),
|
|
|
|
feed.to_string(),
|
|
|
|
))
|
2018-09-01 22:08:26 +02:00
|
|
|
}
|