[functional-tests/regex] Get groups working
Hacked a hotpatch soln.
This commit is contained in:
parent
6e9e71da87
commit
da4bb22b6a
@ -320,59 +320,71 @@
|
|||||||
;; any := "."
|
;; any := "."
|
||||||
(define any (p:>> (p:lit ".") (p:pure (lambda (_) #t))))
|
(define any (p:>> (p:lit ".") (p:pure (lambda (_) #t))))
|
||||||
|
|
||||||
|
(define (combine rs)
|
||||||
|
(fold-left seq (car rs) (cdr rs)))
|
||||||
|
|
||||||
;;-----------------------------------------------------------
|
;;-----------------------------------------------------------
|
||||||
;; Higher level combinators, these build a symbolic rx
|
;; Higher level combinators, these build a symbolic rx
|
||||||
|
|
||||||
;; The definitions start being mutually recursive from here on in, so we make
|
;; FIXME: move a hotpatch form to (utils)
|
||||||
;; them thunks to defer evaluation.
|
(define rx
|
||||||
|
(let ((this (lambda xs #f)))
|
||||||
|
(lambda args
|
||||||
|
(if (and (= (length args) 2)
|
||||||
|
(eq? (car args) 'hotpatch))
|
||||||
|
(set! this (cadr args))
|
||||||
|
(apply this args)))))
|
||||||
|
|
||||||
;; group := "(" rx ")"
|
;; group := "(" rx ")"
|
||||||
(define (group)
|
(define group
|
||||||
(fmt #t (dsp "group") nl)
|
|
||||||
(bracket (p:lit "(")
|
(bracket (p:lit "(")
|
||||||
(p:lit ")")
|
(p:lit ")")
|
||||||
(rx)))
|
rx))
|
||||||
|
|
||||||
;; elementary-rx := group | any | eos | char-rx | set
|
;; elementary-rx := group | any | eos | char-rx | set
|
||||||
;; FIXME: put eos and group back in
|
;; FIXME: put eos and group back in
|
||||||
(define (elementary-rx)
|
(define elementary-rx
|
||||||
(p:lift (lambda (fn)
|
(p:alt (p:lift (lambda (fn)
|
||||||
(list (char-instr fn)))
|
(list (char-instr fn)))
|
||||||
(p:one-of any char-rx set)))
|
(p:one-of any char-rx set))
|
||||||
|
group))
|
||||||
|
|
||||||
;; plus-rx := elementary-rx "+"
|
;; plus-rx := elementary-rx "+"
|
||||||
(define (plus-rx)
|
(define plus-rx
|
||||||
(p:lift plus (p:<* (elementary-rx) (p:lit "+"))))
|
(p:lift plus (p:<* elementary-rx (p:lit "+"))))
|
||||||
|
|
||||||
;; star-rx := elementary-rx "*"
|
;; star-rx := elementary-rx "*"
|
||||||
(define (star-rx)
|
(define star-rx
|
||||||
(p:lift star (p:<* (elementary-rx) (p:lit "*"))))
|
(p:lift star (p:<* elementary-rx (p:lit "*"))))
|
||||||
|
|
||||||
;; basic-rx := star-rx | plus-rx | elementary-rx
|
;; basic-rx := star-rx | plus-rx | elementary-rx
|
||||||
(define (basic-rx)
|
(define basic-rx
|
||||||
(p:one-of (star-rx) (plus-rx) (elementary-rx)))
|
(p:one-of star-rx plus-rx elementary-rx))
|
||||||
|
|
||||||
;; simple-rx := basic-rx+
|
;; simple-rx := basic-rx+
|
||||||
(define (simple-rx)
|
(define simple-rx
|
||||||
(define (combine rs)
|
(p:lift combine (p:many+ basic-rx)))
|
||||||
(fold-left seq (car rs) (cdr rs)))
|
|
||||||
|
|
||||||
(p:lift combine (p:many+ (basic-rx))))
|
|
||||||
|
|
||||||
;; rx := simple-rx ("|" simple-rx)*
|
|
||||||
(define (rx)
|
|
||||||
(p:lift2 (lambda (r rs)
|
|
||||||
(fold-left alt r rs))
|
|
||||||
(simple-rx)
|
|
||||||
(p:many* (p:>> (p:lit "|")
|
|
||||||
(simple-rx)))))
|
|
||||||
|
|
||||||
;;-----------------------------------------------------------------------
|
;;-----------------------------------------------------------------------
|
||||||
;; The top level routine, parses the regex string and compiles it into a
|
;; The top level routine, parses the regex string and compiles it into a
|
||||||
;; matcher, or returns false if the parse failed.
|
;; matcher, or returns false if the parse failed.
|
||||||
;; regex :: string -> (matcher <string>)
|
;; regex :: string -> (matcher <string>)
|
||||||
(define (regex str)
|
;; FIXME: it's tempting to return a function that raises if there's a parse error.
|
||||||
(receive (v st) (p:parse (rx) str)
|
(define regex
|
||||||
(if (p:success? st)
|
(let ((patched #f))
|
||||||
(compile-rx v)
|
(lambda (str)
|
||||||
#f))))
|
(unless patched
|
||||||
|
(set! patched #t)
|
||||||
|
;; rx := simple-rx ("|" simple-rx)*
|
||||||
|
(rx 'hotpatch
|
||||||
|
(p:lift2 (lambda (r rs)
|
||||||
|
(fold-left alt r rs))
|
||||||
|
simple-rx
|
||||||
|
(p:many* (p:>> (p:lit "|") simple-rx)))))
|
||||||
|
|
||||||
|
(receive (v st) (p:parse rx str)
|
||||||
|
(if (p:success? st)
|
||||||
|
(compile-rx v)
|
||||||
|
#f))))))
|
||||||
|
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user