603 lines
21 KiB
Rust
603 lines
21 KiB
Rust
use crate::{
|
|
comment_seers::{CommentSeers, NewCommentSeers},
|
|
instance::Instance,
|
|
medias::Media,
|
|
mentions::Mention,
|
|
notifications::*,
|
|
posts::Post,
|
|
safe_string::SafeString,
|
|
schema::comments,
|
|
users::User,
|
|
Connection, Error, Result, CONFIG,
|
|
};
|
|
use activitystreams::{
|
|
activity::{Create, Delete},
|
|
base::{AnyBase, Base},
|
|
iri_string::types::IriString,
|
|
link::{self, kind::MentionType},
|
|
object::{Note, Tombstone},
|
|
prelude::*,
|
|
primitives::OneOrMany,
|
|
time::OffsetDateTime,
|
|
};
|
|
use chrono::{self, NaiveDateTime};
|
|
use diesel::{self, ExpressionMethods, QueryDsl, RunQueryDsl, SaveChangesDsl};
|
|
use plume_common::{
|
|
activity_pub::{
|
|
inbox::{AsActor, AsObject, FromId},
|
|
sign::Signer,
|
|
IntoId, ToAsString, ToAsUri, PUBLIC_VISIBILITY,
|
|
},
|
|
utils,
|
|
};
|
|
use std::collections::HashSet;
|
|
|
|
#[derive(Queryable, Identifiable, Clone, AsChangeset)]
|
|
pub struct Comment {
|
|
pub id: i32,
|
|
pub content: SafeString,
|
|
pub in_response_to_id: Option<i32>,
|
|
pub post_id: i32,
|
|
pub author_id: i32,
|
|
pub creation_date: NaiveDateTime,
|
|
pub ap_url: Option<String>,
|
|
pub sensitive: bool,
|
|
pub spoiler_text: String,
|
|
pub public_visibility: bool,
|
|
}
|
|
|
|
#[derive(Insertable, Default)]
|
|
#[table_name = "comments"]
|
|
pub struct NewComment {
|
|
pub content: SafeString,
|
|
pub in_response_to_id: Option<i32>,
|
|
pub post_id: i32,
|
|
pub author_id: i32,
|
|
pub ap_url: Option<String>,
|
|
pub sensitive: bool,
|
|
pub spoiler_text: String,
|
|
pub public_visibility: bool,
|
|
}
|
|
|
|
impl Comment {
|
|
insert!(comments, NewComment, |inserted, conn| {
|
|
if inserted.ap_url.is_none() {
|
|
inserted.ap_url = Some(format!(
|
|
"{}/comment/{}",
|
|
inserted.get_post(conn)?.ap_url,
|
|
inserted.id
|
|
));
|
|
let _: Comment = inserted.save_changes(conn)?;
|
|
}
|
|
Ok(inserted)
|
|
});
|
|
get!(comments);
|
|
list_by!(comments, list_by_post, post_id as i32);
|
|
list_by!(comments, list_by_author, author_id as i32);
|
|
find_by!(comments, find_by_ap_url, ap_url as &str);
|
|
|
|
pub fn get_author(&self, conn: &Connection) -> Result<User> {
|
|
User::get(conn, self.author_id)
|
|
}
|
|
|
|
pub fn get_post(&self, conn: &Connection) -> Result<Post> {
|
|
Post::get(conn, self.post_id)
|
|
}
|
|
|
|
pub fn count_local(conn: &Connection) -> Result<i64> {
|
|
use crate::schema::users;
|
|
let local_authors = users::table
|
|
.filter(users::instance_id.eq(Instance::get_local()?.id))
|
|
.select(users::id);
|
|
comments::table
|
|
.filter(comments::author_id.eq_any(local_authors))
|
|
.count()
|
|
.get_result(conn)
|
|
.map_err(Error::from)
|
|
}
|
|
|
|
pub fn get_responses(&self, conn: &Connection) -> Result<Vec<Comment>> {
|
|
comments::table
|
|
.filter(comments::in_response_to_id.eq(self.id))
|
|
.load::<Comment>(conn)
|
|
.map_err(Error::from)
|
|
}
|
|
|
|
pub fn can_see(&self, conn: &Connection, user: Option<&User>) -> bool {
|
|
self.public_visibility
|
|
|| user
|
|
.as_ref()
|
|
.map(|u| CommentSeers::can_see(conn, self, u).unwrap_or(false))
|
|
.unwrap_or(false)
|
|
}
|
|
|
|
pub fn to_activity(&self, conn: &Connection) -> Result<Note> {
|
|
let author = User::get(conn, self.author_id)?;
|
|
let (html, mentions, _hashtags) = utils::md_to_html(
|
|
self.content.get().as_ref(),
|
|
Some(&Instance::get_local()?.public_domain),
|
|
true,
|
|
Some(Media::get_media_processor(conn, vec![&author])),
|
|
);
|
|
|
|
let mut note = Note::new();
|
|
let to = vec![PUBLIC_VISIBILITY.parse::<IriString>()?];
|
|
|
|
note.set_id(
|
|
self.ap_url
|
|
.clone()
|
|
.unwrap_or_default()
|
|
.parse::<IriString>()?,
|
|
);
|
|
note.set_summary(self.spoiler_text.clone());
|
|
note.set_content(html);
|
|
note.set_in_reply_to(self.in_response_to_id.map_or_else(
|
|
|| Post::get(conn, self.post_id).map(|post| post.ap_url),
|
|
|id| Comment::get(conn, id).map(|comment| comment.ap_url.unwrap_or_default()),
|
|
)?);
|
|
note.set_published(
|
|
OffsetDateTime::from_unix_timestamp_nanos(self.creation_date.timestamp_nanos_opt().unwrap().into())
|
|
.expect("OffsetDateTime"),
|
|
);
|
|
note.set_attributed_to(author.into_id().parse::<IriString>()?);
|
|
note.set_many_tos(to);
|
|
note.set_many_tags(mentions.into_iter().filter_map(|m| {
|
|
Mention::build_activity(conn, &m)
|
|
.map(|mention| mention.into_any_base().expect("Can convert"))
|
|
.ok()
|
|
}));
|
|
Ok(note)
|
|
}
|
|
|
|
pub fn create_activity(&self, conn: &Connection) -> Result<Create> {
|
|
let author = User::get(conn, self.author_id)?;
|
|
|
|
let note = self.to_activity(conn)?;
|
|
let note_clone = note.clone();
|
|
|
|
let mut act = Create::new(
|
|
author.into_id().parse::<IriString>()?,
|
|
Base::retract(note)?.into_generic()?,
|
|
);
|
|
act.set_id(
|
|
format!(
|
|
"{}/activity",
|
|
self.ap_url.clone().ok_or(Error::MissingApProperty)?,
|
|
)
|
|
.parse::<IriString>()?,
|
|
);
|
|
act.set_many_tos(
|
|
note_clone
|
|
.to()
|
|
.iter()
|
|
.flat_map(|tos| tos.iter().map(|to| to.to_owned())),
|
|
);
|
|
act.set_many_ccs(vec![self.get_author(conn)?.followers_endpoint]);
|
|
Ok(act)
|
|
}
|
|
|
|
pub fn notify(&self, conn: &Connection) -> Result<()> {
|
|
for author in self.get_post(conn)?.get_authors(conn)? {
|
|
if Mention::list_for_comment(conn, self.id)?
|
|
.iter()
|
|
.all(|m| m.get_mentioned(conn).map(|u| u != author).unwrap_or(true))
|
|
&& author.is_local()
|
|
{
|
|
Notification::insert(
|
|
conn,
|
|
NewNotification {
|
|
kind: notification_kind::COMMENT.to_string(),
|
|
object_id: self.id,
|
|
user_id: author.id,
|
|
},
|
|
)?;
|
|
}
|
|
}
|
|
Ok(())
|
|
}
|
|
|
|
pub fn build_delete(&self, conn: &Connection) -> Result<Delete> {
|
|
let mut tombstone = Tombstone::new();
|
|
tombstone.set_id(
|
|
self.ap_url
|
|
.as_ref()
|
|
.ok_or(Error::MissingApProperty)?
|
|
.parse::<IriString>()?,
|
|
);
|
|
|
|
let mut act = Delete::new(
|
|
self.get_author(conn)?.into_id().parse::<IriString>()?,
|
|
Base::retract(tombstone)?.into_generic()?,
|
|
);
|
|
|
|
act.set_id(format!("{}#delete", self.ap_url.clone().unwrap()).parse::<IriString>()?);
|
|
act.set_many_tos(vec![PUBLIC_VISIBILITY.parse::<IriString>()?]);
|
|
|
|
Ok(act)
|
|
}
|
|
}
|
|
|
|
impl FromId<Connection> for Comment {
|
|
type Error = Error;
|
|
type Object = Note;
|
|
|
|
fn from_db(conn: &Connection, id: &str) -> Result<Self> {
|
|
Self::find_by_ap_url(conn, id)
|
|
}
|
|
|
|
fn from_activity(conn: &Connection, note: Note) -> Result<Self> {
|
|
let comm = {
|
|
let previous_url = note
|
|
.in_reply_to()
|
|
.ok_or(Error::MissingApProperty)?
|
|
.iter()
|
|
.next()
|
|
.ok_or(Error::MissingApProperty)?
|
|
.id()
|
|
.ok_or(Error::MissingApProperty)?;
|
|
let previous_comment = Comment::find_by_ap_url(conn, previous_url.as_str());
|
|
|
|
let is_public = |v: &Option<&OneOrMany<AnyBase>>| match v {
|
|
Some(one_or_many) => one_or_many.iter().any(|any_base| {
|
|
let id = any_base.id();
|
|
id.is_some() && id.unwrap() == PUBLIC_VISIBILITY
|
|
}),
|
|
None => false,
|
|
};
|
|
|
|
let public_visibility = is_public(¬e.to())
|
|
|| is_public(¬e.bto())
|
|
|| is_public(¬e.cc())
|
|
|| is_public(¬e.bcc());
|
|
|
|
let summary = note.summary().and_then(|summary| summary.to_as_string());
|
|
let sensitive = summary.is_some();
|
|
let comm = Comment::insert(
|
|
conn,
|
|
NewComment {
|
|
content: SafeString::new(
|
|
¬e
|
|
.content()
|
|
.ok_or(Error::MissingApProperty)?
|
|
.to_as_string()
|
|
.ok_or(Error::InvalidValue)?,
|
|
),
|
|
spoiler_text: summary.unwrap_or_default(),
|
|
ap_url: Some(
|
|
note.id_unchecked()
|
|
.ok_or(Error::MissingApProperty)?
|
|
.to_string(),
|
|
),
|
|
in_response_to_id: previous_comment.iter().map(|c| c.id).next(),
|
|
post_id: previous_comment.map(|c| c.post_id).or_else(|_| {
|
|
Ok(Post::find_by_ap_url(conn, previous_url.as_str())?.id) as Result<i32>
|
|
})?,
|
|
author_id: User::from_id(
|
|
conn,
|
|
¬e
|
|
.attributed_to()
|
|
.ok_or(Error::MissingApProperty)?
|
|
.to_as_uri()
|
|
.ok_or(Error::MissingApProperty)?,
|
|
None,
|
|
CONFIG.proxy(),
|
|
)
|
|
.map_err(|(_, e)| e)?
|
|
.id,
|
|
sensitive,
|
|
public_visibility,
|
|
},
|
|
)?;
|
|
|
|
// save mentions
|
|
if let Some(tags) = note.tag() {
|
|
let author_url = &Post::get(conn, comm.post_id)?.get_authors(conn)?[0].ap_url;
|
|
for tag in tags.iter() {
|
|
let m = tag.clone().extend::<link::Mention, MentionType>()?; // FIXME: Don't clone
|
|
if m.is_none() {
|
|
continue;
|
|
}
|
|
let m = m.unwrap();
|
|
let not_author = m.href().ok_or(Error::MissingApProperty)? != author_url;
|
|
let _ = Mention::from_activity(conn, &m, comm.id, false, not_author);
|
|
}
|
|
}
|
|
comm
|
|
};
|
|
|
|
if !comm.public_visibility {
|
|
let mut receiver_ids = HashSet::new();
|
|
let mut receivers_id = |v: Option<&'_ OneOrMany<AnyBase>>| {
|
|
if let Some(one_or_many) = v {
|
|
for any_base in one_or_many.iter() {
|
|
if let Some(id) = any_base.id() {
|
|
receiver_ids.insert(id.to_string());
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
receivers_id(note.to());
|
|
receivers_id(note.cc());
|
|
receivers_id(note.bto());
|
|
receivers_id(note.bcc());
|
|
|
|
let receivers_ap_url = receiver_ids
|
|
.into_iter()
|
|
.flat_map(|v| {
|
|
if let Ok(user) = User::from_id(conn, v.as_ref(), None, CONFIG.proxy()) {
|
|
vec![user]
|
|
} else {
|
|
vec![] // TODO try to fetch collection
|
|
}
|
|
})
|
|
.filter(|u| u.get_instance(conn).map(|i| i.local).unwrap_or(false))
|
|
.collect::<HashSet<User>>(); //remove duplicates (prevent db error)
|
|
|
|
for user in &receivers_ap_url {
|
|
CommentSeers::insert(
|
|
conn,
|
|
NewCommentSeers {
|
|
comment_id: comm.id,
|
|
user_id: user.id,
|
|
},
|
|
)?;
|
|
}
|
|
}
|
|
|
|
comm.notify(conn)?;
|
|
Ok(comm)
|
|
}
|
|
|
|
fn get_sender() -> &'static dyn Signer {
|
|
Instance::get_local_instance_user().expect("Failed to local instance user")
|
|
}
|
|
}
|
|
|
|
impl AsObject<User, Create, &Connection> for Comment {
|
|
type Error = Error;
|
|
type Output = Self;
|
|
|
|
fn activity(self, _conn: &Connection, _actor: User, _id: &str) -> Result<Self> {
|
|
// The actual creation takes place in the FromId impl
|
|
Ok(self)
|
|
}
|
|
}
|
|
|
|
impl AsObject<User, Delete, &Connection> for Comment {
|
|
type Error = Error;
|
|
type Output = ();
|
|
|
|
fn activity(self, conn: &Connection, actor: User, _id: &str) -> Result<()> {
|
|
if self.author_id != actor.id {
|
|
return Err(Error::Unauthorized);
|
|
}
|
|
|
|
for m in Mention::list_for_comment(conn, self.id)? {
|
|
for n in Notification::find_for_mention(conn, &m)? {
|
|
n.delete(conn)?;
|
|
}
|
|
m.delete(conn)?;
|
|
}
|
|
|
|
for n in Notification::find_for_comment(conn, &self)? {
|
|
n.delete(conn)?;
|
|
}
|
|
|
|
diesel::update(comments::table)
|
|
.filter(comments::in_response_to_id.eq(self.id))
|
|
.set(comments::in_response_to_id.eq(self.in_response_to_id))
|
|
.execute(conn)?;
|
|
diesel::delete(&self).execute(conn)?;
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
pub struct CommentTree {
|
|
pub comment: Comment,
|
|
pub responses: Vec<CommentTree>,
|
|
}
|
|
|
|
impl CommentTree {
|
|
pub fn from_post(conn: &Connection, p: &Post, user: Option<&User>) -> Result<Vec<Self>> {
|
|
Ok(Comment::list_by_post(conn, p.id)?
|
|
.into_iter()
|
|
.filter(|c| c.in_response_to_id.is_none())
|
|
.filter(|c| c.can_see(conn, user))
|
|
.filter_map(|c| Self::from_comment(conn, c, user).ok())
|
|
.collect())
|
|
}
|
|
|
|
pub fn from_comment(conn: &Connection, comment: Comment, user: Option<&User>) -> Result<Self> {
|
|
let responses = comment
|
|
.get_responses(conn)?
|
|
.into_iter()
|
|
.filter(|c| c.can_see(conn, user))
|
|
.filter_map(|c| Self::from_comment(conn, c, user).ok())
|
|
.collect();
|
|
Ok(CommentTree { comment, responses })
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use super::*;
|
|
use crate::blogs::Blog;
|
|
use crate::db_conn::DbConn;
|
|
use crate::inbox::{inbox, tests::fill_database, InboxResult};
|
|
use crate::safe_string::SafeString;
|
|
use crate::tests::{db, format_datetime};
|
|
use assert_json_diff::assert_json_eq;
|
|
use diesel::Connection;
|
|
use serde_json::{json, to_value};
|
|
|
|
fn prepare_activity(conn: &DbConn) -> (Comment, Vec<Post>, Vec<User>, Vec<Blog>) {
|
|
let (posts, users, blogs) = fill_database(conn);
|
|
|
|
let comment = Comment::insert(
|
|
conn,
|
|
NewComment {
|
|
content: SafeString::new("My comment, mentioning to @user"),
|
|
in_response_to_id: None,
|
|
post_id: posts[0].id,
|
|
author_id: users[0].id,
|
|
ap_url: None,
|
|
sensitive: true,
|
|
spoiler_text: "My CW".into(),
|
|
public_visibility: true,
|
|
},
|
|
)
|
|
.unwrap();
|
|
|
|
(comment, posts, users, blogs)
|
|
}
|
|
|
|
// creates a post, get it's Create activity, delete the post,
|
|
// "send" the Create to the inbox, and check it works
|
|
#[test]
|
|
fn self_federation() {
|
|
let conn = &db();
|
|
conn.test_transaction::<_, (), _>(|| {
|
|
let (original_comm, posts, users, _blogs) = prepare_activity(conn);
|
|
let act = original_comm.create_activity(conn).unwrap();
|
|
|
|
assert_json_eq!(to_value(&act).unwrap(), json!({
|
|
"actor": "https://plu.me/@/admin/",
|
|
"cc": ["https://plu.me/@/admin/followers"],
|
|
"id": format!("https://plu.me/~/BlogName/testing/comment/{}/activity", original_comm.id),
|
|
"object": {
|
|
"attributedTo": "https://plu.me/@/admin/",
|
|
"content": r###"<p dir="auto">My comment, mentioning to <a href="https://plu.me/@/user/" title="user">@user</a></p>
|
|
"###,
|
|
"id": format!("https://plu.me/~/BlogName/testing/comment/{}", original_comm.id),
|
|
"inReplyTo": "https://plu.me/~/BlogName/testing",
|
|
"published": format_datetime(&original_comm.creation_date),
|
|
"summary": "My CW",
|
|
"tag": [
|
|
{
|
|
"href": "https://plu.me/@/user/",
|
|
"name": "@user",
|
|
"type": "Mention"
|
|
}
|
|
],
|
|
"to": ["https://www.w3.org/ns/activitystreams#Public"],
|
|
"type": "Note"
|
|
},
|
|
"to": ["https://www.w3.org/ns/activitystreams#Public"],
|
|
"type": "Create",
|
|
}));
|
|
|
|
let reply = Comment::insert(
|
|
conn,
|
|
NewComment {
|
|
content: SafeString::new(""),
|
|
in_response_to_id: Some(original_comm.id),
|
|
post_id: posts[0].id,
|
|
author_id: users[1].id,
|
|
ap_url: None,
|
|
sensitive: false,
|
|
spoiler_text: "".into(),
|
|
public_visibility: true,
|
|
},
|
|
)
|
|
.unwrap();
|
|
let reply_act = reply.create_activity(conn).unwrap();
|
|
|
|
assert_json_eq!(to_value(&reply_act).unwrap(), json!({
|
|
"actor": "https://plu.me/@/user/",
|
|
"cc": ["https://plu.me/@/user/followers"],
|
|
"id": format!("https://plu.me/~/BlogName/testing/comment/{}/activity", reply.id),
|
|
"object": {
|
|
"attributedTo": "https://plu.me/@/user/",
|
|
"content": "",
|
|
"id": format!("https://plu.me/~/BlogName/testing/comment/{}", reply.id),
|
|
"inReplyTo": format!("https://plu.me/~/BlogName/testing/comment/{}", original_comm.id),
|
|
"published": format_datetime(&reply.creation_date),
|
|
"summary": "",
|
|
"tag": [],
|
|
"to": ["https://www.w3.org/ns/activitystreams#Public"],
|
|
"type": "Note"
|
|
},
|
|
"to": ["https://www.w3.org/ns/activitystreams#Public"],
|
|
"type": "Create"
|
|
}));
|
|
|
|
inbox(
|
|
conn,
|
|
serde_json::to_value(original_comm.build_delete(conn).unwrap()).unwrap(),
|
|
)
|
|
.unwrap();
|
|
|
|
match inbox(conn, to_value(act).unwrap()).unwrap() {
|
|
InboxResult::Commented(c) => {
|
|
// TODO: one is HTML, the other markdown: assert_eq!(c.content, original_comm.content);
|
|
assert_eq!(c.in_response_to_id, original_comm.in_response_to_id);
|
|
assert_eq!(c.post_id, original_comm.post_id);
|
|
assert_eq!(c.author_id, original_comm.author_id);
|
|
assert_eq!(c.ap_url, original_comm.ap_url);
|
|
assert_eq!(c.spoiler_text, original_comm.spoiler_text);
|
|
assert_eq!(c.public_visibility, original_comm.public_visibility);
|
|
}
|
|
_ => panic!("Unexpected result"),
|
|
};
|
|
Ok(())
|
|
})
|
|
}
|
|
|
|
#[test]
|
|
fn to_activity() {
|
|
let conn = db();
|
|
conn.test_transaction::<_, Error, _>(|| {
|
|
let (comment, _posts, _users, _blogs) = prepare_activity(&conn);
|
|
let act = comment.to_activity(&conn)?;
|
|
|
|
let expected = json!({
|
|
"attributedTo": "https://plu.me/@/admin/",
|
|
"content": r###"<p dir="auto">My comment, mentioning to <a href="https://plu.me/@/user/" title="user">@user</a></p>
|
|
"###,
|
|
"id": format!("https://plu.me/~/BlogName/testing/comment/{}", comment.id),
|
|
"inReplyTo": "https://plu.me/~/BlogName/testing",
|
|
"published": format_datetime(&comment.creation_date),
|
|
"summary": "My CW",
|
|
"tag": [
|
|
{
|
|
"href": "https://plu.me/@/user/",
|
|
"name": "@user",
|
|
"type": "Mention"
|
|
}
|
|
],
|
|
"to": ["https://www.w3.org/ns/activitystreams#Public"],
|
|
"type": "Note"
|
|
});
|
|
|
|
assert_json_eq!(to_value(act)?, expected);
|
|
|
|
Ok(())
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn build_delete() {
|
|
let conn = db();
|
|
conn.test_transaction::<_, Error, _>(|| {
|
|
let (comment, _posts, _users, _blogs) = prepare_activity(&conn);
|
|
let act = comment.build_delete(&conn)?;
|
|
|
|
let expected = json!({
|
|
"actor": "https://plu.me/@/admin/",
|
|
"id": format!("https://plu.me/~/BlogName/testing/comment/{}#delete", comment.id),
|
|
"object": {
|
|
"id": format!("https://plu.me/~/BlogName/testing/comment/{}", comment.id),
|
|
"type": "Tombstone"
|
|
},
|
|
"to": ["https://www.w3.org/ns/activitystreams#Public"],
|
|
"type": "Delete"
|
|
});
|
|
|
|
assert_json_eq!(to_value(act)?, expected);
|
|
|
|
Ok(())
|
|
});
|
|
}
|
|
}
|