(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) (eq? (car form) 'lambda)) (cadr form) #f)) #f))) (define (operator-arglist operator) (let ((binding (eval operator))) (if binding (let ((arglist (procedure-parameter-list binding))) (let loop ((arglist arglist) (optionals? #f) (required '()) (optional '())) (cond ((null? arglist) `(,operator ("args" (("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)))))) '()))) (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) (syntax->datum (expand form))))