Metadata-Version: 2.4
Name: usps-v3
Version: 1.0.2
Summary: USPS v3 REST API client — OAuth 2.0, address validation, tracking, labels, rates. Drop-in replacement for usps-api (retired Jan 2026).
Project-URL: Homepage, https://revaddress.com
Project-URL: Documentation, https://revaddress.com/docs
Project-URL: Repository, https://github.com/revereveal/usps-v3
Project-URL: Issues, https://github.com/revereveal/usps-v3/issues
Author-email: James Lambert <james@revasser.nyc>
License-Expression: MIT
License-File: LICENSE
Keywords: address-validation,address-verification,api,ecommerce,labels,logistics,oauth2,package-tracking,postage,postal,rates,rest-api,shipping,shipping-labels,shipping-rates,tracking,usps,usps-api,usps-rest-api,usps-v3,usps-webtools,web-tools
Classifier: Development Status :: 5 - Production/Stable
Classifier: Intended Audience :: Developers
Classifier: License :: OSI Approved :: MIT License
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3.9
Classifier: Programming Language :: Python :: 3.10
Classifier: Programming Language :: Python :: 3.11
Classifier: Programming Language :: Python :: 3.12
Classifier: Programming Language :: Python :: 3.13
Classifier: Topic :: Office/Business :: Financial :: Point-Of-Sale
Classifier: Topic :: Software Development :: Libraries :: Python Modules
Requires-Python: >=3.9
Requires-Dist: httpx>=0.24.0
Provides-Extra: dev
Requires-Dist: pytest-httpx>=0.21; extra == 'dev'
Requires-Dist: pytest>=7.0; extra == 'dev'
Requires-Dist: respx>=0.20; extra == 'dev'
Requires-Dist: ruff>=0.3; extra == 'dev'
Description-Content-Type: text/markdown

# usps-v3

