Seller-facing tools for UnitySVC: HTTP SDK + usvc_seller CLI
Project description
unitysvc-sellers
Python SDK and CLI for the UnitySVC seller
API (https://seller.unitysvc.com/v1). This package provides:
unitysvc_sellers— a typed Python package (syncClient+ asyncAsyncClient) that wraps the upstream REST API into importable, type-checked method calls.usvc_seller— a CLI built on top of the SDK for day-to-day seller operations (catalog management, secret rotation, service lifecycle) without writing code.
| Guide | Reference | |
|---|---|---|
| Upstream API | Swagger UI · ReDoc | |
| Python SDK | SDK Guide | SDK Reference (auto-generated from docstrings) |
| CLI | CLI Guide | CLI Reference (auto-generated from typer) |
Install
pip install unitysvc-sellers
This pulls in unitysvc-core
for the shared data models, JSON schemas, and generic file validator;
plus httpx, attrs, typer, and rich.
Programmatic usage
from unitysvc_sellers import Client
client = Client(api_key="svcpass_...") # or Client.from_env()
# List services — iterate the page directly (the result is a ServiceList).
for svc in client.services.list(limit=50, status="active"):
print(svc.id, svc.name, svc.status)
# Fetch one — `svc` is a Service active-record bound to its id.
svc = client.services.get(service_id)
print(svc.name, svc.status)
# Mutate via the bound handle (no need to re-pass the id).
svc.update({"status": "pending"})
svc.submit() # shortcut for {"status": "pending"}
# Promotions — upsert returns a Promotion bound to its id.
promo = client.promotions.upsert({
"name": "summer-2026",
"scope": {"customers": "*"},
"pricing": {"type": "multiply", "factor": "0.80"},
"status": "active",
})
promo.update({"status": "paused"})
# Service groups — same pattern.
grp = client.groups.upsert({"name": "premium", "service_ids": [...]})
grp.update({"display_name": "Premium tier"})
# Push an entire catalog directory.
result = client.upload("./my-catalog", dryrun=False)
print(f"services: {result.services.success}/{result.services.total}")
Configuration
| Source | Default | Override |
|---|---|---|
| API key | (required) | Client(api_key=...) / UNITYSVC_SELLER_API_KEY |
| Base URL | https://seller.unitysvc.com/v1 |
Client(base_url=...) / UNITYSVC_SELLER_API_URL |
| Timeout | 30 s | Client(timeout=...) |
The seller context is encoded entirely in the API key (svcpass_...),
so no separate seller_id argument is required.
Env var naming: env vars are namespaced to the seller role
(UNITYSVC_SELLER_API_KEY, UNITYSVC_SELLER_API_URL) so a single
host can run both the seller SDK and the future customer SDK side by
side without collision — each picks up its own credentials.
URL layout: the SDK's generated paths are semantic resource paths
(/services/{id}, /documents/{id}, /promotions, /service-groups)
with no /seller wrapper. The seller scope is carried by the
subdomain + API key. The same generated SDK works against any
deployment layout without regeneration:
# Default: production seller subdomain
Client() # reads UNITYSVC_SELLER_API_URL or defaults to
# https://seller.unitysvc.com/v1
# Staging
Client(base_url="https://seller.staging.unitysvc.com/v1")
# Local development against a running backend
Client(base_url="http://localhost:8000/v1/seller")
Secrets
Manage encrypted seller secrets (API keys, tokens, credentials).
Values are write-only — only metadata is ever returned. The API
mirrors GitHub's secrets API: set(name, value) is idempotent and
covers both create and rotate.
# List all secrets (metadata only)
secrets = client.secrets.list()
for s in secrets.data:
print(s.name, s.created_at)
# Get one secret's metadata by name
meta = client.secrets.get("OPENAI_API_KEY")
# Create or rotate (idempotent)
client.secrets.set("OPENAI_API_KEY", "sk-...")
# Delete a secret (immediate effect on running services)
client.secrets.delete("OPENAI_API_KEY")
Methods:
| Method | Parameters | Returns | Description |
|---|---|---|---|
secrets.list(skip=0, limit=100) |
skip, limit |
SecretsPublic |
List secrets (metadata only) |
secrets.get(name) |
name: str |
SecretPublic |
Get one secret's metadata |
secrets.set(name, value) |
name: str, value: str |
SecretPublic |
Idempotent create-or-replace |
secrets.delete(name) |
name: str |
None |
Permanently delete a secret |
Secret names must be uppercase with underscores (e.g. OPENAI_API_KEY,
STRIPE_SECRET). Names starting with __ are reserved for platform use.
Pagination
services.list, promotions.list, and groups.list use
cursor-based pagination. Each call returns an iterable list
wrapper (ServiceList, PromotionList, GroupList) that exposes
data, next_cursor, has_more, and next_page():
# Single page — iterate the wrapper directly
page = client.services.list(limit=50)
for svc in page:
print(svc.name)
# Manual pagination via next_page()
while page.has_more:
page = page.next_page()
for svc in page:
...
# Or let the SDK walk every page for you
for svc in client.services.iter_all(status="active"):
print(svc.name)
The CLI list commands accept --cursor and --all (to auto-follow
cursors and render the combined result).
Async client
The same API surface is exposed as AsyncClient for use in asyncio
contexts (FastAPI, Starlette, Trio-via-anyio, scripts using
asyncio.run). Sync iteration walks the current page; for full
iteration use iter_all():
import asyncio
from unitysvc_sellers import AsyncClient
async def main():
async with AsyncClient(api_key="svcpass_...") as client:
# One page at a time.
services = await client.services.list(limit=50)
for svc in services:
print(svc.id, svc.name)
# Or every page, automatically:
async for svc in client.services.iter_all(status="active"):
print(svc.id, svc.name)
# Active-record mutations work the same way.
promo = await client.promotions.get(promo_id)
await promo.update({"status": "active"})
asyncio.run(main())
Each async method has the exact same signature as its sync counterpart
on Client. The remote usvc_seller services|promotions|groups
commands all use AsyncClient under the hood.
Errors
All errors are subclasses of unitysvc_sellers.SellerSDKError:
from unitysvc_sellers import (
SellerSDKError,
AuthenticationError, # 401
PermissionError, # 403
NotFoundError, # 404
ValidationError, # 400, 422
ConflictError, # 409
RateLimitError, # 429
ServerError, # 5xx
APIError, # base for everything above
)
Each carries status_code, detail (parsed body if JSON), and
response_body for debugging.
CLI: usvc_seller
The CLI has two sets of commands:
usvc_seller data ...— local seller catalog operations (no network)usvc_seller services|promotions|groups ...— remote operations against the seller backend, all using the SDK'sAsyncClientunder the hood
Local commands
usvc_seller data validate [DATA_DIR] # schema + catalog-layout validation
usvc_seller data format [DATA_DIR] # normalize JSON/TOML/MD files
usvc_seller data populate [DATA_DIR] # run provider populate scripts
usvc_seller data show provider|offering|listing|service NAME
usvc_seller data list providers|sellers|offerings|listings|services [DATA_DIR]
usvc_seller data list-tests # list local code-example / connectivity tests
usvc_seller data run-tests # run them locally
usvc_seller data show-test SERVICE # show last local test result
usvc_seller data upload [DATA_DIR] # upload services + promotions + groups
[--api-key svcpass_...] # defaults to $UNITYSVC_SELLER_API_KEY
[--base-url https://...] # defaults to $UNITYSVC_SELLER_API_URL or staging
[--type services|promotions|groups] # restrict to one resource kind
[--dryrun] # validate against backend without persisting
Remote commands (require $UNITYSVC_SELLER_API_KEY or --api-key)
# Services
usvc_seller services list [--status STATUS] [--name NAME] [--provider NAME]
[--fields id,name,...] [--format table|json]
usvc_seller services show SERVICE_ID [--format table|json]
usvc_seller services submit SERVICE_IDS... | --all [--provider NAME] [--yes]
usvc_seller services withdraw SERVICE_IDS... | --all [--provider NAME] [--yes]
usvc_seller services deprecate SERVICE_IDS... | --all [--provider NAME] [--yes]
usvc_seller services delete SERVICE_IDS... | --all [--status STATUS]
[--provider NAME] [--dryrun] [--yes]
usvc_seller services update SERVICE_ID
[--set-routing-var key=value | '{json}'] (repeatable)
[--remove-routing-var key] (repeatable)
[--load-routing-vars path/to.json]
[--set-price key=value | '{json}' | NUMBER] (repeatable)
[--remove-price-field key] (repeatable)
# Document tests (registered under services for parity with the legacy CLI)
usvc_seller services list-tests [SERVICE_ID] [--all] [--status STATUS]
[--format table|json]
usvc_seller services show-test DOCUMENT_ID [--format table|json]
usvc_seller services run-tests SERVICE_ID [--document-id DOC_ID] [--force]
usvc_seller services skip-test DOCUMENT_ID
usvc_seller services unskip-test DOCUMENT_ID
# Promotions
usvc_seller promotions list [--format table|json]
usvc_seller promotions show NAME_OR_ID [--format table|json]
usvc_seller promotions activate NAME_OR_ID
usvc_seller promotions pause NAME_OR_ID
usvc_seller promotions delete NAME_OR_ID [--force]
# Service groups
usvc_seller groups list [--status STATUS] [--format table|json]
usvc_seller groups show NAME_OR_ID [--format table|json]
usvc_seller groups delete NAME_OR_ID [--force]
# Secrets
usvc_seller secrets list [--format table|json]
usvc_seller secrets show NAME [--format table|json]
usvc_seller secrets create NAME [--value VALUE | --value-file PATH | --value-stdin]
usvc_seller secrets rotate NAME [--value VALUE | --value-file PATH | --value-stdin]
usvc_seller secrets delete NAME [--force]
promotions activate / pause are sugar over
PATCH /promotions/{id} with a status field — the backend
consolidated the legacy /activate and /pause routes.
The legacy usvc services dedup command is not ported because the
backing endpoint was removed; use
services delete --all --status draft instead.
The legacy usvc_seller groups refresh command is not ported
either. Dynamic group membership is now refreshed automatically by a
background worker whenever a group is mutated, so there's no manual
refresh step for sellers to invoke.
Layout
src/unitysvc_sellers/
├── client.py # Client (sync) facade
├── aclient.py # AsyncClient (async) facade
├── exceptions.py # SellerSDKError + status-code subclasses
├── _http.py # internal: unwrap generated Response → typed model or APIError
├── resources/
│ ├── services.py # client.services.{list,get,upload,set_status,...}
│ ├── promotions.py # client.promotions.{list,get,upsert,update,delete}
│ ├── groups.py # client.groups.{list,get,upsert,update,delete}
│ ├── documents.py # client.documents.{get,execute,update_test}
│ ├── aservices.py # async mirror of services.py
│ ├── apromotions.py # async mirror of promotions.py
│ ├── agroups.py # async mirror of groups.py
│ ├── adocuments.py # async mirror of documents.py
│ └── upload.py # high-level upload_directory(client, path)
├── _generated/ # openapi-python-client output (do not edit by hand)
│ ├── client.py # AuthenticatedClient (httpx + attrs, sync + async)
│ ├── api/seller_services/ # services_list, services_get, ...
│ ├── api/seller_promotions/ # promotions_list, promotions_upsert, ...
│ ├── api/seller_service_groups/ # groups_list, groups_upsert, ...
│ ├── api/seller_documents/ # documents_get, documents_execute, ...
│ ├── models/ # one model per schema component
│ └── ...
├── commands/ # Typer command groups for the remote CLI
│ ├── _helpers.py # run_async, async_client, model_list,
│ │ # resolve_promotion, resolve_service_id, ...
│ ├── services.py # `usvc_seller services {list,show,submit,...}`
│ ├── tests.py # `usvc_seller services {list,show,run,skip,unskip}-test`
│ ├── promotions.py # `usvc_seller promotions {list,show,activate,pause,delete}`
│ └── groups.py # `usvc_seller groups {list,show,delete}`
├── cli.py # `usvc_seller` Typer entry point
├── data.py # `usvc_seller data` command group (local)
├── _cli_upload.py # `usvc_seller data upload` Typer wrapper
├── validator.py # seller DataValidator (extends unitysvc_core.validator)
├── format_data.py # `usvc_seller data format`
├── populate.py # `usvc_seller data populate`
├── example.py # `usvc_seller data {list,run,show}-test` (local)
├── list.py # `usvc_seller data list *`
├── output.py # shared Rich output helpers
└── utils.py # seller-only helpers + re-exports from unitysvc_core.utils
Regenerating the API client
The low-level client under src/unitysvc_sellers/_generated/ is
produced by openapi-python-client from a filtered copy of the backend
OpenAPI spec at openapi.json. To regenerate after a backend change:
# Requires a sibling checkout of unitysvc/unitysvc with backend/.venv set up
./scripts/generate_client.sh ../unitysvc
This script:
- Dumps
/v1/openapi.jsonfrom the backend's running app viascripts/dump_spec.py. - Filters to seller-tagged operations and sanitizes schema names that
contain characters openapi-python-client cannot parse (e.g. strips
the auto-generated pydantic
titlefrom anonymous inline object schemas to avoidPricing/Termscollisions). - Runs
openapi-python-client generatewith the config inscripts/openapi-python-client.yml.
The hand-written facades in unitysvc_sellers/{client,resources}.py
should rarely change when the spec is regenerated; only the operation
modules under _generated/api/seller/ and the models under
_generated/models/ get refreshed.
History
This package was split out of
unitysvc-services
(see issue #99).
Shared types + schemas live in
unitysvc-core; seller CLI,
the catalog HTTP SDK, and seller-specific catalog utilities live here.
Roadmap
unitysvc_sellers.builders— catalog-builder helpers (populate_from_iterator,render_template_file, etc.) forunitysvc-services-*data repositories.- Attachment-bytes upload — once the backend defines its replacement
for the old
/seller/documents/upload-attachmentendpoint, theclient.upload(...)orchestrator will inline binary file content again instead of requiringexternal_urlreferences.
License
MIT
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file unitysvc_sellers-0.1.6.tar.gz.
File metadata
- Download URL: unitysvc_sellers-0.1.6.tar.gz
- Upload date:
- Size: 181.1 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
d798b507448bcdc4b3a2879b623f3c3338ece576f342f7aa8dae2b0f74059c24
|
|
| MD5 |
c0b1e208fdcb8f1d9061ae189c4b453a
|
|
| BLAKE2b-256 |
9be0cd282966b6f77949872a4efe5c0e239cfd64a22cada6440cfa8ba29d0b6b
|
Provenance
The following attestation bundles were made for unitysvc_sellers-0.1.6.tar.gz:
Publisher:
publish.yml on unitysvc/unitysvc-sellers
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
unitysvc_sellers-0.1.6.tar.gz -
Subject digest:
d798b507448bcdc4b3a2879b623f3c3338ece576f342f7aa8dae2b0f74059c24 - Sigstore transparency entry: 1390638027
- Sigstore integration time:
-
Permalink:
unitysvc/unitysvc-sellers@010e99055b0c138311f856483437ed26a5981114 -
Branch / Tag:
refs/tags/v0.1.6 - Owner: https://github.com/unitysvc
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@010e99055b0c138311f856483437ed26a5981114 -
Trigger Event:
release
-
Statement type:
File details
Details for the file unitysvc_sellers-0.1.6-py3-none-any.whl.
File metadata
- Download URL: unitysvc_sellers-0.1.6-py3-none-any.whl
- Upload date:
- Size: 278.4 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
2d6bc964384927d78ca6fb48eefa4484c94067d4f2baf79be5d25a7bdd0df562
|
|
| MD5 |
9189bbb28e37bc0354ae0228a2d756b1
|
|
| BLAKE2b-256 |
7372ed847f95aa0ae6a23b585709e72e16aac7c6270d911710320e9944478432
|
Provenance
The following attestation bundles were made for unitysvc_sellers-0.1.6-py3-none-any.whl:
Publisher:
publish.yml on unitysvc/unitysvc-sellers
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
unitysvc_sellers-0.1.6-py3-none-any.whl -
Subject digest:
2d6bc964384927d78ca6fb48eefa4484c94067d4f2baf79be5d25a7bdd0df562 - Sigstore transparency entry: 1390638033
- Sigstore integration time:
-
Permalink:
unitysvc/unitysvc-sellers@010e99055b0c138311f856483437ed26a5981114 -
Branch / Tag:
refs/tags/v0.1.6 - Owner: https://github.com/unitysvc
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@010e99055b0c138311f856483437ed26a5981114 -
Trigger Event:
release
-
Statement type: