Skip to content

Commit

Permalink
Avoid making IsolatedDevModeMain and allegates static for no good reason
Browse files Browse the repository at this point in the history
Also avoid a reference to IsolatedDevModeMain from
ApplicationLifecycleManager.
  • Loading branch information
gsmet committed Jul 9, 2024
1 parent 224fb37 commit 485f45d
Show file tree
Hide file tree
Showing 6 changed files with 76 additions and 50 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -40,7 +40,7 @@ public class DevModeMain implements Closeable {

private final DevModeContext context;

private static volatile CuratedApplication curatedApplication;
private volatile CuratedApplication curatedApplication;
private Closeable realCloseable;

public DevModeMain(DevModeContext context) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@
import java.util.ServiceLoader;
import java.util.Set;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.atomic.AtomicReference;
import java.util.function.BiConsumer;
import java.util.function.BiFunction;
import java.util.function.Consumer;
Expand Down Expand Up @@ -61,16 +62,16 @@ public class IsolatedDevModeMain implements BiConsumer<CuratedApplication, Map<S
private volatile DevModeContext context;

private final List<HotReplacementSetup> hotReplacementSetups = new ArrayList<>();
private static volatile RunningQuarkusApplication runner;
static volatile Throwable deploymentProblem;
private static volatile CuratedApplication curatedApplication;
private static volatile AugmentAction augmentAction;
private static volatile boolean restarting;
private static volatile boolean firstStartCompleted;
private static final CountDownLatch shutdownLatch = new CountDownLatch(1);
private volatile RunningQuarkusApplication runner;
final AtomicReference<Throwable> deploymentProblem = new AtomicReference<>();
private volatile CuratedApplication curatedApplication;
private volatile AugmentAction augmentAction;
private volatile boolean restarting;
private volatile boolean firstStartCompleted;
private final CountDownLatch shutdownLatch = new CountDownLatch(1);
private Thread shutdownThread;
private CodeGenWatcher codeGenWatcher;
private static volatile ConsoleStateManager.ConsoleContext consoleContext;
private volatile ConsoleStateManager.ConsoleContext consoleContext;
private final List<DevModeListener> listeners = new ArrayList<>();

private synchronized void firstStart() {
Expand All @@ -85,30 +86,7 @@ private synchronized void firstStart() {
//TODO: look at implementing a common core classloader, that removes the need for this sort of crappy hack
curatedApplication.getOrCreateBaseRuntimeClassLoader().loadClass(ApplicationLifecycleManager.class.getName())
.getMethod("setDefaultExitCodeHandler", Consumer.class)
.invoke(null, new Consumer<Integer>() {
@Override
public void accept(Integer integer) {
if (restarting || ApplicationLifecycleManager.isVmShuttingDown()
|| context.isAbortOnFailedStart() ||
context.isTest()) {
return;
}
if (consoleContext == null) {
consoleContext = ConsoleStateManager.INSTANCE
.createContext("Completed Application");
}
//this sucks, but when we get here logging is gone
//so we just setup basic console logging
InitialConfigurator.DELAYED_HANDLER.addHandler(new ConsoleHandler(
ConsoleHandler.Target.SYSTEM_OUT,
new ColorPatternFormatter("%d{yyyy-MM-dd HH:mm:ss,SSS} %-5p [%c{3.}] (%t) %s%e%n")));
consoleContext.reset(new ConsoleCommand(' ', "Restarts the application", "to restart", 0, null,
() -> {
consoleContext.reset();
RuntimeUpdatesProcessor.INSTANCE.doScan(true, true);
}));
}
});
.invoke(null, getExitCodeHandler());

StartupAction start = augmentAction.createInitialRuntimeApplication();

Expand All @@ -127,7 +105,7 @@ public void accept(Integer integer) {
rootCause = rootCause.getCause();
}
if (!(rootCause instanceof BindException)) {
deploymentProblem = t;
deploymentProblem.set(t);
if (!context.isAbortOnFailedStart()) {
//we need to set this here, while we still have the correct TCCL
//this is so the config is still valid, and we can read HTTP config from application.properties
Expand Down Expand Up @@ -174,6 +152,35 @@ public void accept(Integer integer) {
}
}

private Consumer<Integer> getExitCodeHandler() {
if (context.isTest() || context.isAbortOnFailedStart()) {
return TestExitCodeHandler.INSTANCE;
}

return new Consumer<Integer>() {
@Override
public void accept(Integer integer) {
if (restarting || ApplicationLifecycleManager.isVmShuttingDown()) {
return;
}
if (consoleContext == null) {
consoleContext = ConsoleStateManager.INSTANCE
.createContext("Completed Application");
}
//this sucks, but when we get here logging is gone
//so we just setup basic console logging
InitialConfigurator.DELAYED_HANDLER.addHandler(new ConsoleHandler(
ConsoleHandler.Target.SYSTEM_OUT,
new ColorPatternFormatter("%d{yyyy-MM-dd HH:mm:ss,SSS} %-5p [%c{3.}] (%t) %s%e%n")));
consoleContext.reset(new ConsoleCommand(' ', "Restarts the application", "to restart", 0, null,
() -> {
consoleContext.reset();
RuntimeUpdatesProcessor.INSTANCE.doScan(true, true);
}));
}
};
}

public void restartCallback(Set<String> changedResources, ClassScanResult result) {
restartApp(changedResources,
new ClassChangeInformation(result.changedClassNames, result.deletedClassNames, result.addedClassNames));
Expand All @@ -186,7 +193,7 @@ public synchronized void restartApp(Set<String> changedResources, ClassChangeInf
}
stop();
Timing.restart(curatedApplication.getOrCreateAugmentClassLoader());
deploymentProblem = null;
deploymentProblem.set(null);
ClassLoader old = Thread.currentThread().getContextClassLoader();
try {

Expand All @@ -200,7 +207,7 @@ public synchronized void restartApp(Set<String> changedResources, ClassChangeInf
firstStartCompleted = true;
}
} catch (Throwable t) {
deploymentProblem = t;
deploymentProblem.set(t);
Throwable rootCause = t;
while (rootCause.getCause() != null) {
rootCause = rootCause.getCause();
Expand Down Expand Up @@ -253,7 +260,7 @@ private RuntimeUpdatesProcessor setupRuntimeCompilation(DevModeContext context,
public byte[] apply(String s, byte[] bytes) {
return ClassTransformingBuildStep.transform(s, bytes);
}
}, testSupport);
}, testSupport, deploymentProblem);

for (HotReplacementSetup service : ServiceLoader.load(HotReplacementSetup.class,
curatedApplication.getOrCreateBaseRuntimeClassLoader())) {
Expand Down Expand Up @@ -350,6 +357,7 @@ public void close() {
curatedApplication.close();
curatedApplication = null;
augmentAction = null;
deploymentProblem.set(null);
} finally {
if (shutdownThread != null) {
try {
Expand Down Expand Up @@ -422,10 +430,11 @@ public void run() {
firstStart();

// doStart(false, Collections.emptySet());
if (deploymentProblem != null || RuntimeUpdatesProcessor.INSTANCE.getCompileProblem() != null) {
if (deploymentProblem.get() != null || RuntimeUpdatesProcessor.INSTANCE.getCompileProblem() != null) {
if (context.isAbortOnFailedStart()) {
Throwable throwable = deploymentProblem == null ? RuntimeUpdatesProcessor.INSTANCE.getCompileProblem()
: deploymentProblem;
Throwable throwable = deploymentProblem.get() == null
? RuntimeUpdatesProcessor.INSTANCE.getCompileProblem()
: deploymentProblem.get();

throw (throwable instanceof RuntimeException ? (RuntimeException) throwable
: new RuntimeException(throwable));
Expand Down Expand Up @@ -483,4 +492,14 @@ public boolean test(String s) {
}).produces(ApplicationClassPredicateBuildItem.class).build();
}
}

private static class TestExitCodeHandler implements Consumer<Integer> {

private static final TestExitCodeHandler INSTANCE = new TestExitCodeHandler();

@Override
public void accept(Integer exitCode) {
// do nothing
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@
import java.util.Optional;
import java.util.ServiceLoader;
import java.util.Set;
import java.util.concurrent.atomic.AtomicReference;
import java.util.function.BiConsumer;
import java.util.function.BiFunction;
import java.util.function.Function;
Expand Down Expand Up @@ -57,7 +58,7 @@ public class IsolatedRemoteDevModeMain implements BiConsumer<CuratedApplication,
private volatile DevModeContext context;

private final List<HotReplacementSetup> hotReplacementSetups = new ArrayList<>();
static volatile Throwable deploymentProblem;
private AtomicReference<Throwable> deploymentProblem = new AtomicReference<>();
static volatile RemoteDevClient remoteDevClient;
static volatile Closeable remoteDevClientSession;
private static volatile CuratedApplication curatedApplication;
Expand Down Expand Up @@ -99,7 +100,7 @@ private synchronized JarResult generateApplication() {
curatedApplication.getApplicationModel(), null);
return start.getJar();
} catch (Throwable t) {
deploymentProblem = t;
deploymentProblem.set(t);
log.error("Failed to generate Quarkus application", t);
return null;
}
Expand Down Expand Up @@ -137,7 +138,7 @@ public void accept(DevModeContext.ModuleInfo moduleInfo, String s) {
public byte[] apply(String s, byte[] bytes) {
return ClassTransformingBuildStep.transform(s, bytes);
}
}, null);
}, null, deploymentProblem);

for (HotReplacementSetup service : ServiceLoader.load(HotReplacementSetup.class,
curatedApplication.getOrCreateBaseRuntimeClassLoader())) {
Expand Down Expand Up @@ -189,6 +190,7 @@ public void close() {
}
}
} finally {
deploymentProblem.set(null);
curatedApplication.close();
}

Expand Down Expand Up @@ -248,7 +250,7 @@ public void run() {
}

private Closeable doConnect() {
return remoteDevClient.sendConnectRequest(new RemoteDevState(currentHashes, deploymentProblem),
return remoteDevClient.sendConnectRequest(new RemoteDevState(currentHashes, deploymentProblem.get()),
new Function<Set<String>, Map<String, byte[]>>() {
@Override
public Map<String, byte[]> apply(Set<String> fileNames) {
Expand Down Expand Up @@ -283,6 +285,7 @@ private RemoteDevClient.SyncResult runSync() {
Set<String> removed = new HashSet<>();
Map<String, byte[]> changed = new HashMap<>();
try {
deploymentProblem.set(null);
boolean scanResult = RuntimeUpdatesProcessor.INSTANCE.doScan(true);
if (!scanResult && !copiedStaticResources.isEmpty()) {
scanResult = true;
Expand All @@ -305,7 +308,7 @@ private RemoteDevClient.SyncResult runSync() {
currentHashes = newHashes;
}
} catch (IOException e) {
deploymentProblem = e;
deploymentProblem.set(e);
}
return new RemoteDevClient.SyncResult() {
@Override
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -36,7 +36,6 @@ public class IsolatedTestModeMain extends IsolatedDevModeMain {
private volatile DevModeContext context;

private final List<HotReplacementSetup> hotReplacementSetups = new ArrayList<>();
static volatile Throwable deploymentProblem;
private static volatile CuratedApplication curatedApplication;
private static volatile AugmentAction augmentAction;

Expand Down Expand Up @@ -68,7 +67,7 @@ public void accept(DevModeContext.ModuleInfo moduleInfo, String s) {
public byte[] apply(String s, byte[] bytes) {
return ClassTransformingBuildStep.transform(s, bytes);
}
}, testSupport);
}, testSupport, deploymentProblem);

for (HotReplacementSetup service : ServiceLoader.load(HotReplacementSetup.class,
curatedApplication.getOrCreateBaseRuntimeClassLoader())) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -40,6 +40,7 @@
import java.util.concurrent.ConcurrentSkipListSet;
import java.util.concurrent.CopyOnWriteArrayList;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicReference;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
import java.util.function.BiConsumer;
Expand Down Expand Up @@ -91,6 +92,7 @@ public class RuntimeUpdatesProcessor implements HotReplacementContext, Closeable
volatile Throwable compileProblem;
volatile Throwable testCompileProblem;
volatile Throwable hotReloadProblem;
private final AtomicReference<Throwable> deploymentProblem;

private volatile Predicate<ClassInfo> disableInstrumentationForClassPredicate = new AlwaysFalsePredicate<>();
private volatile Predicate<Index> disableInstrumentationForIndexPredicate = new AlwaysFalsePredicate<>();
Expand Down Expand Up @@ -141,7 +143,7 @@ public RuntimeUpdatesProcessor(Path applicationRoot, DevModeContext context, Qua
DevModeType devModeType, BiConsumer<Set<String>, ClassScanResult> restartCallback,
BiConsumer<DevModeContext.ModuleInfo, String> copyResourceNotification,
BiFunction<String, byte[], byte[]> classTransformers,
TestSupport testSupport) {
TestSupport testSupport, AtomicReference<Throwable> deploymentProblem) {
this.applicationRoot = applicationRoot;
this.context = context;
this.compiler = compiler;
Expand Down Expand Up @@ -180,6 +182,7 @@ public void testsDisabled() {
}
});
}
this.deploymentProblem = deploymentProblem;
}

public TestSupport getTestSupport() {
Expand Down Expand Up @@ -392,7 +395,7 @@ public List<Path> getResourcesDir() {
public Throwable getDeploymentProblem() {
//we differentiate between these internally, however for the error reporting they are the same
return compileProblem != null ? compileProblem
: IsolatedDevModeMain.deploymentProblem != null ? IsolatedDevModeMain.deploymentProblem
: deploymentProblem.get() != null ? deploymentProblem.get()
: hotReloadProblem;
}

Expand Down Expand Up @@ -535,7 +538,7 @@ public boolean doScan(boolean userInitiated, boolean forceRestart) {
//all broken we just assume the reason that they have refreshed is because they have fixed something
//trying to watch all resource files is complex and this is likely a good enough solution for what is already an edge case
boolean restartNeeded = !instrumentationChange && (changedClassResults.isChanged()
|| (IsolatedDevModeMain.deploymentProblem != null && userInitiated) || fileRestartNeeded);
|| (deploymentProblem.get() != null && userInitiated) || fileRestartNeeded);
if (restartNeeded) {
String changeString = changedFilesForRestart.stream().map(Path::getFileName).map(Object::toString)
.collect(Collectors.joining(", "));
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -225,6 +225,7 @@ public static void run(Application application, Class<? extends QuarkusApplicati
if (!alreadyStarted) {
application.stop(); //this could have already been called
}
currentApplication = null;
(exitCodeHandler == null ? defaultExitCodeHandler : exitCodeHandler).accept(getExitCode(), null); //this may not be called if shutdown was initiated by a signal
}

Expand Down Expand Up @@ -435,6 +436,7 @@ public void run() {
currentApplication.stop();
}
currentApplication.awaitShutdown();
currentApplication = null;
System.out.flush();
System.err.flush();
}
Expand Down

0 comments on commit 485f45d

Please sign in to comment.