use std::collections::BTreeMap;
use chrono::{DateTime, UTC};
use rustc_serialize::json;
use rustc_serialize::hex::ToHex;
use spec::{ElementType, BinarySubtype};
#[derive(Debug, Clone)]
pub enum Bson {
FloatingPoint(f64),
String(String),
Array(Array),
Document(Document),
Boolean(bool),
Null,
RegExp(String, String),
JavaScriptCode(String),
JavaScriptCodeWithScope(String, Document),
I32(i32),
I64(i64),
TimeStamp(i64),
Binary(BinarySubtype, Vec<u8>),
ObjectId([u8; 12]),
UtcDatetime(DateTime<UTC>),
}
pub type Array = Vec<Bson>;
pub type Document = BTreeMap<String, Bson>;
impl Bson {
pub fn element_type(&self) -> ElementType {
match self {
&Bson::FloatingPoint(..) => ElementType::FloatingPoint,
&Bson::String(..) => ElementType::Utf8String,
&Bson::Array(..) => ElementType::Array,
&Bson::Document(..) => ElementType::EmbeddedDocument,
&Bson::Boolean(..) => ElementType::Boolean,
&Bson::Null => ElementType::NullValue,
&Bson::RegExp(..) => ElementType::RegularExpression,
&Bson::JavaScriptCode(..) => ElementType::JavaScriptCode,
&Bson::JavaScriptCodeWithScope(..) => ElementType::JavaScriptCodeWithScope,
&Bson::I32(..) => ElementType::Integer32Bit,
&Bson::I64(..) => ElementType::Integer64Bit,
&Bson::TimeStamp(..) => ElementType::TimeStamp,
&Bson::Binary(..) => ElementType::Binary,
&Bson::ObjectId(..) => ElementType::ObjectId,
&Bson::UtcDatetime(..) => ElementType::UtcDatetime,
}
}
pub fn to_json(&self) -> json::Json {
match self {
&Bson::FloatingPoint(v) => json::Json::F64(v),
&Bson::String(ref v) => json::Json::String(v.clone()),
&Bson::Array(ref v) =>
json::Json::Array(v.iter().map(|x| x.to_json()).collect()),
&Bson::Document(ref v) =>
json::Json::Object(v.iter().map(|(k, v)| (k.clone(), v.to_json())).collect()),
&Bson::Boolean(v) => json::Json::Boolean(v),
&Bson::Null => json::Json::Null,
&Bson::RegExp(ref pat, ref opt) => {
let mut re = json::Object::new();
re.insert("pattern".to_owned(), json::Json::String(pat.clone()));
re.insert("options".to_owned(), json::Json::String(opt.clone()));
json::Json::Object(re)
},
&Bson::JavaScriptCode(ref code) => json::Json::String(code.clone()),
&Bson::JavaScriptCodeWithScope(ref code, ref scope) => {
let mut obj = json::Object::new();
obj.insert("code".to_owned(), json::Json::String(code.clone()));
let scope_obj =
scope.iter().map(|(k, v)| (k.clone(), v.to_json())).collect();
obj.insert("scope".to_owned(), json::Json::Object(scope_obj));
json::Json::Object(obj)
},
&Bson::I32(v) => json::Json::I64(v as i64),
&Bson::I64(v) => json::Json::I64(v),
&Bson::TimeStamp(v) => json::Json::I64(v),
&Bson::Binary(t, ref v) => {
let mut obj = json::Object::new();
let tval: u8 = From::from(t);
obj.insert("type".to_owned(), json::Json::I64(tval as i64));
obj.insert("data".to_owned(), json::Json::String(v.to_hex()));
json::Json::Object(obj)
},
&Bson::ObjectId(v) => json::Json::String(v.to_hex()),
&Bson::UtcDatetime(ref v) => json::Json::String(v.to_string()),
}
}
pub fn from_json(j: &json::Json) -> Bson {
match j {
&json::Json::I64(x) => Bson::I64(x),
&json::Json::U64(x) => Bson::I64(x as i64),
&json::Json::F64(x) => Bson::FloatingPoint(x),
&json::Json::String(ref x) => Bson::String(x.clone()),
&json::Json::Boolean(x) => Bson::Boolean(x),
&json::Json::Array(ref x) => Bson::Array(x.iter().map(Bson::from_json).collect()),
&json::Json::Object(ref x) => Bson::Document(x.iter().map(|(k, v)| (k.clone(), Bson::from_json(v))).collect()),
&json::Json::Null => Bson::Null,
}
}
}