Skip to main content

Django adapter for python-sendparcel

Project description

django-sendparcel

PyPI Python Version Django Version License

Django adapter for the python-sendparcel multi-carrier shipping library.

Alpha (0.1.0) — API may change between minor releases. Pin your dependency if you use it in production.

Features

  • Shipment model with FSM — built-in Shipment model with finite-state-machine transitions (new → created → label_ready → in_transit → delivered, etc.)
  • Swappable Shipment model — replace the default Shipment with your own via swapper, similar to Django's AUTH_USER_MODEL
  • Protocol adapterDjangoShipmentAdapter bridges the Django Shipment model to the framework-agnostic core
  • Django ORM repositoryDjangoShipmentRepository provides async-compatible persistence via sync_to_async
  • Provider plugin registry — auto-discovers shipping provider plugins at app startup
  • Callback endpoint — receives provider status webhooks and routes them through ShipmentFlow
  • Admin integrationShipmentAdmin with list filters, search, and bulk actions (mark in transit, mark delivered, cancel)
  • Exception middlewareSendParcelExceptionMiddleware maps sendparcel exceptions to appropriate HTTP status codes
  • Provider choice formProviderChoiceForm dynamically populated from the plugin registry
  • Callback retry persistenceCallbackRetry model stores failed callback attempts for later reprocessing

Installation

Install with pip (or your preferred package manager):

pip install django-sendparcel

This will also install the required dependencies: python-sendparcel, Django, anyio, and swapper.

Quick Start

1. Add to INSTALLED_APPS

INSTALLED_APPS = [
    # ...
    "sendparcel_django",
    # ...
]

2. Configure settings

# Provider-specific configuration, keyed by provider slug
SENDPARCEL_PROVIDER_SETTINGS = {
    "my-provider": {
        "api_url": "https://api.example.com/",
        "api_key": "your-api-key",
    },
}

# Default provider slug (optional)
SENDPARCEL_DEFAULT_PROVIDER = "my-provider"

# Custom shipment model (optional, default: "sendparcel_django.Shipment")
# Uses django-swapper convention: <APP_LABEL>_<MODEL_NAME>
SENDPARCEL_DJANGO_SHIPMENT_MODEL = "myapp.Shipment"

3. (Optional) Create a custom Shipment model

If you need additional fields on the Shipment, extend ShipmentModelMixin and point the setting to your model:

from django.db import models
from sendparcel_django.models import ShipmentModelMixin


class Shipment(ShipmentModelMixin):
    # Add custom fields as needed
    notes = models.TextField(blank=True, default="")

    class Meta:
        verbose_name = "shipment"

Then in settings:

SENDPARCEL_DJANGO_SHIPMENT_MODEL = "myapp.Shipment"

4. Include URL configuration

from django.urls import include, path

urlpatterns = [
    # ...
    path("sendparcel/", include("sendparcel_django.urls")),
]

This exposes the callback endpoint at sendparcel/callback/<shipment_id>/ for receiving provider webhooks.

Successful callback responses include provider, status, shipment, and update. The adapter no longer persists label URLs on shipment models.

5. (Optional) Add the exception middleware

MIDDLEWARE = [
    # ...
    "sendparcel_django.middleware.SendParcelExceptionMiddleware",
]

This catches sendparcel exceptions and returns appropriate JSON error responses:

Exception HTTP Status
CommunicationError 502
ProviderNotFoundError 404
ProviderCapabilityError 409
InvalidCallbackError 400
InvalidTransitionError 409
SendParcelException 400

6. Run migrations

python manage.py migrate

Usage

Creating a shipment

Use ShipmentFlow to create shipments with explicit address and parcel data:

import anyio
from sendparcel.flow import ShipmentFlow
from sendparcel_django.repository import DjangoShipmentRepository


