castor-extractor 0.18.7__tar.gz → 0.19.0__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 (398) hide show
  1. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/CHANGELOG.md +40 -1
  2. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/PKG-INFO +4 -4
  3. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/commands/extract_looker.py +3 -3
  4. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/commands/extract_metabase_api.py +1 -1
  5. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/commands/extract_metabase_db.py +1 -1
  6. castor_extractor-0.19.0/castor_extractor/commands/extract_notion.py +16 -0
  7. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/commands/file_check.py +5 -2
  8. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/commands/upload.py +5 -3
  9. castor_extractor-0.19.0/castor_extractor/knowledge/notion/__init__.py +3 -0
  10. castor_extractor-0.19.0/castor_extractor/knowledge/notion/assets.py +9 -0
  11. castor_extractor-0.19.0/castor_extractor/knowledge/notion/client/__init__.py +2 -0
  12. castor_extractor-0.19.0/castor_extractor/knowledge/notion/client/client.py +145 -0
  13. castor_extractor-0.19.0/castor_extractor/knowledge/notion/client/client_test.py +67 -0
  14. castor_extractor-0.19.0/castor_extractor/knowledge/notion/client/constants.py +3 -0
  15. castor_extractor-0.19.0/castor_extractor/knowledge/notion/client/credentials.py +16 -0
  16. castor_extractor-0.19.0/castor_extractor/knowledge/notion/client/endpoints.py +18 -0
  17. castor_extractor-0.19.0/castor_extractor/knowledge/notion/client/pagination.py +16 -0
  18. castor_extractor-0.19.0/castor_extractor/knowledge/notion/extract.py +59 -0
  19. castor_extractor-0.19.0/castor_extractor/quality/soda/__init__.py +2 -0
  20. castor_extractor-0.19.0/castor_extractor/quality/soda/assets.py +8 -0
  21. castor_extractor-0.19.0/castor_extractor/quality/soda/client/__init__.py +1 -0
  22. castor_extractor-0.19.0/castor_extractor/quality/soda/client/client.py +99 -0
  23. castor_extractor-0.19.0/castor_extractor/quality/soda/client/credentials.py +28 -0
  24. castor_extractor-0.19.0/castor_extractor/quality/soda/client/endpoints.py +13 -0
  25. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/types.py +1 -3
  26. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/uploader/upload.py +0 -1
  27. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/__init__.py +2 -0
  28. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/argument_parser_test.py +0 -1
  29. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/client/api.py +29 -11
  30. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/client/api_test.py +9 -1
  31. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/object_test.py +1 -1
  32. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/pager/pager.py +1 -1
  33. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/pager/pager_on_id.py +11 -6
  34. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/safe_request.py +5 -3
  35. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/safe_request_test.py +1 -3
  36. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/string_test.py +1 -1
  37. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/time.py +11 -0
  38. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/domo/client/client.py +2 -3
  39. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/looker/api/client.py +35 -0
  40. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/looker/api/extraction_parameters.py +2 -1
  41. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/looker/extract.py +2 -2
  42. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/metabase/assets.py +3 -1
  43. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/metabase/extract.py +20 -8
  44. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/mode/client/client.py +1 -1
  45. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/powerbi/client/constants.py +1 -1
  46. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/powerbi/client/rest.py +5 -15
  47. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/qlik/client/engine/client.py +36 -5
  48. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/qlik/client/engine/constants.py +1 -0
  49. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/qlik/client/engine/error.py +18 -1
  50. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/salesforce_reporting/client/soql.py +3 -1
  51. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau/extract.py +40 -16
  52. castor_extractor-0.19.0/castor_extractor/visualization/tableau/tests/unit/rest_api/__init__.py +0 -0
  53. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau_revamp/client/client.py +2 -5
  54. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau_revamp/extract.py +3 -2
  55. castor_extractor-0.19.0/castor_extractor/warehouse/__init__.py +0 -0
  56. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/databricks/client.py +54 -35
  57. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/databricks/client_test.py +44 -31
  58. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/salesforce/format_test.py +0 -1
  59. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/pyproject.toml +62 -62
  60. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/Dockerfile +0 -0
  61. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/DockerfileUsage.md +0 -0
  62. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/LICENCE +0 -0
  63. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/README.md +0 -0
  64. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/__init__.py +0 -0
  65. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/commands/__init__.py +0 -0
  66. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/commands/extract_bigquery.py +0 -0
  67. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/commands/extract_databricks.py +0 -0
  68. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/commands/extract_domo.py +0 -0
  69. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/commands/extract_mode.py +0 -0
  70. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/commands/extract_mysql.py +0 -0
  71. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/commands/extract_postgres.py +0 -0
  72. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/commands/extract_powerbi.py +0 -0
  73. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/commands/extract_qlik.py +0 -0
  74. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/commands/extract_redshift.py +0 -0
  75. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/commands/extract_salesforce.py +0 -0
  76. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/commands/extract_salesforce_reporting.py +0 -0
  77. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/commands/extract_sigma.py +0 -0
  78. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/commands/extract_snowflake.py +0 -0
  79. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/commands/extract_sqlserver.py +0 -0
  80. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/commands/extract_tableau.py +0 -0
  81. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/file_checker/__init__.py +0 -0
  82. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/file_checker/column.py +0 -0
  83. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/file_checker/column_test.py +0 -0
  84. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/file_checker/constants.py +0 -0
  85. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/file_checker/enums.py +0 -0
  86. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/file_checker/file.py +0 -0
  87. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/file_checker/file_test.py +0 -0
  88. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/file_checker/file_test_users.csv +0 -0
  89. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/file_checker/file_test_users_valid.csv +0 -0
  90. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/file_checker/templates/__init__.py +0 -0
  91. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/file_checker/templates/generic_warehouse.py +0 -0
  92. {castor_extractor-0.18.7/castor_extractor/visualization → castor_extractor-0.19.0/castor_extractor/knowledge}/__init__.py +0 -0
  93. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/logger.py +0 -0
  94. {castor_extractor-0.18.7/castor_extractor/visualization/tableau/tests → castor_extractor-0.19.0/castor_extractor/quality}/__init__.py +0 -0
  95. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/uploader/__init__.py +0 -0
  96. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/uploader/constant.py +0 -0
  97. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/uploader/env.py +0 -0
  98. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/uploader/env_test.py +0 -0
  99. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/uploader/upload_test.py +0 -0
  100. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/uploader/utils.py +0 -0
  101. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/argument_parser.py +0 -0
  102. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/client/__init__.py +0 -0
  103. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/client/abstract.py +0 -0
  104. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/client/postgres.py +0 -0
  105. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/client/query.py +0 -0
  106. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/client/uri.py +0 -0
  107. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/client/uri_test.py +0 -0
  108. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/collection.py +0 -0
  109. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/collection_test.py +0 -0
  110. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/constants.py +0 -0
  111. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/dbt/__init__.py +0 -0
  112. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/dbt/assets.py +0 -0
  113. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/dbt/client.py +0 -0
  114. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/dbt/client_test.py +0 -0
  115. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/dbt/credentials.py +0 -0
  116. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/deprecate.py +0 -0
  117. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/env.py +0 -0
  118. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/files.py +0 -0
  119. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/files_test.py +0 -0
  120. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/formatter.py +0 -0
  121. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/formatter_test.csv +0 -0
  122. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/formatter_test.json +0 -0
  123. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/formatter_test.py +0 -0
  124. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/json_stream_write.py +0 -0
  125. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/load.py +0 -0
  126. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/object.py +0 -0
  127. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/pager/__init__.py +0 -0
  128. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/pager/pager_on_id_test.py +0 -0
  129. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/pager/pager_on_token.py +0 -0
  130. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/pager/pager_on_token_test.py +0 -0
  131. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/pager/pager_test.py +0 -0
  132. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/retry.py +0 -0
  133. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/retry_test.py +0 -0
  134. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/safe.py +0 -0
  135. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/safe_test.py +0 -0
  136. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/salesforce/__init__.py +0 -0
  137. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/salesforce/client.py +0 -0
  138. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/salesforce/client_test.py +0 -0
  139. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/salesforce/constants.py +0 -0
  140. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/salesforce/credentials.py +0 -0
  141. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/salesforce/credentials_test.py +0 -0
  142. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/store.py +0 -0
  143. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/string.py +0 -0
  144. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/time_test.py +0 -0
  145. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/type.py +0 -0
  146. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/validation.py +0 -0
  147. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/validation_test.py +0 -0
  148. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/utils/write.py +0 -0
  149. {castor_extractor-0.18.7/castor_extractor/visualization/tableau/tests/unit → castor_extractor-0.19.0/castor_extractor/visualization}/__init__.py +0 -0
  150. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/domo/__init__.py +0 -0
  151. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/domo/assets.py +0 -0
  152. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/domo/client/__init__.py +0 -0
  153. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/domo/client/credentials.py +0 -0
  154. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/domo/client/endpoints.py +0 -0
  155. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/domo/client/pagination.py +0 -0
  156. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/domo/client/pagination_test.py +0 -0
  157. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/domo/extract.py +0 -0
  158. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/looker/__init__.py +0 -0
  159. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/looker/api/__init__.py +0 -0
  160. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/looker/api/client_test.py +0 -0
  161. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/looker/api/constants.py +0 -0
  162. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/looker/api/credentials.py +0 -0
  163. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/looker/api/sdk.py +0 -0
  164. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/looker/api/sdk_test.py +0 -0
  165. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/looker/api/utils.py +0 -0
  166. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/looker/assets.py +0 -0
  167. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/looker/constant.py +0 -0
  168. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/looker/constants.py +0 -0
  169. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/looker/fields.py +0 -0
  170. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/looker/fields_test.py +0 -0
  171. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/looker/multithreading.py +0 -0
  172. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/metabase/__init__.py +0 -0
  173. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/metabase/client/__init__.py +0 -0
  174. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/metabase/client/api/__init__.py +0 -0
  175. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/metabase/client/api/client.py +0 -0
  176. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/metabase/client/api/client_test.py +0 -0
  177. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/metabase/client/api/credentials.py +0 -0
  178. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/metabase/client/db/__init__.py +0 -0
  179. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/metabase/client/db/client.py +0 -0
  180. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/metabase/client/db/credentials.py +0 -0
  181. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/metabase/client/db/queries/.sqlfluff +0 -0
  182. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/metabase/client/db/queries/base_url.sql +0 -0
  183. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/metabase/client/db/queries/card.sql +0 -0
  184. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/metabase/client/db/queries/collection.sql +0 -0
  185. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/metabase/client/db/queries/dashboard.sql +0 -0
  186. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/metabase/client/db/queries/dashboard_cards.sql +0 -0
  187. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/metabase/client/db/queries/database.sql +0 -0
  188. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/metabase/client/db/queries/table.sql +0 -0
  189. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/metabase/client/db/queries/user.sql +0 -0
  190. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/metabase/client/decryption.py +0 -0
  191. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/metabase/client/decryption_test.py +0 -0
  192. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/metabase/client/shared.py +0 -0
  193. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/metabase/errors.py +0 -0
  194. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/metabase/types.py +0 -0
  195. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/mode/__init__.py +0 -0
  196. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/mode/assets.py +0 -0
  197. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/mode/client/__init__.py +0 -0
  198. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/mode/client/client_test.json +0 -0
  199. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/mode/client/client_test.py +0 -0
  200. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/mode/client/constants.py +0 -0
  201. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/mode/client/credentials.py +0 -0
  202. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/mode/errors.py +0 -0
  203. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/mode/extract.py +0 -0
  204. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/powerbi/__init__.py +0 -0
  205. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/powerbi/assets.py +0 -0
  206. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/powerbi/client/__init__.py +0 -0
  207. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/powerbi/client/credentials.py +0 -0
  208. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/powerbi/client/credentials_test.py +0 -0
  209. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/powerbi/client/rest_test.py +0 -0
  210. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/powerbi/client/utils.py +0 -0
  211. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/powerbi/client/utils_test.py +0 -0
  212. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/powerbi/extract.py +0 -0
  213. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/qlik/__init__.py +0 -0
  214. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/qlik/assets.py +0 -0
  215. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/qlik/client/__init__.py +0 -0
  216. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/qlik/client/constants.py +0 -0
  217. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/qlik/client/engine/__init__.py +0 -0
  218. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/qlik/client/engine/credentials.py +0 -0
  219. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/qlik/client/engine/error_test.py +0 -0
  220. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/qlik/client/engine/json_rpc.py +0 -0
  221. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/qlik/client/engine/json_rpc_test.py +0 -0
  222. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/qlik/client/engine/websocket.py +0 -0
  223. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/qlik/client/master.py +0 -0
  224. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/qlik/client/rest.py +0 -0
  225. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/qlik/client/rest_test.py +0 -0
  226. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/qlik/extract.py +0 -0
  227. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/salesforce_reporting/__init__.py +0 -0
  228. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/salesforce_reporting/assets.py +0 -0
  229. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/salesforce_reporting/client/__init__.py +0 -0
  230. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/salesforce_reporting/client/rest.py +0 -0
  231. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/salesforce_reporting/extract.py +0 -0
  232. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/sigma/__init__.py +0 -0
  233. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/sigma/assets.py +0 -0
  234. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/sigma/client/__init__.py +0 -0
  235. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/sigma/client/client.py +0 -0
  236. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/sigma/client/client_test.py +0 -0
  237. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/sigma/client/credentials.py +0 -0
  238. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/sigma/client/endpoints.py +0 -0
  239. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/sigma/client/pagination.py +0 -0
  240. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/sigma/extract.py +0 -0
  241. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau/__init__.py +0 -0
  242. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau/assets.py +0 -0
  243. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau/client/__init__.py +0 -0
  244. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau/client/client.py +0 -0
  245. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau/client/client_utils.py +0 -0
  246. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau/client/credentials.py +0 -0
  247. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau/client/project.py +0 -0
  248. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau/client/safe_mode.py +0 -0
  249. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau/constants.py +0 -0
  250. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau/errors.py +0 -0
  251. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau/gql_fields.py +0 -0
  252. {castor_extractor-0.18.7/castor_extractor/visualization/tableau/tests/unit/graphql → castor_extractor-0.19.0/castor_extractor/visualization/tableau/tests}/__init__.py +0 -0
  253. {castor_extractor-0.18.7/castor_extractor/visualization/tableau/tests/unit/rest_api → castor_extractor-0.19.0/castor_extractor/visualization/tableau/tests/unit}/__init__.py +0 -0
  254. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau/tests/unit/assets/graphql/metadata/metadata_1_get.json +0 -0
  255. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau/tests/unit/assets/graphql/metadata/metadata_2_get.json +0 -0
  256. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau/tests/unit/assets/rest_api/auth.xml +0 -0
  257. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau/tests/unit/assets/rest_api/project_get.xml +0 -0
  258. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau/tests/unit/assets/rest_api/user_get.xml +0 -0
  259. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau/tests/unit/assets/rest_api/view_get_usage.xml +0 -0
  260. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau/tests/unit/assets/rest_api/workbook_get.xml +0 -0
  261. {castor_extractor-0.18.7/castor_extractor/warehouse → castor_extractor-0.19.0/castor_extractor/visualization/tableau/tests/unit/graphql}/__init__.py +0 -0
  262. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau/tests/unit/graphql/paginated_object_test.py +0 -0
  263. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau/tests/unit/rest_api/auth_test.py +0 -0
  264. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau/tests/unit/rest_api/credentials_test.py +0 -0
  265. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau/tests/unit/rest_api/projects_test.py +0 -0
  266. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau/tests/unit/rest_api/usages_test.py +0 -0
  267. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau/tests/unit/rest_api/users_test.py +0 -0
  268. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau/tests/unit/rest_api/workbooks_test.py +0 -0
  269. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau/tests/unit/utils/__init__.py +0 -0
  270. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau/tests/unit/utils/env_key.py +0 -0
  271. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau/tsc_fields.py +0 -0
  272. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau/types.py +0 -0
  273. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau/usage.py +0 -0
  274. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau_revamp/__init__.py +0 -0
  275. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau_revamp/assets.py +0 -0
  276. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau_revamp/client/__init__.py +0 -0
  277. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau_revamp/client/credentials.py +0 -0
  278. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau_revamp/client/errors.py +0 -0
  279. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau_revamp/client/gql_queries.py +0 -0
  280. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau_revamp/client/tsc_fields.py +0 -0
  281. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/visualization/tableau_revamp/constants.py +0 -0
  282. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/abstract/__init__.py +0 -0
  283. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/abstract/asset.py +0 -0
  284. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/abstract/asset_test.py +0 -0
  285. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/abstract/extract.py +0 -0
  286. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/abstract/query.py +0 -0
  287. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/abstract/time_filter.py +0 -0
  288. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/abstract/time_filter_test.py +0 -0
  289. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/bigquery/__init__.py +0 -0
  290. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/bigquery/client.py +0 -0
  291. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/bigquery/client_test.py +0 -0
  292. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/bigquery/credentials.py +0 -0
  293. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/bigquery/extract.py +0 -0
  294. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/bigquery/queries/.sqlfluff +0 -0
  295. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/bigquery/queries/column.sql +0 -0
  296. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/bigquery/queries/cte/sharded.sql +0 -0
  297. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/bigquery/queries/database.sql +0 -0
  298. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/bigquery/queries/query.sql +0 -0
  299. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/bigquery/queries/schema.sql +0 -0
  300. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/bigquery/queries/table.sql +0 -0
  301. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/bigquery/queries/table_with_tags.sql +0 -0
  302. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/bigquery/queries/user.sql +0 -0
  303. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/bigquery/queries/view_ddl.sql +0 -0
  304. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/bigquery/query.py +0 -0
  305. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/bigquery/types.py +0 -0
  306. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/databricks/__init__.py +0 -0
  307. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/databricks/credentials.py +0 -0
  308. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/databricks/extract.py +0 -0
  309. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/databricks/format.py +0 -0
  310. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/databricks/format_test.py +0 -0
  311. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/databricks/test_constants.py +0 -0
  312. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/databricks/types.py +0 -0
  313. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/databricks/utils.py +0 -0
  314. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/databricks/utils_test.py +0 -0
  315. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/mysql/__init__.py +0 -0
  316. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/mysql/client.py +0 -0
  317. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/mysql/client_test.py +0 -0
  318. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/mysql/extract.py +0 -0
  319. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/mysql/queries/.sqlfluff +0 -0
  320. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/mysql/queries/column.sql +0 -0
  321. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/mysql/queries/database.sql +0 -0
  322. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/mysql/queries/query.sql +0 -0
  323. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/mysql/queries/schema.sql +0 -0
  324. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/mysql/queries/table.sql +0 -0
  325. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/mysql/queries/user.sql +0 -0
  326. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/mysql/queries/view_ddl.sql +0 -0
  327. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/mysql/query.py +0 -0
  328. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/postgres/__init__.py +0 -0
  329. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/postgres/extract.py +0 -0
  330. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/postgres/queries/.sqlfluff +0 -0
  331. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/postgres/queries/column.sql +0 -0
  332. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/postgres/queries/database.sql +0 -0
  333. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/postgres/queries/group.sql +0 -0
  334. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/postgres/queries/schema.sql +0 -0
  335. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/postgres/queries/table.sql +0 -0
  336. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/postgres/queries/user.sql +0 -0
  337. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/postgres/query.py +0 -0
  338. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/redshift/__init__.py +0 -0
  339. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/redshift/client.py +0 -0
  340. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/redshift/client_test.py +0 -0
  341. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/redshift/extract.py +0 -0
  342. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/redshift/queries/.sqlfluff +0 -0
  343. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/redshift/queries/column.sql +0 -0
  344. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/redshift/queries/database.sql +0 -0
  345. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/redshift/queries/group.sql +0 -0
  346. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/redshift/queries/query.sql +0 -0
  347. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/redshift/queries/schema.sql +0 -0
  348. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/redshift/queries/table.sql +0 -0
  349. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/redshift/queries/table_freshness.sql +0 -0
  350. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/redshift/queries/user.sql +0 -0
  351. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/redshift/queries/view_ddl.sql +0 -0
  352. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/redshift/query.py +0 -0
  353. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/salesforce/__init__.py +0 -0
  354. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/salesforce/client.py +0 -0
  355. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/salesforce/constants.py +0 -0
  356. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/salesforce/extract.py +0 -0
  357. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/salesforce/format.py +0 -0
  358. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/salesforce/soql.py +0 -0
  359. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/snowflake/__init__.py +0 -0
  360. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/snowflake/client.py +0 -0
  361. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/snowflake/client_test.py +0 -0
  362. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/snowflake/credentials.py +0 -0
  363. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/snowflake/credentials_test.py +0 -0
  364. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/snowflake/extract.py +0 -0
  365. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/snowflake/queries/.sqlfluff +0 -0
  366. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/snowflake/queries/column.sql +0 -0
  367. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/snowflake/queries/column_lineage.sql +0 -0
  368. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/snowflake/queries/database.sql +0 -0
  369. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/snowflake/queries/function.sql +0 -0
  370. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/snowflake/queries/grant_to_role.sql +0 -0
  371. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/snowflake/queries/grant_to_user.sql +0 -0
  372. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/snowflake/queries/query.sql +0 -0
  373. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/snowflake/queries/role.sql +0 -0
  374. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/snowflake/queries/schema.sql +0 -0
  375. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/snowflake/queries/table.sql +0 -0
  376. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/snowflake/queries/user.sql +0 -0
  377. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/snowflake/queries/view_ddl.sql +0 -0
  378. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/snowflake/query.py +0 -0
  379. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/sqlserver/__init__.py +0 -0
  380. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/sqlserver/client.py +0 -0
  381. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/sqlserver/extract.py +0 -0
  382. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/sqlserver/queries/.sqlfluff +0 -0
  383. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/sqlserver/queries/column.sql +0 -0
  384. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/sqlserver/queries/database.sql +0 -0
  385. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/sqlserver/queries/schema.sql +0 -0
  386. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/sqlserver/queries/table.sql +0 -0
  387. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/sqlserver/queries/user.sql +0 -0
  388. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/sqlserver/query.py +0 -0
  389. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/synapse/__init__.py +0 -0
  390. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/synapse/extract.py +0 -0
  391. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/synapse/queries/.sqlfluff +0 -0
  392. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/synapse/queries/column.sql +0 -0
  393. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/synapse/queries/database.sql +0 -0
  394. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/synapse/queries/query.sql +0 -0
  395. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/synapse/queries/schema.sql +0 -0
  396. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/synapse/queries/table.sql +0 -0
  397. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/synapse/queries/user.sql +0 -0
  398. {castor_extractor-0.18.7 → castor_extractor-0.19.0}/castor_extractor/warehouse/synapse/queries/view_ddl.sql +0 -0
