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-05-29 11:24:01 -04:00
|
|
|
pub struct Database<'l> {
|
|
|
|
raw: *mut raw::sqlite3,
|
|
|
|
_phantom: PhantomData<&'l raw::sqlite3>,
|
2015-05-29 09:44:06 -04:00
|
|
|
}
|
|
|
|
|
2015-05-29 13:34:48 -04:00
|
|
|
/// A callback triggered for each row of an executed SQL query.
|
2015-05-29 11:24:01 -04:00
|
|
|
pub type ExecuteCallback<'l> = FnMut(Vec<(String, String)>) -> bool + 'l;
|
2015-05-29 09:44:06 -04:00
|
|
|
|
2015-05-29 11:24:01 -04:00
|
|
|
impl<'l> Database<'l> {
|
2015-05-29 09:44:06 -04:00
|
|
|
/// Open a database.
|
2015-05-29 11:24:01 -04:00
|
|
|
pub fn open(path: &Path) -> Result<Database> {
|
|
|
|
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-05-29 11:24:01 -04:00
|
|
|
Ok(Database { raw: raw, _phantom: PhantomData })
|
2015-05-29 09:44:06 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Execute an SQL statement.
|
|
|
|
pub fn execute<'c>(&mut self, sql: &str,
|
|
|
|
callback: Option<&mut ExecuteCallback<'c>>) -> Result<()> {
|
|
|
|
|
|
|
|
unsafe {
|
|
|
|
match callback {
|
|
|
|
Some(callback) => {
|
|
|
|
let mut callback = Box::new(callback);
|
2015-05-29 16:41:34 -04:00
|
|
|
success!(self, raw::sqlite3_exec(self.raw, str_to_c_str!(sql),
|
|
|
|
Some(execute_callback),
|
|
|
|
&mut callback as *mut _ as *mut _,
|
|
|
|
0 as *mut _));
|
2015-05-29 09:44:06 -04:00
|
|
|
},
|
|
|
|
None => {
|
2015-05-29 16:41:34 -04:00
|
|
|
success!(self, raw::sqlite3_exec(self.raw, str_to_c_str!(sql), None,
|
|
|
|
0 as *mut _, 0 as *mut _));
|
2015-05-29 09:44:06 -04:00
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
2015-05-29 11:24:01 -04:00
|
|
|
|
|
|
|
/// Create a prepared statement.
|
|
|
|
pub fn statement(&mut self, sql: &str) -> Result<Statement<'l>> {
|
|
|
|
let mut raw = 0 as *mut _;
|
|
|
|
unsafe {
|
2015-05-29 16:41:34 -04:00
|
|
|
success!(self, raw::sqlite3_prepare(self.raw, str_to_c_str!(sql), -1, &mut raw,
|
|
|
|
0 as *mut _));
|
2015-05-29 11:24:01 -04:00
|
|
|
}
|
|
|
|
Ok(::statement::from_raw(raw))
|
|
|
|
}
|
2015-05-29 09:44:06 -04:00
|
|
|
}
|
|
|
|
|
2015-05-29 11:24:01 -04:00
|
|
|
impl<'l> Drop for Database<'l> {
|
2015-05-29 09:44:06 -04:00
|
|
|
#[inline]
|
|
|
|
fn drop(&mut self) {
|
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]
|
|
|
|
pub fn as_raw(database: &mut Database) -> *mut raw::sqlite3 {
|
|
|
|
database.raw
|
|
|
|
}
|
|
|
|
|
2015-05-29 09:44:06 -04:00
|
|
|
extern fn execute_callback(callback: *mut c_void, count: c_int, values: *mut *mut c_char,
|
|
|
|
columns: *mut *mut c_char) -> c_int {
|
|
|
|
|
|
|
|
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));
|
|
|
|
}
|
|
|
|
|
|
|
|
let ref mut callback = *(callback as *mut Box<&mut ExecuteCallback>);
|
|
|
|
if callback(pairs) { 0 } else { 1 }
|
|
|
|
}
|
|
|
|
}
|