LLMRing core server: alias sync, receipts, and usage logging (key-scoped, no users)
Project description
LLMRing Server
Self-hostable backend for the LLMRing project. It provides optional persistence and advanced features on top of the lockfile-only workflow.
Key Features
- Usage Tracking: Log LLM usage with costs and statistics
- Receipt System: Cryptographically signed receipts (Ed25519 over RFC 8785 JCS)
- Registry Proxy: Cached access to the public model registry (from GitHub Pages)
- Conversations: Store and retrieve conversation history
- MCP Integration: Persist MCP servers, tools, resources, and prompts
- Templates: Reusable conversation templates
This service is optional. LLMRing works fully in lockfile-only mode; run this server when you need persistence, receipts, usage tracking, or MCP integration.
Quick start
Requirements:
- Python 3.10+
- PostgreSQL (reachable from the server)
Install and run:
# from repo root or this directory
uv run llmring-server --reload
# or
uv run python -m llmring_server.cli --reload
By default the server listens on http://0.0.0.0:8000 and exposes Swagger UI at /docs.
Configuration
Configuration is provided via environment variables (Pydantic Settings). Key variables:
- LLMRING_DATABASE_URL: PostgreSQL connection string (default: postgresql://localhost/llmring)
- LLMRING_DATABASE_SCHEMA: Schema name (default: llmring)
- LLMRING_DATABASE_POOL_SIZE: Connection pool size (default: 20)
- LLMRING_DATABASE_POOL_OVERFLOW: Pool overflow (default: 10)
- LLMRING_REDIS_URL: Redis URL for caching (default: redis://localhost:6379/0)
- LLMRING_CACHE_TTL: Cache TTL seconds (default: 3600)
- LLMRING_CORS_ORIGINS: Comma-separated origins (default: http://localhost:5173,http://localhost:5174,*)
- LLMRING_REGISTRY_BASE_URL: Base URL for the public registry (default: https://llmring.github.io/registry/)
- LLMRING_RECEIPTS_PRIVATE_KEY_B64: Base64url Ed25519 private key (for receipt issuance)
- LLMRING_RECEIPTS_PUBLIC_KEY_B64: Base64url Ed25519 public key (for verification)
- LLMRING_RECEIPTS_KEY_ID: Identifier for current signing key
Minimal required: set LLMRING_DATABASE_URL to a reachable Postgres instance. If you plan to issue receipts, also set the signing key variables.
Authentication model
- Project-scoped via
X-API-Keyheader - No user management in this service
- Aliases are local to each codebase in its lockfile; the server only logs the alias label used
Security notes:
- The
X-API-Keymust be treated as a secret. Do not expose it publicly - The server validates the header is present, non-empty, below 256 chars, and without whitespace
- In production, set narrow
LLMRING_CORS_ORIGINS(avoid*) and deploy behind TLS
Endpoints
Public Endpoints
- GET
/→ service info - GET
/health→ DB health - GET
/registry(and/registry.json) → aggregated provider registry (fetched from GitHub Pages) - GET
/receipts/public-key.pem→ current public key in PEM - GET
/receipts/public-key.jwk→ current public key in JWK format - GET
/receipts/public-keys.json→ list of available public keys
Project-Scoped Endpoints (require header X-API-Key)
Usage Tracking (/api/v1)
- POST
/api/v1/log→ Log LLM usage{ "provider": "openai", "model": "gpt-4", "input_tokens": 100, "output_tokens": 50, "cached_input_tokens": 0, "alias": "summarizer", "profile": "prod", "cost": 0.0025 }
- GET
/api/v1/stats?start_date=&end_date=&group_by=day→ Usage statistics
Receipts (/api/v1/receipts)
- POST
/→ Store a signed receipt - GET
/{receipt_id}→ Fetch stored receipt - POST
/issue→ Issue a signed receipt from unsigned payload
Conversations (/conversations)
- POST
/→ Create new conversation{ "title": "Chat Title", "system_prompt": "You are helpful", "model_alias": "claude-3", "project_id": "uuid" }
- GET
/→ List conversations - GET
/{conversation_id}→ Get conversation with messages - PATCH
/{conversation_id}→ Update conversation metadata - GET
/{conversation_id}/messages→ Get conversation messages - POST
/{conversation_id}/messages/batch→ Add multiple messages - DELETE
/old-messages→ Clean up old messages
Conversation Templates (/api/v1/templates)
- POST
/→ Create template - GET
/→ List all templates - GET
/stats→ Template usage statistics - GET
/{template_id}→ Get specific template - PUT
/{template_id}→ Update template - DELETE
/{template_id}→ Delete template - POST
/{template_id}/use→ Record template usage
MCP Integration (/api/v1/mcp)
MCP Servers
- POST
/servers→ Register MCP server{ "name": "my-server", "url": "http://localhost:8080", "transport_type": "http", "auth_config": {...}, "capabilities": {...}, "project_id": "uuid" }
- GET
/servers→ List MCP servers - GET
/servers/{server_id}→ Get server details - PUT
/servers/{server_id}→ Update server - DELETE
/servers/{server_id}→ Remove server - POST
/servers/{server_id}/refresh→ Refresh server capabilities
MCP Tools
- GET
/tools→ List all tools (with server info) - GET
/tools/{tool_id}→ Get tool details - POST
/tools/{tool_id}/execute→ Execute tool{ "input": {...}, "conversation_id": "uuid" }
- GET
/tools/{tool_id}/history→ Get execution history
MCP Resources
- GET
/resources→ List all resources - GET
/resources/{resource_id}→ Get resource details - GET
/resources/{resource_id}/content→ Get resource content
MCP Prompts
- GET
/prompts→ List all prompts - GET
/prompts/{prompt_id}→ Get prompt details - POST
/prompts/{prompt_id}/render→ Render prompt with arguments
Security notes:
- Stats and logs are key-scoped; ensure you send the right API key to avoid data leakage across projects
- Receipts verification requires
LLMRING_RECEIPTS_PUBLIC_KEY_B64to be configured; otherwise signatures are rejected
Receipts
- Signature: Ed25519 over RFC 8785 JSON Canonicalization Scheme (JCS)
- Signature format:
ed25519:<base64url> - Receipt fields (subset):
id,timestamp,model,alias,profile,lock_digest,key_idtokens: { input, output, cached_input }cost: { amount, calculation }signature
- Public keys are available at
/receipts/public-key.pemand/receipts/public-keys.json.
Registry
The server proxies the public registry hosted at https://llmring.github.io/registry/. Models are returned with provider-prefixed keys (e.g., openai:gpt-4o-mini). Responses are cached in Redis when configured.
Database Schema
The server uses PostgreSQL with two schemas:
llmring schema (core data)
- usage_logs: LLM usage tracking
- receipts: Cryptographically signed receipts
- conversations: Conversation metadata
- messages: Conversation messages
- conversation_templates: Reusable templates
mcp_client schema (MCP data)
- servers: MCP server registrations
- tools: Available tools from MCP servers
- resources: Available resources
- prompts: Available prompts
- tool_executions: Tool execution history
Migrations are managed via pgdbm and applied automatically on startup.
Development
Install dev dependencies and run:
# run tests
uv run pytest -q
# run the server in reload mode
uv run llmring-server --reload
# run migrations manually
uv run llmring-db migrate
The project uses:
- FastAPI for HTTP API
- pgdbm for Postgres migrations and access
- httpx for outbound HTTP
- redis (optional) for caching
- cryptography + pynacl for receipts
- Pydantic for data validation
Security Checklist
- Set
LLMRING_CORS_ORIGINSto explicit origins (not*) in production - Serve behind TLS (reverse proxy like nginx or cloud load balancer)
- Store and rotate
X-API-Keyvalues securely; consider per-env keys - Configure
LLMRING_RECEIPTS_PUBLIC_KEY_B64andLLMRING_RECEIPTS_PRIVATE_KEY_B64for receipts - Restrict egress if running in sensitive environments; registry fetches use outbound HTTP
- Enable Redis with authentication (set
LLMRING_REDIS_URL) if caching is needed
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
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 llmring_server-0.1.1.tar.gz.
File metadata
- Download URL: llmring_server-0.1.1.tar.gz
- Upload date:
- Size: 8.7 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.7.4
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
033b815982a52d71c09aa365e2a69bbd151bc914c9ae12b9dfdb1f7746fb8bd7
|
|
| MD5 |
af1ac1afc9f4e9dfbc27c03eb9c23f62
|
|
| BLAKE2b-256 |
12f8925423aba51cd16060a2ed8de935f783f35bf0561a3f9386917d6b2f9cb5
|
File details
Details for the file llmring_server-0.1.1-py3-none-any.whl.
File metadata
- Download URL: llmring_server-0.1.1-py3-none-any.whl
- Upload date:
- Size: 9.2 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.7.4
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
cc914119e4bc3b2576084f292c07908636b789bd916cd2db8b9e07256d80fc41
|
|
| MD5 |
880595e849018d4c7a609f5113f4c30e
|
|
| BLAKE2b-256 |
4d747c6a0dde986edf0862ea09106d4126fc8721b8e80783ff1727734c7114c3
|