Files
codex/codex-rs/tui/src/streaming/controller.rs
2026-04-28 13:19:57 -03:00

1726 lines
61 KiB
Rust

//! Two-region streaming controllers for agent messages and proposed plans.
//!
//! Each stream partitions markdown source into a *stable prefix* (committed to
//! scrollback via the animation queue in `StreamState`) and a *mutable tail*
//! (displayed in the active-cell slot as a transient stream-tail cell).
//!
//! `StreamCore` owns the shared bookkeeping: source accumulation, source
//! partitioning, commit-animation queue management, and terminal resize
//! handling. `StreamController` and `PlanStreamController` are thin
//! wrappers that add only their `emit()` styling and finalize return types.
//!
//! ## Table holdback
//!
//! Table rendering is inherently non-incremental: adding a new row can change
//! every column's width and reshape all prior rows. The holdback mechanism
//! (`table_holdback_state`) detects pipe-table patterns (header + delimiter
//! pair) in the accumulated source and keeps content from the table header
//! onward as mutable tail until the stream finalizes. Holdback is enabled for
//! agent and proposed-plan streams. Lines in `Outside` and `Markdown` fence
//! contexts are scanned; lines inside non-markdown fences are skipped.
//!
//! ## Resize handling
//!
//! On terminal width change, `StreamCore::set_width` re-renders the stable
//! source prefix at the new width and rebuilds the queued stable region from
//! the current emitted line count. Finalized content is canonicalized by
//! transcript consolidation into source-backed markdown cells.
//!
//! ## Invariants
//!
//! - `emitted_stable_len <= enqueued_stable_len <= rendered_stable_lines.len()`.
//! - `raw_source` is append-only until `reset()`; never modified mid-stream.
//! - `stable_source_len` marks the exact source boundary between stable source
//! and mutable tail source.
//! - During confirmed table streaming, only source from the table header onward
//! stays mutable; pre-table source may remain stable.
use crate::history_cell::HistoryCell;
use crate::history_cell::{self};
use crate::markdown::append_markdown_agent_with_cwd;
use crate::render::line_utils::prefix_lines;
use crate::style::proposed_plan_style;
use ratatui::prelude::Stylize;
use ratatui::text::Line;
use std::path::Path;
use std::path::PathBuf;
use std::time::Duration;
use std::time::Instant;
use super::StreamState;
use super::table_holdback::TableHoldbackScanner;
use super::table_holdback::TableHoldbackState;
#[cfg(test)]
use super::table_holdback::table_holdback_state;
// ---------------------------------------------------------------------------
// StreamCore — shared bookkeeping for both stream controllers
// ---------------------------------------------------------------------------
/// Shared state and logic for the two-region streaming model.
///
/// Both [`StreamController`] (agent messages) and [`PlanStreamController`]
/// (proposed plans) delegate their core bookkeeping here: source
/// accumulation, re-rendering, stable/tail partitioning, commit-animation
/// queue management, and terminal resize handling.
///
/// The wrapping controllers add only their own `emit()` styling and
/// finalize return types.
struct StreamCore {
state: StreamState,
/// Current rendering width (columns available for markdown content).
width: Option<usize>,
/// Accumulated raw markdown source for the current stream.
raw_source: String,
/// Byte length of the stable prefix in `raw_source`.
stable_source_len: usize,
/// Re-render of only the stable source prefix at `width`.
rendered_stable_lines: Vec<Line<'static>>,
/// Lines enqueued into the commit-animation queue.
enqueued_stable_len: usize,
/// Lines actually emitted to scrollback.
emitted_stable_len: usize,
/// Session cwd used to keep local file-link display stable during stream re-renders.
cwd: PathBuf,
/// Incremental holdback scanner state for append-only source updates.
holdback_scanner: TableHoldbackScanner,
}
impl StreamCore {
fn new(width: Option<usize>, cwd: &Path) -> Self {
Self {
state: StreamState::new(width, cwd),
width,
raw_source: String::with_capacity(1024),
stable_source_len: 0,
rendered_stable_lines: Vec::with_capacity(64),
enqueued_stable_len: 0,
emitted_stable_len: 0,
cwd: cwd.to_path_buf(),
holdback_scanner: TableHoldbackScanner::new(),
}
}
/// Push a delta; if it contains a newline, commit completed lines and enqueue newly-stable
/// lines. Returns `true` if new lines were enqueued.
fn push_delta(&mut self, delta: &str) -> bool {
if !delta.is_empty() {
self.state.has_seen_delta = true;
}
self.state.collector.push_delta(delta);
let mut enqueued = false;
if delta.contains('\n')
&& let Some(committed_source) = self.state.collector.commit_complete_source()
{
self.raw_source.push_str(&committed_source);
self.holdback_scanner.push_source_chunk(&committed_source);
self.update_stable_source_len();
self.recompute_stable_render();
enqueued = self.sync_stable_queue();
}
enqueued
}
/// Drain the collector, re-render, and return lines not yet emitted.
/// Does NOT reset state - the caller must call `reset()` afterward.
fn finalize_remaining(&mut self) -> Vec<Line<'static>> {
let remainder_source = self.state.collector.finalize_and_drain_source();
if !remainder_source.is_empty() {
self.raw_source.push_str(&remainder_source);
self.holdback_scanner.push_source_chunk(&remainder_source);
}
let mut rendered = Vec::new();
append_markdown_agent_with_cwd(
&self.raw_source,
self.width,
Some(self.cwd.as_path()),
&mut rendered,
);
if self.emitted_stable_len >= rendered.len() {
Vec::new()
} else {
rendered[self.emitted_stable_len..].to_vec()
}
}
/// Step animation: dequeue one line, update the emitted count.
fn tick(&mut self) -> Vec<Line<'static>> {
let step = self.state.step();
self.emitted_stable_len += step.len();
step
}
/// Batch drain: dequeue up to `max_lines`, update the emitted count.
fn tick_batch(&mut self, max_lines: usize) -> Vec<Line<'static>> {
if max_lines == 0 {
return Vec::new();
}
let step = self.state.drain_n(max_lines);
if step.is_empty() {
return step;
}
self.emitted_stable_len += step.len();
step
}
// Trivial StreamCore accessors inlined — called on every animation tick
// and render frame during active streaming.
#[inline]
fn is_idle(&self) -> bool {
self.state.is_idle()
}
#[inline]
fn queued_lines(&self) -> usize {
self.state.queued_len()
}
#[inline]
fn oldest_queued_age(&self, now: Instant) -> Option<Duration> {
self.state.oldest_queued_age(now)
}
/// Lines that belong to the mutable source tail, not yet queued for stable commit.
fn current_tail_lines(&self) -> Vec<Line<'static>> {
if !self.has_tail() {
return Vec::new();
}
let mut rendered_full_source = Vec::new();
append_markdown_agent_with_cwd(
&self.raw_source,
self.width,
Some(self.cwd.as_path()),
&mut rendered_full_source,
);
let stable_len = self
.rendered_stable_lines
.len()
.min(rendered_full_source.len());
rendered_full_source[stable_len..].to_vec()
}
#[inline]
fn has_tail(&self) -> bool {
self.stable_source_len < self.raw_source.len()
}
/// Update rendering width and rebuild queued stable lines for the new layout.
///
/// Re-renders once at the new width and rebuilds queue state from the
/// current emitted line count.
fn set_width(&mut self, width: Option<usize>) {
if self.width == width {
return;
}
let had_pending_queue = self.state.queued_len() > 0;
let had_live_tail = self.has_tail();
self.width = width;
self.state.collector.set_width(width);
if self.raw_source.is_empty() {
return;
}
self.recompute_stable_render();
self.emitted_stable_len = self
.emitted_stable_len
.min(self.rendered_stable_lines.len());
if had_pending_queue
&& self.emitted_stable_len == self.rendered_stable_lines.len()
&& self.emitted_stable_len > 0
{
// If wrapped remainder compresses into fewer lines at the new width,
// keep at least one line un-emitted so pre-resize pending content is
// not skipped permanently.
self.emitted_stable_len -= 1;
}
self.state.clear_queue();
if self.emitted_stable_len > 0 && !had_pending_queue && !had_live_tail {
// Avoid replaying already-emitted content after resize when no
// stable lines were waiting in the queue and there was no mutable
// tail to preserve.
self.enqueued_stable_len = self.rendered_stable_lines.len();
return;
}
self.rebuild_stable_queue_from_render();
}
/// Clear all accumulated state for current stream.
fn reset(&mut self) {
self.state.clear();
self.raw_source.clear();
self.stable_source_len = 0;
self.rendered_stable_lines.clear();
self.enqueued_stable_len = 0;
self.emitted_stable_len = 0;
self.holdback_scanner.reset();
}
/// Update the direct source boundary between stable output and mutable tail.
fn update_stable_source_len(&mut self) {
self.stable_source_len = match self.holdback_scanner.state() {
TableHoldbackState::Confirmed { table_start }
| TableHoldbackState::PendingHeader {
header_start: table_start,
} => table_start.min(self.raw_source.len()),
TableHoldbackState::None => self.raw_source.len(),
};
}
/// Re-render the stable source prefix at the current width.
fn recompute_stable_render(&mut self) {
self.rendered_stable_lines.clear();
append_markdown_agent_with_cwd(
&self.raw_source[..self.stable_source_len],
self.width,
Some(self.cwd.as_path()),
&mut self.rendered_stable_lines,
);
}
/// Compute how many rendered lines should be in the stable region.
fn compute_target_stable_len(&self) -> usize {
self.rendered_stable_lines
.len()
.max(self.emitted_stable_len)
}
/// Advance `enqueued_stable_len` toward the target stable boundary and enqueue any
/// newly-stable lines. Returns `true` if new lines were enqueued.
fn sync_stable_queue(&mut self) -> bool {
let target_stable_len = self.compute_target_stable_len();
// A structural rewrite moved the stable boundary backward into enqueue-but-unemitted
// lines. Rebuild queue from the latest snapshot.
if target_stable_len < self.enqueued_stable_len {
self.state.clear_queue();
if self.emitted_stable_len < target_stable_len {
self.state.enqueue(
self.rendered_stable_lines[self.emitted_stable_len..target_stable_len].to_vec(),
);
}
self.enqueued_stable_len = target_stable_len;
return self.state.queued_len() > 0;
}
if target_stable_len == self.enqueued_stable_len {
return false;
}
self.state.enqueue(
self.rendered_stable_lines[self.enqueued_stable_len..target_stable_len].to_vec(),
);
self.enqueued_stable_len = target_stable_len;
true
}
/// Rebuild the stable queue from the current render snapshot. Used after `set_width()` where
/// the old queue is stale.
fn rebuild_stable_queue_from_render(&mut self) {
let target_stable_len = self.compute_target_stable_len();
self.state.clear_queue();
if self.emitted_stable_len < target_stable_len {
self.state.enqueue(
self.rendered_stable_lines[self.emitted_stable_len..target_stable_len].to_vec(),
);
}
self.enqueued_stable_len = target_stable_len;
}
}
/// Controller for streaming agent message content with table-aware holdback.
///
/// Wraps [`StreamCore`] and adds `AgentMessageCell` emission styling.
pub(crate) struct StreamController {
core: StreamCore,
header_emitted: bool,
}
impl StreamController {
/// Create a controller whose markdown renderer shortens local file links relative to `cwd`.
///
/// The controller snapshots the path into stream state so later commit ticks and finalization
/// render against the same session cwd that was active when streaming started.
pub(crate) fn new(width: Option<usize>, cwd: &Path) -> Self {
Self {
core: StreamCore::new(width, cwd),
header_emitted: false,
}
}
pub(crate) fn push(&mut self, delta: &str) -> bool {
self.core.push_delta(delta)
}
/// Finalize the active stream. Returns the final cell (if any remaining lines) and the raw
/// markdown source for consolidation.
pub(crate) fn finalize(&mut self) -> (Option<Box<dyn HistoryCell>>, Option<String>) {
let remaining = self.core.finalize_remaining();
if self.core.raw_source.is_empty() {
self.core.reset();
return (None, None);
}
// Move ownership — source is consumed before reset() clears it.
let source = std::mem::take(&mut self.core.raw_source);
let out = self.emit(remaining);
self.core.reset();
(out, Some(source))
}
pub(crate) fn on_commit_tick(&mut self) -> (Option<Box<dyn HistoryCell>>, bool) {
let step = self.core.tick();
(self.emit(step), self.core.is_idle())
}
pub(crate) fn on_commit_tick_batch(
&mut self,
max_lines: usize,
) -> (Option<Box<dyn HistoryCell>>, bool) {
let step = self.core.tick_batch(max_lines);
(self.emit(step), self.core.is_idle())
}
// Thin StreamController accessors inlined — one-liner delegates called
// on every render frame and animation tick.
#[inline]
pub(crate) fn queued_lines(&self) -> usize {
self.core.queued_lines()
}
pub(crate) fn oldest_queued_age(&self, now: Instant) -> Option<Duration> {
self.core.oldest_queued_age(now)
}
#[inline]
pub(crate) fn current_tail_lines(&self) -> Vec<Line<'static>> {
self.core.current_tail_lines()
}
#[inline]
pub(crate) fn tail_starts_stream(&self) -> bool {
!self.header_emitted && self.core.enqueued_stable_len == 0
}
#[inline]
pub(crate) fn has_live_tail(&self) -> bool {
self.core.has_tail()
}
pub(crate) fn clear_queue(&mut self) {
self.core.state.clear_queue();
self.core.enqueued_stable_len = self.core.emitted_stable_len;
}
pub(crate) fn set_width(&mut self, width: Option<usize>) {
self.core.set_width(width);
}
fn emit(&mut self, lines: Vec<Line<'static>>) -> Option<Box<dyn HistoryCell>> {
if lines.is_empty() {
return None;
}
Some(Box::new(history_cell::AgentMessageCell::new(lines, {
let header_emitted = self.header_emitted;
self.header_emitted = true;
!header_emitted
})))
}
}
// ---------------------------------------------------------------------------
// PlanStreamController — proposed plan streams
// ---------------------------------------------------------------------------
/// Controller that streams proposed plan markdown into a styled plan block.
///
/// Wraps [`StreamCore`] and adds plan-specific header, indentation, and
/// background styling.
pub(crate) struct PlanStreamController {
core: StreamCore,
header_emitted: bool,
top_padding_emitted: bool,
}
impl PlanStreamController {
/// Create a plan-stream controller whose markdown renderer shortens local file links relative
/// to `cwd`.
///
/// The controller snapshots the path into stream state so later commit ticks and finalization
/// render against the same session cwd that was active when streaming started.
pub(crate) fn new(width: Option<usize>, cwd: &Path) -> Self {
Self {
core: StreamCore::new(width, cwd),
header_emitted: false,
top_padding_emitted: false,
}
}
pub(crate) fn push(&mut self, delta: &str) -> bool {
self.core.push_delta(delta)
}
/// Finalize the active stream. Returns the final cell (if any remaining
/// lines) plus raw markdown source for consolidation.
pub(crate) fn finalize(&mut self) -> (Option<Box<dyn HistoryCell>>, Option<String>) {
let remaining = self.core.finalize_remaining();
if self.core.raw_source.is_empty() {
self.core.reset();
return (None, None);
}
// Move ownership — source is consumed before reset() clears it.
let source = std::mem::take(&mut self.core.raw_source);
let out = self.emit(remaining, /*include_bottom_padding*/ true);
self.core.reset();
(out, Some(source))
}
pub(crate) fn on_commit_tick(&mut self) -> (Option<Box<dyn HistoryCell>>, bool) {
let step = self.core.tick();
(
self.emit(step, /*include_bottom_padding*/ false),
self.core.is_idle(),
)
}
pub(crate) fn on_commit_tick_batch(
&mut self,
max_lines: usize,
) -> (Option<Box<dyn HistoryCell>>, bool) {
let step = self.core.tick_batch(max_lines);
(
self.emit(step, /*include_bottom_padding*/ false),
self.core.is_idle(),
)
}
#[inline]
pub(crate) fn queued_lines(&self) -> usize {
self.core.queued_lines()
}
#[inline]
pub(crate) fn has_live_tail(&self) -> bool {
self.core.has_tail()
}
#[inline]
pub(crate) fn current_tail_lines(&self) -> Vec<Line<'static>> {
self.core.current_tail_lines()
}
#[inline]
pub(crate) fn tail_starts_stream(&self) -> bool {
!self.header_emitted && self.core.enqueued_stable_len == 0
}
pub(crate) fn current_tail_display_lines(&self) -> Vec<Line<'static>> {
let lines = self.current_tail_lines();
if lines.is_empty() {
return Vec::new();
}
self.render_display_lines(lines, /*include_bottom_padding*/ false)
}
pub(crate) fn oldest_queued_age(&self, now: Instant) -> Option<Duration> {
self.core.oldest_queued_age(now)
}
pub(crate) fn clear_queue(&mut self) {
self.core.state.clear_queue();
self.core.enqueued_stable_len = self.core.emitted_stable_len;
}
pub(crate) fn set_width(&mut self, width: Option<usize>) {
self.core.set_width(width);
}
fn emit(
&mut self,
lines: Vec<Line<'static>>,
include_bottom_padding: bool,
) -> Option<Box<dyn HistoryCell>> {
if lines.is_empty() && !include_bottom_padding {
return None;
}
let is_stream_continuation = self.header_emitted;
let out_lines = self.render_display_lines(lines, include_bottom_padding);
self.header_emitted = true;
self.top_padding_emitted = true;
Some(Box::new(history_cell::new_proposed_plan_stream(
out_lines,
is_stream_continuation,
)))
}
fn render_display_lines(
&self,
lines: Vec<Line<'static>>,
include_bottom_padding: bool,
) -> Vec<Line<'static>> {
let mut out_lines: Vec<Line<'static>> = Vec::with_capacity(4);
if !self.header_emitted {
out_lines.push(vec!["".dim(), "Proposed Plan".bold()].into());
out_lines.push(Line::from(" "));
}
let mut plan_lines: Vec<Line<'static>> = Vec::with_capacity(4);
if !self.top_padding_emitted {
plan_lines.push(Line::from(" "));
}
plan_lines.extend(lines);
if include_bottom_padding {
plan_lines.push(Line::from(" "));
}
let plan_style = proposed_plan_style();
let plan_lines = prefix_lines(plan_lines, " ".into(), " ".into())
.into_iter()
.map(|line| line.style(plan_style))
.collect::<Vec<_>>();
out_lines.extend(plan_lines);
out_lines
}
}
#[cfg(test)]
mod tests {
use super::*;
use pretty_assertions::assert_eq;
use std::path::PathBuf;
fn test_cwd() -> PathBuf {
// These tests only need a stable absolute cwd; using temp_dir() avoids baking Unix- or
// Windows-specific root semantics into the fixtures.
std::env::temp_dir()
}
fn stream_controller(width: Option<usize>) -> StreamController {
let cwd = test_cwd();
StreamController::new(width, &cwd)
}
fn plan_stream_controller(width: Option<usize>) -> PlanStreamController {
let cwd = test_cwd();
PlanStreamController::new(width, &cwd)
}
fn lines_to_plain_strings(lines: &[ratatui::text::Line<'_>]) -> Vec<String> {
lines
.iter()
.map(|l| {
l.spans
.iter()
.map(|s| s.content.clone())
.collect::<Vec<_>>()
.join("")
})
.collect()
}
fn collect_streamed_lines(deltas: &[&str], width: Option<usize>) -> Vec<String> {
let mut ctrl = stream_controller(width);
let mut lines = Vec::new();
for d in deltas {
ctrl.push(d);
while let (Some(cell), idle) = ctrl.on_commit_tick() {
lines.extend(cell.transcript_lines(u16::MAX));
if idle {
break;
}
}
}
if let (Some(cell), _source) = ctrl.finalize() {
lines.extend(cell.transcript_lines(u16::MAX));
}
lines_to_plain_strings(&lines)
.into_iter()
.map(|s| s.chars().skip(2).collect::<String>())
.collect()
}
fn collect_plan_streamed_lines(deltas: &[&str], width: Option<usize>) -> Vec<String> {
let mut ctrl = plan_stream_controller(width);
let mut lines = Vec::new();
for d in deltas {
ctrl.push(d);
while let (Some(cell), idle) = ctrl.on_commit_tick() {
lines.extend(cell.transcript_lines(u16::MAX));
if idle {
break;
}
}
}
if let (Some(cell), _source) = ctrl.finalize() {
lines.extend(cell.transcript_lines(u16::MAX));
}
lines_to_plain_strings(&lines)
}
#[test]
fn controller_set_width_rebuilds_queued_lines() {
let mut ctrl = stream_controller(Some(120));
let delta = "This is a long line that should wrap into multiple rows when resized.\n";
assert!(ctrl.push(delta));
assert_eq!(ctrl.queued_lines(), 1);
ctrl.set_width(Some(24));
let (cell, idle) = ctrl.on_commit_tick_batch(usize::MAX);
let rendered = lines_to_plain_strings(
&cell
.expect("expected resized queued lines")
.transcript_lines(u16::MAX),
);
assert!(idle);
assert!(
rendered.len() > 1,
"expected resized content to occupy multiple lines, got {rendered:?}",
);
}
#[test]
fn controller_set_width_no_duplicate_after_emit() {
let mut ctrl = stream_controller(Some(120));
let line =
"This is a long line that definitely wraps when the terminal shrinks to 24 columns.\n";
ctrl.push(line);
let (cell, _) = ctrl.on_commit_tick_batch(usize::MAX);
assert!(cell.is_some(), "expected emitted cell");
assert_eq!(ctrl.queued_lines(), 0);
ctrl.set_width(Some(24));
assert_eq!(
ctrl.queued_lines(),
0,
"already-emitted content must not be re-queued after resize",
);
}
#[test]
fn controller_tick_batch_zero_is_noop() {
let mut ctrl = stream_controller(Some(80));
assert!(ctrl.push("line one\n"));
assert_eq!(ctrl.queued_lines(), 1);
let (cell, idle) = ctrl.on_commit_tick_batch(/*max_lines*/ 0);
assert!(cell.is_none(), "batch size 0 should not emit lines");
assert!(!idle, "batch size 0 should not drain queued lines");
assert_eq!(
ctrl.queued_lines(),
1,
"queue depth should remain unchanged"
);
}
#[test]
fn controller_has_live_tail_reflects_tail_presence() {
let mut ctrl = stream_controller(Some(80));
assert!(!ctrl.has_live_tail());
ctrl.core.raw_source = "tail line\n".to_string();
ctrl.core.stable_source_len = 0;
assert!(ctrl.has_live_tail());
ctrl.core.stable_source_len = ctrl.core.raw_source.len();
assert!(!ctrl.has_live_tail());
}
#[test]
fn plan_controller_has_live_tail_reflects_tail_presence() {
let mut ctrl = plan_stream_controller(Some(80));
assert!(!ctrl.has_live_tail());
ctrl.core.raw_source = "tail line\n".to_string();
ctrl.core.stable_source_len = 0;
assert!(ctrl.has_live_tail());
ctrl.core.stable_source_len = ctrl.core.raw_source.len();
assert!(!ctrl.has_live_tail());
}
#[test]
fn controller_live_tail_keeps_uncommitted_table_cell_newline_gated() {
let mut ctrl = stream_controller(Some(80));
ctrl.push("| A | B |\n");
ctrl.push("| --- | --- |\n");
ctrl.push("| partial");
let tail = lines_to_plain_strings(&ctrl.current_tail_lines()).join("\n");
assert!(
!tail.contains("partial"),
"expected live tail to remain newline-gated: {tail:?}",
);
}
#[test]
fn controller_live_tail_requires_table_holdback_state() {
let mut ctrl = stream_controller(Some(80));
ctrl.push("plain text without newline");
assert!(
ctrl.current_tail_lines().is_empty(),
"expected no live tail outside table holdback state",
);
assert!(!ctrl.has_live_tail());
}
#[test]
fn controller_live_tail_rerenders_table_tail_after_resize() {
let mut ctrl = stream_controller(Some(96));
ctrl.push("| # | Feature | Details | Link |\n");
ctrl.push("| --- | --- | --- | --- |\n");
ctrl.push(
"| 1 | RESIZE_REPRO_SENTINEL | long wrapped content that should be reflowed | https://example.com/resize |\n",
);
for width in [48, 104, 56] {
ctrl.set_width(Some(width));
let tail = lines_to_plain_strings(&ctrl.current_tail_lines());
let mut expected = Vec::new();
crate::markdown::append_markdown_agent(
&ctrl.core.raw_source,
Some(width),
&mut expected,
);
let expected = lines_to_plain_strings(&expected);
assert_eq!(
tail, expected,
"expected live table tail to be rerendered at width {width}",
);
}
}
#[test]
fn controller_set_width_partial_drain_no_lost_lines() {
let mut ctrl = stream_controller(Some(40));
ctrl.push("AAAA BBBB CCCC DDDD EEEE FFFF GGGG HHHH IIII JJJJ\n");
ctrl.push("second line\n");
let (cell, idle) = ctrl.on_commit_tick();
assert!(cell.is_some(), "expected 1 emitted line");
assert!(!idle, "queue should still have lines");
let remaining_before = ctrl.queued_lines();
assert!(remaining_before > 0, "should have queued lines left");
ctrl.set_width(Some(20));
let (cell, source) = ctrl.finalize();
let final_lines = cell
.map(|c| lines_to_plain_strings(&c.transcript_lines(u16::MAX)))
.unwrap_or_default();
assert!(
final_lines.iter().any(|l| l.contains("second line")),
"un-emitted 'second line' was lost after resize; got: {final_lines:?}",
);
assert!(source.is_some(), "expected source from finalize");
}
#[test]
fn controller_set_width_partial_drain_keeps_pending_queue() {
let mut ctrl = stream_controller(Some(40));
ctrl.push("AAAA BBBB CCCC DDDD EEEE FFFF GGGG HHHH IIII JJJJ\n");
ctrl.push("second line\n");
let (cell, idle) = ctrl.on_commit_tick();
assert!(cell.is_some(), "expected 1 emitted line");
assert!(!idle, "queue should still have lines");
assert!(ctrl.queued_lines() > 0, "expected pending queued lines");
ctrl.set_width(Some(20));
assert!(
ctrl.queued_lines() > 0,
"resize must preserve pending queued lines"
);
let mut drained = Vec::new();
for _ in 0..64 {
let (cell, is_idle) = ctrl.on_commit_tick();
if let Some(cell) = cell {
drained.extend(lines_to_plain_strings(&cell.transcript_lines(u16::MAX)));
}
if is_idle {
break;
}
}
assert!(
drained.iter().any(|l| l.contains("second line")),
"pending lines should continue draining after resize; got {drained:?}",
);
}
#[test]
fn controller_set_width_preserves_in_flight_tail() {
let mut ctrl = stream_controller(Some(80));
ctrl.push("tail without newline");
ctrl.set_width(Some(24));
let (cell, _source) = ctrl.finalize();
let rendered = lines_to_plain_strings(
&cell
.expect("expected finalized tail")
.transcript_lines(u16::MAX),
);
assert_eq!(rendered, vec!["• tail without newline".to_string()]);
}
#[test]
fn controller_set_width_preserves_table_tail_when_queue_is_empty() {
let mut ctrl = stream_controller(Some(80));
ctrl.push("intro line\n");
let (_cell, idle) = ctrl.on_commit_tick();
assert!(idle, "intro line should fully drain");
assert_eq!(ctrl.queued_lines(), 0, "expected empty queue before table");
ctrl.push("| A | B |\n");
assert_eq!(
ctrl.queued_lines(),
0,
"pending table header should remain mutable tail, not queued",
);
assert!(ctrl.has_live_tail(), "expected live tail before resize");
ctrl.set_width(Some(24));
let tail_after = lines_to_plain_strings(&ctrl.current_tail_lines());
assert!(
!tail_after.is_empty(),
"resize must keep mutable tail when queue is empty",
);
let joined = tail_after.join(" ");
assert!(
joined.contains('A') && joined.contains('B'),
"expected table header content to remain in tail after resize: {tail_after:?}",
);
}
#[test]
fn plan_controller_set_width_preserves_in_flight_tail() {
let mut ctrl = plan_stream_controller(Some(80));
ctrl.push("1. Item without newline");
ctrl.set_width(Some(24));
let rendered = lines_to_plain_strings(
&(ctrl
.finalize()
.0
.expect("expected finalized tail")
.transcript_lines(u16::MAX)),
);
assert!(
rendered
.iter()
.any(|line| line.contains("Item without newline")),
"expected finalized plan content after resize, got {rendered:?}",
);
}
#[test]
fn plan_controller_holds_table_header_as_live_tail() {
let mut ctrl = plan_stream_controller(Some(80));
assert!(ctrl.push("Intro\n"));
let (_cell, idle) = ctrl.on_commit_tick_batch(usize::MAX);
assert!(idle, "intro line should fully drain");
assert!(!ctrl.push("| Step | Owner |\n"));
assert!(
ctrl.has_live_tail(),
"expected plan table header to be held"
);
}
#[test]
fn controller_loose_vs_tight_with_commit_ticks_matches_full() {
let mut ctrl = stream_controller(/*width*/ None);
let mut lines = Vec::new();
let deltas = vec![
"\n\n",
"Loose",
" vs",
".",
" tight",
" list",
" items",
":\n",
"1",
".",
" Tight",
" item",
"\n",
"2",
".",
" Another",
" tight",
" item",
"\n\n",
"1",
".",
" Loose",
" item",
" with",
" its",
" own",
" paragraph",
".\n\n",
" ",
" This",
" paragraph",
" belongs",
" to",
" the",
" same",
" list",
" item",
".\n\n",
"2",
".",
" Second",
" loose",
" item",
" with",
" a",
" nested",
" list",
" after",
" a",
" blank",
" line",
".\n\n",
" ",
" -",
" Nested",
" bullet",
" under",
" a",
" loose",
" item",
"\n",
" ",
" -",
" Another",
" nested",
" bullet",
"\n\n",
];
for d in deltas.iter() {
ctrl.push(d);
while let (Some(cell), idle) = ctrl.on_commit_tick() {
lines.extend(cell.transcript_lines(u16::MAX));
if idle {
break;
}
}
}
if let (Some(cell), _source) = ctrl.finalize() {
lines.extend(cell.transcript_lines(u16::MAX));
}
let streamed: Vec<_> = lines_to_plain_strings(&lines)
.into_iter()
.map(|s| s.chars().skip(2).collect::<String>())
.collect();
let source: String = deltas.iter().copied().collect();
let mut rendered: Vec<ratatui::text::Line<'static>> = Vec::new();
crate::markdown::append_markdown_agent(&source, /*width*/ None, &mut rendered);
let rendered_strs = lines_to_plain_strings(&rendered);
assert_eq!(streamed, rendered_strs);
let expected = vec![
"Loose vs. tight list items:".to_string(),
"".to_string(),
"1. Tight item".to_string(),
"2. Another tight item".to_string(),
"3. Loose item with its own paragraph.".to_string(),
"".to_string(),
" This paragraph belongs to the same list item.".to_string(),
"4. Second loose item with a nested list after a blank line.".to_string(),
" - Nested bullet under a loose item".to_string(),
" - Another nested bullet".to_string(),
];
assert_eq!(
streamed, expected,
"expected exact rendered lines for loose/tight section"
);
}
#[test]
fn controller_streamed_table_matches_full_render_widths() {
let deltas = vec![
"| Key | Description |\n",
"| --- | --- |\n",
"| -v | Enable very verbose logging output for debugging |\n",
"\n",
];
let streamed = collect_streamed_lines(&deltas, Some(80));
let source: String = deltas.iter().copied().collect();
let mut rendered = Vec::new();
crate::markdown::append_markdown_agent(&source, /*width*/ Some(80), &mut rendered);
let expected = lines_to_plain_strings(&rendered);
assert_eq!(streamed, expected);
}
#[test]
fn controller_holds_blockquoted_table_tail_until_stable() {
let deltas = vec![
"> | A | B |\n",
"> | --- | --- |\n",
"> | longvalue | ok |\n",
"\n",
];
let streamed = collect_streamed_lines(&deltas, Some(80));
let source: String = deltas.iter().copied().collect();
let mut rendered = Vec::new();
crate::markdown::append_markdown_agent(&source, /*width*/ Some(80), &mut rendered);
let expected = lines_to_plain_strings(&rendered);
assert_eq!(streamed, expected);
}
#[test]
fn controller_keeps_pre_table_lines_queued_when_table_is_confirmed() {
let mut ctrl = stream_controller(Some(80));
ctrl.push("Intro line before table.\n");
assert_eq!(ctrl.queued_lines(), 1);
ctrl.push("| Key | Value |\n");
ctrl.push("| --- | --- |\n");
assert_eq!(
ctrl.queued_lines(),
1,
"pre-table line should remain queued after table confirmation",
);
let (cell, idle) = ctrl.on_commit_tick();
let committed = cell
.map(|cell| lines_to_plain_strings(&cell.transcript_lines(u16::MAX)))
.unwrap_or_default();
assert!(
committed
.iter()
.any(|line| line.contains("Intro line before table.")),
"expected pre-table line to commit independently: {committed:?}",
);
assert!(idle, "only pre-table content should have been queued");
}
#[test]
fn controller_set_width_during_confirmed_table_stream_matches_finalize_render() {
let mut ctrl = stream_controller(Some(120));
let deltas = [
"| Key | Description |\n",
"| --- | --- |\n",
"| one | value that should wrap after resize |\n",
];
for delta in deltas {
ctrl.push(delta);
}
assert_eq!(
ctrl.queued_lines(),
0,
"confirmed table should remain mutable"
);
ctrl.set_width(Some(32));
let (cell, source) = ctrl.finalize();
let source = source.expect("expected finalized source");
let streamed = lines_to_plain_strings(
&cell
.expect("expected finalized table")
.transcript_lines(u16::MAX),
)
.into_iter()
.map(|line| line.chars().skip(2).collect::<String>())
.collect::<Vec<_>>();
let mut rendered = Vec::new();
crate::markdown::append_markdown_agent(&source, /*width*/ Some(32), &mut rendered);
let expected = lines_to_plain_strings(&rendered);
assert_eq!(streamed, expected);
}
#[test]
fn controller_does_not_hold_back_pipe_prose_without_table_delimiter() {
let mut ctrl = stream_controller(Some(80));
ctrl.push("status | owner | note\n");
let (_first_commit, first_idle) = ctrl.on_commit_tick();
assert!(first_idle);
ctrl.push("next line\n");
let (second_commit, _second_idle) = ctrl.on_commit_tick();
assert!(
second_commit.is_some(),
"expected prose lines to be released once no table delimiter follows"
);
}
#[test]
fn controller_does_not_stall_repeated_pipe_prose_paragraphs() {
let mut ctrl = stream_controller(Some(80));
ctrl.push("alpha | beta\n\n");
let (_first_commit, first_idle) = ctrl.on_commit_tick();
assert!(first_idle);
ctrl.push("gamma | delta\n\n");
let (second_commit, _second_idle) = ctrl.on_commit_tick();
let second_lines = second_commit
.map(|cell| lines_to_plain_strings(&cell.transcript_lines(u16::MAX)))
.unwrap_or_default();
assert!(
second_lines
.iter()
.any(|line| line.contains("alpha | beta")),
"expected the first pipe-prose paragraph to stream before finalize; got {second_lines:?}",
);
}
#[test]
fn controller_handles_table_immediately_after_heading() {
let deltas = vec![
"### 1) Basic table\n",
"| Name | Role | Status |\n",
"|---|---|---|\n",
"| Alice | Admin | Active |\n",
"| Bob | Editor | Pending |\n",
"\n",
];
let streamed = collect_streamed_lines(&deltas, Some(100));
let source: String = deltas.iter().copied().collect();
let mut rendered = Vec::new();
crate::markdown::append_markdown_agent(&source, /*width*/ Some(100), &mut rendered);
let expected = lines_to_plain_strings(&rendered);
assert_eq!(streamed, expected);
}
#[test]
fn controller_renders_unicode_for_multi_table_response_shape() {
let source = "Absolutely. Here are several different Markdown table patterns you can use for rendering tests.\n\n| Name | Role |
Location |\n|-------|-----------|----------|\n| Ava | Engineer | NYC |\n| Malik | Designer | Berlin |\n| Priya | PM | Remote
|\n\n| Item | Qty | Price | In Stock |\n|:------------|----:|------:|:--------:|\n| Keyboard | 2 | 49.99 | Yes |\n| Mouse | 10
| 19.50 | Yes |\n| Monitor | 1 | 219.0 | No |\n\n| Field | Example | Notes
|\n|---------------|----------------------------------|--------------------------|\n| Escaped pipe | `foo \\| bar` | Should stay
in one cell |\n| Inline code | `let x = value;` | Monospace inline content |\n| Link | [OpenAI](https://openai.com) |
Standard markdown link |\n";
let chunked = source
.split_inclusive('\n')
.map(ToString::to_string)
.collect::<Vec<_>>();
let deltas = chunked.iter().map(String::as_str).collect::<Vec<_>>();
let streamed = collect_streamed_lines(&deltas, Some(120));
assert!(
streamed.iter().any(|line| line.contains('┌')),
"expected unicode table border in streamed output: {streamed:?}"
);
}
#[test]
fn controller_renders_unicode_for_no_outer_pipes_table_shape() {
let source = "### 1) Basic\n\n| Name | Role | Active |\n|---|---|---|\n| Alice | Engineer | Yes |\n| Bob | Designer | No |\n\n### 2) No outer
pipes\n\nCol A | Col B | Col C\n--- | --- | ---\nx | y | z\n10 | 20 | 30\n\n### 3) Another table\n\n| Key | Value |\n|---|---|\n| a | b |\n";
let chunked = source
.split_inclusive('\n')
.map(ToString::to_string)
.collect::<Vec<_>>();
let deltas = chunked.iter().map(String::as_str).collect::<Vec<_>>();
let streamed = collect_streamed_lines(&deltas, Some(100));
let mut rendered = Vec::new();
crate::markdown::append_markdown_agent(source, /*width*/ Some(100), &mut rendered);
let expected = lines_to_plain_strings(&rendered);
assert_eq!(streamed, expected);
let has_raw_no_outer_header = streamed
.iter()
.any(|line| line.trim() == "Col A | Col B | Col C");
assert!(
!has_raw_no_outer_header,
"no-outer-pipes header should not remain raw in final streamed output: {streamed:?}"
);
}
#[test]
fn controller_stabilizes_first_no_outer_pipes_table_in_response() {
let deltas = vec![
"### No outer pipes first\n\n",
"Col A | Col B | Col C\n",
"--- | --- | ---\n",
"x | y | z\n",
"10 | 20 | 30\n",
"\n",
"After table paragraph.\n",
];
let streamed = collect_streamed_lines(&deltas, Some(100));
let source: String = deltas.iter().copied().collect();
let mut rendered = Vec::new();
crate::markdown::append_markdown_agent(&source, /*width*/ Some(100), &mut rendered);
let expected = lines_to_plain_strings(&rendered);
assert_eq!(streamed, expected);
assert!(
streamed.iter().any(|line| line.contains('┌')),
"expected unicode table border for no-outer-pipes streaming: {streamed:?}"
);
assert!(
!streamed
.iter()
.any(|line| line.trim() == "Col A | Col B | Col C"),
"did not expect raw no-outer-pipes header in final streamed output: {streamed:?}"
);
}
#[test]
fn controller_stabilizes_two_column_no_outer_table_in_response() {
let deltas = vec![
"A | B\n",
"--- | ---\n",
"left | right\n",
"\n",
"After table paragraph.\n",
];
let streamed = collect_streamed_lines(&deltas, Some(80));
let source: String = deltas.iter().copied().collect();
let mut rendered = Vec::new();
crate::markdown::append_markdown_agent(&source, /*width*/ Some(80), &mut rendered);
let expected = lines_to_plain_strings(&rendered);
assert_eq!(streamed, expected);
assert!(
streamed.iter().any(|line| line.contains('┌')),
"expected unicode table border for two-column no-outer table: {streamed:?}"
);
assert!(
!streamed.iter().any(|line| line.trim() == "A | B"),
"did not expect raw two-column no-outer header in final streamed output: {streamed:?}"
);
}
#[test]
fn controller_converts_no_outer_table_between_preboxed_sections() {
let source = " ┌───────┬──────────┬────────┐\n │ Name │ Role │ Active │\n ├───────┼──────────┼────────┤\n │ Alice │ Engineer │ Yes │\n │ Bob │ Designer │ No │\n │ Cara │ PM │ Yes │\n └───────┴──────────┴────────┘\n\n ### 3) No outer pipes\n\n Col A | Col B | Col C\n --- | --- | ---\n x | y | z\n 10 | 20 | 30\n\n ┌─────────────────┬────────┬────────────────────────┐\n │ Example │ Output │ Notes │\n ├─────────────────┼────────┼────────────────────────┤\n │ a | b │ `a │ b` │\n │ npm run test │ ok │ Inline code formatting │\n │ SELECT * FROM t │ 3 rows │ SQL snippet │\n └─────────────────┴────────┴────────────────────────┘\n";
let deltas = source
.split_inclusive('\n')
.map(ToString::to_string)
.collect::<Vec<_>>();
let streamed = collect_streamed_lines(
&deltas.iter().map(String::as_str).collect::<Vec<_>>(),
Some(100),
);
let has_raw_no_outer_header = streamed
.iter()
.any(|line| line.trim() == "Col A | Col B | Col C");
assert!(
!has_raw_no_outer_header,
"no-outer table header remained raw in streamed output: {streamed:?}"
);
assert!(
streamed
.iter()
.any(|line| line.contains("┌───────┬───────┬───────┐")),
"expected converted no-outer table border in streamed output: {streamed:?}"
);
}
#[test]
fn controller_keeps_markdown_fenced_tables_mutable_until_finalize() {
let source = "```md\n| A | B |\n|---|---|\n| 1 | 2 |\n```\n";
let deltas = vec![
"```md\n",
"| A | B |\n",
"|---|---|\n",
"| 1 | 2 |\n",
"```\n",
];
let streamed = collect_streamed_lines(&deltas, Some(80));
let mut rendered = Vec::new();
crate::markdown::append_markdown_agent(source, /*width*/ Some(80), &mut rendered);
let expected = lines_to_plain_strings(&rendered);
assert_eq!(streamed, expected);
assert!(
streamed.iter().any(|line| line.contains('┌')),
"expected unicode table border in streamed output: {streamed:?}"
);
assert!(
!streamed.iter().any(|line| line.trim() == "| A | B |"),
"did not expect raw table header line after finalize: {streamed:?}"
);
}
#[test]
fn controller_keeps_markdown_fenced_no_outer_tables_mutable_until_finalize() {
let source =
"```md\nCol A | Col B | Col C\n--- | --- | ---\nx | y | z\n10 | 20 | 30\n```\n";
let deltas = vec![
"```md\n",
"Col A | Col B | Col C\n",
"--- | --- | ---\n",
"x | y | z\n",
"10 | 20 | 30\n",
"```\n",
];
let streamed = collect_streamed_lines(&deltas, Some(100));
let mut rendered = Vec::new();
crate::markdown::append_markdown_agent(source, /*width*/ Some(100), &mut rendered);
let expected = lines_to_plain_strings(&rendered);
assert_eq!(streamed, expected);
assert!(
streamed.iter().any(|line| line.contains('┌')),
"expected unicode table border in streamed output: {streamed:?}"
);
assert!(
!streamed
.iter()
.any(|line| line.trim() == "Col A | Col B | Col C"),
"did not expect raw no-outer-pipes header line after finalize: {streamed:?}"
);
}
#[test]
fn controller_live_view_matches_render_during_interleaved_table_streaming() {
let source = "Project updates are easier to scan when narrative and structured data alternate.\n\n| Focus Area | Owner | Priority | Status |\n|---|---|---|---|\n| Authentication cleanup | Maya | High | 80% |\n| CLI error messages | Jordan | Medium | 55% |\n| Docs refresh | Lee | Low | 30% |\n\nThe first checkpoint shows progress, but we still have open risks.\n\n| Task | Command / Artifact | Due | State |\n|---|---|---|---|\n| Run unit tests | `cargo test -p codex-core` | Today | ✅ |\n| Snapshot review | `cargo insta pending-snapshots -p codex-tui` | Today | ⏳ |\n| Changelog draft | Release template (https://replacechangelog.com/) | Tomorrow | 📝 |\n\nFinal sign-off criteria are summarized below.\n";
let width = Some(72usize);
let mut ctrl = stream_controller(width);
let mut emitted_lines: Vec<Line<'static>> = Vec::new();
for delta in source.split_inclusive('\n') {
ctrl.push(delta);
loop {
let (cell, idle) = ctrl.on_commit_tick();
if let Some(cell) = cell {
emitted_lines.extend(cell.transcript_lines(u16::MAX).into_iter().map(|line| {
let plain: String = line
.spans
.iter()
.map(|s| s.content.clone())
.collect::<Vec<_>>()
.join("");
Line::from(plain.chars().skip(2).collect::<String>())
}));
}
if idle {
break;
}
}
let mut visible = emitted_lines.clone();
visible.extend(ctrl.current_tail_lines());
let visible_plain = lines_to_plain_strings(&visible);
let mut expected = Vec::new();
crate::markdown::append_markdown_agent(
&ctrl.core.raw_source,
/*width*/ width,
&mut expected,
);
let expected_plain = lines_to_plain_strings(&expected);
assert_eq!(
visible_plain, expected_plain,
"live view diverged after delta: {delta:?}"
);
}
}
#[test]
fn controller_keeps_non_markdown_fenced_tables_as_code() {
let source = "```sh\n| A | B |\n|---|---|\n| 1 | 2 |\n```\n";
let deltas = vec![
"```sh\n",
"| A | B |\n",
"|---|---|\n",
"| 1 | 2 |\n",
"```\n",
];
let streamed = collect_streamed_lines(&deltas, Some(80));
let mut rendered = Vec::new();
crate::markdown::append_markdown_agent(source, /*width*/ Some(80), &mut rendered);
let expected = lines_to_plain_strings(&rendered);
assert_eq!(streamed, expected);
assert!(
streamed.iter().any(|line| line.trim() == "| A | B |"),
"expected code-fenced pipe line to remain raw: {streamed:?}"
);
assert!(
!streamed.iter().any(|line| line.contains('┌')),
"did not expect unicode table border for non-markdown fence: {streamed:?}"
);
}
#[test]
fn plan_controller_streamed_table_matches_final_render() {
let deltas = vec![
"## Build plan\n\n",
"| Step | Owner |\n",
"|---|---|\n",
"| Write tests | Agent |\n",
"| Verify output | User |\n",
"\n",
];
let streamed = collect_plan_streamed_lines(&deltas, Some(80));
let source: String = deltas.iter().copied().collect();
let baseline = collect_plan_streamed_lines(&[source.as_str()], Some(80));
assert_eq!(streamed, baseline);
assert!(
streamed
.iter()
.any(|line| line.contains('│') || line.contains('└') || line.contains('┌')),
"expected unicode table box drawing chars in plan streamed output: {streamed:?}"
);
assert!(
!streamed
.iter()
.any(|line| line.trim() == "| Step | Owner |"),
"did not expect raw table header line in plan output: {streamed:?}"
);
}
#[test]
fn plan_controller_streamed_markdown_fenced_table_matches_final_render() {
let deltas = vec![
"## Build plan\n\n",
"```md\n",
"| Step | Owner |\n",
"|---|---|\n",
"| Write tests | Agent |\n",
"| Verify output | User |\n",
"```\n",
"\n",
];
let streamed = collect_plan_streamed_lines(&deltas, Some(80));
let source: String = deltas.iter().copied().collect();
let baseline = collect_plan_streamed_lines(&[source.as_str()], Some(80));
assert_eq!(streamed, baseline);
assert!(
streamed
.iter()
.any(|line| line.contains('│') || line.contains('└') || line.contains('┌')),
"expected unicode table box drawing chars in fenced plan output: {streamed:?}"
);
assert!(
!streamed
.iter()
.any(|line| line.trim() == "| Step | Owner |"),
"did not expect raw table header line in fenced plan output: {streamed:?}"
);
}
#[test]
fn table_holdback_state_detects_header_plus_delimiter() {
let source = "| Key | Description |\n| --- | --- |\n";
assert!(matches!(
table_holdback_state(source),
TableHoldbackState::Confirmed { .. }
));
}
#[test]
fn table_holdback_state_detects_single_column_header_plus_delimiter() {
let source = "| Only |\n| --- |\n";
assert!(matches!(
table_holdback_state(source),
TableHoldbackState::Confirmed { .. }
));
}
#[test]
fn table_holdback_state_ignores_table_like_lines_inside_unclosed_long_fence() {
let source = "````sh\n```cmd\n| Key | Description |\n| --- | --- |\n````\n";
assert!(
matches!(table_holdback_state(source), TableHoldbackState::None),
"table holdback should ignore pipe lines inside an open non-markdown fence",
);
}
#[test]
fn table_holdback_state_treats_indented_fence_text_as_plain_content() {
let source = " ```sh\n| Key | Description |\n| --- | --- |\n";
assert!(
matches!(
table_holdback_state(source),
TableHoldbackState::Confirmed { .. }
),
"indented fence-like text should not open a fence and should not block table detection",
);
}
#[test]
fn table_holdback_state_ignores_table_like_lines_inside_blockquoted_other_fence() {
let source = "> ```sh\n> | Key | Value |\n> | --- | --- |\n> ```\n";
assert!(
matches!(table_holdback_state(source), TableHoldbackState::None),
"table holdback should ignore pipe lines inside non-markdown blockquoted fences",
);
}
#[test]
fn incremental_holdback_matches_stateless_scan_per_chunk() {
let chunks = [
"status | owner\n",
"\n",
"> ```sh\n",
"> | A | B |\n",
"> | --- | --- |\n",
"> ```\n",
"> | Key | Value |\n",
"> | --- | --- |\n",
];
let mut scanner = TableHoldbackScanner::new();
let mut source = String::new();
for chunk in chunks {
source.push_str(chunk);
scanner.push_source_chunk(chunk);
assert_eq!(
scanner.state(),
table_holdback_state(&source),
"scanner mismatch after chunk: {chunk:?}\nsource:\n{source}",
);
}
}
#[test]
fn incremental_holdback_detects_header_delimiter_across_chunk_boundary() {
let mut scanner = TableHoldbackScanner::new();
scanner.push_source_chunk("| A | B |\n");
assert_eq!(
scanner.state(),
TableHoldbackState::PendingHeader { header_start: 0 }
);
scanner.push_source_chunk("| --- | --- |\n");
assert_eq!(
scanner.state(),
TableHoldbackState::Confirmed { table_start: 0 }
);
}
#[test]
fn controller_set_width_after_first_line_emit_does_not_requeue_first_line() {
let mut ctrl = stream_controller(Some(120));
ctrl.push(
"FIRSTTOKEN contains enough words to wrap once the width is reduced dramatically.\n",
);
ctrl.push("second line remains pending\n");
let (first_emit, _) = ctrl.on_commit_tick();
assert!(first_emit.is_some(), "expected first line emission");
ctrl.set_width(Some(20));
let (cell, _source) = ctrl.finalize();
let remaining = cell
.map(|cell| lines_to_plain_strings(&cell.transcript_lines(u16::MAX)))
.unwrap_or_default()
.into_iter()
.map(|line| line.chars().skip(2).collect::<String>())
.collect::<Vec<_>>();
assert!(
!remaining.iter().any(|line| line.contains("FIRSTTOKEN")),
"first line should not be re-queued after resize: {remaining:?}",
);
assert!(
remaining.iter().any(|line| line.contains("second line")),
"expected pending second line after resize: {remaining:?}",
);
}
#[test]
fn controller_set_width_partial_wrapped_emit_preserves_remaining_content() {
let mut ctrl = stream_controller(Some(20));
ctrl.push("The quick brown fox jumps over the lazy dog near the riverbank.\n");
ctrl.push("tail line\n");
let (first_emit, idle) = ctrl.on_commit_tick();
assert!(first_emit.is_some(), "expected first wrapped line emission");
assert!(!idle, "expected remaining queued content after one tick");
assert!(
ctrl.queued_lines() > 0,
"expected non-empty queue before resize"
);
ctrl.set_width(Some(120));
let (cell, _source) = ctrl.finalize();
let remaining = cell
.map(|c| lines_to_plain_strings(&c.transcript_lines(u16::MAX)))
.unwrap_or_default()
.into_iter()
.map(|line| line.chars().skip(2).collect::<String>())
.collect::<Vec<_>>();
assert!(
remaining.iter().any(|line| line.contains("tail line")),
"un-emitted content should remain after resize remap: {remaining:?}",
);
}
#[test]
fn controller_set_width_partial_wrapped_emit_keeps_wrapped_remainder() {
let mut ctrl = stream_controller(Some(18));
ctrl.push("alpha beta gamma delta epsilon zeta eta theta iota kappa lambda mu\n");
let (first_emit, idle) = ctrl.on_commit_tick();
assert!(first_emit.is_some(), "expected first wrapped line emission");
assert!(!idle, "expected remaining wrapped content after one tick");
assert!(
ctrl.queued_lines() > 0,
"expected queued wrapped remainder before resize"
);
ctrl.set_width(Some(80));
let (cell, _source) = ctrl.finalize();
let remaining = cell
.map(|c| lines_to_plain_strings(&c.transcript_lines(u16::MAX)))
.unwrap_or_default();
let joined = remaining.join(" ");
assert!(
joined.contains("kappa") || joined.contains("lambda") || joined.contains("mu"),
"wrapped remainder from partially emitted source line was lost after resize: {remaining:?}",
);
}
}