(library (geiser) (export geiser:eval geiser:completions geiser:module-completions geiser:autodoc geiser:no-values geiser:load-file geiser:newline geiser:macroexpand) (import (chezscheme)) (define (last-index-of str-list char idx last-idx) (if (null? str-list) last-idx (last-index-of (cdr str-list) char (+ 1 idx) (if (char=? char (car str-list)) idx last-idx)))) (define (obj-file-name name) (let ((idx (last-index-of (string->list name) #\. 0 -1))) (if (= idx -1) (string-append name ".so") (string-append (substring name 0 idx) ".so")))) (define (geiser:load-file filename) (let ((output-filename (obj-file-name filename))) (maybe-compile-file filename output-filename) (load output-filename))) (define string-prefix? (lambda (x y) (let ([n (string-length x)]) (and (fx<= n (string-length y)) (let prefix? ([i 0]) (or (fx= i n) (and (char=? (string-ref x i) (string-ref y i)) (prefix? (fx+ i 1))))))))) (define (geiser:completions prefix . rest) rest (sort string-ci (length form) 2)) (case (car form) [(lambda) (list (cadr form))] [(case-lambda) (map car (cdr form))] [else #f]) #f)) #f))) (define (operator-arglist operator) (define (make-autodoc-arglist arglist) (let loop ([arglist arglist] [optionals? #f] [required '()] [optional '()]) (cond ((null? arglist) `(("required" ,@(reverse required)) ("optional" ,@(reverse optional)) ("key") ;; ("module" ,module) )) ((symbol? arglist) (loop '() #t required (cons "..." (cons arglist optional)))) (else (loop (cdr arglist) optionals? (if optionals? required (cons (car arglist) required)) (if optionals? (cons (car arglist) optional) optional)))))) (let ([binding (eval operator)]) (if binding (let ([arglists (procedure-parameter-list binding)]) `(,operator ("args" ,@(map make-autodoc-arglist arglists)))) '()))) (define (geiser:autodoc ids . rest) (cond ((null? ids) '()) ((not (list? ids)) (geiser:autodoc (list ids))) ((not (symbol? (car ids))) (geiser:autodoc (cdr ids))) (else (map (lambda (id) (operator-arglist id)) ids)))) (define (geiser:no-values) #f) (define (geiser:newline) #f) (define (geiser:macroexpand form . rest) (with-output-to-string (lambda () (pretty-print (syntax->datum (expand form)))))))