Files
hermes-webui/tests/test_skills_category_collapse.py
T
nesquena-hermes 33a145a669 release: v0.50.240
## Release v0.50.240

Batch release of 13 PRs that passed full triage + code review + test suite (3199 tests, 0 failures).

---

### Added

- **Compact tool activity mode** (`simplified_tool_calling`, default on) — groups tool calls and thinking traces into a single collapsed "Activity" disclosure card per assistant turn. Also adds a new **Calm Console** theme with earth/slate palette and serif prose. @Michaelyklam — #1282
- **PDF first-page preview** — `MEDIA:` `.pdf` files render a canvas thumbnail via PDF.js CDN (4 MB cap). **HTML sandbox iframe** — `.html`/`.htm` files render inline in a sandboxed `<iframe srcdoc>` (256 KB cap). 10 i18n keys × 7 locales. @bergeouss — #1280, closes #480 #482
- **Inline Excalidraw diagram preview** — `.excalidraw` files render as pure SVG (no external deps; rectangles, ellipses, diamonds, text, lines, arrows, freehand; 512 KB cap). @bergeouss — #1279, closes #479
- **Inline CSV table rendering** — fenced `csv` blocks and `MEDIA:` CSV files render as scrollable HTML tables with auto-separator detection. @bergeouss — #1277, closes #485
- **Inline SVG, audio, and video rendering** — SVG as `<img>`, audio as `<audio controls>`, video as `<video controls>`. @bergeouss — #1276, closes #481
- **Batch session select mode** — multi-select sessions for bulk Archive/Delete/Move. 11 i18n keys × 7 locales. @bergeouss — #1275, closes #568
- **Collapsible skill category headers** — click to collapse/expand without re-render; state persists across filter cycles. @bergeouss — #1281
- **`providers.only_configured` setting** — opt-in flag to restrict the model picker to explicitly configured providers. @KingBoyAndGirl — #1268
- **OpenCode Go model catalog** — adds Kimi K2.6, DeepSeek V4 Pro/Flash, MiMo V2.5/Pro, Qwen3.6/3.5 Plus. @nesquena-hermes — #1284, closes #1269

### Fixed

- **Profile `TERMINAL_CWD` TypeError** — `_build_agent_thread_env()` helper merges env before `_set_thread_env()` call. @hi-friday — #1266
- **Service worker subpath cache bypass** — regex now matches `/api/*` under any mount prefix. @Michaelyklam — #1278
- **SSE client disconnect leaks** — `TimeoutError`/`OSError` treated as clean disconnects; server backlog 64, threads daemonized; session list renders before saved-session restore. @KayZz69 — #1267
- **i18n locale corrections** — Korean MCP strings (23), Chinese MCP strings (23), zh-Hant missing keys (41), de missing keys (229). @bergeouss — #1274, closes #1273

---

### Test results

```
3199 passed, 2 skipped, 3 xpassed in 72.79s
```

### PRs on hold (not included)

#1265 (draft), #1271 (superseded by #1266), #1272 (skipped XSS tests), #1232 (partial test run), #1222 (review questions open), #1134 (live-server tests), #1132 (superseded by #1134), #1108 (negative UX review), #1084 (empty description)
2026-04-29 17:42:32 -07:00

154 lines
6.3 KiB
Python

