factor/extra/webapps/file/file.factor

136 lines
3.4 KiB
Factor
Raw Normal View History

! Copyright (C) 2004, 2008 Slava Pestov.
2007-09-20 18:09:08 -04:00
! See http://factorcode.org/license.txt for BSD license.
USING: calendar html io io.files kernel math math.parser
http.server.responders http.server.templating namespaces parser
sequences strings assocs hashtables debugger http.mime sorting
2008-02-26 20:18:21 -05:00
html.elements logging calendar.format ;
IN: webapps.file
2007-09-20 18:09:08 -04:00
SYMBOL: doc-root
2007-09-20 18:09:08 -04:00
: serving-path ( filename -- filename )
"" or doc-root get swap path+ ;
2007-09-20 18:09:08 -04:00
2008-02-26 18:22:48 -05:00
: unix-time>timestamp ( n -- timestamp )
>r unix-1970 r> seconds time+ ;
2007-09-20 18:09:08 -04:00
: file-http-date ( filename -- string )
file-modified unix-time>timestamp timestamp>http-string ;
: file-response ( filename mime-type -- )
2007-11-12 23:18:56 -05:00
"200 OK" response
2007-09-20 18:09:08 -04:00
[
"Content-Type" set
dup file-length number>string "Content-Length" set
file-http-date "Last-Modified" set
now timestamp>http-string "Date" set
2007-11-12 23:18:56 -05:00
] H{ } make-assoc print-header ;
2007-09-20 18:09:08 -04:00
: last-modified-matches? ( filename -- bool )
file-http-date dup [
2008-02-25 16:24:48 -05:00
"if-modified-since" header-param =
2007-09-20 18:09:08 -04:00
] when ;
: not-modified-response ( -- )
2007-11-12 23:18:56 -05:00
"304 Not Modified" response
now timestamp>http-string "Date" associate print-header ;
2007-09-20 18:09:08 -04:00
! You can override how files are served in a custom responder
SYMBOL: serve-file-hook
[
2007-12-11 18:44:26 -05:00
dupd
file-response
2007-12-11 18:44:26 -05:00
<file-reader> stdio get stream-copy
] serve-file-hook set-global
2007-09-20 18:09:08 -04:00
: serve-static ( filename mime-type -- )
over last-modified-matches? [
2drop not-modified-response
] [
"method" get "head" = [
file-response
2007-09-20 18:09:08 -04:00
] [
serve-file-hook get call
2007-09-20 18:09:08 -04:00
] if
] if ;
SYMBOL: page
: run-page ( filename -- )
dup
[ [ dup page set run-template-file ] with-scope ] try
drop ;
2008-02-07 18:07:43 -05:00
\ run-page DEBUG add-input-logging
2007-09-20 18:09:08 -04:00
: include-page ( filename -- )
serving-path run-page ;
2007-09-20 18:09:08 -04:00
: serve-fhtml ( filename -- )
serving-html
"method" get "head" = [ drop ] [ run-page ] if ;
2007-09-20 18:09:08 -04:00
: serve-file ( filename -- )
dup mime-type dup "application/x-factor-server-page" =
[ drop serve-fhtml ] [ serve-static ] if ;
2007-09-20 18:09:08 -04:00
2008-02-07 18:07:43 -05:00
\ serve-file NOTICE add-input-logging
: file. ( name dirp -- )
[ "/" append ] when
dup <a =href a> write </a> ;
2007-09-20 18:09:08 -04:00
: directory. ( path request -- )
dup [
<h1> write </h1>
<ul>
directory sort-keys
[ <li> file. </li> ] assoc-each
</ul>
] simple-html-document ;
2007-09-20 18:09:08 -04:00
: list-directory ( directory -- )
serving-html
"method" get "head" = [
drop
] [
"request" get directory.
2007-09-20 18:09:08 -04:00
] 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 -- )
2008-02-07 18:07:43 -05:00
serving-path dup exists? [
dup directory? [ serve-directory ] [ serve-file ] if
] [
drop "404 not found" httpd-error
] if ;
2007-09-20 18:09:08 -04:00
: file-responder ( -- )
doc-root get [
2008-02-07 18:07:43 -05:00
"argument" get serve-object
2007-09-20 18:09:08 -04:00
] [
"404 doc-root not set" httpd-error
] if ;
global [
! Serves files from a directory stored in the doc-root
! variable. You can set the variable in the global
! namespace, or inside the responder.
2007-09-20 18:09:08 -04:00
"file" [ file-responder ] add-simple-responder
! The root directory is served by...
"file" set-default-responder
] bind