@@ -1,5 +1,44 @@
1
1
  # Changelog
2
2
 
3
+ ## 0.19.0 - 2024-08-21
4
+
5
+ * Breaking change Looker CLI:
6
+
7
+ `-u` and `--username` changed to `-c` `--client-id`
8
+
9
+ `-p` and `--password` changed to `-s` `--client-secret`
10
+
11
+ * Breaking change Metabase CLI:
12
+
13
+ `-u` and `--username` changed to `-u` `--user`
14
+
15
+
16
+ * Note: if you use environment variables you shouldn't be impacted
17
+
18
+ ## 0.18.13 - 2024-08-19
19
+
20
+ * Qlik: improve measures extraction
21
+
22
+ ## 0.18.12 - 2024-08-19
23
+
24
+ * Bumps: looker-sdk: 24, setuptools: 72
25
+
26
+ ## 0.18.11 - 2023-08-14
27
+
28
+ * Linting and formatting with Ruff
29
+
30
+ ## 0.18.10 - 2024-08-13
31
+
32
+ * Soda: Added Soda extractor (Castor-Managed integration only)
33
+
34
+ ## 0.18.9 - 2024-08-12
35
+
36
+ * Notion: Added Notion extractor
37
+
38
+ ## 0.18.8 - 2024-08-02
39
+
40
+ * Databricks: more reliable extraction of queries
41
+
3
42
  ## 0.18.7 - 2024-08-01
