You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			348 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			348 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| //----------------------------------------------------------------
 | |
| // Copyright (c) Microsoft Corporation.  All rights reserved.
 | |
| //----------------------------------------------------------------
 | |
| 
 | |
| namespace System.Runtime.Serialization.Json
 | |
| {
 | |
|     using System.Collections.Generic;
 | |
|     using System.Runtime;
 | |
|     using System.Runtime.Serialization;
 | |
|     using System.Security;
 | |
|     using System.Reflection;
 | |
| #if !MONO
 | |
|     using System.ServiceModel;
 | |
| #endif
 | |
|     using System.Xml;
 | |
| 
 | |
| #if USE_REFEMIT
 | |
|     public class JsonDataContract
 | |
| #else
 | |
|     class JsonDataContract
 | |
| #endif
 | |
|     {
 | |
|         [Fx.Tag.SecurityNote(Critical = "Holds instance of CriticalHelper which keeps state that is cached statically for serialization."
 | |
|             + "Static fields are marked SecurityCritical or readonly to prevent data from being modified or leaked to other components in appdomain.")]
 | |
|         [SecurityCritical]
 | |
|         JsonDataContractCriticalHelper helper;
 | |
| 
 | |
|         [Fx.Tag.SecurityNote(Critical = "Initializes SecurityCritical field 'helper'.",
 | |
|             Safe = "Doesn't leak anything.")]
 | |
|         [SecuritySafeCritical]
 | |
|         protected JsonDataContract(DataContract traditionalDataContract)
 | |
|         {
 | |
|             this.helper = new JsonDataContractCriticalHelper(traditionalDataContract);
 | |
|         }
 | |
| 
 | |
|         [Fx.Tag.SecurityNote(Critical = "Initializes SecurityCritical field 'helper'.",
 | |
|             Safe = "Doesn't leak anything.")]
 | |
|         [SecuritySafeCritical]
 | |
|         protected JsonDataContract(JsonDataContractCriticalHelper helper)
 | |
|         {
 | |
|             this.helper = helper;
 | |
|         }
 | |
| 
 | |
|         internal virtual string TypeName
 | |
|         {
 | |
|             get { return null; }
 | |
|         }
 | |
| 
 | |
|         protected JsonDataContractCriticalHelper Helper
 | |
|         {
 | |
|             [Fx.Tag.SecurityNote(Critical = "Holds instance of CriticalHelper which keeps state that is cached statically for serialization."
 | |
|                 + "Static fields are marked SecurityCritical or readonly to prevent data from being modified or leaked to other components in appdomain.")]
 | |
|             [SecurityCritical]
 | |
|             get { return helper; }
 | |
|         }
 | |
| 
 | |
|         protected DataContract TraditionalDataContract
 | |
|         {
 | |
|             [Fx.Tag.SecurityNote(Critical = "Fetches the critical TraditionalDataContract from the helper.",
 | |
|                 Safe = "TraditionalDataContract only needs to be protected for write.")]
 | |
|             [SecuritySafeCritical]
 | |
|             get { return this.helper.TraditionalDataContract; }
 | |
|         }
 | |
| 
 | |
|         Dictionary<XmlQualifiedName, DataContract> KnownDataContracts
 | |
|         {
 | |
|             [Fx.Tag.SecurityNote(Critical = "Fetches the critical KnownDataContracts from the helper.",
 | |
|                 Safe = "KnownDataContracts only needs to be protected for write.")]
 | |
|             [SecuritySafeCritical]
 | |
|             get { return this.helper.KnownDataContracts; }
 | |
|         }
 | |
| 
 | |
|         [Fx.Tag.SecurityNote(Critical = "Fetches the critical JsonDataContract from the helper.",
 | |
|             Safe = "JsonDataContract only needs to be protected for write.")]
 | |
|         [SecuritySafeCritical]
 | |
|         public static JsonDataContract GetJsonDataContract(DataContract traditionalDataContract)
 | |
|         {
 | |
|             return JsonDataContractCriticalHelper.GetJsonDataContract(traditionalDataContract);
 | |
|         }
 | |
| 
 | |
|         public object ReadJsonValue(XmlReaderDelegator jsonReader, XmlObjectSerializerReadContextComplexJson context)
 | |
|         {
 | |
|             PushKnownDataContracts(context);
 | |
|             object deserializedObject = ReadJsonValueCore(jsonReader, context);
 | |
|             PopKnownDataContracts(context);
 | |
|             return deserializedObject;
 | |
|         }
 | |
| 
 | |
|         public virtual object ReadJsonValueCore(XmlReaderDelegator jsonReader, XmlObjectSerializerReadContextComplexJson context)
 | |
|         {
 | |
|             return TraditionalDataContract.ReadXmlValue(jsonReader, context);
 | |
|         }
 | |
| 
 | |
|         public void WriteJsonValue(XmlWriterDelegator jsonWriter, object obj, XmlObjectSerializerWriteContextComplexJson context, RuntimeTypeHandle declaredTypeHandle)
 | |
|         {
 | |
|             PushKnownDataContracts(context);
 | |
|             WriteJsonValueCore(jsonWriter, obj, context, declaredTypeHandle);
 | |
|             PopKnownDataContracts(context);
 | |
|         }
 | |
| 
 | |
|         public virtual void WriteJsonValueCore(XmlWriterDelegator jsonWriter, object obj, XmlObjectSerializerWriteContextComplexJson context, RuntimeTypeHandle declaredTypeHandle)
 | |
|         {
 | |
|             TraditionalDataContract.WriteXmlValue(jsonWriter, obj, context);
 | |
|         }
 | |
| 
 | |
|         protected static object HandleReadValue(object obj, XmlObjectSerializerReadContext context)
 | |
|         {
 | |
|             context.AddNewObject(obj);
 | |
|             return obj;
 | |
|         }
 | |
| 
 | |
|         protected static bool TryReadNullAtTopLevel(XmlReaderDelegator reader)
 | |
|         {
 | |
|             while (reader.MoveToAttribute(JsonGlobals.typeString) && (reader.Value == JsonGlobals.nullString))
 | |
|             {
 | |
|                 reader.Skip();
 | |
|                 reader.MoveToElement();
 | |
|                 return true;
 | |
|             }
 | |
| 
 | |
|             reader.MoveToElement();
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         protected void PopKnownDataContracts(XmlObjectSerializerContext context)
 | |
|         {
 | |
|             if (KnownDataContracts != null)
 | |
|             {
 | |
|                 context.scopedKnownTypes.Pop();
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         protected void PushKnownDataContracts(XmlObjectSerializerContext context)
 | |
|         {
 | |
|             if (KnownDataContracts != null)
 | |
|             {
 | |
|                 context.scopedKnownTypes.Push(KnownDataContracts);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         [Fx.Tag.SecurityNote(Critical = "Holds all state used for (de)serializing types."
 | |
|             + "Since the data is cached statically, we lock down access to it.")]
 | |
| #if !NO_SECURITY_ATTRIBUTES
 | |
| #pragma warning disable 618 // have not moved to the v4 security model yet
 | |
|         [SecurityCritical(SecurityCriticalScope.Everything)]
 | |
| #pragma warning restore 618
 | |
| #endif
 | |
| #if USE_REFEMIT
 | |
|         public class JsonDataContractCriticalHelper
 | |
| #else
 | |
|         internal class JsonDataContractCriticalHelper
 | |
| #endif
 | |
|         {
 | |
| 
 | |
|             static object cacheLock = new object();
 | |
|             static object createDataContractLock = new object();
 | |
| 
 | |
|             static JsonDataContract[] dataContractCache = new JsonDataContract[32];
 | |
|             static int dataContractID = 0;
 | |
| 
 | |
|             static TypeHandleRef typeHandleRef = new TypeHandleRef();
 | |
|             static Dictionary<TypeHandleRef, IntRef> typeToIDCache = new Dictionary<TypeHandleRef, IntRef>(new TypeHandleRefEqualityComparer());
 | |
|             Dictionary<XmlQualifiedName, DataContract> knownDataContracts;
 | |
|             DataContract traditionalDataContract;
 | |
|             string typeName;
 | |
| 
 | |
|             internal JsonDataContractCriticalHelper(DataContract traditionalDataContract)
 | |
|             {
 | |
|                 this.traditionalDataContract = traditionalDataContract;
 | |
|                 AddCollectionItemContractsToKnownDataContracts();
 | |
|                 this.typeName = string.IsNullOrEmpty(traditionalDataContract.Namespace.Value) ? traditionalDataContract.Name.Value : string.Concat(traditionalDataContract.Name.Value, JsonGlobals.NameValueSeparatorString, XmlObjectSerializerWriteContextComplexJson.TruncateDefaultDataContractNamespace(traditionalDataContract.Namespace.Value));
 | |
|             }
 | |
| 
 | |
|             internal Dictionary<XmlQualifiedName, DataContract> KnownDataContracts
 | |
|             {
 | |
|                 get { return this.knownDataContracts; }
 | |
|             }
 | |
| 
 | |
|             internal DataContract TraditionalDataContract
 | |
|             {
 | |
|                 get { return this.traditionalDataContract; }
 | |
|             }
 | |
| 
 | |
|             internal virtual string TypeName
 | |
|             {
 | |
|                 get { return this.typeName; }
 | |
|             }
 | |
| 
 | |
|             public static JsonDataContract GetJsonDataContract(DataContract traditionalDataContract)
 | |
|             {
 | |
|                 int id = JsonDataContractCriticalHelper.GetId(traditionalDataContract.UnderlyingType.TypeHandle);
 | |
|                 JsonDataContract dataContract = dataContractCache[id];
 | |
|                 if (dataContract == null)
 | |
|                 {
 | |
|                     dataContract = CreateJsonDataContract(id, traditionalDataContract);
 | |
|                     dataContractCache[id] = dataContract;
 | |
|                 }
 | |
|                 return dataContract;
 | |
|             }
 | |
| 
 | |
|             internal static int GetId(RuntimeTypeHandle typeHandle)
 | |
|             {
 | |
|                 lock (cacheLock)
 | |
|                 {
 | |
|                     IntRef id;
 | |
|                     typeHandleRef.Value = typeHandle;
 | |
|                     if (!typeToIDCache.TryGetValue(typeHandleRef, out id))
 | |
|                     {
 | |
|                         int value = dataContractID++;
 | |
|                         if (value >= dataContractCache.Length)
 | |
|                         {
 | |
|                             int newSize = (value < Int32.MaxValue / 2) ? value * 2 : Int32.MaxValue;
 | |
|                             if (newSize <= value)
 | |
|                             {
 | |
|                                 Fx.Assert("DataContract cache overflow");
 | |
|                                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new SerializationException(System.Runtime.Serialization.SR.GetString(System.Runtime.Serialization.SR.DataContractCacheOverflow)));
 | |
|                             }
 | |
|                             Array.Resize<JsonDataContract>(ref dataContractCache, newSize);
 | |
|                         }
 | |
|                         id = new IntRef(value);
 | |
|                         try
 | |
|                         {
 | |
|                             typeToIDCache.Add(new TypeHandleRef(typeHandle), id);
 | |
|                         }
 | |
|                         catch (Exception ex)
 | |
|                         {
 | |
|                             if (Fx.IsFatal(ex))
 | |
|                             {
 | |
|                                 throw;
 | |
|                             }
 | |
|                             throw DiagnosticUtility.ExceptionUtility.ThrowHelperFatal(ex.Message, ex);
 | |
|                         }
 | |
|                     }
 | |
|                     return id.Value;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             static JsonDataContract CreateJsonDataContract(int id, DataContract traditionalDataContract)
 | |
|             {
 | |
|                 lock (createDataContractLock)
 | |
|                 {
 | |
|                     JsonDataContract dataContract = dataContractCache[id];
 | |
|                     if (dataContract == null)
 | |
|                     {
 | |
|                         Type traditionalDataContractType = traditionalDataContract.GetType();
 | |
|                         if (traditionalDataContractType == typeof(ObjectDataContract))
 | |
|                         {
 | |
|                             dataContract = new JsonObjectDataContract(traditionalDataContract);
 | |
|                         }
 | |
|                         else if (traditionalDataContractType == typeof(StringDataContract))
 | |
|                         {
 | |
|                             dataContract = new JsonStringDataContract((StringDataContract)traditionalDataContract);
 | |
|                         }
 | |
|                         else if (traditionalDataContractType == typeof(UriDataContract))
 | |
|                         {
 | |
|                             dataContract = new JsonUriDataContract((UriDataContract)traditionalDataContract);
 | |
|                         }
 | |
|                         else if (traditionalDataContractType == typeof(QNameDataContract))
 | |
|                         {
 | |
|                             dataContract = new JsonQNameDataContract((QNameDataContract)traditionalDataContract);
 | |
|                         }
 | |
|                         else if (traditionalDataContractType == typeof(ByteArrayDataContract))
 | |
|                         {
 | |
|                             dataContract = new JsonByteArrayDataContract((ByteArrayDataContract)traditionalDataContract);
 | |
|                         }
 | |
|                         else if (traditionalDataContract.IsPrimitive ||
 | |
|                             traditionalDataContract.UnderlyingType == Globals.TypeOfXmlQualifiedName)
 | |
|                         {
 | |
|                             dataContract = new JsonDataContract(traditionalDataContract);
 | |
|                         }
 | |
|                         else if (traditionalDataContractType == typeof(ClassDataContract))
 | |
|                         {
 | |
|                             dataContract = new JsonClassDataContract((ClassDataContract)traditionalDataContract);
 | |
|                         }
 | |
|                         else if (traditionalDataContractType == typeof(EnumDataContract))
 | |
|                         {
 | |
|                             dataContract = new JsonEnumDataContract((EnumDataContract)traditionalDataContract);
 | |
|                         }
 | |
|                         else if ((traditionalDataContractType == typeof(GenericParameterDataContract)) ||
 | |
|                             (traditionalDataContractType == typeof(SpecialTypeDataContract)))
 | |
|                         {
 | |
|                             dataContract = new JsonDataContract(traditionalDataContract);
 | |
|                         }
 | |
|                         else if (traditionalDataContractType == typeof(CollectionDataContract))
 | |
|                         {
 | |
|                             dataContract = new JsonCollectionDataContract((CollectionDataContract)traditionalDataContract);
 | |
|                         }
 | |
|                         else if (traditionalDataContractType == typeof(XmlDataContract))
 | |
|                         {
 | |
|                             dataContract = new JsonXmlDataContract((XmlDataContract)traditionalDataContract);
 | |
|                         }
 | |
|                         else
 | |
|                         {
 | |
|                             throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgument("traditionalDataContract",
 | |
|                                 SR.GetString(SR.JsonTypeNotSupportedByDataContractJsonSerializer, traditionalDataContract.UnderlyingType));
 | |
|                         }
 | |
|                     }
 | |
|                     return dataContract;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             void AddCollectionItemContractsToKnownDataContracts()
 | |
|             {
 | |
|                 if (traditionalDataContract.KnownDataContracts != null)
 | |
|                 {
 | |
|                     foreach (KeyValuePair<XmlQualifiedName, DataContract> knownDataContract in traditionalDataContract.KnownDataContracts)
 | |
|                     {
 | |
|                         if (!object.ReferenceEquals(knownDataContract, null))
 | |
|                         {
 | |
|                             CollectionDataContract collectionDataContract = knownDataContract.Value as CollectionDataContract;
 | |
|                             while (collectionDataContract != null)
 | |
|                             {
 | |
|                                 DataContract itemContract = collectionDataContract.ItemContract;
 | |
|                                 if (knownDataContracts == null)
 | |
|                                 {
 | |
|                                     knownDataContracts = new Dictionary<XmlQualifiedName, DataContract>();
 | |
|                                 }
 | |
| 
 | |
|                                 if (!knownDataContracts.ContainsKey(itemContract.StableName))
 | |
|                                 {
 | |
|                                     knownDataContracts.Add(itemContract.StableName, itemContract);
 | |
|                                 }
 | |
| 
 | |
|                                 if (collectionDataContract.ItemType.IsGenericType
 | |
|                                     && collectionDataContract.ItemType.GetGenericTypeDefinition() == typeof(KeyValue<,>))
 | |
|                                 {
 | |
|                                     DataContract itemDataContract = DataContract.GetDataContract(Globals.TypeOfKeyValuePair.MakeGenericType(collectionDataContract.ItemType.GetGenericArguments()));
 | |
|                                     if (!knownDataContracts.ContainsKey(itemDataContract.StableName))
 | |
|                                     {
 | |
|                                         knownDataContracts.Add(itemDataContract.StableName, itemDataContract);
 | |
|                                     }
 | |
|                                 }
 | |
| 
 | |
|                                 if (!(itemContract is CollectionDataContract))
 | |
|                                 {
 | |
|                                     break;
 | |
|                                 }
 | |
|                                 collectionDataContract = itemContract as CollectionDataContract;
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|     }
 | |
| }
 |