blob: d1f3e058ac55f96a16d9777ae826ef186e54f8bc [file] [log] [blame]
// Copyright 2014 The Bazel Authors. All rights reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package com.google.devtools.build.lib.events;
import static com.google.common.base.Preconditions.checkNotNull;
import com.google.common.eventbus.EventBus;
import com.google.devtools.build.lib.util.io.OutErr;
import java.io.PrintStream;
import java.util.HashSet;
import java.util.concurrent.ConcurrentLinkedQueue;
/**
* The reporter is the primary means of reporting events such as errors, warnings, progress
* information and diagnostic information to the user. It is not intended as a logging mechanism for
* developer-only messages; use a Logger for that.
*
* <p>The reporter instance is consumed by the build system, and passes events using {@link
* #handle(Event)} or {@link #post(Postable)} to {@link EventHandler} instances. The latter only
* occurs to the {@link EventHandler} instances that are also {@link ExtendedEventHandler}
* instances. Additionally, when events are passed using {@link #post(Postable)} they are also
* posted to the {@link EventBus} registered in this reporter.
*
* <p>The reporter's main use is in the blaze runtime and its lifetime is the lifetime of the blaze
* server.
*
* <p>Thread-safe: calls to {@code #report} may be made on any thread. Handlers may be run in an
* arbitrary thread (but right now, they will not be run concurrently).
*/
public final class Reporter implements ExtendedEventHandler, ExceptionListener {
/** Set of {@link EventHandler} registered in this reporter. */
private final ConcurrentLinkedQueue<EventHandler> eventHandlers = new ConcurrentLinkedQueue<>();
/**
* {@link EventBus} registered in this reporter. Calls to {@link #post(Postable)} will propagate
* events to the event bus.
*/
private EventBus eventBus;
/** An OutErr that sends all of its output to this Reporter.
* Each write will (when flushed) get mapped to an EventKind.STDOUT or EventKind.STDERR event.
*/
private final OutErr outErrToReporter = outErrForReporter(this);
private volatile OutputFilter outputFilter = OutputFilter.OUTPUT_EVERYTHING;
private EventHandler ansiAllowingHandler;
private EventHandler ansiStrippingHandler;
private boolean ansiAllowingHandlerRegistered;
private final HashSet<String> eventsShown = new HashSet<>();
/**
* The tag that indicates to the reporter to show this event exactly once, regardless of the
* output filter, and to suppress this event if it's a duplicate of another event with this tag.
*/
public static final String SHOW_ONCE_TAG = "showOnce";
public Reporter(EventBus eventBus) {
this.eventBus = eventBus;
}
/**
* A copy constructor, to make it convenient to replicate a reporter config for temporary
* configuration changes.
*/
public Reporter(Reporter template) {
eventHandlers.addAll(template.eventHandlers);
this.eventBus = template.eventBus;
}
/** Constructor which configures a reporter with the specified handlers. */
public Reporter(EventBus eventBus, EventHandler... handlers) {
this.eventBus = eventBus;
for (EventHandler handler : handlers) {
addHandler(handler);
}
}
public static OutErr outErrForReporter(EventHandler rep) {
return OutErr.create(
// We don't use BufferedOutputStream here, because in general the Blaze
// code base assumes that the output streams are not buffered.
new ReporterStream(rep, EventKind.STDOUT), new ReporterStream(rep, EventKind.STDERR));
}
/**
* Returns an OutErr that sends all of its output to this Reporter. Each write to the OutErr will
* cause an EventKind.STDOUT or EventKind.STDERR event.
*/
public OutErr getOutErr() {
return outErrToReporter;
}
/** Registers an {@link EventHandler} in this reporter. */
public void addHandler(EventHandler handler) {
checkNotNull(handler);
eventHandlers.add(handler);
}
/**
* Removes an {@link EventHandler} from this reporter. If the handler wasn't registered in this
* reporter this method is a no-op.
*/
public void removeHandler(EventHandler handler) {
checkNotNull(handler);
eventHandlers.remove(handler);
}
/**
* Handle the provided {@link Event} using all the {@link EventHandler} registered in this
* reporter.
*/
@Override
public void handle(Event e) {
if (e.getKind() != EventKind.ERROR
&& e.getKind() != EventKind.DEBUG
&& e.getTag() != null
&& !e.getTag().equals(SHOW_ONCE_TAG)
&& !showOutput(e.getTag())) {
return;
}
if (SHOW_ONCE_TAG.equals(e.getTag())) {
if (eventsShown.contains(e.toString())) {
return;
}
eventsShown.add(e.toString());
}
for (EventHandler handler : eventHandlers) {
handler.handle(e);
}
}
/**
* Post the provided {@link com.google.devtools.build.lib.events.ExtendedEventHandler.Postable} to
* the {@link EventBus} and all the {@link ExtendedEventHandler} registered in this reporter.
*/
@Override
public void post(ExtendedEventHandler.Postable obj) {
if (eventBus != null) {
eventBus.post(obj);
}
for (EventHandler eventHandler : eventHandlers) {
if (eventHandler instanceof ExtendedEventHandler) {
((ExtendedEventHandler) eventHandler).post(obj);
}
}
}
public void clearEventBus() {
eventBus = null;
}
/**
* Reports the start of a particular task.
* Is a wrapper around report() with event kind START.
* Should always be matched by a corresponding call to finishTask()
* with the same message, except that the leading percentage
* progress indicator (if any) in the message may differ.
*/
public void startTask(Location location, String message) {
handle(Event.of(EventKind.START, location, message));
}
/**
* Reports the end of a particular task.
* Is a wrapper around report() with event kind FINISH.
* Should always be matched by a corresponding call to startTask()
* with the same message, except that the leading percentage
* progress indicator (if any) in the message may differ.
*/
public void finishTask(Location location, String message) {
handle(Event.of(EventKind.FINISH, location, message));
}
@Override
public void error(Location location, String message, Throwable error) {
handle(Event.error(location, message));
error.printStackTrace(new PrintStream(getOutErr().getErrorStream()));
}
/**
* Returns true iff the given tag matches the output filter.
*/
public boolean showOutput(String tag) {
return outputFilter.showOutput(tag);
}
public void setOutputFilter(OutputFilter outputFilter) {
this.outputFilter = outputFilter;
}
/**
* Registers an ANSI-control-code-allowing EventHandler with an ANSI-stripping EventHandler
* that is already registered with the reporter. The ANSI-stripping handler can then be replaced
* with the ANSI-allowing handler by calling {@code #switchToAnsiAllowingHandler} which
* calls {@code removeHandler} for the ANSI-stripping handler and then {@code addHandler} for the
* ANSI-allowing handler.
*/
public synchronized void registerAnsiAllowingHandler(
EventHandler ansiStrippingHandler,
EventHandler ansiAllowingHandler) {
this.ansiAllowingHandler = ansiAllowingHandler;
this.ansiStrippingHandler = ansiStrippingHandler;
ansiAllowingHandlerRegistered = true;
}
/**
* Restores the ANSI-allowing EventHandler registered using
* {@link #registerAnsiAllowingHandler}.
*/
public synchronized void switchToAnsiAllowingHandler() {
if (ansiAllowingHandlerRegistered) {
removeHandler(ansiStrippingHandler);
addHandler(ansiAllowingHandler);
ansiStrippingHandler = null;
ansiAllowingHandler = null;
ansiAllowingHandlerRegistered = false;
}
}
}