use crate::{
- naive_now,
newtypes::{CommentId, DbUrl, PersonId},
source::comment::{
Comment,
CommentSavedForm,
},
traits::{Crud, DeleteableOrRemoveable, Likeable, Saveable},
+ utils::naive_now,
};
use diesel::{dsl::*, result::Error, *};
+use diesel_ltree::Ltree;
use url::Url;
impl Comment {
.get_results::<Self>(conn)
}
- pub fn update_read(
+ pub fn update_content(
conn: &PgConnection,
comment_id: CommentId,
- new_read: bool,
+ new_content: &str,
) -> Result<Self, Error> {
use crate::schema::comment::dsl::*;
diesel::update(comment.find(comment_id))
- .set(read.eq(new_read))
+ .set((content.eq(new_content), updated.eq(naive_now())))
.get_result::<Self>(conn)
}
- pub fn update_content(
+ pub fn update_distinguished(
conn: &PgConnection,
comment_id: CommentId,
- new_content: &str,
+ new_distinguished: bool,
) -> Result<Self, Error> {
use crate::schema::comment::dsl::*;
diesel::update(comment.find(comment_id))
- .set((content.eq(new_content), updated.eq(naive_now())))
+ .set((distinguished.eq(new_distinguished), updated.eq(naive_now())))
.get_result::<Self>(conn)
}
- pub fn upsert(conn: &PgConnection, comment_form: &CommentForm) -> Result<Comment, Error> {
+ pub fn create(
+ conn: &PgConnection,
+ comment_form: &CommentForm,
+ parent_path: Option<&Ltree>,
+ ) -> Result<Comment, Error> {
use crate::schema::comment::dsl::*;
- insert_into(comment)
+
+ // Insert, to get the id
+ let inserted_comment = insert_into(comment)
.values(comment_form)
.on_conflict(ap_id)
.do_update()
.set(comment_form)
- .get_result::<Self>(conn)
+ .get_result::<Self>(conn);
+
+ if let Ok(comment_insert) = inserted_comment {
+ let comment_id = comment_insert.id;
+
+ // You need to update the ltree column
+ let ltree = Ltree(if let Some(parent_path) = parent_path {
+ // The previous parent will already have 0 in it
+ // Append this comment id
+ format!("{}.{}", parent_path.0, comment_id)
+ } else {
+ // '0' is always the first path, append to that
+ format!("{}.{}", 0, comment_id)
+ });
+
+ let updated_comment = diesel::update(comment.find(comment_id))
+ .set(path.eq(ltree))
+ .get_result::<Self>(conn);
+
+ // Update the child count for the parent comment_aggregates
+ // You could do this with a trigger, but since you have to do this manually anyway,
+ // you can just have it here
+ if let Some(parent_path) = parent_path {
+ // You have to update counts for all parents, not just the immediate one
+ // TODO if the performance of this is terrible, it might be better to do this as part of a
+ // scheduled query... although the counts would often be wrong.
+ //
+ // The child_count query for reference:
+ // select c.id, c.path, count(c2.id) as child_count from comment c
+ // left join comment c2 on c2.path <@ c.path and c2.path != c.path
+ // group by c.id
+
+ let top_parent = format!("0.{}", parent_path.0.split('.').collect::<Vec<&str>>()[1]);
+ let update_child_count_stmt = format!(
+ "
+update comment_aggregates ca set child_count = c.child_count
+from (
+ select c.id, c.path, count(c2.id) as child_count from comment c
+ join comment c2 on c2.path <@ c.path and c2.path != c.path
+ and c.path <@ '{}'
+ group by c.id
+) as c
+where ca.comment_id = c.id",
+ top_parent
+ );
+
+ sql_query(update_child_count_stmt).execute(conn)?;
+ }
+ updated_comment
+ } else {
+ inserted_comment
+ }
}
pub fn read_from_apub_id(conn: &PgConnection, object_id: Url) -> Result<Option<Self>, Error> {
use crate::schema::comment::dsl::*;
.map(Into::into),
)
}
+
+ pub fn parent_comment_id(&self) -> Option<CommentId> {
+ let mut ltree_split: Vec<&str> = self.path.0.split('.').collect();
+ ltree_split.remove(0); // The first is always 0
+ if ltree_split.len() > 1 {
+ ltree_split[ltree_split.len() - 2]
+ .parse::<i32>()
+ .map(CommentId)
+ .ok()
+ } else {
+ None
+ }
+ }
}
impl Crud for Comment {
diesel::delete(comment.find(comment_id)).execute(conn)
}
- fn create(conn: &PgConnection, comment_form: &CommentForm) -> Result<Self, Error> {
- use crate::schema::comment::dsl::*;
- insert_into(comment)
- .values(comment_form)
- .get_result::<Self>(conn)
+ /// This is unimplemented, use [[Comment::create]]
+ fn create(_conn: &PgConnection, _comment_form: &CommentForm) -> Result<Self, Error> {
+ unimplemented!();
}
fn update(
#[cfg(test)]
mod tests {
use crate::{
- establish_unpooled_connection,
+ newtypes::LanguageId,
source::{
comment::*,
community::{Community, CommunityForm},
post::*,
},
traits::{Crud, Likeable, Saveable},
+ utils::establish_unpooled_connection,
};
+ use diesel_ltree::Ltree;
use serial_test::serial;
#[test]
let new_person = PersonForm {
name: "terry".into(),
+ public_key: Some("pubkey".to_string()),
..PersonForm::default()
};
let new_community = CommunityForm {
name: "test community".to_string(),
title: "nada".to_owned(),
+ public_key: Some("pubkey".to_string()),
..CommunityForm::default()
};
..CommentForm::default()
};
- let inserted_comment = Comment::create(&conn, &comment_form).unwrap();
+ let inserted_comment = Comment::create(&conn, &comment_form, None).unwrap();
let expected_comment = Comment {
id: inserted_comment.id,
post_id: inserted_post.id,
removed: false,
deleted: false,
- read: false,
- parent_id: None,
+ path: Ltree(format!("0.{}", inserted_comment.id)),
published: inserted_comment.published,
updated: None,
ap_id: inserted_comment.ap_id.to_owned(),
+ distinguished: false,
local: true,
+ language_id: LanguageId::default(),
};
let child_comment_form = CommentForm {
content: "A child comment".into(),
creator_id: inserted_person.id,
post_id: inserted_post.id,
- parent_id: Some(inserted_comment.id),
+ // path: Some(text2ltree(inserted_comment.id),
..CommentForm::default()
};
- let inserted_child_comment = Comment::create(&conn, &child_comment_form).unwrap();
+ let inserted_child_comment =
+ Comment::create(&conn, &child_comment_form, Some(&inserted_comment.path)).unwrap();
// Comment Like
let comment_like_form = CommentLikeForm {
assert_eq!(expected_comment_like, inserted_comment_like);
assert_eq!(expected_comment_saved, inserted_comment_saved);
assert_eq!(
- expected_comment.id,
- inserted_child_comment.parent_id.unwrap()
+ format!("0.{}.{}", expected_comment.id, inserted_child_comment.id),
+ inserted_child_comment.path.0,
);
assert_eq!(1, like_removed);
assert_eq!(1, saved_removed);