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