Skip to main content

Official Python SDK for the Ziptax API

Project description

Ziptax Python SDK

Official Python SDK for the Ziptax API - Get accurate sales and use tax rates for any US or Canadian address, with optional TaxCloud order management support.

Python Version License

Features

Core Features (ZipTax API)

  • ๐Ÿš€ Simple and intuitive API
  • ๐Ÿ›’ Cart tax calculation with per-item tax rates
  • ๐Ÿท๏ธ Product code (TIC) search and AI-powered recommendation
  • ๐Ÿ”„ Automatic retry logic with exponential backoff
  • โœ… Input validation
  • ๐Ÿ” Type hints for better IDE support
  • ๐Ÿ“ฆ Pydantic models for response validation
  • ๐Ÿ”’ Comprehensive error handling
  • โšก Support for concurrent operations
  • ๐Ÿงช Well-tested with high code coverage

TaxCloud Integration (Optional)

  • ๐Ÿ“‹ Order Management: Create, retrieve, and update orders
  • ๐Ÿ’ฐ Refund Processing: Full and partial refund support
  • ๐Ÿ”— Dual API Support: Seamlessly integrate both ZipTax and TaxCloud
  • ๐Ÿ” Optional Configuration: TaxCloud features only enabled when credentials provided

Installation

pip install ziptax-sdk

Quick Start

from ziptax import ZipTaxClient

# Initialize the client with your API key
client = ZipTaxClient.api_key("your-api-key-here")

# Get sales tax by address
response = client.request.GetSalesTaxByAddress(
    "200 Spectrum Center Drive, Irvine, CA 92618"
)

print(f"Address: {response.address_detail.normalized_address}")
if response.tax_summaries:
    for summary in response.tax_summaries:
        print(f"{summary.summary_name}: {summary.rate * 100:.2f}%")

# Always close the client when done
client.close()

Usage

Initialize the Client

from ziptax import ZipTaxClient

# Basic initialization
client = ZipTaxClient.api_key("your-api-key-here")

# With custom configuration
client = ZipTaxClient.api_key(
    "your-api-key-here",
    timeout=60,           # Request timeout in seconds
    max_retries=5,        # Maximum retry attempts
    retry_delay=2.0,      # Base delay between retries
)

# Using as a context manager (recommended)
with ZipTaxClient.api_key("your-api-key-here") as client:
    response = client.request.GetSalesTaxByAddress("123 Main St")

Get Sales Tax by Address

response = client.request.GetSalesTaxByAddress(
    address="200 Spectrum Center Drive, Irvine, CA 92618",
    country_code="USA",      # Optional: "USA" or "CAN" (default: "USA")
    historical="202401",     # Optional: Historical date (YYYYMM format)
    format="json",           # Optional: Response format (default: "json")
)

# Access response data
print(response.address_detail.normalized_address)
print(response.address_detail.geo_lat)
print(response.address_detail.geo_lng)

# Response code
print(f"Response: {response.metadata.response.code} - {response.metadata.response.message}")

# Tax summaries with display rates
if response.tax_summaries:
    for summary in response.tax_summaries:
        print(f"{summary.summary_name}: {summary.rate}")
        for display_rate in summary.display_rates:
            print(f"  {display_rate.name}: {display_rate.rate}")

# Base rates by jurisdiction
if response.base_rates:
    for rate in response.base_rates:
        print(f"{rate.jur_name} ({rate.jur_type}): {rate.rate}")

# Sourcing rules
if response.sourcing_rules:
    print(f"Sourcing: {response.sourcing_rules.value}")

Get Sales Tax by Geolocation

response = client.request.GetSalesTaxByGeoLocation(
    lat="33.6489",
    lng="-117.8386",
    country_code="USA",
    format="json",
)

print(response.address_detail.normalized_address)

Get Rates by Postal Code

response = client.request.GetRatesByPostalCode(
    postal_code="92694",
    format="json",
)

# Response includes all tax jurisdictions for the postal code
for result in response.results:
    print(f"{result.geo_city}, {result.geo_state}")
    print(f"Sales Tax: {result.tax_sales * 100:.2f}%")
    print(f"Use Tax: {result.tax_use * 100:.2f}%")

Get Account Metrics

metrics = client.request.GetAccountMetrics()

print(f"Requests: {metrics.request_count:,} / {metrics.request_limit:,}")
print(f"Usage: {metrics.usage_percent:.2f}%")
print(f"Account Active: {metrics.is_active}")
print(f"Message: {metrics.message}")

Search Product Codes (TIC)

