Revision: 4655
Author: hans
URL: http://bknr.net/trac/changeset/4655
Automatically set the charset= attribute in the Content-Type: header
when a string has been returned by the handler. With this change, it
is sufficient to change *HUNCHENTOOT-DEFAULT-EXTERNAL-FORMAT* to the
desired default charset used for responses.
U trunk/thirdparty/hunchentoot/headers.lisp
U trunk/thirdparty/hunchentoot/specials.lisp
Modified: trunk/thirdparty/hunchentoot/headers.lisp
===================================================================
--- trunk/thirdparty/hunchentoot/headers.lisp 2011-02-15 21:45:25 UTC (rev 4654)
+++ trunk/thirdparty/hunchentoot/headers.lisp 2011-02-16 11:47:44 UTC (rev 4655)
@@ -53,6 +53,16 @@
(:method (key value stream)
(write-header-line key (princ-to-string value) stream)))
+(defun maybe-add-charset-to-content-type-header (content-type external-format)
+ "Given the contents of a CONTENT-TYPE header, add a charset=
+ attribute describing the given EXTERNAL-FORMAT if no charset=
+ attribute is already present and the content type is a text content
+ type. Returns the augmented content type."
+ (if (and (cl-ppcre:scan "(?i)^text" content-type)
+ (not (cl-ppcre:scan "(?i);\\s*charset=" content-type)))
+ (format nil "~A; charset=~(~A~)" content-type (flex:external-format-name external-format))
+ content-type))
+
(defun start-output (return-code &optional (content nil content-provided-p))
"Sends all headers and maybe the content body to
*HUNCHENTOOT-STREAM*. Returns immediately and does nothing if called
@@ -115,7 +125,9 @@
(setq content (maybe-rewrite-urls-for-session content)))
(when (stringp content)
;; if the content is a string, convert it to the proper external format
- (setf content (string-to-octets content :external-format (reply-external-format*))))
+ (setf content (string-to-octets content :external-format (reply-external-format*))
+ (content-type*) (maybe-add-charset-to-content-type-header (content-type*)
+ (reply-external-format*))))
(when content
;; whenever we know what we're going to send out as content, set
;; the Content-Length header properly; maybe the user specified
Modified: trunk/thirdparty/hunchentoot/specials.lisp
===================================================================
--- trunk/thirdparty/hunchentoot/specials.lisp 2011-02-15 21:45:25 UTC (rev 4654)
+++ trunk/thirdparty/hunchentoot/specials.lisp 2011-02-16 11:47:44 UTC (rev 4655)
@@ -114,11 +114,11 @@
"The three-character names of the twelve months - needed for cookie
date format.")
-(defvar *rewrite-for-session-urls* t
+(defparameter *rewrite-for-session-urls* t
"Whether HTML pages should possibly be rewritten for cookie-less
session-management.")
-(defvar *content-types-for-url-rewrite*
+(defparameter *content-types-for-url-rewrite*
'("text/html" "application/xhtml+xml")
"The content types for which url-rewriting is OK. See
*REWRITE-FOR-SESSION-URLS*.")
@@ -154,20 +154,20 @@
(defvar *session-db* nil
"The default \(global) session database.")
-(defvar *session-max-time* #.(* 30 60)
+(defparameter *session-max-time* #.(* 30 60)
"The default time \(in seconds) after which a session times out.")
-(defvar *session-gc-frequency* 50
+(defparameter *session-gc-frequency* 50
"A session GC \(see function SESSION-GC) will happen every
*SESSION-GC-FREQUENCY* requests \(counting only requests which create
a new session) if this variable is not NIL. See SESSION-CREATED.")
-(defvar *use-user-agent-for-sessions* t
+(defparameter *use-user-agent-for-sessions* t
"Whether the 'User-Agent' header should be encoded into the session
string. If this value is true, a session will cease to be accessible
if the client sends a different 'User-Agent' header.")
-(defvar *use-remote-addr-for-sessions* nil
+(defparameter *use-remote-addr-for-sessions* nil
"Whether the client's remote IP \(as returned by REAL-REMOTE-ADDR)
should be encoded into the session string. If this value is true, a
session will cease to be accessible if the client's remote IP changes.
@@ -175,39 +175,42 @@
This might for example be an issue if the client uses a proxy server
which doesn't send correct 'X_FORWARDED_FOR' headers.")
-(defvar *default-content-type* "text/html; charset=iso-8859-1"
- "The default content-type header which is returned to the client.")
+(defparameter *default-content-type* "text/html"
+ "The default content-type header which is returned to the client.
+If this is text content type, the character set used for encoding the
+response will automatically be added to the content type in a
+``charset'' attribute.")
-(defvar *methods-for-post-parameters* '(:post)
+(defparameter *methods-for-post-parameters* '(:post)
"A list of the request method types \(as keywords) for which
Hunchentoot will try to compute POST-PARAMETERS.")
-(defvar *header-stream* nil
+(defparameter *header-stream* nil
"If this variable is not NIL, it should be bound to a stream to
which incoming and outgoing headers will be written for debugging
purposes.")
-(defvar *show-lisp-errors-p* nil
+(defparameter *show-lisp-errors-p* nil
"Whether Lisp errors in request handlers should be shown in HTML output.")
-(defvar *show-lisp-backtraces-p* t
+(defparameter *show-lisp-backtraces-p* t
"Whether Lisp errors shown in HTML output should contain backtrace information.")
-(defvar *log-lisp-errors-p* t
+(defparameter *log-lisp-errors-p* t
"Whether Lisp errors in request handlers should be logged.")
-(defvar *log-lisp-backtraces-p* t
+(defparameter *log-lisp-backtraces-p* t
"Whether Lisp backtraces should be logged. Only has an effect if
*LOG-LISP-ERRORS-P* is true as well.")
-(defvar *log-lisp-warnings-p* t
+(defparameter *log-lisp-warnings-p* t
"Whether Lisp warnings in request handlers should be logged.")
-(defvar *lisp-errors-log-level* :error
+(defparameter *lisp-errors-log-level* :error
"Log level for Lisp errors. Should be one of :ERROR \(the default),
:WARNING, or :INFO.")
-(defvar *lisp-warnings-log-level* :warning
+(defparameter *lisp-warnings-log-level* :warning
"Log level for Lisp warnings. Should be one of :ERROR, :WARNING
\(the default), or :INFO.")
@@ -219,7 +222,7 @@
"A global lock to prevent concurrent access to the log file used by
the ACCEPTOR-LOG-ACCESS function.")
-(defvar *catch-errors-p* t
+(defparameter *catch-errors-p* t
"Whether Hunchentoot should catch and log errors \(or rather invoke
the debugger).")
@@ -243,7 +246,7 @@
#+:openmcl "http://openmcl.clozure.com/"
"A link to the website of the underlying Lisp implementation.")
-(defvar *tmp-directory*
+(defparameter *tmp-directory*
#+(or :win32 :mswindows) "c:\\hunchentoot-temp\\"
#-(or :win32 :mswindows) "/tmp/hunchentoot/"
"Directory for temporary files created by MAKE-TMP-FILE-NAME.")
@@ -261,13 +264,13 @@
"A FLEXI-STREAMS external format used internally for logging and to
encode cookie values.")
-(defvar *hunchentoot-default-external-format* +latin-1+
+(defparameter *hunchentoot-default-external-format* +utf-8+
"The external format used to compute the REQUEST object.")
(defconstant +buffer-length+ 8192
"Length of buffers used for internal purposes.")
-(defvar *default-connection-timeout* 20
+(defparameter *default-connection-timeout* 20
"The default connection timeout used when an acceptor is reading
from and writing to a socket stream.")
@@ -292,7 +295,7 @@
;; see <http://common-lisp.net/project/hyperdoc/>
;; and <http://www.cliki.net/hyperdoc>
-(defvar *hyperdoc-base-uri* "http://weitz.de/hunchentoot/")
+(defparameter *hyperdoc-base-uri* "http://weitz.de/hunchentoot/")
(let ((exported-symbols-alist
(loop for symbol being the external-symbols of :hunchentoot
Revision: 4653
Author: hans
URL: http://bknr.net/trac/changeset/4653
Mention RESTAS web framework in documentation
U trunk/thirdparty/hunchentoot/doc/index.xml
Modified: trunk/thirdparty/hunchentoot/doc/index.xml
===================================================================
--- trunk/thirdparty/hunchentoot/doc/index.xml 2011-02-15 16:59:44 UTC (rev 4652)
+++ trunk/thirdparty/hunchentoot/doc/index.xml 2011-02-15 21:45:07 UTC (rev 4653)
@@ -339,6 +339,10 @@
<a href="http://weitz.de/cl-webdav/">CL-WEBDAV</a> is a <a href="http://webdav.org/">WebDAV</a>
server based on Hunchentoot.
</li>
+ <li>
+ <a href="http://restas.lisper.ru/">RESTAS</a> is a web
+ framework based on Hunchentoot.
+ </li>
</ul>
</clix:chapter>
Revision: 4650
Author: hans
URL: http://bknr.net/trac/changeset/4650
Further status message generation improvements
Provide more substitution variables (${error} and ${backtrace} in particular)
Add simple (and ugly) internal server error template
Rename ACCEPTOR-HANDLE-RETURN-CODE to ACCEPTOR-STATUS-MESSAGE, make it return
the HTML message, remove CONTENT argument to simplify things
Update documentation
U trunk/thirdparty/hunchentoot/acceptor.lisp
U trunk/thirdparty/hunchentoot/doc/index.xml
U trunk/thirdparty/hunchentoot/headers.lisp
U trunk/thirdparty/hunchentoot/request.lisp
U trunk/thirdparty/hunchentoot/taskmaster.lisp
U trunk/thirdparty/hunchentoot/test/test-handlers.lisp
A trunk/thirdparty/hunchentoot/www/errors/500.html
Modified: trunk/thirdparty/hunchentoot/acceptor.lisp
===================================================================
--- trunk/thirdparty/hunchentoot/acceptor.lisp 2011-02-10 22:20:18 UTC (rev 4649)
+++ trunk/thirdparty/hunchentoot/acceptor.lisp 2011-02-11 10:10:17 UTC (rev 4650)
@@ -543,84 +543,110 @@
(with-debugger
(acceptor-dispatch-request *acceptor* *request*))))
-(defgeneric acceptor-handle-return-code (acceptor http-return-code content)
+(defgeneric acceptor-status-message (acceptor http-status-code &key &allow-other-keys)
(:documentation
"This function is called after the request's handler has been
- invoked, before starting to send any output to the client. It
- converts the HTTP return code that has been determined as the
- result of the handler invocation into a content body sent to the
- user. The content generated by the handler is passed to this
- function as CONTENT argument. For positive return
- codes (i.e. ``200 OK''), the CONTENT is typically just sent to the
- client. For other return codes, the content can be ignored and/or
- processed, depending on the requirements of the acceptor class.
- Note that the CONTENT argument can be NIL if the handler wants to
- send the data to the client stream itself.
+ invoked to convert the HTTP-STATUS-CODE to a HTML message to be
+ displayed to the user. If this function returns a string, that
+ string is sent to the client instead of the content produced by the
+ handler, if any.
If an ERROR-TEMPLATE-DIRECTORY is set in the current acceptor and
- the directory contains a file corresponding to HTTP-RETURN-CODE,
- that file is sent to the client after variable substitution.
- Variables are referenced by ${<variable-name>}. Currently, only
- the ${script-name} variable is supported which contains the current
- URL relative to the server's base URL."))
+ the directory contains a file corresponding to HTTP-STATUS-CODE
+ named <code>.html, that file is sent to the client after variable
+ substitution. Variables are referenced by ${<variable-name>}.
-(defmethod acceptor-handle-return-code ((acceptor acceptor) http-return-code content)
+ Additional keyword arguments may be provided which are made
+ available to the templating logic as substitution variables. These
+ variables can be interpolated into error message templates in,
+ which contains the current URL relative to the server and without
+ GET parameters.
+
+ In addition to the variables corresponding to keyword arguments,
+ the script-name, lisp-implementation-type,
+ lisp-implementation-version and hunchentoot-version variables are
+ available."))
+
+(defun make-cooked-message (http-status-code &key error backtrace)
+ (labels ((cooked-message (format &rest arguments)
+ (setf (content-type*) "text/html; charset=iso-8859-1")
+ (format nil "<html><head><title>~D ~A</title></head><body><h1>~:*~A</h1>~?<p><hr>~A</p></body></html>"
+ http-status-code (reason-phrase http-status-code)
+ format (mapcar (lambda (arg)
+ (if (stringp arg)
+ (escape-for-html arg)
+ arg))
+ arguments)
+ (address-string))))
+ (case http-status-code
+ ((#.+http-moved-temporarily+
+ #.+http-moved-permanently+)
+ (cooked-message "The document has moved <a href='~A'>here</a>" (header-out :location)))
+ ((#.+http-authorization-required+)
+ (cooked-message "The server could not verify that you are authorized to access the document requested. ~
+ Either you supplied the wrong credentials \(e.g., bad password), or your browser doesn't ~
+ understand how to supply the credentials required."))
+ ((#.+http-forbidden+)
+ (cooked-message "You don't have permission to access ~A on this server."
+ (script-name *request*)))
+ ((#.+http-not-found+)
+ (cooked-message "The requested URL ~A was not found on this server."
+ (script-name *request*)))
+ ((#.+http-bad-request+)
+ (cooked-message "Your browser sent a request that this server could not understand."))
+ ((#.+http-internal-server-error+)
+ (if *show-lisp-errors-p*
+ (cooked-message "<pre>~A~@[~%~%Backtrace:~%~%~A~]</pre>"
+ (escape-for-html (princ-to-string error))
+ (when *show-lisp-backtraces-p*
+ (escape-for-html (princ-to-string backtrace))))
+ (cooked-message "An error has occured"))))))
+
+(defmethod acceptor-status-message ((acceptor t) http-status-code &rest args &key &allow-other-keys)
+ (apply 'make-cooked-message http-status-code args))
+
+(defmethod acceptor-status-message :around ((acceptor acceptor) http-status-code &rest args &key &allow-other-keys)
+ (handler-case
+ (call-next-method)
+ (error (e)
+ (log-message* :error "error ~A during error processing, sending cooked message to client" e)
+ (apply 'make-cooked-message http-status-code args))))
+
+(defmethod acceptor-status-message ((acceptor acceptor) http-status-code &rest properties &key &allow-other-keys)
"Default function to generate error message sent to the client."
(labels
- ((cooked-message (format &rest arguments)
- (setf (content-type*) "text/html; charset=iso-8859-1")
- (format nil "<html><head><title>~D ~A</title></head><body><h1>~:*~A</h1>~?<p><hr>~A</p></body></html>"
- http-return-code (reason-phrase http-return-code)
- format (mapcar (lambda (arg)
- (if (stringp arg)
- (escape-for-html arg)
- arg))
- arguments)
- (address-string)))
- (substitute-request-context-variables (string)
- (cl-ppcre:regex-replace-all "(?i)\\$\\{([a-z0-9-_]+)\\}"
- string
- (lambda (target-string start end match-start match-end reg-starts reg-ends)
- (declare (ignore start end match-start match-end))
- (let ((variable (intern (string-upcase (subseq target-string
- (aref reg-starts 0)
- (aref reg-ends 0)))
- :keyword)))
- (case variable
- (:script-name (script-name*))
- (otherwise (string variable)))))))
+ ((substitute-request-context-variables (string)
+ (let ((properties (append `(:script-name ,(script-name*)
+ :lisp-implementation-type ,(lisp-implementation-type)
+ :lisp-implementation-version ,(lisp-implementation-version)
+ :hunchentoot-version ,*hunchentoot-version*)
+ properties)))
+ (cl-ppcre:regex-replace-all "(?i)\\$\\{([a-z0-9-_]+)\\}"
+ string
+ (lambda (target-string start end match-start match-end reg-starts reg-ends)
+ (declare (ignore start end match-start match-end))
+ (let ((variable-name (intern (string-upcase (subseq target-string
+ (aref reg-starts 0)
+ (aref reg-ends 0)))
+ :keyword)))
+ (escape-for-html (princ-to-string (getf properties variable-name variable-name))))))))
(file-contents (file)
(let ((buf (make-string (file-length file))))
(read-sequence buf file)
buf))
(error-contents-from-template ()
(let ((error-file-template-pathname (and (acceptor-error-template-directory acceptor)
- (probe-file (make-pathname :name (princ-to-string http-return-code)
+ (probe-file (make-pathname :name (princ-to-string http-status-code)
:type "html"
:defaults (acceptor-error-template-directory acceptor))))))
(when error-file-template-pathname
(with-open-file (file error-file-template-pathname :if-does-not-exist nil :element-type 'character)
(when file
(substitute-request-context-variables (file-contents file))))))))
- (or (error-contents-from-template)
- (case http-return-code
- ((#.+http-moved-temporarily+
- #.+http-moved-permanently+)
- (cooked-message "The document has moved <a href='~A'>here</a>" (header-out :location)))
- ((#.+http-authorization-required+)
- (cooked-message "The server could not verify that you are authorized to access the document requested. ~
- Either you supplied the wrong credentials \(e.g., bad password), or your browser doesn't ~
- understand how to supply the credentials required."))
- ((#.+http-forbidden+)
- (cooked-message "You don't have permission to access ~A on this server."
- (script-name *request*)))
- ((#.+http-not-found+)
- (cooked-message "The requested URL ~A was not found on this server."
- (script-name *request*)))
- ((#.+http-bad-request+)
- (cooked-message "Your browser sent a request that this server could not understand."))
- (otherwise
- content)))))
+ (or (unless (< 300 http-status-code)
+ (call-next-method)) ; don't ever try template for positive return codes
+ (error-contents-from-template) ; try template
+ (call-next-method)))) ; fall back to cooked message
(defgeneric acceptor-remove-session (acceptor session)
(:documentation
Modified: trunk/thirdparty/hunchentoot/doc/index.xml
===================================================================
--- trunk/thirdparty/hunchentoot/doc/index.xml 2011-02-10 22:20:18 UTC (rev 4649)
+++ trunk/thirdparty/hunchentoot/doc/index.xml 2011-02-11 10:10:17 UTC (rev 4650)
@@ -704,29 +704,31 @@
</clix:description>
</clix:function>
- <clix:function name="acceptor-handle-return-code" generic="true">
+ <clix:function name="acceptor-status-message" generic="true">
<clix:lambda-list>acceptor http-return-code content</clix:lambda-list>
<clix:description>
This function is called after the request's handler has been
- invoked, before starting to send any output to the client. It
- converts the HTTP return code that has been determined as the
- result of the handler invocation into a content body sent to
- the user. The content generated by the handler is passed to
- this function as <clix:arg>CONTENT</clix:arg> argument. For
- positive return codes (i.e. ``200 OK''), the CONTENT is
- typically just sent to the client. For other return codes,
- the content can be ignored and/or processed, depending on the
- requirements of the acceptor class. Note that the
- <clix:arg>CONTENT</clix:arg> argument can be NIL if the
- handler wants to send the data to the client stream itself.
+ invoked to convert the <clix:arg>HTTP-STATUS-CODE</clix:arg>
+ to a HTML message to be displayed to the user. If this
+ function returns a string, that string is sent to the client
+ instead of the content produced by the handler, if any.
- If an ERROR-TEMPLATE-DIRECTORY is set in the current acceptor
- and the directory contains a file corresponding to
- <clix:arg>HTTP-RETURN-CODE</clix:arg>, that file is sent to
- the client after variable substitution. Variables are
- referenced by ${<variable-name>}. Currently, only the
- ${script-name} variable is supported which contains the
- current URL relative to the server's base URL.
+ If an ERROR-TEMPLATE-DIRECTORY is set in the current
+ acceptor and the directory contains a file corresponding to
+ HTTP-STATUS-CODE named <code>.html, that file is sent
+ to the client after variable substitution. Variables are
+ referenced by ${<variable-name>}.
+
+ Additional keyword arguments may be provided which are made
+ available to the templating logic as substitution variables.
+ These variables can be interpolated into error message
+ templates in, which contains the current URL relative to the
+ server and without GET parameters.
+
+ In addition to the variables corresponding to keyword
+ arguments, the script-name, lisp-implementation-type,
+ lisp-implementation-version and hunchentoot-version
+ variables are available.
</clix:description>
</clix:function>
Modified: trunk/thirdparty/hunchentoot/headers.lisp
===================================================================
--- trunk/thirdparty/hunchentoot/headers.lisp 2011-02-10 22:20:18 UTC (rev 4649)
+++ trunk/thirdparty/hunchentoot/headers.lisp 2011-02-11 10:10:17 UTC (rev 4650)
@@ -145,9 +145,9 @@
(defun send-response (acceptor stream status-code
&key headers cookies content)
"Send a HTTP response to the STREAM and log the event in ACCEPTOR.
- STATUS-CODE is the HTTP status code used in the response. If
- CONTENT-LENGTH, HEADERS and COOKIES are used to create the response
- header. If CONTENT is provided, it is sent as the response body.
+ STATUS-CODE is the HTTP status code used in the response. HEADERS
+ and COOKIES are used to create the response header. If CONTENT is
+ provided, it is sent as the response body.
If *HEADER-STREAM* is not NIL, the response headers are written to
that stream when they are written to the client.
@@ -160,8 +160,7 @@
(setf (cdr (assoc :content-length headers)) (content-length*))
(push (cons :content-length (content-length*)) headers)))
;; access log message
- (acceptor-log-access acceptor
- :return-code status-code)
+ (acceptor-log-access acceptor :return-code status-code)
;; Read post data to clear stream - Force binary mode to avoid OCTETS-TO-STRING overhead.
(raw-post-data :force-binary t)
(let* ((client-header-stream (flex:make-flexi-stream stream :external-format :iso-8859-1))
@@ -180,7 +179,8 @@
(format header-stream "~C~C" #\Return #\Linefeed))
;; now optional content
(when content
- (write-sequence content stream))
+ (write-sequence content stream)
+ (finish-output stream))
stream)
(defun send-headers ()
Modified: trunk/thirdparty/hunchentoot/request.lisp
===================================================================
--- trunk/thirdparty/hunchentoot/request.lisp 2011-02-10 22:20:18 UTC (rev 4649)
+++ trunk/thirdparty/hunchentoot/request.lisp 2011-02-11 10:10:17 UTC (rev 4650)
@@ -224,12 +224,10 @@
(log-message* *lisp-errors-log-level* "~A~@[~%~A~]" error (when *log-lisp-backtraces-p*
backtrace)))
(start-output +http-internal-server-error+
- (if *show-lisp-errors-p*
- (format nil "<pre>~A~@[~%~%Backtrace:~%~%~A~]</pre>"
- (escape-for-html (princ-to-string error))
- (when *show-lisp-backtraces-p*
- (escape-for-html (princ-to-string backtrace))))
- "An error has occured"))))
+ (acceptor-status-message *acceptor*
+ +http-internal-server-error+
+ :error (princ-to-string error)
+ :backtrace (princ-to-string backtrace)))))
(multiple-value-bind (body error backtrace)
;; skip dispatch if bad request
(when (eql (return-code *reply*) +http-ok+)
@@ -242,9 +240,9 @@
(handler-case
(with-debugger
(start-output (return-code *reply*)
- (acceptor-handle-return-code *acceptor*
- (return-code *reply*)
- body)))
+ (or (acceptor-status-message *acceptor*
+ (return-code *reply*))
+ body)))
(error (e)
;; error occured while writing to the client. attempt to report.
(report-error-to-client e)))))))
Modified: trunk/thirdparty/hunchentoot/taskmaster.lisp
===================================================================
--- trunk/thirdparty/hunchentoot/taskmaster.lisp 2011-02-10 22:20:18 UTC (rev 4649)
+++ trunk/thirdparty/hunchentoot/taskmaster.lisp 2011-02-11 10:10:17 UTC (rev 4650)
@@ -322,8 +322,7 @@
(send-response acceptor
(initialize-connection-stream acceptor (make-socket-stream socket acceptor))
+http-service-unavailable+
- :content "<html><head><title>Service unavailable</title></head><body><h1>Service unavailable</h1>Please try later.</body></html>"
- :headers '(("Content-Type" . "text/html")))))
+ :content (acceptor-status-message acceptor +http-service-unavailable+))))
#-:lispworks
(defun client-as-string (socket)
Modified: trunk/thirdparty/hunchentoot/test/test-handlers.lisp
===================================================================
--- trunk/thirdparty/hunchentoot/test/test-handlers.lisp 2011-02-10 22:20:18 UTC (rev 4649)
+++ trunk/thirdparty/hunchentoot/test/test-handlers.lisp 2011-02-11 10:10:17 UTC (rev 4650)
@@ -135,9 +135,9 @@
(defun oops ()
(with-html
- (log-message :error "Oops \(error log level).")
- (log-message :warning "Oops \(warning log level).")
- (log-message :info "Oops \(info log level).")
+ (log-message* :error "Oops \(error log level).")
+ (log-message* :warning "Oops \(warning log level).")
+ (log-message* :info "Oops \(info log level).")
(error "Errors were triggered on purpose. Check your error log.")
(:html
(:body "You should never see this sentence..."))))
Added: trunk/thirdparty/hunchentoot/www/errors/500.html
===================================================================
--- trunk/thirdparty/hunchentoot/www/errors/500.html (rev 0)
+++ trunk/thirdparty/hunchentoot/www/errors/500.html 2011-02-11 10:10:17 UTC (rev 4650)
@@ -0,0 +1,18 @@
+<html>
+ <head>
+ <title>Internal Server Error</title>
+ </head>
+ <body>
+ <h1>Internal Server Error</h1>
+ An error occured while processing your ${script-name} request.
+ <hr/>
+ <h1>Error Message</h1>
+<pre>${error}</pre>
+ <h1>Backtrace</h1>
+<pre>${backtrace}</pre>
+ <hr/>
+<a href="http://weitz.de/hunchentoot">Hunchentoot</a> ${hunchentoot-version} running on ${lisp-implementation-type} ${lisp-implementation-version}
+ <hr/>
+ <img src="/img/made-with-lisp-logo.jpg" width="300" height="100"/>
+ </body>
+</html>
Revision: 4648
Author: hans
URL: http://bknr.net/trac/changeset/4648
reindent documentation source
U trunk/thirdparty/hunchentoot/doc/index.xml
Change set too large, please see URL above