1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57
|
{-# LANGUAGE BangPatterns, OverloadedStrings #-}
{-# OPTIONS_GHC -fno-warn-missing-signatures -fno-warn-orphans #-}
module HeadersByteString.Atto
(
request
, response
) where
import Control.Applicative
import Control.DeepSeq (NFData(..))
import Network.HTTP.Types.Version (HttpVersion, http11)
import qualified Data.Attoparsec.ByteString.Char8 as B
import qualified Data.ByteString.Char8 as B
instance NFData HttpVersion where
rnf !_ = ()
isHeaderChar :: Char -> Bool
isHeaderChar c =
(c >= 'a' && c <= 'z') ||
(c >= 'A' && c <= 'Z') ||
(c >= '0' && c <= '9') ||
(c == '_') ||
(c == '-')
header = do
name <- B.takeWhile1 isHeaderChar <* B.char ':' <* B.skipSpace
body <- bodyLine
return (name, body)
bodyLine = B.takeTill (\c -> c == '\r' || c == '\n') <* B.endOfLine
requestLine = do
m <- (B.takeTill B.isSpace <* B.char ' ')
(p,q) <- B.break (=='?') <$> (B.takeTill B.isSpace <* B.char ' ')
v <- httpVersion
return (m,p,q,v)
httpVersion = http11 <$ "HTTP/1.1"
responseLine = (,,) <$>
(httpVersion <* B.skipSpace) <*>
(int <* B.skipSpace) <*>
bodyLine
int :: B.Parser Int
int = B.decimal
request = (,) <$> (requestLine <* B.endOfLine) <*> manyheader
response = (,) <$> responseLine <*> many header
manyheader = do
c <- B.peekChar'
if c == '\r' || c == '\n'
then return []
else (:) <$> header <*> manyheader
|