querysource 4.3.0__tar.gz → 4.4.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.
Files changed (584) hide show
  1. {querysource-4.3.0 → querysource-4.4.0}/PKG-INFO +1 -1
  2. {querysource-4.3.0 → querysource-4.4.0}/qs-parser/Cargo.lock +12 -12
  3. {querysource-4.3.0 → querysource-4.4.0}/qs-parser/Cargo.toml +1 -1
  4. {querysource-4.3.0 → querysource-4.4.0}/querysource/_version.py +3 -3
  5. {querysource-4.3.0 → querysource-4.4.0}/querysource/conf.py +41 -0
  6. {querysource-4.3.0 → querysource-4.4.0}/querysource/handlers/multi.py +37 -8
  7. querysource-4.4.0/querysource/handlers/scheduler.py +307 -0
  8. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/writers/json.py +16 -0
  9. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/__init__.py +105 -33
  10. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/sources/__init__.py +5 -0
  11. querysource-4.4.0/querysource/queries/multi/sources/executors.py +224 -0
  12. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/sources/query.py +87 -26
  13. {querysource-4.3.0 → querysource-4.4.0}/querysource/scheduler/scheduler.py +231 -107
  14. {querysource-4.3.0 → querysource-4.4.0}/querysource/types/dt/filters.py +23 -4
  15. {querysource-4.3.0 → querysource-4.4.0}/querysource/version.py +1 -1
  16. {querysource-4.3.0 → querysource-4.4.0}/querysource.egg-info/PKG-INFO +1 -1
  17. {querysource-4.3.0 → querysource-4.4.0}/querysource.egg-info/SOURCES.txt +7 -0
  18. querysource-4.4.0/tests/test_catalog_remote.py +33 -0
  19. {querysource-4.3.0 → querysource-4.4.0}/tests/test_column_filters.py +108 -0
  20. querysource-4.4.0/tests/test_local_executor.py +95 -0
  21. querysource-4.4.0/tests/test_multiqs_remote_dispatch.py +92 -0
  22. {querysource-4.3.0 → querysource-4.4.0}/tests/test_multiqs_sources_integration.py +13 -0
  23. querysource-4.4.0/tests/test_qworker_config.py +26 -0
  24. querysource-4.4.0/tests/test_remote_executor.py +107 -0
  25. querysource-4.4.0/tests/test_threadquery_executor.py +79 -0
  26. querysource-4.3.0/querysource/handlers/scheduler.py +0 -179
  27. {querysource-4.3.0 → querysource-4.4.0}/.agent/rules/async-programming-expert.md +0 -0
  28. {querysource-4.3.0 → querysource-4.4.0}/.agent/rules/code-reviewer.md +0 -0
  29. {querysource-4.3.0 → querysource-4.4.0}/.agent/rules/cython-development.md +0 -0
  30. {querysource-4.3.0 → querysource-4.4.0}/.agent/rules/prompt-expert.md +0 -0
  31. {querysource-4.3.0 → querysource-4.4.0}/.agent/rules/python-development.md +0 -0
  32. {querysource-4.3.0 → querysource-4.4.0}/.agent/rules/rust-development.md +0 -0
  33. {querysource-4.3.0 → querysource-4.4.0}/.agent/workflows/create-parrot-tool.md +0 -0
  34. {querysource-4.3.0 → querysource-4.4.0}/.agent/workflows/create-workflow.md +0 -0
  35. {querysource-4.3.0 → querysource-4.4.0}/.agent/workflows/debug-regressions-with-git-bisect-binary-search.md +0 -0
  36. {querysource-4.3.0 → querysource-4.4.0}/.agent/workflows/generate-tests.md +0 -0
  37. {querysource-4.3.0 → querysource-4.4.0}/.agent/workflows/git-new-feature.md +0 -0
  38. {querysource-4.3.0 → querysource-4.4.0}/.agent/workflows/parrot-mcp-server.md +0 -0
  39. {querysource-4.3.0 → querysource-4.4.0}/.agent/workflows/release_package.md +0 -0
  40. {querysource-4.3.0 → querysource-4.4.0}/.agent/workflows/start-flow.md +0 -0
  41. {querysource-4.3.0 → querysource-4.4.0}/.bumpversion.cfg +0 -0
  42. {querysource-4.3.0 → querysource-4.4.0}/.claude/agents/code-reviewer.md +0 -0
  43. {querysource-4.3.0 → querysource-4.4.0}/.claude/agents/sdd-autopilot.md +0 -0
  44. {querysource-4.3.0 → querysource-4.4.0}/.claude/agents/sdd-qa.md +0 -0
  45. {querysource-4.3.0 → querysource-4.4.0}/.claude/agents/sdd-research.md +0 -0
  46. {querysource-4.3.0 → querysource-4.4.0}/.claude/agents/sdd-worker.md +0 -0
  47. {querysource-4.3.0 → querysource-4.4.0}/.claude/commands/pr-review.md +0 -0
  48. {querysource-4.3.0 → querysource-4.4.0}/.claude/commands/sdd-brainstorm.md +0 -0
  49. {querysource-4.3.0 → querysource-4.4.0}/.claude/commands/sdd-codereview.md +0 -0
  50. {querysource-4.3.0 → querysource-4.4.0}/.claude/commands/sdd-done.md +0 -0
  51. {querysource-4.3.0 → querysource-4.4.0}/.claude/commands/sdd-fromjira.md +0 -0
  52. {querysource-4.3.0 → querysource-4.4.0}/.claude/commands/sdd-next.md +0 -0
  53. {querysource-4.3.0 → querysource-4.4.0}/.claude/commands/sdd-proposal.md +0 -0
  54. {querysource-4.3.0 → querysource-4.4.0}/.claude/commands/sdd-spec.md +0 -0
  55. {querysource-4.3.0 → querysource-4.4.0}/.claude/commands/sdd-start.md +0 -0
  56. {querysource-4.3.0 → querysource-4.4.0}/.claude/commands/sdd-status.md +0 -0
  57. {querysource-4.3.0 → querysource-4.4.0}/.claude/commands/sdd-task.md +0 -0
  58. {querysource-4.3.0 → querysource-4.4.0}/.claude/commands/sdd-tojira.md +0 -0
  59. {querysource-4.3.0 → querysource-4.4.0}/.claude/rules/aws-cost-optimization.md +0 -0
  60. {querysource-4.3.0 → querysource-4.4.0}/.claude/rules/code-reviewer.md +0 -0
  61. {querysource-4.3.0 → querysource-4.4.0}/.claude/rules/cython-development.md +0 -0
  62. {querysource-4.3.0 → querysource-4.4.0}/.claude/rules/python-development.md +0 -0
  63. {querysource-4.3.0 → querysource-4.4.0}/.claude/rules/rust-development.md +0 -0
  64. {querysource-4.3.0 → querysource-4.4.0}/.claude/rules/using-git-worktrees.md +0 -0
  65. {querysource-4.3.0 → querysource-4.4.0}/.claude/rules/worktree-pr-and-clean.md +0 -0
  66. {querysource-4.3.0 → querysource-4.4.0}/.claude/rules/worktree-start-feature.md +0 -0
  67. {querysource-4.3.0 → querysource-4.4.0}/.claude/rules/worktree-status.md +0 -0
  68. {querysource-4.3.0 → querysource-4.4.0}/.github/dependabot.yml +0 -0
  69. {querysource-4.3.0 → querysource-4.4.0}/.github/workflows/codeql-analysis.yml +0 -0
  70. {querysource-4.3.0 → querysource-4.4.0}/.github/workflows/release.yml +0 -0
  71. {querysource-4.3.0 → querysource-4.4.0}/.isort.cfg +0 -0
  72. {querysource-4.3.0 → querysource-4.4.0}/.jupyter/jupyter_notebook_config.py +0 -0
  73. {querysource-4.3.0 → querysource-4.4.0}/.pylintrc +0 -0
  74. {querysource-4.3.0 → querysource-4.4.0}/CHANGES.rst +0 -0
  75. {querysource-4.3.0 → querysource-4.4.0}/CODE_OF_CONDUCT.md +0 -0
  76. {querysource-4.3.0 → querysource-4.4.0}/CONTRIBUTING.md +0 -0
  77. {querysource-4.3.0 → querysource-4.4.0}/INSTALL +0 -0
  78. {querysource-4.3.0 → querysource-4.4.0}/JOIN_CONDITIONS_FEATURE.md +0 -0
  79. {querysource-4.3.0 → querysource-4.4.0}/JOIN_CONDITIONS_SUMMARY.md +0 -0
  80. {querysource-4.3.0 → querysource-4.4.0}/LICENSE +0 -0
  81. {querysource-4.3.0 → querysource-4.4.0}/MANIFEST.in +0 -0
  82. {querysource-4.3.0 → querysource-4.4.0}/Makefile +0 -0
  83. {querysource-4.3.0 → querysource-4.4.0}/README.md +0 -0
  84. {querysource-4.3.0 → querysource-4.4.0}/app.py +0 -0
  85. {querysource-4.3.0 → querysource-4.4.0}/bin/README.md +0 -0
  86. {querysource-4.3.0 → querysource-4.4.0}/conftest.py +0 -0
  87. {querysource-4.3.0 → querysource-4.4.0}/generated/AirtableSource.json +0 -0
  88. {querysource-4.3.0 → querysource-4.4.0}/generated/Concat.json +0 -0
  89. {querysource-4.3.0 → querysource-4.4.0}/generated/DWHDestination.json +0 -0
  90. {querysource-4.3.0 → querysource-4.4.0}/generated/DropCols.json +0 -0
  91. {querysource-4.3.0 → querysource-4.4.0}/generated/FileSource.json +0 -0
  92. {querysource-4.3.0 → querysource-4.4.0}/generated/Filter.json +0 -0
  93. {querysource-4.3.0 → querysource-4.4.0}/generated/FilterCols.json +0 -0
  94. {querysource-4.3.0 → querysource-4.4.0}/generated/Forecast.json +0 -0
  95. {querysource-4.3.0 → querysource-4.4.0}/generated/GoogleMaps.json +0 -0
  96. {querysource-4.3.0 → querysource-4.4.0}/generated/GroupBy.json +0 -0
  97. {querysource-4.3.0 → querysource-4.4.0}/generated/Info.json +0 -0
  98. {querysource-4.3.0 → querysource-4.4.0}/generated/Join.json +0 -0
  99. {querysource-4.3.0 → querysource-4.4.0}/generated/Map.json +0 -0
  100. {querysource-4.3.0 → querysource-4.4.0}/generated/Melt.json +0 -0
  101. {querysource-4.3.0 → querysource-4.4.0}/generated/Merge.json +0 -0
  102. {querysource-4.3.0 → querysource-4.4.0}/generated/PluckCols.json +0 -0
  103. {querysource-4.3.0 → querysource-4.4.0}/generated/Query.json +0 -0
  104. {querysource-4.3.0 → querysource-4.4.0}/generated/S3Source.json +0 -0
  105. {querysource-4.3.0 → querysource-4.4.0}/generated/SharepointSource.json +0 -0
  106. {querysource-4.3.0 → querysource-4.4.0}/generated/SmartSheetSource.json +0 -0
  107. {querysource-4.3.0 → querysource-4.4.0}/generated/Table.json +0 -0
  108. {querysource-4.3.0 → querysource-4.4.0}/generated/TableDestination.json +0 -0
  109. {querysource-4.3.0 → querysource-4.4.0}/generated/TableOutput.json +0 -0
  110. {querysource-4.3.0 → querysource-4.4.0}/generated/TableOutputAdapter.json +0 -0
  111. {querysource-4.3.0 → querysource-4.4.0}/generated/TableSource.json +0 -0
  112. {querysource-4.3.0 → querysource-4.4.0}/generated/ThreadQuery.json +0 -0
  113. {querysource-4.3.0 → querysource-4.4.0}/generated/ToS3.json +0 -0
  114. {querysource-4.3.0 → querysource-4.4.0}/generated/ToSharepoint.json +0 -0
  115. {querysource-4.3.0 → querysource-4.4.0}/generated/correlation.json +0 -0
  116. {querysource-4.3.0 → querysource-4.4.0}/generated/crosstab.json +0 -0
  117. {querysource-4.3.0 → querysource-4.4.0}/generated/pivot.json +0 -0
  118. {querysource-4.3.0 → querysource-4.4.0}/generated/tExplode.json +0 -0
  119. {querysource-4.3.0 → querysource-4.4.0}/generated/tOrder.json +0 -0
  120. {querysource-4.3.0 → querysource-4.4.0}/generated/tPandas.json +0 -0
  121. {querysource-4.3.0 → querysource-4.4.0}/gunicorn_config.py +0 -0
  122. {querysource-4.3.0 → querysource-4.4.0}/mypy.ini +0 -0
  123. {querysource-4.3.0 → querysource-4.4.0}/nav.py +0 -0
  124. {querysource-4.3.0 → querysource-4.4.0}/policies/datasources.yaml +0 -0
  125. {querysource-4.3.0 → querysource-4.4.0}/policies/defaults.yaml +0 -0
  126. {querysource-4.3.0 → querysource-4.4.0}/policies/drivers.yaml +0 -0
  127. {querysource-4.3.0 → querysource-4.4.0}/policies/raw_queries.yaml +0 -0
  128. {querysource-4.3.0 → querysource-4.4.0}/policies/slugs.yaml +0 -0
  129. {querysource-4.3.0 → querysource-4.4.0}/policies/superusers.yaml +0 -0
  130. {querysource-4.3.0 → querysource-4.4.0}/pyproject.toml +0 -0
  131. {querysource-4.3.0 → querysource-4.4.0}/pytest.ini +0 -0
  132. {querysource-4.3.0 → querysource-4.4.0}/qs-parser/pyproject.toml +0 -0
  133. {querysource-4.3.0 → querysource-4.4.0}/qs-parser/src/lib.rs +0 -0
  134. {querysource-4.3.0 → querysource-4.4.0}/qs-parser/src/parser.rs +0 -0
  135. {querysource-4.3.0 → querysource-4.4.0}/qs-parser/src/sql.rs +0 -0
  136. {querysource-4.3.0 → querysource-4.4.0}/querysource/__cli__.py +0 -0
  137. {querysource-4.3.0 → querysource-4.4.0}/querysource/__init__.py +0 -0
  138. {querysource-4.3.0 → querysource-4.4.0}/querysource/auth/__init__.py +0 -0
  139. {querysource-4.3.0 → querysource-4.4.0}/querysource/auth/_resource_types.py +0 -0
  140. {querysource-4.3.0 → querysource-4.4.0}/querysource/auth/credentials.py +0 -0
  141. {querysource-4.3.0 → querysource-4.4.0}/querysource/auth/pbac.py +0 -0
  142. {querysource-4.3.0 → querysource-4.4.0}/querysource/cache/__init__.py +0 -0
  143. {querysource-4.3.0 → querysource-4.4.0}/querysource/cache/backends/__init__.py +0 -0
  144. {querysource-4.3.0 → querysource-4.4.0}/querysource/cache/backends/abstract.py +0 -0
  145. {querysource-4.3.0 → querysource-4.4.0}/querysource/cache/backends/memcache.py +0 -0
  146. {querysource-4.3.0 → querysource-4.4.0}/querysource/cache/backends/redis.py +0 -0
  147. {querysource-4.3.0 → querysource-4.4.0}/querysource/cache/base.py +0 -0
  148. {querysource-4.3.0 → querysource-4.4.0}/querysource/cli/__init__.py +0 -0
  149. {querysource-4.3.0 → querysource-4.4.0}/querysource/cli/generate_docs.py +0 -0
  150. {querysource-4.3.0 → querysource-4.4.0}/querysource/connections.py +0 -0
  151. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/__init__.py +0 -0
  152. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/__init__.py +0 -0
  153. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/abstract.py +0 -0
  154. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/arangodb.py +0 -0
  155. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/athena.py +0 -0
  156. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/bigquery.py +0 -0
  157. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/cassandra.py +0 -0
  158. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/clickhouse.py +0 -0
  159. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/cockroachdb.py +0 -0
  160. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/couchbase.py +0 -0
  161. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/couchdb.py +0 -0
  162. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/countries.py +0 -0
  163. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/delta.py +0 -0
  164. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/documentdb.py +0 -0
  165. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/dynamodb.py +0 -0
  166. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/elastic.py +0 -0
  167. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/ga.py +0 -0
  168. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/gcalc.py +0 -0
  169. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/hazel.py +0 -0
  170. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/iceberg.py +0 -0
  171. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/influx.py +0 -0
  172. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/jdbc.py +0 -0
  173. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/jira.py +0 -0
  174. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/mariadb.py +0 -0
  175. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/memcached.py +0 -0
  176. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/mongo.py +0 -0
  177. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/mysql.py +0 -0
  178. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/odbc.py +0 -0
  179. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/openweather.py +0 -0
  180. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/oracle.py +0 -0
  181. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/pg.py +0 -0
  182. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/pg_admin.py +0 -0
  183. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/postgres.py +0 -0
  184. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/py.typed +0 -0
  185. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/qs.py +0 -0
  186. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/redis.py +0 -0
  187. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/rest.py +0 -0
  188. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/rethink.py +0 -0
  189. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/sa.py +0 -0
  190. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/salesforce.py +0 -0
  191. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/scylladb.py +0 -0
  192. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/sqlalchemy.py +0 -0
  193. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/sqlite.py +0 -0
  194. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/sqlserver.py +0 -0
  195. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/upc.py +0 -0
  196. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/drivers/zipcodeapi.py +0 -0
  197. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/handlers/__init__.py +0 -0
  198. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/handlers/datasource.py +0 -0
  199. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/handlers/utils.py +0 -0
  200. {querysource-4.3.0 → querysource-4.4.0}/querysource/datasources/models.py +0 -0
  201. {querysource-4.3.0 → querysource-4.4.0}/querysource/events/__init__.py +0 -0
  202. {querysource-4.3.0 → querysource-4.4.0}/querysource/exceptions.py +0 -0
  203. {querysource-4.3.0 → querysource-4.4.0}/querysource/handlers/__init__.py +0 -0
  204. {querysource-4.3.0 → querysource-4.4.0}/querysource/handlers/_pagination.py +0 -0
  205. {querysource-4.3.0 → querysource-4.4.0}/querysource/handlers/abstract.py +0 -0
  206. {querysource-4.3.0 → querysource-4.4.0}/querysource/handlers/components.py +0 -0
  207. {querysource-4.3.0 → querysource-4.4.0}/querysource/handlers/executor.py +0 -0
  208. {querysource-4.3.0 → querysource-4.4.0}/querysource/handlers/integrations/__init__.py +0 -0
  209. {querysource-4.3.0 → querysource-4.4.0}/querysource/handlers/integrations/airtable.py +0 -0
  210. {querysource-4.3.0 → querysource-4.4.0}/querysource/handlers/log.py +0 -0
  211. {querysource-4.3.0 → querysource-4.4.0}/querysource/handlers/manager.py +0 -0
  212. {querysource-4.3.0 → querysource-4.4.0}/querysource/handlers/outputs/__init__.py +0 -0
  213. {querysource-4.3.0 → querysource-4.4.0}/querysource/handlers/outputs/tableOutput/__init__.py +0 -0
  214. {querysource-4.3.0 → querysource-4.4.0}/querysource/handlers/outputs/tableOutput/postgres.py +0 -0
  215. {querysource-4.3.0 → querysource-4.4.0}/querysource/handlers/outputs/tableOutput/table.py +0 -0
  216. {querysource-4.3.0 → querysource-4.4.0}/querysource/handlers/service.py +0 -0
  217. {querysource-4.3.0 → querysource-4.4.0}/querysource/handlers/variables.py +0 -0
  218. {querysource-4.3.0 → querysource-4.4.0}/querysource/interfaces/__init__.py +0 -0
  219. {querysource-4.3.0 → querysource-4.4.0}/querysource/interfaces/airtable.py +0 -0
  220. {querysource-4.3.0 → querysource-4.4.0}/querysource/interfaces/connections.py +0 -0
  221. {querysource-4.3.0 → querysource-4.4.0}/querysource/interfaces/credentials.py +0 -0
  222. {querysource-4.3.0 → querysource-4.4.0}/querysource/interfaces/databases/__init__.py +0 -0
  223. {querysource-4.3.0 → querysource-4.4.0}/querysource/interfaces/databases/abstract.py +0 -0
  224. {querysource-4.3.0 → querysource-4.4.0}/querysource/interfaces/databases/bigquery.py +0 -0
  225. {querysource-4.3.0 → querysource-4.4.0}/querysource/interfaces/databases/db.py +0 -0
  226. {querysource-4.3.0 → querysource-4.4.0}/querysource/interfaces/databases/mongo.py +0 -0
  227. {querysource-4.3.0 → querysource-4.4.0}/querysource/interfaces/databases/rethink.py +0 -0
  228. {querysource-4.3.0 → querysource-4.4.0}/querysource/interfaces/http.py +0 -0
  229. {querysource-4.3.0 → querysource-4.4.0}/querysource/interfaces/playwright_service.py +0 -0
  230. {querysource-4.3.0 → querysource-4.4.0}/querysource/interfaces/queries.py +0 -0
  231. {querysource-4.3.0 → querysource-4.4.0}/querysource/interfaces/selenium_service.py +0 -0
  232. {querysource-4.3.0 → querysource-4.4.0}/querysource/libs/__init__.py +0 -0
  233. {querysource-4.3.0 → querysource-4.4.0}/querysource/libs/encoders.py +0 -0
  234. {querysource-4.3.0 → querysource-4.4.0}/querysource/libs/functions/__init__.py +0 -0
  235. {querysource-4.3.0 → querysource-4.4.0}/querysource/libs/py.typed +0 -0
  236. {querysource-4.3.0 → querysource-4.4.0}/querysource/models.py +0 -0
  237. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/__init__.py +0 -0
  238. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/destinations/__init__.py +0 -0
  239. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/destinations/abstract.py +0 -0
  240. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/destinations/dwh.py +0 -0
  241. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/destinations/s3.py +0 -0
  242. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/destinations/sharepoint.py +0 -0
  243. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/destinations/table.py +0 -0
  244. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/dt/__init__.py +0 -0
  245. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/dt/abstract.py +0 -0
  246. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/dt/arrow.py +0 -0
  247. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/dt/dt.py +0 -0
  248. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/dt/factory.py +0 -0
  249. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/dt/iter.py +0 -0
  250. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/dt/modin.py +0 -0
  251. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/dt/pandas.py +0 -0
  252. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/dt/polars.py +0 -0
  253. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/output.py +0 -0
  254. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/tables/TableOutput/__init__.py +0 -0
  255. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/tables/TableOutput/abstract.py +0 -0
  256. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/tables/TableOutput/bigquery.py +0 -0
  257. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/tables/TableOutput/documentdb.py +0 -0
  258. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/tables/TableOutput/mongodb.py +0 -0
  259. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/tables/TableOutput/mysql.py +0 -0
  260. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/tables/TableOutput/postgres.py +0 -0
  261. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/tables/TableOutput/rethink.py +0 -0
  262. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/tables/TableOutput/sa.py +0 -0
  263. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/tables/TableOutput/table.py +0 -0
  264. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/tables/__init__.py +0 -0
  265. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/writers/__init__.py +0 -0
  266. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/writers/abstract.py +0 -0
  267. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/writers/bokeh.py +0 -0
  268. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/writers/clustering.py +0 -0
  269. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/writers/csv.py +0 -0
  270. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/writers/describe.py +0 -0
  271. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/writers/eda.py +0 -0
  272. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/writers/excel.py +0 -0
  273. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/writers/html.py +0 -0
  274. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/writers/pdf.py +0 -0
  275. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/writers/pickle.py +0 -0
  276. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/writers/plotly.py +0 -0
  277. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/writers/profiling.py +0 -0
  278. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/writers/report.py +0 -0
  279. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/writers/table.py +0 -0
  280. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/writers/tsv.py +0 -0
  281. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/writers/txt.py +0 -0
  282. {querysource-4.3.0 → querysource-4.4.0}/querysource/outputs/writers/xml.py +0 -0
  283. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/__init__.py +0 -0
  284. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/abstract.c +0 -0
  285. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/abstract.pxd +0 -0
  286. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/abstract.pyx +0 -0
  287. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/arangodb.c +0 -0
  288. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/arangodb.pxd +0 -0
  289. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/arangodb.pyx +0 -0
  290. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/bigquery.c +0 -0
  291. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/bigquery.pxd +0 -0
  292. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/bigquery.pyx +0 -0
  293. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/cql.c +0 -0
  294. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/cql.pxd +0 -0
  295. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/cql.pyx +0 -0
  296. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/deltatbl.c +0 -0
  297. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/deltatbl.pxd +0 -0
  298. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/deltatbl.pyx +0 -0
  299. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/elastic.cpp +0 -0
  300. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/elastic.pxd +0 -0
  301. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/elastic.pyx +0 -0
  302. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/iceberg.c +0 -0
  303. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/iceberg.pxd +0 -0
  304. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/iceberg.pyx +0 -0
  305. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/influx.c +0 -0
  306. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/influx.pxd +0 -0
  307. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/influx.pyx +0 -0
  308. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/mongo.cpp +0 -0
  309. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/mongo.pxd +0 -0
  310. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/mongo.pyx +0 -0
  311. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/parser.c +0 -0
  312. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/parser.pxd +0 -0
  313. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/parser.pyx +0 -0
  314. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/pgsql.c +0 -0
  315. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/pgsql.pxd +0 -0
  316. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/pgsql.pyx +0 -0
  317. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/rethink.c +0 -0
  318. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/rethink.pxd +0 -0
  319. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/rethink.pyx +0 -0
  320. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/sosql.c +0 -0
  321. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/sosql.pxd +0 -0
  322. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/sosql.pyx +0 -0
  323. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/sql.c +0 -0
  324. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/sql.pxd +0 -0
  325. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/sql.pyx +0 -0
  326. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/sqlserver.c +0 -0
  327. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/sqlserver.pxd +0 -0
  328. {querysource-4.3.0 → querysource-4.4.0}/querysource/parsers/sqlserver.pyx +0 -0
  329. {querysource-4.3.0 → querysource-4.4.0}/querysource/plugins/__init__.py +0 -0
  330. {querysource-4.3.0 → querysource-4.4.0}/querysource/plugins/importer.py +0 -0
  331. {querysource-4.3.0 → querysource-4.4.0}/querysource/plugins/sources/__init__.py +0 -0
  332. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/__init__.py +0 -0
  333. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/abstract.py +0 -0
  334. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/arangodb.py +0 -0
  335. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/bigquery.py +0 -0
  336. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/cassandra.py +0 -0
  337. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/db.py +0 -0
  338. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/default.py +0 -0
  339. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/deltatbl.py +0 -0
  340. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/documentdb.py +0 -0
  341. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/dummy.py +0 -0
  342. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/elastic.py +0 -0
  343. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/external.py +0 -0
  344. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/http.py +0 -0
  345. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/iceberg.py +0 -0
  346. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/influx.py +0 -0
  347. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/mysql.py +0 -0
  348. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/pg.py +0 -0
  349. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/py.typed +0 -0
  350. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/rest.py +0 -0
  351. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/rethink.py +0 -0
  352. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/salesforce.py +0 -0
  353. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/scylladb.py +0 -0
  354. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/sources/__init__.py +0 -0
  355. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/sources/abstract.py +0 -0
  356. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/sources/amazon.py +0 -0
  357. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/sources/countries.py +0 -0
  358. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/sources/ga.py +0 -0
  359. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/sources/geofcc.py +0 -0
  360. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/sources/gmaps.py +0 -0
  361. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/sources/graphcountries.py +0 -0
  362. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/sources/graphql.py +0 -0
  363. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/sources/http.py +0 -0
  364. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/sources/hubspot.py +0 -0
  365. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/sources/openweather.py +0 -0
  366. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/sources/parsers/__init__.py +0 -0
  367. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/sources/parsers/amproduct.py +0 -0
  368. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/sources/parsers/xpath.py +0 -0
  369. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/sources/pokemon.py +0 -0
  370. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/sources/populartimes.py +0 -0
  371. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/sources/py.typed +0 -0
  372. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/sources/rest.py +0 -0
  373. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/sources/retailnext.py +0 -0
  374. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/sources/rssapp.py +0 -0
  375. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/sources/salesforce.py +0 -0
  376. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/sources/scrapper.py +0 -0
  377. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/sources/shoppertrack.py +0 -0
  378. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/sources/swop.py +0 -0
  379. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/sources/uap.py +0 -0
  380. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/sources/upc.py +0 -0
  381. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/sources/wm_stores.py +0 -0
  382. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/sources/zammad.py +0 -0
  383. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/sources/zipcodeapi.py +0 -0
  384. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/sql.py +0 -0
  385. {querysource-4.3.0 → querysource-4.4.0}/querysource/providers/sqlserver.py +0 -0
  386. {querysource-4.3.0 → querysource-4.4.0}/querysource/py.typed +0 -0
  387. {querysource-4.3.0 → querysource-4.4.0}/querysource/qs_parsers/__init__.py +0 -0
  388. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/__init__.py +0 -0
  389. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/base.py +0 -0
  390. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/executor.py +0 -0
  391. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/models.py +0 -0
  392. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/_introspect.py +0 -0
  393. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/abstract.py +0 -0
  394. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/components/__init__.py +0 -0
  395. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/components/abstract.py +0 -0
  396. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/destinations/__init__.py +0 -0
  397. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/destinations/dwh.py +0 -0
  398. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/destinations/s3.py +0 -0
  399. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/destinations/sharepoint.py +0 -0
  400. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/destinations/table.py +0 -0
  401. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/operators/Concat.py +0 -0
  402. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/operators/GroupBy.py +0 -0
  403. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/operators/Info.py +0 -0
  404. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/operators/Join.py +0 -0
  405. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/operators/Melt.py +0 -0
  406. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/operators/Merge.py +0 -0
  407. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/operators/__init__.py +0 -0
  408. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/operators/abstract.py +0 -0
  409. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/operators/filter/__init__.py +0 -0
  410. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/operators/filter/flt.py +0 -0
  411. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/registry.py +0 -0
  412. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/sources/airtable.py +0 -0
  413. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/sources/base.py +0 -0
  414. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/sources/file.py +0 -0
  415. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/sources/s3.py +0 -0
  416. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/sources/sharepoint.py +0 -0
  417. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/sources/smartsheet.py +0 -0
  418. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/sources/table.py +0 -0
  419. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/transformations/DropCols.py +0 -0
  420. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/transformations/FilterCols.py +0 -0
  421. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/transformations/Forecast.py +0 -0
  422. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/transformations/Map.py +0 -0
  423. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/transformations/PluckCols.py +0 -0
  424. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/transformations/__init__.py +0 -0
  425. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/transformations/abstract.py +0 -0
  426. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/transformations/correlation.py +0 -0
  427. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/transformations/crosstab.py +0 -0
  428. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/transformations/google/__init__.py +0 -0
  429. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/transformations/google/maps.py +0 -0
  430. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/transformations/pivot.py +0 -0
  431. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/transformations/tExplode.py +0 -0
  432. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/transformations/tOrder.py +0 -0
  433. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/multi/transformations/tPandas.py +0 -0
  434. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/obj.py +0 -0
  435. {querysource-4.3.0 → querysource-4.4.0}/querysource/queries/qs.py +0 -0
  436. {querysource-4.3.0 → querysource-4.4.0}/querysource/scheduler/__init__.py +0 -0
  437. {querysource-4.3.0 → querysource-4.4.0}/querysource/scheduler/jobs.py +0 -0
  438. {querysource-4.3.0 → querysource-4.4.0}/querysource/scheduler/notifications.py +0 -0
  439. {querysource-4.3.0 → querysource-4.4.0}/querysource/services.py +0 -0
  440. {querysource-4.3.0 → querysource-4.4.0}/querysource/template/__init__.py +0 -0
  441. {querysource-4.3.0 → querysource-4.4.0}/querysource/template/parser.py +0 -0
  442. {querysource-4.3.0 → querysource-4.4.0}/querysource/types/__init__.py +0 -0
  443. {querysource-4.3.0 → querysource-4.4.0}/querysource/types/converters.cpp +0 -0
  444. {querysource-4.3.0 → querysource-4.4.0}/querysource/types/converters.pyx +0 -0
  445. {querysource-4.3.0 → querysource-4.4.0}/querysource/types/dt/__init__.py +0 -0
  446. {querysource-4.3.0 → querysource-4.4.0}/querysource/types/dt/transforms.py +0 -0
  447. {querysource-4.3.0 → querysource-4.4.0}/querysource/types/py.typed +0 -0
  448. {querysource-4.3.0 → querysource-4.4.0}/querysource/types/typedefs.c +0 -0
  449. {querysource-4.3.0 → querysource-4.4.0}/querysource/types/typedefs.pyx +0 -0
  450. {querysource-4.3.0 → querysource-4.4.0}/querysource/types/validators.cpp +0 -0
  451. {querysource-4.3.0 → querysource-4.4.0}/querysource/types/validators.pyx +0 -0
  452. {querysource-4.3.0 → querysource-4.4.0}/querysource/utils/__init__.py +0 -0
  453. {querysource-4.3.0 → querysource-4.4.0}/querysource/utils/cache_serialization.py +0 -0
  454. {querysource-4.3.0 → querysource-4.4.0}/querysource/utils/events.py +0 -0
  455. {querysource-4.3.0 → querysource-4.4.0}/querysource/utils/fn.py +0 -0
  456. {querysource-4.3.0 → querysource-4.4.0}/querysource/utils/functions.cpp +0 -0
  457. {querysource-4.3.0 → querysource-4.4.0}/querysource/utils/functions.pyx +0 -0
  458. {querysource-4.3.0 → querysource-4.4.0}/querysource/utils/getfunc.py +0 -0
  459. {querysource-4.3.0 → querysource-4.4.0}/querysource/utils/handlers.py +0 -0
  460. {querysource-4.3.0 → querysource-4.4.0}/querysource/utils/parseqs.cpp +0 -0
  461. {querysource-4.3.0 → querysource-4.4.0}/querysource/utils/parseqs.pyx +0 -0
  462. {querysource-4.3.0 → querysource-4.4.0}/querysource/utils/validators.py +0 -0
  463. {querysource-4.3.0 → querysource-4.4.0}/querysource.egg-info/dependency_links.txt +0 -0
  464. {querysource-4.3.0 → querysource-4.4.0}/querysource.egg-info/entry_points.txt +0 -0
  465. {querysource-4.3.0 → querysource-4.4.0}/querysource.egg-info/not-zip-safe +0 -0
  466. {querysource-4.3.0 → querysource-4.4.0}/querysource.egg-info/requires.txt +0 -0
  467. {querysource-4.3.0 → querysource-4.4.0}/querysource.egg-info/top_level.txt +0 -0
  468. {querysource-4.3.0 → querysource-4.4.0}/run.py +0 -0
  469. {querysource-4.3.0 → querysource-4.4.0}/rust/Cargo.lock +0 -0
  470. {querysource-4.3.0 → querysource-4.4.0}/rust/Cargo.toml +0 -0
  471. {querysource-4.3.0 → querysource-4.4.0}/rust/pyproject.toml +0 -0
  472. {querysource-4.3.0 → querysource-4.4.0}/rust/src/arangodb_parser.rs +0 -0
  473. {querysource-4.3.0 → querysource-4.4.0}/rust/src/bigquery_parser.rs +0 -0
  474. {querysource-4.3.0 → querysource-4.4.0}/rust/src/cql_parser.rs +0 -0
  475. {querysource-4.3.0 → querysource-4.4.0}/rust/src/elastic_parser.rs +0 -0
  476. {querysource-4.3.0 → querysource-4.4.0}/rust/src/filter_common.rs +0 -0
  477. {querysource-4.3.0 → querysource-4.4.0}/rust/src/flux_parser.rs +0 -0
  478. {querysource-4.3.0 → querysource-4.4.0}/rust/src/lib.rs +0 -0
  479. {querysource-4.3.0 → querysource-4.4.0}/rust/src/mongo_parser.rs +0 -0
  480. {querysource-4.3.0 → querysource-4.4.0}/rust/src/mssql_parser.rs +0 -0
  481. {querysource-4.3.0 → querysource-4.4.0}/rust/src/parseqs.rs +0 -0
  482. {querysource-4.3.0 → querysource-4.4.0}/rust/src/pgsql_parser.rs +0 -0
  483. {querysource-4.3.0 → querysource-4.4.0}/rust/src/rethink_parser.rs +0 -0
  484. {querysource-4.3.0 → querysource-4.4.0}/rust/src/safe_dict.rs +0 -0
  485. {querysource-4.3.0 → querysource-4.4.0}/rust/src/soql_parser.rs +0 -0
  486. {querysource-4.3.0 → querysource-4.4.0}/rust/src/sql_parser.rs +0 -0
  487. {querysource-4.3.0 → querysource-4.4.0}/rust/src/validators.rs +0 -0
  488. {querysource-4.3.0 → querysource-4.4.0}/setup.cfg +0 -0
  489. {querysource-4.3.0 → querysource-4.4.0}/setup.py +0 -0
  490. {querysource-4.3.0 → querysource-4.4.0}/static/notebook/bundle.js +0 -0
  491. {querysource-4.3.0 → querysource-4.4.0}/templates/__init__.py +0 -0
  492. {querysource-4.3.0 → querysource-4.4.0}/templates/base.html +0 -0
  493. {querysource-4.3.0 → querysource-4.4.0}/templates/default.html +0 -0
  494. {querysource-4.3.0 → querysource-4.4.0}/templates/default_table.html +0 -0
  495. {querysource-4.3.0 → querysource-4.4.0}/templates/fontlist-v330.json +0 -0
  496. {querysource-4.3.0 → querysource-4.4.0}/templates/fontlist-v390.json +0 -0
  497. {querysource-4.3.0 → querysource-4.4.0}/templates/table_charts.html +0 -0
  498. {querysource-4.3.0 → querysource-4.4.0}/tests/auth/__init__.py +0 -0
  499. {querysource-4.3.0 → querysource-4.4.0}/tests/auth/test_credentials.py +0 -0
  500. {querysource-4.3.0 → querysource-4.4.0}/tests/auth/test_pbac_bootstrap.py +0 -0
  501. {querysource-4.3.0 → querysource-4.4.0}/tests/check_concurrent_get_slug.py +0 -0
  502. {querysource-4.3.0 → querysource-4.4.0}/tests/conftest.py +0 -0
  503. {querysource-4.3.0 → querysource-4.4.0}/tests/datasources/__init__.py +0 -0
  504. {querysource-4.3.0 → querysource-4.4.0}/tests/datasources/test_datasource_view_pbac.py +0 -0
  505. {querysource-4.3.0 → querysource-4.4.0}/tests/datasources/test_driver_factory_session.py +0 -0
  506. {querysource-4.3.0 → querysource-4.4.0}/tests/datasources/test_pg_admin_registration.py +0 -0
  507. {querysource-4.3.0 → querysource-4.4.0}/tests/datasources/test_pg_params_for.py +0 -0
  508. {querysource-4.3.0 → querysource-4.4.0}/tests/handlers/__init__.py +0 -0
  509. {querysource-4.3.0 → querysource-4.4.0}/tests/handlers/conftest.py +0 -0
  510. {querysource-4.3.0 → querysource-4.4.0}/tests/handlers/test_abstract_pbac_helpers.py +0 -0
  511. {querysource-4.3.0 → querysource-4.4.0}/tests/handlers/test_airtable_oauth.py +0 -0
  512. {querysource-4.3.0 → querysource-4.4.0}/tests/handlers/test_multiquery_pbac_smoke.py +0 -0
  513. {querysource-4.3.0 → querysource-4.4.0}/tests/handlers/test_queryexecutor_pbac_smoke.py +0 -0
  514. {querysource-4.3.0 → querysource-4.4.0}/tests/handlers/test_querymanager_pagination.py +0 -0
  515. {querysource-4.3.0 → querysource-4.4.0}/tests/handlers/test_queryservice_pbac_smoke.py +0 -0
  516. {querysource-4.3.0 → querysource-4.4.0}/tests/integration/__init__.py +0 -0
  517. {querysource-4.3.0 → querysource-4.4.0}/tests/integration/test_pbac_credentials.py +0 -0
  518. {querysource-4.3.0 → querysource-4.4.0}/tests/integration/test_pbac_enforcement.py +0 -0
  519. {querysource-4.3.0 → querysource-4.4.0}/tests/integration/test_pbac_listing.py +0 -0
  520. {querysource-4.3.0 → querysource-4.4.0}/tests/interfaces/__init__.py +0 -0
  521. {querysource-4.3.0 → querysource-4.4.0}/tests/interfaces/test_airtable_interface.py +0 -0
  522. {querysource-4.3.0 → querysource-4.4.0}/tests/multi/__init__.py +0 -0
  523. {querysource-4.3.0 → querysource-4.4.0}/tests/multi/sources/__init__.py +0 -0
  524. {querysource-4.3.0 → querysource-4.4.0}/tests/multi/sources/test_airtable_source.py +0 -0
  525. {querysource-4.3.0 → querysource-4.4.0}/tests/multi/sources/test_registry.py +0 -0
  526. {querysource-4.3.0 → querysource-4.4.0}/tests/perf/__init__.py +0 -0
  527. {querysource-4.3.0 → querysource-4.4.0}/tests/perf/test_pbac_overhead.py +0 -0
  528. {querysource-4.3.0 → querysource-4.4.0}/tests/policies/__init__.py +0 -0
  529. {querysource-4.3.0 → querysource-4.4.0}/tests/policies/test_default_policies_load.py +0 -0
  530. {querysource-4.3.0 → querysource-4.4.0}/tests/scheduler/__init__.py +0 -0
  531. {querysource-4.3.0 → querysource-4.4.0}/tests/scheduler/test_parse_trigger_shorthands.py +0 -0
  532. {querysource-4.3.0 → querysource-4.4.0}/tests/scheduler/test_shorthand_loaders.py +0 -0
  533. {querysource-4.3.0 → querysource-4.4.0}/tests/services/__init__.py +0 -0
  534. {querysource-4.3.0 → querysource-4.4.0}/tests/services/test_querysource_setup_pbac.py +0 -0
  535. {querysource-4.3.0 → querysource-4.4.0}/tests/test_abstract_multi.py +0 -0
  536. {querysource-4.3.0 → querysource-4.4.0}/tests/test_abstract_refactor.py +0 -0
  537. {querysource-4.3.0 → querysource-4.4.0}/tests/test_api.py +0 -0
  538. {querysource-4.3.0 → querysource-4.4.0}/tests/test_arangodb_parser.py +0 -0
  539. {querysource-4.3.0 → querysource-4.4.0}/tests/test_cli_generate_docs.py +0 -0
  540. {querysource-4.3.0 → querysource-4.4.0}/tests/test_component_handler.py +0 -0
  541. {querysource-4.3.0 → querysource-4.4.0}/tests/test_component_registry.py +0 -0
  542. {querysource-4.3.0 → querysource-4.4.0}/tests/test_conf_airtable.py +0 -0
  543. {querysource-4.3.0 → querysource-4.4.0}/tests/test_destination_base.py +0 -0
  544. {querysource-4.3.0 → querysource-4.4.0}/tests/test_destination_dwh.py +0 -0
  545. {querysource-4.3.0 → querysource-4.4.0}/tests/test_destination_integration.py +0 -0
  546. {querysource-4.3.0 → querysource-4.4.0}/tests/test_destination_s3.py +0 -0
  547. {querysource-4.3.0 → querysource-4.4.0}/tests/test_destination_sharepoint.py +0 -0
  548. {querysource-4.3.0 → querysource-4.4.0}/tests/test_destination_table.py +0 -0
  549. {querysource-4.3.0 → querysource-4.4.0}/tests/test_destinations_documentation_endpoint.py +0 -0
  550. {querysource-4.3.0 → querysource-4.4.0}/tests/test_elastic_parser.py +0 -0
  551. {querysource-4.3.0 → querysource-4.4.0}/tests/test_eval.py +0 -0
  552. {querysource-4.3.0 → querysource-4.4.0}/tests/test_info_eda.py +0 -0
  553. {querysource-4.3.0 → querysource-4.4.0}/tests/test_info_eda_e2e.py +0 -0
  554. {querysource-4.3.0 → querysource-4.4.0}/tests/test_info_eda_integration.py +0 -0
  555. {querysource-4.3.0 → querysource-4.4.0}/tests/test_join_conditions.py +0 -0
  556. {querysource-4.3.0 → querysource-4.4.0}/tests/test_join_with_column_filter.py +0 -0
  557. {querysource-4.3.0 → querysource-4.4.0}/tests/test_multi_destinations_subpackage.py +0 -0
  558. {querysource-4.3.0 → querysource-4.4.0}/tests/test_multiqs_column_transforms.py +0 -0
  559. {querysource-4.3.0 → querysource-4.4.0}/tests/test_multiqs_destination_dispatch.py +0 -0
  560. {querysource-4.3.0 → querysource-4.4.0}/tests/test_queryslug_concurrency.py +0 -0
  561. {querysource-4.3.0 → querysource-4.4.0}/tests/test_querysource_setup_airtable.py +0 -0
  562. {querysource-4.3.0 → querysource-4.4.0}/tests/test_route_registration.py +0 -0
  563. {querysource-4.3.0 → querysource-4.4.0}/tests/test_rss.py +0 -0
  564. {querysource-4.3.0 → querysource-4.4.0}/tests/test_rust_parsers.py +0 -0
  565. {querysource-4.3.0 → querysource-4.4.0}/tests/test_scheduler_core.py +0 -0
  566. {querysource-4.3.0 → querysource-4.4.0}/tests/test_scheduler_handler_integration.py +0 -0
  567. {querysource-4.3.0 → querysource-4.4.0}/tests/test_scheduler_handler_unit.py +0 -0
  568. {querysource-4.3.0 → querysource-4.4.0}/tests/test_scheduler_integration.py +0 -0
  569. {querysource-4.3.0 → querysource-4.4.0}/tests/test_scheduler_jobs.py +0 -0
  570. {querysource-4.3.0 → querysource-4.4.0}/tests/test_scheduler_multi_routing.py +0 -0
  571. {querysource-4.3.0 → querysource-4.4.0}/tests/test_scheduler_notifications.py +0 -0
  572. {querysource-4.3.0 → querysource-4.4.0}/tests/test_schema_introspectable.py +0 -0
  573. {querysource-4.3.0 → querysource-4.4.0}/tests/test_source_registry.py +0 -0
  574. {querysource-4.3.0 → querysource-4.4.0}/tests/test_source_s3.py +0 -0
  575. {querysource-4.3.0 → querysource-4.4.0}/tests/test_source_sharepoint.py +0 -0
  576. {querysource-4.3.0 → querysource-4.4.0}/tests/test_source_smartsheet.py +0 -0
  577. {querysource-4.3.0 → querysource-4.4.0}/tests/test_source_table.py +0 -0
  578. {querysource-4.3.0 → querysource-4.4.0}/tests/test_sql_parser_combinations.py +0 -0
  579. {querysource-4.3.0 → querysource-4.4.0}/tests/test_texplode.py +0 -0
  580. {querysource-4.3.0 → querysource-4.4.0}/tests/test_thread_file_refactor.py +0 -0
  581. {querysource-4.3.0 → querysource-4.4.0}/tests/test_thread_query_refactor.py +0 -0
  582. {querysource-4.3.0 → querysource-4.4.0}/tests/test_thread_source_base.py +0 -0
  583. {querysource-4.3.0 → querysource-4.4.0}/tests/test_zammad_search.py +0 -0
  584. {querysource-4.3.0 → querysource-4.4.0}/tox.ini +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: querysource