4
43
 
5
44
  * Salesforce: extract table descriptions
@@ -420,7 +459,7 @@
420
459
 
421
460
  ## 0.2.2 - 2023-03-13
422
461
 
423
- * Constrain `setuptools` explicitly added
462
+ * Constraint `setuptools` explicitly added
424
463
 
425
464
  ## 0.2.1 - 2023-02-23
426
465
 
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: castor-extractor
3
- Version: 0.18.7
3
+ Version: 0.19.0
4
4
  Summary: Extract your metadata assets.
5
5
  Home-page: https://www.castordoc.com/
6
6
  License: EULA
@@ -29,7 +29,7 @@ Provides-Extra: redshift
29
29
  Provides-Extra: snowflake
30
30
  Provides-Extra: sqlserver
31
31
  Provides-Extra: tableau
32
- Requires-Dist: cryptography (>=41.0.5) ; extra == "snowflake"
32
+ Requires-Dist: cryptography (>=42.0.0,<43.0.0) ; extra == "snowflake"
33
33
  Requires-Dist: databricks-sql-connector (>=3.2.0,<4.0.0) ; extra == "databricks" or extra == "all"
34
34
  Requires-Dist: google-api-core (>=2.1.1,<3.0.0)
35
35
  Requires-Dist: google-auth (>=2,<3)
