2021-05-26 19:19:26 +03:00
|
|
|
{-# LANGUAGE UndecidableInstances #-}
|
2019-10-21 19:01:05 +03:00
|
|
|
|
2021-09-24 01:56:37 +03:00
|
|
|
module Main (main) where
|
2020-12-28 15:56:00 +03:00
|
|
|
|
2021-09-24 01:56:37 +03:00
|
|
|
import Control.Concurrent.MVar
|
|
|
|
import Control.Natural ((:~>) (..))
|
|
|
|
import Data.Aeson qualified as A
|
|
|
|
import Data.ByteString.Lazy.Char8 qualified as BL
|
|
|
|
import Data.Environment qualified as Env
|
|
|
|
import Data.NonNegativeIntSpec qualified as NonNegetiveIntSpec
|
|
|
|
import Data.Parser.CacheControlSpec qualified as CacheControlParser
|
|
|
|
import Data.Parser.JSONPathSpec qualified as JsonPath
|
|
|
|
import Data.Time.Clock (getCurrentTime)
|
|
|
|
import Data.TimeSpec qualified as TimeSpec
|
|
|
|
import Data.URL.Template
|
|
|
|
import Database.MSSQL.TransactionSpec qualified as TransactionSpec
|
|
|
|
import Database.PG.Query qualified as Q
|
|
|
|
import Hasura.App
|
|
|
|
( PGMetadataStorageAppT (..),
|
|
|
|
mkPgSourceResolver,
|
|
|
|
)
|
|
|
|
import Hasura.EventingSpec qualified as EventingSpec
|
|
|
|
import Hasura.GraphQL.Parser.DirectivesTest qualified as GraphQLDirectivesSpec
|
|
|
|
import Hasura.GraphQL.RemoteServerSpec qualified as RemoteServerSpec
|
|
|
|
import Hasura.GraphQL.Schema.RemoteTest qualified as GraphRemoteSchemaSpec
|
|
|
|
import Hasura.IncrementalSpec qualified as IncrementalSpec
|
|
|
|
import Hasura.Metadata.Class
|
|
|
|
import Hasura.Prelude
|
|
|
|
import Hasura.RQL.DDL.Schema.Cache
|
|
|
|
import Hasura.RQL.DDL.Schema.Cache.Common
|
|
|
|
import Hasura.RQL.PermissionSpec qualified as PermSpec
|
|
|
|
import Hasura.RQL.RequestTransformSpec qualified as RequestTransformSpec
|
|
|
|
import Hasura.RQL.Types
|
|
|
|
import Hasura.RQL.Types.CommonSpec qualified as CommonTypesSpec
|
|
|
|
import Hasura.RQL.Types.EndpointSpec qualified as EndpointSpec
|
|
|
|
import Hasura.SQL.WKTSpec qualified as WKTSpec
|
|
|
|
import Hasura.Server.AuthSpec qualified as AuthSpec
|
|
|
|
import Hasura.Server.Init
|
|
|
|
import Hasura.Server.Migrate
|
|
|
|
import Hasura.Server.MigrateSpec qualified as MigrateSpec
|
|
|
|
import Hasura.Server.TelemetrySpec qualified as TelemetrySpec
|
|
|
|
import Hasura.Server.Types
|
|
|
|
import Hasura.Server.Version
|
|
|
|
import Hasura.Server.Version.TH
|
|
|
|
import Network.HTTP.Client qualified as HTTP
|
|
|
|
import Network.HTTP.Client.TLS qualified as HTTP
|
|
|
|
import Network.HTTP.Client.TransformableSpec qualified as TransformableSpec
|
|
|
|
import Options.Applicative
|
|
|
|
import System.Environment (getEnvironment)
|
|
|
|
import System.Exit (exitFailure)
|
|
|
|
import Test.Hspec
|
|
|
|
import Test.Hspec.Runner qualified as Hspec
|
2019-11-18 21:45:54 +03:00
|
|
|
|
|
|
|
data TestSuites
|
2021-09-09 10:59:04 +03:00
|
|
|
= -- | Run all test suites. It probably doesn't make sense to be able to specify additional
|
|
|
|
-- hspec args here.
|
|
|
|
AllSuites !(Maybe URLTemplate) !(Maybe URLTemplate)
|
|
|
|
| -- | Args to pass through to hspec (as if from 'getArgs'), and the specific suite to run.
|
|
|
|
SingleSuite ![String] !TestSuite
|
2019-11-18 21:45:54 +03:00
|
|
|
|
|
|
|
data TestSuite
|
|
|
|
= UnitSuite
|
2020-12-28 15:56:00 +03:00
|
|
|
| PostgresSuite !(Maybe URLTemplate)
|
2021-09-09 10:59:04 +03:00
|
|
|
| MSSQLSuite !(Maybe URLTemplate)
|
2019-10-21 19:01:05 +03:00
|
|
|
|
2019-12-14 09:47:38 +03:00
|
|
|
main :: IO ()
|
2021-09-09 10:59:04 +03:00
|
|
|
main =
|
|
|
|
withVersion $$(getVersionFromEnvironment) $
|
|
|
|
parseArgs >>= \case
|
|
|
|
AllSuites pgConnOptions mssqlConnOptions -> do
|
|
|
|
postgresSpecs <- buildPostgresSpecs pgConnOptions
|
|
|
|
mssqlSpecs <- buildMSSQLSpecs mssqlConnOptions
|
|
|
|
runHspec [] (unitSpecs *> postgresSpecs *> mssqlSpecs)
|
|
|
|
SingleSuite hspecArgs suite ->
|
|
|
|
runHspec hspecArgs =<< case suite of
|
2021-09-24 01:56:37 +03:00
|
|
|
UnitSuite -> pure unitSpecs
|
2021-09-09 10:59:04 +03:00
|
|
|
PostgresSuite pgConnOptions -> buildPostgresSpecs pgConnOptions
|
|
|
|
MSSQLSuite mssqlConnOptions -> buildMSSQLSpecs mssqlConnOptions
|
2019-11-18 21:45:54 +03:00
|
|
|
|
|
|
|
unitSpecs :: Spec
|
|
|
|
unitSpecs = do
|
2021-05-20 13:03:02 +03:00
|
|
|
describe "Data.NonNegativeInt" NonNegetiveIntSpec.spec
|
2020-01-14 00:56:51 +03:00
|
|
|
describe "Data.Parser.CacheControl" CacheControlParser.spec
|
2020-08-31 19:40:01 +03:00
|
|
|
describe "Data.Parser.JSONPath" JsonPath.spec
|
2020-01-16 04:56:57 +03:00
|
|
|
describe "Data.Time" TimeSpec.spec
|
2021-05-20 13:03:02 +03:00
|
|
|
describe "Hasura.Eventing" EventingSpec.spec
|
|
|
|
describe "Hasura.GraphQL.Parser.Directives" GraphQLDirectivesSpec.spec
|
2021-05-24 23:12:53 +03:00
|
|
|
describe "Hasura.GraphQL.Schema.Remote" GraphRemoteSchemaSpec.spec
|
2021-05-20 13:03:02 +03:00
|
|
|
describe "Hasura.Incremental" IncrementalSpec.spec
|
2021-09-06 19:59:18 +03:00
|
|
|
describe "Hasura.RQL.Types.Common" CommonTypesSpec.spec
|
2021-01-29 04:02:34 +03:00
|
|
|
describe "Hasura.RQL.Types.Endpoint" EndpointSpec.spec
|
2021-07-30 14:33:06 +03:00
|
|
|
describe "Hasura.GraphQL.RemoteServer" RemoteServerSpec.spec
|
2021-03-26 19:59:16 +03:00
|
|
|
describe "Hasura.SQL.WKT" WKTSpec.spec
|
2021-05-20 13:03:02 +03:00
|
|
|
describe "Hasura.Server.Auth" AuthSpec.spec
|
|
|
|
describe "Hasura.Server.Telemetry" TelemetrySpec.spec
|
2021-08-09 13:20:04 +03:00
|
|
|
describe "Hasura.RQL.PermissionSpec" PermSpec.spec
|
2021-09-16 14:03:01 +03:00
|
|
|
describe "Hasura.RQL.RequestTransformSpec" RequestTransformSpec.spec
|
|
|
|
describe "Network.HTTP.Client.TransformableSpec" TransformableSpec.spec
|
2019-11-18 21:45:54 +03:00
|
|
|
|
2021-09-09 10:59:04 +03:00
|
|
|
buildMSSQLSpecs :: Maybe URLTemplate -> IO Spec
|
|
|
|
buildMSSQLSpecs maybeUrlTemplate = do
|
|
|
|
env <- liftIO getEnvironment
|
|
|
|
let envMap = Env.mkEnvironment env
|
|
|
|
|
|
|
|
urlTemplate <- flip onLeft printErrExit $
|
|
|
|
runWithEnv env $ do
|
|
|
|
let envVar = fst mssqlConnectionString
|
|
|
|
maybeV <- withEnv maybeUrlTemplate envVar
|
|
|
|
onNothing maybeV $
|
|
|
|
throwError $
|
|
|
|
"Expected: " <> envVar
|
|
|
|
connStr <- flip onLeft printErrExit $ renderURLTemplate envMap urlTemplate
|
|
|
|
pure $ describe "Database.MSSQL.TransactionSpec" $ TransactionSpec.spec connStr
|
|
|
|
|
|
|
|
mssqlConnectionString :: (String, String)
|
|
|
|
mssqlConnectionString =
|
|
|
|
( "HASURA_MSSQL_CONN_STR",
|
|
|
|
"SQL Server database connection string. Example DRIVER={ODBC Driver 17 for SQL Server};SERVER=$IP_ADDRESS,$PORT;Uid=$USER;Pwd=$PASSWORD;"
|
|
|
|
)
|
|
|
|
|
2020-12-28 15:56:00 +03:00
|
|
|
buildPostgresSpecs :: HasVersion => Maybe URLTemplate -> IO Spec
|
|
|
|
buildPostgresSpecs maybeUrlTemplate = do
|
2019-11-18 21:45:54 +03:00
|
|
|
env <- getEnvironment
|
2020-12-28 15:56:00 +03:00
|
|
|
let envMap = Env.mkEnvironment env
|
|
|
|
|
2021-09-24 01:56:37 +03:00
|
|
|
pgUrlTemplate <- flip onLeft printErrExit $
|
|
|
|
runWithEnv env $ do
|
|
|
|
let envVar = fst databaseUrlEnv
|
|
|
|
maybeV <- withEnv maybeUrlTemplate envVar
|
|
|
|
onNothing maybeV $
|
|
|
|
throwError $
|
|
|
|
"Expected: --database-url or " <> envVar
|
2019-11-18 21:45:54 +03:00
|
|
|
|
2020-12-28 15:56:00 +03:00
|
|
|
pgUrlText <- flip onLeft printErrExit $ renderURLTemplate envMap pgUrlTemplate
|
|
|
|
let pgConnInfo = Q.ConnInfo 1 $ Q.CDDatabaseURI $ txtToBs pgUrlText
|
|
|
|
urlConf = UrlValue $ InputWebhook pgUrlTemplate
|
2021-04-28 19:49:23 +03:00
|
|
|
sourceConnInfo =
|
2021-05-21 04:49:50 +03:00
|
|
|
PostgresSourceConnInfo urlConf (Just setPostgresPoolSettings) True Q.ReadCommitted Nothing
|
2021-02-14 09:07:52 +03:00
|
|
|
sourceConfig = PostgresConnConfiguration sourceConnInfo Nothing
|
2019-11-18 21:45:54 +03:00
|
|
|
|
2021-09-24 01:56:37 +03:00
|
|
|
pgPool <- Q.initPGPool pgConnInfo Q.defaultConnParams {Q.cpConns = 1} print
|
2020-12-28 15:56:00 +03:00
|
|
|
let pgContext = mkPGExecCtx Q.Serializable pgPool
|
|
|
|
|
|
|
|
setupCacheRef = do
|
2019-11-20 21:21:30 +03:00
|
|
|
httpManager <- HTTP.newManager HTTP.tlsManagerSettings
|
2021-04-08 11:25:11 +03:00
|
|
|
let sqlGenCtx = SQLGenCtx False False
|
2021-02-18 19:46:14 +03:00
|
|
|
maintenanceMode = MaintenanceModeDisabled
|
|
|
|
serverConfigCtx =
|
[Preview] Inherited roles for postgres read queries
fixes #3868
docker image - `hasura/graphql-engine:inherited-roles-preview-48b73a2de`
Note:
To be able to use the inherited roles feature, the graphql-engine should be started with the env variable `HASURA_GRAPHQL_EXPERIMENTAL_FEATURES` set to `inherited_roles`.
Introduction
------------
This PR implements the idea of multiple roles as presented in this [paper](https://www.microsoft.com/en-us/research/wp-content/uploads/2016/02/FGALanguageICDE07.pdf). The multiple roles feature in this PR can be used via inherited roles. An inherited role is a role which can be created by combining multiple singular roles. For example, if there are two roles `author` and `editor` configured in the graphql-engine, then we can create a inherited role with the name of `combined_author_editor` role which will combine the select permissions of the `author` and `editor` roles and then make GraphQL queries using the `combined_author_editor`.
How are select permissions of different roles are combined?
------------------------------------------------------------
A select permission includes 5 things:
1. Columns accessible to the role
2. Row selection filter
3. Limit
4. Allow aggregation
5. Scalar computed fields accessible to the role
Suppose there are two roles, `role1` gives access to the `address` column with row filter `P1` and `role2` gives access to both the `address` and the `phone` column with row filter `P2` and we create a new role `combined_roles` which combines `role1` and `role2`.
Let's say the following GraphQL query is queried with the `combined_roles` role.
```graphql
query {
employees {
address
phone
}
}
```
This will translate to the following SQL query:
```sql
select
(case when (P1 or P2) then address else null end) as address,
(case when P2 then phone else null end) as phone
from employee
where (P1 or P2)
```
The other parameters of the select permission will be combined in the following manner:
1. Limit - Minimum of the limits will be the limit of the inherited role
2. Allow aggregations - If any of the role allows aggregation, then the inherited role will allow aggregation
3. Scalar computed fields - same as table column fields, as in the above example
APIs for inherited roles:
----------------------
1. `add_inherited_role`
`add_inherited_role` is the [metadata API](https://hasura.io/docs/1.0/graphql/core/api-reference/index.html#schema-metadata-api) to create a new inherited role. It accepts two arguments
`role_name`: the name of the inherited role to be added (String)
`role_set`: list of roles that need to be combined (Array of Strings)
Example:
```json
{
"type": "add_inherited_role",
"args": {
"role_name":"combined_user",
"role_set":[
"user",
"user1"
]
}
}
```
After adding the inherited role, the inherited role can be used like single roles like earlier
Note:
An inherited role can only be created with non-inherited/singular roles.
2. `drop_inherited_role`
The `drop_inherited_role` API accepts the name of the inherited role and drops it from the metadata. It accepts a single argument:
`role_name`: name of the inherited role to be dropped
Example:
```json
{
"type": "drop_inherited_role",
"args": {
"role_name":"combined_user"
}
}
```
Metadata
---------
The derived roles metadata will be included under the `experimental_features` key while exporting the metadata.
```json
{
"experimental_features": {
"derived_roles": [
{
"role_name": "manager_is_employee_too",
"role_set": [
"employee",
"manager"
]
}
]
}
}
```
Scope
------
Only postgres queries and subscriptions are supported in this PR.
Important points:
-----------------
1. All columns exposed to an inherited role will be marked as `nullable`, this is done so that cell value nullification can be done.
TODOs
-------
- [ ] Tests
- [ ] Test a GraphQL query running with a inherited role without enabling inherited roles in experimental features
- [] Tests for aggregate queries, limit, computed fields, functions, subscriptions (?)
- [ ] Introspection test with a inherited role (nullability changes in a inherited role)
- [ ] Docs
- [ ] Changelog
Co-authored-by: Vamshi Surabhi <6562944+0x777@users.noreply.github.com>
GitOrigin-RevId: 3b8ee1e11f5ceca80fe294f8c074d42fbccfec63
2021-03-08 14:14:13 +03:00
|
|
|
ServerConfigCtx FunctionPermissionsInferred RemoteSchemaPermsDisabled sqlGenCtx maintenanceMode mempty
|
2021-01-29 08:48:17 +03:00
|
|
|
cacheBuildParams = CacheBuildParams httpManager (mkPgSourceResolver print) serverConfigCtx
|
2021-05-26 19:19:26 +03:00
|
|
|
pgLogger = print
|
2019-11-20 21:21:30 +03:00
|
|
|
|
2021-05-26 19:19:26 +03:00
|
|
|
run :: MetadataStorageT (PGMetadataStorageAppT CacheBuild) a -> IO a
|
2020-12-28 15:56:00 +03:00
|
|
|
run =
|
2021-05-26 19:19:26 +03:00
|
|
|
runMetadataStorageT
|
2021-09-24 01:56:37 +03:00
|
|
|
>>> flip runPGMetadataStorageAppT (pgPool, pgLogger)
|
|
|
|
>>> runCacheBuild cacheBuildParams
|
|
|
|
>>> runExceptT
|
|
|
|
>=> flip onLeft printErrJExit
|
|
|
|
>=> flip onLeft printErrJExit
|
2019-11-20 21:21:30 +03:00
|
|
|
|
2020-12-28 15:56:00 +03:00
|
|
|
(metadata, schemaCache) <- run do
|
2021-09-24 01:56:37 +03:00
|
|
|
metadata <-
|
|
|
|
snd
|
|
|
|
<$> (liftEitherM . runExceptT . runTx pgContext Q.ReadWrite)
|
|
|
|
(migrateCatalog (Just sourceConfig) maintenanceMode =<< liftIO getCurrentTime)
|
2021-08-06 20:05:17 +03:00
|
|
|
schemaCache <- lift $ lift $ buildRebuildableSchemaCache envMap metadata
|
2020-12-28 15:56:00 +03:00
|
|
|
pure (metadata, schemaCache)
|
|
|
|
|
2019-11-20 21:21:30 +03:00
|
|
|
cacheRef <- newMVar schemaCache
|
2020-12-28 15:56:00 +03:00
|
|
|
pure $ NT (run . flip MigrateSpec.runCacheRefT cacheRef . fmap fst . runMetadataT metadata)
|
2019-11-20 21:21:30 +03:00
|
|
|
|
2021-09-24 01:56:37 +03:00
|
|
|
pure $
|
|
|
|
beforeAll setupCacheRef $
|
|
|
|
describe "Hasura.Server.Migrate" $ MigrateSpec.spec sourceConfig pgContext pgConnInfo
|
2019-11-18 21:45:54 +03:00
|
|
|
|
|
|
|
parseArgs :: IO TestSuites
|
2021-09-24 01:56:37 +03:00
|
|
|
parseArgs =
|
|
|
|
execParser $
|
|
|
|
info (helper <*> (parseNoCommand <|> parseSubCommand)) $
|
|
|
|
fullDesc <> header "Hasura GraphQL Engine test suite"
|
2019-10-21 19:01:05 +03:00
|
|
|
where
|
2020-12-28 15:56:00 +03:00
|
|
|
parseDbUrlTemplate =
|
|
|
|
parseDatabaseUrl <|> (fmap rawConnDetailsToUrl <$> parseRawConnDetails)
|
2021-09-09 10:59:04 +03:00
|
|
|
parseNoCommand = AllSuites <$> parseDbUrlTemplate <*> parseDbUrlTemplate
|
2020-02-13 20:38:23 +03:00
|
|
|
parseSubCommand = SingleSuite <$> parseHspecPassThroughArgs <*> subCmd
|
|
|
|
where
|
2021-09-09 10:59:04 +03:00
|
|
|
subCmd =
|
|
|
|
subparser $
|
|
|
|
mconcat
|
|
|
|
[ command "unit" $
|
|
|
|
info (pure UnitSuite) $
|
|
|
|
progDesc "Only run unit tests",
|
|
|
|
command "postgres" $
|
|
|
|
info (helper <*> (PostgresSuite <$> parseDbUrlTemplate)) $
|
|
|
|
progDesc "Only run Postgres integration tests",
|
|
|
|
command "mssql" $
|
|
|
|
info (helper <*> (MSSQLSuite <$> parseDbUrlTemplate)) $
|
|
|
|
progDesc "Only run SQL Server unit tests"
|
|
|
|
]
|
2020-01-21 21:12:27 +03:00
|
|
|
-- Add additional arguments and tweak as needed:
|
|
|
|
hspecArgs = ["match", "skip"]
|
|
|
|
-- parse to a list of arguments as they'd appear from 'getArgs':
|
|
|
|
parseHspecPassThroughArgs :: Parser [String]
|
2021-09-24 01:56:37 +03:00
|
|
|
parseHspecPassThroughArgs = fmap concat $
|
|
|
|
for hspecArgs $ \nm ->
|
|
|
|
fmap (maybe [] (\a -> ["--" <> nm, a])) $
|
|
|
|
optional $
|
|
|
|
strOption
|
|
|
|
( long nm
|
|
|
|
<> metavar "<PATTERN>"
|
|
|
|
<> help "Flag passed through to hspec (see hspec docs)."
|
|
|
|
)
|
2020-01-21 21:12:27 +03:00
|
|
|
|
|
|
|
runHspec :: [String] -> Spec -> IO ()
|
|
|
|
runHspec hspecArgs m = do
|
|
|
|
config <- Hspec.readConfig Hspec.defaultConfig hspecArgs
|
2019-11-18 21:45:54 +03:00
|
|
|
Hspec.evaluateSummary =<< Hspec.runSpec m config
|
|
|
|
|
|
|
|
printErrExit :: String -> IO a
|
|
|
|
printErrExit = (*> exitFailure) . putStrLn
|
2019-11-20 21:21:30 +03:00
|
|
|
|
|
|
|
printErrJExit :: (A.ToJSON a) => a -> IO b
|
|
|
|
printErrJExit = (*> exitFailure) . BL.putStrLn . A.encode
|