2021-02-19 05:39:30 +03:00
|
|
|
import ruamel.yaml as yaml
|
|
|
|
from validate import check_query_f
|
|
|
|
import pytest
|
2021-05-20 00:06:06 +03:00
|
|
|
import os
|
2021-02-19 05:39:30 +03:00
|
|
|
|
|
|
|
usefixtures = pytest.mark.usefixtures
|
|
|
|
|
|
|
|
use_mutation_fixtures = usefixtures(
|
|
|
|
'per_class_db_schema_for_mutation_tests',
|
|
|
|
'per_method_db_data_for_mutation_tests'
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
@usefixtures('per_method_tests_db_state')
|
|
|
|
class TestMetadata:
|
|
|
|
|
|
|
|
def test_reload_metadata(self, hge_ctx):
|
|
|
|
check_query_f(hge_ctx, self.dir() + '/reload_metadata.yaml')
|
|
|
|
|
|
|
|
# FIXME:- Using export_metadata will dump
|
|
|
|
# the source configuration dependent on --database-url
|
|
|
|
# def test_export_metadata(self, hge_ctx):
|
|
|
|
# check_query_f(hge_ctx, self.dir() + '/export_metadata.yaml')
|
|
|
|
|
|
|
|
def test_clear_metadata(self, hge_ctx):
|
|
|
|
check_query_f(hge_ctx, self.dir() + '/clear_metadata.yaml')
|
|
|
|
|
2021-07-16 19:08:23 +03:00
|
|
|
def test_clear_metadata_as_user(self, hge_ctx):
|
|
|
|
check_query_f(hge_ctx, self.dir() + '/metadata_as_user_err.yaml')
|
|
|
|
|
2021-02-19 05:39:30 +03:00
|
|
|
def test_replace_metadata(self, hge_ctx):
|
|
|
|
check_query_f(hge_ctx, self.dir() + '/replace_metadata.yaml')
|
|
|
|
|
|
|
|
def test_replace_metadata_wo_remote_schemas(self, hge_ctx):
|
|
|
|
check_query_f(hge_ctx, self.dir() + '/replace_metadata_wo_rs.yaml')
|
|
|
|
|
|
|
|
def test_replace_metadata_v2(self, hge_ctx):
|
|
|
|
check_query_f(hge_ctx, self.dir() + '/replace_metadata_v2.yaml')
|
|
|
|
|
|
|
|
def test_replace_metadata_allow_inconsistent(self, hge_ctx):
|
|
|
|
check_query_f(hge_ctx, self.dir() +
|
|
|
|
'/replace_metadata_allow_inconsistent_inconsistent.yaml')
|
|
|
|
check_query_f(hge_ctx, self.dir() +
|
|
|
|
'/replace_metadata_allow_inconsistent.yaml')
|
|
|
|
|
2021-09-14 15:02:13 +03:00
|
|
|
def test_replace_metadata_disallow_inconsistent_metadata(self, hge_ctx):
|
|
|
|
st_code, resp = hge_ctx.v1metadataq({"type": "export_metadata", "args": {}})
|
|
|
|
assert st_code == 200, resp
|
|
|
|
default_source_config = {}
|
|
|
|
default_source = list(filter(lambda source: (source["name"] == "default"), resp["sources"]))
|
|
|
|
if default_source:
|
|
|
|
default_source_config = default_source[0]["configuration"]
|
|
|
|
else:
|
|
|
|
assert False, "default source config not found"
|
|
|
|
return
|
|
|
|
st_code, resp = hge_ctx.v1metadataq({
|
|
|
|
"type": "replace_metadata",
|
|
|
|
"version": 2,
|
|
|
|
"args": {
|
|
|
|
"metadata": {
|
|
|
|
"version": 3,
|
|
|
|
"sources": [
|
|
|
|
{
|
|
|
|
"name": "default",
|
|
|
|
"kind": "postgres",
|
|
|
|
"tables": [
|
|
|
|
{
|
|
|
|
"table": {
|
|
|
|
"schema": "public",
|
|
|
|
"name": "author"
|
|
|
|
},
|
|
|
|
"insert_permissions": [
|
|
|
|
{
|
|
|
|
"role": "user1",
|
|
|
|
"permission": {
|
|
|
|
"check": {},
|
|
|
|
"columns": [
|
|
|
|
"id",
|
|
|
|
"name"
|
|
|
|
],
|
|
|
|
"backend_only": False
|
|
|
|
}
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"role": "user2",
|
|
|
|
"permission": {
|
|
|
|
"check": {
|
|
|
|
"id": {
|
|
|
|
"_eq": "X-Hasura-User-Id"
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"columns": [
|
|
|
|
"id",
|
|
|
|
"name"
|
|
|
|
],
|
|
|
|
"backend_only": False
|
|
|
|
}
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"configuration": default_source_config
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"inherited_roles": [
|
|
|
|
{
|
|
|
|
"role_name": "users",
|
|
|
|
"role_set": [
|
|
|
|
"user2",
|
|
|
|
"user1"
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
assert st_code == 400, resp
|
|
|
|
assert resp == {
|
|
|
|
"internal": [
|
|
|
|
{
|
|
|
|
"reason": "Could not inherit permission for the role 'users' for the entity: 'insert permission, table: author, source: 'default''",
|
|
|
|
"name": "users",
|
|
|
|
"type": "inherited role permission inconsistency",
|
|
|
|
"entity": {
|
|
|
|
"permission_type": "insert",
|
|
|
|
"source": "default",
|
|
|
|
"table": "author"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"path": "$.args",
|
|
|
|
"error": "cannot continue due to inconsistent metadata",
|
|
|
|
"code": "unexpected"
|
|
|
|
}
|
|
|
|
|
2021-02-19 05:39:30 +03:00
|
|
|
def test_dump_internal_state(self, hge_ctx):
|
|
|
|
check_query_f(hge_ctx, self.dir() + '/dump_internal_state.yaml')
|
|
|
|
|
2021-03-24 09:44:42 +03:00
|
|
|
def test_pg_add_source(self, hge_ctx):
|
|
|
|
check_query_f(hge_ctx, self.dir() + '/pg_add_source.yaml')
|
Disable test_pg_add_source_with_source_parameters locally.
### Description
Our python test suite has several major problems; one of them being that the tests themselves are not responsible for their own setup. We are therefore using environment variables for all matters of configuration, such as _where the postgres instance is_. This is something that should be changed, but in the meantime, it is the test implementer's responsibility to ensure that tests have a consistent setup in CI and locally, or to to add the proper "skip" annotations.
The recently added `test_pg_add_source_with_source_parameters` fails to do so: as it tests adding a postgres source from hardcoded parameters, rather than relying on environment variables, it only works if the postgres instance is at the matching address, which happens to be the one set in the circle ci config. This is undesirable for two reasons:
- it breaks local tests: running tests locally with `dev.sh` sets postgres up differently, and the test fails;
- a change to the circle config would result in failures in that test.
Sadly, there's no good solution here: our tests do not currently support expanding environment variables in the queries' yaml files, meaning it's not possible to set the values of all those parameters differently in each environment. And we haven't yet started working towards having a unified testing environment setup.
As a result, this PR disables the offending test UNLESS the postgres instance happens to be exactly where the test expects it. This is also very inelegant and adds more tech debt to the pile, but I do not see how to fix this with our current test infrastructure. :(
https://github.com/hasura/graphql-engine-mono/pull/2336
GitOrigin-RevId: 8bc9142075d14acaa48e9c4b20de2527185bc75c
2021-09-14 07:10:15 +03:00
|
|
|
|
|
|
|
@pytest.mark.skipif(
|
|
|
|
os.getenv('HASURA_GRAPHQL_PG_SOURCE_URL_1') != 'postgresql://gql_test@localhost:5432/pg_source_1',
|
|
|
|
reason="This test relies on hardcoded connection parameters that match Circle's setup.")
|
2021-09-06 19:59:18 +03:00
|
|
|
def test_pg_add_source_with_source_parameters(self, hge_ctx):
|
|
|
|
check_query_f(hge_ctx, self.dir() + '/pg_add_source_with_parameters.yaml')
|
2021-03-24 09:44:42 +03:00
|
|
|
|
|
|
|
def test_pg_track_table_source(self, hge_ctx):
|
|
|
|
check_query_f(hge_ctx, self.dir() + '/pg_track_table_source.yaml')
|
|
|
|
|
2021-05-24 16:13:08 +03:00
|
|
|
def test_rename_source(self, hge_ctx):
|
|
|
|
check_query_f(hge_ctx, self.dir() + '/rename_source.yaml')
|
|
|
|
|
2021-03-24 09:44:42 +03:00
|
|
|
def test_pg_multisource_query(self, hge_ctx):
|
|
|
|
check_query_f(hge_ctx, self.dir() + '/pg_multisource_query.yaml')
|
|
|
|
|
2021-05-20 00:06:06 +03:00
|
|
|
@pytest.mark.skipif(
|
|
|
|
os.getenv('HASURA_GRAPHQL_PG_SOURCE_URL_1') == os.getenv('HASURA_GRAPHQL_PG_SOURCE_URL_2') or
|
|
|
|
os.getenv('HASURA_GRAPHQL_PG_SOURCE_URL_1') is None or
|
|
|
|
os.getenv('HASURA_GRAPHQL_PG_SOURCE_URL_2') is None,
|
|
|
|
reason="We need two different and valid instances of postgres for this test.")
|
2021-05-18 16:06:42 +03:00
|
|
|
def test_pg_multisource_table_name_conflict(self, hge_ctx):
|
|
|
|
check_query_f(hge_ctx, self.dir() + '/pg_multisource_table_name_conflict.yaml')
|
|
|
|
|
2021-02-19 05:39:30 +03:00
|
|
|
@classmethod
|
|
|
|
def dir(cls):
|
|
|
|
return "queries/v1/metadata"
|
|
|
|
|
|
|
|
# TODO These look like dependent tests. Ideally we should be able to run tests independently
|
|
|
|
|
|
|
|
|
|
|
|
@usefixtures('per_class_tests_db_state')
|
|
|
|
class TestMetadataOrder:
|
|
|
|
@classmethod
|
|
|
|
def dir(cls):
|
|
|
|
return "queries/v1/metadata_order"
|
|
|
|
|
|
|
|
# FIXME:- Using export_metadata will dump
|
|
|
|
# the source configuration dependent on --database-url
|
|
|
|
# def test_export_metadata(self, hge_ctx):
|
|
|
|
# check_query_f(hge_ctx, self.dir() + '/export_metadata.yaml')
|
|
|
|
|
|
|
|
# def test_clear_export_metadata(self, hge_ctx):
|
|
|
|
# In the 'clear_export_metadata.yaml' the metadata is added
|
|
|
|
# using the metadata APIs
|
|
|
|
# check_query_f(hge_ctx, self.dir() + '/clear_export_metadata.yaml')
|
|
|
|
|
|
|
|
def test_export_replace(self, hge_ctx):
|
|
|
|
url = '/v1/query'
|
|
|
|
export_query = {
|
|
|
|
'type': 'export_metadata',
|
|
|
|
'args': {}
|
|
|
|
}
|
|
|
|
headers = {}
|
|
|
|
if hge_ctx.hge_key is not None:
|
|
|
|
headers['X-Hasura-Admin-Secret'] = hge_ctx.hge_key
|
2021-03-03 07:40:03 +03:00
|
|
|
# we are exporting the metadata here after creating it through
|
2021-02-19 05:39:30 +03:00
|
|
|
# the metadata APIs
|
|
|
|
export_code, export_resp, _ = hge_ctx.anyq(url, export_query, headers)
|
|
|
|
assert export_code == 200, export_resp
|
|
|
|
replace_query = {
|
|
|
|
'type': 'replace_metadata',
|
|
|
|
'args': export_resp
|
|
|
|
}
|
|
|
|
# we are replacing the metadata with the exported metadata from the
|
|
|
|
# `export_metadata` response.
|
|
|
|
replace_code, replace_resp, _ = hge_ctx.anyq(
|
|
|
|
url, replace_query, headers)
|
|
|
|
assert replace_code == 200, replace_resp
|
|
|
|
# This test catches incorrect key names(if any) in the export_metadata serialization,
|
|
|
|
# for example, A new query collection is added to the allow list using the
|
|
|
|
# add_collection_to_allowlist metadata API. When
|
|
|
|
# the metadata is exported it will contain the allowlist. Now, when this
|
|
|
|
# metadata is imported, if the graphql-engine is expecting a different key
|
|
|
|
# like allow_list(instead of allowlist) then the allow list won't be imported.
|
|
|
|
# Now, exporting the metadata won't contain the allowlist key
|
|
|
|
# because it wasn't imported properly and hence the two exports will differ.
|
|
|
|
export_code_1, export_resp_1, _ = hge_ctx.anyq(
|
|
|
|
url, export_query, headers)
|
|
|
|
assert export_code_1 == 200
|
|
|
|
assert export_resp == export_resp_1
|
|
|
|
|
|
|
|
def test_export_replace_v2(self, hge_ctx):
|
|
|
|
url = '/v1/metadata'
|
|
|
|
export_query = {
|
|
|
|
'type': 'export_metadata',
|
|
|
|
'version': 2,
|
|
|
|
'args': {}
|
|
|
|
}
|
|
|
|
headers = {}
|
|
|
|
if hge_ctx.hge_key is not None:
|
|
|
|
headers['X-Hasura-Admin-Secret'] = hge_ctx.hge_key
|
2021-03-03 07:40:03 +03:00
|
|
|
# we are exporting the metadata here after creating it through
|
2021-02-19 05:39:30 +03:00
|
|
|
# the metadata APIs
|
|
|
|
export_code, export_resp, _ = hge_ctx.anyq(url, export_query, headers)
|
|
|
|
assert export_code == 200, export_resp
|
|
|
|
|
|
|
|
replace_query = {
|
|
|
|
'type': 'replace_metadata',
|
|
|
|
'version': 2,
|
|
|
|
'resource_version': export_resp['resource_version'],
|
|
|
|
'args': {'metadata': export_resp['metadata']}
|
|
|
|
}
|
|
|
|
# we are replacing the metadata with the exported metadata from the
|
|
|
|
# `export_metadata` response.
|
|
|
|
replace_code, replace_resp, _ = hge_ctx.anyq(
|
|
|
|
url, replace_query, headers)
|
|
|
|
assert replace_code == 200, replace_resp
|
|
|
|
|
|
|
|
export_code_1, export_resp_1, _ = hge_ctx.anyq(
|
|
|
|
url, export_query, headers)
|
|
|
|
assert export_code_1 == 200
|
|
|
|
assert export_resp['metadata'] == export_resp_1['metadata']
|
|
|
|
|
|
|
|
# `resource_version` should have been incremented
|
|
|
|
assert export_resp['resource_version'] + \
|
|
|
|
1 == export_resp_1['resource_version']
|
|
|
|
|
|
|
|
def test_export_replace_v2_conflict(self, hge_ctx):
|
|
|
|
url = '/v1/metadata'
|
|
|
|
export_query = {
|
|
|
|
'type': 'export_metadata',
|
|
|
|
'version': 2,
|
|
|
|
'args': {}
|
|
|
|
}
|
|
|
|
headers = {}
|
|
|
|
if hge_ctx.hge_key is not None:
|
|
|
|
headers['X-Hasura-Admin-Secret'] = hge_ctx.hge_key
|
2021-03-03 07:40:03 +03:00
|
|
|
# we are exporting the metadata here after creating it through
|
2021-02-19 05:39:30 +03:00
|
|
|
# the metadata APIs
|
|
|
|
export_code, export_resp, _ = hge_ctx.anyq(url, export_query, headers)
|
|
|
|
assert export_code == 200, export_resp
|
|
|
|
|
|
|
|
replace_query = {
|
|
|
|
'type': 'replace_metadata',
|
|
|
|
'version': 2,
|
|
|
|
'resource_version': export_resp['resource_version'] - 1,
|
|
|
|
'args': {'metadata': export_resp['metadata']}
|
|
|
|
}
|
|
|
|
# we are replacing the metadata with the exported metadata from the
|
|
|
|
# `export_metadata` response.
|
|
|
|
# Using the wrong `resource_version` should result in a 409 conflict
|
|
|
|
replace_code, replace_resp, _ = hge_ctx.anyq(
|
|
|
|
url, replace_query, headers)
|
|
|
|
assert replace_code == 409, replace_resp
|
|
|
|
|
|
|
|
export_code_1, export_resp_1, _ = hge_ctx.anyq(
|
|
|
|
url, export_query, headers)
|
|
|
|
assert export_code_1 == 200
|
|
|
|
assert export_resp['metadata'] == export_resp_1['metadata']
|
|
|
|
|
|
|
|
# `resource_version` should be unchanged
|
|
|
|
assert export_resp['resource_version'] == export_resp_1['resource_version']
|
2021-03-03 07:40:03 +03:00
|
|
|
|
|
|
|
def test_reload_metadata(self, hge_ctx):
|
|
|
|
url = '/v1/metadata'
|
|
|
|
export_query = {
|
|
|
|
'type': 'export_metadata',
|
|
|
|
'version': 2,
|
|
|
|
'args': {}
|
|
|
|
}
|
|
|
|
headers = {}
|
|
|
|
if hge_ctx.hge_key is not None:
|
|
|
|
headers['X-Hasura-Admin-Secret'] = hge_ctx.hge_key
|
|
|
|
# we are exporting the metadata here after creating it through
|
|
|
|
# the metadata APIs
|
|
|
|
export_code, export_resp, _ = hge_ctx.anyq(url, export_query, headers)
|
|
|
|
assert export_code == 200, export_resp
|
|
|
|
|
|
|
|
reload_query = {
|
|
|
|
'type': 'reload_metadata',
|
|
|
|
'resource_version': export_resp['resource_version'],
|
|
|
|
'args': {}
|
|
|
|
}
|
|
|
|
# we are replacing the metadata with the exported metadata from the
|
|
|
|
# `export_metadata` response.
|
|
|
|
reload_code, reload_resp, _ = hge_ctx.anyq(
|
|
|
|
url, reload_query, headers)
|
|
|
|
assert reload_code == 200, reload_resp
|
|
|
|
|
|
|
|
export_code_1, export_resp_1, _ = hge_ctx.anyq(
|
|
|
|
url, export_query, headers)
|
|
|
|
assert export_code_1 == 200
|
|
|
|
assert export_resp['metadata'] == export_resp_1['metadata']
|
|
|
|
|
|
|
|
# `resource_version` should have been incremented
|
|
|
|
assert export_resp['resource_version'] + \
|
|
|
|
1 == export_resp_1['resource_version']
|
|
|
|
|
|
|
|
def test_reload_metadata_conflict(self, hge_ctx):
|
|
|
|
url = '/v1/metadata'
|
|
|
|
export_query = {
|
|
|
|
'type': 'export_metadata',
|
|
|
|
'version': 2,
|
|
|
|
'args': {}
|
|
|
|
}
|
|
|
|
headers = {}
|
|
|
|
if hge_ctx.hge_key is not None:
|
|
|
|
headers['X-Hasura-Admin-Secret'] = hge_ctx.hge_key
|
|
|
|
# we are exporting the metadata here after creating it through
|
|
|
|
# the metadata APIs
|
|
|
|
export_code, export_resp, _ = hge_ctx.anyq(url, export_query, headers)
|
|
|
|
assert export_code == 200, export_resp
|
|
|
|
|
|
|
|
reload_query = {
|
|
|
|
'type': 'reload_metadata',
|
|
|
|
'resource_version': export_resp['resource_version'] - 1,
|
|
|
|
'args': {}
|
|
|
|
}
|
|
|
|
# we are replacing the metadata with the exported metadata from the
|
|
|
|
# `export_metadata` response.
|
|
|
|
reload_code, reload_resp, _ = hge_ctx.anyq(
|
|
|
|
url, reload_query, headers)
|
|
|
|
assert reload_code == 409, reload_resp
|
|
|
|
|
|
|
|
export_code_1, export_resp_1, _ = hge_ctx.anyq(
|
|
|
|
url, export_query, headers)
|
|
|
|
assert export_code_1 == 200
|
|
|
|
assert export_resp['metadata'] == export_resp_1['metadata']
|
|
|
|
|
|
|
|
# `resource_version` should be unchanged
|
|
|
|
assert export_resp['resource_version'] == export_resp_1['resource_version']
|
2021-03-24 09:44:42 +03:00
|
|
|
|
2021-06-17 16:21:55 +03:00
|
|
|
|
|
|
|
@pytest.mark.parametrize("backend", ['citus', 'mssql', 'postgres'])
|
|
|
|
@usefixtures('per_class_tests_db_state')
|
2021-08-12 19:47:33 +03:00
|
|
|
class TestSetTableCustomizationCommon:
|
2021-06-17 16:21:55 +03:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def dir(cls):
|
|
|
|
return "queries/v1/metadata"
|
|
|
|
|
|
|
|
def test_set_table_customization(self, hge_ctx):
|
|
|
|
check_query_f(hge_ctx, self.dir() + hge_ctx.backend_suffix('/set_table_customization') + '.yaml')
|