|
|
|
use libc::{c_uint, size_t};
|
|
|
|
use std::{fmt, ptr, result, mem};
|
|
|
|
use std::ffi::CString;
|
|
|
|
#[cfg(unix)]
|
|
|
|
use std::os::unix::ffi::OsStrExt;
|
|
|
|
#[cfg(windows)]
|
|
|
|
use std::ffi::OsStr;
|
|
|
|
use std::path::Path;
|
|
|
|
use std::sync::Mutex;
|
|
|
|
|
|
|
|
use ffi;
|
|
|
|
|
|
|
|
use error::{Result, lmdb_result};
|
|
|
|
use database::Database;
|
|
|
|
use transaction::{RoTransaction, RwTransaction, Transaction};
|
|
|
|
use flags::{DatabaseFlags, EnvironmentFlags};
|
|
|
|
|
|
|
|
#[cfg(windows)]
|
|
|
|
/// Adding a 'missing' trait from windows OsStrExt
|
|
|
|
trait OsStrExtLmdb {
|
|
|
|
fn as_bytes(&self) -> &[u8];
|
|
|
|
}
|
|
|
|
#[cfg(windows)]
|
|
|
|
impl OsStrExtLmdb for OsStr {
|
|
|
|
fn as_bytes(&self) -> &[u8] {
|
|
|
|
&self.to_str().unwrap().as_bytes()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// An LMDB environment.
|
|
|
|
///
|
|
|
|
/// An environment supports multiple databases, all residing in the same shared-memory map.
|
|
|
|
pub struct Environment {
|
|
|
|
env: *mut ffi::MDB_env,
|
|
|
|
dbi_open_mutex: Mutex<()>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Environment {
|
|
|
|
|
|
|
|
/// Creates a new builder for specifying options for opening an LMDB environment.
|
|
|
|
pub fn new() -> EnvironmentBuilder {
|
|
|
|
EnvironmentBuilder {
|
|
|
|
flags: EnvironmentFlags::empty(),
|
|
|
|
max_readers: None,
|
|
|
|
max_dbs: None,
|
|
|
|
map_size: None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns a raw pointer to the underlying LMDB environment.
|
|
|
|
///
|
|
|
|
/// The caller **must** ensure that the pointer is not dereferenced after the lifetime of the
|
|
|
|
/// environment.
|
|
|
|
pub fn env(&self) -> *mut ffi::MDB_env {
|
|
|
|
self.env
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Opens a handle to an LMDB database.
|
|
|
|
///
|
|
|
|
/// If `name` is `None`, then the returned handle will be for the default database.
|
|
|
|
///
|
|
|
|
/// If `name` is not `None`, then the returned handle will be for a named database. In this
|
|
|
|
/// case the environment must be configured to allow named databases through
|
|
|
|
/// `EnvironmentBuilder::set_max_dbs`.
|
|
|
|
///
|
|
|
|
/// The returned database handle may be shared among any transaction in the environment.
|
|
|
|
///
|
|
|
|
/// This function will fail with `Error::BadRslot` if called by a thread which has an ongoing
|
|
|
|
/// transaction.
|
|
|
|
///
|
|
|
|
/// The database name may not contain the null character.
|
|
|
|
pub fn open_db<'env>(&'env self, name: Option<&str>) -> Result<Database> {
|
|
|
|
let mutex = self.dbi_open_mutex.lock();
|
|
|
|
let txn = self.begin_ro_txn()?;
|
|
|
|
let db = unsafe { txn.open_db(name)? };
|
|
|
|
txn.commit()?;
|
|
|
|
drop(mutex);
|
|
|
|
Ok(db)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Opens a handle to an LMDB database, creating the database if necessary.
|
|
|
|
///
|
|
|
|
/// If the database is already created, the given option flags will be added to it.
|
|
|
|
///
|
|
|
|
/// If `name` is `None`, then the returned handle will be for the default database.
|
|
|
|
///
|
|
|
|
/// If `name` is not `None`, then the returned handle will be for a named database. In this
|
|
|
|
/// case the environment must be configured to allow named databases through
|
|
|
|
/// `EnvironmentBuilder::set_max_dbs`.
|
|
|
|
///
|
|
|
|
/// The returned database handle may be shared among any transaction in the environment.
|
|
|
|
///
|
|
|
|
/// This function will fail with `Error::BadRslot` if called by a thread with an open
|
|
|
|
/// transaction.
|
|
|
|
pub fn create_db<'env>(&'env self,
|
|
|
|
name: Option<&str>,
|
|
|
|
flags: DatabaseFlags)
|
|
|
|
-> Result<Database> {
|
|
|
|
let mutex = self.dbi_open_mutex.lock();
|
|
|
|
let txn = self.begin_rw_txn()?;
|
|
|
|
let db = unsafe { txn.create_db(name, flags)? };
|
|
|
|
txn.commit()?;
|
|
|
|
drop(mutex);
|
|
|
|
Ok(db)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Retrieves the set of flags which the database is opened with.
|
|
|
|
///
|
|
|
|
/// The database must belong to to this environment.
|
|
|
|
pub fn get_db_flags<'env>(&'env self, db: Database) -> Result<DatabaseFlags> {
|
|
|
|
let txn = self.begin_ro_txn()?;
|
|
|
|
let mut flags: c_uint = 0;
|
|
|
|
unsafe {
|
|
|
|
lmdb_result(ffi::mdb_dbi_flags(txn.txn(), db.dbi(), &mut flags))?;
|
|
|
|
}
|
|
|
|
Ok(DatabaseFlags::from_bits(flags).unwrap())
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Create a read-only transaction for use with the environment.
|
|
|
|
pub fn begin_ro_txn<'env>(&'env self) -> Result<RoTransaction<'env>> {
|
|
|
|
RoTransaction::new(self)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Create a read-write transaction for use with the environment. This method will block while
|
|
|
|
/// there are any other read-write transactions open on the environment.
|
|
|
|
pub fn begin_rw_txn<'env>(&'env self) -> Result<RwTransaction<'env>> {
|
|
|
|
RwTransaction::new(self)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Flush data buffers to disk.
|
|
|
|
///
|
|
|
|
/// Data is always written to disk when `Transaction::commit` is called, but the operating
|
|
|
|
/// system may keep it buffered. LMDB always flushes the OS buffers upon commit as well, unless
|
|
|
|
/// the environment was opened with `MDB_NOSYNC` or in part `MDB_NOMETASYNC`.
|
|
|
|
pub fn sync(&self, force: bool) -> Result<()> {
|
|
|
|
unsafe {
|
|
|
|
lmdb_result(ffi::mdb_env_sync(self.env(), if force { 1 } else { 0 }))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Closes the database handle. Normally unnecessary.
|
|
|
|
///
|
|
|
|
/// Closing a database handle is not necessary, but lets `Transaction::open_database` reuse the
|
|
|
|
/// handle value. Usually it's better to set a bigger `EnvironmentBuilder::set_max_dbs`, unless
|
|
|
|
/// that value would be large.
|
|
|
|
///
|
|
|
|
/// ## Safety
|
|
|
|
///
|
|
|
|
/// This call is not mutex protected. Databases should only be closed by a single thread, and
|
|
|
|
/// only if no other threads are going to reference the database handle or one of its cursors
|
|
|
|
/// any further. Do not close a handle if an existing transaction has modified its database.
|
|
|
|
/// Doing so can cause misbehavior from database corruption to errors like
|
|
|
|
/// `Error::BadValSize` (since the DB name is gone).
|
|
|
|
pub unsafe fn close_db(&mut self, db: Database) {
|
|
|
|
ffi::mdb_dbi_close(self.env, db.dbi());
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Retrieves statistics about this environment.
|
|
|
|
pub fn stat(&self) -> Result<Stat> {
|
|
|
|
unsafe {
|
|
|
|
let mut stat = Stat(mem::zeroed());
|
|
|
|
lmdb_try!(ffi::mdb_env_stat(self.env(), &mut stat.0));
|
|
|
|
Ok(stat)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Environment statistics.
|
|
|
|
///
|
|
|
|
/// Contains information about the size and layout of an LMDB environment.
|
|
|
|
pub struct Stat(ffi::MDB_stat);
|
|
|
|
|
|
|
|
impl Stat {
|
|
|
|
/// Size of a database page. This is the same for all databases in the environment.
|
|
|
|
#[inline]
|
|
|
|
pub fn page_size(&self) -> u32 {
|
|
|
|
self.0.ms_psize
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Depth (height) of the B-tree.
|
|
|
|
#[inline]
|
|
|
|
pub fn depth(&self) -> u32 {
|
|
|
|
self.0.ms_depth
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Number of internal (non-leaf) pages.
|
|
|
|
#[inline]
|
|
|
|
pub fn branch_pages(&self) -> usize {
|
|
|
|
self.0.ms_branch_pages
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Number of leaf pages.
|
|
|
|
#[inline]
|
|
|
|
pub fn leaf_pages(&self) -> usize {
|
|
|
|
self.0.ms_leaf_pages
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Number of overflow pages.
|
|
|
|
#[inline]
|
|
|
|
pub fn overflow_pages(&self) -> usize {
|
|
|
|
self.0.ms_overflow_pages
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Number of data items.
|
|
|
|
#[inline]
|
|
|
|
pub fn entries(&self) -> usize {
|
|
|
|
self.0.ms_entries
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe impl Send for Environment {}
|
|
|
|
unsafe impl Sync for Environment {}
|
|
|
|
|
|
|
|
impl fmt::Debug for Environment {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> result::Result<(), fmt::Error> {
|
|
|
|
f.debug_struct("Environment").finish()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Drop for Environment {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
unsafe { ffi::mdb_env_close(self.env) }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
//// Environment Builder
|
|
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
/// Options for opening or creating an environment.
|
|
|
|
#[derive(Debug, PartialEq, Eq, Copy, Clone)]
|
|
|
|
pub struct EnvironmentBuilder {
|
|
|
|
flags: EnvironmentFlags,
|
|
|
|
max_readers: Option<c_uint>,
|
|
|
|
max_dbs: Option<c_uint>,
|
|
|
|
map_size: Option<size_t>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl EnvironmentBuilder {
|
|
|
|
|
|
|
|
/// Open an environment.
|
|
|
|
///
|
|
|
|
/// On UNIX, the database files will be opened with 644 permissions.
|
|
|
|
///
|
|
|
|
/// The path may not contain the null character.
|
|
|
|
pub fn open(&self, path: &Path) -> Result<Environment> {
|
|
|
|
self.open_with_permissions(path, 0o644)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Open an environment with the provided UNIX permissions.
|
|
|
|
///
|
|
|
|
/// On Windows, the permissions will be ignored.
|
|
|
|
///
|
|
|
|
/// The path may not contain the null character.
|
|
|
|
pub fn open_with_permissions(&self, path: &Path, mode: ffi::mode_t) -> Result<Environment> {
|
|
|
|
let mut env: *mut ffi::MDB_env = ptr::null_mut();
|
|
|
|
unsafe {
|
|
|
|
lmdb_try!(ffi::mdb_env_create(&mut env));
|
|
|
|
if let Some(max_readers) = self.max_readers {
|
|
|
|
lmdb_try_with_cleanup!(ffi::mdb_env_set_maxreaders(env, max_readers),
|
|
|
|
ffi::mdb_env_close(env))
|
|
|
|
}
|
|
|
|
if let Some(max_dbs) = self.max_dbs {
|
|
|
|
lmdb_try_with_cleanup!(ffi::mdb_env_set_maxdbs(env, max_dbs),
|
|
|
|
ffi::mdb_env_close(env))
|
|
|
|
}
|
|
|
|
if let Some(map_size) = self.map_size {
|
|
|
|
lmdb_try_with_cleanup!(ffi::mdb_env_set_mapsize(env, map_size),
|
|
|
|
ffi::mdb_env_close(env))
|
|
|
|
}
|
|
|
|
let path = match CString::new(path.as_os_str().as_bytes()) {
|
|
|
|
Ok(path) => path,
|
|
|
|
Err(..) => return Err(::Error::Invalid),
|
|
|
|
};
|
|
|
|
lmdb_try_with_cleanup!(ffi::mdb_env_open(env, path.as_ptr(), self.flags.bits(), mode),
|
|
|
|
ffi::mdb_env_close(env));
|
|
|
|
}
|
|
|
|
Ok(Environment { env: env, dbi_open_mutex: Mutex::new(()) })
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Sets the provided options in the environment.
|
|
|
|
pub fn set_flags(&mut self, flags: EnvironmentFlags) -> &mut EnvironmentBuilder {
|
|
|
|
self.flags = flags;
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Sets the maximum number of threads or reader slots for the environment.
|
|
|
|
///
|
|
|
|
/// This defines the number of slots in the lock table that is used to track readers in the
|
|
|
|
/// the environment. The default is 126. Starting a read-only transaction normally ties a lock
|
|
|
|
/// table slot to the current thread until the environment closes or the thread exits. If
|
|
|
|
/// `MDB_NOTLS` is in use, `Environment::open_txn` instead ties the slot to the `Transaction`
|
|
|
|
/// object until it or the `Environment` object is destroyed.
|
|
|
|
pub fn set_max_readers(&mut self, max_readers: c_uint) -> &mut EnvironmentBuilder {
|
|
|
|
self.max_readers = Some(max_readers);
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Sets the maximum number of named databases for the environment.
|
|
|
|
///
|
|
|
|
/// This function is only needed if multiple databases will be used in the
|
|
|
|
/// environment. Simpler applications that use the environment as a single
|
|
|
|
/// unnamed database can ignore this option.
|
|
|
|
///
|
|
|
|
/// Currently a moderate number of slots are cheap but a huge number gets
|
|
|
|
/// expensive: 7-120 words per transaction, and every `Transaction::open_db`
|
|
|
|
/// does a linear search of the opened slots.
|
|
|
|
pub fn set_max_dbs(&mut self, max_readers: c_uint) -> &mut EnvironmentBuilder {
|
|
|
|
self.max_dbs = Some(max_readers);
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Sets the size of the memory map to use for the environment.
|
|
|
|
///
|
|
|
|
/// The size should be a multiple of the OS page size. The default is
|
|
|
|
/// 10485760 bytes. The size of the memory map is also the maximum size
|
|
|
|
/// of the database. The value should be chosen as large as possible,
|
|
|
|
/// to accommodate future growth of the database. It may be increased at
|
|
|
|
/// later times.
|
|
|
|
///
|
|
|
|
/// Any attempt to set a size smaller than the space already consumed
|
|
|
|
/// by the environment will be silently changed to the current size of the used space.
|
|
|
|
pub fn set_map_size(&mut self, map_size: size_t) -> &mut EnvironmentBuilder {
|
|
|
|
self.map_size = Some(map_size);
|
|
|
|
self
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod test {
|
|
|
|
|
|
|
|
use tempdir::TempDir;
|
|
|
|
|
|
|
|
use flags::*;
|
|
|
|
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_open() {
|
|
|
|
let dir = TempDir::new("test").unwrap();
|
|
|
|
|
|
|
|
// opening non-existent env with read-only should fail
|
|
|
|
assert!(Environment::new().set_flags(EnvironmentFlags::READ_ONLY)
|
|
|
|
.open(dir.path())
|
|
|
|
.is_err());
|
|
|
|
|
|
|
|
// opening non-existent env should succeed
|
|
|
|
assert!(Environment::new().open(dir.path()).is_ok());
|
|
|
|
|
|
|
|
// opening env with read-only should succeed
|
|
|
|
assert!(Environment::new().set_flags(EnvironmentFlags::READ_ONLY)
|
|
|
|
.open(dir.path())
|
|
|
|
.is_ok());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_begin_txn() {
|
|
|
|
let dir = TempDir::new("test").unwrap();
|
|
|
|
|
|
|
|
{ // writable environment
|
|
|
|
let env = Environment::new().open(dir.path()).unwrap();
|
|
|
|
|
|
|
|
assert!(env.begin_rw_txn().is_ok());
|
|
|
|
assert!(env.begin_ro_txn().is_ok());
|
|
|
|
}
|
|
|
|
|
|
|
|
{ // read-only environment
|
|
|
|
let env = Environment::new().set_flags(EnvironmentFlags::READ_ONLY)
|
|
|
|
.open(dir.path())
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
assert!(env.begin_rw_txn().is_err());
|
|
|
|
assert!(env.begin_ro_txn().is_ok());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_open_db() {
|
|
|
|
let dir = TempDir::new("test").unwrap();
|
|
|
|
let env = Environment::new().set_max_dbs(1)
|
|
|
|
.open(dir.path())
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
assert!(env.open_db(None).is_ok());
|
|
|
|
assert!(env.open_db(Some("testdb")).is_err());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_create_db() {
|
|
|
|
let dir = TempDir::new("test").unwrap();
|
|
|
|
let env = Environment::new().set_max_dbs(11)
|
|
|
|
.open(dir.path())
|
|
|
|
.unwrap();
|
|
|
|
assert!(env.open_db(Some("testdb")).is_err());
|
|
|
|
assert!(env.create_db(Some("testdb"), DatabaseFlags::empty()).is_ok());
|
|
|
|
assert!(env.open_db(Some("testdb")).is_ok())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_close_database() {
|
|
|
|
let dir = TempDir::new("test").unwrap();
|
|
|
|
let mut env = Environment::new().set_max_dbs(10)
|
|
|
|
.open(dir.path())
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
let db = env.create_db(Some("db"), DatabaseFlags::empty()).unwrap();
|
|
|
|
unsafe { env.close_db(db); }
|
|
|
|
assert!(env.open_db(Some("db")).is_ok());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_sync() {
|
|
|
|
let dir = TempDir::new("test").unwrap();
|
|
|
|
{
|
|
|
|
let env = Environment::new().open(dir.path()).unwrap();
|
|
|
|
assert!(env.sync(true).is_ok());
|
|
|
|
} {
|
|
|
|
let env = Environment::new().set_flags(EnvironmentFlags::READ_ONLY)
|
|
|
|
.open(dir.path())
|
|
|
|
.unwrap();
|
|
|
|
assert!(env.sync(true).is_err());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|