module Network.HPACK.Table (
HeaderTable
, newHeaderTableForEncoding
, newHeaderTableForDecoding
, shouldRenew
, renewHeaderTable
, printHeaderTable
, insertEntry
, HeaderCache(..)
, lookupTable
, module Network.HPACK.Table.Entry
, WhichTable(..)
, which
) where
import Control.Applicative ((<$>))
import Control.Exception (throwIO)
import Data.Array.IO (IOArray, newArray, readArray, writeArray)
import qualified Data.ByteString.Char8 as BS
import Network.HPACK.Table.Entry
import qualified Network.HPACK.Table.HashPSQ as HP
import Network.HPACK.Table.Static
import Network.HPACK.Types
import Control.Monad (forM)
type Table = IOArray Index Entry
data HeaderTable = HeaderTable {
circularTable :: !Table
, offset :: !Index
, numOfEntries :: !Int
, maxNumOfEntries :: !Int
, headerTableSize :: !Size
, maxHeaderTableSize :: !Size
, reverseIndex :: Maybe (HP.HashPSQ HIndex)
}
adj :: Int -> Int -> Int
adj maxN x = (x + maxN) `mod` maxN
printHeaderTable :: HeaderTable -> IO ()
printHeaderTable HeaderTable{..} = do
es <- mapM (readArray circularTable . adj maxNumOfEntries) [beg .. end]
let ts = zip [1..] es
mapM_ printEntry ts
putStrLn $ " Table size: " ++ show headerTableSize ++ "/" ++ show maxHeaderTableSize
print reverseIndex
where
beg = offset + 1
end = offset + numOfEntries
printEntry :: (Index,Entry) -> IO ()
printEntry (i,e) = do
putStr "[ "
putStr $ show i
putStr "] (s = "
putStr $ show $ entrySize e
putStr ") "
BS.putStr $ entryHeaderName e
putStr ": "
BS.putStrLn $ entryHeaderValue e
data WhichTable = InHeaderTable | InStaticTable deriving (Eq,Show)
data HeaderCache = None
| KeyOnly WhichTable Index
| KeyValue WhichTable Index deriving Show
newtype HIndex = HIndex Int deriving (Eq, Ord, Show)
fromHIndexToIndex :: HeaderTable -> HIndex -> Index
fromHIndexToIndex HeaderTable{..} (HIndex hidx) = idx
where
idx = adj maxNumOfEntries (hidx offset) + staticTableSize
fromIndexToHIndex :: HeaderTable -> Index -> HIndex
fromIndexToHIndex HeaderTable{..} idx = HIndex hidx
where
hidx = adj maxNumOfEntries (idx + offset staticTableSize)
fromSIndexToIndex :: HeaderTable -> SIndex -> Index
fromSIndexToIndex _ sidx = fromStaticIndex sidx
fromIndexToSIndex :: HeaderTable -> Index -> SIndex
fromIndexToSIndex _ idx = toStaticIndex idx
newHeaderTableForEncoding :: Size -> IO HeaderTable
newHeaderTableForEncoding maxsiz = newHeaderTable maxsiz (Just HP.empty)
newHeaderTableForDecoding :: Size -> IO HeaderTable
newHeaderTableForDecoding maxsiz = newHeaderTable maxsiz Nothing
newHeaderTable :: Size -> Maybe (HP.HashPSQ HIndex) -> IO HeaderTable
newHeaderTable maxsiz mhp = do
tbl <- newArray (0,end) dummyEntry
return HeaderTable {
maxNumOfEntries = maxN
, offset = end
, numOfEntries = 0
, circularTable = tbl
, headerTableSize = 0
, maxHeaderTableSize = maxsiz
, reverseIndex = mhp
}
where
maxN = maxNumbers maxsiz
end = maxN 1
renewHeaderTable :: Size -> HeaderTable -> IO HeaderTable
renewHeaderTable maxsiz oldhdrtbl =
newHeaderTable maxsiz mhp >>= copyTable oldhdrtbl
where
mhp = case reverseIndex oldhdrtbl of
Nothing -> Nothing
_ -> Just HP.empty
copyTable :: HeaderTable -> HeaderTable -> IO HeaderTable
copyTable oldhdrtbl newhdrtbl = getEntries oldhdrtbl >>= copyEntries newhdrtbl
getEntries :: HeaderTable -> IO [Entry]
getEntries HeaderTable{..} = forM [1 .. numOfEntries] readTable
where
readTable i = readArray circularTable $ adj maxNumOfEntries (offset + i)
copyEntries :: HeaderTable -> [Entry] -> IO HeaderTable
copyEntries hdrtbl [] = return hdrtbl
copyEntries hdrtbl@HeaderTable{..} (e:es)
| headerTableSize + entrySize e <= maxHeaderTableSize = do
hdrtbl' <- insertEnd e hdrtbl
copyEntries hdrtbl' es
| otherwise = return hdrtbl
shouldRenew :: HeaderTable -> Size -> Bool
shouldRenew HeaderTable{..} maxsiz = maxHeaderTableSize /= maxsiz
insertEntry :: Entry -> HeaderTable -> IO HeaderTable
insertEntry e hdrtbl = do
(hdrtbl', hs) <- insertFront e hdrtbl >>= adjustTableSize
let hdrtbl'' = case reverseIndex hdrtbl' of
Nothing -> hdrtbl'
Just rev -> hdrtbl' { reverseIndex = Just (HP.deleteList hs rev) }
return hdrtbl''
insertFront :: Entry -> HeaderTable -> IO HeaderTable
insertFront e hdrtbl@HeaderTable{..} = do
writeArray circularTable i e
return $ hdrtbl {
offset = offset'
, numOfEntries = numOfEntries + 1
, headerTableSize = headerTableSize'
, reverseIndex = reverseIndex'
}
where
i = offset
headerTableSize' = headerTableSize + entrySize e
offset' = adj maxNumOfEntries (offset 1)
reverseIndex' = case reverseIndex of
Nothing -> Nothing
Just rev -> Just $ HP.insert (entryHeader e) (HIndex i) rev
adjustTableSize :: HeaderTable -> IO (HeaderTable, [Header])
adjustTableSize hdrtbl = adjust hdrtbl []
adjust :: HeaderTable -> [Header] -> IO (HeaderTable, [Header])
adjust hdrtbl hs
| tsize <= maxtsize = return (hdrtbl, hs)
| otherwise = do
(hdrtbl', h) <- removeEnd hdrtbl
adjust hdrtbl' (h:hs)
where
tsize = headerTableSize hdrtbl
maxtsize = maxHeaderTableSize hdrtbl
insertEnd :: Entry -> HeaderTable -> IO HeaderTable
insertEnd e hdrtbl@HeaderTable{..} = do
writeArray circularTable i e
return $ hdrtbl {
numOfEntries = numOfEntries + 1
, headerTableSize = headerTableSize'
, reverseIndex = reverseIndex'
}
where
i = adj maxNumOfEntries (offset + numOfEntries + 1)
headerTableSize' = headerTableSize + entrySize e
reverseIndex' = case reverseIndex of
Nothing -> Nothing
Just rev -> Just $ HP.insert (entryHeader e) (HIndex i) rev
removeEnd :: HeaderTable -> IO (HeaderTable,Header)
removeEnd hdrtbl@HeaderTable{..} = do
let i = adj maxNumOfEntries (offset + numOfEntries)
e <- readArray circularTable i
writeArray circularTable i dummyEntry
let tsize = headerTableSize entrySize e
h = entryHeader e
hdrtbl' = hdrtbl {
numOfEntries = numOfEntries 1
, headerTableSize = tsize
}
return (hdrtbl', h)
lookupTable :: Header -> HeaderTable -> HeaderCache
lookupTable h hdrtbl = case reverseIndex hdrtbl of
Nothing -> None
Just rev -> case HP.search h rev of
HP.N -> case HP.search h staticHashPSQ of
HP.N -> None
HP.K sidx -> KeyOnly InStaticTable (fromSIndexToIndex hdrtbl sidx)
HP.KV sidx -> KeyValue InStaticTable (fromSIndexToIndex hdrtbl sidx)
HP.K hidx -> case HP.search h staticHashPSQ of
HP.N -> KeyOnly InHeaderTable (fromHIndexToIndex hdrtbl hidx)
HP.K sidx -> KeyOnly InStaticTable (fromSIndexToIndex hdrtbl sidx)
HP.KV sidx -> KeyValue InStaticTable (fromSIndexToIndex hdrtbl sidx)
HP.KV hidx -> KeyValue InHeaderTable (fromHIndexToIndex hdrtbl hidx)
isIn :: Int -> HeaderTable -> Bool
isIn idx HeaderTable{..} = idx > staticTableSize
which :: HeaderTable -> Index -> IO (WhichTable, Entry)
which hdrtbl idx
| idx `isIn` hdrtbl = (InHeaderTable,) <$> toHeaderEntry hdrtbl hidx
| isSIndexValid sidx = return (InStaticTable, toStaticEntry sidx)
| otherwise = throwIO $ IndexOverrun idx
where
hidx = fromIndexToHIndex hdrtbl idx
sidx = fromIndexToSIndex hdrtbl idx
toHeaderEntry :: HeaderTable -> HIndex -> IO Entry
toHeaderEntry HeaderTable{..} (HIndex hidx) = readArray circularTable hidx