+ REQUEST object extended for SUrflets by INPUT-PORT

+ SURFLET-REQUESTs used in SUrflets
+ Use MAKE-INPUT-RESPONSE to generate SUrflet responses
This commit is contained in:
interp 2003-02-17 10:09:24 +00:00
parent bf937b2e74
commit 21f62d5d8e
12 changed files with 111 additions and 66 deletions

View File

@ -118,9 +118,10 @@
(surflet-handler/surflets surflet-handler/surflets-interface) (surflet-handler/surflets surflet-handler/surflets-interface)
(surflet-handler/admin surflet-handler/admin-interface) (surflet-handler/admin surflet-handler/admin-interface)
(surflet-handler/responses surflet-handler/responses-interface)) (surflet-handler/responses surflet-handler/responses-interface))
(open httpd-responses (open httpd-responses ;replies for httpd
httpd-requests httpd-requests ;requests from httpd
httpd-errors surflet-requests ;requests for surflets
httpd-errors ;errors for httpd
uri ;URI-PATH-LIST->PATH uri ;URI-PATH-LIST->PATH
tables ;HASH-TABLES tables ;HASH-TABLES
define-record-types ;DEFINE-RECORD-TYPE define-record-types ;DEFINE-RECORD-TYPE
@ -193,7 +194,7 @@
(open surflet-handler/surflets (open surflet-handler/surflets
surflet-handler/responses surflet-handler/responses
httpd-responses ; STATUS-CODE httpd-responses ; STATUS-CODE
httpd-requests ; HTTP-URL:SEARCH surflet-requests ; HTTP-URL:SEARCH
url ; REQUEST:URL url ; REQUEST:URL
parse-html-forms parse-html-forms
sxml-to-html ; SXML->HTML sxml-to-html ; SXML->HTML
@ -314,3 +315,22 @@
locks locks
define-record-types) define-record-types)
(files thread-safe-counter)) (files thread-safe-counter))
(define-interface surflet-requests-interface
(export make-surflet-request
surflet-request?
surflet-request-request
surflet-request-input-port
surflet-request-method
surflet-request-uri
surflet-request-url
surflet-request-version
surflet-request-headers
surflet-request-socket))
(define-structure surflet-requests surflet-requests-interface
(open scheme
define-record-types
httpd-requests)
(files surflet-request))

View File

