// Copyright 2015 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. import 'dart:async'; import 'package:args/command_runner.dart'; import 'package:intl/intl.dart' as intl; import 'package:intl/intl_standalone.dart' as intl_standalone; import 'package:meta/meta.dart'; import 'src/base/common.dart'; import 'src/base/context.dart'; import 'src/base/file_system.dart'; import 'src/base/io.dart'; import 'src/base/logger.dart'; import 'src/base/process.dart'; import 'src/base/utils.dart'; import 'src/context_runner.dart'; import 'src/doctor.dart'; import 'src/globals.dart'; import 'src/reporting/github_template.dart'; import 'src/reporting/reporting.dart'; import 'src/runner/flutter_command.dart'; import 'src/runner/flutter_command_runner.dart'; import 'src/version.dart'; /// Runs the Flutter tool with support for the specified list of [commands]. Future run( List args, List commands, { bool muteCommandLogging = false, bool verbose = false, bool verboseHelp = false, bool reportCrashes, String flutterVersion, Map overrides, }) { reportCrashes ??= !isRunningOnBot; if (muteCommandLogging) { // Remove the verbose option; for help and doctor, users don't need to see // verbose logs. args = List.from(args); args.removeWhere((String option) => option == '-v' || option == '--verbose'); } final FlutterCommandRunner runner = FlutterCommandRunner(verboseHelp: verboseHelp); commands.forEach(runner.addCommand); return runInContext(() async { // Initialize the system locale. final String systemLocale = await intl_standalone.findSystemLocale(); intl.Intl.defaultLocale = intl.Intl.verifiedLocale( systemLocale, intl.NumberFormat.localeExists, onFailure: (String _) => 'en_US', ); String getVersion() => flutterVersion ?? FlutterVersion.instance.getVersionString(redactUnknownBranches: true); Object firstError; StackTrace firstStackTrace; return await runZoned>(() async { try { await runner.run(args); return await _exit(0); } catch (error, stackTrace) { firstError = error; firstStackTrace = stackTrace; return await _handleToolError( error, stackTrace, verbose, args, reportCrashes, getVersion); } }, onError: (Object error, StackTrace stackTrace) async { // If sending a crash report throws an error into the zone, we don't want // to re-try sending the crash report with *that* error. Rather, we want // to send the original error that triggered the crash report. final Object e = firstError ?? error; final StackTrace s = firstStackTrace ?? stackTrace; await _handleToolError(e, s, verbose, args, reportCrashes, getVersion); }); }, overrides: overrides); } Future _handleToolError( dynamic error, StackTrace stackTrace, bool verbose, List args, bool reportCrashes, String getFlutterVersion(), ) async { if (error is UsageException) { printError('${error.message}\n'); printError("Run 'flutter -h' (or 'flutter -h') for available flutter commands and options."); // Argument error exit code. return _exit(64); } else if (error is ToolExit) { if (error.message != null) { printError(error.message); } if (verbose) { printError('\n$stackTrace\n'); } return _exit(error.exitCode ?? 1); } else if (error is ProcessExit) { // We've caught an exit code. if (error.immediate) { exit(error.exitCode); return error.exitCode; } else { return _exit(error.exitCode); } } else { // We've crashed; emit a log report. stderr.writeln(); if (!reportCrashes) { // Print the stack trace on the bots - don't write a crash report. stderr.writeln('$error'); stderr.writeln(stackTrace.toString()); return _exit(1); } // Report to both [Usage] and [CrashReportSender]. flutterUsage.sendException(error); await CrashReportSender.instance.sendReport( error: error, stackTrace: stackTrace, getFlutterVersion: getFlutterVersion, command: args.join(' '), ); final String errorString = error.toString(); printError('Oops; flutter has exited unexpectedly: "$errorString".'); try { await _informUserOfCrash(args, error, stackTrace, errorString); return _exit(1); } catch (error) { stderr.writeln( 'Unable to generate crash report due to secondary error: $error\n' 'please let us know at https://github.com/flutter/flutter/issues.', ); // Any exception throw here (including one thrown by `_exit()`) will // get caught by our zone's `onError` handler. In order to avoid an // infinite error loop, we throw an error that is recognized above // and will trigger an immediate exit. throw ProcessExit(1, immediate: true); } } } Future _informUserOfCrash(List args, dynamic error, StackTrace stackTrace, String errorString) async { final String doctorText = await _doctorText(); final File file = await _createLocalCrashReport(args, error, stackTrace, doctorText); printError('A crash report has been written to ${file.path}.'); printStatus('This crash may already be reported. Check GitHub for similar crashes.', emphasis: true); final GitHubTemplateCreator gitHubTemplateCreator = context.get() ?? GitHubTemplateCreator(); final String similarIssuesURL = await gitHubTemplateCreator.toolCrashSimilarIssuesGitHubURL(errorString); printStatus('$similarIssuesURL\n', wrap: false); printStatus('To report your crash to the Flutter team, first read the guide to filing a bug.', emphasis: true); printStatus('https://flutter.dev/docs/resources/bug-reports\n', wrap: false); printStatus('Create a new GitHub issue by pasting this link into your browser and completing the issue template. Thank you!', emphasis: true); final String command = _crashCommand(args); final String gitHubTemplateURL = await gitHubTemplateCreator.toolCrashIssueTemplateGitHubURL( command, errorString, _crashException(error), stackTrace, doctorText ); printStatus('$gitHubTemplateURL\n', wrap: false); } String _crashCommand(List args) => 'flutter ${args.join(' ')}'; String _crashException(dynamic error) => '${error.runtimeType}: $error'; /// File system used by the crash reporting logic. /// /// We do not want to use the file system stored in the context because it may /// be recording. Additionally, in the case of a crash we do not trust the /// integrity of the [AppContext]. @visibleForTesting FileSystem crashFileSystem = const LocalFileSystem(); /// Saves the crash report to a local file. Future _createLocalCrashReport(List args, dynamic error, StackTrace stackTrace, String doctorText) async { File crashFile = getUniqueFile(crashFileSystem.currentDirectory, 'flutter', 'log'); final StringBuffer buffer = StringBuffer(); buffer.writeln('Flutter crash report; please file at https://github.com/flutter/flutter/issues.\n'); buffer.writeln('## command\n'); buffer.writeln('${_crashCommand(args)}\n'); buffer.writeln('## exception\n'); buffer.writeln('${_crashException(error)}\n'); buffer.writeln('```\n$stackTrace```\n'); buffer.writeln('## flutter doctor\n'); buffer.writeln('```\n$doctorText```'); try { crashFile.writeAsStringSync(buffer.toString()); } on FileSystemException catch (_) { // Fallback to the system temporary directory. crashFile = getUniqueFile(crashFileSystem.systemTempDirectory, 'flutter', 'log'); try { crashFile.writeAsStringSync(buffer.toString()); } on FileSystemException catch (e) { printError('Could not write crash report to disk: $e'); printError(buffer.toString()); } } return crashFile; } Future _doctorText() async { try { final BufferLogger logger = BufferLogger(); await context.run( body: () => doctor.diagnose(verbose: true, showColor: false), overrides: { Logger: () => logger, }, ); return logger.statusText; } catch (error, trace) { return 'encountered exception: $error\n\n${trace.toString().trim()}\n'; } } Future _exit(int code) async { // Prints the welcome message if needed. flutterUsage.printWelcome(); // Send any last analytics calls that are in progress without overly delaying // the tool's exit (we wait a maximum of 250ms). if (flutterUsage.enabled) { final Stopwatch stopwatch = Stopwatch()..start(); await flutterUsage.ensureAnalyticsSent(); printTrace('ensureAnalyticsSent: ${stopwatch.elapsedMilliseconds}ms'); } // Run shutdown hooks before flushing logs await runShutdownHooks(); final Completer completer = Completer(); // Give the task / timer queue one cycle through before we hard exit. Timer.run(() { try { printTrace('exiting with code $code'); exit(code); completer.complete(); } catch (error, stackTrace) { completer.completeError(error, stackTrace); } }); await completer.future; return code; }