Skip to main content

Shared Python toolkit for OuEstCharlie agents

Project description

OuEstCharlie Python Toolkit

Shared Python library for building OuEstCharlie photo management agents.

Overview

This toolkit provides three core capabilities:

  1. MCP integration — MCP server lifecycle, tool registration, progress reporting, and logging
  2. Manifest read-edit with consistency — hierarchical manifest traversal, atomic read-modify-write with optimistic concurrency
  3. XMP read-edit with consistency — sidecar read-modify-write with optimistic concurrency and field-level semantics

Package Structure

ouestcharlie-toolkit/
├── pyproject.toml
├── hatch_build.py                # Build hook: compiles image-proc and bundles it in the wheel
├── image-proc/                   # Rust CLI: decode + resize + AVIF/JPEG assembly
│   ├── Cargo.toml
│   └── src/main.rs
└── src/
    └── ouestcharlie_toolkit/
        ├── bin/                  # Bundled image-proc binary (populated at build time)
        ├── schema.py             # Data models, exceptions, constants
        ├── backend.py            # Backend protocol
        ├── backends/
        │   └── local.py          # Local filesystem backend
        ├── manifest.py           # ManifestStore for manifest operations
        ├── xmp.py                # XmpStore for XMP sidecar operations
        ├── thumbnail_builder.py  # Thumbnail generation (delegates to image-proc)
        ├── progress.py           # ProgressReporter for MCP progress
        └── server.py             # AgentBase for MCP server lifecycle

Installation

From PyPI (recommended)

pip install ouestcharlie-toolkit

The image-proc binary is compiled and bundled inside the wheel at publish time — no Rust toolchain required at install time.

System prerequisites:

  • macOS: brew install inih (required by pyexiv2 at runtime)
  • Linux/Windows: no extra steps — pyexiv2 and image-proc wheels are self-contained

From source (development)

System prerequisites:

brew install inih nasm    # macOS — inih for pyexiv2, nasm for rav1e AVIF encoder
sudo apt install nasm     # Linux
choco install nasm        # Windows

Create virtual environment and install dependencies:

uv venv --python 3.13
uv pip install -e ".[dev]"

The image-proc binary is not compiled automatically in editable installs. Build it once:

cd image-proc && cargo build --release
# binary: image-proc/target/release/image-proc

With optional features:

cargo build --release --features raw    # RAW format support (pure Rust, no extra deps)
cargo build --release --features heic   # HEIC support (requires brew install libheif)

The toolkit resolves the binary in this order:

  1. IMAGE_PROC_BINARY environment variable
  2. bin/image-proc[.exe] bundled inside the installed wheel
  3. image-proc on $PATH
  4. image-proc/target/release/image-proc relative to this repo (dev build)

Optional features (RAW and HEIC)

To build with RAW or HEIC support, set env vars before hatch build or cargo build:

IMAGE_PROC_FEATURE_RAW=1 hatch build   # enables rawler (pure Rust RAW decoder)
IMAGE_PROC_FEATURE_HEIC=1 hatch build  # enables libheif-rs (requires brew install libheif)

Running Tests

Always use .venv/bin/python -m pytest — do not use .venv/bin/pytest or a system python:

# Run all tests
.venv/bin/python -m pytest tests/ -v

# Run a specific file
.venv/bin/python -m pytest tests/test_photo.py -v --tb=short

Why: pytest on PATH or uv run pytest may resolve to the wrong Python or fail on native dependencies.

To run the Rust tests:

cd image-proc && cargo test

Building a Wheel

The hatch_build.py hook compiles image-proc and bundles the binary inside the wheel:

pip install hatch
hatch build
# produces dist/ouestcharlie_toolkit-*.whl (platform-specific)

Set env vars to enable optional features:

IMAGE_PROC_FEATURE_RAW=1 hatch build
IMAGE_PROC_FEATURE_HEIC=1 hatch build

Dependencies

  • mcp>=1.0 — Official MCP Python SDK
  • pyexiv2>=2.8 — EXIF extraction from image files (wraps Exiv2); requires brew install inih on macOS
  • blake3>=1.0.8 — Fast content hashing

image-proc (Rust binary, bundled in the wheel) handles all image decoding, resizing, AVIF assembly, and JPEG preview generation.

XMP parsing and serialization use stdlib only and have no native dependencies.

