123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378 |
- ;;; HTTP response objects
- ;; Copyright (C) 2010, 2011, 2012, 2013, 2014, 2015 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 (ice-9 binary-ports)
- #:use-module (ice-9 textual-ports)
- #:use-module (ice-9 match)
- #: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
- adapt-response-version
- write-response
- response-must-not-include-body?
- response-body-port
- read-response-body
- write-response-body
- ;; 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
- text-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 (non-negative-integer? n)
- (and (number? n) (>= n 0) (exact? n) (integer? n)))
-
- (define (validate-headers headers)
- (if (pair? headers)
- (let ((h (car headers)))
- (if (pair? h)
- (let ((k (car h)) (v (cdr h)))
- (if (valid-header? k v)
- (validate-headers (cdr headers))
- (bad-response "Bad value for header ~a: ~s" k v)))
- (bad-response "Header not a pair: ~a" h)))
- (if (not (null? headers))
- (bad-response "Headers not a list: ~a" headers))))
- (define* (build-response #:key (version '(1 . 1)) (code 200) reason-phrase
- (headers '()) port (validate-headers? #t))
- "Construct an HTTP response object. If VALIDATE-HEADERS? is true,
- the headers are each run through their respective validators."
- (cond
- ((not (and (pair? version)
- (non-negative-integer? (car version))
- (non-negative-integer? (cdr version))))
- (bad-response "Bad version: ~a" version))
- ((not (and (non-negative-integer? code) (< code 600)))
- (bad-response "Bad code: ~a" code))
- ((and reason-phrase (not (string? reason-phrase)))
- (bad-response "Bad reason phrase" reason-phrase))
- (else
- (if validate-headers?
- (validate-headers headers))))
- (make-response version code reason-phrase headers port))
- (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)
- "Return the reason phrase given in RESPONSE, or the standard
- reason phrase for the response's code."
- (or (%response-reason-phrase response)
- (code->reason-phrase (response-code response))))
- (define (text-content-type? type)
- "Return #t if TYPE, a symbol as returned by `response-content-type',
- represents a textual type such as `text/plain'."
- (let ((type (symbol->string type)))
- (or (string-prefix? "text/" type)
- (string-suffix? "/xml" type)
- (string-suffix? "+xml" type))))
- (define (read-response port)
- "Read an HTTP response from PORT.
- As a side effect, sets the encoding on PORT to
- ISO-8859-1 (latin-1), so that reading one character reads one byte. See
- the discussion of character sets in \"HTTP Responses\" in the manual,
- for more information."
- (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)
- "Adapt the given response to a different HTTP version. Returns a new
- HTTP response.
- The idea is that many applications might just build a response for the
- default HTTP version, and this method could handle a number of
- programmatic transformations to respond to older HTTP versions (0.9 and
- 1.0). But currently this function is a bit heavy-handed, just updating
- the version field."
- (build-response #:code (response-code response)
- #:version version
- #:headers (response-headers response)
- #:port (response-port response)))
- (define (write-response r port)
- "Write the given HTTP response to PORT.
- Returns a new response, whose ‘response-port’ will continue writing
- on PORT, perhaps using some transfer encoding."
- (write-response-line (response-version r) (response-code r)
- (response-reason-phrase r) port)
- (write-headers (response-headers r) port)
- (put-string port "\r\n")
- (if (eq? port (response-port r))
- r
- (make-response (response-version r) (response-code r)
- (response-reason-phrase r) (response-headers r) port)))
- (define (response-must-not-include-body? r)
- "Returns ‘#t’ if the response R is not permitted to have a body.
- This is true for some response types, like those with code 304."
- ;; RFC 2616, section 4.3.
- (or (<= 100 (response-code r) 199)
- (= (response-code r) 204)
- (= (response-code r) 304)))
- (define (make-delimited-input-port port len keep-alive?)
- "Return an input port that reads from PORT, and makes sure that
- exactly LEN bytes are available from PORT. Closing the returned port
- closes PORT, unless KEEP-ALIVE? is true."
- (define bytes-read 0)
- (define (fail)
- (bad-response "EOF while reading response body: ~a bytes of ~a"
- bytes-read len))
- (define (read! bv start count)
- ;; Read at most LEN bytes in total. HTTP/1.1 doesn't say what to do
- ;; when a server provides more than the Content-Length, but it seems
- ;; wise to just stop reading at LEN.
- (let ((count (min count (- len bytes-read))))
- (let loop ((ret (get-bytevector-n! port bv start count)))
- (cond ((eof-object? ret)
- (if (= bytes-read len)
- 0 ; EOF
- (fail)))
- ((and (zero? ret) (> count 0))
- ;; Do not return zero since zero means EOF, so try again.
- (loop (get-bytevector-n! port bv start count)))
- (else
- (set! bytes-read (+ bytes-read ret))
- ret)))))
- (define close
- (and (not keep-alive?)
- (lambda ()
- (close-port port))))
- (make-custom-binary-input-port "delimited input port" read! #f #f close))
- (define* (response-body-port r #:key (decode? #t) (keep-alive? #t))
- "Return an input port from which the body of R can be read. The
- encoding of the returned port is set according to R's ‘content-type’
- header, when it's textual, except if DECODE? is ‘#f’. Return #f when
- no body is available.
- When KEEP-ALIVE? is ‘#f’, closing the returned port also closes R's
- response port."
- (define port
- (cond
- ((member '(chunked) (response-transfer-encoding r))
- (make-chunked-input-port (response-port r)
- #:keep-alive? keep-alive?))
- ((response-content-length r)
- => (lambda (len)
- (make-delimited-input-port (response-port r)
- len keep-alive?)))
- ((response-must-not-include-body? r)
- #f)
- ((or (memq 'close (response-connection r))
- (and (equal? (response-version r) '(1 . 0))
- (not (memq 'keep-alive (response-connection r)))))
- (response-port r))
- (else
- ;; Here we have a message with no transfer encoding, no
- ;; content-length, and a response that won't necessarily be closed
- ;; by the server. Not much we can do; assume that the client
- ;; knows how to handle it.
- (response-port r))))
- (when (and decode? port)
- (match (response-content-type r)
- (((? text-content-type?) . props)
- (set-port-encoding! port
- (or (assq-ref props 'charset)
- "ISO-8859-1")))
- (_ #f)))
- port)
- (define (read-response-body r)
- "Reads the response body from R, as a bytevector. Returns
- ‘#f’ if there was no response body."
- (let ((body (and=> (response-body-port r #:decode? #f)
- get-bytevector-all)))
- ;; Reading a body of length 0 will result in get-bytevector-all
- ;; returning the EOF object.
- (if (eof-object? body)
- #vu8()
- body)))
- (define (write-response-body r bv)
- "Write BV, a bytevector, to the port corresponding to the HTTP
- response R."
- (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)
|