querysource 4.2.0__tar.gz → 4.2.1__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 (440) hide show
  1. querysource-4.2.1/JOIN_CONDITIONS_FEATURE.md +499 -0
  2. querysource-4.2.1/JOIN_CONDITIONS_SUMMARY.md +403 -0
  3. {querysource-4.2.0/querysource.egg-info → querysource-4.2.1}/PKG-INFO +1 -1
  4. {querysource-4.2.0 → querysource-4.2.1}/querysource/_version.py +3 -3
  5. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/handlers/datasource.py +9 -14
  6. {querysource-4.2.0 → querysource-4.2.1}/querysource/handlers/manager.py +11 -16
  7. {querysource-4.2.0 → querysource-4.2.1}/querysource/interfaces/connections.py +7 -8
  8. {querysource-4.2.0 → querysource-4.2.1}/querysource/version.py +1 -1
  9. {querysource-4.2.0 → querysource-4.2.1/querysource.egg-info}/PKG-INFO +1 -1
  10. {querysource-4.2.0 → querysource-4.2.1}/querysource.egg-info/SOURCES.txt +4 -0
  11. querysource-4.2.1/tests/check_concurrent_get_slug.py +78 -0
  12. querysource-4.2.1/tests/test_queryslug_concurrency.py +156 -0
  13. {querysource-4.2.0 → querysource-4.2.1}/.bumpversion.cfg +0 -0
  14. {querysource-4.2.0 → querysource-4.2.1}/.claude/agents/code-reviewer.md +0 -0
  15. {querysource-4.2.0 → querysource-4.2.1}/.claude/agents/sdd-worker.md +0 -0
  16. {querysource-4.2.0 → querysource-4.2.1}/.claude/commands/pr-review.md +0 -0
  17. {querysource-4.2.0 → querysource-4.2.1}/.claude/commands/sdd-brainstorm.md +0 -0
  18. {querysource-4.2.0 → querysource-4.2.1}/.claude/commands/sdd-codereview.md +0 -0
  19. {querysource-4.2.0 → querysource-4.2.1}/.claude/commands/sdd-done.md +0 -0
  20. {querysource-4.2.0 → querysource-4.2.1}/.claude/commands/sdd-fromjira.md +0 -0
  21. {querysource-4.2.0 → querysource-4.2.1}/.claude/commands/sdd-next.md +0 -0
  22. {querysource-4.2.0 → querysource-4.2.1}/.claude/commands/sdd-proposal.md +0 -0
  23. {querysource-4.2.0 → querysource-4.2.1}/.claude/commands/sdd-spec.md +0 -0
  24. {querysource-4.2.0 → querysource-4.2.1}/.claude/commands/sdd-start.md +0 -0
  25. {querysource-4.2.0 → querysource-4.2.1}/.claude/commands/sdd-status.md +0 -0
  26. {querysource-4.2.0 → querysource-4.2.1}/.claude/commands/sdd-task.md +0 -0
  27. {querysource-4.2.0 → querysource-4.2.1}/.claude/commands/sdd-tojira.md +0 -0
  28. {querysource-4.2.0 → querysource-4.2.1}/.claude/rules/aws-cost-optimization.md +0 -0
  29. {querysource-4.2.0 → querysource-4.2.1}/.claude/rules/code-reviewer.md +0 -0
  30. {querysource-4.2.0 → querysource-4.2.1}/.claude/rules/cython-development.md +0 -0
  31. {querysource-4.2.0 → querysource-4.2.1}/.claude/rules/python-development.md +0 -0
  32. {querysource-4.2.0 → querysource-4.2.1}/.claude/rules/rust-development.md +0 -0
  33. {querysource-4.2.0 → querysource-4.2.1}/.claude/rules/using-git-worktrees.md +0 -0
  34. {querysource-4.2.0 → querysource-4.2.1}/.claude/rules/worktree-pr-and-clean.md +0 -0
  35. {querysource-4.2.0 → querysource-4.2.1}/.claude/rules/worktree-start-feature.md +0 -0
  36. {querysource-4.2.0 → querysource-4.2.1}/.claude/rules/worktree-status.md +0 -0
  37. {querysource-4.2.0 → querysource-4.2.1}/.github/dependabot.yml +0 -0
  38. {querysource-4.2.0 → querysource-4.2.1}/.github/workflows/codeql-analysis.yml +0 -0
  39. {querysource-4.2.0 → querysource-4.2.1}/.github/workflows/release.yml +0 -0
  40. {querysource-4.2.0 → querysource-4.2.1}/.isort.cfg +0 -0
  41. {querysource-4.2.0 → querysource-4.2.1}/.jupyter/jupyter_notebook_config.py +0 -0
  42. {querysource-4.2.0 → querysource-4.2.1}/.pylintrc +0 -0
  43. {querysource-4.2.0 → querysource-4.2.1}/CHANGES.rst +0 -0
  44. {querysource-4.2.0 → querysource-4.2.1}/CODE_OF_CONDUCT.md +0 -0
  45. {querysource-4.2.0 → querysource-4.2.1}/CONTRIBUTING.md +0 -0
  46. {querysource-4.2.0 → querysource-4.2.1}/INSTALL +0 -0
  47. {querysource-4.2.0 → querysource-4.2.1}/LICENSE +0 -0
  48. {querysource-4.2.0 → querysource-4.2.1}/MANIFEST.in +0 -0
  49. {querysource-4.2.0 → querysource-4.2.1}/Makefile +0 -0
  50. {querysource-4.2.0 → querysource-4.2.1}/README.md +0 -0
  51. {querysource-4.2.0 → querysource-4.2.1}/app.py +0 -0
  52. {querysource-4.2.0 → querysource-4.2.1}/bin/README.md +0 -0
  53. {querysource-4.2.0 → querysource-4.2.1}/gunicorn_config.py +0 -0
  54. {querysource-4.2.0 → querysource-4.2.1}/mypy.ini +0 -0
  55. {querysource-4.2.0 → querysource-4.2.1}/nav.py +0 -0
  56. {querysource-4.2.0 → querysource-4.2.1}/policies/datasources.yaml +0 -0
  57. {querysource-4.2.0 → querysource-4.2.1}/policies/defaults.yaml +0 -0
  58. {querysource-4.2.0 → querysource-4.2.1}/policies/drivers.yaml +0 -0
  59. {querysource-4.2.0 → querysource-4.2.1}/policies/raw_queries.yaml +0 -0
  60. {querysource-4.2.0 → querysource-4.2.1}/policies/slugs.yaml +0 -0
  61. {querysource-4.2.0 → querysource-4.2.1}/policies/superusers.yaml +0 -0
  62. {querysource-4.2.0 → querysource-4.2.1}/pyproject.toml +0 -0
  63. {querysource-4.2.0 → querysource-4.2.1}/pytest.ini +0 -0
  64. {querysource-4.2.0 → querysource-4.2.1}/querysource/__cli__.py +0 -0
  65. {querysource-4.2.0 → querysource-4.2.1}/querysource/__init__.py +0 -0
  66. {querysource-4.2.0 → querysource-4.2.1}/querysource/auth/__init__.py +0 -0
  67. {querysource-4.2.0 → querysource-4.2.1}/querysource/auth/_resource_types.py +0 -0
  68. {querysource-4.2.0 → querysource-4.2.1}/querysource/auth/credentials.py +0 -0
  69. {querysource-4.2.0 → querysource-4.2.1}/querysource/auth/pbac.py +0 -0
  70. {querysource-4.2.0 → querysource-4.2.1}/querysource/cache/__init__.py +0 -0
  71. {querysource-4.2.0 → querysource-4.2.1}/querysource/cache/backends/__init__.py +0 -0
  72. {querysource-4.2.0 → querysource-4.2.1}/querysource/cache/backends/abstract.py +0 -0
  73. {querysource-4.2.0 → querysource-4.2.1}/querysource/cache/backends/memcache.py +0 -0
  74. {querysource-4.2.0 → querysource-4.2.1}/querysource/cache/backends/redis.py +0 -0
  75. {querysource-4.2.0 → querysource-4.2.1}/querysource/cache/base.py +0 -0
  76. {querysource-4.2.0 → querysource-4.2.1}/querysource/conf.py +0 -0
  77. {querysource-4.2.0 → querysource-4.2.1}/querysource/connections.py +0 -0
  78. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/__init__.py +0 -0
  79. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/__init__.py +0 -0
  80. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/abstract.py +0 -0
  81. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/arangodb.py +0 -0
  82. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/athena.py +0 -0
  83. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/bigquery.py +0 -0
  84. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/cassandra.py +0 -0
  85. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/clickhouse.py +0 -0
  86. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/cockroachdb.py +0 -0
  87. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/couchbase.py +0 -0
  88. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/couchdb.py +0 -0
  89. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/countries.py +0 -0
  90. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/delta.py +0 -0
  91. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/documentdb.py +0 -0
  92. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/dynamodb.py +0 -0
  93. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/elastic.py +0 -0
  94. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/ga.py +0 -0
  95. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/gcalc.py +0 -0
  96. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/hazel.py +0 -0
  97. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/iceberg.py +0 -0
  98. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/influx.py +0 -0
  99. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/jdbc.py +0 -0
  100. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/jira.py +0 -0
  101. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/mariadb.py +0 -0
  102. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/memcached.py +0 -0
  103. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/mongo.py +0 -0
  104. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/mysql.py +0 -0
  105. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/odbc.py +0 -0
  106. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/openweather.py +0 -0
  107. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/oracle.py +0 -0
  108. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/pg.py +0 -0
  109. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/pg_admin.py +0 -0
  110. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/postgres.py +0 -0
  111. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/py.typed +0 -0
  112. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/qs.py +0 -0
  113. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/redis.py +0 -0
  114. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/rest.py +0 -0
  115. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/rethink.py +0 -0
  116. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/sa.py +0 -0
  117. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/salesforce.py +0 -0
  118. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/scylladb.py +0 -0
  119. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/sqlalchemy.py +0 -0
  120. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/sqlite.py +0 -0
  121. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/sqlserver.py +0 -0
  122. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/upc.py +0 -0
  123. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/drivers/zipcodeapi.py +0 -0
  124. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/handlers/__init__.py +0 -0
  125. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/handlers/utils.py +0 -0
  126. {querysource-4.2.0 → querysource-4.2.1}/querysource/datasources/models.py +0 -0
  127. {querysource-4.2.0 → querysource-4.2.1}/querysource/events/__init__.py +0 -0
  128. {querysource-4.2.0 → querysource-4.2.1}/querysource/exceptions.py +0 -0
  129. {querysource-4.2.0 → querysource-4.2.1}/querysource/handlers/__init__.py +0 -0
  130. {querysource-4.2.0 → querysource-4.2.1}/querysource/handlers/_pagination.py +0 -0
  131. {querysource-4.2.0 → querysource-4.2.1}/querysource/handlers/abstract.py +0 -0
  132. {querysource-4.2.0 → querysource-4.2.1}/querysource/handlers/executor.py +0 -0
  133. {querysource-4.2.0 → querysource-4.2.1}/querysource/handlers/log.py +0 -0
  134. {querysource-4.2.0 → querysource-4.2.1}/querysource/handlers/multi.py +0 -0
  135. {querysource-4.2.0 → querysource-4.2.1}/querysource/handlers/outputs/__init__.py +0 -0
  136. {querysource-4.2.0 → querysource-4.2.1}/querysource/handlers/outputs/tableOutput/__init__.py +0 -0
  137. {querysource-4.2.0 → querysource-4.2.1}/querysource/handlers/outputs/tableOutput/postgres.py +0 -0
  138. {querysource-4.2.0 → querysource-4.2.1}/querysource/handlers/outputs/tableOutput/table.py +0 -0
  139. {querysource-4.2.0 → querysource-4.2.1}/querysource/handlers/service.py +0 -0
  140. {querysource-4.2.0 → querysource-4.2.1}/querysource/handlers/variables.py +0 -0
  141. {querysource-4.2.0 → querysource-4.2.1}/querysource/interfaces/__init__.py +0 -0
  142. {querysource-4.2.0 → querysource-4.2.1}/querysource/interfaces/credentials.py +0 -0
  143. {querysource-4.2.0 → querysource-4.2.1}/querysource/interfaces/databases/__init__.py +0 -0
  144. {querysource-4.2.0 → querysource-4.2.1}/querysource/interfaces/databases/abstract.py +0 -0
  145. {querysource-4.2.0 → querysource-4.2.1}/querysource/interfaces/databases/bigquery.py +0 -0
  146. {querysource-4.2.0 → querysource-4.2.1}/querysource/interfaces/databases/db.py +0 -0
  147. {querysource-4.2.0 → querysource-4.2.1}/querysource/interfaces/databases/mongo.py +0 -0
  148. {querysource-4.2.0 → querysource-4.2.1}/querysource/interfaces/databases/rethink.py +0 -0
  149. {querysource-4.2.0 → querysource-4.2.1}/querysource/interfaces/http.py +0 -0
  150. {querysource-4.2.0 → querysource-4.2.1}/querysource/interfaces/playwright_service.py +0 -0
  151. {querysource-4.2.0 → querysource-4.2.1}/querysource/interfaces/queries.py +0 -0
  152. {querysource-4.2.0 → querysource-4.2.1}/querysource/interfaces/selenium_service.py +0 -0
  153. {querysource-4.2.0 → querysource-4.2.1}/querysource/libs/__init__.py +0 -0
  154. {querysource-4.2.0 → querysource-4.2.1}/querysource/libs/encoders.py +0 -0
  155. {querysource-4.2.0 → querysource-4.2.1}/querysource/libs/functions/__init__.py +0 -0
  156. {querysource-4.2.0 → querysource-4.2.1}/querysource/libs/py.typed +0 -0
  157. {querysource-4.2.0 → querysource-4.2.1}/querysource/models.py +0 -0
  158. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/__init__.py +0 -0
  159. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/dt/__init__.py +0 -0
  160. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/dt/abstract.py +0 -0
  161. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/dt/arrow.py +0 -0
  162. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/dt/dt.py +0 -0
  163. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/dt/factory.py +0 -0
  164. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/dt/iter.py +0 -0
  165. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/dt/modin.py +0 -0
  166. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/dt/pandas.py +0 -0
  167. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/dt/polars.py +0 -0
  168. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/output.py +0 -0
  169. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/tables/TableOutput/__init__.py +0 -0
  170. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/tables/TableOutput/abstract.py +0 -0
  171. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/tables/TableOutput/bigquery.py +0 -0
  172. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/tables/TableOutput/documentdb.py +0 -0
  173. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/tables/TableOutput/mongodb.py +0 -0
  174. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/tables/TableOutput/mysql.py +0 -0
  175. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/tables/TableOutput/postgres.py +0 -0
  176. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/tables/TableOutput/rethink.py +0 -0
  177. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/tables/TableOutput/sa.py +0 -0
  178. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/tables/TableOutput/table.py +0 -0
  179. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/tables/__init__.py +0 -0
  180. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/writers/__init__.py +0 -0
  181. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/writers/abstract.py +0 -0
  182. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/writers/bokeh.py +0 -0
  183. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/writers/clustering.py +0 -0
  184. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/writers/csv.py +0 -0
  185. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/writers/describe.py +0 -0
  186. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/writers/eda.py +0 -0
  187. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/writers/excel.py +0 -0
  188. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/writers/html.py +0 -0
  189. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/writers/json.py +0 -0
  190. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/writers/pdf.py +0 -0
  191. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/writers/pickle.py +0 -0
  192. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/writers/plotly.py +0 -0
  193. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/writers/profiling.py +0 -0
  194. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/writers/report.py +0 -0
  195. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/writers/table.py +0 -0
  196. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/writers/tsv.py +0 -0
  197. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/writers/txt.py +0 -0
  198. {querysource-4.2.0 → querysource-4.2.1}/querysource/outputs/writers/xml.py +0 -0
  199. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/__init__.py +0 -0
  200. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/abstract.c +0 -0
  201. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/abstract.pxd +0 -0
  202. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/abstract.pyx +0 -0
  203. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/arangodb.c +0 -0
  204. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/arangodb.pxd +0 -0
  205. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/arangodb.pyx +0 -0
  206. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/bigquery.c +0 -0
  207. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/bigquery.pxd +0 -0
  208. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/bigquery.pyx +0 -0
  209. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/cql.c +0 -0
  210. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/cql.pxd +0 -0
  211. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/cql.pyx +0 -0
  212. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/deltatbl.c +0 -0
  213. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/deltatbl.pxd +0 -0
  214. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/deltatbl.pyx +0 -0
  215. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/elastic.cpp +0 -0
  216. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/elastic.pxd +0 -0
  217. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/elastic.pyx +0 -0
  218. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/iceberg.c +0 -0
  219. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/iceberg.pxd +0 -0
  220. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/iceberg.pyx +0 -0
  221. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/influx.c +0 -0
  222. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/influx.pxd +0 -0
  223. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/influx.pyx +0 -0
  224. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/mongo.cpp +0 -0
  225. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/mongo.pxd +0 -0
  226. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/mongo.pyx +0 -0
  227. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/parser.c +0 -0
  228. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/parser.pxd +0 -0
  229. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/parser.pyx +0 -0
  230. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/pgsql.c +0 -0
  231. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/pgsql.pxd +0 -0
  232. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/pgsql.pyx +0 -0
  233. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/rethink.c +0 -0
  234. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/rethink.pxd +0 -0
  235. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/rethink.pyx +0 -0
  236. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/sosql.c +0 -0
  237. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/sosql.pxd +0 -0
  238. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/sosql.pyx +0 -0
  239. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/sql.c +0 -0
  240. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/sql.pxd +0 -0
  241. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/sql.pyx +0 -0
  242. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/sqlserver.c +0 -0
  243. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/sqlserver.pxd +0 -0
  244. {querysource-4.2.0 → querysource-4.2.1}/querysource/parsers/sqlserver.pyx +0 -0
  245. {querysource-4.2.0 → querysource-4.2.1}/querysource/plugins/__init__.py +0 -0
  246. {querysource-4.2.0 → querysource-4.2.1}/querysource/plugins/importer.py +0 -0
  247. {querysource-4.2.0 → querysource-4.2.1}/querysource/plugins/sources/__init__.py +0 -0
  248. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/__init__.py +0 -0
  249. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/abstract.py +0 -0
  250. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/arangodb.py +0 -0
  251. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/bigquery.py +0 -0
  252. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/cassandra.py +0 -0
  253. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/db.py +0 -0
  254. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/default.py +0 -0
  255. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/deltatbl.py +0 -0
  256. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/documentdb.py +0 -0
  257. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/dummy.py +0 -0
  258. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/elastic.py +0 -0
  259. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/external.py +0 -0
  260. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/http.py +0 -0
  261. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/iceberg.py +0 -0
  262. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/influx.py +0 -0
  263. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/mysql.py +0 -0
  264. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/pg.py +0 -0
  265. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/py.typed +0 -0
  266. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/rest.py +0 -0
  267. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/rethink.py +0 -0
  268. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/salesforce.py +0 -0
  269. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/scylladb.py +0 -0
  270. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/sources/__init__.py +0 -0
  271. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/sources/abstract.py +0 -0
  272. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/sources/amazon.py +0 -0
  273. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/sources/countries.py +0 -0
  274. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/sources/ga.py +0 -0
  275. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/sources/geofcc.py +0 -0
  276. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/sources/gmaps.py +0 -0
  277. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/sources/graphcountries.py +0 -0
  278. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/sources/graphql.py +0 -0
  279. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/sources/http.py +0 -0
  280. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/sources/hubspot.py +0 -0
  281. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/sources/openweather.py +0 -0
  282. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/sources/parsers/__init__.py +0 -0
  283. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/sources/parsers/amproduct.py +0 -0
  284. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/sources/parsers/xpath.py +0 -0
  285. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/sources/pokemon.py +0 -0
  286. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/sources/populartimes.py +0 -0
  287. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/sources/py.typed +0 -0
  288. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/sources/rest.py +0 -0
  289. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/sources/retailnext.py +0 -0
  290. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/sources/rssapp.py +0 -0
  291. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/sources/salesforce.py +0 -0
  292. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/sources/scrapper.py +0 -0
  293. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/sources/shoppertrack.py +0 -0
  294. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/sources/swop.py +0 -0
  295. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/sources/uap.py +0 -0
  296. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/sources/upc.py +0 -0
  297. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/sources/wm_stores.py +0 -0
  298. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/sources/zammad.py +0 -0
  299. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/sources/zipcodeapi.py +0 -0
  300. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/sql.py +0 -0
  301. {querysource-4.2.0 → querysource-4.2.1}/querysource/providers/sqlserver.py +0 -0
  302. {querysource-4.2.0 → querysource-4.2.1}/querysource/py.typed +0 -0
  303. {querysource-4.2.0 → querysource-4.2.1}/querysource/qs_parsers/__init__.py +0 -0
  304. {querysource-4.2.0 → querysource-4.2.1}/querysource/queries/__init__.py +0 -0
  305. {querysource-4.2.0 → querysource-4.2.1}/querysource/queries/base.py +0 -0
  306. {querysource-4.2.0 → querysource-4.2.1}/querysource/queries/executor.py +0 -0
  307. {querysource-4.2.0 → querysource-4.2.1}/querysource/queries/models.py +0 -0
  308. {querysource-4.2.0 → querysource-4.2.1}/querysource/queries/multi/__init__.py +0 -0
  309. {querysource-4.2.0 → querysource-4.2.1}/querysource/queries/multi/components/__init__.py +0 -0
  310. {querysource-4.2.0 → querysource-4.2.1}/querysource/queries/multi/operators/Concat.py +0 -0
  311. {querysource-4.2.0 → querysource-4.2.1}/querysource/queries/multi/operators/GroupBy.py +0 -0
  312. {querysource-4.2.0 → querysource-4.2.1}/querysource/queries/multi/operators/Info.py +0 -0
  313. {querysource-4.2.0 → querysource-4.2.1}/querysource/queries/multi/operators/Join.py +0 -0
  314. {querysource-4.2.0 → querysource-4.2.1}/querysource/queries/multi/operators/Melt.py +0 -0
  315. {querysource-4.2.0 → querysource-4.2.1}/querysource/queries/multi/operators/Merge.py +0 -0
  316. {querysource-4.2.0 → querysource-4.2.1}/querysource/queries/multi/operators/__init__.py +0 -0
  317. {querysource-4.2.0 → querysource-4.2.1}/querysource/queries/multi/operators/abstract.py +0 -0
  318. {querysource-4.2.0 → querysource-4.2.1}/querysource/queries/multi/operators/filter/__init__.py +0 -0
  319. {querysource-4.2.0 → querysource-4.2.1}/querysource/queries/multi/operators/filter/flt.py +0 -0
  320. {querysource-4.2.0 → querysource-4.2.1}/querysource/queries/multi/sources/__init__.py +0 -0
  321. {querysource-4.2.0 → querysource-4.2.1}/querysource/queries/multi/sources/file.py +0 -0
  322. {querysource-4.2.0 → querysource-4.2.1}/querysource/queries/multi/sources/query.py +0 -0
  323. {querysource-4.2.0 → querysource-4.2.1}/querysource/queries/multi/transformations/Forecast.py +0 -0
  324. {querysource-4.2.0 → querysource-4.2.1}/querysource/queries/multi/transformations/Map.py +0 -0
  325. {querysource-4.2.0 → querysource-4.2.1}/querysource/queries/multi/transformations/__init__.py +0 -0
  326. {querysource-4.2.0 → querysource-4.2.1}/querysource/queries/multi/transformations/abstract.py +0 -0
  327. {querysource-4.2.0 → querysource-4.2.1}/querysource/queries/multi/transformations/correlation.py +0 -0
  328. {querysource-4.2.0 → querysource-4.2.1}/querysource/queries/multi/transformations/crosstab.py +0 -0
  329. {querysource-4.2.0 → querysource-4.2.1}/querysource/queries/multi/transformations/google/__init__.py +0 -0
  330. {querysource-4.2.0 → querysource-4.2.1}/querysource/queries/multi/transformations/google/maps.py +0 -0
  331. {querysource-4.2.0 → querysource-4.2.1}/querysource/queries/multi/transformations/pivot.py +0 -0
  332. {querysource-4.2.0 → querysource-4.2.1}/querysource/queries/multi/transformations/tOrder.py +0 -0
  333. {querysource-4.2.0 → querysource-4.2.1}/querysource/queries/multi/transformations/tPandas.py +0 -0
  334. {querysource-4.2.0 → querysource-4.2.1}/querysource/queries/obj.py +0 -0
  335. {querysource-4.2.0 → querysource-4.2.1}/querysource/queries/qs.py +0 -0
  336. {querysource-4.2.0 → querysource-4.2.1}/querysource/scheduler/__init__.py +0 -0
  337. {querysource-4.2.0 → querysource-4.2.1}/querysource/scheduler/jobs.py +0 -0
  338. {querysource-4.2.0 → querysource-4.2.1}/querysource/scheduler/notifications.py +0 -0
  339. {querysource-4.2.0 → querysource-4.2.1}/querysource/scheduler/scheduler.py +0 -0
  340. {querysource-4.2.0 → querysource-4.2.1}/querysource/services.py +0 -0
  341. {querysource-4.2.0 → querysource-4.2.1}/querysource/template/__init__.py +0 -0
  342. {querysource-4.2.0 → querysource-4.2.1}/querysource/template/parser.py +0 -0
  343. {querysource-4.2.0 → querysource-4.2.1}/querysource/types/__init__.py +0 -0
  344. {querysource-4.2.0 → querysource-4.2.1}/querysource/types/converters.cpp +0 -0
  345. {querysource-4.2.0 → querysource-4.2.1}/querysource/types/converters.pyx +0 -0
  346. {querysource-4.2.0 → querysource-4.2.1}/querysource/types/dt/__init__.py +0 -0
  347. {querysource-4.2.0 → querysource-4.2.1}/querysource/types/dt/filters.py +0 -0
  348. {querysource-4.2.0 → querysource-4.2.1}/querysource/types/dt/transforms.py +0 -0
  349. {querysource-4.2.0 → querysource-4.2.1}/querysource/types/py.typed +0 -0
  350. {querysource-4.2.0 → querysource-4.2.1}/querysource/types/typedefs.c +0 -0
  351. {querysource-4.2.0 → querysource-4.2.1}/querysource/types/typedefs.pyx +0 -0
  352. {querysource-4.2.0 → querysource-4.2.1}/querysource/types/validators.cpp +0 -0
  353. {querysource-4.2.0 → querysource-4.2.1}/querysource/types/validators.pyx +0 -0
  354. {querysource-4.2.0 → querysource-4.2.1}/querysource/utils/__init__.py +0 -0
  355. {querysource-4.2.0 → querysource-4.2.1}/querysource/utils/cache_serialization.py +0 -0
  356. {querysource-4.2.0 → querysource-4.2.1}/querysource/utils/events.py +0 -0
  357. {querysource-4.2.0 → querysource-4.2.1}/querysource/utils/fn.py +0 -0
  358. {querysource-4.2.0 → querysource-4.2.1}/querysource/utils/functions.cpp +0 -0
  359. {querysource-4.2.0 → querysource-4.2.1}/querysource/utils/functions.pyx +0 -0
  360. {querysource-4.2.0 → querysource-4.2.1}/querysource/utils/getfunc.py +0 -0
  361. {querysource-4.2.0 → querysource-4.2.1}/querysource/utils/handlers.py +0 -0
  362. {querysource-4.2.0 → querysource-4.2.1}/querysource/utils/parseqs.cpp +0 -0
  363. {querysource-4.2.0 → querysource-4.2.1}/querysource/utils/parseqs.pyx +0 -0
  364. {querysource-4.2.0 → querysource-4.2.1}/querysource/utils/validators.py +0 -0
  365. {querysource-4.2.0 → querysource-4.2.1}/querysource.egg-info/dependency_links.txt +0 -0
  366. {querysource-4.2.0 → querysource-4.2.1}/querysource.egg-info/entry_points.txt +0 -0
  367. {querysource-4.2.0 → querysource-4.2.1}/querysource.egg-info/not-zip-safe +0 -0
  368. {querysource-4.2.0 → querysource-4.2.1}/querysource.egg-info/requires.txt +0 -0
  369. {querysource-4.2.0 → querysource-4.2.1}/querysource.egg-info/top_level.txt +0 -0
  370. {querysource-4.2.0 → querysource-4.2.1}/run.py +0 -0
  371. {querysource-4.2.0 → querysource-4.2.1}/rust/Cargo.lock +0 -0
  372. {querysource-4.2.0 → querysource-4.2.1}/rust/Cargo.toml +0 -0
  373. {querysource-4.2.0 → querysource-4.2.1}/rust/pyproject.toml +0 -0
  374. {querysource-4.2.0 → querysource-4.2.1}/rust/src/arangodb_parser.rs +0 -0
  375. {querysource-4.2.0 → querysource-4.2.1}/rust/src/bigquery_parser.rs +0 -0
  376. {querysource-4.2.0 → querysource-4.2.1}/rust/src/cql_parser.rs +0 -0
  377. {querysource-4.2.0 → querysource-4.2.1}/rust/src/elastic_parser.rs +0 -0
  378. {querysource-4.2.0 → querysource-4.2.1}/rust/src/filter_common.rs +0 -0
  379. {querysource-4.2.0 → querysource-4.2.1}/rust/src/flux_parser.rs +0 -0
  380. {querysource-4.2.0 → querysource-4.2.1}/rust/src/lib.rs +0 -0
  381. {querysource-4.2.0 → querysource-4.2.1}/rust/src/mongo_parser.rs +0 -0
  382. {querysource-4.2.0 → querysource-4.2.1}/rust/src/mssql_parser.rs +0 -0
  383. {querysource-4.2.0 → querysource-4.2.1}/rust/src/parseqs.rs +0 -0
  384. {querysource-4.2.0 → querysource-4.2.1}/rust/src/pgsql_parser.rs +0 -0
  385. {querysource-4.2.0 → querysource-4.2.1}/rust/src/rethink_parser.rs +0 -0
  386. {querysource-4.2.0 → querysource-4.2.1}/rust/src/safe_dict.rs +0 -0
  387. {querysource-4.2.0 → querysource-4.2.1}/rust/src/soql_parser.rs +0 -0
  388. {querysource-4.2.0 → querysource-4.2.1}/rust/src/sql_parser.rs +0 -0
  389. {querysource-4.2.0 → querysource-4.2.1}/rust/src/validators.rs +0 -0
  390. {querysource-4.2.0 → querysource-4.2.1}/setup.cfg +0 -0
  391. {querysource-4.2.0 → querysource-4.2.1}/setup.py +0 -0
  392. {querysource-4.2.0 → querysource-4.2.1}/static/notebook/bundle.js +0 -0
  393. {querysource-4.2.0 → querysource-4.2.1}/templates/__init__.py +0 -0
  394. {querysource-4.2.0 → querysource-4.2.1}/templates/base.html +0 -0
  395. {querysource-4.2.0 → querysource-4.2.1}/templates/default.html +0 -0
  396. {querysource-4.2.0 → querysource-4.2.1}/templates/default_table.html +0 -0
  397. {querysource-4.2.0 → querysource-4.2.1}/templates/fontlist-v330.json +0 -0
  398. {querysource-4.2.0 → querysource-4.2.1}/templates/fontlist-v390.json +0 -0
  399. {querysource-4.2.0 → querysource-4.2.1}/templates/table_charts.html +0 -0
  400. {querysource-4.2.0 → querysource-4.2.1}/tests/auth/__init__.py +0 -0
  401. {querysource-4.2.0 → querysource-4.2.1}/tests/auth/test_credentials.py +0 -0
  402. {querysource-4.2.0 → querysource-4.2.1}/tests/auth/test_pbac_bootstrap.py +0 -0
  403. {querysource-4.2.0 → querysource-4.2.1}/tests/conftest.py +0 -0
  404. {querysource-4.2.0 → querysource-4.2.1}/tests/datasources/__init__.py +0 -0
  405. {querysource-4.2.0 → querysource-4.2.1}/tests/datasources/test_datasource_view_pbac.py +0 -0
  406. {querysource-4.2.0 → querysource-4.2.1}/tests/datasources/test_driver_factory_session.py +0 -0
  407. {querysource-4.2.0 → querysource-4.2.1}/tests/datasources/test_pg_admin_registration.py +0 -0
  408. {querysource-4.2.0 → querysource-4.2.1}/tests/datasources/test_pg_params_for.py +0 -0
  409. {querysource-4.2.0 → querysource-4.2.1}/tests/handlers/__init__.py +0 -0
  410. {querysource-4.2.0 → querysource-4.2.1}/tests/handlers/conftest.py +0 -0
  411. {querysource-4.2.0 → querysource-4.2.1}/tests/handlers/test_abstract_pbac_helpers.py +0 -0
  412. {querysource-4.2.0 → querysource-4.2.1}/tests/handlers/test_multiquery_pbac_smoke.py +0 -0
  413. {querysource-4.2.0 → querysource-4.2.1}/tests/handlers/test_queryexecutor_pbac_smoke.py +0 -0
  414. {querysource-4.2.0 → querysource-4.2.1}/tests/handlers/test_querymanager_pagination.py +0 -0
  415. {querysource-4.2.0 → querysource-4.2.1}/tests/handlers/test_queryservice_pbac_smoke.py +0 -0
  416. {querysource-4.2.0 → querysource-4.2.1}/tests/integration/__init__.py +0 -0
  417. {querysource-4.2.0 → querysource-4.2.1}/tests/integration/test_pbac_credentials.py +0 -0
  418. {querysource-4.2.0 → querysource-4.2.1}/tests/integration/test_pbac_enforcement.py +0 -0
  419. {querysource-4.2.0 → querysource-4.2.1}/tests/integration/test_pbac_listing.py +0 -0
  420. {querysource-4.2.0 → querysource-4.2.1}/tests/perf/__init__.py +0 -0
  421. {querysource-4.2.0 → querysource-4.2.1}/tests/perf/test_pbac_overhead.py +0 -0
  422. {querysource-4.2.0 → querysource-4.2.1}/tests/policies/__init__.py +0 -0
  423. {querysource-4.2.0 → querysource-4.2.1}/tests/policies/test_default_policies_load.py +0 -0
  424. {querysource-4.2.0 → querysource-4.2.1}/tests/services/__init__.py +0 -0
  425. {querysource-4.2.0 → querysource-4.2.1}/tests/services/test_querysource_setup_pbac.py +0 -0
  426. {querysource-4.2.0 → querysource-4.2.1}/tests/test_api.py +0 -0
  427. {querysource-4.2.0 → querysource-4.2.1}/tests/test_arangodb_parser.py +0 -0
  428. {querysource-4.2.0 → querysource-4.2.1}/tests/test_column_filters.py +0 -0
  429. {querysource-4.2.0 → querysource-4.2.1}/tests/test_elastic_parser.py +0 -0
  430. {querysource-4.2.0 → querysource-4.2.1}/tests/test_eval.py +0 -0
  431. {querysource-4.2.0 → querysource-4.2.1}/tests/test_join_conditions.py +0 -0
  432. {querysource-4.2.0 → querysource-4.2.1}/tests/test_join_with_column_filter.py +0 -0
  433. {querysource-4.2.0 → querysource-4.2.1}/tests/test_rss.py +0 -0
  434. {querysource-4.2.0 → querysource-4.2.1}/tests/test_rust_parsers.py +0 -0
  435. {querysource-4.2.0 → querysource-4.2.1}/tests/test_scheduler_core.py +0 -0
  436. {querysource-4.2.0 → querysource-4.2.1}/tests/test_scheduler_integration.py +0 -0
  437. {querysource-4.2.0 → querysource-4.2.1}/tests/test_scheduler_jobs.py +0 -0
  438. {querysource-4.2.0 → querysource-4.2.1}/tests/test_scheduler_notifications.py +0 -0
  439. {querysource-4.2.0 → querysource-4.2.1}/tests/test_sql_parser_combinations.py +0 -0
  440. {querysource-4.2.0 → querysource-4.2.1}/tox.ini +0 -0
