graphql-engine/server/tests-hspec/Harness/Quoter/Yaml.hs
2022-02-21 17:06:04 +00:00

193 lines
6.2 KiB
Haskell

{-# LANGUAGE DeriveAnyClass #-}
{-# LANGUAGE LambdaCase #-}
{-# LANGUAGE NamedFieldPuns #-}
{-# LANGUAGE ScopedTypeVariables #-}
{-# OPTIONS_GHC -fno-warn-orphans #-}
-- | Templating yaml files.
module Harness.Quoter.Yaml
( yaml,
shouldReturnYaml,
shouldReturnOneOfYaml,
shouldBeYaml,
)
where
import Control.Exception.Safe (Exception, impureThrow, throwM)
import Control.Monad.Trans.Resource (ResourceT)
import Data.Aeson (Value)
import Data.Aeson qualified
import Data.Aeson qualified as Aeson
import Data.Aeson.Text qualified as Aeson.Text
import Data.ByteString.Char8 qualified as BS8
import Data.Conduit (runConduitRes, (.|))
import Data.Conduit.List qualified as CL
import Data.HashMap.Strict qualified as HashMap
import Data.Text.Encoding.Error qualified as T
import Data.Text.Lazy qualified as LT
import Data.Vector qualified as Vector
import Data.Yaml qualified
import Data.Yaml.Internal qualified
import Harness.Test.Context qualified as Context (Options (..))
import Instances.TH.Lift ()
import Language.Haskell.TH (Exp, Q, listE, mkName, runIO, varE)
import Language.Haskell.TH.Lift (Lift)
import Language.Haskell.TH.Lift qualified as TH
import Language.Haskell.TH.Quote (QuasiQuoter (..))
import System.IO.Unsafe (unsafePerformIO)
import Test.Hspec (HasCallStack, shouldBe, shouldContain)
import Text.Libyaml qualified as Libyaml
import Prelude
-------------------------------------------------------------------
-- * Expectations
-- | The action @actualIO@ should produce the @expected@ YAML,
-- represented (by the yaml package) as an aeson 'Value'.
--
-- We use 'Visual' internally to easily display the 'Value' as YAML
-- when the test suite uses its 'Show' instance.
shouldReturnYaml :: HasCallStack => Context.Options -> IO Value -> Value -> IO ()
shouldReturnYaml options actualIO rawExpected = do
actual <- actualIO
let Context.Options {stringifyNumbers} = options
expected =
if stringifyNumbers
then stringifyExpectedToActual rawExpected actual
else rawExpected
shouldBeYaml actual expected
-- | TODO(jkachmar): Document.
stringifyExpectedToActual :: Value -> Value -> Value
stringifyExpectedToActual (Aeson.Number n) (Aeson.String _) =
Aeson.String (LT.toStrict . Aeson.Text.encodeToLazyText $ n)
stringifyExpectedToActual (Aeson.Object hm) (Aeson.Object hm') =
let stringifyKV k v =
case HashMap.lookup k hm' of
Just v' -> stringifyExpectedToActual v v'
Nothing -> v
in Aeson.Object (HashMap.mapWithKey stringifyKV hm)
stringifyExpectedToActual (Aeson.Array as) (Aeson.Array bs) =
Aeson.Array (Vector.zipWith stringifyExpectedToActual as bs)
stringifyExpectedToActual expected _ = expected
-- | The action @actualIO@ should produce the @expected@ YAML,
-- represented (by the yaml package) as an aeson 'Value'.
--
-- We use 'Visual' internally to easily display the 'Value' as YAML
-- when the test suite uses its 'Show' instance.
shouldReturnOneOfYaml :: HasCallStack => Context.Options -> IO Value -> [Value] -> IO ()
shouldReturnOneOfYaml options actualIO expecteds = do
actual <- actualIO
let Context.Options {stringifyNumbers} = options
fixNumbers expected =
if stringifyNumbers
then stringifyExpectedToActual expected actual
else expected
shouldContain (map (Visual . fixNumbers) expecteds) [Visual actual]
-- | We use 'Visual' internally to easily display the 'Value' as YAML
-- when the test suite uses its 'Show' instance.
--
-- NOTE(jkachmar): A lot of the matchers we define in this module are
-- implemented in the @hspec-expectations-json@ package.
--
-- Since @Data.Yaml@ uses the same underlying 'Value' type as
-- @Data.Aeson@, we could pull that in as a dependency and alias
-- some of these functions accordingly.
shouldBeYaml :: HasCallStack => Value -> Value -> IO ()
shouldBeYaml actual expected = do
shouldBe (Visual actual) (Visual expected)
-------------------------------------------------------------------
-- * Quasi quoters
yaml :: QuasiQuoter
yaml =
QuasiQuoter
{ quoteExp = templateYaml,
quotePat = \_ -> fail "invalid",
quoteType = \_ -> fail "invalid",
quoteDec = \_ -> fail "invalid"
}
-- | Template a YAML file contents. Throws a bunch of exception types:
-- 'YamlTemplateException' or 'YamlException' or 'ParseException'.
--
-- Produces 'Value'.
templateYaml :: String -> Q Exp
templateYaml inputString = do
events <-
runIO
( runConduitRes
(Libyaml.decode inputBytes .| CL.mapM processor .| CL.consume)
)
[|
unsafePerformIO
( do
bs <-
runConduitRes
(CL.sourceList (concat $(listE events)) .| Libyaml.encode)
case Data.Yaml.decodeEither' bs of
Left e -> impureThrow e
Right (v :: Aeson.Value) -> pure v
)
|]
where
inputBytes = BS8.pack inputString
-- | Process the events as they come in, potentially expanding any
-- aliases to objects.
processor :: Libyaml.Event -> ResourceT IO (Q Exp)
processor =
\case
Libyaml.EventAlias anchorName ->
pure
[|
Data.Yaml.Internal.objToEvents
Data.Yaml.Internal.defaultStringStyle
(Data.Aeson.toJSON $(varE (mkName anchorName)))
[]
|]
-- We disable anchors because aliases are used only to refer to
-- Haskell variables, not YAML anchors.
(Libyaml.EventScalar _ _ _ (Just {})) -> throwM AnchorsAreDisabled
(Libyaml.EventSequenceStart _ _ (Just {})) -> throwM AnchorsAreDisabled
(Libyaml.EventMappingStart _ _ (Just {})) -> throwM AnchorsAreDisabled
event -> pure (TH.lift [event])
-------------------------------------------------------------------
-- * YAML types
-- | Exceptions that will be thrown mercilessly.
data YamlTemplateException
= AnchorsAreDisabled
| YamlEncodingProblem T.UnicodeException
deriving stock (Show)
deriving anyclass (Exception)
deriving instance Lift Libyaml.Event
deriving instance Lift Libyaml.Style
deriving instance Lift Libyaml.Tag
deriving instance Lift Libyaml.SequenceStyle
deriving instance Lift Libyaml.MappingStyle
-- | For the test suite: diff structural, but display in a readable
-- way.
newtype Visual = Visual {unVisual :: Value}
deriving stock (Eq)
instance Show Visual where
show = BS8.unpack . Data.Yaml.encode . unVisual