async def create_shipment(provider_slug):
    repository = DjangoShipmentRepository()
    flow = ShipmentFlow(
        repository=repository,
        config=settings.SENDPARCEL_PROVIDER_SETTINGS,
    )

    outcome = await flow.create_shipment(
        provider_slug,
        sender_address={
            "name": "My Warehouse",
            "line1": "1 Warehouse St",
            "city": "Warsaw",
            "postal_code": "00-001",
            "country_code": "PL",
        },
        receiver_address={
            "name": "Customer Name",
            "line1": "10 Customer Ave",
            "city": "Krakow",
            "postal_code": "30-001",
            "country_code": "PL",
        },
        parcels=[{"weight_kg": 2.5}],
        reference_id="my-order-123",  # optional reference for your system
    )

    shipment = outcome.shipment
    if outcome.label is None:
        label_outcome = await flow.create_label(shipment)
        return label_outcome.shipment

    return shipment

Call from synchronous Django code using anyio.run():

shipment = anyio.run(create_shipment, "my-provider")

Provider choice form

Use ProviderChoiceForm to let users select a shipping provider:

from sendparcel_django.forms import ProviderChoiceForm

form = ProviderChoiceForm(request.POST)
if form.is_valid():
    provider_slug = form.cleaned_data["provider"]

The form choices are dynamically populated from the plugin registry.

Admin

The ShipmentAdmin is auto-registered for the active Shipment model (default or swapped). It provides:

  • List display: ID, reference ID, status, provider, tracking number, creation date
  • Filters: status, provider
  • Search: tracking number, external ID, reference ID
  • Bulk actions: mark as in transit, mark as delivered, cancel — each action triggers FSM transitions with guard validation

Configuration Reference

All settings are read from your Django settings module.

Setting Type Default Description
SENDPARCEL_PROVIDER_SETTINGS dict {} Provider-specific configuration, keyed by provider slug
SENDPARCEL_DEFAULT_PROVIDER str "" Default provider slug
SENDPARCEL_DJANGO_SHIPMENT_MODEL str "sendparcel_django.Shipment" Dotted path to the Shipment model (swappable via django-swapper)

Settings are resolved at call time via sendparcel_django.conf.get_settings(), so @override_settings works correctly in tests.

Shipment Model Fields

The ShipmentModelMixin provides these fields on every Shipment (default or custom):

Field Type Description
reference_id CharField Your system's reference (e.g. order ID)
provider CharField Provider slug
status CharField Current FSM state (default: "new")
external_id CharField Provider-assigned shipment ID
tracking_number CharField Tracking number from provider
created_at DateTimeField Auto-set on creation
updated_at DateTimeField Auto-set on save

The default concrete Shipment model uses these fields directly. When creating a custom model, you can add any additional fields you need.

Example Project

A full working example is included in the example/ directory. It demonstrates:

  • A custom Shipment model with inline address fields
  • Shipment creation through the ShipmentFlow
  • A delivery simulation provider for local testing
  • HTMX-powered shipment tracking UI

To run the example:

cd example
pip install -e ..
pip install -e ../../python-sendparcel
python manage.py migrate
python manage.py runserver

Supported Versions

Dependency Version
Python >= 3.12
Django >= 5.2
python-sendparcel >= 0.1.1
anyio >= 4.0
swapper >= 1.4

Running Tests

The test suite uses pytest with pytest-django:

pip install -e ".[dev]"
pytest

Test configuration is in tests/settings.py. The test suite covers models, protocols, views, middleware, admin, forms, registry, repository, FSM integration, and callback retry logic.

Credits

License

MIT License. See LICENSE for details.

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

django_sendparcel-0.1.1.tar.gz (44.3 kB view details)

Uploaded Source

File details

Details for the file django_sendparcel-0.1.1.tar.gz.

File metadata

  • Download URL: django_sendparcel-0.1.1.tar.gz
  • Upload date:
  • Size: 44.3 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.10.3 {"installer":{"name":"uv","version":"0.10.3","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Manjaro Linux","version":null,"id":null,"libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}

File hashes

Hashes for django_sendparcel-0.1.1.tar.gz
Algorithm Hash digest
SHA256 04d9108565f446a0a285d674be167710951b3d76556dcfa380511c938f291270
MD5 14218ac4836fdb266c506b6a98c30e20
BLAKE2b-256 bf2ce76508ab3f69ec6a8e36c62d24a5ab5420bb646a22a59652f044d04e5f6b

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