mod error;
mod serde;
pub use self::error::{DecoderError, DecoderResult};
pub use self::serde::Decoder;
use std::io::Read;
use std::str;
use byteorder::{LittleEndian, ReadBytesExt};
use chrono::UTC;
use chrono::offset::TimeZone;
use spec::{self, BinarySubtype};
use bson::{Bson, Array, Document};
use oid;
use serde::de::Deserialize;
fn read_string<R: Read + ?Sized>(reader: &mut R) -> DecoderResult<String> {
let len = try!(reader.read_i32::<LittleEndian>());
let mut s = String::with_capacity(len as usize - 1);
try!(reader.take(len as u64 - 1).read_to_string(&mut s));
try!(reader.read_u8());
Ok(s)
}
fn read_cstring<R: Read + ?Sized>(reader: &mut R) -> DecoderResult<String> {
let mut v = Vec::new();
loop {
let c = try!(reader.read_u8());
if c == 0 {
break;
}
v.push(c);
}
Ok(try!(str::from_utf8(&v)).to_owned())
}
#[inline]
fn read_i32<R: Read + ?Sized>(reader: &mut R) -> DecoderResult<i32> {
reader.read_i32::<LittleEndian>().map_err(From::from)
}
#[inline]
fn read_i64<R: Read + ?Sized>(reader: &mut R) -> DecoderResult<i64> {
reader.read_i64::<LittleEndian>().map_err(From::from)
}
pub fn decode_document<R: Read + ?Sized>(reader: &mut R) -> DecoderResult<Document> {
let mut doc = Document::new();
try!(read_i32(reader));
loop {
let tag = try!(reader.read_u8());
if tag == 0 {
break;
}
let key = try!(read_cstring(reader));
let val = try!(decode_bson(reader, tag));
doc.insert(key, val);
}
Ok(doc)
}
fn decode_array<R: Read + ?Sized>(reader: &mut R) -> DecoderResult<Array> {
let mut arr = Array::new();
try!(read_i32(reader));
loop {
let tag = try!(reader.read_u8());
if tag == 0 {
break;
}
let key = try!(read_cstring(reader));
if key != &arr.len().to_string()[..] {
return Err(DecoderError::InvalidArrayKey(arr.len(), key));
}
let val = try!(decode_bson(reader, tag));
arr.push(val)
}
Ok(arr)
}
fn decode_bson<R: Read + ?Sized>(reader: &mut R, tag: u8) -> DecoderResult<Bson> {
use spec::ElementType::*;
match spec::ElementType::from(tag) {
Some(FloatingPoint) => Ok(Bson::FloatingPoint(try!(reader.read_f64::<LittleEndian>()))),
Some(Utf8String) => read_string(reader).map(Bson::String),
Some(EmbeddedDocument) => decode_document(reader).map(Bson::Document),
Some(Array) => decode_array(reader).map(Bson::Array),
Some(Binary) => {
let len = try!(read_i32(reader));
let subtype = BinarySubtype::from(try!(reader.read_u8()));
let mut data = Vec::with_capacity(len as usize);
try!(reader.take(len as u64).read_to_end(&mut data));
Ok(Bson::Binary(subtype, data))
}
Some(ObjectId) => {
let mut objid = [0; 12];
for x in &mut objid {
*x = try!(reader.read_u8());
}
Ok(Bson::ObjectId(oid::ObjectId::with_bytes(objid)))
}
Some(Boolean) => Ok(Bson::Boolean(try!(reader.read_u8()) != 0)),
Some(NullValue) => Ok(Bson::Null),
Some(RegularExpression) => {
let pat = try!(read_cstring(reader));
let opt = try!(read_cstring(reader));
Ok(Bson::RegExp(pat, opt))
}
Some(JavaScriptCode) => read_string(reader).map(Bson::JavaScriptCode),
Some(JavaScriptCodeWithScope) => {
try!(read_i32(reader));
let code = try!(read_string(reader));
let scope = try!(decode_document(reader));
Ok(Bson::JavaScriptCodeWithScope(code, scope))
}
Some(Integer32Bit) => read_i32(reader).map(Bson::I32),
Some(Integer64Bit) => read_i64(reader).map(Bson::I64),
Some(TimeStamp) => read_i64(reader).map(Bson::TimeStamp),
Some(UtcDatetime) => {
let time = try!(read_i64(reader));
Ok(Bson::UtcDatetime(UTC.timestamp(time / 1000, (time % 1000) as u32 * 1000000)))
}
Some(Symbol) => read_string(reader).map(Bson::Symbol),
Some(Undefined) | Some(DbPointer) | Some(MaxKey) | Some(MinKey) | None => {
Err(DecoderError::UnrecognizedElementType(tag))
}
}
}
pub fn from_bson<T>(bson: Bson) -> DecoderResult<T>
where T: Deserialize
{
let mut de = Decoder::new(bson);
Deserialize::deserialize(&mut de)
}