;;; xmltok.el --- XML tokenization -*- lexical-binding:t -*- ;; Copyright (C) 2003, 2007-2021 Free Software Foundation, Inc. ;; Author: James Clark ;; Keywords: wp, hypermedia, languages, XML ;; This file is part of GNU Emacs. ;; GNU Emacs is free software: you can redistribute it and/or modify ;; it under the terms of the GNU General Public License as published by ;; the Free Software Foundation, either version 3 of the License, or ;; (at your option) any later version. ;; GNU Emacs is distributed in the hope that it will be useful, ;; but WITHOUT ANY WARRANTY; without even the implied warranty of ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the ;; GNU General Public License for more details. ;; You should have received a copy of the GNU General Public License ;; along with GNU Emacs. If not, see . ;;; Commentary: ;; This implements an XML 1.0 parser. It also implements the XML ;; Namespaces Recommendation. It is designed to be conforming, but it ;; works a bit differently from a normal XML parser. An XML document ;; consists of the prolog and an instance. The prolog is parsed as a ;; single unit using `xmltok-forward-prolog'. The instance is ;; considered as a sequence of tokens, where a token is something like ;; a start-tag, a comment, a chunk of data or a CDATA section. The ;; tokenization of the instance is stateless: the tokenization of one ;; part of the instance does not depend on tokenization of the ;; preceding part of the instance. This allows the instance to be ;; parsed incrementally. The main entry point is `xmltok-forward': ;; this can be called at any point in the instance provided it is ;; between tokens. ;; ;; This is a non-validating XML 1.0 processor. It does not resolve ;; parameter entities (including the external DTD subset) and it does ;; not resolve external general entities. ;; ;; It is non-conformant by design in the following respects. ;; ;; 1. It expects the client to detect aspects of well-formedness that ;; are not internal to a single token, specifically checking that ;; end-tags match start-tags and that the instance contains exactly ;; one element. ;; ;; 2. It expects the client to detect duplicate attributes. Detection ;; of duplicate attributes after expansion of namespace prefixes ;; requires the namespace processing state. Detection of duplicate ;; attributes before expansion of namespace prefixes does not, but is ;; redundant given that the client will do detection of duplicate ;; attributes after expansion of namespace prefixes. ;; ;; 3. It allows the client to recover from well-formedness errors. ;; This is essential for use in applications where the document is ;; being parsed during the editing process. ;; ;; 4. It does not support documents that do not conform to the lexical ;; requirements of the XML Namespaces Recommendation (e.g. a document ;; with a colon in an entity name). ;; ;; There are also a number of things that have not yet been ;; implemented that make it non-conformant. ;; ;; 1. It does not implement default attributes. ATTLIST declarations ;; are parsed, but no checking is done on the content of attribute ;; value literals specifying default attribute values, and default ;; attribute values are not reported to the client. ;; ;; 2. It does not implement internal entities containing elements. If ;; an internal entity is referenced and parsing its replacement text ;; yields one or more tags, then it will skip the reference and ;; report this to the client. ;; ;; 3. It does not check the syntax of public identifiers in the DTD. ;; ;; 4. It allows some non-ASCII characters in certain situations where ;; it should not. For example, it only enforces XML 1.0's ;; restrictions on name characters strictly for ASCII characters. The ;; problem here is XML's character model is based squarely on Unicode, ;; whereas Emacs's is not (as of version 21). It is not clear what ;; the right thing to do is. ;;; Code: (defvar xmltok-type nil) (defvar xmltok-start nil) (defvar xmltok-name-colon nil) (defvar xmltok-name-end nil) (defvar xmltok-replacement nil "String containing replacement for a character or entity reference.") (defvar xmltok-attributes nil "List containing attributes of last scanned element. Each member of the list is a vector representing an attribute, which can be accessed using the functions `xmltok-attribute-name-start', `xmltok-attribute-name-colon', `xmltok-attribute-name-end', `xmltok-attribute-value-start', `xmltok-attribute-value-end', `xmltok-attribute-raw-normalized-value', `xmltok-attribute-refs'.") (defvar xmltok-namespace-attributes nil "List containing namespace declarations of last scanned element. List has same format as `xmltok-attributes'.") (defvar xmltok-dtd nil "Information about the DTD used by `xmltok-forward'. `xmltok-forward-prolog' sets this up. It consists of an alist of general entity names vs definitions. The first member of the alist is t if references to entities not in the alist are well-formed \(e.g. because there's an external subset that wasn't parsed). Each general entity name is a string. The definition is either nil, a symbol, a string, a cons cell. If the definition is nil, then it means that it's an internal entity but the result of parsing it is unknown. If it is a symbol, then the symbol is either `unparsed', meaning the entity is an unparsed entity, `external', meaning the entity is or references an external entity, `element', meaning the entity includes one or more elements, or `not-well-formed', meaning the replacement text is not well-formed. If the definition is a string, then the replacement text of the entity is that string; this happens only during the parsing of the prolog. If the definition is a cons cell \(ER . AR), then ER specifies the string that results from referencing the entity in element content and AR is either nil, meaning the replacement text included a <, or a string which is the normalized attribute value.") (defvar xmltok-errors nil "List of errors detected by `xmltok-forward' and `xmltok-forward-prolog'. When `xmltok-forward' and `xmltok-forward-prolog' detect a well-formedness error, they will add an entry to the beginning of this list. Each entry is a vector [MESSAGE START END], where MESSAGE is a string giving the error message and START and END are integers indicating the position of the error.") (defmacro xmltok-save (&rest body) (declare (indent 0) (debug t)) `(let (xmltok-type xmltok-start xmltok-name-colon xmltok-name-end xmltok-replacement xmltok-attributes xmltok-namespace-attributes xmltok-errors) ,@body)) (defsubst xmltok-attribute-name-start (att) (aref att 0)) (defsubst xmltok-attribute-name-colon (att) (aref att 1)) (defsubst xmltok-attribute-name-end (att) (aref att 2)) (defsubst xmltok-attribute-value-start (att) (aref att 3)) (defsubst xmltok-attribute-value-end (att) (aref att 4)) (defsubst xmltok-attribute-raw-normalized-value (att) "Return an object representing the normalized value of ATT. This can be t indicating that the normalized value is the same as the buffer substring from the start to the end of the value, or nil indicating that the value is not well-formed or a string." (aref att 5)) (defsubst xmltok-attribute-refs (att) "Return a list of the entity and character references in ATT. Each member is a vector [TYPE START END] where TYPE is either char-ref or entity-ref and START and END are integers giving the start and end of the reference. Nested entity references are not included in the list." (aref att 6)) (defun xmltok-attribute-prefix (att) (let ((colon (xmltok-attribute-name-colon att))) (and colon (buffer-substring-no-properties (xmltok-attribute-name-start att) colon)))) (defun xmltok-attribute-local-name (att) (let ((colon (xmltok-attribute-name-colon att))) (buffer-substring-no-properties (if colon (1+ colon) (xmltok-attribute-name-start att)) (xmltok-attribute-name-end att)))) (defun xmltok-attribute-value (att) (let ((rnv (xmltok-attribute-raw-normalized-value att))) (and rnv (if (stringp rnv) rnv (buffer-substring-no-properties (xmltok-attribute-value-start att) (xmltok-attribute-value-end att)))))) (defun xmltok-start-tag-prefix () (and xmltok-name-colon (buffer-substring-no-properties (1+ xmltok-start) xmltok-name-colon))) (defun xmltok-start-tag-local-name () (buffer-substring-no-properties (1+ (or xmltok-name-colon xmltok-start)) xmltok-name-end)) (defun xmltok-end-tag-prefix () (and xmltok-name-colon (buffer-substring-no-properties (+ 2 xmltok-start) xmltok-name-colon))) (defun xmltok-end-tag-local-name () (buffer-substring-no-properties (if xmltok-name-colon (1+ xmltok-name-colon) (+ 2 xmltok-start)) xmltok-name-end)) (defun xmltok-start-tag-qname () (buffer-substring-no-properties (+ xmltok-start 1) xmltok-name-end)) (defun xmltok-end-tag-qname () (buffer-substring-no-properties (+ xmltok-start 2) xmltok-name-end)) (defsubst xmltok-make-attribute (name-begin name-colon name-end &optional value-begin value-end raw-normalized-value) "Make an attribute. RAW-NORMALIZED-VALUE is nil if the value is not well-formed, t if the normalized value is the string between VALUE-BEGIN and VALUE-END, otherwise a STRING giving the value." (vector name-begin name-colon name-end value-begin value-end raw-normalized-value nil)) (defsubst xmltok-error-message (err) (aref err 0)) (defsubst xmltok-error-start (err) (aref err 1)) (defsubst xmltok-error-end (err) (aref err 2)) (defsubst xmltok-make-error (message start end) (vector message start end)) (defun xmltok-add-error (message &optional start end) (push (xmltok-make-error message (or start xmltok-start) (or end (point))) xmltok-errors)) (defun xmltok-forward () (setq xmltok-start (point)) (let* ((case-fold-search nil) (space-count (skip-chars-forward " \t\r\n")) (ch (char-after))) (cond ((eq ch ?\<) (cond ((> space-count 0) (setq xmltok-type 'space)) (t (forward-char 1) (xmltok-scan-after-lt)))) ((eq ch ?\&) (cond ((> space-count 0) (setq xmltok-type 'space)) (t (forward-char 1) (xmltok-scan-after-amp 'xmltok-handle-entity)))) ((re-search-forward "[<&]\\|\\(]]>\\)" nil t) (cond ((not (match-beginning 1)) (goto-char (match-beginning 0)) ;; must have got a non-space char (setq xmltok-type 'data)) ((= (match-beginning 1) xmltok-start) (xmltok-add-error "Found `]]>' not closing a CDATA section") (setq xmltok-type 'not-well-formed)) (t (goto-char (match-beginning 0)) (setq xmltok-type (if (= (point) (+ xmltok-start space-count)) 'space 'data))))) ((eq ch nil) (setq xmltok-type (if (> space-count 0) 'space nil))) (t (goto-char (point-max)) (setq xmltok-type 'data))))) (eval-when-compile ;; A symbolic regexp is represented by a list whose CAR is the string ;; containing the regexp and whose cdr is a list of symbolic names ;; for the groups in the string. ;; Construct a symbolic regexp from a regexp. (defun xmltok-r (str) (cons str nil)) ;; Concatenate zero of more regexps and symbolic regexps. (defun xmltok+ (&rest args) (let (strs names) (while args (let ((arg (car args))) (if (stringp arg) (setq strs (cons arg strs)) (setq strs (cons (car arg) strs)) (setq names (cons (cdr arg) names))) (setq args (cdr args)))) (cons (apply #'concat (nreverse strs)) (apply #'append (nreverse names)))))) (eval-when-compile ;; Make a symbolic group named NAME from the regexp R. ;; R may be a symbolic regexp or an ordinary regexp. (defmacro xmltok-g (name &rest r) (let ((sym (make-symbol "r"))) `(let ((,sym (xmltok+ ,@r))) (if (stringp ,sym) (cons (concat "\\(" ,sym "\\)") (cons ',name nil)) (cons (concat "\\(" (car ,sym) "\\)") (cons ',name (cdr ,sym))))))) (defun xmltok-p (&rest r) (xmltok+ "\\(?:" (apply #'xmltok+ r) "\\)")) ;; Get the group index of ELEM in a LIST of symbols. (defun xmltok-get-index (elem list) (or elem (error "Missing group name")) (let ((found nil) (i 1)) (while list (cond ((eq elem (car list)) (setq found i) (setq list nil)) (t (setq i (1+ i)) (setq list (cdr list))))) (or found (error "Bad group name %s" elem)))) ;; Define a macro SYM using a symbolic regexp R. ;; SYM can be called in three ways: ;; (SYM regexp) ;; expands to the regexp in R ;; (SYM start G) ;; expands to ;; (match-beginning N) ;; where N is the group index of G in R. ;; (SYM end G) ;; expands to ;; (match-end N) ;; where N is the group index of G in R. (defmacro xmltok-defregexp (sym r) `(defalias ',sym (let ((r ,r)) `(macro . ,(lambda (action &optional group-name) (cond ((eq action 'regexp) (car r)) ((or (eq action 'start) (eq action 'beginning)) (list 'match-beginning (xmltok-get-index group-name (cdr r)))) ((eq action 'end) (list 'match-end (xmltok-get-index group-name (cdr r)))) ((eq action 'string) (list 'match-string (xmltok-get-index group-name (cdr r)))) ((eq action 'string-no-properties) (list 'match-string-no-properties (xmltok-get-index group-name (cdr r)))) (t (error "Invalid action: %s" action))))))))) (eval-when-compile (let* ((or "\\|") (open "\\(?:") (close "\\)") (name-start-char "[_[:alpha:]]") (name-continue-not-start-char "[-.[:digit:]]") (name-continue-char "[-._[:alnum:]]") (* "*") (+ "+") (opt "?") (question "\\?") (s "[ \r\t\n]") (s+ (concat s +)) (s* (concat s *)) (ncname (concat name-start-char name-continue-char *)) (entity-ref (xmltok+ (xmltok-g entity-name ncname) (xmltok-g entity-ref-close ";") opt)) (decimal-ref (xmltok+ (xmltok-g decimal "[0-9]" +) (xmltok-g decimal-ref-close ";") opt)) (hex-ref (xmltok+ "x" open (xmltok-g hex "[[:xdigit:]]" +) (xmltok-g hex-ref-close ";") opt close opt)) (char-ref (xmltok+ (xmltok-g number-sign "#") open decimal-ref or hex-ref close opt)) (start-tag-close (xmltok+ open (xmltok-g start-tag-close s* ">") or open (xmltok-g empty-tag-slash s* "/") (xmltok-g empty-tag-close ">") opt close or (xmltok-g start-tag-s s+) close)) (start-tag (xmltok+ (xmltok-g start-tag-name ncname (xmltok-g start-tag-colon ":" ncname) opt) start-tag-close opt)) (end-tag (xmltok+ (xmltok-g end-tag-slash "/") open (xmltok-g end-tag-name ncname (xmltok-g end-tag-colon ":" ncname) opt) (xmltok-g end-tag-close s* ">") opt close opt)) (comment (xmltok+ (xmltok-g markup-declaration "!") (xmltok-g comment-first-dash "-" (xmltok-g comment-open "-") opt) opt)) (cdata-section (xmltok+ "!" (xmltok-g marked-section-open "\\[") open "C" open "D" open "A" open "T" open "A" (xmltok-g cdata-section-open "\\[" ) opt close opt ; A close opt ; T close opt ; A close opt ; D close opt)) ; C (processing-instruction (xmltok-g processing-instruction-question question))) (xmltok-defregexp xmltok-ncname (xmltok+ open ncname close)) (xmltok-defregexp xmltok-after-amp (xmltok+ entity-ref or char-ref)) (xmltok-defregexp xmltok-after-lt (xmltok+ start-tag or end-tag ;; cdata-section must come before comment ;; because we treat "))) (xmltok-defregexp xmltok-prolog (let* ((single-char (xmltok-g single-char "[[|,(\"'>]")) (internal-subset-close (xmltok-g internal-subset-close "][ \t\r\n]*>")) (starts-with-close-paren (xmltok-g close-paren ")" (xmltok-p (xmltok-g close-paren-occur "[+?]") or (xmltok-g close-paren-star "\\*")) opt)) (starts-with-percent (xmltok-g percent "%" (xmltok-g param-entity-ref ncname (xmltok-g param-entity-ref-close ";") opt) opt)) (starts-with-nmtoken-not-name (xmltok-g nmtoken (xmltok-p name-continue-not-start-char or ":") (xmltok-p name-continue-char or ":") *)) (nmtoken-after-colon (xmltok+ (xmltok-p name-continue-not-start-char or ":") (xmltok-p name-continue-char or ":") * or name-start-char name-continue-char * ":" (xmltok-p name-continue-char or ":") *)) (after-ncname (xmltok+ (xmltok-g ncname-nmtoken ":" (xmltok-p nmtoken-after-colon)) or (xmltok-p (xmltok-g colon ":" ncname) (xmltok-g colon-name-occur "[?+*]") opt) or (xmltok-g ncname-occur "[?+*]") or (xmltok-g ncname-colon ":"))) (starts-with-name (xmltok-g name ncname (xmltok-p after-ncname) opt)) (starts-with-hash (xmltok-g pound "#" (xmltok-g hash-name ncname))) (markup-declaration (xmltok-g markup-declaration "!" (xmltok-p (xmltok-g comment-first-dash "-" (xmltok-g comment-open "-") opt) or (xmltok-g named-markup-declaration ncname)) opt)) (after-lt (xmltok+ markup-declaration or (xmltok-g processing-instruction-question question) or (xmltok-g instance-start ncname))) (starts-with-lt (xmltok-g less-than "<" (xmltok-p after-lt) opt))) (xmltok+ starts-with-lt or single-char or starts-with-close-paren or starts-with-percent or starts-with-name or starts-with-nmtoken-not-name or starts-with-hash or internal-subset-close))))) (defconst xmltok-ncname-regexp (xmltok-ncname regexp)) (defun xmltok-scan-after-lt () (cond ((not (looking-at (xmltok-after-lt regexp))) (xmltok-add-error "`<' that is not markup must be entered as `<'") (setq xmltok-type 'not-well-formed)) (t (goto-char (match-end 0)) (cond ((xmltok-after-lt start start-tag-close) (setq xmltok-name-end (xmltok-after-lt end start-tag-name)) (setq xmltok-name-colon (xmltok-after-lt start start-tag-colon)) (setq xmltok-attributes nil) (setq xmltok-namespace-attributes nil) (setq xmltok-type 'start-tag)) ((xmltok-after-lt start end-tag-close) (setq xmltok-name-end (xmltok-after-lt end end-tag-name)) (setq xmltok-name-colon (xmltok-after-lt start end-tag-colon)) (setq xmltok-type 'end-tag)) ((xmltok-after-lt start start-tag-s) (setq xmltok-name-end (xmltok-after-lt end start-tag-name)) (setq xmltok-name-colon (xmltok-after-lt start start-tag-colon)) (setq xmltok-namespace-attributes nil) (setq xmltok-attributes nil) (xmltok-scan-attributes) xmltok-type) ((xmltok-after-lt start empty-tag-close) (setq xmltok-name-end (xmltok-after-lt end start-tag-name)) (setq xmltok-name-colon (xmltok-after-lt start start-tag-colon)) (setq xmltok-attributes nil) (setq xmltok-namespace-attributes nil) (setq xmltok-type 'empty-element)) ((xmltok-after-lt start cdata-section-open) (setq xmltok-type (progn (search-forward "]]>" nil 'move) 'cdata-section))) ((xmltok-after-lt start processing-instruction-question) (xmltok-scan-after-processing-instruction-open)) ((xmltok-after-lt start comment-open) (xmltok-scan-after-comment-open)) ((xmltok-after-lt start empty-tag-slash) (setq xmltok-name-end (xmltok-after-lt end start-tag-name)) (setq xmltok-name-colon (xmltok-after-lt start start-tag-colon)) (setq xmltok-attributes nil) (setq xmltok-namespace-attributes nil) (xmltok-add-error "Expected `/>'" (1- (point))) (setq xmltok-type 'partial-empty-element)) ((xmltok-after-lt start start-tag-name) (xmltok-add-error "Missing `>'" nil (1+ xmltok-start)) (setq xmltok-name-end (xmltok-after-lt end start-tag-name)) (setq xmltok-name-colon (xmltok-after-lt start start-tag-colon)) (setq xmltok-namespace-attributes nil) (setq xmltok-attributes nil) (setq xmltok-type 'partial-start-tag)) ((xmltok-after-lt start end-tag-name) (setq xmltok-name-end (xmltok-after-lt end end-tag-name)) (setq xmltok-name-colon (xmltok-after-lt start end-tag-colon)) (cond ((and (not xmltok-name-colon) (eq (char-after) ?:)) (goto-char (1+ (point))) (xmltok-add-error "Expected name following `:'" (1- (point)))) (t (xmltok-add-error "Missing `>'" nil (1+ xmltok-start)))) (setq xmltok-type 'partial-end-tag)) ((xmltok-after-lt start end-tag-slash) (xmltok-add-error "Expected name following `) in unclosed PI (defun xmltok-scan-after-processing-instruction-open () (search-forward "?>" nil 'move) (cond ((not (save-excursion (goto-char (+ 2 xmltok-start)) (and (looking-at (xmltok-ncname regexp)) (setq xmltok-name-end (match-end 0))))) (setq xmltok-name-end (+ xmltok-start 2)) (xmltok-add-error "\\)?" nil 'move) (not (match-end 1))) (xmltok-add-error "`--' not followed by `>'" (match-beginning 0))) (setq xmltok-type 'comment)) (defun xmltok-scan-attributes () (let ((recovering nil) (atts-needing-normalization nil)) (while (cond ((or (looking-at (xmltok-attribute regexp)) ;; use non-greedy group (when (looking-at (concat "[^<>\n]+?" (xmltok-attribute regexp))) (unless recovering (xmltok-add-error "Malformed attribute" (point) (save-excursion (goto-char (xmltok-attribute start name)) (skip-chars-backward "\r\n\t ") (point)))) t)) (setq recovering nil) (goto-char (match-end 0)) (let ((att (xmltok-add-attribute))) (when att (setq atts-needing-normalization (cons att atts-needing-normalization)))) (cond ((xmltok-attribute start start-tag-s) t) ((xmltok-attribute start start-tag-close) (setq xmltok-type 'start-tag) nil) ((xmltok-attribute start empty-tag-close) (setq xmltok-type 'empty-element) nil) ((xmltok-attribute start empty-tag-slash) (setq xmltok-type 'partial-empty-element) (xmltok-add-error "Expected `/>'" (1- (point))) nil) ((looking-at "[ \t\r\n]*[\"']") (goto-char (match-end 0)) (xmltok-add-error "Missing closing delimiter" (1- (point))) (setq recovering t) t) ((looking-at "[ \t]*\\([^ \t\r\n\"'=<>/]+\\)[ \t\r\n/>]") (goto-char (match-end 1)) (xmltok-add-error "Attribute value not quoted" (match-beginning 1)) (setq recovering t) t) (t (xmltok-add-error "Missing attribute value" (1- (point))) (setq recovering t) t))) ((looking-at "[^<>\n]*/>") (let ((start (point))) (goto-char (match-end 0)) (unless recovering (xmltok-add-error "Malformed empty-element" start (- (point) 2)))) (setq xmltok-type 'empty-element) nil) ((looking-at "[^<>\n]*>") (let ((start (point))) (goto-char (match-end 0)) (unless recovering (xmltok-add-error "Malformed start-tag" start (1- (point))))) (setq xmltok-type 'start-tag) nil) (t (when recovering (skip-chars-forward "^<>\n")) (xmltok-add-error "Missing `>'" xmltok-start (1+ xmltok-start)) (setq xmltok-type 'partial-start-tag) nil))) (while atts-needing-normalization (xmltok-normalize-attribute (car atts-needing-normalization)) (setq atts-needing-normalization (cdr atts-needing-normalization)))) (setq xmltok-attributes (nreverse xmltok-attributes)) (setq xmltok-namespace-attributes (nreverse xmltok-namespace-attributes))) (defun xmltok-add-attribute () "Return the attribute if it needs normalizing, otherwise nil." (let* ((needs-normalizing nil) (att (if (xmltok-attribute start literal) (progn (setq needs-normalizing (or (xmltok-attribute start complex1) (xmltok-attribute start complex2))) (xmltok-make-attribute (xmltok-attribute start name) (xmltok-attribute start colon) (xmltok-attribute end name) (1+ (xmltok-attribute start literal)) (1- (xmltok-attribute end literal)) (not needs-normalizing))) (xmltok-make-attribute (xmltok-attribute start name) (xmltok-attribute start colon) (xmltok-attribute end name))))) (if (xmltok-attribute start xmlns) (setq xmltok-namespace-attributes (cons att xmltok-namespace-attributes)) (setq xmltok-attributes (cons att xmltok-attributes))) (and needs-normalizing att))) (defun xmltok-normalize-attribute (att) (let ((end (xmltok-attribute-value-end att)) (well-formed t) (value-parts nil) (refs nil)) (save-excursion (goto-char (xmltok-attribute-value-start att)) (while (progn (let ((n (skip-chars-forward "^\r\t\n&" end))) (when (> n 0) (setq value-parts (cons (buffer-substring-no-properties (- (point) n) (point)) value-parts)))) (when (< (point) end) (goto-char (1+ (point))) (cond ((eq (char-before) ?\&) (let ((xmltok-start (1- (point))) xmltok-type xmltok-replacement) (xmltok-scan-after-amp (lambda (start end) (xmltok-handle-entity start end t))) (cond ((or (eq xmltok-type 'char-ref) (eq xmltok-type 'entity-ref)) (setq refs (cons (vector xmltok-type xmltok-start (point)) refs)) (if xmltok-replacement (setq value-parts (cons xmltok-replacement value-parts)) (setq well-formed nil))) (t (setq well-formed nil))))) (t (setq value-parts (cons " " value-parts))))) (< (point) end)))) (when well-formed (aset att 5 (apply #'concat (nreverse value-parts)))) (aset att 6 (nreverse refs)))) (defun xmltok-scan-after-amp (entity-handler) (cond ((not (looking-at (xmltok-after-amp regexp))) (xmltok-add-error "`&' that is not markup must be entered as `&'") (setq xmltok-type 'not-well-formed)) (t (goto-char (match-end 0)) (cond ((xmltok-after-amp start entity-ref-close) (funcall entity-handler (xmltok-after-amp start entity-name) (xmltok-after-amp end entity-name)) (setq xmltok-type 'entity-ref)) ((xmltok-after-amp start decimal-ref-close) (xmltok-scan-char-ref (xmltok-after-amp start decimal) (xmltok-after-amp end decimal) 10)) ((xmltok-after-amp start hex-ref-close) (xmltok-scan-char-ref (xmltok-after-amp start hex) (xmltok-after-amp end hex) 16)) ((xmltok-after-amp start number-sign) (xmltok-add-error "Missing character number") (setq xmltok-type 'not-well-formed)) (t (xmltok-add-error "Missing closing `;'") (setq xmltok-type 'not-well-formed)))))) (defconst xmltok-entity-error-messages '((unparsed . "Referenced entity is unparsed") (not-well-formed . "Referenced entity is not well-formed") (external nil . "Referenced entity is external") (element nil . "Referenced entity contains <"))) (defun xmltok-handle-entity (start end &optional attributep) (let* ((name (buffer-substring-no-properties start end)) (name-def (assoc name xmltok-dtd)) (def (cdr name-def))) (cond ((setq xmltok-replacement (and (consp def) (if attributep (cdr def) (car def))))) ((null name-def) (unless (eq (car xmltok-dtd) t) (xmltok-add-error "Referenced entity has not been defined" start end))) ((and attributep (consp def)) (xmltok-add-error "Referenced entity contains <" start end)) (t (let ((err (cdr (assq def xmltok-entity-error-messages)))) (when (consp err) (setq err (if attributep (cdr err) (car err)))) (when err (xmltok-add-error err start end))))))) (defun xmltok-scan-char-ref (start end base) (setq xmltok-replacement (let ((n (string-to-number (buffer-substring-no-properties start end) base))) (cond ((and (integerp n) (xmltok-valid-char-p n)) (setq n (xmltok-unicode-to-char n)) (and n (string n))) (t (xmltok-add-error "Invalid character code" start end) nil)))) (setq xmltok-type 'char-ref)) (defun xmltok-char-number (start end) (let* ((base (if (eq (char-after (+ start 2)) ?x) 16 10)) (n (string-to-number (buffer-substring-no-properties (+ start (if (= base 16) 3 2)) (1- end)) base))) (and (integerp n) (xmltok-valid-char-p n) n))) (defun xmltok-valid-char-p (n) "Return non-nil if N is the Unicode code of a valid XML character." (cond ((< n #x20) (memq n '(#xA #xD #x9))) ((< n #xD800) t) ((< n #xE000) nil) ((< n #xFFFE) t) (t (and (> n #xFFFF) (< n #x110000))))) (defun xmltok-unicode-to-char (n) "Return the character corresponding to Unicode scalar value N. Return nil if unsupported in Emacs." (decode-char 'ucs n)) ;;; Prolog parsing (defvar xmltok-contains-doctype nil) (defvar xmltok-doctype-external-subset-flag nil) (defvar xmltok-internal-subset-start nil) (defvar xmltok-had-param-entity-ref nil) (defvar xmltok-prolog-regions nil) (defvar xmltok-standalone nil "Non-nil if there was an XML declaration specifying standalone=\"yes\".") (defvar xmltok-markup-declaration-doctype-flag nil) (defconst xmltok-predefined-entity-alist '(("lt" "<" . "<") ("gt" ">" . ">") ("amp" "&" . "&") ("apos" "'" . "'") ("quot" "\"" . "\""))) (defun xmltok-forward-prolog () "Move forward to the end of the XML prolog. Returns a list of vectors [TYPE START END] where TYPE is a symbol and START and END are integers giving the start and end of the region of that type. TYPE can be one of xml-declaration, xml-declaration-attribute-name, xml-declaration-attribute-value, comment, processing-instruction-left, processing-instruction-right, markup-declaration-open, markup-declaration-close, internal-subset-open, internal-subset-close, hash-name, keyword, literal, encoding-name. Adds to `xmltok-errors' as appropriate." (let ((case-fold-search nil) xmltok-start xmltok-type xmltok-prolog-regions xmltok-contains-doctype xmltok-internal-subset-start xmltok-had-param-entity-ref xmltok-standalone xmltok-doctype-external-subset-flag xmltok-markup-declaration-doctype-flag) (setq xmltok-dtd xmltok-predefined-entity-alist) (xmltok-scan-xml-declaration) (xmltok-next-prolog-token) (while (condition-case nil (when (xmltok-parse-prolog-item) (xmltok-next-prolog-token)) (xmltok-markup-declaration-parse-error (xmltok-skip-markup-declaration)))) (when xmltok-internal-subset-start (xmltok-add-error "No closing ]" (1- xmltok-internal-subset-start) xmltok-internal-subset-start)) (xmltok-parse-entities) (nreverse xmltok-prolog-regions))) (defconst xmltok-bad-xml-decl-regexp "[ \t\r\n]*<\\?xml\\(?:[ \t\r\n]\\|\\?>\\)") ;;;###autoload (defun xmltok-get-declared-encoding-position (&optional limit) "Return the position of the encoding in the XML declaration at point. If there is a well-formed XML declaration starting at point and it contains an encoding declaration, then return (START . END) where START and END are the positions of the start and the end of the encoding name; if there is no encoding declaration return the position where and encoding declaration could be inserted. If there is XML that is not well-formed that looks like an XML declaration, return nil. Otherwise, return t. If LIMIT is non-nil, then do not consider characters beyond LIMIT." (cond ((let ((case-fold-search nil)) (and (looking-at (xmltok-xml-declaration regexp)) (or (not limit) (<= (match-end 0) limit)))) (let ((end (xmltok-xml-declaration end encoding-value))) (if end (cons (1+ (xmltok-xml-declaration start encoding-value)) (1- end)) (or (xmltok-xml-declaration end version-value) (+ (point) 5))))) ((not (let ((case-fold-search t)) (looking-at xmltok-bad-xml-decl-regexp)))))) (defun xmltok-scan-xml-declaration () (when (looking-at (xmltok-xml-declaration regexp)) (xmltok-add-prolog-region 'xml-declaration (point) (match-end 0)) (goto-char (match-end 0)) (when (xmltok-xml-declaration start version-name) (xmltok-add-prolog-region 'xml-declaration-attribute-name (xmltok-xml-declaration start version-name) (xmltok-xml-declaration end version-name)) (let ((start (xmltok-xml-declaration start version-value)) (end (xmltok-xml-declaration end version-value))) (xmltok-add-prolog-region 'xml-declaration-attribute-value start end))) ;; XXX need to check encoding name ;; Should start with letter, not contain colon (when (xmltok-xml-declaration start encoding-name) (xmltok-add-prolog-region 'xml-declaration-attribute-name (xmltok-xml-declaration start encoding-name) (xmltok-xml-declaration end encoding-name)) (let ((start (xmltok-xml-declaration start encoding-value)) (end (xmltok-xml-declaration end encoding-value))) (xmltok-add-prolog-region 'encoding-name (1+ start) (1- end)) (xmltok-add-prolog-region 'xml-declaration-attribute-value start end))) (when (xmltok-xml-declaration start standalone-name) (xmltok-add-prolog-region 'xml-declaration-attribute-name (xmltok-xml-declaration start standalone-name) (xmltok-xml-declaration end standalone-name)) (let ((start (xmltok-xml-declaration start standalone-value)) (end (xmltok-xml-declaration end standalone-value))) (xmltok-add-prolog-region 'xml-declaration-attribute-value start end) (setq xmltok-standalone (string= (buffer-substring-no-properties (1+ start) (1- end)) "yes")))) t)) (defconst xmltok-markup-declaration-alist '(("ELEMENT" . xmltok-parse-element-declaration) ("ATTLIST" . xmltok-parse-attlist-declaration) ("ENTITY" . xmltok-parse-entity-declaration) ("NOTATION" . xmltok-parse-notation-declaration))) (defun xmltok-parse-prolog-item () (cond ((eq xmltok-type 'comment) (xmltok-add-prolog-region 'comment xmltok-start (point)) t) ((eq xmltok-type 'processing-instruction)) ((eq xmltok-type 'named-markup-declaration) (setq xmltok-markup-declaration-doctype-flag nil) (xmltok-add-prolog-region 'markup-declaration-open xmltok-start (point)) (let* ((name (buffer-substring-no-properties (+ xmltok-start 2) (point))) (fun (cdr (assoc name xmltok-markup-declaration-alist)))) (cond (fun (unless xmltok-internal-subset-start (xmltok-add-error "Declaration allowed only in internal subset")) (funcall fun)) ((string= name "DOCTYPE") (xmltok-parse-doctype)) (t (xmltok-add-error "Unknown markup declaration" (+ xmltok-start 2)) (xmltok-next-prolog-token) (xmltok-markup-declaration-parse-error)))) t) ((or (eq xmltok-type 'end-prolog) (not xmltok-type)) nil) ((eq xmltok-type 'internal-subset-close) (xmltok-add-prolog-region 'internal-subset-close xmltok-start (1+ xmltok-start)) (xmltok-add-prolog-region 'markup-declaration-close (1- (point)) (point)) (if xmltok-internal-subset-start (setq xmltok-internal-subset-start nil) (xmltok-add-error "]> outside internal subset")) t) ((eq xmltok-type 'param-entity-ref) (if xmltok-internal-subset-start (setq xmltok-had-param-entity-ref t) (xmltok-add-error "Parameter entity reference outside document type declaration")) t) ;; If we don't do this, we can get thousands of errors when ;; a plain text file is parsed. ((not xmltok-internal-subset-start) (when (let ((err (car xmltok-errors))) (or (not err) (<= (xmltok-error-end err) xmltok-start))) (goto-char xmltok-start)) nil) ((eq xmltok-type 'not-well-formed) t) (t (xmltok-add-error "Token allowed only inside markup declaration") t))) (defun xmltok-parse-doctype () (setq xmltok-markup-declaration-doctype-flag t) (xmltok-next-prolog-token) (when xmltok-internal-subset-start (xmltok-add-error "DOCTYPE declaration not allowed in internal subset") (xmltok-markup-declaration-parse-error)) (when xmltok-contains-doctype (xmltok-add-error "Duplicate DOCTYPE declaration") (xmltok-markup-declaration-parse-error)) (setq xmltok-contains-doctype t) (xmltok-require-token 'name 'prefixed-name) (xmltok-require-next-token "SYSTEM" "PUBLIC" ?\[ ?>) (cond ((eq xmltok-type ?\[) (setq xmltok-internal-subset-start (point))) ((eq xmltok-type ?>)) (t (setq xmltok-doctype-external-subset-flag t) (xmltok-parse-external-id) (xmltok-require-token ?\[ ?>) (when (eq xmltok-type ?\[) (setq xmltok-internal-subset-start (point)))))) (defun xmltok-parse-attlist-declaration () (xmltok-require-next-token 'prefixed-name 'name) (while (progn (xmltok-require-next-token ?> 'name 'prefixed-name) (if (eq xmltok-type ?>) nil (xmltok-require-next-token ?\( "CDATA" "ID" "IDREF" "IDREFS" "ENTITY" "ENTITIES" "NMTOKEN" "NMTOKENS" "NOTATION") (cond ((eq xmltok-type ?\() (xmltok-parse-nmtoken-group)) ((string= (xmltok-current-token-string) "NOTATION") (xmltok-require-next-token ?\() (xmltok-parse-nmtoken-group))) (xmltok-require-next-token "#IMPLIED" "#REQUIRED" "#FIXED" 'literal) (when (string= (xmltok-current-token-string) "#FIXED") (xmltok-require-next-token 'literal)) t)))) (defun xmltok-parse-nmtoken-group () (while (progn (xmltok-require-next-token 'nmtoken 'prefixed-name 'name) (xmltok-require-next-token ?| ?\)) (eq xmltok-type ?|)))) (defun xmltok-parse-element-declaration () (xmltok-require-next-token 'name 'prefixed-name) (xmltok-require-next-token "EMPTY" "ANY" ?\() (when (eq xmltok-type ?\() (xmltok-require-next-token "#PCDATA" 'name 'prefixed-name 'name-occur ?\() (cond ((eq xmltok-type 'hash-name) (xmltok-require-next-token ?| ?\) 'close-paren-star) (while (eq xmltok-type ?|) (xmltok-require-next-token 'name 'prefixed-name) (xmltok-require-next-token 'close-paren-star ?|))) (t (xmltok-parse-model-group)))) (xmltok-require-next-token ?>)) (defun xmltok-parse-model-group () (xmltok-parse-model-group-member) (xmltok-require-next-token ?| ?, ?\) 'close-paren-star 'close-paren-occur) (when (memq xmltok-type '(?, ?|)) (let ((connector xmltok-type)) (while (progn (xmltok-next-prolog-token) (xmltok-parse-model-group-member) (xmltok-require-next-token connector ?\) 'close-paren-star 'close-paren-occur) (eq xmltok-type connector)))))) (defun xmltok-parse-model-group-member () (xmltok-require-token 'name 'prefixed-name 'name-occur ?\() (when (eq xmltok-type ?\() (xmltok-next-prolog-token) (xmltok-parse-model-group))) (defun xmltok-parse-entity-declaration () (let (paramp name) (xmltok-require-next-token 'name ?%) (when (eq xmltok-type ?%) (setq paramp t) (xmltok-require-next-token 'name)) (setq name (xmltok-current-token-string)) (xmltok-require-next-token 'literal "SYSTEM" "PUBLIC") (cond ((eq xmltok-type 'literal) (let ((replacement (xmltok-parse-entity-value))) (unless paramp (xmltok-define-entity name replacement))) (xmltok-require-next-token ?>)) (t (xmltok-parse-external-id) (if paramp (xmltok-require-token ?>) (xmltok-require-token ?> "NDATA") (if (eq xmltok-type ?>) (xmltok-define-entity name 'external) (xmltok-require-next-token 'name) (xmltok-require-next-token ?>) (xmltok-define-entity name 'unparsed))))))) (defun xmltok-define-entity (name value) (when (and (or (not xmltok-had-param-entity-ref) xmltok-standalone) (not (assoc name xmltok-dtd))) (setq xmltok-dtd (cons (cons name value) xmltok-dtd)))) (defun xmltok-parse-entity-value () (let ((lim (1- (point))) (well-formed t) value-parts start) (save-excursion (goto-char (1+ xmltok-start)) (setq start (point)) (while (progn (skip-chars-forward "^%&" lim) (when (< (point) lim) (goto-char (1+ (point))) (cond ((eq (char-before) ?%) (xmltok-add-error "Parameter entity references are not allowed in the internal subset" (1- (point)) (point)) (setq well-formed nil)) (t (let ((xmltok-start (1- (point))) xmltok-type xmltok-replacement) (xmltok-scan-after-amp (lambda (_start _end))) (cond ((eq xmltok-type 'char-ref) (setq value-parts (cons (buffer-substring-no-properties start xmltok-start) value-parts)) (setq value-parts (cons xmltok-replacement value-parts)) (setq start (point))) ((eq xmltok-type 'not-well-formed) (setq well-formed nil)))))) t)))) (if (not well-formed) nil (apply #'concat (nreverse (cons (buffer-substring-no-properties start lim) value-parts)))))) (defun xmltok-parse-notation-declaration () (xmltok-require-next-token 'name) (xmltok-require-next-token "SYSTEM" "PUBLIC") (let ((publicp (string= (xmltok-current-token-string) "PUBLIC"))) (xmltok-require-next-token 'literal) (cond (publicp (xmltok-require-next-token 'literal ?>) (unless (eq xmltok-type ?>) (xmltok-require-next-token ?>))) (t (xmltok-require-next-token ?>))))) (defun xmltok-parse-external-id () (xmltok-require-token "SYSTEM" "PUBLIC") (let ((publicp (string= (xmltok-current-token-string) "PUBLIC"))) (xmltok-require-next-token 'literal) (when publicp (xmltok-require-next-token 'literal))) (xmltok-next-prolog-token)) (defun xmltok-require-next-token (&rest types) (xmltok-next-prolog-token) (apply #'xmltok-require-token types)) (defun xmltok-require-token (&rest types) ;; XXX Generate a more helpful error message (while (and (not (let ((type (car types))) (if (stringp (car types)) (string= (xmltok-current-token-string) type) (eq type xmltok-type)))) (setq types (cdr types)))) (unless types (when (and xmltok-type (not (eq xmltok-type 'not-well-formed))) (xmltok-add-error "Unexpected token")) (xmltok-markup-declaration-parse-error)) (let ((region-type (xmltok-prolog-region-type (car types)))) (when region-type (xmltok-add-prolog-region region-type xmltok-start (point))))) (defun xmltok-current-token-string () (buffer-substring-no-properties xmltok-start (point))) (define-error 'xmltok-markup-declaration-parse-error "Syntax error in markup declaration") (defun xmltok-markup-declaration-parse-error () (signal 'xmltok-markup-declaration-parse-error nil)) (defun xmltok-skip-markup-declaration () (while (cond ((eq xmltok-type ?>) (xmltok-next-prolog-token) nil) ((and xmltok-markup-declaration-doctype-flag (eq xmltok-type ?\[)) (setq xmltok-internal-subset-start (point)) (xmltok-next-prolog-token) nil) ((memq xmltok-type '(nil end-prolog named-markup-declaration comment processing-instruction)) nil) ((and xmltok-internal-subset-start (eq xmltok-type 'internal-subset-close)) nil) (t (xmltok-next-prolog-token) t))) xmltok-type) (defun xmltok-prolog-region-type (required) (cond ((cdr (assq xmltok-type '((literal . literal) (?> . markup-declaration-close) (?\[ . internal-subset-open) (hash-name . hash-name))))) ((and (stringp required) (eq xmltok-type 'name)) 'keyword))) ;; Return new token type. (defun xmltok-next-prolog-token () (skip-chars-forward " \t\r\n") (setq xmltok-start (point)) (cond ((not (and (looking-at (xmltok-prolog regexp)) (goto-char (match-end 0)))) (let ((ch (char-after))) (cond (ch (goto-char (1+ (point))) (xmltok-add-error "Illegal char in prolog") (setq xmltok-type 'not-well-formed)) (t (setq xmltok-type nil))))) ((or (xmltok-prolog start ncname-occur) (xmltok-prolog start colon-name-occur)) (setq xmltok-name-end (1- (point))) (setq xmltok-name-colon (xmltok-prolog start colon)) (setq xmltok-type 'name-occur)) ((xmltok-prolog start colon) (setq xmltok-name-end (point)) (setq xmltok-name-colon (xmltok-prolog start colon)) (unless (looking-at "[ \t\r\n>),|[%]") (xmltok-add-error "Missing space after name")) (setq xmltok-type 'prefixed-name)) ((or (xmltok-prolog start ncname-nmtoken) (xmltok-prolog start ncname-colon)) (unless (looking-at "[ \t\r\n>),|[%]") (xmltok-add-error "Missing space after name token")) (setq xmltok-type 'nmtoken)) ((xmltok-prolog start name) (setq xmltok-name-end (point)) (setq xmltok-name-colon nil) (unless (looking-at "[ \t\r\n>),|[%]") (xmltok-add-error "Missing space after name")) (setq xmltok-type 'name)) ((xmltok-prolog start hash-name) (setq xmltok-name-end (point)) (unless (looking-at "[ \t\r\n>)|%]") (xmltok-add-error "Missing space after name")) (setq xmltok-type 'hash-name)) ((xmltok-prolog start processing-instruction-question) (xmltok-scan-prolog-after-processing-instruction-open)) ((xmltok-prolog start comment-open) ;; XXX if not-well-formed, ignore some stuff (xmltok-scan-after-comment-open)) ((xmltok-prolog start named-markup-declaration) (setq xmltok-type 'named-markup-declaration)) ((xmltok-prolog start instance-start) (goto-char xmltok-start) (setq xmltok-type 'end-prolog)) ((xmltok-prolog start close-paren-star) (setq xmltok-type 'close-paren-star)) ((xmltok-prolog start close-paren-occur) (setq xmltok-type 'close-paren-occur)) ((xmltok-prolog start close-paren) (unless (looking-at "[ \t\r\n>,|)]") (xmltok-add-error "Missing space after )")) (setq xmltok-type ?\))) ((xmltok-prolog start single-char) (let ((ch (char-before))) (cond ((memq ch '(?\" ?\')) (xmltok-scan-prolog-literal)) (t (setq xmltok-type ch))))) ((xmltok-prolog start percent) (cond ((xmltok-prolog start param-entity-ref-close) (setq xmltok-name-end (1- (point))) (setq xmltok-type 'param-entity-ref)) ((xmltok-prolog start param-entity-ref) (xmltok-add-error "Missing ;") (setq xmltok-name-end (point)) (setq xmltok-type 'param-entity-ref)) ((looking-at "[ \t\r\n%]") (setq xmltok-type ?%)) (t (xmltok-add-error "Expected name after %") (setq xmltok-type 'not-well-formed)))) ((xmltok-prolog start nmtoken) (unless (looking-at "[ \t\r\n>),|[%]") (xmltok-add-error "Missing space after name token")) (setq xmltok-type 'nmtoken)) ((xmltok-prolog start internal-subset-close) (setq xmltok-type 'internal-subset-close)) ((xmltok-prolog start pound) (xmltok-add-error "Expected name after #") (setq xmltok-type 'not-well-formed)) ((xmltok-prolog start markup-declaration) (xmltok-add-error "Expected name or -- after []" delim)) (point))) (end (save-excursion (goto-char safe-end) (search-forward delim nil t)))) (cond ((or (not end) (save-excursion (goto-char end) (looking-at "[ \t\r\n>%[]"))) (goto-char end)) ((eq (1+ safe-end) end) (goto-char end) (xmltok-add-error (format "Missing space after %s" delim) safe-end))) (setq xmltok-type 'literal))) (defun xmltok-scan-prolog-after-processing-instruction-open () (search-forward "?>" nil 'move) (let* ((end (point)) (target (save-excursion (goto-char (+ xmltok-start 2)) (and (looking-at (xmltok-ncname regexp)) (or (memq (char-after (match-end 0)) '(?\n ?\t ?\r ? )) (= (match-end 0) (- end 2))) (match-string-no-properties 0))))) (cond ((not target) (xmltok-add-error "\ Processing instruction does not start with a name" (+ xmltok-start 2) (+ xmltok-start 3))) ((not (and (= (length target) 3) (let ((case-fold-search t)) (string-match "xml" target))))) ((= xmltok-start 1) (xmltok-add-error "Invalid XML declaration" xmltok-start (point))) ((save-excursion (goto-char xmltok-start) (looking-at (xmltok-xml-declaration regexp))) (xmltok-add-error "XML declaration not at beginning of file" xmltok-start (point))) (t (xmltok-add-error "Processing instruction has target of xml" (+ xmltok-start 2) (+ xmltok-start 5)))) (xmltok-add-prolog-region 'processing-instruction-left xmltok-start (+ xmltok-start 2 (if target (length target) 0))) (xmltok-add-prolog-region 'processing-instruction-right (if target (save-excursion (goto-char (+ xmltok-start (length target) 2)) (skip-chars-forward " \t\r\n") (point)) (+ xmltok-start 2)) (point))) (setq xmltok-type 'processing-instruction)) (defun xmltok-parse-entities () (let ((todo xmltok-dtd)) (when (and (or xmltok-had-param-entity-ref xmltok-doctype-external-subset-flag) (not xmltok-standalone)) (setq xmltok-dtd (cons t xmltok-dtd))) (while todo (xmltok-parse-entity (car todo)) (setq todo (cdr todo))))) (defun xmltok-parse-entity (name-def) (let ((def (cdr name-def)) ;; in case its value is buffer local (xmltok-dtd xmltok-dtd) buf) (when (stringp def) (if (string-match "\\`[^&<\t\r\n]*\\'" def) (setcdr name-def (cons def def)) (setcdr name-def 'not-well-formed) ; avoid infinite expansion loops (setq buf (get-buffer-create (format " *Entity %s*" (car name-def)))) (with-current-buffer buf (erase-buffer) (insert def) (goto-char (point-min)) (setcdr name-def (xmltok-parse-entity-replacement))) (kill-buffer buf))))) (defun xmltok-parse-entity-replacement () (let ((def (cons "" ""))) (while (let* ((start (point)) (found (re-search-forward "[<&\t\r\n]\\|]]>" nil t)) (ch (and found (char-before))) (str (buffer-substring-no-properties start (if found (match-beginning 0) (point-max))))) (setq def (xmltok-append-entity-def def (cons str str))) (cond ((not found) nil) ((eq ch ?>) (setq def 'not-well-formed) nil) ((eq ch ?<) (xmltok-save (setq xmltok-start (1- (point))) (xmltok-scan-after-lt) (setq def (xmltok-append-entity-def def (cond ((memq xmltok-type '(start-tag end-tag empty-element)) 'element) ((memq xmltok-type '(comment processing-instruction)) (cons "" nil)) ((eq xmltok-type 'cdata-section) (cons (buffer-substring-no-properties (+ xmltok-start 9) (- (point) 3)) nil)) (t 'not-well-formed))))) t) ((eq ch ?&) (let ((xmltok-start (1- (point))) xmltok-type xmltok-replacement xmltok-errors) (xmltok-scan-after-amp 'xmltok-handle-nested-entity) (cond ((eq xmltok-type 'entity-ref) (setq def (xmltok-append-entity-def def xmltok-replacement))) ((eq xmltok-type 'char-ref) (setq def (xmltok-append-entity-def def (if xmltok-replacement (cons xmltok-replacement xmltok-replacement) (and xmltok-errors 'not-well-formed))))) (t (setq def 'not-well-formed)))) t) (t (setq def (xmltok-append-entity-def def (cons (match-string-no-properties 0) " "))) t)))) def)) (defun xmltok-handle-nested-entity (start end) (let* ((name-def (assoc (buffer-substring-no-properties start end) xmltok-dtd)) (def (cdr name-def))) (when (stringp def) (xmltok-parse-entity name-def) (setq def (cdr name-def))) (setq xmltok-replacement (cond ((null name-def) (if (eq (car xmltok-dtd) t) nil 'not-well-formed)) ((eq def 'unparsed) 'not-well-formed) (t def))))) (defun xmltok-append-entity-def (d1 d2) (cond ((consp d1) (if (consp d2) (cons (concat (car d1) (car d2)) (and (cdr d1) (cdr d2) (concat (cdr d1) (cdr d2)))) d2)) ((consp d2) d1) (t (let ((defs '(not-well-formed external element))) (while (not (or (eq (car defs) d1) (eq (car defs) d2))) (setq defs (cdr defs))) (car defs))))) (defun xmltok-add-prolog-region (type start end) (setq xmltok-prolog-regions (cons (vector type start end) xmltok-prolog-regions))) (defun xmltok-merge-attributes () "Return a list merging `xmltok-attributes' and `xmltok-namespace-attributes'. The members of the merged list are in order of occurrence in the document. The list may share list structure with `xmltok-attributes' and `xmltok-namespace-attributes'." (cond ((not xmltok-namespace-attributes) xmltok-attributes) ((not xmltok-attributes) xmltok-namespace-attributes) (t (let ((atts1 xmltok-attributes) (atts2 xmltok-namespace-attributes) merged) (while (and atts1 atts2) (cond ((< (xmltok-attribute-name-start (car atts1)) (xmltok-attribute-name-start (car atts2))) (setq merged (cons (car atts1) merged)) (setq atts1 (cdr atts1))) (t (setq merged (cons (car atts2) merged)) (setq atts2 (cdr atts2))))) (setq merged (nreverse merged)) (cond (atts1 (setq merged (nconc merged atts1))) (atts2 (setq merged (nconc merged atts2)))) merged)))) ;;; Testing (defun xmltok-forward-test () (interactive) (if (xmltok-forward) (message "Scanned %s" xmltok-type) (message "Scanned nothing"))) (defun xmltok-next-prolog-token-test () (interactive) (if (xmltok-next-prolog-token) (message "Scanned %s" (if (integerp xmltok-type) (string xmltok-type) xmltok-type)) (message "Scanned end of file"))) (provide 'xmltok) ;;; xmltok.el ends here