| 
									
										
										
										
											2000-02-04 15:39:30 +00:00
										 |  |  | """A parser for XML, using the derived class as static DTD.""" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-11-18 15:27:20 +00:00
										 |  |  | # Author: Sjoerd Mullender. | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | import re | 
					
						
							|  |  |  | import string | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-04-10 02:04:00 +00:00
										 |  |  | import warnings | 
					
						
							| 
									
										
										
										
											2004-05-02 20:37:13 +00:00
										 |  |  | warnings.warn("The xmllib module is obsolete.  Use xml.sax instead.", DeprecationWarning) | 
					
						
							| 
									
										
										
										
											2002-04-10 02:04:00 +00:00
										 |  |  | del warnings | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-08-26 15:52:33 +00:00
										 |  |  | version = '0.3' | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-12-06 10:37:57 +00:00
										 |  |  | class Error(RuntimeError): | 
					
						
							|  |  |  |     pass | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | # Regular expressions used for parsing | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-10-19 13:28:26 +00:00
										 |  |  | _S = '[ \t\r\n]+'                       # white space | 
					
						
							|  |  |  | _opS = '[ \t\r\n]*'                     # optional white space | 
					
						
							|  |  |  | _Name = '[a-zA-Z_:][-a-zA-Z0-9._:]*'    # valid XML name | 
					
						
							|  |  |  | _QStr = "(?:'[^']*'|\"[^\"]*\")"        # quoted XML string | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  | illegal = re.compile('[^\t\r\n -\176\240-\377]') # illegal chars in content | 
					
						
							|  |  |  | interesting = re.compile('[]&<]') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | amp = re.compile('&') | 
					
						
							|  |  |  | ref = re.compile('&(' + _Name + '|#[0-9]+|#x[0-9a-fA-F]+)[^-a-zA-Z0-9._:]') | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | entityref = re.compile('&(?P<name>' + _Name + ')[^-a-zA-Z0-9._:]') | 
					
						
							|  |  |  | charref = re.compile('&#(?P<char>[0-9]+[^0-9]|x[0-9a-fA-F]+[^0-9a-fA-F])') | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  | space = re.compile(_S + '$') | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | newline = re.compile('\n') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-10-19 13:28:26 +00:00
										 |  |  | attrfind = re.compile( | 
					
						
							|  |  |  |     _S + '(?P<name>' + _Name + ')' | 
					
						
							|  |  |  |     '(' + _opS + '=' + _opS + | 
					
						
							| 
									
										
										
										
											2000-06-21 20:01:10 +00:00
										 |  |  |     '(?P<value>'+_QStr+'|[-a-zA-Z0-9.:+*%?!\(\)_#=~]+))?') | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | starttagopen = re.compile('<' + _Name) | 
					
						
							|  |  |  | starttagend = re.compile(_opS + '(?P<slash>/?)>') | 
					
						
							| 
									
										
										
										
											1998-10-19 13:28:26 +00:00
										 |  |  | starttagmatch = re.compile('<(?P<tagname>'+_Name+')' | 
					
						
							|  |  |  |                       '(?P<attrs>(?:'+attrfind.pattern+')*)'+ | 
					
						
							|  |  |  |                       starttagend.pattern) | 
					
						
							|  |  |  | endtagopen = re.compile('</') | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  | endbracket = re.compile(_opS + '>') | 
					
						
							| 
									
										
										
										
											1998-10-19 13:28:26 +00:00
										 |  |  | endbracketfind = re.compile('(?:[^>\'"]|'+_QStr+')*>') | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | tagfind = re.compile(_Name) | 
					
						
							| 
									
										
										
										
											1998-01-29 14:55:24 +00:00
										 |  |  | cdataopen = re.compile(r'<!\[CDATA\[') | 
					
						
							|  |  |  | cdataclose = re.compile(r'\]\]>') | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  | # this matches one of the following: | 
					
						
							|  |  |  | # SYSTEM SystemLiteral | 
					
						
							|  |  |  | # PUBLIC PubidLiteral SystemLiteral | 
					
						
							| 
									
										
										
										
											1998-10-19 13:28:26 +00:00
										 |  |  | _SystemLiteral = '(?P<%s>'+_QStr+')' | 
					
						
							| 
									
										
										
										
											2000-06-21 20:01:10 +00:00
										 |  |  | _PublicLiteral = '(?P<%s>"[-\'\(\)+,./:=?;!*#@$_%% \n\ra-zA-Z0-9]*"|' \ | 
					
						
							|  |  |  |                         "'[-\(\)+,./:=?;!*#@$_%% \n\ra-zA-Z0-9]*')" | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  | _ExternalId = '(?:SYSTEM|' \ | 
					
						
							|  |  |  |                  'PUBLIC'+_S+_PublicLiteral%'pubid'+ \ | 
					
						
							|  |  |  |               ')'+_S+_SystemLiteral%'syslit' | 
					
						
							|  |  |  | doctype = re.compile('<!DOCTYPE'+_S+'(?P<name>'+_Name+')' | 
					
						
							|  |  |  |                      '(?:'+_S+_ExternalId+')?'+_opS) | 
					
						
							|  |  |  | xmldecl = re.compile('<\?xml'+_S+ | 
					
						
							| 
									
										
										
										
											1998-10-19 13:28:26 +00:00
										 |  |  |                      'version'+_opS+'='+_opS+'(?P<version>'+_QStr+')'+ | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |                      '(?:'+_S+'encoding'+_opS+'='+_opS+ | 
					
						
							|  |  |  |                         "(?P<encoding>'[A-Za-z][-A-Za-z0-9._]*'|" | 
					
						
							|  |  |  |                         '"[A-Za-z][-A-Za-z0-9._]*"))?' | 
					
						
							|  |  |  |                      '(?:'+_S+'standalone'+_opS+'='+_opS+ | 
					
						
							|  |  |  |                         '(?P<standalone>\'(?:yes|no)\'|"(?:yes|no)"))?'+ | 
					
						
							|  |  |  |                      _opS+'\?>') | 
					
						
							|  |  |  | procopen = re.compile(r'<\?(?P<proc>' + _Name + ')' + _opS) | 
					
						
							| 
									
										
										
										
											1998-01-29 14:55:24 +00:00
										 |  |  | procclose = re.compile(_opS + r'\?>') | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | commentopen = re.compile('<!--') | 
					
						
							|  |  |  | commentclose = re.compile('-->') | 
					
						
							|  |  |  | doubledash = re.compile('--') | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  | attrtrans = string.maketrans(' \r\n\t', '    ') | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-12-18 20:17:13 +00:00
										 |  |  | # definitions for XML namespaces | 
					
						
							|  |  |  | _NCName = '[a-zA-Z_][-a-zA-Z0-9._]*'    # XML Name, minus the ":" | 
					
						
							|  |  |  | ncname = re.compile(_NCName + '$') | 
					
						
							|  |  |  | qname = re.compile('(?:(?P<prefix>' + _NCName + '):)?' # optional prefix | 
					
						
							|  |  |  |                    '(?P<local>' + _NCName + ')$') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | xmlns = re.compile('xmlns(?::(?P<ncname>'+_NCName+'))?$') | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | # XML parser base class -- find tags and call handler functions. | 
					
						
							|  |  |  | # Usage: p = XMLParser(); p.feed(data); ...; p.close(). | 
					
						
							| 
									
										
										
										
											1997-11-18 15:27:20 +00:00
										 |  |  | # The dtd is defined by deriving a class which defines methods with | 
					
						
							|  |  |  | # special names to handle tags: start_foo and end_foo to handle <foo> | 
					
						
							|  |  |  | # and </foo>, respectively.  The data between tags is passed to the | 
					
						
							|  |  |  | # parser by calling self.handle_data() with some data as argument (the | 
					
						
							| 
									
										
										
										
											2000-06-28 14:48:01 +00:00
										 |  |  | # data may be split up in arbitrary chunks). | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | class XMLParser: | 
					
						
							| 
									
										
										
										
											1998-12-18 20:17:13 +00:00
										 |  |  |     attributes = {}                     # default, to be overridden | 
					
						
							|  |  |  |     elements = {}                       # default, to be overridden | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-08-26 15:52:33 +00:00
										 |  |  |     # parsing options, settable using keyword args in __init__ | 
					
						
							|  |  |  |     __accept_unquoted_attributes = 0 | 
					
						
							|  |  |  |     __accept_missing_endtag_name = 0 | 
					
						
							|  |  |  |     __map_case = 0 | 
					
						
							|  |  |  |     __accept_utf8 = 0 | 
					
						
							| 
									
										
										
										
											2000-07-04 14:53:12 +00:00
										 |  |  |     __translate_attribute_references = 1 | 
					
						
							| 
									
										
										
										
											1999-08-26 15:52:33 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  |     # Interface -- initialize and reset this instance | 
					
						
							| 
									
										
										
										
											1999-08-26 15:52:33 +00:00
										 |  |  |     def __init__(self, **kw): | 
					
						
							| 
									
										
										
										
											1999-02-02 17:54:38 +00:00
										 |  |  |         self.__fixed = 0 | 
					
						
							| 
									
										
										
										
											2002-06-01 14:18:47 +00:00
										 |  |  |         if 'accept_unquoted_attributes' in kw: | 
					
						
							| 
									
										
										
										
											1999-08-26 15:52:33 +00:00
										 |  |  |             self.__accept_unquoted_attributes = kw['accept_unquoted_attributes'] | 
					
						
							| 
									
										
										
										
											2002-06-01 14:18:47 +00:00
										 |  |  |         if 'accept_missing_endtag_name' in kw: | 
					
						
							| 
									
										
										
										
											1999-08-26 15:52:33 +00:00
										 |  |  |             self.__accept_missing_endtag_name = kw['accept_missing_endtag_name'] | 
					
						
							| 
									
										
										
										
											2002-06-01 14:18:47 +00:00
										 |  |  |         if 'map_case' in kw: | 
					
						
							| 
									
										
										
										
											1999-08-26 15:52:33 +00:00
										 |  |  |             self.__map_case = kw['map_case'] | 
					
						
							| 
									
										
										
										
											2002-06-01 14:18:47 +00:00
										 |  |  |         if 'accept_utf8' in kw: | 
					
						
							| 
									
										
										
										
											1999-08-26 15:52:33 +00:00
										 |  |  |             self.__accept_utf8 = kw['accept_utf8'] | 
					
						
							| 
									
										
										
										
											2002-06-01 14:18:47 +00:00
										 |  |  |         if 'translate_attribute_references' in kw: | 
					
						
							| 
									
										
										
										
											2000-07-04 14:53:12 +00:00
										 |  |  |             self.__translate_attribute_references = kw['translate_attribute_references'] | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         self.reset() | 
					
						
							| 
									
										
										
										
											1999-02-01 15:35:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __fixelements(self): | 
					
						
							| 
									
										
										
										
											1999-02-02 17:54:38 +00:00
										 |  |  |         self.__fixed = 1 | 
					
						
							| 
									
										
										
										
											1999-02-01 15:35:13 +00:00
										 |  |  |         self.elements = {} | 
					
						
							|  |  |  |         self.__fixdict(self.__dict__) | 
					
						
							|  |  |  |         self.__fixclass(self.__class__) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __fixclass(self, kl): | 
					
						
							|  |  |  |         self.__fixdict(kl.__dict__) | 
					
						
							|  |  |  |         for k in kl.__bases__: | 
					
						
							|  |  |  |             self.__fixclass(k) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __fixdict(self, dict): | 
					
						
							| 
									
										
										
										
											1999-02-02 22:15:24 +00:00
										 |  |  |         for key in dict.keys(): | 
					
						
							| 
									
										
										
										
											1999-02-01 15:35:13 +00:00
										 |  |  |             if key[:6] == 'start_': | 
					
						
							| 
									
										
										
										
											1999-02-02 22:15:24 +00:00
										 |  |  |                 tag = key[6:] | 
					
						
							|  |  |  |                 start, end = self.elements.get(tag, (None, None)) | 
					
						
							| 
									
										
										
										
											1999-02-01 15:35:13 +00:00
										 |  |  |                 if start is None: | 
					
						
							| 
									
										
										
										
											1999-02-02 22:15:24 +00:00
										 |  |  |                     self.elements[tag] = getattr(self, key), end | 
					
						
							| 
									
										
										
										
											1999-02-01 15:35:13 +00:00
										 |  |  |             elif key[:4] == 'end_': | 
					
						
							| 
									
										
										
										
											1999-02-02 22:15:24 +00:00
										 |  |  |                 tag = key[4:] | 
					
						
							|  |  |  |                 start, end = self.elements.get(tag, (None, None)) | 
					
						
							| 
									
										
										
										
											1999-02-01 15:35:13 +00:00
										 |  |  |                 if end is None: | 
					
						
							| 
									
										
										
										
											1999-02-02 22:15:24 +00:00
										 |  |  |                     self.elements[tag] = start, getattr(self, key) | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Interface -- reset this instance.  Loses all unprocessed data | 
					
						
							|  |  |  |     def reset(self): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         self.rawdata = '' | 
					
						
							|  |  |  |         self.stack = [] | 
					
						
							|  |  |  |         self.nomoretags = 0 | 
					
						
							|  |  |  |         self.literal = 0 | 
					
						
							|  |  |  |         self.lineno = 1 | 
					
						
							|  |  |  |         self.__at_start = 1 | 
					
						
							|  |  |  |         self.__seen_doctype = None | 
					
						
							|  |  |  |         self.__seen_starttag = 0 | 
					
						
							| 
									
										
										
										
											1998-12-21 18:15:28 +00:00
										 |  |  |         self.__use_namespaces = 0 | 
					
						
							| 
									
										
										
										
											1998-12-18 20:17:13 +00:00
										 |  |  |         self.__namespaces = {'xml':None}   # xml is implicitly declared | 
					
						
							| 
									
										
										
										
											2000-07-16 12:04:32 +00:00
										 |  |  |         # backward compatibility hack: if elements not overridden, | 
					
						
							| 
									
										
										
										
											1999-02-02 17:54:38 +00:00
										 |  |  |         # fill it in ourselves | 
					
						
							|  |  |  |         if self.elements is XMLParser.elements: | 
					
						
							|  |  |  |             self.__fixelements() | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # For derived classes only -- enter literal mode (CDATA) till EOF | 
					
						
							|  |  |  |     def setnomoretags(self): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         self.nomoretags = self.literal = 1 | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # For derived classes only -- enter literal mode (CDATA) | 
					
						
							|  |  |  |     def setliteral(self, *args): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         self.literal = 1 | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Interface -- feed some data to the parser.  Call this as | 
					
						
							|  |  |  |     # often as you want, with as little or as much text as you | 
					
						
							|  |  |  |     # want (may include '\n').  (This just saves the text, all the | 
					
						
							|  |  |  |     # processing is done by goahead().) | 
					
						
							|  |  |  |     def feed(self, data): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         self.rawdata = self.rawdata + data | 
					
						
							|  |  |  |         self.goahead(0) | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Interface -- handle the remaining data | 
					
						
							|  |  |  |     def close(self): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         self.goahead(1) | 
					
						
							| 
									
										
										
										
											1999-02-02 17:54:38 +00:00
										 |  |  |         if self.__fixed: | 
					
						
							|  |  |  |             self.__fixed = 0 | 
					
						
							|  |  |  |             # remove self.elements so that we don't leak | 
					
						
							|  |  |  |             del self.elements | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Interface -- translate references | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |     def translate_references(self, data, all = 1): | 
					
						
							| 
									
										
										
										
											2000-07-04 14:53:12 +00:00
										 |  |  |         if not self.__translate_attribute_references: | 
					
						
							|  |  |  |             return data | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         i = 0 | 
					
						
							|  |  |  |         while 1: | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |             res = amp.search(data, i) | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |             if res is None: | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |                 return data | 
					
						
							| 
									
										
										
										
											2000-07-10 08:09:48 +00:00
										 |  |  |             s = res.start(0) | 
					
						
							|  |  |  |             res = ref.match(data, s) | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |             if res is None: | 
					
						
							|  |  |  |                 self.syntax_error("bogus `&'") | 
					
						
							| 
									
										
										
										
											2000-07-10 08:09:48 +00:00
										 |  |  |                 i = s+1 | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |                 continue | 
					
						
							|  |  |  |             i = res.end(0) | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |             str = res.group(1) | 
					
						
							| 
									
										
										
										
											2000-07-10 08:09:48 +00:00
										 |  |  |             rescan = 0 | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |             if str[0] == '#': | 
					
						
							|  |  |  |                 if str[1] == 'x': | 
					
						
							| 
									
										
										
										
											2001-02-09 10:48:30 +00:00
										 |  |  |                     str = chr(int(str[2:], 16)) | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |                 else: | 
					
						
							| 
									
										
										
										
											2001-02-09 10:48:30 +00:00
										 |  |  |                     str = chr(int(str[1:])) | 
					
						
							| 
									
										
										
										
											2000-07-10 08:09:48 +00:00
										 |  |  |                 if data[i - 1] != ';': | 
					
						
							|  |  |  |                     self.syntax_error("`;' missing after char reference") | 
					
						
							|  |  |  |                     i = i-1 | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |             elif all: | 
					
						
							| 
									
										
										
										
											2002-06-01 14:18:47 +00:00
										 |  |  |                 if str in self.entitydefs: | 
					
						
							| 
									
										
										
										
											2000-07-10 08:09:48 +00:00
										 |  |  |                     str = self.entitydefs[str] | 
					
						
							|  |  |  |                     rescan = 1 | 
					
						
							|  |  |  |                 elif data[i - 1] != ';': | 
					
						
							|  |  |  |                     self.syntax_error("bogus `&'") | 
					
						
							|  |  |  |                     i = s + 1 # just past the & | 
					
						
							|  |  |  |                     continue | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |                 else: | 
					
						
							| 
									
										
										
										
											1999-02-02 17:54:38 +00:00
										 |  |  |                     self.syntax_error("reference to unknown entity `&%s;'" % str) | 
					
						
							| 
									
										
										
										
											2000-07-10 08:09:48 +00:00
										 |  |  |                     str = '&' + str + ';' | 
					
						
							|  |  |  |             elif data[i - 1] != ';': | 
					
						
							|  |  |  |                 self.syntax_error("bogus `&'") | 
					
						
							|  |  |  |                 i = s + 1 # just past the & | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # when we get here, str contains the translated text and i points | 
					
						
							|  |  |  |             # to the end of the string that is to be replaced | 
					
						
							|  |  |  |             data = data[:s] + str + data[i:] | 
					
						
							|  |  |  |             if rescan: | 
					
						
							|  |  |  |                 i = s | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |             else: | 
					
						
							| 
									
										
										
										
											2000-07-10 08:09:48 +00:00
										 |  |  |                 i = s + len(str) | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-31 10:27:00 +00:00
										 |  |  |     # Interface - return a dictionary of all namespaces currently valid | 
					
						
							|  |  |  |     def getnamespace(self): | 
					
						
							|  |  |  |         nsdict = {} | 
					
						
							|  |  |  |         for t, d, nst in self.stack: | 
					
						
							|  |  |  |             nsdict.update(d) | 
					
						
							|  |  |  |         return nsdict | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  |     # Internal -- handle data as far as reasonable.  May leave state | 
					
						
							|  |  |  |     # and data to be processed by a subsequent call.  If 'end' is | 
					
						
							|  |  |  |     # true, force handling all data as if followed by EOF marker. | 
					
						
							|  |  |  |     def goahead(self, end): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         rawdata = self.rawdata | 
					
						
							|  |  |  |         i = 0 | 
					
						
							|  |  |  |         n = len(rawdata) | 
					
						
							|  |  |  |         while i < n: | 
					
						
							|  |  |  |             if i > 0: | 
					
						
							|  |  |  |                 self.__at_start = 0 | 
					
						
							|  |  |  |             if self.nomoretags: | 
					
						
							|  |  |  |                 data = rawdata[i:n] | 
					
						
							|  |  |  |                 self.handle_data(data) | 
					
						
							| 
									
										
										
										
											2001-02-09 10:48:30 +00:00
										 |  |  |                 self.lineno = self.lineno + data.count('\n') | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |                 i = n | 
					
						
							|  |  |  |                 break | 
					
						
							|  |  |  |             res = interesting.search(rawdata, i) | 
					
						
							|  |  |  |             if res: | 
					
						
							| 
									
										
										
										
											2001-01-15 03:34:38 +00:00
										 |  |  |                 j = res.start(0) | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |             else: | 
					
						
							| 
									
										
										
										
											2001-01-15 03:34:38 +00:00
										 |  |  |                 j = n | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |             if i < j: | 
					
						
							| 
									
										
										
										
											1998-12-21 18:15:28 +00:00
										 |  |  |                 data = rawdata[i:j] | 
					
						
							|  |  |  |                 if self.__at_start and space.match(data) is None: | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |                     self.syntax_error('illegal data at start of file') | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |                 self.__at_start = 0 | 
					
						
							| 
									
										
										
										
											1998-10-19 13:28:26 +00:00
										 |  |  |                 if not self.stack and space.match(data) is None: | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |                     self.syntax_error('data not in content') | 
					
						
							| 
									
										
										
										
											1999-08-26 15:52:33 +00:00
										 |  |  |                 if not self.__accept_utf8 and illegal.search(data): | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |                     self.syntax_error('illegal character in content') | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |                 self.handle_data(data) | 
					
						
							| 
									
										
										
										
											2001-02-09 10:48:30 +00:00
										 |  |  |                 self.lineno = self.lineno + data.count('\n') | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |             i = j | 
					
						
							|  |  |  |             if i == n: break | 
					
						
							|  |  |  |             if rawdata[i] == '<': | 
					
						
							|  |  |  |                 if starttagopen.match(rawdata, i): | 
					
						
							|  |  |  |                     if self.literal: | 
					
						
							|  |  |  |                         data = rawdata[i] | 
					
						
							|  |  |  |                         self.handle_data(data) | 
					
						
							| 
									
										
										
										
											2001-02-09 10:48:30 +00:00
										 |  |  |                         self.lineno = self.lineno + data.count('\n') | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |                         i = i+1 | 
					
						
							|  |  |  |                         continue | 
					
						
							|  |  |  |                     k = self.parse_starttag(i) | 
					
						
							|  |  |  |                     if k < 0: break | 
					
						
							|  |  |  |                     self.__seen_starttag = 1 | 
					
						
							| 
									
										
										
										
											2001-02-09 10:48:30 +00:00
										 |  |  |                     self.lineno = self.lineno + rawdata[i:k].count('\n') | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |                     i = k | 
					
						
							|  |  |  |                     continue | 
					
						
							|  |  |  |                 if endtagopen.match(rawdata, i): | 
					
						
							|  |  |  |                     k = self.parse_endtag(i) | 
					
						
							|  |  |  |                     if k < 0: break | 
					
						
							| 
									
										
										
										
											2001-02-09 10:48:30 +00:00
										 |  |  |                     self.lineno = self.lineno + rawdata[i:k].count('\n') | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |                     i =  k | 
					
						
							|  |  |  |                     continue | 
					
						
							|  |  |  |                 if commentopen.match(rawdata, i): | 
					
						
							|  |  |  |                     if self.literal: | 
					
						
							|  |  |  |                         data = rawdata[i] | 
					
						
							|  |  |  |                         self.handle_data(data) | 
					
						
							| 
									
										
										
										
											2001-02-09 10:48:30 +00:00
										 |  |  |                         self.lineno = self.lineno + data.count('\n') | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |                         i = i+1 | 
					
						
							|  |  |  |                         continue | 
					
						
							|  |  |  |                     k = self.parse_comment(i) | 
					
						
							|  |  |  |                     if k < 0: break | 
					
						
							| 
									
										
										
										
											2001-02-09 10:48:30 +00:00
										 |  |  |                     self.lineno = self.lineno + rawdata[i:k].count('\n') | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |                     i = k | 
					
						
							|  |  |  |                     continue | 
					
						
							|  |  |  |                 if cdataopen.match(rawdata, i): | 
					
						
							|  |  |  |                     k = self.parse_cdata(i) | 
					
						
							|  |  |  |                     if k < 0: break | 
					
						
							| 
									
										
										
										
											2001-02-09 10:48:30 +00:00
										 |  |  |                     self.lineno = self.lineno + rawdata[i:k].count('\n') | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |                     i = k | 
					
						
							|  |  |  |                     continue | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |                 res = xmldecl.match(rawdata, i) | 
					
						
							|  |  |  |                 if res: | 
					
						
							|  |  |  |                     if not self.__at_start: | 
					
						
							|  |  |  |                         self.syntax_error("<?xml?> declaration not at start of document") | 
					
						
							|  |  |  |                     version, encoding, standalone = res.group('version', | 
					
						
							|  |  |  |                                                               'encoding', | 
					
						
							|  |  |  |                                                               'standalone') | 
					
						
							|  |  |  |                     if version[1:-1] != '1.0': | 
					
						
							| 
									
										
										
										
											2000-12-06 10:37:57 +00:00
										 |  |  |                         raise Error('only XML version 1.0 supported') | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |                     if encoding: encoding = encoding[1:-1] | 
					
						
							|  |  |  |                     if standalone: standalone = standalone[1:-1] | 
					
						
							|  |  |  |                     self.handle_xml(encoding, standalone) | 
					
						
							|  |  |  |                     i = res.end(0) | 
					
						
							|  |  |  |                     continue | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |                 res = procopen.match(rawdata, i) | 
					
						
							|  |  |  |                 if res: | 
					
						
							|  |  |  |                     k = self.parse_proc(i) | 
					
						
							|  |  |  |                     if k < 0: break | 
					
						
							| 
									
										
										
										
											2001-02-09 10:48:30 +00:00
										 |  |  |                     self.lineno = self.lineno + rawdata[i:k].count('\n') | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |                     i = k | 
					
						
							|  |  |  |                     continue | 
					
						
							|  |  |  |                 res = doctype.match(rawdata, i) | 
					
						
							|  |  |  |                 if res: | 
					
						
							|  |  |  |                     if self.literal: | 
					
						
							|  |  |  |                         data = rawdata[i] | 
					
						
							|  |  |  |                         self.handle_data(data) | 
					
						
							| 
									
										
										
										
											2001-02-09 10:48:30 +00:00
										 |  |  |                         self.lineno = self.lineno + data.count('\n') | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |                         i = i+1 | 
					
						
							|  |  |  |                         continue | 
					
						
							|  |  |  |                     if self.__seen_doctype: | 
					
						
							|  |  |  |                         self.syntax_error('multiple DOCTYPE elements') | 
					
						
							|  |  |  |                     if self.__seen_starttag: | 
					
						
							|  |  |  |                         self.syntax_error('DOCTYPE not at beginning of document') | 
					
						
							|  |  |  |                     k = self.parse_doctype(res) | 
					
						
							|  |  |  |                     if k < 0: break | 
					
						
							|  |  |  |                     self.__seen_doctype = res.group('name') | 
					
						
							| 
									
										
										
										
											1999-08-26 15:52:33 +00:00
										 |  |  |                     if self.__map_case: | 
					
						
							| 
									
										
										
										
											2001-02-09 10:48:30 +00:00
										 |  |  |                         self.__seen_doctype = self.__seen_doctype.lower() | 
					
						
							|  |  |  |                     self.lineno = self.lineno + rawdata[i:k].count('\n') | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |                     i = k | 
					
						
							|  |  |  |                     continue | 
					
						
							|  |  |  |             elif rawdata[i] == '&': | 
					
						
							| 
									
										
										
										
											1998-12-08 13:25:00 +00:00
										 |  |  |                 if self.literal: | 
					
						
							|  |  |  |                     data = rawdata[i] | 
					
						
							|  |  |  |                     self.handle_data(data) | 
					
						
							|  |  |  |                     i = i+1 | 
					
						
							|  |  |  |                     continue | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |                 res = charref.match(rawdata, i) | 
					
						
							|  |  |  |                 if res is not None: | 
					
						
							|  |  |  |                     i = res.end(0) | 
					
						
							|  |  |  |                     if rawdata[i-1] != ';': | 
					
						
							|  |  |  |                         self.syntax_error("`;' missing in charref") | 
					
						
							|  |  |  |                         i = i-1 | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |                     if not self.stack: | 
					
						
							|  |  |  |                         self.syntax_error('data not in content') | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |                     self.handle_charref(res.group('char')[:-1]) | 
					
						
							| 
									
										
										
										
											2001-02-09 10:48:30 +00:00
										 |  |  |                     self.lineno = self.lineno + res.group(0).count('\n') | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |                     continue | 
					
						
							|  |  |  |                 res = entityref.match(rawdata, i) | 
					
						
							|  |  |  |                 if res is not None: | 
					
						
							|  |  |  |                     i = res.end(0) | 
					
						
							|  |  |  |                     if rawdata[i-1] != ';': | 
					
						
							|  |  |  |                         self.syntax_error("`;' missing in entityref") | 
					
						
							|  |  |  |                         i = i-1 | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |                     name = res.group('name') | 
					
						
							| 
									
										
										
										
											1999-08-26 15:52:33 +00:00
										 |  |  |                     if self.__map_case: | 
					
						
							| 
									
										
										
										
											2001-02-09 10:48:30 +00:00
										 |  |  |                         name = name.lower() | 
					
						
							| 
									
										
										
										
											2002-06-01 14:18:47 +00:00
										 |  |  |                     if name in self.entitydefs: | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |                         self.rawdata = rawdata = rawdata[:res.start(0)] + self.entitydefs[name] + rawdata[i:] | 
					
						
							|  |  |  |                         n = len(rawdata) | 
					
						
							|  |  |  |                         i = res.start(0) | 
					
						
							|  |  |  |                     else: | 
					
						
							|  |  |  |                         self.unknown_entityref(name) | 
					
						
							| 
									
										
										
										
											2001-02-09 10:48:30 +00:00
										 |  |  |                     self.lineno = self.lineno + res.group(0).count('\n') | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |                     continue | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |             elif rawdata[i] == ']': | 
					
						
							| 
									
										
										
										
											1998-12-08 13:25:00 +00:00
										 |  |  |                 if self.literal: | 
					
						
							|  |  |  |                     data = rawdata[i] | 
					
						
							|  |  |  |                     self.handle_data(data) | 
					
						
							|  |  |  |                     i = i+1 | 
					
						
							|  |  |  |                     continue | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |                 if n-i < 3: | 
					
						
							|  |  |  |                     break | 
					
						
							|  |  |  |                 if cdataclose.match(rawdata, i): | 
					
						
							|  |  |  |                     self.syntax_error("bogus `]]>'") | 
					
						
							|  |  |  |                 self.handle_data(rawdata[i]) | 
					
						
							|  |  |  |                 i = i+1 | 
					
						
							|  |  |  |                 continue | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |             else: | 
					
						
							| 
									
										
										
										
											2000-12-06 10:37:57 +00:00
										 |  |  |                 raise Error('neither < nor & ??') | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |             # We get here only if incomplete matches but | 
					
						
							|  |  |  |             # nothing else | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |             break | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         # end while | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |         if i > 0: | 
					
						
							|  |  |  |             self.__at_start = 0 | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         if end and i < n: | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |             data = rawdata[i] | 
					
						
							|  |  |  |             self.syntax_error("bogus `%s'" % data) | 
					
						
							| 
									
										
										
										
											1999-08-26 15:52:33 +00:00
										 |  |  |             if not self.__accept_utf8 and illegal.search(data): | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |                 self.syntax_error('illegal character in content') | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |             self.handle_data(data) | 
					
						
							| 
									
										
										
										
											2001-02-09 10:48:30 +00:00
										 |  |  |             self.lineno = self.lineno + data.count('\n') | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |             self.rawdata = rawdata[i+1:] | 
					
						
							|  |  |  |             return self.goahead(end) | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         self.rawdata = rawdata[i:] | 
					
						
							|  |  |  |         if end: | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |             if not self.__seen_starttag: | 
					
						
							|  |  |  |                 self.syntax_error('no elements in file') | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |             if self.stack: | 
					
						
							|  |  |  |                 self.syntax_error('missing end tags') | 
					
						
							|  |  |  |                 while self.stack: | 
					
						
							| 
									
										
										
										
											1998-12-18 20:17:13 +00:00
										 |  |  |                     self.finish_endtag(self.stack[-1][0]) | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Internal -- parse comment, return length or -1 if not terminated | 
					
						
							|  |  |  |     def parse_comment(self, i): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         rawdata = self.rawdata | 
					
						
							| 
									
										
										
										
											2000-12-12 23:20:45 +00:00
										 |  |  |         if rawdata[i:i+4] != '<!--': | 
					
						
							| 
									
										
										
										
											2000-12-06 10:37:57 +00:00
										 |  |  |             raise Error('unexpected call to handle_comment') | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         res = commentclose.search(rawdata, i+4) | 
					
						
							| 
									
										
										
										
											1998-10-19 13:28:26 +00:00
										 |  |  |         if res is None: | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |             return -1 | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |         if doubledash.search(rawdata, i+4, res.start(0)): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |             self.syntax_error("`--' inside comment") | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |         if rawdata[res.start(0)-1] == '-': | 
					
						
							|  |  |  |             self.syntax_error('comment cannot end in three dashes') | 
					
						
							| 
									
										
										
										
											1999-08-26 15:52:33 +00:00
										 |  |  |         if not self.__accept_utf8 and \ | 
					
						
							|  |  |  |            illegal.search(rawdata, i+4, res.start(0)): | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |             self.syntax_error('illegal character in comment') | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         self.handle_comment(rawdata[i+4: res.start(0)]) | 
					
						
							|  |  |  |         return res.end(0) | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-01-29 14:55:24 +00:00
										 |  |  |     # Internal -- handle DOCTYPE tag, return length or -1 if not terminated | 
					
						
							|  |  |  |     def parse_doctype(self, res): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         rawdata = self.rawdata | 
					
						
							|  |  |  |         n = len(rawdata) | 
					
						
							|  |  |  |         name = res.group('name') | 
					
						
							| 
									
										
										
										
											1999-08-26 15:52:33 +00:00
										 |  |  |         if self.__map_case: | 
					
						
							| 
									
										
										
										
											2001-02-09 10:48:30 +00:00
										 |  |  |             name = name.lower() | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |         pubid, syslit = res.group('pubid', 'syslit') | 
					
						
							|  |  |  |         if pubid is not None: | 
					
						
							|  |  |  |             pubid = pubid[1:-1]         # remove quotes | 
					
						
							| 
									
										
										
										
											2001-02-09 10:48:30 +00:00
										 |  |  |             pubid = ' '.join(pubid.split()) # normalize | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |         if syslit is not None: syslit = syslit[1:-1] # remove quotes | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         j = k = res.end(0) | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |         if k >= n: | 
					
						
							|  |  |  |             return -1 | 
					
						
							|  |  |  |         if rawdata[k] == '[': | 
					
						
							|  |  |  |             level = 0 | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |             k = k+1 | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |             dq = sq = 0 | 
					
						
							|  |  |  |             while k < n: | 
					
						
							|  |  |  |                 c = rawdata[k] | 
					
						
							|  |  |  |                 if not sq and c == '"': | 
					
						
							|  |  |  |                     dq = not dq | 
					
						
							|  |  |  |                 elif not dq and c == "'": | 
					
						
							|  |  |  |                     sq = not sq | 
					
						
							|  |  |  |                 elif sq or dq: | 
					
						
							|  |  |  |                     pass | 
					
						
							|  |  |  |                 elif level <= 0 and c == ']': | 
					
						
							|  |  |  |                     res = endbracket.match(rawdata, k+1) | 
					
						
							| 
									
										
										
										
											1998-10-19 13:28:26 +00:00
										 |  |  |                     if res is None: | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |                         return -1 | 
					
						
							|  |  |  |                     self.handle_doctype(name, pubid, syslit, rawdata[j+1:k]) | 
					
						
							|  |  |  |                     return res.end(0) | 
					
						
							|  |  |  |                 elif c == '<': | 
					
						
							|  |  |  |                     level = level + 1 | 
					
						
							|  |  |  |                 elif c == '>': | 
					
						
							|  |  |  |                     level = level - 1 | 
					
						
							|  |  |  |                     if level < 0: | 
					
						
							|  |  |  |                         self.syntax_error("bogus `>' in DOCTYPE") | 
					
						
							|  |  |  |                 k = k+1 | 
					
						
							| 
									
										
										
										
											1998-10-19 13:28:26 +00:00
										 |  |  |         res = endbracketfind.match(rawdata, k) | 
					
						
							|  |  |  |         if res is None: | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |             return -1 | 
					
						
							| 
									
										
										
										
											1998-10-19 13:28:26 +00:00
										 |  |  |         if endbracket.match(rawdata, k) is None: | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |             self.syntax_error('garbage in DOCTYPE') | 
					
						
							|  |  |  |         self.handle_doctype(name, pubid, syslit, None) | 
					
						
							|  |  |  |         return res.end(0) | 
					
						
							| 
									
										
										
										
											1998-01-29 14:55:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Internal -- handle CDATA tag, return length or -1 if not terminated | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  |     def parse_cdata(self, i): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         rawdata = self.rawdata | 
					
						
							| 
									
										
										
										
											2000-12-12 23:20:45 +00:00
										 |  |  |         if rawdata[i:i+9] != '<![CDATA[': | 
					
						
							| 
									
										
										
										
											2000-12-06 10:37:57 +00:00
										 |  |  |             raise Error('unexpected call to parse_cdata') | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         res = cdataclose.search(rawdata, i+9) | 
					
						
							| 
									
										
										
										
											1998-10-19 13:28:26 +00:00
										 |  |  |         if res is None: | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |             return -1 | 
					
						
							| 
									
										
										
										
											1999-08-26 15:52:33 +00:00
										 |  |  |         if not self.__accept_utf8 and \ | 
					
						
							|  |  |  |            illegal.search(rawdata, i+9, res.start(0)): | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |             self.syntax_error('illegal character in CDATA') | 
					
						
							|  |  |  |         if not self.stack: | 
					
						
							|  |  |  |             self.syntax_error('CDATA not in content') | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         self.handle_cdata(rawdata[i+9:res.start(0)]) | 
					
						
							|  |  |  |         return res.end(0) | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-12-18 20:17:13 +00:00
										 |  |  |     __xml_namespace_attributes = {'ns':None, 'src':None, 'prefix':None} | 
					
						
							| 
									
										
										
										
											1998-01-29 14:55:24 +00:00
										 |  |  |     # Internal -- handle a processing instruction tag | 
					
						
							|  |  |  |     def parse_proc(self, i): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         rawdata = self.rawdata | 
					
						
							|  |  |  |         end = procclose.search(rawdata, i) | 
					
						
							| 
									
										
										
										
											1998-10-19 13:28:26 +00:00
										 |  |  |         if end is None: | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |             return -1 | 
					
						
							|  |  |  |         j = end.start(0) | 
					
						
							| 
									
										
										
										
											1999-08-26 15:52:33 +00:00
										 |  |  |         if not self.__accept_utf8 and illegal.search(rawdata, i+2, j): | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |             self.syntax_error('illegal character in processing instruction') | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         res = tagfind.match(rawdata, i+2) | 
					
						
							| 
									
										
										
										
											1998-10-19 13:28:26 +00:00
										 |  |  |         if res is None: | 
					
						
							| 
									
										
										
										
											2000-12-06 10:37:57 +00:00
										 |  |  |             raise Error('unexpected call to parse_proc') | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         k = res.end(0) | 
					
						
							|  |  |  |         name = res.group(0) | 
					
						
							| 
									
										
										
										
											1999-08-26 15:52:33 +00:00
										 |  |  |         if self.__map_case: | 
					
						
							| 
									
										
										
										
											2001-02-09 10:48:30 +00:00
										 |  |  |             name = name.lower() | 
					
						
							| 
									
										
										
										
											1998-12-18 20:17:13 +00:00
										 |  |  |         if name == 'xml:namespace': | 
					
						
							|  |  |  |             self.syntax_error('old-fashioned namespace declaration') | 
					
						
							| 
									
										
										
										
											1998-12-21 18:15:28 +00:00
										 |  |  |             self.__use_namespaces = -1 | 
					
						
							| 
									
										
										
										
											1998-12-18 20:17:13 +00:00
										 |  |  |             # namespace declaration | 
					
						
							|  |  |  |             # this must come after the <?xml?> declaration (if any) | 
					
						
							|  |  |  |             # and before the <!DOCTYPE> (if any). | 
					
						
							|  |  |  |             if self.__seen_doctype or self.__seen_starttag: | 
					
						
							|  |  |  |                 self.syntax_error('xml:namespace declaration too late in document') | 
					
						
							|  |  |  |             attrdict, namespace, k = self.parse_attributes(name, k, j) | 
					
						
							|  |  |  |             if namespace: | 
					
						
							|  |  |  |                 self.syntax_error('namespace declaration inside namespace declaration') | 
					
						
							|  |  |  |             for attrname in attrdict.keys(): | 
					
						
							| 
									
										
										
										
											2002-06-01 14:18:47 +00:00
										 |  |  |                 if not attrname in self.__xml_namespace_attributes: | 
					
						
							| 
									
										
										
										
											1998-12-18 20:17:13 +00:00
										 |  |  |                     self.syntax_error("unknown attribute `%s' in xml:namespace tag" % attrname) | 
					
						
							| 
									
										
										
										
											2002-06-01 14:18:47 +00:00
										 |  |  |             if not 'ns' in attrdict or not 'prefix' in attrdict: | 
					
						
							| 
									
										
										
										
											1998-12-18 20:17:13 +00:00
										 |  |  |                 self.syntax_error('xml:namespace without required attributes') | 
					
						
							|  |  |  |             prefix = attrdict.get('prefix') | 
					
						
							|  |  |  |             if ncname.match(prefix) is None: | 
					
						
							|  |  |  |                 self.syntax_error('xml:namespace illegal prefix value') | 
					
						
							|  |  |  |                 return end.end(0) | 
					
						
							| 
									
										
										
										
											2002-06-01 14:18:47 +00:00
										 |  |  |             if prefix in self.__namespaces: | 
					
						
							| 
									
										
										
										
											1998-12-18 20:17:13 +00:00
										 |  |  |                 self.syntax_error('xml:namespace prefix not unique') | 
					
						
							|  |  |  |             self.__namespaces[prefix] = attrdict['ns'] | 
					
						
							|  |  |  |         else: | 
					
						
							| 
									
										
										
										
											2001-02-09 10:48:30 +00:00
										 |  |  |             if name.lower() == 'xml': | 
					
						
							| 
									
										
										
										
											1998-12-18 20:17:13 +00:00
										 |  |  |                 self.syntax_error('illegal processing instruction target name') | 
					
						
							|  |  |  |             self.handle_proc(name, rawdata[k:j]) | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         return end.end(0) | 
					
						
							| 
									
										
										
										
											1998-01-29 14:55:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Internal -- parse attributes between i and j | 
					
						
							| 
									
										
										
										
											1998-12-18 20:17:13 +00:00
										 |  |  |     def parse_attributes(self, tag, i, j): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         rawdata = self.rawdata | 
					
						
							|  |  |  |         attrdict = {} | 
					
						
							| 
									
										
										
										
											1998-12-18 20:17:13 +00:00
										 |  |  |         namespace = {} | 
					
						
							| 
									
										
										
										
											1998-10-19 13:28:26 +00:00
										 |  |  |         while i < j: | 
					
						
							|  |  |  |             res = attrfind.match(rawdata, i) | 
					
						
							|  |  |  |             if res is None: | 
					
						
							|  |  |  |                 break | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |             attrname, attrvalue = res.group('name', 'value') | 
					
						
							| 
									
										
										
										
											1999-08-26 15:52:33 +00:00
										 |  |  |             if self.__map_case: | 
					
						
							| 
									
										
										
										
											2001-02-09 10:48:30 +00:00
										 |  |  |                 attrname = attrname.lower() | 
					
						
							| 
									
										
										
										
											1998-12-18 20:17:13 +00:00
										 |  |  |             i = res.end(0) | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |             if attrvalue is None: | 
					
						
							| 
									
										
										
										
											1998-10-19 13:28:26 +00:00
										 |  |  |                 self.syntax_error("no value specified for attribute `%s'" % attrname) | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |                 attrvalue = attrname | 
					
						
							|  |  |  |             elif attrvalue[:1] == "'" == attrvalue[-1:] or \ | 
					
						
							|  |  |  |                  attrvalue[:1] == '"' == attrvalue[-1:]: | 
					
						
							|  |  |  |                 attrvalue = attrvalue[1:-1] | 
					
						
							| 
									
										
										
										
											1999-08-26 15:52:33 +00:00
										 |  |  |             elif not self.__accept_unquoted_attributes: | 
					
						
							| 
									
										
										
										
											1998-10-19 13:28:26 +00:00
										 |  |  |                 self.syntax_error("attribute `%s' value not quoted" % attrname) | 
					
						
							| 
									
										
										
										
											1998-12-18 20:17:13 +00:00
										 |  |  |             res = xmlns.match(attrname) | 
					
						
							|  |  |  |             if res is not None: | 
					
						
							|  |  |  |                 # namespace declaration | 
					
						
							|  |  |  |                 ncname = res.group('ncname') | 
					
						
							|  |  |  |                 namespace[ncname or ''] = attrvalue or None | 
					
						
							| 
									
										
										
										
											1998-12-21 18:15:28 +00:00
										 |  |  |                 if not self.__use_namespaces: | 
					
						
							|  |  |  |                     self.__use_namespaces = len(self.stack)+1 | 
					
						
							| 
									
										
										
										
											1998-12-18 20:17:13 +00:00
										 |  |  |                 continue | 
					
						
							| 
									
										
										
										
											1998-10-19 13:28:26 +00:00
										 |  |  |             if '<' in attrvalue: | 
					
						
							|  |  |  |                 self.syntax_error("`<' illegal in attribute value") | 
					
						
							| 
									
										
										
										
											2002-06-01 14:18:47 +00:00
										 |  |  |             if attrname in attrdict: | 
					
						
							| 
									
										
										
										
											1998-10-19 13:28:26 +00:00
										 |  |  |                 self.syntax_error("attribute `%s' specified twice" % attrname) | 
					
						
							| 
									
										
										
										
											2001-02-09 10:48:30 +00:00
										 |  |  |             attrvalue = attrvalue.translate(attrtrans) | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |             attrdict[attrname] = self.translate_references(attrvalue) | 
					
						
							| 
									
										
										
										
											1998-12-18 20:17:13 +00:00
										 |  |  |         return attrdict, namespace, i | 
					
						
							| 
									
										
										
										
											1998-01-29 14:55:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Internal -- handle starttag, return length or -1 if not terminated | 
					
						
							|  |  |  |     def parse_starttag(self, i): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         rawdata = self.rawdata | 
					
						
							|  |  |  |         # i points to start of tag | 
					
						
							| 
									
										
										
										
											1998-10-19 13:28:26 +00:00
										 |  |  |         end = endbracketfind.match(rawdata, i+1) | 
					
						
							|  |  |  |         if end is None: | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |             return -1 | 
					
						
							| 
									
										
										
										
											1998-10-19 13:28:26 +00:00
										 |  |  |         tag = starttagmatch.match(rawdata, i) | 
					
						
							|  |  |  |         if tag is None or tag.end(0) != end.end(0): | 
					
						
							|  |  |  |             self.syntax_error('garbage in starttag') | 
					
						
							|  |  |  |             return end.end(0) | 
					
						
							| 
									
										
										
										
											1998-12-18 20:17:13 +00:00
										 |  |  |         nstag = tagname = tag.group('tagname') | 
					
						
							| 
									
										
										
										
											1999-08-26 15:52:33 +00:00
										 |  |  |         if self.__map_case: | 
					
						
							| 
									
										
										
										
											2001-02-09 10:48:30 +00:00
										 |  |  |             nstag = tagname = nstag.lower() | 
					
						
							| 
									
										
										
										
											1998-10-19 13:28:26 +00:00
										 |  |  |         if not self.__seen_starttag and self.__seen_doctype and \ | 
					
						
							|  |  |  |            tagname != self.__seen_doctype: | 
					
						
							|  |  |  |             self.syntax_error('starttag does not match DOCTYPE') | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |         if self.__seen_starttag and not self.stack: | 
					
						
							|  |  |  |             self.syntax_error('multiple elements on top level') | 
					
						
							| 
									
										
										
										
											1998-10-19 13:28:26 +00:00
										 |  |  |         k, j = tag.span('attrs') | 
					
						
							| 
									
										
										
										
											1998-12-18 20:17:13 +00:00
										 |  |  |         attrdict, nsdict, k = self.parse_attributes(tagname, k, j) | 
					
						
							|  |  |  |         self.stack.append((tagname, nsdict, nstag)) | 
					
						
							| 
									
										
										
										
											1998-12-21 18:15:28 +00:00
										 |  |  |         if self.__use_namespaces: | 
					
						
							|  |  |  |             res = qname.match(tagname) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             res = None | 
					
						
							| 
									
										
										
										
											1998-12-18 20:17:13 +00:00
										 |  |  |         if res is not None: | 
					
						
							|  |  |  |             prefix, nstag = res.group('prefix', 'local') | 
					
						
							|  |  |  |             if prefix is None: | 
					
						
							|  |  |  |                 prefix = '' | 
					
						
							|  |  |  |             ns = None | 
					
						
							|  |  |  |             for t, d, nst in self.stack: | 
					
						
							| 
									
										
										
										
											2002-06-01 14:18:47 +00:00
										 |  |  |                 if prefix in d: | 
					
						
							| 
									
										
										
										
											1998-12-18 20:17:13 +00:00
										 |  |  |                     ns = d[prefix] | 
					
						
							|  |  |  |             if ns is None and prefix != '': | 
					
						
							|  |  |  |                 ns = self.__namespaces.get(prefix) | 
					
						
							|  |  |  |             if ns is not None: | 
					
						
							|  |  |  |                 nstag = ns + ' ' + nstag | 
					
						
							|  |  |  |             elif prefix != '': | 
					
						
							|  |  |  |                 nstag = prefix + ':' + nstag # undo split | 
					
						
							|  |  |  |             self.stack[-1] = tagname, nsdict, nstag | 
					
						
							|  |  |  |         # translate namespace of attributes | 
					
						
							| 
									
										
										
										
											2000-12-06 10:37:57 +00:00
										 |  |  |         attrnamemap = {} # map from new name to old name (used for error reporting) | 
					
						
							|  |  |  |         for key in attrdict.keys(): | 
					
						
							|  |  |  |             attrnamemap[key] = key | 
					
						
							| 
									
										
										
										
											1998-12-21 18:15:28 +00:00
										 |  |  |         if self.__use_namespaces: | 
					
						
							|  |  |  |             nattrdict = {} | 
					
						
							|  |  |  |             for key, val in attrdict.items(): | 
					
						
							| 
									
										
										
										
											2000-12-06 10:37:57 +00:00
										 |  |  |                 okey = key | 
					
						
							| 
									
										
										
										
											1998-12-21 18:15:28 +00:00
										 |  |  |                 res = qname.match(key) | 
					
						
							|  |  |  |                 if res is not None: | 
					
						
							|  |  |  |                     aprefix, key = res.group('prefix', 'local') | 
					
						
							| 
									
										
										
										
											1999-08-26 15:52:33 +00:00
										 |  |  |                     if self.__map_case: | 
					
						
							| 
									
										
										
										
											2001-02-09 10:48:30 +00:00
										 |  |  |                         key = key.lower() | 
					
						
							| 
									
										
										
										
											2004-05-02 20:37:13 +00:00
										 |  |  |                     if aprefix is not None: | 
					
						
							|  |  |  |                         ans = None | 
					
						
							|  |  |  |                         for t, d, nst in self.stack: | 
					
						
							|  |  |  |                             if aprefix in d: | 
					
						
							|  |  |  |                                 ans = d[aprefix] | 
					
						
							|  |  |  |                         if ans is None: | 
					
						
							|  |  |  |                             ans = self.__namespaces.get(aprefix) | 
					
						
							|  |  |  |                         if ans is not None: | 
					
						
							|  |  |  |                             key = ans + ' ' + key | 
					
						
							|  |  |  |                         else: | 
					
						
							|  |  |  |                             key = aprefix + ':' + key | 
					
						
							| 
									
										
										
										
											1998-12-21 18:15:28 +00:00
										 |  |  |                 nattrdict[key] = val | 
					
						
							| 
									
										
										
										
											2000-12-06 10:37:57 +00:00
										 |  |  |                 attrnamemap[key] = okey | 
					
						
							| 
									
										
										
										
											1998-12-21 18:15:28 +00:00
										 |  |  |             attrdict = nattrdict | 
					
						
							| 
									
										
										
										
											1998-12-18 20:17:13 +00:00
										 |  |  |         attributes = self.attributes.get(nstag) | 
					
						
							|  |  |  |         if attributes is not None: | 
					
						
							|  |  |  |             for key in attrdict.keys(): | 
					
						
							| 
									
										
										
										
											2002-06-01 14:18:47 +00:00
										 |  |  |                 if not key in attributes: | 
					
						
							| 
									
										
										
										
											2000-12-06 10:37:57 +00:00
										 |  |  |                     self.syntax_error("unknown attribute `%s' in tag `%s'" % (attrnamemap[key], tagname)) | 
					
						
							| 
									
										
										
										
											1998-12-18 20:17:13 +00:00
										 |  |  |             for key, val in attributes.items(): | 
					
						
							| 
									
										
										
										
											2002-06-01 14:18:47 +00:00
										 |  |  |                 if val is not None and not key in attrdict: | 
					
						
							| 
									
										
										
										
											1998-12-18 20:17:13 +00:00
										 |  |  |                     attrdict[key] = val | 
					
						
							|  |  |  |         method = self.elements.get(nstag, (None, None))[0] | 
					
						
							|  |  |  |         self.finish_starttag(nstag, attrdict, method) | 
					
						
							| 
									
										
										
										
											1998-10-19 13:28:26 +00:00
										 |  |  |         if tag.group('slash') == '/': | 
					
						
							|  |  |  |             self.finish_endtag(tagname) | 
					
						
							|  |  |  |         return tag.end(0) | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Internal -- parse endtag | 
					
						
							|  |  |  |     def parse_endtag(self, i): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         rawdata = self.rawdata | 
					
						
							| 
									
										
										
										
											1998-10-19 13:28:26 +00:00
										 |  |  |         end = endbracketfind.match(rawdata, i+1) | 
					
						
							|  |  |  |         if end is None: | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |             return -1 | 
					
						
							|  |  |  |         res = tagfind.match(rawdata, i+2) | 
					
						
							| 
									
										
										
										
											1998-10-19 13:28:26 +00:00
										 |  |  |         if res is None: | 
					
						
							| 
									
										
										
										
											1998-12-07 21:59:56 +00:00
										 |  |  |             if self.literal: | 
					
						
							|  |  |  |                 self.handle_data(rawdata[i]) | 
					
						
							|  |  |  |                 return i+1 | 
					
						
							| 
									
										
										
										
											1999-08-26 15:52:33 +00:00
										 |  |  |             if not self.__accept_missing_endtag_name: | 
					
						
							|  |  |  |                 self.syntax_error('no name specified in end tag') | 
					
						
							| 
									
										
										
										
											2000-06-21 20:01:10 +00:00
										 |  |  |             tag = self.stack[-1][0] | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |             k = i+2 | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             tag = res.group(0) | 
					
						
							| 
									
										
										
										
											1999-08-26 15:52:33 +00:00
										 |  |  |             if self.__map_case: | 
					
						
							| 
									
										
										
										
											2001-02-09 10:48:30 +00:00
										 |  |  |                 tag = tag.lower() | 
					
						
							| 
									
										
										
										
											1998-12-07 21:59:56 +00:00
										 |  |  |             if self.literal: | 
					
						
							| 
									
										
										
										
											1998-12-18 20:17:13 +00:00
										 |  |  |                 if not self.stack or tag != self.stack[-1][0]: | 
					
						
							| 
									
										
										
										
											1998-12-07 21:59:56 +00:00
										 |  |  |                     self.handle_data(rawdata[i]) | 
					
						
							|  |  |  |                     return i+1 | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |             k = res.end(0) | 
					
						
							| 
									
										
										
										
											1998-10-19 13:28:26 +00:00
										 |  |  |         if endbracket.match(rawdata, k) is None: | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |             self.syntax_error('garbage in end tag') | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         self.finish_endtag(tag) | 
					
						
							|  |  |  |         return end.end(0) | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Internal -- finish processing of start tag | 
					
						
							| 
									
										
										
										
											1998-12-18 20:17:13 +00:00
										 |  |  |     def finish_starttag(self, tagname, attrdict, method): | 
					
						
							|  |  |  |         if method is not None: | 
					
						
							|  |  |  |             self.handle_starttag(tagname, method, attrdict) | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											1998-12-18 20:17:13 +00:00
										 |  |  |             self.unknown_starttag(tagname, attrdict) | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Internal -- finish processing of end tag | 
					
						
							|  |  |  |     def finish_endtag(self, tag): | 
					
						
							| 
									
										
										
										
											2001-03-14 17:03:30 +00:00
										 |  |  |         self.literal = 0 | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         if not tag: | 
					
						
							|  |  |  |             self.syntax_error('name-less end tag') | 
					
						
							|  |  |  |             found = len(self.stack) - 1 | 
					
						
							|  |  |  |             if found < 0: | 
					
						
							|  |  |  |                 self.unknown_endtag(tag) | 
					
						
							|  |  |  |                 return | 
					
						
							|  |  |  |         else: | 
					
						
							| 
									
										
										
										
											1998-12-18 20:17:13 +00:00
										 |  |  |             found = -1 | 
					
						
							|  |  |  |             for i in range(len(self.stack)): | 
					
						
							|  |  |  |                 if tag == self.stack[i][0]: | 
					
						
							|  |  |  |                     found = i | 
					
						
							|  |  |  |             if found == -1: | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |                 self.syntax_error('unopened end tag') | 
					
						
							|  |  |  |                 return | 
					
						
							|  |  |  |         while len(self.stack) > found: | 
					
						
							|  |  |  |             if found < len(self.stack) - 1: | 
					
						
							| 
									
										
										
										
											1998-12-18 20:17:13 +00:00
										 |  |  |                 self.syntax_error('missing close tag for %s' % self.stack[-1][2]) | 
					
						
							|  |  |  |             nstag = self.stack[-1][2] | 
					
						
							|  |  |  |             method = self.elements.get(nstag, (None, None))[1] | 
					
						
							|  |  |  |             if method is not None: | 
					
						
							|  |  |  |                 self.handle_endtag(nstag, method) | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |             else: | 
					
						
							| 
									
										
										
										
											1998-12-18 20:17:13 +00:00
										 |  |  |                 self.unknown_endtag(nstag) | 
					
						
							| 
									
										
										
										
											1998-12-21 18:15:28 +00:00
										 |  |  |             if self.__use_namespaces == len(self.stack): | 
					
						
							|  |  |  |                 self.__use_namespaces = 0 | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |             del self.stack[-1] | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-01-29 14:55:24 +00:00
										 |  |  |     # Overridable -- handle xml processing instruction | 
					
						
							|  |  |  |     def handle_xml(self, encoding, standalone): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         pass | 
					
						
							| 
									
										
										
										
											1998-01-29 14:55:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Overridable -- handle DOCTYPE | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |     def handle_doctype(self, tag, pubid, syslit, data): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         pass | 
					
						
							| 
									
										
										
										
											1998-01-29 14:55:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  |     # Overridable -- handle start tag | 
					
						
							|  |  |  |     def handle_starttag(self, tag, method, attrs): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         method(attrs) | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Overridable -- handle end tag | 
					
						
							|  |  |  |     def handle_endtag(self, tag, method): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         method() | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Example -- handle character reference, no need to override | 
					
						
							|  |  |  |     def handle_charref(self, name): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         try: | 
					
						
							|  |  |  |             if name[0] == 'x': | 
					
						
							| 
									
										
										
										
											2001-02-09 10:48:30 +00:00
										 |  |  |                 n = int(name[1:], 16) | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |             else: | 
					
						
							| 
									
										
										
										
											2001-02-09 10:48:30 +00:00
										 |  |  |                 n = int(name) | 
					
						
							|  |  |  |         except ValueError: | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |             self.unknown_charref(name) | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         if not 0 <= n <= 255: | 
					
						
							|  |  |  |             self.unknown_charref(name) | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         self.handle_data(chr(n)) | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Definition of entities -- derived classes may override | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |     entitydefs = {'lt': '<',        # must use charref | 
					
						
							|  |  |  |                   'gt': '>', | 
					
						
							|  |  |  |                   'amp': '&',       # must use charref | 
					
						
							|  |  |  |                   'quot': '"', | 
					
						
							|  |  |  |                   'apos': ''', | 
					
						
							|  |  |  |                   } | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Example -- handle data, should be overridden | 
					
						
							|  |  |  |     def handle_data(self, data): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         pass | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Example -- handle cdata, could be overridden | 
					
						
							|  |  |  |     def handle_cdata(self, data): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         pass | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Example -- handle comment, could be overridden | 
					
						
							|  |  |  |     def handle_comment(self, data): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         pass | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Example -- handle processing instructions, could be overridden | 
					
						
							|  |  |  |     def handle_proc(self, name, data): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         pass | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Example -- handle relatively harmless syntax errors, could be overridden | 
					
						
							| 
									
										
										
										
											1998-01-29 14:55:24 +00:00
										 |  |  |     def syntax_error(self, message): | 
					
						
							| 
									
										
										
										
											2000-12-06 10:37:57 +00:00
										 |  |  |         raise Error('Syntax error at line %d: %s' % (self.lineno, message)) | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # To be overridden -- handlers for unknown objects | 
					
						
							|  |  |  |     def unknown_starttag(self, tag, attrs): pass | 
					
						
							|  |  |  |     def unknown_endtag(self, tag): pass | 
					
						
							|  |  |  |     def unknown_charref(self, ref): pass | 
					
						
							| 
									
										
										
										
											1999-08-26 15:52:33 +00:00
										 |  |  |     def unknown_entityref(self, name): | 
					
						
							|  |  |  |         self.syntax_error("reference to unknown entity `&%s;'" % name) | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class TestXMLParser(XMLParser): | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-08-26 15:52:33 +00:00
										 |  |  |     def __init__(self, **kw): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         self.testdata = "" | 
					
						
							| 
									
										
										
										
											2003-02-27 20:14:51 +00:00
										 |  |  |         XMLParser.__init__(self, **kw) | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-01-29 14:55:24 +00:00
										 |  |  |     def handle_xml(self, encoding, standalone): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         self.flush() | 
					
						
							|  |  |  |         print 'xml: encoding =',encoding,'standalone =',standalone | 
					
						
							| 
									
										
										
										
											1998-01-29 14:55:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |     def handle_doctype(self, tag, pubid, syslit, data): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         self.flush() | 
					
						
							| 
									
										
										
										
											2004-02-12 17:35:32 +00:00
										 |  |  |         print 'DOCTYPE:',tag, repr(data) | 
					
						
							| 
									
										
										
										
											1998-01-29 14:55:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  |     def handle_data(self, data): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         self.testdata = self.testdata + data | 
					
						
							| 
									
										
										
										
											2004-02-12 17:35:32 +00:00
										 |  |  |         if len(repr(self.testdata)) >= 70: | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |             self.flush() | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def flush(self): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         data = self.testdata | 
					
						
							|  |  |  |         if data: | 
					
						
							|  |  |  |             self.testdata = "" | 
					
						
							| 
									
										
										
										
											2004-02-12 17:35:32 +00:00
										 |  |  |             print 'data:', repr(data) | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def handle_cdata(self, data): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         self.flush() | 
					
						
							| 
									
										
										
										
											2004-02-12 17:35:32 +00:00
										 |  |  |         print 'cdata:', repr(data) | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def handle_proc(self, name, data): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         self.flush() | 
					
						
							| 
									
										
										
										
											2004-02-12 17:35:32 +00:00
										 |  |  |         print 'processing:',name,repr(data) | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def handle_comment(self, data): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         self.flush() | 
					
						
							| 
									
										
										
										
											2004-02-12 17:35:32 +00:00
										 |  |  |         r = repr(data) | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         if len(r) > 68: | 
					
						
							|  |  |  |             r = r[:32] + '...' + r[-32:] | 
					
						
							|  |  |  |         print 'comment:', r | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-01-29 14:55:24 +00:00
										 |  |  |     def syntax_error(self, message): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         print 'error at line %d:' % self.lineno, message | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def unknown_starttag(self, tag, attrs): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         self.flush() | 
					
						
							|  |  |  |         if not attrs: | 
					
						
							|  |  |  |             print 'start tag: <' + tag + '>' | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             print 'start tag: <' + tag, | 
					
						
							|  |  |  |             for name, value in attrs.items(): | 
					
						
							|  |  |  |                 print name + '=' + '"' + value + '"', | 
					
						
							|  |  |  |             print '>' | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def unknown_endtag(self, tag): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         self.flush() | 
					
						
							|  |  |  |         print 'end tag: </' + tag + '>' | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def unknown_entityref(self, ref): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         self.flush() | 
					
						
							|  |  |  |         print '*** unknown entity ref: &' + ref + ';' | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def unknown_charref(self, ref): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         self.flush() | 
					
						
							|  |  |  |         print '*** unknown char ref: &#' + ref + ';' | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def close(self): | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         XMLParser.close(self) | 
					
						
							|  |  |  |         self.flush() | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | def test(args = None): | 
					
						
							| 
									
										
										
										
											1998-12-22 14:23:50 +00:00
										 |  |  |     import sys, getopt | 
					
						
							|  |  |  |     from time import time | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if not args: | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         args = sys.argv[1:] | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-12-22 14:23:50 +00:00
										 |  |  |     opts, args = getopt.getopt(args, 'st') | 
					
						
							|  |  |  |     klass = TestXMLParser | 
					
						
							|  |  |  |     do_time = 0 | 
					
						
							|  |  |  |     for o, a in opts: | 
					
						
							|  |  |  |         if o == '-s': | 
					
						
							|  |  |  |             klass = XMLParser | 
					
						
							|  |  |  |         elif o == '-t': | 
					
						
							|  |  |  |             do_time = 1 | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if args: | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         file = args[0] | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  |     else: | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         file = 'test.xml' | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if file == '-': | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         f = sys.stdin | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  |     else: | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         try: | 
					
						
							|  |  |  |             f = open(file, 'r') | 
					
						
							|  |  |  |         except IOError, msg: | 
					
						
							|  |  |  |             print file, ":", msg | 
					
						
							|  |  |  |             sys.exit(1) | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     data = f.read() | 
					
						
							|  |  |  |     if f is not sys.stdin: | 
					
						
							| 
									
										
										
										
											1998-03-26 21:13:24 +00:00
										 |  |  |         f.close() | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     x = klass() | 
					
						
							| 
									
										
										
										
											1998-12-22 14:23:50 +00:00
										 |  |  |     t0 = time() | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |     try: | 
					
						
							| 
									
										
										
										
											1998-12-22 14:23:50 +00:00
										 |  |  |         if do_time: | 
					
						
							|  |  |  |             x.feed(data) | 
					
						
							|  |  |  |             x.close() | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             for c in data: | 
					
						
							|  |  |  |                 x.feed(c) | 
					
						
							|  |  |  |             x.close() | 
					
						
							| 
									
										
										
										
											2000-12-06 10:37:57 +00:00
										 |  |  |     except Error, msg: | 
					
						
							| 
									
										
										
										
											1998-12-22 14:23:50 +00:00
										 |  |  |         t1 = time() | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |         print msg | 
					
						
							| 
									
										
										
										
											1998-12-22 14:23:50 +00:00
										 |  |  |         if do_time: | 
					
						
							|  |  |  |             print 'total time: %g' % (t1-t0) | 
					
						
							| 
									
										
										
										
											1998-04-03 16:02:39 +00:00
										 |  |  |         sys.exit(1) | 
					
						
							| 
									
										
										
										
											1998-12-22 14:23:50 +00:00
										 |  |  |     t1 = time() | 
					
						
							|  |  |  |     if do_time: | 
					
						
							|  |  |  |         print 'total time: %g' % (t1-t0) | 
					
						
							| 
									
										
										
										
											1997-11-18 15:09:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if __name__ == '__main__': | 
					
						
							|  |  |  |     test() |