module Network.MPD.Commands.Parse where
import Network.MPD.Commands.Types
import Control.Monad.Error
import Network.MPD.Utils
import Network.MPD.Core (MonadMPD, MPDError(Unexpected))
parseCount :: [String] -> Either String Count
parseCount = foldM f defaultCount . toAssocList
where f :: Count -> (String, String) -> Either String Count
f a ("songs", x) = return $ parse parseNum
(\x' -> a { cSongs = x'}) a x
f a ("playtime", x) = return $ parse parseNum
(\x' -> a { cPlaytime = x' }) a x
f _ x = Left $ show x
parseOutputs :: [String] -> Either String [Device]
parseOutputs = mapM (foldM f defaultDevice)
. splitGroups [("outputid",id)]
. toAssocList
where f a ("outputid", x) = return $ parse parseNum
(\x' -> a { dOutputID = x' }) a x
f a ("outputname", x) = return a { dOutputName = x }
f a ("outputenabled", x) = return $ parse parseBool
(\x' -> a { dOutputEnabled = x'}) a x
f _ x = fail $ show x
parseStats :: [String] -> Either String Stats
parseStats = foldM f defaultStats . toAssocList
where
f a ("artists", x) = return $ parse parseNum
(\x' -> a { stsArtists = x' }) a x
f a ("albums", x) = return $ parse parseNum
(\x' -> a { stsAlbums = x' }) a x
f a ("songs", x) = return $ parse parseNum
(\x' -> a { stsSongs = x' }) a x
f a ("uptime", x) = return $ parse parseNum
(\x' -> a { stsUptime = x' }) a x
f a ("playtime", x) = return $ parse parseNum
(\x' -> a { stsPlaytime = x' }) a x
f a ("db_playtime", x) = return $ parse parseNum
(\x' -> a { stsDbPlaytime = x' }) a x
f a ("db_update", x) = return $ parse parseNum
(\x' -> a { stsDbUpdate = x' }) a x
f _ x = fail $ show x
parseSong :: [(String, String)] -> Either String Song
parseSong xs = foldM f defaultSong xs
where f a ("Artist", x) = return a { sgArtist = x }
f a ("Album", x) = return a { sgAlbum = x }
f a ("Title", x) = return a { sgTitle = x }
f a ("Genre", x) = return a { sgGenre = x }
f a ("Name", x) = return a { sgName = x }
f a ("Composer", x) = return a { sgComposer = x }
f a ("Performer", x) = return a { sgPerformer = x }
f a ("Date", x) = return $ parse parseDate
(\x' -> a { sgDate = x' }) a x
f a ("Track", x) = return $ parse parseTuple
(\x' -> a { sgTrack = x'}) a x
f a ("Disc", x) = return a { sgDisc = parseTuple x }
f a ("file", x) = return a { sgFilePath = x }
f a ("Time", x) = return $ parse parseNum
(\x' -> a { sgLength = x'}) a x
f a ("Id", x) = return $ parse parseNum
(\x' -> a { sgIndex = Just (ID x') }) a x
f a ("Pos", x) =
maybe (return $ parse parseNum
(\x' -> a { sgIndex = Just (Pos x') }) a x)
(const $ return a)
(sgIndex a)
f a (k, v) = return a { sgAux = (k, v) : sgAux a }
parseTuple s = let (x, y) = breakChar '/' s in
case (parseNum x, parseNum y) of
(Just x', Nothing) -> Just (x', x')
(Just x', Just y') -> Just (x', y')
_ -> Nothing
parseStatus :: [String] -> Either String Status
parseStatus = foldM f defaultStatus . toAssocList
where f a ("state", x)
= return $ parse state (\x' -> a { stState = x' }) a x
f a ("volume", x)
= return $ parse parseNum (\x' -> a { stVolume = x' }) a x
f a ("repeat", x)
= return $ parse parseBool (\x' -> a { stRepeat = x' }) a x
f a ("random", x)
= return $ parse parseBool (\x' -> a { stRandom = x' }) a x
f a ("playlist", x)
= return $ parse parseNum (\x' -> a { stPlaylistVersion = x' }) a x
f a ("playlistlength", x)
= return $ parse parseNum (\x' -> a { stPlaylistLength = x' }) a x
f a ("xfade", x)
= return $ parse parseNum (\x' -> a { stXFadeWidth = x' }) a x
f a ("song", x)
= return $ parse parseNum (\x' -> a { stSongPos = Just (Pos x') }) a x
f a ("songid", x)
= return $ parse parseNum (\x' -> a { stSongID = Just (ID x') }) a x
f a ("time", x)
= return $ parse time (\x' -> a { stTime = x' }) a x
f a ("bitrate", x)
= return $ parse parseNum (\x' -> a { stBitrate = x' }) a x
f a ("audio", x)
= return $ parse audio (\x' -> a { stAudio = x' }) a x
f a ("updating_db", x)
= return $ parse parseNum (\x' -> a { stUpdatingDb = x' }) a x
f a ("error", x)
= return a { stError = x }
f a ("single", x)
= return $ parse parseBool (\x' -> a { stSingle = x' }) a x
f a ("consume", x)
= return $ parse parseBool (\x' -> a { stConsume = x' }) a x
f a ("nextsong", x)
= return $ parse parseNum (\x' -> a { stNextSongPos = Just (Pos x') }) a x
f a ("nextsongid", x)
= return $ parse parseNum (\x' -> a { stNextSongID = Just (ID x') }) a x
f _ x
= fail $ show x
state "play" = Just Playing
state "pause" = Just Paused
state "stop" = Just Stopped
state _ = Nothing
time = pair parseNum . breakChar ':'
audio s = let (u, u') = breakChar ':' s
(v, w) = breakChar ':' u' in
case (parseNum u, parseNum v, parseNum w) of
(Just a, Just b, Just c) -> Just (a, b, c)
_ -> Nothing
runParser :: (MonadMPD m, MonadError MPDError m)
=> (input -> Either String a) -> input -> m a
runParser f = either (throwError . Unexpected) return . f
parse :: (String -> Maybe a) -> (a -> b) -> b -> String -> b
parse parser f x = maybe x f . parser
pair :: (String -> Maybe a) -> (String, String) -> Maybe (a, a)
pair p (x, y) = case (p x, p y) of
(Just a, Just b) -> Just (a, b)
_ -> Nothing