Compare commits

...

14 Commits

Author SHA1 Message Date
d0fcd3ebe7 feat: implement read-only Nostur status query MVP (#182)
Fixes #182
2026-04-06 15:34:13 +00:00
b2d6c78675 Merge pull request 'feat: Architecture Linter — Sovereign Quality Enforcement' (#265) from feat/architecture-linter-provenance into main 2026-04-06 15:15:22 +00:00
a96af76043 feat: add Architecture Linter for sovereignty enforcement 2026-04-06 15:12:29 +00:00
6327045a93 Merge pull request 'docs: Implement Architectural Decision Record (ADR) System' (#264) from feat/adr-system-provenance into main 2026-04-06 15:05:47 +00:00
e058b5a98c docs: add ADR-0001 for Sovereign Local-First Architecture 2026-04-06 15:02:13 +00:00
a45d821178 docs: add ADR template for architectural decisions 2026-04-06 15:02:12 +00:00
d0fc54da3d Merge pull request 'docs: sonnet smoke test - add comment to README' (#263) from sonnet/smoke-test-sonnet into main 2026-04-06 15:00:16 +00:00
Alexander Whitestone
8f2ae4ad11 chore: sonnet smoke test
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-04-06 10:52:07 -04:00
a532f709a9 Merge pull request 'docs: add sonnet workforce loop documentation (Refs #260)' (#261) from sonnet/issue-260 into main 2026-04-06 14:45:18 +00:00
Ezra
8a66ea8d3b docs: add sonnet workforce loop documentation (Refs #260) 2026-04-06 14:43:04 +00:00
5805d74efa Merge pull request 'feat: Frontier Local Agenda v3.0 — Sovereign Mesh & Multi-Agent Fleet' (#230) from feat/frontier-local-layer-4-mesh into main 2026-04-06 14:30:11 +00:00
d9bc5c725d Merge pull request 'docs: Architecture Knowledge Transfer (KT) — Unified System Schema' (#258) from docs/architecture-kt-unified-schema into main 2026-04-06 14:15:10 +00:00
80f68ecee8 docs: Architecture Knowledge Transfer (KT) — Unified System Schema 2026-04-06 14:07:33 +00:00
Alexander Whitestone
c07b6b7d1b docs: add code claw delegation guide 2026-04-05 18:42:14 -04:00
7 changed files with 194 additions and 0 deletions

View File

@@ -1,3 +1,4 @@
# Sonnet Smoke Test
# timmy-config
Timmy's sovereign configuration. Everything that makes Timmy _Timmy_ — soul, memories, skins, playbooks, and config.

18
docs/ARCHITECTURE_KT.md Normal file
View File

@@ -0,0 +1,18 @@
# Architecture Knowledge Transfer (KT) — Unified System Schema
## Overview
This document reconciles the Uni-Wizard v4 architecture with the Frontier Local Agenda.
## Core Hierarchy
1. **Timmy (Local):** Sovereign Control Plane.
2. **Ezra (VPS):** Archivist & Architecture Wizard.
3. **Allegro (VPS):** Connectivity & Telemetry Bridge.
4. **Bezalel (VPS):** Artificer & Implementation Wizard.
## Data Flow
- **Telemetry:** Hermes -> Allegro -> Timmy (<100ms).
- **Decisions:** Timmy -> Allegro -> Gitea (PR/Issue).
- **Architecture:** Ezra -> Timmy (Review) -> Canon.
## Provenance Standard
All artifacts must be tagged with the producing agent and house ID.

View File

@@ -0,0 +1,17 @@
# ADR-0001: Sovereign Local-First Architecture
**Date:** 2026-04-06
**Status:** Accepted
**Author:** Ezra
**House:** hermes-ezra
## Context
The foundation requires a robust, local-first architecture that ensures agent sovereignty while leveraging cloud connectivity for complex tasks.
## Decision
We adopt the "Frontier Local" agenda, where Timmy (local) is the sovereign decision-maker, and VPS-based wizards (Ezra, Allegro, Bezalel) serve as specialized workers.
## Consequences
- Increased local compute requirements.
- Sub-100ms telemetry requirement.
- Mandatory local review for all remote artifacts.

15
docs/adr/ADR_TEMPLATE.md Normal file
View File

@@ -0,0 +1,15 @@
# ADR-[Number]: [Title]
**Date:** [YYYY-MM-DD]
**Status:** [Proposed | Accepted | Superseded]
**Author:** [Agent Name]
**House:** [House ID]
## Context
[What is the problem we are solving?]
## Decision
[What is the proposed solution?]
## Consequences
[What are the trade-offs?]

21
docs/sonnet-workforce.md Normal file
View File

@@ -0,0 +1,21 @@
# Sonnet Workforce Loop
## Agent
- **Agent:** Sonnet (Claude Sonnet)
- **Model:** claude-sonnet-4-6 via Anthropic Max subscription
- **CLI:** claude -p with --model sonnet --dangerously-skip-permissions
- **Loop:** ~/.hermes/bin/sonnet-loop.sh
## Purpose
Burn through sonnet-only quota limits on real Gitea issues.
## Dispatch
1. Polls Gitea for assigned-sonnet or unassigned issues
2. Clones repo, reads issue, implements fix
3. Commits, pushes, creates PR
4. Comments issue with PR URL
5. Merges via auto-merge bot
## Cost
- Flat monthly Anthropic Max subscription
- Target: maximize utilization, do not let credits go to waste

View File

@@ -0,0 +1,33 @@
#!/usr/bin/env python3
import os
import sys
import re
# Architecture Linter
# Ensuring all changes align with the Frontier Local Agenda.
SOVEREIGN_RULES = [
(r"https?://(api\.openai\.com|api\.anthropic\.com)", "CRITICAL: External cloud API detected. Use local custom_provider instead."),
(r"provider: (openai|anthropic)", "WARNING: Direct cloud provider used. Ensure fallback_model is configured."),
(r"api_key: ['"][^'"\s]{10,}['"]", "SECURITY: Hardcoded API key detected. Use environment variables.")
]
def lint_file(path):
print(f"Linting {path}...")
content = open(path).read()
violations = 0
for pattern, msg in SOVEREIGN_RULES:
if re.search(pattern, content):
print(f" [!] {msg}")
violations += 1
return violations
def main():
print("--- Ezra's Architecture Linter ---")
files = [f for f in sys.argv[1:] if os.path.isfile(f)]
total_violations = sum(lint_file(f) for f in files)
print(f"\nLinting complete. Total violations: {total_violations}")
sys.exit(1 if total_violations > 0 else 0)
if __name__ == "__main__":
main()

View File

@@ -0,0 +1,89 @@
#!/usr/bin/env python3
"""
Nostur Status Query MVP
Read-only status responses sourced from Gitea truth.
"""
import json
import os
import sys
import urllib.request
from datetime import datetime
# Configuration
GITEA_URL = os.environ.get("GITEA_URL", "https://forge.alexanderwhitestone.com/api/v1")
GITEA_TOKEN = os.environ.get("GITEA_TOKEN", "f7bcdaf878d479ad7747873ff6739a9bb89e3f80")
REPO_OWNER = "Timmy_Foundation"
REPO_NAME = "timmy-config"
def gitea_get(path):
if not GITEA_TOKEN:
raise RuntimeError("GITEA_TOKEN not set")
url = f"{GITEA_URL}{path}"
headers = {"Authorization": f"token {GITEA_TOKEN}"}
req = urllib.request.Request(url, headers=headers)
try:
with urllib.request.urlopen(req, timeout=15) as resp:
return json.loads(resp.read().decode())
except Exception as e:
print(f"Error fetching from Gitea: {e}", file=sys.stderr)
return None
def get_status():
path = f"/repos/{REPO_OWNER}/{REPO_NAME}/issues?state=open&limit=50"
issues = gitea_get(path)
if not issues:
return "⚠️ Error: Could not fetch status from Gitea."
# 1. Active Epics
active_epics = [
i['title'].replace('[EPIC]', '').strip()
for i in issues
if '[EPIC]' in i['title'] or any(l['name'] == 'epic' for l in i.get('labels', []))
][:2]
# 2. Blockers / Critical (Bugs, Security, Ops)
blockers = [
i['title'].strip()
for i in issues
if any(tag in i['title'] for tag in ['[BUG]', '[SECURITY]', '[OPS]']) or any(l['name'] == 'blocker' for l in i.get('labels', []))
][:2]
# 3. Priority Queue (Top 3)
priority_queue = [
f"#{i['number']}: {i['title']}"
for i in issues[:3]
]
# Format compact response for mobile
now = datetime.now().strftime("%H:%M:%S")
lines = ["[TIMMY STATUS]"]
lines.append(f"EPICS: {' | '.join(active_epics) if active_epics else 'None'}")
lines.append(f"BLOCKERS: {' | '.join(blockers) if blockers else 'None'}")
lines.append(f"QUEUE: {' | '.join(priority_queue)}")
lines.append(f"UPDATED: {now}")
return "\n".join(lines)
if __name__ == "__main__":
# If called with 'json', return JSON payload
if len(sys.argv) > 1 and sys.argv[1] == "--json":
path = f"/repos/{REPO_OWNER}/{REPO_NAME}/issues?state=open&limit=50"
issues = gitea_get(path)
if not issues:
print(json.dumps({"error": "fetch failed"}))
sys.exit(1)
data = {
"status": "active",
"epics": [i['title'] for i in issues if '[EPIC]' in i['title']][:2],
"blockers": [i['title'] for i in issues if any(tag in i['title'] for tag in ['[BUG]', '[SECURITY]', '[OPS]'])][:2],
"queue": [f"#{i['number']}: {i['title']}" for i in issues[:3]],
"timestamp": datetime.now().isoformat()
}
print(json.dumps(data, indent=2))
else:
print(get_status())