diff options
| -rw-r--r-- | luz/src/db/mod.rs | 26 | ||||
| -rw-r--r-- | luz/src/lib.rs | 52 | 
2 files changed, 78 insertions, 0 deletions
| diff --git a/luz/src/db/mod.rs b/luz/src/db/mod.rs index d9fb3e3..aea40ac 100644 --- a/luz/src/db/mod.rs +++ b/luz/src/db/mod.rs @@ -333,6 +333,32 @@ impl Db {          Ok(chats)      } +    /// chats ordered by date of last message +    // greatest-n-per-group +    pub(crate) async fn read_chats_ordered_with_latest_messages( +        &self, +    ) -> Result<Vec<(Chat, Message)>, Error> { +        #[derive(sqlx::FromRow)] +        pub struct ChatWithMessage { +            #[sqlx(flatten)] +            pub chat: Chat, +            #[sqlx(flatten)] +            pub message: Message, +        } + +        // TODO: i don't know if this will assign the right uuid to the latest message or the chat's id. should probably check but i don't think it matters as nothing ever gets called with the id of the latest message in the chats list +        let chats: Vec<ChatWithMessage> = sqlx::query_as("select c.*, m.* from chats c join (select chat_id, max(timestamp) max_timestamp from messages group by chat_id) max_timestamps on c.id = max_timestamps.chat_id join messages m on max_timestamps.chat_id = m.chat_id and max_timestamps.max_timestamp = m.timestamp order by m.timestamp desc") +            .fetch_all(&self.db) +            .await?; + +        let chats = chats +            .into_iter() +            .map(|chat_with_message| (chat_with_message.chat, chat_with_message.message)) +            .collect(); + +        Ok(chats) +    } +      async fn read_chat_id(&self, chat: JID) -> Result<Uuid, Error> {          #[derive(sqlx::FromRow)]          struct Row { diff --git a/luz/src/lib.rs b/luz/src/lib.rs index f7d8a23..b87891d 100644 --- a/luz/src/lib.rs +++ b/luz/src/lib.rs @@ -323,9 +323,20 @@ impl CommandMessage {                  }              }              CommandMessage::GetChats(sender) => { +                let chats = db.read_chats().await.map_err(|e| e.into()); +                sender.send(chats); +            } +            CommandMessage::GetChatsOrdered(sender) => {                  let chats = db.read_chats_ordered().await.map_err(|e| e.into());                  sender.send(chats);              } +            CommandMessage::GetChatsOrderedWithLatestMessages(sender) => { +                let chats = db +                    .read_chats_ordered_with_latest_messages() +                    .await +                    .map_err(|e| e.into()); +                sender.send(chats); +            }              CommandMessage::GetChat(jid, sender) => {                  let chats = db.read_chat(jid).await.map_err(|e| e.into());                  sender.send(chats); @@ -509,9 +520,20 @@ impl CommandMessage {                  }              }              CommandMessage::GetChats(sender) => { +                let chats = db.read_chats().await.map_err(|e| e.into()); +                sender.send(chats); +            } +            CommandMessage::GetChatsOrdered(sender) => {                  let chats = db.read_chats_ordered().await.map_err(|e| e.into());                  sender.send(chats);              } +            CommandMessage::GetChatsOrderedWithLatestMessages(sender) => { +                let chats = db +                    .read_chats_ordered_with_latest_messages() +                    .await +                    .map_err(|e| e.into()); +                sender.send(chats); +            }              CommandMessage::GetChat(jid, sender) => {                  let chats = db.read_chat(jid).await.map_err(|e| e.into());                  sender.send(chats); @@ -1133,6 +1155,32 @@ impl LuzHandle {          Ok(chats)      } +    pub async fn get_chats_ordered(&self) -> Result<Vec<Chat>, CommandError<DatabaseError>> { +        let (send, recv) = oneshot::channel(); +        self.send(CommandMessage::GetChatsOrdered(send)) +            .await +            .map_err(|e| CommandError::Actor(Into::<ActorError>::into(e)))?; +        let chats = timeout(self.timeout, recv) +            .await +            .map_err(|e| CommandError::Actor(Into::<ActorError>::into(e)))? +            .map_err(|e| CommandError::Actor(Into::<ActorError>::into(e)))??; +        Ok(chats) +    } + +    pub async fn get_chats_ordered_with_latest_messages( +        &self, +    ) -> Result<Vec<(Chat, Message)>, CommandError<DatabaseError>> { +        let (send, recv) = oneshot::channel(); +        self.send(CommandMessage::GetChatsOrderedWithLatestMessages(send)) +            .await +            .map_err(|e| CommandError::Actor(Into::<ActorError>::into(e)))?; +        let chats = timeout(self.timeout, recv) +            .await +            .map_err(|e| CommandError::Actor(Into::<ActorError>::into(e)))? +            .map_err(|e| CommandError::Actor(Into::<ActorError>::into(e)))??; +        Ok(chats) +    } +      pub async fn get_chat(&self, jid: JID) -> Result<Chat, CommandError<DatabaseError>> {          let (send, recv) = oneshot::channel();          self.send(CommandMessage::GetChat(jid, send)) @@ -1360,6 +1408,10 @@ pub enum CommandMessage {      /// get all chats. chat will include 10 messages in their message Vec (enough for chat previews)      // TODO: paging and filtering      GetChats(oneshot::Sender<Result<Vec<Chat>, DatabaseError>>), +    // TODO: paging and filtering +    GetChatsOrdered(oneshot::Sender<Result<Vec<Chat>, DatabaseError>>), +    // TODO: paging and filtering +    GetChatsOrderedWithLatestMessages(oneshot::Sender<Result<Vec<(Chat, Message)>, DatabaseError>>),      /// get a specific chat by jid      GetChat(JID, oneshot::Sender<Result<Chat, DatabaseError>>),      /// get message history for chat (does appropriate mam things) | 
