sui_graphql_rpc/
data.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
// Copyright (c) Mysten Labs, Inc.
// SPDX-License-Identifier: Apache-2.0

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;

/// Database Backend in use -- abstracting a specific implementation.
pub(crate) type Db = pg::PgExecutor;

/// Helper types to access associated types on `Db`.
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;

/// Helper types for accessing a shared `DataLoader` instance.
#[derive(Clone)]
pub(crate) struct DataLoader(pub Arc<AGDataLoader<Db>>);

/// A generic boxed query (compatible with the return type of `into_boxed` on diesel's table DSL).
///
/// - ST is the SqlType of the rows selected.
/// - QS is the QuerySource (the table(s) being selected from).
/// - GB is the GroupBy clause.
///
/// These type parameters should usually be inferred by context.
pub(crate) type Query<ST, QS, GB> =
    BoxedSelectStatement<'static, ST, FromClause<QS>, DieselBackend, GB>;

/// Interface for accessing relational data written by the Indexer, agnostic of the database
/// back-end being used.
#[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;

    /// Execute `txn` with read committed isolation. `txn` is supplied a database connection to
    /// issue queries over.
    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;

    /// Execute `txn` with repeatable reads and no phantom reads -- multiple calls to the same query
    /// should produce the same results. `txn` is supplied a database connection to issue queries
    /// over.
    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>;

    /// Run a query that fetches a single value. `query` is a thunk that returns a query when
    /// called.
    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;

    /// Run a query that fetches multiple values. `query` is a thunk that returns a query when
    /// called.
    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;

    /// Helper to limit a query that fetches multiple values to return only its first value. `query`
    /// is a thunk that returns a query when called.
    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)))
    }
}