Files
hermes-webui/tests/test_csv_table_rendering.py
2026-05-13 11:50:31 +08:00

142 lines
5.7 KiB
Python

"""Test: CSV table rendering (#485)"""
import re
def test_csv_extension_regex():
"""Verify _CSV_EXTS regex is defined."""
with open('static/ui.js') as f:
src = f.read()
assert '_CSV_EXTS' in src, "Missing _CSV_EXTS regex"
assert '.csv' in src, "CSV regex should match .csv extension"
def test_csv_fence_block_handler():
"""Verify fenced ```csv blocks are handled."""
with open('static/ui.js') as f:
src = f.read()
assert "lang==='csv'" in src, "Missing csv language detection in fence handler"
assert 'csv-table' in src, "Missing csv-table class for fenced CSV rendering"
assert 'csv-table-wrap' in src, "Missing csv-table-wrap class"
def test_csv_fence_renders_table_structure():
"""Verify fenced CSV blocks produce proper table HTML."""
with open('static/ui.js') as f:
src = f.read()
# Should have thead, tbody, th, td
assert '<thead>' in src, "CSV table should have <thead>"
assert '<tbody>' in src, "CSV table should have <tbody>"
# In the fence handler section
fence_section = src[src.find("lang==='csv'"):src.find("lang==='csv'") + 800]
assert '<th>' in fence_section, "CSV headers should use <th>"
assert '<td>' in fence_section, "CSV body should use <td>"
def test_csv_fence_fallback_for_insufficient_rows():
"""Verify CSV with < 2 rows falls back to code block."""
with open('static/ui.js') as f:
src = f.read()
fence_section = src[src.find("lang==='csv'"):src.find("lang==='csv'") + 800]
assert 'rows.length>=2' in fence_section, "Should check for at least 2 rows"
assert '<pre><code' in fence_section, "Fallback should render as <pre><code>"
def test_csv_media_file_handler():
"""Verify MEDIA: CSV files trigger inline loading."""
with open('static/ui.js') as f:
src = f.read()
assert 'csv-inline-load' in src, "Missing csv-inline-load class for MEDIA: CSV"
assert 'csv_loading' in src, "Missing csv_loading i18n key usage"
def test_loadCsvInline_function():
"""Verify loadCsvInline lazy-load function exists."""
with open('static/ui.js') as f:
src = f.read()
assert 'function loadCsvInline' in src, "Missing loadCsvInline function"
def test_csv_inline_max_size():
"""Verify CSV inline rendering has a size cap."""
with open('static/ui.js') as f:
src = f.read()
csv_section = src[src.find('function loadCsvInline'):src.find('function loadCsvInline') + 2000]
assert 'CSV_MAX_SIZE' in csv_section, "Should have CSV_MAX_SIZE constant"
assert 'csv_too_large' in csv_section, "Should use csv_too_large i18n for oversized files"
def test_csv_auto_detect_separator():
"""Verify CSV handler auto-detects separator."""
with open('static/ui.js') as f:
src = f.read()
csv_section = src[src.find('function loadCsvInline'):src.find('function loadCsvInline') + 2000]
assert 'separators' in csv_section, "Should have separator detection"
assert ';' in csv_section, "Should detect semicolon separator"
assert 'tab' in csv_section.lower() or '\\t' in csv_section, "Should detect tab separator"
def test_csv_quote_stripping():
"""Verify CSV handler strips surrounding quotes from fields."""
with open('static/ui.js') as f:
src = f.read()
assert "replace(/^[\"']|[\"']$/g,'')" in src, "Should strip quotes from CSV fields"
def test_csv_error_handling():
"""Verify CSV error and empty data handling."""
with open('static/ui.js') as f:
src = f.read()
csv_section = src[src.find('function loadCsvInline'):src.find('function loadCsvInline') + 2500]
assert 'csv_error' in csv_section, "Should use csv_error i18n on fetch failure"
assert 'csv_no_data' in csv_section, "Should use csv_no_data i18n for insufficient data"
def test_csv_loadCsvInline_called_after_render():
"""Verify loadCsvInline is called by the consolidated post-render pass."""
with open('static/ui.js') as f:
src = f.read()
assert 'requestAnimationFrame(()=>postProcessRenderedMessages(inner))' in src
idx = src.find('function postProcessRenderedMessages')
body = src[idx:idx + 500]
assert 'loadCsvInline(container)' in body, "post-process should call loadCsvInline once per render"
def test_csv_line_ending_normalization():
"""Verify CSV handler normalizes line endings."""
with open('static/ui.js') as f:
src = f.read()
csv_section = src[src.find('function loadCsvInline'):src.find('function loadCsvInline') + 2000]
assert '\\r\\n' in csv_section, "Should handle \\r\\n line endings"
assert '\\r' in csv_section, "Should handle \\r line endings"
def test_csv_i18n_keys():
"""Verify CSV i18n keys exist in all 7 locales."""
with open('static/i18n.js') as f:
src = f.read()
required_keys = ['csv_loading', 'csv_too_large', 'csv_no_data', 'csv_error']
for key in required_keys:
count = src.count(f"{key}:")
assert count >= 8, f"Key '{key}' found {count} times, expected >= 8 (one per locale)"
def test_csv_css_classes():
"""Verify CSV table CSS classes are defined."""
with open('static/style.css') as f:
src = f.read()
required_classes = ['csv-table-wrap', 'csv-table', 'csv-table th', 'csv-table td']
for cls in required_classes:
assert cls in src, f"Missing CSS: {cls}"
# Check for hover effect
assert 'csv-table tbody tr:hover' in src, "Missing hover effect for CSV rows"
def test_csv_not_matched_by_image_exts():
"""Verify .csv is NOT in _IMAGE_EXTS."""
with open('static/ui.js') as f:
src = f.read()
match = re.search(r"const _IMAGE_EXTS=/([^/]+)/i", src)
assert match
exts = match.group(1)
assert 'csv' not in exts.lower(), ".csv should NOT be in _IMAGE_EXTS"