Search for Taxability Information Codes (TICs) using a natural language product description. Returns all matching codes ranked and scored by relevance.

response = client.request.SearchProductCodes(
    "baked goods sold in plastic packaging"
)

for result in response.results:
    print(f"TIC {result.tic_id}: {result.label}")
    print(f"  Score: {result.score:.4f} (Rank: {result.rank})")
    print(f"  Description: {result.description}")

Use the returned tic_id as the taxability_code parameter in rate requests or cart line items. For v60 rate requests, convert to str first:

# Use with rate requests (taxability_code is a string parameter)
tic = response.results[0].tic_id
tax = client.request.GetSalesTaxByAddress(
    "200 Spectrum Center Dr, Irvine, CA 92618",
    taxability_code=str(tic),
)

# Use with cart line items (taxability_code is an integer)
CartLineItem(item_id="item-1", price=10.00, quantity=1, taxability_code=tic)

Recommend Product Code (TIC)

Get an AI-powered product code recommendation. Returns a single best-match TIC with higher accuracy than the standard search. Has slightly higher latency due to the AI processing step.

response = client.request.RecommendProductCode(
    "baked goods sold in plastic packaging"
)

prediction = response.predictions[0]
if prediction.status == "success":
    print(f"Recommended TIC: {prediction.tic_id} ({prediction.label})")
    print(f"  TIC Description: {prediction.tic_description}")
    print(f"  Product Description: {prediction.product_description}")
else:
    print(f"Recommendation failed: {prediction.error}")

Calculate Cart Tax

Calculate sales tax for a shopping cart with multiple line items. CalculateCart uses dual-routing: when TaxCloud credentials are configured on the client, the request is automatically routed to the TaxCloud API; otherwise it is sent to the ZipTax API. The input is the same CalculateCartRequest in both cases, but the response type differs:

  • Without TaxCloud credentials -- returns a CalculateCartResponse (ZipTax API)
  • With TaxCloud credentials -- returns a TaxCloudCalculateCartResponse (TaxCloud API)
from ziptax.models import (
    CalculateCartRequest,
    CartItem,
    CartAddress,
    CartCurrency,
    CartLineItem,
)

# Build the cart request
request = CalculateCartRequest(
    items=[
        CartItem(
            customer_id="customer-453",
            currency=CartCurrency(currency_code="USD"),
            destination=CartAddress(
                address="200 Spectrum Center Dr, Irvine, CA 92618"
            ),
            origin=CartAddress(
                address="323 Washington Ave N, Minneapolis, MN 55401"
            ),
            line_items=[
                CartLineItem(
                    item_id="item-1",
                    price=10.75,
                    quantity=1.5,
                ),
                CartLineItem(
                    item_id="item-2",
                    price=25.00,
                    quantity=2.0,
                    taxability_code=0,
                ),
            ],
        )
    ]
)

# Calculate tax (routes to ZipTax or TaxCloud based on client config)
result = client.request.CalculateCart(request)

# Access results
cart = result.items[0]
print(f"Cart ID: {cart.cart_id}")
for item in cart.line_items:
    print(f"  {item.item_id}: rate={item.tax.rate}, amount=${item.tax.amount:.2f}")

Validation

The cart models enforce constraints at construction time via Pydantic:

  • items must contain exactly 1 cart
  • line_items must contain 1-250 items
  • price and quantity must be greater than 0
  • currency_code must be "USD"
from pydantic import ValidationError

try:
    CartLineItem(item_id="item-1", price=-5.00, quantity=1.0)
except ValidationError as e:
    print(e)  # price must be greater than 0

TaxCloud Order Management

The SDK includes optional support for TaxCloud order management features. To use these features, you need both a ZipTax API key and TaxCloud credentials (Connection ID and API Key).

Initialize Client with TaxCloud Support

from ziptax import ZipTaxClient

# Initialize with TaxCloud credentials
client = ZipTaxClient.api_key(
    api_key="your-ziptax-api-key",
    taxcloud_connection_id="25eb9b97-5acb-492d-b720-c03e79cf715a",
    taxcloud_api_key="your-taxcloud-api-key",
)

# TaxCloud features are now available via client.request

Create an Order

from ziptax.models import (
    CreateOrderRequest,
    TaxCloudAddress,
    CartItemWithTax,
    Tax,
    Currency,
)

