use bson::{Bson, Document};
use serde::Deserialize;
use super::{Outcome, TestFile};
use crate::{
options::{Collation, ReplaceOptions},
test::{run_spec_test, CLIENT, LOCK},
};
#[derive(Debug, Deserialize)]
#[serde(rename_all = "camelCase")]
struct Arguments {
pub filter: Document,
pub replacement: Document,
pub upsert: Option<bool>,
pub collation: Option<Collation>,
}
#[derive(Debug, Deserialize)]
#[serde(rename_all = "camelCase")]
struct ResultDoc {
pub matched_count: i64,
pub modified_count: i64,
pub upserted_count: Option<i64>,
pub upserted_id: Option<Bson>,
}
#[function_name::named]
fn run_replace_one_test(test_file: TestFile) {
let data = test_file.data;
for test_case in test_file.tests {
if test_case.operation.name != "replaceOne" {
continue;
}
let _guard = LOCK.run_concurrently();
let coll = CLIENT.init_db_and_coll(
function_name!(),
&test_case.description.replace('$', "%").replace(' ', "_"),
);
coll.insert_many(data.clone(), None)
.expect(&test_case.description);
let arguments: Arguments = bson::from_bson(Bson::Document(test_case.operation.arguments))
.expect(&test_case.description);
let outcome: Outcome<ResultDoc> =
bson::from_bson(Bson::Document(test_case.outcome)).expect(&test_case.description);
if let Some(ref c) = outcome.collection {
if let Some(ref name) = c.name {
CLIENT.drop_collection(function_name!(), name);
}
}
let options = ReplaceOptions {
upsert: arguments.upsert,
collation: arguments.collation,
..Default::default()
};
let result = coll
.replace_one(arguments.filter, arguments.replacement, options)
.expect(&test_case.description);
assert_eq!(
outcome.result.matched_count, result.matched_count,
"{}",
test_case.description
);
assert_eq!(
outcome.result.modified_count, result.modified_count,
"{}",
test_case.description
);
assert_eq!(
outcome.result.upserted_count.unwrap_or(0),
if result.upserted_id.is_some() { 1 } else { 0 },
"{}",
test_case.description
);
assert_eq!(
outcome.result.upserted_id, result.upserted_id,
"{}",
test_case.description
);
if let Some(c) = outcome.collection {
let outcome_coll = match c.name {
Some(ref name) => CLIENT.get_coll(function_name!(), name),
None => coll,
};
assert_eq!(
c.data,
super::find_all(&outcome_coll),
"{}",
test_case.description
);
}
}
}
#[test]
fn run() {
run_spec_test(&["crud", "v1", "write"], run_replace_one_test);
}