From 476ee04d9837606b5d38917a83f4559d9b04f24c Mon Sep 17 00:00:00 2001 From: Alberto Schiabel Date: Tue, 16 Jan 2024 15:40:14 +0100 Subject: [PATCH] feat(query-engine-wasm): exclude native-drivers only errors from `wasm32` target (#4616) * feat(query-engine-wasm): start excluding native-drivers only errors from wasm32 target * feat(query-engine-wasm): fix compilation on native targets * chore(query-engine-wasm): update comment * chore(query-engine-wasm): remove comment * chore(query-engine-wasm): fix clippy * chore(query-engine-wasm): apply "repr(transparent)" to "ConnectionInfo" on Wasm, to shove off a few bytes * chore(query-engine-wasm): remove comments * feat: address review comments * chore: clippy * chore: fix quaint test * fix(driver-adapters): fix condition that made Vitess fail --- libs/user-facing-errors/src/quaint.rs | 387 +++++++++--------- quaint/src/connector.rs | 6 + quaint/src/connector/connection_info.rs | 224 +++++----- quaint/src/connector/mssql/native/error.rs | 10 +- quaint/src/connector/mysql/native/error.rs | 12 +- quaint/src/connector/native.rs | 31 ++ quaint/src/connector/postgres/native/error.rs | 21 +- quaint/src/connector/postgres/native/mod.rs | 9 +- quaint/src/connector/postgres/url.rs | 2 +- quaint/src/connector/timeout.rs | 6 +- quaint/src/{error.rs => error/mod.rs} | 95 ++--- quaint/src/error/name.rs | 32 ++ quaint/src/error/native.rs | 30 ++ quaint/src/pooled.rs | 11 +- quaint/src/prelude.rs | 3 + quaint/src/single.rs | 10 +- .../sql-query-connector/src/error.rs | 65 ++- .../src/flavour/mssql/connection.rs | 9 +- .../src/flavour/mysql/connection.rs | 9 +- .../src/flavour/postgres/connection.rs | 9 +- .../src/multi_engine_test_api.rs | 14 +- 21 files changed, 570 insertions(+), 425 deletions(-) create mode 100644 quaint/src/connector/native.rs rename quaint/src/{error.rs => error/mod.rs} (84%) create mode 100644 quaint/src/error/name.rs create mode 100644 quaint/src/error/native.rs diff --git a/libs/user-facing-errors/src/quaint.rs b/libs/user-facing-errors/src/quaint.rs index aab6598d81bc..c2e73948dcfb 100644 --- a/libs/user-facing-errors/src/quaint.rs +++ b/libs/user-facing-errors/src/quaint.rs @@ -1,8 +1,10 @@ use crate::{common, query_engine, KnownError}; -use common::ModelKind; use indoc::formatdoc; use quaint::{error::ErrorKind, prelude::ConnectionInfo}; +#[cfg(not(target_arch = "wasm32"))] +use quaint::{connector::NativeConnectionInfo, error::NativeErrorKind}; + impl From<&quaint::error::DatabaseConstraint> for query_engine::DatabaseConstraint { fn from(other: &quaint::error::DatabaseConstraint) -> Self { match other { @@ -37,91 +39,157 @@ pub fn invalid_connection_string_description(error_details: &str) -> String { } pub fn render_quaint_error(kind: &ErrorKind, connection_info: &ConnectionInfo) -> Option { - match (kind, connection_info) { - (ErrorKind::DatabaseDoesNotExist { .. }, ConnectionInfo::Sqlite { .. }) => { - unreachable!(); // quaint implicitly creates sqlite databases - } - - (ErrorKind::DatabaseDoesNotExist { db_name }, ConnectionInfo::Postgres(url)) => { - Some(KnownError::new(common::DatabaseDoesNotExist::Postgres { - database_name: db_name.to_string(), - database_host: url.host().to_owned(), - database_port: url.port(), - })) - } - - (ErrorKind::DatabaseDoesNotExist { .. }, ConnectionInfo::Mysql(url)) => { - Some(KnownError::new(common::DatabaseDoesNotExist::Mysql { - database_name: url.dbname().to_owned(), - database_host: url.host().to_owned(), - database_port: url.port(), - })) - } - (ErrorKind::DatabaseDoesNotExist { .. }, ConnectionInfo::Mssql(url)) => { - Some(KnownError::new(common::DatabaseDoesNotExist::Mssql { - database_name: url.dbname().to_owned(), - database_host: url.host().to_owned(), - database_port: url.port(), - })) - } - - (ErrorKind::DatabaseAccessDenied { .. }, ConnectionInfo::Postgres(url)) => { - Some(KnownError::new(common::DatabaseAccessDenied { - database_user: url.username().into_owned(), - database_name: format!("{}.{}", url.dbname(), url.schema()), - })) - } - - (ErrorKind::DatabaseAccessDenied { .. }, ConnectionInfo::Mysql(url)) => { - Some(KnownError::new(common::DatabaseAccessDenied { - database_user: url.username().into_owned(), - database_name: url.dbname().to_owned(), - })) - } - - (ErrorKind::DatabaseAlreadyExists { db_name }, ConnectionInfo::Postgres(url)) => { - Some(KnownError::new(common::DatabaseAlreadyExists { - database_name: format!("{db_name}"), - database_host: url.host().to_owned(), - database_port: url.port(), - })) - } + let default_value: Option = None; - (ErrorKind::DatabaseAlreadyExists { db_name }, ConnectionInfo::Mysql(url)) => { - Some(KnownError::new(common::DatabaseAlreadyExists { - database_name: format!("{db_name}"), - database_host: url.host().to_owned(), - database_port: url.port(), - })) - } - - (ErrorKind::AuthenticationFailed { user }, ConnectionInfo::Postgres(url)) => { - Some(KnownError::new(common::IncorrectDatabaseCredentials { - database_user: format!("{user}"), - database_host: url.host().to_owned(), - })) - } - - (ErrorKind::AuthenticationFailed { user }, ConnectionInfo::Mysql(url)) => { - Some(KnownError::new(common::IncorrectDatabaseCredentials { - database_user: format!("{user}"), - database_host: url.host().to_owned(), - })) - } - - (ErrorKind::ConnectionError(_), ConnectionInfo::Postgres(url)) => { - Some(KnownError::new(common::DatabaseNotReachable { - database_port: url.port(), - database_host: url.host().to_owned(), - })) - } - - (ErrorKind::ConnectionError(_), ConnectionInfo::Mysql(url)) => { - Some(KnownError::new(common::DatabaseNotReachable { - database_port: url.port(), - database_host: url.host().to_owned(), - })) - } + match (kind, connection_info) { + (ErrorKind::DatabaseDoesNotExist { .. }, ConnectionInfo::External(_)) => default_value, + #[cfg(not(target_arch = "wasm32"))] + (ErrorKind::DatabaseDoesNotExist { db_name }, _) => match connection_info { + ConnectionInfo::Native(NativeConnectionInfo::Postgres(url)) => { + Some(KnownError::new(common::DatabaseDoesNotExist::Postgres { + database_name: db_name.to_string(), + database_host: url.host().to_owned(), + database_port: url.port(), + })) + } + ConnectionInfo::Native(NativeConnectionInfo::Mysql(url)) => { + Some(KnownError::new(common::DatabaseDoesNotExist::Mysql { + database_name: url.dbname().to_owned(), + database_host: url.host().to_owned(), + database_port: url.port(), + })) + } + ConnectionInfo::Native(NativeConnectionInfo::Mssql(url)) => { + Some(KnownError::new(common::DatabaseDoesNotExist::Mssql { + database_name: url.dbname().to_owned(), + database_host: url.host().to_owned(), + database_port: url.port(), + })) + } + _ => unreachable!(), // quaint implicitly creates sqlite databases + }, + + (ErrorKind::DatabaseAccessDenied { .. }, ConnectionInfo::External(_)) => default_value, + #[cfg(not(target_arch = "wasm32"))] + (ErrorKind::DatabaseAccessDenied { .. }, _) => match connection_info { + ConnectionInfo::Native(NativeConnectionInfo::Postgres(url)) => { + Some(KnownError::new(common::DatabaseAccessDenied { + database_user: url.username().into_owned(), + database_name: format!("{}.{}", url.dbname(), url.schema()), + })) + } + ConnectionInfo::Native(NativeConnectionInfo::Mysql(url)) => { + Some(KnownError::new(common::DatabaseAccessDenied { + database_user: url.username().into_owned(), + database_name: url.dbname().to_owned(), + })) + } + _ => unreachable!(), + }, + + (ErrorKind::DatabaseAlreadyExists { .. }, ConnectionInfo::External(_)) => default_value, + #[cfg(not(target_arch = "wasm32"))] + (ErrorKind::DatabaseAlreadyExists { db_name }, _) => match connection_info { + ConnectionInfo::Native(NativeConnectionInfo::Postgres(url)) => { + Some(KnownError::new(common::DatabaseAlreadyExists { + database_name: format!("{db_name}"), + database_host: url.host().to_owned(), + database_port: url.port(), + })) + } + ConnectionInfo::Native(NativeConnectionInfo::Mysql(url)) => { + Some(KnownError::new(common::DatabaseAlreadyExists { + database_name: format!("{db_name}"), + database_host: url.host().to_owned(), + database_port: url.port(), + })) + } + _ => unreachable!(), + }, + + (ErrorKind::AuthenticationFailed { .. }, ConnectionInfo::External(_)) => default_value, + #[cfg(not(target_arch = "wasm32"))] + (ErrorKind::AuthenticationFailed { user }, _) => match connection_info { + ConnectionInfo::Native(NativeConnectionInfo::Postgres(url)) => { + Some(KnownError::new(common::IncorrectDatabaseCredentials { + database_user: format!("{user}"), + database_host: url.host().to_owned(), + })) + } + ConnectionInfo::Native(NativeConnectionInfo::Mysql(url)) => { + Some(KnownError::new(common::IncorrectDatabaseCredentials { + database_user: format!("{user}"), + database_host: url.host().to_owned(), + })) + } + _ => unreachable!(), + }, + + (ErrorKind::SocketTimeout { .. }, ConnectionInfo::External(_)) => default_value, + #[cfg(not(target_arch = "wasm32"))] + (ErrorKind::SocketTimeout, _) => match connection_info { + ConnectionInfo::Native(NativeConnectionInfo::Postgres(url)) => { + let time = match url.socket_timeout() { + Some(dur) => format!("{}s", dur.as_secs()), + None => String::from("N/A"), + }; + + Some(KnownError::new(common::DatabaseOperationTimeout { + time, + context: "Socket timeout (the database failed to respond to a query within the configured timeout — see https://pris.ly/d/mssql-connector for more details.)." + .into(), + })) + } + ConnectionInfo::Native(NativeConnectionInfo::Mysql(url)) => { + let time = match url.socket_timeout() { + Some(dur) => format!("{}s", dur.as_secs()), + None => String::from("N/A"), + }; + + Some(KnownError::new(common::DatabaseOperationTimeout { + time, + context: "Socket timeout (the database failed to respond to a query within the configured timeout — see https://pris.ly/d/mysql-connector for more details.)." + .into(), + })) + } + ConnectionInfo::Native(NativeConnectionInfo::Mssql(url)) => { + let time = match url.socket_timeout() { + Some(dur) => format!("{}s", dur.as_secs()), + None => String::from("N/A"), + }; + + Some(KnownError::new(common::DatabaseOperationTimeout { + time, + context: "Socket timeout (the database failed to respond to a query within the configured timeout — see https://pris.ly/d/postgres-connector for more details.)." + .into(), + })) + } + _ => unreachable!(), + }, + + (ErrorKind::TableDoesNotExist { .. }, ConnectionInfo::External(_)) => default_value, + #[cfg(not(target_arch = "wasm32"))] + (ErrorKind::TableDoesNotExist { table: model }, _) => match connection_info { + ConnectionInfo::Native(NativeConnectionInfo::Postgres(_)) => Some(KnownError::new(common::InvalidModel { + model: format!("{model}"), + kind: common::ModelKind::Table, + })), + ConnectionInfo::Native(NativeConnectionInfo::Mysql(_)) => Some(KnownError::new(common::InvalidModel { + model: format!("{model}"), + kind: common::ModelKind::Table, + })), + ConnectionInfo::Native(NativeConnectionInfo::Sqlite { .. }) => { + Some(KnownError::new(common::InvalidModel { + model: format!("{model}"), + kind: common::ModelKind::Table, + })) + } + ConnectionInfo::Native(NativeConnectionInfo::Mssql(_)) => Some(KnownError::new(common::InvalidModel { + model: format!("{model}"), + kind: common::ModelKind::Table, + })), + _ => unreachable!(), + }, (ErrorKind::UniqueConstraintViolation { constraint }, _) => { Some(KnownError::new(query_engine::UniqueKeyViolation { @@ -129,75 +197,6 @@ pub fn render_quaint_error(kind: &ErrorKind, connection_info: &ConnectionInfo) - })) } - (ErrorKind::TlsError { message }, _) => Some(KnownError::new(common::TlsConnectionError { - message: message.into(), - })), - - (ErrorKind::ConnectTimeout, ConnectionInfo::Mysql(url)) => { - Some(KnownError::new(common::DatabaseNotReachable { - database_host: url.host().to_owned(), - database_port: url.port(), - })) - } - - (ErrorKind::ConnectTimeout, ConnectionInfo::Postgres(url)) => { - Some(KnownError::new(common::DatabaseNotReachable { - database_host: url.host().to_owned(), - database_port: url.port(), - })) - } - - (ErrorKind::ConnectTimeout, ConnectionInfo::Mssql(url)) => { - Some(KnownError::new(common::DatabaseNotReachable { - database_host: url.host().to_owned(), - database_port: url.port(), - })) - } - - (ErrorKind::SocketTimeout, ConnectionInfo::Mysql(url)) => { - let time = match url.socket_timeout() { - Some(dur) => format!("{}s", dur.as_secs()), - None => String::from("N/A"), - }; - - Some(KnownError::new(common::DatabaseOperationTimeout { - time, - context: "Socket timeout (the database failed to respond to a query within the configured timeout — see https://pris.ly/d/mysql-connector for more details.)." - .into(), - })) - } - - (ErrorKind::SocketTimeout, ConnectionInfo::Postgres(url)) => { - let time = match url.socket_timeout() { - Some(dur) => format!("{}s", dur.as_secs()), - None => String::from("N/A"), - }; - - Some(KnownError::new(common::DatabaseOperationTimeout { - time, - context: "Socket timeout (the database failed to respond to a query within the configured timeout — see https://pris.ly/d/mssql-connector for more details.)." - .into(), - })) - } - - (ErrorKind::SocketTimeout, ConnectionInfo::Mssql(url)) => { - let time = match url.socket_timeout() { - Some(dur) => format!("{}s", dur.as_secs()), - None => String::from("N/A"), - }; - - Some(KnownError::new(common::DatabaseOperationTimeout { - time, - context: "Socket timeout (the database failed to respond to a query within the configured timeout — see https://pris.ly/d/postgres-connector for more details.)." - .into(), - })) - } - - (ErrorKind::PoolTimeout { max_open, timeout, .. }, _) => Some(KnownError::new(query_engine::PoolTimeout { - connection_limit: *max_open, - timeout: *timeout, - })), - (ErrorKind::DatabaseUrlIsInvalid(details), _connection_info) => { Some(KnownError::new(common::InvalidConnectionString { details: details.to_owned(), @@ -216,42 +215,50 @@ pub fn render_quaint_error(kind: &ErrorKind, connection_info: &ConnectionInfo) - })) } - (ErrorKind::TableDoesNotExist { table: model }, ConnectionInfo::Mysql(_)) => { - Some(KnownError::new(common::InvalidModel { - model: format!("{model}"), - kind: ModelKind::Table, - })) - } - - (ErrorKind::TableDoesNotExist { table: model }, ConnectionInfo::Postgres(_)) => { - Some(KnownError::new(common::InvalidModel { - model: format!("{model}"), - kind: ModelKind::Table, - })) - } - - (ErrorKind::TableDoesNotExist { table: model }, ConnectionInfo::Sqlite { .. }) => { - Some(KnownError::new(common::InvalidModel { - model: format!("{model}"), - kind: ModelKind::Table, - })) - } - - (ErrorKind::TableDoesNotExist { table: model }, ConnectionInfo::Mssql(_)) => { - Some(KnownError::new(common::InvalidModel { - model: format!("{model}"), - kind: ModelKind::Table, - })) - } - - (ErrorKind::IncorrectNumberOfParameters { expected, actual }, ConnectionInfo::Mssql(_)) => { - Some(KnownError::new(common::IncorrectNumberOfParameters { - expected: *expected, - actual: *actual, - })) - } - - (ErrorKind::ConnectionClosed, _) => Some(KnownError::new(common::ConnectionClosed)), + #[cfg(not(target_arch = "wasm32"))] + (ErrorKind::Native(native_error_kind), _) => match (native_error_kind, connection_info) { + (NativeErrorKind::ConnectionError(_), ConnectionInfo::Native(NativeConnectionInfo::Postgres(url))) => { + Some(KnownError::new(common::DatabaseNotReachable { + database_port: url.port(), + database_host: url.host().to_owned(), + })) + } + (NativeErrorKind::ConnectionError(_), ConnectionInfo::Native(NativeConnectionInfo::Mysql(url))) => { + Some(KnownError::new(common::DatabaseNotReachable { + database_port: url.port(), + database_host: url.host().to_owned(), + })) + } + (NativeErrorKind::TlsError { message }, _) => Some(KnownError::new(common::TlsConnectionError { + message: message.into(), + })), + (NativeErrorKind::ConnectTimeout, ConnectionInfo::Native(NativeConnectionInfo::Postgres(url))) => { + Some(KnownError::new(common::DatabaseNotReachable { + database_host: url.host().to_owned(), + database_port: url.port(), + })) + } + (NativeErrorKind::ConnectTimeout, ConnectionInfo::Native(NativeConnectionInfo::Mysql(url))) => { + Some(KnownError::new(common::DatabaseNotReachable { + database_host: url.host().to_owned(), + database_port: url.port(), + })) + } + (NativeErrorKind::ConnectTimeout, ConnectionInfo::Native(NativeConnectionInfo::Mssql(url))) => { + Some(KnownError::new(common::DatabaseNotReachable { + database_host: url.host().to_owned(), + database_port: url.port(), + })) + } + (NativeErrorKind::PoolTimeout { max_open, timeout, .. }, _) => { + Some(KnownError::new(query_engine::PoolTimeout { + connection_limit: *max_open, + timeout: *timeout, + })) + } + (NativeErrorKind::ConnectionClosed, _) => Some(KnownError::new(common::ConnectionClosed)), + _ => unreachable!(), + }, _ => None, } diff --git a/quaint/src/connector.rs b/quaint/src/connector.rs index d56ca737bd80..475856936566 100644 --- a/quaint/src/connector.rs +++ b/quaint/src/connector.rs @@ -13,6 +13,8 @@ mod connection_info; pub mod external; pub mod metrics; +#[cfg(feature = "native")] +pub mod native; mod queryable; mod result_set; #[cfg(any(feature = "mssql-native", feature = "postgresql-native", feature = "mysql-native"))] @@ -22,6 +24,10 @@ mod type_identifier; pub use self::result_set::*; pub use connection_info::*; + +#[cfg(feature = "native")] +pub use native::*; + pub use external::*; pub use queryable::*; pub use transaction::*; diff --git a/quaint/src/connector/connection_info.rs b/quaint/src/connector/connection_info.rs index 0abe9adb28f3..50f2301e443e 100644 --- a/quaint/src/connector/connection_info.rs +++ b/quaint/src/connector/connection_info.rs @@ -1,3 +1,6 @@ +#![cfg_attr(target_arch = "wasm32", allow(unused_imports))] +#![cfg_attr(not(target_arch = "wasm32"), allow(clippy::large_enum_variant))] + use crate::error::{Error, ErrorKind}; use std::{borrow::Cow, fmt}; use url::Url; @@ -15,30 +18,15 @@ use std::convert::TryFrom; use super::ExternalConnectionInfo; +#[cfg(not(target_arch = "wasm32"))] +use super::NativeConnectionInfo; + /// General information about a SQL connection. #[derive(Debug, Clone)] +#[cfg_attr(target_arch = "wasm32", repr(transparent))] pub enum ConnectionInfo { - /// A PostgreSQL connection URL. - #[cfg(feature = "postgresql")] - Postgres(PostgresUrl), - /// A MySQL connection URL. - #[cfg(feature = "mysql")] - Mysql(MysqlUrl), - /// A SQL Server connection URL. - #[cfg(feature = "mssql")] - Mssql(MssqlUrl), - /// A SQLite connection URL. - #[cfg(feature = "sqlite")] - Sqlite { - /// The filesystem path of the SQLite database. - file_path: String, - /// The name the database is bound to - Always "main" - db_name: String, - }, - #[cfg(feature = "sqlite")] - InMemorySqlite { - db_name: String, - }, + #[cfg(not(target_arch = "wasm32"))] + Native(NativeConnectionInfo), External(ExternalConnectionInfo), } @@ -47,6 +35,7 @@ impl ConnectionInfo { /// /// Will fail if URI is invalid or the scheme points to an unsupported /// database. + #[cfg(not(target_arch = "wasm32"))] pub fn from_url(url_str: &str) -> crate::Result { let url_result: Result = url_str.parse(); @@ -57,15 +46,17 @@ impl ConnectionInfo { if url_result.is_err() { let params = SqliteParams::try_from(s)?; - return Ok(ConnectionInfo::Sqlite { + return Ok(ConnectionInfo::Native(NativeConnectionInfo::Sqlite { file_path: params.file_path, db_name: params.db_name, - }); + })); } } #[cfg(feature = "mssql")] s if s.starts_with("jdbc:sqlserver") || s.starts_with("sqlserver") => { - return Ok(ConnectionInfo::Mssql(MssqlUrl::new(url_str)?)); + return Ok(ConnectionInfo::Native(NativeConnectionInfo::Mssql(MssqlUrl::new( + url_str, + )?))); } _ => (), } @@ -81,18 +72,20 @@ impl ConnectionInfo { match sql_family { #[cfg(feature = "mysql")] - SqlFamily::Mysql => Ok(ConnectionInfo::Mysql(MysqlUrl::new(url)?)), + SqlFamily::Mysql => Ok(ConnectionInfo::Native(NativeConnectionInfo::Mysql(MysqlUrl::new(url)?))), #[cfg(feature = "sqlite")] SqlFamily::Sqlite => { let params = SqliteParams::try_from(url_str)?; - Ok(ConnectionInfo::Sqlite { + Ok(ConnectionInfo::Native(NativeConnectionInfo::Sqlite { file_path: params.file_path, db_name: params.db_name, - }) + })) } #[cfg(feature = "postgresql")] - SqlFamily::Postgres => Ok(ConnectionInfo::Postgres(PostgresUrl::new(url)?)), + SqlFamily::Postgres => Ok(ConnectionInfo::Native(NativeConnectionInfo::Postgres( + PostgresUrl::new(url)?, + ))), #[allow(unreachable_patterns)] _ => unreachable!(), } @@ -101,14 +94,17 @@ impl ConnectionInfo { /// The provided database name. This will be `None` on SQLite. pub fn dbname(&self) -> Option<&str> { match self { - #[cfg(feature = "postgresql")] - ConnectionInfo::Postgres(url) => Some(url.dbname()), - #[cfg(feature = "mysql")] - ConnectionInfo::Mysql(url) => Some(url.dbname()), - #[cfg(feature = "mssql")] - ConnectionInfo::Mssql(url) => Some(url.dbname()), - #[cfg(feature = "sqlite")] - ConnectionInfo::Sqlite { .. } | ConnectionInfo::InMemorySqlite { .. } => None, + #[cfg(not(target_arch = "wasm32"))] + ConnectionInfo::Native(info) => match info { + #[cfg(feature = "postgresql")] + NativeConnectionInfo::Postgres(url) => Some(url.dbname()), + #[cfg(feature = "mysql")] + NativeConnectionInfo::Mysql(url) => Some(url.dbname()), + #[cfg(feature = "mssql")] + NativeConnectionInfo::Mssql(url) => Some(url.dbname()), + #[cfg(feature = "sqlite")] + NativeConnectionInfo::Sqlite { .. } | NativeConnectionInfo::InMemorySqlite { .. } => None, + }, ConnectionInfo::External(_) => None, } } @@ -120,16 +116,19 @@ impl ConnectionInfo { /// - In MySQL, it is the database name. pub fn schema_name(&self) -> &str { match self { - #[cfg(feature = "postgresql")] - ConnectionInfo::Postgres(url) => url.schema(), - #[cfg(feature = "mysql")] - ConnectionInfo::Mysql(url) => url.dbname(), - #[cfg(feature = "mssql")] - ConnectionInfo::Mssql(url) => url.schema(), - #[cfg(feature = "sqlite")] - ConnectionInfo::Sqlite { db_name, .. } => db_name, - #[cfg(feature = "sqlite")] - ConnectionInfo::InMemorySqlite { db_name } => db_name, + #[cfg(not(target_arch = "wasm32"))] + ConnectionInfo::Native(info) => match info { + #[cfg(feature = "postgresql")] + NativeConnectionInfo::Postgres(url) => url.schema(), + #[cfg(feature = "mysql")] + NativeConnectionInfo::Mysql(url) => url.dbname(), + #[cfg(feature = "mssql")] + NativeConnectionInfo::Mssql(url) => url.schema(), + #[cfg(feature = "sqlite")] + NativeConnectionInfo::Sqlite { db_name, .. } => db_name, + #[cfg(feature = "sqlite")] + NativeConnectionInfo::InMemorySqlite { db_name } => db_name, + }, ConnectionInfo::External(info) => &info.schema_name, } } @@ -137,30 +136,36 @@ impl ConnectionInfo { /// The provided database host. This will be `"localhost"` on SQLite. pub fn host(&self) -> &str { match self { - #[cfg(feature = "postgresql")] - ConnectionInfo::Postgres(url) => url.host(), - #[cfg(feature = "mysql")] - ConnectionInfo::Mysql(url) => url.host(), - #[cfg(feature = "mssql")] - ConnectionInfo::Mssql(url) => url.host(), - #[cfg(feature = "sqlite")] - ConnectionInfo::Sqlite { .. } | ConnectionInfo::InMemorySqlite { .. } => "localhost", - + #[cfg(not(target_arch = "wasm32"))] + ConnectionInfo::Native(info) => match info { + #[cfg(feature = "postgresql")] + NativeConnectionInfo::Postgres(url) => url.host(), + #[cfg(feature = "mysql")] + NativeConnectionInfo::Mysql(url) => url.host(), + #[cfg(feature = "mssql")] + NativeConnectionInfo::Mssql(url) => url.host(), + #[cfg(feature = "sqlite")] + NativeConnectionInfo::Sqlite { .. } | NativeConnectionInfo::InMemorySqlite { .. } => "localhost", + }, ConnectionInfo::External(_) => "external", } } /// The provided database user name. This will be `None` on SQLite. pub fn username(&self) -> Option> { + // TODO: why do some of the native `.username()` methods return an `Option<&str>` and others a `Cow`? match self { - #[cfg(feature = "postgresql")] - ConnectionInfo::Postgres(url) => Some(url.username()), - #[cfg(feature = "mysql")] - ConnectionInfo::Mysql(url) => Some(url.username()), - #[cfg(feature = "mssql")] - ConnectionInfo::Mssql(url) => url.username().map(Cow::from), - #[cfg(feature = "sqlite")] - ConnectionInfo::Sqlite { .. } | ConnectionInfo::InMemorySqlite { .. } => None, + #[cfg(not(target_arch = "wasm32"))] + ConnectionInfo::Native(info) => match info { + #[cfg(feature = "postgresql")] + NativeConnectionInfo::Postgres(url) => Some(url.username()), + #[cfg(feature = "mysql")] + NativeConnectionInfo::Mysql(url) => Some(url.username()), + #[cfg(feature = "mssql")] + NativeConnectionInfo::Mssql(url) => url.username().map(Cow::from), + #[cfg(feature = "sqlite")] + NativeConnectionInfo::Sqlite { .. } | NativeConnectionInfo::InMemorySqlite { .. } => None, + }, ConnectionInfo::External(_) => None, } } @@ -168,16 +173,19 @@ impl ConnectionInfo { /// The database file for SQLite, otherwise `None`. pub fn file_path(&self) -> Option<&str> { match self { - #[cfg(feature = "postgresql")] - ConnectionInfo::Postgres(_) => None, - #[cfg(feature = "mysql")] - ConnectionInfo::Mysql(_) => None, - #[cfg(feature = "mssql")] - ConnectionInfo::Mssql(_) => None, - #[cfg(feature = "sqlite")] - ConnectionInfo::Sqlite { file_path, .. } => Some(file_path), - #[cfg(feature = "sqlite")] - ConnectionInfo::InMemorySqlite { .. } => None, + #[cfg(not(target_arch = "wasm32"))] + ConnectionInfo::Native(info) => match info { + #[cfg(feature = "postgresql")] + NativeConnectionInfo::Postgres(_) => None, + #[cfg(feature = "mysql")] + NativeConnectionInfo::Mysql(_) => None, + #[cfg(feature = "mssql")] + NativeConnectionInfo::Mssql(_) => None, + #[cfg(feature = "sqlite")] + NativeConnectionInfo::Sqlite { file_path, .. } => Some(file_path), + #[cfg(feature = "sqlite")] + NativeConnectionInfo::InMemorySqlite { .. } => None, + }, ConnectionInfo::External(_) => None, } } @@ -185,14 +193,17 @@ impl ConnectionInfo { /// The family of databases connected. pub fn sql_family(&self) -> SqlFamily { match self { - #[cfg(feature = "postgresql")] - ConnectionInfo::Postgres(_) => SqlFamily::Postgres, - #[cfg(feature = "mysql")] - ConnectionInfo::Mysql(_) => SqlFamily::Mysql, - #[cfg(feature = "mssql")] - ConnectionInfo::Mssql(_) => SqlFamily::Mssql, - #[cfg(feature = "sqlite")] - ConnectionInfo::Sqlite { .. } | ConnectionInfo::InMemorySqlite { .. } => SqlFamily::Sqlite, + #[cfg(not(target_arch = "wasm32"))] + ConnectionInfo::Native(info) => match info { + #[cfg(feature = "postgresql")] + NativeConnectionInfo::Postgres(_) => SqlFamily::Postgres, + #[cfg(feature = "mysql")] + NativeConnectionInfo::Mysql(_) => SqlFamily::Mysql, + #[cfg(feature = "mssql")] + NativeConnectionInfo::Mssql(_) => SqlFamily::Mssql, + #[cfg(feature = "sqlite")] + NativeConnectionInfo::Sqlite { .. } | NativeConnectionInfo::InMemorySqlite { .. } => SqlFamily::Sqlite, + }, ConnectionInfo::External(info) => info.sql_family.to_owned(), } } @@ -200,24 +211,26 @@ impl ConnectionInfo { /// The provided database port, if applicable. pub fn port(&self) -> Option { match self { - #[cfg(feature = "postgresql")] - ConnectionInfo::Postgres(url) => Some(url.port()), - #[cfg(feature = "mysql")] - ConnectionInfo::Mysql(url) => Some(url.port()), - #[cfg(feature = "mssql")] - ConnectionInfo::Mssql(url) => Some(url.port()), - #[cfg(feature = "sqlite")] - ConnectionInfo::Sqlite { .. } | ConnectionInfo::InMemorySqlite { .. } => None, + #[cfg(not(target_arch = "wasm32"))] + ConnectionInfo::Native(info) => match info { + #[cfg(feature = "postgresql")] + NativeConnectionInfo::Postgres(url) => Some(url.port()), + #[cfg(feature = "mysql")] + NativeConnectionInfo::Mysql(url) => Some(url.port()), + #[cfg(feature = "mssql")] + NativeConnectionInfo::Mssql(url) => Some(url.port()), + #[cfg(feature = "sqlite")] + NativeConnectionInfo::Sqlite { .. } | NativeConnectionInfo::InMemorySqlite { .. } => None, + }, ConnectionInfo::External(_) => None, } } /// Whether the pgbouncer mode is enabled. pub fn pg_bouncer(&self) -> bool { - #[allow(unreachable_patterns)] match self { - #[cfg(feature = "postgresql")] - ConnectionInfo::Postgres(url) => url.pg_bouncer(), + #[cfg(all(not(target_arch = "wasm32"), feature = "postgresql"))] + ConnectionInfo::Native(NativeConnectionInfo::Postgres(url)) => url.pg_bouncer(), _ => false, } } @@ -226,16 +239,19 @@ impl ConnectionInfo { /// and port on MySQL/Postgres, and the file path on SQLite. pub fn database_location(&self) -> String { match self { - #[cfg(feature = "postgresql")] - ConnectionInfo::Postgres(url) => format!("{}:{}", url.host(), url.port()), - #[cfg(feature = "mysql")] - ConnectionInfo::Mysql(url) => format!("{}:{}", url.host(), url.port()), - #[cfg(feature = "mssql")] - ConnectionInfo::Mssql(url) => format!("{}:{}", url.host(), url.port()), - #[cfg(feature = "sqlite")] - ConnectionInfo::Sqlite { file_path, .. } => file_path.clone(), - #[cfg(feature = "sqlite")] - ConnectionInfo::InMemorySqlite { .. } => "in-memory".into(), + #[cfg(not(target_arch = "wasm32"))] + ConnectionInfo::Native(info) => match info { + #[cfg(feature = "postgresql")] + NativeConnectionInfo::Postgres(url) => format!("{}:{}", url.host(), url.port()), + #[cfg(feature = "mysql")] + NativeConnectionInfo::Mysql(url) => format!("{}:{}", url.host(), url.port()), + #[cfg(feature = "mssql")] + NativeConnectionInfo::Mssql(url) => format!("{}:{}", url.host(), url.port()), + #[cfg(feature = "sqlite")] + NativeConnectionInfo::Sqlite { file_path, .. } => file_path.clone(), + #[cfg(feature = "sqlite")] + NativeConnectionInfo::InMemorySqlite { .. } => "in-memory".into(), + }, ConnectionInfo::External(_) => "external".into(), } } @@ -353,7 +369,7 @@ mod tests { let conn_info = ConnectionInfo::from_url("file:dev.db").unwrap(); #[allow(irrefutable_let_patterns)] - if let ConnectionInfo::Sqlite { file_path, db_name: _ } = conn_info { + if let ConnectionInfo::Native(NativeConnectionInfo::Sqlite { file_path, db_name: _ }) = conn_info { assert_eq!(file_path, "dev.db"); } else { panic!("Wrong type of connection info, should be Sqlite"); @@ -366,7 +382,7 @@ mod tests { let conn_info = ConnectionInfo::from_url("mysql://myuser:my%23pass%23word@lclhst:5432/mydb").unwrap(); #[allow(irrefutable_let_patterns)] - if let ConnectionInfo::Mysql(url) = conn_info { + if let ConnectionInfo::Native(NativeConnectionInfo::Mysql(url)) = conn_info { assert_eq!(url.password().unwrap(), "my#pass#word"); assert_eq!(url.host(), "lclhst"); assert_eq!(url.username(), "myuser"); diff --git a/quaint/src/connector/mssql/native/error.rs b/quaint/src/connector/mssql/native/error.rs index f9b6f5e95ab6..9c16bf9f2952 100644 --- a/quaint/src/connector/mssql/native/error.rs +++ b/quaint/src/connector/mssql/native/error.rs @@ -1,4 +1,4 @@ -use crate::error::{DatabaseConstraint, Error, ErrorKind}; +use crate::error::{DatabaseConstraint, Error, ErrorKind, NativeErrorKind}; use tiberius::error::IoErrorKind; impl From for Error { @@ -8,17 +8,19 @@ impl From for Error { kind: IoErrorKind::UnexpectedEof, message, } => { - let mut builder = Error::builder(ErrorKind::ConnectionClosed); + let mut builder = Error::builder(ErrorKind::Native(NativeErrorKind::ConnectionClosed)); builder.set_original_message(message); builder.build() } - e @ tiberius::error::Error::Io { .. } => Error::builder(ErrorKind::ConnectionError(e.into())).build(), + e @ tiberius::error::Error::Io { .. } => { + Error::builder(ErrorKind::Native(NativeErrorKind::ConnectionError(e.into()))).build() + } tiberius::error::Error::Tls(message) => { let message = format!( "The TLS settings didn't allow the connection to be established. Please review your connection string. (error: {message})" ); - Error::builder(ErrorKind::TlsError { message }).build() + Error::builder(ErrorKind::Native(NativeErrorKind::TlsError { message })).build() } tiberius::error::Error::Server(e) if [3902u32, 3903u32, 3971u32].iter().any(|code| e.code() == *code) => { let kind = ErrorKind::TransactionAlreadyClosed(e.message().to_string()); diff --git a/quaint/src/connector/mysql/native/error.rs b/quaint/src/connector/mysql/native/error.rs index 89c21fb706f6..0d9e58ccd9dc 100644 --- a/quaint/src/connector/mysql/native/error.rs +++ b/quaint/src/connector/mysql/native/error.rs @@ -1,6 +1,6 @@ use crate::{ connector::mysql::error::MysqlError, - error::{Error, ErrorKind}, + error::{Error, ErrorKind, NativeErrorKind}, }; use mysql_async as my; @@ -17,14 +17,16 @@ impl From<&my::ServerError> for MysqlError { impl From for Error { fn from(e: my::Error) -> Error { match e { - my::Error::Io(my::IoError::Tls(err)) => Error::builder(ErrorKind::TlsError { + my::Error::Io(my::IoError::Tls(err)) => Error::builder(ErrorKind::Native(NativeErrorKind::TlsError { message: err.to_string(), - }) + })) .build(), my::Error::Io(my::IoError::Io(err)) if err.kind() == std::io::ErrorKind::UnexpectedEof => { - Error::builder(ErrorKind::ConnectionClosed).build() + Error::builder(ErrorKind::Native(NativeErrorKind::ConnectionClosed)).build() + } + my::Error::Io(io_error) => { + Error::builder(ErrorKind::Native(NativeErrorKind::ConnectionError(io_error.into()))).build() } - my::Error::Io(io_error) => Error::builder(ErrorKind::ConnectionError(io_error.into())).build(), my::Error::Driver(e) => Error::builder(ErrorKind::QueryError(e.into())).build(), my::Error::Server(ref server_error) => { let mysql_error: MysqlError = server_error.into(); diff --git a/quaint/src/connector/native.rs b/quaint/src/connector/native.rs new file mode 100644 index 000000000000..b9cf4b9858e6 --- /dev/null +++ b/quaint/src/connector/native.rs @@ -0,0 +1,31 @@ +#[cfg(feature = "mssql")] +use crate::connector::MssqlUrl; +#[cfg(feature = "mysql")] +use crate::connector::MysqlUrl; +#[cfg(feature = "postgresql")] +use crate::connector::PostgresUrl; + +/// General information about a SQL connection, provided by native Rust drivers. +#[cfg(not(target_arch = "wasm32"))] +#[derive(Debug, Clone)] +pub enum NativeConnectionInfo { + /// A PostgreSQL connection URL. + #[cfg(feature = "postgresql")] + Postgres(PostgresUrl), + /// A MySQL connection URL. + #[cfg(feature = "mysql")] + Mysql(MysqlUrl), + /// A SQL Server connection URL. + #[cfg(feature = "mssql")] + Mssql(MssqlUrl), + /// A SQLite connection URL. + #[cfg(feature = "sqlite")] + Sqlite { + /// The filesystem path of the SQLite database. + file_path: String, + /// The name the database is bound to - Always "main" + db_name: String, + }, + #[cfg(feature = "sqlite")] + InMemorySqlite { db_name: String }, +} diff --git a/quaint/src/connector/postgres/native/error.rs b/quaint/src/connector/postgres/native/error.rs index c353e397705c..6ceb26299691 100644 --- a/quaint/src/connector/postgres/native/error.rs +++ b/quaint/src/connector/postgres/native/error.rs @@ -2,7 +2,7 @@ use tokio_postgres::error::DbError; use crate::{ connector::postgres::error::PostgresError, - error::{Error, ErrorKind}, + error::{Error, ErrorKind, NativeErrorKind}, }; impl From<&DbError> for PostgresError { @@ -21,7 +21,7 @@ impl From<&DbError> for PostgresError { impl From for Error { fn from(e: tokio_postgres::error::Error) -> Error { if e.is_closed() { - return Error::builder(ErrorKind::ConnectionClosed).build(); + return Error::builder(ErrorKind::Native(NativeErrorKind::ConnectionClosed)).build(); } if let Some(db_error) = e.as_db_error() { @@ -46,7 +46,7 @@ impl From for Error { match reason.as_str() { "error connecting to server: timed out" => { - let mut builder = Error::builder(ErrorKind::ConnectTimeout); + let mut builder = Error::builder(ErrorKind::Native(NativeErrorKind::ConnectTimeout)); if let Some(code) = code { builder.set_original_code(code); @@ -57,9 +57,9 @@ impl From for Error { } // sigh... // https://github.com/sfackler/rust-postgres/blob/0c84ed9f8201f4e5b4803199a24afa2c9f3723b2/tokio-postgres/src/connect_tls.rs#L37 "error performing TLS handshake: server does not support TLS" => { - let mut builder = Error::builder(ErrorKind::TlsError { + let mut builder = Error::builder(ErrorKind::Native(NativeErrorKind::TlsError { message: reason.clone(), - }); + })); if let Some(code) = code { builder.set_original_code(code); @@ -105,7 +105,12 @@ fn try_extracting_io_error(err: &tokio_postgres::error::Error) -> Option err.source() .and_then(|err| err.downcast_ref::()) - .map(|err| ErrorKind::ConnectionError(Box::new(std::io::Error::new(err.kind(), format!("{err}"))))) + .map(|err| { + ErrorKind::Native(NativeErrorKind::ConnectionError(Box::new(std::io::Error::new( + err.kind(), + format!("{err}"), + )))) + }) .map(|kind| Error::builder(kind).build()) } @@ -117,9 +122,9 @@ impl From for Error { impl From<&native_tls::Error> for Error { fn from(e: &native_tls::Error) -> Error { - let kind = ErrorKind::TlsError { + let kind = ErrorKind::Native(NativeErrorKind::TlsError { message: format!("{e}"), - }; + }); Error::builder(kind).build() } diff --git a/quaint/src/connector/postgres/native/mod.rs b/quaint/src/connector/postgres/native/mod.rs index d656bceb1e00..2f8496d40ff5 100644 --- a/quaint/src/connector/postgres/native/mod.rs +++ b/quaint/src/connector/postgres/native/mod.rs @@ -7,6 +7,7 @@ mod error; pub(crate) use crate::connector::postgres::url::PostgresUrl; use crate::connector::postgres::url::{Hidden, SslAcceptMode, SslParams}; use crate::connector::{timeout, IsolationLevel, Transaction}; +use crate::error::NativeErrorKind; use crate::{ ast::{Query, Value}, @@ -93,9 +94,9 @@ impl SslParams { if let Some(ref cert_file) = self.certificate_file { let cert = fs::read(cert_file).map_err(|err| { - Error::builder(ErrorKind::TlsError { + Error::builder(ErrorKind::Native(NativeErrorKind::TlsError { message: format!("cert file not found ({err})"), - }) + })) .build() })?; @@ -104,9 +105,9 @@ impl SslParams { if let Some(ref identity_file) = self.identity_file { let db = fs::read(identity_file).map_err(|err| { - Error::builder(ErrorKind::TlsError { + Error::builder(ErrorKind::Native(NativeErrorKind::TlsError { message: format!("identity file not found ({err})"), - }) + })) .build() })?; let password = self.identity_password.0.as_deref().unwrap_or(""); diff --git a/quaint/src/connector/postgres/url.rs b/quaint/src/connector/postgres/url.rs index 4970ae6b1a9d..3d8c803e0954 100644 --- a/quaint/src/connector/postgres/url.rs +++ b/quaint/src/connector/postgres/url.rs @@ -608,7 +608,7 @@ mod tests { match res { Ok(_) => unreachable!(), Err(e) => match e.kind() { - ErrorKind::TlsError { .. } => (), + ErrorKind::Native(NativeErrorKind::TlsError { .. }) => (), other => panic!("{:#?}", other), }, } diff --git a/quaint/src/connector/timeout.rs b/quaint/src/connector/timeout.rs index 7eec9dcd0506..a0445c4c7a26 100644 --- a/quaint/src/connector/timeout.rs +++ b/quaint/src/connector/timeout.rs @@ -2,12 +2,16 @@ use crate::error::{Error, ErrorKind}; use futures::Future; use std::time::Duration; +#[cfg(feature = "native")] pub async fn connect(duration: Option, f: F) -> crate::Result where F: Future>, E: Into, { - timeout(duration, f, || Error::builder(ErrorKind::ConnectTimeout).build()).await + timeout(duration, f, || { + Error::builder(ErrorKind::Native(crate::error::NativeErrorKind::ConnectTimeout)).build() + }) + .await } pub async fn socket(duration: Option, f: F) -> crate::Result diff --git a/quaint/src/error.rs b/quaint/src/error/mod.rs similarity index 84% rename from quaint/src/error.rs rename to quaint/src/error/mod.rs index 99427addbfff..aec1adc1648a 100644 --- a/quaint/src/error.rs +++ b/quaint/src/error/mod.rs @@ -1,14 +1,24 @@ //! Error module + +#[cfg(not(target_arch = "wasm32"))] +pub mod native; + +pub(crate) mod name; + use crate::connector::IsolationLevel; -use std::{borrow::Cow, fmt, io, num}; +use std::{borrow::Cow, fmt, num}; use thiserror::Error; #[cfg(feature = "pooled")] use std::time::Duration; +#[cfg(not(target_arch = "wasm32"))] +pub use native::NativeErrorKind; + pub use crate::connector::mysql::MysqlError; pub use crate::connector::postgres::PostgresError; pub use crate::connector::sqlite::SqliteError; +pub(crate) use name::Name; #[derive(Debug, PartialEq, Eq)] pub enum DatabaseConstraint { @@ -41,39 +51,6 @@ impl fmt::Display for DatabaseConstraint { } } -#[derive(Debug, PartialEq, Eq)] -pub enum Name { - Available(String), - Unavailable, -} - -impl Name { - pub fn available(name: impl ToString) -> Self { - Self::Available(name.to_string()) - } -} - -impl fmt::Display for Name { - fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { - match self { - Self::Available(name) => name.fmt(f), - Self::Unavailable => write!(f, "(not available)"), - } - } -} - -impl From> for Name -where - T: ToString, -{ - fn from(name: Option) -> Self { - match name { - Some(name) => Self::available(name), - None => Self::Unavailable, - } - } -} - #[derive(Debug, Error)] /// The error types for database I/O, connection and query parameter /// construction. @@ -134,8 +111,9 @@ impl Error { } /// Determines if the error was associated with closed connection. + #[cfg(not(target_arch = "wasm32"))] pub fn is_closed(&self) -> bool { - matches!(self.kind, ErrorKind::ConnectionClosed) + matches!(self.kind, ErrorKind::Native(NativeErrorKind::ConnectionClosed)) } // Builds an error from a raw error coming from the connector @@ -157,6 +135,10 @@ impl fmt::Display for Error { #[derive(Debug, Error)] pub enum ErrorKind { + #[cfg(not(target_arch = "wasm32"))] + #[error("Error in the underlying connector")] + Native(NativeErrorKind), + #[error("Error in the underlying connector ({}): {}", status, reason)] RawConnectorError { status: String, reason: String }, @@ -193,9 +175,6 @@ pub enum ErrorKind { #[error("Foreign key constraint failed: {}", constraint)] ForeignKeyConstraintViolation { constraint: DatabaseConstraint }, - #[error("Error creating a database connection.")] - ConnectionError(Box), - #[error("Error reading the column value: {}", _0)] ColumnReadFailure(Box), @@ -220,32 +199,9 @@ pub enum ErrorKind { #[error("The provided arguments are not supported")] InvalidConnectionArguments, - #[error("Error in an I/O operation: {0}")] - IoError(io::Error), - - #[error("Timed out when connecting to the database.")] - ConnectTimeout, - - #[error("The server terminated the connection.")] - ConnectionClosed, - - #[error( - "Timed out fetching a connection from the pool (connection limit: {}, in use: {}, pool timeout {})", - max_open, - in_use, - timeout - )] - PoolTimeout { max_open: u64, in_use: u64, timeout: u64 }, - - #[error("The connection pool has been closed")] - PoolClosed {}, - #[error("Timed out during query execution.")] SocketTimeout, - #[error("Error opening a TLS connection. {}", message)] - TlsError { message: String }, - #[error("Value out of range error. {}", message)] ValueOutOfRange { message: String }, @@ -281,6 +237,13 @@ pub enum ErrorKind { ExternalError(i32), } +#[cfg(not(target_arch = "wasm32"))] +impl From for Error { + fn from(e: std::io::Error) -> Error { + Error::builder(ErrorKind::Native(NativeErrorKind::IoError(e))).build() + } +} + impl ErrorKind { #[cfg(feature = "mysql-native")] pub(crate) fn value_out_of_range(msg: impl Into) -> Self { @@ -298,11 +261,11 @@ impl ErrorKind { #[cfg(feature = "pooled")] pub(crate) fn pool_timeout(max_open: u64, in_use: u64, timeout: Duration) -> Self { - Self::PoolTimeout { + Self::Native(NativeErrorKind::PoolTimeout { max_open, in_use, timeout: timeout.as_secs(), - } + }) } pub fn invalid_isolation_level(isolation_level: &IsolationLevel) -> Self { @@ -357,12 +320,6 @@ impl From for Error { } } -impl From for Error { - fn from(e: io::Error) -> Error { - Error::builder(ErrorKind::IoError(e)).build() - } -} - impl From for Error { fn from(_e: std::num::ParseIntError) -> Error { Error::builder(ErrorKind::conversion("Couldn't convert data to an integer")).build() diff --git a/quaint/src/error/name.rs b/quaint/src/error/name.rs new file mode 100644 index 000000000000..f4d48a47331b --- /dev/null +++ b/quaint/src/error/name.rs @@ -0,0 +1,32 @@ +#[derive(Debug, PartialEq, Eq)] +pub enum Name { + Available(String), + Unavailable, +} + +impl Name { + pub fn available(name: impl ToString) -> Self { + Self::Available(name.to_string()) + } +} + +impl std::fmt::Display for Name { + fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { + match self { + Self::Available(name) => name.fmt(f), + Self::Unavailable => write!(f, "(not available)"), + } + } +} + +impl From> for Name +where + T: ToString, +{ + fn from(name: Option) -> Self { + match name { + Some(name) => Self::available(name), + None => Self::Unavailable, + } + } +} diff --git a/quaint/src/error/native.rs b/quaint/src/error/native.rs new file mode 100644 index 000000000000..e9479b64f200 --- /dev/null +++ b/quaint/src/error/native.rs @@ -0,0 +1,30 @@ +use thiserror::Error; + +#[derive(Debug, Error)] +pub enum NativeErrorKind { + #[error("Error creating a database connection.")] + ConnectionError(Box), + + #[error("The server terminated the connection.")] + ConnectionClosed, + + #[error("The connection pool has been closed")] + PoolClosed {}, + + #[error( + "Timed out fetching a connection from the pool (connection limit: {}, in use: {}, pool timeout {})", + max_open, + in_use, + timeout + )] + PoolTimeout { max_open: u64, in_use: u64, timeout: u64 }, + + #[error("Error in an I/O operation: {0}")] + IoError(std::io::Error), + + #[error("Timed out when connecting to the database.")] + ConnectTimeout, + + #[error("Error opening a TLS connection. {}", message)] + TlsError { message: String }, +} diff --git a/quaint/src/pooled.rs b/quaint/src/pooled.rs index 4c4152923377..2dc1a843eba1 100644 --- a/quaint/src/pooled.rs +++ b/quaint/src/pooled.rs @@ -152,6 +152,9 @@ mod manager; pub use manager::*; +#[cfg(feature = "native")] +use crate::{connector::NativeConnectionInfo, error::NativeErrorKind}; + use crate::{ connector::{ConnectionInfo, PostgresFlavour}, error::{Error, ErrorKind}, @@ -303,7 +306,7 @@ impl Builder { /// /// - Defaults to `PostgresFlavour::Unknown`. pub fn set_postgres_flavour(&mut self, flavour: PostgresFlavour) { - if let ConnectionInfo::Postgres(ref mut url) = self.connection_info { + if let ConnectionInfo::Native(NativeConnectionInfo::Postgres(ref mut url)) = self.connection_info { url.set_flavour(flavour); } @@ -484,7 +487,9 @@ impl Quaint { let inner = match res { Ok(conn) => conn, - Err(mobc::Error::PoolClosed) => return Err(Error::builder(ErrorKind::PoolClosed {}).build()), + Err(mobc::Error::PoolClosed) => { + return Err(Error::builder(ErrorKind::Native(NativeErrorKind::PoolClosed {})).build()) + } Err(mobc::Error::Timeout) => { let state = self.inner.state().await; // We can use unwrap here because a pool timeout has to be set to use a connection pool @@ -495,7 +500,7 @@ impl Quaint { } Err(mobc::Error::Inner(e)) => return Err(e), Err(e @ mobc::Error::BadConn) => { - let error = Error::builder(ErrorKind::ConnectionError(Box::new(e))).build(); + let error = Error::builder(ErrorKind::Native(NativeErrorKind::ConnectionError(Box::new(e)))).build(); return Err(error); } }; diff --git a/quaint/src/prelude.rs b/quaint/src/prelude.rs index adaa701210f1..6b28926a5f43 100644 --- a/quaint/src/prelude.rs +++ b/quaint/src/prelude.rs @@ -5,3 +5,6 @@ pub use crate::connector::{ TransactionCapable, }; pub use crate::{col, val, values}; + +#[cfg(feature = "native")] +pub use crate::connector::NativeConnectionInfo; diff --git a/quaint/src/single.rs b/quaint/src/single.rs index b819259d81c7..653ac990b2e2 100644 --- a/quaint/src/single.rs +++ b/quaint/src/single.rs @@ -10,6 +10,9 @@ use std::{fmt, sync::Arc}; #[cfg(feature = "sqlite-native")] use std::convert::TryFrom; +#[cfg(feature = "native")] +use crate::connector::NativeConnectionInfo; + /// The main entry point and an abstraction over a database connection. #[derive(Clone)] pub struct Quaint { @@ -125,7 +128,7 @@ impl Quaint { /// - `isolationLevel` the transaction isolation level. Possible values: /// `READ UNCOMMITTED`, `READ COMMITTED`, `REPEATABLE READ`, `SNAPSHOT`, /// `SERIALIZABLE`. - #[cfg_attr(target_arch = "wasm32", allow(unused_variables))] + #[cfg(feature = "native")] #[allow(unreachable_code)] pub async fn new(url_str: &str) -> crate::Result { let inner = match url_str { @@ -172,9 +175,9 @@ impl Quaint { Ok(Quaint { inner: Arc::new(connector::Sqlite::new_in_memory()?), - connection_info: Arc::new(ConnectionInfo::InMemorySqlite { + connection_info: Arc::new(ConnectionInfo::Native(NativeConnectionInfo::InMemorySqlite { db_name: DEFAULT_SQLITE_DATABASE.to_owned(), - }), + })), }) } @@ -183,6 +186,7 @@ impl Quaint { &self.connection_info } + #[cfg(feature = "native")] fn log_start(info: &ConnectionInfo) { let family = info.sql_family(); let pg_bouncer = if info.pg_bouncer() { " in PgBouncer mode" } else { "" }; diff --git a/query-engine/connectors/sql-query-connector/src/error.rs b/query-engine/connectors/sql-query-connector/src/error.rs index f3ce28abbdc7..feb47fcbbb44 100644 --- a/query-engine/connectors/sql-query-connector/src/error.rs +++ b/query-engine/connectors/sql-query-connector/src/error.rs @@ -5,13 +5,17 @@ use std::{any::Any, string::FromUtf8Error}; use thiserror::Error; use user_facing_errors::query_engine::DatabaseConstraint; -pub(crate) enum RawError { - IncorrectNumberOfParameters { - expected: usize, - actual: usize, - }, - QueryInvalidInput(String), +#[cfg(not(target_arch = "wasm32"))] +use quaint::error::NativeErrorKind; + +#[cfg(not(target_arch = "wasm32"))] +pub(crate) enum NativeRawError { ConnectionClosed, +} + +pub(crate) enum RawError { + #[cfg(not(target_arch = "wasm32"))] + Native(NativeRawError), Database { code: Option, message: Option, @@ -19,6 +23,11 @@ pub(crate) enum RawError { UnsupportedColumnType { column_type: String, }, + IncorrectNumberOfParameters { + expected: usize, + actual: usize, + }, + QueryInvalidInput(String), External { id: i32, }, @@ -27,6 +36,10 @@ pub(crate) enum RawError { impl From for SqlError { fn from(re: RawError) -> SqlError { match re { + #[cfg(not(target_arch = "wasm32"))] + RawError::Native(native) => match native { + NativeRawError::ConnectionClosed => SqlError::ConnectionClosed, + }, RawError::IncorrectNumberOfParameters { expected, actual } => { Self::IncorrectNumberOfParameters { expected, actual } } @@ -37,7 +50,6 @@ impl From for SqlError { r#"Failed to deserialize column of type '{column_type}'. If you're using $queryRaw and this column is explicitly marked as `Unsupported` in your Prisma schema, try casting this column to any supported Prisma type such as `String`."# ), }, - RawError::ConnectionClosed => Self::ConnectionClosed, RawError::Database { code, message } => Self::RawError { code: code.unwrap_or_else(|| String::from("N/A")), message: message.unwrap_or_else(|| String::from("N/A")), @@ -49,23 +61,28 @@ impl From for SqlError { impl From for RawError { fn from(e: quaint::error::Error) -> Self { + let default_value: RawError = Self::Database { + code: e.original_code().map(ToString::to_string), + message: e.original_message().map(ToString::to_string), + }; + match e.kind() { + #[cfg(not(target_arch = "wasm32"))] + quaint::error::ErrorKind::Native(NativeErrorKind::ConnectionClosed) => { + Self::Native(NativeRawError::ConnectionClosed) + } quaint::error::ErrorKind::IncorrectNumberOfParameters { expected, actual } => { Self::IncorrectNumberOfParameters { expected: *expected, actual: *actual, } } - quaint::error::ErrorKind::ConnectionClosed => Self::ConnectionClosed, quaint::error::ErrorKind::UnsupportedColumnType { column_type } => Self::UnsupportedColumnType { column_type: column_type.to_owned(), }, quaint::error::ErrorKind::QueryInvalidInput(message) => Self::QueryInvalidInput(message.to_owned()), quaint::error::ErrorKind::ExternalError(id) => Self::External { id: *id }, - _ => Self::Database { - code: e.original_code().map(ToString::to_string), - message: e.original_message().map(ToString::to_string), - }, + _ => default_value, } } } @@ -276,15 +293,26 @@ impl From for SqlError { } impl From for SqlError { - fn from(e: quaint::error::Error) -> Self { - match QuaintKind::from(e) { + fn from(error: quaint::error::Error) -> Self { + let quaint_kind = QuaintKind::from(error); + + match quaint_kind { + #[cfg(not(target_arch = "wasm32"))] + QuaintKind::Native(ref native_error_kind) => match native_error_kind { + NativeErrorKind::IoError(_) | NativeErrorKind::ConnectionError(_) => Self::ConnectionError(quaint_kind), + NativeErrorKind::ConnectionClosed => SqlError::ConnectionClosed, + NativeErrorKind::ConnectTimeout => SqlError::ConnectionError(quaint_kind), + NativeErrorKind::PoolTimeout { .. } => SqlError::ConnectionError(quaint_kind), + NativeErrorKind::PoolClosed { .. } => SqlError::ConnectionError(quaint_kind), + NativeErrorKind::TlsError { .. } => Self::ConnectionError(quaint_kind), + }, + QuaintKind::RawConnectorError { status, reason } => Self::RawError { code: status, message: reason, }, QuaintKind::QueryError(qe) => Self::QueryError(qe), QuaintKind::QueryInvalidInput(qe) => Self::QueryInvalidInput(qe), - e @ QuaintKind::IoError(_) => Self::ConnectionError(e), QuaintKind::NotFound => Self::RecordDoesNotExist { cause: "Record not found".to_owned(), }, @@ -300,11 +328,10 @@ impl From for SqlError { constraint: constraint.into(), }, QuaintKind::MissingFullTextSearchIndex => Self::MissingFullTextSearchIndex, - e @ QuaintKind::ConnectionError(_) => Self::ConnectionError(e), QuaintKind::ColumnReadFailure(e) => Self::ColumnReadFailure(e), QuaintKind::ColumnNotFound { column } => SqlError::ColumnDoesNotExist(format!("{column}")), QuaintKind::TableDoesNotExist { table } => SqlError::TableDoesNotExist(format!("{table}")), - QuaintKind::ConnectionClosed => SqlError::ConnectionClosed, + QuaintKind::InvalidIsolationLevel(msg) => Self::InvalidIsolationLevel(msg), QuaintKind::TransactionWriteConflict => Self::TransactionWriteConflict, QuaintKind::RollbackWithoutBegin => Self::RollbackWithoutBegin, @@ -324,11 +351,7 @@ impl From for SqlError { e @ QuaintKind::DatabaseAccessDenied { .. } => SqlError::ConnectionError(e), e @ QuaintKind::DatabaseAlreadyExists { .. } => SqlError::ConnectionError(e), e @ QuaintKind::InvalidConnectionArguments => SqlError::ConnectionError(e), - e @ QuaintKind::ConnectTimeout => SqlError::ConnectionError(e), e @ QuaintKind::SocketTimeout => SqlError::ConnectionError(e), - e @ QuaintKind::PoolTimeout { .. } => SqlError::ConnectionError(e), - e @ QuaintKind::PoolClosed { .. } => SqlError::ConnectionError(e), - e @ QuaintKind::TlsError { .. } => Self::ConnectionError(e), } } } diff --git a/schema-engine/connectors/sql-schema-connector/src/flavour/mssql/connection.rs b/schema-engine/connectors/sql-schema-connector/src/flavour/mssql/connection.rs index 88094a0c3b4d..580c3a186381 100644 --- a/schema-engine/connectors/sql-schema-connector/src/flavour/mssql/connection.rs +++ b/schema-engine/connectors/sql-schema-connector/src/flavour/mssql/connection.rs @@ -2,7 +2,7 @@ use quaint::{ connector::{self, MssqlUrl}, - prelude::{ConnectionInfo, Queryable}, + prelude::{ConnectionInfo, NativeConnectionInfo, Queryable}, }; use schema_connector::{ConnectorError, ConnectorResult, Namespaces}; use sql_schema_describer::{mssql as describer, DescriberErrorKind, SqlSchema, SqlSchemaDescriberBackend}; @@ -104,5 +104,10 @@ fn quaint_err(params: &super::Params) -> impl (Fn(quaint::error::Error) -> Conne } fn quaint_err_url(url: &MssqlUrl) -> impl (Fn(quaint::error::Error) -> ConnectorError) + '_ { - |err| crate::flavour::quaint_error_to_connector_error(err, &ConnectionInfo::Mssql(url.clone())) + |err| { + crate::flavour::quaint_error_to_connector_error( + err, + &ConnectionInfo::Native(NativeConnectionInfo::Mssql(url.clone())), + ) + } } diff --git a/schema-engine/connectors/sql-schema-connector/src/flavour/mysql/connection.rs b/schema-engine/connectors/sql-schema-connector/src/flavour/mysql/connection.rs index f52dc91aff93..c0d216ef4a4d 100644 --- a/schema-engine/connectors/sql-schema-connector/src/flavour/mysql/connection.rs +++ b/schema-engine/connectors/sql-schema-connector/src/flavour/mysql/connection.rs @@ -7,7 +7,7 @@ use quaint::{ mysql_async::{self as my, prelude::Query}, MysqlUrl, }, - prelude::{ConnectionInfo, Queryable}, + prelude::{ConnectionInfo, NativeConnectionInfo, Queryable}, }; use schema_connector::{ConnectorError, ConnectorResult}; use sql_schema_describer::{DescriberErrorKind, SqlSchema}; @@ -160,7 +160,12 @@ impl Connection { } fn quaint_err(url: &MysqlUrl) -> impl (Fn(quaint::error::Error) -> ConnectorError) + '_ { - |err| crate::flavour::quaint_error_to_connector_error(err, &ConnectionInfo::Mysql(url.clone())) + |err| { + crate::flavour::quaint_error_to_connector_error( + err, + &ConnectionInfo::Native(NativeConnectionInfo::Mysql(url.clone())), + ) + } } fn convert_server_error(circumstances: BitFlags, error: &my::Error) -> Option { diff --git a/schema-engine/connectors/sql-schema-connector/src/flavour/postgres/connection.rs b/schema-engine/connectors/sql-schema-connector/src/flavour/postgres/connection.rs index 9db4ed56b859..c5f4c645916e 100644 --- a/schema-engine/connectors/sql-schema-connector/src/flavour/postgres/connection.rs +++ b/schema-engine/connectors/sql-schema-connector/src/flavour/postgres/connection.rs @@ -5,7 +5,7 @@ use indoc::indoc; use psl::PreviewFeature; use quaint::{ connector::{self, tokio_postgres::error::ErrorPosition, PostgresUrl}, - prelude::{ConnectionInfo, Queryable}, + prelude::{ConnectionInfo, NativeConnectionInfo, Queryable}, }; use schema_connector::{ConnectorError, ConnectorResult, Namespaces}; use sql_schema_describer::{postgres::PostgresSchemaExt, SqlSchema}; @@ -202,5 +202,10 @@ fn normalize_sql_schema(schema: &mut SqlSchema, preview_features: BitFlags impl (Fn(quaint::error::Error) -> ConnectorError) + '_ { - |err| crate::flavour::quaint_error_to_connector_error(err, &ConnectionInfo::Postgres(url.clone())) + |err| { + crate::flavour::quaint_error_to_connector_error( + err, + &ConnectionInfo::Native(NativeConnectionInfo::Postgres(url.clone())), + ) + } } diff --git a/schema-engine/sql-migration-tests/src/multi_engine_test_api.rs b/schema-engine/sql-migration-tests/src/multi_engine_test_api.rs index aeaa059bccfd..79c745aa86d3 100644 --- a/schema-engine/sql-migration-tests/src/multi_engine_test_api.rs +++ b/schema-engine/sql-migration-tests/src/multi_engine_test_api.rs @@ -14,7 +14,7 @@ use crate::{ }; use psl::PreviewFeature; use quaint::{ - prelude::{ConnectionInfo, Queryable, ResultSet}, + prelude::{ConnectionInfo, NativeConnectionInfo, Queryable, ResultSet}, single::Quaint, }; use schema_core::schema_connector::{ConnectorParams, SchemaConnector}; @@ -196,17 +196,19 @@ impl TestApi { }; let mut connector = match &connection_info { - ConnectionInfo::Postgres(_) => { + ConnectionInfo::Native(NativeConnectionInfo::Postgres(_)) => { if self.args.provider() == "cockroachdb" { SqlSchemaConnector::new_cockroach() } else { SqlSchemaConnector::new_postgres() } } - ConnectionInfo::Mysql(_) => SqlSchemaConnector::new_mysql(), - ConnectionInfo::Mssql(_) => SqlSchemaConnector::new_mssql(), - ConnectionInfo::Sqlite { .. } => SqlSchemaConnector::new_sqlite(), - ConnectionInfo::InMemorySqlite { .. } | ConnectionInfo::External(_) => unreachable!(), + ConnectionInfo::Native(NativeConnectionInfo::Mysql(_)) => SqlSchemaConnector::new_mysql(), + ConnectionInfo::Native(NativeConnectionInfo::Mssql(_)) => SqlSchemaConnector::new_mssql(), + ConnectionInfo::Native(NativeConnectionInfo::Sqlite { .. }) => SqlSchemaConnector::new_sqlite(), + ConnectionInfo::Native(NativeConnectionInfo::InMemorySqlite { .. }) | ConnectionInfo::External(_) => { + unreachable!() + } }; connector.set_params(params).unwrap();