querysource 4.1.12__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.
- querysource-4.2.1/JOIN_CONDITIONS_FEATURE.md +499 -0
- querysource-4.2.1/JOIN_CONDITIONS_SUMMARY.md +403 -0
- {querysource-4.1.12/querysource.egg-info → querysource-4.2.1}/PKG-INFO +1 -1
- {querysource-4.1.12 → querysource-4.2.1}/app.py +9 -6
- querysource-4.2.1/policies/datasources.yaml +24 -0
- querysource-4.2.1/policies/defaults.yaml +31 -0
- querysource-4.2.1/policies/drivers.yaml +24 -0
- querysource-4.2.1/policies/raw_queries.yaml +24 -0
- querysource-4.2.1/policies/slugs.yaml +24 -0
- querysource-4.2.1/policies/superusers.yaml +33 -0
- {querysource-4.1.12 → querysource-4.2.1}/pyproject.toml +6 -0
- {querysource-4.1.12 → querysource-4.2.1}/pytest.ini +2 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/_version.py +3 -3
- querysource-4.2.1/querysource/auth/__init__.py +21 -0
- querysource-4.2.1/querysource/auth/_resource_types.py +48 -0
- querysource-4.2.1/querysource/auth/credentials.py +224 -0
- querysource-4.2.1/querysource/auth/pbac.py +149 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/conf.py +5 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/connections.py +6 -8
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/__init__.py +5 -0
- querysource-4.2.1/querysource/datasources/drivers/pg.py +124 -0
- querysource-4.2.1/querysource/datasources/drivers/pg_admin.py +44 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/postgres.py +1 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/handlers/datasource.py +97 -14
- {querysource-4.1.12 → querysource-4.2.1}/querysource/handlers/abstract.py +141 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/handlers/executor.py +60 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/handlers/manager.py +11 -16
- {querysource-4.1.12 → querysource-4.2.1}/querysource/handlers/multi.py +110 -5
- {querysource-4.1.12 → querysource-4.2.1}/querysource/handlers/service.py +31 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/interfaces/connections.py +68 -22
- {querysource-4.1.12 → querysource-4.2.1}/querysource/interfaces/queries.py +6 -8
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/bigquery.c +0 -1
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/cql.c +170 -191
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/deltatbl.c +176 -197
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/iceberg.c +173 -194
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/pgsql.c +148 -169
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/sosql.c +167 -188
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/sql.c +3111 -2429
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/sql.pxd +0 -1
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/sql.pyx +82 -12
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/sqlserver.c +150 -171
- {querysource-4.1.12 → querysource-4.2.1}/querysource/queries/multi/__init__.py +3 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/queries/obj.py +25 -3
- {querysource-4.1.12 → querysource-4.2.1}/querysource/queries/qs.py +29 -3
- {querysource-4.1.12 → querysource-4.2.1}/querysource/services.py +56 -7
- {querysource-4.1.12 → querysource-4.2.1}/querysource/version.py +1 -1
- {querysource-4.1.12 → querysource-4.2.1/querysource.egg-info}/PKG-INFO +1 -1
- {querysource-4.1.12 → querysource-4.2.1}/querysource.egg-info/SOURCES.txt +40 -1
- {querysource-4.1.12 → querysource-4.2.1}/rust/src/sql_parser.rs +127 -10
- querysource-4.2.1/tests/auth/test_credentials.py +145 -0
- querysource-4.2.1/tests/auth/test_pbac_bootstrap.py +121 -0
- querysource-4.2.1/tests/check_concurrent_get_slug.py +78 -0
- querysource-4.2.1/tests/conftest.py +177 -0
- querysource-4.2.1/tests/datasources/__init__.py +0 -0
- querysource-4.2.1/tests/datasources/test_datasource_view_pbac.py +119 -0
- querysource-4.2.1/tests/datasources/test_driver_factory_session.py +206 -0
- querysource-4.2.1/tests/datasources/test_pg_admin_registration.py +39 -0
- querysource-4.2.1/tests/datasources/test_pg_params_for.py +95 -0
- querysource-4.2.1/tests/handlers/__init__.py +0 -0
- querysource-4.2.1/tests/handlers/test_abstract_pbac_helpers.py +229 -0
- querysource-4.2.1/tests/handlers/test_multiquery_pbac_smoke.py +149 -0
- querysource-4.2.1/tests/handlers/test_queryexecutor_pbac_smoke.py +113 -0
- querysource-4.2.1/tests/handlers/test_queryservice_pbac_smoke.py +75 -0
- querysource-4.2.1/tests/integration/__init__.py +0 -0
- querysource-4.2.1/tests/integration/test_pbac_credentials.py +172 -0
- querysource-4.2.1/tests/integration/test_pbac_enforcement.py +314 -0
- querysource-4.2.1/tests/integration/test_pbac_listing.py +196 -0
- querysource-4.2.1/tests/perf/__init__.py +0 -0
- querysource-4.2.1/tests/perf/test_pbac_overhead.py +183 -0
- querysource-4.2.1/tests/policies/__init__.py +0 -0
- querysource-4.2.1/tests/policies/test_default_policies_load.py +108 -0
- querysource-4.2.1/tests/services/__init__.py +0 -0
- querysource-4.2.1/tests/services/test_querysource_setup_pbac.py +50 -0
- querysource-4.2.1/tests/test_queryslug_concurrency.py +156 -0
- {querysource-4.1.12 → querysource-4.2.1}/tests/test_rust_parsers.py +53 -0
- querysource-4.2.1/tests/test_sql_parser_combinations.py +425 -0
- querysource-4.1.12/querysource/datasources/drivers/pg.py +0 -66
- {querysource-4.1.12 → querysource-4.2.1}/.bumpversion.cfg +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/.claude/agents/code-reviewer.md +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/.claude/agents/sdd-worker.md +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/.claude/commands/pr-review.md +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/.claude/commands/sdd-brainstorm.md +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/.claude/commands/sdd-codereview.md +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/.claude/commands/sdd-done.md +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/.claude/commands/sdd-fromjira.md +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/.claude/commands/sdd-next.md +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/.claude/commands/sdd-proposal.md +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/.claude/commands/sdd-spec.md +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/.claude/commands/sdd-start.md +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/.claude/commands/sdd-status.md +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/.claude/commands/sdd-task.md +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/.claude/commands/sdd-tojira.md +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/.claude/rules/aws-cost-optimization.md +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/.claude/rules/code-reviewer.md +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/.claude/rules/cython-development.md +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/.claude/rules/python-development.md +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/.claude/rules/rust-development.md +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/.claude/rules/using-git-worktrees.md +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/.claude/rules/worktree-pr-and-clean.md +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/.claude/rules/worktree-start-feature.md +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/.claude/rules/worktree-status.md +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/.github/dependabot.yml +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/.github/workflows/codeql-analysis.yml +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/.github/workflows/release.yml +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/.isort.cfg +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/.jupyter/jupyter_notebook_config.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/.pylintrc +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/CHANGES.rst +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/CODE_OF_CONDUCT.md +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/CONTRIBUTING.md +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/INSTALL +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/LICENSE +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/MANIFEST.in +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/Makefile +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/README.md +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/bin/README.md +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/gunicorn_config.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/mypy.ini +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/nav.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/__cli__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/cache/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/cache/backends/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/cache/backends/abstract.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/cache/backends/memcache.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/cache/backends/redis.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/cache/base.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/abstract.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/arangodb.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/athena.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/bigquery.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/cassandra.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/clickhouse.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/cockroachdb.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/couchbase.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/couchdb.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/countries.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/delta.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/documentdb.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/dynamodb.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/elastic.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/ga.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/gcalc.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/hazel.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/iceberg.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/influx.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/jdbc.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/jira.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/mariadb.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/memcached.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/mongo.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/mysql.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/odbc.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/openweather.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/oracle.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/py.typed +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/qs.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/redis.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/rest.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/rethink.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/sa.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/salesforce.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/scylladb.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/sqlalchemy.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/sqlite.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/sqlserver.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/upc.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/drivers/zipcodeapi.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/handlers/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/handlers/utils.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/datasources/models.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/events/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/exceptions.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/handlers/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/handlers/_pagination.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/handlers/log.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/handlers/outputs/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/handlers/outputs/tableOutput/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/handlers/outputs/tableOutput/postgres.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/handlers/outputs/tableOutput/table.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/handlers/variables.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/interfaces/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/interfaces/credentials.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/interfaces/databases/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/interfaces/databases/abstract.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/interfaces/databases/bigquery.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/interfaces/databases/db.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/interfaces/databases/mongo.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/interfaces/databases/rethink.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/interfaces/http.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/interfaces/playwright_service.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/interfaces/selenium_service.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/libs/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/libs/encoders.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/libs/functions/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/libs/py.typed +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/models.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/dt/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/dt/abstract.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/dt/arrow.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/dt/dt.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/dt/factory.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/dt/iter.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/dt/modin.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/dt/pandas.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/dt/polars.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/output.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/tables/TableOutput/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/tables/TableOutput/abstract.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/tables/TableOutput/bigquery.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/tables/TableOutput/documentdb.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/tables/TableOutput/mongodb.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/tables/TableOutput/mysql.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/tables/TableOutput/postgres.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/tables/TableOutput/rethink.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/tables/TableOutput/sa.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/tables/TableOutput/table.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/tables/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/writers/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/writers/abstract.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/writers/bokeh.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/writers/clustering.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/writers/csv.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/writers/describe.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/writers/eda.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/writers/excel.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/writers/html.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/writers/json.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/writers/pdf.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/writers/pickle.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/writers/plotly.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/writers/profiling.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/writers/report.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/writers/table.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/writers/tsv.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/writers/txt.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/outputs/writers/xml.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/abstract.c +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/abstract.pxd +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/abstract.pyx +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/arangodb.c +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/arangodb.pxd +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/arangodb.pyx +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/bigquery.pxd +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/bigquery.pyx +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/cql.pxd +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/cql.pyx +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/deltatbl.pxd +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/deltatbl.pyx +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/elastic.cpp +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/elastic.pxd +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/elastic.pyx +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/iceberg.pxd +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/iceberg.pyx +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/influx.c +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/influx.pxd +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/influx.pyx +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/mongo.cpp +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/mongo.pxd +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/mongo.pyx +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/parser.c +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/parser.pxd +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/parser.pyx +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/pgsql.pxd +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/pgsql.pyx +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/rethink.c +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/rethink.pxd +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/rethink.pyx +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/sosql.pxd +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/sosql.pyx +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/sqlserver.pxd +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/parsers/sqlserver.pyx +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/plugins/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/plugins/importer.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/plugins/sources/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/abstract.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/arangodb.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/bigquery.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/cassandra.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/db.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/default.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/deltatbl.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/documentdb.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/dummy.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/elastic.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/external.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/http.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/iceberg.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/influx.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/mysql.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/pg.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/py.typed +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/rest.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/rethink.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/salesforce.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/scylladb.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/sources/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/sources/abstract.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/sources/amazon.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/sources/countries.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/sources/ga.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/sources/geofcc.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/sources/gmaps.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/sources/graphcountries.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/sources/graphql.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/sources/http.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/sources/hubspot.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/sources/openweather.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/sources/parsers/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/sources/parsers/amproduct.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/sources/parsers/xpath.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/sources/pokemon.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/sources/populartimes.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/sources/py.typed +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/sources/rest.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/sources/retailnext.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/sources/rssapp.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/sources/salesforce.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/sources/scrapper.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/sources/shoppertrack.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/sources/swop.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/sources/uap.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/sources/upc.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/sources/wm_stores.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/sources/zammad.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/sources/zipcodeapi.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/sql.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/providers/sqlserver.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/py.typed +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/qs_parsers/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/queries/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/queries/base.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/queries/executor.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/queries/models.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/queries/multi/components/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/queries/multi/operators/Concat.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/queries/multi/operators/GroupBy.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/queries/multi/operators/Info.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/queries/multi/operators/Join.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/queries/multi/operators/Melt.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/queries/multi/operators/Merge.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/queries/multi/operators/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/queries/multi/operators/abstract.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/queries/multi/operators/filter/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/queries/multi/operators/filter/flt.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/queries/multi/sources/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/queries/multi/sources/file.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/queries/multi/sources/query.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/queries/multi/transformations/Forecast.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/queries/multi/transformations/Map.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/queries/multi/transformations/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/queries/multi/transformations/abstract.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/queries/multi/transformations/correlation.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/queries/multi/transformations/crosstab.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/queries/multi/transformations/google/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/queries/multi/transformations/google/maps.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/queries/multi/transformations/pivot.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/queries/multi/transformations/tOrder.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/queries/multi/transformations/tPandas.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/scheduler/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/scheduler/jobs.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/scheduler/notifications.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/scheduler/scheduler.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/template/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/template/parser.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/types/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/types/converters.cpp +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/types/converters.pyx +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/types/dt/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/types/dt/filters.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/types/dt/transforms.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/types/py.typed +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/types/typedefs.c +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/types/typedefs.pyx +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/types/validators.cpp +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/types/validators.pyx +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/utils/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/utils/cache_serialization.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/utils/events.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/utils/fn.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/utils/functions.cpp +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/utils/functions.pyx +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/utils/getfunc.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/utils/handlers.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/utils/parseqs.cpp +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/utils/parseqs.pyx +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource/utils/validators.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource.egg-info/dependency_links.txt +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource.egg-info/entry_points.txt +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource.egg-info/not-zip-safe +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource.egg-info/requires.txt +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/querysource.egg-info/top_level.txt +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/run.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/rust/Cargo.lock +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/rust/Cargo.toml +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/rust/pyproject.toml +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/rust/src/arangodb_parser.rs +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/rust/src/bigquery_parser.rs +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/rust/src/cql_parser.rs +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/rust/src/elastic_parser.rs +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/rust/src/filter_common.rs +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/rust/src/flux_parser.rs +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/rust/src/lib.rs +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/rust/src/mongo_parser.rs +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/rust/src/mssql_parser.rs +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/rust/src/parseqs.rs +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/rust/src/pgsql_parser.rs +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/rust/src/rethink_parser.rs +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/rust/src/safe_dict.rs +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/rust/src/soql_parser.rs +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/rust/src/validators.rs +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/setup.cfg +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/setup.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/static/notebook/bundle.js +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/templates/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/templates/base.html +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/templates/default.html +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/templates/default_table.html +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/templates/fontlist-v330.json +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/templates/fontlist-v390.json +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/templates/table_charts.html +0 -0
- {querysource-4.1.12/tests/handlers → querysource-4.2.1/tests/auth}/__init__.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/tests/handlers/conftest.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/tests/handlers/test_querymanager_pagination.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/tests/test_api.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/tests/test_arangodb_parser.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/tests/test_column_filters.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/tests/test_elastic_parser.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/tests/test_eval.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/tests/test_join_conditions.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/tests/test_join_with_column_filter.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/tests/test_rss.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/tests/test_scheduler_core.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/tests/test_scheduler_integration.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/tests/test_scheduler_jobs.py +0 -0
- {querysource-4.1.12 → querysource-4.2.1}/tests/test_scheduler_notifications.py +0 -0
- {querysource-4.1.12 → 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)
|