Skip to content

Instantly share code, notes, and snippets.

@kageurufu
Created May 3, 2015 21:30
Show Gist options
  • Save kageurufu/cae42a105d20d52dc907 to your computer and use it in GitHub Desktop.
Save kageurufu/cae42a105d20d52dc907 to your computer and use it in GitHub Desktop.
#lang racket
(require xml net/url)
(define (serve port-no)
(define main-cust (make-custodian))
(parameterize ([current-custodian main-cust])
(define listener (tcp-listen port-no 5 #t))
(define (loop)
(accept-and-handle listener)
(loop))
(thread loop))
(lambda ()
(custodian-shutdown-all main-cust)))
(define (accept-and-handle listener)
(define cust (make-custodian))
(custodian-limit-memory cust (* 50 1024 1024))
(parameterize ([current-custodian cust])
(define-values (in out) (tcp-accept listener))
(thread (lambda ()
(handle in out)
(close-input-port in)
(close-output-port out))))
(thread (lambda ()
(sleep 10)
(custodian-shutdown-all cust))))
(define (handle in out)
(define req
(regexp-match #rx"^GET (.+) HTTP/[0-9]+\\.[0-9]+"
(read-line in)))
(when req
(regexp-match #rx"(\r\n|^)\r\n" in)
(let ([xexpr (dispatch (list-ref req 1))])
(headers out)
(display (xexpr->string xexpr) out))))
(define (headers out)
(display "HTTP/1.0 200 Okay\r\n" out)
(display "Server: k\r\n" out)
(display "Content-Type: text/html\r\n\r\n" out))
(define (dispatch str-path)
(define url (string->url str-path))
(define path (map path/param-path (url-path url)))
(define h (hash-ref dispatch-table (car path) #f))
(if h
(h (url-query url))
`(html (head (title "Error"))
(body
(font ((color "red"))
"Unknown page: ", str-path)))))
(define dispatch-table (make-hash))
(hash-set! dispatch-table "hello"
(lambda (query)
`(html (body "Hello, World!"))))
(define (build-request-page label next-url hidden)
`(html
(head (title "Enter a number to add"))
(body ([bgcolor "white"])
(form ([action ,next-url] [method "get"])
,label
(input ([type "text"] [name "number"] [value ""]))
(input ([type "hidden"] [name "hidden"] [value ,hidden]))
(input ([type "submit"] [name "enter"] [value "Enter"]))))))
(define (many query)
(build-request-page "Number of greetings:" "/reply" ""))
(define (reply query)
(define n (string->number (cdr (assq 'number query))))
`(html
(body
,@(for/list ([i (in-range n)])
" hello"))))
(hash-set! dispatch-table "many" many)
(hash-set! dispatch-table "reply" reply)
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment