pub(crate) mod apys;
pub(crate) mod move_registry_data_loader;
pub(crate) mod package_resolver;
pub(crate) mod pg;
use std::sync::Arc;
use async_graphql::dataloader::DataLoader as AGDataLoader;
use async_trait::async_trait;
use diesel::{
query_builder::{BoxedSelectStatement, FromClause, QueryFragment, QueryId},
query_dsl::methods::LimitDsl,
QueryResult,
};
use diesel_async::{methods::LoadQuery, scoped_futures::ScopedBoxFuture};
use crate::error::Error;
pub(crate) type Db = pg::PgExecutor;
pub(crate) type Conn<'c> = <Db as QueryExecutor>::DbConnection<'c>;
pub(crate) type DieselConn = <Db as QueryExecutor>::Connection;
pub(crate) type DieselBackend = <Db as QueryExecutor>::Backend;
#[derive(Clone)]
pub(crate) struct DataLoader(pub Arc<AGDataLoader<Db>>);
pub(crate) type Query<ST, QS, GB> =
BoxedSelectStatement<'static, ST, FromClause<QS>, DieselBackend, GB>;
#[async_trait]
pub(crate) trait QueryExecutor {
type Backend: diesel::backend::Backend;
type Connection: diesel_async::AsyncConnection;
type DbConnection<'c>: DbConnection<Connection = Self::Connection, Backend = Self::Backend>
where
Self: 'c;
async fn execute<'c, T, U, E>(&self, txn: T) -> Result<U, Error>
where
T: for<'r> FnOnce(
&'r mut Self::DbConnection<'_>,
) -> ScopedBoxFuture<'static, 'r, Result<U, E>>
+ Send
+ 'c,
E: From<diesel::result::Error> + std::error::Error,
T: Send + 'static,
U: Send + 'static,
E: Send + 'static;
async fn execute_repeatable<'c, T, U, E>(&self, txn: T) -> Result<U, Error>
where
T: for<'r> FnOnce(
&'r mut Self::DbConnection<'_>,
) -> ScopedBoxFuture<'static, 'r, Result<U, E>>
+ Send
+ 'c,
E: From<diesel::result::Error> + std::error::Error,
T: Send + 'static,
U: Send + 'static,
E: Send + 'static;
}
#[async_trait]
pub(crate) trait DbConnection {
type Backend: diesel::backend::Backend;
type Connection: diesel_async::AsyncConnection<Backend = Self::Backend>;
async fn result<T, Q, U>(&mut self, query: T) -> QueryResult<U>
where
T: Fn() -> Q + Send,
Q: diesel::query_builder::Query + Send + 'static,
Q: LoadQuery<'static, Self::Connection, U>,
Q: QueryId + QueryFragment<Self::Backend>,
U: Send;
async fn results<T, Q, U>(&mut self, query: T) -> QueryResult<Vec<U>>
where
T: Fn() -> Q + Send,
Q: diesel::query_builder::Query + Send + 'static,
Q: LoadQuery<'static, Self::Connection, U>,
Q: QueryId + QueryFragment<Self::Backend>,
U: Send;
async fn first<T, Q: LimitDsl, U>(&mut self, query: T) -> QueryResult<U>
where
T: Fn() -> Q + Send,
<Q as LimitDsl>::Output: diesel::query_builder::Query + Send,
<Q as LimitDsl>::Output: LoadQuery<'static, Self::Connection, U>,
<Q as LimitDsl>::Output: QueryId + QueryFragment<Self::Backend> + 'static,
U: Send,
{
self.result(move || query().limit(1i64)).await
}
}
impl DataLoader {
pub(crate) fn new(db: Db) -> Self {
Self(Arc::new(AGDataLoader::new(db, tokio::spawn)))
}
}