684 lines
27 KiB
TeX
684 lines
27 KiB
TeX
\chapter{HTTP server}\label{cha:httpd}
|
|
%
|
|
The SUnet HTTP Server is a complete industrial-strength implementation
|
|
of the HTTP 1.0 protocol. It is highly configurable and allows the writing
|
|
of dynamic web pages that run inside the server without going through
|
|
complicated and slow protocols like CGI or Fast/CGI.
|
|
|
|
\section{Starting and configuring the server}
|
|
|
|
All procedures described in this section are exported by the
|
|
\texttt{httpd} structure.
|
|
|
|
The Web server is started by calling the \ex{httpd} procedure, which takes
|
|
one argument, an options value:
|
|
|
|
\defun{httpd}{options}{\noreturn}
|
|
\begin{desc}
|
|
This procedure starts the server. The \var{options} argument
|
|
specifies various configuration parameters, explained below.
|
|
|
|
The server's basic loop is to wait on the port for a connection from
|
|
an HTTP client. When it receives a connection, it reads in and
|
|
parses the request into a special request data structure. Then the
|
|
server forks a thread which binds the current I/O ports to the
|
|
connection socket, and then hands off to the top-level
|
|
request handler (which must be specified in the options). The
|
|
request handler is responsible for actually serving
|
|
the request---it can be any arbitrary computation. Its output goes
|
|
directly back to the HTTP client that sent the request.
|
|
|
|
Before calling the request handler to service the request, the HTTP
|
|
server installs an error handler that fields any uncaught error,
|
|
sends an error reply to the client, and aborts the request
|
|
transaction. Hence any error caused by a request handler will be
|
|
handled in a reasonable and robust fashion.
|
|
\end{desc}
|
|
%
|
|
The \var{options} argument can be constructed through a number of procedures
|
|
with names of the form \texttt{with-\ldots}. Each of these procedures
|
|
either creates a fresh options value or adds a configuration parameter
|
|
to an old options argument. The configuration parameter value is
|
|
always the first argument, the (old) options value the optional second
|
|
one. Here they are:
|
|
|
|
\defun{with-port}{port [options]}{options}
|
|
\begin{desc}
|
|
This specifies the port on which the server listens. Defaults to 80.
|
|
\end{desc}
|
|
|
|
\defun{with-root-directory}{root-directory [options]}{options}
|
|
\begin{desc}
|
|
This specifies the current directory of the server. Note that this
|
|
is \emph{not} the document root directory. Defaults to \texttt{/}.
|
|
\end{desc}
|
|
|
|
\defun{with-fqdn}{fqdn [options]}{options}
|
|
\begin{desc}
|
|
This specifies the fully-qualified domain name the server uses in
|
|
automatically generated replies, or \ex{\#f} if the server should
|
|
query DNS for the fully-qualified domain name.. Defaults to \ex{\#f}.
|
|
\end{desc}
|
|
|
|
\defun{with-reported-port}{reported-port [options]}{options}
|
|
\begin{desc}
|
|
This specifies the port number the server uses in automatically
|
|
generated replies or \ex{\#f} if the reported port is the same as
|
|
the port the server is listening on. (This is useful if you're
|
|
running the server through an accelerating proxy.) Defaults to
|
|
\ex{\#f}.
|
|
\end{desc}
|
|
|
|
\defun{with-server-admin}{mail-address [options]}{options}
|
|
\begin{desc}
|
|
This specifies the email address of the server administrator the
|
|
server uses in automatically generated replies. Defaults to \ex{\#f}.
|
|
\end{desc}
|
|
|
|
\defun{with-request-handler}{request-handler [options]}{options}
|
|
\begin{desc}
|
|
This specifies the request handler of the server to which the server
|
|
delegates the actual work. More on that subject below in
|
|
Section~\ref{httpd:request-handlers}. This parameter must be specified.
|
|
\end{desc}
|
|
|
|
\defun{with-simultaneous-requests}{requests [options]}{options}
|
|
\begin{desc}
|
|
This specifies a limit on the number of simultaneous requests the
|
|
server servers. If that limit is exceeded during operation, the
|
|
server will hold off on new requests until the number of
|
|
simultaneous requests has sunk below the limit again. If this
|
|
parameter is \ex{\#f}, no limit is imposed. Defaults to \ex{\#f}.
|
|
\end{desc}
|
|
|
|
\defun{with-logfile}{logfile [options]}{options}
|
|
\begin{desc}
|
|
This specifies the name of a log file for the server where it writes
|
|
Common Log Format logging information. It can also be a port in
|
|
which case the information is logged to that port, or \ex{\#f} for
|
|
no logging. Defaults to \ex{\#f}.
|
|
|
|
To allow rotation of logfiles, the server re-opens the logfile
|
|
whenever it receives a \texttt{USR1} signal.
|
|
\end{desc}
|
|
|
|
\defun{with-syslog?}{syslog? [options]}{options}
|
|
\begin{desc}
|
|
This specifies whether the server will log information about
|
|
incoming to the Unix syslog facility. Defaults to \ex{\#t}.
|
|
\end{desc}
|
|
|
|
\defun{with-resolve-ip?}{resolve-ip? [options]}{options}
|
|
\begin{desc}
|
|
This specifies whether the server writes the domain names rather
|
|
than numerical IPs to the output log it produces. Defaults to
|
|
\ex{\#t}.
|
|
\end{desc}
|
|
|
|
To avoid paranthitis, the \ex{make-httpd-options} procedure eases the
|
|
construction of the options argument:
|
|
|
|
\defun{make-httpd-options}{transformer value \ldots}{options}
|
|
\begin{desc}
|
|
This constructs an options value from an argument list of parameter
|
|
transformers and parameter values. The arguments come in pairs,
|
|
each an option transformer from the list above, and a value for that
|
|
parameter. \ex{Make-httpd-options} returns the resulting options value.
|
|
\end{desc}
|
|
|
|
For example,
|
|
\begin{alltt}
|
|
(httpd (make-httpd-options
|
|
with-request-handler (rooted-file-handler "/usr/local/etc/httpd")
|
|
with-root-directory "/usr/local/etc/httpd"))
|
|
\end{alltt}
|
|
%
|
|
starts the server on port 80 with
|
|
\ex{/usr/local/etc/httpd} as its root directory and
|
|
lets it serve any file out from this directory.
|
|
% #### note about rooted-file-handler
|
|
|
|
|
|
\section{Requests}
|
|
\label{httpd:requests}
|
|
|
|
Request handlers operate on \textit{requests} which contain the
|
|
information needed to generate a page. The relevant procedures to
|
|
dissect requests are defined in the \texttt{httpd-requests} structure:
|
|
|
|
\defun{request?}{value}{boolean}
|
|
\defunx{request-method}{request}{string}
|
|
\defunx{request-uri}{request}{string}
|
|
\defunx{request-url}{request}{url}
|
|
\defunx{request-version}{request}{pair}
|
|
\defunx{request-headers}{request}{list}
|
|
\defunx{request-socket}{request}{socket}
|
|
\begin{desc}
|
|
The procedure inspect request values. \ex{Request?} is a predicate
|
|
for requests. \ex{Request-method} extracts the method of the HTTP
|
|
request; it's a string such as \verb|"GET"|, \verb|"PUT"|.
|
|
\ex{Request-uri} returns the escaped URI string as read from request
|
|
line. \ex{Request-url} returns an HTTP URL value (see the
|
|
description of the \ex{url} structure in \ref{cha:url}).
|
|
\ex{Request-version} returns \verb|(major . minor)| integer pair
|
|
representing the version specified in the HTTP request.
|
|
\ex{Request-headers} returns an association lists of header field
|
|
names and their values, each represented by a list of strings, one
|
|
for each line. \ex{Request-socket} returns the the socket connected
|
|
to the client.\footnote{Request handlers should not perform I/O on the
|
|
request record's socket. Request handlers are frequently called
|
|
recursively, and doing I/O directly to the socket might bypass a
|
|
filtering or other processing step interposed on the current I/O ports
|
|
by some superior request handler.}
|
|
\end{desc}
|
|
|
|
\section{Responses}
|
|
\label{sec:http-responses}
|
|
|
|
A path handler must return a \textit{response} value representing the
|
|
content to be sent to the client. The machinery presented here for
|
|
constructing responses lives in the \ex{httpd-responses} structure.
|
|
|
|
\defun{make-response}{status-code maybe-message seconds mime extras
|
|
body}{response}
|
|
\begin{desc}
|
|
This procedure constructs a response value. \var{Status-code} is an
|
|
HTTP status code (more on that below). \var{Maybe-message} is a a
|
|
message elaborating on the circumstances of the status code; it can
|
|
also be \sharpf{} meaning that the server should send a default
|
|
message associated with the status code. \var{Seconds} natural
|
|
number indicating the time the content was created, typically the
|
|
value of \verb|(time)|. \var{Mime} is a string indicating the MIME
|
|
type of the response (such as \verb|"text/html"| or
|
|
\verb|"application/octet-stream"|). \var{Extras} is an association
|
|
list with extra headers to be added to the response; its elements
|
|
are pairs, each of which consists of a symbol representing the field
|
|
name and a string representing the field value. \var{Body}
|
|
represents the body of the response; more on that below.
|
|
\end{desc}
|
|
|
|
\defun{make-redirect-response}{location}{response}
|
|
\begin{desc}
|
|
This is a helper procedure for constructing HTTP redirections. The
|
|
server will serve the new file indicated by \var{location}.
|
|
\var{Location} must be URI-encoded and begin with a slash.
|
|
\end{desc}
|
|
|
|
\defun{make-error-response}{status-code request [message] extras \ldots}{response}
|
|
\begin{desc}
|
|
This is a helper procedure for constructing error responses.
|
|
\var{code} is status code of the response (see below). \var{Request}
|
|
is the request that led to the error. \var{Message} is an optional
|
|
string containing an error message written in HTML, and \var{extras}
|
|
are further optional arguments containing further message lines to
|
|
be added to the web page that's generated.
|
|
|
|
\ex{Make-error-response} constructs a response value which generates
|
|
a web page containg a short explanatory message for the error at hand.
|
|
\end{desc}
|
|
|
|
\begin{table}[htb]
|
|
\centering
|
|
\begin{tabular}{|l|l|l|}
|
|
\hline
|
|
ok & 200 & OK\\\hline
|
|
created & 201 & Created\\\hline
|
|
accepted & 202 & Accepted\\\hline
|
|
prov-info & 203 & Provisional Information\\\hline
|
|
no-content & 204 & No Content\\\hline
|
|
|
|
mult-choice & 300 & Multiple Choices\\\hline
|
|
moved-perm & 301 & Moved Permanently\\\hline
|
|
moved-temp & 302 & Moved Temporarily\\\hline
|
|
method & 303 & Method (obsolete)\\\hline
|
|
not-mod & 304 & Not Modified\\\hline
|
|
|
|
bad-request & 400 & Bad Request\\\hline
|
|
unauthorized & 401 & Unauthorized\\\hline
|
|
payment-req & 402 & Payment Required\\\hline
|
|
forbidden & 403 & Forbidden\\\hline
|
|
not-found & 404 & Not Found\\\hline
|
|
method-not-allowed & 405 & Method Not Allowed\\\hline
|
|
none-acceptable & 406 & None Acceptable\\\hline
|
|
proxy-auth-required & 407 & Proxy Authentication Required\\\hline
|
|
timeout & 408 & Request Timeout\\\hline
|
|
conflict & 409 & Conflict\\\hline
|
|
gone & 410 & Gone\\\hline
|
|
internal-error & 500 & Internal Server Error\\\hline
|
|
not-implemented & 501 & Not Implemented\\\hline
|
|
bad-gateway & 502 & Bad Gateway\\\hline
|
|
service-unavailable & 503 & Service Unavailable\\\hline
|
|
gateway-timeout & 504 & Gateway Timeout\\\hline
|
|
\end{tabular}
|
|
\caption{HTTP status codes}
|
|
\label{tab:status-code-names}
|
|
\end{table}
|
|
|
|
\dfn{status-code}{\synvar{name}}{status-code}{syntax}
|
|
\defunx{name->status-code}{symbol}{status-code}
|
|
\defunx{status-code-number}{status-code}{integer}
|
|
\defunx{status-code-message}{status-code}{string}
|
|
\begin{desc}
|
|
The \ex{status-code} syntax returns a status code where
|
|
\synvar{name} is the name from Table~\ref{tab:status-code-names}.
|
|
\ex{Name->status-code} also returns a status code for a name
|
|
represented as a symbol. For a given status code,
|
|
\ex{status-code-number} extracts its number, and
|
|
\ex{status-code-message} extracts its associated default message.
|
|
\end{desc}
|
|
|
|
\section{Response Bodies}
|
|
\label{httpd:response-bodies}
|
|
|
|
A \textit{response body} represents the body of an HTTP response.
|
|
There are several types of response bodies, depending on the
|
|
requirements on content generation.
|
|
|
|
\defun{make-writer-body}{proc}{body}
|
|
\begin{desc}
|
|
This constructs a response body from a \textit{writer}---a procedure
|
|
that prints the page contents to a port. The \var{proc} argument
|
|
must be a procedure accepting an output port (to which \var{proc}
|
|
prints the body) and the options value passed to the \ex{httpd}
|
|
invocation.
|
|
\end{desc}
|
|
|
|
\defun{make-reader-writer-body}{proc}{body}
|
|
\begin{desc}
|
|
This constructs a response body from a \textit{reader/writer}---a
|
|
procedure that prints the page contents to a port, possibly after
|
|
reading input from the socket of the HTTP connection. The
|
|
\var{proc} argument must be a procedure accepting three arguments:
|
|
an input port (associated with the HTTP connection socket), an
|
|
output port (to which \var{proc} prints the body), and the options
|
|
value passed to the \ex{httpd} invocation.
|
|
\end{desc}
|
|
|
|
\section{Request Handlers}
|
|
\label{httpd:request-handlers}
|
|
|
|
A request handler generates the actual content for a request; request
|
|
handlers form a simple algebra and may be combined and composed in
|
|
various ways.
|
|
|
|
|
|
A request handler is a procedure of two arguments like this:
|
|
\defun{request-handler}{path req}{response}
|
|
\begin{desc}
|
|
\var{Req} is a request. The \semvar{path} argument is the URL's
|
|
path, parsed and split at slashes into a string list. For example,
|
|
if the Web client dereferences URL
|
|
%
|
|
\begin{verbatim}
|
|
http://clark.lcs.mit.edu:8001/h/shivers/code/web.tar.gz
|
|
\end{verbatim}
|
|
then the server would pass the following path to the top-level
|
|
handler:
|
|
%
|
|
\begin{verbatim}
|
|
("h" "shivers" "code" "web.tar.gz")
|
|
\end{verbatim}
|
|
%
|
|
The \var{path} argument's pre-parsed representation as a string
|
|
list makes it easy for the request handler to implement recursive
|
|
operations dispatch on URL paths.
|
|
|
|
The request handler must return an HTTP response.
|
|
\end{desc}
|
|
|
|
\subsection{Basic Request Handlers}
|
|
|
|
The web server comes with a useful toolbox of basic request handlers
|
|
that can be used and built upon. The following procedures are
|
|
exported by the \ex{httpd\=basic\=handlers} structure:
|
|
|
|
\defvar{null-request-handler}{request-handler}
|
|
\begin{desc}
|
|
This request handler always generated a \ex{not-found} error
|
|
response, no patter what the request is.
|
|
\end{desc}
|
|
|
|
\defun{make-predicate-handler}{predicate handler
|
|
default-handler}{request-handler}
|
|
\begin{desc}
|
|
The request handler returned by this procedure first calls
|
|
\var{predicate} on its path and request; it then acts like
|
|
\var{handler} if the predicate returned a true vale, and like
|
|
\var{default-handler} if the predicate returned \sharpf.
|
|
\end{desc}
|
|
|
|
\defun{make-host-name-handler}{hostname handler default-handler}{request-handler}
|
|
\begin{desc}
|
|
The request handler returned by this procedure compares the host
|
|
name specified in the request with \var{hostname}: if they match, it
|
|
acts like \var{handler}, otherwise, it acts like
|
|
\var{default-handler}.
|
|
\end{desc}
|
|
|
|
\defun{make-path-predicate-handler}{predicate handler
|
|
default-handler}{request-handler}
|
|
\begin{desc}
|
|
The request handler returned by this procedure first calls
|
|
\var{predicate} on its path; it then acts like \var{handler} if the
|
|
predicate returned a true vale, and like \var{default-handler} if
|
|
the predicate returned \sharpf.
|
|
\end{desc}
|
|
|
|
\defun{make-path-prefix-handler}{path-prefix handler default-handler}{request-handler}
|
|
\begin{desc}
|
|
This constructs a request handler that calls \var{handler} on its
|
|
argument if \var{path-prefix} (a string) is the first element of the
|
|
requested path; it calls \var{handler} on the rest of the path and
|
|
the original request. Otherwise, the handler acts like
|
|
\var{default-handler}.
|
|
\end{desc}
|
|
|
|
\defun{alist-path-dispatcher}{handler-alist default-handler}{request-handler}
|
|
\begin{desc}
|
|
This procedure takes as arguments an alist mapping strings to path
|
|
handlers, and a default request handler, and returns a handler that
|
|
dispatches on its path argument. When the new request handler is
|
|
applied to a path
|
|
\begin{verbatim}
|
|
("foo" "bar" "baz")
|
|
\end{verbatim}
|
|
it uses the
|
|
first element of the path---\ex{foo}---to index into the
|
|
alist. If it finds an associated request handler in the alist, it
|
|
hands the request off to that handler, passing it the tail of the
|
|
path, in this case
|
|
\begin{verbatim}
|
|
("bar" "baz")
|
|
\end{verbatim}
|
|
%
|
|
On the other hand, if the path is
|
|
empty, or the alist search does not yield a hit, we hand off to the
|
|
default path handler, passing it the entire original path,
|
|
\begin{verbatim}
|
|
("foo" "bar" "baz")
|
|
\end{verbatim}
|
|
%
|
|
This procedure is how you say: ``If the first element of the URL's
|
|
path is `foo', do X; if it's `bar', do Y; otherwise, do Z.''
|
|
The slash-delimited URI path structure implies an associated tree of
|
|
names. The request-handler system and the alist dispatcher allow you to
|
|
procedurally define the server's response to any arbitrary subtree
|
|
of the path space.
|
|
|
|
Example: A typical top-level request handler is
|
|
\begin{alltt}
|
|
(define ph
|
|
(alist-path-dispatcher
|
|
`(("h" . ,(home-dir-handler "public\_html"))
|
|
("cgi-bin" . ,(cgi-handler "/usr/local/etc/httpd/cgi-bin"))
|
|
("seval" . ,seval-handler))
|
|
(rooted-file-handler "/usr/local/etc/httpd/htdocs")))
|
|
\end{alltt}
|
|
|
|
This means:
|
|
\begin{itemize}
|
|
\item If the path looks like \ex{("h"\ob{} "shivers"\ob{}
|
|
"code"\ob{} "web.\ob{}tar.\ob{}gz")}, pass the path
|
|
\ex{("shivers"\ob{} "code"\ob{} "web.\ob{}tar.\ob{}gz")} to a
|
|
home-directory request handler.
|
|
\item If the path looks like \ex{("cgi-\ob{}bin"\ob{} "calendar")},
|
|
pass ("calendar") off to the CGI request handler.
|
|
\item If the path looks like \ex{("seval"\ob{} \ldots)}, the tail
|
|
of the path is passed off to the code-uploading \ex{seval} path
|
|
handler.
|
|
\item Otherwise, the whole path is passed to a rooted file handler,
|
|
who will convert it into a filename, rooted at
|
|
\ex{/usr/\ob{}lo\ob{}cal/\ob{}etc/\ob{}httpd/\ob{}htdocs},
|
|
and serve that file.
|
|
\end{itemize}
|
|
\end{desc}
|
|
|
|
\subsection{Static Content Request Handlers}
|
|
|
|
The request handlers described in this section are for serving static
|
|
content off directory trees in the file system. They live in the
|
|
\ex{httpd-file-directory-handlers} structure.
|
|
|
|
The request handlers in this section eventually call an internal
|
|
procedure named \ex{file\=serve} for serving files which implements a
|
|
simple directory-generation service using the following rules:
|
|
\begin{itemize}
|
|
\item If the filename has the form of a directory (i.e., it ends with
|
|
a slash), then \ex{file\=serve} actually looks for a file named
|
|
\ex{index.html} in that directory.
|
|
\item If the filename names a directory, but is not in directory form
|
|
(i.e., it doesn't end in a slash, as in
|
|
``\ex{/usr\ob{}in\ob{}clu\ob{}de}'' or ``\ex{/usr\ob{}raj}''),
|
|
then \ex{file\=serve} sends back a ``301 moved permanently''
|
|
message, redirecting the client to a slash-terminated version of the
|
|
original URL. For example, the URL
|
|
\ex{http://\ob{}clark.\ob{}lcs.\ob{}mit.\ob{}edu/\ob{}~shi\ob{}vers}
|
|
would be redirected to
|
|
\ex{http://\ob{}clark.\ob{}lcs.\ob{}mit.\ob{}edu/\ob{}~shi\ob{}vers/}
|
|
\item If the filename names a regular file, it is served to the
|
|
client.
|
|
\end{itemize}
|
|
%
|
|
The \ex{httpd-file-directory-handlers} all take an options value as an
|
|
argument, similar to the options for \ex{httpd} itself.
|
|
|
|
The \var{options} argument can be constructed through a number of procedures
|
|
with names of the form \texttt{with-\ldots}. Each of these procedures
|
|
either creates a fresh options value or adds a configuration parameter
|
|
to an old options argument. The configuration parameter value is
|
|
always the first argument, the (old) options value the optional second
|
|
one. Here they are:
|
|
|
|
\defun{with-file-name->content-type}{proc [options]}{options}
|
|
\begin{desc}
|
|
This specifies a procedure for determining the MIME content type
|
|
(``\ex{text/html},'' ``\ex{application/octet-stream}'' etc.)
|
|
from a file name. \var{Proc} takes a file name as an argument and
|
|
must return a string. (This is relevant in directory listings.) The default is a procedure able to handle the
|
|
more common file extensions.
|
|
\end{desc}
|
|
|
|
\defun{with-file-name->content-encoding}{proc [options]}{options}
|
|
\begin{desc}
|
|
This specifies a procedure for determining the MIME content encoding
|
|
(if the file is compressed, gzipped, etc.) from a file name.
|
|
(This is relevant in directory listings.)
|
|
\var{Proc} takes a file name as an argument and must return two
|
|
values: the equivalent, unencoded file name (i.e., without the
|
|
trailing \ex{.Z} or \ex{.gz}) and a string representing the content
|
|
encoding.
|
|
\end{desc}
|
|
|
|
\defun{with-file-name->icon-url}{proc [options]}{options}
|
|
\begin{desc}
|
|
This specifies a procedure for determining the icon to be displayed
|
|
next to a file name in a directory listing.
|
|
\var{Proc} takes a file name as an argument and must return a URI
|
|
representing the corresponding icon.
|
|
\end{desc}
|
|
|
|
\defun{with-blank-icon-url}{file-name [options]}{options}
|
|
\begin{desc}
|
|
This specifies a file name for the special icon that must be as wide
|
|
as the icons returned by the previous procedure but that is blank.
|
|
\end{desc}
|
|
|
|
\defun{with-back-icon-url}{file-name [options]}{options}
|
|
\begin{desc}
|
|
This specifies a file name for the special icon that is displayed
|
|
next to the ``parent directory'' link in directory listings.
|
|
\end{desc}
|
|
|
|
\defun{with-unknown-icon-url}{file-name [options]}{options}
|
|
\begin{desc}
|
|
This specifies a file name for the special icon that is displayed
|
|
next to the unknown entries in directory listings.
|
|
\end{desc}
|
|
|
|
The \ex{make-file-directory-options} procedure eases the construction
|
|
of the options argument:
|
|
|
|
\defun{make-file-directory-options}{transformer value \ldots}{options}
|
|
\begin{desc}
|
|
This constructs an options value from an argument list of parameter
|
|
transformers and parameter values. The arguments come in pairs,
|
|
each an option transformer from the list above, and a value for that
|
|
parameter. \ex{Make-file-directory-options} returns the resulting
|
|
options value.
|
|
\end{desc}
|
|
%
|
|
Here are procedure for constructing static content request handlers:
|
|
%
|
|
\defun{rooted-file-handler}{root options}{request-handler}
|
|
\begin{desc}
|
|
This returns a request handler that serves files from a particular
|
|
root in the file system. Only the \ex{GET} operation is provided.
|
|
The path argument passed to the handler is converted into a
|
|
filename, and appended to \var{root}. The file name is checked for
|
|
\ex{..} components, and the transaction is aborted if it does.
|
|
Otherwise, the file is served to the client.
|
|
\end{desc}
|
|
|
|
\defun{rooted-file-or-directory-handler}{root options}{request-handler}
|
|
\begin{desc}
|
|
Dito, but also serve directory indices for directories without
|
|
\ex{index.html}.
|
|
\end{desc}
|
|
|
|
\defun{home-dir-handler}{subdir options}{request-handler}
|
|
\begin{desc}
|
|
This procedure builds a request handler that does basic file serving
|
|
out of home directories. If the resulting \var{request-handler} is
|
|
passed a path of the form \ex{(\var{user} . \var{file-path})}, then it serves the file
|
|
\ex{\var{subdir}/\var{file-path}} inside the user's home directory.
|
|
|
|
The request handler only handles GET requests; the filename is not
|
|
allowed to contain \ex{..} elements.
|
|
\end{desc}
|
|
|
|
\defun{tilde-home-dir-handler}{subdir default-request-handler options}{request-handler}
|
|
\begin{desc}
|
|
This returns request handler that examines the car of the path. If
|
|
it is a string beginning with a tilde, e.g., \ex{"~ziggy"}, then the
|
|
string is taken to mean a home directory, and the request is served
|
|
similarly to a home-dir-handler request handler. Otherwise, the
|
|
request is passed off in its entirety to the
|
|
\var{default-request-handler}.
|
|
\end{desc}
|
|
|
|
\section{CGI Server}
|
|
|
|
The procedure(s) described here live in the \ex{httpd-cgi-handlers}
|
|
structure.
|
|
|
|
\defun{cgi-handler}{bin-dir [cgi-bin-path]}{request-handler}
|
|
\begin{desc}
|
|
Returns a request handler for CGI scripts located in
|
|
\var{bin-dir}. \var{Cgi-bin-dir} specifies the value of the
|
|
\ex{PATH} variable of the environment the CGI scripts run in. It defaults
|
|
to
|
|
\begin{verbatim}
|
|
/bin:/usr/bin:/usr/ucb:/usr/bsd:/usr/local/bin
|
|
\end{verbatim}
|
|
The CGI scripts are called as specified by CGI/1.1\footnote{see
|
|
\url{http://hoohoo.ncsa.uiuc.edu/cgi/interface.html} for a sort of
|
|
specification.}.
|
|
|
|
Note that the CGI handler looks at the name of the CGI script to
|
|
determine how it should be handled:
|
|
\begin{itemize}
|
|
\item If the name of the script starts with `\ex{nph-}', its reply
|
|
is read, the RFC~822-fields like \ex{Content-Type} and \ex{Status}
|
|
are parsed and the client is sent back a real HTTP reply,
|
|
containing the rest of the script's output.
|
|
|
|
\item If the name of the script doesn't start with `\ex{nph-}',
|
|
its output is sent back to the client directly. If its return code
|
|
is not zero, an error message is generated.
|
|
\end{itemize}
|
|
\end{desc}
|
|
|
|
\section{Scheme-Evaluating Request Handlers}
|
|
|
|
The \ex{httpd-seval-handlers} structure contains a handler which
|
|
demonstrates how to safely evaluate Scheme code uploaded from the
|
|
client to the server.
|
|
|
|
\defvar{seval-handler}{request-handler}
|
|
\begin{desc}
|
|
This request handler is suitable for receiving code entered into an
|
|
HTML text form. The Scheme code being uploaded is being \ex{POST}ed
|
|
to the server (from a form). The code should be URI-encoded in the
|
|
URL as \texttt{program=}$\left<\mathrm{stuff}\right>$.
|
|
$\mathrm{stuff}$ must be an (URI-encoded) Scheme expression which
|
|
the handler evaluates in a separate subprocess. (It waits for 10
|
|
seconds for a result, then kills the subprocess.) The handler then
|
|
prints the return values of the Scheme code.
|
|
\end{desc}
|
|
|
|
The following structures define environments that are \RnRS without
|
|
features that could examine or effect the file system. You can also
|
|
use them as models of how to execute code in other protected
|
|
environments in \scm.
|
|
|
|
\subsection{The \protect{\texttt{loser}} structure}
|
|
The \ex{loser} package exports only one procedure:
|
|
|
|
\begin{defundesc}{loser}{name}{nothing}
|
|
Raises an error like ``Illegal call \var{name}''.
|
|
\end{defundesc}
|
|
|
|
\subsection{The \protect{\texttt{toothless}} structure}
|
|
The \ex{toothless} structure contains everything of \RnRS except
|
|
that following procedure cause an error if called:
|
|
\begin{itemize}
|
|
\item \ex{call-with-input-file}
|
|
\item \ex{call-with-output-file}
|
|
\item \ex{load}
|
|
\item \ex{open-input-file}
|
|
\item \ex{open-output-file}
|
|
\item \ex{transcript-on}
|
|
\item \ex{with-input-from-file}
|
|
\item \ex{with-input-to-file}
|
|
\item \ex{eval}
|
|
\item \ex{interaction-environment}
|
|
\item \ex{scheme-report-environment}
|
|
\end{itemize}
|
|
|
|
\subsection{The \protect{\texttt{toothless-eval}} structure}
|
|
|
|
\begin{defundesc}{eval-safely} {expression} {any result}
|
|
Creates a brand-new structure, imports the \ex{toothless} structure,
|
|
and evaluates \semvar{expression} in it. When the evaluation is
|
|
done, the environment is thrown away, so \semvar{expression}'s
|
|
side-effects don't persist from one \ex{eval\=safely} call to the
|
|
next. If \semvar{expression} raises an error exception,
|
|
\ex{eval-safely} returns \sharpf.
|
|
\end{defundesc}
|
|
|
|
\section{Writing Request Handlers}
|
|
|
|
\subsection{Parsing HTML Forms}
|
|
|
|
In HTML forms, field data are turned into a single string, of the form
|
|
\texttt{\synvar{name}=\synvar{val}\&\synvar{name}=\synvar{val}\ldots}.
|
|
The \ex{parse-html-forms} structure provides simple functionality to
|
|
parse these strings.
|
|
|
|
\defun{parse-html-form-query}{string}{alist}
|
|
\begin{desc}
|
|
This parses \verb|"foo=x&bar=y"| into \verb|(("foo" . "x") ("bar" .
|
|
"y"))|. Substrings are plus-decoded (i.-e.\ plus characters are
|
|
turned into spaces) and then URI-decoded.
|
|
|
|
This implementation is
|
|
slightly sleazy as it will successfully parse a string like
|
|
\verb|"a&b=c&d=f"| into \verb|(("a&b" . "c") ("d" . "f"))| without
|
|
a complaint.
|
|
\end{desc}
|
|
|
|
%%% Local Variables:
|
|
%%% mode: latex
|
|
%%% TeX-master: "man"
|
|
%%% End:
|