fedimovies/src/activitypub/activity.rs

600 lines
16 KiB
Rust
Raw Normal View History

2021-04-09 00:22:17 +00:00
use chrono::{DateTime, Utc};
use serde::{Deserialize, Serialize};
use serde_json::{json, Value};
use uuid::Uuid;
use crate::frontend::get_tag_page_url;
use crate::models::posts::types::{Post, Visibility};
2021-04-09 00:22:17 +00:00
use crate::models::profiles::types::DbActorProfile;
use crate::models::users::types::User;
2021-04-09 00:22:17 +00:00
use crate::utils::files::get_file_url;
use crate::utils::id::new_uuid;
use super::actor::{get_local_actor, ActorKeyError};
2021-04-09 00:22:17 +00:00
use super::constants::{AP_CONTEXT, AP_PUBLIC};
use super::views::{get_actor_url, get_followers_url, get_object_url};
2021-04-09 00:22:17 +00:00
use super::vocabulary::*;
#[derive(Deserialize, Serialize)]
#[serde(rename_all = "camelCase")]
pub struct Attachment {
pub name: Option<String>,
2021-04-09 00:22:17 +00:00
#[serde(rename = "type")]
pub attachment_type: String,
pub media_type: Option<String>,
2021-04-09 00:22:17 +00:00
pub url: String,
}
2021-11-11 21:51:47 +00:00
#[derive(Deserialize, Serialize)]
#[serde(rename_all = "camelCase")]
pub struct Tag {
pub name: String,
#[serde(rename = "type")]
pub tag_type: String,
pub href: Option<String>,
2021-11-11 21:51:47 +00:00
}
2021-10-29 16:33:50 +00:00
#[derive(Default, Deserialize, Serialize)]
2021-04-09 00:22:17 +00:00
#[serde(rename_all = "camelCase")]
pub struct Object {
#[serde(rename = "@context")]
pub context: Option<Value>,
pub id: String,
#[serde(rename = "type")]
pub object_type: String,
#[serde(skip_serializing_if = "Option::is_none")]
pub actor: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub attachment: Option<Vec<Attachment>>,
#[serde(skip_serializing_if = "Option::is_none")]
pub cc: Option<Value>,
#[serde(skip_serializing_if = "Option::is_none")]
pub former_type: Option<String>,
2021-04-09 00:22:17 +00:00
#[serde(skip_serializing_if = "Option::is_none")]
pub object: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub published: Option<DateTime<Utc>>,
#[serde(skip_serializing_if = "Option::is_none")]
pub attributed_to: Option<Value>,
2021-04-09 00:22:17 +00:00
#[serde(skip_serializing_if = "Option::is_none")]
pub in_reply_to: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub content: Option<String>,
2021-11-11 21:51:47 +00:00
#[serde(skip_serializing_if = "Option::is_none")]
pub tag: Option<Vec<Tag>>,
2021-04-09 00:22:17 +00:00
#[serde(skip_serializing_if = "Option::is_none")]
pub to: Option<Value>,
}
#[derive(Serialize)]
#[serde(rename_all = "camelCase")]
pub struct Note {
#[serde(rename = "@context")]
context: String,
id: String,
#[serde(rename = "type")]
object_type: String,
#[serde(skip_serializing_if = "Vec::is_empty")]
attachment: Vec<Attachment>,
attributed_to: String,
content: String,
#[serde(skip_serializing_if = "Option::is_none")]
in_reply_to: Option<String>,
published: DateTime<Utc>,
#[serde(skip_serializing_if = "Vec::is_empty")]
tag: Vec<Tag>,
to: Vec<String>,
cc: Vec<String>,
}
2021-04-09 00:22:17 +00:00
#[derive(Deserialize, Serialize)]
#[serde(rename_all = "camelCase")]
pub struct Activity {
#[serde(rename = "@context")]
pub context: Value,
pub id: String,
#[serde(rename = "type")]
pub activity_type: String,
pub actor: String,
pub object: Value,
pub to: Option<Value>,
pub cc: Option<Value>,
2021-04-09 00:22:17 +00:00
}
fn create_activity(
instance_url: &str,
actor_name: &str,
activity_type: &str,
internal_activity_id: Option<&Uuid>,
object: impl Serialize,
primary_audience: Vec<String>,
secondary_audience: Vec<String>,
2021-04-09 00:22:17 +00:00
) -> Activity {
let actor_id = get_actor_url(
instance_url,
2021-11-13 17:37:31 +00:00
actor_name,
2021-04-09 00:22:17 +00:00
);
2021-12-23 21:51:01 +00:00
let mut activity_id = get_object_url(
2021-04-09 00:22:17 +00:00
instance_url,
internal_activity_id.unwrap_or(&new_uuid()),
2021-04-09 00:22:17 +00:00
);
2021-12-23 21:51:01 +00:00
if activity_type == CREATE {
activity_id.push_str("/create");
};
2021-11-13 17:37:31 +00:00
Activity {
2021-04-09 00:22:17 +00:00
context: json!(AP_CONTEXT),
id: activity_id,
activity_type: activity_type.to_string(),
actor: actor_id,
object: serde_json::to_value(object).unwrap(),
to: Some(json!(primary_audience)),
cc: Some(json!(secondary_audience)),
2021-11-13 17:37:31 +00:00
}
2021-04-09 00:22:17 +00:00
}
pub fn create_note(
2021-11-11 21:51:47 +00:00
instance_host: &str,
2021-10-09 12:53:53 +00:00
instance_url: &str,
2021-04-09 00:22:17 +00:00
post: &Post,
) -> Note {
2021-04-09 00:22:17 +00:00
let object_id = get_object_url(
2021-10-09 12:53:53 +00:00
instance_url,
2021-04-09 00:22:17 +00:00
&post.id,
);
let actor_id = get_actor_url(
2021-10-09 12:53:53 +00:00
instance_url,
2021-04-09 00:22:17 +00:00
&post.author.username,
);
let attachments: Vec<Attachment> = post.attachments.iter().map(|db_item| {
2021-10-09 12:53:53 +00:00
let url = get_file_url(instance_url, &db_item.file_name);
let media_type = db_item.media_type.clone();
2021-04-09 00:22:17 +00:00
Attachment {
name: None,
2021-04-09 00:22:17 +00:00
attachment_type: DOCUMENT.to_string(),
media_type,
url,
}
}).collect();
let mut primary_audience = vec![];
let mut secondary_audience = vec![];
2021-12-07 23:28:58 +00:00
let mut tags = vec![];
if matches!(post.visibility, Visibility::Public) {
primary_audience.push(AP_PUBLIC.to_string());
secondary_audience.push(get_followers_url(
instance_url, &post.author.username,
));
};
2021-12-07 23:28:58 +00:00
for profile in &post.mentions {
let actor_id = profile.actor_id(instance_url);
primary_audience.push(actor_id);
let actor_url = profile.actor_url(instance_url);
2021-12-07 23:28:58 +00:00
let tag = Tag {
name: format!("@{}", profile.actor_address(instance_host)),
2021-11-11 21:51:47 +00:00
tag_type: MENTION.to_string(),
href: Some(actor_url),
2021-12-07 23:28:58 +00:00
};
tags.push(tag);
};
for tag_name in &post.tags {
let tag_page_url = get_tag_page_url(instance_url, tag_name);
2021-12-07 23:28:58 +00:00
let tag = Tag {
name: format!("#{}", tag_name),
tag_type: HASHTAG.to_string(),
href: Some(tag_page_url),
2021-12-07 23:28:58 +00:00
};
tags.push(tag);
};
let in_reply_to_object_id = match post.in_reply_to_id {
Some(in_reply_to_id) => {
let in_reply_to = post.in_reply_to.as_ref().unwrap();
assert_eq!(in_reply_to.id, in_reply_to_id);
let in_reply_to_actor_id = in_reply_to.author.actor_id(instance_url);
if !primary_audience.contains(&in_reply_to_actor_id) {
primary_audience.push(in_reply_to_actor_id);
};
Some(in_reply_to.get_object_id(instance_url))
},
None => None,
};
Note {
context: AP_CONTEXT.to_string(),
2021-04-09 00:22:17 +00:00
id: object_id,
object_type: NOTE.to_string(),
attachment: attachments,
published: post.created_at,
attributed_to: actor_id,
in_reply_to: in_reply_to_object_id,
content: post.content.clone(),
2021-12-07 23:28:58 +00:00
tag: tags,
to: primary_audience,
cc: secondary_audience,
}
}
pub fn create_activity_note(
2021-11-11 21:51:47 +00:00
instance_host: &str,
2021-10-09 12:53:53 +00:00
instance_url: &str,
post: &Post,
) -> Activity {
let object = create_note(instance_host, instance_url, post);
let primary_audience = object.to.clone();
let secondary_audience = object.cc.clone();
2021-04-09 00:22:17 +00:00
let activity = create_activity(
2021-10-09 12:53:53 +00:00
instance_url,
2021-04-09 00:22:17 +00:00
&post.author.username,
CREATE,
2021-12-23 21:51:01 +00:00
Some(&post.id),
object,
primary_audience,
secondary_audience,
2021-04-09 00:22:17 +00:00
);
activity
}
2021-10-29 19:21:26 +00:00
pub fn create_activity_like(
instance_url: &str,
actor_profile: &DbActorProfile,
note_id: &str,
reaction_id: &Uuid,
recipient_id: &str,
2021-10-29 19:21:26 +00:00
) -> Activity {
let activity = create_activity(
instance_url,
&actor_profile.username,
LIKE,
Some(reaction_id),
note_id,
vec![AP_PUBLIC.to_string(), recipient_id.to_string()],
vec![],
2021-10-29 19:21:26 +00:00
);
activity
}
pub fn create_activity_undo_like(
instance_url: &str,
actor_profile: &DbActorProfile,
reaction_id: &Uuid,
recipient_id: &str,
) -> Activity {
let object_id = get_object_url(
instance_url,
reaction_id,
);
create_activity(
instance_url,
&actor_profile.username,
UNDO,
None,
object_id,
vec![AP_PUBLIC.to_string(), recipient_id.to_string()],
vec![],
)
}
pub fn create_activity_announce(
instance_url: &str,
actor_profile: &DbActorProfile,
post: &Post,
repost_id: &Uuid,
) -> Activity {
let object_id = post.get_object_id(instance_url);
let recipient_id = post.author.actor_id(instance_url);
let activity = create_activity(
instance_url,
&actor_profile.username,
ANNOUNCE,
Some(repost_id),
object_id,
vec![AP_PUBLIC.to_string(), recipient_id],
vec![get_followers_url(instance_url, &actor_profile.username)],
);
activity
}
pub fn create_activity_undo_announce(
instance_url: &str,
actor_profile: &DbActorProfile,
repost_id: &Uuid,
recipient_id: &str,
) -> Activity {
let object_id = get_object_url(
instance_url,
repost_id,
);
let primary_audience = vec![
AP_PUBLIC.to_string(),
recipient_id.to_string(),
];
create_activity(
instance_url,
&actor_profile.username,
UNDO,
None,
object_id,
primary_audience,
vec![get_followers_url(instance_url, &actor_profile.username)],
)
}
2021-12-04 00:42:36 +00:00
pub fn create_activity_delete_note(
instance_url: &str,
post: &Post,
) -> Activity {
let object_id = post.get_object_id(instance_url);
let object = Object {
context: Some(json!(AP_CONTEXT)),
id: object_id,
object_type: TOMBSTONE.to_string(),
former_type: Some(NOTE.to_string()),
2021-12-04 00:42:36 +00:00
..Default::default()
};
let activity = create_activity(
instance_url,
&post.author.username,
2021-12-04 00:42:36 +00:00
DELETE,
None,
object,
vec![AP_PUBLIC.to_string()],
vec![],
2021-12-04 00:42:36 +00:00
);
activity
}
2021-04-09 00:22:17 +00:00
pub fn create_activity_follow(
2021-10-09 12:53:53 +00:00
instance_url: &str,
2021-04-09 00:22:17 +00:00
actor_profile: &DbActorProfile,
follow_request_id: &Uuid,
2021-10-29 16:33:50 +00:00
target_actor_id: &str,
2021-04-09 00:22:17 +00:00
) -> Activity {
let object = Object {
context: Some(json!(AP_CONTEXT)),
2021-10-29 16:33:50 +00:00
id: target_actor_id.to_owned(),
2021-04-09 00:22:17 +00:00
object_type: PERSON.to_string(),
2021-10-29 16:33:50 +00:00
..Default::default()
2021-04-09 00:22:17 +00:00
};
let activity = create_activity(
2021-10-09 12:53:53 +00:00
instance_url,
2021-04-09 00:22:17 +00:00
&actor_profile.username,
FOLLOW,
Some(follow_request_id),
object,
vec![target_actor_id.to_string()],
vec![],
2021-04-09 00:22:17 +00:00
);
activity
}
pub fn create_activity_accept_follow(
2021-10-09 12:53:53 +00:00
instance_url: &str,
2021-04-09 00:22:17 +00:00
actor_profile: &DbActorProfile,
follow_activity_id: &str,
2021-11-18 21:28:50 +00:00
source_actor_id: &str,
2021-04-09 00:22:17 +00:00
) -> Activity {
// TODO: use received activity as object
let object = Object {
context: Some(json!(AP_CONTEXT)),
id: follow_activity_id.to_string(),
object_type: FOLLOW.to_string(),
2021-10-29 16:33:50 +00:00
..Default::default()
2021-04-09 00:22:17 +00:00
};
let activity = create_activity(
2021-10-09 12:53:53 +00:00
instance_url,
2021-04-09 00:22:17 +00:00
&actor_profile.username,
ACCEPT,
None,
object,
vec![source_actor_id.to_string()],
vec![],
2021-04-09 00:22:17 +00:00
);
activity
}
pub fn create_activity_undo_follow(
2021-10-09 12:53:53 +00:00
instance_url: &str,
2021-04-09 00:22:17 +00:00
actor_profile: &DbActorProfile,
follow_request_id: &Uuid,
2021-10-29 16:33:50 +00:00
target_actor_id: &str,
2021-04-09 00:22:17 +00:00
) -> Activity {
// TODO: retrieve 'Follow' activity from database
let follow_activity_id = get_object_url(
2021-10-09 12:53:53 +00:00
instance_url,
2021-04-09 00:22:17 +00:00
follow_request_id,
);
let follow_actor_id = get_actor_url(
2021-10-09 12:53:53 +00:00
instance_url,
2021-04-09 00:22:17 +00:00
&actor_profile.username,
);
let object = Object {
context: Some(json!(AP_CONTEXT)),
id: follow_activity_id,
object_type: FOLLOW.to_string(),
actor: Some(follow_actor_id),
2021-10-29 16:33:50 +00:00
object: Some(target_actor_id.to_owned()),
..Default::default()
2021-04-09 00:22:17 +00:00
};
let activity = create_activity(
2021-10-09 12:53:53 +00:00
instance_url,
2021-04-09 00:22:17 +00:00
&actor_profile.username,
UNDO,
None,
object,
vec![target_actor_id.to_string()],
vec![],
2021-04-09 00:22:17 +00:00
);
activity
}
pub fn create_activity_update_person(
user: &User,
instance_url: &str,
) -> Result<Activity, ActorKeyError> {
let actor = get_local_actor(user, instance_url)?;
let activity = create_activity(
instance_url,
&user.profile.username,
UPDATE,
None,
actor,
vec![
AP_PUBLIC.to_string(),
get_followers_url(instance_url, &user.profile.username),
],
vec![],
);
Ok(activity)
}
2021-10-09 12:53:53 +00:00
#[cfg(test)]
mod tests {
use crate::activitypub::actor::Actor;
2021-10-09 12:53:53 +00:00
use super::*;
2021-11-11 21:51:47 +00:00
const INSTANCE_HOST: &str = "example.com";
2021-10-09 12:53:53 +00:00
const INSTANCE_URL: &str = "https://example.com";
#[test]
fn test_create_note() {
let author = DbActorProfile {
username: "author".to_string(),
..Default::default()
};
let post = Post { author, ..Default::default() };
let note = create_note(INSTANCE_HOST, INSTANCE_URL, &post);
2021-10-09 12:53:53 +00:00
assert_eq!(
note.id,
format!("{}/objects/{}", INSTANCE_URL, post.id),
);
assert_eq!(note.attachment.len(), 0);
2021-10-09 12:53:53 +00:00
assert_eq!(
note.attributed_to,
2021-10-09 12:53:53 +00:00
format!("{}/users/{}", INSTANCE_URL, post.author.username),
);
assert_eq!(note.in_reply_to.is_none(), true);
assert_eq!(note.content, post.content);
assert_eq!(note.to, vec![AP_PUBLIC]);
assert_eq!(note.cc, vec![
get_followers_url(INSTANCE_URL, "author"),
]);
2021-10-09 12:53:53 +00:00
}
#[test]
fn test_create_note_with_local_parent() {
let parent = Post::default();
let post = Post {
in_reply_to_id: Some(parent.id),
in_reply_to: Some(Box::new(parent.clone())),
2021-10-09 12:53:53 +00:00
..Default::default()
};
let note = create_note(INSTANCE_HOST, INSTANCE_URL, &post);
2021-10-09 12:53:53 +00:00
assert_eq!(
note.in_reply_to.unwrap(),
format!("{}/objects/{}", INSTANCE_URL, parent.id),
);
assert_eq!(note.to, vec![
AP_PUBLIC.to_string(),
get_actor_url(INSTANCE_URL, &parent.author.username),
]);
2021-10-09 12:53:53 +00:00
}
#[test]
fn test_create_note_with_remote_parent() {
2021-11-11 21:51:47 +00:00
let parent_author_acct = "test@test.net";
let parent_author_actor_id = "https://test.net/user/test";
let parent_author_actor_url = "https://test.net/@test";
2021-10-09 12:53:53 +00:00
let parent_author = DbActorProfile {
2021-11-11 21:51:47 +00:00
acct: parent_author_acct.to_string(),
actor_json: Some(Actor {
id: parent_author_actor_id.to_string(),
url: Some(parent_author_actor_url.to_string()),
..Default::default()
}),
2021-10-09 12:53:53 +00:00
..Default::default()
};
let parent = Post {
2021-11-11 21:51:47 +00:00
author: parent_author.clone(),
2021-10-09 12:53:53 +00:00
object_id: Some("https://test.net/obj/123".to_string()),
..Default::default()
};
let post = Post {
in_reply_to_id: Some(parent.id),
in_reply_to: Some(Box::new(parent.clone())),
2021-11-11 21:51:47 +00:00
mentions: vec![parent_author],
2021-10-09 12:53:53 +00:00
..Default::default()
};
let note = create_note(INSTANCE_HOST, INSTANCE_URL, &post);
2021-10-09 12:53:53 +00:00
assert_eq!(
note.in_reply_to.unwrap(),
parent.object_id.unwrap(),
);
let tags = note.tag;
2021-11-11 21:51:47 +00:00
assert_eq!(tags.len(), 1);
assert_eq!(tags[0].name, format!("@{}", parent_author_acct));
assert_eq!(tags[0].href.as_ref().unwrap(), parent_author_actor_url);
assert_eq!(note.to, vec![AP_PUBLIC, parent_author_actor_id]);
2021-10-09 12:53:53 +00:00
}
2021-10-29 16:33:50 +00:00
#[test]
fn test_create_activity_follow() {
let follower = DbActorProfile {
username: "follower".to_string(),
..Default::default()
};
let follow_request_id = new_uuid();
2021-10-29 16:33:50 +00:00
let target_actor_id = "https://example.com/actor/test";
let activity = create_activity_follow(
INSTANCE_URL,
&follower,
&follow_request_id,
target_actor_id,
);
assert_eq!(
activity.id,
format!("{}/objects/{}", INSTANCE_URL, follow_request_id),
);
assert_eq!(activity.activity_type, "Follow");
assert_eq!(
activity.actor,
format!("{}/users/{}", INSTANCE_URL, follower.username),
);
assert_eq!(activity.object["id"], target_actor_id);
assert_eq!(activity.object["type"], "Person");
assert_eq!(activity.object["actor"], Value::Null);
assert_eq!(activity.object["object"], Value::Null);
assert_eq!(activity.object["content"], Value::Null);
assert_eq!(activity.to.unwrap(), json!([target_actor_id]));
assert_eq!(activity.cc.unwrap(), json!([]));
2021-10-29 16:33:50 +00:00
}
2021-10-09 12:53:53 +00:00
}