castor-extractor 0.22.0__tar.gz → 0.22.5__tar.gz

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

Potentially problematic release.


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

Files changed (444) hide show
  1. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/CHANGELOG.md +20 -0
  2. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/PKG-INFO +23 -1
  3. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/__init__.py +1 -0
  4. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/time.py +4 -0
  5. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/time_test.py +8 -1
  6. castor_extractor-0.22.5/castor_extractor/visualization/looker_studio/__init__.py +6 -0
  7. castor_extractor-0.22.5/castor_extractor/visualization/looker_studio/assets.py +6 -0
  8. castor_extractor-0.22.5/castor_extractor/visualization/looker_studio/client/__init__.py +3 -0
  9. castor_extractor-0.22.5/castor_extractor/visualization/looker_studio/client/admin_sdk_client.py +90 -0
  10. castor_extractor-0.22.5/castor_extractor/visualization/looker_studio/client/client.py +37 -0
  11. castor_extractor-0.22.5/castor_extractor/visualization/looker_studio/client/credentials.py +20 -0
  12. castor_extractor-0.22.5/castor_extractor/visualization/looker_studio/client/endpoints.py +18 -0
  13. castor_extractor-0.22.5/castor_extractor/visualization/looker_studio/client/enums.py +8 -0
  14. castor_extractor-0.22.5/castor_extractor/visualization/looker_studio/client/looker_studio_api_client.py +102 -0
  15. castor_extractor-0.22.5/castor_extractor/visualization/looker_studio/client/pagination.py +31 -0
  16. castor_extractor-0.22.5/castor_extractor/visualization/looker_studio/client/scopes.py +6 -0
  17. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/sigma/client/client.py +64 -10
  18. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/thoughtspot/assets.py +3 -1
  19. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/thoughtspot/client/client.py +67 -14
  20. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/thoughtspot/client/utils.py +10 -4
  21. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/thoughtspot/client/utils_test.py +22 -4
  22. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/databricks/api_client.py +2 -60
  23. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/databricks/client.py +4 -47
  24. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/databricks/client_test.py +1 -35
  25. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/databricks/credentials.py +4 -6
  26. castor_extractor-0.22.5/castor_extractor/warehouse/databricks/enums.py +15 -0
  27. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/databricks/extract.py +13 -11
  28. castor_extractor-0.22.5/castor_extractor/warehouse/databricks/lineage.py +69 -0
  29. castor_extractor-0.22.5/castor_extractor/warehouse/databricks/lineage_test.py +89 -0
  30. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/databricks/sql_client.py +23 -8
  31. castor_extractor-0.22.5/castor_extractor/warehouse/databricks/types.py +1 -0
  32. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/salesforce/format.py +12 -5
  33. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/salesforce/format_test.py +22 -6
  34. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/pyproject.toml +4 -1
  35. castor_extractor-0.22.0/castor_extractor/warehouse/databricks/lineage.py +0 -141
  36. castor_extractor-0.22.0/castor_extractor/warehouse/databricks/lineage_test.py +0 -34
  37. castor_extractor-0.22.0/castor_extractor/warehouse/databricks/test_constants.py +0 -79
  38. castor_extractor-0.22.0/castor_extractor/warehouse/databricks/types.py +0 -8
  39. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/Dockerfile +0 -0
  40. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/DockerfileUsage.md +0 -0
  41. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/LICENCE +0 -0
  42. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/README.md +0 -0
  43. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/__init__.py +0 -0
  44. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/__init__.py +0 -0
  45. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_bigquery.py +0 -0
  46. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_confluence.py +0 -0
  47. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_databricks.py +0 -0
  48. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_domo.py +0 -0
  49. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_looker.py +0 -0
  50. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_metabase_api.py +0 -0
  51. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_metabase_db.py +0 -0
  52. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_mode.py +0 -0
  53. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_mysql.py +0 -0
  54. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_notion.py +0 -0
  55. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_postgres.py +0 -0
  56. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_powerbi.py +0 -0
  57. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_qlik.py +0 -0
  58. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_redshift.py +0 -0
  59. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_salesforce.py +0 -0
  60. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_salesforce_reporting.py +0 -0
  61. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_sigma.py +0 -0
  62. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_snowflake.py +0 -0
  63. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_sqlserver.py +0 -0
  64. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_tableau.py +0 -0
  65. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/extract_thoughtspot.py +0 -0
  66. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/file_check.py +0 -0
  67. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/commands/upload.py +0 -0
  68. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/file_checker/__init__.py +0 -0
  69. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/file_checker/column.py +0 -0
  70. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/file_checker/column_test.py +0 -0
  71. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/file_checker/constants.py +0 -0
  72. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/file_checker/enums.py +0 -0
  73. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/file_checker/file.py +0 -0
  74. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/file_checker/file_test.py +0 -0
  75. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/file_checker/file_test_users.csv +0 -0
  76. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/file_checker/file_test_users_valid.csv +0 -0
  77. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/file_checker/templates/__init__.py +0 -0
  78. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/file_checker/templates/generic_warehouse.py +0 -0
  79. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/__init__.py +0 -0
  80. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/confluence/__init__.py +0 -0
  81. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/confluence/assets.py +0 -0
  82. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/confluence/client/__init__.py +0 -0
  83. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/confluence/client/client.py +0 -0
  84. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/confluence/client/credentials.py +0 -0
  85. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/confluence/client/endpoints.py +0 -0
  86. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/confluence/client/pagination.py +0 -0
  87. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/confluence/extract.py +0 -0
  88. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/notion/__init__.py +0 -0
  89. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/notion/assets.py +0 -0
  90. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/notion/client/__init__.py +0 -0
  91. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/notion/client/client.py +0 -0
  92. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/notion/client/client_test.py +0 -0
  93. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/notion/client/constants.py +0 -0
  94. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/notion/client/credentials.py +0 -0
  95. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/notion/client/endpoints.py +0 -0
  96. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/notion/client/pagination.py +0 -0
  97. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/knowledge/notion/extract.py +0 -0
  98. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/logger.py +0 -0
  99. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/quality/__init__.py +0 -0
  100. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/quality/soda/__init__.py +0 -0
  101. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/quality/soda/assets.py +0 -0
  102. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/quality/soda/client/__init__.py +0 -0
  103. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/quality/soda/client/client.py +0 -0
  104. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/quality/soda/client/credentials.py +0 -0
  105. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/quality/soda/client/endpoints.py +0 -0
  106. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/quality/soda/client/pagination.py +0 -0
  107. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/types.py +0 -0
  108. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/uploader/__init__.py +0 -0
  109. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/uploader/constant.py +0 -0
  110. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/uploader/env.py +0 -0
  111. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/uploader/env_test.py +0 -0
  112. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/uploader/settings.py +0 -0
  113. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/uploader/upload.py +0 -0
  114. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/uploader/upload_test.py +0 -0
  115. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/uploader/utils.py +0 -0
  116. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/argument_parser.py +0 -0
  117. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/argument_parser_test.py +0 -0
  118. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/__init__.py +0 -0
  119. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/abstract.py +0 -0
  120. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/api/__init__.py +0 -0
  121. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/api/auth.py +0 -0
  122. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/api/auth_test.py +0 -0
  123. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/api/client.py +0 -0
  124. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/api/client_test.py +0 -0
  125. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/api/pagination.py +0 -0
  126. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/api/pagination_test.py +0 -0
  127. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/api/safe_request.py +0 -0
  128. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/api/safe_request_test.py +0 -0
  129. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/api/utils.py +0 -0
  130. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/api/utils_test.py +0 -0
  131. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/postgres.py +0 -0
  132. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/query.py +0 -0
  133. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/uri.py +0 -0
  134. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/client/uri_test.py +0 -0
  135. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/collection.py +0 -0
  136. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/collection_test.py +0 -0
  137. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/constants.py +0 -0
  138. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/dbt/__init__.py +0 -0
  139. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/dbt/assets.py +0 -0
  140. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/dbt/client.py +0 -0
  141. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/dbt/client_test.py +0 -0
  142. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/dbt/credentials.py +0 -0
  143. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/deprecate.py +0 -0
  144. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/env.py +0 -0
  145. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/files.py +0 -0
  146. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/files_test.py +0 -0
  147. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/formatter.py +0 -0
  148. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/formatter_test.csv +0 -0
  149. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/formatter_test.json +0 -0
  150. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/formatter_test.py +0 -0
  151. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/json_stream_write.py +0 -0
  152. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/load.py +0 -0
  153. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/object.py +0 -0
  154. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/object_test.py +0 -0
  155. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/pager/__init__.py +0 -0
  156. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/pager/pager.py +0 -0
  157. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/pager/pager_on_id.py +0 -0
  158. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/pager/pager_on_id_test.py +0 -0
  159. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/pager/pager_test.py +0 -0
  160. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/retry.py +0 -0
  161. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/retry_test.py +0 -0
  162. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/safe.py +0 -0
  163. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/safe_test.py +0 -0
  164. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/salesforce/__init__.py +0 -0
  165. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/salesforce/client.py +0 -0
  166. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/salesforce/client_test.py +0 -0
  167. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/salesforce/constants.py +0 -0
  168. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/salesforce/credentials.py +0 -0
  169. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/salesforce/credentials_test.py +0 -0
  170. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/salesforce/pagination.py +0 -0
  171. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/store.py +0 -0
  172. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/string.py +0 -0
  173. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/string_test.py +0 -0
  174. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/type.py +0 -0
  175. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/validation.py +0 -0
  176. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/validation_test.py +0 -0
  177. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/utils/write.py +0 -0
  178. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/__init__.py +0 -0
  179. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/domo/__init__.py +0 -0
  180. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/domo/assets.py +0 -0
  181. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/domo/client/__init__.py +0 -0
  182. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/domo/client/client.py +0 -0
  183. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/domo/client/credentials.py +0 -0
  184. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/domo/client/endpoints.py +0 -0
  185. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/domo/client/pagination.py +0 -0
  186. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/domo/client/pagination_test.py +0 -0
  187. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/domo/extract.py +0 -0
  188. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/__init__.py +0 -0
  189. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/api/__init__.py +0 -0
  190. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/api/client.py +0 -0
  191. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/api/client_test.py +0 -0
  192. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/api/constants.py +0 -0
  193. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/api/credentials.py +0 -0
  194. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/api/extraction_parameters.py +0 -0
  195. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/api/sdk.py +0 -0
  196. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/api/sdk_test.py +0 -0
  197. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/api/utils.py +0 -0
  198. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/assets.py +0 -0
  199. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/constant.py +0 -0
  200. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/constants.py +0 -0
  201. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/extract.py +0 -0
  202. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/fields.py +0 -0
  203. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/fields_test.py +0 -0
  204. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/looker/multithreading.py +0 -0
  205. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/__init__.py +0 -0
  206. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/assets.py +0 -0
  207. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/__init__.py +0 -0
  208. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/api/__init__.py +0 -0
  209. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/api/client.py +0 -0
  210. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/api/client_test.py +0 -0
  211. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/api/credentials.py +0 -0
  212. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/db/__init__.py +0 -0
  213. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/db/client.py +0 -0
  214. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/db/credentials.py +0 -0
  215. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/db/queries/.sqlfluff +0 -0
  216. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/db/queries/base_url.sql +0 -0
  217. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/db/queries/card.sql +0 -0
  218. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/db/queries/collection.sql +0 -0
  219. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/db/queries/dashboard.sql +0 -0
  220. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/db/queries/dashboard_cards.sql +0 -0
  221. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/db/queries/database.sql +0 -0
  222. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/db/queries/table.sql +0 -0
  223. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/db/queries/user.sql +0 -0
  224. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/decryption.py +0 -0
  225. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/decryption_test.py +0 -0
  226. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/client/shared.py +0 -0
  227. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/errors.py +0 -0
  228. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/extract.py +0 -0
  229. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/metabase/types.py +0 -0
  230. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/mode/__init__.py +0 -0
  231. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/mode/assets.py +0 -0
  232. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/mode/client/__init__.py +0 -0
  233. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/mode/client/client.py +0 -0
  234. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/mode/client/client_test.json +0 -0
  235. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/mode/client/client_test.py +0 -0
  236. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/mode/client/constants.py +0 -0
  237. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/mode/client/credentials.py +0 -0
  238. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/mode/errors.py +0 -0
  239. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/mode/extract.py +0 -0
  240. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/powerbi/__init__.py +0 -0
  241. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/powerbi/assets.py +0 -0
  242. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/powerbi/client/__init__.py +0 -0
  243. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/powerbi/client/authentication.py +0 -0
  244. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/powerbi/client/client.py +0 -0
  245. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/powerbi/client/client_test.py +0 -0
  246. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/powerbi/client/constants.py +0 -0
  247. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/powerbi/client/credentials.py +0 -0
  248. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/powerbi/client/credentials_test.py +0 -0
  249. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/powerbi/client/endpoints.py +0 -0
  250. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/powerbi/client/pagination.py +0 -0
  251. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/powerbi/extract.py +0 -0
  252. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/__init__.py +0 -0
  253. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/assets.py +0 -0
  254. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/client/__init__.py +0 -0
  255. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/client/constants.py +0 -0
  256. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/client/engine/__init__.py +0 -0
  257. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/client/engine/client.py +0 -0
  258. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/client/engine/constants.py +0 -0
  259. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/client/engine/credentials.py +0 -0
  260. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/client/engine/error.py +0 -0
  261. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/client/engine/error_test.py +0 -0
  262. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/client/engine/json_rpc.py +0 -0
  263. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/client/engine/json_rpc_test.py +0 -0
  264. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/client/engine/websocket.py +0 -0
  265. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/client/master.py +0 -0
  266. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/client/rest.py +0 -0
  267. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/client/rest_test.py +0 -0
  268. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/qlik/extract.py +0 -0
  269. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/salesforce_reporting/__init__.py +0 -0
  270. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/salesforce_reporting/assets.py +0 -0
  271. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/salesforce_reporting/client/__init__.py +0 -0
  272. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/salesforce_reporting/client/rest.py +0 -0
  273. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/salesforce_reporting/client/soql.py +0 -0
  274. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/salesforce_reporting/extract.py +0 -0
  275. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/sigma/__init__.py +0 -0
  276. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/sigma/assets.py +0 -0
  277. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/sigma/client/__init__.py +0 -0
  278. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/sigma/client/credentials.py +0 -0
  279. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/sigma/client/endpoints.py +0 -0
  280. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/sigma/client/pagination.py +0 -0
  281. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/sigma/extract.py +0 -0
  282. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/__init__.py +0 -0
  283. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/assets.py +0 -0
  284. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/client/__init__.py +0 -0
  285. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/client/client.py +0 -0
  286. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/client/client_utils.py +0 -0
  287. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/client/credentials.py +0 -0
  288. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/client/project.py +0 -0
  289. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/client/safe_mode.py +0 -0
  290. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/constants.py +0 -0
  291. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/errors.py +0 -0
  292. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/extract.py +0 -0
  293. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/gql_fields.py +0 -0
  294. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/__init__.py +0 -0
  295. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/__init__.py +0 -0
  296. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/assets/graphql/metadata/metadata_1_get.json +0 -0
  297. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/assets/graphql/metadata/metadata_2_get.json +0 -0
  298. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/assets/rest_api/auth.xml +0 -0
  299. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/assets/rest_api/project_get.xml +0 -0
  300. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/assets/rest_api/user_get.xml +0 -0
  301. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/assets/rest_api/view_get_usage.xml +0 -0
  302. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/assets/rest_api/workbook_get.xml +0 -0
  303. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/graphql/__init__.py +0 -0
  304. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/graphql/paginated_object_test.py +0 -0
  305. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/rest_api/__init__.py +0 -0
  306. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/rest_api/auth_test.py +0 -0
  307. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/rest_api/credentials_test.py +0 -0
  308. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/rest_api/projects_test.py +0 -0
  309. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/rest_api/usages_test.py +0 -0
  310. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/rest_api/users_test.py +0 -0
  311. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/rest_api/workbooks_test.py +0 -0
  312. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/utils/__init__.py +0 -0
  313. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tests/unit/utils/env_key.py +0 -0
  314. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/tsc_fields.py +0 -0
  315. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/types.py +0 -0
  316. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau/usage.py +0 -0
  317. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau_revamp/__init__.py +0 -0
  318. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau_revamp/assets.py +0 -0
  319. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau_revamp/client/__init__.py +0 -0
  320. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau_revamp/client/client.py +0 -0
  321. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau_revamp/client/client_metadata_api.py +0 -0
  322. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau_revamp/client/client_rest_api.py +0 -0
  323. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau_revamp/client/client_tsc.py +0 -0
  324. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau_revamp/client/credentials.py +0 -0
  325. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau_revamp/client/errors.py +0 -0
  326. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau_revamp/client/gql_queries.py +0 -0
  327. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau_revamp/client/rest_fields.py +0 -0
  328. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau_revamp/constants.py +0 -0
  329. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/tableau_revamp/extract.py +0 -0
  330. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/thoughtspot/__init__.py +0 -0
  331. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/thoughtspot/client/__init__.py +0 -0
  332. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/thoughtspot/client/credentials.py +0 -0
  333. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/thoughtspot/client/endpoints.py +0 -0
  334. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/visualization/thoughtspot/extract.py +0 -0
  335. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/__init__.py +0 -0
  336. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/abstract/__init__.py +0 -0
  337. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/abstract/asset.py +0 -0
  338. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/abstract/asset_test.py +0 -0
  339. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/abstract/extract.py +0 -0
  340. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/abstract/query.py +0 -0
  341. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/abstract/time_filter.py +0 -0
  342. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/abstract/time_filter_test.py +0 -0
  343. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/__init__.py +0 -0
  344. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/client.py +0 -0
  345. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/client_test.py +0 -0
  346. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/credentials.py +0 -0
  347. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/extract.py +0 -0
  348. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/queries/.sqlfluff +0 -0
  349. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/queries/column.sql +0 -0
  350. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/queries/cte/sharded.sql +0 -0
  351. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/queries/database.sql +0 -0
  352. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/queries/query.sql +0 -0
  353. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/queries/schema.sql +0 -0
  354. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/queries/table.sql +0 -0
  355. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/queries/table_with_tags.sql +0 -0
  356. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/queries/user.sql +0 -0
  357. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/queries/view_ddl.sql +0 -0
  358. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/query.py +0 -0
  359. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/bigquery/types.py +0 -0
  360. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/databricks/__init__.py +0 -0
  361. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/databricks/api_client_test.py +0 -0
  362. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/databricks/endpoints.py +0 -0
  363. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/databricks/format.py +0 -0
  364. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/databricks/format_test.py +0 -0
  365. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/databricks/pagination.py +0 -0
  366. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/databricks/utils.py +0 -0
  367. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/databricks/utils_test.py +0 -0
  368. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/mysql/__init__.py +0 -0
  369. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/mysql/client.py +0 -0
  370. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/mysql/client_test.py +0 -0
  371. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/mysql/extract.py +0 -0
  372. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/mysql/queries/.sqlfluff +0 -0
  373. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/mysql/queries/column.sql +0 -0
  374. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/mysql/queries/database.sql +0 -0
  375. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/mysql/queries/query.sql +0 -0
  376. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/mysql/queries/schema.sql +0 -0
  377. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/mysql/queries/table.sql +0 -0
  378. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/mysql/queries/user.sql +0 -0
  379. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/mysql/queries/view_ddl.sql +0 -0
  380. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/mysql/query.py +0 -0
  381. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/postgres/__init__.py +0 -0
  382. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/postgres/extract.py +0 -0
  383. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/postgres/queries/.sqlfluff +0 -0
  384. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/postgres/queries/column.sql +0 -0
  385. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/postgres/queries/database.sql +0 -0
  386. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/postgres/queries/group.sql +0 -0
  387. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/postgres/queries/schema.sql +0 -0
  388. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/postgres/queries/table.sql +0 -0
  389. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/postgres/queries/user.sql +0 -0
  390. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/postgres/query.py +0 -0
  391. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/__init__.py +0 -0
  392. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/client.py +0 -0
  393. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/client_test.py +0 -0
  394. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/extract.py +0 -0
  395. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/extract_test.py +0 -0
  396. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/queries/.sqlfluff +0 -0
  397. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/queries/column.sql +0 -0
  398. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/queries/database.sql +0 -0
  399. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/queries/group.sql +0 -0
  400. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/queries/query.sql +0 -0
  401. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/queries/query_serverless.sql +0 -0
  402. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/queries/schema.sql +0 -0
  403. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/queries/table.sql +0 -0
  404. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/queries/table_freshness.sql +0 -0
  405. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/queries/user.sql +0 -0
  406. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/queries/view_ddl.sql +0 -0
  407. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/redshift/query.py +0 -0
  408. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/salesforce/__init__.py +0 -0
  409. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/salesforce/client.py +0 -0
  410. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/salesforce/constants.py +0 -0
  411. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/salesforce/extract.py +0 -0
  412. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/salesforce/pagination.py +0 -0
  413. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/salesforce/soql.py +0 -0
  414. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/__init__.py +0 -0
  415. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/client.py +0 -0
  416. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/client_test.py +0 -0
  417. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/credentials.py +0 -0
  418. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/credentials_test.py +0 -0
  419. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/extract.py +0 -0
  420. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/queries/.sqlfluff +0 -0
  421. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/queries/column.sql +0 -0
  422. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/queries/column_lineage.sql +0 -0
  423. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/queries/database.sql +0 -0
  424. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/queries/function.sql +0 -0
  425. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/queries/grant_to_role.sql +0 -0
  426. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/queries/grant_to_user.sql +0 -0
  427. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/queries/query.sql +0 -0
  428. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/queries/role.sql +0 -0
  429. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/queries/schema.sql +0 -0
  430. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/queries/table.sql +0 -0
  431. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/queries/user.sql +0 -0
  432. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/queries/view_ddl.sql +0 -0
  433. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/snowflake/query.py +0 -0
  434. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/sqlserver/__init__.py +0 -0
  435. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/sqlserver/client.py +0 -0
  436. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/sqlserver/extract.py +0 -0
  437. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/sqlserver/queries/.sqlfluff +0 -0
  438. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/sqlserver/queries/column.sql +0 -0
  439. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/sqlserver/queries/database.sql +0 -0
  440. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/sqlserver/queries/schema.sql +0 -0
  441. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/sqlserver/queries/table.sql +0 -0
  442. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/sqlserver/queries/user.sql +0 -0
  443. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/sqlserver/query.py +0 -0
  444. {castor_extractor-0.22.0 → castor_extractor-0.22.5}/castor_extractor/warehouse/synapse/queries/column.sql +0 -0
