Skip to main content

Official Python SDK for NextGIS Toolbox providing convenient access to geographical data processing tools

Project description

NextGIS Toolbox SDK

A Python SDK for interacting with NextGIS Toolbox API, providing convenient access to geographical data processing tools.

Features

  • Easy-to-use interface for NextGIS Toolbox tools
  • Synchronous and asynchronous task execution
  • Robust file upload and download capabilities with rich metadata
  • Simple dict-based file references - what the server returns is what you send
  • Access to file metadata: original filename, UUID, and storage type
  • Built-in retry mechanism for API operations
  • Progress tracking for file operations
  • Comprehensive logging system
  • Tool chaining for complex workflows
  • Intelligent file name handling from HTTP headers
  • Serializable task results for easy storage and sharing

Installation

pip install toolbox-sdk

Quickstart

Run the hello tool and use the result.value property to get the result of the run:

from toolbox_sdk import ToolboxClient

# Initialize client with your API key, use default base url
toolbox = ToolboxClient("your-api-key")

# Create the tool
hello = toolbox.tool("hello")

# Run the tool with the correct parameter
result = hello({"name": "Natalia"})

# Print the result
print(result.value)

Running convert operation and configure the logger to watch the progress:

from toolbox_sdk import ToolboxClient

# Enable basic debug logging to stderr
ToolboxClient.configure_logger()

# Initialize client with your API key, use default base url
toolbox = ToolboxClient("your-api-key")

# Run a tool synchronously
convert = toolbox.tool("convert")
result = convert({
    "source": toolbox.upload_file("input.geojson"),
    "format": "GPKG"
})

# Download the resulting file into the current directory
toolbox.download_results(result, ".")

# Access the downloaded file path
output_path = result.get_file_path("output")
print(f"Downloaded file: {output_path}")

Running generalization operation:

from toolbox_sdk import ToolboxClient

# Create the client with API key and on premise Toolbox URL
toolbox = ToolboxClient(
    api_key="your-api-key",
    base_url="https://toolbox.example.com",
)

# Create the tool
generalization = toolbox.tool("generalization")

# Run the tool with the correct parameter
result = generalization({
    "vector": toolbox.upload_file("generalization_input.zip"),
    "threshold": 0.005,
    "method": "douglas"
})

# Download all results into the current directory
toolbox.download_results(result, ".")

# Get all downloaded file paths
file_paths = result.get_all_file_paths()
for name, path in file_paths.items():
    print(f"{name}: {path}")

Tool Chaining

You can chain tools together by using the output of one tool as the input to another:

from dotenv import load_dotenv

from toolbox_sdk import ToolboxClient

# Load environment variables from .env file
load_dotenv()

# Enable basic debug logging to stderr
ToolboxClient.configure_logger()

# Initialize client with API key from environment variable
toolbox = ToolboxClient()

# Create tool instances
mapcalc = toolbox.tool("r_mapcalc")
ndi = toolbox.tool("ndi")

# Upload the input bands
band4_id = toolbox.upload_file("band4.tif")
band5_id = toolbox.upload_file("band5.tif")

# Process band4 with mapcalc - scale to 0-1 range
band4_result = mapcalc({
    "A": band4_id,
    "expression": "A / 10000",  # Scale NIR band to 0-1 range
})

# Process band5 with mapcalc - scale to 0-1 range
band5_result = mapcalc({
    "A": band5_id,
    "expression": "A / 10000",  # Scale Red band to 0-1 range
})

# Download intermediate results into separate directories, since they have the same file name
toolbox.download_results(band4_result, "band4_result")
toolbox.download_results(band5_result, "band5_result")
# Upload the result
band4_id = toolbox.upload_file(band4_result.get_file_path_as_string("result_raster"))
band5_id = toolbox.upload_file(band5_result.get_file_path_as_string("result_raster"))

# Calculate NDVI using the ndi tool
ndvi_result = ndi({
    "raster_1": band4_id,  # Use output from first tool
    "raster_2": band5_id,  # Use output from second tool
    "formula": "(A-B)/(A+B)"  # NDVI formula
})

# Download the final NDVI result
toolbox.download_results(ndvi_result, "ndvi_output")

# Access specific output path
ndvi_path = ndvi_result.get_file_path_as_string("result_file")
print(f"NDVI raster is available at: {ndvi_path}")

File Uploads

When you upload a file using upload_file(), the SDK returns a dictionary containing file metadata. This dict can be passed directly to tools.

Basic Usage

from toolbox_sdk import ToolboxClient

toolbox = ToolboxClient("your-api-key")

# Upload a file - returns a dict with metadata
file_data = toolbox.upload_file("band4.tif")
# Returns: {"name": "band4.tif", "local": {"uuid": "..."}, "s3": null}

# Pass the dict directly to tools
result = toolbox.tool("r_mapcalc")({"A": file_data, "expression": "A * 2"})

Accessing File Metadata

The returned dictionary contains all file metadata:

# Upload a file
file_data = toolbox.upload_file("my_data.geojson")

# Access the original filename
print(f"Filename: {file_data['name']}")
# Output: Filename: my_data.geojson

# Get the unique file identifier
if file_data["local"]:
    uuid = file_data["local"]["uuid"]
    storage = "local"
elif file_data["s3"]:
    uuid = file_data["s3"]["uuid"]
    storage = "s3"

print(f"UUID: {uuid}")
# Output: UUID: 01961f79-ac76-9c7d-fae6-dddff5638b47

print(f"Storage: {storage}")
# Output: Storage: local (or s3)

File Data Structure

The upload returns a dictionary with the following structure:

{
    "name": str,              # Original filename
    "local": {                # Present if stored locally
        "uuid": str           # Unique file identifier
    } or None,
    "s3": {                   # Present if stored in S3
        "uuid": str           # Unique file identifier
    } or None
}

Example: Tracking Uploaded Files

from toolbox_sdk import ToolboxClient

toolbox = ToolboxClient("your-api-key")

# Upload multiple files
files = []
for filename in ["band4.tif", "band5.tif", "mask.tif"]:
    file_data = toolbox.upload_file(filename)
    files.append(file_data)

# Use the uploaded files in a tool
result = toolbox.tool("r_mapcalc")({
    "A": files[0],  # Pass dicts directly
    "B": files[1],
    "expression": "(A - B) / (A + B)"
})

Environment Variables

ToolboxClient uses the TOOLBOX_API_KEY and TOOLBOX_BASE_URL environment variables as default values for the corresponding parameters. Thus, you can configure it using these variables.

$ export TOOLBOX_API_KEY=your-api-key
$ python
>>> from toolbox_sdk import ToolboxClient
>>> toolbox = ToolboxClient()

This also means that you can use .env files to configure the SDK via the dotenv library:

$ unset TOOLBOX_API_KEY
$ echo TOOLBOX_API_KEY=your-api-key > .env
$ pip install python-dotenv
$ python
>>> from dotenv import load_env
>>> load_env()
>>>
>>> from toolbox_sdk import ToolboxClient
>>> toolbox = ToolboxClient()

Advanced Usage

Asynchronous Operations

The .env file must be present that contains the API key:

TOOLBOX_API_KEY=your-api-key
from dotenv import load_dotenv
from toolbox_sdk import ToolboxClient

# Get the API key from the .env file
load_dotenv()

# Configure logger and create the client
ToolboxClient.configure_logger()
toolbox = ToolboxClient()

# Create the tool
mapcalc = toolbox.tool("r_mapcalc")

# Set the correct parameter
task = mapcalc.submit({
    "A": toolbox.upload_file("band4.tif"),
    "B": toolbox.upload_file("band5.tif"),
    "expression": "A + B"
})

# Run the task
result = task.wait_for_completion(timeout=120)

