castor-extractor 0.15.4__tar.gz → 0.16.3__tar.gz
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Potentially problematic release.
This version of castor-extractor might be problematic. Click here for more details.
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/CHANGELOG.md +15 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/Dockerfile +1 -1
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/PKG-INFO +5 -4
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/dbt/client_test.py +2 -2
- castor_extractor-0.16.3/castor_extractor/utils/dbt/credentials.py +15 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/formatter.py +14 -11
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/formatter_test.py +7 -4
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/domo/client/client.py +1 -1
- castor_extractor-0.16.3/castor_extractor/visualization/powerbi/client/credentials.py +20 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/powerbi/client/rest.py +18 -8
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/powerbi/client/rest_test.py +61 -27
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/assets.py +5 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/client/client.py +10 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/gql_fields.py +30 -9
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/databricks/client.py +19 -2
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/databricks/client_test.py +14 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/databricks/extract.py +4 -1
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/databricks/format.py +5 -4
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/pyproject.toml +25 -15
- castor_extractor-0.15.4/castor_extractor/utils/dbt/credentials.py +0 -16
- castor_extractor-0.15.4/castor_extractor/visualization/powerbi/client/credentials.py +0 -22
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/LICENCE +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/README.md +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/commands/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/commands/extract_bigquery.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/commands/extract_databricks.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/commands/extract_domo.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/commands/extract_looker.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/commands/extract_metabase_api.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/commands/extract_metabase_db.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/commands/extract_mode.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/commands/extract_mysql.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/commands/extract_postgres.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/commands/extract_powerbi.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/commands/extract_qlik.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/commands/extract_redshift.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/commands/extract_salesforce_reporting.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/commands/extract_sigma.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/commands/extract_snowflake.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/commands/extract_sqlserver.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/commands/extract_tableau.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/commands/file_check.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/commands/upload.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/file_checker/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/file_checker/column.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/file_checker/column_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/file_checker/constants.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/file_checker/enums.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/file_checker/file.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/file_checker/file_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/file_checker/file_test_users.csv +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/file_checker/file_test_users_valid.csv +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/file_checker/templates/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/file_checker/templates/generic_warehouse.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/logger.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/types.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/uploader/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/uploader/constant.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/uploader/env.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/uploader/env_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/uploader/upload.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/uploader/upload_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/uploader/utils.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/client/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/client/abstract.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/client/api.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/client/api_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/client/postgres.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/client/query.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/client/uri.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/client/uri_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/collection.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/constants.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/dbt/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/dbt/assets.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/dbt/client.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/deprecate.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/env.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/files.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/files_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/formatter_test.csv +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/formatter_test.json +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/json_stream_write.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/load.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/object.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/object_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/pager/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/pager/pager.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/pager/pager_on_id.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/pager/pager_on_id_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/pager/pager_on_token.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/pager/pager_on_token_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/pager/pager_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/retry.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/retry_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/safe.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/safe_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/store.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/string.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/string_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/time.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/time_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/type.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/validation.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/validation_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/write.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/domo/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/domo/assets.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/domo/client/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/domo/client/client_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/domo/client/credentials.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/domo/client/endpoints.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/domo/client/pagination.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/domo/client/pagination_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/domo/constants.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/domo/extract.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/looker/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/looker/api/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/looker/api/client.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/looker/api/client_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/looker/api/constants.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/looker/api/sdk.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/looker/api/sdk_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/looker/api/utils.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/looker/assets.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/looker/constant.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/looker/env.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/looker/extract.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/looker/fields.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/looker/fields_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/looker/multithreading.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/looker/parameters.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/metabase/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/metabase/assets.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/metabase/client/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/metabase/client/api/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/metabase/client/api/client.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/metabase/client/api/client_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/metabase/client/api/credentials.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/metabase/client/db/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/metabase/client/db/client.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/metabase/client/db/credentials.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/metabase/client/db/queries/.sqlfluff +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/metabase/client/db/queries/base_url.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/metabase/client/db/queries/card.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/metabase/client/db/queries/collection.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/metabase/client/db/queries/dashboard.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/metabase/client/db/queries/dashboard_cards.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/metabase/client/db/queries/database.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/metabase/client/db/queries/table.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/metabase/client/db/queries/user.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/metabase/client/decryption.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/metabase/client/decryption_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/metabase/client/shared.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/metabase/errors.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/metabase/extract.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/metabase/types.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/mode/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/mode/assets.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/mode/client/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/mode/client/client.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/mode/client/client_test.json +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/mode/client/client_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/mode/client/constants.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/mode/client/credentials.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/mode/errors.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/mode/extract.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/powerbi/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/powerbi/assets.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/powerbi/client/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/powerbi/client/constants.py +3 -3
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/powerbi/client/credentials_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/powerbi/client/utils.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/powerbi/client/utils_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/powerbi/extract.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/qlik/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/qlik/assets.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/qlik/client/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/qlik/client/constants.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/qlik/client/engine/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/qlik/client/engine/client.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/qlik/client/engine/constants.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/qlik/client/engine/error.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/qlik/client/engine/error_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/qlik/client/engine/json_rpc.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/qlik/client/engine/json_rpc_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/qlik/client/engine/websocket.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/qlik/client/master.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/qlik/client/rest.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/qlik/client/rest_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/qlik/constants.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/qlik/extract.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/salesforce_reporting/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/salesforce_reporting/assets.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/salesforce_reporting/client/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/salesforce_reporting/client/constants.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/salesforce_reporting/client/credentials.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/salesforce_reporting/client/credentials_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/salesforce_reporting/client/rest.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/salesforce_reporting/client/soql.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/salesforce_reporting/extract.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/sigma/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/sigma/assets.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/sigma/client/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/sigma/client/client.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/sigma/client/client_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/sigma/client/credentials.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/sigma/client/endpoints.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/sigma/client/pagination.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/sigma/constants.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/sigma/extract.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/client/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/client/client_utils.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/client/credentials.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/client/project.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/client/safe_mode.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/constants.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/errors.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/extract.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/tests/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/tests/unit/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/tests/unit/assets/graphql/metadata/metadata_1_get.json +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/tests/unit/assets/graphql/metadata/metadata_2_get.json +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/tests/unit/assets/rest_api/auth.xml +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/tests/unit/assets/rest_api/project_get.xml +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/tests/unit/assets/rest_api/user_get.xml +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/tests/unit/assets/rest_api/view_get_usage.xml +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/tests/unit/assets/rest_api/workbook_get.xml +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/tests/unit/graphql/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/tests/unit/graphql/paginated_object_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/tests/unit/rest_api/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/tests/unit/rest_api/auth_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/tests/unit/rest_api/credentials_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/tests/unit/rest_api/projects_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/tests/unit/rest_api/usages_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/tests/unit/rest_api/users_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/tests/unit/rest_api/workbooks_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/tests/unit/utils/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/tests/unit/utils/env_key.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/tsc_fields.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/types.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/usage.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/abstract/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/abstract/asset.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/abstract/asset_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/abstract/extract.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/abstract/query.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/abstract/time_filter.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/abstract/time_filter_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/bigquery/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/bigquery/client.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/bigquery/client_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/bigquery/credentials.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/bigquery/extract.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/bigquery/queries/.sqlfluff +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/bigquery/queries/column.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/bigquery/queries/cte/sharded.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/bigquery/queries/database.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/bigquery/queries/query.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/bigquery/queries/schema.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/bigquery/queries/table.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/bigquery/queries/table_with_tags.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/bigquery/queries/user.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/bigquery/queries/view_ddl.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/bigquery/query.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/bigquery/types.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/databricks/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/databricks/credentials.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/databricks/format_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/databricks/types.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/mysql/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/mysql/client.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/mysql/client_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/mysql/extract.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/mysql/queries/.sqlfluff +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/mysql/queries/column.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/mysql/queries/database.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/mysql/queries/query.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/mysql/queries/schema.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/mysql/queries/table.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/mysql/queries/user.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/mysql/queries/view_ddl.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/mysql/query.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/postgres/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/postgres/extract.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/postgres/queries/.sqlfluff +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/postgres/queries/column.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/postgres/queries/database.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/postgres/queries/group.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/postgres/queries/schema.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/postgres/queries/table.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/postgres/queries/user.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/postgres/query.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/redshift/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/redshift/client.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/redshift/client_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/redshift/extract.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/redshift/queries/.sqlfluff +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/redshift/queries/column.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/redshift/queries/database.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/redshift/queries/group.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/redshift/queries/query.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/redshift/queries/schema.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/redshift/queries/table.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/redshift/queries/table_freshness.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/redshift/queries/user.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/redshift/queries/view_ddl.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/redshift/query.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/snowflake/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/snowflake/client.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/snowflake/client_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/snowflake/credentials.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/snowflake/credentials_test.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/snowflake/extract.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/snowflake/queries/.sqlfluff +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/snowflake/queries/column.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/snowflake/queries/column_lineage.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/snowflake/queries/database.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/snowflake/queries/grant_to_role.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/snowflake/queries/grant_to_user.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/snowflake/queries/query.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/snowflake/queries/role.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/snowflake/queries/schema.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/snowflake/queries/table.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/snowflake/queries/user.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/snowflake/queries/view_ddl.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/snowflake/query.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/sqlserver/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/sqlserver/client.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/sqlserver/extract.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/sqlserver/queries/.sqlfluff +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/sqlserver/queries/column.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/sqlserver/queries/database.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/sqlserver/queries/schema.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/sqlserver/queries/table.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/sqlserver/queries/user.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/sqlserver/query.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/synapse/__init__.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/synapse/extract.py +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/synapse/queries/.sqlfluff +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/synapse/queries/column.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/synapse/queries/database.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/synapse/queries/query.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/synapse/queries/schema.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/synapse/queries/table.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/synapse/queries/user.sql +0 -0
- {castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/synapse/queries/view_ddl.sql +0 -0
|
@@ -1,5 +1,20 @@
|
|
|
1
1
|
# Changelog
|
|
2
2
|
|
|
3
|
+
## 0.16.3 - 2024-04-24
|
|
4
|
+
|
|
5
|
+
* Databricks: Extract table owners
|
|
6
|
+
|
|
7
|
+
## 0.16.2 - 2024-04-09
|
|
8
|
+
|
|
9
|
+
* PowerBI: Extract pages from report
|
|
10
|
+
|
|
11
|
+
## 0.16.1 - 2024-04-02
|
|
12
|
+
|
|
13
|
+
* Systematically escape nul bytes on CSV write
|
|
14
|
+
|
|
15
|
+
## 0.16.0 - 2024-03-26
|
|
16
|
+
|
|
17
|
+
* Use pydantic v2
|
|
3
18
|
|
|
4
19
|
## 0.15.4 - 2024-03-25
|
|
5
20
|
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
Metadata-Version: 2.1
|
|
2
2
|
Name: castor-extractor
|
|
3
|
-
Version: 0.
|
|
3
|
+
Version: 0.16.3
|
|
4
4
|
Summary: Extract your metadata assets.
|
|
5
5
|
Home-page: https://www.castordoc.com/
|
|
6
6
|
License: EULA
|
|
@@ -16,6 +16,7 @@ Classifier: Programming Language :: Python :: 3.10
|
|
|
16
16
|
Classifier: Programming Language :: Python :: 3.11
|
|
17
17
|
Provides-Extra: all
|
|
18
18
|
Provides-Extra: bigquery
|
|
19
|
+
Provides-Extra: dbt
|
|
19
20
|
Provides-Extra: looker
|
|
20
21
|
Provides-Extra: metabase
|
|
21
22
|
Provides-Extra: mysql
|
|
@@ -26,10 +27,9 @@ Provides-Extra: redshift
|
|
|
26
27
|
Provides-Extra: snowflake
|
|
27
28
|
Provides-Extra: sqlserver
|
|
28
29
|
Provides-Extra: tableau
|
|
29
|
-
Requires-Dist: click (>=8.1,<8.2)
|
|
30
30
|
Requires-Dist: cryptography (>=41.0.5) ; extra == "snowflake"
|
|
31
31
|
Requires-Dist: google-api-core (>=2.1.1,<3.0.0)
|
|
32
|
-
Requires-Dist: google-auth (>=
|
|
32
|
+
Requires-Dist: google-auth (>=2,<3)
|
|
33
33
|
Requires-Dist: google-cloud-core (>=2.1.0,<3.0.0)
|
|
34
34
|
Requires-Dist: google-cloud-storage (>=2,<3)
|
|
35
35
|
Requires-Dist: google-resumable-media (>=2.0.3,<3.0.0)
|
|
@@ -38,7 +38,8 @@ Requires-Dist: looker-sdk (>=22.4.0,<=23.0.0) ; extra == "looker" or extra == "a
|
|
|
38
38
|
Requires-Dist: msal (>=1.20.0,<2.0.0) ; extra == "powerbi" or extra == "all"
|
|
39
39
|
Requires-Dist: psycopg2-binary (>=2.0.0,<3.0.0) ; extra == "metabase" or extra == "postgres" or extra == "redshift" or extra == "all"
|
|
40
40
|
Requires-Dist: pycryptodome (>=3.0.0,<4.0.0) ; extra == "metabase" or extra == "all"
|
|
41
|
-
Requires-Dist: pydantic (>=
|
|
41
|
+
Requires-Dist: pydantic (>=2.6,<3.0)
|
|
42
|
+
Requires-Dist: pydantic-settings (>=2.2,<3.0)
|
|
42
43
|
Requires-Dist: pymssql (>=2.2.11,<3.0.0) ; extra == "sqlserver" or extra == "all"
|
|
43
44
|
Requires-Dist: pymysql[rsa] (>=1.1.0,<2.0.0) ; extra == "mysql" or extra == "all"
|
|
44
45
|
Requires-Dist: python-dateutil (>=2.0.0,<=3.0.0)
|
{castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/dbt/client_test.py
RENAMED
|
@@ -89,7 +89,7 @@ def test_DbtClient_list_job_identifiers():
|
|
|
89
89
|
|
|
90
90
|
with patch(infer_path, return_value=40), patch(call_path) as mocked_call:
|
|
91
91
|
mocked_call.return_value = jobs
|
|
92
|
-
credentials = DbtCredentials(token="some-token", job_id=1)
|
|
92
|
+
credentials = DbtCredentials(token="some-token", job_id="1")
|
|
93
93
|
dbt_client = DbtClient(credentials=credentials)
|
|
94
94
|
|
|
95
95
|
jobs_ids = dbt_client.list_job_identifiers()
|
|
@@ -103,7 +103,7 @@ def test_DbtClient_fetch_artifacts():
|
|
|
103
103
|
url = "https://cloud.getdbt.com/api/v2/accounts/40/runs/{}/artifacts/{}"
|
|
104
104
|
|
|
105
105
|
with patch(infer_path, return_value=40), patch(call_path) as mocked_call:
|
|
106
|
-
credentials = DbtCredentials(token="some-token", job_id=1)
|
|
106
|
+
credentials = DbtCredentials(token="some-token", job_id="1")
|
|
107
107
|
dbt_client = DbtClient(credentials=credentials)
|
|
108
108
|
|
|
109
109
|
dbt_client.fetch_run_results(run_id)
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
from pydantic import Field
|
|
2
|
+
from pydantic_settings import BaseSettings, SettingsConfigDict
|
|
3
|
+
|
|
4
|
+
DEFAULT_DBT_CLOUD_URL = "https://cloud.getdbt.com"
|
|
5
|
+
|
|
6
|
+
|
|
7
|
+
class DbtCredentials(BaseSettings):
|
|
8
|
+
"""dbt credentials: host has default value"""
|
|
9
|
+
|
|
10
|
+
host: str = Field(
|
|
11
|
+
default=DEFAULT_DBT_CLOUD_URL, validation_alias="CASTOR_DBT_HOST"
|
|
12
|
+
)
|
|
13
|
+
job_id: str = Field(..., validation_alias="CASTOR_DBT_JOB_ID")
|
|
14
|
+
token: str = Field(..., validation_alias="CASTOR_DBT_TOKEN")
|
|
15
|
+
model_config = SettingsConfigDict(extra="ignore", populate_by_name=True)
|
|
@@ -30,7 +30,14 @@ def _header(row: dict) -> Sequence[str]:
|
|
|
30
30
|
|
|
31
31
|
|
|
32
32
|
def _scalar(value: Any) -> ScalarValue:
|
|
33
|
-
if isinstance(value,
|
|
33
|
+
if isinstance(value, str):
|
|
34
|
+
if "\x00" in value: # infrequent error caused by bad encoding
|
|
35
|
+
value = remove_unsupported_byte(value)
|
|
36
|
+
logger.warning("Removed unsupported byte to write to csv")
|
|
37
|
+
return value
|
|
38
|
+
|
|
39
|
+
return value
|
|
40
|
+
if isinstance(value, (int, float)):
|
|
34
41
|
return value
|
|
35
42
|
if isinstance(value, (date, datetime)):
|
|
36
43
|
return value.isoformat()
|
|
@@ -46,11 +53,11 @@ def _row(header: Sequence[str], row: dict) -> List[ScalarValue]:
|
|
|
46
53
|
return [_scalar(row.get(h)) for h in header]
|
|
47
54
|
|
|
48
55
|
|
|
49
|
-
def remove_unsupported_byte(
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
56
|
+
def remove_unsupported_byte(element: ScalarValue) -> ScalarValue:
|
|
57
|
+
if not isinstance(element, str):
|
|
58
|
+
return element
|
|
59
|
+
|
|
60
|
+
return re.sub("\x00", "", element)
|
|
54
61
|
|
|
55
62
|
|
|
56
63
|
def to_string_array(arr_json: str) -> List[str]:
|
|
@@ -85,11 +92,7 @@ def to_csv(buffer: IO[str], data: Iterable[dict]) -> bool:
|
|
|
85
92
|
header = _header(row)
|
|
86
93
|
writer.writerow(header)
|
|
87
94
|
converted = _row(header, row)
|
|
88
|
-
|
|
89
|
-
writer.writerow(converted)
|
|
90
|
-
except csv.Error: # infrequent error caused by bad encoding
|
|
91
|
-
writer.writerow(remove_unsupported_byte(converted))
|
|
92
|
-
logger.warning("Removed unsupported byte to write to csv")
|
|
95
|
+
writer.writerow(converted)
|
|
93
96
|
return True
|
|
94
97
|
|
|
95
98
|
|
{castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/utils/formatter_test.py
RENAMED
|
@@ -68,7 +68,10 @@ def test__json_formatter():
|
|
|
68
68
|
_test(formatter)
|
|
69
69
|
|
|
70
70
|
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
71
|
+
@pytest.mark.parametrize(
|
|
72
|
+
"element, expected_output",
|
|
73
|
+
[(1, 1), ("foo", "foo"), ("bar\x00bie", "barbie")],
|
|
74
|
+
)
|
|
75
|
+
def test__remove_unsupported_byte(element, expected_output):
|
|
76
|
+
cleaned = remove_unsupported_byte(element)
|
|
77
|
+
assert cleaned == expected_output
|
|
@@ -14,7 +14,7 @@ RawData = Iterator[dict]
|
|
|
14
14
|
|
|
15
15
|
DOMO_PUBLIC_URL = "https://api.domo.com"
|
|
16
16
|
FORMAT = "%Y-%m-%d %I:%M:%S %p"
|
|
17
|
-
DEFAULT_TIMEOUT =
|
|
17
|
+
DEFAULT_TIMEOUT = 500
|
|
18
18
|
TOKEN_EXPIRATION_SECONDS = timedelta(seconds=3000) # auth token lasts 1 hour
|
|
19
19
|
|
|
20
20
|
IGNORED_ERROR_CODES = (
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
from dataclasses import field
|
|
2
|
+
from typing import List, Optional
|
|
3
|
+
|
|
4
|
+
from pydantic.dataclasses import dataclass
|
|
5
|
+
|
|
6
|
+
from .constants import Urls
|
|
7
|
+
|
|
8
|
+
|
|
9
|
+
@dataclass
|
|
10
|
+
class Credentials:
|
|
11
|
+
"""Class to handle PowerBI rest API permissions"""
|
|
12
|
+
|
|
13
|
+
client_id: str
|
|
14
|
+
tenant_id: str
|
|
15
|
+
secret: str = field(metadata={"sensitive": True})
|
|
16
|
+
scopes: Optional[List[str]] = None
|
|
17
|
+
|
|
18
|
+
def __post_init__(self):
|
|
19
|
+
if self.scopes is None:
|
|
20
|
+
self.scopes = [Urls.DEFAULT_SCOPE]
|
|
@@ -64,19 +64,15 @@ class Client:
|
|
|
64
64
|
|
|
65
65
|
def __init__(self, credentials: Credentials):
|
|
66
66
|
self.creds = credentials
|
|
67
|
-
|
|
68
|
-
def _access_token(self) -> dict:
|
|
69
67
|
client_app = f"{Urls.CLIENT_APP_BASE}{self.creds.tenant_id}"
|
|
70
|
-
app = msal.ConfidentialClientApplication(
|
|
68
|
+
self.app = msal.ConfidentialClientApplication(
|
|
71
69
|
client_id=self.creds.client_id,
|
|
72
70
|
authority=client_app,
|
|
73
71
|
client_credential=self.creds.secret,
|
|
74
72
|
)
|
|
75
73
|
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
if not token:
|
|
79
|
-
token = app.acquire_token_for_client(scopes=self.creds.scopes)
|
|
74
|
+
def _access_token(self) -> dict:
|
|
75
|
+
token = self.app.acquire_token_for_client(scopes=self.creds.scopes)
|
|
80
76
|
|
|
81
77
|
if Keys.ACCESS_TOKEN not in token:
|
|
82
78
|
raise ValueError(f"No access token in token response: {token}")
|
|
@@ -248,7 +244,17 @@ class Client:
|
|
|
248
244
|
Returns a list of reports for the organization.
|
|
249
245
|
https://learn.microsoft.com/en-us/rest/api/power-bi/admin/reports-get-reports-as-admin
|
|
250
246
|
"""
|
|
251
|
-
|
|
247
|
+
reports = self._get(Urls.REPORTS)[Keys.VALUE]
|
|
248
|
+
for report in reports:
|
|
249
|
+
report_id = report.get("id")
|
|
250
|
+
try:
|
|
251
|
+
url = Urls.REPORTS + f"/{report_id}/pages"
|
|
252
|
+
pages = self._get(url)[Keys.VALUE]
|
|
253
|
+
report["pages"] = pages
|
|
254
|
+
except (requests.HTTPError, requests.exceptions.Timeout) as e:
|
|
255
|
+
logger.debug(e)
|
|
256
|
+
continue
|
|
257
|
+
return reports
|
|
252
258
|
|
|
253
259
|
def _dashboards(self) -> List[Dict]:
|
|
254
260
|
"""
|
|
@@ -273,6 +279,10 @@ class Client:
|
|
|
273
279
|
self._wait_for_scan_result(scan_id)
|
|
274
280
|
yield self._get_scan(scan_id)
|
|
275
281
|
|
|
282
|
+
def test_connection(self) -> None:
|
|
283
|
+
"""Use credentials & verify requesting the API doesn't raise an error"""
|
|
284
|
+
self._header()
|
|
285
|
+
|
|
276
286
|
def fetch(
|
|
277
287
|
self,
|
|
278
288
|
asset: PowerBiAsset,
|
|
@@ -31,7 +31,6 @@ def test__access_token(mock_app):
|
|
|
31
31
|
# init mocks
|
|
32
32
|
valid_response = {"access_token": "mock_token"}
|
|
33
33
|
returning_valid_token = Mock(return_value=valid_response)
|
|
34
|
-
mock_app.return_value.acquire_token_silent = Mock(return_value=None)
|
|
35
34
|
mock_app.return_value.acquire_token_for_client = returning_valid_token
|
|
36
35
|
|
|
37
36
|
# init client
|
|
@@ -40,30 +39,29 @@ def test__access_token(mock_app):
|
|
|
40
39
|
# generated token
|
|
41
40
|
assert client._access_token() == valid_response
|
|
42
41
|
|
|
43
|
-
# via silent endpoint token
|
|
44
|
-
mock_app.return_value.acquire_token_silent = returning_valid_token
|
|
45
|
-
mock_app.return_value.acquire_token_for_client = None
|
|
46
|
-
assert client._access_token() == valid_response
|
|
47
|
-
|
|
48
42
|
# token missing in response
|
|
49
43
|
invalid_response = {"not_access_token": "666"}
|
|
50
44
|
returning_invalid_token = Mock(return_value=invalid_response)
|
|
51
|
-
mock_app.return_value.
|
|
45
|
+
mock_app.return_value.acquire_token_for_client = returning_invalid_token
|
|
52
46
|
|
|
53
47
|
with pytest.raises(ValueError):
|
|
54
48
|
client._access_token()
|
|
55
49
|
|
|
56
50
|
|
|
51
|
+
@patch.object(msal, "ConfidentialClientApplication")
|
|
57
52
|
@patch.object(Client, "_access_token")
|
|
58
|
-
def test__headers(mock_access_token):
|
|
53
|
+
def test__headers(mock_access_token, mock_app):
|
|
54
|
+
mock_app.return_value = None
|
|
59
55
|
client = _client()
|
|
60
56
|
mock_access_token.return_value = {Keys.ACCESS_TOKEN: "666"}
|
|
61
57
|
assert client._header() == {"Authorization": "Bearer 666"}
|
|
62
58
|
|
|
63
59
|
|
|
60
|
+
@patch.object(msal, "ConfidentialClientApplication")
|
|
64
61
|
@patch("requests.request")
|
|
65
62
|
@patch.object(Client, "_access_token")
|
|
66
|
-
def test__get(mocked_access_token, mocked_request):
|
|
63
|
+
def test__get(mocked_access_token, mocked_request, mock_app):
|
|
64
|
+
mock_app.return_value = None
|
|
67
65
|
client = _client()
|
|
68
66
|
mocked_access_token.return_value = {Keys.ACCESS_TOKEN: "666"}
|
|
69
67
|
fact = {"fact": "Approximately 24 cat skins can make a coat.", "length": 43}
|
|
@@ -81,9 +79,11 @@ def test__get(mocked_access_token, mocked_request):
|
|
|
81
79
|
result = client._get("https/whatev.er")
|
|
82
80
|
|
|
83
81
|
|
|
82
|
+
@patch.object(msal, "ConfidentialClientApplication")
|
|
84
83
|
@patch("requests.request")
|
|
85
84
|
@patch.object(Client, "_access_token")
|
|
86
|
-
def test__workspace_ids(_, mocked_request):
|
|
85
|
+
def test__workspace_ids(_, mocked_request, mock_app):
|
|
86
|
+
mock_app.return_value = None
|
|
87
87
|
client = _client()
|
|
88
88
|
mocked_request.return_value = Mock(
|
|
89
89
|
json=lambda: [{"id": 1000}, {"id": 1001}, {"id": 1003}],
|
|
@@ -112,9 +112,11 @@ def test__workspace_ids(_, mocked_request):
|
|
|
112
112
|
)
|
|
113
113
|
|
|
114
114
|
|
|
115
|
+
@patch.object(msal, "ConfidentialClientApplication")
|
|
115
116
|
@patch("requests.request")
|
|
116
117
|
@patch.object(Client, "_access_token")
|
|
117
|
-
def test__post_default(_, mocked_request):
|
|
118
|
+
def test__post_default(_, mocked_request, mock_app):
|
|
119
|
+
mock_app.return_value = None
|
|
118
120
|
client = _client()
|
|
119
121
|
url = "https://estcequecestbientotleweekend.fr/"
|
|
120
122
|
params = QueryParams.METADATA_SCAN
|
|
@@ -129,9 +131,11 @@ def test__post_default(_, mocked_request):
|
|
|
129
131
|
)
|
|
130
132
|
|
|
131
133
|
|
|
134
|
+
@patch.object(msal, "ConfidentialClientApplication")
|
|
132
135
|
@patch("requests.request")
|
|
133
136
|
@patch.object(Client, "_access_token")
|
|
134
|
-
def test__post_with_processor(_, mocked_request):
|
|
137
|
+
def test__post_with_processor(_, mocked_request, mock_app):
|
|
138
|
+
mock_app.return_value = None
|
|
135
139
|
client = _client()
|
|
136
140
|
url = "https://estcequecestbientotleweekend.fr/"
|
|
137
141
|
params = QueryParams.METADATA_SCAN
|
|
@@ -146,9 +150,11 @@ def test__post_with_processor(_, mocked_request):
|
|
|
146
150
|
assert result == 1000
|
|
147
151
|
|
|
148
152
|
|
|
153
|
+
@patch.object(msal, "ConfidentialClientApplication")
|
|
149
154
|
@patch("requests.request")
|
|
150
155
|
@patch.object(Client, "_access_token")
|
|
151
|
-
def test__datasets(_, mocked_request):
|
|
156
|
+
def test__datasets(_, mocked_request, mock_app):
|
|
157
|
+
mock_app.return_value = None
|
|
152
158
|
client = _client()
|
|
153
159
|
mocked_request.return_value = Mock(
|
|
154
160
|
json=lambda: {"value": [{"id": 1, "type": "dataset"}]},
|
|
@@ -164,27 +170,50 @@ def test__datasets(_, mocked_request):
|
|
|
164
170
|
assert datasets == [{"id": 1, "type": "dataset"}]
|
|
165
171
|
|
|
166
172
|
|
|
173
|
+
@patch.object(msal, "ConfidentialClientApplication")
|
|
167
174
|
@patch("requests.request")
|
|
168
175
|
@patch.object(Client, "_access_token")
|
|
169
|
-
def test__reports(_, mocked_request):
|
|
176
|
+
def test__reports(_, mocked_request, mock_app):
|
|
177
|
+
mock_app.return_value = None
|
|
170
178
|
client = _client()
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
179
|
+
page_url = f"{Urls.REPORTS}/1/pages"
|
|
180
|
+
calls = [
|
|
181
|
+
call(GET, Urls.REPORTS, data=None, headers=ANY, params=None),
|
|
182
|
+
call(
|
|
183
|
+
GET,
|
|
184
|
+
page_url,
|
|
185
|
+
data=None,
|
|
186
|
+
headers=ANY,
|
|
187
|
+
params=None,
|
|
188
|
+
),
|
|
189
|
+
]
|
|
190
|
+
mocked_request.side_effect = [
|
|
191
|
+
Mock(json=lambda: {"value": [{"id": 1, "type": "report"}]}),
|
|
192
|
+
Mock(
|
|
193
|
+
json=lambda: {
|
|
194
|
+
"value": [
|
|
195
|
+
{"name": "page_name", "displayName": "page", "order": 0}
|
|
196
|
+
]
|
|
197
|
+
}
|
|
198
|
+
),
|
|
199
|
+
]
|
|
174
200
|
reports = client._reports()
|
|
175
|
-
mocked_request.
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
201
|
+
mocked_request.assert_has_calls(calls)
|
|
202
|
+
|
|
203
|
+
assert reports == [
|
|
204
|
+
{
|
|
205
|
+
"id": 1,
|
|
206
|
+
"type": "report",
|
|
207
|
+
"pages": [{"name": "page_name", "displayName": "page", "order": 0}],
|
|
208
|
+
}
|
|
209
|
+
]
|
|
183
210
|
|
|
184
211
|
|
|
212
|
+
@patch.object(msal, "ConfidentialClientApplication")
|
|
185
213
|
@patch("requests.request")
|
|
186
214
|
@patch.object(Client, "_access_token")
|
|
187
|
-
def test__dashboards(_, mocked_request):
|
|
215
|
+
def test__dashboards(_, mocked_request, mock_app):
|
|
216
|
+
mock_app.return_value = None
|
|
188
217
|
client = _client()
|
|
189
218
|
mocked_request.return_value = Mock(
|
|
190
219
|
json=lambda: {"value": [{"id": 1, "type": "dashboard"}]},
|
|
@@ -200,6 +229,7 @@ def test__dashboards(_, mocked_request):
|
|
|
200
229
|
assert dashboards == [{"id": 1, "type": "dashboard"}]
|
|
201
230
|
|
|
202
231
|
|
|
232
|
+
@patch.object(msal, "ConfidentialClientApplication")
|
|
203
233
|
@patch.object(Client, "_workspace_ids")
|
|
204
234
|
@patch.object(Client, "_create_scan")
|
|
205
235
|
@patch.object(Client, "_wait_for_scan_result")
|
|
@@ -209,7 +239,9 @@ def test__metadata(
|
|
|
209
239
|
mocked_wait,
|
|
210
240
|
mocked_create_scan,
|
|
211
241
|
mocked_workspace_ids,
|
|
242
|
+
mock_app,
|
|
212
243
|
):
|
|
244
|
+
mock_app.return_value = None
|
|
213
245
|
mocked_workspace_ids.return_value = list(range(200))
|
|
214
246
|
mocked_create_scan.return_value = 314
|
|
215
247
|
mocked_wait.return_value = True
|
|
@@ -240,8 +272,10 @@ _CALLS = [
|
|
|
240
272
|
]
|
|
241
273
|
|
|
242
274
|
|
|
275
|
+
@patch.object(msal, "ConfidentialClientApplication")
|
|
243
276
|
@patch.object(Client, "_call")
|
|
244
|
-
def test__activity_events(mocked):
|
|
277
|
+
def test__activity_events(mocked, mock_app):
|
|
278
|
+
mock_app.return_value = None
|
|
245
279
|
client = _client()
|
|
246
280
|
mocked.side_effect = _CALLS
|
|
247
281
|
|
{castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/visualization/tableau/assets.py
RENAMED
|
@@ -12,10 +12,12 @@ class TableauAsset(ExternalAsset):
|
|
|
12
12
|
CUSTOM_SQL_TABLE = "custom_sql_tables"
|
|
13
13
|
CUSTOM_SQL_QUERY = "custom_sql_queries"
|
|
14
14
|
DASHBOARD = "dashboards"
|
|
15
|
+
DASHBOARD_SHEET = "dashboards_sheets"
|
|
15
16
|
DATASOURCE = "datasources"
|
|
16
17
|
FIELD = "fields"
|
|
17
18
|
PROJECT = "projects"
|
|
18
19
|
PUBLISHED_DATASOURCE = "published_datasources"
|
|
20
|
+
SHEET = "sheets"
|
|
19
21
|
USAGE = "views"
|
|
20
22
|
USER = "users"
|
|
21
23
|
WORKBOOK = "workbooks"
|
|
@@ -25,7 +27,9 @@ class TableauAsset(ExternalAsset):
|
|
|
25
27
|
def optional(cls) -> Set["TableauAsset"]:
|
|
26
28
|
return {
|
|
27
29
|
TableauAsset.DASHBOARD,
|
|
30
|
+
TableauAsset.DASHBOARD_SHEET,
|
|
28
31
|
TableauAsset.FIELD,
|
|
32
|
+
TableauAsset.SHEET,
|
|
29
33
|
TableauAsset.PUBLISHED_DATASOURCE,
|
|
30
34
|
}
|
|
31
35
|
|
|
@@ -42,4 +46,5 @@ class TableauGraphqlAsset(Enum):
|
|
|
42
46
|
DASHBOARD = "dashboards"
|
|
43
47
|
DATASOURCE = "datasources"
|
|
44
48
|
GROUP_FIELD = "groupFields"
|
|
49
|
+
SHEETS = "sheets"
|
|
45
50
|
WORKBOOK_TO_DATASOURCE = "workbooks"
|
|
@@ -173,6 +173,13 @@ class ApiClient:
|
|
|
173
173
|
TableauAsset.DASHBOARD,
|
|
174
174
|
)
|
|
175
175
|
|
|
176
|
+
def _fetch_sheets(self) -> SerializedAsset:
|
|
177
|
+
"""Fetches sheets"""
|
|
178
|
+
|
|
179
|
+
return self._fetch_paginated_objects(
|
|
180
|
+
TableauAsset.SHEET,
|
|
181
|
+
)
|
|
182
|
+
|
|
176
183
|
def _fetch_paginated_objects(self, asset: TableauAsset) -> SerializedAsset:
|
|
177
184
|
"""Fetches paginated objects"""
|
|
178
185
|
|
|
@@ -203,6 +210,9 @@ class ApiClient:
|
|
|
203
210
|
if asset == TableauAsset.PUBLISHED_DATASOURCE:
|
|
204
211
|
assets = self._fetch_published_datasources()
|
|
205
212
|
|
|
213
|
+
if asset == TableauAsset.SHEET:
|
|
214
|
+
assets = self._fetch_sheets()
|
|
215
|
+
|
|
206
216
|
if asset == TableauAsset.USAGE:
|
|
207
217
|
assets = self._fetch_usages(self._safe_mode)
|
|
208
218
|
|
|
@@ -111,15 +111,15 @@ class GQLQueryFields(Enum):
|
|
|
111
111
|
"""
|
|
112
112
|
|
|
113
113
|
DASHBOARDS: str = """
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
114
|
+
id
|
|
115
|
+
name
|
|
116
|
+
path
|
|
117
|
+
tags {
|
|
118
|
+
name
|
|
119
|
+
}
|
|
120
|
+
workbook {
|
|
121
|
+
luid # to retrieve the parent
|
|
122
|
+
}
|
|
123
123
|
"""
|
|
124
124
|
|
|
125
125
|
DATASOURCE: str = """
|
|
@@ -160,6 +160,21 @@ class GQLQueryFields(Enum):
|
|
|
160
160
|
role
|
|
161
161
|
"""
|
|
162
162
|
|
|
163
|
+
SHEET: str = """
|
|
164
|
+
containedInDashboards {
|
|
165
|
+
id
|
|
166
|
+
}
|
|
167
|
+
id
|
|
168
|
+
index
|
|
169
|
+
name
|
|
170
|
+
upstreamFields{
|
|
171
|
+
name
|
|
172
|
+
}
|
|
173
|
+
workbook {
|
|
174
|
+
luid
|
|
175
|
+
}
|
|
176
|
+
"""
|
|
177
|
+
|
|
163
178
|
WORKBOOK_TO_DATASOURCE: str = """
|
|
164
179
|
luid
|
|
165
180
|
id
|
|
@@ -219,6 +234,12 @@ QUERY_FIELDS: Dict[TableauAsset, QueryInfo] = {
|
|
|
219
234
|
OBJECT_TYPE: TableauGraphqlAsset.GROUP_FIELD,
|
|
220
235
|
},
|
|
221
236
|
],
|
|
237
|
+
TableauAsset.SHEET: [
|
|
238
|
+
{
|
|
239
|
+
FIELDS: GQLQueryFields.SHEET,
|
|
240
|
+
OBJECT_TYPE: TableauGraphqlAsset.SHEETS,
|
|
241
|
+
},
|
|
242
|
+
],
|
|
222
243
|
TableauAsset.WORKBOOK_TO_DATASOURCE: [
|
|
223
244
|
{
|
|
224
245
|
FIELDS: GQLQueryFields.WORKBOOK_TO_DATASOURCE,
|
{castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/databricks/client.py
RENAMED
|
@@ -87,15 +87,32 @@ class DatabricksClient(APIClient):
|
|
|
87
87
|
content.get("tables", []), schema
|
|
88
88
|
)
|
|
89
89
|
|
|
90
|
-
|
|
90
|
+
@staticmethod
|
|
91
|
+
def _match_table_with_user(table: dict, user_id_by_email: dict) -> dict:
|
|
92
|
+
table_owner_email = table.get("owner_email")
|
|
93
|
+
if not table_owner_email:
|
|
94
|
+
return table
|
|
95
|
+
owner_external_id = user_id_by_email.get(table_owner_email)
|
|
96
|
+
if not owner_external_id:
|
|
97
|
+
return table
|
|
98
|
+
return {**table, "owner_external_id": owner_external_id}
|
|
99
|
+
|
|
100
|
+
def tables_and_columns(
|
|
101
|
+
self, schemas: List[dict], users: List[dict]
|
|
102
|
+
) -> TablesColumns:
|
|
91
103
|
"""
|
|
92
104
|
Get the databricks tables & columns leveraging the unity catalog API
|
|
93
105
|
"""
|
|
94
106
|
tables: List[dict] = []
|
|
95
107
|
columns: List[dict] = []
|
|
108
|
+
user_id_by_email = {user.get("email"): user.get("id") for user in users}
|
|
96
109
|
for schema in schemas:
|
|
97
110
|
t_to_add, c_to_add = self._tables_columns_of_schema(schema)
|
|
98
|
-
|
|
111
|
+
t_with_owner = [
|
|
112
|
+
self._match_table_with_user(table, user_id_by_email)
|
|
113
|
+
for table in t_to_add
|
|
114
|
+
]
|
|
115
|
+
tables.extend(t_with_owner)
|
|
99
116
|
columns.extend(c_to_add)
|
|
100
117
|
return tables, columns
|
|
101
118
|
|
|
@@ -64,3 +64,17 @@ def test_DatabricksClient__keep_catalog():
|
|
|
64
64
|
assert client._keep_catalog("staging")
|
|
65
65
|
assert not client._keep_catalog("dev")
|
|
66
66
|
assert not client._keep_catalog("something_unknown")
|
|
67
|
+
|
|
68
|
+
|
|
69
|
+
def test_DatabricksClient__match_table_with_user():
|
|
70
|
+
client = MockDatabricksClient()
|
|
71
|
+
users_by_email = {"bob@castordoc.com": 3}
|
|
72
|
+
|
|
73
|
+
table = {"id": 1, "owner_email": "bob@castordoc.com"}
|
|
74
|
+
table_with_owner = client._match_table_with_user(table, users_by_email)
|
|
75
|
+
|
|
76
|
+
assert table_with_owner == {**table, "owner_external_id": 3}
|
|
77
|
+
|
|
78
|
+
table_without_owner = {"id": 1, "owner_email": None}
|
|
79
|
+
actual = client._match_table_with_user(table_without_owner, users_by_email)
|
|
80
|
+
assert actual == table_without_owner
|
{castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/databricks/extract.py
RENAMED
|
@@ -4,6 +4,7 @@ from typing import Dict, Optional
|
|
|
4
4
|
from ...utils import AbstractStorage, LocalStorage, write_summary
|
|
5
5
|
from ..abstract import (
|
|
6
6
|
CATALOG_ASSETS,
|
|
7
|
+
EXTERNAL_LINEAGE_ASSETS,
|
|
7
8
|
QUERIES_ASSETS,
|
|
8
9
|
VIEWS_ASSETS,
|
|
9
10
|
SupportedAssets,
|
|
@@ -20,6 +21,7 @@ DATABRICKS_ASSETS: SupportedAssets = {
|
|
|
20
21
|
WarehouseAssetGroup.QUERY: QUERIES_ASSETS,
|
|
21
22
|
WarehouseAssetGroup.ROLE: (WarehouseAsset.USER,),
|
|
22
23
|
WarehouseAssetGroup.VIEW_DDL: VIEWS_ASSETS,
|
|
24
|
+
WarehouseAssetGroup.EXTERNAL_LINEAGE: EXTERNAL_LINEAGE_ASSETS,
|
|
23
25
|
}
|
|
24
26
|
|
|
25
27
|
logger = logging.getLogger(__name__)
|
|
@@ -80,7 +82,8 @@ class DatabricksExtractionProcessor:
|
|
|
80
82
|
|
|
81
83
|
del databases
|
|
82
84
|
|
|
83
|
-
|
|
85
|
+
users = self._client.users()
|
|
86
|
+
tables, columns = self._client.tables_and_columns(schemas, users)
|
|
84
87
|
|
|
85
88
|
location = self._storage.put(WarehouseAsset.TABLE.value, tables)
|
|
86
89
|
catalog_locations[WarehouseAsset.TABLE.value] = location
|
{castor_extractor-0.15.4 → castor_extractor-0.16.3}/castor_extractor/warehouse/databricks/format.py
RENAMED
|
@@ -19,10 +19,11 @@ def _to_datetime_or_none(time_ms: Optional[int]) -> Optional[datetime]:
|
|
|
19
19
|
|
|
20
20
|
def _table_payload(schema: dict, table: dict) -> dict:
|
|
21
21
|
return {
|
|
22
|
+
"description": table.get("comment"),
|
|
22
23
|
"id": table["table_id"],
|
|
24
|
+
"owner_email": table.get("owner"),
|
|
23
25
|
"schema_id": f"{schema['id']}",
|
|
24
26
|
"table_name": table["name"],
|
|
25
|
-
"description": table.get("comment"),
|
|
26
27
|
"tags": [],
|
|
27
28
|
"type": table.get("table_type"),
|
|
28
29
|
}
|
|
@@ -30,12 +31,12 @@ def _table_payload(schema: dict, table: dict) -> dict:
|
|
|
30
31
|
|
|
31
32
|
def _column_payload(table: dict, column: dict) -> dict:
|
|
32
33
|
return {
|
|
33
|
-
"id": f"`{table['id']}`.`{column['name']}`",
|
|
34
34
|
"column_name": column["name"],
|
|
35
|
-
"table_id": table["id"],
|
|
36
|
-
"description": column.get("comment"),
|
|
37
35
|
"data_type": column["type_name"],
|
|
36
|
+
"description": column.get("comment"),
|
|
37
|
+
"id": f"`{table['id']}`.`{column['name']}`",
|
|
38
38
|
"ordinal_position": column["position"],
|
|
39
|
+
"table_id": table["id"],
|
|
39
40
|
}
|
|
40
41
|
|
|
41
42
|
|