factor/libs/httpd/file-responder.factor

100 lines
2.5 KiB
Factor

! Copyright (C) 2004, 2006 Slava Pestov.
! See http://factorcode.org/license.txt for BSD license.
USING: calendar embedded errors html httpd io kernel math
namespaces parser sequences strings hashtables ;
IN: file-responder
: serving-path ( filename -- filename )
[ "" ] unless* "doc-root" get swap path+ ;
: file-http-date ( filename -- string )
file-modified unix>gmt timestamp>http-string ;
: file-response ( filename mime-type -- )
[
"Content-Type" set
dup file-length number>string "Content-Length" set
file-http-date "Last-Modified" set
now timestamp>http-string "Date" set
] make-hash "200 OK" response terpri ;
: last-modified-matches? ( filename -- bool )
file-http-date dup [
"If-Modified-Since" "header" get hash =
] when ;
: not-modified-response ( -- )
now timestamp>http-string "Date" associate
"304 Not Modified" response terpri ;
: serve-static ( filename mime-type -- )
over last-modified-matches? [
2drop not-modified-response
] [
dupd file-response
"method" get "head" = [
drop
] [
<file-reader> stdio get stream-copy
] if
] if ;
SYMBOL: page
: run-page ( filename -- )
dup
[ [ dup page set run-embedded-file ] with-scope ] try
drop ;
: include-page ( filename -- )
"doc-root" get swap path+ run-page ;
: serve-file ( filename -- )
dup mime-type dup "application/x-factor-server-page" =
[ drop serving-html run-page ] [ serve-static ] if ;
: file. ( path name -- )
tuck path+
directory? "[DIR] " " " ? write
write-pathname terpri ;
: directory. ( path -- )
dup directory natural-sort [ file. ] each-with ;
: list-directory ( directory -- )
serving-html
"method" get "head" = [
drop
] [
"request" get [
"" swap directory.
] simple-html-document
] if ;
: find-index ( filename -- path )
{ "index.html" "index.fhtml" }
[ dupd path+ exists? ] find nip
dup [ path+ ] [ nip ] if ;
: serve-directory ( filename -- )
dup "/" tail? [
dup find-index
[ serve-file ] [ list-directory ] ?if
] [
drop directory-no/
] if ;
: serve-object ( filename -- )
dup directory? [ serve-directory ] [ serve-file ] if ;
: file-responder ( -- )
"doc-root" get [
"argument" get serving-path dup exists? [
serve-object
] [
drop "404 not found" httpd-error
] if
] [
"404 doc-root not set" httpd-error
] if ;