Usage

Creating an Agent

from ouestcharlie_toolkit import AgentBase

class HousekeepingAgent(AgentBase):
    def __init__(self):
        super().__init__(name="ouestcharlie-housekeeping", version="1.0.0")

        # Register tools using the FastMCP instance
        @self.mcp.tool()
        async def rebuild_partition(backend: str, partition: str, mode: str = "lazy"):
            """Rebuild partition manifest and thumbnails."""
            # Agent logic here
            photos = await self.backend.list_files(partition, suffix=".jpg")
            progress = self.progress(total=len(photos))

            for photo in photos:
                await self.check_cancelled()
                # Process photo...
                await progress.advance(message=f"Processing {photo.path}")

            return {"photosProcessed": len(photos), "errors": 0}

if __name__ == "__main__":
    agent = HousekeepingAgent()
    agent.run()  # Runs on stdio transport

Working with Manifests

from ouestcharlie_toolkit import ManifestStore, PhotoEntry

# Read-modify-write pattern
async def add_photo_to_manifest(store: ManifestStore, partition: str, photo: PhotoEntry):
    def modify(manifest):
        manifest.photos.append(photo)
        # Recompute summary...
        return manifest

    await store.read_modify_write_leaf(partition, modify)

Working with XMP Sidecars

from ouestcharlie_toolkit import XmpStore

# Read-modify-write pattern
async def add_face_tags(store: XmpStore, photo_path: str, faces: list[str]):
    def modify(xmp):
        for face in faces:
            tag = f"ouestcharlie:faces/{face}"
            if tag not in xmp.tags:
                xmp.tags.append(tag)
        return xmp

    await store.read_modify_write(photo_path, modify)

Backend Configuration

The toolkit reads backend configuration from the WOOF_BACKEND_CONFIG environment variable:

export WOOF_BACKEND_CONFIG='{"type": "filesystem", "root": "/Users/alice/Photos"}'

Implementation Status

✅ Completed

  • Package structure and build configuration
  • Backend protocol and local filesystem implementation
  • ManifestStore with optimistic concurrency
  • XmpStore with optimistic concurrency
  • ProgressReporter
  • AgentBase with MCP server lifecycle
  • Thumbnail generation: per-partition AVIF grid via avif-grid Rust binary
    • Parallel decode (rayon) for JPEG, PNG, WebP, TIFF
    • Orientation correction (TIFF values 1–8)
    • Crop and pad fit modes
    • Stubbed RAW (--features raw) and HEIC (--features heic) support

📋 Future Work

  • Cloud backend implementations (S3, GCS, ADLS Gen2, OneDrive, Kdrive)
  • Bloom filter implementation for partition summaries

Architecture

See ouestcharlie/agent/agent_LLD_rationale.md for technology selection rationale.

Key design principles:

  • Optimistic concurrency — All manifest and XMP writes use version tokens to detect conflicts
  • Unknown field preservation — Schema evolution via _extra dict in dataclasses
  • Async throughout — All I/O operations are async
  • Backend abstraction — Swappable storage backends (local, S3, GCS, etc.)
  • MCP-native — Built on FastMCP for clean agent implementation

References

License

MIT license

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

ouestcharlie_py_toolkit-0.2.1.tar.gz (1.2 MB view details)

Uploaded Source

Built Distributions

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

ouestcharlie_py_toolkit-0.2.1-cp312-cp312-win_amd64.whl (1.6 MB view details)

Uploaded CPython 3.12Windows x86-64

ouestcharlie_py_toolkit-0.2.1-cp312-cp312-manylinux_2_34_x86_64.whl (1.8 MB view details)

Uploaded CPython 3.12manylinux: glibc 2.34+ x86-64

ouestcharlie_py_toolkit-0.2.1-cp312-cp312-macosx_15_0_x86_64.whl (1.8 MB view details)

Uploaded CPython 3.12macOS 15.0+ x86-64

ouestcharlie_py_toolkit-0.2.1-cp312-cp312-macosx_15_0_arm64.whl (1.5 MB view details)

Uploaded CPython 3.12macOS 15.0+ ARM64

File details

Details for the file ouestcharlie_py_toolkit-0.2.1.tar.gz.

