using System; using System.Collections.Generic; using System.Globalization; using System.Linq; using System.Text; namespace UniJSON { public class JsonFormatter : IFormatter { IStore m_w; protected IStore Store { get { return m_w; } } enum Current { ROOT, ARRAY, OBJECT } class Context { public Current Current; public int Count; public Context(Current current) { Current = current; Count = 0; } } Stack m_stack = new Stack(); string m_indent; void Indent() { if (!string.IsNullOrEmpty(m_indent)) { m_w.Write('\n'); for (int i = 0; i < m_stack.Count - 1; ++i) { m_w.Write(m_indent); } } } string m_colon; public JsonFormatter(int indent = 0) : this(new BytesStore(128), indent) { } public JsonFormatter(IStore w, int indent = 0) { m_w = w; m_stack.Push(new Context(Current.ROOT)); m_indent = new string(Enumerable.Range(0, indent).Select(x => ' ').ToArray()); m_colon = indent == 0 ? ":" : ": "; } public override string ToString() { var bytes = this.GetStoreBytes(); return Encoding.UTF8.GetString(bytes.Array, bytes.Offset, bytes.Count); } public IStore GetStore() { return m_w; } public void Clear() { m_w.Clear(); m_stack.Clear(); m_stack.Push(new Context(Current.ROOT)); } protected void CommaCheck(bool isKey = false) { var top = m_stack.Pop(); switch (top.Current) { case Current.ROOT: { if (top.Count != 0) throw new FormatterException("multiple root value"); } break; case Current.ARRAY: { if (top.Count != 0) { m_w.Write(','); } } break; case Current.OBJECT: { if (top.Count % 2 == 0) { if (!isKey) throw new FormatterException("key expected"); if (top.Count != 0) { m_w.Write(','); } } else { if (isKey) throw new FormatterException("key not expected"); } } break; } top.Count += 1; /* { var debug = string.Format("{0} {1} = {2}", m_stack.Count, top.Current, top.Count); Debug.Log(debug); } */ m_stack.Push(top); } static Utf8String s_null = Utf8String.From("null"); public void Null() { CommaCheck(); m_w.Write(s_null.Bytes); } public void BeginList(int _ = 0) { CommaCheck(); m_w.Write('['); m_stack.Push(new Context(Current.ARRAY)); } public void EndList() { if (m_stack.Peek().Current != Current.ARRAY) { throw new InvalidOperationException(); } m_w.Write(']'); m_stack.Pop(); } public void BeginMap(int _ = 0) { CommaCheck(); m_w.Write('{'); m_stack.Push(new Context(Current.OBJECT)); } public void EndMap() { if (m_stack.Peek().Current != Current.OBJECT) { throw new InvalidOperationException(); } m_stack.Pop(); Indent(); m_w.Write('}'); } public void Key(Utf8String key) { _Value(key, true); m_w.Write(m_colon); } public void Value(string x) { Value(Utf8String.From(x)); } public void Value(Utf8String key) { _Value(key, false); } void _Value(Utf8String key, bool isKey) { CommaCheck(isKey); if (isKey) { Indent(); } JsonString.Quote(key, m_w); } static Utf8String s_true = Utf8String.From("true"); static Utf8String s_false = Utf8String.From("false"); public void Value(Boolean x) { CommaCheck(); m_w.Write(x ? s_true.Bytes : s_false.Bytes); } public void Value(SByte x) { CommaCheck(); m_w.Write(x.ToString()); } public void Value(Int16 x) { CommaCheck(); m_w.Write(x.ToString()); } public void Value(Int32 x) { CommaCheck(); m_w.Write(x.ToString()); } public void Value(Int64 x) { CommaCheck(); m_w.Write(x.ToString()); } public void Value(Byte x) { CommaCheck(); m_w.Write(x.ToString()); } public void Value(UInt16 x) { CommaCheck(); m_w.Write(x.ToString()); } public void Value(UInt32 x) { CommaCheck(); m_w.Write(x.ToString()); } public void Value(UInt64 x) { CommaCheck(); m_w.Write(x.ToString()); } public void Value(Single x) { CommaCheck(); m_w.Write(x.ToString("R", CultureInfo.InvariantCulture)); } public void Value(Double x) { CommaCheck(); m_w.Write(x.ToString("R", CultureInfo.InvariantCulture)); } public void Value(ArraySegment x) { CommaCheck(); m_w.Write('"'); m_w.Write(Convert.ToBase64String(x.Array, x.Offset, x.Count)); m_w.Write('"'); } public void Raw(ArraySegment x) { CommaCheck(); m_w.Write(x); } // ISO-8601: YYYY-MM-DD“T”hh:mm:ss“Z” public void Value(DateTimeOffset x) { Value(x.ToString("yyyy-MM-ddTHH:mm:ssZ")); } public void Value(JsonNode node) { CommaCheck(); m_w.Write(node.Value.Bytes); } } }