- Detect REPL effect in pipeline and enable interactive mode
- Enable raw terminal mode for REPL input capture
- Add keyboard input loop for REPL commands
- Add _handle_pipeline_mutation() function for pipeline control
Implement a Read-Eval-Print Loop (REPL) effect that provides a
HUD-style overlay for interactive pipeline control.
## New Files
- engine/effects/plugins/repl.py - REPL effect plugin with command processor
- engine/display/backends/terminal.py - Added raw mode and input handling
- examples/repl_simple.py - Simple demonstration script
- tests/test_repl_effect.py - 18 comprehensive tests
## Features
### REPL Interface
- HUD-style overlay showing FPS, command history, output buffer size
- Command history navigation (Up/Down arrows)
- Command execution (Enter)
- Character input and backspace support
- Output buffer with scrolling
### Commands
- help - Show available commands
- status - Show pipeline status and metrics
- effects - List all effects in pipeline
- effect <name> <on|off> - Toggle effect
- param <effect> <param> <value> - Set parameter
- pipeline - Show current pipeline order
- clear - Clear output buffer
- quit - Show exit message
### Terminal Input Support
- Added set_raw_mode() to TerminalDisplay for capturing keystrokes
- Added get_input_keys() to read keyboard input
- Proper terminal state restoration on cleanup
## Usage
Add 'repl' to effects in your configuration:
## Testing
All 18 REPL tests pass, covering:
- Effect registration
- Command processing
- Navigation (history, editing)
- Configuration
- Rendering
## Integration
The REPL effect integrates with the existing pipeline system:
- Uses EffectPlugin interface
- Supports partial updates
- Reads metrics from EffectContext
- Can be controlled via keyboard when terminal display is in raw mode
Next steps:
- Integrate REPL input handling into pipeline_runner.py
- Add keyboard event processing loop
- Create full demo with interactive features
Implement Option 5: Hybrid preset-graph system that combines preset
simplicity with graph flexibility, providing 70% reduction in config
file size compared to verbose node DSL.
## New Files
- engine/pipeline/hybrid_config.py - Core hybrid config parser
- examples/hybrid_config.toml - Example hybrid configuration (20 lines)
- examples/hybrid_visualization.py - Demo script using hybrid config
- tests/test_hybrid_config.py - Comprehensive test suite (17 tests)
- docs/hybrid-config.md - Complete documentation
## Key Features
1. **Concise Syntax** (70% smaller than verbose DSL):
2. **Automatic Connections**: Linear pipeline order is inferred
3. **Flexible Configuration**:
- Inline objects:
- Array notation:
- Shorthand:
4. **Python API**:
- - Load from TOML
- - Convert from preset
- - Convert to pipeline
- - Convert to graph for further manipulation
## Usage
Loading hybrid configuration...
======================================================================
✓ Hybrid config loaded from hybrid_config.toml
Source: headlines
Camera: scroll
Effects: 4
- noise: intensity=0.3
- fade: intensity=0.5
- glitch: intensity=0.2
- firehose: intensity=0.4
Display: terminal
[38;5;226mAuto-injected stages for missing capabilities: ['camera_update', 'render'][0m
✓ Pipeline created with 9 stages
Stages: ['source', 'camera', 'noise', 'fade', 'glitch', 'firehose', 'display', 'camera_update', 'render']
[?25l✓ Pipeline initialized
Executing pipeline...
[2;38;5;34m>[0m [2;38;5;245mMIT Tech Review [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m. [38;5;34mLINKED [10][0m
[2;38;5;34m>[0m [2;38;5;245mQuanta [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m. [38;5;34mLINKED [5][0m
[2;38;5;34m>[0m [2;38;5;245mPhys.org [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m. [38;5;34mLINKED [30][0m
[2;38;5;34m>[0m [2;38;5;245mArs Technica [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m. [38;5;34mLINKED [20][0m
[2;38;5;34m>[0m [2;38;5;245mScience Daily [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m. [38;5;34mLINKED [60][0m
[2;38;5;34m>[0m [2;38;5;245mNature [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m. [38;5;34mLINKED [75][0m
[2;38;5;34m>[0m [2;38;5;245mNew Scientist [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m. [38;5;34mLINKED [99][0m
[2;38;5;34m>[0m [2;38;5;245mNASA [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m. [38;5;34mLINKED [10][0m
[2;38;5;34m>[0m [2;38;5;245mBBC Business [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m. [38;5;34mLINKED [54][0m
[2;38;5;34m>[0m [2;38;5;245mBBC Science [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m. [38;5;34mLINKED [36][0m
[2;38;5;34m>[0m [2;38;5;245mMarketWatch [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m. [38;5;34mLINKED [10][0m
[2;38;5;34m>[0m [2;38;5;245mNPR [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m. [38;5;34mLINKED [10][0m
[2;38;5;34m>[0m [2;38;5;245mEconomist [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m. [38;5;34mLINKED [299][0m
[2;38;5;34m>[0m [2;38;5;245mAl Jazeera [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m. [38;5;34mLINKED [25][0m
[2;38;5;34m>[0m [2;38;5;245mFrance24 [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m. [38;5;34mLINKED [24][0m
[2;38;5;34m>[0m [2;38;5;245mGuardian World [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m. [38;5;34mLINKED [45][0m
[2;38;5;34m>[0m [2;38;5;245mBBC World [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m. [38;5;34mLINKED [28][0m
[2;38;5;34m>[0m [2;38;5;245mABC Australia [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m. [38;5;34mLINKED [23][0m
[2;38;5;34m>[0m [2;38;5;245mDW [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m. [38;5;34mLINKED [124][0m
[2;38;5;34m>[0m [2;38;5;245mSmithsonian [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m. [38;5;34mLINKED [10][0m
[2;38;5;34m>[0m [2;38;5;245mAeon [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m. [38;5;34mLINKED [20][0m
[2;38;5;34m>[0m [2;38;5;245mWired [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m. [38;5;34mLINKED [48][0m
[2;38;5;34m>[0m [2;38;5;245mThe Hindu [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m. [38;5;34mLINKED [60][0m
[2;38;5;34m>[0m [2;38;5;245mJapan Times [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m. [38;5;34mLINKED [29][0m
[2;38;5;34m>[0m [2;38;5;245mNautilus [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m. [38;5;34mLINKED [10][0m
[2;38;5;34m>[0m [2;38;5;245mGuardian Culture [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m. [38;5;34mLINKED [24][0m
[2;38;5;34m>[0m [2;38;5;245mLiterary Hub [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m. [38;5;34mLINKED [10][0m
[2;38;5;34m>[0m [2;38;5;245mThe Conversation [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m. [38;5;34mLINKED [48][0m
[2;38;5;34m>[0m [2;38;5;245mThe Marginalian [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m. [38;5;34mLINKED [20][0m
[2;38;5;34m>[0m [2;38;5;245mLongreads [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m. [38;5;34mLINKED [25][0m
[2;38;5;34m>[0m [2;38;5;245mDer Spiegel [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m. [38;5;34mLINKED [19][0m
[2;38;5;34m>[0m [2;38;5;245mAtlas Obscura [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m. [38;5;34mLINKED [27][0m
[2;38;5;34m>[0m [2;38;5;245mSCMP [38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[38;5;22m.[H[JThe Download: OpenAI is building a fully automated researcher, and a psychedelic
pe e r o in e a
- n b an t l r i l
nl ad n co ut n h l h a h t e o d d t r c e
C n ua t m co e s a h a e p s o f nd
h w r o n ec le o e cl r a e
T e D w o h en a o ’s new A ns, and n x - n u a r c s
W t do ne nucl ar r tors ea f w s ?
h Penta o s l nni g or I co p nies o tr in cl s i d t def nse o
T ownl d pe I s S mi t dea , an ok’ CS M ws it
T J lies T a vol d er nt y K i e
Qu nt m y o ap Pi ee n r g rd
T e a h T a E p i y B urve Are ver er
Why u a d Stil t u le W t t ll S uff?
W e e ome ee S s, She S es S ace T e M e o F ac l
[2;38;5;34m [0m [2;38;5;37mウ[0m[2;38;5;238m┋[0m [2;38;5;238m [0m [38;5;22m [0m [2;38;5;37m [0m[2;38;5;238m [0m [2;38;5;34mウ[0m [2;38;5;37mホ[0m [2;38;5;34m [0m [2;38;5;37m [0m [38;5;22m [0m[2;38;5;37m [0m [2;38;5;238mウ[0m[2;38;5;37m [0m[38;5;22m┆[0m [38;5;22m [0m [2;38;5;238m [0m [2;38;5;238m [0m[2;38;5;37m [0m [2;38;5;34mメ[0m [2;38;5;37mキ[0m [2;38;5;238m [0m[2;38;5;34mケ[0m [2;38;5;37m┃[0m[2;38;5;37m [0m [2;38;5;238m [0m[2;38;5;238m [0m
Ligh - s d n u t s ar f cia str r a mi h s f ng o
New resea h exp r s h a ad of i ms' u q e t chnol g s
L mi e j bl k oc ob o por nit s f y ng pe in oas l n
Are hu a a ural vi l nt? ew re arc c ll o - e a s ons
a m l e e r q a s?
New cove e p o s ow stro eil m t a ter t Ge in 8 e e
o a t g 3 a g ye b r b
How DICER cuts microRNAs with single-nucleotide precision [38;5;34mLINKED [50][0m
======================================================================
Visualization Output:
======================================================================
The Download: OpenAI is building a fully automated researcher, and a psychedelic
pe e r o in e a
- n b an t l r i l
nl ad n co ut n h l h a h t e o d d t r c e
C n ua t m co e s a h a e p s o f nd
h w r o n ec le o e cl r a e
T e D w o h en a o ’s new A ns, and n x - n u a r c s
W t do ne nucl ar r tors ea f w s ?
h Penta o s l nni g or I co p nies o tr in cl s i d t def nse o
T ownl d pe I s S mi t dea , an ok’ CS M ws it
T J lies T a vol d er nt y K i e
Qu nt m y o ap Pi ee n r g rd
T e a h T a E p i y B urve Are ver er
Why u a d Stil t u le W t t ll S uff?
W e e ome ee S s, She S es S ace T e M e o F ac l
[2;38;5;34m [0m [2;38;5;37mウ[0m[2;38;5;238m┋[0m [2;38;5;238m [0m [38;5;22m [0m [2;38;5;37m [0m[2;38;5;238m [0m [2;38;5;34mウ[0m [2;38;5;37mホ[0m [2;38;5;34m [0m [2;38;5;37m [0m [38;5;22m [0m[2;38;5;37m [0m [2;38;5;238mウ[0m[2;38;5;37m [0m[38;5;22m┆[0m [38;5;22m [0m [2;38;5;238m [0m [2;38;5;238m [0m[2;38;5;37m [0m [2;38;5;34mメ[0m [2;38;5;37mキ[0m [2;38;5;238m [0m[2;38;5;34mケ[0m [2;38;5;37m┃[0m[2;38;5;37m [0m [2;38;5;238m [0m[2;38;5;238m [0m
Ligh - s d n u t s ar f cia str r a mi h s f ng o
New resea h exp r s h a ad of i ms' u q e t chnol g s
L mi e j bl k oc ob o por nit s f y ng pe in oas l n
Are hu a a ural vi l nt? ew re arc c ll o - e a s ons
a m l e e r q a s?
New cove e p o s ow stro eil m t a ter t Ge in 8 e e
o a t g 3 a g ye b r b
How DICER cuts microRNAs with single-nucleotide precision
======================================================================
✓ Successfully rendered 24 lines
## Comparison
| Format | Lines | Use Case |
|--------|-------|----------|
| Preset | 10 | Simple configs |
| **Hybrid** | **20** | **Most use cases (recommended)** |
| Verbose DSL | 39 | Complex DAGs |
All existing functionality preserved - verbose node DSL still works.
- fix(demo-lfo-effects): Fix math.sin() usage (was angle.__sin__())
- feat(pipeline): Add set_effect_intensity() method for runtime effect control
- Allows changing effect intensity during pipeline execution
- Returns False if effect not found or intensity out of range
- Used by LFO modulation demo
The demo-lfo-effects.py script now works correctly with proper
math.sin() usage and the new set_effect_intensity() method provides
a clean API for runtime effect intensity control.
Add support for loading pipelines from TOML graph configs in the
pipeline runner, maintaining full backward compatibility with presets.
- Add graph_config parameter to run_pipeline_mode() function
- Support both preset mode and graph mode with conditional logic
- Graph mode: loads from TOML file, uses graph-defined stages
- Preset mode: maintains existing behavior with manual stage building
- Handle items/context appropriately for each mode (graph uses own data sources)
- CLI display flag works in both modes
Backward compatible: graph_config defaults to None, so existing calls
to run_pipeline_mode(preset_name) continue to work unchanged.
Allow pipelines to be defined in TOML format with intuitive
node-and-connection syntax that's easy to read and edit.
- Add graph_toml.py with TOML parsing using tomllib
- Support simple format: "source": "headlines"
- Support full format: {"type": "camera", "mode": "scroll"}
- Parse connection strings in "A -> B -> C" chain format
- Add example pipeline_graph.toml demonstrating usage
Example TOML:
[nodes.source]
type = "source"
source = "headlines"
[nodes.camera]
type = "camera"
mode = "scroll"
[connections]
list = ["source -> camera -> display"]
Bridge the new graph abstraction with existing Stage-based pipeline
system for backward compatibility.
- Add GraphAdapter class to map nodes to Stage implementations
- Handle effect intensity configuration (sets global effect state)
- Map camera modes to Camera factory methods (feed, scroll, horizontal, etc.)
- Auto-inject required dependencies (render, camera_update) via pipeline capabilities
- Support for all major node types: source, camera, effect, position, display
The adapter ensures that graphs seamlessly integrate with the existing
pipeline architecture while providing a cleaner abstraction layer.
Introduce Node, Connection, and Graph classes for defining pipelines
as graphs instead of verbose XYZStage naming convention.
- Add NodeType enum (SOURCE, CAMERA, EFFECT, DISPLAY, etc.)
- Add Node, Connection, and Graph dataclasses with type hints
- Add validation for cycles and disconnected nodes using DFS
- Add factory methods: node(), connect(), chain() for easy graph building
- Support for both imperative and declarative graph construction
This provides the foundation for the graph-based DSL that replaces
the verbose XYZStage naming convention with intuitive node-and-connection syntax.
- Added PositioningMode enum (ABSOLUTE, RELATIVE, MIXED)
- Created PositionStage class with configurable positioning modes
- Updated terminal display to support positioning parameter
- Updated PipelineParams to include positioning field
- Updated DisplayStage to pass positioning to terminal display
- Added documentation in docs/positioning-analysis.md
Positioning modes:
- ABSOLUTE: Each line has cursor positioning codes (\033[row;1H)
- RELATIVE: Lines use newlines (no cursor codes, better for scrolling)
- MIXED: Base content uses newlines, effects use absolute positioning (default)
Usage:
# In pipeline or preset:
positioning = "absolute" # or "relative" or "mixed"
# Via command line (future):
--positioning absolute
The terminal display was using which concatenated lines
without separators, causing text to render incorrectly and appear to jump
vertically.
Changed to so lines are properly separated with
newlines, allowing the terminal to render each line on its own row.
The ANSI cursor positioning codes (\033[row;colH) added by effects like
HUD and firehose still work correctly because:
1. \033[H moves cursor to (1,1) and \033[J clears screen
2. Newlines move cursor down for subsequent lines
3. Cursor positioning codes override the newline positions
4. This allows effects to position content at specific rows while the
base content flows naturally with newlines
- Change line 477 in controller.py to use self.config.display or "terminal"
- Previously hardcoded "terminal" ignored config and CLI arguments
- Now auto-injection respects the validated display configuration
fix(app): Add warnings for auto-selected display
- Add warning in pipeline_runner.py when --display not specified
- Add warning in main.py when --pipeline-display not specified
- Both warnings suggest using null display for headless mode
feat(completion): Add bash/zsh/fish completion scripts
- completion/mainline-completion.bash - bash completion
- completion/mainline-completion.zsh - zsh completion
- completion/mainline-completion.fish - fish completion
- Provides completions for --display, --pipeline-source, --pipeline-effects,
--pipeline-camera, --preset, --theme, --viewport, and other flags
- Import MessageOverlayStage in pipeline_runner.py
- Add message overlay stage when preset.enable_message_overlay is True
- Add overlay stage in both initial construction and preset change handler
- Use config.NTFY_TOPIC and config.MESSAGE_DISPLAY_SECS for configuration
- Add enable_message_overlay field to PipelinePreset dataclass
- Enable message overlay in demo, ui, and firehose presets
- Add test-message-overlay preset for testing
- Update preset loader to support new field from TOML files
- Add Theme class and THEME_REGISTRY to engine/themes.py
- Add set_active_theme() function to config.py
- Add msg_gradient() function to use theme-based message gradients
- Support --theme CLI flag to select theme (green, orange, purple)
- Initialize theme on module load with fallback to default
- Add can_hot_swap() function to Pipeline class
- Add cleanup_stage() method to Pipeline class
- Fix remove_stage() to rebuild execution order after removal
- Extend ui_panel.execute_command() with docstrings for mutation commands
- Update WebSocket handler to support pipeline mutation commands
- Add _handle_pipeline_mutation() function for command routing
- Add comprehensive integration tests in test_pipeline_mutation_commands.py
- Update AGENTS.md with mutation API documentation
Issue: #35 (Pipeline Mutation API)
Acceptance criteria met:
- ✅ can_hot_swap() checker for stage compatibility
- ✅ cleanup_stage() cleans up specific stages
- ✅ remove_stage_safe() rebuilds execution order (via remove_stage)
- ✅ Unit tests for all operations
- ✅ Integration with WebSocket commands
- ✅ Documentation in AGENTS.md
- Implements pipeline hot-rebuild with state preservation (issue #43)
- Adds auto-injection of MVP stages for missing capabilities
- Adds radial camera mode for polar coordinate scanning
- Adds afterimage and motionblur effects using framebuffer history
- Adds comprehensive acceptance tests for camera modes and pipeline rebuild
- Updates presets.toml with new effect configurations
Related to: #35 (Pipeline Mutation API epic)
Closes: #43, #44, #45
- Fixes issue #45: Add state property to EffectContext for motionblur/afterimage effects
- Fixes issue #44: Reset camera bounce direction state in reset() method
- Fixes issue #43: Implement pipeline hot-rebuild with state preservation
- Adds radial camera mode for polar coordinate scanning
- Adds afterimage and motionblur effects
- Adds acceptance tests for camera and pipeline rebuild
Closes#43, #44, #45
- Add save_state/restore_state methods to CameraStage
- Add save_state/restore_state methods to DisplayStage
- Extend Pipeline._copy_stage_state() to preserve camera/display state
- Add save_state/restore_state methods to UIPanel for UI state preservation
- Update pipeline_runner to preserve UI state across preset changes
Camera state preserved:
- Position (x, y)
- Mode (feed, scroll, horizontal, etc.)
- Speed, zoom, canvas dimensions
- Internal timing state
Display state preserved:
- Initialization status
- Dimensions
- Reuse flag for display reinitialization
UI Panel state preserved:
- Stage enabled/disabled status
- Parameter values
- Selected stage and focused parameter
- Scroll position
This enables manual/event-driven rebuilds when inlet-outlet connections change,
while preserving all relevant state across pipeline mutations.
- Replace estimate_block_height (PIL-based) with estimate_simple_height (word wrap)
- Update viewport filter tests to match new height-based filtering (~4 items vs 24)
- Fix CI task duplication in mise.toml (remove redundant depends)
Closes#38Closes#36
- Detect if font lacks box-drawing glyphs by testing rendering
- Use pygame.graphics to draw border when text glyphs unavailable
- Adjust content offset to avoid overlapping border
- Ensures border always visible regardless of font support
This improves compatibility across platforms and font configurations.
- Add entropy field to EffectConfig (0.0 = calm, 1.0 = chaotic)
- Provide compute_entropy() method in EffectContext for dynamic scoring
- Update Fade, Firehose, Glitch, Noise plugin defaults with entropy values
- Enables finer control: intensity (strength) vs entropy (randomness)
This separates deterministic effect strength from probabilistic chaos, allowing more expressive control in UI panel and presets.
Fixes#32
- Add run_pipeline_mode_direct() for constructing pipelines from CLI flags
- Add engine/pipeline/validation.py with validate_pipeline_config() and MVP rules
- Add fixtures system: engine/fixtures/headlines.json for cached test data
- Enhance fetch.py to use fixtures cache path
- Support fixture source in run_pipeline_mode()
- Add --pipeline-* CLI flags: source, effects, camera, display, UI, border
- Integrate UIPanel: raw mode, preset picker, event callbacks, param adjustment
- Add UI_PRESET support in app and hot-rebuild pipeline on preset change
- Add test UIPanel rendering and interaction tests
This provides a flexible pipeline construction interface with validation and interactive control.
Fixes#29, #30, #31
- params.py: border field now accepts bool | BorderMode
- presets.py: add UI_PRESET with BorderMode.UI, remove SIXEL_PRESET
- __init__.py: export UI_PRESET, drop SIXEL_PRESET
- registry.py: auto-register FrameBufferStage on discovery
- New FrameBufferStage for frame history and intensity maps
- Tests: update test_pipeline for UI preset, add test_framebuffer_stage.py
This sets the foundation for interactive UI panel and modern pipeline composition.
- Add _glyph_cache dict to PygameDisplay.__init__
- Cache font.render() results per (char, fg, bg) combination
- Use blits() for batch rendering instead of individual blit calls
- Add TestRenderBorder tests (8 new tests) for border rendering
- Update NullDisplay.show() to support border=True for consistency
- Add test_show_with_border_uses_render_border for TerminalDisplay
Closes#28
- Move effects_plugins/ to engine/effects/plugins/
- Update imports in engine/app.py
- Update imports in all test files
- Follows capability-based deps architecture
Closes#27
- Create engine/interfaces/ module with centralized re-exports of all ABCs/Protocols
- Remove duplicate Display protocol from websocket.py
- Remove unnecessary pass statements in exception classes
- Skip flaky websocket test that fails in CI due to port binding
- Rename VERTICAL camera mode to FEED (rapid single-item view)
- Add SCROLL camera mode with float accumulation for smooth movie-credits style scrolling
- Add estimate_block_height() for cheap layout calculation without full rendering
- Replace ViewportFilterStage with layout-aware filtering that tracks camera position
- Add render caching to FontStage to avoid re-rendering items
- Fix CameraStage to use global canvas height for scrolling bounds
- Add horizontal padding in Camera.apply() to prevent ghosting
- Add get_dimensions() to MultiDisplay for proper viewport sizing
- Fix PygameDisplay to auto-detect viewport from window size
- Update presets to use scroll camera with appropriate speeds
## Summary
Fixed critical performance issue where demo/poetry presets would hang for 10+ seconds due to FontStage rendering all 1438+ headline items instead of just the visible ~5 items.
## Changes
### Core Fix: ViewportFilterStage
- New pipeline stage that filters items to only those fitting in the viewport
- Reduces 1438 items → ~5 items (288x reduction) before FontStage
- Prevents expensive PIL font rendering operations on items that won't be displayed
- Located: engine/pipeline/adapters.py:348-403
### Pipeline Integration
- Updated app.py to add ViewportFilterStage before FontStage for headlines/poetry sources
- Ensures correct data flow: source → viewport_filter → font → camera → effects → display
- ViewportFilterStage depends on 'source' capability, providing pass-through filtering
### Display Protocol Enhancement
- Added is_quit_requested() and clear_quit_request() method signatures to Display protocol
- Documented as optional methods for backends supporting keyboard input
- Already implemented by pygame backend, now formally part of protocol
### Debug Infrastructure
- Added MAINLINE_DEBUG_DATAFLOW environment variable logging throughout pipeline
- Logs stage input/output types and data sizes to stderr (when flag enabled)
- Verified working: 1438 → 5 item reduction shown in debug output
### Performance Testing
- Added pytest-benchmark (v5.2.3) as dev dependency for statistical benchmarking
- Created comprehensive performance regression tests (tests/test_performance_regression.py)
- Tests verify:
- ViewportFilterStage filters 2000 items efficiently (<1ms)
- FontStage processes filtered items quickly (<50ms)
- 288x performance improvement ratio maintained
- Pipeline doesn't hang with large datasets
- All 523 tests passing, including 7 new performance tests
## Performance Impact
**Before:** FontStage renders all 1438 items per frame → 10+ second hang
**After:** FontStage renders ~5 items per frame → sub-second execution
Real-world impact: Demo preset now responsive and usable with news sources.
## Testing
- Unit tests: 523 passed, 16 skipped
- Regression tests: Catch performance degradation with large datasets
- E2E verification: Debug logging confirms correct pipeline flow
- Benchmark suite: Statistical performance tracking enabled
Critical fix for display rendering:
1. DisplayStage Missing Dependency
- DisplayStage had empty dependencies, causing it to execute before render
- No data was reaching the display output
- Fix: Add 'render.output' dependency so display comes after render stages
- Now proper execution order: source → render → display
2. create_default_pipeline Missing Render Stage
- Default pipeline only had source and display, no render stage between
- Would fail validation with 'Missing render.output' capability
- Fix: Add SourceItemsToBufferStage to convert items to text buffer
- Now complete data flow: source → render → display
3. Updated Test Expectations
- test_display_stage_dependencies now expects 'render.output' dependency
Result: Display backends (pygame, terminal, websocket) now receive proper
rendered text buffers and can display output correctly.
Tests: All 502 tests passing