Commit Graph

417 Commits

Author SHA1 Message Date
GitHub Copilot d4acdbb4f1 Fix SVG clip path count by increasing console height
Rich's SVG export creates fewer clip paths than content lines.
Work around by adding +2 to console height.
2026-01-24 17:00:23 +00:00
GitHub Copilot 1a5222ab2a Simplify terminal resize - no more size toggling
Remove all the -1 column toggle tricks for forcing tmux redraws.
Just set the size directly.
2026-01-24 16:53:53 +00:00
GitHub Copilot a771e1469e Fix screenshot rendering and reconnect behavior
- Skip empty placeholder cells for wide characters in SVG
- Single redraw on reconnect (integrated into set_terminal_size)
- Sync pyte to PTY size with redraw trigger for screenshots
- Fix extra line in SVG by not adding newline after last row

Bump version to 0.2.10
2026-01-24 16:49:23 +00:00
GitHub Copilot 3f265f19dc Fix double redraw flash on reconnect
- Remove separate force_redraw on WebSocket connect
- Integrate size toggle into set_terminal_size for single redraw
- Update test to expect two executor calls (toggle pattern)
2026-01-24 16:48:53 +00:00
GitHub Copilot 032d46b2a9 Skip empty placeholder cells in SVG rendering
Wide characters (emoji, CJK) in pyte are followed by empty placeholder
cells. Skipping these fixes Rich's SVG character positioning.
2026-01-24 16:46:54 +00:00
GitHub Copilot 8285e9f910 Force tmux redraw when pyte syncs to PTY size
Toggle PTY size after resize to trigger tmux redraw, with brief
delay to let redraw data arrive before taking screenshot
2026-01-24 16:45:19 +00:00
GitHub Copilot a6e37b87c3 Fix extra line in SVG screenshot
Don't add newline after last row to prevent Rich from creating an extra line
2026-01-24 16:43:43 +00:00
GitHub Copilot 98b5a1fc2a Fix terminal size sync and tmux redraw
- Sync pyte screen to actual PTY size before screenshots
- Toggle terminal size to force full tmux redraw on reconnect
- Query PTY size with TIOCGWINSZ to detect external resizes

Bump version to 0.2.9
2026-01-24 16:40:12 +00:00
GitHub Copilot 9ae00e113d Sync pyte screen to actual PTY size before screenshots
Query PTY size with TIOCGWINSZ and resize pyte screen to match.
Fixes tmux status bar wrapping when terminal was resized externally.
2026-01-24 16:39:29 +00:00
GitHub Copilot 3da68eaaf8 Improve tmux redraw by toggling terminal size
Toggle width -1 then back to force tmux to fully redraw all panes
2026-01-24 16:32:49 +00:00
GitHub Copilot c01b7c1091 Fix terminal resize and reconnect behavior
- Force terminal redraw on WebSocket reconnect (fixes tmux display)
- Simplify screenshot dimensions (use DEFAULT_TERMINAL_SIZE for new sessions)
- Track last known terminal size for reconnection
- Fix trailing whitespace in tests

Bump version to 0.2.8
2026-01-24 16:30:14 +00:00
GitHub Copilot ffd0f91d9d Simplify screenshot dimensions
Remove width/height query params from screenshot endpoint.
New sessions created by screenshot use DEFAULT_TERMINAL_SIZE.
Existing sessions keep their current size.
2026-01-24 16:29:26 +00:00
GitHub Copilot 728681a195 Force terminal redraw on reconnect
- Track last known terminal size in TerminalSession
- Add force_redraw() method that re-sends SIGWINCH to trigger redraw
- Call force_redraw() when WebSocket reconnects to existing session
- Helps tmux and similar apps restore proper display after disconnect
2026-01-24 16:28:50 +00:00
GitHub Copilot 2f61bd7747 Don't resize terminal on session disconnect
Rename DISCONNECT_RESIZE to DEFAULT_TERMINAL_SIZE
Update tests for removed _resize_on_disconnect and stricter available check

