mirror of
https://git.savannah.gnu.org/git/guile.git
synced 2025-04-29 19:30:36 +02:00
These were found with: make GUILE_WARNINGS='-W1 -Wunused-module' * module/ice-9/copy-tree.scm: * module/ice-9/eval-string.scm: * module/ice-9/getopt-long.scm: * module/ice-9/poll.scm: * module/ice-9/popen.scm: * module/ice-9/sandbox.scm: * module/ice-9/threads.scm: * module/sxml/apply-templates.scm: * module/sxml/simple.scm: * module/system/base/types.scm: * module/system/repl/command.scm: * module/system/repl/common.scm: * module/system/repl/coop-server.scm: * module/system/repl/debug.scm: * module/system/repl/error-handling.scm: * module/system/repl/repl.scm: * module/system/repl/server.scm: * module/system/vm/assembler.scm: * module/system/vm/disassembler.scm: * module/system/vm/dwarf.scm: * module/system/vm/elf.scm: * module/system/vm/frame.scm: * module/system/vm/inspect.scm: * module/system/vm/linker.scm: * module/system/vm/program.scm: * module/system/vm/trace.scm: * module/system/vm/trap-state.scm: * module/system/vm/traps.scm: * module/system/xref.scm: * module/texinfo/indexing.scm: * module/texinfo/plain-text.scm: * module/texinfo/reflection.scm: * module/texinfo/string-utils.scm: * module/web/client.scm: * module/web/http.scm: * module/web/request.scm: * module/web/response.scm: Remove imports of unused modules.
377 lines
13 KiB
Scheme
377 lines
13 KiB
Scheme
;;; 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)
|