1 use crate::structs::{CommunityModeratorView, CommunityView, PersonView};
7 NullableExpressionMethods,
8 PgTextExpressionMethods,
11 use diesel_async::RunQueryDsl;
12 use lemmy_db_schema::{
13 aggregates::structs::CommunityAggregates,
14 newtypes::{CommunityId, PersonId},
15 schema::{community, community_aggregates, community_block, community_follower, local_user},
17 community::{Community, CommunityFollower},
18 community_block::CommunityBlock,
19 local_user::LocalUser,
22 utils::{fuzzy_search, get_conn, limit_and_offset, DbPool},
26 use typed_builder::TypedBuilder;
28 type CommunityViewTuple = (
31 Option<CommunityFollower>,
32 Option<CommunityBlock>,
38 community_id: CommunityId,
39 my_person_id: Option<PersonId>,
40 is_mod_or_admin: Option<bool>,
41 ) -> Result<Self, Error> {
42 let conn = &mut get_conn(pool).await?;
43 // The left join below will return None in this case
44 let person_id_join = my_person_id.unwrap_or(PersonId(-1));
46 let mut query = community::table
48 .inner_join(community_aggregates::table)
50 community_follower::table.on(
52 .eq(community_follower::community_id)
53 .and(community_follower::person_id.eq(person_id_join)),
57 community_block::table.on(
59 .eq(community_block::community_id)
60 .and(community_block::person_id.eq(person_id_join)),
64 community::all_columns,
65 community_aggregates::all_columns,
66 community_follower::all_columns.nullable(),
67 community_block::all_columns.nullable(),
71 // Hide deleted and removed for non-admins or mods
72 if !is_mod_or_admin.unwrap_or(false) {
74 .filter(community::removed.eq(false))
75 .filter(community::deleted.eq(false));
78 let (community, counts, follower, blocked) = query.first::<CommunityViewTuple>(conn).await?;
82 subscribed: CommunityFollower::to_subscribed_type(&follower),
83 blocked: blocked.is_some(),
88 pub async fn is_mod_or_admin(
91 community_id: CommunityId,
92 ) -> Result<bool, Error> {
93 let is_mod = CommunityModeratorView::for_community(pool, community_id)
97 .map(|m| m.moderator.id)
98 .collect::<Vec<PersonId>>()
101 .contains(&person_id);
106 let is_admin = PersonView::admins(pool)
110 .map(|a| a.person.id)
111 .collect::<Vec<PersonId>>()
114 .contains(&person_id);
119 #[derive(TypedBuilder)]
120 #[builder(field_defaults(default))]
121 pub struct CommunityQuery<'a> {
124 listing_type: Option<ListingType>,
125 sort: Option<SortType>,
126 local_user: Option<&'a LocalUser>,
127 search_term: Option<String>,
128 is_mod_or_admin: Option<bool>,
129 show_nsfw: Option<bool>,
134 impl<'a> CommunityQuery<'a> {
135 pub async fn list(self) -> Result<Vec<CommunityView>, Error> {
138 let conn = &mut get_conn(self.pool).await?;
140 // The left join below will return None in this case
141 let person_id_join = self.local_user.map(|l| l.person_id).unwrap_or(PersonId(-1));
143 let mut query = community::table
144 .inner_join(community_aggregates::table)
145 .left_join(local_user::table.on(local_user::person_id.eq(person_id_join)))
147 community_follower::table.on(
149 .eq(community_follower::community_id)
150 .and(community_follower::person_id.eq(person_id_join)),
154 community_block::table.on(
156 .eq(community_block::community_id)
157 .and(community_block::person_id.eq(person_id_join)),
161 community::all_columns,
162 community_aggregates::all_columns,
163 community_follower::all_columns.nullable(),
164 community_block::all_columns.nullable(),
168 if let Some(search_term) = self.search_term {
169 let searcher = fuzzy_search(&search_term);
171 .filter(community::name.ilike(searcher.clone()))
172 .or_filter(community::title.ilike(searcher));
175 // Hide deleted and removed for non-admins or mods
176 if !self.is_mod_or_admin.unwrap_or(false) {
178 .filter(community::removed.eq(false))
179 .filter(community::deleted.eq(false))
183 .or(community_follower::person_id.eq(person_id_join)),
186 match self.sort.unwrap_or(Hot) {
187 Hot | Active => query = query.order_by(community_aggregates::hot_rank.desc()),
188 NewComments | TopDay | TopTwelveHour | TopSixHour | TopHour => {
189 query = query.order_by(community_aggregates::users_active_day.desc())
191 New => query = query.order_by(community::published.desc()),
192 Old => query = query.order_by(community::published.asc()),
193 MostComments => query = query.order_by(community_aggregates::comments.desc()),
194 TopAll | TopYear | TopNineMonths => {
195 query = query.order_by(community_aggregates::subscribers.desc())
197 TopSixMonths | TopThreeMonths => {
198 query = query.order_by(community_aggregates::users_active_half_year.desc())
200 TopMonth => query = query.order_by(community_aggregates::users_active_month.desc()),
201 TopWeek => query = query.order_by(community_aggregates::users_active_week.desc()),
204 if let Some(listing_type) = self.listing_type {
205 query = match listing_type {
206 ListingType::Subscribed => query.filter(community_follower::person_id.is_not_null()), // TODO could be this: and(community_follower::person_id.eq(person_id_join)),
207 ListingType::Local => query.filter(community::local.eq(true)),
212 // Don't show blocked communities or nsfw communities if not enabled in profile
213 if self.local_user.is_some() {
214 query = query.filter(community_block::person_id.is_null());
215 query = query.filter(community::nsfw.eq(false).or(local_user::show_nsfw.eq(true)));
217 // No person in request, only show nsfw communities if show_nsfw is passed into request
218 if !self.show_nsfw.unwrap_or(false) {
219 query = query.filter(community::nsfw.eq(false));
223 let (limit, offset) = limit_and_offset(self.page, self.limit)?;
227 .load::<CommunityViewTuple>(conn)
230 Ok(res.into_iter().map(CommunityView::from_tuple).collect())
234 impl JoinView for CommunityView {
235 type JoinTuple = CommunityViewTuple;
236 fn from_tuple(a: Self::JoinTuple) -> Self {
240 subscribed: CommunityFollower::to_subscribed_type(&a.2),
241 blocked: a.3.is_some(),