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.

Files changed (367) hide show
  1. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/CHANGELOG.md +4 -0
  2. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/PKG-INFO +1 -1
  3. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_databricks.py +3 -0
  4. castor_extractor-0.16.4/castor_extractor/commands/extract_salesforce.py +43 -0
  5. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_salesforce_reporting.py +6 -6
  6. castor_extractor-0.16.4/castor_extractor/utils/client/api.py +59 -0
  7. castor_extractor-0.16.4/castor_extractor/utils/salesforce/__init__.py +3 -0
  8. castor_extractor-0.16.4/castor_extractor/utils/salesforce/client.py +84 -0
  9. castor_extractor-0.16.4/castor_extractor/utils/salesforce/client_test.py +21 -0
  10. castor_extractor-0.16.4/castor_extractor/utils/salesforce/constants.py +13 -0
  11. castor_extractor-0.16.4/castor_extractor/utils/salesforce/credentials.py +65 -0
  12. {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
  13. castor_extractor-0.16.4/castor_extractor/visualization/salesforce_reporting/__init__.py +3 -0
  14. castor_extractor-0.16.4/castor_extractor/visualization/salesforce_reporting/client/__init__.py +1 -0
  15. castor_extractor-0.16.4/castor_extractor/visualization/salesforce_reporting/client/rest.py +60 -0
  16. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/salesforce_reporting/extract.py +10 -8
  17. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/databricks/client.py +1 -1
  18. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/databricks/credentials.py +1 -4
  19. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/databricks/extract.py +1 -1
  20. castor_extractor-0.16.4/castor_extractor/warehouse/salesforce/__init__.py +6 -0
  21. castor_extractor-0.16.4/castor_extractor/warehouse/salesforce/client.py +112 -0
  22. castor_extractor-0.16.4/castor_extractor/warehouse/salesforce/constants.py +2 -0
  23. castor_extractor-0.16.4/castor_extractor/warehouse/salesforce/extract.py +111 -0
  24. castor_extractor-0.16.4/castor_extractor/warehouse/salesforce/format.py +67 -0
  25. castor_extractor-0.16.4/castor_extractor/warehouse/salesforce/format_test.py +32 -0
  26. castor_extractor-0.16.4/castor_extractor/warehouse/salesforce/soql.py +45 -0
  27. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/pyproject.toml +3 -2
  28. castor_extractor-0.16.3/castor_extractor/utils/client/api.py +0 -50
  29. castor_extractor-0.16.3/castor_extractor/visualization/salesforce_reporting/__init__.py +0 -4
  30. castor_extractor-0.16.3/castor_extractor/visualization/salesforce_reporting/client/__init__.py +0 -2
  31. castor_extractor-0.16.3/castor_extractor/visualization/salesforce_reporting/client/constants.py +0 -2
  32. castor_extractor-0.16.3/castor_extractor/visualization/salesforce_reporting/client/credentials.py +0 -33
  33. castor_extractor-0.16.3/castor_extractor/visualization/salesforce_reporting/client/rest.py +0 -143
  34. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/Dockerfile +0 -0
  35. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/LICENCE +0 -0
  36. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/README.md +0 -0
  37. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/__init__.py +0 -0
  38. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/__init__.py +0 -0
  39. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_bigquery.py +0 -0
  40. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_domo.py +0 -0
  41. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_looker.py +0 -0
  42. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_metabase_api.py +0 -0
  43. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_metabase_db.py +0 -0
  44. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_mode.py +0 -0
  45. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_mysql.py +0 -0
  46. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_postgres.py +0 -0
  47. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_powerbi.py +0 -0
  48. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_qlik.py +0 -0
  49. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_redshift.py +0 -0
  50. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_sigma.py +0 -0
  51. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_snowflake.py +0 -0
  52. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_sqlserver.py +0 -0
  53. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/extract_tableau.py +0 -0
  54. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/file_check.py +0 -0
  55. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/commands/upload.py +0 -0
  56. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/file_checker/__init__.py +0 -0
  57. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/file_checker/column.py +0 -0
  58. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/file_checker/column_test.py +0 -0
  59. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/file_checker/constants.py +0 -0
  60. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/file_checker/enums.py +0 -0
  61. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/file_checker/file.py +0 -0
  62. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/file_checker/file_test.py +0 -0
  63. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/file_checker/file_test_users.csv +0 -0
  64. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/file_checker/file_test_users_valid.csv +0 -0
  65. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/file_checker/templates/__init__.py +0 -0
  66. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/file_checker/templates/generic_warehouse.py +0 -0
  67. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/logger.py +0 -0
  68. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/types.py +0 -0
  69. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/uploader/__init__.py +0 -0
  70. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/uploader/constant.py +0 -0
  71. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/uploader/env.py +0 -0
  72. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/uploader/env_test.py +0 -0
  73. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/uploader/upload.py +0 -0
  74. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/uploader/upload_test.py +0 -0
  75. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/uploader/utils.py +0 -0
  76. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/__init__.py +0 -0
  77. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/client/__init__.py +0 -0
  78. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/client/abstract.py +0 -0
  79. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/client/api_test.py +0 -0
  80. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/client/postgres.py +0 -0
  81. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/client/query.py +0 -0
  82. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/client/uri.py +0 -0
  83. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/client/uri_test.py +0 -0
  84. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/collection.py +0 -0
  85. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/constants.py +0 -0
  86. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/dbt/__init__.py +0 -0
  87. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/dbt/assets.py +0 -0
  88. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/dbt/client.py +0 -0
  89. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/dbt/client_test.py +0 -0
  90. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/dbt/credentials.py +0 -0
  91. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/deprecate.py +0 -0
  92. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/env.py +0 -0
  93. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/files.py +0 -0
  94. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/files_test.py +0 -0
  95. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/formatter.py +0 -0
  96. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/formatter_test.csv +0 -0
  97. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/formatter_test.json +0 -0
  98. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/formatter_test.py +0 -0
  99. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/json_stream_write.py +0 -0
  100. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/load.py +0 -0
  101. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/object.py +0 -0
  102. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/object_test.py +0 -0
  103. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/pager/__init__.py +0 -0
  104. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/pager/pager.py +0 -0
  105. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/pager/pager_on_id.py +0 -0
  106. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/pager/pager_on_id_test.py +0 -0
  107. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/pager/pager_on_token.py +0 -0
  108. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/pager/pager_on_token_test.py +0 -0
  109. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/pager/pager_test.py +0 -0
  110. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/retry.py +0 -0
  111. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/retry_test.py +0 -0
  112. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/safe.py +0 -0
  113. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/safe_test.py +0 -0
  114. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/store.py +0 -0
  115. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/string.py +0 -0
  116. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/string_test.py +0 -0
  117. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/time.py +0 -0
  118. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/time_test.py +0 -0
  119. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/type.py +0 -0
  120. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/validation.py +0 -0
  121. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/validation_test.py +0 -0
  122. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/utils/write.py +0 -0
  123. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/__init__.py +0 -0
  124. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/domo/__init__.py +0 -0
  125. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/domo/assets.py +0 -0
  126. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/domo/client/__init__.py +0 -0
  127. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/domo/client/client.py +0 -0
  128. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/domo/client/client_test.py +0 -0
  129. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/domo/client/credentials.py +0 -0
  130. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/domo/client/endpoints.py +0 -0
  131. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/domo/client/pagination.py +0 -0
  132. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/domo/client/pagination_test.py +0 -0
  133. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/domo/constants.py +0 -0
  134. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/domo/extract.py +0 -0
  135. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/looker/__init__.py +0 -0
  136. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/looker/api/__init__.py +0 -0
  137. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/looker/api/client.py +0 -0
  138. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/looker/api/client_test.py +0 -0
  139. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/looker/api/constants.py +0 -0
  140. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/looker/api/sdk.py +0 -0
  141. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/looker/api/sdk_test.py +0 -0
  142. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/looker/api/utils.py +0 -0
  143. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/looker/assets.py +0 -0
  144. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/looker/constant.py +0 -0
  145. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/looker/env.py +0 -0
  146. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/looker/extract.py +0 -0
  147. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/looker/fields.py +0 -0
  148. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/looker/fields_test.py +0 -0
  149. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/looker/multithreading.py +0 -0
  150. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/looker/parameters.py +0 -0
  151. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/__init__.py +0 -0
  152. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/assets.py +0 -0
  153. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/__init__.py +0 -0
  154. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/api/__init__.py +0 -0
  155. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/api/client.py +0 -0
  156. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/api/client_test.py +0 -0
  157. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/api/credentials.py +0 -0
  158. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/db/__init__.py +0 -0
  159. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/db/client.py +0 -0
  160. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/db/credentials.py +0 -0
  161. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/db/queries/.sqlfluff +0 -0
  162. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/db/queries/base_url.sql +0 -0
  163. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/db/queries/card.sql +0 -0
  164. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/db/queries/collection.sql +0 -0
  165. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/db/queries/dashboard.sql +0 -0
  166. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/db/queries/dashboard_cards.sql +0 -0
  167. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/db/queries/database.sql +0 -0
  168. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/db/queries/table.sql +0 -0
  169. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/db/queries/user.sql +0 -0
  170. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/decryption.py +0 -0
  171. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/decryption_test.py +0 -0
  172. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/client/shared.py +0 -0
  173. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/errors.py +0 -0
  174. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/extract.py +0 -0
  175. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/metabase/types.py +0 -0
  176. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/mode/__init__.py +0 -0
  177. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/mode/assets.py +0 -0
  178. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/mode/client/__init__.py +0 -0
  179. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/mode/client/client.py +0 -0
  180. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/mode/client/client_test.json +0 -0
  181. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/mode/client/client_test.py +0 -0
  182. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/mode/client/constants.py +0 -0
  183. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/mode/client/credentials.py +0 -0
  184. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/mode/errors.py +0 -0
  185. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/mode/extract.py +0 -0
  186. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/powerbi/__init__.py +0 -0
  187. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/powerbi/assets.py +0 -0
  188. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/powerbi/client/__init__.py +0 -0
  189. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/powerbi/client/constants.py +0 -0
  190. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/powerbi/client/credentials.py +0 -0
  191. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/powerbi/client/credentials_test.py +0 -0
  192. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/powerbi/client/rest.py +0 -0
  193. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/powerbi/client/rest_test.py +0 -0
  194. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/powerbi/client/utils.py +0 -0
  195. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/powerbi/client/utils_test.py +0 -0
  196. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/powerbi/extract.py +0 -0
  197. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/__init__.py +0 -0
  198. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/assets.py +0 -0
  199. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/client/__init__.py +0 -0
  200. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/client/constants.py +0 -0
  201. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/client/engine/__init__.py +0 -0
  202. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/client/engine/client.py +0 -0
  203. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/client/engine/constants.py +0 -0
  204. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/client/engine/error.py +0 -0
  205. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/client/engine/error_test.py +0 -0
  206. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/client/engine/json_rpc.py +0 -0
  207. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/client/engine/json_rpc_test.py +0 -0
  208. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/client/engine/websocket.py +0 -0
  209. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/client/master.py +0 -0
  210. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/client/rest.py +0 -0
  211. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/client/rest_test.py +0 -0
  212. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/constants.py +0 -0
  213. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/qlik/extract.py +0 -0
  214. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/salesforce_reporting/assets.py +0 -0
  215. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/salesforce_reporting/client/soql.py +0 -0
  216. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/sigma/__init__.py +0 -0
  217. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/sigma/assets.py +0 -0
  218. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/sigma/client/__init__.py +0 -0
  219. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/sigma/client/client.py +0 -0
  220. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/sigma/client/client_test.py +0 -0
  221. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/sigma/client/credentials.py +0 -0
  222. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/sigma/client/endpoints.py +0 -0
  223. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/sigma/client/pagination.py +0 -0
  224. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/sigma/constants.py +0 -0
  225. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/sigma/extract.py +0 -0
  226. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/__init__.py +0 -0
  227. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/assets.py +0 -0
  228. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/client/__init__.py +0 -0
  229. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/client/client.py +0 -0
  230. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/client/client_utils.py +0 -0
  231. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/client/credentials.py +0 -0
  232. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/client/project.py +0 -0
  233. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/client/safe_mode.py +0 -0
  234. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/constants.py +0 -0
  235. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/errors.py +0 -0
  236. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/extract.py +0 -0
  237. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/gql_fields.py +0 -0
  238. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/__init__.py +0 -0
  239. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/__init__.py +0 -0
  240. {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
  241. {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
  242. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/assets/rest_api/auth.xml +0 -0
  243. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/assets/rest_api/project_get.xml +0 -0
  244. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/assets/rest_api/user_get.xml +0 -0
  245. {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
  246. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/assets/rest_api/workbook_get.xml +0 -0
  247. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/graphql/__init__.py +0 -0
  248. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/graphql/paginated_object_test.py +0 -0
  249. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/rest_api/__init__.py +0 -0
  250. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/rest_api/auth_test.py +0 -0
  251. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/rest_api/credentials_test.py +0 -0
  252. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/rest_api/projects_test.py +0 -0
  253. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/rest_api/usages_test.py +0 -0
  254. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/rest_api/users_test.py +0 -0
  255. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/rest_api/workbooks_test.py +0 -0
  256. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/utils/__init__.py +0 -0
  257. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tests/unit/utils/env_key.py +0 -0
  258. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/tsc_fields.py +0 -0
  259. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/types.py +0 -0
  260. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/visualization/tableau/usage.py +0 -0
  261. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/__init__.py +0 -0
  262. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/abstract/__init__.py +0 -0
  263. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/abstract/asset.py +0 -0
  264. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/abstract/asset_test.py +0 -0
  265. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/abstract/extract.py +0 -0
  266. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/abstract/query.py +0 -0
  267. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/abstract/time_filter.py +0 -0
  268. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/abstract/time_filter_test.py +0 -0
  269. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/__init__.py +0 -0
  270. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/client.py +0 -0
  271. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/client_test.py +0 -0
  272. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/credentials.py +0 -0
  273. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/extract.py +0 -0
  274. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/queries/.sqlfluff +0 -0
  275. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/queries/column.sql +0 -0
  276. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/queries/cte/sharded.sql +0 -0
  277. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/queries/database.sql +0 -0
  278. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/queries/query.sql +0 -0
  279. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/queries/schema.sql +0 -0
  280. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/queries/table.sql +0 -0
  281. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/queries/table_with_tags.sql +0 -0
  282. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/queries/user.sql +0 -0
  283. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/queries/view_ddl.sql +0 -0
  284. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/query.py +0 -0
  285. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/bigquery/types.py +0 -0
  286. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/databricks/__init__.py +0 -0
  287. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/databricks/client_test.py +0 -0
  288. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/databricks/format.py +0 -0
  289. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/databricks/format_test.py +0 -0
  290. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/databricks/types.py +0 -0
  291. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/mysql/__init__.py +0 -0
  292. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/mysql/client.py +0 -0
  293. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/mysql/client_test.py +0 -0
  294. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/mysql/extract.py +0 -0
  295. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/mysql/queries/.sqlfluff +0 -0
  296. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/mysql/queries/column.sql +0 -0
  297. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/mysql/queries/database.sql +0 -0
  298. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/mysql/queries/query.sql +0 -0
  299. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/mysql/queries/schema.sql +0 -0
  300. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/mysql/queries/table.sql +0 -0
  301. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/mysql/queries/user.sql +0 -0
  302. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/mysql/queries/view_ddl.sql +0 -0
  303. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/mysql/query.py +0 -0
  304. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/postgres/__init__.py +0 -0
  305. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/postgres/extract.py +0 -0
  306. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/postgres/queries/.sqlfluff +0 -0
  307. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/postgres/queries/column.sql +0 -0
  308. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/postgres/queries/database.sql +0 -0
  309. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/postgres/queries/group.sql +0 -0
  310. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/postgres/queries/schema.sql +0 -0
  311. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/postgres/queries/table.sql +0 -0
  312. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/postgres/queries/user.sql +0 -0
  313. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/postgres/query.py +0 -0
  314. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/redshift/__init__.py +0 -0
  315. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/redshift/client.py +0 -0
  316. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/redshift/client_test.py +0 -0
  317. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/redshift/extract.py +0 -0
  318. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/redshift/queries/.sqlfluff +0 -0
  319. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/redshift/queries/column.sql +0 -0
  320. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/redshift/queries/database.sql +0 -0
  321. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/redshift/queries/group.sql +0 -0
  322. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/redshift/queries/query.sql +0 -0
  323. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/redshift/queries/schema.sql +0 -0
  324. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/redshift/queries/table.sql +0 -0
  325. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/redshift/queries/table_freshness.sql +0 -0
  326. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/redshift/queries/user.sql +0 -0
  327. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/redshift/queries/view_ddl.sql +0 -0
  328. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/redshift/query.py +0 -0
  329. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/__init__.py +0 -0
  330. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/client.py +0 -0
  331. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/client_test.py +0 -0
  332. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/credentials.py +0 -0
  333. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/credentials_test.py +0 -0
  334. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/extract.py +0 -0
  335. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/queries/.sqlfluff +0 -0
  336. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/queries/column.sql +0 -0
  337. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/queries/column_lineage.sql +0 -0
  338. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/queries/database.sql +0 -0
  339. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/queries/grant_to_role.sql +0 -0
  340. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/queries/grant_to_user.sql +0 -0
  341. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/queries/query.sql +0 -0
  342. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/queries/role.sql +0 -0
  343. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/queries/schema.sql +0 -0
  344. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/queries/table.sql +0 -0
  345. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/queries/user.sql +0 -0
  346. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/queries/view_ddl.sql +0 -0
  347. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/snowflake/query.py +0 -0
  348. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/sqlserver/__init__.py +0 -0
  349. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/sqlserver/client.py +0 -0
  350. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/sqlserver/extract.py +0 -0
  351. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/sqlserver/queries/.sqlfluff +0 -0
  352. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/sqlserver/queries/column.sql +0 -0
  353. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/sqlserver/queries/database.sql +0 -0
  354. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/sqlserver/queries/schema.sql +0 -0
  355. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/sqlserver/queries/table.sql +0 -0
  356. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/sqlserver/queries/user.sql +0 -0
  357. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/sqlserver/query.py +0 -0
  358. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/synapse/__init__.py +0 -0
  359. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/synapse/extract.py +0 -0
  360. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/synapse/queries/.sqlfluff +0 -0
  361. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/synapse/queries/column.sql +0 -0
  362. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/synapse/queries/database.sql +0 -0
  363. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/synapse/queries/query.sql +0 -0
  364. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/synapse/queries/schema.sql +0 -0
  365. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/synapse/queries/table.sql +0 -0
  366. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/synapse/queries/user.sql +0 -0
  367. {castor_extractor-0.16.3 → castor_extractor-0.16.4}/castor_extractor/warehouse/synapse/queries/view_ddl.sql +0 -0
@@ -1,5 +1,9 @@
1
1
  # Changelog
2
2
 
3
+ ## 0.16.4 - 2024-04-25
4
+
5
+ * Salesforce: extract sobjects and fields
6
+
3
7
  ## 0.16.3 - 2024-04-24
4
8
 
5
9
  * Databricks: Extract table owners
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: castor-extractor
3
- Version: 0.16.3
3
+ Version: 0.16.4
4
4
  Summary: Extract your metadata assets.
5
5
  Home-page: https://www.castordoc.com/
6
6
  License: EULA
@@ -1,7 +1,10 @@
1
+ import logging
1
2
  from argparse import ArgumentParser
2
3
 
3
4
  from castor_extractor.warehouse import databricks # type: ignore
4
5
 
6
+ logging.basicConfig(level=logging.INFO, format="%(levelname)s - %(message)s")
7
+
5
8
 
6
9
  def main():
7
10
  parser = ArgumentParser()
@@ -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("-k", "--consumer-key", help="Salesforce consumer key")
14
+ parser.add_argument("-c", "--client-id", help="Salesforce client id")
15
15
  parser.add_argument(
16
- "-s", "--consumer-secret", help="Salesforce consumer secret"
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("-l", "--url", help="Salesforce instance URL")
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
- consumer_key=args.consumer_key,
29
- consumer_secret=args.consumer_secret,
28
+ client_id=args.client_id,
29
+ client_secret=args.client_secret,
30
30
  security_token=args.security_token,
31
- instance_url=args.url,
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,3 @@
1
+ from .client import SalesforceBaseClient
2
+ from .constants import Keys
3
+ from .credentials import SalesforceCredentials, to_credentials
@@ -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
- consumer_key="degenie",
9
- consumer_secret="fautpasledire",
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()
@@ -0,0 +1,3 @@
1
+ from .assets import SalesforceReportingAsset
2
+ from .client import SalesforceReportingClient
3
+ from .extract import extract_all
@@ -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 SalesforceClient, SalesforceCredentials
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: SalesforceClient,
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
- consumer_key: str,
34
- consumer_secret: str,
34
+ client_id: str,
35
+ client_secret: str,
35
36
  security_token: str,
36
- instance_url: str,
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
- consumer_key=consumer_key,
48
- consumer_secret=consumer_secret,
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 = SalesforceClient(credentials=creds, instance_url=instance_url)
53
+ client = SalesforceReportingClient(credentials=creds)
52
54
  ts = current_timestamp()
53
55
 
54
56
  for key, data in iterate_all_data(client):
@@ -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)
@@ -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,6 @@
1
+ from .client import SalesforceClient
2
+ from .extract import (
3
+ SALESFORCE_ASSETS,
4
+ SalesforceExtractionProcessor,
5
+ extract_all,
6
+ )
@@ -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)
@@ -0,0 +1,2 @@
1
+ DATABASE_NAME = "salesforce"
2
+ SCHEMA_NAME = "schema"