Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(1197)

Unified Diff: utils/pub/log.dart

Issue 11437019: Add logging system to pub and sprinkle some logging in. (Closed) Base URL: https://dart.googlecode.com/svn/branches/bleeding_edge/dart
Patch Set: print() -> log.message(). Created 8 years ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: utils/pub/log.dart
diff --git a/utils/pub/log.dart b/utils/pub/log.dart
new file mode 100644
index 0000000000000000000000000000000000000000..8ef2a6445a13af4bfef3c9b4e2d04c7040230013
--- /dev/null
+++ b/utils/pub/log.dart
@@ -0,0 +1,194 @@
+// 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 logging;
nweiz 2012/12/05 23:56:54 It makes me sad that we're rolling our own logging
Bob Nystrom 2012/12/06 01:33:26 I spent some time integrating pkg/logging in first
+
+import 'dart:io';
+import 'io.dart';
+
+typedef LogFn(Level level, message);
+final Map<Level, LogFn> _loggers = new Map<Level, LogFn>();
+
+/// The list of recorded log messages. Will only be recorded if
+/// [recordTranscript()] is called.
+List<Entry> _transcript;
+
+/// An enum type for defining the different logging levels.
nweiz 2012/12/05 23:56:54 Mention which levels are enabled by default and wh
Bob Nystrom 2012/12/06 01:33:26 Done.
+class Level {
+ /// An error occurred and an operation could not be completed. Usually shown
+ /// to the user on stderr.
+ static const ERROR = const Level("Error");
+
+ /// Something unexpected happened, but the program was able to continue,
+ /// though possibly in a degraded fashion.
+ static const WARNING = const Level("Warning");
+
+ /// A message intended specifically to be shown to the user.
+ static const MESSAGE = const Level("Message");
+
+ /// Some interaction with the external world occurred, such as a network
+ /// operation, process spawning, or file IO.
+ static const IO = const Level("IO");
+
+ /// Fine-grained and verbose additional information. Can be 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.
+ static const FINE = const Level("FINE");
+
+ const Level(this.name);
nweiz 2012/12/05 23:56:54 This should probably be private.
Bob Nystrom 2012/12/06 01:33:26 Done.
+ final String name;
+
+ String toString() => name;
+ int get hashCode => name.hashCode;
+}
+
+/// A single log entry.
+class Entry {
nweiz 2012/12/05 23:56:54 I'm not sure this class is more useful than Pair<L
Bob Nystrom 2012/12/06 01:33:26 I like nice named types and Dart is pretty compact
+ final Level level;
+ final String message;
+
+ Entry(this.level, this.message);
+}
+
+/// Logs [message] at [Level.ERROR].
+void error(message) => write(Level.ERROR, message);
+
+/// Logs [message] at [Level.WARNING].
+void warning(message) => write(Level.WARNING, message);
+
+/// Logs [message] at [Level.MESSAGE].
+void message(message) => write(Level.MESSAGE, message);
+
+/// Logs [message] at [Level.IO].
+void io(message) => write(Level.IO, message);
+
+/// Logs [message] at [Level.FINE].
+void fine(message) => write(Level.FINE, message);
+
+/// Logs [message] at [level].
+void write(Level level, message) {
+ if (_loggers.isEmpty) showNormal();
+
+ var logFn = _loggers[level];
+ if (logFn != null) logFn(level, message);
+
+ if (_transcript != null) {
+ _transcript.add(new Entry(level, '$message'));
+ }
+}
+
+/// Logs an asynchronous IO operation. Logs [startMessage] before the operation
+/// starts, then when [operation] completes, invokes [endMessage] with the
+/// completion value and logs the result of that. Returns a future that
+/// completes after the logging is done.
+///
+/// If [endMessage] is omitted, then logs "Begin [startMessage]" before the
+/// operation and "End [startMessage]" after it.
+Future ioAsync(String startMessage, Future operation,
nweiz 2012/12/05 23:56:54 I'm not sure these arguments are in the right orde
Bob Nystrom 2012/12/06 01:33:26 I don't mind stuff coming before functions as much
nweiz 2012/12/06 19:40:02 But the future actually starts before either messa
+ [String endMessage(value)]) {
+ if (endMessage == null) {
+ io("Begin $startMessage.");
+ } else {
+ io(startMessage);
+ }
+
+ return operation.transform((result) {
+ if (endMessage == null) {
+ io("End $startMessage.");
+ } else {
+ io(endMessage(result));
+ }
+ return result;
+ });
+}
+
+/// Logs the spawning of an [executable] process with [arguments] at [INFO]
nweiz 2012/12/05 23:56:54 "INFO" -> "IO"
Bob Nystrom 2012/12/06 01:33:26 Done.
+/// level.
+void process(String executable, List<String> arguments) {
+ io("Spawning $executable ${Strings.join(arguments, ' ')}");
+}
+
+/// Logs the results of running [executable] with [arguments].
+void processResult(String executable, List<String> arguments,
nweiz 2012/12/05 23:56:54 "arguments" is unused.
Bob Nystrom 2012/12/06 01:33:26 Done.
+ PubProcessResult result) {
+ io("Finished $executable. Exit code ${result.exitCode}.");
+
+ dumpOutput(String name, List<String> output) {
+ if (output.length == 0) {
nweiz 2012/12/05 23:56:54 Add a sanity check for length.
Bob Nystrom 2012/12/06 01:33:26 Done.
+ fine("Nothing output on $name.");
+ } else {
+ fine("$name:");
+ output.forEach(fine);
nweiz 2012/12/05 23:56:54 Another good place to indent.
Bob Nystrom 2012/12/06 01:33:26 See other comments.
+ }
+ }
+
+ dumpOutput("stdout", result.stdout);
+ dumpOutput("stderr", result.stderr);
+}
+
+/// Enables recording of log entries.
+void recordTranscript() {
+ _transcript = <Entry>[];
+}
+
+/// If [recordTranscript()] was called, then prints the previously recorded log
+/// transcript to stderr.
+void dumpTranscript() {
+ if (_transcript == null) return;
+
+ stderr.writeString('---- Log transcript ----\n');
+ for (var entry in _transcript) {
+ _logToStderrWithLabel(entry.level, entry.message);
+ }
+ stderr.writeString('---- End log transcript ----\n');
+}
+
+/// Sets the verbosity to "normal", which shows errors, warnings, and messages.
+void showNormal() {
+ _loggers[Level.ERROR] = _logToStderr;
+ _loggers[Level.WARNING] = _logToStderrWithLabel;
+ _loggers[Level.MESSAGE] = _logToStdout;
+ _loggers[Level.IO] = null;
+ _loggers[Level.FINE] = null;
+}
+
+/// Sets the verbosity to "io", which shows errors, warnings, messages, and IO
+/// event logs.
+void showIO() {
+ _loggers[Level.ERROR] = _logToStderrWithLabel;
+ _loggers[Level.WARNING] = _logToStderrWithLabel;
+ _loggers[Level.MESSAGE] = _logToStdoutWithLabel;
+ _loggers[Level.IO] = _logToStderrWithLabel;
+ _loggers[Level.FINE] = null;
+}
+
+/// Sets the verbosity to "all", which logs ALL the things.
+void showAll() {
+ _loggers[Level.ERROR] = _logToStderrWithLabel;
+ _loggers[Level.WARNING] = _logToStderrWithLabel;
+ _loggers[Level.MESSAGE] = _logToStdoutWithLabel;
+ _loggers[Level.IO] = _logToStderrWithLabel;
+ _loggers[Level.FINE] = _logToStderrWithLabel;
+}
+
+/// Log function that prints the message to stdout.
+void _logToStdout(Level level, message) {
+ print('$message');
nweiz 2012/12/05 23:56:54 You could automatically handle that indentation I'
Bob Nystrom 2012/12/06 01:33:26 See other comments. :)
+}
+
+/// Log function that prints the message to stdout with the level name.
+void _logToStdoutWithLabel(Level level, message) {
+ print('$level: $message');
+}
+
+/// Log function that prints the message to stderr.
+void _logToStderr(Level level, message) {
+ stderr.writeString('$message\n');
+}
+
+/// Log function that prints the message to stderr with the level name.
+void _logToStderrWithLabel(Level level, message) {
+ stderr.writeString('$level: $message\n');
+}

Powered by Google App Engine
This is Rietveld 408576698