602 lines
		
	
	
		
			21 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
			
		
		
	
	
			602 lines
		
	
	
		
			21 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
use crate::{
 | 
						|
    comment_seers::{CommentSeers, NewCommentSeers},
 | 
						|
    db_conn::DbConn,
 | 
						|
    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);
 | 
						|
    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: &DbConn) -> 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().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: &DbConn) -> 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<DbConn> for Comment {
 | 
						|
    type Error = Error;
 | 
						|
    type Object = Note;
 | 
						|
 | 
						|
    fn from_db(conn: &DbConn, id: &str) -> Result<Self> {
 | 
						|
        Self::find_by_ap_url(conn, id)
 | 
						|
    }
 | 
						|
 | 
						|
    fn from_activity(conn: &DbConn, 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, &DbConn> for Comment {
 | 
						|
    type Error = Error;
 | 
						|
    type Output = Self;
 | 
						|
 | 
						|
    fn activity(self, _conn: &DbConn, _actor: User, _id: &str) -> Result<Self> {
 | 
						|
        // The actual creation takes place in the FromId impl
 | 
						|
        Ok(self)
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
impl AsObject<User, Delete, &DbConn> for Comment {
 | 
						|
    type Error = Error;
 | 
						|
    type Output = ();
 | 
						|
 | 
						|
    fn activity(self, conn: &DbConn, 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::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(())
 | 
						|
        });
 | 
						|
    }
 | 
						|
}
 |