You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			676 lines
		
	
	
		
			28 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
		
		
			
		
	
	
			676 lines
		
	
	
		
			28 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
|   | //------------------------------------------------------------------------------ | ||
|  | // <copyright file="ValidateNames.cs" company="Microsoft"> | ||
|  | //     Copyright (c) Microsoft Corporation.  All rights reserved. | ||
|  | // </copyright> | ||
|  | // <owner current="true" primary="true">[....]</owner> | ||
|  | //------------------------------------------------------------------------------ | ||
|  | 
 | ||
|  | using System; | ||
|  | #if !SILVERLIGHT | ||
|  | using System.Xml.XPath; | ||
|  | #endif | ||
|  | using System.Diagnostics; | ||
|  | using System.Globalization; | ||
|  | 
 | ||
|  | #if SILVERLIGHT_XPATH | ||
|  | namespace System.Xml.XPath { | ||
|  | #else | ||
|  | namespace System.Xml { | ||
|  | #endif | ||
|  | 
 | ||
|  |     /// <summary> | ||
|  |     /// Contains various static functions and methods for parsing and validating: | ||
|  |     ///     NCName (not namespace-aware, no colons allowed) | ||
|  |     ///     QName (prefix:local-name) | ||
|  |     /// </summary> | ||
|  |     internal static class ValidateNames { | ||
|  | 
 | ||
|  |         internal enum Flags { | ||
|  |             NCNames = 0x1,              // Validate that each non-empty prefix and localName is a valid NCName | ||
|  |             CheckLocalName = 0x2,       // Validate the local-name | ||
|  |             CheckPrefixMapping = 0x4,   // Validate the prefix --> namespace mapping | ||
|  |             All = 0x7, | ||
|  |             AllExceptNCNames = 0x6, | ||
|  |             AllExceptPrefixMapping = 0x3, | ||
|  |         }; | ||
|  | 
 | ||
|  |         static XmlCharType xmlCharType = XmlCharType.Instance; | ||
|  | 
 | ||
|  | #if !SILVERLIGHT | ||
|  |         //----------------------------------------------- | ||
|  |         // Nmtoken parsing | ||
|  |         //----------------------------------------------- | ||
|  |         /// <summary> | ||
|  |         /// Attempts to parse the input string as an Nmtoken (see the XML spec production [7] && XML Namespaces spec). | ||
|  |         /// Quits parsing when an invalid Nmtoken char is reached or the end of string is reached. | ||
|  |         /// Returns the number of valid Nmtoken chars that were parsed. | ||
|  |         /// </summary> | ||
|  |         internal static unsafe int ParseNmtoken(string s, int offset) { | ||
|  |             Debug.Assert(s != null && offset <= s.Length); | ||
|  | 
 | ||
|  |             // Keep parsing until the end of string or an invalid NCName character is reached | ||
|  |             int i = offset; | ||
|  |             while (i < s.Length) { | ||
|  |                 if ((xmlCharType.charProperties[s[i]] & XmlCharType.fNCNameSC) != 0) { // if (xmlCharType.IsNCNameSingleChar(s[i])) { | ||
|  |                     i++; | ||
|  |                 } | ||
|  | #if XML10_FIFTH_EDITION | ||
|  |                 else if (xmlCharType.IsNCNameSurrogateChar(s, i)) { | ||
|  |                     i += 2; | ||
|  |                 } | ||
|  | #endif | ||
|  |                 else { | ||
|  |                     break; | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             return i - offset; | ||
|  |         } | ||
|  | #endif | ||
|  | 
 | ||
|  |         //----------------------------------------------- | ||
|  |         // Nmtoken parsing (no XML namespaces support) | ||
|  |         //----------------------------------------------- | ||
|  |         /// <summary> | ||
|  |         /// Attempts to parse the input string as an Nmtoken (see the XML spec production [7]) without taking  | ||
|  |         /// into account the XML Namespaces spec. What it means is that the ':' character is allowed at any  | ||
|  |         /// position and any number of times in the token. | ||
|  |         /// Quits parsing when an invalid Nmtoken char is reached or the end of string is reached. | ||
|  |         /// Returns the number of valid Nmtoken chars that were parsed. | ||
|  |         /// </summary> | ||
|  | #if SILVERLIGHT && !SILVERLIGHT_DISABLE_SECURITY && XMLCHARTYPE_USE_RESOURCE | ||
|  |         [System.Security.SecuritySafeCritical] | ||
|  | #endif | ||
|  |         internal static unsafe int ParseNmtokenNoNamespaces(string s, int offset) { | ||
|  | 
 | ||
|  |             Debug.Assert(s != null && offset <= s.Length); | ||
|  | 
 | ||
|  |             // Keep parsing until the end of string or an invalid Name character is reached | ||
|  |             int i = offset; | ||
|  |             while (i < s.Length) { | ||
|  |                 if ((xmlCharType.charProperties[s[i]] & XmlCharType.fNCNameSC) != 0 || s[i] == ':') { // if (xmlCharType.IsNameSingleChar(s[i])) { | ||
|  |                     i++; | ||
|  |                 } | ||
|  | #if XML10_FIFTH_EDITION | ||
|  |                 else if (xmlCharType.IsNCNameSurrogateChar(s, i)) { | ||
|  |                     i += 2; | ||
|  |                 } | ||
|  | #endif | ||
|  |                 else { | ||
|  |                     break; | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             return i - offset; | ||
|  |         } | ||
|  | 
 | ||
|  |         // helper methods | ||
|  |         internal static bool IsNmtokenNoNamespaces(string s) { | ||
|  |             int endPos = ParseNmtokenNoNamespaces(s, 0); | ||
|  |             return endPos > 0 && endPos == s.Length; | ||
|  |         } | ||
|  | 
 | ||
|  |         //----------------------------------------------- | ||
|  |         // Name parsing (no XML namespaces support) | ||
|  |         //----------------------------------------------- | ||
|  |         /// <summary> | ||
|  |         /// Attempts to parse the input string as a Name without taking into account the XML Namespaces spec. | ||
|  |         /// What it means is that the ':' character does not delimiter prefix and local name, but it is a regular | ||
|  |         /// name character, which is allowed to appear at any position and any number of times in the name. | ||
|  |         /// Quits parsing when an invalid Name char is reached or the end of string is reached. | ||
|  |         /// Returns the number of valid Name chars that were parsed. | ||
|  |         /// </summary> | ||
|  | #if SILVERLIGHT && !SILVERLIGHT_DISABLE_SECURITY && XMLCHARTYPE_USE_RESOURCE | ||
|  |         [System.Security.SecuritySafeCritical] | ||
|  | #endif | ||
|  |         internal static unsafe int ParseNameNoNamespaces(string s, int offset) { | ||
|  | 
 | ||
|  |             Debug.Assert(s != null && offset <= s.Length); | ||
|  | 
 | ||
|  |             // Quit if the first character is not a valid NCName starting character | ||
|  |             int i = offset; | ||
|  |             if (i < s.Length) { | ||
|  |                 if ((xmlCharType.charProperties[s[i]] & XmlCharType.fNCStartNameSC) != 0 || s[i] == ':') { // xmlCharType.IsStartNCNameSingleChar(s[i])) { | ||
|  |                     i++; | ||
|  |                 } | ||
|  | #if XML10_FIFTH_EDITION | ||
|  |                 else if (xmlCharType.IsNCNameSurrogateChar(s, i)) { | ||
|  |                     i += 2; | ||
|  |                 } | ||
|  | #endif | ||
|  |                 else { | ||
|  |                     return 0; // no valid StartNCName char | ||
|  |                 } | ||
|  | 
 | ||
|  |                 // Keep parsing until the end of string or an invalid NCName character is reached | ||
|  |                 while (i < s.Length) { | ||
|  |                     if ((xmlCharType.charProperties[s[i]] & XmlCharType.fNCNameSC) != 0 || s[i] == ':') { // if (xmlCharType.IsNCNameSingleChar(s[i])) | ||
|  |                         i++; | ||
|  |                     } | ||
|  | #if XML10_FIFTH_EDITION | ||
|  |                     else if (xmlCharType.IsNCNameSurrogateChar(s, i)) { | ||
|  |                         i += 2; | ||
|  |                     } | ||
|  | #endif | ||
|  |                     else { | ||
|  |                         break; | ||
|  |                     } | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             return i - offset; | ||
|  |         } | ||
|  | 
 | ||
|  |         // helper methods | ||
|  |         internal static bool IsNameNoNamespaces(string s) { | ||
|  |             int endPos = ParseNameNoNamespaces(s, 0); | ||
|  |             return endPos > 0 && endPos == s.Length; | ||
|  |         } | ||
|  | 
 | ||
|  |         //----------------------------------------------- | ||
|  |         // NCName parsing | ||
|  |         //----------------------------------------------- | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// Attempts to parse the input string as an NCName (see the XML Namespace spec). | ||
|  |         /// Quits parsing when an invalid NCName char is reached or the end of string is reached. | ||
|  |         /// Returns the number of valid NCName chars that were parsed. | ||
|  |         /// </summary> | ||
|  | #if SILVERLIGHT && !SILVERLIGHT_DISABLE_SECURITY && XMLCHARTYPE_USE_RESOURCE | ||
|  |         [System.Security.SecuritySafeCritical] | ||
|  | #endif | ||
|  |         internal static unsafe int ParseNCName(string s, int offset) { | ||
|  | 
 | ||
|  |             Debug.Assert(s != null && offset <= s.Length); | ||
|  | 
 | ||
|  |             // Quit if the first character is not a valid NCName starting character | ||
|  |             int i = offset; | ||
|  |             if (i < s.Length) { | ||
|  |                 if ((xmlCharType.charProperties[s[i]] & XmlCharType.fNCStartNameSC) != 0) { // xmlCharType.IsStartNCNameSingleChar(s[i])) { | ||
|  |                     i++; | ||
|  |                 } | ||
|  | #if XML10_FIFTH_EDITION | ||
|  |                 else if (xmlCharType.IsNCNameSurrogateChar(s, i)) { | ||
|  |                     i += 2; | ||
|  |                 } | ||
|  | #endif | ||
|  |                 else { | ||
|  |                     return 0; // no valid StartNCName char | ||
|  |                 } | ||
|  | 
 | ||
|  |                 // Keep parsing until the end of string or an invalid NCName character is reached | ||
|  |                 while (i < s.Length) { | ||
|  |                     if ((xmlCharType.charProperties[s[i]] & XmlCharType.fNCNameSC) != 0) { // if (xmlCharType.IsNCNameSingleChar(s[i])) | ||
|  |                         i++; | ||
|  |                     } | ||
|  | #if XML10_FIFTH_EDITION | ||
|  |                     else if (xmlCharType.IsNCNameSurrogateChar(s, i)) { | ||
|  |                         i += 2; | ||
|  |                     } | ||
|  | #endif | ||
|  |                     else { | ||
|  |                         break; | ||
|  |                     } | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             return i - offset; | ||
|  |         } | ||
|  | 
 | ||
|  |         internal static int ParseNCName(string s) { | ||
|  |             return ParseNCName(s, 0); | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// Calls parseName and throws exception if the resulting name is not a valid NCName. | ||
|  |         /// Returns the input string if there is no error. | ||
|  |         /// </summary> | ||
|  |         internal static string ParseNCNameThrow(string s) { | ||
|  |             // throwOnError = true | ||
|  |             ParseNCNameInternal(s, true); | ||
|  |             return s; | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// Calls parseName and returns false or throws exception if the resulting name is not | ||
|  |         /// a valid NCName.  Returns the input string if there is no error. | ||
|  |         /// </summary> | ||
|  |         private static bool ParseNCNameInternal(string s, bool throwOnError) { | ||
|  |             int len = ParseNCName(s, 0); | ||
|  | 
 | ||
|  |             if (len == 0 || len != s.Length) { | ||
|  |                 // If the string is not a valid NCName, then throw or return false | ||
|  |                 if (throwOnError) ThrowInvalidName(s, 0, len); | ||
|  |                 return false; | ||
|  |             } | ||
|  | 
 | ||
|  |             return true; | ||
|  |         } | ||
|  | 
 | ||
|  |         //----------------------------------------------- | ||
|  |         // QName parsing | ||
|  |         //----------------------------------------------- | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// Attempts to parse the input string as a QName (see the XML Namespace spec). | ||
|  |         /// Quits parsing when an invalid QName char is reached or the end of string is reached. | ||
|  |         /// Returns the number of valid QName chars that were parsed. | ||
|  |         /// Sets colonOffset to the offset of a colon character if it exists, or 0 otherwise. | ||
|  |         /// </summary> | ||
|  |         internal static int ParseQName(string s, int offset, out int colonOffset) { | ||
|  |             int len, lenLocal; | ||
|  | 
 | ||
|  |             // Assume no colon | ||
|  |             colonOffset = 0; | ||
|  | 
 | ||
|  |             // Parse NCName (may be prefix, may be local name) | ||
|  |             len = ParseNCName(s, offset); | ||
|  |             if (len != 0) { | ||
|  | 
 | ||
|  |                 // Non-empty NCName, so look for colon if there are any characters left | ||
|  |                 offset += len; | ||
|  |                 if (offset < s.Length && s[offset] == ':') { | ||
|  | 
 | ||
|  |                     // First NCName was prefix, so look for local name part | ||
|  |                     lenLocal = ParseNCName(s, offset + 1); | ||
|  |                     if (lenLocal != 0) { | ||
|  |                         // Local name part found, so increase total QName length (add 1 for colon) | ||
|  |                         colonOffset = offset; | ||
|  |                         len += lenLocal + 1; | ||
|  |                     } | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             return len; | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// Calls parseQName and throws exception if the resulting name is not a valid QName. | ||
|  |         /// Returns the prefix and local name parts. | ||
|  |         /// </summary> | ||
|  |         internal static void ParseQNameThrow(string s, out string prefix, out string localName) { | ||
|  |             int colonOffset; | ||
|  |             int len = ParseQName(s, 0, out colonOffset); | ||
|  | 
 | ||
|  |             if (len == 0 || len != s.Length) { | ||
|  |                 // If the string is not a valid QName, then throw | ||
|  |                 ThrowInvalidName(s, 0, len); | ||
|  |             } | ||
|  | 
 | ||
|  |             if (colonOffset != 0) { | ||
|  |                 prefix = s.Substring(0, colonOffset); | ||
|  |                 localName = s.Substring(colonOffset + 1); | ||
|  |             } | ||
|  |             else { | ||
|  |                 prefix = ""; | ||
|  |                 localName = s; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  | #if !SILVERLIGHT | ||
|  |         /// <summary> | ||
|  |         /// Parses the input string as a NameTest (see the XPath spec), returning the prefix and | ||
|  |         /// local name parts.  Throws an exception if the given string is not a valid NameTest. | ||
|  |         /// If the NameTest contains a star, null values for localName (case NCName':*'), or for | ||
|  |         /// both localName and prefix (case '*') are returned. | ||
|  |         /// </summary> | ||
|  |         internal static void ParseNameTestThrow(string s, out string prefix, out string localName) { | ||
|  |             int len, lenLocal, offset; | ||
|  | 
 | ||
|  |             if (s.Length != 0 && s[0] == '*') { | ||
|  |                 // '*' as a NameTest | ||
|  |                 prefix = localName = null; | ||
|  |                 len = 1; | ||
|  |             } | ||
|  |             else { | ||
|  |                 // Parse NCName (may be prefix, may be local name) | ||
|  |                 len = ParseNCName(s, 0); | ||
|  |                 if (len != 0) { | ||
|  | 
 | ||
|  |                     // Non-empty NCName, so look for colon if there are any characters left | ||
|  |                     localName = s.Substring(0, len); | ||
|  |                     if (len < s.Length && s[len] == ':') { | ||
|  | 
 | ||
|  |                         // First NCName was prefix, so look for local name part | ||
|  |                         prefix = localName; | ||
|  |                         offset = len + 1; | ||
|  |                         if (offset < s.Length && s[offset] == '*') { | ||
|  |                             // '*' as a local name part, add 2 to len for colon and star | ||
|  |                             localName = null; | ||
|  |                             len += 2; | ||
|  |                         } | ||
|  |                         else { | ||
|  |                             lenLocal = ParseNCName(s, offset); | ||
|  |                             if (lenLocal != 0) { | ||
|  |                                 // Local name part found, so increase total NameTest length | ||
|  |                                 localName = s.Substring(offset, lenLocal); | ||
|  |                                 len += lenLocal + 1; | ||
|  |                             } | ||
|  |                         } | ||
|  |                     } | ||
|  |                     else { | ||
|  |                         prefix = string.Empty; | ||
|  |                     } | ||
|  |                 } | ||
|  |                 else { | ||
|  |                     // Make the compiler happy | ||
|  |                     prefix = localName = null; | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             if (len == 0 || len != s.Length) { | ||
|  |                 // If the string is not a valid NameTest, then throw | ||
|  |                 ThrowInvalidName(s, 0, len); | ||
|  |             } | ||
|  |         } | ||
|  | #endif | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// Throws an invalid name exception. | ||
|  |         /// </summary> | ||
|  |         /// <param name="s">String that was parsed.</param> | ||
|  |         /// <param name="offsetStartChar">Offset in string where parsing began.</param> | ||
|  |         /// <param name="offsetBadChar">Offset in string where parsing failed.</param> | ||
|  |         internal static void ThrowInvalidName(string s, int offsetStartChar, int offsetBadChar) { | ||
|  |             // If the name is empty, throw an exception | ||
|  |             if (offsetStartChar >= s.Length) | ||
|  | #if !SILVERLIGHT_XPATH | ||
|  |                 throw new XmlException(Res.Xml_EmptyName, string.Empty); | ||
|  | #else | ||
|  |                 throw new XmlException(Res.GetString(Res.Xml_EmptyName, string.Empty)); | ||
|  | #endif | ||
|  | 
 | ||
|  |             Debug.Assert(offsetBadChar < s.Length); | ||
|  | 
 | ||
|  |             if (xmlCharType.IsNCNameSingleChar(s[offsetBadChar]) && !XmlCharType.Instance.IsStartNCNameSingleChar(s[offsetBadChar])) { | ||
|  |                 // The error character is a valid name character, but is not a valid start name character | ||
|  | #if !SILVERLIGHT_XPATH | ||
|  |                 throw new XmlException(Res.Xml_BadStartNameChar, XmlException.BuildCharExceptionArgs(s, offsetBadChar)); | ||
|  | #else | ||
|  |                 throw new XmlException(Res.GetString(Res.Xml_BadStartNameChar, XmlExceptionHelper.BuildCharExceptionArgs(s, offsetBadChar))); | ||
|  | #endif | ||
|  |             } | ||
|  |             else { | ||
|  |                 // The error character is an invalid name character | ||
|  | #if !SILVERLIGHT_XPATH | ||
|  |                 throw new XmlException(Res.Xml_BadNameChar, XmlException.BuildCharExceptionArgs(s, offsetBadChar)); | ||
|  | #else | ||
|  |                 throw new XmlException(Res.GetString(Res.Xml_BadNameChar, XmlExceptionHelper.BuildCharExceptionArgs(s, offsetBadChar))); | ||
|  | #endif | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  | #if !SILVERLIGHT | ||
|  |         internal static Exception GetInvalidNameException(string s, int offsetStartChar, int offsetBadChar) { | ||
|  |             // If the name is empty, throw an exception | ||
|  |             if (offsetStartChar >= s.Length) | ||
|  |                 return new XmlException(Res.Xml_EmptyName, string.Empty); | ||
|  | 
 | ||
|  |             Debug.Assert(offsetBadChar < s.Length); | ||
|  | 
 | ||
|  |             if (xmlCharType.IsNCNameSingleChar(s[offsetBadChar]) && !xmlCharType.IsStartNCNameSingleChar(s[offsetBadChar])) { | ||
|  |                 // The error character is a valid name character, but is not a valid start name character | ||
|  |                 return new XmlException(Res.Xml_BadStartNameChar, XmlException.BuildCharExceptionArgs(s, offsetBadChar)); | ||
|  |             } | ||
|  |             else { | ||
|  |                 // The error character is an invalid name character | ||
|  |                 return new XmlException(Res.Xml_BadNameChar, XmlException.BuildCharExceptionArgs(s, offsetBadChar)); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// Returns true if "prefix" starts with the characters 'x', 'm', 'l' (case-insensitive). | ||
|  |         /// </summary> | ||
|  |         internal static bool StartsWithXml(string s) { | ||
|  |             if (s.Length < 3) | ||
|  |                 return false; | ||
|  | 
 | ||
|  |             if (s[0] != 'x' && s[0] != 'X') | ||
|  |                 return false; | ||
|  | 
 | ||
|  |             if (s[1] != 'm' && s[1] != 'M') | ||
|  |                 return false; | ||
|  | 
 | ||
|  |             if (s[2] != 'l' && s[2] != 'L') | ||
|  |                 return false; | ||
|  | 
 | ||
|  |             return true; | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// Returns true if "s" is a namespace that is reserved by Xml 1.0 or Namespace 1.0. | ||
|  |         /// </summary> | ||
|  |         internal static bool IsReservedNamespace(string s) { | ||
|  |             return s.Equals(XmlReservedNs.NsXml) || s.Equals(XmlReservedNs.NsXmlNs); | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// Throw if the specified name parts are not valid according to the rules of "nodeKind".  Check only rules that are | ||
|  |         /// specified by the Flags. | ||
|  |         /// NOTE: Namespaces should be passed using a prefix, ns pair.  "localName" is always string.Empty. | ||
|  |         /// </summary> | ||
|  |         internal static void ValidateNameThrow(string prefix, string localName, string ns, XPathNodeType nodeKind, Flags flags) { | ||
|  |             // throwOnError = true | ||
|  |             ValidateNameInternal(prefix, localName, ns, nodeKind, flags, true); | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// Return false if the specified name parts are not valid according to the rules of "nodeKind".  Check only rules that are | ||
|  |         /// specified by the Flags. | ||
|  |         /// NOTE: Namespaces should be passed using a prefix, ns pair.  "localName" is always string.Empty. | ||
|  |         /// </summary> | ||
|  |         internal static bool ValidateName(string prefix, string localName, string ns, XPathNodeType nodeKind, Flags flags) { | ||
|  |             // throwOnError = false | ||
|  |             return ValidateNameInternal(prefix, localName, ns, nodeKind, flags, false); | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// Return false or throw if the specified name parts are not valid according to the rules of "nodeKind".  Check only rules | ||
|  |         /// that are specified by the Flags. | ||
|  |         /// NOTE: Namespaces should be passed using a prefix, ns pair.  "localName" is always string.Empty. | ||
|  |         /// </summary> | ||
|  |         private static bool ValidateNameInternal(string prefix, string localName, string ns, XPathNodeType nodeKind, Flags flags, bool throwOnError) { | ||
|  |             Debug.Assert(prefix != null && localName != null && ns != null); | ||
|  | 
 | ||
|  |             if ((flags & Flags.NCNames) != 0) { | ||
|  | 
 | ||
|  |                 // 1. Verify that each non-empty prefix and localName is a valid NCName | ||
|  |                 if (prefix.Length != 0) | ||
|  |                     if (!ParseNCNameInternal(prefix, throwOnError)) { | ||
|  |                         return false; | ||
|  |                     } | ||
|  | 
 | ||
|  |                 if (localName.Length != 0) | ||
|  |                     if (!ParseNCNameInternal(localName, throwOnError)) { | ||
|  |                         return false; | ||
|  |                     } | ||
|  |             } | ||
|  | 
 | ||
|  |             if ((flags & Flags.CheckLocalName) != 0) { | ||
|  | 
 | ||
|  |                 // 2. Determine whether the local name is valid | ||
|  |                 switch (nodeKind) { | ||
|  |                     case XPathNodeType.Element: | ||
|  |                         // Elements and attributes must have a non-empty local name | ||
|  |                         if (localName.Length == 0) { | ||
|  |                             if (throwOnError) throw new XmlException(Res.Xdom_Empty_LocalName, string.Empty); | ||
|  |                             return false; | ||
|  |                         } | ||
|  |                         break; | ||
|  | 
 | ||
|  |                     case XPathNodeType.Attribute: | ||
|  |                         // Attribute local name cannot be "xmlns" if namespace is empty | ||
|  |                         if (ns.Length == 0 && localName.Equals("xmlns")) { | ||
|  |                             if (throwOnError) throw new XmlException(Res.XmlBadName, new string[] {nodeKind.ToString(), localName}); | ||
|  |                             return false; | ||
|  |                         } | ||
|  |                         goto case XPathNodeType.Element; | ||
|  | 
 | ||
|  |                     case XPathNodeType.ProcessingInstruction: | ||
|  |                         // PI's local-name must be non-empty and cannot be 'xml' (case-insensitive) | ||
|  |                         if (localName.Length == 0 || (localName.Length == 3 && StartsWithXml(localName))) { | ||
|  |                             if (throwOnError) throw new XmlException(Res.Xml_InvalidPIName, localName); | ||
|  |                             return false; | ||
|  |                         } | ||
|  |                         break; | ||
|  | 
 | ||
|  |                     default: | ||
|  |                         // All other node types must have empty local-name | ||
|  |                         if (localName.Length != 0) { | ||
|  |                             if (throwOnError) throw new XmlException(Res.XmlNoNameAllowed, nodeKind.ToString()); | ||
|  |                             return false; | ||
|  |                         } | ||
|  |                         break; | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             if ((flags & Flags.CheckPrefixMapping) != 0) { | ||
|  | 
 | ||
|  |                 // 3. Determine whether the prefix is valid | ||
|  |                 switch (nodeKind) { | ||
|  |                     case XPathNodeType.Element: | ||
|  |                     case XPathNodeType.Attribute: | ||
|  |                     case XPathNodeType.Namespace: | ||
|  |                         if (ns.Length == 0) { | ||
|  |                             // If namespace is empty, then prefix must be empty | ||
|  |                             if (prefix.Length != 0) { | ||
|  |                                 if (throwOnError) throw new XmlException(Res.Xml_PrefixForEmptyNs, string.Empty); | ||
|  |                                 return false; | ||
|  |                             } | ||
|  |                         } | ||
|  |                         else { | ||
|  |                             // Don't allow empty attribute prefix since namespace is non-empty | ||
|  |                             if (prefix.Length == 0 && nodeKind == XPathNodeType.Attribute) { | ||
|  |                                 if (throwOnError) throw new XmlException(Res.XmlBadName, new string[] {nodeKind.ToString(), localName}); | ||
|  |                                 return false; | ||
|  |                             } | ||
|  | 
 | ||
|  |                             if (prefix.Equals("xml")) { | ||
|  |                                 // xml prefix must be mapped to the xml namespace | ||
|  |                                 if (!ns.Equals(XmlReservedNs.NsXml)) { | ||
|  |                                     if (throwOnError) throw new XmlException(Res.Xml_XmlPrefix, string.Empty); | ||
|  |                                     return false; | ||
|  |                                 } | ||
|  |                             } | ||
|  |                             else if (prefix.Equals("xmlns")) { | ||
|  |                                 // Prefix may never be 'xmlns' | ||
|  |                                 if (throwOnError) throw new XmlException(Res.Xml_XmlnsPrefix, string.Empty); | ||
|  |                                 return false; | ||
|  |                             } | ||
|  |                             else if (IsReservedNamespace(ns)) { | ||
|  |                                 // Don't allow non-reserved prefixes to map to xml or xmlns namespaces | ||
|  |                                 if (throwOnError) throw new XmlException(Res.Xml_NamespaceDeclXmlXmlns, string.Empty); | ||
|  |                                 return false; | ||
|  |                             } | ||
|  |                         } | ||
|  |                         break; | ||
|  | 
 | ||
|  |                     case XPathNodeType.ProcessingInstruction: | ||
|  |                         // PI's prefix and namespace must be empty | ||
|  |                         if (prefix.Length != 0 || ns.Length != 0) { | ||
|  |                             if (throwOnError) throw new XmlException(Res.Xml_InvalidPIName, CreateName(prefix, localName)); | ||
|  |                             return false; | ||
|  |                         } | ||
|  |                         break; | ||
|  | 
 | ||
|  |                     default: | ||
|  |                         // All other node types must have empty prefix and namespace | ||
|  |                         if (prefix.Length != 0 || ns.Length != 0) { | ||
|  |                             if (throwOnError) throw new XmlException(Res.XmlNoNameAllowed, nodeKind.ToString()); | ||
|  |                             return false; | ||
|  |                         } | ||
|  |                         break; | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             return true; | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// Creates a colon-delimited qname from prefix and local name parts. | ||
|  |         /// </summary> | ||
|  |         private static string CreateName(string prefix, string localName) { | ||
|  |             return (prefix.Length != 0) ? prefix + ":" + localName : localName; | ||
|  |         } | ||
|  | #endif | ||
|  | 
 | ||
|  | 
 | ||
|  | #if !SILVERLIGHT || SILVERLIGHT_XPATH | ||
|  |         /// <summary> | ||
|  |         /// Split a QualifiedName into prefix and localname, w/o any checking. | ||
|  |         /// (Used for XmlReader/XPathNavigator MoveTo(name) methods) | ||
|  |         /// </summary> | ||
|  |         internal static void SplitQName(string name, out string prefix, out string lname) { | ||
|  |             int colonPos = name.IndexOf(':'); | ||
|  |             if (-1 == colonPos) { | ||
|  |                 prefix = string.Empty; | ||
|  |                 lname = name; | ||
|  |             } | ||
|  |             else if (0 == colonPos || (name.Length-1) == colonPos) { | ||
|  | #if !SILVERLIGHT_XPATH | ||
|  |                 throw new ArgumentException(Res.GetString(Res.Xml_BadNameChar, XmlException.BuildCharExceptionArgs(':', '\0')), "name"); | ||
|  | #else | ||
|  |                 throw new ArgumentException(Res.GetString(Res.Xml_BadNameChar, XmlExceptionHelper.BuildCharExceptionArgs(':', '\0')), "name"); | ||
|  | #endif | ||
|  |             } | ||
|  |             else { | ||
|  |                 prefix = name.Substring(0, colonPos); | ||
|  |                 colonPos++; // move after colon | ||
|  |                 lname = name.Substring(colonPos, name.Length - colonPos); | ||
|  |             } | ||
|  |         } | ||
|  | #endif | ||
|  |     } | ||
|  | 
 | ||
|  | #if SILVERLIGHT_XPATH | ||
|  |     internal class XmlExceptionHelper | ||
|  |     { | ||
|  |         internal static string[] BuildCharExceptionArgs(string data, int invCharIndex) | ||
|  |         { | ||
|  |             return BuildCharExceptionArgs(data[invCharIndex], invCharIndex + 1 < data.Length ? data[invCharIndex + 1] : '\0'); | ||
|  |         } | ||
|  | 
 | ||
|  |         internal static string[] BuildCharExceptionArgs(char[] data, int invCharIndex) | ||
|  |         { | ||
|  |             return BuildCharExceptionArgs(data, data.Length, invCharIndex); | ||
|  |         } | ||
|  | 
 | ||
|  |         internal static string[] BuildCharExceptionArgs(char[] data, int length, int invCharIndex) | ||
|  |         { | ||
|  |             Debug.Assert(invCharIndex < data.Length); | ||
|  |             Debug.Assert(invCharIndex < length); | ||
|  |             Debug.Assert(length <= data.Length); | ||
|  | 
 | ||
|  |             return BuildCharExceptionArgs(data[invCharIndex], invCharIndex + 1 < length ? data[invCharIndex + 1] : '\0'); | ||
|  |         } | ||
|  | 
 | ||
|  |         internal static string[] BuildCharExceptionArgs(char invChar, char nextChar) | ||
|  |         { | ||
|  |             string[] aStringList = new string[2]; | ||
|  | 
 | ||
|  |             // for surrogate characters include both high and low char in the message so that a full character is displayed | ||
|  |             if (XmlCharType.IsHighSurrogate(invChar) && nextChar != 0) | ||
|  |             { | ||
|  |                 int combinedChar = XmlCharType.CombineSurrogateChar(nextChar, invChar); | ||
|  |                 aStringList[0] = new string(new char[] { invChar, nextChar }); | ||
|  |                 aStringList[1] = string.Format(CultureInfo.InvariantCulture, "0x{0:X2}", combinedChar); | ||
|  |             } | ||
|  |             else | ||
|  |             { | ||
|  |                 // don't include 0 character in the string - in means eof-of-string in native code, where this may bubble up to | ||
|  |                 if ((int)invChar == 0) | ||
|  |                 { | ||
|  |                     aStringList[0] = "."; | ||
|  |                 } | ||
|  |                 else | ||
|  |                 { | ||
|  |                     aStringList[0] = invChar.ToString(CultureInfo.InvariantCulture); | ||
|  |                 } | ||
|  |                 aStringList[1] = string.Format(CultureInfo.InvariantCulture, "0x{0:X2}", (int)invChar); | ||
|  |             } | ||
|  |             return aStringList; | ||
|  |         } | ||
|  |     } | ||
|  | #endif | ||
|  | } |