From 484d72ae36c611ca56c1d13eb5e0d47c5ef4532c Mon Sep 17 00:00:00 2001 From: Valeria Bulanova Date: Thu, 5 Dec 2024 12:50:31 +0300 Subject: [PATCH 1/3] feat(connectors): BI-5975 Connection export --- .../api_schema/connection_base_fields.py | 2 +- .../api_schema/connection_sql.py | 2 +- .../dl_api_connector/api_schema/extras.py | 5 ++++ .../dl_api_connector/api_schema/top_level.py | 18 +++++++++++++ .../dl_api_lib/app/control_api/app.py | 16 ++++++++++++ .../app/control_api/resources/connections.py | 22 ++++++++++++++++ .../connector/connection_suite.py | 26 +++++++++++++++++++ .../api/api_schema/connection.py | 2 +- .../api/api_schema/connection.py | 1 - .../dl_connector_mssql/core/us_connection.py | 1 + .../dl_connector_mysql/core/us_connection.py | 1 + .../dl_connector_oracle/core/us_connection.py | 1 + .../core/postgresql_base/us_connection.py | 1 + .../api/api_schema/connection.py | 1 - .../dl_connector_promql/core/us_connection.py | 1 + .../api/api_schema/connection.py | 2 -- .../core/ydb/us_connection.py | 1 + lib/dl_core/dl_core/us_connection_base.py | 1 + 18 files changed, 97 insertions(+), 7 deletions(-) diff --git a/lib/dl_api_connector/dl_api_connector/api_schema/connection_base_fields.py b/lib/dl_api_connector/dl_api_connector/api_schema/connection_base_fields.py index 245673590..582ddc16a 100644 --- a/lib/dl_api_connector/dl_api_connector/api_schema/connection_base_fields.py +++ b/lib/dl_api_connector/dl_api_connector/api_schema/connection_base_fields.py @@ -47,7 +47,7 @@ def secret_string_field( required: bool = True, allow_none: bool = False, default: Optional[str] = None, - bi_extra: FieldExtra = FieldExtra(editable=True), # noqa: B008 + bi_extra: FieldExtra = FieldExtra(editable=True, export_fake=True), # noqa: B008 ) -> ma_fields.String: return ma_fields.String( attribute=attribute, diff --git a/lib/dl_api_connector/dl_api_connector/api_schema/connection_sql.py b/lib/dl_api_connector/dl_api_connector/api_schema/connection_sql.py index 9b9f02a6d..d1bcdcad4 100644 --- a/lib/dl_api_connector/dl_api_connector/api_schema/connection_sql.py +++ b/lib/dl_api_connector/dl_api_connector/api_schema/connection_sql.py @@ -75,7 +75,7 @@ class ClassicSQLConnectionSchema(ConnectionSchema): host = DBHostField(attribute="data.host", required=True, bi_extra=FieldExtra(editable=True)) port = ma_fields.Integer(attribute="data.port", required=True, bi_extra=FieldExtra(editable=True)) username = ma_fields.String(attribute="data.username", required=True, bi_extra=FieldExtra(editable=True)) - password = secret_string_field(attribute="data.password", bi_extra=FieldExtra(editable=True)) + password = secret_string_field(attribute="data.password") db_name = ma_fields.String( attribute="data.db_name", allow_none=True, bi_extra=FieldExtra(editable=True), validate=db_name_no_query_params ) diff --git a/lib/dl_api_connector/dl_api_connector/api_schema/extras.py b/lib/dl_api_connector/dl_api_connector/api_schema/extras.py index 0fa1b12e4..38e43b084 100644 --- a/lib/dl_api_connector/dl_api_connector/api_schema/extras.py +++ b/lib/dl_api_connector/dl_api_connector/api_schema/extras.py @@ -25,6 +25,10 @@ class EditMode(OperationsMode): test = enum.auto() +class ExportMode(OperationsMode): + export = enum.auto() + + class SchemaKWArgs(TypedDict): only: Optional[Sequence[str]] partial: Union[Sequence[str], bool] @@ -38,3 +42,4 @@ class FieldExtra: partial_in: Sequence[OperationsMode] = () exclude_in: Sequence[OperationsMode] = () editable: Union[bool, Sequence[OperationsMode]] = () + export_fake: Optional[bool] = False diff --git a/lib/dl_api_connector/dl_api_connector/api_schema/top_level.py b/lib/dl_api_connector/dl_api_connector/api_schema/top_level.py index e04a34dd6..3dfdcb1be 100644 --- a/lib/dl_api_connector/dl_api_connector/api_schema/top_level.py +++ b/lib/dl_api_connector/dl_api_connector/api_schema/top_level.py @@ -1,6 +1,7 @@ from __future__ import annotations import copy +from copy import deepcopy import itertools import logging import os @@ -19,6 +20,7 @@ import marshmallow from marshmallow import ( missing, + post_dump, post_load, pre_load, ) @@ -28,6 +30,7 @@ from dl_api_connector.api_schema.extras import ( CreateMode, EditMode, + ExportMode, FieldExtra, OperationsMode, SchemaKWArgs, @@ -98,6 +101,13 @@ def all_fields_with_extra_info(cls) -> Iterable[tuple[str, ma_fields.Field, Fiel if extra is not None: yield field_name, field, extra + @classmethod + def fieldnames_with_extra_export_fake_info(cls) -> Iterable[str]: + for field_name, field in cls.all_fields_dict().items(): + extra = cls.get_field_extra(field) + if extra is not None and extra.export_fake is True: + yield field_name + def _refine_init_kwargs(self, kw_args: SchemaKWArgs, operations_mode: Optional[OperationsMode]) -> SchemaKWArgs: if operations_mode is None: return kw_args @@ -232,6 +242,14 @@ def pre_load(self, data: dict[str, Any], **_: Any) -> dict[str, Any]: ) return self.handle_unknown_fields(data) + @post_dump(pass_many=False) + def post_dump(self, data: dict[str, Any], **_: Any) -> dict[str, Any]: + if isinstance(self.operations_mode, ExportMode): + data = deepcopy(data) + for secret_field in self.fieldnames_with_extra_export_fake_info(): + data[secret_field] = "******" + return data + _US_ENTRY_TV = TypeVar("_US_ENTRY_TV", bound=USEntry) diff --git a/lib/dl_api_lib/dl_api_lib/app/control_api/app.py b/lib/dl_api_lib/dl_api_lib/app/control_api/app.py index 7f34c719b..786e812d3 100644 --- a/lib/dl_api_lib/dl_api_lib/app/control_api/app.py +++ b/lib/dl_api_lib/dl_api_lib/app/control_api/app.py @@ -6,6 +6,7 @@ Generic, Optional, TypeVar, + final, ) import attr @@ -49,6 +50,12 @@ from dl_core.connection_models import ConnectOptions from dl_core.us_connection_base import ConnectionBase +from dl_api_lib.app.control_api.resources.connections import ( + BIResource, + ConnectionExportItem, +) +from dl_api_lib.app.control_api.resources.connections import ns as connections_namespace + @attr.s(frozen=True) class EnvSetupResult: @@ -62,6 +69,14 @@ class EnvSetupResult: class ControlApiAppFactory(SRFactoryBuilder, Generic[TControlApiAppSettings], abc.ABC): _settings: TControlApiAppSettings = attr.ib() + def get_connection_export_resource(self) -> type[BIResource]: + return ConnectionExportItem + + @final + def register_additional_handlers(self) -> None: + connection_export_resource = self.get_connection_export_resource() + connections_namespace.add_resource(connection_export_resource, "/export/") + @abc.abstractmethod def set_up_environment( self, @@ -159,6 +174,7 @@ def create_app( ma = Marshmallow() ma.init_app(app) + app.before_first_request(self.register_additional_handlers) init_apis(app) return app diff --git a/lib/dl_api_lib/dl_api_lib/app/control_api/resources/connections.py b/lib/dl_api_lib/dl_api_lib/app/control_api/resources/connections.py index cae40e5d0..ee0e7c4a0 100644 --- a/lib/dl_api_lib/dl_api_lib/app/control_api/resources/connections.py +++ b/lib/dl_api_lib/dl_api_lib/app/control_api/resources/connections.py @@ -15,6 +15,7 @@ from dl_api_connector.api_schema.extras import ( CreateMode, EditMode, + ExportMode, ) from dl_api_lib import exc from dl_api_lib.api_decorators import schematic_request @@ -205,6 +206,27 @@ def put(self, connection_id): # type: ignore # TODO: fix us_manager.save(conn) +class ConnectionExportItem(BIResource): + @put_to_request_context(endpoint_code="ConnectionGet") + @schematic_request( + ns=ns, + responses={ + # 200: ('Success', GetConnectionResponseSchema()), + }, + ) + def get(self, connection_id: str) -> dict: + conn = self.get_us_manager().get_by_id(connection_id, expected_type=ConnectionBase) + need_permission_on_entry(conn, USPermissionKind.read) + assert isinstance(conn, ConnectionBase) + + if not conn.allow_export: + raise exc.UnsupportedForEntityType(f"Connector {conn.conn_type.name} does not support export") + + result = GenericConnectionSchema(context=self.get_schema_ctx(ExportMode.export)).dump(conn) + result.update(options=ConnectionOptionsSchema().dump(conn.get_options())) + return result + + def _dump_source_templates(tpls) -> dict: # type: ignore # TODO: fix if tpls is None: return None # type: ignore # TODO: fix diff --git a/lib/dl_api_lib_testing/dl_api_lib_testing/connector/connection_suite.py b/lib/dl_api_lib_testing/dl_api_lib_testing/connector/connection_suite.py index 6ae2677fd..f758f0f25 100644 --- a/lib/dl_api_lib_testing/dl_api_lib_testing/connector/connection_suite.py +++ b/lib/dl_api_lib_testing/dl_api_lib_testing/connector/connection_suite.py @@ -6,6 +6,8 @@ from dl_api_client.dsmaker.api.http_sync_base import SyncHttpClientBase from dl_api_lib_testing.connection_base import ConnectionTestBase +from dl_core.us_connection_base import ConnectionBase +from dl_core.us_manager.us_manager_sync import SyncUSManager from dl_testing.regulated_test import RegulatedTestCase @@ -23,6 +25,30 @@ def test_create_connection( ) assert resp.status_code == 200, resp.json + def test_export_connection( + self, + control_api_sync_client: SyncHttpClientBase, + saved_connection_id: str, + bi_headers: Optional[dict[str, str]], + sync_us_manager: SyncUSManager, + ) -> None: + conn = sync_us_manager.get_by_id(saved_connection_id, expected_type=ConnectionBase) + assert isinstance(conn, ConnectionBase) + + resp = control_api_sync_client.get( + url=f"/api/v1/connections/export/{saved_connection_id}", + headers=bi_headers, + ) + + if not conn.allow_export: + assert resp.status_code == 400 + return + + assert resp.status_code == 200, resp.json + if hasattr(conn.data, "password"): + password = resp.json.get("password", None) + assert password == "******" + def test_test_connection( self, control_api_sync_client: SyncHttpClientBase, diff --git a/lib/dl_connector_chyt/dl_connector_chyt/api/api_schema/connection.py b/lib/dl_connector_chyt/dl_connector_chyt/api/api_schema/connection.py index d460e384e..3982f3687 100644 --- a/lib/dl_connector_chyt/dl_connector_chyt/api/api_schema/connection.py +++ b/lib/dl_connector_chyt/dl_connector_chyt/api/api_schema/connection.py @@ -28,7 +28,7 @@ class CHYTConnectionSchema(ConnectionMetaMixin, RawSQLLevelMixin, DataExportForb host = DBHostField(attribute="data.host", required=True, bi_extra=FieldExtra(editable=True)) port = ma.fields.Integer(attribute="data.port", required=True, bi_extra=FieldExtra(editable=True)) - token = secret_string_field(attribute="data.token", required=True, bi_extra=FieldExtra(editable=True)) + token = secret_string_field(attribute="data.token", required=True) alias = alias_string_field(attribute="data.alias") secure = ma.fields.Boolean(attribute="data.secure", bi_extra=FieldExtra(editable=True)) cache_ttl_sec = cache_ttl_field(attribute="data.cache_ttl_sec") diff --git a/lib/dl_connector_clickhouse/dl_connector_clickhouse/api/api_schema/connection.py b/lib/dl_connector_clickhouse/dl_connector_clickhouse/api/api_schema/connection.py index 5c4ff732b..24ed735b2 100644 --- a/lib/dl_connector_clickhouse/dl_connector_clickhouse/api/api_schema/connection.py +++ b/lib/dl_connector_clickhouse/dl_connector_clickhouse/api/api_schema/connection.py @@ -26,7 +26,6 @@ class ClickHouseConnectionSchema( attribute="data.password", required=False, allow_none=True, - bi_extra=FieldExtra(editable=True), ) secure = core_ma_fields.OnOffField(attribute="data.secure", bi_extra=FieldExtra(editable=True)) diff --git a/lib/dl_connector_mssql/dl_connector_mssql/core/us_connection.py b/lib/dl_connector_mssql/dl_connector_mssql/core/us_connection.py index 4e5376287..eb7377ab9 100644 --- a/lib/dl_connector_mssql/dl_connector_mssql/core/us_connection.py +++ b/lib/dl_connector_mssql/dl_connector_mssql/core/us_connection.py @@ -29,6 +29,7 @@ class ConnectionMSSQL(ClassicConnectionSQL): allowed_source_types = frozenset((SOURCE_TYPE_MSSQL_TABLE, SOURCE_TYPE_MSSQL_SUBSELECT)) allow_dashsql: ClassVar[bool] = True allow_cache: ClassVar[bool] = True + allow_export: ClassVar[bool] = True is_always_user_source: ClassVar[bool] = True @attr.s(kw_only=True) diff --git a/lib/dl_connector_mysql/dl_connector_mysql/core/us_connection.py b/lib/dl_connector_mysql/dl_connector_mysql/core/us_connection.py index 852a9e3d6..97df33f26 100644 --- a/lib/dl_connector_mysql/dl_connector_mysql/core/us_connection.py +++ b/lib/dl_connector_mysql/dl_connector_mysql/core/us_connection.py @@ -22,6 +22,7 @@ class ConnectionMySQL(ClassicConnectionSQL): allowed_source_types = frozenset((SOURCE_TYPE_MYSQL_TABLE, SOURCE_TYPE_MYSQL_SUBSELECT)) allow_dashsql: ClassVar[bool] = True allow_cache: ClassVar[bool] = True + allow_export: ClassVar[bool] = True is_always_user_source: ClassVar[bool] = True @attr.s(kw_only=True) diff --git a/lib/dl_connector_oracle/dl_connector_oracle/core/us_connection.py b/lib/dl_connector_oracle/dl_connector_oracle/core/us_connection.py index 20f5badb0..22ec7a088 100644 --- a/lib/dl_connector_oracle/dl_connector_oracle/core/us_connection.py +++ b/lib/dl_connector_oracle/dl_connector_oracle/core/us_connection.py @@ -32,6 +32,7 @@ class ConnectionSQLOracle(ClassicConnectionSQL): allowed_source_types = frozenset((SOURCE_TYPE_ORACLE_TABLE, SOURCE_TYPE_ORACLE_SUBSELECT)) allow_dashsql: ClassVar[bool] = True allow_cache: ClassVar[bool] = True + allow_export: ClassVar[bool] = True is_always_user_source: ClassVar[bool] = True @attr.s(kw_only=True) diff --git a/lib/dl_connector_postgresql/dl_connector_postgresql/core/postgresql_base/us_connection.py b/lib/dl_connector_postgresql/dl_connector_postgresql/core/postgresql_base/us_connection.py index 9ae7b4292..3eabf8ffa 100644 --- a/lib/dl_connector_postgresql/dl_connector_postgresql/core/postgresql_base/us_connection.py +++ b/lib/dl_connector_postgresql/dl_connector_postgresql/core/postgresql_base/us_connection.py @@ -19,6 +19,7 @@ class ConnectionPostgreSQLBase(ClassicConnectionSQL): has_schema = True default_schema_name = "public" + allow_export = True @attr.s(kw_only=True) class DataModel(ClassicConnectionSQL.DataModel): diff --git a/lib/dl_connector_promql/dl_connector_promql/api/api_schema/connection.py b/lib/dl_connector_promql/dl_connector_promql/api/api_schema/connection.py index 24afdb406..062221db7 100644 --- a/lib/dl_connector_promql/dl_connector_promql/api/api_schema/connection.py +++ b/lib/dl_connector_promql/dl_connector_promql/api/api_schema/connection.py @@ -56,7 +56,6 @@ class PromQLConnectionSchema(ConnectionMetaMixin, ClassicSQLConnectionSchema): attribute="data.password", required=False, allow_none=True, - bi_extra=FieldExtra(editable=True), ) path = DBPathField( attribute="data.path", diff --git a/lib/dl_connector_promql/dl_connector_promql/core/us_connection.py b/lib/dl_connector_promql/dl_connector_promql/core/us_connection.py index b104c3cd0..4d63ba4f3 100644 --- a/lib/dl_connector_promql/dl_connector_promql/core/us_connection.py +++ b/lib/dl_connector_promql/dl_connector_promql/core/us_connection.py @@ -17,6 +17,7 @@ class PromQLConnection(ClassicConnectionSQL): allow_cache: ClassVar[bool] = True is_always_user_source: ClassVar[bool] = True allow_dashsql: ClassVar[bool] = True + allow_export: ClassVar[bool] = True source_type = SOURCE_TYPE_PROMQL @attr.s(kw_only=True) diff --git a/lib/dl_connector_snowflake/dl_connector_snowflake/api/api_schema/connection.py b/lib/dl_connector_snowflake/dl_connector_snowflake/api/api_schema/connection.py index 9411a7c3f..6bdbea3d6 100644 --- a/lib/dl_connector_snowflake/dl_connector_snowflake/api/api_schema/connection.py +++ b/lib/dl_connector_snowflake/dl_connector_snowflake/api/api_schema/connection.py @@ -35,7 +35,6 @@ class SnowFlakeConnectionSchema(ConnectionSchema, RawSQLLevelMixin): client_secret = secret_string_field( attribute="data.client_secret", required=True, - bi_extra=FieldExtra(editable=True), ) schema = ma_fields.String( attribute="data.schema", @@ -55,7 +54,6 @@ class SnowFlakeConnectionSchema(ConnectionSchema, RawSQLLevelMixin): refresh_token = secret_string_field( attribute="data.refresh_token", required=False, - bi_extra=FieldExtra(editable=True), ) refresh_token_expire_time = ma_fields.DateTime( attribute="data.refresh_token_expire_time", diff --git a/lib/dl_connector_ydb/dl_connector_ydb/core/ydb/us_connection.py b/lib/dl_connector_ydb/dl_connector_ydb/core/ydb/us_connection.py index 360237f8d..fa0c5894f 100644 --- a/lib/dl_connector_ydb/dl_connector_ydb/core/ydb/us_connection.py +++ b/lib/dl_connector_ydb/dl_connector_ydb/core/ydb/us_connection.py @@ -36,6 +36,7 @@ class YDBConnection(ClassicConnectionSQL): allow_cache: ClassVar[bool] = True is_always_user_source: ClassVar[bool] = True allow_dashsql: ClassVar[bool] = True + allow_export: ClassVar[bool] = True source_type = SOURCE_TYPE_YDB_TABLE diff --git a/lib/dl_core/dl_core/us_connection_base.py b/lib/dl_core/dl_core/us_connection_base.py index 589fcb3ae..d8140fecf 100644 --- a/lib/dl_core/dl_core/us_connection_base.py +++ b/lib/dl_core/dl_core/us_connection_base.py @@ -142,6 +142,7 @@ class ConnectionBase(USEntry, metaclass=abc.ABCMeta): allowed_source_types: ClassVar[Optional[frozenset[DataSourceType]]] = None allow_dashsql: ClassVar[bool] = False allow_cache: ClassVar[bool] = False + allow_export: ClassVar[bool] = False is_always_internal_source: ClassVar[bool] = False is_always_user_source: ClassVar[bool] = False From c21865dfd3bfd56a63d2eb656717e877d1a32ce8 Mon Sep 17 00:00:00 2001 From: Valeria Bulanova Date: Mon, 9 Dec 2024 16:12:18 +0300 Subject: [PATCH 2/3] Fix --- .../db/api/test_connection.py | 30 ++++++++++++++++++- 1 file changed, 29 insertions(+), 1 deletion(-) diff --git a/lib/dl_connector_ydb/dl_connector_ydb_tests/db/api/test_connection.py b/lib/dl_connector_ydb/dl_connector_ydb_tests/db/api/test_connection.py index 89221454b..3c253918b 100644 --- a/lib/dl_connector_ydb/dl_connector_ydb_tests/db/api/test_connection.py +++ b/lib/dl_connector_ydb/dl_connector_ydb_tests/db/api/test_connection.py @@ -1,7 +1,35 @@ +from typing import Optional + +from dl_api_client.dsmaker.api.http_sync_base import SyncHttpClientBase from dl_api_lib_testing.connector.connection_suite import DefaultConnectorConnectionTestSuite +from dl_core.us_connection_base import ConnectionBase +from dl_core.us_manager.us_manager_sync import SyncUSManager from dl_connector_ydb_tests.db.api.base import YDBConnectionTestBase class TestYDBConnection(YDBConnectionTestBase, DefaultConnectorConnectionTestSuite): - pass + # a separate test since password=self.data.token + def test_export_connection( + self, + control_api_sync_client: SyncHttpClientBase, + saved_connection_id: str, + bi_headers: Optional[dict[str, str]], + sync_us_manager: SyncUSManager, + ) -> None: + conn = sync_us_manager.get_by_id(saved_connection_id, expected_type=ConnectionBase) + assert isinstance(conn, ConnectionBase) + + resp = control_api_sync_client.get( + url=f"/api/v1/connections/export/{saved_connection_id}", + headers=bi_headers, + ) + + if not conn.allow_export: + assert resp.status_code == 400 + return + + assert resp.status_code == 200, resp.json + if hasattr(conn.data, "token"): + token = resp.json.get("token", None) + assert token == "******" From d1d6696fb8cfc4dd97d894c16ed36f77e2e1ac1f Mon Sep 17 00:00:00 2001 From: Valeria Bulanova Date: Tue, 10 Dec 2024 11:45:49 +0300 Subject: [PATCH 3/3] Fix --- .../dl_connector_clickhouse/core/clickhouse/us_connection.py | 1 + 1 file changed, 1 insertion(+) diff --git a/lib/dl_connector_clickhouse/dl_connector_clickhouse/core/clickhouse/us_connection.py b/lib/dl_connector_clickhouse/dl_connector_clickhouse/core/clickhouse/us_connection.py index c5cf8e160..e4e0e789a 100644 --- a/lib/dl_connector_clickhouse/dl_connector_clickhouse/core/clickhouse/us_connection.py +++ b/lib/dl_connector_clickhouse/dl_connector_clickhouse/core/clickhouse/us_connection.py @@ -28,6 +28,7 @@ class ConnectionClickhouse(ConnectionClickhouseBase): allowed_source_types = frozenset((SOURCE_TYPE_CH_TABLE, SOURCE_TYPE_CH_SUBSELECT)) allow_dashsql: ClassVar[bool] = True allow_cache: ClassVar[bool] = True + allow_export: ClassVar[bool] = True is_always_user_source: ClassVar[bool] = False # TODO: should be `True`, but need some cleanup for that. def get_data_source_template_templates(self, localizer: Localizer) -> list[DataSourceTemplate]: