summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorSébastien Lesaint <sebastien.lesaint@sonarsource.com>2016-01-13 11:24:01 +0100
committerSébastien Lesaint <sebastien.lesaint@sonarsource.com>2016-01-18 11:16:44 +0100
commite7c4a306ac47843029d183d8e3f5ca7730184cf2 (patch)
treed0ad97d750a92e90c60c280a856f6a2b75a1b6b5
parent5e12a8cf6011e4f95405a1f8abeee38c92245141 (diff)
downloadsonarqube-e7c4a306ac47843029d183d8e3f5ca7730184cf2.tar.gz
sonarqube-e7c4a306ac47843029d183d8e3f5ca7730184cf2.zip
SONAR-7168 fileSystem must be reset during a restart
deletion of temp directory is required for uninstalling plugins but also to cleanly start web server (eg. deleting unsuccessfully processed analysis reports)
-rw-r--r--server/sonar-process-monitor/src/main/java/org/sonar/process/monitor/FileSystem.java29
-rw-r--r--server/sonar-process-monitor/src/main/java/org/sonar/process/monitor/JavaProcessLauncher.java8
-rw-r--r--server/sonar-process-monitor/src/main/java/org/sonar/process/monitor/Monitor.java70
-rw-r--r--server/sonar-process-monitor/src/test/java/org/sonar/process/monitor/JavaProcessLauncherTest.java3
-rw-r--r--server/sonar-process-monitor/src/test/java/org/sonar/process/monitor/MonitorTest.java124
-rw-r--r--sonar-application/src/main/java/org/sonar/application/App.java16
-rw-r--r--sonar-application/src/main/java/org/sonar/application/AppFileSystem.java120
-rw-r--r--sonar-application/src/main/java/org/sonar/application/PropsBuilder.java46
-rw-r--r--sonar-application/src/test/java/org/sonar/application/AppFileSystemTest.java202
-rw-r--r--sonar-application/src/test/java/org/sonar/application/PropsBuilderTest.java64
10 files changed, 507 insertions, 175 deletions
diff --git a/server/sonar-process-monitor/src/main/java/org/sonar/process/monitor/FileSystem.java b/server/sonar-process-monitor/src/main/java/org/sonar/process/monitor/FileSystem.java
new file mode 100644
index 00000000000..4efe53bf489
--- /dev/null
+++ b/server/sonar-process-monitor/src/main/java/org/sonar/process/monitor/FileSystem.java
@@ -0,0 +1,29 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2016 SonarSource SA
+ * mailto:contact AT sonarsource DOT com
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 3 of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+package org.sonar.process.monitor;
+
+import java.io.File;
+import java.io.IOException;
+
+public interface FileSystem {
+ void reset() throws IOException;
+
+ File getTempDir();
+}
diff --git a/server/sonar-process-monitor/src/main/java/org/sonar/process/monitor/JavaProcessLauncher.java b/server/sonar-process-monitor/src/main/java/org/sonar/process/monitor/JavaProcessLauncher.java
index 2f9490b1844..019d6388cd9 100644
--- a/server/sonar-process-monitor/src/main/java/org/sonar/process/monitor/JavaProcessLauncher.java
+++ b/server/sonar-process-monitor/src/main/java/org/sonar/process/monitor/JavaProcessLauncher.java
@@ -39,16 +39,20 @@ public class JavaProcessLauncher {
private final File tempDir;
private final AllProcessesCommands allProcessesCommands;
- public JavaProcessLauncher(Timeouts timeouts, File tempDir, AllProcessesCommands allProcessesCommands) {
+ public JavaProcessLauncher(Timeouts timeouts, File tempDir) {
this.timeouts = timeouts;
this.tempDir = tempDir;
- this.allProcessesCommands = allProcessesCommands;
+ this.allProcessesCommands = new AllProcessesCommands(tempDir);
}
public void close() {
allProcessesCommands.close();
}
+ public ProcessCommands getProcessCommand(int processNumber, boolean clean) {
+ return allProcessesCommands.getProcessCommand(processNumber, clean);
+ }
+
ProcessRef launch(JavaCommand command) {
Process process = null;
try {
diff --git a/server/sonar-process-monitor/src/main/java/org/sonar/process/monitor/Monitor.java b/server/sonar-process-monitor/src/main/java/org/sonar/process/monitor/Monitor.java
index 8c3e459106f..481495b7d14 100644
--- a/server/sonar-process-monitor/src/main/java/org/sonar/process/monitor/Monitor.java
+++ b/server/sonar-process-monitor/src/main/java/org/sonar/process/monitor/Monitor.java
@@ -20,6 +20,7 @@
package org.sonar.process.monitor;
import java.io.File;
+import java.io.IOException;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
@@ -28,7 +29,6 @@ import javax.annotation.CheckForNull;
import javax.annotation.Nullable;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
-import org.sonar.process.AllProcessesCommands;
import org.sonar.process.Lifecycle;
import org.sonar.process.Lifecycle.State;
import org.sonar.process.ProcessCommands;
@@ -43,32 +43,34 @@ public class Monitor {
private static int restartorInstanceCounter = 0;
- private final AllProcessesCommands allProcessesCommands;
- private final JavaProcessLauncher launcher;
-
+ private final FileSystem fileSystem;
private final SystemExit systemExit;
+ private final boolean watchForHardStop;
private final Thread shutdownHook = new Thread(new MonitorShutdownHook(), "Monitor Shutdown Hook");
private final List<WatcherThread> watcherThreads = new CopyOnWriteArrayList<>();
private final Lifecycle lifecycle = new Lifecycle();
+
private final TerminatorThread terminator = new TerminatorThread();
private final RestartRequestWatcherThread restartWatcher = new RestartRequestWatcherThread();
@CheckForNull
private List<JavaCommand> javaCommands;
@CheckForNull
+ private JavaProcessLauncher launcher;
+ @CheckForNull
private RestartorThread restartor;
@CheckForNull
HardStopWatcherThread hardStopWatcher;
static int nextProcessId = 1;
- Monitor(File tempDir, SystemExit exit) {
- this.allProcessesCommands = new AllProcessesCommands(tempDir);
- this.launcher = new JavaProcessLauncher(TIMEOUTS, tempDir, allProcessesCommands);
+ Monitor(FileSystem fileSystem, SystemExit exit, boolean watchForHardStop) {
+ this.fileSystem = fileSystem;
this.systemExit = exit;
+ this.watchForHardStop = watchForHardStop;
}
- public static Monitor create(File tempDir) {
- return new Monitor(tempDir, new SystemExit());
+ public static Monitor create(FileSystem fileSystem, boolean watchForHardStop) {
+ return new Monitor(fileSystem, new SystemExit(), watchForHardStop);
}
/**
@@ -88,6 +90,8 @@ public class Monitor {
// intercepts CTRL-C
Runtime.getRuntime().addShutdownHook(shutdownHook);
+
+ // start watching for restart requested by child process
this.restartWatcher.start();
this.javaCommands = commands;
@@ -97,12 +101,42 @@ public class Monitor {
private void startProcesses() {
// do no start any child process if not in state INIT or RESTARTING (a stop could be in progress too)
if (lifecycle.tryToMoveTo(State.STARTING)) {
+ resetFileSystem();
+
startAndMonitorProcesses();
stopIfAnyProcessDidNotStart();
+ watchForHardStop();
+ }
+ }
+
+ private void watchForHardStop() {
+ if (!watchForHardStop) {
+ return;
+ }
+
+ if (this.hardStopWatcher != null) {
+ this.hardStopWatcher.stopWatching();
+ awaitTermination(this.hardStopWatcher);
+ }
+ ProcessCommands processCommand = this.launcher.getProcessCommand(CURRENT_PROCESS_NUMBER, true);
+ this.hardStopWatcher = new HardStopWatcherThread(processCommand);
+ this.hardStopWatcher.start();
+ }
+
+ private void resetFileSystem() {
+ // since JavaLauncher depends on temp directory, which is reset below, we need to close it first
+ closeJavaLauncher();
+ try {
+ fileSystem.reset();
+ } catch (IOException e) {
+ // failed to reset FileSystem
+ throw new RuntimeException("Failed to reset file system", e);
}
}
private void startAndMonitorProcesses() {
+ File tempDir = fileSystem.getTempDir();
+ this.launcher = new JavaProcessLauncher(TIMEOUTS, tempDir);
for (JavaCommand command : javaCommands) {
try {
ProcessRef processRef = launcher.launch(command);
@@ -115,10 +149,11 @@ public class Monitor {
}
}
- public void watchForHardStop() {
- ProcessCommands processCommand = this.allProcessesCommands.getProcessCommand(CURRENT_PROCESS_NUMBER, true);
- this.hardStopWatcher = new HardStopWatcherThread(processCommand);
- this.hardStopWatcher.start();
+ private void closeJavaLauncher() {
+ if (this.launcher != null) {
+ this.launcher.close();
+ this.launcher = null;
+ }
}
private void monitor(ProcessRef processRef) {
@@ -213,7 +248,7 @@ public class Monitor {
Runtime.getRuntime().removeShutdownHook(shutdownHook);
}
// cleanly close JavaLauncher
- launcher.close();
+ closeJavaLauncher();
}
}
@@ -227,6 +262,7 @@ public class Monitor {
}
private void stopAsync(State stoppingState) {
+ assert stoppingState == State.STOPPING || stoppingState == State.HARD_STOPPING;
if (lifecycle.tryToMoveTo(stoppingState)) {
terminator.start();
}
@@ -310,6 +346,7 @@ public class Monitor {
public class HardStopWatcherThread extends Thread {
private final ProcessCommands processCommands;
+ private boolean watch = true;
public HardStopWatcherThread(ProcessCommands processCommands) {
super("Hard stop watcher");
@@ -318,7 +355,7 @@ public class Monitor {
@Override
public void run() {
- while (lifecycle.getState() != Lifecycle.State.STOPPED) {
+ while (watch && lifecycle.getState() != Lifecycle.State.STOPPED) {
if (processCommands.askedForStop()) {
trace("Stopping process");
Monitor.this.stop();
@@ -332,6 +369,9 @@ public class Monitor {
}
}
+ public void stopWatching() {
+ this.watch = false;
+ }
}
private void stopProcesses() {
diff --git a/server/sonar-process-monitor/src/test/java/org/sonar/process/monitor/JavaProcessLauncherTest.java b/server/sonar-process-monitor/src/test/java/org/sonar/process/monitor/JavaProcessLauncherTest.java
index 20053415c15..af32f3f759b 100644
--- a/server/sonar-process-monitor/src/test/java/org/sonar/process/monitor/JavaProcessLauncherTest.java
+++ b/server/sonar-process-monitor/src/test/java/org/sonar/process/monitor/JavaProcessLauncherTest.java
@@ -23,7 +23,6 @@ import java.io.File;
import org.junit.Rule;
import org.junit.Test;
import org.junit.rules.TemporaryFolder;
-import org.sonar.process.AllProcessesCommands;
import static org.assertj.core.api.Assertions.assertThat;
import static org.junit.Assert.fail;
@@ -37,7 +36,7 @@ public class JavaProcessLauncherTest {
public void fail_to_launch() throws Exception {
File tempDir = temp.newFolder();
JavaCommand command = new JavaCommand("test");
- JavaProcessLauncher launcher = new JavaProcessLauncher(new Timeouts(), tempDir, new AllProcessesCommands(tempDir));
+ JavaProcessLauncher launcher = new JavaProcessLauncher(new Timeouts(), tempDir);
try {
// command is not correct (missing options), java.lang.ProcessBuilder#start()
// throws an exception
diff --git a/server/sonar-process-monitor/src/test/java/org/sonar/process/monitor/MonitorTest.java b/server/sonar-process-monitor/src/test/java/org/sonar/process/monitor/MonitorTest.java
index 713ace625dc..ea25a399cfe 100644
--- a/server/sonar-process-monitor/src/test/java/org/sonar/process/monitor/MonitorTest.java
+++ b/server/sonar-process-monitor/src/test/java/org/sonar/process/monitor/MonitorTest.java
@@ -45,16 +45,23 @@ import org.sonar.process.NetworkUtils;
import org.sonar.process.ProcessCommands;
import org.sonar.process.SystemExit;
+import static java.util.Collections.singletonList;
import static org.assertj.core.api.Assertions.assertThat;
import static org.junit.Assert.fail;
import static org.mockito.Mockito.mock;
+import static org.mockito.Mockito.times;
+import static org.mockito.Mockito.verify;
+import static org.mockito.Mockito.when;
import static org.sonar.process.monitor.MonitorTest.HttpProcessClientAssert.assertThat;
public class MonitorTest {
- static File testJar;
- Monitor monitor;
- SystemExit exit = mock(SystemExit.class);
+ private static File testJar;
+
+ private FileSystem fileSystem = mock(FileSystem.class);
+ private SystemExit exit = mock(SystemExit.class);
+
+ private Monitor underTest;
/**
* Find the JAR file containing the test apps. Classes can't be moved in sonar-process-monitor because
@@ -105,8 +112,8 @@ public class MonitorTest {
@After
public void tearDown() {
try {
- if (monitor != null) {
- monitor.stop();
+ if (underTest != null) {
+ underTest.stop();
}
} catch (Throwable ignored) {
}
@@ -114,9 +121,9 @@ public class MonitorTest {
@Test
public void fail_to_start_if_no_commands() throws Exception {
- monitor = newDefaultMonitor(tempDir);
+ underTest = newDefaultMonitor(tempDir);
try {
- monitor.start(Collections.<JavaCommand>emptyList());
+ underTest.start(Collections.<JavaCommand>emptyList());
fail();
} catch (IllegalArgumentException e) {
assertThat(e).hasMessage("At least one command is required");
@@ -125,51 +132,52 @@ public class MonitorTest {
@Test
public void fail_to_start_multiple_times() throws Exception {
- monitor = newDefaultMonitor(tempDir);
- monitor.start(Arrays.asList(newStandardProcessCommand()));
+ underTest = newDefaultMonitor(tempDir);
+ underTest.start(singletonList(newStandardProcessCommand()));
boolean failed = false;
try {
- monitor.start(Arrays.asList(newStandardProcessCommand()));
+ underTest.start(singletonList(newStandardProcessCommand()));
} catch (IllegalStateException e) {
failed = e.getMessage().equals("Can not start multiple times");
}
- monitor.stop();
+ underTest.stop();
assertThat(failed).isTrue();
}
@Test
public void start_then_stop_gracefully() throws Exception {
- monitor = newDefaultMonitor(tempDir);
+ underTest = newDefaultMonitor(tempDir);
HttpProcessClient client = new HttpProcessClient(tempDir, "test");
// blocks until started
- monitor.start(Arrays.asList(client.newCommand()));
+ underTest.start(singletonList(client.newCommand()));
assertThat(client).isReady()
.wasStartedBefore(System.currentTimeMillis());
// blocks until stopped
- monitor.stop();
+ underTest.stop();
assertThat(client)
.isNotReady()
.wasGracefullyTerminated();
- assertThat(monitor.getState()).isEqualTo(State.STOPPED);
+ assertThat(underTest.getState()).isEqualTo(State.STOPPED);
+ verify(fileSystem).reset();
}
@Test
public void start_then_stop_sequence_of_commands() throws Exception {
- monitor = newDefaultMonitor(tempDir);
+ underTest = newDefaultMonitor(tempDir);
HttpProcessClient p1 = new HttpProcessClient(tempDir, "p1");
HttpProcessClient p2 = new HttpProcessClient(tempDir, "p2");
- monitor.start(Arrays.asList(p1.newCommand(), p2.newCommand()));
+ underTest.start(Arrays.asList(p1.newCommand(), p2.newCommand()));
// start p2 when p1 is fully started (ready)
assertThat(p1)
.isReady()
.wasStartedBefore(p2);
assertThat(p2)
- .isReady();
+ .isReady();
- monitor.stop();
+ underTest.stop();
// stop in inverse order
assertThat(p1)
@@ -178,68 +186,73 @@ public class MonitorTest {
assertThat(p2)
.isNotReady()
.wasGracefullyTerminatedBefore(p1);
+ verify(fileSystem).reset();
}
@Test
public void stop_all_processes_if_monitor_shutdowns() throws Exception {
- monitor = newDefaultMonitor(tempDir);
+ underTest = newDefaultMonitor(tempDir);
HttpProcessClient p1 = new HttpProcessClient(tempDir, "p1");
HttpProcessClient p2 = new HttpProcessClient(tempDir, "p2");
- monitor.start(Arrays.asList(p1.newCommand(), p2.newCommand()));
+ underTest.start(Arrays.asList(p1.newCommand(), p2.newCommand()));
assertThat(p1).isReady();
assertThat(p2).isReady();
// emulate CTRL-C
- monitor.getShutdownHook().run();
- monitor.getShutdownHook().join();
+ underTest.getShutdownHook().run();
+ underTest.getShutdownHook().join();
assertThat(p1).wasGracefullyTerminated();
assertThat(p2).wasGracefullyTerminated();
+
+ verify(fileSystem).reset();
}
@Test
public void restart_all_processes_if_one_asks_for_restart() throws Exception {
- monitor = newDefaultMonitor(tempDir);
+ underTest = newDefaultMonitor(tempDir);
HttpProcessClient p1 = new HttpProcessClient(tempDir, "p1");
HttpProcessClient p2 = new HttpProcessClient(tempDir, "p2");
- monitor.start(Arrays.asList(p1.newCommand(), p2.newCommand()));
+ underTest.start(Arrays.asList(p1.newCommand(), p2.newCommand()));
assertThat(p1).isReady();
assertThat(p2).isReady();
p2.restart();
- assertThat(monitor.waitForOneRestart()).isTrue();
+ assertThat(underTest.waitForOneRestart()).isTrue();
assertThat(p1)
- .wasStarted(2)
- .wasGracefullyTerminated(1);
+ .wasStarted(2)
+ .wasGracefullyTerminated(1);
assertThat(p2)
- .wasStarted(2)
- .wasGracefullyTerminated(1);
+ .wasStarted(2)
+ .wasGracefullyTerminated(1);
- monitor.stop();
+ underTest.stop();
assertThat(p1)
- .wasStarted(2)
- .wasGracefullyTerminated(2);
+ .wasStarted(2)
+ .wasGracefullyTerminated(2);
assertThat(p2)
- .wasStarted(2)
- .wasGracefullyTerminated(2);
+ .wasStarted(2)
+ .wasGracefullyTerminated(2);
+
+ verify(fileSystem, times(2)).reset();
}
@Test
public void stop_all_processes_if_one_shutdowns() throws Exception {
- monitor = newDefaultMonitor(tempDir);
+ underTest = newDefaultMonitor(tempDir);
HttpProcessClient p1 = new HttpProcessClient(tempDir, "p1");
HttpProcessClient p2 = new HttpProcessClient(tempDir, "p2");
- monitor.start(Arrays.asList(p1.newCommand(), p2.newCommand()));
+ underTest.start(Arrays.asList(p1.newCommand(), p2.newCommand()));
assertThat(p1.isReady()).isTrue();
assertThat(p2.isReady()).isTrue();
// kill p1 -> waiting for detection by monitor than termination of p2
p1.kill();
- monitor.awaitTermination();
+ underTest.awaitTermination();
assertThat(p1)
.isNotReady()
@@ -247,15 +260,17 @@ public class MonitorTest {
assertThat(p2)
.isNotReady()
.wasGracefullyTerminated();
+
+ verify(fileSystem).reset();
}
@Test
public void stop_all_processes_if_one_fails_to_start() throws Exception {
- monitor = newDefaultMonitor(tempDir);
+ underTest = newDefaultMonitor(tempDir);
HttpProcessClient p1 = new HttpProcessClient(tempDir, "p1");
HttpProcessClient p2 = new HttpProcessClient(tempDir, "p2", -1);
try {
- monitor.start(Arrays.asList(p1.newCommand(), p2.newCommand()));
+ underTest.start(Arrays.asList(p1.newCommand(), p2.newCommand()));
fail();
} catch (Exception expected) {
assertThat(p1)
@@ -283,13 +298,13 @@ public class MonitorTest {
@Test
public void fail_to_start_if_bad_class_name() throws Exception {
- monitor = newDefaultMonitor(tempDir);
+ underTest = newDefaultMonitor(tempDir);
JavaCommand command = new JavaCommand("test")
.addClasspath(testJar.getAbsolutePath())
.setClassName("org.sonar.process.test.Unknown");
try {
- monitor.start(Arrays.asList(command));
+ underTest.start(singletonList(command));
fail();
} catch (Exception e) {
// expected
@@ -299,17 +314,28 @@ public class MonitorTest {
@Test
public void watchForHardStop_adds_a_hardStopWatcher_thread_and_starts_it() throws Exception {
- Monitor monitor = newDefaultMonitor(tempDir);
- assertThat(monitor.hardStopWatcher).isNull();
+ underTest = newDefaultMonitor(tempDir, true);
+ assertThat(underTest.hardStopWatcher).isNull();
- monitor.watchForHardStop();
+ HttpProcessClient p1 = new HttpProcessClient(tempDir, "p1");
+ underTest.start(singletonList(p1.newCommand()));
- assertThat(monitor.hardStopWatcher).isNotNull();
- assertThat(monitor.hardStopWatcher.isAlive()).isTrue();
+ assertThat(underTest.hardStopWatcher).isNotNull();
+ assertThat(underTest.hardStopWatcher.isAlive()).isTrue();
+
+ p1.kill();
+ underTest.awaitTermination();
+
+ assertThat(underTest.hardStopWatcher.isAlive()).isFalse();
}
private Monitor newDefaultMonitor(File tempDir) throws IOException {
- return new Monitor(tempDir, exit);
+ return newDefaultMonitor(tempDir, false);
+ }
+
+ private Monitor newDefaultMonitor(File tempDir, boolean watchForHardStop) throws IOException {
+ when(fileSystem.getTempDir()).thenReturn(tempDir);
+ return new Monitor(fileSystem, exit, watchForHardStop);
}
/**
@@ -369,7 +395,7 @@ public class MonitorTest {
public void restart() {
try {
HttpRequest httpRequest = HttpRequest.post("http://localhost:" + httpPort + "/" + "restart")
- .readTimeout(5000).connectTimeout(5000);
+ .readTimeout(5000).connectTimeout(5000);
if (!httpRequest.ok() || !"ok".equals(httpRequest.body())) {
throw new IllegalStateException("Wrong response calling restart");
}
diff --git a/sonar-application/src/main/java/org/sonar/application/App.java b/sonar-application/src/main/java/org/sonar/application/App.java
index e09899b1fa4..ca2f98d62b8 100644
--- a/sonar-application/src/main/java/org/sonar/application/App.java
+++ b/sonar-application/src/main/java/org/sonar/application/App.java
@@ -39,8 +39,8 @@ public class App implements Stoppable {
private final Monitor monitor;
- public App(File tempDir) {
- this(Monitor.create(tempDir));
+ public App(AppFileSystem appFileSystem, boolean watchForHardStop) {
+ this(Monitor.create(appFileSystem, watchForHardStop));
}
App(Monitor monitor) {
@@ -48,14 +48,11 @@ public class App implements Stoppable {
}
public void start(Props props) {
- if (props.valueAsBoolean(ProcessProperties.ENABLE_STOP_COMMAND, false)) {
- monitor.watchForHardStop();
- }
monitor.start(createCommands(props));
monitor.awaitTermination();
}
- private List<JavaCommand> createCommands(Props props) {
+ private static List<JavaCommand> createCommands(Props props) {
List<JavaCommand> commands = new ArrayList<>();
File homeDir = props.nonNullValueAsFile(ProcessProperties.PATH_HOME);
JavaCommand elasticsearch = new JavaCommand("search");
@@ -102,11 +99,14 @@ public class App implements Stoppable {
CommandLineParser cli = new CommandLineParser();
Properties rawProperties = cli.parseArguments(args);
Props props = new PropsBuilder(rawProperties, new JdbcSettings()).build();
+ AppFileSystem appFileSystem = new AppFileSystem(props);
+ appFileSystem.verifyProps();
AppLogging logging = new AppLogging();
logging.configure(props);
- File tempDir = props.nonNullValueAsFile(ProcessProperties.PATH_TEMP);
- App app = new App(tempDir);
+ // used by orchestrator
+ boolean watchForHardStop = props.valueAsBoolean(ProcessProperties.ENABLE_STOP_COMMAND, false);
+ App app = new App(appFileSystem, watchForHardStop);
app.start(props);
}
diff --git a/sonar-application/src/main/java/org/sonar/application/AppFileSystem.java b/sonar-application/src/main/java/org/sonar/application/AppFileSystem.java
new file mode 100644
index 00000000000..32f563737b2
--- /dev/null
+++ b/sonar-application/src/main/java/org/sonar/application/AppFileSystem.java
@@ -0,0 +1,120 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2016 SonarSource SA
+ * mailto:contact AT sonarsource DOT com
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 3 of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+package org.sonar.application;
+
+import java.io.File;
+import java.io.IOException;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+import org.sonar.process.Props;
+import org.sonar.process.monitor.FileSystem;
+
+import static org.apache.commons.io.FileUtils.deleteQuietly;
+import static org.apache.commons.io.FileUtils.forceMkdir;
+import static org.sonar.process.ProcessProperties.PATH_DATA;
+import static org.sonar.process.ProcessProperties.PATH_HOME;
+import static org.sonar.process.ProcessProperties.PATH_LOGS;
+import static org.sonar.process.ProcessProperties.PATH_TEMP;
+import static org.sonar.process.ProcessProperties.PATH_WEB;
+
+public class AppFileSystem implements FileSystem {
+ private static final Logger LOG = LoggerFactory.getLogger(AppFileSystem.class);
+
+ private static final String DEFAULT_DATA_DIRECTORY_NAME = "data";
+ private static final String DEFAULT_WEB_DIRECTORY_NAME = "web";
+ private static final String DEFAULT_LOGS_DIRECTORY_NAME = "logs";
+ private static final String DEFAULT_TEMP_DIRECTORY_NAME = "temp";
+
+ private final Props props;
+ private final File homeDir;
+ private boolean initialized = false;
+
+ public AppFileSystem(Props props) {
+ this.props = props;
+ this.homeDir = props.nonNullValueAsFile(PATH_HOME);
+ }
+
+ public void verifyProps() {
+ ensurePropertyIsAbsolutePath(props, PATH_DATA, DEFAULT_DATA_DIRECTORY_NAME);
+ ensurePropertyIsAbsolutePath(props, PATH_WEB, DEFAULT_WEB_DIRECTORY_NAME);
+ ensurePropertyIsAbsolutePath(props, PATH_LOGS, DEFAULT_LOGS_DIRECTORY_NAME);
+ ensurePropertyIsAbsolutePath(props, PATH_TEMP, DEFAULT_TEMP_DIRECTORY_NAME);
+ this.initialized = true;
+ }
+
+ /**
+ * Must be called after {@link #verifyProps()}
+ */
+ @Override
+ public void reset() throws IOException {
+ if (!initialized) {
+ throw new IllegalStateException("method verifyProps must be called first");
+ }
+ ensureDirectoryExists(props, PATH_DATA);
+ ensureDirectoryExists(props, PATH_WEB);
+ ensureDirectoryExists(props, PATH_LOGS);
+ createOrCleanDirectory(props, PATH_TEMP);
+ }
+
+ @Override
+ public File getTempDir() {
+ return props.nonNullValueAsFile(PATH_TEMP);
+ }
+
+ private File ensurePropertyIsAbsolutePath(Props props, String propKey, String defaultRelativePath) {
+ String path = props.value(propKey, defaultRelativePath);
+ File d = new File(path);
+ if (!d.isAbsolute()) {
+ d = new File(homeDir, path);
+ LOG.trace("Overriding property {} from relative path '{}' to absolute path '{}'", path, d.getAbsolutePath());
+ props.set(propKey, d.getAbsolutePath());
+ }
+ return d;
+ }
+
+ private static boolean ensureDirectoryExists(Props props, String propKey) throws IOException {
+ File dir = props.nonNullValueAsFile(propKey);
+ if (dir.exists()) {
+ ensureIsNotAFile(propKey, dir);
+ return false;
+ } else {
+ LOG.trace("forceMkdir {}", dir.getAbsolutePath());
+ forceMkdir(dir);
+ ensureIsNotAFile(propKey, dir);
+ return true;
+ }
+ }
+
+ private static void ensureIsNotAFile(String propKey, File dir) {
+ if (!dir.isDirectory()) {
+ throw new IllegalStateException(String.format("Property '%s' is not valid, not a directory: %s",
+ propKey, dir.getAbsolutePath()));
+ }
+ }
+
+ private static void createOrCleanDirectory(Props props, String propKey) throws IOException {
+ File dir = props.nonNullValueAsFile(propKey);
+ LOG.info("Deleting and/or creating temp directory {}", dir.getAbsolutePath());
+ if (!ensureDirectoryExists(props, propKey)) {
+ deleteQuietly(dir);
+ forceMkdir(dir);
+ }
+ }
+}
diff --git a/sonar-application/src/main/java/org/sonar/application/PropsBuilder.java b/sonar-application/src/main/java/org/sonar/application/PropsBuilder.java
index d915e1609ad..1e2d25778e2 100644
--- a/sonar-application/src/main/java/org/sonar/application/PropsBuilder.java
+++ b/sonar-application/src/main/java/org/sonar/application/PropsBuilder.java
@@ -19,12 +19,6 @@
*/
package org.sonar.application;
-import org.apache.commons.io.FileUtils;
-import org.apache.commons.io.IOUtils;
-import org.sonar.process.ConfigurationUtils;
-import org.sonar.process.ProcessProperties;
-import org.sonar.process.Props;
-
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
@@ -33,6 +27,10 @@ import java.io.Reader;
import java.net.URISyntaxException;
import java.nio.charset.StandardCharsets;
import java.util.Properties;
+import org.apache.commons.io.IOUtils;
+import org.sonar.process.ConfigurationUtils;
+import org.sonar.process.ProcessProperties;
+import org.sonar.process.Props;
class PropsBuilder {
@@ -51,8 +49,7 @@ class PropsBuilder {
}
/**
- * Load optional conf/sonar.properties, interpolates environment variables and
- * initializes file system
+ * Load optional conf/sonar.properties, interpolates environment variables
*/
Props build() throws IOException {
Properties p = loadPropertiesFile(homeDir);
@@ -66,12 +63,6 @@ class PropsBuilder {
Props props = new Props(p);
ProcessProperties.completeDefaults(props);
- // init file system
- initExistingDir(props, ProcessProperties.PATH_DATA, "data");
- initExistingDir(props, ProcessProperties.PATH_WEB, "web");
- initExistingDir(props, ProcessProperties.PATH_LOGS, "logs");
- initTempDir(props);
-
// check JDBC properties and set path to driver
jdbcSettings.checkAndComplete(homeDir, props);
@@ -96,31 +87,4 @@ class PropsBuilder {
}
return p;
}
-
- private void initTempDir(Props props) throws IOException {
- File dir = configureDir(props, ProcessProperties.PATH_TEMP, "temp");
- FileUtils.deleteQuietly(dir);
- FileUtils.forceMkdir(dir);
- }
-
- private void initExistingDir(Props props, String propKey, String defaultRelativePath) throws IOException {
- File dir = configureDir(props, propKey, defaultRelativePath);
- if (!dir.exists()) {
- FileUtils.forceMkdir(dir);
- }
- if (!dir.isDirectory()) {
- throw new IllegalStateException(String.format("Property '%s' is not valid, not a directory: %s",
- propKey, dir.getAbsolutePath()));
- }
- }
-
- private File configureDir(Props props, String propKey, String defaultRelativePath) {
- String path = props.value(propKey, defaultRelativePath);
- File d = new File(path);
- if (!d.isAbsolute()) {
- d = new File(homeDir, path);
- }
- props.set(propKey, d.getAbsolutePath());
- return d;
- }
}
diff --git a/sonar-application/src/test/java/org/sonar/application/AppFileSystemTest.java b/sonar-application/src/test/java/org/sonar/application/AppFileSystemTest.java
new file mode 100644
index 00000000000..baac93fa203
--- /dev/null
+++ b/sonar-application/src/test/java/org/sonar/application/AppFileSystemTest.java
@@ -0,0 +1,202 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2016 SonarSource SA
+ * mailto:contact AT sonarsource DOT com
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 3 of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+package org.sonar.application;
+
+import java.io.File;
+import java.io.IOException;
+import java.util.Properties;
+import org.junit.Before;
+import org.junit.Rule;
+import org.junit.Test;
+import org.junit.rules.ExpectedException;
+import org.junit.rules.TemporaryFolder;
+import org.sonar.process.Props;
+
+import static org.assertj.core.api.Assertions.assertThat;
+
+public class AppFileSystemTest {
+
+ private static final String PROPERTY_SONAR_PATH_WEB = "sonar.path.web";
+ private static final String PROPERTY_SONAR_PATH_DATA = "sonar.path.data";
+ private static final String PROPERTY_SONAR_PATH_LOGS = "sonar.path.logs";
+ private static final String PROPERTY_SONAR_PATH_TEMP = "sonar.path.temp";
+ private static final String NON_DEFAULT_DATA_DIR_NAME = "toto";
+ private static final String NON_DEFAULT_WEB_DIR_NAME = "tutu";
+ private static final String NON_DEFAULT_LOGS_DIR_NAME = "titi";
+ private static final String NON_DEFAULT_TEMP_DIR_NAME = "tatta";
+ private static final String DEFAULT_DATA_DIR_NAME = "data";
+ private static final String DEFAULT_WEB_DIR_NAME = "web";
+ private static final String DEFAULT_LOGS_DIR_NAME = "logs";
+ private static final String DEFAULT_TEMP_DIR_NAME = "temp";
+
+ @Rule
+ public TemporaryFolder temp = new TemporaryFolder();
+ @Rule
+ public ExpectedException expectedException = ExpectedException.none();
+
+ private File homeDir;
+ private Properties properties;
+
+ @Before
+ public void before() throws IOException {
+ homeDir = temp.newFolder();
+
+ properties = new Properties();
+ properties.setProperty("sonar.path.home", homeDir.getAbsolutePath());
+ }
+
+ @Test
+ public void verifyProps_set_dir_path_absolute_based_on_home_dir_and_default_names_when_no_property() {
+ Props props = new Props(properties);
+ AppFileSystem underTest = new AppFileSystem(props);
+
+ underTest.verifyProps();
+
+ assertThat(props.nonNullValue(PROPERTY_SONAR_PATH_DATA)).isEqualTo(new File(homeDir, DEFAULT_DATA_DIR_NAME).getAbsolutePath());
+ assertThat(props.nonNullValue(PROPERTY_SONAR_PATH_WEB)).isEqualTo(new File(homeDir, DEFAULT_WEB_DIR_NAME).getAbsolutePath());
+ assertThat(props.nonNullValue(PROPERTY_SONAR_PATH_LOGS)).isEqualTo(new File(homeDir, DEFAULT_LOGS_DIR_NAME).getAbsolutePath());
+ assertThat(props.nonNullValue(PROPERTY_SONAR_PATH_TEMP)).isEqualTo(new File(homeDir, DEFAULT_TEMP_DIR_NAME).getAbsolutePath());
+ }
+
+ @Test
+ public void verifyProps_can_be_called_multiple_times() {
+ AppFileSystem underTest = new AppFileSystem(new Props(properties));
+
+ underTest.verifyProps();
+ underTest.verifyProps();
+ }
+
+ @Test
+ public void reset_throws_ISE_if_verifyProps_not_called_first() throws Exception {
+ AppFileSystem underTest = new AppFileSystem(new Props(properties));
+
+ expectedException.expect(IllegalStateException.class);
+ expectedException.expectMessage("method verifyProps must be called first");
+
+ underTest.reset();
+ }
+
+ @Test
+ public void verifyProps_makes_dir_path_absolute_based_on_home_dir_when_relative() throws Exception {
+ properties.setProperty(PROPERTY_SONAR_PATH_WEB, NON_DEFAULT_WEB_DIR_NAME);
+ properties.setProperty(PROPERTY_SONAR_PATH_DATA, NON_DEFAULT_DATA_DIR_NAME);
+ properties.setProperty(PROPERTY_SONAR_PATH_LOGS, NON_DEFAULT_LOGS_DIR_NAME);
+ properties.setProperty(PROPERTY_SONAR_PATH_TEMP, NON_DEFAULT_TEMP_DIR_NAME);
+
+ Props props = new Props(properties);
+ AppFileSystem underTest = new AppFileSystem(props);
+
+ underTest.verifyProps();
+
+ assertThat(props.nonNullValue(PROPERTY_SONAR_PATH_DATA)).isEqualTo(new File(homeDir, NON_DEFAULT_DATA_DIR_NAME).getAbsolutePath());
+ assertThat(props.nonNullValue(PROPERTY_SONAR_PATH_WEB)).isEqualTo(new File(homeDir, NON_DEFAULT_WEB_DIR_NAME).getAbsolutePath());
+ assertThat(props.nonNullValue(PROPERTY_SONAR_PATH_LOGS)).isEqualTo(new File(homeDir, NON_DEFAULT_LOGS_DIR_NAME).getAbsolutePath());
+ assertThat(props.nonNullValue(PROPERTY_SONAR_PATH_TEMP)).isEqualTo(new File(homeDir, NON_DEFAULT_TEMP_DIR_NAME).getAbsolutePath());
+ }
+
+ @Test
+ public void reset_creates_dir_all_dirs_if_they_don_t_exist() throws Exception {
+ AppFileSystem underTest = new AppFileSystem(new Props(properties));
+
+ underTest.verifyProps();
+
+ File dataDir = new File(homeDir, DEFAULT_DATA_DIR_NAME);
+ File webDir = new File(homeDir, DEFAULT_WEB_DIR_NAME);
+ File logsDir = new File(homeDir, DEFAULT_LOGS_DIR_NAME);
+ File tempDir = new File(homeDir, DEFAULT_TEMP_DIR_NAME);
+ assertThat(dataDir).doesNotExist();
+ assertThat(webDir).doesNotExist();
+ assertThat(logsDir).doesNotExist();
+ assertThat(tempDir).doesNotExist();
+
+ underTest.reset();
+
+ assertThat(dataDir).exists().isDirectory();
+ assertThat(webDir).exists().isDirectory();
+ assertThat(logsDir).exists().isDirectory();
+ assertThat(tempDir).exists().isDirectory();
+ }
+
+ @Test
+ public void reset_delete_temp_dir_if_already_exists() throws Exception {
+ File tempDir = new File(homeDir, DEFAULT_TEMP_DIR_NAME);
+ assertThat(tempDir.mkdir()).isTrue();
+ File fileInTempDir = new File(tempDir, "someFile.txt");
+ assertThat(fileInTempDir.createNewFile()).isTrue();
+
+ AppFileSystem underTest = new AppFileSystem(new Props(properties));
+ underTest.verifyProps();
+ underTest.reset();
+
+ assertThat(tempDir).exists();
+ assertThat(fileInTempDir).doesNotExist();
+ }
+
+ @Test
+ public void reset_throws_ISE_if_data_dir_is_a_file() throws Exception {
+ resetThrowsISEIfDirIsAFile(PROPERTY_SONAR_PATH_DATA);
+ }
+
+ @Test
+ public void reset_throws_ISE_if_web_dir_is_a_file() throws Exception {
+ resetThrowsISEIfDirIsAFile(PROPERTY_SONAR_PATH_WEB);
+ }
+
+ @Test
+ public void reset_throws_ISE_if_logs_dir_is_a_file() throws Exception {
+ resetThrowsISEIfDirIsAFile(PROPERTY_SONAR_PATH_LOGS);
+ }
+
+ @Test
+ public void reset_throws_ISE_if_temp_dir_is_a_file() throws Exception {
+ resetThrowsISEIfDirIsAFile(PROPERTY_SONAR_PATH_TEMP);
+ }
+
+ private void resetThrowsISEIfDirIsAFile(String property) throws IOException {
+ File file = new File(homeDir, "zoom.store");
+ assertThat(file.createNewFile()).isTrue();
+
+ properties.setProperty(property, "zoom.store");
+
+ AppFileSystem underTest = new AppFileSystem(new Props(properties));
+
+ underTest.verifyProps();
+
+ expectedException.expect(IllegalStateException.class);
+ expectedException.expectMessage("Property '" + property + "' is not valid, not a directory: " + file.getAbsolutePath());
+
+ underTest.reset();
+ }
+
+ // @Test
+// public void fail_if_required_directory_is_a_file() throws Exception {
+// // <home>/data is missing
+// FileUtils.forceMkdir(webDir);
+// FileUtils.forceMkdir(logsDir);
+// try {
+// FileUtils.touch(dataDir);
+// new PropsBuilder(new Properties(), jdbcSettings, homeDir).build();
+// fail();
+// } catch (IllegalStateException e) {
+// assertThat(e.getMessage()).startsWith("Property 'sonar.path.data' is not valid, not a directory: " + dataDir.getAbsolutePath());
+// }
+// }
+
+}
diff --git a/sonar-application/src/test/java/org/sonar/application/PropsBuilderTest.java b/sonar-application/src/test/java/org/sonar/application/PropsBuilderTest.java
index de5069940d2..ad5fb1ddb8a 100644
--- a/sonar-application/src/test/java/org/sonar/application/PropsBuilderTest.java
+++ b/sonar-application/src/test/java/org/sonar/application/PropsBuilderTest.java
@@ -19,6 +19,10 @@
*/
package org.sonar.application;
+import java.io.File;
+import java.io.IOException;
+import java.nio.charset.StandardCharsets;
+import java.util.Properties;
import org.apache.commons.io.FileUtils;
import org.junit.Before;
import org.junit.Rule;
@@ -26,13 +30,7 @@ import org.junit.Test;
import org.junit.rules.TemporaryFolder;
import org.sonar.process.Props;
-import java.io.File;
-import java.io.IOException;
-import java.nio.charset.StandardCharsets;
-import java.util.Properties;
-
import static org.assertj.core.api.Assertions.assertThat;
-import static org.junit.Assert.fail;
import static org.mockito.Mockito.mock;
public class PropsBuilderTest {
@@ -40,39 +38,21 @@ public class PropsBuilderTest {
@Rule
public TemporaryFolder temp = new TemporaryFolder();
- File homeDir;
- File dataDir;
- File webDir;
- File logsDir;
- JdbcSettings jdbcSettings = mock(JdbcSettings.class);
+ private File homeDir;
+ private JdbcSettings jdbcSettings = mock(JdbcSettings.class);
@Before
public void before() throws IOException {
homeDir = temp.newFolder();
- dataDir = new File(homeDir, "data");
- webDir = new File(homeDir, "web");
- logsDir = new File(homeDir, "logs");
}
@Test
public void build_props() throws Exception {
- FileUtils.forceMkdir(dataDir);
- FileUtils.forceMkdir(webDir);
- FileUtils.forceMkdir(logsDir);
Properties rawProperties = new Properties();
rawProperties.setProperty("foo", "bar");
Props props = new PropsBuilder(rawProperties, jdbcSettings, homeDir).build();
- assertThat(props.nonNullValueAsFile("sonar.path.logs")).isEqualTo(logsDir);
- assertThat(props.nonNullValueAsFile("sonar.path.home")).isEqualTo(homeDir);
-
- // create <HOME>/temp
- File tempDir = props.nonNullValueAsFile("sonar.path.temp");
- assertThat(tempDir).isDirectory().exists();
- assertThat(tempDir.getName()).isEqualTo("temp");
- assertThat(tempDir.getParentFile()).isEqualTo(homeDir);
-
assertThat(props.value("foo")).isEqualTo("bar");
assertThat(props.value("unknown")).isNull();
@@ -81,36 +61,8 @@ public class PropsBuilderTest {
}
@Test
- public void create_missing_required_directory() throws Exception {
- // <home>/data is missing
- FileUtils.forceMkdir(webDir);
- FileUtils.forceMkdir(logsDir);
-
- File dataDir = new File(homeDir, "data");
- new PropsBuilder(new Properties(), jdbcSettings, homeDir).build();
- assertThat(dataDir).isDirectory().exists();
- }
-
- @Test
- public void fail_if_required_directory_is_a_file() throws Exception {
- // <home>/data is missing
- FileUtils.forceMkdir(webDir);
- FileUtils.forceMkdir(logsDir);
- try {
- FileUtils.touch(dataDir);
- new PropsBuilder(new Properties(), jdbcSettings, homeDir).build();
- fail();
- } catch (IllegalStateException e) {
- assertThat(e.getMessage()).startsWith("Property 'sonar.path.data' is not valid, not a directory: " + dataDir.getAbsolutePath());
- }
- }
-
- @Test
public void load_properties_file_if_exists() throws Exception {
FileUtils.write(new File(homeDir, "conf/sonar.properties"), "sonar.jdbc.username=angela\nsonar.origin=file");
- FileUtils.forceMkdir(dataDir);
- FileUtils.forceMkdir(webDir);
- FileUtils.forceMkdir(logsDir);
Properties rawProperties = new Properties();
rawProperties.setProperty("sonar.origin", "raw");
@@ -132,10 +84,6 @@ public class PropsBuilderTest {
@Test
public void do_not_load_properties_file_if_not_exists() throws Exception {
- FileUtils.forceMkdir(dataDir);
- FileUtils.forceMkdir(webDir);
- FileUtils.forceMkdir(logsDir);
-
Properties rawProperties = new Properties();
rawProperties.setProperty("sonar.origin", "raw");
Props props = new PropsBuilder(rawProperties, jdbcSettings, homeDir).build();