terminal.test.ts 12.0 KB
Newer Older
1 2 3 4 5
/*---------------------------------------------------------------------------------------------
 *  Copyright (c) Microsoft Corporation. All rights reserved.
 *  Licensed under the MIT License. See License.txt in the project root for license information.
 *--------------------------------------------------------------------------------------------*/

6
import { window, Pseudoterminal, EventEmitter, TerminalDimensions, workspace, ConfigurationTarget, Disposable } from 'vscode';
7
import { doesNotThrow, equal, ok, deepEqual } from 'assert';
8 9

suite('window namespace tests', () => {
10 11 12 13 14
	suiteSetup(async () => {
		// Disable conpty in integration tests because of https://github.com/microsoft/vscode/issues/76548
		await workspace.getConfiguration('terminal.integrated').update('windowsEnableConpty', false, ConfigurationTarget.Global);
	});
	suite('Terminal', () => {
15 16 17 18 19 20 21
		let disposables: Disposable[] = [];

		teardown(() => {
			disposables.forEach(d => d.dispose());
			disposables.length = 0;
		});

22
		test('sendText immediately after createTerminal should not throw', (done) => {
23
			disposables.push(window.onDidOpenTerminal(term => {
D
Daniel Imms 已提交
24 25 26 27 28
				try {
					equal(terminal, term);
				} catch (e) {
					done(e);
				}
29
				terminal.dispose();
30 31
				disposables.push(window.onDidCloseTerminal(() => done()));
			}));
32 33 34 35 36
			const terminal = window.createTerminal();
			doesNotThrow(terminal.sendText.bind(terminal, 'echo "foo"'));
		});

		test('onDidCloseTerminal event fires when terminal is disposed', (done) => {
37
			disposables.push(window.onDidOpenTerminal(term => {
D
Daniel Imms 已提交
38 39 40 41 42
				try {
					equal(terminal, term);
				} catch (e) {
					done(e);
				}
43
				terminal.dispose();
44 45
				disposables.push(window.onDidCloseTerminal(() => done()));
			}));
46
			const terminal = window.createTerminal();
47 48 49
		});

		test('processId immediately after createTerminal should fetch the pid', (done) => {
50
			disposables.push(window.onDidOpenTerminal(term => {
D
Daniel Imms 已提交
51 52 53 54 55
				try {
					equal(terminal, term);
				} catch (e) {
					done(e);
				}
56
				terminal.processId.then(id => {
D
Daniel Imms 已提交
57 58 59 60 61
					try {
						ok(id > 0);
					} catch (e) {
						done(e);
					}
62
					terminal.dispose();
63
					disposables.push(window.onDidCloseTerminal(() => done()));
64
				});
65
			}));
66
			const terminal = window.createTerminal();
67 68
		});

69
		test('name in constructor should set terminal.name', (done) => {
70
			disposables.push(window.onDidOpenTerminal(term => {
D
Daniel Imms 已提交
71 72 73 74 75
				try {
					equal(terminal, term);
				} catch (e) {
					done(e);
				}
76
				terminal.dispose();
77 78
				disposables.push(window.onDidCloseTerminal(() => done()));
			}));
79
			const terminal = window.createTerminal('a');
D
Daniel Imms 已提交
80 81 82 83 84
			try {
				equal(terminal.name, 'a');
			} catch (e) {
				done(e);
			}
85 86 87
		});

		test('onDidOpenTerminal should fire when a terminal is created', (done) => {
88
			disposables.push(window.onDidOpenTerminal(term => {
D
Daniel Imms 已提交
89 90 91 92 93
				try {
					equal(term.name, 'b');
				} catch (e) {
					done(e);
				}
94
				disposables.push(window.onDidCloseTerminal(() => done()));
95
				terminal.dispose();
96
			}));
97 98
			const terminal = window.createTerminal('b');
		});
99

100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115
		// test('onDidChangeActiveTerminal should fire when new terminals are created', (done) => {
		// 	const reg1 = window.onDidChangeActiveTerminal((active: Terminal | undefined) => {
		// 		equal(active, terminal);
		// 		equal(active, window.activeTerminal);
		// 		reg1.dispose();
		// 		const reg2 = window.onDidChangeActiveTerminal((active: Terminal | undefined) => {
		// 			equal(active, undefined);
		// 			equal(active, window.activeTerminal);
		// 			reg2.dispose();
		// 			done();
		// 		});
		// 		terminal.dispose();
		// 	});
		// 	const terminal = window.createTerminal();
		// 	terminal.show();
		// });
116

117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168
		// test('onDidChangeTerminalDimensions should fire when new terminals are created', (done) => {
		// 	const reg1 = window.onDidChangeTerminalDimensions(async (event: TerminalDimensionsChangeEvent) => {
		// 		equal(event.terminal, terminal1);
		// 		equal(typeof event.dimensions.columns, 'number');
		// 		equal(typeof event.dimensions.rows, 'number');
		// 		ok(event.dimensions.columns > 0);
		// 		ok(event.dimensions.rows > 0);
		// 		reg1.dispose();
		// 		let terminal2: Terminal;
		// 		const reg2 = window.onDidOpenTerminal((newTerminal) => {
		// 			// This is guarantees to fire before dimensions change event
		// 			if (newTerminal !== terminal1) {
		// 				terminal2 = newTerminal;
		// 				reg2.dispose();
		// 			}
		// 		});
		// 		let firstCalled = false;
		// 		let secondCalled = false;
		// 		const reg3 = window.onDidChangeTerminalDimensions((event: TerminalDimensionsChangeEvent) => {
		// 			if (event.terminal === terminal1) {
		// 				// The original terminal should fire dimension change after a split
		// 				firstCalled = true;
		// 			} else if (event.terminal !== terminal1) {
		// 				// The new split terminal should fire dimension change
		// 				secondCalled = true;
		// 			}
		// 			if (firstCalled && secondCalled) {
		// 				let firstDisposed = false;
		// 				let secondDisposed = false;
		// 				const reg4 = window.onDidCloseTerminal(term => {
		// 					if (term === terminal1) {
		// 						firstDisposed = true;
		// 					}
		// 					if (term === terminal2) {
		// 						secondDisposed = true;
		// 					}
		// 					if (firstDisposed && secondDisposed) {
		// 						reg4.dispose();
		// 						done();
		// 					}
		// 				});
		// 				terminal1.dispose();
		// 				terminal2.dispose();
		// 				reg3.dispose();
		// 			}
		// 		});
		// 		await timeout(500);
		// 		commands.executeCommand('workbench.action.terminal.split');
		// 	});
		// 	const terminal1 = window.createTerminal({ name: 'test' });
		// 	terminal1.show();
		// });
169

170 171 172
		suite('hideFromUser', () => {
			test('should be available to terminals API', done => {
				const terminal = window.createTerminal({ name: 'bg', hideFromUser: true });
173
				disposables.push(window.onDidOpenTerminal(t => {
D
Daniel Imms 已提交
174 175 176 177 178 179 180
					try {
						equal(t, terminal);
						equal(t.name, 'bg');
						ok(window.terminals.indexOf(terminal) !== -1);
					} catch (e) {
						done(e);
					}
181 182
					disposables.push(window.onDidCloseTerminal(() => {
						// reg3.dispose();
183
						done();
184
					}));
185
					terminal.dispose();
186
				}));
187 188
			});
		});
189

190 191 192 193 194
		suite('window.onDidWriteTerminalData', () => {
			test('should listen to all future terminal data events', (done) => {
				const openEvents: string[] = [];
				const dataEvents: { name: string, data: string }[] = [];
				const closeEvents: string[] = [];
195
				disposables.push(window.onDidOpenTerminal(e => openEvents.push(e.name)));
196 197

				let resolveOnceDataWritten: (() => void) | undefined;
E
Eric Amodio 已提交
198
				let resolveOnceClosed: (() => void) | undefined;
199

200
				disposables.push(window.onDidWriteTerminalData(e => {
201 202 203
					dataEvents.push({ name: e.terminal.name, data: e.data });

					resolveOnceDataWritten!();
204
				}));
205

206
				disposables.push(window.onDidCloseTerminal(e => {
207
					closeEvents.push(e.name);
D
Daniel Imms 已提交
208 209 210 211 212 213 214 215 216 217
					try {
						if (closeEvents.length === 1) {
							deepEqual(openEvents, ['test1']);
							deepEqual(dataEvents, [{ name: 'test1', data: 'write1' }]);
							deepEqual(closeEvents, ['test1']);
						} else if (closeEvents.length === 2) {
							deepEqual(openEvents, ['test1', 'test2']);
							deepEqual(dataEvents, [{ name: 'test1', data: 'write1' }, { name: 'test2', data: 'write2' }]);
							deepEqual(closeEvents, ['test1', 'test2']);
						}
E
Eric Amodio 已提交
218
						resolveOnceClosed!();
D
Daniel Imms 已提交
219 220
					} catch (e) {
						done(e);
221
					}
222
				}));
223 224 225

				const term1Write = new EventEmitter<string>();
				const term1Close = new EventEmitter<void>();
E
Eric Amodio 已提交
226 227 228 229 230 231 232 233 234 235 236 237 238 239
				window.createTerminal({
					name: 'test1', pty: {
						onDidWrite: term1Write.event,
						onDidClose: term1Close.event,
						open: async () => {
							term1Write.fire('write1');

							// Wait until the data is written
							await new Promise(resolve => { resolveOnceDataWritten = resolve; });

							term1Close.fire();

							// Wait until the terminal is closed
							await new Promise<void>(resolve => { resolveOnceClosed = resolve; });
D
Daniel Imms 已提交
240

E
Eric Amodio 已提交
241 242 243 244 245 246 247 248
							const term2Write = new EventEmitter<string>();
							const term2Close = new EventEmitter<void>();
							window.createTerminal({
								name: 'test2', pty: {
									onDidWrite: term2Write.event,
									onDidClose: term2Close.event,
									open: async () => {
										term2Write.fire('write2');
D
Daniel Imms 已提交
249

E
Eric Amodio 已提交
250 251
										// Wait until the data is written
										await new Promise<void>(resolve => { resolveOnceDataWritten = resolve; });
D
Daniel Imms 已提交
252

E
Eric Amodio 已提交
253
										term2Close.fire();
D
Daniel Imms 已提交
254

E
Eric Amodio 已提交
255 256 257 258 259 260 261 262 263 264 265 266
										// Wait until the terminal is closed
										await new Promise<void>(resolve => { resolveOnceClosed = resolve; });

										done();
									},
									close: () => { }
								}
							});
						},
						close: () => { }
					}
				});
267 268 269
			});
		});

D
Daniel Imms 已提交
270
		suite('Extension pty terminals', () => {
271
			test('should fire onDidOpenTerminal and onDidCloseTerminal', (done) => {
272
				disposables.push(window.onDidOpenTerminal(term => {
D
Daniel Imms 已提交
273 274 275 276 277
					try {
						equal(term.name, 'c');
					} catch (e) {
						done(e);
					}
278
					disposables.push(window.onDidCloseTerminal(() => done()));
279
					term.dispose();
280
				}));
281
				const pty: Pseudoterminal = {
D
Daniel Imms 已提交
282
					onDidWrite: new EventEmitter<string>().event,
283 284
					open: () => { },
					close: () => { }
285
				};
286
				window.createTerminal({ name: 'c', pty });
287 288
			});

289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334
			// The below tests depend on global UI state and each other
			// test('should not provide dimensions on start as the terminal has not been shown yet', (done) => {
			// 	const reg1 = window.onDidOpenTerminal(term => {
			// 		equal(terminal, term);
			// 		reg1.dispose();
			// 	});
			// 	const pty: Pseudoterminal = {
			// 		onDidWrite: new EventEmitter<string>().event,
			// 		open: (dimensions) => {
			// 			equal(dimensions, undefined);
			// 			const reg3 = window.onDidCloseTerminal(() => {
			// 				reg3.dispose();
			// 				done();
			// 			});
			// 			// Show a terminal and wait a brief period before dispose, this will cause
			// 			// the panel to init it's dimenisons and be provided to following terminals.
			// 			// The following test depends on this.
			// 			terminal.show();
			// 			setTimeout(() => terminal.dispose(), 200);
			// 		},
			// 		close: () => {}
			// 	};
			// 	const terminal = window.createTerminal({ name: 'foo', pty });
			// });
			// test('should provide dimensions on start as the terminal has been shown', (done) => {
			// 	const reg1 = window.onDidOpenTerminal(term => {
			// 		equal(terminal, term);
			// 		reg1.dispose();
			// 	});
			// 	const pty: Pseudoterminal = {
			// 		onDidWrite: new EventEmitter<string>().event,
			// 		open: (dimensions) => {
			// 			// This test depends on Terminal.show being called some time before such
			// 			// that the panel dimensions are initialized and cached.
			// 			ok(dimensions!.columns > 0);
			// 			ok(dimensions!.rows > 0);
			// 			const reg3 = window.onDidCloseTerminal(() => {
			// 				reg3.dispose();
			// 				done();
			// 			});
			// 			terminal.dispose();
			// 		},
			// 		close: () => {}
			// 	};
			// 	const terminal = window.createTerminal({ name: 'foo', pty });
			// });
D
Daniel Imms 已提交
335 336

			test('should respect dimension overrides', (done) => {
337
				disposables.push(window.onDidOpenTerminal(term => {
D
Daniel Imms 已提交
338 339 340 341 342
					try {
						equal(terminal, term);
					} catch (e) {
						done(e);
					}
D
Daniel Imms 已提交
343
					term.show();
344
					disposables.push(window.onDidChangeTerminalDimensions(e => {
345 346 347 348
						if (e.dimensions.columns === 0 || e.dimensions.rows === 0) {
							// HACK: Ignore the event if dimension(s) are zero (#83778)
							return;
						}
D
Daniel Imms 已提交
349 350 351 352 353 354 355
						try {
							equal(e.dimensions.columns, 10);
							equal(e.dimensions.rows, 5);
							equal(e.terminal, terminal);
						} catch (e) {
							done(e);
						}
356
						disposables.push(window.onDidCloseTerminal(() => done()));
D
Daniel Imms 已提交
357
						terminal.dispose();
358 359
					}));
				}));
D
Daniel Imms 已提交
360 361
				const writeEmitter = new EventEmitter<string>();
				const overrideDimensionsEmitter = new EventEmitter<TerminalDimensions>();
362
				const pty: Pseudoterminal = {
363
					onDidWrite: writeEmitter.event,
D
Daniel Imms 已提交
364
					onDidOverrideDimensions: overrideDimensionsEmitter.event,
D
Daniel Imms 已提交
365 366 367
					open: () => {
						overrideDimensionsEmitter.fire({ columns: 10, rows: 5 });
					},
368
					close: () => { }
D
Daniel Imms 已提交
369
				};
370
				const terminal = window.createTerminal({ name: 'foo', pty });
D
Daniel Imms 已提交
371
			});
372
		});
373 374
	});
});