org.gradle.internal.operations.notify.BuildOperationNotificationBridge Maven / Gradle / Ivy
/*
* Copyright 2017 the original author or authors.
*
* 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 org.gradle.internal.operations.notify;
import org.gradle.BuildListener;
import org.gradle.api.initialization.Settings;
import org.gradle.api.internal.InternalAction;
import org.gradle.api.invocation.Gradle;
import org.gradle.internal.InternalBuildAdapter;
import org.gradle.internal.event.ListenerManager;
import org.gradle.internal.operations.BuildOperationDescriptor;
import org.gradle.internal.operations.BuildOperationListener;
import org.gradle.internal.operations.BuildOperationListenerManager;
import org.gradle.internal.operations.OperationFinishEvent;
import org.gradle.internal.operations.OperationIdentifier;
import org.gradle.internal.operations.OperationProgressEvent;
import org.gradle.internal.operations.OperationStartEvent;
import org.gradle.internal.service.scopes.Scopes;
import org.gradle.internal.service.scopes.ServiceScope;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import javax.annotation.Nullable;
import java.util.Map;
import java.util.Queue;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentLinkedQueue;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
@ServiceScope(Scopes.BuildTree.class)
public class BuildOperationNotificationBridge implements BuildOperationNotificationListenerRegistrar {
private static final Logger LOGGER = LoggerFactory.getLogger(BuildOperationNotificationBridge.class);
private final BuildOperationListenerManager buildOperationListenerManager;
private final ListenerManager listenerManager;
private class State {
private final ReplayAndAttachListener replayAndAttachListener = new ReplayAndAttachListener();
private final BuildOperationListener buildOperationListener = new Adapter(replayAndAttachListener);
private BuildOperationNotificationListener notificationListener;
private void assignSingleListener(BuildOperationNotificationListener notificationListener) {
if (this.notificationListener != null) {
throw new IllegalStateException("listener is already registered (implementation class " + this.notificationListener.getClass().getName() + ")");
}
this.notificationListener = notificationListener;
}
private void stop() {
buildOperationListenerManager.removeListener(state.buildOperationListener);
listenerManager.removeListener(buildListener);
}
}
private State state;
private final BuildOperationNotificationValve valve = new BuildOperationNotificationValve() {
@Override
public void start() {
if (state != null) {
throw new IllegalStateException("build operation notification valve already started");
}
state = new State();
buildOperationListenerManager.addListener(state.buildOperationListener);
}
@Override
public void stop() {
if (state != null) {
state.stop();
state = null;
}
}
};
// Notification listeners are expected to register before projectsLoaded.
// This avoids buffering until the end of the build when no listener comes.
private final BuildListener buildListener = new InternalBuildAdapter() {
@Override
public void beforeSettings(Settings settings) {
if (settings.getGradle().getParent() == null) {
settings.getGradle().projectsLoaded((InternalAction) project -> {
State s = state;
if (s != null && s.notificationListener == null) {
valve.stop();
}
});
}
}
};
public BuildOperationNotificationBridge(BuildOperationListenerManager buildOperationListenerManager, ListenerManager listenerManager) {
this.buildOperationListenerManager = buildOperationListenerManager;
this.listenerManager = listenerManager;
listenerManager.addListener(buildListener);
}
@Override
public void register(BuildOperationNotificationListener listener) {
State state = requireState();
state.assignSingleListener(listener);
state.replayAndAttachListener.attach(listener);
}
private State requireState() {
State s = state;
if (s == null) {
throw new IllegalStateException("state is null");
}
return s;
}
public BuildOperationNotificationValve getValve() {
return valve;
}
/*
Note: the intention here is to work towards not having to create new objects
to meet the notification object interfaces.
Instead, the base types like BuildOperationDescriptor should implement them natively.
However, this will require restructuring this type and associated things such as
OperationStartEvent. This will happen later.
*/
private static class Adapter implements BuildOperationListener {
private final BuildOperationNotificationListener notificationListener;
private final Map parents = new ConcurrentHashMap<>();
private final Map active = new ConcurrentHashMap<>();
private Adapter(BuildOperationNotificationListener notificationListener) {
this.notificationListener = notificationListener;
}
@Override
public void started(BuildOperationDescriptor buildOperation, OperationStartEvent startEvent) {
OperationIdentifier id = buildOperation.getId();
OperationIdentifier parentId = buildOperation.getParentId();
if (parentId != null) {
if (active.containsKey(parentId)) {
parents.put(id, parentId);
} else {
parentId = parents.get(parentId);
if (parentId != null) {
parents.put(id, parentId);
}
}
}
if (buildOperation.getDetails() == null) {
return;
}
active.put(id, "");
Started notification = new Started(startEvent.getStartTime(), id, parentId, buildOperation.getDetails());
try {
notificationListener.started(notification);
} catch (Throwable e) {
LOGGER.debug("Build operation notification listener threw an error on " + notification, e);
maybeThrow(e);
}
}
private void maybeThrow(Throwable e) {
if (e instanceof Error && !(e instanceof LinkageError)) {
throw (Error) e;
}
}
@Override
public void progress(OperationIdentifier buildOperationId, OperationProgressEvent progressEvent) {
Object details = progressEvent.getDetails();
if (details == null) {
return;
}
// Find the nearest parent up that we care about and use that as the parent.
OperationIdentifier owner = findOwner(buildOperationId);
if (owner == null) {
return;
}
notificationListener.progress(new Progress(owner, progressEvent.getTime(), details));
}
private OperationIdentifier findOwner(OperationIdentifier id) {
if (active.containsKey(id)) {
return id;
} else {
return parents.get(id);
}
}
@Override
public void finished(BuildOperationDescriptor buildOperation, OperationFinishEvent finishEvent) {
OperationIdentifier id = buildOperation.getId();
OperationIdentifier parentId = parents.remove(id);
if (active.remove(id) == null) {
return;
}
Finished notification = new Finished(finishEvent.getEndTime(), id, parentId, buildOperation.getDetails(), finishEvent.getResult(), finishEvent.getFailure());
try {
notificationListener.finished(notification);
} catch (Throwable e) {
LOGGER.debug("Build operation notification listener threw an error on " + notification, e);
maybeThrow(e);
}
}
}
private static class RecordingListener implements BuildOperationNotificationListener {
private final Queue
© 2015 - 2025 Weber Informatics LLC | Privacy Policy