@@ -1,6 +1,26 @@
1
1
 
2
2
  # Changelog
3
3
 
4
+ ## 0.22.5 - 2025-01-09
5
+
6
+ * Databricks: validate and deduplicate lineage links
7
+
8
+ ## 0.22.4 - 2025-01-08
9
+
10
+ * ThoughtSpot: extract answers
11
+
12
+ ## 0.22.3 - 2024-12-10
13
+
14
+ * Databricks: extract lineage from system tables
15
+
16
+ ## 0.22.2 - 2024-12-06
17
+
18
+ * Sigma: multithreading to retrieve lineage
19
+
20
+ ## 0.22.1 - 2024-12-05
21
+
22
+ * Salesforce: deduplicate tables
23
+
4
24
  ## 0.22.0 - 2024-12-04
5
25
 
6
26
  * Stop supporting python3.8
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: castor-extractor
3
- Version: 0.22.0
3
+ Version: 0.22.5
4
4
  Summary: Extract your metadata assets.
5
5
  Home-page: https://www.castordoc.com/
6
6
  License: EULA
@@ -19,6 +19,7 @@ Provides-Extra: bigquery
19
19
  Provides-Extra: databricks
20
20
  Provides-Extra: dbt
21
21
  Provides-Extra: looker
22
+ Provides-Extra: lookerstudio
22
23
  Provides-Extra: metabase