# Prepare order request
order_request = CreateOrderRequest(
    order_id="my-order-1",
    customer_id="customer-453",
    transaction_date="2024-01-15T09:30:00Z",
    completed_date="2024-01-15T09:30:00Z",
    origin=TaxCloudAddress(
        line1="323 Washington Ave N",
        city="Minneapolis",
        state="MN",
        zip="55401-2427",
    ),
    destination=TaxCloudAddress(
        line1="323 Washington Ave N",
        city="Minneapolis",
        state="MN",
        zip="55401-2427",
    ),
    line_items=[
        CartItemWithTax(
            index=0,
            item_id="item-1",
            price=10.8,
            quantity=1.5,
            tax=Tax(amount=1.31, rate=0.0813),
        )
    ],
    currency=Currency(currency_code="USD"),
)

# Create the order
order = client.request.CreateOrder(order_request)
print(f"Created order: {order.order_id}")
print(f"Tax amount: ${order.line_items[0].tax.amount}")

Retrieve an Order

# Get an existing order by ID
order = client.request.GetOrder("my-order-1")

print(f"Order ID: {order.order_id}")
print(f"Customer ID: {order.customer_id}")
print(f"Completed Date: {order.completed_date}")
print(f"Total Tax: ${sum(item.tax.amount for item in order.line_items)}")

Update an Order

from ziptax.models import UpdateOrderRequest

# Update the order's completed date
update_request = UpdateOrderRequest(
    completed_date="2024-01-16T10:00:00Z"
)

updated_order = client.request.UpdateOrder("my-order-1", update_request)
print(f"Updated completed date: {updated_order.completed_date}")

Create a Refund

from ziptax.models import (
    RefundTransactionRequest,
    CartItemRefundWithTaxRequest,
)

# Partial refund - specify items and quantities
refund_request = RefundTransactionRequest(
    items=[
        CartItemRefundWithTaxRequest(
            item_id="item-1",
            quantity=1.0,
        )
    ]
)
refunds = client.request.RefundOrder("my-order-1", refund_request)
print(f"Refunded tax: ${refunds[0].items[0].tax.amount}")

# Full refund - omit items parameter
full_refunds = client.request.RefundOrder("my-order-2")
print("Full refund created")

Create an Order from Cart

If you've already calculated cart tax via CalculateCart with TaxCloud credentials, you can convert that cart directly into a finalized order using the returned cart_id:

from ziptax.models import CreateOrderFromCartRequest, UpdateOrderRequest

# Use the cart_id from a previous CalculateCart response
request = CreateOrderFromCartRequest(
    cart_id="ce4a1234-5678-90ab-cdef-1234567890ab",
    order_id="my-order-1",
)

order = client.request.CreateOrderFromCart(request)
print(f"Created order: {order.order_id}")
print(f"Transaction date: {order.transaction_date}")
print(f"Tax amount: ${order.line_items[0].tax.amount}")

# TaxCloud automatically commits the order at creation time.
# To set a completed date, use UpdateOrder after creation:
update_request = UpdateOrderRequest(completed_date="2024-01-16T10:00:00Z")
updated = client.request.UpdateOrder(order.order_id, update_request)

TaxCloud Error Handling

from ziptax import ZipTaxCloudConfigError

try:
    # Attempt to use TaxCloud feature without credentials
    order = client.request.GetOrder("my-order-1")

except ZipTaxCloudConfigError as e:
    # TaxCloud credentials not configured
    print(f"TaxCloud error: {e.message}")
    print("Please provide taxcloud_connection_id and taxcloud_api_key")

except ZipTaxNotFoundError as e:
    # Order not found
    print(f"Order not found: {e.message}")

Configuration

You can configure the client using dict-style access:

client = ZipTaxClient.api_key("your-api-key-here")

# Set configuration options
client.config["format"] = "json"
client.config["timeout"] = 60

# Get configuration options
timeout = client.config["timeout"]

Error Handling

The SDK provides comprehensive error handling with specific exception types:

from ziptax import (
    ZipTaxClient,
    ZipTaxValidationError,
    ZipTaxAuthenticationError,
    ZipTaxRateLimitError,
    ZipTaxServerError,
    ZipTaxError,
)

client = ZipTaxClient.api_key("your-api-key-here")

try:
    response = client.request.GetSalesTaxByAddress("123 Main St")

except ZipTaxValidationError as e:
    # Input validation errors
    print(f"Validation error: {e.message}")

except ZipTaxAuthenticationError as e:
    # Authentication failures (401)
    print(f"Authentication error: {e.message}")

except ZipTaxRateLimitError as e:
    # Rate limit exceeded (429)
    print(f"Rate limit error: {e.message}")
    if e.retry_after:
        print(f"Retry after {e.retry_after} seconds")

except ZipTaxServerError as e:
    # Server errors (5xx)
    print(f"Server error: {e.message}")

