| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393 |
- //
- // JsonSerializationReader.cs
- //
- // Author:
- // Atsushi Enomoto <[email protected]>
- //
- // Copyright (C) 2008 Novell, Inc (http://www.novell.com)
- //
- // Permission is hereby granted, free of charge, to any person obtaining
- // a copy of this software and associated documentation files (the
- // "Software"), to deal in the Software without restriction, including
- // without limitation the rights to use, copy, modify, merge, publish,
- // distribute, sublicense, and/or sell copies of the Software, and to
- // permit persons to whom the Software is furnished to do so, subject to
- // the following conditions:
- //
- // The above copyright notice and this permission notice shall be
- // included in all copies or substantial portions of the Software.
- //
- // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
- // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
- // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
- // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
- // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
- // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
- // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
- //
- using System;
- using System.Collections;
- using System.Collections.Generic;
- using System.Collections.ObjectModel;
- using System.Globalization;
- using System.IO;
- using System.Linq;
- using System.Reflection;
- using System.Text;
- using System.Xml;
- namespace System.Runtime.Serialization.Json
- {
- class JsonSerializationReader
- {
- DataContractJsonSerializer serializer;
- XmlReader reader;
- int serialized_object_count;
- bool verify_object_name;
- Dictionary<Type, TypeMap> typemaps = new Dictionary<Type, TypeMap> ();
- Type root_type;
- public JsonSerializationReader (DataContractJsonSerializer serializer, XmlReader reader, Type rootType, bool verifyObjectName)
- {
- this.serializer = serializer;
- this.reader = reader;
- this.root_type = rootType;
- this.verify_object_name = verifyObjectName;
- }
- public XmlReader Reader {
- get { return reader; }
- }
- public object ReadRoot ()
- {
- TypeMap rootMap = GetTypeMap (root_type);
- object v = ReadObject (root_type);
- return v;
- }
- public object ReadObject (Type type)
- {
- if (serialized_object_count ++ == serializer.MaxItemsInObjectGraph)
- throw SerializationError (String.Format ("The object graph exceeded the maximum object count '{0}' specified in the serializer", serializer.MaxItemsInObjectGraph));
- bool nullable = false;
- if (type.IsGenericType && type.GetGenericTypeDefinition () == typeof (Nullable<>)) {
- nullable = true;
- type = Nullable.GetUnderlyingType (type);
- }
- bool isNull = reader.GetAttribute ("type") == "null";
- switch (Type.GetTypeCode (type)) {
- case TypeCode.DBNull:
- string dbn = reader.ReadElementContentAsString ();
- if (dbn != String.Empty)
- throw new SerializationException (String.Format ("The only expected DBNull value string is '{{}}'. Tha actual input was '{0}'.", dbn));
- return DBNull.Value;
- case TypeCode.String:
- if (isNull) {
- reader.ReadElementContentAsString ();
- return null;
- }
- else
- return reader.ReadElementContentAsString ();
- case TypeCode.Char:
- var c = reader.ReadElementContentAsString ();
- if (c.Length > 1)
- throw new XmlException ("Invalid JSON char");
- return Char.Parse(c);
- case TypeCode.Single:
- return reader.ReadElementContentAsFloat ();
- case TypeCode.Double:
- return reader.ReadElementContentAsDouble ();
- case TypeCode.Decimal:
- return reader.ReadElementContentAsDecimal ();
- case TypeCode.Byte:
- case TypeCode.SByte:
- case TypeCode.Int16:
- case TypeCode.Int32:
- case TypeCode.UInt16:
- if (type.IsEnum)
- return Enum.ToObject (type, Convert.ChangeType (reader.ReadElementContentAsLong (), Enum.GetUnderlyingType (type), null));
- else
- return Convert.ChangeType (reader.ReadElementContentAsDecimal (), type, null);
- case TypeCode.UInt32:
- case TypeCode.Int64:
- case TypeCode.UInt64:
- if (type.IsEnum)
- return Enum.ToObject (type, Convert.ChangeType (reader.ReadElementContentAsLong (), Enum.GetUnderlyingType (type), null));
- else
- return Convert.ChangeType (reader.ReadElementContentAsDecimal (), type, null);
- case TypeCode.Boolean:
- return reader.ReadElementContentAsBoolean ();
- case TypeCode.DateTime:
- // it does not use ReadElementContentAsDateTime(). Different string format.
- var s = reader.ReadElementContentAsString ();
- if (s.Length < 2 || !s.StartsWith ("/Date(", StringComparison.Ordinal) || !s.EndsWith (")/", StringComparison.Ordinal)) {
- if (nullable)
- return null;
- throw new XmlException ("Invalid JSON DateTime format. The value format should be '/Date(UnixTime)/'");
- }
- // The date can contain [SIGN]LONG, [SIGN]LONG+HOURSMINUTES or [SIGN]LONG-HOURSMINUTES
- // the format for HOURSMINUTES is DDDD
- int tidx = s.IndexOf ('-', 8);
- if (tidx == -1)
- tidx = s.IndexOf ('+', 8);
- int minutes = 0;
- if (tidx == -1){
- s = s.Substring (6, s.Length - 8);
- } else {
- int offset;
- int.TryParse (s.Substring (tidx+1, s.Length-3-tidx), out offset);
- minutes = (offset % 100) + (offset / 100) * 60;
- if (s [tidx] == '-')
- minutes = -minutes;
- s = s.Substring (6, tidx-6);
- }
- var date = new DateTime (1970, 1, 1).AddMilliseconds (long.Parse (s));
- if (minutes != 0)
- date = date.AddMinutes (minutes);
- return date;
- default:
- if (type == typeof (Guid)) {
- return new Guid (reader.ReadElementContentAsString ());
- } else if (type == typeof (Uri)) {
- if (isNull) {
- reader.ReadElementContentAsString ();
- return null;
- }
- else
- return new Uri (reader.ReadElementContentAsString ());
- } else if (type == typeof (XmlQualifiedName)) {
- s = reader.ReadElementContentAsString ();
- int idx = s.IndexOf (':');
- return idx < 0 ? new XmlQualifiedName (s) : new XmlQualifiedName (s.Substring (0, idx), s.Substring (idx + 1));
- } else if (type != typeof (object)) {
- // strongly-typed object
- if (reader.IsEmptyElement) {
- // empty -> null array or object
- reader.Read ();
- return null;
- }
- Type ct = GetCollectionElementType (type);
- if (ct != null) {
- return DeserializeGenericCollection (type, ct);
- } else {
- TypeMap map = GetTypeMap (type);
- return map.Deserialize (this);
- }
- }
- else
- return ReadInstanceDrivenObject ();
- }
- }
- Type GetRuntimeType (string name)
- {
- name = ToRuntimeTypeName (name);
- if (serializer.KnownTypes != null)
- foreach (Type t in serializer.KnownTypes)
- if (t.FullName == name)
- return t;
- var ret = root_type.Assembly.GetType (name, false) ?? Type.GetType (name, false);
- if (ret != null)
- return ret;
- // We probably have to iterate all the existing
- // assemblies that are loaded in current domain.
- foreach (var ass in AppDomain.CurrentDomain.GetAssemblies ()) {
- ret = ass.GetType (name, false);
- if (ret != null)
- return ret;
- }
- return null;
- }
- object ReadInstanceDrivenObject ()
- {
- string type = reader.GetAttribute ("type");
- switch (type) {
- case "null":
- reader.Skip ();
- return null;
- case "object":
- string runtimeType = reader.GetAttribute ("__type");
- if (runtimeType != null) {
- Type t = GetRuntimeType (runtimeType);
- if (t == null)
- throw SerializationError (String.Format ("Cannot load type '{0}'", runtimeType));
- return ReadObject (t);
- }
- break;
- }
- string v = reader.ReadElementContentAsString ();
- switch (type) {
- case "boolean":
- switch (v) {
- case "true":
- return true;
- case "false":
- return false;
- default:
- throw SerializationError (String.Format ("Invalid JSON boolean value: {0}", v));
- }
- case "string":
- return v;
- case "number":
- int i;
- if (int.TryParse (v, NumberStyles.None, CultureInfo.InvariantCulture, out i))
- return i;
- long l;
- if (long.TryParse (v, NumberStyles.None, CultureInfo.InvariantCulture, out l))
- return l;
- ulong ul;
- if (ulong.TryParse (v, NumberStyles.None, CultureInfo.InvariantCulture, out ul))
- return ul;
- double dbl;
- if (double.TryParse (v, NumberStyles.None, CultureInfo.InvariantCulture, out dbl))
- return dbl;
- decimal dec;
- if (decimal.TryParse (v, NumberStyles.None, CultureInfo.InvariantCulture, out dec))
- return dec;
- throw SerializationError (String.Format ("Invalid JSON input: {0}", v));
- default:
- throw SerializationError (String.Format ("Unexpected type: {0}", type));
- }
- }
- string FormatTypeName (Type type)
- {
- return type.Namespace == null ? type.Name : String.Format ("{0}:#{1}", type.Name, type.Namespace);
- }
- string ToRuntimeTypeName (string s)
- {
- int idx = s.IndexOf (":#", StringComparison.Ordinal);
- return idx < 0 ? s : String.Concat (s.Substring (idx + 2), ".", s.Substring (0, idx));
- }
- IEnumerable<Type> GetInterfaces2 (Type type)
- {
- if (type.IsInterface)
- yield return type;
- foreach (var t in type.GetInterfaces ())
- yield return t;
- }
- Type GetCollectionElementType (Type type)
- {
- if (type.IsArray)
- return type.GetElementType ();
- if (type.IsGenericType) {
- // returns T for IEnumerable<T>
- foreach (Type i in GetInterfaces2 (type))
- if (i.IsGenericType && i.GetGenericTypeDefinition ().Equals (typeof (IEnumerable<>)))
- return i.GetGenericArguments () [0];
- }
- if (typeof (IEnumerable).IsAssignableFrom (type))
- // return typeof(object) for mere collection.
- return typeof (object);
- else
- return null;
- }
- object DeserializeGenericCollection (Type collectionType, Type elementType)
- {
- reader.ReadStartElement ();
- object ret;
- if (collectionType.IsInterface)
- collectionType = typeof (List<>).MakeGenericType (elementType);
- if (TypeMap.IsDictionary (collectionType)) {
- object dic = Activator.CreateInstance (collectionType);
- var itemSetter = dic.GetType ().GetProperty ("Item");
- var keyarr = new object [1];
- for (reader.MoveToContent (); reader.NodeType != XmlNodeType.EndElement; reader.MoveToContent ()) {
- if (!reader.IsStartElement ("item"))
- throw SerializationError (String.Format ("Expected element 'item', but found '{0}' in namespace '{1}'", reader.LocalName, reader.NamespaceURI));
- // reading a KeyValuePair in the form of <Key .../><Value .../>
- reader.Read ();
- reader.MoveToContent ();
- object key = ReadObject (elementType.GetGenericArguments () [0]);
- reader.MoveToContent ();
- object val = ReadObject (elementType.GetGenericArguments () [1]);
- reader.Read ();
- keyarr [0] = key;
- itemSetter.SetValue (dic, val, keyarr);
- }
- ret = dic;
- } else if (typeof (IList).IsAssignableFrom (collectionType)) {
- #if NET_2_1
- Type listType = collectionType.IsArray ? typeof (List<>).MakeGenericType (elementType) : null;
- #else
- Type listType = collectionType.IsArray ? typeof (ArrayList) : null;
- #endif
- IList c = (IList) Activator.CreateInstance (listType ?? collectionType);
- for (reader.MoveToContent (); reader.NodeType != XmlNodeType.EndElement; reader.MoveToContent ()) {
- if (!reader.IsStartElement ("item"))
- throw SerializationError (String.Format ("Expected element 'item', but found '{0}' in namespace '{1}'", reader.LocalName, reader.NamespaceURI));
- Type et = elementType == typeof (object) || elementType.IsAbstract ? null : elementType;
- object elem = ReadObject (et ?? typeof (object));
- c.Add (elem);
- }
- #if NET_2_1
- if (collectionType.IsArray) {
- Array array = Array.CreateInstance (elementType, c.Count);
- c.CopyTo (array, 0);
- ret = array;
- }
- else
- ret = c;
- #else
- ret = collectionType.IsArray ? ((ArrayList) c).ToArray (elementType) : c;
- #endif
- } else {
- object c = Activator.CreateInstance (collectionType);
- MethodInfo add = collectionType.GetMethod ("Add", new Type [] {elementType});
- if (add == null) {
- var icoll = typeof (ICollection<>).MakeGenericType (elementType);
- if (icoll.IsAssignableFrom (c.GetType ()))
- add = icoll.GetMethod ("Add");
- }
-
- for (reader.MoveToContent (); reader.NodeType != XmlNodeType.EndElement; reader.MoveToContent ()) {
- if (!reader.IsStartElement ("item"))
- throw SerializationError (String.Format ("Expected element 'item', but found '{0}' in namespace '{1}'", reader.LocalName, reader.NamespaceURI));
- object elem = ReadObject (elementType);
- add.Invoke (c, new object [] {elem});
- }
- ret = c;
- }
- reader.ReadEndElement ();
- return ret;
- }
- TypeMap GetTypeMap (Type type)
- {
- TypeMap map;
- if (!typemaps.TryGetValue (type, out map)) {
- map = TypeMap.CreateTypeMap (type);
- typemaps [type] = map;
- }
- return map;
- }
- Exception SerializationError (string basemsg)
- {
- IXmlLineInfo li = reader as IXmlLineInfo;
- if (li == null || !li.HasLineInfo ())
- return new SerializationException (basemsg);
- else
- return new SerializationException (String.Format ("{0}. Error at {1} ({2},{3})", basemsg, reader.BaseURI, li.LineNumber, li.LinePosition));
- }
- }
- }
|