Thin inference substrate: user-authored profiles, LangChain as a handler, optional provenance.
Project description
chumak
A thin inference substrate for Python projects: user-authored profiles, LangChain as a handler, optional provenance/meta on every response.
Chumaks (Чумаки) were wandering Ukrainian salt-traders who traversed the steppe between distant places. They named the Milky Way after themselves — Чумацький Шлях, the Chumaks' Way — because they navigated by it.
What it is
A small library that abstracts away which LLM you're calling and how.
- Author profiles (TOML files) under the app's XDG config dir.
- Load a profile via
ProfileLoader(with inheritance + env-var overrides) - Call
infer(prompt=..., output_schema=..., profile=...)and get back a validated pydantic payload, normalised citations, and (optionally) a provenanceMetastamp.
Two built-in handlers:
langchain— useslangchain.chat_models.init_chat_model(profile.model)so a single identifier (anthropic:claude-opus-4-7,openai:gpt-5, …) routes to the right provider. Structured output, citations, and token usage all handled.subprocess— shells out to a CLI (claude --print,codex exec, etc.). Useful for prompt iteration via an existing, authorised tool. Schema is injected into the prompt as JSON Schema; stdout is parsed and validated.
Profiles
Profiles are user-authored. chumak ships at most one generic example
(anthropic-claude-opus-4-7 via the LangChain handler). Everything else is yours.
Profiles live in consumer app directory, e.g. ~/.config/<your-app>/chumak/profiles/.
chumak does not impose a config dir; the app passes search_paths to ProfileLoader.
File shape
# ~/.config/galops-vision/chumak/profiles/claude.toml
handler = "langchain"
model = "anthropic:claude-opus-4-7"
temperature = 0.0
[model_kwargs]
max_tokens = 4096
Inheritance
# claude-account-b.toml
extends = "claude"
[model_kwargs]
# api_key sourced from env — see below
Env-var overlay
Every field on a profile is overridable from the environment. chumak does not provision special fields; the convention is uniform:
{APP_PREFIX}_PROFILE_{PROFILE_NAME}_{FIELD_PATH}
with __ as the nested-field delimiter (single _ stays inside field names):
# top-level field
export MYAPP_VISION_PROFILE_CLAUDE_MODEL=anthropic:claude-haiku-4-5
# nested into model_kwargs
export MYAPP_VISION_PROFILE_CLAUDE_ACCOUNT_B_MODEL_KWARGS__API_KEY=sk-ant-...
This means a profile file can be effectively empty on disk (just declaring the
profile's existence and maybe an extends), with all values supplied by the
environment. You decide which fields are sensitive and never touch disk.
Usage
from pathlib import Path
from chumak import ProfileLoader, infer
loader = ProfileLoader(
search_paths=[Path.home() / ".config/my-app/chumak/profiles"],
env_prefix="MYAPP",
)
loader.names() # -> ["claude", "claude-creative", ...]
profile = loader.load("claude")
from pydantic import BaseModel
class AnneSchema(BaseModel):
title: str
value: int
result = infer(
prompt="Extract title and value from this text: ...",
output_schema=AnneSchema,
profile=profile,
)
result.payload # -> MissionTitle(title=..., bounty=...)
result.citations # -> [Citation, ...] (if the model supplied any)
result.meta # -> Meta with cost, generated_at, model identity
With provenance
from chumak import Provenance
result = infer(
prompt="...",
output_schema=AnneSchema,
profile=profile,
provenance=Provenance(
artefact_type="model@v1",
artefact_id="artifact-type:2026-05-20T12:34:56Z",
),
)
result.meta.artefact_type # "mission_title@v1"
result.meta.derived_from # [...]
Design notes
- No domain knowledge: chumak carries no built-in prompts, no role concepts (tactical/narrator etc. — that's an app concern; just name your profile).
- LangChain is a handler, not the spine: subprocess CLIs are first-class.
- Provenance is opt-in: omit
provenance=andmeta.artefact_typeisNone. - The lib never reads env directly for its own settings. The env overlay for profiles is a deliberate, scoped exception, gated on the prefix the consumer passes in.
Tooling
uv, Python 3.12+, ruff, ty, pytest. See CONTRIBUTING.md for dev setup, the integration test, and quality-check commands.
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 chumak-0.1.1.tar.gz.
File metadata
- Download URL: chumak-0.1.1.tar.gz
- Upload date:
- Size: 101.3 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: uv/0.11.19 {"installer":{"name":"uv","version":"0.11.19","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
ca718421a0cb2decfad82eeac9ff1e15c262db9eaca073b684c01ec54a77076d
|
|
| MD5 |
8216addec35a94abd86e0d1d6af2141a
|
|
| BLAKE2b-256 |
58d50eeea0818bae7041a3ec9d16408ee255e963dc046209fbab7014926e8bcb
|
File details
Details for the file chumak-0.1.1-py3-none-any.whl.
File metadata
- Download URL: chumak-0.1.1-py3-none-any.whl
- Upload date:
- Size: 17.0 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: uv/0.11.19 {"installer":{"name":"uv","version":"0.11.19","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
050f70a346ca9d0a1f5f941a6e0b2fac223e1344ddb2db18643d37a09c8b9723
|
|
| MD5 |
e77fee39c7c896f27c4fae7f44185f01
|
|
| BLAKE2b-256 |
e8014a092faffa8f1831e747fc0079cf4c26881e8f718736a426596482e0e7bf
|