Bump version to 0.2.7
2026-01-24 16:11:39 +00:00
GitHub Copilot b5b144dcd0 Initialize sparklines to start at zero
Bump version to 0.2.6
2026-01-24 12:55:45 +00:00
GitHub Copilot cc2ab79859 Fix Docker sparklines and Ctrl-C exit
- Fix Ctrl-C to exit immediately by setting exit_event before cleanup
- Filter Docker containers by compose project name to match correct stack
- Derive compose project from manifest directory (matches docker-compose default)
- Improve Docker socket availability check to test actual connectivity
- Add DOCKER_HOST env var support for alternate socket paths
- Better error logging for socket permission issues

Bump version to 0.2.5
2026-01-24 12:53:09 +00:00
GitHub Copilot 6ab7503748 Clean up Docker stats logging
- Remove excessive debug logging
- Add single warning when no containers found (with hint about socket mount)
- Use HTTP/1.0 to avoid chunked encoding complexity
- Simplify response parsing
2026-01-24 12:30:39 +00:00
GitHub Copilot 13d2569c37 Bump version to 0.2.4 2026-01-24 12:27:23 +00:00
GitHub Copilot 83b6503501 Improve Docker API response parsing
- Refactor into separate methods for cleaner code
- Better handling of chunked transfer encoding
- Add more debug logging to diagnose parsing failures
- Log body preview when JSON not found
2026-01-24 12:25:08 +00:00
GitHub Copilot bf54d44c7e Bump version to 0.2.3 2026-01-24 12:22:25 +00:00
GitHub Copilot 8d966ca203 Add debug logging for Docker stats collection
Helps diagnose empty sparklines by logging:
- Container discovery results and matches
- Stats request failures
- CPU calculation results
- What containers/services were found vs expected
2026-01-24 12:21:55 +00:00
GitHub Copilot de5ea155a0 Throttle SSE notifications and debounce client refreshes
Server-side:
- Limit SSE notifications to max once per second per route
- Track last notification time per route

Client-side:
- Debounce screenshot refreshes with 2s minimum interval per tile
- Pending refreshes are scheduled if events arrive during debounce window
2026-01-24 12:20:49 +00:00
GitHub Copilot 73c520b0c6 Bump version to 0.2.2 2026-01-24 12:16:28 +00:00
GitHub Copilot 92651e236b Fix sparkline slug-to-service mapping initialization
- Initialize _slug_to_service as instance variable in __init__
- Add logging to help debug empty sparklines
- Log the mapping at startup
2026-01-24 12:13:38 +00:00
GitHub Copilot 231bf69b3d Bump version to 0.2.1 2026-01-24 11:59:40 +00:00
GitHub Copilot 5536c4ac51 Invalidate screenshot cache on terminal resize
When terminal resizes, old screenshot content is stale until
the app (tmux etc) re-renders at new dimensions. Clear cache
to force re-capture after resize.
2026-01-24 11:59:23 +00:00
GitHub Copilot 02d941af6a Fix sparkline service name mapping
- Pass service names (not slugs) to DockerStatsCollector
- Create slug->name mapping for sparkline lookups
- Stats are stored by service name, looked up by slug
- Add debug logging when no containers found
2026-01-24 11:57:16 +00:00
GitHub Copilot 901a2e4cfa Release v0.2.0
New features:
- CPU sparklines on dashboard showing 30-minute container history
- Real-time screenshot updates via Server-Sent Events
- Auto-focus terminals on page load
- Tab reuse when clicking dashboard tiles
- pyte-based screenshot rendering with proper ANSI interpretation
- Dirty tracking for efficient screenshot caching

Documentation:
- Updated README with new features and API endpoints
- Added Dashboard Features section
- Documented compose mode CPU sparklines
- Added API endpoints table

Technical improvements:
- Docker stats collection via Unix socket (no new deps)
- SSE endpoint for activity notifications
- Proper pyte-to-Rich color mapping
- Terminal lifecycle race condition fixes
2026-01-24 11:53:44 +00:00
GitHub Copilot be2f18c8ba Auto-focus terminal on page load
- Try to focus xterm textarea after initialization
- Retry with delays (100ms, 500ms, 1s) as terminal loads async
- Re-focus when window regains focus (tab switching)
2026-01-24 11:45:07 +00:00
GitHub Copilot bd477c1b3c Use SSE for real-time screenshot updates
- New /events SSE endpoint pushes activity notifications to browsers
- Dashboard subscribes to SSE stream instead of polling
- Screenshots refresh instantly when terminal activity occurs
- Sparklines still poll every 30s (appropriate for 30min history)
- SSE includes keepalive every 30s and auto-reconnect on error
- Removes inefficient 5s polling; updates only on actual changes
2026-01-24 11:44:29 +00:00
GitHub Copilot 1ba4ce2a34 Adjust sparkline and screenshot timing
Sparklines:
- Poll interval: 2s -> 10s
- History size: 30 -> 180 readings
- Now shows 30 minutes of CPU history

