2018-07-10 05:51:12 +00:00
|
|
|
// Copyright 2018 The Chromium Authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
|
|
// found in the LICENSE file.
|
|
|
|
|
2018-10-30 16:59:49 +00:00
|
|
|
import 'dart:async';
|
|
|
|
|
2018-07-10 05:51:12 +00:00
|
|
|
import 'package:file/file.dart';
|
2019-02-05 23:00:51 +00:00
|
|
|
import 'package:flutter_tools/src/base/common.dart';
|
2018-07-10 05:51:12 +00:00
|
|
|
import 'package:flutter_tools/src/base/file_system.dart';
|
2019-08-09 16:58:49 +00:00
|
|
|
import 'package:vm_service/vm_service.dart';
|
2018-07-10 05:51:12 +00:00
|
|
|
|
2018-08-15 03:33:58 +00:00
|
|
|
import '../src/common.dart';
|
2018-10-30 16:59:49 +00:00
|
|
|
import 'test_data/hot_reload_project.dart';
|
2018-07-10 05:51:12 +00:00
|
|
|
import 'test_driver.dart';
|
2018-10-19 11:51:31 +00:00
|
|
|
import 'test_utils.dart';
|
2018-07-10 05:51:12 +00:00
|
|
|
|
|
|
|
void main() {
|
2019-10-18 23:35:39 +00:00
|
|
|
Directory tempDir;
|
|
|
|
final HotReloadProject _project = HotReloadProject();
|
|
|
|
FlutterRunTestDriver _flutter;
|
2018-08-17 20:17:23 +00:00
|
|
|
|
2019-10-18 23:35:39 +00:00
|
|
|
setUp(() async {
|
|
|
|
tempDir = createResolvedTempDirectorySync('hot_reload_test.');
|
|
|
|
await _project.setUpIn(tempDir);
|
|
|
|
_flutter = FlutterRunTestDriver(tempDir);
|
|
|
|
});
|
2018-07-10 05:51:12 +00:00
|
|
|
|
2019-10-18 23:35:39 +00:00
|
|
|
tearDown(() async {
|
|
|
|
await _flutter?.stop();
|
|
|
|
tryToDelete(tempDir);
|
|
|
|
});
|
2018-07-10 05:51:12 +00:00
|
|
|
|
2019-10-18 23:35:39 +00:00
|
|
|
test('hot reload works without error', () async {
|
|
|
|
await _flutter.run();
|
|
|
|
await _flutter.hotReload();
|
|
|
|
});
|
|
|
|
|
|
|
|
test('newly added code executes during hot reload', () async {
|
|
|
|
await _flutter.run();
|
|
|
|
_project.uncommentHotReloadPrint();
|
|
|
|
final StringBuffer stdout = StringBuffer();
|
|
|
|
final StreamSubscription<String> subscription = _flutter.stdout.listen(stdout.writeln);
|
|
|
|
try {
|
2018-08-01 16:50:38 +00:00
|
|
|
await _flutter.hotReload();
|
2019-10-18 23:35:39 +00:00
|
|
|
expect(stdout.toString(), contains('(((((RELOAD WORKED)))))'));
|
|
|
|
} finally {
|
|
|
|
await subscription.cancel();
|
|
|
|
}
|
|
|
|
});
|
2018-07-10 05:51:12 +00:00
|
|
|
|
2019-10-18 23:35:39 +00:00
|
|
|
test('hot restart works without error', () async {
|
|
|
|
await _flutter.run();
|
|
|
|
await _flutter.hotRestart();
|
|
|
|
});
|
2018-10-30 16:59:49 +00:00
|
|
|
|
2019-10-18 23:35:39 +00:00
|
|
|
test('breakpoints are hit after hot reload', () async {
|
|
|
|
Isolate isolate;
|
|
|
|
await _flutter.run(withDebugger: true, startPaused: true);
|
|
|
|
final Completer<void> sawTick1 = Completer<void>();
|
|
|
|
final Completer<void> sawTick3 = Completer<void>();
|
|
|
|
final Completer<void> sawDebuggerPausedMessage = Completer<void>();
|
|
|
|
final StreamSubscription<String> subscription = _flutter.stdout.listen(
|
|
|
|
(String line) {
|
|
|
|
if (line.contains('((((TICK 1))))')) {
|
|
|
|
expect(sawTick1.isCompleted, isFalse);
|
|
|
|
sawTick1.complete();
|
|
|
|
}
|
|
|
|
if (line.contains('((((TICK 3))))')) {
|
|
|
|
expect(sawTick3.isCompleted, isFalse);
|
|
|
|
sawTick3.complete();
|
|
|
|
}
|
|
|
|
if (line.contains('The application is paused in the debugger on a breakpoint.')) {
|
|
|
|
expect(sawDebuggerPausedMessage.isCompleted, isFalse);
|
|
|
|
sawDebuggerPausedMessage.complete();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
);
|
|
|
|
await _flutter.resume(); // we start paused so we can set up our TICK 1 listener before the app starts
|
|
|
|
unawaited(sawTick1.future.timeout(
|
|
|
|
const Duration(seconds: 5),
|
|
|
|
onTimeout: () { print('The test app is taking longer than expected to print its synchronization line...'); },
|
|
|
|
));
|
|
|
|
await sawTick1.future; // after this, app is in steady state
|
|
|
|
await _flutter.addBreakpoint(
|
|
|
|
_project.scheduledBreakpointUri,
|
|
|
|
_project.scheduledBreakpointLine,
|
|
|
|
);
|
|
|
|
await _flutter.hotReload(); // reload triggers code which eventually hits the breakpoint
|
|
|
|
isolate = await _flutter.waitForPause();
|
|
|
|
expect(isolate.pauseEvent.kind, equals(EventKind.kPauseBreakpoint));
|
|
|
|
await _flutter.resume();
|
|
|
|
await _flutter.addBreakpoint(
|
|
|
|
_project.buildBreakpointUri,
|
|
|
|
_project.buildBreakpointLine,
|
|
|
|
);
|
|
|
|
bool reloaded = false;
|
|
|
|
final Future<void> reloadFuture = _flutter.hotReload().then((void value) { reloaded = true; });
|
|
|
|
await sawTick3.future; // this should happen before it pauses
|
|
|
|
isolate = await _flutter.waitForPause();
|
|
|
|
expect(isolate.pauseEvent.kind, equals(EventKind.kPauseBreakpoint));
|
|
|
|
await sawDebuggerPausedMessage.future;
|
|
|
|
expect(reloaded, isFalse);
|
|
|
|
await _flutter.resume();
|
|
|
|
await reloadFuture;
|
|
|
|
expect(reloaded, isTrue);
|
|
|
|
reloaded = false;
|
|
|
|
await subscription.cancel();
|
|
|
|
});
|
2018-09-04 15:24:10 +00:00
|
|
|
|
2019-10-18 23:35:39 +00:00
|
|
|
test('hot reload doesn\'t reassemble if paused', () async {
|
2019-11-22 20:26:31 +00:00
|
|
|
final Future<void> setup = _flutter.run(withDebugger: true);
|
|
|
|
final Completer<void> sawTick1 = Completer<void>();
|
2019-10-18 23:35:39 +00:00
|
|
|
final Completer<void> sawTick2 = Completer<void>();
|
|
|
|
final Completer<void> sawTick3 = Completer<void>();
|
|
|
|
final Completer<void> sawDebuggerPausedMessage1 = Completer<void>();
|
|
|
|
final Completer<void> sawDebuggerPausedMessage2 = Completer<void>();
|
|
|
|
final StreamSubscription<String> subscription = _flutter.stdout.listen(
|
|
|
|
(String line) {
|
2019-11-22 20:26:31 +00:00
|
|
|
if (line.contains('((((TICK 1))))')) {
|
|
|
|
expect(sawTick1.isCompleted, isFalse);
|
|
|
|
sawTick1.complete();
|
|
|
|
}
|
2019-10-18 23:35:39 +00:00
|
|
|
if (line.contains('((((TICK 2))))')) {
|
|
|
|
expect(sawTick2.isCompleted, isFalse);
|
|
|
|
sawTick2.complete();
|
|
|
|
}
|
|
|
|
if (line.contains('The application is paused in the debugger on a breakpoint.')) {
|
|
|
|
expect(sawDebuggerPausedMessage1.isCompleted, isFalse);
|
|
|
|
sawDebuggerPausedMessage1.complete();
|
|
|
|
}
|
|
|
|
if (line.contains('The application is paused in the debugger on a breakpoint; interface might not update.')) {
|
|
|
|
expect(sawDebuggerPausedMessage2.isCompleted, isFalse);
|
|
|
|
sawDebuggerPausedMessage2.complete();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
);
|
2019-11-22 20:26:31 +00:00
|
|
|
await setup;
|
|
|
|
await sawTick1.future;
|
2019-10-18 23:35:39 +00:00
|
|
|
await _flutter.addBreakpoint(
|
|
|
|
_project.buildBreakpointUri,
|
|
|
|
_project.buildBreakpointLine,
|
|
|
|
);
|
|
|
|
bool reloaded = false;
|
|
|
|
final Future<void> reloadFuture = _flutter.hotReload().then((void value) { reloaded = true; });
|
|
|
|
await sawTick2.future; // this should happen before it pauses
|
|
|
|
final Isolate isolate = await _flutter.waitForPause();
|
|
|
|
expect(isolate.pauseEvent.kind, equals(EventKind.kPauseBreakpoint));
|
|
|
|
expect(reloaded, isFalse);
|
|
|
|
await sawDebuggerPausedMessage1.future; // this is the one where it say "uh, you broke into the debugger while reloading"
|
|
|
|
await reloadFuture; // this is the one where it times out because you're in the debugger
|
|
|
|
expect(reloaded, isTrue);
|
|
|
|
await _flutter.hotReload(); // now we're already paused
|
|
|
|
expect(sawTick3.isCompleted, isFalse);
|
|
|
|
await sawDebuggerPausedMessage2.future; // so we just get told that nothing is going to happen
|
|
|
|
await _flutter.resume();
|
|
|
|
await subscription.cancel();
|
|
|
|
});
|
2018-07-10 05:51:12 +00:00
|
|
|
}
|