2021-10-24 15:46:30 +00:00
# include "ssb.js.h"
2021-01-02 18:10:00 +00:00
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:41:27 +00:00
# include "ssb.db.h"
2021-01-02 18:10:00 +00:00
# include "ssb.h"
2023-07-20 02:20:38 +00:00
# include "trace.h"
2021-11-03 22:15:46 +00:00
# include "util.js.h"
2021-01-02 18:10:00 +00:00
2023-05-21 21:36:51 +00:00
# include "sodium/crypto_box.h"
# include "sodium/crypto_scalarmult.h"
# include "sodium/crypto_scalarmult_curve25519.h"
2024-01-04 01:17:30 +00:00
# include "sodium/crypto_scalarmult_ed25519.h"
2023-05-21 21:36:51 +00:00
# include "sodium/crypto_secretbox.h"
# include "sodium/crypto_sign.h"
# include "sodium/randombytes.h"
# include "string.h"
# include "sqlite3.h"
# include "uv.h"
2021-01-02 18:10:00 +00:00
2023-02-26 19:51:54 +00:00
# include <assert.h>
2023-02-08 01:29:44 +00:00
# include <inttypes.h>
2023-08-25 18:22:09 +00:00
# if !defined(_countof)
# define _countof(a) ((int)(sizeof((a)) / sizeof(*(a))))
# endif
2023-08-25 20:23:40 +00:00
static const int k_sql_async_timeout_ms = 60 * 1000 ;
2021-01-02 18:10:00 +00:00
static JSClassID _tf_ssb_classId ;
2023-10-20 14:37:24 +00:00
static JSValue _tf_ssb_appendMessageWithIdentity ( JSContext * context , JSValueConst this_val , int argc , JSValueConst * argv ) ;
2021-12-22 19:57:34 +00:00
2022-07-14 01:01:14 +00:00
static JSValue _tf_ssb_createIdentity ( JSContext * context , JSValueConst this_val , int argc , JSValueConst * argv )
{
tf_ssb_t * ssb = JS_GetOpaque ( this_val , _tf_ssb_classId ) ;
JSValue result = JS_UNDEFINED ;
if ( ssb )
{
const char * user = JS_ToCString ( context , argv [ 0 ] ) ;
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 ) ) ;
2022-08-03 23:52:13 +00:00
if ( tf_ssb_db_identity_add ( ssb , user , public , private ) )
{
char id [ 513 ] ;
snprintf ( id , sizeof ( id ) , " @%s " , public ) ;
result = JS_NewString ( context , id ) ;
}
else
2022-07-14 01:01:14 +00:00
{
result = JS_ThrowInternalError ( context , " Unable to add identity. " ) ;
}
}
else
{
result = JS_ThrowInternalError ( context , " Too many identities for user. " ) ;
}
JS_FreeCString ( context , user ) ;
}
return result ;
}
2024-01-04 01:17:30 +00:00
static JSValue _tf_ssb_addIdentity ( JSContext * context , JSValueConst this_val , int argc , JSValueConst * argv )
{
tf_ssb_t * ssb = JS_GetOpaque ( this_val , _tf_ssb_classId ) ;
JSValue result = JS_UNDEFINED ;
if ( ssb )
{
const char * user = JS_ToCString ( context , argv [ 0 ] ) ;
JSValue buffer = JS_UNDEFINED ;
size_t length = 0 ;
uint8_t * array = tf_util_try_get_array_buffer ( context , & length , argv [ 1 ] ) ;
if ( ! array )
{
size_t offset ;
size_t element_size ;
buffer = tf_util_try_get_typed_array_buffer ( context , argv [ 1 ] , & offset , & length , & element_size ) ;
if ( ! JS_IsException ( buffer ) )
{
array = tf_util_try_get_array_buffer ( context , & length , buffer ) ;
}
}
if ( array )
{
if ( length = = crypto_sign_SECRETKEYBYTES / 2 )
{
uint8_t public_key [ crypto_sign_PUBLICKEYBYTES ] ;
unsigned char seed [ crypto_sign_SEEDBYTES ] ;
uint8_t secret_key [ crypto_sign_SECRETKEYBYTES ] = { 0 } ;
memcpy ( secret_key , array , sizeof ( secret_key ) / 2 ) ;
2024-02-15 23:35:01 +00:00
if ( crypto_sign_ed25519_sk_to_seed ( seed , secret_key ) = = 0 & & crypto_sign_seed_keypair ( public_key , secret_key , seed ) = = 0 )
2024-01-04 01:17:30 +00:00
{
char public_key_b64 [ 512 ] ;
tf_base64_encode ( public_key , sizeof ( public_key ) , public_key_b64 , sizeof ( public_key_b64 ) ) ;
snprintf ( public_key_b64 + strlen ( public_key_b64 ) , sizeof ( public_key_b64 ) - strlen ( public_key_b64 ) , " .ed25519 " ) ;
uint8_t combined [ crypto_sign_SECRETKEYBYTES ] ;
memcpy ( combined , array , length ) ;
memcpy ( combined + length , public_key , sizeof ( public_key ) ) ;
char combined_b64 [ 512 ] ;
tf_base64_encode ( combined , sizeof ( combined ) , combined_b64 , sizeof ( combined_b64 ) ) ;
snprintf ( combined_b64 + strlen ( combined_b64 ) , sizeof ( combined_b64 ) - strlen ( combined_b64 ) , " .ed25519 " ) ;
if ( tf_ssb_db_identity_add ( ssb , user , public_key_b64 , combined_b64 ) )
{
result = JS_TRUE ;
}
else
{
tf_printf ( " Unable to add the identity. " ) ;
}
}
}
else
{
tf_printf ( " Unexpected private key size: %zd vs. %d \n " , length , crypto_sign_SECRETKEYBYTES ) ;
}
}
else
{
tf_printf ( " didn't find array \n " ) ;
}
JS_FreeValue ( context , buffer ) ;
JS_FreeCString ( context , user ) ;
}
else
{
tf_printf ( " no ssb \n " ) ;
}
return result ;
}
2024-01-06 19:22:49 +00:00
static JSValue _tf_ssb_deleteIdentity ( JSContext * context , JSValueConst this_val , int argc , JSValueConst * argv )
{
tf_ssb_t * ssb = JS_GetOpaque ( this_val , _tf_ssb_classId ) ;
JSValue result = JS_UNDEFINED ;
if ( ssb )
{
const char * user = JS_ToCString ( context , argv [ 0 ] ) ;
const char * id = JS_ToCString ( context , argv [ 1 ] ) ;
if ( id & & user )
{
if ( tf_ssb_db_identity_delete ( ssb , user , * id = = ' @ ' ? id + 1 : id ) )
{
result = JS_TRUE ;
}
}
JS_FreeCString ( context , id ) ;
JS_FreeCString ( context , user ) ;
}
return result ;
}
2023-10-20 14:37:24 +00:00
static JSValue _set_server_following_internal ( tf_ssb_t * ssb , JSValueConst this_val , JSValue id , JSValue following )
{
JSContext * context = tf_ssb_get_context ( ssb ) ;
JSValue message = JS_NewObject ( context ) ;
JSValue server_user = JS_NewString ( context , " :admin " ) ;
char server_id_buffer [ k_id_base64_len ] = { 0 } ;
tf_ssb_whoami ( ssb , server_id_buffer , sizeof ( server_id_buffer ) ) ;
JSValue server_id = JS_NewString ( context , server_id_buffer ) ;
JS_SetPropertyStr ( context , message , " type " , JS_NewString ( context , " contact " ) ) ;
JS_SetPropertyStr ( context , message , " contact " , JS_DupValue ( context , id ) ) ;
JS_SetPropertyStr ( context , message , " following " , JS_DupValue ( context , following ) ) ;
2024-02-15 23:35:01 +00:00
JSValue args [ ] = {
2023-10-20 14:37:24 +00:00
server_user ,
server_id ,
message ,
} ;
JSValue result = _tf_ssb_appendMessageWithIdentity ( context , this_val , _countof ( args ) , args ) ;
JS_FreeValue ( context , server_id ) ;
JS_FreeValue ( context , server_user ) ;
JS_FreeValue ( context , message ) ;
return result ;
}
static JSValue _tf_ssb_set_server_following_me ( JSContext * context , JSValueConst this_val , int argc , JSValueConst * argv )
{
tf_ssb_t * ssb = JS_GetOpaque ( this_val , _tf_ssb_classId ) ;
JSValue result = JS_UNDEFINED ;
if ( ssb )
{
char server_id [ k_id_base64_len ] ;
tf_ssb_whoami ( ssb , server_id , sizeof ( server_id ) ) ;
const char * user = JS_ToCString ( context , argv [ 0 ] ) ;
const char * key = JS_ToCString ( context , argv [ 1 ] ) ;
if ( ! tf_ssb_db_identity_get_private_key ( ssb , user , key , NULL , 0 ) )
{
result = JS_ThrowInternalError ( context , " User %s does not own key %s. " , user , key ) ;
}
else
{
const char * server_id_ptr = server_id ;
2023-11-03 00:45:30 +00:00
const char * * current_following = tf_ssb_db_following_deep_ids ( ssb , & server_id_ptr , 1 , 1 ) ;
2023-10-20 14:37:24 +00:00
bool is_following = false ;
for ( const char * * it = current_following ; * it ; it + + )
{
if ( strcmp ( key , * it ) = = 0 )
{
is_following = true ;
break ;
}
}
tf_free ( current_following ) ;
bool want_following = JS_ToBool ( context , argv [ 2 ] ) ;
if ( ( want_following & & ! is_following ) | | ( ! want_following & & is_following ) )
{
result = _set_server_following_internal ( ssb , this_val , argv [ 1 ] , argv [ 2 ] ) ;
}
}
JS_FreeCString ( context , key ) ;
JS_FreeCString ( context , user ) ;
}
return result ;
}
2022-07-14 01:01:14 +00:00
typedef struct _identities_visit_t
{
JSContext * context ;
JSValue array ;
int count ;
} identities_visit_t ;
static void _tf_ssb_getIdentities_visit ( const char * identity , void * data )
{
identities_visit_t * state = data ;
2022-07-31 19:01:08 +00:00
char id [ k_id_base64_len ] ;
snprintf ( id , sizeof ( id ) , " @%s " , identity ) ;
JS_SetPropertyUint32 ( state - > context , state - > array , state - > count + + , JS_NewString ( state - > context , id ) ) ;
2022-07-14 01:01:14 +00:00
}
static JSValue _tf_ssb_getIdentities ( JSContext * context , JSValueConst this_val , int argc , JSValueConst * argv )
{
JSValue result = JS_NewArray ( context ) ;
tf_ssb_t * ssb = JS_GetOpaque ( this_val , _tf_ssb_classId ) ;
if ( ssb )
{
const char * user = JS_ToCString ( context , argv [ 0 ] ) ;
2024-02-15 23:35:01 +00:00
identities_visit_t state = {
2022-07-14 01:01:14 +00:00
. context = context ,
. array = result ,
} ;
tf_ssb_db_identity_visit ( ssb , user , _tf_ssb_getIdentities_visit , & state ) ;
JS_FreeCString ( context , user ) ;
}
return result ;
}
2024-01-04 00:21:15 +00:00
static JSValue _tf_ssb_getPrivateKey ( JSContext * context , JSValueConst this_val , int argc , JSValueConst * argv )
{
JSValue result = JS_UNDEFINED ;
tf_ssb_t * ssb = JS_GetOpaque ( this_val , _tf_ssb_classId ) ;
const char * user = JS_ToCString ( context , argv [ 0 ] ) ;
const char * id = JS_ToCString ( context , argv [ 1 ] ) ;
uint8_t private_key [ crypto_sign_SECRETKEYBYTES ] ;
if ( tf_ssb_db_identity_get_private_key ( ssb , user , id , private_key , sizeof ( private_key ) ) )
{
result = tf_util_new_uint8_array ( context , private_key , sizeof ( private_key ) / 2 ) ;
}
JS_FreeCString ( context , user ) ;
JS_FreeCString ( context , id ) ;
return result ;
}
2023-10-20 12:55:05 +00:00
static JSValue _tf_ssb_getServerIdentity ( JSContext * context , JSValueConst this_val , int argc , JSValueConst * argv )
{
2024-03-03 12:20:03 -05:00
JSValue result = JS_UNDEFINED ;
2023-10-20 12:55:05 +00:00
tf_ssb_t * ssb = JS_GetOpaque ( this_val , _tf_ssb_classId ) ;
if ( ssb )
{
char id [ k_id_base64_len ] = { 0 } ;
if ( tf_ssb_whoami ( ssb , id , sizeof ( id ) ) )
{
result = JS_NewString ( context , id ) ;
}
}
return result ;
}
2022-07-31 19:01:08 +00:00
static JSValue _tf_ssb_getAllIdentities ( JSContext * context , JSValueConst this_val , int argc , JSValueConst * argv )
{
JSValue result = JS_NewArray ( context ) ;
tf_ssb_t * ssb = JS_GetOpaque ( this_val , _tf_ssb_classId ) ;
if ( ssb )
{
2024-02-15 23:35:01 +00:00
identities_visit_t state = {
2022-07-31 19:01:08 +00:00
. context = context ,
. array = result ,
} ;
tf_ssb_db_identity_visit_all ( ssb , _tf_ssb_getIdentities_visit , & state ) ;
}
return result ;
}
2024-05-05 12:55:32 -04:00
typedef struct _active_identity_work_t
{
uv_work_t request ;
tf_ssb_t * ssb ;
JSContext * context ;
const char * name ;
const char * package_owner ;
const char * package_name ;
char identity [ k_id_base64_len ] ;
int result ;
JSValue promise [ 2 ] ;
} active_identity_work_t ;
static void _tf_ssb_getActiveIdentity_visit ( const char * identity , void * user_data )
{
active_identity_work_t * request = user_data ;
if ( ! * request - > identity )
{
snprintf ( request - > identity , sizeof ( request - > identity ) , " %s " , identity ) ;
}
}
static void _tf_ssb_getActiveIdentity_work ( uv_work_t * work )
{
active_identity_work_t * request = work - > data ;
tf_ssb_record_thread_busy ( request - > ssb , true ) ;
tf_trace_t * trace = tf_ssb_get_trace ( request - > ssb ) ;
tf_trace_begin ( trace , " _tf_ssb_getActiveIdentity_work " ) ;
sqlite3 * db = tf_ssb_acquire_db_reader ( request - > ssb ) ;
2024-05-05 13:48:22 -04:00
tf_ssb_db_identity_get_active ( db , request - > name , request - > package_owner , request - > package_name , request - > identity , sizeof ( request - > identity ) ) ;
2024-05-05 12:55:32 -04:00
tf_ssb_release_db_reader ( request - > ssb , db ) ;
if ( ! * request - > identity )
{
tf_ssb_db_identity_visit ( request - > ssb , request - > name , _tf_ssb_getActiveIdentity_visit , request ) ;
}
tf_trace_end ( trace ) ;
tf_ssb_record_thread_busy ( request - > ssb , false ) ;
}
static void _tf_ssb_getActiveIdentity_after_work ( uv_work_t * work , int status )
{
active_identity_work_t * request = work - > data ;
JSContext * context = request - > context ;
if ( request - > result = = 0 )
{
JSValue identity = JS_NewString ( context , request - > identity ) ;
JSValue error = JS_Call ( context , request - > promise [ 0 ] , JS_UNDEFINED , 1 , & identity ) ;
JS_FreeValue ( context , identity ) ;
tf_util_report_error ( context , error ) ;
JS_FreeValue ( context , error ) ;
}
else
{
JSValue error = JS_Call ( context , request - > promise [ 1 ] , JS_UNDEFINED , 0 , NULL ) ;
tf_util_report_error ( context , error ) ;
JS_FreeValue ( context , error ) ;
}
JS_FreeValue ( context , request - > promise [ 0 ] ) ;
JS_FreeValue ( context , request - > promise [ 1 ] ) ;
tf_free ( ( void * ) request - > name ) ;
tf_free ( ( void * ) request - > package_owner ) ;
tf_free ( ( void * ) request - > package_name ) ;
tf_free ( request ) ;
}
static JSValue _tf_ssb_getActiveIdentity ( JSContext * context , JSValueConst this_val , int argc , JSValueConst * argv )
{
const char * name = JS_ToCString ( context , argv [ 0 ] ) ;
const char * package_owner = JS_ToCString ( context , argv [ 1 ] ) ;
const char * package_name = JS_ToCString ( context , argv [ 2 ] ) ;
active_identity_work_t * work = tf_malloc ( sizeof ( active_identity_work_t ) ) ;
2024-05-08 12:20:57 -04:00
* work = ( active_identity_work_t ) {
2024-05-05 12:55:32 -04:00
. request = { . data = work } ,
. ssb = JS_GetOpaque ( this_val , _tf_ssb_classId ) ,
. context = context ,
. name = tf_strdup ( name ) ,
. package_owner = tf_strdup ( package_owner ) ,
. package_name = tf_strdup ( package_name ) ,
} ;
JSValue result = JS_NewPromiseCapability ( context , work - > promise ) ;
JS_FreeCString ( context , name ) ;
JS_FreeCString ( context , package_owner ) ;
JS_FreeCString ( context , package_name ) ;
int r = uv_queue_work ( tf_ssb_get_loop ( work - > ssb ) , & work - > request , _tf_ssb_getActiveIdentity_work , _tf_ssb_getActiveIdentity_after_work ) ;
if ( r )
{
_tf_ssb_getActiveIdentity_after_work ( & work - > request , r ) ;
}
return result ;
}
2024-05-05 13:48:22 -04:00
typedef struct _identity_info_work_t
{
uv_work_t request ;
tf_ssb_t * ssb ;
JSContext * context ;
const char * name ;
const char * package_owner ;
const char * package_name ;
int count ;
char * * identities ;
char * * names ;
int result ;
char active_identity [ k_id_base64_len ] ;
JSValue promise [ 2 ] ;
} identity_info_work_t ;
static void _tf_ssb_getIdentityInfo_visit ( const char * identity , void * data )
{
identity_info_work_t * request = data ;
request - > identities = tf_resize_vec ( request - > identities , ( request - > count + 1 ) * sizeof ( char * ) ) ;
request - > names = tf_resize_vec ( request - > names , ( request - > count + 1 ) * sizeof ( char * ) ) ;
request - > identities [ request - > count ] = tf_strdup ( identity ) ;
request - > names [ request - > count ] = NULL ;
2024-05-08 12:20:57 -04:00
request - > count + + ;
;
2024-05-05 13:48:22 -04:00
}
static void _tf_ssb_getIdentityInfo_work ( uv_work_t * work )
{
identity_info_work_t * request = work - > data ;
tf_ssb_db_identity_visit ( request - > ssb , request - > name , _tf_ssb_getIdentityInfo_visit , request ) ;
sqlite3 * db = tf_ssb_acquire_db_reader ( request - > ssb ) ;
sqlite3_stmt * statement = NULL ;
request - > result = sqlite3_prepare ( db ,
" SELECT author, name FROM ( "
" SELECT "
" messages.author, "
" RANK() OVER (PARTITION BY messages.author ORDER BY messages.sequence DESC) AS author_rank, "
" messages.content ->> 'name' AS name "
" FROM messages "
" JOIN identities ON messages.author = ids.value "
" WHERE WHERE identities.user = ? AND json_extract(messages.content, '$.type') = 'about' AND content ->> 'about' = messages.author AND name IS NOT NULL) "
2024-05-08 12:20:57 -04:00
" WHERE author_rank = 1 " ,
- 1 , & statement , NULL ) ;
2024-05-05 13:48:22 -04:00
if ( request - > result = = SQLITE_OK )
{
if ( sqlite3_bind_text ( statement , 1 , request - > name , - 1 , NULL ) = = SQLITE_OK )
{
int r = SQLITE_OK ;
while ( ( r = sqlite3_step ( statement ) ) = = SQLITE_OK )
{
for ( int i = 0 ; i < request - > count ; i + + )
{
const char * identity = ( const char * ) sqlite3_column_text ( statement , 0 ) ;
const char * name = ( const char * ) sqlite3_column_text ( statement , 1 ) ;
if ( strcmp ( request - > identities [ i ] , identity ) = = 0 & & ! request - > names [ i ] )
{
request - > names [ i ] = tf_strdup ( name ) ;
}
break ;
}
}
}
sqlite3_finalize ( statement ) ;
}
tf_ssb_db_identity_get_active ( db , request - > name , request - > package_owner , request - > package_name , request - > active_identity , sizeof ( request - > active_identity ) ) ;
if ( ! * request - > active_identity & & request - > count )
{
snprintf ( request - > active_identity , sizeof ( request - > active_identity ) , " %s " , request - > identities [ 0 ] ) ;
}
tf_ssb_release_db_reader ( request - > ssb , db ) ;
}
static void _tf_ssb_getIdentityInfo_after_work ( uv_work_t * work , int status )
{
identity_info_work_t * request = work - > data ;
JSContext * context = request - > context ;
JSValue result = JS_NewObject ( context ) ;
JSValue identities = JS_NewArray ( context ) ;
for ( int i = 0 ; i < request - > count ; i + + )
{
JS_SetPropertyUint32 ( context , identities , i , JS_NewString ( context , request - > identities [ i ] ) ) ;
}
JS_SetPropertyStr ( context , result , " identities " , identities ) ;
JSValue names = JS_NewObject ( context ) ;
for ( int i = 0 ; i < request - > count ; i + + )
{
JS_SetPropertyStr ( context , names , request - > identities [ i ] , JS_NewString ( context , request - > names [ i ] ? request - > names [ i ] : request - > identities [ i ] ) ) ;
}
JS_SetPropertyStr ( context , result , " names " , names ) ;
JS_SetPropertyStr ( context , result , " identity " , JS_NewString ( context , request - > active_identity ) ) ;
JSValue error = JS_Call ( context , request - > promise [ 0 ] , JS_UNDEFINED , 1 , & result ) ;
tf_util_report_error ( context , error ) ;
JS_FreeValue ( context , error ) ;
JS_FreeValue ( context , result ) ;
JS_FreeValue ( context , request - > promise [ 0 ] ) ;
JS_FreeValue ( context , request - > promise [ 1 ] ) ;
for ( int i = 0 ; i < request - > count ; i + + )
{
tf_free ( request - > identities [ i ] ) ;
tf_free ( request - > names [ i ] ) ;
}
tf_free ( request - > identities ) ;
tf_free ( request - > names ) ;
tf_free ( ( void * ) request - > name ) ;
tf_free ( ( void * ) request - > package_owner ) ;
tf_free ( ( void * ) request - > package_name ) ;
tf_free ( request ) ;
}
static JSValue _tf_ssb_getIdentityInfo ( JSContext * context , JSValueConst this_val , int argc , JSValueConst * argv )
{
const char * name = JS_ToCString ( context , argv [ 0 ] ) ;
const char * package_owner = JS_ToCString ( context , argv [ 1 ] ) ;
const char * package_name = JS_ToCString ( context , argv [ 2 ] ) ;
identity_info_work_t * work = tf_malloc ( sizeof ( identity_info_work_t ) ) ;
2024-05-08 12:20:57 -04:00
* work = ( identity_info_work_t ) {
2024-05-05 13:48:22 -04:00
. request = { . data = work } ,
. ssb = JS_GetOpaque ( this_val , _tf_ssb_classId ) ,
. context = context ,
. name = tf_strdup ( name ) ,
. package_owner = tf_strdup ( package_owner ) ,
. package_name = tf_strdup ( package_name ) ,
} ;
JSValue result = JS_NewPromiseCapability ( context , work - > promise ) ;
JS_FreeCString ( context , name ) ;
JS_FreeCString ( context , package_owner ) ;
JS_FreeCString ( context , package_name ) ;
int r = uv_queue_work ( tf_ssb_get_loop ( work - > ssb ) , & work - > request , _tf_ssb_getIdentityInfo_work , _tf_ssb_getIdentityInfo_after_work ) ;
if ( r )
{
_tf_ssb_getIdentityInfo_after_work ( & work - > request , r ) ;
}
return result ;
}
2023-07-20 01:02:50 +00:00
typedef struct _append_message_t
{
JSContext * context ;
JSValue promise [ 2 ] ;
} append_message_t ;
static void _tf_ssb_appendMessage_finish ( append_message_t * async , bool success , JSValue result )
2022-07-14 01:01:14 +00:00
{
2023-07-20 01:02:50 +00:00
JSValue error = JS_Call ( async - > context , success ? async - > promise [ 0 ] : async - > promise [ 1 ] , JS_UNDEFINED , 1 , & result ) ;
tf_util_report_error ( async - > context , error ) ;
JS_FreeValue ( async - > context , error ) ;
JS_FreeValue ( async - > context , async - > promise [ 0 ] ) ;
JS_FreeValue ( async - > context , async - > promise [ 1 ] ) ;
tf_free ( async ) ;
}
static void _tf_ssb_appendMessageWithIdentity_callback ( const char * id , bool verified , bool is_new , void * user_data )
{
append_message_t * async = user_data ;
2022-07-14 01:01:14 +00:00
JSValue result = JS_UNDEFINED ;
2023-07-20 01:02:50 +00:00
if ( verified )
{
result = is_new ? JS_TRUE : JS_FALSE ;
}
_tf_ssb_appendMessage_finish ( async , verified , result ) ;
}
static JSValue _tf_ssb_appendMessageWithIdentity ( JSContext * context , JSValueConst this_val , int argc , JSValueConst * argv )
{
append_message_t * async = tf_malloc ( sizeof ( append_message_t ) ) ;
2023-07-22 01:33:06 +00:00
* async = ( append_message_t ) { . context = context } ;
2023-07-20 01:02:50 +00:00
JSValue result = JS_NewPromiseCapability ( context , async - > promise ) ;
2022-07-14 01:01:14 +00:00
tf_ssb_t * ssb = JS_GetOpaque ( this_val , _tf_ssb_classId ) ;
if ( ssb )
{
const char * user = JS_ToCString ( context , argv [ 0 ] ) ;
const char * id = JS_ToCString ( context , argv [ 1 ] ) ;
uint8_t private_key [ crypto_sign_SECRETKEYBYTES ] ;
if ( tf_ssb_db_identity_get_private_key ( ssb , user , id , private_key , sizeof ( private_key ) ) )
{
2024-01-27 16:37:22 +00:00
JSValue signed_message = tf_ssb_sign_message ( ssb , id , private_key , argv [ 2 ] ) ;
tf_ssb_verify_strip_and_store_message ( ssb , signed_message , _tf_ssb_appendMessageWithIdentity_callback , async ) ;
JS_FreeValue ( context , signed_message ) ;
2022-07-14 01:01:14 +00:00
}
else
{
2023-07-20 01:02:50 +00:00
_tf_ssb_appendMessage_finish ( async , false , JS_ThrowInternalError ( context , " Unable to get private key for user %s with identity %s. " , user , id ) ) ;
2022-07-14 01:01:14 +00:00
}
JS_FreeCString ( context , id ) ;
JS_FreeCString ( context , user ) ;
}
2023-07-20 01:02:50 +00:00
else
{
_tf_ssb_appendMessage_finish ( async , false , JS_ThrowInternalError ( context , " No SSB instance. " ) ) ;
}
2022-07-14 01:01:14 +00:00
return result ;
}
2021-01-02 18:10:00 +00:00
static JSValue _tf_ssb_getMessage ( JSContext * context , JSValueConst this_val , int argc , JSValueConst * argv )
{
JSValue result = JS_NULL ;
tf_ssb_t * ssb = JS_GetOpaque ( this_val , _tf_ssb_classId ) ;
2021-10-10 21:51:38 +00:00
if ( ssb )
{
2021-01-02 18:10:00 +00:00
const char * id = JS_ToCString ( context , argv [ 0 ] ) ;
int64_t sequence = 0 ;
JS_ToInt64 ( context , & sequence , argv [ 1 ] ) ;
2022-02-12 01:44:11 +00:00
double timestamp = - 1.0 ;
2021-01-02 18:10:00 +00:00
char * contents = NULL ;
2021-10-10 21:51:38 +00:00
if ( tf_ssb_db_get_message_by_author_and_sequence ( ssb , id , sequence , NULL , 0 , & timestamp , & contents ) )
{
2021-01-02 18:10:00 +00:00
result = JS_NewObject ( context ) ;
2022-02-12 01:44:11 +00:00
JS_SetPropertyStr ( context , result , " timestamp " , JS_NewFloat64 ( context , timestamp ) ) ;
2021-01-02 18:10:00 +00:00
JS_SetPropertyStr ( context , result , " content " , JS_NewString ( context , contents ) ) ;
2022-06-04 17:04:51 +00:00
tf_free ( contents ) ;
2021-01-02 18:10:00 +00:00
}
JS_FreeCString ( context , id ) ;
}
return result ;
}
static JSValue _tf_ssb_blobGet ( JSContext * context , JSValueConst this_val , int argc , JSValueConst * argv )
{
JSValue result = JS_NULL ;
tf_ssb_t * ssb = JS_GetOpaque ( this_val , _tf_ssb_classId ) ;
2021-10-10 21:51:38 +00:00
if ( ssb )
{
2021-01-02 18:10:00 +00:00
const char * id = JS_ToCString ( context , argv [ 0 ] ) ;
uint8_t * blob = NULL ;
size_t size = 0 ;
2021-10-10 21:51:38 +00:00
if ( tf_ssb_db_blob_get ( ssb , id , & blob , & size ) )
{
2021-01-02 18:10:00 +00:00
result = JS_NewArrayBufferCopy ( context , blob , size ) ;
2022-06-04 17:04:51 +00:00
tf_free ( blob ) ;
2021-01-02 18:10:00 +00:00
}
2021-09-06 20:54:44 +00:00
JS_FreeCString ( context , id ) ;
2021-01-02 18:10:00 +00:00
}
return result ;
}
2023-07-18 23:46:15 +00:00
typedef struct _blob_store_t
{
JSContext * context ;
JSValue promise [ 2 ] ;
uint8_t * buffer ;
} blob_store_t ;
2024-02-10 16:50:00 +00:00
static void _tf_ssb_blob_store_complete ( blob_store_t * store , const char * id )
2023-07-18 23:46:15 +00:00
{
2023-07-22 01:33:28 +00:00
JSValue result = JS_UNDEFINED ;
if ( id )
{
JSValue id_value = JS_NewString ( store - > context , id ) ;
2024-01-08 02:30:08 +00:00
JSValue result = JS_Call ( store - > context , store - > promise [ 0 ] , JS_UNDEFINED , 1 , & id_value ) ;
2024-03-18 12:46:12 -04:00
JS_FreeValue ( store - > context , id_value ) ;
2024-01-08 02:30:08 +00:00
tf_util_report_error ( store - > context , result ) ;
JS_FreeValue ( store - > context , result ) ;
2023-07-22 01:33:28 +00:00
}
else
{
2024-01-08 02:30:08 +00:00
JSValue result = JS_Call ( store - > context , store - > promise [ 1 ] , JS_UNDEFINED , 0 , NULL ) ;
tf_util_report_error ( store - > context , result ) ;
JS_FreeValue ( store - > context , result ) ;
2023-07-22 01:33:28 +00:00
}
2023-07-18 23:46:15 +00:00
tf_util_report_error ( store - > context , result ) ;
JS_FreeValue ( store - > context , result ) ;
JS_FreeValue ( store - > context , store - > promise [ 0 ] ) ;
JS_FreeValue ( store - > context , store - > promise [ 1 ] ) ;
tf_free ( store - > buffer ) ;
tf_free ( store ) ;
}
2024-02-10 16:50:00 +00:00
static void _tf_ssb_blob_store_callback ( const char * id , bool is_new , void * user_data )
2023-07-18 23:46:15 +00:00
{
blob_store_t * store = user_data ;
_tf_ssb_blob_store_complete ( store , id ) ;
}
2021-01-02 18:10:00 +00:00
static JSValue _tf_ssb_blobStore ( JSContext * context , JSValueConst this_val , int argc , JSValueConst * argv )
{
2023-07-18 23:46:15 +00:00
blob_store_t * store = tf_malloc ( sizeof ( blob_store_t ) ) ;
2023-07-22 01:24:58 +00:00
* store = ( blob_store_t ) { . context = context } ;
2023-07-18 23:46:15 +00:00
JSValue result = JS_NewPromiseCapability ( context , store - > promise ) ;
2021-01-02 18:10:00 +00:00
tf_ssb_t * ssb = JS_GetOpaque ( this_val , _tf_ssb_classId ) ;
2021-10-10 21:51:38 +00:00
if ( ssb )
{
2021-01-02 18:10:00 +00:00
uint8_t * blob = NULL ;
size_t size = 0 ;
2021-10-10 21:51:38 +00:00
if ( JS_IsString ( argv [ 0 ] ) )
{
2021-01-02 18:10:00 +00:00
const char * text = JS_ToCStringLen ( context , & size , argv [ 0 ] ) ;
2023-07-18 23:46:15 +00:00
store - > buffer = tf_malloc ( size ) ;
memcpy ( store - > buffer , text , size ) ;
tf_ssb_db_blob_store_async ( ssb , store - > buffer , size , _tf_ssb_blob_store_callback , store ) ;
2021-01-02 18:10:00 +00:00
JS_FreeCString ( context , text ) ;
2021-10-10 21:51:38 +00:00
}
2021-11-03 22:15:46 +00:00
else if ( ( blob = tf_util_try_get_array_buffer ( context , & size , argv [ 0 ] ) ) ! = 0 )
2021-10-10 21:51:38 +00:00
{
2023-07-18 23:46:15 +00:00
store - > buffer = tf_malloc ( size ) ;
memcpy ( store - > buffer , blob , size ) ;
tf_ssb_db_blob_store_async ( ssb , store - > buffer , size , _tf_ssb_blob_store_callback , store ) ;
2021-01-02 18:10:00 +00:00
}
2021-10-31 21:15:18 +00:00
else
{
size_t offset ;
size_t element_size ;
2021-11-03 22:15:46 +00:00
JSValue buffer = tf_util_try_get_typed_array_buffer ( context , argv [ 0 ] , & offset , & size , & element_size ) ;
2021-10-31 21:15:18 +00:00
if ( ! JS_IsException ( buffer ) )
{
2021-11-03 22:15:46 +00:00
blob = tf_util_try_get_array_buffer ( context , & size , buffer ) ;
2021-10-31 21:15:18 +00:00
if ( blob )
{
2023-07-18 23:46:15 +00:00
store - > buffer = tf_malloc ( size ) ;
memcpy ( store - > buffer , blob , size ) ;
tf_ssb_db_blob_store_async ( ssb , store - > buffer , size , _tf_ssb_blob_store_callback , store ) ;
2021-10-31 21:15:18 +00:00
}
2023-07-18 23:46:15 +00:00
else
{
_tf_ssb_blob_store_complete ( store , NULL ) ;
}
}
else
{
_tf_ssb_blob_store_complete ( store , NULL ) ;
2021-10-31 21:15:18 +00:00
}
JS_FreeValue ( context , buffer ) ;
}
2021-01-02 18:10:00 +00:00
}
return result ;
}
2021-01-09 23:06:33 +00:00
static JSValue _tf_ssb_messageContentGet ( JSContext * context , JSValueConst this_val , int argc , JSValueConst * argv )
{
JSValue result = JS_NULL ;
tf_ssb_t * ssb = JS_GetOpaque ( this_val , _tf_ssb_classId ) ;
2021-10-10 21:51:38 +00:00
if ( ssb )
{
2021-01-09 23:06:33 +00:00
const char * id = JS_ToCString ( context , argv [ 0 ] ) ;
uint8_t * blob = NULL ;
size_t size = 0 ;
2021-10-10 21:51:38 +00:00
if ( tf_ssb_db_message_content_get ( ssb , id , & blob , & size ) )
{
2021-01-09 23:06:33 +00:00
result = JS_NewArrayBufferCopy ( context , blob , size ) ;
2022-06-04 17:04:51 +00:00
tf_free ( blob ) ;
2021-01-09 23:06:33 +00:00
}
2021-09-06 20:54:44 +00:00
JS_FreeCString ( context , id ) ;
2021-01-09 23:06:33 +00:00
}
return result ;
}
2021-01-02 18:10:00 +00:00
static JSValue _tf_ssb_connections ( JSContext * context , JSValueConst this_val , int argc , JSValueConst * argv )
{
JSValue result = JS_NULL ;
tf_ssb_t * ssb = JS_GetOpaque ( this_val , _tf_ssb_classId ) ;
2021-10-10 21:51:38 +00:00
if ( ssb )
{
2023-08-25 18:22:09 +00:00
tf_ssb_connection_t * connections [ 32 ] ;
int count = tf_ssb_get_connections ( ssb , connections , _countof ( connections ) ) ;
result = JS_NewArray ( context ) ;
for ( int i = 0 ; i < count ; i + + )
2021-10-10 21:51:38 +00:00
{
2023-08-25 18:22:09 +00:00
char id [ k_id_base64_len ] = { 0 } ;
tf_ssb_connection_t * connection = connections [ i ] ;
JSValue object = JS_NewObject ( context ) ;
tf_ssb_connection_get_id ( connection , id , sizeof ( id ) ) ;
JS_SetPropertyStr ( context , object , " id " , JS_NewString ( context , id ) ) ;
JS_SetPropertyStr ( context , object , " host " , JS_NewString ( context , tf_ssb_connection_get_host ( connection ) ) ) ;
JS_SetPropertyStr ( context , object , " port " , JS_NewInt32 ( context , tf_ssb_connection_get_port ( connection ) ) ) ;
tf_ssb_connection_t * tunnel = tf_ssb_connection_get_tunnel ( connection ) ;
if ( tunnel )
2021-10-10 21:51:38 +00:00
{
2023-08-25 18:22:09 +00:00
int tunnel_index = - 1 ;
for ( int j = 0 ; j < count ; j + + )
{
if ( connections [ j ] = = tunnel )
{
tunnel_index = j ;
break ;
}
}
JS_SetPropertyStr ( context , object , " tunnel " , JS_NewInt32 ( context , tunnel_index ) ) ;
2021-01-02 18:10:00 +00:00
}
2024-05-02 19:02:23 -04:00
JS_SetPropertyStr ( context , object , " requests " , tf_ssb_connection_requests_to_object ( connection ) ) ;
2023-08-25 18:22:09 +00:00
JS_SetPropertyUint32 ( context , result , i , object ) ;
2021-01-02 18:10:00 +00:00
}
}
return result ;
}
2023-01-18 00:37:45 +00:00
static JSValue _tf_ssb_storedConnections ( JSContext * context , JSValueConst this_val , int argc , JSValueConst * argv )
{
JSValue result = JS_NULL ;
tf_ssb_t * ssb = JS_GetOpaque ( this_val , _tf_ssb_classId ) ;
if ( ssb )
{
int count = 0 ;
tf_ssb_db_stored_connection_t * connections = tf_ssb_db_get_stored_connections ( ssb , & count ) ;
result = JS_NewArray ( context ) ;
for ( int i = 0 ; i < count ; i + + )
{
JSValue connection = JS_NewObject ( context ) ;
JS_SetPropertyStr ( context , connection , " address " , JS_NewString ( context , connections [ i ] . address ) ) ;
JS_SetPropertyStr ( context , connection , " port " , JS_NewInt32 ( context , connections [ i ] . port ) ) ;
JS_SetPropertyStr ( context , connection , " pubkey " , JS_NewString ( context , connections [ i ] . pubkey ) ) ;
JS_SetPropertyUint32 ( context , result , i , connection ) ;
}
tf_free ( connections ) ;
}
return result ;
}
2022-11-02 23:34:44 +00:00
static JSValue _tf_ssb_getConnection ( JSContext * context , JSValueConst this_val , int argc , JSValueConst * argv )
{
tf_ssb_t * ssb = JS_GetOpaque ( this_val , _tf_ssb_classId ) ;
const char * id = JS_ToCString ( context , argv [ 0 ] ) ;
tf_ssb_connection_t * connection = tf_ssb_connection_get ( ssb , id ) ;
JS_FreeCString ( context , id ) ;
return JS_DupValue ( context , tf_ssb_connection_get_object ( connection ) ) ;
}
2022-11-09 01:31:18 +00:00
static JSValue _tf_ssb_closeConnection ( JSContext * context , JSValueConst this_val , int argc , JSValueConst * argv )
{
tf_ssb_t * ssb = JS_GetOpaque ( this_val , _tf_ssb_classId ) ;
const char * id = JS_ToCString ( context , argv [ 0 ] ) ;
tf_ssb_connection_t * connection = tf_ssb_connection_get ( ssb , id ) ;
if ( connection )
{
tf_ssb_connection_close ( connection ) ;
}
JS_FreeCString ( context , id ) ;
return connection ? JS_TRUE : JS_FALSE ;
}
2023-02-08 01:29:44 +00:00
typedef struct _sql_work_t
{
tf_ssb_t * ssb ;
2023-08-25 20:23:40 +00:00
sqlite3 * db ;
2024-01-11 01:38:30 +00:00
char * error ;
2023-02-08 01:29:44 +00:00
const char * query ;
uint8_t * binds ;
uint8_t * rows ;
2024-01-11 01:38:30 +00:00
size_t binds_count ;
2023-02-08 01:29:44 +00:00
size_t rows_count ;
2024-01-11 01:38:30 +00:00
uv_work_t request ;
uv_async_t async ;
uv_timer_t timeout ;
uv_mutex_t lock ;
2023-02-08 01:29:44 +00:00
JSValue callback ;
JSValue promise [ 2 ] ;
int result ;
} sql_work_t ;
static void _tf_ssb_sql_append ( uint8_t * * rows , size_t * rows_count , const void * data , size_t size )
{
* rows = tf_resize_vec ( * rows , * rows_count + size ) ;
memcpy ( * rows + * rows_count , data , size ) ;
* rows_count + = size ;
}
static void _tf_ssb_sqlAsync_work ( uv_work_t * work )
{
sql_work_t * sql_work = work - > data ;
2023-08-17 00:01:59 +00:00
tf_ssb_record_thread_busy ( sql_work - > ssb , true ) ;
2023-07-20 02:20:38 +00:00
tf_trace_t * trace = tf_ssb_get_trace ( sql_work - > ssb ) ;
tf_trace_begin ( trace , " sql_async_work " ) ;
2023-08-25 19:41:54 +00:00
sqlite3 * db = tf_ssb_acquire_db_reader_restricted ( sql_work - > ssb ) ;
2023-08-25 20:23:40 +00:00
uv_mutex_lock ( & sql_work - > lock ) ;
sql_work - > db = db ;
uv_mutex_unlock ( & sql_work - > lock ) ;
uv_async_send ( & sql_work - > async ) ;
2023-02-08 01:29:44 +00:00
sqlite3_stmt * statement = NULL ;
2023-02-16 00:06:45 +00:00
sql_work - > result = sqlite3_prepare ( db , sql_work - > query , - 1 , & statement , NULL ) ;
if ( sql_work - > result = = SQLITE_OK )
2023-02-08 01:29:44 +00:00
{
const uint8_t * p = sql_work - > binds ;
int column = 0 ;
while ( p < sql_work - > binds + sql_work - > binds_count )
{
switch ( * p + + )
{
case SQLITE_INTEGER :
{
int64_t value = 0 ;
memcpy ( & value , p , sizeof ( value ) ) ;
sqlite3_bind_int64 ( statement , column + 1 , value ) ;
p + = sizeof ( value ) ;
}
break ;
case SQLITE_TEXT :
{
size_t length = 0 ;
memcpy ( & length , p , sizeof ( length ) ) ;
p + = sizeof ( length ) ;
sqlite3_bind_text ( statement , column + 1 , ( const char * ) p , length , NULL ) ;
2023-02-14 02:13:08 +00:00
p + = length ;
2023-02-08 01:29:44 +00:00
}
break ;
case SQLITE_NULL :
sqlite3_bind_null ( statement , column + 1 ) ;
break ;
default :
abort ( ) ;
}
column + + ;
}
int r = SQLITE_OK ;
while ( ( r = sqlite3_step ( statement ) ) = = SQLITE_ROW )
{
_tf_ssb_sql_append ( & sql_work - > rows , & sql_work - > rows_count , & ( uint8_t [ ] ) { ' r ' } , 1 ) ;
for ( int i = 0 ; i < sqlite3_column_count ( statement ) ; i + + )
{
_tf_ssb_sql_append ( & sql_work - > rows , & sql_work - > rows_count , & ( uint8_t [ ] ) { ' c ' } , 1 ) ;
const char * name = sqlite3_column_name ( statement , i ) ;
_tf_ssb_sql_append ( & sql_work - > rows , & sql_work - > rows_count , name , strlen ( name ) + 1 ) ;
uint8_t type = sqlite3_column_type ( statement , i ) ;
_tf_ssb_sql_append ( & sql_work - > rows , & sql_work - > rows_count , & type , sizeof ( type ) ) ;
switch ( type )
{
case SQLITE_INTEGER :
{
int64_t value = sqlite3_column_int64 ( statement , i ) ;
_tf_ssb_sql_append ( & sql_work - > rows , & sql_work - > rows_count , & value , sizeof ( value ) ) ;
}
break ;
case SQLITE_FLOAT :
{
double value = sqlite3_column_double ( statement , i ) ;
_tf_ssb_sql_append ( & sql_work - > rows , & sql_work - > rows_count , & value , sizeof ( value ) ) ;
}
break ;
case SQLITE_TEXT :
{
size_t bytes = sqlite3_column_bytes ( statement , i ) ;
_tf_ssb_sql_append ( & sql_work - > rows , & sql_work - > rows_count , & bytes , sizeof ( bytes ) ) ;
_tf_ssb_sql_append ( & sql_work - > rows , & sql_work - > rows_count , sqlite3_column_text ( statement , i ) , bytes ) ;
}
break ;
case SQLITE_BLOB :
{
size_t bytes = sqlite3_column_bytes ( statement , i ) ;
_tf_ssb_sql_append ( & sql_work - > rows , & sql_work - > rows_count , & bytes , sizeof ( bytes ) ) ;
_tf_ssb_sql_append ( & sql_work - > rows , & sql_work - > rows_count , sqlite3_column_blob ( statement , i ) , bytes ) ;
}
break ;
case SQLITE_NULL :
break ;
default :
abort ( ) ;
}
}
}
2023-02-17 02:04:48 +00:00
sql_work - > result = r ;
if ( r ! = SQLITE_OK & & r ! = SQLITE_DONE )
{
2023-08-25 20:23:40 +00:00
if ( sqlite3_is_interrupted ( db ) )
{
sql_work - > error = tf_strdup ( " Timed out " ) ;
}
else
{
sql_work - > error = tf_strdup ( sqlite3_errmsg ( db ) ) ;
}
2023-02-17 02:04:48 +00:00
}
2023-02-08 01:29:44 +00:00
_tf_ssb_sql_append ( & sql_work - > rows , & sql_work - > rows_count , & ( uint8_t [ ] ) { 0 } , 1 ) ;
sqlite3_finalize ( statement ) ;
}
else
{
2023-02-16 00:06:45 +00:00
sql_work - > error = tf_strdup ( sqlite3_errmsg ( db ) ) ;
2023-02-08 01:29:44 +00:00
}
2023-08-25 20:23:40 +00:00
uv_mutex_lock ( & sql_work - > lock ) ;
sql_work - > db = NULL ;
uv_mutex_unlock ( & sql_work - > lock ) ;
2023-02-08 01:29:44 +00:00
tf_ssb_release_db_reader ( sql_work - > ssb , db ) ;
2023-08-17 00:01:59 +00:00
tf_ssb_record_thread_busy ( sql_work - > ssb , false ) ;
2023-07-20 02:20:38 +00:00
tf_trace_end ( trace ) ;
2023-02-08 01:29:44 +00:00
}
2023-08-25 20:23:40 +00:00
static void _tf_ssb_sqlAsync_handle_close ( uv_handle_t * handle )
{
sql_work_t * work = handle - > data ;
handle - > data = NULL ;
2024-02-15 23:35:01 +00:00
if ( ! work - > async . data & & ! work - > timeout . data )
2023-08-25 20:23:40 +00:00
{
tf_free ( work ) ;
}
}
static void _tf_ssb_sqlAsync_destroy ( sql_work_t * work )
{
tf_free ( work - > binds ) ;
tf_free ( work - > error ) ;
if ( work - > rows )
{
tf_free ( work - > rows ) ;
}
uv_mutex_destroy ( & work - > lock ) ;
uv_close ( ( uv_handle_t * ) & work - > timeout , _tf_ssb_sqlAsync_handle_close ) ;
uv_close ( ( uv_handle_t * ) & work - > async , _tf_ssb_sqlAsync_handle_close ) ;
}
2023-02-08 01:29:44 +00:00
static void _tf_ssb_sqlAsync_after_work ( uv_work_t * work , int status )
{
sql_work_t * sql_work = work - > data ;
2023-07-20 02:20:38 +00:00
tf_trace_t * trace = tf_ssb_get_trace ( sql_work - > ssb ) ;
tf_trace_begin ( trace , " sql_async_after_work " ) ;
2023-02-08 01:29:44 +00:00
JSContext * context = tf_ssb_get_context ( sql_work - > ssb ) ;
uint8_t * p = sql_work - > rows ;
while ( p < sql_work - > rows + sql_work - > rows_count )
{
if ( * p + + = = ' r ' )
{
JSValue row = JS_NewObject ( context ) ;
while ( * p = = ' c ' )
{
p + + ;
const char * column_name = ( const char * ) p ;
size_t length = strlen ( ( char * ) p ) ;
p + = length + 1 ;
switch ( * p + + )
{
case SQLITE_INTEGER :
{
int64_t value = 0 ;
memcpy ( & value , p , sizeof ( value ) ) ;
JS_SetPropertyStr ( context , row , column_name , JS_NewInt64 ( context , value ) ) ;
p + = sizeof ( value ) ;
}
break ;
case SQLITE_FLOAT :
{
double value = 0.0 ;
memcpy ( & value , p , sizeof ( value ) ) ;
JS_SetPropertyStr ( context , row , column_name , JS_NewFloat64 ( context , value ) ) ;
p + = sizeof ( value ) ;
}
break ;
case SQLITE_TEXT :
case SQLITE_BLOB :
{
size_t length = 0 ;
memcpy ( & length , p , sizeof ( length ) ) ;
p + = sizeof ( length ) ;
JS_SetPropertyStr ( context , row , column_name , JS_NewStringLen ( context , ( const char * ) p , length ) ) ;
p + = length ;
}
break ;
case SQLITE_NULL :
JS_SetPropertyStr ( context , row , column_name , JS_NULL ) ;
break ;
}
}
JSValue response = JS_Call ( context , sql_work - > callback , JS_UNDEFINED , 1 , & row ) ;
2023-08-25 20:23:40 +00:00
bool is_error = tf_util_report_error ( context , response ) ;
2023-02-15 02:56:01 +00:00
JS_FreeValue ( context , response ) ;
2023-02-08 01:29:44 +00:00
JS_FreeValue ( context , row ) ;
2023-08-25 20:23:40 +00:00
if ( is_error )
{
break ;
}
2023-02-08 01:29:44 +00:00
}
else
{
break ;
}
}
2023-02-16 00:06:45 +00:00
JSValue result = JS_UNDEFINED ;
2023-02-17 02:04:48 +00:00
if ( sql_work - > result = = SQLITE_OK | | sql_work - > result = = SQLITE_DONE )
2023-02-16 00:06:45 +00:00
{
result = JS_Call ( context , sql_work - > promise [ 0 ] , JS_UNDEFINED , 0 , NULL ) ;
tf_util_report_error ( context , result ) ;
}
else
{
JSValue error = JS_ThrowInternalError ( context , " SQL Error %s: %s " , sql_work - > error , sql_work - > query ) ;
JSValue exception = JS_GetException ( context ) ;
result = JS_Call ( context , sql_work - > promise [ 1 ] , JS_UNDEFINED , 1 , & exception ) ;
tf_util_report_error ( context , result ) ;
JS_FreeValue ( context , exception ) ;
JS_FreeValue ( context , error ) ;
}
2023-02-15 02:59:46 +00:00
JS_FreeValue ( context , result ) ;
2023-02-08 01:29:44 +00:00
JS_FreeValue ( context , sql_work - > promise [ 0 ] ) ;
JS_FreeValue ( context , sql_work - > promise [ 1 ] ) ;
2023-02-08 01:50:47 +00:00
JS_FreeValue ( context , sql_work - > callback ) ;
2024-01-27 18:26:01 +00:00
JS_FreeCString ( context , sql_work - > query ) ;
2023-08-25 20:23:40 +00:00
_tf_ssb_sqlAsync_destroy ( sql_work ) ;
2023-07-20 02:20:38 +00:00
tf_trace_end ( trace ) ;
2023-02-08 01:29:44 +00:00
}
2023-08-25 20:23:40 +00:00
static void _tf_ssb_sqlAsync_timeout ( uv_timer_t * timer )
{
sql_work_t * work = timer - > data ;
uv_mutex_lock ( & work - > lock ) ;
if ( work - > db )
{
sqlite3_interrupt ( work - > db ) ;
}
uv_mutex_unlock ( & work - > lock ) ;
}
static void _tf_ssb_sqlAsync_start_timer ( uv_async_t * async )
{
sql_work_t * work = async - > data ;
uv_timer_start ( & work - > timeout , _tf_ssb_sqlAsync_timeout , k_sql_async_timeout_ms , 0 ) ;
}
2023-02-08 01:29:44 +00:00
static JSValue _tf_ssb_sqlAsync ( JSContext * context , JSValueConst this_val , int argc , JSValueConst * argv )
{
tf_ssb_t * ssb = JS_GetOpaque ( this_val , _tf_ssb_classId ) ;
2023-02-14 02:13:08 +00:00
const char * query = JS_ToCString ( context , argv [ 0 ] ) ;
sql_work_t * work = tf_malloc ( sizeof ( sql_work_t ) ) ;
* work = ( sql_work_t )
2023-02-08 01:29:44 +00:00
{
2023-02-14 02:13:08 +00:00
. request =
2023-02-08 01:29:44 +00:00
{
2023-02-14 02:13:08 +00:00
. data = work ,
} ,
2023-08-25 20:23:40 +00:00
. async =
{
. data = work ,
} ,
. timeout =
{
. data = work ,
} ,
2023-02-14 02:13:08 +00:00
. ssb = ssb ,
. callback = JS_DupValue ( context , argv [ 2 ] ) ,
. query = query ,
} ;
2023-08-25 20:23:40 +00:00
uv_mutex_init ( & work - > lock ) ;
uv_async_init ( tf_ssb_get_loop ( ssb ) , & work - > async , _tf_ssb_sqlAsync_start_timer ) ;
uv_timer_init ( tf_ssb_get_loop ( ssb ) , & work - > timeout ) ;
2023-02-14 02:13:08 +00:00
JSValue result = JS_NewPromiseCapability ( context , work - > promise ) ;
JSValue error_value = JS_UNDEFINED ;
if ( ssb )
{
2023-02-08 01:29:44 +00:00
int32_t length = tf_util_get_length ( context , argv [ 1 ] ) ;
for ( int i = 0 ; i < length ; i + + )
{
JSValue value = JS_GetPropertyUint32 ( context , argv [ 1 ] , i ) ;
if ( JS_IsNumber ( value ) )
{
uint8_t type = SQLITE_INTEGER ;
int64_t number = 0 ;
JS_ToInt64 ( context , & number , value ) ;
_tf_ssb_sql_append ( & work - > binds , & work - > binds_count , & type , sizeof ( type ) ) ;
_tf_ssb_sql_append ( & work - > binds , & work - > binds_count , & number , sizeof ( number ) ) ;
}
else if ( JS_IsBool ( value ) )
{
uint8_t type = SQLITE_INTEGER ;
int64_t number = JS_ToBool ( context , value ) ? 1 : 0 ;
_tf_ssb_sql_append ( & work - > binds , & work - > binds_count , & type , sizeof ( type ) ) ;
_tf_ssb_sql_append ( & work - > binds , & work - > binds_count , & number , sizeof ( number ) ) ;
}
2023-02-14 02:13:08 +00:00
else if ( JS_IsNull ( value ) )
2023-02-08 01:29:44 +00:00
{
uint8_t type = SQLITE_NULL ;
_tf_ssb_sql_append ( & work - > binds , & work - > binds_count , & type , sizeof ( type ) ) ;
}
else
{
uint8_t type = SQLITE_TEXT ;
size_t length = 0 ;
const char * string = JS_ToCStringLen ( context , & length , value ) ;
if ( ! string )
{
string = " " ;
}
_tf_ssb_sql_append ( & work - > binds , & work - > binds_count , & type , sizeof ( type ) ) ;
2023-02-14 02:13:08 +00:00
_tf_ssb_sql_append ( & work - > binds , & work - > binds_count , & length , sizeof ( length ) ) ;
2023-02-08 01:29:44 +00:00
_tf_ssb_sql_append ( & work - > binds , & work - > binds_count , string , length ) ;
JS_FreeCString ( context , string ) ;
}
2023-02-19 13:51:06 +00:00
JS_FreeValue ( context , value ) ;
2023-02-08 01:29:44 +00:00
}
int r = uv_queue_work ( tf_ssb_get_loop ( ssb ) , & work - > request , _tf_ssb_sqlAsync_work , _tf_ssb_sqlAsync_after_work ) ;
if ( r )
{
2023-02-14 02:13:08 +00:00
error_value = JS_ThrowInternalError ( context , " uv_queue_work failed: %s " , uv_strerror ( r ) ) ;
2023-02-08 01:29:44 +00:00
}
}
2023-02-14 02:13:08 +00:00
if ( ! JS_IsUndefined ( error_value ) )
{
JSValue call_result = JS_Call ( context , work - > promise [ 1 ] , JS_UNDEFINED , 1 , & error_value ) ;
tf_util_report_error ( context , call_result ) ;
2023-02-15 02:59:46 +00:00
JS_FreeValue ( context , call_result ) ;
2023-08-25 20:23:40 +00:00
JS_FreeValue ( context , error_value ) ;
JS_FreeCString ( context , query ) ;
2023-02-14 02:13:08 +00:00
JS_FreeValue ( context , work - > promise [ 0 ] ) ;
JS_FreeValue ( context , work - > promise [ 1 ] ) ;
JS_FreeValue ( context , work - > callback ) ;
2023-08-25 20:23:40 +00:00
_tf_ssb_sqlAsync_destroy ( work ) ;
2023-02-14 02:13:08 +00:00
}
2023-02-08 01:29:44 +00:00
return result ;
}
2023-07-20 01:02:50 +00:00
typedef struct _message_store_t
{
JSContext * context ;
JSValue promise [ 2 ] ;
} message_store_t ;
2024-02-10 16:50:00 +00:00
static void _tf_ssb_message_store_callback ( const char * id , bool verified , bool is_new , void * user_data )
2023-07-20 01:02:50 +00:00
{
message_store_t * store = user_data ;
JSValue result = JS_Call ( store - > context , id ? store - > promise [ 0 ] : store - > promise [ 1 ] , JS_UNDEFINED , 0 , NULL ) ;
tf_util_report_error ( store - > context , result ) ;
JS_FreeValue ( store - > context , result ) ;
JS_FreeValue ( store - > context , store - > promise [ 0 ] ) ;
JS_FreeValue ( store - > context , store - > promise [ 1 ] ) ;
tf_free ( store ) ;
}
2021-09-06 17:50:38 +00:00
static JSValue _tf_ssb_storeMessage ( JSContext * context , JSValueConst this_val , int argc , JSValueConst * argv )
{
tf_ssb_t * ssb = JS_GetOpaque ( this_val , _tf_ssb_classId ) ;
2023-07-20 01:02:50 +00:00
message_store_t * store = tf_malloc ( sizeof ( message_store_t ) ) ;
* store = ( message_store_t ) { . context = context } ;
JSValue result = JS_NewPromiseCapability ( context , store - > promise ) ;
tf_ssb_verify_strip_and_store_message ( ssb , argv [ 0 ] , _tf_ssb_message_store_callback , store ) ;
return result ;
2021-09-06 17:50:38 +00:00
}
2021-01-02 18:10:00 +00:00
typedef struct _broadcasts_t
{
JSContext * context ;
JSValue array ;
int length ;
} broadcasts_t ;
2022-11-09 03:51:31 +00:00
static void _tf_ssb_broadcasts_visit ( const char * host , const struct sockaddr_in * addr , tf_ssb_connection_t * tunnel , const uint8_t * pub , void * user_data )
2021-01-02 18:10:00 +00:00
{
broadcasts_t * broadcasts = user_data ;
JSValue entry = JS_NewObject ( broadcasts - > context ) ;
char pubkey [ k_id_base64_len ] ;
tf_ssb_id_bin_to_str ( pubkey , sizeof ( pubkey ) , pub ) ;
2022-11-02 23:34:44 +00:00
if ( tunnel )
{
JS_SetPropertyStr ( broadcasts - > context , entry , " tunnel " , JS_DupValue ( broadcasts - > context , tf_ssb_connection_get_object ( tunnel ) ) ) ;
}
else
{
2022-11-09 03:51:31 +00:00
JS_SetPropertyStr ( broadcasts - > context , entry , " address " , JS_NewString ( broadcasts - > context , host ) ) ;
2022-11-02 23:34:44 +00:00
JS_SetPropertyStr ( broadcasts - > context , entry , " port " , JS_NewInt32 ( broadcasts - > context , ntohs ( addr - > sin_port ) ) ) ;
}
2021-01-02 18:10:00 +00:00
JS_SetPropertyStr ( broadcasts - > context , entry , " pubkey " , JS_NewString ( broadcasts - > context , pubkey ) ) ;
JS_SetPropertyUint32 ( broadcasts - > context , broadcasts - > array , broadcasts - > length + + , entry ) ;
}
static JSValue _tf_ssb_getBroadcasts ( JSContext * context , JSValueConst this_val , int argc , JSValueConst * argv )
{
JSValue result = JS_UNDEFINED ;
tf_ssb_t * ssb = JS_GetOpaque ( this_val , _tf_ssb_classId ) ;
2021-10-10 21:51:38 +00:00
if ( ssb )
{
2021-01-02 18:10:00 +00:00
result = JS_NewArray ( context ) ;
broadcasts_t broadcasts = {
. context = context ,
. array = result ,
. length = 0 ,
} ;
tf_ssb_visit_broadcasts ( ssb , _tf_ssb_broadcasts_visit , & broadcasts ) ;
}
return result ;
}
static JSValue _tf_ssb_connect ( JSContext * context , JSValueConst this_val , int argc , JSValueConst * argv )
{
JSValue args = argv [ 0 ] ;
tf_ssb_t * ssb = JS_GetOpaque ( this_val , _tf_ssb_classId ) ;
2021-10-10 21:51:38 +00:00
if ( ssb )
{
if ( JS_IsString ( args ) )
{
2021-01-02 18:10:00 +00:00
const char * address_str = JS_ToCString ( context , args ) ;
2023-03-07 17:50:17 +00:00
tf_printf ( " Connecting to %s \n " , address_str ) ;
2021-01-02 18:10:00 +00:00
tf_ssb_connect_str ( ssb , address_str ) ;
JS_FreeCString ( context , address_str ) ;
2021-10-10 21:51:38 +00:00
}
else
{
2021-01-02 18:10:00 +00:00
JSValue address = JS_GetPropertyStr ( context , args , " address " ) ;
JSValue port = JS_GetPropertyStr ( context , args , " port " ) ;
JSValue pubkey = JS_GetPropertyStr ( context , args , " pubkey " ) ;
const char * address_str = JS_ToCString ( context , address ) ;
int32_t port_int = 0 ;
JS_ToInt32 ( context , & port_int , port ) ;
const char * pubkey_str = JS_ToCString ( context , pubkey ) ;
2022-11-09 01:31:18 +00:00
if ( pubkey_str )
{
2023-03-07 17:50:17 +00:00
tf_printf ( " Connecting to %s:%d \n " , address_str , port_int ) ;
2022-11-09 01:31:18 +00:00
uint8_t pubkey_bin [ k_id_bin_len ] ;
tf_ssb_id_str_to_bin ( pubkey_bin , pubkey_str ) ;
tf_ssb_connect ( ssb , address_str , port_int , pubkey_bin ) ;
}
else
{
2023-03-07 17:50:17 +00:00
tf_printf ( " Not connecting to null. \n " ) ;
2022-11-09 01:31:18 +00:00
}
2021-01-02 18:10:00 +00:00
JS_FreeCString ( context , pubkey_str ) ;
JS_FreeCString ( context , address_str ) ;
JS_FreeValue ( context , address ) ;
JS_FreeValue ( context , port ) ;
JS_FreeValue ( context , pubkey ) ;
}
}
return JS_UNDEFINED ;
}
2023-01-18 00:37:45 +00:00
static JSValue _tf_ssb_forgetStoredConnection ( JSContext * context , JSValueConst this_val , int argc , JSValueConst * argv )
{
JSValue args = argv [ 0 ] ;
tf_ssb_t * ssb = JS_GetOpaque ( this_val , _tf_ssb_classId ) ;
if ( ssb )
{
JSValue address = JS_GetPropertyStr ( context , args , " address " ) ;
JSValue port = JS_GetPropertyStr ( context , args , " port " ) ;
JSValue pubkey = JS_GetPropertyStr ( context , args , " pubkey " ) ;
const char * address_str = JS_ToCString ( context , address ) ;
int32_t port_int = 0 ;
JS_ToInt32 ( context , & port_int , port ) ;
const char * pubkey_str = JS_ToCString ( context , pubkey ) ;
if ( pubkey_str )
{
tf_ssb_db_forget_stored_connection ( ssb , address_str , port_int , pubkey_str ) ;
}
JS_FreeCString ( context , pubkey_str ) ;
JS_FreeCString ( context , address_str ) ;
JS_FreeValue ( context , address ) ;
JS_FreeValue ( context , port ) ;
JS_FreeValue ( context , pubkey ) ;
}
return JS_UNDEFINED ;
}
2022-11-02 23:34:44 +00:00
static void _tf_ssb_cleanup_value ( tf_ssb_t * ssb , void * user_data )
{
JSValue callback = JS_MKPTR ( JS_TAG_OBJECT , user_data ) ;
JS_FreeValue ( tf_ssb_get_context ( ssb ) , callback ) ;
}
2021-11-11 00:05:07 +00:00
static void _tf_ssb_on_message_added_callback ( tf_ssb_t * ssb , const char * id , void * user_data )
{
JSContext * context = tf_ssb_get_context ( ssb ) ;
JSValue callback = JS_MKPTR ( JS_TAG_OBJECT , user_data ) ;
JSValue string = JS_NewString ( context , id ) ;
JSValue response = JS_Call ( context , callback , JS_UNDEFINED , 1 , & string ) ;
2021-12-27 21:48:16 +00:00
if ( tf_util_report_error ( context , response ) )
{
tf_ssb_remove_message_added_callback ( ssb , _tf_ssb_on_message_added_callback , user_data ) ;
}
2021-11-11 00:05:07 +00:00
JS_FreeValue ( context , response ) ;
JS_FreeValue ( context , string ) ;
}
2021-11-07 22:28:58 +00:00
static void _tf_ssb_on_blob_want_added_callback ( tf_ssb_t * ssb , const char * id , void * user_data )
2021-09-06 17:50:38 +00:00
{
JSContext * context = tf_ssb_get_context ( ssb ) ;
JSValue callback = JS_MKPTR ( JS_TAG_OBJECT , user_data ) ;
JSValue string = JS_NewString ( context , id ) ;
JSValue response = JS_Call ( context , callback , JS_UNDEFINED , 1 , & string ) ;
2021-12-27 21:48:16 +00:00
if ( tf_util_report_error ( context , response ) )
{
tf_ssb_remove_blob_want_added_callback ( ssb , _tf_ssb_on_blob_want_added_callback , user_data ) ;
}
2021-09-06 17:50:38 +00:00
JS_FreeValue ( context , response ) ;
JS_FreeValue ( context , string ) ;
}
2021-11-07 22:28:58 +00:00
static void _tf_ssb_on_connections_changed_callback ( tf_ssb_t * ssb , tf_ssb_change_t change , tf_ssb_connection_t * connection , void * user_data )
2021-09-06 17:50:38 +00:00
{
JSContext * context = tf_ssb_get_context ( ssb ) ;
JSValue callback = JS_MKPTR ( JS_TAG_OBJECT , user_data ) ;
JSValue response = JS_UNDEFINED ;
switch ( change )
{
case k_tf_ssb_change_create :
break ;
2024-05-02 19:02:23 -04:00
case k_tf_ssb_change_update :
{
JSValue object = JS_DupValue ( context , tf_ssb_connection_get_object ( connection ) ) ;
JSValue args [ ] = {
JS_NewString ( context , " update " ) ,
object ,
} ;
response = JS_Call ( context , callback , JS_UNDEFINED , 2 , args ) ;
if ( tf_util_report_error ( context , response ) )
{
tf_ssb_remove_connections_changed_callback ( ssb , _tf_ssb_on_connections_changed_callback , user_data ) ;
}
JS_FreeValue ( context , args [ 0 ] ) ;
JS_FreeValue ( context , object ) ;
}
break ;
2021-09-06 17:50:38 +00:00
case k_tf_ssb_change_connect :
{
2021-09-06 18:23:22 +00:00
JSValue object = JS_DupValue ( context , tf_ssb_connection_get_object ( connection ) ) ;
2024-02-15 23:35:01 +00:00
JSValue args [ ] = {
2021-09-06 17:50:38 +00:00
JS_NewString ( context , " add " ) ,
object ,
} ;
response = JS_Call ( context , callback , JS_UNDEFINED , 2 , args ) ;
2021-12-27 21:48:16 +00:00
if ( tf_util_report_error ( context , response ) )
{
tf_ssb_remove_connections_changed_callback ( ssb , _tf_ssb_on_connections_changed_callback , user_data ) ;
}
2021-09-06 18:23:22 +00:00
JS_FreeValue ( context , args [ 0 ] ) ;
JS_FreeValue ( context , object ) ;
2021-09-06 17:50:38 +00:00
}
break ;
case k_tf_ssb_change_remove :
{
2021-09-06 18:23:22 +00:00
JSValue object = JS_DupValue ( context , tf_ssb_connection_get_object ( connection ) ) ;
2024-02-15 23:35:01 +00:00
JSValue args [ ] = {
2021-09-06 17:50:38 +00:00
JS_NewString ( context , " remove " ) ,
object ,
} ;
response = JS_Call ( context , callback , JS_UNDEFINED , 2 , args ) ;
2021-12-27 21:48:16 +00:00
if ( tf_util_report_error ( context , response ) )
{
tf_ssb_remove_connections_changed_callback ( ssb , _tf_ssb_on_connections_changed_callback , user_data ) ;
}
2021-09-06 18:23:22 +00:00
JS_FreeValue ( context , args [ 0 ] ) ;
JS_FreeValue ( context , object ) ;
2021-09-06 17:50:38 +00:00
}
break ;
}
JS_FreeValue ( context , response ) ;
}
2021-11-07 22:28:58 +00:00
static void _tf_ssb_on_broadcasts_changed_callback ( tf_ssb_t * ssb , void * user_data )
2021-09-06 17:50:38 +00:00
{
2021-11-07 22:28:58 +00:00
JSContext * context = tf_ssb_get_context ( ssb ) ;
JSValue callback = JS_MKPTR ( JS_TAG_OBJECT , user_data ) ;
2021-12-17 23:58:59 +00:00
JSValue argv = JS_UNDEFINED ;
JSValue response = JS_Call ( context , callback , JS_UNDEFINED , 1 , & argv ) ;
2021-12-27 21:48:16 +00:00
if ( tf_util_report_error ( context , response ) )
{
2021-12-27 22:00:37 +00:00
tf_ssb_remove_broadcasts_changed_callback ( ssb , _tf_ssb_on_broadcasts_changed_callback , user_data ) ;
2021-12-27 21:48:16 +00:00
}
2021-11-07 22:28:58 +00:00
JS_FreeValue ( context , response ) ;
2021-09-06 17:50:38 +00:00
}
2021-11-07 22:28:58 +00:00
static JSValue _tf_ssb_add_event_listener ( JSContext * context , JSValueConst this_val , int argc , JSValueConst * argv )
{
tf_ssb_t * ssb = JS_GetOpaque ( this_val , _tf_ssb_classId ) ;
const char * event_name = JS_ToCString ( context , argv [ 0 ] ) ;
JSValue callback = argv [ 1 ] ;
JSValue result = JS_UNDEFINED ;
if ( ! event_name )
{
result = JS_ThrowTypeError ( context , " Expected argument 1 to be a string event name. " ) ;
}
else if ( ! JS_IsFunction ( context , callback ) )
{
result = JS_ThrowTypeError ( context , " Expected argument 2 to be a function. " ) ;
}
else
{
if ( strcmp ( event_name , " connections " ) = = 0 )
{
void * ptr = JS_VALUE_GET_PTR ( JS_DupValue ( context , callback ) ) ;
tf_ssb_add_connections_changed_callback ( ssb , _tf_ssb_on_connections_changed_callback , _tf_ssb_cleanup_value , ptr ) ;
}
else if ( strcmp ( event_name , " broadcasts " ) = = 0 )
{
void * ptr = JS_VALUE_GET_PTR ( JS_DupValue ( context , callback ) ) ;
tf_ssb_add_broadcasts_changed_callback ( ssb , _tf_ssb_on_broadcasts_changed_callback , _tf_ssb_cleanup_value , ptr ) ;
}
2021-11-11 00:05:07 +00:00
else if ( strcmp ( event_name , " message " ) = = 0 )
{
void * ptr = JS_VALUE_GET_PTR ( JS_DupValue ( context , callback ) ) ;
tf_ssb_add_message_added_callback ( ssb , _tf_ssb_on_message_added_callback , _tf_ssb_cleanup_value , ptr ) ;
}
2021-11-07 22:28:58 +00:00
else if ( strcmp ( event_name , " blob_want_added " ) = = 0 )
{
void * ptr = JS_VALUE_GET_PTR ( JS_DupValue ( context , callback ) ) ;
tf_ssb_add_blob_want_added_callback ( ssb , _tf_ssb_on_blob_want_added_callback , _tf_ssb_cleanup_value , ptr ) ;
}
}
JS_FreeCString ( context , event_name ) ;
return result ;
}
static JSValue _tf_ssb_remove_event_listener ( JSContext * context , JSValueConst this_val , int argc , JSValueConst * argv )
{
tf_ssb_t * ssb = JS_GetOpaque ( this_val , _tf_ssb_classId ) ;
const char * event_name = JS_ToCString ( context , argv [ 0 ] ) ;
JSValue callback = argv [ 1 ] ;
JSValue result = JS_UNDEFINED ;
if ( ! event_name )
{
result = JS_ThrowTypeError ( context , " Expected argument 1 to be a string event name. " ) ;
}
else if ( ! JS_IsFunction ( context , callback ) )
{
result = JS_ThrowTypeError ( context , " Expected argument 2 to be a function. " ) ;
}
else
{
if ( strcmp ( event_name , " connections " ) = = 0 )
{
void * ptr = JS_VALUE_GET_PTR ( JS_DupValue ( context , callback ) ) ;
tf_ssb_remove_connections_changed_callback ( ssb , _tf_ssb_on_connections_changed_callback , ptr ) ;
}
else if ( strcmp ( event_name , " broadcasts " ) = = 0 )
{
void * ptr = JS_VALUE_GET_PTR ( JS_DupValue ( context , callback ) ) ;
tf_ssb_remove_broadcasts_changed_callback ( ssb , _tf_ssb_on_broadcasts_changed_callback , ptr ) ;
}
2021-11-17 23:47:55 +00:00
else if ( strcmp ( event_name , " message " ) = = 0 )
2021-11-11 00:05:07 +00:00
{
void * ptr = JS_VALUE_GET_PTR ( JS_DupValue ( context , callback ) ) ;
tf_ssb_remove_message_added_callback ( ssb , _tf_ssb_on_message_added_callback , ptr ) ;
}
2021-11-07 22:28:58 +00:00
else if ( strcmp ( event_name , " blob_want_added " ) = = 0 )
{
void * ptr = JS_VALUE_GET_PTR ( JS_DupValue ( context , callback ) ) ;
tf_ssb_remove_blob_want_added_callback ( ssb , _tf_ssb_on_blob_want_added_callback , ptr ) ;
}
}
JS_FreeCString ( context , event_name ) ;
return result ;
}
2022-12-01 01:19:35 +00:00
static JSValue _tf_ssb_createTunnel ( JSContext * context , JSValueConst this_val , int argc , JSValueConst * argv )
{
2023-01-17 23:10:17 +00:00
JSValue result = JS_UNDEFINED ;
2022-12-01 01:19:35 +00:00
tf_ssb_t * ssb = JS_GetOpaque ( this_val , _tf_ssb_classId ) ;
const char * portal_id = JS_ToCString ( context , argv [ 0 ] ) ;
2023-01-17 23:10:17 +00:00
const char * target_id = JS_ToCString ( context , argv [ 1 ] ) ;
tf_ssb_connection_t * connection = tf_ssb_connection_get ( ssb , portal_id ) ;
if ( connection )
{
int32_t request_number = tf_ssb_connection_next_request_number ( connection ) ;
JSValue message = JS_NewObject ( context ) ;
JSValue name = JS_NewArray ( context ) ;
JS_SetPropertyUint32 ( context , name , 0 , JS_NewString ( context , " tunnel " ) ) ;
JS_SetPropertyUint32 ( context , name , 1 , JS_NewString ( context , " connect " ) ) ;
JS_SetPropertyStr ( context , message , " name " , name ) ;
JSValue arg = JS_NewObject ( context ) ;
JS_SetPropertyStr ( context , arg , " portal " , JS_NewString ( context , portal_id ) ) ;
JS_SetPropertyStr ( context , arg , " target " , JS_NewString ( context , target_id ) ) ;
JSValue args = JS_NewArray ( context ) ;
JS_SetPropertyUint32 ( context , args , 0 , arg ) ;
JS_SetPropertyStr ( context , message , " args " , args ) ;
JS_SetPropertyStr ( context , message , " type " , JS_NewString ( context , " duplex " ) ) ;
2024-05-02 19:02:23 -04:00
tf_ssb_connection_rpc_send_json ( connection , k_ssb_rpc_flag_stream | k_ssb_rpc_flag_new_request , request_number , " tunnel.connect " , message , NULL , NULL , NULL ) ;
2023-01-17 23:10:17 +00:00
JS_FreeValue ( context , message ) ;
tf_ssb_connection_tunnel_create ( ssb , portal_id , request_number , target_id ) ;
result = JS_TRUE ;
}
2022-12-01 01:19:35 +00:00
JS_FreeCString ( context , target_id ) ;
JS_FreeCString ( context , portal_id ) ;
2023-01-17 23:10:17 +00:00
return result ;
2022-12-01 01:19:35 +00:00
}
2024-02-15 23:35:01 +00:00
enum
{
k_max_private_message_recipients = 8
} ;
2023-02-26 19:51:54 +00:00
static bool _tf_ssb_get_private_key_curve25519 ( sqlite3 * db , const char * user , const char * identity , uint8_t out_private_key [ static crypto_sign_SECRETKEYBYTES ] )
{
if ( ! user | | ! identity | | ! out_private_key )
{
2023-07-20 05:06:15 +00:00
tf_printf ( " user=%p identity=%p out_private_key=%p \n " , user , identity , out_private_key ) ;
2023-02-26 19:51:54 +00:00
return false ;
}
bool success = false ;
sqlite3_stmt * statement = NULL ;
if ( sqlite3_prepare ( db , " SELECT private_key FROM identities WHERE user = ? AND public_key = ? " , - 1 , & statement , NULL ) = = SQLITE_OK )
{
2024-02-17 19:22:02 +00:00
if ( sqlite3_bind_text ( statement , 1 , user , - 1 , NULL ) = = SQLITE_OK & & sqlite3_bind_text ( statement , 2 , * identity = = ' @ ' ? identity + 1 : identity , - 1 , NULL ) = = SQLITE_OK )
2023-02-26 19:51:54 +00:00
{
while ( sqlite3_step ( statement ) = = SQLITE_ROW )
{
uint8_t key [ crypto_sign_SECRETKEYBYTES ] = { 0 } ;
2024-02-17 19:22:02 +00:00
int length = tf_base64_decode ( ( const char * ) sqlite3_column_text ( statement , 0 ) , sqlite3_column_bytes ( statement , 0 ) - strlen ( " .ed25519 " ) , key , sizeof ( key ) ) ;
2023-02-26 19:51:54 +00:00
if ( length = = crypto_sign_SECRETKEYBYTES )
{
success = crypto_sign_ed25519_sk_to_curve25519 ( out_private_key , key ) = = 0 ;
}
}
}
sqlite3_finalize ( statement ) ;
}
return success ;
}
static JSValue _tf_ssb_private_message_encrypt ( JSContext * context , JSValueConst this_val , int argc , JSValueConst * argv )
{
JSValue result = JS_UNDEFINED ;
int recipient_count = tf_util_get_length ( context , argv [ 2 ] ) ;
if ( recipient_count < 1 | | recipient_count > k_max_private_message_recipients )
{
return JS_ThrowRangeError ( context , " Number of recipients must be between 1 and %d. " , k_max_private_message_recipients ) ;
}
const char * signer_user = JS_ToCString ( context , argv [ 0 ] ) ;
const char * signer_identity = JS_ToCString ( context , argv [ 1 ] ) ;
uint8_t recipients [ k_max_private_message_recipients ] [ crypto_scalarmult_curve25519_SCALARBYTES ] ;
size_t message_size = 0 ;
const char * message = JS_ToCStringLen ( context , & message_size , argv [ 3 ] ) ;
for ( int i = 0 ; i < recipient_count & & JS_IsUndefined ( result ) ; i + + )
{
JSValue recipient = JS_GetPropertyUint32 ( context , argv [ 2 ] , i ) ;
const char * id = JS_ToCString ( context , recipient ) ;
if ( id )
{
const char * type = strstr ( id , " .ed25519 " ) ;
const char * id_start = * id = = ' @ ' ? id + 1 : id ;
uint8_t key [ crypto_box_PUBLICKEYBYTES ] = { 0 } ;
if ( tf_base64_decode ( id_start , type ? ( size_t ) ( type - id_start ) : strlen ( id_start ) , key , sizeof ( key ) ) ! = sizeof ( key ) )
{
result = JS_ThrowInternalError ( context , " Invalid recipient: %s. \n " , id ) ;
}
else if ( crypto_sign_ed25519_pk_to_curve25519 ( recipients [ i ] , key ) ! = 0 )
{
result = JS_ThrowInternalError ( context , " Failed to convert recipient ID. \n " ) ;
}
JS_FreeCString ( context , id ) ;
}
JS_FreeValue ( context , recipient ) ;
}
if ( JS_IsUndefined ( result ) )
{
tf_ssb_t * ssb = JS_GetOpaque ( this_val , _tf_ssb_classId ) ;
2023-07-20 05:06:15 +00:00
uint8_t private_key [ crypto_sign_SECRETKEYBYTES ] = { 0 } ;
2023-06-15 00:27:49 +00:00
sqlite3 * db = tf_ssb_acquire_db_reader ( ssb ) ;
2023-07-20 05:06:15 +00:00
bool found = _tf_ssb_get_private_key_curve25519 ( db , signer_user , signer_identity , private_key ) ;
tf_ssb_release_db_reader ( ssb , db ) ;
2023-02-26 19:51:54 +00:00
2023-07-20 05:06:15 +00:00
if ( found )
2023-02-26 19:51:54 +00:00
{
uint8_t public_key [ crypto_box_PUBLICKEYBYTES ] = { 0 } ;
uint8_t secret_key [ crypto_box_SECRETKEYBYTES ] = { 0 } ;
uint8_t nonce [ crypto_box_NONCEBYTES ] = { 0 } ;
uint8_t body_key [ crypto_box_SECRETKEYBYTES ] = { 0 } ;
crypto_box_keypair ( public_key , secret_key ) ;
randombytes_buf ( nonce , sizeof ( nonce ) ) ;
randombytes_buf ( body_key , sizeof ( body_key ) ) ;
uint8_t length_and_key [ 1 + sizeof ( body_key ) ] ;
length_and_key [ 0 ] = ( uint8_t ) recipient_count ;
memcpy ( length_and_key + 1 , body_key , sizeof ( body_key ) ) ;
2024-02-15 23:35:01 +00:00
size_t payload_size =
2024-02-17 19:22:02 +00:00
sizeof ( nonce ) + sizeof ( public_key ) + ( crypto_secretbox_MACBYTES + sizeof ( length_and_key ) ) * recipient_count + crypto_secretbox_MACBYTES + message_size ;
2023-02-26 19:51:54 +00:00
uint8_t * payload = tf_malloc ( payload_size ) ;
uint8_t * p = payload ;
memcpy ( p , nonce , sizeof ( nonce ) ) ;
p + = sizeof ( nonce ) ;
memcpy ( p , public_key , sizeof ( public_key ) ) ;
p + = sizeof ( public_key ) ;
for ( int i = 0 ; i < recipient_count & & JS_IsUndefined ( result ) ; i + + )
{
uint8_t shared_secret [ crypto_secretbox_KEYBYTES ] = { 0 } ;
if ( crypto_scalarmult ( shared_secret , secret_key , recipients [ i ] ) = = 0 )
{
if ( crypto_secretbox_easy ( p , length_and_key , sizeof ( length_and_key ) , nonce , shared_secret ) ! = 0 )
{
result = JS_ThrowInternalError ( context , " crypto_secretbox_easy failed " ) ;
}
else
{
p + = crypto_secretbox_MACBYTES + sizeof ( length_and_key ) ;
}
}
else
{
result = JS_ThrowInternalError ( context , " crypto_scalarmult failed " ) ;
}
}
if ( JS_IsUndefined ( result ) )
{
if ( crypto_secretbox_easy ( p , ( const uint8_t * ) message , message_size , nonce , body_key ) ! = 0 )
{
result = JS_ThrowInternalError ( context , " crypto_secretbox_easy failed for the message. \n " ) ;
}
else
{
p + = crypto_secretbox_MACBYTES + message_size ;
assert ( ( size_t ) ( p - payload ) = = payload_size ) ;
char * encoded = tf_malloc ( payload_size * 2 + 5 ) ;
size_t encoded_length = tf_base64_encode ( payload , payload_size , encoded , payload_size * 2 + 5 ) ;
memcpy ( encoded + encoded_length , " .box " , 5 ) ;
encoded_length + = 4 ;
result = JS_NewStringLen ( context , encoded , encoded_length ) ;
tf_free ( encoded ) ;
}
}
tf_free ( payload ) ;
}
else
{
result = JS_ThrowInternalError ( context , " Unable to get key for ID %s of user %s. " , signer_identity , signer_user ) ;
}
}
JS_FreeCString ( context , signer_user ) ;
JS_FreeCString ( context , signer_identity ) ;
JS_FreeCString ( context , message ) ;
return result ;
}
static JSValue _tf_ssb_private_message_decrypt ( JSContext * context , JSValueConst this_val , int argc , JSValueConst * argv )
{
JSValue result = JS_UNDEFINED ;
const char * user = JS_ToCString ( context , argv [ 0 ] ) ;
const char * identity = JS_ToCString ( context , argv [ 1 ] ) ;
size_t message_size = 0 ;
const char * message = JS_ToCStringLen ( context , & message_size , argv [ 2 ] ) ;
uint8_t private_key [ crypto_sign_SECRETKEYBYTES ] = { 0 } ;
tf_ssb_t * ssb = JS_GetOpaque ( this_val , _tf_ssb_classId ) ;
2023-06-15 00:27:49 +00:00
sqlite3 * db = tf_ssb_acquire_db_reader ( ssb ) ;
2023-07-20 05:06:15 +00:00
bool found = _tf_ssb_get_private_key_curve25519 ( db , user , identity , private_key ) ;
tf_ssb_release_db_reader ( ssb , db ) ;
if ( found )
2023-02-26 19:51:54 +00:00
{
uint8_t * decoded = tf_malloc ( message_size ) ;
int decoded_length = tf_base64_decode ( message , message_size - strlen ( " .box " ) , decoded , message_size ) ;
uint8_t * nonce = decoded ;
uint8_t * public_key = decoded + crypto_box_NONCEBYTES ;
if ( public_key + crypto_secretbox_KEYBYTES < decoded + decoded_length )
{
uint8_t shared_secret [ crypto_secretbox_KEYBYTES ] = { 0 } ;
if ( crypto_scalarmult ( shared_secret , private_key , public_key ) = = 0 )
{
2024-02-15 23:35:01 +00:00
enum
{
k_recipient_header_bytes = crypto_secretbox_MACBYTES + sizeof ( uint8_t ) + crypto_secretbox_KEYBYTES
} ;
for ( uint8_t * p = decoded + crypto_box_NONCEBYTES + crypto_secretbox_KEYBYTES ; p < = decoded + decoded_length - k_recipient_header_bytes ;
2024-02-17 19:22:02 +00:00
p + = k_recipient_header_bytes )
2023-02-26 19:51:54 +00:00
{
uint8_t out [ k_recipient_header_bytes ] = { 0 } ;
int opened = crypto_secretbox_open_easy ( out , p , k_recipient_header_bytes , nonce , shared_secret ) ;
if ( opened ! = - 1 )
{
int recipients = ( int ) out [ 0 ] ;
uint8_t * body = decoded + crypto_box_NONCEBYTES + crypto_secretbox_KEYBYTES + k_recipient_header_bytes * recipients ;
size_t body_size = decoded + decoded_length - body ;
uint8_t * decrypted = tf_malloc ( body_size ) ;
uint8_t * key = out + 1 ;
if ( crypto_secretbox_open_easy ( decrypted , body , body_size , nonce , key ) ! = - 1 )
{
result = JS_NewStringLen ( context , ( const char * ) decrypted , body_size - crypto_secretbox_MACBYTES ) ;
}
else
{
result = JS_ThrowInternalError ( context , " Received key to open secret box containing message body, but it did not work. " ) ;
}
tf_free ( decrypted ) ;
}
}
}
else
{
result = JS_ThrowInternalError ( context , " crypto_scalarmult failed. " ) ;
}
}
else
{
result = JS_ThrowInternalError ( context , " Encrypted message was not long enough to contains its one-time public key. " ) ;
}
tf_free ( decoded ) ;
}
else
{
result = JS_ThrowInternalError ( context , " Private key not found for user %s with id %s. " , user , identity ) ;
}
JS_FreeCString ( context , user ) ;
JS_FreeCString ( context , identity ) ;
JS_FreeCString ( context , message ) ;
return result ;
}
2023-10-29 19:05:32 +00:00
typedef struct _following_t
{
uv_work_t work ;
tf_ssb_t * ssb ;
JSContext * context ;
JSValue promise [ 2 ] ;
2023-11-03 00:45:30 +00:00
tf_ssb_following_t * out_following ;
2023-10-29 19:05:32 +00:00
int depth ;
int ids_count ;
const char * ids [ ] ;
} following_t ;
static void _tf_ssb_following_work ( uv_work_t * work )
{
following_t * following = work - > data ;
2023-12-30 20:34:35 +00:00
tf_ssb_record_thread_busy ( following - > ssb , true ) ;
2023-11-03 00:45:30 +00:00
following - > out_following = tf_ssb_db_following_deep ( following - > ssb , following - > ids , following - > ids_count , following - > depth ) ;
2023-12-30 20:34:35 +00:00
tf_ssb_record_thread_busy ( following - > ssb , false ) ;
2023-10-29 19:05:32 +00:00
}
static void _tf_ssb_following_after_work ( uv_work_t * work , int status )
{
following_t * following = work - > data ;
JSContext * context = following - > context ;
if ( status = = 0 )
{
2023-11-03 00:45:30 +00:00
JSValue object = JS_NewObject ( context ) ;
for ( int i = 0 ; * following - > out_following [ i ] . id ; i + + )
2023-10-29 19:05:32 +00:00
{
2023-11-03 00:45:30 +00:00
JSValue entry = JS_NewObject ( context ) ;
JS_SetPropertyStr ( context , entry , " of " , JS_NewInt32 ( context , following - > out_following [ i ] . following_count ) ) ;
JS_SetPropertyStr ( context , entry , " ob " , JS_NewInt32 ( context , following - > out_following [ i ] . blocking_count ) ) ;
JS_SetPropertyStr ( context , entry , " if " , JS_NewInt32 ( context , following - > out_following [ i ] . followed_by_count ) ) ;
JS_SetPropertyStr ( context , entry , " ib " , JS_NewInt32 ( context , following - > out_following [ i ] . blocked_by_count ) ) ;
JS_SetPropertyStr ( context , object , following - > out_following [ i ] . id , entry ) ;
2023-10-29 19:05:32 +00:00
}
2024-01-08 02:30:08 +00:00
JSValue result = JS_Call ( context , following - > promise [ 0 ] , JS_UNDEFINED , 1 , & object ) ;
tf_util_report_error ( context , result ) ;
JS_FreeValue ( context , result ) ;
2023-11-03 00:45:30 +00:00
JS_FreeValue ( context , object ) ;
2023-10-29 19:05:32 +00:00
}
else
{
char buffer [ 256 ] ;
uv_strerror_r ( status , buffer , sizeof ( buffer ) ) ;
JSValue message = JS_NewString ( context , buffer ) ;
2024-01-08 02:30:08 +00:00
JSValue result = JS_Call ( context , following - > promise [ 1 ] , JS_UNDEFINED , 1 , & message ) ;
tf_util_report_error ( context , result ) ;
JS_FreeValue ( context , result ) ;
2023-10-29 19:05:32 +00:00
JS_FreeValue ( context , message ) ;
}
2024-01-10 02:49:44 +00:00
JS_FreeValue ( context , following - > promise [ 0 ] ) ;
JS_FreeValue ( context , following - > promise [ 1 ] ) ;
2023-10-29 19:05:32 +00:00
for ( int i = 0 ; i < following - > ids_count ; i + + )
{
tf_free ( ( void * ) following - > ids [ i ] ) ;
}
2023-11-03 00:45:30 +00:00
tf_free ( following - > out_following ) ;
2023-10-29 19:05:32 +00:00
tf_free ( following ) ;
}
static JSValue _tf_ssb_following ( JSContext * context , JSValueConst this_val , int argc , JSValueConst * argv )
{
tf_ssb_t * ssb = JS_GetOpaque ( this_val , _tf_ssb_classId ) ;
int ids_count = tf_util_get_length ( context , argv [ 0 ] ) ;
following_t * following = tf_malloc ( sizeof ( following_t ) + sizeof ( char * ) * ids_count ) ;
* following = ( following_t )
{
. work =
{
. data = following ,
} ,
. context = context ,
. ssb = ssb ,
} ;
JS_ToInt32 ( context , & following - > depth , argv [ 1 ] ) ;
JSValue result = JS_NewPromiseCapability ( context , following - > promise ) ;
for ( int i = 0 ; i < ids_count ; i + + )
{
JSValue id_value = JS_GetPropertyUint32 ( context , argv [ 0 ] , i ) ;
if ( ! JS_IsUndefined ( id_value ) )
{
const char * id_string = JS_ToCString ( context , id_value ) ;
following - > ids [ following - > ids_count + + ] = tf_strdup ( id_string ) ;
JS_FreeCString ( context , id_string ) ;
JS_FreeValue ( context , id_value ) ;
}
}
int r = uv_queue_work ( tf_ssb_get_loop ( ssb ) , & following - > work , _tf_ssb_following_work , _tf_ssb_following_after_work ) ;
if ( r )
{
_tf_ssb_following_after_work ( & following - > work , r ) ;
}
return result ;
}
2021-10-24 15:46:30 +00:00
void tf_ssb_register ( JSContext * context , tf_ssb_t * ssb )
2021-01-02 18:10:00 +00:00
{
JS_NewClassID ( & _tf_ssb_classId ) ;
2024-02-15 23:35:01 +00:00
JSClassDef def = {
2021-01-02 18:10:00 +00:00
. class_name = " ssb " ,
} ;
2021-10-10 21:51:38 +00:00
if ( JS_NewClass ( JS_GetRuntime ( context ) , _tf_ssb_classId , & def ) ! = 0 )
{
2021-01-02 18:10:00 +00:00
fprintf ( stderr , " Failed to register ssb. \n " ) ;
}
JSValue global = JS_GetGlobalObject ( context ) ;
JSValue object = JS_NewObjectClass ( context , _tf_ssb_classId ) ;
JS_SetPropertyStr ( context , global , " ssb " , object ) ;
JS_SetOpaque ( object , ssb ) ;
2022-07-31 19:01:08 +00:00
/* Requires an identity. */
2022-07-14 01:01:14 +00:00
JS_SetPropertyStr ( context , object , " createIdentity " , JS_NewCFunction ( context , _tf_ssb_createIdentity , " createIdentity " , 1 ) ) ;
2024-01-04 01:17:30 +00:00
JS_SetPropertyStr ( context , object , " addIdentity " , JS_NewCFunction ( context , _tf_ssb_addIdentity , " addIdentity " , 2 ) ) ;
2024-01-06 19:22:49 +00:00
JS_SetPropertyStr ( context , object , " deleteIdentity " , JS_NewCFunction ( context , _tf_ssb_deleteIdentity , " deleteIdentity " , 2 ) ) ;
2023-10-20 14:37:24 +00:00
JS_SetPropertyStr ( context , object , " setServerFollowingMe " , JS_NewCFunction ( context , _tf_ssb_set_server_following_me , " setServerFollowingMe " , 3 ) ) ;
2022-07-14 01:01:14 +00:00
JS_SetPropertyStr ( context , object , " getIdentities " , JS_NewCFunction ( context , _tf_ssb_getIdentities , " getIdentities " , 1 ) ) ;
2024-01-04 00:21:15 +00:00
JS_SetPropertyStr ( context , object , " getPrivateKey " , JS_NewCFunction ( context , _tf_ssb_getPrivateKey , " getPrivateKey " , 2 ) ) ;
2023-02-26 19:51:54 +00:00
JS_SetPropertyStr ( context , object , " privateMessageEncrypt " , JS_NewCFunction ( context , _tf_ssb_private_message_encrypt , " privateMessageEncrypt " , 4 ) ) ;
JS_SetPropertyStr ( context , object , " privateMessageDecrypt " , JS_NewCFunction ( context , _tf_ssb_private_message_decrypt , " privateMessageDecrypt " , 3 ) ) ;
2023-07-18 23:46:15 +00:00
/* Write. */
JS_SetPropertyStr ( context , object , " appendMessageWithIdentity " , JS_NewCFunction ( context , _tf_ssb_appendMessageWithIdentity , " appendMessageWithIdentity " , 3 ) ) ;
2022-07-14 01:01:14 +00:00
2022-07-31 19:01:08 +00:00
/* Does not require an identity. */
2023-10-20 12:55:05 +00:00
JS_SetPropertyStr ( context , object , " getServerIdentity " , JS_NewCFunction ( context , _tf_ssb_getServerIdentity , " getServerIdentity " , 0 ) ) ;
2022-07-31 19:01:08 +00:00
JS_SetPropertyStr ( context , object , " getAllIdentities " , JS_NewCFunction ( context , _tf_ssb_getAllIdentities , " getAllIdentities " , 0 ) ) ;
2024-05-05 12:55:32 -04:00
JS_SetPropertyStr ( context , object , " getActiveIdentity " , JS_NewCFunction ( context , _tf_ssb_getActiveIdentity , " getActiveIdentity " , 3 ) ) ;
2024-05-05 13:48:22 -04:00
JS_SetPropertyStr ( context , object , " getIdentityInfo " , JS_NewCFunction ( context , _tf_ssb_getIdentityInfo , " getIdentityInfo " , 3 ) ) ;
2021-01-02 18:10:00 +00:00
JS_SetPropertyStr ( context , object , " getMessage " , JS_NewCFunction ( context , _tf_ssb_getMessage , " getMessage " , 2 ) ) ;
JS_SetPropertyStr ( context , object , " blobGet " , JS_NewCFunction ( context , _tf_ssb_blobGet , " blobGet " , 1 ) ) ;
2021-01-09 23:06:33 +00:00
JS_SetPropertyStr ( context , object , " messageContentGet " , JS_NewCFunction ( context , _tf_ssb_messageContentGet , " messageContentGet " , 1 ) ) ;
2021-01-02 18:10:00 +00:00
JS_SetPropertyStr ( context , object , " connections " , JS_NewCFunction ( context , _tf_ssb_connections , " connections " , 0 ) ) ;
2023-01-18 00:37:45 +00:00
JS_SetPropertyStr ( context , object , " storedConnections " , JS_NewCFunction ( context , _tf_ssb_storedConnections , " storedConnections " , 0 ) ) ;
2022-11-02 23:34:44 +00:00
JS_SetPropertyStr ( context , object , " getConnection " , JS_NewCFunction ( context , _tf_ssb_getConnection , " getConnection " , 1 ) ) ;
2022-11-09 01:31:18 +00:00
JS_SetPropertyStr ( context , object , " closeConnection " , JS_NewCFunction ( context , _tf_ssb_closeConnection , " closeConnection " , 1 ) ) ;
2023-01-18 00:37:45 +00:00
JS_SetPropertyStr ( context , object , " forgetStoredConnection " , JS_NewCFunction ( context , _tf_ssb_forgetStoredConnection , " forgetStoredConnection " , 1 ) ) ;
2023-02-08 01:29:44 +00:00
JS_SetPropertyStr ( context , object , " sqlAsync " , JS_NewCFunction ( context , _tf_ssb_sqlAsync , " sqlAsync " , 3 ) ) ;
2021-01-02 18:10:00 +00:00
JS_SetPropertyStr ( context , object , " getBroadcasts " , JS_NewCFunction ( context , _tf_ssb_getBroadcasts , " getBroadcasts " , 0 ) ) ;
JS_SetPropertyStr ( context , object , " connect " , JS_NewCFunction ( context , _tf_ssb_connect , " connect " , 1 ) ) ;
2022-12-01 01:19:35 +00:00
JS_SetPropertyStr ( context , object , " createTunnel " , JS_NewCFunction ( context , _tf_ssb_createTunnel , " createTunnel " , 3 ) ) ;
2023-10-29 19:05:32 +00:00
JS_SetPropertyStr ( context , object , " following " , JS_NewCFunction ( context , _tf_ssb_following , " following " , 2 ) ) ;
2023-07-18 23:46:15 +00:00
/* Write. */
JS_SetPropertyStr ( context , object , " storeMessage " , JS_NewCFunction ( context , _tf_ssb_storeMessage , " storeMessage " , 1 ) ) ;
2023-09-07 22:44:18 +00:00
JS_SetPropertyStr ( context , object , " blobStore " , JS_NewCFunction ( context , _tf_ssb_blobStore , " blobStore " , 1 ) ) ;
2021-11-07 22:28:58 +00:00
2022-07-31 19:01:08 +00:00
/* Should be trusted only. */
2021-11-07 22:28:58 +00:00
JS_SetPropertyStr ( context , object , " addEventListener " , JS_NewCFunction ( context , _tf_ssb_add_event_listener , " addEventListener " , 2 ) ) ;
JS_SetPropertyStr ( context , object , " removeEventListener " , JS_NewCFunction ( context , _tf_ssb_remove_event_listener , " removeEventListener " , 2 ) ) ;
2021-09-06 17:50:38 +00:00
2021-01-02 18:10:00 +00:00
JS_FreeValue ( context , global ) ;
}