castor-extractor 0.22.0__tar.gz → 0.22.5__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.22.0 → castor_extractor-0.22.5}/CHANGELOG.md +20 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/PKG-INFO +23 -1
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/__init__.py +1 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/time.py +4 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/time_test.py +8 -1
- castor_extractor-0.22.5/castor_extractor/visualization/looker_studio/__init__.py +6 -0
- castor_extractor-0.22.5/castor_extractor/visualization/looker_studio/assets.py +6 -0
- castor_extractor-0.22.5/castor_extractor/visualization/looker_studio/client/__init__.py +3 -0
- castor_extractor-0.22.5/castor_extractor/visualization/looker_studio/client/admin_sdk_client.py +90 -0
- castor_extractor-0.22.5/castor_extractor/visualization/looker_studio/client/client.py +37 -0
- castor_extractor-0.22.5/castor_extractor/visualization/looker_studio/client/credentials.py +20 -0
- castor_extractor-0.22.5/castor_extractor/visualization/looker_studio/client/endpoints.py +18 -0
- castor_extractor-0.22.5/castor_extractor/visualization/looker_studio/client/enums.py +8 -0
- castor_extractor-0.22.5/castor_extractor/visualization/looker_studio/client/looker_studio_api_client.py +102 -0
- castor_extractor-0.22.5/castor_extractor/visualization/looker_studio/client/pagination.py +31 -0
- castor_extractor-0.22.5/castor_extractor/visualization/looker_studio/client/scopes.py +6 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/sigma/client/client.py +64 -10
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/thoughtspot/assets.py +3 -1
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/thoughtspot/client/client.py +67 -14
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/thoughtspot/client/utils.py +10 -4
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/thoughtspot/client/utils_test.py +22 -4
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/databricks/api_client.py +2 -60
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/databricks/client.py +4 -47
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/databricks/client_test.py +1 -35
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/databricks/credentials.py +4 -6
- castor_extractor-0.22.5/castor_extractor/warehouse/databricks/enums.py +15 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/databricks/extract.py +13 -11
- castor_extractor-0.22.5/castor_extractor/warehouse/databricks/lineage.py +69 -0
- castor_extractor-0.22.5/castor_extractor/warehouse/databricks/lineage_test.py +89 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/databricks/sql_client.py +23 -8
- castor_extractor-0.22.5/castor_extractor/warehouse/databricks/types.py +1 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/salesforce/format.py +12 -5
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/salesforce/format_test.py +22 -6
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/pyproject.toml +4 -1
- castor_extractor-0.22.0/castor_extractor/warehouse/databricks/lineage.py +0 -141
- castor_extractor-0.22.0/castor_extractor/warehouse/databricks/lineage_test.py +0 -34
- castor_extractor-0.22.0/castor_extractor/warehouse/databricks/test_constants.py +0 -79
- castor_extractor-0.22.0/castor_extractor/warehouse/databricks/types.py +0 -8
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/Dockerfile +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/DockerfileUsage.md +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/LICENCE +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/README.md +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_bigquery.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_confluence.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_databricks.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_domo.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_looker.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_metabase_api.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_metabase_db.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_mode.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_mysql.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_notion.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_postgres.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_powerbi.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_qlik.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_redshift.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_salesforce.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_salesforce_reporting.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_sigma.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_snowflake.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_sqlserver.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_tableau.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_thoughtspot.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/file_check.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/upload.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/file_checker/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/file_checker/column.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/file_checker/column_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/file_checker/constants.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/file_checker/enums.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/file_checker/file.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/file_checker/file_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/file_checker/file_test_users.csv +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/file_checker/file_test_users_valid.csv +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/file_checker/templates/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/file_checker/templates/generic_warehouse.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/confluence/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/confluence/assets.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/confluence/client/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/confluence/client/client.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/confluence/client/credentials.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/confluence/client/endpoints.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/confluence/client/pagination.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/confluence/extract.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/notion/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/notion/assets.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/notion/client/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/notion/client/client.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/notion/client/client_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/notion/client/constants.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/notion/client/credentials.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/notion/client/endpoints.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/notion/client/pagination.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/notion/extract.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/logger.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/quality/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/quality/soda/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/quality/soda/assets.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/quality/soda/client/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/quality/soda/client/client.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/quality/soda/client/credentials.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/quality/soda/client/endpoints.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/quality/soda/client/pagination.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/types.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/uploader/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/uploader/constant.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/uploader/env.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/uploader/env_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/uploader/settings.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/uploader/upload.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/uploader/upload_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/uploader/utils.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/argument_parser.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/argument_parser_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/abstract.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/api/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/api/auth.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/api/auth_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/api/client.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/api/client_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/api/pagination.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/api/pagination_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/api/safe_request.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/api/safe_request_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/api/utils.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/api/utils_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/postgres.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/query.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/uri.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/uri_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/collection.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/collection_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/constants.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/dbt/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/dbt/assets.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/dbt/client.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/dbt/client_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/dbt/credentials.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/deprecate.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/env.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/files.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/files_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/formatter.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/formatter_test.csv +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/formatter_test.json +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/formatter_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/json_stream_write.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/load.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/object.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/object_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/pager/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/pager/pager.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/pager/pager_on_id.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/pager/pager_on_id_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/pager/pager_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/retry.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/retry_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/safe.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/safe_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/salesforce/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/salesforce/client.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/salesforce/client_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/salesforce/constants.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/salesforce/credentials.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/salesforce/credentials_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/salesforce/pagination.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/store.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/string.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/string_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/type.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/validation.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/validation_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/write.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/domo/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/domo/assets.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/domo/client/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/domo/client/client.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/domo/client/credentials.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/domo/client/endpoints.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/domo/client/pagination.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/domo/client/pagination_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/domo/extract.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/api/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/api/client.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/api/client_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/api/constants.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/api/credentials.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/api/extraction_parameters.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/api/sdk.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/api/sdk_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/api/utils.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/assets.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/constant.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/constants.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/extract.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/fields.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/fields_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/multithreading.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/assets.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/api/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/api/client.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/api/client_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/api/credentials.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/db/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/db/client.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/db/credentials.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/db/queries/.sqlfluff +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/db/queries/base_url.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/db/queries/card.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/db/queries/collection.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/db/queries/dashboard.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/db/queries/dashboard_cards.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/db/queries/database.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/db/queries/table.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/db/queries/user.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/decryption.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/decryption_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/shared.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/errors.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/extract.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/types.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/mode/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/mode/assets.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/mode/client/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/mode/client/client.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/mode/client/client_test.json +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/mode/client/client_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/mode/client/constants.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/mode/client/credentials.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/mode/errors.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/mode/extract.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/powerbi/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/powerbi/assets.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/powerbi/client/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/powerbi/client/authentication.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/powerbi/client/client.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/powerbi/client/client_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/powerbi/client/constants.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/powerbi/client/credentials.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/powerbi/client/credentials_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/powerbi/client/endpoints.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/powerbi/client/pagination.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/powerbi/extract.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/assets.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/client/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/client/constants.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/client/engine/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/client/engine/client.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/client/engine/constants.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/client/engine/credentials.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/client/engine/error.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/client/engine/error_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/client/engine/json_rpc.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/client/engine/json_rpc_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/client/engine/websocket.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/client/master.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/client/rest.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/client/rest_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/extract.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/salesforce_reporting/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/salesforce_reporting/assets.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/salesforce_reporting/client/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/salesforce_reporting/client/rest.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/salesforce_reporting/client/soql.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/salesforce_reporting/extract.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/sigma/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/sigma/assets.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/sigma/client/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/sigma/client/credentials.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/sigma/client/endpoints.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/sigma/client/pagination.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/sigma/extract.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/assets.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/client/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/client/client.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/client/client_utils.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/client/credentials.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/client/project.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/client/safe_mode.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/constants.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/errors.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/extract.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/gql_fields.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/assets/graphql/metadata/metadata_1_get.json +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/assets/graphql/metadata/metadata_2_get.json +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/assets/rest_api/auth.xml +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/assets/rest_api/project_get.xml +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/assets/rest_api/user_get.xml +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/assets/rest_api/view_get_usage.xml +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/assets/rest_api/workbook_get.xml +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/graphql/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/graphql/paginated_object_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/rest_api/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/rest_api/auth_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/rest_api/credentials_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/rest_api/projects_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/rest_api/usages_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/rest_api/users_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/rest_api/workbooks_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/utils/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/utils/env_key.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tsc_fields.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/types.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/usage.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau_revamp/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau_revamp/assets.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau_revamp/client/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau_revamp/client/client.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau_revamp/client/client_metadata_api.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau_revamp/client/client_rest_api.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau_revamp/client/client_tsc.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau_revamp/client/credentials.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau_revamp/client/errors.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau_revamp/client/gql_queries.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau_revamp/client/rest_fields.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau_revamp/constants.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau_revamp/extract.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/thoughtspot/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/thoughtspot/client/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/thoughtspot/client/credentials.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/thoughtspot/client/endpoints.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/thoughtspot/extract.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/abstract/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/abstract/asset.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/abstract/asset_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/abstract/extract.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/abstract/query.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/abstract/time_filter.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/abstract/time_filter_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/client.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/client_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/credentials.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/extract.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/queries/.sqlfluff +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/queries/column.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/queries/cte/sharded.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/queries/database.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/queries/query.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/queries/schema.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/queries/table.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/queries/table_with_tags.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/queries/user.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/queries/view_ddl.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/query.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/types.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/databricks/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/databricks/api_client_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/databricks/endpoints.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/databricks/format.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/databricks/format_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/databricks/pagination.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/databricks/utils.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/databricks/utils_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/mysql/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/mysql/client.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/mysql/client_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/mysql/extract.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/mysql/queries/.sqlfluff +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/mysql/queries/column.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/mysql/queries/database.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/mysql/queries/query.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/mysql/queries/schema.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/mysql/queries/table.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/mysql/queries/user.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/mysql/queries/view_ddl.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/mysql/query.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/postgres/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/postgres/extract.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/postgres/queries/.sqlfluff +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/postgres/queries/column.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/postgres/queries/database.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/postgres/queries/group.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/postgres/queries/schema.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/postgres/queries/table.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/postgres/queries/user.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/postgres/query.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/client.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/client_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/extract.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/extract_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/queries/.sqlfluff +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/queries/column.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/queries/database.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/queries/group.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/queries/query.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/queries/query_serverless.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/queries/schema.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/queries/table.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/queries/table_freshness.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/queries/user.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/queries/view_ddl.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/query.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/salesforce/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/salesforce/client.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/salesforce/constants.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/salesforce/extract.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/salesforce/pagination.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/salesforce/soql.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/client.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/client_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/credentials.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/credentials_test.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/extract.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/queries/.sqlfluff +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/queries/column.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/queries/column_lineage.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/queries/database.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/queries/function.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/queries/grant_to_role.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/queries/grant_to_user.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/queries/query.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/queries/role.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/queries/schema.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/queries/table.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/queries/user.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/queries/view_ddl.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/query.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/sqlserver/__init__.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/sqlserver/client.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/sqlserver/extract.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/sqlserver/queries/.sqlfluff +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/sqlserver/queries/column.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/sqlserver/queries/database.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/sqlserver/queries/schema.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/sqlserver/queries/table.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/sqlserver/queries/user.sql +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/sqlserver/query.py +0 -0
- {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/synapse/queries/column.sql +0 -0
|
@@ -1,6 +1,26 @@
|
|
|
1
1
|
|
|
2
2
|
# Changelog
|
|
3
3
|
|
|
4
|
+
## 0.22.5 - 2025-01-09
|
|
5
|
+
|
|
6
|
+
* Databricks: validate and deduplicate lineage links
|
|
7
|
+
|
|
8
|
+
## 0.22.4 - 2025-01-08
|
|
9
|
+
|
|
10
|
+
* ThoughtSpot: extract answers
|
|
11
|
+
|
|
12
|
+
## 0.22.3 - 2024-12-10
|
|
13
|
+
|
|
14
|
+
* Databricks: extract lineage from system tables
|
|
15
|
+
|
|
16
|
+
## 0.22.2 - 2024-12-06
|
|
17
|
+
|
|
18
|
+
* Sigma: multithreading to retrieve lineage
|
|
19
|
+
|
|
20
|
+
## 0.22.1 - 2024-12-05
|
|
21
|
+
|
|
22
|
+
* Salesforce: deduplicate tables
|
|
23
|
+
|
|
4
24
|
## 0.22.0 - 2024-12-04
|
|
5
25
|
|
|
6
26
|
* Stop supporting python3.8
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
Metadata-Version: 2.1
|
|
2
2
|
Name: castor-extractor
|
|
3
|
-
Version: 0.22.
|
|
3
|
+
Version: 0.22.5
|
|
4
4
|
Summary: Extract your metadata assets.
|
|
5
5
|
Home-page: https://www.castordoc.com/
|
|
6
6
|
License: EULA
|
|
@@ -19,6 +19,7 @@ Provides-Extra: bigquery
|
|
|
19
19
|
Provides-Extra: databricks
|
|
20
20
|
Provides-Extra: dbt
|
|
21
21
|
Provides-Extra: looker
|
|
22
|
+
Provides-Extra: lookerstudio
|
|
22
23
|
Provides-Extra: metabase
|
|
23
24
|
Provides-Extra: mysql
|
|
24
25
|
Provides-Extra: postgres
|
|
@@ -31,6 +32,7 @@ Provides-Extra: tableau
|
|
|
31
32
|
Requires-Dist: cryptography (>=43.0.0,<44.0.0) ; extra == "snowflake"
|
|
32
33
|
Requires-Dist: databricks-sql-connector (>=3.2.0,<4.0.0) ; extra == "databricks" or extra == "all"
|
|
33
34
|
Requires-Dist: google-api-core (>=2.1.1,<3.0.0)
|
|
35
|
+
Requires-Dist: google-api-python-client (>=2.121.0,<3.0.0) ; extra == "lookerstudio" or extra == "all"
|
|
34
36
|
Requires-Dist: google-auth (>=2,<3)
|
|
35
37
|
Requires-Dist: google-cloud-core (>=2.1.0,<3.0.0)
|
|
36
38
|
Requires-Dist: google-cloud-storage (>=2,<3)
|
|
@@ -205,6 +207,26 @@ For any questions or bug report, contact us at [support@castordoc.com](mailto:su
|
|
|
205
207
|
|
|
206
208
|
# Changelog
|
|
207
209
|
|
|
210
|
+
## 0.22.5 - 2025-01-09
|
|
211
|
+
|
|
212
|
+
* Databricks: validate and deduplicate lineage links
|
|
213
|
+
|
|
214
|
+
## 0.22.4 - 2025-01-08
|
|
215
|
+
|
|
216
|
+
* ThoughtSpot: extract answers
|
|
217
|
+
|
|
218
|
+
## 0.22.3 - 2024-12-10
|
|
219
|
+
|
|
220
|
+
* Databricks: extract lineage from system tables
|
|
221
|
+
|
|
222
|
+
## 0.22.2 - 2024-12-06
|
|
223
|
+
|
|
224
|
+
* Sigma: multithreading to retrieve lineage
|
|
225
|
+
|
|
226
|
+
## 0.22.1 - 2024-12-05
|
|
227
|
+
|
|
228
|
+
* Salesforce: deduplicate tables
|
|
229
|
+
|
|
208
230
|
## 0.22.0 - 2024-12-04
|
|
209
231
|
|
|
210
232
|
* Stop supporting python3.8
|
|
@@ -63,5 +63,9 @@ def format_date(timestamp: Union[datetime, date]) -> str:
|
|
|
63
63
|
return timestamp.strftime(ISO_FORMAT)
|
|
64
64
|
|
|
65
65
|
|
|
66
|
+
def format_rfc_3339_date(timestamp: datetime) -> str:
|
|
67
|
+
return timestamp.isoformat(timespec="seconds") + "Z"
|
|
68
|
+
|
|
69
|
+
|
|
66
70
|
def yesterday() -> date:
|
|
67
71
|
return current_date() - timedelta(days=1)
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
from datetime import date, datetime
|
|
2
2
|
|
|
3
|
-
from .time import at_midnight, date_after, timestamp_ms
|
|
3
|
+
from .time import at_midnight, date_after, format_rfc_3339_date, timestamp_ms
|
|
4
4
|
|
|
5
5
|
|
|
6
6
|
def test_at_midnight():
|
|
@@ -17,3 +17,10 @@ def test_timestamp_ms():
|
|
|
17
17
|
result = timestamp_ms(dt)
|
|
18
18
|
expected = 670636800000
|
|
19
19
|
assert result == expected
|
|
20
|
+
|
|
21
|
+
|
|
22
|
+
def test_format_rfc_3339_date():
|
|
23
|
+
dt = datetime(1995, 4, 3, 2, 1)
|
|
24
|
+
result = format_rfc_3339_date(dt)
|
|
25
|
+
expected = "1995-04-03T02:01:00Z"
|
|
26
|
+
assert result == expected
|
castor_extractor-0.22.5/castor_extractor/visualization/looker_studio/client/admin_sdk_client.py
ADDED
|
@@ -0,0 +1,90 @@
|
|
|
1
|
+
from typing import Iterator, Optional
|
|
2
|
+
|
|
3
|
+
from google.oauth2.service_account import Credentials
|
|
4
|
+
from googleapiclient import discovery # type: ignore
|
|
5
|
+
|
|
6
|
+
from ....utils import (
|
|
7
|
+
at_midnight,
|
|
8
|
+
current_date,
|
|
9
|
+
fetch_all_pages,
|
|
10
|
+
format_rfc_3339_date,
|
|
11
|
+
past_date,
|
|
12
|
+
)
|
|
13
|
+
from .credentials import LookerStudioCredentials
|
|
14
|
+
from .pagination import LookerStudioPagination
|
|
15
|
+
from .scopes import SCOPES
|
|
16
|
+
|
|
17
|
+
USER_EMAIL_FIELD = "primaryEmail"
|
|
18
|
+
|
|
19
|
+
|
|
20
|
+
class AdminSDKClient:
|
|
21
|
+
"""
|
|
22
|
+
Client to call the Report API and Directory API.
|
|
23
|
+
The service account must impersonate and admin account.
|
|
24
|
+
"""
|
|
25
|
+
|
|
26
|
+
def __init__(self, credentials: LookerStudioCredentials):
|
|
27
|
+
self._credentials = Credentials.from_service_account_info(
|
|
28
|
+
credentials.model_dump(),
|
|
29
|
+
scopes=SCOPES,
|
|
30
|
+
subject=credentials.admin_email, # impersonates an admin
|
|
31
|
+
)
|
|
32
|
+
self.directory_api = discovery.build(
|
|
33
|
+
"admin", "directory_v1", credentials=self._credentials
|
|
34
|
+
)
|
|
35
|
+
self.report_api = discovery.build(
|
|
36
|
+
"admin", "reports_v1", credentials=self._credentials
|
|
37
|
+
)
|
|
38
|
+
|
|
39
|
+
def list_users(self) -> Iterator[dict]:
|
|
40
|
+
"""
|
|
41
|
+
Lists all users in the domain; only the primaryEmail field is selected.
|
|
42
|
+
Note:
|
|
43
|
+
* `my_customer` is an alias to represent the account's `customerId`
|
|
44
|
+
* `domain_public` allows non-admins to list users. This is technically
|
|
45
|
+
not necessary here because an admin account is impersonated, but it
|
|
46
|
+
avoids tapping into unnecessary data & serves for future reference.
|
|
47
|
+
See
|
|
48
|
+
https://googleapis.github.io/google-api-python-client/docs/dyn/admin_directory_v1.users.html#list
|
|
49
|
+
https://developers.google.com/admin-sdk/directory/reference/rest/v1/users/list
|
|
50
|
+
https://developers.google.com/admin-sdk/directory/v1/guides/manage-users#retrieve_users_non_admin
|
|
51
|
+
https://stackoverflow.com/a/71083443/14448410
|
|
52
|
+
"""
|
|
53
|
+
|
|
54
|
+
def _users(pagination_params: Optional[dict] = None) -> dict:
|
|
55
|
+
parameters = {
|
|
56
|
+
"viewType": "domain_public",
|
|
57
|
+
"customer": "my_customer",
|
|
58
|
+
"fields": f"users({USER_EMAIL_FIELD}), nextPageToken",
|
|
59
|
+
**(pagination_params or {}),
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
return self.directory_api.users().list(**parameters).execute()
|
|
63
|
+
|
|
64
|
+
yield from fetch_all_pages(_users, LookerStudioPagination)
|
|
65
|
+
|
|
66
|
+
def list_view_events(self) -> Iterator[dict]:
|
|
67
|
+
"""
|
|
68
|
+
Lists all Data Studio View events of the past day.
|
|
69
|
+
See
|
|
70
|
+
https://googleapis.github.io/google-api-python-client/docs/dyn/admin_reports_v1.activities.html
|
|
71
|
+
https://developers.google.com/admin-sdk/reports/reference/rest/v1/activities/list
|
|
72
|
+
https://developers.google.com/admin-sdk/reports/v1/appendix/activity/data-studio#VIEW
|
|
73
|
+
"""
|
|
74
|
+
|
|
75
|
+
def _activity(pagination_params: Optional[dict] = None) -> dict:
|
|
76
|
+
yesterday = format_rfc_3339_date(at_midnight(past_date(1)))
|
|
77
|
+
today = format_rfc_3339_date(at_midnight(current_date()))
|
|
78
|
+
|
|
79
|
+
parameters = {
|
|
80
|
+
"userKey": "all",
|
|
81
|
+
"applicationName": "data_studio",
|
|
82
|
+
"eventName": "VIEW",
|
|
83
|
+
"startTime": yesterday,
|
|
84
|
+
"endTime": today,
|
|
85
|
+
**(pagination_params or {}),
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
return self.report_api.activities().list(**parameters).execute()
|
|
89
|
+
|
|
90
|
+
yield from fetch_all_pages(_activity, LookerStudioPagination)
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
from typing import Iterator
|
|
2
|
+
|
|
3
|
+
from .. import LookerStudioAsset
|
|
4
|
+
from .admin_sdk_client import USER_EMAIL_FIELD, AdminSDKClient
|
|
5
|
+
from .credentials import LookerStudioCredentials
|
|
6
|
+
from .looker_studio_api_client import LookerStudioAPIClient
|
|
7
|
+
|
|
8
|
+
|
|
9
|
+
class LookerStudioClient:
|
|
10
|
+
"""
|
|
11
|
+
Acts as a wrapper class to fetch Looker Studio assets, which requires
|
|
12
|
+
coordinating calls between the Admin SDK API and the Looker Studio API.
|
|
13
|
+
"""
|
|
14
|
+
|
|
15
|
+
def __init__(self, credentials: LookerStudioCredentials):
|
|
16
|
+
self.admin_sdk_client = AdminSDKClient(credentials)
|
|
17
|
+
self.looker_studio_client = LookerStudioAPIClient(credentials)
|
|
18
|
+
|
|
19
|
+
def _get_assets(self) -> Iterator[dict]:
|
|
20
|
+
"""
|
|
21
|
+
Extracts reports and data sources user by user.
|
|
22
|
+
"""
|
|
23
|
+
users = self.admin_sdk_client.list_users()
|
|
24
|
+
|
|
25
|
+
for user in users:
|
|
26
|
+
email = user[USER_EMAIL_FIELD]
|
|
27
|
+
yield from self.looker_studio_client.fetch_user_assets(email)
|
|
28
|
+
|
|
29
|
+
def fetch(self, asset: LookerStudioAsset) -> Iterator[dict]:
|
|
30
|
+
if asset == LookerStudioAsset.VIEW_ACTIVITY:
|
|
31
|
+
yield from self.admin_sdk_client.list_view_events()
|
|
32
|
+
|
|
33
|
+
elif asset == LookerStudioAsset.ASSETS:
|
|
34
|
+
yield from self._get_assets()
|
|
35
|
+
|
|
36
|
+
else:
|
|
37
|
+
raise ValueError(f"The asset {asset}, is not supported")
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
from pydantic import BaseModel, SecretStr, field_serializer
|
|
2
|
+
|
|
3
|
+
|
|
4
|
+
class LookerStudioCredentials(BaseModel):
|
|
5
|
+
admin_email: str
|
|
6
|
+
auth_provider_x509_cert_url: str
|
|
7
|
+
auth_uri: str
|
|
8
|
+
client_email: str
|
|
9
|
+
client_id: str
|
|
10
|
+
client_x509_cert_url: str
|
|
11
|
+
private_key: SecretStr
|
|
12
|
+
private_key_id: str
|
|
13
|
+
project_id: str
|
|
14
|
+
token_uri: str
|
|
15
|
+
type: str
|
|
16
|
+
|
|
17
|
+
@field_serializer("private_key")
|
|
18
|
+
def dump_secret(self, pk):
|
|
19
|
+
"""When using model_dump, show private_key value"""
|
|
20
|
+
return pk.get_secret_value()
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
class LookerStudioAPIEndpoint:
|
|
2
|
+
BASE_PATH = "https://datastudio.googleapis.com"
|
|
3
|
+
|
|
4
|
+
@classmethod
|
|
5
|
+
def search(cls) -> str:
|
|
6
|
+
"""
|
|
7
|
+
Search a user's assets.
|
|
8
|
+
See https://developers.google.com/looker-studio/integrate/api/reference/assets/search
|
|
9
|
+
"""
|
|
10
|
+
return f"{cls.BASE_PATH}/v1/assets:search"
|
|
11
|
+
|
|
12
|
+
@classmethod
|
|
13
|
+
def permissions(cls, asset_name: str) -> str:
|
|
14
|
+
"""
|
|
15
|
+
Get the permissions of an asset. The user must be the owner of the asset.
|
|
16
|
+
See https://developers.google.com/looker-studio/integrate/api/reference/permissions/get
|
|
17
|
+
"""
|
|
18
|
+
return f"{cls.BASE_PATH}/v1/assets/{asset_name}/permissions"
|
|
@@ -0,0 +1,102 @@
|
|
|
1
|
+
from functools import partial
|
|
2
|
+
from typing import Iterator, Optional
|
|
3
|
+
|
|
4
|
+
from google.auth.transport.requests import Request
|
|
5
|
+
from google.oauth2.service_account import Credentials
|
|
6
|
+
|
|
7
|
+
from ....utils import (
|
|
8
|
+
APIClient,
|
|
9
|
+
BearerAuth,
|
|
10
|
+
fetch_all_pages,
|
|
11
|
+
)
|
|
12
|
+
from .credentials import LookerStudioCredentials
|
|
13
|
+
from .endpoints import LookerStudioAPIEndpoint
|
|
14
|
+
from .enums import LookerStudioAssetType
|
|
15
|
+
from .pagination import LookerStudioPagination
|
|
16
|
+
from .scopes import SCOPES
|
|
17
|
+
|
|
18
|
+
|
|
19
|
+
class LookerStudioAPIAuth(BearerAuth):
|
|
20
|
+
def __init__(
|
|
21
|
+
self,
|
|
22
|
+
credentials: LookerStudioCredentials,
|
|
23
|
+
subject: Optional[str] = None,
|
|
24
|
+
):
|
|
25
|
+
"""
|
|
26
|
+
Instantiates the service account credentials.
|
|
27
|
+
If a `subject` email is passed, the service account will impersonate
|
|
28
|
+
that user and make requests on that user's behalf.
|
|
29
|
+
"""
|
|
30
|
+
self._credentials = Credentials.from_service_account_info(
|
|
31
|
+
credentials.model_dump(), scopes=SCOPES
|
|
32
|
+
)
|
|
33
|
+
if subject:
|
|
34
|
+
self._credentials = self._credentials.with_subject(subject)
|
|
35
|
+
|
|
36
|
+
def fetch_token(self):
|
|
37
|
+
self._credentials.refresh(Request())
|
|
38
|
+
return self._credentials.token
|
|
39
|
+
|
|
40
|
+
|
|
41
|
+
class LookerStudioAPIClient(APIClient):
|
|
42
|
+
def __init__(self, credentials: LookerStudioCredentials):
|
|
43
|
+
auth = LookerStudioAPIAuth(credentials=credentials)
|
|
44
|
+
super().__init__(auth=auth)
|
|
45
|
+
|
|
46
|
+
self._credentials = credentials
|
|
47
|
+
|
|
48
|
+
def _is_private_asset(self, asset_name: str) -> bool:
|
|
49
|
+
"""
|
|
50
|
+
Returns True if the asset is not viewable by anyone other than the owner.
|
|
51
|
+
|
|
52
|
+
The permissions dict contains `Role: Member[]` key-value pairs and has
|
|
53
|
+
at least one key-value pair to define the asset's unique OWNER.
|
|
54
|
+
If another key is present, it means the asset was shared with
|
|
55
|
+
another person or group.
|
|
56
|
+
|
|
57
|
+
See also https://developers.google.com/looker-studio/integrate/api/reference/types#Permissions
|
|
58
|
+
"""
|
|
59
|
+
data = self._get(LookerStudioAPIEndpoint.permissions(asset_name))
|
|
60
|
+
permissions = data["permissions"]
|
|
61
|
+
return len(permissions.keys()) == 1
|
|
62
|
+
|
|
63
|
+
def _user_assets(
|
|
64
|
+
self, asset_type: LookerStudioAssetType, user_email: str
|
|
65
|
+
) -> Iterator[dict]:
|
|
66
|
+
"""
|
|
67
|
+
Yields all assets of the given type, owned by the given user and visible
|
|
68
|
+
by other members.
|
|
69
|
+
"""
|
|
70
|
+
request = partial(
|
|
71
|
+
self._get,
|
|
72
|
+
LookerStudioAPIEndpoint.search(),
|
|
73
|
+
params={"assetTypes": [asset_type.value]},
|
|
74
|
+
)
|
|
75
|
+
assets = fetch_all_pages(request, LookerStudioPagination)
|
|
76
|
+
|
|
77
|
+
for asset in assets:
|
|
78
|
+
asset_name = asset["name"]
|
|
79
|
+
owner = asset["owner"]
|
|
80
|
+
if owner == user_email and not self._is_private_asset(asset_name):
|
|
81
|
+
yield asset
|
|
82
|
+
|
|
83
|
+
def _impersonate_user(self, user_email: str):
|
|
84
|
+
self._auth = LookerStudioAPIAuth(
|
|
85
|
+
credentials=self._credentials, subject=user_email
|
|
86
|
+
)
|
|
87
|
+
|
|
88
|
+
def fetch_user_assets(self, user_email: str) -> Iterator[dict]:
|
|
89
|
+
"""Yields assets (reports and data sources) shared by the given user."""
|
|
90
|
+
self._impersonate_user(user_email)
|
|
91
|
+
|
|
92
|
+
reports = self._user_assets(
|
|
93
|
+
asset_type=LookerStudioAssetType.REPORT,
|
|
94
|
+
user_email=user_email,
|
|
95
|
+
)
|
|
96
|
+
data_sources = self._user_assets(
|
|
97
|
+
asset_type=LookerStudioAssetType.DATA_SOURCE,
|
|
98
|
+
user_email=user_email,
|
|
99
|
+
)
|
|
100
|
+
|
|
101
|
+
yield from reports
|
|
102
|
+
yield from data_sources
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
from typing import Optional
|
|
2
|
+
|
|
3
|
+
from pydantic import AliasChoices, ConfigDict, Field
|
|
4
|
+
from pydantic.alias_generators import to_camel
|
|
5
|
+
|
|
6
|
+
from ....utils import PaginationModel
|
|
7
|
+
|
|
8
|
+
NEXT_PAGE_KEY = "pageToken"
|
|
9
|
+
|
|
10
|
+
|
|
11
|
+
class LookerStudioPagination(PaginationModel):
|
|
12
|
+
items: list = Field(
|
|
13
|
+
default_factory=list,
|
|
14
|
+
validation_alias=AliasChoices("items", "users", "assets"),
|
|
15
|
+
)
|
|
16
|
+
next_page_token: Optional[str] = None
|
|
17
|
+
|
|
18
|
+
model_config = ConfigDict(
|
|
19
|
+
alias_generator=to_camel,
|
|
20
|
+
populate_by_name=True,
|
|
21
|
+
from_attributes=True,
|
|
22
|
+
)
|
|
23
|
+
|
|
24
|
+
def is_last(self) -> bool:
|
|
25
|
+
return self.next_page_token is None
|
|
26
|
+
|
|
27
|
+
def next_page_payload(self) -> dict:
|
|
28
|
+
return {NEXT_PAGE_KEY: self.next_page_token}
|
|
29
|
+
|
|
30
|
+
def page_results(self) -> list:
|
|
31
|
+
return self.items
|
|
@@ -1,9 +1,11 @@
|
|
|
1
1
|
from collections.abc import Iterator
|
|
2
|
+
from concurrent.futures import ThreadPoolExecutor
|
|
2
3
|
from functools import partial
|
|
3
4
|
from http import HTTPStatus
|
|
4
5
|
from typing import Callable, Optional
|
|
5
6
|
|
|
6
7
|
import requests
|
|
8
|
+
from pydantic import BaseModel
|
|
7
9
|
|
|
8
10
|
from ....utils import (
|
|
9
11
|
APIClient,
|
|
@@ -12,6 +14,7 @@ from ....utils import (
|
|
|
12
14
|
build_url,
|
|
13
15
|
fetch_all_pages,
|
|
14
16
|
handle_response,
|
|
17
|
+
retry,
|
|
15
18
|
)
|
|
16
19
|
from ..assets import SigmaAsset
|
|
17
20
|
from .credentials import SigmaCredentials
|
|
@@ -29,7 +32,7 @@ _DATA_ELEMENTS: tuple[str, ...] = (
|
|
|
29
32
|
)
|
|
30
33
|
|
|
31
34
|
_AUTH_TIMEOUT_S = 60
|
|
32
|
-
|
|
35
|
+
_SIGMA_TIMEOUT_S = 300
|
|
33
36
|
|
|
34
37
|
_SIGMA_HEADERS = {
|
|
35
38
|
"Content-Type": _CONTENT_TYPE,
|
|
@@ -47,6 +50,23 @@ SIGMA_SAFE_MODE = RequestSafeMode(
|
|
|
47
50
|
max_errors=_VOLUME_IGNORED,
|
|
48
51
|
status_codes=_IGNORED_ERROR_CODES,
|
|
49
52
|
)
|
|
53
|
+
_THREADS_LINEAGE = 10 # empirically found; hit the rate limit with 20 workers
|
|
54
|
+
_RETRY_NUMBER = 1
|
|
55
|
+
_RETRY_BASE_MS = 60_000
|
|
56
|
+
|
|
57
|
+
|
|
58
|
+
class LineageContext(BaseModel):
|
|
59
|
+
"""all info needed to build the endpoint for lineage retrieval"""
|
|
60
|
+
|
|
61
|
+
workbook_id: str
|
|
62
|
+
element_id: str
|
|
63
|
+
|
|
64
|
+
|
|
65
|
+
class Lineage(BaseModel):
|
|
66
|
+
"""holds response from lineage API and context used to retrieve it"""
|
|
67
|
+
|
|
68
|
+
lineage: dict
|
|
69
|
+
context: LineageContext
|
|
50
70
|
|
|
51
71
|
|
|
52
72
|
class SigmaBearerAuth(BearerAuth):
|
|
@@ -77,7 +97,7 @@ class SigmaClient(APIClient):
|
|
|
77
97
|
host=credentials.host,
|
|
78
98
|
auth=auth,
|
|
79
99
|
headers=_SIGMA_HEADERS,
|
|
80
|
-
timeout=
|
|
100
|
+
timeout=_SIGMA_TIMEOUT_S,
|
|
81
101
|
safe_mode=safe_mode or SIGMA_SAFE_MODE,
|
|
82
102
|
)
|
|
83
103
|
|
|
@@ -133,17 +153,51 @@ class SigmaClient(APIClient):
|
|
|
133
153
|
page=page, workbook_id=workbook_id
|
|
134
154
|
)
|
|
135
155
|
|
|
136
|
-
|
|
156
|
+
@retry(
|
|
157
|
+
(ConnectionError,),
|
|
158
|
+
max_retries=_RETRY_NUMBER,
|
|
159
|
+
base_ms=_RETRY_BASE_MS,
|
|
160
|
+
log_exc_info=True,
|
|
161
|
+
)
|
|
162
|
+
def _get_lineage(self, lineage_context: LineageContext) -> Lineage:
|
|
163
|
+
"""
|
|
164
|
+
return the lineage from API and other ids needed to characterize
|
|
165
|
+
lineage in castor
|
|
166
|
+
"""
|
|
167
|
+
workbook_id = lineage_context.workbook_id
|
|
168
|
+
element_id = lineage_context.element_id
|
|
169
|
+
endpoint = SigmaEndpointFactory.lineage(workbook_id, element_id)
|
|
170
|
+
return Lineage(lineage=self._get(endpoint), context=lineage_context)
|
|
171
|
+
|
|
172
|
+
@staticmethod
|
|
173
|
+
def _lineage_context(elements: list[dict]) -> list[LineageContext]:
|
|
174
|
+
"""
|
|
175
|
+
Helper function to prepare context for lineage retrieval.
|
|
176
|
+
Elements without associated columns are skipped.
|
|
177
|
+
"""
|
|
178
|
+
contexts: list[LineageContext] = []
|
|
137
179
|
for element in elements:
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
180
|
+
if element.get("columns") is None:
|
|
181
|
+
continue
|
|
182
|
+
|
|
183
|
+
context = LineageContext(
|
|
184
|
+
workbook_id=element["workbook_id"],
|
|
185
|
+
element_id=element["elementId"],
|
|
142
186
|
)
|
|
187
|
+
contexts.append(context)
|
|
188
|
+
return contexts
|
|
189
|
+
|
|
190
|
+
def _get_all_lineages(self, elements: list[dict]) -> Iterator[dict]:
|
|
191
|
+
lineage_context = self._lineage_context(elements)
|
|
192
|
+
|
|
193
|
+
with ThreadPoolExecutor(max_workers=_THREADS_LINEAGE) as executor:
|
|
194
|
+
results = executor.map(self._get_lineage, lineage_context)
|
|
195
|
+
|
|
196
|
+
for lineage in results:
|
|
143
197
|
yield {
|
|
144
|
-
**lineage,
|
|
145
|
-
"workbook_id": workbook_id,
|
|
146
|
-
"element_id": element_id,
|
|
198
|
+
**lineage.lineage,
|
|
199
|
+
"workbook_id": lineage.context.workbook_id,
|
|
200
|
+
"element_id": lineage.context.element_id,
|
|
147
201
|
}
|
|
148
202
|
|
|
149
203
|
def _get_all_queries(self, workbooks: list[dict]) -> Iterator[dict]:
|
|
@@ -4,6 +4,8 @@ from ...types import ExternalAsset
|
|
|
4
4
|
class ThoughtspotAsset(ExternalAsset):
|
|
5
5
|
"""Thoughtspot assets"""
|
|
6
6
|
|
|
7
|
+
ANSWERS = "answers"
|
|
8
|
+
ANSWER_USAGES = "answer_usages"
|
|
7
9
|
LIVEBOARDS = "liveboards"
|
|
10
|
+
LIVEBOARD_USAGES = "liveboard_usages"
|
|
8
11
|
LOGICAL_TABLES = "logical_tables"
|
|
9
|
-
USAGES = "usages"
|