2007-09-20 18:09:08 -04:00
|
|
|
! Copyright (C) 2005, 2006 Daniel Ehrenberg
|
|
|
|
! See http://factorcode.org/license.txt for BSD license.
|
2009-01-19 23:25:15 -05:00
|
|
|
USING: accessors arrays ascii assocs combinators locals
|
2009-01-15 16:25:00 -05:00
|
|
|
combinators.short-circuit fry io.encodings io.encodings.iana
|
|
|
|
io.encodings.string io.encodings.utf16 io.encodings.utf8 kernel make
|
2009-01-21 00:54:33 -05:00
|
|
|
math math.parser namespaces sequences sets splitting xml.state
|
2009-01-19 23:25:15 -05:00
|
|
|
strings xml.char-classes xml.data xml.entities xml.errors hashtables
|
2009-01-21 00:54:33 -05:00
|
|
|
circular io sbufs ;
|
2007-09-20 18:09:08 -04:00
|
|
|
IN: xml.tokenize
|
|
|
|
|
2009-01-21 00:54:33 -05:00
|
|
|
! Originally from state-parser
|
|
|
|
|
|
|
|
SYMBOL: prolog-data
|
|
|
|
|
|
|
|
: version=1.0? ( -- ? )
|
|
|
|
prolog-data get [ version>> "1.0" = ] [ t ] if* ;
|
|
|
|
|
|
|
|
: assure-good-char ( ch -- ch )
|
|
|
|
[
|
|
|
|
version=1.0? over text? not get-check and
|
|
|
|
[ disallowed-char ] when
|
|
|
|
] [ f ] if* ;
|
|
|
|
|
|
|
|
! * Basic utility words
|
|
|
|
|
|
|
|
: record ( char -- )
|
|
|
|
CHAR: \n =
|
|
|
|
[ 0 get-line 1+ set-line ] [ get-column 1+ ] if
|
|
|
|
set-column ;
|
|
|
|
|
|
|
|
! (next) normalizes \r\n and \r
|
|
|
|
: (next) ( -- char )
|
|
|
|
get-next read1
|
|
|
|
2dup swap CHAR: \r = [
|
|
|
|
CHAR: \n =
|
|
|
|
[ nip read1 ] [ nip CHAR: \n swap ] if
|
|
|
|
] [ drop ] if
|
|
|
|
set-next dup set-char assure-good-char ;
|
|
|
|
|
|
|
|
: next ( -- )
|
|
|
|
#! Increment spot.
|
|
|
|
get-char [ unexpected-end ] unless (next) record ;
|
|
|
|
|
|
|
|
: skip-until ( quot: ( -- ? ) -- )
|
|
|
|
get-char [
|
|
|
|
[ call ] keep swap [ drop ] [
|
|
|
|
next skip-until
|
|
|
|
] if
|
|
|
|
] [ drop ] if ; inline recursive
|
|
|
|
|
|
|
|
: take-until ( quot -- string )
|
|
|
|
#! Take the substring of a string starting at spot
|
|
|
|
#! from code until the quotation given is true and
|
|
|
|
#! advance spot to after the substring.
|
|
|
|
10 <sbuf> [
|
|
|
|
'[ @ [ t ] [ get-char _ push f ] if ] skip-until
|
|
|
|
] keep >string ; inline
|
|
|
|
|
|
|
|
: take-char ( ch -- string )
|
|
|
|
[ dup get-char = ] take-until nip ;
|
|
|
|
|
|
|
|
: pass-blank ( -- )
|
|
|
|
#! Advance code past any whitespace, including newlines
|
|
|
|
[ get-char blank? not ] skip-until ;
|
|
|
|
|
|
|
|
: string-matches? ( string circular -- ? )
|
|
|
|
get-char over push-circular
|
|
|
|
sequence= ;
|
|
|
|
|
|
|
|
: take-string ( match -- string )
|
|
|
|
dup length <circular-string>
|
|
|
|
[ 2dup string-matches? ] take-until nip
|
|
|
|
dup length rot length 1- - head
|
|
|
|
get-char [ missing-close ] unless next ;
|
|
|
|
|
|
|
|
: expect ( ch -- )
|
|
|
|
get-char 2dup = [ 2drop ] [
|
|
|
|
[ 1string ] bi@ expected
|
|
|
|
] if next ;
|
|
|
|
|
|
|
|
: expect-string ( string -- )
|
|
|
|
dup [ get-char next ] replicate 2dup =
|
|
|
|
[ 2drop ] [ expected ] if ;
|
|
|
|
|
|
|
|
: init-parser ( -- )
|
|
|
|
0 1 0 f f <spot> spot set
|
|
|
|
read1 set-next next ;
|
|
|
|
|
|
|
|
: state-parse ( stream quot -- )
|
|
|
|
! with-input-stream implicitly creates a new scope which we use
|
|
|
|
swap [ init-parser call ] with-input-stream ; inline
|
|
|
|
|
2007-09-20 18:09:08 -04:00
|
|
|
! XML namespace processing: ns = namespace
|
|
|
|
|
|
|
|
! A stack of hashtables
|
|
|
|
SYMBOL: ns-stack
|
|
|
|
|
2009-01-20 16:37:21 -05:00
|
|
|
SYMBOL: depth
|
|
|
|
|
2007-09-20 18:09:08 -04:00
|
|
|
: attrs>ns ( attrs-alist -- hash )
|
|
|
|
! this should check to make sure URIs are valid
|
|
|
|
[
|
|
|
|
[
|
2008-08-27 18:02:54 -04:00
|
|
|
swap dup space>> "xmlns" =
|
|
|
|
[ main>> set ]
|
2007-09-20 18:09:08 -04:00
|
|
|
[
|
|
|
|
T{ name f "" "xmlns" f } names-match?
|
|
|
|
[ "" set ] [ drop ] if
|
|
|
|
] if
|
|
|
|
] assoc-each
|
|
|
|
] { } make-assoc f like ;
|
|
|
|
|
|
|
|
: add-ns ( name -- )
|
2008-08-27 18:02:54 -04:00
|
|
|
dup space>> dup ns-stack get assoc-stack
|
2008-12-02 20:59:16 -05:00
|
|
|
[ nip ] [ nonexist-ns ] if* >>url drop ;
|
2007-09-20 18:09:08 -04:00
|
|
|
|
|
|
|
: push-ns ( hash -- )
|
|
|
|
ns-stack get push ;
|
|
|
|
|
|
|
|
: pop-ns ( -- )
|
|
|
|
ns-stack get pop* ;
|
|
|
|
|
|
|
|
: init-ns-stack ( -- )
|
|
|
|
V{ H{
|
|
|
|
{ "xml" "http://www.w3.org/XML/1998/namespace" }
|
|
|
|
{ "xmlns" "http://www.w3.org/2000/xmlns" }
|
|
|
|
{ "" "" }
|
|
|
|
} } clone
|
|
|
|
ns-stack set ;
|
|
|
|
|
|
|
|
: tag-ns ( name attrs-alist -- name attrs )
|
|
|
|
dup attrs>ns push-ns
|
2008-12-02 20:59:16 -05:00
|
|
|
[ dup add-ns ] dip dup [ drop add-ns ] assoc-each <attrs> ;
|
2007-09-20 18:09:08 -04:00
|
|
|
|
|
|
|
! Parsing names
|
|
|
|
|
2009-01-20 16:37:21 -05:00
|
|
|
: valid-name? ( str -- ? )
|
|
|
|
[ f ] [
|
|
|
|
version=1.0? swap {
|
|
|
|
[ first name-start? ]
|
|
|
|
[ rest-slice [ name-char? ] with all? ]
|
|
|
|
} 2&&
|
|
|
|
] if-empty ;
|
|
|
|
|
|
|
|
: prefixed-name ( str -- name/f )
|
|
|
|
":" split dup length 2 = [
|
|
|
|
[ [ valid-name? ] all? ]
|
|
|
|
[ first2 f <name> ] bi and
|
|
|
|
] [ drop f ] if ;
|
|
|
|
|
|
|
|
: interpret-name ( str -- name )
|
|
|
|
dup prefixed-name [ ] [
|
|
|
|
dup valid-name?
|
|
|
|
[ <simple-name> ] [ bad-name ] if
|
|
|
|
] ?if ;
|
2007-09-20 18:09:08 -04:00
|
|
|
|
2009-01-20 16:37:21 -05:00
|
|
|
: take-name ( -- string )
|
|
|
|
version=1.0? '[ _ get-char name-char? not ] take-until ;
|
2007-09-20 18:09:08 -04:00
|
|
|
|
2009-01-20 16:37:21 -05:00
|
|
|
: parse-name ( -- name )
|
|
|
|
take-name interpret-name ;
|
2009-01-15 16:25:00 -05:00
|
|
|
|
2009-01-20 16:37:21 -05:00
|
|
|
: parse-name-starting ( string -- name )
|
|
|
|
take-name append interpret-name ;
|
2009-01-15 16:25:00 -05:00
|
|
|
|
2007-09-20 18:09:08 -04:00
|
|
|
! -- Parsing strings
|
|
|
|
|
2009-01-15 23:20:24 -05:00
|
|
|
: parse-named-entity ( string -- )
|
2009-01-19 23:25:15 -05:00
|
|
|
dup entities at [ , ] [
|
2009-01-15 23:20:24 -05:00
|
|
|
dup extra-entities get at
|
2009-01-19 23:25:15 -05:00
|
|
|
[ % ] [ no-entity ] ?if
|
2007-09-20 18:09:08 -04:00
|
|
|
] ?if ;
|
|
|
|
|
|
|
|
: parse-entity ( -- )
|
|
|
|
next CHAR: ; take-char next
|
|
|
|
"#" ?head [
|
|
|
|
"x" ?head 16 10 ? base> ,
|
2009-01-15 23:20:24 -05:00
|
|
|
] [ parse-named-entity ] if ;
|
2007-09-20 18:09:08 -04:00
|
|
|
|
2009-01-21 00:54:33 -05:00
|
|
|
SYMBOL: pe-table
|
|
|
|
SYMBOL: in-dtd?
|
|
|
|
|
|
|
|
: parse-pe ( -- )
|
|
|
|
next CHAR: ; take-char dup next
|
|
|
|
pe-table get at [ % ] [ no-entity ] ?if ;
|
|
|
|
|
2009-01-19 23:25:15 -05:00
|
|
|
:: (parse-char) ( quot: ( ch -- ? ) -- )
|
|
|
|
get-char :> char
|
|
|
|
{
|
|
|
|
{ [ char not ] [ ] }
|
|
|
|
{ [ char quot call ] [ next ] }
|
|
|
|
{ [ char CHAR: & = ] [ parse-entity quot (parse-char) ] }
|
2009-01-21 00:54:33 -05:00
|
|
|
{ [ in-dtd? get char CHAR: % = and ] [ parse-pe quot (parse-char) ] }
|
2009-01-19 23:25:15 -05:00
|
|
|
[ char , next quot (parse-char) ]
|
|
|
|
} cond ; inline recursive
|
|
|
|
|
|
|
|
: parse-char ( quot: ( ch -- ? ) -- seq )
|
|
|
|
[ (parse-char) ] "" make ; inline
|
2007-09-20 18:09:08 -04:00
|
|
|
|
2009-01-19 23:25:15 -05:00
|
|
|
: assure-no-]]> ( circular -- )
|
|
|
|
"]]>" sequence= [ text-w/]]> ] when ;
|
2007-09-20 18:09:08 -04:00
|
|
|
|
2009-01-20 16:37:21 -05:00
|
|
|
:: parse-text ( -- string )
|
|
|
|
3 f <array> <circular> :> circ
|
|
|
|
depth get zero? :> no-text [| char |
|
|
|
|
char circ push-circular
|
|
|
|
circ assure-no-]]>
|
|
|
|
no-text [ char blank? char CHAR: < = or [
|
|
|
|
char 1string t pre/post-content
|
|
|
|
] unless ] when
|
|
|
|
char CHAR: < =
|
2009-01-19 23:25:15 -05:00
|
|
|
] parse-char ;
|
|
|
|
|
2007-09-20 18:09:08 -04:00
|
|
|
! Parsing tags
|
|
|
|
|
|
|
|
: start-tag ( -- name ? )
|
|
|
|
#! Outputs the name and whether this is a closing tag
|
|
|
|
get-char CHAR: / = dup [ next ] when
|
|
|
|
parse-name swap ;
|
|
|
|
|
2009-01-21 00:54:33 -05:00
|
|
|
: normalize-quote ( str -- str )
|
|
|
|
[ dup "\t\r\n" member? [ drop CHAR: \s ] when ] map ;
|
|
|
|
|
2009-01-19 23:25:15 -05:00
|
|
|
: (parse-quote) ( <-disallowed? ch -- string )
|
|
|
|
swap '[
|
|
|
|
dup _ = [ drop t ]
|
|
|
|
[ CHAR: < = _ and [ attr-w/< ] [ f ] if ] if
|
2009-01-21 00:54:33 -05:00
|
|
|
] parse-char normalize-quote get-char
|
2009-01-19 23:25:15 -05:00
|
|
|
[ unclosed-quote ] unless ; inline
|
2009-01-15 23:20:24 -05:00
|
|
|
|
2009-01-19 23:25:15 -05:00
|
|
|
: parse-quote* ( <-disallowed? -- seq )
|
2009-01-15 23:20:24 -05:00
|
|
|
pass-blank get-char dup "'\"" member?
|
2009-01-19 23:25:15 -05:00
|
|
|
[ next (parse-quote) ] [ quoteless-attr ] if ; inline
|
|
|
|
|
|
|
|
: parse-quote ( -- seq )
|
|
|
|
f parse-quote* ;
|
|
|
|
|
2007-09-20 18:09:08 -04:00
|
|
|
: parse-attr ( -- )
|
2009-01-20 16:37:21 -05:00
|
|
|
parse-name pass-blank CHAR: = expect pass-blank
|
2009-01-21 00:54:33 -05:00
|
|
|
t parse-quote* 2array , ;
|
2007-09-20 18:09:08 -04:00
|
|
|
|
|
|
|
: (middle-tag) ( -- )
|
|
|
|
pass-blank version=1.0? get-char name-start?
|
|
|
|
[ parse-attr (middle-tag) ] when ;
|
|
|
|
|
2009-01-19 23:25:15 -05:00
|
|
|
: assure-no-duplicates ( attrs-alist -- attrs-alist )
|
|
|
|
H{ } clone 2dup '[ swap _ push-at ] assoc-each
|
|
|
|
[ nip length 2 >= ] assoc-filter >alist
|
|
|
|
[ first first2 duplicate-attr ] unless-empty ;
|
|
|
|
|
2007-09-20 18:09:08 -04:00
|
|
|
: middle-tag ( -- attrs-alist )
|
2007-12-23 14:57:39 -05:00
|
|
|
! f make will make a vector if it has any elements
|
2009-01-19 23:25:15 -05:00
|
|
|
[ (middle-tag) ] f make pass-blank
|
|
|
|
assure-no-duplicates ;
|
2007-09-20 18:09:08 -04:00
|
|
|
|
2009-01-20 16:37:21 -05:00
|
|
|
: close ( -- )
|
|
|
|
pass-blank CHAR: > expect ;
|
|
|
|
|
2007-09-20 18:09:08 -04:00
|
|
|
: end-tag ( name attrs-alist -- tag )
|
|
|
|
tag-ns pass-blank get-char CHAR: / =
|
2009-01-20 16:37:21 -05:00
|
|
|
[ pop-ns <contained> next CHAR: > expect ]
|
|
|
|
[ depth inc <opener> close ] if ;
|
2007-09-20 18:09:08 -04:00
|
|
|
|
|
|
|
: take-comment ( -- comment )
|
|
|
|
"--" expect-string
|
|
|
|
"--" take-string
|
|
|
|
<comment>
|
|
|
|
CHAR: > expect ;
|
|
|
|
|
|
|
|
: take-cdata ( -- string )
|
2009-01-20 16:37:21 -05:00
|
|
|
depth get zero? [ bad-cdata ] when
|
2007-10-12 16:28:23 -04:00
|
|
|
"[CDATA[" expect-string "]]>" take-string ;
|
2007-09-20 18:09:08 -04:00
|
|
|
|
2009-01-19 23:25:15 -05:00
|
|
|
: take-word ( -- string )
|
|
|
|
[ get-char blank? ] take-until ;
|
|
|
|
|
|
|
|
: take-decl-contents ( -- first second )
|
|
|
|
pass-blank take-word pass-blank ">" take-string ;
|
|
|
|
|
2008-12-02 20:59:16 -05:00
|
|
|
: take-element-decl ( -- element-decl )
|
2009-01-19 23:25:15 -05:00
|
|
|
take-decl-contents <element-decl> ;
|
2008-12-02 20:59:16 -05:00
|
|
|
|
2009-01-21 00:54:33 -05:00
|
|
|
: take-attlist-decl ( -- attlist-decl )
|
2009-01-19 23:25:15 -05:00
|
|
|
take-decl-contents <attlist-decl> ;
|
2008-12-02 20:59:16 -05:00
|
|
|
|
2009-01-20 16:37:21 -05:00
|
|
|
: take-notation-decl ( -- notation-decl )
|
|
|
|
take-decl-contents <notation-decl> ;
|
|
|
|
|
2008-12-02 20:59:16 -05:00
|
|
|
: take-until-one-of ( seps -- str sep )
|
|
|
|
'[ get-char _ member? ] take-until get-char ;
|
|
|
|
|
|
|
|
: take-system-id ( -- system-id )
|
2009-01-20 16:37:21 -05:00
|
|
|
parse-quote <system-id> close ;
|
2008-12-02 20:59:16 -05:00
|
|
|
|
|
|
|
: take-public-id ( -- public-id )
|
2009-01-20 16:37:21 -05:00
|
|
|
parse-quote parse-quote <public-id> close ;
|
2008-12-02 20:59:16 -05:00
|
|
|
|
|
|
|
DEFER: direct
|
|
|
|
|
|
|
|
: (take-internal-subset) ( -- )
|
|
|
|
pass-blank get-char {
|
|
|
|
{ CHAR: ] [ next ] }
|
|
|
|
[ drop "<!" expect-string direct , (take-internal-subset) ]
|
|
|
|
} case ;
|
|
|
|
|
|
|
|
: take-internal-subset ( -- seq )
|
2009-01-21 00:54:33 -05:00
|
|
|
[
|
|
|
|
H{ } pe-table set
|
|
|
|
t in-dtd? set
|
|
|
|
(take-internal-subset)
|
|
|
|
] { } make ;
|
2008-12-02 20:59:16 -05:00
|
|
|
|
|
|
|
: (take-external-id) ( token -- external-id )
|
|
|
|
pass-blank {
|
|
|
|
{ "SYSTEM" [ take-system-id ] }
|
|
|
|
{ "PUBLIC" [ take-public-id ] }
|
|
|
|
[ bad-external-id ]
|
|
|
|
} case ;
|
|
|
|
|
|
|
|
: take-external-id ( -- external-id )
|
2009-01-19 23:25:15 -05:00
|
|
|
take-word (take-external-id) ;
|
|
|
|
|
|
|
|
: only-blanks ( str -- )
|
|
|
|
[ blank? ] all? [ bad-decl ] unless ;
|
2008-12-02 20:59:16 -05:00
|
|
|
|
2009-01-21 00:54:33 -05:00
|
|
|
: nontrivial-doctype ( -- external-id internal-subset )
|
|
|
|
pass-blank get-char CHAR: [ = [
|
|
|
|
next take-internal-subset f swap close
|
|
|
|
] [
|
|
|
|
" >" take-until-one-of {
|
|
|
|
{ CHAR: \s [ (take-external-id) ] }
|
|
|
|
{ CHAR: > [ only-blanks f ] }
|
|
|
|
} case f
|
|
|
|
] if ;
|
|
|
|
|
2008-12-02 20:59:16 -05:00
|
|
|
: take-doctype-decl ( -- doctype-decl )
|
|
|
|
pass-blank " >" take-until-one-of {
|
2009-01-21 00:54:33 -05:00
|
|
|
{ CHAR: \s [ nontrivial-doctype ] }
|
2008-12-02 20:59:16 -05:00
|
|
|
{ CHAR: > [ f f ] }
|
|
|
|
} case <doctype-decl> ;
|
|
|
|
|
2009-01-21 00:54:33 -05:00
|
|
|
: take-entity-def ( var -- entity-name entity-def )
|
2009-01-21 01:17:25 -05:00
|
|
|
[
|
|
|
|
take-word pass-blank get-char {
|
|
|
|
{ CHAR: ' [ parse-quote ] }
|
|
|
|
{ CHAR: " [ parse-quote ] }
|
|
|
|
[ drop take-external-id ]
|
|
|
|
} case swap
|
|
|
|
] dip [ [ ?set-at ] change ] 2keep swap ;
|
2009-01-15 23:20:24 -05:00
|
|
|
|
2008-12-02 20:59:16 -05:00
|
|
|
: take-entity-decl ( -- entity-decl )
|
|
|
|
pass-blank get-char {
|
2009-01-21 00:54:33 -05:00
|
|
|
{ CHAR: % [ next pass-blank pe-table take-entity-def ] }
|
|
|
|
[ drop extra-entities take-entity-def ]
|
2008-12-02 20:59:16 -05:00
|
|
|
} case
|
2009-01-20 16:37:21 -05:00
|
|
|
close <entity-decl> ;
|
2008-12-02 20:59:16 -05:00
|
|
|
|
2007-09-20 18:09:08 -04:00
|
|
|
: take-directive ( -- directive )
|
2009-01-20 16:37:21 -05:00
|
|
|
take-name {
|
2008-12-02 20:59:16 -05:00
|
|
|
{ "ELEMENT" [ take-element-decl ] }
|
|
|
|
{ "ATTLIST" [ take-attlist-decl ] }
|
|
|
|
{ "DOCTYPE" [ take-doctype-decl ] }
|
|
|
|
{ "ENTITY" [ take-entity-decl ] }
|
2009-01-20 16:37:21 -05:00
|
|
|
{ "NOTATION" [ take-notation-decl ] }
|
2008-12-02 20:59:16 -05:00
|
|
|
[ bad-directive ]
|
|
|
|
} case ;
|
2007-09-20 18:09:08 -04:00
|
|
|
|
|
|
|
: direct ( -- object )
|
|
|
|
get-char {
|
|
|
|
{ CHAR: - [ take-comment ] }
|
|
|
|
{ CHAR: [ [ take-cdata ] }
|
|
|
|
[ drop take-directive ]
|
|
|
|
} case ;
|
|
|
|
|
|
|
|
: assure-no-extra ( seq -- )
|
|
|
|
[ first ] map {
|
|
|
|
T{ name f "" "version" f }
|
|
|
|
T{ name f "" "encoding" f }
|
|
|
|
T{ name f "" "standalone" f }
|
2008-04-26 03:01:43 -04:00
|
|
|
} diff
|
2008-12-02 20:59:16 -05:00
|
|
|
[ extra-attrs ] unless-empty ;
|
2007-09-20 18:09:08 -04:00
|
|
|
|
|
|
|
: good-version ( version -- version )
|
2008-12-02 20:59:16 -05:00
|
|
|
dup { "1.0" "1.1" } member? [ bad-version ] unless ;
|
2007-09-20 18:09:08 -04:00
|
|
|
|
2009-01-15 23:20:24 -05:00
|
|
|
: prolog-version ( alist -- version )
|
|
|
|
T{ name f "" "version" f } swap at
|
|
|
|
[ good-version ] [ versionless-prolog ] if* ;
|
|
|
|
|
|
|
|
: prolog-encoding ( alist -- encoding )
|
|
|
|
T{ name f "" "encoding" f } swap at "UTF-8" or ;
|
|
|
|
|
2009-01-21 00:54:33 -05:00
|
|
|
: yes/no>bool ( string -- t/f )
|
|
|
|
{
|
|
|
|
{ "yes" [ t ] }
|
|
|
|
{ "no" [ f ] }
|
|
|
|
[ not-yes/no ]
|
|
|
|
} case ;
|
|
|
|
|
2009-01-15 23:20:24 -05:00
|
|
|
: prolog-standalone ( alist -- version )
|
2007-09-20 18:09:08 -04:00
|
|
|
T{ name f "" "standalone" f } swap at
|
2009-01-15 23:20:24 -05:00
|
|
|
[ yes/no>bool ] [ f ] if* ;
|
|
|
|
|
|
|
|
: prolog-attrs ( alist -- prolog )
|
|
|
|
[ prolog-version ]
|
|
|
|
[ prolog-encoding ]
|
|
|
|
[ prolog-standalone ]
|
|
|
|
tri <prolog> ;
|
2007-09-20 18:09:08 -04:00
|
|
|
|
2009-01-15 01:11:23 -05:00
|
|
|
SYMBOL: string-input?
|
|
|
|
: decode-input-if ( encoding -- )
|
|
|
|
string-input? get [ drop ] [ decode-input ] if ;
|
|
|
|
|
2007-09-20 18:09:08 -04:00
|
|
|
: parse-prolog ( -- prolog )
|
|
|
|
pass-blank middle-tag "?>" expect-string
|
|
|
|
dup assure-no-extra prolog-attrs
|
2009-01-15 01:11:23 -05:00
|
|
|
dup encoding>> dup "UTF-16" =
|
|
|
|
[ drop ] [ name>encoding [ decode-input-if ] when* ] if
|
2007-09-20 18:09:08 -04:00
|
|
|
dup prolog-data set ;
|
|
|
|
|
|
|
|
: instruct ( -- instruction )
|
2009-01-20 16:37:21 -05:00
|
|
|
take-name {
|
|
|
|
{ [ dup "xml" = ] [ drop parse-prolog ] }
|
|
|
|
{ [ dup >lower "xml" = ] [ capitalized-prolog ] }
|
|
|
|
{ [ dup valid-name? not ] [ bad-name ] }
|
|
|
|
[ "?>" take-string append <instruction> ]
|
|
|
|
} cond ;
|
2007-09-20 18:09:08 -04:00
|
|
|
|
|
|
|
: make-tag ( -- tag )
|
|
|
|
{
|
|
|
|
{ [ get-char dup CHAR: ! = ] [ drop next direct ] }
|
2009-01-15 23:20:24 -05:00
|
|
|
{ [ CHAR: ? = ] [ next instruct ] }
|
2008-04-11 13:57:43 -04:00
|
|
|
[
|
2009-01-20 16:37:21 -05:00
|
|
|
start-tag [ dup add-ns pop-ns <closer> depth dec close ]
|
2007-09-20 18:09:08 -04:00
|
|
|
[ middle-tag end-tag ] if
|
2008-04-11 13:57:43 -04:00
|
|
|
]
|
2007-09-20 18:09:08 -04:00
|
|
|
} cond ;
|
2009-01-15 01:11:23 -05:00
|
|
|
|
|
|
|
! Autodetecting encodings
|
|
|
|
|
2009-01-15 16:25:00 -05:00
|
|
|
: continue-make-tag ( str -- tag )
|
2009-01-20 16:37:21 -05:00
|
|
|
parse-name-starting middle-tag end-tag ;
|
2009-01-15 16:25:00 -05:00
|
|
|
|
2009-01-15 01:11:23 -05:00
|
|
|
: start-utf16le ( -- tag )
|
|
|
|
utf16le decode-input-if
|
|
|
|
CHAR: ? expect
|
2009-01-19 23:25:15 -05:00
|
|
|
0 expect check instruct ;
|
2009-01-15 01:11:23 -05:00
|
|
|
|
2009-01-15 16:25:00 -05:00
|
|
|
: 10xxxxxx? ( ch -- ? )
|
|
|
|
-6 shift 3 bitand 2 = ;
|
|
|
|
|
|
|
|
: start<name ( ch -- tag )
|
|
|
|
ascii?
|
|
|
|
[ utf8 decode-input-if next make-tag ] [
|
|
|
|
next
|
|
|
|
[ get-next 10xxxxxx? not ] take-until
|
|
|
|
get-char suffix utf8 decode
|
|
|
|
utf8 decode-input-if next
|
|
|
|
continue-make-tag
|
|
|
|
] if ;
|
|
|
|
|
2009-01-15 01:11:23 -05:00
|
|
|
: start< ( -- tag )
|
|
|
|
get-next {
|
|
|
|
{ 0 [ next next start-utf16le ] }
|
2009-01-19 23:25:15 -05:00
|
|
|
{ CHAR: ? [ check next next instruct ] } ! XML prolog parsing sets the encoding
|
|
|
|
{ CHAR: ! [ check utf8 decode-input next next direct ] }
|
|
|
|
[ check start<name ]
|
2009-01-15 01:11:23 -05:00
|
|
|
} case ;
|
|
|
|
|
|
|
|
: skip-utf8-bom ( -- tag )
|
|
|
|
"\u0000bb\u0000bf" expect utf8 decode-input
|
2009-01-19 23:25:15 -05:00
|
|
|
CHAR: < expect check make-tag ;
|
2009-01-15 01:11:23 -05:00
|
|
|
|
2009-01-15 23:20:24 -05:00
|
|
|
: decode-expecting ( encoding string -- tag )
|
2009-01-19 23:25:15 -05:00
|
|
|
[ decode-input-if next ] [ expect-string ] bi* check make-tag ;
|
2009-01-15 23:20:24 -05:00
|
|
|
|
2009-01-15 01:11:23 -05:00
|
|
|
: start-utf16be ( -- tag )
|
2009-01-15 23:20:24 -05:00
|
|
|
utf16be "<" decode-expecting ;
|
2009-01-15 01:11:23 -05:00
|
|
|
|
|
|
|
: skip-utf16le-bom ( -- tag )
|
2009-01-15 23:20:24 -05:00
|
|
|
utf16le "\u0000fe<" decode-expecting ;
|
2009-01-15 01:11:23 -05:00
|
|
|
|
|
|
|
: skip-utf16be-bom ( -- tag )
|
2009-01-15 23:20:24 -05:00
|
|
|
utf16be "\u0000ff<" decode-expecting ;
|
2009-01-15 01:11:23 -05:00
|
|
|
|
|
|
|
: start-document ( -- tag )
|
|
|
|
get-char {
|
|
|
|
{ CHAR: < [ start< ] }
|
|
|
|
{ 0 [ start-utf16be ] }
|
|
|
|
{ HEX: EF [ skip-utf8-bom ] }
|
|
|
|
{ HEX: FF [ skip-utf16le-bom ] }
|
|
|
|
{ HEX: FE [ skip-utf16be-bom ] }
|
|
|
|
{ f [ "" ] }
|
2009-01-15 23:20:24 -05:00
|
|
|
[ drop utf8 decode-input-if f ]
|
|
|
|
! Same problem as with <e`>, in the case of XML chunks?
|
2009-01-19 23:25:15 -05:00
|
|
|
} case check ;
|