diff options
author | Joey Hess <joey@kitenet.net> | 2014-04-05 13:29:28 -0400 |
---|---|---|
committer | Joey Hess <joey@kitenet.net> | 2014-04-05 13:29:28 -0400 |
commit | 37de1fc1863379c4bc8f71210e63530a06ff8335 (patch) | |
tree | 9a400c1e5bc9a30069732af8a2a6784dd8996c2e /Utility/SimpleProtocol.hs | |
parent | 5ea17d174ac989c2bcadf5052be616ac4bc74988 (diff) |
factored out Utility.SimpleProtocol from the external special remote implementation
Diffstat (limited to 'Utility/SimpleProtocol.hs')
-rw-r--r-- | Utility/SimpleProtocol.hs | 75 |
1 files changed, 75 insertions, 0 deletions
diff --git a/Utility/SimpleProtocol.hs b/Utility/SimpleProtocol.hs new file mode 100644 index 000000000..9cc25bc91 --- /dev/null +++ b/Utility/SimpleProtocol.hs @@ -0,0 +1,75 @@ +{- Simple line-based protocols. + - + - Copyright 2013-2014 Joey Hess <joey@kitenet.net> + - + - Licensed under the GNU GPL version 3 or higher. + -} + +module Utility.SimpleProtocol ( + Sendable(..), + Receivable(..), + parseMessage, + Serializable(..), + Parser, + parseFail, + parse0, + parse1, + parse2, + parse3, +) where + +import Control.Applicative +import Data.Char + +import Utility.Misc + +-- Messages that can be sent. +class Sendable m where + formatMessage :: m -> [String] + +-- Messages that can be received. +class Receivable m where + -- Passed the first word of the message, returns + -- a Parser that can be be fed the rest of the message to generate + -- the value. + parseCommand :: String -> Parser m + +parseMessage :: (Receivable m) => String -> Maybe m +parseMessage s = parseCommand command rest + where + (command, rest) = splitWord s + +class Serializable a where + serialize :: a -> String + deserialize :: String -> Maybe a + +{- Parsing the parameters of messages. Using the right parseN ensures + - that the string is split into exactly the requested number of words, + - which allows the last parameter of a message to contain arbitrary + - whitespace, etc, without needing any special quoting. + -} +type Parser a = String -> Maybe a + +parseFail :: Parser a +parseFail _ = Nothing + +parse0 :: a -> Parser a +parse0 mk "" = Just mk +parse0 _ _ = Nothing + +parse1 :: Serializable p1 => (p1 -> a) -> Parser a +parse1 mk p1 = mk <$> deserialize p1 + +parse2 :: (Serializable p1, Serializable p2) => (p1 -> p2 -> a) -> Parser a +parse2 mk s = mk <$> deserialize p1 <*> deserialize p2 + where + (p1, p2) = splitWord s + +parse3 :: (Serializable p1, Serializable p2, Serializable p3) => (p1 -> p2 -> p3 -> a) -> Parser a +parse3 mk s = mk <$> deserialize p1 <*> deserialize p2 <*> deserialize p3 + where + (p1, rest) = splitWord s + (p2, p3) = splitWord rest + +splitWord :: String -> (String, String) +splitWord = separate isSpace |