@ -46,11 +46,14 @@
(path-string (uri-path->uri path))) (path-string (uri-path->uri path)))
(if (or (string=? request-method "GET") (if (or (string=? request-method "GET")
(string=? request-method "POST")) (string=? request-method "POST"))
(if (resume-url? path-string) (make-input-response
(resume-url path-string surflet-path req) (lambda (input-port)
(launch-new-session path-string surflet-path req)) (let ((s-req (make-surflet-request req input-port)))
(if (resume-url? path-string)
(resume-url path-string surflet-path s-req)
(launch-new-session path-string surflet-path s-req)))))
(make-error-response (status-code method-not-allowed) req (make-error-response (status-code method-not-allowed) req
request-method))) request-method)))
(make-error-response (status-code bad-request) req (make-error-response (status-code bad-request) req
(format #f "Bad path: ~s" path))))) (format #f "Bad path: ~s" path)))))
@ -58,13 +61,14 @@
;; Loads and runs a new session of a SUrflet installing the RESET ;; Loads and runs a new session of a SUrflet installing the RESET
;; boundary; returns a (HTTP-)RESPONSE. PATH-STRING is the virtual ;; boundary; returns a (HTTP-)RESPONSE. PATH-STRING is the virtual
;; path of the request, SURFLET-PATH is a string pointing to the real ;; path of the request, SURFLET-PATH is a string pointing to the real
;; directory of the SUrflets, and REQ the request of the browser. ;; directory of the SUrflets, and S-REQ the request of the browser.
(define (launch-new-session path-string surflet-path req) (define (launch-new-session path-string surflet-path s-req)
(cond (cond
((file-not-exists? (absolute-file-name path-string surflet-path)) ((file-not-exists? (absolute-file-name path-string surflet-path))
(make-error-response (status-code not-found) req path-string)) (make-error-response (status-code not-found)
(surflet-request-request s-req) path-string))
((string=? (file-name-extension path-string) ".scm") ((string=? (file-name-extension path-string) ".scm")
(obtain-lock *session-table-lock*) (obtain-lock *session-table-lock*)
@ -86,7 +90,7 @@
;; Catch conditions from get-surflet-rt-structure. ;; Catch conditions from get-surflet-rt-structure.
(lambda (condition decline) (lambda (condition decline)
(delete-session! session-id) (delete-session! session-id)
(bad-gateway-error-response req path-string condition)) (bad-gateway-error-response s-req path-string condition))
(let ((surflet (get-surflet-rt-structure path-string surflet-path))) (let ((surflet (get-surflet-rt-structure path-string surflet-path)))
(fork-thread (fork-thread
(session-surveillance session-id (session-surveillance session-id
@ -99,23 +103,17 @@
(delete-session! session-id) (delete-session! session-id)
;; Restore correct continuation with shift. ;; Restore correct continuation with shift.
(shift unused (shift unused
(bad-gateway-error-response req path-string condition))) (bad-gateway-error-response s-req path-string condition)))
(with-cwd surflet-path (with-cwd surflet-path
(with-names-from-rt-structure (with-names-from-rt-structure
surflet surflet-interface surflet surflet-interface
(main req))))))))) ; Launch serlvet's main procedure. (main s-req))))))))) ; Launch serlvet's main procedure.
(else ; We'll serve every non-scm file. (else ; We'll serve every non-scm file.
;; We need access to SEND-FILE-RESPONSE of (make-error-response (status-code forbidden)
;; HTTPD-FILE-DIR-HANDLERS. In the official SUnet release, we (surflet-request-request s-req)
;; don't have it, so we disable this feature here. "Can't serve other than Scheme files."
; (let ((full-file-name (absolute-file-name path-string surflet-path))) path-string))
; (send-file-response full-file-name
; (file-info full-file-name)
; req))
(make-error-response (status-code forbidden) req
"Can't serve other than Scheme files."
path-string))
)) ))
@ -156,12 +154,13 @@
;;; RESUME-URL ;;; RESUME-URL
;; Resumes a suspended URL and returns a (HTTP-)RESPONSE. PATH-STRING ;; Resumes a suspended URL and returns a (HTTP-)RESPONSE. PATH-STRING
;; is the virtual path, SURFLET-PATH a string pointing to the real ;; is the virtual path, SURFLET-PATH a string pointing to the real
;; directory of the SUrflets and REQ the request of the browser. ;; directory of the SUrflets and S-REQ the request of the browser.
(define resume-url (define resume-url
(let ((bad-request (let ((bad-request
(lambda (path-string req) (lambda (path-string s-req)
(make-error-response (make-error-response
(status-code bad-request) req (status-code bad-request)
(surflet-request-request s-req)
(format #f (format #f
"<br> "<br>
<p>There may be several reasons, why your request for a SUrflet was denied: <p>There may be several reasons, why your request for a SUrflet was denied:
@ -181,7 +180,7 @@
(release-lock continuation-table-lock) (release-lock continuation-table-lock)
result))))) result)))))
(lambda (path-string surflet-path req) (lambda (path-string surflet-path s-req)
(receive (session-id continuation-id) (receive (session-id continuation-id)
;; Searches ids only in file-name. ;; Searches ids only in file-name.
(resume-url-ids (file-name-nondirectory path-string)) (resume-url-ids (file-name-nondirectory path-string))
@ -196,9 +195,9 @@
(reset (reset
(begin (begin
(register-instance! session-id) (register-instance! session-id)
(resume req)))) (resume s-req))))
(bad-request path-string req))) (bad-request path-string s-req)))
(bad-request path-string req))) (bad-request path-string s-req)))
)))) ))))
@ -250,12 +249,14 @@
;;; SEND-ERROR ;;; SEND-ERROR
;; Stops current computation, and leaves current continuation via ;; Stops current computation, and leaves current continuation via
;; SHIFT with a (HTTP-)(ERROR-)RESPONSE. STATUS-CODE is a status code ;; SHIFT with a (HTTP-)(ERROR-)RESPONSE. STATUS-CODE is a status code
;; from HTTP-RESPONSES, REQ a request (may be #f) and MESSAGES ;; from HTTP-RESPONSES, S-REQ a surflet-request (may be #f) and
;; contains further informations (arbitrary types). ;; MESSAGES contains further informations (arbitrary types).
(define (send-error status-code req . messages) (define (send-error status-code s-req . messages)
(shift unused (apply make-error-response (shift unused (apply make-error-response
(cons status-code (cons status-code
(cons #f messages))))) (cons (and (surflet-request? s-req)
(surflet-request-request s-req))
messages)))))
;;; MAKE-HTTP-RESPONSE ;;; MAKE-HTTP-RESPONSE
;; Converts a SURFLET-RESPONSE to a (HTTP-)RESPONSE. Returns a ;; Converts a SURFLET-RESPONSE to a (HTTP-)RESPONSE. Returns a
@ -579,9 +580,10 @@
(define (resume-url? resume-url) (define (resume-url? resume-url)
(regexp-search? *resume-url-regexp* resume-url)) (regexp-search? *resume-url-regexp* resume-url))
(define (bad-gateway-error-response req path-string condition) (define (bad-gateway-error-response s-req path-string condition)
(make-error-response (make-error-response
(status-code bad-gateway) req (status-code bad-gateway)
(surflet-request-request s-req)
(format #f "Error in SUrflet ~s." path-string) (format #f "Error in SUrflet ~s." path-string)
condition)) condition))

View File

@ -0,0 +1,19 @@
;; Copyright Andras Bernauer (2003)
;; somehow `extend' httpd-request
(define-record-type surflet-request :surflet-request
(make-surflet-request request input-port)
surflet-request?
(request surflet-request-request)
(input-port surflet-request-input-port))
(define (make-fake-selector request-selector)
(lambda (surflet-request)
(request-selector (surflet-request-request surflet-request))))
(define surflet-request-method (make-fake-selector request-method))
(define surflet-request-uri (make-fake-selector request-uri))
(define surflet-request-url (make-fake-selector request-url))
(define surflet-request-version (make-fake-selector request-version))
(define surflet-request-headers (make-fake-selector request-headers))
(define surflet-request-socket (make-fake-selector request-socket))

View File

@ -41,19 +41,19 @@
(define *POST-bindings-cache* '()) (define *POST-bindings-cache* '())
(define *cache-lock* (make-lock)) (define *cache-lock* (make-lock))
(define (get-bindings request) (define (get-bindings surflet-request)
(let ((request-method (request-method request))) (let ((request-method (surflet-request-method surflet-request)))
(cond (cond
((string=? request-method "GET") ((string=? request-method "GET")
(form-query (http-url-search (request-url request)))) (form-query (http-url-search (surflet-request-url surflet-request))))
((string=? request-method "POST") ((string=? request-method "POST")
(or (cached-bindings request) (or (cached-bindings surflet-request)
(let* ((content-length (get-content-length (request-headers request))) (let* ((content-length (get-content-length (surflet-request-headers surflet-request)))
(input-port (socket:inport (request-socket request))) (input-port (surflet-request-input-port surflet-request))
(form-data (read-string content-length input-port))) (form-data (read-string content-length input-port)))
(let ((form-bindings (form-query form-data))) (let ((form-bindings (form-query form-data)))
(obtain-lock *cache-lock*) (obtain-lock *cache-lock*)
(set! *POST-bindings-cache* (cons (cons (make-weak-pointer request) (set! *POST-bindings-cache* (cons (cons (make-weak-pointer surflet-request)
form-bindings) form-bindings)
*POST-bindings-cache*)) *POST-bindings-cache*))
(release-lock *cache-lock*) (release-lock *cache-lock*)
@ -64,17 +64,19 @@
;; Looking up, if we have cached this request. While going through the ;; Looking up, if we have cached this request. While going through the
;; list, we remove entries to request objects, that are no longer ;; list, we remove entries to request objects, that are no longer
;; valid. Expecting a call for an uncached request every now and then, ;; valid. Expecting a call for an uncached request every now and then,
;; it is guaranteed, that the list is cleaned up every now and then. ;; it is guaranteed, that the list is cleaned up every now and
(define (cached-bindings request) ;; then. The cache is a list of pairs
;;; (surflet-request . computed-binding)
(define (cached-bindings surflet-request)
(obtain-lock *cache-lock*) (obtain-lock *cache-lock*)
(let ((result (let ((result
(let loop ((cache *POST-bindings-cache*)) (let loop ((cache *POST-bindings-cache*))
(if (null? cache) (if (null? cache)
#f ; no such request cached #f ; no such request cached
(let* ((head (car cache)) (let* ((head (car cache))
(req (weak-pointer-ref (car head)))) (s-req (weak-pointer-ref (car head))))
(if req (if s-req
(if (eq? req request) (if (eq? s-req surflet-request)
(cdar cache) ; request is cached (cdar cache) ; request is cached
(loop (cdr cache))) ; request isn't cached (loop (cdr cache))) ; request isn't cached
(begin (begin
@ -271,7 +273,7 @@
(send-html (send-html
`(html (title "Outdated Data") `(html (title "Outdated Data")
(body (h1 "Outdated Data") (body (h1 "Outdated Data")
(p "The page or action you requested relies on outdated data") (p "The page or action you requested relies on outdated data.")
,(if url ,(if url
`(p "Try to " `(p "Try to "
(URL ,url "reload") (URL ,url "reload")

View File

@ -1,6 +1,6 @@
(define-structure surflet surflet-interface (define-structure surflet surflet-interface
(open surflets (open surflets
httpd-requests surflet-requests
httpd-responses httpd-responses
url url
scheme-with-scsh) scheme-with-scsh)
@ -26,7 +26,7 @@
(p (URL "/" "Return to main menu") (br) (p (URL "/" "Return to main menu") (br)
(URL "add.scm" "Start new calculation.")))))))) (URL "add.scm" "Start new calculation."))))))))
(let* ((bindings (form-query (let* ((bindings (form-query
(http-url-search (request-url result)))) (http-url-search (surflet-request-url result))))
(number (string->number (number (string->number
(extract-single-binding "number" bindings)))) (extract-single-binding "number" bindings))))
(if number (if number

View File

@ -1,5 +1,5 @@
(define-structure surflet surflet-interface (define-structure surflet surflet-interface
(open httpd-requests ; REQUEST-URL (open surflet-requests ; SURFLET-REQUEST-URL
httpd-responses ; MAKE-RESPONSE httpd-responses ; MAKE-RESPONSE
parse-html-forms ; PARSE-HTML-FORM-QUERY parse-html-forms ; PARSE-HTML-FORM-QUERY
url ; HTTP-URL-SEARCH url ; HTTP-URL-SEARCH
@ -79,7 +79,7 @@
(let* ((title (if (pair? maybe-title) (car maybe-title) #f)) (let* ((title (if (pair? maybe-title) (car maybe-title) #f))
(result (send/suspend (make-get-number-page input-text title))) (result (send/suspend (make-get-number-page input-text title)))
(bindings (parse-html-form-query (bindings (parse-html-form-query
(http-url-search (request-url result)))) (http-url-search (surflet-request-url result))))
(number (string->number (number (string->number
(extract-single-binding "number" bindings)))) (extract-single-binding "number" bindings))))
(if number (if number

View File

@ -1,6 +1,6 @@
(define-structure surflet surflet-interface (define-structure surflet surflet-interface
(open surflets (open surflets
httpd-requests surflet-requests
url url
handle-fatal-error handle-fatal-error
let-opt let-opt
@ -26,7 +26,8 @@
(URL "add2.scm" "Start new calculation.")))))))) (URL "add2.scm" "Start new calculation."))))))))
(if result (if result
(or (input-field-value number-input-field (or (input-field-value number-input-field
(form-query (http-url-search (request-url result)))) (form-query (http-url-search
(surflet-request-url result))))
(get-number title "Please enter a valid number.")) (get-number title "Please enter a valid number."))
(get-number title "Please enter a number.")))) (get-number title "Please enter a number."))))

View File

@ -3,7 +3,7 @@
surflets surflets
surflet-handler/admin surflet-handler/admin
httpd-responses httpd-responses
httpd-requests surflet-requests
url url
handle-fatal-error handle-fatal-error
let-opt let-opt
@ -152,7 +152,7 @@
(current-sessions (sort-list! (get-sessions) session-surflet-name<?))) (current-sessions (sort-list! (get-sessions) session-surflet-name<?)))
(real-sessions current-sessions update-text (real-sessions current-sessions update-text
(resume-url-session-id (resume-url-session-id
(last (http-url-path (request-url req))))))) (last (http-url-path (surflet-request-url req)))))))
(define (real-sessions current-sessions update-text this-session-id) (define (real-sessions current-sessions update-text this-session-id)
(let ((outdated? (make-outdater)) (let ((outdated? (make-outdater))
@ -258,7 +258,7 @@
(session-id (car session-pair)) (session-id (car session-pair))
(session-entry (cdr session-pair)) (session-entry (cdr session-pair))
(this-continuation-id (resume-url-continuation-id (this-continuation-id (resume-url-continuation-id
(last (http-url-path (request-url req))))) (last (http-url-path (surflet-request-url req)))))
(update-text (:optional maybe-update-text ""))) (update-text (:optional maybe-update-text "")))
(let* ((current-continuations (let* ((current-continuations
(sort-list! (get-continuations session-id) (sort-list! (get-continuations session-id)

View File

@ -1,6 +1,6 @@
(define-structure surflet surflet-interface (define-structure surflet surflet-interface
(open surflets (open surflets
httpd-requests surflet-requests
handle-fatal-error handle-fatal-error
url url
scheme-with-scsh) scheme-with-scsh)
@ -55,7 +55,7 @@
,(make-submit-button)) ,(make-submit-button))
(hr) (hr)
(p (URL "/" "Return to main menu."))))))) (p (URL "/" "Return to main menu.")))))))
(bindings (form-query (http-url-search (request-url req))))) (bindings (form-query (http-url-search (surflet-request-url req)))))
(input-field-value byte-input-fields bindings))) (input-field-value byte-input-fields bindings)))
(define (main req) (define (main req)

View File

@ -1,6 +1,6 @@
(define-structure surflet surflet-interface (define-structure surflet surflet-interface
(open surflets (open surflets
httpd-requests surflet-requests
handle-fatal-error handle-fatal-error
let-opt let-opt
scheme-with-scsh) scheme-with-scsh)

View File

@ -1,6 +1,6 @@
(define-structure surflet surflet-interface (define-structure surflet surflet-interface
(open surflets (open surflets
httpd-requests surflet-requests
handle-fatal-error handle-fatal-error
let-opt let-opt
scheme-with-scsh) scheme-with-scsh)

View File

@ -7,7 +7,7 @@
(subset srfi-13 (string-downcase string-join)) (subset srfi-13 (string-downcase string-join))
(subset srfi-1 (find filter-map split-at remove)) (subset srfi-1 (find filter-map split-at remove))
sunet-utilities sunet-utilities
httpd-requests) surflet-requests)
(begin (begin
;;; Spaceship components ;;; Spaceship components
@ -462,7 +462,8 @@ spaceships of class " ,class ":")
(p "Thank you for your ordering.") (p "Thank you for your ordering.")
(p "Your order has been registered. " (p "Your order has been registered. "
"We will contact you (" "We will contact you ("
,(host-name-or-ip (socket-remote-address (request-socket req))) ,(host-name-or-ip (socket-remote-address
(surflet-request-socket req)))
") as soon as the ship is built.") ") as soon as the ship is built.")
(p "This will take about " ,months " months.") (p "This will take about " ,months " months.")
,(return-links first-page-return-link main-return-link)))) ,(return-links first-page-return-link main-return-link))))