org.gradle.internal.operations.notify.BuildOperationNotificationBridge Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of gradle-api Show documentation
Show all versions of gradle-api Show documentation
Gradle 6.9.1 API redistribution.
/*
* 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.Project;
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.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;
public class BuildOperationNotificationBridge {
private static final Logger LOGGER = LoggerFactory.getLogger(BuildOperationNotificationBridge.class);
private final BuildOperationListenerManager buildOperationListenerManager;
private final ListenerManager listenerManager;
private class State {
private ReplayAndAttachListener replayAndAttachListener = new ReplayAndAttachListener();
private 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;
}
}
};
// Listen for the end of configuration of the root project of the root build,
// and discard buffered notifications if no listeners have yet appeared.
// This avoids buffering until the end of the build when no listener comes.
private final BuildListener buildListener = new InternalBuildAdapter() {
public void buildStarted(@SuppressWarnings("NullableProblems") Gradle gradle) {
if (gradle.getParent() == null) {
gradle.rootProject(new InternalAction() {
@Override
public void execute(@SuppressWarnings("NullableProblems") Project project) {
project.afterEvaluate(new InternalAction() {
@Override
public void execute(@SuppressWarnings("NullableProblems") Project project) {
State s = state;
if (s != null && s.notificationListener == null) {
valve.stop();
}
}
});
}
});
}
}
};
private final BuildOperationNotificationListenerRegistrar registrar = new BuildOperationNotificationListenerRegistrar() {
@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 BuildOperationNotificationBridge(BuildOperationListenerManager buildOperationListenerManager, ListenerManager listenerManager) {
this.buildOperationListenerManager = buildOperationListenerManager;
this.listenerManager = listenerManager;
listenerManager.addListener(buildListener);
}
public BuildOperationNotificationListenerRegistrar getRegistrar() {
return registrar;
}
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