1 use actix_web::{guard, web, Error, HttpResponse, Result};
3 comment::{distinguish::distinguish_comment, like::like_comment, save::save_comment},
4 comment_report::{list::list_comment_reports, resolve::resolve_comment_report},
5 local_user::notifications::mark_reply_read::mark_reply_as_read,
9 use lemmy_api_common::{
10 comment::CreateCommentReport,
23 context::LemmyContext,
24 custom_emoji::{CreateCustomEmoji, DeleteCustomEmoji, EditCustomEmoji},
39 MarkPersonMentionAsRead,
40 PasswordChangeAfterReset,
60 CreatePrivateMessageReport,
63 ListPrivateMessageReports,
64 MarkPrivateMessageAsRead,
65 ResolvePrivateMessageReport,
68 ApproveRegistrationApplication,
69 GetFederatedInstances,
71 GetUnreadRegistrationApplicationCount,
73 ListRegistrationApplications,
82 create::create_comment,
83 delete::delete_comment,
85 remove::remove_comment,
86 update::update_comment,
88 community::list::list_communities,
89 post::{create::create_post, read::get_post, update::update_post},
90 private_message::read::get_private_message,
91 site::{create::create_site, read::get_site, update::update_site},
96 list_comments::list_comments,
97 list_posts::list_posts,
98 read_community::get_community,
99 read_person::read_person,
100 resolve_object::resolve_object,
105 use lemmy_utils::{rate_limit::RateLimitCell, spawn_try_task, SYNCHRONOUS_FEDERATION};
106 use serde::Deserialize;
108 pub fn config(cfg: &mut web::ServiceConfig, rate_limit: &RateLimitCell) {
110 web::scope("/api/v3")
114 .wrap(rate_limit.message())
115 .route("", web::get().to(get_site))
117 .route("", web::post().to(create_site))
118 .route("", web::put().to(update_site)),
121 web::resource("/modlog")
122 .wrap(rate_limit.message())
123 .route(web::get().to(route_get::<GetModlog>)),
126 web::resource("/search")
127 .wrap(rate_limit.search())
128 .route(web::get().to(search)),
131 web::resource("/resolve_object")
132 .wrap(rate_limit.message())
133 .route(web::get().to(resolve_object)),
137 web::resource("/community")
138 .guard(guard::Post())
139 .wrap(rate_limit.register())
140 .route(web::post().to(route_post_crud::<CreateCommunity>)),
143 web::scope("/community")
144 .wrap(rate_limit.message())
145 .route("", web::get().to(get_community))
146 .route("", web::put().to(route_post_crud::<EditCommunity>))
147 .route("/hide", web::put().to(route_post::<HideCommunity>))
148 .route("/list", web::get().to(list_communities))
149 .route("/follow", web::post().to(route_post::<FollowCommunity>))
150 .route("/block", web::post().to(route_post::<BlockCommunity>))
153 web::post().to(route_post_crud::<DeleteCommunity>),
158 web::post().to(route_post_crud::<RemoveCommunity>),
160 .route("/transfer", web::post().to(route_post::<TransferCommunity>))
161 .route("/ban_user", web::post().to(route_post::<BanFromCommunity>))
162 .route("/mod", web::post().to(route_post::<AddModToCommunity>)),
165 web::scope("/federated_instances")
166 .wrap(rate_limit.message())
167 .route("", web::get().to(route_get::<GetFederatedInstances>)),
171 // Handle POST to /post separately to add the post() rate limitter
172 web::resource("/post")
173 .guard(guard::Post())
174 .wrap(rate_limit.post())
175 .route(web::post().to(create_post)),
179 .wrap(rate_limit.message())
180 .route("", web::get().to(get_post))
181 .route("", web::put().to(update_post))
182 .route("/delete", web::post().to(route_post_crud::<DeletePost>))
183 .route("/remove", web::post().to(route_post_crud::<RemovePost>))
186 web::post().to(route_post::<MarkPostAsRead>),
188 .route("/lock", web::post().to(route_post::<LockPost>))
189 .route("/feature", web::post().to(route_post::<FeaturePost>))
190 .route("/list", web::get().to(list_posts))
191 .route("/like", web::post().to(like_post))
192 .route("/save", web::put().to(route_post::<SavePost>))
193 .route("/report", web::post().to(route_post::<CreatePostReport>))
196 web::put().to(route_post::<ResolvePostReport>),
198 .route("/report/list", web::get().to(route_get::<ListPostReports>))
201 web::get().to(route_get::<GetSiteMetadata>),
206 // Handle POST to /comment separately to add the comment() rate limitter
207 web::resource("/comment")
208 .guard(guard::Post())
209 .wrap(rate_limit.comment())
210 .route(web::post().to(create_comment)),
213 web::scope("/comment")
214 .wrap(rate_limit.message())
215 .route("", web::get().to(get_comment))
216 .route("", web::put().to(update_comment))
217 .route("/delete", web::post().to(delete_comment))
218 .route("/remove", web::post().to(remove_comment))
219 .route("/mark_as_read", web::post().to(mark_reply_as_read))
220 .route("/distinguish", web::post().to(distinguish_comment))
221 .route("/like", web::post().to(like_comment))
222 .route("/save", web::put().to(save_comment))
223 .route("/list", web::get().to(list_comments))
224 .route("/report", web::post().to(route_post::<CreateCommentReport>))
225 .route("/report/resolve", web::put().to(resolve_comment_report))
226 .route("/report/list", web::get().to(list_comment_reports)),
230 web::scope("/private_message")
231 .wrap(rate_limit.message())
232 .route("/list", web::get().to(get_private_message))
233 .route("", web::post().to(route_post_crud::<CreatePrivateMessage>))
234 .route("", web::put().to(route_post_crud::<EditPrivateMessage>))
237 web::post().to(route_post_crud::<DeletePrivateMessage>),
241 web::post().to(route_post::<MarkPrivateMessageAsRead>),
245 web::post().to(route_post::<CreatePrivateMessageReport>),
249 web::put().to(route_post::<ResolvePrivateMessageReport>),
253 web::get().to(route_get::<ListPrivateMessageReports>),
258 // Account action, I don't like that it's in /user maybe /accounts
259 // Handle /user/register separately to add the register() rate limitter
260 web::resource("/user/register")
261 .guard(guard::Post())
262 .wrap(rate_limit.register())
263 .route(web::post().to(route_post_crud::<Register>)),
266 // Handle captcha separately
267 web::resource("/user/get_captcha")
268 .wrap(rate_limit.post())
269 .route(web::get().to(route_get::<GetCaptcha>)),
274 .wrap(rate_limit.message())
275 .route("", web::get().to(read_person))
276 .route("/mention", web::get().to(route_get::<GetPersonMentions>))
278 "/mention/mark_as_read",
279 web::post().to(route_post::<MarkPersonMentionAsRead>),
281 .route("/replies", web::get().to(route_get::<GetReplies>))
282 // Admin action. I don't like that it's in /user
283 .route("/ban", web::post().to(route_post::<BanPerson>))
284 .route("/banned", web::get().to(route_get::<GetBannedPersons>))
285 .route("/block", web::post().to(route_post::<BlockPerson>))
286 // Account actions. I don't like that they're in /user maybe /accounts
287 .route("/login", web::post().to(route_post::<Login>))
290 web::post().to(route_post_crud::<DeleteAccount>),
294 web::post().to(route_post::<PasswordReset>),
298 web::post().to(route_post::<PasswordChangeAfterReset>),
300 // mark_all_as_read feels off being in this section as well
303 web::post().to(route_post::<MarkAllAsRead>),
306 "/save_user_settings",
307 web::put().to(route_post::<SaveUserSettings>),
311 web::put().to(route_post::<ChangePassword>),
313 .route("/report_count", web::get().to(route_get::<GetReportCount>))
314 .route("/unread_count", web::get().to(route_get::<GetUnreadCount>))
315 .route("/verify_email", web::post().to(route_post::<VerifyEmail>))
316 .route("/leave_admin", web::post().to(route_post::<LeaveAdmin>)),
321 .wrap(rate_limit.message())
322 .route("/add", web::post().to(route_post::<AddAdmin>))
324 "/registration_application/count",
325 web::get().to(route_get::<GetUnreadRegistrationApplicationCount>),
328 "/registration_application/list",
329 web::get().to(route_get::<ListRegistrationApplications>),
332 "/registration_application/approve",
333 web::put().to(route_post::<ApproveRegistrationApplication>),
337 .route("/person", web::post().to(route_post::<PurgePerson>))
338 .route("/community", web::post().to(route_post::<PurgeCommunity>))
339 .route("/post", web::post().to(route_post::<PurgePost>))
340 .route("/comment", web::post().to(route_post::<PurgeComment>)),
344 web::scope("/custom_emoji")
345 .wrap(rate_limit.message())
346 .route("", web::post().to(route_post_crud::<CreateCustomEmoji>))
347 .route("", web::put().to(route_post_crud::<EditCustomEmoji>))
350 web::post().to(route_post_crud::<DeleteCustomEmoji>),
356 async fn perform<'a, Data>(
358 context: web::Data<LemmyContext>,
359 apub_data: activitypub_federation::config::Data<LemmyContext>,
360 ) -> Result<HttpResponse, Error>
363 + SendActivity<Response = <Data as Perform>::Response>
369 let res = data.perform(&context).await?;
370 let res_clone = res.clone();
371 let fed_task = async move { SendActivity::send_activity(&data, &res_clone, &apub_data).await };
372 if *SYNCHRONOUS_FEDERATION {
375 spawn_try_task(fed_task);
377 Ok(HttpResponse::Ok().json(&res))
380 async fn route_get<'a, Data>(
381 data: web::Query<Data>,
382 context: web::Data<LemmyContext>,
383 apub_data: activitypub_federation::config::Data<LemmyContext>,
384 ) -> Result<HttpResponse, Error>
387 + SendActivity<Response = <Data as Perform>::Response>
393 perform::<Data>(data.0, context, apub_data).await
396 async fn route_post<'a, Data>(
397 data: web::Json<Data>,
398 context: web::Data<LemmyContext>,
399 apub_data: activitypub_federation::config::Data<LemmyContext>,
400 ) -> Result<HttpResponse, Error>
403 + SendActivity<Response = <Data as Perform>::Response>
409 perform::<Data>(data.0, context, apub_data).await
412 async fn perform_crud<'a, Data>(
414 context: web::Data<LemmyContext>,
415 apub_data: activitypub_federation::config::Data<LemmyContext>,
416 ) -> Result<HttpResponse, Error>
419 + SendActivity<Response = <Data as PerformCrud>::Response>
425 let res = data.perform(&context).await?;
426 let res_clone = res.clone();
427 let fed_task = async move { SendActivity::send_activity(&data, &res_clone, &apub_data).await };
428 if *SYNCHRONOUS_FEDERATION {
431 spawn_try_task(fed_task);
433 Ok(HttpResponse::Ok().json(&res))
436 async fn route_post_crud<'a, Data>(
437 data: web::Json<Data>,
438 context: web::Data<LemmyContext>,
439 apub_data: activitypub_federation::config::Data<LemmyContext>,
440 ) -> Result<HttpResponse, Error>
443 + SendActivity<Response = <Data as PerformCrud>::Response>
449 perform_crud::<Data>(data.0, context, apub_data).await