Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Sqlite bind review #7

Closed
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -186,6 +186,11 @@ for Rust libraries in [RFC #1105](https://github.com/rust-lang/rfcs/blob/master/

* The MySQL connection is using the CLIENT_FOUND_ROWS from now on. This means that updating rows without changing any values will return the number of matched rows (like most other SQL servers do), as opposed to the number of changed rows.

* The definition of `ToSql::to_sql` and `QueryFragment::walk_ast` has changed to allow serializing values without
copying the value itself. This is useful for database backends like sqlite where you can directly share a buffer
with the database. Beside of the changed signature, existing impls of this trait should remain unchanged in almost
all cases.

### Fixed

* Many types were incorrectly considered non-aggregate when they should not
Expand Down
8 changes: 4 additions & 4 deletions diesel/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@ categories = ["database"]
edition = "2018"

[dependencies]
byteorder = "1.0"
byteorder = {version = "1.0", optional = true}
chrono = { version = "0.4.19", optional = true, default-features = false, features = ["clock", "std"] }
libc = { version = "0.2.0", optional = true }
libsqlite3-sys = { version = ">=0.8.0, <0.24.0", optional = true, features = ["bundled_bindings"] }
Expand Down Expand Up @@ -44,7 +44,7 @@ ipnetwork = ">=0.12.2, <0.19.0"
quickcheck = "0.9"

[features]
default = ["32-column-tables", "with-deprecated"]
default = ["with-deprecated", "32-column-tables"]
extras = ["chrono", "serde_json", "uuid", "network-address", "numeric", "r2d2"]
unstable = ["diesel_derives/nightly"]
large-tables = ["32-column-tables"]
Expand All @@ -59,8 +59,8 @@ without-deprecated = []
with-deprecated = []
network-address = ["ipnetwork", "libc"]
numeric = ["num-bigint", "bigdecimal", "num-traits", "num-integer"]
postgres_backend = ["diesel_derives/postgres", "bitflags"]
mysql_backend = ["diesel_derives/mysql"]
postgres_backend = ["diesel_derives/postgres", "bitflags", "byteorder"]
mysql_backend = ["diesel_derives/mysql", "byteorder"]

[package.metadata.docs.rs]
features = ["postgres", "mysql", "sqlite", "extras"]
Expand Down
34 changes: 15 additions & 19 deletions diesel/src/backend.rs
Original file line number Diff line number Diff line change
@@ -1,10 +1,6 @@
//! Types which represent various database backends

use byteorder::ByteOrder;

use crate::query_builder::bind_collector::BindCollector;
use crate::query_builder::QueryBuilder;
use crate::sql_types::{self, HasSqlType};
use crate::sql_types::{self, HasSqlType, TypeMetadata};

/// A database backend
///
Expand Down Expand Up @@ -33,19 +29,24 @@ where
Self: HasSqlType<sql_types::Time>,
Self: HasSqlType<sql_types::Timestamp>,
Self: for<'a> HasRawValue<'a>,
Self: for<'a> HasBindCollector<'a>,
{
/// The concrete `QueryBuilder` implementation for this backend.
type QueryBuilder: QueryBuilder<Self>;
}

/// The bind collector type used to collect query binds for this backend
///
/// This trait is separate from `Backend` to imitate `type BindCollector<'a>`. It
/// should only be referenced directly by implementors. Users of this type
/// should instead use the [`BindCollector`] helper type instead.
pub trait HasBindCollector<'a>: TypeMetadata + Sized {
/// The concrete `BindCollector` implementation for this backend.
///
/// Most backends should use [`RawBytesBindCollector`].
///
/// [`RawBytesBindCollector`]: crate::query_builder::bind_collector::RawBytesBindCollector
type BindCollector: BindCollector<Self>;
/// What byte order is used to transmit integers?
///
/// This type is only used if `RawValue` is `[u8]`.
type ByteOrder: ByteOrder;
type BindCollector: crate::query_builder::bind_collector::BindCollector<'a, Self> + 'a;
}

/// The raw representation of a database value given to `FromSql`.
Expand All @@ -60,19 +61,14 @@ pub trait HasRawValue<'a> {
type RawValue;
}

/// A trait indicating that the provided raw value uses a binary representation internally
// That's a false positive, `HasRawValue<'a>` is essentially
// a reference wrapper
#[allow(clippy::wrong_self_convention)]
pub trait BinaryRawValue<'a>: HasRawValue<'a> {
/// Get the underlying binary representation of the raw value
fn as_bytes(value: Self::RawValue) -> &'a [u8];
}