except ZipTaxError as e:
    # General Ziptax errors
    print(f"Ziptax error: {e.message}")

Exception Hierarchy

ZipTaxError
โ”œโ”€โ”€ ZipTaxAPIError
โ”‚   โ”œโ”€โ”€ ZipTaxAuthenticationError (401)
โ”‚   โ”œโ”€โ”€ ZipTaxAuthorizationError (403)
โ”‚   โ”œโ”€โ”€ ZipTaxNotFoundError (404)
โ”‚   โ”œโ”€โ”€ ZipTaxRateLimitError (429)
โ”‚   โ””โ”€โ”€ ZipTaxServerError (5xx)
โ”œโ”€โ”€ ZipTaxValidationError
โ”œโ”€โ”€ ZipTaxConnectionError
โ”œโ”€โ”€ ZipTaxTimeoutError
โ”œโ”€โ”€ ZipTaxRetryError
โ””โ”€โ”€ ZipTaxCloudConfigError (TaxCloud credentials not configured)

Async Operations

For concurrent operations, you can use asyncio with the SDK:

import asyncio
from concurrent.futures import ThreadPoolExecutor
from ziptax import ZipTaxClient

async def get_tax_rates_async(client, addresses):
    loop = asyncio.get_event_loop()
    with ThreadPoolExecutor() as executor:
        tasks = [
            loop.run_in_executor(
                executor,
                client.request.GetSalesTaxByAddress,
                address
            )
            for address in addresses
        ]
        return await asyncio.gather(*tasks)

# Usage
client = ZipTaxClient.api_key("your-api-key-here")
addresses = ["123 Main St, CA", "456 Oak Ave, NY"]
responses = asyncio.run(get_tax_rates_async(client, addresses))

See examples/async_usage.py for more examples.

Response Models

All API responses are validated using Pydantic models:

V60Response

class V60Response:
    metadata: V60Metadata                           # Response metadata with code/message
    base_rates: Optional[List[V60BaseRate]]        # Tax rates by jurisdiction
    service: Optional[V60Service]                   # Service taxability (None for some regions)
    shipping: Optional[V60Shipping]                 # Shipping taxability (None for some regions)
    sourcing_rules: Optional[V60SourcingRules]     # Origin/Destination rules
    tax_summaries: Optional[List[V60TaxSummary]]   # Tax summaries with display rates
    address_detail: V60AddressDetail                # Address details

V60Metadata

class V60Metadata:
    version: str                    # API version (e.g., "v60")
    response: V60ResponseInfo       # Response info object

class V60ResponseInfo:
    code: int                       # Response code (100 = success)
    name: str                       # Response code name
    message: str                    # Response message
    definition: str                 # Schema definition URL

V60TaxSummary

class V60TaxSummary:
    rate: float                                    # Summary tax rate
    tax_type: str                                  # Tax type (e.g., "SALES_TAX")
    summary_name: str                              # Summary description
    display_rates: List[V60DisplayRate]           # Display rates breakdown

class V60DisplayRate:
    name: str                                      # Display rate name
    rate: float                                    # Display rate value

V60AccountMetrics

class V60AccountMetrics:
    request_count: int       # Number of API requests made
    request_limit: int       # Maximum allowed API requests
    usage_percent: float     # Percentage of request limit used
    is_active: bool          # Whether the account is currently active
    message: str             # Account status or informational message

Note: Uses extra="allow" to accept any additional fields the API may return.

ProductCodeSearchResponse

class ProductCodeSearchResponse:
    query: str                                          # The original search query
    results: List[ProductCodeSearchResult]              # Ranked results

class ProductCodeSearchResult:
    tic_id: int             # Taxability Information Code (parsed from string)
    label: str              # TIC label
    natural_label: str      # Natural language label
    description: str        # Full TIC description
    documentation: str      # Long-form TIC documentation
    rank: int               # Result rank (1 = best match, parsed from string)
    score: float            # Confidence score 0.0-1.0 (parsed from string)

ProductCodeRecommendationResponse

class ProductCodeRecommendationResponse:
    predictions: List[ProductCodeRecommendation]        # AI recommendations

class ProductCodeRecommendation:
    status: str                    # "success" or "fail"
    error: Optional[str]           # Error message when status is "fail"
    tic_id: int                    # Recommended TIC (parsed from string)
    label: str                     # TIC label
    natural_label: str             # Natural language label
    tic_description: str           # Full TIC description
    product_description: str       # Original product description from query

See the models documentation for complete model definitions.

Development

Setup

# Clone the repository
git clone https://github.com/ziptax/ziptax-python.git
cd ziptax-python

