Hammer createTokenizer with the worst stalls a terminal can produce — split + flush at every interior byte, and a 200-report byte-by-byte feed that flushes after every single byte — and assert the two invariants that make the SGR-leak class structurally impossible: nothing ever leaks as a text token, and every complete report reassembles whole. A mixed mouse+keystroke variant proves real input survives the same storm.
186 lines
6.1 KiB
TypeScript
186 lines
6.1 KiB
TypeScript
import { describe, expect, it } from 'vitest'
|
|
|
|
import { createTokenizer, type Token } from './tokenize.js'
|
|
|
|
describe('tokenizer escape-sequence boundaries', () => {
|
|
it('reassembles a CSI mouse sequence split across two feeds', () => {
|
|
const t = createTokenizer({ x10Mouse: true })
|
|
|
|
expect(t.feed('\x1b[<0;35;')).toEqual([])
|
|
expect(t.feed('46M')).toEqual([{ type: 'sequence', value: '\x1b[<0;35;46M' }])
|
|
expect(t.buffer()).toBe('')
|
|
})
|
|
})
|
|
|
|
describe('tokenizer state-aware flush', () => {
|
|
it('does not emit an incomplete CSI on flush — it keeps it for reassembly', () => {
|
|
const t = createTokenizer({ x10Mouse: true })
|
|
|
|
// A render stall lets App's watchdog flush mid-sequence. The buffered CSI
|
|
// prefix must NOT be emitted (that is the `46M…` leak); it stays buffered.
|
|
expect(t.feed('\x1b[<0;35;')).toEqual([])
|
|
expect(t.flush()).toEqual([])
|
|
expect(t.buffer()).toBe('\x1b[<0;35;')
|
|
|
|
// The continuation arrives on the next feed and the whole report
|
|
// reassembles into a single clean sequence token — nothing leaked.
|
|
expect(t.feed('46M')).toEqual([{ type: 'sequence', value: '\x1b[<0;35;46M' }])
|
|
expect(t.buffer()).toBe('')
|
|
})
|
|
|
|
it('drops a partial control sequence that survives a second flush (truncation)', () => {
|
|
const t = createTokenizer({ x10Mouse: true })
|
|
|
|
expect(t.feed('\x1b[<0;35;')).toEqual([])
|
|
expect(t.flush()).toEqual([]) // first flush keeps the buffer
|
|
expect(t.buffer()).toBe('\x1b[<0;35;')
|
|
|
|
// Continuation never arrived: the next flush sees the same buffer and
|
|
// drops it so it can't fuse with the next keypress's bytes.
|
|
expect(t.flush()).toEqual([])
|
|
expect(t.buffer()).toBe('')
|
|
})
|
|
|
|
it('still emits a bare ESC on flush so the Escape key works', () => {
|
|
const t = createTokenizer({ x10Mouse: true })
|
|
|
|
expect(t.feed('\x1b')).toEqual([])
|
|
expect(t.flush()).toEqual([{ type: 'sequence', value: '\x1b' }])
|
|
expect(t.buffer()).toBe('')
|
|
})
|
|
|
|
it('reassembles even when a flush fires between every byte of the report', () => {
|
|
const t = createTokenizer({ x10Mouse: true })
|
|
|
|
// Pathological stall: a flush between each chunk. As long as the
|
|
// continuation eventually arrives, no fragment is ever emitted as input.
|
|
for (const chunk of ['\x1b[', '<', '0;', '35;', '46']) {
|
|
expect(t.feed(chunk)).toEqual([])
|
|
expect(t.flush()).toEqual([])
|
|
}
|
|
|
|
expect(t.feed('M')).toEqual([{ type: 'sequence', value: '\x1b[<0;35;46M' }])
|
|
expect(t.buffer()).toBe('')
|
|
})
|
|
})
|
|
|
|
// Battle-test: prove the leak class is structurally impossible, not just that
|
|
// the known cases are patched. We hammer the tokenizer with the worst stalls a
|
|
// terminal can produce (split + flush at every byte) and assert the two hard
|
|
// invariants: nothing leaks as text, and every complete report reassembles.
|
|
describe('tokenizer fuzz: fragments never leak under a flush storm', () => {
|
|
const sgr = (btn: number, col: number, row: number, press: boolean): string =>
|
|
`\x1b[<${btn};${col};${row}${press ? 'M' : 'm'}`
|
|
|
|
it('reassembles a report split + flushed at every interior byte', () => {
|
|
const seq = sgr(0, 35, 46, true)
|
|
|
|
// Start at 2: an earlier split is the lone-ESC ESCDELAY boundary, which
|
|
// intentionally flushes to the Escape key. Terminals never split a mouse
|
|
// report there — a report is one atomic write — so it's not a real case.
|
|
for (let i = 2; i < seq.length; i++) {
|
|
const t = createTokenizer({ x10Mouse: true })
|
|
const tokens: Token[] = [...t.feed(seq.slice(0, i)), ...t.flush(), ...t.feed(seq.slice(i))]
|
|
|
|
expect(tokens).toEqual([{ type: 'sequence', value: seq }])
|
|
expect(t.buffer()).toBe('')
|
|
}
|
|
})
|
|
|
|
it('feeds 200 random reports one byte at a time, flushing after every byte', () => {
|
|
// Deterministic PRNG so a failure is reproducible.
|
|
let s = 0x1234567
|
|
|
|
const rnd = (n: number): number => {
|
|
s = (s * 1103515245 + 12345) & 0x7fffffff
|
|
|
|
return s % n
|
|
}
|
|
|
|
const reports = Array.from({ length: 200 }, () => sgr(rnd(120), 1 + rnd(300), 1 + rnd(200), rnd(2) === 0))
|
|
const stream = reports.join('')
|
|
|
|
const t = createTokenizer({ x10Mouse: true })
|
|
const seqTokens: string[] = []
|
|
let textLeak = ''
|
|
|
|
const drain = (tokens: Token[]): void => {
|
|
for (const tok of tokens) {
|
|
if (tok.type === 'sequence') {
|
|
seqTokens.push(tok.value)
|
|
} else {
|
|
textLeak += tok.value
|
|
}
|
|
}
|
|
}
|
|
|
|
for (const ch of stream) {
|
|
drain(t.feed(ch))
|
|
|
|
// Flush storm — but not at a lone-ESC boundary (the real watchdog
|
|
// re-arms while bytes are pending; a single flush between feeds never
|
|
// hits the truncation valve).
|
|
if (t.buffer() !== '\x1b') {
|
|
drain(t.flush())
|
|
}
|
|
}
|
|
|
|
expect(textLeak).toBe('')
|
|
expect(seqTokens.join('')).toBe(stream)
|
|
})
|
|
|
|
it('keeps real keystrokes intact while mouse reports reassemble around them', () => {
|
|
let s = 0x0badf00d
|
|
|
|
const rnd = (n: number): number => {
|
|
s = (s * 1103515245 + 12345) & 0x7fffffff
|
|
|
|
return s % n
|
|
}
|
|
|
|
const typed = 'abc 123 xyz'
|
|
const expectedKeys: string[] = []
|
|
const expectedSeqs: string[] = []
|
|
const parts: string[] = []
|
|
|
|
for (let k = 0; k < 120; k++) {
|
|
if (rnd(3) === 0) {
|
|
const ch = typed[rnd(typed.length)]!
|
|
expectedKeys.push(ch)
|
|
parts.push(ch)
|
|
} else {
|
|
const seq = sgr(rnd(64), 1 + rnd(200), 1 + rnd(100), rnd(2) === 0)
|
|
expectedSeqs.push(seq)
|
|
parts.push(seq)
|
|
}
|
|
}
|
|
|
|
const stream = parts.join('')
|
|
const t = createTokenizer({ x10Mouse: true })
|
|
const seqTokens: string[] = []
|
|
let text = ''
|
|
|
|
const drain = (tokens: Token[]): void => {
|
|
for (const tok of tokens) {
|
|
if (tok.type === 'sequence') {
|
|
seqTokens.push(tok.value)
|
|
} else {
|
|
text += tok.value
|
|
}
|
|
}
|
|
}
|
|
|
|
for (const ch of stream) {
|
|
drain(t.feed(ch))
|
|
|
|
if (t.buffer() !== '\x1b') {
|
|
drain(t.flush())
|
|
}
|
|
}
|
|
|
|
// Every typed character survives, in order; every report reassembles whole.
|
|
expect(text).toBe(expectedKeys.join(''))
|
|
expect(seqTokens).toEqual(expectedSeqs)
|
|
})
|
|
})
|