3
- Version: 4.3.0
3
+ Version: 4.4.0
4
4
  Summary: Aiohttp web service for querying several databases easily
5
5
  Author-email: Jesus Lara <jesuslarag@gmail.com>
6
6
  License: BSD-3-Clause
@@ -119,9 +119,9 @@ dependencies = [
119
119
 
120
120
  [[package]]
121
121
  name = "pyo3"
122
- version = "0.23.5"
122
+ version = "0.24.1"
123
123
  source = "registry+https://github.com/rust-lang/crates.io-index"
124
- checksum = "7778bffd85cf38175ac1f545509665d0b9b92a198ca7941f131f85f7a4f9a872"
124
+ checksum = "17da310086b068fbdcefbba30aeb3721d5bb9af8db4987d6735b2183ca567229"
125
125
  dependencies = [
126
126
  "cfg-if",
127
127
  "indoc",
@@ -137,9 +137,9 @@ dependencies = [
137
137
 
138
138
  [[package]]
139
139
  name = "pyo3-build-config"
140
- version = "0.23.5"
140
+ version = "0.24.1"
141
141
  source = "registry+https://github.com/rust-lang/crates.io-index"
142
- checksum = "94f6cbe86ef3bf18998d9df6e0f3fc1050a8c5efa409bf712e661a4366e010fb"
142
+ checksum = "e27165889bd793000a098bb966adc4300c312497ea25cf7a690a9f0ac5aa5fc1"
143
143
  dependencies = [
144
144
  "once_cell",
145
145
  "target-lexicon",
@@ -147,9 +147,9 @@ dependencies = [
147
147
 
148
148
  [[package]]
149
149
  name = "pyo3-ffi"
150
- version = "0.23.5"
150
+ version = "0.24.1"
151
151
  source = "registry+https://github.com/rust-lang/crates.io-index"
152
- checksum = "e9f1b4c431c0bb1c8fb0a338709859eed0d030ff6daa34368d3b152a63dfdd8d"
152
+ checksum = "05280526e1dbf6b420062f3ef228b78c0c54ba94e157f5cb724a609d0f2faabc"
153
153
  dependencies = [
154
154
  "libc",
155
155
  "pyo3-build-config",
@@ -157,9 +157,9 @@ dependencies = [
157
157
 
158
158
  [[package]]
159
159
  name = "pyo3-macros"
160
- version = "0.23.5"
160
+ version = "0.24.1"
161
161
  source = "registry+https://github.com/rust-lang/crates.io-index"
162
- checksum = "fbc2201328f63c4710f68abdf653c89d8dbc2858b88c5d88b0ff38a75288a9da"
162
+ checksum = "5c3ce5686aa4d3f63359a5100c62a127c9f15e8398e5fdeb5deef1fed5cd5f44"
163
163
  dependencies = [
164
164
  "proc-macro2",
165
165
  "pyo3-macros-backend",
@@ -169,9 +169,9 @@ dependencies = [
169
169
 
170
170
  [[package]]
171
171
  name = "pyo3-macros-backend"
172
- version = "0.23.5"
172
+ version = "0.24.1"
173
173
  source = "registry+https://github.com/rust-lang/crates.io-index"
174
- checksum = "fca6726ad0f3da9c9de093d6f116a93c1a38e417ed73bf138472cf4064f72028"
174
+ checksum = "f4cf6faa0cbfb0ed08e89beb8103ae9724eb4750e3a78084ba4017cbe94f3855"
175
175
  dependencies = [
176
176
  "heck",
177
177
  "proc-macro2",
@@ -311,9 +311,9 @@ dependencies = [
311
311
 
312
312
  [[package]]
313
313
  name = "target-lexicon"
314
- version = "0.12.16"
314
+ version = "0.13.5"
315
315
  source = "registry+https://github.com/rust-lang/crates.io-index"
316
- checksum = "61c41af27dd6d1e27b1b16b489db798443478cef1f06a660c96db617ba5de3b1"
316
+ checksum = "adb6935a6f5c20170eeceb1a3835a49e12e19d792f6dd344ccc76a985ca5a6ca"
317
317
 
318
318
  [[package]]
319
319
  name = "unicode-ident"
@@ -10,7 +10,7 @@ name = "qs_parser"
10
10
  crate-type = ["cdylib"]
11
11
 
12
12
  [dependencies]
13
- pyo3 = { version = "0.23.4", features = ["extension-module"] }
13
+ pyo3 = { version = "0.24.1", features = ["extension-module"] }
14
14
  serde = { version = "1.0", features = ["derive"] }
15
15
  serde_json = "1.0"
16
16
  rayon = "1.10"
@@ -18,7 +18,7 @@ version_tuple: tuple[int | str, ...]
18
18
  commit_id: str | None
19
19
  __commit_id__: str | None
20
20
 
21
- __version__ = version = '4.3.1.dev0+gec7dea294.d20260523'
22
- __version_tuple__ = version_tuple = (4, 3, 1, 'dev0', 'gec7dea294.d20260523')
21
+ __version__ = version = '4.4.1.dev0+g74f4a954b.d20260527'
22
+ __version_tuple__ = version_tuple = (4, 4, 1, 'dev0', 'g74f4a954b.d20260527')
23
23
 
24
- __commit_id__ = commit_id = 'gec7dea294'
24
+ __commit_id__ = commit_id = 'g74f4a954b'
@@ -280,6 +280,20 @@ JIRA_CERT = config.get('JIRA_CERT')
280
280
  HTTPCLIENT_MAX_SEMAPHORE = config.getint("HTTPCLIENT_MAX_SEMAPHORE", fallback=5)
281
281
  HTTPCLIENT_MAX_WORKERS = config.getint("HTTPCLIENT_MAX_WORKERS", fallback=1)
282
282
 
283
+ ## MultiQS thread guardrails:
284
+ MULTIQS_MAX_CONCURRENT_THREADS = config.getint(
285
+ "MULTIQS_MAX_CONCURRENT_THREADS",
286
+ fallback=4,
287
+ )
288
+ MULTIQS_MAX_SOURCES_PER_REQUEST = config.getint(
289
+ "MULTIQS_MAX_SOURCES_PER_REQUEST",
290
+ fallback=25,
291
+ )
292
+ MULTIQS_SOURCE_TIMEOUT_SECONDS = config.getint(
293
+ "MULTIQS_SOURCE_TIMEOUT_SECONDS",
294
+ fallback=30,
295
+ )
296
+
283
297
  ## Google API:
284
298
  GOOGLE_API_KEY = config.get('GOOGLE_API_KEY')
285
299
  GOOGLE_SEARCH_API_KEY = config.get('GOOGLE_SEARCH_API_KEY')
@@ -439,6 +453,33 @@ QS_AIRTABLE_OAUTH_ENABLED = config.getboolean(
439
453
  'QS_AIRTABLE_OAUTH_ENABLED', fallback=False
440
454
  )
441
455
 
456
+ # ── QWorker Remote Execution (FEAT-101) ───────────────────────────────────────
457
+ # Host/port of the remote qworker server used when MultiQS queries have
458
+ # ``remote: true``. QWORKER_HOST=None means remote execution is not configured
459
+ # and any ``remote: true`` query without an explicit ``worker:`` key will raise
460
+ # DriverError at dispatch time.
461
+ QWORKER_HOST = config.get('QWORKER_HOST', fallback=None)
462
+ QWORKER_PORT = config.getint('QWORKER_PORT', fallback=8888)
463
+ # TCP connection timeout in seconds (how long QClient waits to establish a
464
+ # connection to the qworker server).
465
+ QWORKER_TIMEOUT = config.getint('QWORKER_TIMEOUT', fallback=5)
466
+ # Query execution timeout in seconds (wraps the entire client.run() call via
467
+ # asyncio.wait_for; independent from the TCP connection timeout above).
468
+ QWORKER_QUERY_TIMEOUT = config.getint('QWORKER_QUERY_TIMEOUT', fallback=60)
469
+
470
+ # Pre-parsed worker list for QClient.
471
+ # Supports "host1:port1,host2:port2" or plain "host1,host2" (uses QWORKER_PORT).
472
+ _qworker_hosts_raw = config.get('QWORKER_HOST', fallback=None)
473
+ QWORKER_WORKERS: list = []
474
+ if _qworker_hosts_raw:
475
+ for _h in _qworker_hosts_raw.split(','):
476
+ _h = _h.strip()
477
+ if ':' in _h:
478
+ _hh, _hp = _h.rsplit(':', 1)
479
+ QWORKER_WORKERS.append((_hh.strip(), int(_hp)))
480
+ else:
481
+ QWORKER_WORKERS.append((_h, QWORKER_PORT))
482
+
442
483
  try:
443
484
  from settings.settings import * # pylint: disable=W0614,W0401 # noqa
444
485
  except ImportError:
@@ -231,15 +231,25 @@ class QueryHandler(AbstractHandler):
231
231
  result, options = await qs.query()
232
232
  except DataNotFound as dnf:
233
233
  total_time = time.monotonic() - started_at
234
+ _remote_queries_on_err = getattr(qs, '_remote_queries', [])
235
+ if _remote_queries_on_err:
236
+ self.logger.warning(
237
+ "MultiQuery DataNotFound after remote queries %s: %s",
238
+ _remote_queries_on_err,
239
+ dnf,
240
+ )
241
+ _err_headers = {
242
+ 'Content-Type': 'application/json',
243
+ 'X-Slug': slug,
244
+ 'X-Format': queryformat,
245
+ 'X-Total-Time': f'{total_time:.2f} seconds',
246
+ 'X-Error': str(dnf),
247
+ }
248
+ if _remote_queries_on_err:
249
+ _err_headers['X-Remote-Queries'] = ','.join(_remote_queries_on_err)
234
250
  return self.NoData(
235
251
  message=str(dnf),
236
- headers={
237
- 'Content-Type': 'application/json',
238
- 'X-Slug': slug,
239
- 'X-Format': queryformat,
240
- 'X-Total-Time': f'{total_time:.2f} seconds',
241
- 'X-Error': str(dnf),
242
- },
252
+ headers=_err_headers,
243
253
  )
244
254
  except SlugNotFound as snf:
245
255
  raise self.Error(
@@ -255,6 +265,13 @@ class QueryHandler(AbstractHandler):
255
265
  )
256
266
  except (QueryException, DriverError) as qe:
257
267
  trace = traceback.format_exc()
268
+ _remote_queries_on_err = getattr(qs, '_remote_queries', [])
269
+ if _remote_queries_on_err:
270
+ self.logger.warning(
271
+ "MultiQuery error after remote queries %s: %s",
272
+ _remote_queries_on_err,
273
+ qe,
274
+ )
258
275
  self.logger.exception(qe, stack_info=True)
259
276
  raise self.Error(
260
277
  message="Query Error",
@@ -264,6 +281,13 @@ class QueryHandler(AbstractHandler):
264
281
  )
265
282
  except Exception as ex:
266
283
  trace = traceback.format_exc()
284
+ _remote_queries_on_err = getattr(qs, '_remote_queries', [])
285
+ if _remote_queries_on_err:
286
+ self.logger.warning(
287
+ "MultiQuery unexpected error after remote queries %s: %s",
288
+ _remote_queries_on_err,
289
+ ex,
290
+ )
267
291
  self.logger.exception(ex, stack_info=True)
268
292
  raise self.Except(
269
293
  message=f"Unknown Error on Query: {ex!s}",
@@ -381,7 +405,12 @@ class QueryHandler(AbstractHandler):
381
405
  self.logger.debug(
382
406
  f'Query Duration: {total_time:.2f} seconds'
383
407
  )
384
- return await output.response()
408
+ response = await output.response()
409
+ # Add X-Remote-Queries header if any queries ran remotely (FEAT-101).
410
+ remote_queries = getattr(qs, '_remote_queries', [])
411
+ if remote_queries:
412
+ response.headers['X-Remote-Queries'] = ','.join(remote_queries)
413
+ return response
385
414
  except (DataNotFound) as ex:
386
415
  return self.NoData(
387
416
  message="No Data was Found",
@@ -0,0 +1,307 @@
1
+ """Scheduler jobstore handler — FEAT-100.
2
+
3
+ HTTP surface on top of QSScheduler's APScheduler jobstore.
4
+
5
+ Routes (registered in querysource/services.py when ENABLE_QS_SCHEDULER=True):
6
+ GET /api/v1/qs/scheduler/jobs -> list all jobs
7
+ GET /api/v1/qs/scheduler/jobs/{job_id} -> single job, or 404
8
+ POST /api/v1/qs/scheduler/jobs -> sync a slug's job(s) live
9
+ DELETE /api/v1/qs/scheduler/jobs/{job_id} -> remove a job live
10
+ PATCH /api/v1/qs/scheduler/jobs/{job_id} -> pause/resume a job live
11
+
12
+ The write verbs mutate the running APScheduler so schedules take effect
13
+ without a restart. The DB (public.queries) stays the source of truth:
14
+ startup rebuilds every job, and POST mirrors a single slug's row into the
15
+ live scheduler.
16
+
17
+ Note: Per-process visibility only. MemoryJobStore lives in the running
18
+ process — multiple QS instances each report (and mutate) their own jobs.
19
+ """
20
+ from __future__ import annotations
21
+
22
+ import logging
23
+ from typing import TYPE_CHECKING, Optional
24
+
25
+ from aiohttp import web
26
+ from navigator.views import BaseView
27
+
28
+ if TYPE_CHECKING:
29
+ from apscheduler.job import Job
30
+ from querysource.scheduler import QSScheduler
31
+
32
+ logger = logging.getLogger("QS.SchedulerJobsView")
33
+
34
+
35
+ def _kind_from_id(job_id: str) -> str:
36
+ """Map job ID prefix to its kind.
37
+
38
+ Args:
39
+ job_id: APScheduler job ID, e.g. ``"query_foo"``.
40
+
41
+ Returns:
42
+ One of ``"query"``, ``"multi"``, ``"cache"``, ``"unknown"``.
43
+ """
44
+ if job_id.startswith("query_"):
45
+ return "query"
46
+ if job_id.startswith("multi_"):
47
+ return "multi"
48
+ if job_id.startswith("cache_"):
49
+ return "cache"
50
+ return "unknown"
51
+
52
+
53
+ class SchedulerJobsView(BaseView):
54
+ """Class-based aiohttp view exposing the QSScheduler jobstore.
55
+
56
+ GET /api/v1/qs/scheduler/jobs — list all jobs
57
+ GET /api/v1/qs/scheduler/jobs/{job_id} — single job by ID
58
+ POST /api/v1/qs/scheduler/jobs — sync a slug's job(s) live
59
+ DELETE /api/v1/qs/scheduler/jobs/{job_id} — remove a job live
60
+ PATCH /api/v1/qs/scheduler/jobs/{job_id} — pause/resume a job live
61
+
62
+ Note: Per-process visibility only. MemoryJobStore lives in the running
63
+ process — if multiple QS instances run with the scheduler enabled, each
64
+ instance answers only for (and mutates) its own jobstore.
65
+ """
66
+
67
+ def _get_scheduler(self) -> Optional["QSScheduler"]:
68
+ """Retrieve the QSScheduler instance from the app state.
69
+
70
+ Returns:
71
+ The QSScheduler stored in ``app["qs_scheduler"]``, or None if
72
+ not present (e.g. during startup or if scheduler was not
73
+ initialised).
74
+ """
75
+ return self.request.app.get("qs_scheduler")
76
+
77
+ def _serialize_job(self, job: "Job") -> dict:
78
+ """Serialize an APScheduler Job to a plain dict.
79
+
80
+ Only extracts ``slug`` from ``job.kwargs`` — never includes the raw
81
+ kwargs dict because it contains non-serialisable internal objects
82
+ such as ``notification_manager``.
83
+
84
+ Args:
85
+ job: An APScheduler ``Job`` instance.
86
+
87
+ Returns:
88
+ A dict with keys: ``id``, ``name``, ``kind``, ``slug``,
89
+ ``next_run_time``, ``trigger``, ``coalesce``, ``max_instances``,
90
+ ``misfire_grace_time``, ``pending``.
91
+ """
92
+ # APScheduler Job uses __slots__; some slots are only set after the
93
+ # scheduler starts (e.g. next_run_time). Use getattr with a default
94
+ # to avoid AttributeError when the scheduler is pending/unstarted.
95
+ nrt = getattr(job, "next_run_time", None)
96
+ trigger_cls_name = type(job.trigger).__name__ # e.g. "IntervalTrigger"
97
+ # Strip the "Trigger" suffix and lowercase to get "interval" / "cron"
98
+ trigger_type = trigger_cls_name.lower()
99
+ if trigger_type.endswith("trigger"):
100
+ trigger_type = trigger_type[: -len("trigger")]
101
+
102
+ slug: Optional[str] = (job.kwargs or {}).get("slug")
103
+
104
+ return {
105
+ "id": job.id,
106
+ "name": job.name,
107
+ "kind": _kind_from_id(job.id),
108
+ "slug": slug,
109
+ "next_run_time": nrt.isoformat() if nrt is not None else None,
110
+ "trigger": {
111
+ "type": trigger_type,
112
+ "repr": str(job.trigger),
113
+ },
114
+ "coalesce": bool(getattr(job, "coalesce", False)),
115
+ "max_instances": int(getattr(job, "max_instances", 1)),
116
+ "misfire_grace_time": getattr(job, "misfire_grace_time", None),
117
+ "pending": bool(getattr(job, "pending", True)),
118
+ }
119
+
120
+ async def get(self) -> web.Response:
121
+ """Handle GET requests for the scheduler jobs endpoint.
122
+
123
+ Dispatches based on the presence of ``{job_id}`` in the URL:
124
+
125
+ - ``GET /api/v1/qs/scheduler/jobs`` — list all jobs.
126
+ - ``GET /api/v1/qs/scheduler/jobs/{job_id}`` — single job lookup.
127
+
128
+ Returns:
129
+ - **200** with the list envelope or single job dict.
130
+ - **404** when ``job_id`` is provided but no matching job exists.
131
+ - **503** when ``app["qs_scheduler"]`` is missing (e.g.
132
+ mid-startup race between route registration and
133
+ ``QSScheduler.startup``).
134
+ """
135
+ scheduler = self._get_scheduler()
136
+ if scheduler is None:
137
+ return self._unavailable_response()
138
+
139
+ aps = scheduler._scheduler # AsyncIOScheduler instance
140
+ params = self.match_parameters(self.request)
141
+ job_id = params.get("job_id")
142
+
143
+ if job_id:
144
+ job = aps.get_job(job_id) if aps is not None else None
145
+ if job is None:
146
+ logger.debug("Job '%s' not found in scheduler.", job_id)
147
+ return self.json_response(
148
+ response={"error": f"Job '{job_id}' not found."},
149
+ status=404,
150
+ )
151
+ return self.json_response(
152
+ response=self._serialize_job(job),
153
+ status=200,
154
+ )
155
+
156
+ # List all jobs
157
+ jobs = aps.get_jobs() if aps is not None else []
158
+ logger.debug("Listing %d scheduler job(s).", len(jobs))
159
+ return self.json_response(
160
+ response={
161
+ "scheduler": {
162
+ "enabled": True,
163
+ "running": bool(aps.running) if aps is not None else False,
164
+ "timezone": str(scheduler._timezone),
165
+ "job_count": len(jobs),
166
+ },
167
+ "jobs": [self._serialize_job(j) for j in jobs],
168
+ },
169
+ status=200,
170
+ )
171
+
172
+ def _unavailable_response(self) -> web.Response:
173
+ """503 envelope used when the scheduler isn't in app state yet."""
174
+ logger.warning(
175
+ "scheduler endpoint called but 'qs_scheduler' is not in app state — "
176
+ "scheduler may still be starting up."
177
+ )
178
+ return self.json_response(
179
+ response={
180
+ "error": (
181
+ "QSScheduler is not available yet — the scheduler is "
182
+ "enabled but the application has not finished starting up."
183
+ ),
184
+ "scheduler": {"enabled": True, "running": False},
185
+ },
186
+ status=503,
187
+ )
188
+
189
+ async def post(self) -> web.Response:
190
+ """Register/sync a slug's scheduled job(s) into the live scheduler.
191
+
192
+ Body: ``{"slug": "<query_slug>"}``.
193
+
194
+ Reads the slug's current ``public.queries`` row and (re)registers its
195
+ query/multi/cache jobs without a restart. If the slug no longer has a
196
+ scheduler definition, its jobs are removed. The DB row written by the
197
+ slug-management API is the source of truth; this only mirrors it into
198
+ the running scheduler.
199
+
200
+ Returns:
201
+ - **200** ``{status, slug, registered: [...], removed: [...]}``.
202
+ - **400** when the body is not JSON or ``slug`` is missing.
203
+ - **503** when the scheduler isn't available yet.
204
+ """
205
+ scheduler = self._get_scheduler()
206
+ if scheduler is None:
207
+ return self._unavailable_response()
208
+
209
+ try:
210
+ body = await self.request.json()
211
+ except Exception: # noqa: BLE001
212
+ return self.json_response(
213
+ response={"error": "Request body must be valid JSON."},
214
+ status=400,
215
+ )
216
+ slug = body.get("slug") if isinstance(body, dict) else None
217
+ if not slug or not isinstance(slug, str):
218
+ return self.json_response(
219
+ response={"error": "Body must include a non-empty 'slug' string."},
220
+ status=400,
221
+ )
222
+
223
+ try:
224
+ result = await scheduler.register_slug(slug)
225
+ except Exception as exc: # noqa: BLE001
226
+ logger.error("Error registering slug '%s': %s", slug, exc)
227
+ return self.json_response(
228
+ response={"error": str(exc)},
229
+ status=500,
230
+ )
231
+ return self.json_response(response={"status": "ok", **result}, status=200)
232
+
233
+ async def delete(self) -> web.Response:
234
+ """Remove a single job from the live scheduler.
235
+
236
+ ``DELETE /api/v1/qs/scheduler/jobs/{job_id}``.
237
+
238
+ Returns:
239
+ - **200** ``{status: "removed", id}`` on success.
240
+ - **400** when ``job_id`` is missing from the path.
241
+ - **404** when no such job exists.
242
+ - **503** when the scheduler isn't available yet.
243
+ """
244
+ scheduler = self._get_scheduler()
245
+ if scheduler is None:
246
+ return self._unavailable_response()
247
+
248
+ job_id = self.match_parameters(self.request).get("job_id")
249
+ if not job_id:
250
+ return self.json_response(
251
+ response={"error": "job_id is required in the path."},
252
+ status=400,
253
+ )
254
+ if not scheduler.remove_job(job_id):
255
+ return self.json_response(
256
+ response={"error": f"Job '{job_id}' not found."},
257
+ status=404,
258
+ )
259
+ return self.json_response(
260
+ response={"status": "removed", "id": job_id},
261
+ status=200,
262
+ )
263
+
264
+ async def patch(self) -> web.Response:
265
+ """Pause or resume a live job.
266
+
267
+ ``PATCH /api/v1/qs/scheduler/jobs/{job_id}`` with body
268
+ ``{"action": "pause" | "resume"}``.
269
+
270
+ Returns:
271
+ - **200** ``{status: "paused"|"resumed", id}`` on success.
272
+ - **400** when ``job_id`` or a valid ``action`` is missing.
273
+ - **404** when no such job exists.
274
+ - **503** when the scheduler isn't available yet.
275
+ """
276
+ scheduler = self._get_scheduler()
277
+ if scheduler is None:
278
+ return self._unavailable_response()
279
+
280
+ job_id = self.match_parameters(self.request).get("job_id")
281
+ if not job_id:
282
+ return self.json_response(
283
+ response={"error": "job_id is required in the path."},
284
+ status=400,
285
+ )
286
+ try:
287
+ body = await self.request.json()
288
+ except Exception: # noqa: BLE001
289
+ return self.json_response(
290
+ response={"error": "Request body must be valid JSON."},
291
+ status=400,
292
+ )
293
+ action = body.get("action") if isinstance(body, dict) else None
294
+ if action not in ("pause", "resume"):
295
+ return self.json_response(
296
+ response={"error": "Body 'action' must be 'pause' or 'resume'."},
297
+ status=400,
298
+ )
299
+ if not scheduler.set_job_paused(job_id, action == "pause"):
300
+ return self.json_response(
301
+ response={"error": f"Job '{job_id}' not found."},
302
+ status=404,
303
+ )
304
+ return self.json_response(
305
+ response={"status": f"{action}d", "id": job_id},
306
+ status=200,
307
+ )
@@ -15,11 +15,27 @@ class jsonWriter(AbstractWriter):
15
15
  from pandas import DataFrame
16
16
  is_dataframe = isinstance(self.data, DataFrame)
17
17
  except ImportError:
18
+ DataFrame = None # type: ignore
18
19
  is_dataframe = False
20
+ # Multi-source pipelines without an operator (no Join/Concat/Melt/Merge)
21
+ # return a dict of DataFrames (one per source). Serialize each frame to
22
+ # records so the frontend can render per-source tabs.
23
+ is_multi_df = (
24
+ DataFrame is not None
25
+ and isinstance(self.data, dict)
26
+ and len(self.data) > 0
27
+ and all(isinstance(v, DataFrame) for v in self.data.values())
28
+ )
19
29
  if is_dataframe:
20
30
  # Convert to a list of dictionaries
21
31
  data_dict = self.data.to_dict(orient='records')
22
32
  data = self._json.dumps(data_dict)
33
+ elif is_multi_df:
34
+ data_dict = {
35
+ key: frame.to_dict(orient='records')
36
+ for key, frame in self.data.items()
37
+ }
38
+ data = self._json.dumps(data_dict)
23
39
  else:
24
40
  try:
25
41
  data = self._json.dumps(self.data)