You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			807 lines
		
	
	
		
			27 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			807 lines
		
	
	
		
			27 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| //---------------------------------------------------------------------
 | |
| // <copyright file="SchemaElement.cs" company="Microsoft">
 | |
| //      Copyright (c) Microsoft Corporation.  All rights reserved.
 | |
| // </copyright>
 | |
| //
 | |
| // @owner       [....]
 | |
| // @backupOwner [....]
 | |
| //---------------------------------------------------------------------
 | |
| 
 | |
| namespace System.Data.EntityModel.SchemaObjectModel
 | |
| {
 | |
|     using System;
 | |
|     using System.Collections.Generic;
 | |
|     using System.Data;
 | |
|     using System.Data.Entity;
 | |
|     using System.Data.Metadata.Edm;
 | |
|     using System.Diagnostics;
 | |
|     using System.Globalization;
 | |
|     using System.IO;
 | |
|     using System.Xml;
 | |
|     using System.Xml.Linq;
 | |
| 
 | |
|     /// <summary>
 | |
|     /// Summary description for SchemaElement.
 | |
|     /// </summary>
 | |
|     [DebuggerDisplay("Name={Name}")]
 | |
|     internal abstract class SchemaElement
 | |
|     {
 | |
|         // see http://www.w3.org/TR/2006/REC-xml-names-20060816/
 | |
|         internal const string XmlNamespaceNamespace = "http://www.w3.org/2000/xmlns/";
 | |
| 
 | |
| 
 | |
|         #region Instance Fields
 | |
|         private SchemaElement _parentElement = null;
 | |
|         private Schema _schema = null;
 | |
|         private int _lineNumber = 0;
 | |
|         private int _linePosition = 0;
 | |
|         private string _name = null;
 | |
|         private DocumentationElement _documentation = null;
 | |
| 
 | |
|         private List<MetadataProperty> _otherContent;
 | |
| 
 | |
|         #endregion
 | |
| 
 | |
|         #region Static Fields
 | |
|         /// <summary></summary>
 | |
|         protected const int MaxValueVersionComponent = short.MaxValue;
 | |
|         #endregion
 | |
| 
 | |
|         #region Public Properties
 | |
|         /// <summary>
 | |
|         /// 
 | |
|         /// </summary>
 | |
|         internal  int LineNumber
 | |
|         {
 | |
|             get
 | |
|             {
 | |
|                 return _lineNumber;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// 
 | |
|         /// </summary>
 | |
|         internal  int LinePosition
 | |
|         {
 | |
|             get
 | |
|             {
 | |
|                 return _linePosition;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// 
 | |
|         /// </summary>
 | |
|         public virtual string Name
 | |
|         {
 | |
|             get
 | |
|             {
 | |
|                 return _name;
 | |
|             }
 | |
|             set
 | |
|             {
 | |
|                 _name = value;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// 
 | |
|         /// </summary>
 | |
|         internal  DocumentationElement Documentation
 | |
|         {
 | |
|             get
 | |
|             {
 | |
|                 return _documentation;
 | |
|             }
 | |
|             set
 | |
|             {
 | |
|                 _documentation = value;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// 
 | |
|         /// </summary>
 | |
|         internal  SchemaElement ParentElement
 | |
|         {
 | |
|             get
 | |
|             {
 | |
|                 return _parentElement;
 | |
|             }
 | |
|             private set
 | |
|             {
 | |
|                 _parentElement = value;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// 
 | |
|         /// </summary>
 | |
|         internal Schema Schema
 | |
|         {
 | |
|             get
 | |
|             {
 | |
|                 return _schema;
 | |
|             }
 | |
|             set
 | |
|             {
 | |
|                 _schema = value;
 | |
|             }
 | |
|         }
 | |
|         
 | |
|         /// <summary>
 | |
|         /// 
 | |
|         /// </summary>
 | |
|         public  virtual string FQName
 | |
|         {
 | |
|             get
 | |
|             {
 | |
|                 return Name;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// 
 | |
|         /// </summary>
 | |
|         public virtual string Identity
 | |
|         {
 | |
|             get
 | |
|             {
 | |
|                 return Name;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|        
 | |
|         public List<MetadataProperty> OtherContent
 | |
|         {
 | |
|             get 
 | |
|             {
 | |
|                 if (_otherContent == null)
 | |
|                 {
 | |
|                     _otherContent = new List<MetadataProperty>();
 | |
|                 }
 | |
| 
 | |
|                 return _otherContent; 
 | |
|             }
 | |
|         }
 | |
|         #endregion
 | |
| 
 | |
|         #region Internal Methods
 | |
|         /// <summary>
 | |
|         /// Validates this element and its children
 | |
|         /// </summary>
 | |
|         
 | |
|         internal virtual void Validate()
 | |
|         {
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// 
 | |
|         /// </summary>
 | |
|         /// <param name="errorCode"></param>
 | |
|         /// <param name="severity"></param>
 | |
|         /// <param name="lineNumber"></param>
 | |
|         /// <param name="linePosition"></param>
 | |
|         /// <param name="message"></param>
 | |
|         internal void AddError( ErrorCode errorCode, EdmSchemaErrorSeverity severity, int lineNumber, int linePosition, object message )
 | |
|         {
 | |
|             AddError(errorCode,severity,SchemaLocation,lineNumber,linePosition,message);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// 
 | |
|         /// </summary>
 | |
|         /// <param name="errorCode"></param>
 | |
|         /// <param name="severity"></param>
 | |
|         /// <param name="reader"></param>
 | |
|         /// <param name="message"></param>
 | |
|         internal void AddError( ErrorCode errorCode, EdmSchemaErrorSeverity severity, XmlReader reader, object message )
 | |
|         {
 | |
|             int lineNumber;
 | |
|             int linePosition;
 | |
|             GetPositionInfo(reader, out lineNumber, out linePosition);
 | |
|             AddError(errorCode,severity,SchemaLocation,lineNumber,linePosition,message);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// 
 | |
|         /// </summary>
 | |
|         /// <param name="errorCode"></param>
 | |
|         /// <param name="severity"></param>
 | |
|         /// <param name="message"></param>
 | |
|         internal void AddError( ErrorCode errorCode, EdmSchemaErrorSeverity severity, object message )
 | |
|         {
 | |
|             AddError(errorCode,severity,SchemaLocation,LineNumber,LinePosition,message);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// 
 | |
|         /// </summary>
 | |
|         /// <param name="errorCode"></param>
 | |
|         /// <param name="severity"></param>
 | |
|         /// <param name="element"></param>
 | |
|         /// <param name="message"></param>
 | |
|         internal void AddError( ErrorCode errorCode, EdmSchemaErrorSeverity severity, SchemaElement element, object message )
 | |
|         {
 | |
|             AddError(errorCode,severity,element.Schema.Location,element.LineNumber,element.LinePosition,message);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// 
 | |
|         /// </summary>
 | |
|         /// <param name="reader"></param>
 | |
|         /// <returns></returns>
 | |
|         internal void Parse(XmlReader reader)
 | |
|         {
 | |
|             GetPositionInfo(reader);
 | |
| 
 | |
|             bool hasEndElement = !reader.IsEmptyElement;
 | |
| 
 | |
|             Debug.Assert(reader.NodeType == XmlNodeType.Element);
 | |
|             for ( bool more = reader.MoveToFirstAttribute(); more; more = reader.MoveToNextAttribute() )
 | |
|             {
 | |
|                 ParseAttribute(reader);
 | |
|             }
 | |
|             HandleAttributesComplete();
 | |
| 
 | |
|             bool done = !hasEndElement;
 | |
|             bool skipToNextElement = false;
 | |
|             while ( !done )
 | |
|             {
 | |
|                 if ( skipToNextElement )
 | |
|                 {
 | |
|                     skipToNextElement = false;
 | |
|                     reader.Skip();
 | |
|                     if ( reader.EOF )
 | |
|                         break;
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     if ( !reader.Read() )
 | |
|                         break;
 | |
|                 }
 | |
|                 switch ( reader.NodeType )
 | |
|                 {
 | |
|                     case XmlNodeType.Element:
 | |
|                         skipToNextElement = ParseElement(reader);
 | |
|                         break;
 | |
| 
 | |
|                     case XmlNodeType.EndElement:
 | |
|                     {
 | |
|                         done = true;
 | |
|                         break;
 | |
|                     }
 | |
| 
 | |
|                     case XmlNodeType.CDATA:
 | |
|                     case XmlNodeType.Text:
 | |
|                     case XmlNodeType.SignificantWhitespace:
 | |
|                         ParseText(reader);
 | |
|                         break;
 | |
| 
 | |
|                         // we ignore these childless elements
 | |
|                     case XmlNodeType.Whitespace:
 | |
|                     case XmlNodeType.XmlDeclaration:
 | |
|                     case XmlNodeType.Comment:
 | |
|                     case XmlNodeType.Notation:
 | |
|                     case XmlNodeType.ProcessingInstruction:
 | |
|                     {
 | |
|                         break;
 | |
|                     }
 | |
| 
 | |
|                         // we ignore these elements that can have children
 | |
|                     case XmlNodeType.DocumentType:
 | |
|                     case XmlNodeType.EntityReference:
 | |
|                     {
 | |
|                         skipToNextElement = true;
 | |
|                         break;
 | |
|                     }
 | |
| 
 | |
|                     default:
 | |
|                     {
 | |
|                         AddError( ErrorCode.UnexpectedXmlNodeType, EdmSchemaErrorSeverity.Error, reader,
 | |
|                             System.Data.Entity.Strings.UnexpectedXmlNodeType(reader.NodeType));
 | |
|                         skipToNextElement = true;
 | |
|                         break;
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|             HandleChildElementsComplete();
 | |
|             if ( reader.EOF && reader.Depth > 0 )
 | |
|             {
 | |
|                 AddError( ErrorCode.MalformedXml, EdmSchemaErrorSeverity.Error, 0, 0,
 | |
|                     System.Data.Entity.Strings.MalformedXml(LineNumber,LinePosition));
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Set the current line number and position for an XmlReader
 | |
|         /// </summary>
 | |
|         /// <param name="reader">the reader whose position is desired</param>
 | |
|         internal void GetPositionInfo(XmlReader reader)
 | |
|         {
 | |
|             GetPositionInfo(reader,out _lineNumber,out _linePosition);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Get the current line number and position for an XmlReader
 | |
|         /// </summary>
 | |
|         /// <param name="reader">the reader whose position is desired</param>
 | |
|         /// <param name="lineNumber">the line number</param>
 | |
|         /// <param name="linePosition">the line position</param>
 | |
|         internal static void GetPositionInfo(XmlReader reader, out int lineNumber, out int linePosition)
 | |
|         {
 | |
|             IXmlLineInfo xmlLineInfo = reader as IXmlLineInfo;
 | |
|             if ( xmlLineInfo != null && xmlLineInfo.HasLineInfo() )
 | |
|             {
 | |
|                 lineNumber = xmlLineInfo.LineNumber;
 | |
|                 linePosition = xmlLineInfo.LinePosition;
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 lineNumber = 0;
 | |
|                 linePosition = 0;
 | |
|             }
 | |
|         }
 | |
|         
 | |
|         /// <summary>
 | |
|         /// 
 | |
|         /// </summary>
 | |
|         internal virtual void ResolveTopLevelNames()
 | |
|         {
 | |
|         }
 | |
|         internal virtual void ResolveSecondLevelNames()
 | |
|         {
 | |
|         }
 | |
|         #endregion
 | |
| 
 | |
|         #region Protected Methods
 | |
| 
 | |
|         /// <summary>
 | |
|         /// 
 | |
|         /// </summary>
 | |
|         /// <param name="parentElement"></param>
 | |
|         internal SchemaElement(SchemaElement parentElement)
 | |
|         {
 | |
|             if ( parentElement != null )
 | |
|             {
 | |
|                 ParentElement = parentElement;
 | |
|                 for ( SchemaElement element = parentElement; element != null; element = element.ParentElement )
 | |
|                 {
 | |
|                     Schema schema = element as Schema;
 | |
|                     if ( schema != null )
 | |
|                     {
 | |
|                         Schema = schema;
 | |
|                         break;
 | |
|                     }
 | |
|                 }
 | |
|                 
 | |
|                 if (Schema == null)
 | |
|                 {
 | |
|                     throw EntityUtil.InvalidOperation(System.Data.Entity.Strings.AllElementsMustBeInSchema);
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         internal SchemaElement(SchemaElement parentElement, string name)
 | |
|             : this(parentElement)
 | |
|         {
 | |
|             _name = name;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// 
 | |
|         /// </summary>
 | |
|         protected virtual void HandleAttributesComplete()
 | |
|         {
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// 
 | |
|         /// </summary>
 | |
|         protected virtual void HandleChildElementsComplete()
 | |
|         {
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// 
 | |
|         /// </summary>
 | |
|         /// <param name="reader"></param>
 | |
|         /// <param name="field"></param>
 | |
|         /// <returns></returns>
 | |
|         protected string HandleUndottedNameAttribute(XmlReader reader, string field)
 | |
|         {
 | |
|             string name = field;
 | |
|             Debug.Assert(string.IsNullOrEmpty(field), string.Format(CultureInfo.CurrentCulture, "{0} is already defined", reader.Name));
 | |
| 
 | |
|             bool success = Utils.GetUndottedName(Schema, reader, out name);
 | |
|             if ( !success )
 | |
|                 return name;
 | |
| 
 | |
|             return name;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// 
 | |
|         /// </summary>
 | |
|         /// <param name="reader"></param>
 | |
|         /// <param name="field"></param>
 | |
|         /// <param name="errorMessageId"></param>
 | |
|         /// <returns></returns>
 | |
|         protected ReturnValue<string> HandleDottedNameAttribute(XmlReader reader, string field, Func<object, string> errorFormat)
 | |
|         {
 | |
|             ReturnValue<string> returnValue = new ReturnValue<string>();
 | |
|             Debug.Assert(string.IsNullOrEmpty(field), string.Format(CultureInfo.CurrentCulture, "{0} is already defined", reader.Name));
 | |
| 
 | |
|             string value;
 | |
|             if ( !Utils.GetDottedName(Schema,reader,out value) )
 | |
|                 return returnValue;
 | |
| 
 | |
|             returnValue.Value = value;
 | |
|             return returnValue;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Use to handle an attribute with an int data type
 | |
|         /// </summary>
 | |
|         /// <param name="reader">the reader positioned at the int attribute</param>
 | |
|         /// <param name="field">The int field to be given the value found</param>
 | |
|         /// <returns>true if an int value was successfuly extracted from the attribute, false otherwise.</returns>
 | |
|         internal bool HandleIntAttribute(XmlReader reader, ref int field)
 | |
|         {
 | |
|             int value;
 | |
|             if ( !Utils.GetInt(Schema, reader, out value) )
 | |
|                 return false;
 | |
| 
 | |
|             field = value;
 | |
|             return true;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Use to handle an attribute with an int data type
 | |
|         /// </summary>
 | |
|         /// <param name="reader">the reader positioned at the int attribute</param>
 | |
|         /// <param name="field">The int field to be given the value found</param>
 | |
|         /// <returns>true if an int value was successfuly extracted from the attribute, false otherwise.</returns>
 | |
|         internal bool HandleByteAttribute(XmlReader reader, ref byte field)
 | |
|         {
 | |
|             byte value;
 | |
|             if ( !Utils.GetByte(Schema, reader, out value) )
 | |
|                 return false;
 | |
| 
 | |
|             field = value;
 | |
|             return true;
 | |
|         }
 | |
|         /// <summary>
 | |
|         /// 
 | |
|         /// </summary>
 | |
|         /// <param name="reader"></param>
 | |
|         /// <param name="field"></param>
 | |
|         /// <returns></returns>
 | |
|         internal bool HandleBoolAttribute(XmlReader reader, ref bool field)
 | |
|         {
 | |
|             bool value;
 | |
|             if ( !Utils.GetBool(Schema,reader,out value) )
 | |
|                 return false;
 | |
| 
 | |
|             field = value;
 | |
|             return true;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Use this to jump through an element that doesn't need any processing
 | |
|         /// </summary>
 | |
|         /// <param name="reader">xml reader currently positioned at an element</param>
 | |
|         protected virtual void SkipThroughElement(XmlReader reader)
 | |
|         {
 | |
|             Debug.Assert(reader != null);
 | |
|             Parse(reader);
 | |
|         }
 | |
| 
 | |
|         protected void SkipElement(XmlReader reader)
 | |
|         {
 | |
|             using (XmlReader subtree = reader.ReadSubtree())
 | |
|             {
 | |
|                 while (subtree.Read()) ;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         #endregion
 | |
| 
 | |
|         #region Protected Properties
 | |
|         /// <summary>
 | |
|         /// 
 | |
|         /// </summary>
 | |
|         protected string SchemaLocation
 | |
|         {
 | |
|             get
 | |
|             {
 | |
|                 if ( Schema != null )
 | |
|                     return Schema.Location;
 | |
|                 return null;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         protected virtual bool HandleText(XmlReader reader)
 | |
|         {
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         internal virtual SchemaElement Clone(SchemaElement parentElement)
 | |
|         {
 | |
|             throw Error.NotImplemented();
 | |
|         }
 | |
|         #endregion
 | |
| 
 | |
|         #region Private Methods
 | |
|         /// <summary>
 | |
|         /// 
 | |
|         /// </summary>
 | |
|         /// <param name="reader"></param>
 | |
|         /// <returns></returns>
 | |
|         private void HandleDocumentationElement(XmlReader reader)
 | |
|         {
 | |
|             Documentation = new DocumentationElement(this);
 | |
|             Documentation.Parse(reader);
 | |
|         }
 | |
|         
 | |
|         /// <summary>
 | |
|         /// 
 | |
|         /// </summary>
 | |
|         /// <param name="reader"></param>
 | |
|         protected virtual void HandleNameAttribute(XmlReader reader)
 | |
|         {
 | |
|             Name = HandleUndottedNameAttribute(reader, Name);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// 
 | |
|         /// </summary>
 | |
|         /// <param name="errorCode"></param>
 | |
|         /// <param name="severity"></param>
 | |
|         /// <param name="source"></param>
 | |
|         /// <param name="lineNumber"></param>
 | |
|         /// <param name="linePosition"></param>
 | |
|         /// <param name="message"></param>
 | |
|         private void AddError( ErrorCode errorCode, EdmSchemaErrorSeverity severity, string sourceLocation, int lineNumber, int linePosition, object message )
 | |
|         {
 | |
|             EdmSchemaError error = null;
 | |
|             string messageString = message as string;
 | |
|             if ( messageString != null )
 | |
|                 error = new EdmSchemaError( messageString, (int)errorCode, severity, sourceLocation, lineNumber, linePosition );
 | |
|             else
 | |
|             {
 | |
|                 Exception ex = message as Exception;
 | |
|                 if ( ex != null )
 | |
|                     error = new EdmSchemaError( ex.Message, (int)errorCode, severity, sourceLocation, lineNumber, linePosition, ex );
 | |
|                 else
 | |
|                     error = new EdmSchemaError( message.ToString(), (int)errorCode, severity, sourceLocation, lineNumber, linePosition );
 | |
|             }
 | |
|             Schema.AddError(error);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Call handler for the current attribute
 | |
|         /// </summary>
 | |
|         /// <param name="reader">XmlReader positioned at the attribute</param>
 | |
|         private void ParseAttribute(XmlReader reader)
 | |
|         {
 | |
| #if false
 | |
|             // the attribute value is schema invalid, just skip it; this avoids some duplicate errors at the expense of better error messages...
 | |
|             if ( reader.SchemaInfo != null && reader.SchemaInfo.Validity == System.Xml.Schema.XmlSchemaValidity.Invalid )
 | |
|                 continue;
 | |
| #endif
 | |
|             string attributeNamespace = reader.NamespaceURI;
 | |
|             if (attributeNamespace == XmlConstants.AnnotationNamespace 
 | |
|                 && reader.LocalName == XmlConstants.UseStrongSpatialTypes
 | |
|                 && !ProhibitAttribute(attributeNamespace, reader.LocalName) 
 | |
|                 && HandleAttribute(reader))
 | |
|             {
 | |
|                 return;
 | |
|             }
 | |
|             else if (!Schema.IsParseableXmlNamespace(attributeNamespace, true))
 | |
|             {
 | |
|                 AddOtherContent(reader);                
 | |
|             }
 | |
|             else if (!ProhibitAttribute(attributeNamespace, reader.LocalName)&&
 | |
|                      HandleAttribute(reader))
 | |
|             {
 | |
|                 return;
 | |
|             }
 | |
|             else if (reader.SchemaInfo == null || reader.SchemaInfo.Validity != System.Xml.Schema.XmlSchemaValidity.Invalid)
 | |
|             {
 | |
|                 // there's no handler for (namespace,name) and there wasn't a validation error. 
 | |
|                 // Report an error of our own if the node is in no namespace or if it is in one of our xml schemas tartget namespace.
 | |
|                 if (string.IsNullOrEmpty(attributeNamespace) || Schema.IsParseableXmlNamespace(attributeNamespace, true))
 | |
|                 {
 | |
|                     AddError(ErrorCode.UnexpectedXmlAttribute, EdmSchemaErrorSeverity.Error, reader, System.Data.Entity.Strings.UnexpectedXmlAttribute(reader.Name));
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         protected virtual bool ProhibitAttribute(string namespaceUri, string localName)
 | |
|         {
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// This overload assumes the default namespace
 | |
|         /// </summary>
 | |
|         /// <param name="reader"></param>
 | |
|         /// <param name="localName"></param>
 | |
|         /// <returns></returns>
 | |
|         internal static bool CanHandleAttribute(XmlReader reader, string localName)
 | |
|         {
 | |
|             Debug.Assert(reader.NamespaceURI != null);
 | |
|             return reader.NamespaceURI.Length == 0 && reader.LocalName == localName;
 | |
|         }
 | |
| 
 | |
|         protected virtual bool HandleAttribute(XmlReader reader)
 | |
|         {
 | |
|             if(CanHandleAttribute(reader, XmlConstants.Name))
 | |
|             {
 | |
|                 HandleNameAttribute(reader);
 | |
|                 return true;
 | |
|             }
 | |
| 
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         private bool AddOtherContent(XmlReader reader)
 | |
|         {
 | |
|             int lineNumber;
 | |
|             int linePosition;
 | |
|             GetPositionInfo(reader, out lineNumber, out linePosition);
 | |
| 
 | |
|             MetadataProperty property;
 | |
|             if (reader.NodeType == XmlNodeType.Element)
 | |
|             {
 | |
| 
 | |
|                 if (this._schema.SchemaVersion == XmlConstants.EdmVersionForV1 ||
 | |
|                     this._schema.SchemaVersion == XmlConstants.EdmVersionForV1_1)
 | |
|                 {
 | |
|                     // skip this element
 | |
|                     // we don't support element annotations in v1 and v1.1
 | |
|                     return true;
 | |
|                 }
 | |
| 
 | |
|                 // in V1 and V1.1 the codegen can only appear as the attribute annotation and we want to maintain
 | |
|                 // the same behavior for V2, thus we throw if we encounter CodeGen namespace 
 | |
|                 // in structural annotation in V2 and furthur version
 | |
|                 if (this._schema.SchemaVersion >= XmlConstants.EdmVersionForV2 
 | |
|                     && reader.NamespaceURI == XmlConstants.CodeGenerationSchemaNamespace)
 | |
|                 {
 | |
|                     Debug.Assert(
 | |
|                         XmlConstants.SchemaVersionLatest == XmlConstants.EdmVersionForV3, 
 | |
|                         "Please add checking for the latest namespace");
 | |
| 
 | |
|                     AddError(ErrorCode.NoCodeGenNamespaceInStructuralAnnotation, EdmSchemaErrorSeverity.Error, lineNumber, linePosition, Strings.NoCodeGenNamespaceInStructuralAnnotation(XmlConstants.CodeGenerationSchemaNamespace));
 | |
|                     return true;
 | |
|                 }
 | |
| 
 | |
|                 Debug.Assert(
 | |
|                         !Schema.IsParseableXmlNamespace(reader.NamespaceURI, false),
 | |
|                         "Structural annotation cannot use any edm reserved namespaces");
 | |
| 
 | |
|                 // using this subtree aproach because when I call 
 | |
|                 // reader.ReadOuterXml() it positions me at the Node beyond
 | |
|                 // the end of the node I am starting on
 | |
|                 // which doesn't work with the parsing logic
 | |
|                 using (XmlReader subtree = reader.ReadSubtree())
 | |
|                 {
 | |
|                     subtree.Read();
 | |
|                     XElement element = XElement.Load(new StringReader(subtree.ReadOuterXml()));
 | |
| 
 | |
|                     property = CreateMetadataPropertyFromOtherNamespaceXmlArtifact(element.Name.NamespaceName, element.Name.LocalName, element);
 | |
|                 }
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 if (reader.NamespaceURI == XmlNamespaceNamespace)
 | |
|                 {
 | |
|                     // we don't bring in namespace definitions
 | |
|                     return true;
 | |
|                 }
 | |
| 
 | |
|                 Debug.Assert(reader.NodeType == XmlNodeType.Attribute, "called an attribute function when not on an attribute");
 | |
|                 property = CreateMetadataPropertyFromOtherNamespaceXmlArtifact(reader.NamespaceURI, reader.LocalName, reader.Value);
 | |
|             }
 | |
| 
 | |
|             if (!OtherContent.Exists(mp => mp.Identity == property.Identity))
 | |
|             {
 | |
|                 OtherContent.Add(property);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 AddError(ErrorCode.AlreadyDefined, EdmSchemaErrorSeverity.Error, lineNumber, linePosition, Strings.DuplicateAnnotation(property.Identity, this.FQName));
 | |
|             }
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         internal static MetadataProperty CreateMetadataPropertyFromOtherNamespaceXmlArtifact(string xmlNamespaceUri, string artifactName, object value)
 | |
|         {
 | |
|             MetadataProperty property;
 | |
|             property = new MetadataProperty(xmlNamespaceUri + ":" + artifactName,
 | |
|                                        TypeUsage.Create(EdmProviderManifest.Instance.GetPrimitiveType(PrimitiveTypeKind.String)),
 | |
|                                        value);
 | |
|             return property;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Call handler for the current element
 | |
|         /// </summary>
 | |
|         /// <param name="reader">XmlReader positioned at the element</param>
 | |
|         /// <returns>true if element content should be skipped</returns>
 | |
|         private bool ParseElement(XmlReader reader)
 | |
|         {
 | |
|             string elementNamespace = reader.NamespaceURI;
 | |
|             // for schema element that right under the schema, we just ignore them, since schema does not
 | |
|             // have metadataproperties
 | |
|             if (!Schema.IsParseableXmlNamespace(elementNamespace, true) && this.ParentElement != null)
 | |
|             {
 | |
|                 return AddOtherContent(reader);
 | |
|             }
 | |
|             if (HandleElement(reader))
 | |
|             {
 | |
|                 return false;
 | |
|             }
 | |
|             else
 | |
|             {
 | |
| 
 | |
|                 // we need to report an error if the namespace for this element is a target namespace for the xml schemas we are parsing against.
 | |
|                 // otherwise we assume that this is either a valid 'any' element or that the xsd validator has generated an error
 | |
|                 if (string.IsNullOrEmpty(elementNamespace) || Schema.IsParseableXmlNamespace(reader.NamespaceURI, false))
 | |
|                 {
 | |
|                     AddError(ErrorCode.UnexpectedXmlElement, EdmSchemaErrorSeverity.Error, reader, System.Data.Entity.Strings.UnexpectedXmlElement(reader.Name));
 | |
|                 }
 | |
|                 return true;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         protected bool CanHandleElement(XmlReader reader, string localName)
 | |
|         {
 | |
|             return reader.NamespaceURI == Schema.SchemaXmlNamespace && reader.LocalName == localName;
 | |
|         }
 | |
| 
 | |
|         protected virtual bool HandleElement(XmlReader reader)
 | |
|         {
 | |
|             if (CanHandleElement(reader, XmlConstants.Documentation))
 | |
|             {
 | |
|                 HandleDocumentationElement(reader);
 | |
|                 return true;
 | |
|             }
 | |
| 
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Handle text data.
 | |
|         /// </summary>
 | |
|         /// <param name="reader">XmlReader positioned at Text, CData, or SignificantWhitespace </param>
 | |
|         private void ParseText(XmlReader reader)
 | |
|         {
 | |
|             if (HandleText(reader))
 | |
|             {
 | |
|                 return;
 | |
|             }
 | |
|             else if (reader.Value != null && reader.Value.Trim().Length == 0)
 | |
|             {
 | |
|                 // just ignore this text.  Don't add an error, since the value is just whitespace.
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 AddError( ErrorCode.TextNotAllowed, EdmSchemaErrorSeverity.Error, reader, System.Data.Entity.Strings.TextNotAllowed(reader.Value ) );
 | |
|             }
 | |
|         }
 | |
|         #endregion
 | |
| 
 | |
|         [Conditional("DEBUG")]
 | |
|         internal static void AssertReaderConsidersSchemaInvalid(XmlReader reader)
 | |
|         {
 | |
|             Debug.Assert(reader.SchemaInfo == null ||
 | |
|                          reader.SchemaInfo.Validity != System.Xml.Schema.XmlSchemaValidity.Valid, "The xsd should see this as not acceptable");
 | |
|         }
 | |
| 
 | |
|     }
 | |
| }
 |