{-# LANGUAGE DeriveGeneric #-} {-# LANGUAGE GeneralizedNewtypeDeriving #-} {-# LANGUAGE OverloadedStrings #-} -- | Several newtypes and combinators for dealing with less-than-cleanly JSON input. module Data.Aeson.Filthy ( -- * Double-Encodings JSONString(..) , (.:$) , (.=$) -- * Booleans -- $booleans , OneOrZero(..) , YesOrNo(..) , OnOrOff(..) , AnyBool(..) -- * Maybe , EmptyAsNothing(..) -- * Case Insensitive Keys , (.:~) ) where import Control.Applicative (Alternative (..)) import Control.Monad (MonadPlus) import Control.Monad.Fix (MonadFix) import Data.Aeson import Data.Aeson.Types (Pair, Parser) import Data.Bits (Bits, FiniteBits) import qualified Data.ByteString.Lazy as BL import qualified Data.HashMap.Lazy as HM import Data.Ix (Ix) import Data.String (IsString) import Data.Text (Text) import qualified Data.Text as T import qualified Data.Text.Encoding as T import Foreign.Storable (Storable) import GHC.Generics (Generic, Generic1) -- | A double-encoded JSON value. -- -- >>> encode (JSONString True) -- "\"true\"" -- -- >>> decode "\"true\"" :: Maybe (JSONString Bool) -- Just (JSONString {jsonString = True}) newtype JSONString a = JSONString { jsonString :: a } deriving (Bounded, Enum, Eq, Ord, Read, Show, Ix, Generic, FiniteBits, Bits, Storable, Num, Integral, Real, Floating, Fractional, RealFrac, RealFloat, IsString) instance ToJSON a => ToJSON (JSONString a) where toJSON = String . T.decodeUtf8 . BL.toStrict . encode . jsonString instance FromJSON a => FromJSON (JSONString a) where parseJSON = withText "a double-encoded json value (JSONString)" (maybe (error "couldn't decode string") return . evil) where evil = fmap JSONString . decodeStrict . T.encodeUtf8 -- | Works like aeson's ('.:'), but assumes the value being parsed is double-encoded. Mnemonic: @$@ -- sorta looks like an "S" (for "String"). (.:$) :: FromJSON a => Object -> Text -> Parser a o .:$ t = jsonString <$> o .: t -- | Works like aeson's ('.='), but double-encodes the value being serialized. (.=$) :: ToJSON a => Text -> a -> Pair n .=$ o = n .= JSONString o -- $booleans There's a surprising number of ways people like to encode Booleans. At present, the -- docs below lie a bit in that values which don't parse to a 'True' value are considered false. -- For instance, -- -- >>> oneOrZero <$> decode "0" -- Just False -- -- >>> oneOrZero <$> decode "1" -- Just True -- -- >>> oneOrZero <$> decode "2" -- Just False -- | 'Bool's rendered "yes" or "no" -- -- >>> yesOrNo <$> decode "\"yes\"" -- Just True -- -- >>> yesOrNo <$> decode "\"no\"" -- Just False newtype YesOrNo = YesOrNo { yesOrNo :: Bool } deriving (Bounded, Enum, Eq, Ord, Read, Show, Ix, Generic, FiniteBits, Bits, Storable) -- | 'Bool's rendered "on" or "off" -- -- >>> onOrOff <$> decode "\"on\"" -- Just True -- -- >>> onOrOff <$> decode "\"off\"" -- Just False newtype OnOrOff = OnOrOff { onOrOff :: Bool } deriving (Bounded, Enum, Eq, Ord, Read, Show, Ix, Generic, FiniteBits, Bits, Storable) -- | 'Bool's rendered 0 or 1 -- -- >>> oneOrZero <$> decode "1" -- Just True -- -- >>> oneOrZero <$> decode "0" -- Just False newtype OneOrZero = OneOrZero { oneOrZero :: Bool } deriving (Bounded, Enum, Eq, Ord, Read, Show, Ix, Generic, FiniteBits, Bits, Storable) -- | 'Bool's rendered as more-or-less anything. -- -- >>> let Just bs = decode "[1, \"1\", \"true\", \"yes\", \"on\", true]" -- >>> and $ map anyBool bs -- True newtype AnyBool = AnyBool { anyBool :: Bool } deriving (Bounded, Enum, Eq, Ord, Read, Show, Ix, Generic, FiniteBits, Bits, Storable) instance ToJSON YesOrNo where toJSON (YesOrNo True) = "yes" toJSON _ = "no" instance FromJSON YesOrNo where parseJSON "yes" = pure $ YesOrNo True parseJSON _ = pure $ YesOrNo False instance ToJSON OnOrOff where toJSON (OnOrOff True) = "on" toJSON _ = "off" instance FromJSON OnOrOff where parseJSON "on" = pure $ OnOrOff True parseJSON _ = pure $ OnOrOff False instance ToJSON OneOrZero where toJSON (OneOrZero True) = Number 1 toJSON _ = Number 0 instance FromJSON OneOrZero where parseJSON (Number 1) = pure $ OneOrZero True parseJSON _ = pure $ OneOrZero False instance FromJSON AnyBool where parseJSON (Number 1) = pure $ AnyBool True parseJSON (String "1") = pure $ AnyBool True parseJSON (String "true") = pure $ AnyBool True parseJSON (String "yes") = pure $ AnyBool True parseJSON (String "on") = pure $ AnyBool True parseJSON (Bool b) = pure $ AnyBool b parseJSON _ = pure $ AnyBool False -- | Sometimes an empty string in a JSON object actually means 'Nothing' -- -- >>> emptyAsNothing <$> decode "\"\"" :: Maybe (Maybe Text) -- Just Nothing -- -- >>> emptyAsNothing <$> decode "\"something\"" :: Maybe (Maybe Text) -- Just (Just "something") newtype EmptyAsNothing a = EmptyAsNothing { emptyAsNothing :: Maybe a} deriving (Eq, Ord, Read, Show, Functor, Applicative, Alternative, Monad, MonadPlus, Foldable, Monoid, MonadFix, Generic, Generic1) instance Traversable EmptyAsNothing where traverse f = fmap EmptyAsNothing . traverse f . emptyAsNothing instance ToJSON a => ToJSON (EmptyAsNothing a) where toJSON = maybe "" toJSON . emptyAsNothing instance FromJSON a => FromJSON (EmptyAsNothing a) where parseJSON "" = pure $ EmptyAsNothing Nothing parseJSON x = EmptyAsNothing <$> parseJSON x -- | Some systems attempt to treat keys in JSON objects case-insensitively(ish). Golang's JSON -- marshalling is a prominent example: . The -- ('.:~') combinator works like ('.:'), but if it fails to match, attempts to find a -- case-insensitive variant of the key being sought. If there is an exact match, ('.:~') will -- take that; if there are multiple non-exact matches, the choice of selected value is -- unspecified. Mnemonic: @~@ swaps case in vi. -- -- >>> data Foo = Foo Int deriving (Read, Show) -- >>> instance FromJSON Foo where parseJSON (Object o) = Foo <$> o .:~ "foo" -- >>> decode "{\"FOO\": 12}" :: Maybe Foo -- Just (Foo 12) -- >>> decode "{\"foo\": 17, \"FOO\": 12}" :: Maybe Foo -- Just (Foo 17) (.:~) :: FromJSON a => Object -> Text -> Parser a o .:~ key = o .: key <|> maybe empty parseJSON go where go = lookup (T.toLower key) [(T.toLower k, v) | (k,v) <- HM.toList o]