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
64 XmlSchemaCollection schemas;
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 XmlSchemaCollection (NameTable);
297 public object SchemaType {
298 get { return schemaInfo.SchemaType; }
303 public override XmlReaderSettings Settings {
304 get { return validatingReader == null ? sourceReader.Settings : validatingReader.Settings; }
308 [MonoTODO ("We decided not to support XDR schema that spec is obsolete.")]
309 public ValidationType ValidationType {
310 get { return validationType; }
312 if (ReadState != ReadState.Initial)
313 throw new InvalidOperationException ("ValidationType cannot be set after the first call to Read method.");
314 switch (validationType) {
315 case ValidationType.Auto:
316 case ValidationType.DTD:
317 case ValidationType.None:
318 case ValidationType.Schema:
319 validationType = value;
321 case ValidationType.XDR:
322 throw new NotSupportedException ();
327 public override string Value {
328 get { return validatingReader == null ? String.Empty : validatingReader.Value; }
331 public override string XmlLang {
332 get { return validatingReader == null ? String.Empty : validatingReader.XmlLang; }
335 public XmlResolver XmlResolver {
337 resolverSpecified = true;
339 if (xmlTextReader != null)
340 xmlTextReader.XmlResolver = value;
342 XsdValidatingReader xsvr = validatingReader as XsdValidatingReader;
344 xsvr.XmlResolver = value;
345 DTDValidatingReader dvr = validatingReader as DTDValidatingReader;
347 dvr.XmlResolver = value;
351 public override XmlSpace XmlSpace {
352 get { return validatingReader == null ? XmlSpace.None : validatingReader.XmlSpace; }
355 #endregion // Properties
359 public override void Close ()
361 if (validatingReader == null)
362 sourceReader.Close ();
364 validatingReader.Close ();
367 public override string GetAttribute (int i)
369 if (validatingReader == null)
370 throw new IndexOutOfRangeException ("Reader is not started.");
371 return validatingReader [i];
374 public override string GetAttribute (string name)
376 return validatingReader == null ? null : validatingReader [name];
379 public override string GetAttribute (string localName, string namespaceName)
381 return validatingReader == null ? null : validatingReader [localName, namespaceName];
384 XmlParserContext IHasXmlParserContext.ParserContext {
385 get { return dtdReader != null ? dtdReader.ParserContext : null; }
389 IDictionary<string, string> IXmlNamespaceResolver.GetNamespacesInScope (XmlNamespaceScope scope)
391 return ((IHasXmlParserContext) this).ParserContext.NamespaceManager.GetNamespacesInScope (scope);
396 public bool HasLineInfo ()
398 bool IXmlLineInfo.HasLineInfo ()
401 IXmlLineInfo info = validatingReader as IXmlLineInfo;
402 return info != null ? info.HasLineInfo () : false;
405 public override string LookupNamespace (string prefix)
407 if (validatingReader != null)
408 return validatingReader.LookupNamespace (prefix);
410 return sourceReader.LookupNamespace (prefix);
414 string IXmlNamespaceResolver.LookupPrefix (string ns)
416 IXmlNamespaceResolver res = null;
417 if (validatingReader != null)
418 res = sourceReader as IXmlNamespaceResolver;
420 res = validatingReader as IXmlNamespaceResolver;
422 res.LookupNamespace (ns) :
428 public override void MoveToAttribute (int i)
430 if (validatingReader == null)
431 throw new IndexOutOfRangeException ("Reader is not started.");
433 validatingReader.MoveToAttribute (i);
436 public override bool MoveToAttribute (string name)
438 if (validatingReader == null)
440 return validatingReader.MoveToAttribute (name);
443 public override bool MoveToAttribute (string localName, string namespaceName)
445 if (validatingReader == null)
447 return validatingReader.MoveToAttribute (localName, namespaceName);
450 public override bool MoveToElement ()
452 if (validatingReader == null)
454 return validatingReader.MoveToElement ();
457 public override bool MoveToFirstAttribute ()
459 if (validatingReader == null)
461 return validatingReader.MoveToFirstAttribute ();
464 public override bool MoveToNextAttribute ()
466 if (validatingReader == null)
468 return validatingReader.MoveToNextAttribute ();
471 [MonoTODO ("We decided not to support XDR schema that spec is obsolete.")]
472 public override bool Read ()
474 if (validatingReader == null) {
475 switch (ValidationType) {
476 case ValidationType.Auto:
477 case ValidationType.None:
478 goto case ValidationType.Schema; // might be specified by xsi:schemaLocation.
479 case ValidationType.DTD:
480 validatingReader = dtdReader = new DTDValidatingReader (sourceReader, this);
481 dtdReader.XmlResolver = Resolver;
483 case ValidationType.Schema:
484 dtdReader = new DTDValidatingReader (sourceReader, this);
485 XsdValidatingReader xsvr = new XsdValidatingReader (dtdReader);
486 xsvr.ValidationEventHandler +=
487 new ValidationEventHandler (
489 xsvr.ValidationType = ValidationType;
490 xsvr.Schemas = Schemas.SchemaSet;
491 xsvr.XmlResolver = Resolver;
492 validatingReader = xsvr;
493 dtdReader.XmlResolver = Resolver;
495 case ValidationType.XDR:
496 throw new NotSupportedException ();
498 schemaInfo = validatingReader as IHasXmlSchemaInfo;
500 return validatingReader.Read ();
503 public override bool ReadAttributeValue ()
505 if (validatingReader == null)
507 return validatingReader.ReadAttributeValue ();
510 public override string ReadString ()
512 return base.ReadString ();
515 public object ReadTypedValue ()
517 if (dtdReader == null)
519 XmlSchemaDatatype dt = schemaInfo.SchemaType as XmlSchemaDatatype;
521 XmlSchemaType st = schemaInfo.SchemaType as XmlSchemaType;
528 case XmlNodeType.Element:
532 storedCharacters.Length = 0;
537 case XmlNodeType.Whitespace:
538 case XmlNodeType.SignificantWhitespace:
539 case XmlNodeType.Text:
540 case XmlNodeType.CDATA:
541 storedCharacters.Append (Value);
543 case XmlNodeType.Comment:
549 } while (loop && !EOF);
550 return dt.ParseValue (storedCharacters.ToString (), NameTable, dtdReader.ParserContext.NamespaceManager);
551 case XmlNodeType.Attribute:
552 return dt.ParseValue (Value, NameTable, dtdReader.ParserContext.NamespaceManager);
557 public override void ResolveEntity ()
559 validatingReader.ResolveEntity ();
562 // It should be "protected" as usual "event model"
563 // methods are, but validation event is not exposed,
564 // so it is no other way to make it "internal".
565 internal void OnValidationEvent (object o, ValidationEventArgs e)
567 if (ValidationEventHandler != null)
568 ValidationEventHandler (o, e);
569 else if (ValidationType != ValidationType.None && e.Severity == XmlSeverityType.Error)
574 [MonoTODO ("Check how expanded entity is handled here.")]
575 public override int ReadContentAsBase64 (byte [] buffer, int offset, int length)
577 if (validatingReader != null)
578 return validatingReader.ReadContentAsBase64 (buffer, offset, length);
580 return sourceReader.ReadContentAsBase64 (buffer, offset, length);
583 [MonoTODO ("Check how expanded entity is handled here.")]
584 public override int ReadContentAsBinHex (byte [] buffer, int offset, int length)
586 if (validatingReader != null)
587 return validatingReader.ReadContentAsBinHex (buffer, offset, length);
589 return sourceReader.ReadContentAsBinHex (buffer, offset, length);
592 [MonoTODO ("Check how expanded entity is handled here.")]
593 public override int ReadElementContentAsBase64 (byte [] buffer, int offset, int length)
595 if (validatingReader != null)
596 return validatingReader.ReadElementContentAsBase64 (buffer, offset, length);
598 return sourceReader.ReadElementContentAsBase64 (buffer, offset, length);
601 [MonoTODO ("Check how expanded entity is handled here.")]
602 public override int ReadElementContentAsBinHex (byte [] buffer, int offset, int length)
604 if (validatingReader != null)
605 return validatingReader.ReadElementContentAsBinHex (buffer, offset, length);
607 return sourceReader.ReadElementContentAsBinHex (buffer, offset, length);
610 #endregion // Methods
612 #region Events and Delegates
614 public event ValidationEventHandler ValidationEventHandler;
616 #endregion // Events and Delegates