/// A helper type to get the raw representation of a database type given to
/// `FromSql`. Equivalent to `<DB as Backend>::RawValue<'a>`.
pub type RawValue<'a, DB> = <DB as HasRawValue<'a>>::RawValue;

/// A helper type to get the bind collector for a database backend.
/// Equivalent to `<DB as HasBindCollector<'a>>::BindCollector<'a>`j
pub type BindCollector<'a, DB> = <DB as HasBindCollector<'a>>::BindCollector;

/// This trait provides various options to configure the
/// generated SQL for a specific backend.
///
Expand Down
14 changes: 7 additions & 7 deletions diesel/src/connection/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -29,20 +29,20 @@ pub trait SimpleConnection {
/// implementation. This trait is only useful in combination with [`Connection`].
///
/// Implementation wise this is a workaround for GAT's
pub trait ConnectionGatWorkaround<'a, DB: Backend> {
pub trait ConnectionGatWorkaround<'conn, 'query, DB: Backend> {
/// The cursor type returned by [`Connection::load`]
///
/// Users should handle this as opaque type that implements [`Iterator`]
type Cursor: Iterator<Item = QueryResult<Self::Row>>;
/// The row type used as [`Iterator::Item`] for the iterator implementation
/// of [`ConnectionGatWorkaround::Cursor`]
type Row: crate::row::Row<'a, DB>;
type Row: crate::row::Row<'conn, DB>;
}

/// A connection to a database
pub trait Connection: SimpleConnection + Sized + Send
where
Self: for<'a> ConnectionGatWorkaround<'a, <Self as Connection>::Backend>,
Self: for<'a, 'b> ConnectionGatWorkaround<'a, 'b, <Self as Connection>::Backend>,
{
/// The backend this type connects to
type Backend: Backend;
Expand Down Expand Up @@ -192,13 +192,13 @@ where
fn execute(&mut self, query: &str) -> QueryResult<usize>;

#[doc(hidden)]
fn load<T>(
&mut self,
fn load<'conn, 'query, T>(
&'conn mut self,
source: T,
) -> QueryResult<<Self as ConnectionGatWorkaround<Self::Backend>>::Cursor>
) -> QueryResult<<Self as ConnectionGatWorkaround<'conn, 'query, Self::Backend>>::Cursor>
where
T: AsQuery,
T::Query: QueryFragment<Self::Backend> + QueryId,
T::Query: QueryFragment<Self::Backend> + QueryId + 'query,
Self::Backend: QueryMetadata<T::SqlType>;

#[doc(hidden)]
Expand Down
4 changes: 2 additions & 2 deletions diesel/src/connection/statement_cache.rs
Original file line number Diff line number Diff line change
Expand Up @@ -136,7 +136,7 @@ where
source: &T,
bind_types: &[DB::TypeMetadata],
prepare_fn: F,
) -> QueryResult<MaybeCached<Statement>>
) -> QueryResult<MaybeCached<'_, Statement>>
where
T: QueryFragment<DB> + QueryId,
F: FnOnce(&str, PrepareForCache) -> QueryResult<Statement>,
Expand Down Expand Up @@ -226,7 +226,7 @@ where
}
}

