Unverified Commit d98213c4 authored by Jonah Williams's avatar Jonah Williams Committed by GitHub

[flutter_tools] reland: Update background isolates when performing hot reload/restart (#52479)

Reland of #52149
parent 02769001
...@@ -74,7 +74,7 @@ void main() { ...@@ -74,7 +74,7 @@ void main() {
run.stdin.write('P'); run.stdin.write('P');
await driver.drive('none'); await driver.drive('none');
final Future<String> reloadStartingText = final Future<String> reloadStartingText =
stdout.stream.firstWhere((String line) => line.endsWith('] Initializing hot reload...')); stdout.stream.firstWhere((String line) => line.endsWith('] Performing hot reload...'));
final Future<String> reloadEndingText = final Future<String> reloadEndingText =
stdout.stream.firstWhere((String line) => line.contains('] Reloaded ') && line.endsWith('ms.')); stdout.stream.firstWhere((String line) => line.contains('] Reloaded ') && line.endsWith('ms.'));
print('test: pressing "r" to perform a hot reload...'); print('test: pressing "r" to perform a hot reload...');
......
...@@ -283,8 +283,8 @@ class FlutterDevice { ...@@ -283,8 +283,8 @@ class FlutterDevice {
final Uri deviceEntryUri = devFS.baseUri.resolveUri(globals.fs.path.toUri(entryPath)); final Uri deviceEntryUri = devFS.baseUri.resolveUri(globals.fs.path.toUri(entryPath));
final Uri devicePackagesUri = devFS.baseUri.resolve('.packages'); final Uri devicePackagesUri = devFS.baseUri.resolve('.packages');
return <Future<Map<String, dynamic>>>[ return <Future<Map<String, dynamic>>>[
for (final FlutterView view in views) for (final Isolate isolate in vmService.vm.isolates)
view.uiIsolate.reloadSources( isolate.reloadSources(
pause: pause, pause: pause,
rootLibUri: deviceEntryUri, rootLibUri: deviceEntryUri,
packagesUri: devicePackagesUri, packagesUri: devicePackagesUri,
......
...@@ -93,9 +93,9 @@ class HotRunner extends ResidentRunner { ...@@ -93,9 +93,9 @@ class HotRunner extends ResidentRunner {
bool _didAttach = false; bool _didAttach = false;
final Map<String, List<int>> benchmarkData = <String, List<int>>{}; final Map<String, List<int>> benchmarkData = <String, List<int>>{};
// The initial launch is from a snapshot.
bool _runningFromSnapshot = true;
DateTime firstBuildTime; DateTime firstBuildTime;
bool _shouldResetAssetDirectory = true;
void _addBenchmarkData(String name, int value) { void _addBenchmarkData(String name, int value) {
benchmarkData[name] ??= <int>[]; benchmarkData[name] ??= <int>[];
...@@ -520,14 +520,16 @@ class HotRunner extends ResidentRunner { ...@@ -520,14 +520,16 @@ class HotRunner extends ResidentRunner {
} }
} }
// Check if the isolate is paused and resume it. // Check if the isolate is paused and resume it.
final List<Future<void>> futures = <Future<void>>[]; final List<Future<void>> operations = <Future<void>>[];
for (final FlutterDevice device in flutterDevices) { for (final FlutterDevice device in flutterDevices) {
final Set<Isolate> uiIsolates = <Isolate>{};
for (final FlutterView view in device.views) { for (final FlutterView view in device.views) {
if (view.uiIsolate == null) { if (view.uiIsolate == null) {
continue; continue;
} }
uiIsolates.add(view.uiIsolate);
// Reload the isolate. // Reload the isolate.
futures.add(view.uiIsolate.reload().then((ServiceObject _) { operations.add(view.uiIsolate.reload().then((ServiceObject _) {
final ServiceEvent pauseEvent = view.uiIsolate.pauseEvent; final ServiceEvent pauseEvent = view.uiIsolate.pauseEvent;
if ((pauseEvent != null) && pauseEvent.isPauseEvent) { if ((pauseEvent != null) && pauseEvent.isPauseEvent) {
// Resume the isolate so that it can be killed by the embedder. // Resume the isolate so that it can be killed by the embedder.
...@@ -536,16 +538,22 @@ class HotRunner extends ResidentRunner { ...@@ -536,16 +538,22 @@ class HotRunner extends ResidentRunner {
return null; return null;
})); }));
} }
// The engine handles killing and recreating isolates that it has spawned
// ("uiIsolates"). The isolates that were spawned from these uiIsolates
// will not be restared, and so they must be manually killed.
for (final Isolate isolate in device?.vmService?.vm?.isolates ?? <Isolate>[]) {
if (!uiIsolates.contains(isolate)) {
operations.add(isolate.invokeRpcRaw('kill', params: <String, dynamic>{
'isolateId': isolate.id,
}));
}
} }
await Future.wait(futures); }
await Future.wait(operations);
// We are now running from source.
_runningFromSnapshot = false;
await _launchFromDevFS(mainPath + '.dill'); await _launchFromDevFS(mainPath + '.dill');
restartTimer.stop(); restartTimer.stop();
globals.printTrace('Hot restart performed in ${getElapsedAsMilliseconds(restartTimer.elapsed)}.'); globals.printTrace('Hot restart performed in ${getElapsedAsMilliseconds(restartTimer.elapsed)}.');
// We are now running from sources.
_runningFromSnapshot = false;
_addBenchmarkData('hotRestartMillisecondsToFrame', _addBenchmarkData('hotRestartMillisecondsToFrame',
restartTimer.elapsed.inMilliseconds); restartTimer.elapsed.inMilliseconds);
...@@ -734,10 +742,8 @@ class HotRunner extends ResidentRunner { ...@@ -734,10 +742,8 @@ class HotRunner extends ResidentRunner {
String reason, String reason,
bool pause, bool pause,
}) async { }) async {
final bool reloadOnTopOfSnapshot = _runningFromSnapshot;
final String progressPrefix = reloadOnTopOfSnapshot ? 'Initializing' : 'Performing';
Status status = globals.logger.startProgress( Status status = globals.logger.startProgress(
'$progressPrefix hot reload...', 'Performing hot reload...',
timeout: timeoutConfiguration.fastOperation, timeout: timeoutConfiguration.fastOperation,
progressId: 'hot.reload', progressId: 'hot.reload',
); );
...@@ -788,13 +794,6 @@ class HotRunner extends ResidentRunner { ...@@ -788,13 +794,6 @@ class HotRunner extends ResidentRunner {
} }
} }
// The initial launch is from a script snapshot. When we reload from source
// on top of a script snapshot, the first reload will be a worst case reload
// because all of the sources will end up being dirty (library paths will
// change from host path to a device path). Subsequent reloads will
// not be affected, so we resume reporting reload times on the second
// reload.
bool shouldReportReloadTime = !_runningFromSnapshot;
final Stopwatch reloadTimer = Stopwatch()..start(); final Stopwatch reloadTimer = Stopwatch()..start();
if (!_isPaused()) { if (!_isPaused()) {
...@@ -805,6 +804,7 @@ class HotRunner extends ResidentRunner { ...@@ -805,6 +804,7 @@ class HotRunner extends ResidentRunner {
final Stopwatch devFSTimer = Stopwatch()..start(); final Stopwatch devFSTimer = Stopwatch()..start();
final UpdateFSReport updatedDevFS = await _updateDevFS(); final UpdateFSReport updatedDevFS = await _updateDevFS();
// Record time it took to synchronize to DevFS. // Record time it took to synchronize to DevFS.
bool shouldReportReloadTime = true;
_addBenchmarkData('hotReloadDevFSSyncMilliseconds', devFSTimer.elapsed.inMilliseconds); _addBenchmarkData('hotReloadDevFSSyncMilliseconds', devFSTimer.elapsed.inMilliseconds);
if (!updatedDevFS.success) { if (!updatedDevFS.success) {
return OperationResult(1, 'DevFS synchronization failed'); return OperationResult(1, 'DevFS synchronization failed');
...@@ -819,10 +819,11 @@ class HotRunner extends ResidentRunner { ...@@ -819,10 +819,11 @@ class HotRunner extends ResidentRunner {
); );
final List<Future<DeviceReloadReport>> allReportsFutures = <Future<DeviceReloadReport>>[]; final List<Future<DeviceReloadReport>> allReportsFutures = <Future<DeviceReloadReport>>[];
for (final FlutterDevice device in flutterDevices) { for (final FlutterDevice device in flutterDevices) {
if (_runningFromSnapshot) { if (_shouldResetAssetDirectory) {
// Asset directory has to be set only once when we switch from // Asset directory has to be set only once when we switch from
// running from snapshot to running from uploaded files. // running from bundle to uploaded files.
await device.resetAssetDirectory(); await device.resetAssetDirectory();
_shouldResetAssetDirectory = false;
} }
final List<Future<Map<String, dynamic>>> reportFutures = device.reloadSources( final List<Future<Map<String, dynamic>>> reportFutures = device.reloadSources(
entryPath, pause: pause, entryPath, pause: pause,
...@@ -905,8 +906,6 @@ class HotRunner extends ResidentRunner { ...@@ -905,8 +906,6 @@ class HotRunner extends ResidentRunner {
} }
await Future.wait(allDevices); await Future.wait(allDevices);
// We are now running from source.
_runningFromSnapshot = false;
// Check if any isolates are paused. // Check if any isolates are paused.
final List<FlutterView> reassembleViews = <FlutterView>[]; final List<FlutterView> reassembleViews = <FlutterView>[];
String serviceEventKind; String serviceEventKind;
......
...@@ -92,6 +92,9 @@ void main() { ...@@ -92,6 +92,9 @@ void main() {
]); ]);
when(mockFlutterDevice.device).thenReturn(mockDevice); when(mockFlutterDevice.device).thenReturn(mockDevice);
when(mockFlutterView.uiIsolate).thenReturn(mockIsolate); when(mockFlutterView.uiIsolate).thenReturn(mockIsolate);
final MockVM mockVM = MockVM();
when(mockVMService.vm).thenReturn(mockVM);
when(mockVM.isolates).thenReturn(<Isolate>[mockIsolate]);
when(mockFlutterView.runFromSource(any, any, any)).thenAnswer((Invocation invocation) async {}); when(mockFlutterView.runFromSource(any, any, any)).thenAnswer((Invocation invocation) async {});
when(mockFlutterDevice.stopEchoingDeviceLog()).thenAnswer((Invocation invocation) async { }); when(mockFlutterDevice.stopEchoingDeviceLog()).thenAnswer((Invocation invocation) async { });
when(mockFlutterDevice.observatoryUris).thenAnswer((_) => Stream<Uri>.value(testUri)); when(mockFlutterDevice.observatoryUris).thenAnswer((_) => Stream<Uri>.value(testUri));
...@@ -744,6 +747,7 @@ class MockDevicePortForwarder extends Mock implements DevicePortForwarder {} ...@@ -744,6 +747,7 @@ class MockDevicePortForwarder extends Mock implements DevicePortForwarder {}
class MockUsage extends Mock implements Usage {} class MockUsage extends Mock implements Usage {}
class MockProcessManager extends Mock implements ProcessManager {} class MockProcessManager extends Mock implements ProcessManager {}
class MockServiceEvent extends Mock implements ServiceEvent {} class MockServiceEvent extends Mock implements ServiceEvent {}
class MockVM extends Mock implements VM {}
class TestFlutterDevice extends FlutterDevice { class TestFlutterDevice extends FlutterDevice {
TestFlutterDevice(Device device, this.views, { Stream<Uri> observatoryUris }) TestFlutterDevice(Device device, this.views, { Stream<Uri> observatoryUris })
: super(device, buildInfo: BuildInfo.debug) { : super(device, buildInfo: BuildInfo.debug) {
......
// Copyright 2014 The Flutter Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
import 'dart:async';
import 'package:file/file.dart';
import 'package:flutter_tools/src/base/file_system.dart';
import '../src/common.dart';
import 'test_data/background_project.dart';
import 'test_driver.dart';
import 'test_utils.dart';
void main() {
Directory tempDir;
setUp(() async {
tempDir = createResolvedTempDirectorySync('hot_reload_test.');
});
tearDown(() async {
tryToDelete(tempDir);
});
test('Hot restart kills background isolates', () async {
final BackgroundProject project = BackgroundProject();
await project.setUpIn(tempDir);
final FlutterRunTestDriver flutter = FlutterRunTestDriver(tempDir);
const String newBackgroundMessage = 'New Background';
final Completer<void> sawForgroundMessage = Completer<void>.sync();
final Completer<void> sawBackgroundMessage = Completer<void>.sync();
final Completer<void> sawNewBackgroundMessage = Completer<void>.sync();
final StreamSubscription<String> subscription = flutter.stdout.listen((String line) {
print('[LOG]:"$line"');
if (line.contains('Main thread') && !sawForgroundMessage.isCompleted) {
sawForgroundMessage.complete();
}
if (line.contains('Isolate thread')) {
sawBackgroundMessage.complete();
}
if (line.contains(newBackgroundMessage)) {
sawNewBackgroundMessage.complete();
}
},
);
await flutter.run();
await sawForgroundMessage.future;
await sawBackgroundMessage.future;
project.updateTestIsolatePhrase(newBackgroundMessage);
await flutter.hotRestart();
await sawBackgroundMessage.future;
// Wait a tiny amount of time in case we did not kill the background isolate.
await Future<void>.delayed(const Duration(milliseconds: 10));
await subscription.cancel();
await flutter?.stop();
});
test('Hot reload updates background isolates', () async {
final RepeatingBackgroundProject project = RepeatingBackgroundProject();
await project.setUpIn(tempDir);
final FlutterRunTestDriver flutter = FlutterRunTestDriver(tempDir);
const String newBackgroundMessage = 'New Background';
final Completer<void> sawBackgroundMessage = Completer<void>.sync();
final Completer<void> sawNewBackgroundMessage = Completer<void>.sync();
final StreamSubscription<String> subscription = flutter.stdout.listen((String line) {
print('[LOG]:"$line"');
if (line.contains('Isolate thread') && !sawBackgroundMessage.isCompleted) {
sawBackgroundMessage.complete();
}
if (line.contains(newBackgroundMessage) && !sawNewBackgroundMessage.isCompleted) {
sawNewBackgroundMessage.complete();
}
},
);
await flutter.run();
await sawBackgroundMessage.future;
project.updateTestIsolatePhrase(newBackgroundMessage);
await flutter.hotReload();
await sawNewBackgroundMessage.future;
await subscription.cancel();
await flutter?.stop();
});
}
// Copyright 2014 The Flutter Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
import 'package:flutter_tools/src/globals.dart' as globals;
import '../test_utils.dart';
import 'project.dart';
/// Spawns a background isolate that prints a debug message.
class BackgroundProject extends Project {
@override
final String pubspec = '''
name: test
environment:
sdk: ">=2.0.0-dev.68.0 <3.0.0"
dependencies:
flutter:
sdk: flutter
''';
@override
final String main = r'''
import 'dart:async';
import 'dart:isolate';
import 'package:flutter/widgets.dart';
import 'package:flutter/material.dart';
void main() {
Isolate.spawn<void>(background, null, debugName: 'background');
TestMain();
}
void background(void message) {
TestIsolate();
}
class TestMain {
TestMain() {
debugPrint('Main thread');
}
}
class TestIsolate {
TestIsolate() {
debugPrint('Isolate thread');
}
}
''';
void updateTestIsolatePhrase(String message) {
final String newMainContents = main.replaceFirst('Isolate thread', message);
writeFile(globals.fs.path.join(dir.path, 'lib', 'main.dart'), newMainContents);
}
}
// Spawns a background isolate that repeats a message.
class RepeatingBackgroundProject extends Project {
@override
final String pubspec = '''
name: test
environment:
sdk: ">=2.0.0-dev.68.0 <3.0.0"
dependencies:
flutter:
sdk: flutter
''';
@override
final String main = r'''
import 'dart:async';
import 'dart:isolate';
import 'package:flutter/widgets.dart';
import 'package:flutter/material.dart';
void main() {
Isolate.spawn<void>(background, null, debugName: 'background');
TestMain();
}
void background(void message) {
Timer.periodic(const Duration(milliseconds: 500), (Timer timer) => TestIsolate());
}
class TestMain {
TestMain() {
debugPrint('Main thread');
}
}
class TestIsolate {
TestIsolate() {
debugPrint('Isolate thread');
}
}
''';
void updateTestIsolatePhrase(String message) {
final String newMainContents = main.replaceFirst('Isolate thread', message);
writeFile(globals.fs.path.join(dir.path, 'lib', 'main.dart'), newMainContents);
}
}
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment