2022-09-25 22:04:53 +00:00
|
|
|
use super::*;
|
2022-11-26 21:17:30 +00:00
|
|
|
|
2022-09-25 22:04:53 +00:00
|
|
|
///////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2023-09-02 01:13:05 +00:00
|
|
|
/// Valid destinations for a message sent over a routing context
|
2022-09-25 22:04:53 +00:00
|
|
|
#[derive(Clone, Debug)]
|
|
|
|
pub enum Target {
|
2023-09-02 01:13:05 +00:00
|
|
|
/// Node by its public key
|
|
|
|
NodeId(TypedKey),
|
|
|
|
/// Remote private route by its id
|
|
|
|
PrivateRoute(RouteId),
|
2022-09-25 22:04:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub struct RoutingContextInner {}
|
|
|
|
|
|
|
|
pub struct RoutingContextUnlockedInner {
|
2022-10-22 01:27:07 +00:00
|
|
|
/// Safety routing requirements
|
|
|
|
safety_selection: SafetySelection,
|
2022-09-25 22:04:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Drop for RoutingContextInner {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
// self.api
|
|
|
|
// .borrow_mut()
|
|
|
|
// .routing_contexts
|
|
|
|
// //.remove(&self.id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-02 01:13:05 +00:00
|
|
|
/// Routing contexts are the way you specify the communication preferences for Veilid.
|
|
|
|
///
|
|
|
|
/// By default routing contexts are 'direct' from node to node, offering no privacy. To enable sender
|
|
|
|
/// privacy, use [RoutingContext::with_privacy()]. To enable receiver privacy, you should send to a private route RouteId that you have
|
|
|
|
/// imported, rather than directly to a NodeId.
|
|
|
|
///
|
2022-09-25 22:04:53 +00:00
|
|
|
#[derive(Clone)]
|
|
|
|
pub struct RoutingContext {
|
|
|
|
/// Veilid API handle
|
|
|
|
api: VeilidAPI,
|
|
|
|
inner: Arc<Mutex<RoutingContextInner>>,
|
|
|
|
unlocked_inner: Arc<RoutingContextUnlockedInner>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl RoutingContext {
|
|
|
|
////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
pub(super) fn new(api: VeilidAPI) -> Self {
|
|
|
|
Self {
|
|
|
|
api,
|
|
|
|
inner: Arc::new(Mutex::new(RoutingContextInner {})),
|
|
|
|
unlocked_inner: Arc::new(RoutingContextUnlockedInner {
|
2022-11-26 19:16:02 +00:00
|
|
|
safety_selection: SafetySelection::Unsafe(Sequencing::default()),
|
2022-09-25 22:04:53 +00:00
|
|
|
}),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-02 01:13:05 +00:00
|
|
|
/// Turn on sender privacy, enabling the use of safety routes.
|
|
|
|
///
|
|
|
|
/// 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
|
|
|
|
///
|
|
|
|
/// To modify these defaults, use [RoutingContext::with_custom_privacy()].
|
2023-05-29 19:24:57 +00:00
|
|
|
pub fn with_privacy(self) -> VeilidAPIResult<Self> {
|
2022-10-19 01:53:45 +00:00
|
|
|
let config = self.api.config()?;
|
|
|
|
let c = config.get();
|
2022-11-26 19:16:02 +00:00
|
|
|
|
2023-06-28 15:40:02 +00:00
|
|
|
self.with_custom_privacy(SafetySelection::Safe(SafetySpec {
|
|
|
|
preferred_route: None,
|
|
|
|
hop_count: c.network.rpc.default_route_hop_count as usize,
|
|
|
|
stability: Stability::default(),
|
|
|
|
sequencing: Sequencing::default(),
|
|
|
|
}))
|
|
|
|
}
|
|
|
|
|
2023-09-02 01:13:05 +00:00
|
|
|
/// Turn on privacy using a custom [SafetySelection]
|
2023-06-28 15:40:02 +00:00
|
|
|
pub fn with_custom_privacy(self, safety_selection: SafetySelection) -> VeilidAPIResult<Self> {
|
2022-10-19 01:53:45 +00:00
|
|
|
Ok(Self {
|
2022-09-25 22:04:53 +00:00
|
|
|
api: self.api.clone(),
|
|
|
|
inner: Arc::new(Mutex::new(RoutingContextInner {})),
|
2023-06-28 15:40:02 +00:00
|
|
|
unlocked_inner: Arc::new(RoutingContextUnlockedInner { safety_selection }),
|
2022-10-19 01:53:45 +00:00
|
|
|
})
|
|
|
|
}
|
2023-02-26 03:02:13 +00:00
|
|
|
|
2023-09-02 01:13:05 +00:00
|
|
|
/// Use a specified [Sequencing] preference, with or without privacy
|
2022-10-22 01:27:07 +00:00
|
|
|
pub fn with_sequencing(self, sequencing: Sequencing) -> Self {
|
2022-09-25 22:04:53 +00:00
|
|
|
Self {
|
|
|
|
api: self.api.clone(),
|
|
|
|
inner: Arc::new(Mutex::new(RoutingContextInner {})),
|
|
|
|
unlocked_inner: Arc::new(RoutingContextUnlockedInner {
|
2022-10-22 01:27:07 +00:00
|
|
|
safety_selection: match self.unlocked_inner.safety_selection {
|
|
|
|
SafetySelection::Unsafe(_) => SafetySelection::Unsafe(sequencing),
|
|
|
|
SafetySelection::Safe(safety_spec) => SafetySelection::Safe(SafetySpec {
|
|
|
|
preferred_route: safety_spec.preferred_route,
|
|
|
|
hop_count: safety_spec.hop_count,
|
|
|
|
stability: safety_spec.stability,
|
|
|
|
sequencing,
|
|
|
|
}),
|
|
|
|
},
|
2022-09-25 22:04:53 +00:00
|
|
|
}),
|
|
|
|
}
|
|
|
|
}
|
2022-11-26 19:16:02 +00:00
|
|
|
|
|
|
|
fn sequencing(&self) -> Sequencing {
|
2022-10-22 01:27:07 +00:00
|
|
|
match self.unlocked_inner.safety_selection {
|
|
|
|
SafetySelection::Unsafe(sequencing) => sequencing,
|
|
|
|
SafetySelection::Safe(safety_spec) => safety_spec.sequencing,
|
|
|
|
}
|
|
|
|
}
|
2022-09-25 22:04:53 +00:00
|
|
|
|
2023-09-02 01:13:05 +00:00
|
|
|
/// Get the [VeilidAPI] object that created this [RoutingContext]
|
2022-09-25 22:04:53 +00:00
|
|
|
pub fn api(&self) -> VeilidAPI {
|
|
|
|
self.api.clone()
|
|
|
|
}
|
|
|
|
|
2023-05-29 19:24:57 +00:00
|
|
|
async fn get_destination(&self, target: Target) -> VeilidAPIResult<rpc_processor::Destination> {
|
2022-09-25 22:04:53 +00:00
|
|
|
let rpc_processor = self.api.rpc_processor()?;
|
|
|
|
|
|
|
|
match target {
|
|
|
|
Target::NodeId(node_id) => {
|
|
|
|
// Resolve node
|
2023-05-29 19:24:57 +00:00
|
|
|
let mut nr = match rpc_processor
|
|
|
|
.resolve_node(node_id, self.unlocked_inner.safety_selection)
|
|
|
|
.await
|
|
|
|
{
|
2022-09-25 22:04:53 +00:00
|
|
|
Ok(Some(nr)) => nr,
|
2023-02-26 03:02:13 +00:00
|
|
|
Ok(None) => apibail_invalid_target!(),
|
2022-09-25 22:04:53 +00:00
|
|
|
Err(e) => return Err(e.into()),
|
|
|
|
};
|
2022-10-22 01:27:07 +00:00
|
|
|
// Apply sequencing to match safety selection
|
|
|
|
nr.set_sequencing(self.sequencing());
|
|
|
|
|
2022-09-25 22:04:53 +00:00
|
|
|
Ok(rpc_processor::Destination::Direct {
|
|
|
|
target: nr,
|
2022-10-22 01:27:07 +00:00
|
|
|
safety_selection: self.unlocked_inner.safety_selection,
|
2022-09-25 22:04:53 +00:00
|
|
|
})
|
|
|
|
}
|
2023-02-25 02:02:24 +00:00
|
|
|
Target::PrivateRoute(rsid) => {
|
2022-11-22 23:26:39 +00:00
|
|
|
// Get remote private route
|
|
|
|
let rss = self.api.routing_table()?.route_spec_store();
|
2023-02-26 23:11:10 +00:00
|
|
|
|
|
|
|
let Some(private_route) = rss.best_remote_private_route(&rsid) else {
|
2023-02-26 03:02:13 +00:00
|
|
|
apibail_invalid_target!();
|
|
|
|
};
|
2022-11-25 19:21:55 +00:00
|
|
|
|
2022-11-22 23:26:39 +00:00
|
|
|
Ok(rpc_processor::Destination::PrivateRoute {
|
2023-02-26 23:11:10 +00:00
|
|
|
private_route,
|
2022-11-22 23:26:39 +00:00
|
|
|
safety_selection: self.unlocked_inner.safety_selection,
|
|
|
|
})
|
|
|
|
}
|
2022-09-25 22:04:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////
|
|
|
|
// App-level Messaging
|
|
|
|
|
2023-09-02 01:13:05 +00:00
|
|
|
/// App-level bidirectional call that expects a response to be returned.
|
|
|
|
///
|
|
|
|
/// Veilid apps may use this for arbitrary message passing.
|
|
|
|
///
|
|
|
|
/// * `target` - can be either a direct node id or a private route
|
|
|
|
/// * `message` - an arbitrary message blob of up to 32768 bytes
|
|
|
|
///
|
|
|
|
/// Returns an answer blob of up to 32768 bytes
|
2023-06-27 01:29:02 +00:00
|
|
|
pub async fn app_call(&self, target: Target, message: Vec<u8>) -> VeilidAPIResult<Vec<u8>> {
|
2022-09-25 22:04:53 +00:00
|
|
|
let rpc_processor = self.api.rpc_processor()?;
|
|
|
|
|
|
|
|
// Get destination
|
|
|
|
let dest = self.get_destination(target).await?;
|
|
|
|
|
|
|
|
// Send app message
|
2023-06-27 01:29:02 +00:00
|
|
|
let answer = match rpc_processor.rpc_call_app_call(dest, message).await {
|
2022-09-25 22:04:53 +00:00
|
|
|
Ok(NetworkResult::Value(v)) => v,
|
2022-11-26 21:17:30 +00:00
|
|
|
Ok(NetworkResult::Timeout) => apibail_timeout!(),
|
2023-06-21 17:40:12 +00:00
|
|
|
Ok(NetworkResult::ServiceUnavailable(e)) => {
|
|
|
|
log_network_result!(format!("app_call: ServiceUnavailable({})", e));
|
|
|
|
apibail_try_again!()
|
|
|
|
}
|
2022-10-04 17:09:03 +00:00
|
|
|
Ok(NetworkResult::NoConnection(e)) | Ok(NetworkResult::AlreadyExists(e)) => {
|
2022-11-26 21:17:30 +00:00
|
|
|
apibail_no_connection!(e);
|
2022-09-25 22:04:53 +00:00
|
|
|
}
|
2022-10-04 17:09:03 +00:00
|
|
|
|
2022-09-25 22:04:53 +00:00
|
|
|
Ok(NetworkResult::InvalidMessage(message)) => {
|
2022-11-26 21:17:30 +00:00
|
|
|
apibail_generic!(message);
|
2022-09-25 22:04:53 +00:00
|
|
|
}
|
|
|
|
Err(e) => return Err(e.into()),
|
|
|
|
};
|
|
|
|
|
|
|
|
Ok(answer.answer)
|
|
|
|
}
|
|
|
|
|
2023-09-02 01:13:05 +00:00
|
|
|
/// App-level unidirectional message that does not expect any value to be returned.
|
|
|
|
///
|
|
|
|
/// Veilid apps may use this for arbitrary message passing.
|
|
|
|
///
|
|
|
|
/// * `target` - can be either a direct node id or a private route
|
|
|
|
/// * `message` - an arbitrary message blob of up to 32768 bytes
|
2023-05-29 19:24:57 +00:00
|
|
|
pub async fn app_message(&self, target: Target, message: Vec<u8>) -> VeilidAPIResult<()> {
|
2022-09-25 22:04:53 +00:00
|
|
|
let rpc_processor = self.api.rpc_processor()?;
|
|
|
|
|
|
|
|
// Get destination
|
|
|
|
let dest = self.get_destination(target).await?;
|
|
|
|
|
|
|
|
// Send app message
|
|
|
|
match rpc_processor.rpc_call_app_message(dest, message).await {
|
|
|
|
Ok(NetworkResult::Value(())) => {}
|
2022-11-26 21:17:30 +00:00
|
|
|
Ok(NetworkResult::Timeout) => apibail_timeout!(),
|
2023-06-21 17:40:12 +00:00
|
|
|
Ok(NetworkResult::ServiceUnavailable(e)) => {
|
|
|
|
log_network_result!(format!("app_message: ServiceUnavailable({})", e));
|
|
|
|
apibail_try_again!()
|
|
|
|
}
|
2022-10-04 17:09:03 +00:00
|
|
|
Ok(NetworkResult::NoConnection(e)) | Ok(NetworkResult::AlreadyExists(e)) => {
|
2022-11-26 21:17:30 +00:00
|
|
|
apibail_no_connection!(e);
|
2022-09-25 22:04:53 +00:00
|
|
|
}
|
|
|
|
Ok(NetworkResult::InvalidMessage(message)) => {
|
2022-11-26 21:17:30 +00:00
|
|
|
apibail_generic!(message);
|
2022-09-25 22:04:53 +00:00
|
|
|
}
|
|
|
|
Err(e) => return Err(e.into()),
|
|
|
|
};
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
///////////////////////////////////
|
2023-05-29 19:24:57 +00:00
|
|
|
/// DHT Records
|
2022-09-25 22:04:53 +00:00
|
|
|
|
2023-05-29 19:24:57 +00:00
|
|
|
/// Creates a new DHT record a specified crypto kind and schema
|
2023-09-02 01:13:05 +00:00
|
|
|
///
|
|
|
|
/// The record is considered 'open' after the create operation succeeds.
|
|
|
|
///
|
|
|
|
/// Returns the newly allocated DHT record's key if successful.
|
2023-05-29 19:24:57 +00:00
|
|
|
pub async fn create_dht_record(
|
2023-02-26 03:02:13 +00:00
|
|
|
&self,
|
2023-05-29 19:24:57 +00:00
|
|
|
schema: DHTSchema,
|
2023-07-09 02:50:44 +00:00
|
|
|
kind: Option<CryptoKind>,
|
2023-05-29 19:24:57 +00:00
|
|
|
) -> VeilidAPIResult<DHTRecordDescriptor> {
|
2023-07-09 02:50:44 +00:00
|
|
|
let kind = kind.unwrap_or(best_crypto_kind());
|
2023-08-05 15:33:27 +00:00
|
|
|
Crypto::validate_crypto_kind(kind)?;
|
2023-05-29 19:24:57 +00:00
|
|
|
let storage_manager = self.api.storage_manager()?;
|
|
|
|
storage_manager
|
|
|
|
.create_record(kind, schema, self.unlocked_inner.safety_selection)
|
|
|
|
.await
|
2022-09-25 22:04:53 +00:00
|
|
|
}
|
|
|
|
|
2023-09-02 01:13:05 +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.
|
|
|
|
///
|
2023-05-29 19:24:57 +00:00
|
|
|
/// Returns the DHT record descriptor for the opened record if successful
|
|
|
|
pub async fn open_dht_record(
|
2022-09-25 22:04:53 +00:00
|
|
|
&self,
|
2023-05-29 19:24:57 +00:00
|
|
|
key: TypedKey,
|
|
|
|
writer: Option<KeyPair>,
|
|
|
|
) -> VeilidAPIResult<DHTRecordDescriptor> {
|
2023-08-05 15:33:27 +00:00
|
|
|
Crypto::validate_crypto_kind(key.kind)?;
|
2023-05-29 19:24:57 +00:00
|
|
|
let storage_manager = self.api.storage_manager()?;
|
|
|
|
storage_manager
|
|
|
|
.open_record(key, writer, self.unlocked_inner.safety_selection)
|
|
|
|
.await
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Closes a DHT record at a specific key that was opened with create_dht_record or open_dht_record.
|
2023-09-02 01:13:05 +00:00
|
|
|
///
|
2023-05-29 19:24:57 +00:00
|
|
|
/// Closing a record allows you to re-open it with a different routing context
|
|
|
|
pub async fn close_dht_record(&self, key: TypedKey) -> VeilidAPIResult<()> {
|
2023-08-05 15:33:27 +00:00
|
|
|
Crypto::validate_crypto_kind(key.kind)?;
|
2023-05-29 19:24:57 +00:00
|
|
|
let storage_manager = self.api.storage_manager()?;
|
|
|
|
storage_manager.close_record(key).await
|
|
|
|
}
|
|
|
|
|
2023-09-02 01:13:05 +00:00
|
|
|
/// Deletes a DHT record at a specific key
|
|
|
|
///
|
|
|
|
/// If the record is opened, it must be closed before it is deleted.
|
2023-05-29 19:24:57 +00:00
|
|
|
/// 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 delete_dht_record(&self, key: TypedKey) -> VeilidAPIResult<()> {
|
2023-08-05 15:33:27 +00:00
|
|
|
Crypto::validate_crypto_kind(key.kind)?;
|
2023-05-29 19:24:57 +00:00
|
|
|
let storage_manager = self.api.storage_manager()?;
|
|
|
|
storage_manager.delete_record(key).await
|
2022-09-25 22:04:53 +00:00
|
|
|
}
|
|
|
|
|
2023-05-29 19:24:57 +00:00
|
|
|
/// Gets the latest value of a subkey
|
2023-09-02 01:13:05 +00:00
|
|
|
///
|
2023-09-05 16:05:18 +00:00
|
|
|
/// May pull the latest value from the network, but by setting 'force_refresh' you can force a network data refresh
|
2023-09-02 01:13:05 +00:00
|
|
|
///
|
|
|
|
/// Returns `None` if the value subkey has not yet been set
|
|
|
|
/// Returns `Some(data)` if the value subkey has valid data
|
2023-05-29 19:24:57 +00:00
|
|
|
pub async fn get_dht_value(
|
2022-09-25 22:04:53 +00:00
|
|
|
&self,
|
2023-05-29 19:24:57 +00:00
|
|
|
key: TypedKey,
|
|
|
|
subkey: ValueSubkey,
|
|
|
|
force_refresh: bool,
|
|
|
|
) -> VeilidAPIResult<Option<ValueData>> {
|
2023-08-05 15:33:27 +00:00
|
|
|
Crypto::validate_crypto_kind(key.kind)?;
|
2023-05-29 19:24:57 +00:00
|
|
|
let storage_manager = self.api.storage_manager()?;
|
|
|
|
storage_manager.get_value(key, subkey, force_refresh).await
|
2022-09-25 22:04:53 +00:00
|
|
|
}
|
|
|
|
|
2023-05-29 19:24:57 +00:00
|
|
|
/// Pushes a changed subkey value to the network
|
2023-09-02 01:13:05 +00:00
|
|
|
///
|
|
|
|
/// Returns `None` if the value was successfully put
|
|
|
|
/// Returns `Some(data)` if the value put was older than the one available on the network
|
2023-05-29 19:24:57 +00:00
|
|
|
pub async fn set_dht_value(
|
2023-02-26 03:02:13 +00:00
|
|
|
&self,
|
2023-05-29 19:24:57 +00:00
|
|
|
key: TypedKey,
|
|
|
|
subkey: ValueSubkey,
|
|
|
|
data: Vec<u8>,
|
|
|
|
) -> VeilidAPIResult<Option<ValueData>> {
|
2023-08-05 15:33:27 +00:00
|
|
|
Crypto::validate_crypto_kind(key.kind)?;
|
2023-05-29 19:24:57 +00:00
|
|
|
let storage_manager = self.api.storage_manager()?;
|
|
|
|
storage_manager.set_value(key, subkey, data).await
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Watches changes to an opened or created value
|
2023-09-02 01:13:05 +00:00
|
|
|
///
|
2023-05-29 19:24:57 +00:00
|
|
|
/// Changes to subkeys within the subkey range are returned via a ValueChanged callback
|
|
|
|
/// If the subkey range is empty, all subkey changes are considered
|
|
|
|
/// Expiration can be infinite to keep the watch for the maximum amount of time
|
2023-09-02 01:13:05 +00:00
|
|
|
///
|
2023-05-29 19:24:57 +00:00
|
|
|
/// Return value upon success is the amount of time allowed for the watch
|
|
|
|
pub async fn watch_dht_values(
|
|
|
|
&self,
|
|
|
|
key: TypedKey,
|
|
|
|
subkeys: ValueSubkeyRangeSet,
|
|
|
|
expiration: Timestamp,
|
|
|
|
count: u32,
|
|
|
|
) -> VeilidAPIResult<Timestamp> {
|
2023-08-05 15:33:27 +00:00
|
|
|
Crypto::validate_crypto_kind(key.kind)?;
|
2023-05-29 19:24:57 +00:00
|
|
|
let storage_manager = self.api.storage_manager()?;
|
|
|
|
storage_manager
|
|
|
|
.watch_values(key, subkeys, expiration, count)
|
|
|
|
.await
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Cancels a watch early
|
2023-09-02 01:13:05 +00:00
|
|
|
///
|
2023-05-29 19:24:57 +00:00
|
|
|
/// This is a convenience function that cancels watching all subkeys in a range
|
|
|
|
pub async fn cancel_dht_watch(
|
|
|
|
&self,
|
|
|
|
key: TypedKey,
|
|
|
|
subkeys: ValueSubkeyRangeSet,
|
|
|
|
) -> VeilidAPIResult<bool> {
|
2023-08-05 15:33:27 +00:00
|
|
|
Crypto::validate_crypto_kind(key.kind)?;
|
2023-05-29 19:24:57 +00:00
|
|
|
let storage_manager = self.api.storage_manager()?;
|
|
|
|
storage_manager.cancel_watch_values(key, subkeys).await
|
2022-09-25 22:04:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
///////////////////////////////////
|
|
|
|
/// Block Store
|
|
|
|
|
2023-06-05 01:22:55 +00:00
|
|
|
#[cfg(feature = "unstable-blockstore")]
|
2023-05-29 19:24:57 +00:00
|
|
|
pub async fn find_block(&self, _block_id: PublicKey) -> VeilidAPIResult<Vec<u8>> {
|
2022-09-25 22:04:53 +00:00
|
|
|
panic!("unimplemented");
|
|
|
|
}
|
|
|
|
|
2023-06-05 01:22:55 +00:00
|
|
|
#[cfg(feature = "unstable-blockstore")]
|
2023-05-29 19:24:57 +00:00
|
|
|
pub async fn supply_block(&self, _block_id: PublicKey) -> VeilidAPIResult<bool> {
|
2022-09-25 22:04:53 +00:00
|
|
|
panic!("unimplemented");
|
|
|
|
}
|
|
|
|
}
|