use crate::{ diesel::{BoolExpressionMethods, OptionalExtension}, newtypes::{CommunityId, DbUrl, PersonId, PostId}, schema::{community, person, post, post_hide, post_like, post_read, post_saved}, source::post::{ Post, PostHide, PostHideForm, PostInsertForm, PostLike, PostLikeForm, PostRead, PostReadForm, PostSaved, PostSavedForm, PostUpdateForm, }, traits::{Crud, Likeable, Saveable}, utils::{ functions::coalesce, get_conn, naive_now, now, DbPool, DELETED_REPLACEMENT_TEXT, FETCH_LIMIT_MAX, SITEMAP_DAYS, SITEMAP_LIMIT, }, }; use ::url::Url; use chrono::{DateTime, Utc}; use diesel::{ dsl::{count, insert_into, not}, result::Error, DecoratableTarget, ExpressionMethods, QueryDsl, TextExpressionMethods, }; use diesel_async::RunQueryDsl; #[async_trait] impl Crud for Post { type InsertForm = PostInsertForm; type UpdateForm = PostUpdateForm; type IdType = PostId; async fn create(pool: &mut DbPool<'_>, form: &Self::InsertForm) -> Result { let conn = &mut get_conn(pool).await?; insert_into(post::table) .values(form) .get_result::(conn) .await } async fn update( pool: &mut DbPool<'_>, post_id: PostId, new_post: &Self::UpdateForm, ) -> Result { let conn = &mut get_conn(pool).await?; diesel::update(post::table.find(post_id)) .set(new_post) .get_result::(conn) .await } } impl Post { pub async fn read_xx(pool: &mut DbPool<'_>, id: PostId) -> Result { let conn = &mut *get_conn(pool).await?; post::table.find(id).first(conn).await } pub async fn insert_apub( pool: &mut DbPool<'_>, timestamp: DateTime, form: &PostInsertForm, ) -> Result { let conn = &mut get_conn(pool).await?; insert_into(post::table) .values(form) .on_conflict(post::ap_id) .filter_target(coalesce(post::updated, post::published).lt(timestamp)) .do_update() .set(form) .get_result::(conn) .await } pub async fn list_featured_for_community( pool: &mut DbPool<'_>, the_community_id: CommunityId, ) -> Result, Error> { let conn = &mut get_conn(pool).await?; post::table .filter(post::community_id.eq(the_community_id)) .filter(post::deleted.eq(false)) .filter(post::removed.eq(false)) .filter(post::featured_community.eq(true)) .then_order_by(post::published.desc()) .limit(FETCH_LIMIT_MAX) .load::(conn) .await } pub async fn list_for_sitemap( pool: &mut DbPool<'_>, ) -> Result)>, Error> { let conn = &mut get_conn(pool).await?; post::table .select((post::ap_id, coalesce(post::updated, post::published))) .filter(post::local.eq(true)) .filter(post::deleted.eq(false)) .filter(post::removed.eq(false)) .filter( post::published.ge(Utc::now().naive_utc() - SITEMAP_DAYS.expect("TimeDelta out of bounds")), ) .order(post::published.desc()) .limit(SITEMAP_LIMIT) .load::<(DbUrl, chrono::DateTime)>(conn) .await } pub async fn permadelete_for_creator( pool: &mut DbPool<'_>, for_creator_id: PersonId, ) -> Result, Error> { let conn = &mut get_conn(pool).await?; diesel::update(post::table.filter(post::creator_id.eq(for_creator_id))) .set(( post::name.eq(DELETED_REPLACEMENT_TEXT), post::url.eq(Option::<&str>::None), post::body.eq(DELETED_REPLACEMENT_TEXT), post::deleted.eq(true), post::updated.eq(naive_now()), )) .get_results::(conn) .await } pub async fn update_removed_for_creator( pool: &mut DbPool<'_>, for_creator_id: PersonId, for_community_id: Option, new_removed: bool, ) -> Result, Error> { let conn = &mut get_conn(pool).await?; let mut update = diesel::update(post::table).into_boxed(); update = update.filter(post::creator_id.eq(for_creator_id)); if let Some(for_community_id) = for_community_id { update = update.filter(post::community_id.eq(for_community_id)); } update .set((post::removed.eq(new_removed), post::updated.eq(naive_now()))) .get_results::(conn) .await } pub fn is_post_creator(person_id: PersonId, post_creator_id: PersonId) -> bool { person_id == post_creator_id } pub async fn read_from_apub_id( pool: &mut DbPool<'_>, object_id: Url, ) -> Result, Error> { let conn = &mut get_conn(pool).await?; let object_id: DbUrl = object_id.into(); post::table .filter(post::ap_id.eq(object_id)) .filter(post::scheduled_publish_time.is_null()) .first(conn) .await .optional() } pub async fn fetch_pictrs_posts_for_creator( pool: &mut DbPool<'_>, for_creator_id: PersonId, ) -> Result, Error> { let conn = &mut get_conn(pool).await?; let pictrs_search = "%pictrs/image%"; post::table .filter(post::creator_id.eq(for_creator_id)) .filter(post::url.like(pictrs_search)) .load::(conn) .await } /// Sets the url and thumbnails fields to None pub async fn remove_pictrs_post_images_and_thumbnails_for_creator( pool: &mut DbPool<'_>, for_creator_id: PersonId, ) -> Result, Error> { let conn = &mut get_conn(pool).await?; let pictrs_search = "%pictrs/image%"; diesel::update( post::table .filter(post::creator_id.eq(for_creator_id)) .filter(post::url.like(pictrs_search)), ) .set(( post::url.eq::>(None), post::thumbnail_url.eq::>(None), )) .get_results::(conn) .await } pub async fn fetch_pictrs_posts_for_community( pool: &mut DbPool<'_>, for_community_id: CommunityId, ) -> Result, Error> { let conn = &mut get_conn(pool).await?; let pictrs_search = "%pictrs/image%"; post::table .filter(post::community_id.eq(for_community_id)) .filter(post::url.like(pictrs_search)) .load::(conn) .await } /// Sets the url and thumbnails fields to None pub async fn remove_pictrs_post_images_and_thumbnails_for_community( pool: &mut DbPool<'_>, for_community_id: CommunityId, ) -> Result, Error> { let conn = &mut get_conn(pool).await?; let pictrs_search = "%pictrs/image%"; diesel::update( post::table .filter(post::community_id.eq(for_community_id)) .filter(post::url.like(pictrs_search)), ) .set(( post::url.eq::>(None), post::thumbnail_url.eq::>(None), )) .get_results::(conn) .await } pub async fn user_scheduled_post_count( person_id: PersonId, pool: &mut DbPool<'_>, ) -> Result { let conn = &mut get_conn(pool).await?; post::table .inner_join(person::table) .inner_join(community::table) // find all posts which have scheduled_publish_time that is in the past .filter(post::scheduled_publish_time.is_not_null()) .filter(coalesce(post::scheduled_publish_time, now()).lt(now())) // make sure the post and community are still around .filter(not(post::deleted.or(post::removed))) .filter(not(community::removed.or(community::deleted))) // only posts by specified user .filter(post::creator_id.eq(person_id)) .select(count(post::id)) .first::(conn) .await } } #[async_trait] impl Likeable for PostLike { type Form = PostLikeForm; type IdType = PostId; async fn like(pool: &mut DbPool<'_>, post_like_form: &PostLikeForm) -> Result { let conn = &mut get_conn(pool).await?; insert_into(post_like::table) .values(post_like_form) .on_conflict((post_like::post_id, post_like::person_id)) .do_update() .set(post_like_form) .get_result::(conn) .await } async fn remove( pool: &mut DbPool<'_>, person_id: PersonId, post_id: PostId, ) -> Result { let conn = &mut get_conn(pool).await?; diesel::delete(post_like::table.find((person_id, post_id))) .execute(conn) .await } } #[async_trait] impl Saveable for PostSaved { type Form = PostSavedForm; async fn save(pool: &mut DbPool<'_>, post_saved_form: &PostSavedForm) -> Result { let conn = &mut get_conn(pool).await?; insert_into(post_saved::table) .values(post_saved_form) .on_conflict((post_saved::post_id, post_saved::person_id)) .do_update() .set(post_saved_form) .get_result::(conn) .await } async fn unsave(pool: &mut DbPool<'_>, post_saved_form: &PostSavedForm) -> Result { let conn = &mut get_conn(pool).await?; diesel::delete(post_saved::table.find((post_saved_form.person_id, post_saved_form.post_id))) .execute(conn) .await } } impl PostRead { pub async fn mark_as_read( pool: &mut DbPool<'_>, post_id: PostId, person_id: PersonId, ) -> Result { let conn = &mut get_conn(pool).await?; let form = PostReadForm { post_id, person_id }; insert_into(post_read::table) .values(form) .on_conflict_do_nothing() .execute(conn) .await } pub async fn mark_as_unread( pool: &mut DbPool<'_>, post_id_: PostId, person_id_: PersonId, ) -> Result { let conn = &mut get_conn(pool).await?; let read_post = post_read::table .filter(post_read::post_id.eq(post_id_)) .filter(post_read::person_id.eq(person_id_)); diesel::delete(read_post).execute(conn).await } } impl PostHide { pub async fn hide( pool: &mut DbPool<'_>, post_id: PostId, person_id: PersonId, ) -> Result { let conn = &mut get_conn(pool).await?; let form = PostHideForm { post_id, person_id }; insert_into(post_hide::table) .values(form) .on_conflict_do_nothing() .execute(conn) .await } pub async fn unhide( pool: &mut DbPool<'_>, post_id_: PostId, person_id_: PersonId, ) -> Result { let conn = &mut get_conn(pool).await?; let hidden_post = post_hide::table .filter(post_hide::post_id.eq(post_id_)) .filter(post_hide::person_id.eq(person_id_)); diesel::delete(hidden_post).execute(conn).await } } #[cfg(test)] #[allow(clippy::indexing_slicing)] mod tests { use crate::{ source::{ community::{Community, CommunityInsertForm}, instance::Instance, person::{Person, PersonInsertForm}, post::{ Post, PostInsertForm, PostLike, PostLikeForm, PostRead, PostSaved, PostSavedForm, PostUpdateForm, }, }, traits::{Crud, Likeable, Saveable}, utils::build_db_pool_for_tests, }; use lemmy_utils::error::LemmyResult; use pretty_assertions::assert_eq; use serial_test::serial; use url::Url; #[tokio::test] #[serial] async fn test_crud() -> LemmyResult<()> { let pool = &build_db_pool_for_tests().await; let pool = &mut pool.into(); let inserted_instance = Instance::read_or_create(pool, "my_domain.tld".to_string()).await?; let new_person = PersonInsertForm::test_form(inserted_instance.id, "jim"); let inserted_person = Person::create(pool, &new_person).await?; let new_community = CommunityInsertForm::new( inserted_instance.id, "test community_3".to_string(), "nada".to_owned(), "pubkey".to_string(), ); let inserted_community = Community::create(pool, &new_community).await?; let new_post = PostInsertForm::new( "A test post".into(), inserted_person.id, inserted_community.id, ); let inserted_post = Post::create(pool, &new_post).await?; let new_post2 = PostInsertForm::new( "A test post 2".into(), inserted_person.id, inserted_community.id, ); let inserted_post2 = Post::create(pool, &new_post2).await?; let expected_post = Post { id: inserted_post.id, name: "A test post".into(), url: None, body: None, alt_text: None, creator_id: inserted_person.id, community_id: inserted_community.id, published: inserted_post.published, removed: false, locked: false, nsfw: false, deleted: false, updated: None, embed_title: None, embed_description: None, embed_video_url: None, thumbnail_url: None, ap_id: Url::parse(&format!("https://lemmy-alpha/post/{}", inserted_post.id))?.into(), local: true, language_id: Default::default(), featured_community: false, featured_local: false, url_content_type: None, scheduled_publish_time: None, }; // Post Like let post_like_form = PostLikeForm { post_id: inserted_post.id, person_id: inserted_person.id, score: 1, }; let inserted_post_like = PostLike::like(pool, &post_like_form).await?; let expected_post_like = PostLike { post_id: inserted_post.id, person_id: inserted_person.id, published: inserted_post_like.published, score: 1, }; // Post Save let post_saved_form = PostSavedForm { post_id: inserted_post.id, person_id: inserted_person.id, }; let inserted_post_saved = PostSaved::save(pool, &post_saved_form).await?; let expected_post_saved = PostSaved { post_id: inserted_post.id, person_id: inserted_person.id, published: inserted_post_saved.published, }; // Mark 2 posts as read PostRead::mark_as_read(pool, inserted_post.id, inserted_person.id).await?; PostRead::mark_as_read(pool, inserted_post2.id, inserted_person.id).await?; let read_post = Post::read(pool, inserted_post.id).await?; let new_post_update = PostUpdateForm { name: Some("A test post".into()), ..Default::default() }; let updated_post = Post::update(pool, inserted_post.id, &new_post_update).await?; let like_removed = PostLike::remove(pool, inserted_person.id, inserted_post.id).await?; assert_eq!(1, like_removed); let saved_removed = PostSaved::unsave(pool, &post_saved_form).await?; assert_eq!(1, saved_removed); // mark some posts as unread let read_removed_1 = PostRead::mark_as_unread(pool, inserted_post.id, inserted_person.id).await?; assert_eq!(1, read_removed_1); let read_removed_2 = PostRead::mark_as_unread(pool, inserted_post2.id, inserted_person.id).await?; assert_eq!(1, read_removed_2); let num_deleted = Post::delete(pool, inserted_post.id).await? + Post::delete(pool, inserted_post2.id).await?; assert_eq!(2, num_deleted); Community::delete(pool, inserted_community.id).await?; Person::delete(pool, inserted_person.id).await?; Instance::delete(pool, inserted_instance.id).await?; assert_eq!(expected_post, read_post); assert_eq!(expected_post, inserted_post); assert_eq!(expected_post, updated_post); assert_eq!(expected_post_like, inserted_post_like); assert_eq!(expected_post_saved, inserted_post_saved); Ok(()) } }