12. Paragraph wrapping: remaining double-newline-separated blocks ->
Known gaps:
- Tables: not supported, render as plain text
- Nested lists: single regex pass, multi-level indentation not handled
- Mixed bold+link in same line: may produce garbled output
- Inline HTML: not sanitized (esc() only runs on code content)
### 5.5 Model Chip Label (Fixed in Sprint 1)
B3 was resolved in Sprint 1. Current code uses a MODEL_LABELS dict:
const MODEL_LABELS = {
'openai/gpt-5.4-mini': 'GPT-5.4 Mini', 'openai/gpt-4o': 'GPT-4o',
'openai/o3': 'o3', 'openai/o4-mini': 'o4-mini',
'anthropic/claude-sonnet-4.6': 'Sonnet 4.6', 'anthropic/claude-sonnet-4-5': 'Sonnet 4.5',
'anthropic/claude-haiku-3-5': 'Haiku 3.5', 'google/gemini-2.5-pro': 'Gemini 2.5 Pro',
'deepseek/deepseek-chat-v3-0324': 'DeepSeek V3', 'meta-llama/llama-4-scout': 'Llama 4 Scout',
};
$('modelChip').textContent = MODEL_LABELS[m] || (m.split('/').pop() || 'Unknown');
Fallback: any unlisted model shows its short ID (after the last /) rather than a wrong label.
To add a new model: add an entry to MODEL_LABELS and add an to the .
### 5.6 Session Delete Rules (from skill)
These rules are critical. GPT-5.4-mini has repeatedly re-introduced broken versions.
1. deleteSession() NEVER calls newSession(). Deleting does not create.
2. If deleted session was active AND other sessions exist: load sessions[0] (most recent).
3. If deleted session was active AND no sessions remain: show empty state.
4. If deleted session was not active: just re-render the list.
5. Always show toast("Conversation deleted") after any delete.
### 5.7 Send() Session Guard
Before any async operations in send():
const activeSid = S.session.session_id;
After the agent completes:
if (S.session && S.session.session_id === activeSid) {
// apply result, re-render
setBusy(false);
} else {
// user switched sessions mid-flight
// only refresh sidebar, do NOT call setBusy(false) on the new session
await renderSessionList();
}
This prevents a session switch mid-flight from either clobbering the new session's state
or unlocking the Send button on the wrong session.
---
## 6. Data Flow: Full Chat Round Trip
Step-by-step trace of what happens when you type a message and press Send:
1. User types, presses Enter. send() is called.
2. Guard: return if (!text && !pendingFiles) || S.busy
3. If S.session is null: await newSession(), await renderSessionList()
4. Capture activeSid = S.session.session_id (before any awaits)
5. uploadPendingFiles(): POST each file in S.pendingFiles to /api/upload
- Shows upload progress bar
- Clears S.pendingFiles on completion
- Returns array of uploaded filenames
6. Build msgText from text + file note
7. Build userMsg {role:'user', content: displayText, attachments?: filenames}
8. Push userMsg to S.messages, call renderMessages(), appendThinking()
9. setBusy(true), setStatus('Hermes is thinking...')
10. INFLIGHT[activeSid] = {messages: [...S.messages], uploaded}
11. startApprovalPolling(activeSid)
12. POST /api/chat/start {session_id, message, model, workspace}
Server: saves session, creates queue.Queue, starts daemon thread, returns {stream_id}
13. Browser opens EventSource('/api/chat/stream?stream_id=X')
14. In the SSE loop:
- 'token': assistantText += d.text, ensureAssistantRow(), render markdown
- 'tool': setStatus('tool name...')
- 'approval': showApprovalCard(d)
- 'done': sync S from d.session, renderMessages(), loadDir, renderSessionList,
setBusy(false), delete INFLIGHT[activeSid]
- 'error': show error message, setBusy(false)
- es.onerror: handle network drops (show error, setBusy(false))
15. If approval needed: user clicks a button, respondApproval() fires
POST /api/approval/respond -> server pops _pending, calls approve_*
Agent retries the command (now is_approved() returns True) and continues
---
## 7. Dependency Map
Direct imports in server.py:
run_agent.AIAgent Main agent class. Wraps LLM + tool execution.
tools.approval.* Module-level approval state.
yaml Config loading.
Standard library: json, os, re, sys, threading, time, traceback, uuid,
http.server, pathlib, urllib.parse, email.parser, queue
AIAgent constructor parameters used:
model= OpenRouter model ID string
platform='cli' Sets the platform context for tool selection
quiet_mode=True Suppresses agent's own stdout output
enabled_toolsets= List of toolset names from config.yaml
session_id= Used for tool state keying (memory, todos, etc.)
stream_delta_callback= Called per token delta (or None as sentinel)
tool_progress_callback= Called per tool invocation (name, preview, args)
AIAgent.run_conversation() parameters:
user_message= The human turn text
conversation_history= Prior messages list (OpenAI format)
task_id= Session ID (NOTE: NOT session_id=, it is task_id=)
Return value:
{
'messages': [...], Full conversation including new turns
'final_response': '...', Last assistant text response
'completed': True/False, Whether the conversation completed normally
...other fields
}
---
## 8. Configuration Loading
On startup, server.py reads ~/.hermes/config.yaml:
cfg = yaml.safe_load(CONFIG_PATH.read_text())
CLI_TOOLSETS = cfg.get('platform_toolsets', {}).get('cli', [...default...])
Default toolset list (hardcoded fallback):
browser, clarify, code_execution, cronjob, delegation, file,
image_gen, memory, session_search, skills, terminal, todo, tts, vision, web
The web UI always runs with the full CLI toolset. There is no per-session toolset
restriction from the UI yet (see ROADMAP.md Wave 4 for the plan).
---
## 9. Known Bugs and Technical Debt Summary
| ID | Severity | Description | Status | Fix |
|-----|----------|------------------------------------------------------|------------------|------------------|
| B1 | Critical | Approval wiring untested; pattern_keys not shown | FIXED Sprint 1 | Card shows keys; inject_test endpoint added for verification |
| B2 | High | File input no accept attribute | FIXED Sprint 1 | accept= added with image/*, text/*, pdf, code extensions |
| B3 | High | Model chip label hardcodes sonnet substring check | FIXED Sprint 1 | MODEL_LABELS map; fallback to short model ID |
| B4 | High | Reload mid-stream: stream_id lost, no reconnect | FIXED Sprint 1 | stream/status endpoint; reconnect banner via localStorage |
| B5 | High | INFLIGHT in-memory only, lost on reload | FIXED Sprint 1 | markInflight/clearInflight in localStorage |
| B6 | Medium | New sessions always use DEFAULT_WORKSPACE | FIXED Sprint 3 | newSession() passes S.session.workspace to /api/session/new |
| B7 | Medium | Sidebar title overflow: missing min-width:0 | FIXED Sprint 1 | min-width:0 on .session-item |
| B8 | Medium | renderMd missing tables, nested lists | PARTIAL Sprint 4 | Tables Sprint 2; nested lists improved Sprint 4; full fix still Phase E |
| B9 | Medium | Empty assistant messages can render | FIXED Sprint 1 | loadSession() filters empty-text assistant messages |
| B10 | Low | Thinking dots stay during tool-running | FIXED Sprint 3 | removeThinking() on first tool event; compact 'Running X...' row shown |
| B11 | Low | GET /api/session no-ID silently creates session | FIXED Sprint 1 | Returns 400 with error message |
| B12 | Low | Preview panel display:none to flex layout jump | FIXED Sprint 4 | visibility/opacity transition replaces display:none toggle |
| B13 | Low | No CORS headers | Open | Phase H |
| B14 | Low | No keyboard shortcut for new chat | FIXED Sprint 3 | Cmd/Ctrl+K triggers newSession() from anywhere |
| TD1 | Critical | Env vars are process-global (concurrent request bug) | PARTIAL Sprint 5 | Thread-local _set_thread_env() added. Per-session lock from Sprint 4. Process-level env still written as fallback. Full fix needs terminal tool to read thread-local. |
| TD2 | High | SESSIONS cache: no eviction, locking missing | FIXED Sprint 5 | OrderedDict + LRU cap 100 + move_to_end on access. LOCK from Sprint 1. Complete. |
| TD3 | High | No test coverage | PARTIAL Sprint 1 | 19 HTTP integration tests added; unit tests pending Phase A split |
| TD4 | Medium | All code in one file (HTML/CSS/JS/Python mingled) | FIXED Sprint 5 | JS extracted to static/app.js in Sprint 5 (Sprint 9: app.js deleted, replaced by 6 modules). Phase A complete. |
| TD5 | Medium | No request validation (KeyError -> 500 + traceback) | FIXED Sprint 4 | All endpoints hardened: /api/list, /api/file, /api/crons/* all return clean 400/404 |
| TD6 | Low | all_sessions() full directory scan every call | FIXED Sprint 5 | Session index file (_index.json) built on every save. all_sessions() reads index O(1). Phase C partial. |
| TD7 | Low | No structured logging | FIXED Sprint 1 | log_request() override emits JSON per request |
---
## 10. Architecture Improvement Roadmap
These phases run in parallel with the feature roadmap. Each phase targets software
quality: testability, resilience, maintainability, and modularity.
### Phase A: File Separation (Priority: High, Effort: Medium)
Split server.py into a proper package.
Target structure:
webui-mvp/
server.py Entry point: starts server, imports api/
api/
__init__.py
handlers.py do_GET / do_POST routing and dispatch
session_store.py Session class, get_session, new_session, all_sessions, SESSIONS
streaming.py _run_agent_streaming, STREAMS, STREAMS_LOCK, _sse()
upload.py parse_multipart, handle_upload
files.py safe_resolve, list_dir, read_file_content
approval.py Thin wrapper around tools.approval for the HTTP API
config.py Configuration loading (env vars, config.yaml)
static/
index.html HTML document (served directly from disk)
style.css All CSS
[app.js deleted] Replaced by 6 modules: ui.js, workspace.js, sessions.js,
messages.js, panels.js, boot.js
tests/
test_session_crud.py
test_upload.py
test_streaming.py
test_approval.py
test_files.py
frontend/
test_markdown.html
test_session_state.html
Implementation steps:
1. Extract CSS and HTML to static/style.css and static/index.html. No content changes.
Server serves index.html from disk: handler reads Path('static/index.html').read_text()
2. Extract JS to 6 static modules (complete -- app.js deleted Sprint 9)
Add GET /static/* handler in do_GET.
3. Extract Session class and helpers to api/session_store.py
4. Extract _run_agent_streaming and SSE helpers to api/streaming.py
5. Extract parse_multipart and handle_upload to api/upload.py
6. Extract list_dir and friends to api/files.py
7. Refactor handlers.py to import from the above modules
8. server.py becomes: config setup, start server, import Handler from handlers.py
Benefit: Each file is under ~200 lines. Agents can read and modify individual files
without loading the full 1100-line blob.
### Phase B: Thread-Safe Request Context (Priority: Critical, Effort: Medium)
Replace process-global env vars with thread-local or explicit parameter passing.
Root cause: TERMINAL_CWD, HERMES_EXEC_ASK, HERMES_SESSION_KEY are set via os.environ
in _run_agent_streaming(). Two concurrent sessions clobber each other.
Fix options (in order of preference):
Option 1 (best): Check if AIAgent constructor accepts a context dict. Pass workspace,
exec_ask, and session_key directly. Zero env var usage in server code.
Option 2: Use threading.local():
_ctx = threading.local()
# In _run_agent_streaming:
_ctx.workspace = str(workspace)
_ctx.session_key = session_id
# In tools that read env vars: check _ctx first, fall back to os.environ
Option 3 (interim, safe for single-user): Wrap the env var block in a per-session lock:
SESSION_AGENT_LOCKS = {} # session_id -> Lock
# Only one agent run per session at a time
with SESSION_AGENT_LOCKS.setdefault(session_id, threading.Lock()):
os.environ[...] = ...
result = agent.run_conversation(...)
Phase B also includes: review all other os.environ reads/writes in the codebase for
similar thread-safety issues.
### Phase C: Session Store Improvements (Priority: Medium, Effort: Medium)
Three problems to fix:
1. Unbounded SESSIONS cache:
Replace dict with functools.lru_cache wrapper or a simple OrderedDict with max size.
Evict LRU entries when size exceeds 100.
2. No locking around SESSIONS:
Wrap all SESSIONS dict reads and writes with LOCK (already defined, just unused).
Pattern: with LOCK: s = SESSIONS.get(sid)
3. O(n) directory scan in all_sessions():
Add an index file: SESSION_DIR/index.json
Contents: list of compact() dicts, sorted by updated_at
Maintained on every Session.save() and every delete.
all_sessions() reads index.json (one file read) instead of scanning all JSONs.
get_session() still loads the full {session_id}.json on cache miss.
Index rebuild tool: a function that regenerates index.json from all *.json files.
### Phase D: Input Validation and Error Handling (Priority: Medium, Effort: Low)
1. Add a validate() helper:
def validate(body, *required_fields):
missing = [f for f in required_fields if not body.get(f)]
if missing: raise ValueError(f"Missing required fields: {missing}")
2. Refine the outer try/except in do_GET and do_POST:
except ValueError as e:
return j(self, {'error': str(e)}, status=400)
except KeyError as e:
return j(self, {'error': f'Not found: {e}'}, status=404)
except Exception as e:
log.exception('Unhandled error')
return j(self, {'error': 'Internal server error'}, status=500)
# Never expose tracebacks to the client (security risk even on localhost)
3. Add request duration logging:
Log at INFO level: {method} {path} -> {status} in {duration}ms
### Phase E: Frontend Modularization (Priority: Medium, Effort: High)
After Phase A splits the HTML/JS into files, Phase E improves the JavaScript itself.
1. Switch to ES Modules (type="module"):
app.js deleted Sprint 9 -- replaced by 6 modules:
- state.js: export S, INFLIGHT
- sessions.js: session CRUD functions
- chat.js: send(), SSE handling
- files.js: loadDir(), openFile()
- upload.js: uploadPendingFiles(), addFiles(), renderTray()
- approval.js: approval card and polling
- markdown.js: renderMd()
- ui.js: setStatus, setBusy, showToast, syncTopbar
Each module imports what it needs from state.js and other modules.
2. Replace renderMd with marked.js:
CDN: https://cdn.jsdelivr.net/npm/marked/marked.min.js
No bundler needed, ~50KB, handles tables, nested lists, HTML sanitization.
Usage: marked.parse(raw) -- drop-in replacement.
Add DOMPurify alongside for XSS sanitization of rendered HTML.
3. Add Prism.js for syntax highlighting:
CDN: https://cdn.jsdelivr.net/npm/prismjs
Apply after renderMd: Prism.highlightAllUnder(element)
Supports 200+ languages with auto-detection.
### Phase F: API Design Cleanup (Priority: Low, Effort: Medium)
1. Version prefix: add /api/v1/ to all new endpoints.
Keep /api/* as aliases for backward compatibility.
2. Standard response envelope:
Success: {"ok": true, "data": {...}}
Error: {"ok": false, "error": "message", "code": "ERROR_CODE"}
3. Session list pagination:
GET /api/v1/sessions?limit=30&offset=0
Response: {"ok": true, "data": {"sessions": [...], "total": N, "has_more": false}}
4. Consistent naming: use snake_case for all JSON keys.
### Phase G: Observability (Priority: Low, Effort: Low)
1. Structured JSON logging to /tmp/webui-mvp.log:
{"ts": "...", "method": "POST", "path": "/api/chat/start", "status": 200, "ms": 12}
2. Enhanced /health response:
{"status": "ok", "sessions": 10, "active_streams": 2, "uptime_s": 3600, "version": "0.3"}
3. GET /api/debug/stats (localhost only):
{"sessions_cached": N, "streams_active": M, "memory_mb": X}
### Phase H: Authentication (Priority: Low, Effort: Medium)
Optional password gate for non-SSH-tunnel deployments.
1. HERMES_WEBUI_PASSWORD env var enables auth
2. Login page: minimal dark form, POST /api/auth/login
3. Server sets HttpOnly + SameSite=Strict cookie on successful login
4. All API endpoints check cookie if HERMES_WEBUI_PASSWORD is set
5. Cookie validity: 30 days from last activity
### Phase I: Test Infrastructure (Priority: High, Effort: High)
No tests exist today. This is the highest-risk technical debt.
1. Python unit tests (pytest):
- tests/test_session_crud.py: Session class, get_session, new_session, all_sessions
- tests/test_upload.py: parse_multipart directly with known byte payloads
- tests/test_files.py: safe_resolve, list_dir, read_file_content with tmp dirs
- tests/test_streaming.py: mock AIAgent, verify event sequence
- tests/test_approval.py: approval state machine
2. HTTP integration tests:
- Start a test server on a random port
- Drive it with httpx or requests
- Verify all API endpoints return correct shapes and status codes
3. Frontend tests (no build step):
- tests/frontend/test_markdown.html: known input -> expected HTML output assertions
- Run via: python3 -m http.server and open in browser, or use playwright
4. CI (GitHub Actions):
- .github/workflows/test.yml: on push, run pytest + ruff lint
- Target: zero test failures before merging any feature branch
### Phase J: Performance (Priority: Low, Effort: High)
For scale beyond single-user casual use.
1. Session index (Phase C prerequisite): O(1) session list loads
2. Message pagination: /api/session returns last 50 messages, paginate older ones
3. Frontend virtual scroll: IntersectionObserver for both message list and session list
4. Stream cleanup background thread: evict STREAMS entries older than 5 minutes
5. File tree lazy loading: expand-on-click fetches subdirectory contents
---
## 11. How To Add a New API Endpoint
Follow this exact pattern. Review existing handlers in do_GET/do_POST for reference.
### Backend (server.py -> future: api/handlers.py)
GET endpoint:
# Inside do_GET, before the 404 fallback line:
if parsed.path == '/api/your/endpoint':
qs = parse_qs(parsed.query)
param = qs.get('param', [''])[0]
if not param:
return j(self, {'error': 'param is required'}, status=400)
# do work
return j(self, {'result': value})
POST endpoint (AFTER /api/upload check, body already parsed):
if parsed.path == '/api/your/endpoint':
value = body.get('field', '')
if not value:
return j(self, {'error': 'field is required'}, status=400)
# do work
return j(self, {'ok': True, 'data': result})
Endpoint requiring a valid session:
sid = body.get('session_id', '')
try:
s = get_session(sid)
except KeyError:
return j(self, {'error': 'Session not found'}, status=404)
Endpoint that calls Hermes Python modules:
# Example: calling cron.jobs
import sys
sys.path.insert(0, str(Path(__file__).parent.parent))
from cron.jobs import list_jobs
jobs = list_jobs(include_disabled=True)
return j(self, {'jobs': jobs})
### Frontend (6 static JS modules: ui.js, workspace.js, sessions.js, messages.js, panels.js, boot.js)
Simple GET fetch:
const data = await api('/api/your/endpoint?param=' + encodeURIComponent(value));
// data is parsed JSON response, throws on error
POST:
const data = await api('/api/your/endpoint', {
method: 'POST',
body: JSON.stringify({field: value})
});
The api() helper:
async function api(path, opts={}) {
const r = await fetch(path, {headers:{'Content-Type':'application/json'},...opts});
const d = await r.json();
if (!r.ok) throw new Error(d.error || r.statusText);
return d;
}
---
## 12. Common Debugging Commands
# Server health and session count
curl -s http://127.0.0.1:8787/health | python3 -m json.tool
# Tail the server log live
tail -f /tmp/webui-mvp.log
# List all sessions (metadata only)
curl -s http://127.0.0.1:8787/api/sessions | python3 -m json.tool
# Inspect a full session with messages
SID=your_session_id_here
curl -s "http://127.0.0.1:8787/api/session?session_id=$SID" | python3 -m json.tool
# Kill and restart server cleanly
pkill -f "python.*webui-mvp/server.py"
/webui-mvp/start.sh
# Check if server process is running
ps aux | grep "webui-mvp/server.py"
# Inspect session files on disk
ls -lt ~/.hermes/webui-mvp/sessions/
cat ~/.hermes/webui-mvp/sessions/SESSION_ID.json | python3 -m json.tool
# Count messages in a session
python3 -c "import json; d=json.load(open('sessions/SID.json')); print(len(d['messages']))"
# Check approval module state
cd
venv/bin/python -c "from tools.approval import _pending; print(_pending)"
# Check active SSE streams (requires server access)
curl -s http://127.0.0.1:8787/health # streams not exposed yet, add in Phase G
# Find all sessions with messages (not Untitled empty)
ls ~/.hermes/webui-mvp/sessions/ | xargs -I{} python3 -c "
import json, sys
d = json.load(open('~/.hermes/webui-mvp/sessions/{}'))
if d['messages']: print('{}', d['title'][:50])
" 2>/dev/null
---
## 13. Architecture Decision Records
### ADR-001: Single-File Server
Decision: All code in server.py
Rationale: No build step, easy agent modification, zero deployment complexity.
Trade-off: Maintenance burden grows with file size.
Resolution: Phase A splits the file.
### ADR-002: HTML as Python Raw String
Decision: Frontend embedded in server.py as r"""..."""
Rationale: Simplest way to serve frontend without static file server or build system.
Trade-off: No editor syntax highlighting, complex patching, base64 gymnastics for large edits.
Resolution: Phase A moves to static/index.html served from disk.
### ADR-003: ThreadingHTTPServer
Decision: Python stdlib, synchronous threads, not asyncio.
Rationale: No dependencies, synchronous agent calls fit naturally in threads.
Trade-off: Memory scales linearly with concurrent users. Thread pool is unbounded.
Resolution: Acceptable for single-user. Phase J adds concurrency limits if needed.
### ADR-004: SSE over WebSockets
Decision: Server-Sent Events for streaming.
Rationale: Simpler than WebSockets, unidirectional, no upgrade handshake, EventSource is
standard browser API.
Trade-off: Server-to-client only. Approval events use SSE from agent thread + polling fallback.
Resolution: No plan to switch. SSE is sufficient.
### ADR-005: Module-Level Approval State
Decision: tools/approval.py uses module-level _pending dict shared across all threads.
Rationale: The approval system was pre-existing; sharing state via same Python process works.
Trade-off: Breaks if ever moved to multi-process (gunicorn workers) or subprocess.
Resolution: Document the constraint. Move to SQLite if scaling is ever needed.
### ADR-006: No Authentication
Decision: No auth initially.
Rationale: Localhost-only via SSH tunnel. Auth adds complexity without security benefit
when the transport layer (SSH) is already authenticated.
Trade-off: Anyone on the VPS with localhost access can use the server.
Resolution: Phase H adds optional password gate for direct-access deployments.
### ADR-007: Approval State via Environment Variables
Decision: HERMES_EXEC_ASK and HERMES_SESSION_KEY passed via os.environ.
Rationale: tools/approval.py and terminal_tool.py already read these env vars.
Trade-off: Process-global. Two concurrent chat requests clobber each other.
Resolution: Phase B replaces with thread-local or explicit parameter passing.
---
## 14. Version History
v0.1 Initial MVP: single-file server, sync /api/chat, no streaming
v0.2 SSE streaming via /api/chat/start + /api/chat/stream
v0.2 INFLIGHT session guard, session delete rules, toast UI
v0.2 Binary file upload fixed (replaced cgi.FieldStorage with parse_multipart)
v0.2 Approval card UI wired to tools/approval.py
v0.2 Approval SSE event (immediate surface on tool invocation)
v0.3 Sprint 1 (March 30, 2026):
Bug fixes: B1 B2 B3 B4/B5 B7 B9 B11 all resolved
Architecture: LOCK on SESSIONS, section headers, structured JSON logging
Tests: 19/19 HTTP integration tests passing
Features: 10-model dropdown with provider groups, reconnect banner,
GET /api/chat/stream/status, GET /api/approval/inject_test
v0.4 Sprint 2 (March 30, 2026):
Features: image preview via /api/file/raw, rendered markdown in right panel,
table support in renderMd(), smart file icons, type badge in path bar
Tests: 8 new tests, 27/27 total passing
v0.5 [Planned] Wave 1 features: cron viewer, skills viewer, memory viewer
v0.5 Sprint 3 (March 30, 2026):
Features: sidebar nav tabs (Chat/Tasks/Skills/Memory), cron viewer,
skills viewer (search + SKILL.md preview), memory viewer
Bug fixes: B6, B10, B14
Arch: Phase D partial (require()/bad() validation helpers)
New endpoints: /api/crons, /api/crons/output, /api/crons/run, /api/crons/pause,
/api/crons/resume, /api/skills, /api/skills/content, /api/memory
Tests: 21 new tests, 48/48 total
v0.6 Sprint 4 (March 30, 2026):
Relocation: source moved to /, symlink back
Phase A partial: CSS extracted to static/style.css, served from disk
Phase B partial: per-session agent lock (SESSION_AGENT_LOCKS)
Features: session rename (inline), session search, file delete, file create
Bug fixes: B12, B8 improved, TD5 completed
New endpoints: /api/session/rename, /api/sessions/search, /api/file/delete, /api/file/create, GET /static/*
Tests: 20 new tests, 68/68 total
v0.7 Sprint 5 (March 30, 2026):
Arch: Phase A complete (JS -> static/app.js), TD2 LRU cache, TD1 thread-local, Phase C index
Features: workspace management panel + topbar quick-switch, copy message, inline file editor
New endpoints: /api/workspaces, /api/workspaces/add, /api/workspaces/remove, /api/workspaces/rename, /api/file/save
New state files: workspaces.json, last_workspace.txt, sessions/_index.json
Tests: 18 new tests, 86/86 total
v0.8 Sprint 6 (March 31, 2026):
Phase E complete: HTML to static/index.html (server.py now 903 lines, pure Python)
Phase D complete: all endpoints validated
Features: resizable panels (localStorage), cron create from UI, session JSON export
Bug fix: Escape from file editor now cancels edits
New endpoints: POST /api/crons/create, GET /api/session/export
Tests: 16 new, 106/106 total
v0.0.6 Sprint 8 (March 31, 2026):
Features: edit+regenerate messages, regenerate last response, clear conversation,
Prism.js syntax highlighting, message queue (MSG_QUEUE + drain on idle),
INFLIGHT-first loadSession (message persists on switch-away/back)
Bug fixes: A1 (reconnect banner false positive), A2 (session list scroll clip)
New endpoints: POST /api/session/clear, POST /api/session/truncate
Tests: 14 new, 139/139 total
JS: MSG_QUEUE global, updateQueueBadge(), setBusy drain logic, send() queues when busy,
loadSession checks INFLIGHT before server fetch
v0.1.0 Concurrency sweeps (March 31, 2026):
R10-R15: approval cross-session, activity bar per-session, live card
restore on switch-back, settled cards after done, model source,
newSession card clear. 190/190 tests.
v0.0.8 Sprint 10 (March 31, 2026):
Arch: server.py split into api/ modules (config, helpers, models, workspace, upload, streaming)
Features: background task cancel, cron run history, tool card UX polish
Post-sprint fixes: SSE cancel event breaks loop, Cancel button always hidden on setBusy(false),
S.activeStreamId initialized, tool-card show-more uses data attributes, version label v0.0.8,
Session.__init__ **kwargs forward-compat, test cron isolation via HERMES_HOME,
last_workspace reset in conftest between tests, tool cards grouped by assistant turn
Tests: 18 new, 167/167 total
Regressions fixed: uuid, AIAgent, has_pending, SSE cancel loop, Session.__init__ tool_calls
test_regressions.py: 10 tests -- one per introduced bug, permanent regression gate
Total after fixes: 177/177
v0.0.7 Sprint 9 (March 31, 2026):
Arch: app.js deleted; replaced by ui.js, workspace.js, sessions.js, messages.js, panels.js, boot.js
Features: tool call cards (inline collapsible, live + history), attachment persistence,
todo list panel (parses tool results from session history)
Tests: 10 new, 149/149 total
v0.9 Sprint 7 (March 31, 2026):
Features: cron edit+delete, skill create/edit/delete, memory write, session content search
Arch: Phase G partial (active_streams+uptime in /health), git init
Bug fixes: A1 (activity bar min-height), A2 (model chip sync), A3 (cron output overflow)
New endpoints: /api/crons/update, /api/crons/delete, /api/skills/save, /api/skills/delete,
/api/memory/write, /api/sessions/search (extended)
Tests: 19 new, 125/125 total
---
## 15. Sprint Log
This section records what was actually built and changed in each sprint. It is the
permanent history of the codebase. Update it at the end of every sprint.
### Sprint 1 (March 30, 2026): Bug Fixes, Arch Foundations, First Tests
**Tracks:** Bug fixes (7), Architecture (3), Tests (1)
**Test result:** 19/19 passing
**Backup:** server.py.sprint1.bak
#### Bug Fixes Applied
| ID | Description | Change |
|-----|--------------------------------------|------------------------------------------------------------------------|
| B3 | Model chip label wrong for new models | Replaced substring check with MODEL_LABELS dict; 10 models supported |
| B7 | Sidebar title overflow | Added min-width:0 to .session-item |
| B11 | /api/session GET creates session silently | Returns 400 with error message when session_id is missing |
| B2 | File input no accept attribute | Added accept= with image/*, text/*, pdf, json, common code extensions |
| B9 | Empty assistant messages render | loadSession() filters out empty-text assistant messages before render |
| B1 | Approval card missing pattern context | showApprovalCard() now appends pattern_keys to description text |
| B4/B5 | Reload mid-stream loses context | markInflight/clearInflight in localStorage; checkInflightOnBoot() shows gold reconnect banner; GET /api/chat/stream/status endpoint added |
Model dropdown also expanded from 2 options to 10, grouped by provider in .
#### Architecture Improvements Applied
| Item | Description | Change |
|--------|--------------------------------------|---------------------------------------------------------------------------|
| Arch-1 | Section headers | 8 clear # === SECTION === banners dividing server.py into logical zones |
| Arch-2 | LOCK around SESSIONS dict | get_session, new_session, delete now hold LOCK; eliminates race condition |
| Arch-3 | Structured request logging | log_request() override emits JSON per request to /tmp/webui-mvp.log |
Request log format:
{"ts": "2026-03-30T17:30:08Z", "method": "GET", "path": "/health", "status": 200, "ms": 0.1}
#### Test Suite Added
File: webui-mvp/tests/test_sprint1.py (19 tests)
File: webui-mvp/tests/__init__.py
Test categories:
Health check (1)
Session CRUD: create, load, update, delete, sort, B11 footgun (6)
Multipart parser unit tests: text file, binary/PNG (2)
HTTP upload: success, too large, no file, bad session (4)
Approval API: pending/none, inject+deny, inject+session-approve (3)
Stream status endpoint (1)
File browser: list dir, path traversal block (2)
Run tests:
cd
venv/bin/python -m pytest webui-mvp/tests/test_sprint1.py -v
#### Section 5.5 Update (B3 resolved)
The model chip label bug is now fixed. The MODEL_LABELS object in syncTopbar():
const MODEL_LABELS = {
'openai/gpt-5.4-mini': 'GPT-5.4 Mini',
'openai/gpt-4o': 'GPT-4o',
'openai/o3': 'o3',
'openai/o4-mini': 'o4-mini',
'anthropic/claude-sonnet-4.6': 'Sonnet 4.6',
'anthropic/claude-sonnet-4-5': 'Sonnet 4.5',
'anthropic/claude-haiku-3-5': 'Haiku 3.5',
'google/gemini-2.5-pro': 'Gemini 2.5 Pro',
'deepseek/deepseek-chat-v3-0324': 'DeepSeek V3',
'meta-llama/llama-4-scout': 'Llama 4 Scout',
};
$('modelChip').textContent = MODEL_LABELS[m] || (m.split('/').pop() || 'Unknown');
Fallback: splits on '/' and uses the last segment, so any unlisted model shows its
short identifier rather than a wrong hardcoded label.
#### Version History Update
v0.3 Sprint 1: B3/B7/B11/B2/B9/B1/B4/B5 bug fixes
v0.3 Sprint 1: Model dropdown expanded to 10 models in provider groups
v0.3 Sprint 1: LOCK added around SESSIONS dict (thread safety)
v0.3 Sprint 1: Section headers added throughout server.py
v0.3 Sprint 1: Structured JSON request logging via log_request() override
v0.3 Sprint 1: GET /api/chat/stream/status endpoint
v0.3 Sprint 1: Reconnect banner (markInflight/clearInflight/checkInflightOnBoot)
v0.3 Sprint 1: GET /api/approval/inject_test endpoint (test-only)
v0.3 Sprint 1: First pytest suite, 19 tests, all passing
---
## 16. Architecture Phase Priority Matrix
Quick-reference table for prioritizing architecture work. Phases are from Section 10.
| Phase | Name | Priority | Effort | Blocks | Status |
|-------|-----------------------------|----------|--------|----------------|------------|
| A+E | File Separation + Frontend | High | Medium | F | COMPLETE Sprint 6+9 (HTML->index.html, JS->6 modules, app.js deleted; server.py pure Python ~1150 lines) |
| B | Thread-Safe Request Context | Critical | Medium | nothing | PARTIAL (Sprint 4: per-session lock added; global env vars still used) |
| C | Session Store Improvements | Medium | Medium | J | PARTIAL Sprint 5 (index file + LRU cache; LRU eviction policy and pagination still open) |
| D | Input Validation | Medium | Low | nothing | COMPLETE Sprint 6 (approval/respond + file/raw hardened; all endpoints validated) |
| E | Frontend Modularization | Medium | High | requires A | Pending |
| F | API Design Cleanup | Low | Medium | requires A | Pending |
| G | Observability | Low | Low | nothing | Partial (Sprint 7: active_streams+uptime added to /health; log rotation still pending) |
| H | Authentication | Low | Medium | nothing | Pending |
| I | Test Infrastructure | High | High | requires A,D | Partial(*) |
| J | Performance | Low | High | requires C | Pending |
(*) Phase G is partial: structured request logging done in Sprint 1. Full observability
(health detail, debug/stats endpoint, log rotation) remains.
(*) Phase I is partial: HTTP integration test suite started in Sprint 1. Unit tests for
isolated modules require Phase A file split first.
Recommended execution order:
1. Phase B (thread safety): critical, low risk, no file changes needed
2. Phase D (input validation): low effort, improves error messages immediately
3. Phase A (file split): enables E, F, and full Phase I
4. Phase G remainder (health detail, debug endpoint): 1-2 hours
5. Phase C (session index): needed as session count grows
6. Phase E (frontend modules + marked.js): biggest UX improvement
7. Phase I (full test suite): after A gives us importable modules
8. Phase F, H, J: lower priority, tackle when needed
---
## 17. Working Conventions for Agent Contributors
This section is specifically for agents (Hermes instances, subagents, Codex, etc.) that
will be working on this codebase. Read this before touching any file.
### Before Making Any Change
1. Read this document (ARCHITECTURE.md) fully. Especially sections 4, 5, and the ADRs.
2. Read the relevant section of server.py by searching for the SECTION header.
3. Check the Sprint Log (Section 15) to understand what was recently changed.
4. Run the test suite first to confirm baseline: cd &&
venv/bin/python -m pytest webui-mvp/tests/test_sprint1.py -v
5. Check server health: curl -s http://127.0.0.1:8787/health
### Making Changes
Always back up server.py before a non-trivial change:
cp server.py server.py.$(date +%Y%m%d_%H%M).bak
Use exact string matching when patching. The pitfalls are documented in the
hermes-webui-mvp skill. Key ones:
- Never use sed on this file from the shell. Use execute_code with Python string replace.
- Always assert the old string is found before replacing (prevents silent no-op patches).
- Unicode escape sequences in JS (\u2026) exist as literal backslash-u in the file.
Match the file's raw content, not interpreted Python strings.
- The HTML block is a Python raw string (r"""..."""). Standard triple-quote escaping
rules do not apply inside it, but Python escape sequences \n etc. work in JS strings
inside it as literal two-character sequences.
After any change:
venv/bin/python -m py_compile webui-mvp/server.py # syntax check
curl -s http://127.0.0.1:8787/health # server still alive
venv/bin/python -m pytest webui-mvp/tests/ -v # tests still pass
### Critical Rules (do NOT regress these)
These patterns have been broken and fixed multiple times. Do not re-introduce them.
RULE-1: deleteSession() must NEVER call newSession().
Deleting does not create. If the deleted session was active and others remain,
load sessions[0]. If none remain, show empty state. See Section 5.6.
RULE-2: /api/upload must be checked BEFORE read_body() in do_POST.
read_body() consumes the request body. Upload parsing also needs the body.
Order matters. See Section 4.1.
RULE-3: run_conversation() takes task_id=, NOT session_id=.
task_id is the correct keyword argument. session_id= raises TypeError silently.
RULE-4: stream_delta_callback receives None as end-of-stream sentinel.
The on_token callback must guard: if text is None: return
RULE-5: send() must capture activeSid BEFORE any await.
The active session can change while awaits are pending. Capture first, guard on return.
RULE-6: Boot IIFE must never auto-create a session.
Only two places create sessions: the + button and send() when S.session is null.
RULE-7: All SESSIONS dict accesses must hold LOCK.
LOCK is a module-level threading.Lock(). Use: with LOCK: ...
RULE-8: do NOT expose tracebacks to API clients.
500 responses should return {"error": "Internal server error"}, not the full traceback.
(Currently traceback is exposed; fix in Phase D. Do not make it worse.)
RULE-9: Pattern_keys, not pattern_key, for multi-pattern approvals.
The approval module may include both pattern_key (singular, legacy) and pattern_keys
(plural, all matched patterns). Always iterate pattern_keys when approving.
### Adding New API Endpoints
See Section 11 for the exact code pattern. Short version:
- GET: add before the 404 fallback in do_GET
- POST: add after /api/upload check and after read_body(), before 404 fallback in do_POST
- Always validate required fields, return 400 for missing/invalid input
- Always use get_session(sid) with try/except KeyError -> 400 or 404
- Add a test in test_sprint1.py or a new test file
### Updating This Document
Update ARCHITECTURE.md whenever you:
- Fix a bug listed in Section 9 (update its row, mark resolved)
- Complete an architecture phase (update Section 16 matrix)
- Add a new endpoint (add to Section 4.1 routing table)
- Discover a new pitfall or rule (add to Section 17)
- Complete a sprint (add a new entry to Section 15)
This document is the memory of the codebase. If it is not updated, future agents will
make the same mistakes again.
---
## 18. Endpoint Reference (Current)
Complete list of all HTTP endpoints as of Sprint 1 (v0.3).
### GET Endpoints
/ Returns full HTML app (index page)
/index.html Same as /
/health {"status":"ok","sessions":N}
/api/session ?session_id=X -> full session + messages. 400 if no ID.
/api/sessions List of all session compact() dicts, sorted by updated_at
/api/list ?session_id=X&path=. -> directory listing for session workspace
/api/file ?session_id=X&path=rel -> file content (text, 200KB limit)
/api/chat/stream ?stream_id=X -> SSE stream. Long-lived. Emits token/tool/
approval/done/error events.
/api/chat/stream/status ?stream_id=X -> {"active": true/false, "stream_id": X}
/api/approval/pending ?session_id=X -> {"pending": entry_or_null}
/api/approval/inject_test ?session_id=X&pattern_key=K&command=C -> test-only endpoint.
Injects a pending approval entry into the server process.
/api/file/raw ?session_id=X&path=P -> raw file bytes with correct MIME type.
Used for image preview. Path traversal protected via safe_resolve.
Returns 404 JSON if file not found.
### POST Endpoints
/api/upload multipart/form-data. Fields: session_id, file. Returns filename.
/api/session/new {"model"?, "workspace"?} -> new session
/api/session/update {"session_id", "workspace"?, "model"?} -> updated session
/api/session/delete {"session_id"} -> {"ok": true}
/api/chat/start {"session_id", "message", "model"?, "workspace"?}
-> {"stream_id", "session_id"}. Starts agent daemon thread.
/api/chat (fallback, sync) {"session_id", "message", "model"?, "workspace"?}
-> blocks until agent finishes. Returns full result.
/api/approval/respond {"session_id", "choice": once|session|always|deny}
-> {"ok": true, "choice": choice}
### GET Endpoints Added in Sprint 3
/api/crons All cron jobs. Returns {jobs: [...]}.
/api/crons/output ?job_id=X&limit=N -> {outputs: [{filename, content}]}
/api/skills All skills. Returns {skills: [{name, description, category}]}
/api/skills/content ?name=X -> full skill data including SKILL.md content
/api/memory MEMORY.md + USER.md. Returns {memory, user, *_path, *_mtime}
### POST Endpoints Added in Sprint 3
/api/crons/run {job_id} -> triggers run in daemon thread. Returns {ok, status}.
/api/crons/pause {job_id} -> {ok, job} or 404.
/api/crons/resume {job_id} -> {ok, job} or 404.
---
## Sprint 2 Log Entry (March 30, 2026)
Added to Section 15 Sprint Log.
### Sprint 2: Rich File Preview (March 30, 2026)
**Tracks:** Features (4 sub-features), Tests (8 new)
**Test result:** 27/27 passing (19 Sprint 1 + 8 Sprint 2)
**Backup:** server.py.sprint1.bak (Sprint 1 backup; Sprint 2 is incremental)
#### Features Implemented
**Image Preview (GET /api/file/raw)**
New endpoint in do_GET:
GET /api/file/raw?session_id=X&path=relative/path
- Reads raw bytes from workspace file via safe_resolve() (path traversal protected)
- Looks up MIME type from MIME_MAP constant keyed by lowercase extension
- Falls back to 'application/octet-stream' for unknown types
- Serves bytes directly with correct Content-Type header
- No MAX_FILE_BYTES size limit (images can be large; the browser handles progressive load)
- Returns JSON 404 if file not found or not a file
Frontend: openFile() checks IMAGE_EXTS set. If image, sets
and calls showPreview('image'). The browser loads the image natively. onerror handler
shows a status message if load fails.
**Rendered Markdown Preview**
Frontend only -- uses existing GET /api/file endpoint for text content.
openFile() checks MD_EXTS set. If markdown, fetches text then calls:
$('previewMd').innerHTML = renderMd(data.content);
Preview renders in .preview-md container with full typography CSS separate from the
chat bubble .msg-body CSS (allows different sizing/spacing for the narrower side panel).
**Table Support in renderMd()**
Added a regex pass before paragraph wrapping:
- Detects blocks of pipe-delimited rows where row[1] is a separator (|---|---|)
- Converts to HTML
- Handles any number of columns
- This partially resolves B8 (renderMd missing tables)
**Smart File Icons in renderFileTree()**
New fileIcon(name, type) function maps extensions to emoji icons:
- Directories: folder icon
- Images: camera icon
- Markdown: notepad icon
- Python: snake icon
- JS/TS/JSX/TSX: circuit icon
- JSON/YAML/TOML: gear icon
- Shell scripts: terminal icon
- Everything else: document icon
**Preview Path Bar with Type Badge**
previewPath bar now has two elements:
- #previewPathText: the relative file path
- #previewBadge: colored badge with type label (image/md/extension)
Blue for images, gold for markdown, gray for code
#### New Constants Added
IMAGE_EXTS set of image extensions: .png .jpg .jpeg .gif .svg .webp .ico .bmp
MD_EXTS set of markdown extensions: .md .markdown .mdown
CODE_EXTS set of code/text extensions for reference
MIME_MAP dict: extension -> MIME type string
#### New HTML Elements
#previewPathText span inside preview path bar (was direct textContent on #previewPath)
#previewBadge colored type badge span
#previewImgWrap div centering the preview image
#previewImg element for image preview
#previewMd div for rendered markdown HTML
#### Endpoint Reference Update
Added to Section 18:
GET /api/file/raw ?session_id=X&path=P -> raw file bytes with correct MIME type.
Path traversal protected. 404 JSON if not found.
#### B8 Status Update (Section 9)
B8 (renderMd missing tables) is now PARTIAL: table parsing added in Sprint 2.
Nested lists and complex inline HTML still not handled. Full fix remains Phase E
(replace renderMd with marked.js).
### Sprint 3 (March 30, 2026): Panel Navigation + Feature Viewers
**Tracks:** Bug fixes (3), Features (3 panels + 8 API endpoints), Arch Phase D (partial)
**Tests:** 48/48 passing
**Backup:** server.py.sprint2.bak
#### New Sidebar Navigation
Four tabs at the top of the sidebar: Chat (default), Tasks, Skills, Memory.
Implemented via `.nav-tab` / `.panel-view` CSS classes. `switchPanel(name)` activates
the correct tab and panel-view, then lazy-loads panel data on first open.
#### Tasks Panel (Cron viewer)
`loadCrons()` fetches GET /api/crons, renders each job as a collapsible `.cron-item`.
`toggleCron(id)` expands/collapses the body. `loadCronOutput(jobId)` auto-loads the last
output file from GET /api/crons/output for each job.
Run Now: POST /api/crons/run starts the job in a daemon thread, returns immediately.
Pause/Resume: POST /api/crons/pause and /api/crons/resume call the cron.jobs functions.
#### Skills Panel
`loadSkills()` fetches GET /api/skills, caches in `_skillsData`. `renderSkills()` groups
by category, filters by search input. Clicking a skill calls `openSkill(name)` which
fetches GET /api/skills/content and renders in the right panel using `showPreview('md')`.
#### Memory Panel
`loadMemory()` fetches GET /api/memory (reads MEMORY.md + USER.md from
~/.hermes/memories/), renders both as markdown via renderMd() with timestamps.
#### New API Endpoints (Section 18 update)
GET /api/crons All jobs from cron.jobs.list_jobs(include_disabled=True)
GET /api/crons/output ?job_id=X&limit=N -> last N output .md files for a job
POST /api/crons/run {job_id} -> triggers run_job() in daemon thread
POST /api/crons/pause {job_id} -> pause_job(job_id)
POST /api/crons/resume {job_id} -> resume_job(job_id)
GET /api/skills All skills via tools.skills_tool.skills_list()
GET /api/skills/content ?name=X -> full skill data via skill_view(name)
GET /api/memory MEMORY.md + USER.md content and mtimes
#### Phase D Input Validation Applied
require(body, *fields) raises ValueError with clean message on missing fields
bad(handler, msg, status=400) returns clean JSON error response
Endpoints hardened: /api/session/update, /api/session/delete, /api/chat/start.
Unknown session ID on /api/session/update now returns 404 instead of 500.
#### Bug Fix Details
B6: `newSession()` now passes `inheritWs = S.session?.workspace` to /api/session/new.
Backend already accepted `workspace` param in session/new but it was never sent.
B10: `es.addEventListener('tool', ...)` now calls `removeThinking()` before updating
status and shows a compact `.msg-role + .msg-body` tool-running row. `ensureAssistantRow()`
also removes `#toolRunningRow` when first token arrives.
B14: `document.addEventListener('keydown', ...)` at global scope catches Cmd/Ctrl+K
and calls `newSession()` if not busy.
### Sprint 4 (March 30, 2026): Relocation + Session Power Features + Phase A/B
**Tracks:** Bugs (B12, B8, TD5), Features (rename, search, file ops), Arch (Phase A/B start), Relocation
**Tests:** 68/68 passing
**Backup:** server.py.sprint2.bak (last full backup; Sprint 3 and 4 are incremental)
#### Source Relocation
Moved /webui-mvp/ to /.
Symlink: /webui-mvp ->
The symlink means all existing import paths (sys.path.insert for hermes-agent modules)
continue working unchanged. start.sh updated to reference new canonical path.
Safe from: git pull, git reset --hard, git stash on hermes-agent repo.
NOT safe from: git clean -fd (would delete symlink but not the target).
Disk failure: still a single-copy risk. Use git init + push when ready.
#### Phase A: CSS Extracted
/static/style.css: the 23KB CSS block from the Python raw string.
server.py no longer contains any CSS. GET /static/* handler serves disk files.
server.py shrunk by ~200 lines.
#### Phase B: Per-Session Agent Lock
SESSION_AGENT_LOCKS = {} keyed by session_id, each value is a threading.Lock().
_get_session_agent_lock(sid) returns the lock, creating it if needed.
_run_agent_streaming() wraps the env var block with: with _agent_lock: ...
This prevents two concurrent requests for the same session from overwriting env vars
mid-execution. Two concurrent requests for DIFFERENT sessions are still unsafe (env vars
are process-global). Full fix requires removing env var usage entirely (Phase B complete).
#### New Endpoints
GET /static/* Serves files from /static/ with
correct Content-Type. Currently serves style.css.
POST /api/session/rename {session_id, title} -> {session: compact}. Truncates to 80 chars.
GET /api/sessions/search ?q=X -> sessions whose title contains q (case-insensitive).
Empty q returns all sessions (same as /api/sessions).
POST /api/file/delete {session_id, path} -> {ok: true}. Path traversal protected.
POST /api/file/create {session_id, path, content?} -> {ok, path}. Errors if exists.
### Sprint 5 (March 30, 2026): Phase A Complete + Workspace + Edit + Copy
**Tracks:** Arch (Phase A complete, TD1/TD2/TD6/Phase C), Features (3), Tests (18)
**Tests:** 86/86 passing
#### Phase A Complete: static/app.js
Extracted 902-line JavaScript from server.py HTML string to /static/app.js.
server.py now: Python code + thin HTML skeleton (~875 lines, down from 1778).
Layout: server.py imports nothing from static/; the HTML just has and <script src>.
Served via GET /static/* handler added in Sprint 4.
node --check validates app.js on every sprint.
#### TD2: LRU SESSIONS Cache
SESSIONS changed to collections.OrderedDict.
get_session(): SESSIONS.move_to_end(sid) on hit; on miss: load from disk, add, move_to_end, evict if over SESSIONS_MAX=100.
new_session(): same eviction logic on insert.
Result: memory usage capped regardless of session count.
#### TD1: Thread-Local Env Context
_thread_ctx = threading.local() added to Server Globals.
_set_thread_env(**kwargs) and _clear_thread_env() set/clear _thread_ctx.env.
_run_agent_streaming() calls _set_thread_env() before env var writes, _clear_thread_env() in outer finally.
Process-level os.environ writes still exist as fallback (needed until terminal tool reads thread-local).
#### Phase C: Session Index File
SESSION_INDEX_FILE = SESSION_DIR / '_index.json'.
_write_session_index(): builds compact() list from SESSIONS + disk files, writes JSON.
Called in Session.save() -- keeps index always current.
all_sessions(): reads index JSON first (one file read); overlays in-memory SESSIONS; falls back to full glob scan on error.
Index files starting with '_' are skipped during full scan to avoid recursion.
#### New Workspace Infrastructure
WORKSPACES_FILE = ~/.hermes/webui-mvp/workspaces.json
LAST_WORKSPACE_FILE = ~/.hermes/webui-mvp/last_workspace.txt
load_workspaces() / save_workspaces() / get_last_workspace() / set_last_workspace() helpers.
new_session() now calls get_last_workspace() as default instead of DEFAULT_WORKSPACE.
set_last_workspace() called in /api/session/update and /api/chat/start.
#### New Endpoints (Sprint 5)
GET /api/workspaces {workspaces: [...], last: path}
POST /api/workspaces/add {path, name?} -- validates exists+dir, no duplicates
POST /api/workspaces/remove {path} -- removes from list, ok even if not present
POST /api/workspaces/rename {path, name} -- updates display name, 404 if not found
POST /api/file/save {session_id, path, content} -- write text to existing file
### Sprint 6 (March 31, 2026): Polish + Resize + Cron Create + Phase E
**Tests:** 106/106 passing
**Backup:** server.py.sprint5.bak
#### Phase E Complete: static/index.html
The HTML = r triple-quoted string (197 lines, 12682 chars) was extracted to
/static/index.html and served via disk read on each request.
server.py is now pure Python: zero HTML/CSS/JS inline. All static content is in static/.
Static file layout (final):
static/index.html (Sprint 6) -- HTML template
static/style.css (Sprint 4) -- all CSS
static/app.js (Sprint 5) -- all JavaScript
server.py line count progression: 1778 (S1) -> 1042 (S5) -> 903 (S6)
#### Phase D Complete
/api/approval/respond: validates session_id present; choice must be one of
(once, session, always, deny); returns 400 on invalid.
/api/file/raw: validates session_id present; try/except KeyError returns 404.
#### New Endpoints
POST /api/crons/create {prompt, schedule, name?, deliver?, skills?, model?}
-> {ok: true, job: {...}} or 400 on invalid schedule/missing fields.
Uses cron.jobs.create_job() directly.
GET /api/session/export ?session_id=X
-> full session JSON with Content-Disposition: attachment header.
Includes all messages, workspace, model, timestamps.
#### Resizable Panels
_initResizePanels() called from boot IIFE. Creates mousedown listeners on #sidebarResize
and #rightpanelResize. On mousemove: computes delta and clamps to min/max. On mouseup:
saves width to localStorage. Widths restored at boot via localStorage.getItem().
CSS: .resize-handle with position:absolute, width:5px, cursor:col-resize.
body.resizing added during drag to suppress text selection.