]> Untitled Git - lemmy.git/blob - crates/apub/src/http/person.rs
Extract Activitypub logic into separate library (#2288)
[lemmy.git] / crates / apub / src / http / person.rs
1 use crate::{
2   activity_lists::PersonInboxActivities,
3   fetcher::user_or_community::UserOrCommunity,
4   generate_outbox_url,
5   http::{create_apub_response, create_apub_tombstone_response, receive_lemmy_activity},
6   objects::person::ApubPerson,
7   protocol::collections::empty_outbox::EmptyOutbox,
8 };
9 use activitypub_federation::{deser::context::WithContext, traits::ApubObject};
10 use actix_web::{web, HttpRequest, HttpResponse};
11 use lemmy_api_common::utils::blocking;
12 use lemmy_db_schema::{source::person::Person, traits::ApubActor};
13 use lemmy_utils::error::LemmyError;
14 use lemmy_websocket::LemmyContext;
15 use serde::Deserialize;
16
17 #[derive(Deserialize)]
18 pub struct PersonQuery {
19   user_name: String,
20 }
21
22 /// Return the ActivityPub json representation of a local person over HTTP.
23 #[tracing::instrument(skip_all)]
24 pub(crate) async fn get_apub_person_http(
25   info: web::Path<PersonQuery>,
26   context: web::Data<LemmyContext>,
27 ) -> Result<HttpResponse, LemmyError> {
28   let user_name = info.into_inner().user_name;
29   // TODO: this needs to be able to read deleted persons, so that it can send tombstones
30   let person: ApubPerson = blocking(context.pool(), move |conn| {
31     Person::read_from_name(conn, &user_name)
32   })
33   .await??
34   .into();
35
36   if !person.deleted {
37     let apub = person.into_apub(&context).await?;
38
39     Ok(create_apub_response(&apub))
40   } else {
41     Ok(create_apub_tombstone_response(person.actor_id.clone()))
42   }
43 }
44
45 #[tracing::instrument(skip_all)]
46 pub async fn person_inbox(
47   request: HttpRequest,
48   payload: String,
49   context: web::Data<LemmyContext>,
50 ) -> Result<HttpResponse, LemmyError> {
51   receive_lemmy_activity::<WithContext<PersonInboxActivities>, UserOrCommunity>(
52     request, payload, context,
53   )
54   .await
55 }
56
57 #[tracing::instrument(skip_all)]
58 pub(crate) async fn get_apub_person_outbox(
59   info: web::Path<PersonQuery>,
60   context: web::Data<LemmyContext>,
61 ) -> Result<HttpResponse, LemmyError> {
62   let person = blocking(context.pool(), move |conn| {
63     Person::read_from_name(conn, &info.user_name)
64   })
65   .await??;
66   let outbox_id = generate_outbox_url(&person.actor_id)?.into();
67   let outbox = EmptyOutbox::new(outbox_id).await?;
68   Ok(create_apub_response(&outbox))
69 }