{-# LANGUAGE DeriveFunctor #-} {-# LANGUAGE GADTs #-} {-# LANGUAGE GeneralizedNewtypeDeriving #-} {-# LANGUAGE MultiParamTypeClasses #-} {-# LANGUAGE OverloadedStrings #-} {-# LANGUAGE StandaloneDeriving #-} {-# LANGUAGE TypeFamilies #-} module Stratosphere.Values ( Val (..) , sub , ValList (..) , Integer' (..) , Bool' (..) , Double' (..) , ToRef (..) ) where import Data.Aeson import Data.HashMap.Strict (HashMap) import Data.String (IsString (..)) import Data.Text (Text) import GHC.Exts (IsList(..)) -- GADTs are cool, but I couldn't get this to work with FromJSON. Now that we -- don't have FromJSON though, we can revisit this. -- data Val a where -- Literal :: a -> Val a -- Ref :: Text -> Val a -- If :: Text -> Val a -> Val a -> Val a -- And :: Val Bool -> Val Bool -> Val Bool -- Equals :: (Show a, ToJSON a) => Val a -> Val a -> Val Bool -- Or :: Val Bool -> Val Bool -> Val Bool -- | This type is a wrapper around any values in a template. A value can be a -- 'Literal', a 'Ref', or an intrinsic function. See: -- http://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/intrinsic-function-reference.html data Val a = Literal a | Ref Text | If Text (Val a) (Val a) | And (Val Bool) (Val Bool) | Equals (Val a) (Val a) | Or (Val Bool) (Val Bool) | GetAtt Text Text | Base64 (Val Text) | Join Text (ValList Text) | Select Integer (ValList a) | FindInMap (Val a) (Val a) (Val a) -- ^ Map name, top level key, and second level key | ImportValue Text -- ^ The account-and-region-unique exported name of the value to import | Sub Text (Maybe (HashMap Text (Val Text))) -- ^ Substitution string and optional map of values deriving instance (Show a) => Show (Val a) deriving instance (Eq a) => Eq (Val a) deriving instance Functor Val instance (IsString a) => IsString (Val a) where fromString s = Literal (fromString s) instance (ToJSON a) => ToJSON (Val a) where toJSON (Literal v) = toJSON v toJSON (Ref r) = refToJSON r toJSON (If i x y) = mkFunc "Fn::If" [toJSON i, toJSON x, toJSON y] toJSON (And x y) = mkFunc "Fn::And" [toJSON (Bool' <$> x), toJSON (Bool' <$> y)] toJSON (Equals x y) = mkFunc "Fn::Equals" [toJSON x, toJSON y] toJSON (Or x y) = mkFunc "Fn::Or" [toJSON (Bool' <$> x), toJSON (Bool' <$> y)] toJSON (GetAtt x y) = mkFunc "Fn::GetAtt" [toJSON x, toJSON y] toJSON (Base64 v) = object [("Fn::Base64", toJSON v)] toJSON (Join d vs) = mkFunc "Fn::Join" [toJSON d, toJSON vs] toJSON (Select i vs) = mkFunc "Fn::Select" [toJSON (Integer' i), toJSON vs] toJSON (FindInMap mapName topKey secondKey) = object [("Fn::FindInMap", toJSON [toJSON mapName, toJSON topKey, toJSON secondKey])] toJSON (ImportValue t) = importValueToJSON t toJSON (Sub s Nothing) = object [("Fn::Sub", toJSON s)] toJSON (Sub s (Just vals)) = mkFunc "Fn::Sub" [toJSON s, Object (toJSON <$> vals)] -- | Simple version of 'Sub' without a map of values. sub :: Text -> Val Text sub s = Sub s Nothing refToJSON :: Text -> Value refToJSON ref = object [("Ref", toJSON ref)] importValueToJSON :: Text -> Value importValueToJSON ref = object [("Fn::ImportValue", toJSON ref)] mkFunc :: Text -> [Value] -> Value mkFunc name args = object [(name, Array $ fromList args)] -- | 'ValList' is like 'Val', except it is used in place of lists of Vals in -- templates. For example, if you have a parameter called @SubnetIds@ of type -- @List@ then, you can use @RefList "SubnetIds"@ to -- reference it. data ValList a = ValList [Val a] | RefList Text | ImportValueList Text | Split Text (Val a) | GetAZs (Val Text) deriving (Show, Eq) deriving instance Functor ValList instance IsList (ValList a) where type Item (ValList a) = Val a fromList = ValList toList (ValList xs) = xs -- This is obviously not meaningful, but the IsList instance is so useful -- that I decided to allow it. toList (RefList _) = [] toList (ImportValueList _) = [] toList (Split _ _) = [] toList (GetAZs _) = [] instance (ToJSON a) => ToJSON (ValList a) where toJSON (ValList vals) = toJSON vals toJSON (RefList ref) = refToJSON ref toJSON (ImportValueList ref) = importValueToJSON ref toJSON (Split d s) = mkFunc "Fn::Split" [toJSON d, toJSON s] toJSON (GetAZs r) = object [("Fn::GetAZs", toJSON r)] -- | We need to wrap integers so we can override the Aeson type-classes. This -- is necessary because CloudFront made the silly decision to represent numbers -- as JSON strings. newtype Integer' = Integer' { unInteger' :: Integer } deriving (Show, Eq, Num) instance ToJSON Integer' where toJSON (Integer' i) = toJSON $ show i -- | We need to wrap Bools for the same reason we need to wrap Ints. newtype Bool' = Bool' { unBool' :: Bool } deriving (Show, Bounded, Enum, Eq, Ord) instance ToJSON Bool' where toJSON (Bool' True) = "true" toJSON (Bool' False) = "false" -- | Class used to create a 'Ref' from another type. class ToRef a b where toRef :: a -> Val b -- | We need to wrap Doubles for the same reason we need to wrap Ints. newtype Double' = Double' { unDouble' :: Double } deriving (Show, Eq, Num) instance ToJSON Double' where toJSON (Double' i) = toJSON $ show i