aboutsummaryrefslogtreecommitdiffstats
path: root/tests/src
diff options
context:
space:
mode:
Diffstat (limited to 'tests/src')
-rw-r--r--tests/src/test/java/org/sonarqube/tests/Category5Suite.java5
-rw-r--r--tests/src/test/java/org/sonarqube/tests/LogsTailer.java162
-rw-r--r--tests/src/test/java/org/sonarqube/tests/ce/CeShutdownTest.java176
-rw-r--r--tests/src/test/java/org/sonarqube/tests/cluster/Cluster.java332
-rw-r--r--tests/src/test/java/org/sonarqube/tests/cluster/ClusterTest.java261
-rw-r--r--tests/src/test/java/org/sonarqube/tests/cluster/DataCenterEdition.java53
-rw-r--r--tests/src/test/java/org/sonarqube/tests/cluster/DataCenterEditionTest.java157
-rw-r--r--tests/src/test/java/org/sonarqube/tests/cluster/Node.java185
-rw-r--r--tests/src/test/java/org/sonarqube/tests/cluster/NodeConfig.java187
-rw-r--r--tests/src/test/java/org/sonarqube/tests/cluster/StartupLogWatcherImpl.java33
-rw-r--r--tests/src/test/java/org/sonarqube/tests/serverSystem/ClusterTest.java215
11 files changed, 1046 insertions, 720 deletions
diff --git a/tests/src/test/java/org/sonarqube/tests/Category5Suite.java b/tests/src/test/java/org/sonarqube/tests/Category5Suite.java
index 55556648e85..7bc3331d1d3 100644
--- a/tests/src/test/java/org/sonarqube/tests/Category5Suite.java
+++ b/tests/src/test/java/org/sonarqube/tests/Category5Suite.java
@@ -21,11 +21,12 @@ package org.sonarqube.tests;
import org.junit.runner.RunWith;
import org.junit.runners.Suite;
+import org.sonarqube.tests.ce.CeShutdownTest;
import org.sonarqube.tests.ce.CeWorkersTest;
import org.sonarqube.tests.qualityProfile.ActiveRuleEsResilienceTest;
import org.sonarqube.tests.qualityProfile.BuiltInQualityProfilesNotificationTest;
import org.sonarqube.tests.rule.RuleEsResilienceTest;
-import org.sonarqube.tests.serverSystem.ClusterTest;
+import org.sonarqube.tests.cluster.ClusterTest;
import org.sonarqube.tests.serverSystem.RestartTest;
import org.sonarqube.tests.serverSystem.ServerSystemRestartingOrchestrator;
import org.sonarqube.tests.settings.ElasticsearchSettingsTest;
@@ -63,7 +64,9 @@ import org.sonarqube.tests.user.UserEsResilienceTest;
TelemetryUploadTest.class,
TelemetryOptOutTest.class,
// ce
+ CeShutdownTest.class,
CeWorkersTest.class,
+
// elasticsearch
ElasticsearchSettingsTest.class
})
diff --git a/tests/src/test/java/org/sonarqube/tests/LogsTailer.java b/tests/src/test/java/org/sonarqube/tests/LogsTailer.java
new file mode 100644
index 00000000000..06f4d7d189a
--- /dev/null
+++ b/tests/src/test/java/org/sonarqube/tests/LogsTailer.java
@@ -0,0 +1,162 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2017 SonarSource SA
+ * mailto:info 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.sonarqube.tests;
+
+import java.io.File;
+import java.util.ArrayList;
+import java.util.Collections;
+import java.util.List;
+import java.util.Optional;
+import java.util.concurrent.CountDownLatch;
+import java.util.function.Consumer;
+import java.util.stream.Collectors;
+import org.apache.commons.io.input.Tailer;
+import org.apache.commons.io.input.TailerListenerAdapter;
+
+import static java.util.Objects.requireNonNull;
+
+/**
+ * Watch log files, usually server logs (see Orchestrator.getServer().get*Logs()).
+ * This class allows to not load the full content in memory.
+ */
+public class LogsTailer implements AutoCloseable {
+
+ private final List<Tailer> tailers;
+ private final LogConsumer logConsumer;
+
+ private LogsTailer(Builder builder) {
+ logConsumer = new LogConsumer(builder.consumers);
+ tailers = builder.files.stream()
+ .map(file -> Tailer.create(file, logConsumer, 500))
+ .collect(Collectors.toList());
+ }
+
+ public Watch watch(String text) {
+ return new Watch(text);
+ }
+
+ @Override
+ public void close() throws Exception {
+ for (Tailer tailer : tailers) {
+ tailer.stop();
+ }
+ }
+
+ public static Builder builder() {
+ return new Builder();
+ }
+
+ public static class Builder {
+ private final List<File> files = new ArrayList<>();
+ private final List<Consumer<String>> consumers = new ArrayList<>();
+
+ public Builder addFile(File file) {
+ this.files.add(file);
+ return this;
+ }
+
+ public Builder addFiles(File file, File... otherFiles) {
+ this.files.add(file);
+ Collections.addAll(this.files, otherFiles);
+ return this;
+ }
+
+ public Builder addConsumer(Consumer<String> consumer) {
+ this.consumers.add(consumer);
+ return this;
+ }
+
+ public Builder doOnFind(String text, Runnable runnable) {
+ return addConsumer(log -> {
+ if (log.contains(text)) {
+ runnable.run();
+ }
+ });
+ }
+
+ public LogsTailer build() {
+ return new LogsTailer(this);
+ }
+ }
+
+ private static class LogConsumer extends TailerListenerAdapter {
+ private final List<Consumer<String>> consumers = Collections.synchronizedList(new ArrayList<>());
+
+ private LogConsumer(List<Consumer<String>> consumers) {
+ this.consumers.addAll(consumers);
+ }
+
+ @Override
+ public void handle(String line) {
+ synchronized (consumers) {
+ for (Consumer<String> consumer : consumers) {
+ try {
+ consumer.accept(line);
+ } catch (Exception e) {
+ // do not prevent other consumers to handle the log
+ e.printStackTrace();
+ }
+ }
+ }
+ }
+
+ public void add(Consumer<String> consumer) {
+ this.consumers.add(consumer);
+ }
+
+ public void remove(Consumer<String> consumer) {
+ this.consumers.remove(consumer);
+ }
+ }
+
+ public class Watch implements AutoCloseable {
+ private final String expectedText;
+ private final CountDownLatch foundSignal = new CountDownLatch(1);
+ private String log = null;
+ private final Consumer<String> consumer;
+
+ private Watch(String expectedText) {
+ this.expectedText = requireNonNull(expectedText);
+ this.consumer = log -> {
+ if (log.contains(this.expectedText)) {
+ this.log = log;
+ foundSignal.countDown();
+ }
+ };
+ logConsumer.add(consumer);
+ }
+
+ /**
+ * Blocks until the expected log appears in watched files.
+ */
+ public void waitForLog() throws InterruptedException {
+ foundSignal.await();
+ }
+
+ public Optional<String> getLog() {
+ return Optional.ofNullable(log);
+ }
+
+ @Override
+ public void close() {
+ logConsumer.remove(consumer);
+ }
+ }
+}
diff --git a/tests/src/test/java/org/sonarqube/tests/ce/CeShutdownTest.java b/tests/src/test/java/org/sonarqube/tests/ce/CeShutdownTest.java
new file mode 100644
index 00000000000..29c237ca7ca
--- /dev/null
+++ b/tests/src/test/java/org/sonarqube/tests/ce/CeShutdownTest.java
@@ -0,0 +1,176 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2017 SonarSource SA
+ * mailto:info 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.sonarqube.tests.ce;
+
+import com.sonar.orchestrator.Orchestrator;
+import com.sonar.orchestrator.build.SonarScanner;
+import java.io.File;
+import java.io.IOException;
+import org.apache.commons.io.FileUtils;
+import org.junit.Ignore;
+import org.junit.Rule;
+import org.junit.Test;
+import org.junit.rules.DisableOnDebug;
+import org.junit.rules.TemporaryFolder;
+import org.junit.rules.TestRule;
+import org.junit.rules.Timeout;
+import org.sonarqube.tests.LogsTailer;
+import org.sonarqube.ws.client.WsClient;
+import org.sonarqube.ws.client.ce.ActivityStatusWsRequest;
+import util.ItUtils;
+
+import static com.google.common.base.Preconditions.checkState;
+import static org.assertj.core.api.Assertions.assertThat;
+
+public class CeShutdownTest {
+
+ @Rule
+ public TestRule safeguard = new DisableOnDebug(Timeout.seconds(600));
+ @Rule
+ public TemporaryFolder temp = new TemporaryFolder();
+
+ @Test
+ public void stopping_CE_waits_for_in_progress_task_to_be_finished() throws Exception {
+ try (ComputeEngine ce = new ComputeEngine()) {
+
+ try (LogsTailer.Watch watch = ce.logs().watch("CE analysis is paused")) {
+ ce.triggerTask();
+ watch.waitForLog();
+ assertThat(ce.countInProgressTasks()).isEqualTo(1);
+ }
+
+ // stop does not kill in-progress workers. It waits
+ // for them (at least a few dozens of seconds)
+ try (LogsTailer.Watch watch = ce.logs().watch("Waiting for workers to finish in-progress tasks")) {
+ ce.triggerStop();
+ watch.waitForLog();
+ assertThat(ce.countInProgressTasks()).isEqualTo(1);
+ }
+
+ // resume the in-progress task, so that it can
+ // finish successfully
+ try (LogsTailer.Watch watch = ce.logs().watch("Process [ce] is stopped")) {
+ ce.resumeTask();
+ watch.waitForLog();
+ assertThat(ce.isTaskFinished()).isTrue();
+ assertThat(ce.hasErrorLogs()).isFalse();
+ }
+ }
+ }
+
+ @Test
+ @Ignore("TODO make the graceful stop timeout configurable. 40 seconds is too long for a test.")
+ public void stopping_CE_kills_in_progress_tasks_if_too_long_to_gracefully_stop() throws Exception {
+ try (ComputeEngine ce = new ComputeEngine()) {
+
+ try (LogsTailer.Watch watch = ce.logs().watch("CE analysis is paused")) {
+ ce.triggerTask();
+ watch.waitForLog();
+ assertThat(ce.countInProgressTasks()).isEqualTo(1);
+ }
+
+ // stop does not kill in-progress workers. It waits
+ // for them (at least a few dozens of seconds)
+ try (LogsTailer.Watch watch = ce.logs().watch("Waiting for workers to finish in-progress tasks")) {
+ ce.triggerStop();
+ watch.waitForLog();
+ assertThat(ce.countInProgressTasks()).isEqualTo(1);
+ }
+
+ // resume the in-progress task, so that it can
+ // finish successfully
+ try (LogsTailer.Watch watch = ce.logs().watch("Process [ce] is stopped")) {
+ watch.waitForLog();
+ assertThat(ce.isTaskFinished()).isTrue();
+ assertThat(ce.hasErrorLogs()).isTrue();
+ }
+ }
+ }
+
+ private class ComputeEngine implements AutoCloseable {
+ private final Orchestrator orchestrator;
+ private final File pauseFile;
+ private final WsClient adminWsClient;
+ private Thread stopper;
+ private final LogsTailer logsTailer;
+
+ ComputeEngine() throws Exception {
+ pauseFile = temp.newFile();
+ FileUtils.touch(pauseFile);
+
+ orchestrator = Orchestrator.builderEnv()
+ .setServerProperty("sonar.ce.pauseTask.path", pauseFile.getAbsolutePath())
+ .addPlugin(ItUtils.xooPlugin())
+ .addPlugin(ItUtils.pluginArtifact("server-plugin"))
+ .build();
+ orchestrator.start();
+ adminWsClient = ItUtils.newAdminWsClient(orchestrator);
+ logsTailer = LogsTailer.builder()
+ .addFile(orchestrator.getServer().getCeLogs())
+ .addFile(orchestrator.getServer().getAppLogs())
+ .build();
+ }
+
+ LogsTailer logs() {
+ return logsTailer;
+ }
+
+ void triggerTask() throws InterruptedException {
+ orchestrator.executeBuild(SonarScanner.create(new File("projects/shared/xoo-sample"), "sonar.projectKey", "foo"), false);
+ }
+
+ void resumeTask() throws Exception {
+ FileUtils.forceDelete(pauseFile);
+ }
+
+ int countInProgressTasks() {
+ return adminWsClient.ce().activityStatus(ActivityStatusWsRequest.newBuilder().build()).getInProgress();
+ }
+
+ boolean isTaskFinished() throws Exception {
+ String ceLogs = FileUtils.readFileToString(orchestrator.getServer().getCeLogs());
+ return ceLogs.contains("Executed task | project=foo | type=REPORT");
+ }
+
+ boolean hasErrorLogs() throws IOException {
+ String ceLogs = FileUtils.readFileToString(orchestrator.getServer().getCeLogs());
+ return ceLogs.contains(" ERROR ");
+ }
+
+ /**
+ * non-blocking stop
+ */
+ void triggerStop() {
+ checkState(stopper == null);
+ stopper = new Thread(orchestrator::stop);
+ stopper.start();
+ }
+
+ @Override
+ public void close() throws Exception {
+ if (stopper != null) {
+ stopper.interrupt();
+ }
+ if (orchestrator != null) {
+ orchestrator.stop();
+ }
+ }
+ }
+}
diff --git a/tests/src/test/java/org/sonarqube/tests/cluster/Cluster.java b/tests/src/test/java/org/sonarqube/tests/cluster/Cluster.java
index 6c8416cbc8d..9340ce53d8a 100644
--- a/tests/src/test/java/org/sonarqube/tests/cluster/Cluster.java
+++ b/tests/src/test/java/org/sonarqube/tests/cluster/Cluster.java
@@ -17,301 +17,111 @@
* 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.sonarqube.tests.cluster;
-import com.google.common.net.HostAndPort;
import com.sonar.orchestrator.Orchestrator;
import com.sonar.orchestrator.OrchestratorBuilder;
-import com.sonar.orchestrator.util.NetworkUtils;
-import java.net.Inet4Address;
-import java.net.InetAddress;
-import java.net.NetworkInterface;
-import java.net.SocketException;
import java.util.ArrayList;
-import java.util.Arrays;
-import java.util.Collections;
-import java.util.EnumSet;
-import java.util.Enumeration;
import java.util.List;
-import java.util.Properties;
-import java.util.concurrent.ExecutionException;
-import java.util.concurrent.ForkJoinPool;
import java.util.function.Consumer;
-import java.util.function.Predicate;
-import java.util.stream.Collectors;
-import javax.annotation.CheckForNull;
-
-import static org.sonarqube.tests.cluster.Cluster.NodeType.APPLICATION;
-import static org.sonarqube.tests.cluster.Cluster.NodeType.SEARCH;
-
-public class Cluster {
-
- protected static final String CLUSTER_ENABLED = "sonar.cluster.enabled";
- protected static final String CLUSTER_NODE_TYPE = "sonar.cluster.node.type";
- protected static final String CLUSTER_SEARCH_HOSTS = "sonar.cluster.search.hosts";
- protected static final String CLUSTER_HOSTS = "sonar.cluster.hosts";
- protected static final String CLUSTER_NODE_PORT = "sonar.cluster.node.port";
- protected static final String CLUSTER_NODE_HOST = "sonar.cluster.node.host";
- protected static final String CLUSTER_NAME = "sonar.cluster.name";
-
- protected static final String SEARCH_HOST = "sonar.search.host";
- protected static final String SEARCH_PORT = "sonar.search.port";
- protected static final String SEARCH_JAVA_OPTS = "sonar.search.javaOpts";
-
- protected static final String WEB_JAVA_OPTS = "sonar.web.javaOpts";
- protected static final String WEB_PORT = "sonar.web.port";
-
- protected static final String CE_JAVA_OPTS = "sonar.ce.javaOpts";
+import java.util.stream.Stream;
+import javax.annotation.Nullable;
+import org.slf4j.LoggerFactory;
- public enum NodeType {
- SEARCH("search"), APPLICATION("application");
+import static java.util.stream.Collectors.joining;
- final String value;
+class Cluster implements AutoCloseable {
- NodeType(String value) {
- this.value = value;
- }
+ @Nullable
+ private final String clusterName;
- public String getValue() {
- return value;
- }
+ private final List<Node> nodes = new ArrayList<>();
- public static final EnumSet<NodeType> ALL = EnumSet.allOf(NodeType.class);
+ Cluster(@Nullable String name) {
+ this.clusterName = name;
}
- private final List<Node> nodes;
- private final ForkJoinPool forkJoinPool = new ForkJoinPool(5);
-
- private Cluster(List<Node> nodes) {
- this.nodes = nodes;
- assignPorts();
- completeNodesConfiguration();
- buildOrchestrators();
+ Node startNode(NodeConfig config, Consumer<OrchestratorBuilder> consumer) {
+ Node node = addNode(config, consumer);
+ node.start();
+ return node;
}
- public void start() throws ExecutionException, InterruptedException {
- forkJoinPool.submit(
- () -> nodes.parallelStream().forEach(
- node -> node.getOrchestrator().start()
- )
- ).get();
- }
+ Node addNode(NodeConfig config, Consumer<OrchestratorBuilder> consumer) {
+ OrchestratorBuilder builder = newOrchestratorBuilder(config);
- public void stop() throws ExecutionException, InterruptedException {
- forkJoinPool.submit(
- () -> nodes.parallelStream().forEach(
- node -> node.getOrchestrator().stop()
- )
- ).get();
- }
-
- public void stopAll(Predicate<Node> predicate) throws ExecutionException, InterruptedException {
- forkJoinPool.submit(
- () -> nodes.parallelStream()
- .filter(predicate)
- .forEach(n -> n.getOrchestrator().stop())
- ).get();
- }
-
- public void startAll(Predicate<Node> predicate) throws ExecutionException, InterruptedException {
- forkJoinPool.submit(
- () -> nodes.parallelStream()
- .filter(predicate)
- .forEach(n -> n.getOrchestrator().start())
- ).get();
- }
-
- public List<Node> getNodes() {
- return Collections.unmodifiableList(nodes);
- }
-
- private void assignPorts() {
- nodes.stream().forEach(
- node -> {
- node.setHzPort(NetworkUtils.getNextAvailablePort(getNonloopbackIPv4Address()));
- if (node.getType() == SEARCH) {
- node.setEsPort(NetworkUtils.getNextAvailablePort(getNonloopbackIPv4Address()));
- } else if (node.getType() == APPLICATION) {
- node.setWebPort(NetworkUtils.getNextAvailablePort(getNonloopbackIPv4Address()));
- }
- }
- );
- }
-
- public static InetAddress getNonloopbackIPv4Address() {
- try {
- Enumeration<NetworkInterface> nets = NetworkInterface.getNetworkInterfaces();
- for (NetworkInterface networkInterface : Collections.list(nets)) {
- if (!networkInterface.isLoopback() && networkInterface.isUp() && !isBlackListed(networkInterface)) {
- Enumeration<InetAddress> inetAddresses = networkInterface.getInetAddresses();
- while (inetAddresses.hasMoreElements()) {
- InetAddress inetAddress = inetAddresses.nextElement();
- if (inetAddress instanceof Inet4Address) {
- return inetAddress;
- }
- }
- }
- }
- } catch (SocketException se) {
- throw new RuntimeException("Cannot find a non loopback card required for tests", se);
+ switch (config.getType()) {
+ case SEARCH:
+ builder
+ .setServerProperty("sonar.cluster.node.type", "search")
+ .setServerProperty("sonar.search.host", config.getAddress().getHostAddress())
+ .setServerProperty("sonar.search.port", "" + config.getSearchPort().get())
+ .setServerProperty("sonar.search.javaOpts", "-Xmx64m -Xms64m -XX:+HeapDumpOnOutOfMemoryError");
+ break;
+ case APPLICATION:
+ builder
+ .setServerProperty("sonar.cluster.node.type", "application")
+ .setServerProperty("sonar.web.host", config.getAddress().getHostAddress())
+ .setServerProperty("sonar.web.port", "" + config.getWebPort().get())
+ .setServerProperty("sonar.web.javaOpts", "-Xmx128m -Xms16m -XX:+HeapDumpOnOutOfMemoryError")
+ .setServerProperty("sonar.auth.jwtBase64Hs256Secret", "HrPSavOYLNNrwTY+SOqpChr7OwvbR/zbDLdVXRN0+Eg=")
+ .setServerProperty("sonar.ce.javaOpts", "-Xmx32m -Xms16m -XX:+HeapDumpOnOutOfMemoryError");
+ break;
}
- throw new RuntimeException("Cannot find a non loopback card required for tests");
+ consumer.accept(builder);
+ Orchestrator orchestrator = builder.build();
+ Node node = new Node(config, orchestrator);
+ nodes.add(node);
+ return node;
}
- private static boolean isBlackListed(NetworkInterface networkInterface) {
- return networkInterface.getName().startsWith("docker") ||
- networkInterface.getName().startsWith("vboxnet");
+ Stream<Node> getNodes() {
+ return nodes.stream();
}
- private void completeNodesConfiguration() {
- String inet = getNonloopbackIPv4Address().getHostAddress();
- String clusterHosts = nodes.stream()
- .map(node -> HostAndPort.fromParts(inet, node.getHzPort()).toString())
- .collect(Collectors.joining(","));
- String elasticsearchHosts = nodes.stream()
- .filter(node -> node.getType() == SEARCH)
- .map(node -> HostAndPort.fromParts(inet, node.getEsPort()).toString())
- .collect(Collectors.joining(","));
-
- nodes.forEach(
- node -> {
- node.addProperty(CLUSTER_NODE_HOST, inet);
- node.addProperty(CLUSTER_HOSTS, clusterHosts);
- node.addProperty(CLUSTER_NODE_PORT, Integer.toString(node.getHzPort() == null ? -1 : node.getHzPort()));
- node.addProperty(CLUSTER_SEARCH_HOSTS, elasticsearchHosts);
- node.addProperty(SEARCH_PORT, Integer.toString(node.getEsPort() == null ? -1 : node.getEsPort()));
- node.addProperty(SEARCH_HOST, inet);
- node.addProperty(WEB_PORT, Integer.toString(node.getWebPort() == null ? -1 : node.getWebPort()));
- node.addProperty(CLUSTER_NODE_TYPE, node.getType().getValue());
- }
- );
+ Stream<Node> getAppNodes() {
+ return nodes.stream().filter(n -> n.getConfig().getType() == NodeConfig.NodeType.APPLICATION);
}
- private void buildOrchestrators() {
- nodes.stream().limit(1).forEach(
- node -> buildOrchestrator(node, false)
- );
- nodes.stream().skip(1).forEach(
- node -> buildOrchestrator(node, true)
- );
+ Stream<Node> getSearchNodes() {
+ return nodes.stream().filter(n -> n.getConfig().getType() == NodeConfig.NodeType.SEARCH);
}
- private void buildOrchestrator(Node node, boolean keepDatabase) {
- OrchestratorBuilder builder = Orchestrator.builderEnv()
- .setOrchestratorProperty("orchestrator.keepDatabase", Boolean.toString(keepDatabase))
- .setStartupLogWatcher(new StartupLogWatcherImpl());
-
- node.getProperties().entrySet().stream().forEach(
- e -> builder.setServerProperty((String) e.getKey(), (String) e.getValue())
- );
-
- node.setOrchestrator(builder.build());
+ Node getAppNode(int index) {
+ return getAppNodes().skip(index).findFirst().orElseThrow(IllegalArgumentException::new);
}
- public static Builder builder() {
- return new Builder();
+ Node getSearchNode(int index) {
+ return getSearchNodes().skip(index).findFirst().orElseThrow(IllegalArgumentException::new);
}
- public static class Builder {
- private final List<Node> nodes = new ArrayList<>();
-
- public Cluster build() {
- return new Cluster(nodes);
- }
-
- public Builder addNode(NodeType type, Consumer<Node>... consumers) {
- Node node = new Node(type);
- Arrays.stream(consumers).forEach(c -> c.accept(node));
- nodes.add(node);
- return this;
+ @Override
+ public void close() throws Exception {
+ // nodes are stopped in order of creation
+ for (Node node : nodes) {
+ try {
+ node.stop();
+ } catch (Exception e) {
+ LoggerFactory.getLogger(getClass()).error("Fail to stop node", e);
+ }
}
}
- /**
- * A cluster node
- */
- public static class Node {
- private final NodeType type;
- private Integer webPort;
- private Integer esPort;
- private Integer hzPort;
- private Orchestrator orchestrator = null;
- private Properties properties = new Properties();
-
- public Node(NodeType type) {
- this.type = type;
-
- // Default properties
- properties.setProperty(CLUSTER_ENABLED, "true");
- properties.setProperty(CLUSTER_NAME, "sonarqube");
- properties.setProperty(CE_JAVA_OPTS, "-Xmx256m");
- properties.setProperty(WEB_JAVA_OPTS, "-Xmx256m");
- properties.setProperty(SEARCH_JAVA_OPTS, "-Xmx256m -Xms256m " +
- "-XX:+UseConcMarkSweepGC " +
- "-XX:CMSInitiatingOccupancyFraction=75 " +
- "-XX:+UseCMSInitiatingOccupancyOnly " +
- "-XX:+AlwaysPreTouch " +
- "-server " +
- "-Xss1m " +
- "-Djava.awt.headless=true " +
- "-Dfile.encoding=UTF-8 " +
- "-Djna.nosys=true " +
- "-Djdk.io.permissionsUseCanonicalPath=true " +
- "-Dio.netty.noUnsafe=true " +
- "-Dio.netty.noKeySetOptimization=true " +
- "-Dio.netty.recycler.maxCapacityPerThread=0 " +
- "-Dlog4j.shutdownHookEnabled=false " +
- "-Dlog4j2.disable.jmx=true " +
- "-Dlog4j.skipJansi=true " +
- "-XX:+HeapDumpOnOutOfMemoryError");
- }
-
- public Properties getProperties() {
- return properties;
- }
-
- public Orchestrator getOrchestrator() {
- return orchestrator;
- }
-
- private void setOrchestrator(Orchestrator orchestrator) {
- this.orchestrator = orchestrator;
- }
-
- public NodeType getType() {
- return type;
+ private OrchestratorBuilder newOrchestratorBuilder(NodeConfig node) {
+ OrchestratorBuilder builder = Orchestrator.builderEnv();
+ builder.setOrchestratorProperty("orchestrator.keepDatabase", "true");
+ builder.setServerProperty("sonar.cluster.enabled", "true");
+ builder.setServerProperty("sonar.cluster.node.host", node.getAddress().getHostAddress());
+ builder.setServerProperty("sonar.cluster.node.port", "" + node.getHzPort());
+ builder.setServerProperty("sonar.cluster.hosts", node.getConnectedNodes().stream().map(NodeConfig::getHzHost).collect(joining(",")));
+ builder.setServerProperty("sonar.cluster.search.hosts", node.getSearchNodes().stream().map(NodeConfig::getSearchHost).collect(joining(",")));
+ if (clusterName != null) {
+ builder.setServerProperty("sonar.cluster.name", clusterName);
}
-
- @CheckForNull
- public Integer getWebPort() {
- return webPort;
- }
-
- @CheckForNull
- public Integer getEsPort() {
- return esPort;
- }
-
- @CheckForNull
- public Integer getHzPort() {
- return hzPort;
- }
-
- private void setWebPort(Integer webPort) {
- this.webPort = webPort;
- }
-
- private void setEsPort(Integer esPort) {
- this.esPort = esPort;
- }
-
- private void setHzPort(Integer hzPort) {
- this.hzPort = hzPort;
- }
-
- private void addProperty(String key, String value) {
- properties.setProperty(key, value);
+ if (node.getName().isPresent()) {
+ builder.setServerProperty("sonar.cluster.node.name", node.getName().get());
}
+ builder.setStartupLogWatcher(logLine -> true);
+ return builder;
}
}
diff --git a/tests/src/test/java/org/sonarqube/tests/cluster/ClusterTest.java b/tests/src/test/java/org/sonarqube/tests/cluster/ClusterTest.java
new file mode 100644
index 00000000000..0cf07e3fda5
--- /dev/null
+++ b/tests/src/test/java/org/sonarqube/tests/cluster/ClusterTest.java
@@ -0,0 +1,261 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2017 SonarSource SA
+ * mailto:info 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.sonarqube.tests.cluster;
+
+import com.sonar.orchestrator.Orchestrator;
+import com.sonar.orchestrator.OrchestratorBuilder;
+import com.sonar.orchestrator.db.DefaultDatabase;
+import java.util.ArrayList;
+import java.util.List;
+import java.util.function.BinaryOperator;
+import java.util.function.Consumer;
+import java.util.stream.IntStream;
+import org.junit.BeforeClass;
+import org.junit.Rule;
+import org.junit.Test;
+import org.junit.rules.DisableOnDebug;
+import org.junit.rules.ExpectedException;
+import org.junit.rules.TestRule;
+import org.junit.rules.Timeout;
+
+import static com.google.common.base.Preconditions.checkState;
+import static org.assertj.core.api.Assertions.assertThat;
+import static org.junit.Assert.fail;
+import static org.sonarqube.tests.cluster.NodeConfig.newApplicationConfig;
+import static org.sonarqube.tests.cluster.NodeConfig.newSearchConfig;
+
+public class ClusterTest {
+
+ @Rule
+ public TestRule safeguard = new DisableOnDebug(Timeout.seconds(300));
+
+ @Rule
+ public ExpectedException expectedException = ExpectedException.none();
+
+ @BeforeClass
+ public static void initDbSchema() throws Exception {
+ Orchestrator orchestrator = Orchestrator.builderEnv()
+ // enforce (re-)creation of database schema
+ .setOrchestratorProperty("orchestrator.keepDatabase", "false")
+ .build();
+ DefaultDatabase db = new DefaultDatabase(orchestrator.getConfiguration());
+ checkState(!db.getClient().getDialect().equals("h2"), "H2 is not supported in cluster mode");
+ db.start();
+ db.stop();
+ }
+
+ @Test
+ public void test_high_availability_topology() throws Exception {
+ try (Cluster cluster = newCluster(3, 2)) {
+ cluster.getNodes().forEach(Node::start);
+ cluster.getAppNodes().forEach(Node::waitForStatusUp);
+
+ // TODO verify cluster health to be green
+ // TODO verify that ES cluster is green
+
+ cluster.getNodes().forEach(node -> {
+ node.assertThatProcessesAreUp();
+ assertThat(node.anyLogsContain(" ERROR ")).isFalse();
+ assertThat(node.anyLogsContain("MessageException")).isFalse();
+ });
+
+ // verify that there's a single web startup leader
+ Node startupLeader = cluster.getAppNodes()
+ .filter(Node::isStartupLeader)
+ .reduce(singleElement())
+ .get();
+ assertThat(startupLeader.hasStartupLeaderOperations()).isTrue();
+ assertThat(startupLeader.hasCreatedSearchIndices()).isTrue();
+
+ // verify that the second app node is a startup follower
+ Node startupFollower = cluster.getAppNodes()
+ .filter(Node::isStartupFollower)
+ .reduce(singleElement())
+ .get();
+ assertThat(startupFollower.hasStartupLeaderOperations()).isFalse();
+ assertThat(startupFollower.hasCreatedSearchIndices()).isFalse();
+
+ cluster.getAppNodes().forEach(app -> {
+ // compute engine is being started when app node is already in status UP
+ app.waitForCeLogsContain("Compute Engine is operational");
+ assertThat(app.anyLogsContain("Process[ce] is up")).isTrue();
+ });
+ }
+ }
+
+ @Test
+ public void minimal_cluster_is_2_search_and_1_application_nodes() throws Exception {
+ try (Cluster cluster = newCluster(2, 1)) {
+ cluster.getNodes().forEach(Node::start);
+
+ Node app = cluster.getAppNode(0);
+ app.waitForStatusUp();
+ app.waitForCeLogsContain("Compute Engine is operational");
+ assertThat(app.isStartupLeader()).isTrue();
+ assertThat(app.hasStartupLeaderOperations()).isTrue();
+
+ // TODO verify cluster health to be yellow
+ // TODO verify that ES cluster is yellow
+
+ cluster.getNodes().forEach(node -> {
+ assertThat(node.anyLogsContain(" ERROR ")).isFalse();
+ node.assertThatProcessesAreUp();
+ });
+ }
+ }
+
+ @Test
+ public void configuration_of_connection_to_other_nodes_can_be_non_exhaustive() throws Exception {
+ try (Cluster cluster = new Cluster(null)) {
+ NodeConfig searchConfig1 = newSearchConfig();
+ NodeConfig searchConfig2 = newSearchConfig();
+ NodeConfig appConfig = newApplicationConfig();
+
+ // HZ bus : app -> search 2 -> search1, which is not recommended at all !!!
+ searchConfig2.addConnectionToBus(searchConfig1);
+ appConfig.addConnectionToBus(searchConfig2);
+
+ // search1 is not configured to connect search2
+ // app is not configured to connect to search 1
+ // --> not recommended at all !!!
+ searchConfig2.addConnectionToSearch(searchConfig1);
+ appConfig.addConnectionToSearch(searchConfig2);
+
+ cluster.startNode(searchConfig1, nothing());
+ cluster.startNode(searchConfig2, nothing());
+ Node app = cluster.startNode(appConfig, nothing());
+
+ app.waitForStatusUp();
+ assertThat(app.isStartupLeader()).isTrue();
+ assertThat(app.hasStartupLeaderOperations()).isTrue();
+ // TODO verify cluster health to be yellow
+ // TODO verify that ES cluster is yellow
+
+ // no errors
+ cluster.getNodes().forEach(node -> {
+ assertThat(node.anyLogsContain(" ERROR ")).isFalse();
+ node.assertThatProcessesAreUp();
+ });
+ }
+ }
+
+ @Test
+ public void node_fails_to_join_cluster_if_different_cluster_name() throws Exception {
+ try (Cluster cluster = new Cluster("foo")) {
+ NodeConfig searchConfig1 = newSearchConfig();
+ NodeConfig searchConfig2 = newSearchConfig();
+ NodeConfig.interconnectBus(searchConfig1, searchConfig2);
+ NodeConfig.interconnectSearch(searchConfig1, searchConfig2);
+ cluster.startNode(searchConfig1, nothing());
+ cluster.startNode(searchConfig2, nothing());
+
+ NodeConfig searchConfig3 = newSearchConfig()
+ .addConnectionToSearch(searchConfig1)
+ .addConnectionToBus(searchConfig1, searchConfig2);
+ Node search3 = cluster.addNode(searchConfig3, b -> b
+ .setServerProperty("sonar.cluster.name", "bar")
+ .setStartupLogWatcher(logLine -> logLine.contains("SonarQube is up")));
+ try {
+ search3.start();
+ fail();
+ } catch (IllegalStateException e) {
+ assertThat(e).hasMessage("Server startup failure");
+ // TODO how to force process to write into sonar.log, even if sonar.log.console=true ?
+ // assertThat(search3.anyLogsContain("This node has a cluster name [bar], which does not match [foo] from the cluster")).isTrue();
+ }
+ }
+ }
+
+ @Test
+ public void restarting_all_application_nodes_elects_a_new_startup_leader() throws Exception {
+ // no need for 3 search nodes, 2 is enough for the test
+ try (Cluster cluster = newCluster(2, 2)) {
+ cluster.getNodes().forEach(Node::start);
+ cluster.getAppNodes().forEach(Node::waitForStatusUp);
+
+ // stop application nodes only
+ cluster.getAppNodes().forEach(app -> {
+ app.stop();
+ app.cleanUpLogs();
+ // logs are empty, no more possible to know if node was startup leader/follower
+ assertThat(app.isStartupLeader()).isFalse();
+ assertThat(app.isStartupFollower()).isFalse();
+ });
+
+ // restart application nodes
+ cluster.getAppNodes().forEach(Node::start);
+ cluster.getAppNodes().forEach(Node::waitForStatusUp);
+
+ // one app node is elected as startup leader. It does some initialization stuff,
+ // like registration of rules. Search indices already exist and are up-to-date.
+ Node startupLeader = cluster.getAppNodes()
+ .filter(Node::isStartupLeader)
+ .reduce(singleElement())
+ .get();
+ assertThat(startupLeader.hasStartupLeaderOperations()).isTrue();
+ assertThat(startupLeader.hasCreatedSearchIndices()).isFalse();
+
+ Node startupFollower = cluster.getAppNodes()
+ .filter(Node::isStartupFollower)
+ .reduce(singleElement())
+ .get();
+ assertThat(startupFollower.hasStartupLeaderOperations()).isFalse();
+ assertThat(startupFollower.hasCreatedSearchIndices()).isFalse();
+ assertThat(startupFollower).isNotSameAs(startupLeader);
+ }
+ }
+
+ /**
+ * Used to have non-blocking {@link Node#start()}. Orchestrator considers
+ * node to be up as soon as the first log is generated.
+ */
+ private static Consumer<OrchestratorBuilder> nothing() {
+ return b -> {
+ };
+ }
+
+ /**
+ * Configure a cluster with recommended configuration (each node has references
+ * to other nodes)
+ */
+ private static Cluster newCluster(int nbOfSearchNodes, int nbOfAppNodes) {
+ Cluster cluster = new Cluster(null);
+
+ List<NodeConfig> configs = new ArrayList<>();
+ IntStream.range(0, nbOfSearchNodes).forEach(i -> configs.add(newSearchConfig()));
+ IntStream.range(0, nbOfAppNodes).forEach(i -> configs.add(newApplicationConfig()));
+ NodeConfig[] configsArray = configs.toArray(new NodeConfig[configs.size()]);
+
+ // a node is connected to all nodes, including itself (see sonar.cluster.hosts)
+ NodeConfig.interconnectBus(configsArray);
+
+ // search nodes are interconnected, and app nodes connect to all search nodes
+ NodeConfig.interconnectSearch(configsArray);
+
+ configs.forEach(c -> cluster.addNode(c, nothing()));
+ return cluster;
+ }
+
+ private static BinaryOperator<Node> singleElement() {
+ return (a, b) -> {
+ throw new IllegalStateException("More than one element");
+ };
+ }
+}
diff --git a/tests/src/test/java/org/sonarqube/tests/cluster/DataCenterEdition.java b/tests/src/test/java/org/sonarqube/tests/cluster/DataCenterEdition.java
deleted file mode 100644
index b431bcf90bb..00000000000
--- a/tests/src/test/java/org/sonarqube/tests/cluster/DataCenterEdition.java
+++ /dev/null
@@ -1,53 +0,0 @@
-/*
- * SonarQube
- * Copyright (C) 2009-2017 SonarSource SA
- * mailto:info 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.sonarqube.tests.cluster;
-
-import java.util.concurrent.ExecutionException;
-
-import static org.sonarqube.tests.cluster.Cluster.NodeType.APPLICATION;
-import static org.sonarqube.tests.cluster.Cluster.NodeType.SEARCH;
-
-public class DataCenterEdition {
-
- private final Cluster cluster;
-
- public DataCenterEdition() {
- cluster = Cluster.builder()
- .addNode(SEARCH)
- .addNode(SEARCH)
- .addNode(SEARCH)
- .addNode(APPLICATION)
- .addNode(APPLICATION)
- .build();
- }
-
- public void stop() throws ExecutionException, InterruptedException {
- cluster.stop();
- }
-
- public void start() throws ExecutionException, InterruptedException {
- cluster.start();
- }
-
- public Cluster getCluster() {
- return cluster;
- }
-}
diff --git a/tests/src/test/java/org/sonarqube/tests/cluster/DataCenterEditionTest.java b/tests/src/test/java/org/sonarqube/tests/cluster/DataCenterEditionTest.java
deleted file mode 100644
index 52f7930e434..00000000000
--- a/tests/src/test/java/org/sonarqube/tests/cluster/DataCenterEditionTest.java
+++ /dev/null
@@ -1,157 +0,0 @@
-/*
- * SonarQube
- * Copyright (C) 2009-2017 SonarSource SA
- * mailto:info 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.sonarqube.tests.cluster;
-
-import com.sonar.orchestrator.db.Database;
-import com.sonar.orchestrator.db.DatabaseClient;
-import java.io.IOException;
-import java.net.InetAddress;
-import java.net.ServerSocket;
-import java.sql.Connection;
-import java.sql.SQLException;
-import java.sql.Statement;
-import java.util.concurrent.ExecutionException;
-import java.util.concurrent.TimeUnit;
-import javax.annotation.Nullable;
-import org.junit.Rule;
-import org.junit.Test;
-import org.junit.rules.DisableOnDebug;
-import org.junit.rules.TestRule;
-import org.junit.rules.Timeout;
-
-import static org.assertj.core.api.Assertions.assertThat;
-import static org.assertj.core.api.Assertions.tuple;
-import static org.junit.Assert.fail;
-import static org.sonarqube.tests.cluster.Cluster.NodeType.APPLICATION;
-import static org.sonarqube.tests.cluster.Cluster.NodeType.SEARCH;
-
-public class DataCenterEditionTest {
-
- @Rule
- public TestRule timeout = new DisableOnDebug(Timeout.builder()
- .withLookingForStuckThread(true)
- .withTimeout(5, TimeUnit.MINUTES)
- .build());
-
- @Test
- public void launch() throws ExecutionException, InterruptedException {
- DataCenterEdition dce = new DataCenterEdition();
- Cluster cluster = dce.getCluster();
- dce.start();
- assertThat(cluster.getNodes())
- .extracting(Cluster.Node::getType, n -> isPortBound(false, n.getEsPort()), n -> isPortBound(true, n.getWebPort()))
- .containsExactlyInAnyOrder(
- tuple(SEARCH, true, false),
- tuple(SEARCH, true, false),
- tuple(SEARCH, true, false),
- tuple(APPLICATION, false, true),
- tuple(APPLICATION, false, true)
- );
- dce.stop();
- }
-
- @Test
- public void upgrade_application_nodes_without_stopping_search_nodes_must_work() throws ExecutionException, InterruptedException, SQLException {
- DataCenterEdition dce = new DataCenterEdition();
- Cluster cluster = dce.getCluster();
- dce.start();
-
- // Stop all Application nodes
- cluster.stopAll(n -> n.getType() == APPLICATION);
-
- // Drop the schema
- Database database = cluster.getNodes().get(0).getOrchestrator().getDatabase();
- dropAndCreate(database.getClient());
- assertDatabaseDropped(database);
-
- // Start all Application nodes
- cluster.startAll(n -> n.getType() == APPLICATION);
-
- // We are expecting a new leader to be elected which will recreate the database
- assertDatabaseInitialized(database);
-
- dce.stop();
- }
-
- @Test
- public void using_different_cluster_names_should_fail() throws ExecutionException, InterruptedException, SQLException {
- Cluster cluster = Cluster.builder()
- .addNode(SEARCH, n -> n.getProperties().setProperty(Cluster.CLUSTER_NAME, "goodClusterName"))
- .addNode(SEARCH, n -> n.getProperties().setProperty(Cluster.CLUSTER_NAME, "goodClusterName"))
- .addNode(SEARCH, n -> n.getProperties().setProperty(Cluster.CLUSTER_NAME, "goodClusterName"))
- .addNode(APPLICATION, n -> n.getProperties().setProperty(Cluster.CLUSTER_NAME, "goodClusterName"))
- .addNode(APPLICATION, n -> n.getProperties().setProperty(Cluster.CLUSTER_NAME, "badClusterName"))
- .build();
- cluster.startAll(n -> "goodClusterName".equals(n.getProperties().getProperty(Cluster.CLUSTER_NAME)));
-
- try {
- cluster.startAll(n -> "badClusterName".equals(n.getProperties().getProperty(Cluster.CLUSTER_NAME)));
- fail("A node with a bad cluster name was able to join the cluster");
- } catch (Exception e) {
- // we expect, that joining the cluster fails
- System.out.println(e);
- }
- }
-
- private void assertDatabaseInitialized(Database database) {
- assertThat(countRowsOfMigration(database)).isGreaterThan(0);
- }
-
- private int countRowsOfMigration(Database database) {
- return database.countSql("select count(*) from schema_migrations");
- }
-
- private void assertDatabaseDropped(Database database) {
- try {
- countRowsOfMigration(database);
- fail("Table 'schema_migrations' has not been dropped");
- } catch (Exception e) {
- // we expect the table to not exist
- }
- }
-
- private static boolean isPortBound(boolean loopback, @Nullable Integer port) {
- if (port == null) {
- return false;
- }
- InetAddress inetAddress = loopback ? InetAddress.getLoopbackAddress() : Cluster.getNonloopbackIPv4Address();
- try (ServerSocket socket = new ServerSocket(port, 50, inetAddress)) {
- throw new IllegalStateException("A port was set explicitly, but was not bound (port="+port+")");
- } catch (IOException e) {
- return true;
- }
- }
-
- private static void dropAndCreate(DatabaseClient databaseClient) throws SQLException {
- try (Connection connection = databaseClient.openRootConnection()) {
- executeDdl(connection, databaseClient.getDropDdl());
- executeDdl(connection, databaseClient.getCreateDdl());
- }
- }
-
- private static void executeDdl(Connection connection, String... ddls) throws SQLException {
- try (Statement stmt = connection.createStatement()) {
- for (String ddl : ddls) {
- stmt.executeUpdate(ddl);
- }
- }
- }
-}
diff --git a/tests/src/test/java/org/sonarqube/tests/cluster/Node.java b/tests/src/test/java/org/sonarqube/tests/cluster/Node.java
new file mode 100644
index 00000000000..03dd4daf456
--- /dev/null
+++ b/tests/src/test/java/org/sonarqube/tests/cluster/Node.java
@@ -0,0 +1,185 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2017 SonarSource SA
+ * mailto:info 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.sonarqube.tests.cluster;
+
+import com.google.common.util.concurrent.Uninterruptibles;
+import com.sonar.orchestrator.Orchestrator;
+import java.io.File;
+import java.io.IOException;
+import java.net.ServerSocket;
+import java.util.Map;
+import java.util.concurrent.TimeUnit;
+import javax.annotation.Nullable;
+import org.apache.commons.io.FileUtils;
+import util.ItUtils;
+
+import static org.assertj.core.api.Assertions.assertThat;
+
+class Node {
+
+ private final NodeConfig config;
+ private final Orchestrator orchestrator;
+
+ Node(NodeConfig config, Orchestrator orchestrator) {
+ this.config = config;
+ this.orchestrator = orchestrator;
+ }
+
+ NodeConfig getConfig() {
+ return config;
+ }
+
+ /**
+ * Non-blocking startup of node. The method does not wait for
+ * node to be started because Orchestrator uses a StartupLogWatcher
+ * that returns as soon as a log is generated.
+ */
+ void start() {
+ orchestrator.start();
+ }
+
+ void stop() {
+ orchestrator.stop();
+ }
+
+ void cleanUpLogs() {
+ if (orchestrator.getServer() != null) {
+ FileUtils.deleteQuietly(orchestrator.getServer().getWebLogs());
+ FileUtils.deleteQuietly(orchestrator.getServer().getCeLogs());
+ FileUtils.deleteQuietly(orchestrator.getServer().getEsLogs());
+ FileUtils.deleteQuietly(orchestrator.getServer().getAppLogs());
+ }
+ }
+
+ boolean isStartupLeader() {
+ return webLogsContain("Cluster enabled (startup leader)");
+ }
+
+ boolean isStartupFollower() {
+ return webLogsContain("Cluster enabled (startup follower)");
+ }
+
+ void waitForStatusUp() {
+ waitForStatus("UP");
+ }
+
+ void waitForStatus(String expectedStatus) {
+ String status = null;
+ try {
+ while (!expectedStatus.equals(status)) {
+ if (orchestrator.getServer() != null) {
+ try {
+ Map<String, Object> json = ItUtils.jsonToMap(orchestrator.getServer().newHttpCall("api/system/status").executeUnsafely().getBodyAsString());
+ status = (String) json.get("status");
+ } catch (Exception e) {
+ // ignored
+ }
+ }
+
+ Thread.sleep(500);
+ }
+ } catch (InterruptedException e) {
+ Thread.currentThread().interrupt();
+ }
+ }
+
+ void assertThatProcessesAreUp() {
+ assertThat(arePortsBound()).as(getConfig().getType().toString()).isTrue();
+ switch (config.getType()) {
+ case SEARCH:
+ assertThat(anyLogsContain("Process[es] is up")).isTrue();
+ assertThat(anyLogsContain("Process[web] is up")).isFalse();
+ assertThat(anyLogsContain("Elasticsearch cluster enabled")).isTrue();
+ break;
+ case APPLICATION:
+ assertThat(anyLogsContain("Process[es] is up")).isFalse();
+ assertThat(anyLogsContain("Process[web] is up")).isTrue();
+ assertThat(anyLogsContain("Elasticsearch cluster enabled")).isFalse();
+ break;
+ }
+ }
+
+ void waitForCeLogsContain(String expectedMessage) {
+ boolean found = false;
+ while (!found) {
+ found = orchestrator.getServer() != null && fileContains(orchestrator.getServer().getCeLogs(), expectedMessage);
+ if (!found) {
+ Uninterruptibles.sleepUninterruptibly(1_000, TimeUnit.MILLISECONDS);
+ }
+ }
+ }
+
+ boolean hasStartupLeaderOperations() throws IOException {
+ if (orchestrator.getServer() == null) {
+ return false;
+ }
+ String logs = FileUtils.readFileToString(orchestrator.getServer().getWebLogs());
+ return logs.contains("Register metrics") &&
+ logs.contains("Register rules");
+ }
+
+ boolean hasCreatedSearchIndices() throws IOException {
+ if (orchestrator.getServer() == null) {
+ return false;
+ }
+ String logs = FileUtils.readFileToString(orchestrator.getServer().getWebLogs());
+ return logs.contains("[o.s.s.e.IndexCreator] Create index");
+ }
+
+ boolean anyLogsContain(String message) {
+ if (orchestrator.getServer() == null) {
+ return false;
+ }
+ return fileContains(orchestrator.getServer().getAppLogs(), message) ||
+ fileContains(orchestrator.getServer().getWebLogs(), message) ||
+ fileContains(orchestrator.getServer().getEsLogs(), message) ||
+ fileContains(orchestrator.getServer().getCeLogs(), message);
+ }
+
+ private boolean webLogsContain(String message) {
+ if (orchestrator.getServer() == null) {
+ return false;
+ }
+ return fileContains(orchestrator.getServer().getWebLogs(), message);
+ }
+
+ private static boolean fileContains(@Nullable File logFile, String message) {
+ try {
+ return logFile != null && logFile.exists() && FileUtils.readFileToString(logFile).contains(message);
+ } catch (Exception e) {
+ throw new IllegalStateException(e);
+ }
+ }
+
+ private boolean arePortsBound() {
+ return isPortBound(config.getHzPort()) &&
+ config.getSearchPort().map(this::isPortBound).orElse(true) &&
+ config.getWebPort().map(this::isPortBound).orElse(true);
+ }
+
+ private boolean isPortBound(int port) {
+ try (ServerSocket socket = new ServerSocket(port, 50, config.getAddress())) {
+ return false;
+ } catch (IOException e) {
+ return true;
+ }
+ }
+
+}
diff --git a/tests/src/test/java/org/sonarqube/tests/cluster/NodeConfig.java b/tests/src/test/java/org/sonarqube/tests/cluster/NodeConfig.java
new file mode 100644
index 00000000000..df3d2ed7dcf
--- /dev/null
+++ b/tests/src/test/java/org/sonarqube/tests/cluster/NodeConfig.java
@@ -0,0 +1,187 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2017 SonarSource SA
+ * mailto:info 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.sonarqube.tests.cluster;
+
+import com.sonar.orchestrator.util.NetworkUtils;
+import java.net.Inet4Address;
+import java.net.InetAddress;
+import java.net.NetworkInterface;
+import java.net.SocketException;
+import java.util.ArrayList;
+import java.util.Arrays;
+import java.util.Collections;
+import java.util.Enumeration;
+import java.util.List;
+import java.util.Optional;
+import javax.annotation.Nullable;
+
+import static com.google.common.base.Preconditions.checkArgument;
+
+class NodeConfig {
+
+ enum NodeType {
+ SEARCH("search"), APPLICATION("application");
+
+ final String value;
+
+ NodeType(String value) {
+ this.value = value;
+ }
+
+ String getValue() {
+ return value;
+ }
+ }
+
+ private final NodeType type;
+ @Nullable
+ private final String name;
+ private final InetAddress address;
+ private final int hzPort;
+ @Nullable
+ private final Integer searchPort;
+ @Nullable
+ private final Integer webPort;
+ private final List<NodeConfig> connectedNodes = new ArrayList<>();
+ private final List<NodeConfig> searchNodes = new ArrayList<>();
+
+ private NodeConfig(NodeType type, @Nullable String name) {
+ this.type = type;
+ this.name = name;
+ this.address = getNonLoopbackIpv4Address();
+ this.hzPort = NetworkUtils.getNextAvailablePort(this.address);
+ this.connectedNodes.add(this);
+ switch (type) {
+ case SEARCH:
+ this.searchPort = NetworkUtils.getNextAvailablePort(this.address);
+ this.webPort = null;
+ this.searchNodes.add(this);
+ break;
+ case APPLICATION:
+ this.searchPort = null;
+ this.webPort = NetworkUtils.getNextAvailablePort(this.address);
+ break;
+ default:
+ throw new IllegalArgumentException();
+ }
+ }
+
+ NodeType getType() {
+ return type;
+ }
+
+ Optional<String> getName() {
+ return Optional.ofNullable(name);
+ }
+
+ InetAddress getAddress() {
+ return address;
+ }
+
+ int getHzPort() {
+ return hzPort;
+ }
+
+ Optional<Integer> getSearchPort() {
+ return Optional.ofNullable(searchPort);
+ }
+
+ Optional<Integer> getWebPort() {
+ return Optional.ofNullable(webPort);
+ }
+
+ String getHzHost() {
+ return address.getHostAddress() + ":" + hzPort;
+ }
+
+ String getSearchHost() {
+ return address.getHostAddress() + ":" + searchPort;
+ }
+
+ NodeConfig addConnectionToBus(NodeConfig... configs) {
+ connectedNodes.addAll(Arrays.asList(configs));
+ return this;
+ }
+
+ NodeConfig addConnectionToSearch(NodeConfig... configs) {
+ Arrays.stream(configs).forEach(config -> {
+ checkArgument(config.getType() == NodeType.SEARCH);
+ searchNodes.add(config);
+ });
+ return this;
+ }
+
+ List<NodeConfig> getConnectedNodes() {
+ return connectedNodes;
+ }
+
+ List<NodeConfig> getSearchNodes() {
+ return searchNodes;
+ }
+
+ static NodeConfig newApplicationConfig() {
+ return new NodeConfig(NodeType.APPLICATION, null);
+ }
+
+ static NodeConfig newSearchConfig() {
+ return new NodeConfig(NodeType.SEARCH, null);
+ }
+
+ /**
+ * See property sonar.cluster.hosts
+ */
+ static void interconnectBus(NodeConfig... configs) {
+ Arrays.stream(configs).forEach(config -> Arrays.stream(configs).filter(c -> c != config).forEach(config::addConnectionToBus));
+ }
+
+ /**
+ * See property sonar.cluster.search.hosts
+ */
+ static void interconnectSearch(NodeConfig... configs) {
+ Arrays.stream(configs).forEach(config -> Arrays.stream(configs)
+ .filter(c -> c.getType() == NodeType.SEARCH)
+ .forEach(config::addConnectionToSearch));
+ }
+
+ private static InetAddress getNonLoopbackIpv4Address() {
+ try {
+ Enumeration<NetworkInterface> nets = NetworkInterface.getNetworkInterfaces();
+ for (NetworkInterface networkInterface : Collections.list(nets)) {
+ if (!networkInterface.isLoopback() && networkInterface.isUp() && !isBlackListed(networkInterface)) {
+ Enumeration<InetAddress> inetAddresses = networkInterface.getInetAddresses();
+ while (inetAddresses.hasMoreElements()) {
+ InetAddress inetAddress = inetAddresses.nextElement();
+ if (inetAddress instanceof Inet4Address) {
+ return inetAddress;
+ }
+ }
+ }
+ }
+ } catch (SocketException se) {
+ throw new RuntimeException("Cannot find a non loopback card required for tests", se);
+ }
+ throw new RuntimeException("Cannot find a non loopback card required for tests");
+ }
+
+ private static boolean isBlackListed(NetworkInterface networkInterface) {
+ return networkInterface.getName().startsWith("docker") ||
+ networkInterface.getName().startsWith("vboxnet");
+ }
+}
diff --git a/tests/src/test/java/org/sonarqube/tests/cluster/StartupLogWatcherImpl.java b/tests/src/test/java/org/sonarqube/tests/cluster/StartupLogWatcherImpl.java
deleted file mode 100644
index 170ddc16872..00000000000
--- a/tests/src/test/java/org/sonarqube/tests/cluster/StartupLogWatcherImpl.java
+++ /dev/null
@@ -1,33 +0,0 @@
-/*
- * SonarQube
- * Copyright (C) 2009-2017 SonarSource SA
- * mailto:info 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.sonarqube.tests.cluster;
-
-
-import com.sonar.orchestrator.server.StartupLogWatcher;
-
-public class StartupLogWatcherImpl implements StartupLogWatcher {
- private static final String STARTUP_EXPECTED_MESSAGE = "SonarQube is up";
-
- @Override
- public boolean isStarted(String logLine) {
- return logLine.contains(STARTUP_EXPECTED_MESSAGE);
- }
-}
diff --git a/tests/src/test/java/org/sonarqube/tests/serverSystem/ClusterTest.java b/tests/src/test/java/org/sonarqube/tests/serverSystem/ClusterTest.java
deleted file mode 100644
index ff110377844..00000000000
--- a/tests/src/test/java/org/sonarqube/tests/serverSystem/ClusterTest.java
+++ /dev/null
@@ -1,215 +0,0 @@
-/*
- * SonarQube
- * Copyright (C) 2009-2017 SonarSource SA
- * mailto:info 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.sonarqube.tests.serverSystem;
-
-import com.google.common.base.Joiner;
-import com.google.common.collect.ImmutableMap;
-import com.sonar.orchestrator.Orchestrator;
-import com.sonar.orchestrator.server.StartupLogWatcher;
-import java.io.File;
-import java.io.FileInputStream;
-import java.io.FileOutputStream;
-import java.io.IOException;
-import java.nio.file.Files;
-import java.util.List;
-import java.util.Map;
-import java.util.Properties;
-import java.util.regex.Matcher;
-import java.util.regex.Pattern;
-import java.util.stream.Collectors;
-import java.util.stream.Stream;
-import org.apache.commons.io.FileUtils;
-import org.apache.commons.lang.StringUtils;
-import org.junit.Ignore;
-import org.junit.Test;
-import org.sonarqube.ws.Issues;
-import org.sonarqube.ws.Settings;
-import org.sonarqube.ws.client.rule.SearchWsRequest;
-import org.sonarqube.ws.client.setting.ValuesRequest;
-import util.ItUtils;
-
-import static org.apache.commons.lang3.StringUtils.containsIgnoreCase;
-import static org.assertj.core.api.Assertions.assertThat;
-import static util.ItUtils.newWsClient;
-
-@Ignore("temporarily ignored")
-public class ClusterTest {
-
- private static final String CONF_FILE_PATH = "conf/sonar.properties";
-
- /**
- * SONAR-7899
- */
- @Test
- public void secondary_nodes_do_not_write_to_datastores_at_startup() throws Exception {
- // start "startup leader", which creates and populates datastores
- Orchestrator orchestrator = Orchestrator.builderEnv()
- .setServerProperty("sonar.cluster.enabled", "true")
- .setServerProperty("sonar.cluster.name", "secondary_nodes_do_not_write_to_datastores_at_startup")
- .setServerProperty("sonar.cluster.web.startupLeader", "true")
- .setServerProperty("sonar.log.level", "TRACE")
- .addPlugin(ItUtils.xooPlugin())
- .build();
- orchestrator.start();
-
- expectLog(orchestrator, "Cluster enabled (startup leader)");
- expectWriteOperations(orchestrator, true);
- // verify that datastores are populated by requesting rules
- assertThat(newWsClient(orchestrator).rules().search(new SearchWsRequest()).getTotal()).isGreaterThan(0);
-
- FileUtils.write(orchestrator.getServer().getWebLogs(), "", false);
- updateSonarPropertiesFile(orchestrator, ImmutableMap.of("sonar.cluster.web.startupLeader", "false"));
- orchestrator.restartServer();
-
- expectLog(orchestrator, "Cluster enabled (startup follower)");
- expectWriteOperations(orchestrator, false);
-
- orchestrator.stop();
- }
-
- @Test
- public void start_cluster_of_elasticsearch_and_web_nodes() throws IOException {
- Orchestrator elasticsearch = null;
- Orchestrator web = null;
-
- try {
- ElasticsearchStartupWatcher esWatcher = new ElasticsearchStartupWatcher();
- elasticsearch = Orchestrator.builderEnv()
- .setServerProperty("sonar.cluster.enabled", "true")
- .setServerProperty("sonar.cluster.name", "start_cluster_of_elasticsearch_and_web_nodes")
- .setServerProperty("sonar.cluster.web.disabled", "true")
- .setServerProperty("sonar.cluster.ce.disabled", "true")
- .setStartupLogWatcher(esWatcher)
- .build();
- elasticsearch.start();
- assertThat(esWatcher.port).isGreaterThan(0);
- assertThat(FileUtils.readFileToString(elasticsearch.getServer().getAppLogs())).doesNotContain("Process[web]");
-
- web = Orchestrator.builderEnv()
- .setServerProperty("sonar.cluster.enabled", "true")
- .setServerProperty("sonar.cluster.name", "start_cluster_of_elasticsearch_and_web_nodes")
- .setServerProperty("sonar.cluster.web.startupLeader", "true")
- .setServerProperty("sonar.cluster.search.disabled", "true")
- .setServerProperty("sonar.cluster.search.hosts", "localhost:" + esWatcher.port)
- // no need for compute engine in this test. Disable it for faster test.
- .setServerProperty("sonar.cluster.ce.disabled", "true")
- // override the default watcher provided by Orchestrator
- // which waits for Compute Engine to be up
- .setStartupLogWatcher(log -> log.contains("SonarQube is up"))
- .build();
- web.start();
-
- String coreId = getPropertyValue(web, "sonar.core.id");
- String startTime = getPropertyValue(web, "sonar.core.startTime");
-
- assertThat(FileUtils.readFileToString(web.getServer().getAppLogs())).doesNotContain("Process[es]");
- // call a web service that requires Elasticsearch
- Issues.SearchWsResponse wsResponse = newWsClient(web).issues().search(new org.sonarqube.ws.client.issue.SearchWsRequest());
- assertThat(wsResponse.getIssuesCount()).isEqualTo(0);
-
- web.restartServer();
-
- // sonar core id must not change after restart
- assertThat(getPropertyValue(web, "sonar.core.id")).isEqualTo(coreId);
- // startTime must change at each startup
- assertThat(getPropertyValue(web, "sonar.core.startTime")).isNotEqualTo(startTime);
- } finally {
- if (web != null) {
- web.stop();
- }
- if (elasticsearch != null) {
- elasticsearch.stop();
- }
- }
- }
-
- private static String getPropertyValue(Orchestrator web, String property) {
- Settings.ValuesWsResponse response = ItUtils.newAdminWsClient(web).settings().values(ValuesRequest.builder().setKeys(property).build());
- List<Settings.Setting> settingsList = response.getSettingsList();
- if (settingsList.isEmpty()) {
- return null;
- }
- assertThat(settingsList).hasSize(1);
- return settingsList.iterator().next().getValue();
- }
-
- private static class ElasticsearchStartupWatcher implements StartupLogWatcher {
- private final Pattern pattern = Pattern.compile("Elasticsearch listening on .*:(\\d+)");
- private int port = -1;
-
- @Override
- public boolean isStarted(String log) {
- Matcher matcher = pattern.matcher(log);
- if (matcher.find()) {
- port = Integer.parseInt(matcher.group(1));
- }
- return log.contains("Process[es] is up");
- }
- }
-
- private static void expectLog(Orchestrator orchestrator, String expectedLog) throws IOException {
- File logFile = orchestrator.getServer().getWebLogs();
- try (Stream<String> lines = Files.lines(logFile.toPath())) {
- assertThat(lines.anyMatch(s -> StringUtils.containsIgnoreCase(s, expectedLog))).isTrue();
- }
- }
-
- private static void expectWriteOperations(Orchestrator orchestrator, boolean expected) throws IOException {
- try (Stream<String> lines = Files.lines(orchestrator.getServer().getWebLogs().toPath())) {
- List<String> writeOperations = lines.filter(ClusterTest::isWriteOperation).collect(Collectors.toList());
- if (expected) {
- assertThat(writeOperations).isNotEmpty();
- } else {
- assertThat(writeOperations).as("Unexpected write operations: " + Joiner.on('\n').join(writeOperations)).isEmpty();
-
- }
- }
- }
-
- private static boolean isWriteOperation(String log) {
- return isDbWriteOperation(log) || isEsWriteOperation(log);
- }
-
- private static boolean isDbWriteOperation(String log) {
- return log.contains("web[][sql]") && (containsIgnoreCase(log, "sql=insert") ||
- containsIgnoreCase(log, "sql=update") ||
- containsIgnoreCase(log, "sql=delete") ||
- containsIgnoreCase(log, "sql=create"));
- }
-
- private static boolean isEsWriteOperation(String log) {
- return log.contains("web[][es]") && (containsIgnoreCase(log, "Create index") ||
- containsIgnoreCase(log, "Create type") ||
- containsIgnoreCase(log, "put mapping request") ||
- containsIgnoreCase(log, "refresh request") ||
- containsIgnoreCase(log, "index request"));
- }
-
- private static void updateSonarPropertiesFile(Orchestrator orchestrator, Map<String, String> props) throws IOException {
- Properties propsFile = new Properties();
- try (FileInputStream conf = FileUtils.openInputStream(new File(orchestrator.getServer().getHome(), CONF_FILE_PATH))) {
- propsFile.load(conf);
- propsFile.putAll(props);
- }
- try (FileOutputStream conf = FileUtils.openOutputStream(new File(orchestrator.getServer().getHome(), CONF_FILE_PATH))) {
- propsFile.store(conf, "");
- }
- }
-}