mirror of
https://git.savannah.gnu.org/git/guile.git
synced 2025-04-30 20:00:19 +02:00
* module/web/request.scm (read-response-body/latin-1): * module/web/response.scm (read-response-body/latin-1): Avoid the craziness of the read-delimited! interface and hand-roll our own. Fixes errors if read-delimited returns #f or EOF.
275 lines
8.8 KiB
Scheme
275 lines
8.8 KiB
Scheme
;;; HTTP response objects
|
|
|
|
;; Copyright (C) 2010 Free Software Foundation, Inc.
|
|
|
|
;; This library is free software; you can redistribute it and/or
|
|
;; modify it under the terms of the GNU Lesser General Public
|
|
;; License as published by the Free Software Foundation; either
|
|
;; version 3 of the License, or (at your option) any later version.
|
|
;;
|
|
;; This library is distributed in the hope that it will be useful,
|
|
;; but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
;; Lesser General Public License for more details.
|
|
;;
|
|
;; You should have received a copy of the GNU Lesser General Public
|
|
;; License along with this library; if not, write to the Free Software
|
|
;; Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
|
|
;; 02110-1301 USA
|
|
|
|
;;; Code:
|
|
|
|
(define-module (web response)
|
|
#:use-module (rnrs bytevectors)
|
|
#:use-module (rnrs io ports)
|
|
#:use-module (ice-9 rdelim)
|
|
#:use-module (srfi srfi-9)
|
|
#:use-module (web http)
|
|
#:export (response?
|
|
response-version
|
|
response-code
|
|
response-reason-phrase
|
|
response-headers
|
|
response-port
|
|
read-response
|
|
build-response
|
|
extend-response
|
|
adapt-response-version
|
|
write-response
|
|
|
|
read-response-body/latin-1
|
|
write-response-body/latin-1
|
|
|
|
read-response-body/bytevector
|
|
write-response-body/bytevector
|
|
|
|
;; General headers
|
|
;;
|
|
response-cache-control
|
|
response-connection
|
|
response-date
|
|
response-pragma
|
|
response-trailer
|
|
response-transfer-encoding
|
|
response-upgrade
|
|
response-via
|
|
response-warning
|
|
|
|
;; Entity headers
|
|
;;
|
|
response-allow
|
|
response-content-encoding
|
|
response-content-language
|
|
response-content-length
|
|
response-content-location
|
|
response-content-md5
|
|
response-content-range
|
|
response-content-type
|
|
response-expires
|
|
response-last-modified
|
|
|
|
;; Response headers
|
|
;;
|
|
response-accept-ranges
|
|
response-age
|
|
response-etag
|
|
response-location
|
|
response-proxy-authenticate
|
|
response-retry-after
|
|
response-server
|
|
response-vary
|
|
response-www-authenticate))
|
|
|
|
|
|
(define-record-type <response>
|
|
(make-response version code reason-phrase headers port)
|
|
response?
|
|
(version response-version)
|
|
(code response-code)
|
|
(reason-phrase %response-reason-phrase)
|
|
(headers response-headers)
|
|
(port response-port))
|
|
|
|
(define (bad-response message . args)
|
|
(throw 'bad-response message args))
|
|
|
|
(define* (build-response #:key (version '(1 . 1)) (code 200) reason-phrase
|
|
(headers '()) port)
|
|
(make-response version code reason-phrase headers port))
|
|
|
|
(define (extend-response r k v . additional)
|
|
(let ((r (build-response #:version (response-version r)
|
|
#:code (response-code r)
|
|
#:reason-phrase (%response-reason-phrase r)
|
|
#:headers
|
|
(assoc-set! (copy-tree (response-headers r))
|
|
k v)
|
|
#:port (response-port r))))
|
|
(if (null? additional)
|
|
r
|
|
(apply extend-response r additional))))
|
|
|
|
(define *reason-phrases*
|
|
'((100 . "Continue")
|
|
(101 . "Switching Protocols")
|
|
(200 . "OK")
|
|
(201 . "Created")
|
|
(202 . "Accepted")
|
|
(203 . "Non-Authoritative Information")
|
|
(204 . "No Content")
|
|
(205 . "Reset Content")
|
|
(206 . "Partial Content")
|
|
(300 . "Multiple Choices")
|
|
(301 . "Moved Permanently")
|
|
(302 . "Found")
|
|
(303 . "See Other")
|
|
(304 . "Not Modified")
|
|
(305 . "Use Proxy")
|
|
(307 . "Temporary Redirect")
|
|
(400 . "Bad Request")
|
|
(401 . "Unauthorized")
|
|
(402 . "Payment Required")
|
|
(403 . "Forbidden")
|
|
(404 . "Not Found")
|
|
(405 . "Method Not Allowed")
|
|
(406 . "Not Acceptable")
|
|
(407 . "Proxy Authentication Required")
|
|
(408 . "Request Timeout")
|
|
(409 . "Conflict")
|
|
(410 . "Gone")
|
|
(411 . "Length Required")
|
|
(412 . "Precondition Failed")
|
|
(413 . "Request Entity Too Large")
|
|
(414 . "Request-URI Too Long")
|
|
(415 . "Unsupported Media Type")
|
|
(416 . "Requested Range Not Satisfiable")
|
|
(417 . "Expectation Failed")
|
|
(500 . "Internal Server Error")
|
|
(501 . "Not Implemented")
|
|
(502 . "Bad Gateway")
|
|
(503 . "Service Unavailable")
|
|
(504 . "Gateway Timeout")
|
|
(505 . "HTTP Version Not Supported")))
|
|
|
|
(define (code->reason-phrase code)
|
|
(or (assv-ref *reason-phrases* code)
|
|
"(Unknown)"))
|
|
|
|
(define (response-reason-phrase response)
|
|
(or (%response-reason-phrase response)
|
|
(code->reason-phrase (response-code response))))
|
|
|
|
(define (read-response port)
|
|
(set-port-encoding! port "ISO-8859-1")
|
|
(call-with-values (lambda () (read-response-line port))
|
|
(lambda (version code reason-phrase)
|
|
(make-response version code reason-phrase (read-headers port) port))))
|
|
|
|
(define (adapt-response-version response version)
|
|
(build-response #:code (response-code response)
|
|
#:version version
|
|
#:headers (response-headers response)
|
|
#:port (response-port response)))
|
|
|
|
(define (write-response r port)
|
|
(write-response-line (response-version r) (response-code r)
|
|
(response-reason-phrase r) port)
|
|
(write-headers (response-headers r) port)
|
|
(display "\r\n" port)
|
|
(if (eq? port (response-port r))
|
|
r
|
|
(make-response (response-version r) (response-code r)
|
|
(response-reason-phrase r) (response-headers r) port)))
|
|
|
|
;; Probably not what you want to use "in production". Relies on one byte
|
|
;; per char because we are in latin-1 encoding.
|
|
;;
|
|
(define (read-response-body/latin-1 r)
|
|
(cond
|
|
((response-content-length r) =>
|
|
(lambda (nbytes)
|
|
(let ((buf (make-string nbytes))
|
|
(port (response-port r)))
|
|
(let lp ((i 0))
|
|
(cond
|
|
((< i nbytes)
|
|
(let ((c (read-char port)))
|
|
(cond
|
|
((eof-object? c)
|
|
(bad-response "EOF while reading response body: ~a bytes of ~a"
|
|
i nbytes))
|
|
(else
|
|
(string-set! buf i c)
|
|
(lp (1+ i))))))
|
|
(else buf))))))
|
|
(else #f)))
|
|
|
|
;; Likewise, assumes that body can be written in the latin-1 encoding,
|
|
;; and that the latin-1 encoding is what is expected by the server.
|
|
;;
|
|
(define (write-response-body/latin-1 r body)
|
|
(display body (response-port r)))
|
|
|
|
(define (read-response-body/bytevector r)
|
|
(let ((nbytes (response-content-length r)))
|
|
(and nbytes
|
|
(let ((bv (get-bytevector-n (response-port r) nbytes)))
|
|
(if (= (bytevector-length bv) nbytes)
|
|
bv
|
|
(bad-response "EOF while reading response body: ~a bytes of ~a"
|
|
(bytevector-length bv) nbytes))))))
|
|
|
|
(define (write-response-body/bytevector r bv)
|
|
(put-bytevector (response-port r) bv))
|
|
|
|
(define-syntax define-response-accessor
|
|
(lambda (x)
|
|
(syntax-case x ()
|
|
((_ field)
|
|
#'(define-response-accessor field #f))
|
|
((_ field def) (identifier? #'field)
|
|
#`(define* (#,(datum->syntax
|
|
#'field
|
|
(symbol-append 'response- (syntax->datum #'field)))
|
|
response
|
|
#:optional (default def))
|
|
(cond
|
|
((assq 'field (response-headers response)) => cdr)
|
|
(else default)))))))
|
|
|
|
;; General headers
|
|
;;
|
|
(define-response-accessor cache-control '())
|
|
(define-response-accessor connection '())
|
|
(define-response-accessor date #f)
|
|
(define-response-accessor pragma '())
|
|
(define-response-accessor trailer '())
|
|
(define-response-accessor transfer-encoding '())
|
|
(define-response-accessor upgrade '())
|
|
(define-response-accessor via '())
|
|
(define-response-accessor warning '())
|
|
|
|
;; Entity headers
|
|
;;
|
|
(define-response-accessor allow '())
|
|
(define-response-accessor content-encoding '())
|
|
(define-response-accessor content-language '())
|
|
(define-response-accessor content-length #f)
|
|
(define-response-accessor content-location #f)
|
|
(define-response-accessor content-md5 #f)
|
|
(define-response-accessor content-range #f)
|
|
(define-response-accessor content-type #f)
|
|
(define-response-accessor expires #f)
|
|
(define-response-accessor last-modified #f)
|
|
|
|
;; Response headers
|
|
;;
|
|
(define-response-accessor accept-ranges #f)
|
|
(define-response-accessor age #f)
|
|
(define-response-accessor etag #f)
|
|
(define-response-accessor location #f)
|
|
(define-response-accessor proxy-authenticate #f)
|
|
(define-response-accessor retry-after #f)
|
|
(define-response-accessor server #f)
|
|
(define-response-accessor vary '())
|
|
(define-response-accessor www-authenticate #f)
|