|  | // Copyright (c) 2012, the Dart project authors.  Please see the AUTHORS file | 
|  | // for details. All rights reserved. Use of this source code is governed by a | 
|  | // BSD-style license that can be found in the LICENSE file. | 
|  |  | 
|  | /// Message logging. | 
|  | library; | 
|  |  | 
|  | import 'dart:async'; | 
|  | import 'dart:convert'; | 
|  | import 'dart:io'; | 
|  |  | 
|  | import 'package:args/command_runner.dart'; | 
|  | import 'package:path/path.dart' as p; | 
|  | import 'package:source_span/source_span.dart'; | 
|  | import 'package:stack_trace/stack_trace.dart'; | 
|  |  | 
|  | import 'entrypoint.dart'; | 
|  | import 'exceptions.dart'; | 
|  | import 'io.dart'; | 
|  | import 'progress.dart'; | 
|  | import 'sdk.dart'; | 
|  | import 'transcript.dart'; | 
|  | import 'utils.dart'; | 
|  |  | 
|  | /// The singleton instance so that we can have a nice api like: | 
|  | /// | 
|  | ///     log.json.error(...); | 
|  | final json = _JsonLogger(); | 
|  |  | 
|  | /// The current logging verbosity. | 
|  | Verbosity verbosity = Verbosity.normal; | 
|  |  | 
|  | /// In cases where there's a ton of log spew, make sure we don't eat infinite | 
|  | /// memory. | 
|  | /// | 
|  | /// This can occur when the backtracking solver stumbles into a pathological | 
|  | /// dependency graph. It generally will find a solution, but it may log | 
|  | /// thousands and thousands of entries to get there. | 
|  | const _maxTranscript = 10000; | 
|  |  | 
|  | /// The list of recorded log messages. Will only be recorded if | 
|  | /// [recordTranscript()] is called. | 
|  | final Transcript<_Entry> _transcript = Transcript(_maxTranscript); | 
|  |  | 
|  | /// The currently-animated progress indicator, if any. | 
|  | /// | 
|  | /// This will also be in [_progresses]. | 
|  | Progress? _animatedProgress; | 
|  |  | 
|  | final _cyan = getAnsi('\u001b[36m'); | 
|  | final _green = getAnsi('\u001b[32m'); | 
|  | final _magenta = getAnsi('\u001b[35m'); | 
|  | final _red = getAnsi('\u001b[31m'); | 
|  | final _yellow = getAnsi('\u001b[33m'); | 
|  | //final _blue = getSpecial('\u001b[34m'); | 
|  | final _gray = getAnsi('\u001b[38;5;245m'); | 
|  |  | 
|  | final _none = getAnsi('\u001b[0m'); | 
|  | final _noColor = getAnsi('\u001b[39m'); | 
|  | final _bold = getAnsi('\u001b[1m'); | 
|  |  | 
|  | /// An enum type for defining the different logging levels a given message can | 
|  | /// be associated with. | 
|  | /// | 
|  | /// By default, [error] and [warning] messages are printed to stderr. | 
|  | /// [message] messages are printed to stdout, and others are ignored. | 
|  | enum Level { | 
|  | /// An error occurred and an operation could not be completed. | 
|  | /// | 
|  | /// Usually shown to the user on stderr. | 
|  | error('ERR '), | 
|  |  | 
|  | /// Something unexpected happened, but the program was able to continue, | 
|  | /// though possibly in a degraded fashion. | 
|  | warning('WARN'), | 
|  |  | 
|  | /// A message intended specifically to be shown to the user. | 
|  | message('MSG '), | 
|  |  | 
|  | /// Some interaction with the external world occurred, such as a network | 
|  | /// operation, process spawning, or file IO. | 
|  | io('IO  '), | 
|  |  | 
|  | /// Incremental output during pub's version constraint solver. | 
|  | solver('SLVR'), | 
|  |  | 
|  | /// Fine-grained and verbose additional information. | 
|  | /// | 
|  | /// Used to provide program state context for other logs (such as what pub | 
|  | /// was doing when an IO operation occurred) or just more detail for an | 
|  | /// operation. | 
|  | fine('FINE'); | 
|  |  | 
|  | const Level(this.name); | 
|  |  | 
|  | final String name; | 
|  |  | 
|  | @override | 
|  | String toString() => name; | 
|  | } | 
|  |  | 
|  | /// An enum type to control which log levels are displayed and how they are | 
|  | /// displayed. | 
|  | enum Verbosity { | 
|  | /// Silence all logging. | 
|  | none('none', { | 
|  | Level.error: null, | 
|  | Level.warning: null, | 
|  | Level.message: null, | 
|  | Level.io: null, | 
|  | Level.solver: null, | 
|  | Level.fine: null, | 
|  | }), | 
|  |  | 
|  | /// Shows only errors. | 
|  | error('error', { | 
|  | Level.error: _logToStderr, | 
|  | Level.warning: null, | 
|  | Level.message: null, | 
|  | Level.io: null, | 
|  | Level.solver: null, | 
|  | Level.fine: null, | 
|  | }), | 
|  |  | 
|  | /// Shows only errors and warnings. | 
|  | warning('warning', { | 
|  | Level.error: _logToStderr, | 
|  | Level.warning: _logToStderr, | 
|  | Level.message: null, | 
|  | Level.io: null, | 
|  | Level.solver: null, | 
|  | Level.fine: null, | 
|  | }), | 
|  |  | 
|  | /// The default verbosity which shows errors, warnings, and messages. | 
|  | normal('normal', { | 
|  | Level.error: _logToStderr, | 
|  | Level.warning: _logToStderr, | 
|  | Level.message: _logToStdout, | 
|  | Level.io: null, | 
|  | Level.solver: null, | 
|  | Level.fine: null, | 
|  | }), | 
|  |  | 
|  | /// Shows errors, warnings, messages, and IO event logs. | 
|  | io('io', { | 
|  | Level.error: _logToStderrWithLabel, | 
|  | Level.warning: _logToStderrWithLabel, | 
|  | Level.message: _logToStdoutWithLabel, | 
|  | Level.io: _logToStderrWithLabel, | 
|  | Level.solver: null, | 
|  | Level.fine: null, | 
|  | }), | 
|  |  | 
|  | /// Shows errors, warnings, messages, and version solver logs. | 
|  | solver('solver', { | 
|  | Level.error: _logToStderr, | 
|  | Level.warning: _logToStderr, | 
|  | Level.message: _logToStdout, | 
|  | Level.io: null, | 
|  | Level.solver: _logToStdout, | 
|  | Level.fine: null, | 
|  | }), | 
|  |  | 
|  | /// Shows all logs. | 
|  | all('all', { | 
|  | Level.error: _logToStderrWithLabel, | 
|  | Level.warning: _logToStderrWithLabel, | 
|  | Level.message: _logToStdoutWithLabel, | 
|  | Level.io: _logToStderrWithLabel, | 
|  | Level.solver: _logToStderrWithLabel, | 
|  | Level.fine: _logToStderrWithLabel, | 
|  | }), | 
|  |  | 
|  | /// Shows all logs. | 
|  | testing('testing', { | 
|  | Level.error: _logToStderrWithLabel, | 
|  | Level.warning: _logToStderrWithLabel, | 
|  | Level.message: _logToStdoutWithLabel, | 
|  | Level.io: _logToStderrWithLabel, | 
|  | Level.solver: _logToStderrWithLabel, | 
|  | Level.fine: _logToStderrWithLabel, | 
|  | }); | 
|  |  | 
|  | const Verbosity(this.name, this._loggers); | 
|  |  | 
|  | final String name; | 
|  | final Map<Level, void Function(_Entry entry)?> _loggers; | 
|  |  | 
|  | /// Returns whether or not logs at [level] will be printed. | 
|  | bool isLevelVisible(Level level) => _loggers[level] != null; | 
|  |  | 
|  | @override | 
|  | String toString() => name; | 
|  | } | 
|  |  | 
|  | /// A single log entry. | 
|  | class _Entry { | 
|  | final Level level; | 
|  | final List<String> lines; | 
|  |  | 
|  | _Entry(this.level, this.lines); | 
|  | } | 
|  |  | 
|  | /// Logs [message] at [Level.error]. | 
|  | /// | 
|  | /// If [error] is passed, it's appended to [message]. If [trace] is passed, it's | 
|  | /// printed at log level fine. | 
|  | void error(String message, [Object? error, StackTrace? trace]) { | 
|  | if (error != null) { | 
|  | message = message.isEmpty ? '$error' : '$message: $error'; | 
|  | if (error is Error && trace == null) trace = error.stackTrace; | 
|  | } | 
|  | write(Level.error, message); | 
|  | if (trace != null) write(Level.fine, Chain.forTrace(trace).toString()); | 
|  | } | 
|  |  | 
|  | /// Logs [message] at [Level.warning]. | 
|  | void warning(String message) => write(Level.warning, message); | 
|  |  | 
|  | /// Logs [message] at [Level.message]. | 
|  | void message(String message) => write(Level.message, message); | 
|  |  | 
|  | /// Logs [message] at [Level.io]. | 
|  | void io(String message) => write(Level.io, message); | 
|  |  | 
|  | /// Logs [message] at [Level.solver]. | 
|  | void solver(String message) => write(Level.solver, message); | 
|  |  | 
|  | /// Logs [message] at [Level.fine]. | 
|  | void fine(String message) => write(Level.fine, message); | 
|  |  | 
|  | /// Logs [message] at [level]. | 
|  | void write(Level level, String message) { | 
|  | message = message.toString(); | 
|  | final lines = splitLines(message); | 
|  |  | 
|  | // Discard a trailing newline. This is useful since StringBuffers often end | 
|  | // up with an extra newline at the end from using [writeln]. | 
|  | if (lines.isNotEmpty && lines.last == '') { | 
|  | lines.removeLast(); | 
|  | } | 
|  |  | 
|  | final entry = _Entry(level, lines); | 
|  |  | 
|  | final logFn = verbosity._loggers[level]; | 
|  | if (logFn != null) logFn(entry); | 
|  |  | 
|  | _transcript.add(entry); | 
|  | } | 
|  |  | 
|  | /// Logs the spawning of an [executable] process with [arguments] at [io] | 
|  | /// level. | 
|  | void process( | 
|  | String executable, | 
|  | List<String> arguments, | 
|  | String workingDirectory, | 
|  | ) { | 
|  | io("Spawning \"$executable ${arguments.join(' ')}\" in " | 
|  | '${p.absolute(workingDirectory)}'); | 
|  | } | 
|  |  | 
|  | /// Logs the results of running [executable]. | 
|  | void processResult(String executable, PubProcessResult result) { | 
|  | // Log it all as one message so that it shows up as a single unit in the logs. | 
|  | final buffer = StringBuffer(); | 
|  | buffer.writeln('Finished $executable. Exit code ${result.exitCode}.'); | 
|  |  | 
|  | void dumpOutput(String name, List<String> output) { | 
|  | if (output.isEmpty) { | 
|  | buffer.writeln('Nothing output on $name.'); | 
|  | } else { | 
|  | buffer.writeln('$name:'); | 
|  | var numLines = 0; | 
|  | for (var line in output) { | 
|  | if (++numLines > 1000) { | 
|  | buffer.writeln('[${output.length - 1000}] more lines of output ' | 
|  | 'truncated...]'); | 
|  | break; | 
|  | } | 
|  |  | 
|  | buffer.writeln('| $line'); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | dumpOutput('stdout', result.stdout); | 
|  | dumpOutput('stderr', result.stderr); | 
|  |  | 
|  | io(buffer.toString().trim()); | 
|  | } | 
|  |  | 
|  | /// Logs an exception. | 
|  | void exception(Object exception, [StackTrace? trace]) { | 
|  | if (exception is SilentException) return; | 
|  |  | 
|  | final chain = trace == null ? Chain.current() : Chain.forTrace(trace); | 
|  |  | 
|  | // This is basically the top-level exception handler so that we don't | 
|  | // spew a stack trace on our users. | 
|  | if (exception is SourceSpanException) { | 
|  | error(exception.toString(color: canUseAnsiCodes)); | 
|  | } else { | 
|  | error(getErrorMessage(exception)); | 
|  | } | 
|  | fine('Exception type: ${exception.runtimeType}'); | 
|  |  | 
|  | if (json.enabled) { | 
|  | if (exception is UsageException) { | 
|  | // Don't print usage info in JSON output. | 
|  | json.error(exception.message); | 
|  | } else { | 
|  | json.error(exception); | 
|  | } | 
|  | } | 
|  |  | 
|  | if (!isUserFacingException(exception)) { | 
|  | error(chain.terse.toString()); | 
|  | } else { | 
|  | fine(chain.terse.toString()); | 
|  | } | 
|  |  | 
|  | if (exception is WrappedException && exception.innerError != null) { | 
|  | var message = 'Wrapped exception: ${exception.innerError}'; | 
|  | if (exception.innerChain != null) { | 
|  | message = '$message\n${exception.innerChain}'; | 
|  | } | 
|  | fine(message); | 
|  | } | 
|  | } | 
|  |  | 
|  | /// Prints the recorded log transcript to stderr. | 
|  | void dumpTranscriptToStdErr() { | 
|  | stderr.writeln('---- Log transcript ----'); | 
|  | _transcript.forEach((entry) { | 
|  | _printToStream(stderr, entry, showLabel: true); | 
|  | }, (discarded) { | 
|  | stderr.writeln('---- ($discarded discarded) ----'); | 
|  | }); | 
|  | stderr.writeln('---- End log transcript ----'); | 
|  | } | 
|  |  | 
|  | /// Shortens [input] to at most [limit] characters by omitting the middle part | 
|  | /// replacing it with '[...]' if it is too long. | 
|  | /// | 
|  | /// [limit] must be more than 5. | 
|  | String limitLength(String input, int limit) { | 
|  | const snip = '[...]'; | 
|  | assert(limit > snip.length); | 
|  | if (input.length <= limit) return input; | 
|  | final half = (limit - snip.length) ~/ 2; | 
|  | final extra = (limit - snip.length).isOdd ? 1 : 0; | 
|  | return '${input.substring(0, half + extra)}' | 
|  | '$snip' | 
|  | '${input.substring(input.length - half)}'; | 
|  | } | 
|  |  | 
|  | /// Prints relevant system information and the log transcript to [path]. | 
|  | void dumpTranscriptToFile(String path, String command, Entrypoint? entrypoint) { | 
|  | final buffer = StringBuffer(); | 
|  | buffer.writeln(''' | 
|  | Information about the latest pub run. | 
|  |  | 
|  | If you believe something is not working right, you can go to | 
|  | https://github.com/dart-lang/pub/issues/new to post a new issue and attach this file. | 
|  |  | 
|  | Before making this file public, make sure to remove any sensitive information! | 
|  |  | 
|  | Pub version: ${sdk.version} | 
|  | Created: ${DateTime.now().toIso8601String()} | 
|  | FLUTTER_ROOT: ${Platform.environment['FLUTTER_ROOT'] ?? '<not set>'} | 
|  | PUB_HOSTED_URL: ${Platform.environment['PUB_HOSTED_URL'] ?? '<not set>'} | 
|  | PUB_CACHE: "${Platform.environment['PUB_CACHE'] ?? '<not set>'}" | 
|  | Command: $command | 
|  | Platform: ${Platform.operatingSystem} | 
|  | '''); | 
|  |  | 
|  | if (entrypoint != null) { | 
|  | // TODO(https://github.com/dart-lang/pub/issues/4127): We probably want to | 
|  | // log all pubspecs in workspace? | 
|  |  | 
|  | if (entrypoint.canFindWorkspaceRoot) { | 
|  | buffer.writeln( | 
|  | '---- ${p.absolute(entrypoint.workspaceRoot.pubspecPath)} ----', | 
|  | ); | 
|  | buffer.writeln( | 
|  | limitLength( | 
|  | readTextFile(entrypoint.workspaceRoot.pubspecPath), | 
|  | 5000, | 
|  | ), | 
|  | ); | 
|  | buffer.writeln('---- End pubspec.yaml ----'); | 
|  | } else { | 
|  | buffer.writeln('<No pubspec.yaml>'); | 
|  | } | 
|  | if (entrypoint.canFindWorkspaceRoot && | 
|  | fileExists(entrypoint.lockFilePath)) { | 
|  | buffer.writeln('---- ${p.absolute(entrypoint.lockFilePath)} ----'); | 
|  | buffer.writeln(limitLength(readTextFile(entrypoint.lockFilePath), 5000)); | 
|  | buffer.writeln('---- End pubspec.lock ----'); | 
|  | } else { | 
|  | buffer.writeln('<No pubspec.lock>'); | 
|  | } | 
|  | } | 
|  |  | 
|  | buffer.writeln('---- Log transcript ----'); | 
|  |  | 
|  | _transcript.forEach((entry) { | 
|  | _printToStream(buffer, entry, showLabel: true); | 
|  | }, (discarded) { | 
|  | buffer.writeln('---- ($discarded entries discarded) ----'); | 
|  | }); | 
|  | buffer.writeln('---- End log transcript ----'); | 
|  | ensureDir(p.dirname(path)); | 
|  | try { | 
|  | writeTextFile(path, buffer.toString(), dontLogContents: true); | 
|  | } on IOException catch (e) { | 
|  | stderr.writeln('Failed writing log to `$path` ($e), writing it to stderr:'); | 
|  | dumpTranscriptToStdErr(); | 
|  | } | 
|  | } | 
|  |  | 
|  | /// Filter out normal pub output when not attached to a terminal | 
|  | /// | 
|  | /// Unless the user has overridden the verbosity, | 
|  | /// | 
|  | /// This is useful to not pollute stdout when the output is piped somewhere. | 
|  | Future<T> errorsOnlyUnlessTerminal<T>(FutureOr<T> Function() callback) async { | 
|  | final oldVerbosity = verbosity; | 
|  | if (verbosity == Verbosity.normal && !terminalOutputForStdout) { | 
|  | verbosity = Verbosity.error; | 
|  | } | 
|  | final result = await callback(); | 
|  | verbosity = oldVerbosity; | 
|  | return result; | 
|  | } | 
|  |  | 
|  | /// Prints [message] then displays an updated elapsed time until the future | 
|  | /// returned by [callback] completes. | 
|  | /// | 
|  | /// If anything else is logged during this (including another call to | 
|  | /// [progress]) that cancels the progress animation, although the total time | 
|  | /// will still be printed once it finishes. If [fine] is passed, the progress | 
|  | /// information will only be visible at [Level.fine]. | 
|  | Future<T> progress<T>(String message, Future<T> Function() callback) { | 
|  | _stopProgress(); | 
|  |  | 
|  | final progress = Progress(message); | 
|  | _animatedProgress = progress; | 
|  | return callback().whenComplete(progress.stop); | 
|  | } | 
|  |  | 
|  | /// Like [progress] but erases the message once done. | 
|  | Future<T> spinner<T>( | 
|  | String message, | 
|  | Future<T> Function() callback, { | 
|  | bool condition = true, | 
|  | }) { | 
|  | if (condition) { | 
|  | _stopProgress(); | 
|  |  | 
|  | final progress = Progress(message); | 
|  | _animatedProgress = progress; | 
|  | return callback().whenComplete(progress.stopAndClear); | 
|  | } | 
|  | return callback(); | 
|  | } | 
|  |  | 
|  | /// Stops animating the running progress indicator, if currently running. | 
|  | void _stopProgress() { | 
|  | if (_animatedProgress != null) _animatedProgress!.stopAnimating(); | 
|  | _animatedProgress = null; | 
|  | } | 
|  |  | 
|  | /// The number of outstanding calls to [muteProgress] that have not been unmuted | 
|  | /// yet. | 
|  | int _numMutes = 0; | 
|  |  | 
|  | /// Whether progress animation should be muted or not. | 
|  | bool get isMuted => _numMutes > 0; | 
|  |  | 
|  | /// Stops animating any ongoing progress. | 
|  | /// | 
|  | /// This is called before spawning Git since Git sometimes writes directly to | 
|  | /// the terminal to ask for login credentials, which would then get overwritten | 
|  | /// by the progress animation. | 
|  | /// | 
|  | /// Each call to this must be paired with a call to [unmuteProgress]. | 
|  | void muteProgress() { | 
|  | _numMutes++; | 
|  | } | 
|  |  | 
|  | /// Resumes animating any ongoing progress once all calls to [muteProgress] | 
|  | /// have made their matching [unmuteProgress]. | 
|  | void unmuteProgress() { | 
|  | assert(_numMutes > 0); | 
|  | _numMutes--; | 
|  | } | 
|  |  | 
|  | /// Wraps [text] in the ANSI escape codes to make it bold when on a platform | 
|  | /// that supports that. | 
|  | /// | 
|  | /// Use this to highlight the most important piece of a long chunk of text. | 
|  | String bold(String text) => '$_bold$text$_none'; | 
|  |  | 
|  | /// Wraps [text] in the ANSI escape codes to make it gray when on a platform | 
|  | /// that supports that. | 
|  | /// | 
|  | /// Use this for text that's less important than the text around it. | 
|  | String gray(String text) => '$_gray$text$_none'; | 
|  |  | 
|  | /// Wraps [text] in the ANSI escape codes to color it cyan when on a platform | 
|  | /// that supports that. | 
|  | /// | 
|  | /// Use this to highlight something interesting but neither good nor bad. | 
|  | String cyan(String text) => _addColor(text, _cyan); | 
|  |  | 
|  | /// Wraps [text] in the ANSI escape codes to color it green when on a platform | 
|  | /// that supports that. | 
|  | /// | 
|  | /// Use this to highlight something successful or otherwise positive. | 
|  | String green(String text) => _addColor(text, _green); | 
|  |  | 
|  | /// Wraps [text] in the ANSI escape codes to color it magenta when on a | 
|  | /// platform that supports that. | 
|  | /// | 
|  | /// Use this to highlight something risky that the user should be aware of but | 
|  | /// may intend to do. | 
|  | String magenta(String text) => _addColor(text, _magenta); | 
|  |  | 
|  | /// Wraps [text] in the ANSI escape codes to color it red when on a platform | 
|  | /// that supports that. | 
|  | /// | 
|  | /// Use this to highlight unequivocal errors, problems, or failures. | 
|  | String red(String text) => _addColor(text, _red); | 
|  |  | 
|  | /// Wraps [text] in the ANSI escape codes to color it yellow when on a platform | 
|  | /// that supports that. | 
|  | /// | 
|  | /// Use this to highlight warnings, cautions or other things that are bad but | 
|  | /// do not prevent the user's goal from being reached. | 
|  | String yellow(String text) => _addColor(text, _yellow); | 
|  |  | 
|  | /// Returns [text] colored using the given [colorCode]. | 
|  | /// | 
|  | /// This is resilient to the text containing other colors or bold text. | 
|  | String _addColor(String text, String colorCode) { | 
|  | return colorCode + | 
|  | text | 
|  | .toString() | 
|  | .replaceAll(_none, _none + colorCode) | 
|  | .replaceAll(_noColor, _none + colorCode) + | 
|  | _noColor; | 
|  | } | 
|  |  | 
|  | /// Log function that prints the message to stdout. | 
|  | void _logToStdout(_Entry entry) { | 
|  | _logToStream(stdout, entry, showLabel: false); | 
|  | } | 
|  |  | 
|  | /// Log function that prints the message to stdout with the level name. | 
|  | void _logToStdoutWithLabel(_Entry entry) { | 
|  | _logToStream(stdout, entry, showLabel: true); | 
|  | } | 
|  |  | 
|  | /// Log function that prints the message to stderr. | 
|  | void _logToStderr(_Entry entry) { | 
|  | _logToStream(stderr, entry, showLabel: false); | 
|  | } | 
|  |  | 
|  | /// Log function that prints the message to stderr with the level name. | 
|  | void _logToStderrWithLabel(_Entry entry) { | 
|  | _logToStream(stderr, entry, showLabel: true); | 
|  | } | 
|  |  | 
|  | void _logToStream(IOSink sink, _Entry entry, {required bool showLabel}) { | 
|  | if (json.enabled) return; | 
|  |  | 
|  | _printToStream(sink, entry, showLabel: showLabel); | 
|  | } | 
|  |  | 
|  | void _printToStream(StringSink sink, _Entry entry, {required bool showLabel}) { | 
|  | _stopProgress(); | 
|  |  | 
|  | var firstLine = true; | 
|  | for (var line in entry.lines) { | 
|  | if (showLabel) { | 
|  | if (firstLine) { | 
|  | sink.write('${entry.level.name}: '); | 
|  | } else { | 
|  | sink.write('    | '); | 
|  | } | 
|  | } | 
|  |  | 
|  | sink.writeln(line); | 
|  |  | 
|  | firstLine = false; | 
|  | } | 
|  | } | 
|  |  | 
|  | /// Namespace-like class for collecting the methods for JSON logging. | 
|  | class _JsonLogger { | 
|  | /// Whether logging should use machine-friendly JSON output or human-friendly | 
|  | /// text. | 
|  | /// | 
|  | /// If set to `true`, then no regular logging is printed. Logged messages | 
|  | /// will still be recorded and displayed if the transcript is printed. | 
|  | bool enabled = false; | 
|  |  | 
|  | /// Creates an error JSON object for [error] and prints it if JSON output | 
|  | /// is enabled. | 
|  | /// | 
|  | /// Always prints to stdout. | 
|  | void error(Object error, [StackTrace? stackTrace]) { | 
|  | final errorJson = {'error': error.toString()}; | 
|  |  | 
|  | if (stackTrace == null && error is Error) stackTrace = error.stackTrace; | 
|  | if (stackTrace != null) { | 
|  | errorJson['stackTrace'] = Chain.forTrace(stackTrace).toString(); | 
|  | } | 
|  |  | 
|  | // If the error came from a file, include the path. | 
|  | if (error is SourceSpanException && error.span?.sourceUrl != null) { | 
|  | // Normalize paths and make them absolute for backwards compatibility with | 
|  | // the protocol used by the analyzer. | 
|  | errorJson['path'] = | 
|  | p.normalize(p.absolute(p.fromUri(error.span!.sourceUrl))); | 
|  | } | 
|  |  | 
|  | if (error is FileException) { | 
|  | errorJson['path'] = p.normalize(p.absolute(error.path)); | 
|  | } | 
|  |  | 
|  | message(errorJson); | 
|  | } | 
|  |  | 
|  | /// Encodes [message] to JSON and prints it if JSON output is enabled. | 
|  | void message(Map<String, String> message) { | 
|  | if (!enabled) return; | 
|  |  | 
|  | stdout.writeln(jsonEncode(message)); | 
|  | } | 
|  | } |