1 {-# LANGUAGE BangPatterns        #-}
    2 {-# LANGUAGE CPP                 #-}
    3 {-# LANGUAGE OverloadedStrings   #-}
    4 {-# LANGUAGE ScopedTypeVariables #-}
    5 
    6 -- | Contains web handlers to serve files from a directory.
    7 module Snap.Internal.Util.FileServe
    8   ( -- * Helper functions
    9     getSafePath
   10     -- * Configuration for directory serving
   11   , MimeMap
   12   , HandlerMap
   13   , DirectoryConfig(..)
   14   , simpleDirectoryConfig
   15   , defaultDirectoryConfig
   16   , fancyDirectoryConfig
   17   , defaultIndexGenerator
   18   , defaultMimeTypes
   19   , fileType
   20     -- * File servers
   21   , serveDirectory
   22   , serveDirectoryWith
   23   , serveFile
   24   , serveFileAs
   25     -- * Internal functions
   26   , decodeFilePath
   27   ) where
   28 
   29 ------------------------------------------------------------------------------
   30 import           Control.Applicative              (Alternative ((<|>)), Applicative ((*>), (<*)), (<$>))
   31 import           Control.Exception.Lifted         (SomeException, catch, evaluate)
   32 import           Control.Monad                    (Monad ((>>), (>>=), return), filterM, forM_, liftM, unless, void, when, (=<<))
   33 import           Control.Monad.IO.Class           (MonadIO (..))
   34 import           Data.Attoparsec.ByteString.Char8 (Parser, char, endOfInput, option, string)
   35 import           Data.ByteString.Builder          (Builder, byteString, char8, stringUtf8, toLazyByteString)
   36 import           Data.ByteString.Char8            (ByteString)
   37 import qualified Data.ByteString.Char8            as S (append, concat, intercalate, isSuffixOf, null, pack, takeWhile)
   38 import qualified Data.ByteString.Lazy.Char8       as L
   39 import           Data.HashMap.Strict              (HashMap)
   40 import qualified Data.HashMap.Strict              as Map (empty, fromList, lookup)
   41 import           Data.List                        (drop, dropWhile, elem, filter, foldl', null, sort, (++))
   42 import           Data.Maybe                       (fromMaybe, isNothing)
   43 import           Data.Monoid                      (Monoid (mappend, mconcat))
   44 import qualified Data.Text                        as T (Text, pack, unpack)
   45 import qualified Data.Text.Encoding               as T (decodeUtf8, encodeUtf8)
   46 import           Data.Word                        (Word64)
   47 import           Prelude                          (Bool (..), Eq (..), FilePath, IO, Maybe (Just, Nothing), Num (..), Ord (..), Show (show), String, const, either, flip, fromIntegral, id, maybe, not, ($), ($!), (.), (||))
   48 import qualified Prelude
   49 import           Snap.Core                        (MonadSnap (..), Request (rqPathInfo, rqQueryString, rqURI), deleteHeader, emptyResponse, finishWith, formatHttpTime, getHeader, getRequest, modifyResponse, parseHttpTime, pass, redirect, sendFile, sendFilePartial, setContentLength, setContentType, setHeader, setResponseBody, setResponseCode, urlDecode, writeBS)
   50 import           Snap.Internal.Debug              (debug)
   51 import           Snap.Internal.Parsing            (fullyParse, parseNum)
   52 import           System.Directory                 (doesDirectoryExist, doesFileExist, getDirectoryContents)
   53 import           System.FilePath                  (isRelative, joinPath, splitDirectories, takeExtensions, takeFileName, (</>))
   54 import           System.PosixCompat.Files         (fileSize, getFileStatus, modificationTime)
   55 
   56 
   57 ------------------------------------------------------------------------------
   58 -- | Gets a path from the 'Request' using 'rqPathInfo' and makes sure it is
   59 -- safe to use for opening files.  A path is safe if it is a relative path
   60 -- and has no ".." elements to escape the intended directory structure.
   61 --
   62 -- Example:
   63 --
   64 -- @
   65 -- ghci> :set -XOverloadedStrings
   66 -- ghci> import qualified "Data.Map" as M
   67 -- ghci> import qualified "Snap.Test" as T
   68 -- ghci> import qualified "Data.ByteString.Char8" as B8
   69 -- ghci> T.runHandler (T.get \"\/foo\/bar\" M.empty) ('getSafePath' >>= 'writeBS' . B8.pack)
   70 -- HTTP\/1.1 200 OK
   71 -- server: Snap\/test
   72 -- date: Fri, 08 Aug 2014 16:13:20 GMT
   73 --
   74 -- foo\/bar
   75 -- ghci> T.runHandler (T.get \"\/foo\/..\/bar\" M.empty) ('getSafePath' >>= 'writeBS' . B8.pack)
   76 -- HTTP\/1.1 404 Not Found
   77 -- ...
   78 -- @
   79 getSafePath :: MonadSnap m => m FilePath
   80 getSafePath = do
   81     req <- getRequest
   82     let mp = urlDecode $ rqPathInfo req
   83 
   84     p <- maybe pass (return . T.unpack . T.decodeUtf8) mp
   85 
   86     -- relative paths only!
   87     when (not $ isRelative p) pass
   88 
   89     -- check that we don't have any sneaky .. paths
   90     let dirs = splitDirectories p
   91     when (elem ".." dirs) pass
   92 
   93     return $! joinPath dirs
   94 
   95 
   96 ------------------------------------------------------------------------------
   97 -- | A type alias for dynamic handlers
   98 type HandlerMap m = HashMap FilePath (FilePath -> m ())
   99 
  100 
  101 ------------------------------------------------------------------------------
  102 -- | A type alias for MIME type
  103 type MimeMap = HashMap FilePath ByteString
  104 
  105 
  106 ------------------------------------------------------------------------------
  107 -- | The default set of mime type mappings we use when serving files. Its
  108 -- value:
  109 --
  110 -- > Map.fromList [
  111 -- >   ( ".asc"     , "text/plain"                                             ),
  112 -- >   ( ".asf"     , "video/x-ms-asf"                                         ),
  113 -- >   ( ".asx"     , "video/x-ms-asf"                                         ),
  114 -- >   ( ".au"      , "audio/basic"                                            ),
  115 -- >   ( ".avi"     , "video/x-msvideo"                                        ),
  116 -- >   ( ".bmp"     , "image/bmp"                                              ),
  117 -- >   ( ".bz2"     , "application/x-bzip"                                     ),
  118 -- >   ( ".c"       , "text/plain"                                             ),
  119 -- >   ( ".class"   , "application/octet-stream"                               ),
  120 -- >   ( ".conf"    , "text/plain"                                             ),
  121 -- >   ( ".cpp"     , "text/plain"                                             ),
  122 -- >   ( ".css"     , "text/css"                                               ),
  123 -- >   ( ".cxx"     , "text/plain"                                             ),
  124 -- >   ( ".doc"     , "application/msword"                                     ),
  125 -- >   ( ".docx"    , S.append "application/vnd.openxmlformats-officedocument"
  126 -- >                           ".wordprocessingml.document"                    ),
  127 -- >   ( ".dotx"    , S.append "application/vnd.openxmlformats-officedocument"
  128 -- >                           ".wordprocessingml.template"                    ),
  129 -- >   ( ".dtd"     , "application/xml-dtd"                                    ),
  130 -- >   ( ".dvi"     , "application/x-dvi"                                      ),
  131 -- >   ( ".exe"     , "application/octet-stream"                               ),
  132 -- >   ( ".flv"     , "video/x-flv"                                            ),
  133 -- >   ( ".gif"     , "image/gif"                                              ),
  134 -- >   ( ".gz"      , "application/x-gzip"                                     ),
  135 -- >   ( ".hs"      , "text/plain"                                             ),
  136 -- >   ( ".htm"     , "text/html"                                              ),
  137 -- >   ( ".html"    , "text/html"                                              ),
  138 -- >   ( ".ico"     , "image/x-icon"                                           ),
  139 -- >   ( ".jar"     , "application/x-java-archive"                             ),
  140 -- >   ( ".jpeg"    , "image/jpeg"                                             ),
  141 -- >   ( ".jpg"     , "image/jpeg"                                             ),
  142 -- >   ( ".js"      , "text/javascript"                                        ),
  143 -- >   ( ".json"    , "application/json"                                       ),
  144 -- >   ( ".log"     , "text/plain"                                             ),
  145 -- >   ( ".m3u"     , "audio/x-mpegurl"                                        ),
  146 -- >   ( ".m3u8"    , "application/x-mpegURL"                                  ),
  147 -- >   ( ".mka"     , "audio/x-matroska"                                       ),
  148 -- >   ( ".mk3d"    , "video/x-matroska"                                       ),
  149 -- >   ( ".mkv"     , "video/x-matroska"                                       ),
  150 -- >   ( ".mov"     , "video/quicktime"                                        ),
  151 -- >   ( ".mp3"     , "audio/mpeg"                                             ),
  152 -- >   ( ".mp4"     , "video/mp4"                                              ),
  153 -- >   ( ".mpeg"    , "video/mpeg"                                             ),
  154 -- >   ( ".mpg"     , "video/mpeg"                                             ),
  155 -- >   ( ".ogg"     , "application/ogg"                                        ),
  156 -- >   ( ".pac"     , "application/x-ns-proxy-autoconfig"                      ),
  157 -- >   ( ".pdf"     , "application/pdf"                                        ),
  158 -- >   ( ".png"     , "image/png"                                              ),
  159 -- >   ( ".potx"    , S.append "application/vnd.openxmlformats-officedocument"
  160 -- >                           ".presentationml.template"                      ),
  161 -- >   ( ".ppsx"    , S.append "application/vnd.openxmlformats-officedocument"
  162 -- >                           ".presentationml.slideshow"                     ),
  163 -- >   ( ".ppt"     , "application/vnd.ms-powerpoint"                          ),
  164 -- >   ( ".pptx"    , S.append "application/vnd.openxmlformats-officedocument"
  165 -- >                           ".presentationml.presentation"                  ),
  166 -- >   ( ".ps"      , "application/postscript"                                 ),
  167 -- >   ( ".qt"      , "video/quicktime"                                        ),
  168 -- >   ( ".rtf"     , "text/rtf"                                               ),
  169 -- >   ( ".sig"     , "application/pgp-signature"                              ),
  170 -- >   ( ".sldx"    , S.append "application/vnd.openxmlformats-officedocument"
  171 -- >                           ".presentationml.slide"                         ),
  172 -- >   ( ".spl"     , "application/futuresplash"                               ),
  173 -- >   ( ".svg"     , "image/svg+xml"                                          ),
  174 -- >   ( ".swf"     , "application/x-shockwave-flash"                          ),
  175 -- >   ( ".tar"     , "application/x-tar"                                      ),
  176 -- >   ( ".tar.bz2" , "application/x-bzip-compressed-tar"                      ),
  177 -- >   ( ".tar.gz"  , "application/x-tgz"                                      ),
  178 -- >   ( ".tbz"     , "application/x-bzip-compressed-tar"                      ),
  179 -- >   ( ".text"    , "text/plain"                                             ),
  180 -- >   ( ".tif"     , "image/tiff"                                             ),
  181 -- >   ( ".tiff"    , "image/tiff"                                             ),
  182 -- >   ( ".tgz"     , "application/x-tgz"                                      ),
  183 -- >   ( ".torrent" , "application/x-bittorrent"                               ),
  184 -- >   ( ".ts"      , "video/mp2t"                                             ),
  185 -- >   ( ".txt"     , "text/plain"                                             ),
  186 -- >   ( ".wav"     , "audio/x-wav"                                            ),
  187 -- >   ( ".wax"     , "audio/x-ms-wax"                                         ),
  188 -- >   ( ".webm"    , "video/webm"                                             ),
  189 -- >   ( ".wma"     , "audio/x-ms-wma"                                         ),
  190 -- >   ( ".wmv"     , "video/x-ms-wmv"                                         ),
  191 -- >   ( ".xbm"     , "image/x-xbitmap"                                        ),
  192 -- >   ( ".xlam"    , "application/vnd.ms-excel.addin.macroEnabled.12"         ),
  193 -- >   ( ".xls"     , "application/vnd.ms-excel"                               ),
  194 -- >   ( ".xlsb"    , "application/vnd.ms-excel.sheet.binary.macroEnabled.12"  ),
  195 -- >   ( ".xlsx"    , S.append "application/vnd.openxmlformats-officedocument."
  196 -- >                           "spreadsheetml.sheet"                           ),
  197 -- >   ( ".xltx"    , S.append "application/vnd.openxmlformats-officedocument."
  198 -- >                           "spreadsheetml.template"                        ),
  199 -- >   ( ".xml"     , "text/xml"                                               ),
  200 -- >   ( ".xpm"     , "image/x-xpixmap"                                        ),
  201 -- >   ( ".xwd"     , "image/x-xwindowdump"                                    ),
  202 -- >   ( ".zip"     , "application/zip"                                        ) ]
  203 
  204 defaultMimeTypes :: MimeMap
  205 defaultMimeTypes =
  206   Map.fromList [
  207     ( ".asc"     , "text/plain"                                             ),
  208     ( ".asf"     , "video/x-ms-asf"                                         ),
  209     ( ".asx"     , "video/x-ms-asf"                                         ),
  210     ( ".au"      , "audio/basic"                                            ),
  211     ( ".avi"     , "video/x-msvideo"                                        ),
  212     ( ".bmp"     , "image/bmp"                                              ),
  213     ( ".bz2"     , "application/x-bzip"                                     ),
  214     ( ".c"       , "text/plain"                                             ),
  215     ( ".class"   , "application/octet-stream"                               ),
  216     ( ".conf"    , "text/plain"                                             ),
  217     ( ".cpp"     , "text/plain"                                             ),
  218     ( ".css"     , "text/css"                                               ),
  219     ( ".cxx"     , "text/plain"                                             ),
  220     ( ".doc"     , "application/msword"                                     ),
  221     ( ".docx"    , S.append "application/vnd.openxmlformats-officedocument"
  222                             ".wordprocessingml.document"                    ),
  223     ( ".dotx"    , S.append "application/vnd.openxmlformats-officedocument"
  224                             ".wordprocessingml.template"                    ),
  225     ( ".dtd"     , "application/xml-dtd"                                    ),
  226     ( ".dvi"     , "application/x-dvi"                                      ),
  227     ( ".exe"     , "application/octet-stream"                               ),
  228     ( ".flv"     , "video/x-flv"                                            ),
  229     ( ".gif"     , "image/gif"                                              ),
  230     ( ".gz"      , "application/x-gzip"                                     ),
  231     ( ".hs"      , "text/plain"                                             ),
  232     ( ".htm"     , "text/html"                                              ),
  233     ( ".html"    , "text/html"                                              ),
  234     ( ".ico"     , "image/x-icon"                                           ),
  235     ( ".jar"     , "application/x-java-archive"                             ),
  236     ( ".jpeg"    , "image/jpeg"                                             ),
  237     ( ".jpg"     , "image/jpeg"                                             ),
  238     ( ".js"      , "text/javascript"                                        ),
  239     ( ".json"    , "application/json"                                       ),
  240     ( ".log"     , "text/plain"                                             ),
  241     ( ".m3u"     , "audio/x-mpegurl"                                        ),
  242     ( ".m3u8"    , "application/x-mpegURL"                                  ),
  243     ( ".mka"     , "audio/x-matroska"                                       ),
  244     ( ".mk3d"    , "video/x-matroska"                                       ),
  245     ( ".mkv"     , "video/x-matroska"                                       ),
  246     ( ".mov"     , "video/quicktime"                                        ),
  247     ( ".mp3"     , "audio/mpeg"                                             ),
  248     ( ".mp4"     , "video/mp4"                                              ),
  249     ( ".mpeg"    , "video/mpeg"                                             ),
  250     ( ".mpg"     , "video/mpeg"                                             ),
  251     ( ".ogg"     , "application/ogg"                                        ),
  252     ( ".pac"     , "application/x-ns-proxy-autoconfig"                      ),
  253     ( ".pdf"     , "application/pdf"                                        ),
  254     ( ".png"     , "image/png"                                              ),
  255     ( ".potx"    , S.append "application/vnd.openxmlformats-officedocument"
  256                             ".presentationml.template"                      ),
  257     ( ".ppsx"    , S.append "application/vnd.openxmlformats-officedocument"
  258                             ".presentationml.slideshow"                     ),
  259     ( ".ppt"     , "application/vnd.ms-powerpoint"                          ),
  260     ( ".pptx"    , S.append "application/vnd.openxmlformats-officedocument"
  261                             ".presentationml.presentation"                  ),
  262     ( ".ps"      , "application/postscript"                                 ),
  263     ( ".qt"      , "video/quicktime"                                        ),
  264     ( ".rtf"     , "text/rtf"                                               ),
  265     ( ".sig"     , "application/pgp-signature"                              ),
  266     ( ".sldx"    , S.append "application/vnd.openxmlformats-officedocument"
  267                             ".presentationml.slide"                         ),
  268     ( ".spl"     , "application/futuresplash"                               ),
  269     ( ".svg"     , "image/svg+xml"                                          ),
  270     ( ".swf"     , "application/x-shockwave-flash"                          ),
  271     ( ".tar"     , "application/x-tar"                                      ),
  272     ( ".tar.bz2" , "application/x-bzip-compressed-tar"                      ),
  273     ( ".tar.gz"  , "application/x-tgz"                                      ),
  274     ( ".tbz"     , "application/x-bzip-compressed-tar"                      ),
  275     ( ".text"    , "text/plain"                                             ),
  276     ( ".tif"     , "image/tiff"                                             ),
  277     ( ".tiff"    , "image/tiff"                                             ),
  278     ( ".tgz"     , "application/x-tgz"                                      ),
  279     ( ".torrent" , "application/x-bittorrent"                               ),
  280     ( ".ts"      , "video/mp2t"                                             ),
  281     ( ".txt"     , "text/plain"                                             ),
  282     ( ".wav"     , "audio/x-wav"                                            ),
  283     ( ".wax"     , "audio/x-ms-wax"                                         ),
  284     ( ".webm"    , "video/webm"                                             ),
  285     ( ".wma"     , "audio/x-ms-wma"                                         ),
  286     ( ".wmv"     , "video/x-ms-wmv"                                         ),
  287     ( ".xbm"     , "image/x-xbitmap"                                        ),
  288     ( ".xlam"    , "application/vnd.ms-excel.addin.macroEnabled.12"         ),
  289     ( ".xls"     , "application/vnd.ms-excel"                               ),
  290     ( ".xlsb"    , "application/vnd.ms-excel.sheet.binary.macroEnabled.12"  ),
  291     ( ".xlsx"    , S.append "application/vnd.openxmlformats-officedocument."
  292                             "spreadsheetml.sheet"                           ),
  293     ( ".xltx"    , S.append "application/vnd.openxmlformats-officedocument."
  294                             "spreadsheetml.template"                        ),
  295     ( ".xml"     , "text/xml"                                               ),
  296     ( ".xpm"     , "image/x-xpixmap"                                        ),
  297     ( ".xwd"     , "image/x-xwindowdump"                                    ),
  298     ( ".zip"     , "application/zip"                                        ) ]
  299 
  300 
  301 ------------------------------------------------------------------------------
  302 -- | A collection of options for serving static files out of a directory.
  303 data DirectoryConfig m = DirectoryConfig {
  304     -- | Files to look for when a directory is requested (e.g., index.html)
  305     indexFiles      :: [FilePath],
  306 
  307     -- | Handler to generate a directory listing if there is no index.
  308     indexGenerator  :: FilePath -> m (),
  309 
  310     -- | Map of extensions to pass to dynamic file handlers.  This could be
  311     -- used, for example, to implement CGI dispatch, pretty printing of source
  312     -- code, etc.
  313     dynamicHandlers :: HandlerMap m,
  314 
  315     -- | MIME type map to look up content types.
  316     mimeTypes       :: MimeMap,
  317 
  318     -- | Handler that is called before a file is served.  It will only be
  319     -- called when a file is actually found, not for generated index pages.
  320     preServeHook    :: FilePath -> m ()
  321     }
  322 
  323 
  324 ------------------------------------------------------------------------------
  325 -- | Style information for the default directory index generator.
  326 snapIndexStyles :: ByteString
  327 snapIndexStyles =
  328     S.intercalate "\n"
  329         [ "body { margin: 0px 0px 0px 0px; font-family: sans-serif }"
  330         , "div.header {"
  331         ,     "padding: 40px 40px 0px 40px; height:35px;"
  332         ,     "background:rgb(25,50,87);"
  333         ,     "background-image:-webkit-gradient("
  334         ,         "linear,left bottom,left top,"
  335         ,         "color-stop(0.00, rgb(31,62,108)),"
  336         ,         "color-stop(1.00, rgb(19,38,66)));"
  337         ,     "background-image:-moz-linear-gradient("
  338         ,         "center bottom,rgb(31,62,108) 0%,rgb(19,38,66) 100%);"
  339         ,     "text-shadow:-1px 3px 1px rgb(16,33,57);"
  340         ,     "font-size:16pt; letter-spacing: 2pt; color:white;"
  341         ,     "border-bottom:10px solid rgb(46,93,156) }"
  342         , "div.content {"
  343         ,     "background:rgb(255,255,255);"
  344         ,     "background-image:-webkit-gradient("
  345         ,         "linear,left bottom, left top,"
  346         ,         "color-stop(0.50, rgb(255,255,255)),"
  347         ,         "color-stop(1.00, rgb(224,234,247)));"
  348         ,     "background-image:-moz-linear-gradient("
  349         ,         "center bottom, white 50%, rgb(224,234,247) 100%);"
  350         ,     "padding: 40px 40px 40px 40px }"
  351         , "div.footer {"
  352         ,     "padding: 16px 0px 10px 10px; height:31px;"
  353         ,     "border-top: 1px solid rgb(194,209,225);"
  354         ,     "color: rgb(160,172,186); font-size:10pt;"
  355         ,     "background: rgb(245,249,255) }"
  356         , "table { max-width:100%; margin: 0 auto;" `S.append`
  357           " border-collapse: collapse; }"
  358         , "tr:hover { background:rgb(256,256,224) }"
  359         , "td { border:0; font-family:monospace; padding: 2px 0; }"
  360         , "td.filename, td.type { padding-right: 2em; }"
  361         , "th { border:0; background:rgb(28,56,97);"
  362         ,      "text-shadow:-1px 3px 1px rgb(16,33,57); color: white}"
  363         ]
  364 
  365 
  366 ------------------------------------------------------------------------------
  367 -- | An automatic index generator, which is fairly small and does not rely on
  368 -- any external files (which may not be there depending on external request
  369 -- routing).
  370 --
  371 -- A 'MimeMap' is passed in to display the types of files in the directory
  372 -- listing based on their extension.  Preferably, this is the same as the map
  373 -- in the 'DirectoryConfig'
  374 --
  375 -- The styles parameter allows you to apply styles to the directory listing.
  376 -- The listing itself consists of a table, containing a header row using
  377 -- th elements, and one row per file using td elements, so styles for those
  378 -- pieces may be attached to the appropriate tags.
  379 defaultIndexGenerator :: MonadSnap m
  380                       => MimeMap    -- ^ MIME type mapping for reporting types
  381                       -> ByteString -- ^ Style info to insert in header
  382                       -> FilePath   -- ^ Directory to generate index for
  383                       -> m ()
  384 defaultIndexGenerator mm styles d = do
  385     modifyResponse $ setContentType "text/html; charset=utf-8"
  386     rq      <- getRequest
  387 
  388     let uri   = uriWithoutQueryString rq
  389     let pInfo = rqPathInfo rq
  390 
  391     writeBS "<!DOCTYPE html>\n<html>\n<head>"
  392     writeBS "<title>Directory Listing: "
  393     writeBS uri
  394     writeBS "</title>"
  395     writeBS "<style type='text/css'>"
  396     writeBS styles
  397     writeBS "</style></head><body>"
  398     writeBS "<div class=\"header\">Directory Listing: "
  399     writeBS uri
  400     writeBS "</div><div class=\"content\">"
  401     writeBS "<table><tr><th>File Name</th><th>Type</th><th>Last Modified"
  402     writeBS "</th></tr>"
  403 
  404     when (pInfo /= "") $
  405         writeBS "<tr><td><a href='../'>..</a></td><td colspan=2>DIR</td></tr>"
  406 
  407     entries <- liftIO $ getDirectoryContents d
  408     dirs    <- liftIO $ filterM (doesDirectoryExist . (d </>)) entries
  409     files   <- liftIO $ filterM (doesFileExist . (d </>)) entries
  410 
  411     forM_ (sort $ filter (not . (`elem` ["..", "."])) dirs) $ \f0 -> do
  412         f <- liftIO $ liftM (\s -> T.encodeUtf8 s `mappend` "/")
  413                     $ decodeFilePath f0
  414         writeBS "<tr><td class='filename'><a href='"
  415         writeBS f
  416         writeBS "'>"
  417         writeBS f
  418         writeBS "</a></td><td class='type' colspan=2>DIR</td></tr>"
  419 
  420     forM_ (sort files) $ \f0 -> do
  421         f <- liftIO $ liftM T.encodeUtf8 $ decodeFilePath f0
  422         stat <- liftIO $ getFileStatus (d </> f0)
  423         tm   <- liftIO $ formatHttpTime (modificationTime stat)
  424         writeBS "<tr><td class='filename'><a href='"
  425         writeBS f
  426         writeBS "'>"
  427         writeBS f
  428         writeBS "</a></td><td class='type'>"
  429         writeBS (fileType mm f0)
  430         writeBS "</td><td>"
  431         writeBS tm
  432         writeBS "</tr>"
  433 
  434     writeBS "</table></div><div class=\"footer\">Powered by "
  435     writeBS "<b><a href=\"http://snapframework.com/\">Snap</a></b></div>"
  436     writeBS "</body>"
  437 
  438 
  439 ------------------------------------------------------------------------------
  440 decodeFilePath :: FilePath -> IO T.Text
  441 decodeFilePath fp = do
  442     evaluate (T.decodeUtf8 bs) `catch`
  443         (\(_::SomeException) -> return (T.pack fp))
  444   where
  445     bs = S.pack fp
  446 
  447 ------------------------------------------------------------------------------
  448 -- | A very simple configuration for directory serving.  This configuration
  449 -- uses built-in MIME types from 'defaultMimeTypes', and has no index files,
  450 -- index generator, dynamic file handlers, or 'preServeHook'.
  451 simpleDirectoryConfig :: MonadSnap m => DirectoryConfig m
  452 simpleDirectoryConfig = DirectoryConfig {
  453     indexFiles      = [],
  454     indexGenerator  = const pass,
  455     dynamicHandlers = Map.empty,
  456     mimeTypes       = defaultMimeTypes,
  457     preServeHook    = const $ return $! ()
  458     }
  459 
  460 
  461 ------------------------------------------------------------------------------
  462 -- | A reasonable default configuration for directory serving.  This
  463 -- configuration uses built-in MIME types from 'defaultMimeTypes', serves
  464 -- common index files @index.html@ and @index.htm@, but does not autogenerate
  465 -- directory indexes, nor have any dynamic file handlers. The 'preServeHook'
  466 -- will not do anything.
  467 defaultDirectoryConfig :: MonadSnap m => DirectoryConfig m
  468 defaultDirectoryConfig = DirectoryConfig {
  469     indexFiles      = ["index.html", "index.htm"],
  470     indexGenerator  = const pass,
  471     dynamicHandlers = Map.empty,
  472     mimeTypes       = defaultMimeTypes,
  473     preServeHook    = const $ return $! ()
  474     }
  475 
  476 
  477 ------------------------------------------------------------------------------
  478 -- | A more elaborate configuration for file serving.  This configuration
  479 -- uses built-in MIME types from 'defaultMimeTypes', serves common index files
  480 -- @index.html@ and @index.htm@, and autogenerates directory indexes with a
  481 -- Snap-like feel.  It still has no dynamic file handlers, nor 'preServeHook',
  482 -- which should be added as needed.
  483 --
  484 -- Files recognized as indexes include @index.html@, @index.htm@,
  485 -- @default.html@, @default.htm@, @home.html@
  486 --
  487 -- Example of how the autogenerated directory index looks like:
  488 --
  489 -- <<>>
  490 
  491 fancyDirectoryConfig :: MonadSnap m => DirectoryConfig m
  492 fancyDirectoryConfig = DirectoryConfig {
  493     indexFiles      = ["index.html", "index.htm"],
  494     indexGenerator  = defaultIndexGenerator defaultMimeTypes snapIndexStyles,
  495     dynamicHandlers = Map.empty,
  496     mimeTypes       = defaultMimeTypes,
  497     preServeHook    = const $ return $! ()
  498     }
  499 
  500 
  501 ------------------------------------------------------------------------------
  502 -- | Serves static files from a directory using the default configuration
  503 -- as given in 'defaultDirectoryConfig'.
  504 serveDirectory :: MonadSnap m
  505                => FilePath           -- ^ Directory to serve from
  506                -> m ()
  507 serveDirectory = serveDirectoryWith defaultDirectoryConfig
  508 {-# INLINE serveDirectory #-}
  509 
  510 
  511 ------------------------------------------------------------------------------
  512 -- | Serves static files from a directory.  Configuration options are
  513 -- passed in a 'DirectoryConfig' that captures various choices about desired
  514 -- behavior.  The relative path given in 'rqPathInfo' is searched for a
  515 -- requested file, and the file is served with the appropriate mime type if it
  516 -- is found. Absolute paths and \"@..@\" are prohibited to prevent files from
  517 -- being served from outside the sandbox.
  518 serveDirectoryWith :: MonadSnap m
  519                    => DirectoryConfig m  -- ^ Configuration options
  520                    -> FilePath           -- ^ Directory to serve from
  521                    -> m ()
  522 serveDirectoryWith cfg base = do
  523     b <- directory <|> file <|> redir
  524     when (not b) pass
  525 
  526   where
  527     idxs     = indexFiles cfg
  528     generate = indexGenerator cfg
  529     mimes    = mimeTypes cfg
  530     dyns     = dynamicHandlers cfg
  531     pshook   = preServeHook cfg
  532 
  533     -- Serves a file if it exists; passes if not
  534     serve f = do
  535         liftIO (doesFileExist f) >>= flip unless pass
  536         let fname          = takeFileName f
  537         let staticServe f' = pshook f >> serveFileAs (fileType mimes fname) f'
  538         lookupExt staticServe dyns fname f >> return True
  539 
  540     -- Serves a directory via indices if available.  Returns True on success,
  541     -- False on failure to find an index.  Passes /only/ if the request was
  542     -- not for a directory (no trailing slash).
  543     directory = do
  544         rq  <- getRequest
  545         let uri = uriWithoutQueryString rq
  546         unless ("/" `S.isSuffixOf` uri) pass
  547         rel <- (base </>) <$> getSafePath
  548         b   <- liftIO $ doesDirectoryExist rel
  549         if b then do let serveRel f = serve (rel </> f)
  550                      foldl' (<|>) pass (Prelude.map serveRel idxs)
  551                          <|> (generate rel >> return True)
  552                          <|> return False
  553              else return False
  554 
  555     -- Serves a file requested by name.  Passes if the file doesn't exist.
  556     file = serve =<< ((base </>) <$> getSafePath)
  557 
  558     -- If the request is for a directory but lacks a trailing slash, redirects
  559     -- to the directory name with a trailing slash.
  560     redir = do
  561         rel <- (base </>) <$> getSafePath
  562         liftIO (doesDirectoryExist rel) >>= flip unless pass
  563         rq <- getRequest
  564         let uri = uriWithoutQueryString rq
  565         let qss = queryStringSuffix rq
  566         let u = S.concat [uri, "/", qss]
  567         redirect u
  568 
  569 
  570 ------------------------------------------------------------------------------
  571 -- | Serves a single file specified by a full or relative path.  If the file
  572 -- does not exist, throws an exception (not that it does /not/ pass to the
  573 -- next handler).   The path restrictions on 'serveDirectory' don't apply to
  574 -- this function since the path is not being supplied by the user.
  575 serveFile :: MonadSnap m
  576           => FilePath          -- ^ path to file
  577           -> m ()
  578 serveFile fp = serveFileAs (fileType defaultMimeTypes (takeFileName fp)) fp
  579 {-# INLINE serveFile #-}
  580 
  581 
  582 ------------------------------------------------------------------------------
  583 -- | Same as 'serveFile', with control over the MIME mapping used.
  584 serveFileAs :: MonadSnap m
  585             => ByteString        -- ^ MIME type
  586             -> FilePath          -- ^ path to file
  587             -> m ()
  588 serveFileAs mime fp = do
  589     reqOrig <- getRequest
  590 
  591     -- If-Range header must be ignored if there is no Range: header in the
  592     -- request (RFC 2616 section 14.27)
  593     let req = if isNothing $ getHeader "range" reqOrig
  594                 then deleteHeader "if-range" reqOrig
  595                 else reqOrig
  596 
  597     -- check "If-Modified-Since" and "If-Range" headers
  598     let mbH = getHeader "if-modified-since" req
  599     mbIfModified <- liftIO $ case mbH of
  600                                Nothing  -> return Nothing
  601                                (Just s) -> liftM Just $ parseHttpTime s
  602 
  603     -- If-Range header could contain an entity, but then parseHttpTime will
  604     -- fail and return 0 which means a 200 response will be generated anyways
  605     mbIfRange <- liftIO $ case getHeader "if-range" req of
  606                             Nothing  -> return Nothing
  607                             (Just s) -> liftM Just $ parseHttpTime s
  608 
  609     dbg $ "mbIfModified: " ++ Prelude.show mbIfModified
  610     dbg $ "mbIfRange: " ++ Prelude.show mbIfRange
  611 
  612     -- check modification time and bug out early if the file is not modified.
  613     --
  614     -- TODO: a stat cache would be nice here, but it'd need the date thread
  615     -- stuff from snap-server to be folded into snap-core
  616     filestat <- liftIO $ getFileStatus fp
  617     let mt = modificationTime filestat
  618     maybe (return $! ()) (\lt -> when (mt <= lt) notModified) mbIfModified
  619 
  620     let sz = fromIntegral $ fileSize filestat
  621     lm <- liftIO $ formatHttpTime mt
  622 
  623     -- ok, at this point we know the last-modified time and the
  624     -- content-type. set those.
  625     modifyResponse $ setHeader "Last-Modified" lm
  626                    . setHeader "Accept-Ranges" "bytes"
  627                    . setContentType mime
  628 
  629 
  630     -- now check: is this a range request? If there is an 'If-Range' header
  631     -- with an old modification time we skip this check and send a 200
  632     -- response
  633     let skipRangeCheck = maybe (False)
  634                                (\lt -> mt > lt)
  635                                mbIfRange
  636 
  637     -- checkRangeReq checks for a Range: header in the request and sends a
  638     -- partial response if it matches.
  639     wasRange <- if skipRangeCheck
  640                   then return False
  641                   else liftSnap $ checkRangeReq req fp sz
  642 
  643     dbg $ "was this a range request? " ++ Prelude.show wasRange
  644 
  645     -- if we didn't have a range request, we just do normal sendfile
  646     unless wasRange $ do
  647       modifyResponse $ setResponseCode 200
  648                      . setContentLength sz
  649       liftSnap $ sendFile fp
  650 
  651   where
  652     --------------------------------------------------------------------------
  653     notModified = finishWith $
  654                   setResponseCode 304 emptyResponse
  655 
  656 
  657 ------------------------------------------------------------------------------
  658 lookupExt :: a -> HashMap FilePath a -> FilePath -> a
  659 lookupExt def m f =
  660     if null ext
  661       then def
  662       else fromMaybe (lookupExt def m (next ext)) mbe
  663 
  664   where
  665     next            = dropWhile (/= '.') . drop 1
  666     ext             = takeExtensions f
  667     mbe             = Map.lookup ext m
  668 
  669 
  670 ------------------------------------------------------------------------------
  671 -- | Determine a given file's MIME type from its filename and the provided MIME
  672 -- map.
  673 fileType :: MimeMap -> FilePath -> ByteString
  674 fileType = lookupExt defaultMimeType
  675 
  676 
  677 ------------------------------------------------------------------------------
  678 defaultMimeType :: ByteString
  679 defaultMimeType = "application/octet-stream"
  680 
  681 
  682 ------------------------------------------------------------------------------
  683 data RangeReq = RangeReq !Word64 !(Maybe Word64)
  684               | SuffixRangeReq !Word64
  685 
  686 
  687 ------------------------------------------------------------------------------
  688 rangeParser :: Parser RangeReq
  689 rangeParser = string "bytes=" *>
  690               (byteRangeSpec <|> suffixByteRangeSpec) <*
  691               endOfInput
  692   where
  693     byteRangeSpec = do
  694         start <- fromIntegral <$> parseNum
  695         void $! char '-'
  696         end   <- option Nothing $ liftM Just parseNum
  697 
  698         return $! RangeReq start (fromIntegral <$> end)
  699 
  700     suffixByteRangeSpec =
  701         liftM (SuffixRangeReq . fromIntegral) $ char '-' *> parseNum
  702 
  703 
  704 ------------------------------------------------------------------------------
  705 checkRangeReq :: (MonadSnap m) => Request -> FilePath -> Word64 -> m Bool
  706 checkRangeReq req fp sz = do
  707     -- TODO/FIXME: multiple ranges
  708     maybe (return False)
  709           (\s -> either (const $ return False)
  710                         withRange
  711                         (fullyParse s rangeParser))
  712           (getHeader "range" req)
  713 
  714   where
  715     withRange (RangeReq start mend) = do
  716         let end = fromMaybe (sz-1) mend
  717         dbg $ "withRange: start=" ++ Prelude.show start
  718                   ++ ", end=" ++ Prelude.show end
  719 
  720         if start < 0 || end < start || start >= sz || end >= sz
  721            then send416
  722            else send206 start end
  723 
  724     withRange (SuffixRangeReq nbytes) = do
  725         let end   = sz-1
  726         let start = sz - nbytes
  727 
  728         dbg $ "withRange: start=" ++ Prelude.show start
  729                   ++ ", end=" ++ Prelude.show end
  730 
  731         if start < 0 || end < start || start >= sz || end >= sz
  732            then send416
  733            else send206 start end
  734 
  735     -- note: start and end INCLUSIVE here
  736     send206 start end = do
  737         dbg "inside send206"
  738         let !len = end-start+1
  739         let crng = S.concat . L.toChunks $
  740                    toLazyByteString $
  741                    mconcat [ byteString "bytes "
  742                            , fromShow start
  743                            , char8 '-'
  744                            , fromShow end
  745                            , char8 '/'
  746                            , fromShow sz ]
  747 
  748         modifyResponse $ setResponseCode 206
  749                        . setHeader "Content-Range" crng
  750                        . setContentLength len
  751 
  752         dbg $ "send206: sending range (" ++ Prelude.show start
  753                 ++ "," ++ Prelude.show (end+1) ++ ") to sendFilePartial"
  754 
  755         -- end here was inclusive, sendFilePartial is exclusive
  756         sendFilePartial fp (start,end+1)
  757         return True
  758 
  759 
  760     send416 = do
  761         dbg "inside send416"
  762         -- if there's an "If-Range" header in the request, then we just send
  763         -- back 200
  764         if getHeader "If-Range" req /= Nothing
  765            then return False
  766            else do
  767                let crng = S.concat . L.toChunks $
  768                           toLazyByteString $
  769                           mconcat [ byteString "bytes */"
  770                                   , fromShow sz ]
  771 
  772                modifyResponse $ setResponseCode 416
  773                               . setHeader "Content-Range" crng
  774                               . setContentLength 0
  775                               . deleteHeader "Content-Type"
  776                               . deleteHeader "Content-Encoding"
  777                               . deleteHeader "Transfer-Encoding"
  778                               . setResponseBody (return . id)
  779 
  780                return True
  781 
  782 
  783 ------------------------------------------------------------------------------
  784 dbg :: (MonadIO m) => String -> m ()
  785 dbg s = debug $ "FileServe:" ++ s
  786 
  787 
  788 ------------------------------------------------------------------------------
  789 uriWithoutQueryString :: Request -> ByteString
  790 uriWithoutQueryString rq = S.takeWhile (/= '?') uri
  791   where
  792     uri   = rqURI rq
  793 
  794 
  795 ------------------------------------------------------------------------------
  796 queryStringSuffix :: Request -> ByteString
  797 queryStringSuffix rq = S.concat [ s, qs ]
  798   where
  799     qs = rqQueryString rq
  800     s  = if S.null qs then "" else "?"
  801 
  802 
  803 ------------------------------------------------------------------------------
  804 fromShow :: Show a => a -> Builder
  805 fromShow = stringUtf8 . show