_markupbase.py 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395
  1. """Shared support for scanning document type declarations in HTML and XHTML.
  2. This module is used as a foundation for the html.parser module. It has no
  3. documented public API and should not be used directly.
  4. """
  5. import re
  6. _declname_match = re.compile(r'[a-zA-Z][-_.a-zA-Z0-9]*\s*').match
  7. _declstringlit_match = re.compile(r'(\'[^\']*\'|"[^"]*")\s*').match
  8. _commentclose = re.compile(r'--\s*>')
  9. _markedsectionclose = re.compile(r']\s*]\s*>')
  10. # An analysis of the MS-Word extensions is available at
  11. # http://www.planetpublish.com/xmlarena/xap/Thursday/WordtoXML.pdf
  12. _msmarkedsectionclose = re.compile(r']\s*>')
  13. del re
  14. class ParserBase:
  15. """Parser base class which provides some common support methods used
  16. by the SGML/HTML and XHTML parsers."""
  17. def __init__(self):
  18. if self.__class__ is ParserBase:
  19. raise RuntimeError(
  20. "_markupbase.ParserBase must be subclassed")
  21. def error(self, message):
  22. raise NotImplementedError(
  23. "subclasses of ParserBase must override error()")
  24. def reset(self):
  25. self.lineno = 1
  26. self.offset = 0
  27. def getpos(self):
  28. """Return current line number and offset."""
  29. return self.lineno, self.offset
  30. # Internal -- update line number and offset. This should be
  31. # called for each piece of data exactly once, in order -- in other
  32. # words the concatenation of all the input strings to this
  33. # function should be exactly the entire input.
  34. def updatepos(self, i, j):
  35. if i >= j:
  36. return j
  37. rawdata = self.rawdata
  38. nlines = rawdata.count("\n", i, j)
  39. if nlines:
  40. self.lineno = self.lineno + nlines
  41. pos = rawdata.rindex("\n", i, j) # Should not fail
  42. self.offset = j-(pos+1)
  43. else:
  44. self.offset = self.offset + j-i
  45. return j
  46. _decl_otherchars = ''
  47. # Internal -- parse declaration (for use by subclasses).
  48. def parse_declaration(self, i):
  49. # This is some sort of declaration; in "HTML as
  50. # deployed," this should only be the document type
  51. # declaration ("<!DOCTYPE html...>").
  52. # ISO 8879:1986, however, has more complex
  53. # declaration syntax for elements in <!...>, including:
  54. # --comment--
  55. # [marked section]
  56. # name in the following list: ENTITY, DOCTYPE, ELEMENT,
  57. # ATTLIST, NOTATION, SHORTREF, USEMAP,
  58. # LINKTYPE, LINK, IDLINK, USELINK, SYSTEM
  59. rawdata = self.rawdata
  60. j = i + 2
  61. assert rawdata[i:j] == "<!", "unexpected call to parse_declaration"
  62. if rawdata[j:j+1] == ">":
  63. # the empty comment <!>
  64. return j + 1
  65. if rawdata[j:j+1] in ("-", ""):
  66. # Start of comment followed by buffer boundary,
  67. # or just a buffer boundary.
  68. return -1
  69. # A simple, practical version could look like: ((name|stringlit) S*) + '>'
  70. n = len(rawdata)
  71. if rawdata[j:j+2] == '--': #comment
  72. # Locate --.*-- as the body of the comment
  73. return self.parse_comment(i)
  74. elif rawdata[j] == '[': #marked section
  75. # Locate [statusWord [...arbitrary SGML...]] as the body of the marked section
  76. # Where statusWord is one of TEMP, CDATA, IGNORE, INCLUDE, RCDATA
  77. # Note that this is extended by Microsoft Office "Save as Web" function
  78. # to include [if...] and [endif].
  79. return self.parse_marked_section(i)
  80. else: #all other declaration elements
  81. decltype, j = self._scan_name(j, i)
  82. if j < 0:
  83. return j
  84. if decltype == "doctype":
  85. self._decl_otherchars = ''
  86. while j < n:
  87. c = rawdata[j]
  88. if c == ">":
  89. # end of declaration syntax
  90. data = rawdata[i+2:j]
  91. if decltype == "doctype":
  92. self.handle_decl(data)
  93. else:
  94. # According to the HTML5 specs sections "8.2.4.44 Bogus
  95. # comment state" and "8.2.4.45 Markup declaration open
  96. # state", a comment token should be emitted.
  97. # Calling unknown_decl provides more flexibility though.
  98. self.unknown_decl(data)
  99. return j + 1
  100. if c in "\"'":
  101. m = _declstringlit_match(rawdata, j)
  102. if not m:
  103. return -1 # incomplete
  104. j = m.end()
  105. elif c in "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ":
  106. name, j = self._scan_name(j, i)
  107. elif c in self._decl_otherchars:
  108. j = j + 1
  109. elif c == "[":
  110. # this could be handled in a separate doctype parser
  111. if decltype == "doctype":
  112. j = self._parse_doctype_subset(j + 1, i)
  113. elif decltype in {"attlist", "linktype", "link", "element"}:
  114. # must tolerate []'d groups in a content model in an element declaration
  115. # also in data attribute specifications of attlist declaration
  116. # also link type declaration subsets in linktype declarations
  117. # also link attribute specification lists in link declarations
  118. self.error("unsupported '[' char in %s declaration" % decltype)
  119. else:
  120. self.error("unexpected '[' char in declaration")
  121. else:
  122. self.error(
  123. "unexpected %r char in declaration" % rawdata[j])
  124. if j < 0:
  125. return j
  126. return -1 # incomplete
  127. # Internal -- parse a marked section
  128. # Override this to handle MS-word extension syntax <![if word]>content<![endif]>
  129. def parse_marked_section(self, i, report=1):
  130. rawdata= self.rawdata
  131. assert rawdata[i:i+3] == '<![', "unexpected call to parse_marked_section()"
  132. sectName, j = self._scan_name( i+3, i )
  133. if j < 0:
  134. return j
  135. if sectName in {"temp", "cdata", "ignore", "include", "rcdata"}:
  136. # look for standard ]]> ending
  137. match= _markedsectionclose.search(rawdata, i+3)
  138. elif sectName in {"if", "else", "endif"}:
  139. # look for MS Office ]> ending
  140. match= _msmarkedsectionclose.search(rawdata, i+3)
  141. else:
  142. self.error('unknown status keyword %r in marked section' % rawdata[i+3:j])
  143. if not match:
  144. return -1
  145. if report:
  146. j = match.start(0)
  147. self.unknown_decl(rawdata[i+3: j])
  148. return match.end(0)
  149. # Internal -- parse comment, return length or -1 if not terminated
  150. def parse_comment(self, i, report=1):
  151. rawdata = self.rawdata
  152. if rawdata[i:i+4] != '<!--':
  153. self.error('unexpected call to parse_comment()')
  154. match = _commentclose.search(rawdata, i+4)
  155. if not match:
  156. return -1
  157. if report:
  158. j = match.start(0)
  159. self.handle_comment(rawdata[i+4: j])
  160. return match.end(0)
  161. # Internal -- scan past the internal subset in a <!DOCTYPE declaration,
  162. # returning the index just past any whitespace following the trailing ']'.
  163. def _parse_doctype_subset(self, i, declstartpos):
  164. rawdata = self.rawdata
  165. n = len(rawdata)
  166. j = i
  167. while j < n:
  168. c = rawdata[j]
  169. if c == "<":
  170. s = rawdata[j:j+2]
  171. if s == "<":
  172. # end of buffer; incomplete
  173. return -1
  174. if s != "<!":
  175. self.updatepos(declstartpos, j + 1)
  176. self.error("unexpected char in internal subset (in %r)" % s)
  177. if (j + 2) == n:
  178. # end of buffer; incomplete
  179. return -1
  180. if (j + 4) > n:
  181. # end of buffer; incomplete
  182. return -1
  183. if rawdata[j:j+4] == "<!--":
  184. j = self.parse_comment(j, report=0)
  185. if j < 0:
  186. return j
  187. continue
  188. name, j = self._scan_name(j + 2, declstartpos)
  189. if j == -1:
  190. return -1
  191. if name not in {"attlist", "element", "entity", "notation"}:
  192. self.updatepos(declstartpos, j + 2)
  193. self.error(
  194. "unknown declaration %r in internal subset" % name)
  195. # handle the individual names
  196. meth = getattr(self, "_parse_doctype_" + name)
  197. j = meth(j, declstartpos)
  198. if j < 0:
  199. return j
  200. elif c == "%":
  201. # parameter entity reference
  202. if (j + 1) == n:
  203. # end of buffer; incomplete
  204. return -1
  205. s, j = self._scan_name(j + 1, declstartpos)
  206. if j < 0:
  207. return j
  208. if rawdata[j] == ";":
  209. j = j + 1
  210. elif c == "]":
  211. j = j + 1
  212. while j < n and rawdata[j].isspace():
  213. j = j + 1
  214. if j < n:
  215. if rawdata[j] == ">":
  216. return j
  217. self.updatepos(declstartpos, j)
  218. self.error("unexpected char after internal subset")
  219. else:
  220. return -1
  221. elif c.isspace():
  222. j = j + 1
  223. else:
  224. self.updatepos(declstartpos, j)
  225. self.error("unexpected char %r in internal subset" % c)
  226. # end of buffer reached
  227. return -1
  228. # Internal -- scan past <!ELEMENT declarations
  229. def _parse_doctype_element(self, i, declstartpos):
  230. name, j = self._scan_name(i, declstartpos)
  231. if j == -1:
  232. return -1
  233. # style content model; just skip until '>'
  234. rawdata = self.rawdata
  235. if '>' in rawdata[j:]:
  236. return rawdata.find(">", j) + 1
  237. return -1
  238. # Internal -- scan past <!ATTLIST declarations
  239. def _parse_doctype_attlist(self, i, declstartpos):
  240. rawdata = self.rawdata
  241. name, j = self._scan_name(i, declstartpos)
  242. c = rawdata[j:j+1]
  243. if c == "":
  244. return -1
  245. if c == ">":
  246. return j + 1
  247. while 1:
  248. # scan a series of attribute descriptions; simplified:
  249. # name type [value] [#constraint]
  250. name, j = self._scan_name(j, declstartpos)
  251. if j < 0:
  252. return j
  253. c = rawdata[j:j+1]
  254. if c == "":
  255. return -1
  256. if c == "(":
  257. # an enumerated type; look for ')'
  258. if ")" in rawdata[j:]:
  259. j = rawdata.find(")", j) + 1
  260. else:
  261. return -1
  262. while rawdata[j:j+1].isspace():
  263. j = j + 1
  264. if not rawdata[j:]:
  265. # end of buffer, incomplete
  266. return -1
  267. else:
  268. name, j = self._scan_name(j, declstartpos)
  269. c = rawdata[j:j+1]
  270. if not c:
  271. return -1
  272. if c in "'\"":
  273. m = _declstringlit_match(rawdata, j)
  274. if m:
  275. j = m.end()
  276. else:
  277. return -1
  278. c = rawdata[j:j+1]
  279. if not c:
  280. return -1
  281. if c == "#":
  282. if rawdata[j:] == "#":
  283. # end of buffer
  284. return -1
  285. name, j = self._scan_name(j + 1, declstartpos)
  286. if j < 0:
  287. return j
  288. c = rawdata[j:j+1]
  289. if not c:
  290. return -1
  291. if c == '>':
  292. # all done
  293. return j + 1
  294. # Internal -- scan past <!NOTATION declarations
  295. def _parse_doctype_notation(self, i, declstartpos):
  296. name, j = self._scan_name(i, declstartpos)
  297. if j < 0:
  298. return j
  299. rawdata = self.rawdata
  300. while 1:
  301. c = rawdata[j:j+1]
  302. if not c:
  303. # end of buffer; incomplete
  304. return -1
  305. if c == '>':
  306. return j + 1
  307. if c in "'\"":
  308. m = _declstringlit_match(rawdata, j)
  309. if not m:
  310. return -1
  311. j = m.end()
  312. else:
  313. name, j = self._scan_name(j, declstartpos)
  314. if j < 0:
  315. return j
  316. # Internal -- scan past <!ENTITY declarations
  317. def _parse_doctype_entity(self, i, declstartpos):
  318. rawdata = self.rawdata
  319. if rawdata[i:i+1] == "%":
  320. j = i + 1
  321. while 1:
  322. c = rawdata[j:j+1]
  323. if not c:
  324. return -1
  325. if c.isspace():
  326. j = j + 1
  327. else:
  328. break
  329. else:
  330. j = i
  331. name, j = self._scan_name(j, declstartpos)
  332. if j < 0:
  333. return j
  334. while 1:
  335. c = self.rawdata[j:j+1]
  336. if not c:
  337. return -1
  338. if c in "'\"":
  339. m = _declstringlit_match(rawdata, j)
  340. if m:
  341. j = m.end()
  342. else:
  343. return -1 # incomplete
  344. elif c == ">":
  345. return j + 1
  346. else:
  347. name, j = self._scan_name(j, declstartpos)
  348. if j < 0:
  349. return j
  350. # Internal -- scan a name token and the new position and the token, or
  351. # return -1 if we've reached the end of the buffer.
  352. def _scan_name(self, i, declstartpos):
  353. rawdata = self.rawdata
  354. n = len(rawdata)
  355. if i == n:
  356. return None, -1
  357. m = _declname_match(rawdata, i)
  358. if m:
  359. s = m.group()
  360. name = s.strip()
  361. if (i + len(s)) == n:
  362. return None, -1 # end of buffer
  363. return name.lower(), m.end()
  364. else:
  365. self.updatepos(declstartpos, i)
  366. self.error("expected name token at %r"
  367. % rawdata[declstartpos:declstartpos+20])
  368. # To be overridden -- handlers for unknown objects
  369. def unknown_decl(self, data):
  370. pass