#lang racket/base (require racket/list web-server/servlet web-server/servlet-env) ;; A hacky way to test the evaluation. ;; ;; Sets up a web server and opens a browser window. ;; The page on screen periodically polls the server to see if a program has ;; come in to be evaluated. Whenever code does come in, evaluates and returns the ;; value to the user, along with the time it took to evaluate. (provide make-evaluate (struct-out error-happened) (struct-out evaluated)) (define-struct error-happened (str t) #:transparent) (define-struct evaluated (stdout value t browser) #:transparent) ;; make-evaluate: (Any output-port) -> void ;; Produce a JavaScript evaluator that cooperates with a browser. ;; The JavaScript-compiler is expected to write out a thunk. When invoked, ;; the thunk should return a function that consumes three values, corresponding ;; to success, failure, and other parameters to evaluation. For example: ;; ;; (make-evaluate (lambda (program op) ;; (fprintf op "(function() { ;; return function(success, fail, params) { ;; success('ok'); ;; }})"))) ;; ;; is a do-nothing evaluator that will always give back 'ok'. ;; ;; At the moment, the evaluator will pass in a parameter that binds 'currentDisplayer' to a function ;; that captures output. (define (make-evaluate javascript-compiler) (define port (+ 8000 (random 8000))) ;; This channel's meant to serialize use of the web server. (define ch (make-channel)) ;; start up the web server ;; The web server responds to two types of requests ;; ?comet Starting up the comet request path. ;; ?v Getting a value back from evaluation. ;; ?e Got an error. (void (thread (lambda () (define (start req) (cond ;; Server-side sync for a program [(exists-binding? 'comet (request-bindings req)) (handle-comet req)] ;; Normal result came back [(exists-binding? 'v (request-bindings req)) (handle-normal-response req)] ;; Error occurred [(exists-binding? 'e (request-bindings req)) (handle-error-response req)] [else (make-on-first-load-response)])) (serve/servlet start #:banner? #f #:launch-browser? #t #:quit? #f #:port port #:servlet-path "/eval")))) (define *alarm-timeout* 30000) (define (handle-comet req) (let/ec return (let* ([alarm (alarm-evt (+ (current-inexact-milliseconds) *alarm-timeout*))] [program (sync ch alarm)] [op (open-output-bytes)]) (cond [(eq? program alarm) (try-again-response)] [else (with-handlers ([exn:fail? (lambda (exn) (let ([sentinel (format #<bytes/utf-8 sentinel))))))]) (javascript-compiler program op)) (response/full 200 #"Okay" (current-seconds) #"text/plain; charset=utf-8" empty (list #"" (get-output-bytes op)))])))) (define (try-again-response) (response/full 200 #"Try again" (current-seconds) #"text/plain; charset=utf-8" empty (list #"" #""))) (define (ok-response) (response/full 200 #"Okay" (current-seconds) TEXT/HTML-MIME-TYPE empty (list #"" #"

ok

"))) (define (handle-normal-response req) (channel-put ch (make-evaluated (extract-binding/single 'o (request-bindings req)) (extract-binding/single 'v (request-bindings req)) (string->number (extract-binding/single 't (request-bindings req))) (extract-binding/single 'b (request-bindings req)))) (ok-response)) (define (handle-error-response req) (channel-put ch (make-error-happened (extract-binding/single 'e (request-bindings req)) (string->number (extract-binding/single 't (request-bindings req))))) (ok-response)) (define (make-on-first-load-response) (let ([op (open-output-bytes)]) (fprintf op #<

Harness loaded. Do not close this window.

EOF ) (response/full 200 #"Okay" (current-seconds) TEXT/HTML-MIME-TYPE empty (list #"" (get-output-bytes op))))) ;; evaluate: sexp -> (values string number) ;; A little driver to test the evalution of expressions, using a browser to help. ;; Returns the captured result of stdout, plus # of milliseconds it took to execute. (define (evaluate e) ;; Send the program to the web browser, and wait for the thread to send back (channel-put ch e) (let ([result (channel-get ch)]) (cond [(error-happened? result) (raise result)] [else result]))) evaluate)