[![PyPI](https://img.shields.io/pypi/v/usps-v3)](https://pypi.org/project/usps-v3/)
[![License: MIT](https://img.shields.io/badge/License-MIT-blue.svg)](LICENSE)
[![RevAddress](https://img.shields.io/badge/Managed%20API-RevAddress-6366f1)](https://revaddress.com)

USPS v3 REST API client — OAuth 2.0, address validation, tracking, labels, rates. Drop-in replacement for the `usps-api` PyPI package (retired January 2026).

Direct USPS integration. OAuth 2.0. No middleman. No per-label fees.

> **Migrating from `usps-api`?** That package (13,700+ downloads/month) uses the retired USPS Web Tools XML API (shut down January 25, 2026). This SDK uses the new v3 REST API with OAuth 2.0. See [Migration from usps-api](#migration-from-usps-api) below.

> **Don't want to manage USPS credentials?** [RevAddress](https://revaddress.com) provides a managed USPS v3 API with flat monthly pricing, rate limit handling, and BYOK support. [Get a free sandbox key](https://revaddress.com/signup/) — no credit card required.

## Migrating from EasyPost or USPS Web Tools?

RevAddress provides a managed USPS v3 API with flat monthly pricing — no per-label fees. If you're migrating from EasyPost (shutting down March 17, 2026) or the legacy USPS Web Tools XML API:

- **[Migration Guide](https://revaddress.com/blog/usps-migration-guide/)** — Step-by-step from XML to REST
- **[EasyPost vs RevAddress](https://revaddress.com/blog/easypost-vs-revaddress/)** — Feature and pricing comparison
- **[Endpoint Mapping](https://revaddress.com/blog/usps-web-tools-endpoint-mapping/)** — Every legacy endpoint mapped to v3

Save 81% vs EasyPost at 5,000 labels/mo ($79/mo flat vs $420 in per-label fees). [Get started →](https://revaddress.com/signup/)

## Install

```bash
pip install usps-v3
```

## Quick Start

```python
from usps_v3 import Client

# Credentials from USPS Business Customer Gateway
client = Client(client_id="your-id", client_secret="your-secret")
# Or set USPS_CLIENT_ID and USPS_CLIENT_SECRET environment variables:
# client = Client()

# Validate an address (FREE)
result = client.addresses.validate(
    street_address="1600 Pennsylvania Ave NW",
    city="Washington",
    state="DC",
    zip_code="20500",
)
print(result["address"]["ZIPPlus4"])  # "0005"

# Track a package (FREE)
info = client.tracking.track("9400111899223033005282")
print(info["statusCategory"])  # "Delivered"

# Get delivery time estimates (FREE)
standards = client.standards.estimates("10001", "90210")

# Find drop-off locations (FREE)
locations = client.locations.dropoff("20500", mail_class="PRIORITY_MAIL")

# Get rate quotes
rates = client.prices.domestic("10001", "90210", weight=2.5)
print(rates["rates"]["rateOptions"][0]["totalPrice"])

# International rates
intl = client.prices.international("10001", "CA", weight=3.0)

# Create shipping labels (requires USPS enrollment + COP claims)
label = client.labels.create(
    from_address={"streetAddress": "123 Sender St", "city": "New York", "state": "NY", "ZIPCode": "10001"},
    to_address={"streetAddress": "456 Recipient Ave", "city": "LA", "state": "CA", "ZIPCode": "90001"},
    mail_class="PRIORITY_MAIL",
    weight=2.0,
)
print(label["trackingNumber"])

# Void a label
client.labels.void("9400111899223033005282")
```

## Features

| Feature | Endpoint | Auth Required |
|---------|----------|--------------|
| Address Validation | `addresses.validate()` | OAuth only |
| City/State Lookup | `addresses.city_state()` | OAuth only |
| Package Tracking | `tracking.track()` | OAuth only |
| Service Standards | `standards.estimates()` | OAuth only |
| Drop-off Locations | `locations.dropoff()` | OAuth only |
| Domestic Prices | `prices.domestic()` | OAuth only |
| International Prices | `prices.international()` | OAuth only |
| Label Creation | `labels.create()` | OAuth + Payment Auth |
| Label Void | `labels.void()` | OAuth only |

## Authentication

The SDK handles OAuth 2.0 token lifecycle automatically:

- **Token caching**: Tokens are cached in memory and on disk (`~/.usps-v3/tokens.json`)
- **Auto-refresh**: Tokens refresh automatically 30 minutes before expiry
- **Thread-safe**: Safe for concurrent use across threads

### Getting Credentials

1. Register at [USPS Business Customer Gateway](https://gateway.usps.com)
2. Create an application in the API developer portal
3. Note your `client_id` and `client_secret`

For label creation, you also need:
- **CRID** (Customer Registration ID)
- **MIDs** (Mailer IDs — master + label owner)
- **EPA** (Enterprise Payment Account)
- **COP claims linking** at [cop.usps.com](https://cop.usps.com)

See our [USPS CRID/MID enrollment guide](https://revaddress.com/blog/usps-crid-mid-enrollment/) for step-by-step instructions.

```python
client = Client(
    client_id="...",
    client_secret="...",
    crid="56982563",
    master_mid="904128936",
    label_mid="904128937",
    epa_account="1000405525",
)
```

## Error Handling

```python
from usps_v3 import Client, AuthError, ValidationError, RateLimitError, APIError

try:
    result = client.addresses.validate(street_address="123 Main St")
except ValidationError as e:
    print(f"Bad input: {e} (field: {e.field})")
except RateLimitError as e:
    print(f"Rate limited — retry after {e.retry_after}s")
except AuthError as e:
    print(f"Auth failed: {e}")
except APIError as e:
    print(f"USPS error ({e.status_code}): {e}")
```

## USPS Rate Limits

The v3 API defaults to **60 requests/hour** (down from unlimited in Web Tools). The SDK does not enforce this limit — USPS returns 429 when exceeded.

To request a higher limit, contact USPS at [emailus.usps.com](https://emailus.usps.com). See our [USPS rate limits guide](https://revaddress.com/blog/usps-rate-limits/) for details.

## Migration from usps-api

The `usps-api` PyPI package (13,700+ downloads/month) uses the retired USPS Web Tools XML API (shut down January 25, 2026). If you're migrating from `usps-api` or any Web Tools XML integration:

| Web Tools (XML) | v3 SDK (Python) |
|-----------------|-----------------|
| `<AddressValidateRequest>` | `client.addresses.validate(...)` |
| `<CityStateLookupRequest>` | `client.addresses.city_state(...)` |
| `<TrackFieldRequest>` | `client.tracking.track(...)` |
| `<RateV4Request>` | `client.prices.domestic(...)` |
| User ID auth | OAuth 2.0 (automatic) |
| XML response parsing | Python dicts (automatic) |
| Unlimited requests | 60/hr default (request increase) |

See our [complete migration guide](https://revaddress.com/blog/usps-migration-guide/) and [Web Tools endpoint mapping](https://revaddress.com/blog/usps-web-tools-endpoint-mapping/) for detailed migration instructions.

## RevAddress Managed API

If you'd rather not manage USPS OAuth credentials, rate limits, and enrollment yourself, **[RevAddress](https://revaddress.com)** offers a managed REST API:

- **Drop-in USPS v3 API** — same endpoints, managed OAuth
- **Flat monthly pricing** — no per-label fees ([from $29/mo](https://revaddress.com/pricing/))
- **Rate limit handling** — 120-600 req/min depending on plan
- **BYOK support** — bring your own USPS credentials
- **208 tests, 38 routes** — production-grade infrastructure

[Get a free sandbox key](https://revaddress.com/signup/) — address validation, tracking, and rate shopping included. No credit card required.

## Development

```bash
git clone https://github.com/revereveal/usps-v3.git
cd usps-v3
pip install -e ".[dev]"
pytest -v
```

## Links

- [Node.js SDK](https://github.com/revereveal/usps-v3-node) — same API, Node.js edition
- [RevAddress API](https://revaddress.com) — managed USPS API with BYOK support
- [RevAddress Docs](https://revaddress.com/docs/) — API reference and guides
- [USPS v3 API Docs](https://developer.usps.com/api/81)
- [PyPI Package](https://pypi.org/project/usps-v3/)

## License

MIT — see [LICENSE](LICENSE).

Built by [RevAddress](https://revaddress.com) — managed USPS API infrastructure for developers and businesses.
