1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
//! **json_in_type** is a library for *very fast* [JSON](http://json.org/) serialization.
//! It does only serialization, not parsing.
//!
//! # Principles
//! This library is fast at runtime because it tries to do more at compile time,
//! using rust's powerful type system, and macros.
//!
//! The base idea when starting to write this library was that a json object
//! with a number of properties that is known at compile-time can be efficiently
//! stored in an ad-hoc type, that will have a compact representation in memory,
//! and will be serialized much faster than a HashMap.
//!
//! # How to use
//! This crate has two main macros, [`json_object!`](macro.json_object.html),
//! and [`json_list!`](macro.json_list.html).
//! Use them to create [json values](trait.JSONValue.html), that you can then serialize.
//!
//!

pub mod base_types;
pub mod list;
pub mod object;
pub mod string;
pub mod utils;

use std::fmt;
use std::fmt::Display;
use std::fmt::Formatter;
use std::io;

/// A trait implemented by types that can be serialized to JSON
///
/// This trait can be derived for custom structs using
/// [json_in_type_derive](https://docs.rs/json_in_type_derive/)
pub trait JSONValue {
    /// Write the object as json to the given writer
    ///
    /// # Examples
    ///
    /// Write a JSON object to a file
    ///
    /// ```
    /// # let mut my_file = ::std::io::sink();
    /// use json_in_type::JSONValue;
    ///
    /// vec![1, 2, 3].write_json(&mut my_file);
    /// ```
    fn write_json<W: io::Write>(&self, w: &mut W) -> io::Result<()>;

    /// Returns the object formatted as a json string
    ///
    /// # Panics
    /// If you implement JSONValue on your own types and emit invalid UTF-8
    /// in write_json. If you use the implementations of JSONValue provided
    /// in this library, this function will never panic.
    fn to_json_string(&self) -> String {
        // This is safe because the bytes we emit are all valid UTF-8
        String::from_utf8(self.to_json_buffer()).unwrap()
    }

    /// Returns a buffer containing the bytes of a json representation of the object
    fn to_json_buffer(&self) -> Vec<u8> {
        let mut buffer = Vec::with_capacity(512);
        self.write_json(&mut buffer).unwrap();
        buffer
    }
}

impl<'a, S: JSONValue + ?Sized> JSONValue for &'a S {
    fn write_json<W: io::Write>(&self, w: &mut W) -> io::Result<()> {
        (**self).write_json(w)
    }
}

impl<'a, S: JSONValue + ?Sized> JSONValue for Box<S> {
    fn write_json<W: io::Write>(&self, w: &mut W) -> io::Result<()> {
        (**self).write_json(w)
    }
}

/// Encapsulates a [JSONValue](trait.JSONValue.html) and implements useful traits.
///
/// # Examples
///
/// ```
/// use json_in_type::JSON;
///
/// let x_json = JSON(vec![(), (), ()]);
/// assert_eq!("[null,null,null]", x_json.to_string()); // JSON format
///
/// println!("{}", x_json); // just works. Displays [null,null,null]
///
/// let my_buffer : Vec<u8> = x_json.into();
/// assert_eq!(b"[null,null,null]".to_vec(), my_buffer);
/// ```
pub struct JSON<T: JSONValue>(pub T);

impl<T: JSONValue> Display for JSON<T> {
    fn fmt(&self, f: &mut Formatter) -> fmt::Result {
        let mut writer = utils::FormatterWriter(f);
        self.0
            .write_json(&mut writer)
            .map(|_size| ())
            .map_err(|_err| fmt::Error {})
    }
}

impl<T: JSONValue> From<JSON<T>> for Vec<u8> {
    fn from(json: JSON<T>) -> Self {
        json.0.to_json_buffer()
    }
}