Ian Hickson | 449f4a6 | 2019-11-27 15:04:02 -0800 | [diff] [blame] | 1 | // Copyright 2014 The Flutter Authors. All rights reserved. |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 2 | // Use of this source code is governed by a BSD-style license that can be |
| 3 | // found in the LICENSE file. |
| 4 | |
Jonah Williams | 9203448 | 2022-06-15 13:02:07 -0700 | [diff] [blame] | 5 | |
Jonah Williams | 74bd7b6 | 2021-01-27 15:17:53 -0800 | [diff] [blame] | 6 | |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 7 | import 'dart:async'; |
| 8 | |
| 9 | import 'package:args/command_runner.dart'; |
Devon Carew | 1c6078c | 2018-05-24 13:42:46 -0700 | [diff] [blame] | 10 | import 'package:intl/intl.dart' as intl; |
| 11 | import 'package:intl/intl_standalone.dart' as intl_standalone; |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 12 | |
Jonah Williams | b1e9918 | 2021-01-13 14:14:07 -0800 | [diff] [blame] | 13 | import 'src/base/async_guard.dart'; |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 14 | import 'src/base/common.dart'; |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 15 | import 'src/base/context.dart'; |
| 16 | import 'src/base/file_system.dart'; |
| 17 | import 'src/base/io.dart'; |
| 18 | import 'src/base/logger.dart'; |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 19 | import 'src/base/process.dart'; |
Todd Volkert | 8d11f5c | 2018-03-28 10:58:28 -0700 | [diff] [blame] | 20 | import 'src/context_runner.dart'; |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 21 | import 'src/doctor.dart'; |
Jenn Magder | 868e417 | 2021-11-03 12:53:04 -0700 | [diff] [blame] | 22 | import 'src/globals.dart' as globals; |
Jenn Magder | e29595d | 2021-04-09 15:24:15 -0700 | [diff] [blame] | 23 | import 'src/reporting/crash_reporting.dart'; |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 24 | import 'src/runner/flutter_command.dart'; |
| 25 | import 'src/runner/flutter_command_runner.dart'; |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 26 | |
| 27 | /// Runs the Flutter tool with support for the specified list of [commands]. |
| 28 | Future<int> run( |
| 29 | List<String> args, |
Jonah Williams | bd3eee7 | 2020-09-03 11:55:55 -0700 | [diff] [blame] | 30 | List<FlutterCommand> Function() commands, { |
Jenn Magder | 8109dcc | 2020-04-20 15:23:09 -0700 | [diff] [blame] | 31 | bool muteCommandLogging = false, |
| 32 | bool verbose = false, |
| 33 | bool verboseHelp = false, |
Jonah Williams | 9203448 | 2022-06-15 13:02:07 -0700 | [diff] [blame] | 34 | bool? reportCrashes, |
| 35 | String? flutterVersion, |
| 36 | Map<Type, Generator>? overrides, |
Christopher Fujino | 08d5d51 | 2022-09-02 10:38:56 -0700 | [diff] [blame] | 37 | required ShutdownHooks shutdownHooks, |
Jenn Magder | 8109dcc | 2020-04-20 15:23:09 -0700 | [diff] [blame] | 38 | }) async { |
jcollins-g | 26102c9 | 2018-02-01 10:27:55 -0800 | [diff] [blame] | 39 | if (muteCommandLogging) { |
| 40 | // Remove the verbose option; for help and doctor, users don't need to see |
| 41 | // verbose logs. |
James D. Lin | 566c1d1 | 2020-04-21 22:09:50 -0700 | [diff] [blame] | 42 | args = List<String>.of(args); |
Greg Spencer | 52ae102 | 2021-11-10 16:13:04 -0800 | [diff] [blame] | 43 | args.removeWhere((String option) => option == '-vv' || option == '-v' || option == '--verbose'); |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 44 | } |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 45 | |
Todd Volkert | 8d11f5c | 2018-03-28 10:58:28 -0700 | [diff] [blame] | 46 | return runInContext<int>(() async { |
Jonah Williams | 2133343 | 2020-03-10 11:35:52 -0700 | [diff] [blame] | 47 | reportCrashes ??= !await globals.isRunningOnBot; |
Jonah Williams | f5de6aa | 2020-06-04 16:35:36 -0700 | [diff] [blame] | 48 | final FlutterCommandRunner runner = FlutterCommandRunner(verboseHelp: verboseHelp); |
Jonah Williams | bd3eee7 | 2020-09-03 11:55:55 -0700 | [diff] [blame] | 49 | commands().forEach(runner.addCommand); |
Jonah Williams | 2133343 | 2020-03-10 11:35:52 -0700 | [diff] [blame] | 50 | |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 51 | // Initialize the system locale. |
Devon Carew | 1c6078c | 2018-05-24 13:42:46 -0700 | [diff] [blame] | 52 | final String systemLocale = await intl_standalone.findSystemLocale(); |
| 53 | intl.Intl.defaultLocale = intl.Intl.verifiedLocale( |
| 54 | systemLocale, intl.NumberFormat.localeExists, |
Alexandre Ardhuin | 387f885 | 2019-03-01 08:17:55 +0100 | [diff] [blame] | 55 | onFailure: (String _) => 'en_US', |
Devon Carew | 1c6078c | 2018-05-24 13:42:46 -0700 | [diff] [blame] | 56 | ); |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 57 | |
Jenn Magder | 7f71562 | 2020-01-27 17:38:01 -0800 | [diff] [blame] | 58 | String getVersion() => flutterVersion ?? globals.flutterVersion.getVersionString(redactUnknownBranches: true); |
Jonah Williams | 9203448 | 2022-06-15 13:02:07 -0700 | [diff] [blame] | 59 | Object? firstError; |
| 60 | StackTrace? firstStackTrace; |
Michael Goderbauer | cb867bb | 2021-03-05 18:38:15 -0800 | [diff] [blame] | 61 | return runZoned<Future<int>>(() async { |
Alexander Aprelev | d917978 | 2019-06-10 09:52:22 -0700 | [diff] [blame] | 62 | try { |
| 63 | await runner.run(args); |
James D. Lin | c21b323 | 2020-06-15 09:35:04 -0700 | [diff] [blame] | 64 | |
| 65 | // Triggering [runZoned]'s error callback does not necessarily mean that |
Pierre-Louis | 0c2f7bc | 2022-09-02 06:00:58 +0200 | [diff] [blame] | 66 | // we stopped executing the body. See https://github.com/dart-lang/sdk/issues/42150. |
James D. Lin | c21b323 | 2020-06-15 09:35:04 -0700 | [diff] [blame] | 67 | if (firstError == null) { |
Christopher Fujino | 08d5d51 | 2022-09-02 10:38:56 -0700 | [diff] [blame] | 68 | return await _exit(0, shutdownHooks: shutdownHooks); |
James D. Lin | c21b323 | 2020-06-15 09:35:04 -0700 | [diff] [blame] | 69 | } |
| 70 | |
Pierre-Louis | 0c2f7bc | 2022-09-02 06:00:58 +0200 | [diff] [blame] | 71 | // We already hit some error, so don't return success. The error path |
James D. Lin | c21b323 | 2020-06-15 09:35:04 -0700 | [diff] [blame] | 72 | // (which should be in progress) is responsible for calling _exit(). |
| 73 | return 1; |
Ian Hickson | 299d484 | 2021-10-14 22:03:03 -0700 | [diff] [blame] | 74 | } catch (error, stackTrace) { // ignore: avoid_catches_without_on_clauses |
| 75 | // This catches all exceptions to send to crash logging, etc. |
Zachary Anderson | c7596da | 2019-07-31 13:51:19 -0700 | [diff] [blame] | 76 | firstError = error; |
| 77 | firstStackTrace = stackTrace; |
Christopher Fujino | 08d5d51 | 2022-09-02 10:38:56 -0700 | [diff] [blame] | 78 | return _handleToolError(error, stackTrace, verbose, args, reportCrashes!, getVersion, shutdownHooks); |
Alexander Aprelev | d917978 | 2019-06-10 09:52:22 -0700 | [diff] [blame] | 79 | } |
Lasse R.H. Nielsen | af5194d | 2020-03-24 16:34:17 +0100 | [diff] [blame] | 80 | }, onError: (Object error, StackTrace stackTrace) async { // ignore: deprecated_member_use |
Zachary Anderson | c7596da | 2019-07-31 13:51:19 -0700 | [diff] [blame] | 81 | // If sending a crash report throws an error into the zone, we don't want |
| 82 | // to re-try sending the crash report with *that* error. Rather, we want |
| 83 | // to send the original error that triggered the crash report. |
James D. Lin | c21b323 | 2020-06-15 09:35:04 -0700 | [diff] [blame] | 84 | firstError ??= error; |
| 85 | firstStackTrace ??= stackTrace; |
Christopher Fujino | 08d5d51 | 2022-09-02 10:38:56 -0700 | [diff] [blame] | 86 | await _handleToolError(firstError!, firstStackTrace, verbose, args, reportCrashes!, getVersion, shutdownHooks); |
Alexander Aprelev | d917978 | 2019-06-10 09:52:22 -0700 | [diff] [blame] | 87 | }); |
Jonah Williams | 81c7af3 | 2018-11-06 14:36:35 -0800 | [diff] [blame] | 88 | }, overrides: overrides); |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 89 | } |
| 90 | |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 91 | Future<int> _handleToolError( |
Jonah Williams | 9203448 | 2022-06-15 13:02:07 -0700 | [diff] [blame] | 92 | Object error, |
| 93 | StackTrace? stackTrace, |
Alexandre Ardhuin | 5169ab5 | 2019-02-21 09:27:07 +0100 | [diff] [blame] | 94 | bool verbose, |
| 95 | List<String> args, |
| 96 | bool reportCrashes, |
Michael Goderbauer | 7b251f5 | 2021-03-04 08:59:17 -0800 | [diff] [blame] | 97 | String Function() getFlutterVersion, |
Christopher Fujino | 08d5d51 | 2022-09-02 10:38:56 -0700 | [diff] [blame] | 98 | ShutdownHooks shutdownHooks, |
Alexandre Ardhuin | 5169ab5 | 2019-02-21 09:27:07 +0100 | [diff] [blame] | 99 | ) async { |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 100 | if (error is UsageException) { |
Zachary Anderson | 23a3d10 | 2020-01-13 10:12:06 -0800 | [diff] [blame] | 101 | globals.printError('${error.message}\n'); |
| 102 | globals.printError("Run 'flutter -h' (or 'flutter <command> -h') for available flutter commands and options."); |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 103 | // Argument error exit code. |
Christopher Fujino | 08d5d51 | 2022-09-02 10:38:56 -0700 | [diff] [blame] | 104 | return _exit(64, shutdownHooks: shutdownHooks); |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 105 | } else if (error is ToolExit) { |
Zachary Anderson | e2340c6 | 2019-09-13 14:51:35 -0700 | [diff] [blame] | 106 | if (error.message != null) { |
Jonah Williams | 9203448 | 2022-06-15 13:02:07 -0700 | [diff] [blame] | 107 | globals.printError(error.message!); |
Zachary Anderson | e2340c6 | 2019-09-13 14:51:35 -0700 | [diff] [blame] | 108 | } |
| 109 | if (verbose) { |
Zachary Anderson | 23a3d10 | 2020-01-13 10:12:06 -0800 | [diff] [blame] | 110 | globals.printError('\n$stackTrace\n'); |
Zachary Anderson | e2340c6 | 2019-09-13 14:51:35 -0700 | [diff] [blame] | 111 | } |
Christopher Fujino | 08d5d51 | 2022-09-02 10:38:56 -0700 | [diff] [blame] | 112 | return _exit(error.exitCode ?? 1, shutdownHooks: shutdownHooks); |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 113 | } else if (error is ProcessExit) { |
| 114 | // We've caught an exit code. |
| 115 | if (error.immediate) { |
| 116 | exit(error.exitCode); |
| 117 | return error.exitCode; |
| 118 | } else { |
Christopher Fujino | 08d5d51 | 2022-09-02 10:38:56 -0700 | [diff] [blame] | 119 | return _exit(error.exitCode, shutdownHooks: shutdownHooks); |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 120 | } |
| 121 | } else { |
| 122 | // We've crashed; emit a log report. |
Zachary Anderson | 68ed5c2 | 2020-01-28 07:58:02 -0800 | [diff] [blame] | 123 | globals.stdio.stderrWrite('\n'); |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 124 | |
| 125 | if (!reportCrashes) { |
| 126 | // Print the stack trace on the bots - don't write a crash report. |
Zachary Anderson | 68ed5c2 | 2020-01-28 07:58:02 -0800 | [diff] [blame] | 127 | globals.stdio.stderrWrite('$error\n'); |
| 128 | globals.stdio.stderrWrite('$stackTrace\n'); |
Christopher Fujino | 08d5d51 | 2022-09-02 10:38:56 -0700 | [diff] [blame] | 129 | return _exit(1, shutdownHooks: shutdownHooks); |
Zachary Anderson | 36e8b93 | 2019-08-21 13:07:52 -0700 | [diff] [blame] | 130 | } |
| 131 | |
| 132 | // Report to both [Usage] and [CrashReportSender]. |
Jenn Magder | f6a5512 | 2020-03-06 12:14:29 -0800 | [diff] [blame] | 133 | globals.flutterUsage.sendException(error); |
Jonah Williams | b1e9918 | 2021-01-13 14:14:07 -0800 | [diff] [blame] | 134 | await asyncGuard(() async { |
| 135 | final CrashReportSender crashReportSender = CrashReportSender( |
Jonah Williams | b1e9918 | 2021-01-13 14:14:07 -0800 | [diff] [blame] | 136 | usage: globals.flutterUsage, |
| 137 | platform: globals.platform, |
| 138 | logger: globals.logger, |
| 139 | operatingSystemUtils: globals.os, |
| 140 | ); |
| 141 | await crashReportSender.sendReport( |
| 142 | error: error, |
Jonah Williams | 9203448 | 2022-06-15 13:02:07 -0700 | [diff] [blame] | 143 | stackTrace: stackTrace!, |
Jonah Williams | b1e9918 | 2021-01-13 14:14:07 -0800 | [diff] [blame] | 144 | getFlutterVersion: getFlutterVersion, |
| 145 | command: args.join(' '), |
| 146 | ); |
| 147 | }, onError: (dynamic error) { |
| 148 | globals.printError('Error sending crash report: $error'); |
| 149 | }); |
Zachary Anderson | 36e8b93 | 2019-08-21 13:07:52 -0700 | [diff] [blame] | 150 | |
James D. Lin | b7fd24a | 2020-04-28 10:34:03 -0700 | [diff] [blame] | 151 | globals.printError('Oops; flutter has exited unexpectedly: "$error".'); |
Zachary Anderson | 36e8b93 | 2019-08-21 13:07:52 -0700 | [diff] [blame] | 152 | |
| 153 | try { |
Jenn Magder | 1c0eade | 2022-01-10 13:20:21 -0800 | [diff] [blame] | 154 | final BufferLogger logger = BufferLogger( |
| 155 | terminal: globals.terminal, |
| 156 | outputPreferences: globals.outputPreferences, |
| 157 | ); |
| 158 | |
| 159 | final DoctorText doctorText = DoctorText(logger); |
| 160 | |
James D. Lin | b7fd24a | 2020-04-28 10:34:03 -0700 | [diff] [blame] | 161 | final CrashDetails details = CrashDetails( |
| 162 | command: _crashCommand(args), |
| 163 | error: error, |
Jonah Williams | 9203448 | 2022-06-15 13:02:07 -0700 | [diff] [blame] | 164 | stackTrace: stackTrace!, |
Jenn Magder | 1c0eade | 2022-01-10 13:20:21 -0800 | [diff] [blame] | 165 | doctorText: doctorText, |
James D. Lin | b7fd24a | 2020-04-28 10:34:03 -0700 | [diff] [blame] | 166 | ); |
| 167 | final File file = await _createLocalCrashReport(details); |
Jonah Williams | 9203448 | 2022-06-15 13:02:07 -0700 | [diff] [blame] | 168 | await globals.crashReporter!.informUser(details, file); |
Jenn Magder | 7d8f820 | 2019-11-26 14:06:31 -0800 | [diff] [blame] | 169 | |
Christopher Fujino | 08d5d51 | 2022-09-02 10:38:56 -0700 | [diff] [blame] | 170 | return _exit(1, shutdownHooks: shutdownHooks); |
Zachary Anderson | 6c408a0 | 2020-03-06 10:22:12 -0800 | [diff] [blame] | 171 | // This catch catches all exceptions to ensure the message below is printed. |
Jonah Williams | 9203448 | 2022-06-15 13:02:07 -0700 | [diff] [blame] | 172 | } catch (error, st) { // ignore: avoid_catches_without_on_clauses |
Zachary Anderson | 68ed5c2 | 2020-01-28 07:58:02 -0800 | [diff] [blame] | 173 | globals.stdio.stderrWrite( |
Jonah Williams | 9203448 | 2022-06-15 13:02:07 -0700 | [diff] [blame] | 174 | 'Unable to generate crash report due to secondary error: $error\n$st\n' |
Ian Hickson | 299d484 | 2021-10-14 22:03:03 -0700 | [diff] [blame] | 175 | '${globals.userMessages.flutterToolBugInstructions}\n', |
| 176 | ); |
James D. Lin | b7fd24a | 2020-04-28 10:34:03 -0700 | [diff] [blame] | 177 | // Any exception thrown here (including one thrown by `_exit()`) will |
Zachary Anderson | 36e8b93 | 2019-08-21 13:07:52 -0700 | [diff] [blame] | 178 | // get caught by our zone's `onError` handler. In order to avoid an |
| 179 | // infinite error loop, we throw an error that is recognized above |
| 180 | // and will trigger an immediate exit. |
| 181 | throw ProcessExit(1, immediate: true); |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 182 | } |
| 183 | } |
| 184 | } |
| 185 | |
Jenn Magder | 7d8f820 | 2019-11-26 14:06:31 -0800 | [diff] [blame] | 186 | String _crashCommand(List<String> args) => 'flutter ${args.join(' ')}'; |
| 187 | |
| 188 | String _crashException(dynamic error) => '${error.runtimeType}: $error'; |
| 189 | |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 190 | /// Saves the crash report to a local file. |
James D. Lin | b7fd24a | 2020-04-28 10:34:03 -0700 | [diff] [blame] | 191 | Future<File> _createLocalCrashReport(CrashDetails details) async { |
Zachary Anderson | 2c51efe | 2020-01-31 12:51:20 -0800 | [diff] [blame] | 192 | File crashFile = globals.fsUtils.getUniqueFile( |
Jenn Magder | 8109dcc | 2020-04-20 15:23:09 -0700 | [diff] [blame] | 193 | globals.fs.currentDirectory, |
Zachary Anderson | 2c51efe | 2020-01-31 12:51:20 -0800 | [diff] [blame] | 194 | 'flutter', |
| 195 | 'log', |
| 196 | ); |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 197 | |
Alexandre Ardhuin | d927c93 | 2018-09-12 08:29:29 +0200 | [diff] [blame] | 198 | final StringBuffer buffer = StringBuffer(); |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 199 | |
James D. Lin | b7fd24a | 2020-04-28 10:34:03 -0700 | [diff] [blame] | 200 | buffer.writeln('Flutter crash report.'); |
| 201 | buffer.writeln('${globals.userMessages.flutterToolBugInstructions}\n'); |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 202 | |
| 203 | buffer.writeln('## command\n'); |
James D. Lin | b7fd24a | 2020-04-28 10:34:03 -0700 | [diff] [blame] | 204 | buffer.writeln('${details.command}\n'); |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 205 | |
| 206 | buffer.writeln('## exception\n'); |
James D. Lin | b7fd24a | 2020-04-28 10:34:03 -0700 | [diff] [blame] | 207 | buffer.writeln('${_crashException(details.error)}\n'); |
| 208 | buffer.writeln('```\n${details.stackTrace}```\n'); |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 209 | |
| 210 | buffer.writeln('## flutter doctor\n'); |
Jenn Magder | 1c0eade | 2022-01-10 13:20:21 -0800 | [diff] [blame] | 211 | buffer.writeln('```\n${await details.doctorText.text}```'); |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 212 | |
| 213 | try { |
Zachary Anderson | 398ac1f | 2019-08-20 13:15:08 -0700 | [diff] [blame] | 214 | crashFile.writeAsStringSync(buffer.toString()); |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 215 | } on FileSystemException catch (_) { |
| 216 | // Fallback to the system temporary directory. |
Zachary Anderson | 2c51efe | 2020-01-31 12:51:20 -0800 | [diff] [blame] | 217 | crashFile = globals.fsUtils.getUniqueFile( |
Jenn Magder | 8109dcc | 2020-04-20 15:23:09 -0700 | [diff] [blame] | 218 | globals.fs.systemTempDirectory, |
Zachary Anderson | 2c51efe | 2020-01-31 12:51:20 -0800 | [diff] [blame] | 219 | 'flutter', |
| 220 | 'log', |
| 221 | ); |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 222 | try { |
Zachary Anderson | 398ac1f | 2019-08-20 13:15:08 -0700 | [diff] [blame] | 223 | crashFile.writeAsStringSync(buffer.toString()); |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 224 | } on FileSystemException catch (e) { |
Zachary Anderson | 23a3d10 | 2020-01-13 10:12:06 -0800 | [diff] [blame] | 225 | globals.printError('Could not write crash report to disk: $e'); |
| 226 | globals.printError(buffer.toString()); |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 227 | } |
| 228 | } |
| 229 | |
| 230 | return crashFile; |
| 231 | } |
| 232 | |
Christopher Fujino | 08d5d51 | 2022-09-02 10:38:56 -0700 | [diff] [blame] | 233 | Future<int> _exit(int code, {required ShutdownHooks shutdownHooks}) async { |
Zachary Anderson | 372fe29 | 2019-11-05 10:43:52 -0800 | [diff] [blame] | 234 | // Prints the welcome message if needed. |
Jenn Magder | f6a5512 | 2020-03-06 12:14:29 -0800 | [diff] [blame] | 235 | globals.flutterUsage.printWelcome(); |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 236 | |
| 237 | // Send any last analytics calls that are in progress without overly delaying |
| 238 | // the tool's exit (we wait a maximum of 250ms). |
Jenn Magder | f6a5512 | 2020-03-06 12:14:29 -0800 | [diff] [blame] | 239 | if (globals.flutterUsage.enabled) { |
Alexandre Ardhuin | d927c93 | 2018-09-12 08:29:29 +0200 | [diff] [blame] | 240 | final Stopwatch stopwatch = Stopwatch()..start(); |
Jenn Magder | f6a5512 | 2020-03-06 12:14:29 -0800 | [diff] [blame] | 241 | await globals.flutterUsage.ensureAnalyticsSent(); |
Zachary Anderson | 23a3d10 | 2020-01-13 10:12:06 -0800 | [diff] [blame] | 242 | globals.printTrace('ensureAnalyticsSent: ${stopwatch.elapsedMilliseconds}ms'); |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 243 | } |
| 244 | |
| 245 | // Run shutdown hooks before flushing logs |
Christopher Fujino | 08d5d51 | 2022-09-02 10:38:56 -0700 | [diff] [blame] | 246 | await shutdownHooks.runShutdownHooks(globals.logger); |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 247 | |
Alexandre Ardhuin | 2d3ff10 | 2018-10-05 07:54:56 +0200 | [diff] [blame] | 248 | final Completer<void> completer = Completer<void>(); |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 249 | |
| 250 | // Give the task / timer queue one cycle through before we hard exit. |
| 251 | Timer.run(() { |
| 252 | try { |
Zachary Anderson | 23a3d10 | 2020-01-13 10:12:06 -0800 | [diff] [blame] | 253 | globals.printTrace('exiting with code $code'); |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 254 | exit(code); |
| 255 | completer.complete(); |
Jonah Williams | 08576cb | 2020-10-12 09:31:02 -0700 | [diff] [blame] | 256 | // This catches all exceptions because the error is propagated on the |
Zachary Anderson | 6c408a0 | 2020-03-06 10:22:12 -0800 | [diff] [blame] | 257 | // completer. |
| 258 | } catch (error, stackTrace) { // ignore: avoid_catches_without_on_clauses |
Todd Volkert | 454db9d | 2017-11-09 21:45:31 -0800 | [diff] [blame] | 259 | completer.completeError(error, stackTrace); |
| 260 | } |
| 261 | }); |
| 262 | |
| 263 | await completer.future; |
| 264 | return code; |
| 265 | } |