{-# 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 ) 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 :: m FilePath getSafePath = do Request req <- m Request forall (m :: * -> *). MonadSnap m => m Request getRequest let mp :: Maybe ByteString mp = ByteString -> Maybe ByteString urlDecode (ByteString -> Maybe ByteString) -> ByteString -> Maybe ByteString forall a b. (a -> b) -> a -> b $ Request -> ByteString rqPathInfo Request req FilePath p <- m FilePath -> (ByteString -> m FilePath) -> Maybe ByteString -> m FilePath forall b a. b -> (a -> b) -> Maybe a -> b maybe m FilePath forall (m :: * -> *) a. MonadSnap m => m a pass (FilePath -> m FilePath forall (m :: * -> *) a. Monad m => a -> m a return (FilePath -> m FilePath) -> (ByteString -> FilePath) -> ByteString -> m FilePath forall b c a. (b -> c) -> (a -> b) -> a -> c . Text -> FilePath T.unpack (Text -> FilePath) -> (ByteString -> Text) -> ByteString -> FilePath forall b c a. (b -> c) -> (a -> b) -> a -> c . ByteString -> Text T.decodeUtf8) Maybe ByteString mp -- relative paths only! Bool -> m () -> m () forall (f :: * -> *). Applicative f => Bool -> f () -> f () when (Bool -> Bool not (Bool -> Bool) -> Bool -> Bool forall a b. (a -> b) -> a -> b $ FilePath -> Bool isRelative FilePath p) m () 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 Bool -> m () -> m () forall (f :: * -> *). Applicative f => Bool -> f () -> f () when (FilePath -> [FilePath] -> Bool forall (t :: * -> *) a. (Foldable t, Eq a) => a -> t a -> Bool elem ".." [FilePath] dirs) m () forall (m :: * -> *) a. MonadSnap m => m a pass FilePath -> m FilePath forall (m :: * -> *) a. Monad m => a -> m a return (FilePath -> m FilePath) -> FilePath -> m FilePath 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 = [(FilePath, ByteString)] -> MimeMap forall k v. (Eq k, Hashable k) => [(k, v)] -> HashMap k v 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" , ByteString -> ByteString -> ByteString S.append "application/vnd.openxmlformats-officedocument" ".wordprocessingml.document" ), ( ".dotx" , ByteString -> ByteString -> ByteString 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" , ByteString -> ByteString -> ByteString S.append "application/vnd.openxmlformats-officedocument" ".presentationml.template" ), ( ".ppsx" , ByteString -> ByteString -> ByteString S.append "application/vnd.openxmlformats-officedocument" ".presentationml.slideshow" ), ( ".ppt" , "application/vnd.ms-powerpoint" ), ( ".pptx" , ByteString -> ByteString -> ByteString S.append "application/vnd.openxmlformats-officedocument" ".presentationml.presentation" ), ( ".ps" , "application/postscript" ), ( ".qt" , "video/quicktime" ), ( ".rtf" , "text/rtf" ), ( ".sig" , "application/pgp-signature" ), ( ".sldx" , ByteString -> ByteString -> ByteString 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" ), ( ".tiff" , "image/tiff" ), ( ".tif" , "image/tiff" ), ( ".torrent" , "application/x-bittorrent" ), ( ".ts" , "video/mp2t" ), ( ".ttf" , "font/ttf" ), ( ".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" , ByteString -> ByteString -> ByteString S.append "application/vnd.openxmlformats-officedocument." "spreadsheetml.sheet" ), ( ".xltx" , ByteString -> ByteString -> ByteString S.append "application/vnd.openxmlformats-officedocument." "spreadsheetml.template" ), ( ".xml" , "text/xml" ), ( ".xpm" , "image/x-xpixmap" ), ( ".xwd" , "image/x-xwindowdump" ), ( ".zip" , "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) DirectoryConfig m -> [FilePath] indexFiles :: [FilePath], -- | Handler to generate a directory listing if there is no index. 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. DirectoryConfig m -> HandlerMap m dynamicHandlers :: HandlerMap m, -- | MIME type map to look up content types. 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. 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 "\n" [ "body { margin: 0px 0px 0px 0px; font-family: sans-serif }" , "div.header {" , "padding: 40px 40px 0px 40px; height:35px;" , "background:rgb(25,50,87);" , "background-image:-webkit-gradient(" , "linear,left bottom,left top," , "color-stop(0.00, rgb(31,62,108))," , "color-stop(1.00, rgb(19,38,66)));" , "background-image:-moz-linear-gradient(" , "center bottom,rgb(31,62,108) 0%,rgb(19,38,66) 100%);" , "text-shadow:-1px 3px 1px rgb(16,33,57);" , "font-size:16pt; letter-spacing: 2pt; color:white;" , "border-bottom:10px solid rgb(46,93,156) }" , "div.content {" , "background:rgb(255,255,255);" , "background-image:-webkit-gradient(" , "linear,left bottom, left top," , "color-stop(0.50, rgb(255,255,255))," , "color-stop(1.00, rgb(224,234,247)));" , "background-image:-moz-linear-gradient(" , "center bottom, white 50%, rgb(224,234,247) 100%);" , "padding: 40px 40px 40px 40px }" , "div.footer {" , "padding: 16px 0px 10px 10px; height:31px;" , "border-top: 1px solid rgb(194,209,225);" , "color: rgb(160,172,186); font-size:10pt;" , "background: rgb(245,249,255) }" , "table { max-width:100%; margin: 0 auto;" ByteString -> ByteString -> ByteString `S.append` " border-collapse: collapse; }" , "tr:hover { background:rgb(256,256,224) }" , "td { border:0; font-family:monospace; padding: 2px 0; }" , "td.filename, td.type { padding-right: 2em; }" , "th { border:0; background:rgb(28,56,97);" , "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 :: MimeMap -> ByteString -> FilePath -> m () defaultIndexGenerator mm :: MimeMap mm styles :: ByteString styles d :: FilePath d = do (Response -> Response) -> m () forall (m :: * -> *). MonadSnap m => (Response -> Response) -> m () modifyResponse ((Response -> Response) -> m ()) -> (Response -> Response) -> m () forall a b. (a -> b) -> a -> b $ ByteString -> Response -> Response setContentType "text/html; charset=utf-8" Request rq <- m Request 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 ByteString -> m () forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS "<!DOCTYPE html>\n<html>\n<head>" ByteString -> m () forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS "<title>Directory Listing: " ByteString -> m () forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString uri ByteString -> m () forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS "</title>" ByteString -> m () forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS "<style type='text/css'>" ByteString -> m () forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString styles ByteString -> m () forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS "</style></head><body>" ByteString -> m () forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS "<div class=\"header\">Directory Listing: " ByteString -> m () forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString uri ByteString -> m () forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS "</div><div class=\"content\">" ByteString -> m () forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS "<table><tr><th>File Name</th><th>Type</th><th>Last Modified" ByteString -> m () forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS "</th></tr>" Bool -> m () -> m () forall (f :: * -> *). Applicative f => Bool -> f () -> f () when (ByteString pInfo ByteString -> ByteString -> Bool forall a. Eq a => a -> a -> Bool /= "") (m () -> m ()) -> m () -> m () forall a b. (a -> b) -> a -> b $ ByteString -> m () forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS "<tr><td><a href='../'>..</a></td><td colspan=2>DIR</td></tr>" [FilePath] entries <- IO [FilePath] -> m [FilePath] forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO (IO [FilePath] -> m [FilePath]) -> IO [FilePath] -> m [FilePath] forall a b. (a -> b) -> a -> b $ FilePath -> IO [FilePath] getDirectoryContents FilePath d [FilePath] dirs <- IO [FilePath] -> m [FilePath] forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO (IO [FilePath] -> m [FilePath]) -> IO [FilePath] -> m [FilePath] forall a b. (a -> b) -> a -> b $ (FilePath -> IO Bool) -> [FilePath] -> IO [FilePath] forall (m :: * -> *) a. Applicative m => (a -> m Bool) -> [a] -> m [a] filterM (FilePath -> IO Bool doesDirectoryExist (FilePath -> IO Bool) -> (FilePath -> FilePath) -> FilePath -> IO Bool forall b c a. (b -> c) -> (a -> b) -> a -> c . (FilePath d FilePath -> FilePath -> FilePath </>)) [FilePath] entries [FilePath] files <- IO [FilePath] -> m [FilePath] forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO (IO [FilePath] -> m [FilePath]) -> IO [FilePath] -> m [FilePath] forall a b. (a -> b) -> a -> b $ (FilePath -> IO Bool) -> [FilePath] -> IO [FilePath] forall (m :: * -> *) a. Applicative m => (a -> m Bool) -> [a] -> m [a] filterM (FilePath -> IO Bool doesFileExist (FilePath -> IO Bool) -> (FilePath -> FilePath) -> FilePath -> IO Bool forall b c a. (b -> c) -> (a -> b) -> a -> c . (FilePath d FilePath -> FilePath -> FilePath </>)) [FilePath] entries [FilePath] -> (FilePath -> m ()) -> m () forall (t :: * -> *) (m :: * -> *) a b. (Foldable t, Monad m) => t a -> (a -> m b) -> m () forM_ ([FilePath] -> [FilePath] forall a. Ord a => [a] -> [a] sort ([FilePath] -> [FilePath]) -> [FilePath] -> [FilePath] forall a b. (a -> b) -> a -> b $ (FilePath -> Bool) -> [FilePath] -> [FilePath] forall a. (a -> Bool) -> [a] -> [a] filter (Bool -> Bool not (Bool -> Bool) -> (FilePath -> Bool) -> FilePath -> Bool forall b c a. (b -> c) -> (a -> b) -> a -> c . (FilePath -> [FilePath] -> Bool forall (t :: * -> *) a. (Foldable t, Eq a) => a -> t a -> Bool `elem` ["..", "."])) [FilePath] dirs) ((FilePath -> m ()) -> m ()) -> (FilePath -> m ()) -> m () forall a b. (a -> b) -> a -> b $ \f0 :: FilePath f0 -> do ByteString f <- IO ByteString -> m ByteString forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO (IO ByteString -> m ByteString) -> IO ByteString -> m ByteString forall a b. (a -> b) -> a -> b $ (Text -> ByteString) -> IO Text -> IO ByteString forall (m :: * -> *) a1 r. Monad m => (a1 -> r) -> m a1 -> m r liftM (\s :: Text s -> Text -> ByteString T.encodeUtf8 Text s ByteString -> ByteString -> ByteString forall a. Monoid a => a -> a -> a `mappend` "/") (IO Text -> IO ByteString) -> IO Text -> IO ByteString forall a b. (a -> b) -> a -> b $ FilePath -> IO Text decodeFilePath FilePath f0 ByteString -> m () forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS "<tr><td class='filename'><a href='" ByteString -> m () forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString f ByteString -> m () forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS "'>" ByteString -> m () forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString f ByteString -> m () forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS "</a></td><td class='type' colspan=2>DIR</td></tr>" [FilePath] -> (FilePath -> m ()) -> m () forall (t :: * -> *) (m :: * -> *) a b. (Foldable t, Monad m) => t a -> (a -> m b) -> m () forM_ ([FilePath] -> [FilePath] forall a. Ord a => [a] -> [a] sort [FilePath] files) ((FilePath -> m ()) -> m ()) -> (FilePath -> m ()) -> m () forall a b. (a -> b) -> a -> b $ \f0 :: FilePath f0 -> do ByteString f <- IO ByteString -> m ByteString forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO (IO ByteString -> m ByteString) -> IO ByteString -> m ByteString forall a b. (a -> b) -> a -> b $ (Text -> ByteString) -> IO Text -> IO ByteString forall (m :: * -> *) a1 r. Monad m => (a1 -> r) -> m a1 -> m r liftM Text -> ByteString T.encodeUtf8 (IO Text -> IO ByteString) -> IO Text -> IO ByteString forall a b. (a -> b) -> a -> b $ FilePath -> IO Text decodeFilePath FilePath f0 FileStatus stat <- IO FileStatus -> m FileStatus forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO (IO FileStatus -> m FileStatus) -> IO FileStatus -> m FileStatus forall a b. (a -> b) -> a -> b $ FilePath -> IO FileStatus getFileStatus (FilePath d FilePath -> FilePath -> FilePath </> FilePath f0) ByteString tm <- IO ByteString -> m ByteString forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO (IO ByteString -> m ByteString) -> IO ByteString -> m ByteString forall a b. (a -> b) -> a -> b $ CTime -> IO ByteString formatHttpTime (FileStatus -> CTime modificationTime FileStatus stat) ByteString -> m () forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS "<tr><td class='filename'><a href='" ByteString -> m () forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString f ByteString -> m () forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS "'>" ByteString -> m () forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString f ByteString -> m () forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS "</a></td><td class='type'>" ByteString -> m () forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS (MimeMap -> FilePath -> ByteString fileType MimeMap mm FilePath f0) ByteString -> m () forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS "</td><td>" ByteString -> m () forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS ByteString tm ByteString -> m () forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS "</tr>" ByteString -> m () forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS "</table></div><div class=\"footer\">Powered by " ByteString -> m () forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS "<b><a href=\"http://snapframework.com/\">Snap</a></b></div>" ByteString -> m () forall (m :: * -> *). MonadSnap m => ByteString -> m () writeBS "</body>" ------------------------------------------------------------------------------ decodeFilePath :: FilePath -> IO T.Text decodeFilePath :: FilePath -> IO Text decodeFilePath fp :: FilePath fp = do Text -> IO Text forall (m :: * -> *) a. MonadBase IO m => a -> m a evaluate (ByteString -> Text T.decodeUtf8 ByteString bs) IO Text -> (SomeException -> IO Text) -> IO Text forall (m :: * -> *) e a. (MonadBaseControl IO m, Exception e) => m a -> (e -> m a) -> m a `catch` (\(SomeException _::SomeException) -> Text -> IO Text 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 :: DirectoryConfig m simpleDirectoryConfig = DirectoryConfig :: forall (m :: * -> *). [FilePath] -> (FilePath -> m ()) -> HandlerMap m -> MimeMap -> (FilePath -> m ()) -> DirectoryConfig m DirectoryConfig { indexFiles :: [FilePath] indexFiles = [], indexGenerator :: FilePath -> m () indexGenerator = m () -> FilePath -> m () forall a b. a -> b -> a const m () forall (m :: * -> *) a. MonadSnap m => m a pass, dynamicHandlers :: HandlerMap m dynamicHandlers = HandlerMap m forall k v. HashMap k v Map.empty, mimeTypes :: MimeMap mimeTypes = MimeMap defaultMimeTypes, preServeHook :: FilePath -> m () preServeHook = m () -> FilePath -> m () forall a b. a -> b -> a const (m () -> FilePath -> m ()) -> m () -> FilePath -> m () forall a b. (a -> b) -> a -> b $ () -> m () forall (m :: * -> *) a. Monad m => a -> m a return (() -> m ()) -> () -> m () 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 :: DirectoryConfig m defaultDirectoryConfig = DirectoryConfig :: forall (m :: * -> *). [FilePath] -> (FilePath -> m ()) -> HandlerMap m -> MimeMap -> (FilePath -> m ()) -> DirectoryConfig m DirectoryConfig { indexFiles :: [FilePath] indexFiles = ["index.html", "index.htm"], indexGenerator :: FilePath -> m () indexGenerator = m () -> FilePath -> m () forall a b. a -> b -> a const m () forall (m :: * -> *) a. MonadSnap m => m a pass, dynamicHandlers :: HandlerMap m dynamicHandlers = HandlerMap m forall k v. HashMap k v Map.empty, mimeTypes :: MimeMap mimeTypes = MimeMap defaultMimeTypes, preServeHook :: FilePath -> m () preServeHook = m () -> FilePath -> m () forall a b. a -> b -> a const (m () -> FilePath -> m ()) -> m () -> FilePath -> m () forall a b. (a -> b) -> a -> b $ () -> m () forall (m :: * -> *) a. Monad m => a -> m a return (() -> m ()) -> () -> m () 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 :: DirectoryConfig m fancyDirectoryConfig = DirectoryConfig :: forall (m :: * -> *). [FilePath] -> (FilePath -> m ()) -> HandlerMap m -> MimeMap -> (FilePath -> m ()) -> DirectoryConfig m DirectoryConfig { indexFiles :: [FilePath] indexFiles = ["index.html", "index.htm"], indexGenerator :: FilePath -> m () indexGenerator = MimeMap -> ByteString -> FilePath -> m () forall (m :: * -> *). MonadSnap m => MimeMap -> ByteString -> FilePath -> m () defaultIndexGenerator MimeMap defaultMimeTypes ByteString snapIndexStyles, dynamicHandlers :: HandlerMap m dynamicHandlers = HandlerMap m forall k v. HashMap k v Map.empty, mimeTypes :: MimeMap mimeTypes = MimeMap defaultMimeTypes, preServeHook :: FilePath -> m () preServeHook = m () -> FilePath -> m () forall a b. a -> b -> a const (m () -> FilePath -> m ()) -> m () -> FilePath -> m () forall a b. (a -> b) -> a -> b $ () -> m () forall (m :: * -> *) a. Monad m => a -> m a return (() -> m ()) -> () -> m () 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 :: FilePath -> m () serveDirectory = DirectoryConfig m -> FilePath -> m () forall (m :: * -> *). MonadSnap m => DirectoryConfig m -> FilePath -> m () serveDirectoryWith DirectoryConfig m 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 :: DirectoryConfig m -> FilePath -> m () serveDirectoryWith cfg :: DirectoryConfig m cfg base :: FilePath base = do Bool b <- m Bool directory m Bool -> m Bool -> m Bool forall (f :: * -> *) a. Alternative f => f a -> f a -> f a <|> m Bool file m Bool -> m Bool -> m Bool forall (f :: * -> *) a. Alternative f => f a -> f a -> f a <|> m Bool forall b. m b redir Bool -> m () -> m () forall (f :: * -> *). Applicative f => Bool -> f () -> f () when (Bool -> Bool not Bool b) m () forall (m :: * -> *) a. MonadSnap m => m a pass where idxs :: [FilePath] idxs = DirectoryConfig m -> [FilePath] forall (m :: * -> *). DirectoryConfig m -> [FilePath] indexFiles DirectoryConfig m cfg generate :: FilePath -> m () generate = DirectoryConfig m -> FilePath -> m () forall (m :: * -> *). DirectoryConfig m -> FilePath -> m () indexGenerator DirectoryConfig m cfg mimes :: MimeMap mimes = DirectoryConfig m -> MimeMap forall (m :: * -> *). DirectoryConfig m -> MimeMap mimeTypes DirectoryConfig m cfg dyns :: HandlerMap m dyns = DirectoryConfig m -> HandlerMap m forall (m :: * -> *). DirectoryConfig m -> HandlerMap m dynamicHandlers DirectoryConfig m cfg pshook :: FilePath -> m () pshook = DirectoryConfig m -> FilePath -> m () forall (m :: * -> *). DirectoryConfig m -> FilePath -> m () preServeHook DirectoryConfig m cfg -- Serves a file if it exists; passes if not serve :: FilePath -> m Bool serve f :: FilePath f = do IO Bool -> m Bool forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO (FilePath -> IO Bool doesFileExist FilePath f) m Bool -> (Bool -> m ()) -> m () forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b >>= (Bool -> m () -> m ()) -> m () -> Bool -> m () forall a b c. (a -> b -> c) -> b -> a -> c flip Bool -> m () -> m () forall (f :: * -> *). Applicative f => Bool -> f () -> f () unless m () forall (m :: * -> *) a. MonadSnap m => m a pass let fname :: FilePath fname = FilePath -> FilePath takeFileName FilePath f let staticServe :: FilePath -> m () staticServe f' :: FilePath f' = FilePath -> m () pshook FilePath f m () -> m () -> m () forall (m :: * -> *) a b. Monad m => m a -> m b -> m b >> ByteString -> FilePath -> m () forall (m :: * -> *). MonadSnap m => ByteString -> FilePath -> m () serveFileAs (MimeMap -> FilePath -> ByteString fileType MimeMap mimes FilePath fname) FilePath f' (FilePath -> m ()) -> HandlerMap m -> FilePath -> FilePath -> m () forall a. a -> HashMap FilePath a -> FilePath -> a lookupExt FilePath -> m () staticServe HandlerMap m dyns FilePath fname FilePath f m () -> m Bool -> m Bool forall (m :: * -> *) a b. Monad m => m a -> m b -> m b >> Bool -> m Bool 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 <- m Request forall (m :: * -> *). MonadSnap m => m Request getRequest let uri :: ByteString uri = Request -> ByteString uriWithoutQueryString Request rq Bool -> m () -> m () forall (f :: * -> *). Applicative f => Bool -> f () -> f () unless ("/" ByteString -> ByteString -> Bool `S.isSuffixOf` ByteString uri) m () forall (m :: * -> *) a. MonadSnap m => m a pass FilePath rel <- (FilePath base FilePath -> FilePath -> FilePath </>) (FilePath -> FilePath) -> m FilePath -> m FilePath forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b <$> m FilePath forall (m :: * -> *). MonadSnap m => m FilePath getSafePath Bool b <- IO Bool -> m Bool forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO (IO Bool -> m Bool) -> IO Bool -> m Bool forall a b. (a -> b) -> a -> b $ FilePath -> IO Bool doesDirectoryExist FilePath rel if Bool b then do let serveRel :: FilePath -> m Bool serveRel f :: FilePath f = FilePath -> m Bool serve (FilePath rel FilePath -> FilePath -> FilePath </> FilePath f) (m Bool -> m Bool -> m Bool) -> m Bool -> [m Bool] -> m Bool forall (t :: * -> *) b a. Foldable t => (b -> a -> b) -> b -> t a -> b foldl' m Bool -> m Bool -> m Bool forall (f :: * -> *) a. Alternative f => f a -> f a -> f a (<|>) m Bool forall (m :: * -> *) a. MonadSnap m => m a pass ((FilePath -> m Bool) -> [FilePath] -> [m Bool] forall a b. (a -> b) -> [a] -> [b] Prelude.map FilePath -> m Bool serveRel [FilePath] idxs) m Bool -> m Bool -> m Bool forall (f :: * -> *) a. Alternative f => f a -> f a -> f a <|> (FilePath -> m () generate FilePath rel m () -> m Bool -> m Bool forall (m :: * -> *) a b. Monad m => m a -> m b -> m b >> Bool -> m Bool forall (m :: * -> *) a. Monad m => a -> m a return Bool True) m Bool -> m Bool -> m Bool forall (f :: * -> *) a. Alternative f => f a -> f a -> f a <|> Bool -> m Bool forall (m :: * -> *) a. Monad m => a -> m a return Bool False else Bool -> m Bool 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 (FilePath -> m Bool) -> m FilePath -> m Bool forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b =<< ((FilePath base FilePath -> FilePath -> FilePath </>) (FilePath -> FilePath) -> m FilePath -> m FilePath forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b <$> m FilePath 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 </>) (FilePath -> FilePath) -> m FilePath -> m FilePath forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b <$> m FilePath forall (m :: * -> *). MonadSnap m => m FilePath getSafePath IO Bool -> m Bool forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO (FilePath -> IO Bool doesDirectoryExist FilePath rel) m Bool -> (Bool -> m ()) -> m () forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b >>= (Bool -> m () -> m ()) -> m () -> Bool -> m () forall a b c. (a -> b -> c) -> b -> a -> c flip Bool -> m () -> m () forall (f :: * -> *). Applicative f => Bool -> f () -> f () unless m () forall (m :: * -> *) a. MonadSnap m => m a pass Request rq <- m Request 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 qss] ByteString -> m b 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 :: FilePath -> m () serveFile fp :: FilePath fp = ByteString -> FilePath -> m () 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 :: ByteString -> FilePath -> m () serveFileAs mime :: ByteString mime fp :: FilePath fp = do Request reqOrig <- m Request 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 Maybe ByteString -> Bool forall a. Maybe a -> Bool isNothing (Maybe ByteString -> Bool) -> Maybe ByteString -> Bool forall a b. (a -> b) -> a -> b $ CI ByteString -> Request -> Maybe ByteString forall a. HasHeaders a => CI ByteString -> a -> Maybe ByteString getHeader "range" Request reqOrig then CI ByteString -> Request -> Request forall a. HasHeaders a => CI ByteString -> a -> a deleteHeader "if-range" Request reqOrig else Request reqOrig -- check "If-Modified-Since" and "If-Range" headers let mbH :: Maybe ByteString mbH = CI ByteString -> Request -> Maybe ByteString forall a. HasHeaders a => CI ByteString -> a -> Maybe ByteString getHeader "if-modified-since" Request req Maybe CTime mbIfModified <- IO (Maybe CTime) -> m (Maybe CTime) forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO (IO (Maybe CTime) -> m (Maybe CTime)) -> IO (Maybe CTime) -> m (Maybe CTime) forall a b. (a -> b) -> a -> b $ case Maybe ByteString mbH of Nothing -> Maybe CTime -> IO (Maybe CTime) forall (m :: * -> *) a. Monad m => a -> m a return Maybe CTime forall a. Maybe a Nothing (Just s :: ByteString s) -> (CTime -> Maybe CTime) -> IO CTime -> IO (Maybe CTime) forall (m :: * -> *) a1 r. Monad m => (a1 -> r) -> m a1 -> m r liftM CTime -> Maybe CTime forall a. a -> Maybe a Just (IO CTime -> IO (Maybe CTime)) -> IO CTime -> IO (Maybe CTime) 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 <- IO (Maybe CTime) -> m (Maybe CTime) forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO (IO (Maybe CTime) -> m (Maybe CTime)) -> IO (Maybe CTime) -> m (Maybe CTime) forall a b. (a -> b) -> a -> b $ case CI ByteString -> Request -> Maybe ByteString forall a. HasHeaders a => CI ByteString -> a -> Maybe ByteString getHeader "if-range" Request req of Nothing -> Maybe CTime -> IO (Maybe CTime) forall (m :: * -> *) a. Monad m => a -> m a return Maybe CTime forall a. Maybe a Nothing (Just s :: ByteString s) -> (CTime -> Maybe CTime) -> IO CTime -> IO (Maybe CTime) forall (m :: * -> *) a1 r. Monad m => (a1 -> r) -> m a1 -> m r liftM CTime -> Maybe CTime forall a. a -> Maybe a Just (IO CTime -> IO (Maybe CTime)) -> IO CTime -> IO (Maybe CTime) forall a b. (a -> b) -> a -> b $ ByteString -> IO CTime parseHttpTime ByteString s FilePath -> m () forall (m :: * -> *). MonadIO m => FilePath -> m () dbg (FilePath -> m ()) -> FilePath -> m () forall a b. (a -> b) -> a -> b $ "mbIfModified: " FilePath -> FilePath -> FilePath forall a. [a] -> [a] -> [a] ++ Maybe CTime -> FilePath forall a. Show a => a -> FilePath Prelude.show Maybe CTime mbIfModified FilePath -> m () forall (m :: * -> *). MonadIO m => FilePath -> m () dbg (FilePath -> m ()) -> FilePath -> m () forall a b. (a -> b) -> a -> b $ "mbIfRange: " FilePath -> FilePath -> FilePath forall a. [a] -> [a] -> [a] ++ Maybe CTime -> FilePath 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 <- IO FileStatus -> m FileStatus forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO (IO FileStatus -> m FileStatus) -> IO FileStatus -> m FileStatus forall a b. (a -> b) -> a -> b $ FilePath -> IO FileStatus getFileStatus FilePath fp let mt :: CTime mt = FileStatus -> CTime modificationTime FileStatus filestat m () -> (CTime -> m ()) -> Maybe CTime -> m () forall b a. b -> (a -> b) -> Maybe a -> b maybe (() -> m () forall (m :: * -> *) a. Monad m => a -> m a return (() -> m ()) -> () -> m () forall a b. (a -> b) -> a -> b $! ()) (\lt :: CTime lt -> Bool -> m () -> m () forall (f :: * -> *). Applicative f => Bool -> f () -> f () when (CTime mt CTime -> CTime -> Bool forall a. Ord a => a -> a -> Bool <= CTime lt) m () forall a. m a notModified) Maybe CTime mbIfModified let sz :: Word64 sz = FileOffset -> Word64 forall a b. (Integral a, Num b) => a -> b fromIntegral (FileOffset -> Word64) -> FileOffset -> Word64 forall a b. (a -> b) -> a -> b $ FileStatus -> FileOffset fileSize FileStatus filestat ByteString lm <- IO ByteString -> m ByteString forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO (IO ByteString -> m ByteString) -> IO ByteString -> m ByteString 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. (Response -> Response) -> m () forall (m :: * -> *). MonadSnap m => (Response -> Response) -> m () modifyResponse ((Response -> Response) -> m ()) -> (Response -> Response) -> m () forall a b. (a -> b) -> a -> b $ CI ByteString -> ByteString -> Response -> Response forall a. HasHeaders a => CI ByteString -> ByteString -> a -> a setHeader "Last-Modified" ByteString lm (Response -> Response) -> (Response -> Response) -> Response -> Response forall b c a. (b -> c) -> (a -> b) -> a -> c . CI ByteString -> ByteString -> Response -> Response forall a. HasHeaders a => CI ByteString -> ByteString -> a -> a setHeader "Accept-Ranges" "bytes" (Response -> Response) -> (Response -> Response) -> Response -> Response 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 = Bool -> (CTime -> Bool) -> Maybe CTime -> Bool forall b a. b -> (a -> b) -> Maybe a -> b maybe (Bool False) (\lt :: CTime lt -> CTime mt CTime -> CTime -> Bool 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 Bool -> m Bool forall (m :: * -> *) a. Monad m => a -> m a return Bool False else Snap Bool -> m Bool forall (m :: * -> *) a. MonadSnap m => Snap a -> m a liftSnap (Snap Bool -> m Bool) -> Snap Bool -> m Bool forall a b. (a -> b) -> a -> b $ Request -> FilePath -> Word64 -> Snap Bool forall (m :: * -> *). MonadSnap m => Request -> FilePath -> Word64 -> m Bool checkRangeReq Request req FilePath fp Word64 sz FilePath -> m () forall (m :: * -> *). MonadIO m => FilePath -> m () dbg (FilePath -> m ()) -> FilePath -> m () forall a b. (a -> b) -> a -> b $ "was this a range request? " FilePath -> FilePath -> FilePath forall a. [a] -> [a] -> [a] ++ Bool -> FilePath forall a. Show a => a -> FilePath Prelude.show Bool wasRange -- if we didn't have a range request, we just do normal sendfile Bool -> m () -> m () forall (f :: * -> *). Applicative f => Bool -> f () -> f () unless Bool wasRange (m () -> m ()) -> m () -> m () forall a b. (a -> b) -> a -> b $ do (Response -> Response) -> m () forall (m :: * -> *). MonadSnap m => (Response -> Response) -> m () modifyResponse ((Response -> Response) -> m ()) -> (Response -> Response) -> m () forall a b. (a -> b) -> a -> b $ Int -> Response -> Response setResponseCode 200 (Response -> Response) -> (Response -> Response) -> Response -> Response forall b c a. (b -> c) -> (a -> b) -> a -> c . Word64 -> Response -> Response setContentLength Word64 sz Snap () -> m () forall (m :: * -> *) a. MonadSnap m => Snap a -> m a liftSnap (Snap () -> m ()) -> Snap () -> m () forall a b. (a -> b) -> a -> b $ FilePath -> Snap () forall (m :: * -> *). MonadSnap m => FilePath -> m () sendFile FilePath fp where -------------------------------------------------------------------------- notModified :: m a notModified = Response -> m a forall (m :: * -> *) a. MonadSnap m => Response -> m a finishWith (Response -> m a) -> Response -> m a forall a b. (a -> b) -> a -> b $ Int -> Response -> Response setResponseCode 304 Response emptyResponse ------------------------------------------------------------------------------ lookupExt :: a -> HashMap FilePath a -> FilePath -> a lookupExt :: a -> HashMap FilePath a -> FilePath -> a lookupExt def :: a def m :: HashMap FilePath a m f :: FilePath f = if FilePath -> Bool forall (t :: * -> *) a. Foldable t => t a -> Bool null FilePath ext then a def else a -> Maybe a -> a forall a. a -> Maybe a -> a fromMaybe (a -> HashMap FilePath a -> FilePath -> a 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 = (Char -> Bool) -> FilePath -> FilePath forall a. (a -> Bool) -> [a] -> [a] dropWhile (Char -> Char -> Bool forall a. Eq a => a -> a -> Bool /= '.') (FilePath -> FilePath) -> (FilePath -> FilePath) -> FilePath -> FilePath forall b c a. (b -> c) -> (a -> b) -> a -> c . Int -> FilePath -> FilePath forall a. Int -> [a] -> [a] drop 1 ext :: FilePath ext = FilePath -> FilePath takeExtensions FilePath f mbe :: Maybe a mbe = FilePath -> HashMap FilePath a -> Maybe a 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 = ByteString -> MimeMap -> FilePath -> ByteString forall a. a -> HashMap FilePath a -> FilePath -> a lookupExt ByteString defaultMimeType ------------------------------------------------------------------------------ defaultMimeType :: ByteString defaultMimeType :: ByteString defaultMimeType = "application/octet-stream" ------------------------------------------------------------------------------ data RangeReq = RangeReq !Word64 !(Maybe Word64) | SuffixRangeReq !Word64 ------------------------------------------------------------------------------ rangeParser :: Parser RangeReq rangeParser :: Parser RangeReq rangeParser = ByteString -> Parser ByteString string "bytes=" Parser ByteString -> Parser RangeReq -> Parser RangeReq forall (f :: * -> *) a b. Applicative f => f a -> f b -> f b *> (Parser RangeReq byteRangeSpec Parser RangeReq -> Parser RangeReq -> Parser RangeReq forall (f :: * -> *) a. Alternative f => f a -> f a -> f a <|> Parser RangeReq suffixByteRangeSpec) Parser RangeReq -> Parser ByteString () -> Parser RangeReq forall (f :: * -> *) a b. Applicative f => f a -> f b -> f a <* Parser ByteString () forall t. Chunk t => Parser t () endOfInput where byteRangeSpec :: Parser RangeReq byteRangeSpec = do Word64 start <- Int64 -> Word64 forall a b. (Integral a, Num b) => a -> b fromIntegral (Int64 -> Word64) -> Parser ByteString Int64 -> Parser ByteString Word64 forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b <$> Parser ByteString Int64 parseNum Parser ByteString Char -> Parser ByteString () forall (f :: * -> *) a. Functor f => f a -> f () void (Parser ByteString Char -> Parser ByteString ()) -> Parser ByteString Char -> Parser ByteString () forall a b. (a -> b) -> a -> b $! Char -> Parser ByteString Char char '-' Maybe Int64 end <- Maybe Int64 -> Parser ByteString (Maybe Int64) -> Parser ByteString (Maybe Int64) forall (f :: * -> *) a. Alternative f => a -> f a -> f a option Maybe Int64 forall a. Maybe a Nothing (Parser ByteString (Maybe Int64) -> Parser ByteString (Maybe Int64)) -> Parser ByteString (Maybe Int64) -> Parser ByteString (Maybe Int64) forall a b. (a -> b) -> a -> b $ (Int64 -> Maybe Int64) -> Parser ByteString Int64 -> Parser ByteString (Maybe Int64) forall (m :: * -> *) a1 r. Monad m => (a1 -> r) -> m a1 -> m r liftM Int64 -> Maybe Int64 forall a. a -> Maybe a Just Parser ByteString Int64 parseNum RangeReq -> Parser RangeReq forall (m :: * -> *) a. Monad m => a -> m a return (RangeReq -> Parser RangeReq) -> RangeReq -> Parser RangeReq forall a b. (a -> b) -> a -> b $! Word64 -> Maybe Word64 -> RangeReq RangeReq Word64 start (Int64 -> Word64 forall a b. (Integral a, Num b) => a -> b fromIntegral (Int64 -> Word64) -> Maybe Int64 -> Maybe Word64 forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b <$> Maybe Int64 end) suffixByteRangeSpec :: Parser RangeReq suffixByteRangeSpec = (Int64 -> RangeReq) -> Parser ByteString Int64 -> Parser RangeReq forall (m :: * -> *) a1 r. Monad m => (a1 -> r) -> m a1 -> m r liftM (Word64 -> RangeReq SuffixRangeReq (Word64 -> RangeReq) -> (Int64 -> Word64) -> Int64 -> RangeReq forall b c a. (b -> c) -> (a -> b) -> a -> c . Int64 -> Word64 forall a b. (Integral a, Num b) => a -> b fromIntegral) (Parser ByteString Int64 -> Parser RangeReq) -> Parser ByteString Int64 -> Parser RangeReq forall a b. (a -> b) -> a -> b $ Char -> Parser ByteString Char char '-' Parser ByteString Char -> Parser ByteString Int64 -> Parser ByteString Int64 forall (f :: * -> *) a b. Applicative f => f a -> f b -> f b *> Parser ByteString Int64 parseNum ------------------------------------------------------------------------------ checkRangeReq :: (MonadSnap m) => Request -> FilePath -> Word64 -> m Bool checkRangeReq :: Request -> FilePath -> Word64 -> m Bool checkRangeReq req :: Request req fp :: FilePath fp sz :: Word64 sz = do -- TODO/FIXME: multiple ranges m Bool -> (ByteString -> m Bool) -> Maybe ByteString -> m Bool forall b a. b -> (a -> b) -> Maybe a -> b maybe (Bool -> m Bool forall (m :: * -> *) a. Monad m => a -> m a return Bool False) (\s :: ByteString s -> (FilePath -> m Bool) -> (RangeReq -> m Bool) -> Either FilePath RangeReq -> m Bool forall a c b. (a -> c) -> (b -> c) -> Either a b -> c either (m Bool -> FilePath -> m Bool forall a b. a -> b -> a const (m Bool -> FilePath -> m Bool) -> m Bool -> FilePath -> m Bool forall a b. (a -> b) -> a -> b $ Bool -> m Bool forall (m :: * -> *) a. Monad m => a -> m a return Bool False) RangeReq -> m Bool withRange (ByteString -> Parser RangeReq -> Either FilePath RangeReq forall a. ByteString -> Parser a -> Either FilePath a fullyParse ByteString s Parser RangeReq rangeParser)) (CI ByteString -> Request -> Maybe ByteString forall a. HasHeaders a => CI ByteString -> a -> Maybe ByteString getHeader "range" Request req) where withRange :: RangeReq -> m Bool withRange (RangeReq start :: Word64 start mend :: Maybe Word64 mend) = do let end :: Word64 end = Word64 -> Maybe Word64 -> Word64 forall a. a -> Maybe a -> a fromMaybe (Word64 szWord64 -> Word64 -> Word64 forall a. Num a => a -> a -> a -1) Maybe Word64 mend FilePath -> m () forall (m :: * -> *). MonadIO m => FilePath -> m () dbg (FilePath -> m ()) -> FilePath -> m () forall a b. (a -> b) -> a -> b $ "withRange: start=" FilePath -> FilePath -> FilePath forall a. [a] -> [a] -> [a] ++ Word64 -> FilePath forall a. Show a => a -> FilePath Prelude.show Word64 start FilePath -> FilePath -> FilePath forall a. [a] -> [a] -> [a] ++ ", end=" FilePath -> FilePath -> FilePath forall a. [a] -> [a] -> [a] ++ Word64 -> FilePath forall a. Show a => a -> FilePath Prelude.show Word64 end if Word64 start Word64 -> Word64 -> Bool forall a. Ord a => a -> a -> Bool < 0 Bool -> Bool -> Bool || Word64 end Word64 -> Word64 -> Bool forall a. Ord a => a -> a -> Bool < Word64 start Bool -> Bool -> Bool || Word64 start Word64 -> Word64 -> Bool forall a. Ord a => a -> a -> Bool >= Word64 sz Bool -> Bool -> Bool || Word64 end Word64 -> Word64 -> Bool forall a. Ord a => a -> a -> Bool >= Word64 sz then m Bool send416 else Word64 -> Word64 -> m Bool forall (m :: * -> *). MonadSnap m => Word64 -> Word64 -> m Bool send206 Word64 start Word64 end withRange (SuffixRangeReq nbytes :: Word64 nbytes) = do let end :: Word64 end = Word64 szWord64 -> Word64 -> Word64 forall a. Num a => a -> a -> a -1 let start :: Word64 start = Word64 sz Word64 -> Word64 -> Word64 forall a. Num a => a -> a -> a - Word64 nbytes FilePath -> m () forall (m :: * -> *). MonadIO m => FilePath -> m () dbg (FilePath -> m ()) -> FilePath -> m () forall a b. (a -> b) -> a -> b $ "withRange: start=" FilePath -> FilePath -> FilePath forall a. [a] -> [a] -> [a] ++ Word64 -> FilePath forall a. Show a => a -> FilePath Prelude.show Word64 start FilePath -> FilePath -> FilePath forall a. [a] -> [a] -> [a] ++ ", end=" FilePath -> FilePath -> FilePath forall a. [a] -> [a] -> [a] ++ Word64 -> FilePath forall a. Show a => a -> FilePath Prelude.show Word64 end if Word64 start Word64 -> Word64 -> Bool forall a. Ord a => a -> a -> Bool < 0 Bool -> Bool -> Bool || Word64 end Word64 -> Word64 -> Bool forall a. Ord a => a -> a -> Bool < Word64 start Bool -> Bool -> Bool || Word64 start Word64 -> Word64 -> Bool forall a. Ord a => a -> a -> Bool >= Word64 sz Bool -> Bool -> Bool || Word64 end Word64 -> Word64 -> Bool forall a. Ord a => a -> a -> Bool >= Word64 sz then m Bool send416 else Word64 -> Word64 -> m Bool 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 start :: Word64 start end :: Word64 end = do FilePath -> m () forall (m :: * -> *). MonadIO m => FilePath -> m () dbg "inside send206" let !len :: Word64 len = Word64 endWord64 -> Word64 -> Word64 forall a. Num a => a -> a -> a -Word64 startWord64 -> Word64 -> Word64 forall a. Num a => a -> a -> a +1 let crng :: ByteString crng = [ByteString] -> ByteString S.concat ([ByteString] -> ByteString) -> (ByteString -> [ByteString]) -> ByteString -> ByteString forall b c a. (b -> c) -> (a -> b) -> a -> c . ByteString -> [ByteString] L.toChunks (ByteString -> ByteString) -> ByteString -> ByteString forall a b. (a -> b) -> a -> b $ Builder -> ByteString toLazyByteString (Builder -> ByteString) -> Builder -> ByteString forall a b. (a -> b) -> a -> b $ [Builder] -> Builder forall a. Monoid a => [a] -> a mconcat [ ByteString -> Builder byteString "bytes " , Word64 -> Builder forall a. Show a => a -> Builder fromShow Word64 start , Char -> Builder char8 '-' , Word64 -> Builder forall a. Show a => a -> Builder fromShow Word64 end , Char -> Builder char8 '/' , Word64 -> Builder forall a. Show a => a -> Builder fromShow Word64 sz ] (Response -> Response) -> m () forall (m :: * -> *). MonadSnap m => (Response -> Response) -> m () modifyResponse ((Response -> Response) -> m ()) -> (Response -> Response) -> m () forall a b. (a -> b) -> a -> b $ Int -> Response -> Response setResponseCode 206 (Response -> Response) -> (Response -> Response) -> Response -> Response forall b c a. (b -> c) -> (a -> b) -> a -> c . CI ByteString -> ByteString -> Response -> Response forall a. HasHeaders a => CI ByteString -> ByteString -> a -> a setHeader "Content-Range" ByteString crng (Response -> Response) -> (Response -> Response) -> Response -> Response forall b c a. (b -> c) -> (a -> b) -> a -> c . Word64 -> Response -> Response setContentLength Word64 len FilePath -> m () forall (m :: * -> *). MonadIO m => FilePath -> m () dbg (FilePath -> m ()) -> FilePath -> m () forall a b. (a -> b) -> a -> b $ "send206: sending range (" FilePath -> FilePath -> FilePath forall a. [a] -> [a] -> [a] ++ Word64 -> FilePath forall a. Show a => a -> FilePath Prelude.show Word64 start FilePath -> FilePath -> FilePath forall a. [a] -> [a] -> [a] ++ "," FilePath -> FilePath -> FilePath forall a. [a] -> [a] -> [a] ++ Word64 -> FilePath forall a. Show a => a -> FilePath Prelude.show (Word64 endWord64 -> Word64 -> Word64 forall a. Num a => a -> a -> a +1) FilePath -> FilePath -> FilePath forall a. [a] -> [a] -> [a] ++ ") to sendFilePartial" -- end here was inclusive, sendFilePartial is exclusive FilePath -> (Word64, Word64) -> m () forall (m :: * -> *). MonadSnap m => FilePath -> (Word64, Word64) -> m () sendFilePartial FilePath fp (Word64 start,Word64 endWord64 -> Word64 -> Word64 forall a. Num a => a -> a -> a +1) Bool -> m Bool forall (m :: * -> *) a. Monad m => a -> m a return Bool True send416 :: m Bool send416 = do FilePath -> m () forall (m :: * -> *). MonadIO m => FilePath -> m () dbg "inside send416" -- if there's an "If-Range" header in the request, then we just send -- back 200 if CI ByteString -> Request -> Maybe ByteString forall a. HasHeaders a => CI ByteString -> a -> Maybe ByteString getHeader "If-Range" Request req Maybe ByteString -> Maybe ByteString -> Bool forall a. Eq a => a -> a -> Bool /= Maybe ByteString forall a. Maybe a Nothing then Bool -> m Bool forall (m :: * -> *) a. Monad m => a -> m a return Bool False else do let crng :: ByteString crng = [ByteString] -> ByteString S.concat ([ByteString] -> ByteString) -> (ByteString -> [ByteString]) -> ByteString -> ByteString forall b c a. (b -> c) -> (a -> b) -> a -> c . ByteString -> [ByteString] L.toChunks (ByteString -> ByteString) -> ByteString -> ByteString forall a b. (a -> b) -> a -> b $ Builder -> ByteString toLazyByteString (Builder -> ByteString) -> Builder -> ByteString forall a b. (a -> b) -> a -> b $ [Builder] -> Builder forall a. Monoid a => [a] -> a mconcat [ ByteString -> Builder byteString "bytes */" , Word64 -> Builder forall a. Show a => a -> Builder fromShow Word64 sz ] (Response -> Response) -> m () forall (m :: * -> *). MonadSnap m => (Response -> Response) -> m () modifyResponse ((Response -> Response) -> m ()) -> (Response -> Response) -> m () forall a b. (a -> b) -> a -> b $ Int -> Response -> Response setResponseCode 416 (Response -> Response) -> (Response -> Response) -> Response -> Response forall b c a. (b -> c) -> (a -> b) -> a -> c . CI ByteString -> ByteString -> Response -> Response forall a. HasHeaders a => CI ByteString -> ByteString -> a -> a setHeader "Content-Range" ByteString crng (Response -> Response) -> (Response -> Response) -> Response -> Response forall b c a. (b -> c) -> (a -> b) -> a -> c . Word64 -> Response -> Response setContentLength 0 (Response -> Response) -> (Response -> Response) -> Response -> Response forall b c a. (b -> c) -> (a -> b) -> a -> c . CI ByteString -> Response -> Response forall a. HasHeaders a => CI ByteString -> a -> a deleteHeader "Content-Type" (Response -> Response) -> (Response -> Response) -> Response -> Response forall b c a. (b -> c) -> (a -> b) -> a -> c . CI ByteString -> Response -> Response forall a. HasHeaders a => CI ByteString -> a -> a deleteHeader "Content-Encoding" (Response -> Response) -> (Response -> Response) -> Response -> Response forall b c a. (b -> c) -> (a -> b) -> a -> c . CI ByteString -> Response -> Response forall a. HasHeaders a => CI ByteString -> a -> a deleteHeader "Transfer-Encoding" (Response -> Response) -> (Response -> Response) -> Response -> Response forall b c a. (b -> c) -> (a -> b) -> a -> c . (OutputStream Builder -> IO (OutputStream Builder)) -> Response -> Response setResponseBody (OutputStream Builder -> IO (OutputStream Builder) forall (m :: * -> *) a. Monad m => a -> m a return (OutputStream Builder -> IO (OutputStream Builder)) -> (OutputStream Builder -> OutputStream Builder) -> OutputStream Builder -> IO (OutputStream Builder) forall b c a. (b -> c) -> (a -> b) -> a -> c . OutputStream Builder -> OutputStream Builder forall a. a -> a id) Bool -> m Bool forall (m :: * -> *) a. Monad m => a -> m a return Bool True ------------------------------------------------------------------------------ dbg :: (MonadIO m) => String -> m () dbg :: FilePath -> m () dbg s :: FilePath s = FilePath -> m () forall (m :: * -> *). MonadIO m => FilePath -> m () debug (FilePath -> m ()) -> FilePath -> m () forall a b. (a -> b) -> a -> b $ "FileServe:" FilePath -> FilePath -> FilePath forall a. [a] -> [a] -> [a] ++ FilePath s ------------------------------------------------------------------------------ uriWithoutQueryString :: Request -> ByteString uriWithoutQueryString :: Request -> ByteString uriWithoutQueryString rq :: Request rq = (Char -> Bool) -> ByteString -> ByteString S.takeWhile (Char -> Char -> Bool forall a. Eq a => a -> a -> Bool /= '?') ByteString uri where uri :: ByteString uri = Request -> ByteString rqURI Request rq ------------------------------------------------------------------------------ queryStringSuffix :: Request -> ByteString queryStringSuffix :: Request -> ByteString queryStringSuffix rq :: 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 "" else "?" ------------------------------------------------------------------------------ fromShow :: Show a => a -> Builder fromShow :: a -> Builder fromShow = FilePath -> Builder stringUtf8 (FilePath -> Builder) -> (a -> FilePath) -> a -> Builder forall b c a. (b -> c) -> (a -> b) -> a -> c . a -> FilePath forall a. Show a => a -> FilePath show