2019-10-18 19:05:28 +08:00
|
|
|
use std::collections::hash_map::{Entry, HashMap};
|
2019-10-18 18:52:45 +08:00
|
|
|
use std::fs::File;
|
2019-10-04 22:49:17 +08:00
|
|
|
use std::path::Path;
|
|
|
|
use std::sync::{Arc, RwLock};
|
2019-10-07 22:16:04 +08:00
|
|
|
use std::{fs, thread};
|
|
|
|
|
2019-11-06 17:49:13 +08:00
|
|
|
use crossbeam_channel::{Receiver, Sender};
|
2019-10-21 18:05:53 +08:00
|
|
|
use heed::types::{Str, Unit};
|
|
|
|
use heed::{CompactionOption, Result as ZResult};
|
2019-11-04 20:18:07 +08:00
|
|
|
use log::debug;
|
2019-10-07 22:16:04 +08:00
|
|
|
|
|
|
|
use crate::{store, update, Index, MResult};
|
2019-10-04 22:49:17 +08:00
|
|
|
|
2019-11-16 00:33:06 +08:00
|
|
|
pub type BoxUpdateFn = Box<dyn Fn(&str, update::ProcessedUpdateResult) + Send + Sync + 'static>;
|
2019-10-09 17:45:19 +08:00
|
|
|
type ArcSwapFn = arc_swap::ArcSwapOption<BoxUpdateFn>;
|
|
|
|
|
2019-10-04 22:49:17 +08:00
|
|
|
pub struct Database {
|
2019-10-21 18:05:53 +08:00
|
|
|
pub env: heed::Env,
|
|
|
|
common_store: heed::PolyDatabase,
|
|
|
|
indexes_store: heed::Database<Str, Unit>,
|
2019-11-16 00:33:06 +08:00
|
|
|
indexes: RwLock<HashMap<String, (Index, thread::JoinHandle<()>)>>,
|
|
|
|
update_fn: Arc<ArcSwapFn>,
|
2019-10-07 22:16:04 +08:00
|
|
|
}
|
|
|
|
|
2019-11-04 18:11:58 +08:00
|
|
|
macro_rules! r#break_try {
|
|
|
|
($expr:expr, $msg:tt) => {
|
|
|
|
match $expr {
|
|
|
|
core::result::Result::Ok(val) => val,
|
|
|
|
core::result::Result::Err(err) => {
|
|
|
|
log::error!(concat!($msg, ": {}"), err);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2019-11-06 17:49:13 +08:00
|
|
|
pub enum UpdateEvent {
|
|
|
|
NewUpdate,
|
|
|
|
MustStop,
|
|
|
|
}
|
|
|
|
|
|
|
|
pub type UpdateEvents = Receiver<UpdateEvent>;
|
|
|
|
pub type UpdateEventsEmitter = Sender<UpdateEvent>;
|
|
|
|
|
2019-11-16 00:33:06 +08:00
|
|
|
fn update_awaiter(
|
|
|
|
receiver: UpdateEvents,
|
|
|
|
env: heed::Env,
|
2019-11-19 23:15:49 +08:00
|
|
|
index_uid: &str,
|
2019-11-16 00:33:06 +08:00
|
|
|
update_fn: Arc<ArcSwapFn>,
|
|
|
|
index: Index,
|
|
|
|
) {
|
2019-11-06 17:49:13 +08:00
|
|
|
let mut receiver = receiver.into_iter();
|
|
|
|
while let Some(UpdateEvent::NewUpdate) = receiver.next() {
|
2019-10-07 22:16:04 +08:00
|
|
|
loop {
|
2019-11-04 18:11:58 +08:00
|
|
|
// instantiate a main/parent transaction
|
|
|
|
let mut writer = break_try!(env.write_txn(), "LMDB write transaction begin failed");
|
|
|
|
|
|
|
|
// retrieve the update that needs to be processed
|
|
|
|
let result = index.updates.pop_front(&mut writer);
|
|
|
|
let (update_id, update) = match break_try!(result, "pop front update failed") {
|
|
|
|
Some(value) => value,
|
|
|
|
None => {
|
2019-11-01 00:27:58 +08:00
|
|
|
debug!("no more updates");
|
|
|
|
writer.abort();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-11-04 18:11:58 +08:00
|
|
|
// instantiate a nested transaction
|
|
|
|
let result = env.nested_write_txn(&mut writer);
|
|
|
|
let mut nested_writer = break_try!(result, "LMDB nested write transaction failed");
|
2019-11-01 00:27:58 +08:00
|
|
|
|
2019-11-04 18:11:58 +08:00
|
|
|
// try to apply the update to the database using the nested transaction
|
|
|
|
let result = update::update_task(&mut nested_writer, index.clone(), update_id, update);
|
|
|
|
let status = break_try!(result, "update task failed");
|
|
|
|
|
|
|
|
// commit the nested transaction if the update was successful, abort it otherwise
|
2019-11-12 23:18:53 +08:00
|
|
|
if status.error.is_none() {
|
2019-11-04 18:11:58 +08:00
|
|
|
break_try!(nested_writer.commit(), "commit nested transaction failed");
|
|
|
|
} else {
|
|
|
|
nested_writer.abort()
|
|
|
|
}
|
|
|
|
|
|
|
|
// write the result of the update in the updates-results store
|
|
|
|
let updates_results = index.updates_results;
|
|
|
|
let result = updates_results.put_update_result(&mut writer, update_id, &status);
|
|
|
|
|
|
|
|
// always commit the main/parent transaction, even if the update was unsuccessful
|
|
|
|
break_try!(result, "update result store commit failed");
|
|
|
|
break_try!(writer.commit(), "update parent transaction failed");
|
|
|
|
|
|
|
|
// call the user callback when the update and the result are written consistently
|
|
|
|
if let Some(ref callback) = *update_fn.load() {
|
2019-11-19 23:15:49 +08:00
|
|
|
(callback)(index_uid, status);
|
2019-10-07 22:16:04 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-11-06 17:49:13 +08:00
|
|
|
|
|
|
|
debug!("update loop system stopped");
|
2019-10-04 22:49:17 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Database {
|
|
|
|
pub fn open_or_create(path: impl AsRef<Path>) -> MResult<Database> {
|
2019-10-07 22:16:04 +08:00
|
|
|
fs::create_dir_all(path.as_ref())?;
|
|
|
|
|
2019-10-21 18:05:53 +08:00
|
|
|
let env = heed::EnvOpenOptions::new()
|
2019-10-16 23:05:24 +08:00
|
|
|
.map_size(10 * 1024 * 1024 * 1024) // 10GB
|
|
|
|
.max_dbs(3000)
|
|
|
|
.open(path)?;
|
2019-10-04 22:49:17 +08:00
|
|
|
|
2019-10-31 22:48:29 +08:00
|
|
|
let common_store = env.create_poly_database(Some("common"))?;
|
2019-10-16 23:05:24 +08:00
|
|
|
let indexes_store = env.create_database::<Str, Unit>(Some("indexes"))?;
|
2019-11-16 00:33:06 +08:00
|
|
|
let update_fn = Arc::new(ArcSwapFn::empty());
|
2019-10-04 22:49:17 +08:00
|
|
|
|
|
|
|
// list all indexes that needs to be opened
|
|
|
|
let mut must_open = Vec::new();
|
2019-10-16 23:05:24 +08:00
|
|
|
let reader = env.read_txn()?;
|
|
|
|
for result in indexes_store.iter(&reader)? {
|
2019-11-19 23:15:49 +08:00
|
|
|
let (index_uid, _) = result?;
|
|
|
|
must_open.push(index_uid.to_owned());
|
2019-10-04 22:49:17 +08:00
|
|
|
}
|
|
|
|
|
2019-10-16 23:05:24 +08:00
|
|
|
reader.abort();
|
2019-10-04 22:49:17 +08:00
|
|
|
|
|
|
|
// open the previously aggregated indexes
|
|
|
|
let mut indexes = HashMap::new();
|
2019-11-19 23:15:49 +08:00
|
|
|
for index_uid in must_open {
|
2019-10-07 22:16:04 +08:00
|
|
|
let (sender, receiver) = crossbeam_channel::bounded(100);
|
2019-11-19 23:15:49 +08:00
|
|
|
let index = match store::open(&env, &index_uid, sender.clone())? {
|
2019-10-16 23:05:24 +08:00
|
|
|
Some(index) => index,
|
|
|
|
None => {
|
2019-10-18 19:05:28 +08:00
|
|
|
log::warn!(
|
|
|
|
"the index {} doesn't exist or has not all the databases",
|
2019-11-19 23:15:49 +08:00
|
|
|
index_uid
|
2019-10-18 19:05:28 +08:00
|
|
|
);
|
2019-10-16 23:05:24 +08:00
|
|
|
continue;
|
2019-10-18 19:05:28 +08:00
|
|
|
}
|
2019-10-16 23:05:24 +08:00
|
|
|
};
|
2019-10-09 17:45:19 +08:00
|
|
|
|
2019-10-16 23:05:24 +08:00
|
|
|
let env_clone = env.clone();
|
2019-10-07 22:16:04 +08:00
|
|
|
let index_clone = index.clone();
|
2019-11-19 23:15:49 +08:00
|
|
|
let name_clone = index_uid.clone();
|
2019-10-09 17:45:19 +08:00
|
|
|
let update_fn_clone = update_fn.clone();
|
|
|
|
|
|
|
|
let handle = thread::spawn(move || {
|
2019-11-16 00:33:06 +08:00
|
|
|
update_awaiter(
|
|
|
|
receiver,
|
|
|
|
env_clone,
|
|
|
|
&name_clone,
|
|
|
|
update_fn_clone,
|
|
|
|
index_clone,
|
|
|
|
)
|
2019-10-09 17:45:19 +08:00
|
|
|
});
|
2019-10-07 22:16:04 +08:00
|
|
|
|
|
|
|
// send an update notification to make sure that
|
2019-10-16 23:05:24 +08:00
|
|
|
// possible pre-boot updates are consumed
|
2019-11-06 17:49:13 +08:00
|
|
|
sender.send(UpdateEvent::NewUpdate).unwrap();
|
2019-10-07 22:16:04 +08:00
|
|
|
|
2019-11-19 23:15:49 +08:00
|
|
|
let result = indexes.insert(index_uid, (index, handle));
|
2019-10-18 19:05:28 +08:00
|
|
|
assert!(
|
|
|
|
result.is_none(),
|
|
|
|
"The index should not have been already open"
|
|
|
|
);
|
2019-10-04 22:49:17 +08:00
|
|
|
}
|
|
|
|
|
2019-10-18 19:05:28 +08:00
|
|
|
Ok(Database {
|
|
|
|
env,
|
|
|
|
common_store,
|
|
|
|
indexes_store,
|
|
|
|
indexes: RwLock::new(indexes),
|
2019-11-16 00:33:06 +08:00
|
|
|
update_fn,
|
2019-10-18 19:05:28 +08:00
|
|
|
})
|
2019-10-04 22:49:17 +08:00
|
|
|
}
|
|
|
|
|
2019-10-10 19:38:58 +08:00
|
|
|
pub fn open_index(&self, name: impl AsRef<str>) -> Option<Index> {
|
2019-10-04 22:49:17 +08:00
|
|
|
let indexes_lock = self.indexes.read().unwrap();
|
2019-10-10 19:38:58 +08:00
|
|
|
match indexes_lock.get(name.as_ref()) {
|
|
|
|
Some((index, ..)) => Some(index.clone()),
|
|
|
|
None => None,
|
|
|
|
}
|
|
|
|
}
|
2019-10-04 22:49:17 +08:00
|
|
|
|
2019-10-10 19:38:58 +08:00
|
|
|
pub fn create_index(&self, name: impl AsRef<str>) -> MResult<Index> {
|
|
|
|
let name = name.as_ref();
|
|
|
|
let mut indexes_lock = self.indexes.write().unwrap();
|
2019-10-04 22:49:17 +08:00
|
|
|
|
2019-10-10 19:38:58 +08:00
|
|
|
match indexes_lock.entry(name.to_owned()) {
|
|
|
|
Entry::Occupied(_) => Err(crate::Error::IndexAlreadyExists),
|
|
|
|
Entry::Vacant(entry) => {
|
2019-10-07 22:16:04 +08:00
|
|
|
let (sender, receiver) = crossbeam_channel::bounded(100);
|
2019-10-16 23:05:24 +08:00
|
|
|
let index = store::create(&self.env, name, sender)?;
|
2019-10-04 22:49:17 +08:00
|
|
|
|
2019-10-16 23:05:24 +08:00
|
|
|
let mut writer = self.env.write_txn()?;
|
|
|
|
self.indexes_store.put(&mut writer, name, &())?;
|
2019-10-09 17:45:19 +08:00
|
|
|
|
2019-10-16 23:05:24 +08:00
|
|
|
let env_clone = self.env.clone();
|
2019-10-10 19:38:58 +08:00
|
|
|
let index_clone = index.clone();
|
2019-11-16 00:33:06 +08:00
|
|
|
let name_clone = name.to_owned();
|
|
|
|
let update_fn_clone = self.update_fn.clone();
|
2019-10-09 17:45:19 +08:00
|
|
|
|
2019-10-10 19:38:58 +08:00
|
|
|
let handle = thread::spawn(move || {
|
2019-11-16 00:33:06 +08:00
|
|
|
update_awaiter(
|
|
|
|
receiver,
|
|
|
|
env_clone,
|
|
|
|
&name_clone,
|
|
|
|
update_fn_clone,
|
|
|
|
index_clone,
|
|
|
|
)
|
2019-10-10 19:38:58 +08:00
|
|
|
});
|
2019-10-04 22:49:17 +08:00
|
|
|
|
|
|
|
writer.commit()?;
|
2019-11-16 00:33:06 +08:00
|
|
|
entry.insert((index.clone(), handle));
|
2019-10-04 22:49:17 +08:00
|
|
|
|
|
|
|
Ok(index)
|
2019-10-10 19:38:58 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-06 17:49:13 +08:00
|
|
|
pub fn delete_index(&self, name: impl AsRef<str>) -> MResult<bool> {
|
|
|
|
let name = name.as_ref();
|
|
|
|
let mut indexes_lock = self.indexes.write().unwrap();
|
|
|
|
|
|
|
|
match indexes_lock.remove_entry(name) {
|
2019-11-16 00:33:06 +08:00
|
|
|
Some((name, (index, handle))) => {
|
2019-11-06 17:49:13 +08:00
|
|
|
// remove the index name from the list of indexes
|
|
|
|
// and clear all the LMDB dbi
|
|
|
|
let mut writer = self.env.write_txn()?;
|
|
|
|
self.indexes_store.delete(&mut writer, &name)?;
|
2019-11-20 18:20:50 +08:00
|
|
|
|
2019-11-06 17:49:13 +08:00
|
|
|
store::clear(&mut writer, &index)?;
|
|
|
|
writer.commit()?;
|
|
|
|
|
2019-11-20 18:20:50 +08:00
|
|
|
drop(indexes_lock);
|
|
|
|
|
2019-11-06 17:49:13 +08:00
|
|
|
// join the update loop thread to ensure it is stopped
|
|
|
|
handle.join().unwrap();
|
|
|
|
|
|
|
|
Ok(true)
|
|
|
|
}
|
|
|
|
None => Ok(false),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-16 00:33:06 +08:00
|
|
|
pub fn set_update_callback(&self, update_fn: BoxUpdateFn) {
|
|
|
|
let update_fn = Some(Arc::new(update_fn));
|
|
|
|
self.update_fn.swap(update_fn);
|
2019-10-10 19:38:58 +08:00
|
|
|
}
|
|
|
|
|
2019-11-16 00:33:06 +08:00
|
|
|
pub fn unset_update_callback(&self) {
|
|
|
|
self.update_fn.swap(None);
|
2019-10-04 22:49:17 +08:00
|
|
|
}
|
|
|
|
|
2019-10-18 18:52:45 +08:00
|
|
|
pub fn copy_and_compact_to_path<P: AsRef<Path>>(&self, path: P) -> ZResult<File> {
|
|
|
|
self.env.copy_to_path(path, CompactionOption::Enabled)
|
|
|
|
}
|
|
|
|
|
2019-11-20 18:21:21 +08:00
|
|
|
pub fn indexes_uids(&self) -> Vec<String> {
|
2019-10-04 22:49:17 +08:00
|
|
|
let indexes = self.indexes.read().unwrap();
|
2019-11-20 18:21:21 +08:00
|
|
|
indexes.keys().cloned().collect()
|
2019-10-04 22:49:17 +08:00
|
|
|
}
|
2019-10-07 22:16:04 +08:00
|
|
|
|
2019-10-21 18:05:53 +08:00
|
|
|
pub fn common_store(&self) -> heed::PolyDatabase {
|
2019-10-10 21:14:32 +08:00
|
|
|
self.common_store
|
2019-10-07 22:16:04 +08:00
|
|
|
}
|
2019-10-04 22:49:17 +08:00
|
|
|
}
|
2019-11-04 20:18:07 +08:00
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use super::*;
|
2019-11-05 22:03:18 +08:00
|
|
|
|
2019-11-04 20:18:07 +08:00
|
|
|
use crate::update::{ProcessedUpdateResult, UpdateStatus};
|
2019-11-05 22:03:18 +08:00
|
|
|
use crate::DocumentId;
|
|
|
|
use serde::de::IgnoredAny;
|
2019-11-04 20:18:07 +08:00
|
|
|
use std::sync::mpsc;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn valid_updates() {
|
|
|
|
let dir = tempfile::tempdir().unwrap();
|
|
|
|
|
|
|
|
let database = Database::open_or_create(dir.path()).unwrap();
|
|
|
|
let env = &database.env;
|
|
|
|
|
|
|
|
let (sender, receiver) = mpsc::sync_channel(100);
|
2019-11-16 00:33:06 +08:00
|
|
|
let update_fn = move |_name: &str, update: ProcessedUpdateResult| {
|
|
|
|
sender.send(update.update_id).unwrap()
|
|
|
|
};
|
2019-11-04 20:18:07 +08:00
|
|
|
let index = database.create_index("test").unwrap();
|
|
|
|
|
2019-11-16 00:33:06 +08:00
|
|
|
database.set_update_callback(Box::new(update_fn));
|
2019-11-04 20:18:07 +08:00
|
|
|
|
|
|
|
let schema = {
|
|
|
|
let data = r#"
|
|
|
|
identifier = "id"
|
|
|
|
|
|
|
|
[attributes."name"]
|
|
|
|
displayed = true
|
|
|
|
indexed = true
|
|
|
|
|
|
|
|
[attributes."description"]
|
|
|
|
displayed = true
|
|
|
|
indexed = true
|
|
|
|
"#;
|
|
|
|
toml::from_str(data).unwrap()
|
|
|
|
};
|
|
|
|
|
|
|
|
let mut writer = env.write_txn().unwrap();
|
|
|
|
let _update_id = index.schema_update(&mut writer, schema).unwrap();
|
|
|
|
writer.commit().unwrap();
|
|
|
|
|
|
|
|
let mut additions = index.documents_addition();
|
|
|
|
|
|
|
|
let doc1 = serde_json::json!({
|
|
|
|
"id": 123,
|
|
|
|
"name": "Marvin",
|
|
|
|
"description": "My name is Marvin",
|
|
|
|
});
|
|
|
|
|
|
|
|
let doc2 = serde_json::json!({
|
|
|
|
"id": 234,
|
|
|
|
"name": "Kevin",
|
|
|
|
"description": "My name is Kevin",
|
|
|
|
});
|
|
|
|
|
|
|
|
additions.update_document(doc1);
|
|
|
|
additions.update_document(doc2);
|
|
|
|
|
|
|
|
let mut writer = env.write_txn().unwrap();
|
|
|
|
let update_id = additions.finalize(&mut writer).unwrap();
|
|
|
|
writer.commit().unwrap();
|
|
|
|
|
|
|
|
// block until the transaction is processed
|
|
|
|
let _ = receiver.into_iter().find(|id| *id == update_id);
|
|
|
|
|
|
|
|
let reader = env.read_txn().unwrap();
|
|
|
|
let result = index.update_status(&reader, update_id).unwrap();
|
2019-11-13 01:00:47 +08:00
|
|
|
assert_matches!(result, Some(UpdateStatus::Processed { content }) if content.error.is_none());
|
2019-11-04 20:18:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn invalid_updates() {
|
|
|
|
let dir = tempfile::tempdir().unwrap();
|
|
|
|
|
|
|
|
let database = Database::open_or_create(dir.path()).unwrap();
|
|
|
|
let env = &database.env;
|
|
|
|
|
|
|
|
let (sender, receiver) = mpsc::sync_channel(100);
|
2019-11-16 00:33:06 +08:00
|
|
|
let update_fn = move |_name: &str, update: ProcessedUpdateResult| {
|
|
|
|
sender.send(update.update_id).unwrap()
|
|
|
|
};
|
2019-11-04 20:18:07 +08:00
|
|
|
let index = database.create_index("test").unwrap();
|
|
|
|
|
2019-11-16 00:33:06 +08:00
|
|
|
database.set_update_callback(Box::new(update_fn));
|
2019-11-04 20:18:07 +08:00
|
|
|
|
|
|
|
let schema = {
|
|
|
|
let data = r#"
|
|
|
|
identifier = "id"
|
|
|
|
|
|
|
|
[attributes."name"]
|
|
|
|
displayed = true
|
|
|
|
indexed = true
|
|
|
|
|
|
|
|
[attributes."description"]
|
|
|
|
displayed = true
|
|
|
|
indexed = true
|
|
|
|
"#;
|
|
|
|
toml::from_str(data).unwrap()
|
|
|
|
};
|
|
|
|
|
|
|
|
let mut writer = env.write_txn().unwrap();
|
|
|
|
let _update_id = index.schema_update(&mut writer, schema).unwrap();
|
|
|
|
writer.commit().unwrap();
|
|
|
|
|
|
|
|
let mut additions = index.documents_addition();
|
|
|
|
|
|
|
|
let doc1 = serde_json::json!({
|
|
|
|
"id": 123,
|
|
|
|
"name": "Marvin",
|
|
|
|
"description": "My name is Marvin",
|
|
|
|
});
|
|
|
|
|
|
|
|
let doc2 = serde_json::json!({
|
|
|
|
"name": "Kevin",
|
|
|
|
"description": "My name is Kevin",
|
|
|
|
});
|
|
|
|
|
|
|
|
additions.update_document(doc1);
|
|
|
|
additions.update_document(doc2);
|
|
|
|
|
|
|
|
let mut writer = env.write_txn().unwrap();
|
|
|
|
let update_id = additions.finalize(&mut writer).unwrap();
|
|
|
|
writer.commit().unwrap();
|
|
|
|
|
|
|
|
// block until the transaction is processed
|
|
|
|
let _ = receiver.into_iter().find(|id| *id == update_id);
|
|
|
|
|
|
|
|
let reader = env.read_txn().unwrap();
|
|
|
|
let result = index.update_status(&reader, update_id).unwrap();
|
2019-11-13 01:00:47 +08:00
|
|
|
assert_matches!(result, Some(UpdateStatus::Processed { content }) if content.error.is_some());
|
2019-11-04 20:18:07 +08:00
|
|
|
}
|
2019-11-05 19:03:37 +08:00
|
|
|
|
2019-11-11 00:41:32 +08:00
|
|
|
#[test]
|
2019-11-12 23:18:53 +08:00
|
|
|
fn ignored_words_too_long() {
|
2019-11-11 00:41:32 +08:00
|
|
|
let dir = tempfile::tempdir().unwrap();
|
|
|
|
|
|
|
|
let database = Database::open_or_create(dir.path()).unwrap();
|
|
|
|
let env = &database.env;
|
|
|
|
|
|
|
|
let (sender, receiver) = mpsc::sync_channel(100);
|
2019-11-16 00:33:06 +08:00
|
|
|
let update_fn = move |_name: &str, update: ProcessedUpdateResult| {
|
|
|
|
sender.send(update.update_id).unwrap()
|
|
|
|
};
|
2019-11-11 00:41:32 +08:00
|
|
|
let index = database.create_index("test").unwrap();
|
|
|
|
|
2019-11-16 00:33:06 +08:00
|
|
|
database.set_update_callback(Box::new(update_fn));
|
2019-11-11 00:41:32 +08:00
|
|
|
|
|
|
|
let schema = {
|
|
|
|
let data = r#"
|
|
|
|
identifier = "id"
|
|
|
|
|
|
|
|
[attributes."name"]
|
|
|
|
displayed = true
|
|
|
|
indexed = true
|
|
|
|
"#;
|
|
|
|
toml::from_str(data).unwrap()
|
|
|
|
};
|
|
|
|
|
|
|
|
let mut writer = env.write_txn().unwrap();
|
|
|
|
let _update_id = index.schema_update(&mut writer, schema).unwrap();
|
|
|
|
writer.commit().unwrap();
|
|
|
|
|
|
|
|
let mut additions = index.documents_addition();
|
|
|
|
|
|
|
|
let doc1 = serde_json::json!({
|
|
|
|
"id": 123,
|
|
|
|
"name": "s̷̡̢̡̧̺̜̞͕͉͉͕̜͔̟̼̥̝͍̟̖͔͔̪͉̲̹̝̣̖͎̞̤̥͓͎̭̩͕̙̩̿̀̋̅̈́̌́̏̍̄̽͂̆̾̀̿̕̚̚͜͠͠ͅͅļ̵̨̨̨̰̦̻̳̖̳͚̬̫͚̦͖͈̲̫̣̩̥̻̙̦̱̼̠̖̻̼̘̖͉̪̜̠̙͖̙̩͔̖̯̩̲̿̽͋̔̿̍̓͂̍̿͊͆̃͗̔̎͐͌̾̆͗́̆̒̔̾̅̚̚͜͜ͅͅī̵̛̦̅̔̓͂͌̾́͂͛̎̋͐͆̽̂̋̋́̾̀̉̓̏̽́̑̀͒̇͋͛̈́̃̉̏͊̌̄̽̿̏̇͘̕̚̕p̶̧̛̛̖̯̗͕̝̗̭̱͙̖̗̟̟̐͆̊̂͐̋̓̂̈́̓͊̆͌̾̾͐͋͗͌̆̿̅͆̈́̈́̉͋̍͊͗̌̓̅̈̎̇̃̎̈́̉̐̋͑̃͘̕͘d̴̢̨̛͕̘̯͖̭̮̝̝̐̊̈̅̐̀͒̀́̈́̀͌̽͛͆͑̀̽̿͛̃̋̇̎̀́̂́͘͠͝ǫ̵̨̛̮̩̘͚̬̯̖̱͍̼͑͑̓̐́̑̿̈́̔͌̂̄͐͝ģ̶̧̜͇̣̭̺̪̺̖̻͖̮̭̣̙̻͒͊͗̓̓͒̀̀ͅ",
|
|
|
|
});
|
|
|
|
|
|
|
|
additions.update_document(doc1);
|
|
|
|
|
|
|
|
let mut writer = env.write_txn().unwrap();
|
|
|
|
let update_id = additions.finalize(&mut writer).unwrap();
|
|
|
|
writer.commit().unwrap();
|
|
|
|
|
|
|
|
// block until the transaction is processed
|
|
|
|
let _ = receiver.into_iter().find(|id| *id == update_id);
|
|
|
|
|
|
|
|
let reader = env.read_txn().unwrap();
|
|
|
|
let result = index.update_status(&reader, update_id).unwrap();
|
2019-11-13 01:00:47 +08:00
|
|
|
assert_matches!(result, Some(UpdateStatus::Processed { content }) if content.error.is_none());
|
2019-11-11 00:41:32 +08:00
|
|
|
}
|
|
|
|
|
2019-11-05 19:03:37 +08:00
|
|
|
#[test]
|
|
|
|
fn add_schema_attributes_at_end() {
|
|
|
|
let dir = tempfile::tempdir().unwrap();
|
|
|
|
|
|
|
|
let database = Database::open_or_create(dir.path()).unwrap();
|
|
|
|
let env = &database.env;
|
|
|
|
|
|
|
|
let (sender, receiver) = mpsc::sync_channel(100);
|
2019-11-16 00:33:06 +08:00
|
|
|
let update_fn = move |_name: &str, update: ProcessedUpdateResult| {
|
|
|
|
sender.send(update.update_id).unwrap()
|
|
|
|
};
|
2019-11-05 19:03:37 +08:00
|
|
|
let index = database.create_index("test").unwrap();
|
|
|
|
|
2019-11-16 00:33:06 +08:00
|
|
|
database.set_update_callback(Box::new(update_fn));
|
2019-11-05 19:03:37 +08:00
|
|
|
|
|
|
|
let schema = {
|
|
|
|
let data = r#"
|
|
|
|
identifier = "id"
|
|
|
|
|
|
|
|
[attributes."name"]
|
|
|
|
displayed = true
|
|
|
|
indexed = true
|
|
|
|
|
|
|
|
[attributes."description"]
|
|
|
|
displayed = true
|
|
|
|
indexed = true
|
|
|
|
"#;
|
|
|
|
toml::from_str(data).unwrap()
|
|
|
|
};
|
|
|
|
|
|
|
|
let mut writer = env.write_txn().unwrap();
|
|
|
|
let _update_id = index.schema_update(&mut writer, schema).unwrap();
|
|
|
|
writer.commit().unwrap();
|
|
|
|
|
|
|
|
let mut additions = index.documents_addition();
|
|
|
|
|
|
|
|
let doc1 = serde_json::json!({
|
|
|
|
"id": 123,
|
|
|
|
"name": "Marvin",
|
|
|
|
"description": "My name is Marvin",
|
|
|
|
});
|
|
|
|
|
|
|
|
let doc2 = serde_json::json!({
|
|
|
|
"id": 234,
|
|
|
|
"name": "Kevin",
|
|
|
|
"description": "My name is Kevin",
|
|
|
|
});
|
|
|
|
|
|
|
|
additions.update_document(doc1);
|
|
|
|
additions.update_document(doc2);
|
|
|
|
|
|
|
|
let mut writer = env.write_txn().unwrap();
|
|
|
|
let _update_id = additions.finalize(&mut writer).unwrap();
|
|
|
|
writer.commit().unwrap();
|
|
|
|
|
|
|
|
let schema = {
|
|
|
|
let data = r#"
|
|
|
|
identifier = "id"
|
|
|
|
|
|
|
|
[attributes."name"]
|
|
|
|
displayed = true
|
|
|
|
indexed = true
|
|
|
|
|
|
|
|
[attributes."description"]
|
|
|
|
displayed = true
|
|
|
|
indexed = true
|
|
|
|
|
|
|
|
[attributes."age"]
|
|
|
|
displayed = true
|
|
|
|
indexed = true
|
|
|
|
|
|
|
|
[attributes."sex"]
|
|
|
|
displayed = true
|
|
|
|
indexed = true
|
|
|
|
"#;
|
|
|
|
toml::from_str(data).unwrap()
|
|
|
|
};
|
|
|
|
|
|
|
|
let mut writer = env.write_txn().unwrap();
|
|
|
|
let update_id = index.schema_update(&mut writer, schema).unwrap();
|
|
|
|
writer.commit().unwrap();
|
|
|
|
|
|
|
|
// block until the transaction is processed
|
|
|
|
let _ = receiver.iter().find(|id| *id == update_id);
|
|
|
|
|
|
|
|
// check if it has been accepted
|
|
|
|
let reader = env.read_txn().unwrap();
|
|
|
|
let result = index.update_status(&reader, update_id).unwrap();
|
2019-11-13 01:00:47 +08:00
|
|
|
assert_matches!(result, Some(UpdateStatus::Processed { content }) if content.error.is_none());
|
2019-11-05 19:03:37 +08:00
|
|
|
reader.abort();
|
|
|
|
|
|
|
|
let mut additions = index.documents_addition();
|
|
|
|
|
|
|
|
let doc1 = serde_json::json!({
|
|
|
|
"id": 123,
|
|
|
|
"name": "Marvin",
|
|
|
|
"description": "My name is Marvin",
|
|
|
|
"age": 21,
|
|
|
|
"sex": "Male",
|
|
|
|
});
|
|
|
|
|
|
|
|
let doc2 = serde_json::json!({
|
|
|
|
"id": 234,
|
|
|
|
"name": "Kevin",
|
|
|
|
"description": "My name is Kevin",
|
|
|
|
"age": 23,
|
|
|
|
"sex": "Male",
|
|
|
|
});
|
|
|
|
|
|
|
|
additions.update_document(doc1);
|
|
|
|
additions.update_document(doc2);
|
|
|
|
|
|
|
|
let mut writer = env.write_txn().unwrap();
|
|
|
|
let update_id = additions.finalize(&mut writer).unwrap();
|
|
|
|
writer.commit().unwrap();
|
|
|
|
|
|
|
|
// block until the transaction is processed
|
|
|
|
let _ = receiver.iter().find(|id| *id == update_id);
|
|
|
|
|
|
|
|
// check if it has been accepted
|
|
|
|
let reader = env.read_txn().unwrap();
|
|
|
|
let result = index.update_status(&reader, update_id).unwrap();
|
2019-11-13 01:00:47 +08:00
|
|
|
assert_matches!(result, Some(UpdateStatus::Processed { content }) if content.error.is_none());
|
2019-11-05 19:03:37 +08:00
|
|
|
|
|
|
|
// even try to search for a document
|
|
|
|
let results = index.query_builder().query(&reader, "21 ", 0..20).unwrap();
|
|
|
|
assert_matches!(results.len(), 1);
|
|
|
|
|
|
|
|
reader.abort();
|
|
|
|
|
|
|
|
// try to introduce attributes in the middle of the schema
|
|
|
|
let schema = {
|
|
|
|
let data = r#"
|
|
|
|
identifier = "id"
|
|
|
|
|
|
|
|
[attributes."name"]
|
|
|
|
displayed = true
|
|
|
|
indexed = true
|
|
|
|
|
|
|
|
[attributes."description"]
|
|
|
|
displayed = true
|
|
|
|
indexed = true
|
|
|
|
|
|
|
|
[attributes."city"]
|
|
|
|
displayed = true
|
|
|
|
indexed = true
|
|
|
|
|
|
|
|
[attributes."age"]
|
|
|
|
displayed = true
|
|
|
|
indexed = true
|
|
|
|
|
|
|
|
[attributes."sex"]
|
|
|
|
displayed = true
|
|
|
|
indexed = true
|
|
|
|
"#;
|
|
|
|
toml::from_str(data).unwrap()
|
|
|
|
};
|
|
|
|
|
|
|
|
let mut writer = env.write_txn().unwrap();
|
|
|
|
let update_id = index.schema_update(&mut writer, schema).unwrap();
|
|
|
|
writer.commit().unwrap();
|
|
|
|
|
|
|
|
// block until the transaction is processed
|
|
|
|
let _ = receiver.iter().find(|id| *id == update_id);
|
|
|
|
|
|
|
|
// check if it has been accepted
|
|
|
|
let reader = env.read_txn().unwrap();
|
|
|
|
let result = index.update_status(&reader, update_id).unwrap();
|
2019-11-13 01:00:47 +08:00
|
|
|
assert_matches!(result, Some(UpdateStatus::Processed { content }) if content.error.is_some());
|
2019-11-05 19:03:37 +08:00
|
|
|
}
|
2019-11-05 22:03:18 +08:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn deserialize_documents() {
|
|
|
|
let dir = tempfile::tempdir().unwrap();
|
|
|
|
|
|
|
|
let database = Database::open_or_create(dir.path()).unwrap();
|
|
|
|
let env = &database.env;
|
|
|
|
|
|
|
|
let (sender, receiver) = mpsc::sync_channel(100);
|
2019-11-16 00:33:06 +08:00
|
|
|
let update_fn = move |_name: &str, update: ProcessedUpdateResult| {
|
|
|
|
sender.send(update.update_id).unwrap()
|
|
|
|
};
|
2019-11-05 22:03:18 +08:00
|
|
|
let index = database.create_index("test").unwrap();
|
|
|
|
|
2019-11-16 00:33:06 +08:00
|
|
|
database.set_update_callback(Box::new(update_fn));
|
2019-11-05 22:03:18 +08:00
|
|
|
|
|
|
|
let schema = {
|
|
|
|
let data = r#"
|
|
|
|
identifier = "id"
|
|
|
|
|
|
|
|
[attributes."name"]
|
|
|
|
displayed = true
|
|
|
|
indexed = true
|
|
|
|
|
|
|
|
[attributes."description"]
|
|
|
|
displayed = true
|
|
|
|
indexed = true
|
|
|
|
"#;
|
|
|
|
toml::from_str(data).unwrap()
|
|
|
|
};
|
|
|
|
|
|
|
|
let mut writer = env.write_txn().unwrap();
|
|
|
|
let _update_id = index.schema_update(&mut writer, schema).unwrap();
|
|
|
|
writer.commit().unwrap();
|
|
|
|
|
|
|
|
let mut additions = index.documents_addition();
|
|
|
|
|
|
|
|
// DocumentId(7900334843754999545)
|
|
|
|
let doc1 = serde_json::json!({
|
|
|
|
"id": 123,
|
|
|
|
"name": "Marvin",
|
|
|
|
"description": "My name is Marvin",
|
|
|
|
});
|
|
|
|
|
|
|
|
// DocumentId(8367468610878465872)
|
|
|
|
let doc2 = serde_json::json!({
|
|
|
|
"id": 234,
|
|
|
|
"name": "Kevin",
|
|
|
|
"description": "My name is Kevin",
|
|
|
|
});
|
|
|
|
|
|
|
|
additions.update_document(doc1);
|
|
|
|
additions.update_document(doc2);
|
|
|
|
|
|
|
|
let mut writer = env.write_txn().unwrap();
|
|
|
|
let update_id = additions.finalize(&mut writer).unwrap();
|
|
|
|
writer.commit().unwrap();
|
|
|
|
|
|
|
|
// block until the transaction is processed
|
|
|
|
let _ = receiver.into_iter().find(|id| *id == update_id);
|
|
|
|
|
|
|
|
let reader = env.read_txn().unwrap();
|
|
|
|
let result = index.update_status(&reader, update_id).unwrap();
|
2019-11-13 01:00:47 +08:00
|
|
|
assert_matches!(result, Some(UpdateStatus::Processed { content }) if content.error.is_none());
|
2019-11-05 22:03:18 +08:00
|
|
|
|
|
|
|
let document: Option<IgnoredAny> = index.document(&reader, None, DocumentId(25)).unwrap();
|
|
|
|
assert!(document.is_none());
|
|
|
|
|
|
|
|
let document: Option<IgnoredAny> = index
|
|
|
|
.document(&reader, None, DocumentId(7900334843754999545))
|
|
|
|
.unwrap();
|
|
|
|
assert!(document.is_some());
|
|
|
|
|
|
|
|
let document: Option<IgnoredAny> = index
|
|
|
|
.document(&reader, None, DocumentId(8367468610878465872))
|
|
|
|
.unwrap();
|
|
|
|
assert!(document.is_some());
|
|
|
|
}
|
2019-11-05 22:23:41 +08:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn partial_document_update() {
|
|
|
|
let dir = tempfile::tempdir().unwrap();
|
|
|
|
|
|
|
|
let database = Database::open_or_create(dir.path()).unwrap();
|
|
|
|
let env = &database.env;
|
|
|
|
|
|
|
|
let (sender, receiver) = mpsc::sync_channel(100);
|
2019-11-16 00:33:06 +08:00
|
|
|
let update_fn = move |_name: &str, update: ProcessedUpdateResult| {
|
|
|
|
sender.send(update.update_id).unwrap()
|
|
|
|
};
|
2019-11-05 22:23:41 +08:00
|
|
|
let index = database.create_index("test").unwrap();
|
|
|
|
|
2019-11-16 00:33:06 +08:00
|
|
|
database.set_update_callback(Box::new(update_fn));
|
2019-11-05 22:23:41 +08:00
|
|
|
|
|
|
|
let schema = {
|
|
|
|
let data = r#"
|
|
|
|
identifier = "id"
|
|
|
|
|
|
|
|
[attributes."id"]
|
|
|
|
displayed = true
|
|
|
|
|
|
|
|
[attributes."name"]
|
|
|
|
displayed = true
|
|
|
|
indexed = true
|
|
|
|
|
|
|
|
[attributes."description"]
|
|
|
|
displayed = true
|
|
|
|
indexed = true
|
|
|
|
"#;
|
|
|
|
toml::from_str(data).unwrap()
|
|
|
|
};
|
|
|
|
|
|
|
|
let mut writer = env.write_txn().unwrap();
|
|
|
|
let _update_id = index.schema_update(&mut writer, schema).unwrap();
|
|
|
|
writer.commit().unwrap();
|
|
|
|
|
|
|
|
let mut additions = index.documents_addition();
|
|
|
|
|
|
|
|
// DocumentId(7900334843754999545)
|
|
|
|
let doc1 = serde_json::json!({
|
|
|
|
"id": 123,
|
|
|
|
"name": "Marvin",
|
|
|
|
"description": "My name is Marvin",
|
|
|
|
});
|
|
|
|
|
|
|
|
// DocumentId(8367468610878465872)
|
|
|
|
let doc2 = serde_json::json!({
|
|
|
|
"id": 234,
|
|
|
|
"name": "Kevin",
|
|
|
|
"description": "My name is Kevin",
|
|
|
|
});
|
|
|
|
|
|
|
|
additions.update_document(doc1);
|
|
|
|
additions.update_document(doc2);
|
|
|
|
|
|
|
|
let mut writer = env.write_txn().unwrap();
|
|
|
|
let update_id = additions.finalize(&mut writer).unwrap();
|
|
|
|
writer.commit().unwrap();
|
|
|
|
|
|
|
|
// block until the transaction is processed
|
|
|
|
let _ = receiver.iter().find(|id| *id == update_id);
|
|
|
|
|
|
|
|
let reader = env.read_txn().unwrap();
|
|
|
|
let result = index.update_status(&reader, update_id).unwrap();
|
2019-11-13 01:00:47 +08:00
|
|
|
assert_matches!(result, Some(UpdateStatus::Processed { content }) if content.error.is_none());
|
2019-11-05 22:23:41 +08:00
|
|
|
|
|
|
|
let document: Option<IgnoredAny> = index.document(&reader, None, DocumentId(25)).unwrap();
|
|
|
|
assert!(document.is_none());
|
|
|
|
|
|
|
|
let document: Option<IgnoredAny> = index
|
|
|
|
.document(&reader, None, DocumentId(7900334843754999545))
|
|
|
|
.unwrap();
|
|
|
|
assert!(document.is_some());
|
|
|
|
|
|
|
|
let document: Option<IgnoredAny> = index
|
|
|
|
.document(&reader, None, DocumentId(8367468610878465872))
|
|
|
|
.unwrap();
|
|
|
|
assert!(document.is_some());
|
|
|
|
|
|
|
|
reader.abort();
|
|
|
|
|
|
|
|
let mut partial_additions = index.documents_partial_addition();
|
|
|
|
|
|
|
|
// DocumentId(7900334843754999545)
|
|
|
|
let partial_doc1 = serde_json::json!({
|
|
|
|
"id": 123,
|
|
|
|
"description": "I am the new Marvin",
|
|
|
|
});
|
|
|
|
|
|
|
|
// DocumentId(8367468610878465872)
|
|
|
|
let partial_doc2 = serde_json::json!({
|
|
|
|
"id": 234,
|
|
|
|
"description": "I am the new Kevin",
|
|
|
|
});
|
|
|
|
|
|
|
|
partial_additions.update_document(partial_doc1);
|
|
|
|
partial_additions.update_document(partial_doc2);
|
|
|
|
|
|
|
|
let mut writer = env.write_txn().unwrap();
|
|
|
|
let update_id = partial_additions.finalize(&mut writer).unwrap();
|
|
|
|
writer.commit().unwrap();
|
|
|
|
|
|
|
|
// block until the transaction is processed
|
|
|
|
let _ = receiver.iter().find(|id| *id == update_id);
|
|
|
|
|
|
|
|
let reader = env.read_txn().unwrap();
|
|
|
|
let result = index.update_status(&reader, update_id).unwrap();
|
2019-11-13 01:00:47 +08:00
|
|
|
assert_matches!(result, Some(UpdateStatus::Processed { content }) if content.error.is_none());
|
2019-11-05 22:23:41 +08:00
|
|
|
|
|
|
|
let document: Option<serde_json::Value> = index
|
|
|
|
.document(&reader, None, DocumentId(7900334843754999545))
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
let new_doc1 = serde_json::json!({
|
|
|
|
"id": 123,
|
|
|
|
"name": "Marvin",
|
|
|
|
"description": "I am the new Marvin",
|
|
|
|
});
|
|
|
|
assert_eq!(document, Some(new_doc1));
|
|
|
|
|
|
|
|
let document: Option<serde_json::Value> = index
|
|
|
|
.document(&reader, None, DocumentId(8367468610878465872))
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
let new_doc2 = serde_json::json!({
|
|
|
|
"id": 234,
|
|
|
|
"name": "Kevin",
|
|
|
|
"description": "I am the new Kevin",
|
|
|
|
});
|
|
|
|
assert_eq!(document, Some(new_doc2));
|
|
|
|
}
|
2019-11-06 17:49:31 +08:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn delete_index() {
|
|
|
|
let dir = tempfile::tempdir().unwrap();
|
|
|
|
|
2019-11-20 18:20:50 +08:00
|
|
|
let database = Arc::new(Database::open_or_create(dir.path()).unwrap());
|
|
|
|
let env = &database.env;
|
|
|
|
|
|
|
|
let (sender, receiver) = mpsc::sync_channel(100);
|
|
|
|
let db_cloned = database.clone();
|
|
|
|
let update_fn = move |name: &str, update: ProcessedUpdateResult| {
|
|
|
|
// try to open index to trigger a lock
|
|
|
|
let _ = db_cloned.open_index(name);
|
|
|
|
sender.send(update.update_id).unwrap()
|
|
|
|
};
|
|
|
|
|
|
|
|
// create the index
|
|
|
|
let index = database.create_index("test").unwrap();
|
|
|
|
|
|
|
|
database.set_update_callback(Box::new(update_fn));
|
|
|
|
|
|
|
|
let schema = {
|
|
|
|
let data = r#"
|
|
|
|
identifier = "id"
|
|
|
|
|
|
|
|
[attributes."name"]
|
|
|
|
displayed = true
|
|
|
|
indexed = true
|
|
|
|
|
|
|
|
[attributes."description"]
|
|
|
|
displayed = true
|
|
|
|
indexed = true
|
|
|
|
"#;
|
|
|
|
toml::from_str(data).unwrap()
|
|
|
|
};
|
2019-11-06 17:49:31 +08:00
|
|
|
|
2019-11-20 18:20:50 +08:00
|
|
|
// add a schema to the index
|
|
|
|
let mut writer = env.write_txn().unwrap();
|
|
|
|
let _update_id = index.schema_update(&mut writer, schema).unwrap();
|
|
|
|
writer.commit().unwrap();
|
|
|
|
|
|
|
|
// add documents to the index
|
|
|
|
let mut additions = index.documents_addition();
|
|
|
|
|
|
|
|
let doc1 = serde_json::json!({
|
|
|
|
"id": 123,
|
|
|
|
"name": "Marvin",
|
|
|
|
"description": "My name is Marvin",
|
|
|
|
});
|
|
|
|
|
|
|
|
let doc2 = serde_json::json!({
|
|
|
|
"id": 234,
|
|
|
|
"name": "Kevin",
|
|
|
|
"description": "My name is Kevin",
|
|
|
|
});
|
|
|
|
|
|
|
|
additions.update_document(doc1);
|
|
|
|
additions.update_document(doc2);
|
|
|
|
|
|
|
|
let mut writer = env.write_txn().unwrap();
|
|
|
|
let update_id = additions.finalize(&mut writer).unwrap();
|
|
|
|
writer.commit().unwrap();
|
|
|
|
|
|
|
|
// delete the index
|
2019-11-06 17:49:31 +08:00
|
|
|
let deleted = database.delete_index("test").unwrap();
|
|
|
|
assert!(deleted);
|
|
|
|
|
2019-11-20 18:20:50 +08:00
|
|
|
// block until the transaction is processed
|
|
|
|
let _ = receiver.into_iter().find(|id| *id == update_id);
|
|
|
|
|
2019-11-06 17:49:31 +08:00
|
|
|
let result = database.open_index("test");
|
|
|
|
assert!(result.is_none());
|
|
|
|
}
|
2019-11-04 20:18:07 +08:00
|
|
|
}
|