"""Tests for collapsible skill categories in the Skills panel.
Validates that renderSkills() produces collapsible category headers
with chevron toggles, click handlers, and persisted collapse state.
"""
import os
import re
import pytest
def _readpanels():
with open(os.path.join('static', 'panels.js')) as f:
return f.read()
def _readcss():
with open(os.path.join('static', 'style.css')) as f:
return f.read()
# ── State variable ──────────────────────────────────────────────────────────
class TestCollapseState:
"""A Set must track collapsed categories across re-renders."""
def test_collapsed_cats_set_exists(self):
p = _readpanels()
assert '_collapsedCats' in p, '_collapsedCats Set must exist'
assert 'new Set()' in p, '_collapsedCats must be initialized as Set'
def test_toggle_function_exists(self):
p = _readpanels()
assert '_toggleCatCollapse' in p, '_toggleCatCollapse() function must exist'
# ── renderSkills produces collapsible headers ──────────────────────────────
class TestRenderSkillsCollapse:
"""renderSkills() must render category headers with chevron icons and click handlers."""
def test_chevron_icon_used_instead_of_folder(self):
p = _readpanels()
idx = p.find('function renderSkills(')
body = p[idx:idx + 2000]
assert 'chevron-right' in body, 'Must use chevron-right icon instead of folder'
assert "li('folder'" not in body, 'Must not use folder icon anymore'
def test_cat_header_has_dataset_cat(self):
p = _readpanels()
idx = p.find('function renderSkills(')
body = p[idx:idx + 2000]
assert 'dataset.cat' in body, 'Header must store category in data-cat attribute'
def test_cat_header_has_click_handler(self):
p = _readpanels()
idx = p.find('function renderSkills(')
body = p[idx:idx + 2000]
assert 'hdr.onclick' in body or 'onclick' in body, 'Header must have onclick handler'
def test_collapsed_class_toggled(self):
p = _readpanels()
idx = p.find('function renderSkills(')
body = p[idx:idx + 2000]
assert 'collapsed' in body, 'Must apply collapsed class based on state'
def test_skill_items_hidden_when_collapsed(self):
p = _readpanels()
idx = p.find('function renderSkills(')
body = p[idx:idx + 2000]
assert "'none'" in body and "style.display" in body, 'Skill items must be hidden when category is collapsed'
def test_chevron_rotation_on_collapse(self):
p = _readpanels()
idx = p.find('function renderSkills(')
body = p[idx:idx + 2000]
assert 'rotate(90deg)' in body, 'Chevron must rotate 90deg when expanded'
def test_renderSkills_preserves_search_query(self):
"""Search query must still be read and applied before grouping."""
p = _readpanels()
idx = p.find('function renderSkills(')
body = p[idx:idx + 500]
assert 'skillsSearch' in body, 'Must read search input value'
assert 'toLowerCase().includes(query)' in body, 'Must filter by name/description/category'
# ── _toggleCatCollapse DOM manipulation ────────────────────────────────────
class TestToggleCatCollapse:
"""_toggleCatCollapse() must toggle DOM without full re-render."""
def test_toggles_set_membership(self):
p = _readpanels()
idx = p.find('function _toggleCatCollapse(')
body = p[idx:idx + 500]
assert '_collapsedCats.has(cat)' in body
assert '_collapsedCats.delete(cat)' in body
assert '_collapsedCats.add(cat)' in body
def test_queries_skills_category_elements(self):
p = _readpanels()
idx = p.find('function _toggleCatCollapse(')
body = p[idx:idx + 800]
assert '.skills-category' in body, 'Must query .skills-category elements'
def test_matches_by_dataset_cat(self):
p = _readpanels()
idx = p.find('function _toggleCatCollapse(')
body = p[idx:idx + 800]
assert 'header.dataset.cat === cat' in body or 'dataset.cat' in body, 'Must match category by data attribute'
def test_toggles_skill_item_display(self):
p = _readpanels()
idx = p.find('function _toggleCatCollapse(')
body = p[idx:idx + 800]
assert '.skill-item' in body, 'Must query .skill-item elements'
assert "display = collapsed ? 'none'" in body or "style.display" in body, 'Must toggle display property'
def test_toggles_chevron_rotation(self):
p = _readpanels()
idx = p.find('function _toggleCatCollapse(')
body = p[idx:idx + 800]
assert '.cat-chevron' in body, 'Must select chevron element'
assert 'rotate' in body, 'Must toggle rotation on chevron'
# ── CSS ────────────────────────────────────────────────────────────────────
class TestCSSClasses:
"""CSS must support collapsible categories."""
def test_cat_chevron_class(self):
css = _readcss()
assert '.cat-chevron' in css, '.cat-chevron class must exist in CSS'
def test_cat_chevron_has_fixed_size(self):
css = _readcss()
m = re.search(r'\.cat-chevron\{[^}]+\}', css)
assert m, '.cat-chevron rule must exist'
assert 'width' in m.group(), 'Chevron must have fixed width'
assert 'flex-shrink' in m.group(), 'Chevron must not shrink'
def test_skills_cat_header_user_select_none(self):
css = _readcss()
m = re.search(r'\.skills-cat-header\{[^}]+\}', css)
assert m, '.skills-cat-header rule must exist'
assert 'user-select' in m.group(), 'Header must have user-select:none to prevent text selection on click'
def test_skills_cat_header_has_cursor_pointer(self):
css = _readcss()
m = re.search(r'\.skills-cat-header\{[^}]+\}', css)
assert m, '.skills-cat-header rule must exist'
assert 'cursor:pointer' in m.group(), 'Header must have cursor:pointer'