@@ -37,7 +37,7 @@ Requires-Dist: google-cloud-core (>=2.1.0,<3.0.0)
37
37
  Requires-Dist: google-cloud-storage (>=2,<3)
38
38
  Requires-Dist: google-resumable-media (>=2.0.3,<3.0.0)
39
39
  Requires-Dist: googleapis-common-protos (>=1.53.0,<2.0.0)
40
- Requires-Dist: looker-sdk (>=23.0.0,<24.0.0) ; extra == "looker" or extra == "all"
40
+ Requires-Dist: looker-sdk (>=24.0.0,<25.0.0) ; extra == "looker" or extra == "all"
41
41
  Requires-Dist: msal (>=1.20.0,<2.0.0) ; extra == "powerbi" or extra == "all"
42
42
  Requires-Dist: numpy (<1.25) ; (python_version >= "3.8" and python_version < "3.9") and (extra == "bigquery" or extra == "databricks" or extra == "all")
43
43
  Requires-Dist: numpy (<2) ; extra == "bigquery" or extra == "databricks" or extra == "all"
@@ -52,7 +52,7 @@ Requires-Dist: pymssql (>=2.2.11,<3.0.0) ; extra == "sqlserver" or extra == "all
52
52
  Requires-Dist: pymysql[rsa] (>=1.1.0,<2.0.0) ; extra == "mysql" or extra == "all"
