castor-extractor 0.16.3__tar.gz → 0.16.4__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.16.3 → castor_extractor-0.16.4}/CHANGELOG.md +4 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/PKG-INFO +1 -1
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_databricks.py +3 -0
- castor_extractor-0.16.4/castor_extractor/commands/extract_salesforce.py +43 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_salesforce_reporting.py +6 -6
- castor_extractor-0.16.4/castor_extractor/utils/client/api.py +59 -0
- castor_extractor-0.16.4/castor_extractor/utils/salesforce/__init__.py +3 -0
- castor_extractor-0.16.4/castor_extractor/utils/salesforce/client.py +84 -0
- castor_extractor-0.16.4/castor_extractor/utils/salesforce/client_test.py +21 -0
- castor_extractor-0.16.4/castor_extractor/utils/salesforce/constants.py +13 -0
- castor_extractor-0.16.4/castor_extractor/utils/salesforce/credentials.py +65 -0
- {castor_extractor-0.16.3/castor_extractor/visualization/salesforce_reporting/client → castor_extractor-0.16.4/castor_extractor/utils/salesforce}/credentials_test.py +3 -2
- castor_extractor-0.16.4/castor_extractor/visualization/salesforce_reporting/__init__.py +3 -0
- castor_extractor-0.16.4/castor_extractor/visualization/salesforce_reporting/client/__init__.py +1 -0
- castor_extractor-0.16.4/castor_extractor/visualization/salesforce_reporting/client/rest.py +60 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/salesforce_reporting/extract.py +10 -8
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/databricks/client.py +1 -1
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/databricks/credentials.py +1 -4
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/databricks/extract.py +1 -1
- castor_extractor-0.16.4/castor_extractor/warehouse/salesforce/__init__.py +6 -0
- castor_extractor-0.16.4/castor_extractor/warehouse/salesforce/client.py +112 -0
- castor_extractor-0.16.4/castor_extractor/warehouse/salesforce/constants.py +2 -0
- castor_extractor-0.16.4/castor_extractor/warehouse/salesforce/extract.py +111 -0
- castor_extractor-0.16.4/castor_extractor/warehouse/salesforce/format.py +67 -0
- castor_extractor-0.16.4/castor_extractor/warehouse/salesforce/format_test.py +32 -0
- castor_extractor-0.16.4/castor_extractor/warehouse/salesforce/soql.py +45 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/pyproject.toml +3 -2
- castor_extractor-0.16.3/castor_extractor/utils/client/api.py +0 -50
- castor_extractor-0.16.3/castor_extractor/visualization/salesforce_reporting/__init__.py +0 -4
- castor_extractor-0.16.3/castor_extractor/visualization/salesforce_reporting/client/__init__.py +0 -2
- castor_extractor-0.16.3/castor_extractor/visualization/salesforce_reporting/client/constants.py +0 -2
- castor_extractor-0.16.3/castor_extractor/visualization/salesforce_reporting/client/credentials.py +0 -33
- castor_extractor-0.16.3/castor_extractor/visualization/salesforce_reporting/client/rest.py +0 -143
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/Dockerfile +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/LICENCE +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/README.md +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_bigquery.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_domo.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_looker.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_metabase_api.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_metabase_db.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_mode.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_mysql.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_postgres.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_powerbi.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_qlik.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_redshift.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_sigma.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_snowflake.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_sqlserver.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_tableau.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/file_check.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/upload.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/file_checker/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/file_checker/column.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/file_checker/column_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/file_checker/constants.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/file_checker/enums.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/file_checker/file.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/file_checker/file_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/file_checker/file_test_users.csv +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/file_checker/file_test_users_valid.csv +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/file_checker/templates/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/file_checker/templates/generic_warehouse.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/logger.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/types.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/uploader/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/uploader/constant.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/uploader/env.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/uploader/env_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/uploader/upload.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/uploader/upload_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/uploader/utils.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/client/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/client/abstract.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/client/api_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/client/postgres.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/client/query.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/client/uri.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/client/uri_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/collection.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/constants.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/dbt/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/dbt/assets.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/dbt/client.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/dbt/client_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/dbt/credentials.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/deprecate.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/env.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/files.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/files_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/formatter.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/formatter_test.csv +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/formatter_test.json +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/formatter_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/json_stream_write.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/load.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/object.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/object_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/pager/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/pager/pager.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/pager/pager_on_id.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/pager/pager_on_id_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/pager/pager_on_token.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/pager/pager_on_token_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/pager/pager_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/retry.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/retry_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/safe.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/safe_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/store.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/string.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/string_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/time.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/time_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/type.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/validation.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/validation_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/write.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/domo/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/domo/assets.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/domo/client/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/domo/client/client.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/domo/client/client_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/domo/client/credentials.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/domo/client/endpoints.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/domo/client/pagination.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/domo/client/pagination_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/domo/constants.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/domo/extract.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/looker/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/looker/api/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/looker/api/client.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/looker/api/client_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/looker/api/constants.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/looker/api/sdk.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/looker/api/sdk_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/looker/api/utils.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/looker/assets.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/looker/constant.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/looker/env.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/looker/extract.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/looker/fields.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/looker/fields_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/looker/multithreading.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/looker/parameters.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/assets.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/api/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/api/client.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/api/client_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/api/credentials.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/db/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/db/client.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/db/credentials.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/db/queries/.sqlfluff +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/db/queries/base_url.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/db/queries/card.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/db/queries/collection.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/db/queries/dashboard.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/db/queries/dashboard_cards.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/db/queries/database.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/db/queries/table.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/db/queries/user.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/decryption.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/decryption_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/shared.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/errors.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/extract.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/types.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/mode/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/mode/assets.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/mode/client/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/mode/client/client.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/mode/client/client_test.json +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/mode/client/client_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/mode/client/constants.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/mode/client/credentials.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/mode/errors.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/mode/extract.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/powerbi/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/powerbi/assets.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/powerbi/client/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/powerbi/client/constants.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/powerbi/client/credentials.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/powerbi/client/credentials_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/powerbi/client/rest.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/powerbi/client/rest_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/powerbi/client/utils.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/powerbi/client/utils_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/powerbi/extract.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/assets.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/client/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/client/constants.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/client/engine/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/client/engine/client.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/client/engine/constants.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/client/engine/error.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/client/engine/error_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/client/engine/json_rpc.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/client/engine/json_rpc_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/client/engine/websocket.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/client/master.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/client/rest.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/client/rest_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/constants.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/extract.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/salesforce_reporting/assets.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/salesforce_reporting/client/soql.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/sigma/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/sigma/assets.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/sigma/client/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/sigma/client/client.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/sigma/client/client_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/sigma/client/credentials.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/sigma/client/endpoints.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/sigma/client/pagination.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/sigma/constants.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/sigma/extract.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/assets.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/client/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/client/client.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/client/client_utils.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/client/credentials.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/client/project.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/client/safe_mode.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/constants.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/errors.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/extract.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/gql_fields.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/assets/graphql/metadata/metadata_1_get.json +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/assets/graphql/metadata/metadata_2_get.json +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/assets/rest_api/auth.xml +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/assets/rest_api/project_get.xml +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/assets/rest_api/user_get.xml +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/assets/rest_api/view_get_usage.xml +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/assets/rest_api/workbook_get.xml +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/graphql/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/graphql/paginated_object_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/rest_api/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/rest_api/auth_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/rest_api/credentials_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/rest_api/projects_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/rest_api/usages_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/rest_api/users_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/rest_api/workbooks_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/utils/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/utils/env_key.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tsc_fields.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/types.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/usage.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/abstract/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/abstract/asset.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/abstract/asset_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/abstract/extract.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/abstract/query.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/abstract/time_filter.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/abstract/time_filter_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/client.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/client_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/credentials.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/extract.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/queries/.sqlfluff +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/queries/column.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/queries/cte/sharded.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/queries/database.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/queries/query.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/queries/schema.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/queries/table.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/queries/table_with_tags.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/queries/user.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/queries/view_ddl.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/query.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/types.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/databricks/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/databricks/client_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/databricks/format.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/databricks/format_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/databricks/types.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/mysql/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/mysql/client.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/mysql/client_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/mysql/extract.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/mysql/queries/.sqlfluff +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/mysql/queries/column.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/mysql/queries/database.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/mysql/queries/query.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/mysql/queries/schema.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/mysql/queries/table.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/mysql/queries/user.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/mysql/queries/view_ddl.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/mysql/query.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/postgres/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/postgres/extract.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/postgres/queries/.sqlfluff +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/postgres/queries/column.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/postgres/queries/database.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/postgres/queries/group.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/postgres/queries/schema.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/postgres/queries/table.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/postgres/queries/user.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/postgres/query.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/redshift/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/redshift/client.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/redshift/client_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/redshift/extract.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/redshift/queries/.sqlfluff +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/redshift/queries/column.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/redshift/queries/database.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/redshift/queries/group.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/redshift/queries/query.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/redshift/queries/schema.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/redshift/queries/table.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/redshift/queries/table_freshness.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/redshift/queries/user.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/redshift/queries/view_ddl.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/redshift/query.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/client.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/client_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/credentials.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/credentials_test.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/extract.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/queries/.sqlfluff +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/queries/column.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/queries/column_lineage.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/queries/database.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/queries/grant_to_role.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/queries/grant_to_user.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/queries/query.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/queries/role.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/queries/schema.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/queries/table.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/queries/user.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/queries/view_ddl.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/query.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/sqlserver/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/sqlserver/client.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/sqlserver/extract.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/sqlserver/queries/.sqlfluff +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/sqlserver/queries/column.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/sqlserver/queries/database.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/sqlserver/queries/schema.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/sqlserver/queries/table.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/sqlserver/queries/user.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/sqlserver/query.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/synapse/__init__.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/synapse/extract.py +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/synapse/queries/.sqlfluff +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/synapse/queries/column.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/synapse/queries/database.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/synapse/queries/query.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/synapse/queries/schema.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/synapse/queries/table.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/synapse/queries/user.sql +0 -0
- {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/synapse/queries/view_ddl.sql +0 -0
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
import logging
|
|
2
|
+
from argparse import ArgumentParser
|
|
3
|
+
|
|
4
|
+
from castor_extractor.warehouse import salesforce # type: ignore
|
|
5
|
+
|
|
6
|
+
logging.basicConfig(level=logging.INFO, format="%(levelname)s - %(message)s")
|
|
7
|
+
|
|
8
|
+
|
|
9
|
+
def main():
|
|
10
|
+
parser = ArgumentParser()
|
|
11
|
+
|
|
12
|
+
parser.add_argument("-u", "--username", help="Salesforce username")
|
|
13
|
+
parser.add_argument("-p", "--password", help="Salesforce password")
|
|
14
|
+
parser.add_argument("-c", "--client-id", help="Salesforce client id")
|
|
15
|
+
parser.add_argument(
|
|
16
|
+
"-s", "--client-secret", help="Salesforce client secret"
|
|
17
|
+
)
|
|
18
|
+
parser.add_argument(
|
|
19
|
+
"-t", "--security-token", help="Salesforce security token"
|
|
20
|
+
)
|
|
21
|
+
parser.add_argument("-b", "--base-url", help="Salesforce instance URL")
|
|
22
|
+
parser.add_argument("-o", "--output", help="Directory to write to")
|
|
23
|
+
|
|
24
|
+
parser.add_argument(
|
|
25
|
+
"--skip-existing",
|
|
26
|
+
dest="skip_existing",
|
|
27
|
+
action="store_true",
|
|
28
|
+
help="Skips files already extracted instead of replacing them",
|
|
29
|
+
)
|
|
30
|
+
parser.set_defaults(skip_existing=False)
|
|
31
|
+
|
|
32
|
+
args = parser.parse_args()
|
|
33
|
+
|
|
34
|
+
salesforce.extract_all(
|
|
35
|
+
username=args.username,
|
|
36
|
+
password=args.password,
|
|
37
|
+
client_id=args.client_id,
|
|
38
|
+
client_secret=args.client_secret,
|
|
39
|
+
security_token=args.security_token,
|
|
40
|
+
base_url=args.base_url,
|
|
41
|
+
output_directory=args.output,
|
|
42
|
+
skip_existing=args.skip_existing,
|
|
43
|
+
)
|
|
@@ -11,23 +11,23 @@ def main():
|
|
|
11
11
|
|
|
12
12
|
parser.add_argument("-u", "--username", help="Salesforce username")
|
|
13
13
|
parser.add_argument("-p", "--password", help="Salesforce password")
|
|
14
|
-
parser.add_argument("-
|
|
14
|
+
parser.add_argument("-c", "--client-id", help="Salesforce client id")
|
|
15
15
|
parser.add_argument(
|
|
16
|
-
"-s", "--
|
|
16
|
+
"-s", "--client-secret", help="Salesforce client secret"
|
|
17
17
|
)
|
|
18
18
|
parser.add_argument(
|
|
19
19
|
"-t", "--security-token", help="Salesforce security token"
|
|
20
20
|
)
|
|
21
|
-
parser.add_argument("-
|
|
21
|
+
parser.add_argument("-b", "--base-url", help="Salesforce instance URL")
|
|
22
22
|
parser.add_argument("-o", "--output", help="Directory to write to")
|
|
23
23
|
|
|
24
24
|
args = parser.parse_args()
|
|
25
25
|
salesforce_reporting.extract_all(
|
|
26
26
|
username=args.username,
|
|
27
27
|
password=args.password,
|
|
28
|
-
|
|
29
|
-
|
|
28
|
+
client_id=args.client_id,
|
|
29
|
+
client_secret=args.client_secret,
|
|
30
30
|
security_token=args.security_token,
|
|
31
|
-
|
|
31
|
+
base_url=args.base_url,
|
|
32
32
|
output_directory=args.output,
|
|
33
33
|
)
|
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
import logging
|
|
2
|
+
from typing import Any, Callable, Dict, Literal, Optional
|
|
3
|
+
|
|
4
|
+
import requests
|
|
5
|
+
|
|
6
|
+
logger = logging.getLogger(__name__)
|
|
7
|
+
|
|
8
|
+
DEFAULT_TIMEOUT_MS = 30_000
|
|
9
|
+
|
|
10
|
+
# https://requests.readthedocs.io/en/latest/api/#requests.request
|
|
11
|
+
HttpMethod = Literal["GET", "OPTIONS", "HEAD", "POST", "PUT", "PATCH", "DELETE"]
|
|
12
|
+
|
|
13
|
+
|
|
14
|
+
class APIClient:
|
|
15
|
+
"""
|
|
16
|
+
API client
|
|
17
|
+
- authentication via access token
|
|
18
|
+
"""
|
|
19
|
+
|
|
20
|
+
def __init__(self, host: str, token: Optional[str] = None):
|
|
21
|
+
self._host = host
|
|
22
|
+
self._token = token or ""
|
|
23
|
+
self._timeout = DEFAULT_TIMEOUT_MS
|
|
24
|
+
|
|
25
|
+
@staticmethod
|
|
26
|
+
def build_url(host: str, path: str):
|
|
27
|
+
if not host.startswith("https://"):
|
|
28
|
+
host = "https://" + host
|
|
29
|
+
return f"{host.strip('/')}/{path}"
|
|
30
|
+
|
|
31
|
+
def _headers(self) -> Dict[str, str]:
|
|
32
|
+
if self._token:
|
|
33
|
+
return {"Authorization": f"Bearer {self._token}"}
|
|
34
|
+
return dict()
|
|
35
|
+
|
|
36
|
+
def _call(
|
|
37
|
+
self,
|
|
38
|
+
url: str,
|
|
39
|
+
method: HttpMethod = "GET",
|
|
40
|
+
*,
|
|
41
|
+
params: Optional[dict] = None,
|
|
42
|
+
data: Optional[dict] = None,
|
|
43
|
+
processor: Optional[Callable] = None,
|
|
44
|
+
) -> Any:
|
|
45
|
+
logger.debug(f"Calling {method} on {url}")
|
|
46
|
+
result = requests.request(
|
|
47
|
+
method, url, headers=self._headers(), params=params, json=data
|
|
48
|
+
)
|
|
49
|
+
result.raise_for_status()
|
|
50
|
+
|
|
51
|
+
if processor:
|
|
52
|
+
return processor(result)
|
|
53
|
+
|
|
54
|
+
return result.json()
|
|
55
|
+
|
|
56
|
+
def get(self, path: str, payload: Optional[dict] = None) -> dict:
|
|
57
|
+
"""path: REST API operation path, such as /api/2.0/clusters/get"""
|
|
58
|
+
url = self.build_url(self._host, path)
|
|
59
|
+
return self._call(url=url, data=payload)
|
|
@@ -0,0 +1,84 @@
|
|
|
1
|
+
import logging
|
|
2
|
+
from typing import Iterator, Optional, Tuple
|
|
3
|
+
|
|
4
|
+
from requests import Response
|
|
5
|
+
|
|
6
|
+
from ...utils.client.api import APIClient
|
|
7
|
+
from .constants import DEFAULT_API_VERSION, DEFAULT_PAGINATION_LIMIT
|
|
8
|
+
from .credentials import SalesforceCredentials
|
|
9
|
+
|
|
10
|
+
logger = logging.getLogger(__name__)
|
|
11
|
+
|
|
12
|
+
|
|
13
|
+
class SalesforceBaseClient(APIClient):
|
|
14
|
+
"""
|
|
15
|
+
Salesforce API client.
|
|
16
|
+
https://developer.salesforce.com/docs/atlas.en-us.api_rest.meta/api_rest/intro_rest.htm
|
|
17
|
+
"""
|
|
18
|
+
|
|
19
|
+
api_version = DEFAULT_API_VERSION
|
|
20
|
+
pagination_limit = DEFAULT_PAGINATION_LIMIT
|
|
21
|
+
|
|
22
|
+
PATH_TPL = "services/data/v{version}/{suffix}"
|
|
23
|
+
|
|
24
|
+
def __init__(self, credentials: SalesforceCredentials):
|
|
25
|
+
super().__init__(host=credentials.base_url)
|
|
26
|
+
self._token = self._access_token(credentials)
|
|
27
|
+
|
|
28
|
+
def _access_token(self, credentials: SalesforceCredentials) -> str:
|
|
29
|
+
url = self.build_url(self._host, "services/oauth2/token")
|
|
30
|
+
response = self._call(
|
|
31
|
+
url, "POST", params=credentials.token_request_payload()
|
|
32
|
+
)
|
|
33
|
+
return response["access_token"]
|
|
34
|
+
|
|
35
|
+
def _full_url(self, suffix: str) -> str:
|
|
36
|
+
path = self.PATH_TPL.format(version=self.api_version, suffix=suffix)
|
|
37
|
+
return self.build_url(self._host, path)
|
|
38
|
+
|
|
39
|
+
@property
|
|
40
|
+
def query_url(self) -> str:
|
|
41
|
+
"""Returns the query API url"""
|
|
42
|
+
return self._full_url("query")
|
|
43
|
+
|
|
44
|
+
@property
|
|
45
|
+
def tooling_url(self) -> str:
|
|
46
|
+
"""Returns the tooling API url"""
|
|
47
|
+
return self._full_url("tooling/query")
|
|
48
|
+
|
|
49
|
+
@staticmethod
|
|
50
|
+
def _query_processor(response: Response) -> Tuple[dict, Optional[str]]:
|
|
51
|
+
results = response.json()
|
|
52
|
+
return results["records"], results.get("nextRecordsUrl")
|
|
53
|
+
|
|
54
|
+
def _has_reached_pagination_limit(self, page_number: int) -> bool:
|
|
55
|
+
return page_number > self.pagination_limit
|
|
56
|
+
|
|
57
|
+
def _query_first_page(self, query: str) -> Tuple[Iterator[dict], str]:
|
|
58
|
+
url = self.query_url
|
|
59
|
+
logger.info("querying page 0")
|
|
60
|
+
records, next_page_url = self._call(
|
|
61
|
+
url, params={"q": query}, processor=self._query_processor
|
|
62
|
+
)
|
|
63
|
+
return records, next_page_url
|
|
64
|
+
|
|
65
|
+
def _query_all(self, query: str) -> Iterator[dict]:
|
|
66
|
+
"""
|
|
67
|
+
Run a SOQL query over salesforce API.
|
|
68
|
+
|
|
69
|
+
more: https://developer.salesforce.com/docs/atlas.en-us.api_rest.meta/api_rest/dome_query.htm
|
|
70
|
+
"""
|
|
71
|
+
records, next_page_path = self._query_first_page(query)
|
|
72
|
+
yield from records
|
|
73
|
+
|
|
74
|
+
page_count = 1
|
|
75
|
+
while next_page_path and not self._has_reached_pagination_limit(
|
|
76
|
+
page_count
|
|
77
|
+
):
|
|
78
|
+
logger.info(f"querying page {page_count}")
|
|
79
|
+
url = self.build_url(self._host, next_page_path)
|
|
80
|
+
records, next_page = self._call(
|
|
81
|
+
url, processor=self._query_processor
|
|
82
|
+
)
|
|
83
|
+
yield from records
|
|
84
|
+
page_count += 1
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
from unittest.mock import patch
|
|
2
|
+
|
|
3
|
+
from .client import SalesforceBaseClient
|
|
4
|
+
from .credentials import SalesforceCredentials
|
|
5
|
+
|
|
6
|
+
|
|
7
|
+
@patch.object(SalesforceBaseClient, "_call")
|
|
8
|
+
def test_SalesforceBaseClient__urls(mock_call):
|
|
9
|
+
mock_call.return_value = {"access_token": "the_token"}
|
|
10
|
+
credentials = SalesforceCredentials(
|
|
11
|
+
username="usr",
|
|
12
|
+
password="pw",
|
|
13
|
+
client_id="key",
|
|
14
|
+
client_secret="secret",
|
|
15
|
+
security_token="token",
|
|
16
|
+
base_url="url",
|
|
17
|
+
)
|
|
18
|
+
client = SalesforceBaseClient(credentials)
|
|
19
|
+
|
|
20
|
+
assert client.query_url == "https://url/services/data/v59.0/query"
|
|
21
|
+
assert client.tooling_url == "https://url/services/data/v59.0/tooling/query"
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
DEFAULT_API_VERSION = 59.0
|
|
2
|
+
DEFAULT_PAGINATION_LIMIT = 100
|
|
3
|
+
|
|
4
|
+
|
|
5
|
+
class Keys:
|
|
6
|
+
"""Salesforce's credentials keys"""
|
|
7
|
+
|
|
8
|
+
USERNAME = "username"
|
|
9
|
+
PASSWORD = "password" # noqa: S105
|
|
10
|
+
CLIENT_ID = "client_id"
|
|
11
|
+
CLIENT_SECRET = "client_secret" # noqa: S105
|
|
12
|
+
SECURITY_TOKEN = "security_token" # noqa: S105
|
|
13
|
+
BASE_URL = "base_url"
|
|
@@ -0,0 +1,65 @@
|
|
|
1
|
+
from typing import Dict
|
|
2
|
+
|
|
3
|
+
from ...utils import from_env
|
|
4
|
+
from .constants import Keys
|
|
5
|
+
|
|
6
|
+
_USERNAME = "CASTOR_SALESFORCE_USERNAME"
|
|
7
|
+
_PASSWORD = "CASTOR_SALESFORCE_PASSWORD" # noqa: S105
|
|
8
|
+
_SECURITY_TOKEN = "CASTOR_SALESFORCE_SECURITY_TOKEN" # noqa: S105
|
|
9
|
+
_CLIENT_ID = "CASTOR_SALESFORCE_CLIENT_ID"
|
|
10
|
+
_CLIENT_SECRET = "CASTOR_SALESFORCE_CLIENT_SECRET" # noqa: S105
|
|
11
|
+
_BASE_URL = "CASTOR_SALESFORCE_BASE_URL"
|
|
12
|
+
|
|
13
|
+
|
|
14
|
+
class SalesforceCredentials:
|
|
15
|
+
"""
|
|
16
|
+
Class to handle Salesforce rest API permissions
|
|
17
|
+
"""
|
|
18
|
+
|
|
19
|
+
def __init__(
|
|
20
|
+
self,
|
|
21
|
+
*,
|
|
22
|
+
username: str,
|
|
23
|
+
password: str,
|
|
24
|
+
security_token: str,
|
|
25
|
+
client_id: str,
|
|
26
|
+
client_secret: str,
|
|
27
|
+
base_url: str,
|
|
28
|
+
):
|
|
29
|
+
self.username = username
|
|
30
|
+
self.password = password + security_token
|
|
31
|
+
self.client_id = client_id
|
|
32
|
+
self.client_secret = client_secret
|
|
33
|
+
self.base_url = base_url
|
|
34
|
+
|
|
35
|
+
def token_request_payload(self) -> Dict[str, str]:
|
|
36
|
+
"""
|
|
37
|
+
Params to post to the API in order to retrieve the authentication token
|
|
38
|
+
"""
|
|
39
|
+
return {
|
|
40
|
+
"grant_type": "password",
|
|
41
|
+
"client_id": self.client_id,
|
|
42
|
+
"client_secret": self.client_secret,
|
|
43
|
+
"username": self.username,
|
|
44
|
+
"password": self.password,
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
|
|
48
|
+
def to_credentials(params: dict) -> SalesforceCredentials:
|
|
49
|
+
"""extract Salesforce credentials"""
|
|
50
|
+
username = params.get(Keys.USERNAME) or from_env(_USERNAME)
|
|
51
|
+
password = params.get(Keys.PASSWORD) or from_env(_PASSWORD)
|
|
52
|
+
security_token = params.get(Keys.SECURITY_TOKEN) or from_env(
|
|
53
|
+
_SECURITY_TOKEN
|
|
54
|
+
)
|
|
55
|
+
client_id = params.get(Keys.CLIENT_ID) or from_env(_CLIENT_ID)
|
|
56
|
+
client_secret = params.get(Keys.CLIENT_SECRET) or from_env(_CLIENT_SECRET)
|
|
57
|
+
base_url = params.get(Keys.BASE_URL) or from_env(_BASE_URL)
|
|
58
|
+
return SalesforceCredentials(
|
|
59
|
+
username=username,
|
|
60
|
+
password=password,
|
|
61
|
+
client_id=client_id,
|
|
62
|
+
client_secret=client_secret,
|
|
63
|
+
security_token=security_token,
|
|
64
|
+
base_url=base_url,
|
|
65
|
+
)
|
|
@@ -5,9 +5,10 @@ def test_Credentials_token_request_payload():
|
|
|
5
5
|
creds = SalesforceCredentials(
|
|
6
6
|
username="giphy",
|
|
7
7
|
password="1312",
|
|
8
|
-
|
|
9
|
-
|
|
8
|
+
client_id="degenie",
|
|
9
|
+
client_secret="fautpasledire",
|
|
10
10
|
security_token="yo",
|
|
11
|
+
base_url="man",
|
|
11
12
|
)
|
|
12
13
|
|
|
13
14
|
payload = creds.token_request_payload()
|
castor_extractor-0.16.4/castor_extractor/visualization/salesforce_reporting/client/__init__.py
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
from .rest import SalesforceReportingClient
|
|
@@ -0,0 +1,60 @@
|
|
|
1
|
+
import logging
|
|
2
|
+
from typing import Dict, Iterator, List, Optional
|
|
3
|
+
|
|
4
|
+
from ....utils.salesforce import SalesforceBaseClient
|
|
5
|
+
from ..assets import SalesforceReportingAsset
|
|
6
|
+
from .soql import queries
|
|
7
|
+
|
|
8
|
+
logger = logging.getLogger(__name__)
|
|
9
|
+
|
|
10
|
+
REQUIRING_URL_ASSETS = (
|
|
11
|
+
SalesforceReportingAsset.REPORTS,
|
|
12
|
+
SalesforceReportingAsset.DASHBOARDS,
|
|
13
|
+
SalesforceReportingAsset.FOLDERS,
|
|
14
|
+
)
|
|
15
|
+
|
|
16
|
+
|
|
17
|
+
class SalesforceReportingClient(SalesforceBaseClient):
|
|
18
|
+
"""
|
|
19
|
+
Salesforce Reporting API client
|
|
20
|
+
"""
|
|
21
|
+
|
|
22
|
+
def _get_asset_url(
|
|
23
|
+
self, asset_type: SalesforceReportingAsset, asset: dict
|
|
24
|
+
) -> Optional[str]:
|
|
25
|
+
"""
|
|
26
|
+
Fetch the given Asset + add the corresponding URL.
|
|
27
|
+
"""
|
|
28
|
+
|
|
29
|
+
if asset_type == SalesforceReportingAsset.DASHBOARDS:
|
|
30
|
+
path = f"lightning/r/Dashboard/{asset['Id']}/view"
|
|
31
|
+
return self.build_url(self._host, path)
|
|
32
|
+
|
|
33
|
+
if asset_type == SalesforceReportingAsset.FOLDERS:
|
|
34
|
+
path = asset["attributes"]["url"].lstrip("/")
|
|
35
|
+
return self.build_url(self._host, path)
|
|
36
|
+
|
|
37
|
+
if asset_type == SalesforceReportingAsset.REPORTS:
|
|
38
|
+
path = f"lightning/r/Report/{asset['Id']}/view"
|
|
39
|
+
return self.build_url(self._host, path)
|
|
40
|
+
|
|
41
|
+
return None
|
|
42
|
+
|
|
43
|
+
def _fetch_and_add_url(
|
|
44
|
+
self, asset_type: SalesforceReportingAsset
|
|
45
|
+
) -> Iterator[dict]:
|
|
46
|
+
assets = self._query_all(queries[asset_type])
|
|
47
|
+
for asset in assets:
|
|
48
|
+
url = self._get_asset_url(asset_type, asset)
|
|
49
|
+
yield {**asset, "Url": url}
|
|
50
|
+
|
|
51
|
+
def fetch(self, asset: SalesforceReportingAsset) -> List[Dict]:
|
|
52
|
+
"""
|
|
53
|
+
Fetch Salesforce Reporting assets
|
|
54
|
+
"""
|
|
55
|
+
logger.info(f"Starting extraction of {asset}")
|
|
56
|
+
|
|
57
|
+
if asset in REQUIRING_URL_ASSETS:
|
|
58
|
+
return list(self._fetch_and_add_url(asset))
|
|
59
|
+
|
|
60
|
+
return list(self._query_all(queries[asset]))
|
|
@@ -10,14 +10,15 @@ from ...utils import (
|
|
|
10
10
|
write_json,
|
|
11
11
|
write_summary,
|
|
12
12
|
)
|
|
13
|
+
from ...utils.salesforce import SalesforceCredentials
|
|
13
14
|
from .assets import SalesforceReportingAsset
|
|
14
|
-
from .client import
|
|
15
|
+
from .client import SalesforceReportingClient
|
|
15
16
|
|
|
16
17
|
logger = logging.getLogger(__name__)
|
|
17
18
|
|
|
18
19
|
|
|
19
20
|
def iterate_all_data(
|
|
20
|
-
client:
|
|
21
|
+
client: SalesforceReportingClient,
|
|
21
22
|
) -> Iterable[Tuple[str, Union[list, dict]]]:
|
|
22
23
|
"""Iterate over the extracted data from Salesforce"""
|
|
23
24
|
|
|
@@ -30,10 +31,10 @@ def iterate_all_data(
|
|
|
30
31
|
def extract_all(
|
|
31
32
|
username: str,
|
|
32
33
|
password: str,
|
|
33
|
-
|
|
34
|
-
|
|
34
|
+
client_id: str,
|
|
35
|
+
client_secret: str,
|
|
35
36
|
security_token: str,
|
|
36
|
-
|
|
37
|
+
base_url: str,
|
|
37
38
|
output_directory: Optional[str] = None,
|
|
38
39
|
) -> None:
|
|
39
40
|
"""
|
|
@@ -44,11 +45,12 @@ def extract_all(
|
|
|
44
45
|
creds = SalesforceCredentials(
|
|
45
46
|
username=username,
|
|
46
47
|
password=password,
|
|
47
|
-
|
|
48
|
-
|
|
48
|
+
client_id=client_id,
|
|
49
|
+
client_secret=client_secret,
|
|
49
50
|
security_token=security_token,
|
|
51
|
+
base_url=base_url,
|
|
50
52
|
)
|
|
51
|
-
client =
|
|
53
|
+
client = SalesforceReportingClient(credentials=creds)
|
|
52
54
|
ts = current_timestamp()
|
|
53
55
|
|
|
54
56
|
for key, data in iterate_all_data(client):
|
{castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/databricks/client.py
RENAMED
|
@@ -31,7 +31,7 @@ class DatabricksClient(APIClient):
|
|
|
31
31
|
db_allowed: Optional[Set[str]] = None,
|
|
32
32
|
db_blocked: Optional[Set[str]] = None,
|
|
33
33
|
):
|
|
34
|
-
super().__init__(credentials)
|
|
34
|
+
super().__init__(host=credentials.host, token=credentials.token)
|
|
35
35
|
self._db_allowed = db_allowed
|
|
36
36
|
self._db_blocked = db_blocked
|
|
37
37
|
self.formatter = DatabricksFormatter()
|
|
@@ -25,7 +25,4 @@ def to_credentials(params: dict) -> DatabricksCredentials:
|
|
|
25
25
|
"""extract Databricks credentials"""
|
|
26
26
|
host = params.get("host") or from_env(_HOST)
|
|
27
27
|
token = params.get("token") or from_env(_TOKEN)
|
|
28
|
-
return DatabricksCredentials(
|
|
29
|
-
host=host,
|
|
30
|
-
token=token,
|
|
31
|
-
)
|
|
28
|
+
return DatabricksCredentials(host=host, token=token)
|
{castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/databricks/extract.py
RENAMED
|
@@ -43,7 +43,7 @@ class DatabricksExtractionProcessor:
|
|
|
43
43
|
self._storage = storage
|
|
44
44
|
self._skip_existing = skip_existing
|
|
45
45
|
|
|
46
|
-
def _should_not_reextract(self, asset_group) -> bool:
|
|
46
|
+
def _should_not_reextract(self, asset_group: WarehouseAssetGroup) -> bool:
|
|
47
47
|
"""helper function to determine whether we need to extract"""
|
|
48
48
|
if not self._skip_existing:
|
|
49
49
|
return False
|
|
@@ -0,0 +1,112 @@
|
|
|
1
|
+
import logging
|
|
2
|
+
from typing import Dict, Iterator, List
|
|
3
|
+
|
|
4
|
+
from tqdm import tqdm # type: ignore
|
|
5
|
+
|
|
6
|
+
from ...utils.salesforce import SalesforceBaseClient, SalesforceCredentials
|
|
7
|
+
from .format import SalesforceFormatter
|
|
8
|
+
from .soql import SOBJECT_FIELDS_QUERY_TPL, SOBJECTS_QUERY_TPL
|
|
9
|
+
|
|
10
|
+
logger = logging.getLogger(__name__)
|
|
11
|
+
|
|
12
|
+
|
|
13
|
+
class SalesforceClient(SalesforceBaseClient):
|
|
14
|
+
"""
|
|
15
|
+
Salesforce API client to extract sobjects
|
|
16
|
+
"""
|
|
17
|
+
|
|
18
|
+
# Implicit (hard-coded in Salesforce) limitation when using SOQL of 2,000 rows
|
|
19
|
+
LIMIT_RECORDS_PER_PAGE = 2000
|
|
20
|
+
|
|
21
|
+
def __init__(self, credentials: SalesforceCredentials):
|
|
22
|
+
super().__init__(credentials)
|
|
23
|
+
self.formatter = SalesforceFormatter()
|
|
24
|
+
|
|
25
|
+
@staticmethod
|
|
26
|
+
def name() -> str:
|
|
27
|
+
return "Salesforce"
|
|
28
|
+
|
|
29
|
+
def _format_query(self, query_template: str, start_durable_id: str) -> str:
|
|
30
|
+
return query_template.format(
|
|
31
|
+
start_durable_id=start_durable_id,
|
|
32
|
+
limit=self.LIMIT_RECORDS_PER_PAGE,
|
|
33
|
+
)
|
|
34
|
+
|
|
35
|
+
def _next_records(
|
|
36
|
+
self, url: str, query_template: str, start_durable_id: str = "0000"
|
|
37
|
+
) -> List[dict]:
|
|
38
|
+
query = self._format_query(
|
|
39
|
+
query_template, start_durable_id=start_durable_id
|
|
40
|
+
)
|
|
41
|
+
records, _ = self._call(
|
|
42
|
+
url, params={"q": query}, processor=self._query_processor
|
|
43
|
+
)
|
|
44
|
+
return records
|
|
45
|
+
|
|
46
|
+
def _is_last_page(self, records: List[dict]) -> bool:
|
|
47
|
+
return len(records) < self.LIMIT_RECORDS_PER_PAGE
|
|
48
|
+
|
|
49
|
+
def _should_query_next_page(
|
|
50
|
+
self, records: List[dict], page_number: int
|
|
51
|
+
) -> bool:
|
|
52
|
+
return not (
|
|
53
|
+
self._is_last_page(records)
|
|
54
|
+
or self._has_reached_pagination_limit(page_number)
|
|
55
|
+
)
|
|
56
|
+
|
|
57
|
+
def _query_all(self, query_template: str) -> Iterator[dict]:
|
|
58
|
+
"""
|
|
59
|
+
Run a SOQL query over salesforce API
|
|
60
|
+
|
|
61
|
+
Note, pagination is performed via a LIMIT in the SOQL query and requires
|
|
62
|
+
that ids are sorted. The SOQL query must support `limit` and
|
|
63
|
+
`start_durable_id` as parameters.
|
|
64
|
+
"""
|
|
65
|
+
url = self.query_url
|
|
66
|
+
logger.info("querying page 0")
|
|
67
|
+
records = self._next_records(url, query_template)
|
|
68
|
+
yield from records
|
|
69
|
+
|
|
70
|
+
page_count = 1
|
|
71
|
+
while self._should_query_next_page(records, page_count):
|
|
72
|
+
logger.info(f"querying page {page_count}")
|
|
73
|
+
last_durable_id = records[-1]["DurableId"]
|
|
74
|
+
records = self._next_records(
|
|
75
|
+
url, query_template, start_durable_id=last_durable_id
|
|
76
|
+
)
|
|
77
|
+
yield from records
|
|
78
|
+
page_count += 1
|
|
79
|
+
|
|
80
|
+
def fetch_sobjects(self) -> List[dict]:
|
|
81
|
+
"""Fetch all sobjects"""
|
|
82
|
+
logger.info("Extracting sobjects")
|
|
83
|
+
return list(self._query_all(SOBJECTS_QUERY_TPL))
|
|
84
|
+
|
|
85
|
+
def fetch_fields(self, sobject_name: str) -> List[dict]:
|
|
86
|
+
"""Fetches fields of a given sobject"""
|
|
87
|
+
query = SOBJECT_FIELDS_QUERY_TPL.format(
|
|
88
|
+
entity_definition_id=sobject_name
|
|
89
|
+
)
|
|
90
|
+
response = self._call(self.tooling_url, params={"q": query})
|
|
91
|
+
return response["records"]
|
|
92
|
+
|
|
93
|
+
def tables(self) -> List[dict]:
|
|
94
|
+
"""
|
|
95
|
+
Get Salesforce sobjects as tables
|
|
96
|
+
"""
|
|
97
|
+
sobjects = self.fetch_sobjects()
|
|
98
|
+
logger.info(f"Extracted {len(sobjects)} sobjects")
|
|
99
|
+
return self.formatter.tables(sobjects)
|
|
100
|
+
|
|
101
|
+
def columns(
|
|
102
|
+
self, sobject_names: List[str], show_progress: bool = True
|
|
103
|
+
) -> List[dict]:
|
|
104
|
+
"""
|
|
105
|
+
Get salesforce sobject fields as columns
|
|
106
|
+
show_progress: optionally deactivate the tqdm progress bar
|
|
107
|
+
"""
|
|
108
|
+
sobject_fields: Dict[str, List[dict]] = dict()
|
|
109
|
+
for sobject_name in tqdm(sobject_names, disable=not show_progress):
|
|
110
|
+
fields = self.fetch_fields(sobject_name)
|
|
111
|
+
sobject_fields[sobject_name] = fields
|
|
112
|
+
return self.formatter.columns(sobject_fields)
|