Files
hermes-webui/tests/test_session_batch_select.py
2026-05-02 23:05:55 +08:00

257 lines
11 KiB
Python

"""Test: session batch select mode functions exist in sessions.js (#568)"""
import re
def test_batch_select_state_variables():
"""Verify batch select state variables are declared."""
with open('static/sessions.js') as f:
src = f.read()
assert '_sessionSelectMode' in src, "Missing _sessionSelectMode variable"
assert '_selectedSessions' in src, "Missing _selectedSessions variable"
assert 'new Set()' in src, "Selected sessions should use Set"
def test_batch_select_functions_exist():
"""Verify all batch select functions are defined."""
with open('static/sessions.js') as f:
src = f.read()
required_funcs = [
'toggleSessionSelectMode',
'exitSessionSelectMode',
'toggleSessionSelect',
'selectAllSessions',
'deselectAllSessions',
'_updateBatchActionBar',
'_renderBatchActionBar',
'_showBatchProjectPicker',
]
for fn in required_funcs:
assert f'function {fn}(' in src, f"Missing function: {fn}"
def test_batch_select_checkbox_rendering():
"""Verify checkbox is rendered when in select mode."""
with open('static/sessions.js') as f:
src = f.read()
assert 'session-select-cb' in src, "Missing session-select-cb class"
assert 'session-select-cb-wrapper' in src, "Missing session-select-cb-wrapper class"
assert "cb.type='checkbox'" in src, "Checkbox should be type checkbox"
def test_batch_select_intercepts_navigation():
"""Verify select mode intercepts session navigation."""
with open('static/sessions.js') as f:
src = f.read()
assert "_sessionSelectMode" in src
# Should have early return when in select mode
assert 'toggleSessionSelect(s.session_id)' in src, \
"Pointerup handler should call toggleSessionSelect in select mode"
def test_batch_checkbox_sets_selection_without_row_double_toggle():
"""Clicking the checkbox itself must not also trigger row-level toggling."""
with open('static/sessions.js') as f:
src = f.read()
assert 'function setSessionSelected(sid, selected)' in src, \
"Checkbox changes should set explicit state instead of toggling blindly"
assert 'cb.onchange=(e)=>{e.stopPropagation();setSessionSelected(s.session_id,cb.checked);};' in src, \
"Checkbox change must use its checked state"
assert 'cb.onpointerup=(e)=>{e.stopPropagation();};' in src, \
"Checkbox pointerup must not bubble to the row pointerup handler"
assert 'cbWrapper.onpointerup=(e)=>{e.stopPropagation();};' in src, \
"Checkbox wrapper pointerup must not bubble to the row pointerup handler"
def test_batch_select_escape_handler():
"""Verify Escape key exits select mode."""
with open('static/sessions.js') as f:
src = f.read()
assert "e.key==='Escape'&&_sessionSelectMode" in src, \
"Should have Escape key handler for select mode"
def test_batch_select_toggle_button():
"""Verify select mode toggle button is rendered."""
with open('static/sessions.js') as f:
src = f.read()
assert 'session-select-toggle' in src, "Missing session-select-toggle class"
assert 'toggleSessionSelectMode' in src, "Missing toggleSessionSelectMode call"
def test_batch_select_bar_element():
"""Verify batch action bar DOM element is created."""
with open('static/sessions.js') as f:
src = f.read()
assert 'batchActionBar' in src, "Missing batchActionBar element"
assert 'batch-action-bar' in src, "Missing batch-action-bar CSS class"
assert 'batch-action-btn' in src, "Missing batch-action-btn class"
def test_batch_action_bar_overrides_css_hidden_state():
"""Selected sessions must make the fixed action bar visible."""
with open('static/sessions.js') as f:
src = f.read()
assert "if(count>0){_renderBatchActionBar();}" in src, \
"Updating selected count must render action buttons, not just reveal an empty bar"
assert "t('session_selected_count',_selectedSessions.size)" in src, \
"Selected count must pass the selected session count to i18n"
assert "t('session_batch_archive_confirm',ids.length)" in src, \
"Batch archive confirmation must pass selected session count to i18n"
assert "t('session_batch_delete_confirm',ids.length)" in src, \
"Batch delete confirmation must pass selected session count to i18n"
assert "bar.innerHTML='';bar.style.display=_selectedSessions.size>0?'flex':'none'" in src, \
"Rendering the action bar must explicitly show it when selections exist"
assert "batchBar.style.display='flex'" in src, \
"Session list render must explicitly show the action bar in select mode"
def test_batch_action_bar_is_sidebar_inline_not_global_footer():
"""Batch actions should appear in the session list, not over the composer."""
with open('static/sessions.js') as f:
js = f.read()
with open('static/style.css') as f:
css = f.read()
assert "list.appendChild(batchBar)" in js, \
"Batch action bar should be rendered inside the session list"
assert "document.body.appendChild(batchBar)" not in js, \
"Batch action bar must not be mounted as a global footer"
assert ".batch-action-bar{display:none;margin:" in css, \
"Batch action bar should use inline sidebar spacing"
assert "position:fixed" not in css[css.find(".batch-action-bar{"):css.find(".batch-count{")], \
"Batch action bar must not be fixed to the bottom of the viewport"
def test_batch_project_picker_is_anchored_to_batch_actions():
"""Batch move project picker should open inside the sidebar action bar."""
with open('static/sessions.js') as f:
js = f.read()
with open('static/style.css') as f:
css = f.read()
assert "const bar=$('batchActionBar');if(!bar)return;" in js, \
"Batch project picker should anchor to the batch action bar"
assert "picker.className='project-picker batch-project-picker'" in js, \
"Batch project picker needs its own inline styling hook"
assert "bar.appendChild(picker)" in js, \
"Batch project picker should render inside the batch action bar"
assert "document.body.appendChild(picker);picker.style.cssText='position:fixed" not in js, \
"Batch project picker must not use the old global fixed placement"
assert ".batch-action-bar .batch-project-picker{position:static;" in css, \
"Batch project picker should override the shared absolute project picker style"
assert css.find(".project-picker{") < css.find(".batch-action-bar .batch-project-picker{"), \
"Batch project picker override must come after the shared project picker rule"
def test_streaming_zero_message_sessions_stay_visible_after_reload():
"""In-flight sessions may have zero saved messages during reload recovery."""
with open('static/sessions.js') as f:
src = f.read()
assert "_isSessionEffectivelyStreaming(s)" in src, \
"Streaming sessions must bypass the zero-message sidebar filter"
assert "!!s.active_stream_id" in src, \
"Sessions with persisted active stream IDs must remain visible after reload"
assert "!!s.pending_user_message" in src, \
"Sessions with pending user turns must remain visible after reload"
def test_boot_does_not_drop_zero_message_inflight_session():
"""Reloading /session/<id> during a running turn must keep the session open."""
with open('static/boot.js') as f:
src = f.read()
assert "const _restoredInFlight = S.session && (" in src, \
"Boot must detect restored in-flight sessions before ephemeral cleanup"
assert "S.session.active_stream_id" in src, \
"Boot must treat active stream IDs as real sessions"
assert "S.session.pending_user_message" in src, \
"Boot must treat pending user messages as real sessions"
assert "&& !_restoredInFlight" in src, \
"Zero-message cleanup must not run for in-flight sessions"
def test_batch_select_i18n_keys():
"""Verify all batch select i18n keys exist in all locales."""
with open('static/i18n.js') as f:
src = f.read()
required_keys = [
'session_select_mode',
'session_select_mode_desc',
'session_select_all',
'session_deselect_all',
'session_selected_count',
'session_batch_archive',
'session_batch_delete',
'session_batch_move',
'session_batch_delete_confirm',
'session_batch_archive_confirm',
'session_no_selection',
]
locales = ['en', 'ru', 'es', 'de', 'zh', 'zh-Hant', 'ko']
for key in required_keys:
for locale in locales:
# Check if the key exists in the locale block
if locale == 'zh-Hant':
pattern = rf"'{locale}'\s*:.*?{key}"
else:
pattern = rf"{locale}\s*:.*?{key}"
# Simpler check: just verify the key string with colon exists
assert f"{key}:" in src, f"Missing i18n key '{key}' in i18n.js"
# Count occurrences - each key should appear in all 7 locales
for key in required_keys:
count = src.count(f"{key}:")
assert count >= 8, f"Key '{key}' found {count} times, expected >= 8 (one per locale) (one per locale)"
def test_i18n_string_placeholder_interpolation_supported():
"""String-valued translations with {0} placeholders should interpolate args."""
with open('static/i18n.js') as f:
src = f.read()
assert "String(val).replace(/\\{(\\d+)\\}/g" in src, \
"t() must interpolate {0}-style placeholders for string-valued translations"
assert "Object.prototype.hasOwnProperty.call(args, idx)" in src, \
"t() must preserve unknown placeholders instead of replacing with undefined"
def test_batch_select_css_exists():
"""Verify batch select CSS classes are defined."""
with open('static/style.css') as f:
src = f.read()
required_classes = [
'session-select-toggle',
'session-select-bar',
'batch-exit-btn',
'batch-select-all-btn',
'session-select-cb-wrapper',
'session-select-cb',
'session-item.selected',
'batch-action-bar',
'batch-count',
'batch-action-btn',
'batch-action-btn-danger',
]
for cls in required_classes:
assert cls in src, f"Missing CSS class: .{cls}"
def test_batch_select_mode_flags():
"""Verify select mode properly toggles state."""
with open('static/sessions.js') as f:
src = f.read()
# toggleSessionSelectMode should flip the flag
assert '_sessionSelectMode=!_sessionSelectMode' in src, \
"toggleSessionSelectMode should flip _sessionSelectMode"
# exitSessionSelectMode should clear state
assert '_sessionSelectMode=false' in src, \
"exitSessionSelectMode should set _sessionSelectMode=false"
assert '_selectedSessions.clear()' in src, \
"Exit should clear selected sessions"
def test_batch_delete_uses_confirm_dialog():
"""Verify batch delete shows confirmation dialog."""
with open('static/sessions.js') as f:
src = f.read()
# The delete handler should call showConfirmDialog with batch message
assert "session_batch_delete_confirm" in src, \
"Batch delete should use session_batch_delete_confirm i18n key"
assert "showConfirmDialog" in src, \
"Should use showConfirmDialog for batch operations"