| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600 |
- //
- // System.Xml.XmlTextReader
- //
- // Author:
- // Jason Diamond ([email protected])
- // Adam Treat ([email protected])
- // Atsushi Enomoto ([email protected])
- //
- // (C) 2001, 2002 Jason Diamond http://injektilo.org/
- //
- // FIXME:
- //
- // Some thought needs to be given to performance.
- //
- // If current node is on an Attribute, Prefix might be null, and
- // in several fields which uses XmlReader, it should be considered.
- //
- using System;
- using System.Collections;
- using System.IO;
- using System.Security.Policy;
- using System.Text;
- using System.Xml.Schema;
- using Mono.Xml;
- using Mono.Xml.Native;
- namespace System.Xml
- {
- public class XmlTextReader : XmlReader, IXmlLineInfo
- {
- #region Constructors
- protected XmlTextReader ()
- {
- }
- public XmlTextReader (Stream input)
- : this (new XmlStreamReader (input))
- {
- }
- public XmlTextReader (string url)
- : this(url, new NameTable ())
- {
- }
- public XmlTextReader (TextReader input)
- : this (input, new NameTable ())
- {
- }
- protected XmlTextReader (XmlNameTable nt)
- : this (String.Empty, null, XmlNodeType.None, null)
- {
- }
- public XmlTextReader (Stream input, XmlNameTable nt)
- : this(new XmlStreamReader (input), nt)
- {
- }
- public XmlTextReader (string url, Stream input)
- : this (url, new XmlStreamReader (input))
- {
- }
- public XmlTextReader (string url, TextReader input)
- : this (url, input, new NameTable ())
- {
- }
- public XmlTextReader (string url, XmlNameTable nt)
- {
- Uri uri = resolver.ResolveUri (null, url);
- Stream s = resolver.GetEntity (uri, null, typeof (Stream)) as Stream;
- XmlParserContext ctx = new XmlParserContext (nt,
- new XmlNamespaceManager (nt),
- String.Empty,
- XmlSpace.None);
- this.InitializeContext (uri.ToString(), ctx, new XmlStreamReader (s), XmlNodeType.Document);
- }
- public XmlTextReader (TextReader input, XmlNameTable nt)
- : this (String.Empty, input, nt)
- {
- }
- public XmlTextReader (Stream xmlFragment, XmlNodeType fragType, XmlParserContext context)
- : this (context != null ? context.BaseURI : String.Empty,
- new XmlStreamReader (xmlFragment),
- fragType,
- context)
- {
- }
- public XmlTextReader (string url, Stream input, XmlNameTable nt)
- : this (url, new XmlStreamReader (input), nt)
- {
- }
- public XmlTextReader (string url, TextReader input, XmlNameTable nt)
- : this (url, input, XmlNodeType.Document, null)
- {
- }
- public XmlTextReader (string xmlFragment, XmlNodeType fragType, XmlParserContext context)
- : this (context != null ? context.BaseURI : String.Empty,
- new StringReader (xmlFragment),
- fragType,
- context)
- {
- }
- XmlTextReader (string url, TextReader fragment, XmlNodeType fragType, XmlParserContext context)
- {
- InitializeContext (url, context, fragment, fragType);
- }
- #endregion
- #region Properties
- public override int AttributeCount
- {
- get { return attributeCount; }
- }
- public override string BaseURI
- {
- get { return parserContext.BaseURI; }
- }
- public override int Depth
- {
- get {
- int nodeTypeMod = currentToken.NodeType == XmlNodeType.Element ? 0 : -1;
- if (currentAttributeValue >= 0)
- return nodeTypeMod + elementDepth + 2; // inside attribute value.
- else if (currentAttribute >= 0)
- return nodeTypeMod + elementDepth + 1;
- return elementDepth;
- }
- }
- public Encoding Encoding
- {
- get { return parserContext.Encoding; }
- }
- #if NET_2_0
- [MonoTODO]
- public EntityHandling EntityHandling {
- get { throw new NotImplementedException (); }
- }
- #endif
- public override bool EOF
- {
- get
- {
- return
- readState == ReadState.EndOfFile ||
- readState == ReadState.Closed;
- }
- }
- #if NET_2_0
- [MonoTODO]
- public override Evidence [] Evidences {
- get { return base.Evidences; }
- }
- #endif
- public override bool HasValue {
- get { return cursorToken.Value != null; }
- }
- public override bool IsDefault {
- // XmlTextReader does not expand default attributes.
- get { return false; }
- }
- public override bool IsEmptyElement {
- get { return cursorToken.IsEmptyElement; }
- }
- public override string this [int i] {
- get { return GetAttribute (i); }
- }
- public override string this [string name] {
- get { return GetAttribute (name); }
- }
- public override string this [string localName, string namespaceName] {
- get { return GetAttribute (localName, namespaceName); }
- }
- public int LineNumber {
- get {
- if (useProceedingLineInfo)
- return line;
- else
- return cursorToken.LineNumber;
- }
- }
- public int LinePosition {
- get {
- if (useProceedingLineInfo)
- return column;
- else
- return cursorToken.LinePosition;
- }
- }
- public override string LocalName {
- get { return cursorToken.LocalName; }
- }
- public override string Name {
- get { return cursorToken.Name; }
- }
- public bool Namespaces {
- get { return namespaces; }
- set {
- if (readState != ReadState.Initial)
- throw new InvalidOperationException ("Namespaces have to be set before reading.");
- namespaces = value;
- }
- }
- public override string NamespaceURI {
- get { return cursorToken.NamespaceURI; }
- }
- public override XmlNameTable NameTable {
- get { return parserContext.NameTable; }
- }
- public override XmlNodeType NodeType {
- get { return cursorToken.NodeType; }
- }
- public bool Normalization {
- get { return normalization; }
- set { normalization = value; }
- }
- public override string Prefix {
- get { return cursorToken.Prefix; }
- }
- public override char QuoteChar {
- get { return cursorToken.QuoteChar; }
- }
- public override ReadState ReadState {
- get { return readState; }
- }
- public override string Value {
- get { return cursorToken.Value != null ? cursorToken.Value : String.Empty; }
- }
- public WhitespaceHandling WhitespaceHandling {
- get { return whitespaceHandling; }
- set { whitespaceHandling = value; }
- }
- public override string XmlLang {
- get { return parserContext.XmlLang; }
- }
- public XmlResolver XmlResolver {
- set { resolver = value; }
- }
- public override XmlSpace XmlSpace {
- get { return parserContext.XmlSpace; }
- }
- #endregion
- #region Methods
- public override void Close ()
- {
- readState = ReadState.Closed;
- cursorToken.Clear ();
- currentToken.Clear ();
- attributeCount = 0;
- if (reader != null)
- reader.Close ();
- }
- public override string GetAttribute (int i)
- {
- if (i >= attributeCount)
- throw new ArgumentOutOfRangeException ("i is smaller than AttributeCount");
- else {
- return attributeTokens [i].Value;
- }
- }
- // MS.NET 1.0 msdn says that this method returns String.Empty
- // for absent attribute, but in fact it returns null.
- // This description is corrected in MS.NET 1.1 msdn.
- public override string GetAttribute (string name)
- {
- for (int i = 0; i < attributeCount; i++)
- if (attributeTokens [i].Name == name)
- return attributeTokens [i].Value;
- return null;
- }
- private int GetIndexOfQualifiedAttribute (string localName, string namespaceURI)
- {
- for (int i = 0; i < attributeCount; i++) {
- XmlAttributeTokenInfo ti = attributeTokens [i];
- if (ti.LocalName == localName && ti.NamespaceURI == namespaceURI)
- return i;
- }
- return -1;
- }
- internal XmlParserContext GetInternalParserContext ()
- {
- return parserContext;
- }
- public override string GetAttribute (string localName, string namespaceURI)
- {
- int idx = this.GetIndexOfQualifiedAttribute (localName, namespaceURI);
- if (idx < 0)
- return null;
- return attributeTokens [idx].Value;
- }
- public TextReader GetRemainder ()
- {
- if (peekCharsIndex == peekCharsLength)
- return reader;
- return new StringReader (new string (peekChars, peekCharsIndex, peekCharsLength - peekCharsIndex) + reader.ReadToEnd ());
- }
- bool IXmlLineInfo.HasLineInfo ()
- {
- return true;
- }
- public override string LookupNamespace (string prefix)
- {
- return LookupNamespace (prefix, false);
- }
- internal string LookupNamespace (string prefix, bool atomizedName)
- {
- return parserContext.NamespaceManager.LookupNamespace (prefix, atomizedName);
- }
- public override void MoveToAttribute (int i)
- {
- if (i >= attributeCount)
- throw new ArgumentOutOfRangeException ("attribute index out of range.");
- currentAttribute = i;
- currentAttributeValue = -1;
- cursorToken = attributeTokens [i];
- }
- public override bool MoveToAttribute (string name)
- {
- for (int i = 0; i < attributeCount; i++) {
- XmlAttributeTokenInfo ti = attributeTokens [i];
- if (ti.Name == name) {
- MoveToAttribute (i);
- return true;
- }
- }
- return false;
- }
- public override bool MoveToAttribute (string localName, string namespaceName)
- {
- int idx = GetIndexOfQualifiedAttribute (localName, namespaceName);
- if (idx < 0)
- return false;
- MoveToAttribute (idx);
- return true;
- }
- public override bool MoveToElement ()
- {
- if (currentToken == null) // for attribute .ctor()
- return false;
- if (cursorToken == currentToken)
- return false;
- if (currentAttribute >= 0) {
- currentAttribute = -1;
- currentAttributeValue = -1;
- cursorToken = currentToken;
- return true;
- }
- else
- return false;
- }
- public override bool MoveToFirstAttribute ()
- {
- if (attributeCount == 0)
- return false;
- MoveToElement ();
- return MoveToNextAttribute ();
- }
- public override bool MoveToNextAttribute ()
- {
- if (currentAttribute == 0 && attributeCount == 0)
- return false;
- if (currentAttribute + 1 < attributeCount) {
- currentAttribute++;
- currentAttributeValue = -1;
- cursorToken = attributeTokens [currentAttribute];
- return true;
- }
- else
- return false;
- }
- public override bool Read ()
- {
- if (startNodeType == XmlNodeType.Attribute) {
- if (currentAttribute == 0)
- return false; // already read.
- ClearAttributes ();
- IncrementAttributeToken ();
- ReadAttributeValueTokens ('"');
- cursorToken = attributeTokens [0];
- currentAttributeValue = -1;
- readState = ReadState.Interactive;
- return true;
- }
- bool more = false;
- readState = ReadState.Interactive;
- currentLinkedNodeLineNumber = line;
- currentLinkedNodeLinePosition = column;
- useProceedingLineInfo = true;
- cursorToken = currentToken;
- attributeCount = 0;
- currentAttribute = currentAttributeValue = -1;
- currentToken.Clear ();
- // It was moved from end of ReadStartTag ().
- if (depthUp)
- ++depth;
- depthUp = false;
- if (shouldSkipUntilEndTag) {
- shouldSkipUntilEndTag = false;
- return ReadUntilEndTag ();
- }
- base64CacheStartsAt = -1;
- more = ReadContent ();
- if (depth == 0 && !allowMultipleRoot && (IsEmptyElement || NodeType == XmlNodeType.EndElement))
- currentState = XmlNodeType.EndElement;
- if (!more && startNodeType == XmlNodeType.Document && currentState != XmlNodeType.EndElement)
- throw new XmlException ("Document element did not appear.");
- useProceedingLineInfo = false;
- return more;
- }
- public override bool ReadAttributeValue ()
- {
- if (readState == ReadState.Initial && startNodeType == XmlNodeType.Attribute) {
- Read ();
- }
- if (currentAttribute < 0)
- return false;
- XmlAttributeTokenInfo ti = attributeTokens [currentAttribute];
- if (currentAttributeValue < 0)
- currentAttributeValue = ti.ValueTokenStartIndex - 1;
- if (currentAttributeValue < ti.ValueTokenEndIndex) {
- currentAttributeValue++;
- cursorToken = attributeValueTokens [currentAttributeValue];
- return true;
- }
- else
- return false;
- }
- public int ReadBase64 (byte [] buffer, int offset, int length)
- {
- if (offset < 0)
- throw new ArgumentOutOfRangeException ("offset", offset, "Offset must be non-negative integer.");
- else if (length < 0)
- throw new ArgumentOutOfRangeException ("length", length, "Length must be non-negative integer.");
- else if (buffer.Length < offset + length)
- throw new ArgumentOutOfRangeException ("buffer length is smaller than the sum of offset and length.");
- if (length == 0) // It does not raise an error.
- return 0;
- int bufIndex = offset;
- int bufLast = offset + length;
- if (base64CacheStartsAt >= 0) {
- for (int i = base64CacheStartsAt; i < 3; i++) {
- buffer [bufIndex++] = base64Cache [base64CacheStartsAt++];
- if (bufIndex == bufLast)
- return bufLast - offset;
- }
- }
- for (int i = 0; i < 3; i++)
- base64Cache [i] = 0;
- base64CacheStartsAt = -1;
- int max = (int) System.Math.Ceiling (4.0 / 3 * length);
- int additional = max % 4;
- if (additional > 0)
- max += 4 - additional;
- char [] chars = new char [max];
- int charsLength = ReadChars (chars, 0, max);
- byte b = 0;
- byte work = 0;
- for (int i = 0; i < charsLength - 3; i += 4) {
- b = (byte) (GetBase64Byte (chars [i]) << 2);
- if (bufIndex < bufLast)
- buffer [bufIndex] = b;
- else {
- if (base64CacheStartsAt < 0)
- base64CacheStartsAt = 0;
- base64Cache [0] = b;
- }
- // charsLength mod 4 might not equals to 0.
- if (i + 1 == charsLength)
- break;
- b = GetBase64Byte (chars [i + 1]);
- work = (byte) (b >> 4);
- if (bufIndex < bufLast) {
- buffer [bufIndex] += work;
- bufIndex++;
- }
- else
- base64Cache [0] += work;
- work = (byte) ((b & 0xf) << 4);
- if (bufIndex < bufLast) {
- buffer [bufIndex] = work;
- }
- else {
- if (base64CacheStartsAt < 0)
- base64CacheStartsAt = 1;
- base64Cache [1] = work;
- }
- if (i + 2 == charsLength)
- break;
- b = GetBase64Byte (chars [i + 2]);
- work = (byte) (b >> 2);
- if (bufIndex < bufLast) {
- buffer [bufIndex] += work;
- bufIndex++;
- }
- else
- base64Cache [1] += work;
- work = (byte) ((b & 3) << 6);
- if (bufIndex < bufLast)
- buffer [bufIndex] = work;
- else {
- if (base64CacheStartsAt < 0)
- base64CacheStartsAt = 2;
- base64Cache [2] = work;
- }
- if (i + 3 == charsLength)
- break;
- work = GetBase64Byte (chars [i + 3]);
- if (bufIndex < bufLast) {
- buffer [bufIndex] += work;
- bufIndex++;
- }
- else
- base64Cache [2] += work;
- }
- return System.Math.Min (bufLast - offset, bufIndex - offset);
- }
- public int ReadBinHex (byte [] buffer, int offset, int length)
- {
- if (offset < 0)
- throw new ArgumentOutOfRangeException ("offset", offset, "Offset must be non-negative integer.");
- else if (length < 0)
- throw new ArgumentOutOfRangeException ("length", length, "Length must be non-negative integer.");
- else if (buffer.Length < offset + length)
- throw new ArgumentOutOfRangeException ("buffer length is smaller than the sum of offset and length.");
- if (length == 0)
- return 0;
- char [] chars = new char [length * 2];
- int charsLength = ReadChars (chars, 0, length * 2);
- return XmlConvert.FromBinHexString (chars, offset, charsLength, buffer);
- }
- public int ReadChars (char [] buffer, int offset, int length)
- {
- return ReadCharsInternal (buffer, offset, length);
- }
- #if NET_1_1
- #else
- public override string ReadInnerXml ()
- {
- return ReadInnerXmlInternal ();
- }
- public override string ReadOuterXml ()
- {
- return ReadOuterXmlInternal ();
- }
- public override string ReadString ()
- {
- return ReadStringInternal ();
- }
- #endif
- public void ResetState ()
- {
- Init ();
- }
- public override void ResolveEntity ()
- {
- // XmlTextReader does not resolve entities.
- throw new InvalidOperationException ("XmlTextReader cannot resolve external entities.");
- }
- #if NET_2_0
- [MonoTODO ("Implement for performance reason")]
- public override void Skip ()
- {
- base.Skip ();
- }
- #endif
- #endregion
- #region Internals
- // Parsed DTD Objects
- #if DTD_HANDLE_EVENTS
- internal event ValidationEventHandler ValidationEventHandler;
- #endif
- internal DTDObjectModel DTD {
- get { return parserContext.Dtd; }
- }
- internal XmlResolver Resolver {
- get { return resolver; }
- }
- #endregion
- #region Privates
- internal class XmlTokenInfo
- {
- public XmlTokenInfo (XmlTextReader xtr, bool isPrimaryToken)
- {
- this.isPrimaryToken = isPrimaryToken;
- Reader = xtr;
- Clear ();
- }
- bool isPrimaryToken;
- string valueCache;
- protected XmlTextReader Reader;
- public string Name;
- public string LocalName;
- public string Prefix;
- public string NamespaceURI;
- public bool IsEmptyElement;
- public char QuoteChar;
- public int LineNumber;
- public int LinePosition;
- public XmlNodeType NodeType;
- public virtual string Value {
- get {
- if (valueCache != null)
- return valueCache;
- switch (NodeType) {
- case XmlNodeType.Text:
- case XmlNodeType.SignificantWhitespace:
- case XmlNodeType.Whitespace:
- case XmlNodeType.Comment:
- case XmlNodeType.CDATA:
- case XmlNodeType.ProcessingInstruction:
- valueCache = Reader.CreateValueString ();
- return valueCache;
- }
- return null;
- }
- set { valueCache = value; }
- }
- public virtual void Clear ()
- {
- valueCache = null;
- NodeType = XmlNodeType.None;
- Name = LocalName = Prefix = NamespaceURI = String.Empty;
- IsEmptyElement = false;
- QuoteChar = '"';
- LineNumber = LinePosition = 0;
- }
- internal virtual void FillNames ()
- {
- if (Reader.Namespaces) {
- int indexOfColon = -1;
- switch (NodeType) {
- case XmlNodeType.Attribute:
- case XmlNodeType.Element:
- case XmlNodeType.EndElement:
- indexOfColon = Name.IndexOf (':');
- break;
- }
- if (indexOfColon == -1) {
- Prefix = String.Empty;
- LocalName = Name;
- } else {
- // This improves speed by at least nearly 5%, but eats more memory at least nearly 0.3%
- // However, this might be reverted if NameTable is got improved.
- char [] nameArr = Name.ToCharArray ();
- Prefix = Reader.NameTable.Add (nameArr, 0, indexOfColon);
- LocalName = Reader.NameTable.Add (nameArr, indexOfColon + 1, nameArr.Length - indexOfColon - 1);
- // Prefix = Reader.NameTable.Add (Name.Substring (0, indexOfColon));
- // LocalName = Reader.NameTable.Add (Name.Substring (indexOfColon + 1));
- }
- // NamespaceURI
- switch (NodeType) {
- case XmlNodeType.Attribute:
- if (Prefix.Length == 0)
- NamespaceURI = string.Empty;
- else
- NamespaceURI = Reader.LookupNamespace (Prefix, true);
- break;
- case XmlNodeType.Element:
- case XmlNodeType.EndElement:
- NamespaceURI = Reader.LookupNamespace (Prefix, true);
- break;
- default:
- NamespaceURI = "";
- break;
- }
- } else {
- Prefix = String.Empty;
- LocalName = Name;
- }
- }
- }
- internal class XmlAttributeTokenInfo : XmlTokenInfo
- {
- public XmlAttributeTokenInfo (XmlTextReader reader)
- : base (reader, false)
- {
- NodeType = XmlNodeType.Attribute;
- }
- public int ValueTokenStartIndex;
- public int ValueTokenEndIndex;
- string valueCache;
- bool cachedNormalization;
- StringBuilder tmpBuilder = new StringBuilder ();
- public override string Value {
- get {
- if (cachedNormalization != Reader.Normalization)
- valueCache = null;
- if (valueCache != null)
- return valueCache;
- cachedNormalization = Reader.Normalization;
- // An empty value should return String.Empty.
- if (ValueTokenStartIndex == ValueTokenEndIndex) {
- XmlTokenInfo ti = Reader.attributeValueTokens [ValueTokenStartIndex];
- if (ti.NodeType == XmlNodeType.EntityReference)
- valueCache = String.Concat ("&", ti.Name, ";");
- else
- valueCache = ti.Value;
- if (cachedNormalization)
- NormalizeSpaces ();
- return valueCache;
- }
- tmpBuilder.Length = 0;
- for (int i = ValueTokenStartIndex; i <= ValueTokenEndIndex; i++) {
- XmlTokenInfo ti = Reader.attributeValueTokens [i];
- if (ti.NodeType == XmlNodeType.Text)
- tmpBuilder.Append (ti.Value);
- else {
- tmpBuilder.Append ('&');
- tmpBuilder.Append (ti.Name);
- tmpBuilder.Append (';');
- }
- }
- valueCache = tmpBuilder.ToString ();
- if (cachedNormalization)
- NormalizeSpaces ();
- return valueCache;
- }
- set { valueCache = value; }
- }
- public override void Clear ()
- {
- base.Clear ();
- valueCache = null;
- NodeType = XmlNodeType.Attribute;
- ValueTokenStartIndex = ValueTokenEndIndex = 0;
- }
- internal override void FillNames ()
- {
- base.FillNames ();
- if (Prefix == "xmlns" || Name == "xmlns")
- NamespaceURI = XmlNamespaceManager.XmlnsXmlns;
- }
- private void NormalizeSpaces ()
- {
- tmpBuilder.Length = 0;
- for (int i = 0; i < valueCache.Length; i++)
- switch (valueCache [i]) {
- case '\r':
- if (i + 1 < valueCache.Length && valueCache [i + 1] == '\n')
- i++;
- goto case '\n';
- case '\t':
- case '\n':
- tmpBuilder.Append (' ');
- break;
- default:
- tmpBuilder.Append (valueCache [i]);
- break;
- }
- valueCache = tmpBuilder.ToString ();
- }
- }
- private XmlTokenInfo cursorToken;
- private XmlTokenInfo currentToken;
- private XmlAttributeTokenInfo currentAttributeToken;
- private XmlTokenInfo currentAttributeValueToken;
- private XmlAttributeTokenInfo [] attributeTokens = new XmlAttributeTokenInfo [10];
- private XmlTokenInfo [] attributeValueTokens = new XmlTokenInfo [10];
- private int currentAttribute;
- private int currentAttributeValue;
- private int attributeCount;
- private XmlParserContext parserContext;
- private ReadState readState;
- private int depth;
- private int elementDepth;
- private bool depthUp;
- private bool popScope;
- private string [] elementNames;
- int elementNameStackPos;
- private bool allowMultipleRoot;
- private bool isStandalone;
- private bool returnEntityReference;
- private string entityReferenceName;
- private char [] nameBuffer;
- private int nameLength;
- private int nameCapacity;
- private const int initialNameCapacity = 32;
- private char [] valueBuffer;
- private int valueLength;
- private int valueCapacity;
- private const int initialValueCapacity = 256;
- private char [] currentTagBuffer;
- private int currentTagLength;
- private int currentTagCapacity;
- private const int initialCurrentTagCapacity = 256;
- private TextReader reader;
- private char [] peekChars;
- private int peekCharsIndex;
- private int peekCharsLength;
- private const int peekCharCapacity = 1024;
- private int line;
- private int column;
- private int currentLinkedNodeLineNumber;
- private int currentLinkedNodeLinePosition;
- private bool useProceedingLineInfo;
- private XmlNodeType startNodeType;
- // State machine attribute.
- // XmlDeclaration: after the first node.
- // DocumentType: after doctypedecl
- // Element: inside document element
- // EndElement: after document element
- private XmlNodeType currentState;
- // For ReadChars()/ReadBase64()/ReadBinHex()
- private bool shouldSkipUntilEndTag;
- private byte [] base64Cache = new byte [3];
- private int base64CacheStartsAt;
- // These values are never re-initialized.
- private bool namespaces = true;
- private WhitespaceHandling whitespaceHandling = WhitespaceHandling.All;
- private XmlResolver resolver = new XmlUrlResolver ();
- private bool normalization = false;
- private void Init ()
- {
- currentToken = new XmlTokenInfo (this, true);
- cursorToken = currentToken;
- currentAttribute = -1;
- currentAttributeValue = -1;
- attributeCount = 0;
- readState = ReadState.Initial;
- allowMultipleRoot = false;
- depth = 0;
- elementDepth = 0;
- depthUp = false;
- popScope = allowMultipleRoot = false;
- elementNames = new string [10];
- elementNameStackPos = 0;
- isStandalone = false;
- returnEntityReference = false;
- entityReferenceName = String.Empty;
- nameBuffer = new char [initialNameCapacity];
- nameLength = 0;
- nameCapacity = initialNameCapacity;
- valueBuffer = new char [initialValueCapacity];
- valueLength = 0;
- valueCapacity = initialValueCapacity;
- currentTagBuffer = new char [initialCurrentTagCapacity];
- currentTagLength = 0;
- currentTagCapacity = initialCurrentTagCapacity;
- peekCharsIndex = 0;
- peekCharsLength = 0;
- if (peekChars == null)
- peekChars = new char [peekCharCapacity];
- line = 1;
- column = 0;
- currentTagLength = 0;
- currentLinkedNodeLineNumber = currentLinkedNodeLinePosition = 0;
- useProceedingLineInfo = false;
- currentState = XmlNodeType.None;
- shouldSkipUntilEndTag = false;
- base64CacheStartsAt = -1;
- }
- private void InitializeContext (string url, XmlParserContext context, TextReader fragment, XmlNodeType fragType)
- {
- startNodeType = fragType;
- parserContext = context;
- if (context == null) {
- XmlNameTable nt = new NameTable ();
- parserContext = new XmlParserContext (nt,
- new XmlNamespaceManager (nt),
- String.Empty,
- XmlSpace.None);
- }
- if (url != null && url.Length > 0) {
- Uri uri = null;
- try {
- uri = new Uri (url);
- } catch (Exception) {
- string path = Path.GetFullPath ("./a");
- uri = new Uri (new Uri (path), url);
- }
- parserContext.BaseURI = uri.ToString ();
- }
- Init ();
- switch (fragType) {
- case XmlNodeType.Attribute:
- fragment = new StringReader (fragment.ReadToEnd ().Replace ("\"", """));
- break;
- case XmlNodeType.Element:
- currentState = XmlNodeType.Element;
- allowMultipleRoot = true;
- break;
- case XmlNodeType.Document:
- break;
- default:
- throw new XmlException (String.Format ("NodeType {0} is not allowed to create XmlTextReader.", fragType));
- }
- reader = fragment;
- }
- // Use this method rather than setting the properties
- // directly so that all the necessary properties can
- // be changed in harmony with each other. Maybe the
- // fields should be in a seperate class to help enforce
- // this.
- private void SetProperties (
- XmlNodeType nodeType,
- string name,
- bool isEmptyElement,
- string value,
- bool clearAttributes)
- {
- SetProperties (currentToken, nodeType, name, isEmptyElement, value, clearAttributes);
- currentToken.LineNumber = this.currentLinkedNodeLineNumber;
- currentToken.LinePosition = this.currentLinkedNodeLinePosition;
- }
- private void SetProperties (
- XmlTokenInfo token,
- XmlNodeType nodeType,
- string name,
- bool isEmptyElement,
- string value,
- bool clearAttributes)
- {
- token.Clear ();
- token.NodeType = nodeType;
- token.Name = name;
- token.IsEmptyElement = isEmptyElement;
- token.Value = value;
- this.elementDepth = depth;
- if (clearAttributes)
- ClearAttributes ();
- token.FillNames ();
- }
- private void ClearAttributes ()
- {
- for (int i = 0; i < attributeCount; i++)
- attributeTokens [i].Clear ();
- attributeCount = 0;
- currentAttribute = -1;
- currentAttributeValue = -1;
- }
- private int PeekChar ()
- {
- if (peekCharsLength == peekCharsIndex) {
- if (!ReadTextReader ())
- return -1;
- return PeekChar ();
- }
- else
- return peekChars [peekCharsIndex];
- }
- private int ReadChar ()
- {
- int ch;
- if (peekCharsLength == peekCharsIndex) {
- if (!ReadTextReader ())
- return -1;
- return ReadChar ();
- }
- ch = peekChars [peekCharsIndex++];
- if (ch == '\n') {
- line++;
- column = 1;
- } else {
- column++;
- }
- if (currentState != XmlNodeType.Element)
- AppendCurrentTagChar (ch);
- return ch;
- }
- private bool ReadTextReader ()
- {
- peekCharsIndex = 0;
- peekCharsLength = reader.Read (peekChars, 0, peekCharCapacity);
- if (peekCharsLength == 0)
- return false;
- return true;
- }
- private string ExpandSurrogateChar (int ch)
- {
- if (ch < Char.MaxValue)
- return ((char) ch).ToString ();
- else {
- char [] tmp = new char [] {(char) (ch / 0x10000 + 0xD800 - 1), (char) (ch % 0x10000 + 0xDC00)};
- return new string (tmp);
- }
- }
- // This should really keep track of some state so
- // that it's not possible to have more than one document
- // element or text outside of the document element.
- private bool ReadContent ()
- {
- currentTagLength = 0;
- if (popScope) {
- parserContext.NamespaceManager.PopScope ();
- popScope = false;
- }
- if (returnEntityReference)
- SetEntityReferenceProperties ();
- else {
- switch (PeekChar ()) {
- case '<':
- ReadChar ();
- ReadTag ();
- break;
- case '\r': goto case ' ';
- case '\n': goto case ' ';
- case '\t': goto case ' ';
- case ' ':
- if (whitespaceHandling == WhitespaceHandling.All ||
- whitespaceHandling == WhitespaceHandling.Significant)
- ReadWhitespace ();
- else {
- SkipWhitespace ();
- return ReadContent ();
- }
- break;
- case -1:
- readState = ReadState.EndOfFile;
- ClearValueBuffer ();
- SetProperties (
- XmlNodeType.None, // nodeType
- String.Empty, // name
- false, // isEmptyElement
- null, // value
- true // clearAttributes
- );
- if (depth > 0)
- throw new XmlException ("unexpected end of file. Current depth is " + depth);
- return false;
- default:
- ReadText (true);
- break;
- }
- }
- return this.ReadState != ReadState.EndOfFile;
- }
- private void SetEntityReferenceProperties ()
- {
- DTDEntityDeclaration decl = DTD != null ? DTD.EntityDecls [entityReferenceName] : null;
- if (this.isStandalone)
- if (DTD == null || decl == null || !decl.IsInternalSubset)
- throw new XmlException (this as IXmlLineInfo,
- "Standalone document must not contain any references to an non-internally declared entity.");
- if (decl != null && decl.NotationName != null)
- throw new XmlException (this as IXmlLineInfo,
- "Reference to any unparsed entities is not allowed here.");
- ClearValueBuffer ();
- SetProperties (
- XmlNodeType.EntityReference, // nodeType
- entityReferenceName, // name
- false, // isEmptyElement
- null, // value
- true // clearAttributes
- );
- returnEntityReference = false;
- entityReferenceName = String.Empty;
- }
- // The leading '<' has already been consumed.
- private void ReadTag ()
- {
- switch (PeekChar ())
- {
- case '/':
- ReadChar ();
- ReadEndTag ();
- break;
- case '?':
- ReadChar ();
- ReadProcessingInstruction ();
- break;
- case '!':
- ReadChar ();
- ReadDeclaration ();
- break;
- default:
- ReadStartTag ();
- break;
- }
- }
- // The leading '<' has already been consumed.
- private void ReadStartTag ()
- {
- if (currentState == XmlNodeType.EndElement)
- throw new XmlException (this as IXmlLineInfo,
- "Element cannot appear in this state.");
- currentState = XmlNodeType.Element;
- parserContext.NamespaceManager.PushScope ();
- string name = ReadName ();
- if (currentState == XmlNodeType.EndElement)
- throw new XmlException (this as IXmlLineInfo,"document has terminated, cannot open new element");
- bool isEmptyElement = false;
- ClearAttributes ();
- SkipWhitespace ();
- if (XmlChar.IsFirstNameChar (PeekChar ()))
- ReadAttributes (false);
- cursorToken = this.currentToken;
- // fill namespaces
- for (int i = 0; i < attributeCount; i++)
- attributeTokens [i].FillNames ();
- // quick name check
- for (int i = 0; i < attributeCount; i++)
- for (int j = i + 1; j < attributeCount; j++)
- if (Object.ReferenceEquals (attributeTokens [i].Name, attributeTokens [j].Name) ||
- (Object.ReferenceEquals (attributeTokens [i].LocalName, attributeTokens [j].LocalName) &&
- Object.ReferenceEquals (attributeTokens [i].NamespaceURI, attributeTokens [j].NamespaceURI)))
- throw new XmlException (this as IXmlLineInfo,
- "Attribute name and qualified name must be identical.");
- string baseUri = GetAttribute ("xml:base");
- if (baseUri != null) {
- if (this.resolver != null)
- parserContext.BaseURI = resolver.ResolveUri (new Uri (BaseURI), baseUri).ToString ();
- else
- parserContext.BaseURI = baseUri;
- }
- string xmlLang = GetAttribute ("xml:lang");
- if (xmlLang != null)
- parserContext.XmlLang = xmlLang;
- string xmlSpaceAttr = GetAttribute ("xml:space");
- if (xmlSpaceAttr != null) {
- if (xmlSpaceAttr == "preserve")
- parserContext.XmlSpace = XmlSpace.Preserve;
- else if (xmlSpaceAttr == "default")
- parserContext.XmlSpace = XmlSpace.Default;
- else
- throw new XmlException (this as IXmlLineInfo,String.Format ("Invalid xml:space value: {0}", xmlSpaceAttr));
- }
- if (PeekChar () == '/') {
- ReadChar ();
- isEmptyElement = true;
- popScope = true;
- }
- else {
- depthUp = true;
- PushElementName (name);
- parserContext.PushScope ();
- }
- Expect ('>');
- SetProperties (
- XmlNodeType.Element, // nodeType
- name, // name
- isEmptyElement, // isEmptyElement
- null, // value
- false // clearAttributes
- );
- }
- private void PushElementName (string name)
- {
- if (elementNames.Length == elementNameStackPos) {
- string [] newArray = new string [elementNames.Length * 2];
- Array.Copy (elementNames, 0, newArray, 0, elementNameStackPos);
- elementNames = newArray;
- }
- elementNames [elementNameStackPos++] = name;
- }
- // The reader is positioned on the first character
- // of the element's name.
- private void ReadEndTag ()
- {
- if (currentState != XmlNodeType.Element)
- throw new XmlException (this as IXmlLineInfo,
- "End tag cannot appear in this state.");
- string name = ReadName ();
- if (elementNameStackPos == 0)
- throw new XmlException (this as IXmlLineInfo,"closing element without matching opening element");
- string expected = elementNames [--elementNameStackPos];
- if (expected != name)
- throw new XmlException (this as IXmlLineInfo,String.Format ("unmatched closing element: expected {0} but found {1}", expected, name));
- parserContext.PopScope ();
- ExpectAfterWhitespace ('>');
- --depth;
- SetProperties (
- XmlNodeType.EndElement, // nodeType
- name, // name
- false, // isEmptyElement
- null, // value
- true // clearAttributes
- );
- popScope = true;
- }
- private void AppendNameChar (int ch)
- {
- if (nameLength == nameCapacity)
- ExpandNameCapacity ();
- if (ch < Char.MaxValue)
- nameBuffer [nameLength++] = (char) ch;
- else {
- nameBuffer [nameLength++] = (char) (ch / 0x10000 + 0xD800 - 1);
- if (nameLength == nameCapacity)
- ExpandNameCapacity ();
- nameBuffer [nameLength++] = (char) (ch % 0x10000 + 0xDC00);
- }
- }
- private void ExpandNameCapacity ()
- {
- nameCapacity = nameCapacity * 2;
- char [] oldNameBuffer = nameBuffer;
- nameBuffer = new char [nameCapacity];
- Array.Copy (oldNameBuffer, nameBuffer, nameLength);
- }
- private string CreateNameString ()
- {
- return parserContext.NameTable.Add (nameBuffer, 0, nameLength);
- }
- private void AppendValueChar (int ch)
- {
- if (valueLength == valueCapacity)
- ExpandValueCapacity ();
- if (ch < Char.MaxValue)
- valueBuffer [valueLength++] = (char) ch;
- else {
- valueBuffer [valueLength++] = (char) (ch / 0x10000 + 0xD800 - 1);
- if (valueLength == valueCapacity)
- ExpandValueCapacity ();
- valueBuffer [valueLength++] = (char) (ch % 0x10000 + 0xDC00);
- }
- }
- private void ExpandValueCapacity ()
- {
- valueCapacity = valueCapacity * 2;
- char [] oldValueBuffer = valueBuffer;
- valueBuffer = new char [valueCapacity];
- Array.Copy (oldValueBuffer, valueBuffer, valueLength);
- }
- private string CreateValueString ()
- {
- return new string (valueBuffer, 0, valueLength);
- }
- private void ClearValueBuffer ()
- {
- valueLength = 0;
- }
- private void AppendCurrentTagChar (int ch)
- {
- if (currentTagLength == currentTagCapacity)
- ExpandCurrentTagCapacity ();
- if (ch < Char.MaxValue)
- currentTagBuffer [currentTagLength++] = (char) ch;
- else {
- currentTagBuffer [currentTagLength++] = (char) (ch / 0x10000 + 0xD800 - 1);
- if (currentTagLength == currentTagCapacity)
- ExpandCurrentTagCapacity ();
- currentTagBuffer [currentTagLength++] = (char) (ch % 0x10000 + 0xDC00);
- }
- }
- private void ExpandCurrentTagCapacity ()
- {
- currentTagCapacity = currentTagCapacity * 2;
- char [] oldCurrentTagBuffer = currentTagBuffer;
- currentTagBuffer = new char [currentTagCapacity];
- Array.Copy (oldCurrentTagBuffer, currentTagBuffer, currentTagLength);
- }
- private string CreateCurrentTagString ()
- {
- return new string (currentTagBuffer, 0, currentTagLength);
- }
- private void ClearCurrentTagBuffer ()
- {
- currentTagLength = 0;
- }
- // The reader is positioned on the first character
- // of the text.
- private void ReadText (bool notWhitespace)
- {
- if (currentState != XmlNodeType.Element)
- throw new XmlException (this as IXmlLineInfo,
- "Text node cannot appear in this state.");
- if (notWhitespace)
- ClearValueBuffer ();
- int ch = PeekChar ();
- bool previousWasCloseBracket = false;
- while (ch != '<' && ch != -1) {
- if (ch == '&') {
- ReadChar ();
- ch = ReadReference (false);
- if (returnEntityReference) // Returns -1 if char validation should not be done
- break;
- }
- else
- ch = ReadChar ();
- if (normalization && XmlChar.IsInvalid (ch))
- throw new XmlException (this, "Not allowed character was found.");
- AppendValueChar (ch);
- // Block "]]>"
- if (ch == ']') {
- if (previousWasCloseBracket)
- if (PeekChar () == '>')
- throw new XmlException (this as IXmlLineInfo,
- "Inside text content, character sequence ']]>' is not allowed.");
- previousWasCloseBracket = true;
- }
- else if (previousWasCloseBracket)
- previousWasCloseBracket = false;
- ch = PeekChar ();
- notWhitespace = true;
- }
- if (returnEntityReference && valueLength == 0) {
- SetEntityReferenceProperties ();
- } else {
- XmlNodeType nodeType = notWhitespace ? XmlNodeType.Text :
- this.XmlSpace == XmlSpace.Preserve ? XmlNodeType.SignificantWhitespace : XmlNodeType.Whitespace;
- SetProperties (
- nodeType, // nodeType
- String.Empty, // name
- false, // isEmptyElement
- null, // value: create only when required
- true // clearAttributes
- );
- }
- }
- // The leading '&' has already been consumed.
- // Returns true if the entity reference isn't a simple
- // character reference or one of the predefined entities.
- // This allows the ReadText method to break so that the
- // next call to Read will return the EntityReference node.
- private int ReadReference (bool ignoreEntityReferences)
- {
- if (PeekChar () == '#') {
- ReadChar ();
- return ReadCharacterReference ();
- } else
- return ReadEntityReference (ignoreEntityReferences);
- }
- private int ReadCharacterReference ()
- {
- int value = 0;
- if (PeekChar () == 'x') {
- ReadChar ();
- while (PeekChar () != ';' && PeekChar () != -1) {
- int ch = ReadChar ();
- if (ch >= '0' && ch <= '9')
- value = (value << 4) + ch - '0';
- else if (ch >= 'A' && ch <= 'F')
- value = (value << 4) + ch - 'A' + 10;
- else if (ch >= 'a' && ch <= 'f')
- value = (value << 4) + ch - 'a' + 10;
- else
- throw new XmlException (this as IXmlLineInfo,
- String.Format (
- "invalid hexadecimal digit: {0} (#x{1:X})",
- (char) ch,
- ch));
- }
- } else {
- while (PeekChar () != ';' && PeekChar () != -1) {
- int ch = ReadChar ();
- if (ch >= '0' && ch <= '9')
- value = value * 10 + ch - '0';
- else
- throw new XmlException (this as IXmlLineInfo,
- String.Format (
- "invalid decimal digit: {0} (#x{1:X})",
- (char) ch,
- ch));
- }
- }
- ReadChar (); // ';'
- // There is no way to save surrogate pairs...
- if (normalization && XmlChar.IsInvalid (value))
- throw new XmlException (this as IXmlLineInfo,
- "Referenced character was not allowed in XML.");
- return value;
- }
- // Returns -1 if it should not be validated.
- // Real EOF must not be detected here.
- private int ReadEntityReference (bool ignoreEntityReferences)
- {
- string name = ReadName ();
- Expect (';');
- int predefined = XmlChar.GetPredefinedEntity (name);
- if (predefined >= 0)
- return predefined;
- else {
- if (ignoreEntityReferences) {
- AppendValueChar ('&');
- for (int i = 0; i < name.Length; i++)
- AppendValueChar (name [i]);
- AppendValueChar (';');
- } else {
- returnEntityReference = true;
- entityReferenceName = name;
- }
- }
- return -1;
- }
- // The reader is positioned on the first character of
- // the attribute name.
- private void ReadAttributes (bool isXmlDecl)
- {
- int peekChar = -1;
- bool requireWhitespace = false;
- currentAttribute = -1;
- currentAttributeValue = -1;
- do {
- if (!SkipWhitespace () && requireWhitespace)
- throw new XmlException ("Unexpected token. Name is required here.");
- IncrementAttributeToken ();
- currentAttributeToken.LineNumber = line;
- currentAttributeToken.LinePosition = column;
- currentAttributeToken.LocalName =
- currentAttributeToken.Name = ReadName ();
- ExpectAfterWhitespace ('=');
- SkipWhitespace ();
- ReadAttributeValueTokens (-1);
- attributeCount++;
- if (currentAttributeToken.Name == "xmlns")
- parserContext.NamespaceManager.AddNamespace (String.Empty, GetAttribute (currentAttribute));
- else if (currentAttributeToken.Name.StartsWith ("xmlns:")) {
- string nsPrefix = currentAttributeToken.Name.Substring (6);
- parserContext.NamespaceManager.AddNamespace (nsPrefix, GetAttribute (currentAttribute));
- }
- if (!SkipWhitespace ())
- requireWhitespace = true;
- peekChar = PeekChar ();
- if (isXmlDecl) {
- if (peekChar == '?')
- break;
- }
- else if (peekChar == '/' || peekChar == '>')
- break;
- } while (peekChar != -1);
- currentAttribute = -1;
- currentAttributeValue = -1;
- }
- private void AddAttribute (string name, string value)
- {
- IncrementAttributeToken ();
- XmlAttributeTokenInfo ati = attributeTokens [currentAttribute];
- ati.Name = "SYSTEM";
- ati.FillNames ();
- IncrementAttributeValueToken ();
- XmlTokenInfo vti = attributeValueTokens [currentAttributeValue];
- vti.Value = value;
- SetProperties (vti, XmlNodeType.Text, String.Empty, false, value, false);
- attributeCount++;
- }
- private void IncrementAttributeToken ()
- {
- currentAttribute++;
- if (attributeTokens.Length == currentAttribute) {
- XmlAttributeTokenInfo [] newArray =
- new XmlAttributeTokenInfo [attributeTokens.Length * 2];
- attributeTokens.CopyTo (newArray, 0);
- attributeTokens = newArray;
- }
- if (attributeTokens [currentAttribute] == null)
- attributeTokens [currentAttribute] = new XmlAttributeTokenInfo (this);
- currentAttributeToken = attributeTokens [currentAttribute];
- currentAttributeToken.Clear ();
- }
- private void IncrementAttributeValueToken ()
- {
- ClearValueBuffer ();
- currentAttributeValue++;
- if (attributeValueTokens.Length == currentAttributeValue) {
- XmlTokenInfo [] newArray = new XmlTokenInfo [attributeValueTokens.Length * 2];
- attributeValueTokens.CopyTo (newArray, 0);
- attributeValueTokens = newArray;
- }
- if (attributeValueTokens [currentAttributeValue] == null)
- attributeValueTokens [currentAttributeValue] = new XmlTokenInfo (this, false);
- currentAttributeValueToken = attributeValueTokens [currentAttributeValue];
- currentAttributeValueToken.Clear ();
- }
- // FIXME: normalize here
- private void ReadAttributeValueTokens (int dummyQuoteChar)
- {
- int quoteChar = (dummyQuoteChar < 0) ? ReadChar () : dummyQuoteChar;
- if (quoteChar != '\'' && quoteChar != '\"')
- throw new XmlException (this as IXmlLineInfo,"an attribute value was not quoted");
- currentAttributeToken.QuoteChar = (char) quoteChar;
- IncrementAttributeValueToken ();
- currentAttributeToken.ValueTokenStartIndex = currentAttributeValue;
- currentAttributeValueToken.LineNumber = line;
- currentAttributeValueToken.LinePosition = column;
- bool incrementToken = false;
- bool isNewToken = true;
- bool loop = true;
- int ch = 0;
- while (loop) {
- ch = ReadChar ();
- if (ch == quoteChar)
- break;
- if (incrementToken) {
- IncrementAttributeValueToken ();
- currentAttributeValueToken.LineNumber = line;
- currentAttributeValueToken.LinePosition = column;
- incrementToken = false;
- isNewToken = true;
- }
- switch (ch)
- {
- case '<':
- throw new XmlException (this as IXmlLineInfo,"attribute values cannot contain '<'");
- case -1:
- if (dummyQuoteChar < 0)
- throw new XmlException (this as IXmlLineInfo,"unexpected end of file in an attribute value");
- else // Attribute value constructor.
- loop = false;
- break;
- case '&':
- int startPosition = currentTagLength - 1;
- if (PeekChar () == '#') {
- ReadChar ();
- ch = ReadCharacterReference ();
- if (normalization && XmlChar.IsInvalid (ch))
- throw new XmlException (this as IXmlLineInfo,
- "Not allowed character was found.");
- AppendValueChar (ch);
- break;
- }
- // Check XML 1.0 section 3.1 WFC.
- string entName = ReadName ();
- Expect (';');
- int predefined = XmlChar.GetPredefinedEntity (entName);
- if (predefined < 0) {
- CheckAttributeEntityReferenceWFC (entName);
- currentAttributeValueToken.Value = CreateValueString ();
- currentAttributeValueToken.NodeType = XmlNodeType.Text;
- if (!isNewToken)
- IncrementAttributeValueToken ();
- currentAttributeValueToken.Name = entName;
- currentAttributeValueToken.Value = String.Empty;
- currentAttributeValueToken.NodeType = XmlNodeType.EntityReference;
- incrementToken = true;
- }
- else
- AppendValueChar (predefined);
- break;
- default:
- if (normalization && XmlChar.IsInvalid (ch))
- throw new XmlException (this, "Invalid character was found.");
- AppendValueChar (ch);
- break;
- }
- isNewToken = false;
- }
- if (!incrementToken) {
- currentAttributeValueToken.Value = CreateValueString ();
- currentAttributeValueToken.NodeType = XmlNodeType.Text;
- }
- currentAttributeToken.ValueTokenEndIndex = currentAttributeValue;
- }
- private void CheckAttributeEntityReferenceWFC (string entName)
- {
- DTDEntityDeclaration entDecl =
- DTD == null ? null : DTD.EntityDecls [entName];
- if (DTD != null && resolver != null && entDecl == null)
- throw new XmlException (this, "Referenced entity does not exist.");
- if (entDecl == null)
- return;
- if (entDecl.HasExternalReference)
- throw new XmlException (this, "Reference to external entities is not allowed in the value of an attribute.");
- if (isStandalone && !entDecl.IsInternalSubset)
- throw new XmlException (this, "Reference to external entities is not allowed in the internal subset.");
- if (entDecl.EntityValue.IndexOf ('<') >= 0)
- throw new XmlException (this, "Attribute must not contain character '<' either directly or indirectly by way of entity references.");
- }
- // The reader is positioned on the first character
- // of the target.
- //
- // It may be xml declaration or processing instruction.
- private void ReadProcessingInstruction ()
- {
- string target = ReadName ();
- if (target == "xml") {
- ReadXmlDeclaration ();
- return;
- } else if (target.ToLower () == "xml")
- throw new XmlException (this as IXmlLineInfo,
- "Not allowed processing instruction name which starts with 'X', 'M', 'L' was found.");
- if (currentState == XmlNodeType.None)
- currentState = XmlNodeType.XmlDeclaration;
- if (!SkipWhitespace ())
- if (PeekChar () != '?')
- throw new XmlException (this as IXmlLineInfo,
- "Invalid processing instruction name was found.");
- ClearValueBuffer ();
- while (PeekChar () != -1) {
- int ch = ReadChar ();
- if (ch == '?' && PeekChar () == '>') {
- ReadChar ();
- break;
- }
- if (normalization && XmlChar.IsInvalid (ch))
- throw new XmlException (this, "Invalid character was found.");
- AppendValueChar (ch);
- }
- SetProperties (
- XmlNodeType.ProcessingInstruction, // nodeType
- target, // name
- false, // isEmptyElement
- null, // value: create only when required
- true // clearAttributes
- );
- }
- // The reader is positioned after "<?xml "
- private void ReadXmlDeclaration ()
- {
- if (currentState != XmlNodeType.None) {
- throw new XmlException (this as IXmlLineInfo,
- "XML declaration cannot appear in this state.");
- }
- currentState = XmlNodeType.XmlDeclaration;
- ClearAttributes ();
- ReadAttributes (true); // They must have "version."
- string version = GetAttribute ("version");
- string message = null;
- if (attributeTokens [0].Name != "version" || version != "1.0")
- message = "Version 1.0 declaration is required in XML Declaration.";
- else if (attributeCount > 1 &&
- (attributeTokens [1].Name != "encoding" &&
- attributeTokens [1].Name != "standalone"))
- message = "Invalid Xml Declaration markup was found.";
- else if (attributeCount > 2 && attributeTokens [2].Name != "standalone")
- message = "Invalid Xml Declaration markup was found.";
- string sa = GetAttribute ("standalone");
- if (sa != null && sa != "yes" && sa != "no")
- message = "Only 'yes' or 'no' is allowed for standalone.";
- this.isStandalone = (sa == "yes");
- if (message != null)
- throw new XmlException (this as IXmlLineInfo, message);
- SetProperties (
- XmlNodeType.XmlDeclaration, // nodeType
- "xml", // name
- false, // isEmptyElement
- new string (currentTagBuffer, 6, currentTagLength - 6), // value
- false // clearAttributes
- );
- Expect ("?>");
- }
- internal void SkipTextDeclaration ()
- {
- this.currentState = XmlNodeType.Element;
- if (PeekChar () != '<')
- return;
- ReadChar ();
- if (PeekChar () != '?') {
- peekCharsIndex = 0;
- return;
- }
- ReadChar ();
- while (peekCharsIndex < 6) {
- if (PeekChar () < 0)
- break;
- else
- ReadChar ();
- }
- if (new string (peekChars, 2, 4) != "xml ") {
- if (new string (peekChars, 2, 3).ToLower () == "xml") {
- throw new XmlException (this as IXmlLineInfo,
- "Processing instruction name must not be character sequence 'X' 'M' 'L' with case insensitivity.");
- }
- peekCharsIndex = 0;
- return;
- }
- SkipWhitespace ();
- // version decl
- if (PeekChar () == 'v') {
- Expect ("version");
- ExpectAfterWhitespace ('=');
- SkipWhitespace ();
- int quoteChar = ReadChar ();
- char [] expect1_0 = new char [3];
- int versionLength = 0;
- switch (quoteChar) {
- case '\'':
- case '"':
- while (PeekChar () != quoteChar) {
- if (PeekChar () == -1)
- throw new XmlException (this as IXmlLineInfo,
- "Invalid version declaration inside text declaration.");
- else if (versionLength == 3)
- throw new XmlException (this as IXmlLineInfo,
- "Invalid version number inside text declaration.");
- else {
- expect1_0 [versionLength] = (char) ReadChar ();
- versionLength++;
- if (versionLength == 3 && new String (expect1_0) != "1.0")
- throw new XmlException (this as IXmlLineInfo,
- "Invalid version number inside text declaration.");
- }
- }
- ReadChar ();
- SkipWhitespace ();
- break;
- default:
- throw new XmlException (this as IXmlLineInfo,
- "Invalid version declaration inside text declaration.");
- }
- }
- if (PeekChar () == 'e') {
- Expect ("encoding");
- ExpectAfterWhitespace ('=');
- SkipWhitespace ();
- int quoteChar = ReadChar ();
- switch (quoteChar) {
- case '\'':
- case '"':
- while (PeekChar () != quoteChar)
- if (ReadChar () == -1)
- throw new XmlException (this as IXmlLineInfo,
- "Invalid encoding declaration inside text declaration.");
- ReadChar ();
- SkipWhitespace ();
- break;
- default:
- throw new XmlException (this as IXmlLineInfo,
- "Invalid encoding declaration inside text declaration.");
- }
- // Encoding value should be checked inside XmlInputStream.
- }
- else
- throw new XmlException (this as IXmlLineInfo,
- "Encoding declaration is mandatory in text declaration.");
- Expect ("?>");
- }
- // The reader is positioned on the first character after
- // the leading '<!'.
- private void ReadDeclaration ()
- {
- int ch = PeekChar ();
- switch (ch)
- {
- case '-':
- Expect ("--");
- ReadComment ();
- break;
- case '[':
- ReadChar ();
- Expect ("CDATA[");
- ReadCDATA ();
- break;
- case 'D':
- Expect ("DOCTYPE");
- ReadDoctypeDecl ();
- break;
- default:
- throw new XmlException (this as IXmlLineInfo,
- "Unexpected declaration markup was found.");
- }
- }
- // The reader is positioned on the first character after
- // the leading '<!--'.
- private void ReadComment ()
- {
- if (currentState == XmlNodeType.None)
- currentState = XmlNodeType.XmlDeclaration;
- ClearValueBuffer ();
- while (PeekChar () != -1) {
- int ch = ReadChar ();
- if (ch == '-' && PeekChar () == '-') {
- ReadChar ();
- if (PeekChar () != '>')
- throw new XmlException (this as IXmlLineInfo,"comments cannot contain '--'");
- ReadChar ();
- break;
- }
- if (XmlChar.IsInvalid (ch))
- throw new XmlException (this as IXmlLineInfo,
- "Not allowed character was found.");
- AppendValueChar (ch);
- }
- SetProperties (
- XmlNodeType.Comment, // nodeType
- String.Empty, // name
- false, // isEmptyElement
- null, // value: create only when required
- true // clearAttributes
- );
- }
- // The reader is positioned on the first character after
- // the leading '<![CDATA['.
- private void ReadCDATA ()
- {
- if (currentState != XmlNodeType.Element)
- throw new XmlException (this as IXmlLineInfo,
- "CDATA section cannot appear in this state.");
- ClearValueBuffer ();
- bool skip = false;
- int ch = 0;
- while (PeekChar () != -1) {
- if (!skip)
- ch = ReadChar ();
- skip = false;
- if (ch == ']' && PeekChar () == ']') {
- ch = ReadChar (); // ']'
- if (PeekChar () == '>') {
- ReadChar (); // '>'
- break;
- } else {
- skip = true;
- }
- }
- if (normalization && XmlChar.IsInvalid (ch))
- throw new XmlException (this, "Invalid character was found.");
- AppendValueChar (ch);
- }
- SetProperties (
- XmlNodeType.CDATA, // nodeType
- String.Empty, // name
- false, // isEmptyElement
- null, // value: create only when required
- true // clearAttributes
- );
- }
- // The reader is positioned on the first character after
- // the leading '<!DOCTYPE'.
- private void ReadDoctypeDecl ()
- {
- switch (currentState) {
- case XmlNodeType.DocumentType:
- case XmlNodeType.Element:
- case XmlNodeType.EndElement:
- throw new XmlException (this as IXmlLineInfo,
- "Document type cannot appear in this state.");
- }
- currentState = XmlNodeType.DocumentType;
- string doctypeName = null;
- string publicId = null;
- string systemId = null;
- int intSubsetStartLine = 0;
- int intSubsetStartColumn = 0;
- SkipWhitespace ();
- doctypeName = ReadName ();
- SkipWhitespace ();
- switch(PeekChar ())
- {
- case 'S':
- systemId = ReadSystemLiteral (true);
- break;
- case 'P':
- publicId = ReadPubidLiteral ();
- if (!SkipWhitespace ())
- throw new XmlException (this as IXmlLineInfo,
- "Whitespace is required between PUBLIC id and SYSTEM id.");
- systemId = ReadSystemLiteral (false);
- break;
- }
- SkipWhitespace ();
- if(PeekChar () == '[')
- {
- // read markupdecl etc. or end of decl
- ReadChar ();
- intSubsetStartLine = this.LineNumber;
- intSubsetStartColumn = this.LinePosition;
- int startPos = currentTagLength;
- ReadInternalSubset ();
- int endPos = currentTagLength - 1;
- parserContext.InternalSubset = new string (currentTagBuffer, startPos, endPos - startPos);
- }
- // end of DOCTYPE decl.
- ExpectAfterWhitespace ('>');
- GenerateDTDObjectModel (doctypeName, publicId,
- systemId, parserContext.InternalSubset,
- intSubsetStartLine, intSubsetStartColumn);
- // set properties for <!DOCTYPE> node
- SetProperties (
- XmlNodeType.DocumentType, // nodeType
- doctypeName, // name
- false, // isEmptyElement
- parserContext.InternalSubset, // value
- true // clearAttributes
- );
- if (publicId != null)
- AddAttribute ("PUBLIC", publicId);
- if (systemId != null)
- AddAttribute ("SYSTEM", systemId);
- currentAttribute = currentAttributeValue = -1;
- }
- internal DTDObjectModel GenerateDTDObjectModel (string name, string publicId,
- string systemId, string internalSubset)
- {
- return GenerateDTDObjectModel (name, publicId, systemId, internalSubset, 0, 0);
- }
- internal DTDObjectModel GenerateDTDObjectModel (string name, string publicId,
- string systemId, string internalSubset, int intSubsetStartLine, int intSubsetStartColumn)
- {
- // now compile DTD
- parserContext.Dtd = new DTDObjectModel (this.NameTable); // merges both internal and external subsets in the meantime,
- DTD.BaseURI = BaseURI;
- DTD.Name = name;
- DTD.PublicId = publicId;
- DTD.SystemId = systemId;
- DTD.InternalSubset = internalSubset;
- DTD.XmlResolver = resolver;
- DTD.IsStandalone = isStandalone;
- DTD.LineNumber = line;
- DTD.LinePosition = column;
- DTDReader dr = new DTDReader (DTD, intSubsetStartLine, intSubsetStartColumn);
- dr.Normalization = this.normalization;
- #if DTD_HANDLE_EVENTS
- dr.ValidationEventHandler += new ValidationEventHandler (OnValidationEvent);
- #endif
- return dr.GenerateDTDObjectModel ();
- }
- private void OnValidationEvent (object o, ValidationEventArgs e)
- {
- #if DTD_HANDLE_EVENTS
- if (ValidationEventHandler != null)
- // Override object as this.
- ValidationEventHandler (this, e);
- #endif
- }
- private enum DtdInputState
- {
- Free = 1,
- ElementDecl,
- AttlistDecl,
- EntityDecl,
- NotationDecl,
- PI,
- Comment,
- InsideSingleQuoted,
- InsideDoubleQuoted,
- }
- private class DtdInputStateStack
- {
- Stack intern = new Stack ();
- public DtdInputStateStack ()
- {
- Push (DtdInputState.Free);
- }
- public DtdInputState Peek ()
- {
- return (DtdInputState) intern.Peek ();
- }
- public DtdInputState Pop ()
- {
- return (DtdInputState) intern.Pop ();
- }
- public void Push (DtdInputState val)
- {
- intern.Push (val);
- }
- }
- DtdInputStateStack stateStack = new DtdInputStateStack ();
- DtdInputState State {
- get { return stateStack.Peek (); }
- }
- // Simply read but not generate any result.
- private void ReadInternalSubset ()
- {
- bool continueParse = true;
- while (continueParse) {
- switch (ReadChar ()) {
- case ']':
- switch (State) {
- case DtdInputState.Free:
- continueParse = false;
- break;
- case DtdInputState.InsideDoubleQuoted:
- continue;
- case DtdInputState.InsideSingleQuoted:
- continue;
- default:
- throw new XmlException (this as IXmlLineInfo,"unexpected end of file at DTD.");
- }
- break;
- case -1:
- throw new XmlException (this as IXmlLineInfo,"unexpected end of file at DTD.");
- case '<':
- if (State == DtdInputState.InsideDoubleQuoted ||
- State == DtdInputState.InsideSingleQuoted)
- continue; // well-formed
- switch (ReadChar ()) {
- case '?':
- stateStack.Push (DtdInputState.PI);
- break;
- case '!':
- switch (ReadChar ()) {
- case 'E':
- switch (ReadChar ()) {
- case 'L':
- Expect ("EMENT");
- stateStack.Push (DtdInputState.ElementDecl);
- break;
- case 'N':
- Expect ("TITY");
- stateStack.Push (DtdInputState.EntityDecl);
- break;
- default:
- throw new XmlException (this as IXmlLineInfo,"unexpected token '<!E'.");
- }
- break;
- case 'A':
- Expect ("TTLIST");
- stateStack.Push (DtdInputState.AttlistDecl);
- break;
- case 'N':
- Expect ("OTATION");
- stateStack.Push (DtdInputState.NotationDecl);
- break;
- case '-':
- Expect ("-");
- stateStack.Push (DtdInputState.Comment);
- break;
- }
- break;
- default:
- throw new XmlException (this as IXmlLineInfo,"unexpected '>'.");
- }
- break;
- case '\'':
- if (State == DtdInputState.InsideSingleQuoted)
- stateStack.Pop ();
- else if (State != DtdInputState.InsideDoubleQuoted && State != DtdInputState.Comment)
- stateStack.Push (DtdInputState.InsideSingleQuoted);
- break;
- case '"':
- if (State == DtdInputState.InsideDoubleQuoted)
- stateStack.Pop ();
- else if (State != DtdInputState.InsideSingleQuoted && State != DtdInputState.Comment)
- stateStack.Push (DtdInputState.InsideDoubleQuoted);
- break;
- case '>':
- switch (State) {
- case DtdInputState.ElementDecl:
- goto case DtdInputState.NotationDecl;
- case DtdInputState.AttlistDecl:
- goto case DtdInputState.NotationDecl;
- case DtdInputState.EntityDecl:
- goto case DtdInputState.NotationDecl;
- case DtdInputState.NotationDecl:
- stateStack.Pop ();
- break;
- case DtdInputState.InsideDoubleQuoted:
- continue;
- case DtdInputState.InsideSingleQuoted:
- continue; // well-formed
- case DtdInputState.Comment:
- continue;
- default:
- throw new XmlException (this as IXmlLineInfo,"unexpected token '>'");
- }
- break;
- case '?':
- if (State == DtdInputState.PI) {
- if (ReadChar () == '>')
- stateStack.Pop ();
- }
- break;
- case '-':
- if (State == DtdInputState.Comment) {
- if (PeekChar () == '-') {
- ReadChar ();
- Expect ('>');
- stateStack.Pop ();
- }
- }
- break;
- case '%':
- if (State != DtdInputState.Free && State != DtdInputState.EntityDecl && State != DtdInputState.Comment && State != DtdInputState.InsideDoubleQuoted && State != DtdInputState.InsideSingleQuoted)
- throw new XmlException (this as IXmlLineInfo,"Parameter Entity Reference cannot appear as a part of markupdecl (see XML spec 2.8).");
- break;
- }
- }
- }
- // The reader is positioned on the first 'S' of "SYSTEM".
- private string ReadSystemLiteral (bool expectSYSTEM)
- {
- if(expectSYSTEM) {
- Expect ("SYSTEM");
- if (!SkipWhitespace ())
- throw new XmlException (this as IXmlLineInfo,
- "Whitespace is required after 'SYSTEM'.");
- }
- else
- SkipWhitespace ();
- int quoteChar = ReadChar (); // apos or quot
- int startPos = currentTagLength;
- int c = 0;
- ClearValueBuffer ();
- while (c != quoteChar) {
- c = ReadChar ();
- if (c < 0)
- throw new XmlException (this as IXmlLineInfo,"Unexpected end of stream in ExternalID.");
- if (c != quoteChar)
- AppendValueChar (c);
- }
- return CreateValueString ();
- }
- private string ReadPubidLiteral()
- {
- Expect ("PUBLIC");
- if (!SkipWhitespace ())
- throw new XmlException (this as IXmlLineInfo,
- "Whitespace is required after 'PUBLIC'.");
- int quoteChar = ReadChar ();
- int startPos = currentTagLength;
- int c = 0;
- ClearValueBuffer ();
- while(c != quoteChar)
- {
- c = ReadChar ();
- if(c < 0) throw new XmlException (this as IXmlLineInfo,"Unexpected end of stream in ExternalID.");
- if(c != quoteChar && !XmlChar.IsPubidChar (c))
- throw new XmlException (this as IXmlLineInfo,"character '" + (char) c + "' not allowed for PUBLIC ID");
- if (c != quoteChar)
- AppendValueChar (c);
- }
- return CreateValueString ();
- }
- // The reader is positioned on the first character
- // of the name.
- private string ReadName ()
- {
- int ch = PeekChar ();
- if (!XmlChar.IsFirstNameChar (ch))
- throw new XmlException (this as IXmlLineInfo,String.Format ("a name did not start with a legal character {0} ({1})", ch, (char) ch));
- nameLength = 0;
- AppendNameChar (ReadChar ());
- while (XmlChar.IsNameChar (PeekChar ())) {
- AppendNameChar (ReadChar ());
- }
- return CreateNameString ();
- }
- // Read the next character and compare it against the
- // specified character.
- private void Expect (int expected)
- {
- int ch = ReadChar ();
- if (ch != expected) {
- throw new XmlException (this as IXmlLineInfo,
- String.Format (
- "expected '{0}' ({1:X}) but found '{2}' ({3:X})",
- (char) expected,
- expected,
- (char) ch,
- ch));
- }
- }
- private void Expect (string expected)
- {
- int len = expected.Length;
- for(int i=0; i< len; i++)
- Expect (expected[i]);
- }
- private void ExpectAfterWhitespace (char c)
- {
- while (true) {
- int i = ReadChar ();
- if (i < 0x21 && XmlChar.IsWhitespace (i))
- continue;
- if (c != i)
- throw new XmlException (this, String.Join (String.Empty, new string [] {"Expected ", c.ToString (), ", but found " + (char) i, "[", i.ToString (), "]"}));
- break;
- }
- }
- // Does not consume the first non-whitespace character.
- private bool SkipWhitespace ()
- {
- bool skipped = XmlChar.IsWhitespace (PeekChar ());
- if (!skipped)
- return false;
- while (XmlChar.IsWhitespace (PeekChar ()))
- ReadChar ();
- return skipped;
- }
- private void ReadWhitespace ()
- {
- if (currentState == XmlNodeType.None)
- currentState = XmlNodeType.XmlDeclaration;
- ClearValueBuffer ();
- int ch = PeekChar ();
- do {
- AppendValueChar (ReadChar ());
- } while ((ch = PeekChar ()) != -1 && XmlChar.IsWhitespace (ch));
- if (currentState == XmlNodeType.Element && ch != -1 && ch != '<')
- ReadText (false);
- else {
- XmlNodeType nodeType = (this.XmlSpace == XmlSpace.Preserve) ?
- XmlNodeType.SignificantWhitespace : XmlNodeType.Whitespace;
- SetProperties (nodeType,
- String.Empty,
- false,
- null, // value: create only when required
- true);
- }
- return;
- }
- private byte GetBase64Byte (char ch)
- {
- switch (ch) {
- case '+':
- return 62;
- case '/':
- return 63;
- case '=':
- return 0;
- default:
- if (ch >= 'A' && ch <= 'Z')
- return (byte) (ch - 'A');
- else if (ch >= 'a' && ch <= 'z')
- return (byte) (ch - 'a' + 26);
- else if (ch >= '0' && ch <= '9')
- return (byte) (ch - '0' + 52);
- else
- throw new XmlException ("Invalid Base64 character was found.");
- }
- }
- // Returns -1 if it should throw an error.
- private int ReadCharsInternal (char [] buffer, int offset, int length)
- {
- shouldSkipUntilEndTag = true;
- if (offset < 0)
- throw new ArgumentOutOfRangeException ("offset", offset, "Offset must be non-negative integer.");
- else if (length < 0)
- throw new ArgumentOutOfRangeException ("length", length, "Length must be non-negative integer.");
- else if (buffer.Length < offset + length)
- throw new ArgumentOutOfRangeException ("buffer length is smaller than the sum of offset and length.");
- if (NodeType != XmlNodeType.Element)
- return 0;
- int bufIndex = offset;
- for (int i = 0; i < length; i++) {
- int c = PeekChar ();
- switch (c) {
- case -1:
- throw new XmlException (this as IXmlLineInfo, "Unexpected end of xml.");
- case '<':
- ReadChar ();
- if (PeekChar () != '/') {
- buffer [bufIndex++] = '<';
- continue;
- }
- // Seems to skip immediate EndElement
- Expect ('/');
- string name = ReadName ();
- if (name != elementNames [elementNameStackPos - 1]) {
- if (i + 1 < length) {
- buffer [bufIndex++] = '<';
- buffer [bufIndex++] = '/';
- }
- for (int n = 0; n < name.Length && i + n + 1 < length; n++)
- buffer [bufIndex++] = name [n];
- continue;
- }
- Expect ('>');
- depth--;
- elementNames [--elementNameStackPos] = null;
- shouldSkipUntilEndTag = false;
- Read ();
- return i;
- default:
- ReadChar ();
- if (c < Char.MaxValue)
- buffer [bufIndex++] = (char) c;
- else {
- buffer [bufIndex++] = (char) (c / 0x10000 + 0xD800 - 1);
- buffer [bufIndex++] = (char) (c % 0x10000 + 0xDC00);
- }
- break;
- }
- }
- return length;
- }
- private bool ReadUntilEndTag ()
- {
- int ch;
- do {
- ch = ReadChar ();
- switch (ch) {
- case -1:
- throw new XmlException (this as IXmlLineInfo,
- "Unexpected end of xml.");
- case '<':
- if (PeekChar () != '/')
- continue;
- ReadChar ();
- string name = ReadName ();
- if (name != elementNames [elementNameStackPos - 1])
- continue;
- Expect ('>');
- depth--;
- elementNames [--elementNameStackPos] = null;
- return Read ();
- }
- } while (true);
- }
- #endregion
- }
- }
|