# Install dependencies
pip install -e ".[dev]"

Running Tests

# Run all tests
pytest

# Run with coverage
pytest --cov=src/ziptax --cov-report=html

# Run specific test file
pytest tests/test_client.py

Code Quality

# Format code
black src/ tests/

# Lint code
ruff src/ tests/

# Type checking
mypy src/

Examples

See the examples/ directory for complete examples:

API Reference

ZipTaxClient

Main client for interacting with the Ziptax API.

Methods

  • api_key(api_key, **kwargs) - Create a client instance with an API key
  • close() - Close the HTTP client session

Properties

  • config - Configuration object (dict-like access)
  • request - Functions object for making API requests

Functions

API endpoint functions accessible via client.request.

ZipTax API Methods

  • GetSalesTaxByAddress(address, **kwargs) - Get tax rates by address
  • GetSalesTaxByGeoLocation(lat, lng, **kwargs) - Get tax rates by coordinates
  • GetRatesByPostalCode(postal_code, **kwargs) - Get tax rates by US postal code
  • GetAccountMetrics(**kwargs) - Get account usage metrics
  • SearchProductCodes(query) - Search for product codes (TICs) by description
  • RecommendProductCode(query) - Get an AI-powered TIC recommendation
  • CalculateCart(request) - Calculate sales tax for a shopping cart

TaxCloud API Methods (Optional)

Requires taxcloud_connection_id and taxcloud_api_key in client initialization.

  • CreateOrder(request, **kwargs) - Create an order in TaxCloud
  • CreateOrderFromCart(request) - Create an order from a previously calculated cart
  • GetOrder(order_id) - Retrieve an order by ID
  • UpdateOrder(order_id, request) - Update an order's completed date
  • RefundOrder(order_id, request) - Create a full or partial refund

Requirements

  • Python 3.8+
  • requests >= 2.28.0
  • pydantic >= 2.0.0

License

This project is licensed under the MIT License - see the LICENSE file for details.

Support

Contributing

Contributions are welcome! Please feel free to submit a Pull Request.

  1. Fork the repository
  2. Create your feature branch (git checkout -b feature/amazing-feature)
  3. Make your changes and write tests
  4. Bump the version using python scripts/bump_version.py patch (or minor/major)
  5. Update CHANGELOG.md with your changes
  6. Commit your changes (git commit -m 'Add some amazing feature')
  7. Push to the branch (git push origin feature/amazing-feature)
  8. Open a Pull Request

Note: All PRs require a version bump. See docs/VERSIONING.md for details on our versioning strategy.

Changelog

See CHANGELOG.md for version history and changes.


Made with โค๏ธ by the Ziptax Team

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

ziptax_sdk-0.2.6b0.tar.gz (46.4 kB view details)

Uploaded Source

Built Distribution

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

ziptax_sdk-0.2.6b0-py3-none-any.whl (31.8 kB view details)

Uploaded Python 3

File details

Details for the file ziptax_sdk-0.2.6b0.tar.gz.

File metadata

  • Download URL: ziptax_sdk-0.2.6b0.tar.gz
  • Upload date:
  • Size: 46.4 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for ziptax_sdk-0.2.6b0.tar.gz
Algorithm Hash digest
SHA256 a322665bf01835c8df4a293c8e6933e0a15c9f4bbeaa727c5f72d19f05492e81
MD5 03dd2569b78bcac5c685d466e36e843e
BLAKE2b-256 cdf8673a905f85035d555e50ee17dd96a8ea728672f19367dbc3181ef475183e

See more details on using hashes here.

Provenance

The following attestation bundles were made for ziptax_sdk-0.2.6b0.tar.gz:

Publisher: publish.yml on ZipTax/ziptax-python

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file ziptax_sdk-0.2.6b0-py3-none-any.whl.

File metadata

  • Download URL: ziptax_sdk-0.2.6b0-py3-none-any.whl
  • Upload date:
  • Size: 31.8 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for ziptax_sdk-0.2.6b0-py3-none-any.whl
Algorithm Hash digest
SHA256 2132d362b673f64cadde0c5e4380e1ffd2f1433d7d87c486b8da6a58892b6f0b
MD5 60b4a98eec8cfbc87907ede145434888
BLAKE2b-256 f6c7971ce27c340b84c43ae3b4335ff9b22042d43e8e5fc744a2e01a39fed356

See more details on using hashes here.

Provenance

The following attestation bundles were made for ziptax_sdk-0.2.6b0-py3-none-any.whl:

Publisher: publish.yml on ZipTax/ziptax-python

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

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