Replace several define-macro' uses with
syntax-rules' and `syntax-case'
This commit is contained in:
parent
1c0d0f2a85
commit
2649cadc43
4 changed files with 92 additions and 100 deletions
|
@ -56,9 +56,11 @@
|
||||||
(output git-condition-output)
|
(output git-condition-output)
|
||||||
(status git-condition-status))
|
(status git-condition-status))
|
||||||
|
|
||||||
(define-macro (false-if-git-error . body)
|
(define-syntax false-if-git-error
|
||||||
`(,guard (c ((,git-condition? c) #f))
|
(syntax-rules ()
|
||||||
,@body))
|
((_ body0 body ...)
|
||||||
|
(guard (c ((git-condition? c) #f))
|
||||||
|
body0 body ...))))
|
||||||
|
|
||||||
;;;
|
;;;
|
||||||
;;; running git
|
;;; running git
|
||||||
|
|
|
@ -128,68 +128,80 @@
|
||||||
#f)
|
#f)
|
||||||
#f)))))
|
#f)))))
|
||||||
|
|
||||||
(define-macro (let-request request bindings . body)
|
(define-syntax let-request
|
||||||
(let ((request-var (gensym)))
|
(lambda (stx)
|
||||||
(define (make-binding b)
|
(define (make-binding b)
|
||||||
(cond
|
(syntax-case b ()
|
||||||
((symbol? b) `(,b (,rref ,request-var ',b)))
|
((id option ...)
|
||||||
((list? b) `(,(car b) (,rref ,request-var ',(car b) ,@(cdr b))))
|
(identifier? #'id)
|
||||||
(else (error "what" b))))
|
#'(id (rref request-var 'id option ...)))
|
||||||
`(let ((,request-var ,request))
|
(id
|
||||||
(let (,@(map make-binding bindings))
|
(identifier? #'id)
|
||||||
,@body))))
|
#'(id (rref request-var 'id)))))
|
||||||
|
(syntax-case stx ()
|
||||||
|
((_ request (binding ...) body ...)
|
||||||
|
(with-syntax (((binding ...) (map make-binding #'(binding ...))))
|
||||||
|
#`(let ((request-var request))
|
||||||
|
(let (binding ...)
|
||||||
|
body ...)))))))
|
||||||
|
|
||||||
(define-macro (path-proc-case path . clauses)
|
(define-syntax path-proc-case
|
||||||
(let ((path-var (gensym)))
|
(lambda (stx)
|
||||||
(define (optional-argument? arg)
|
(define (optional-argument? arg)
|
||||||
(eqv? (string-ref arg (1- (string-length arg))) #\?))
|
(eqv? (string-ref arg (- (string-length arg) 1)) #\?))
|
||||||
(define (required-argument? arg)
|
(define (required-argument? arg)
|
||||||
(eqv? (string-ref arg (1- (string-length arg))) #\!))
|
(eqv? (string-ref arg (- (string-length arg) 1)) #\!))
|
||||||
(define (output-argument? arg)
|
(define (output-argument? arg)
|
||||||
(or (optional-argument? arg) (required-argument? arg)))
|
(or (optional-argument? arg) (required-argument? arg)))
|
||||||
(define (process-clause clause)
|
(define (process-clause clause)
|
||||||
(or (list-has-length? clause 2) (error "foo"))
|
(syntax-case clause (else)
|
||||||
(if (eq? (car clause) 'else)
|
((else expr ...) clause)
|
||||||
clause
|
(((p ...) proc)
|
||||||
(let ((pat (map symbol->string (car clause)))
|
(let ((pat (map (lambda (p)
|
||||||
(proc (cadr clause)))
|
(symbol->string (syntax->datum p)))
|
||||||
(cond
|
#'(p ...))))
|
||||||
((find-tail output-argument? pat)
|
(cond
|
||||||
=> (lambda (tail)
|
((find-tail output-argument? pat)
|
||||||
(define test
|
=> (lambda (tail)
|
||||||
(let* ((npat (length pat))
|
(let* ((req (find-tail required-argument? tail))
|
||||||
(ntail (length tail))
|
(opt (find-tail optional-argument? tail))
|
||||||
(req (find-tail required-argument? tail))
|
(npat (length pat))
|
||||||
(opt (find-tail optional-argument? tail))
|
(ntail (length tail))
|
||||||
(nopt (if opt (length opt) 0))
|
(nopt (if opt (length opt) 0))
|
||||||
(nreq (if req (- (length req) nopt) 0)))
|
(nreq (if req (- (length req) nopt) 0)))
|
||||||
(lambda (path)
|
#`((let ((pathtail (list-head-match '#,pat
|
||||||
(let ((pathtail (list-head-match pat path (- npat ntail))))
|
path-var
|
||||||
;(pk pat npat ntail req opt nopt nreq path pathtail)
|
(- #,npat #,ntail))))
|
||||||
(if (and pathtail (>= (length pathtail) nreq)
|
;;(pk pat npat ntail req opt nopt nreq path pathtail)
|
||||||
(<= (length pathtail) (+ nreq nopt)))
|
(if (and pathtail (>= (length pathtail) #,nreq)
|
||||||
(append pathtail
|
(<= (length pathtail) (+ #,nreq #,nopt)))
|
||||||
(make-list (- (+ nreq nopt) (length pathtail)) #f))
|
(append
|
||||||
#f)))))
|
pathtail
|
||||||
`((,test ,path-var)
|
(make-list (- (+ #,nreq #,nopt) (length pathtail)) #f))
|
||||||
=> (lambda (outargs)
|
#f))
|
||||||
(lambda args
|
=> (lambda (outargs)
|
||||||
(apply ,proc (append args outargs)))))))
|
(lambda args
|
||||||
(else
|
(apply proc (append args outargs))))))))
|
||||||
`((equal? ,path-var ',pat) ,proc))))))
|
(else
|
||||||
`(let ((,path-var ,path))
|
#`((equal? path-var '#,pat) proc)))))))
|
||||||
(cond ,@(map process-clause clauses)))))
|
(syntax-case stx ()
|
||||||
|
((_ path clause ...)
|
||||||
|
(with-syntax (((cond-clause ...) (map process-clause #'(clause ...))))
|
||||||
|
#`(let ((path-var path))
|
||||||
|
(cond cond-clause ...)))))))
|
||||||
|
|
||||||
(define (rcons*-fold request . keys-and-procs)
|
(define (rcons*-fold request . keys-and-procs)
|
||||||
(foldn (lambda (request k proc)
|
(foldn (lambda (request k proc)
|
||||||
(rcons k (proc request) request))
|
(rcons k (proc request) request))
|
||||||
2 request keys-and-procs))
|
2 request keys-and-procs))
|
||||||
|
|
||||||
(define-macro (request-path-case request . clauses)
|
(define-syntax request-path-case
|
||||||
`(,path-proc-case
|
(syntax-rules ()
|
||||||
(,let-request ,request (method path)
|
((_ request clause ...)
|
||||||
(cons method path))
|
(path-proc-case
|
||||||
,@clauses))
|
(let-request request (method path)
|
||||||
|
(cons method path))
|
||||||
|
clause ...))))
|
||||||
|
|
||||||
(define (request-server-name request)
|
(define (request-server-name request)
|
||||||
(let ((headers (rref request 'headers)))
|
(let ((headers (rref request 'headers)))
|
||||||
|
|
|
@ -36,7 +36,7 @@
|
||||||
date-increment date-comparator date-before? date-after? compose1
|
date-increment date-comparator date-before? date-after? compose1
|
||||||
rfc822-date->timestamp timestamp->rfc822-date timestamp->atom-date
|
rfc822-date->timestamp timestamp->rfc822-date timestamp->atom-date
|
||||||
timestamp->date string-split/trimming
|
timestamp->date string-split/trimming
|
||||||
list-intersperse with-backtrace with-time-debugging define-memoized))
|
list-intersperse with-backtrace with-time-debugging))
|
||||||
|
|
||||||
(define (emailish? x)
|
(define (emailish? x)
|
||||||
(match-bind "^([a-zA-Z0-9._+-]+)@([a-zA-Z0-9-]+\\.)+[a-zA-Z]+$"
|
(match-bind "^([a-zA-Z0-9._+-]+)@([a-zA-Z0-9-]+\\.)+[a-zA-Z]+$"
|
||||||
|
@ -88,36 +88,26 @@
|
||||||
(string-join (cons (passwd:dir (getpwnam user)) (cdr parts)) "/"))
|
(string-join (cons (passwd:dir (getpwnam user)) (cdr parts)) "/"))
|
||||||
path)))
|
path)))
|
||||||
|
|
||||||
(define-macro (match-lines string pattern bindings expr)
|
(define-syntax match-lines
|
||||||
(let ((line (gensym)) (seed (gensym)))
|
(syntax-rules ()
|
||||||
`(,fold
|
((_ string pattern bindings expr)
|
||||||
(lambda (,line ,seed)
|
(let ((rx (irregex pattern)))
|
||||||
(,match-bind ,pattern ,line ,bindings
|
(fold
|
||||||
(cons ,expr ,seed)
|
(lambda (line seed)
|
||||||
,seed))
|
(match-bind rx line bindings
|
||||||
'() (string-split ,string #\newline))))
|
(cons expr seed)
|
||||||
|
seed))
|
||||||
;; clause := ((pat args) body...)
|
'() (string-split string #\newline))))))
|
||||||
(define-macro (match-case string . clauses)
|
|
||||||
(let ((str (gensym)))
|
|
||||||
`(let ((,str ,string))
|
|
||||||
,(let lp ((in clauses))
|
|
||||||
(let ((clause (car in)))
|
|
||||||
(if (eq? (car clause) 'else)
|
|
||||||
`(begin ,@(cdr clause))
|
|
||||||
`(match-bind ,(caar clause) ,str ,(cadar clause)
|
|
||||||
(begin ,@(cdr clause))
|
|
||||||
,(if (null? (cdr in))
|
|
||||||
#f
|
|
||||||
(lp (cdr in))))))))))
|
|
||||||
|
|
||||||
(define (dbg fmt . args)
|
(define (dbg fmt . args)
|
||||||
(apply format (current-error-port) fmt args))
|
(apply format (current-error-port) fmt args))
|
||||||
|
|
||||||
(define-macro (unwind-protect form . cleanups)
|
(define-syntax unwind-protect
|
||||||
`(dynamic-wind (lambda () #t)
|
(syntax-rules ()
|
||||||
(lambda () ,form)
|
((_ form cleanup0 cleanups ...)
|
||||||
(lambda () ,@cleanups)))
|
(dynamic-wind (lambda () #t)
|
||||||
|
(lambda () form)
|
||||||
|
(lambda () cleanup0 cleanups ...)))))
|
||||||
|
|
||||||
(define (dsu-sort list key less)
|
(define (dsu-sort list key less)
|
||||||
(map cdr
|
(map cdr
|
||||||
|
@ -196,8 +186,10 @@
|
||||||
(fluid-set! the-last-stack (make-stack #t 2 0))
|
(fluid-set! the-last-stack (make-stack #t 2 0))
|
||||||
(apply throw args)))))
|
(apply throw args)))))
|
||||||
|
|
||||||
(define-macro (with-backtrace . forms)
|
(define-syntax with-backtrace
|
||||||
`(,with-backtrace* (lambda () ,@forms)))
|
(syntax-rules ()
|
||||||
|
((_ form0 forms ...)
|
||||||
|
(with-backtrace* (lambda () form0 forms ...)))))
|
||||||
|
|
||||||
(define (gettimeofday-diff prev)
|
(define (gettimeofday-diff prev)
|
||||||
(let ((now (gettimeofday)))
|
(let ((now (gettimeofday)))
|
||||||
|
@ -211,8 +203,10 @@
|
||||||
(proc)
|
(proc)
|
||||||
(pk 'stop-clock (gettimeofday-diff start)))))
|
(pk 'stop-clock (gettimeofday-diff start)))))
|
||||||
|
|
||||||
(define-macro (with-time-debugging . forms)
|
(define-syntax with-time-debugging
|
||||||
`(,with-time-debugging* (lambda () ,@forms)))
|
(syntax-rules ()
|
||||||
|
((_ form0 forms ...)
|
||||||
|
(with-time-debugging* (lambda () form0 forms ...)))))
|
||||||
|
|
||||||
(define (memoize1 proc)
|
(define (memoize1 proc)
|
||||||
(let ((old-args #f) (cache #f) (proc proc))
|
(let ((old-args #f) (cache #f) (proc proc))
|
||||||
|
@ -224,11 +218,6 @@
|
||||||
(set! cache val)
|
(set! cache val)
|
||||||
val)))))
|
val)))))
|
||||||
|
|
||||||
(define-macro (define-memoized form . body)
|
|
||||||
`(begin
|
|
||||||
(define ,form ,@body)
|
|
||||||
(set! ,(car form) (,memoize1 ,(car form)))))
|
|
||||||
|
|
||||||
(define (write-hash h)
|
(define (write-hash h)
|
||||||
(write (hash-fold acons '() h)))
|
(write (hash-fold acons '() h)))
|
||||||
|
|
||||||
|
|
|
@ -32,7 +32,7 @@
|
||||||
#:use-module (tekuti template)
|
#:use-module (tekuti template)
|
||||||
#:use-module (tekuti page)
|
#:use-module (tekuti page)
|
||||||
#:use-module (srfi srfi-1)
|
#:use-module (srfi srfi-1)
|
||||||
#:export (let-headers header-ref
|
#:export (header-ref
|
||||||
handle-request))
|
handle-request))
|
||||||
|
|
||||||
(define *status-names*
|
(define *status-names*
|
||||||
|
@ -53,17 +53,6 @@
|
||||||
"<!DOCTYPE html PUBLIC \"-//W3C//DTD XHTML 1.0 Strict//EN\" "
|
"<!DOCTYPE html PUBLIC \"-//W3C//DTD XHTML 1.0 Strict//EN\" "
|
||||||
"\"http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd\">\n"))
|
"\"http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd\">\n"))
|
||||||
|
|
||||||
;;; useless macro
|
|
||||||
(define-macro (let-headers headers bindings . body)
|
|
||||||
(let ((headers-var (gensym)))
|
|
||||||
`(let ((,headers-var ,headers))
|
|
||||||
(let (,@(map (lambda (binding)
|
|
||||||
`(,(car binding)
|
|
||||||
(or (assoc-ref ,headers-var ,(cadr binding))
|
|
||||||
(error "Missing header:" ,(cadr binding)))))
|
|
||||||
bindings))
|
|
||||||
,@body))))
|
|
||||||
|
|
||||||
(define (make-output request)
|
(define (make-output request)
|
||||||
(lambda (port)
|
(lambda (port)
|
||||||
(let ((sxml (or (rref request 'sxml #f)
|
(let ((sxml (or (rref request 'sxml #f)
|
||||||
|
|
Loading…
Reference in a new issue