"""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/ 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"