Three new modules land in tests/ alongside the v0.1 pure-module tests: - test_store_protocol.py: InMemoryStore round-trips + Protocol conformance. Covers create / get / update / list_by_status / next_derivation_index / record_tx idempotency + defensive-copy semantics. - test_mint_metadata.py: mint_nft_cert end-to-end against a stubbed ChainContext (no live Ogmios). Exercises the 2-of-2 native-script policy shape, tx body construction, CIP-25 envelope CBOR round-trip, and the oversize-asset-name guard. - test_monitor_with_inmemory_store.py: monitor loop driven against InMemoryStore with Koios + the oracle monkeypatched. Covers every status transition the scheduler cares about (confirm, overpay, underpay, stay-pending, record_tx, expiry skip) and the two reprice code paths (successful reprice bumps expected_lovelace + expires_at; max_repricings flips to EXPIRED). All 42 tests pass on pycardano 0.16 with pytest-asyncio auto mode. |
||
|---|---|---|
| cardano_checkout | ||
| tests | ||
| .gitignore | ||
| LICENSE | ||
| pyproject.toml | ||
| README.md | ||
cardano-checkout
Python SDK for merchant-side Cardano payments + NFT certificate-of-authenticity minting.
Zero-custody by design: the merchant provides a wallet xpub. The SDK derives unique receive addresses per invoice, polls the chain for payment, and optionally mints a CIP-25 NFT cert on confirmation. The platform never holds or moves funds.
Extracted from TradeCraft's
services/cardano_*.py modules (2,400+ lines of production code running on the
Cardano mainnet) and packaged for reuse across the Sulkta Coop product family.
Status
v0.1.0-dev — alpha extraction. Pure modules lifted verbatim from TradeCraft.
DB-coupled modules (monitor, scheduler) ship with a TODO: refactor to Store protocol marker — they work as-is when paired with TradeCraft's SQLAlchemy models
but will be refactored to the generic InvoiceStore Protocol in v0.2.
| Module | Status | Notes |
|---|---|---|
addresses |
✅ stable | CIP-1852 HD derivation; pure pycardano |
oracles |
✅ stable | ADA/USD price via Koios with 5-min cache |
invoice + store |
✅ new | Framework-agnostic invoice + persistence Protocol |
mint |
⏳ stub | CIP-25 v2 metadata builder works; tx submission in v0.2 |
ipfs |
✅ working | kubo HTTP API client w/ optional mirror-pin |
monitor |
🟡 SQLAlchemy-coupled | v0.2 target: refactor around InvoiceStore |
scheduler |
🟡 SQLAlchemy-coupled | v0.2 target: same |
txbuild |
❌ v0.2 | Full PyCardano tx construction via Ogmios |
Design
┌────────────────────────────────────────────────────────┐
│ Merchant App │
│ (TradeCraft / chromaticcraft / your-product) │
└──────────────┬───────────────────────┬─────────────────┘
│ │
uses │ implements │ imports
▼ ▼
┌──────────────┐ ┌────────────────────────┐
│ InvoiceStore │ ◄────── │ cardano_checkout SDK │
│ (your DB) │ │ │
└──────────────┘ │ addresses ← pure │
│ oracles ← pure │
│ invoice ← dataclass │
│ monitor ← polls chain │
│ scheduler ← bg loop │
│ mint ← NFT cert │
│ ipfs ← upload │
│ txbuild ← PyCardano wrappers │
└────────────────────────┘
│
talks to │
▼
┌────────────────────────┐
│ Koios + Ogmios + kubo │
└────────────────────────┘
The merchant app provides:
- A wallet xpub (account-level extended public key).
- An
InvoiceStoreimplementation (SQLAlchemy, Postgres, SQLite, in-memory — whatever).
The SDK provides:
- Address derivation from the xpub.
- Per-invoice payment monitoring against Koios.
- ADA ↔ USD price conversion.
- CIP-25 v2 NFT cert minting (v0.2).
- IPFS upload + pinning for NFT image metadata.
Quick start
import asyncio
from cardano_checkout import addresses, oracles
# Derive a receive address for invoice #42
addr = addresses.derive_address(
xpub_hex="<your wallet xpub>",
index=42,
network="mainnet",
)
# Convert a USD price to lovelace at current market
async def main() -> None:
lovelace = await oracles.convert_usd_to_lovelace(99.00)
ada = lovelace / 1_000_000
print(f"Customer owes {ada:.4f} ADA for $99")
asyncio.run(main())
IPFS: bake-then-mirror pattern
The SDK's IPFSClient expects a local kubo daemon (typically in the same
Docker image as the web app) for upload and primary pin, and takes an
optional list of mirror endpoints to pin add the CID on a second node
for archival redundancy.
Typical chromaticcraft deployment:
from cardano_checkout import ipfs
client = ipfs.IPFSClient(
api_url="http://127.0.0.1:5001", # local kubo in the same container
mirror_api_urls=["http://192.168.254.5:5001"], # Lucy's kubo over the LAN/VPN
)
cid = await client.add(photo_bytes, filename="order-0001.jpg")
# Image now served by Rackham (low latency) AND pinned on Lucy (durability)
NFT cert-of-authenticity design
One minting policy per merchant studio. Policy is a native script (no Plutus required), optionally time-locked to make "no more editions after X" a cryptographically verifiable claim.
CIP-25 v2 metadata. Single NFT per order. Policy skey never leaves the custody host (Lucy in Sulkta's pattern). The SDK builds the metadata envelope + tx; external signer does the signature.
Installation
pip install 'cardano-checkout[sqlalchemy]' # if you're using SQLAlchemy
pip install cardano-checkout # core only
License
Apache-2.0 — matches upstream Cardano tooling.