import Prelude hiding (lookup)
import Data.ByteString.Char8 (ByteString, unpack)
import Data.Map as M ((!), lookup)
+import Data.List (intersperse)
import Crypto.Hash.SHA1 (hash)
import Data.Maybe (maybeToList)
import FuncTorrent.Bencode (BVal(..), encode, decode, bstrToString, bValToInteger)
--- only single file mode supported for the time being.
+data FileMeta = FileMeta { lengthInBytes :: !Integer
+ , md5sum :: !(Maybe String)
+ , path :: String
+ } deriving (Eq, Show)
+
data Info = Info { pieceLength :: !Integer
, pieces :: !ByteString
, private :: !(Maybe Integer)
, name :: !String
- , lengthInBytes :: !Integer
- , md5sum :: !(Maybe String)
+ , filemeta :: [FileMeta]
} deriving (Eq, Show)
data Metainfo = Metainfo { info :: !(Maybe Info)
(Bstr pieces') = m ! "pieces"
private' = Nothing
(Bstr name') = m ! "name"
+ -- is the key "files" present? If so, it is a multi-file torrent
+ -- if not, it is a single file torrent.
+ filesIfMulti = lookup "files" m
(Bint length') = m ! "length"
md5sum' = Nothing
- in Just Info { pieceLength = pieceLength'
- , pieces = pieces'
- , private = private'
- , name = unpack name'
- , lengthInBytes = length'
- , md5sum = md5sum'}
+ partialInfo = Info { pieceLength = pieceLength'
+ , pieces = pieces'
+ , private = private'
+ , name = unpack name'
+ }
+ in
+ case filesIfMulti of
+ Nothing -> let (Bint length') = m ! "length"
+ filemeta' = FileMeta { lengthInBytes = length'
+ , md5sum = Nothing
+ , path = unpack name' }
+ in Just (partialInfo { filemeta = [filemeta'] })
+ Just (Blist files) -> mapM toFileMeta files >>=
+ \filemeta' ->
+ Just partialInfo { filemeta = filemeta' }
bvalToInfo _ = Nothing
+toFileMeta :: BVal -> Maybe FileMeta
+toFileMeta (Bdict fm) = let (Bint length) = fm ! "length"
+ (Blist pathElems) = fm ! "path"
+ pathStrings = fmap bstrToString pathElems
+ in
+ sequence pathStrings >>=
+ \pathList -> let path' = concat $ intersperse "/" pathList
+ in Just (FileMeta { lengthInBytes = length
+ , md5sum = Nothing
+ , path = path' })
+toFileMeta _ = Nothing
+
mkMetaInfo :: BVal -> Either String Metainfo
mkMetaInfo (Bdict m) =
let info' = bvalToInfo $ m ! "info"