-
Notifications
You must be signed in to change notification settings - Fork 134
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Browse files
Browse the repository at this point in the history
* add test case * add Web.Scotty.Body module * test case fixed but other test fails :D * move request max size test case under ScottyT and cleanup imports * add counterexample test to setMaxRequestBodySize --------- Co-authored-by: Marco Zocca <[email protected]>
- Loading branch information
Showing
9 changed files
with
214 additions
and
129 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,101 @@ | ||
{-# LANGUAGE FlexibleContexts, FlexibleInstances, RecordWildCards, | ||
OverloadedStrings, MultiWayIf #-} | ||
module Web.Scotty.Body ( | ||
newBodyInfo, | ||
cloneBodyInfo | ||
|
||
, getFormParamsAndFilesAction | ||
, getBodyAction | ||
, getBodyChunkAction | ||
) where | ||
|
||
import Control.Concurrent.MVar | ||
import Control.Monad.IO.Class | ||
import qualified Data.ByteString as BS | ||
import qualified Data.ByteString.Char8 as B | ||
import qualified Data.ByteString.Lazy.Char8 as BL | ||
import Data.Maybe | ||
import GHC.Exception | ||
import Network.Wai (Request(..), getRequestBodyChunk) | ||
import qualified Network.Wai.Parse as W (File, Param, getRequestBodyType, BackEnd, lbsBackEnd, sinkRequestBody) | ||
import Web.Scotty.Action | ||
import Web.Scotty.Internal.Types (BodyInfo(..), BodyChunkBuffer(..), BodyPartiallyStreamed(..), RouteOptions(..)) | ||
import Web.Scotty.Util | ||
|
||
-- | Make a new BodyInfo with readProgress at 0 and an empty BodyChunkBuffer. | ||
newBodyInfo :: (MonadIO m) => Request -> m BodyInfo | ||
newBodyInfo req = liftIO $ do | ||
readProgress <- newMVar 0 | ||
chunkBuffer <- newMVar (BodyChunkBuffer False []) | ||
return $ BodyInfo readProgress chunkBuffer (getRequestBodyChunk req) | ||
|
||
-- | Make a copy of a BodyInfo, sharing the previous BodyChunkBuffer but with the | ||
-- readProgress MVar reset to 0. | ||
cloneBodyInfo :: (MonadIO m) => BodyInfo -> m BodyInfo | ||
cloneBodyInfo (BodyInfo _ chunkBufferVar getChunk) = liftIO $ do | ||
cleanReadProgressVar <- newMVar 0 | ||
return $ BodyInfo cleanReadProgressVar chunkBufferVar getChunk | ||
|
||
-- | Get the form params and files from the request. Requires reading the whole body. | ||
getFormParamsAndFilesAction :: Request -> BodyInfo -> RouteOptions -> IO ([Param], [W.File BL.ByteString]) | ||
getFormParamsAndFilesAction req bodyInfo opts = do | ||
let shouldParseBody = isJust $ W.getRequestBodyType req | ||
|
||
if shouldParseBody | ||
then | ||
do | ||
bs <- getBodyAction bodyInfo opts | ||
let wholeBody = BL.toChunks bs | ||
(formparams, fs) <- parseRequestBody wholeBody W.lbsBackEnd req -- NB this loads the whole body into memory | ||
let convert (k, v) = (strictByteStringToLazyText k, strictByteStringToLazyText v) | ||
return (convert <$> formparams, fs) | ||
else | ||
return ([], []) | ||
|
||
-- | Retrieve the entire body, using the cached chunks in the BodyInfo and reading any other | ||
-- chunks if they still exist. | ||
-- Mimic the previous behavior by throwing BodyPartiallyStreamed if the user has already | ||
-- started reading the body by chunks. | ||
getBodyAction :: BodyInfo -> RouteOptions -> IO (BL.ByteString) | ||
getBodyAction (BodyInfo readProgress chunkBufferVar getChunk) opts = | ||
modifyMVar readProgress $ \index -> | ||
modifyMVar chunkBufferVar $ \bcb@(BodyChunkBuffer hasFinished chunks) -> do | ||
if | index > 0 -> throw BodyPartiallyStreamed | ||
| hasFinished -> return (bcb, (index, BL.fromChunks chunks)) | ||
| otherwise -> do | ||
newChunks <- readRequestBody getChunk return (maxRequestBodySize opts) | ||
return $ (BodyChunkBuffer True (chunks ++ newChunks), (index, BL.fromChunks (chunks ++ newChunks))) | ||
|
||
-- | Retrieve a chunk from the body at the index stored in the readProgress MVar. | ||
-- Serve the chunk from the cached array if it's already present; otherwise read another | ||
-- chunk from WAI and advance the index. | ||
getBodyChunkAction :: BodyInfo -> IO BS.ByteString | ||
getBodyChunkAction (BodyInfo readProgress chunkBufferVar getChunk) = | ||
modifyMVar readProgress $ \index -> | ||
modifyMVar chunkBufferVar $ \bcb@(BodyChunkBuffer hasFinished chunks) -> do | ||
if | index < length chunks -> return (bcb, (index + 1, chunks !! index)) | ||
| hasFinished -> return (bcb, (index, mempty)) | ||
| otherwise -> do | ||
newChunk <- getChunk | ||
return (BodyChunkBuffer (newChunk == mempty) (chunks ++ [newChunk]), (index + 1, newChunk)) | ||
|
||
|
||
-- Stolen from wai-extra's Network.Wai.Parse, modified to accept body as list of Bytestrings. | ||
-- Reason: WAI's requestBody is an IO action that returns the body as chunks. Once read, | ||
-- they can't be read again. We read them into a lazy Bytestring, so Scotty user can get | ||
-- the raw body, even if they also want to call wai-extra's parsing routines. | ||
parseRequestBody :: MonadIO m | ||
=> [B.ByteString] | ||
-> W.BackEnd y | ||
-> Request | ||
-> m ([W.Param], [W.File y]) | ||
parseRequestBody bl s r = | ||
case W.getRequestBodyType r of | ||
Nothing -> return ([], []) | ||
Just rbt -> do | ||
mvar <- liftIO $ newMVar bl -- MVar is a bit of a hack so we don't have to inline | ||
-- large portions of Network.Wai.Parse | ||
let provider = modifyMVar mvar $ \bsold -> case bsold of | ||
[] -> return ([], B.empty) | ||
(b:bs) -> return (bs, b) | ||
liftIO $ W.sinkRequestBody s rbt provider |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.