File metadata

  • Download URL: ouestcharlie_py_toolkit-0.2.1.tar.gz
  • Upload date:
  • Size: 1.2 MB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for ouestcharlie_py_toolkit-0.2.1.tar.gz
Algorithm Hash digest
SHA256 e8fb0d10d036be8c57a4e40fed75ac6b2f11e25614f300403bc4f319359bfa94
MD5 bc4400b4a0adde66f17d64929784dd8c
BLAKE2b-256 256c8ac7645b6068c90cb42999410bb3789070fcd0b291015891a5513f599bbb

See more details on using hashes here.

Provenance

The following attestation bundles were made for ouestcharlie_py_toolkit-0.2.1.tar.gz:

Publisher: publish.yml on ouestcharlie/ouestcharlie-py-toolkit

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

File details

Details for the file ouestcharlie_py_toolkit-0.2.1-cp312-cp312-win_amd64.whl.

File metadata

File hashes

Hashes for ouestcharlie_py_toolkit-0.2.1-cp312-cp312-win_amd64.whl
Algorithm Hash digest
SHA256 b510de8b5a308e6614d039b70228101b54aafab8f96f2710070810564ae61d3e
MD5 c8c26f829ea60c13840dce8a405811e5
BLAKE2b-256 5b26d428ffc23f829dc505796be2d7e2e91891b23e22b10fef7a49e2f5cc3df5

See more details on using hashes here.

Provenance

The following attestation bundles were made for ouestcharlie_py_toolkit-0.2.1-cp312-cp312-win_amd64.whl:

Publisher: publish.yml on ouestcharlie/ouestcharlie-py-toolkit

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

File details

Details for the file ouestcharlie_py_toolkit-0.2.1-cp312-cp312-manylinux_2_34_x86_64.whl.

File metadata

File hashes

Hashes for ouestcharlie_py_toolkit-0.2.1-cp312-cp312-manylinux_2_34_x86_64.whl
Algorithm Hash digest
SHA256 7f422aefb3d5210c2fdbae993231c0bbed6564eda362fb3947c2035e0b41f83f
MD5 937ae69f4ddd5ace6bb7232fff8a5bb4
BLAKE2b-256 e5e40b6319e08a82310a1539058d8b048130c8c44e63478ae39b04559d0186ce

See more details on using hashes here.

Provenance

The following attestation bundles were made for ouestcharlie_py_toolkit-0.2.1-cp312-cp312-manylinux_2_34_x86_64.whl:

Publisher: publish.yml on ouestcharlie/ouestcharlie-py-toolkit

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

File details

Details for the file ouestcharlie_py_toolkit-0.2.1-cp312-cp312-macosx_15_0_x86_64.whl.

File metadata

File hashes

Hashes for ouestcharlie_py_toolkit-0.2.1-cp312-cp312-macosx_15_0_x86_64.whl
Algorithm Hash digest
SHA256 d8ea7db7254a01cded37bf9db56d4beaf53888d794cbe7d99c94ea9afbf6ea07
MD5 25710e761bf3555787ddd2fe51f77630
BLAKE2b-256 3414ec3b656ee2aa985f915a27dbc655e5ae09e1f24244870c2d05591d1a3940

See more details on using hashes here.

Provenance

The following attestation bundles were made for ouestcharlie_py_toolkit-0.2.1-cp312-cp312-macosx_15_0_x86_64.whl:

Publisher: publish.yml on ouestcharlie/ouestcharlie-py-toolkit

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

File details

Details for the file ouestcharlie_py_toolkit-0.2.1-cp312-cp312-macosx_15_0_arm64.whl.

File metadata

File hashes

Hashes for ouestcharlie_py_toolkit-0.2.1-cp312-cp312-macosx_15_0_arm64.whl
Algorithm Hash digest
SHA256 a560c83cf988da6b0ad80d37849aa087cbb5dc0e9afe3e0e0658f3ef78d13cc4
MD5 33ba17022a313eb25fd9df97328bf3df
BLAKE2b-256 63277a2beb61c586ee20a5f0cab7e905ae9d2e36dfc629bcb9d80404ca420296

See more details on using hashes here.

Provenance

The following attestation bundles were made for ouestcharlie_py_toolkit-0.2.1-cp312-cp312-macosx_15_0_arm64.whl:

Publisher: publish.yml on ouestcharlie/ouestcharlie-py-toolkit

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