chibi-scheme/lib/chibi/doc.scm
2021-05-14 13:49:07 +09:00

1144 lines
44 KiB
Scheme

;;> A library for generating SXML docs from Scribble, directly or
;;> extracted from literate docs.
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;; utils
(define (write-to-string x)
(call-with-output-string (lambda (out) (write x out))))
(define (string-concatenate-reverse ls)
(string-concatenate (reverse ls)))
(define (string-strip str . o)
(let ((bad (if (pair? o) (car o) " \t\n")))
(call-with-output-string
(lambda (out)
(call-with-input-string str
(lambda (in)
(let lp ()
(let ((ch (read-char in)))
(cond
((not (eof-object? ch))
(if (not (string-find? bad ch))
(write-char ch out))
(lp)))))))))))
(define (string-first-token str sep)
(let ((len (string-length str)))
(let lp ((i 0))
(cond ((= i len) str)
((not (string-find? sep (string-ref str i))) (lp (+ i 1)))
(else
(let lp ((j (+ i 1)))
(cond ((= j len) "")
((string-find? sep (string-ref str j)) (lp (+ j 1)))
(else
(let lp ((k (+ j 1)))
(cond
((or (= k len) (string-find? sep (string-ref str k)))
(substring str j k))
(else
(lp (+ k 1)))))))))))))
(define (intersperse ls x)
(if (or (null? ls) (null? (cdr ls)))
ls
(let lp ((ls (cdr ls)) (res (list (car ls))))
(let ((res (cons (car ls) (cons x res))))
(if (null? (cdr ls))
(reverse res)
(lp (cdr ls) res))))))
(define (normalize-sxml x)
(cond
((pair? x)
(let lp ((ls x) (res '()))
(cond ((null? ls)
(string-concatenate-reverse res))
((string? (car ls))
(lp (cdr ls) (cons (car ls) res)))
((pair? res)
(cons (string-concatenate-reverse res)
(cons (car ls) (normalize-sxml (cdr ls)))))
(else
(cons (car ls) (normalize-sxml (cdr ls)))))))
(else x)))
(define (map-sxml proc x)
(if (pair? x)
(cons (map-sxml proc (car x)) (map-sxml proc (cdr x)))
(proc x)))
(define (sxml-body x)
(cond ((not (and (pair? x) (pair? (cdr x)))) '())
((and (pair? (cadr x)) (eq? '@ (car (cadr x)))) (cddr x))
(else (cdr x))))
(define (sxml->sexp-list x)
(call-with-input-string (sxml-strip x) port->sexp-list))
;;> Replace ansi escape sequences in a \var{str} with the corresponding sxml.
(define (ansi->sxml str)
;; TODO: ick
(let ((start (string-cursor-start str))
(end (string-cursor-end str)))
(let lp1 ((from start)
(to start)
(res '()))
(define (lookup str)
(case (string->number str)
((0) '/) ((1) 'b) ((3) 'i) ((4) 'u) ((9) 's)
((22) '/b) ((23) '/i) ((24) '/u) ((29) '/s)
((30) 'black) ((31) 'red) ((32) 'green) ((33) 'yellow)
((34) 'blue) ((35) 'magenta) ((36) 'cyan) ((37) 'white)
((39) '/color)
(else #f)))
(define (collect from to res)
(if (string-cursor<? from to)
(cons (substring-cursor str from to) res)
res))
(define (finish)
(let ((ls (reverse (collect from to res))))
(if (and (= 1 (length ls)) (string? (car ls)))
(car ls)
(let lp1 ((ls ls) (cur '()) (res '()))
(define (close to)
(let lp2 ((ls cur) (tmp '()))
(cond
((null? ls)
(list '() `(,@(reverse tmp) ,@res)))
((eq? to (car ls))
(list (cdr ls) `((,to ,@tmp) ,@res)))
((and (eq? to 'color) (memq (car ls) '(b i u s)))
;; color close came to an open non-color
;; back off and leave this open
(let ((s `(,(car ls) ,@(take-while string? tmp)))
(tmp (drop-while string? tmp)))
(list `(,@(reverse tmp) ,@(reverse s)) res)))
((symbol? (car ls))
(lp2 (cdr ls) `((,(car ls) ,@(reverse tmp)))))
((and (pair? (car ls)) (eq? 'color to))
(lp2 (cdr ls) `((,@(car ls) ,@(reverse tmp)))))
((pair? (car ls))
(lp2 (cdr ls) `(,(car ls) ,@(reverse tmp))))
(else
(lp2 (cdr ls) `(,(car ls) ,@tmp))))))
(cond
((null? ls)
`(span ,@(reverse (cadr (close #f)))))
((and (string? (car ls)) (pair? cur))
(lp1 (cdr ls) (cons (car ls) cur) res))
((string? (car ls))
(lp1 (cdr ls) cur (cons (car ls) res)))
(else
(case (car ls)
((b i u s) (lp1 (cdr ls) (cons (car ls) cur) res))
((/b) (apply lp1 (cdr ls) (close 'b)))
((/i) (apply lp1 (cdr ls) (close 'i)))
((/u) (apply lp1 (cdr ls) (close 'u)))
((/s) (apply lp1 (cdr ls) (close 's)))
((/) (apply lp1 (cdr ls) (close 'all)))
((/color) (apply lp1 (cdr ls) (close 'color)))
(else
(let ((style (string-append "color:"
(symbol->string (car ls)))))
(lp1 (cdr ls)
(cons `(span (@ (style . ,style))) cur)
res))))))))))
(if (string-cursor>=? to end)
(finish)
(let ((c (string-cursor-ref str to))
(sc2 (string-cursor-next str to)))
(if (and (= 27 (char->integer c))
(string-cursor<? sc2 end)
(eqv? #\[ (string-cursor-ref str sc2)))
(let ((sc3 (string-cursor-next str sc2)))
(let lp2 ((sc4 sc3))
(if (string-cursor>=? sc4 end)
(finish)
(let ((c2 (string-cursor-ref str sc4))
(sc5 (string-cursor-next str sc4)))
(if (eqv? #\m c2)
(let ((code (lookup
(substring-cursor str sc3 sc4)))
(res (collect from to res)))
(lp1 sc5 sc5 (if code (cons code res) res)))
(lp2 sc5))))))
(lp1 from sc2 res)))))))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;> Extract the literate Scribble docs for module \var{mod-name} and
;;> print them to \var{out}, rendered with \var{render} which defaults
;;> to \scheme{sxml-display-as-text}.
(define (print-module-docs mod-name . o)
(let ((out (if (pair? o) (car o) (current-output-port)))
(render (or (and (pair? o) (pair? (cdr o)) (cadr o))
sxml-display-as-text))
(unexpanded?
(and (pair? o) (pair? (cdr o)) (pair? (cddr o)) (car (cddr o)))))
(render
((if unexpanded? (lambda (sxml env) (fixup-docs sxml)) generate-docs)
`((title ,(write-to-string mod-name))
,@(extract-module-docs mod-name #f))
(make-module-doc-env mod-name))
out)))
;;> Extract the literate Scribble docs for just the binding for
;;> \var{var} in module \var{mod-name}, and print them as in
;;> \scheme{print-module-docs}.
(define (print-module-binding-docs mod-name var . o)
(let ((out (if (pair? o) (car o) (current-output-port)))
(render (or (and (pair? o) (pair? (cdr o)) (cadr o))
sxml-display-as-text)))
(render
(generate-docs
(extract-module-docs mod-name #t (list var))
(make-module-doc-env mod-name))
out)))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;> Extract the literate Scribble docs for \var{proc} which should be
;;> a procedure and return them as sxml.
(define (procedure-docs proc)
(let ((mod (and (procedure? proc) (containing-module proc))))
(and mod
(generate-docs
(extract-module-docs (car mod) #t (list (procedure-name proc)))
(make-module-doc-env (car mod))))))
;;> Extract the literate Scribble docs for \var{proc} which should be
;;> a procedure and render them as in \scheme{print-module-docs}.
(define (print-procedure-docs proc . o)
(let ((out (if (pair? o) (car o) (current-output-port)))
(render (or (and (pair? o) (pair? (cdr o)) (cadr o))
sxml-display-as-text))
(docs (procedure-docs proc)))
(if docs (render docs out))))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;; doc environments
(define (env-ref env name . o)
(cond ((assq name (car env)) => cdr)
((pair? o) (car o))
(else #f)))
(define (env-set! env name value)
(cond ((assq name (car env)) => (lambda (cell) (set-cdr! cell value)))
(else (set-car! env (cons (cons name value) (car env))))))
(define (env-extend env vars vals)
(list (append (map cons vars vals) (car env))))
;;> Return a new document environment suitable for passing to
;;> \scheme{expand-docs}, with default rules for sections, code
;;> blocks, procedure and macro signatures, etc.
(define (make-default-doc-env)
`(((title . ,(expand-section 'h1))
(section . ,(expand-section 'h2))
(subsection . ,(expand-section 'h3))
(subsubsection . ,(expand-section 'h4))
(subsubsubsection . ,(expand-section 'h5))
(procedure . ,expand-procedure)
(macro . ,expand-macro)
(centered . center)
(smaller . small)
(larger . large)
(bold . b)
(italic . i)
(emph . em)
(subscript . sub)
(superscript . sup)
(itemlist . ul)
(item . li)
(var . code)
(cfun . code)
(cmacro . code)
(ctype . code)
(url . ,expand-url)
(hyperlink . ,expand-hyperlink)
(rawcode . code)
(pre . pre)
(bibitem . ,(lambda (x env) '())) ;; TODO: bibtex
(code . ,expand-code)
(codeblock . ,expand-codeblock)
(ccode
. ,(lambda (x env)
(expand-code `(,(car x) language: c ,@(cdr x)) env)))
(ccodeblock
. ,(lambda (x env)
(expand-codeblock `(,(car x) language: c ,@(cdr x)) env)))
(scheme
. ,(lambda (x env)
(expand-code `(,(car x) language: scheme ,@(cdr x)) env)))
(schemeblock
. ,(lambda (x env)
(expand-codeblock `(,(car x) language: scheme ,@(cdr x)) env)))
(command . ,expand-command)
(author . ,expand-author)
(margin-note . ,expand-note)
(example . ,expand-example)
(example-import . ,expand-example-import)
(example-import-only . ,expand-example-import-only)
)))
;;> Return a new document environment as in
;;> \scheme{make-default-doc-env}, with an \scheme{example-env}
;;> binding mapped to an environment importing \scheme{(scheme base)}
;;> and the module \var{mod-name}. This binding is used when
;;> expanding examples in the docs.
(define (make-module-doc-env mod-name)
(env-extend (make-default-doc-env)
'(example-env)
(list (delay (environment '(scheme small)
'(only (chibi) import)
mod-name)))))
(define (section-name tag name)
(string-strip
(call-with-output-string
(lambda (out)
(display tag out)
(write-char #\_ out)
(display name out)))))
(define (expand-section tag)
(lambda (sxml env)
(if (null? (cdr sxml))
(error "section must not be empty" sxml)
(let* ((name (and (eq? 'tag: (cadr sxml))
(pair? (cddr sxml))
(sxml-strip (car (cddr sxml)))))
(body (map (lambda (x) (expand-docs x env))
(if name (cdr (cddr sxml)) (cdr sxml))))
(name (or name (sxml-strip (cons tag body)))))
`(div (a (@ (name . ,(section-name tag name)))) (,tag ,@body))))))
(define (expand-url sxml env)
(if (not (= 2 (length sxml)))
(error "url expects one argument" sxml)
(let ((url (expand-docs (cadr sxml) env)))
`(a (@ (href . ,url)) ,url))))
(define (expand-hyperlink sxml env)
(if (not (>= (length sxml) 3))
(error "hyperlink expects at least two arguments" sxml)
(let ((url (expand-docs (cadr sxml) env)))
`(a (@ (href . ,url))
,(map (lambda (x) (expand-docs x env)) (cddr sxml))))))
(define (expand-note sxml env)
`(div (@ (id . "notes"))
,@(map (lambda (x) (expand-docs x env)) (cdr sxml))))
(define (expand-author sxml env)
`(div (@ (id . "notes"))
,@(map (lambda (x) (expand-docs x env)) (cdr sxml))
(br)
,(seconds->string (current-seconds))))
(define (expand-code sxml env)
(let* ((hl (if (and (pair? (cdr sxml)) (eq? 'language: (cadr sxml)))
(highlighter-for (car (cddr sxml)))
highlight))
(body (if (and (pair? (cdr sxml)) (eq? 'language: (cadr sxml)))
(cdr (cddr sxml))
(cdr sxml))))
`(code ,@(map-sxml (lambda (x) (if (string? x) (hl x) x))
(normalize-sxml
(map (lambda (x) (expand-docs x env)) body))))))
(define (expand-codeblock sxml env)
`(pre ,(expand-code sxml env)))
(define (expand-example x env)
(let ((expr `(begin ,@(sxml->sexp-list x)))
(example-env
(force (or (env-ref env 'example-env) (current-environment)))))
`(div
,(expand-codeblock `(,(car x) language: scheme ,@(cdr x)) env)
,(let* ((res-out (open-output-string))
(tmp-out (open-output-string))
(tmp-err (open-output-string))
(res (parameterize ((current-output-port tmp-out)
(current-error-port tmp-err))
(protect (exn (#t (print-exception exn tmp-err)))
(eval expr example-env)))))
(display "=> " res-out)
(write res res-out)
(let ((res-str (get-output-string res-out))
(out-str (get-output-string tmp-out))
(err-str (get-output-string tmp-err)))
`(,@(if (string-null? out-str)
'()
`((div (@ (class . "output")) (pre ,(ansi->sxml out-str)))))
,@(if (string-null? err-str)
'()
`((div (@ (class . "error")) (pre ,(ansi->sxml err-str)))))
,@(if (and (or (not (string-null? err-str))
(not (string-null? out-str)))
(eq? res (if #f #f)))
'()
`((div (@ (class . "result")) (code ,res-str))))))))))
(define (expand-example-import x env)
(eval `(import ,@(cdr x))
(force (or (env-ref env 'example-env) (current-environment))))
"")
(define (expand-example-import-only x env)
(env-set! env 'example-env (apply environment (cdr x)))
"")
(define (expand-command sxml env)
`(pre (@ (class . "command"))
(code ,@(map (lambda (x) (expand-docs x env)) (cdr sxml)))))
(define (expand-tagged tag ls env)
(cons tag (map (lambda (x) (expand-docs x env)) ls)))
;;> Given the sxml document \var{sxml}, expands macros defined in the
;;> document environment \var{env} into standard html tags.
(define (expand-docs sxml env)
(cond
((pair? sxml)
(cond
((symbol? (car sxml))
(let ((op (env-ref env (car sxml))))
(cond
((procedure? op)
(op sxml env))
((symbol? op)
(expand-tagged op (cdr sxml) env))
(else
(expand-tagged (car sxml) (cdr sxml) env)))))
(else
(map (lambda (x) (expand-docs x env)) sxml))))
(else
sxml)))
(define (expand-procedure sxml env)
((expand-section 'h4) `(,(car sxml) (rawcode ,@(cdr sxml))) env))
(define (expand-macro sxml env)
(expand-procedure sxml env))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;; adjustments for html
(define header-index
(let* ((headers '(h1 h2 h3 h4 h5 h6))
(len (length headers)))
(lambda (h) (- len (length (memq h headers))))))
;; return a list of (index . link-to-header) for all headers
(define (extract-contents x)
(match x
(('div ('a ('@ ('name . name)) . _)
((and h (or 'h1 'h2 'h3 'h4 'h5 'h6)) . section))
(let* ((raw-text (sxml-strip (cons h section)))
(text (if (string-prefix? "(" raw-text)
(let ((end (string-find
raw-text
(lambda (ch)
(or (char-whitespace? ch)
(eqv? ch #\)))))))
(substring raw-text 1 end))
raw-text)))
`((,(header-index h)
(a (@ (href . ,(string-append "#" name)))
,text)))))
((a . b)
(append (extract-contents a) (extract-contents b)))
(else
'())))
;; nest the (index . link-to-header)s into ol
(define (get-contents x)
(if (null? x)
'()
(let lp ((ls (cdr x))
(depth (caar x))
(parent (cadr (car x)))
(kids '())
(res '()))
(define (collect)
(cons `(li ,parent ,(get-contents (reverse kids))) res))
;; take a span of all sub-headers, recurse and repeat on next span
(cond
((null? ls)
`(ol ,@(reverse (collect))))
((> (caar ls) depth)
(lp (cdr ls) depth parent (cons (car ls) kids) res))
(else
(lp (cdr ls) (caar ls) (cadr (car ls)) '() (collect)))))))
(define (fix-header x)
`((!DOCTYPE html)
(html (head ,@(cond ((assq 'title x) => (lambda (x) (list x)))
(else '()))
"\n"
(meta (@ (charset . "UTF-8")))
(style (@ (type . "text/css"))
"
body {color: #000; background-color: #FFFFF8;}
div#menu {font-size: smaller; position: absolute; top: 50px; left: 0; width: 250px; height: 100%}
div#menu a:link {text-decoration: none}
div#main {font-size: large; position: absolute; top: 0; left: 260px; max-width: 590px; height: 100%}
div#notes {position: relative; top: 2em; left: 620px; width: 200px; height: 0px; font-size: smaller;}
div#footer {padding-bottom: 50px}
div#menu ol {list-style-position:inside; padding-left: 5px; margin-left: 5px}
div#menu ol ol {list-style: lower-alpha; padding-left: 15px; margin-left: 15px}
div#menu ol ol ol {list-style: decimal; padding-left: 5px; margin-left: 5px}
h2 { color: #888888; border-top: 3px solid #4588ba; }
h3 { color: #666666; border-top: 2px solid #4588ba; }
h4 { color: #222288; border-top: 1px solid #4588ba; }
.result { color: #000; background-color: #FFEADF; width: 100%; padding: 3px}
.output { color: #000; background-color: beige; width: 100%; padding: 3px}
.error { color: #000; background-color: #F0B0B0; width: 100%; padding: 3px}
.command { color: #000; background-color: #FFEADF; width: 100%; padding: 5px}
"
,(highlight-style))
"\n")
(body
(div (@ (id . "menu"))
,(let ((contents (get-contents (extract-contents x))))
(match contents
;; flatten if we have only a single heading
(('ol (li y sections ...))
sections)
(else contents))))
(div (@ (id . "main"))
,@(map (lambda (x)
(if (and (pair? x) (eq? 'title (car x)))
(cons 'h1 (cdr x))
x))
x)
(div (@ (id . "footer"))))))))
(define (fix-paragraphs x)
(let lp ((ls x) (p '()) (res '()))
(define (collect)
(if (pair? p) (cons `(p ,@(reverse p)) res) res))
(define (inline? x)
(or (string? x)
(and (pair? x)
(or (string? (car x))
(memq (car x)
'(a b i u span code small large sub sup em))))))
(define (enclosing? x)
(and (pair? x) (symbol? (car x))
(memq (car x) '(div body))))
(cond
((null? ls)
(reverse (collect)))
((equal? "\n" (car ls))
(if (and (pair? p) (equal? "\n" (car p)))
(let lp2 ((ls (cdr ls)))
(if (and (pair? ls) (equal? "\n" (car ls)))
(lp2 (cdr ls))
(lp ls '() (collect))))
(lp (cdr ls) (cons (car ls) p) res)))
((inline? (car ls))
(lp (cdr ls) (cons (car ls) p) res))
((enclosing? (car ls))
(lp (cdr ls) '() (cons (car ls) (collect))))
(else
(lp (cdr ls) '() (cons (car ls) (collect)))))))
(define (fix-begins x)
x)
;;> Resolves paragraphs and adds a header to convert \var{sxml} to a
;;> standalone document renderable in html.
(define (fixup-docs sxml)
(fix-header (fix-paragraphs (fix-begins sxml))))
;;> Composes \scheme{expand-docs} and \scheme{fixup-docs}.
(define (generate-docs sxml . o)
(let ((env (if (pair? o) (car o) (make-default-doc-env))))
(fixup-docs (expand-docs sxml env))))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;; extraction
(define (skip-horizontal-whitespace in)
(cond ((memv (peek-char in) '(#\space #\tab))
(read-char in)
(skip-horizontal-whitespace in))))
(define (external-clause? x)
(not (and (pair? (cdr x)) (pair? (cadr x)) (string? (car (cadr x))))))
(define (contains? tree x)
(or (eq? tree x)
(and (pair? tree)
(or (contains? (car tree) x)
(contains? (cdr tree) x)))))
(define (form-defined-name form)
(match form
(('define (name . x) . y) name)
(((or 'define 'define-syntax) name . x)
name)
(((or 'define-c 'define-c-const)
t (name . x) . y)
name)
(((or 'define-c 'define-c-const)
t name . x)
name)
(else #f)))
;; Try to determine the names of optional parameters checking common
;; patterns.
(define (get-optionals-signature ls body)
(let lp ((ls ls) (pre '()))
(cond
((pair? ls) (lp (cdr ls) (cons (car ls) pre)))
((null? ls) (reverse pre))
(else
(let* ((o ls)
(o? (lambda (x) (eq? x o))))
(let extract ((x body)
(vars '())
(i 0))
(match x
((('define x val) . rest)
(if (contains? val o)
(extract #f vars i)
(extract rest vars i)))
((((or 'let 'let* 'letrec 'letrec*) (y ...) . body) . rest)
(let ((ordered? (memq (car x) '(let* letrec*))))
(let lp ((ls y) (vars vars) (j i))
(cond
((pair? ls)
(match (car ls)
;; handle rebinding o
(((? o?) ('if ('pair? (? o?)) ('cdr (? o?)) default))
(lp (cdr ls) vars (+ j 1)))
(((? o?) expr)
(extract #f vars i))
;; binding vars to o
((v ('if ('pair? (? o?)) ('car (? o?)) default))
(lp (cdr ls) (cons (cons v (if ordered? j i)) vars) j))
((v ('and ('pair? (? o?)) ('car (? o?))))
(lp (cdr ls) (cons (cons v (if ordered? j i)) vars) j))
((v ('or ('and ('pair? (? o?)) ('car (? o?))) default))
(lp (cdr ls) (cons (cons v (if ordered? j i)) vars) j))
((v ('if ('and ('pair? (? o?)) ('pair? ('cdr (? o?))))
('cadr (? o?))
default))
(lp (cdr ls)
(cons (cons v (+ 1 (if ordered? j i))) vars)
j))
((v ('and ('pair? (? o?)) ('pair? ('cdr (? o?)))
('cadr (? o?))))
(lp (cdr ls)
(cons (cons v (+ 1 (if ordered? j i))) vars)
j))
(else
(lp (cdr ls) vars j))))
(else
(extract body vars j))))))
((('let (? symbol?) (y ...) . body) . rest)
(extract `((let ,y . ,body) . ,rest) vars i))
((((or 'let-optionals 'let-optionals*) ls ((var default) ...)
. body) . rest)
(let lp ((ls var) (vars vars) (i i))
(cond
((pair? ls)
(lp (cdr ls) (cons (cons (car ls) i) vars) (+ i 1)))
(else
(extract body vars i)))))
(_
(let* ((opts (map car (sort vars < cdr)))
(rest-var? (contains? x o))
(tail (cond
((and (pair? opts) rest-var?)
(list (append opts o)))
(rest-var?
o)
((pair? opts)
(list opts))
(else
o))))
(append (reverse pre) tail))))))))))
(define (get-procedure-signature mod id proc)
(protect (exn (else '()))
(cond ((and mod (procedure? proc) (procedure-signature id mod))
=> (lambda (sig)
(list (cons (or id (procedure-name proc)) (cdr sig)))))
(else '()))))
(define (get-value-signature mod id proc name value)
(match value
(((or 'let 'let* 'letrec 'letrec*) vars body0 ... body)
(get-value-signature mod id proc name body))
(('lambda args . body)
(list (cons name (get-optionals-signature args body))))
((('lambda args body0 ... body) vals ...)
(get-value-signature mod id proc name body))
(('begin body0 ... body) (get-value-signature mod id proc name body))
(else (get-procedure-signature mod id proc))))
(define (get-signature mod id proc source form)
(match form
(('define (name args ...) . body)
(list (cons name args)))
(('define (name . args) . body)
(list (cons name (get-optionals-signature args body))))
(('define name value)
(get-value-signature mod id proc name value))
(('define-syntax name ('syntax-rules () (clause . body) ...))
;; TODO: smarter summary merging forms
(map (lambda (x) (cons name (cdr x)))
(filter external-clause? clause)))
(else
(cond
((procedure-analysis proc mod)
=> (lambda (lam) (list (cons (lambda-name lam) (lambda-params lam)))))
(else
(get-procedure-signature mod id proc))))))
(define (get-ffi-signatures form)
(match form
(('define-c ret-type (or (name _) name) (args ...))
(list (cons name
(map (lambda (x) (if (pair? x) (last x) x))
(remove (lambda (x)
(and (pair? x)
(memq (car x) '(value result))))
args)))))
(('define-c-const type (or (name _) name))
(list (list 'const: type name)))
(('cond-expand (test . clauses) . rest)
(append-map get-ffi-signatures clauses))
(((or 'define-c-struct 'define-c-class 'define-c-type) name . rest)
(let lp ((ls rest) (res '()))
(cond
((null? ls)
(reverse res))
((eq? 'predicate: (car ls))
(lp (cddr ls) (cons (list (cadr ls) 'obj) res)))
((eq? 'constructor: (car ls))
(lp (cddr ls)
(cons (if (pair? (cadr ls)) (cadr ls) (list (cadr ls))) res)))
((pair? (car ls))
(lp (cdr ls)
(append (if (pair? (cddr (cdar ls)))
(list (list (car (cddr (cdar ls))) name (caar ls)))
'())
(list (list (cadr (cdar ls)) name))
res)))
((symbol? (car ls))
(lp (cddr ls) res))
(else
(lp (cdr ls) res)))))
(else
'())))
(define section-number
(let ((sections '(section subsection subsubsection subsubsubsection)))
(lambda (x)
(cond ((memq x sections) => length)
((memq x '(procedure macro)) (section-number 'subsubsection))
(else 0)))))
(define (section>=? x n)
(and (pair? x)
(if (memq (car x) '(div))
(find (lambda (y) (section>=? y n)) (sxml-body x))
(>= (section-number (car x)) n))))
(define (extract-sxml tag x)
(and (pair? x)
(cond ((if (pair? tag) (memq (car x) tag) (eq? tag (car x))) x)
((memq (car x) '(div))
(any (lambda (y) (extract-sxml tag y)) (sxml-body x)))
(else #f))))
(define (section-describes? x name)
(let ((name (symbol->string name)))
(and (pair? x) (pair? (cdr x))
(let* ((str (sxml-strip (cadr x)))
(op (string-first-token str " \t\r\n()#")))
(or (string=? op name)
;; FIXME: hack for loop iterators
(and (string=? op "for")
(string-contains str (string-append "(" name " "))))))))
;; write a signature handling a trailing list as [optional] parameters
(define (write-signature sig)
(if (and (list? sig)
(> (length sig) 1)
(pair? (last sig))
(not (any pair? (drop-right sig 1))))
(call-with-output-string
(lambda (out)
(display "(" out)
(write (car sig) out)
(let lp ((ls (cdr sig)))
(cond
((pair? (car ls))
(display " [" out)
(write (caar ls) out)
(let lp ((ls (cdar ls)))
(cond
((pair? ls)
(display " " out)
(write (car ls) out)
(lp (cdr ls)))
((not (null? ls))
(display " . " out)
(write ls out))))
(display "])" out))
(else
(display " " out)
(write (car ls) out)
(lp (cdr ls)))))))
(write-to-string sig)))
(define (insert-signature orig-ls name sig)
(let ((sig (if (pair? sig) sig (and name (list name)))))
(cond
((not (pair? sig))
'())
(else
(let ((name
(cond
(name)
((not (pair? (car sig))) (car sig))
((eq? 'const: (caar sig)) (cadr (cdar sig)))
(else (caar sig)))))
(let lp ((ls orig-ls) (rev-pre '()))
(cond
((or (null? ls)
(section>=? (car ls) (section-number 'subsubsection)))
`(,@(reverse rev-pre)
,@(if (and (pair? ls)
(section-describes?
(extract-sxml
'(subsubsection procedure macro)
(car ls))
name))
'()
`((subsubsection
tag: ,(write-to-string name)
(rawcode
,@(if (and (pair? (car sig)) (eq? 'const: (caar sig)))
`((i ,(write-to-string (car (cdar sig))) ": ")
,(write-to-string (cadr (cdar sig))))
(intersperse (map write-signature sig) '(br)))))))
,@ls))
(else
(lp (cdr ls) (cons (car ls) rev-pre))))))))))
;;> Extract inline Scribble documentation (with the ;;> prefix) from
;;> the source file \var{file}, associating any signatures from the
;;> provided defs when available and not overridden in the docs.
(define (extract-file-docs mod file all-defs strict? . o)
;; extract (<file> . <line>) macro source or
;; (<offset> <file . <line>) procedure source or
;; ((<offset> <file . <line>) ...) bytecode sources
(define (source-line source)
(and (pair? source)
(cond
((string? (car source))
(and (equal? file (car source))
(number? (cdr source))
(cdr source)))
((pair? (car source))
(source-line (car source)))
(else
(and (number? (car source))
(pair? (cdr source))
(equal? file (cadr source))
(cddr source))))))
(define (read-to-paren in)
(let lp1 ((res '()))
(let ((ch (peek-char in)))
(cond
((or (eof-object? ch) (eqv? #\) ch)) (read-char in) (reverse res))
((char-whitespace? ch) (read-char in) (lp1 res))
((eq? ch #\;)
(let lp2 ()
(let ((ch2 (read-char in)))
(if (or (eof-object? ch2) (eqv? #\newline ch2))
(lp1 res)
(lp2)))))
;; TODO: support #; and #| comments at end of list
(else (lp1 (cons (read in) res)))))))
(call-with-input-file file
(lambda (in)
(let* ((lang (or (and (pair? o) (car o)) 'scheme))
;; filter to only defs found in this file
(defs (filter-map
(lambda (x)
(let ((line (source-line (third x))))
(and line
;; (name value line)
`(,(car x) ,(cadr x) ,line))))
all-defs)))
(let lp ((lines '())
(cur '())
(res '())
(ids '())
(depth 0)
(last-line 0))
(define (collect)
(if (pair? lines)
(append
(reverse
(call-with-input-string
(string-concatenate (reverse lines) "\n")
scribble-parse))
cur)
cur))
(define (get-ids sxml)
(match sxml
(((or 'procedure 'macro) x)
(cond
((string? x)
(list
(string->symbol
(string-trim (car (string-split x))
(lambda (ch) (or (eq? ch #\() (eq? ch #\))))))))
(else
'())))
((x ...) (append-map get-ids x))
(else '())))
(skip-horizontal-whitespace in)
(cond
((eof-object? (peek-char in))
(append (collect) res))
((eqv? #\newline (peek-char in))
(read-char in)
(lp lines cur res ids depth last-line))
((eqv? #\; (peek-char in))
(read-char in)
(cond
((and (eqv? #\; (peek-char in))
(begin (read-char in) (eqv? #\> (peek-char in))))
(read-char in)
(if (eqv? #\space (peek-char in)) (read-char in))
(lp (cons (read-line in) lines) cur res ids depth last-line))
(else
(let lp ()
(cond ((eqv? #\; (peek-char in))
(read-char in)
(lp))))
(let* ((line (read-line in))
(cur (collect))
(ids (append (get-ids cur) ids)))
;; ";;/" attaches the docs to the preceding form
;; rather than the next
(cond
((equal? line "/")
(lp '() '() (append cur res) ids depth last-line))
(else
(cond
((and (not (equal? line ""))
(eqv? #\/ (string-ref line 0)))
(display "WARNING: ;;/ line should be empty"
(current-error-port))
(write line (current-error-port))
(newline (current-error-port))))
(lp '() cur res ids depth last-line)))))))
((eqv? #\) (peek-char in))
(read-char in)
(if (zero? depth)
(error "unexpected ) at line" last-line)
(lp lines cur res ids (- depth 1) last-line)))
((not (eqv? #\( (peek-char in)))
;; ignore non-list top-level expression
(read in)
(lp lines cur res ids depth (port-line in)))
(else ;; found a top-level expression
(read-char in)
(let ((op (read in)))
(case op
((begin define-library)
;; allowed nested docs in these forms
(lp lines cur res ids (+ depth 1) (port-line in)))
(else
;; read until closing paren
(let* ((cur (collect))
(ids (append (get-ids cur) ids))
(form (cons op (read-to-paren in)))
(id (form-defined-name form))
(line (port-line in))
;; find all procedures defined by form
(procs2 (filter (lambda (x) (<= last-line (third x) line))
(filter third defs)))
(procs (if (= 2 (length procs2))
(cdr procs2)
procs2))
;; the the signature for the form
(sigs
(cond
((eq? lang 'ffi)
(filter
(lambda (x)
(assq (if (eq? 'const: (car x)) (third x) (car x))
all-defs))
(get-ffi-signatures form)))
((= 1 (length procs))
(get-signature
mod id (caar procs) (cdar procs) form))
(else
(get-signature
mod id (and id mod (module-ref mod id)) #f form)))))
(cond
((and strict?
(or (not (pair? sigs)) (not (assq (caar sigs) defs))))
;; drop unrelated docs in strict mode
(lp '() '() res ids depth line))
((and (eq? lang 'ffi) (pair? sigs))
(lp '() '() (append (insert-signature cur #f sigs) res)
ids depth line))
((and (memq lang '(scheme module)) (= 1 (length procs)))
(lp '() '()
(append (insert-signature cur (caar procs) sigs) res)
ids depth line))
((and (null? procs)
(and (not (memq id ids)) (assq id all-defs)))
(let ((sigs (if (and (null? sigs) id)
(list id)
sigs)))
(lp '() '() (append (insert-signature cur #f sigs) res)
ids depth line)))
(else
(lp '() '() (append cur res) ids depth line))))))))))))))
;; utility to get the source position of an object
(define (object-source x)
(cond ((opcode? x) #f)
((bytecode? x)
(let ((src (bytecode-source x)))
(if (and (vector? src) (positive? (vector-length src)))
(vector-ref src 0)
src)))
((procedure? x) (object-source (procedure-code x)))
((macro? x) (macro-source x))
(else #f)))
;; helper for below functions
(define (extract-module-docs-from-files mod srcs includes stubs strict? exports . o)
(let ((dir (or (and (pair? o) (car o)) (module-dir mod)))
(defs (map (lambda (x)
(let ((val (and mod (protect (exn (else #f))
(module-ref mod x)))))
`(,x ,val ,(object-source val))))
exports)))
(define (resolve-file file)
(let ((res (make-path dir file)))
(if (file-exists? res)
res
file)))
(append
(reverse
(append-map (lambda (x)
(extract-file-docs mod (resolve-file x) defs strict? 'module))
srcs))
(reverse
(append-map (lambda (x) (extract-file-docs mod (resolve-file x) defs strict?))
includes))
(reverse
(append-map (lambda (x) (extract-file-docs mod (resolve-file x) defs strict? 'ffi))
stubs)))))
;;> Extract the literate Scribble docs from module \var{mod-name} and
;;> return them as sxml. If \var{strict?} is true ignore docs for
;;> unexported values, defined by the optional \var{exports} which
;;> defaults to all the module exports.
(define (extract-module-docs mod-name strict? . o)
(let ((mod (load-module mod-name)))
(if (not mod)
(error "couldn't find module" mod-name))
(let* ((exports (if (pair? o) (car o) (module-exports mod)))
(srcs
(append
(cond ((find-module-file (module-name->file mod-name)) => list)
(else '()))
(module-include-library-declarations mod))))
(extract-module-docs-from-files
mod srcs (module-includes mod) (module-shared-includes mod)
strict? exports))))
;;> As above, but extracts docs for the module defined in \var{file},
;;> which need not be in the search path.
(define (extract-module-file-docs file strict? . o)
(let ((forms (file->sexp-list file)))
(if (not (and (pair? forms) (pair? (car forms))
(memq (caar forms) '(define-library library))))
(error "file doesn't define a library" file))
(let* ((mod-form (car forms))
(mod-name (cadr mod-form))
(lib-dir (module-lib-dir file mod-name))
(orig-mod-path (current-module-path))
(new-mod-path (cons lib-dir orig-mod-path))
(mod (protect (exn (else #f))
(dynamic-wind
(lambda () (current-module-path new-mod-path))
(lambda ()
(let ((mod (load-module mod-name)))
(protect (exn (else #f)) (analyze-module mod-name))
mod))
(lambda () (current-module-path orig-mod-path)))))
(dir (path-directory file)))
(define (get-forms ls names dir . o)
(let ((resolve? (and (pair? o) (car o))))
(let lp ((ls ls) (res '()))
(if (null? ls)
(reverse res)
(let ((x (car ls)))
(lp (cdr ls)
(append
(if (and (pair? x) (memq (car x) names))
(map (lambda (y)
(if (and resolve? (string? y))
(make-path dir y)
y))
(reverse (cdr x)))
'())
(if (and (pair? x)
(eq? 'include-library-declarations (car x)))
(append-map
(lambda (inc)
(let* ((file (make-path dir inc))
(sexps (file->sexp-list file))
(dir (path-directory file)))
(get-forms sexps names dir resolve?)))
(cdr x))
'())
res)))))))
(define (get-exports)
(if mod (module-exports mod) (get-forms (cddr mod-form) '(exports) dir)))
(define (get-decls)
(get-forms (cddr mod-form) '(include-library-declarations) dir #t))
(define (get-includes)
(get-forms (cddr mod-form) '(include include-ci) dir #t))
(define (get-shared-includes)
(map (lambda (f) (string-append f ".stub"))
(get-forms (cddr mod-form) '(include-shared) dir #t)))
(let* ((exports (if (pair? o) (car o) (get-exports)))
(srcs (cons file (get-decls))))
(extract-module-docs-from-files
mod srcs (get-includes) (get-shared-includes) strict? exports)))))