Dependency-light Python foundation (logging, exceptions, config, utils) with LLM clients, anti-slop guards, cost tracking, and an opinionated FastAPI+SQLAlchemy framework available as opt-in extras
Project description
pf-core
A dependency-light Python foundation for building LLM applications — and one built to be worked on by AI coding agents as much as by people. The base install provides structured logging, an exception hierarchy, config-from-env, and a service/repo architecture; opt-in extras add LLM clients, output validation, cost tracking and budgets, an eval harness, and a FastAPI + SQLAlchemy app framework. Capabilities compose orthogonally — the foundation alone, the LLM layer without a database, or the web layer without LLMs.
Built for AI-assisted development
The conventions that keep a codebase legible to an AI agent are enforced, not suggested. A build gate fails CI when a file grows past its line budget — small files stay within a model's working context and edit cleanly — and a companion checker flags imports that cross the layered architecture the wrong way. Logging, errors, config, and data access each have one obvious way to do them, documented one-module-per-file for retrieval, so generated code lands in the same shapes as hand-written code instead of drifting. The result is a substrate where an agent can do real work and the guardrails hold.
One interface over every LLM backend — including Claude Code
OpenRouter (paid API), the Anthropic SDK, and Claude Code (a local Claude Max session, $0 per call) sit behind the same chat(messages, model) -> (content, usage) interface. A YAML model router assigns a backend per agent and falls back to the next available one; a registry accepts custom backends (Ollama, direct OpenAI, …). Because the clients are interchangeable and pf_core.parallel fans work across a thread pool, a batch of LLM calls can run concurrently and route anywhere — a large batch pushed onto a Claude Max subscription instead of spending API credits, or spread across providers — while every call is still tracked and budget-checked the same way.
Output guards and observability
LLMs return fenced, truncated, or not-quite-JSON output; pf-core recovers it (pf_core.llm.parse) and validates the result against a schema with optional semantic and cross-field checks (pf_core.llm.validate) — available without the client stack, so output from any transport can be guarded. Every call can record one database row (prompt, tokens, cost, validations, and the job it belongs to), making spend and quality queryable and runs replayable. Pre-call budget checks enforce daily/monthly caps with a kill-switch, a cache skips paying for identical calls, prompts are versioned and linked to the runs they produced, and an eval harness replays golden sets against a new model or prompt to show whether a change is an improvement before it ships.
The rest of the framework
A multi-dialect database layer (SQLite / MySQL / PostgreSQL, identical API) with a shared Alembic runner; a FastAPI app factory with self-contained error pages and content negotiation; a job tracker with a state machine, idempotent step history, and worker leases so multi-step work survives restarts; a mountable admin dashboard for runs, costs, and budgets; and pipeline helpers for run-records, baselines, and stage-cascade cache invalidation. See docs/modules.md for the full index.
Install
pip install pf-core # foundation only — no LLM, no DB, no web
pip install pf-core[validate] # + output guards (no clients/HTTP)
pip install pf-core[llm] # + LLM clients (includes [validate])
pip install pf-core[full,postgres] # the whole app framework
Until the first PyPI release, install from a tagged commit:
pip install "pf-core[llm] @ git+https://github.com/phierceweb/pf-core.git@v0.1.0"
Pinning to a tagged release is recommended for stability — main is the development line and may contain unreleased work between tags. Extras compose orthogonally ([db] without LLM, [web] without [db], [llm] standalone); importing a gated module without its extra raises an ImportError naming the extra and the pip command. Full matrix and release/update flow: docs/INSTALLATION.md.
Documentation
- docs/INSTALLATION.md — extras matrix, install/release/update flows, verification
- docs/modules.md — one-line-per-module index, grouped by concern
- docs/ — per-module reference with usage and parameter detail
- CHANGELOG.md — release history
Development
python -m venv .venv && source .venv/bin/activate
pip install -e ".[full,articles,anthropic,image-phash,dev]" # everything, so the full suite runs
pre-commit install
pytest
python bin/verify-bare-install # confirm the base install stays dependency-light
Pytest fixtures auto-register as a plugin via the pf_core entry point — no conftest.py import needed in consumers. Contribution guidelines: CONTRIBUTING.md.
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 pf_core-0.1.0.tar.gz.
File metadata
- Download URL: pf_core-0.1.0.tar.gz
- Upload date:
- Size: 525.5 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.12.13
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
afd968b5a88bab9b333d5ec93f35b18d921a1171ee4f879af8418031a737146b
|
|
| MD5 |
a4f8307c567986dde36869df6ae06179
|
|
| BLAKE2b-256 |
219d7811c5b9941d03c1b0595f321ef8b9c6baa9c83a552b7760258abf94f117
|
File details
Details for the file pf_core-0.1.0-py3-none-any.whl.
File metadata
- Download URL: pf_core-0.1.0-py3-none-any.whl
- Upload date:
- Size: 476.1 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.12.13
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
4e5877ba161256c75621bb07b9cbd77d02b234cc1d7f985e99e040f93323dfcd
|
|
| MD5 |
7dd373304a20c73f2327f4a853c43673
|
|
| BLAKE2b-256 |
1dabb67dbb8be5f6832f3f36f9a03845ca7cb04c24ef98a930d9aeae2ff528e4
|