use std::io;
use std::num::FpCategory;
use serde::ser;
use super::error::{Error, ErrorCode, Result};
pub struct Serializer<W, F=CompactFormatter> {
writer: W,
formatter: F,
first: bool,
}
impl<W> Serializer<W>
where W: io::Write,
{
#[inline]
pub fn new(writer: W) -> Self {
Serializer::with_formatter(writer, CompactFormatter)
}
}
impl<'a, W> Serializer<W, PrettyFormatter<'a>>
where W: io::Write,
{
#[inline]
pub fn pretty(writer: W) -> Self {
Serializer::with_formatter(writer, PrettyFormatter::new())
}
}
impl<W, F> Serializer<W, F>
where W: io::Write,
F: Formatter,
{
#[inline]
pub fn with_formatter(writer: W, formatter: F) -> Self {
Serializer {
writer: writer,
formatter: formatter,
first: false,
}
}
#[inline]
pub fn into_inner(self) -> W {
self.writer
}
}
impl<W, F> ser::Serializer for Serializer<W, F>
where W: io::Write,
F: Formatter,
{
type Error = Error;
#[inline]
fn serialize_bool(&mut self, value: bool) -> Result<()> {
if value {
self.writer.write_all(b"true").map_err(From::from)
} else {
self.writer.write_all(b"false").map_err(From::from)
}
}
#[inline]
fn serialize_isize(&mut self, value: isize) -> Result<()> {
write!(&mut self.writer, "{}", value).map_err(From::from)
}
#[inline]
fn serialize_i8(&mut self, value: i8) -> Result<()> {
write!(&mut self.writer, "{}", value).map_err(From::from)
}
#[inline]
fn serialize_i16(&mut self, value: i16) -> Result<()> {
write!(&mut self.writer, "{}", value).map_err(From::from)
}
#[inline]
fn serialize_i32(&mut self, value: i32) -> Result<()> {
write!(&mut self.writer, "{}", value).map_err(From::from)
}
#[inline]
fn serialize_i64(&mut self, value: i64) -> Result<()> {
write!(&mut self.writer, "{}", value).map_err(From::from)
}
#[inline]
fn serialize_usize(&mut self, value: usize) -> Result<()> {
write!(&mut self.writer, "{}", value).map_err(From::from)
}
#[inline]
fn serialize_u8(&mut self, value: u8) -> Result<()> {
write!(&mut self.writer, "{}", value).map_err(From::from)
}
#[inline]
fn serialize_u16(&mut self, value: u16) -> Result<()> {
write!(&mut self.writer, "{}", value).map_err(From::from)
}
#[inline]
fn serialize_u32(&mut self, value: u32) -> Result<()> {
write!(&mut self.writer, "{}", value).map_err(From::from)
}
#[inline]
fn serialize_u64(&mut self, value: u64) -> Result<()> {
write!(&mut self.writer, "{}", value).map_err(From::from)
}
#[inline]
fn serialize_f32(&mut self, value: f32) -> Result<()> {
fmt_f32_or_null(&mut self.writer, value).map_err(From::from)
}
#[inline]
fn serialize_f64(&mut self, value: f64) -> Result<()> {
fmt_f64_or_null(&mut self.writer, value).map_err(From::from)
}
#[inline]
fn serialize_char(&mut self, value: char) -> Result<()> {
escape_char(&mut self.writer, value).map_err(From::from)
}
#[inline]
fn serialize_str(&mut self, value: &str) -> Result<()> {
escape_str(&mut self.writer, value).map_err(From::from)
}
#[inline]
fn serialize_none(&mut self) -> Result<()> {
self.serialize_unit()
}
#[inline]
fn serialize_some<V>(&mut self, value: V) -> Result<()>
where V: ser::Serialize
{
value.serialize(self)
}
#[inline]
fn serialize_unit(&mut self) -> Result<()> {
self.writer.write_all(b"null").map_err(From::from)
}
#[inline]
fn serialize_newtype_struct<T>(&mut self,
_name: &'static str,
value: T) -> Result<()>
where T: ser::Serialize,
{
value.serialize(self)
}
#[inline]
fn serialize_unit_variant(&mut self,
_name: &str,
_variant_index: usize,
variant: &str) -> Result<()> {
try!(self.formatter.open(&mut self.writer, b'{'));
try!(self.formatter.comma(&mut self.writer, true));
try!(self.serialize_str(variant));
try!(self.formatter.colon(&mut self.writer));
try!(self.writer.write_all(b"[]"));
self.formatter.close(&mut self.writer, b'}')
}
#[inline]
fn serialize_newtype_variant<T>(&mut self,
_name: &str,
_variant_index: usize,
variant: &str,
value: T) -> Result<()>
where T: ser::Serialize,
{
try!(self.formatter.open(&mut self.writer, b'{'));
try!(self.formatter.comma(&mut self.writer, true));
try!(self.serialize_str(variant));
try!(self.formatter.colon(&mut self.writer));
try!(value.serialize(self));
self.formatter.close(&mut self.writer, b'}')
}
#[inline]
fn serialize_seq<V>(&mut self, mut visitor: V) -> Result<()>
where V: ser::SeqVisitor,
{
match visitor.len() {
Some(len) if len == 0 => {
self.writer.write_all(b"[]").map_err(From::from)
}
_ => {
try!(self.formatter.open(&mut self.writer, b'['));
self.first = true;
while let Some(()) = try!(visitor.visit(self)) { }
self.formatter.close(&mut self.writer, b']').map_err(From::from)
}
}
}
#[inline]
fn serialize_tuple_variant<V>(&mut self,
_name: &str,
_variant_index: usize,
variant: &str,
visitor: V) -> Result<()>
where V: ser::SeqVisitor,
{
try!(self.formatter.open(&mut self.writer, b'{'));
try!(self.formatter.comma(&mut self.writer, true));
try!(self.serialize_str(variant));
try!(self.formatter.colon(&mut self.writer));
try!(self.serialize_seq(visitor));
self.formatter.close(&mut self.writer, b'}')
}
#[inline]
fn serialize_seq_elt<T>(&mut self, value: T) -> Result<()>
where T: ser::Serialize,
{
try!(self.formatter.comma(&mut self.writer, self.first));
try!(value.serialize(self));
self.first = false;
Ok(())
}
#[inline]
fn serialize_map<V>(&mut self, mut visitor: V) -> Result<()>
where V: ser::MapVisitor,
{
match visitor.len() {
Some(len) if len == 0 => {
self.writer.write_all(b"{}").map_err(From::from)
}
_ => {
try!(self.formatter.open(&mut self.writer, b'{'));
self.first = true;
while let Some(()) = try!(visitor.visit(self)) { }
self.formatter.close(&mut self.writer, b'}')
}
}
}
#[inline]
fn serialize_struct_variant<V>(&mut self,
_name: &str,
_variant_index: usize,
variant: &str,
visitor: V) -> Result<()>
where V: ser::MapVisitor,
{
try!(self.formatter.open(&mut self.writer, b'{'));
try!(self.formatter.comma(&mut self.writer, true));
try!(self.serialize_str(variant));
try!(self.formatter.colon(&mut self.writer));
try!(self.serialize_map(visitor));
self.formatter.close(&mut self.writer, b'}')
}
#[inline]
fn serialize_map_elt<K, V>(&mut self, key: K, value: V) -> Result<()>
where K: ser::Serialize,
V: ser::Serialize,
{
try!(self.formatter.comma(&mut self.writer, self.first));
try!(key.serialize(&mut MapKeySerializer { ser: self }));
try!(self.formatter.colon(&mut self.writer));
try!(value.serialize(self));
self.first = false;
Ok(())
}
}
struct MapKeySerializer<'a, W: 'a, F: 'a> {
ser: &'a mut Serializer<W, F>,
}
impl<'a, W, F> ser::Serializer for MapKeySerializer<'a, W, F>
where W: io::Write,
F: Formatter,
{
type Error = Error;
#[inline]
fn serialize_str(&mut self, value: &str) -> Result<()> {
self.ser.serialize_str(value)
}
fn serialize_bool(&mut self, _value: bool) -> Result<()> {
Err(Error::Syntax(ErrorCode::KeyMustBeAString, 0, 0))
}
fn serialize_i64(&mut self, _value: i64) -> Result<()> {
Err(Error::Syntax(ErrorCode::KeyMustBeAString, 0, 0))
}
fn serialize_u64(&mut self, _value: u64) -> Result<()> {
Err(Error::Syntax(ErrorCode::KeyMustBeAString, 0, 0))
}
fn serialize_f64(&mut self, _value: f64) -> Result<()> {
Err(Error::Syntax(ErrorCode::KeyMustBeAString, 0, 0))
}
fn serialize_unit(&mut self) -> Result<()> {
Err(Error::Syntax(ErrorCode::KeyMustBeAString, 0, 0))
}
fn serialize_none(&mut self) -> Result<()> {
Err(Error::Syntax(ErrorCode::KeyMustBeAString, 0, 0))
}
fn serialize_some<V>(&mut self, _value: V) -> Result<()>
where V: ser::Serialize
{
Err(Error::Syntax(ErrorCode::KeyMustBeAString, 0, 0))
}
fn serialize_seq<V>(&mut self, _visitor: V) -> Result<()>
where V: ser::SeqVisitor,
{
Err(Error::Syntax(ErrorCode::KeyMustBeAString, 0, 0))
}
fn serialize_seq_elt<T>(&mut self, _value: T) -> Result<()>
where T: ser::Serialize,
{
Err(Error::Syntax(ErrorCode::KeyMustBeAString, 0, 0))
}
fn serialize_map<V>(&mut self, _visitor: V) -> Result<()>
where V: ser::MapVisitor,
{
Err(Error::Syntax(ErrorCode::KeyMustBeAString, 0, 0))
}
fn serialize_map_elt<K, V>(&mut self, _key: K, _value: V) -> Result<()>
where K: ser::Serialize,
V: ser::Serialize,
{
Err(Error::Syntax(ErrorCode::KeyMustBeAString, 0, 0))
}
}
pub trait Formatter {
fn open<W>(&mut self, writer: &mut W, ch: u8) -> Result<()>
where W: io::Write;
fn comma<W>(&mut self, writer: &mut W, first: bool) -> Result<()>
where W: io::Write;
fn colon<W>(&mut self, writer: &mut W) -> Result<()>
where W: io::Write;
fn close<W>(&mut self, writer: &mut W, ch: u8) -> Result<()>
where W: io::Write;
}
pub struct CompactFormatter;
impl Formatter for CompactFormatter {
fn open<W>(&mut self, writer: &mut W, ch: u8) -> Result<()>
where W: io::Write,
{
writer.write_all(&[ch]).map_err(From::from)
}
fn comma<W>(&mut self, writer: &mut W, first: bool) -> Result<()>
where W: io::Write,
{
if first {
Ok(())
} else {
writer.write_all(b",").map_err(From::from)
}
}
fn colon<W>(&mut self, writer: &mut W) -> Result<()>
where W: io::Write,
{
writer.write_all(b":").map_err(From::from)
}
fn close<W>(&mut self, writer: &mut W, ch: u8) -> Result<()>
where W: io::Write,
{
writer.write_all(&[ch]).map_err(From::from)
}
}
pub struct PrettyFormatter<'a> {
current_indent: usize,
indent: &'a [u8],
}
impl<'a> PrettyFormatter<'a> {
pub fn new() -> Self {
PrettyFormatter::with_indent(b" ")
}
pub fn with_indent(indent: &'a [u8]) -> Self {
PrettyFormatter {
current_indent: 0,
indent: indent,
}
}
}
impl<'a> Formatter for PrettyFormatter<'a> {
fn open<W>(&mut self, writer: &mut W, ch: u8) -> Result<()>
where W: io::Write,
{
self.current_indent += 1;
writer.write_all(&[ch]).map_err(From::from)
}
fn comma<W>(&mut self, writer: &mut W, first: bool) -> Result<()>
where W: io::Write,
{
if first {
try!(writer.write_all(b"\n"));
} else {
try!(writer.write_all(b",\n"));
}
indent(writer, self.current_indent, self.indent)
}
fn colon<W>(&mut self, writer: &mut W) -> Result<()>
where W: io::Write,
{
writer.write_all(b": ").map_err(From::from)
}
fn close<W>(&mut self, writer: &mut W, ch: u8) -> Result<()>
where W: io::Write,
{
self.current_indent -= 1;
try!(writer.write(b"\n"));
try!(indent(writer, self.current_indent, self.indent));
writer.write_all(&[ch]).map_err(From::from)
}
}
#[inline]
pub fn escape_bytes<W>(wr: &mut W, bytes: &[u8]) -> Result<()>
where W: io::Write
{
try!(wr.write_all(b"\""));
let mut start = 0;
for (i, byte) in bytes.iter().enumerate() {
let escaped = match *byte {
b'"' => b"\\\"",
b'\\' => b"\\\\",
b'\x08' => b"\\b",
b'\x0c' => b"\\f",
b'\n' => b"\\n",
b'\r' => b"\\r",
b'\t' => b"\\t",
_ => { continue; }
};
if start < i {
try!(wr.write_all(&bytes[start..i]));
}
try!(wr.write_all(escaped));
start = i + 1;
}
if start != bytes.len() {
try!(wr.write_all(&bytes[start..]));
}
try!(wr.write_all(b"\""));
Ok(())
}
#[inline]
pub fn escape_str<W>(wr: &mut W, value: &str) -> Result<()>
where W: io::Write
{
escape_bytes(wr, value.as_bytes())
}
#[inline]
fn escape_char<W>(wr: &mut W, value: char) -> Result<()>
where W: io::Write
{
let mut s = String::new();
s.push(value);
escape_bytes(wr, s.as_bytes())
}
fn fmt_f32_or_null<W>(wr: &mut W, value: f32) -> Result<()>
where W: io::Write
{
match value.classify() {
FpCategory::Nan | FpCategory::Infinite => {
try!(wr.write_all(b"null"))
}
_ => {
try!(write!(wr, "{:?}", value))
}
}
Ok(())
}
fn fmt_f64_or_null<W>(wr: &mut W, value: f64) -> Result<()>
where W: io::Write
{
match value.classify() {
FpCategory::Nan | FpCategory::Infinite => {
try!(wr.write_all(b"null"))
}
_ => {
try!(write!(wr, "{:?}", value))
}
}
Ok(())
}
#[inline]
pub fn to_writer<W, T>(writer: &mut W, value: &T) -> Result<()>
where W: io::Write,
T: ser::Serialize,
{
let mut ser = Serializer::new(writer);
try!(value.serialize(&mut ser));
Ok(())
}
#[inline]
pub fn to_writer_pretty<W, T>(writer: &mut W, value: &T) -> Result<()>
where W: io::Write,
T: ser::Serialize,
{
let mut ser = Serializer::pretty(writer);
try!(value.serialize(&mut ser));
Ok(())
}
#[inline]
pub fn to_vec<T>(value: &T) -> Result<Vec<u8>>
where T: ser::Serialize,
{
let mut writer = Vec::with_capacity(128);
try!(to_writer(&mut writer, value));
Ok(writer)
}
#[inline]
pub fn to_vec_pretty<T>(value: &T) -> Result<Vec<u8>>
where T: ser::Serialize,
{
let mut writer = Vec::with_capacity(128);
try!(to_writer_pretty(&mut writer, value));
Ok(writer)
}
#[inline]
pub fn to_string<T>(value: &T) -> Result<String>
where T: ser::Serialize
{
let vec = try!(to_vec(value));
let string = try!(String::from_utf8(vec));
Ok(string)
}
#[inline]
pub fn to_string_pretty<T>(value: &T) -> Result<String>
where T: ser::Serialize
{
let vec = try!(to_vec_pretty(value));
let string = try!(String::from_utf8(vec));
Ok(string)
}
fn indent<W>(wr: &mut W, n: usize, s: &[u8]) -> Result<()>
where W: io::Write,
{
for _ in 0 .. n {
try!(wr.write_all(s));
}
Ok(())
}