Skip to content

Commit 712752d

Browse files
[flutter_tools] return terminal to echo and line mode before exiting resident_runner (#105283)
1 parent c181e45 commit 712752d

File tree

2 files changed

+52
-4
lines changed

2 files changed

+52
-4
lines changed

packages/flutter_tools/lib/src/commands/run.dart

Lines changed: 4 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -716,6 +716,10 @@ class RunCommand extends RunCommandBase {
716716
throwToolExit('Lost connection to device.');
717717
}
718718
rethrow;
719+
} finally {
720+
// However we exited from the runner, ensure the terminal has line mode
721+
// and echo mode enabled before we return the user to the shell.
722+
globals.terminal.singleCharMode = false;
719723
}
720724
return FlutterCommandResult(
721725
ExitStatus.success,

packages/flutter_tools/test/commands.shard/hermetic/run_test.dart

Lines changed: 48 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,7 @@ import 'package:flutter_tools/src/base/file_system.dart';
2323
import 'package:flutter_tools/src/base/io.dart';
2424
import 'package:flutter_tools/src/base/logger.dart';
2525
import 'package:flutter_tools/src/base/platform.dart';
26+
import 'package:flutter_tools/src/base/terminal.dart';
2627
import 'package:flutter_tools/src/base/user_messages.dart';
2728
import 'package:flutter_tools/src/build_info.dart';
2829
import 'package:flutter_tools/src/cache.dart';
@@ -47,14 +48,14 @@ import '../../src/fakes.dart';
4748
import '../../src/test_flutter_command_runner.dart';
4849

4950
void main() {
51+
setUpAll(() {
52+
Cache.disableLocking();
53+
});
54+
5055
group('run', () {
5156
FakeDeviceManager mockDeviceManager;
5257
FileSystem fileSystem;
5358

54-
setUpAll(() {
55-
Cache.disableLocking();
56-
});
57-
5859
setUp(() {
5960
mockDeviceManager = FakeDeviceManager();
6061
fileSystem = MemoryFileSystem.test();
@@ -657,6 +658,35 @@ void main() {
657658
});
658659
});
659660

661+
group('terminal', () {
662+
FakeAnsiTerminal fakeTerminal;
663+
664+
setUp(() {
665+
fakeTerminal = FakeAnsiTerminal();
666+
});
667+
668+
testUsingContext('Flutter run sets terminal singleCharMode to false on exit', () async {
669+
final FakeResidentRunner residentRunner = FakeResidentRunner();
670+
final TestRunCommandWithFakeResidentRunner command = TestRunCommandWithFakeResidentRunner();
671+
command.fakeResidentRunner = residentRunner;
672+
673+
await createTestCommandRunner(command).run(<String>[
674+
'run',
675+
'--no-pub',
676+
]);
677+
// The sync completer where we initially set `terminal.singleCharMode` to
678+
// `true` does not execute in unit tests, so explicitly check the
679+
// `setSingleCharModeHistory` that the finally block ran, setting this
680+
// back to `false`.
681+
expect(fakeTerminal.setSingleCharModeHistory, contains(false));
682+
}, overrides: <Type, Generator>{
683+
AnsiTerminal: () => fakeTerminal,
684+
Cache: () => Cache.test(processManager: FakeProcessManager.any()),
685+
FileSystem: () => MemoryFileSystem.test(),
686+
ProcessManager: () => FakeProcessManager.any(),
687+
});
688+
});
689+
660690
testUsingContext('Flutter run catches service has disappear errors and throws a tool exit', () async {
661691
final FakeResidentRunner residentRunner = FakeResidentRunner();
662692
residentRunner.rpcError = RPCError('flutter._listViews', RPCErrorCodes.kServiceDisappeared, '');
@@ -1021,3 +1051,17 @@ class CapturingAppDomain extends AppDomain {
10211051
throwToolExit('');
10221052
}
10231053
}
1054+
1055+
class FakeAnsiTerminal extends Fake implements AnsiTerminal {
1056+
@override
1057+
bool usesTerminalUi = false;
1058+
1059+
/// A list of all the calls to the [singleCharMode] setter.
1060+
List<bool> setSingleCharModeHistory = <bool>[];
1061+
1062+
@override
1063+
set singleCharMode(bool value) => setSingleCharModeHistory.add(value);
1064+
1065+
@override
1066+
bool get singleCharMode => setSingleCharModeHistory.last;
1067+
}

0 commit comments

Comments
 (0)