2023-09-06 17:21:29 +02:00
package database
import (
2023-09-07 22:48:34 +02:00
"context"
2023-09-06 17:21:29 +02:00
"database/sql"
"fmt"
2023-09-25 03:49:40 +02:00
"strings"
2023-09-27 05:11:06 +02:00
"time"
2023-09-12 21:56:05 +02:00
2023-09-06 17:21:29 +02:00
"github.com/ddworken/hishtory/shared"
"github.com/jackc/pgx/v4/stdlib"
_ "github.com/lib/pq"
sqltrace "gopkg.in/DataDog/dd-trace-go.v1/contrib/database/sql"
gormtrace "gopkg.in/DataDog/dd-trace-go.v1/contrib/gorm.io/gorm.v1"
"gorm.io/driver/postgres"
"gorm.io/driver/sqlite"
"gorm.io/gorm"
)
type DB struct {
* gorm . DB
}
func OpenSQLite ( dsn string , config * gorm . Config ) ( * DB , error ) {
db , err := gorm . Open ( sqlite . Open ( dsn ) , config )
if err != nil {
return nil , fmt . Errorf ( "gorm.Open: %w" , err )
}
return & DB { db } , nil
}
func OpenPostgres ( dsn string , config * gorm . Config ) ( * DB , error ) {
sqltrace . Register ( "pgx" , & stdlib . Driver { } , sqltrace . WithServiceName ( "hishtory-api" ) )
sqlDb , err := sqltrace . Open ( "pgx" , dsn )
if err != nil {
return nil , fmt . Errorf ( "sqltrace.Open: %w" , err )
}
db , err := gormtrace . Open ( postgres . New ( postgres . Config { Conn : sqlDb } ) , config )
if err != nil {
return nil , fmt . Errorf ( "gormtrace.Open: %w" , err )
}
return & DB { db } , nil
}
func ( db * DB ) AddDatabaseTables ( ) error {
models := [ ] any {
& shared . EncHistoryEntry { } ,
& shared . Device { } ,
& shared . UsageData { } ,
& shared . DumpRequest { } ,
& shared . DeletionRequest { } ,
& shared . Feedback { } ,
2023-09-27 05:11:06 +02:00
& ActiveUserStats { } ,
2023-09-06 17:21:29 +02:00
}
for _ , model := range models {
if err := db . AutoMigrate ( model ) ; err != nil {
return fmt . Errorf ( "db.AutoMigrate: %w" , err )
}
}
return nil
}
2023-09-23 04:11:43 +02:00
func ( db * DB ) CreateIndices ( ) error {
// Note: If adding a new index here, consider manually running it on the prod DB using CONCURRENTLY to
// make server startup non-blocking. The benefit of this function is primarily for other people so they
// don't have to manually create these indexes.
2023-09-25 03:49:40 +02:00
indices := [ ] struct {
name string
table string
columns [ ] string
} {
{ "entry_id_idx" , "enc_history_entries" , [ ] string { "encrypted_id" } } ,
{ "device_id_idx" , "enc_history_entries" , [ ] string { "device_id" } } ,
{ "read_count_idx" , "enc_history_entries" , [ ] string { "read_count" } } ,
{ "redact_idx" , "enc_history_entries" , [ ] string { "user_id" , "device_id" , "date" } } ,
{ "del_user_idx" , "deletion_requests" , [ ] string { "user_id" } } ,
2023-09-23 04:11:43 +02:00
}
for _ , index := range indices {
2023-09-25 03:49:40 +02:00
sql := ""
if db . Name ( ) == "sqlite" {
sql = fmt . Sprintf ( "CREATE INDEX IF NOT EXISTS %s ON %s (%s)" , index . name , index . table , strings . Join ( index . columns , "," ) )
} else {
sql = fmt . Sprintf ( "CREATE INDEX IF NOT EXISTS %s ON %s USING btree(%s)" , index . name , index . table , strings . Join ( index . columns , "," ) )
}
r := db . Exec ( sql )
2023-09-23 04:11:43 +02:00
if r . Error != nil {
return fmt . Errorf ( "failed to execute index creation sql=%#v: %w" , index , r . Error )
}
}
return nil
}
2023-09-06 17:21:29 +02:00
func ( db * DB ) Close ( ) error {
rawDB , err := db . DB . DB ( )
if err != nil {
return fmt . Errorf ( "db.DB.DB: %w" , err )
}
if err := rawDB . Close ( ) ; err != nil {
return fmt . Errorf ( "rawDB.Close: %w" , err )
}
return nil
}
func ( db * DB ) Ping ( ) error {
rawDB , err := db . DB . DB ( )
if err != nil {
return fmt . Errorf ( "db.DB.DB: %w" , err )
}
if err := rawDB . Ping ( ) ; err != nil {
return fmt . Errorf ( "rawDB.Ping: %w" , err )
}
return nil
}
2023-09-08 16:57:44 +02:00
func ( db * DB ) SetMaxIdleConns ( n int ) error {
rawDB , err := db . DB . DB ( )
if err != nil {
return err
}
rawDB . SetMaxIdleConns ( n )
return nil
}
2023-09-06 17:21:29 +02:00
func ( db * DB ) Stats ( ) ( sql . DBStats , error ) {
rawDB , err := db . DB . DB ( )
if err != nil {
return sql . DBStats { } , fmt . Errorf ( "db.DB.DB: %w" , err )
}
return rawDB . Stats ( ) , nil
}
2023-09-07 22:48:34 +02:00
func ( db * DB ) DistinctUsers ( ctx context . Context ) ( int64 , error ) {
row := db . WithContext ( ctx ) . Raw ( "SELECT COUNT(DISTINCT devices.user_id) FROM devices" ) . Row ( )
var numDistinctUsers int64
err := row . Scan ( & numDistinctUsers )
if err != nil {
return 0 , fmt . Errorf ( "row.Scan: %w" , err )
}
return numDistinctUsers , nil
}
2023-09-08 16:57:44 +02:00
func ( db * DB ) DumpRequestCreate ( ctx context . Context , req * shared . DumpRequest ) error {
tx := db . WithContext ( ctx ) . Create ( req )
2023-09-07 22:48:34 +02:00
if tx . Error != nil {
2023-09-08 16:57:44 +02:00
return fmt . Errorf ( "tx.Error: %w" , tx . Error )
2023-09-07 22:48:34 +02:00
}
2023-09-08 16:57:44 +02:00
return nil
2023-09-07 22:48:34 +02:00
}
2023-09-08 16:57:44 +02:00
func ( db * DB ) DumpRequestForUserAndDevice ( ctx context . Context , userID , deviceID string ) ( [ ] * shared . DumpRequest , error ) {
var dumpRequests [ ] * shared . DumpRequest
// Filter out ones requested by the hishtory instance that sent this request
tx := db . WithContext ( ctx ) . Where ( "user_id = ? AND requesting_device_id != ?" , userID , deviceID ) . Find ( & dumpRequests )
2023-09-07 22:48:34 +02:00
if tx . Error != nil {
2023-09-08 16:57:44 +02:00
return nil , fmt . Errorf ( "tx.Error: %w" , tx . Error )
2023-09-07 22:48:34 +02:00
}
2023-09-08 16:57:44 +02:00
return dumpRequests , nil
2023-09-07 22:48:34 +02:00
}
2023-09-08 16:57:44 +02:00
func ( db * DB ) DumpRequestDeleteForUserAndDevice ( ctx context . Context , userID , deviceID string ) error {
tx := db . WithContext ( ctx ) . Delete ( & shared . DumpRequest { } , "user_id = ? AND requesting_device_id = ?" , userID , deviceID )
2023-09-07 22:48:34 +02:00
if tx . Error != nil {
return fmt . Errorf ( "tx.Error: %w" , tx . Error )
}
return nil
}
2023-09-08 16:57:44 +02:00
func ( db * DB ) ApplyDeletionRequestsToBackend ( ctx context . Context , request * shared . DeletionRequest ) ( int64 , error ) {
tx := db . WithContext ( ctx ) . Where ( "false" )
for _ , message := range request . Messages . Ids {
2023-09-22 23:03:41 +02:00
// Note that we do an OR with date or the ID matching since the ID is not always recorded for older history entries.
2023-09-23 03:59:37 +02:00
tx = tx . Or ( db . WithContext ( ctx ) . Where ( "user_id = ? AND device_id = ? AND (date = ? OR encrypted_id = ?)" , request . UserId , message . DeviceId , message . EndTime , message . EntryId ) )
2023-09-08 16:57:44 +02:00
}
result := tx . Delete ( & shared . EncHistoryEntry { } )
if tx . Error != nil {
return 0 , fmt . Errorf ( "tx.Error: %w" , tx . Error )
}
return result . RowsAffected , nil
}
func ( db * DB ) DeletionRequestInc ( ctx context . Context , userID , deviceID string ) error {
tx := db . WithContext ( ctx ) . Exec ( "UPDATE deletion_requests SET read_count = read_count + 1 WHERE user_id = ? AND destination_device_id = ?" , userID , deviceID )
2023-09-07 22:48:34 +02:00
if tx . Error != nil {
return fmt . Errorf ( "tx.Error: %w" , tx . Error )
}
return nil
}
2023-09-08 16:57:44 +02:00
func ( db * DB ) DeletionRequestsForUserAndDevice ( ctx context . Context , userID , deviceID string ) ( [ ] * shared . DeletionRequest , error ) {
var deletionRequests [ ] * shared . DeletionRequest
tx := db . WithContext ( ctx ) . Where ( "user_id = ? AND destination_device_id = ?" , userID , deviceID ) . Find ( & deletionRequests )
2023-09-07 22:48:34 +02:00
if tx . Error != nil {
2023-09-08 16:57:44 +02:00
return nil , fmt . Errorf ( "tx.Error: %w" , tx . Error )
}
return deletionRequests , nil
}
func ( db * DB ) DeletionRequestCreate ( ctx context . Context , request * shared . DeletionRequest ) error {
userID := request . UserId
devices , err := db . DevicesForUser ( ctx , userID )
if err != nil {
return fmt . Errorf ( "db.DevicesForUser: %w" , err )
2023-09-07 22:48:34 +02:00
}
2023-09-08 16:57:44 +02:00
if len ( devices ) == 0 {
return fmt . Errorf ( "found no devices associated with user_id=%s, can't save history entry" , userID )
}
fmt . Printf ( "db.DeletionRequestCreate: Found %d devices\n" , len ( devices ) )
// TODO: maybe this should be a transaction?
for _ , device := range devices {
request . DestinationDeviceId = device . DeviceId
tx := db . WithContext ( ctx ) . Create ( & request )
if tx . Error != nil {
return fmt . Errorf ( "tx.Error: %w" , tx . Error )
}
}
numDeleted , err := db . ApplyDeletionRequestsToBackend ( ctx , request )
if err != nil {
return fmt . Errorf ( "db.ApplyDeletionRequestsToBackend: %w" , err )
}
fmt . Printf ( "addDeletionRequestHandler: Deleted %d rows in the backend\n" , numDeleted )
return nil
}
func ( db * DB ) FeedbackCreate ( ctx context . Context , feedback * shared . Feedback ) error {
tx := db . WithContext ( ctx ) . Create ( feedback )
if tx . Error != nil {
return fmt . Errorf ( "tx.Error: %w" , tx . Error )
}
return nil
}
func ( db * DB ) Clean ( ctx context . Context ) error {
r := db . WithContext ( ctx ) . Exec ( "DELETE FROM enc_history_entries WHERE read_count > 10" )
if r . Error != nil {
return r . Error
}
2023-09-23 04:19:06 +02:00
r = db . WithContext ( ctx ) . Exec ( "DELETE FROM deletion_requests WHERE read_count > 100" )
2023-09-08 16:57:44 +02:00
if r . Error != nil {
return r . Error
}
return nil
}
2023-09-27 05:11:06 +02:00
func extractInt64FromRow ( row * sql . Row ) ( int64 , error ) {
var ret int64
err := row . Scan ( & ret )
if err != nil {
return 0 , fmt . Errorf ( "extractInt64FromRow: %w" , err )
}
return ret , nil
}
type ActiveUserStats struct {
Date time . Time
TotalNumDevices int64
TotalNumUsers int64
DailyActiveSubmitUsers int64
DailyActiveQueryUsers int64
WeeklyActiveSubmitUsers int64
WeeklyActiveQueryUsers int64
DailyInstalls int64
DailyUninstalls int64
}
func ( db * DB ) GenerateAndStoreActiveUserStats ( ctx context . Context ) error {
totalNumDevices , err := extractInt64FromRow ( db . WithContext ( ctx ) . Raw ( "SELECT COUNT(DISTINCT devices.device_id) FROM devices" ) . Row ( ) )
if err != nil {
return err
}
totalNumUsers , err := extractInt64FromRow ( db . WithContext ( ctx ) . Raw ( "SELECT COUNT(DISTINCT devices.user_id) FROM devices" ) . Row ( ) )
if err != nil {
return err
}
dauSubmit , err := extractInt64FromRow ( db . WithContext ( ctx ) . Raw ( "SELECT COUNT(DISTINCT user_id) FROM usage_data WHERE last_used > (now()::date-1)::timestamp" ) . Row ( ) )
if err != nil {
return err
}
dauQuery , err := extractInt64FromRow ( db . WithContext ( ctx ) . Raw ( "SELECT COUNT(DISTINCT user_id) FROM usage_data WHERE last_queried > (now()::date-1)::timestamp" ) . Row ( ) )
if err != nil {
return err
}
wauSubmit , err := extractInt64FromRow ( db . WithContext ( ctx ) . Raw ( "SELECT COUNT(DISTINCT user_id) FROM usage_data WHERE last_used > (now()::date-7)::timestamp" ) . Row ( ) )
if err != nil {
return err
}
wauQuery , err := extractInt64FromRow ( db . WithContext ( ctx ) . Raw ( "SELECT COUNT(DISTINCT user_id) FROM usage_data WHERE last_queried > (now()::date-7)::timestamp" ) . Row ( ) )
if err != nil {
return err
}
dailyInstalls , err := extractInt64FromRow ( db . WithContext ( ctx ) . Raw ( "SELECT count(distinct device_id) FROM devices WHERE registration_date > (now()::date-1)::timestamp" ) . Row ( ) )
if err != nil {
return err
}
dailyUninstalls , err := extractInt64FromRow ( db . WithContext ( ctx ) . Raw ( "SELECT COUNT(*) FROM feedbacks WHERE date > (now()::date-1)::timestamp" ) . Row ( ) )
if err != nil {
return err
}
return db . Create ( ActiveUserStats {
Date : time . Now ( ) ,
TotalNumDevices : totalNumDevices ,
TotalNumUsers : totalNumUsers ,
DailyActiveSubmitUsers : dauSubmit ,
DailyActiveQueryUsers : dauQuery ,
WeeklyActiveSubmitUsers : wauSubmit ,
WeeklyActiveQueryUsers : wauQuery ,
DailyInstalls : dailyInstalls ,
DailyUninstalls : dailyUninstalls ,
} ) . Error
}
2023-09-08 16:57:44 +02:00
func ( db * DB ) DeepClean ( ctx context . Context ) error {
return db . WithContext ( ctx ) . Transaction ( func ( tx * gorm . DB ) error {
r := tx . Exec ( `
CREATE TEMP TABLE temp_users_with_one_device AS (
SELECT user_id
FROM devices
GROUP BY user_id
HAVING COUNT ( DISTINCT device_id ) > 1
)
` )
if r . Error != nil {
return r . Error
}
r = tx . Exec ( `
CREATE TEMP TABLE temp_inactive_users AS (
SELECT user_id
FROM usage_data
WHERE last_used <= ( now ( ) - INTERVAL ' 90 days ' )
)
` )
if r . Error != nil {
return r . Error
}
r = tx . Exec ( `
SELECT COUNT ( * ) FROM enc_history_entries WHERE
date <= ( now ( ) - INTERVAL ' 90 days ' )
AND user_id IN ( SELECT * FROM temp_users_with_one_device )
AND user_id IN ( SELECT * FROM temp_inactive_users )
` )
if r . Error != nil {
return r . Error
}
fmt . Printf ( "Ran deep clean and deleted %d rows\n" , r . RowsAffected )
return nil
} )
2023-09-07 22:48:34 +02:00
}