use crate::{
DatabaseConnection, DatabaseConnectionType, DbBackend, ExecResult, ProxyDatabaseTrait,
QueryResult, Statement, debug_print, error::*,
};
use std::{fmt::Debug, sync::Arc};
use tracing::instrument;
#[derive(Debug)]
pub struct ProxyDatabaseConnector;
#[derive(Debug)]
pub struct ProxyDatabaseConnection {
db_backend: DbBackend,
proxy: Arc<Box<dyn ProxyDatabaseTrait>>,
}
impl ProxyDatabaseConnector {
#[allow(unused_variables)]
pub fn accepts(string: &str) -> bool {
true
}
#[allow(unused_variables)]
#[instrument(level = "trace")]
pub fn connect(
db_type: DbBackend,
func: Arc<Box<dyn ProxyDatabaseTrait>>,
) -> Result<DatabaseConnection, DbErr> {
Ok(
DatabaseConnectionType::ProxyDatabaseConnection(Arc::new(
ProxyDatabaseConnection::new(db_type, func),
))
.into(),
)
}
}
impl ProxyDatabaseConnection {
pub fn new(db_backend: DbBackend, funcs: Arc<Box<dyn ProxyDatabaseTrait>>) -> Self {
Self {
db_backend,
proxy: funcs.to_owned(),
}
}
pub fn get_database_backend(&self) -> DbBackend {
self.db_backend
}
#[instrument(level = "trace")]
pub fn execute(&self, statement: Statement) -> Result<ExecResult, DbErr> {
debug_print!("{}", statement);
Ok(self.proxy.execute(statement)?.into())
}
#[instrument(level = "trace")]
pub fn query_one(&self, statement: Statement) -> Result<Option<QueryResult>, DbErr> {
debug_print!("{}", statement);
let result = self.proxy.query(statement)?;
if let Some(first) = result.first() {
return Ok(Some(QueryResult {
row: crate::QueryResultRow::Proxy(first.to_owned()),
}));
} else {
return Ok(None);
}
}
#[instrument(level = "trace")]
pub fn query_all(&self, statement: Statement) -> Result<Vec<QueryResult>, DbErr> {
debug_print!("{}", statement);
let result = self.proxy.query(statement)?;
Ok(result
.into_iter()
.map(|row| QueryResult {
row: crate::QueryResultRow::Proxy(row),
})
.collect())
}
#[instrument(level = "trace")]
pub fn begin(&self) {
self.proxy.begin()
}
#[instrument(level = "trace")]
pub fn commit(&self) {
self.proxy.commit()
}
#[instrument(level = "trace")]
pub fn rollback(&self) {
self.proxy.rollback()
}
#[instrument(level = "trace")]
pub fn start_rollback(&self) {
self.proxy.start_rollback()
}
pub fn ping(&self) -> Result<(), DbErr> {
self.proxy.ping()
}
}
impl
From<(
Arc<crate::ProxyDatabaseConnection>,
Statement,
Option<crate::metric::Callback>,
)> for crate::QueryStream
{
fn from(
(conn, stmt, metric_callback): (
Arc<crate::ProxyDatabaseConnection>,
Statement,
Option<crate::metric::Callback>,
),
) -> Self {
crate::QueryStream::build(stmt, crate::InnerConnection::Proxy(conn), metric_callback)
}
}
impl crate::DatabaseTransaction {
pub(crate) fn new_proxy(
inner: Arc<crate::ProxyDatabaseConnection>,
metric_callback: Option<crate::metric::Callback>,
) -> Result<crate::DatabaseTransaction, DbErr> {
use std::sync::Mutex;
let backend = inner.get_database_backend();
Self::begin(
Arc::new(Mutex::new(crate::InnerConnection::Proxy(inner))),
backend,
metric_callback,
None,
None,
None,
)
}
}