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