2021-08-22 19:34:28 +00:00
# include "ssb.db.h"
2023-03-07 17:50:17 +00:00
# include "log.h"
2022-06-04 17:04:51 +00:00
# include "mem.h"
2021-08-22 19:34:28 +00:00
# include "ssb.h"
# include "trace.h"
2022-07-09 15:13:35 +00:00
# include "util.js.h"
2021-08-22 19:34:28 +00:00
2024-04-04 21:00:59 -04:00
# include "ow-crypt.h"
2023-05-21 21:36:51 +00:00
# include "sodium/crypto_hash_sha256.h"
# include "sodium/crypto_sign.h"
# include "sqlite3.h"
2023-07-18 23:46:15 +00:00
# include "uv.h"
2023-05-21 21:36:51 +00:00
2021-08-22 19:34:28 +00:00
# include <stdlib.h>
# include <string.h>
2023-07-20 01:02:50 +00:00
# include <unistd.h>
2023-07-20 02:20:38 +00:00
typedef struct _message_store_t message_store_t ;
static void _tf_ssb_db_store_message_work_finish ( message_store_t * store ) ;
static void _tf_ssb_db_store_message_after_work ( uv_work_t * work , int status ) ;
2022-02-12 01:44:11 +00:00
static void _tf_ssb_db_exec ( sqlite3 * db , const char * statement )
{
char * error = NULL ;
int result = sqlite3_exec ( db , statement , NULL , NULL , & error ) ;
if ( result ! = SQLITE_OK )
{
2023-07-20 01:02:50 +00:00
tf_printf ( " Error running '%s': %s. \n " , statement , error ? error : sqlite3_errmsg ( db ) ) ;
2022-02-12 01:44:11 +00:00
abort ( ) ;
}
}
2022-09-10 17:56:54 +00:00
static bool _tf_ssb_db_has_rows ( sqlite3 * db , const char * query )
{
bool found = false ;
sqlite3_stmt * statement = NULL ;
if ( sqlite3_prepare ( db , query , - 1 , & statement , NULL ) = = SQLITE_OK )
{
int result = SQLITE_OK ;
while ( ( result = sqlite3_step ( statement ) ) = = SQLITE_ROW )
{
found = true ;
}
if ( result ! = SQLITE_DONE )
{
2023-03-07 17:50:17 +00:00
tf_printf ( " %s \n " , sqlite3_errmsg ( db ) ) ;
2022-09-10 17:56:54 +00:00
abort ( ) ;
}
sqlite3_finalize ( statement ) ;
}
else
{
2023-03-07 17:50:17 +00:00
tf_printf ( " %s \n " , sqlite3_errmsg ( db ) ) ;
2022-09-10 17:56:54 +00:00
abort ( ) ;
}
return found ;
}
2023-02-08 01:29:44 +00:00
static void _tf_ssb_db_init_internal ( sqlite3 * db )
2021-08-22 19:34:28 +00:00
{
2023-02-17 22:43:19 +00:00
sqlite3_extended_result_codes ( db , 1 ) ;
2022-02-12 01:44:11 +00:00
_tf_ssb_db_exec ( db , " PRAGMA journal_mode = WAL " ) ;
_tf_ssb_db_exec ( db , " PRAGMA synchronous = NORMAL " ) ;
2023-02-08 01:29:44 +00:00
}
2023-02-17 22:43:19 +00:00
void tf_ssb_db_init_reader ( sqlite3 * db )
{
_tf_ssb_db_init_internal ( db ) ;
}
2023-02-08 01:29:44 +00:00
void tf_ssb_db_init ( tf_ssb_t * ssb )
{
2023-06-15 00:27:49 +00:00
sqlite3 * db = tf_ssb_acquire_db_writer ( ssb ) ;
2023-02-08 01:29:44 +00:00
_tf_ssb_db_init_internal ( db ) ;
2023-07-27 12:22:37 +00:00
sqlite3_stmt * statement = NULL ;
int auto_vacuum = 0 ;
if ( sqlite3_prepare ( db , " PRAGMA auto_vacuum " , - 1 , & statement , NULL ) = = SQLITE_OK )
{
if ( sqlite3_step ( statement ) = = SQLITE_ROW )
{
auto_vacuum = sqlite3_column_int ( statement , 0 ) ;
}
sqlite3_finalize ( statement ) ;
}
if ( auto_vacuum ! = 1 /* FULL */ )
{
tf_printf ( " Enabling auto-vacuum and performing full vacuum. \n " ) ;
_tf_ssb_db_exec ( db , " PRAGMA auto_vacuum = FULL " ) ;
_tf_ssb_db_exec ( db , " VACUUM main " ) ;
tf_printf ( " All clean. \n " ) ;
}
2022-02-12 01:44:11 +00:00
_tf_ssb_db_exec ( db ,
2024-02-17 19:22:02 +00:00
" CREATE TABLE IF NOT EXISTS messages ( "
" author TEXT, "
" id TEXT PRIMARY KEY, "
" sequence INTEGER, "
" timestamp REAL, "
" previous TEXT, "
" hash TEXT, "
2024-02-28 20:01:52 -05:00
" content BLOB, "
2024-02-17 19:22:02 +00:00
" signature TEXT, "
2024-03-13 19:40:09 -04:00
" flags INTEGER, "
2024-02-17 19:22:02 +00:00
" UNIQUE(author, sequence) "
" ) " ) ;
2024-02-28 20:01:52 -05:00
if ( _tf_ssb_db_has_rows ( db , " SELECT name FROM pragma_table_info('messages') WHERE name = 'content' AND type == 'TEXT' " ) )
{
tf_printf ( " converting to JSONB \n " ) ;
_tf_ssb_db_exec ( db , " DROP TRIGGER IF EXISTS messages_ai_refs " ) ;
_tf_ssb_db_exec ( db , " DROP TRIGGER IF EXISTS messages_ad_refs " ) ;
_tf_ssb_db_exec ( db , " DROP TRIGGER IF EXISTS messages_ai " ) ;
_tf_ssb_db_exec ( db , " DROP TRIGGER IF EXISTS messages_ad " ) ;
_tf_ssb_db_exec ( db , " DROP TABLE IF EXISTS messages_fts " ) ;
_tf_ssb_db_exec ( db , " BEGIN TRANSACTION " ) ;
_tf_ssb_db_exec ( db , " ALTER TABLE messages ADD COLUMN contentb BLOB " ) ;
_tf_ssb_db_exec ( db , " UPDATE messages SET contentb = jsonb(content) " ) ;
_tf_ssb_db_exec ( db , " ALTER TABLE messages DROP COLUMN content " ) ;
_tf_ssb_db_exec ( db , " ALTER TABLE messages RENAME COLUMN contentb TO content " ) ;
_tf_ssb_db_exec ( db , " COMMIT TRANSACTION " ) ;
}
2024-03-13 19:40:09 -04:00
if ( _tf_ssb_db_has_rows ( db , " SELECT name FROM pragma_table_info('messages') WHERE name = 'sequence_before_author' " ) )
{
tf_printf ( " Renaming sequence_before_author -> flags. \n " ) ;
_tf_ssb_db_exec ( db , " ALTER TABLE messages RENAME COLUMN sequence_before_author TO flags " ) ;
}
2022-02-12 01:44:11 +00:00
_tf_ssb_db_exec ( db , " CREATE INDEX IF NOT EXISTS messages_author_id_index ON messages (author, id) " ) ;
_tf_ssb_db_exec ( db , " CREATE INDEX IF NOT EXISTS messages_author_sequence_index ON messages (author, sequence) " ) ;
_tf_ssb_db_exec ( db , " CREATE INDEX IF NOT EXISTS messages_author_timestamp_index ON messages (author, timestamp) " ) ;
2023-08-16 22:57:16 +00:00
_tf_ssb_db_exec ( db , " CREATE INDEX IF NOT EXISTS messages_timestamp_index ON messages (timestamp) " ) ;
2022-02-12 01:44:11 +00:00
_tf_ssb_db_exec ( db ,
2024-02-17 19:22:02 +00:00
" CREATE TABLE IF NOT EXISTS blobs ( "
" id TEXT PRIMARY KEY, "
" content BLOB, "
" created INTEGER "
" ) " ) ;
2024-02-15 23:35:01 +00:00
_tf_ssb_db_exec ( db , " DROP TABLE IF EXISTS blob_wants " ) ;
2022-02-12 01:44:11 +00:00
_tf_ssb_db_exec ( db ,
2024-02-17 19:22:02 +00:00
" CREATE TABLE IF NOT EXISTS properties ( "
" id TEXT, "
" key TEXT, "
" value TEXT, "
" UNIQUE(id, key) "
" ) " ) ;
2022-02-12 01:44:11 +00:00
_tf_ssb_db_exec ( db ,
2024-02-17 19:22:02 +00:00
" CREATE TABLE IF NOT EXISTS connections ( "
" host TEXT, "
" port INTEGER, "
" key TEXT, "
" last_attempt INTEGER, "
" last_success INTEGER, "
" UNIQUE(host, port, key) "
" ) " ) ;
2022-07-14 01:01:14 +00:00
_tf_ssb_db_exec ( db ,
2024-02-17 19:22:02 +00:00
" CREATE TABLE IF NOT EXISTS identities ( "
" user TEXT, "
" public_key TEXT UNIQUE, "
" private_key TEXT UNIQUE "
" ) " ) ;
2022-07-14 01:01:14 +00:00
_tf_ssb_db_exec ( db , " CREATE INDEX IF NOT EXISTS identities_user ON identities (user, public_key) " ) ;
2022-02-12 01:44:11 +00:00
2022-10-12 13:04:45 +00:00
bool populate_fts = false ;
if ( ! _tf_ssb_db_has_rows ( db , " PRAGMA table_list('messages_fts') " ) )
2022-09-10 17:56:54 +00:00
{
_tf_ssb_db_exec ( db , " CREATE VIRTUAL TABLE IF NOT EXISTS messages_fts USING fts5(content, content=messages, content_rowid=rowid) " ) ;
2022-10-12 13:04:45 +00:00
populate_fts = true ;
}
2022-11-02 23:34:44 +00:00
if ( ! populate_fts & & /* HACK */ false )
2022-10-12 13:04:45 +00:00
{
2023-03-07 17:50:17 +00:00
tf_printf ( " Checking FTS5 integrity... \n " ) ;
2022-10-12 13:04:45 +00:00
if ( sqlite3_exec ( db , " INSERT INTO messages_fts(messages_fts, rank) VALUES ('integrity-check', 0) " , NULL , NULL , NULL ) = = SQLITE_CORRUPT_VTAB )
{
populate_fts = true ;
}
2023-03-07 17:50:17 +00:00
tf_printf ( " Done. \n " ) ;
2022-10-12 13:04:45 +00:00
}
if ( populate_fts )
{
2023-03-07 17:50:17 +00:00
tf_printf ( " Populating full-text search... \n " ) ;
2024-03-03 18:55:58 +00:00
_tf_ssb_db_exec ( db , " INSERT INTO messages_fts (rowid, content) SELECT rowid, json(content) FROM messages " ) ;
2023-03-07 17:50:17 +00:00
tf_printf ( " Done. \n " ) ;
2022-09-10 17:56:54 +00:00
}
2024-03-05 20:49:30 -05:00
_tf_ssb_db_exec (
db , " CREATE TRIGGER IF NOT EXISTS messages_ai AFTER INSERT ON messages BEGIN INSERT INTO messages_fts(rowid, content) VALUES (new.rowid, json(new.content)); END " ) ;
2024-02-15 23:35:01 +00:00
_tf_ssb_db_exec ( db ,
2024-02-17 19:22:02 +00:00
" CREATE TRIGGER IF NOT EXISTS messages_ad AFTER DELETE ON messages BEGIN INSERT INTO messages_fts(messages_fts, rowid, content) VALUES ('delete', old.rowid, "
" old.content); END " ) ;
2022-09-10 17:56:54 +00:00
2022-10-09 12:53:59 +00:00
if ( ! _tf_ssb_db_has_rows ( db , " PRAGMA table_list('messages_refs') " ) )
{
_tf_ssb_db_exec ( db ,
2024-02-17 19:22:02 +00:00
" CREATE TABLE IF NOT EXISTS messages_refs ( "
" message TEXT, "
" ref TEXT, "
" UNIQUE(message, ref) "
" ) " ) ;
2023-03-07 17:50:17 +00:00
tf_printf ( " Populating messages_refs... \n " ) ;
2024-02-15 23:35:01 +00:00
_tf_ssb_db_exec ( db ,
2024-02-17 19:22:02 +00:00
" INSERT INTO messages_refs(message, ref) "
" SELECT messages.id, j.value FROM messages, json_tree(messages.content) as j WHERE "
" j.value LIKE '&%.sha256' OR "
" j.value LIKE '%%%.sha256' OR "
" j.value LIKE '@%.ed25519' "
" ON CONFLICT DO NOTHING " ) ;
2023-03-07 17:50:17 +00:00
tf_printf ( " Done. \n " ) ;
2022-10-09 12:53:59 +00:00
}
2022-10-15 19:28:57 +00:00
_tf_ssb_db_exec ( db , " DROP TRIGGER IF EXISTS messages_ai_refs " ) ;
2024-02-15 23:35:01 +00:00
_tf_ssb_db_exec ( db ,
2024-02-17 19:22:02 +00:00
" CREATE TRIGGER IF NOT EXISTS messages_ai_refs AFTER INSERT ON messages BEGIN "
" INSERT INTO messages_refs(message, ref) "
" SELECT DISTINCT new.id, j.value FROM json_tree(new.content) as j WHERE "
" j.value LIKE '&%.sha256' OR "
" j.value LIKE '%%%.sha256' OR "
" j.value LIKE '@%.ed25519' "
" ON CONFLICT DO NOTHING; END " ) ;
2022-10-14 16:42:31 +00:00
_tf_ssb_db_exec ( db , " DROP TRIGGER IF EXISTS messages_ad_refs " ) ;
_tf_ssb_db_exec ( db , " CREATE TRIGGER IF NOT EXISTS messages_ad_refs AFTER DELETE ON messages BEGIN DELETE FROM messages_refs WHERE messages_refs.message = old.id; END " ) ;
2022-10-21 23:30:22 +00:00
_tf_ssb_db_exec ( db , " CREATE INDEX IF NOT EXISTS messages_refs_message_idx ON messages_refs (message) " ) ;
_tf_ssb_db_exec ( db , " CREATE INDEX IF NOT EXISTS messages_refs_ref_idx ON messages_refs (ref) " ) ;
2023-07-07 12:08:14 +00:00
_tf_ssb_db_exec ( db , " DROP VIEW IF EXISTS blob_wants_view " ) ;
2023-01-14 23:25:56 +00:00
_tf_ssb_db_exec ( db ,
2024-02-17 19:22:02 +00:00
" CREATE VIEW IF NOT EXISTS blob_wants_view (id, timestamp) AS "
" SELECT messages_refs.ref AS id, messages.timestamp AS timestamp "
" FROM messages_refs "
" JOIN messages ON messages.id = messages_refs.message "
" LEFT OUTER JOIN blobs ON messages_refs.ref = blobs.id "
" WHERE blobs.id IS NULL "
" AND LENGTH(messages_refs.ref) = 52 "
" AND messages_refs.ref LIKE '&%.sha256' " ) ;
2022-10-21 23:30:22 +00:00
2024-03-13 19:40:09 -04:00
bool need_add_flags = true ;
2022-02-12 01:44:11 +00:00
bool need_convert_timestamp_to_real = false ;
if ( sqlite3_prepare ( db , " PRAGMA table_info(messages) " , - 1 , & statement , NULL ) = = SQLITE_OK )
{
2022-02-12 02:51:43 +00:00
int result = SQLITE_OK ;
while ( ( result = sqlite3_step ( statement ) ) = = SQLITE_ROW )
2022-02-12 01:44:11 +00:00
{
const char * name = ( const char * ) sqlite3_column_text ( statement , 1 ) ;
2022-02-12 02:51:43 +00:00
const char * type = ( const char * ) sqlite3_column_text ( statement , 2 ) ;
2022-02-12 01:44:11 +00:00
if ( name & & type & & strcmp ( name , " timestamp " ) = = 0 & & strcmp ( type , " INTEGER " ) = = 0 )
{
need_convert_timestamp_to_real = true ;
}
2024-03-13 19:40:09 -04:00
if ( name & & strcmp ( name , " flags " ) = = 0 )
2022-02-12 01:44:11 +00:00
{
2024-03-13 19:40:09 -04:00
need_add_flags = false ;
2022-02-12 01:44:11 +00:00
}
}
sqlite3_finalize ( statement ) ;
}
if ( need_convert_timestamp_to_real )
{
2023-03-07 17:50:17 +00:00
tf_printf ( " Converting timestamp column from INTEGER to REAL. \n " ) ;
2022-02-12 02:51:43 +00:00
_tf_ssb_db_exec ( db , " BEGIN TRANSACTION " ) ;
_tf_ssb_db_exec ( db , " DROP INDEX IF EXISTS messages_author_timestamp_index " ) ;
2022-02-12 01:44:11 +00:00
_tf_ssb_db_exec ( db , " ALTER TABLE messages ADD COLUMN timestamp_real REAL " ) ;
_tf_ssb_db_exec ( db , " UPDATE messages SET timestamp_real = timestamp " ) ;
2022-02-12 02:51:43 +00:00
_tf_ssb_db_exec ( db , " ALTER TABLE messages DROP COLUMN timestamp " ) ;
2022-02-12 01:44:11 +00:00
_tf_ssb_db_exec ( db , " ALTER TABLE messages RENAME COLUMN timestamp_real TO timestamp " ) ;
2022-02-12 02:51:43 +00:00
_tf_ssb_db_exec ( db , " CREATE INDEX IF NOT EXISTS messages_author_timestamp_index ON messages (author, timestamp) " ) ;
_tf_ssb_db_exec ( db , " COMMIT TRANSACTION " ) ;
2022-02-12 01:44:11 +00:00
}
2024-03-13 19:40:09 -04:00
if ( need_add_flags )
2022-02-12 01:44:11 +00:00
{
2024-03-13 19:40:09 -04:00
tf_printf ( " Adding flags column. \n " ) ;
_tf_ssb_db_exec ( db , " ALTER TABLE messages ADD COLUMN flags INTEGER " ) ;
2022-02-12 01:44:11 +00:00
}
2023-06-15 00:27:49 +00:00
tf_ssb_release_db_writer ( ssb , db ) ;
2021-08-22 19:34:28 +00:00
}
2022-02-06 03:51:25 +00:00
static bool _tf_ssb_db_previous_message_exists ( sqlite3 * db , const char * author , int64_t sequence , const char * previous )
{
bool exists = false ;
if ( sequence = = 1 )
{
exists = true ;
}
else
{
sqlite3_stmt * statement ;
if ( sqlite3_prepare ( db , " SELECT COUNT(*) FROM messages WHERE author = ?1 AND sequence = ?2 AND id = ?3 " , - 1 , & statement , NULL ) = = SQLITE_OK )
{
2024-02-15 23:35:01 +00:00
if ( sqlite3_bind_text ( statement , 1 , author , - 1 , NULL ) = = SQLITE_OK & & sqlite3_bind_int64 ( statement , 2 , sequence - 1 ) = = SQLITE_OK & &
2024-02-17 19:22:02 +00:00
sqlite3_bind_text ( statement , 3 , previous , - 1 , NULL ) = = SQLITE_OK & & sqlite3_step ( statement ) = = SQLITE_ROW )
2022-02-06 03:51:25 +00:00
{
exists = sqlite3_column_int ( statement , 0 ) ! = 0 ;
}
sqlite3_finalize ( statement ) ;
}
}
return exists ;
}
2024-02-15 23:35:01 +00:00
static int64_t _tf_ssb_db_store_message_raw ( tf_ssb_t * ssb , const char * id , const char * previous , const char * author , int64_t sequence , double timestamp , const char * content ,
2024-03-13 19:40:09 -04:00
size_t content_len , const char * signature , int flags )
2021-08-22 19:34:28 +00:00
{
2023-06-15 00:27:49 +00:00
sqlite3 * db = tf_ssb_acquire_db_writer ( ssb ) ;
2021-09-06 17:50:38 +00:00
int64_t last_row_id = - 1 ;
2022-02-06 03:51:25 +00:00
if ( _tf_ssb_db_previous_message_exists ( db , author , sequence , previous ) )
2021-10-10 21:51:38 +00:00
{
2024-03-13 19:40:09 -04:00
const char * query = " INSERT INTO messages (id, previous, author, sequence, timestamp, content, hash, signature, flags) VALUES (?, ?, ?, ?, ?, jsonb(?), "
2024-02-17 19:22:02 +00:00
" ?, ?, ?) ON CONFLICT DO NOTHING " ;
2023-07-20 01:02:50 +00:00
sqlite3_stmt * statement ;
2022-02-06 03:51:25 +00:00
if ( sqlite3_prepare ( db , query , - 1 , & statement , NULL ) = = SQLITE_OK )
2021-10-10 21:51:38 +00:00
{
2022-02-06 03:51:25 +00:00
if ( sqlite3_bind_text ( statement , 1 , id , - 1 , NULL ) = = SQLITE_OK & &
2024-02-17 19:22:02 +00:00
( previous ? sqlite3_bind_text ( statement , 2 , previous , - 1 , NULL ) : sqlite3_bind_null ( statement , 2 ) ) = = SQLITE_OK & &
sqlite3_bind_text ( statement , 3 , author , - 1 , NULL ) = = SQLITE_OK & & sqlite3_bind_int64 ( statement , 4 , sequence ) = = SQLITE_OK & &
sqlite3_bind_double ( statement , 5 , timestamp ) = = SQLITE_OK & & sqlite3_bind_text ( statement , 6 , content , content_len , NULL ) = = SQLITE_OK & &
sqlite3_bind_text ( statement , 7 , " sha256 " , 6 , NULL ) = = SQLITE_OK & & sqlite3_bind_text ( statement , 8 , signature , - 1 , NULL ) = = SQLITE_OK & &
2024-03-13 19:40:09 -04:00
sqlite3_bind_int ( statement , 9 , flags ) = = SQLITE_OK )
2021-09-06 17:50:38 +00:00
{
2022-02-06 03:51:25 +00:00
int r = sqlite3_step ( statement ) ;
if ( r ! = SQLITE_DONE )
{
2023-07-20 01:02:50 +00:00
tf_printf ( " _tf_ssb_db_store_message_raw: %s \n " , sqlite3_errmsg ( db ) ) ;
2022-02-06 03:51:25 +00:00
}
2023-07-20 01:02:50 +00:00
if ( r = = SQLITE_DONE & & sqlite3_changes ( db ) ! = 0 )
2022-02-06 03:51:25 +00:00
{
last_row_id = sqlite3_last_insert_rowid ( db ) ;
}
2021-09-06 17:50:38 +00:00
}
2022-05-21 01:38:13 +00:00
else
{
2023-03-07 17:50:17 +00:00
tf_printf ( " bind failed \n " ) ;
2022-05-21 01:38:13 +00:00
}
2022-02-06 03:51:25 +00:00
sqlite3_finalize ( statement ) ;
}
else
{
2023-06-15 00:27:49 +00:00
tf_printf ( " %s: prepare failed: %s \n " , __FUNCTION__ , sqlite3_errmsg ( db ) ) ;
2021-08-22 19:34:28 +00:00
}
2023-07-20 01:02:50 +00:00
}
else
{
tf_printf ( " %p: Previous message doesn't exist for author=%s sequence=% " PRId64 " . \n " , db , author , sequence ) ;
}
tf_ssb_release_db_writer ( ssb , db ) ;
return last_row_id ;
}
static char * _tf_ssb_db_get_message_blob_wants ( tf_ssb_t * ssb , int64_t rowid )
{
sqlite3 * db = tf_ssb_acquire_db_reader ( ssb ) ;
sqlite3_stmt * statement ;
char * result = NULL ;
size_t size = 0 ;
2023-06-14 21:59:04 +00:00
2024-02-15 23:35:01 +00:00
if ( sqlite3_prepare ( db ,
2024-02-17 19:22:02 +00:00
" SELECT DISTINCT json.value FROM messages, json_tree(messages.content) AS json LEFT OUTER JOIN blobs ON json.value = blobs.id WHERE messages.rowid = ?1 AND "
" json.value LIKE '&%%.sha256' AND length(json.value) = ?2 AND blobs.content IS NULL " ,
- 1 , & statement , NULL ) = = SQLITE_OK )
2023-07-20 01:02:50 +00:00
{
2024-02-15 23:35:01 +00:00
if ( sqlite3_bind_int64 ( statement , 1 , rowid ) = = SQLITE_OK & & sqlite3_bind_int ( statement , 2 , k_blob_id_len - 1 ) = = SQLITE_OK )
2023-07-20 01:02:50 +00:00
{
int r = SQLITE_OK ;
while ( ( r = sqlite3_step ( statement ) ) = = SQLITE_ROW )
{
int id_size = sqlite3_column_bytes ( statement , 0 ) ;
const uint8_t * id = sqlite3_column_text ( statement , 0 ) ;
2024-01-27 21:29:06 +00:00
result = tf_resize_vec ( result , size + id_size + 1 ) ;
2023-07-20 01:02:50 +00:00
memcpy ( result + size , id , id_size + 1 ) ;
size + = id_size + 1 ;
}
if ( r ! = SQLITE_DONE )
{
tf_printf ( " %s \n " , sqlite3_errmsg ( db ) ) ;
}
}
2023-07-20 05:06:15 +00:00
else
{
tf_printf ( " bind failed: %s \n " , sqlite3_errmsg ( db ) ) ;
}
2023-07-20 01:02:50 +00:00
sqlite3_finalize ( statement ) ;
2021-10-10 21:51:38 +00:00
}
else
{
2023-07-20 01:02:50 +00:00
tf_printf ( " %s: prepare failed: %s \n " , __FUNCTION__ , sqlite3_errmsg ( db ) ) ;
2021-08-22 19:34:28 +00:00
}
2023-07-20 01:02:50 +00:00
result = tf_realloc ( result , size + 1 ) ;
result [ size ] = ' \0 ' ;
tf_ssb_release_db_reader ( ssb , db ) ;
return result ;
}
typedef struct _message_store_t
{
uv_work_t work ;
tf_ssb_t * ssb ;
char id [ k_id_base64_len ] ;
char signature [ 512 ] ;
2024-03-13 19:40:09 -04:00
int flags ;
2023-07-20 01:02:50 +00:00
char previous [ k_id_base64_len ] ;
char author [ k_id_base64_len ] ;
int64_t sequence ;
double timestamp ;
const char * content ;
size_t length ;
bool out_stored ;
char * out_blob_wants ;
tf_ssb_db_store_message_callback_t * callback ;
void * user_data ;
2021-08-22 19:34:28 +00:00
2023-07-20 01:02:50 +00:00
message_store_t * next ;
} message_store_t ;
static void _tf_ssb_db_store_message_work ( uv_work_t * work )
{
message_store_t * store = work - > data ;
2023-08-17 00:01:59 +00:00
tf_ssb_record_thread_busy ( store - > ssb , true ) ;
2023-07-20 02:20:38 +00:00
tf_trace_t * trace = tf_ssb_get_trace ( store - > ssb ) ;
tf_trace_begin ( trace , " message_store_work " ) ;
2024-02-15 23:35:01 +00:00
int64_t last_row_id = _tf_ssb_db_store_message_raw ( store - > ssb , store - > id , * store - > previous ? store - > previous : NULL , store - > author , store - > sequence , store - > timestamp ,
2024-03-13 19:40:09 -04:00
store - > content , store - > length , store - > signature , store - > flags ) ;
2023-06-14 21:59:04 +00:00
if ( last_row_id ! = - 1 )
{
2023-07-20 01:02:50 +00:00
store - > out_stored = true ;
store - > out_blob_wants = _tf_ssb_db_get_message_blob_wants ( store - > ssb , last_row_id ) ;
}
2023-07-20 02:20:38 +00:00
tf_trace_end ( trace ) ;
2023-08-17 00:01:59 +00:00
tf_ssb_record_thread_busy ( store - > ssb , false ) ;
2023-07-20 01:02:50 +00:00
}
static void _wake_up_queue ( tf_ssb_t * ssb , tf_ssb_store_queue_t * queue )
{
if ( ! queue - > running )
{
message_store_t * next = queue - > head ;
if ( next )
2023-06-14 21:59:04 +00:00
{
2023-07-20 01:02:50 +00:00
queue - > head = next - > next ;
if ( queue - > tail = = next )
2023-06-14 21:59:04 +00:00
{
2023-07-20 01:02:50 +00:00
queue - > tail = NULL ;
}
next - > next = NULL ;
queue - > running = true ;
int r = uv_queue_work ( tf_ssb_get_loop ( ssb ) , & next - > work , _tf_ssb_db_store_message_work , _tf_ssb_db_store_message_after_work ) ;
if ( r )
{
_tf_ssb_db_store_message_work_finish ( next ) ;
2023-06-14 21:59:04 +00:00
}
}
2023-07-20 01:02:50 +00:00
}
}
static void _tf_ssb_db_store_message_work_finish ( message_store_t * store )
{
JSContext * context = tf_ssb_get_context ( store - > ssb ) ;
if ( store - > callback )
{
store - > callback ( store - > id , store - > out_stored , store - > user_data ) ;
}
JS_FreeCString ( context , store - > content ) ;
tf_ssb_store_queue_t * queue = tf_ssb_get_store_queue ( store - > ssb ) ;
queue - > running = false ;
_wake_up_queue ( store - > ssb , queue ) ;
tf_free ( store ) ;
}
static void _tf_ssb_db_store_message_after_work ( uv_work_t * work , int status )
{
message_store_t * store = work - > data ;
2023-07-20 02:20:38 +00:00
tf_trace_t * trace = tf_ssb_get_trace ( store - > ssb ) ;
tf_trace_begin ( trace , " message_store_after_work " ) ;
2023-07-20 01:02:50 +00:00
if ( store - > out_stored )
{
2023-07-20 05:06:15 +00:00
tf_trace_begin ( trace , " notify_message_added " ) ;
2023-10-08 13:52:49 +00:00
JSContext * context = tf_ssb_get_context ( store - > ssb ) ;
2024-03-13 19:40:09 -04:00
JSValue formatted =
tf_ssb_format_message ( context , store - > previous , store - > author , store - > sequence , store - > timestamp , " sha256 " , store - > content , store - > signature , store - > flags ) ;
2023-10-08 13:52:49 +00:00
JSValue message = JS_NewObject ( context ) ;
JS_SetPropertyStr ( context , message , " key " , JS_NewString ( context , store - > id ) ) ;
JS_SetPropertyStr ( context , message , " value " , formatted ) ;
char timestamp_string [ 256 ] ;
snprintf ( timestamp_string , sizeof ( timestamp_string ) , " %f " , store - > timestamp ) ;
JS_SetPropertyStr ( context , message , " timestamp " , JS_NewString ( context , timestamp_string ) ) ;
tf_ssb_notify_message_added ( store - > ssb , store - > id , message ) ;
JS_FreeValue ( context , message ) ;
2023-07-20 05:06:15 +00:00
tf_trace_end ( trace ) ;
2023-07-20 01:02:50 +00:00
}
if ( store - > out_blob_wants )
{
2023-07-20 05:06:15 +00:00
tf_trace_begin ( trace , " notify_blob_wants_added " ) ;
2023-07-20 01:02:50 +00:00
for ( char * p = store - > out_blob_wants ; * p ; p = p + strlen ( p ) )
2023-06-14 21:59:04 +00:00
{
2023-07-20 01:02:50 +00:00
tf_ssb_notify_blob_want_added ( store - > ssb , p ) ;
2023-06-14 21:59:04 +00:00
}
2023-07-20 01:02:50 +00:00
tf_free ( store - > out_blob_wants ) ;
2023-07-20 05:06:15 +00:00
tf_trace_end ( trace ) ;
2023-06-14 21:59:04 +00:00
}
2023-07-20 01:02:50 +00:00
_tf_ssb_db_store_message_work_finish ( store ) ;
2023-07-20 02:20:38 +00:00
tf_trace_end ( trace ) ;
2023-07-20 01:02:50 +00:00
}
2021-09-06 17:50:38 +00:00
2024-03-13 19:40:09 -04:00
void tf_ssb_db_store_message (
tf_ssb_t * ssb , JSContext * context , const char * id , JSValue val , const char * signature , int flags , tf_ssb_db_store_message_callback_t * callback , void * user_data )
2023-07-20 01:02:50 +00:00
{
JSValue previousval = JS_GetPropertyStr ( context , val , " previous " ) ;
const char * previous = JS_IsNull ( previousval ) ? NULL : JS_ToCString ( context , previousval ) ;
JS_FreeValue ( context , previousval ) ;
JSValue authorval = JS_GetPropertyStr ( context , val , " author " ) ;
const char * author = JS_ToCString ( context , authorval ) ;
JS_FreeValue ( context , authorval ) ;
int64_t sequence = - 1 ;
JSValue sequenceval = JS_GetPropertyStr ( context , val , " sequence " ) ;
JS_ToInt64 ( context , & sequence , sequenceval ) ;
JS_FreeValue ( context , sequenceval ) ;
double timestamp = - 1.0 ;
JSValue timestampval = JS_GetPropertyStr ( context , val , " timestamp " ) ;
JS_ToFloat64 ( context , & timestamp , timestampval ) ;
JS_FreeValue ( context , timestampval ) ;
JSValue contentval = JS_GetPropertyStr ( context , val , " content " ) ;
JSValue content = JS_JSONStringify ( context , contentval , JS_NULL , JS_NULL ) ;
size_t content_len ;
const char * contentstr = JS_ToCStringLen ( context , & content_len , content ) ;
JS_FreeValue ( context , content ) ;
JS_FreeValue ( context , contentval ) ;
message_store_t * store = tf_malloc ( sizeof ( message_store_t ) ) ;
* store = ( message_store_t )
{
. work =
{
. data = store ,
} ,
. ssb = ssb ,
. sequence = sequence ,
. timestamp = timestamp ,
. content = contentstr ,
. length = content_len ,
2024-03-13 19:40:09 -04:00
. flags = flags ,
2023-07-20 01:02:50 +00:00
. callback = callback ,
. user_data = user_data ,
} ;
snprintf ( store - > id , sizeof ( store - > id ) , " %s " , id ) ;
snprintf ( store - > previous , sizeof ( store - > previous ) , " %s " , previous ? previous : " " ) ;
snprintf ( store - > author , sizeof ( store - > author ) , " %s " , author ) ;
snprintf ( store - > signature , sizeof ( store - > signature ) , " %s " , signature ) ;
2021-08-22 19:34:28 +00:00
JS_FreeCString ( context , author ) ;
JS_FreeCString ( context , previous ) ;
2023-07-20 01:02:50 +00:00
tf_ssb_store_queue_t * queue = tf_ssb_get_store_queue ( ssb ) ;
if ( queue - > tail )
{
message_store_t * tail = queue - > tail ;
tail - > next = store ;
queue - > tail = store ;
}
else
{
queue - > head = store ;
queue - > tail = store ;
}
_wake_up_queue ( ssb , queue ) ;
2021-08-22 19:34:28 +00:00
}
2021-08-22 19:41:27 +00:00
bool tf_ssb_db_message_content_get ( tf_ssb_t * ssb , const char * id , uint8_t * * out_blob , size_t * out_size )
2021-08-22 19:34:28 +00:00
{
bool result = false ;
sqlite3_stmt * statement ;
2023-06-15 00:27:49 +00:00
sqlite3 * db = tf_ssb_acquire_db_reader ( ssb ) ;
2024-02-28 20:01:52 -05:00
const char * query = " SELECT json(content) FROM messages WHERE id = ? " ;
2023-06-15 00:27:49 +00:00
if ( sqlite3_prepare ( db , query , - 1 , & statement , NULL ) = = SQLITE_OK )
2021-10-10 21:51:38 +00:00
{
2024-02-15 23:35:01 +00:00
if ( sqlite3_bind_text ( statement , 1 , id , - 1 , NULL ) = = SQLITE_OK & & sqlite3_step ( statement ) = = SQLITE_ROW )
2021-10-10 21:51:38 +00:00
{
2021-08-22 19:34:28 +00:00
const uint8_t * blob = sqlite3_column_blob ( statement , 0 ) ;
int size = sqlite3_column_bytes ( statement , 0 ) ;
2021-10-10 21:51:38 +00:00
if ( out_blob )
{
2022-06-04 17:04:51 +00:00
* out_blob = tf_malloc ( size + 1 ) ;
2021-08-22 19:34:28 +00:00
memcpy ( * out_blob , blob , size ) ;
( * out_blob ) [ size ] = ' \0 ' ;
}
2021-10-10 21:51:38 +00:00
if ( out_size )
{
2021-08-22 19:34:28 +00:00
* out_size = size ;
}
result = true ;
}
sqlite3_finalize ( statement ) ;
}
2023-06-15 00:27:49 +00:00
tf_ssb_release_db_reader ( ssb , db ) ;
2021-08-22 19:34:28 +00:00
return result ;
}
2022-11-17 01:49:34 +00:00
bool tf_ssb_db_blob_has ( tf_ssb_t * ssb , const char * id )
{
bool result = false ;
sqlite3_stmt * statement ;
2023-06-15 00:27:49 +00:00
sqlite3 * db = tf_ssb_acquire_db_reader ( ssb ) ;
2023-10-24 01:27:35 +00:00
const char * query = " SELECT COUNT(*) FROM blobs WHERE id = ?1 " ;
2023-06-15 00:27:49 +00:00
if ( sqlite3_prepare ( db , query , - 1 , & statement , NULL ) = = SQLITE_OK )
2022-11-17 01:49:34 +00:00
{
2024-02-15 23:35:01 +00:00
if ( sqlite3_bind_text ( statement , 1 , id , - 1 , NULL ) = = SQLITE_OK & & sqlite3_step ( statement ) = = SQLITE_ROW )
2022-11-17 01:49:34 +00:00
{
result = sqlite3_column_int64 ( statement , 0 ) ! = 0 ;
}
sqlite3_finalize ( statement ) ;
}
2023-06-15 00:27:49 +00:00
tf_ssb_release_db_reader ( ssb , db ) ;
2022-11-17 01:49:34 +00:00
return result ;
}
2021-08-22 19:41:27 +00:00
bool tf_ssb_db_blob_get ( tf_ssb_t * ssb , const char * id , uint8_t * * out_blob , size_t * out_size )
2021-08-22 19:34:28 +00:00
{
bool result = false ;
sqlite3_stmt * statement ;
2023-06-15 00:27:49 +00:00
sqlite3 * db = tf_ssb_acquire_db_reader ( ssb ) ;
2023-10-24 01:27:35 +00:00
const char * query = " SELECT content FROM blobs WHERE id = ?1 " ;
2023-06-15 00:27:49 +00:00
if ( sqlite3_prepare ( db , query , - 1 , & statement , NULL ) = = SQLITE_OK )
2021-10-10 21:51:38 +00:00
{
2024-02-15 23:35:01 +00:00
if ( sqlite3_bind_text ( statement , 1 , id , - 1 , NULL ) = = SQLITE_OK & & sqlite3_step ( statement ) = = SQLITE_ROW )
2021-10-10 21:51:38 +00:00
{
2021-08-22 19:34:28 +00:00
const uint8_t * blob = sqlite3_column_blob ( statement , 0 ) ;
int size = sqlite3_column_bytes ( statement , 0 ) ;
2021-10-10 21:51:38 +00:00
if ( out_blob )
{
2022-06-04 17:04:51 +00:00
* out_blob = tf_malloc ( size + 1 ) ;
2022-01-17 21:46:32 +00:00
if ( size )
{
memcpy ( * out_blob , blob , size ) ;
}
2021-08-22 19:34:28 +00:00
( * out_blob ) [ size ] = ' \0 ' ;
}
2021-10-10 21:51:38 +00:00
if ( out_size )
{
2021-08-22 19:34:28 +00:00
* out_size = size ;
}
result = true ;
}
sqlite3_finalize ( statement ) ;
}
2023-06-15 00:27:49 +00:00
tf_ssb_release_db_reader ( ssb , db ) ;
2021-08-22 19:34:28 +00:00
return result ;
}
2023-07-18 23:46:15 +00:00
typedef struct _blob_store_work_t
{
uv_work_t work ;
tf_ssb_t * ssb ;
const uint8_t * blob ;
size_t size ;
char id [ k_blob_id_len ] ;
bool is_new ;
tf_ssb_db_blob_store_callback_t * callback ;
void * user_data ;
} blob_store_work_t ;
static void _tf_ssb_db_blob_store_work ( uv_work_t * work )
{
blob_store_work_t * blob_work = work - > data ;
2023-08-17 00:01:59 +00:00
tf_ssb_record_thread_busy ( blob_work - > ssb , true ) ;
2023-07-18 23:56:20 +00:00
tf_trace_t * trace = tf_ssb_get_trace ( blob_work - > ssb ) ;
tf_trace_begin ( trace , " blob_store_work " ) ;
2023-07-18 23:46:15 +00:00
tf_ssb_db_blob_store ( blob_work - > ssb , blob_work - > blob , blob_work - > size , blob_work - > id , sizeof ( blob_work - > id ) , & blob_work - > is_new ) ;
2023-07-18 23:56:20 +00:00
tf_trace_end ( trace ) ;
2023-08-17 00:01:59 +00:00
tf_ssb_record_thread_busy ( blob_work - > ssb , false ) ;
2023-07-18 23:46:15 +00:00
}
static void _tf_ssb_db_blob_store_after_work ( uv_work_t * work , int status )
{
blob_store_work_t * blob_work = work - > data ;
2023-07-18 23:56:20 +00:00
tf_trace_t * trace = tf_ssb_get_trace ( blob_work - > ssb ) ;
tf_trace_begin ( trace , " blob_store_after_work " ) ;
2023-07-19 00:58:20 +00:00
if ( status = = 0 & & * blob_work - > id )
{
tf_ssb_notify_blob_stored ( blob_work - > ssb , blob_work - > id ) ;
}
2023-07-18 23:46:15 +00:00
if ( status ! = 0 )
{
tf_printf ( " tf_ssb_db_blob_store_async -> uv_queue_work failed asynchronously: %s \n " , uv_strerror ( status ) ) ;
}
if ( blob_work - > callback )
{
blob_work - > callback ( status = = 0 ? blob_work - > id : NULL , blob_work - > is_new , blob_work - > user_data ) ;
}
2023-07-18 23:56:20 +00:00
tf_trace_end ( trace ) ;
2023-07-18 23:46:15 +00:00
tf_free ( blob_work ) ;
}
void tf_ssb_db_blob_store_async ( tf_ssb_t * ssb , const uint8_t * blob , size_t size , tf_ssb_db_blob_store_callback_t * callback , void * user_data )
{
blob_store_work_t * work = tf_malloc ( sizeof ( blob_store_work_t ) ) ;
* work = ( blob_store_work_t )
{
. work =
{
. data = work ,
} ,
. ssb = ssb ,
. blob = blob ,
. size = size ,
. callback = callback ,
. user_data = user_data ,
} ;
int r = uv_queue_work ( tf_ssb_get_loop ( ssb ) , & work - > work , _tf_ssb_db_blob_store_work , _tf_ssb_db_blob_store_after_work ) ;
if ( r )
{
tf_printf ( " tf_ssb_db_blob_store_async -> uv_queue_work failed immediately: %s \n " , uv_strerror ( r ) ) ;
if ( callback )
{
callback ( NULL , false , user_data ) ;
}
tf_free ( work ) ;
}
}
2022-10-12 12:27:32 +00:00
bool tf_ssb_db_blob_store ( tf_ssb_t * ssb , const uint8_t * blob , size_t size , char * out_id , size_t out_id_size , bool * out_new )
2021-08-22 19:34:28 +00:00
{
bool result = false ;
uint8_t hash [ crypto_hash_sha256_BYTES ] ;
crypto_hash_sha256 ( hash , blob , size ) ;
char hash64 [ 256 ] ;
2023-02-14 03:15:24 +00:00
tf_base64_encode ( hash , sizeof ( hash ) , hash64 , sizeof ( hash64 ) ) ;
2021-08-22 19:34:28 +00:00
char id [ 512 ] ;
snprintf ( id , sizeof ( id ) , " &%s.sha256 " , hash64 ) ;
2022-01-09 01:32:33 +00:00
int rows = 0 ;
2023-07-08 13:43:44 +00:00
sqlite3 * db = tf_ssb_acquire_db_writer ( ssb ) ;
sqlite3_stmt * statement ;
2024-02-17 19:22:02 +00:00
if ( sqlite3_prepare ( db , " INSERT INTO blobs (id, content, created) VALUES (?1, ?2, CAST(strftime('%s') AS INTEGER)) ON CONFLICT DO NOTHING " , - 1 , & statement , NULL ) = = SQLITE_OK )
2021-10-10 21:51:38 +00:00
{
2024-02-15 23:35:01 +00:00
if ( sqlite3_bind_text ( statement , 1 , id , - 1 , NULL ) = = SQLITE_OK & & sqlite3_bind_blob ( statement , 2 , blob , size , NULL ) = = SQLITE_OK )
2021-10-10 21:51:38 +00:00
{
2021-08-22 19:34:28 +00:00
result = sqlite3_step ( statement ) = = SQLITE_DONE ;
2022-01-09 01:32:33 +00:00
rows = sqlite3_changes ( db ) ;
2021-10-10 21:51:38 +00:00
}
else
{
2023-03-07 17:50:17 +00:00
tf_printf ( " bind failed: %s \n " , sqlite3_errmsg ( db ) ) ;
2021-08-22 19:34:28 +00:00
}
sqlite3_finalize ( statement ) ;
2021-10-10 21:51:38 +00:00
}
else
{
2023-06-15 00:27:49 +00:00
tf_printf ( " %s: prepare failed: %s \n " , __FUNCTION__ , sqlite3_errmsg ( db ) ) ;
2021-08-22 19:34:28 +00:00
}
2023-06-15 00:27:49 +00:00
tf_ssb_release_db_writer ( ssb , db ) ;
2021-08-22 19:34:28 +00:00
2023-01-18 22:52:54 +00:00
if ( rows )
2022-01-09 01:32:33 +00:00
{
2023-01-18 22:52:54 +00:00
if ( ! out_new )
{
2023-03-07 17:50:17 +00:00
tf_printf ( " blob stored %s %zd => %d \n " , id , size , result ) ;
2023-01-18 22:52:54 +00:00
}
2022-01-09 01:32:33 +00:00
}
2021-10-31 21:15:18 +00:00
2021-10-10 21:51:38 +00:00
if ( result & & out_id )
{
2021-08-22 19:34:28 +00:00
snprintf ( out_id , out_id_size , " %s " , id ) ;
}
2022-10-12 12:27:32 +00:00
if ( out_new )
{
* out_new = rows ! = 0 ;
}
2021-08-22 19:34:28 +00:00
return result ;
}
2024-02-15 23:35:01 +00:00
bool tf_ssb_db_get_message_by_author_and_sequence (
2024-02-17 19:22:02 +00:00
tf_ssb_t * ssb , const char * author , int64_t sequence , char * out_message_id , size_t out_message_id_size , double * out_timestamp , char * * out_content )
2021-08-22 19:34:28 +00:00
{
bool found = false ;
sqlite3_stmt * statement ;
2024-02-28 20:01:52 -05:00
const char * query = " SELECT id, timestamp, json(content) FROM messages WHERE author = ?1 AND sequence = ?2 " ;
2023-06-15 00:27:49 +00:00
sqlite3 * db = tf_ssb_acquire_db_reader ( ssb ) ;
if ( sqlite3_prepare ( db , query , - 1 , & statement , NULL ) = = SQLITE_OK )
2021-10-10 21:51:38 +00:00
{
2024-02-17 19:22:02 +00:00
if ( sqlite3_bind_text ( statement , 1 , author , - 1 , NULL ) = = SQLITE_OK & & sqlite3_bind_int64 ( statement , 2 , sequence ) = = SQLITE_OK & & sqlite3_step ( statement ) = = SQLITE_ROW )
2021-10-10 21:51:38 +00:00
{
if ( out_message_id )
{
2021-08-22 19:34:28 +00:00
strncpy ( out_message_id , ( const char * ) sqlite3_column_text ( statement , 0 ) , out_message_id_size - 1 ) ;
}
2021-10-10 21:51:38 +00:00
if ( out_timestamp )
{
2022-02-12 01:44:11 +00:00
* out_timestamp = sqlite3_column_double ( statement , 1 ) ;
2021-08-22 19:34:28 +00:00
}
2021-10-10 21:51:38 +00:00
if ( out_content )
{
2022-06-04 17:04:51 +00:00
* out_content = tf_strdup ( ( const char * ) sqlite3_column_text ( statement , 2 ) ) ;
2021-08-22 19:34:28 +00:00
}
found = true ;
}
sqlite3_finalize ( statement ) ;
2021-10-10 21:51:38 +00:00
}
else
{
2023-06-15 00:27:49 +00:00
tf_printf ( " %s: prepare failed: %s \n " , __FUNCTION__ , sqlite3_errmsg ( db ) ) ;
2021-08-22 19:34:28 +00:00
}
2023-06-15 00:27:49 +00:00
tf_ssb_release_db_reader ( ssb , db ) ;
2021-08-22 19:34:28 +00:00
return found ;
}
2021-08-22 19:41:27 +00:00
bool tf_ssb_db_get_latest_message_by_author ( tf_ssb_t * ssb , const char * author , int64_t * out_sequence , char * out_message_id , size_t out_message_id_size )
2021-08-22 19:34:28 +00:00
{
bool found = false ;
sqlite3_stmt * statement ;
2023-06-15 00:27:49 +00:00
sqlite3 * db = tf_ssb_acquire_db_reader ( ssb ) ;
2023-10-24 01:27:35 +00:00
const char * query = " SELECT id, sequence FROM messages WHERE author = ?1 AND sequence = (SELECT MAX(sequence) FROM messages WHERE author = ?1) " ;
2023-06-15 00:27:49 +00:00
if ( sqlite3_prepare ( db , query , - 1 , & statement , NULL ) = = SQLITE_OK )
2021-10-10 21:51:38 +00:00
{
2024-02-15 23:35:01 +00:00
if ( sqlite3_bind_text ( statement , 1 , author , - 1 , NULL ) = = SQLITE_OK & & sqlite3_step ( statement ) = = SQLITE_ROW )
2021-10-10 21:51:38 +00:00
{
if ( out_sequence )
{
2021-08-22 19:34:28 +00:00
* out_sequence = sqlite3_column_int64 ( statement , 1 ) ;
}
2021-10-10 21:51:38 +00:00
if ( out_message_id )
{
2021-08-22 19:34:28 +00:00
strncpy ( out_message_id , ( const char * ) sqlite3_column_text ( statement , 0 ) , out_message_id_size - 1 ) ;
}
found = true ;
}
sqlite3_finalize ( statement ) ;
2021-10-10 21:51:38 +00:00
}
else
{
2023-06-15 00:27:49 +00:00
tf_printf ( " %s: prepare failed: %s \n " , __FUNCTION__ , sqlite3_errmsg ( db ) ) ;
2021-08-22 19:34:28 +00:00
}
2023-06-15 00:27:49 +00:00
tf_ssb_release_db_reader ( ssb , db ) ;
2021-08-22 19:34:28 +00:00
return found ;
}
2022-09-04 01:58:11 +00:00
static JSValue _tf_ssb_sqlite_bind_json ( JSContext * context , sqlite3 * db , sqlite3_stmt * statement , JSValue binds )
2021-10-10 21:51:38 +00:00
{
if ( JS_IsUndefined ( binds ) )
{
2022-09-04 01:58:11 +00:00
return JS_UNDEFINED ;
2021-08-22 19:34:28 +00:00
}
2022-07-09 15:13:35 +00:00
if ( ! JS_IsArray ( context , binds ) )
{
2022-09-04 01:58:11 +00:00
return JS_ThrowTypeError ( context , " Expected bind parameters to be an array. " ) ;
2022-07-09 15:13:35 +00:00
}
2022-09-04 01:58:11 +00:00
JSValue result = JS_UNDEFINED ;
2022-07-09 15:13:35 +00:00
int32_t length = tf_util_get_length ( context , binds ) ;
2022-09-04 01:58:11 +00:00
for ( int i = 0 ; i < length & & JS_IsUndefined ( result ) ; i + + )
2021-10-10 21:51:38 +00:00
{
2022-07-09 15:13:35 +00:00
JSValue value = JS_GetPropertyUint32 ( context , binds , i ) ;
2022-09-04 01:58:11 +00:00
if ( JS_IsNumber ( value ) )
2021-10-10 21:51:38 +00:00
{
2022-09-04 01:58:11 +00:00
int64_t number = 0 ;
JS_ToInt64 ( context , & number , value ) ;
if ( sqlite3_bind_int64 ( statement , i + 1 , number ) ! = SQLITE_OK )
2021-10-10 21:51:38 +00:00
{
2022-09-04 01:58:11 +00:00
result = JS_ThrowInternalError ( context , " Failed to bind: %s. " , sqlite3_errmsg ( db ) ) ;
2021-10-10 21:51:38 +00:00
}
2022-07-09 15:13:35 +00:00
}
2022-09-04 01:58:11 +00:00
else if ( JS_IsBool ( value ) )
2022-07-09 15:13:35 +00:00
{
2022-09-04 01:58:11 +00:00
if ( sqlite3_bind_int ( statement , i + 1 , JS_ToBool ( context , value ) ? 1 : 0 ) ! = SQLITE_OK )
2021-10-10 21:51:38 +00:00
{
2022-09-04 01:58:11 +00:00
result = JS_ThrowInternalError ( context , " Failed to bind: %s. " , sqlite3_errmsg ( db ) ) ;
2021-08-22 19:34:28 +00:00
}
}
2022-07-09 15:13:35 +00:00
else if ( JS_IsNull ( value ) )
{
if ( sqlite3_bind_null ( statement , i + 1 ) ! = SQLITE_OK )
{
2022-09-04 01:58:11 +00:00
result = JS_ThrowInternalError ( context , " Failed to bind: %s. " , sqlite3_errmsg ( db ) ) ;
2022-07-09 15:13:35 +00:00
}
}
else
{
2022-09-04 01:58:11 +00:00
size_t str_len = 0 ;
const char * str = JS_ToCStringLen ( context , & str_len , value ) ;
if ( str )
{
if ( sqlite3_bind_text ( statement , i + 1 , str , str_len , SQLITE_TRANSIENT ) ! = SQLITE_OK )
{
result = JS_ThrowInternalError ( context , " Failed to bind: %s. " , sqlite3_errmsg ( db ) ) ;
}
JS_FreeCString ( context , str ) ;
}
else
{
result = JS_ThrowInternalError ( context , " Could not convert bind argument %d to string. " , i ) ;
}
2022-07-09 15:13:35 +00:00
}
JS_FreeValue ( context , value ) ;
2021-10-10 21:51:38 +00:00
}
2022-09-04 01:58:11 +00:00
return result ;
2021-08-22 19:34:28 +00:00
}
2021-10-10 21:51:38 +00:00
static JSValue _tf_ssb_sqlite_row_to_json ( JSContext * context , sqlite3_stmt * row )
{
2021-08-22 19:34:28 +00:00
JSValue result = JS_NewObject ( context ) ;
2021-10-10 21:51:38 +00:00
for ( int i = 0 ; i < sqlite3_column_count ( row ) ; i + + )
{
2021-08-22 19:34:28 +00:00
const char * name = sqlite3_column_name ( row , i ) ;
2021-10-10 21:51:38 +00:00
switch ( sqlite3_column_type ( row , i ) )
{
2021-08-22 19:34:28 +00:00
case SQLITE_INTEGER :
JS_SetPropertyStr ( context , result , name , JS_NewInt64 ( context , sqlite3_column_int64 ( row , i ) ) ) ;
break ;
case SQLITE_FLOAT :
JS_SetPropertyStr ( context , result , name , JS_NewFloat64 ( context , sqlite3_column_double ( row , i ) ) ) ;
break ;
case SQLITE_TEXT :
JS_SetPropertyStr ( context , result , name , JS_NewStringLen ( context , ( const char * ) sqlite3_column_text ( row , i ) , sqlite3_column_bytes ( row , i ) ) ) ;
break ;
case SQLITE_BLOB :
JS_SetPropertyStr ( context , result , name , JS_NewArrayBufferCopy ( context , sqlite3_column_blob ( row , i ) , sqlite3_column_bytes ( row , i ) ) ) ;
break ;
case SQLITE_NULL :
JS_SetPropertyStr ( context , result , name , JS_NULL ) ;
break ;
}
}
return result ;
}
2023-08-16 22:43:08 +00:00
int tf_ssb_sqlite_authorizer ( void * user_data , int action_code , const char * arg0 , const char * arg1 , const char * arg2 , const char * arg3 )
2021-08-22 19:34:28 +00:00
{
2022-10-21 23:30:22 +00:00
int result = SQLITE_DENY ;
2021-10-10 21:51:38 +00:00
switch ( action_code )
{
2021-08-22 19:34:28 +00:00
case SQLITE_SELECT :
case SQLITE_FUNCTION :
2022-10-21 23:30:22 +00:00
result = SQLITE_OK ;
break ;
2021-08-22 19:34:28 +00:00
case SQLITE_READ :
2024-02-15 23:35:01 +00:00
result = ( strcmp ( arg0 , " blob_wants_view " ) = = 0 | | strcmp ( arg0 , " json_each " ) = = 0 | | strcmp ( arg0 , " json_tree " ) = = 0 | | strcmp ( arg0 , " messages " ) = = 0 | |
2024-02-17 19:22:02 +00:00
strcmp ( arg0 , " messages_fts " ) = = 0 | | strcmp ( arg0 , " messages_fts_idx " ) = = 0 | | strcmp ( arg0 , " messages_fts_config " ) = = 0 | | strcmp ( arg0 , " messages_refs " ) = = 0 | |
strcmp ( arg0 , " messages_refs_message_idx " ) = = 0 | | strcmp ( arg0 , " messages_refs_ref_idx " ) = = 0 | | strcmp ( arg0 , " sqlite_master " ) = = 0 | | false )
? SQLITE_OK
: SQLITE_DENY ;
2021-08-22 19:34:28 +00:00
break ;
2022-09-10 17:56:54 +00:00
case SQLITE_PRAGMA :
2022-10-21 23:30:22 +00:00
result = strcmp ( arg0 , " data_version " ) = = 0 ? SQLITE_OK : SQLITE_DENY ;
break ;
2022-09-10 18:09:10 +00:00
case SQLITE_UPDATE :
2022-10-21 23:30:22 +00:00
result = strcmp ( arg0 , " sqlite_master " ) = = 0 ? SQLITE_OK : SQLITE_DENY ;
break ;
2021-08-22 19:34:28 +00:00
}
2022-10-21 23:30:22 +00:00
if ( result ! = SQLITE_OK )
{
2023-03-07 17:50:17 +00:00
tf_printf ( " Denying sqlite access to %d %s %s %s %s \n " , action_code , arg0 , arg1 , arg2 , arg3 ) ;
2022-10-21 23:30:22 +00:00
fflush ( stdout ) ;
}
return result ;
2021-08-22 19:34:28 +00:00
}
2022-04-18 00:24:00 +00:00
JSValue tf_ssb_db_visit_query ( tf_ssb_t * ssb , const char * query , const JSValue binds , void ( * callback ) ( JSValue row , void * user_data ) , void * user_data )
2021-08-22 19:34:28 +00:00
{
2022-04-18 00:24:00 +00:00
JSValue result = JS_UNDEFINED ;
2023-08-25 19:41:54 +00:00
sqlite3 * db = tf_ssb_acquire_db_reader_restricted ( ssb ) ;
2022-04-18 00:24:00 +00:00
JSContext * context = tf_ssb_get_context ( ssb ) ;
2021-08-22 19:34:28 +00:00
sqlite3_stmt * statement ;
2021-10-10 21:51:38 +00:00
if ( sqlite3_prepare ( db , query , - 1 , & statement , NULL ) = = SQLITE_OK )
{
2022-09-04 01:58:11 +00:00
JSValue bind_result = _tf_ssb_sqlite_bind_json ( context , db , statement , binds ) ;
if ( JS_IsUndefined ( bind_result ) )
2021-10-10 21:51:38 +00:00
{
2022-04-18 00:24:00 +00:00
int r = SQLITE_OK ;
while ( ( r = sqlite3_step ( statement ) ) = = SQLITE_ROW )
2021-10-10 21:51:38 +00:00
{
2021-08-22 19:34:28 +00:00
JSValue row = _tf_ssb_sqlite_row_to_json ( context , statement ) ;
tf_trace_t * trace = tf_ssb_get_trace ( ssb ) ;
tf_trace_begin ( trace , " callback " ) ;
callback ( row , user_data ) ;
tf_trace_end ( trace ) ;
JS_FreeValue ( context , row ) ;
}
2022-04-18 00:24:00 +00:00
if ( r ! = SQLITE_DONE )
{
result = JS_ThrowInternalError ( context , " SQL Error %s: running \" %s \" . " , sqlite3_errmsg ( db ) , query ) ;
}
2021-08-22 19:34:28 +00:00
}
2022-09-04 01:58:11 +00:00
else
{
result = bind_result ;
}
2021-08-22 19:34:28 +00:00
sqlite3_finalize ( statement ) ;
2021-10-10 21:51:38 +00:00
}
else
{
2022-04-18 00:24:00 +00:00
result = JS_ThrowInternalError ( context , " SQL Error %s: preparing \" %s \" . " , sqlite3_errmsg ( db ) , query ) ;
2021-08-22 19:34:28 +00:00
}
2023-06-15 00:27:49 +00:00
tf_ssb_release_db_reader ( ssb , db ) ;
2022-04-18 00:24:00 +00:00
return result ;
2021-08-22 19:34:28 +00:00
}
2022-02-10 03:58:33 +00:00
2024-03-13 19:40:09 -04:00
JSValue tf_ssb_format_message (
JSContext * context , const char * previous , const char * author , int64_t sequence , double timestamp , const char * hash , const char * content , const char * signature , int flags )
2022-02-10 03:58:33 +00:00
{
JSValue value = JS_NewObject ( context ) ;
2023-07-20 01:02:50 +00:00
JS_SetPropertyStr ( context , value , " previous " , ( previous & & * previous ) ? JS_NewString ( context , previous ) : JS_NULL ) ;
2024-03-13 19:40:09 -04:00
if ( flags & k_tf_ssb_message_flag_sequence_before_author )
2022-02-10 03:58:33 +00:00
{
JS_SetPropertyStr ( context , value , " sequence " , JS_NewInt64 ( context , sequence ) ) ;
JS_SetPropertyStr ( context , value , " author " , JS_NewString ( context , author ) ) ;
}
else
{
JS_SetPropertyStr ( context , value , " author " , JS_NewString ( context , author ) ) ;
JS_SetPropertyStr ( context , value , " sequence " , JS_NewInt64 ( context , sequence ) ) ;
}
2022-02-12 01:44:11 +00:00
JS_SetPropertyStr ( context , value , " timestamp " , JS_NewFloat64 ( context , timestamp ) ) ;
2022-02-10 03:58:33 +00:00
JS_SetPropertyStr ( context , value , " hash " , JS_NewString ( context , hash ) ) ;
JS_SetPropertyStr ( context , value , " content " , JS_ParseJSON ( context , content , strlen ( content ) , NULL ) ) ;
JS_SetPropertyStr ( context , value , " signature " , JS_NewString ( context , signature ) ) ;
return value ;
}
2022-07-14 01:01:14 +00:00
int tf_ssb_db_identity_get_count_for_user ( tf_ssb_t * ssb , const char * user )
{
int count = 0 ;
2023-06-15 00:27:49 +00:00
sqlite3 * db = tf_ssb_acquire_db_reader ( ssb ) ;
2022-07-14 01:01:14 +00:00
sqlite3_stmt * statement = NULL ;
if ( sqlite3_prepare ( db , " SELECT COUNT(*) FROM identities WHERE user = ? " , - 1 , & statement , NULL ) = = SQLITE_OK )
{
if ( sqlite3_bind_text ( statement , 1 , user , - 1 , NULL ) = = SQLITE_OK )
{
if ( sqlite3_step ( statement ) = = SQLITE_ROW )
{
count = sqlite3_column_int ( statement , 0 ) ;
}
}
sqlite3_finalize ( statement ) ;
}
2023-06-15 00:27:49 +00:00
tf_ssb_release_db_reader ( ssb , db ) ;
2022-07-14 01:01:14 +00:00
return count ;
}
2022-10-14 12:27:34 +00:00
bool tf_ssb_db_identity_create ( tf_ssb_t * ssb , const char * user , uint8_t * out_public_key , uint8_t * out_private_key )
{
int count = tf_ssb_db_identity_get_count_for_user ( ssb , user ) ;
if ( count < 16 )
{
char public [ 512 ] ;
char private [ 512 ] ;
tf_ssb_generate_keys_buffer ( public , sizeof ( public ) , private , sizeof ( private ) ) ;
if ( tf_ssb_db_identity_add ( ssb , user , public , private ) )
{
2024-04-01 12:53:00 -04:00
if ( out_public_key )
{
tf_ssb_id_str_to_bin ( out_public_key , public ) ;
}
if ( out_private_key )
{
tf_ssb_id_str_to_bin ( out_private_key , private ) ;
}
2022-10-14 12:27:34 +00:00
return true ;
}
}
return false ;
}
2022-07-14 01:01:14 +00:00
bool tf_ssb_db_identity_add ( tf_ssb_t * ssb , const char * user , const char * public_key , const char * private_key )
{
bool added = false ;
2023-06-15 00:27:49 +00:00
sqlite3 * db = tf_ssb_acquire_db_writer ( ssb ) ;
2022-07-14 01:01:14 +00:00
sqlite3_stmt * statement = NULL ;
if ( sqlite3_prepare ( db , " INSERT INTO identities (user, public_key, private_key) VALUES (?, ?, ?) ON CONFLICT DO NOTHING " , - 1 , & statement , NULL ) = = SQLITE_OK )
{
2024-02-15 23:35:01 +00:00
if ( sqlite3_bind_text ( statement , 1 , user , - 1 , NULL ) = = SQLITE_OK & & sqlite3_bind_text ( statement , 2 , public_key , - 1 , NULL ) = = SQLITE_OK & &
2024-02-17 19:22:02 +00:00
sqlite3_bind_text ( statement , 3 , private_key , - 1 , NULL ) = = SQLITE_OK )
2022-07-14 01:01:14 +00:00
{
2024-02-15 23:35:01 +00:00
added = sqlite3_step ( statement ) = = SQLITE_DONE & & sqlite3_changes ( db ) ! = 0 ;
2022-07-14 01:01:14 +00:00
if ( ! added )
{
2023-03-07 17:50:17 +00:00
tf_printf ( " Unable to add identity: %s. \n " , sqlite3_errmsg ( db ) ) ;
2022-07-14 01:01:14 +00:00
}
}
sqlite3_finalize ( statement ) ;
}
2023-06-15 00:27:49 +00:00
tf_ssb_release_db_writer ( ssb , db ) ;
2022-07-14 01:01:14 +00:00
return added ;
}
2024-01-06 19:22:49 +00:00
bool tf_ssb_db_identity_delete ( tf_ssb_t * ssb , const char * user , const char * public_key )
{
bool removed = false ;
sqlite3 * db = tf_ssb_acquire_db_writer ( ssb ) ;
sqlite3_stmt * statement = NULL ;
tf_printf ( " deleting [%s] [%s] \n " , user , public_key ) ;
if ( sqlite3_prepare ( db , " DELETE FROM identities WHERE user = ? AND public_key = ? " , - 1 , & statement , NULL ) = = SQLITE_OK )
{
2024-02-15 23:35:01 +00:00
if ( sqlite3_bind_text ( statement , 1 , user , - 1 , NULL ) = = SQLITE_OK & & sqlite3_bind_text ( statement , 2 , public_key , - 1 , NULL ) = = SQLITE_OK )
2024-01-06 19:22:49 +00:00
{
2024-02-15 23:35:01 +00:00
removed = sqlite3_step ( statement ) = = SQLITE_DONE & & sqlite3_changes ( db ) ! = 0 ;
2024-01-06 19:22:49 +00:00
if ( ! removed )
{
tf_printf ( " Unable to delete identity: %s. \n " , sqlite3_errmsg ( db ) ) ;
}
}
sqlite3_finalize ( statement ) ;
}
tf_ssb_release_db_writer ( ssb , db ) ;
return removed ;
}
2022-07-14 01:01:14 +00:00
void tf_ssb_db_identity_visit ( tf_ssb_t * ssb , const char * user , void ( * callback ) ( const char * identity , void * user_data ) , void * user_data )
{
2023-06-15 00:27:49 +00:00
sqlite3 * db = tf_ssb_acquire_db_reader ( ssb ) ;
2022-07-14 01:01:14 +00:00
sqlite3_stmt * statement = NULL ;
if ( sqlite3_prepare ( db , " SELECT public_key FROM identities WHERE user = ? ORDER BY public_key " , - 1 , & statement , NULL ) = = SQLITE_OK )
{
if ( sqlite3_bind_text ( statement , 1 , user , - 1 , NULL ) = = SQLITE_OK )
{
while ( sqlite3_step ( statement ) = = SQLITE_ROW )
{
callback ( ( const char * ) sqlite3_column_text ( statement , 0 ) , user_data ) ;
}
}
sqlite3_finalize ( statement ) ;
}
2023-06-15 00:27:49 +00:00
tf_ssb_release_db_reader ( ssb , db ) ;
2022-07-14 01:01:14 +00:00
}
2022-07-31 19:01:08 +00:00
void tf_ssb_db_identity_visit_all ( tf_ssb_t * ssb , void ( * callback ) ( const char * identity , void * user_data ) , void * user_data )
{
2023-06-15 00:27:49 +00:00
sqlite3 * db = tf_ssb_acquire_db_reader ( ssb ) ;
2022-07-31 19:01:08 +00:00
sqlite3_stmt * statement = NULL ;
if ( sqlite3_prepare ( db , " SELECT public_key FROM identities ORDER BY public_key " , - 1 , & statement , NULL ) = = SQLITE_OK )
{
while ( sqlite3_step ( statement ) = = SQLITE_ROW )
{
callback ( ( const char * ) sqlite3_column_text ( statement , 0 ) , user_data ) ;
}
sqlite3_finalize ( statement ) ;
}
2023-06-15 00:27:49 +00:00
tf_ssb_release_db_reader ( ssb , db ) ;
2022-07-31 19:01:08 +00:00
}
2022-07-14 01:01:14 +00:00
bool tf_ssb_db_identity_get_private_key ( tf_ssb_t * ssb , const char * user , const char * public_key , uint8_t * out_private_key , size_t private_key_size )
{
bool success = false ;
2023-10-20 14:37:24 +00:00
if ( out_private_key )
{
memset ( out_private_key , 0 , private_key_size ) ;
}
2023-06-15 00:27:49 +00:00
sqlite3 * db = tf_ssb_acquire_db_reader ( ssb ) ;
2022-07-14 01:01:14 +00:00
sqlite3_stmt * statement = NULL ;
if ( sqlite3_prepare ( db , " SELECT private_key FROM identities WHERE user = ? AND public_key = ? " , - 1 , & statement , NULL ) = = SQLITE_OK )
{
if ( sqlite3_bind_text ( statement , 1 , user , - 1 , NULL ) = = SQLITE_OK & &
2024-02-17 19:22:02 +00:00
sqlite3_bind_text ( statement , 2 , ( public_key & & * public_key = = ' @ ' ) ? public_key + 1 : public_key , - 1 , NULL ) = = SQLITE_OK )
2022-07-14 01:01:14 +00:00
{
if ( sqlite3_step ( statement ) = = SQLITE_ROW )
{
2023-02-14 03:15:24 +00:00
const char * key = ( const char * ) sqlite3_column_text ( statement , 0 ) ;
2023-10-20 14:37:24 +00:00
if ( out_private_key & & private_key_size )
{
int r = tf_base64_decode ( key , sqlite3_column_bytes ( statement , 0 ) - strlen ( " .ed25519 " ) , out_private_key , private_key_size ) ;
success = r > 0 ;
}
else
{
success = true ;
}
2022-07-14 01:01:14 +00:00
}
}
2024-04-02 20:11:36 -04:00
else
{
tf_printf ( " Bind failed: %s. \n " , sqlite3_errmsg ( db ) ) ;
}
2022-07-14 01:01:14 +00:00
sqlite3_finalize ( statement ) ;
}
2024-04-02 20:11:36 -04:00
else
{
tf_printf ( " Prepare failed: %s. \n " , sqlite3_errmsg ( db ) ) ;
}
2023-06-15 00:27:49 +00:00
tf_ssb_release_db_reader ( ssb , db ) ;
2022-07-14 01:01:14 +00:00
return success ;
}
2022-08-15 02:23:45 +00:00
2022-12-31 21:44:48 +00:00
typedef struct _following_t following_t ;
typedef struct _following_t
{
char id [ k_id_base64_len ] ;
following_t * * following ;
following_t * * blocking ;
int following_count ;
int blocking_count ;
int depth ;
2023-10-20 14:37:24 +00:00
int ref_count ;
2023-11-03 00:45:30 +00:00
int block_ref_count ;
2023-12-07 02:28:49 +00:00
bool populated ;
2022-12-31 21:44:48 +00:00
} following_t ;
static int _following_compare ( const void * a , const void * b )
{
const char * ida = a ;
const following_t * const * followingb = b ;
return strcmp ( ida , ( * followingb ) - > id ) ;
}
2023-12-07 02:28:49 +00:00
static bool _has_following_entry ( const char * id , following_t * * list , int count )
{
2023-12-29 17:45:07 +00:00
return count ? bsearch ( id , list , count , sizeof ( following_t * ) , _following_compare ) ! = 0 : false ;
2023-12-07 02:28:49 +00:00
}
2023-11-03 00:45:30 +00:00
static bool _add_following_entry ( following_t * * * list , int * count , following_t * add )
2022-12-31 21:44:48 +00:00
{
int index = tf_util_insert_index ( add - > id , * list , * count , sizeof ( following_t * ) , _following_compare ) ;
2023-10-20 14:37:24 +00:00
if ( index > = * count | | strcmp ( add - > id , ( * list ) [ index ] - > id ) ! = 0 )
2022-12-31 21:44:48 +00:00
{
* list = tf_resize_vec ( * list , sizeof ( * * list ) * ( * count + 1 ) ) ;
if ( * count - index )
{
memmove ( * list + index + 1 , * list + index , sizeof ( following_t * ) * ( * count - index ) ) ;
}
( * list ) [ index ] = add ;
2023-10-20 14:37:24 +00:00
( * count ) + + ;
2023-11-03 00:45:30 +00:00
return true ;
2023-10-20 14:37:24 +00:00
}
2023-11-03 00:45:30 +00:00
return false ;
2023-10-20 14:37:24 +00:00
}
2023-11-03 00:45:30 +00:00
static bool _remove_following_entry ( following_t * * * list , int * count , following_t * remove )
2023-10-20 14:37:24 +00:00
{
int index = tf_util_insert_index ( remove - > id , * list , * count , sizeof ( following_t * ) , _following_compare ) ;
if ( index < * count & & strcmp ( remove - > id , ( * list ) [ index ] - > id ) = = 0 )
{
if ( * count - index > 1 )
{
2023-11-09 00:08:04 +00:00
memmove ( * list + index , * list + index + 1 , sizeof ( following_t * ) * ( * count - index - 1 ) ) ;
2023-10-20 14:37:24 +00:00
}
* list = tf_resize_vec ( * list , sizeof ( * * list ) * ( * count - 1 ) ) ;
( * count ) - - ;
2023-11-03 00:45:30 +00:00
return true ;
2022-12-31 21:44:48 +00:00
}
2023-11-03 00:45:30 +00:00
return false ;
2022-12-31 21:44:48 +00:00
}
2023-12-07 02:28:49 +00:00
typedef struct _block_node_t block_node_t ;
typedef struct _block_node_t
{
following_t * entry ;
block_node_t * parent ;
} block_node_t ;
static bool _is_blocked_by_active_blocks ( const char * id , const block_node_t * blocks )
2022-12-31 21:44:48 +00:00
{
2023-12-07 02:28:49 +00:00
for ( const block_node_t * b = blocks ; b ; b = b - > parent )
{
if ( _has_following_entry ( id , b - > entry - > blocking , b - > entry - > blocking_count ) )
{
return true ;
}
}
return false ;
}
static following_t * _make_following_node ( const char * id , following_t * * * following , int * following_count , block_node_t * blocks )
{
if ( _is_blocked_by_active_blocks ( id , blocks ) )
{
return NULL ;
}
2022-12-31 21:44:48 +00:00
int index = tf_util_insert_index ( id , * following , * following_count , sizeof ( following_t * ) , _following_compare ) ;
following_t * entry = NULL ;
if ( index < * following_count & & strcmp ( id , ( * following ) [ index ] - > id ) = = 0 )
{
entry = ( * following ) [ index ] ;
}
else
{
* following = tf_resize_vec ( * following , sizeof ( * following ) * ( * following_count + 1 ) ) ;
if ( * following_count - index )
{
memmove ( * following + index + 1 , * following + index , sizeof ( following_t * ) * ( * following_count - index ) ) ;
}
entry = tf_malloc ( sizeof ( following_t ) ) ;
( * following ) [ index ] = entry ;
( * following_count ) + + ;
2023-01-03 00:49:21 +00:00
memset ( entry , 0 , sizeof ( * entry ) ) ;
2023-12-07 02:28:49 +00:00
entry - > depth = INT_MAX ;
2022-12-31 21:44:48 +00:00
snprintf ( entry - > id , sizeof ( entry - > id ) , " %s " , id ) ;
2023-01-04 02:59:35 +00:00
}
2023-12-07 02:28:49 +00:00
return entry ;
}
2022-12-31 21:44:48 +00:00
2023-12-07 02:28:49 +00:00
static void _populate_follows_and_blocks ( tf_ssb_t * ssb , following_t * entry , following_t * * * following , int * following_count , block_node_t * active_blocks )
{
sqlite3 * db = tf_ssb_acquire_db_reader ( ssb ) ;
sqlite3_stmt * statement = NULL ;
if ( sqlite3_prepare ( db ,
2024-02-17 19:22:02 +00:00
" SELECT json_extract(content, '$.contact') AS contact, json_extract(content, '$.following'), json_extract(content, '$.blocking') "
" FROM messages "
" WHERE contact IS NOT NULL AND author = ? AND json_extract(content, '$.type') = 'contact' "
" ORDER BY sequence " ,
- 1 , & statement , NULL ) = = SQLITE_OK )
2023-12-07 02:28:49 +00:00
{
if ( sqlite3_bind_text ( statement , 1 , entry - > id , - 1 , NULL ) = = SQLITE_OK )
2022-12-31 21:44:48 +00:00
{
2023-12-07 02:28:49 +00:00
while ( sqlite3_step ( statement ) = = SQLITE_ROW )
2022-12-31 21:44:48 +00:00
{
2023-12-07 02:28:49 +00:00
const char * contact = ( const char * ) sqlite3_column_text ( statement , 0 ) ;
if ( sqlite3_column_type ( statement , 1 ) ! = SQLITE_NULL )
2022-12-31 21:44:48 +00:00
{
2023-12-07 02:28:49 +00:00
bool is_following = sqlite3_column_int ( statement , 1 ) ! = 0 ;
following_t * next = _make_following_node ( contact , following , following_count , active_blocks ) ;
if ( next )
2023-11-09 00:08:04 +00:00
{
2023-01-04 02:59:35 +00:00
if ( is_following )
2022-12-31 21:44:48 +00:00
{
2023-11-03 00:45:30 +00:00
if ( _add_following_entry ( & entry - > following , & entry - > following_count , next ) )
{
next - > ref_count + + ;
}
2023-10-20 14:37:24 +00:00
}
else
{
2023-11-03 00:45:30 +00:00
if ( _remove_following_entry ( & entry - > following , & entry - > following_count , next ) )
{
next - > ref_count - - ;
}
2022-12-31 21:44:48 +00:00
}
2023-01-04 02:59:35 +00:00
}
2023-12-07 02:28:49 +00:00
}
if ( sqlite3_column_type ( statement , 2 ) ! = SQLITE_NULL )
{
bool is_blocking = sqlite3_column_int ( statement , 2 ) ! = 0 ;
following_t * next = _make_following_node ( contact , following , following_count , active_blocks ) ;
if ( next )
2023-01-04 02:59:35 +00:00
{
if ( is_blocking )
2022-12-31 21:44:48 +00:00
{
2023-11-03 00:45:30 +00:00
if ( _add_following_entry ( & entry - > blocking , & entry - > blocking_count , next ) )
{
next - > block_ref_count + + ;
}
2022-12-31 21:44:48 +00:00
}
2023-10-20 14:37:24 +00:00
else
{
2023-11-03 00:45:30 +00:00
if ( _remove_following_entry ( & entry - > blocking , & entry - > blocking_count , next ) )
{
next - > block_ref_count - - ;
}
2023-10-20 14:37:24 +00:00
}
2022-12-31 21:44:48 +00:00
}
}
}
}
2023-12-07 02:28:49 +00:00
sqlite3_finalize ( statement ) ;
}
tf_ssb_release_db_reader ( ssb , db ) ;
}
static void _get_following ( tf_ssb_t * ssb , following_t * entry , following_t * * * following , int * following_count , int depth , int max_depth , block_node_t * active_blocks )
{
entry - > depth = tf_min ( depth , entry - > depth ) ;
if ( depth < max_depth & & ! entry - > populated & & ! _is_blocked_by_active_blocks ( entry - > id , active_blocks ) )
{
entry - > populated = true ;
_populate_follows_and_blocks ( ssb , entry , following , following_count , active_blocks ) ;
if ( depth < max_depth )
{
block_node_t blocks = { . entry = entry , . parent = active_blocks } ;
for ( int i = 0 ; i < entry - > following_count ; i + + )
{
if ( ! _has_following_entry ( entry - > following [ i ] - > id , entry - > blocking , entry - > blocking_count ) )
{
_get_following ( ssb , entry - > following [ i ] , following , following_count , depth + 1 , max_depth , & blocks ) ;
}
}
}
2022-12-31 21:44:48 +00:00
}
}
2023-11-03 00:45:30 +00:00
tf_ssb_following_t * tf_ssb_db_following_deep ( tf_ssb_t * ssb , const char * * ids , int count , int depth )
{
following_t * * following = NULL ;
int following_count = 0 ;
for ( int i = 0 ; i < count ; i + + )
{
2023-12-07 02:28:49 +00:00
following_t * entry = _make_following_node ( ids [ i ] , & following , & following_count , NULL ) ;
_get_following ( ssb , entry , & following , & following_count , 0 , depth , NULL ) ;
2023-11-03 00:45:30 +00:00
entry - > ref_count + + ;
}
int actual_following_count = 0 ;
for ( int i = 0 ; i < following_count ; i + + )
{
if ( following [ i ] - > ref_count > 0 )
{
actual_following_count + + ;
}
}
tf_ssb_following_t * result = tf_malloc ( sizeof ( tf_ssb_following_t ) * ( actual_following_count + 1 ) ) ;
memset ( result , 0 , sizeof ( tf_ssb_following_t ) * ( actual_following_count + 1 ) ) ;
int write_index = 0 ;
for ( int i = 0 ; i < following_count ; i + + )
{
if ( following [ i ] - > ref_count > 0 )
{
snprintf ( result [ write_index ] . id , sizeof ( result [ write_index ] . id ) , " %s " , following [ i ] - > id ) ;
result [ write_index ] . following_count = following [ i ] - > following_count ;
result [ write_index ] . blocking_count = following [ i ] - > blocking_count ;
result [ write_index ] . followed_by_count = following [ i ] - > ref_count ;
result [ write_index ] . blocked_by_count = following [ i ] - > block_ref_count ;
write_index + + ;
}
}
for ( int i = 0 ; i < following_count ; i + + )
{
tf_free ( following [ i ] - > following ) ;
tf_free ( following [ i ] - > blocking ) ;
tf_free ( following [ i ] ) ;
}
tf_free ( following ) ;
return result ;
}
const char * * tf_ssb_db_following_deep_ids ( tf_ssb_t * ssb , const char * * ids , int count , int depth )
2022-12-31 21:44:48 +00:00
{
following_t * * following = NULL ;
int following_count = 0 ;
for ( int i = 0 ; i < count ; i + + )
{
2023-12-07 02:28:49 +00:00
following_t * entry = _make_following_node ( ids [ i ] , & following , & following_count , NULL ) ;
_get_following ( ssb , entry , & following , & following_count , 0 , depth , NULL ) ;
2023-10-20 14:37:24 +00:00
entry - > ref_count + + ;
}
int actual_following_count = 0 ;
for ( int i = 0 ; i < following_count ; i + + )
{
if ( following [ i ] - > ref_count > 0 )
{
actual_following_count + + ;
}
2022-12-31 21:44:48 +00:00
}
2023-10-20 14:37:24 +00:00
char * * result = tf_malloc ( sizeof ( char * ) * ( actual_following_count + 1 ) + k_id_base64_len * actual_following_count ) ;
char * result_ids = ( char * ) result + sizeof ( char * ) * ( actual_following_count + 1 ) ;
2022-12-31 21:44:48 +00:00
2023-10-20 14:37:24 +00:00
int write_index = 0 ;
2022-12-31 21:44:48 +00:00
for ( int i = 0 ; i < following_count ; i + + )
{
2023-10-20 14:37:24 +00:00
if ( following [ i ] - > ref_count > 0 )
{
result [ write_index ] = result_ids + k_id_base64_len * write_index ;
snprintf ( result [ write_index ] , k_id_base64_len , " %s " , following [ i ] - > id ) ;
write_index + + ;
}
2022-12-31 21:44:48 +00:00
}
2023-10-20 14:37:24 +00:00
result [ actual_following_count ] = NULL ;
2022-12-31 21:44:48 +00:00
for ( int i = 0 ; i < following_count ; i + + )
{
tf_free ( following [ i ] - > following ) ;
tf_free ( following [ i ] - > blocking ) ;
tf_free ( following [ i ] ) ;
}
tf_free ( following ) ;
return ( const char * * ) result ;
}
2023-01-08 20:01:35 +00:00
typedef struct _identities_t
{
const char * * ids ;
int count ;
} identities_t ;
static void _add_identity ( const char * identity , void * user_data )
{
identities_t * identities = user_data ;
char full_id [ k_id_base64_len ] ;
snprintf ( full_id , sizeof ( full_id ) , " @%s " , identity ) ;
identities - > ids = tf_resize_vec ( identities - > ids , sizeof ( const char * ) * ( identities - > count + 1 ) ) ;
identities - > ids [ identities - > count + + ] = tf_strdup ( full_id ) ;
}
const char * * tf_ssb_db_get_all_visible_identities ( tf_ssb_t * ssb , int depth )
{
identities_t identities = { 0 } ;
tf_ssb_db_identity_visit_all ( ssb , _add_identity , & identities ) ;
2023-11-03 00:45:30 +00:00
const char * * following = tf_ssb_db_following_deep_ids ( ssb , identities . ids , identities . count , depth ) ;
2023-01-08 20:01:35 +00:00
for ( int i = 0 ; i < identities . count ; i + + )
{
tf_free ( ( void * ) identities . ids [ i ] ) ;
}
tf_free ( identities . ids ) ;
return following ;
}
2024-02-15 23:35:01 +00:00
JSValue tf_ssb_db_get_message_by_id ( tf_ssb_t * ssb , const char * id , bool is_keys )
2023-01-08 17:45:15 +00:00
{
JSValue result = JS_UNDEFINED ;
JSContext * context = tf_ssb_get_context ( ssb ) ;
2023-06-15 00:27:49 +00:00
sqlite3 * db = tf_ssb_acquire_db_reader ( ssb ) ;
2023-01-08 17:45:15 +00:00
sqlite3_stmt * statement ;
2024-03-13 19:40:09 -04:00
if ( sqlite3_prepare ( db , " SELECT previous, author, id, sequence, timestamp, hash, json(content), signature, flags FROM messages WHERE id = ? " , - 1 , & statement , NULL ) = =
SQLITE_OK )
2023-01-08 17:45:15 +00:00
{
if ( sqlite3_bind_text ( statement , 1 , id , - 1 , NULL ) = = SQLITE_OK )
{
if ( sqlite3_step ( statement ) = = SQLITE_ROW )
{
JSValue message = JS_UNDEFINED ;
2024-02-15 23:35:01 +00:00
JSValue formatted = tf_ssb_format_message ( context , ( const char * ) sqlite3_column_text ( statement , 0 ) , ( const char * ) sqlite3_column_text ( statement , 1 ) ,
2024-02-17 19:22:02 +00:00
sqlite3_column_int64 ( statement , 3 ) , sqlite3_column_double ( statement , 4 ) , ( const char * ) sqlite3_column_text ( statement , 5 ) ,
( const char * ) sqlite3_column_text ( statement , 6 ) , ( const char * ) sqlite3_column_text ( statement , 7 ) , sqlite3_column_int ( statement , 8 ) ) ;
2023-01-08 17:45:15 +00:00
if ( is_keys )
{
message = JS_NewObject ( context ) ;
JS_SetPropertyStr ( context , message , " key " , JS_NewString ( context , ( const char * ) sqlite3_column_text ( statement , 2 ) ) ) ;
JS_SetPropertyStr ( context , message , " value " , formatted ) ;
JS_SetPropertyStr ( context , message , " timestamp " , JS_NewString ( context , ( const char * ) sqlite3_column_text ( statement , 4 ) ) ) ;
}
else
{
message = formatted ;
}
result = message ;
}
}
sqlite3_finalize ( statement ) ;
}
2023-06-15 00:27:49 +00:00
tf_ssb_release_db_reader ( ssb , db ) ;
2023-01-08 17:45:15 +00:00
return result ;
}
2023-01-18 00:37:45 +00:00
tf_ssb_db_stored_connection_t * tf_ssb_db_get_stored_connections ( tf_ssb_t * ssb , int * out_count )
{
2023-06-15 00:27:49 +00:00
sqlite3 * db = tf_ssb_acquire_db_reader ( ssb ) ;
2023-01-18 00:37:45 +00:00
tf_ssb_db_stored_connection_t * result = NULL ;
int count = 0 ;
sqlite3_stmt * statement ;
if ( sqlite3_prepare ( db , " SELECT host, port, key FROM connections ORDER BY host, port, key " , - 1 , & statement , NULL ) = = SQLITE_OK )
{
while ( sqlite3_step ( statement ) = = SQLITE_ROW )
{
result = tf_resize_vec ( result , sizeof ( tf_ssb_db_stored_connection_t ) * ( count + 1 ) ) ;
2024-02-15 23:35:01 +00:00
result [ count ] = ( tf_ssb_db_stored_connection_t ) {
2023-01-18 00:37:45 +00:00
. port = sqlite3_column_int ( statement , 1 ) ,
} ;
snprintf ( result [ count ] . address , sizeof ( result [ count ] . address ) , " %s " , ( const char * ) sqlite3_column_text ( statement , 0 ) ) ;
snprintf ( result [ count ] . pubkey , sizeof ( result [ count ] . pubkey ) , " %s " , ( const char * ) sqlite3_column_text ( statement , 2 ) ) ;
count + + ;
}
sqlite3_finalize ( statement ) ;
}
2023-06-15 00:27:49 +00:00
tf_ssb_release_db_reader ( ssb , db ) ;
2023-01-18 00:37:45 +00:00
* out_count = count ;
return result ;
}
void tf_ssb_db_forget_stored_connection ( tf_ssb_t * ssb , const char * address , int port , const char * pubkey )
{
2023-06-15 00:27:49 +00:00
sqlite3 * db = tf_ssb_acquire_db_writer ( ssb ) ;
2023-01-18 00:37:45 +00:00
sqlite3_stmt * statement ;
if ( sqlite3_prepare ( db , " DELETE FROM connections WHERE host = ? AND port = ? AND key = ? " , - 1 , & statement , NULL ) = = SQLITE_OK )
{
2024-02-15 23:35:01 +00:00
if ( sqlite3_bind_text ( statement , 1 , address , - 1 , NULL ) ! = SQLITE_OK | | sqlite3_bind_int ( statement , 2 , port ) ! = SQLITE_OK | |
2024-02-17 19:22:02 +00:00
sqlite3_bind_text ( statement , 3 , pubkey , - 1 , NULL ) ! = SQLITE_OK | | sqlite3_step ( statement ) ! = SQLITE_DONE )
2023-01-18 00:37:45 +00:00
{
2023-03-07 17:50:17 +00:00
tf_printf ( " Delete stored connection: %s. \n " , sqlite3_errmsg ( db ) ) ;
2023-01-18 00:37:45 +00:00
}
sqlite3_finalize ( statement ) ;
}
2023-06-15 00:27:49 +00:00
tf_ssb_release_db_writer ( ssb , db ) ;
2023-01-18 00:37:45 +00:00
}
2024-04-04 21:00:59 -04:00
bool tf_ssb_db_get_account_password_hash ( tf_ssb_t * ssb , const char * name , char * out_password , size_t password_size )
{
bool result = false ;
sqlite3 * db = tf_ssb_acquire_db_reader ( ssb ) ;
sqlite3_stmt * statement = NULL ;
if ( sqlite3_prepare ( db , " SELECT value ->> '$.password' FROM properties WHERE id = 'auth' AND key = 'user:' || ? " , - 1 , & statement , NULL ) = = SQLITE_OK )
{
if ( sqlite3_bind_text ( statement , 1 , name , - 1 , NULL ) = = SQLITE_OK )
{
if ( sqlite3_step ( statement ) = = SQLITE_ROW )
{
snprintf ( out_password , password_size , " %s " , ( const char * ) sqlite3_column_text ( statement , 0 ) ) ;
result = true ;
}
}
sqlite3_finalize ( statement ) ;
}
tf_ssb_release_db_reader ( ssb , db ) ;
return result ;
}
bool tf_ssb_db_set_account_password ( tf_ssb_t * ssb , const char * name , const char * password )
{
JSContext * context = tf_ssb_get_context ( ssb ) ;
bool result = false ;
static const int k_salt_length = 12 ;
char buffer [ 16 ] ;
size_t bytes = uv_random ( tf_ssb_get_loop ( ssb ) , & ( uv_random_t ) { 0 } , buffer , sizeof ( buffer ) , 0 , NULL ) = = 0 ? sizeof ( buffer ) : 0 ;
char output [ 7 + 22 + 1 ] ;
char * salt = crypt_gensalt_rn ( " $2b$ " , k_salt_length , buffer , bytes , output , sizeof ( output ) ) ;
char hash_output [ 7 + 22 + 31 + 1 ] ;
char * hash = crypt_rn ( password , salt , hash_output , sizeof ( hash_output ) ) ;
JSValue user_entry = JS_NewObject ( context ) ;
JS_SetPropertyStr ( context , user_entry , " password " , JS_NewString ( context , hash ) ) ;
JSValue user_json = JS_JSONStringify ( context , user_entry , JS_NULL , JS_NULL ) ;
size_t user_length = 0 ;
const char * user_string = JS_ToCStringLen ( context , & user_length , user_json ) ;
sqlite3 * db = tf_ssb_acquire_db_writer ( ssb ) ;
sqlite3_stmt * statement = NULL ;
if ( sqlite3_prepare ( db , " INSERT OR REPLACE INTO properties (id, key, value) VALUES ('auth', 'user:' || ?, ?) " , - 1 , & statement , NULL ) = = SQLITE_OK )
{
if ( sqlite3_bind_text ( statement , 1 , name , - 1 , NULL ) = = SQLITE_OK & & sqlite3_bind_text ( statement , 2 , user_string , user_length , NULL ) = = SQLITE_OK )
{
result = sqlite3_step ( statement ) = = SQLITE_DONE ;
}
sqlite3_finalize ( statement ) ;
}
tf_ssb_release_db_writer ( ssb , db ) ;
JS_FreeCString ( context , user_string ) ;
JS_FreeValue ( context , user_json ) ;
JS_FreeValue ( context , user_entry ) ;
return result ;
}
bool tf_ssb_db_register_account ( tf_ssb_t * ssb , const char * name , const char * password )
{
bool result = false ;
JSContext * context = tf_ssb_get_context ( ssb ) ;
JSValue users_array = JS_UNDEFINED ;
sqlite3 * db = tf_ssb_acquire_db_writer ( ssb ) ;
sqlite3_stmt * statement = NULL ;
if ( sqlite3_prepare ( db , " SELECT value FROM properties WHERE id = 'auth' AND key = 'users' " , - 1 , & statement , NULL ) = = SQLITE_OK )
{
if ( sqlite3_step ( statement ) = = SQLITE_ROW )
{
users_array = JS_ParseJSON ( context , ( const char * ) sqlite3_column_text ( statement , 0 ) , sqlite3_column_bytes ( statement , 0 ) , NULL ) ;
}
sqlite3_finalize ( statement ) ;
}
if ( JS_IsUndefined ( users_array ) )
{
users_array = JS_NewArray ( context ) ;
}
int length = tf_util_get_length ( context , users_array ) ;
JS_SetPropertyUint32 ( context , users_array , length , JS_NewString ( context , name ) ) ;
JSValue json = JS_JSONStringify ( context , users_array , JS_NULL , JS_NULL ) ;
JS_FreeValue ( context , users_array ) ;
size_t value_length = 0 ;
const char * value = JS_ToCStringLen ( context , & value_length , json ) ;
if ( sqlite3_prepare ( db , " INSERT OR REPLACE INTO properties (id, key, value) VALUES ('auth', 'users', ?) " , - 1 , & statement , NULL ) = = SQLITE_OK )
{
if ( sqlite3_bind_text ( statement , 1 , value , value_length , NULL ) = = SQLITE_OK )
{
result = sqlite3_step ( statement ) = = SQLITE_DONE ;
}
sqlite3_finalize ( statement ) ;
}
JS_FreeCString ( context , value ) ;
JS_FreeValue ( context , json ) ;
tf_ssb_release_db_writer ( ssb , db ) ;
result = result & & tf_ssb_db_set_account_password ( ssb , name , password ) ;
return result ;
}
const char * tf_ssb_db_get_property ( tf_ssb_t * ssb , const char * id , const char * key )
{
char * result = NULL ;
sqlite3 * db = tf_ssb_acquire_db_reader ( ssb ) ;
sqlite3_stmt * statement = NULL ;
if ( sqlite3_prepare ( db , " SELECT value FROM properties WHERE id = ? AND key = ? " , - 1 , & statement , NULL ) = = SQLITE_OK )
{
if ( sqlite3_bind_text ( statement , 1 , id , - 1 , NULL ) = = SQLITE_OK & & sqlite3_bind_text ( statement , 2 , key , - 1 , NULL ) = = SQLITE_OK )
{
if ( sqlite3_step ( statement ) = = SQLITE_ROW )
{
size_t length = sqlite3_column_bytes ( statement , 0 ) ;
result = tf_malloc ( length + 1 ) ;
memcpy ( result , sqlite3_column_text ( statement , 0 ) , length ) ;
result [ length ] = ' \0 ' ;
}
}
sqlite3_finalize ( statement ) ;
}
tf_ssb_release_db_reader ( ssb , db ) ;
return result ;
}
bool tf_ssb_db_set_property ( tf_ssb_t * ssb , const char * id , const char * key , const char * value )
{
bool result = false ;
sqlite3 * db = tf_ssb_acquire_db_writer ( ssb ) ;
sqlite3_stmt * statement = NULL ;
if ( sqlite3_prepare ( db , " INSERT OR REPLACE INTO properties (id, key, value) VALUES (?, ?, ?) " , - 1 , & statement , NULL ) = = SQLITE_OK )
{
if ( sqlite3_bind_text ( statement , 1 , id , - 1 , NULL ) = = SQLITE_OK & & sqlite3_bind_text ( statement , 2 , key , - 1 , NULL ) = = SQLITE_OK & &
sqlite3_bind_text ( statement , 3 , value , - 1 , NULL ) = = SQLITE_OK )
{
result = sqlite3_step ( statement ) = = SQLITE_DONE ;
}
sqlite3_finalize ( statement ) ;
}
tf_ssb_release_db_writer ( ssb , db ) ;
return result ;
}