2021-12-28 15:25:04 +00:00
"use strict" ;
2021-09-06 17:50:38 +00:00
var g _wants _requests = { } ;
2021-11-03 23:23:20 +00:00
var g _database = new Database ( 'core' ) ;
2022-11-02 23:34:44 +00:00
let g _attendants = { } ;
2021-12-27 19:52:42 +00:00
const k _use _create _history _stream = false ;
2021-12-28 15:25:04 +00:00
const k _blobs _concurrent _target = 8 ;
2021-09-06 17:50:38 +00:00
2021-12-27 19:52:42 +00:00
function following ( db , id ) {
var o = db . get ( id + ":following" ) ;
2021-09-09 00:15:57 +00:00
const k _version = 5 ;
var f = o ? JSON . parse ( o ) : o ;
if ( ! f || f . version != k _version ) {
f = { users : [ ] , sequence : 0 , version : k _version } ;
}
f . users = new Set ( f . users ) ;
2021-12-27 19:52:42 +00:00
ssb . sqlStream (
2021-09-09 00:15:57 +00:00
"SELECT " +
" sequence, " +
" json_extract(content, '$.contact') AS contact, " +
" json_extract(content, '$.following') AS following " +
"FROM messages " +
"WHERE " +
" author = ?1 AND " +
" sequence > ?2 AND " +
" json_extract(content, '$.type') = 'contact' " +
"UNION SELECT MAX(sequence) AS sequence, NULL, NULL FROM messages WHERE author = ?1 " +
"ORDER BY sequence" ,
[ id , f . sequence ] ,
function ( row ) {
if ( row . following ) {
f . users . add ( row . contact ) ;
} else {
f . users . delete ( row . contact ) ;
}
f . sequence = row . sequence ;
} ) ;
2022-02-06 03:28:29 +00:00
f . users = Array . from ( f . users ) . sort ( ) ;
2021-09-09 00:15:57 +00:00
var j = JSON . stringify ( f ) ;
if ( o != j ) {
2021-12-27 19:52:42 +00:00
db . set ( id + ":following" , j ) ;
2021-09-09 00:15:57 +00:00
}
return f . users ;
}
2021-12-27 19:52:42 +00:00
function followingDeep ( db , seed _ids , depth ) {
2021-09-09 00:15:57 +00:00
if ( depth <= 0 ) {
return seed _ids ;
}
2021-12-27 19:52:42 +00:00
var f = seed _ids . map ( x => following ( db , x ) ) ;
2021-09-09 00:15:57 +00:00
var ids = [ ] . concat ( ... f ) ;
2021-12-27 19:52:42 +00:00
var x = followingDeep ( db , [ ... new Set ( ids ) ] . sort ( ) , depth - 1 ) ;
2021-11-06 02:10:13 +00:00
x = [ ... new Set ( [ ] . concat ( ... x , ... seed _ids ) ) ] . sort ( ) ;
2021-09-09 00:15:57 +00:00
return x ;
}
2021-10-31 19:39:16 +00:00
function get _latest _sequence _for _author ( author ) {
2021-10-30 21:07:01 +00:00
var sequence = 0 ;
2021-10-31 19:39:16 +00:00
ssb . sqlStream (
2022-02-06 03:28:29 +00:00
'SELECT MAX(sequence) AS sequence FROM messages WHERE author = ?1' ,
2021-10-30 21:07:01 +00:00
[ author ] ,
function ( row ) {
2021-10-31 19:39:16 +00:00
if ( row . sequence ) {
2021-12-27 19:52:42 +00:00
sequence = row . sequence ;
2021-10-31 19:39:16 +00:00
}
2021-10-30 21:07:01 +00:00
} ) ;
return sequence ;
}
2021-12-22 19:57:34 +00:00
function storeMessage ( message ) {
var payload = message . message . value ? message . message . value : message . message ;
if ( typeof ( payload ) == 'object' ) {
ssb . storeMessage ( payload ) ;
}
}
2022-11-02 23:34:44 +00:00
function tunnel _attendants ( request ) {
if ( request . message . type !== 'state' ) {
throw Error ( 'Unexpected type: ' + request . message . type ) ;
}
let state = new Set ( request . message . ids ) ;
for ( let id of state ) {
request . add _room _attendant ( id ) ;
}
request . more ( function attendants ( message ) {
if ( message . message . type === 'joined' ) {
request . add _room _attendant ( message . message . id ) ;
state . add ( message . message . id ) ;
} else if ( message . message . type === 'left' ) {
request . remove _room _attendant ( message . message . id ) ;
state . delete ( message . message . id ) ;
} else {
throw Error ( 'Unexpected type: ' + message . type ) ;
}
} ) ;
}
ssb . addEventListener ( 'connections' , function on _connections _changed ( change , connection ) {
2021-09-06 17:50:38 +00:00
if ( change == 'add' ) {
2021-10-31 19:39:16 +00:00
var sequence = get _latest _sequence _for _author ( connection . id ) ;
2021-12-27 19:52:42 +00:00
if ( k _use _create _history _stream ) {
connection . send _json ( { 'name' : [ 'createHistoryStream' ] , 'type' : 'source' , 'args' : [ { 'id' : connection . id , 'seq' : sequence , 'live' : true , 'keys' : false } ] } , storeMessage ) ;
2022-07-31 19:01:08 +00:00
var identities = ssb . getAllIdentities ( ) ;
followingDeep ( g _database , identities , 2 ) . then ( function ( ids ) {
2021-12-27 19:52:42 +00:00
for ( let id of ids ) {
2022-07-31 19:01:08 +00:00
if ( identities . indexOf ( id ) != - 1 ) {
2021-12-27 19:52:42 +00:00
continue ;
}
var sequence = get _latest _sequence _for _author ( id ) ;
connection . send _json ( { 'name' : [ 'createHistoryStream' ] , 'type' : 'source' , 'args' : [ { 'id' : id , 'seq' : sequence , 'live' : true , 'keys' : false } ] } , storeMessage ) ;
}
} ) ;
} else {
if ( connection . is _client ) {
2022-11-09 23:25:22 +00:00
connection . send _json ( { 'name' : [ 'tunnel' , 'isRoom' ] , 'args' : [ ] } , function tunnel _is _room ( request ) {
2022-11-02 23:34:44 +00:00
if ( request . message ) {
connection . send _json ( { 'name' : [ 'room' , 'attendants' ] , 'args' : [ ] , 'type' : 'source' } , tunnel _attendants ) ;
}
} ) ;
2021-12-27 19:52:42 +00:00
connection . send _json ( { "name" : [ "ebt" , "replicate" ] , "args" : [ { "version" : 3 , "format" : "classic" } ] , "type" : "duplex" } , ebtReplicateClient ) ;
}
}
2021-12-28 15:25:04 +00:00
connection . active _blob _wants = { } ;
2022-11-02 23:34:44 +00:00
connection . send _json ( { 'name' : [ 'blobs' , 'createWants' ] , 'type' : 'source' , 'args' : [ ] } , function on _blob _create _wants ( message ) {
2021-09-06 17:50:38 +00:00
Object . keys ( message . message ) . forEach ( function ( id ) {
if ( message . message [ id ] < 0 ) {
2022-02-02 02:13:38 +00:00
if ( g _wants _requests [ connection . id ] ) {
delete connection . active _blob _wants [ id ] ;
var blob = ssb . blobGet ( id ) ;
if ( blob ) {
var out _message = { } ;
out _message [ id ] = blob . byteLength ;
g _wants _requests [ connection . id ] . send _json ( out _message ) ;
}
2021-09-06 17:50:38 +00:00
}
} else {
2021-10-31 21:15:18 +00:00
var received _bytes = 0 ;
var expected _bytes = message . message [ id ] ;
var buffer = new Uint8Array ( expected _bytes ) ;
2021-10-30 21:07:01 +00:00
connection . send _json ( { 'name' : [ 'blobs' , 'get' ] , 'type' : 'source' , 'args' : [ id ] } , function ( message ) {
2021-12-28 15:25:04 +00:00
if ( message . flags & 0x4 /* end */ ) {
delete connection . active _blob _wants [ id ] ;
} else {
buffer . set ( new Uint8Array ( message . message , 0 , message . message . byteLength ) , received _bytes ) ;
received _bytes += message . message . byteLength ;
if ( received _bytes == expected _bytes ) {
ssb . blobStore ( buffer ) ;
}
2021-10-31 21:15:18 +00:00
}
2021-09-06 17:50:38 +00:00
} ) ;
2021-12-28 15:25:04 +00:00
if ( Object . keys ( connection . active _blob _wants ) . length < k _blobs _concurrent _target ) {
requestMoreBlobs ( g _wants _requests [ connection . id ] ) ;
}
2021-09-06 17:50:38 +00:00
}
} ) ;
} ) ;
} else if ( change == 'remove' ) {
2021-11-03 22:15:46 +00:00
print ( 'REMOVE' , connection . id ) ;
2022-11-02 23:34:44 +00:00
notify _attendant _changed ( connection . id , 'left' ) ;
delete g _attendants [ connection . id ] ;
2021-09-06 17:50:38 +00:00
delete g _wants _requests [ connection . id ] ;
} else {
2021-11-03 22:15:46 +00:00
print ( 'CHANGE' , change ) ;
2021-09-06 17:50:38 +00:00
}
} ) ;
2021-12-28 15:25:04 +00:00
function blob _want _discovered ( request , id ) {
2022-11-09 23:25:22 +00:00
if ( ! request || ! request . connection || Object . keys ( request . connection . active _blob _wants ) . length > k _blobs _concurrent _target ) {
2021-12-28 15:25:04 +00:00
return ;
2021-09-06 17:50:38 +00:00
}
2021-12-28 15:25:04 +00:00
var message = { } ;
message [ id ] = - 1 ;
request . send _json ( message ) ;
request . connection . active _blob _wants [ id ] = true ;
}
function requestMoreBlobs ( request ) {
2021-09-06 17:50:38 +00:00
ssb . sqlStream (
2021-12-28 15:25:04 +00:00
'SELECT id FROM blob_wants LIMIT ' + k _blobs _concurrent _target ,
2021-09-06 17:50:38 +00:00
[ ] ,
2021-12-28 15:25:04 +00:00
row => blob _want _discovered ( request , row . id ) ) ;
}
2022-11-09 23:25:22 +00:00
ssb . addRpc ( [ 'manifest' ] , function ( request ) {
} ) ;
2021-12-28 15:25:04 +00:00
ssb . addRpc ( [ 'blobs' , 'createWants' ] , function ( request ) {
g _wants _requests [ request . connection . id ] = request ;
ssb . addEventListener ( 'blob_want_added' , id => blob _want _discovered ( request , id ) ) ;
requestMoreBlobs ( request ) ;
2021-09-06 17:50:38 +00:00
} ) ;
2021-11-07 22:28:58 +00:00
ssb . addRpc ( [ 'blobs' , 'has' ] , function ( request ) {
2021-09-06 17:50:38 +00:00
var found = false ;
ssb . sqlStream (
'SELECT 1 FROM blobs where id = ?1' ,
[ request . args [ 0 ] ] ,
function ( row ) {
found = true ;
} ) ;
request . send _json ( found ) ;
} ) ;
2021-11-07 22:28:58 +00:00
ssb . addRpc ( [ 'blobs' , 'get' ] , function ( request ) {
2021-12-28 21:48:03 +00:00
for ( let arg of request . args ) {
var blob ;
if ( arg . key ) {
blob = ssb . blobGet ( arg . key ) ;
} else {
blob = ssb . blobGet ( arg ) ;
}
const k _send _max = 8192 ;
if ( blob . byteLength > k _send _max ) {
for ( var i = 0 ; i < blob . byteLength ; i += k _send _max ) {
var buffer = new Uint8Array ( blob , i , Math . min ( blob . byteLength - i , k _send _max ) ) ;
request . send _binary ( buffer ) ;
}
} else {
request . send _binary ( blob ) ;
}
request . send _json _end ( true ) ;
2021-10-31 19:39:16 +00:00
}
2021-09-06 17:50:38 +00:00
} ) ;
2021-12-22 19:57:34 +00:00
ssb . addRpc ( [ 'gossip' , 'ping' ] , function ( request ) {
2022-11-02 23:34:44 +00:00
request . more ( function ping ( message ) {
2021-12-27 19:52:42 +00:00
message . send _json ( Date . now ( ) ) ;
2021-12-22 19:57:34 +00:00
} ) ;
} ) ;
2021-12-27 19:52:42 +00:00
ssb . addRpc ( [ 'tunnel' , 'isRoom' ] , function ( request ) {
2022-11-02 23:34:44 +00:00
request . send _json ( true ) ;
} ) ;
function notify _attendant _changed ( id , type ) {
for ( let r of Object . values ( g _attendants ) ) {
try {
r . send _json ( {
type : type ,
id : id ,
} ) ;
} catch ( e ) {
print ( ` Removing ${ r . connection . id } from g_attendants in ${ type } . ` , e ) ;
delete g _attendants [ r . connection . id ] ;
}
}
}
ssb . addRpc ( [ 'room' , 'attendants' ] , function ( request ) {
let ids = Object . keys ( g _attendants ) . sort ( ) ;
request . send _json ( {
type : 'state' ,
ids : ids ,
} ) ;
notify _attendant _changed ( request . connection . id , 'joined' ) ;
g _attendants [ request . connection . id ] = request ;
} ) ;
ssb . addRpc ( [ 'tunnel' , 'connect' ] , function ( request ) {
if ( ! request . args [ 0 ] . origin &&
request . args [ 0 ] . portal &&
request . args [ 0 ] . target ) {
let target _connection = ssb . getConnection ( request . args [ 0 ] . target ) ;
let target _request _number = target _connection . send _json ( {
'name' : [ 'tunnel' , 'connect' ] ,
'args' : [ {
'origin' : request . connection . id ,
'portal' : request . args [ 0 ] . portal ,
'target' : request . args [ 0 ] . target ,
} ] ,
'type' : 'duplex' ,
} ) ;
ssb . tunnel ( request . connection , - request . request _number , target _connection , target _request _number ) ;
} else if ( request . args [ 0 ] . origin &&
request . args [ 0 ] . portal &&
request . args [ 0 ] . target ) {
ssb . createTunnel ( request . connection , - request . request _number , request . args [ 0 ] . origin ) ;
}
2021-12-27 19:52:42 +00:00
} ) ;
function ebtReplicateSendClock ( request , have ) {
2022-07-31 19:01:08 +00:00
var identities = ssb . getAllIdentities ( ) ;
2021-12-27 19:52:42 +00:00
var message = { } ;
2022-01-22 20:13:14 +00:00
var last _sent = request . connection . sent _clock || { } ;
2022-07-31 19:01:08 +00:00
var ids = followingDeep ( g _database , identities , 2 ) . concat ( [ request . connection . id ] ) ;
2022-01-22 20:47:10 +00:00
if ( ! Object . keys ( last _sent ) . length ) {
2022-01-22 20:13:14 +00:00
for ( let id of ids ) {
2022-02-06 03:49:47 +00:00
message [ id ] = get _latest _sequence _for _author ( id ) ;
2022-01-22 20:13:14 +00:00
}
2021-12-27 19:52:42 +00:00
}
2021-12-28 15:25:04 +00:00
for ( let id of Object . keys ( have ) ) {
if ( message [ id ] === undefined ) {
var sequence = get _latest _sequence _for _author ( id ) ;
message [ id ] = sequence ? sequence : - 1 ;
}
}
2021-12-27 19:52:42 +00:00
var to _send = { }
2022-02-11 02:44:27 +00:00
var offset = Math . floor ( Math . random ( ) * ids . length ) ;
for ( var i = 0 ; i < ids . length ; i ++ ) {
var id = ids [ ( i + offset ) % ids . length ] ;
2021-12-27 19:52:42 +00:00
if ( last _sent [ id ] === undefined || message [ id ] > last _sent [ id ] ) {
last _sent [ id ] = to _send [ id ] = message [ id ] === - 1 ? - 1 : message [ id ] << 1 ;
}
2021-12-28 15:25:04 +00:00
if ( Object . keys ( to _send ) . length >= 32 ) {
request . send _json ( to _send ) ;
to _send = { } ;
}
2021-12-27 19:52:42 +00:00
}
request . connection . sent _clock = last _sent ;
if ( Object . keys ( to _send ) . length ) {
request . send _json ( to _send ) ;
}
}
function formatMessage ( row ) {
2022-02-03 02:00:05 +00:00
if ( row . sequence _before _author ) {
return {
previous : row . previous ,
sequence : row . sequence ,
author : row . author ,
timestamp : row . timestamp ,
hash : row . hash ,
content : JSON . parse ( row . content ) ,
signature : row . signature ,
} ;
} else {
return {
previous : row . previous ,
author : row . author ,
sequence : row . sequence ,
timestamp : row . timestamp ,
hash : row . hash ,
content : JSON . parse ( row . content ) ,
signature : row . signature ,
} ;
}
2021-12-27 19:52:42 +00:00
}
function ebtReplicateRegisterMessageCallback ( request ) {
ssb . addEventListener ( 'message' , function ( message _id ) {
ssb . sqlStream (
2022-07-31 19:01:08 +00:00
'SELECT previous, author, id, sequence, timestamp, hash, content, signature FROM messages WHERE id = ?1' ,
[ message _id ] ,
2021-12-27 19:52:42 +00:00
function ( row ) {
request . send _json ( formatMessage ( row ) ) ;
} ) ;
} ) ;
}
function ebtReplicateCommon ( request ) {
if ( request . message . author ) {
storeMessage ( request ) ;
} else {
ebtReplicateSendClock ( request , request . message ) ;
for ( let id of Object . keys ( request . message ) ) {
if ( request . message [ id ] >= 0 && ( request . message [ id ] & 1 ) == 0 ) {
ssb . sqlStream (
'SELECT previous, author, id, sequence, timestamp, hash, content, signature FROM messages WHERE author = ?1 AND sequence >= ?2 ORDER BY sequence' ,
[ id , request . message [ id ] >> 1 ] ,
function ( row ) {
request . send _json ( formatMessage ( row ) ) ;
} ) ;
}
}
}
}
function ebtReplicateClient ( request ) {
if ( ! request . connection . message _registered ) {
ebtReplicateRegisterMessageCallback ( request ) ;
request . connection . message _registered = true ;
}
ebtReplicateCommon ( request ) ;
}
function ebtReplicateServer ( request ) {
ebtReplicateRegisterMessageCallback ( request ) ;
ebtReplicateSendClock ( request , { } ) ;
request . more ( ebtReplicateCommon ) ;
}
ssb . addRpc ( [ 'ebt' , 'replicate' ] , ebtReplicateServer ) ;
2021-11-07 22:28:58 +00:00
ssb . addRpc ( [ 'createHistoryStream' ] , function ( request ) {
2021-09-06 17:50:38 +00:00
var id = request . args [ 0 ] . id ;
var seq = request . args [ 0 ] . seq ;
2021-11-06 02:10:13 +00:00
var keys = request . args [ 0 ] . keys || request . args [ 0 ] . keys === undefined ;
2021-11-11 00:05:07 +00:00
function sendMessage ( row ) {
if ( keys ) {
var message = {
key : row . id ,
2022-02-03 02:00:05 +00:00
value : formatMessage ( row ) ,
2021-11-11 00:05:07 +00:00
timestamp : row . timestamp ,
} ;
} else {
2022-02-03 02:00:05 +00:00
var message = formatMessage ( row ) ;
2021-11-11 00:05:07 +00:00
}
request . send _json ( message ) ;
}
ssb . sqlStream (
2022-02-03 02:00:05 +00:00
'SELECT previous, author, id, sequence, timestamp, hash, content, signature, sequence_before_author FROM messages WHERE author = ?1 AND sequence >= ?2 ORDER BY sequence' ,
2021-11-11 00:05:07 +00:00
[ id , seq ? ? 0 ] ,
sendMessage ) ;
2021-12-20 17:00:25 +00:00
ssb . addEventListener ( 'message' , function ( message _id ) {
2021-11-11 00:05:07 +00:00
ssb . sqlStream (
2022-02-03 02:00:05 +00:00
'SELECT previous, author, id, sequence, timestamp, hash, content, signature, sequence_before_author FROM messages WHERE id = ?1 AND author = ?2' ,
2021-12-27 19:52:42 +00:00
[ message _id , id ] ,
2021-12-20 17:00:25 +00:00
function ( row ) {
2021-12-27 19:52:42 +00:00
sendMessage ( row ) ;
2021-12-20 17:00:25 +00:00
} ) ;
2021-11-11 00:05:07 +00:00
} ) ;
2021-09-06 17:50:38 +00:00
} ) ;