2022-05-26 17:05:13 +03:00
|
|
|
-- | Build expectations for GraphQL field parsers. For now it focuses on updates
|
|
|
|
-- only.
|
|
|
|
--
|
|
|
|
-- See 'runUpdateFieldTest'.
|
|
|
|
module Test.Parser.Expectation
|
|
|
|
( UpdateTestSetup (..),
|
|
|
|
UpdateExpectationBuilder (..),
|
2022-07-18 18:15:34 +03:00
|
|
|
BackendUpdateBuilder (..),
|
|
|
|
MultiRowUpdateBuilder (..),
|
2022-05-26 17:05:13 +03:00
|
|
|
runUpdateFieldTest,
|
|
|
|
module I,
|
2022-08-06 00:40:41 +03:00
|
|
|
AnnotatedUpdateBuilder (..),
|
|
|
|
mkAnnotatedUpdate,
|
2022-08-11 13:45:39 +03:00
|
|
|
toBoolExp,
|
2022-05-26 17:05:13 +03:00
|
|
|
)
|
|
|
|
where
|
|
|
|
|
|
|
|
import Data.Bifunctor (bimap)
|
|
|
|
import Data.HashMap.Strict qualified as HM
|
|
|
|
import Hasura.Backends.Postgres.SQL.Types (QualifiedTable)
|
2022-07-18 18:15:34 +03:00
|
|
|
import Hasura.Backends.Postgres.Types.Update (BackendUpdate (..), MultiRowUpdate (..), UpdateOpExpression (..))
|
2022-05-26 17:05:13 +03:00
|
|
|
import Hasura.GraphQL.Parser.Internal.Parser (FieldParser (..))
|
2022-07-18 18:15:34 +03:00
|
|
|
import Hasura.GraphQL.Parser.Schema (Definition (..))
|
|
|
|
import Hasura.GraphQL.Parser.Variable (Variable (..))
|
2022-07-19 09:55:42 +03:00
|
|
|
import Hasura.GraphQL.Schema.NamingCase
|
2022-05-26 17:05:13 +03:00
|
|
|
import Hasura.Prelude
|
|
|
|
import Hasura.RQL.IR.BoolExp (AnnBoolExpFld (..), GBoolExp (..), OpExpG (..))
|
|
|
|
import Hasura.RQL.IR.Returning (MutationOutputG (..))
|
|
|
|
import Hasura.RQL.IR.Root (RemoteRelationshipField)
|
|
|
|
import Hasura.RQL.IR.Update (AnnotatedUpdateG (..))
|
2022-05-31 01:07:02 +03:00
|
|
|
import Hasura.RQL.IR.Value (UnpreparedValue)
|
2022-05-26 17:05:13 +03:00
|
|
|
import Hasura.RQL.Types.Column (ColumnInfo (..))
|
|
|
|
import Hasura.RQL.Types.Instances ()
|
|
|
|
import Hasura.SQL.Backend (BackendType (Postgres), PostgresKind (Vanilla))
|
|
|
|
import Language.GraphQL.Draft.Syntax qualified as Syntax
|
|
|
|
import Test.Hspec
|
|
|
|
import Test.Parser.Internal
|
2022-08-06 00:40:41 +03:00
|
|
|
import Test.Parser.Internal as I (ColumnInfoBuilder (..), mkColumnInfo, mkTable)
|
2022-05-26 17:05:13 +03:00
|
|
|
import Test.Parser.Monad
|
|
|
|
|
|
|
|
type PG = 'Postgres 'Vanilla
|
|
|
|
|
|
|
|
type BoolExp = GBoolExp PG (AnnBoolExpFld PG (UnpreparedValue PG))
|
|
|
|
|
2022-08-06 00:40:41 +03:00
|
|
|
type Output r = MutationOutputG PG r (UnpreparedValue PG)
|
2022-05-26 17:05:13 +03:00
|
|
|
|
|
|
|
type Field = Syntax.Field Syntax.NoFragments Variable
|
|
|
|
|
|
|
|
type Where = (ColumnInfoBuilder, [OpExpG PG (UnpreparedValue PG)])
|
|
|
|
|
2022-07-18 18:15:34 +03:00
|
|
|
type Update = BackendUpdateBuilder ColumnInfoBuilder
|
2022-05-26 17:05:13 +03:00
|
|
|
|
|
|
|
-- | Holds all the information required to setup and run a field parser update
|
|
|
|
-- test.
|
|
|
|
data UpdateTestSetup = UpdateTestSetup
|
|
|
|
{ -- | name of the table
|
|
|
|
utsTable :: Text,
|
|
|
|
-- | table columns
|
|
|
|
utsColumns :: [ColumnInfoBuilder],
|
|
|
|
-- | expectation
|
|
|
|
utsExpect :: UpdateExpectationBuilder,
|
|
|
|
-- | GrqphQL field, see Test.Parser.Parser
|
|
|
|
utsField :: Field
|
|
|
|
}
|
|
|
|
|
|
|
|
-- | Build the expected output columns, where and update clauses.
|
|
|
|
data UpdateExpectationBuilder = UpdateExpectationBuilder
|
|
|
|
{ -- | build the expected selection set/output, e.g.
|
|
|
|
--
|
|
|
|
-- > MOutMultirowFields [("affected_rows", MCount)]
|
2022-08-06 00:40:41 +03:00
|
|
|
utbOutput :: Output (RemoteRelationshipFieldWrapper UnpreparedValue),
|
2022-05-26 17:05:13 +03:00
|
|
|
-- | expected where condition(s), e.g. given a @nameColumn ::
|
|
|
|
-- ColumnInfoBuilder@ and @oldValue :: UnpreparedValue PG@:
|
|
|
|
--
|
|
|
|
-- > [(nameColumn, [AEQ true oldvalue])]
|
|
|
|
utbWhere :: [Where],
|
|
|
|
-- | expected update clause(s), e.g. given a @nameColumn ::
|
|
|
|
-- ColumnInfoBuilder@ and @newValue :: UnpreparedValue PG@:
|
|
|
|
--
|
|
|
|
-- > [(namecolumn, UpdateSet newValue)]
|
2022-07-18 18:15:34 +03:00
|
|
|
utbUpdate :: Update
|
2022-05-26 17:05:13 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
-- | Run a test given the schema and field.
|
|
|
|
runUpdateFieldTest :: UpdateTestSetup -> Expectation
|
|
|
|
runUpdateFieldTest UpdateTestSetup {..} =
|
2022-08-25 20:54:40 +03:00
|
|
|
case mkParser (TableInfoBuilder table utsColumns) of
|
2022-05-26 17:05:13 +03:00
|
|
|
SchemaTestT [] -> expectationFailure "expected at least one parser"
|
2022-07-18 18:15:34 +03:00
|
|
|
SchemaTestT parsers ->
|
|
|
|
case find (byName (Syntax._fName utsField)) parsers of
|
|
|
|
Nothing -> expectationFailure $ "could not find parser " <> show (Syntax._fName utsField)
|
|
|
|
Just FieldParser {..} ->
|
|
|
|
case fParser utsField of
|
|
|
|
ParserTestT (Right annUpdate) ->
|
|
|
|
coerce annUpdate `shouldBe` expected
|
|
|
|
ParserTestT (Left err) -> err
|
2022-05-26 17:05:13 +03:00
|
|
|
where
|
|
|
|
UpdateExpectationBuilder {..} = utsExpect
|
|
|
|
|
2022-07-18 18:15:34 +03:00
|
|
|
byName :: Syntax.Name -> Parser -> Bool
|
|
|
|
byName name FieldParser {..} = name == dName fDefinition
|
|
|
|
|
2022-05-26 17:05:13 +03:00
|
|
|
table :: QualifiedTable
|
|
|
|
table = mkTable utsTable
|
|
|
|
|
|
|
|
expected :: AnnotatedUpdateG PG (RemoteRelationshipFieldWrapper UnpreparedValue) (UnpreparedValue PG)
|
|
|
|
expected =
|
|
|
|
mkAnnotatedUpdate
|
|
|
|
AnnotatedUpdateBuilder
|
|
|
|
{ aubTable = table,
|
|
|
|
aubOutput = utbOutput,
|
|
|
|
aubColumns = mkColumnInfo <$> utsColumns,
|
|
|
|
aubWhere = first mkColumnInfo <$> utbWhere,
|
2022-07-18 18:15:34 +03:00
|
|
|
aubUpdate = mkUpdateColumns utbUpdate
|
2022-05-26 17:05:13 +03:00
|
|
|
}
|
2022-07-18 18:15:34 +03:00
|
|
|
mkUpdateColumns :: BackendUpdateBuilder ColumnInfoBuilder -> BackendUpdateBuilder (ColumnInfo PG)
|
|
|
|
mkUpdateColumns = fmap mkColumnInfo
|
2022-05-26 17:05:13 +03:00
|
|
|
|
|
|
|
-- | Internal use only. The intended use is through 'runUpdateFieldTest'.
|
|
|
|
--
|
|
|
|
-- Build an 'AnnotatedUpdateG', to be used with 'mkAnnotatedUpdate'.
|
2022-08-06 00:40:41 +03:00
|
|
|
data AnnotatedUpdateBuilder r = AnnotatedUpdateBuilder
|
2022-05-26 17:05:13 +03:00
|
|
|
{ -- | the main table for the update
|
|
|
|
aubTable :: QualifiedTable,
|
|
|
|
-- | the 'Output' clause, e.g., selection set, affected_rows, etc.
|
2022-08-06 00:40:41 +03:00
|
|
|
aubOutput :: Output r,
|
2022-05-26 17:05:13 +03:00
|
|
|
-- | the table columns (all of them)
|
|
|
|
aubColumns :: [ColumnInfo PG],
|
|
|
|
-- | the where clause(s)
|
|
|
|
aubWhere :: [(ColumnInfo PG, [OpExpG PG (UnpreparedValue PG)])],
|
|
|
|
-- | the update statement(s)
|
2022-07-18 18:15:34 +03:00
|
|
|
aubUpdate :: BackendUpdateBuilder (ColumnInfo PG)
|
2022-05-26 17:05:13 +03:00
|
|
|
}
|
|
|
|
|
2022-07-18 18:15:34 +03:00
|
|
|
data BackendUpdateBuilder col
|
|
|
|
= UpdateTable [(col, UpdateOpExpression (UnpreparedValue PG))]
|
|
|
|
| UpdateMany [MultiRowUpdateBuilder col]
|
|
|
|
deriving stock (Functor)
|
|
|
|
|
|
|
|
data MultiRowUpdateBuilder col = MultiRowUpdateBuilder
|
|
|
|
{ mrubWhere :: [(col, [OpExpG PG (UnpreparedValue PG)])],
|
|
|
|
mrubUpdate :: [(col, UpdateOpExpression (UnpreparedValue PG))]
|
|
|
|
}
|
|
|
|
deriving stock (Functor)
|
|
|
|
|
2022-05-26 17:05:13 +03:00
|
|
|
-- | 'RemoteRelationshipField' cannot have Eq/Show instances, so we're wrapping
|
|
|
|
-- it.
|
|
|
|
newtype RemoteRelationshipFieldWrapper vf = RemoteRelationshipFieldWrapper (RemoteRelationshipField vf)
|
|
|
|
|
|
|
|
instance Show (RemoteRelationshipFieldWrapper vf) where
|
|
|
|
show =
|
|
|
|
error "Test.Parser.Expectation: no Show implementation for RemoteRelationshipFieldWrapper"
|
|
|
|
|
|
|
|
instance Eq (RemoteRelationshipFieldWrapper vf) where
|
|
|
|
(==) =
|
|
|
|
error "Test.Parser.Expectation: no Eq implementation for RemoteRelationshipFieldWrapper"
|
|
|
|
|
|
|
|
-- | Internal use, see 'runUpdateFieldTest'.
|
|
|
|
mkAnnotatedUpdate ::
|
2022-08-06 00:40:41 +03:00
|
|
|
forall r.
|
|
|
|
AnnotatedUpdateBuilder r ->
|
|
|
|
AnnotatedUpdateG PG r (UnpreparedValue PG)
|
2022-05-26 17:05:13 +03:00
|
|
|
mkAnnotatedUpdate AnnotatedUpdateBuilder {..} = AnnotatedUpdateG {..}
|
|
|
|
where
|
|
|
|
_auTable :: QualifiedTable
|
|
|
|
_auTable = aubTable
|
|
|
|
|
|
|
|
_auWhere :: (BoolExp, BoolExp)
|
2022-08-11 13:45:39 +03:00
|
|
|
_auWhere = (column, toBoolExp aubWhere)
|
2022-05-26 17:05:13 +03:00
|
|
|
|
|
|
|
_auCheck :: BoolExp
|
|
|
|
_auCheck = BoolAnd []
|
|
|
|
|
2022-07-18 18:15:34 +03:00
|
|
|
_auBackend :: BackendUpdate 'Vanilla (UnpreparedValue PG)
|
2022-05-26 17:05:13 +03:00
|
|
|
_auBackend =
|
2022-07-18 18:15:34 +03:00
|
|
|
case aubUpdate of
|
|
|
|
UpdateTable items ->
|
|
|
|
BackendUpdate $
|
|
|
|
HM.fromList $
|
|
|
|
fmap (first ciColumn) items
|
|
|
|
UpdateMany rows ->
|
|
|
|
BackendMultiRowUpdate $ fmap mapRows rows
|
|
|
|
|
|
|
|
mapRows :: MultiRowUpdateBuilder (ColumnInfo PG) -> MultiRowUpdate 'Vanilla (UnpreparedValue PG)
|
|
|
|
mapRows MultiRowUpdateBuilder {..} =
|
|
|
|
MultiRowUpdate
|
|
|
|
{ mruWhere = toBoolExp mrubWhere,
|
|
|
|
mruExpression = HM.fromList $ fmap (bimap ciColumn id) mrubUpdate
|
2022-05-26 17:05:13 +03:00
|
|
|
}
|
|
|
|
|
2022-08-06 00:40:41 +03:00
|
|
|
_auOutput :: Output r
|
2022-05-26 17:05:13 +03:00
|
|
|
_auOutput = aubOutput
|
|
|
|
|
|
|
|
_auAllCols :: [ColumnInfo PG]
|
|
|
|
_auAllCols = aubColumns
|
|
|
|
|
2022-08-11 13:45:39 +03:00
|
|
|
column :: BoolExp
|
|
|
|
column =
|
2022-05-26 17:05:13 +03:00
|
|
|
BoolAnd
|
2022-08-11 13:45:39 +03:00
|
|
|
. fmap (\c -> BoolField . AVColumn c $ [])
|
2022-05-26 17:05:13 +03:00
|
|
|
$ aubColumns
|
2022-07-19 09:55:42 +03:00
|
|
|
|
|
|
|
_auNamingConvention :: Maybe NamingCase
|
|
|
|
_auNamingConvention = Just HasuraCase
|
2022-08-11 13:45:39 +03:00
|
|
|
|
|
|
|
toBoolExp :: [(ColumnInfo PG, [OpExpG PG (UnpreparedValue PG)])] -> BoolExp
|
|
|
|
toBoolExp = BoolAnd . fmap (\(c, ops) -> BoolField $ AVColumn c ops)
|