castor-extractor 0.15.4__tar.gz → 0.16.3__tar.gz

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.

Potentially problematic release.


This version of castor-extractor might be problematic. Click here for more details.

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