23
24
  Provides-Extra: mysql
24
25
  Provides-Extra: postgres
@@ -31,6 +32,7 @@ Provides-Extra: tableau
31
32
  Requires-Dist: cryptography (>=43.0.0,<44.0.0) ; extra == "snowflake"
32
33
  Requires-Dist: databricks-sql-connector (>=3.2.0,<4.0.0) ; extra == "databricks" or extra == "all"
33
34
  Requires-Dist: google-api-core (>=2.1.1,<3.0.0)
35
+ Requires-Dist: google-api-python-client (>=2.121.0,<3.0.0) ; extra == "lookerstudio" or extra == "all"
34
36
  Requires-Dist: google-auth (>=2,<3)
35
37
  Requires-Dist: google-cloud-core (>=2.1.0,<3.0.0)
36
38
  Requires-Dist: google-cloud-storage (>=2,<3)
@@ -205,6 +207,26 @@ For any questions or bug report, contact us at [support@castordoc.com](mailto:su
205
207
 
206
208
  # Changelog
207
209
 
210
+ ## 0.22.5 - 2025-01-09
211
+
212
+ * Databricks: validate and deduplicate lineage links
213
+
214
+ ## 0.22.4 - 2025-01-08
215
+
216
+ * ThoughtSpot: extract answers
217
+
218
+ ## 0.22.3 - 2024-12-10
219
+
220
+ * Databricks: extract lineage from system tables
221
+
222
+ ## 0.22.2 - 2024-12-06
223
+
224
+ * Sigma: multithreading to retrieve lineage
225
+
226
+ ## 0.22.1 - 2024-12-05
227
+
228
+ * Salesforce: deduplicate tables
229
+
208
230
  ## 0.22.0 - 2024-12-04
209
231
 
210
232
  * Stop supporting python3.8
@@ -45,6 +45,7 @@ from .time import (
45
45
  current_timestamp,
46
46
  date_after,
47
47
  format_date,
48
+ format_rfc_3339_date,
48
49
  past_date,
49
50
  timestamp_ms,
50
51
  yesterday,
@@ -63,5 +63,9 @@ def format_date(timestamp: Union[datetime, date]) -> str:
63
63
  return timestamp.strftime(ISO_FORMAT)
64
64
 
65
65
 
66
+ def format_rfc_3339_date(timestamp: datetime) -> str:
67
+ return timestamp.isoformat(timespec="seconds") + "Z"
68
+
69
+
66
70
  def yesterday() -> date:
67
71
  return current_date() - timedelta(days=1)
@@ -1,6 +1,6 @@
1
1
  from datetime import date, datetime
2
2
 
3
- from .time import at_midnight, date_after, timestamp_ms
3
+ from .time import at_midnight, date_after, format_rfc_3339_date, timestamp_ms
4
4
 
5
5
 
6
6
  def test_at_midnight():
@@ -17,3 +17,10 @@ def test_timestamp_ms():
17
17
  result = timestamp_ms(dt)
18
18
  expected = 670636800000
19
19
  assert result == expected
20
+
21
+
22
+ def test_format_rfc_3339_date():
23
+ dt = datetime(1995, 4, 3, 2, 1)
24
+ result = format_rfc_3339_date(dt)
25
+ expected = "1995-04-03T02:01:00Z"
26
+ assert result == expected
@@ -0,0 +1,6 @@
1
+ from .assets import LookerStudioAsset
2
+ from .client import (
3
+ LookerStudioAssetType,
4
+ LookerStudioClient,
5
+ LookerStudioCredentials,
6
+ )
@@ -0,0 +1,6 @@
1
+ from ...types import ExternalAsset
2
+
3
+
4
+ class LookerStudioAsset(ExternalAsset):
5
+ ASSETS = "assets"
6
+ VIEW_ACTIVITY = "view_activity"
@@ -0,0 +1,3 @@
1
+ from .client import LookerStudioClient
2
+ from .credentials import LookerStudioCredentials
3
+ from .enums import LookerStudioAssetType
@@ -0,0 +1,90 @@
1
+ from typing import Iterator, Optional
2
+
3
+ from google.oauth2.service_account import Credentials
4
+ from googleapiclient import discovery # type: ignore
5
+
6
+ from ....utils import (
7
+ at_midnight,
8
+ current_date,
9
+ fetch_all_pages,
10
+ format_rfc_3339_date,
11
+ past_date,
12
+ )
13
+ from .credentials import LookerStudioCredentials
14
+ from .pagination import LookerStudioPagination
15
+ from .scopes import SCOPES
16
+
17
+ USER_EMAIL_FIELD = "primaryEmail"
18
+
19
+
20
+ class AdminSDKClient:
21
+ """
22
+ Client to call the Report API and Directory API.
23
+ The service account must impersonate and admin account.
24
+ """
25
+
26
+ def __init__(self, credentials: LookerStudioCredentials):
27
+ self._credentials = Credentials.from_service_account_info(
28
+ credentials.model_dump(),
29
+ scopes=SCOPES,
30
+ subject=credentials.admin_email, # impersonates an admin
31
+ )
32
+ self.directory_api = discovery.build(
33
+ "admin", "directory_v1", credentials=self._credentials
34
+ )
35
+ self.report_api = discovery.build(
36
+ "admin", "reports_v1", credentials=self._credentials
37
+ )
38
+
39
+ def list_users(self) -> Iterator[dict]:
40
+ """
41
+ Lists all users in the domain; only the primaryEmail field is selected.
42
+ Note:
43
+ * `my_customer` is an alias to represent the account's `customerId`
44
+ * `domain_public` allows non-admins to list users. This is technically
45
+ not necessary here because an admin account is impersonated, but it
46
+ avoids tapping into unnecessary data & serves for future reference.
47
+ See
48
+ https://googleapis.github.io/google-api-python-client/docs/dyn/admin_directory_v1.users.html#list
49
+ https://developers.google.com/admin-sdk/directory/reference/rest/v1/users/list
50
+ https://developers.google.com/admin-sdk/directory/v1/guides/manage-users#retrieve_users_non_admin
51
+ https://stackoverflow.com/a/71083443/14448410
52
+ """
53
+
54
+ def _users(pagination_params: Optional[dict] = None) -> dict:
55
+ parameters = {
56
+ "viewType": "domain_public",
57
+ "customer": "my_customer",
58
+ "fields": f"users({USER_EMAIL_FIELD}), nextPageToken",
59
+ **(pagination_params or {}),
60
+ }
61
+
62
+ return self.directory_api.users().list(**parameters).execute()
63
+
64
+ yield from fetch_all_pages(_users, LookerStudioPagination)
65
+
66
+ def list_view_events(self) -> Iterator[dict]:
67
+ """
68
+ Lists all Data Studio View events of the past day.
69
+ See
70
+ https://googleapis.github.io/google-api-python-client/docs/dyn/admin_reports_v1.activities.html
71
+ https://developers.google.com/admin-sdk/reports/reference/rest/v1/activities/list
72
+ https://developers.google.com/admin-sdk/reports/v1/appendix/activity/data-studio#VIEW
73
+ """
74
+
75
+ def _activity(pagination_params: Optional[dict] = None) -> dict:
76
+ yesterday = format_rfc_3339_date(at_midnight(past_date(1)))
77
+ today = format_rfc_3339_date(at_midnight(current_date()))
78
+
79
+ parameters = {
80
+ "userKey": "all",
81
+ "applicationName": "data_studio",
82
+ "eventName": "VIEW",
83
+ "startTime": yesterday,
84
+ "endTime": today,
85
+ **(pagination_params or {}),
86
+ }
87
+
88
+ return self.report_api.activities().list(**parameters).execute()
89
+
90
+ yield from fetch_all_pages(_activity, LookerStudioPagination)
@@ -0,0 +1,37 @@
1
+ from typing import Iterator
2
+
3
+ from .. import LookerStudioAsset
4
+ from .admin_sdk_client import USER_EMAIL_FIELD, AdminSDKClient
5
+ from .credentials import LookerStudioCredentials
6
+ from .looker_studio_api_client import LookerStudioAPIClient
7
+
8
+
9
+ class LookerStudioClient:
10
+ """
11
+ Acts as a wrapper class to fetch Looker Studio assets, which requires
12
+ coordinating calls between the Admin SDK API and the Looker Studio API.
13
+ """
14
+
15
+ def __init__(self, credentials: LookerStudioCredentials):
16
+ self.admin_sdk_client = AdminSDKClient(credentials)
17
+ self.looker_studio_client = LookerStudioAPIClient(credentials)
18
+
19
+ def _get_assets(self) -> Iterator[dict]:
20
+ """
21
+ Extracts reports and data sources user by user.
22
+ """
23
+ users = self.admin_sdk_client.list_users()
24
+
25
+ for user in users:
26
+ email = user[USER_EMAIL_FIELD]
27
+ yield from self.looker_studio_client.fetch_user_assets(email)
28
+
29
+ def fetch(self, asset: LookerStudioAsset) -> Iterator[dict]:
30
+ if asset == LookerStudioAsset.VIEW_ACTIVITY:
31
+ yield from self.admin_sdk_client.list_view_events()
32
+
33
+ elif asset == LookerStudioAsset.ASSETS:
34
+ yield from self._get_assets()
35
+
36
+ else:
37
+ raise ValueError(f"The asset {asset}, is not supported")
@@ -0,0 +1,20 @@
1
+ from pydantic import BaseModel, SecretStr, field_serializer
2
+
3
+
4
+ class LookerStudioCredentials(BaseModel):
5
+ admin_email: str
6
+ auth_provider_x509_cert_url: str
7
+ auth_uri: str
8
+ client_email: str
9
+ client_id: str
10
+ client_x509_cert_url: str
11
+ private_key: SecretStr
12
+ private_key_id: str
13
+ project_id: str
14
+ token_uri: str
15
+ type: str
16
+
17
+ @field_serializer("private_key")
18
+ def dump_secret(self, pk):
19
+ """When using model_dump, show private_key value"""
20
+ return pk.get_secret_value()
@@ -0,0 +1,18 @@
1
+ class LookerStudioAPIEndpoint:
2
+ BASE_PATH = "https://datastudio.googleapis.com"
3
+
4
+ @classmethod
5
+ def search(cls) -> str:
6
+ """
7
+ Search a user's assets.
8
+ See https://developers.google.com/looker-studio/integrate/api/reference/assets/search
9
+ """
10
+ return f"{cls.BASE_PATH}/v1/assets:search"
11
+
12
+ @classmethod
13
+ def permissions(cls, asset_name: str) -> str:
14
+ """
15
+ Get the permissions of an asset. The user must be the owner of the asset.
16
+ See https://developers.google.com/looker-studio/integrate/api/reference/permissions/get
17
+ """
18
+ return f"{cls.BASE_PATH}/v1/assets/{asset_name}/permissions"
@@ -0,0 +1,8 @@
1
+ from enum import Enum
2
+
3
+
4
+ class LookerStudioAssetType(Enum):
5
+ DATA_SOURCE = "DATA_SOURCE"
6
+ EXPLORER = "EXPLORER"
7
+ REPORT = "REPORT"
8
+ WORKSPACE = "WORKSPACE"
@@ -0,0 +1,102 @@
1
+ from functools import partial
2
+ from typing import Iterator, Optional
3
+
4
+ from google.auth.transport.requests import Request
5
+ from google.oauth2.service_account import Credentials
6
+
7
+ from ....utils import (
8
+ APIClient,
9
+ BearerAuth,
10
+ fetch_all_pages,
11
+ )
12
+ from .credentials import LookerStudioCredentials
13
+ from .endpoints import LookerStudioAPIEndpoint
14
+ from .enums import LookerStudioAssetType
15
+ from .pagination import LookerStudioPagination
16
+ from .scopes import SCOPES
17
+
18
+
19
+ class LookerStudioAPIAuth(BearerAuth):
20
+ def __init__(
21
+ self,
22
+ credentials: LookerStudioCredentials,
23
+ subject: Optional[str] = None,
24
+ ):
25
+ """
26
+ Instantiates the service account credentials.
27
+ If a `subject` email is passed, the service account will impersonate
28
+ that user and make requests on that user's behalf.
29
+ """
30
+ self._credentials = Credentials.from_service_account_info(
31
+ credentials.model_dump(), scopes=SCOPES
32
+ )
33
+ if subject:
34
+ self._credentials = self._credentials.with_subject(subject)
35
+
36
+ def fetch_token(self):
37
+ self._credentials.refresh(Request())
38
+ return self._credentials.token
39
+
40
+
41
+ class LookerStudioAPIClient(APIClient):
42
+ def __init__(self, credentials: LookerStudioCredentials):
43
+ auth = LookerStudioAPIAuth(credentials=credentials)
44
+ super().__init__(auth=auth)
45
+
46
+ self._credentials = credentials
47
+
48
+ def _is_private_asset(self, asset_name: str) -> bool:
49
+ """
50
+ Returns True if the asset is not viewable by anyone other than the owner.
51
+
52
+ The permissions dict contains `Role: Member[]` key-value pairs and has
53
+ at least one key-value pair to define the asset's unique OWNER.
54
+ If another key is present, it means the asset was shared with
55
+ another person or group.
56
+
57
+ See also https://developers.google.com/looker-studio/integrate/api/reference/types#Permissions
58
+ """
59
+ data = self._get(LookerStudioAPIEndpoint.permissions(asset_name))
60
+ permissions = data["permissions"]
61
+ return len(permissions.keys()) == 1
62
+
63
+ def _user_assets(
64
+ self, asset_type: LookerStudioAssetType, user_email: str
65
+ ) -> Iterator[dict]:
66
+ """
67
+ Yields all assets of the given type, owned by the given user and visible
68
+ by other members.
69
+ """
70
+ request = partial(
71
+ self._get,
72
+ LookerStudioAPIEndpoint.search(),
73
+ params={"assetTypes": [asset_type.value]},
74
+ )
75
+ assets = fetch_all_pages(request, LookerStudioPagination)
76
+
77
+ for asset in assets:
78
+ asset_name = asset["name"]
79
+ owner = asset["owner"]
80
+ if owner == user_email and not self._is_private_asset(asset_name):
81
+ yield asset
82
+
83
+ def _impersonate_user(self, user_email: str):
84
+ self._auth = LookerStudioAPIAuth(
85
+ credentials=self._credentials, subject=user_email
86
+ )
87
+
88
+ def fetch_user_assets(self, user_email: str) -> Iterator[dict]:
89
+ """Yields assets (reports and data sources) shared by the given user."""
90
+ self._impersonate_user(user_email)
91
+
92
+ reports = self._user_assets(
93
+ asset_type=LookerStudioAssetType.REPORT,
94
+ user_email=user_email,
95
+ )
96
+ data_sources = self._user_assets(
97
+ asset_type=LookerStudioAssetType.DATA_SOURCE,
98
+ user_email=user_email,
99
+ )
100
+
101
+ yield from reports
102
+ yield from data_sources
@@ -0,0 +1,31 @@
1
+ from typing import Optional
2
+
3
+ from pydantic import AliasChoices, ConfigDict, Field
4
+ from pydantic.alias_generators import to_camel
5
+
6
+ from ....utils import PaginationModel
7
+
8
+ NEXT_PAGE_KEY = "pageToken"
9
+
10
+
11
+ class LookerStudioPagination(PaginationModel):
12
+ items: list = Field(
13
+ default_factory=list,
14
+ validation_alias=AliasChoices("items", "users", "assets"),
15
+ )
16
+ next_page_token: Optional[str] = None
17
+
18
+ model_config = ConfigDict(
19
+ alias_generator=to_camel,
20
+ populate_by_name=True,
21
+ from_attributes=True,
22
+ )
23
+
24
+ def is_last(self) -> bool:
25
+ return self.next_page_token is None
26
+
27
+ def next_page_payload(self) -> dict:
28
+ return {NEXT_PAGE_KEY: self.next_page_token}
29
+
30
+ def page_results(self) -> list:
31
+ return self.items
@@ -0,0 +1,6 @@
1
+ SCOPES = (
2
+ "https://www.googleapis.com/auth/datastudio",
3
+ "https://www.googleapis.com/auth/userinfo.profile",
4
+ "https://www.googleapis.com/auth/admin.reports.audit.readonly",
5
+ "https://www.googleapis.com/auth/admin.directory.user.readonly",
6
+ )
@@ -1,9 +1,11 @@
1
1
  from collections.abc import Iterator
2
+ from concurrent.futures import ThreadPoolExecutor
2
3
  from functools import partial
3
4
  from http import HTTPStatus
4
5
  from typing import Callable, Optional
5
6
 
6
7
  import requests
8
+ from pydantic import BaseModel
7
9
 
8
10
  from ....utils import (
9
11
  APIClient,
@@ -12,6 +14,7 @@ from ....utils import (
12
14
  build_url,
13
15
  fetch_all_pages,
14
16
  handle_response,
17
+ retry,
15
18
  )
16
19
  from ..assets import SigmaAsset
17
20
  from .credentials import SigmaCredentials
@@ -29,7 +32,7 @@ _DATA_ELEMENTS: tuple[str, ...] = (
29
32
  )
30
33
 
31
34
  _AUTH_TIMEOUT_S = 60
32
- _SIGMA_TIMEOUT = 120
35
+ _SIGMA_TIMEOUT_S = 300
33
36
 
34
37
  _SIGMA_HEADERS = {
35
38
  "Content-Type": _CONTENT_TYPE,
@@ -47,6 +50,23 @@ SIGMA_SAFE_MODE = RequestSafeMode(
47
50
  max_errors=_VOLUME_IGNORED,
48
51
  status_codes=_IGNORED_ERROR_CODES,
49
52
  )
53
+ _THREADS_LINEAGE = 10 # empirically found; hit the rate limit with 20 workers
54
+ _RETRY_NUMBER = 1
55
+ _RETRY_BASE_MS = 60_000
56
+
57
+
58
+ class LineageContext(BaseModel):
59
+ """all info needed to build the endpoint for lineage retrieval"""
60
+
61
+ workbook_id: str
62
+ element_id: str
63
+
64
+
65
+ class Lineage(BaseModel):
66
+ """holds response from lineage API and context used to retrieve it"""
67
+
68
+ lineage: dict
69
+ context: LineageContext
50
70
 
51
71
 
52
72
  class SigmaBearerAuth(BearerAuth):
@@ -77,7 +97,7 @@ class SigmaClient(APIClient):
77
97
  host=credentials.host,
78
98
  auth=auth,
79
99
  headers=_SIGMA_HEADERS,
80
- timeout=_SIGMA_TIMEOUT,
100
+ timeout=_SIGMA_TIMEOUT_S,
81
101
  safe_mode=safe_mode or SIGMA_SAFE_MODE,
82
102
  )
83
103
 
@@ -133,17 +153,51 @@ class SigmaClient(APIClient):
133
153
  page=page, workbook_id=workbook_id
134
154
  )
135
155
 
136
- def _get_all_lineages(self, elements: list[dict]) -> Iterator[dict]:
156
+ @retry(
157
+ (ConnectionError,),
158
+ max_retries=_RETRY_NUMBER,
159
+ base_ms=_RETRY_BASE_MS,
160
+ log_exc_info=True,
161
+ )
162
+ def _get_lineage(self, lineage_context: LineageContext) -> Lineage:
163
+ """
164
+ return the lineage from API and other ids needed to characterize
165
+ lineage in castor
166
+ """
167
+ workbook_id = lineage_context.workbook_id
168
+ element_id = lineage_context.element_id
169
+ endpoint = SigmaEndpointFactory.lineage(workbook_id, element_id)
170
+ return Lineage(lineage=self._get(endpoint), context=lineage_context)
171
+
172
+ @staticmethod
173
+ def _lineage_context(elements: list[dict]) -> list[LineageContext]:
174
+ """
175
+ Helper function to prepare context for lineage retrieval.
176
+ Elements without associated columns are skipped.
177
+ """
178
+ contexts: list[LineageContext] = []
137
179
  for element in elements:
138
- workbook_id = element["workbook_id"]
139
- element_id = element["elementId"]
140
- lineage = self._get(
141
- endpoint=SigmaEndpointFactory.lineage(workbook_id, element_id)
180
+ if element.get("columns") is None:
181
+ continue
182
+
183
+ context = LineageContext(
184
+ workbook_id=element["workbook_id"],
185
+ element_id=element["elementId"],
142
186
  )
187
+ contexts.append(context)
188
+ return contexts
189
+
190
+ def _get_all_lineages(self, elements: list[dict]) -> Iterator[dict]:
191
+ lineage_context = self._lineage_context(elements)
192
+
193
+ with ThreadPoolExecutor(max_workers=_THREADS_LINEAGE) as executor:
194
+ results = executor.map(self._get_lineage, lineage_context)
195
+
196
+ for lineage in results:
143
197
  yield {
144
- **lineage,
145
- "workbook_id": workbook_id,
146
- "element_id": element_id,
198
+ **lineage.lineage,
199
+ "workbook_id": lineage.context.workbook_id,
200
+ "element_id": lineage.context.element_id,
147
201
  }
148
202
 
149
203
  def _get_all_queries(self, workbooks: list[dict]) -> Iterator[dict]:
@@ -4,6 +4,8 @@ from ...types import ExternalAsset
4
4
  class ThoughtspotAsset(ExternalAsset):
5
5
  """Thoughtspot assets"""
6
6
 
7
+ ANSWERS = "answers"
8
+ ANSWER_USAGES = "answer_usages"
7
9
  LIVEBOARDS = "liveboards"
10
+ LIVEBOARD_USAGES = "liveboard_usages"
8
11
  LOGICAL_TABLES = "logical_tables"
9
- USAGES = "usages"