2000-09-26 10:35:26 -04:00
|
|
|
;;; http server in the Scheme Shell -*- Scheme -*-
|
2002-08-26 11:14:10 -04:00
|
|
|
|
|
|
|
;;; This file is part of the Scheme Untergrund Networking package.
|
2000-09-26 10:35:26 -04:00
|
|
|
|
|
|
|
;;; Copyright (c) 1994 by Brian D. Carlstrom and Olin Shivers.
|
2002-08-26 11:14:10 -04:00
|
|
|
;;; Copyright (c) 1996-2002 by Mike Sperber.
|
|
|
|
;;; Copyright (c) 2000-2002 by Martin Gasbichler.
|
|
|
|
;;; Copyright (c) 2002 by Andreas Bernauer.
|
|
|
|
;;; For copyright information, see the file COPYING which comes with
|
|
|
|
;;; the distribution.
|
|
|
|
|
2000-09-26 10:35:26 -04:00
|
|
|
|
|
|
|
;;; This file implements the core of an HTTP server: code to establish
|
|
|
|
;;; net connections, read and parse requests, and handler errors.
|
|
|
|
;;; It does not have the code to actually handle requests. That's up
|
|
|
|
;;; to other modules, and could vary from server to server. To build
|
|
|
|
;;; a complete server, you need to define path handlers (see below) --
|
|
|
|
;;; they determine how requests are to be handled.
|
|
|
|
;;;
|
2002-08-22 10:59:49 -04:00
|
|
|
;;; The RFC detailing the HTTP 1.0 protocol, RFC 1945, can be found at
|
|
|
|
;;; http://www.w3.org/Protocols/rfc1945/rfc1945
|
2000-09-26 10:35:26 -04:00
|
|
|
|
2002-02-23 09:42:50 -05:00
|
|
|
;;; (httpd options)
|
2000-09-26 10:35:26 -04:00
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
;;; The server top-level. PATH-HANDLER is the top-level request path handler --
|
2002-02-23 09:42:50 -05:00
|
|
|
;;; the procedure that actually deals with the request.
|
2000-09-26 10:35:26 -04:00
|
|
|
|
2002-08-26 07:44:02 -04:00
|
|
|
(define server/protocol "HTTP/1.0")
|
|
|
|
|
2002-02-23 09:42:50 -05:00
|
|
|
(define (httpd options)
|
|
|
|
(let ((port (httpd-options-port options))
|
2002-03-01 03:54:48 -05:00
|
|
|
(root-dir (httpd-options-root-directory options))
|
|
|
|
(rate-limiter
|
|
|
|
(cond
|
|
|
|
((httpd-options-simultaneous-requests options)
|
|
|
|
=> make-rate-limiter)
|
|
|
|
(else #f))))
|
2002-03-25 06:35:05 -05:00
|
|
|
(init-http-log! options)
|
|
|
|
(with-syslog-destination
|
|
|
|
"httpd" #f #f #f
|
|
|
|
(lambda ()
|
|
|
|
(with-cwd
|
|
|
|
root-dir
|
|
|
|
(bind-listen-accept-loop
|
|
|
|
protocol-family/internet
|
|
|
|
;; Why is the output socket unbuffered? So that if the client
|
|
|
|
;; closes the connection, we won't lose when we try to close the
|
|
|
|
;; socket by trying to flush the output buffer.
|
|
|
|
(lambda (sock addr)
|
2002-03-01 03:54:48 -05:00
|
|
|
(if rate-limiter
|
2002-03-25 06:35:05 -05:00
|
|
|
(begin
|
|
|
|
(rate-limit-block rate-limiter)
|
|
|
|
(rate-limit-open rate-limiter)))
|
|
|
|
|
|
|
|
(with-fatal-error-handler
|
|
|
|
(lambda (c decline)
|
|
|
|
(http-syslog (syslog-level notice) "error during connection negotiation~%")
|
|
|
|
(if rate-limiter
|
|
|
|
(rate-limit-close rate-limiter)))
|
|
|
|
(call-with-values
|
|
|
|
(lambda ()
|
|
|
|
(socket-address->internet-address (socket-remote-address sock)))
|
|
|
|
(lambda (host-address service-port)
|
2002-04-02 06:34:53 -05:00
|
|
|
(if (and rate-limiter *http-syslog?*)
|
2002-03-25 06:35:05 -05:00
|
|
|
(http-syslog (syslog-level info) "<~a>~a: concurrent request #~a~%"
|
2002-03-01 03:54:48 -05:00
|
|
|
(pid)
|
|
|
|
(format-internet-host-address host-address)
|
2002-03-25 06:35:05 -05:00
|
|
|
(rate-limiter-current-requests rate-limiter)))
|
|
|
|
|
|
|
|
(set-port-buffering (socket:outport sock) bufpol/none) ; No buffering
|
|
|
|
(fork-thread
|
|
|
|
(lambda ()
|
2002-08-26 05:46:11 -04:00
|
|
|
(set-port-buffering (current-input-port) bufpol/none)
|
|
|
|
(process-toplevel-request sock host-address options)
|
2002-04-02 06:34:53 -05:00
|
|
|
(if *http-syslog?*
|
2002-03-25 06:35:05 -05:00
|
|
|
(http-syslog (syslog-level debug) "<~a>~a [closing]~%"
|
|
|
|
(pid)
|
|
|
|
(format-internet-host-address host-address)))
|
|
|
|
(with-fatal-error-handler
|
|
|
|
(lambda (c decline)
|
2002-04-02 06:34:53 -05:00
|
|
|
(if *http-syslog?*
|
2002-03-25 06:35:05 -05:00
|
|
|
(http-syslog (syslog-level notice) "<~a>~a [error closing (~a)]~%"
|
|
|
|
(pid)
|
|
|
|
(format-internet-host-address host-address)
|
|
|
|
c)))
|
|
|
|
(close-socket sock))
|
|
|
|
(if rate-limiter
|
|
|
|
(rate-limit-close rate-limiter))
|
2002-04-02 06:34:53 -05:00
|
|
|
(if *http-syslog?*
|
2002-03-25 06:35:05 -05:00
|
|
|
(http-syslog (syslog-level info) "<~a>~a [closed]~%"
|
|
|
|
(pid)
|
|
|
|
(format-internet-host-address host-address)))))))))
|
|
|
|
port))))))
|
|
|
|
|
2000-09-26 10:35:26 -04:00
|
|
|
|
|
|
|
;;; Top-level http request processor
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
;;; Read, parse, and handle a single http request. The only thing that makes
|
|
|
|
;;; this complicated is handling errors -- as a server, we can't just let the
|
|
|
|
;;; standard error handlers toss us into a breakpoint. We have to catch the
|
2002-08-26 05:59:14 -04:00
|
|
|
;;; error, send an error response back to the client if we can, and then keep
|
2000-09-26 10:35:26 -04:00
|
|
|
;;; on trucking. This means using the S48's condition system to catch and
|
2002-08-26 11:14:10 -04:00
|
|
|
;;; handle the various errors, which introduces a major point of R5RS
|
|
|
|
;;; incompatibiliy -- R5RS has no exception system. So if you were to port
|
2000-09-26 10:35:26 -04:00
|
|
|
;;; this code to some other Scheme, you'd really have to sit down and think
|
|
|
|
;;; about this issue for a minute.
|
|
|
|
|
2002-03-01 03:54:48 -05:00
|
|
|
(define (process-toplevel-request sock host-address options)
|
2000-09-26 10:35:26 -04:00
|
|
|
;; This top-level error-handler catches *all* uncaught errors and warnings.
|
2002-08-26 05:59:14 -04:00
|
|
|
;; If the error condition is a reportable HTTP error, we send a response back
|
2000-09-26 10:35:26 -04:00
|
|
|
;; to the client. In any event, we abort the transaction, and return from
|
|
|
|
;; PROCESS-TOPLEVEL-REQUEST.
|
|
|
|
;;
|
|
|
|
;; We *oughta* map non-http-errors into replies anyway.
|
2002-08-26 05:46:11 -04:00
|
|
|
(with-fatal-error-handler*
|
|
|
|
(lambda (c decline)
|
|
|
|
(http-syslog (syslog-level notice) "<~a>~a: error: ~s~%"
|
|
|
|
(pid)
|
|
|
|
(format-internet-host-address host-address)
|
|
|
|
c)
|
|
|
|
(with-fatal-error-handler*
|
|
|
|
(lambda (c decline)
|
|
|
|
(http-syslog (syslog-level notice) "<~a>~a [error shutting down: ~s]~%"
|
|
|
|
(pid)
|
|
|
|
(format-internet-host-address host-address)
|
|
|
|
c))
|
|
|
|
(lambda ()
|
|
|
|
(shutdown-socket sock shutdown/sends+receives)
|
|
|
|
(http-syslog (syslog-level info) "<~a>~a [shut down]~%"
|
|
|
|
(pid)
|
|
|
|
(format-internet-host-address host-address))
|
|
|
|
(decline))))
|
|
|
|
(lambda ()
|
|
|
|
(call-with-values
|
|
|
|
(lambda ()
|
|
|
|
(with-fatal-error-handler*
|
|
|
|
(lambda (c decline)
|
|
|
|
(http-syslog (syslog-level notice) "<~a>~a: error: ~s~%"
|
|
|
|
(pid)
|
|
|
|
(format-internet-host-address host-address)
|
|
|
|
c)
|
|
|
|
(cond
|
|
|
|
((http-error? c)
|
2002-08-26 05:59:14 -04:00
|
|
|
(apply (lambda (status-code req . args)
|
2002-08-26 05:46:11 -04:00
|
|
|
(values req
|
|
|
|
(apply make-http-error-response
|
2002-08-26 05:59:14 -04:00
|
|
|
status-code req
|
2002-08-26 05:46:11 -04:00
|
|
|
args)))
|
|
|
|
(condition-stuff c)))
|
|
|
|
((fatal-syntax-error? c)
|
|
|
|
(values #f
|
2002-08-26 05:59:14 -04:00
|
|
|
(apply make-http-error-response http-status/bad-request
|
2002-08-26 05:46:11 -04:00
|
|
|
#f ; No request yet.
|
|
|
|
"Request parsing error -- report to client maintainer."
|
|
|
|
(condition-stuff c))))
|
|
|
|
(else
|
|
|
|
(decline))))
|
|
|
|
(lambda ()
|
|
|
|
(let* ((req (parse-http-request sock options))
|
|
|
|
(response ((httpd-options-path-handler options)
|
|
|
|
(http-url:path (request:url req))
|
|
|
|
req)))
|
|
|
|
(values req response)))))
|
|
|
|
(lambda (req response)
|
2002-08-26 07:11:40 -04:00
|
|
|
|
|
|
|
(send-http-response req response (socket:outport sock) options)
|
|
|
|
|
2002-08-26 05:59:14 -04:00
|
|
|
(http-log req http-status/ok))))))
|
2000-09-26 10:35:26 -04:00
|
|
|
|
|
|
|
;;;; HTTP request parsing
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
2002-05-26 13:56:56 -04:00
|
|
|
;;;; This code provides procedures to read requests from an input
|
|
|
|
;;;; port.
|
2000-09-26 10:35:26 -04:00
|
|
|
|
|
|
|
;;; Read and parse an http request from INPORT.
|
|
|
|
;;;
|
|
|
|
;;; Note: this parser parses the URI into an http URL record. If the URI
|
|
|
|
;;; isn't an http URL, the parser fails. This may not be right. There's
|
|
|
|
;;; nothing in the http protocol to prevent you from passing a non-http
|
|
|
|
;;; URI -- what this would mean, however, is not clear. Like so much of
|
|
|
|
;;; the Web, the protocols are redundant, underconstrained, and ill-specified.
|
|
|
|
|
2002-02-23 09:42:50 -05:00
|
|
|
(define (parse-http-request sock options)
|
2002-08-26 05:46:11 -04:00
|
|
|
(let ((line (read-crlf-line (socket:inport sock))))
|
2000-09-26 10:35:26 -04:00
|
|
|
;; Blat out some logging info.
|
2002-04-02 06:34:53 -05:00
|
|
|
(if *http-syslog?*
|
2002-02-21 11:12:22 -05:00
|
|
|
(call-with-values
|
|
|
|
(lambda ()
|
|
|
|
(socket-address->internet-address (socket-remote-address sock)))
|
|
|
|
(lambda (host-address service-port)
|
2002-03-25 06:35:05 -05:00
|
|
|
(http-syslog (syslog-level info) "<~a>~a: ~a~%"
|
2002-03-01 03:54:48 -05:00
|
|
|
(pid)
|
2002-02-21 11:12:22 -05:00
|
|
|
(format-internet-host-address host-address)
|
|
|
|
line))))
|
|
|
|
|
2000-09-26 10:35:26 -04:00
|
|
|
(if (eof-object? line)
|
|
|
|
(fatal-syntax-error "EOF while parsing request.")
|
|
|
|
|
|
|
|
(let* ((elts (string->words line)) ; Split at white-space.
|
2001-08-20 07:31:03 -04:00
|
|
|
(version (case (length elts)
|
2000-09-26 10:35:26 -04:00
|
|
|
((2) '(0 . 9))
|
|
|
|
((3) (parse-http-version (caddr elts)))
|
|
|
|
(else (fatal-syntax-error "Bad HTTP version.")))))
|
|
|
|
|
|
|
|
(let* ((meth (car elts))
|
|
|
|
(uri-string (cadr elts))
|
2002-02-23 09:42:50 -05:00
|
|
|
(url (parse-http-servers-url-fragment uri-string sock options))
|
2000-09-26 10:35:26 -04:00
|
|
|
(headers (if (equal? version '(0 . 9)) '()
|
2002-08-26 05:46:11 -04:00
|
|
|
(read-rfc822-headers (socket:inport sock)))))
|
2000-09-26 10:35:26 -04:00
|
|
|
(make-request meth uri-string url version headers sock))))))
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
;;; Parse the URL, but if it begins without the "http://host:port" prefix,
|
|
|
|
;;; interpolate one from SOCKET. It would sleazier but faster if we just
|
|
|
|
;;; computed the default host and port at server-startup time, instead of
|
|
|
|
;;; on every request.
|
|
|
|
|
2002-02-23 09:42:50 -05:00
|
|
|
(define (parse-http-servers-url-fragment uri-string socket options)
|
2000-09-26 10:35:26 -04:00
|
|
|
(receive (scheme path search frag-id) (parse-uri uri-string)
|
|
|
|
(if frag-id ; Can't have a #frag part.
|
|
|
|
(fatal-syntax-error "HTTP URL contains illegal #<fragment> suffix."
|
|
|
|
uri-string)
|
|
|
|
|
|
|
|
(if scheme
|
|
|
|
(if (string-ci=? scheme "http") ; Better be an http url.
|
|
|
|
(parse-http-url path search #f)
|
|
|
|
(fatal-syntax-error "Non-HTTP URL" uri-string))
|
|
|
|
|
|
|
|
;; Interpolate the userhost struct from our net connection.
|
|
|
|
(if (and (pair? path) (string=? (car path) ""))
|
|
|
|
(let* ((addr (socket-local-address socket))
|
2002-02-23 09:42:50 -05:00
|
|
|
(local-name (my-reported-fqdn addr options))
|
|
|
|
(portnum (my-reported-port addr options)))
|
2000-09-26 10:35:26 -04:00
|
|
|
(make-http-url (make-userhost #f #f
|
|
|
|
local-name
|
|
|
|
(number->string portnum))
|
2002-06-08 11:29:29 -04:00
|
|
|
(map unescape-uri (cdr path)) ; Skip initial /.
|
2000-09-26 10:35:26 -04:00
|
|
|
search
|
|
|
|
#f))
|
|
|
|
|
|
|
|
(fatal-syntax-error "Path fragment must begin with slash"
|
|
|
|
uri-string))))))
|
|
|
|
|
|
|
|
|
|
|
|
(define parse-http-version
|
|
|
|
(let ((re (make-regexp "^HTTP/([0-9]+)\\.([0-9]+)$"))
|
|
|
|
(lose (lambda (s) (fatal-syntax-error "Bad HTTP version" s))))
|
|
|
|
(lambda (vstring)
|
|
|
|
(let ((m (regexp-exec re vstring)))
|
|
|
|
(if m
|
|
|
|
(cons (or (string->number (match:substring m 1) 10) (lose vstring))
|
|
|
|
(or (string->number (match:substring m 2) 10) (lose vstring)))
|
|
|
|
(lose vstring))))))
|
|
|
|
|
|
|
|
|
|
|
|
;;; Split string into a list of whitespace-separated strings.
|
|
|
|
;;; This could have been trivially defined in scsh as (field-splitter " \t\n")
|
|
|
|
;;; but I hand-coded it because it's short, and I didn't want invoke the
|
|
|
|
;;; regexp machinery for something so simple.
|
|
|
|
|
2001-05-17 12:48:41 -04:00
|
|
|
(define non-whitespace (char-set-complement char-set:whitespace))
|
2000-09-26 10:35:26 -04:00
|
|
|
|
|
|
|
(define (string->words s)
|
|
|
|
(let recur ((start 0))
|
2002-04-21 14:55:18 -04:00
|
|
|
(cond ((string-index s non-whitespace start) =>
|
2000-09-26 10:35:26 -04:00
|
|
|
(lambda (start)
|
2002-04-21 14:55:18 -04:00
|
|
|
(cond ((string-index s char-set:whitespace start) =>
|
2000-09-26 10:35:26 -04:00
|
|
|
(lambda (end)
|
|
|
|
(cons (substring s start end)
|
|
|
|
(recur end))))
|
|
|
|
(else (list (substring s start (string-length s)))))))
|
|
|
|
(else '()))))
|
|
|
|
|
2002-08-26 07:11:40 -04:00
|
|
|
(define (send-http-headers response port)
|
2002-08-26 05:46:11 -04:00
|
|
|
(display server/protocol port)
|
|
|
|
(write-char #\space port)
|
|
|
|
(display (response-code response) port)
|
|
|
|
(write-char #\space port)
|
|
|
|
(display (response-message response) port)
|
|
|
|
(write-crlf port)
|
|
|
|
|
2002-08-26 07:11:40 -04:00
|
|
|
(send-http-header-fields
|
2002-08-26 11:14:10 -04:00
|
|
|
(list (cons 'server (string-append "Scheme Untergrund " sunet-version-identifier))
|
2002-08-26 05:46:11 -04:00
|
|
|
(cons 'content-type (response-mime response))
|
|
|
|
(cons 'date (time->http-date-string (response-seconds response))))
|
|
|
|
port)
|
2002-08-26 07:11:40 -04:00
|
|
|
(send-http-header-fields (response-extras response) port)
|
2002-08-26 05:46:11 -04:00
|
|
|
|
2002-08-26 07:11:40 -04:00
|
|
|
(write-crlf port))
|
|
|
|
|
|
|
|
(define (send-http-response request response port options)
|
|
|
|
|
|
|
|
(if (not (v0.9-request? request))
|
|
|
|
(send-http-headers response port)
|
2000-09-26 10:35:26 -04:00
|
|
|
|
2002-08-26 07:11:40 -04:00
|
|
|
(if (not (string=? (request:method request) "HEAD"))
|
|
|
|
(display-http-body (response-body response) port options))))
|
2002-08-26 05:46:11 -04:00
|
|
|
|
2002-08-26 07:11:40 -04:00
|
|
|
(define (send-http-header-fields headers port)
|
2002-08-26 05:46:11 -04:00
|
|
|
(for-each (lambda (pair)
|
|
|
|
(display (car pair) port)
|
|
|
|
(display ": " port)
|
|
|
|
(display (cdr pair) port)
|
|
|
|
(write-crlf port))
|
|
|
|
headers))
|
|
|
|
|
2002-08-26 07:21:53 -04:00
|
|
|
(define (time->http-date-string time)
|
|
|
|
(format-date "~A, ~d-~b-~y ~H:~M:~S GMT" (date time 0)))
|
|
|
|
|
2002-08-26 05:59:14 -04:00
|
|
|
;;; (make-http-error-response status-code req [message . extras])
|
2000-09-26 10:35:26 -04:00
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
2002-08-26 05:59:14 -04:00
|
|
|
;;; Take an http-error condition, and format it into a response to the client.
|
2000-09-26 10:35:26 -04:00
|
|
|
;;;
|
|
|
|
;;; As a special case, request REQ is allowed to be #f, meaning we haven't
|
|
|
|
;;; even had a chance to parse and construct the request. This is only used
|
2002-08-26 05:46:11 -04:00
|
|
|
;;; for 400 BAD-REQUEST error report.
|
2000-09-26 10:35:26 -04:00
|
|
|
|
2002-08-26 05:46:11 -04:00
|
|
|
;;; MAKE-HTTP-ERROR-RESPONSE is called from error handlers, so to avoid
|
2000-09-26 10:35:26 -04:00
|
|
|
;;; infinite looping, if an error occurs while it is running, we just
|
|
|
|
;;; silently return. (We no longer need to do this; I have changed
|
|
|
|
;;; WITH-FATAL-ERROR-HANDLER* so that this is not necessary, but I'll
|
|
|
|
;;; leave it in to play it safe.)
|
|
|
|
|
2002-08-26 05:59:14 -04:00
|
|
|
(define (make-http-error-response status-code req . args)
|
2002-08-26 05:46:11 -04:00
|
|
|
(ignore-errors
|
|
|
|
(lambda () ; Ignore errors -- see note above.
|
2002-08-26 05:59:14 -04:00
|
|
|
(apply really-make-http-error-response status-code req args))))
|
2000-09-26 10:35:26 -04:00
|
|
|
|
2002-08-26 05:59:14 -04:00
|
|
|
(define (really-make-http-error-response status-code req . args)
|
|
|
|
(http-log req status-code)
|
2001-04-27 12:19:34 -04:00
|
|
|
|
2002-08-26 05:46:11 -04:00
|
|
|
(let* ((message (and (pair? args) (car args)))
|
2000-09-26 10:35:26 -04:00
|
|
|
(extras (if (pair? args) (cdr args) '()))
|
|
|
|
|
2002-08-26 05:46:11 -04:00
|
|
|
(generic-title (lambda (port)
|
|
|
|
(title-html port
|
2002-08-26 05:59:14 -04:00
|
|
|
(status-code->text status-code))))
|
2002-08-26 05:46:11 -04:00
|
|
|
(close-html (lambda (port)
|
|
|
|
(for-each (lambda (x) (format port "<BR>~s~%" x)) extras)
|
|
|
|
(write-string "</BODY>\n" port)))
|
|
|
|
|
|
|
|
(create-response
|
|
|
|
(lambda (headers writer-proc)
|
2002-08-26 05:59:14 -04:00
|
|
|
(make-response status-code
|
|
|
|
(status-code->text status-code)
|
2002-08-26 05:46:11 -04:00
|
|
|
(time)
|
|
|
|
"text/html"
|
|
|
|
headers
|
|
|
|
(make-writer-body writer-proc)))))
|
2000-09-26 10:35:26 -04:00
|
|
|
|
2001-08-20 07:31:03 -04:00
|
|
|
(cond
|
2002-08-26 05:59:14 -04:00
|
|
|
;; This error response requires two args: message is the new URI: field,
|
2002-08-26 05:46:11 -04:00
|
|
|
;; and the first EXTRA is the older Location: field.
|
2002-08-26 05:59:14 -04:00
|
|
|
((or (= status-code http-status/moved-temp)
|
|
|
|
(= status-code http-status/moved-perm))
|
2002-08-26 05:46:11 -04:00
|
|
|
(create-response
|
|
|
|
(list (cons 'uri message)
|
|
|
|
(cons 'location (car extras)))
|
|
|
|
(lambda (port options)
|
|
|
|
(title-html port "Document moved")
|
|
|
|
(format port
|
|
|
|
"This document has ~A moved to a <A HREF=\"~A\">new location</A>.~%"
|
2002-08-26 05:59:14 -04:00
|
|
|
(if (= status-code http-status/moved-temp) "temporarily" "permanently")
|
2002-08-26 05:46:11 -04:00
|
|
|
message)
|
|
|
|
(close-html port))))
|
|
|
|
|
2002-08-26 05:59:14 -04:00
|
|
|
((= status-code http-status/bad-request)
|
2002-08-26 05:46:11 -04:00
|
|
|
(create-response
|
|
|
|
'()
|
|
|
|
(lambda (port options)
|
|
|
|
(generic-title port)
|
|
|
|
(write-string "<P>Client sent a query that this server could not understand.\n"
|
|
|
|
port)
|
|
|
|
(if message (format port "<BR>~%Reason: ~A~%" message))
|
|
|
|
(close-html port))))
|
|
|
|
|
2002-08-26 05:59:14 -04:00
|
|
|
((= status-code http-status/unauthorized)
|
2002-08-26 05:46:11 -04:00
|
|
|
(create-response
|
|
|
|
(list (cons 'WWW-Authenticate message)) ; Vas is das?
|
|
|
|
(lambda (port options)
|
|
|
|
(title-html port "Authorization Required")
|
|
|
|
(write-string "<P>Browser not authentication-capable or\n" port)
|
|
|
|
(write-string "authentication failed.\n" port)
|
|
|
|
(if message (format port "~a~%" message))
|
|
|
|
(close-html port))))
|
|
|
|
|
2002-08-26 05:59:14 -04:00
|
|
|
((= status-code http-status/forbidden)
|
2002-08-26 05:46:11 -04:00
|
|
|
(create-response
|
|
|
|
'()
|
|
|
|
(lambda (port options)
|
|
|
|
(title-html port "Request not allowed.")
|
|
|
|
(format port
|
|
|
|
"Your client does not have permission to perform a ~A~%"
|
|
|
|
(request:method req))
|
|
|
|
(format port "operation on url ~a.~%" (request:uri req))
|
|
|
|
(if message (format port "<P>~%~a~%" message))
|
|
|
|
(close-html port))))
|
2001-08-20 07:31:03 -04:00
|
|
|
|
2002-08-26 05:59:14 -04:00
|
|
|
((= status-code http-status/not-found)
|
2002-08-26 05:46:11 -04:00
|
|
|
(create-response
|
|
|
|
'()
|
|
|
|
(lambda (port options)
|
|
|
|
(title-html port "URL not found")
|
|
|
|
(write-string
|
|
|
|
"<P>The requested URL was not found on this server.\n"
|
|
|
|
port)
|
|
|
|
(if message (format port "<P>~%~a~%" message))
|
|
|
|
(close-html port))))
|
|
|
|
|
2002-08-26 05:59:14 -04:00
|
|
|
((= status-code http-status/internal-error)
|
2002-08-26 05:46:11 -04:00
|
|
|
(http-syslog (syslog-level error) "internal-error: ~A" message)
|
|
|
|
(create-response
|
|
|
|
'()
|
|
|
|
(lambda (port options)
|
|
|
|
(generic-title port)
|
|
|
|
(format port "The server encountered an internal error or
|
2000-09-26 10:35:26 -04:00
|
|
|
misconfiguration and was unable to complete your request.
|
|
|
|
<P>
|
|
|
|
Please inform the server administrator, ~A, of the circumstances leading to
|
|
|
|
the error, and time it occured.~%"
|
2002-08-26 05:46:11 -04:00
|
|
|
(httpd-options-server-admin options))
|
|
|
|
(if message (format port "<P>~%~a~%" message))
|
|
|
|
(close-html port))))
|
|
|
|
|
2002-08-26 05:59:14 -04:00
|
|
|
((= status-code http-status/not-implemented)
|
2002-08-26 05:46:11 -04:00
|
|
|
(create-response
|
|
|
|
'()
|
|
|
|
(lambda (port options)
|
|
|
|
(generic-title port)
|
|
|
|
(format port "This server does not currently implement
|
2000-09-26 10:35:26 -04:00
|
|
|
the requested method (~A).~%"
|
2002-08-26 05:46:11 -04:00
|
|
|
(request:method req))
|
|
|
|
(if message (format port "<P>~a~%" message))
|
|
|
|
(close-html port))))
|
|
|
|
|
|
|
|
(else
|
2002-08-26 05:59:14 -04:00
|
|
|
(http-syslog (syslog-level info) "Skipping unhandled status code ~A.~%" status-code)
|
2002-08-26 05:46:11 -04:00
|
|
|
(create-response
|
|
|
|
'()
|
|
|
|
(lambda (port options)
|
|
|
|
(generic-title port)
|
|
|
|
(close-html port)))))))
|
2000-09-26 10:35:26 -04:00
|
|
|
|
2002-08-26 07:21:53 -04:00
|
|
|
(define (title-html out message)
|
|
|
|
(format out "<HEAD>~%<TITLE>~%~A~%</TITLE>~%</HEAD>~%~%" message)
|
|
|
|
(format out "<BODY>~%<H1>~A</H1>~%" message))
|
2000-09-26 10:35:26 -04:00
|
|
|
|
|
|
|
;;; Return my Internet host name (my fully-qualified domain name).
|
|
|
|
;;; This works only if an actual resolver is behind host-info.
|
|
|
|
;;;
|
2002-08-24 13:26:56 -04:00
|
|
|
;;; In case of aliased names, you just might get the wrong one.
|
|
|
|
;;; Furthermore, you may get screwed in the presence of a server
|
|
|
|
;;; accelerator such as Squid.
|
|
|
|
|
|
|
|
|
|
|
|
(define my-reported-fqdn
|
|
|
|
(let ((fqdn-lock (make-lock))
|
|
|
|
(fqdn-cache #f)
|
|
|
|
(used-addr #f)
|
|
|
|
(used-options #f))
|
|
|
|
(lambda (addr options)
|
|
|
|
(obtain-lock fqdn-lock)
|
|
|
|
(let ((result
|
|
|
|
(if fqdn-cache
|
|
|
|
(or (and (equal? used-addr addr)
|
2002-08-24 13:45:37 -04:00
|
|
|
(equal? used-options options))
|
|
|
|
fqdn-cache)
|
|
|
|
(begin
|
|
|
|
(set! fqdn-cache (or (httpd-options-fqdn options)
|
|
|
|
(dns-lookup-ip (socket-address->string addr))
|
|
|
|
(host-info:name (host-info addr))))
|
|
|
|
(set! used-addr addr)
|
|
|
|
(set! used-options options)
|
|
|
|
fqdn-cache))))
|
2002-08-24 13:26:56 -04:00
|
|
|
(release-lock fqdn-lock)
|
2002-08-24 13:45:37 -04:00
|
|
|
|
2002-08-24 13:26:56 -04:00
|
|
|
result))))
|
2000-09-26 10:35:26 -04:00
|
|
|
|
2002-02-23 09:42:50 -05:00
|
|
|
(define (my-reported-port addr options)
|
|
|
|
(or (httpd-options-reported-port options)
|
2000-09-26 10:35:26 -04:00
|
|
|
(receive (ip-addr portnum) (socket-address->internet-address addr)
|
|
|
|
portnum)))
|
2002-08-24 12:46:34 -04:00
|
|
|
|