{-# LANGUAGE AllowAmbiguousTypes #-}
{-# LANGUAGE DataKinds #-}
{-# LANGUAGE GADTs #-}
{-# LANGUAGE KindSignatures #-}
{-# LANGUAGE MultiParamTypeClasses #-}
{-# LANGUAGE ScopedTypeVariables #-}
{-# LANGUAGE TypeApplications #-}
module Antiope.Messages.Types
( WithEncoded(..)
, With(..)
, FromWith(..)
, fromWith2, fromWith3
) where
import Data.Aeson (FromJSON (..), ToJSON (..), eitherDecodeStrict, encode, withObject, (.:), (.=))
import Data.Text (Text)
import GHC.TypeLits
import qualified Data.Aeson as Aeson
import qualified Data.Aeson.Types as Aeson
import qualified Data.ByteString.Lazy as LBS
import Data.Proxy
import qualified Data.Text as Text
import qualified Data.Text.Encoding as Text
class FromWith f where
fromWith :: f a -> a
instance FromWith (With x) where
fromWith (With a) = a
instance FromWith (WithEncoded x) where
fromWith (WithEncoded a) = a
fromWith2 :: (FromWith f, FromWith g) => f (g a) -> a
fromWith2 = fromWith . fromWith
{-# INLINE fromWith2 #-}
fromWith3 :: (FromWith f, FromWith g, FromWith h) => f (g (h a)) -> a
fromWith3 = fromWith . fromWith . fromWith
{-# INLINE fromWith3 #-}
data WithEncoded (fld :: Symbol) a where
WithEncoded :: forall fld a. KnownSymbol fld => a -> WithEncoded fld a
data With (fld :: Symbol) a where
With :: forall fld a. KnownSymbol fld => a -> With fld a
instance Show a => Show (WithEncoded fld a) where
show (WithEncoded a) = show a
instance Show a => Show (With fld a) where
show (With a) = show a
instance Eq a => Eq (WithEncoded fld a) where
(WithEncoded a) == (WithEncoded b) = a == b
instance Eq a => Eq (With fld a) where
(With a) == (With b) = a == b
instance Ord a => Ord (WithEncoded fld a) where
compare (WithEncoded a) (WithEncoded b) = compare a b
instance Ord a => Ord (With fld a) where
compare (With a) (With b) = compare a b
instance (KnownSymbol fld, FromJSON a) => FromJSON (WithEncoded fld a) where
parseJSON =
let name = symbolVal @fld Proxy
in withObject name $ \obj ->
WithEncoded <$> decodeEscaped obj (Text.pack name)
instance (KnownSymbol fld, ToJSON a) => ToJSON (WithEncoded fld a) where
toJSON (WithEncoded a) =
let name = Text.pack (symbolVal @fld Proxy)
in Aeson.object [ name .= (Text.decodeUtf8 . LBS.toStrict . encode) a ]
instance (KnownSymbol fld, FromJSON a) => FromJSON (With fld a) where
parseJSON =
let name = symbolVal @fld Proxy
in withObject name $ \obj ->
With <$> obj .: Text.pack name
instance (KnownSymbol fld, ToJSON a) => ToJSON (With fld a) where
toJSON (With a) =
let name = Text.pack (symbolVal @fld Proxy)
in Aeson.object [ name .= a ]
decodeEscaped :: FromJSON b => Aeson.Object -> Text -> Aeson.Parser b
decodeEscaped o t =
(o .: t) >>= (either fail pure . eitherDecodeStrict . Text.encodeUtf8)