mirror of
https://github.com/hasura/graphql-engine.git
synced 2024-12-15 09:22:43 +03:00
Reverse stringifyNumbers
assertions
PR-URL: https://github.com/hasura/graphql-engine-mono/pull/6079 GitOrigin-RevId: 2db16d057f5ff2ea261d23b43eeced22ebd0235a
This commit is contained in:
parent
fabbd134b2
commit
a9f7c1ef80
@ -16,13 +16,13 @@ import Data.Aeson
|
|||||||
)
|
)
|
||||||
import Data.Aeson qualified as Aeson
|
import Data.Aeson qualified as Aeson
|
||||||
import Data.Aeson.KeyMap qualified as KM
|
import Data.Aeson.KeyMap qualified as KM
|
||||||
import Data.Aeson.KeyMap.Extended qualified as KM
|
import Data.Aeson.KeyMap.Extended qualified as KM (mapWithKey)
|
||||||
import Data.Aeson.Text qualified as Aeson.Text
|
|
||||||
import Data.List (permutations)
|
import Data.List (permutations)
|
||||||
|
import Data.Set (Set)
|
||||||
|
import Data.Set qualified as Set
|
||||||
import Data.Text qualified as T
|
import Data.Text qualified as T
|
||||||
import Data.Text.Encoding (decodeUtf8With)
|
import Data.Text.Encoding (decodeUtf8With)
|
||||||
import Data.Text.Encoding.Error qualified as TE
|
import Data.Text.Encoding.Error qualified as TE
|
||||||
import Data.Text.Lazy qualified as LT
|
|
||||||
import Data.Vector qualified as V
|
import Data.Vector qualified as V
|
||||||
import Data.Vector qualified as Vector
|
import Data.Vector qualified as Vector
|
||||||
import Data.Yaml qualified
|
import Data.Yaml qualified
|
||||||
@ -66,6 +66,43 @@ combinationsObjectUsingValue fn variants = combinationsObject fn (map fromObject
|
|||||||
shouldReturnYaml :: HasCallStack => Fixture.Options -> IO Value -> Value -> IO ()
|
shouldReturnYaml :: HasCallStack => Fixture.Options -> IO Value -> Value -> IO ()
|
||||||
shouldReturnYaml = shouldReturnYamlF pure
|
shouldReturnYaml = shouldReturnYamlF pure
|
||||||
|
|
||||||
|
-- | Because JSON supports numbers only up to 32 bits, some backends (such as
|
||||||
|
-- BigQuery) send numbers as strings instead. So, for example, the floating
|
||||||
|
-- point @2.0@ will reach us as @'2.0'@.
|
||||||
|
--
|
||||||
|
-- This presents an issue when we want to write tests across multiple backends:
|
||||||
|
-- what do we do if a test should match @2.0@ for Postgres and @'2.0'@ for
|
||||||
|
-- BigQuery? Worse still, what if it should match @'2.0'@ for BigQuery and
|
||||||
|
-- @'2.00000000'@ for CockroachDB?
|
||||||
|
--
|
||||||
|
-- This function attempts to solve the problem by zipping the expected output
|
||||||
|
-- and the actual output together, looking for any instance where we expect a
|
||||||
|
-- number, but find a string. In these instances, we replace the string with
|
||||||
|
-- the result of parsing that string _into_ a number - specifically, a
|
||||||
|
-- @Scientific@. This works because @Scientific@ can support 64-bit numbers
|
||||||
|
-- (and any arbitrary precision, for that matter), so it should be able to
|
||||||
|
-- handle any stringified number we receive.
|
||||||
|
--
|
||||||
|
-- If the zipping doesn't line up, we assume this is probably a bad result and
|
||||||
|
-- consequently should result in a failing test. In these cases, we leave the
|
||||||
|
-- actual output exactly as-is, and wait for the test to fail.
|
||||||
|
parseToMatch :: Value -> Value -> Value
|
||||||
|
parseToMatch (Array expected) (Array actual) =
|
||||||
|
Array (Vector.zipWith parseToMatch expected actual)
|
||||||
|
parseToMatch (Number _) (String text) =
|
||||||
|
case readMaybe (T.unpack text) of
|
||||||
|
Just actual -> Number actual
|
||||||
|
Nothing -> String text
|
||||||
|
parseToMatch (Object expected) (Object actual) = do
|
||||||
|
let walk :: KM.KeyMap Value -> Aeson.Key -> Value -> Value
|
||||||
|
walk reference key current =
|
||||||
|
case KM.lookup key reference of
|
||||||
|
Just this -> parseToMatch this current
|
||||||
|
Nothing -> current
|
||||||
|
|
||||||
|
Object (KM.mapWithKey (walk expected) actual)
|
||||||
|
parseToMatch _ actual = actual
|
||||||
|
|
||||||
-- | The function @transform@ converts the returned YAML
|
-- | The function @transform@ converts the returned YAML
|
||||||
-- prior to comparison. It exists in IO in order to be able
|
-- prior to comparison. It exists in IO in order to be able
|
||||||
-- to easily throw exceptions for hspec purposes.
|
-- to easily throw exceptions for hspec purposes.
|
||||||
@ -76,32 +113,15 @@ shouldReturnYaml = shouldReturnYamlF pure
|
|||||||
-- We use 'Visual' internally to easily display the 'Value' as YAML
|
-- We use 'Visual' internally to easily display the 'Value' as YAML
|
||||||
-- when the test suite uses its 'Show' instance.
|
-- when the test suite uses its 'Show' instance.
|
||||||
shouldReturnYamlF :: HasCallStack => (Value -> IO Value) -> Fixture.Options -> IO Value -> Value -> IO ()
|
shouldReturnYamlF :: HasCallStack => (Value -> IO Value) -> Fixture.Options -> IO Value -> Value -> IO ()
|
||||||
shouldReturnYamlF transform options actualIO rawExpected = do
|
shouldReturnYamlF transform options actualIO expected = do
|
||||||
actual <- transform =<< actualIO
|
actual <-
|
||||||
|
actualIO >>= transform >>= \actual ->
|
||||||
|
pure
|
||||||
|
if Fixture.stringifyNumbers options
|
||||||
|
then parseToMatch expected actual
|
||||||
|
else actual
|
||||||
|
|
||||||
let Fixture.Options {stringifyNumbers} = options
|
actual `shouldBe` expected
|
||||||
expected' =
|
|
||||||
if stringifyNumbers
|
|
||||||
then stringifyExpectedToActual rawExpected actual
|
|
||||||
else rawExpected
|
|
||||||
|
|
||||||
expected <- transform expected'
|
|
||||||
|
|
||||||
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 km) (Aeson.Object km') =
|
|
||||||
let stringifyKV k v =
|
|
||||||
case KM.lookup k km' of
|
|
||||||
Just v' -> stringifyExpectedToActual v v'
|
|
||||||
Nothing -> v
|
|
||||||
in Aeson.Object (KM.mapWithKey stringifyKV km)
|
|
||||||
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,
|
-- | The action @actualIO@ should produce the @expected@ YAML,
|
||||||
-- represented (by the yaml package) as an aeson 'Value'.
|
-- represented (by the yaml package) as an aeson 'Value'.
|
||||||
@ -109,16 +129,20 @@ stringifyExpectedToActual expected _ = expected
|
|||||||
-- We use 'Visual' internally to easily display the 'Value' as YAML
|
-- We use 'Visual' internally to easily display the 'Value' as YAML
|
||||||
-- when the test suite uses its 'Show' instance.
|
-- when the test suite uses its 'Show' instance.
|
||||||
shouldReturnOneOfYaml :: HasCallStack => Fixture.Options -> IO Value -> [Value] -> IO ()
|
shouldReturnOneOfYaml :: HasCallStack => Fixture.Options -> IO Value -> [Value] -> IO ()
|
||||||
shouldReturnOneOfYaml options actualIO expecteds = do
|
shouldReturnOneOfYaml Fixture.Options {stringifyNumbers} actualIO candidates = do
|
||||||
actual <- actualIO
|
actual <- actualIO
|
||||||
|
|
||||||
let Fixture.Options {stringifyNumbers} = options
|
let expecteds :: Set Value
|
||||||
fixNumbers expected =
|
expecteds = Set.fromList candidates
|
||||||
if stringifyNumbers
|
|
||||||
then stringifyExpectedToActual expected actual
|
|
||||||
else expected
|
|
||||||
|
|
||||||
shouldContain (map (Visual . fixNumbers) expecteds) [Visual actual]
|
actuals :: Set Value
|
||||||
|
actuals
|
||||||
|
| stringifyNumbers = Set.map (`parseToMatch` actual) expecteds
|
||||||
|
| otherwise = Set.singleton actual
|
||||||
|
|
||||||
|
case Set.lookupMin (Set.intersection expecteds actuals) of
|
||||||
|
Just match -> Visual match `shouldBe` Visual actual
|
||||||
|
Nothing -> map Visual (Set.toList expecteds) `shouldContain` [Visual actual]
|
||||||
|
|
||||||
-- | We use 'Visual' internally to easily display the 'Value' as YAML
|
-- | We use 'Visual' internally to easily display the 'Value' as YAML
|
||||||
-- when the test suite uses its 'Show' instance.
|
-- when the test suite uses its 'Show' instance.
|
||||||
|
@ -81,8 +81,8 @@ schemaInspectionTests opts = describe "Schema and Source Inspection" $ do
|
|||||||
|]
|
|]
|
||||||
)
|
)
|
||||||
[yaml|
|
[yaml|
|
||||||
- - Artist
|
|
||||||
- - Album
|
- - Album
|
||||||
|
- - Artist
|
||||||
- - Customer
|
- - Customer
|
||||||
- - Employee
|
- - Employee
|
||||||
- - Genre
|
- - Genre
|
||||||
|
@ -58,8 +58,7 @@ spec = do
|
|||||||
Fixture.customOptions =
|
Fixture.customOptions =
|
||||||
Just $
|
Just $
|
||||||
Fixture.defaultOptions
|
Fixture.defaultOptions
|
||||||
{ Fixture.stringifyNumbers = True,
|
{ Fixture.stringifyNumbers = True
|
||||||
Fixture.skipTests = Just "Skipping whilst we fix numerical comparison of stringified results in Hspec tests"
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
(Fixture.fixture $ Fixture.Backend Fixture.SQLServer)
|
(Fixture.fixture $ Fixture.Backend Fixture.SQLServer)
|
||||||
|
Loading…
Reference in New Issue
Block a user