Skip to content

Commit

Permalink
feat(driver-adapters): remove "queryable" into its own module
Browse files Browse the repository at this point in the history
  • Loading branch information
jkomyno committed Nov 22, 2023
1 parent d6d09d6 commit 2347cb1
Show file tree
Hide file tree
Showing 12 changed files with 392 additions and 22 deletions.
2 changes: 1 addition & 1 deletion query-engine/driver-adapters/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@ tracing = "0.1"
tracing-core = "0.1"
metrics = "0.18"
uuid = { version = "1", features = ["v4"] }
pin-project = "1"

# Note: these deps are temporarily specified here to avoid importing them from tiberius (the SQL server driver).
# They will be imported from quaint-core instead in a future PR.
Expand All @@ -37,4 +38,3 @@ serde-wasm-bindgen.workspace = true
wasm-bindgen.workspace = true
wasm-bindgen-futures.workspace = true
tsify.workspace = true
pin-project = "1"
2 changes: 2 additions & 0 deletions query-engine/driver-adapters/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,8 @@
pub(crate) mod conversion;
pub(crate) mod error;
pub(crate) mod queryable;
pub(crate) mod send_future;
pub(crate) mod types;

#[cfg(not(target_arch = "wasm32"))]
Expand Down
2 changes: 1 addition & 1 deletion query-engine/driver-adapters/src/napi/conversion.rs
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
pub(crate) use crate::conversion::{mysql, postgres, sqlite, JSArg};
pub(crate) use crate::conversion::JSArg;

use napi::bindgen_prelude::{FromNapiValue, ToNapiValue};
use napi::NapiValue;
Expand Down
6 changes: 3 additions & 3 deletions query-engine/driver-adapters/src/napi/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -3,8 +3,8 @@
mod async_js_function;
mod conversion;
mod error;
mod proxy;
mod queryable;
pub(crate) mod proxy;
mod result;
mod transaction;
pub use queryable::{from_napi, JsQueryable};

pub use crate::queryable::{from_napi, JsQueryable};
6 changes: 2 additions & 4 deletions query-engine/driver-adapters/src/napi/transaction.rs
Original file line number Diff line number Diff line change
Expand Up @@ -7,10 +7,8 @@ use quaint::{
Value,
};

use super::{
proxy::{CommonProxy, TransactionOptions, TransactionProxy},
queryable::JsBaseQueryable,
};
use super::proxy::{CommonProxy, TransactionOptions, TransactionProxy};
use crate::queryable::JsBaseQueryable;

// Wrapper around JS transaction objects that implements Queryable
// and quaint::Transaction. Can be used in place of quaint transaction,
Expand Down
310 changes: 310 additions & 0 deletions query-engine/driver-adapters/src/queryable/mod.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,310 @@
#[cfg(not(target_arch = "wasm32"))]
pub(crate) mod napi;

#[cfg(not(target_arch = "wasm32"))]
pub use napi::from_napi;

#[cfg(not(target_arch = "wasm32"))]
pub(crate) use napi::JsBaseQueryable;

#[cfg(target_arch = "wasm32")]
pub(crate) mod wasm;

#[cfg(target_arch = "wasm32")]
pub use wasm::from_wasm;

#[cfg(target_arch = "wasm32")]
pub(crate) use wasm::JsBaseQueryable;

use super::{
conversion,
proxy::{CommonProxy, DriverProxy, Query},
};
use crate::send_future::SendFuture;
use async_trait::async_trait;
use futures::Future;
use psl::datamodel_connector::Flavour;
use quaint::{
connector::{metrics, IsolationLevel, Transaction},
error::{Error, ErrorKind},
prelude::{Query as QuaintQuery, Queryable as QuaintQueryable, ResultSet, TransactionCapable},
visitor::{self, Visitor},
};
use tracing::{info_span, Instrument};

impl JsBaseQueryable {
pub(crate) fn new(proxy: CommonProxy) -> Self {
let flavour: Flavour = proxy.flavour.parse().unwrap();
Self { proxy, flavour }
}

/// visit a quaint query AST according to the flavour of the JS connector
fn visit_quaint_query<'a>(&self, q: QuaintQuery<'a>) -> quaint::Result<(String, Vec<quaint::Value<'a>>)> {
match self.flavour {
Flavour::Mysql => visitor::Mysql::build(q),
Flavour::Postgres => visitor::Postgres::build(q),
Flavour::Sqlite => visitor::Sqlite::build(q),
_ => unimplemented!("Unsupported flavour for JS connector {:?}", self.flavour),
}
}

async fn build_query(&self, sql: &str, values: &[quaint::Value<'_>]) -> quaint::Result<Query> {
let sql: String = sql.to_string();

let converter = match self.flavour {
Flavour::Postgres => conversion::postgres::value_to_js_arg,
Flavour::Sqlite => conversion::sqlite::value_to_js_arg,
Flavour::Mysql => conversion::mysql::value_to_js_arg,
_ => unreachable!("Unsupported flavour for JS connector {:?}", self.flavour),
};

let args = values
.iter()
.map(converter)
.collect::<serde_json::Result<Vec<conversion::JSArg>>>()?;

Ok(Query { sql, args })
}
}

