module Data.MessagePack.Types.Class
( MessagePack (..)
, GMessagePack (..)
) where
import Control.Applicative (Applicative, (<$>), (<*>))
import Control.Arrow ((***))
import qualified Data.ByteString as S
import qualified Data.ByteString.Lazy as L
import Data.Hashable (Hashable)
import qualified Data.HashMap.Strict as HashMap
import Data.Int (Int16, Int32, Int64, Int8)
import qualified Data.IntMap.Strict as IntMap
import qualified Data.Map as Map
import qualified Data.Text as T
import qualified Data.Text.Lazy as LT
import qualified Data.Vector as V
import qualified Data.Vector.Storable as VS
import qualified Data.Vector.Unboxed as VU
import Data.Word (Word, Word16, Word32, Word64,
Word8)
import GHC.Generics
import Data.MessagePack.Types.Assoc
import Data.MessagePack.Types.Object
class GMessagePack f where
gToObject :: f a -> Object
gFromObject :: (Applicative m, Monad m) => Object -> m (f a)
class MessagePack a where
toObject :: a -> Object
fromObject :: (Applicative m, Monad m) => Object -> m a
default toObject :: (Generic a, GMessagePack (Rep a))
=> a -> Object
toObject = genericToObject
default fromObject :: ( Applicative m, Monad m
, Generic a, GMessagePack (Rep a))
=> Object -> m a
fromObject = genericFromObject
genericToObject :: (Generic a, GMessagePack (Rep a))
=> a -> Object
genericToObject = gToObject . from
genericFromObject :: ( Applicative m, Monad m
, Generic a, GMessagePack (Rep a))
=> Object -> m a
genericFromObject x = to <$> gFromObject x
toInt :: Integral a => a -> Int64
toInt = fromIntegral
fromInt :: Integral a => Int64 -> a
fromInt = fromIntegral
toWord :: Integral a => a -> Word64
toWord = fromIntegral
fromWord :: Integral a => Word64 -> a
fromWord = fromIntegral
instance MessagePack Int64 where
toObject i
| i < 0 = ObjectInt i
| otherwise = ObjectWord $ toWord i
fromObject = \case
ObjectInt n -> return n
ObjectWord n -> return $ toInt n
_ -> fail "invalid encoding for integer type"
instance MessagePack Word64 where
toObject = ObjectWord
fromObject = \case
ObjectWord n -> return n
_ -> fail "invalid encoding for integer type"
instance MessagePack Int where { toObject = toObject . toInt; fromObject o = fromInt <$> fromObject o }
instance MessagePack Int8 where { toObject = toObject . toInt; fromObject o = fromInt <$> fromObject o }
instance MessagePack Int16 where { toObject = toObject . toInt; fromObject o = fromInt <$> fromObject o }
instance MessagePack Int32 where { toObject = toObject . toInt; fromObject o = fromInt <$> fromObject o }
instance MessagePack Word where { toObject = toObject . toWord; fromObject o = fromWord <$> fromObject o }
instance MessagePack Word8 where { toObject = toObject . toWord; fromObject o = fromWord <$> fromObject o }
instance MessagePack Word16 where { toObject = toObject . toWord; fromObject o = fromWord <$> fromObject o }
instance MessagePack Word32 where { toObject = toObject . toWord; fromObject o = fromWord <$> fromObject o }
instance MessagePack Object where
toObject = id
fromObject = return
instance MessagePack () where
toObject _ = ObjectNil
fromObject = \case
ObjectNil -> return ()
ObjectArray [] -> return ()
_ -> fail "invalid encoding for ()"
instance MessagePack Bool where
toObject = ObjectBool
fromObject = \case
ObjectBool b -> return b
_ -> fail "invalid encoding for Bool"
instance MessagePack Float where
toObject = ObjectFloat
fromObject = \case
ObjectInt n -> return $ fromIntegral n
ObjectWord n -> return $ fromIntegral n
ObjectFloat f -> return f
ObjectDouble d -> return $ realToFrac d
_ -> fail "invalid encoding for Float"
instance MessagePack Double where
toObject = ObjectDouble
fromObject = \case
ObjectInt n -> return $ fromIntegral n
ObjectWord n -> return $ fromIntegral n
ObjectFloat f -> return $ realToFrac f
ObjectDouble d -> return d
_ -> fail "invalid encoding for Double"
instance MessagePack String where
toObject = toObject . T.pack
fromObject obj = T.unpack <$> fromObject obj
instance MessagePack S.ByteString where
toObject = ObjectBin
fromObject = \case
ObjectBin r -> return r
_ -> fail "invalid encoding for ByteString"
instance MessagePack L.ByteString where
toObject = ObjectBin . L.toStrict
fromObject obj = L.fromStrict <$> fromObject obj
instance MessagePack T.Text where
toObject = ObjectStr
fromObject = \case
ObjectStr s -> return s
_ -> fail "invalid encoding for Text"
instance MessagePack LT.Text where
toObject = toObject . LT.toStrict
fromObject obj = LT.fromStrict <$> fromObject obj
instance MessagePack a => MessagePack [a] where
toObject = ObjectArray . map toObject
fromObject = \case
ObjectArray xs -> mapM fromObject xs
_ -> fail "invalid encoding for list"
instance MessagePack a => MessagePack (V.Vector a) where
toObject = ObjectArray . map toObject . V.toList
fromObject = \case
ObjectArray o -> V.fromList <$> mapM fromObject o
_ -> fail "invalid encoding for Vector"
instance (MessagePack a, VU.Unbox a) => MessagePack (VU.Vector a) where
toObject = ObjectArray . map toObject . VU.toList
fromObject = \case
ObjectArray o -> VU.fromList <$> mapM fromObject o
_ -> fail "invalid encoding for Unboxed Vector"
instance (MessagePack a, VS.Storable a) => MessagePack (VS.Vector a) where
toObject = ObjectArray . map toObject . VS.toList
fromObject = \case
ObjectArray o -> VS.fromList <$> mapM fromObject o
_ -> fail "invalid encoding for Storable Vector"
instance (MessagePack a, MessagePack b) => MessagePack (Assoc [(a, b)]) where
toObject (Assoc xs) = ObjectMap $ map (toObject *** toObject) xs
fromObject = \case
ObjectMap xs ->
Assoc <$> mapM (\(k, v) -> (,) <$> fromObject k <*> fromObject v) xs
_ ->
fail "invalid encoding for Assoc"
instance (MessagePack k, MessagePack v, Ord k) => MessagePack (Map.Map k v) where
toObject = toObject . Assoc . Map.toList
fromObject obj = Map.fromList . unAssoc <$> fromObject obj
instance MessagePack v => MessagePack (IntMap.IntMap v) where
toObject = toObject . Assoc . IntMap.toList
fromObject obj = IntMap.fromList . unAssoc <$> fromObject obj
instance (MessagePack k, MessagePack v, Hashable k, Eq k) => MessagePack (HashMap.HashMap k v) where
toObject = toObject . Assoc . HashMap.toList
fromObject obj = HashMap.fromList . unAssoc <$> fromObject obj
instance (MessagePack a1, MessagePack a2) => MessagePack (a1, a2) where
toObject (a1, a2) = ObjectArray [toObject a1, toObject a2]
fromObject (ObjectArray [a1, a2]) = (,) <$> fromObject a1 <*> fromObject a2
fromObject _ = fail "invalid encoding for tuple"
instance (MessagePack a1, MessagePack a2, MessagePack a3) => MessagePack (a1, a2, a3) where
toObject (a1, a2, a3) = ObjectArray [toObject a1, toObject a2, toObject a3]
fromObject (ObjectArray [a1, a2, a3]) = (,,) <$> fromObject a1 <*> fromObject a2 <*> fromObject a3
fromObject _ = fail "invalid encoding for tuple"
instance (MessagePack a1, MessagePack a2, MessagePack a3, MessagePack a4) => MessagePack (a1, a2, a3, a4) where
toObject (a1, a2, a3, a4) = ObjectArray [toObject a1, toObject a2, toObject a3, toObject a4]
fromObject (ObjectArray [a1, a2, a3, a4]) = (,,,) <$> fromObject a1 <*> fromObject a2 <*> fromObject a3 <*> fromObject a4
fromObject _ = fail "invalid encoding for tuple"
instance (MessagePack a1, MessagePack a2, MessagePack a3, MessagePack a4, MessagePack a5) => MessagePack (a1, a2, a3, a4, a5) where
toObject (a1, a2, a3, a4, a5) = ObjectArray [toObject a1, toObject a2, toObject a3, toObject a4, toObject a5]
fromObject (ObjectArray [a1, a2, a3, a4, a5]) = (,,,,) <$> fromObject a1 <*> fromObject a2 <*> fromObject a3 <*> fromObject a4 <*> fromObject a5
fromObject _ = fail "invalid encoding for tuple"
instance (MessagePack a1, MessagePack a2, MessagePack a3, MessagePack a4, MessagePack a5, MessagePack a6) => MessagePack (a1, a2, a3, a4, a5, a6) where
toObject (a1, a2, a3, a4, a5, a6) = ObjectArray [toObject a1, toObject a2, toObject a3, toObject a4, toObject a5, toObject a6]
fromObject (ObjectArray [a1, a2, a3, a4, a5, a6]) = (,,,,,) <$> fromObject a1 <*> fromObject a2 <*> fromObject a3 <*> fromObject a4 <*> fromObject a5 <*> fromObject a6
fromObject _ = fail "invalid encoding for tuple"
instance (MessagePack a1, MessagePack a2, MessagePack a3, MessagePack a4, MessagePack a5, MessagePack a6, MessagePack a7) => MessagePack (a1, a2, a3, a4, a5, a6, a7) where
toObject (a1, a2, a3, a4, a5, a6, a7) = ObjectArray [toObject a1, toObject a2, toObject a3, toObject a4, toObject a5, toObject a6, toObject a7]
fromObject (ObjectArray [a1, a2, a3, a4, a5, a6, a7]) = (,,,,,,) <$> fromObject a1 <*> fromObject a2 <*> fromObject a3 <*> fromObject a4 <*> fromObject a5 <*> fromObject a6 <*> fromObject a7
fromObject _ = fail "invalid encoding for tuple"
instance (MessagePack a1, MessagePack a2, MessagePack a3, MessagePack a4, MessagePack a5, MessagePack a6, MessagePack a7, MessagePack a8) => MessagePack (a1, a2, a3, a4, a5, a6, a7, a8) where
toObject (a1, a2, a3, a4, a5, a6, a7, a8) = ObjectArray [toObject a1, toObject a2, toObject a3, toObject a4, toObject a5, toObject a6, toObject a7, toObject a8]
fromObject (ObjectArray [a1, a2, a3, a4, a5, a6, a7, a8]) = (,,,,,,,) <$> fromObject a1 <*> fromObject a2 <*> fromObject a3 <*> fromObject a4 <*> fromObject a5 <*> fromObject a6 <*> fromObject a7 <*> fromObject a8
fromObject _ = fail "invalid encoding for tuple"
instance (MessagePack a1, MessagePack a2, MessagePack a3, MessagePack a4, MessagePack a5, MessagePack a6, MessagePack a7, MessagePack a8, MessagePack a9) => MessagePack (a1, a2, a3, a4, a5, a6, a7, a8, a9) where
toObject (a1, a2, a3, a4, a5, a6, a7, a8, a9) = ObjectArray [toObject a1, toObject a2, toObject a3, toObject a4, toObject a5, toObject a6, toObject a7, toObject a8, toObject a9]
fromObject (ObjectArray [a1, a2, a3, a4, a5, a6, a7, a8, a9]) = (,,,,,,,,) <$> fromObject a1 <*> fromObject a2 <*> fromObject a3 <*> fromObject a4 <*> fromObject a5 <*> fromObject a6 <*> fromObject a7 <*> fromObject a8 <*> fromObject a9
fromObject _ = fail "invalid encoding for tuple"