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-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 ) ;
}
} ) ;
}
2023-01-02 00:33:11 +00:00
function get _more _blobs ( connection ) {
while ( Object . keys ( connection . active _blob _gets ) . length < k _blobs _concurrent _target ) {
let next = Object . keys ( connection . blob _get _queue ) . pop ( ) ;
let expected _bytes = connection . blob _get _queue [ next ] ;
if ( ! next ) {
break ;
}
delete connection . blob _get _queue [ next ] ;
connection . active _blob _gets [ next ] = true ;
let received _bytes = 0 ;
let buffer = new Uint8Array ( expected _bytes ) ;
connection . send _json ( { 'name' : [ 'blobs' , 'get' ] , 'type' : 'source' , 'args' : [ next ] } , function ( message ) {
if ( message . flags & 0x4 /* end */ ) {
delete connection . active _blob _gets [ next ] ;
} 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 ) ;
delete connection . active _blob _gets [ next ] ;
get _more _blobs ( connection ) ;
}
}
} ) ;
}
}
2022-11-12 02:00:49 +00:00
function send _blobs _create _wants ( connection ) {
connection . send _json ( { 'name' : [ 'blobs' , 'createWants' ] , 'type' : 'source' , 'args' : [ ] } , function on _blob _create _wants ( message ) {
2022-11-13 22:02:54 +00:00
if ( message . message ? . name === 'Error' ) {
return ;
}
2022-11-12 02:00:49 +00:00
Object . keys ( message . message ) . forEach ( function ( id ) {
if ( message . message [ id ] < 0 ) {
if ( g _wants _requests [ connection . id ] ) {
2023-01-02 00:33:11 +00:00
let blob = ssb . blobGet ( id ) ;
2022-11-12 02:00:49 +00:00
if ( blob ) {
2023-01-02 00:33:11 +00:00
let out _message = { } ;
2022-11-12 02:00:49 +00:00
out _message [ id ] = blob . byteLength ;
g _wants _requests [ connection . id ] . send _json ( out _message ) ;
}
}
} else {
2023-01-02 00:33:11 +00:00
let expected _bytes = message . message [ id ] ;
connection . blob _get _queue [ id ] = expected _bytes ;
get _more _blobs ( connection ) ;
2022-11-12 02:00:49 +00:00
}
} ) ;
} ) ;
}
2022-11-02 23:34:44 +00:00
ssb . addEventListener ( 'connections' , function on _connections _changed ( change , connection ) {
2021-09-06 17:50:38 +00:00
if ( change == 'add' ) {
2023-01-02 00:33:11 +00:00
connection . active _blob _gets = { } ;
connection . blob _get _queue = { } ;
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 ( ) ;
2022-12-31 21:44:48 +00:00
ssb . followingDeep ( 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-12 02:00:49 +00:00
connection . send _json ( { "name" : [ "ebt" , "replicate" ] , "args" : [ { "version" : 3 , "format" : "classic" } ] , "type" : "duplex" } , ebtReplicateClient ) ;
connection . send _json _async ( { 'name' : [ 'tunnel' , 'isRoom' ] , 'args' : [ ] } , function tunnel _is _room ( request ) {
2022-11-02 23:34:44 +00:00
if ( request . message ) {
2022-11-12 02:00:49 +00:00
request . connection . send _json ( { 'name' : [ 'room' , 'attendants' ] , 'args' : [ ] , 'type' : 'source' } , tunnel _attendants ) ;
2022-11-02 23:34:44 +00:00
}
} ) ;
2021-12-27 19:52:42 +00:00
}
2022-11-12 02:00:49 +00:00
send _blobs _create _wants ( connection ) ;
2021-12-27 19:52:42 +00:00
}
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 ) {
2023-01-02 00:33:11 +00:00
if ( ! request || ! request . connection ) {
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 ) ;
}
2023-01-02 00:33:11 +00:00
function sleep ( ms ) {
return new Promise ( function ( resolve , reject ) {
setTimeout ( x => resolve ( ) , ms ) ;
} ) ;
}
async function requestMoreBlobs ( request ) {
let last = '' ;
while ( true ) {
await ssb . sqlStream (
2023-01-02 00:35:37 +00:00
'SELECT id FROM blob_wants WHERE id > ? ORDER BY id LIMIT 32' ,
2023-01-02 00:33:11 +00:00
[ last ] ,
function ( row ) {
blob _want _discovered ( request , row . id ) ;
last = row . id ;
} ) ;
await sleep ( 1000 ) ;
}
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
} ) ;
2022-11-02 23:34:44 +00:00
function notify _attendant _changed ( id , type ) {
2022-11-13 22:02:54 +00:00
if ( ! id ) {
print ( ` notify_attendant_changed called with id= ${ id } ` ) ;
2022-11-16 00:11:03 +00:00
return ;
2022-11-13 22:02:54 +00:00
}
2022-11-02 23:34:44 +00:00
for ( let r of Object . values ( g _attendants ) ) {
try {
r . send _json ( {
type : type ,
id : id ,
} ) ;
} catch ( e ) {
2022-11-13 22:02:54 +00:00
print ( ` Removing ${ id } from g_attendants in ${ type } . ` , e ) ;
delete g _attendants [ id ] ;
2022-11-02 23:34:44 +00:00
}
}
}
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 ;
} ) ;
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-12-31 21:44:48 +00:00
var ids = ssb . followingDeep ( 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 ) {
2022-12-31 18:59:29 +00:00
if ( request . connection . send _clock ) {
ssb . sqlStream (
'SELECT previous, author, id, sequence, timestamp, hash, content, signature FROM messages WHERE id = ?1' ,
[ message _id ] ,
function ( row ) {
if ( request . connection . send _clock [ row . author ] < row . sequence ) {
request . send _json ( formatMessage ( row ) ) ;
}
} ) ;
}
2021-12-27 19:52:42 +00:00
} ) ;
}
function ebtReplicateCommon ( request ) {
if ( request . message . author ) {
storeMessage ( request ) ;
} else {
ebtReplicateSendClock ( request , request . message ) ;
2022-12-31 18:59:29 +00:00
if ( ! request . connection . send _clock ) {
request . connection . send _clock = { } ;
}
2021-12-27 19:52:42 +00:00
for ( let id of Object . keys ( request . message ) ) {
if ( request . message [ id ] >= 0 && ( request . message [ id ] & 1 ) == 0 ) {
2022-12-31 18:59:29 +00:00
request . connection . send _clock [ id ] = request . message [ id ] >> 1 ;
2021-12-27 19:52:42 +00:00
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 ) ) ;
2022-12-31 18:59:29 +00:00
request . connection . send _clock [ id ] = row . sequence ;
2021-12-27 19:52:42 +00:00
} ) ;
2022-12-31 18:59:29 +00:00
} else {
delete request . connection . send _clock [ id ] ;
2021-12-27 19:52:42 +00:00
}
}
}
}
function ebtReplicateClient ( request ) {
2022-11-12 02:00:49 +00:00
if ( request . message ? . name !== 'Error' ) {
if ( ! request . connection . message _registered ) {
ebtReplicateRegisterMessageCallback ( request ) ;
request . connection . message _registered = true ;
}
ebtReplicateCommon ( request ) ;
2021-12-27 19:52:42 +00:00
}
}
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
} ) ;