aboutsummaryrefslogtreecommitdiffstats
path: root/server
diff options
context:
space:
mode:
authorSébastien Lesaint <sebastien.lesaint@sonarsource.com>2019-05-03 17:36:18 +0200
committerSonarTech <sonartech@sonarsource.com>2019-06-03 20:21:21 +0200
commite4db1c35e0bead63a16875c5f5d8e9ad149a3c4d (patch)
treece04467f2caf2cf518f45a8859636bbc7918e9c9 /server
parentb7c48f52ec450937c610242b7f3232223e3236b3 (diff)
downloadsonarqube-e4db1c35e0bead63a16875c5f5d8e9ad149a3c4d.tar.gz
sonarqube-e4db1c35e0bead63a16875c5f5d8e9ad149a3c4d.zip
SONAR-12043 SchedulerImpl correctly handle Thread interrupt
Diffstat (limited to 'server')
-rw-r--r--server/sonar-main/src/main/java/org/sonar/application/Scheduler.java2
-rw-r--r--server/sonar-main/src/main/java/org/sonar/application/SchedulerImpl.java71
-rw-r--r--server/sonar-main/src/main/java/org/sonar/application/process/ManagedProcessHandler.java8
-rw-r--r--server/sonar-main/src/test/java/org/sonar/application/process/ManagedProcessHandlerTest.java8
4 files changed, 61 insertions, 28 deletions
diff --git a/server/sonar-main/src/main/java/org/sonar/application/Scheduler.java b/server/sonar-main/src/main/java/org/sonar/application/Scheduler.java
index 1f4e19d75da..b01448b4787 100644
--- a/server/sonar-main/src/main/java/org/sonar/application/Scheduler.java
+++ b/server/sonar-main/src/main/java/org/sonar/application/Scheduler.java
@@ -21,7 +21,7 @@ package org.sonar.application;
public interface Scheduler {
- void schedule();
+ void schedule() throws InterruptedException;
/**
* Stops all processes and waits for them to be down.
diff --git a/server/sonar-main/src/main/java/org/sonar/application/SchedulerImpl.java b/server/sonar-main/src/main/java/org/sonar/application/SchedulerImpl.java
index 80b4b7f408a..b218afc3ff1 100644
--- a/server/sonar-main/src/main/java/org/sonar/application/SchedulerImpl.java
+++ b/server/sonar-main/src/main/java/org/sonar/application/SchedulerImpl.java
@@ -32,7 +32,6 @@ import org.sonar.application.command.AbstractCommand;
import org.sonar.application.command.CommandFactory;
import org.sonar.application.config.AppSettings;
import org.sonar.application.config.ClusterSettings;
-import org.sonar.application.process.ManagedProcess;
import org.sonar.application.process.ManagedProcessEventListener;
import org.sonar.application.process.ManagedProcessHandler;
import org.sonar.application.process.ManagedProcessLifecycle;
@@ -77,7 +76,7 @@ public class SchedulerImpl implements Scheduler, ManagedProcessEventListener, Pr
}
@Override
- public void schedule() {
+ public void schedule() throws InterruptedException {
if (!nodeLifecycle.tryToMoveTo(NodeLifecycle.State.STARTING)) {
return;
}
@@ -98,20 +97,20 @@ public class SchedulerImpl implements Scheduler, ManagedProcessEventListener, Pr
tryToStartAll();
}
- private void tryToStartAll() {
+ private void tryToStartAll() throws InterruptedException {
tryToStartEs();
tryToStartWeb();
tryToStartCe();
}
- private void tryToStartEs() {
+ private void tryToStartEs() throws InterruptedException {
ManagedProcessHandler process = processesById.get(ProcessId.ELASTICSEARCH);
if (process != null) {
tryToStartProcess(process, commandFactory::createEsCommand);
}
}
- private void tryToStartWeb() {
+ private void tryToStartWeb() throws InterruptedException {
ManagedProcessHandler process = processesById.get(ProcessId.WEB_SERVER);
if (process == null) {
return;
@@ -136,7 +135,7 @@ public class SchedulerImpl implements Scheduler, ManagedProcessEventListener, Pr
}
}
- private void tryToStartCe() {
+ private void tryToStartCe() throws InterruptedException {
ManagedProcessHandler process = processesById.get(ProcessId.COMPUTE_ENGINE);
if (process != null && appState.isOperational(ProcessId.WEB_SERVER, true) && isEsClientStartable()) {
tryToStartProcess(process, commandFactory::createCeCommand);
@@ -148,16 +147,17 @@ public class SchedulerImpl implements Scheduler, ManagedProcessEventListener, Pr
return appState.isOperational(ProcessId.ELASTICSEARCH, requireLocalEs);
}
- private void tryToStartProcess(ManagedProcessHandler process, Supplier<AbstractCommand> commandSupplier) {
- tryToStart(process, () -> {
+ private void tryToStartProcess(ManagedProcessHandler processHandler, Supplier<AbstractCommand> commandSupplier) throws InterruptedException {
+ // starter or restarter thread was interrupted, we should not proceed with starting the process
+ if (Thread.currentThread().isInterrupted()) {
+ throw new InterruptedException();
+ }
+
+ try {
+ processHandler.start(() -> {
AbstractCommand command = commandSupplier.get();
return processLauncher.launch(command);
});
- }
-
- private void tryToStart(ManagedProcessHandler process, Supplier<ManagedProcess> processMonitorSupplier) {
- try {
- process.start(processMonitorSupplier);
} catch (RuntimeException e) {
// failed to start command -> stop everything
hardStop();
@@ -165,7 +165,7 @@ public class SchedulerImpl implements Scheduler, ManagedProcessEventListener, Pr
}
}
- private void hardStopAll() {
+ private void hardStopAll() throws InterruptedException {
// order is important for non-cluster mode
hardStopProcess(ProcessId.COMPUTE_ENGINE);
hardStopProcess(ProcessId.WEB_SERVER);
@@ -175,8 +175,10 @@ public class SchedulerImpl implements Scheduler, ManagedProcessEventListener, Pr
/**
* Request for quick stop then blocks until process is stopped.
* Returns immediately if the process is disabled in configuration.
+ *
+ * @throws InterruptedException if {@link ManagedProcessHandler#hardStop()} throws a {@link InterruptedException}
*/
- private void hardStopProcess(ProcessId processId) {
+ private void hardStopProcess(ProcessId processId) throws InterruptedException {
ManagedProcessHandler process = processesById.get(processId);
if (process != null) {
process.hardStop();
@@ -191,12 +193,19 @@ public class SchedulerImpl implements Scheduler, ManagedProcessEventListener, Pr
if (nodeLifecycle.tryToMoveTo(NodeLifecycle.State.STOPPING)) {
LOG.info("Stopping SonarQube");
}
- hardStopAll();
- if (hardStopperThread != null && Thread.currentThread() != hardStopperThread) {
- hardStopperThread.interrupt();
- }
- if (restarterThread != null && Thread.currentThread() != restarterThread) {
- restarterThread.interrupt();
+ try {
+ hardStopAll();
+ if (hardStopperThread != null && Thread.currentThread() != hardStopperThread) {
+ hardStopperThread.interrupt();
+ }
+ if (restarterThread != null && Thread.currentThread() != restarterThread) {
+ restarterThread.interrupt();
+ }
+ } catch (InterruptedException e) {
+ // ignore and assume SQ stop is handled by another thread
+ LOG.debug("Stopping all processes was interrupted in the middle of a hard stop" +
+ " (current thread name is \"{}\")", Thread.currentThread().getName());
+ Thread.currentThread().interrupt();
}
awaitTermination.countDown();
}
@@ -231,7 +240,14 @@ public class SchedulerImpl implements Scheduler, ManagedProcessEventListener, Pr
@Override
public void onAppStateOperational(ProcessId processId) {
if (nodeLifecycle.getState() == NodeLifecycle.State.STARTING) {
- tryToStartAll();
+ try {
+ tryToStartAll();
+ } catch (InterruptedException e) {
+ // startup process was interrupted, let's assume it means shutdown was requested
+ LOG.debug("Startup process was interrupted on notification that process [{}] was operation", processId.getKey(), e);
+ hardStopAsync();
+ Thread.currentThread().interrupt();
+ }
}
}
@@ -293,6 +309,10 @@ public class SchedulerImpl implements Scheduler, ManagedProcessEventListener, Pr
try {
appReloader.reload(settings);
schedule();
+ } catch (InterruptedException e) {
+ // restart was interrupted, most likely by a stop thread, restart must be aborted
+ LOG.debug("{} thread was interrupted", getName(), e);
+ super.interrupt();
} catch (Exception e) {
LOG.error("Fail to restart", e);
hardStop();
@@ -307,7 +327,12 @@ public class SchedulerImpl implements Scheduler, ManagedProcessEventListener, Pr
@Override
public void run() {
- hardStopAll();
+ try {
+ hardStopAll();
+ } catch (InterruptedException e) {
+ LOG.debug("{} thread was interrupted", getName(), e);
+ interrupt();
+ }
}
}
}
diff --git a/server/sonar-main/src/main/java/org/sonar/application/process/ManagedProcessHandler.java b/server/sonar-main/src/main/java/org/sonar/application/process/ManagedProcessHandler.java
index 720a3802823..791bc351ae4 100644
--- a/server/sonar-main/src/main/java/org/sonar/application/process/ManagedProcessHandler.java
+++ b/server/sonar-main/src/main/java/org/sonar/application/process/ManagedProcessHandler.java
@@ -98,7 +98,7 @@ public class ManagedProcessHandler {
* Sends kill signal and awaits termination. No guarantee that process is gracefully terminated (=shutdown hooks
* executed). It depends on OS.
*/
- public void hardStop() {
+ public void hardStop() throws InterruptedException {
if (lifecycle.tryToMoveTo(ManagedProcessLifecycle.State.HARD_STOPPING)) {
hardStopImpl();
if (process != null && process.isAlive()) {
@@ -123,7 +123,7 @@ public class ManagedProcessHandler {
}
}
- private void hardStopImpl() {
+ private void hardStopImpl() throws InterruptedException {
if (process == null) {
return;
}
@@ -132,8 +132,10 @@ public class ManagedProcessHandler {
process.waitFor(hardStopTimeout.getDuration(), hardStopTimeout.getUnit());
} catch (InterruptedException e) {
// can't wait for the termination of process. Let's assume it's down.
- LOG.warn("Interrupted while hard stopping process {}", processId, e);
+ String errorMessage = format("Interrupted while hard stopping process %s", processId);
+ LOG.warn(errorMessage, e);
Thread.currentThread().interrupt();
+ throw new InterruptedException(errorMessage);
} catch (Throwable e) {
LOG.error("Failed while asking for hard stop of process {}", processId, e);
}
diff --git a/server/sonar-main/src/test/java/org/sonar/application/process/ManagedProcessHandlerTest.java b/server/sonar-main/src/test/java/org/sonar/application/process/ManagedProcessHandlerTest.java
index 60e3f8285dd..882ac36a1f9 100644
--- a/server/sonar-main/src/test/java/org/sonar/application/process/ManagedProcessHandlerTest.java
+++ b/server/sonar-main/src/test/java/org/sonar/application/process/ManagedProcessHandlerTest.java
@@ -191,7 +191,13 @@ public class ManagedProcessHandlerTest {
try (TestManagedProcess testProcess = new TestManagedProcess()) {
underTest.start(() -> testProcess);
- Thread stopperThread = new Thread(underTest::hardStop);
+ Thread stopperThread = new Thread(() -> {
+ try {
+ underTest.hardStop();
+ } catch (InterruptedException e) {
+ throw new RuntimeException(e);
+ }
+ });
stopperThread.start();
// thread is blocked until process stopped