pub fn sql<T: QueryFragment<DB>>(&self, source: &T) -> QueryResult<Cow<str>> {
pub fn sql<T: QueryFragment<DB>>(&self, source: &T) -> QueryResult<Cow<'_, str>> {
match *self {
StatementCacheKey::Type(_) => Self::construct_sql(source).map(Cow::Owned),
StatementCacheKey::Sql { ref sql, .. } => Ok(Cow::Borrowed(sql)),
Expand Down
4 changes: 2 additions & 2 deletions diesel/src/deserialize.rs
Original file line number Diff line number Diff line change
Expand Up @@ -411,7 +411,7 @@ pub use diesel_derives::QueryableByName;
/// ```
pub trait FromSql<A, DB: Backend>: Sized {
/// See the trait documentation.
fn from_sql(bytes: backend::RawValue<DB>) -> Result<Self>;
fn from_sql(bytes: backend::RawValue<'_, DB>) -> Result<Self>;

/// A specialized variant of `from_sql` for handling null values.
///
Expand All @@ -421,7 +421,7 @@ pub trait FromSql<A, DB: Backend>: Sized {
/// If your custom type supports null values you need to provide a
/// custom implementation.
#[inline(always)]
fn from_nullable_sql(bytes: Option<backend::RawValue<DB>>) -> Result<Self> {
fn from_nullable_sql(bytes: Option<backend::RawValue<'_, DB>>) -> Result<Self> {
match bytes {
Some(bytes) => Self::from_sql(bytes),
None => Err(Box::new(crate::result::UnexpectedNullError)),
Expand Down
37 changes: 28 additions & 9 deletions diesel/src/expression/array_comparison.rs
Original file line number Diff line number Diff line change
@@ -1,12 +1,12 @@
use crate::backend::sql_dialect;
use crate::backend::Backend;
use crate::backend::SqlDialect;
use crate::expression::bound::Bound;
use crate::expression::subselect::Subselect;
use crate::expression::*;
use crate::query_builder::*;
use crate::query_builder::{BoxedSelectStatement, SelectStatement};
use crate::result::QueryResult;
use crate::serialize::ToSql;
use crate::sql_types::Bool;
use std::marker::PhantomData;

Expand Down Expand Up @@ -55,7 +55,10 @@ where
DB: Backend,
Self: QueryFragment<DB, DB::ArrayComparision>,
{
fn walk_ast(&self, pass: AstPass<DB>) -> QueryResult<()> {
fn walk_ast<'a, 'b>(&'a self, pass: AstPass<'_, 'b, DB>) -> QueryResult<()>
where
'a: 'b,
{
<Self as QueryFragment<DB, DB::ArrayComparision>>::walk_ast(self, pass)
}
}
Expand All @@ -68,7 +71,10 @@ where
T: QueryFragment<DB>,
U: QueryFragment<DB> + MaybeEmpty,
{
fn walk_ast(&self, mut out: AstPass<DB>) -> QueryResult<()> {
fn walk_ast<'a, 'b>(&'a self, mut out: AstPass<'_, 'b, DB>) -> QueryResult<()>
where
'a: 'b,
{
if self.values.is_empty() {
out.push_sql("1=0");
} else {
Expand All @@ -86,7 +92,10 @@ where
DB: Backend,
Self: QueryFragment<DB, DB::ArrayComparision>,
{
fn walk_ast(&self, pass: AstPass<DB>) -> QueryResult<()> {
fn walk_ast<'a, 'b>(&'a self, pass: AstPass<'_, 'b, DB>) -> QueryResult<()>
where
'a: 'b,
{
<Self as QueryFragment<DB, DB::ArrayComparision>>::walk_ast(self, pass)
}
}
Expand All @@ -99,7 +108,10 @@ where
T: QueryFragment<DB>,
U: QueryFragment<DB> + MaybeEmpty,
{
fn walk_ast(&self, mut out: AstPass<DB>) -> QueryResult<()> {
fn walk_ast<'a, 'b>(&'a self, mut out: AstPass<'_, 'b, DB>) -> QueryResult<()>
where
'a: 'b,
{
if self.values.is_empty() {
out.push_sql("1=1");
} else {
Expand Down Expand Up @@ -215,7 +227,10 @@ where
Self: QueryFragment<DB, DB::ArrayComparision>,
DB: Backend,
{
fn walk_ast(&self, pass: AstPass<DB>) -> QueryResult<()> {
fn walk_ast<'a, 'b>(&'a self, pass: AstPass<'_, 'b, DB>) -> QueryResult<()>
where
'a: 'b,
{
<Self as QueryFragment<DB, DB::ArrayComparision>>::walk_ast(self, pass)
}
}
Expand All @@ -224,11 +239,15 @@ impl<ST, I, DB> QueryFragment<DB, sql_dialect::array_comparision::AnsiSqlArrayCo
for Many<ST, I>
where
DB: Backend
+ HasSqlType<ST>
+ SqlDialect<ArrayComparision = sql_dialect::array_comparision::AnsiSqlArrayComparison>,
ST: SingleValue,
for<'a> Bound<ST, &'a I>: QueryFragment<DB>,
I: ToSql<ST, DB>,
{
fn walk_ast(&self, mut out: AstPass<DB>) -> QueryResult<()> {
fn walk_ast<'a, 'b>(&'a self, mut out: AstPass<'_, 'b, DB>) -> QueryResult<()>
where
'a: 'b,
{
out.unsafe_to_cache_prepared();
let mut first = true;
for value in &self.0 {
Expand All @@ -237,7 +256,7 @@ where
} else {
out.push_sql(", ");
}
Bound::new(value).walk_ast(out.reborrow())?;
out.push_bind_param(value)?;
}
Ok(())
}
Expand Down
6 changes: 4 additions & 2 deletions diesel/src/expression/assume_not_null.rs
Original file line number Diff line number Diff line change
@@ -1,7 +1,6 @@
use crate::backend::Backend;
use crate::expression::TypedExpressionType;
use crate::expression::*;
use crate::query_builder::select_statement::NoFromClause;
use crate::query_builder::*;
use crate::query_source::joins::ToInnerJoin;
use crate::result::QueryResult;
Expand Down Expand Up @@ -30,7 +29,10 @@ where
DB: Backend,
T: QueryFragment<DB>,
{
fn walk_ast(&self, pass: AstPass<DB>) -> QueryResult<()> {
fn walk_ast<'a, 'b>(&'a self, pass: AstPass<'_, 'b, DB>) -> QueryResult<()>
where
'a: 'b,
{
self.0.walk_ast(pass)
}
}
Expand Down
7 changes: 5 additions & 2 deletions diesel/src/expression/bound.rs
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@ pub struct Bound<T, U> {
impl<T, U> Bound<T, U> {
pub fn new(item: U) -> Self {
Bound {
item: item,
item,
_marker: PhantomData,
}
}
Expand All @@ -34,7 +34,10 @@ where
DB: Backend + HasSqlType<T>,
U: ToSql<T, DB>,
{
fn walk_ast(&self, mut pass: AstPass<DB>) -> QueryResult<()> {
fn walk_ast<'a, 'b>(&'a self, mut pass: AstPass<'_, 'b, DB>) -> QueryResult<()>
where
'a: 'b,
{
pass.push_bind_param(&self.item)?;
Ok(())
}
Expand Down
5 changes: 4 additions & 1 deletion diesel/src/expression/coerce.rs
Original file line number Diff line number Diff line change
Expand Up @@ -60,7 +60,10 @@ where
T: QueryFragment<DB>,
DB: Backend,
{
fn walk_ast(&self, pass: AstPass<DB>) -> QueryResult<()> {
fn walk_ast<'a, 'b>(&'a self, pass: AstPass<'_, 'b, DB>) -> QueryResult<()>
where
'a: 'b,
{
self.expr.walk_ast(pass)
}
}
Expand Down
14 changes: 10 additions & 4 deletions diesel/src/expression/count.rs
Original file line number Diff line number Diff line change
Expand Up @@ -68,7 +68,10 @@ impl Expression for CountStar {
}

impl<DB: Backend> QueryFragment<DB> for CountStar {
fn walk_ast(&self, mut out: AstPass<DB>) -> QueryResult<()> {
fn walk_ast<'a, 'b>(&'a self, mut out: AstPass<'_, 'b, DB>) -> QueryResult<()>
where
'a: 'b,
{
out.push_sql("COUNT(*)");
Ok(())
}
Expand Down Expand Up @@ -147,11 +150,14 @@ impl<T, E, DB> QueryFragment<DB> for CountDistinct<T, E>
where
T: SqlType + SingleValue,
DB: Backend,
for<'a> &'a E: QueryFragment<DB>,
E: QueryFragment<DB>,
{
fn walk_ast(&self, mut out: AstPass<DB>) -> QueryResult<()> {
fn walk_ast<'a, 'b>(&'a self, mut out: AstPass<'_, 'b, DB>) -> QueryResult<()>
where
'a: 'b,
{
out.push_sql("COUNT(DISTINCT ");
(&self.expr).walk_ast(out.reborrow())?;
self.expr.walk_ast(out.reborrow())?;
out.push_sql(")");
Ok(())
}
Expand Down
10 changes: 8 additions & 2 deletions diesel/src/expression/exists.rs
Original file line number Diff line number Diff line change
Expand Up @@ -55,7 +55,10 @@ where
DB: Backend,
Self: QueryFragment<DB, DB::ExistsSyntax>,
{
fn walk_ast(&self, pass: AstPass<DB>) -> QueryResult<()> {
fn walk_ast<'a, 'b>(&'a self, pass: AstPass<'_, 'b, DB>) -> QueryResult<()>
where
'a: 'b,
{
<Self as QueryFragment<DB, DB::ExistsSyntax>>::walk_ast(self, pass)
}
}
Expand All @@ -65,7 +68,10 @@ where
DB: Backend + SqlDialect<ExistsSyntax = sql_dialect::exists_syntax::AnsiSqlExistsSyntax>,
T: QueryFragment<DB>,
{
fn walk_ast(&self, mut out: AstPass<DB>) -> QueryResult<()> {
fn walk_ast<'a, 'b>(&'a self, mut out: AstPass<'_, 'b, DB>) -> QueryResult<()>
where
'a: 'b,
{
out.push_sql("EXISTS (");
self.0.walk_ast(out.reborrow())?;
out.push_sql(")");
Expand Down
Loading