2022-12-30 08:23:19 +00:00
|
|
|
use crate::api;
|
2023-04-28 11:20:57 +00:00
|
|
|
use crate::api::err::Error;
|
2022-12-30 08:23:19 +00:00
|
|
|
use crate::api::method::query::Response;
|
|
|
|
use crate::api::opt::Endpoint;
|
|
|
|
use crate::api::ExtraFeatures;
|
|
|
|
use crate::api::Result;
|
|
|
|
use crate::api::Surreal;
|
2023-11-13 17:19:47 +00:00
|
|
|
use crate::dbs::Notification;
|
2023-04-25 14:22:34 +00:00
|
|
|
use crate::opt::from_value;
|
2022-12-30 08:23:19 +00:00
|
|
|
use crate::sql::Query;
|
|
|
|
use crate::sql::Value;
|
|
|
|
use flume::Receiver;
|
|
|
|
use flume::Sender;
|
|
|
|
use serde::de::DeserializeOwned;
|
|
|
|
use serde::Serialize;
|
|
|
|
use std::collections::BTreeMap;
|
|
|
|
use std::collections::HashSet;
|
|
|
|
use std::future::Future;
|
|
|
|
use std::marker::PhantomData;
|
|
|
|
use std::path::PathBuf;
|
|
|
|
use std::pin::Pin;
|
|
|
|
use std::sync::atomic::AtomicI64;
|
|
|
|
use std::sync::atomic::Ordering;
|
|
|
|
|
|
|
|
#[derive(Debug)]
|
|
|
|
#[allow(dead_code)] // used by the embedded and remote connections
|
|
|
|
pub(crate) struct Route {
|
|
|
|
pub(crate) request: (i64, Method, Param),
|
|
|
|
pub(crate) response: Sender<Result<DbResponse>>,
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Message router
|
|
|
|
#[derive(Debug)]
|
|
|
|
pub struct Router<C: api::Connection> {
|
|
|
|
pub(crate) conn: PhantomData<C>,
|
|
|
|
pub(crate) sender: Sender<Option<Route>>,
|
|
|
|
pub(crate) last_id: AtomicI64,
|
|
|
|
pub(crate) features: HashSet<ExtraFeatures>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<C> Router<C>
|
|
|
|
where
|
|
|
|
C: api::Connection,
|
|
|
|
{
|
|
|
|
pub(crate) fn next_id(&self) -> i64 {
|
|
|
|
self.last_id.fetch_add(1, Ordering::SeqCst)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<C> Drop for Router<C>
|
|
|
|
where
|
|
|
|
C: api::Connection,
|
|
|
|
{
|
|
|
|
fn drop(&mut self) {
|
|
|
|
let _res = self.sender.send(None);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// The query method
|
|
|
|
#[derive(Debug, Clone, Copy, Serialize, PartialEq, Eq, Hash)]
|
|
|
|
#[serde(rename_all = "lowercase")]
|
|
|
|
pub enum Method {
|
|
|
|
/// Sends an authentication token to the server
|
|
|
|
Authenticate,
|
2023-09-28 09:17:29 +00:00
|
|
|
/// Performs a merge update operation
|
2022-12-30 08:23:19 +00:00
|
|
|
Merge,
|
|
|
|
/// Creates a record in a table
|
|
|
|
Create,
|
|
|
|
/// Deletes a record from a table
|
|
|
|
Delete,
|
|
|
|
/// Exports a database
|
|
|
|
Export,
|
|
|
|
/// Checks the health of the server
|
|
|
|
Health,
|
|
|
|
/// Imports a database
|
|
|
|
Import,
|
|
|
|
/// Invalidates a session
|
|
|
|
Invalidate,
|
|
|
|
/// Kills a live query
|
|
|
|
#[doc(hidden)] // Not supported yet
|
|
|
|
Kill,
|
|
|
|
/// Starts a live query
|
|
|
|
#[doc(hidden)] // Not supported yet
|
|
|
|
Live,
|
2023-09-28 09:17:29 +00:00
|
|
|
/// Performs a patch update operation
|
2022-12-30 08:23:19 +00:00
|
|
|
Patch,
|
|
|
|
/// Sends a raw query to the database
|
|
|
|
Query,
|
|
|
|
/// Selects a record or records from a table
|
|
|
|
Select,
|
|
|
|
/// Sets a parameter on the connection
|
|
|
|
Set,
|
|
|
|
/// Signs into the server
|
|
|
|
Signin,
|
|
|
|
/// Signs up on the server
|
|
|
|
Signup,
|
|
|
|
/// Removes a parameter from a connection
|
|
|
|
Unset,
|
2023-09-28 09:17:29 +00:00
|
|
|
/// Performs an update operation
|
2022-12-30 08:23:19 +00:00
|
|
|
Update,
|
|
|
|
/// Selects a namespace and database to use
|
|
|
|
Use,
|
|
|
|
/// Queries the version of the server
|
|
|
|
Version,
|
|
|
|
}
|
|
|
|
|
|
|
|
/// The database response sent from the router to the caller
|
|
|
|
#[derive(Debug)]
|
|
|
|
pub enum DbResponse {
|
|
|
|
/// The response sent for the `query` method
|
|
|
|
Query(Response),
|
|
|
|
/// The response sent for any method except `query`
|
|
|
|
Other(Value),
|
|
|
|
}
|
|
|
|
|
2023-12-12 13:51:43 +00:00
|
|
|
#[derive(Debug)]
|
|
|
|
#[allow(dead_code)] // used by ML model import and export functions
|
|
|
|
pub(crate) enum MlConfig {
|
|
|
|
Import,
|
|
|
|
Export {
|
|
|
|
name: String,
|
|
|
|
version: String,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2022-12-30 08:23:19 +00:00
|
|
|
/// Holds the parameters given to the caller
|
2023-11-13 17:19:47 +00:00
|
|
|
#[derive(Debug, Default)]
|
2022-12-30 08:23:19 +00:00
|
|
|
#[allow(dead_code)] // used by the embedded and remote connections
|
|
|
|
pub struct Param {
|
|
|
|
pub(crate) query: Option<(Query, BTreeMap<String, Value>)>,
|
|
|
|
pub(crate) other: Vec<Value>,
|
|
|
|
pub(crate) file: Option<PathBuf>,
|
2023-11-13 17:19:47 +00:00
|
|
|
pub(crate) bytes_sender: Option<channel::Sender<Result<Vec<u8>>>>,
|
|
|
|
pub(crate) notification_sender: Option<channel::Sender<Notification>>,
|
2023-12-12 13:51:43 +00:00
|
|
|
pub(crate) ml_config: Option<MlConfig>,
|
2022-12-30 08:23:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Param {
|
|
|
|
pub(crate) fn new(other: Vec<Value>) -> Self {
|
|
|
|
Self {
|
2023-08-09 12:15:20 +00:00
|
|
|
other,
|
2023-11-13 17:19:47 +00:00
|
|
|
..Default::default()
|
2022-12-30 08:23:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub(crate) fn query(query: Query, bindings: BTreeMap<String, Value>) -> Self {
|
|
|
|
Self {
|
|
|
|
query: Some((query, bindings)),
|
2023-11-13 17:19:47 +00:00
|
|
|
..Default::default()
|
2022-12-30 08:23:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub(crate) fn file(file: PathBuf) -> Self {
|
|
|
|
Self {
|
|
|
|
file: Some(file),
|
2023-11-13 17:19:47 +00:00
|
|
|
..Default::default()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub(crate) fn bytes_sender(send: channel::Sender<Result<Vec<u8>>>) -> Self {
|
|
|
|
Self {
|
|
|
|
bytes_sender: Some(send),
|
|
|
|
..Default::default()
|
2023-08-09 12:15:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-11-13 17:19:47 +00:00
|
|
|
pub(crate) fn notification_sender(send: channel::Sender<Notification>) -> Self {
|
2023-08-09 12:15:20 +00:00
|
|
|
Self {
|
2023-11-13 17:19:47 +00:00
|
|
|
notification_sender: Some(send),
|
|
|
|
..Default::default()
|
2022-12-30 08:23:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Connection trait implemented by supported protocols
|
|
|
|
pub trait Connection: Sized + Send + Sync + 'static {
|
|
|
|
/// Constructs a new client without connecting to the server
|
|
|
|
fn new(method: Method) -> Self;
|
|
|
|
|
|
|
|
/// Connect to the server
|
|
|
|
fn connect(
|
|
|
|
address: Endpoint,
|
|
|
|
capacity: usize,
|
|
|
|
) -> Pin<Box<dyn Future<Output = Result<Surreal<Self>>> + Send + Sync + 'static>>
|
|
|
|
where
|
|
|
|
Self: api::Connection;
|
|
|
|
|
|
|
|
/// Send a query to the server
|
|
|
|
#[allow(clippy::type_complexity)]
|
|
|
|
fn send<'r>(
|
|
|
|
&'r mut self,
|
|
|
|
router: &'r Router<Self>,
|
|
|
|
param: Param,
|
|
|
|
) -> Pin<Box<dyn Future<Output = Result<Receiver<Result<DbResponse>>>> + Send + Sync + 'r>>
|
|
|
|
where
|
|
|
|
Self: api::Connection;
|
|
|
|
|
|
|
|
/// Receive responses for all methods except `query`
|
2023-04-28 11:20:57 +00:00
|
|
|
fn recv(
|
2022-12-30 08:23:19 +00:00
|
|
|
&mut self,
|
|
|
|
receiver: Receiver<Result<DbResponse>>,
|
2023-04-28 11:20:57 +00:00
|
|
|
) -> Pin<Box<dyn Future<Output = Result<Value>> + Send + Sync + '_>> {
|
2023-04-25 14:22:34 +00:00
|
|
|
Box::pin(async move {
|
|
|
|
let response = receiver.into_recv_async().await?;
|
|
|
|
match response? {
|
2023-04-28 11:20:57 +00:00
|
|
|
DbResponse::Other(value) => Ok(value),
|
2023-04-25 14:22:34 +00:00
|
|
|
DbResponse::Query(..) => unreachable!(),
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
2022-12-30 08:23:19 +00:00
|
|
|
|
|
|
|
/// Receive the response of the `query` method
|
|
|
|
fn recv_query(
|
|
|
|
&mut self,
|
|
|
|
receiver: Receiver<Result<DbResponse>>,
|
2023-04-25 14:22:34 +00:00
|
|
|
) -> Pin<Box<dyn Future<Output = Result<Response>> + Send + Sync + '_>> {
|
|
|
|
Box::pin(async move {
|
|
|
|
let response = receiver.into_recv_async().await?;
|
|
|
|
match response? {
|
|
|
|
DbResponse::Query(results) => Ok(results),
|
|
|
|
DbResponse::Other(..) => unreachable!(),
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
2022-12-30 08:23:19 +00:00
|
|
|
|
|
|
|
/// Execute all methods except `query`
|
|
|
|
fn execute<'r, R>(
|
|
|
|
&'r mut self,
|
|
|
|
router: &'r Router<Self>,
|
|
|
|
param: Param,
|
|
|
|
) -> Pin<Box<dyn Future<Output = Result<R>> + Send + Sync + 'r>>
|
|
|
|
where
|
|
|
|
R: DeserializeOwned,
|
|
|
|
Self: api::Connection,
|
2023-04-28 11:20:57 +00:00
|
|
|
{
|
|
|
|
Box::pin(async move {
|
|
|
|
let rx = self.send(router, param).await?;
|
|
|
|
let value = self.recv(rx).await?;
|
|
|
|
from_value(value).map_err(Into::into)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Execute methods that return an optional single response
|
|
|
|
fn execute_opt<'r, R>(
|
|
|
|
&'r mut self,
|
|
|
|
router: &'r Router<Self>,
|
|
|
|
param: Param,
|
|
|
|
) -> Pin<Box<dyn Future<Output = Result<Option<R>>> + Send + Sync + 'r>>
|
|
|
|
where
|
|
|
|
R: DeserializeOwned,
|
|
|
|
Self: api::Connection,
|
|
|
|
{
|
|
|
|
Box::pin(async move {
|
|
|
|
let rx = self.send(router, param).await?;
|
|
|
|
match self.recv(rx).await? {
|
|
|
|
Value::None | Value::Null => Ok(None),
|
|
|
|
value => from_value(value).map_err(Into::into),
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Execute methods that return multiple responses
|
|
|
|
fn execute_vec<'r, R>(
|
|
|
|
&'r mut self,
|
|
|
|
router: &'r Router<Self>,
|
|
|
|
param: Param,
|
|
|
|
) -> Pin<Box<dyn Future<Output = Result<Vec<R>>> + Send + Sync + 'r>>
|
|
|
|
where
|
|
|
|
R: DeserializeOwned,
|
|
|
|
Self: api::Connection,
|
|
|
|
{
|
|
|
|
Box::pin(async move {
|
|
|
|
let rx = self.send(router, param).await?;
|
|
|
|
let value = match self.recv(rx).await? {
|
|
|
|
Value::None | Value::Null => Value::Array(Default::default()),
|
|
|
|
Value::Array(array) => Value::Array(array),
|
|
|
|
value => vec![value].into(),
|
|
|
|
};
|
|
|
|
from_value(value).map_err(Into::into)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Execute methods that return nothing
|
|
|
|
fn execute_unit<'r>(
|
|
|
|
&'r mut self,
|
|
|
|
router: &'r Router<Self>,
|
|
|
|
param: Param,
|
|
|
|
) -> Pin<Box<dyn Future<Output = Result<()>> + Send + Sync + 'r>>
|
|
|
|
where
|
|
|
|
Self: api::Connection,
|
|
|
|
{
|
|
|
|
Box::pin(async move {
|
|
|
|
let rx = self.send(router, param).await?;
|
|
|
|
match self.recv(rx).await? {
|
|
|
|
Value::None | Value::Null => Ok(()),
|
|
|
|
Value::Array(array) if array.is_empty() => Ok(()),
|
|
|
|
value => Err(Error::FromValue {
|
|
|
|
value,
|
|
|
|
error: "expected the database to return nothing".to_owned(),
|
|
|
|
}
|
|
|
|
.into()),
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Execute methods that return a raw value
|
|
|
|
fn execute_value<'r>(
|
|
|
|
&'r mut self,
|
|
|
|
router: &'r Router<Self>,
|
|
|
|
param: Param,
|
|
|
|
) -> Pin<Box<dyn Future<Output = Result<Value>> + Send + Sync + 'r>>
|
|
|
|
where
|
|
|
|
Self: api::Connection,
|
2022-12-30 08:23:19 +00:00
|
|
|
{
|
|
|
|
Box::pin(async move {
|
|
|
|
let rx = self.send(router, param).await?;
|
|
|
|
self.recv(rx).await
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Execute the `query` method
|
|
|
|
fn execute_query<'r>(
|
|
|
|
&'r mut self,
|
|
|
|
router: &'r Router<Self>,
|
|
|
|
param: Param,
|
|
|
|
) -> Pin<Box<dyn Future<Output = Result<Response>> + Send + Sync + 'r>>
|
|
|
|
where
|
|
|
|
Self: api::Connection,
|
|
|
|
{
|
|
|
|
Box::pin(async move {
|
|
|
|
let rx = self.send(router, param).await?;
|
|
|
|
self.recv_query(rx).await
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|