use tokio::sync::{mpsc, oneshot};
use super::conn::pooled::PooledConnection;
use crate::{
error::{Error, Result},
runtime::{AsyncJoinHandle, WorkerHandle},
};
pub(super) fn channel(handle: WorkerHandle) -> (ConnectionRequester, ConnectionRequestReceiver) {
let (sender, receiver) = mpsc::unbounded_channel();
(
ConnectionRequester {
sender,
_handle: handle,
},
ConnectionRequestReceiver { receiver },
)
}
#[derive(Clone, Debug)]
pub(super) struct ConnectionRequester {
sender: mpsc::UnboundedSender<ConnectionRequest>,
_handle: WorkerHandle,
}
impl ConnectionRequester {
pub(super) async fn request(&self) -> ConnectionRequestResult {
let (sender, receiver) = oneshot::channel();
self.sender
.send(ConnectionRequest {
sender,
warm_pool: false,
})
.unwrap();
receiver.await.unwrap()
}
pub(super) fn weak(&self) -> WeakConnectionRequester {
WeakConnectionRequester {
sender: self.sender.clone(),
}
}
}
#[derive(Clone, Debug)]
pub(super) struct WeakConnectionRequester {
sender: mpsc::UnboundedSender<ConnectionRequest>,
}
impl WeakConnectionRequester {
pub(super) async fn request_warm_pool(&self) -> Option<ConnectionRequestResult> {
let (sender, receiver) = oneshot::channel();
if self
.sender
.send(ConnectionRequest {
sender,
warm_pool: true,
})
.is_err()
{
return None;
}
receiver.await.ok()
}
}
#[derive(Debug)]
pub(super) struct ConnectionRequestReceiver {
receiver: mpsc::UnboundedReceiver<ConnectionRequest>,
}
impl ConnectionRequestReceiver {
pub(super) async fn recv(&mut self) -> Option<ConnectionRequest> {
self.receiver.recv().await
}
}
#[derive(Debug)]
pub(super) struct ConnectionRequest {
sender: oneshot::Sender<ConnectionRequestResult>,
warm_pool: bool,
}
impl ConnectionRequest {
pub(super) fn fulfill(
self,
result: ConnectionRequestResult,
) -> std::result::Result<(), ConnectionRequestResult> {
self.sender.send(result)
}
pub(super) fn is_warm_pool(&self) -> bool {
self.warm_pool
}
}
#[derive(Debug)]
pub(super) enum ConnectionRequestResult {
Pooled(Box<PooledConnection>),
Establishing(AsyncJoinHandle<Result<PooledConnection>>),
PoolCleared(Error),
PoolWarmed,
}
impl ConnectionRequestResult {
pub(super) fn unwrap_pooled_connection(self) -> PooledConnection {
match self {
ConnectionRequestResult::Pooled(c) => *c,
_ => panic!("attempted to unwrap pooled connection when was establishing"),
}
}
}