53
53
  Requires-Dist: python-dateutil (>=2.0.0,<=3.0.0)
54
54
  Requires-Dist: requests (>=2.0.0,<3.0.0)
55
- Requires-Dist: setuptools (>=70,<71)
55
+ Requires-Dist: setuptools (>=72,<73)
56
56
  Requires-Dist: snowflake-connector-python (>=3.4.0,<4.0.0) ; extra == "snowflake" or extra == "all"
57
57
  Requires-Dist: snowflake-sqlalchemy (!=1.2.5,<2.0.0) ; extra == "snowflake" or extra == "all"
58
58
  Requires-Dist: sqlalchemy (>=1.4,<1.5)
@@ -7,8 +7,8 @@ from castor_extractor.visualization import looker # type: ignore
7
7
  def main():
8
8
  parser = ArgumentParser()
9
9
  parser.add_argument("-b", "--base-url", help="Looker base url")
10
- parser.add_argument("-u", "--username", help="Looker client id")
11
- parser.add_argument("-p", "--password", help="Looker client secret")
10
+ parser.add_argument("-c", "--client-id", help="Looker client id")
11
+ parser.add_argument("-s", "--client-secret", help="Looker client secret")
12
12
  parser.add_argument("-o", "--output", help="Directory to write to")
13
13
  parser.add_argument("-t", "--timeout", type=int, help="Timeout in seconds")
14
14
  parser.add_argument(
@@ -18,7 +18,7 @@ def main():
18
18
  )
19
19
  parser.add_argument(
20
20
  "--safe-mode",
21
- "-s",
21
+ "-S",
22
22
  help="Looker safe mode",
23
23
  action="store_true",
24
24
  )
@@ -11,7 +11,7 @@ def main():
11
11
  parser = ArgumentParser()
12
12
 
13
13
  parser.add_argument("-b", "--base-url", help="Metabase base url")
14
- parser.add_argument("-u", "--username", help="Metabase username")
14
+ parser.add_argument("-u", "--user", help="Metabase username")
15
15
  parser.add_argument("-p", "--password", help="Metabase password")
16
16
 
17
17
  parser.add_argument("-o", "--output", help="Directory to write to")
@@ -19,7 +19,7 @@ def main():
19
19
  parser.add_argument("-P", "--port", help="TCP/IP port number")
20
20
  parser.add_argument("-d", "--database", help="Database name")
21
21
  parser.add_argument("-s", "--schema", help="Schema name")
22
- parser.add_argument("-u", "--username", help="Username")
22
+ parser.add_argument("-u", "--user", help="Username")
23
23
  parser.add_argument("-p", "--password", help="Password")
