{-# LANGUAGE BangPatterns #-} {-# LANGUAGE CPP #-} {-# LANGUAGE OverloadedStrings #-} {-# LANGUAGE ScopedTypeVariables #-} -- | Contains web handlers to serve files from a directory. module Snap.Internal.Util.FileServe ( -- * Helper functions getSafePath -- * Configuration for directory serving , MimeMap , HandlerMap , DirectoryConfig(..) , simpleDirectoryConfig , defaultDirectoryConfig , fancyDirectoryConfig , defaultIndexGenerator , defaultMimeTypes , fileType -- * File servers , serveDirectory , serveDirectoryWith , serveFile , serveFileAs -- * Internal functions , decodeFilePath , checkRangeReq ) where ------------------------------------------------------------------------------ import Control.Applicative (Alternative ((<|>)), Applicative ((*>), (<*)), (<$>)) import Control.Exception.Lifted (SomeException, catch, evaluate) import Control.Monad (Monad ((>>), (>>=), return), filterM, forM_, liftM, unless, void, when, (=<<)) import Control.Monad.IO.Class (MonadIO (..)) import Data.Attoparsec.ByteString.Char8 (Parser, char, endOfInput, option, string) import Data.ByteString.Builder (Builder, byteString, char8, stringUtf8, toLazyByteString) import Data.ByteString.Char8 (ByteString) import qualified Data.ByteString.Char8 as S (append, concat, intercalate, isSuffixOf, null, pack, takeWhile) import qualified Data.ByteString.Lazy.Char8 as L import Data.HashMap.Strict (HashMap) import qualified Data.HashMap.Strict as Map (empty, fromList, lookup) import Data.List (drop, dropWhile, elem, filter, foldl', null, sort, (++)) import Data.Maybe (fromMaybe, isNothing) import Data.Monoid (Monoid (mappend, mconcat)) import qualified Data.Text as T (Text, pack, unpack) import qualified Data.Text.Encoding as T (decodeUtf8, encodeUtf8) import Data.Word (Word64) import Prelude (Bool (..), Eq (..), FilePath, IO, Maybe (Just, Nothing), Num (..), Ord (..), Show (show), String, const, either, flip, fromIntegral, id, maybe, not, ($), ($!), (.), (||)) import qualified Prelude 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) import Snap.Internal.Debug (debug) import Snap.Internal.Parsing (fullyParse, parseNum) import System.Directory (doesDirectoryExist, doesFileExist, getDirectoryContents) import System.FilePath (isRelative, joinPath, splitDirectories, takeExtensions, takeFileName, (</>)) import System.PosixCompat.Files (fileSize, getFileStatus, modificationTime) ------------------------------------------------------------------------------ -- | Gets a path from the 'Request' using 'rqPathInfo' and makes sure it is -- safe to use for opening files. A path is safe if it is a relative path -- and has no ".." elements to escape the intended directory structure. -- -- Example: -- -- @ -- ghci> :set -XOverloadedStrings -- ghci> import qualified "Data.Map" as M -- ghci> import qualified "Snap.Test" as T -- ghci> import qualified "Data.ByteString.Char8" as B8 -- ghci> T.runHandler (T.get \"\/foo\/bar\" M.empty) ('getSafePath' >>= 'writeBS' . B8.pack) -- HTTP\/1.1 200 OK -- server: Snap\/test -- date: Fri, 08 Aug 2014 16:13:20 GMT -- -- foo\/bar -- ghci> T.runHandler (T.get \"\/foo\/..\/bar\" M.empty) ('getSafePath' >>= 'writeBS' . B8.pack) -- HTTP\/1.1 404 Not Found -- ... -- @ getSafePath :: MonadSnap m => m FilePath getSafePath :: forall (m :: * -> *). MonadSnap m => m FilePath getSafePath = do Request req <- forall (m :: * -> *). MonadSnap m => m Request getRequest let mp :: Maybe ByteString mp = ByteString -> Maybe ByteString urlDecode forall a b. (a -> b) -> a -> b $ Request -> ByteString rqPathInfo Request req FilePath p <- forall b a. b -> (a -> b) -> Maybe a -> b maybe forall (m :: * -> *) a. MonadSnap m => m a pass (forall (m :: * -> *) a. Monad m => a -> m a return forall b c a. (b -> c) -> (a -> b) -> a -> c . Text -> FilePath T.unpack forall b c a. (b -> c) -> (a -> b) -> a -> c . ByteString -> Text T.decodeUtf8) Maybe ByteString mp -- relative paths only! forall (f :: * -> *). Applicative f => Bool -> f () -> f () when (Bool -> Bool not forall a b. (a -> b) -> a -> b $ FilePath -> Bool isRelative FilePath p) forall (m :: * -> *) a. MonadSnap m => m a pass -- check that we don't have any sneaky .. paths let dirs :: [FilePath] dirs = FilePath -> [FilePath] splitDirectories FilePath p forall (f :: * -> *). Applicative f => Bool -> f () -> f () when (forall (t :: * -> *) a. (Foldable t, Eq a) => a -> t a -> Bool elem FilePath ".." [FilePath] dirs) forall (m :: * -> *) a. MonadSnap m => m a pass forall (m :: * -> *) a. Monad m => a -> m a return forall a b. (a -> b) -> a -> b $! [FilePath] -> FilePath joinPath [FilePath] dirs ------------------------------------------------------------------------------ -- | A type alias for dynamic handlers type HandlerMap m = HashMap FilePath (FilePath -> m ()) ------------------------------------------------------------------------------ -- | A type alias for MIME type type MimeMap = HashMap FilePath ByteString ------------------------------------------------------------------------------ -- | The default set of mime type mappings we use when serving files. Its -- value: -- -- > Map.fromList [ -- > ( ".asc" , "text/plain" ), -- > ( ".asf" , "video/x-ms-asf" ), -- > ( ".asx" , "video/x-ms-asf" ), -- > ( ".au" , "audio/basic" ), -- > ( ".avi" , "video/x-msvideo" ), -- > ( ".bmp" , "image/bmp" ), -- > ( ".bz2" , "application/x-bzip" ), -- > ( ".c" , "text/plain" ), -- > ( ".class" , "application/octet-stream" ), -- > ( ".conf" , "text/plain" ), -- > ( ".cpp" , "text/plain" ), -- > ( ".css" , "text/css" ), -- > ( ".csv" , "text/csv" ), -- > ( ".cxx" , "text/plain" ), -- > ( ".doc" , "application/msword" ), -- > ( ".docx" , S.append "application/vnd.openxmlformats-officedocument" -- > ".wordprocessingml.document" ), -- > ( ".dotx" , S.append "application/vnd.openxmlformats-officedocument" -- > ".wordprocessingml.template" ), -- > ( ".dtd" , "application/xml-dtd" ), -- > ( ".dvi" , "application/x-dvi" ), -- > ( ".exe" , "application/octet-stream" ), -- > ( ".flv" , "video/x-flv" ), -- > ( ".gif" , "image/gif" ), -- > ( ".gz" , "application/x-gzip" ), -- > ( ".hs" , "text/plain" ), -- > ( ".htm" , "text/html" ), -- > ( ".html" , "text/html" ), -- > ( ".ico" , "image/x-icon" ), -- > ( ".jar" , "application/x-java-archive" ), -- > ( ".jpeg" , "image/jpeg" ), -- > ( ".jpg" , "image/jpeg" ), -- > ( ".js" , "text/javascript" ), -- > ( ".json" , "application/json" ), -- > ( ".log" , "text/plain" ), -- > ( ".m3u" , "audio/x-mpegurl" ), -- > ( ".m3u8" , "application/x-mpegURL" ), -- > ( ".mka" , "audio/x-matroska" ), -- > ( ".mk3d" , "video/x-matroska" ), -- > ( ".mkv" , "video/x-matroska" ), -- > ( ".mov" , "video/quicktime" ), -- > ( ".mp3" , "audio/mpeg" ), -- > ( ".mp4" , "video/mp4" ), -- > ( ".mpeg" , "video/mpeg" ), -- > ( ".mpg" , "video/mpeg" ), -- > ( ".ogg" , "application/ogg" ), -- > ( ".pac" , "application/x-ns-proxy-autoconfig" ), -- > ( ".pdf" , "application/pdf" ), -- > ( ".png" , "image/png" ), -- > ( ".potx" , S.append "application/vnd.openxmlformats-officedocument" -- > ".presentationml.template" ), -- > ( ".ppsx" , S.append "application/vnd.openxmlformats-officedocument" -- > ".presentationml.slideshow" ), -- > ( ".ppt" , "application/vnd.ms-powerpoint" ), -- > ( ".pptx" , S.append "application/vnd.openxmlformats-officedocument" -- > ".presentationml.presentation" ), -- > ( ".ps" , "application/postscript" ), -- > ( ".qt" , "video/quicktime" ), -- > ( ".rtf" , "text/rtf" ), -- > ( ".sig" , "application/pgp-signature" ), -- > ( ".sldx" , S.append "application/vnd.openxmlformats-officedocument" -- > ".presentationml.slide" ), -- > ( ".spl" , "application/futuresplash" ), -- > ( ".svg" , "image/svg+xml" ), -- > ( ".swf" , "application/x-shockwave-flash" ), -- > ( ".tar" , "application/x-tar" ), -- > ( ".tar.bz2" , "application/x-bzip-compressed-tar" ), -- > ( ".tar.gz" , "application/x-tgz" ), -- > ( ".tbz" , "application/x-bzip-compressed-tar" ), -- > ( ".text" , "text/plain" ), -- > ( ".tgz" , "application/x-tgz" ), -- > ( ".tif" , "image/tiff" ), -- > ( ".tiff" , "image/tiff" ), -- > ( ".torrent" , "application/x-bittorrent" ), -- > ( ".ts" , "video/mp2t" ), -- > ( ".txt" , "text/plain" ), -- > ( ".wav" , "audio/x-wav" ), -- > ( ".wax" , "audio/x-ms-wax" ), -- > ( ".webm" , "video/webm" ), -- > ( ".wma" , "audio/x-ms-wma" ), -- > ( ".wmv" , "video/x-ms-wmv" ), -- > ( ".xbm" , "image/x-xbitmap" ), -- > ( ".xlam" , "application/vnd.ms-excel.addin.macroEnabled.12" ), -- > ( ".xls" , "application/vnd.ms-excel" ), -- > ( ".xlsb" , "application/vnd.ms-excel.sheet.binary.macroEnabled.12" ), -- > ( ".xlsx" , S.append "application/vnd.openxmlformats-officedocument." -- > "spreadsheetml.sheet" ), -- > ( ".xltx" , S.append "application/vnd.openxmlformats-officedocument." -- > "spreadsheetml.template" ), -- > ( ".xml" , "text/xml" ), -- > ( ".xpm" , "image/x-xpixmap" ), -- > ( ".xwd" , "image/x-xwindowdump" ), -- > ( ".zip" , "application/zip" ) ] defaultMimeTypes :: MimeMap defaultMimeTypes :: MimeMap defaultMimeTypes = forall k v. (Eq k, Hashable k) => [(k, v)] -> HashMap k v Map.fromList [ ( FilePath ".asc" , ByteString "text/plain" ), ( FilePath ".asf" , ByteString "video/x-ms-asf" ), ( FilePath ".asx" , ByteString "video/x-ms-asf" ), ( FilePath ".au" , ByteString "audio/basic" ), ( FilePath ".avi" , ByteString "video/x-msvideo" ), ( FilePath ".bmp" , ByteString "image/bmp" ), ( FilePath ".bz2" , ByteString "application/x-bzip" ), ( FilePath ".c" , ByteString "text/plain" ), ( FilePath ".class" , ByteString "application/octet-stream" ), ( FilePath ".conf" , ByteString "text/plain" ), ( FilePath ".cpp" , ByteString "text/plain" ), ( FilePath ".css" , ByteString "text/css" ), ( FilePath ".csv" , ByteString "text/csv" ), ( FilePath ".cxx" , ByteString "text/plain" ), ( FilePath ".doc" , ByteString "application/msword" ), ( FilePath ".docx" , ByteString -> ByteString -> ByteString S.append ByteString "application/vnd.openxmlformats-officedocument" ByteString ".wordprocessingml.document" ), ( FilePath ".dotx" , ByteString -> ByteString -> ByteString S.append ByteString "application/vnd.openxmlformats-officedocument" ByteString ".wordprocessingml.template" ), ( FilePath ".dtd" , ByteString "application/xml-dtd" ), ( FilePath ".dvi" , ByteString "application/x-dvi" ), ( FilePath ".exe" , ByteString "application/octet-stream" ), ( FilePath ".flv" , ByteString "video/x-flv" ), ( FilePath ".gif" , ByteString "image/gif" ), ( FilePath ".gz" , ByteString "application/x-gzip" ), ( FilePath ".hs" , ByteString "text/plain" ), ( FilePath ".htm" , ByteString "text/html" ), ( FilePath ".html" , ByteString "text/html" ), ( FilePath ".ico" , ByteString "image/x-icon" ), ( FilePath ".jar" , ByteString "application/x-java-archive" ), ( FilePath ".jpeg" , ByteString "image/jpeg" ), ( FilePath ".jpg" , ByteString "image/jpeg" ), ( FilePath ".js" , ByteString "text/javascript" ), ( FilePath ".json" , ByteString "application/json" ), ( FilePath ".log" , ByteString "text/plain" ), ( FilePath ".m3u" , ByteString "audio/x-mpegurl" ), ( FilePath ".m3u8" , ByteString "application/x-mpegURL" ), ( FilePath ".mka" , ByteString "audio/x-matroska" ), ( FilePath ".mk3d" , ByteString "video/x-matroska" ), ( FilePath ".mkv" , ByteString "video/x-matroska" ), ( FilePath ".mov" , ByteString "video/quicktime" ), ( FilePath ".mp3" , ByteString "audio/mpeg" ), ( FilePath ".mp4" , ByteString "video/mp4" ), ( FilePath ".mpeg" , ByteString "video/mpeg" ), ( FilePath ".mpg" , ByteString "video/mpeg" ), ( FilePath ".ogg" , ByteString "application/ogg" ), ( FilePath ".pac" , ByteString "application/x-ns-proxy-autoconfig" ), ( FilePath ".pdf" , ByteString "application/pdf" ), ( FilePath ".png" , ByteString "image/png" ), ( FilePath ".potx" , ByteString -> ByteString -> ByteString S.append ByteString "application/vnd.openxmlformats-officedocument" ByteString ".presentationml.template" ), ( FilePath ".ppsx" , ByteString -> ByteString -> ByteString S.append ByteString "application/vnd.openxmlformats-officedocument" ByteString ".presentationml.slideshow" ), ( FilePath ".ppt" , ByteString "application/vnd.ms-powerpoint" ), ( FilePath ".pptx" , ByteString -> ByteString -> ByteString S.append ByteString "application/vnd.openxmlformats-officedocument" ByteString ".presentationml.presentation" ), ( FilePath ".ps" , ByteString "application/postscript" ), ( FilePath ".qt" , ByteString "video/quicktime" ), ( FilePath ".rtf" , ByteString "text/rtf" ), ( FilePath ".sig" , ByteString "application/pgp-signature" ), ( FilePath ".sldx" , ByteString -> ByteString -> ByteString S.append ByteString "application/vnd.openxmlformats-officedocument" ByteString ".presentationml.slide" ), ( FilePath ".spl" , ByteString "application/futuresplash" ), ( FilePath ".svg" , ByteString "image/svg+xml" ), ( FilePath ".swf" , ByteString "application/x-shockwave-flash" ), ( FilePath ".tar" , ByteString "application/x-tar" ), ( FilePath ".tar.bz2" , ByteString "application/x-bzip-compressed-tar" ), ( FilePath ".tar.gz" , ByteString "application/x-tgz" ), ( FilePath ".tbz" , ByteString "application/x-bzip-compressed-tar" ), ( FilePath ".text" , ByteString "text/plain" ), ( FilePath ".tgz" , ByteString "application/x-tgz" ), ( FilePath ".tiff" , ByteString "image/tiff" ), ( FilePath ".tif" , ByteString "image/tiff" ), ( FilePath ".torrent" , ByteString "application/x-bittorrent" ), ( FilePath ".ts" , ByteString "video/mp2t" ), ( FilePath ".ttf" , ByteString "font/ttf" ), ( FilePath ".txt" , ByteString "text/plain" ), ( FilePath ".wav" , ByteString "audio/x-wav" ), ( FilePath ".wax" , ByteString "audio/x-ms-wax" ), ( FilePath ".webm" , ByteString "video/webm" ), ( FilePath ".wma" , ByteString "audio/x-ms-wma" ), ( FilePath ".wmv" , ByteString "video/x-ms-wmv" ), ( FilePath ".xbm" , ByteString "image/x-xbitmap" ), ( FilePath ".xlam" , ByteString "application/vnd.ms-excel.addin.macroEnabled.12" ), ( FilePath ".xls" , ByteString "application/vnd.ms-excel" ), ( FilePath ".xlsb" , ByteString "application/vnd.ms-excel.sheet.binary.macroEnabled.12" ), ( FilePath ".xlsx" , ByteString -> ByteString -> ByteString S.append ByteString "application/vnd.openxmlformats-officedocument." ByteString "spreadsheetml.sheet" ), ( FilePath ".xltx" , ByteString -> ByteString -> ByteString S.append ByteString "application/vnd.openxmlformats-officedocument." ByteString "spreadsheetml.template" ), ( FilePath ".xml" , ByteString "text/xml" ), ( FilePath ".xpm" , ByteString "image/x-xpixmap" ), ( FilePath ".xwd" , ByteString "image/x-xwindowdump" ), ( FilePath ".zip" , ByteString "application/zip" ) ] ------------------------------------------------------------------------------ -- | A collection of options for serving static files out of a directory. data DirectoryConfig m = DirectoryConfig { -- | Files to look for when a directory is requested (e.g., index.html) forall (m :: * -> *). DirectoryConfig m -> [FilePath] indexFiles :: [FilePath], -- | Handler to generate a directory listing if there is no index. forall (m :: * -> *). DirectoryConfig m -> FilePath -> m () indexGenerator :: FilePath -> m (), -- | Map of extensions to pass to dynamic file handlers. This could be -- used, for example, to implement CGI dispatch, pretty printing of source -- code, etc. forall (m :: * -> *). DirectoryConfig m -> HandlerMap m dynamicHandlers :: HandlerMap m, -- | MIME type map to look up content types. forall (m :: * -> *). DirectoryConfig m -> MimeMap mimeTypes :: MimeMap, -- | Handler that is called before a file is served. It will only be -- called when a file is actually found, not for generated index pages. forall (m :: * -> *). DirectoryConfig m -> FilePath -> m () preServeHook :: FilePath -> m () } ------------------------------------------------------------------------------ -- | Style information for the default directory index generator. snapIndexStyles :: ByteString snapIndexStyles :: ByteString snapIndexStyles = ByteString -> [ByteString] -> ByteString S.intercalate ByteString "\n" [ ByteString "body { margin: 0px 0px 0px 0px; font-family: sans-serif }" , ByteString "div.header {" , ByteString "padding: 40px 40px 0px 40px; height:35px;" , ByteString "background:rgb(25,50,87);" , ByteString "background-image:-webkit-gradient(" , ByteString "linear,left bottom,left top," , ByteString "color-stop(0.00, rgb(31,62,108))," , ByteString "color-stop(1.00, rgb(19,38,66)));" , ByteString "background-image:-moz-linear-gradient(" , ByteString "center bottom,rgb(31,62,108) 0%,rgb(19,38,66) 100%);" , ByteString "text-shadow:-1px 3px 1px rgb(16,33,57);" , ByteString "font-size:16pt; letter-spacing: 2pt; color:white;" , ByteString "border-bottom:10px solid rgb(46,93,156) }" , ByteString "div.content {" , ByteString "background:rgb(255,255,255);" , ByteString "background-image:-webkit-gradient(" , ByteString "linear,left bottom, left top," , ByteString "color-stop(0.50, rgb(255,255,255))," , ByteString "color-stop(1.00, rgb(224,234,247)));" , ByteString "background-image:-moz-linear-gradient(" , ByteString "center bottom, white 50%, rgb(224,234,247) 100%);" , ByteString "padding: 40px 40px 40px 40px }" , ByteString "div.footer {" , ByteString "padding: 16px 0px 10px 10px; height:31px;" , ByteString "border-top: 1px solid rgb(194,209,225);" , ByteString "color: rgb(160,172,186); font-size:10pt;" , ByteString "background: rgb(245,249,255) }" , ByteString "table { max-width:100%; margin: 0 auto;" ByteString -> ByteString -> ByteString `S.append` ByteString " border-collapse: collapse; }" , ByteString "tr:hover { background:rgb(256,256,224) }" , ByteString "td { border:0; font-family:monospace; padding: 2px 0; }" , ByteString "td.filename, td.type { padding-right: 2em; }" , ByteString "th { border:0; background:rgb(28,56,97);" , ByteString "text-shadow:-1px 3px 1px rgb(16,33,57); color: white}" ] ------------------------------------------------------------------------------ -- | An automatic index generator, which is fairly small and does not rely on -- any external files (which may not be there depending on external request -- routing). -- -- A 'MimeMap' is passed in to display the types of files in the directory -- listing based on their extension. Preferably, this is the same as the map -- in the 'DirectoryConfig' -- -- The styles parameter allows you to apply styles to the directory listing. -- The listing itself consists of a table, containing a header row using -- th elements, and one row per file using td elements, so styles for those -- pieces may be attached to the appropriate tags. defaultIndexGenerator :: MonadSnap m => MimeMap -- ^ MIME type mapping for reporting types -> ByteString -- ^ Style info to insert in header -> FilePath -- ^ Directory to generate index for -> m () defaultIndexGenerator :: forall (m :: * -> *). MonadSnap m => MimeMap -> ByteString -> FilePath -> m () defaultIndexGenerator MimeMap mm ByteString styles FilePath d = do forall (m :: * -> *). MonadSnap m => (Response -> Response) -> m () modifyResponse forall a b. (a -> b) -> a -> b $ ByteString -> Response -> Response setContentType ByteString "text/html; charset=utf-8" Request rq <- forall (m :: * -> *). MonadSnap m => m Request getRequest let uri :: ByteString uri = Request -> ByteString uriWithoutQueryString Request rq let pInfo :: ByteString pInfo = Request -> ByteString rqPathInfo Request rq forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString "<!DOCTYPE html>\n<html>\n<head>" forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString "<title>Directory Listing: " forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString uri forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString "</title>" forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString "<style type='text/css'>" forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString styles forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString "</style></head><body>" forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString "<div class=\"header\">Directory Listing: " forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString uri forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString "</div><div class=\"content\">" forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString "<table><tr><th>File Name</th><th>Type</th><th>Last Modified" forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString "</th></tr>" forall (f :: * -> *). Applicative f => Bool -> f () -> f () when (ByteString pInfo forall a. Eq a => a -> a -> Bool /= ByteString "") forall a b. (a -> b) -> a -> b $ forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString "<tr><td><a href='../'>..</a></td><td colspan=2>DIR</td></tr>" [FilePath] entries <- forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO forall a b. (a -> b) -> a -> b $ FilePath -> IO [FilePath] getDirectoryContents FilePath d [FilePath] dirs <- forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO forall a b. (a -> b) -> a -> b $ forall (m :: * -> *) a. Applicative m => (a -> m Bool) -> [a] -> m [a] filterM (FilePath -> IO Bool doesDirectoryExist forall b c a. (b -> c) -> (a -> b) -> a -> c . (FilePath d FilePath -> FilePath -> FilePath </>)) [FilePath] entries [FilePath] files <- forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO forall a b. (a -> b) -> a -> b $ forall (m :: * -> *) a. Applicative m => (a -> m Bool) -> [a] -> m [a] filterM (FilePath -> IO Bool doesFileExist forall b c a. (b -> c) -> (a -> b) -> a -> c . (FilePath d FilePath -> FilePath -> FilePath </>)) [FilePath] entries forall (t :: * -> *) (m :: * -> *) a b. (Foldable t, Monad m) => t a -> (a -> m b) -> m () forM_ (forall a. Ord a => [a] -> [a] sort forall a b. (a -> b) -> a -> b $ forall a. (a -> Bool) -> [a] -> [a] filter (Bool -> Bool not forall b c a. (b -> c) -> (a -> b) -> a -> c . (forall (t :: * -> *) a. (Foldable t, Eq a) => a -> t a -> Bool `elem` [FilePath "..", FilePath "."])) [FilePath] dirs) forall a b. (a -> b) -> a -> b $ \FilePath f0 -> do ByteString f <- forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO forall a b. (a -> b) -> a -> b $ forall (m :: * -> *) a1 r. Monad m => (a1 -> r) -> m a1 -> m r liftM (\Text s -> Text -> ByteString T.encodeUtf8 Text s forall a. Monoid a => a -> a -> a `mappend` ByteString "/") forall a b. (a -> b) -> a -> b $ FilePath -> IO Text decodeFilePath FilePath f0 forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString "<tr><td class='filename'><a href='" forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString f forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString "'>" forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString f forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString "</a></td><td class='type' colspan=2>DIR</td></tr>" forall (t :: * -> *) (m :: * -> *) a b. (Foldable t, Monad m) => t a -> (a -> m b) -> m () forM_ (forall a. Ord a => [a] -> [a] sort [FilePath] files) forall a b. (a -> b) -> a -> b $ \FilePath f0 -> do ByteString f <- forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO forall a b. (a -> b) -> a -> b $ forall (m :: * -> *) a1 r. Monad m => (a1 -> r) -> m a1 -> m r liftM Text -> ByteString T.encodeUtf8 forall a b. (a -> b) -> a -> b $ FilePath -> IO Text decodeFilePath FilePath f0 FileStatus stat <- forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO forall a b. (a -> b) -> a -> b $ FilePath -> IO FileStatus getFileStatus (FilePath d FilePath -> FilePath -> FilePath </> FilePath f0) ByteString tm <- forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO forall a b. (a -> b) -> a -> b $ CTime -> IO ByteString formatHttpTime (FileStatus -> CTime modificationTime FileStatus stat) forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString "<tr><td class='filename'><a href='" forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString f forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString "'>" forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString f forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString "</a></td><td class='type'>" forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS (MimeMap -> FilePath -> ByteString fileType MimeMap mm FilePath f0) forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString "</td><td>" forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString tm forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString "</tr>" forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString "</table></div><div class=\"footer\">Powered by " forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString "<b><a href=\"http://snapframework.com/\">Snap</a></b></div>" forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString "</body>" ------------------------------------------------------------------------------ decodeFilePath :: FilePath -> IO T.Text decodeFilePath :: FilePath -> IO Text decodeFilePath FilePath fp = do forall (m :: * -> *) a. MonadBase IO m => a -> m a evaluate (ByteString -> Text T.decodeUtf8 ByteString bs) forall (m :: * -> *) e a. (MonadBaseControl IO m, Exception e) => m a -> (e -> m a) -> m a `catch` (\(SomeException _::SomeException) -> forall (m :: * -> *) a. Monad m => a -> m a return (FilePath -> Text T.pack FilePath fp)) where bs :: ByteString bs = FilePath -> ByteString S.pack FilePath fp ------------------------------------------------------------------------------ -- | A very simple configuration for directory serving. This configuration -- uses built-in MIME types from 'defaultMimeTypes', and has no index files, -- index generator, dynamic file handlers, or 'preServeHook'. simpleDirectoryConfig :: MonadSnap m => DirectoryConfig m simpleDirectoryConfig :: forall (m :: * -> *). MonadSnap m => DirectoryConfig m simpleDirectoryConfig = DirectoryConfig { indexFiles :: [FilePath] indexFiles = [], indexGenerator :: FilePath -> m () indexGenerator = forall a b. a -> b -> a const forall (m :: * -> *) a. MonadSnap m => m a pass, dynamicHandlers :: HandlerMap m dynamicHandlers = forall k v. HashMap k v Map.empty, mimeTypes :: MimeMap mimeTypes = MimeMap defaultMimeTypes, preServeHook :: FilePath -> m () preServeHook = forall a b. a -> b -> a const forall a b. (a -> b) -> a -> b $ forall (m :: * -> *) a. Monad m => a -> m a return forall a b. (a -> b) -> a -> b $! () } ------------------------------------------------------------------------------ -- | A reasonable default configuration for directory serving. This -- configuration uses built-in MIME types from 'defaultMimeTypes', serves -- common index files @index.html@ and @index.htm@, but does not autogenerate -- directory indexes, nor have any dynamic file handlers. The 'preServeHook' -- will not do anything. defaultDirectoryConfig :: MonadSnap m => DirectoryConfig m defaultDirectoryConfig :: forall (m :: * -> *). MonadSnap m => DirectoryConfig m defaultDirectoryConfig = DirectoryConfig { indexFiles :: [FilePath] indexFiles = [FilePath "index.html", FilePath "index.htm"], indexGenerator :: FilePath -> m () indexGenerator = forall a b. a -> b -> a const forall (m :: * -> *) a. MonadSnap m => m a pass, dynamicHandlers :: HandlerMap m dynamicHandlers = forall k v. HashMap k v Map.empty, mimeTypes :: MimeMap mimeTypes = MimeMap defaultMimeTypes, preServeHook :: FilePath -> m () preServeHook = forall a b. a -> b -> a const forall a b. (a -> b) -> a -> b $ forall (m :: * -> *) a. Monad m => a -> m a return forall a b. (a -> b) -> a -> b $! () } ------------------------------------------------------------------------------ -- | A more elaborate configuration for file serving. This configuration -- uses built-in MIME types from 'defaultMimeTypes', serves common index files -- @index.html@ and @index.htm@, and autogenerates directory indexes with a -- Snap-like feel. It still has no dynamic file handlers, nor 'preServeHook', -- which should be added as needed. -- -- Files recognized as indexes include @index.html@, @index.htm@, -- @default.html@, @default.htm@, @home.html@ -- -- Example of how the autogenerated directory index looks like: -- -- <<>> fancyDirectoryConfig :: MonadSnap m => DirectoryConfig m fancyDirectoryConfig :: forall (m :: * -> *). MonadSnap m => DirectoryConfig m fancyDirectoryConfig = DirectoryConfig { indexFiles :: [FilePath] indexFiles = [FilePath "index.html", FilePath "index.htm"], indexGenerator :: FilePath -> m () indexGenerator = forall (m :: * -> *). MonadSnap m => MimeMap -> ByteString -> FilePath -> m () defaultIndexGenerator MimeMap defaultMimeTypes ByteString snapIndexStyles, dynamicHandlers :: HandlerMap m dynamicHandlers = forall k v. HashMap k v Map.empty, mimeTypes :: MimeMap mimeTypes = MimeMap defaultMimeTypes, preServeHook :: FilePath -> m () preServeHook = forall a b. a -> b -> a const forall a b. (a -> b) -> a -> b $ forall (m :: * -> *) a. Monad m => a -> m a return forall a b. (a -> b) -> a -> b $! () } ------------------------------------------------------------------------------ -- | Serves static files from a directory using the default configuration -- as given in 'defaultDirectoryConfig'. serveDirectory :: MonadSnap m => FilePath -- ^ Directory to serve from -> m () serveDirectory :: forall (m :: * -> *). MonadSnap m => FilePath -> m () serveDirectory = forall (m :: * -> *). MonadSnap m => DirectoryConfig m -> FilePath -> m () serveDirectoryWith forall (m :: * -> *). MonadSnap m => DirectoryConfig m defaultDirectoryConfig {-# INLINE serveDirectory #-} ------------------------------------------------------------------------------ -- | Serves static files from a directory. Configuration options are -- passed in a 'DirectoryConfig' that captures various choices about desired -- behavior. The relative path given in 'rqPathInfo' is searched for a -- requested file, and the file is served with the appropriate mime type if it -- is found. Absolute paths and \"@..@\" are prohibited to prevent files from -- being served from outside the sandbox. serveDirectoryWith :: MonadSnap m => DirectoryConfig m -- ^ Configuration options -> FilePath -- ^ Directory to serve from -> m () serveDirectoryWith :: forall (m :: * -> *). MonadSnap m => DirectoryConfig m -> FilePath -> m () serveDirectoryWith DirectoryConfig m cfg FilePath base = do Bool b <- m Bool directory forall (f :: * -> *) a. Alternative f => f a -> f a -> f a <|> m Bool file forall (f :: * -> *) a. Alternative f => f a -> f a -> f a <|> forall {b}. m b redir forall (f :: * -> *). Applicative f => Bool -> f () -> f () when (Bool -> Bool not Bool b) forall (m :: * -> *) a. MonadSnap m => m a pass where idxs :: [FilePath] idxs = forall (m :: * -> *). DirectoryConfig m -> [FilePath] indexFiles DirectoryConfig m cfg generate :: FilePath -> m () generate = forall (m :: * -> *). DirectoryConfig m -> FilePath -> m () indexGenerator DirectoryConfig m cfg mimes :: MimeMap mimes = forall (m :: * -> *). DirectoryConfig m -> MimeMap mimeTypes DirectoryConfig m cfg dyns :: HandlerMap m dyns = forall (m :: * -> *). DirectoryConfig m -> HandlerMap m dynamicHandlers DirectoryConfig m cfg pshook :: FilePath -> m () pshook = forall (m :: * -> *). DirectoryConfig m -> FilePath -> m () preServeHook DirectoryConfig m cfg -- Serves a file if it exists; passes if not serve :: FilePath -> m Bool serve FilePath f = do forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO (FilePath -> IO Bool doesFileExist FilePath f) forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b >>= forall a b c. (a -> b -> c) -> b -> a -> c flip forall (f :: * -> *). Applicative f => Bool -> f () -> f () unless forall (m :: * -> *) a. MonadSnap m => m a pass let fname :: FilePath fname = FilePath -> FilePath takeFileName FilePath f let staticServe :: FilePath -> m () staticServe FilePath f' = FilePath -> m () pshook FilePath f forall (m :: * -> *) a b. Monad m => m a -> m b -> m b >> forall (m :: * -> *). MonadSnap m => ByteString -> FilePath -> m () serveFileAs (MimeMap -> FilePath -> ByteString fileType MimeMap mimes FilePath fname) FilePath f' forall a. a -> HashMap FilePath a -> FilePath -> a lookupExt FilePath -> m () staticServe HandlerMap m dyns FilePath fname FilePath f forall (m :: * -> *) a b. Monad m => m a -> m b -> m b >> forall (m :: * -> *) a. Monad m => a -> m a return Bool True -- Serves a directory via indices if available. Returns True on success, -- False on failure to find an index. Passes /only/ if the request was -- not for a directory (no trailing slash). directory :: m Bool directory = do Request rq <- forall (m :: * -> *). MonadSnap m => m Request getRequest let uri :: ByteString uri = Request -> ByteString uriWithoutQueryString Request rq forall (f :: * -> *). Applicative f => Bool -> f () -> f () unless (ByteString "/" ByteString -> ByteString -> Bool `S.isSuffixOf` ByteString uri) forall (m :: * -> *) a. MonadSnap m => m a pass FilePath rel <- (FilePath base FilePath -> FilePath -> FilePath </>) forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b <$> forall (m :: * -> *). MonadSnap m => m FilePath getSafePath Bool b <- forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO forall a b. (a -> b) -> a -> b $ FilePath -> IO Bool doesDirectoryExist FilePath rel if Bool b then do let serveRel :: FilePath -> m Bool serveRel FilePath f = FilePath -> m Bool serve (FilePath rel FilePath -> FilePath -> FilePath </> FilePath f) forall (t :: * -> *) b a. Foldable t => (b -> a -> b) -> b -> t a -> b foldl' forall (f :: * -> *) a. Alternative f => f a -> f a -> f a (<|>) forall (m :: * -> *) a. MonadSnap m => m a pass (forall a b. (a -> b) -> [a] -> [b] Prelude.map FilePath -> m Bool serveRel [FilePath] idxs) forall (f :: * -> *) a. Alternative f => f a -> f a -> f a <|> (FilePath -> m () generate FilePath rel forall (m :: * -> *) a b. Monad m => m a -> m b -> m b >> forall (m :: * -> *) a. Monad m => a -> m a return Bool True) forall (f :: * -> *) a. Alternative f => f a -> f a -> f a <|> forall (m :: * -> *) a. Monad m => a -> m a return Bool False else forall (m :: * -> *) a. Monad m => a -> m a return Bool False -- Serves a file requested by name. Passes if the file doesn't exist. file :: m Bool file = FilePath -> m Bool serve forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b =<< ((FilePath base FilePath -> FilePath -> FilePath </>) forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b <$> forall (m :: * -> *). MonadSnap m => m FilePath getSafePath) -- If the request is for a directory but lacks a trailing slash, redirects -- to the directory name with a trailing slash. redir :: m b redir = do FilePath rel <- (FilePath base FilePath -> FilePath -> FilePath </>) forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b <$> forall (m :: * -> *). MonadSnap m => m FilePath getSafePath forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO (FilePath -> IO Bool doesDirectoryExist FilePath rel) forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b >>= forall a b c. (a -> b -> c) -> b -> a -> c flip forall (f :: * -> *). Applicative f => Bool -> f () -> f () unless forall (m :: * -> *) a. MonadSnap m => m a pass Request rq <- forall (m :: * -> *). MonadSnap m => m Request getRequest let uri :: ByteString uri = Request -> ByteString uriWithoutQueryString Request rq let qss :: ByteString qss = Request -> ByteString queryStringSuffix Request rq let u :: ByteString u = [ByteString] -> ByteString S.concat [ByteString uri, ByteString "/", ByteString qss] forall (m :: * -> *) a. MonadSnap m => ByteString -> m a redirect ByteString u ------------------------------------------------------------------------------ -- | Serves a single file specified by a full or relative path. If the file -- does not exist, throws an exception (not that it does /not/ pass to the -- next handler). The path restrictions on 'serveDirectory' don't apply to -- this function since the path is not being supplied by the user. serveFile :: MonadSnap m => FilePath -- ^ path to file -> m () serveFile :: forall (m :: * -> *). MonadSnap m => FilePath -> m () serveFile FilePath fp = forall (m :: * -> *). MonadSnap m => ByteString -> FilePath -> m () serveFileAs (MimeMap -> FilePath -> ByteString fileType MimeMap defaultMimeTypes (FilePath -> FilePath takeFileName FilePath fp)) FilePath fp {-# INLINE serveFile #-} ------------------------------------------------------------------------------ -- | Same as 'serveFile', with control over the MIME mapping used. serveFileAs :: MonadSnap m => ByteString -- ^ MIME type -> FilePath -- ^ path to file -> m () serveFileAs :: forall (m :: * -> *). MonadSnap m => ByteString -> FilePath -> m () serveFileAs ByteString mime FilePath fp = do Request reqOrig <- forall (m :: * -> *). MonadSnap m => m Request getRequest -- If-Range header must be ignored if there is no Range: header in the -- request (RFC 2616 section 14.27) let req :: Request req = if forall a. Maybe a -> Bool isNothing forall a b. (a -> b) -> a -> b $ forall a. HasHeaders a => CI ByteString -> a -> Maybe ByteString getHeader CI ByteString "range" Request reqOrig then forall a. HasHeaders a => CI ByteString -> a -> a deleteHeader CI ByteString "if-range" Request reqOrig else Request reqOrig -- check "If-Modified-Since" and "If-Range" headers let mbH :: Maybe ByteString mbH = forall a. HasHeaders a => CI ByteString -> a -> Maybe ByteString getHeader CI ByteString "if-modified-since" Request req Maybe CTime mbIfModified <- forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO forall a b. (a -> b) -> a -> b $ case Maybe ByteString mbH of Maybe ByteString Nothing -> forall (m :: * -> *) a. Monad m => a -> m a return forall a. Maybe a Nothing (Just ByteString s) -> forall (m :: * -> *) a1 r. Monad m => (a1 -> r) -> m a1 -> m r liftM forall a. a -> Maybe a Just forall a b. (a -> b) -> a -> b $ ByteString -> IO CTime parseHttpTime ByteString s -- If-Range header could contain an entity, but then parseHttpTime will -- fail and return 0 which means a 200 response will be generated anyways Maybe CTime mbIfRange <- forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO forall a b. (a -> b) -> a -> b $ case forall a. HasHeaders a => CI ByteString -> a -> Maybe ByteString getHeader CI ByteString "if-range" Request req of Maybe ByteString Nothing -> forall (m :: * -> *) a. Monad m => a -> m a return forall a. Maybe a Nothing (Just ByteString s) -> forall (m :: * -> *) a1 r. Monad m => (a1 -> r) -> m a1 -> m r liftM forall a. a -> Maybe a Just forall a b. (a -> b) -> a -> b $ ByteString -> IO CTime parseHttpTime ByteString s forall (m :: * -> *). MonadIO m => FilePath -> m () dbg forall a b. (a -> b) -> a -> b $ FilePath "mbIfModified: " forall a. [a] -> [a] -> [a] ++ forall a. Show a => a -> FilePath Prelude.show Maybe CTime mbIfModified forall (m :: * -> *). MonadIO m => FilePath -> m () dbg forall a b. (a -> b) -> a -> b $ FilePath "mbIfRange: " forall a. [a] -> [a] -> [a] ++ forall a. Show a => a -> FilePath Prelude.show Maybe CTime mbIfRange -- check modification time and bug out early if the file is not modified. -- -- TODO: a stat cache would be nice here, but it'd need the date thread -- stuff from snap-server to be folded into snap-core FileStatus filestat <- forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO forall a b. (a -> b) -> a -> b $ FilePath -> IO FileStatus getFileStatus FilePath fp let mt :: CTime mt = FileStatus -> CTime modificationTime FileStatus filestat forall b a. b -> (a -> b) -> Maybe a -> b maybe (forall (m :: * -> *) a. Monad m => a -> m a return forall a b. (a -> b) -> a -> b $! ()) (\CTime lt -> forall (f :: * -> *). Applicative f => Bool -> f () -> f () when (CTime mt forall a. Ord a => a -> a -> Bool <= CTime lt) forall {a}. m a notModified) Maybe CTime mbIfModified let sz :: Word64 sz = forall a b. (Integral a, Num b) => a -> b fromIntegral forall a b. (a -> b) -> a -> b $ FileStatus -> FileOffset fileSize FileStatus filestat ByteString lm <- forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO forall a b. (a -> b) -> a -> b $ CTime -> IO ByteString formatHttpTime CTime mt -- ok, at this point we know the last-modified time and the -- content-type. set those. forall (m :: * -> *). MonadSnap m => (Response -> Response) -> m () modifyResponse forall a b. (a -> b) -> a -> b $ forall a. HasHeaders a => CI ByteString -> ByteString -> a -> a setHeader CI ByteString "Last-Modified" ByteString lm forall b c a. (b -> c) -> (a -> b) -> a -> c . forall a. HasHeaders a => CI ByteString -> ByteString -> a -> a setHeader CI ByteString "Accept-Ranges" ByteString "bytes" forall b c a. (b -> c) -> (a -> b) -> a -> c . ByteString -> Response -> Response setContentType ByteString mime -- now check: is this a range request? If there is an 'If-Range' header -- with an old modification time we skip this check and send a 200 -- response let skipRangeCheck :: Bool skipRangeCheck = forall b a. b -> (a -> b) -> Maybe a -> b maybe (Bool False) (\CTime lt -> CTime mt forall a. Ord a => a -> a -> Bool > CTime lt) Maybe CTime mbIfRange -- checkRangeReq checks for a Range: header in the request and sends a -- partial response if it matches. Bool wasRange <- if Bool skipRangeCheck then forall (m :: * -> *) a. Monad m => a -> m a return Bool False else forall (m :: * -> *) a. MonadSnap m => Snap a -> m a liftSnap forall a b. (a -> b) -> a -> b $ forall (m :: * -> *). MonadSnap m => Request -> FilePath -> Word64 -> m Bool checkRangeReq Request req FilePath fp Word64 sz forall (m :: * -> *). MonadIO m => FilePath -> m () dbg forall a b. (a -> b) -> a -> b $ FilePath "was this a range request? " forall a. [a] -> [a] -> [a] ++ forall a. Show a => a -> FilePath Prelude.show Bool wasRange -- if we didn't have a range request, we just do normal sendfile forall (f :: * -> *). Applicative f => Bool -> f () -> f () unless Bool wasRange forall a b. (a -> b) -> a -> b $ do forall (m :: * -> *). MonadSnap m => (Response -> Response) -> m () modifyResponse forall a b. (a -> b) -> a -> b $ Int -> Response -> Response setResponseCode Int 200 forall b c a. (b -> c) -> (a -> b) -> a -> c . Word64 -> Response -> Response setContentLength Word64 sz forall (m :: * -> *) a. MonadSnap m => Snap a -> m a liftSnap forall a b. (a -> b) -> a -> b $ forall (m :: * -> *). MonadSnap m => FilePath -> m () sendFile FilePath fp where -------------------------------------------------------------------------- notModified :: m a notModified = forall (m :: * -> *) a. MonadSnap m => Response -> m a finishWith forall a b. (a -> b) -> a -> b $ Int -> Response -> Response setResponseCode Int 304 Response emptyResponse ------------------------------------------------------------------------------ lookupExt :: a -> HashMap FilePath a -> FilePath -> a lookupExt :: forall a. a -> HashMap FilePath a -> FilePath -> a lookupExt a def HashMap FilePath a m FilePath f = if forall (t :: * -> *) a. Foldable t => t a -> Bool null FilePath ext then a def else forall a. a -> Maybe a -> a fromMaybe (forall a. a -> HashMap FilePath a -> FilePath -> a lookupExt a def HashMap FilePath a m (FilePath -> FilePath next FilePath ext)) Maybe a mbe where next :: FilePath -> FilePath next = forall a. (a -> Bool) -> [a] -> [a] dropWhile (forall a. Eq a => a -> a -> Bool /= Char '.') forall b c a. (b -> c) -> (a -> b) -> a -> c . forall a. Int -> [a] -> [a] drop Int 1 ext :: FilePath ext = FilePath -> FilePath takeExtensions FilePath f mbe :: Maybe a mbe = forall k v. (Eq k, Hashable k) => k -> HashMap k v -> Maybe v Map.lookup FilePath ext HashMap FilePath a m ------------------------------------------------------------------------------ -- | Determine a given file's MIME type from its filename and the provided MIME -- map. fileType :: MimeMap -> FilePath -> ByteString fileType :: MimeMap -> FilePath -> ByteString fileType = forall a. a -> HashMap FilePath a -> FilePath -> a lookupExt ByteString defaultMimeType ------------------------------------------------------------------------------ defaultMimeType :: ByteString defaultMimeType :: ByteString defaultMimeType = ByteString "application/octet-stream" ------------------------------------------------------------------------------ data RangeReq = RangeReq !Word64 !(Maybe Word64) | SuffixRangeReq !Word64 ------------------------------------------------------------------------------ rangeParser :: Parser RangeReq rangeParser :: Parser RangeReq rangeParser = ByteString -> Parser ByteString string ByteString "bytes=" forall (f :: * -> *) a b. Applicative f => f a -> f b -> f b *> (Parser RangeReq byteRangeSpec forall (f :: * -> *) a. Alternative f => f a -> f a -> f a <|> Parser RangeReq suffixByteRangeSpec) forall (f :: * -> *) a b. Applicative f => f a -> f b -> f a <* forall t. Chunk t => Parser t () endOfInput where byteRangeSpec :: Parser RangeReq byteRangeSpec = do Word64 start <- forall a b. (Integral a, Num b) => a -> b fromIntegral forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b <$> Parser Int64 parseNum forall (f :: * -> *) a. Functor f => f a -> f () void forall a b. (a -> b) -> a -> b $! Char -> Parser Char char Char '-' Maybe Int64 end <- forall (f :: * -> *) a. Alternative f => a -> f a -> f a option forall a. Maybe a Nothing forall a b. (a -> b) -> a -> b $ forall (m :: * -> *) a1 r. Monad m => (a1 -> r) -> m a1 -> m r liftM forall a. a -> Maybe a Just Parser Int64 parseNum forall (m :: * -> *) a. Monad m => a -> m a return forall a b. (a -> b) -> a -> b $! Word64 -> Maybe Word64 -> RangeReq RangeReq Word64 start (forall a b. (Integral a, Num b) => a -> b fromIntegral forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b <$> Maybe Int64 end) suffixByteRangeSpec :: Parser RangeReq suffixByteRangeSpec = forall (m :: * -> *) a1 r. Monad m => (a1 -> r) -> m a1 -> m r liftM (Word64 -> RangeReq SuffixRangeReq forall b c a. (b -> c) -> (a -> b) -> a -> c . forall a b. (Integral a, Num b) => a -> b fromIntegral) forall a b. (a -> b) -> a -> b $ Char -> Parser Char char Char '-' forall (f :: * -> *) a b. Applicative f => f a -> f b -> f b *> Parser Int64 parseNum ------------------------------------------------------------------------------ checkRangeReq :: (MonadSnap m) => Request -> FilePath -> Word64 -> m Bool checkRangeReq :: forall (m :: * -> *). MonadSnap m => Request -> FilePath -> Word64 -> m Bool checkRangeReq Request req FilePath fp Word64 sz = do -- TODO/FIXME: multiple ranges forall b a. b -> (a -> b) -> Maybe a -> b maybe (forall (m :: * -> *) a. Monad m => a -> m a return Bool False) (\ByteString s -> forall a c b. (a -> c) -> (b -> c) -> Either a b -> c either (forall a b. a -> b -> a const forall a b. (a -> b) -> a -> b $ forall (m :: * -> *) a. Monad m => a -> m a return Bool False) RangeReq -> m Bool withRange (forall a. ByteString -> Parser a -> Either FilePath a fullyParse ByteString s Parser RangeReq rangeParser)) (forall a. HasHeaders a => CI ByteString -> a -> Maybe ByteString getHeader CI ByteString "range" Request req) where withRange :: RangeReq -> m Bool withRange (RangeReq Word64 start Maybe Word64 mend) = do let end :: Word64 end = forall a. a -> Maybe a -> a fromMaybe (Word64 szforall a. Num a => a -> a -> a -Word64 1) Maybe Word64 mend forall (m :: * -> *). MonadIO m => FilePath -> m () dbg forall a b. (a -> b) -> a -> b $ FilePath "withRange: start=" forall a. [a] -> [a] -> [a] ++ forall a. Show a => a -> FilePath Prelude.show Word64 start forall a. [a] -> [a] -> [a] ++ FilePath ", end=" forall a. [a] -> [a] -> [a] ++ forall a. Show a => a -> FilePath Prelude.show Word64 end if Word64 start forall a. Ord a => a -> a -> Bool < Word64 0 Bool -> Bool -> Bool || Word64 end forall a. Ord a => a -> a -> Bool < Word64 start Bool -> Bool -> Bool || Word64 start forall a. Ord a => a -> a -> Bool >= Word64 sz Bool -> Bool -> Bool || Word64 end forall a. Ord a => a -> a -> Bool >= Word64 sz then m Bool send416 else forall {m :: * -> *}. MonadSnap m => Word64 -> Word64 -> m Bool send206 Word64 start Word64 end withRange (SuffixRangeReq Word64 nbytes) = do let end :: Word64 end = Word64 szforall a. Num a => a -> a -> a -Word64 1 let start :: Word64 start = Word64 sz forall a. Num a => a -> a -> a - Word64 nbytes forall (m :: * -> *). MonadIO m => FilePath -> m () dbg forall a b. (a -> b) -> a -> b $ FilePath "withRange: start=" forall a. [a] -> [a] -> [a] ++ forall a. Show a => a -> FilePath Prelude.show Word64 start forall a. [a] -> [a] -> [a] ++ FilePath ", end=" forall a. [a] -> [a] -> [a] ++ forall a. Show a => a -> FilePath Prelude.show Word64 end if Word64 start forall a. Ord a => a -> a -> Bool < Word64 0 Bool -> Bool -> Bool || Word64 end forall a. Ord a => a -> a -> Bool < Word64 start Bool -> Bool -> Bool || Word64 start forall a. Ord a => a -> a -> Bool >= Word64 sz Bool -> Bool -> Bool || Word64 end forall a. Ord a => a -> a -> Bool >= Word64 sz then m Bool send416 else forall {m :: * -> *}. MonadSnap m => Word64 -> Word64 -> m Bool send206 Word64 start Word64 end -- note: start and end INCLUSIVE here send206 :: Word64 -> Word64 -> m Bool send206 Word64 start Word64 end = do forall (m :: * -> *). MonadIO m => FilePath -> m () dbg FilePath "inside send206" let !len :: Word64 len = Word64 endforall a. Num a => a -> a -> a -Word64 startforall a. Num a => a -> a -> a +Word64 1 let crng :: ByteString crng = [ByteString] -> ByteString S.concat forall b c a. (b -> c) -> (a -> b) -> a -> c . ByteString -> [ByteString] L.toChunks forall a b. (a -> b) -> a -> b $ Builder -> ByteString toLazyByteString forall a b. (a -> b) -> a -> b $ forall a. Monoid a => [a] -> a mconcat [ ByteString -> Builder byteString ByteString "bytes " , forall a. Show a => a -> Builder fromShow Word64 start , Char -> Builder char8 Char '-' , forall a. Show a => a -> Builder fromShow Word64 end , Char -> Builder char8 Char '/' , forall a. Show a => a -> Builder fromShow Word64 sz ] forall (m :: * -> *). MonadSnap m => (Response -> Response) -> m () modifyResponse forall a b. (a -> b) -> a -> b $ Int -> Response -> Response setResponseCode Int 206 forall b c a. (b -> c) -> (a -> b) -> a -> c . forall a. HasHeaders a => CI ByteString -> ByteString -> a -> a setHeader CI ByteString "Content-Range" ByteString crng forall b c a. (b -> c) -> (a -> b) -> a -> c . Word64 -> Response -> Response setContentLength Word64 len forall (m :: * -> *). MonadIO m => FilePath -> m () dbg forall a b. (a -> b) -> a -> b $ FilePath "send206: sending range (" forall a. [a] -> [a] -> [a] ++ forall a. Show a => a -> FilePath Prelude.show Word64 start forall a. [a] -> [a] -> [a] ++ FilePath "," forall a. [a] -> [a] -> [a] ++ forall a. Show a => a -> FilePath Prelude.show (Word64 endforall a. Num a => a -> a -> a +Word64 1) forall a. [a] -> [a] -> [a] ++ FilePath ") to sendFilePartial" -- end here was inclusive, sendFilePartial is exclusive forall (m :: * -> *). MonadSnap m => FilePath -> (Word64, Word64) -> m () sendFilePartial FilePath fp (Word64 start,Word64 endforall a. Num a => a -> a -> a +Word64 1) forall (m :: * -> *) a. Monad m => a -> m a return Bool True send416 :: m Bool send416 = do forall (m :: * -> *). MonadIO m => FilePath -> m () dbg FilePath "inside send416" -- if there's an "If-Range" header in the request, then we just send -- back 200 if forall a. HasHeaders a => CI ByteString -> a -> Maybe ByteString getHeader CI ByteString "If-Range" Request req forall a. Eq a => a -> a -> Bool /= forall a. Maybe a Nothing then forall (m :: * -> *) a. Monad m => a -> m a return Bool False else do let crng :: ByteString crng = [ByteString] -> ByteString S.concat forall b c a. (b -> c) -> (a -> b) -> a -> c . ByteString -> [ByteString] L.toChunks forall a b. (a -> b) -> a -> b $ Builder -> ByteString toLazyByteString forall a b. (a -> b) -> a -> b $ forall a. Monoid a => [a] -> a mconcat [ ByteString -> Builder byteString ByteString "bytes */" , forall a. Show a => a -> Builder fromShow Word64 sz ] forall (m :: * -> *). MonadSnap m => (Response -> Response) -> m () modifyResponse forall a b. (a -> b) -> a -> b $ Int -> Response -> Response setResponseCode Int 416 forall b c a. (b -> c) -> (a -> b) -> a -> c . forall a. HasHeaders a => CI ByteString -> ByteString -> a -> a setHeader CI ByteString "Content-Range" ByteString crng forall b c a. (b -> c) -> (a -> b) -> a -> c . Word64 -> Response -> Response setContentLength Word64 0 forall b c a. (b -> c) -> (a -> b) -> a -> c . forall a. HasHeaders a => CI ByteString -> a -> a deleteHeader CI ByteString "Content-Type" forall b c a. (b -> c) -> (a -> b) -> a -> c . forall a. HasHeaders a => CI ByteString -> a -> a deleteHeader CI ByteString "Content-Encoding" forall b c a. (b -> c) -> (a -> b) -> a -> c . forall a. HasHeaders a => CI ByteString -> a -> a deleteHeader CI ByteString "Transfer-Encoding" forall b c a. (b -> c) -> (a -> b) -> a -> c . (OutputStream Builder -> IO (OutputStream Builder)) -> Response -> Response setResponseBody (forall (m :: * -> *) a. Monad m => a -> m a return forall b c a. (b -> c) -> (a -> b) -> a -> c . forall a. a -> a id) forall (m :: * -> *) a. Monad m => a -> m a return Bool True ------------------------------------------------------------------------------ dbg :: (MonadIO m) => String -> m () dbg :: forall (m :: * -> *). MonadIO m => FilePath -> m () dbg FilePath s = forall (m :: * -> *). MonadIO m => FilePath -> m () debug forall a b. (a -> b) -> a -> b $ FilePath "FileServe:" forall a. [a] -> [a] -> [a] ++ FilePath s ------------------------------------------------------------------------------ uriWithoutQueryString :: Request -> ByteString uriWithoutQueryString :: Request -> ByteString uriWithoutQueryString Request rq = (Char -> Bool) -> ByteString -> ByteString S.takeWhile (forall a. Eq a => a -> a -> Bool /= Char '?') ByteString uri where uri :: ByteString uri = Request -> ByteString rqURI Request rq ------------------------------------------------------------------------------ queryStringSuffix :: Request -> ByteString queryStringSuffix :: Request -> ByteString queryStringSuffix Request rq = [ByteString] -> ByteString S.concat [ ByteString s, ByteString qs ] where qs :: ByteString qs = Request -> ByteString rqQueryString Request rq s :: ByteString s = if ByteString -> Bool S.null ByteString qs then ByteString "" else ByteString "?" ------------------------------------------------------------------------------ fromShow :: Show a => a -> Builder fromShow :: forall a. Show a => a -> Builder fromShow = FilePath -> Builder stringUtf8 forall b c a. (b -> c) -> (a -> b) -> a -> c . forall a. Show a => a -> FilePath show