Screenshots:
- Dashboard refresh interval: 15s -> 5s
- Combined with dirty tracking, updates on activity with 5s cap
2026-01-24 11:41:01 +00:00
GitHub Copilot 34aee378d9 Bump version to 0.1.18 2026-01-24 11:34:27 +00:00
GitHub Copilot 3e29f925e2 Rename dashboard title to 'Session Dashboard' 2026-01-24 11:34:02 +00:00
GitHub Copilot 1f51d878c8 Add CPU sparkline to dashboard in compose mode
- New docker_stats.py module reads container stats from Docker socket
  using only asyncio + stdlib (no new dependencies)
- Calculates CPU % from delta of cpu_usage and system_cpu_usage
- Maintains ring buffer of last 30 CPU readings per container
- render_sparkline_svg() generates mini SVG chart from history
- DockerStatsCollector polls containers every 2 seconds
- New /cpu-sparkline.svg endpoint serves sparkline for a container
- Dashboard shows sparkline in tile header next to container name
- Only active in compose mode (--compose-manifest flag)
- Graceful degradation if Docker socket unavailable

Bump version to 0.1.17
2026-01-24 11:33:27 +00:00
GitHub Copilot ff8f5efabd Optimize screenshot updates using pyte dirty tracking
- get_screen_state() now returns has_changes flag indicating if screen changed
- pyte's dirty set tracks which rows have been modified since last read
- Screenshot handler returns cached SVG immediately when no changes detected
- Removed _screenshot_last_rendered_activity tracking (replaced by dirty flag)
- Added test for dirty flag behavior

Bump version to 0.1.16
2026-01-24 11:27:33 +00:00
GitHub Copilot 8ae3f77f23 fix: reuse same browser tab when clicking dashboard tiles
Use tile slug as window name in window.open() so clicking the same
tile twice focuses the existing tab instead of opening a new one.

Changed: window.open(url, '_blank') -> window.open(url, 'webterm-{slug}')
2026-01-24 11:23:43 +00:00
GitHub Copilot 0cae07cba6 fix: complete pyte-to-Rich color mapping
Audited all color names from both libraries and added complete mappings:

pyte standard ANSI:
- brown -> yellow (pyte uses 'brown' for ANSI color 33)