24
24
  parser.add_argument(
25
25
  "-k",
@@ -0,0 +1,16 @@
1
+ import logging
2
+ from argparse import ArgumentParser
3
+
4
+ from castor_extractor.knowledge import notion # type: ignore
5
+ from castor_extractor.utils import parse_filled_arguments # type: ignore
6
+
7
+ logging.basicConfig(level=logging.INFO, format="%(levelname)s - %(message)s")
8
+
9
+
10
+ def main():
11
+ parser = ArgumentParser()
12
+
13
+ parser.add_argument("-t", "--token", help="Notion token")
14
+ parser.add_argument("-o", "--output", help="Directory to write to")
15
+
16
+ notion.extract_all(**parse_filled_arguments(parser))
@@ -4,8 +4,11 @@ from argparse import ArgumentParser
4
4
  from typing import Set
5
5
 
6
6
  from castor_extractor import file_checker # type: ignore
7
- from castor_extractor.utils import LocalStorage # type: ignore
8
- from castor_extractor.utils import explode, search_files # type: ignore
7
+ from castor_extractor.utils import ( # type: ignore
8
+ LocalStorage, # type: ignore
9
+ explode,
10
+ search_files,
11
+ )
9
12
 
10
13
  logging.basicConfig(level=logging.INFO, format="%(levelname)s - %(message)s")
11
14
  logger = logging.getLogger(__name__)
@@ -1,9 +1,11 @@
1
1
  import argparse
2
2
  import logging
3
3
 
4
- from castor_extractor.uploader import FileType # type: ignore
5
- from castor_extractor.uploader import upload # type: ignore
6
- from castor_extractor.uploader import upload_manifest # type: ignore
4
+ from castor_extractor.uploader import ( # type: ignore
5
+ FileType,
6
+ upload,
7
+ upload_manifest,
8
+ )
7
9
 
8
10
  FILE_TYPES = {FileType.QUALITY, FileType.VIZ, FileType.WAREHOUSE}
9
11
 
@@ -0,0 +1,3 @@
1
+ from .assets import NotionAsset
2
+ from .client import NotionClient, NotionCredentials
3
+ from .extract import extract_all
@@ -0,0 +1,9 @@
1
+ from ...types import ExternalAsset
2
+
3
+
4
+ class NotionAsset(ExternalAsset):
5
+ """Notion assets"""
6
+
7
+ DATABASES = "databases"
8
+ PAGES = "pages"
9
+ USERS = "users"
@@ -0,0 +1,2 @@
1
+ from .client import NotionClient
2
+ from .credentials import NotionCredentials
@@ -0,0 +1,145 @@
1
+ import logging
2
+ from http import HTTPStatus
3
+ from typing import Dict, Iterator
4
+
5
+ from pydantic import ValidationError
6
+
7
+ from ....utils import RequestSafeMode, empty_iterator
8
+ from ....utils.client.api import APIClient, HttpMethod
9
+ from ..assets import NotionAsset
10
+ from .constants import (
11
+ CASTOR_NOTION_USER_AGENT,
12
+ NOTION_BASE_URL,
13
+ NOTION_TIMEOUT_MS,
14
+ )
15
+ from .credentials import NotionCredentials
16
+ from .endpoints import EndpointFactory
17
+ from .pagination import PaginatedResponse
18
+
19
+ logger = logging.getLogger("__name__")
20
+
21
+ NOTION_VERSION = "2021-08-16"
22
+
23
+ VOLUME_IGNORED = 10
24
+ IGNORED_ERROR_CODES = (HTTPStatus.BAD_GATEWAY,)
25
+ NOTION_SAFE_MODE = RequestSafeMode(
26
+ max_errors=VOLUME_IGNORED,
27
+ status_codes=IGNORED_ERROR_CODES,
28
+ )
29
+
30
+
31
+ def _search_filter(asset: str) -> Dict[str, Dict[str, str]]:
32
+ return {"filter": {"value": asset, "property": "object"}}
33
+
34
+
35
+ class NotionClient(APIClient):
36
+ """Client fetching data from Notion"""
37
+
38
+ def __init__(
39
+ self,
40
+ credentials: NotionCredentials,
41
+ timeout: int = NOTION_TIMEOUT_MS,
42
+ base_url: str = NOTION_BASE_URL,
43
+ user_agent: str = CASTOR_NOTION_USER_AGENT,
44
+ safe_mode: RequestSafeMode = NOTION_SAFE_MODE,
45
+ ) -> None:
46
+ base_headers = {
47
+ "Notion-Version": NOTION_VERSION,
48
+ "User-Agent": user_agent,
49
+ }
50
+
51
+ super().__init__(
52
+ host=base_url,
53
+ token=credentials.token,
54
+ headers=base_headers,
55
+ timeout=timeout,
56
+ safe_mode=safe_mode,
57
+ )
58
+
59
+ def _request(
60
+ self, method: HttpMethod, endpoint: str, params: dict
61
+ ) -> Iterator[dict]:
62
+ """
63
+ API call to Notion:
64
+ - If result is paginated, yield all pages
65
+ - If not, yield only the response payload
66
+ """
67
+ built_url = self.build_url(self._host, endpoint)
68
+ response_payload = self._call(
69
+ method=method,
70
+ url=built_url,
71
+ params=params if params and method == "GET" else None,
72
+ data=params if params and method == "POST" else None,
73
+ )
74
+ try:
75
+ paginated_response = PaginatedResponse(**response_payload)
76
+ yield from paginated_response.results
77
+
78
+ if not paginated_response.has_more:
79
+ return empty_iterator()
80
+
81
+ yield from self._request(
82
+ method=method,
83
+ endpoint=endpoint,
84
+ params=paginated_response.start_cursor,
85
+ )
86
+
87
+ except ValidationError:
88
+ yield response_payload
89
+
90
+ def _page_listing(self) -> Iterator[dict]:
91
+ return self._request(
92
+ method="POST",
93
+ endpoint=EndpointFactory.search(),
94
+ params=_search_filter("page"),
95
+ )
96
+
97
+ def _blocks(self, block_id: str) -> Iterator[dict]:
98
+ return self._request("GET", EndpointFactory.blocks(block_id), {})
99
+
100
+ def databases(self) -> Iterator[dict]:
101
+ return self._request(
102
+ method="POST",
103
+ endpoint=EndpointFactory.search(),
104
+ params=_search_filter("database"),
105
+ )
106
+
107
+ def recursive_blocks(self, block_id: str) -> Iterator[dict]:
108
+ """Fetch recursively all children blocks of a given block or page"""
109
+ blocks = self._blocks(block_id)
110
+ for block in blocks:
111
+ if block["has_children"] and block.get("type") != "child_page":
112
+ children = self.recursive_blocks(block["id"])
113
+ block["child_blocks"] = list(children)
114
+
115
+ yield block
116
+
117
+ def pages(self) -> Iterator[dict]:
118
+ """Fetch all pages with its whole content"""
119
+ pages = list(self._page_listing())
120
+ logger.info(f"Extracting {len(pages)} pages ...")
121
+ for page in pages:
122
+ if page.get("object") == "database":
123
+ # Notion Search API filter for page doesn't work
124
+ continue
125
+ content = list(self.recursive_blocks(page["id"]))
126
+ page["child_blocks"] = content
127
+ yield page
128
+
129
+ def users(self) -> Iterator[dict]:
130
+ """Fetch all users"""
131
+ return self._request("GET", EndpointFactory.users(), {})
132
+
133
+ def fetch(self, asset: NotionAsset) -> Iterator[dict]:
134
+ """Returns the needed metadata for the queried asset"""
135
+ if asset == NotionAsset.PAGES:
136
+ yield from self.pages()
137
+
138
+ elif asset == NotionAsset.DATABASES:
139
+ yield from self.databases()
140
+
141
+ elif asset == NotionAsset.USERS:
142
+ yield from self.users()
143
+
144
+ else:
145
+ raise ValueError(f"This asset {asset} is unknown")
@@ -0,0 +1,67 @@
1
+ from unittest.mock import patch
2
+
3
+ from .client import NotionClient
4
+ from .credentials import NotionCredentials
5
+
6
+ MOCK_PAGINATED_RESPONSE = {
7
+ "results": [{"result_id": 1}],
8
+ "next_cursor": "2",
9
+ "has_more": True,
10
+ }
11
+ MOCK_PAGINATED_RESPONSE_2 = {
12
+ "results": [{"result_id": 2}],
13
+ "next_cursor": None,
14
+ "has_more": False,
15
+ }
16
+ MOCK_RESPONSE = {"result_id": 3}
17
+
18
+
19
+ @patch.object(NotionClient, "_call")
20
+ def test_NotionClient__request(mock_call):
21
+ mock_call.side_effect = [
22
+ MOCK_PAGINATED_RESPONSE,
23
+ MOCK_PAGINATED_RESPONSE_2,
24
+ MOCK_RESPONSE,
25
+ ]
26
+
27
+ client = NotionClient(NotionCredentials(token="MockToken"))
28
+ response = list(client._request("GET", "fake_endpoint", {}))
29
+ assert response == [{"result_id": 1}, {"result_id": 2}]
30
+
31
+ response = list(client._request("GET", "fake_endpoint", {}))
32
+ assert response == [{"result_id": 3}]
33
+
34
+
35
+ MOCK_BLOCK = [{"object": "block", "id": "1", "has_children": True}]
36
+ MOCK_BLOCK_CHILDREN = [
37
+ {"object": "block", "id": "2", "has_children": False},
38
+ {"object": "block", "id": "3", "has_children": True, "type": "child_page"},
39
+ ]
40
+
41
+
42
+ @patch.object(NotionClient, "_blocks")
43
+ def test_NotionClient__recursive_blocks(mock_blocks):
44
+ mock_blocks.side_effect = [
45
+ MOCK_BLOCK,
46
+ MOCK_BLOCK_CHILDREN,
47
+ ]
48
+
49
+ client = NotionClient(NotionCredentials(token="MockToken"))
50
+ response = list(client.recursive_blocks("1"))
51
+
52
+ assert response == [
53
+ {
54
+ "object": "block",
55
+ "id": "1",
56
+ "has_children": True,
57
+ "child_blocks": [
58
+ {"object": "block", "id": "2", "has_children": False},
59
+ {
60
+ "object": "block",
61
+ "id": "3",
62
+ "has_children": True,
63
+ "type": "child_page",
64
+ },
65
+ ],
66
+ }
67
+ ]
@@ -0,0 +1,3 @@
1
+ NOTION_BASE_URL = "https://api.notion.com/v1/"
2
+ CASTOR_NOTION_USER_AGENT = "castor-extractor"
3
+ NOTION_TIMEOUT_MS = 300
@@ -0,0 +1,16 @@
1
+ from pydantic import Field
2
+ from pydantic_settings import BaseSettings, SettingsConfigDict
3
+
4
+ CASTOR_ENV_PREFIX = "CASTOR_NOTION_"
5
+
6
+
7
+ class NotionCredentials(BaseSettings):
8
+ """Class to handle Notion rest API permissions"""
9
+
10
+ model_config = SettingsConfigDict(
11
+ env_prefix=CASTOR_ENV_PREFIX,
12
+ extra="ignore",
13
+ populate_by_name=True,
14
+ )
15
+
16
+ token: str = Field(repr=False)
@@ -0,0 +1,18 @@
1
+ class EndpointFactory:
2
+ """Wrapper class around all endpoints we're using"""
3
+
4
+ BLOCKS = "blocks"
5
+ SEARCH = "search"
6
+ USERS = "users"
7
+
8
+ @classmethod
9
+ def users(cls) -> str:
10
+ return cls.USERS
11
+
12
+ @classmethod
13
+ def blocks(cls, block_id: str) -> str:
14
+ return f"{cls.BLOCKS}/{block_id}/children"
15
+
16
+ @classmethod
17
+ def search(cls) -> str:
18
+ return cls.SEARCH
@@ -0,0 +1,16 @@
1
+ from typing import Optional
2
+
3
+ from pydantic.dataclasses import dataclass
4
+
5
+
6
+ @dataclass
7
+ class PaginatedResponse:
8
+ """Class to handle paginated results"""
9
+
10
+ results: list
11
+ next_cursor: Optional[str]
12
+ has_more: bool
13
+
14
+ @property
15
+ def start_cursor(self) -> dict:
16
+ return {"start_cursor": self.next_cursor}
@@ -0,0 +1,59 @@
1
+ import logging
2
+ from typing import Iterable, Iterator, Tuple, Union
3
+
4
+ from ...utils import (
5
+ OUTPUT_DIR,
6
+ current_timestamp,
7
+ deep_serialize,
8
+ from_env,
9
+ get_output_filename,
10
+ write_json,
11
+ write_summary,
12
+ )
13
+ from .assets import NotionAsset
14
+ from .client import NotionClient, NotionCredentials
15
+
16
+ logger = logging.getLogger(__name__)
17
+
18
+
19
+ def iterate_all_data(
20
+ client: NotionClient,
21
+ ) -> Iterable[Tuple[NotionAsset, Union[list, Iterator, dict]]]:
22
+ """Iterate over the extracted data from Notion"""
23
+
24
+ logger.info("Extracting USERS from API")
25
+ users = list(deep_serialize(client.users()))
26
+ yield NotionAsset.USERS, users
27
+ logger.info(f"Extracted {len(users)} users from API")
28
+
29
+ logger.info("Extracting PAGES from API")
30
+ pages = list(deep_serialize(client.pages()))
31
+ yield NotionAsset.PAGES, pages
32
+ logger.info(f"Extracted {len(pages)} pages from API")
33
+
34
+ logger.info("Extracting DATABASES from API")
35
+ databases = list(deep_serialize(client.databases()))
36
+ yield NotionAsset.DATABASES, databases
37
+ logger.info(f"Extracted {len(databases)} databases from API")
38
+
39
+
40
+ def extract_all(**kwargs) -> None:
41
+ """
42
+ Extract data from Notion API
43
+ Store the output files locally under the given output_directory
44
+ """
45
+
46
+ output_directory = kwargs.get("output") or from_env(OUTPUT_DIR)
47
+
48
+ credentials = NotionCredentials(**kwargs)
49
+ client = NotionClient(credentials=credentials)
50
+
51
+ ts = current_timestamp()
52
+
53
+ for key, data in iterate_all_data(client):
54
+ filename = get_output_filename(key.name.lower(), output_directory, ts)
55
+ write_json(filename, data)
56
+
57
+ es = current_timestamp()
58
+
59
+ write_summary(output_directory, ts, duration_second=es - ts)
@@ -0,0 +1,2 @@
1
+ from .assets import SodaAsset
2
+ from .client import SodaClient, SodaCredentials
@@ -0,0 +1,8 @@
1
+ from ...types import ExternalAsset
2
+
3
+
4
+ class SodaAsset(ExternalAsset):
5
+ """Soda assets"""
6
+
7
+ CHECK_RESULTS = "check_results"
8
+ DATASETS = "datasets"
@@ -0,0 +1 @@
1
+ from .client import SodaClient, SodaCredentials
@@ -0,0 +1,99 @@
1
+ from time import sleep
2
+ from typing import Any, Iterator, Optional
3
+
4
+ import requests
5
+
6
+ from ....utils import format_date, yesterday
7
+ from ....utils.client.api import DEFAULT_TIMEOUT_S, HttpMethod
8
+ from ..assets import SodaAsset
9
+ from .credentials import SodaCredentials
10
+ from .endpoints import SodaEndpointFactory
11
+
12
+ _REPORTING_PAGE_SIZE = 400
13
+ _CLOUD_PAGE_SIZE = 100
14
+ _REQUESTS_PER_MINUTE = 10
15
+ _SECONDS_PER_MINUTE = 60
16
+ _RATE_LIMIT_S = (_SECONDS_PER_MINUTE // _REQUESTS_PER_MINUTE) + 1
17
+
18
+
19
+ class SodaClient:
20
+ def __init__(
21
+ self,
22
+ credentials: SodaCredentials,
23
+ ):
24
+ self._timeout = DEFAULT_TIMEOUT_S
25
+ self._reporting_headers = credentials.reporting_headers
26
+ self._auth = (credentials.api_key, credentials.secret)
27
+
28
+ def _call(
29
+ self,
30
+ url: str,
31
+ headers: dict,
32
+ method: HttpMethod = "GET",
33
+ *,
34
+ params: Optional[dict] = None,
35
+ data: Optional[dict] = None,
36
+ auth: Optional[tuple] = None,
37
+ ) -> Any:
38
+ response = requests.request(
39
+ method,
40
+ url,
41
+ headers=headers,
42
+ params=params,
43
+ json=data,
44
+ timeout=self._timeout,
45
+ auth=auth,
46
+ )
47
+ response.raise_for_status()
48
+
49
+ return response.json()
50
+
51
+ def _get_results_paginated(self, url: str, additional: dict) -> Iterator:
52
+ page_number = 1
53
+ next_page = True
54
+ while next_page:
55
+ json_data = {
56
+ **{"page": page_number, "size": _REPORTING_PAGE_SIZE},
57
+ **additional,
58
+ }
59
+ _check_results_page = self._call(
60
+ url=url,
61
+ method="POST",
62
+ data=json_data,
63
+ headers=self._reporting_headers,
64
+ )
65
+ yield from _check_results_page["data"]
66
+
67
+ next_page = len(_check_results_page["data"]) == _REPORTING_PAGE_SIZE
68
+ page_number += 1
69
+
70
+ def _datasets(self) -> Iterator[dict]:
71
+ url = SodaEndpointFactory.datasets()
72
+ next_page = True
73
+ page_number = 0
74
+ while next_page:
75
+ data = self._call(
76
+ url=url,
77
+ method="GET",
78
+ headers={},
79
+ params={"size": _CLOUD_PAGE_SIZE, "page": page_number},
80
+ auth=self._auth,
81
+ )
82
+ yield from data["content"]
83
+ next_page = not data["last"]
84
+ page_number += 1
85
+ sleep(_RATE_LIMIT_S)
86
+
87
+ def _check_results(self) -> Iterator:
88
+ url = SodaEndpointFactory.check_results()
89
+ _date = format_date(timestamp=yesterday())
90
+ return self._get_results_paginated(
91
+ url, additional={"from_datetime": _date}
92
+ )
93
+
94
+ def fetch(self, asset: SodaAsset) -> Iterator[dict]:
95
+ if asset == SodaAsset.DATASETS:
96
+ yield from self._datasets()
97
+ if asset == SodaAsset.CHECK_RESULTS:
98
+ yield from self._check_results()
99
+ raise ValueError(f"The asset {asset}, is not supported")
@@ -0,0 +1,28 @@
1
+ from typing import Dict
2
+
3
+ from pydantic import Field
4
+ from pydantic_settings import BaseSettings, SettingsConfigDict
5
+ from requests.auth import HTTPBasicAuth
6
+
7
+ SODA_ENV_PREFIX = "CASTOR_SODA_"
8
+
9
+
10
+ class SodaCredentials(BaseSettings):
11
+ """Class to handle Soda rest API permissions"""
12
+
13
+ model_config = SettingsConfigDict(
14
+ env_prefix=SODA_ENV_PREFIX,
15
+ extra="ignore",
16
+ populate_by_name=True,
17
+ )
18
+
19
+ api_key: str = Field(repr=False)
20
+ secret: str = Field(repr=False)
21
+
22
+ @property
23
+ def reporting_headers(self) -> Dict[str, str]:
24
+ return {
25
+ "Content-Type": "application/json",
26
+ "API_KEY_ID": self.api_key,
27
+ "API_KEY_SECRET": self.secret,
28
+ }