# Download all results into the current directory
toolbox.download_results(result, ".")

# Check the outputs of the tool
print(result.outputs)

Working with File Paths

The SDK now intelligently handles file names from HTTP headers and provides methods to access downloaded file paths:

# Download results
result = toolbox.download_results(task_result, "output_dir")

# Get path to a specific output file
file_path = result.get_file_path("output_name")
print(f"File path: {file_path}")

# Get path as string (useful for serialization or passing to other libraries)
file_path_str = result.get_file_path_as_string("output_name")
print(f"File path string: {file_path_str}")

# Get all file paths
all_paths = result.get_all_file_paths()
for name, path in all_paths.items():
    print(f"{name}: {path}")

# Get serializable file paths (as strings)
serializable_paths = result.get_serializable_file_paths()

Serializing Task Results

Task results can be serialized to JSON for storage or sharing:

import json

# Get a serializable representation of the result
result_dict = result.to_dict()

# Convert to JSON
json_str = json.dumps(result_dict)

# Save to file
with open("task_result.json", "w") as f:
    f.write(json_str)

# Later, you can load and use the paths
with open("task_result.json", "r") as f:
    loaded_result = json.loads(f.read())

print(f"Task ID: {loaded_result['task_id']}")
print(f"File paths: {loaded_result['file_paths']}")

Key Components

  • ToolboxClient: Main client for API interaction
  • Tool: Represents individual Toolbox tools
  • Task: Handles asynchronous operations
  • TaskResult: Stores tool outputs and downloaded file paths

Error Handling

The SDK provides specific exceptions:

  • ToolboxError: Base exception
  • ToolboxAPIError: API-related errors
  • ToolboxTimeoutError: Timeout errors

Requirements

  • Python ≥ 3.8
  • requests ≥ 2.28.0
  • pytest ≥ 8.3.4 (for testing)
  • responses ≥ 0.25.3 (for testing)
  • python-dotenv ≥ 1.0.1 (for examples)

License

MIT License

Support

For issues and feature requests, please use the GitHub issue tracker.

Development

First, install hatch, then clone the repository:

$ git clone git@github.com:nextgis/toolbox_sdk.git
$ cd toolbox_sdk

And then use hatch commands: hatch test, hatch fmt, hatch build, etc. To run integration tests, provide TOOLBOX_API_KEY (and optionally TOOLBOX_BASE_URL) in the .env file to avoid setting them using environment variables each time. The tests will use load_env() as described above.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

toolbox_sdk-0.1.0b4.tar.gz (22.6 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

toolbox_sdk-0.1.0b4-py3-none-any.whl (13.7 kB view details)

Uploaded Python 3

File details

Details for the file toolbox_sdk-0.1.0b4.tar.gz.

File metadata

  • Download URL: toolbox_sdk-0.1.0b4.tar.gz
  • Upload date:
  • Size: 22.6 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: python-httpx/0.28.1

File hashes

Hashes for toolbox_sdk-0.1.0b4.tar.gz
Algorithm Hash digest
SHA256 fd172f3d3c0610881d0588f8cadf4edc05544284a92cf7f5071be889557dbbda
MD5 3e238324bedabae6141ee268537015a9
BLAKE2b-256 39fcfb72b36fe4704cd5839c5770d772f4ab0668bdcee2bbf1c33cd44a5cef46

See more details on using hashes here.

File details

Details for the file toolbox_sdk-0.1.0b4-py3-none-any.whl.

File metadata

File hashes

Hashes for toolbox_sdk-0.1.0b4-py3-none-any.whl
Algorithm Hash digest
SHA256 9af4bdc2732cd2cead171b6a0332f2a70e5c5b657dfdc7ad79be4cdcc48128ea
MD5 a3f641d4ade668e15f780a29eea5b3c0
BLAKE2b-256 8a11fadb91ea6417ff3ab7a4ebc78b9bbaf3dfa42c351ce476ff2c33b24acd41

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page