251 lines
6.4 KiB
Rust
Raw Normal View History

2015-06-14 11:33:55 -04:00
//! Interface to [SQLite][1].
//!
//! ## Example
//!
2015-08-03 16:12:09 -04:00
//! Create a table, insert a couple of rows, and fetch one:
//!
2015-06-14 11:33:55 -04:00
//! ```
2015-07-04 08:53:26 -04:00
//! let connection = sqlite::open(":memory:").unwrap();
2015-06-14 11:33:55 -04:00
//!
2015-07-04 09:29:46 -04:00
//! connection.execute("
2015-08-03 16:12:09 -04:00
//! CREATE TABLE users (name TEXT, age INTEGER);
//! INSERT INTO users (name, age) VALUES ('Alice', 42);
//! INSERT INTO users (name, age) VALUES ('Bob', 69);
2015-07-04 09:29:46 -04:00
//! ").unwrap();
2015-06-14 11:33:55 -04:00
//!
2015-08-03 16:12:09 -04:00
//! connection.process("SELECT * FROM users WHERE age > 50", |pairs| {
2015-06-14 11:33:55 -04:00
//! for &(column, value) in pairs.iter() {
//! println!("{} = {}", column, value.unwrap());
//! }
//! true
//! }).unwrap();
2015-07-30 08:12:43 -04:00
//! ```
//!
//! The same example using prepared statements:
//!
//! ```
2015-08-03 16:52:51 -04:00
//! use sqlite::State;
//!
2015-07-30 08:12:43 -04:00
//! let connection = sqlite::open(":memory:").unwrap();
//!
//! connection.execute("
2015-08-03 16:12:09 -04:00
//! CREATE TABLE users (name TEXT, age INTEGER)
//! ").unwrap();
//!
//! let mut statement = connection.prepare("
//! INSERT INTO users (name, age) VALUES (?, ?)
//! ").unwrap();
//!
//! statement.bind(1, "Alice").unwrap();
//! statement.bind(2, 42).unwrap();
//! assert_eq!(statement.next().unwrap(), State::Done);
2015-08-03 16:12:09 -04:00
//!
//! statement.reset().unwrap();
//!
//! statement.bind(1, "Bob").unwrap();
//! statement.bind(2, 69).unwrap();
//! assert_eq!(statement.next().unwrap(), State::Done);
2015-07-30 08:12:43 -04:00
//!
//! let mut statement = connection.prepare("
2015-08-03 16:12:09 -04:00
//! SELECT * FROM users WHERE age > 50
2015-07-30 08:12:43 -04:00
//! ").unwrap();
2015-07-04 09:27:13 -04:00
//!
//! while let State::Row = statement.next().unwrap() {
2015-08-03 16:52:51 -04:00
//! println!("name = {}", statement.read::<String>(0).unwrap());
//! println!("age = {}", statement.read::<i64>(1).unwrap());
//! }
//! ```
//!
2015-08-03 17:10:30 -04:00
//! The same example using cursors:
2015-08-03 16:52:51 -04:00
//!
//! ```
//! use sqlite::Value;
//!
//! let connection = sqlite::open(":memory:").unwrap();
//!
//! connection.execute("
//! CREATE TABLE users (name TEXT, age INTEGER)
//! ").unwrap();
//!
2015-08-03 17:10:30 -04:00
//! let mut cursor = connection.prepare("
2015-08-03 16:52:51 -04:00
//! INSERT INTO users (name, age) VALUES (?, ?)
2015-08-03 17:10:30 -04:00
//! ").unwrap().cursor().unwrap();
2015-08-03 16:52:51 -04:00
//!
2015-08-03 17:10:30 -04:00
//! cursor.bind(&[
2015-08-03 16:52:51 -04:00
//! Value::String("Alice".to_string()), Value::Integer(42),
//! ]).unwrap();
//!
2015-08-03 17:10:30 -04:00
//! cursor.bind(&[
2015-08-03 16:52:51 -04:00
//! Value::String("Bob".to_string()), Value::Integer(69),
//! ]).unwrap();
//!
2015-08-03 17:10:30 -04:00
//! let mut cursor = connection.prepare("
2015-08-03 16:52:51 -04:00
//! SELECT * FROM users WHERE age > 50
2015-08-03 17:10:30 -04:00
//! ").unwrap().cursor().unwrap();
2015-08-03 16:52:51 -04:00
//!
2015-08-03 17:10:30 -04:00
//! while let Some(row) = cursor.next().unwrap() {
2015-08-03 16:52:51 -04:00
//! match (&row[0], &row[1]) {
//! (&Value::String(ref name), &Value::Integer(age)) => {
//! println!("name = {}", name);
//! println!("age = {}", age);
//! },
//! _ => unreachable!(),
//! }
2015-07-04 09:27:13 -04:00
//! }
2015-06-14 11:33:55 -04:00
//! ```
//!
//! [1]: https://www.sqlite.org
2015-05-28 17:21:43 -04:00
extern crate libc;
2015-06-12 14:23:18 -04:00
extern crate sqlite3_sys as ffi;
2015-05-28 17:21:43 -04:00
2015-08-01 17:38:08 -04:00
use std::{error, fmt};
2015-05-29 09:44:06 -04:00
macro_rules! raise(
2015-08-01 17:38:08 -04:00
($message:expr) => (return Err(::Error { code: None, message: Some($message.to_string()) }));
2015-05-29 09:44:06 -04:00
);
2015-05-28 21:30:02 -04:00
2015-06-19 20:30:28 -04:00
macro_rules! error(
2015-08-01 17:38:08 -04:00
($connection:expr, $code:expr) => (match ::last_error($connection) {
2015-07-27 11:38:54 -04:00
Some(error) => return Err(error),
2015-08-01 17:38:08 -04:00
_ => return Err(::Error { code: Some($code as isize), message: None }),
2015-07-27 11:38:54 -04:00
});
2015-06-08 17:43:31 -04:00
);
2015-06-19 20:30:28 -04:00
macro_rules! ok(
2015-07-27 11:38:54 -04:00
($connection:expr, $result:expr) => (match $result {
::ffi::SQLITE_OK => {},
code => error!($connection, code),
});
($result:expr) => (match $result {
::ffi::SQLITE_OK => {},
2015-08-01 17:38:08 -04:00
code => return Err(::Error { code: Some(code as isize), message: None }),
2015-05-29 09:44:06 -04:00
});
);
2015-05-28 21:30:02 -04:00
2015-06-19 20:23:12 -04:00
macro_rules! c_str_to_str(
($string:expr) => (::std::str::from_utf8(::std::ffi::CStr::from_ptr($string).to_bytes()));
);
macro_rules! c_str_to_string(
($string:expr) => (
String::from_utf8_lossy(::std::ffi::CStr::from_ptr($string as *const _).to_bytes())
.into_owned()
);
);
2015-07-27 11:38:54 -04:00
macro_rules! path_to_cstr(
($path:expr) => (match $path.to_str() {
Some(path) => match ::std::ffi::CString::new(path) {
Ok(string) => string,
_ => raise!("failed to process a path"),
},
_ => raise!("failed to process a path"),
});
);
macro_rules! str_to_cstr(
($string:expr) => (match ::std::ffi::CString::new($string) {
Ok(string) => string,
_ => raise!("failed to process a string"),
});
2015-05-29 09:44:06 -04:00
);
2015-05-28 21:30:02 -04:00
2015-08-01 17:38:08 -04:00
/// An error.
#[derive(Debug)]
pub struct Error {
/// The error code.
pub code: Option<isize>,
/// The error message.
pub message: Option<String>,
}
/// A result.
pub type Result<T> = std::result::Result<T, Error>;
/// A data type.
#[derive(Clone, Copy, Debug, Eq, PartialEq)]
pub enum Type {
/// The binary type.
2015-08-01 17:51:43 -04:00
Binary,
/// The floating-point type.
Float,
/// The integer type.
Integer,
/// The string type.
String,
/// The null type.
Null,
}
/// A typed value.
#[derive(Clone, Debug, PartialEq)]
pub enum Value {
/// Binary data.
Binary(Vec<u8>),
/// A floating-point number.
Float(f64),
/// An integer.
Integer(i64),
/// A string.
String(String),
/// A null value.
Null,
}
2015-08-01 17:38:08 -04:00
impl fmt::Display for Error {
fn fmt(&self, formatter: &mut fmt::Formatter) -> fmt::Result {
match (self.code, &self.message) {
(Some(code), &Some(ref message)) => write!(formatter, "{} (code {})", message, code),
(Some(code), _) => write!(formatter, "an SQLite error (code {})", code),
(_, &Some(ref message)) => message.fmt(formatter),
_ => write!(formatter, "an SQLite error"),
}
}
}
impl error::Error for Error {
fn description(&self) -> &str {
match self.message {
Some(ref message) => message,
_ => "an SQLite error",
}
}
}
2015-07-04 08:53:26 -04:00
mod connection;
2015-08-03 17:10:30 -04:00
mod cursor;
2015-05-29 11:24:01 -04:00
mod statement;
2015-05-28 19:19:08 -04:00
2015-07-04 08:53:26 -04:00
pub use connection::Connection;
2015-08-03 17:10:30 -04:00
pub use cursor::Cursor;
pub use statement::{Statement, State, Bindable, Readable};
2015-05-28 19:19:08 -04:00
2015-06-12 14:21:29 -04:00
/// Open a connection to a new or existing database.
2015-05-28 19:19:08 -04:00
#[inline]
pub fn open<T: AsRef<std::path::Path>>(path: T) -> Result<Connection> {
2015-07-04 08:53:26 -04:00
Connection::open(path)
2015-05-28 17:21:43 -04:00
}
2015-08-01 17:38:08 -04:00
2015-08-01 21:38:54 -04:00
/// Return the version number of SQLite.
///
/// For instance, the version `3.8.11.1` corresponds to the integer `3008011`.
#[inline]
pub fn version() -> usize {
unsafe { ffi::sqlite3_libversion_number() as usize }
}
2015-08-01 17:38:08 -04:00
fn last_error(raw: *mut ffi::sqlite3) -> Option<Error> {
unsafe {
let code = ffi::sqlite3_errcode(raw);
if code == ffi::SQLITE_OK {
return None;
}
let message = ffi::sqlite3_errmsg(raw);
if message.is_null() {
return None;
}
Some(Error { code: Some(code as isize), message: Some(c_str_to_string!(message)) })
}
}