]> Untitled Git - lemmy.git/blob - crates/apub/src/api/read_person.rs
71f4abd519a90293ceea0b462f31f1bf6399a964
[lemmy.git] / crates / apub / src / api / read_person.rs
1 use crate::{fetcher::resolve_actor_identifier, objects::person::ApubPerson};
2 use activitypub_federation::config::Data;
3 use actix_web::web::{Json, Query};
4 use lemmy_api_common::{
5   context::LemmyContext,
6   person::{GetPersonDetails, GetPersonDetailsResponse},
7   utils::{check_private_instance, is_admin, local_user_view_from_jwt_opt},
8 };
9 use lemmy_db_schema::{
10   source::{local_site::LocalSite, person::Person},
11   utils::post_to_comment_sort_type,
12 };
13 use lemmy_db_views::{comment_view::CommentQuery, post_view::PostQuery};
14 use lemmy_db_views_actor::structs::{CommunityModeratorView, PersonView};
15 use lemmy_utils::error::{LemmyError, LemmyErrorExt2, LemmyErrorType};
16
17 #[tracing::instrument(skip(context))]
18 pub async fn read_person(
19   data: Query<GetPersonDetails>,
20   context: Data<LemmyContext>,
21 ) -> Result<Json<GetPersonDetailsResponse>, LemmyError> {
22   // Check to make sure a person name or an id is given
23   if data.username.is_none() && data.person_id.is_none() {
24     return Err(LemmyErrorType::NoIdGiven)?;
25   }
26
27   let local_user_view = local_user_view_from_jwt_opt(data.auth.as_ref(), &context).await;
28   let local_site = LocalSite::read(&mut context.pool()).await?;
29   let is_admin = local_user_view.as_ref().map(|luv| is_admin(luv).is_ok());
30
31   check_private_instance(&local_user_view, &local_site)?;
32
33   let person_details_id = match data.person_id {
34     Some(id) => id,
35     None => {
36       if let Some(username) = &data.username {
37         resolve_actor_identifier::<ApubPerson, Person>(username, &context, &local_user_view, true)
38           .await
39           .with_lemmy_type(LemmyErrorType::CouldntFindPerson)?
40           .id
41       } else {
42         return Err(LemmyErrorType::CouldntFindPerson)?;
43       }
44     }
45   };
46
47   // You don't need to return settings for the user, since this comes back with GetSite
48   // `my_user`
49   let person_view = PersonView::read(&mut context.pool(), person_details_id).await?;
50
51   let sort = data.sort;
52   let page = data.page;
53   let limit = data.limit;
54   let saved_only = data.saved_only;
55   let community_id = data.community_id;
56   let local_user = local_user_view.map(|l| l.local_user);
57   let local_user_clone = local_user.clone();
58
59   let posts = PostQuery::builder()
60     .pool(&mut context.pool())
61     .sort(sort)
62     .saved_only(saved_only)
63     .local_user(local_user.as_ref())
64     .community_id(community_id)
65     .is_mod_or_admin(is_admin)
66     .page(page)
67     .limit(limit)
68     .creator_id(
69       // If its saved only, you don't care what creator it was
70       // Or, if its not saved, then you only want it for that specific creator
71       if !saved_only.unwrap_or(false) {
72         Some(person_details_id)
73       } else {
74         None
75       },
76     )
77     .build()
78     .list()
79     .await?;
80
81   let comments = CommentQuery::builder()
82     .pool(&mut context.pool())
83     .local_user(local_user_clone.as_ref())
84     .sort(sort.map(post_to_comment_sort_type))
85     .saved_only(saved_only)
86     .show_deleted_and_removed(Some(false))
87     .community_id(community_id)
88     .page(page)
89     .limit(limit)
90     .creator_id(
91       // If its saved only, you don't care what creator it was
92       // Or, if its not saved, then you only want it for that specific creator
93       if !saved_only.unwrap_or(false) {
94         Some(person_details_id)
95       } else {
96         None
97       },
98     )
99     .build()
100     .list()
101     .await?;
102
103   let moderates =
104     CommunityModeratorView::for_person(&mut context.pool(), person_details_id).await?;
105
106   // Return the jwt
107   Ok(Json(GetPersonDetailsResponse {
108     person_view,
109     moderates,
110     comments,
111     posts,
112   }))
113 }