// Copyright (c) Microsoft Corporation. All rights reserved. See License.txt in the project root for license information. using System.Collections.Generic; using System.Globalization; using System.IO; using System.Runtime.Serialization.Json; using System.Text; using System.Xml; namespace System.Json { internal static class JXmlToJsonValueConverter { internal const string RootElementName = "root"; internal const string ItemElementName = "item"; internal const string TypeAttributeName = "type"; internal const string ArrayAttributeValue = "array"; internal const string BooleanAttributeValue = "boolean"; internal const string NullAttributeValue = "null"; internal const string NumberAttributeValue = "number"; internal const string ObjectAttributeValue = "object"; internal const string StringAttributeValue = "string"; private const string TypeHintAttributeName = "__type"; private static readonly char[] _floatingPointChars = new char[] { '.', 'e', 'E' }; public static JsonValue JXMLToJsonValue(Stream jsonStream) { if (jsonStream == null) { throw new ArgumentNullException("jsonStream"); } return JXMLToJsonValue(jsonStream, null); } public static JsonValue JXMLToJsonValue(string jsonString) { if (jsonString == null) { throw new ArgumentNullException("jsonString"); } if (jsonString.Length == 0) { throw new ArgumentException(Properties.Resources.JsonStringCannotBeEmpty, "jsonString"); } byte[] jsonBytes = Encoding.UTF8.GetBytes(jsonString); return JXMLToJsonValue(null, jsonBytes); } public static JsonValue JXMLToJsonValue(XmlDictionaryReader jsonReader) { if (jsonReader == null) { throw new ArgumentNullException("jsonReader"); } const string RootObjectName = "RootObject"; Stack jsonStack = new Stack(); string nodeType = null; bool isEmptyElement = false; JsonValue parent = new JsonObject(); jsonStack.Push(parent); string currentName = RootObjectName; try { MoveToRootNode(jsonReader); while (jsonStack.Count > 0 && jsonReader.NodeType != XmlNodeType.None) { if (parent is JsonObject && currentName == null) { currentName = GetMemberName(jsonReader); } nodeType = jsonReader.GetAttribute(TypeAttributeName) ?? StringAttributeValue; if (parent is JsonArray) { // For arrays, the element name has to be "item" if (jsonReader.Name != ItemElementName) { throw new FormatException(Properties.Resources.IncorrectJsonFormat); } } switch (nodeType) { case NullAttributeValue: case BooleanAttributeValue: case StringAttributeValue: case NumberAttributeValue: JsonPrimitive jsonPrimitive = ReadPrimitive(nodeType, jsonReader); InsertJsonValue(jsonStack, ref parent, ref currentName, jsonPrimitive, true); break; case ArrayAttributeValue: JsonArray jsonArray = CreateJsonArray(jsonReader, ref isEmptyElement); InsertJsonValue(jsonStack, ref parent, ref currentName, jsonArray, isEmptyElement); break; case ObjectAttributeValue: JsonObject jsonObject = CreateObjectWithTypeHint(jsonReader, ref isEmptyElement); InsertJsonValue(jsonStack, ref parent, ref currentName, jsonObject, isEmptyElement); break; default: throw new FormatException(Properties.Resources.IncorrectJsonFormat); } while (jsonReader.NodeType == XmlNodeType.EndElement && jsonStack.Count > 0) { jsonReader.Read(); SkipWhitespace(jsonReader); jsonStack.Pop(); if (jsonStack.Count > 0) { parent = jsonStack.Peek(); } } } } catch (XmlException xmlException) { throw new FormatException(Properties.Resources.IncorrectJsonFormat, xmlException); } if (jsonStack.Count != 1) { throw new FormatException(Properties.Resources.IncorrectJsonFormat); } return parent[RootObjectName]; } private static JsonValue JXMLToJsonValue(Stream jsonStream, byte[] jsonBytes) { try { using (XmlDictionaryReader jsonReader = jsonStream != null ? JsonReaderWriterFactory.CreateJsonReader(jsonStream, XmlDictionaryReaderQuotas.Max) : JsonReaderWriterFactory.CreateJsonReader(jsonBytes, XmlDictionaryReaderQuotas.Max)) { return JXMLToJsonValue(jsonReader); } } catch (XmlException) { throw new FormatException(Properties.Resources.IncorrectJsonFormat); } } private static void InsertJsonValue(Stack jsonStack, ref JsonValue parent, ref string currentName, JsonValue jsonValue, bool isEmptyElement) { if (parent is JsonArray) { ((JsonArray)parent).Add(jsonValue); } else { if (currentName != null) { ((JsonObject)parent)[currentName] = jsonValue; currentName = null; } } if (!isEmptyElement) { jsonStack.Push(jsonValue); parent = jsonValue; } } private static string GetMemberName(XmlDictionaryReader jsonReader) { string name; if (jsonReader.NamespaceURI == ItemElementName && jsonReader.LocalName == ItemElementName) { // JXML special case for names which aren't valid XML names name = jsonReader.GetAttribute(ItemElementName); if (name == null) { throw new FormatException(Properties.Resources.IncorrectJsonFormat); } } else { name = jsonReader.Name; } return name; } private static JsonObject CreateObjectWithTypeHint(XmlDictionaryReader jsonReader, ref bool isEmptyElement) { JsonObject jsonObject = new JsonObject(); string typeHintAttribute = jsonReader.GetAttribute(TypeHintAttributeName); isEmptyElement = jsonReader.IsEmptyElement; jsonReader.ReadStartElement(); SkipWhitespace(jsonReader); if (typeHintAttribute != null) { jsonObject.Add(TypeHintAttributeName, typeHintAttribute); } return jsonObject; } private static JsonArray CreateJsonArray(XmlDictionaryReader jsonReader, ref bool isEmptyElement) { JsonArray jsonArray = new JsonArray(); isEmptyElement = jsonReader.IsEmptyElement; jsonReader.ReadStartElement(); SkipWhitespace(jsonReader); return jsonArray; } private static void MoveToRootNode(XmlDictionaryReader jsonReader) { while (!jsonReader.EOF && (jsonReader.NodeType == XmlNodeType.None || jsonReader.NodeType == XmlNodeType.XmlDeclaration)) { // read into node jsonReader.Read(); SkipWhitespace(jsonReader); } if (jsonReader.NodeType != XmlNodeType.Element || !String.IsNullOrEmpty(jsonReader.NamespaceURI) || jsonReader.Name != RootElementName) { throw new FormatException(Properties.Resources.IncorrectJsonFormat); } } private static JsonPrimitive ReadPrimitive(string type, XmlDictionaryReader jsonReader) { JsonValue result = null; switch (type) { case NullAttributeValue: jsonReader.Skip(); result = null; break; case BooleanAttributeValue: result = jsonReader.ReadElementContentAsBoolean(); break; case StringAttributeValue: result = jsonReader.ReadElementContentAsString(); break; case NumberAttributeValue: string temp = jsonReader.ReadElementContentAsString(); result = ConvertStringToJsonNumber(temp); break; } SkipWhitespace(jsonReader); return (JsonPrimitive)result; } private static void SkipWhitespace(XmlDictionaryReader reader) { while (!reader.EOF && (reader.NodeType == XmlNodeType.Whitespace || reader.NodeType == XmlNodeType.SignificantWhitespace)) { reader.Read(); } } private static JsonValue ConvertStringToJsonNumber(string value) { if (value.IndexOfAny(_floatingPointChars) < 0) { int intVal; if (Int32.TryParse(value, NumberStyles.Float, CultureInfo.InvariantCulture, out intVal)) { return intVal; } long longVal; if (Int64.TryParse(value, NumberStyles.Float, CultureInfo.InvariantCulture, out longVal)) { return longVal; } } decimal decValue; if (Decimal.TryParse(value, NumberStyles.Float, CultureInfo.InvariantCulture, out decValue) && decValue != 0) { return decValue; } double dblValue; if (Double.TryParse(value, NumberStyles.Float, CultureInfo.InvariantCulture, out dblValue)) { return dblValue; } throw new ArgumentException(RS.Format(Properties.Resources.InvalidJsonPrimitive, value.ToString()), "value"); } } }