2015-05-29 09:44:06 -04:00
|
|
|
use libc::{c_char, c_int, c_void};
|
|
|
|
use raw;
|
|
|
|
use std::marker::PhantomData;
|
|
|
|
use std::path::Path;
|
|
|
|
|
2015-05-29 11:24:01 -04:00
|
|
|
use {Result, Statement};
|
2015-05-29 09:44:06 -04:00
|
|
|
|
|
|
|
/// A database.
|
2015-06-08 14:52:13 -04:00
|
|
|
pub struct Database<'l> {
|
2015-05-29 11:24:01 -04:00
|
|
|
raw: *mut raw::sqlite3,
|
2015-06-08 14:52:13 -04:00
|
|
|
busy_callback: Option<Box<FnMut(usize) -> bool + 'l>>,
|
2015-06-01 11:39:22 -04:00
|
|
|
phantom: PhantomData<raw::sqlite3>,
|
2015-05-29 09:44:06 -04:00
|
|
|
}
|
|
|
|
|
2015-06-08 14:52:13 -04:00
|
|
|
impl<'l> Database<'l> {
|
2015-06-08 14:58:05 -04:00
|
|
|
/// Open a database.
|
2015-06-08 14:52:13 -04:00
|
|
|
pub fn open(path: &Path) -> Result<Database<'l>> {
|
2015-05-29 11:24:01 -04:00
|
|
|
let mut raw = 0 as *mut _;
|
2015-05-29 09:44:06 -04:00
|
|
|
unsafe {
|
2015-05-29 11:24:01 -04:00
|
|
|
success!(raw::sqlite3_open(path_to_c_str!(path), &mut raw));
|
2015-05-29 09:44:06 -04:00
|
|
|
}
|
2015-06-08 14:52:13 -04:00
|
|
|
Ok(Database {
|
|
|
|
raw: raw,
|
|
|
|
busy_callback: None,
|
|
|
|
phantom: PhantomData,
|
|
|
|
})
|
2015-05-29 09:44:06 -04:00
|
|
|
}
|
|
|
|
|
2015-06-08 13:27:07 -04:00
|
|
|
/// Execute a query without processing the resulting rows if any.
|
|
|
|
#[inline]
|
2015-06-08 15:00:44 -04:00
|
|
|
pub fn execute(&self, sql: &str) -> Result<()> {
|
2015-06-08 13:27:07 -04:00
|
|
|
unsafe {
|
|
|
|
success!(self, raw::sqlite3_exec(self.raw, str_to_c_str!(sql), None, 0 as *mut _,
|
|
|
|
0 as *mut _));
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Execute a query and process the resulting rows if any.
|
|
|
|
///
|
|
|
|
/// The callback is triggered for each row. If the callback returns `false`,
|
|
|
|
/// no more rows will be processed.
|
|
|
|
#[inline]
|
2015-06-08 15:00:44 -04:00
|
|
|
pub fn process<F>(&self, sql: &str, callback: F) -> Result<()>
|
2015-06-08 13:27:07 -04:00
|
|
|
where F: FnMut(Vec<(String, String)>) -> bool
|
|
|
|
{
|
|
|
|
unsafe {
|
2015-06-08 14:52:13 -04:00
|
|
|
let callback = Box::new(callback);
|
2015-06-08 13:27:07 -04:00
|
|
|
success!(self, raw::sqlite3_exec(self.raw, str_to_c_str!(sql),
|
|
|
|
Some(execute_callback::<F>),
|
2015-06-08 14:52:13 -04:00
|
|
|
&*callback as *const _ as *mut _ as *mut _,
|
2015-06-08 13:27:07 -04:00
|
|
|
0 as *mut _));
|
2015-05-29 09:44:06 -04:00
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
2015-05-29 11:24:01 -04:00
|
|
|
|
2015-06-08 13:27:07 -04:00
|
|
|
/// Create a prepared statement.
|
2015-05-29 16:58:48 -04:00
|
|
|
#[inline]
|
2015-06-08 14:52:13 -04:00
|
|
|
pub fn prepare(&'l self, sql: &str) -> Result<Statement<'l>> {
|
2015-05-29 16:58:48 -04:00
|
|
|
::statement::new(self, sql)
|
2015-05-29 11:24:01 -04:00
|
|
|
}
|
2015-06-07 22:15:59 -04:00
|
|
|
|
2015-06-08 13:27:07 -04:00
|
|
|
/// Set a callback for handling busy events.
|
2015-06-08 11:28:43 -04:00
|
|
|
///
|
|
|
|
/// The callback is triggered when the database cannot perform an operation
|
2015-06-08 13:27:07 -04:00
|
|
|
/// due to processing of some other request. If the callback returns `true`,
|
|
|
|
/// the operation will be repeated.
|
|
|
|
pub fn set_busy_handler<F>(&mut self, callback: F) -> Result<()>
|
2015-06-08 14:52:13 -04:00
|
|
|
where F: FnMut(usize) -> bool + 'l
|
2015-06-08 13:27:07 -04:00
|
|
|
{
|
2015-06-08 14:52:13 -04:00
|
|
|
try!(self.remove_busy_handler());
|
2015-06-08 13:27:07 -04:00
|
|
|
unsafe {
|
2015-06-08 14:52:13 -04:00
|
|
|
let callback = Box::new(callback);
|
|
|
|
let result = raw::sqlite3_busy_handler(self.raw, Some(busy_callback::<F>),
|
|
|
|
&*callback as *const _ as *mut _ as *mut _);
|
|
|
|
self.busy_callback = Some(callback);
|
|
|
|
success!(result);
|
2015-06-07 22:15:59 -04:00
|
|
|
}
|
|
|
|
Ok(())
|
2015-06-08 07:53:28 -04:00
|
|
|
}
|
|
|
|
|
2015-06-08 13:27:07 -04:00
|
|
|
/// Set an implicit callback for handling busy events that tries to repeat
|
|
|
|
/// rejected operations until a timeout expires.
|
2015-06-08 07:53:28 -04:00
|
|
|
#[inline]
|
|
|
|
pub fn set_busy_timeout(&mut self, milliseconds: usize) -> Result<()> {
|
|
|
|
unsafe { success!(self, raw::sqlite3_busy_timeout(self.raw, milliseconds as c_int)) };
|
|
|
|
Ok(())
|
2015-06-07 22:15:59 -04:00
|
|
|
}
|
2015-06-08 13:27:07 -04:00
|
|
|
|
|
|
|
/// Remove the callback handling busy events.
|
|
|
|
#[inline]
|
|
|
|
pub fn remove_busy_handler(&mut self) -> Result<()> {
|
2015-06-08 14:52:13 -04:00
|
|
|
::std::mem::replace(&mut self.busy_callback, None);
|
|
|
|
unsafe { success!(raw::sqlite3_busy_handler(self.raw, None, 0 as *mut _)) };
|
2015-06-08 13:27:07 -04:00
|
|
|
Ok(())
|
|
|
|
}
|
2015-05-29 09:44:06 -04:00
|
|
|
}
|
|
|
|
|
2015-06-08 14:52:13 -04:00
|
|
|
impl<'l> Drop for Database<'l> {
|
2015-05-29 09:44:06 -04:00
|
|
|
#[inline]
|
2015-06-08 14:52:13 -04:00
|
|
|
#[allow(unused_must_use)]
|
2015-05-29 09:44:06 -04:00
|
|
|
fn drop(&mut self) {
|
2015-06-08 14:52:13 -04:00
|
|
|
self.remove_busy_handler();
|
2015-05-29 14:15:01 -04:00
|
|
|
unsafe { raw::sqlite3_close(self.raw) };
|
2015-05-29 09:44:06 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-29 14:15:01 -04:00
|
|
|
#[inline]
|
2015-06-01 16:27:09 -04:00
|
|
|
pub fn as_raw(database: &Database) -> *mut raw::sqlite3 {
|
2015-05-29 14:15:01 -04:00
|
|
|
database.raw
|
|
|
|
}
|
|
|
|
|
2015-06-08 13:27:07 -04:00
|
|
|
extern fn busy_callback<F>(callback: *mut c_void, attempts: c_int) -> c_int
|
|
|
|
where F: FnMut(usize) -> bool
|
|
|
|
{
|
2015-06-08 14:52:13 -04:00
|
|
|
unsafe { if (*(callback as *mut F))(attempts as usize) { 1 } else { 0 } }
|
2015-06-07 22:15:59 -04:00
|
|
|
}
|
|
|
|
|
2015-06-08 13:27:07 -04:00
|
|
|
extern fn execute_callback<F>(callback: *mut c_void, count: c_int, values: *mut *mut c_char,
|
|
|
|
columns: *mut *mut c_char) -> c_int
|
|
|
|
where F: FnMut(Vec<(String, String)>) -> bool
|
|
|
|
{
|
2015-05-29 09:44:06 -04:00
|
|
|
unsafe {
|
|
|
|
let mut pairs = Vec::with_capacity(count as usize);
|
|
|
|
|
|
|
|
for i in 0..(count as isize) {
|
2015-05-29 13:08:02 -04:00
|
|
|
let column = c_str_to_string!(*columns.offset(i));
|
|
|
|
let value = c_str_to_string!(*values.offset(i));
|
2015-05-29 09:44:06 -04:00
|
|
|
pairs.push((column, value));
|
|
|
|
}
|
|
|
|
|
2015-06-08 13:27:07 -04:00
|
|
|
if (*(callback as *mut F))(pairs) { 0 } else { 1 }
|
2015-05-29 09:44:06 -04:00
|
|
|
}
|
|
|
|
}
|
2015-06-08 09:37:44 -04:00
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use super::Database;
|
|
|
|
use tests::setup;
|
|
|
|
|
|
|
|
macro_rules! ok(
|
|
|
|
($result:expr) => ($result.unwrap());
|
|
|
|
);
|
|
|
|
|
|
|
|
#[test]
|
2015-06-08 15:00:44 -04:00
|
|
|
fn execute() {
|
2015-06-08 09:37:44 -04:00
|
|
|
let (path, _directory) = setup();
|
|
|
|
let database = ok!(Database::open(&path));
|
2015-06-08 15:00:44 -04:00
|
|
|
match database.execute(":)") {
|
2015-06-08 13:27:07 -04:00
|
|
|
Err(error) => assert_eq!(error.message,
|
|
|
|
Some(String::from(r#"unrecognized token: ":""#))),
|
2015-06-08 09:37:44 -04:00
|
|
|
_ => assert!(false),
|
|
|
|
}
|
|
|
|
}
|
2015-06-08 09:43:13 -04:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn set_busy_handler() {
|
|
|
|
let (path, _directory) = setup();
|
|
|
|
let mut database = ok!(Database::open(&path));
|
2015-06-08 13:27:07 -04:00
|
|
|
ok!(database.set_busy_handler(|_| true));
|
2015-06-08 09:43:13 -04:00
|
|
|
}
|
2015-06-08 09:37:44 -04:00
|
|
|
}
|