1 use crate::structs::PostReportView;
2 use diesel::{dsl::*, result::Error, *};
4 aggregates::structs::PostAggregates,
5 newtypes::{CommunityId, PersonId, PostReportId},
19 community::{Community, CommunityPersonBan, CommunitySafe},
20 person::{Person, PersonAlias1, PersonAlias2, PersonSafe, PersonSafeAlias1, PersonSafeAlias2},
22 post_report::PostReport,
24 traits::{MaybeOptional, ToSafe, ViewToVec},
25 utils::limit_and_offset,
28 type PostReportViewTuple = (
34 Option<CommunityPersonBan>,
37 Option<PersonSafeAlias2>,
41 /// returns the PostReportView for the provided report_id
43 /// * `report_id` - the report id to obtain
46 report_id: PostReportId,
47 my_person_id: PersonId,
48 ) -> Result<Self, Error> {
55 creator_banned_from_community,
59 ) = post_report::table
61 .inner_join(post::table)
62 .inner_join(community::table.on(post::community_id.eq(community::id)))
63 .inner_join(person::table.on(post_report::creator_id.eq(person::id)))
64 .inner_join(person_alias_1::table.on(post::creator_id.eq(person_alias_1::id)))
66 community_person_ban::table.on(
68 .eq(community_person_ban::community_id)
69 .and(community_person_ban::person_id.eq(post::creator_id))
71 community_person_ban::expires
73 .or(community_person_ban::expires.gt(now)),
80 .eq(post_like::post_id)
81 .and(post_like::person_id.eq(my_person_id)),
84 .inner_join(post_aggregates::table.on(post_report::post_id.eq(post_aggregates::post_id)))
86 person_alias_2::table.on(post_report::resolver_id.eq(person_alias_2::id.nullable())),
89 post_report::all_columns,
91 Community::safe_columns_tuple(),
92 Person::safe_columns_tuple(),
93 PersonAlias1::safe_columns_tuple(),
94 community_person_ban::all_columns.nullable(),
95 post_like::score.nullable(),
96 post_aggregates::all_columns,
97 PersonAlias2::safe_columns_tuple().nullable(),
99 .first::<PostReportViewTuple>(conn)?;
101 let my_vote = post_like;
109 creator_banned_from_community: creator_banned_from_community.is_some(),
116 /// returns the current unresolved post report count for the communities you mod
117 pub fn get_report_count(
119 my_person_id: PersonId,
121 community_id: Option<CommunityId>,
122 ) -> Result<i64, Error> {
124 let mut query = post_report::table
125 .inner_join(post::table)
126 .filter(post_report::resolved.eq(false))
129 if let Some(community_id) = community_id {
130 query = query.filter(post::community_id.eq(community_id))
133 // If its not an admin, get only the ones you mod
137 community_moderator::table.on(
138 community_moderator::community_id
139 .eq(post::community_id)
140 .and(community_moderator::person_id.eq(my_person_id)),
143 .select(count(post_report::id))
146 query.select(count(post_report::id)).first::<i64>(conn)
151 pub struct PostReportQueryBuilder<'a> {
152 conn: &'a PgConnection,
153 my_person_id: PersonId,
155 community_id: Option<CommunityId>,
158 unresolved_only: Option<bool>,
161 impl<'a> PostReportQueryBuilder<'a> {
162 pub fn create(conn: &'a PgConnection, my_person_id: PersonId, admin: bool) -> Self {
163 PostReportQueryBuilder {
170 unresolved_only: Some(true),
174 pub fn community_id<T: MaybeOptional<CommunityId>>(mut self, community_id: T) -> Self {
175 self.community_id = community_id.get_optional();
179 pub fn page<T: MaybeOptional<i64>>(mut self, page: T) -> Self {
180 self.page = page.get_optional();
184 pub fn limit<T: MaybeOptional<i64>>(mut self, limit: T) -> Self {
185 self.limit = limit.get_optional();
189 pub fn unresolved_only<T: MaybeOptional<bool>>(mut self, unresolved_only: T) -> Self {
190 self.unresolved_only = unresolved_only.get_optional();
194 pub fn list(self) -> Result<Vec<PostReportView>, Error> {
195 let mut query = post_report::table
196 .inner_join(post::table)
197 .inner_join(community::table.on(post::community_id.eq(community::id)))
198 .inner_join(person::table.on(post_report::creator_id.eq(person::id)))
199 .inner_join(person_alias_1::table.on(post::creator_id.eq(person_alias_1::id)))
201 community_person_ban::table.on(
203 .eq(community_person_ban::community_id)
204 .and(community_person_ban::person_id.eq(post::creator_id))
206 community_person_ban::expires
208 .or(community_person_ban::expires.gt(now)),
215 .eq(post_like::post_id)
216 .and(post_like::person_id.eq(self.my_person_id)),
219 .inner_join(post_aggregates::table.on(post_report::post_id.eq(post_aggregates::post_id)))
221 person_alias_2::table.on(post_report::resolver_id.eq(person_alias_2::id.nullable())),
224 post_report::all_columns,
226 Community::safe_columns_tuple(),
227 Person::safe_columns_tuple(),
228 PersonAlias1::safe_columns_tuple(),
229 community_person_ban::all_columns.nullable(),
230 post_like::score.nullable(),
231 post_aggregates::all_columns,
232 PersonAlias2::safe_columns_tuple().nullable(),
236 if let Some(community_id) = self.community_id {
237 query = query.filter(post::community_id.eq(community_id));
240 if self.unresolved_only.unwrap_or(false) {
241 query = query.filter(post_report::resolved.eq(false));
244 let (limit, offset) = limit_and_offset(self.page, self.limit)?;
247 .order_by(post_report::published.desc())
251 // If its not an admin, get only the ones you mod
252 let res = if !self.admin {
255 community_moderator::table.on(
256 community_moderator::community_id
257 .eq(post::community_id)
258 .and(community_moderator::person_id.eq(self.my_person_id)),
261 .load::<PostReportViewTuple>(self.conn)?
263 query.load::<PostReportViewTuple>(self.conn)?
266 Ok(PostReportView::from_tuple_to_vec(res))
270 impl ViewToVec for PostReportView {
271 type DbTuple = PostReportViewTuple;
272 fn from_tuple_to_vec(items: Vec<Self::DbTuple>) -> Vec<Self> {
276 post_report: a.0.to_owned(),
277 post: a.1.to_owned(),
278 community: a.2.to_owned(),
279 creator: a.3.to_owned(),
280 post_creator: a.4.to_owned(),
281 creator_banned_from_community: a.5.is_some(),
283 counts: a.7.to_owned(),
284 resolver: a.8.to_owned(),
286 .collect::<Vec<Self>>()
292 use crate::post_report_view::{PostReportQueryBuilder, PostReportView};
293 use lemmy_db_schema::{
294 aggregates::structs::PostAggregates,
299 post_report::{PostReport, PostReportForm},
301 traits::{Crud, Joinable, Reportable},
302 utils::establish_unpooled_connection,
304 use serial_test::serial;
309 let conn = establish_unpooled_connection();
311 let new_person = PersonForm {
312 name: "timmy_prv".into(),
313 public_key: Some("pubkey".to_string()),
314 ..PersonForm::default()
317 let inserted_timmy = Person::create(&conn, &new_person).unwrap();
319 let new_person_2 = PersonForm {
320 name: "sara_prv".into(),
321 public_key: Some("pubkey".to_string()),
322 ..PersonForm::default()
325 let inserted_sara = Person::create(&conn, &new_person_2).unwrap();
327 // Add a third person, since new ppl can only report something once.
328 let new_person_3 = PersonForm {
329 name: "jessica_prv".into(),
330 public_key: Some("pubkey".to_string()),
331 ..PersonForm::default()
334 let inserted_jessica = Person::create(&conn, &new_person_3).unwrap();
336 let new_community = CommunityForm {
337 name: "test community prv".to_string(),
338 title: "nada".to_owned(),
339 public_key: Some("pubkey".to_string()),
340 ..CommunityForm::default()
343 let inserted_community = Community::create(&conn, &new_community).unwrap();
346 let timmy_moderator_form = CommunityModeratorForm {
347 community_id: inserted_community.id,
348 person_id: inserted_timmy.id,
351 let _inserted_moderator = CommunityModerator::join(&conn, &timmy_moderator_form).unwrap();
353 let new_post = PostForm {
354 name: "A test post crv".into(),
355 creator_id: inserted_timmy.id,
356 community_id: inserted_community.id,
357 ..PostForm::default()
360 let inserted_post = Post::create(&conn, &new_post).unwrap();
363 let sara_report_form = PostReportForm {
364 creator_id: inserted_sara.id,
365 post_id: inserted_post.id,
366 original_post_name: "Orig post".into(),
367 original_post_url: None,
368 original_post_body: None,
369 reason: "from sara".into(),
372 let inserted_sara_report = PostReport::report(&conn, &sara_report_form).unwrap();
375 let jessica_report_form = PostReportForm {
376 creator_id: inserted_jessica.id,
377 post_id: inserted_post.id,
378 original_post_name: "Orig post".into(),
379 original_post_url: None,
380 original_post_body: None,
381 reason: "from jessica".into(),
384 let inserted_jessica_report = PostReport::report(&conn, &jessica_report_form).unwrap();
386 let agg = PostAggregates::read(&conn, inserted_post.id).unwrap();
388 let read_jessica_report_view =
389 PostReportView::read(&conn, inserted_jessica_report.id, inserted_timmy.id).unwrap();
390 let expected_jessica_report_view = PostReportView {
391 post_report: inserted_jessica_report.to_owned(),
392 post: inserted_post.to_owned(),
393 community: CommunitySafe {
394 id: inserted_community.id,
395 name: inserted_community.name,
400 actor_id: inserted_community.actor_id.to_owned(),
402 title: inserted_community.title,
407 posting_restricted_to_mods: false,
408 published: inserted_community.published,
410 creator: PersonSafe {
411 id: inserted_jessica.id,
412 name: inserted_jessica.name,
414 published: inserted_jessica.published,
416 actor_id: inserted_jessica.actor_id.to_owned(),
425 inbox_url: inserted_jessica.inbox_url.to_owned(),
426 shared_inbox_url: None,
427 matrix_user_id: None,
430 post_creator: PersonSafeAlias1 {
431 id: inserted_timmy.id,
432 name: inserted_timmy.name.to_owned(),
434 published: inserted_timmy.published,
436 actor_id: inserted_timmy.actor_id.to_owned(),
445 inbox_url: inserted_timmy.inbox_url.to_owned(),
446 shared_inbox_url: None,
447 matrix_user_id: None,
450 creator_banned_from_community: false,
452 counts: PostAggregates {
454 post_id: inserted_post.id,
460 published: agg.published,
461 newest_comment_time_necro: inserted_post.published,
462 newest_comment_time: inserted_post.published,
467 assert_eq!(read_jessica_report_view, expected_jessica_report_view);
469 let mut expected_sara_report_view = expected_jessica_report_view.clone();
470 expected_sara_report_view.post_report = inserted_sara_report;
471 expected_sara_report_view.my_vote = None;
472 expected_sara_report_view.creator = PersonSafe {
473 id: inserted_sara.id,
474 name: inserted_sara.name,
476 published: inserted_sara.published,
478 actor_id: inserted_sara.actor_id.to_owned(),
487 inbox_url: inserted_sara.inbox_url.to_owned(),
488 shared_inbox_url: None,
489 matrix_user_id: None,
493 // Do a batch read of timmys reports
494 let reports = PostReportQueryBuilder::create(&conn, inserted_timmy.id, false)
501 expected_jessica_report_view.to_owned(),
502 expected_sara_report_view.to_owned()
506 // Make sure the counts are correct
508 PostReportView::get_report_count(&conn, inserted_timmy.id, false, None).unwrap();
509 assert_eq!(2, report_count);
511 // Try to resolve the report
512 PostReport::resolve(&conn, inserted_jessica_report.id, inserted_timmy.id).unwrap();
513 let read_jessica_report_view_after_resolve =
514 PostReportView::read(&conn, inserted_jessica_report.id, inserted_timmy.id).unwrap();
516 let mut expected_jessica_report_view_after_resolve = expected_jessica_report_view;
517 expected_jessica_report_view_after_resolve
520 expected_jessica_report_view_after_resolve
522 .resolver_id = Some(inserted_timmy.id);
523 expected_jessica_report_view_after_resolve
525 .updated = read_jessica_report_view_after_resolve.post_report.updated;
526 expected_jessica_report_view_after_resolve.resolver = Some(PersonSafeAlias2 {
527 id: inserted_timmy.id,
528 name: inserted_timmy.name.to_owned(),
530 published: inserted_timmy.published,
532 actor_id: inserted_timmy.actor_id.to_owned(),
541 inbox_url: inserted_timmy.inbox_url.to_owned(),
542 shared_inbox_url: None,
543 matrix_user_id: None,
548 read_jessica_report_view_after_resolve,
549 expected_jessica_report_view_after_resolve
552 // Do a batch read of timmys reports
553 // It should only show saras, which is unresolved
554 let reports_after_resolve = PostReportQueryBuilder::create(&conn, inserted_timmy.id, false)
557 assert_eq!(reports_after_resolve[0], expected_sara_report_view);
559 // Make sure the counts are correct
560 let report_count_after_resolved =
561 PostReportView::get_report_count(&conn, inserted_timmy.id, false, None).unwrap();
562 assert_eq!(1, report_count_after_resolved);
564 Person::delete(&conn, inserted_timmy.id).unwrap();
565 Person::delete(&conn, inserted_sara.id).unwrap();
566 Person::delete(&conn, inserted_jessica.id).unwrap();
567 Community::delete(&conn, inserted_community.id).unwrap();