Python SDK for DATAQUERY Data API - Query, download, and check availability of economic data files
Project description
DataQuery SDK
Python SDK for the J.P. Morgan DataQuery API — authenticated file downloads, time-series queries, and real-time notification-driven downloads with OAuth 2.0, rate limiting, and automatic retries built in.
Contents
- Features
- New here? Three steps to your first download
- Installation
- Configure credentials
- Quick start
- Auto-download (coming soon)
- CLI
- Configuration
- Logging
- Error handling
- Troubleshooting
- Date formats
- Performance tuning
- API reference (most-used methods)
- Examples · Development · Requirements · Support
Features
- Streaming file downloads — single streaming GET per file
- Historical and date-range downloads — fetch every file in a group (optionally filtered to one or many
file-group-ids) between two dates - Notification-driven downloads (SSE) — subscribe to the
/events/notificationstream and auto-download files as soon as they are published - Time-series queries — by expression, by instrument, or by group with attribute / filter projections
- OAuth 2.0 with token caching and refresh — or supply a bearer token directly
- Token-bucket rate limiter — 300 rpm / 5 tps defaults (configurable up to API limits)
- Retry + circuit breaker — exponential backoff, configurable failure threshold
- Sync and async APIs — every operation has
_asyncand sync variants - Optional pandas integration —
to_dataframe(...)on any response - CLI —
dataquery groups | files | availability | download | download-group | auth | config
New here? Three steps to your first download
-
pip install dataquery-sdk -
Put your OAuth credentials in a
.envfile (see Configure credentials) -
Run a one-liner to confirm everything works:
import asyncio from dataquery import DataQuery async def main(): async with DataQuery() as dq: groups = await dq.list_groups_async(limit=5) for g in groups: print(g.group_id, "—", g.group_name) asyncio.run(main())
If that prints groups, auth and networking are working. From there, jump to Quick start (date-range download) or the CLI.
Installation
# Core install
pip install dataquery-sdk
# With pandas DataFrame conversion
pip install "dataquery-sdk[pandas]"
# With dev tooling (ruff, mypy, pytest)
pip install "dataquery-sdk[dev]"
Python 3.11+ is required.
Configure credentials
Set OAuth client credentials via environment variables:
export DATAQUERY_CLIENT_ID="your_client_id"
export DATAQUERY_CLIENT_SECRET="your_client_secret"
Or create a .env file in the working directory:
DATAQUERY_CLIENT_ID=your_client_id
DATAQUERY_CLIENT_SECRET=your_client_secret
Or pass them directly to the constructor:
from dataquery import DataQuery
dq = DataQuery(client_id="...", client_secret="...")
A starter .env can be generated with dataquery config template --output .env.
Quick start
Download files for a date range
from dataquery import DataQuery
# async
async with DataQuery() as dq:
result = await dq.run_group_download_async(
group_id="JPMAQS_GENERIC_RETURNS",
start_date="20250101",
end_date="20250131",
destination_dir="./data",
)
# OperationReport (Pydantic model) — counts/timing/data/details are dicts on it.
print(f"{result.counts['successful_downloads']}/{result.counts['total_files']} files downloaded")
# sync — same arguments, drop the _async suffix
with DataQuery() as dq:
result = dq.run_group_download(
group_id="JPMAQS_GENERIC_RETURNS",
start_date="20250101",
end_date="20250131",
destination_dir="./data",
)
Restrict a date-range download to specific file-group-ids
file_group_id accepts a single id or a list. When a list is supplied, availability
queries run in parallel per id and the union of dates is downloaded.
async with DataQuery() as dq:
result = await dq.run_group_download_async(
group_id="JPMAQS_GENERIC_RETURNS",
start_date="20250101",
end_date="20250131",
destination_dir="./data",
file_group_id=["FG_ABC", "FG_DEF", "FG_XYZ"],
)
Download a single file
from pathlib import Path
from dataquery import DataQuery
async with DataQuery() as dq:
result = await dq.download_file_async(
file_group_id="JPMAQS_GENERIC_RETURNS",
file_datetime="20250115",
destination_path=Path("./downloads"),
)
print(f"Downloaded: {result.local_path} ({result.file_size} bytes)")
Time-series queries
async with DataQuery() as dq:
# By expression
ts = await dq.get_expressions_time_series_async(
expressions=["DB(MTE,IRISH EUR 1.100 15-May-2029 LON,,IE00BH3SQ895,MIDPRC)"],
start_date="20240101",
end_date="20240131",
)
# By group with attributes + filter
ts = await dq.get_group_time_series_async(
group_id="FI_GO_BO_EA",
attributes=["MIDPRC", "REPO_1M"],
filter="country(IRL)",
start_date="20240101",
end_date="20240131",
)
df = dq.to_dataframe(ts) # requires pandas extra
Discover available data
async with DataQuery() as dq:
groups = await dq.list_groups_async(limit=100)
files = await dq.list_files_async(group_id="JPMAQS_GENERIC_RETURNS")
available = await dq.list_available_files_async(
group_id="JPMAQS_GENERIC_RETURNS",
start_date="20250101",
end_date="20250131",
)
instruments = await dq.search_instruments_async(
group_id="FI_GO_BO_EA", keywords="irish"
)
Auto-download (coming soon)
Real-time notification-driven downloads via the DataQuery SSE stream are under
active development and will be available in a future release. Once released,
auto_download_async will subscribe to the notification endpoint and download
files automatically as they are published — no polling required.
CLI
The installer registers a dataquery script:
# List / search groups
dataquery groups --limit 100
dataquery groups --search "fixed income" --json
# List files in a group
dataquery files --group-id JPMAQS_GENERIC_RETURNS --json
# Check availability for a single file
dataquery availability --file-group-id JPMAQS_GENERIC_RETURNS --file-datetime 20250115
# Download a single file
dataquery download --file-group-id JPMAQS_GENERIC_RETURNS \
--file-datetime 20250115 \
--destination ./downloads
# Watch a group and download as new files arrive (calls auto_download_async under the hood —
# same SSE subscription, same event-replay state files)
dataquery download --watch --group-id JPMAQS_GENERIC_RETURNS --destination ./downloads
# Download everything in a date range
dataquery download-group --group-id JPMAQS_GENERIC_RETURNS \
--start-date 20250101 --end-date 20250131 \
--destination ./data
# Restrict to one or more file-group-ids
dataquery download-group --group-id JPMAQS_GENERIC_RETURNS \
--file-group-id FG_ABC FG_DEF \
--start-date 20250101 --end-date 20250131
# Config utilities
dataquery config show
dataquery config validate
dataquery config template --output .env
# Verify auth
dataquery auth test
Every subcommand accepts --env-file PATH (to point at a non-default .env) and
most accept --json for machine-readable output.
Configuration
Environment variables
All environment variables use the DATAQUERY_ prefix.
Credentials
| Variable | Default | Notes |
|---|---|---|
DATAQUERY_CLIENT_ID |
(none) | Required for OAuth |
DATAQUERY_CLIENT_SECRET |
(none) | Required for OAuth |
DATAQUERY_BEARER_TOKEN |
(none) | Alternative to OAuth |
DATAQUERY_OAUTH_ENABLED |
true |
Set false to use bearer-token mode |
API endpoints
| Variable | Default |
|---|---|
DATAQUERY_BASE_URL |
https://api-developer.jpmorgan.com |
DATAQUERY_FILES_BASE_URL |
https://api-dataquery.jpmchase.com |
DATAQUERY_CONTEXT_PATH |
/research/dataquery-authe/api/v2 |
HTTP / retry / rate limit
| Variable | Default |
|---|---|
DATAQUERY_TIMEOUT |
600.0 |
DATAQUERY_MAX_RETRIES |
3 |
DATAQUERY_RETRY_DELAY |
1.0 |
DATAQUERY_CIRCUIT_BREAKER_THRESHOLD |
5 |
DATAQUERY_REQUESTS_PER_MINUTE |
300 |
DATAQUERY_BURST_CAPACITY |
5 |
DATAQUERY_POOL_CONNECTIONS |
10 |
DATAQUERY_POOL_MAXSIZE |
20 |
Proxy (optional)
DATAQUERY_PROXY_ENABLED, DATAQUERY_PROXY_URL, DATAQUERY_PROXY_USERNAME,
DATAQUERY_PROXY_PASSWORD, DATAQUERY_PROXY_VERIFY_SSL.
Programmatic configuration
from dataquery import ClientConfig, DataQuery
config = ClientConfig(
client_id="...",
client_secret="...",
base_url="https://api-developer.jpmorgan.com",
timeout=60.0,
max_retries=3,
requests_per_minute=300,
)
async with DataQuery(config) as dq:
...
# Or pass overrides as kwargs on top of env/.env resolution:
async with DataQuery(client_id="...", client_secret="...", timeout=60.0) as dq:
...
Logging
The SDK logs through structlog and emits structured events for requests, retries, rate-limit waits, SSE reconnects, and download progress. Two ways to drive it:
Standard Python logging — works without extra setup:
import logging
logging.basicConfig(level=logging.INFO)
logging.getLogger("dataquery").setLevel(logging.DEBUG) # SDK-only DEBUG
DEBUG includes per-chunk download progress and SSE keepalives — useful while debugging, noisy in production.
Structured (JSON) output — recommended for long-running auto_download
services so a log shipper can parse the events:
from pathlib import Path
from dataquery.config.logging import (
LogFormat, LogLevel, create_logging_config, create_logging_manager,
)
cfg = create_logging_config(
level=LogLevel.INFO,
format=LogFormat.JSON, # or LogFormat.CONSOLE for humans
enable_file=True,
log_file=Path("./dataquery.log"),
enable_request_logging=False, # set True to log every HTTP request/response
)
create_logging_manager(cfg) # installs handlers on the root logger
examples/system/enable_request_logging.py is a runnable version showing
request/response logging for traffic debugging.
Health snapshots for auto_download — manager.get_stats() returns
notifications received, files downloaded / skipped / failed, the last event
id, and a bounded ring of recent errors. Wire it into a /healthz endpoint
for daemon-style deployments.
Error handling
All errors inherit from DataQueryError:
from dataquery import DataQuery
from dataquery.types.exceptions import (
DataQueryError,
AuthenticationError,
NotFoundError,
RateLimitError,
NetworkError,
DownloadError,
ConfigurationError,
)
async with DataQuery() as dq:
try:
ts = await dq.get_expressions_time_series_async(
expressions=["DB(...)"], start_date="20240101", end_date="20240131"
)
except AuthenticationError:
... # check credentials
except RateLimitError:
... # back off — SDK already retried
except NotFoundError:
... # group / file / instrument not found
except NetworkError:
... # transient; SDK already retried
except DataQueryError:
... # any other SDK-level failure
Troubleshooting
AuthenticationError / HTTP 401 on the first call
- Verify both
DATAQUERY_CLIENT_IDandDATAQUERY_CLIENT_SECRETare set:dataquery config showwill print the resolved config (secrets masked). - Confirm OAuth is reaching the right endpoint:
dataquery auth testperforms a token exchange and reports the failure mode. - If the credentials are correct but the audience is wrong, set
DATAQUERY_OAUTH_AUDto the value provisioned for your client.
.env file isn't picked up
- The SDK looks for
.envin the current working directory at instantiation. Eithercdto the directory containing.envbefore running, or pass the path explicitly:DataQuery(env_file=".env.production"). - Variables already set in the shell environment win over the
.envfile — unset them (unset DATAQUERY_CLIENT_ID) if you want the file to take effect. - The CLI accepts
--env-file PATHon every subcommand for the same reason.
Connection / proxy / SSL failures
- Behind a corporate proxy, set
DATAQUERY_PROXY_ENABLED=trueandDATAQUERY_PROXY_URL=http://proxy.host:port. AddDATAQUERY_PROXY_USERNAME/DATAQUERY_PROXY_PASSWORDif auth is required. - For self-signed proxy CAs, set
DATAQUERY_PROXY_VERIFY_SSL=false(insecure; prefer pointingSSL_CERT_FILEat the corporate root CA bundle). - Sporadic
NetworkErrorafter long idle periods usually means a stateful middlebox is dropping the SSE socket — setheartbeat_timeout=90.0onauto_download_asyncto force a reconnect when no bytes arrive within the window.
Rate-limit pauses
- Default is 300 rpm / 5 tps. The SDK self-throttles via the token-bucket
limiter; if you see long sleeps before requests, lower
DATAQUERY_REQUESTS_PER_MINUTEis not the cure — it's likely working as designed. Raise it (up to your provisioned limit) to go faster. dq.get_rate_limit_info()shows the current bucket state.
SSE auto-download "missed" events after a restart
- Confirm
enable_event_replay=True(the default). - Replay state lives under
<destination>/.sse_state/sse_<fingerprint>.json— if that directory was wiped, the next start has nothing to resume from. Usemanager.clear_event_id()(ordataquery download --watch --reset-event-id) only when you intentionally want a clean slate.
Date formats
start_date="20240101" # absolute, YYYYMMDD
start_date="TODAY"
start_date="TODAY-1D" # yesterday
start_date="TODAY-1W"
start_date="TODAY-1M"
start_date="TODAY-1Y"
Performance tuning
run_group_download_async streams each file as a single GET. The SDK
automatically inserts delays between file starts so the configured
requests_per_minute is not exceeded.
await dq.run_group_download_async(
group_id="JPMAQS_GENERIC_RETURNS",
start_date="20250101",
end_date="20250131",
destination_dir="./data",
max_retries=3,
)
Tune throughput via DATAQUERY_REQUESTS_PER_MINUTE and DATAQUERY_BURST_CAPACITY
(see Configuration) rather than per-call concurrency
flags.
API reference (most-used methods)
| Area | Method | Notes |
|---|---|---|
| Discovery | list_groups_async(limit) |
|
search_groups_async(keywords, limit, offset) |
||
list_files_async(group_id, file_group_id=None) |
||
list_available_files_async(group_id, file_group_id, start_date, end_date) |
||
list_instruments_async(group_id, instrument_id=None, page=None) |
||
search_instruments_async(group_id, keywords, page=None) |
||
get_group_attributes_async(group_id, ...) |
||
get_group_filters_async(group_id, page=None) |
||
| Downloads | download_file_async(file_group_id, file_datetime, ...) |
single file |
run_group_download_async(group_id, start_date, end_date, file_group_id=None, ...) |
date range, single or list of ids | |
download_historical_async(...) |
chunked historical backfill | |
auto_download_async(group_id, ...) |
SSE notifications (the only watch path) | |
| Time series | get_expressions_time_series_async(expressions, start_date, end_date) |
|
get_instrument_time_series_async(instruments, attributes, start_date, end_date) |
||
get_group_time_series_async(group_id, attributes, filter, start_date, end_date) |
||
| Grid data | get_grid_data_async(...) |
|
| Utilities | check_availability_async(file_group_id, file_datetime) |
|
health_check_async() |
||
to_dataframe(response) |
requires pandas extra |
|
get_stats() / get_pool_stats() / get_rate_limit_info() |
diagnostics |
Every async method has a sync counterpart with the same name minus the
_async suffix — list_groups_async ↔ list_groups, download_file_async ↔
download_file, etc. Sync calls run the coroutine internally via
asyncio.run, so do not call them from inside an existing event loop.
Examples
The examples/ directory is organised by feature:
examples/files/— single-file and date-range downloadsexamples/expressions/— expression time seriesexamples/instruments/— instrument discovery + time seriesexamples/groups/andexamples/groups_advanced/— group discovery and time seriesexamples/grid/— grid dataexamples/system/— SSE notification subscriber (single + multi-group), diagnostics
Run any example directly:
python examples/files/download_file.py
python examples/system/auto_download_example.py # single group
python examples/system/auto_download_multi_group_example.py # several groups in parallel
Development
# Clone
git clone https://github.com/jpmorganchase/dataquery-sdk.git
cd dataquery-sdk
# Install with dev + all extras
uv sync --all-extras --dev # using uv
# or
pip install -e ".[dev,pandas]"
# Run tests
pytest tests/ -v
pytest tests/ --cov=dataquery --cov-report=term-missing
# Lint / format / type-check
ruff check dataquery/ tests/ examples/
ruff format dataquery/ tests/ examples/
mypy dataquery/
Pytest markers: slow, integration, unit, asyncio.
Requirements
- Python 3.11+
aiohttp>=3.8,<4,pydantic>=2,<3,structlog>=23,python-dotenv>=1- Optional:
pandas>=2(forto_dataframe)
Support
- GitHub Issues: https://github.com/jpmorganchase/dataquery-sdk/issues
- Email: dataquery_support@jpmorgan.com
License
MIT — see LICENSE.
Changelog
See docs/changelog.md.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file dataquery_sdk-0.2.0.tar.gz.
File metadata
- Download URL: dataquery_sdk-0.2.0.tar.gz
- Upload date:
- Size: 125.2 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.13.13
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
d4b390fae65d27e5b821d38398ef41becf4699370d78466af10d507ce80e43a2
|
|
| MD5 |
de17e0ec75058b2d15cfe679ccdb5477
|
|
| BLAKE2b-256 |
08921d7cf17c935cafefe5b02b269be152ea80d1fd42e3caaa44990245597a5a
|
File details
Details for the file dataquery_sdk-0.2.0-py3-none-any.whl.
File metadata
- Download URL: dataquery_sdk-0.2.0-py3-none-any.whl
- Upload date:
- Size: 128.4 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.13.13
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
55d23616f3741a5f5c2b8d925b58de4648f95d4c082f2484a606c549a537c9a9
|
|
| MD5 |
cff75a5829864ffb43027118053629e4
|
|
| BLAKE2b-256 |
a2119856e9590b0e8d510a2647664915714fdbdbf2a912861c147da442a604bf
|