2 // System.Xml.XmlValidatingReader.cs
5 // Tim Coleman (tim@timcoleman.com)
6 // Atsushi Enomoto (ginga@kit.hi-ho.ne.jp)
8 // Copyright (C) Tim Coleman, 2002
9 // (C)2003 Atsushi Enomoto
10 // Copyright (C) 2005 Novell, Inc (http://www.novell.com)
12 // Permission is hereby granted, free of charge, to any person obtaining
13 // a copy of this software and associated documentation files (the
14 // "Software"), to deal in the Software without restriction, including
15 // without limitation the rights to use, copy, modify, merge, publish,
16 // distribute, sublicense, and/or sell copies of the Software, and to
17 // permit persons to whom the Software is furnished to do so, subject to
18 // the following conditions:
20 // The above copyright notice and this permission notice shall be
21 // included in all copies or substantial portions of the Software.
23 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
24 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
25 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
26 // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
27 // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
28 // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
29 // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
33 using System.Collections.Generic;
36 using System.Security.Permissions;
38 using System.Xml.Schema;
40 using Mono.Xml.Schema;
44 [PermissionSet (SecurityAction.InheritanceDemand, Unrestricted = true)]
46 [Obsolete("Use XmlReader created by XmlReader.Create() method using"
47 + " appropriate XmlReaderSettings instead.")]
48 public class XmlValidatingReader : XmlReader, IXmlLineInfo, IXmlNamespaceResolver, IHasXmlParserContext
50 public class XmlValidatingReader : XmlReader, IXmlLineInfo, IHasXmlParserContext
56 EntityHandling entityHandling;
57 XmlReader sourceReader;
58 XmlTextReader xmlTextReader;
59 XmlReader validatingReader;
60 XmlResolver resolver; // Only used to non-XmlTextReader XmlReader
61 bool resolverSpecified;
62 ValidationType validationType;
63 // for 2.0: Now it is obsolete. It is allocated only when it is required
65 DTDValidatingReader dtdReader;
66 IHasXmlSchemaInfo schemaInfo;
67 StringBuilder storedCharacters;
73 public XmlValidatingReader (XmlReader reader)
75 sourceReader = reader;
76 xmlTextReader = reader as XmlTextReader;
77 if (xmlTextReader == null)
78 resolver = new XmlUrlResolver ();
79 entityHandling = EntityHandling.ExpandEntities;
80 validationType = ValidationType.Auto;
81 storedCharacters = new StringBuilder ();
84 public XmlValidatingReader (Stream xmlFragment, XmlNodeType fragType, XmlParserContext context)
85 : this (new XmlTextReader (xmlFragment, fragType, context))
89 public XmlValidatingReader (string xmlFragment, XmlNodeType fragType, XmlParserContext context)
90 : this (new XmlTextReader (xmlFragment, fragType, context))
94 #endregion // Constructors
98 public override int AttributeCount {
99 get { return validatingReader == null ? 0 : validatingReader.AttributeCount; }
102 public override string BaseURI {
103 get { return validatingReader == null ? sourceReader.BaseURI : validatingReader.BaseURI; }
107 public override bool CanReadBinaryContent {
112 // This property for this class always return true.
113 public override bool CanResolveEntity {
117 public override int Depth {
118 get { return validatingReader == null ? 0 : validatingReader.Depth; }
121 public Encoding Encoding {
123 if (xmlTextReader != null)
124 return xmlTextReader.Encoding;
126 throw new NotSupportedException ("Encoding is supported only for XmlTextReader.");
130 public EntityHandling EntityHandling {
131 get { return entityHandling; }
133 entityHandling = value;
134 if (dtdReader != null)
135 dtdReader.EntityHandling = value;
139 public override bool EOF {
140 get { return validatingReader == null ? false : validatingReader.EOF; }
143 #if DTD_HANDLE_EVENTS
144 internal bool HasValidationEvent {
145 get { return ValidationEventHandler != null; }
149 public override bool HasValue {
150 get { return validatingReader == null ? false : validatingReader.HasValue; }
153 public override bool IsDefault {
154 get { return validatingReader == null ? false : validatingReader.IsDefault; }
157 public override bool IsEmptyElement {
158 get { return validatingReader == null ? false : validatingReader.IsEmptyElement; }
163 public override string this [int i] {
164 get { return GetAttribute (i); }
167 public override string this [string name] {
168 get { return GetAttribute (name); }
171 public override string this [string localName, string namespaceName] {
172 get { return GetAttribute (localName, namespaceName); }
177 public int LineNumber {
179 int IXmlLineInfo.LineNumber {
184 IXmlLineInfo info = validatingReader as IXmlLineInfo;
185 return info != null ? info.LineNumber : 0;
190 public int LinePosition {
192 int IXmlLineInfo.LinePosition {
197 IXmlLineInfo info = validatingReader as IXmlLineInfo;
198 return info != null ? info.LinePosition : 0;
202 public override string LocalName {
204 if (validatingReader == null)
207 return validatingReader.LocalName;
209 return validatingReader.Name;
213 public override string Name {
214 get { return validatingReader == null ? String.Empty : validatingReader.Name; }
217 public bool Namespaces {
219 if (xmlTextReader != null)
220 return xmlTextReader.Namespaces;
225 if (ReadState != ReadState.Initial)
226 throw new InvalidOperationException ("Namespaces have to be set before reading.");
228 if (xmlTextReader != null)
229 xmlTextReader.Namespaces = value;
231 throw new NotSupportedException ("Property 'Namespaces' is supported only for XmlTextReader.");
235 public override string NamespaceURI {
237 if (validatingReader == null)
240 return validatingReader.NamespaceURI;
246 public override XmlNameTable NameTable {
247 get { return validatingReader == null ? sourceReader.NameTable : validatingReader.NameTable; }
250 public override XmlNodeType NodeType {
251 get { return validatingReader == null ? XmlNodeType.None : validatingReader.NodeType; }
254 public override string Prefix {
255 get { return validatingReader == null ? String.Empty : validatingReader.Prefix; }
258 public override char QuoteChar {
259 get { return validatingReader == null ? sourceReader.QuoteChar : validatingReader.QuoteChar; }
262 public XmlReader Reader {
263 get { return sourceReader; }
266 public override ReadState ReadState {
268 if (validatingReader == null)
269 return ReadState.Initial;
270 return validatingReader.ReadState;
274 internal XmlResolver Resolver {
276 // This is special rule... MS.NET shares the
277 // XmlResolver between XmlTextReader and
278 // XmlValidatingReader, so we mimick that
279 // silly behavior here.
280 if (this.xmlTextReader != null)
281 return this.xmlTextReader.Resolver;
282 else if (resolverSpecified)
289 public XmlSchemaCollection Schemas {
292 schemas = new XmlSchemaSet ();
293 return schemas.SchemaCollection;
297 internal void SetSchemas (XmlSchemaSet schemas)
299 this.schemas = schemas;
302 public object SchemaType {
303 get { return schemaInfo.SchemaType; }
308 public override XmlReaderSettings Settings {
309 get { return validatingReader == null ? sourceReader.Settings : validatingReader.Settings; }
313 [MonoTODO ("We decided not to support XDR schema that spec is obsolete.")]
314 public ValidationType ValidationType {
315 get { return validationType; }
317 if (ReadState != ReadState.Initial)
318 throw new InvalidOperationException ("ValidationType cannot be set after the first call to Read method.");
319 switch (validationType) {
320 case ValidationType.Auto:
321 case ValidationType.DTD:
322 case ValidationType.None:
323 case ValidationType.Schema:
324 validationType = value;
326 case ValidationType.XDR:
327 throw new NotSupportedException ();
332 public override string Value {
333 get { return validatingReader == null ? String.Empty : validatingReader.Value; }
336 public override string XmlLang {
337 get { return validatingReader == null ? String.Empty : validatingReader.XmlLang; }
340 public XmlResolver XmlResolver {
342 resolverSpecified = true;
344 if (xmlTextReader != null)
345 xmlTextReader.XmlResolver = value;
347 XsdValidatingReader xsvr = validatingReader as XsdValidatingReader;
349 xsvr.XmlResolver = value;
350 DTDValidatingReader dvr = validatingReader as DTDValidatingReader;
352 dvr.XmlResolver = value;
356 public override XmlSpace XmlSpace {
357 get { return validatingReader == null ? XmlSpace.None : validatingReader.XmlSpace; }
360 #endregion // Properties
364 public override void Close ()
366 if (validatingReader == null)
367 sourceReader.Close ();
369 validatingReader.Close ();
372 public override string GetAttribute (int i)
374 if (validatingReader == null)
375 throw new IndexOutOfRangeException ("Reader is not started.");
376 return validatingReader [i];
379 public override string GetAttribute (string name)
381 return validatingReader == null ? null : validatingReader [name];
384 public override string GetAttribute (string localName, string namespaceName)
386 return validatingReader == null ? null : validatingReader [localName, namespaceName];
389 XmlParserContext IHasXmlParserContext.ParserContext {
390 get { return dtdReader != null ? dtdReader.ParserContext : null; }
394 IDictionary<string, string> IXmlNamespaceResolver.GetNamespacesInScope (XmlNamespaceScope scope)
396 return ((IHasXmlParserContext) this).ParserContext.NamespaceManager.GetNamespacesInScope (scope);
401 public bool HasLineInfo ()
403 bool IXmlLineInfo.HasLineInfo ()
406 IXmlLineInfo info = validatingReader as IXmlLineInfo;
407 return info != null ? info.HasLineInfo () : false;
410 public override string LookupNamespace (string prefix)
412 if (validatingReader != null)
413 return validatingReader.LookupNamespace (prefix);
415 return sourceReader.LookupNamespace (prefix);
419 string IXmlNamespaceResolver.LookupPrefix (string ns)
421 IXmlNamespaceResolver res = null;
422 if (validatingReader != null)
423 res = sourceReader as IXmlNamespaceResolver;
425 res = validatingReader as IXmlNamespaceResolver;
427 res.LookupNamespace (ns) :
433 public override void MoveToAttribute (int i)
435 if (validatingReader == null)
436 throw new IndexOutOfRangeException ("Reader is not started.");
438 validatingReader.MoveToAttribute (i);
441 public override bool MoveToAttribute (string name)
443 if (validatingReader == null)
445 return validatingReader.MoveToAttribute (name);
448 public override bool MoveToAttribute (string localName, string namespaceName)
450 if (validatingReader == null)
452 return validatingReader.MoveToAttribute (localName, namespaceName);
455 public override bool MoveToElement ()
457 if (validatingReader == null)
459 return validatingReader.MoveToElement ();
462 public override bool MoveToFirstAttribute ()
464 if (validatingReader == null)
466 return validatingReader.MoveToFirstAttribute ();
469 public override bool MoveToNextAttribute ()
471 if (validatingReader == null)
473 return validatingReader.MoveToNextAttribute ();
476 [MonoTODO ("We decided not to support XDR schema that spec is obsolete.")]
477 public override bool Read ()
479 if (ReadState == ReadState.Initial) {
480 switch (ValidationType) {
481 case ValidationType.Auto:
482 case ValidationType.None:
483 goto case ValidationType.Schema; // might be specified by xsi:schemaLocation.
484 case ValidationType.DTD:
485 validatingReader = dtdReader = new DTDValidatingReader (sourceReader, this);
486 dtdReader.XmlResolver = Resolver;
488 case ValidationType.Schema:
489 dtdReader = new DTDValidatingReader (sourceReader, this);
490 XsdValidatingReader xsvr = new XsdValidatingReader (dtdReader);
491 xsvr.ValidationEventHandler +=
492 new ValidationEventHandler (
494 xsvr.ValidationType = ValidationType;
495 xsvr.Schemas = Schemas.SchemaSet;
496 xsvr.XmlResolver = Resolver;
497 validatingReader = xsvr;
498 dtdReader.XmlResolver = Resolver;
500 case ValidationType.XDR:
501 throw new NotSupportedException ();
503 schemaInfo = validatingReader as IHasXmlSchemaInfo;
505 return validatingReader.Read ();
508 public override bool ReadAttributeValue ()
510 if (validatingReader == null)
512 return validatingReader.ReadAttributeValue ();
516 // LAMESPEC: MS.NET 1.0 has critical bug here.
517 // After calling these methods, validation does not work anymore!
518 public override string ReadInnerXml ()
520 if (validatingReader == null)
522 return validatingReader.ReadInnerXml ();
525 public override string ReadOuterXml ()
527 if (validatingReader == null)
529 return validatingReader.ReadOuterXml ();
534 public override string ReadString ()
536 return base.ReadStringInternal ();
539 public override string ReadString ()
541 return base.ReadString ();
545 public object ReadTypedValue ()
547 if (dtdReader == null)
549 XmlSchemaDatatype dt = schemaInfo.SchemaType as XmlSchemaDatatype;
553 case XmlNodeType.Element:
557 storedCharacters.Length = 0;
562 case XmlNodeType.SignificantWhitespace:
563 case XmlNodeType.Text:
564 case XmlNodeType.CDATA:
565 storedCharacters.Append (Value);
567 case XmlNodeType.Comment:
573 } while (loop && !EOF);
574 return dt.ParseValue (storedCharacters.ToString (), NameTable, dtdReader.ParserContext.NamespaceManager);
575 case XmlNodeType.Attribute:
576 return dt.ParseValue (Value, NameTable, dtdReader.ParserContext.NamespaceManager);
581 public override void ResolveEntity ()
583 validatingReader.ResolveEntity ();
586 // It should be "protected" as usual "event model"
587 // methods are, but validation event is not exposed,
588 // so it is no other way to make it "internal".
589 internal void OnValidationEvent (object o, ValidationEventArgs e)
591 if (ValidationEventHandler != null)
592 ValidationEventHandler (o, e);
593 else if (ValidationType != ValidationType.None && e.Severity == XmlSeverityType.Error)
598 [MonoTODO ("Check how expanded entity is handled here.")]
599 public override int ReadContentAsBase64 (byte [] buffer, int offset, int length)
601 if (validatingReader != null)
602 return validatingReader.ReadContentAsBase64 (buffer, offset, length);
604 return sourceReader.ReadContentAsBase64 (buffer, offset, length);
607 [MonoTODO ("Check how expanded entity is handled here.")]
608 public override int ReadContentAsBinHex (byte [] buffer, int offset, int length)
610 if (validatingReader != null)
611 return validatingReader.ReadContentAsBinHex (buffer, offset, length);
613 return sourceReader.ReadContentAsBinHex (buffer, offset, length);
616 [MonoTODO ("Check how expanded entity is handled here.")]
617 public override int ReadElementContentAsBase64 (byte [] buffer, int offset, int length)
619 if (validatingReader != null)
620 return validatingReader.ReadElementContentAsBase64 (buffer, offset, length);
622 return sourceReader.ReadElementContentAsBase64 (buffer, offset, length);
625 [MonoTODO ("Check how expanded entity is handled here.")]
626 public override int ReadElementContentAsBinHex (byte [] buffer, int offset, int length)
628 if (validatingReader != null)
629 return validatingReader.ReadElementContentAsBinHex (buffer, offset, length);
631 return sourceReader.ReadElementContentAsBinHex (buffer, offset, length);
634 #endregion // Methods
636 #region Events and Delegates
638 public event ValidationEventHandler ValidationEventHandler;
640 #endregion // Events and Delegates