1use std::fmt::Debug;
2use std::marker::PhantomData;
3
4use async_trait::async_trait;
5use bonsaidb_core::api::{self, Api, ApiError, Infallible};
6use bonsaidb_core::arc_bytes::serde::Bytes;
7use bonsaidb_core::permissions::PermissionDenied;
8use bonsaidb_core::schema::{InsertError, InvalidNameError};
9
10use crate::{Backend, ConnectedClient, CustomServer, Error, NoBackend};
11
12#[async_trait]
14pub trait Handler<Api: api::Api, B: Backend = NoBackend>: Send + Sync {
15 async fn handle(session: HandlerSession<'_, B>, request: Api) -> HandlerResult<Api>;
19}
20
21pub struct HandlerSession<'a, B: Backend = NoBackend> {
24 pub server: &'a CustomServer<B>,
27 pub as_client: CustomServer<B>,
31 pub client: &'a ConnectedClient<B>,
33}
34
35#[async_trait]
36pub(crate) trait AnyHandler<B: Backend>: Send + Sync + Debug {
37 async fn handle(&self, session: HandlerSession<'_, B>, request: &[u8]) -> Result<Bytes, Error>;
38}
39
40pub(crate) struct AnyWrapper<D: Handler<A, B>, B: Backend, A: Api>(
41 pub(crate) PhantomData<(D, B, A)>,
42);
43
44impl<D, B, A> Debug for AnyWrapper<D, B, A>
45where
46 D: Handler<A, B>,
47 B: Backend,
48 A: Api,
49{
50 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
51 f.debug_tuple("AnyWrapper").finish()
52 }
53}
54
55#[async_trait]
56impl<T, B, A> AnyHandler<B> for AnyWrapper<T, B, A>
57where
58 B: Backend,
59 T: Handler<A, B>,
60 A: Api,
61{
62 async fn handle(&self, client: HandlerSession<'_, B>, request: &[u8]) -> Result<Bytes, Error> {
63 let request = pot::from_slice(request)?;
64 let response = match T::handle(client, request).await {
65 Ok(response) => Ok(response),
66 Err(HandlerError::Api(err)) => Err(err),
67 Err(HandlerError::Server(err)) => return Err(err),
68 };
69 Ok(Bytes::from(pot::to_vec(&response)?))
70 }
71}
72
73#[derive(thiserror::Error, Debug)]
75pub enum HandlerError<E: ApiError = Infallible> {
76 #[error("api error: {0}")]
78 Api(E),
79 #[error("server error: {0}")]
81 Server(#[from] Error),
82}
83
84impl<E: ApiError> From<PermissionDenied> for HandlerError<E> {
85 fn from(permission_denied: PermissionDenied) -> Self {
86 Self::Server(Error::from(permission_denied))
87 }
88}
89
90impl<E: ApiError> From<bonsaidb_core::Error> for HandlerError<E> {
91 fn from(err: bonsaidb_core::Error) -> Self {
92 Self::Server(Error::from(err))
93 }
94}
95
96impl<E: ApiError> From<bonsaidb_local::Error> for HandlerError<E> {
97 fn from(err: bonsaidb_local::Error) -> Self {
98 Self::Server(Error::from(err))
99 }
100}
101
102impl<E: ApiError> From<InvalidNameError> for HandlerError<E> {
103 fn from(err: InvalidNameError) -> Self {
104 Self::Server(Error::from(err))
105 }
106}
107
108#[cfg(feature = "websockets")]
109impl<E: ApiError> From<bincode::Error> for HandlerError<E> {
110 fn from(other: bincode::Error) -> Self {
111 Self::Server(Error::from(bonsaidb_local::Error::from(other)))
112 }
113}
114
115impl<E: ApiError> From<pot::Error> for HandlerError<E> {
116 fn from(other: pot::Error) -> Self {
117 Self::Server(Error::from(other))
118 }
119}
120
121impl<E: ApiError> From<std::io::Error> for HandlerError<E> {
122 fn from(err: std::io::Error) -> Self {
123 Self::Server(Error::from(err))
124 }
125}
126
127impl<T, E> From<InsertError<T>> for HandlerError<E>
128where
129 E: ApiError,
130{
131 fn from(error: InsertError<T>) -> Self {
132 Self::Server(Error::from(error.error))
133 }
134}
135
136pub type HandlerResult<Api> =
139 Result<<Api as api::Api>::Response, HandlerError<<Api as api::Api>::Error>>;