Unverified Commit 2daac920 authored by Jonah Williams's avatar Jonah Williams Committed by GitHub

[flutter_tools] simplify shutdown hooks (#77969)

parent 66768f8c
...@@ -347,9 +347,6 @@ abstract class IOSApp extends ApplicationPackage { ...@@ -347,9 +347,6 @@ abstract class IOSApp extends ApplicationPackage {
} else { } else {
// Try to unpack as an ipa. // Try to unpack as an ipa.
final Directory tempDir = globals.fs.systemTempDirectory.createTempSync('flutter_app.'); final Directory tempDir = globals.fs.systemTempDirectory.createTempSync('flutter_app.');
shutdownHooks.addShutdownHook(() async {
await tempDir.delete(recursive: true);
}, ShutdownStage.STILL_RECORDING);
globals.os.unzip(globals.fs.file(applicationBinary), tempDir); globals.os.unzip(globals.fs.file(applicationBinary), tempDir);
final Directory payloadDir = globals.fs.directory( final Directory payloadDir = globals.fs.directory(
globals.fs.path.join(tempDir.path, 'Payload'), globals.fs.path.join(tempDir.path, 'Payload'),
......
...@@ -235,7 +235,6 @@ class LocalFileSystem extends local_fs.LocalFileSystem { ...@@ -235,7 +235,6 @@ class LocalFileSystem extends local_fs.LocalFileSystem {
// exits normally. // exits normally.
shutdownHooks?.addShutdownHook( shutdownHooks?.addShutdownHook(
_tryToDeleteTemp, _tryToDeleteTemp,
ShutdownStage.CLEANUP,
); );
} }
return _systemTemp; return _systemTemp;
......
...@@ -26,35 +26,6 @@ typedef ShutdownHook = FutureOr<dynamic> Function(); ...@@ -26,35 +26,6 @@ typedef ShutdownHook = FutureOr<dynamic> Function();
// See [here](https://github.com/flutter/flutter/pull/14535#discussion_r167041161) // See [here](https://github.com/flutter/flutter/pull/14535#discussion_r167041161)
// for more details. // for more details.
/// The stage in which a [ShutdownHook] will be run. All shutdown hooks within
/// a given stage will be started in parallel and will be guaranteed to run to
/// completion before shutdown hooks in the next stage are started.
class ShutdownStage implements Comparable<ShutdownStage> {
const ShutdownStage._(this.priority);
/// The stage priority. Smaller values will be run before larger values.
final int priority;
/// The stage before the invocation recording (if one exists) is serialized
/// to disk. Tasks performed during this stage *will* be recorded.
static const ShutdownStage STILL_RECORDING = ShutdownStage._(1);
/// The stage during which the invocation recording (if one exists) will be
/// serialized to disk. Invocations performed after this stage will not be
/// recorded.
static const ShutdownStage SERIALIZE_RECORDING = ShutdownStage._(2);
/// The stage during which a serialized recording will be refined (e.g.
/// cleansed for tests, zipped up for bug reporting purposes, etc.).
static const ShutdownStage POST_PROCESS_RECORDING = ShutdownStage._(3);
/// The stage during which temporary files and directories will be deleted.
static const ShutdownStage CLEANUP = ShutdownStage._(4);
@override
int compareTo(ShutdownStage other) => priority.compareTo(other.priority);
}
ShutdownHooks get shutdownHooks => ShutdownHooks.instance; ShutdownHooks get shutdownHooks => ShutdownHooks.instance;
abstract class ShutdownHooks { abstract class ShutdownHooks {
...@@ -67,14 +38,9 @@ abstract class ShutdownHooks { ...@@ -67,14 +38,9 @@ abstract class ShutdownHooks {
static ShutdownHooks get instance => context.get<ShutdownHooks>(); static ShutdownHooks get instance => context.get<ShutdownHooks>();
/// Registers a [ShutdownHook] to be executed before the VM exits. /// Registers a [ShutdownHook] to be executed before the VM exits.
///
/// If [stage] is specified, the shutdown hook will be run during the specified
/// stage. By default, the shutdown hook will be run during the
/// [ShutdownStage.CLEANUP] stage.
void addShutdownHook( void addShutdownHook(
ShutdownHook shutdownHook, [ ShutdownHook shutdownHook
ShutdownStage stage = ShutdownStage.CLEANUP, );
]);
/// Runs all registered shutdown hooks and returns a future that completes when /// Runs all registered shutdown hooks and returns a future that completes when
/// all such hooks have finished. /// all such hooks have finished.
...@@ -92,18 +58,16 @@ class _DefaultShutdownHooks implements ShutdownHooks { ...@@ -92,18 +58,16 @@ class _DefaultShutdownHooks implements ShutdownHooks {
}) : _logger = logger; }) : _logger = logger;
final Logger _logger; final Logger _logger;
final List<ShutdownHook> _shutdownHooks = <ShutdownHook>[];
final Map<ShutdownStage, List<ShutdownHook>> _shutdownHooks = <ShutdownStage, List<ShutdownHook>>{};
bool _shutdownHooksRunning = false; bool _shutdownHooksRunning = false;
@override @override
void addShutdownHook( void addShutdownHook(
ShutdownHook shutdownHook, [ ShutdownHook shutdownHook
ShutdownStage stage = ShutdownStage.CLEANUP, ) {
]) {
assert(!_shutdownHooksRunning); assert(!_shutdownHooksRunning);
_shutdownHooks.putIfAbsent(stage, () => <ShutdownHook>[]).add(shutdownHook); _shutdownHooks.add(shutdownHook);
} }
@override @override
...@@ -111,22 +75,17 @@ class _DefaultShutdownHooks implements ShutdownHooks { ...@@ -111,22 +75,17 @@ class _DefaultShutdownHooks implements ShutdownHooks {
_logger.printTrace('Running shutdown hooks'); _logger.printTrace('Running shutdown hooks');
_shutdownHooksRunning = true; _shutdownHooksRunning = true;
try { try {
for (final ShutdownStage stage in _shutdownHooks.keys.toList()..sort()) {
_logger.printTrace('Shutdown hook priority ${stage.priority}');
final List<ShutdownHook> hooks = _shutdownHooks.remove(stage);
final List<Future<dynamic>> futures = <Future<dynamic>>[]; final List<Future<dynamic>> futures = <Future<dynamic>>[];
for (final ShutdownHook shutdownHook in hooks) { for (final ShutdownHook shutdownHook in _shutdownHooks) {
final FutureOr<dynamic> result = shutdownHook(); final FutureOr<dynamic> result = shutdownHook();
if (result is Future<dynamic>) { if (result is Future<dynamic>) {
futures.add(result); futures.add(result);
} }
} }
await Future.wait<dynamic>(futures); await Future.wait<dynamic>(futures);
}
} finally { } finally {
_shutdownHooksRunning = false; _shutdownHooksRunning = false;
} }
assert(_shutdownHooks.isEmpty);
_logger.printTrace('Shutdown hooks complete'); _logger.printTrace('Shutdown hooks complete');
} }
} }
......
...@@ -48,35 +48,17 @@ void main() { ...@@ -48,35 +48,17 @@ void main() {
group('shutdownHooks', () { group('shutdownHooks', () {
testWithoutContext('runInExpectedOrder', () async { testWithoutContext('runInExpectedOrder', () async {
int i = 1; int i = 1;
int serializeRecording1;
int serializeRecording2;
int postProcessRecording;
int cleanup; int cleanup;
final ShutdownHooks shutdownHooks = ShutdownHooks(logger: BufferLogger.test()); final ShutdownHooks shutdownHooks = ShutdownHooks(logger: BufferLogger.test());
shutdownHooks.addShutdownHook(() async {
serializeRecording1 = i++;
}, ShutdownStage.SERIALIZE_RECORDING);
shutdownHooks.addShutdownHook(() async { shutdownHooks.addShutdownHook(() async {
cleanup = i++; cleanup = i++;
}, ShutdownStage.CLEANUP); });
shutdownHooks.addShutdownHook(() async {
postProcessRecording = i++;
}, ShutdownStage.POST_PROCESS_RECORDING);
shutdownHooks.addShutdownHook(() async {
serializeRecording2 = i++;
}, ShutdownStage.SERIALIZE_RECORDING);
await shutdownHooks.runShutdownHooks(); await shutdownHooks.runShutdownHooks();
expect(serializeRecording1, lessThanOrEqualTo(2)); expect(cleanup, 1);
expect(serializeRecording2, lessThanOrEqualTo(2));
expect(postProcessRecording, 3);
expect(cleanup, 4);
}); });
}); });
......
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