pyte bright/AIXTERM colors:
- brightblack -> bright_black
- brightred -> bright_red
- brightgreen -> bright_green
- brightbrown -> bright_yellow
- brightblue -> bright_blue
- brightmagenta -> bright_magenta
- bfightmagenta -> bright_magenta (typo in pyte's BG_AIXTERM)
- brightcyan -> bright_cyan
- brightwhite -> bright_white

Also handles hex colors from 256-color and truecolor modes
by adding '#' prefix (e.g., 'ff8700' -> '#ff8700').

Bumps version to 0.1.15.
2026-01-24 11:23:01 +00:00
GitHub Copilot 0d53952ff7 fix: handle all pyte color formats for Rich compatibility
- Add 'brown' -> 'yellow' mapping (pyte uses 'brown' for ANSI yellow)
- Add helper function to convert hex colors (pyte outputs 'ff8700',
  Rich needs '#ff8700')
- Handles 256-color and truecolor (24-bit) ANSI codes properly

Bumps version to 0.1.14.
2026-01-24 11:21:09 +00:00
GitHub Copilot 4566d03aa5 chore: bump version to 0.1.13
Changes since 0.1.12:
- Fix pyte color name mapping for Rich compatibility
- Use session's actual screen state for screenshots
- Use requested dimensions when creating screenshot sessions
- Fix terminal lifecycle race conditions
2026-01-24 11:17:55 +00:00
GitHub Copilot 4d3a13f6ef fix: resolve terminal lifecycle race conditions
1. Lock pyte screen initialization in open() to prevent races with
   concurrent _update_screen() calls

2. Reorder session registration: call open() BEFORE adding to
   sessions/routes dicts, so sessions are fully initialized before
   other code can access them

3. Add clarifying comment that PTY resize completes before pyte resize

These fixes prevent dimension mismatches between PTY and pyte screen
that could cause content wrapping in screenshots.
2026-01-24 11:17:18 +00:00
GitHub Copilot db22fd9357 fix: use requested dimensions when creating screenshot sessions
When creating a new session for screenshot:
1. Use width/height query params instead of hardcoded DISCONNECT_RESIZE
2. Add a small delay (0.5s) after creating session to allow initial output

This ensures new sessions are created with the correct dimensions
matching what the screenshot expects.
2026-01-24 11:15:01 +00:00
GitHub Copilot e85213315e fix: use session's actual screen state for screenshots
The screenshot was creating a new pyte screen with arbitrary dimensions
from query params, but the replay buffer contains ANSI sequences meant
for the session's actual terminal size. This mismatch caused wrapping.

Now we use get_screen_state() which returns the actual screen buffer
from the terminal session's pyte screen, with the correct dimensions.
This ensures the screenshot matches exactly what the terminal rendered.
2026-01-24 11:14:12 +00:00
GitHub Copilot 3e433f5af5 fix: map pyte color names to Rich-compatible names
Pyte uses 'brightblack', 'brightred', etc. but Rich expects
'bright_black', 'bright_red' with underscores. Added PYTE_TO_RICH_COLOR
mapping to translate color names in screenshot rendering.
2026-01-24 11:11:35 +00:00
GitHub Copilot 4f8c7d88d5 fix: repair broken tests and remove unused dependencies
Test fixes:
- Fix app_session.py to use 'textual-webterm' package name (not 'textual-web')
- Fix CLI version test to not hardcode version number
- Fix static path test to not use removed Path._flavour attribute

Removed unused dependencies:
- xdg
- msgpack
- httpx

All 209 tests pass with 86% coverage.
2026-01-24 10:40:26 +00:00
GitHub Copilot f9196da9f8 fix: use pyte+Rich hybrid for colored SVG screenshots
Screenshots now properly preserve terminal colors:
1. Replay buffer provides raw ANSI data with color codes
2. pyte interprets escape sequences for accurate screen state
3. Rich renders the pyte buffer with colors to SVG

This gives us both accurate terminal state (no creeping/wrapping)
and proper color preservation in screenshots.

Bumps version to 0.1.12.
2026-01-24 10:37:54 +00:00
GitHub Copilot 8ef48bdb86 chore: remove completed TODO.md and REFACTORING.md 2026-01-24 10:34:34 +00:00
GitHub Copilot fb88a4f6b9 docs: update TODO.md with screenshot fix details 2026-01-24 10:33:41 +00:00
GitHub Copilot 894fb2eaaf fix: maintain pyte screen state in TerminalSession for accurate screenshots
Instead of trying to replay a truncated byte buffer through pyte, this
change maintains a pyte Screen object within TerminalSession that gets
updated as terminal data flows through. This provides accurate terminal
state for screenshots without issues from buffer truncation.

Key changes:
- Add pyte Screen and Stream to TerminalSession
- Update screen state as data arrives via _update_screen()
- Add get_screen_lines() to return current screen state
- Resize pyte screen when terminal size changes
- Update local_server to use get_screen_lines() directly
- Remove _apply_carriage_returns() workaround

This properly fixes the tmux status bar 'creeping up' issue by ensuring
the screenshot always reflects the actual terminal state.
2026-01-24 10:33:31 +00:00
GitHub Copilot a58c434eaf docs: mark completed items in TODO.md and REFACTORING.md 2026-01-24 10:27:25 +00:00
GitHub Copilot ea78ba7ff2 refactor: normalize logging to use %-style formatting
Convert f-string logging to lazy %-style interpolation throughout:
- session_manager.py
- cli.py
- terminal_session.py

This follows Python logging best practices for performance (lazy
evaluation) and consistency across the codebase.

Addresses REFACTORING.md item about normalizing logging style.
2026-01-24 10:26:53 +00:00