{-# 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