You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			791 lines
		
	
	
		
			30 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
		
		
			
		
	
	
			791 lines
		
	
	
		
			30 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
|   | //------------------------------------------------------------------------------ | ||
|  | // <copyright file="XmlSchemas.cs" company="Microsoft"> | ||
|  | //     Copyright (c) Microsoft Corporation.  All rights reserved. | ||
|  | // </copyright> | ||
|  | // <owner current="true" primary="true">[....]</owner>                                                                 | ||
|  | //------------------------------------------------------------------------------ | ||
|  | 
 | ||
|  | namespace System.Xml.Serialization { | ||
|  | 
 | ||
|  |     using System.Collections; | ||
|  |     using System.Collections.Generic; | ||
|  |     using System.IO; | ||
|  |     using System; | ||
|  |     using System.Globalization; | ||
|  |     using System.ComponentModel; | ||
|  |     using System.Xml.Serialization; | ||
|  |     using System.Xml.Schema; | ||
|  |     using System.Diagnostics; | ||
|  |     using System.Threading; | ||
|  |     using System.Security.Permissions; | ||
|  |     using System.Security.Policy; | ||
|  |     using System.Security; | ||
|  |     using System.Net; | ||
|  | 
 | ||
|  |     /// <include file='doc\XmlSchemas.uex' path='docs/doc[@for="XmlSchemas"]/*' /> | ||
|  |     /// <internalonly/> | ||
|  |     /// <devdoc> | ||
|  |     ///    <para>[To be supplied.]</para> | ||
|  |     /// </devdoc> | ||
|  |     public class XmlSchemas : CollectionBase, IEnumerable<XmlSchema> { | ||
|  |         XmlSchemaSet schemaSet; | ||
|  |         Hashtable references; | ||
|  |         SchemaObjectCache cache; // cached schema top-level items | ||
|  |         bool shareTypes; | ||
|  |         Hashtable mergedSchemas; | ||
|  |         internal Hashtable delayedSchemas = new Hashtable(); | ||
|  |         bool isCompiled = false; | ||
|  |         static volatile XmlSchema xsd; | ||
|  |         static volatile XmlSchema xml; | ||
|  | 
 | ||
|  |         /// <include file='doc\XmlSchemas.uex' path='docs/doc[@for="XmlSchemas.this"]/*' /> | ||
|  |         /// <devdoc> | ||
|  |         ///    <para>[To be supplied.]</para> | ||
|  |         /// </devdoc> | ||
|  |         public XmlSchema this[int index] { | ||
|  |             get { return (XmlSchema)List[index]; } | ||
|  |             set { List[index] = value; } | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <include file='doc\XmlSchemas.uex' path='docs/doc[@for="XmlSchemas.this1"]/*' /> | ||
|  |         /// <devdoc> | ||
|  |         ///    <para>[To be supplied.]</para> | ||
|  |         /// </devdoc> | ||
|  |         public XmlSchema this[string ns] { | ||
|  |             get { | ||
|  |                 IList values = (IList)SchemaSet.Schemas(ns); | ||
|  |                 if (values.Count == 0) | ||
|  |                     return null; | ||
|  |                 if (values.Count == 1) | ||
|  |                     return (XmlSchema)values[0]; | ||
|  | 
 | ||
|  |                 throw new InvalidOperationException(Res.GetString(Res.XmlSchemaDuplicateNamespace, ns)); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <include file='doc\XmlSchemas.uex' path='docs/doc[@for="XmlSchemas.GetSchemas"]/*' /> | ||
|  |         /// <devdoc> | ||
|  |         ///    <para>[To be supplied.]</para> | ||
|  |         /// </devdoc> | ||
|  |         public IList GetSchemas(string ns) { | ||
|  |             return (IList)SchemaSet.Schemas(ns); | ||
|  |         } | ||
|  | 
 | ||
|  |         internal SchemaObjectCache Cache { | ||
|  |             get { | ||
|  |                 if (cache == null) | ||
|  |                     cache = new SchemaObjectCache(); | ||
|  |                 return cache;  | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         internal Hashtable MergedSchemas { | ||
|  |             get { | ||
|  |                 if (mergedSchemas == null) | ||
|  |                     mergedSchemas = new Hashtable(); | ||
|  |                 return mergedSchemas; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         internal Hashtable References { | ||
|  |             get {  | ||
|  |                 if (references == null) | ||
|  |                     references = new Hashtable(); | ||
|  |                 return references; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         internal XmlSchemaSet SchemaSet { | ||
|  |             get {  | ||
|  |                 if (schemaSet == null) { | ||
|  |                     schemaSet = new XmlSchemaSet(); | ||
|  |                     schemaSet.XmlResolver = null; | ||
|  |                     schemaSet.ValidationEventHandler += new ValidationEventHandler(IgnoreCompileErrors); | ||
|  |                 } | ||
|  |                 return schemaSet; | ||
|  |             } | ||
|  |         } | ||
|  |         internal int Add(XmlSchema schema, bool delay) { | ||
|  |             if (delay) { | ||
|  |                 if (delayedSchemas[schema] == null) | ||
|  |                     delayedSchemas.Add(schema, schema); | ||
|  |                 return -1; | ||
|  |             } | ||
|  |             else { | ||
|  |                 return Add(schema); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <include file='doc\XmlSchemas.uex' path='docs/doc[@for="XmlSchemas.Add"]/*' /> | ||
|  |         /// <devdoc> | ||
|  |         ///    <para>[To be supplied.]</para> | ||
|  |         /// </devdoc> | ||
|  |         public int Add(XmlSchema schema) { | ||
|  |             if (List.Contains(schema)) | ||
|  |                 return List.IndexOf(schema); | ||
|  |             return List.Add(schema); | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <include file='doc\XmlSchemas.uex' path='docs/doc[@for="XmlSchemas.Add"]/*' /> | ||
|  |         /// <devdoc> | ||
|  |         ///    <para>[To be supplied.]</para> | ||
|  |         /// </devdoc> | ||
|  |         public int Add(XmlSchema schema, Uri baseUri) { | ||
|  |             if (List.Contains(schema)) | ||
|  |                 return List.IndexOf(schema); | ||
|  |             if (baseUri != null) | ||
|  |                 schema.BaseUri = baseUri; | ||
|  |             return List.Add(schema); | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <include file='doc\XmlSchemas.uex' path='docs/doc[@for="XmlSchemas.Add1"]/*' /> | ||
|  |         /// <devdoc> | ||
|  |         ///    <para>[To be supplied.]</para> | ||
|  |         /// </devdoc> | ||
|  |         public void Add(XmlSchemas schemas) { | ||
|  |             foreach (XmlSchema schema in schemas) { | ||
|  |                 Add(schema); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <include file='doc\XmlSchemas.uex' path='docs/doc[@for="XmlSchemas.AddReference"]/*' /> | ||
|  |         /// <devdoc> | ||
|  |         ///    <para>[To be supplied.]</para> | ||
|  |         /// </devdoc> | ||
|  |         public void AddReference(XmlSchema schema) { | ||
|  |             References[schema] = schema; | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <include file='doc\XmlSchemas.uex' path='docs/doc[@for="XmlSchemas.Insert"]/*' /> | ||
|  |         /// <devdoc> | ||
|  |         ///    <para>[To be supplied.]</para> | ||
|  |         /// </devdoc> | ||
|  |         public void Insert(int index, XmlSchema schema) { | ||
|  |             List.Insert(index, schema); | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <include file='doc\XmlSchemas.uex' path='docs/doc[@for="XmlSchemas.IndexOf"]/*' /> | ||
|  |         /// <devdoc> | ||
|  |         ///    <para>[To be supplied.]</para> | ||
|  |         /// </devdoc> | ||
|  |         public int IndexOf(XmlSchema schema) { | ||
|  |             return List.IndexOf(schema); | ||
|  |         } | ||
|  |          | ||
|  |         /// <include file='doc\XmlSchemas.uex' path='docs/doc[@for="XmlSchemas.Contains"]/*' /> | ||
|  |         /// <devdoc> | ||
|  |         ///    <para>[To be supplied.]</para> | ||
|  |         /// </devdoc> | ||
|  |         public bool Contains(XmlSchema schema) { | ||
|  |             return List.Contains(schema); | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <include file='doc\XmlSchemas.uex' path='docs/doc[@for="XmlSchemas.Contains1"]/*' /> | ||
|  |         /// <devdoc> | ||
|  |         ///    <para>[To be supplied.]</para> | ||
|  |         /// </devdoc> | ||
|  |         public bool Contains(string targetNamespace) { | ||
|  |             return SchemaSet.Contains(targetNamespace); | ||
|  |         } | ||
|  |          | ||
|  |         /// <include file='doc\XmlSchemas.uex' path='docs/doc[@for="XmlSchemas.Remove"]/*' /> | ||
|  |         /// <devdoc> | ||
|  |         ///    <para>[To be supplied.]</para> | ||
|  |         /// </devdoc> | ||
|  |         public void Remove(XmlSchema schema) { | ||
|  |             List.Remove(schema); | ||
|  |         } | ||
|  |          | ||
|  |         /// <include file='doc\XmlSchemas.uex' path='docs/doc[@for="XmlSchemas.CopyTo"]/*' /> | ||
|  |         /// <devdoc> | ||
|  |         ///    <para>[To be supplied.]</para> | ||
|  |         /// </devdoc> | ||
|  |         public void CopyTo(XmlSchema[] array, int index) { | ||
|  |             List.CopyTo(array, index); | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <include file='doc\XmlSchemas.uex' path='docs/doc[@for="XmlSchemas.OnInsert"]/*' /> | ||
|  |         /// <devdoc> | ||
|  |         ///    <para>[To be supplied.]</para> | ||
|  |         /// </devdoc> | ||
|  |         protected override void OnInsert(int index, object value) { | ||
|  |             AddName((XmlSchema)value); | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <include file='doc\XmlSchemas.uex' path='docs/doc[@for="XmlSchemas.OnRemove"]/*' /> | ||
|  |         /// <devdoc> | ||
|  |         ///    <para>[To be supplied.]</para> | ||
|  |         /// </devdoc> | ||
|  |         protected override void OnRemove(int index, object value) { | ||
|  |             RemoveName((XmlSchema)value); | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <include file='doc\XmlSchemas.uex' path='docs/doc[@for="XmlSchemas.OnClear"]/*' /> | ||
|  |         /// <devdoc> | ||
|  |         ///    <para>[To be supplied.]</para> | ||
|  |         /// </devdoc> | ||
|  |         protected override void OnClear() { | ||
|  |             schemaSet = null; | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <include file='doc\XmlSchemas.uex' path='docs/doc[@for="XmlSchemas.OnSet"]/*' /> | ||
|  |         /// <devdoc> | ||
|  |         ///    <para>[To be supplied.]</para> | ||
|  |         /// </devdoc> | ||
|  |         protected override void OnSet(int index, object oldValue, object newValue) { | ||
|  |             RemoveName((XmlSchema)oldValue); | ||
|  |             AddName((XmlSchema)newValue); | ||
|  |         } | ||
|  | 
 | ||
|  |         void AddName(XmlSchema schema) { | ||
|  |             if (isCompiled) throw new InvalidOperationException(Res.GetString(Res.XmlSchemaCompiled)); | ||
|  |             if (SchemaSet.Contains(schema)) | ||
|  |                 SchemaSet.Reprocess(schema); | ||
|  |             else { | ||
|  |                 Prepare(schema); | ||
|  |                 SchemaSet.Add(schema); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         void Prepare(XmlSchema schema) { | ||
|  |             // need to remove illegal <import> externals; | ||
|  |             ArrayList removes = new ArrayList(); | ||
|  |             string ns = schema.TargetNamespace; | ||
|  |             foreach (XmlSchemaExternal external in schema.Includes) { | ||
|  |                 if (external is XmlSchemaImport) { | ||
|  |                     if (ns == ((XmlSchemaImport)external).Namespace) { | ||
|  |                         removes.Add(external); | ||
|  |                     } | ||
|  |                 } | ||
|  |             } | ||
|  |             foreach(XmlSchemaObject o in removes) { | ||
|  |                 schema.Includes.Remove(o); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         void RemoveName(XmlSchema schema) { | ||
|  |             SchemaSet.Remove(schema); | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <include file='doc\XmlSchemas.uex' path='docs/doc[@for="XmlSchemas.Find"]/*' /> | ||
|  |         /// <devdoc> | ||
|  |         ///    <para>[To be supplied.]</para> | ||
|  |         /// </devdoc> | ||
|  |         public object Find(XmlQualifiedName name, Type type) { | ||
|  |             return Find(name, type, true); | ||
|  |         } | ||
|  |         internal object Find(XmlQualifiedName name, Type type, bool checkCache) { | ||
|  |             if (!IsCompiled) { | ||
|  |                  foreach (XmlSchema schema in List) { | ||
|  |                     Preprocess(schema); | ||
|  |                 } | ||
|  |             } | ||
|  |             IList values = (IList)SchemaSet.Schemas(name.Namespace); | ||
|  |             if (values == null) return null; | ||
|  | 
 | ||
|  |             foreach (XmlSchema schema in values) { | ||
|  |                 Preprocess(schema); | ||
|  | 
 | ||
|  |                 XmlSchemaObject ret = null; | ||
|  |                 if (typeof(XmlSchemaType).IsAssignableFrom(type)) { | ||
|  |                     ret = schema.SchemaTypes[name]; | ||
|  |                     if (ret == null || !type.IsAssignableFrom(ret.GetType())) { | ||
|  |                         continue; | ||
|  |                     } | ||
|  |                 } | ||
|  |                 else if (type == typeof(XmlSchemaGroup)) { | ||
|  |                     ret = schema.Groups[name]; | ||
|  |                 } | ||
|  |                 else if (type == typeof(XmlSchemaAttributeGroup)) { | ||
|  |                     ret = schema.AttributeGroups[name]; | ||
|  |                 } | ||
|  |                 else if (type == typeof(XmlSchemaElement)) { | ||
|  |                     ret = schema.Elements[name]; | ||
|  |                 } | ||
|  |                 else if (type == typeof(XmlSchemaAttribute)) { | ||
|  |                     ret = schema.Attributes[name]; | ||
|  |                 } | ||
|  |                 else if (type == typeof(XmlSchemaNotation)) { | ||
|  |                     ret = schema.Notations[name]; | ||
|  |                 } | ||
|  | #if DEBUG | ||
|  |             else { | ||
|  |                 // use exception in the place of Debug.Assert to avoid throwing asserts from a server process such as aspnet_ewp.exe | ||
|  |                 throw new InvalidOperationException(Res.GetString(Res.XmlInternalErrorDetails, "XmlSchemas.Find: Invalid object type " + type.FullName)); | ||
|  |             } | ||
|  | #endif | ||
|  | 
 | ||
|  |                 if (ret != null && shareTypes && checkCache && !IsReference(ret)) | ||
|  |                     ret = Cache.AddItem(ret, name, this); | ||
|  |                 if (ret != null) { | ||
|  |                     return ret; | ||
|  |                 } | ||
|  |             } | ||
|  |             return null; | ||
|  |         } | ||
|  | 
 | ||
|  |         IEnumerator<XmlSchema> IEnumerable<XmlSchema>.GetEnumerator() { | ||
|  |             return new XmlSchemaEnumerator(this); | ||
|  |         } | ||
|  | 
 | ||
|  |         internal static void Preprocess(XmlSchema schema) { | ||
|  |             if (!schema.IsPreprocessed) { | ||
|  |                 try { | ||
|  |                     XmlNameTable nameTable = new System.Xml.NameTable(); | ||
|  |                     Preprocessor prep = new Preprocessor(nameTable, new SchemaNames(nameTable), null); | ||
|  |                     prep.SchemaLocations = new Hashtable(); | ||
|  |                     prep.Execute(schema, schema.TargetNamespace, false); | ||
|  |                 } | ||
|  |                 catch(XmlSchemaException e) { | ||
|  |                     throw CreateValidationException(e, e.Message); | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <include file='doc\XmlSchemas.uex' path='docs/doc[@for="XmlSchemas.IsDataSet"]/*' /> | ||
|  |         /// <devdoc> | ||
|  |         ///    <para>[To be supplied.]</para> | ||
|  |         /// </devdoc> | ||
|  |         public static bool IsDataSet(XmlSchema schema) { | ||
|  |             foreach (XmlSchemaObject o in schema.Items) { | ||
|  |                 if (o is XmlSchemaElement) { | ||
|  |                     XmlSchemaElement e = (XmlSchemaElement)o; | ||
|  |                     if (e.UnhandledAttributes != null) { | ||
|  |                         foreach (XmlAttribute a in e.UnhandledAttributes) { | ||
|  |                             if (a.LocalName == "IsDataSet" && a.NamespaceURI == "urn:schemas-microsoft-com:xml-msdata") { | ||
|  |                                 // currently the msdata:IsDataSet uses its own format for the boolean values | ||
|  |                                 if (a.Value == "True" || a.Value == "true" || a.Value == "1") return true; | ||
|  |                             } | ||
|  |                         } | ||
|  |                     } | ||
|  |                 } | ||
|  |             } | ||
|  |             return false; | ||
|  |         } | ||
|  | 
 | ||
|  |         void Merge(XmlSchema schema) { | ||
|  |             if (MergedSchemas[schema] != null) | ||
|  |                 return; | ||
|  |             IList originals = (IList)SchemaSet.Schemas(schema.TargetNamespace); | ||
|  |             if (originals != null && originals.Count > 0) { | ||
|  |                 MergedSchemas.Add(schema, schema); | ||
|  |                 Merge(originals, schema); | ||
|  |             } | ||
|  |             else { | ||
|  |                 Add(schema);  | ||
|  |                 MergedSchemas.Add(schema, schema); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         void AddImport(IList schemas, string ns) { | ||
|  |             foreach(XmlSchema s in schemas) { | ||
|  |                 bool add = true; | ||
|  |                 foreach (XmlSchemaExternal external in s.Includes) { | ||
|  |                     if (external is XmlSchemaImport && ((XmlSchemaImport)external).Namespace == ns) { | ||
|  |                         add = false; | ||
|  |                         break; | ||
|  |                     } | ||
|  |                 } | ||
|  |                 if (add) { | ||
|  |                     XmlSchemaImport import = new XmlSchemaImport(); | ||
|  |                     import.Namespace = ns; | ||
|  |                     s.Includes.Add(import); | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         void Merge(IList originals, XmlSchema schema) { | ||
|  |             foreach (XmlSchema s in originals) { | ||
|  |                 if (schema == s) { | ||
|  |                     return; | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             foreach (XmlSchemaExternal external in schema.Includes) { | ||
|  |                 if (external is XmlSchemaImport) { | ||
|  |                     external.SchemaLocation = null; | ||
|  |                     if (external.Schema != null) { | ||
|  |                         Merge(external.Schema); | ||
|  |                     } | ||
|  |                     else { | ||
|  |                         AddImport(originals, ((XmlSchemaImport)external).Namespace); | ||
|  |                     } | ||
|  |                 } | ||
|  |                 else { | ||
|  |                     if (external.Schema == null) { | ||
|  |                         // we do not process includes or redefines by the schemaLocation | ||
|  |                         if (external.SchemaLocation != null) { | ||
|  |                             throw new InvalidOperationException(Res.GetString(Res.XmlSchemaIncludeLocation, this.GetType().Name, external.SchemaLocation)); | ||
|  |                         } | ||
|  |                     } | ||
|  |                     else { | ||
|  |                         external.SchemaLocation = null; | ||
|  |                         Merge(originals, external.Schema); | ||
|  |                     } | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             // bring all included items to the parent schema; | ||
|  |             bool[] matchedItems = new bool[schema.Items.Count]; | ||
|  |             int count = 0; | ||
|  |             for (int i = 0; i < schema.Items.Count; i++) { | ||
|  |                 XmlSchemaObject o  = schema.Items[i]; | ||
|  |                 XmlSchemaObject dest = Find(o, originals); | ||
|  |                 if (dest != null) { | ||
|  |                     if (!Cache.Match(dest, o, shareTypes)) { | ||
|  |                         //  | ||
|  |                         Debug.WriteLineIf(DiagnosticsSwitches.XmlSerialization.TraceVerbose, "XmlSerialization::Failed to Merge " + MergeFailedMessage(o, dest, schema.TargetNamespace) | ||
|  |                             + "' Plase Compare hash:\r\n" + Cache.looks[dest] + "\r\n" + Cache.looks[o]); | ||
|  |                         throw new InvalidOperationException(MergeFailedMessage(o, dest, schema.TargetNamespace)); | ||
|  |                     } | ||
|  |                     matchedItems[i] = true; | ||
|  |                     count++; | ||
|  |                 } | ||
|  |             } | ||
|  |             if (count != schema.Items.Count) { | ||
|  |                 XmlSchema destination = (XmlSchema)originals[0]; | ||
|  |                 for (int i = 0; i < schema.Items.Count; i++) { | ||
|  |                     if (!matchedItems[i]) { | ||
|  |                         destination.Items.Add(schema.Items[i]); | ||
|  |                     } | ||
|  |                 } | ||
|  |                 destination.IsPreprocessed = false; | ||
|  |                 Preprocess(destination); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         static string ItemName(XmlSchemaObject o) { | ||
|  |             if (o is XmlSchemaNotation) { | ||
|  |                 return ((XmlSchemaNotation)o).Name; | ||
|  |             } | ||
|  |             else if (o is XmlSchemaGroup) { | ||
|  |                 return ((XmlSchemaGroup)o).Name; | ||
|  |             } | ||
|  |             else if (o is XmlSchemaElement) { | ||
|  |                 return ((XmlSchemaElement)o).Name; | ||
|  |             } | ||
|  |             else if (o is XmlSchemaType) { | ||
|  |                 return ((XmlSchemaType)o).Name; | ||
|  |             } | ||
|  |             else if (o is XmlSchemaAttributeGroup) { | ||
|  |                 return ((XmlSchemaAttributeGroup)o).Name; | ||
|  |             } | ||
|  |             else if (o is XmlSchemaAttribute) { | ||
|  |                 return ((XmlSchemaAttribute)o).Name; | ||
|  |             } | ||
|  |             return null; | ||
|  |         } | ||
|  | 
 | ||
|  |         internal static XmlQualifiedName GetParentName(XmlSchemaObject item) { | ||
|  |             while (item.Parent != null) { | ||
|  |                 if (item.Parent is XmlSchemaType) { | ||
|  |                     XmlSchemaType type = (XmlSchemaType)item.Parent; | ||
|  |                     if (type.Name != null && type.Name.Length != 0) { | ||
|  |                         return type.QualifiedName; | ||
|  |                     } | ||
|  |                 } | ||
|  |                 item = item.Parent; | ||
|  |             } | ||
|  |             return XmlQualifiedName.Empty; | ||
|  |         } | ||
|  | 
 | ||
|  |         static string GetSchemaItem(XmlSchemaObject o, string ns, string details) { | ||
|  |             if (o == null) { | ||
|  |                 return null; | ||
|  |             } | ||
|  |             while (o.Parent != null && !(o.Parent is XmlSchema)) { | ||
|  |                 o = o.Parent; | ||
|  |             } | ||
|  |             if (ns == null || ns.Length == 0) { | ||
|  |                 XmlSchemaObject tmp = o; | ||
|  |                 while (tmp.Parent != null) { | ||
|  |                     tmp = tmp.Parent; | ||
|  |                 } | ||
|  |                 if (tmp is XmlSchema) { | ||
|  |                     ns = ((XmlSchema)tmp).TargetNamespace; | ||
|  |                 } | ||
|  |             } | ||
|  |             string item = null; | ||
|  |             if (o is XmlSchemaNotation) { | ||
|  |                 item = Res.GetString(Res.XmlSchemaNamedItem, ns, "notation", ((XmlSchemaNotation)o).Name, details); | ||
|  |             } | ||
|  |             else if (o is XmlSchemaGroup) { | ||
|  |                 item = Res.GetString(Res.XmlSchemaNamedItem, ns, "group", ((XmlSchemaGroup)o).Name, details); | ||
|  |             } | ||
|  |             else if (o is XmlSchemaElement) { | ||
|  |                 XmlSchemaElement e = ((XmlSchemaElement)o); | ||
|  |                 if (e.Name == null || e.Name.Length == 0) { | ||
|  |                     XmlQualifiedName parentName = XmlSchemas.GetParentName(o); | ||
|  |                     // Element reference '{0}' declared in schema type '{1}' from namespace '{2}' | ||
|  |                     item = Res.GetString(Res.XmlSchemaElementReference, e.RefName.ToString(), parentName.Name, parentName.Namespace); | ||
|  |                 } | ||
|  |                 else { | ||
|  |                     item = Res.GetString(Res.XmlSchemaNamedItem, ns, "element", e.Name, details); | ||
|  |                 } | ||
|  |             } | ||
|  |             else if (o is XmlSchemaType) { | ||
|  |                 item = Res.GetString(Res.XmlSchemaNamedItem, ns, o.GetType() == typeof(XmlSchemaSimpleType) ? "simpleType" : "complexType", ((XmlSchemaType)o).Name, null); | ||
|  |             } | ||
|  |             else if (o is XmlSchemaAttributeGroup) { | ||
|  |                 item = Res.GetString(Res.XmlSchemaNamedItem, ns, "attributeGroup", ((XmlSchemaAttributeGroup)o).Name, details); | ||
|  |             } | ||
|  |             else if (o is XmlSchemaAttribute) { | ||
|  |                 XmlSchemaAttribute a = ((XmlSchemaAttribute)o); | ||
|  |                 if (a.Name == null || a.Name.Length == 0) { | ||
|  |                     XmlQualifiedName parentName = XmlSchemas.GetParentName(o); | ||
|  |                     // Attribure reference '{0}' declared in schema type '{1}' from namespace '{2}' | ||
|  |                     return Res.GetString(Res.XmlSchemaAttributeReference, a.RefName.ToString(), parentName.Name, parentName.Namespace); | ||
|  |                 } | ||
|  |                 else  { | ||
|  |                     item = Res.GetString(Res.XmlSchemaNamedItem, ns, "attribute", a.Name, details); | ||
|  |                 } | ||
|  | 
 | ||
|  |             } | ||
|  |             else if (o is XmlSchemaContent) { | ||
|  |                 XmlQualifiedName parentName = XmlSchemas.GetParentName(o); | ||
|  |                 // Check content definition of schema type '{0}' from namespace '{1}'. {2} | ||
|  |                 item = Res.GetString(Res.XmlSchemaContentDef, parentName.Name, parentName.Namespace, null); | ||
|  |             } | ||
|  |             else if (o is XmlSchemaExternal) { | ||
|  |                 string itemType = o is XmlSchemaImport ? "import" : o is XmlSchemaInclude ? "include" : o is XmlSchemaRedefine ? "redefine" : o.GetType().Name; | ||
|  |                 item = Res.GetString(Res.XmlSchemaItem, ns, itemType, details); | ||
|  |             } | ||
|  |             else if (o is XmlSchema) { | ||
|  |                 item = Res.GetString(Res.XmlSchema, ns, details); | ||
|  |             } | ||
|  |             else { | ||
|  |                 item = Res.GetString(Res.XmlSchemaNamedItem, ns, o.GetType().Name, null, details); | ||
|  |             } | ||
|  | 
 | ||
|  |             return item; | ||
|  |         } | ||
|  | 
 | ||
|  |         static string Dump(XmlSchemaObject o) { | ||
|  |             XmlWriterSettings settings = new XmlWriterSettings(); | ||
|  |             settings.OmitXmlDeclaration = true; | ||
|  |             settings.Indent = true; | ||
|  |             XmlSerializer s = new XmlSerializer(o.GetType()); | ||
|  |             StringWriter sw = new StringWriter(CultureInfo.InvariantCulture); | ||
|  |             XmlWriter xmlWriter = XmlWriter.Create(sw, settings); | ||
|  |             XmlSerializerNamespaces ns = new XmlSerializerNamespaces(); | ||
|  |             ns.Add("xs", XmlSchema.Namespace); | ||
|  |             s.Serialize(xmlWriter, o, ns); | ||
|  |             return sw.ToString(); | ||
|  |         } | ||
|  |         static string MergeFailedMessage(XmlSchemaObject src, XmlSchemaObject dest, string ns) { | ||
|  |             string err = Res.GetString(Res.XmlSerializableMergeItem, ns, GetSchemaItem(src, ns, null)); | ||
|  |             err += "\r\n" + Dump(src); | ||
|  |             err += "\r\n" + Dump(dest); | ||
|  |             return err; | ||
|  |         } | ||
|  | 
 | ||
|  |         internal XmlSchemaObject Find(XmlSchemaObject o, IList originals) { | ||
|  |             string name = ItemName(o); | ||
|  |             if (name == null) | ||
|  |                 return null; | ||
|  | 
 | ||
|  |             Type type = o.GetType(); | ||
|  | 
 | ||
|  |             foreach (XmlSchema s in originals) { | ||
|  |                 foreach(XmlSchemaObject item in s.Items) { | ||
|  |                     if (item.GetType() == type && name == ItemName(item)) { | ||
|  |                         return item; | ||
|  |                     } | ||
|  |                 } | ||
|  |             } | ||
|  |             return null; | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <include file='doc\XmlSchemas.uex' path='docs/doc[@for="XmlSchemas.IsCompiled"]/*' /> | ||
|  |         public bool IsCompiled { | ||
|  |             get { return isCompiled; } | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <include file='doc\XmlSchemas.uex' path='docs/doc[@for="XmlSchemas.Compile"]/*' /> | ||
|  |         public void Compile(ValidationEventHandler handler, bool fullCompile) { | ||
|  |             if (isCompiled) | ||
|  |                 return; | ||
|  | 
 | ||
|  |             foreach(XmlSchema s in delayedSchemas.Values) | ||
|  |                 Merge(s); | ||
|  |             delayedSchemas.Clear(); | ||
|  | 
 | ||
|  |             if (fullCompile) { | ||
|  |                 schemaSet = new XmlSchemaSet(); | ||
|  |                 schemaSet.XmlResolver = null; | ||
|  |                 schemaSet.ValidationEventHandler += handler; | ||
|  | 
 | ||
|  |                 foreach (XmlSchema s in References.Values) | ||
|  |                     schemaSet.Add(s); | ||
|  |                 int schemaCount = schemaSet.Count; | ||
|  | 
 | ||
|  |                 foreach (XmlSchema s in List) { | ||
|  |                     if (!SchemaSet.Contains(s)) { | ||
|  |                         schemaSet.Add(s); | ||
|  |                         schemaCount++; | ||
|  |                     } | ||
|  |                 } | ||
|  | 
 | ||
|  |                 if (!SchemaSet.Contains(XmlSchema.Namespace)) { | ||
|  |                     AddReference(XsdSchema); | ||
|  |                     schemaSet.Add(XsdSchema); | ||
|  |                     schemaCount++; | ||
|  |                 } | ||
|  | 
 | ||
|  |                 if (!SchemaSet.Contains(XmlReservedNs.NsXml)) { | ||
|  |                     AddReference(XmlSchema); | ||
|  |                     schemaSet.Add(XmlSchema); | ||
|  |                     schemaCount++; | ||
|  |                 } | ||
|  |                 schemaSet.Compile(); | ||
|  |                 schemaSet.ValidationEventHandler -= handler; | ||
|  |                 isCompiled = schemaSet.IsCompiled && schemaCount == schemaSet.Count; | ||
|  |             } | ||
|  |             else { | ||
|  |                 try { | ||
|  |                     XmlNameTable nameTable = new System.Xml.NameTable(); | ||
|  |                     Preprocessor prep = new Preprocessor(nameTable, new SchemaNames(nameTable), null); | ||
|  |                     prep.XmlResolver = null; | ||
|  |                     prep.SchemaLocations = new Hashtable(); | ||
|  |                     prep.ChameleonSchemas = new Hashtable(); | ||
|  |                     foreach (XmlSchema schema in SchemaSet.Schemas()) { | ||
|  |                         prep.Execute(schema, schema.TargetNamespace, true); | ||
|  |                     } | ||
|  |                 } | ||
|  |                 catch(XmlSchemaException e) { | ||
|  |                     throw CreateValidationException(e, e.Message); | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         internal static Exception CreateValidationException(XmlSchemaException exception, string message) { | ||
|  |             XmlSchemaObject source = exception.SourceSchemaObject; | ||
|  |             if (exception.LineNumber == 0 && exception.LinePosition == 0) { | ||
|  |                 throw new InvalidOperationException(GetSchemaItem(source, null, message), exception); | ||
|  |             } | ||
|  |             else { | ||
|  |                 string ns = null; | ||
|  |                 if (source != null) { | ||
|  |                     while (source.Parent != null) { | ||
|  |                         source = source.Parent; | ||
|  |                     } | ||
|  |                     if (source is XmlSchema) { | ||
|  |                         ns = ((XmlSchema)source).TargetNamespace; | ||
|  |                     } | ||
|  |                 } | ||
|  |                 throw new InvalidOperationException(Res.GetString(Res.XmlSchemaSyntaxErrorDetails, ns, message, exception.LineNumber, exception.LinePosition), exception); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         internal static void IgnoreCompileErrors(object sender, ValidationEventArgs args) { | ||
|  |             return; | ||
|  |         } | ||
|  | 
 | ||
|  |         internal static XmlSchema XsdSchema { | ||
|  |             get { | ||
|  |                 if (xsd == null) { | ||
|  |                     xsd = CreateFakeXsdSchema(XmlSchema.Namespace, "schema"); | ||
|  |                 } | ||
|  |                 return xsd; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         internal static XmlSchema XmlSchema { | ||
|  |             get { | ||
|  |                 if (xml == null) { | ||
|  |                     xml = XmlSchema.Read(new StringReader(xmlSchema), null); | ||
|  |                 } | ||
|  |                 return xml; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         private static XmlSchema CreateFakeXsdSchema(string ns, string name) { | ||
|  |             /* Create fake xsd schema to fool the XmlSchema.Compiler | ||
|  |                 <xsd:schema targetNamespace="http://www.w3.org/2001/XMLSchema" xmlns:xsd="http://www.w3.org/2001/XMLSchema"> | ||
|  |                   <xsd:element name="schema"> | ||
|  |                     <xsd:complexType /> | ||
|  |                   </xsd:element> | ||
|  |                 </xsd:schema> | ||
|  |             */ | ||
|  |             XmlSchema schema = new XmlSchema(); | ||
|  |             schema.TargetNamespace = ns; | ||
|  |             XmlSchemaElement element = new XmlSchemaElement(); | ||
|  |             element.Name = name; | ||
|  |             XmlSchemaComplexType type = new XmlSchemaComplexType(); | ||
|  |             element.SchemaType = type; | ||
|  |             schema.Items.Add(element); | ||
|  |             return schema; | ||
|  |         } | ||
|  | 
 | ||
|  |         internal void SetCache(SchemaObjectCache cache, bool shareTypes) { | ||
|  |             this.shareTypes = shareTypes; | ||
|  |             this.cache = cache; | ||
|  |             if (shareTypes) { | ||
|  |                 cache.GenerateSchemaGraph(this); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         internal bool IsReference(XmlSchemaObject type) { | ||
|  |             XmlSchemaObject parent = type; | ||
|  |             while (parent.Parent != null) { | ||
|  |                 parent = parent.Parent; | ||
|  |             } | ||
|  |             return References.Contains(parent); | ||
|  |         } | ||
|  | 
 | ||
|  |         internal const string xmlSchema = @"<?xml version='1.0' encoding='UTF-8' ?> 
 | ||
|  | <xs:schema targetNamespace='http://www.w3.org/XML/1998/namespace' xmlns:xs='http://www.w3.org/2001/XMLSchema' xml:lang='en'> | ||
|  |  <xs:attribute name='lang' type='xs:language'/> | ||
|  |  <xs:attribute name='space'> | ||
|  |   <xs:simpleType> | ||
|  |    <xs:restriction base='xs:NCName'> | ||
|  |     <xs:enumeration value='default'/> | ||
|  |     <xs:enumeration value='preserve'/> | ||
|  |    </xs:restriction> | ||
|  |   </xs:simpleType> | ||
|  |  </xs:attribute> | ||
|  |  <xs:attribute name='base' type='xs:anyURI'/> | ||
|  |  <xs:attribute name='id' type='xs:ID' /> | ||
|  |  <xs:attributeGroup name='specialAttrs'> | ||
|  |   <xs:attribute ref='xml:base'/> | ||
|  |   <xs:attribute ref='xml:lang'/> | ||
|  |   <xs:attribute ref='xml:space'/> | ||
|  |  </xs:attributeGroup> | ||
|  | </xs:schema>";
 | ||
|  |     } | ||
|  | 
 | ||
|  |     public class XmlSchemaEnumerator : IEnumerator<XmlSchema>, System.Collections.IEnumerator { | ||
|  |         private XmlSchemas list; | ||
|  |         private int idx, end; | ||
|  | 
 | ||
|  |         public XmlSchemaEnumerator(XmlSchemas list) { | ||
|  |             this.list = list; | ||
|  |             this.idx = -1; | ||
|  |             this.end = list.Count - 1; | ||
|  |         } | ||
|  | 
 | ||
|  |         public void Dispose() { | ||
|  |         } | ||
|  | 
 | ||
|  |         public bool MoveNext() { | ||
|  |             if (this.idx >= this.end) | ||
|  |                 return false; | ||
|  | 
 | ||
|  |             this.idx++; | ||
|  |             return true; | ||
|  |         } | ||
|  | 
 | ||
|  |         public XmlSchema Current { | ||
|  |             get { return this.list[this.idx]; } | ||
|  |         } | ||
|  | 
 | ||
|  |         object System.Collections.IEnumerator.Current { | ||
|  |             get { return this.list[this.idx]; } | ||
|  |         } | ||
|  | 
 | ||
|  |         void System.Collections.IEnumerator.Reset() { | ||
|  |             this.idx = -1; | ||
|  |         } | ||
|  |     } | ||
|  | } |