2 activity_lists::AnnouncableActivities,
3 objects::{community::ApubCommunity, post::ApubPost},
6 community::announce::AnnounceActivity,
7 create_or_update::page::CreateOrUpdatePage,
10 collections::group_outbox::GroupOutbox,
13 use activitypub_federation::{
15 kinds::collection::OrderedCollectionType,
16 protocol::verification::verify_domains_match,
17 traits::{ActivityHandler, Collection},
19 use futures::future::join_all;
20 use lemmy_api_common::{context::LemmyContext, utils::generate_outbox_url};
21 use lemmy_db_schema::{
22 source::{person::Person, post::Post},
24 utils::FETCH_LIMIT_MAX,
26 use lemmy_utils::error::LemmyError;
29 #[derive(Clone, Debug)]
30 pub(crate) struct ApubCommunityOutbox(Vec<ApubPost>);
32 #[async_trait::async_trait]
33 impl Collection for ApubCommunityOutbox {
34 type Owner = ApubCommunity;
35 type DataType = LemmyContext;
36 type Kind = GroupOutbox;
37 type Error = LemmyError;
39 #[tracing::instrument(skip_all)]
42 data: &Data<Self::DataType>,
43 ) -> Result<Self::Kind, LemmyError> {
44 let post_list: Vec<ApubPost> = Post::list_for_community(data.pool(), owner.id)
49 let mut ordered_items = vec![];
50 for post in post_list {
51 let person = Person::read(data.pool(), post.creator_id).await?.into();
53 CreateOrUpdatePage::new(post, &person, owner, CreateOrUpdateType::Create, data).await?;
54 let announcable = AnnouncableActivities::CreateOrUpdatePost(create);
55 let announce = AnnounceActivity::new(announcable.try_into()?, owner, data)?;
56 ordered_items.push(announce);
60 r#type: OrderedCollectionType::OrderedCollection,
61 id: generate_outbox_url(&owner.actor_id)?.into(),
62 total_items: ordered_items.len() as i32,
67 #[tracing::instrument(skip_all)]
69 group_outbox: &GroupOutbox,
70 expected_domain: &Url,
71 _data: &Data<Self::DataType>,
72 ) -> Result<(), LemmyError> {
73 verify_domains_match(expected_domain, &group_outbox.id)?;
77 #[tracing::instrument(skip_all)]
81 data: &Data<Self::DataType>,
82 ) -> Result<Self, LemmyError> {
83 let mut outbox_activities = apub.ordered_items;
84 if outbox_activities.len() as i64 > FETCH_LIMIT_MAX {
85 outbox_activities = outbox_activities
86 .get(0..(FETCH_LIMIT_MAX as usize))
91 // We intentionally ignore errors here. This is because the outbox might contain posts from old
92 // Lemmy versions, or from other software which we cant parse. In that case, we simply skip the
93 // item and only parse the ones that work.
94 // process items in parallel, to avoid long delay from fetch_site_metadata() and other processing
95 join_all(outbox_activities.into_iter().map(|activity| {
97 // use separate request counter for each item, otherwise there will be problems with
98 // parallel processing
99 let verify = activity.verify(data).await;
101 activity.receive(data).await.ok();
107 // This return value is unused, so just set an empty vec
108 Ok(ApubCommunityOutbox(Vec::new()))