factor/basis/xml/tokenize/tokenize.factor

365 lines
9.7 KiB
Factor
Raw Normal View History

2007-09-20 18:09:08 -04:00
! Copyright (C) 2005, 2006 Daniel Ehrenberg
! See http://factorcode.org/license.txt for BSD license.
USING: accessors arrays ascii assocs combinators
combinators.short-circuit fry io.encodings io.encodings.iana
io.encodings.string io.encodings.utf16 io.encodings.utf8 kernel make
math math.parser namespaces sequences sets splitting state-parser
2009-01-15 23:20:24 -05:00
strings xml.char-classes xml.data xml.entities xml.errors hashtables ;
2007-09-20 18:09:08 -04:00
IN: xml.tokenize
! XML namespace processing: ns = namespace
! A stack of hashtables
SYMBOL: ns-stack
: 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
[ 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
[ dup add-ns ] dip dup [ drop add-ns ] assoc-each <attrs> ;
2007-09-20 18:09:08 -04:00
! Parsing names
: version=1.0? ( -- ? )
2008-09-01 19:43:52 -04:00
prolog-data get version>> "1.0" = ;
2007-09-20 18:09:08 -04:00
! version=1.0? is calculated once and passed around for efficiency
: assure-name ( str version=1.0? -- str )
over {
[ first name-start? ]
[ rest-slice [ name-char? ] with all? ]
2009-01-15 17:35:55 -05:00
} 2&& [ bad-name ] unless ;
2007-09-20 18:09:08 -04:00
: (parse-name) ( start -- str )
version=1.0?
[ [ get-char name-char? not ] curry take-until append ]
[ assure-name ] bi ;
: parse-name-starting ( start -- name )
2007-09-20 18:09:08 -04:00
(parse-name) get-char CHAR: : =
[ next "" (parse-name) ] [ "" swap ] if f <name> ;
: parse-name ( -- name )
"" parse-name-starting ;
2007-09-20 18:09:08 -04:00
! -- Parsing strings
2009-01-15 23:20:24 -05:00
: parse-named-entity ( string -- )
2007-09-20 18:09:08 -04:00
dup entities at [ , ] [
2009-01-15 23:20:24 -05:00
dup extra-entities get at
[ dup number? [ , ] [ % ] if ] [ no-entity ] ?if ! Make less hackish
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
: (parse-char) ( ch -- )
get-char {
{ [ dup not ] [ 2drop ] }
{ [ 2dup = ] [ 2drop next ] }
{ [ dup CHAR: & = ] [ drop parse-entity (parse-char) ] }
2008-04-11 13:57:43 -04:00
[ , next (parse-char) ]
2007-09-20 18:09:08 -04:00
} cond ;
: parse-char ( ch -- string )
[ (parse-char) ] "" make ;
: parse-text ( -- string )
CHAR: < 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-15 23:20:24 -05:00
: (parse-quote) ( ch -- string )
parse-char get-char
[ unclosed-quote ] unless ;
: parse-quote ( -- seq )
pass-blank get-char dup "'\"" member?
[ next (parse-quote) ] [ quoteless-attr ] if ;
2007-09-20 18:09:08 -04:00
: parse-attr ( -- )
2009-01-15 23:20:24 -05:00
parse-name
pass-blank CHAR: = expect
parse-quote
2007-09-20 18:09:08 -04:00
2array , ;
: (middle-tag) ( -- )
pass-blank version=1.0? get-char name-start?
[ parse-attr (middle-tag) ] when ;
: middle-tag ( -- attrs-alist )
2007-12-23 14:57:39 -05:00
! f make will make a vector if it has any elements
[ (middle-tag) ] f make pass-blank ;
2007-09-20 18:09:08 -04:00
: end-tag ( name attrs-alist -- tag )
tag-ns pass-blank get-char CHAR: / =
[ pop-ns <contained> next ] [ <opener> ] if ;
: take-comment ( -- comment )
"--" expect-string
"--" take-string
<comment>
CHAR: > expect ;
: take-cdata ( -- string )
2007-10-12 16:28:23 -04:00
"[CDATA[" expect-string "]]>" take-string ;
2007-09-20 18:09:08 -04:00
: take-element-decl ( -- element-decl )
pass-blank " " take-string pass-blank ">" take-string <element-decl> ;
: take-attlist-decl ( -- doctype-decl )
pass-blank " " take-string pass-blank ">" take-string <attlist-decl> ;
: take-until-one-of ( seps -- str sep )
'[ get-char _ member? ] take-until get-char ;
: only-blanks ( str -- )
[ blank? ] all? [ bad-doctype-decl ] unless ;
2009-01-15 23:20:24 -05:00
: take-system-literal ( -- str ) ! replace with parse-quote?
pass-blank get-char next {
{ CHAR: ' [ "'" take-string ] }
{ CHAR: " [ "\"" take-string ] }
} case ;
: take-system-id ( -- system-id )
take-system-literal <system-id>
">" take-string only-blanks ;
: take-public-id ( -- public-id )
take-system-literal
take-system-literal <public-id>
">" take-string only-blanks ;
DEFER: direct
: (take-internal-subset) ( -- )
pass-blank get-char {
{ CHAR: ] [ next ] }
[ drop "<!" expect-string direct , (take-internal-subset) ]
} case ;
: take-internal-subset ( -- seq )
[ (take-internal-subset) ] { } make ;
: (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 )
" " take-string (take-external-id) ;
: take-doctype-decl ( -- doctype-decl )
pass-blank " >" take-until-one-of {
{ CHAR: \s [
pass-blank get-char CHAR: [ = [
next take-internal-subset f swap
">" take-string only-blanks
] [
" >" take-until-one-of {
{ CHAR: \s [ (take-external-id) ] }
{ CHAR: > [ only-blanks f ] }
} case f
] if
] }
{ CHAR: > [ f f ] }
} case <doctype-decl> ;
: take-entity-def ( -- entity-name entity-def )
" " take-string pass-blank get-char {
2009-01-15 23:20:24 -05:00
{ CHAR: ' [ parse-quote ] }
{ CHAR: " [ parse-quote ] }
[ drop take-external-id ]
} case ;
2009-01-15 23:20:24 -05:00
: associate-entity ( entity-name entity-def -- )
swap extra-entities [ ?set-at ] change ;
: take-entity-decl ( -- entity-decl )
pass-blank get-char {
{ CHAR: % [ next pass-blank take-entity-def ] }
2009-01-15 23:20:24 -05:00
[ drop take-entity-def 2dup associate-entity ]
} case
">" take-string only-blanks <entity-decl> ;
2007-09-20 18:09:08 -04:00
: take-directive ( -- directive )
" " take-string {
{ "ELEMENT" [ take-element-decl ] }
{ "ATTLIST" [ take-attlist-decl ] }
{ "DOCTYPE" [ take-doctype-decl ] }
{ "ENTITY" [ take-entity-decl ] }
[ 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 ;
: yes/no>bool ( string -- t/f )
{
{ "yes" [ t ] }
{ "no" [ f ] }
[ not-yes/no ]
2007-09-20 18:09:08 -04:00
} case ;
: assure-no-extra ( seq -- )
[ first ] map {
T{ name f "" "version" f }
T{ name f "" "encoding" f }
T{ name f "" "standalone" f }
} diff
[ extra-attrs ] unless-empty ;
2007-09-20 18:09:08 -04:00
: good-version ( version -- version )
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 ;
: 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 )
"" (parse-name) dup "xml" =
2007-09-20 18:09:08 -04:00
[ drop parse-prolog ] [
dup >lower "xml" =
[ capitalized-prolog ]
2007-09-20 18:09:08 -04:00
[ "?>" take-string append <instruction> ] if
] if ;
: 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
[
2007-09-20 18:09:08 -04:00
start-tag [ dup add-ns pop-ns <closer> ]
[ middle-tag end-tag ] if
CHAR: > expect
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
: continue-make-tag ( str -- tag )
parse-name-starting middle-tag end-tag CHAR: > expect ;
2009-01-15 01:11:23 -05:00
: start-utf16le ( -- tag )
utf16le decode-input-if
CHAR: ? expect
0 expect instruct ;
: 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 ] }
{ CHAR: ? [ next next instruct ] } ! XML prolog parsing sets the encoding
{ CHAR: ! [ utf8 decode-input next next direct ] }
[ start<name ]
2009-01-15 01:11:23 -05:00
} case ;
: skip-utf8-bom ( -- tag )
"\u0000bb\u0000bf" expect utf8 decode-input
CHAR: < expect make-tag ;
2009-01-15 23:20:24 -05:00
: decode-expecting ( encoding string -- tag )
[ decode-input-if next ] [ expect-string ] bi* make-tag ;
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-15 01:11:23 -05:00
} case ;