#[async_trait]
impl QuaintQueryable for JsBaseQueryable {
async fn query(&self, q: QuaintQuery<'_>) -> quaint::Result<ResultSet> {
let (sql, params) = self.visit_quaint_query(q)?;
self.query_raw(&sql, &params).await
}

async fn query_raw(&self, sql: &str, params: &[quaint::Value<'_>]) -> quaint::Result<ResultSet> {
metrics::query("js.query_raw", sql, params, move || async move {
self.do_query_raw(sql, params).await
})
.await
}

async fn query_raw_typed(&self, sql: &str, params: &[quaint::Value<'_>]) -> quaint::Result<ResultSet> {
self.query_raw(sql, params).await
}

async fn execute(&self, q: QuaintQuery<'_>) -> quaint::Result<u64> {
let (sql, params) = self.visit_quaint_query(q)?;
self.execute_raw(&sql, &params).await
}

async fn execute_raw(&self, sql: &str, params: &[quaint::Value<'_>]) -> quaint::Result<u64> {
metrics::query("js.execute_raw", sql, params, move || async move {
self.do_execute_raw(sql, params).await
})
.await
}

async fn execute_raw_typed(&self, sql: &str, params: &[quaint::Value<'_>]) -> quaint::Result<u64> {
self.execute_raw(sql, params).await
}

async fn raw_cmd(&self, cmd: &str) -> quaint::Result<()> {
let params = &[];
metrics::query("js.raw_cmd", cmd, params, move || async move {
self.do_execute_raw(cmd, params).await?;
Ok(())
})
.await
}

async fn version(&self) -> quaint::Result<Option<String>> {
// Note: JS Connectors don't use this method.
Ok(None)
}

fn is_healthy(&self) -> bool {
// Note: JS Connectors don't use this method.
true
}

/// Sets the transaction isolation level to given value.
/// Implementers have to make sure that the passed isolation level is valid for the underlying database.
async fn set_tx_isolation_level(&self, isolation_level: IsolationLevel) -> quaint::Result<()> {
if matches!(isolation_level, IsolationLevel::Snapshot) {
return Err(Error::builder(ErrorKind::invalid_isolation_level(&isolation_level)).build());
}

if self.flavour == Flavour::Sqlite {
return match isolation_level {
IsolationLevel::Serializable => Ok(()),
_ => Err(Error::builder(ErrorKind::invalid_isolation_level(&isolation_level)).build()),
};
}

self.raw_cmd(&format!("SET TRANSACTION ISOLATION LEVEL {isolation_level}"))
.await
}

fn requires_isolation_first(&self) -> bool {
match self.flavour {
Flavour::Mysql => true,
Flavour::Postgres | Flavour::Sqlite => false,
_ => unreachable!(),
}
}
}

impl JsBaseQueryable {
pub fn phantom_query_message(stmt: &str) -> String {
format!(r#"-- Implicit "{}" query via underlying driver"#, stmt)
}

async fn do_query_raw_inner(&self, sql: &str, params: &[quaint::Value<'_>]) -> quaint::Result<ResultSet> {
let len = params.len();
let serialization_span = info_span!("js:query:args", user_facing = true, "length" = %len);
let query = self.build_query(sql, params).instrument(serialization_span).await?;

let sql_span = info_span!("js:query:sql", user_facing = true, "db.statement" = %sql);
let result_set = self.proxy.query_raw(query).instrument(sql_span).await?;

let len = result_set.len();
let _deserialization_span = info_span!("js:query:result", user_facing = true, "length" = %len).entered();

result_set.try_into()
}

fn do_query_raw<'a>(
&'a self,
sql: &'a str,
params: &'a [quaint::Value<'a>],
) -> SendFuture<impl Future<Output = quaint::Result<ResultSet>> + 'a> {
SendFuture(self.do_query_raw_inner(sql, params))
}

async fn do_execute_raw_inner(&self, sql: &str, params: &[quaint::Value<'_>]) -> quaint::Result<u64> {
let len = params.len();
let serialization_span = info_span!("js:query:args", user_facing = true, "length" = %len);
let query = self.build_query(sql, params).instrument(serialization_span).await?;

let sql_span = info_span!("js:query:sql", user_facing = true, "db.statement" = %sql);
let affected_rows = self.proxy.execute_raw(query).instrument(sql_span).await?;

Ok(affected_rows as u64)
}

fn do_execute_raw<'a>(
&'a self,
sql: &'a str,
params: &'a [quaint::Value<'a>],
) -> SendFuture<impl Future<Output = quaint::Result<u64>> + 'a> {
SendFuture(self.do_execute_raw_inner(sql, params))
}
}

/// A JsQueryable adapts a Proxy to implement quaint's Queryable interface. It has the
/// responsibility of transforming inputs and outputs of `query` and `execute` methods from quaint
/// types to types that can be translated into javascript and viceversa. This is to let the rest of
/// the query engine work as if it was using quaint itself. The aforementioned transformations are:
///
/// Transforming a `quaint::ast::Query` into SQL by visiting it for the specific flavour of SQL
/// expected by the client connector. (eg. using the mysql visitor for the Planetscale client
/// connector)
///
/// Transforming a `JSResultSet` (what client connectors implemented in javascript provide)
/// into a `quaint::connector::result_set::ResultSet`. A quaint `ResultSet` is basically a vector
/// of `quaint::Value` but said type is a tagged enum, with non-unit variants that cannot be converted to javascript as is.
///
pub struct JsQueryable {
inner: JsBaseQueryable,
driver_proxy: DriverProxy,
}

impl std::fmt::Display for JsQueryable {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(f, "JSQueryable(driver)")
}
}

