use std::collections::HashMap;
use std::marker::PhantomData;
use std::sync::Arc;
use std::time::Duration;
use bonsaidb_core::api;
use bonsaidb_core::api::ApiName;
use bonsaidb_core::networking::CURRENT_PROTOCOL_VERSION;
#[cfg(not(target_arch = "wasm32"))]
use fabruic::Certificate;
#[cfg(not(target_arch = "wasm32"))]
use tokio::runtime::Handle;
use url::Url;
use crate::client::{AnyApiCallback, ApiCallback};
#[cfg(not(target_arch = "wasm32"))]
use crate::BlockingClient;
use crate::{AsyncClient, Error};
pub struct Async;
#[cfg(not(target_arch = "wasm32"))]
pub struct Blocking;
#[must_use]
pub struct Builder<AsyncMode> {
url: Url,
protocol_version: &'static str,
custom_apis: HashMap<ApiName, Option<Arc<dyn AnyApiCallback>>>,
connect_timeout: Option<Duration>,
request_timeout: Option<Duration>,
#[cfg(not(target_arch = "wasm32"))]
certificate: Option<fabruic::Certificate>,
#[cfg(not(target_arch = "wasm32"))]
tokio: Option<Handle>,
mode: PhantomData<AsyncMode>,
}
impl<AsyncMode> Builder<AsyncMode> {
pub(crate) fn new(url: Url) -> Self {
Self {
url,
protocol_version: CURRENT_PROTOCOL_VERSION,
custom_apis: HashMap::new(),
request_timeout: None,
connect_timeout: None,
#[cfg(not(target_arch = "wasm32"))]
certificate: None,
#[cfg(not(target_arch = "wasm32"))]
tokio: None,
mode: PhantomData,
}
}
#[cfg(not(target_arch = "wasm32"))]
#[allow(clippy::missing_const_for_fn)]
pub fn with_runtime(mut self, handle: Handle) -> Self {
self.tokio = Some(handle);
self
}
pub fn with_api<Api: api::Api>(mut self) -> Self {
self.custom_apis.insert(Api::name(), None);
self
}
pub fn with_api_callback<Api: api::Api>(mut self, callback: ApiCallback<Api>) -> Self {
self.custom_apis
.insert(Api::name(), Some(Arc::new(callback)));
self
}
#[cfg(not(target_arch = "wasm32"))]
#[allow(clippy::missing_const_for_fn)]
pub fn with_certificate(mut self, certificate: Certificate) -> Self {
self.certificate = Some(certificate);
self
}
#[cfg(feature = "test-util")]
#[allow(clippy::missing_const_for_fn)]
pub fn with_protocol_version(mut self, version: &'static str) -> Self {
self.protocol_version = version;
self
}
pub fn with_request_timeout(mut self, timeout: impl Into<Duration>) -> Self {
self.request_timeout = Some(timeout.into());
self
}
pub fn with_connect_timeout(mut self, timeout: impl Into<Duration>) -> Self {
self.connect_timeout = Some(timeout.into());
self
}
fn finish_internal(self) -> Result<AsyncClient, Error> {
AsyncClient::new_from_parts(
self.url,
self.protocol_version,
self.custom_apis,
self.connect_timeout,
self.request_timeout,
#[cfg(not(target_arch = "wasm32"))]
self.certificate,
#[cfg(not(target_arch = "wasm32"))]
self.tokio.or_else(|| Handle::try_current().ok()),
)
}
}
#[cfg(not(target_arch = "wasm32"))]
impl Builder<Blocking> {
pub fn build(self) -> Result<BlockingClient, Error> {
self.finish_internal().map(BlockingClient)
}
}
impl Builder<Async> {
pub fn build(self) -> Result<AsyncClient, Error> {
self.finish_internal()
}
}