@@ -0,0 +1,499 @@
1
+ # Join Conditions Feature
2
+
3
+ ## Overview
4
+
5
+ **Join Conditions** allow you to specify column-to-column comparison conditions directly in the Join operator, without needing a separate Filter operator.
6
+
7
+ This is syntactically cleaner and semantically clearer than Join + Filter for complex join scenarios.
8
+
9
+ ---
10
+
11
+ ## Problem It Solves
12
+
13
+ ### Before (Join + Filter Approach)
14
+ ```json
15
+ {
16
+ "Join": {
17
+ "left": "calls",
18
+ "right": "podcasts",
19
+ "on": "usuario_id"
20
+ },
21
+ "Filter": {
22
+ "filter": [
23
+ {
24
+ "column": "call_date",
25
+ "expression": ">=",
26
+ "value": {"$column": "podcast_date"}
27
+ }
28
+ ]
29
+ }
30
+ }
31
+ ```
32
+
33
+ **Issues:**
34
+ - Requires 2 operators
35
+ - Intent is unclear (is the filter part of the join or applied after?)
36
+ - Harder to optimize
37
+
38
+ ### After (Join with Conditions)
39
+ ```json
40
+ {
41
+ "Join": {
42
+ "left": "calls",
43
+ "right": "podcasts",
44
+ "on": "usuario_id",
45
+ "join_conditions": [
46
+ {
47
+ "left": "call_date",
48
+ "expression": ">=",
49
+ "right": "podcast_date"
50
+ }
51
+ ]
52
+ }
53
+ }
54
+ ```
55
+
56
+ **Benefits:**
57
+ - Single operator
58
+ - Intent is crystal clear
59
+ - Conditions are semantically part of the join
60
+ - Easier to understand and maintain
61
+
62
+ ---
63
+
64
+ ## Syntax
65
+
66
+ ### Basic Structure
67
+
68
+ ```json
69
+ {
70
+ "Join": {
71
+ "left": "table1",
72
+ "right": "table2",
73
+ "on": "join_key",
74
+ "join_conditions": [
75
+ {
76
+ "left": "column_from_table1",
77
+ "expression": ">=",
78
+ "right": "column_from_table2"
79
+ }
80
+ ]
81
+ }
82
+ }
83
+ ```
84
+
85
+ ### Parameters
86
+
87
+ | Parameter | Type | Description |
88
+ |-----------|------|-------------|
89
+ | **left** | string | Name of left table (required) |
90
+ | **right** | string | Name of right table (required) |
91
+ | **on** | string \| list | Join key(s) - same as standard join |
92
+ | **join_conditions** | array | Array of join condition objects |
93
+ | **type** | string | Join type: `inner`, `left`, `right`, `outer` (default: `inner`) |
94
+
95
+ ### Join Condition Object
96
+
97
+ ```json
98
+ {
99
+ "left": "column_name_from_left_table",
100
+ "expression": ">=",
101
+ "right": "column_name_from_right_table"
102
+ }
103
+ ```
104
+
105
+ | Field | Type | Description |
106
+ |-------|------|-------------|
107
+ | **left** | string | Column from left table |
108
+ | **expression** | string | Comparison operator: `>`, `>=`, `<`, `<=`, `==`, `!=` |
109
+ | **right** | string | Column from right table |
110
+
111
+ ---
112
+
113
+ ## Supported Operators
114
+
115
+ All comparison operators work with join_conditions:
116
+
117
+ - **Numeric**: `>`, `>=`, `<`, `<=`, `==`, `!=`
118
+ - **String**: `==`, `!=` (others via string methods not yet supported in join context)
119
+ - **Date**: `>`, `>=`, `<`, `<=`, `==`, `!=`
120
+
121
+ ---
122
+
123
+ ## Examples
124
+
125
+ ### Example 1: Simple Date Comparison
126
+
127
+ ```json
128
+ {
129
+ "queries": {
130
+ "calls": {"slug": "get-calls"},
131
+ "podcasts": {"slug": "get-podcasts"}
132
+ },
133
+ "Join": {
134
+ "left": "calls",
135
+ "right": "podcasts",
136
+ "on": "usuario_id",
137
+ "join_conditions": [
138
+ {
139
+ "left": "call_date",
140
+ "expression": ">=",
141
+ "right": "podcast_date"
142
+ }
143
+ ]
144
+ }
145
+ }
146
+ ```
147
+
148
+ **SQL Equivalent:**
149
+ ```sql
150
+ SELECT *
151
+ FROM calls c
152
+ INNER JOIN podcasts p
153
+ ON c.usuario_id = p.usuario_id
154
+ AND c.call_date >= p.podcast_date
155
+ ```
156
+
157
+ ---
158
+
159
+ ### Example 2: Multiple Join Conditions (AND Logic)
160
+
161
+ ```json
162
+ {
163
+ "Join": {
164
+ "left": "calls",
165
+ "right": "podcasts",
166
+ "on": "usuario_id",
167
+ "join_conditions": [
168
+ {
169
+ "left": "call_date",
170
+ "expression": ">=",
171
+ "right": "podcast_date"
172
+ },
173
+ {
174
+ "left": "call_duration",
175
+ "expression": "<",
176
+ "right": "podcast_duration"
177
+ }
178
+ ]
179
+ }
180
+ }
181
+ ```
182
+
183
+ **SQL Equivalent:**
184
+ ```sql
185
+ SELECT *
186
+ FROM calls c
187
+ INNER JOIN podcasts p
188
+ ON c.usuario_id = p.usuario_id
189
+ AND c.call_date >= p.podcast_date
190
+ AND c.call_duration < p.podcast_duration
191
+ ```
192
+
193
+ ---
194
+
195
+ ### Example 3: Range Checking Between Columns
196
+
197
+ ```json
198
+ {
199
+ "Join": {
200
+ "left": "orders",
201
+ "right": "price_tiers",
202
+ "on": "product_id",
203
+ "join_conditions": [
204
+ {
205
+ "left": "order_amount",
206
+ "expression": ">=",
207
+ "right": "min_price"
208
+ },
209
+ {
210
+ "left": "order_amount",
211
+ "expression": "<=",
212
+ "right": "max_price"
213
+ }
214
+ ]
215
+ }
216
+ }
217
+ ```
218
+
219
+ **SQL Equivalent:**
220
+ ```sql
221
+ SELECT *
222
+ FROM orders o
223
+ INNER JOIN price_tiers pt
224
+ ON o.product_id = pt.product_id
225
+ AND o.order_amount >= pt.min_price
226
+ AND o.order_amount <= pt.max_price
227
+ ```
228
+
229
+ ---
230
+
231
+ ### Example 4: All Join Types with Conditions
232
+
233
+ ```json
234
+ {
235
+ "Join": {
236
+ "left": "calls",
237
+ "right": "podcasts",
238
+ "on": "usuario_id",
239
+ "type": "left",
240
+ "join_conditions": [
241
+ {
242
+ "left": "call_date",
243
+ "expression": ">",
244
+ "right": "podcast_date"
245
+ }
246
+ ]
247
+ }
248
+ }
249
+ ```
250
+
251
+ Join types supported:
252
+ - `inner` - Only matching rows with condition satisfied
253
+ - `left` - All from left, matching from right where condition satisfied
254
+ - `right` - Matching from left, all from right where condition satisfied
255
+ - `outer` - All rows from both where condition satisfied or one side is NULL
256
+
257
+ ---
258
+
259
+ ## How It Works
260
+
261
+ ### Execution Flow
262
+
263
+ ```
264
+ 1. Merge on join key(s) using specified join type
265
+ └─► Intermediate result (before conditions)
266
+
267
+ 2. Apply join_conditions to filtered result
268
+ └─► Reuses Filter expression builder
269
+ └─► AND logic combines all conditions
270
+ └─► Final result
271
+
272
+ 3. Return joined and conditioned DataFrame
273
+ ```
274
+
275
+ ### Under the Hood
276
+
277
+ Join conditions internally reuse the `create_filter()` function from the Filter module:
278
+
279
+ ```python
280
+ # Join condition:
281
+ {"left": "col_a", "expression": ">=", "right": "col_b"}
282
+
283
+ # Converted to Filter format:
284
+ {"column": "col_a", "expression": ">=", "value": {"$column": "col_b"}}
285
+
286
+ # Applied via:
287
+ df = df.loc[eval("(df['col_a'] >= df['col_b'])")]
288
+ ```
289
+
290
+ This ensures **consistent behavior** between Filter and Join conditions.
291
+
292
+ ---
293
+
294
+ ## Performance
295
+
296
+ - **Complexity**: O(n log n) for merge + O(n) for conditions
297
+ - **Memory**: One boolean mask array per condition
298
+ - **Optimization**: Conditions are applied immediately after merge, not in post-processing
299
+
300
+ Join conditions have **no performance penalty** vs Join + Filter approach.
301
+
302
+ ---
303
+
304
+ ## Error Handling
305
+
306
+ ### Missing Required Fields
307
+
308
+ ```json
309
+ {
310
+ "join_conditions": [
311
+ {
312
+ "left": "col_a"
313
+ // Missing: "expression" and "right"
314
+ }
315
+ ]
316
+ }
317
+ ```
318
+
319
+ **Error:**
320
+ ```
321
+ QueryException: Join condition must have 'left', 'expression', and 'right' fields
322
+ ```
323
+
324
+ ### Column Not Found
325
+
326
+ ```json
327
+ {
328
+ "join_conditions": [
329
+ {
330
+ "left": "nonexistent_column",
331
+ "expression": ">=",
332
+ "right": "podcast_date"
333
+ }
334
+ ]
335
+ }
336
+ ```
337
+
338
+ **Error:**
339
+ ```
340
+ QueryException: tFilter: Column nonexistent_column not found in DataFrame.
341
+ ```
342
+
343
+ ### Empty Result After Conditions
344
+
345
+ **Status**: VALID (not an error)
346
+
347
+ If all rows are filtered out by conditions, an empty DataFrame is returned. This is semantically valid - no rows matched the join conditions.
348
+
349
+ ---
350
+
351
+ ## Comparison: Join Conditions vs Filter
352
+
353
+ | Aspect | Join Conditions | Join + Filter |
354
+ |--------|-----------------|---------------|
355
+ | **Semantics** | Conditions are part of join | Filter is post-join |
356
+ | **Operators** | 1 (Join) | 2 (Join + Filter) |
357
+ | **Clarity** | Very clear intent | Slightly ambiguous |
358
+ | **Performance** | Same | Same |
359
+ | **SQL Equivalence** | INNER/LEFT/RIGHT/OUTER JOIN ... ON ... AND ... | SELECT ... WHERE ... |
360
+ | **Use Case** | Explicit join conditions | General filtering |
361
+
362
+ ---
363
+
364
+ ## When to Use
365
+
366
+ ### Use Join Conditions When:
367
+ - ✅ Conditions are logically part of the join
368
+ - ✅ Comparing columns from both tables
369
+ - ✅ Want clear, single-operator syntax
370
+ - ✅ Multiple join conditions (cleaner than sequential filters)
371
+
372
+ ### Use Join + Filter When:
373
+ - ✅ Filter is applied post-join for different reasons
374
+ - ✅ Want to use advanced filter operators (contains, regex, etc.)
375
+ - ✅ Filter logic is complex and deserves its own operator
376
+
377
+ ---
378
+
379
+ ## Complete Example
380
+
381
+ ### Data Setup
382
+
383
+ **calls table:**
384
+ ```
385
+ | id | user_id | date | duration |
386
+ |----|---------|------------|----------|
387
+ | 1 | 100 | 2024-01-15 | 45 |
388
+ | 2 | 101 | 2024-01-20 | 30 |
389
+ ```
390
+
391
+ **podcasts table:**
392
+ ```
393
+ | id | user_id | date | duration |
394
+ |----|---------|------------|----------|
395
+ | 200 | 100 | 2024-01-10 | 90 |
396
+ | 201 | 101 | 2024-01-25 | 45 |
397
+ ```
398
+
399
+ ### Query with Join Conditions
400
+
401
+ ```json
402
+ {
403
+ "queries": {
404
+ "calls": {"slug": "get-calls"},
405
+ "podcasts": {"slug": "get-podcasts"}
406
+ },
407
+ "Join": {
408
+ "left": "calls",
409
+ "right": "podcasts",
410
+ "on": "user_id",
411
+ "type": "inner",
412
+ "join_conditions": [
413
+ {
414
+ "left": "date",
415
+ "expression": ">=",
416
+ "right": "date"
417
+ },
418
+ {
419
+ "left": "duration",
420
+ "expression": "<",
421
+ "right": "duration"
422
+ }
423
+ ]
424
+ }
425
+ }
426
+ ```
427
+
428
+ ### Result
429
+
430
+ ```
431
+ | id (call) | user_id | date (call) | duration (call) | id (podcast) | date (podcast) | duration (podcast) |
432
+ |-----------|---------|-------------|-----------------|--------------|----------------|--------------------|
433
+ | 1 | 100 | 2024-01-15 | 45 | 200 | 2024-01-10 | 90 |
434
+ | 2 | 101 | 2024-01-20 | 30 | 201 | 2024-01-25 | 45 |
435
+ ```
436
+
437
+ (Row 2 included: call_date 2024-01-20 >= podcast_date 2024-01-25? No... wait, let me recalculate)
438
+
439
+ Actually:
440
+ - Row 1: 2024-01-15 >= 2024-01-10 ✓ AND 45 < 90 ✓ → KEEP
441
+ - Row 2: 2024-01-20 >= 2024-01-25 ✗ → DROP
442
+
443
+ **Final Result:**
444
+ ```
445
+ | id | user_id | date | duration | id | date | duration |
446
+ |----|---------|------------|----------|-----|------------|----------|
447
+ | 1 | 100 | 2024-01-15 | 45 | 200 | 2024-01-10 | 90 |
448
+ ```
449
+
450
+ ---
451
+
452
+ ## Integration with MultiQuery
453
+
454
+ Join Conditions are fully integrated into the MultiQuery workflow:
455
+
456
+ ```
457
+ Query Execution
458
+
459
+ Join (with optional conditions)
460
+ ├─ Merge on keys
461
+ └─ Apply conditions
462
+
463
+ Filter (optional, for post-join filtering)
464
+
465
+ GroupBy (optional)
466
+
467
+ Output
468
+ ```
469
+
470
+ ---
471
+
472
+ ## Limitations
473
+
474
+ - ✅ AND logic only (not OR between conditions)
475
+ - Workaround: use separate Join operators in sequence
476
+ - ✅ No aggregate functions in conditions
477
+ - Workaround: pre-compute aggregates in Query
478
+ - ✅ No string pattern matching in join context
479
+ - Workaround: use Filter operator post-join
480
+
481
+ ---
482
+
483
+ ## Testing
484
+
485
+ Tests are included in:
486
+ - `tests/test_join_conditions.py` - 10 comprehensive tests
487
+
488
+ Run with:
489
+ ```bash
490
+ pytest tests/test_join_conditions.py -v
491
+ ```
492
+
493
+ ---
494
+
495
+ ## See Also
496
+
497
+ - [Column Filter Reference](./COLUMN_FILTER_EXAMPLE.md)
498
+ - [Join + Column Filter Integration](./JOIN_AND_COLUMN_FILTER_INTEGRATION.md)
499
+ - [Join Operator Implementation](./querysource/queries/multi/operators/Join.py)