racket/collects/xml/private/syntax.ss
2008-02-23 09:42:03 +00:00

214 lines
7.5 KiB
Scheme

(module syntax mzscheme
(provide syntax-structs@)
(require mzlib/unitsig
"sig.ss")
; to make error-raising functions named like structure mutators
(define-syntax (struct! stx)
(syntax-case stx ()
[(struct-src name (field ...))
(with-syntax ([struct:name (datum->syntax-object
(syntax name)
(string->symbol (string-append "struct:" (symbol->string (syntax-object->datum (syntax name))))))]
[(setter-name ...)
(let ([struct-name
(symbol->string (syntax-object->datum (syntax name)))])
(map (lambda (field-name)
(datum->syntax-object
field-name
(string->symbol
(string-append
"set-"
struct-name
"-"
(symbol->string (syntax-object->datum field-name))
"!"))))
(syntax->list (syntax (field ...)))))])
(syntax
(begin
(define struct:name void)
(define (setter-name s v)
(error (quote setter-name) "cannot mutate XML syntax"))
...)))]))
(define syntax-structs@
(unit/sig xml-structs^
(import)
; The locations from the two sets of structures shouldn't mingle, so I'm
; re-defining the location structure. Maybe this is not a good idea, but I
; think it's okay.
(define-struct location (line char offset))
(define-struct source (start stop))
; make-document : prolog element ? -> document
(define (make-document p e ?) e)
; make-prolog : ? #f -> prolog
(define (make-prolog ? ??) #f)
; make-element : src src sym (listof attribute) (listof content) -> element
(define (make-element from to name attrs content)
(wrap (list* name attrs content) from to))
; make-pcdata : src src str -> pcdata
(define (make-pcdata from to x)
(wrap x from to))
; make-cdata : src src str -> cdata
(define (make-cdata from to x)
(wrap x from to))
; make-entity : src src (U sym num) -> entity
(define (make-entity from to entity)
(wrap entity from to))
; make-comment : str -> comment
; There is no syntax object representation for comments
(define (make-comment x) #f)
; make-pi : src src sym str -> pi
; There's not really a syntax object representation for pi's either
(define (make-pi from to name val) #f)
; make-attribute : src src sym str -> attribute
(define (make-attribute from to name val)
(wrap (list name val) from to))
(define (make-document-type . x) #f)
(define (make-external-dtd . x) #f)
(define (make-external-dtd/public . x) #f)
(define (make-external-dtd/system . x) #f)
; wrap : tst src src -> syntax
(define (wrap x from to)
(datum->syntax-object #f x (position from to)))
; position : src src -> (list #f nat nat nat nat)
(define (position from to)
(let ([start-offset (location-offset from)])
(list #f (location-line from) (location-char from) start-offset
(- (location-offset to) start-offset))))
; : syntax -> syntax
(define (attribute-name a) (car (syntax->list a)))
(define (attribute-value a) (cadr (syntax->list a)))
; : syntax -> syntax
(define (element-name e) (car (syntax->list e)))
(define (element-attributes e) (cadr (syntax->list e)))
(define (element-content e) (cddr (syntax->list e)))
(define (entity-text e) (syntax-e e))
(define (pcdata-string x) (syntax-e x))
(define (cdata-string x) (syntax-e x))
(define (comment-text c)
(error 'comment-text "expected a syntax representation of an XML comment, received ~e" c))
; conflate documents with their root elements
(define (document-element d) d)
; more here - spoof document pieces better?
(define (document-misc d) null)
(define (document-prolog d) null)
(define (document-type-external dtd)
(error 'document-type-external "expected a dtd, given ~e" dtd))
(define (document-type-inlined dtd)
(error 'document-type-inlined "expected a dtd, given ~e" dtd))
(define (document-type-name dtd)
(error 'document-type-name "expected a dtd, given ~e" dtd))
(define (external-dtd-system x)
(error 'external-dtd-system "expected an external dtd, given ~e" x))
(define (external-dtd/public-public x)
(error 'external-dtd/public-public "expected an external dtd, given ~e" x))
(define (pi-instruction x)
(error 'pi-instruction "expected a pi, given ~e" x))
(define (pi-target-name x)
(error 'pi-target-name "expected a pi, given ~e" x))
(define (prolog-dtd x)
(error 'prolog-dtd "expected a prolog, given ~e" x))
(define (prolog-misc x)
(error 'prolog-misc "expected a prolog, given ~e" x))
(define (prolog-misc2 x)
(error 'prolog-misc2 "expected a prolog, given ~e" x))
; : tst -> bool
(define (attribute? a)
(and (syntax? a)
(let ([x (syntax-object->datum a)])
(and (pair? x) (symbol? (car x))
(pair? (cdr x)) (string? (cadr x))
(null? (cddr x))))))
; : tst -> bool
(define (comment? x) #f)
; : tst -> bool
(define (content? x)
(and (syntax? x)
(or (string? (syntax-object->datum x))
(element? x))))
; : tst -> bool
(define (element? x)
(and (syntax? x)
(let ([e (syntax-e x)])
(and (pair? e) (symbol? (car e))
(pair? (cdr e)) (list? (cadr e))
(andmap attribute? (cadr e))
(list? (cddr e))
(andmap content? (cddr e))))))
; : tst -> bool
(define document? element?)
; : tst -> bool
(define (document-type? x) #f)
; : tst -> bool
(define (external-dtd/public? x) #f)
(define (external-dtd/system? x) #f)
(define (external-dtd? x) #f)
(define (prolog? x) #f)
(define (pi? x) #f)
; : tst -> bool
(define (pcdata? x)
(and (syntax? x) (string (syntax-e x))))
(define (cdata? x)
(and (syntax? x) (string (syntax-e x))))
; : tst -> bool
(define (entity? x)
(and (syntax? x) (let ([r (syntax-e x)]) (or (symbol? r) (number? r)))))
;(struct! location (line char offset))
(struct! document (prolog element misc))
(struct! comment (text))
(struct! prolog (misc dtd misc2))
(struct! document-type (name external inlined))
(struct! external-dtd (system))
(struct! external-dtd/public (public))
(struct! external-dtd/system ())
(struct! element (name attributes content))
(struct! attribute (name value))
(struct! pi (target-name instruction))
;(struct! source (start stop))
(struct! pcdata (string))
(struct! cdata (string))
(struct! entity (text))
)))