aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorSimon Brandhof <simon.brandhof@sonarsource.com>2016-03-16 23:01:11 +0100
committerSimon Brandhof <simon.brandhof@sonarsource.com>2016-03-25 09:58:44 +0100
commit6bc55b17ebd87802cb9b55d5b6625e1a935f3909 (patch)
tree0a50457cc294c211ad7a3ee081dabb93e4231286
parent4250bb53f050b5a7f86b211ec2b34f85f7cc3ddc (diff)
downloadsonarqube-6bc55b17ebd87802cb9b55d5b6625e1a935f3909.tar.gz
sonarqube-6bc55b17ebd87802cb9b55d5b6625e1a935f3909.zip
SONAR-7436 Monitoring of CE process in system WS and console
-rw-r--r--it/it-tests/src/test/java/it/serverSystem/ServerSystemTest.java19
-rw-r--r--server/sonar-ce/src/main/java/org/sonar/ce/app/WebServerWatcherImpl.java4
-rw-r--r--server/sonar-ce/src/main/java/org/sonar/ce/container/ComputeEngineContainerImpl.java4
-rw-r--r--server/sonar-ce/src/test/java/org/sonar/ce/app/WebServerWatcherImplTest.java4
-rw-r--r--server/sonar-ce/src/test/java/org/sonar/ce/container/ComputeEngineContainerImplTest.java6
-rw-r--r--server/sonar-process-monitor/src/main/java/org/sonar/process/monitor/JavaCommand.java24
-rw-r--r--server/sonar-process-monitor/src/main/java/org/sonar/process/monitor/JavaProcessLauncher.java40
-rw-r--r--server/sonar-process-monitor/src/test/java/org/sonar/process/monitor/JavaCommandTest.java6
-rw-r--r--server/sonar-process-monitor/src/test/java/org/sonar/process/monitor/JavaProcessLauncherTest.java5
-rw-r--r--server/sonar-process-monitor/src/test/java/org/sonar/process/monitor/MonitorTest.java65
-rw-r--r--server/sonar-process/pom.xml14
-rw-r--r--server/sonar-process/src/main/java/org/sonar/process/AllProcessesCommands.java52
-rw-r--r--server/sonar-process/src/main/java/org/sonar/process/DefaultProcessCommands.java10
-rw-r--r--server/sonar-process/src/main/java/org/sonar/process/ProcessCommands.java6
-rw-r--r--server/sonar-process/src/main/java/org/sonar/process/ProcessEntryPoint.java12
-rw-r--r--server/sonar-process/src/main/java/org/sonar/process/ProcessId.java53
-rw-r--r--server/sonar-process/src/main/java/org/sonar/process/jmx/CeDatabaseMBean.java43
-rw-r--r--server/sonar-process/src/main/java/org/sonar/process/jmx/CeTasksMBean.java (renamed from server/sonar-server/src/main/java/org/sonar/server/computation/monitoring/ComputeEngineQueueMonitorMBean.java)11
-rw-r--r--server/sonar-process/src/main/java/org/sonar/process/jmx/EsSettingsMBean.java38
-rw-r--r--server/sonar-process/src/main/java/org/sonar/process/jmx/Jmx.java86
-rw-r--r--server/sonar-process/src/main/java/org/sonar/process/jmx/JmxConnection.java97
-rw-r--r--server/sonar-process/src/main/java/org/sonar/process/jmx/JmxConnectionFactory.java57
-rw-r--r--server/sonar-process/src/main/java/org/sonar/process/jmx/package-info.java23
-rw-r--r--server/sonar-process/src/test/java/org/sonar/process/AllProcessesCommandsTest.java14
-rw-r--r--server/sonar-process/src/test/java/org/sonar/process/ProcessEntryPointTest.java12
-rw-r--r--server/sonar-process/src/test/java/org/sonar/process/ProcessIdTest.java47
-rw-r--r--server/sonar-process/src/test/java/org/sonar/process/jmx/Fake.java27
-rw-r--r--server/sonar-process/src/test/java/org/sonar/process/jmx/FakeMBean.java24
-rw-r--r--server/sonar-process/src/test/java/org/sonar/process/jmx/JmxConnectionTest.java36
-rw-r--r--server/sonar-process/src/test/java/org/sonar/process/jmx/JmxTest.java87
-rw-r--r--server/sonar-process/src/test/java/org/sonar/process/jmx/test/Fake.java30
-rw-r--r--server/sonar-search/src/main/java/org/sonar/search/EsSettings.java (renamed from server/sonar-search/src/main/java/org/sonar/search/SearchSettings.java)35
-rw-r--r--server/sonar-search/src/main/java/org/sonar/search/SearchServer.java8
-rw-r--r--server/sonar-search/src/test/java/org/sonar/search/EsSettingsTest.java (renamed from server/sonar-search/src/test/java/org/sonar/search/SearchSettingsTest.java)30
-rw-r--r--server/sonar-search/src/test/java/org/sonar/search/SearchServerTest.java7
-rw-r--r--server/sonar-server/src/main/java/org/sonar/ce/CeModule.java4
-rw-r--r--server/sonar-server/src/main/java/org/sonar/ce/monitoring/CEQueueStatus.java52
-rw-r--r--server/sonar-server/src/main/java/org/sonar/ce/monitoring/DummyCEQueueStatusImpl.java93
-rw-r--r--server/sonar-server/src/main/java/org/sonar/ce/queue/CeQueueImpl.java11
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/app/ProcessCommandWrapper.java1
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/app/ProcessCommandWrapperImpl.java24
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/computation/CeModule.java2
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/computation/monitoring/CEQueueStatusImpl.java65
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/computation/monitoring/CeDatabaseMBeanImpl.java97
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/computation/monitoring/CeTasksMBeanImpl.java (renamed from server/sonar-server/src/main/java/org/sonar/server/computation/monitoring/ComputeEngineQueueMonitor.java)39
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/computation/queue/CeQueueInitializer.java10
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/computation/queue/CeQueueModule.java6
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/computation/queue/InternalCeQueueImpl.java12
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/BaseMonitorMBean.java29
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/CeDatabaseMonitor.java58
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/CeStateMonitor.java46
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/CeTasksMonitor.java55
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/DatabaseMonitor.java13
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/EsMonitor.java25
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/EsStateMonitor.java46
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/JmxConnectorProvider.java46
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/JvmPropsMonitor.java (renamed from server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/JvmPropertiesMonitor.java)3
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/platform/platformlevel/PlatformLevel4.java14
-rw-r--r--server/sonar-server/src/test/java/org/sonar/ce/monitoring/CEQueueStatusImplConcurrentTest.java19
-rw-r--r--server/sonar-server/src/test/java/org/sonar/ce/queue/CeQueueImplTest.java41
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/app/ProcessCommandWrapperImplTest.java12
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/computation/monitoring/CEQueueStatusImplTest.java138
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/computation/monitoring/CeTasksMBeanImplTest.java (renamed from server/sonar-server/src/test/java/org/sonar/server/computation/monitoring/ComputeEngineQueueMonitorTest.java)64
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/computation/queue/CeQueueInitializerTest.java56
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/computation/queue/InternalCeQueueImplTest.java32
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/es/EsServerHolder.java7
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/BaseMonitorMBeanTest.java8
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/CeDatabaseMonitorTest.java53
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/CeStateMonitorTest.java53
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/CeTasksMonitorTest.java53
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/DatabaseMonitorTest.java5
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/EsMonitorTest.java39
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/EsStateMonitorTest.java52
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/JmxConnectionFactoryProviderTest.java56
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/JvmPropsMonitorTest.java (renamed from server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/JvmPropertiesMonitorTest.java)13
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/SonarQubeMonitorTest.java5
-rw-r--r--server/sonar-web/src/main/js/apps/system/main.js4
-rw-r--r--sonar-application/src/main/java/org/sonar/application/App.java16
-rw-r--r--sonar-application/src/test/java/org/sonar/application/AppTest.java7
79 files changed, 1757 insertions, 763 deletions
diff --git a/it/it-tests/src/test/java/it/serverSystem/ServerSystemTest.java b/it/it-tests/src/test/java/it/serverSystem/ServerSystemTest.java
index 64c72d3faec..51192aac4bc 100644
--- a/it/it-tests/src/test/java/it/serverSystem/ServerSystemTest.java
+++ b/it/it-tests/src/test/java/it/serverSystem/ServerSystemTest.java
@@ -40,12 +40,15 @@ import org.junit.Test;
import org.junit.experimental.categories.Category;
import org.sonar.wsclient.services.Server;
import org.sonar.wsclient.services.ServerQuery;
+import org.sonarqube.ws.client.GetRequest;
+import org.sonarqube.ws.client.WsResponse;
import util.ItUtils;
import util.QaOnly;
import util.selenium.SeleneseTest;
import static org.assertj.core.api.Assertions.assertThat;
import static org.junit.Assert.fail;
+import static util.ItUtils.newAdminWsClient;
@Category(QaOnly.class)
public class ServerSystemTest {
@@ -90,6 +93,22 @@ public class ServerSystemTest {
}
/**
+ * SONAR-7436
+ */
+ @Test
+ public void monitor_compute_engine_and_elasticsearch_processes() throws Exception {
+ WsResponse response = newAdminWsClient(orchestrator).wsConnector().call(
+ new GetRequest("api/system/info"));
+ assertThat(response.code()).isEqualTo(200);
+
+ assertThat(response.content()).containsSequence("\"Compute Engine Database\":", "\"Pool Active Connections\"");
+ assertThat(response.content()).containsSequence("\"Compute Engine State\":", "\"Heap Used\"");
+ assertThat(response.content()).containsSequence("\"Compute Engine Tasks\":", "\"Pending\"", "\"In Progress\"");
+
+ assertThat(response.content()).containsSequence("\"Elasticsearch\":", "\"State\":\"GREEN\"");
+ }
+
+ /**
* See http://jira.codehaus.org/browse/SONAR-2727
*/
@Test
diff --git a/server/sonar-ce/src/main/java/org/sonar/ce/app/WebServerWatcherImpl.java b/server/sonar-ce/src/main/java/org/sonar/ce/app/WebServerWatcherImpl.java
index 4ebc8f80030..73bde24aa0e 100644
--- a/server/sonar-ce/src/main/java/org/sonar/ce/app/WebServerWatcherImpl.java
+++ b/server/sonar-ce/src/main/java/org/sonar/ce/app/WebServerWatcherImpl.java
@@ -23,10 +23,10 @@ import java.io.File;
import org.sonar.api.utils.log.Logger;
import org.sonar.api.utils.log.Loggers;
import org.sonar.process.DefaultProcessCommands;
+import org.sonar.process.ProcessId;
public class WebServerWatcherImpl implements WebServerWatcher {
private static final Logger LOG = Loggers.get(WebServerWatcherImpl.class);
- private static final int WEB_SERVER_PROCESS_NUMBER = 2;
private static final int POLL_DELAY = 200;
// accounting only every 5 log calls so that only one every second (because delay is 200ms) is taken into account
private static final int CALL_RATIO = 5;
@@ -39,7 +39,7 @@ public class WebServerWatcherImpl implements WebServerWatcher {
@Override
public boolean waitForOperational() {
- try (DefaultProcessCommands processCommands = DefaultProcessCommands.secondary(sharedDir, WEB_SERVER_PROCESS_NUMBER)) {
+ try (DefaultProcessCommands processCommands = DefaultProcessCommands.secondary(sharedDir, ProcessId.WEB_SERVER.getIpcIndex())) {
if (processCommands.isOperational()) {
return true;
}
diff --git a/server/sonar-ce/src/main/java/org/sonar/ce/container/ComputeEngineContainerImpl.java b/server/sonar-ce/src/main/java/org/sonar/ce/container/ComputeEngineContainerImpl.java
index 42527bf4e8b..be0bda6e858 100644
--- a/server/sonar-ce/src/main/java/org/sonar/ce/container/ComputeEngineContainerImpl.java
+++ b/server/sonar-ce/src/main/java/org/sonar/ce/container/ComputeEngineContainerImpl.java
@@ -108,6 +108,7 @@ import org.sonar.server.platform.ServerImpl;
import org.sonar.server.platform.ServerLifecycleNotifier;
import org.sonar.server.platform.ServerLogging;
import org.sonar.server.platform.TempFolderProvider;
+import org.sonar.server.platform.monitoring.JmxConnectorProvider;
import org.sonar.server.plugins.InstalledPluginReferentialFactory;
import org.sonar.server.plugins.ServerExtensionInstaller;
import org.sonar.server.properties.ProjectSettingsFactory;
@@ -138,6 +139,7 @@ public class ComputeEngineContainerImpl implements ComputeEngineContainer {
private static final Object[] LEVEL_1_COMPONENTS = new Object[] {
ComputeEngineSettings.class,
new SonarQubeVersionProvider(),
+ new JmxConnectorProvider(),
ServerImpl.class,
UuidFactoryImpl.INSTANCE,
// no EmbeddedDatabaseFactory.class, creating H2 DB if responsibility of WebServer
@@ -521,7 +523,7 @@ public class ComputeEngineContainerImpl implements ComputeEngineContainer {
// SonarQubeMonitor.class, no Monitor in CE, responsibility of Web Server
// EsMonitor.class, no Monitor in CE, responsibility of Web Server
// PluginsMonitor.class, no Monitor in CE, responsibility of Web Server
- // JvmPropertiesMonitor.class, no Monitor in CE, responsibility of Web Server
+ // JvmPropsMonitor.class, no Monitor in CE, responsibility of Web Server
// DatabaseMonitor.class, no Monitor in CE, responsibility of Web Server
// MigrateDbAction.class, no Web Service in CE
// LogsAction.class, no Web Service in CE
diff --git a/server/sonar-ce/src/test/java/org/sonar/ce/app/WebServerWatcherImplTest.java b/server/sonar-ce/src/test/java/org/sonar/ce/app/WebServerWatcherImplTest.java
index 32007dc60f1..d4a9ae285a4 100644
--- a/server/sonar-ce/src/test/java/org/sonar/ce/app/WebServerWatcherImplTest.java
+++ b/server/sonar-ce/src/test/java/org/sonar/ce/app/WebServerWatcherImplTest.java
@@ -30,12 +30,12 @@ import org.junit.rules.Timeout;
import org.sonar.api.utils.log.LogTester;
import org.sonar.api.utils.log.LoggerLevel;
import org.sonar.process.DefaultProcessCommands;
+import org.sonar.process.ProcessId;
import static java.util.concurrent.TimeUnit.MILLISECONDS;
import static org.assertj.core.api.Assertions.assertThat;
public class WebServerWatcherImplTest {
- private static final int WEB_SERVER_PROCESS_NUMBER = 2;
@Rule
public Timeout timeout = Timeout.seconds(1);
@@ -102,7 +102,7 @@ public class WebServerWatcherImplTest {
}
private void setWebServerOperational() {
- try (DefaultProcessCommands processCommands = DefaultProcessCommands.secondary(sharedDir, WEB_SERVER_PROCESS_NUMBER)) {
+ try (DefaultProcessCommands processCommands = DefaultProcessCommands.secondary(sharedDir, ProcessId.WEB_SERVER.getIpcIndex())) {
processCommands.setOperational();
}
}
diff --git a/server/sonar-ce/src/test/java/org/sonar/ce/container/ComputeEngineContainerImplTest.java b/server/sonar-ce/src/test/java/org/sonar/ce/container/ComputeEngineContainerImplTest.java
index b56bd13daf1..291093f2b45 100644
--- a/server/sonar-ce/src/test/java/org/sonar/ce/container/ComputeEngineContainerImplTest.java
+++ b/server/sonar-ce/src/test/java/org/sonar/ce/container/ComputeEngineContainerImplTest.java
@@ -34,6 +34,7 @@ import org.sonar.db.DbTester;
import org.sonar.process.Props;
import static org.assertj.core.api.Assertions.assertThat;
+import static org.sonar.process.ProcessEntryPoint.PROPERTY_SHARED_PATH;
import static org.sonar.process.ProcessProperties.PATH_DATA;
import static org.sonar.process.ProcessProperties.PATH_HOME;
import static org.sonar.process.ProcessProperties.PATH_TEMP;
@@ -65,6 +66,7 @@ public class ComputeEngineContainerImplTest {
properties.setProperty(PATH_HOME, homeDir.getAbsolutePath());
properties.setProperty(PATH_DATA, dataDir.getAbsolutePath());
properties.setProperty(PATH_TEMP, tmpDir.getAbsolutePath());
+ properties.setProperty(PROPERTY_SHARED_PATH, tmpDir.getAbsolutePath());
String url = ((BasicDataSource) dbTester.database().getDataSource()).getUrl();
properties.setProperty(DatabaseProperties.PROP_URL, url);
properties.setProperty(DatabaseProperties.PROP_USER, "sonar");
@@ -77,7 +79,7 @@ public class ComputeEngineContainerImplTest {
assertThat(picoContainer.getComponentAdapters())
.hasSize(
CONTAINER_ITSELF
- + 77 // level 4
+ + 78 // level 4
+ 5 // content of CeModule
+ 7 // content of CeQueueModule
+ 4 // content of ReportProcessingModule
@@ -93,7 +95,7 @@ public class ComputeEngineContainerImplTest {
);
assertThat(picoContainer.getParent().getParent().getParent().getComponentAdapters()).hasSize(
COMPONENTS_IN_LEVEL_1_AT_CONSTRUCTION
- + 22 // level 1
+ + 23 // level 1
+ 47 // content of DaoModule
+ 1 // content of EsSearchModule
+ 56 // content of CorePropertyDefinitions
diff --git a/server/sonar-process-monitor/src/main/java/org/sonar/process/monitor/JavaCommand.java b/server/sonar-process-monitor/src/main/java/org/sonar/process/monitor/JavaCommand.java
index 6a4f77607d9..98caa87d620 100644
--- a/server/sonar-process-monitor/src/main/java/org/sonar/process/monitor/JavaCommand.java
+++ b/server/sonar-process-monitor/src/main/java/org/sonar/process/monitor/JavaCommand.java
@@ -19,10 +19,6 @@
*/
package org.sonar.process.monitor;
-import org.apache.commons.lang.StringUtils;
-
-import javax.annotation.Nullable;
-
import java.io.File;
import java.util.ArrayList;
import java.util.HashMap;
@@ -30,11 +26,14 @@ import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Properties;
+import javax.annotation.Nullable;
+import org.apache.commons.lang.StringUtils;
+import org.sonar.process.ProcessId;
public class JavaCommand {
// unique key among the group of commands to launch
- private final String key;
+ private final ProcessId id;
private File workDir;
@@ -52,19 +51,12 @@ public class JavaCommand {
private final Map<String, String> envVariables = new HashMap<>(System.getenv());
- private final int processIndex;
-
- public JavaCommand(String key, int processIndex) {
- this.key = key;
- this.processIndex = processIndex;
- }
-
- public String getKey() {
- return key;
+ public JavaCommand(ProcessId id) {
+ this.id = id;
}
- public int getProcessIndex() {
- return processIndex;
+ public ProcessId getProcessId() {
+ return id;
}
public File getWorkDir() {
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 7a5869abb38..c586d7c9df0 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
@@ -33,18 +33,24 @@ import org.sonar.process.ProcessCommands;
import org.sonar.process.ProcessEntryPoint;
import org.sonar.process.ProcessUtils;
-public class JavaProcessLauncher {
+import static org.sonar.process.ProcessEntryPoint.PROPERTY_PROCESS_INDEX;
+import static org.sonar.process.ProcessEntryPoint.PROPERTY_PROCESS_KEY;
+import static org.sonar.process.ProcessEntryPoint.PROPERTY_SHARED_PATH;
+import static org.sonar.process.ProcessEntryPoint.PROPERTY_TERMINATION_TIMEOUT;
+
+class JavaProcessLauncher implements AutoCloseable {
private final Timeouts timeouts;
private final File tempDir;
private final AllProcessesCommands allProcessesCommands;
- public JavaProcessLauncher(Timeouts timeouts, File tempDir) {
+ JavaProcessLauncher(Timeouts timeouts, File tempDir) {
this.timeouts = timeouts;
this.tempDir = tempDir;
this.allProcessesCommands = new AllProcessesCommands(tempDir);
}
+ @Override
public void close() {
allProcessesCommands.close();
}
@@ -52,21 +58,21 @@ public class JavaProcessLauncher {
ProcessRef launch(JavaCommand command) {
Process process = null;
try {
- ProcessCommands commands = allProcessesCommands.createAfterClean(command.getProcessIndex());
+ ProcessCommands commands = allProcessesCommands.createAfterClean(command.getProcessId().getIpcIndex());
ProcessBuilder processBuilder = create(command);
LoggerFactory.getLogger(getClass()).info("Launch process[{}]: {}",
- command.getKey(), StringUtils.join(processBuilder.command(), " "));
+ command.getProcessId().getKey(), StringUtils.join(processBuilder.command(), " "));
process = processBuilder.start();
- StreamGobbler inputGobbler = new StreamGobbler(process.getInputStream(), command.getKey());
+ StreamGobbler inputGobbler = new StreamGobbler(process.getInputStream(), command.getProcessId().getKey());
inputGobbler.start();
- return new ProcessRef(command.getKey(), commands, process, inputGobbler);
+ return new ProcessRef(command.getProcessId().getKey(), commands, process, inputGobbler);
} catch (Exception e) {
// just in case
ProcessUtils.sendKillSignal(process);
- throw new IllegalStateException("Fail to launch " + command.getKey(), e);
+ throw new IllegalStateException("Fail to launch [" + command.getProcessId().getKey() + "]", e);
}
}
@@ -76,6 +82,7 @@ public class JavaProcessLauncher {
commands.addAll(javaCommand.getJavaOptions());
// TODO warning - does it work if temp dir contains a whitespace ?
commands.add(String.format("-Djava.io.tmpdir=%s", tempDir.getAbsolutePath()));
+ commands.add(getJmxAgentCommand());
commands.addAll(buildClasspath(javaCommand));
commands.add(javaCommand.getClassName());
commands.add(buildPropertiesFile(javaCommand).getAbsolutePath());
@@ -88,6 +95,15 @@ public class JavaProcessLauncher {
return processBuilder;
}
+ /**
+ * JVM option to enable the agent that allows inter-process communication through JMX without
+ * opening new ports. The agent is available in JRE of OpenJDK/OracleJDK only.
+ * @see ProcessEntryPoint
+ */
+ private static String getJmxAgentCommand() {
+ return "-javaagent:" + System.getProperty("java.home") + File.separator + "lib" + File.separator + "management-agent.jar";
+ }
+
private String buildJavaPath() {
String separator = System.getProperty("file.separator");
return new File(new File(System.getProperty("java.home")),
@@ -104,12 +120,12 @@ public class JavaProcessLauncher {
propertiesFile = File.createTempFile("sq-process", "properties");
Properties props = new Properties();
props.putAll(javaCommand.getArguments());
- props.setProperty(ProcessEntryPoint.PROPERTY_PROCESS_KEY, javaCommand.getKey());
- props.setProperty(ProcessEntryPoint.PROPERTY_PROCESS_INDEX, Integer.toString(javaCommand.getProcessIndex()));
- props.setProperty(ProcessEntryPoint.PROPERTY_TERMINATION_TIMEOUT, String.valueOf(timeouts.getTerminationTimeout()));
- props.setProperty(ProcessEntryPoint.PROPERTY_SHARED_PATH, tempDir.getAbsolutePath());
+ props.setProperty(PROPERTY_PROCESS_KEY, javaCommand.getProcessId().getKey());
+ props.setProperty(PROPERTY_PROCESS_INDEX, Integer.toString(javaCommand.getProcessId().getIpcIndex()));
+ props.setProperty(PROPERTY_TERMINATION_TIMEOUT, String.valueOf(timeouts.getTerminationTimeout()));
+ props.setProperty(PROPERTY_SHARED_PATH, tempDir.getAbsolutePath());
OutputStream out = new FileOutputStream(propertiesFile);
- props.store(out, String.format("Temporary properties file for command [%s]", javaCommand.getKey()));
+ props.store(out, String.format("Temporary properties file for command [%s]", javaCommand.getProcessId().getKey()));
out.close();
return propertiesFile;
} catch (Exception e) {
diff --git a/server/sonar-process-monitor/src/test/java/org/sonar/process/monitor/JavaCommandTest.java b/server/sonar-process-monitor/src/test/java/org/sonar/process/monitor/JavaCommandTest.java
index e53c564cdac..c0ddbce5709 100644
--- a/server/sonar-process-monitor/src/test/java/org/sonar/process/monitor/JavaCommandTest.java
+++ b/server/sonar-process-monitor/src/test/java/org/sonar/process/monitor/JavaCommandTest.java
@@ -25,6 +25,7 @@ import org.junit.rules.TemporaryFolder;
import java.io.File;
import java.util.Properties;
+import org.sonar.process.ProcessId;
import static org.assertj.core.api.Assertions.assertThat;
@@ -32,11 +33,10 @@ public class JavaCommandTest {
@Rule
public TemporaryFolder temp = new TemporaryFolder();
- private int processIndex = 0;
@Test
public void test_parameters() throws Exception {
- JavaCommand command = new JavaCommand("es", processIndex++);
+ JavaCommand command = new JavaCommand(ProcessId.ELASTICSEARCH);
command.setArgument("first_arg", "val1");
Properties args = new Properties();
@@ -64,7 +64,7 @@ public class JavaCommandTest {
@Test
public void add_java_options() {
- JavaCommand command = new JavaCommand("foo", processIndex++);
+ JavaCommand command = new JavaCommand(ProcessId.ELASTICSEARCH);
assertThat(command.getJavaOptions()).isEmpty();
command.addJavaOptions("");
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 499b22a04bd..df9ae973d17 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,6 +23,7 @@ import java.io.File;
import org.junit.Rule;
import org.junit.Test;
import org.junit.rules.TemporaryFolder;
+import org.sonar.process.ProcessId;
import static org.assertj.core.api.Assertions.assertThat;
import static org.junit.Assert.fail;
@@ -35,7 +36,7 @@ public class JavaProcessLauncherTest {
@Test
public void fail_to_launch() throws Exception {
File tempDir = temp.newFolder();
- JavaCommand command = new JavaCommand("test", 0);
+ JavaCommand command = new JavaCommand(ProcessId.ELASTICSEARCH);
JavaProcessLauncher launcher = new JavaProcessLauncher(new Timeouts(), tempDir);
try {
// command is not correct (missing options), java.lang.ProcessBuilder#start()
@@ -43,7 +44,7 @@ public class JavaProcessLauncherTest {
launcher.launch(command);
fail();
} catch (IllegalStateException e) {
- assertThat(e).hasMessage("Fail to launch test");
+ assertThat(e).hasMessage("Fail to launch [es]");
}
}
}
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 f278e4978bb..475fe2c0867 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
@@ -42,6 +42,7 @@ import org.junit.rules.TestRule;
import org.junit.rules.Timeout;
import org.sonar.process.Lifecycle.State;
import org.sonar.process.NetworkUtils;
+import org.sonar.process.ProcessId;
import org.sonar.process.SystemExit;
import static java.util.Collections.singletonList;
@@ -147,7 +148,7 @@ public class MonitorTest {
@Test
public void start_then_stop_gracefully() throws Exception {
underTest = newDefaultMonitor(tempDir);
- HttpProcessClient client = new HttpProcessClient(tempDir, "test");
+ HttpProcessClient client = new HttpProcessClient(tempDir, ProcessId.ELASTICSEARCH);
// blocks until started
underTest.start(singletonList(client.newCommand()));
@@ -157,7 +158,7 @@ public class MonitorTest {
// blocks until stopped
underTest.stop();
assertThat(client)
- .isNotReady()
+ .isNotUp()
.wasGracefullyTerminated();
assertThat(underTest.getState()).isEqualTo(State.STOPPED);
verify(fileSystem).reset();
@@ -166,8 +167,8 @@ public class MonitorTest {
@Test
public void start_then_stop_sequence_of_commands() throws Exception {
underTest = newDefaultMonitor(tempDir);
- HttpProcessClient p1 = new HttpProcessClient(tempDir, "p1");
- HttpProcessClient p2 = new HttpProcessClient(tempDir, "p2");
+ HttpProcessClient p1 = new HttpProcessClient(tempDir, ProcessId.ELASTICSEARCH);
+ HttpProcessClient p2 = new HttpProcessClient(tempDir, ProcessId.WEB_SERVER);
underTest.start(Arrays.asList(p1.newCommand(), p2.newCommand()));
// start p2 when p1 is fully started (ready)
@@ -181,10 +182,10 @@ public class MonitorTest {
// stop in inverse order
assertThat(p1)
- .isNotReady()
+ .isNotUp()
.wasGracefullyTerminated();
assertThat(p2)
- .isNotReady()
+ .isNotUp()
.wasGracefullyTerminatedBefore(p1);
verify(fileSystem).reset();
}
@@ -192,8 +193,8 @@ public class MonitorTest {
@Test
public void stop_all_processes_if_monitor_shutdowns() throws Exception {
underTest = newDefaultMonitor(tempDir);
- HttpProcessClient p1 = new HttpProcessClient(tempDir, "p1");
- HttpProcessClient p2 = new HttpProcessClient(tempDir, "p2");
+ HttpProcessClient p1 = new HttpProcessClient(tempDir, ProcessId.ELASTICSEARCH);
+ HttpProcessClient p2 = new HttpProcessClient(tempDir, ProcessId.WEB_SERVER);
underTest.start(Arrays.asList(p1.newCommand(), p2.newCommand()));
assertThat(p1).isUp();
assertThat(p2).isUp();
@@ -211,8 +212,8 @@ public class MonitorTest {
@Test
public void restart_all_processes_if_one_asks_for_restart() throws Exception {
underTest = newDefaultMonitor(tempDir);
- HttpProcessClient p1 = new HttpProcessClient(tempDir, "p1");
- HttpProcessClient p2 = new HttpProcessClient(tempDir, "p2");
+ HttpProcessClient p1 = new HttpProcessClient(tempDir, ProcessId.ELASTICSEARCH);
+ HttpProcessClient p2 = new HttpProcessClient(tempDir, ProcessId.WEB_SERVER);
underTest.start(Arrays.asList(p1.newCommand(), p2.newCommand()));
assertThat(p1).isUp();
@@ -244,8 +245,8 @@ public class MonitorTest {
@Test
public void stop_all_processes_if_one_shutdowns() throws Exception {
underTest = newDefaultMonitor(tempDir);
- HttpProcessClient p1 = new HttpProcessClient(tempDir, "p1");
- HttpProcessClient p2 = new HttpProcessClient(tempDir, "p2");
+ HttpProcessClient p1 = new HttpProcessClient(tempDir, ProcessId.ELASTICSEARCH);
+ HttpProcessClient p2 = new HttpProcessClient(tempDir, ProcessId.WEB_SERVER);
underTest.start(Arrays.asList(p1.newCommand(), p2.newCommand()));
assertThat(p1.isUp()).isTrue();
assertThat(p2.isUp()).isTrue();
@@ -255,10 +256,10 @@ public class MonitorTest {
underTest.awaitTermination();
assertThat(p1)
- .isNotReady()
+ .isNotUp()
.wasNotGracefullyTerminated();
assertThat(p2)
- .isNotReady()
+ .isNotUp()
.wasGracefullyTerminated();
verify(fileSystem).reset();
@@ -267,8 +268,8 @@ public class MonitorTest {
@Test
public void stop_all_processes_if_one_fails_to_start() throws Exception {
underTest = newDefaultMonitor(tempDir);
- HttpProcessClient p1 = new HttpProcessClient(tempDir, "p1");
- HttpProcessClient p2 = new HttpProcessClient(tempDir, "p2", -1);
+ HttpProcessClient p1 = new HttpProcessClient(tempDir, ProcessId.ELASTICSEARCH);
+ HttpProcessClient p2 = new HttpProcessClient(tempDir, ProcessId.WEB_SERVER, -1);
try {
underTest.start(Arrays.asList(p1.newCommand(), p2.newCommand()));
fail();
@@ -286,7 +287,7 @@ public class MonitorTest {
@Test
public void fail_to_start_if_bad_class_name() throws Exception {
underTest = newDefaultMonitor(tempDir);
- JavaCommand command = new JavaCommand("test", processIndex++)
+ JavaCommand command = new JavaCommand(ProcessId.ELASTICSEARCH)
.addClasspath(testJar.getAbsolutePath())
.setClassName("org.sonar.process.test.Unknown");
@@ -304,7 +305,7 @@ public class MonitorTest {
underTest = newDefaultMonitor(tempDir, true);
assertThat(underTest.hardStopWatcher).isNull();
- HttpProcessClient p1 = new HttpProcessClient(tempDir, "p1");
+ HttpProcessClient p1 = new HttpProcessClient(tempDir, ProcessId.COMPUTE_ENGINE);
underTest.start(singletonList(p1.newCommand()));
assertThat(underTest.hardStopWatcher).isNotNull();
@@ -330,24 +331,24 @@ public class MonitorTest {
*/
private class HttpProcessClient {
private final int httpPort;
- private final String commandKey;
+ private final ProcessId processId;
private final File tempDir;
- private HttpProcessClient(File tempDir, String commandKey) throws IOException {
- this(tempDir, commandKey, NetworkUtils.freePort());
+ private HttpProcessClient(File tempDir, ProcessId processId) throws IOException {
+ this(tempDir, processId, NetworkUtils.freePort());
}
/**
* Use httpPort=-1 to make server fail to start
*/
- private HttpProcessClient(File tempDir, String commandKey, int httpPort) throws IOException {
+ private HttpProcessClient(File tempDir, ProcessId processId, int httpPort) throws IOException {
this.tempDir = tempDir;
- this.commandKey = commandKey;
+ this.processId = processId;
this.httpPort = httpPort;
}
JavaCommand newCommand() {
- return new JavaCommand(commandKey, processIndex++)
+ return new JavaCommand(processId)
.addClasspath(testJar.getAbsolutePath())
.setClassName("org.sonar.process.test.HttpProcess")
.setArgument("httpPort", String.valueOf(httpPort));
@@ -490,7 +491,7 @@ public class MonitorTest {
isNotNull();
if (!actual.wasGracefullyTerminated()) {
- failWithMessage("HttpClient %s should have been gracefully terminated", actual.commandKey);
+ failWithMessage("HttpClient %s should have been gracefully terminated", actual.processId.getKey());
}
return this;
@@ -500,7 +501,7 @@ public class MonitorTest {
isNotNull();
if (actual.wasGracefullyTerminated()) {
- failWithMessage("HttpClient %s should not have been gracefully terminated", actual.commandKey);
+ failWithMessage("HttpClient %s should not have been gracefully terminated", actual.processId.getKey());
}
return this;
@@ -530,17 +531,17 @@ public class MonitorTest {
// check condition
if (!actual.isUp()) {
- failWithMessage("HttpClient %s should be up", actual.commandKey);
+ failWithMessage("HttpClient %s should be up", actual.processId.getKey());
}
return this;
}
- public HttpProcessClientAssert isNotReady() {
+ public HttpProcessClientAssert isNotUp() {
isNotNull();
if (actual.isUp()) {
- failWithMessage("HttpClient %s should not be ready", actual.commandKey);
+ failWithMessage("HttpClient %s should not be up", actual.processId.getKey());
}
return this;
@@ -551,7 +552,7 @@ public class MonitorTest {
// check condition
if (!actual.wasReady()) {
- failWithMessage("HttpClient %s should been ready at least once", actual.commandKey);
+ failWithMessage("HttpClient %s should been ready at least once", actual.processId.getKey());
}
return this;
@@ -562,7 +563,7 @@ public class MonitorTest {
// check condition
if (actual.wasReady()) {
- failWithMessage("HttpClient %s should never been ready", actual.commandKey);
+ failWithMessage("HttpClient %s should never been ready", actual.processId.getKey());
}
return this;
@@ -570,7 +571,7 @@ public class MonitorTest {
}
private JavaCommand newStandardProcessCommand() throws IOException {
- return new JavaCommand("standard", processIndex++)
+ return new JavaCommand(ProcessId.ELASTICSEARCH)
.addClasspath(testJar.getAbsolutePath())
.setClassName("org.sonar.process.test.StandardProcess");
}
diff --git a/server/sonar-process/pom.xml b/server/sonar-process/pom.xml
index dad1c57192b..df90101c3c0 100644
--- a/server/sonar-process/pom.xml
+++ b/server/sonar-process/pom.xml
@@ -93,7 +93,19 @@
<skipTests>${skipServerTests}</skipTests>
</configuration>
</plugin>
-
</plugins>
+ <pluginManagement>
+ <plugins>
+ <plugin>
+ <groupId>org.codehaus.mojo</groupId>
+ <artifactId>animal-sniffer-maven-plugin</artifactId>
+ <configuration>
+ <ignores>
+ <ignore>sun.misc.VMSupport</ignore>
+ </ignores>
+ </configuration>
+ </plugin>
+ </plugins>
+ </pluginManagement>
</build>
</project>
diff --git a/server/sonar-process/src/main/java/org/sonar/process/AllProcessesCommands.java b/server/sonar-process/src/main/java/org/sonar/process/AllProcessesCommands.java
index a3b0b8364c9..9652aca6795 100644
--- a/server/sonar-process/src/main/java/org/sonar/process/AllProcessesCommands.java
+++ b/server/sonar-process/src/main/java/org/sonar/process/AllProcessesCommands.java
@@ -24,8 +24,11 @@ import java.io.IOException;
import java.io.RandomAccessFile;
import java.nio.MappedByteBuffer;
import java.nio.channels.FileChannel;
+import java.nio.charset.StandardCharsets;
import org.apache.commons.io.IOUtils;
+import static java.lang.String.format;
+import static org.apache.commons.lang.StringUtils.rightPad;
import static org.sonar.process.ProcessCommands.MAX_PROCESSES;
/**
@@ -60,14 +63,17 @@ import static org.sonar.process.ProcessCommands.MAX_PROCESSES;
* </ul>
* </p>
*/
-public class AllProcessesCommands {
+public class AllProcessesCommands implements AutoCloseable {
private static final int UP_BYTE_OFFSET = 0;
private static final int STOP_BYTE_OFFSET = 1;
private static final int RESTART_BYTE_OFFSET = 2;
private static final int OPERATIONAL_BYTE_OFFSET = 3;
private static final int PING_BYTE_OFFSET = 4;
+ private static final int JMX_URL_BYTE_OFFSET = PING_BYTE_OFFSET + 8;
- private static final int BYTE_LENGTH_FOR_ONE_PROCESS = 1 + 1 + 1 + 1 + 8;
+ private static final int JMX_URL_SIZE_IN_BYTES = 500;
+
+ private static final int BYTE_LENGTH_FOR_ONE_PROCESS = 1 + 1 + 1 + 1 + 8 + JMX_URL_SIZE_IN_BYTES;
// With this shared memory we can handle up to MAX_PROCESSES processes
private static final int MAX_SHARED_MEMORY = BYTE_LENGTH_FOR_ONE_PROCESS * MAX_PROCESSES;
@@ -142,6 +148,19 @@ public class AllProcessesCommands {
return readLong(processNumber, PING_BYTE_OFFSET);
}
+ String getJmxUrl(int processNumber) {
+ byte[] urlBytes = readBytes(processNumber, JMX_URL_BYTE_OFFSET, JMX_URL_SIZE_IN_BYTES);
+ return new String(urlBytes, StandardCharsets.US_ASCII).trim();
+ }
+
+ void setJmxUrl(int processNumber, String jmxUrl) {
+ byte[] urlBytes = rightPad(jmxUrl, JMX_URL_SIZE_IN_BYTES).getBytes(StandardCharsets.US_ASCII);
+ if (urlBytes.length > JMX_URL_SIZE_IN_BYTES) {
+ throw new IllegalArgumentException(format("JMX URL is too long. Max is %d bytes. Got: %s", JMX_URL_SIZE_IN_BYTES, jmxUrl));
+ }
+ writeBytes(processNumber, JMX_URL_BYTE_OFFSET, urlBytes);
+ }
+
/**
* To be executed by monitor process to ask for child process termination
*/
@@ -165,13 +184,14 @@ public class AllProcessesCommands {
writeByte(processNumber, RESTART_BYTE_OFFSET, EMPTY);
}
+ @Override
public void close() {
IOUtils.closeQuietly(sharedMemory);
}
public void checkProcessNumber(int processNumber) {
if (processNumber < 0 || processNumber >= MAX_PROCESSES) {
- throw new IllegalArgumentException(String.format("Process number %s is not valid", processNumber));
+ throw new IllegalArgumentException(format("Process number %s is not valid", processNumber));
}
}
@@ -185,10 +205,26 @@ public class AllProcessesCommands {
mappedByteBuffer.put(offset(processNumber) + offset, value);
}
+ private void writeBytes(int processNumber, int offset, byte[] value) {
+ int bufferOffset = offset(processNumber) + offset;
+ for (int i = 0; i < value.length; i++) {
+ mappedByteBuffer.put(bufferOffset + i, value[i]);
+ }
+ }
+
private byte readByte(int processNumber, int offset) {
return mappedByteBuffer.get(offset(processNumber) + offset);
}
+ private byte[] readBytes(int processNumber, int offset, int length) {
+ int bufferOffset = offset(processNumber) + offset;
+ byte[] bytes = new byte[length];
+ for (int i = 0; i < length; i++) {
+ bytes[i] = mappedByteBuffer.get(bufferOffset + i);
+ }
+ return bytes;
+ }
+
private void writeLong(int processNumber, int offset, long value) {
mappedByteBuffer.putLong(offset(processNumber) + offset, value);
}
@@ -241,6 +277,16 @@ public class AllProcessesCommands {
}
@Override
+ public void setJmxUrl(String s) {
+ AllProcessesCommands.this.setJmxUrl(processNumber, s);
+ }
+
+ @Override
+ public String getJmxUrl() {
+ return AllProcessesCommands.this.getJmxUrl(processNumber);
+ }
+
+ @Override
public void askForStop() {
AllProcessesCommands.this.askForStop(processNumber);
}
diff --git a/server/sonar-process/src/main/java/org/sonar/process/DefaultProcessCommands.java b/server/sonar-process/src/main/java/org/sonar/process/DefaultProcessCommands.java
index 04a608fef06..d8c9975a4ba 100644
--- a/server/sonar-process/src/main/java/org/sonar/process/DefaultProcessCommands.java
+++ b/server/sonar-process/src/main/java/org/sonar/process/DefaultProcessCommands.java
@@ -83,6 +83,16 @@ public class DefaultProcessCommands implements ProcessCommands {
}
@Override
+ public void setJmxUrl(String s) {
+ delegate.setJmxUrl(s);
+ }
+
+ @Override
+ public String getJmxUrl() {
+ return delegate.getJmxUrl();
+ }
+
+ @Override
public void askForStop() {
delegate.askForStop();
}
diff --git a/server/sonar-process/src/main/java/org/sonar/process/ProcessCommands.java b/server/sonar-process/src/main/java/org/sonar/process/ProcessCommands.java
index 482029b81d8..01ece9a7060 100644
--- a/server/sonar-process/src/main/java/org/sonar/process/ProcessCommands.java
+++ b/server/sonar-process/src/main/java/org/sonar/process/ProcessCommands.java
@@ -33,7 +33,7 @@ import java.io.File;
*/
public interface ProcessCommands extends AutoCloseable {
- int MAX_PROCESSES = 50;
+ int MAX_PROCESSES = 5;
boolean isUp();
@@ -55,6 +55,10 @@ public interface ProcessCommands extends AutoCloseable {
long getLastPing();
+ void setJmxUrl(String s);
+
+ String getJmxUrl();
+
/**
* To be executed by monitor process to ask for child process termination
*/
diff --git a/server/sonar-process/src/main/java/org/sonar/process/ProcessEntryPoint.java b/server/sonar-process/src/main/java/org/sonar/process/ProcessEntryPoint.java
index 10930daf679..98190b78e2c 100644
--- a/server/sonar-process/src/main/java/org/sonar/process/ProcessEntryPoint.java
+++ b/server/sonar-process/src/main/java/org/sonar/process/ProcessEntryPoint.java
@@ -21,6 +21,7 @@ package org.sonar.process;
import java.io.File;
import org.slf4j.LoggerFactory;
+import sun.misc.VMSupport;
public class ProcessEntryPoint implements Stoppable {
@@ -104,6 +105,8 @@ public class ProcessEntryPoint implements Stoppable {
Thread.sleep(20L);
}
+ commands.setJmxUrl(guessJmxUrl());
+
// notify monitor that process is ready
commands.setUp();
@@ -118,6 +121,15 @@ public class ProcessEntryPoint implements Stoppable {
}
}
+ private static String guessJmxUrl() {
+ // this property is set by the agent management-agent.jar enabled by org.sonar.process.monitor.JavaProcessLauncher.
+ String jmxUrl = VMSupport.getAgentProperties().getProperty("com.sun.management.jmxremote.localConnectorAddress");
+ if (jmxUrl == null) {
+ throw new IllegalStateException("Fail to load the JMX URL of JVM " + System.getProperty("java.vm.name"));
+ }
+ return jmxUrl;
+ }
+
boolean isStarted() {
return lifecycle.getState() == Lifecycle.State.STARTED;
}
diff --git a/server/sonar-process/src/main/java/org/sonar/process/ProcessId.java b/server/sonar-process/src/main/java/org/sonar/process/ProcessId.java
new file mode 100644
index 00000000000..9189ada6cd2
--- /dev/null
+++ b/server/sonar-process/src/main/java/org/sonar/process/ProcessId.java
@@ -0,0 +1,53 @@
+/*
+ * 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;
+
+public enum ProcessId {
+
+ APP("app", 0), ELASTICSEARCH("es", 1), WEB_SERVER("web", 2), COMPUTE_ENGINE("ce", 3);
+
+ private final String key;
+ private final int ipcIndex;
+
+ ProcessId(String key, int ipcIndex) {
+ this.key = key;
+ this.ipcIndex = ipcIndex;
+ }
+
+ public String getKey() {
+ return key;
+ }
+
+ /**
+ * Index used for inter-process communication
+ */
+ public int getIpcIndex() {
+ return ipcIndex;
+ }
+
+ @Override
+ public String toString() {
+ StringBuilder sb = new StringBuilder("[");
+ sb.append("key='").append(key).append('\'');
+ sb.append(", ipcIndex=").append(ipcIndex);
+ sb.append(']');
+ return sb.toString();
+ }
+}
diff --git a/server/sonar-process/src/main/java/org/sonar/process/jmx/CeDatabaseMBean.java b/server/sonar-process/src/main/java/org/sonar/process/jmx/CeDatabaseMBean.java
new file mode 100644
index 00000000000..354ce0db734
--- /dev/null
+++ b/server/sonar-process/src/main/java/org/sonar/process/jmx/CeDatabaseMBean.java
@@ -0,0 +1,43 @@
+/*
+ * 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.jmx;
+
+public interface CeDatabaseMBean {
+
+ String OBJECT_NAME = "SonarQube:name=ComputeEngineDatabase";
+
+ int getPoolActiveConnections();
+
+ int getPoolMaxActiveConnections();
+
+ int getPoolIdleConnections();
+
+ int getPoolMaxIdleConnections();
+
+ int getPoolMinIdleConnections();
+
+ int getPoolInitialSize();
+
+ long getPoolMaxWaitMillis();
+
+ boolean getPoolRemoveAbandoned();
+
+ int getPoolRemoveAbandonedTimeoutSeconds();
+}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/computation/monitoring/ComputeEngineQueueMonitorMBean.java b/server/sonar-process/src/main/java/org/sonar/process/jmx/CeTasksMBean.java
index f1d694ded59..968dabc52b2 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/computation/monitoring/ComputeEngineQueueMonitorMBean.java
+++ b/server/sonar-process/src/main/java/org/sonar/process/jmx/CeTasksMBean.java
@@ -17,14 +17,11 @@
* 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.server.computation.monitoring;
+package org.sonar.process.jmx;
-public interface ComputeEngineQueueMonitorMBean {
+public interface CeTasksMBean {
- /**
- * Count of received batch reports since instance startup
- */
- long getReceivedCount();
+ String OBJECT_NAME = "SonarQube:name=ComputeEngineTasks";
/**
* Count of batch reports waiting for processing since startup, including reports received before instance startup.
@@ -47,7 +44,7 @@ public interface ComputeEngineQueueMonitorMBean {
long getSuccessCount();
/**
- * Time spent processing reports since startup.
+ * Time spent processing reports since startup, in milliseconds.
*/
long getProcessingTime();
diff --git a/server/sonar-process/src/main/java/org/sonar/process/jmx/EsSettingsMBean.java b/server/sonar-process/src/main/java/org/sonar/process/jmx/EsSettingsMBean.java
new file mode 100644
index 00000000000..197dc35051c
--- /dev/null
+++ b/server/sonar-process/src/main/java/org/sonar/process/jmx/EsSettingsMBean.java
@@ -0,0 +1,38 @@
+/*
+ * 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.jmx;
+
+/**
+ * MBean registered in the Elasticsearch process
+ */
+public interface EsSettingsMBean {
+
+ String OBJECT_NAME = "SonarQube:name=ElasticsearchSettings";
+
+ /**
+ * @return the enabled HTTP port, -1 if disabled
+ */
+ int getHttpPort();
+
+ String getClusterName();
+
+ String getNodeName();
+
+}
diff --git a/server/sonar-process/src/main/java/org/sonar/process/jmx/Jmx.java b/server/sonar-process/src/main/java/org/sonar/process/jmx/Jmx.java
new file mode 100644
index 00000000000..62375a32e83
--- /dev/null
+++ b/server/sonar-process/src/main/java/org/sonar/process/jmx/Jmx.java
@@ -0,0 +1,86 @@
+/*
+ * 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.jmx;
+
+import java.lang.management.ManagementFactory;
+import javax.management.InstanceAlreadyExistsException;
+import javax.management.MBeanRegistrationException;
+import javax.management.MalformedObjectNameException;
+import javax.management.NotCompliantMBeanException;
+import javax.management.ObjectName;
+import javax.management.StandardMBean;
+import org.slf4j.LoggerFactory;
+
+/**
+ * JMX utilities to register MBeans to JMX server
+ */
+public class Jmx {
+
+ private Jmx() {
+ // only statics
+ }
+
+ /**
+ * Register a MBean to JMX server
+ */
+ public static void register(String name, Object instance) {
+ try {
+ Class mbeanInterface = guessMBeanInterface(instance);
+ ManagementFactory.getPlatformMBeanServer().registerMBean(new StandardMBean(instance, mbeanInterface), new ObjectName(name));
+
+ } catch (MalformedObjectNameException | NotCompliantMBeanException | InstanceAlreadyExistsException | MBeanRegistrationException e) {
+ throw new IllegalStateException("Can not register MBean [" + name + "]", e);
+ }
+ }
+
+ /**
+ * MBeans have multiple conventions, including:
+ * 1. name of interface is suffixed by "MBean"
+ * 2. name of implementation is the name of the interface without "MBean"
+ * 3. implementation and interface must be in the same package
+ * To avoid the last convention, we wrap the mbean within a StandardMBean. That
+ * requires to find the related interface.
+ */
+ private static Class guessMBeanInterface(Object instance) {
+ Class mbeanInterface = null;
+ Class<?>[] interfaces = instance.getClass().getInterfaces();
+ for (Class<?> anInterface : interfaces) {
+ if (anInterface.getName().endsWith("MBean")) {
+ mbeanInterface = anInterface;
+ break;
+ }
+ }
+ if (mbeanInterface == null) {
+ throw new IllegalArgumentException("Can not find the MBean interface of class " + instance.getClass().getName());
+ }
+ return mbeanInterface;
+ }
+
+ /**
+ * Unregister a MBean from JMX server. Errors are ignored and logged as warnings.
+ */
+ public static void unregister(String name) {
+ try {
+ ManagementFactory.getPlatformMBeanServer().unregisterMBean(new ObjectName(name));
+ } catch (Exception e) {
+ LoggerFactory.getLogger(Jmx.class).warn("Can not unregister MBean [" + name + "]", e);
+ }
+ }
+}
diff --git a/server/sonar-process/src/main/java/org/sonar/process/jmx/JmxConnection.java b/server/sonar-process/src/main/java/org/sonar/process/jmx/JmxConnection.java
new file mode 100644
index 00000000000..378c135af03
--- /dev/null
+++ b/server/sonar-process/src/main/java/org/sonar/process/jmx/JmxConnection.java
@@ -0,0 +1,97 @@
+/*
+ * 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.jmx;
+
+import java.io.IOException;
+import java.lang.management.ManagementFactory;
+import java.lang.management.MemoryMXBean;
+import java.lang.management.MemoryUsage;
+import java.lang.management.ThreadMXBean;
+import java.util.SortedMap;
+import java.util.TreeMap;
+import javax.annotation.CheckForNull;
+import javax.management.JMX;
+import javax.management.MBeanServerConnection;
+import javax.management.ObjectName;
+import javax.management.remote.JMXConnector;
+
+public class JmxConnection implements AutoCloseable {
+
+ private static final long MEGABYTE = 1024L * 1024L;
+ private final JMXConnector jmxConnector;
+
+ JmxConnection(JMXConnector jmxConnector) {
+ this.jmxConnector = jmxConnector;
+ }
+
+ /**
+ * Get a MBean from a remote JMX server.
+ * @throws IllegalStateException if a valid
+ * connection to remote server cannot be created, for instance because the connection to has
+ * not yet been established (with {@link JMXConnector#connect()}), or it has been closed/broken.
+ */
+ public <M> M getMBean(String mBeanName, Class<M> mBeanInterfaceClass) {
+ try {
+ MBeanServerConnection connection = jmxConnector.getMBeanServerConnection();
+ if (mBeanName.startsWith("java.lang")) {
+ return ManagementFactory.newPlatformMXBeanProxy(connection, mBeanName, mBeanInterfaceClass);
+ }
+ return JMX.newMBeanProxy(connection, new ObjectName(mBeanName), mBeanInterfaceClass);
+ } catch (Exception e) {
+ throw new IllegalStateException("Fail to connect to MBean " + mBeanName, e);
+ }
+ }
+
+ public SortedMap<String, Object> getSystemState() {
+ SortedMap<String, Object> props = new TreeMap<>();
+ MemoryMXBean memory = getMBean(ManagementFactory.MEMORY_MXBEAN_NAME, MemoryMXBean.class);
+ MemoryUsage heap = memory.getHeapMemoryUsage();
+ props.put("Heap Committed (MB)", toMegaBytes(heap.getCommitted()));
+ props.put("Heap Init (MB)", toMegaBytes(heap.getInit()));
+ props.put("Heap Max (MB)", toMegaBytes(heap.getMax()));
+ props.put("Heap Used (MB)", toMegaBytes(heap.getUsed()));
+ MemoryUsage nonHeap = memory.getNonHeapMemoryUsage();
+ props.put("Non Heap Committed (MB)", toMegaBytes(nonHeap.getCommitted()));
+ props.put("Non Heap Init (MB)", toMegaBytes(nonHeap.getInit()));
+ props.put("Non Heap Max (MB)", toMegaBytes(nonHeap.getMax()));
+ props.put("Non Heap Used (MB)", toMegaBytes(nonHeap.getUsed()));
+ ThreadMXBean thread = getMBean(ManagementFactory.THREAD_MXBEAN_NAME, ThreadMXBean.class);
+ props.put("Thread Count", thread.getThreadCount());
+ return props;
+ }
+
+ // visible for testing
+ @CheckForNull
+ static Long toMegaBytes(long bytes) {
+ if (bytes < 0L) {
+ return null;
+ }
+ return bytes / MEGABYTE;
+ }
+
+ @Override
+ public void close() {
+ try {
+ jmxConnector.close();
+ } catch (IOException e) {
+ throw new IllegalStateException("Can not close JMX connector", e);
+ }
+ }
+}
diff --git a/server/sonar-process/src/main/java/org/sonar/process/jmx/JmxConnectionFactory.java b/server/sonar-process/src/main/java/org/sonar/process/jmx/JmxConnectionFactory.java
new file mode 100644
index 00000000000..ca906b9540d
--- /dev/null
+++ b/server/sonar-process/src/main/java/org/sonar/process/jmx/JmxConnectionFactory.java
@@ -0,0 +1,57 @@
+/*
+ * 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.jmx;
+
+import java.io.File;
+import javax.annotation.concurrent.Immutable;
+import javax.management.remote.JMXConnector;
+import javax.management.remote.JMXConnectorFactory;
+import javax.management.remote.JMXServiceURL;
+import org.sonar.process.DefaultProcessCommands;
+import org.sonar.process.ProcessEntryPoint;
+import org.sonar.process.ProcessId;
+import org.sonar.process.Props;
+
+/**
+ * Connects to JMX of other JVM processes
+ */
+@Immutable
+public class JmxConnectionFactory {
+ private final File ipcSharedDir;
+
+ public JmxConnectionFactory(File ipcSharedDir) {
+ this.ipcSharedDir = ipcSharedDir;
+ }
+
+ public JmxConnectionFactory(Props props) {
+ this.ipcSharedDir = props.nonNullValueAsFile(ProcessEntryPoint.PROPERTY_SHARED_PATH);
+ }
+
+ public JmxConnection create(ProcessId processId) {
+ try (DefaultProcessCommands commands = DefaultProcessCommands.secondary(ipcSharedDir, processId.getIpcIndex())) {
+ String url = commands.getJmxUrl();
+ JMXConnector jmxConnector = JMXConnectorFactory.newJMXConnector(new JMXServiceURL(url), null);
+ jmxConnector.connect();
+ return new JmxConnection(jmxConnector);
+ } catch (Exception e) {
+ throw new IllegalStateException("Can not connect to process " + processId, e);
+ }
+ }
+}
diff --git a/server/sonar-process/src/main/java/org/sonar/process/jmx/package-info.java b/server/sonar-process/src/main/java/org/sonar/process/jmx/package-info.java
new file mode 100644
index 00000000000..bb4fb97207f
--- /dev/null
+++ b/server/sonar-process/src/main/java/org/sonar/process/jmx/package-info.java
@@ -0,0 +1,23 @@
+/*
+ * 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.
+ */
+@ParametersAreNonnullByDefault
+package org.sonar.process.jmx;
+
+import javax.annotation.ParametersAreNonnullByDefault;
diff --git a/server/sonar-process/src/test/java/org/sonar/process/AllProcessesCommandsTest.java b/server/sonar-process/src/test/java/org/sonar/process/AllProcessesCommandsTest.java
index bda42cf07e7..e834fce905f 100644
--- a/server/sonar-process/src/test/java/org/sonar/process/AllProcessesCommandsTest.java
+++ b/server/sonar-process/src/test/java/org/sonar/process/AllProcessesCommandsTest.java
@@ -97,6 +97,20 @@ public class AllProcessesCommandsTest {
}
@Test
+ public void write_and_read_jmx_url() throws IOException {
+ AllProcessesCommands commands = new AllProcessesCommands(temp.newFolder());
+
+ int offset = 12;
+ for (int i = 0; i < 500; i++) {
+ assertThat(readByte(commands, offset + i)).isEqualTo(EMPTY);
+ }
+
+ commands.setJmxUrl(PROCESS_NUMBER, "jmx:foo");
+ assertThat(readByte(commands, offset)).isNotEqualTo(EMPTY);
+ assertThat(commands.getJmxUrl(PROCESS_NUMBER)).isEqualTo("jmx:foo");
+ }
+
+ @Test
public void ask_for_stop() throws Exception {
AllProcessesCommands commands = new AllProcessesCommands(temp.newFolder());
int offset = 1;
diff --git a/server/sonar-process/src/test/java/org/sonar/process/ProcessEntryPointTest.java b/server/sonar-process/src/test/java/org/sonar/process/ProcessEntryPointTest.java
index 6acf205b52e..f6e2bfd916f 100644
--- a/server/sonar-process/src/test/java/org/sonar/process/ProcessEntryPointTest.java
+++ b/server/sonar-process/src/test/java/org/sonar/process/ProcessEntryPointTest.java
@@ -54,6 +54,8 @@ public class ProcessEntryPointTest {
@Rule
public TemporaryFolder temp = new TemporaryFolder();
+ ProcessCommands commands = mock(ProcessCommands.class);
+
@Test
public void load_properties_from_file() throws Exception {
File propsFile = temp.newFile();
@@ -67,7 +69,7 @@ public class ProcessEntryPointTest {
@Test
public void test_initial_state() throws Exception {
Props props = createProps();
- ProcessEntryPoint entryPoint = new ProcessEntryPoint(props, exit, mock(ProcessCommands.class));
+ ProcessEntryPoint entryPoint = new ProcessEntryPoint(props, exit, commands);
assertThat(entryPoint.getProps()).isSameAs(props);
assertThat(entryPoint.isStarted()).isFalse();
@@ -77,7 +79,7 @@ public class ProcessEntryPointTest {
@Test
public void fail_to_launch_multiple_times() throws IOException {
Props props = createProps();
- ProcessEntryPoint entryPoint = new ProcessEntryPoint(props, exit, mock(ProcessCommands.class));
+ ProcessEntryPoint entryPoint = new ProcessEntryPoint(props, exit, commands);
entryPoint.launch(new NoopProcess());
try {
@@ -91,7 +93,7 @@ public class ProcessEntryPointTest {
@Test
public void launch_then_request_graceful_stop() throws Exception {
Props props = createProps();
- final ProcessEntryPoint entryPoint = new ProcessEntryPoint(props, exit, mock(ProcessCommands.class));
+ final ProcessEntryPoint entryPoint = new ProcessEntryPoint(props, exit, commands);
final StandardProcess process = new StandardProcess();
Thread runner = new Thread() {
@@ -117,7 +119,7 @@ public class ProcessEntryPointTest {
@Test
public void terminate_if_unexpected_shutdown() throws Exception {
Props props = createProps();
- final ProcessEntryPoint entryPoint = new ProcessEntryPoint(props, exit, mock(ProcessCommands.class));
+ final ProcessEntryPoint entryPoint = new ProcessEntryPoint(props, exit, commands);
final StandardProcess process = new StandardProcess();
Thread runner = new Thread() {
@@ -147,7 +149,7 @@ public class ProcessEntryPointTest {
@Test
public void terminate_if_startup_error() throws IOException {
Props props = createProps();
- final ProcessEntryPoint entryPoint = new ProcessEntryPoint(props, exit, mock(ProcessCommands.class));
+ final ProcessEntryPoint entryPoint = new ProcessEntryPoint(props, exit, commands);
final Monitored process = new StartupErrorProcess();
entryPoint.launch(process);
diff --git a/server/sonar-process/src/test/java/org/sonar/process/ProcessIdTest.java b/server/sonar-process/src/test/java/org/sonar/process/ProcessIdTest.java
new file mode 100644
index 00000000000..4eebf2cd124
--- /dev/null
+++ b/server/sonar-process/src/test/java/org/sonar/process/ProcessIdTest.java
@@ -0,0 +1,47 @@
+/*
+ * 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;
+
+import java.util.HashSet;
+import java.util.Set;
+import org.junit.Test;
+
+import static org.assertj.core.api.Assertions.assertThat;
+
+public class ProcessIdTest {
+
+ @Test
+ public void test_constants() {
+ assertThat(ProcessId.COMPUTE_ENGINE.getKey()).isEqualTo("ce");
+ assertThat(ProcessId.COMPUTE_ENGINE.getIpcIndex()).isEqualTo(3);
+ }
+
+ @Test
+ public void all_values_are_unique() {
+ Set<Integer> ipcIndices = new HashSet<>();
+ Set<String> keys = new HashSet<>();
+ for (ProcessId processId : ProcessId.values()) {
+ ipcIndices.add(processId.getIpcIndex());
+ keys.add(processId.getKey());
+ }
+ assertThat(ipcIndices).hasSize(ProcessId.values().length);
+ assertThat(keys).hasSize(ProcessId.values().length);
+ }
+}
diff --git a/server/sonar-process/src/test/java/org/sonar/process/jmx/Fake.java b/server/sonar-process/src/test/java/org/sonar/process/jmx/Fake.java
new file mode 100644
index 00000000000..e7bf9912c78
--- /dev/null
+++ b/server/sonar-process/src/test/java/org/sonar/process/jmx/Fake.java
@@ -0,0 +1,27 @@
+/*
+ * 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.jmx;
+
+public class Fake implements FakeMBean {
+ @Override
+ public void foo() {
+
+ }
+}
diff --git a/server/sonar-process/src/test/java/org/sonar/process/jmx/FakeMBean.java b/server/sonar-process/src/test/java/org/sonar/process/jmx/FakeMBean.java
new file mode 100644
index 00000000000..b5cc660511c
--- /dev/null
+++ b/server/sonar-process/src/test/java/org/sonar/process/jmx/FakeMBean.java
@@ -0,0 +1,24 @@
+/*
+ * 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.jmx;
+
+public interface FakeMBean {
+ void foo();
+}
diff --git a/server/sonar-process/src/test/java/org/sonar/process/jmx/JmxConnectionTest.java b/server/sonar-process/src/test/java/org/sonar/process/jmx/JmxConnectionTest.java
new file mode 100644
index 00000000000..c8aa351524d
--- /dev/null
+++ b/server/sonar-process/src/test/java/org/sonar/process/jmx/JmxConnectionTest.java
@@ -0,0 +1,36 @@
+/*
+ * 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.jmx;
+
+import org.junit.Test;
+
+import static org.assertj.core.api.Assertions.assertThat;
+
+public class JmxConnectionTest {
+
+ @Test
+ public void toMegaBytes() {
+ assertThat(JmxConnection.toMegaBytes(-1)).isNull();
+ assertThat(JmxConnection.toMegaBytes(0L)).isEqualTo(0L);
+ assertThat(JmxConnection.toMegaBytes(500L)).isEqualTo(0L);
+ assertThat(JmxConnection.toMegaBytes(500_000L)).isEqualTo(0L);
+ assertThat(JmxConnection.toMegaBytes(500_000_000L)).isEqualTo(476L);
+ }
+}
diff --git a/server/sonar-process/src/test/java/org/sonar/process/jmx/JmxTest.java b/server/sonar-process/src/test/java/org/sonar/process/jmx/JmxTest.java
new file mode 100644
index 00000000000..9b241accac3
--- /dev/null
+++ b/server/sonar-process/src/test/java/org/sonar/process/jmx/JmxTest.java
@@ -0,0 +1,87 @@
+/*
+ * 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.jmx;
+
+import java.lang.management.ManagementFactory;
+import javax.annotation.CheckForNull;
+import javax.management.InstanceNotFoundException;
+import javax.management.ObjectInstance;
+import javax.management.ObjectName;
+import org.junit.Rule;
+import org.junit.Test;
+import org.junit.rules.ExpectedException;
+
+import static org.assertj.core.api.Assertions.assertThat;
+
+public class JmxTest {
+
+ static final String FAKE_NAME = "SonarQube:name=Fake";
+
+ @Rule
+ public ExpectedException expectedException = ExpectedException.none();
+
+ FakeMBean mbean = new Fake();
+
+ @Test
+ public void register_and_unregister() throws Exception {
+ assertThat(lookupMBean()).isNull();
+
+ Jmx.register(FAKE_NAME, mbean);
+ assertThat(lookupMBean()).isNotNull();
+
+ Jmx.unregister(FAKE_NAME);
+ assertThat(lookupMBean()).isNull();
+ }
+
+ @Test
+ public void do_not_fail_when_unregistering_a_non_registered_bean() throws Exception {
+ Jmx.unregister(FAKE_NAME);
+ assertThat(lookupMBean()).isNull();
+ }
+
+ @Test
+ public void register_fails_if_mbean_interface_can_not_be_found() {
+ expectedException.expect(IllegalArgumentException.class);
+ expectedException.expectMessage("Can not find the MBean interface of class java.lang.String");
+
+ Jmx.register(FAKE_NAME, "not a mbean");
+ }
+
+ @Test
+ public void support_implementation_in_different_package_than_interface() throws Exception {
+ assertThat(lookupMBean()).isNull();
+
+ Jmx.register(FAKE_NAME, new org.sonar.process.jmx.test.Fake());
+ assertThat(lookupMBean()).isNotNull();
+
+ Jmx.unregister(FAKE_NAME);
+ assertThat(lookupMBean()).isNull();
+ }
+
+ @CheckForNull
+ private ObjectInstance lookupMBean() throws Exception {
+ try {
+ return ManagementFactory.getPlatformMBeanServer().getObjectInstance(new ObjectName(FAKE_NAME));
+ } catch (InstanceNotFoundException e) {
+ return null;
+ }
+ }
+
+}
diff --git a/server/sonar-process/src/test/java/org/sonar/process/jmx/test/Fake.java b/server/sonar-process/src/test/java/org/sonar/process/jmx/test/Fake.java
new file mode 100644
index 00000000000..d5e15ab0ca9
--- /dev/null
+++ b/server/sonar-process/src/test/java/org/sonar/process/jmx/test/Fake.java
@@ -0,0 +1,30 @@
+/*
+ * 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.jmx.test;
+
+import org.sonar.process.jmx.FakeMBean;
+
+// implementation is in a different package than interface
+public class Fake implements FakeMBean {
+ @Override
+ public void foo() {
+
+ }
+}
diff --git a/server/sonar-search/src/main/java/org/sonar/search/SearchSettings.java b/server/sonar-search/src/main/java/org/sonar/search/EsSettings.java
index db634f5c587..2f0986e1e45 100644
--- a/server/sonar-search/src/main/java/org/sonar/search/SearchSettings.java
+++ b/server/sonar-search/src/main/java/org/sonar/search/EsSettings.java
@@ -33,17 +33,18 @@ import org.slf4j.LoggerFactory;
import org.sonar.process.MessageException;
import org.sonar.process.ProcessProperties;
import org.sonar.process.Props;
+import org.sonar.process.jmx.EsSettingsMBean;
-class SearchSettings {
+public class EsSettings implements EsSettingsMBean {
- private static final Logger LOGGER = LoggerFactory.getLogger(SearchSettings.class);
+ private static final Logger LOGGER = LoggerFactory.getLogger(EsSettings.class);
public static final String PROP_MARVEL_HOSTS = "sonar.search.marvelHosts";
private final Props props;
private final Set<String> masterHosts = new LinkedHashSet<>();
- SearchSettings(Props props) {
+ EsSettings(Props props) {
this.props = props;
masterHosts.addAll(Arrays.asList(StringUtils.split(props.value(ProcessProperties.CLUSTER_MASTER_HOST, ""), ",")));
}
@@ -56,6 +57,21 @@ class SearchSettings {
return props.valueAsBoolean(ProcessProperties.CLUSTER_MASTER, false);
}
+ @Override
+ public int getHttpPort() {
+ return props.valueAsInt(ProcessProperties.SEARCH_HTTP_PORT, -1);
+ }
+
+ @Override
+ public String getClusterName() {
+ return props.value(ProcessProperties.CLUSTER_NAME);
+ }
+
+ @Override
+ public String getNodeName() {
+ return props.value(ProcessProperties.CLUSTER_NODE_NAME);
+ }
+
Settings build() {
ImmutableSettings.Builder builder = ImmutableSettings.settingsBuilder();
configureFileSystem(builder);
@@ -115,8 +131,8 @@ class SearchSettings {
// Elasticsearch sets the default value of TCP reuse address to true only on non-MSWindows machines, but why ?
builder.put("network.tcp.reuse_address", true);
- Integer httpPort = props.valueAsInt(ProcessProperties.SEARCH_HTTP_PORT);
- if (httpPort == null || httpPort < 0) {
+ int httpPort = getHttpPort();
+ if (httpPort < 0) {
// standard configuration
builder.put("http.enabled", false);
} else {
@@ -130,7 +146,7 @@ class SearchSettings {
}
}
- private void configureIndexDefaults(ImmutableSettings.Builder builder) {
+ private static void configureIndexDefaults(ImmutableSettings.Builder builder) {
builder
.put("index.number_of_shards", "1")
.put("index.refresh_interval", "30s")
@@ -156,10 +172,11 @@ class SearchSettings {
}
}
builder.put(IndexMetaData.SETTING_NUMBER_OF_REPLICAS, replicationFactor);
- builder.put("cluster.name", props.value(ProcessProperties.CLUSTER_NAME));
+ builder.put("cluster.name", getClusterName());
builder.put("cluster.routing.allocation.awareness.attributes", "rack_id");
- builder.put("node.rack_id", props.value(ProcessProperties.CLUSTER_NODE_NAME, "unknown"));
- builder.put("node.name", props.value(ProcessProperties.CLUSTER_NODE_NAME));
+ String nodeName = getNodeName();
+ builder.put("node.rack_id", nodeName);
+ builder.put("node.name", nodeName);
}
private void configureMarvel(ImmutableSettings.Builder builder) {
diff --git a/server/sonar-search/src/main/java/org/sonar/search/SearchServer.java b/server/sonar-search/src/main/java/org/sonar/search/SearchServer.java
index 0a7939abedf..31541ad259f 100644
--- a/server/sonar-search/src/main/java/org/sonar/search/SearchServer.java
+++ b/server/sonar-search/src/main/java/org/sonar/search/SearchServer.java
@@ -26,14 +26,16 @@ import org.sonar.process.MinimumViableSystem;
import org.sonar.process.Monitored;
import org.sonar.process.ProcessEntryPoint;
import org.sonar.process.Props;
+import org.sonar.process.jmx.EsSettingsMBean;
+import org.sonar.process.jmx.Jmx;
public class SearchServer implements Monitored {
- private final SearchSettings settings;
+ private final EsSettings settings;
private InternalNode node;
public SearchServer(Props props) {
- this.settings = new SearchSettings(props);
+ this.settings = new EsSettings(props);
new MinimumViableSystem()
.checkJavaVersion()
.checkWritableTempDir();
@@ -41,6 +43,7 @@ public class SearchServer implements Monitored {
@Override
public void start() {
+ Jmx.register(EsSettingsMBean.OBJECT_NAME, settings);
node = new InternalNode(settings.build(), false);
node.start();
}
@@ -70,6 +73,7 @@ public class SearchServer implements Monitored {
if (node != null && !node.isClosed()) {
node.close();
}
+ Jmx.unregister(EsSettingsMBean.OBJECT_NAME);
}
public static void main(String... args) {
diff --git a/server/sonar-search/src/test/java/org/sonar/search/SearchSettingsTest.java b/server/sonar-search/src/test/java/org/sonar/search/EsSettingsTest.java
index 607eab9d92e..f8f12f25272 100644
--- a/server/sonar-search/src/test/java/org/sonar/search/SearchSettingsTest.java
+++ b/server/sonar-search/src/test/java/org/sonar/search/EsSettingsTest.java
@@ -34,7 +34,7 @@ import java.util.Properties;
import static org.assertj.core.api.Assertions.assertThat;
import static org.junit.Assert.fail;
-public class SearchSettingsTest {
+public class EsSettingsTest {
@Rule
public TemporaryFolder temp = new TemporaryFolder();
@@ -49,10 +49,10 @@ public class SearchSettingsTest {
props.set(ProcessProperties.CLUSTER_NAME, "tests");
props.set(ProcessProperties.CLUSTER_NODE_NAME, "test");
- SearchSettings searchSettings = new SearchSettings(props);
- assertThat(searchSettings.inCluster()).isFalse();
+ EsSettings esSettings = new EsSettings(props);
+ assertThat(esSettings.inCluster()).isFalse();
- Settings generated = searchSettings.build();
+ Settings generated = esSettings.build();
assertThat(generated.get("transport.tcp.port")).isEqualTo("1234");
assertThat(generated.get("transport.host")).isEqualTo("127.0.0.1");
assertThat(generated.get("cluster.name")).isEqualTo("tests");
@@ -74,10 +74,10 @@ public class SearchSettingsTest {
public void test_default_hosts() throws Exception {
Props props = minProps();
- SearchSettings searchSettings = new SearchSettings(props);
- assertThat(searchSettings.inCluster()).isFalse();
+ EsSettings esSettings = new EsSettings(props);
+ assertThat(esSettings.inCluster()).isFalse();
- Settings generated = searchSettings.build();
+ Settings generated = esSettings.build();
assertThat(generated.get("transport.tcp.port")).isEqualTo("9001");
assertThat(generated.get("transport.host")).isEqualTo("127.0.0.1");
assertThat(generated.get("cluster.name")).isEqualTo("sonarqube");
@@ -94,7 +94,7 @@ public class SearchSettingsTest {
props.set(ProcessProperties.PATH_LOGS, logDir.getAbsolutePath());
props.set(ProcessProperties.PATH_TEMP, tempDir.getAbsolutePath());
- Settings settings = new SearchSettings(props).build();
+ Settings settings = new EsSettings(props).build();
assertThat(settings.get("path.data")).isEqualTo(new File(dataDir, "es").getAbsolutePath());
assertThat(settings.get("path.logs")).isEqualTo(logDir.getAbsolutePath());
@@ -106,7 +106,7 @@ public class SearchSettingsTest {
Props props = minProps();
props.set(ProcessProperties.CLUSTER_ACTIVATE, "true");
props.set(ProcessProperties.CLUSTER_MASTER, "true");
- Settings settings = new SearchSettings(props).build();
+ Settings settings = new EsSettings(props).build();
assertThat(settings.get("index.number_of_replicas")).isEqualTo("1");
assertThat(settings.get("discovery.zen.ping.unicast.hosts")).isNull();
@@ -118,7 +118,7 @@ public class SearchSettingsTest {
Props props = minProps();
props.set(ProcessProperties.CLUSTER_ACTIVATE, "true");
props.set(ProcessProperties.CLUSTER_MASTER_HOST, "127.0.0.2,127.0.0.3");
- Settings settings = new SearchSettings(props).build();
+ Settings settings = new EsSettings(props).build();
assertThat(settings.get("discovery.zen.ping.unicast.hosts")).isEqualTo("127.0.0.2,127.0.0.3");
assertThat(settings.get("node.master")).isEqualTo("false");
@@ -129,7 +129,7 @@ public class SearchSettingsTest {
Props props = minProps();
props.set(ProcessProperties.CLUSTER_ACTIVATE, "true");
try {
- new SearchSettings(props).build();
+ new EsSettings(props).build();
fail();
} catch (MessageException ignored) {
// expected
@@ -139,8 +139,8 @@ public class SearchSettingsTest {
@Test
public void enable_marvel() throws Exception {
Props props = minProps();
- props.set(SearchSettings.PROP_MARVEL_HOSTS, "127.0.0.2,127.0.0.3");
- Settings settings = new SearchSettings(props).build();
+ props.set(EsSettings.PROP_MARVEL_HOSTS, "127.0.0.2,127.0.0.3");
+ Settings settings = new EsSettings(props).build();
assertThat(settings.get("marvel.agent.exporter.es.hosts")).isEqualTo("127.0.0.2,127.0.0.3");
}
@@ -149,7 +149,7 @@ public class SearchSettingsTest {
public void enable_http_connector() throws Exception {
Props props = minProps();
props.set(ProcessProperties.SEARCH_HTTP_PORT, "9010");
- Settings settings = new SearchSettings(props).build();
+ Settings settings = new EsSettings(props).build();
assertThat(settings.get("http.port")).isEqualTo("9010");
assertThat(settings.get("http.host")).isEqualTo("127.0.0.1");
@@ -161,7 +161,7 @@ public class SearchSettingsTest {
Props props = minProps();
props.set(ProcessProperties.SEARCH_HTTP_PORT, "9010");
props.set(ProcessProperties.SEARCH_HOST, "127.0.0.2");
- Settings settings = new SearchSettings(props).build();
+ Settings settings = new EsSettings(props).build();
assertThat(settings.get("http.port")).isEqualTo("9010");
assertThat(settings.get("http.host")).isEqualTo("127.0.0.2");
diff --git a/server/sonar-search/src/test/java/org/sonar/search/SearchServerTest.java b/server/sonar-search/src/test/java/org/sonar/search/SearchServerTest.java
index 5ce6cf13a0b..ebf7e106f4b 100644
--- a/server/sonar-search/src/test/java/org/sonar/search/SearchServerTest.java
+++ b/server/sonar-search/src/test/java/org/sonar/search/SearchServerTest.java
@@ -19,6 +19,8 @@
*/
package org.sonar.search;
+import java.net.InetAddress;
+import java.util.Properties;
import org.elasticsearch.action.admin.cluster.health.ClusterHealthStatus;
import org.elasticsearch.client.Client;
import org.elasticsearch.client.transport.NoNodeAvailableException;
@@ -34,12 +36,10 @@ import org.junit.rules.TemporaryFolder;
import org.junit.rules.TestRule;
import org.junit.rules.Timeout;
import org.sonar.process.NetworkUtils;
+import org.sonar.process.ProcessEntryPoint;
import org.sonar.process.ProcessProperties;
import org.sonar.process.Props;
-import java.net.InetAddress;
-import java.util.Properties;
-
import static org.assertj.core.api.Assertions.assertThat;
import static org.junit.Assert.fail;
@@ -78,6 +78,7 @@ public class SearchServerTest {
props.set(ProcessProperties.CLUSTER_NAME, CLUSTER_NAME);
props.set(ProcessProperties.CLUSTER_NODE_NAME, "test");
props.set(ProcessProperties.PATH_HOME, temp.newFolder().getAbsolutePath());
+ props.set(ProcessEntryPoint.PROPERTY_SHARED_PATH, temp.newFolder().getAbsolutePath());
searchServer = new SearchServer(props);
searchServer.start();
diff --git a/server/sonar-server/src/main/java/org/sonar/ce/CeModule.java b/server/sonar-server/src/main/java/org/sonar/ce/CeModule.java
index f4f05b14fba..1f4bb4b7f35 100644
--- a/server/sonar-server/src/main/java/org/sonar/ce/CeModule.java
+++ b/server/sonar-server/src/main/java/org/sonar/ce/CeModule.java
@@ -20,7 +20,6 @@
package org.sonar.ce;
import org.sonar.ce.log.CeLogging;
-import org.sonar.ce.monitoring.DummyCEQueueStatusImpl;
import org.sonar.ce.queue.CeQueueImpl;
import org.sonar.ce.queue.report.ReportFiles;
import org.sonar.ce.queue.report.ReportSubmitter;
@@ -32,9 +31,6 @@ public class CeModule extends Module {
protected void configureModule() {
add(CeLogging.class,
- // queue monitoring
- DummyCEQueueStatusImpl.class,
-
// Queue
CeQueueImpl.class,
ReportSubmitter.class,
diff --git a/server/sonar-server/src/main/java/org/sonar/ce/monitoring/CEQueueStatus.java b/server/sonar-server/src/main/java/org/sonar/ce/monitoring/CEQueueStatus.java
index 1ffac1e5ffd..695f50c2d47 100644
--- a/server/sonar-server/src/main/java/org/sonar/ce/monitoring/CEQueueStatus.java
+++ b/server/sonar-server/src/main/java/org/sonar/ce/monitoring/CEQueueStatus.java
@@ -22,57 +22,12 @@ package org.sonar.ce.monitoring;
public interface CEQueueStatus {
/**
- * Sets the count of reports waiting for processing at startup. This method can be called only once.
- *
- * @param initialPendingCount the count of reports, must be {@literal >=} 0
- *
- * @return the new count of batch reports waiting for processing (which is the same as the argument)
- *
- * @throws IllegalStateException if this method has already been called or is called after {@link #getPendingCount()}
- * @throws IllegalArgumentException if the argument is {@literal <} 0
- */
- long initPendingCount(long initialPendingCount);
-
- /**
- * Adds 1 to the count of received batch reports and 1 to the count of batch reports waiting for processing.
- * <p>
- * Calling this method is equivalent to calling {@link #addReceived(long)} with {@code 1} as argument but will
- * trigger no parameter check. So, it can be faster.
- * </p>
- *
- * @return the new count of received batch reports
- *
- * @see #getReceivedCount()
- * @see #getPendingCount()
- *
- * @throws IllegalStateException if {@link #initPendingCount(long)} has not been called yet
- */
- long addReceived();
-
- /**
- * Adds {@code numberOfReceived} to the count of received batch reports and {@code numberOfReceived} to the count of
- * batch reports waiting for processing.
- *
- * @return the new count of received batch reports
- *
- * @see #getReceivedCount()
- * @see #getPendingCount()
- *
- * @throws IllegalStateException if {@link #initPendingCount(long)} has not been called yet
- * @throws IllegalArgumentException if {@code numberOfReceived} is less or equal to 0
- */
- long addReceived(long numberOfReceived);
-
- /**
* Adds 1 to the count of batch reports under processing and removes 1 from the count of batch reports waiting for
* processing.
*
* @return the new count of batch reports under processing
*
* @see #getInProgressCount()
- * @see #getPendingCount()
- *
- * @throws IllegalStateException if {@link #initPendingCount(long)} has not been called yet
*/
long addInProgress();
@@ -107,11 +62,6 @@ public interface CEQueueStatus {
long addError(long processingTime);
/**
- * Count of received batch reports since instance startup
- */
- long getReceivedCount();
-
- /**
* Count of batch reports waiting for processing since startup, including reports received before instance startup.
*/
long getPendingCount();
@@ -132,7 +82,7 @@ public interface CEQueueStatus {
long getSuccessCount();
/**
- * Time spent processing batch reports since startup.
+ * Time spent processing batch reports since startup, in milliseconds.
*/
long getProcessingTime();
}
diff --git a/server/sonar-server/src/main/java/org/sonar/ce/monitoring/DummyCEQueueStatusImpl.java b/server/sonar-server/src/main/java/org/sonar/ce/monitoring/DummyCEQueueStatusImpl.java
deleted file mode 100644
index 9ed4fcc721a..00000000000
--- a/server/sonar-server/src/main/java/org/sonar/ce/monitoring/DummyCEQueueStatusImpl.java
+++ /dev/null
@@ -1,93 +0,0 @@
-/*
- * 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.ce.monitoring;
-
-import java.util.concurrent.atomic.AtomicLong;
-
-/**
- * FIXME fix this dummy CEQueueStatus implementation, probably by removing its use from
- */
-public class DummyCEQueueStatusImpl implements CEQueueStatus {
- private final AtomicLong received = new AtomicLong(0);
-
- @Override
- public long initPendingCount(long initialPendingCount) {
- return notImplemented();
- }
-
- @Override
- public long addReceived() {
- return received.incrementAndGet();
- }
-
- @Override
- public long addReceived(long numberOfReceived) {
- return received.addAndGet(numberOfReceived);
- }
-
- @Override
- public long addInProgress() {
- return notImplemented();
- }
-
- @Override
- public long addSuccess(long processingTime) {
- return notImplemented();
- }
-
- @Override
- public long addError(long processingTime) {
- return notImplemented();
- }
-
- @Override
- public long getReceivedCount() {
- return received.get();
- }
-
- @Override
- public long getPendingCount() {
- return notImplemented();
- }
-
- @Override
- public long getInProgressCount() {
- return notImplemented();
- }
-
- @Override
- public long getErrorCount() {
- return notImplemented();
- }
-
- @Override
- public long getSuccessCount() {
- return notImplemented();
- }
-
- @Override
- public long getProcessingTime() {
- return notImplemented();
- }
-
- private static long notImplemented() {
- throw new UnsupportedOperationException("Not implemented!");
- }
-}
diff --git a/server/sonar-server/src/main/java/org/sonar/ce/queue/CeQueueImpl.java b/server/sonar-server/src/main/java/org/sonar/ce/queue/CeQueueImpl.java
index a6aff101fc0..7aa61ad53ad 100644
--- a/server/sonar-server/src/main/java/org/sonar/ce/queue/CeQueueImpl.java
+++ b/server/sonar-server/src/main/java/org/sonar/ce/queue/CeQueueImpl.java
@@ -37,7 +37,6 @@ import org.sonar.db.DbSession;
import org.sonar.db.ce.CeActivityDto;
import org.sonar.db.ce.CeQueueDto;
import org.sonar.db.component.ComponentDto;
-import org.sonar.ce.monitoring.CEQueueStatus;
import static com.google.common.base.Preconditions.checkState;
import static com.google.common.base.Predicates.notNull;
@@ -49,7 +48,6 @@ public class CeQueueImpl implements CeQueue {
private final DbClient dbClient;
private final UuidFactory uuidFactory;
- private final CEQueueStatus queueStatus;
private final CeQueueListener[] listeners;
// state
@@ -58,14 +56,13 @@ public class CeQueueImpl implements CeQueue {
/**
* Constructor in case there is no CeQueueListener
*/
- public CeQueueImpl(DbClient dbClient, UuidFactory uuidFactory, CEQueueStatus queueStatus) {
- this(dbClient, uuidFactory, queueStatus, new CeQueueListener[]{});
+ public CeQueueImpl(DbClient dbClient, UuidFactory uuidFactory) {
+ this(dbClient, uuidFactory, new CeQueueListener[] {});
}
- public CeQueueImpl(DbClient dbClient, UuidFactory uuidFactory, CEQueueStatus queueStatus, CeQueueListener[] listeners) {
+ public CeQueueImpl(DbClient dbClient, UuidFactory uuidFactory, CeQueueListener[] listeners) {
this.dbClient = dbClient;
this.uuidFactory = uuidFactory;
- this.queueStatus = queueStatus;
this.listeners = listeners;
}
@@ -83,7 +80,6 @@ public class CeQueueImpl implements CeQueue {
CeQueueDto dto = new CeTaskSubmitToInsertedCeQueueDto(dbSession, dbClient).apply(submission);
CeTask task = loadTask(dbSession, dto);
dbSession.commit();
- queueStatus.addReceived();
return task;
} finally {
@@ -105,7 +101,6 @@ public class CeQueueImpl implements CeQueue {
.toList();
List<CeTask> tasks = loadTasks(dbSession, ceQueueDtos);
dbSession.commit();
- queueStatus.addReceived(tasks.size());
return tasks;
} finally {
diff --git a/server/sonar-server/src/main/java/org/sonar/server/app/ProcessCommandWrapper.java b/server/sonar-server/src/main/java/org/sonar/server/app/ProcessCommandWrapper.java
index 3310debda32..bb57d8e563e 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/app/ProcessCommandWrapper.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/app/ProcessCommandWrapper.java
@@ -29,4 +29,5 @@ public interface ProcessCommandWrapper {
* Notifies any listening process that the WebServer is operational.
*/
void notifyOperational();
+
}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/app/ProcessCommandWrapperImpl.java b/server/sonar-server/src/main/java/org/sonar/server/app/ProcessCommandWrapperImpl.java
index 9936a9f7e93..de29b5607e7 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/app/ProcessCommandWrapperImpl.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/app/ProcessCommandWrapperImpl.java
@@ -37,37 +37,41 @@ public class ProcessCommandWrapperImpl implements ProcessCommandWrapper {
@Override
public void requestSQRestart() {
- call(VoidMethod.ASK_FOR_RESTART);
+ call(VoidMethod.ASK_FOR_RESTART, selfProcessNumber());
}
@Override
public void notifyOperational() {
- call(VoidMethod.SET_OPERATIONAL);
+ call(VoidMethod.SET_OPERATIONAL, selfProcessNumber());
}
- private void call(VoidMethod command) {
+ private int selfProcessNumber() {
+ return nonNullAsInt(PROPERTY_PROCESS_INDEX);
+ }
+
+ private <T> T call(VoidMethod command, int processNumber) {
File shareDir = nonNullValueAsFile(PROPERTY_SHARED_PATH);
- int processNumber = nonNullAsInt(PROPERTY_PROCESS_INDEX);
try (DefaultProcessCommands commands = DefaultProcessCommands.secondary(shareDir, processNumber)) {
- command.callOn(commands);
+ return command.callOn(commands);
}
}
private enum VoidMethod {
SET_OPERATIONAL() {
@Override
- void callOn(ProcessCommands processCommands) {
+ <T> T callOn(ProcessCommands processCommands) {
processCommands.setOperational();
+ return null;
}
},
ASK_FOR_RESTART() {
@Override
- void callOn(ProcessCommands processCommands) {
+ <T> T callOn(ProcessCommands processCommands) {
processCommands.askForRestart();
+ return null;
}
};
-
- abstract void callOn(ProcessCommands processCommands);
+ abstract <T> T callOn(ProcessCommands processCommands);
}
private int nonNullAsInt(String key) {
@@ -76,7 +80,7 @@ public class ProcessCommandWrapperImpl implements ProcessCommandWrapper {
return Integer.parseInt(s);
}
- public File nonNullValueAsFile(String key) {
+ private File nonNullValueAsFile(String key) {
String s = settings.getString(key);
checkArgument(s != null, "Property %s is not set", key);
return new File(s);
diff --git a/server/sonar-server/src/main/java/org/sonar/server/computation/CeModule.java b/server/sonar-server/src/main/java/org/sonar/server/computation/CeModule.java
index 25c04f2e25b..4cdf80747d4 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/computation/CeModule.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/computation/CeModule.java
@@ -25,6 +25,7 @@ import org.sonar.db.purge.period.DefaultPeriodCleaner;
import org.sonar.server.computation.configuration.CeConfigurationImpl;
import org.sonar.server.computation.dbcleaner.IndexPurgeListener;
import org.sonar.server.computation.dbcleaner.ProjectCleaner;
+import org.sonar.server.computation.monitoring.CeDatabaseMBeanImpl;
/**
* Globally available components in CE
@@ -35,6 +36,7 @@ public class CeModule extends Module {
add(
CeConfigurationImpl.class,
CeLogging.class,
+ CeDatabaseMBeanImpl.class,
DefaultPeriodCleaner.class,
ProjectCleaner.class,
diff --git a/server/sonar-server/src/main/java/org/sonar/server/computation/monitoring/CEQueueStatusImpl.java b/server/sonar-server/src/main/java/org/sonar/server/computation/monitoring/CEQueueStatusImpl.java
index d9e61e25e15..90a5074f6c2 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/computation/monitoring/CEQueueStatusImpl.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/computation/monitoring/CEQueueStatusImpl.java
@@ -21,86 +21,53 @@ package org.sonar.server.computation.monitoring;
import java.util.concurrent.atomic.AtomicLong;
import org.sonar.ce.monitoring.CEQueueStatus;
+import org.sonar.db.DbClient;
+import org.sonar.db.DbSession;
+import org.sonar.db.ce.CeQueueDto;
import static com.google.common.base.Preconditions.checkArgument;
-import static com.google.common.base.Preconditions.checkState;
public class CEQueueStatusImpl implements CEQueueStatus {
- private static final long PENDING_INITIAL_VALUE = Long.MIN_VALUE;
- private final AtomicLong received = new AtomicLong(0);
- private final AtomicLong pending = new AtomicLong(PENDING_INITIAL_VALUE);
+ private final DbClient dbClient;
private final AtomicLong inProgress = new AtomicLong(0);
private final AtomicLong error = new AtomicLong(0);
private final AtomicLong success = new AtomicLong(0);
private final AtomicLong processingTime = new AtomicLong(0);
- @Override
- public long initPendingCount(long initialPendingCount) {
- checkArgument(initialPendingCount >= 0, "Initial pending count must be >= 0");
- checkState(
- pending.compareAndSet(PENDING_INITIAL_VALUE, initialPendingCount),
- "Method initPendingCount must be used before any other method and can not be called twice");
- return initialPendingCount;
- }
-
- @Override
- public long addReceived() {
- ensurePendingInitialized("addReceived");
-
- pending.incrementAndGet();
- return received.incrementAndGet();
- }
-
- @Override
- public long addReceived(long numberOfReceived) {
- ensurePendingInitialized("addReceived");
- checkArgument(numberOfReceived > 0, "numberOfReceived must be > 0");
-
- pending.addAndGet(numberOfReceived);
- return received.addAndGet(numberOfReceived);
+ public CEQueueStatusImpl(DbClient dbClient) {
+ this.dbClient = dbClient;
}
@Override
public long addInProgress() {
- ensurePendingInitialized("addInProgress");
-
- pending.decrementAndGet();
return inProgress.incrementAndGet();
}
- private void ensurePendingInitialized(String methodName) {
- checkState(pending.get() != PENDING_INITIAL_VALUE, "Method initPendingCount must be used before %s can be called", methodName);
- }
-
@Override
- public long addError(long processingTime) {
- addProcessingTime(processingTime);
+ public long addError(long processingTimeInMs) {
+ addProcessingTime(processingTimeInMs);
inProgress.decrementAndGet();
return error.incrementAndGet();
}
@Override
- public long addSuccess(long processingTime) {
- addProcessingTime(processingTime);
+ public long addSuccess(long processingTimeInMs) {
+ addProcessingTime(processingTimeInMs);
inProgress.decrementAndGet();
return success.incrementAndGet();
}
- private void addProcessingTime(long time) {
- checkArgument(time >= 0, "Processing time can not be < 0");
- processingTime.addAndGet(time);
- }
-
- @Override
- public long getReceivedCount() {
- return received.get();
+ private void addProcessingTime(long ms) {
+ checkArgument(ms >= 0, "Processing time can not be < 0");
+ processingTime.addAndGet(ms);
}
@Override
public long getPendingCount() {
- long currentValue = pending.get();
- return currentValue == PENDING_INITIAL_VALUE ? 0 : currentValue;
+ try (DbSession dbSession = dbClient.openSession(false)) {
+ return dbClient.ceQueueDao().countByStatus(dbSession, CeQueueDto.Status.PENDING);
+ }
}
@Override
diff --git a/server/sonar-server/src/main/java/org/sonar/server/computation/monitoring/CeDatabaseMBeanImpl.java b/server/sonar-server/src/main/java/org/sonar/server/computation/monitoring/CeDatabaseMBeanImpl.java
new file mode 100644
index 00000000000..4b7e43a3fa5
--- /dev/null
+++ b/server/sonar-server/src/main/java/org/sonar/server/computation/monitoring/CeDatabaseMBeanImpl.java
@@ -0,0 +1,97 @@
+/*
+ * 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.server.computation.monitoring;
+
+import org.apache.commons.dbcp.BasicDataSource;
+import org.picocontainer.Startable;
+import org.sonar.db.DbClient;
+import org.sonar.process.jmx.CeDatabaseMBean;
+import org.sonar.process.jmx.Jmx;
+
+public class CeDatabaseMBeanImpl implements CeDatabaseMBean, Startable {
+ private final DbClient dbClient;
+
+ public CeDatabaseMBeanImpl(DbClient dbClient) {
+ this.dbClient = dbClient;
+ }
+
+ @Override
+ public void start() {
+ Jmx.register(OBJECT_NAME, this);
+ }
+
+ /**
+ * Unregister, if needed
+ */
+ @Override
+ public void stop() {
+ Jmx.unregister(OBJECT_NAME);
+ }
+
+ @Override
+ public int getPoolActiveConnections() {
+ return commonsDbcp().getNumActive();
+ }
+
+ @Override
+ public int getPoolMaxActiveConnections() {
+ return commonsDbcp().getMaxActive();
+ }
+
+ @Override
+ public int getPoolIdleConnections() {
+ return commonsDbcp().getNumIdle();
+ }
+
+ @Override
+ public int getPoolMaxIdleConnections() {
+ return commonsDbcp().getMaxIdle();
+ }
+
+ @Override
+ public int getPoolMinIdleConnections() {
+ return commonsDbcp().getMinIdle();
+ }
+
+ @Override
+ public int getPoolInitialSize() {
+ return commonsDbcp().getInitialSize();
+ }
+
+ @Override
+ public long getPoolMaxWaitMillis() {
+ return commonsDbcp().getMaxWait();
+ }
+
+ @Override
+ public boolean getPoolRemoveAbandoned() {
+ return commonsDbcp().getRemoveAbandoned();
+ }
+
+ @Override
+ public int getPoolRemoveAbandonedTimeoutSeconds() {
+ return commonsDbcp().getRemoveAbandonedTimeout();
+ }
+
+ private BasicDataSource commonsDbcp() {
+ return (BasicDataSource) dbClient.getDatabase().getDataSource();
+ }
+
+}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/computation/monitoring/ComputeEngineQueueMonitor.java b/server/sonar-server/src/main/java/org/sonar/server/computation/monitoring/CeTasksMBeanImpl.java
index 94d2322dc1c..57b08e356de 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/computation/monitoring/ComputeEngineQueueMonitor.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/computation/monitoring/CeTasksMBeanImpl.java
@@ -19,47 +19,32 @@
*/
package org.sonar.server.computation.monitoring;
-import java.util.LinkedHashMap;
+import org.picocontainer.Startable;
import org.sonar.ce.monitoring.CEQueueStatus;
+import org.sonar.process.jmx.CeTasksMBean;
+import org.sonar.process.jmx.Jmx;
import org.sonar.server.computation.configuration.CeConfiguration;
-import org.sonar.ce.queue.CeQueue;
-import org.sonar.server.platform.monitoring.BaseMonitorMBean;
-public class ComputeEngineQueueMonitor extends BaseMonitorMBean implements ComputeEngineQueueMonitorMBean {
+public class CeTasksMBeanImpl implements CeTasksMBean, Startable {
private final CEQueueStatus queueStatus;
private final CeConfiguration ceConfiguration;
- public ComputeEngineQueueMonitor(CEQueueStatus queueStatus,
- // ReportQueue initializes CEQueueStatus and is therefor a dependency of
- // ComputeEngineQueueMonitor.
- // Do not remove this parameter, it ensures start order of components
- CeQueue ceQueue,
- CeConfiguration ceConfiguration) {
+ public CeTasksMBeanImpl(CEQueueStatus queueStatus, CeConfiguration ceConfiguration) {
this.queueStatus = queueStatus;
this.ceConfiguration = ceConfiguration;
}
@Override
- public String name() {
- return "ComputeEngine";
+ public void start() {
+ Jmx.register(OBJECT_NAME, this);
}
+ /**
+ * Unregister, if needed
+ */
@Override
- public LinkedHashMap<String, Object> attributes() {
- LinkedHashMap<String, Object> attributes = new LinkedHashMap<>();
- attributes.put("Received", getReceivedCount());
- attributes.put("Pending", getPendingCount());
- attributes.put("In progress", getInProgressCount());
- attributes.put("Successfully processed", getSuccessCount());
- attributes.put("Processed with error", getErrorCount());
- attributes.put("Processing time", getProcessingTime());
- attributes.put("Worker count", getWorkerCount());
- return attributes;
- }
-
- @Override
- public long getReceivedCount() {
- return queueStatus.getReceivedCount();
+ public void stop() {
+ Jmx.unregister(OBJECT_NAME);
}
@Override
diff --git a/server/sonar-server/src/main/java/org/sonar/server/computation/queue/CeQueueInitializer.java b/server/sonar-server/src/main/java/org/sonar/server/computation/queue/CeQueueInitializer.java
index b6f277f8f8f..eaa67a7fc5d 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/computation/queue/CeQueueInitializer.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/computation/queue/CeQueueInitializer.java
@@ -24,7 +24,6 @@ import org.sonar.api.platform.ServerStartHandler;
import org.sonar.api.server.ServerSide;
import org.sonar.db.DbClient;
import org.sonar.db.DbSession;
-import org.sonar.ce.monitoring.CEQueueStatus;
import org.sonar.server.computation.taskprocessor.CeProcessingScheduler;
/**
@@ -36,14 +35,12 @@ import org.sonar.server.computation.taskprocessor.CeProcessingScheduler;
public class CeQueueInitializer implements ServerStartHandler {
private final DbClient dbClient;
- private final CEQueueStatus queueStatus;
private final CeQueueCleaner cleaner;
private final CeProcessingScheduler scheduler;
private boolean done = false;
- public CeQueueInitializer(DbClient dbClient, CEQueueStatus queueStatus, CeQueueCleaner cleaner, CeProcessingScheduler scheduler) {
+ public CeQueueInitializer(DbClient dbClient, CeQueueCleaner cleaner, CeProcessingScheduler scheduler) {
this.dbClient = dbClient;
- this.queueStatus = queueStatus;
this.cleaner = cleaner;
this.scheduler = scheduler;
}
@@ -59,7 +56,6 @@ public class CeQueueInitializer implements ServerStartHandler {
private void initCe() {
DbSession dbSession = dbClient.openSession(false);
try {
- initJmxCounters(dbSession);
cleaner.clean(dbSession);
scheduler.startScheduling();
@@ -67,8 +63,4 @@ public class CeQueueInitializer implements ServerStartHandler {
dbClient.closeSession(dbSession);
}
}
-
- private void initJmxCounters(DbSession dbSession) {
- queueStatus.initPendingCount(dbClient.ceQueueDao().countAll(dbSession));
- }
}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/computation/queue/CeQueueModule.java b/server/sonar-server/src/main/java/org/sonar/server/computation/queue/CeQueueModule.java
index 4be44d6936b..77b676f7dfc 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/computation/queue/CeQueueModule.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/computation/queue/CeQueueModule.java
@@ -19,11 +19,11 @@
*/
package org.sonar.server.computation.queue;
-import org.sonar.server.computation.queue.report.CleanReportQueueListener;
import org.sonar.ce.queue.report.ReportFiles;
import org.sonar.core.platform.Module;
import org.sonar.server.computation.monitoring.CEQueueStatusImpl;
-import org.sonar.server.computation.monitoring.ComputeEngineQueueMonitor;
+import org.sonar.server.computation.monitoring.CeTasksMBeanImpl;
+import org.sonar.server.computation.queue.report.CleanReportQueueListener;
public class CeQueueModule extends Module {
@Override
@@ -34,7 +34,7 @@ public class CeQueueModule extends Module {
// queue monitoring
CEQueueStatusImpl.class,
- ComputeEngineQueueMonitor.class,
+ CeTasksMBeanImpl.class,
// queue cleaning
CeQueueCleaner.class,
diff --git a/server/sonar-server/src/main/java/org/sonar/server/computation/queue/InternalCeQueueImpl.java b/server/sonar-server/src/main/java/org/sonar/server/computation/queue/InternalCeQueueImpl.java
index d8b10f5c5d3..3e0f988270a 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/computation/queue/InternalCeQueueImpl.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/computation/queue/InternalCeQueueImpl.java
@@ -24,6 +24,7 @@ import java.util.concurrent.atomic.AtomicBoolean;
import javax.annotation.Nullable;
import org.sonar.api.server.ServerSide;
import org.sonar.api.utils.System2;
+import org.sonar.ce.monitoring.CEQueueStatus;
import org.sonar.ce.queue.CeQueueImpl;
import org.sonar.ce.queue.CeQueueListener;
import org.sonar.ce.queue.CeTask;
@@ -33,7 +34,6 @@ import org.sonar.db.DbClient;
import org.sonar.db.DbSession;
import org.sonar.db.ce.CeActivityDto;
import org.sonar.db.ce.CeQueueDto;
-import org.sonar.ce.monitoring.CEQueueStatus;
import static java.lang.String.format;
@@ -49,7 +49,7 @@ public class InternalCeQueueImpl extends CeQueueImpl implements InternalCeQueue
public InternalCeQueueImpl(System2 system2, DbClient dbClient, UuidFactory uuidFactory,
CEQueueStatus queueStatus, CeQueueListener[] listeners) {
- super(dbClient, uuidFactory, queueStatus, listeners);
+ super(dbClient, uuidFactory, listeners);
this.system2 = system2;
this.dbClient = dbClient;
this.queueStatus = queueStatus;
@@ -114,12 +114,12 @@ public class InternalCeQueueImpl extends CeQueueImpl implements InternalCeQueue
return;
}
activityDto.setExecutedAt(system2.now());
- long executionTime = activityDto.getExecutedAt() - startedAt;
- activityDto.setExecutionTimeMs(executionTime);
+ long executionTimeInMs = activityDto.getExecutedAt() - startedAt;
+ activityDto.setExecutionTimeMs(executionTimeInMs);
if (status == CeActivityDto.Status.SUCCESS) {
- queueStatus.addSuccess(executionTime);
+ queueStatus.addSuccess(executionTimeInMs);
} else {
- queueStatus.addError(executionTime);
+ queueStatus.addError(executionTimeInMs);
}
}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/BaseMonitorMBean.java b/server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/BaseMonitorMBean.java
index bbc8ef67e6e..b56d9a9346e 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/BaseMonitorMBean.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/BaseMonitorMBean.java
@@ -20,17 +20,10 @@
package org.sonar.server.platform.monitoring;
import org.picocontainer.Startable;
-
-import javax.management.InstanceNotFoundException;
-import javax.management.MBeanRegistrationException;
-import javax.management.MalformedObjectNameException;
-import javax.management.ObjectName;
-import javax.management.OperationsException;
-
-import java.lang.management.ManagementFactory;
+import org.sonar.process.jmx.Jmx;
/**
- * Base implementation of {@link org.sonar.server.platform.monitoring.Monitor}
+ * Base implementation of a {@link org.sonar.server.platform.monitoring.Monitor}
* that is exported as a JMX bean
*/
public abstract class BaseMonitorMBean implements Monitor, Startable {
@@ -40,11 +33,7 @@ public abstract class BaseMonitorMBean implements Monitor, Startable {
*/
@Override
public void start() {
- try {
- ManagementFactory.getPlatformMBeanServer().registerMBean(this, objectName());
- } catch (OperationsException | MBeanRegistrationException e) {
- throw new IllegalStateException("Fail to register MBean " + name(), e);
- }
+ Jmx.register(objectName(), this);
}
/**
@@ -52,16 +41,10 @@ public abstract class BaseMonitorMBean implements Monitor, Startable {
*/
@Override
public void stop() {
- try {
- ManagementFactory.getPlatformMBeanServer().unregisterMBean(objectName());
- } catch (InstanceNotFoundException ignored) {
- // ignore, was not correctly started
- } catch (Exception e) {
- throw new IllegalStateException("Fail to unregister MBean " + name(), e);
- }
+ Jmx.unregister(objectName());
}
- ObjectName objectName() throws MalformedObjectNameException {
- return new ObjectName(String.format("SonarQube:name=%s", name()));
+ String objectName() {
+ return "SonarQube:name=" + name();
}
}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/CeDatabaseMonitor.java b/server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/CeDatabaseMonitor.java
new file mode 100644
index 00000000000..1b6b548e379
--- /dev/null
+++ b/server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/CeDatabaseMonitor.java
@@ -0,0 +1,58 @@
+/*
+ * 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.server.platform.monitoring;
+
+import java.util.LinkedHashMap;
+import org.sonar.process.ProcessId;
+import org.sonar.process.jmx.CeDatabaseMBean;
+import org.sonar.process.jmx.JmxConnection;
+import org.sonar.process.jmx.JmxConnectionFactory;
+
+public class CeDatabaseMonitor implements Monitor {
+
+ private final JmxConnectionFactory jmxConnectionFactory;
+
+ public CeDatabaseMonitor(JmxConnectionFactory jmxConnectionFactory) {
+ this.jmxConnectionFactory = jmxConnectionFactory;
+ }
+
+ @Override
+ public String name() {
+ return "Compute Engine Database";
+ }
+
+ @Override
+ public LinkedHashMap<String, Object> attributes() {
+ try (JmxConnection connection = jmxConnectionFactory.create(ProcessId.COMPUTE_ENGINE)) {
+ LinkedHashMap<String, Object> result = new LinkedHashMap<>();
+ CeDatabaseMBean mbean = connection.getMBean(CeDatabaseMBean.OBJECT_NAME, CeDatabaseMBean.class);
+ result.put("Pool Initial Size", mbean.getPoolInitialSize());
+ result.put("Pool Active Connections", mbean.getPoolActiveConnections());
+ result.put("Pool Idle Connections", mbean.getPoolIdleConnections());
+ result.put("Pool Max Active Connections", mbean.getPoolMaxActiveConnections());
+ result.put("Pool Max Idle Connections", mbean.getPoolMaxIdleConnections());
+ result.put("Pool Min Idle Connections", mbean.getPoolMinIdleConnections());
+ result.put("Pool Max Wait (ms)", mbean.getPoolMaxWaitMillis());
+ result.put("Pool Remove Abandoned", mbean.getPoolRemoveAbandoned());
+ result.put("Pool Remove Abandoned Timeout (sec)", mbean.getPoolRemoveAbandonedTimeoutSeconds());
+ return result;
+ }
+ }
+}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/CeStateMonitor.java b/server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/CeStateMonitor.java
new file mode 100644
index 00000000000..2c834fe20b7
--- /dev/null
+++ b/server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/CeStateMonitor.java
@@ -0,0 +1,46 @@
+/*
+ * 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.server.platform.monitoring;
+
+import java.util.LinkedHashMap;
+import org.sonar.process.ProcessId;
+import org.sonar.process.jmx.JmxConnection;
+import org.sonar.process.jmx.JmxConnectionFactory;
+
+public class CeStateMonitor implements Monitor {
+
+ private final JmxConnectionFactory jmxConnectionFactory;
+
+ public CeStateMonitor(JmxConnectionFactory jmxConnectionFactory) {
+ this.jmxConnectionFactory = jmxConnectionFactory;
+ }
+
+ @Override
+ public String name() {
+ return "Compute Engine State";
+ }
+
+ @Override
+ public LinkedHashMap<String, Object> attributes() {
+ try (JmxConnection connection = jmxConnectionFactory.create(ProcessId.COMPUTE_ENGINE)) {
+ return new LinkedHashMap<>(connection.getSystemState());
+ }
+ }
+}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/CeTasksMonitor.java b/server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/CeTasksMonitor.java
new file mode 100644
index 00000000000..77c1af84c53
--- /dev/null
+++ b/server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/CeTasksMonitor.java
@@ -0,0 +1,55 @@
+/*
+ * 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.server.platform.monitoring;
+
+import java.util.LinkedHashMap;
+import org.sonar.process.ProcessId;
+import org.sonar.process.jmx.CeTasksMBean;
+import org.sonar.process.jmx.JmxConnection;
+import org.sonar.process.jmx.JmxConnectionFactory;
+
+public class CeTasksMonitor implements Monitor {
+
+ private final JmxConnectionFactory jmxConnectionFactory;
+
+ public CeTasksMonitor(JmxConnectionFactory jmxConnectionFactory) {
+ this.jmxConnectionFactory = jmxConnectionFactory;
+ }
+
+ @Override
+ public String name() {
+ return "Compute Engine Tasks";
+ }
+
+ @Override
+ public LinkedHashMap<String, Object> attributes() {
+ try (JmxConnection connection = jmxConnectionFactory.create(ProcessId.COMPUTE_ENGINE)) {
+ LinkedHashMap<String, Object> result = new LinkedHashMap<>();
+ CeTasksMBean ceMBean = connection.getMBean(CeTasksMBean.OBJECT_NAME, CeTasksMBean.class);
+ result.put("Pending", ceMBean.getPendingCount());
+ result.put("In Progress", ceMBean.getInProgressCount());
+ result.put("Processed With Success", ceMBean.getSuccessCount());
+ result.put("Processed With Error", ceMBean.getErrorCount());
+ result.put("Processing Time (ms)", ceMBean.getProcessingTime());
+ result.put("Worker Count", ceMBean.getWorkerCount());
+ return result;
+ }
+ }
+}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/DatabaseMonitor.java b/server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/DatabaseMonitor.java
index cdadf46aa1a..ceb0caf6ff0 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/DatabaseMonitor.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/DatabaseMonitor.java
@@ -19,18 +19,17 @@
*/
package org.sonar.server.platform.monitoring;
-import org.apache.commons.dbcp.BasicDataSource;
-import org.apache.commons.dbutils.DbUtils;
-import org.sonar.db.version.DatabaseVersion;
-import org.sonar.db.DbSession;
-import org.sonar.db.MyBatis;
-import org.sonar.db.DbClient;
-
import java.sql.Connection;
import java.sql.DatabaseMetaData;
import java.sql.SQLException;
import java.util.LinkedHashMap;
import java.util.Map;
+import org.apache.commons.dbcp.BasicDataSource;
+import org.apache.commons.dbutils.DbUtils;
+import org.sonar.db.DbClient;
+import org.sonar.db.DbSession;
+import org.sonar.db.MyBatis;
+import org.sonar.db.version.DatabaseVersion;
/**
* Information about database and connection pool
diff --git a/server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/EsMonitor.java b/server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/EsMonitor.java
index 470676bdd76..0317ba2c1b0 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/EsMonitor.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/EsMonitor.java
@@ -27,22 +27,29 @@ import org.elasticsearch.action.admin.cluster.node.stats.NodesStatsResponse;
import org.elasticsearch.action.admin.cluster.stats.ClusterStatsResponse;
import org.elasticsearch.action.admin.indices.stats.IndexStats;
import org.elasticsearch.action.admin.indices.stats.IndicesStatsResponse;
+import org.elasticsearch.cluster.node.DiscoveryNode;
import org.elasticsearch.common.breaker.CircuitBreaker;
+import org.sonar.process.ProcessId;
+import org.sonar.process.jmx.EsSettingsMBean;
+import org.sonar.process.jmx.JmxConnection;
+import org.sonar.process.jmx.JmxConnectionFactory;
import org.sonar.server.es.EsClient;
import static org.apache.commons.io.FileUtils.byteCountToDisplaySize;
public class EsMonitor extends BaseMonitorMBean implements EsMonitorMBean {
+ private final JmxConnectionFactory jmxConnectionFactory;
private final EsClient esClient;
- public EsMonitor(EsClient esClient) {
+ public EsMonitor(JmxConnectionFactory jmxConnectionFactory, EsClient esClient) {
+ this.jmxConnectionFactory = jmxConnectionFactory;
this.esClient = esClient;
}
@Override
public String name() {
- return "ElasticSearch";
+ return "Elasticsearch";
}
/**
@@ -66,6 +73,13 @@ public class EsMonitor extends BaseMonitorMBean implements EsMonitorMBean {
@Override
public LinkedHashMap<String, Object> attributes() {
LinkedHashMap<String, Object> attributes = new LinkedHashMap<>();
+
+ try (JmxConnection connection = jmxConnectionFactory.create(ProcessId.ELASTICSEARCH)) {
+ EsSettingsMBean mbean = connection.getMBean(EsSettingsMBean.OBJECT_NAME, EsSettingsMBean.class);
+ attributes.put("Cluster Name", mbean.getClusterName());
+ attributes.put("Node Name", mbean.getNodeName());
+ attributes.put("HTTP Port", mbean.getHttpPort());
+ }
attributes.put("State", getStateAsEnum());
attributes.put("Indices", indexAttributes());
attributes.put("Number of Nodes", getNumberOfNodes());
@@ -96,10 +110,11 @@ public class EsMonitor extends BaseMonitorMBean implements EsMonitorMBean {
for (Map.Entry<String, NodeStats> entry : nodesStats.getNodesMap().entrySet()) {
LinkedHashMap<String, Object> nodeAttributes = new LinkedHashMap<>();
- nodes.put(entry.getKey(), nodeAttributes);
NodeStats stats = entry.getValue();
- nodeAttributes.put("Address", stats.getNode().getAddress().toString());
- nodeAttributes.put("Type", stats.getNode().isMasterNode() ? "Master" : "Slave");
+ DiscoveryNode node = stats.getNode();
+ nodes.put(node.getName(), nodeAttributes);
+ nodeAttributes.put("Address", node.getAddress().toString());
+ nodeAttributes.put("Type", node.isMasterNode() ? "Master" : "Slave");
nodeAttributes.put("Disk Available", byteCountToDisplaySize(stats.getFs().getTotal().getAvailable().bytes()));
nodeAttributes.put("Store Size", byteCountToDisplaySize(stats.getIndices().getStore().getSizeInBytes()));
nodeAttributes.put("Open Files", stats.getProcess().getOpenFileDescriptors());
diff --git a/server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/EsStateMonitor.java b/server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/EsStateMonitor.java
new file mode 100644
index 00000000000..0f8d60c4b07
--- /dev/null
+++ b/server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/EsStateMonitor.java
@@ -0,0 +1,46 @@
+/*
+ * 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.server.platform.monitoring;
+
+import java.util.LinkedHashMap;
+import org.sonar.process.ProcessId;
+import org.sonar.process.jmx.JmxConnection;
+import org.sonar.process.jmx.JmxConnectionFactory;
+
+public class EsStateMonitor implements Monitor {
+
+ private final JmxConnectionFactory jmxConnectionFactory;
+
+ public EsStateMonitor(JmxConnectionFactory jmxConnectionFactory) {
+ this.jmxConnectionFactory = jmxConnectionFactory;
+ }
+
+ @Override
+ public String name() {
+ return "Elasticsearch State";
+ }
+
+ @Override
+ public LinkedHashMap<String, Object> attributes() {
+ try (JmxConnection connection = jmxConnectionFactory.create(ProcessId.ELASTICSEARCH)) {
+ return new LinkedHashMap<>(connection.getSystemState());
+ }
+ }
+}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/JmxConnectorProvider.java b/server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/JmxConnectorProvider.java
new file mode 100644
index 00000000000..2bc9d247607
--- /dev/null
+++ b/server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/JmxConnectorProvider.java
@@ -0,0 +1,46 @@
+/*
+ * 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.server.platform.monitoring;
+
+import java.io.File;
+import org.picocontainer.injectors.ProviderAdapter;
+import org.sonar.api.config.Settings;
+import org.sonar.process.jmx.JmxConnectionFactory;
+
+import static com.google.common.base.Preconditions.checkArgument;
+import static org.sonar.process.ProcessEntryPoint.PROPERTY_SHARED_PATH;
+
+public class JmxConnectorProvider extends ProviderAdapter {
+
+ private JmxConnectionFactory singleton = null;
+
+ public synchronized JmxConnectionFactory provide(Settings settings) {
+ if (singleton == null) {
+ singleton = new JmxConnectionFactory(nonNullValueAsFile(settings, PROPERTY_SHARED_PATH));
+ }
+ return singleton;
+ }
+
+ private static File nonNullValueAsFile(Settings settings, String key) {
+ String s = settings.getString(key);
+ checkArgument(s != null, "Property %s is not set", key);
+ return new File(s);
+ }
+}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/JvmPropertiesMonitor.java b/server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/JvmPropsMonitor.java
index 0a0a8f8cb9c..3d4b33e292b 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/JvmPropertiesMonitor.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/platform/monitoring/JvmPropsMonitor.java
@@ -20,13 +20,12 @@
package org.sonar.server.platform.monitoring;
import com.google.common.collect.Maps;
-
import java.util.LinkedHashMap;
import java.util.Map;
import java.util.Objects;
import java.util.SortedMap;
-public class JvmPropertiesMonitor implements Monitor {
+public class JvmPropsMonitor implements Monitor {
@Override
public String name() {
return "JvmProperties";
diff --git a/server/sonar-server/src/main/java/org/sonar/server/platform/platformlevel/PlatformLevel4.java b/server/sonar-server/src/main/java/org/sonar/server/platform/platformlevel/PlatformLevel4.java
index 84e8156aa3c..7a39ad8281a 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/platform/platformlevel/PlatformLevel4.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/platform/platformlevel/PlatformLevel4.java
@@ -158,9 +158,14 @@ import org.sonar.server.permission.ws.PermissionsWsModule;
import org.sonar.server.platform.BackendCleanup;
import org.sonar.server.platform.ServerLogging;
import org.sonar.server.platform.SettingsChangeNotifier;
+import org.sonar.server.platform.monitoring.CeDatabaseMonitor;
+import org.sonar.server.platform.monitoring.CeStateMonitor;
+import org.sonar.server.platform.monitoring.CeTasksMonitor;
import org.sonar.server.platform.monitoring.DatabaseMonitor;
import org.sonar.server.platform.monitoring.EsMonitor;
-import org.sonar.server.platform.monitoring.JvmPropertiesMonitor;
+import org.sonar.server.platform.monitoring.EsStateMonitor;
+import org.sonar.server.platform.monitoring.JmxConnectorProvider;
+import org.sonar.server.platform.monitoring.JvmPropsMonitor;
import org.sonar.server.platform.monitoring.PluginsMonitor;
import org.sonar.server.platform.monitoring.SonarQubeMonitor;
import org.sonar.server.platform.monitoring.SystemMonitor;
@@ -658,6 +663,7 @@ public class PlatformLevel4 extends PlatformLevel {
TypeValidationModule.class,
// System
+ new JmxConnectorProvider(),
ServerLogging.class,
RestartAction.class,
InfoAction.class,
@@ -668,8 +674,12 @@ public class PlatformLevel4 extends PlatformLevel {
SonarQubeMonitor.class,
EsMonitor.class,
PluginsMonitor.class,
- JvmPropertiesMonitor.class,
+ JvmPropsMonitor.class,
DatabaseMonitor.class,
+ EsStateMonitor.class,
+ CeStateMonitor.class,
+ CeTasksMonitor.class,
+ CeDatabaseMonitor.class,
MigrateDbAction.class,
LogsAction.class,
ChangeLogLevelAction.class,
diff --git a/server/sonar-server/src/test/java/org/sonar/ce/monitoring/CEQueueStatusImplConcurrentTest.java b/server/sonar-server/src/test/java/org/sonar/ce/monitoring/CEQueueStatusImplConcurrentTest.java
index edaaf9cd3f8..63c39f6d995 100644
--- a/server/sonar-server/src/test/java/org/sonar/ce/monitoring/CEQueueStatusImplConcurrentTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/ce/monitoring/CEQueueStatusImplConcurrentTest.java
@@ -28,9 +28,11 @@ import java.util.concurrent.ThreadFactory;
import java.util.concurrent.TimeUnit;
import org.junit.After;
import org.junit.Test;
+import org.sonar.db.DbClient;
import org.sonar.server.computation.monitoring.CEQueueStatusImpl;
import static org.assertj.core.api.Assertions.assertThat;
+import static org.mockito.Mockito.mock;
public class CEQueueStatusImplConcurrentTest {
private ExecutorService executorService = Executors.newFixedThreadPool(10, new ThreadFactory() {
@@ -41,7 +43,7 @@ public class CEQueueStatusImplConcurrentTest {
return new Thread(r, CEQueueStatusImplConcurrentTest.class.getSimpleName() + cnt++);
}
});
- private CEQueueStatusImpl underTest = new CEQueueStatusImpl();
+ private CEQueueStatusImpl underTest = new CEQueueStatusImpl(mock(DbClient.class));
@After
public void tearDown() throws Exception {
@@ -50,17 +52,12 @@ public class CEQueueStatusImplConcurrentTest {
@Test
public void test_concurrent_modifications_in_any_order() throws InterruptedException {
- long initialPendingCount = 9963L;
- underTest.initPendingCount(initialPendingCount);
-
for (Runnable runnable : buildShuffleCallsToUnderTest()) {
executorService.submit(runnable);
}
executorService.awaitTermination(1, TimeUnit.SECONDS);
- assertThat(underTest.getReceivedCount()).isEqualTo(100);
- assertThat(underTest.getPendingCount()).isEqualTo(initialPendingCount + 2);
assertThat(underTest.getInProgressCount()).isEqualTo(1);
assertThat(underTest.getErrorCount()).isEqualTo(17);
assertThat(underTest.getSuccessCount()).isEqualTo(80);
@@ -69,9 +66,6 @@ public class CEQueueStatusImplConcurrentTest {
private List<Runnable> buildShuffleCallsToUnderTest() {
List<Runnable> res = new ArrayList<>();
- for (int i = 0; i < 100; i++) {
- res.add(new AddReceivedRunnable());
- }
for (int i = 0; i < 98; i++) {
res.add(new AddInProgressRunnable());
}
@@ -85,13 +79,6 @@ public class CEQueueStatusImplConcurrentTest {
return res;
}
- private class AddReceivedRunnable implements Runnable {
- @Override
- public void run() {
- underTest.addReceived();
- }
- }
-
private class AddInProgressRunnable implements Runnable {
@Override
public void run() {
diff --git a/server/sonar-server/src/test/java/org/sonar/ce/queue/CeQueueImplTest.java b/server/sonar-server/src/test/java/org/sonar/ce/queue/CeQueueImplTest.java
index d18f19edac8..e069e6c296b 100644
--- a/server/sonar-server/src/test/java/org/sonar/ce/queue/CeQueueImplTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/ce/queue/CeQueueImplTest.java
@@ -22,14 +22,11 @@ package org.sonar.ce.queue;
import com.google.common.base.Optional;
import java.util.List;
import javax.annotation.Nullable;
-import org.junit.Before;
import org.junit.Rule;
import org.junit.Test;
import org.junit.rules.ExpectedException;
import org.sonar.api.utils.System2;
import org.sonar.api.utils.internal.TestSystem2;
-import org.sonar.ce.monitoring.CEQueueStatus;
-import org.sonar.server.computation.monitoring.CEQueueStatusImpl;
import org.sonar.core.util.UuidFactory;
import org.sonar.core.util.UuidFactoryImpl;
import org.sonar.db.DbSession;
@@ -45,7 +42,6 @@ import static org.hamcrest.Matchers.startsWith;
import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.verify;
import static org.mockito.Mockito.verifyZeroInteractions;
-import static org.mockito.Mockito.when;
public class CeQueueImplTest {
@@ -59,14 +55,8 @@ public class CeQueueImplTest {
DbSession session = dbTester.getSession();
UuidFactory uuidFactory = UuidFactoryImpl.INSTANCE;
- CEQueueStatus queueStatus = new CEQueueStatusImpl();
CeQueueListener listener = mock(CeQueueListener.class);
- CeQueue underTest = new CeQueueImpl(dbTester.getDbClient(), uuidFactory, queueStatus, new CeQueueListener[] {listener});
-
- @Before
- public void setUp() throws Exception {
- queueStatus.initPendingCount(0);
- }
+ CeQueue underTest = new CeQueueImpl(dbTester.getDbClient(), uuidFactory, new CeQueueListener[] {listener});
@Test
public void submit_returns_task_populated_from_CeTaskSubmit_and_creates_CeQueue_row() {
@@ -78,17 +68,6 @@ public class CeQueueImplTest {
}
@Test
- public void submit_increments_receivedCount_of_QueueStatus() {
- underTest.submit(createTaskSubmit(CeTaskTypes.REPORT, "PROJECT_1", "rob"));
-
- assertThat(queueStatus.getReceivedCount()).isEqualTo(1L);
-
- underTest.submit(createTaskSubmit(CeTaskTypes.REPORT, "PROJECT_2", "rob"));
-
- assertThat(queueStatus.getReceivedCount()).isEqualTo(2L);
- }
-
- @Test
public void submit_populates_component_name_and_key_of_CeTask_if_component_exists() {
ComponentDto componentDto = insertComponent(newComponentDto("PROJECT_1"));
CeTaskSubmit taskSubmit = createTaskSubmit(CeTaskTypes.REPORT, componentDto.uuid(), null);
@@ -145,18 +124,6 @@ public class CeQueueImplTest {
}
@Test
- public void massSubmit_increments_receivedCount_of_QueueStatus() {
- underTest.massSubmit(asList(createTaskSubmit("type 1"), createTaskSubmit("type 2")));
-
- assertThat(queueStatus.getReceivedCount()).isEqualTo(2L);
-
- underTest.massSubmit(asList(createTaskSubmit("a"), createTaskSubmit("a"), createTaskSubmit("b")));
-
- assertThat(queueStatus.getReceivedCount()).isEqualTo(5L);
- }
-
-
- @Test
public void cancel_pending() throws Exception {
CeTask task = submit(CeTaskTypes.REPORT, "PROJECT_1");
@@ -259,12 +226,6 @@ public class CeQueueImplTest {
return submission.build();
}
- private CeTaskResult newTaskResult(Long snapshotId) {
- CeTaskResult taskResult = mock(CeTaskResult.class);
- when(taskResult.getSnapshotId()).thenReturn(snapshotId);
- return taskResult;
- }
-
private ComponentDto insertComponent(ComponentDto componentDto) {
dbTester.getDbClient().componentDao().insert(session, componentDto);
session.commit();
diff --git a/server/sonar-server/src/test/java/org/sonar/server/app/ProcessCommandWrapperImplTest.java b/server/sonar-server/src/test/java/org/sonar/server/app/ProcessCommandWrapperImplTest.java
index 8a394785681..e819ec37398 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/app/ProcessCommandWrapperImplTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/app/ProcessCommandWrapperImplTest.java
@@ -43,22 +43,22 @@ public class ProcessCommandWrapperImplTest {
private Settings settings = new Settings();
@Test
- public void requestSQRestart_throws_IAE_if_process_sharedDir_property_not_set() throws Exception {
+ public void requestSQRestart_throws_IAE_if_process_index_property_not_set() throws Exception {
ProcessCommandWrapperImpl processCommandWrapper = new ProcessCommandWrapperImpl(settings);
expectedException.expect(IllegalArgumentException.class);
- expectedException.expectMessage("Property process.sharedDir is not set");
+ expectedException.expectMessage("Property process.index is not set");
processCommandWrapper.requestSQRestart();
}
@Test
- public void requestSQRestart_throws_IAE_if_process_index_property_not_set() throws Exception {
- settings.setProperty(PROPERTY_SHARED_PATH, temp.newFolder().getAbsolutePath());
+ public void requestSQRestart_throws_IAE_if_process_shared_path_property_not_set() throws Exception {
+ settings.setProperty(PROPERTY_PROCESS_INDEX, 1);
ProcessCommandWrapperImpl processCommandWrapper = new ProcessCommandWrapperImpl(settings);
expectedException.expect(IllegalArgumentException.class);
- expectedException.expectMessage("Property process.index is not set");
+ expectedException.expectMessage("Property process.sharedDir is not set");
processCommandWrapper.requestSQRestart();
}
@@ -79,6 +79,7 @@ public class ProcessCommandWrapperImplTest {
@Test
public void notifyOperational_throws_IAE_if_process_sharedDir_property_not_set() throws Exception {
+ settings.setProperty(PROPERTY_PROCESS_INDEX, 1);
ProcessCommandWrapperImpl processCommandWrapper = new ProcessCommandWrapperImpl(settings);
expectedException.expect(IllegalArgumentException.class);
@@ -89,7 +90,6 @@ public class ProcessCommandWrapperImplTest {
@Test
public void notifyOperational_throws_IAE_if_process_index_property_not_set() throws Exception {
- settings.setProperty(PROPERTY_SHARED_PATH, temp.newFolder().getAbsolutePath());
ProcessCommandWrapperImpl processCommandWrapper = new ProcessCommandWrapperImpl(settings);
expectedException.expect(IllegalArgumentException.class);
diff --git a/server/sonar-server/src/test/java/org/sonar/server/computation/monitoring/CEQueueStatusImplTest.java b/server/sonar-server/src/test/java/org/sonar/server/computation/monitoring/CEQueueStatusImplTest.java
index b1d82c150b4..733383fc65c 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/computation/monitoring/CEQueueStatusImplTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/computation/monitoring/CEQueueStatusImplTest.java
@@ -23,23 +23,29 @@ import java.util.Random;
import org.junit.Rule;
import org.junit.Test;
import org.junit.rules.ExpectedException;
+import org.mockito.Mockito;
+import org.sonar.db.DbClient;
+import org.sonar.db.DbSession;
+import org.sonar.db.ce.CeQueueDto;
import static org.assertj.core.api.Assertions.assertThat;
+import static org.mockito.Matchers.any;
+import static org.mockito.Matchers.eq;
+import static org.mockito.Mockito.mock;
+import static org.mockito.Mockito.when;
public class CEQueueStatusImplTest {
private static final int SOME_RANDOM_MAX = 96535;
private static final int SOME_PROCESSING_TIME = 8723;
- private static final long INITIAL_PENDING_COUNT = 996L;
@Rule
public ExpectedException expectedException = ExpectedException.none();
- private CEQueueStatusImpl underTest = new CEQueueStatusImpl();
+ DbClient dbClient = mock(DbClient.class, Mockito.RETURNS_DEEP_STUBS);
+ private CEQueueStatusImpl underTest = new CEQueueStatusImpl(dbClient);
@Test
public void verify_just_created_instance_metrics() {
- assertThat(underTest.getReceivedCount()).isEqualTo(0);
- assertThat(underTest.getPendingCount()).isEqualTo(0);
assertThat(underTest.getInProgressCount()).isEqualTo(0);
assertThat(underTest.getErrorCount()).isEqualTo(0);
assertThat(underTest.getSuccessCount()).isEqualTo(0);
@@ -47,117 +53,9 @@ public class CEQueueStatusImplTest {
}
@Test
- public void initPendingCount_sets_value_of_pendingCount() {
- underTest.initPendingCount(10);
-
- assertThat(underTest.getPendingCount()).isEqualTo(10);
- }
-
- @Test
- public void initPendingCount_throws_ISE_if_called_twice() {
- expectedException.expect(IllegalStateException.class);
- expectedException.expectMessage("Method initPendingCount must be used before any other method and can not be called twice");
-
- underTest.initPendingCount(10);
- underTest.initPendingCount(10);
- }
-
- @Test
- public void addReceived_throws_ISE_if_called_before_initPendingCount() {
- expectedException.expect(IllegalStateException.class);
- expectedException.expectMessage("Method initPendingCount must be used before addReceived can be called");
-
- underTest.addReceived();
- }
-
- @Test
- public void addReceived_sets_received_to_1_and_pending_counts_to_initial_value_plus_1() {
- underTest.initPendingCount(INITIAL_PENDING_COUNT);
-
- underTest.addReceived();
-
- assertThat(underTest.getReceivedCount()).isEqualTo(1);
- assertThat(underTest.getPendingCount()).isEqualTo(INITIAL_PENDING_COUNT + 1);
- }
-
- @Test
- public void addReceived_any_number_of_call_adds_1_per_call() {
- underTest.initPendingCount(INITIAL_PENDING_COUNT);
-
- int calls = new Random().nextInt(SOME_RANDOM_MAX);
- for (int i = 0; i < calls; i++) {
- underTest.addReceived();
- }
-
- assertThat(underTest.getReceivedCount()).isEqualTo(calls);
- assertThat(underTest.getPendingCount()).isEqualTo(INITIAL_PENDING_COUNT + calls);
- }
-
- @Test
- public void addReceived_with_argument_throws_IAE_if_parameter_is_0() {
- underTest.initPendingCount(INITIAL_PENDING_COUNT);
-
- expectedException.expect(IllegalArgumentException.class);
- expectedException.expectMessage("numberOfReceived must be > 0");
-
- underTest.addReceived(0);
- }
-
- @Test
- public void addReceived_with_argument_throws_IAE_if_parameter_less_than_0() {
- underTest.initPendingCount(INITIAL_PENDING_COUNT);
-
- expectedException.expect(IllegalArgumentException.class);
- expectedException.expectMessage("numberOfReceived must be > 0");
-
- underTest.addReceived(-1 * (new Random().nextInt(SOME_RANDOM_MAX)));
- }
-
- @Test
- public void addReceived_with_argument_sets_received_to_value_and_pending_counts_to_initial_value_plus_value() {
- long value = 123;
- underTest.initPendingCount(INITIAL_PENDING_COUNT);
-
- underTest.addReceived(value);
-
- assertThat(underTest.getReceivedCount()).isEqualTo(value);
- assertThat(underTest.getPendingCount()).isEqualTo(INITIAL_PENDING_COUNT + value);
- }
-
- @Test
- public void addReceived_with_argument_any_number_of_call_adds_some_number_per_call() {
- underTest.initPendingCount(INITIAL_PENDING_COUNT);
-
- Random random = new Random();
- int calls = random.nextInt(SOME_RANDOM_MAX);
- long added = 0;
- for (int i = 0; i < calls; i++) {
- // adding 1 to random number because value can not be 0
- long numberOfReceived = random.nextInt(SOME_RANDOM_MAX) + 1;
- underTest.addReceived(numberOfReceived);
- added += numberOfReceived;
- }
-
- assertThat(underTest.getReceivedCount()).isEqualTo(added);
- assertThat(underTest.getPendingCount()).isEqualTo(INITIAL_PENDING_COUNT + added);
- }
-
- @Test
- public void addInProgress_throws_ISE_if_called_before_initPendingCount() {
- expectedException.expect(IllegalStateException.class);
- expectedException.expectMessage("Method initPendingCount must be used before addInProgress can be called");
-
+ public void addInProgress_increases_InProgress() {
underTest.addInProgress();
- }
-
- @Test
- public void addInProgress_increases_InProgress_and_decreases_Pending_by_1_without_check_on_Pending() {
- underTest.initPendingCount(INITIAL_PENDING_COUNT);
- underTest.addInProgress();
-
- assertThat(underTest.getReceivedCount()).isEqualTo(0);
- assertThat(underTest.getPendingCount()).isEqualTo(INITIAL_PENDING_COUNT - 1);
assertThat(underTest.getInProgressCount()).isEqualTo(1);
assertThat(underTest.getErrorCount()).isEqualTo(0);
assertThat(underTest.getSuccessCount()).isEqualTo(0);
@@ -166,15 +64,12 @@ public class CEQueueStatusImplTest {
@Test
public void addInProgress_any_number_of_call_change_by_1_per_call() {
- underTest.initPendingCount(INITIAL_PENDING_COUNT);
-
int calls = new Random().nextInt(SOME_RANDOM_MAX);
for (int i = 0; i < calls; i++) {
underTest.addInProgress();
}
assertThat(underTest.getInProgressCount()).isEqualTo(calls);
- assertThat(underTest.getPendingCount()).isEqualTo(INITIAL_PENDING_COUNT - calls);
assertThat(underTest.getProcessingTime()).isEqualTo(0);
}
@@ -190,8 +85,6 @@ public class CEQueueStatusImplTest {
public void addError_increases_Error_and_decreases_InProgress_by_1_without_check_on_InProgress() {
underTest.addError(SOME_PROCESSING_TIME);
- assertThat(underTest.getReceivedCount()).isEqualTo(0);
- assertThat(underTest.getPendingCount()).isEqualTo(0);
assertThat(underTest.getInProgressCount()).isEqualTo(-1);
assertThat(underTest.getErrorCount()).isEqualTo(1);
assertThat(underTest.getSuccessCount()).isEqualTo(0);
@@ -222,8 +115,6 @@ public class CEQueueStatusImplTest {
public void addSuccess_increases_Error_and_decreases_InProgress_by_1_without_check_on_InProgress() {
underTest.addSuccess(SOME_PROCESSING_TIME);
- assertThat(underTest.getReceivedCount()).isEqualTo(0);
- assertThat(underTest.getPendingCount()).isEqualTo(0);
assertThat(underTest.getInProgressCount()).isEqualTo(-1);
assertThat(underTest.getErrorCount()).isEqualTo(0);
assertThat(underTest.getSuccessCount()).isEqualTo(1);
@@ -241,4 +132,11 @@ public class CEQueueStatusImplTest {
assertThat(underTest.getInProgressCount()).isEqualTo(-calls);
assertThat(underTest.getProcessingTime()).isEqualTo(calls);
}
+
+ @Test
+ public void count_Pending_from_database() {
+ when(dbClient.ceQueueDao().countByStatus(any(DbSession.class), eq(CeQueueDto.Status.PENDING))).thenReturn(42);
+
+ assertThat(underTest.getPendingCount()).isEqualTo(42);
+ }
}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/computation/monitoring/ComputeEngineQueueMonitorTest.java b/server/sonar-server/src/test/java/org/sonar/server/computation/monitoring/CeTasksMBeanImplTest.java
index bca30441ecc..9b5bc7becc8 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/computation/monitoring/ComputeEngineQueueMonitorTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/computation/monitoring/CeTasksMBeanImplTest.java
@@ -19,17 +19,19 @@
*/
package org.sonar.server.computation.monitoring;
+import java.lang.management.ManagementFactory;
+import javax.annotation.CheckForNull;
+import javax.management.InstanceNotFoundException;
+import javax.management.ObjectInstance;
+import javax.management.ObjectName;
import org.junit.Test;
import org.sonar.ce.monitoring.CEQueueStatus;
+import org.sonar.process.jmx.CeTasksMBean;
import org.sonar.server.computation.configuration.CeConfiguration;
-import org.sonar.ce.queue.CeQueueImpl;
import static org.assertj.core.api.Assertions.assertThat;
-import static org.assertj.core.api.Assertions.entry;
-import static org.mockito.Mockito.mock;
-public class ComputeEngineQueueMonitorTest {
- private static final long RECEIVED_COUNT = 30;
+public class CeTasksMBeanImplTest {
private static final long PENDING_COUNT = 2;
private static final long IN_PROGRESS_COUNT = 5;
private static final long ERROR_COUNT = 10;
@@ -37,28 +39,21 @@ public class ComputeEngineQueueMonitorTest {
private static final long PROCESSING_TIME = 987;
private static final int WORKER_COUNT = 56;
- private ComputeEngineQueueMonitor underTest = new ComputeEngineQueueMonitor(new DumbCEQueueStatus(), mock(CeQueueImpl.class), new DumbCeConfiguration());
+ private CeTasksMBeanImpl underTest = new CeTasksMBeanImpl(new DumbCEQueueStatus(), new DumbCeConfiguration());
@Test
- public void name_is_ComputeEngine() {
- assertThat(underTest.name()).isEqualTo("ComputeEngine");
- }
+ public void register_and_unregister() throws Exception {
+ assertThat(getMBean()).isNull();
- @Test
- public void attributes_has_entry_for_each_get_method() {
- assertThat(underTest.attributes()).containsOnly(
- entry("Received", RECEIVED_COUNT),
- entry("Pending", PENDING_COUNT),
- entry("In progress", IN_PROGRESS_COUNT),
- entry("Successfully processed", SUCCESS_COUNT),
- entry("Processed with error", ERROR_COUNT),
- entry("Processing time", PROCESSING_TIME),
- entry("Worker count", WORKER_COUNT));
+ underTest.start();
+ assertThat(getMBean()).isNotNull();
+
+ underTest.stop();
+ assertThat(getMBean()).isNull();
}
@Test
public void get_methods_delegate_to_the_CEQueueStatus_instance() {
- assertThat(underTest.getReceivedCount()).isEqualTo(RECEIVED_COUNT);
assertThat(underTest.getPendingCount()).isEqualTo(PENDING_COUNT);
assertThat(underTest.getInProgressCount()).isEqualTo(IN_PROGRESS_COUNT);
assertThat(underTest.getErrorCount()).isEqualTo(ERROR_COUNT);
@@ -78,26 +73,6 @@ public class ComputeEngineQueueMonitorTest {
private static class DumbCEQueueStatus implements CEQueueStatus {
@Override
- public long addReceived() {
- return methodNotImplemented();
- }
-
- @Override
- public long addReceived(long numberOfReceived) {
- return methodNotImplemented();
- }
-
- @Override
- public long getReceivedCount() {
- return RECEIVED_COUNT;
- }
-
- @Override
- public long initPendingCount(long initialPendingCount) {
- return methodNotImplemented();
- }
-
- @Override
public long getPendingCount() {
return PENDING_COUNT;
}
@@ -153,4 +128,13 @@ public class ComputeEngineQueueMonitorTest {
throw new UnsupportedOperationException("getQueuePollingDelay is not implemented");
}
}
+
+ @CheckForNull
+ private ObjectInstance getMBean() throws Exception {
+ try {
+ return ManagementFactory.getPlatformMBeanServer().getObjectInstance(new ObjectName(CeTasksMBean.OBJECT_NAME));
+ } catch (InstanceNotFoundException e) {
+ return null;
+ }
+ }
}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/computation/queue/CeQueueInitializerTest.java b/server/sonar-server/src/test/java/org/sonar/server/computation/queue/CeQueueInitializerTest.java
index 41f42da6364..473ebc092ed 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/computation/queue/CeQueueInitializerTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/computation/queue/CeQueueInitializerTest.java
@@ -19,64 +19,31 @@
*/
package org.sonar.server.computation.queue;
-import java.io.File;
import java.io.IOException;
import org.junit.Rule;
import org.junit.Test;
-import org.junit.rules.TemporaryFolder;
import org.mockito.InOrder;
import org.mockito.Mockito;
import org.sonar.api.platform.Server;
import org.sonar.api.utils.System2;
import org.sonar.db.DbSession;
import org.sonar.db.DbTester;
-import org.sonar.db.ce.CeQueueDto;
-import org.sonar.db.ce.CeTaskTypes;
-import org.sonar.ce.monitoring.CEQueueStatus;
-import org.sonar.server.computation.monitoring.CEQueueStatusImpl;
-import org.sonar.ce.queue.report.ReportFiles;
import org.sonar.server.computation.taskprocessor.CeProcessingScheduler;
-import static org.assertj.core.api.Assertions.assertThat;
import static org.mockito.Matchers.any;
import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.reset;
import static org.mockito.Mockito.verifyZeroInteractions;
-import static org.mockito.Mockito.when;
public class CeQueueInitializerTest {
@Rule
public DbTester dbTester = DbTester.create(System2.INSTANCE);
- @Rule
- public TemporaryFolder tempFolder = new TemporaryFolder();
-
Server server = mock(Server.class);
- ReportFiles reportFiles = mock(ReportFiles.class, Mockito.RETURNS_DEEP_STUBS);
- CEQueueStatus queueStatus = new CEQueueStatusImpl();
CeQueueCleaner cleaner = mock(CeQueueCleaner.class);
CeProcessingScheduler scheduler = mock(CeProcessingScheduler.class);
- CeQueueInitializer underTest = new CeQueueInitializer(dbTester.getDbClient(), queueStatus, cleaner, scheduler);
-
- @Test
- public void init_jmx_counters() throws IOException {
- insertInQueue("TASK_1", CeQueueDto.Status.PENDING);
- insertInQueue("TASK_2", CeQueueDto.Status.PENDING);
- // this in-progress task is going to be moved to PENDING
- insertInQueue("TASK_3", CeQueueDto.Status.IN_PROGRESS);
-
- underTest.onServerStart(server);
-
- assertThat(queueStatus.getPendingCount()).isEqualTo(3);
- }
-
- @Test
- public void init_jmx_counters_when_queue_is_empty() {
- underTest.onServerStart(server);
-
- assertThat(queueStatus.getPendingCount()).isEqualTo(0);
- }
+ CeQueueInitializer underTest = new CeQueueInitializer(dbTester.getDbClient(), cleaner, scheduler);
@Test
public void clean_queue_then_start_scheduler_of_workers() throws IOException {
@@ -100,25 +67,4 @@ public class CeQueueInitializerTest {
verifyZeroInteractions(cleaner, scheduler);
}
-
- private void insertInQueue(String taskUuid, CeQueueDto.Status status) throws IOException {
- insertInQueue(taskUuid, status, true);
- }
-
- private CeQueueDto insertInQueue(String taskUuid, CeQueueDto.Status status, boolean createFile) throws IOException {
- CeQueueDto queueDto = new CeQueueDto();
- queueDto.setTaskType(CeTaskTypes.REPORT);
- queueDto.setComponentUuid("PROJECT_1");
- queueDto.setUuid(taskUuid);
- queueDto.setStatus(status);
- dbTester.getDbClient().ceQueueDao().insert(dbTester.getSession(), queueDto);
- dbTester.getSession().commit();
-
- File file = tempFolder.newFile();
- when(reportFiles.fileForUuid(taskUuid)).thenReturn(file);
- if (!createFile) {
- file.delete();
- }
- return queueDto;
- }
}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/computation/queue/InternalCeQueueImplTest.java b/server/sonar-server/src/test/java/org/sonar/server/computation/queue/InternalCeQueueImplTest.java
index 8220ef23961..aeedba846ce 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/computation/queue/InternalCeQueueImplTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/computation/queue/InternalCeQueueImplTest.java
@@ -22,14 +22,12 @@ package org.sonar.server.computation.queue;
import com.google.common.base.Optional;
import java.util.List;
import javax.annotation.Nullable;
-import org.junit.Before;
import org.junit.Rule;
import org.junit.Test;
import org.junit.rules.ExpectedException;
import org.sonar.api.utils.System2;
import org.sonar.api.utils.internal.TestSystem2;
import org.sonar.ce.monitoring.CEQueueStatus;
-import org.sonar.server.computation.monitoring.CEQueueStatusImpl;
import org.sonar.ce.queue.CeQueueListener;
import org.sonar.ce.queue.CeTask;
import org.sonar.ce.queue.CeTaskResult;
@@ -42,6 +40,7 @@ import org.sonar.db.ce.CeActivityDto;
import org.sonar.db.ce.CeQueueDto;
import org.sonar.db.ce.CeTaskTypes;
import org.sonar.db.component.ComponentDto;
+import org.sonar.server.computation.monitoring.CEQueueStatusImpl;
import static java.util.Arrays.asList;
import static org.assertj.core.api.Assertions.assertThat;
@@ -66,15 +65,10 @@ public class InternalCeQueueImplTest {
DbSession session = dbTester.getSession();
UuidFactory uuidFactory = UuidFactoryImpl.INSTANCE;
- CEQueueStatus queueStatus = new CEQueueStatusImpl();
+ CEQueueStatus queueStatus = new CEQueueStatusImpl(dbTester.getDbClient());
CeQueueListener listener = mock(CeQueueListener.class);
InternalCeQueue underTest = new InternalCeQueueImpl(system2, dbTester.getDbClient(), uuidFactory, queueStatus, new CeQueueListener[] {listener});
- @Before
- public void setUp() throws Exception {
- queueStatus.initPendingCount(0);
- }
-
@Test
public void submit_returns_task_populated_from_CeTaskSubmit_and_creates_CeQueue_row() {
CeTaskSubmit taskSubmit = createTaskSubmit(CeTaskTypes.REPORT, "PROJECT_1", "rob");
@@ -85,17 +79,6 @@ public class InternalCeQueueImplTest {
}
@Test
- public void submit_increments_receivedCount_of_QueueStatus() {
- underTest.submit(createTaskSubmit(CeTaskTypes.REPORT, "PROJECT_1", "rob"));
-
- assertThat(queueStatus.getReceivedCount()).isEqualTo(1L);
-
- underTest.submit(createTaskSubmit(CeTaskTypes.REPORT, "PROJECT_2", "rob"));
-
- assertThat(queueStatus.getReceivedCount()).isEqualTo(2L);
- }
-
- @Test
public void submit_populates_component_name_and_key_of_CeTask_if_component_exists() {
ComponentDto componentDto = insertComponent(newComponentDto("PROJECT_1"));
CeTaskSubmit taskSubmit = createTaskSubmit(CeTaskTypes.REPORT, componentDto.uuid(), null);
@@ -152,17 +135,6 @@ public class InternalCeQueueImplTest {
}
@Test
- public void massSubmit_increments_receivedCount_of_QueueStatus() {
- underTest.massSubmit(asList(createTaskSubmit("type 1"), createTaskSubmit("type 2")));
-
- assertThat(queueStatus.getReceivedCount()).isEqualTo(2L);
-
- underTest.massSubmit(asList(createTaskSubmit("a"), createTaskSubmit("a"), createTaskSubmit("b")));
-
- assertThat(queueStatus.getReceivedCount()).isEqualTo(5L);
- }
-
- @Test
public void test_remove() {
CeTask task = submit(CeTaskTypes.REPORT, "PROJECT_1");
Optional<CeTask> peek = underTest.peek();
diff --git a/server/sonar-server/src/test/java/org/sonar/server/es/EsServerHolder.java b/server/sonar-server/src/test/java/org/sonar/server/es/EsServerHolder.java
index 41a5c8b891c..be6fb6a1259 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/es/EsServerHolder.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/es/EsServerHolder.java
@@ -19,20 +19,20 @@
*/
package org.sonar.server.es;
+import java.io.File;
+import java.util.Properties;
import org.elasticsearch.action.admin.indices.delete.DeleteIndexResponse;
import org.elasticsearch.client.transport.TransportClient;
import org.elasticsearch.common.settings.ImmutableSettings;
import org.elasticsearch.common.transport.InetSocketTransportAddress;
import org.sonar.process.LoopbackAddress;
import org.sonar.process.NetworkUtils;
+import org.sonar.process.ProcessEntryPoint;
import org.sonar.process.ProcessProperties;
import org.sonar.process.Props;
import org.sonar.search.SearchServer;
import org.sonar.test.TestUtils;
-import java.io.File;
-import java.util.Properties;
-
public class EsServerHolder {
private static EsServerHolder HOLDER = null;
@@ -114,6 +114,7 @@ public class EsServerHolder {
properties.setProperty(ProcessProperties.SEARCH_PORT, String.valueOf(port));
properties.setProperty(ProcessProperties.SEARCH_HOST, hostName);
properties.setProperty(ProcessProperties.PATH_HOME, homeDir.getAbsolutePath());
+ properties.setProperty(ProcessEntryPoint.PROPERTY_SHARED_PATH, homeDir.getAbsolutePath());
SearchServer server = new SearchServer(new Props(properties));
server.start();
HOLDER = new EsServerHolder(server, clusterName, nodeName, port, hostName, homeDir);
diff --git a/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/BaseMonitorMBeanTest.java b/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/BaseMonitorMBeanTest.java
index 38adaa34662..514bb7fb7c6 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/BaseMonitorMBeanTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/BaseMonitorMBeanTest.java
@@ -19,12 +19,12 @@
*/
package org.sonar.server.platform.monitoring;
-import org.junit.Test;
-
+import java.lang.management.ManagementFactory;
import javax.annotation.CheckForNull;
import javax.management.InstanceNotFoundException;
import javax.management.ObjectInstance;
-import java.lang.management.ManagementFactory;
+import javax.management.ObjectName;
+import org.junit.Test;
import static org.assertj.core.api.Assertions.assertThat;
@@ -52,7 +52,7 @@ public class BaseMonitorMBeanTest {
@CheckForNull
private ObjectInstance getMBean() throws Exception {
try {
- return ManagementFactory.getPlatformMBeanServer().getObjectInstance(underTest.objectName());
+ return ManagementFactory.getPlatformMBeanServer().getObjectInstance(new ObjectName(underTest.objectName()));
} catch (InstanceNotFoundException e) {
return null;
}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/CeDatabaseMonitorTest.java b/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/CeDatabaseMonitorTest.java
new file mode 100644
index 00000000000..4c2ce9ed035
--- /dev/null
+++ b/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/CeDatabaseMonitorTest.java
@@ -0,0 +1,53 @@
+/*
+ * 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.server.platform.monitoring;
+
+import java.util.LinkedHashMap;
+import org.junit.Test;
+import org.mockito.Mockito;
+import org.sonar.process.ProcessId;
+import org.sonar.process.jmx.CeDatabaseMBean;
+import org.sonar.process.jmx.JmxConnectionFactory;
+
+import static org.assertj.core.api.Assertions.assertThat;
+import static org.mockito.Mockito.mock;
+import static org.mockito.Mockito.when;
+
+public class CeDatabaseMonitorTest {
+
+ JmxConnectionFactory jmxConnectionFactory = mock(JmxConnectionFactory.class, Mockito.RETURNS_DEEP_STUBS);
+ CeDatabaseMonitor underTest = new CeDatabaseMonitor(jmxConnectionFactory);
+
+ @Test
+ public void testName() {
+ assertThat(underTest.name()).isNotEmpty();
+ }
+
+ @Test
+ public void testAttributes() {
+ CeDatabaseMBean mbean = mock(CeDatabaseMBean.class, Mockito.RETURNS_DEFAULTS);
+
+ when(jmxConnectionFactory.create(ProcessId.COMPUTE_ENGINE).getMBean(CeDatabaseMBean.OBJECT_NAME, CeDatabaseMBean.class))
+ .thenReturn(mbean);
+ LinkedHashMap<String, Object> attributes = underTest.attributes();
+ assertThat(attributes).containsKeys("Pool Initial Size", "Pool Active Connections");
+ assertThat(attributes).hasSize(9);
+ }
+}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/CeStateMonitorTest.java b/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/CeStateMonitorTest.java
new file mode 100644
index 00000000000..5db51a693fa
--- /dev/null
+++ b/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/CeStateMonitorTest.java
@@ -0,0 +1,53 @@
+/*
+ * 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.server.platform.monitoring;
+
+import com.google.common.collect.ImmutableSortedMap;
+import java.util.LinkedHashMap;
+import org.assertj.core.data.MapEntry;
+import org.junit.Test;
+import org.mockito.Mockito;
+import org.sonar.process.ProcessId;
+import org.sonar.process.jmx.JmxConnectionFactory;
+
+import static org.assertj.core.api.Assertions.assertThat;
+import static org.mockito.Mockito.mock;
+import static org.mockito.Mockito.when;
+
+public class CeStateMonitorTest {
+
+ JmxConnectionFactory jmxConnectionFactory = mock(JmxConnectionFactory.class, Mockito.RETURNS_DEEP_STUBS);
+ CeStateMonitor underTest = new CeStateMonitor(jmxConnectionFactory);
+
+ @Test
+ public void testName() {
+ assertThat(underTest.name()).isNotEmpty();
+ }
+
+ @Test
+ public void testAttributes() {
+ when(jmxConnectionFactory.create(ProcessId.COMPUTE_ENGINE).getSystemState()).thenReturn(ImmutableSortedMap.<String, Object>of(
+ "foo", "foo_val", "bar", "bar_val"));
+ LinkedHashMap<String, Object> attributes = underTest.attributes();
+ assertThat(attributes).containsExactly(
+ MapEntry.entry("bar", "bar_val"),
+ MapEntry.entry("foo", "foo_val"));
+ }
+}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/CeTasksMonitorTest.java b/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/CeTasksMonitorTest.java
new file mode 100644
index 00000000000..cf1902601ae
--- /dev/null
+++ b/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/CeTasksMonitorTest.java
@@ -0,0 +1,53 @@
+/*
+ * 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.server.platform.monitoring;
+
+import java.util.LinkedHashMap;
+import org.junit.Test;
+import org.mockito.Mockito;
+import org.sonar.process.ProcessId;
+import org.sonar.process.jmx.CeTasksMBean;
+import org.sonar.process.jmx.JmxConnectionFactory;
+
+import static org.assertj.core.api.Assertions.assertThat;
+import static org.mockito.Mockito.mock;
+import static org.mockito.Mockito.when;
+
+public class CeTasksMonitorTest {
+
+ JmxConnectionFactory jmxConnectionFactory = mock(JmxConnectionFactory.class, Mockito.RETURNS_DEEP_STUBS);
+ CeTasksMonitor underTest = new CeTasksMonitor(jmxConnectionFactory);
+
+ @Test
+ public void testName() {
+ assertThat(underTest.name()).isNotEmpty();
+ }
+
+ @Test
+ public void testAttributes() {
+ CeTasksMBean mbean = mock(CeTasksMBean.class, Mockito.RETURNS_DEFAULTS);
+
+ when(jmxConnectionFactory.create(ProcessId.COMPUTE_ENGINE).getMBean(CeTasksMBean.OBJECT_NAME, CeTasksMBean.class))
+ .thenReturn(mbean);
+ LinkedHashMap<String, Object> attributes = underTest.attributes();
+ assertThat(attributes).containsKeys("Pending");
+ assertThat(attributes).hasSize(6);
+ }
+}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/DatabaseMonitorTest.java b/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/DatabaseMonitorTest.java
index 82023198119..1e7cb5eccc5 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/DatabaseMonitorTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/DatabaseMonitorTest.java
@@ -43,6 +43,11 @@ public class DatabaseMonitorTest {
}
@Test
+ public void name_is_not_empty() {
+ assertThat(underTest.name()).isNotEmpty();
+ }
+
+ @Test
public void db_info() {
LinkedHashMap<String, Object> attributes = underTest.attributes();
assertThat(attributes.get("Database")).isEqualTo("H2");
diff --git a/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/EsMonitorTest.java b/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/EsMonitorTest.java
index 76f4b1dd1a3..0716a6415a8 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/EsMonitorTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/EsMonitorTest.java
@@ -22,58 +22,69 @@ package org.sonar.server.platform.monitoring;
import java.util.LinkedHashMap;
import java.util.Map;
import org.elasticsearch.action.admin.cluster.health.ClusterHealthStatus;
+import org.junit.Before;
import org.junit.ClassRule;
import org.junit.Test;
+import org.mockito.Mockito;
import org.sonar.api.config.Settings;
+import org.sonar.process.ProcessId;
+import org.sonar.process.jmx.EsSettingsMBean;
+import org.sonar.process.jmx.JmxConnectionFactory;
import org.sonar.server.es.EsTester;
import org.sonar.server.es.NewIndex;
import org.sonar.server.issue.index.IssueIndexDefinition;
import static org.assertj.core.api.Assertions.assertThat;
+import static org.mockito.Mockito.mock;
+import static org.mockito.Mockito.when;
public class EsMonitorTest {
@ClassRule
public static EsTester esTester = new EsTester().addDefinitions(new IssueIndexDefinition(new Settings()));
+ JmxConnectionFactory jmxConnectionFactory = mock(JmxConnectionFactory.class, Mockito.RETURNS_DEEP_STUBS);
+ EsSettingsMBean settingsMBean = mock(EsSettingsMBean.class);
+ EsMonitor underTest = new EsMonitor(jmxConnectionFactory, esTester.client());
+
+ @Before
+ public void setUp() throws Exception {
+ when(jmxConnectionFactory.create(ProcessId.ELASTICSEARCH).getMBean(EsSettingsMBean.OBJECT_NAME, EsSettingsMBean.class)).thenReturn(settingsMBean);
+ }
+
@Test
public void name() {
- EsMonitor monitor = new EsMonitor(esTester.client());
- assertThat(monitor.name()).isEqualTo("ElasticSearch");
+ assertThat(underTest.name()).isEqualTo("Elasticsearch");
}
-
@Test
public void cluster_attributes() {
- EsMonitor monitor = new EsMonitor(esTester.client());
- LinkedHashMap<String, Object> attributes = monitor.attributes();
- assertThat(monitor.getState()).isEqualTo(ClusterHealthStatus.GREEN.name());
+ LinkedHashMap<String, Object> attributes = underTest.attributes();
+ assertThat(underTest.getState()).isEqualTo(ClusterHealthStatus.GREEN.name());
assertThat(attributes.get("State")).isEqualTo(ClusterHealthStatus.GREEN);
assertThat(attributes.get("Number of Nodes")).isEqualTo(1);
}
@Test
public void node_attributes() {
- EsMonitor monitor = new EsMonitor(esTester.client());
- LinkedHashMap<String, Object> attributes = monitor.attributes();
- Map nodesAttributes = (Map)attributes.get("Nodes");
+ LinkedHashMap<String, Object> attributes = underTest.attributes();
+ Map nodesAttributes = (Map) attributes.get("Nodes");
// one node
assertThat(nodesAttributes).hasSize(1);
- Map nodeAttributes = (Map)nodesAttributes.values().iterator().next();
+ Map nodeAttributes = (Map) nodesAttributes.values().iterator().next();
assertThat(nodeAttributes.get("Type")).isEqualTo("Master");
assertThat(nodeAttributes.get("Store Size")).isNotNull();
}
@Test
public void index_attributes() {
- EsMonitor monitor = new EsMonitor(esTester.client());
- LinkedHashMap<String, Object> attributes = monitor.attributes();
- Map indicesAttributes = (Map)attributes.get("Indices");
+ LinkedHashMap<String, Object> attributes = underTest.attributes();
+ Map indicesAttributes = (Map) attributes.get("Indices");
// one index "issues"
assertThat(indicesAttributes).hasSize(1);
- Map indexAttributes = (Map)indicesAttributes.values().iterator().next();
+ Map indexAttributes = (Map) indicesAttributes.values().iterator().next();
assertThat(indexAttributes.get("Docs")).isEqualTo(0L);
assertThat(indexAttributes.get("Shards")).isEqualTo(NewIndex.DEFAULT_NUMBER_OF_SHARDS);
assertThat(indexAttributes.get("Store Size")).isNotNull();
diff --git a/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/EsStateMonitorTest.java b/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/EsStateMonitorTest.java
new file mode 100644
index 00000000000..e06f5c9c111
--- /dev/null
+++ b/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/EsStateMonitorTest.java
@@ -0,0 +1,52 @@
+/*
+ * 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.server.platform.monitoring;
+
+import com.google.common.collect.ImmutableSortedMap;
+import java.util.LinkedHashMap;
+import org.assertj.core.data.MapEntry;
+import org.junit.Test;
+import org.mockito.Mockito;
+import org.sonar.process.ProcessId;
+import org.sonar.process.jmx.JmxConnectionFactory;
+
+import static org.assertj.core.api.Assertions.assertThat;
+import static org.mockito.Mockito.mock;
+import static org.mockito.Mockito.when;
+
+public class EsStateMonitorTest {
+ JmxConnectionFactory jmxConnectionFactory = mock(JmxConnectionFactory.class, Mockito.RETURNS_DEEP_STUBS);
+ EsStateMonitor underTest = new EsStateMonitor(jmxConnectionFactory);
+
+ @Test
+ public void testName() {
+ assertThat(underTest.name()).isNotEmpty();
+ }
+
+ @Test
+ public void testAttributes() {
+ when(jmxConnectionFactory.create(ProcessId.ELASTICSEARCH).getSystemState()).thenReturn(ImmutableSortedMap.<String, Object>of(
+ "foo", "foo_val", "bar", "bar_val"));
+ LinkedHashMap<String, Object> attributes = underTest.attributes();
+ assertThat(attributes).containsExactly(
+ MapEntry.entry("bar", "bar_val"),
+ MapEntry.entry("foo", "foo_val"));
+ }
+}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/JmxConnectionFactoryProviderTest.java b/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/JmxConnectionFactoryProviderTest.java
new file mode 100644
index 00000000000..5a461bc21c3
--- /dev/null
+++ b/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/JmxConnectionFactoryProviderTest.java
@@ -0,0 +1,56 @@
+/*
+ * 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.server.platform.monitoring;
+
+import org.junit.Rule;
+import org.junit.Test;
+import org.junit.rules.ExpectedException;
+import org.sonar.api.config.Settings;
+import org.sonar.process.ProcessEntryPoint;
+import org.sonar.process.jmx.JmxConnectionFactory;
+
+import static org.assertj.core.api.Assertions.assertThat;
+
+public class JmxConnectionFactoryProviderTest {
+
+ @Rule
+ public ExpectedException expectedException = ExpectedException.none();
+
+ Settings settings = new Settings();
+ JmxConnectorProvider underTest = new JmxConnectorProvider();
+
+ @Test
+ public void provide_JmxConnector() {
+ settings.setProperty(ProcessEntryPoint.PROPERTY_SHARED_PATH, "path/");
+ JmxConnectionFactory connector = underTest.provide(settings);
+
+ assertThat(connector).isNotNull();
+ // cache
+ assertThat(underTest.provide(settings)).isSameAs(connector);
+ }
+
+ @Test
+ public void throw_IAE_if_ipc_shared_path_is_not_set() {
+ expectedException.expect(IllegalArgumentException.class);
+ expectedException.expectMessage("Property process.sharedDir is not set");
+
+ underTest.provide(settings);
+ }
+}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/JvmPropertiesMonitorTest.java b/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/JvmPropsMonitorTest.java
index e1692321e98..48ce163e151 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/JvmPropertiesMonitorTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/JvmPropsMonitorTest.java
@@ -19,17 +19,22 @@
*/
package org.sonar.server.platform.monitoring;
-import org.junit.Test;
-
import java.util.LinkedHashMap;
+import org.junit.Test;
import static org.assertj.core.api.Assertions.assertThat;
-public class JvmPropertiesMonitorTest {
+public class JvmPropsMonitorTest {
+
+ JvmPropsMonitor underTest = new JvmPropsMonitor();
+
+ @Test
+ public void name_is_not_empty() {
+ assertThat(underTest.name()).isNotEmpty();
+ }
@Test
public void attributes() {
- JvmPropertiesMonitor underTest = new JvmPropertiesMonitor();
LinkedHashMap<String, Object> attributes = underTest.attributes();
assertThat(attributes).containsKeys("java.vm.vendor", "os.name");
diff --git a/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/SonarQubeMonitorTest.java b/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/SonarQubeMonitorTest.java
index 16889862087..9c9f91999d9 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/SonarQubeMonitorTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/platform/monitoring/SonarQubeMonitorTest.java
@@ -52,6 +52,11 @@ public class SonarQubeMonitorTest {
}
@Test
+ public void name_is_not_empty() {
+ assertThat(new SonarQubeMonitor(settings, new SecurityRealmFactory(settings), server, serverLogging).name()).isNotEmpty();
+ }
+
+ @Test
public void getServerId() {
when(server.getStartedAt()).thenReturn(DateUtils.parseDate("2015-01-01"));
SonarQubeMonitor monitor = new SonarQubeMonitor(settings, new SecurityRealmFactory(settings), server, serverLogging);
diff --git a/server/sonar-web/src/main/js/apps/system/main.js b/server/sonar-web/src/main/js/apps/system/main.js
index 0fbabea94bd..ea003bb7de6 100644
--- a/server/sonar-web/src/main/js/apps/system/main.js
+++ b/server/sonar-web/src/main/js/apps/system/main.js
@@ -24,8 +24,8 @@ import Section from './section';
import { translate } from '../../helpers/l10n';
import RestartModal from '../../components/RestartModal';
-const SECTIONS_ORDER = ['SonarQube', 'Database', 'Plugins', 'System', 'ElasticSearch', 'JvmProperties',
- 'ComputeEngine'];
+const SECTIONS_ORDER = ['SonarQube', 'Database', 'Plugins', 'System', 'Elasticsearch State', 'Elasticsearch',
+ 'Compute Engine Tasks', 'Compute Engine State', 'Compute Engine Database', 'JvmProperties'];
export default React.createClass({
componentDidMount() {
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 35e9f491694..de5db3c03e4 100644
--- a/sonar-application/src/main/java/org/sonar/application/App.java
+++ b/sonar-application/src/main/java/org/sonar/application/App.java
@@ -27,26 +27,24 @@ import java.util.Properties;
import org.apache.commons.io.FilenameUtils;
import org.apache.commons.lang.StringUtils;
import org.sonar.process.MinimumViableSystem;
+import org.sonar.process.ProcessId;
import org.sonar.process.ProcessProperties;
import org.sonar.process.Props;
import org.sonar.process.Stoppable;
import org.sonar.process.monitor.JavaCommand;
import org.sonar.process.monitor.Monitor;
+import static org.sonar.process.ProcessId.APP;
+
/**
* Entry-point of process that starts and monitors ElasticSearch, the Web Server and the Compute Engine.
*/
public class App implements Stoppable {
- public static final int APP_PROCESS_NUMBER = 0;
- public static final int ES_PROCESS_INDEX = 1;
- public static final int WEBSERVER_PROCESS_INDEX = 2;
- public static final int CESERVER_PROCESS_INDEX = 3;
-
private final Monitor monitor;
public App(AppFileSystem appFileSystem, boolean watchForHardStop) {
- this(Monitor.create(APP_PROCESS_NUMBER, appFileSystem, watchForHardStop));
+ this(Monitor.create(APP.getIpcIndex(), appFileSystem, watchForHardStop));
}
App(Monitor monitor) {
@@ -74,7 +72,7 @@ public class App implements Stoppable {
}
private static JavaCommand createESCommand(Props props, File homeDir) {
- JavaCommand elasticsearch = new JavaCommand("search", ES_PROCESS_INDEX);
+ JavaCommand elasticsearch = new JavaCommand(ProcessId.ELASTICSEARCH);
elasticsearch
.setWorkDir(homeDir)
.addJavaOptions("-Djava.awt.headless=true")
@@ -88,7 +86,7 @@ public class App implements Stoppable {
}
private static JavaCommand createWebServerCommand(Props props, File homeDir) {
- JavaCommand webServer = new JavaCommand("web", WEBSERVER_PROCESS_INDEX)
+ JavaCommand webServer = new JavaCommand(ProcessId.WEB_SERVER)
.setWorkDir(homeDir)
.addJavaOptions(ProcessProperties.WEB_ENFORCED_JVM_ARGS)
.addJavaOptions(props.nonNullValue(ProcessProperties.WEB_JAVA_OPTS))
@@ -107,7 +105,7 @@ public class App implements Stoppable {
}
private static JavaCommand createCeServerCommand(Props props, File homeDir) {
- JavaCommand webServer = new JavaCommand("ce", CESERVER_PROCESS_INDEX)
+ JavaCommand webServer = new JavaCommand(ProcessId.COMPUTE_ENGINE)
.setWorkDir(homeDir)
.addJavaOptions(ProcessProperties.CE_ENFORCED_JVM_ARGS)
.addJavaOptions(props.nonNullValue(ProcessProperties.CE_JAVA_OPTS))
diff --git a/sonar-application/src/test/java/org/sonar/application/AppTest.java b/sonar-application/src/test/java/org/sonar/application/AppTest.java
index 70f694c97dc..8150639c865 100644
--- a/sonar-application/src/test/java/org/sonar/application/AppTest.java
+++ b/sonar-application/src/test/java/org/sonar/application/AppTest.java
@@ -28,6 +28,7 @@ import org.junit.Rule;
import org.junit.Test;
import org.junit.rules.TemporaryFolder;
import org.mockito.ArgumentCaptor;
+import org.sonar.process.ProcessId;
import org.sonar.process.ProcessProperties;
import org.sonar.process.Props;
import org.sonar.process.monitor.JavaCommand;
@@ -52,7 +53,7 @@ public class AppTest {
}
@Test
- public void start_elasticsearch_and_tomcat_by_default() throws Exception {
+ public void start_all_processes_by_default() throws Exception {
Monitor monitor = mock(Monitor.class);
App app = new App(monitor);
Props props = initDefaultProps();
@@ -61,7 +62,7 @@ public class AppTest {
ArgumentCaptor<List<JavaCommand>> argument = newJavaCommandArgumentCaptor();
verify(monitor).start(argument.capture());
- assertThat(argument.getValue()).extracting("key").containsExactly("search", "web", "ce");
+ assertThat(argument.getValue()).extracting("processId").containsExactly(ProcessId.ELASTICSEARCH, ProcessId.WEB_SERVER, ProcessId.COMPUTE_ENGINE);
}
@Test
@@ -75,7 +76,7 @@ public class AppTest {
ArgumentCaptor<List<JavaCommand>> argument = newJavaCommandArgumentCaptor();
verify(monitor).start(argument.capture());
- assertThat(argument.getValue()).extracting("key").containsOnly("search");
+ assertThat(argument.getValue()).extracting("processId").containsOnly(ProcessId.ELASTICSEARCH);
}
@Test