module Network.Wai.Parse
( parseHttpAccept
, parseRequestBody
, Sink (..)
, lbsSink
, tempFileSink
, Param
, File
, FileInfo (..)
#if TEST
, Bound (..)
, findBound
, sinkTillBound
, killCR
, killCRLF
, takeLine
#endif
) where
import qualified Data.ByteString as S
import qualified Data.ByteString.Lazy as L
import qualified Data.ByteString.Char8 as S8
import Data.Word (Word8)
import Data.Bits
import Data.Maybe (fromMaybe)
import Data.List (sortBy)
import Data.Function (on)
import System.Directory (removeFile, getTemporaryDirectory)
import System.IO (hClose, openBinaryTempFile, Handle)
import Network.Wai
import Data.Enumerator (Iteratee, yield)
import qualified Data.Enumerator as E
import qualified Data.Enumerator.List as EL
import Control.Monad.IO.Class (liftIO)
import qualified Network.HTTP.Types as H
uncons :: S.ByteString -> Maybe (Word8, S.ByteString)
uncons s
| S.null s = Nothing
| otherwise = Just (S.head s, S.tail s)
breakDiscard :: Word8 -> S.ByteString -> (S.ByteString, S.ByteString)
breakDiscard w s =
let (x, y) = S.break (== w) s
in (x, S.drop 1 y)
qsDecode :: S.ByteString -> S.ByteString
qsDecode z = fst $ S.unfoldrN (S.length z) go z
where
go bs =
case uncons bs of
Nothing -> Nothing
Just (43, ws) -> Just (32, ws)
Just (37, ws) -> Just $ fromMaybe (37, ws) $ do
(x, xs) <- uncons ws
x' <- hexVal x
(y, ys) <- uncons xs
y' <- hexVal y
Just $ (combine x' y', ys)
Just (w, ws) -> Just (w, ws)
hexVal w
| 48 <= w && w <= 57 = Just $ w 48
| 65 <= w && w <= 70 = Just $ w 55
| 97 <= w && w <= 102 = Just $ w 87
| otherwise = Nothing
combine :: Word8 -> Word8 -> Word8
combine a b = shiftL a 4 .|. b
parseHttpAccept :: S.ByteString -> [S.ByteString]
parseHttpAccept = map fst
. sortBy (rcompare `on` snd)
. map grabQ
. S.split 44
where
rcompare :: Double -> Double -> Ordering
rcompare = flip compare
grabQ s =
let (s', q) = breakDiscard 59 s
(_, q') = breakDiscard 61 q
in (trimWhite s', readQ $ trimWhite q')
readQ s = case reads $ S8.unpack s of
(x, _):_ -> x
_ -> 1.0
trimWhite = S.dropWhile (== 32)
data Sink x y = Sink
{ sinkInit :: IO x
, sinkAppend :: x -> S.ByteString -> IO x
, sinkClose :: x -> IO y
, sinkFinalize :: y -> IO ()
}
lbsSink :: Sink ([S.ByteString] -> [S.ByteString]) L.ByteString
lbsSink = Sink
{ sinkInit = return id
, sinkAppend = \front bs -> return $ front . (:) bs
, sinkClose = \front -> return $ L.fromChunks $ front []
, sinkFinalize = \_ -> return ()
}
tempFileSink :: Sink (FilePath, Handle) FilePath
tempFileSink = Sink
{ sinkInit = do
tempDir <- getTemporaryDirectory
openBinaryTempFile tempDir "webenc.buf"
, sinkAppend = \(fp, h) bs -> S.hPut h bs >> return (fp, h)
, sinkClose = \(fp, h) -> do
hClose h
return fp
, sinkFinalize = \fp -> removeFile fp
}
data FileInfo c = FileInfo
{ fileName :: S.ByteString
, fileContentType :: S.ByteString
, fileContent :: c
}
deriving (Eq, Show)
type Param = (S.ByteString, S.ByteString)
type File y = (S.ByteString, FileInfo y)
parseRequestBody :: Sink x y
-> Request
-> Iteratee S.ByteString IO ([Param], [File y])
parseRequestBody sink req = do
case ctype of
Nothing -> return ([], [])
Just Nothing -> do
bs <- EL.consume
return (H.parseSimpleQuery $ S.concat bs, [])
Just (Just bound) ->
let bound'' = S8.pack "--" `S.append` bound
in parsePieces sink bound''
where
urlenc = S8.pack "application/x-www-form-urlencoded"
formBound = S8.pack "multipart/form-data;"
bound' = "boundary="
boundary s =
if "multipart/form-data;" `S.isPrefixOf` s
then
let s' = S.dropWhile (== 32) $ S.drop (S.length formBound) s
in if bound' `S.isPrefixOf` s'
then Just $ S.drop (S.length bound') s'
else Nothing
else Nothing
ctype = do
ctype' <- lookup "Content-Type" $ requestHeaders req
if urlenc `S.isPrefixOf` ctype'
then Just Nothing
else case boundary ctype' of
Just x -> Just $ Just x
Nothing -> Nothing
takeLine :: Iteratee S.ByteString IO (Maybe S.ByteString)
takeLine = do
mbs <- EL.head
case mbs of
Nothing -> return Nothing
Just bs ->
let (x, y) = S.break (== 10) bs
in if S.null y
then do
x' <- takeLine
case x' of
Nothing -> return $ Just $ killCR x
Just x'' -> return $ Just $ killCR $ S.append x x''
else do
E.yield () $ E.Chunks [S.drop 1 y]
return $ Just $ killCR x
takeLines :: Iteratee S.ByteString IO [S.ByteString]
takeLines = do
res <- takeLine
case res of
Nothing -> return []
Just l
| S.null l -> return []
| otherwise -> do
ls <- takeLines
return $ l : ls
parsePieces :: Sink x y -> S.ByteString
-> Iteratee S.ByteString IO ([Param], [File y])
parsePieces sink bound = do
_boundLine <- takeLine
res' <- takeLines
case res' of
[] -> return ([], [])
_ -> do
let ls' = map parsePair res'
let x = do
cd <- lookup contDisp ls'
let ct = lookup contType ls'
let attrs = parseAttrs cd
name <- lookup "name" attrs
return (ct, name, lookup "filename" attrs)
case x of
Just (mct, name, Just filename) -> do
let ct = fromMaybe "application/octet-stream" mct
seed <- liftIO $ sinkInit sink
(seed', wasFound) <-
sinkTillBound bound (sinkAppend sink) seed
y <- liftIO $ sinkClose sink seed'
let fi = FileInfo filename ct y
let y' = (name, fi)
(xs, ys) <-
if wasFound
then parsePieces sink bound
else return ([], [])
return (xs, y' : ys)
Just (_ct, name, Nothing) -> do
let seed = id
let iter front bs = return $ front . (:) bs
(front, wasFound) <-
sinkTillBound bound iter seed
let bs = S.concat $ front []
let x' = (name, qsDecode bs)
(xs, ys) <-
if wasFound
then parsePieces sink bound
else return ([], [])
return (x' : xs, ys)
_ -> do
let seed = ()
iter () _ = return ()
((), wasFound) <- sinkTillBound bound iter seed
if wasFound
then parsePieces sink bound
else return ([], [])
where
contDisp = S8.pack "Content-Disposition"
contType = S8.pack "Content-Type"
parsePair s =
let (x, y) = breakDiscard 58 s
in (x, S.dropWhile (== 32) y)
data Bound = FoundBound S.ByteString S.ByteString
| NoBound
| PartialBound
deriving (Eq, Show)
findBound :: S.ByteString -> S.ByteString -> Bound
findBound b bs = go [0..S.length bs 1]
where
go [] = NoBound
go (i:is)
| mismatch [0..S.length b 1] [i..S.length bs 1] = go is
| otherwise =
let endI = i + S.length b
in if endI > S.length bs
then PartialBound
else FoundBound (S.take i bs) (S.drop endI bs)
mismatch [] _ = False
mismatch _ [] = False
mismatch (x:xs) (y:ys)
| S.index b x == S.index bs y = mismatch xs ys
| otherwise = True
sinkTillBound :: S.ByteString
-> (x -> S.ByteString -> IO x)
-> x
-> Iteratee S.ByteString IO (x, Bool)
sinkTillBound bound iter seed = do
mbs <- EL.head
case mbs of
Nothing -> return (seed, False)
Just bs -> go bs
where
go bs =
case findBound bound bs of
FoundBound before after -> do
let before' = killCRLF before
seed' <- liftIO $ iter seed before'
yield () $ E.Chunks [after]
return (seed', True)
PartialBound -> do
mbs <- EL.head
case mbs of
Nothing -> do
seed' <- liftIO $ iter seed bs
return (seed', False)
Just bs2 -> do
let bs' = bs `S.append` bs2
yield () $ E.Chunks [bs']
sinkTillBound bound iter seed
NoBound -> do
mbs <- EL.head
case mbs of
Nothing -> do
seed' <- liftIO $ iter seed bs
sinkTillBound bound iter seed'
Just bs' -> do
(seed', bs'') <-
if not (S8.null bs) && S8.last bs `elem` "\n\r"
then do
let (front, back) =
S.splitAt (S.length bs 2) bs
seed' <- liftIO $ iter seed front
return (seed', back `S.append` bs')
else do
seed' <- liftIO $ iter seed bs
return (seed', bs')
yield () $ E.Chunks [bs'']
sinkTillBound bound iter seed'
parseAttrs :: S.ByteString -> [(S.ByteString, S.ByteString)]
parseAttrs = map go . S.split 59
where
tw = S.dropWhile (== 32)
dq s = if S.length s > 2 && S.head s == 34 && S.last s == 34
then S.tail $ S.init s
else s
go s =
let (x, y) = breakDiscard 61 s
in (tw x, dq $ tw y)
killCRLF :: S.ByteString -> S.ByteString
killCRLF bs
| S.null bs || S.last bs /= 10 = bs
| otherwise = killCR $ S.init bs
killCR :: S.ByteString -> S.ByteString
killCR bs
| S.null bs || S.last bs /= 13 = bs
| otherwise = S.init bs