2023-09-02 18:15:38 +00:00
#![ allow(non_snake_case) ]
use super ::* ;
#[ wasm_bindgen() ]
pub struct VeilidRoutingContext {
2023-09-11 05:25:37 +00:00
inner_routing_context : RoutingContext ,
2023-09-02 18:15:38 +00:00
}
#[ wasm_bindgen() ]
impl VeilidRoutingContext {
2023-09-11 05:25:37 +00:00
/// Create a new VeilidRoutingContext, without any privacy or sequencing settings.
2023-09-02 18:15:38 +00:00
#[ wasm_bindgen(constructor) ]
2023-09-11 05:25:37 +00:00
pub fn new ( ) -> APIResult < VeilidRoutingContext > {
2023-09-02 18:15:38 +00:00
let veilid_api = get_veilid_api ( ) ? ;
2023-09-11 05:25:37 +00:00
APIResult ::Ok ( VeilidRoutingContext {
inner_routing_context : veilid_api . routing_context ( ) ,
2023-09-04 21:34:49 +00:00
} )
2023-09-02 18:15:38 +00:00
}
2023-09-11 05:25:37 +00:00
/// Same as `new VeilidRoutingContext()` except easier to chain.
pub fn create ( ) -> APIResult < VeilidRoutingContext > {
VeilidRoutingContext ::new ( )
2023-09-02 18:15:38 +00:00
}
2023-09-04 18:04:57 +00:00
// --------------------------------
2023-09-04 03:25:04 +00:00
// Static methods
2023-09-04 18:04:57 +00:00
// --------------------------------
/// Allocate a new private route set with default cryptography and network options.
/// Returns a route id and a publishable 'blob' with the route encrypted with each crypto kind.
/// Those nodes importing the blob will have their choice of which crypto kind to use.
///
/// Returns a route id and 'blob' that can be published over some means (DHT or otherwise) to be imported by another Veilid node.
pub async fn newPrivateRoute ( ) -> APIResult < VeilidRouteBlob > {
2023-09-04 03:25:04 +00:00
let veilid_api = get_veilid_api ( ) ? ;
let ( route_id , blob ) = veilid_api . new_private_route ( ) . await ? ;
let route_blob = VeilidRouteBlob { route_id , blob } ;
APIResult ::Ok ( route_blob )
}
2023-09-11 05:25:37 +00:00
/// Import a private route blob as a remote private route.
///
/// Returns a route id that can be used to send private messages to the node creating this route.
pub fn importRemotePrivateRoute ( & self , blob : String ) -> APIResult < RouteId > {
let blob : Vec < u8 > = data_encoding ::BASE64URL_NOPAD
. decode ( blob . as_bytes ( ) )
. unwrap ( ) ;
let veilid_api = get_veilid_api ( ) ? ;
let route_id = veilid_api . import_remote_private_route ( blob ) ? ;
APIResult ::Ok ( route_id )
}
2023-09-04 18:04:57 +00:00
/// Allocate a new private route and specify a specific cryptosystem, stability and sequencing preference.
/// Returns a route id and a publishable 'blob' with the route encrypted with each crypto kind.
/// Those nodes importing the blob will have their choice of which crypto kind to use.
///
/// Returns a route id and 'blob' that can be published over some means (DHT or otherwise) to be imported by another Veilid node.
2023-09-04 03:25:04 +00:00
pub async fn newCustomPrivateRoute (
stability : Stability ,
sequencing : Sequencing ,
2023-09-04 18:04:57 +00:00
) -> APIResult < VeilidRouteBlob > {
2023-09-04 03:25:04 +00:00
let veilid_api = get_veilid_api ( ) ? ;
let ( route_id , blob ) = veilid_api
. new_custom_private_route ( & veilid_core ::VALID_CRYPTO_KINDS , stability , sequencing )
. await ? ;
let route_blob = VeilidRouteBlob { route_id , blob } ;
APIResult ::Ok ( route_blob )
}
2023-09-04 18:04:57 +00:00
/// Release either a locally allocated or remotely imported private route.
///
/// This will deactivate the route and free its resources and it can no longer be sent to or received from.
pub fn releasePrivateRoute ( route_id : String ) -> APIResult < ( ) > {
2023-09-11 05:25:37 +00:00
let route_id : veilid_core ::RouteId = RouteId ::from_str ( & route_id ) ? ;
2023-09-04 03:25:04 +00:00
let veilid_api = get_veilid_api ( ) ? ;
veilid_api . release_private_route ( route_id ) ? ;
APIRESULT_UNDEFINED
}
2023-09-04 18:04:57 +00:00
/// Respond to an AppCall received over a VeilidUpdate::AppCall.
///
/// * `call_id` - specifies which call to reply to, and it comes from a VeilidUpdate::AppCall, specifically the VeilidAppCall::id() value.
/// * `message` - is an answer blob to be returned by the remote node's RoutingContext::app_call() function, and may be up to 32768 bytes
pub async fn appCallReply ( call_id : String , message : String ) -> APIResult < ( ) > {
2023-09-04 21:34:49 +00:00
let message = unmarshall ( message ) ;
2023-09-04 18:04:57 +00:00
let call_id = match call_id . parse ( ) {
2023-09-04 03:25:04 +00:00
Ok ( v ) = > v ,
Err ( e ) = > {
return APIResult ::Err ( veilid_core ::VeilidAPIError ::invalid_argument (
2023-09-04 18:04:57 +00:00
e , " call_id " , call_id ,
2023-09-04 03:25:04 +00:00
) )
}
} ;
let veilid_api = get_veilid_api ( ) ? ;
2023-09-04 21:34:49 +00:00
veilid_api . app_call_reply ( call_id , message ) . await ? ;
2023-09-04 03:25:04 +00:00
APIRESULT_UNDEFINED
}
2023-09-04 18:04:57 +00:00
// --------------------------------
2023-09-04 03:25:04 +00:00
// Instance methods
2023-09-04 18:04:57 +00:00
// --------------------------------
fn getRoutingContext ( & self ) -> APIResult < RoutingContext > {
2023-09-11 05:25:37 +00:00
APIResult ::Ok ( self . inner_routing_context . clone ( ) )
}
/// Turn on sender privacy, enabling the use of safety routes.
/// Returns a new instance of VeilidRoutingContext - does not mutate.
///
/// Default values for hop count, stability and sequencing preferences are used.
///
/// Hop count default is dependent on config, but is set to 1 extra hop.
/// Stability default is to choose 'low latency' routes, preferring them over long-term reliability.
/// Sequencing default is to have no preference for ordered vs unordered message delivery
pub fn withPrivacy ( & self ) -> APIResult < VeilidRoutingContext > {
let routing_context = self . getRoutingContext ( ) ? ;
APIResult ::Ok ( VeilidRoutingContext {
inner_routing_context : routing_context . with_privacy ( ) ? ,
} )
}
/// Turn on privacy using a custom `SafetySelection`.
/// Returns a new instance of VeilidRoutingContext - does not mutate.
pub fn withCustomPrivacy (
& self ,
safety_selection : SafetySelection ,
) -> APIResult < VeilidRoutingContext > {
let routing_context = self . getRoutingContext ( ) ? ;
APIResult ::Ok ( VeilidRoutingContext {
inner_routing_context : routing_context . with_custom_privacy ( safety_selection ) ? ,
} )
}
/// Use a specified `Sequencing` preference.
/// Returns a new instance of VeilidRoutingContext - does not mutate.
pub fn withSequencing ( & self , sequencing : Sequencing ) -> APIResult < VeilidRoutingContext > {
let routing_context = self . getRoutingContext ( ) ? ;
APIResult ::Ok ( VeilidRoutingContext {
inner_routing_context : routing_context . with_sequencing ( sequencing ) ,
} )
2023-09-04 18:04:57 +00:00
}
/// App-level unidirectional message that does not expect any value to be returned.
///
/// Veilid apps may use this for arbitrary message passing.
///
/// @param {string} target - can be either a direct node id or a private route.
/// @param {string} message - an arbitrary message blob of up to `32768` bytes.
#[ wasm_bindgen(skip_jsdoc) ]
pub async fn appMessage ( & self , target_string : String , message : String ) -> APIResult < ( ) > {
let routing_context = self . getRoutingContext ( ) ? ;
2023-09-04 21:34:49 +00:00
let message = unmarshall ( message ) ;
2023-09-02 18:15:38 +00:00
let veilid_api = get_veilid_api ( ) ? ;
let target = veilid_api . parse_as_target ( target_string ) . await ? ;
2023-09-04 21:34:49 +00:00
routing_context . app_message ( target , message ) . await ? ;
2023-09-02 18:15:38 +00:00
APIRESULT_UNDEFINED
}
2023-09-04 18:04:57 +00:00
/// App-level bidirectional call that expects a response to be returned.
///
/// Veilid apps may use this for arbitrary message passing.
///
/// @param {string} target_string - can be either a direct node id or a private route, base64Url encoded.
/// @param {string} message - an arbitrary message blob of up to `32768` bytes, base64Url encoded.
/// @returns an answer blob of up to `32768` bytes, base64Url encoded.
#[ wasm_bindgen(skip_jsdoc) ]
pub async fn appCall ( & self , target_string : String , request : String ) -> APIResult < String > {
2023-09-04 21:34:49 +00:00
let request : Vec < u8 > = unmarshall ( request ) ;
2023-09-04 18:04:57 +00:00
let routing_context = self . getRoutingContext ( ) ? ;
2023-09-02 18:15:38 +00:00
let veilid_api = get_veilid_api ( ) ? ;
let target = veilid_api . parse_as_target ( target_string ) . await ? ;
2023-09-04 18:04:57 +00:00
let answer = routing_context . app_call ( target , request ) . await ? ;
2023-09-04 21:34:49 +00:00
let answer = marshall ( & answer ) ;
2023-09-02 18:15:38 +00:00
APIResult ::Ok ( answer )
}
2023-09-04 18:04:57 +00:00
/// DHT Records Creates a new DHT record a specified crypto kind and schema
///
/// The record is considered 'open' after the create operation succeeds.
///
/// @returns the newly allocated DHT record's key if successful.
2023-09-04 03:25:04 +00:00
pub async fn createDhtRecord (
& self ,
schema : DHTSchema ,
kind : String ,
2023-09-04 18:04:57 +00:00
) -> APIResult < DHTRecordDescriptor > {
2023-09-04 03:25:04 +00:00
let crypto_kind = if kind . is_empty ( ) {
2023-09-02 18:15:38 +00:00
None
} else {
2023-09-04 03:25:04 +00:00
Some ( veilid_core ::FourCC ::from_str ( & kind ) ? )
2023-09-02 18:15:38 +00:00
} ;
2023-09-04 18:04:57 +00:00
let routing_context = self . getRoutingContext ( ) ? ;
2023-09-02 18:15:38 +00:00
let dht_record_descriptor = routing_context
. create_dht_record ( schema , crypto_kind )
. await ? ;
2023-09-04 03:25:04 +00:00
APIResult ::Ok ( dht_record_descriptor )
2023-09-02 18:15:38 +00:00
}
2023-09-04 18:04:57 +00:00
/// Opens a DHT record at a specific key.
///
/// Associates a secret if one is provided to provide writer capability. Records may only be opened or created. To re-open with a different routing context, first close the value.
///
/// @returns the DHT record descriptor for the opened record if successful.
/// @param {string} writer - Stringified key pair, in the form of `key:secret` where `key` and `secret` are base64Url encoded.
/// @param {string} key - key of the DHT record.
#[ wasm_bindgen(skip_jsdoc) ]
2023-09-02 18:15:38 +00:00
pub async fn openDhtRecord (
& self ,
key : String ,
writer : Option < String > ,
2023-09-04 18:04:57 +00:00
) -> APIResult < DHTRecordDescriptor > {
2023-09-04 03:25:04 +00:00
let key = TypedKey ::from_str ( & key ) . unwrap ( ) ;
let writer = match writer {
Some ( writer ) = > Some ( KeyPair ::from_str ( & writer ) . unwrap ( ) ) ,
_ = > None ,
} ;
2023-09-04 18:04:57 +00:00
let routing_context = self . getRoutingContext ( ) ? ;
2023-09-02 18:15:38 +00:00
let dht_record_descriptor = routing_context . open_dht_record ( key , writer ) . await ? ;
2023-09-04 03:25:04 +00:00
APIResult ::Ok ( dht_record_descriptor )
2023-09-02 18:15:38 +00:00
}
2023-09-04 18:04:57 +00:00
/// Closes a DHT record at a specific key that was opened with create_dht_record or open_dht_record.
///
/// Closing a record allows you to re-open it with a different routing context
pub async fn closeDhtRecord ( & self , key : String ) -> APIResult < ( ) > {
2023-09-04 03:25:04 +00:00
let key = TypedKey ::from_str ( & key ) . unwrap ( ) ;
2023-09-04 18:04:57 +00:00
let routing_context = self . getRoutingContext ( ) ? ;
2023-09-02 18:15:38 +00:00
routing_context . close_dht_record ( key ) . await ? ;
APIRESULT_UNDEFINED
}
2023-09-04 18:04:57 +00:00
/// Deletes a DHT record at a specific key
///
/// If the record is opened, it must be closed before it is deleted.
/// Deleting a record does not delete it from the network, but will remove the storage of the record locally,
/// and will prevent its value from being refreshed on the network by this node.
pub async fn deleteDhtRecord ( & self , key : String ) -> APIResult < ( ) > {
2023-09-04 03:25:04 +00:00
let key = TypedKey ::from_str ( & key ) . unwrap ( ) ;
2023-09-04 18:04:57 +00:00
let routing_context = self . getRoutingContext ( ) ? ;
2023-09-02 18:15:38 +00:00
routing_context . delete_dht_record ( key ) . await ? ;
APIRESULT_UNDEFINED
}
2023-09-04 18:04:57 +00:00
/// Gets the latest value of a subkey.
///
/// May pull the latest value from the network, but by settings 'force_refresh' you can force a network data refresh.
///
/// Returns `undefined` if the value subkey has not yet been set.
/// Returns base64Url encoded `data` if the value subkey has valid data.
2023-09-02 18:15:38 +00:00
pub async fn getDhtValue (
& self ,
key : String ,
subKey : u32 ,
forceRefresh : bool ,
2023-09-04 18:04:57 +00:00
) -> APIResult < Option < ValueData > > {
2023-09-04 03:25:04 +00:00
let key = TypedKey ::from_str ( & key ) . unwrap ( ) ;
2023-09-04 18:04:57 +00:00
let routing_context = self . getRoutingContext ( ) ? ;
2023-09-02 18:15:38 +00:00
let res = routing_context
. get_dht_value ( key , subKey , forceRefresh )
. await ? ;
2023-09-04 03:25:04 +00:00
APIResult ::Ok ( res )
2023-09-02 18:15:38 +00:00
}
2023-09-04 18:04:57 +00:00
/// Pushes a changed subkey value to the network
///
/// Returns `undefined` if the value was successfully put.
/// Returns base64Url encoded `data` if the value put was older than the one available on the network.
2023-09-02 18:15:38 +00:00
pub async fn setDhtValue (
& self ,
key : String ,
subKey : u32 ,
data : String ,
2023-09-04 18:04:57 +00:00
) -> APIResult < Option < ValueData > > {
2023-09-04 03:25:04 +00:00
let key = TypedKey ::from_str ( & key ) . unwrap ( ) ;
2023-09-02 18:15:38 +00:00
let data : Vec < u8 > = data_encoding ::BASE64URL_NOPAD
. decode ( & data . as_bytes ( ) )
. unwrap ( ) ;
2023-09-04 18:04:57 +00:00
let routing_context = self . getRoutingContext ( ) ? ;
2023-09-02 18:15:38 +00:00
let res = routing_context . set_dht_value ( key , subKey , data ) . await ? ;
2023-09-04 03:25:04 +00:00
APIResult ::Ok ( res )
2023-09-02 18:15:38 +00:00
}
// pub async fn watchDhtValues(
// &self,
// key: String,
// subKeys: ValueSubkeyRangeSet,
// expiration: Timestamp,
// count: u32,
2023-09-04 18:04:57 +00:00
// ) -> APIResult<String> {
2023-09-02 18:15:38 +00:00
// let key: veilid_core::TypedKey = veilid_core::deserialize_json(&key).unwrap();
// let subkeys: veilid_core::ValueSubkeyRangeSet =
// veilid_core::deserialize_json(&subkeys).unwrap();
// let expiration = veilid_core::Timestamp::from_str(&expiration).unwrap();
// let routing_context = {
// let rc = (*ROUTING_CONTEXTS).borrow();
// let Some(routing_context) = rc.get(&id) else {
// return APIResult::Err(veilid_core::VeilidAPIError::invalid_argument("routing_context_watch_dht_values", "id", self.id));
// };
// routing_context.clone()
// };
// let res = routing_context
// .watch_dht_values(key, subkeys, expiration, count)
// .await?;
// APIResult::Ok(res.to_string())
// }
// pub async fn cancelDhtWatch(id: u32, key: String, subkeys: String) -> Promise {
// let key: veilid_core::TypedKey = veilid_core::deserialize_json(&key).unwrap();
// let subkeys: veilid_core::ValueSubkeyRangeSet =
// veilid_core::deserialize_json(&subkeys).unwrap();
// let routing_context = {
// let rc = (*ROUTING_CONTEXTS).borrow();
// let Some(routing_context) = rc.get(&id) else {
// return APIResult::Err(veilid_core::VeilidAPIError::invalid_argument("routing_context_cancel_dht_watch", "id", self.id));
// };
// routing_context.clone()
// };
// let res = routing_context.cancel_dht_watch(key, subkeys).await?;
// APIResult::Ok(res)
// }
}