You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			772 lines
		
	
	
		
			30 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			772 lines
		
	
	
		
			30 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| #region Copyright (c) Microsoft Corporation
 | |
| /// <copyright company='Microsoft Corporation'>
 | |
| ///    Copyright (c) Microsoft Corporation. All Rights Reserved.
 | |
| ///    Information Contained Herein is Proprietary and Confidential.
 | |
| /// </copyright>
 | |
| #endregion
 | |
| 
 | |
| using System;
 | |
| using System.Collections.Generic;
 | |
| using System.Diagnostics;
 | |
| using System.Globalization;
 | |
| using System.Reflection;
 | |
| using System.Xml;
 | |
| using System.Xml.Schema;
 | |
| 
 | |
| #if WEB_EXTENSIONS_CODE
 | |
| using System.Web.Resources;
 | |
| #else
 | |
| using Microsoft.VSDesigner.WCF.Resources;
 | |
| #endif
 | |
| 
 | |
| #if WEB_EXTENSIONS_CODE
 | |
| namespace System.Web.Compilation.WCFModel
 | |
| #else
 | |
| namespace Microsoft.VSDesigner.WCFModel
 | |
| #endif
 | |
| {
 | |
|     /// <summary>
 | |
|     /// a utility class to merge schema files, and remove duplicated part
 | |
|     /// </summary>
 | |
|     internal class SchemaMerger
 | |
|     {
 | |
|         // xml serializable attributes
 | |
|         private static Type[] xmlSerializationAttributes = new Type[] {
 | |
|                             typeof(System.Xml.Serialization.XmlElementAttribute),
 | |
|                             typeof(System.Xml.Serialization.XmlAttributeAttribute),
 | |
|                             typeof(System.Xml.Serialization.XmlAnyAttributeAttribute),
 | |
|                             typeof(System.Xml.Serialization.XmlAnyElementAttribute),
 | |
|                             typeof(System.Xml.Serialization.XmlTextAttribute),
 | |
|         };
 | |
| 
 | |
|         // elements in the schema (we don't process annotation node)
 | |
|         private static SchemaTopLevelItemType[] schemaTopLevelItemTypes = new SchemaTopLevelItemType[] {
 | |
|             new SchemaTopLevelItemType(typeof(XmlSchemaType), "type"),
 | |
|             new SchemaTopLevelItemType(typeof(XmlSchemaElement), "element"),
 | |
|             new SchemaTopLevelItemType(typeof(XmlSchemaAttribute), "attribute"),
 | |
|             new SchemaTopLevelItemType(typeof(XmlSchemaGroup), "group"),
 | |
|             new SchemaTopLevelItemType(typeof(XmlSchemaAttributeGroup), "attributeGroup"),
 | |
|         };
 | |
| 
 | |
|         // when properties defined in those types are different, we only report warnings, but not error messages
 | |
|         private static Type[] ignorablePropertyTypes = new Type[] {
 | |
|             typeof(XmlAttribute[]),
 | |
|             typeof(XmlElement[]),
 | |
|             typeof(XmlNode[]),
 | |
|             typeof(XmlSchemaAnnotation),
 | |
|         };
 | |
| 
 | |
|         private readonly static XmlAttribute[] emptyXmlAttributeCollection = new XmlAttribute[0];
 | |
|         private readonly static object[] emptyCollection = new object[0];
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Merge and remove duplicated part from the schema list
 | |
|         /// </summary>
 | |
|         /// <param name="schemaList">schemas with names</param>
 | |
|         /// <param name="importErrors">error messages</param>
 | |
|         /// <param name="duplicatedSchemas">error messages</param>
 | |
|         /// <remarks></remarks>
 | |
|         internal static void MergeSchemas(IEnumerable<XmlSchema> schemaList, IList<ProxyGenerationError> importErrors, out IEnumerable<XmlSchema> duplicatedSchemas)
 | |
|         {
 | |
|             if (schemaList == null)
 | |
|             {
 | |
|                 throw new ArgumentNullException("schemaList");
 | |
|             }
 | |
|             if (importErrors == null)
 | |
|             {
 | |
|                 throw new ArgumentNullException("importErrors");
 | |
|             }
 | |
| 
 | |
|             List<XmlSchema> duplicatedSchemaList = new List<XmlSchema>();
 | |
|             duplicatedSchemas = duplicatedSchemaList;
 | |
| 
 | |
|             // types, elements, groups have their own name space
 | |
|             Dictionary<XmlQualifiedName, XmlSchemaObject>[] knownItemTables = new Dictionary<XmlQualifiedName, XmlSchemaObject>[schemaTopLevelItemTypes.Length];
 | |
|             for (int i = 0; i < schemaTopLevelItemTypes.Length; i++)
 | |
|             {
 | |
|                 knownItemTables[i] = new Dictionary<XmlQualifiedName, XmlSchemaObject>();
 | |
|             }
 | |
| 
 | |
|             foreach (XmlSchema schema in schemaList)
 | |
|             {
 | |
| 
 | |
|                 bool hasNewDefinedItems = false;
 | |
|                 List<XmlSchemaObject> duplicatedItems = new List<XmlSchemaObject>();
 | |
| 
 | |
|                 for (int i = 0; i < schemaTopLevelItemTypes.Length; i++)
 | |
|                 {
 | |
|                     Dictionary<XmlQualifiedName, XmlSchemaObject> knownItemTable = knownItemTables[i];
 | |
|                     int knownItemCount = knownItemTable.Count;
 | |
|                     FindDuplicatedItems(schema, schemaTopLevelItemTypes[i].ItemType, schemaTopLevelItemTypes[i].Name, knownItemTable, duplicatedItems, importErrors);
 | |
|                     if (knownItemTable.Count > knownItemCount)
 | |
|                     {
 | |
|                         hasNewDefinedItems = true;
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 if (duplicatedItems.Count > 0)
 | |
|                 {
 | |
|                     if (!hasNewDefinedItems)
 | |
|                     {
 | |
|                         // remove the whole schema...
 | |
|                         duplicatedSchemaList.Add(schema);
 | |
|                     }
 | |
|                     else
 | |
|                     {
 | |
|                         // remove duplicated items only
 | |
|                         foreach (XmlSchemaObject item in duplicatedItems)
 | |
|                         {
 | |
|                             schema.Items.Remove(item);
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Find duplicated items in a schema
 | |
|         /// </summary>
 | |
|         /// <remarks></remarks>
 | |
|         private static void FindDuplicatedItems(
 | |
|                     XmlSchema schema,
 | |
|                     Type itemType,
 | |
|                     string itemTypeName,
 | |
|                     Dictionary<XmlQualifiedName, XmlSchemaObject> knownItemTable,
 | |
|                     List<XmlSchemaObject> duplicatedItems,
 | |
|                     IList<ProxyGenerationError> importErrors)
 | |
|         {
 | |
| 
 | |
|             string targetNamespace = schema.TargetNamespace;
 | |
|             if (String.IsNullOrEmpty(targetNamespace))
 | |
|             {
 | |
|                 targetNamespace = String.Empty;
 | |
|             }
 | |
| 
 | |
|             foreach (XmlSchemaObject item in schema.Items)
 | |
|             {
 | |
|                 if (itemType.IsInstanceOfType(item))
 | |
|                 {
 | |
| 
 | |
|                     XmlQualifiedName combinedName = new XmlQualifiedName(GetSchemaItemName(item), targetNamespace);
 | |
| 
 | |
|                     XmlSchemaObject originalItem = null;
 | |
|                     if (knownItemTable.TryGetValue(combinedName, out originalItem))
 | |
|                     {
 | |
|                         string differentLocation;
 | |
| 
 | |
|                         if (!AreSchemaObjectsEquivalent(originalItem, item, out differentLocation))
 | |
|                         {
 | |
|                             differentLocation = CombinePath(".", differentLocation);
 | |
|                             importErrors.Add(
 | |
|                                     new ProxyGenerationError(
 | |
|                                         ProxyGenerationError.GeneratorState.MergeMetadata,
 | |
|                                         String.Empty,
 | |
|                                         new InvalidOperationException(
 | |
|                                             String.Format(CultureInfo.CurrentCulture, WCFModelStrings.ReferenceGroup_DuplicatedSchemaItems, itemTypeName, combinedName.ToString(), schema.SourceUri, originalItem.SourceUri, differentLocation)
 | |
|                                         )
 | |
|                                     )
 | |
|                             );
 | |
|                         }
 | |
|                         else if (!String.IsNullOrEmpty(differentLocation))
 | |
|                         {
 | |
|                             // warning: ignorable difference found
 | |
|                             differentLocation = CombinePath(".", differentLocation);
 | |
|                             importErrors.Add(
 | |
|                                     new ProxyGenerationError(
 | |
|                                         ProxyGenerationError.GeneratorState.MergeMetadata,
 | |
|                                         String.Empty,
 | |
|                                         new InvalidOperationException(
 | |
|                                             String.Format(CultureInfo.CurrentCulture, WCFModelStrings.ReferenceGroup_DuplicatedSchemaItemsIgnored, itemTypeName, combinedName.ToString(), schema.SourceUri, originalItem.SourceUri, differentLocation)
 | |
|                                         ),
 | |
|                                         true        // isWarning = true
 | |
|                                     )
 | |
|                             );
 | |
|                         }
 | |
|                         duplicatedItems.Add(item);
 | |
|                     }
 | |
|                     else
 | |
|                     {
 | |
|                         item.SourceUri = schema.SourceUri;
 | |
|                         knownItemTable.Add(combinedName, item);
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Compare two schema objects
 | |
|         /// </summary>
 | |
|         /// <return></return>
 | |
|         /// <remarks>
 | |
|         /// For all those functions, we follow the same pattern:
 | |
|         ///  return false: find not-ignorable difference between them.  differentLocation will contain the path
 | |
|         ///  return true, with empty differentLocation -- no difference found
 | |
|         ///  return true, with non-empty differentLocation -- ignorable difference found under that location
 | |
|         /// </remarks>
 | |
|         private static bool AreSchemaObjectsEquivalent(XmlSchemaObject originalItem, XmlSchemaObject item, out string differentLocation)
 | |
|         {
 | |
|             differentLocation = String.Empty;
 | |
| 
 | |
|             Type itemType = originalItem.GetType();
 | |
|             if (itemType != item.GetType())
 | |
|             {
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|             string ignorableDifferenceLocation = String.Empty;
 | |
| 
 | |
|             PropertyInfo[] properties = itemType.GetProperties();
 | |
|             foreach (PropertyInfo property in properties)
 | |
|             {
 | |
|                 if (IsPersistedProperty(property))
 | |
|                 {
 | |
| 
 | |
|                     bool ignorableProperty = ShouldIgnoreSchemaProperty(property);
 | |
| 
 | |
|                     object originalValue = property.GetValue(originalItem, new object[] { });
 | |
|                     object newValue = property.GetValue(item, new object[] { });
 | |
| 
 | |
|                     if (!CompareSchemaPropertyValues(property, originalValue, newValue, out differentLocation) && !ignorableProperty)
 | |
|                     {
 | |
|                         return false;
 | |
|                     }
 | |
|                     if (String.IsNullOrEmpty(ignorableDifferenceLocation))
 | |
|                     {
 | |
|                         ignorableDifferenceLocation = differentLocation;
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             differentLocation = ignorableDifferenceLocation;
 | |
|             return true;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Compare two property of a schema object
 | |
|         /// </summary>
 | |
|         /// <return>true: if the value are same</return>
 | |
|         /// <remarks></remarks>
 | |
|         private static bool CompareSchemaPropertyValues(PropertyInfo propertyInfo, object originalValue, object newValue, out string differentLocation)
 | |
|         {
 | |
|             differentLocation = String.Empty;
 | |
| 
 | |
|             if (originalValue == null && newValue == null)
 | |
|             {
 | |
|                 return true;
 | |
|             }
 | |
| 
 | |
|             // we create empty collection so a meaningful differentLocation could be generated
 | |
|             if (typeof(XmlAttribute[]) == propertyInfo.PropertyType)
 | |
|             {
 | |
|                 if (originalValue == null)
 | |
|                 {
 | |
|                     originalValue = emptyXmlAttributeCollection;
 | |
|                 }
 | |
|                 if (newValue == null)
 | |
|                 {
 | |
|                     newValue = emptyXmlAttributeCollection;
 | |
|                 }
 | |
| 
 | |
|                 XmlAttribute differentAttribute1, differentAttribute2;
 | |
|                 if (!CompareXmlAttributeCollections((XmlAttribute[])originalValue, (XmlAttribute[])newValue, out differentAttribute1, out differentAttribute2))
 | |
|                 {
 | |
|                     differentLocation = GetSchemaPropertyNameInXml(propertyInfo, differentAttribute1, differentAttribute2);
 | |
|                     return false;
 | |
|                 }
 | |
|                 return true;
 | |
|             }
 | |
| 
 | |
|             if (typeof(System.Collections.ICollection).IsAssignableFrom(propertyInfo.PropertyType))
 | |
|             {
 | |
|                 if (originalValue == null)
 | |
|                 {
 | |
|                     originalValue = emptyCollection;
 | |
|                 }
 | |
|                 if (newValue == null)
 | |
|                 {
 | |
|                     newValue = emptyCollection;
 | |
|                 }
 | |
| 
 | |
|                 object differentItem1, differentItem2;
 | |
|                 if (!CompareSchemaCollections((System.Collections.ICollection)originalValue, (System.Collections.ICollection)newValue, out differentItem1, out differentItem2, out differentLocation))
 | |
|                 {
 | |
|                     differentLocation = CombinePath(GetSchemaPropertyNameInXml(propertyInfo, differentItem1, differentItem2), differentLocation);
 | |
|                     return false;
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     if (!String.IsNullOrEmpty(differentLocation))
 | |
|                     {
 | |
|                         // ignorable difference...
 | |
|                         differentLocation = CombinePath(GetSchemaPropertyNameInXml(propertyInfo, differentItem1, differentItem2), differentLocation);
 | |
|                     }
 | |
|                     return true;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if (originalValue == null || newValue == null)
 | |
|             {
 | |
|                 differentLocation = CombinePath(GetSchemaPropertyNameInXml(propertyInfo, originalValue, newValue), differentLocation);
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|             if (originalValue.GetType() != newValue.GetType())
 | |
|             {
 | |
|                 differentLocation = CombinePath(GetSchemaPropertyNameInXml(propertyInfo, originalValue, newValue), differentLocation);
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|             if (!CompareSchemaValues(originalValue, newValue, out differentLocation))
 | |
|             {
 | |
|                 differentLocation = CombinePath(GetSchemaPropertyNameInXml(propertyInfo, originalValue, newValue), differentLocation);
 | |
|                 return false;
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 // ignorable difference...
 | |
|                 if (!String.IsNullOrEmpty(differentLocation))
 | |
|                 {
 | |
|                     differentLocation = CombinePath(GetSchemaPropertyNameInXml(propertyInfo, originalValue, newValue), differentLocation);
 | |
|                 }
 | |
|                 return true;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Compare two schema values
 | |
|         /// </summary>
 | |
|         /// <return></return>
 | |
|         /// <remarks></remarks>
 | |
|         private static bool CompareSchemaValues(object originalValue, object newValue, out string differentLocation)
 | |
|         {
 | |
|             differentLocation = String.Empty;
 | |
| 
 | |
|             if (originalValue == null || newValue == null)
 | |
|             {
 | |
|                 return (originalValue == null && newValue == null);
 | |
|             }
 | |
| 
 | |
|             if (originalValue.GetType() != newValue.GetType())
 | |
|             {
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|             if (originalValue is XmlSchemaObject)
 | |
|             {
 | |
|                 return AreSchemaObjectsEquivalent((XmlSchemaObject)originalValue, (XmlSchemaObject)newValue, out differentLocation);
 | |
|             }
 | |
| 
 | |
|             if (originalValue is XmlAttribute)
 | |
|             {
 | |
|                 return CompareXmlAttributes((XmlAttribute)originalValue, (XmlAttribute)newValue);
 | |
|             }
 | |
| 
 | |
|             if (originalValue is XmlElement)
 | |
|             {
 | |
|                 return CompareXmlElements((XmlElement)originalValue, (XmlElement)newValue, out differentLocation);
 | |
|             }
 | |
| 
 | |
|             if (originalValue is XmlText)
 | |
|             {
 | |
|                 return CompareXmlTexts((XmlText)originalValue, (XmlText)newValue);
 | |
|             }
 | |
| 
 | |
|             return originalValue.Equals(newValue);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Compare two collections of items
 | |
|         /// </summary>
 | |
|         /// <return></return>
 | |
|         /// <remarks></remarks>
 | |
|         private static bool CompareSchemaCollections(System.Collections.IEnumerable originalCollection, System.Collections.IEnumerable newCollection,
 | |
|                 out object differentItem1, out object differentItem2, out string differentLocation)
 | |
|         {
 | |
| 
 | |
|             differentLocation = String.Empty;
 | |
| 
 | |
|             System.Collections.IEnumerator list1 = originalCollection.GetEnumerator();
 | |
|             System.Collections.IEnumerator list2 = newCollection.GetEnumerator();
 | |
| 
 | |
|             string ignorableDifferenceLocation = String.Empty;
 | |
|             object ignorableDifferenceItem1 = null;
 | |
|             object ignorableDifferenceItem2 = null;
 | |
| 
 | |
|             do
 | |
|             {
 | |
|                 differentItem1 = list1.MoveNext() ? list1.Current : null;
 | |
|                 differentItem2 = list2.MoveNext() ? list2.Current : null;
 | |
| 
 | |
|                 if (!CompareSchemaValues(differentItem1, differentItem2, out differentLocation))
 | |
|                 {
 | |
|                     return false;
 | |
|                 }
 | |
| 
 | |
|                 if (String.IsNullOrEmpty(ignorableDifferenceLocation))
 | |
|                 {
 | |
|                     ignorableDifferenceItem1 = differentItem1;
 | |
|                     ignorableDifferenceItem2 = differentItem2;
 | |
|                     ignorableDifferenceLocation = differentLocation;
 | |
|                 }
 | |
|             }
 | |
|             while (differentItem1 != null && differentItem2 != null);
 | |
| 
 | |
|             Debug.Assert(differentItem1 == null && differentItem2 == null);
 | |
| 
 | |
|             differentLocation = ignorableDifferenceLocation;
 | |
|             differentItem1 = ignorableDifferenceItem1;
 | |
|             differentItem2 = ignorableDifferenceItem2;
 | |
| 
 | |
|             return true;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Compare two attributes
 | |
|         /// </summary>
 | |
|         /// <return></return>
 | |
|         /// <remarks></remarks>
 | |
|         private static bool CompareXmlAttributes(XmlAttribute attribute1, XmlAttribute attribute2)
 | |
|         {
 | |
|             return String.Equals(attribute1.LocalName, attribute2.LocalName, StringComparison.Ordinal) &&
 | |
|                     String.Equals(attribute1.NamespaceURI, attribute2.NamespaceURI, StringComparison.Ordinal) &&
 | |
|                     String.Equals(attribute1.Value, attribute2.Value, StringComparison.Ordinal);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Compare two attribute collections
 | |
|         /// </summary>
 | |
|         /// <return></return>
 | |
|         /// <remarks></remarks>
 | |
|         private static bool CompareXmlAttributeCollections(System.Collections.ICollection attributeCollection1, System.Collections.ICollection attributeCollection2, out XmlAttribute differentAttribute1, out XmlAttribute differentAttribute2)
 | |
|         {
 | |
|             differentAttribute1 = null;
 | |
|             differentAttribute2 = null;
 | |
| 
 | |
|             XmlAttribute[] attributeArray1 = GetSortedAttributeArray(attributeCollection1);
 | |
|             XmlAttribute[] attributeArray2 = GetSortedAttributeArray(attributeCollection2);
 | |
| 
 | |
|             object differentItem1, differentItem2;
 | |
|             string differentLocation;
 | |
|             if (!CompareSchemaCollections(attributeArray1, attributeArray2, out differentItem1, out differentItem2, out differentLocation))
 | |
|             {
 | |
|                 differentAttribute1 = (XmlAttribute)differentItem1;
 | |
|                 differentAttribute2 = (XmlAttribute)differentItem2;
 | |
|                 return false;
 | |
|             }
 | |
|             return true;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// sort XmlAttribute array, so we can compare two collections without being affected by the order
 | |
|         /// </summary>
 | |
|         /// <return></return>
 | |
|         /// <remarks></remarks>
 | |
|         private static XmlAttribute[] GetSortedAttributeArray(System.Collections.ICollection attributeCollection)
 | |
|         {
 | |
|             XmlAttribute[] attributeArray = new XmlAttribute[attributeCollection.Count];
 | |
|             int index = 0;
 | |
|             foreach (XmlAttribute attribute in attributeCollection)
 | |
|             {
 | |
|                 attributeArray[index++] = attribute;
 | |
|             }
 | |
| 
 | |
|             Array.Sort(attributeArray, new AttributeComparer());
 | |
|             return attributeArray;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Compare two elements
 | |
|         /// </summary>
 | |
|         /// <return></return>
 | |
|         /// <remarks></remarks>
 | |
|         private static bool CompareXmlElements(XmlElement element1, XmlElement element2, out string differentLocation)
 | |
|         {
 | |
|             differentLocation = String.Empty;
 | |
| 
 | |
|             if (!String.Equals(element1.LocalName, element2.LocalName, StringComparison.Ordinal) ||
 | |
|                     !String.Equals(element1.NamespaceURI, element2.NamespaceURI, StringComparison.Ordinal))
 | |
|             {
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|             XmlAttribute differentAttribute1, differentAttribute2;
 | |
|             if (!CompareXmlAttributeCollections(element1.Attributes, element2.Attributes, out differentAttribute1, out differentAttribute2))
 | |
|             {
 | |
|                 string attributeName1 = differentAttribute1 != null ? "@" + differentAttribute1.LocalName : String.Empty;
 | |
|                 string attributeName2 = differentAttribute2 != null ? "@" + differentAttribute2.LocalName : String.Empty;
 | |
|                 differentLocation = CombineTwoNames(attributeName1, attributeName2);
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|             object differentChild1, differentChild2;
 | |
|             if (!CompareSchemaCollections(element1.ChildNodes, element2.ChildNodes, out differentChild1, out differentChild2, out differentLocation))
 | |
|             {
 | |
|                 string child1Name = differentChild1 != null ? ((XmlNode)differentChild1).LocalName : String.Empty;
 | |
|                 string child2Name = differentChild2 != null ? ((XmlNode)differentChild2).LocalName : String.Empty;
 | |
|                 differentLocation = CombinePath(CombineTwoNames(child1Name, child2Name), differentLocation);
 | |
|                 return false;
 | |
|             }
 | |
|             return true;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Compare two text nodes
 | |
|         /// </summary>
 | |
|         /// <return></return>
 | |
|         /// <remarks></remarks>
 | |
|         private static bool CompareXmlTexts(XmlText text1, XmlText text2)
 | |
|         {
 | |
|             return String.Equals(text1.Value, text2.Value, StringComparison.Ordinal);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Combine two path (similar to xpath) in error messages
 | |
|         /// </summary>
 | |
|         /// <return></return>
 | |
|         /// <remarks></remarks>
 | |
|         private static string CombinePath(string path1, string path2)
 | |
|         {
 | |
|             if (String.IsNullOrEmpty(path1))
 | |
|             {
 | |
|                 return path2;
 | |
|             }
 | |
|             else if (String.IsNullOrEmpty(path2))
 | |
|             {
 | |
|                 return path1;
 | |
|             }
 | |
| 
 | |
|             return path1 + "/" + path2;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Get Name of a top level schema item
 | |
|         /// </summary>
 | |
|         /// <param name="item"></param>
 | |
|         /// <return></return>
 | |
|         /// <remarks></remarks>
 | |
|         private static string GetSchemaItemName(XmlSchemaObject item)
 | |
|         {
 | |
|             if (item == null)
 | |
|             {
 | |
|                 throw new ArgumentNullException("item");
 | |
|             }
 | |
| 
 | |
|             Type itemType = item.GetType();
 | |
|             PropertyInfo nameProperty = itemType.GetProperty("Name");
 | |
|             if (nameProperty != null)
 | |
|             {
 | |
|                 object nameValue = nameProperty.GetValue(item, new object[] { });
 | |
|                 if (nameValue is string)
 | |
|                 {
 | |
|                     return (string)nameValue;
 | |
|                 }
 | |
|                 return String.Empty;
 | |
|             }
 | |
| 
 | |
|             return String.Empty;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Generate end-user unstandable property name -- we will use name in the schema file, but not name in object model
 | |
|         /// </summary>
 | |
|         /// <return></return>
 | |
|         /// <remarks></remarks>
 | |
|         private static string GetSchemaPropertyNameInXml(PropertyInfo property, object value1, object value2)
 | |
|         {
 | |
|             object[] propertyAttributes = property.GetCustomAttributes(true);
 | |
|             string name = String.Empty;
 | |
| 
 | |
|             if (propertyAttributes != null)
 | |
|             {
 | |
|                 string name1 = GetSchemaPropertyNameInXmlHelper(propertyAttributes, value1);
 | |
|                 string name2 = GetSchemaPropertyNameInXmlHelper(propertyAttributes, value2);
 | |
| 
 | |
|                 name = CombineTwoNames(name1, name2);
 | |
|             }
 | |
| 
 | |
|             if (String.IsNullOrEmpty(name))
 | |
|             {
 | |
|                 Debug.Fail("Why we didn't get a property name with normal routine?");
 | |
|                 name = property.Name;
 | |
|             }
 | |
| 
 | |
|             return name;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Combine names of two properties in error messages
 | |
|         /// </summary>
 | |
|         /// <remarks></remarks>
 | |
|         private static string CombineTwoNames(string name1, string name2)
 | |
|         {
 | |
|             string name = String.Empty;
 | |
|             if (name1.Length > 0)
 | |
|             {
 | |
|                 if (name2.Length > 0)
 | |
|                 {
 | |
|                     if (String.Equals(name1, name2, StringComparison.Ordinal))
 | |
|                     {
 | |
|                         name = name1;
 | |
|                     }
 | |
|                     else
 | |
|                     {
 | |
|                         name = name1 + "|" + name2;
 | |
|                     }
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     name = name1;
 | |
|                 }
 | |
|             }
 | |
|             else if (name2.Length > 0)
 | |
|             {
 | |
|                 name = name2;
 | |
|             }
 | |
|             return name;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// a helper function to generate names
 | |
|         /// </summary>
 | |
|         /// <remarks></remarks>
 | |
|         private static string GetSchemaPropertyNameInXmlHelper(object[] propertyAttributes, object value)
 | |
|         {
 | |
|             if (value != null)
 | |
|             {
 | |
|                 foreach (object attribute in propertyAttributes)
 | |
|                 {
 | |
|                     if (attribute is System.Xml.Serialization.XmlAttributeAttribute)
 | |
|                     {
 | |
|                         return "@" + ((System.Xml.Serialization.XmlAttributeAttribute)attribute).AttributeName;
 | |
|                     }
 | |
|                     if (attribute is System.Xml.Serialization.XmlElementAttribute)
 | |
|                     {
 | |
|                         System.Xml.Serialization.XmlElementAttribute elementAttribute = (System.Xml.Serialization.XmlElementAttribute)attribute;
 | |
|                         Type elementType = elementAttribute.Type;
 | |
|                         if (elementType == null || elementType.IsInstanceOfType(value))
 | |
|                         {
 | |
|                             if (value is XmlSchemaObject)
 | |
|                             {
 | |
|                                 string itemName = GetSchemaItemName((XmlSchemaObject)value);
 | |
|                                 if (itemName.Length > 0)
 | |
|                                 {
 | |
|                                     return String.Format(System.Globalization.CultureInfo.InvariantCulture, "{0}[@name='{1}']", elementAttribute.ElementName, itemName);
 | |
|                                 }
 | |
|                             }
 | |
|                             return elementAttribute.ElementName;
 | |
|                         }
 | |
|                     }
 | |
|                     if (attribute is System.Xml.Serialization.XmlAnyAttributeAttribute)
 | |
|                     {
 | |
|                         if (value is XmlAttribute)
 | |
|                         {
 | |
|                             return "@" + ((XmlAttribute)value).LocalName;
 | |
|                         }
 | |
|                     }
 | |
|                     if (attribute is System.Xml.Serialization.XmlAnyElementAttribute)
 | |
|                     {
 | |
|                         if (value is XmlElement)
 | |
|                         {
 | |
|                             return ((XmlElement)value).LocalName;
 | |
|                         }
 | |
|                     }
 | |
|                     if (attribute is System.Xml.Serialization.XmlTextAttribute)
 | |
|                     {
 | |
|                         if (value is XmlText)
 | |
|                         {
 | |
|                             return ((XmlText)value).Name;
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             return String.Empty;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Check whether a property is persisted with XmlSerialization
 | |
|         /// </summary>
 | |
|         /// <param name="property"></param>
 | |
|         /// <return></return>
 | |
|         /// <remarks></remarks>
 | |
|         private static bool IsPersistedProperty(PropertyInfo property)
 | |
|         {
 | |
|             object[] propertyAttributes = property.GetCustomAttributes(true);
 | |
|             if (propertyAttributes != null)
 | |
|             {
 | |
|                 foreach (object attribute in propertyAttributes)
 | |
|                 {
 | |
|                     foreach (Type serializationAttibuteType in xmlSerializationAttributes)
 | |
|                     {
 | |
|                         if (serializationAttibuteType.IsInstanceOfType(attribute))
 | |
|                         {
 | |
|                             return true;
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// check whether we should report warning but not error messages, when the property is different
 | |
|         /// </summary>
 | |
|         /// <remarks></remarks>
 | |
|         private static bool ShouldIgnoreSchemaProperty(PropertyInfo property)
 | |
|         {
 | |
|             Type propertyType = property.PropertyType;
 | |
|             foreach (Type ignoreableType in ignorablePropertyTypes)
 | |
|             {
 | |
|                 if (propertyType == ignoreableType || propertyType.IsSubclassOf(ignoreableType))
 | |
|                 {
 | |
|                     return true;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             // special case constraints...
 | |
|             if (String.Equals(property.Name, "Constraints", StringComparison.Ordinal))
 | |
|             {
 | |
|                 return true;
 | |
|             }
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// a helper structure to hold top level items we want to scan
 | |
|         /// </summary>
 | |
|         /// <remarks></remarks>
 | |
|         private struct SchemaTopLevelItemType
 | |
|         {
 | |
|             public Type ItemType;
 | |
|             public string Name;
 | |
| 
 | |
|             public SchemaTopLevelItemType(Type itemType, string name)
 | |
|             {
 | |
|                 this.ItemType = itemType;
 | |
|                 this.Name = name;
 | |
|             }
 | |
|         };
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Helper class to compare two XmlAttributes
 | |
|         /// </summary>
 | |
|         /// <remarks></remarks>
 | |
|         private class AttributeComparer : System.Collections.Generic.IComparer<XmlAttribute>
 | |
|         {
 | |
| 
 | |
|             public int Compare(System.Xml.XmlAttribute x, System.Xml.XmlAttribute y)
 | |
|             {
 | |
|                 int namespaceResult = String.Compare(x.NamespaceURI, y.NamespaceURI, StringComparison.Ordinal);
 | |
|                 if (namespaceResult != 0)
 | |
|                 {
 | |
|                     return namespaceResult;
 | |
|                 }
 | |
| 
 | |
|                 return String.Compare(x.Name, y.Name, StringComparison.Ordinal);
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| 
 |