/* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ use rusqlite::{ self, types::{FromSql, ToSql}, Connection, Result as SqlResult, Row, Savepoint, Transaction, TransactionBehavior, NO_PARAMS, }; use std::iter::FromIterator; use std::ops::Deref; use std::time::Instant; use crate::maybe_cached::MaybeCached; pub struct Conn(rusqlite::Connection); /// This trait exists so that we can use these helpers on `rusqlite::{Transaction, Connection}`. /// Note that you must import ConnExt in order to call these methods on anything. pub trait ConnExt { /// The method you need to implement to opt in to all of this. fn conn(&self) -> &Connection; /// Set the value of the pragma on the main database. Returns the same object, for chaining. fn set_pragma(&self, pragma_name: &str, pragma_value: T) -> SqlResult<&Self> where T: ToSql, Self: Sized, { // None == Schema name, e.g. `PRAGMA some_attached_db.something = blah` self.conn() .pragma_update(None, pragma_name, &pragma_value)?; Ok(self) } /// Get a cached or uncached statement based on a flag. fn prepare_maybe_cached<'conn>( &'conn self, sql: &str, cache: bool, ) -> SqlResult> { MaybeCached::prepare(self.conn(), sql, cache) } /// Execute all the provided statements. fn execute_all(&self, stmts: &[&str]) -> SqlResult<()> { let conn = self.conn(); for sql in stmts { let r = conn.execute(sql, NO_PARAMS); match r { Ok(_) => {} // Ignore ExecuteReturnedResults error because they're pointless // and annoying. Err(rusqlite::Error::ExecuteReturnedResults) => {} Err(e) => return Err(e), } } Ok(()) } /// Equivalent to `Connection::execute_named` but caches the statement so that subsequent /// calls to `execute_cached` will have improved performance. fn execute_cached

(&self, sql: &str, params: P) -> SqlResult where P: IntoIterator, P::Item: ToSql, { let mut stmt = self.conn().prepare_cached(sql)?; stmt.execute(params) } /// Equivalent to `Connection::execute_named` but caches the statement so that subsequent /// calls to `execute_named_cached` will have imprroved performance. fn execute_named_cached(&self, sql: &str, params: &[(&str, &dyn ToSql)]) -> SqlResult { crate::maybe_log_plan(self.conn(), sql, params); let mut stmt = self.conn().prepare_cached(sql)?; stmt.execute_named(params) } /// Execute a query that returns a single result column, and return that result. fn query_one(&self, sql: &str) -> SqlResult { crate::maybe_log_plan(self.conn(), sql, &[]); let res: T = self .conn() .query_row_and_then(sql, NO_PARAMS, |row| row.get(0))?; Ok(res) } /// Execute a query that returns 0 or 1 result columns, returning None /// if there were no rows, or if the only result was NULL. fn try_query_one( &self, sql: &str, params: &[(&str, &dyn ToSql)], cache: bool, ) -> SqlResult> where Self: Sized, { crate::maybe_log_plan(self.conn(), sql, params); use rusqlite::OptionalExtension; // The outer option is if we got rows, the inner option is // if the first row was null. let res: Option> = self .conn() .query_row_and_then_named(sql, params, |row| row.get(0), cache) .optional()?; // go from Option> to Option Ok(res.unwrap_or_default()) } /// Equivalent to `rusqlite::Connection::query_row_and_then` but allows use /// of named parameters, and allows passing a flag to indicate that it's cached. fn query_row_and_then_named( &self, sql: &str, params: &[(&str, &dyn ToSql)], mapper: F, cache: bool, ) -> Result where Self: Sized, E: From, F: FnOnce(&Row<'_>) -> Result, { crate::maybe_log_plan(self.conn(), sql, params); Ok(self .try_query_row(sql, params, mapper, cache)? .ok_or(rusqlite::Error::QueryReturnedNoRows)?) } /// Helper for when you'd like to get a Vec of all the rows returned by a /// query that takes named arguments. See also /// `query_rows_and_then_named_cached`. fn query_rows_and_then_named( &self, sql: &str, params: &[(&str, &dyn ToSql)], mapper: F, ) -> Result, E> where Self: Sized, E: From, F: FnMut(&Row<'_>) -> Result, { crate::maybe_log_plan(self.conn(), sql, params); query_rows_and_then_named(self.conn(), sql, params, mapper, false) } /// Helper for when you'd like to get a Vec of all the rows returned by a /// query that takes named arguments. fn query_rows_and_then_named_cached( &self, sql: &str, params: &[(&str, &dyn ToSql)], mapper: F, ) -> Result, E> where Self: Sized, E: From, F: FnMut(&Row<'_>) -> Result, { crate::maybe_log_plan(self.conn(), sql, params); query_rows_and_then_named(self.conn(), sql, params, mapper, true) } /// Like `query_rows_and_then_named`, but works if you want a non-Vec as a result. /// # Example: /// ```rust,no_run /// # use std::collections::HashSet; /// # use sql_support::ConnExt; /// # use rusqlite::Connection; /// fn get_visit_tombstones(conn: &Connection, id: i64) -> rusqlite::Result> { /// Ok(conn.query_rows_into( /// "SELECT visit_date FROM moz_historyvisit_tombstones /// WHERE place_id = :place_id", /// &[(":place_id", &id)], /// |row| row.get::<_, i64>(0))?) /// } /// ``` /// Note if the type isn't inferred, you'll have to do something gross like /// `conn.query_rows_into::, _, _, _>(...)`. fn query_rows_into( &self, sql: &str, params: &[(&str, &dyn ToSql)], mapper: F, ) -> Result where Self: Sized, E: From, F: FnMut(&Row<'_>) -> Result, Coll: FromIterator, { crate::maybe_log_plan(self.conn(), sql, params); query_rows_and_then_named(self.conn(), sql, params, mapper, false) } /// Same as `query_rows_into`, but caches the stmt if possible. fn query_rows_into_cached( &self, sql: &str, params: &[(&str, &dyn ToSql)], mapper: F, ) -> Result where Self: Sized, E: From, F: FnMut(&Row<'_>) -> Result, Coll: FromIterator, { crate::maybe_log_plan(self.conn(), sql, params); query_rows_and_then_named(self.conn(), sql, params, mapper, true) } // This should probably have a longer name... /// Like `query_row_and_then_named` but returns None instead of erroring if no such row exists. fn try_query_row( &self, sql: &str, params: &[(&str, &dyn ToSql)], mapper: F, cache: bool, ) -> Result, E> where Self: Sized, E: From, F: FnOnce(&Row<'_>) -> Result, { crate::maybe_log_plan(self.conn(), sql, params); let conn = self.conn(); let mut stmt = MaybeCached::prepare(conn, sql, cache)?; let mut rows = stmt.query_named(params)?; rows.next()?.map(mapper).transpose() } fn unchecked_transaction(&self) -> SqlResult> { UncheckedTransaction::new(self.conn(), TransactionBehavior::Deferred) } /// Begin `unchecked_transaction` with `TransactionBehavior::Immediate`. Use /// when the first operation will be a read operation, that further writes /// depend on for correctness. fn unchecked_transaction_imm(&self) -> SqlResult> { UncheckedTransaction::new(self.conn(), TransactionBehavior::Immediate) } } impl ConnExt for Connection { #[inline] fn conn(&self) -> &Connection { self } } impl<'conn> ConnExt for Transaction<'conn> { #[inline] fn conn(&self) -> &Connection { &*self } } impl<'conn> ConnExt for Savepoint<'conn> { #[inline] fn conn(&self) -> &Connection { &*self } } /// rusqlite, in an attempt to save us from ourselves, needs a mutable ref to /// a connection to start a transaction. That is a bit of a PITA in some cases, /// so we offer this as an alternative - but the responsibility of ensuring /// there are no concurrent transactions is on our head. /// /// This is very similar to the rusqlite `Transaction` - it doesn't prevent /// against nested transactions but does allow you to use an immutable /// `Connection`. pub struct UncheckedTransaction<'conn> { pub conn: &'conn Connection, pub started_at: Instant, pub finished: bool, // we could add drop_behavior etc too, but we don't need it yet - we // always rollback. } impl<'conn> UncheckedTransaction<'conn> { /// Begin a new unchecked transaction. Cannot be nested, but this is not /// enforced by Rust (hence 'unchecked') - however, it is enforced by /// SQLite; use a rusqlite `savepoint` for nested transactions. pub fn new(conn: &'conn Connection, behavior: TransactionBehavior) -> SqlResult { let query = match behavior { TransactionBehavior::Deferred => "BEGIN DEFERRED", TransactionBehavior::Immediate => "BEGIN IMMEDIATE", TransactionBehavior::Exclusive => "BEGIN EXCLUSIVE", _ => unreachable!(), }; conn.execute_batch(query) .map(move |_| UncheckedTransaction { conn, started_at: Instant::now(), finished: false, }) } /// Consumes and commits an unchecked transaction. pub fn commit(mut self) -> SqlResult<()> { if self.finished { log::warn!("ignoring request to commit an already finished transaction"); return Ok(()); } self.finished = true; self.conn.execute_batch("COMMIT")?; log::debug!("Transaction commited after {:?}", self.started_at.elapsed()); Ok(()) } /// Consumes and rolls back an unchecked transaction. pub fn rollback(mut self) -> SqlResult<()> { if self.finished { log::warn!("ignoring request to rollback an already finished transaction"); return Ok(()); } self.rollback_() } fn rollback_(&mut self) -> SqlResult<()> { self.finished = true; self.conn.execute_batch("ROLLBACK")?; Ok(()) } fn finish_(&mut self) -> SqlResult<()> { if self.finished || self.conn.is_autocommit() { return Ok(()); } self.rollback_()?; Ok(()) } } impl<'conn> Deref for UncheckedTransaction<'conn> { type Target = Connection; #[inline] fn deref(&self) -> &Connection { self.conn } } impl<'conn> Drop for UncheckedTransaction<'conn> { fn drop(&mut self) { if let Err(e) = self.finish_() { log::warn!("Error dropping an unchecked transaction: {}", e); } } } impl<'conn> ConnExt for UncheckedTransaction<'conn> { #[inline] fn conn(&self) -> &Connection { &*self } } fn query_rows_and_then_named( conn: &Connection, sql: &str, params: &[(&str, &dyn ToSql)], mapper: F, cache: bool, ) -> Result where E: From, F: FnMut(&Row<'_>) -> Result, Coll: FromIterator, { let mut stmt = conn.prepare_maybe_cached(sql, cache)?; let iter = stmt.query_and_then_named(params, mapper)?; Ok(iter.collect::>()?) }