# Build (warcraft-core first, then opencode-warcraft plugin)
bun run build
# Run all workspace tests + smoke E2E (~2,500 tests)
bun run test
# Run a single test file
bun test packages/warcraft-core/src/feature/feature-service.test.ts
# Run one package's tests only
bun run test --filter warcraft-core
bun run test --filter opencode-warcraft
# Type-check both packages + Biome lint
bun run lint
# Auto-fix formatting
bun run format
# Check formatting without writing (CI mode)
bun run format:check
# Full pre-release gate (install → build → test → lint → format:check)
bun run release:check
# Regenerate skill registry after adding/editing skills/*.md
bun run generate-skills # run from packages/opencode-warcraft/
# E2E lanes (optional; host lane requires git + br CLI)
bun run test:e2e:host- Monorepo: Bun workspaces with two packages
packages/warcraft-core— domain logic: services, stores, types, state machine, utilitiespackages/opencode-warcraft— OpenCode plugin: agents, tools, hooks, skills, MCP, DI container
- DI: Composition root in
packages/opencode-warcraft/src/container.tswires all services and tools - Storage: Pluggable via
beadsMode— filesystem JSON files (docs/<feature>/) or beads-backed (brCLI +.beads/) - Agents (6): Khadgar (hybrid), Mimiron (planner), Saurfang (orchestrator), Brann (researcher), Mekkatorque (worker), Algalon (reviewer)
- Tools (16): across feature, plan, task, execute, context, status, agents-md, doctor, skill, hashline-edit domains
- Data flow:
feature_create → plan_write → plan_approve (SHA-256) → tasks_sync → warcraft_execute → task() call → worker → task_update → status - hashline-edit: Vendored in
packages/opencode-warcraft/src/tools/hashline-edit/. Uses xxHash32LINE#HASHtags. Path-validated (absolute + project root containment). Available to Mekkatorque workers only.
- 2-space indent, single quotes, semicolons always, trailing commas always, 120-char line width
- Config:
biome.jsonat repo root - Run
bun run formatto auto-fix; CI checks withbun run format:check
- ESM only — local imports MUST use
.jsextension:import { Foo } from './foo.js'; - Type-only imports — use
import typefor types:import type { FeatureJson } from '../types.js'; - Separate value and type imports on different lines when both are needed from the same module
- Node builtins use bare specifiers:
import * as fs from 'fs';
| Kind | Convention | Example |
|---|---|---|
| Types / Classes | PascalCase | FeatureService, TaskStatus, OperationOutcome |
| Functions / Variables | camelCase | createStores, sanitizeName, beadsMode |
| Constants | UPPER_SNAKE_CASE | ALLOWED_TRANSITIONS, DEFAULT_AGENT_MODELS |
| Files | kebab-case | feature-service.ts, bead-gateway.ts |
| Test files | Co-located *.test.ts |
feature-service.test.ts next to feature-service.ts |
- Prefer
interfacefor object shapes,typefor unions/intersections/aliases - Explicit return types on all public/exported functions
- Use discriminated unions for result types (e.g.,
OperationOutcomewithseverity: 'ok' | 'degraded' | 'fatal') - No
any— preferunknownwith type narrowing;anyis linter-allowed but discouraged
- Services return
OperationOutcome<T>(ok/degraded/fatal) withDiagnosticentries — never throw for expected failures - Stores may throw — services catch and wrap into outcomes
- Tools return JSON strings via
toolSuccess(data)ortoolError(message, hints?, options?)— never raw strings - State machine violations throw
InvalidTransitionErrorwithfromandtofields - Beads layer uses
Result<T>discriminated union ({ success: true, value }|{ success: false, error })
- async/await over raw promises; no
.then()chains - Constructor injection for dependencies (services take stores/factories as params)
- Store factory pattern:
createStores(root, beadsMode)returns{ featureStore, taskStore, planStore } spyOnfor mocking in tests — no test framework beyondbun:test- Atomic file writes via
writeJsonLockedSync/patchJsonLockedSyncin filesystem stores
// Standard test structure
import { afterEach, beforeEach, describe, expect, it } from 'bun:test';
describe('FeatureService', () => {
let testRoot: string;
beforeEach(() => {
testRoot = fs.mkdtempSync(path.join(os.tmpdir(), 'warcraft-test-'));
});
afterEach(() => {
fs.rmSync(testRoot, { recursive: true, force: true });
});
it('creates feature at canonical path', () => {
const stores = createStores(testRoot, 'off');
const service = new FeatureService(testRoot, stores.featureStore, 'off');
const outcome = service.create('my-feature');
expect(outcome.severity).toBe('ok');
});
});- Test temp dirs with
fs.mkdtempSync+ cleanup inafterEach - Stub stores with partial overrides:
createFeatureStoreStub(feature, { list: () => [] }) - E2E helpers in
packages/opencode-warcraft/src/e2e/helpers/test-env.ts - Use
isUsable(outcome)guard before accessingoutcome.value
- Shared logic →
warcraft-core: services, stores, types, utilities, state machine - Plugin wiring →
opencode-warcraft: agents, tools, hooks, skills, MCP, prompt assembly - Export new public APIs via
warcraft-core/src/index.ts; consume in plugin viawarcraft-coreworkspace dependency - Keep
warcraft-coreplugin-agnostic — no OpenCode imports
This project is indexed by GitNexus as opencode-warcraft (2503 symbols, 6920 relationships, 201 execution flows). Use the GitNexus MCP tools to understand code, assess impact, and navigate safely.
If any GitNexus tool warns the index is stale, run
npx gitnexus analyzein terminal first.
- MUST run impact analysis before editing any symbol. Before modifying a function, class, or method, run
gitnexus_impact({target: "symbolName", direction: "upstream"})and report the blast radius (direct callers, affected processes, risk level) to the user. - MUST run
gitnexus_detect_changes()before committing to verify your changes only affect expected symbols and execution flows. - MUST warn the user if impact analysis returns HIGH or CRITICAL risk before proceeding with edits.
- When exploring unfamiliar code, use
gitnexus_query({query: "concept"})to find execution flows instead of grepping. It returns process-grouped results ranked by relevance. - When you need full context on a specific symbol — callers, callees, which execution flows it participates in — use
gitnexus_context({name: "symbolName"}).
gitnexus_query({query: "<error or symptom>"})— find execution flows related to the issuegitnexus_context({name: "<suspect function>"})— see all callers, callees, and process participationREAD gitnexus://repo/opencode-warcraft/process/{processName}— trace the full execution flow step by step- For regressions:
gitnexus_detect_changes({scope: "compare", base_ref: "main"})— see what your branch changed
- Renaming: MUST use
gitnexus_rename({symbol_name: "old", new_name: "new", dry_run: true})first. Review the preview — graph edits are safe, text_search edits need manual review. Then run withdry_run: false. - Extracting/Splitting: MUST run
gitnexus_context({name: "target"})to see all incoming/outgoing refs, thengitnexus_impact({target: "target", direction: "upstream"})to find all external callers before moving code. - After any refactor: run
gitnexus_detect_changes({scope: "all"})to verify only expected files changed.
- NEVER edit a function, class, or method without first running
gitnexus_impacton it. - NEVER ignore HIGH or CRITICAL risk warnings from impact analysis.
- NEVER rename symbols with find-and-replace — use
gitnexus_renamewhich understands the call graph. - NEVER commit changes without running
gitnexus_detect_changes()to check affected scope.
| Tool | When to use | Command |
|---|---|---|
query |
Find code by concept | gitnexus_query({query: "auth validation"}) |
context |
360-degree view of one symbol | gitnexus_context({name: "validateUser"}) |
impact |
Blast radius before editing | gitnexus_impact({target: "X", direction: "upstream"}) |
detect_changes |
Pre-commit scope check | gitnexus_detect_changes({scope: "staged"}) |
rename |
Safe multi-file rename | gitnexus_rename({symbol_name: "old", new_name: "new", dry_run: true}) |
cypher |
Custom graph queries | gitnexus_cypher({query: "MATCH ..."}) |
| Depth | Meaning | Action |
|---|---|---|
| d=1 | WILL BREAK — direct callers/importers | MUST update these |
| d=2 | LIKELY AFFECTED — indirect deps | Should test |
| d=3 | MAY NEED TESTING — transitive | Test if critical path |
| Resource | Use for |
|---|---|
gitnexus://repo/opencode-warcraft/context |
Codebase overview, check index freshness |
gitnexus://repo/opencode-warcraft/clusters |
All functional areas |
gitnexus://repo/opencode-warcraft/processes |
All execution flows |
gitnexus://repo/opencode-warcraft/process/{name} |
Step-by-step execution trace |
Before completing any code modification task, verify:
gitnexus_impactwas run for all modified symbols- No HIGH/CRITICAL risk warnings were ignored
gitnexus_detect_changes()confirms changes match expected scope- All d=1 (WILL BREAK) dependents were updated
After committing code changes, the GitNexus index becomes stale. Re-run analyze to update it:
npx gitnexus analyzeIf the index previously included embeddings, preserve them by adding --embeddings:
npx gitnexus analyze --embeddingsTo check whether embeddings exist, inspect .gitnexus/meta.json — the stats.embeddings field shows the count (0 means no embeddings). Running analyze without --embeddings will delete any previously generated embeddings.
Claude Code users: A PostToolUse hook handles this automatically after
git commitandgit merge.
| Task | Read this skill file |
|---|---|
| Understand architecture / "How does X work?" | .claude/skills/gitnexus/gitnexus-exploring/SKILL.md |
| Blast radius / "What breaks if I change X?" | .claude/skills/gitnexus/gitnexus-impact-analysis/SKILL.md |
| Trace bugs / "Why is X failing?" | .claude/skills/gitnexus/gitnexus-debugging/SKILL.md |
| Rename / extract / split / refactor | .claude/skills/gitnexus/gitnexus-refactoring/SKILL.md |
| Tools, resources, schema reference | .claude/skills/gitnexus/gitnexus-guide/SKILL.md |
| Index, status, clean, wiki CLI commands | .claude/skills/gitnexus/gitnexus-cli/SKILL.md |