org.openrewrite.ExecutionContext Maven / Gradle / Ivy
Show all versions of rewrite-core Show documentation
/*
* Copyright 2020 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
*
* https://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.openrewrite;
import org.jspecify.annotations.Nullable;
import org.openrewrite.scheduling.RecipeRunCycle;
import java.util.*;
import java.util.function.BiConsumer;
import java.util.function.BiFunction;
import java.util.function.Consumer;
import java.util.function.Supplier;
import static java.util.Objects.requireNonNull;
/**
* Passes messages between individual visitors or parsing operations and allows errors to be propagated
* back to the process controlling parsing or recipe execution.
*/
public interface ExecutionContext {
String CURRENT_CYCLE = "org.openrewrite.currentCycle";
String CURRENT_RECIPE = "org.openrewrite.currentRecipe";
String DATA_TABLES = "org.openrewrite.dataTables";
String RUN_TIMEOUT = "org.openrewrite.runTimeout";
String REQUIRE_PRINT_EQUALS_INPUT = "org.openrewrite.requirePrintEqualsInput";
@Incubating(since = "7.20.0")
default ExecutionContext addObserver(TreeObserver.Subscription observer) {
putMessageInCollection("org.openrewrite.internal.treeObservers", observer,
() -> Collections.newSetFromMap(new IdentityHashMap<>()));
return this;
}
@Incubating(since = "7.20.0")
default Set getObservers() {
return getMessage("org.openrewrite.internal.treeObservers", Collections.emptySet());
}
void putMessage(String key, @Nullable Object value);
@Nullable T getMessage(String key);
default T computeMessage(String key, V value, Supplier defaultValue, BiFunction remappingFunction) {
T oldMessage = getMessage(key);
if (oldMessage == null) {
oldMessage = defaultValue.get();
}
T newMessage = remappingFunction.apply(value, oldMessage);
putMessage(key, newMessage);
return newMessage;
}
default > C putMessageInCollection(String key, V value, Supplier newCollection) {
return computeMessage(key, value, newCollection, (v, acc) -> {
C c = newCollection.get();
c.addAll(acc);
c.add(value);
return c;
});
}
@SuppressWarnings("unused")
default Set putMessageInSet(String key, T value) {
return putMessageInCollection(key, value, HashSet::new);
}
default T getMessage(String key, @Nullable T defaultValue) {
T t = getMessage(key);
//noinspection DataFlowIssue
return t == null ? defaultValue : t;
}
@Nullable T pollMessage(String key);
@SuppressWarnings("unused")
default T pollMessage(String key, T defaultValue) {
T t = pollMessage(key);
return t == null ? defaultValue : t;
}
@SuppressWarnings("unused")
default void putCurrentRecipe(Recipe recipe) {
putMessage(CURRENT_RECIPE, recipe);
}
Consumer getOnError();
BiConsumer getOnTimeout();
default int getCycle() {
return getCycleDetails().getCycle();
}
default RecipeRunCycle> getCycleDetails() {
return requireNonNull(getMessage(CURRENT_CYCLE));
}
}