pi-neuralwatt-provider
pi-neuralwatt-provider
Models + energy tracking via Neuralwatt
Kimi, GLM, Qwen, DeepSeek — with real-time energy/cost per session for pi.

Features
- OpenAI-compatible API - Uses Neuralwatt's
/v1/chat/completionsendpoint - Reasoning models - Support for thinking models with
reasoning_effortparameter - Vision models - Image input support on Kimi K2.5, K2.6, and Devstral
- Tool use - Function calling support
- Streaming - Real-time token streaming
- Fast variants - Optimized "Fast" versions of popular models for quicker responses
- Energy reporting - Displays energy consumption (J/mWh/Wh/kWh) and actual billed cost ($) in a dedicated status widget below the editor, tracked per-session
- Quota display - Shows subscription plan, kWh allocation, and credits remaining from your Neuralwatt account, right-aligned in the status widget
- Configurable display - Energy and quota can each be shown in the below-editor widget, the built-in status bar, or turned off entirely via a config file
Available Models
| Model | Context | Vision | Reasoning | Input $/M | Output $/M |
|---|---|---|---|---|---|
| GLM-5.2 | 1.0M | $1.45 | $4.50 | ||
| GLM-5.2 (fast) | 1.0M | $1.45 | $4.50 | ||
| GLM-5.2 (flex) | 1.0M | $1.45 | $4.50 | ||
| GLM-5.2 (short, fast, flex) | 200K | $1.45 | $4.50 | ||
| GLM-5.2 (short, fast) | 200K | $1.45 | $4.50 | ||
| GLM-5.2 (short, flex) | 200K | $1.45 | $4.50 | ||
| GLM-5.2 (short) | 200K | $1.45 | $4.50 | ||
| Kimi K2.6 | 262K | $0.69 | $3.22 | ||
| Kimi K2.6 (flex) | 262K | $0.69 | $3.22 | ||
| Kimi K2.6 Fast | 262K | $0.69 | $3.22 | ||
| Kimi K2.6 Long (Virtual Context) | 1.0M | $0.69 | $3.22 | ||
| Kimi K2.7 Code | 262K | $0.95 | $4.00 | ||
| Kimi K2.7 Code (flex) | 262K | $0.95 | $4.00 | ||
| Nemotron-3-Nano (MCR) | 66K | $0.00 | $0.00 | ||
| Qwen3.5 397B | 262K | $0.69 | $4.14 | ||
| Qwen3.5 397B Fast | 262K | $0.69 | $4.14 | ||
| Qwen3.6 35B | 131K | $0.29 | $1.15 | ||
| Qwen3.6 35B Fast | 131K | $0.29 | $1.15 | ||
| GLM-5 Long (MCR 1M) | 1.0M | $1.10 | $3.60 | ||
| GLM-5.1 Fast Long (MCR 1M) | 1.0M | $1.10 | $3.60 | ||
| Kimi K2.5 Long (MCR 1M) | 1.0M | $0.52 | $2.59 | ||
| Claude Opus (Cached) | 1.0M | $0.00 | $0.00 |
Authentication
The Neuralwatt API key can be configured in multiple ways (resolved in this order):
auth.json(recommended) — Add to~/.pi/agent/auth.json:The{ "neuralwatt": { "type": "api_key", "key": "your-api-key" } }keyfield supports literal values, env var names, and shell commands (prefix with!). See pi's auth file docs for details.- Runtime override — Use the
--api-keyCLI flag - Environment variable — Set
NEURALWATT_API_KEY
Get your API key from neuralwatt.com.
Installation
Option 1: Using pi install (Recommended)
Install from npm:
pi install npm:pi-neuralwatt-provider
Or install directly from GitHub:
pi install https://github.com/monotykamary/pi-neuralwatt-provider
Option 2: With npm
Install from npm:
npm install npm:pi-neuralwatt-provider
Option 3: Manual Clone
Then authenticate and run pi:
# Recommended: add to auth.json
# See Authentication section below
# Or set as environment variable
export NEURALWATT_API_KEY=your-api-key-here
pi
Clone this repository:
git clone git@github.com:monotykamary/pi-neuralwatt-provider.git cd pi-neuralwatt-providerConfigure your Neuralwatt API key:
# Recommended: add to auth.json # See Authentication section below # Or set as environment variable export NEURALWATT_API_KEY=your-api-key-hereRun pi with the extension:
pi -e /path/to/pi-neuralwatt-provider
Environment Variables
| Variable | Required | Description |
|---|---|---|
NEURALWATT_API_KEY |
No | Your Neuralwatt API key (fallback if not in auth.json) |
Configuration
Compat Settings
Neuralwatt's API provides compatibility and capability metadata (pricing, reasoning, vision, developer_role, reasoning_effort, max_images) directly in the /v1/models response. The update-models.js script reads these and writes them into models.json. Only genuinely incorrect API data needs a manual override in patch.json.
Currently configured compat settings:
supportsDeveloperRole: false— All models. vLLM doesn't support thedeveloperrole; pi sends system prompts assystemmessages instead.supportsReasoningEffort: true— GLM-5.2. Sends thereasoning_effortparameter (maps pi's/reasoninglevels onto GLM-5.2's nativehigh/max/minimalviathinkingLevelMap).requiresReasoningContentOnAssistantMessages: true— Kimi K2.6/K2.7 reasoning variants. Pi-ai replays the model's prior-turnreasoningfield on every assistant message so the model can continue its chain-of-thought across turns. All Neuralwatt reasoning models get this Layer-A replay automatically (the gateway aliasesreasoningreasoning_content); this flag adds an emptyreasoning_contentscaffold for turns with no thinking block.chatTemplateKwargs— Rawchat_template_kwargsmerged into every request via pi-ai'sonPayloadhook, mirroring vLLM's request field of the same name. Used to opt reasoning models into full-history reasoning preservation (vLLM's Jinja templates otherwise trim older assistant reasoning in alternating chat). The flags are template-level and family-specific — NOT a generic boolean:- Kimi K2.6 / K2.7 →
{ "preserve_thinking": true }— keeps the full reasoning history across turns (doc-backed; behavioral E2E: 0/6 → 6/6 recall). - GLM-5.2 family →
{ "clear_thinking": false }— stops the template clearing older reasoning (functional: 1/4 → 4/4 recall, confirmed family-wide). - GLM-5.1 / Qwen3.x / non-reasoning
-fast— no kwarg (their templates expose no flag; they rely on Layer-A replay only).
These are injected alongside
reasoning_effort(NOT viathinkingFormat: "chat-template", which would displace the OpenAIreasoning_effortpath) so thinking-level control and full-history preservation coexist.- Kimi K2.6 / K2.7 →
Custom Stream Handler
This extension registers a custom streamSimple provider (api: "neuralwatt") that wraps pi-ai's built-in streamOpenAICompletions. A temporary globalThis.fetch override tees the HTTP response body so the OpenAI SDK handles all standard chunk parsing (text, thinking, tool calls, usage) while the extension reads the tee for Neuralwatt's SSE comment lines (: energy {...}, : cost {...}) that the SDK discards.
Pi Configuration
Add to your pi configuration for automatic loading:
{
"extensions": [
"/path/to/pi-neuralwatt-provider"
]
}
Usage
Once loaded, select a model with:
/model neuralwatt kimi-k2.5
Or use /models to browse all available Neuralwatt models.
Reasoning Effort
For reasoning models, control thinking depth:
/reasoning high
Values: none, low, medium, high
Full-history reasoning preservation is on by default for Kimi K2.6/K2.7 and the GLM-5.2 family (see Compat Settings). Override it per-model via Model Overrides.
Display Configuration
Energy and quota are independently configurable. Create ~/.pi/agent/extensions/neuralwatt.json:
{
"energy": "widget",
"quota": "widget"
}
The file is auto-populated with defaults on first run.
| Key | Values | Default | Description |
|---|---|---|---|
energy |
"widget", "statusbar", "off" |
"widget" |
Energy/cost display mode |
quota |
"widget", "statusbar", "off" |
"widget" |
Quota display mode |
Display modes:
"widget"— Shown in the dedicated below-editor status line. Energy on the left, quota on the right, padded to terminal width."statusbar"— Shown in the built-in pi status bar. When both are set to"statusbar", they're combined with a|separator:⚡X J $Y | plan ● kWh ∙ $bal."off"— Hidden entirely. For"quota": "off", the/v1/quotaAPI fetch is also skipped (saving a network round-trip). Energy data is still parsed from the SSE stream and persisted to the session even when"off".
Example — custom quota footer: If you use your own unified quota footer extension, disable the built-in quota display to avoid duplication:
{
"energy": "widget",
"quota": "off"
}
Model Overrides
modelOverrides lets you override compat flags and other model properties per model id, on top of patch.json + custom-models.json, without editing the extension. Keyed by model id; compat and thinkingLevelMap are deep-merged (toggle one flag without redeclaring the rest), scalars are replaced. Applied at session start, so edits take effect on the next pi session.
{
"energy": "widget",
"quota": "widget",
"mcr": "widget",
"hideOnOtherProvider": false,
"modelOverrides": {
// Disable full-history reasoning for kimi-k2.6 (e.g. to save tokens):
"kimi-k2.6": { "compat": { "chatTemplateKwargs": { "preserve_thinking": false } } },
// Override a single thinking level without redeclaring the map:
"glm-5.2": { "thinkingLevelMap": { "high": "max" }, "compat": { "chatTemplateKwargs": { "clear_thinking": true } } },
// Force a smaller image cap:
"kimi-k2.7-code": { "vision": { "maxImagesPerRequest": 4 } }
}
}
The full set of overridable fields matches the model schema (compat, thinkingLevelMap, vision, cost, contextWindow, maxTokens, reasoning, input). See Compat Settings for the catalog of compat flags and what chatTemplateKwargs values mean per family.
Energy Reporting
Neuralwatt provides real-time energy consumption data with every API response. This extension captures it and displays a running total in a dedicated status widget between the editor and the pi footer:
| Segment | Meaning |
|---|---|
⚡0.8mWh |
Cumulative session energy consumption (auto-scaled: J → mWh → Wh → kWh) |
$0.003952 |
Cumulative session actual billed cost from Neuralwatt |
pro |
Your Neuralwatt subscription plan |
● |
Subscription status indicator (● = active, ⊘ = past due/paused) |
31.7/33.0 kWh |
kWh remaining / kWh included in your plan |
∙ $64.55 |
Credits remaining on your account |
🔑 .../.../mo |
Key allowance usage (if set on your API key) |
The energy and cost data comes from Neuralwatt's SSE stream comments (: energy and : cost), which the standard OpenAI SDK discards. This extension uses a custom stream handler that parses raw SSE to capture them.
Energy is measured directly from GPU hardware using NVIDIA's NVML. For concurrent requests, Neuralwatt uses token-weighted attribution to fairly calculate your share. See Neuralwatt's energy methodology for details.
Persistence
Energy and cost data is persisted per-request as custom session entries. On session resume or tree navigation, the totals are rebuilt by replaying all events in the current branch. This means:
- Session resume — Energy/cost totals are restored when you continue a session
- Branching — Navigating to a different point in the session tree shows the correct totals for that branch
- Forking — Forked sessions carry their energy history forward
Per-turn energy event
After every Neuralwatt turn (in the turn_end handler, once the SSE tee has drained), the extension emits a neuralwatt:turn-energy event on pi's shared event bus so other extensions can surface the energy-billed cost without re-parsing the session. The payload:
| Field | Type | Description |
|---|---|---|
costUsd |
number |
Actual billed cost for this request (USD) |
energyJoules |
number |
Energy consumed for this request (Joules) |
turnIndex |
number | null |
pi's turn index for correlation. null if the event didn't carry one. |
The event is only emitted for turns with Neuralwatt activity (the pending* state is per-request), so non-Neuralwatt turns never produce a spurious zero-cost signal. Consumers should correlate on turnIndex and treat a missing/null index defensively.
API Documentation
- Neuralwatt API:
https://api.neuralwatt.com/v1 - Models endpoint:
https://api.neuralwatt.com/v1/models - Chat completions:
https://api.neuralwatt.com/v1/chat/completions
License
MIT