1 use diesel::{dsl::*, result::Error, *};
3 aggregates::post_aggregates::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},
26 use serde::{Deserialize, Serialize};
28 #[derive(Debug, PartialEq, Serialize, Deserialize, Clone)]
29 pub struct PostReportView {
30 pub post_report: PostReport,
32 pub community: CommunitySafe,
33 pub creator: PersonSafe,
34 pub post_creator: PersonSafeAlias1,
35 pub creator_banned_from_community: bool,
36 pub my_vote: Option<i16>,
37 pub counts: PostAggregates,
38 pub resolver: Option<PersonSafeAlias2>,
41 type PostReportViewTuple = (
47 Option<CommunityPersonBan>,
50 Option<PersonSafeAlias2>,
54 /// returns the PostReportView for the provided report_id
56 /// * `report_id` - the report id to obtain
59 report_id: PostReportId,
60 my_person_id: PersonId,
61 ) -> Result<Self, Error> {
68 creator_banned_from_community,
72 ) = post_report::table
74 .inner_join(post::table)
75 .inner_join(community::table.on(post::community_id.eq(community::id)))
76 .inner_join(person::table.on(post_report::creator_id.eq(person::id)))
77 .inner_join(person_alias_1::table.on(post::creator_id.eq(person_alias_1::id)))
79 community_person_ban::table.on(
81 .eq(community_person_ban::community_id)
82 .and(community_person_ban::person_id.eq(post::creator_id))
84 community_person_ban::expires
86 .or(community_person_ban::expires.gt(now)),
93 .eq(post_like::post_id)
94 .and(post_like::person_id.eq(my_person_id)),
97 .inner_join(post_aggregates::table.on(post_report::post_id.eq(post_aggregates::post_id)))
99 person_alias_2::table.on(post_report::resolver_id.eq(person_alias_2::id.nullable())),
102 post_report::all_columns,
104 Community::safe_columns_tuple(),
105 Person::safe_columns_tuple(),
106 PersonAlias1::safe_columns_tuple(),
107 community_person_ban::all_columns.nullable(),
108 post_like::score.nullable(),
109 post_aggregates::all_columns,
110 PersonAlias2::safe_columns_tuple().nullable(),
112 .first::<PostReportViewTuple>(conn)?;
114 let my_vote = if post_like.is_none() { None } else { post_like };
122 creator_banned_from_community: creator_banned_from_community.is_some(),
129 /// returns the current unresolved post report count for the communities you mod
130 pub fn get_report_count(
132 my_person_id: PersonId,
134 community_id: Option<CommunityId>,
135 ) -> Result<i64, Error> {
137 let mut query = post_report::table
138 .inner_join(post::table)
139 .filter(post_report::resolved.eq(false))
142 if let Some(community_id) = community_id {
143 query = query.filter(post::community_id.eq(community_id))
146 // If its not an admin, get only the ones you mod
150 community_moderator::table.on(
151 community_moderator::community_id
152 .eq(post::community_id)
153 .and(community_moderator::person_id.eq(my_person_id)),
156 .select(count(post_report::id))
159 query.select(count(post_report::id)).first::<i64>(conn)
164 pub struct PostReportQueryBuilder<'a> {
165 conn: &'a PgConnection,
166 my_person_id: PersonId,
168 community_id: Option<CommunityId>,
171 unresolved_only: Option<bool>,
174 impl<'a> PostReportQueryBuilder<'a> {
175 pub fn create(conn: &'a PgConnection, my_person_id: PersonId, admin: bool) -> Self {
176 PostReportQueryBuilder {
183 unresolved_only: Some(true),
187 pub fn community_id<T: MaybeOptional<CommunityId>>(mut self, community_id: T) -> Self {
188 self.community_id = community_id.get_optional();
192 pub fn page<T: MaybeOptional<i64>>(mut self, page: T) -> Self {
193 self.page = page.get_optional();
197 pub fn limit<T: MaybeOptional<i64>>(mut self, limit: T) -> Self {
198 self.limit = limit.get_optional();
202 pub fn unresolved_only<T: MaybeOptional<bool>>(mut self, unresolved_only: T) -> Self {
203 self.unresolved_only = unresolved_only.get_optional();
207 pub fn list(self) -> Result<Vec<PostReportView>, Error> {
208 let mut query = post_report::table
209 .inner_join(post::table)
210 .inner_join(community::table.on(post::community_id.eq(community::id)))
211 .inner_join(person::table.on(post_report::creator_id.eq(person::id)))
212 .inner_join(person_alias_1::table.on(post::creator_id.eq(person_alias_1::id)))
214 community_person_ban::table.on(
216 .eq(community_person_ban::community_id)
217 .and(community_person_ban::person_id.eq(post::creator_id))
219 community_person_ban::expires
221 .or(community_person_ban::expires.gt(now)),
228 .eq(post_like::post_id)
229 .and(post_like::person_id.eq(self.my_person_id)),
232 .inner_join(post_aggregates::table.on(post_report::post_id.eq(post_aggregates::post_id)))
234 person_alias_2::table.on(post_report::resolver_id.eq(person_alias_2::id.nullable())),
237 post_report::all_columns,
239 Community::safe_columns_tuple(),
240 Person::safe_columns_tuple(),
241 PersonAlias1::safe_columns_tuple(),
242 community_person_ban::all_columns.nullable(),
243 post_like::score.nullable(),
244 post_aggregates::all_columns,
245 PersonAlias2::safe_columns_tuple().nullable(),
249 if let Some(community_id) = self.community_id {
250 query = query.filter(post::community_id.eq(community_id));
253 if self.unresolved_only.unwrap_or(false) {
254 query = query.filter(post_report::resolved.eq(false));
257 let (limit, offset) = limit_and_offset(self.page, self.limit);
260 .order_by(post_report::published.desc())
264 // If its not an admin, get only the ones you mod
265 let res = if !self.admin {
268 community_moderator::table.on(
269 community_moderator::community_id
270 .eq(post::community_id)
271 .and(community_moderator::person_id.eq(self.my_person_id)),
274 .load::<PostReportViewTuple>(self.conn)?
276 query.load::<PostReportViewTuple>(self.conn)?
279 Ok(PostReportView::from_tuple_to_vec(res))
283 impl ViewToVec for PostReportView {
284 type DbTuple = PostReportViewTuple;
285 fn from_tuple_to_vec(items: Vec<Self::DbTuple>) -> Vec<Self> {
289 post_report: a.0.to_owned(),
290 post: a.1.to_owned(),
291 community: a.2.to_owned(),
292 creator: a.3.to_owned(),
293 post_creator: a.4.to_owned(),
294 creator_banned_from_community: a.5.is_some(),
296 counts: a.7.to_owned(),
297 resolver: a.8.to_owned(),
299 .collect::<Vec<Self>>()
305 use crate::post_report_view::{PostReportQueryBuilder, PostReportView};
306 use lemmy_db_schema::{
307 aggregates::post_aggregates::PostAggregates,
308 establish_unpooled_connection,
313 post_report::{PostReport, PostReportForm},
315 traits::{Crud, Joinable, Reportable},
317 use serial_test::serial;
322 let conn = establish_unpooled_connection();
324 let new_person = PersonForm {
325 name: "timmy_prv".into(),
326 ..PersonForm::default()
329 let inserted_timmy = Person::create(&conn, &new_person).unwrap();
331 let new_person_2 = PersonForm {
332 name: "sara_prv".into(),
333 ..PersonForm::default()
336 let inserted_sara = Person::create(&conn, &new_person_2).unwrap();
338 // Add a third person, since new ppl can only report something once.
339 let new_person_3 = PersonForm {
340 name: "jessica_prv".into(),
341 ..PersonForm::default()
344 let inserted_jessica = Person::create(&conn, &new_person_3).unwrap();
346 let new_community = CommunityForm {
347 name: "test community prv".to_string(),
348 title: "nada".to_owned(),
349 ..CommunityForm::default()
352 let inserted_community = Community::create(&conn, &new_community).unwrap();
355 let timmy_moderator_form = CommunityModeratorForm {
356 community_id: inserted_community.id,
357 person_id: inserted_timmy.id,
360 let _inserted_moderator = CommunityModerator::join(&conn, &timmy_moderator_form).unwrap();
362 let new_post = PostForm {
363 name: "A test post crv".into(),
364 creator_id: inserted_timmy.id,
365 community_id: inserted_community.id,
366 ..PostForm::default()
369 let inserted_post = Post::create(&conn, &new_post).unwrap();
372 let sara_report_form = PostReportForm {
373 creator_id: inserted_sara.id,
374 post_id: inserted_post.id,
375 original_post_name: "Orig post".into(),
376 original_post_url: None,
377 original_post_body: None,
378 reason: "from sara".into(),
381 let inserted_sara_report = PostReport::report(&conn, &sara_report_form).unwrap();
384 let jessica_report_form = PostReportForm {
385 creator_id: inserted_jessica.id,
386 post_id: inserted_post.id,
387 original_post_name: "Orig post".into(),
388 original_post_url: None,
389 original_post_body: None,
390 reason: "from jessica".into(),
393 let inserted_jessica_report = PostReport::report(&conn, &jessica_report_form).unwrap();
395 let agg = PostAggregates::read(&conn, inserted_post.id).unwrap();
397 let read_jessica_report_view =
398 PostReportView::read(&conn, inserted_jessica_report.id, inserted_timmy.id).unwrap();
399 let expected_jessica_report_view = PostReportView {
400 post_report: inserted_jessica_report.to_owned(),
401 post: inserted_post.to_owned(),
402 community: CommunitySafe {
403 id: inserted_community.id,
404 name: inserted_community.name,
409 actor_id: inserted_community.actor_id.to_owned(),
411 title: inserted_community.title,
415 published: inserted_community.published,
417 creator: PersonSafe {
418 id: inserted_jessica.id,
419 name: inserted_jessica.name,
421 published: inserted_jessica.published,
423 actor_id: inserted_jessica.actor_id.to_owned(),
432 inbox_url: inserted_jessica.inbox_url.to_owned(),
433 shared_inbox_url: None,
434 matrix_user_id: None,
437 post_creator: PersonSafeAlias1 {
438 id: inserted_timmy.id,
439 name: inserted_timmy.name.to_owned(),
441 published: inserted_timmy.published,
443 actor_id: inserted_timmy.actor_id.to_owned(),
452 inbox_url: inserted_timmy.inbox_url.to_owned(),
453 shared_inbox_url: None,
454 matrix_user_id: None,
457 creator_banned_from_community: false,
459 counts: PostAggregates {
461 post_id: inserted_post.id,
467 published: agg.published,
468 newest_comment_time_necro: inserted_post.published,
469 newest_comment_time: inserted_post.published,
474 assert_eq!(read_jessica_report_view, expected_jessica_report_view);
476 let mut expected_sara_report_view = expected_jessica_report_view.clone();
477 expected_sara_report_view.post_report = inserted_sara_report;
478 expected_sara_report_view.my_vote = None;
479 expected_sara_report_view.creator = PersonSafe {
480 id: inserted_sara.id,
481 name: inserted_sara.name,
483 published: inserted_sara.published,
485 actor_id: inserted_sara.actor_id.to_owned(),
494 inbox_url: inserted_sara.inbox_url.to_owned(),
495 shared_inbox_url: None,
496 matrix_user_id: None,
500 // Do a batch read of timmys reports
501 let reports = PostReportQueryBuilder::create(&conn, inserted_timmy.id, false)
508 expected_jessica_report_view.to_owned(),
509 expected_sara_report_view.to_owned()
513 // Make sure the counts are correct
515 PostReportView::get_report_count(&conn, inserted_timmy.id, false, None).unwrap();
516 assert_eq!(2, report_count);
518 // Try to resolve the report
519 PostReport::resolve(&conn, inserted_jessica_report.id, inserted_timmy.id).unwrap();
520 let read_jessica_report_view_after_resolve =
521 PostReportView::read(&conn, inserted_jessica_report.id, inserted_timmy.id).unwrap();
523 let mut expected_jessica_report_view_after_resolve = expected_jessica_report_view;
524 expected_jessica_report_view_after_resolve
527 expected_jessica_report_view_after_resolve
529 .resolver_id = Some(inserted_timmy.id);
530 expected_jessica_report_view_after_resolve
532 .updated = read_jessica_report_view_after_resolve.post_report.updated;
533 expected_jessica_report_view_after_resolve.resolver = Some(PersonSafeAlias2 {
534 id: inserted_timmy.id,
535 name: inserted_timmy.name.to_owned(),
537 published: inserted_timmy.published,
539 actor_id: inserted_timmy.actor_id.to_owned(),
548 inbox_url: inserted_timmy.inbox_url.to_owned(),
549 shared_inbox_url: None,
550 matrix_user_id: None,
555 read_jessica_report_view_after_resolve,
556 expected_jessica_report_view_after_resolve
559 // Do a batch read of timmys reports
560 // It should only show saras, which is unresolved
561 let reports_after_resolve = PostReportQueryBuilder::create(&conn, inserted_timmy.id, false)
564 assert_eq!(reports_after_resolve[0], expected_sara_report_view);
566 // Make sure the counts are correct
567 let report_count_after_resolved =
568 PostReportView::get_report_count(&conn, inserted_timmy.id, false, None).unwrap();
569 assert_eq!(1, report_count_after_resolved);
571 Person::delete(&conn, inserted_timmy.id).unwrap();
572 Person::delete(&conn, inserted_sara.id).unwrap();
573 Person::delete(&conn, inserted_jessica.id).unwrap();
574 Community::delete(&conn, inserted_community.id).unwrap();