Python SDK for the DSIS (DecisionSpace Integration Server) API. Handles dual-token authentication (Azure AD + DSIS) and provides a fluent query builder for OData access.
pip install dsis-clientFor protobuf bulk data decoding (grids, horizons, seismic):
pip install dsis-schemas[protobuf]import os
from dsis_client import DSISClient, DSISConfig, Environment, QueryBuilder
config = DSISConfig(
environment=Environment.DEV,
tenant_id=os.getenv("DSIS_TENANT_ID"),
client_id=os.getenv("DSIS_CLIENT_ID"),
client_secret=os.getenv("DSIS_CLIENT_SECRET"),
access_app_id=os.getenv("DSIS_ACCESS_APP_ID"),
dsis_username=os.getenv("DSIS_USERNAME"),
dsis_password=os.getenv("DSIS_PASSWORD"),
subscription_key_dsauth=os.getenv("DSIS_SUBSCRIPTION_KEY_DSAUTH"),
subscription_key_dsdata=os.getenv("DSIS_SUBSCRIPTION_KEY_DSDATA"),
dsis_site="qa",
auth_timeout=(5, 30), # Optional: bound Azure AD + DSIS token requests
)
client = DSISClient(config)
query = (
QueryBuilder(
model_name="OpenWorksCommonModel",
district_id="OpenWorksCommonModel_OW_SV4TSTA-OW_SV4TSTA",
project="SNORRE",
)
.schema("Well")
.select("name", "depth", "status")
.filter("depth gt 1000")
.expand("wellbores")
)
for well in client.execute_query(query):
print(well)QueryBuilder is the primary way to query data. It uses method chaining and IS the query object (no .build() needed).
query = (
QueryBuilder(
model_name="OW5000",
district_id="OpenWorks_OW_SV4TSTA_SingleSource-OW_SV4TSTA",
project="SNORRE",
)
.schema("Fault")
.select("fault_id,fault_type,fault_name")
.filter("fault_type eq 'NORMAL'")
.expand("interpretations")
)
for item in client.execute_query(query):
print(item)Pass a model class to .schema() and use cast=True to get typed results:
from dsis_model_sdk.models.common import Basin
query = (
QueryBuilder(model_name="OpenWorksCommonModel", district_id=dist, project=prj)
.schema(Basin)
.select("basin_name,basin_id,native_uid")
)
for basin in client.execute_query(query, cast=True):
print(basin.basin_name) # IDE autocomplete worksexecute_query() automatically follows odata.nextLink across all pages. Control with max_pages:
# All pages (default)
all_items = list(client.execute_query(query))
# First page only (max 1000 items)
first_page = list(client.execute_query(query, max_pages=1))Fetch binary data (horizons, log curves, seismic) with get_bulk_data() or stream large datasets with get_bulk_data_stream():
from dsis_model_sdk.models.common import HorizonData3D
from dsis_model_sdk.protobuf import decode_horizon_data
binary_data = client.get_bulk_data(
schema=HorizonData3D,
native_uid="46075",
district_id=district_id,
project=project,
)
decoded = decode_horizon_data(binary_data)For large datasets, stream in chunks:
for chunk in client.get_bulk_data_stream(
schema=SeismicDataSet3D,
native_uid=seismic,
query=query,
chunk_size=10 * 1024 * 1024, # 10 MB
stream_retries=2,
):
process(chunk)Set stream_retries to retry transient failures while reading streamed chunks.
Retries reopen the stream and assume the endpoint returns the same bytes across reconnects.
The timeout value on streamed downloads applies to connection setup and waiting for the next bytes, not to the full transfer duration.
For authentication, set auth_timeout on DSISConfig to bound Azure AD token acquisition and DSIS token refresh requests:
config = DSISConfig(
...,
auth_timeout=(5, 30), # (connect_timeout, read_timeout)
)from dsis_client import DSISAuthenticationError, DSISAPIError, DSISConfigurationError
try:
client = DSISClient(config)
for item in client.execute_query(query):
print(item)
except DSISConfigurationError as e:
print(f"Bad config: {e}")
except DSISAuthenticationError as e:
print(f"Auth failed: {e}")
except DSISAPIError as e:
print(f"Request failed: {e}")import logging
logging.basicConfig(level=logging.DEBUG)
logging.getLogger("dsis_client").setLevel(logging.DEBUG)- Getting Started
- QueryBuilder Guide
- Common vs Native Model
- Advanced Serialization
- Working with Binary Data
See CONTRIBUTING.md.