2023-03-31 18:33:32 +03:00
|
|
|
{-# HLINT ignore "avoid Language.GraphQL.Draft.Syntax.unsafeMkName" #-}
|
2023-01-31 15:52:26 +03:00
|
|
|
{-# LANGUAGE BlockArguments #-}
|
|
|
|
{-# LANGUAGE OverloadedStrings #-}
|
2023-03-13 21:13:56 +03:00
|
|
|
{-# OPTIONS_GHC -Wno-incomplete-uni-patterns #-}
|
|
|
|
{-# OPTIONS_GHC -Wno-unrecognised-pragmas #-}
|
2023-01-31 15:52:26 +03:00
|
|
|
|
2023-04-13 19:10:38 +03:00
|
|
|
module Hasura.Backends.Postgres.NativeQueries.NativeQueriesSpec (spec) where
|
2023-01-31 15:52:26 +03:00
|
|
|
|
2023-03-13 21:13:56 +03:00
|
|
|
import Data.Bifunctor
|
|
|
|
import Data.Either
|
2023-04-26 18:42:13 +03:00
|
|
|
import Data.HashMap.Strict qualified as HashMap
|
2023-04-13 19:10:38 +03:00
|
|
|
import Hasura.Backends.Postgres.Instances.NativeQueries
|
2023-03-13 21:13:56 +03:00
|
|
|
import Hasura.Backends.Postgres.SQL.Types
|
|
|
|
import Hasura.Base.Error
|
2023-04-19 12:03:36 +03:00
|
|
|
import Hasura.LogicalModel.Metadata
|
2023-04-13 19:10:38 +03:00
|
|
|
import Hasura.NativeQuery.Metadata
|
2023-03-13 21:13:56 +03:00
|
|
|
import Hasura.Prelude hiding (first)
|
|
|
|
import Language.GraphQL.Draft.Syntax qualified as G
|
|
|
|
import Test.Hspec (Spec, describe, it, shouldBe, shouldSatisfy)
|
2023-01-31 15:52:26 +03:00
|
|
|
|
|
|
|
spec :: Spec
|
|
|
|
spec = do
|
2023-02-03 14:15:08 +03:00
|
|
|
describe "Parses raw query into InterpolatedQuery" do
|
2023-01-31 15:52:26 +03:00
|
|
|
it "Parses SQL with no parameters in it" $ do
|
|
|
|
let rawSQL = "SELECT * FROM dogs"
|
|
|
|
parseInterpolatedQuery rawSQL `shouldBe` Right (InterpolatedQuery [IIText "SELECT * FROM dogs"])
|
|
|
|
|
|
|
|
it "Parses only a variable" $ do
|
|
|
|
let rawSQL = "{{dogs}}"
|
2023-04-13 19:10:38 +03:00
|
|
|
parseInterpolatedQuery rawSQL `shouldBe` Right (InterpolatedQuery [IIVariable (NativeQueryArgumentName "dogs")])
|
2023-01-31 15:52:26 +03:00
|
|
|
|
|
|
|
it "Parses SQL with one parameter in it" $ do
|
|
|
|
let rawSQL = "SELECT * FROM dogs WHERE name = {{name}}"
|
|
|
|
parseInterpolatedQuery rawSQL
|
|
|
|
`shouldBe` Right
|
|
|
|
( InterpolatedQuery
|
|
|
|
[ IIText "SELECT * FROM dogs WHERE name = ",
|
2023-04-13 19:10:38 +03:00
|
|
|
IIVariable (NativeQueryArgumentName "name")
|
2023-01-31 15:52:26 +03:00
|
|
|
]
|
|
|
|
)
|
|
|
|
|
|
|
|
it "Parses SQL with one parameter in the middle of the string" $ do
|
|
|
|
let rawSQL = "SELECT * FROM dogs WHERE {{name}} = name"
|
|
|
|
parseInterpolatedQuery rawSQL
|
|
|
|
`shouldBe` Right
|
|
|
|
( InterpolatedQuery
|
|
|
|
[ IIText "SELECT * FROM dogs WHERE ",
|
2023-04-13 19:10:38 +03:00
|
|
|
IIVariable (NativeQueryArgumentName "name"),
|
2023-01-31 15:52:26 +03:00
|
|
|
IIText " = name"
|
|
|
|
]
|
|
|
|
)
|
|
|
|
|
|
|
|
it "Parses SQL with one parameter and a loose { hanging around" $ do
|
|
|
|
let rawSQL = "SELECT * FROM dogs WHERE {{name}} = '{doggy friend}'"
|
|
|
|
parseInterpolatedQuery rawSQL
|
|
|
|
`shouldBe` Right
|
|
|
|
( InterpolatedQuery
|
|
|
|
[ IIText "SELECT * FROM dogs WHERE ",
|
2023-04-13 19:10:38 +03:00
|
|
|
IIVariable (NativeQueryArgumentName "name"),
|
2023-01-31 15:52:26 +03:00
|
|
|
IIText " = '{doggy friend}'"
|
|
|
|
]
|
|
|
|
)
|
|
|
|
|
|
|
|
it "What should happen for unclosed variable" $ do
|
|
|
|
let rawSQL = "SELECT * FROM dogs WHERE {{name"
|
|
|
|
parseInterpolatedQuery rawSQL `shouldBe` Left "Found '{{' without a matching closing '}}'"
|
2023-03-13 21:13:56 +03:00
|
|
|
|
|
|
|
describe "Validation" do
|
2023-04-19 12:03:36 +03:00
|
|
|
let lmm =
|
|
|
|
LogicalModelMetadata
|
|
|
|
{ _lmmName = LogicalModelName (G.unsafeMkName "logical_model_name"),
|
|
|
|
_lmmFields = mempty,
|
|
|
|
_lmmDescription = Nothing,
|
|
|
|
_lmmSelectPermissions = mempty
|
2023-03-31 18:33:32 +03:00
|
|
|
}
|
|
|
|
|
2023-04-13 19:10:38 +03:00
|
|
|
nqm =
|
|
|
|
NativeQueryMetadata
|
|
|
|
{ _nqmRootFieldName = NativeQueryName (G.unsafeMkName "root_field_name"),
|
|
|
|
_nqmCode = InterpolatedQuery mempty,
|
2023-04-19 12:03:36 +03:00
|
|
|
_nqmReturns = LogicalModelName (G.unsafeMkName "logical_model_name"),
|
2023-04-13 19:10:38 +03:00
|
|
|
_nqmArguments = mempty,
|
2023-04-17 14:30:10 +03:00
|
|
|
_nqmArrayRelationships = mempty,
|
2023-04-13 19:10:38 +03:00
|
|
|
_nqmDescription = mempty
|
2023-03-13 21:13:56 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
it "Rejects undeclared variables" do
|
|
|
|
let Right code = parseInterpolatedQuery "SELECT {{hey}}"
|
2023-04-19 12:03:36 +03:00
|
|
|
let actual :: Either QErr Text = fmap snd $ runExcept $ nativeQueryToPreparedStatement lmm nqm {_nqmCode = code}
|
2023-03-13 21:13:56 +03:00
|
|
|
|
|
|
|
(first showQErr actual) `shouldSatisfy` isLeft
|
|
|
|
let Left err = actual
|
|
|
|
qeCode err `shouldBe` ValidationFailed
|
|
|
|
|
|
|
|
it "Handles multiple occurences of variables " do
|
|
|
|
let Right code = parseInterpolatedQuery "SELECT {{hey}}, {{hey}}"
|
|
|
|
let actual :: Either QErr Text =
|
2023-04-04 17:01:17 +03:00
|
|
|
fmap snd $
|
|
|
|
runExcept $
|
2023-04-13 19:10:38 +03:00
|
|
|
nativeQueryToPreparedStatement
|
2023-04-19 12:03:36 +03:00
|
|
|
lmm
|
2023-04-13 19:10:38 +03:00
|
|
|
nqm
|
|
|
|
{ _nqmCode = code,
|
|
|
|
_nqmArguments =
|
2023-04-26 18:42:13 +03:00
|
|
|
HashMap.fromList
|
2023-04-13 19:10:38 +03:00
|
|
|
[ (NativeQueryArgumentName "hey", NullableScalarType PGVarchar False Nothing)
|
2023-04-04 17:01:17 +03:00
|
|
|
]
|
|
|
|
}
|
2023-03-13 21:13:56 +03:00
|
|
|
|
|
|
|
(first showQErr actual) `shouldSatisfy` isRight
|
|
|
|
let Right rendered = actual
|
2023-03-16 13:44:14 +03:00
|
|
|
rendered
|
2023-04-18 19:08:07 +03:00
|
|
|
`shouldBe` "PREPARE _logimo_vali_(varchar) AS WITH _cte_logimo_vali_ AS (\nSELECT $1, $1\n)\nSELECT \nFROM _cte_logimo_vali_"
|
2023-03-13 21:13:56 +03:00
|
|
|
|
|
|
|
it "Handles multiple variables " do
|
|
|
|
let Right code = parseInterpolatedQuery "SELECT {{hey}}, {{ho}}"
|
|
|
|
let actual :: Either QErr Text =
|
2023-04-04 17:01:17 +03:00
|
|
|
fmap snd $
|
|
|
|
runExcept $
|
2023-04-13 19:10:38 +03:00
|
|
|
nativeQueryToPreparedStatement
|
2023-04-19 12:03:36 +03:00
|
|
|
lmm
|
2023-04-13 19:10:38 +03:00
|
|
|
nqm
|
|
|
|
{ _nqmCode = code,
|
|
|
|
_nqmArguments =
|
2023-04-26 18:42:13 +03:00
|
|
|
HashMap.fromList
|
2023-04-13 19:10:38 +03:00
|
|
|
[ (NativeQueryArgumentName "hey", NullableScalarType PGVarchar False Nothing),
|
|
|
|
(NativeQueryArgumentName "ho", NullableScalarType PGInteger False Nothing)
|
2023-04-04 17:01:17 +03:00
|
|
|
]
|
|
|
|
}
|
2023-03-13 21:13:56 +03:00
|
|
|
|
|
|
|
(first showQErr actual) `shouldSatisfy` isRight
|
|
|
|
let Right rendered = actual
|
2023-03-16 13:44:14 +03:00
|
|
|
rendered
|
2023-04-18 19:08:07 +03:00
|
|
|
`shouldBe` "PREPARE _logimo_vali_(varchar, integer) AS WITH _cte_logimo_vali_ AS (\nSELECT $1, $2\n)\nSELECT \nFROM _cte_logimo_vali_"
|