impl std::fmt::Debug for JsQueryable {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(f, "JSQueryable(driver)")
}
}

#[async_trait]
impl QuaintQueryable for JsQueryable {
async fn query(&self, q: QuaintQuery<'_>) -> quaint::Result<ResultSet> {
self.inner.query(q).await
}

async fn query_raw(&self, sql: &str, params: &[quaint::Value<'_>]) -> quaint::Result<ResultSet> {
self.inner.query_raw(sql, params).await
}

async fn query_raw_typed(&self, sql: &str, params: &[quaint::Value<'_>]) -> quaint::Result<ResultSet> {
self.inner.query_raw_typed(sql, params).await
}

async fn execute(&self, q: QuaintQuery<'_>) -> quaint::Result<u64> {
self.inner.execute(q).await
}

async fn execute_raw(&self, sql: &str, params: &[quaint::Value<'_>]) -> quaint::Result<u64> {
self.inner.execute_raw(sql, params).await
}

async fn execute_raw_typed(&self, sql: &str, params: &[quaint::Value<'_>]) -> quaint::Result<u64> {
self.inner.execute_raw_typed(sql, params).await
}

async fn raw_cmd(&self, cmd: &str) -> quaint::Result<()> {
self.inner.raw_cmd(cmd).await
}

async fn version(&self) -> quaint::Result<Option<String>> {
self.inner.version().await
}

fn is_healthy(&self) -> bool {
self.inner.is_healthy()
}

async fn set_tx_isolation_level(&self, isolation_level: IsolationLevel) -> quaint::Result<()> {
self.inner.set_tx_isolation_level(isolation_level).await
}

fn requires_isolation_first(&self) -> bool {
self.inner.requires_isolation_first()
}
}

#[async_trait]
impl TransactionCapable for JsQueryable {
async fn start_transaction<'a>(
&'a self,
isolation: Option<IsolationLevel>,
) -> quaint::Result<Box<dyn Transaction + 'a>> {
let tx = self.driver_proxy.start_transaction().await?;

let isolation_first = tx.requires_isolation_first();

if isolation_first {
if let Some(isolation) = isolation {
tx.set_tx_isolation_level(isolation).await?;
}
}

let begin_stmt = tx.begin_statement();

let tx_opts = tx.options();
if tx_opts.use_phantom_query {
let begin_stmt = JsBaseQueryable::phantom_query_message(begin_stmt);
tx.raw_phantom_cmd(begin_stmt.as_str()).await?;
} else {
tx.raw_cmd(begin_stmt).await?;
}

if !isolation_first {
if let Some(isolation) = isolation {
tx.set_tx_isolation_level(isolation).await?;
}
}

self.server_reset_query(tx.as_ref()).await?;

Ok(tx)
}
}
32 changes: 32 additions & 0 deletions query-engine/driver-adapters/src/queryable/napi.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
use crate::napi::proxy::{CommonProxy, DriverProxy};
use crate::JsQueryable;
use napi::JsObject;
use psl::datamodel_connector::Flavour;

/// A JsQueryable adapts a Proxy to implement quaint's Queryable interface. It has the
/// responsibility of transforming inputs and outputs of `query` and `execute` methods from quaint
/// types to types that can be translated into javascript and viceversa. This is to let the rest of
/// the query engine work as if it was using quaint itself. The aforementioned transformations are:
///
/// Transforming a `quaint::ast::Query` into SQL by visiting it for the specific flavour of SQL
/// expected by the client connector. (eg. using the mysql visitor for the Planetscale client
/// connector)
///
/// Transforming a `JSResultSet` (what client connectors implemented in javascript provide)
/// into a `quaint::connector::result_set::ResultSet`. A quaint `ResultSet` is basically a vector
/// of `quaint::Value` but said type is a tagged enum, with non-unit variants that cannot be converted to javascript as is.
///
pub(crate) struct JsBaseQueryable {
pub(crate) proxy: CommonProxy,
pub flavour: Flavour,
}

pub fn from_napi(driver: JsObject) -> JsQueryable {
let common = CommonProxy::new(&driver).unwrap();
let driver_proxy = DriverProxy::new(&driver).unwrap();

JsQueryable {
inner: JsBaseQueryable::new(common),
driver_proxy,
}
}
Loading

0 comments on commit 2347cb1

Please sign in to comment.