import { expect } from '@jest/globals';
import { type Stdio, CaptureStdio, StdioMultiplexer, TargetedStdio } from '../../src/stdio';
describe('CaptureStdio', () => {
it('captured streams are initialized to empty string', () => {
let stdio = new CaptureStdio();
expect(stdio.captured_stdout).toBe('');
expect(stdio.captured_stderr).toBe('');
});
it('stdout() and stderr() captures', () => {
let stdio = new CaptureStdio();
stdio.stdout_writeline('hello');
stdio.stdout_writeline('world');
stdio.stderr_writeline('this is an error');
expect(stdio.captured_stdout).toBe('hello\nworld\n');
expect(stdio.captured_stderr).toBe('this is an error\n');
});
it('reset() works', () => {
let stdio = new CaptureStdio();
stdio.stdout_writeline('aaa');
stdio.stderr_writeline('bbb');
stdio.reset();
expect(stdio.captured_stdout).toBe('');
expect(stdio.captured_stderr).toBe('');
});
});
describe('StdioMultiplexer', () => {
let a: CaptureStdio;
let b: CaptureStdio;
let multi: StdioMultiplexer;
beforeEach(() => {
a = new CaptureStdio();
b = new CaptureStdio();
multi = new StdioMultiplexer();
});
it('works without listeners', () => {
// no listeners, messages are ignored
multi.stdout_writeline('out 1');
multi.stderr_writeline('err 1');
expect(a.captured_stdout).toBe('');
expect(a.captured_stderr).toBe('');
expect(b.captured_stdout).toBe('');
expect(b.captured_stderr).toBe('');
});
it('redirects to multiple listeners', () => {
multi.addListener(a);
multi.stdout_writeline('out 1');
multi.stderr_writeline('err 1');
multi.addListener(b);
multi.stdout_writeline('out 2');
multi.stderr_writeline('err 2');
expect(a.captured_stdout).toBe('out 1\nout 2\n');
expect(a.captured_stderr).toBe('err 1\nerr 2\n');
expect(b.captured_stdout).toBe('out 2\n');
expect(b.captured_stderr).toBe('err 2\n');
});
});
describe('TargetedStdio', () => {
let capture: CaptureStdio;
let targeted: TargetedStdio;
let error_targeted: TargetedStdio;
let multi: StdioMultiplexer;
beforeEach(() => {
//DOM element to capture stdout and stderr
let target_div = document.getElementById('output-id');
if (target_div === null) {
target_div = document.createElement('div');
target_div.id = 'output-id';
document.body.appendChild(target_div);
} else {
target_div.innerHTML = '';
}
//DOM element to capture stderr
let error_div = document.getElementById('error-id');
if (error_div === null) {
error_div = document.createElement('div');
error_div.id = 'error-id';
document.body.appendChild(error_div);
} else {
error_div.innerHTML = '';
}
const tag = document.createElement('div');
tag.setAttribute('output', 'output-id');
tag.setAttribute('stderr', 'error-id');
capture = new CaptureStdio();
targeted = new TargetedStdio(tag, 'output', true, true);
error_targeted = new TargetedStdio(tag, 'stderr', false, true);
multi = new StdioMultiplexer();
multi.addListener(capture);
multi.addListener(targeted);
multi.addListener(error_targeted);
});
it('targeted id is set by constructor', () => {
expect(targeted.source_attribute).toBe('output');
});
it('targeted stdio/stderr also goes to multiplexer', () => {
multi.stdout_writeline('out 1');
multi.stderr_writeline('out 2');
expect(capture.captured_stdout).toBe('out 1\n');
expect(capture.captured_stderr).toBe('out 2\n');
expect(document.getElementById('output-id')?.innerHTML).toBe('out 1
out 2
');
expect(document.getElementById('error-id')?.innerHTML).toBe('out 2
');
});
it('Add and remove targeted listener', () => {
multi.stdout_writeline('out 1');
multi.removeListener(targeted);
multi.stdout_writeline('out 2');
multi.addListener(targeted);
multi.stdout_writeline('out 3');
//all three should be captured by multiplexer
expect(capture.captured_stdout).toBe('out 1\nout 2\nout 3\n');
//out 2 should not be present in the DOM element
expect(document.getElementById('output-id')?.innerHTML).toBe('out 1
out 3
');
});
});