0abdbe5a7d
Former-commit-id: 7467d4b717762eeaf652d77f1486dd11ffb1ff1f
710 lines
26 KiB
C#
710 lines
26 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using System.Reflection;
|
|
using System.Runtime;
|
|
using System.Xml;
|
|
|
|
namespace System.Runtime.Serialization.Json
|
|
{
|
|
internal partial class JsonFormatReaderGenerator
|
|
{
|
|
partial class CriticalHelper
|
|
{
|
|
internal JsonFormatClassReaderDelegate GenerateClassReader(ClassDataContract classContract)
|
|
{
|
|
return (XmlReaderDelegator xr, XmlObjectSerializerReadContextComplexJson ctx, XmlDictionaryString emptyDictionaryString, XmlDictionaryString [] memberNames) => new JsonFormatReaderInterpreter (classContract).ReadFromJson (xr, ctx, emptyDictionaryString, memberNames);
|
|
}
|
|
|
|
internal JsonFormatCollectionReaderDelegate GenerateCollectionReader(CollectionDataContract collectionContract)
|
|
{
|
|
return (XmlReaderDelegator xr, XmlObjectSerializerReadContextComplexJson ctx, XmlDictionaryString emptyDS, XmlDictionaryString inm, CollectionDataContract cc) => new JsonFormatReaderInterpreter (collectionContract, false).ReadCollectionFromJson (xr, ctx, emptyDS, inm, cc);
|
|
}
|
|
|
|
internal JsonFormatGetOnlyCollectionReaderDelegate GenerateGetOnlyCollectionReader(CollectionDataContract collectionContract)
|
|
{
|
|
return (XmlReaderDelegator xr, XmlObjectSerializerReadContextComplexJson ctx, XmlDictionaryString emptyDS, XmlDictionaryString inm, CollectionDataContract cc) => new JsonFormatReaderInterpreter (collectionContract, true).ReadGetOnlyCollectionFromJson (xr, ctx, emptyDS, inm, cc);
|
|
}
|
|
}
|
|
}
|
|
|
|
class JsonFormatReaderInterpreter
|
|
{
|
|
public JsonFormatReaderInterpreter (ClassDataContract classContract)
|
|
{
|
|
this.classContract = classContract;
|
|
}
|
|
|
|
public JsonFormatReaderInterpreter (CollectionDataContract collectionContract, bool isGetOnly)
|
|
{
|
|
this.collectionContract = collectionContract;
|
|
this.is_get_only_collection = isGetOnly;
|
|
}
|
|
|
|
bool is_get_only_collection;
|
|
|
|
ClassDataContract classContract;
|
|
|
|
CollectionDataContract collectionContract;
|
|
|
|
object objectLocal;
|
|
Type objectType;
|
|
XmlReaderDelegator xmlReader;
|
|
XmlObjectSerializerReadContextComplexJson context;
|
|
|
|
XmlDictionaryString [] memberNames = null;
|
|
XmlDictionaryString emptyDictionaryString = null;
|
|
XmlDictionaryString itemName = null;
|
|
XmlDictionaryString itemNamespace = null;
|
|
|
|
public object ReadFromJson (XmlReaderDelegator xmlReader, XmlObjectSerializerReadContextComplexJson context, XmlDictionaryString emptyDictionaryString, XmlDictionaryString[] memberNames)
|
|
{
|
|
// InitArgs()
|
|
this.xmlReader = xmlReader;
|
|
this.context = context;
|
|
this.emptyDictionaryString = emptyDictionaryString;
|
|
this.memberNames = memberNames;
|
|
|
|
//DemandSerializationFormatterPermission(classContract);
|
|
//DemandMemberAccessPermission(memberAccessFlag);
|
|
CreateObject (classContract);
|
|
|
|
context.AddNewObject (objectLocal);
|
|
InvokeOnDeserializing (classContract);
|
|
|
|
string objectId = null;
|
|
|
|
if (classContract.IsISerializable)
|
|
ReadISerializable (classContract);
|
|
else
|
|
ReadClass (classContract);
|
|
if (Globals.TypeOfIDeserializationCallback.IsAssignableFrom (classContract.UnderlyingType))
|
|
((IDeserializationCallback) objectLocal).OnDeserialization (null);
|
|
InvokeOnDeserialized(classContract);
|
|
if (!InvokeFactoryMethod (classContract)) {
|
|
|
|
// Do a conversion back from DateTimeOffsetAdapter to DateTimeOffset after deserialization.
|
|
// DateTimeOffsetAdapter is used here for deserialization purposes to bypass the ISerializable implementation
|
|
// on DateTimeOffset; which does not work in partial trust.
|
|
|
|
if (classContract.UnderlyingType == Globals.TypeOfDateTimeOffsetAdapter)
|
|
objectLocal = DateTimeOffsetAdapter.GetDateTimeOffset ((DateTimeOffsetAdapter) objectLocal);
|
|
// else - do we have to call CodeInterpreter.ConvertValue()? I guess not...
|
|
}
|
|
return objectLocal;
|
|
}
|
|
|
|
public object ReadCollectionFromJson (XmlReaderDelegator xmlReader, XmlObjectSerializerReadContextComplexJson context, XmlDictionaryString emptyDictionaryString, XmlDictionaryString itemName, CollectionDataContract collectionContract)
|
|
{
|
|
#region GenerateCollectionReaderHelper
|
|
// InitArgs()
|
|
this.xmlReader = xmlReader;
|
|
this.context = context;
|
|
this.emptyDictionaryString = emptyDictionaryString;
|
|
this.itemName = itemName;
|
|
|
|
this.collectionContract = collectionContract;
|
|
|
|
#endregion
|
|
|
|
ReadCollection (collectionContract);
|
|
|
|
return objectLocal;
|
|
}
|
|
|
|
public void ReadGetOnlyCollectionFromJson (XmlReaderDelegator xmlReader, XmlObjectSerializerReadContextComplexJson context, XmlDictionaryString emptyDictionaryString, XmlDictionaryString itemName, CollectionDataContract collectionContract)
|
|
{
|
|
#region GenerateCollectionReaderHelper
|
|
// InitArgs()
|
|
this.xmlReader = xmlReader;
|
|
this.context = context;
|
|
this.emptyDictionaryString = emptyDictionaryString;
|
|
this.itemName = itemName;
|
|
|
|
this.collectionContract = collectionContract;
|
|
|
|
#endregion
|
|
|
|
ReadGetOnlyCollection (collectionContract);
|
|
}
|
|
|
|
void CreateObject (ClassDataContract classContract)
|
|
{
|
|
Type type = objectType = classContract.UnderlyingType;
|
|
if (type.IsValueType && !classContract.IsNonAttributedType)
|
|
type = Globals.TypeOfValueType;
|
|
|
|
if (classContract.UnderlyingType == Globals.TypeOfDBNull)
|
|
objectLocal = DBNull.Value;
|
|
else if (classContract.IsNonAttributedType) {
|
|
if (type.IsValueType)
|
|
objectLocal = FormatterServices.GetUninitializedObject (type);
|
|
else
|
|
objectLocal = classContract.GetNonAttributedTypeConstructor ().Invoke (new object [0]);
|
|
}
|
|
else
|
|
objectLocal = CodeInterpreter.ConvertValue (XmlFormatReaderGenerator.UnsafeGetUninitializedObject (DataContract.GetIdForInitialization (classContract)), Globals.TypeOfObject, type);
|
|
}
|
|
|
|
void InvokeOnDeserializing (ClassDataContract classContract)
|
|
{
|
|
if (classContract.BaseContract != null)
|
|
InvokeOnDeserializing (classContract.BaseContract);
|
|
if (classContract.OnDeserializing != null)
|
|
classContract.OnDeserializing.Invoke (objectLocal, new object [] {context.GetStreamingContext ()});
|
|
}
|
|
|
|
void InvokeOnDeserialized (ClassDataContract classContract)
|
|
{
|
|
if (classContract.BaseContract != null)
|
|
InvokeOnDeserialized (classContract.BaseContract);
|
|
if (classContract.OnDeserialized != null)
|
|
classContract.OnDeserialized.Invoke (objectLocal, new object [] {context.GetStreamingContext ()});
|
|
}
|
|
|
|
bool HasFactoryMethod (ClassDataContract classContract)
|
|
{
|
|
return Globals.TypeOfIObjectReference.IsAssignableFrom (classContract.UnderlyingType);
|
|
}
|
|
|
|
bool InvokeFactoryMethod (ClassDataContract classContract)
|
|
{
|
|
if (HasFactoryMethod (classContract)) {
|
|
objectLocal = CodeInterpreter.ConvertValue (context.GetRealObject ((IObjectReference) objectLocal, Globals.NewObjectId), Globals.TypeOfObject, classContract.UnderlyingType);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void ReadISerializable (ClassDataContract classContract)
|
|
{
|
|
ConstructorInfo ctor = classContract.UnderlyingType.GetConstructor (Globals.ScanAllMembers, null, JsonFormatGeneratorStatics.SerInfoCtorArgs, null);
|
|
if (ctor == null)
|
|
throw System.Runtime.Serialization.DiagnosticUtility.ExceptionUtility.ThrowHelperError (XmlObjectSerializer.CreateSerializationException (SR.GetString (SR.SerializationInfo_ConstructorNotFound, DataContract.GetClrTypeFullName (classContract.UnderlyingType))));
|
|
context.ReadSerializationInfo (xmlReader, classContract.UnderlyingType);
|
|
ctor.Invoke (objectLocal, new object [] {context.GetStreamingContext ()});
|
|
}
|
|
|
|
void ReadClass (ClassDataContract classContract)
|
|
{
|
|
if (classContract.HasExtensionData) {
|
|
ExtensionDataObject extensionData = new ExtensionDataObject ();
|
|
ReadMembers (classContract, extensionData);
|
|
ClassDataContract currentContract = classContract;
|
|
while (currentContract != null) {
|
|
MethodInfo extensionDataSetMethod = currentContract.ExtensionDataSetMethod;
|
|
if (extensionDataSetMethod != null)
|
|
extensionDataSetMethod.Invoke (objectLocal, new object [] {extensionData});
|
|
currentContract = currentContract.BaseContract;
|
|
}
|
|
}
|
|
else
|
|
ReadMembers (classContract, null);
|
|
}
|
|
|
|
void ReadMembers (ClassDataContract classContract, ExtensionDataObject extensionData)
|
|
{
|
|
int memberCount = classContract.MemberNames.Length;
|
|
context.IncrementItemCount (memberCount);
|
|
|
|
int memberIndex = -1;
|
|
|
|
// JSON intrinsic part.
|
|
BitFlagsGenerator expectedElements = new BitFlagsGenerator (memberCount);
|
|
byte [] requiredElements = new byte [expectedElements.GetLocalCount ()];
|
|
SetRequiredElements (classContract, requiredElements);
|
|
SetExpectedElements (expectedElements, 0 /*startIndex*/);
|
|
|
|
while (XmlObjectSerializerReadContext.MoveToNextElement (xmlReader)) {
|
|
int idx; // used as in "switch (idx)" in the original source.
|
|
idx = context.GetJsonMemberIndex (xmlReader, memberNames, memberIndex, extensionData);
|
|
|
|
if (memberCount > 0)
|
|
ReadMembers (idx, classContract, expectedElements, ref memberIndex);
|
|
}
|
|
|
|
if (!CheckRequiredElements (expectedElements, requiredElements))
|
|
XmlObjectSerializerReadContextComplexJson.ThrowMissingRequiredMembers (objectLocal, memberNames, expectedElements.LoadArray (), requiredElements);
|
|
}
|
|
|
|
int ReadMembers (int index, ClassDataContract classContract, BitFlagsGenerator expectedElements, ref int memberIndex)
|
|
{
|
|
int memberCount = (classContract.BaseContract == null) ? 0 : ReadMembers (index, classContract.BaseContract, expectedElements,
|
|
ref memberIndex);
|
|
|
|
if (memberCount <= index && index < memberCount + classContract.Members.Count) {
|
|
DataMember dataMember = classContract.Members [index - memberCount];
|
|
Type memberType = dataMember.MemberType;
|
|
|
|
memberIndex = memberCount;
|
|
if (!expectedElements.Load (index))
|
|
XmlObjectSerializerReadContextComplexJson.ThrowDuplicateMemberException (objectLocal, memberNames, memberIndex);
|
|
|
|
if (dataMember.IsGetOnlyCollection) {
|
|
var value = CodeInterpreter.GetMember (dataMember.MemberInfo, objectLocal);
|
|
context.StoreCollectionMemberInfo (value);
|
|
ReadValue (memberType, dataMember.Name);
|
|
} else {
|
|
var value = ReadValue (memberType, dataMember.Name);
|
|
CodeInterpreter.SetMember (dataMember.MemberInfo, objectLocal, value);
|
|
}
|
|
memberIndex = index;
|
|
ResetExpectedElements (expectedElements, index);
|
|
}
|
|
return memberCount + classContract.Members.Count;
|
|
}
|
|
|
|
bool CheckRequiredElements (BitFlagsGenerator expectedElements, byte [] requiredElements)
|
|
{
|
|
for (int i = 0; i < requiredElements.Length; i++)
|
|
if ((expectedElements.GetLocal(i) & requiredElements[i]) != 0)
|
|
return false;
|
|
return true;
|
|
}
|
|
|
|
int SetRequiredElements (ClassDataContract contract, byte [] requiredElements)
|
|
{
|
|
int memberCount = (contract.BaseContract == null) ? 0 :
|
|
SetRequiredElements (contract.BaseContract, requiredElements);
|
|
List<DataMember> members = contract.Members;
|
|
for (int i = 0; i < members.Count; i++, memberCount++) {
|
|
if (members[i].IsRequired)
|
|
BitFlagsGenerator.SetBit (requiredElements, memberCount);
|
|
}
|
|
return memberCount;
|
|
}
|
|
|
|
void SetExpectedElements (BitFlagsGenerator expectedElements, int startIndex)
|
|
{
|
|
int memberCount = expectedElements.GetBitCount ();
|
|
for (int i = startIndex; i < memberCount; i++)
|
|
expectedElements.Store (i, true);
|
|
}
|
|
|
|
void ResetExpectedElements (BitFlagsGenerator expectedElements, int index)
|
|
{
|
|
expectedElements.Store (index, false);
|
|
}
|
|
|
|
object ReadValue (Type type, string name)
|
|
{
|
|
var valueType = type;
|
|
object value = null;
|
|
bool shouldAssignNullableValue = false;
|
|
int nullables = 0;
|
|
while (type.IsGenericType && type.GetGenericTypeDefinition () == Globals.TypeOfNullable) {
|
|
nullables++;
|
|
type = type.GetGenericArguments () [0];
|
|
}
|
|
|
|
PrimitiveDataContract primitiveContract = PrimitiveDataContract.GetPrimitiveDataContract (type);
|
|
if ((primitiveContract != null && primitiveContract.UnderlyingType != Globals.TypeOfObject) || nullables != 0 || type.IsValueType) {
|
|
context.ReadAttributes (xmlReader);
|
|
string objectId = context.ReadIfNullOrRef (xmlReader, type, DataContract.IsTypeSerializable (type));
|
|
// Deserialize null
|
|
if (objectId == Globals.NullObjectId) {
|
|
|
|
if (nullables != 0)
|
|
value = Activator.CreateInstance (valueType);
|
|
else if (type.IsValueType)
|
|
throw new SerializationException (SR.GetString (SR.ValueTypeCannotBeNull, DataContract.GetClrTypeFullName (type)));
|
|
else
|
|
value = null;
|
|
} else if (objectId == string.Empty) {
|
|
// Deserialize value
|
|
|
|
// Compare against Globals.NewObjectId, which is set to string.Empty
|
|
|
|
objectId = context.GetObjectId ();
|
|
|
|
if (type.IsValueType) {
|
|
if (!string.IsNullOrEmpty (objectId))
|
|
throw new SerializationException (SR.GetString (SR.ValueTypeCannotHaveId, DataContract.GetClrTypeFullName(type)));
|
|
}
|
|
object innerValueRead = null;
|
|
if (nullables != 0)
|
|
shouldAssignNullableValue = true;
|
|
|
|
if (primitiveContract != null && primitiveContract.UnderlyingType != Globals.TypeOfObject) {
|
|
value = primitiveContract.XmlFormatReaderMethod.Invoke (xmlReader, new object [0]);
|
|
if (!type.IsValueType)
|
|
context.AddNewObject (value);
|
|
}
|
|
else
|
|
value = InternalDeserialize (type, name);
|
|
} else {
|
|
// Deserialize ref
|
|
if (type.IsValueType)
|
|
throw new SerializationException (SR.GetString (SR.ValueTypeCannotHaveRef, DataContract.GetClrTypeFullName (type)));
|
|
else
|
|
value = CodeInterpreter.ConvertValue (context.GetExistingObject (objectId, type, name, string.Empty), Globals.TypeOfObject, type);
|
|
}
|
|
|
|
if (shouldAssignNullableValue) {
|
|
if (objectId != Globals.NullObjectId)
|
|
value = WrapNullableObject (type, value, valueType, nullables);
|
|
}
|
|
}
|
|
else
|
|
value = InternalDeserialize (type, name);
|
|
|
|
return value;
|
|
}
|
|
|
|
object InternalDeserialize (Type type, string name)
|
|
{
|
|
Type declaredType = type.IsPointer ? Globals.TypeOfReflectionPointer : type;
|
|
var obj = context.InternalDeserialize (xmlReader, DataContract.GetId (declaredType.TypeHandle), declaredType.TypeHandle, name, string.Empty);
|
|
|
|
if (type.IsPointer)
|
|
// wow, there is no way to convert void* to object in strongly typed way...
|
|
return JsonFormatGeneratorStatics.UnboxPointer.Invoke (null, new object [] {obj});
|
|
else
|
|
return CodeInterpreter.ConvertValue (obj, Globals.TypeOfObject, type);
|
|
}
|
|
|
|
object WrapNullableObject (Type innerType, object innerValue, Type outerType, int nullables)
|
|
{
|
|
var outerValue = innerValue;
|
|
for (int i = 1; i < nullables; i++) {
|
|
Type type = Globals.TypeOfNullable.MakeGenericType (innerType);
|
|
outerValue = Activator.CreateInstance (type, new object[] { outerValue });
|
|
innerType = type;
|
|
}
|
|
return Activator.CreateInstance (outerType, new object[] { outerValue });
|
|
}
|
|
|
|
|
|
void ReadCollection (CollectionDataContract collectionContract)
|
|
{
|
|
Type type = collectionContract.UnderlyingType;
|
|
Type itemType = collectionContract.ItemType;
|
|
bool isArray = (collectionContract.Kind == CollectionKind.Array);
|
|
|
|
ConstructorInfo constructor = collectionContract.Constructor;
|
|
|
|
if (type.IsInterface) {
|
|
switch (collectionContract.Kind) {
|
|
case CollectionKind.GenericDictionary:
|
|
type = Globals.TypeOfDictionaryGeneric.MakeGenericType (itemType.GetGenericArguments ());
|
|
constructor = type.GetConstructor (BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic, null, Globals.EmptyTypeArray, null);
|
|
break;
|
|
case CollectionKind.Dictionary:
|
|
type = Globals.TypeOfHashtable;
|
|
constructor = type.GetConstructor (BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic, null, Globals.EmptyTypeArray, null);
|
|
break;
|
|
case CollectionKind.Collection:
|
|
case CollectionKind.GenericCollection:
|
|
case CollectionKind.Enumerable:
|
|
case CollectionKind.GenericEnumerable:
|
|
case CollectionKind.List:
|
|
case CollectionKind.GenericList:
|
|
type = itemType.MakeArrayType ();
|
|
isArray = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!isArray) {
|
|
if (type.IsValueType)
|
|
// FIXME: this is not what the original code does.
|
|
objectLocal = FormatterServices.GetUninitializedObject (type);
|
|
else {
|
|
objectLocal = constructor.Invoke (new object [0]);
|
|
context.AddNewObject (objectLocal);
|
|
}
|
|
}
|
|
|
|
bool canReadSimpleDictionary = collectionContract.Kind == CollectionKind.Dictionary ||
|
|
collectionContract.Kind == CollectionKind.GenericDictionary;
|
|
|
|
bool readSimpleDictionary = canReadSimpleDictionary & context.UseSimpleDictionaryFormat;
|
|
if (readSimpleDictionary)
|
|
ReadSimpleDictionary (collectionContract, itemType);
|
|
else {
|
|
string objectId = context.GetObjectId ();
|
|
|
|
bool canReadPrimitiveArray = false, readResult = false;
|
|
if (isArray && TryReadPrimitiveArray (itemType, out readResult))
|
|
canReadPrimitiveArray = true;
|
|
|
|
if (!canReadPrimitiveArray) {
|
|
object growingCollection = null;
|
|
if (isArray)
|
|
growingCollection = Array.CreateInstance (itemType, 32);
|
|
|
|
int i = 0;
|
|
// FIXME: I cannot find i++ part, but without that it won't work as expected.
|
|
for (; i < int.MaxValue; i++) {
|
|
if (IsStartElement (this.itemName, this.emptyDictionaryString)) {
|
|
context.IncrementItemCount (1);
|
|
object value = ReadCollectionItem (collectionContract, itemType);
|
|
if (isArray) {
|
|
MethodInfo ensureArraySizeMethod = XmlFormatGeneratorStatics.EnsureArraySizeMethod.MakeGenericMethod (itemType);
|
|
growingCollection = ensureArraySizeMethod.Invoke (null, new object [] {growingCollection, i});
|
|
((Array) growingCollection).SetValue (value, i);
|
|
} else {
|
|
StoreCollectionValue (objectLocal, itemType, value, collectionContract);
|
|
}
|
|
}
|
|
else if (IsEndElement ())
|
|
break;
|
|
else
|
|
HandleUnexpectedItemInCollection (ref i);
|
|
}
|
|
|
|
if (isArray) {
|
|
MethodInfo trimArraySizeMethod = XmlFormatGeneratorStatics.TrimArraySizeMethod.MakeGenericMethod (itemType);
|
|
objectLocal = trimArraySizeMethod.Invoke (null, new object [] {growingCollection, i});
|
|
context.AddNewObjectWithId (objectId, objectLocal);
|
|
}
|
|
}
|
|
else
|
|
context.AddNewObjectWithId (objectId, objectLocal);
|
|
}
|
|
}
|
|
|
|
void ReadSimpleDictionary (CollectionDataContract collectionContract, Type keyValueType)
|
|
{
|
|
Type[] keyValueTypes = keyValueType.GetGenericArguments ();
|
|
Type keyType = keyValueTypes [0];
|
|
Type valueType = keyValueTypes [1];
|
|
|
|
int keyTypeNullableDepth = 0;
|
|
Type keyTypeOriginal = keyType;
|
|
while (keyType.IsGenericType && keyType.GetGenericTypeDefinition () == Globals.TypeOfNullable) {
|
|
keyTypeNullableDepth++;
|
|
keyType = keyType.GetGenericArguments () [0];
|
|
}
|
|
|
|
ClassDataContract keyValueDataContract = (ClassDataContract)collectionContract.ItemContract;
|
|
DataContract keyDataContract = keyValueDataContract.Members [0].MemberTypeContract;
|
|
|
|
KeyParseMode keyParseMode = KeyParseMode.Fail;
|
|
|
|
if (keyType == Globals.TypeOfString || keyType == Globals.TypeOfObject) {
|
|
keyParseMode = KeyParseMode.AsString;
|
|
} else if (keyType.IsEnum) {
|
|
keyParseMode = KeyParseMode.UsingParseEnum;
|
|
} else if (keyDataContract.ParseMethod != null) {
|
|
keyParseMode = KeyParseMode.UsingCustomParse;
|
|
}
|
|
|
|
if (keyParseMode == KeyParseMode.Fail) {
|
|
ThrowSerializationException (
|
|
SR.GetString (
|
|
SR.KeyTypeCannotBeParsedInSimpleDictionary,
|
|
DataContract.GetClrTypeFullName (collectionContract.UnderlyingType),
|
|
DataContract.GetClrTypeFullName (keyType)));
|
|
} else {
|
|
XmlNodeType nodeType;
|
|
|
|
while ((nodeType = xmlReader.MoveToContent ()) != XmlNodeType.EndElement) {
|
|
if (nodeType != XmlNodeType.Element)
|
|
ThrowUnexpectedStateException (XmlNodeType.Element);
|
|
|
|
context.IncrementItemCount (1);
|
|
|
|
var jsonMemberName = XmlObjectSerializerReadContextComplexJson.GetJsonMemberName (xmlReader);
|
|
object key = null;
|
|
|
|
if (keyParseMode == KeyParseMode.AsString)
|
|
key = jsonMemberName;
|
|
else if (keyParseMode == KeyParseMode.UsingParseEnum)
|
|
key = Enum.Parse (keyType, jsonMemberName);
|
|
else if (keyParseMode == KeyParseMode.UsingCustomParse)
|
|
key = keyDataContract.ParseMethod.Invoke (null, new object [] {jsonMemberName});
|
|
|
|
if (keyTypeNullableDepth > 0) {
|
|
var keyOriginal = WrapNullableObject (keyType, key, valueType, keyTypeNullableDepth);
|
|
key = keyOriginal;
|
|
}
|
|
|
|
var value = ReadValue (valueType, String.Empty);
|
|
collectionContract.AddMethod.Invoke (objectLocal, new object[] {key, value});
|
|
}
|
|
}
|
|
}
|
|
|
|
void ReadGetOnlyCollection (CollectionDataContract collectionContract)
|
|
{
|
|
Type type = collectionContract.UnderlyingType;
|
|
Type itemType = collectionContract.ItemType;
|
|
bool isArray = (collectionContract.Kind == CollectionKind.Array);
|
|
int size = 0;
|
|
|
|
objectLocal = context.GetCollectionMember ();
|
|
|
|
bool canReadSimpleDictionary =
|
|
collectionContract.Kind == CollectionKind.Dictionary ||
|
|
collectionContract.Kind == CollectionKind.GenericDictionary;
|
|
|
|
bool readSimple = canReadSimpleDictionary && context.UseSimpleDictionaryFormat;
|
|
if (readSimple) {
|
|
if (objectLocal == null)
|
|
XmlObjectSerializerReadContext.ThrowNullValueReturnedForGetOnlyCollectionException (type);
|
|
else {
|
|
ReadSimpleDictionary(collectionContract, itemType);
|
|
context.CheckEndOfArray (xmlReader, size, this.itemName, emptyDictionaryString);
|
|
}
|
|
} else {
|
|
|
|
//check that items are actually going to be deserialized into the collection
|
|
if (IsStartElement (this.itemName, this.emptyDictionaryString)) {
|
|
if (objectLocal == null)
|
|
XmlObjectSerializerReadContext.ThrowNullValueReturnedForGetOnlyCollectionException (type);
|
|
else {
|
|
size = 0;
|
|
if (isArray)
|
|
size = ((Array) objectLocal).Length;
|
|
for (int i = 0; i < int.MaxValue;) {
|
|
if (IsStartElement (this.itemName, this.emptyDictionaryString)) {
|
|
context.IncrementItemCount (1);
|
|
var value = ReadCollectionItem (collectionContract, itemType);
|
|
if (isArray) {
|
|
if (size == i)
|
|
XmlObjectSerializerReadContext.ThrowArrayExceededSizeException (size, type);
|
|
else
|
|
((Array) objectLocal).SetValue (value, i);
|
|
} else {
|
|
StoreCollectionValue (objectLocal, itemType, value, collectionContract);
|
|
}
|
|
}
|
|
else if (IsEndElement())
|
|
break;
|
|
else
|
|
HandleUnexpectedItemInCollection (ref i);
|
|
}
|
|
context.CheckEndOfArray (xmlReader, size, this.itemName, this.emptyDictionaryString);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
bool TryReadPrimitiveArray (Type itemType, out bool readResult)
|
|
{
|
|
readResult = false;
|
|
PrimitiveDataContract primitiveContract = PrimitiveDataContract.GetPrimitiveDataContract (itemType);
|
|
if (primitiveContract == null)
|
|
return false;
|
|
|
|
string readArrayMethod = null;
|
|
switch (Type.GetTypeCode (itemType))
|
|
{
|
|
case TypeCode.Boolean:
|
|
readArrayMethod = "TryReadBooleanArray";
|
|
break;
|
|
case TypeCode.Decimal:
|
|
readArrayMethod = "TryReadDecimalArray";
|
|
break;
|
|
case TypeCode.Int32:
|
|
readArrayMethod = "TryReadInt32Array";
|
|
break;
|
|
case TypeCode.Int64:
|
|
readArrayMethod = "TryReadInt64Array";
|
|
break;
|
|
case TypeCode.Single:
|
|
readArrayMethod = "TryReadSingleArray";
|
|
break;
|
|
case TypeCode.Double:
|
|
readArrayMethod = "TryReadDoubleArray";
|
|
break;
|
|
case TypeCode.DateTime:
|
|
readArrayMethod = "TryReadJsonDateTimeArray";
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
if (readArrayMethod != null) {
|
|
var mi = typeof (JsonReaderDelegator).GetMethod (readArrayMethod, Globals.ScanAllMembers);
|
|
var args = new object [] {context, itemName, emptyDictionaryString, -1, objectLocal};
|
|
readResult = (bool) mi.Invoke ((JsonReaderDelegator) xmlReader, args);
|
|
objectLocal = args.Last ();
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
object ReadCollectionItem (CollectionDataContract collectionContract, Type itemType)
|
|
{
|
|
if (collectionContract.Kind == CollectionKind.Dictionary || collectionContract.Kind == CollectionKind.GenericDictionary) {
|
|
context.ResetAttributes ();
|
|
var revisedContract = XmlObjectSerializerWriteContextComplexJson.GetRevisedItemContract (collectionContract.ItemContract);
|
|
var v = DataContractJsonSerializer.ReadJsonValue (revisedContract, xmlReader, context);
|
|
return CodeInterpreter.ConvertValue (v, Globals.TypeOfObject, itemType);
|
|
}
|
|
else
|
|
return ReadValue (itemType, JsonGlobals.itemString);
|
|
}
|
|
|
|
void StoreCollectionValue (object collection, Type valueType, object value, CollectionDataContract collectionContract)
|
|
{
|
|
if (collectionContract.Kind == CollectionKind.GenericDictionary || collectionContract.Kind == CollectionKind.Dictionary) {
|
|
ClassDataContract keyValuePairContract = DataContract.GetDataContract (valueType) as ClassDataContract;
|
|
if (keyValuePairContract == null)
|
|
Fx.Assert ("Failed to create contract for KeyValuePair type");
|
|
DataMember keyMember = keyValuePairContract.Members [0];
|
|
DataMember valueMember = keyValuePairContract.Members [1];
|
|
object pkey = CodeInterpreter.GetMember (keyMember.MemberInfo, value);
|
|
object pvalue = CodeInterpreter.GetMember (valueMember.MemberInfo, value);
|
|
|
|
try {
|
|
collectionContract.AddMethod.Invoke (collection, new object [] {pkey, pvalue});
|
|
} catch (TargetInvocationException ex) {
|
|
if (ex.InnerException != null)
|
|
throw ex.InnerException;
|
|
else
|
|
throw;
|
|
}
|
|
}
|
|
else
|
|
collectionContract.AddMethod.Invoke (collection, new object [] {value});
|
|
}
|
|
|
|
void HandleUnexpectedItemInCollection (ref int iterator)
|
|
{
|
|
if (IsStartElement ()) {
|
|
context.SkipUnknownElement (xmlReader);
|
|
iterator--;
|
|
}
|
|
else
|
|
throw XmlObjectSerializerReadContext.CreateUnexpectedStateException (XmlNodeType.Element, xmlReader);
|
|
}
|
|
|
|
bool IsStartElement(XmlDictionaryString name, XmlDictionaryString ns)
|
|
{
|
|
return xmlReader.IsStartElement (name, ns);
|
|
}
|
|
|
|
bool IsStartElement()
|
|
{
|
|
return xmlReader.IsStartElement ();
|
|
}
|
|
|
|
bool IsEndElement ()
|
|
{
|
|
return xmlReader.NodeType == XmlNodeType.EndElement;
|
|
}
|
|
|
|
void ThrowUnexpectedStateException (XmlNodeType expectedState)
|
|
{
|
|
throw XmlObjectSerializerReadContext.CreateUnexpectedStateException (expectedState, xmlReader);
|
|
}
|
|
|
|
void ThrowSerializationException (string msg, params object [] values)
|
|
{
|
|
if (values != null && values.Length > 0)
|
|
msg = string.Format (msg, values);
|
|
throw new SerializationException (msg);
|
|
}
|
|
|
|
enum KeyParseMode
|
|
{
|
|
Fail,
|
|
AsString,
|
|
UsingParseEnum,
|
|
UsingCustomParse
|
|
}
|
|
}
|
|
}
|