2023-12-15 16:16:18 +00:00
package peersyncing
import (
"database/sql"
"encoding/hex"
"fmt"
"strings"
)
type SyncMessagePersistence interface {
Add ( SyncMessage ) error
All ( ) ( [ ] SyncMessage , error )
Complement ( [ ] SyncMessage ) ( [ ] SyncMessage , error )
2024-05-14 12:20:13 +02:00
ByChatIDs ( [ ] [ ] byte , int ) ( [ ] SyncMessage , error )
2023-12-15 16:16:18 +00:00
ByMessageIDs ( [ ] [ ] byte ) ( [ ] SyncMessage , error )
}
type SyncMessageSQLitePersistence struct {
db * sql . DB
}
func NewSyncMessageSQLitePersistence ( db * sql . DB ) * SyncMessageSQLitePersistence {
return & SyncMessageSQLitePersistence { db : db }
}
func ( p * SyncMessageSQLitePersistence ) Add ( message SyncMessage ) error {
if err := message . Valid ( ) ; err != nil {
return err
}
2024-05-14 12:20:13 +02:00
_ , err := p . db . Exec ( ` INSERT INTO peersyncing_messages (id, type, chat_id, payload, timestamp) VALUES (?, ?, ?, ?, ?) ` , message . ID , message . Type , message . ChatID , message . Payload , message . Timestamp )
2023-12-15 16:16:18 +00:00
return err
}
func ( p * SyncMessageSQLitePersistence ) All ( ) ( [ ] SyncMessage , error ) {
var messages [ ] SyncMessage
2024-05-14 12:20:13 +02:00
rows , err := p . db . Query ( ` SELECT id, type, chat_id, payload, timestamp FROM peersyncing_messages ` )
2023-12-15 16:16:18 +00:00
if err != nil {
return nil , err
}
defer rows . Close ( )
for rows . Next ( ) {
var m SyncMessage
2024-05-14 12:20:13 +02:00
err := rows . Scan ( & m . ID , & m . Type , & m . ChatID , & m . Payload , & m . Timestamp )
2023-12-15 16:16:18 +00:00
if err != nil {
return nil , err
}
messages = append ( messages , m )
}
return messages , nil
}
2024-05-14 12:20:13 +02:00
func ( p * SyncMessageSQLitePersistence ) ByChatIDs ( ids [ ] [ ] byte , limit int ) ( [ ] SyncMessage , error ) {
if len ( ids ) == 0 {
return nil , nil
}
queryArgs := make ( [ ] interface { } , 0 , len ( ids ) )
for _ , id := range ids {
queryArgs = append ( queryArgs , id )
}
queryArgs = append ( queryArgs , limit )
inVector := strings . Repeat ( "?, " , len ( ids ) - 1 ) + "?"
query := "SELECT id, type, chat_id, payload, timestamp FROM peersyncing_messages WHERE chat_id IN (" + inVector + ") ORDER BY timestamp DESC LIMIT ?" // nolint: gosec
2023-12-15 16:16:18 +00:00
var messages [ ] SyncMessage
2024-05-14 12:20:13 +02:00
rows , err := p . db . Query ( query , queryArgs ... )
2023-12-15 16:16:18 +00:00
if err != nil {
return nil , err
}
defer rows . Close ( )
for rows . Next ( ) {
var m SyncMessage
2024-05-14 12:20:13 +02:00
err := rows . Scan ( & m . ID , & m . Type , & m . ChatID , & m . Payload , & m . Timestamp )
2023-12-15 16:16:18 +00:00
if err != nil {
return nil , err
}
messages = append ( messages , m )
}
return messages , nil
}
func ( p * SyncMessageSQLitePersistence ) Complement ( messages [ ] SyncMessage ) ( [ ] SyncMessage , error ) {
if len ( messages ) == 0 {
return nil , nil
}
ids := make ( [ ] interface { } , 0 , len ( messages ) )
for _ , m := range messages {
ids = append ( ids , m . ID )
}
inVector := strings . Repeat ( "?, " , len ( ids ) - 1 ) + "?"
2024-05-14 12:20:13 +02:00
query := "SELECT id, type, chat_id, payload, timestamp FROM peersyncing_messages WHERE id IN (" + inVector + ")" // nolint: gosec
2023-12-15 16:16:18 +00:00
availableMessages := make ( map [ string ] SyncMessage )
rows , err := p . db . Query ( query , ids ... )
if err != nil {
return nil , err
}
defer rows . Close ( )
for rows . Next ( ) {
var m SyncMessage
2024-05-14 12:20:13 +02:00
err := rows . Scan ( & m . ID , & m . Type , & m . ChatID , & m . Payload , & m . Timestamp )
2023-12-15 16:16:18 +00:00
if err != nil {
return nil , err
}
fmt . Printf ( "GOT MESSAGE: %x\n" , m . ID )
availableMessages [ hex . EncodeToString ( m . ID ) ] = m
}
var complement [ ] SyncMessage
for _ , m := range messages {
fmt . Printf ( "CHECKING MESSAGE: %x\n" , m . ID )
if _ , ok := availableMessages [ hex . EncodeToString ( m . ID ) ] ; ! ok {
complement = append ( complement , m )
}
}
return complement , nil
}
func ( p * SyncMessageSQLitePersistence ) ByMessageIDs ( ids [ ] [ ] byte ) ( [ ] SyncMessage , error ) {
if len ( ids ) == 0 {
return nil , nil
}
queryArgs := make ( [ ] interface { } , 0 , len ( ids ) )
for _ , id := range ids {
queryArgs = append ( queryArgs , id )
}
inVector := strings . Repeat ( "?, " , len ( ids ) - 1 ) + "?"
2024-05-14 12:20:13 +02:00
query := "SELECT id, type, chat_id, payload, timestamp FROM peersyncing_messages WHERE id IN (" + inVector + ")" // nolint: gosec
2023-12-15 16:16:18 +00:00
var messages [ ] SyncMessage
rows , err := p . db . Query ( query , queryArgs ... )
if err != nil {
return nil , err
}
defer rows . Close ( )
for rows . Next ( ) {
var m SyncMessage
2024-05-14 12:20:13 +02:00
err := rows . Scan ( & m . ID , & m . Type , & m . ChatID , & m . Payload , & m . Timestamp )
2023-12-15 16:16:18 +00:00
if err != nil {
return nil , err
}
messages = append ( messages , m )
}
return messages , nil
}