2 libmpd for Haskell, an MPD client library.
3 Copyright (C) 2005 Ben Sinclair <bsinclai@turing.une.edu.au>
5 This library is free software; you can redistribute it and/or
6 modify it under the terms of the GNU Lesser General Public
7 License as published by the Free Software Foundation; either
8 version 2.1 of the License, or (at your option) any later version.
10 This library is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 Lesser General Public License for more details.
15 You should have received a copy of the GNU Lesser General Public
16 License along with this library; if not, write to the Free Software
17 Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
21 -- Copyright : (c) Ben Sinclair 2005
23 -- Maintainer : bsinclai@turing.une.edu.au
25 -- Portability : Haskell 98
27 -- MPD client library.
34 -- * Running an action
41 getResponse
, clearerror
, close
, reconnect
, kill
,
43 module Control
.Monad
.Trans
46 import Control
.Monad
(liftM, unless)
47 import Control
.Exception
(finally
)
48 import Control
.Monad
.Trans
49 import Prelude
hiding (repeat)
50 import Data
.IORef
(IORef
, newIORef
, readIORef
, writeIORef
)
51 import Data
.List
(isPrefixOf)
55 import System
.IO.Error
(isEOFError)
61 -- | A connection to an MPD server.
62 -- don't export the field names.
63 data Connection
= Conn
{ connHostName
:: String
64 , connPortNum
:: Integer
65 , connHandle
:: IORef
(Maybe Handle)
66 , connGetPass
:: IO (Maybe String)
69 -- | Represents various MPD errors (aka. ACKs).
70 data ACK
= NoMPD
-- ^ MPD not responding
71 | TimedOut
-- ^ The connection timed out.
72 | Auth
-- ^ ACK [4\@0]
73 | Busy
-- ^ ACK [54\@0]
74 | UnknownCommand
String -- ^ ACK [5\@0]
75 | FileNotFound
-- ^ ACK [50\@0]
76 | FileExists
String -- ^ ACK [56\@0]
77 | System
String -- ^ ACK [52\@0]
78 | PlaylistLoad
-- ^ ACK [53\@0]
79 | NotPlaying
-- ^ ACK [55\@0]
80 | PlaylistMax
-- ^ ACK [51\@0]
81 | InvalidArgument
String -- ^ ACK [2\@0]
82 | InvalidPassword
-- ^ ACK [3\@0]
85 instance Show ACK
where
86 show NoMPD
= "Could not connect to MPD"
87 show TimedOut
= "MPD connection timed out"
88 show Auth
= "Password needed"
89 show Busy
= "Already updating"
90 show (UnknownCommand s
) = s
91 show FileNotFound
= "File or directory does not exist"
92 show (FileExists s
) = s
93 show (System s
) = "System error: " ++ s
94 show PlaylistLoad
= "Failed to load playlist"
95 show PlaylistMax
= "Playlist full"
96 show (InvalidArgument s
) = "Invalid argument: " ++ s
97 show InvalidPassword
= "Invalid password"
98 show NotPlaying
= "Playback stopped"
101 -- Export the type name but not the constructor or the field.
103 -- This is basically a state and an error monad combined. It's just
104 -- nice if we can have a few custom functions that fiddle with the
106 newtype MPD a
= MPD
{ runMPD
:: Connection
-> IO (Either ACK a
) }
108 instance Functor MPD
where
109 fmap f m
= MPD
$ \conn
-> either Left
(Right
. f
) `
liftM` runMPD m conn
111 instance Monad MPD
where
112 return a
= MPD
$ \_
-> return (Right a
)
113 m
>>= f
= MPD
$ \conn
-> runMPD m conn
>>=
114 either (return . Left
) (flip runMPD conn
. f
)
115 fail err
= MPD
$ \_
-> return $ Left
(Custom err
)
117 instance MonadIO MPD
where
118 liftIO m
= MPD
$ \_
-> liftM Right m
120 -- | Throw an exception.
121 throwMPD
:: ACK
-> MPD
()
122 throwMPD e
= MPD
$ \_
-> return (Left e
)
124 -- | Catch an exception from an action.
125 catchMPD
:: MPD a
-> (ACK
-> MPD a
) -> MPD a
126 catchMPD m h
= MPD
$ \conn
->
127 runMPD m conn
>>= either (flip runMPD conn
. h
) (return . Right
)
131 -- Basic connection functions
135 -- | Run an MPD action against a server.
136 withMPDEx
:: String -- ^ Host name.
137 -> Integer -- ^ Port number.
138 -> IO (Maybe String) -- ^ An action that supplies passwords.
139 -> MPD a
-- ^ The action to run.
141 withMPDEx host port getpw m
= do
142 hRef
<- newIORef Nothing
143 connect host port hRef
144 readIORef hRef
>>= maybe (return $ Left NoMPD
)
145 (\_
-> finally
(runMPD m
(Conn host port hRef getpw
)) (closeIO hRef
))
147 -- Connect to an MPD server.
148 connect
:: String -> Integer -- host and port
149 -> IORef
(Maybe Handle) -> IO ()
150 connect host port hRef
=
153 handle
<- connectTo host
. PortNumber
$ fromInteger port
154 writeIORef hRef
(Just handle
)
155 checkConn handle
>>= flip unless (closeIO hRef
)
157 -- Check that an MPD daemon is at the other end of a connection.
158 checkConn
:: Handle -> IO Bool
159 checkConn h
= isPrefixOf "OK MPD" `
liftM`
hGetLine h
161 -- Close a connection.
162 closeIO
:: IORef
(Maybe Handle) -> IO ()
164 readIORef hRef
>>= maybe (return ())
165 (\h
-> hPutStrLn h
"close" >> hClose h
)
166 writeIORef hRef Nothing
168 -- | Refresh a connection.
170 reconnect
= MPD
$ \(Conn host port hRef _
) -> do
171 connect host port hRef
172 liftM (maybe (Left NoMPD
) (const $ Right
())) (readIORef hRef
)
174 -- XXX this doesn't use the password supplying feature.
176 -- | Kill the server. Obviously, the connection is then invalid.
178 kill
= MPD
$ \conn
-> do
179 readIORef
(connHandle conn
) >>=
180 maybe (return ()) (\h
-> hPutStrLn h
"kill" >> hClose h
)
181 writeIORef
(connHandle conn
) Nothing
184 -- | Send a command to the MPD and return the result.
185 getResponse
:: String -> MPD
[String]
186 getResponse cmd
= MPD
$ \conn
-> do
187 readIORef
(connHandle conn
) >>=
188 maybe (return $ Left NoMPD
)
189 (\h
-> hPutStrLn h cmd
>> hFlush h
>>
190 loop h
(tryPassword conn
(getResponse cmd
)) [])
191 where loop h tryPw acc
= do
192 getln h
(\l
-> parseResponse
(loop h tryPw
) l acc
>>= either
193 (\x
-> case x
of Auth
-> tryPw
; _
-> return (Left x
))
196 catch (liftM Right
$ hGetLine h
) (return . Left
) >>=
197 either (\e
-> if isEOFError e
then return (Left TimedOut
)
201 -- Send a password to MPD and run an action on success, return an ACK
203 tryPassword
:: Connection
204 -> MPD a
-- run on success
206 tryPassword conn cont
= do
207 readIORef
(connHandle conn
) >>= maybe (return $ Left NoMPD
)
208 (\h
-> connGetPass conn
>>= maybe (return $ Left Auth
)
209 (\pw
-> do hPutStrLn h
("password " ++ pw
) >> hFlush h
211 case result
of "OK" -> runMPD cont conn
212 _
-> tryPassword conn cont
))
214 splitAck
:: String -> (String, String, String)
215 splitAck s
= (take 3 prefix
, code
, drop 2 msg
)
217 (_
, msg
) = break (== '}') msg
'
218 (code
, msg
') = break (== ' ') rest
219 (prefix
, rest
) = splitAt 4 s
221 -- > parseAck "ACK [5@0] {} unknown command \"pong\"" = Custom "unknown
223 parseAck
:: String -> ACK
224 parseAck s
= case code
of
227 "[2@0]" -> InvalidArgument msg
228 "[3@0]" -> InvalidPassword
229 "[51@0]" -> PlaylistMax
230 "[52@0]" -> System msg
231 "[53@0]" -> PlaylistLoad
232 "[55@0]" -> NotPlaying
233 "[5@0]" -> UnknownCommand msg
234 "[50@0]" -> FileNotFound
235 "[56@0]" -> FileExists msg
237 where (_
, code
, msg
) = splitAck s
239 -- Consume response and return a Response.
240 parseResponse
:: ([String] -> IO (Either ACK
[String]))
241 -> String -> [String] -> IO (Either ACK
[String])
242 parseResponse f s acc
243 |
isPrefixOf "ACK" s
= return $ Left
(parseAck s
)
244 |
isPrefixOf "OK" s
= return $ Right
(reverse acc
)
245 |
otherwise = f
(s
:acc
)
247 -- XXX this doesn't use the password supplying feature.
249 -- | Clear the current error message in status.
251 clearerror
= MPD
$ \conn
-> do
252 readIORef
(connHandle conn
) >>= maybe (return $ Left NoMPD
)
253 (\h
-> hPutStrLn h
"clearerror" >> hFlush h
>> return (Right
()))
255 -- | Close an MPD connection.
257 close
= MPD
$ \conn
-> closeIO
(connHandle conn
) >> return (Right
())