veilid/veilid-core/src/veilid_api/mod.rs

1403 lines
43 KiB
Rust
Raw Normal View History

2021-12-24 23:02:53 +00:00
#![allow(dead_code)]
2021-12-14 14:48:33 +00:00
mod debug;
pub use debug::*;
2021-11-22 16:28:30 +00:00
use crate::*;
pub use crate::xx::{
IpAddr, Ipv4Addr, Ipv6Addr, SocketAddr, SocketAddrV4, SocketAddrV6, SystemPinBoxFuture,
ToSocketAddrs,
};
pub use alloc::string::ToString;
2022-02-07 02:18:42 +00:00
pub use attachment_manager::AttachmentManager;
2021-11-22 16:28:30 +00:00
pub use core::str::FromStr;
2022-02-07 02:18:42 +00:00
pub use dht::crypto::Crypto;
pub use dht::key::{generate_secret, DHTKey, DHTKeySecret};
pub use intf::BlockStore;
pub use intf::ProtectedStore;
pub use intf::TableStore;
pub use network_manager::NetworkManager;
pub use routing_table::RoutingTable;
pub use rpc_processor::InfoAnswer;
use api_logger::*;
use core::fmt;
use core_context::{api_shutdown, VeilidCoreContext};
use rpc_processor::{RPCError, RPCProcessor};
use serde::*;
use xx::*;
2021-11-22 16:28:30 +00:00
/////////////////////////////////////////////////////////////////////////////////////////////////////
2022-02-07 02:18:42 +00:00
#[derive(Clone, Debug, PartialOrd, PartialEq, Eq, Ord, Serialize, Deserialize)]
#[serde(tag = "kind")]
2021-12-21 00:12:30 +00:00
pub enum VeilidAPIError {
2022-02-07 02:18:42 +00:00
NotInitialized,
AlreadyInitialized,
2021-12-21 00:12:30 +00:00
Timeout,
Shutdown,
2022-02-09 14:47:36 +00:00
NodeNotFound {
node_id: NodeId,
},
NoDialInfo {
node_id: NodeId,
},
Internal {
message: String,
},
Unimplemented {
message: String,
},
2021-12-21 00:12:30 +00:00
ParseError {
message: String,
value: String,
},
InvalidArgument {
context: String,
argument: String,
value: String,
},
MissingArgument {
context: String,
argument: String,
},
}
impl fmt::Display for VeilidAPIError {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> Result<(), fmt::Error> {
match self {
2022-02-07 02:18:42 +00:00
VeilidAPIError::NotInitialized => write!(f, "VeilidAPIError::NotInitialized"),
VeilidAPIError::AlreadyInitialized => write!(f, "VeilidAPIError::AlreadyInitialized"),
2021-12-21 00:12:30 +00:00
VeilidAPIError::Timeout => write!(f, "VeilidAPIError::Timeout"),
VeilidAPIError::Shutdown => write!(f, "VeilidAPIError::Shutdown"),
2022-02-09 14:47:36 +00:00
VeilidAPIError::NodeNotFound { node_id } => {
write!(f, "VeilidAPIError::NodeNotFound({})", node_id)
}
VeilidAPIError::NoDialInfo { node_id } => {
write!(f, "VeilidAPIError::NoDialInfo({})", node_id)
}
VeilidAPIError::Internal { message } => {
write!(f, "VeilidAPIError::Internal({})", message)
}
VeilidAPIError::Unimplemented { message } => {
write!(f, "VeilidAPIError::Unimplemented({})", message)
}
2021-12-21 00:12:30 +00:00
VeilidAPIError::ParseError { message, value } => {
write!(f, "VeilidAPIError::ParseError({}: {})", message, value)
}
VeilidAPIError::InvalidArgument {
context,
argument,
value,
} => {
write!(
f,
"VeilidAPIError::InvalidArgument({}: {} = {})",
context, argument, value
)
}
VeilidAPIError::MissingArgument { context, argument } => {
write!(
f,
"VeilidAPIError::MissingArgument({}: {})",
context, argument
)
}
}
}
}
fn convert_rpc_error(x: RPCError) -> VeilidAPIError {
match x {
RPCError::Timeout => VeilidAPIError::Timeout,
2022-02-09 14:47:36 +00:00
RPCError::Unimplemented(s) => VeilidAPIError::Unimplemented { message: s },
RPCError::Internal(s) => VeilidAPIError::Internal { message: s },
RPCError::Protocol(s) => VeilidAPIError::Internal { message: s },
RPCError::InvalidFormat => VeilidAPIError::Internal {
message: "Invalid packet format".to_owned(),
},
2021-12-21 00:12:30 +00:00
}
}
macro_rules! map_rpc_error {
() => {
|x| convert_rpc_error(x)
};
}
macro_rules! parse_error {
($msg:expr, $val:expr) => {
VeilidAPIError::ParseError {
message: $msg.to_string(),
value: $val.to_string(),
}
};
}
/////////////////////////////////////////////////////////////////////////////////////////////////////
2022-02-07 02:18:42 +00:00
#[derive(Debug, Clone, Copy, Serialize, Deserialize)]
2022-01-31 15:11:18 +00:00
pub enum VeilidLogLevel {
Error = 1,
Warn,
Info,
Debug,
Trace,
}
impl VeilidLogLevel {
pub fn from_log_level(level: log::Level) -> VeilidLogLevel {
match level {
Level::Error => VeilidLogLevel::Error,
Level::Warn => VeilidLogLevel::Warn,
Level::Info => VeilidLogLevel::Info,
Level::Debug => VeilidLogLevel::Debug,
Level::Trace => VeilidLogLevel::Trace,
}
}
}
2022-02-07 02:18:42 +00:00
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(tag = "kind")]
pub enum VeilidUpdate {
2022-02-02 01:22:01 +00:00
Log {
log_level: VeilidLogLevel,
message: String,
},
2022-02-09 14:47:36 +00:00
Attachment {
state: AttachmentState,
},
2022-02-15 18:40:17 +00:00
Shutdown,
}
2022-02-07 02:18:42 +00:00
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct VeilidState {
pub attachment: AttachmentState,
}
/////////////////////////////////////////////////////////////////////////////////////////////////////
///
2022-02-07 02:18:42 +00:00
#[derive(Clone, Debug, Default, PartialOrd, PartialEq, Eq, Ord, Serialize, Deserialize)]
2021-11-22 16:28:30 +00:00
pub struct NodeId {
pub key: DHTKey,
}
impl NodeId {
pub fn new(key: DHTKey) -> Self {
2021-12-21 00:12:30 +00:00
assert!(key.valid);
2021-11-27 17:44:21 +00:00
Self { key }
2021-11-22 16:28:30 +00:00
}
}
2021-12-21 00:12:30 +00:00
impl fmt::Display for NodeId {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> Result<(), fmt::Error> {
write!(f, "{}", self.key.encode())
}
}
2021-11-22 16:28:30 +00:00
2022-02-07 02:18:42 +00:00
#[derive(Clone, Debug, Default, PartialOrd, PartialEq, Eq, Ord, Serialize, Deserialize)]
2021-11-22 16:28:30 +00:00
pub struct ValueKey {
pub key: DHTKey,
pub subkey: Option<String>,
}
impl ValueKey {
pub fn new(key: DHTKey) -> Self {
2021-11-27 17:44:21 +00:00
Self { key, subkey: None }
2021-11-22 16:28:30 +00:00
}
pub fn new_subkey(key: DHTKey, subkey: String) -> Self {
Self {
2021-11-27 17:44:21 +00:00
key,
2021-11-28 00:56:56 +00:00
subkey: if subkey.is_empty() {
2021-11-22 16:28:30 +00:00
None
} else {
Some(subkey)
},
}
}
}
2022-02-07 02:18:42 +00:00
#[derive(Clone, Debug, Default, PartialOrd, PartialEq, Eq, Ord, Serialize, Deserialize)]
2021-11-22 16:28:30 +00:00
pub struct BlockId {
pub key: DHTKey,
}
impl BlockId {
pub fn new(key: DHTKey) -> Self {
2021-11-27 17:44:21 +00:00
Self { key }
2021-11-22 16:28:30 +00:00
}
}
/////////////////////////////////////////////////////////////////////////////////////////////////////
2022-02-07 02:18:42 +00:00
#[derive(Clone, Debug, PartialEq, PartialOrd, Ord, Eq, Hash, Default, Serialize, Deserialize)]
2021-11-22 16:28:30 +00:00
pub struct SenderInfo {
2021-12-22 03:20:55 +00:00
pub socket_address: Option<SocketAddress>,
2021-11-22 16:28:30 +00:00
}
2022-02-07 02:18:42 +00:00
#[derive(Clone, Debug, Default, Serialize, Deserialize)]
2021-11-22 16:28:30 +00:00
pub struct NodeInfo {
pub can_route: bool,
pub will_route: bool,
pub can_tunnel: bool,
pub will_tunnel: bool,
pub can_signal_lease: bool,
pub will_signal_lease: bool,
pub can_relay_lease: bool,
pub will_relay_lease: bool,
pub can_validate_dial_info: bool,
pub will_validate_dial_info: bool,
}
2022-02-07 02:18:42 +00:00
#[derive(Copy, Clone, Debug, PartialEq, PartialOrd, Ord, Eq, Hash, Serialize, Deserialize)]
2021-12-22 03:20:55 +00:00
// The derived ordering here is the order of preference, lower is preferred for connections
// Must match DialInfo order
2021-11-22 16:28:30 +00:00
pub enum ProtocolType {
UDP,
TCP,
WS,
WSS,
}
2022-02-07 02:18:42 +00:00
#[derive(Copy, Clone, Debug, PartialEq, PartialOrd, Ord, Eq, Hash, Serialize, Deserialize)]
2021-12-22 03:20:55 +00:00
pub enum AddressType {
IPV4,
IPV6,
2021-11-22 16:28:30 +00:00
}
2022-02-07 02:18:42 +00:00
#[derive(Copy, Clone, Debug, PartialEq, PartialOrd, Ord, Eq, Hash, Serialize, Deserialize)]
2021-11-22 16:28:30 +00:00
pub enum Address {
IPV4(Ipv4Addr),
IPV6(Ipv6Addr),
}
2021-12-21 00:12:30 +00:00
impl Default for Address {
fn default() -> Self {
Address::IPV4(Ipv4Addr::new(0, 0, 0, 0))
2021-12-14 14:48:33 +00:00
}
2021-12-21 00:12:30 +00:00
}
impl Address {
2021-11-22 16:28:30 +00:00
pub fn from_socket_addr(sa: SocketAddr) -> Address {
match sa {
SocketAddr::V4(v4) => Address::IPV4(*v4.ip()),
SocketAddr::V6(v6) => Address::IPV6(*v6.ip()),
}
}
2021-12-22 03:20:55 +00:00
pub fn address_type(&self) -> AddressType {
match self {
2021-12-24 23:02:53 +00:00
Address::IPV4(_) => AddressType::IPV4,
Address::IPV6(_) => AddressType::IPV6,
2021-12-22 03:20:55 +00:00
}
}
2021-11-22 16:28:30 +00:00
pub fn address_string(&self) -> String {
match self {
Address::IPV4(v4) => v4.to_string(),
Address::IPV6(v6) => v6.to_string(),
}
}
pub fn address_string_with_port(&self, port: u16) -> String {
match self {
Address::IPV4(v4) => format!("{}:{}", v4.to_string(), port),
Address::IPV6(v6) => format!("[{}]:{}", v6.to_string(), port),
}
}
2021-12-22 03:20:55 +00:00
pub fn is_global(&self) -> bool {
2021-11-22 16:28:30 +00:00
match self {
2021-12-24 23:02:53 +00:00
Address::IPV4(v4) => ipv4addr_is_global(v4),
Address::IPV6(v6) => ipv6addr_is_global(v6),
2021-11-22 16:28:30 +00:00
}
}
2021-12-22 03:20:55 +00:00
pub fn is_local(&self) -> bool {
2021-11-22 16:28:30 +00:00
match self {
2021-12-24 23:02:53 +00:00
Address::IPV4(v4) => ipv4addr_is_private(v4),
Address::IPV6(v6) => ipv6addr_is_unicast_site_local(v6),
2021-11-22 16:28:30 +00:00
}
}
2021-12-21 00:12:30 +00:00
pub fn to_ip_addr(&self) -> IpAddr {
match self {
Self::IPV4(a) => IpAddr::V4(*a),
Self::IPV6(a) => IpAddr::V6(*a),
}
}
pub fn to_socket_addr(&self, port: u16) -> SocketAddr {
SocketAddr::new(self.to_ip_addr(), port)
}
pub fn to_canonical(&self) -> Address {
match self {
Address::IPV4(v4) => Address::IPV4(*v4),
Address::IPV6(v6) => match v6.to_ipv4() {
Some(v4) => Address::IPV4(v4),
None => Address::IPV6(*v6),
},
}
2021-11-22 16:28:30 +00:00
}
}
2021-12-21 00:12:30 +00:00
impl FromStr for Address {
type Err = VeilidAPIError;
fn from_str(host: &str) -> Result<Address, VeilidAPIError> {
2021-11-28 00:56:56 +00:00
if let Ok(addr) = Ipv4Addr::from_str(host) {
2021-11-22 16:28:30 +00:00
Ok(Address::IPV4(addr))
2021-11-28 00:56:56 +00:00
} else if let Ok(addr) = Ipv6Addr::from_str(host) {
2021-11-22 16:28:30 +00:00
Ok(Address::IPV6(addr))
} else {
2021-12-21 00:12:30 +00:00
Err(parse_error!("Address::from_str failed", host))
2021-11-22 16:28:30 +00:00
}
}
}
2022-02-07 02:18:42 +00:00
#[derive(
Copy, Default, Clone, Debug, PartialEq, PartialOrd, Ord, Eq, Hash, Serialize, Deserialize,
)]
2021-12-21 00:12:30 +00:00
pub struct SocketAddress {
address: Address,
port: u16,
}
impl SocketAddress {
pub fn new(address: Address, port: u16) -> Self {
Self { address, port }
}
pub fn from_socket_addr(sa: SocketAddr) -> SocketAddress {
Self {
address: Address::from_socket_addr(sa),
port: sa.port(),
}
}
pub fn address(&self) -> Address {
self.address
}
2021-12-22 03:20:55 +00:00
pub fn address_type(&self) -> AddressType {
self.address.address_type()
}
2021-12-21 00:12:30 +00:00
pub fn port(&self) -> u16 {
self.port
}
pub fn to_canonical(&self) -> SocketAddress {
SocketAddress {
address: self.address.to_canonical(),
port: self.port,
}
}
pub fn to_ip_addr(&self) -> IpAddr {
self.address.to_ip_addr()
}
pub fn to_socket_addr(&self) -> SocketAddr {
self.address.to_socket_addr(self.port)
}
}
impl fmt::Display for SocketAddress {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> Result<(), fmt::Error> {
2021-12-24 01:34:52 +00:00
write!(f, "{}", self.to_socket_addr())
2021-12-21 00:12:30 +00:00
}
}
impl FromStr for SocketAddress {
type Err = VeilidAPIError;
fn from_str(s: &str) -> Result<SocketAddress, VeilidAPIError> {
2021-12-24 01:34:52 +00:00
let sa = SocketAddr::from_str(s)
.map_err(|e| parse_error!("Failed to parse SocketAddress", e))?;
Ok(SocketAddress::from_socket_addr(sa))
}
}
//////////////////////////////////////////////////////////////////
2022-02-07 02:18:42 +00:00
#[derive(Clone, Default, PartialEq, Eq, PartialOrd, Ord, Serialize, Deserialize)]
2021-12-24 01:34:52 +00:00
pub struct DialInfoFilter {
pub peer_scope: PeerScope,
pub protocol_type: Option<ProtocolType>,
pub address_type: Option<AddressType>,
}
impl DialInfoFilter {
2021-12-24 23:02:53 +00:00
pub fn all() -> Self {
2021-12-24 01:34:52 +00:00
Self {
peer_scope: PeerScope::All,
protocol_type: None,
address_type: None,
}
}
2021-12-24 23:02:53 +00:00
pub fn global() -> Self {
2021-12-24 01:34:52 +00:00
Self {
2021-12-24 23:02:53 +00:00
peer_scope: PeerScope::Global,
protocol_type: None,
2021-12-24 01:34:52 +00:00
address_type: None,
}
}
2021-12-24 23:02:53 +00:00
pub fn local() -> Self {
2021-12-24 01:34:52 +00:00
Self {
2021-12-24 23:02:53 +00:00
peer_scope: PeerScope::Local,
protocol_type: None,
address_type: None,
2021-12-24 01:34:52 +00:00
}
}
2021-12-24 23:02:53 +00:00
pub fn scoped(peer_scope: PeerScope) -> Self {
Self {
peer_scope,
protocol_type: None,
address_type: None,
}
}
pub fn with_protocol_type(mut self, protocol_type: ProtocolType) -> Self {
self.protocol_type = Some(protocol_type);
self
}
pub fn with_address_type(mut self, address_type: AddressType) -> Self {
self.address_type = Some(address_type);
self
}
2021-12-24 01:34:52 +00:00
pub fn is_empty(&self) -> bool {
self.peer_scope == PeerScope::All
&& self.protocol_type.is_none()
&& self.address_type.is_none()
}
}
impl fmt::Debug for DialInfoFilter {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> Result<(), fmt::Error> {
let mut out = String::new();
out += &format!("{:?}", self.peer_scope);
if let Some(pt) = self.protocol_type {
out += &format!("+{:?}", pt);
}
if let Some(at) = self.address_type {
out += &format!("+{:?}", at);
}
write!(f, "[{}]", out)
2021-12-21 00:12:30 +00:00
}
}
2021-12-24 23:02:53 +00:00
pub trait MatchesDialInfoFilter {
fn matches_filter(&self, filter: &DialInfoFilter) -> bool;
}
2022-02-07 02:18:42 +00:00
#[derive(Clone, Default, Debug, PartialEq, PartialOrd, Ord, Eq, Serialize, Deserialize)]
2021-11-22 16:28:30 +00:00
pub struct DialInfoUDP {
2021-12-21 00:12:30 +00:00
pub socket_address: SocketAddress,
2021-11-22 16:28:30 +00:00
}
2022-02-07 02:18:42 +00:00
#[derive(Clone, Default, Debug, PartialEq, PartialOrd, Ord, Eq, Serialize, Deserialize)]
2021-11-22 16:28:30 +00:00
pub struct DialInfoTCP {
2021-12-21 00:12:30 +00:00
pub socket_address: SocketAddress,
2021-11-22 16:28:30 +00:00
}
2022-02-07 02:18:42 +00:00
#[derive(Clone, Default, Debug, PartialEq, PartialOrd, Ord, Eq, Serialize, Deserialize)]
2021-11-22 16:28:30 +00:00
pub struct DialInfoWS {
2021-12-21 00:12:30 +00:00
pub socket_address: SocketAddress,
pub request: String,
2021-11-22 16:28:30 +00:00
}
2022-02-07 02:18:42 +00:00
#[derive(Clone, Default, Debug, PartialEq, PartialOrd, Ord, Eq, Serialize, Deserialize)]
2021-11-22 16:28:30 +00:00
pub struct DialInfoWSS {
2021-12-21 00:12:30 +00:00
pub socket_address: SocketAddress,
pub request: String,
2021-11-22 16:28:30 +00:00
}
2022-02-07 02:18:42 +00:00
#[derive(Clone, Debug, PartialEq, PartialOrd, Ord, Eq, Serialize, Deserialize)]
#[serde(tag = "kind")]
2021-12-22 03:20:55 +00:00
// The derived ordering here is the order of preference, lower is preferred for connections
// Must match ProtocolType order
2021-11-22 16:28:30 +00:00
pub enum DialInfo {
UDP(DialInfoUDP),
TCP(DialInfoTCP),
WS(DialInfoWS),
WSS(DialInfoWSS),
}
2021-12-21 00:12:30 +00:00
impl Default for DialInfo {
fn default() -> Self {
DialInfo::UDP(DialInfoUDP::default())
}
}
impl fmt::Display for DialInfo {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> Result<(), fmt::Error> {
match self {
DialInfo::UDP(di) => write!(f, "udp|{}", di.socket_address),
DialInfo::TCP(di) => write!(f, "tcp|{}", di.socket_address),
DialInfo::WS(di) => write!(f, "ws|{}|{}", di.socket_address, di.request),
DialInfo::WSS(di) => write!(f, "wss|{}|{}", di.socket_address, di.request),
}
}
}
impl FromStr for DialInfo {
type Err = VeilidAPIError;
fn from_str(s: &str) -> Result<DialInfo, VeilidAPIError> {
2021-12-24 23:02:53 +00:00
let (proto, rest) = s
.split_once('|')
.ok_or_else(|| parse_error!("DialInfo::from_str missing protocol '|' separator", s))?;
2021-12-21 00:12:30 +00:00
match proto {
"udp" => {
let socket_address = SocketAddress::from_str(rest)?;
Ok(DialInfo::udp(socket_address))
}
"tcp" => {
let socket_address = SocketAddress::from_str(rest)?;
Ok(DialInfo::tcp(socket_address))
}
"ws" => {
2022-01-05 21:58:18 +00:00
let (sa, rest) = rest.split_once('|').ok_or_else(|| {
2021-12-24 23:02:53 +00:00
parse_error!("DialInfo::from_str missing socket address '|' separator", s)
2021-12-21 00:12:30 +00:00
})?;
let socket_address = SocketAddress::from_str(sa)?;
2022-01-05 21:58:18 +00:00
DialInfo::try_ws(socket_address, format!("ws://{}", rest))
2021-12-21 00:12:30 +00:00
}
"wss" => {
2022-01-05 21:58:18 +00:00
let (sa, rest) = rest.split_once('|').ok_or_else(|| {
2021-12-24 23:02:53 +00:00
parse_error!("DialInfo::from_str missing socket address '|' separator", s)
2021-12-21 00:12:30 +00:00
})?;
let socket_address = SocketAddress::from_str(sa)?;
2022-01-05 21:58:18 +00:00
DialInfo::try_wss(socket_address, format!("wss://{}", rest))
2021-12-21 00:12:30 +00:00
}
2021-12-24 23:02:53 +00:00
_ => Err(parse_error!("DialInfo::from_str has invalid scheme", s)),
2021-12-21 00:12:30 +00:00
}
}
}
2021-11-22 16:28:30 +00:00
impl DialInfo {
2021-12-21 00:12:30 +00:00
pub fn udp_from_socketaddr(socket_addr: SocketAddr) -> Self {
2021-11-22 16:28:30 +00:00
Self::UDP(DialInfoUDP {
2021-12-21 00:12:30 +00:00
socket_address: SocketAddress::from_socket_addr(socket_addr).to_canonical(),
2021-11-22 16:28:30 +00:00
})
}
2021-12-21 00:12:30 +00:00
pub fn tcp_from_socketaddr(socket_addr: SocketAddr) -> Self {
2021-11-22 16:28:30 +00:00
Self::TCP(DialInfoTCP {
2021-12-21 00:12:30 +00:00
socket_address: SocketAddress::from_socket_addr(socket_addr).to_canonical(),
2021-11-22 16:28:30 +00:00
})
}
2021-12-21 00:12:30 +00:00
pub fn udp(socket_address: SocketAddress) -> Self {
Self::UDP(DialInfoUDP {
socket_address: socket_address.to_canonical(),
})
2021-11-22 16:28:30 +00:00
}
2021-12-21 00:12:30 +00:00
pub fn tcp(socket_address: SocketAddress) -> Self {
Self::TCP(DialInfoTCP {
socket_address: socket_address.to_canonical(),
})
2021-11-22 16:28:30 +00:00
}
2021-12-21 00:12:30 +00:00
pub fn try_ws(socket_address: SocketAddress, url: String) -> Result<Self, VeilidAPIError> {
let split_url = SplitUrl::from_str(&url)
.map_err(|e| parse_error!(format!("unable to split WS url: {}", e), url))?;
if split_url.scheme != "ws" || !url.starts_with("ws://") {
return Err(parse_error!("incorrect scheme for WS dialinfo", url));
}
let url_port = split_url.port.unwrap_or(80u16);
if url_port != socket_address.port() {
return Err(parse_error!(
"socket address port doesn't match url port",
url
));
}
Ok(Self::WS(DialInfoWS {
socket_address: socket_address.to_canonical(),
request: url[5..].to_string(),
}))
}
pub fn try_wss(socket_address: SocketAddress, url: String) -> Result<Self, VeilidAPIError> {
let split_url = SplitUrl::from_str(&url)
.map_err(|e| parse_error!(format!("unable to split WSS url: {}", e), url))?;
if split_url.scheme != "wss" || !url.starts_with("wss://") {
return Err(parse_error!("incorrect scheme for WSS dialinfo", url));
}
let url_port = split_url.port.unwrap_or(443u16);
if url_port != socket_address.port() {
return Err(parse_error!(
"socket address port doesn't match url port",
url
));
}
2021-12-24 23:02:53 +00:00
if Address::from_str(&split_url.host).is_ok() {
2021-12-21 00:12:30 +00:00
return Err(parse_error!(
"WSS url can not use address format, only hostname format",
url
));
}
Ok(Self::WSS(DialInfoWSS {
socket_address: socket_address.to_canonical(),
request: url[6..].to_string(),
}))
2021-11-22 16:28:30 +00:00
}
pub fn protocol_type(&self) -> ProtocolType {
match self {
Self::UDP(_) => ProtocolType::UDP,
Self::TCP(_) => ProtocolType::TCP,
Self::WS(_) => ProtocolType::WS,
Self::WSS(_) => ProtocolType::WSS,
}
}
2021-12-22 03:20:55 +00:00
pub fn address_type(&self) -> AddressType {
self.socket_address().address_type()
2021-11-22 16:28:30 +00:00
}
2021-12-21 00:12:30 +00:00
pub fn socket_address(&self) -> SocketAddress {
2021-11-22 16:28:30 +00:00
match self {
2021-12-21 00:12:30 +00:00
Self::UDP(di) => di.socket_address,
Self::TCP(di) => di.socket_address,
Self::WS(di) => di.socket_address,
Self::WSS(di) => di.socket_address,
2021-11-22 16:28:30 +00:00
}
}
2021-12-21 00:12:30 +00:00
pub fn to_ip_addr(&self) -> IpAddr {
2021-11-22 16:28:30 +00:00
match self {
2021-12-21 00:12:30 +00:00
Self::UDP(di) => di.socket_address.to_ip_addr(),
Self::TCP(di) => di.socket_address.to_ip_addr(),
Self::WS(di) => di.socket_address.to_ip_addr(),
Self::WSS(di) => di.socket_address.to_ip_addr(),
2021-11-22 16:28:30 +00:00
}
}
pub fn port(&self) -> u16 {
match self {
2021-12-21 00:12:30 +00:00
Self::UDP(di) => di.socket_address.port,
Self::TCP(di) => di.socket_address.port,
Self::WS(di) => di.socket_address.port,
Self::WSS(di) => di.socket_address.port,
2021-11-22 16:28:30 +00:00
}
}
2022-01-01 03:09:30 +00:00
pub fn set_port(&mut self, port: u16) {
match self {
Self::UDP(di) => di.socket_address.port = port,
Self::TCP(di) => di.socket_address.port = port,
Self::WS(di) => di.socket_address.port = port,
Self::WSS(di) => di.socket_address.port = port,
}
}
2021-12-21 00:12:30 +00:00
pub fn to_socket_addr(&self) -> SocketAddr {
2021-11-22 16:28:30 +00:00
match self {
2021-12-21 00:12:30 +00:00
Self::UDP(di) => di.socket_address.to_socket_addr(),
Self::TCP(di) => di.socket_address.to_socket_addr(),
Self::WS(di) => di.socket_address.to_socket_addr(),
Self::WSS(di) => di.socket_address.to_socket_addr(),
2021-11-22 16:28:30 +00:00
}
}
2021-12-25 15:58:43 +00:00
pub fn to_peer_address(&self) -> PeerAddress {
match self {
Self::UDP(di) => PeerAddress::new(di.socket_address, ProtocolType::UDP),
Self::TCP(di) => PeerAddress::new(di.socket_address, ProtocolType::TCP),
Self::WS(di) => PeerAddress::new(di.socket_address, ProtocolType::WS),
Self::WSS(di) => PeerAddress::new(di.socket_address, ProtocolType::WSS),
}
}
2021-12-21 00:12:30 +00:00
pub fn request(&self) -> Option<String> {
2021-11-22 16:28:30 +00:00
match self {
2021-12-21 00:12:30 +00:00
Self::UDP(_) => None,
Self::TCP(_) => None,
Self::WS(di) => Some(format!("ws://{}", di.request)),
Self::WSS(di) => Some(format!("wss://{}", di.request)),
2021-11-22 16:28:30 +00:00
}
}
2021-12-22 03:20:55 +00:00
pub fn is_global(&self) -> bool {
self.socket_address().address().is_global()
2021-11-22 16:28:30 +00:00
}
2021-12-22 03:20:55 +00:00
pub fn is_local(&self) -> bool {
self.socket_address().address().is_local()
2021-11-22 16:28:30 +00:00
}
2021-12-21 00:12:30 +00:00
pub fn is_valid(&self) -> bool {
let socket_address = self.socket_address();
let address = socket_address.address();
let port = socket_address.port();
2021-12-22 03:20:55 +00:00
(address.is_global() || address.is_local()) && port > 0
}
pub fn matches_peer_scope(&self, scope: PeerScope) -> bool {
match scope {
PeerScope::All => true,
PeerScope::Global => self.is_global(),
PeerScope::Local => self.is_local(),
}
2021-11-22 16:28:30 +00:00
}
2021-12-24 01:34:52 +00:00
pub fn make_filter(&self, scoped: bool) -> DialInfoFilter {
DialInfoFilter {
peer_scope: if scoped {
if self.is_global() {
PeerScope::Global
} else if self.is_local() {
PeerScope::Local
} else {
PeerScope::All
}
} else {
PeerScope::All
},
protocol_type: Some(self.protocol_type()),
address_type: Some(self.address_type()),
}
}
2021-11-22 16:28:30 +00:00
}
2021-12-24 23:02:53 +00:00
impl MatchesDialInfoFilter for DialInfo {
fn matches_filter(&self, filter: &DialInfoFilter) -> bool {
if !self.matches_peer_scope(filter.peer_scope) {
return false;
}
if let Some(pt) = filter.protocol_type {
if self.protocol_type() != pt {
return false;
}
}
if let Some(at) = filter.address_type {
if self.address_type() != at {
return false;
}
}
true
}
}
2021-11-22 16:28:30 +00:00
//////////////////////////////////////////////////////////////////////////
2022-02-07 02:18:42 +00:00
#[derive(Clone, Copy, Debug, Eq, PartialEq, Ord, PartialOrd, Serialize, Deserialize)]
2021-11-22 16:28:30 +00:00
pub enum PeerScope {
All,
2021-12-08 03:09:45 +00:00
Global,
Local,
2021-11-22 16:28:30 +00:00
}
2021-12-24 01:34:52 +00:00
impl Default for PeerScope {
fn default() -> Self {
PeerScope::All
}
}
2021-11-22 16:28:30 +00:00
2022-02-07 02:18:42 +00:00
#[derive(Clone, Debug, Default, Serialize, Deserialize)]
2021-11-22 16:28:30 +00:00
pub struct PeerInfo {
pub node_id: NodeId,
pub dial_infos: Vec<DialInfo>,
}
2022-02-07 02:18:42 +00:00
#[derive(Copy, Clone, Debug, PartialEq, PartialOrd, Eq, Ord, Hash, Serialize, Deserialize)]
2021-11-22 16:28:30 +00:00
pub struct PeerAddress {
2021-12-21 00:12:30 +00:00
pub socket_address: SocketAddress,
2021-11-22 16:28:30 +00:00
pub protocol_type: ProtocolType,
}
impl PeerAddress {
2021-12-21 00:12:30 +00:00
pub fn new(socket_address: SocketAddress, protocol_type: ProtocolType) -> Self {
2021-11-22 16:28:30 +00:00
Self {
2021-12-22 03:20:55 +00:00
socket_address: socket_address.to_canonical(),
2021-11-27 17:44:21 +00:00
protocol_type,
2021-11-22 16:28:30 +00:00
}
}
2021-12-21 00:12:30 +00:00
pub fn to_socket_addr(&self) -> SocketAddr {
self.socket_address.to_socket_addr()
2021-11-22 16:28:30 +00:00
}
2021-12-21 00:12:30 +00:00
2021-12-22 03:20:55 +00:00
pub fn address_type(&self) -> AddressType {
self.socket_address.address_type()
2021-11-22 16:28:30 +00:00
}
}
2022-02-07 02:18:42 +00:00
#[derive(Copy, Clone, Debug, PartialEq, Eq, PartialOrd, Ord, Hash, Serialize, Deserialize)]
2021-11-22 16:28:30 +00:00
pub struct ConnectionDescriptor {
pub remote: PeerAddress,
2021-12-22 03:20:55 +00:00
pub local: Option<SocketAddress>,
2021-11-22 16:28:30 +00:00
}
impl ConnectionDescriptor {
2021-12-22 03:20:55 +00:00
pub fn new(remote: PeerAddress, local: SocketAddress) -> Self {
2021-11-22 16:28:30 +00:00
Self {
2021-11-27 17:44:21 +00:00
remote,
2021-11-22 16:28:30 +00:00
local: Some(local),
}
}
pub fn new_no_local(remote: PeerAddress) -> Self {
Self {
2021-11-27 17:44:21 +00:00
remote,
2021-11-22 16:28:30 +00:00
local: None,
}
}
pub fn protocol_type(&self) -> ProtocolType {
self.remote.protocol_type
}
2021-12-22 03:20:55 +00:00
pub fn address_type(&self) -> AddressType {
self.remote.address_type()
2021-11-22 16:28:30 +00:00
}
2021-12-24 01:34:52 +00:00
pub fn matches_peer_scope(&self, scope: PeerScope) -> bool {
match scope {
PeerScope::All => true,
PeerScope::Global => self.remote.socket_address.address().is_global(),
PeerScope::Local => self.remote.socket_address.address().is_local(),
}
}
2021-12-24 23:02:53 +00:00
}
impl MatchesDialInfoFilter for ConnectionDescriptor {
fn matches_filter(&self, filter: &DialInfoFilter) -> bool {
2021-12-24 01:34:52 +00:00
if !self.matches_peer_scope(filter.peer_scope) {
return false;
}
if let Some(pt) = filter.protocol_type {
if self.protocol_type() != pt {
return false;
}
}
if let Some(at) = filter.address_type {
if self.address_type() != at {
return false;
}
}
true
}
2021-11-22 16:28:30 +00:00
}
//////////////////////////////////////////////////////////////////////////
2022-02-07 02:18:42 +00:00
#[derive(Clone, Debug, Default, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
2022-01-01 03:09:30 +00:00
pub struct NodeDialInfo {
2021-11-22 16:28:30 +00:00
pub node_id: NodeId,
pub dial_info: DialInfo,
}
2022-01-01 03:09:30 +00:00
impl fmt::Display for NodeDialInfo {
2021-12-21 00:12:30 +00:00
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> Result<(), fmt::Error> {
write!(f, "{}@{}", self.node_id, self.dial_info)
2021-11-22 16:28:30 +00:00
}
}
2022-01-01 03:09:30 +00:00
impl FromStr for NodeDialInfo {
2021-12-21 00:12:30 +00:00
type Err = VeilidAPIError;
2022-01-01 03:09:30 +00:00
fn from_str(s: &str) -> Result<NodeDialInfo, VeilidAPIError> {
2021-12-21 00:12:30 +00:00
// split out node id from the dial info
2022-01-01 03:09:30 +00:00
let (node_id_str, rest) = s
.split_once('@')
.ok_or_else(|| parse_error!("NodeDialInfo::from_str missing @ node id separator", s))?;
2021-12-21 00:12:30 +00:00
// parse out node id
let node_id = NodeId::new(DHTKey::try_decode(node_id_str).map_err(|e| {
parse_error!(
2022-01-01 03:09:30 +00:00
format!("NodeDialInfo::from_str couldn't parse node id: {}", e),
2021-12-21 00:12:30 +00:00
s
)
})?);
// parse out dial info
let dial_info = DialInfo::from_str(rest)?;
2022-01-01 03:09:30 +00:00
// return completed NodeDialInfo
Ok(NodeDialInfo { node_id, dial_info })
2021-11-22 16:28:30 +00:00
}
}
2022-02-07 02:18:42 +00:00
#[derive(Clone, Debug, Default, Serialize, Deserialize)]
2021-11-22 16:28:30 +00:00
pub struct LatencyStats {
pub fastest: u64, // fastest latency in the ROLLING_LATENCIES_SIZE last latencies
pub average: u64, // average latency over the ROLLING_LATENCIES_SIZE last latencies
pub slowest: u64, // slowest latency in the ROLLING_LATENCIES_SIZE last latencies
}
2022-02-07 02:18:42 +00:00
#[derive(Clone, Debug, Default, Serialize, Deserialize)]
pub struct TransferStatsDownUp {
pub down: TransferStats,
pub up: TransferStats,
}
2022-02-07 02:18:42 +00:00
#[derive(Clone, Debug, Default, Serialize, Deserialize)]
2021-11-22 16:28:30 +00:00
pub struct TransferStats {
pub total: u64, // total amount transferred ever
pub maximum: u64, // maximum rate over the ROLLING_TRANSFERS_SIZE last amounts
pub average: u64, // average rate over the ROLLING_TRANSFERS_SIZE last amounts
pub minimum: u64, // minimum rate over the ROLLING_TRANSFERS_SIZE last amounts
}
2022-02-07 02:18:42 +00:00
#[derive(Clone, Debug, Default, Serialize, Deserialize)]
2021-11-22 16:28:30 +00:00
pub struct PingStats {
pub in_flight: u32, // number of pings issued that have yet to be answered
pub total_sent: u32, // number of pings that have been sent in the total_time range
pub total_returned: u32, // number of pings that have been returned by the node in the total_time range
pub consecutive_pongs: u32, // number of pongs that have been received and returned consecutively without a lost ping
pub last_pinged: Option<u64>, // when the peer was last pinged
pub first_consecutive_pong_time: Option<u64>, // the timestamp of the first pong in a series of consecutive pongs
pub recent_lost_pings: u32, // number of pings that have been lost since we lost reliability
}
2022-02-07 02:18:42 +00:00
#[derive(Clone, Debug, Default, Serialize, Deserialize)]
2021-11-22 16:28:30 +00:00
pub struct PeerStats {
pub time_added: u64, // when the peer was added to the routing table
2021-12-14 14:48:33 +00:00
pub last_seen: Option<u64>, // when the peer was last seen for any reason, including when we first attempted to reach out to it
pub ping_stats: PingStats, // information about pings
2021-11-22 16:28:30 +00:00
pub latency: Option<LatencyStats>, // latencies for communications with the peer
pub transfer: TransferStatsDownUp, // Stats for communications with the peer
2021-12-14 14:48:33 +00:00
pub node_info: Option<NodeInfo>, // Last known node info
2021-11-22 16:28:30 +00:00
}
cfg_if! {
if #[cfg(target_arch = "wasm32")] {
pub type ValueChangeCallback =
Arc<dyn Fn(ValueKey, Vec<u8>) -> SystemPinBoxFuture<()> + 'static>;
} else {
pub type ValueChangeCallback =
Arc<dyn Fn(ValueKey, Vec<u8>) -> SystemPinBoxFuture<()> + Send + Sync + 'static>;
}
}
2022-02-07 02:18:42 +00:00
#[derive(Clone, Debug, PartialOrd, PartialEq, Eq, Ord, Serialize, Deserialize)]
2021-11-22 16:28:30 +00:00
pub enum TunnelMode {
Raw,
Turn,
}
type TunnelId = u64;
2022-02-07 02:18:42 +00:00
#[derive(Clone, Debug, Serialize, Deserialize)]
2021-11-22 16:28:30 +00:00
pub struct TunnelEndpoint {
pub node_id: NodeId, // the node id of the tunnel endpoint
pub dial_info: Vec<DialInfo>, // multiple ways of how to get to the node
pub mode: TunnelMode,
}
impl Default for TunnelEndpoint {
fn default() -> Self {
Self {
node_id: NodeId::default(),
dial_info: Vec::new(),
mode: TunnelMode::Raw,
}
}
}
2022-02-07 02:18:42 +00:00
#[derive(Clone, Debug, Default, Serialize, Deserialize)]
2021-11-22 16:28:30 +00:00
pub struct FullTunnel {
pub id: TunnelId,
pub timeout: u64,
pub local: TunnelEndpoint,
pub remote: TunnelEndpoint,
}
2022-02-07 02:18:42 +00:00
#[derive(Clone, Debug, Default, Serialize, Deserialize)]
2021-11-22 16:28:30 +00:00
pub struct PartialTunnel {
pub id: TunnelId,
pub timeout: u64,
pub local: TunnelEndpoint,
}
/////////////////////////////////////////////////////////////////////////////////////////////////////
2022-02-07 02:18:42 +00:00
#[derive(Clone, Debug, Default, Serialize, Deserialize)]
2021-11-22 16:28:30 +00:00
pub struct RouteHopSpec {
2022-01-01 03:09:30 +00:00
pub dial_info: NodeDialInfo,
2021-11-22 16:28:30 +00:00
}
2022-02-07 02:18:42 +00:00
#[derive(Clone, Debug, Default, Serialize, Deserialize)]
2021-11-22 16:28:30 +00:00
pub struct PrivateRouteSpec {
//
pub public_key: DHTKey,
pub secret_key: DHTKeySecret,
pub hops: Vec<RouteHopSpec>,
}
2022-02-07 02:18:42 +00:00
#[derive(Clone, Debug, Default, Serialize, Deserialize)]
2021-11-22 16:28:30 +00:00
pub struct SafetyRouteSpec {
pub public_key: DHTKey,
pub secret_key: DHTKeySecret,
pub hops: Vec<RouteHopSpec>,
}
impl SafetyRouteSpec {
pub fn new() -> Self {
let (pk, sk) = generate_secret();
SafetyRouteSpec {
public_key: pk,
secret_key: sk,
hops: Vec::new(),
}
}
}
2022-02-07 02:18:42 +00:00
#[derive(Clone, Debug, Default, Serialize, Deserialize)]
2021-11-22 16:28:30 +00:00
pub struct RoutingContextOptions {
pub safety_route_spec: Option<SafetyRouteSpec>,
pub private_route_spec: Option<PrivateRouteSpec>,
}
/////////////////////////////////////////////////////////////////////////////////////////////////////
2022-02-07 02:18:42 +00:00
#[derive(Clone, Debug, Default, Serialize, Deserialize)]
2021-11-22 16:28:30 +00:00
pub struct SearchDHTAnswer {
pub node_id: NodeId,
pub dial_info: Vec<DialInfo>,
}
/////////////////////////////////////////////////////////////////////////////////////////////////////
pub struct RoutingContextInner {
api: VeilidAPI,
options: RoutingContextOptions,
}
impl Drop for RoutingContextInner {
fn drop(&mut self) {
// self.api
// .borrow_mut()
// .routing_contexts
// //.remove(&self.id);
}
}
#[derive(Clone)]
pub struct RoutingContext {
inner: Arc<Mutex<RoutingContextInner>>,
}
impl RoutingContext {
fn new(api: VeilidAPI, options: RoutingContextOptions) -> Self {
Self {
2021-11-27 17:44:21 +00:00
inner: Arc::new(Mutex::new(RoutingContextInner { api, options })),
2021-11-22 16:28:30 +00:00
}
}
pub fn api(&self) -> VeilidAPI {
self.inner.lock().api.clone()
}
///////////////////////////////////
///
pub async fn get_value(&self, _value_key: ValueKey) -> Result<Vec<u8>, VeilidAPIError> {
panic!("unimplemented");
}
pub async fn set_value(
&self,
_value_key: ValueKey,
_value: Vec<u8>,
) -> Result<bool, VeilidAPIError> {
panic!("unimplemented");
}
pub async fn watch_value(
&self,
_value_key: ValueKey,
_callback: ValueChangeCallback,
) -> Result<bool, VeilidAPIError> {
panic!("unimplemented");
}
pub async fn cancel_watch_value(&self, _value_key: ValueKey) -> Result<bool, VeilidAPIError> {
panic!("unimplemented");
}
pub async fn find_block(&self, _block_id: BlockId) -> Result<Vec<u8>, VeilidAPIError> {
panic!("unimplemented");
}
pub async fn supply_block(&self, _block_id: BlockId) -> Result<bool, VeilidAPIError> {
panic!("unimplemented");
}
pub async fn signal(&self, _data: Vec<u8>) -> Result<bool, VeilidAPIError> {
panic!("unimplemented");
}
}
/////////////////////////////////////////////////////////////////////////////////////////////////////
struct VeilidAPIInner {
2022-02-07 02:18:42 +00:00
context: Option<VeilidCoreContext>,
2021-12-08 03:09:45 +00:00
}
impl fmt::Debug for VeilidAPIInner {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
2022-02-07 02:18:42 +00:00
write!(f, "VeilidAPIInner")
2021-12-08 03:09:45 +00:00
}
2021-11-22 16:28:30 +00:00
}
impl Drop for VeilidAPIInner {
fn drop(&mut self) {
2022-02-07 02:18:42 +00:00
if let Some(context) = self.context.take() {
intf::spawn_local(api_shutdown(context)).detach();
2021-11-22 16:28:30 +00:00
}
}
}
2021-12-08 03:09:45 +00:00
#[derive(Clone, Debug)]
2021-11-22 16:28:30 +00:00
pub struct VeilidAPI {
inner: Arc<Mutex<VeilidAPIInner>>,
}
impl VeilidAPI {
2022-02-07 02:18:42 +00:00
pub(crate) fn new(context: VeilidCoreContext) -> Self {
2021-11-22 16:28:30 +00:00
Self {
2022-02-07 02:18:42 +00:00
inner: Arc::new(Mutex::new(VeilidAPIInner {
context: Some(context),
})),
2021-11-22 16:28:30 +00:00
}
}
2022-02-07 02:18:42 +00:00
pub async fn shutdown(self) {
let context = { self.inner.lock().context.take() };
if let Some(context) = context {
api_shutdown(context).await;
2021-12-08 03:09:45 +00:00
}
2021-11-22 16:28:30 +00:00
}
2022-02-07 02:18:42 +00:00
pub fn is_shutdown(&self) -> bool {
self.inner.lock().context.is_none()
2021-11-22 16:28:30 +00:00
}
2022-02-07 02:18:42 +00:00
////////////////////////////////////////////////////////////////
// Accessors
pub fn config(&self) -> Result<VeilidConfig, VeilidAPIError> {
let inner = self.inner.lock();
if let Some(context) = &inner.context {
return Ok(context.config.clone());
}
Err(VeilidAPIError::NotInitialized)
2021-11-22 16:28:30 +00:00
}
2022-02-07 02:18:42 +00:00
pub fn crypto(&self) -> Result<Crypto, VeilidAPIError> {
let inner = self.inner.lock();
if let Some(context) = &inner.context {
return Ok(context.crypto.clone());
}
Err(VeilidAPIError::NotInitialized)
2021-11-22 16:28:30 +00:00
}
2022-02-07 02:18:42 +00:00
pub fn table_store(&self) -> Result<TableStore, VeilidAPIError> {
let inner = self.inner.lock();
if let Some(context) = &inner.context {
return Ok(context.table_store.clone());
}
Err(VeilidAPIError::NotInitialized)
2021-12-08 03:09:45 +00:00
}
2022-02-07 02:18:42 +00:00
pub fn block_store(&self) -> Result<BlockStore, VeilidAPIError> {
let inner = self.inner.lock();
if let Some(context) = &inner.context {
return Ok(context.block_store.clone());
}
Err(VeilidAPIError::NotInitialized)
2021-11-22 16:28:30 +00:00
}
2022-02-07 02:18:42 +00:00
pub fn protected_store(&self) -> Result<ProtectedStore, VeilidAPIError> {
let inner = self.inner.lock();
if let Some(context) = &inner.context {
return Ok(context.protected_store.clone());
2021-11-22 16:28:30 +00:00
}
2022-02-07 02:18:42 +00:00
Err(VeilidAPIError::NotInitialized)
2021-12-08 03:09:45 +00:00
}
2022-02-07 02:18:42 +00:00
pub fn attachment_manager(&self) -> Result<AttachmentManager, VeilidAPIError> {
let inner = self.inner.lock();
if let Some(context) = &inner.context {
return Ok(context.attachment_manager.clone());
}
Err(VeilidAPIError::NotInitialized)
}
pub fn network_manager(&self) -> Result<NetworkManager, VeilidAPIError> {
let inner = self.inner.lock();
if let Some(context) = &inner.context {
return Ok(context.attachment_manager.network_manager());
}
Err(VeilidAPIError::NotInitialized)
}
pub fn rpc_processor(&self) -> Result<RPCProcessor, VeilidAPIError> {
let inner = self.inner.lock();
if let Some(context) = &inner.context {
return Ok(context.attachment_manager.network_manager().rpc_processor());
}
Err(VeilidAPIError::NotInitialized)
2021-11-22 16:28:30 +00:00
}
////////////////////////////////////////////////////////////////
// Attach/Detach
// get a full copy of the current state
pub async fn get_state(&self) -> Result<VeilidState, VeilidAPIError> {
trace!("VeilidCore::get_state");
2021-12-08 03:09:45 +00:00
let attachment_manager = self.attachment_manager()?;
Ok(VeilidState {
attachment: attachment_manager.get_state(),
})
2021-11-22 16:28:30 +00:00
}
// connect to the network
2021-12-08 03:09:45 +00:00
pub async fn attach(&self) -> Result<(), VeilidAPIError> {
2021-11-22 16:28:30 +00:00
trace!("VeilidCore::attach");
2021-12-08 03:09:45 +00:00
let attachment_manager = self.attachment_manager()?;
2021-11-22 16:28:30 +00:00
attachment_manager.request_attach().await;
2021-12-08 03:09:45 +00:00
Ok(())
2021-11-22 16:28:30 +00:00
}
// disconnect from the network
2021-12-08 03:09:45 +00:00
pub async fn detach(&self) -> Result<(), VeilidAPIError> {
2021-11-22 16:28:30 +00:00
trace!("VeilidCore::detach");
2021-12-08 03:09:45 +00:00
let attachment_manager = self.attachment_manager()?;
2021-11-22 16:28:30 +00:00
attachment_manager.request_detach().await;
2021-12-08 03:09:45 +00:00
Ok(())
2021-11-22 16:28:30 +00:00
}
2022-02-01 03:47:17 +00:00
// Change api logging level if it is enabled
pub async fn change_api_log_level(&self, log_level: VeilidConfigLogLevel) {
ApiLogger::change_log_level(log_level.to_level_filter());
}
2021-11-22 16:28:30 +00:00
////////////////////////////////////////////////////////////////
// Direct Node Access (pretty much for testing only)
pub async fn info(&self, node_id: NodeId) -> Result<InfoAnswer, VeilidAPIError> {
2021-12-08 03:09:45 +00:00
let rpc = self.rpc_processor()?;
2021-11-22 16:28:30 +00:00
let routing_table = rpc.routing_table();
let node_ref = match routing_table.lookup_node_ref(node_id.key) {
2022-02-09 14:47:36 +00:00
None => return Err(VeilidAPIError::NodeNotFound { node_id }),
2021-11-22 16:28:30 +00:00
Some(nr) => nr,
};
let info_answer = rpc
.rpc_call_info(node_ref)
.await
.map_err(map_rpc_error!())?;
Ok(info_answer)
}
pub async fn validate_dial_info(
&self,
node_id: NodeId,
dial_info: DialInfo,
redirect: bool,
alternate_port: bool,
) -> Result<bool, VeilidAPIError> {
2021-12-08 03:09:45 +00:00
let rpc = self.rpc_processor()?;
2021-11-22 16:28:30 +00:00
let routing_table = rpc.routing_table();
let node_ref = match routing_table.lookup_node_ref(node_id.key) {
2022-02-09 14:47:36 +00:00
None => return Err(VeilidAPIError::NodeNotFound { node_id }),
2021-11-22 16:28:30 +00:00
Some(nr) => nr,
};
rpc.rpc_call_validate_dial_info(node_ref.clone(), dial_info, redirect, alternate_port)
.await
.map_err(map_rpc_error!())
}
pub async fn search_dht(&self, node_id: NodeId) -> Result<SearchDHTAnswer, VeilidAPIError> {
2021-12-08 03:09:45 +00:00
let rpc_processor = self.rpc_processor()?;
let config = self.config()?;
2021-11-22 16:28:30 +00:00
let (count, fanout, timeout) = {
let c = config.get();
(
c.network.dht.resolve_node_count,
c.network.dht.resolve_node_fanout,
2022-01-27 14:53:01 +00:00
c.network.dht.resolve_node_timeout_ms.map(ms_to_us),
2021-11-22 16:28:30 +00:00
)
};
let node_ref = rpc_processor
.search_dht_single_key(node_id.key, count, fanout, timeout)
.await
.map_err(map_rpc_error!())?;
let answer = node_ref.operate(|e| SearchDHTAnswer {
node_id: NodeId::new(node_ref.node_id()),
2021-12-22 03:20:55 +00:00
dial_info: e.dial_infos().to_vec(),
2021-11-22 16:28:30 +00:00
});
Ok(answer)
}
pub async fn search_dht_multi(
&self,
node_id: NodeId,
) -> Result<Vec<SearchDHTAnswer>, VeilidAPIError> {
2021-12-08 03:09:45 +00:00
let rpc_processor = self.rpc_processor()?;
let config = self.config()?;
2021-11-22 16:28:30 +00:00
let (count, fanout, timeout) = {
let c = config.get();
(
c.network.dht.resolve_node_count,
c.network.dht.resolve_node_fanout,
2022-01-27 14:53:01 +00:00
c.network.dht.resolve_node_timeout_ms.map(ms_to_us),
2021-11-22 16:28:30 +00:00
)
};
let node_refs = rpc_processor
.search_dht_multi_key(node_id.key, count, fanout, timeout)
.await
.map_err(map_rpc_error!())?;
let mut answer = Vec::<SearchDHTAnswer>::new();
for nr in node_refs {
let a = nr.operate(|e| SearchDHTAnswer {
node_id: NodeId::new(nr.node_id()),
2021-12-22 03:20:55 +00:00
dial_info: e.dial_infos().to_vec(),
2021-11-22 16:28:30 +00:00
});
answer.push(a);
}
Ok(answer)
}
////////////////////////////////////////////////////////////////
// Safety / Private Route Handling
pub async fn new_safety_route_spec(
&self,
_hops: u8,
) -> Result<SafetyRouteSpec, VeilidAPIError> {
panic!("unimplemented");
}
pub async fn new_private_route_spec(
&self,
_hops: u8,
) -> Result<PrivateRouteSpec, VeilidAPIError> {
panic!("unimplemented");
}
////////////////////////////////////////////////////////////////
// Routing Contexts
//
// Safety route specified here is for _this_ node's anonymity as a sender, used via the 'route' operation
// Private route specified here is for _this_ node's anonymity as a receiver, passed out via the 'respond_to' field for replies
pub async fn safe_private(
&self,
safety_route_spec: SafetyRouteSpec,
private_route_spec: PrivateRouteSpec,
) -> RoutingContext {
self.routing_context(RoutingContextOptions {
safety_route_spec: Some(safety_route_spec),
private_route_spec: Some(private_route_spec),
})
.await
}
pub async fn safe_public(&self, safety_route_spec: SafetyRouteSpec) -> RoutingContext {
self.routing_context(RoutingContextOptions {
safety_route_spec: Some(safety_route_spec),
private_route_spec: None,
})
.await
}
pub async fn unsafe_private(&self, private_route_spec: PrivateRouteSpec) -> RoutingContext {
self.routing_context(RoutingContextOptions {
safety_route_spec: None,
private_route_spec: Some(private_route_spec),
})
.await
}
pub async fn unsafe_public(&self) -> RoutingContext {
self.routing_context(RoutingContextOptions {
safety_route_spec: None,
private_route_spec: None,
})
.await
}
pub async fn routing_context(&self, options: RoutingContextOptions) -> RoutingContext {
RoutingContext::new(self.clone(), options)
}
////////////////////////////////////////////////////////////////
// Tunnel Building
pub async fn start_tunnel(
&self,
_endpoint_mode: TunnelMode,
_depth: u8,
) -> Result<PartialTunnel, VeilidAPIError> {
panic!("unimplemented");
}
pub async fn complete_tunnel(
&self,
_endpoint_mode: TunnelMode,
_depth: u8,
_partial_tunnel: PartialTunnel,
) -> Result<FullTunnel, VeilidAPIError> {
panic!("unimplemented");
}
pub async fn cancel_tunnel(&self, _tunnel_id: TunnelId) -> Result<bool, VeilidAPIError> {
panic!("unimplemented");
}
}