aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/computation/task/projectanalysis/container/ProjectAnalysisTaskContainerPopulator.java2
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/computation/task/projectanalysis/webhook/WebhookPostTask.java30
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/webhook/Branch.java82
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/webhook/CeTask.java72
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/webhook/Project.java76
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/webhook/ProjectAnalysis.java106
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/webhook/QualityGate.java211
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/webhook/WebhookModule.java3
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/webhook/WebhookPayloadFactory.java (renamed from server/sonar-server/src/main/java/org/sonar/server/computation/task/projectanalysis/webhook/WebhookPayloadFactory.java)7
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/webhook/WebhookPayloadFactoryImpl.java (renamed from server/sonar-server/src/main/java/org/sonar/server/computation/task/projectanalysis/webhook/WebhookPayloadFactoryImpl.java)67
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/computation/task/projectanalysis/webhook/WebhookPostTaskTest.java20
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/webhook/BranchTest.java55
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/webhook/CeTaskTest.java55
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/webhook/ProjectAnalysisTest.java150
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/webhook/ProjectTest.java64
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/webhook/QualityGateTest.java122
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/webhook/TestWebhookPayloadFactory.java33
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/webhook/WebhookModuleTest.java2
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/webhook/WebhookPayloadFactoryImplTest.java (renamed from server/sonar-server/src/test/java/org/sonar/server/computation/task/projectanalysis/webhook/WebhookPayloadFactoryImplTest.java)148
19 files changed, 1097 insertions, 208 deletions
diff --git a/server/sonar-server/src/main/java/org/sonar/server/computation/task/projectanalysis/container/ProjectAnalysisTaskContainerPopulator.java b/server/sonar-server/src/main/java/org/sonar/server/computation/task/projectanalysis/container/ProjectAnalysisTaskContainerPopulator.java
index 590813d7df7..267a4523e0f 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/computation/task/projectanalysis/container/ProjectAnalysisTaskContainerPopulator.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/computation/task/projectanalysis/container/ProjectAnalysisTaskContainerPopulator.java
@@ -117,7 +117,6 @@ import org.sonar.server.computation.task.projectanalysis.source.SourceHashReposi
import org.sonar.server.computation.task.projectanalysis.source.SourceLinesRepositoryImpl;
import org.sonar.server.computation.task.projectanalysis.step.ReportComputationSteps;
import org.sonar.server.computation.task.projectanalysis.step.SmallChangesetQualityGateSpecialCase;
-import org.sonar.server.computation.task.projectanalysis.webhook.WebhookPayloadFactoryImpl;
import org.sonar.server.computation.task.projectanalysis.webhook.WebhookPostTask;
import org.sonar.server.computation.task.step.ComputationStepExecutor;
import org.sonar.server.computation.task.step.ComputationSteps;
@@ -274,7 +273,6 @@ public final class ProjectAnalysisTaskContainerPopulator implements ContainerPop
// webhooks
WebhookModule.class,
- WebhookPayloadFactoryImpl.class,
WebhookPostTask.class);
}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/computation/task/projectanalysis/webhook/WebhookPostTask.java b/server/sonar-server/src/main/java/org/sonar/server/computation/task/projectanalysis/webhook/WebhookPostTask.java
index 41de76dc710..f6112804a54 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/computation/task/projectanalysis/webhook/WebhookPostTask.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/computation/task/projectanalysis/webhook/WebhookPostTask.java
@@ -19,10 +19,18 @@
*/
package org.sonar.server.computation.task.projectanalysis.webhook;
+import java.util.Date;
+import java.util.Optional;
+import java.util.stream.Collectors;
import org.sonar.api.ce.posttask.PostProjectAnalysisTask;
import org.sonar.api.config.Configuration;
import org.sonar.server.computation.task.projectanalysis.component.ConfigurationRepository;
+import org.sonar.server.webhook.Branch;
+import org.sonar.server.webhook.CeTask;
+import org.sonar.server.webhook.Project;
+import org.sonar.server.webhook.QualityGate;
import org.sonar.server.webhook.WebHooks;
+import org.sonar.server.webhook.WebhookPayloadFactory;
public class WebhookPostTask implements PostProjectAnalysisTask {
@@ -43,7 +51,27 @@ public class WebhookPostTask implements PostProjectAnalysisTask {
webHooks.sendProjectAnalysisUpdate(
config,
new WebHooks.Analysis(analysis.getProject().getUuid(), analysis.getCeTask().getId()),
- () -> payloadFactory.create(analysis));
+ () -> payloadFactory.create(convert(analysis)));
+ }
+
+ private static org.sonar.server.webhook.ProjectAnalysis convert(ProjectAnalysis analysis) {
+ return new org.sonar.server.webhook.ProjectAnalysis(
+ new CeTask(analysis.getCeTask().getId(), CeTask.Status.valueOf(analysis.getCeTask().getStatus().name())),
+ new Project(analysis.getProject().getUuid(), analysis.getProject().getKey(), analysis.getProject().getName()),
+ analysis.getBranch().map(b -> new Branch(b.isMain(), b.getName().orElse(null), Branch.Type.valueOf(b.getType().name()))).orElse(null),
+ Optional.ofNullable(analysis.getQualityGate())
+ .map(qg -> new QualityGate(
+ qg.getId(),
+ qg.getName(),
+ QualityGate.Status.valueOf(qg.getStatus().name()),
+ qg.getConditions().stream()
+ .map(c -> new QualityGate.Condition(QualityGate.EvaluationStatus.valueOf(c.getStatus().name()), c.getMetricKey(), QualityGate.Operator.valueOf(c.getOperator().name()),
+ c.getErrorThreshold(), c.getWarningThreshold(), c.isOnLeakPeriod(),
+ c.getStatus() == org.sonar.api.ce.posttask.QualityGate.EvaluationStatus.NO_VALUE ? null : c.getValue()))
+ .collect(Collectors.toSet())))
+ .orElse(null),
+ analysis.getAnalysisDate().map(Date::getTime).orElse(null),
+ analysis.getScannerContext().getProperties());
}
}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/webhook/Branch.java b/server/sonar-server/src/main/java/org/sonar/server/webhook/Branch.java
new file mode 100644
index 00000000000..cbb1b6cd053
--- /dev/null
+++ b/server/sonar-server/src/main/java/org/sonar/server/webhook/Branch.java
@@ -0,0 +1,82 @@
+/*
+ * 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.sonar.server.webhook;
+
+import java.util.Objects;
+import java.util.Optional;
+import javax.annotation.Nullable;
+
+import static java.util.Objects.requireNonNull;
+
+public final class Branch {
+ private final boolean main;
+ private final String name;
+ private final Type type;
+
+ public Branch(boolean main, @Nullable String name, Type type) {
+ this.main = main;
+ this.name = name;
+ this.type = requireNonNull(type, "type can't be null");
+ }
+
+ public boolean isMain() {
+ return main;
+ }
+
+ public Optional<String> getName() {
+ return Optional.ofNullable(name);
+ }
+
+ public Type getType() {
+ return type;
+ }
+
+ public enum Type {
+ LONG, SHORT
+ }
+
+ @Override
+ public boolean equals(Object o) {
+ if (this == o) {
+ return true;
+ }
+ if (o == null || getClass() != o.getClass()) {
+ return false;
+ }
+ Branch branch = (Branch) o;
+ return main == branch.main &&
+ Objects.equals(name, branch.name) &&
+ type == branch.type;
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(main, name, type);
+ }
+
+ @Override
+ public String toString() {
+ return "Branch{" +
+ "main=" + main +
+ ", name='" + name + '\'' +
+ ", type=" + type +
+ '}';
+ }
+}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/webhook/CeTask.java b/server/sonar-server/src/main/java/org/sonar/server/webhook/CeTask.java
new file mode 100644
index 00000000000..85053978109
--- /dev/null
+++ b/server/sonar-server/src/main/java/org/sonar/server/webhook/CeTask.java
@@ -0,0 +1,72 @@
+/*
+ * 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.sonar.server.webhook;
+
+import java.util.Objects;
+
+import static java.util.Objects.requireNonNull;
+
+public final class CeTask {
+ private final String id;
+ private final Status status;
+
+ public CeTask(String id, Status status) {
+ this.id = requireNonNull(id, "id can't be null");
+ this.status = requireNonNull(status, "status can't be null");
+ }
+
+ public String getId() {
+ return id;
+ }
+
+ public Status getStatus() {
+ return status;
+ }
+
+ public enum Status {
+ SUCCESS, FAILED
+ }
+
+ @Override
+ public boolean equals(Object o) {
+ if (this == o) {
+ return true;
+ }
+ if (o == null || getClass() != o.getClass()) {
+ return false;
+ }
+ CeTask task = (CeTask) o;
+ return Objects.equals(id, task.id) &&
+ status == task.status;
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(id, status);
+ }
+
+ @Override
+ public String toString() {
+ return "CeTask{" +
+ "id='" + id + '\'' +
+ ", status=" + status +
+ '}';
+ }
+}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/webhook/Project.java b/server/sonar-server/src/main/java/org/sonar/server/webhook/Project.java
new file mode 100644
index 00000000000..769f8f9d0b1
--- /dev/null
+++ b/server/sonar-server/src/main/java/org/sonar/server/webhook/Project.java
@@ -0,0 +1,76 @@
+/*
+ * 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.sonar.server.webhook;
+
+import java.util.Objects;
+
+import static java.util.Objects.requireNonNull;
+
+public final class Project {
+ private final String uuid;
+ private final String key;
+ private final String name;
+
+ public Project(String uuid, String key, String name) {
+ this.uuid = requireNonNull(uuid, "uuid can't be null");
+ this.key = requireNonNull(key, "key can't be null");
+ this.name = requireNonNull(name, "name can't be null");
+ }
+
+ public String getUuid() {
+ return uuid;
+ }
+
+ public String getKey() {
+ return key;
+ }
+
+ public String getName() {
+ return name;
+ }
+
+ @Override
+ public boolean equals(Object o) {
+ if (this == o) {
+ return true;
+ }
+ if (o == null || getClass() != o.getClass()) {
+ return false;
+ }
+ Project project = (Project) o;
+ return Objects.equals(uuid, project.uuid) &&
+ Objects.equals(key, project.key) &&
+ Objects.equals(name, project.name);
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(uuid, key, name);
+ }
+
+ @Override
+ public String toString() {
+ return "Project{" +
+ "uuid='" + uuid + '\'' +
+ ", key='" + key + '\'' +
+ ", name='" + name + '\'' +
+ '}';
+ }
+}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/webhook/ProjectAnalysis.java b/server/sonar-server/src/main/java/org/sonar/server/webhook/ProjectAnalysis.java
new file mode 100644
index 00000000000..7184c57c9f2
--- /dev/null
+++ b/server/sonar-server/src/main/java/org/sonar/server/webhook/ProjectAnalysis.java
@@ -0,0 +1,106 @@
+/*
+ * 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.sonar.server.webhook;
+
+import java.util.Date;
+import java.util.Map;
+import java.util.Objects;
+import java.util.Optional;
+import javax.annotation.Nullable;
+
+import static com.google.common.collect.ImmutableMap.copyOf;
+import static java.util.Objects.requireNonNull;
+
+public class ProjectAnalysis {
+ private final CeTask ceTask;
+ private final Project project;
+ private final Branch branch;
+ private final QualityGate qualityGate;
+ private final Long date;
+ private final Map<String, String> properties;
+
+ public ProjectAnalysis(CeTask ceTask, Project project,
+ @Nullable Branch branch, @Nullable QualityGate qualityGate, @Nullable Long date, Map<String, String> properties) {
+ this.ceTask = requireNonNull(ceTask, "ceTask can't be null");
+ this.project = requireNonNull(project, "project can't be null");
+ this.branch = branch;
+ this.qualityGate = qualityGate;
+ this.date = date;
+ this.properties = copyOf(requireNonNull(properties, "properties can't be null"));
+ }
+
+ public CeTask getCeTask() {
+ return ceTask;
+ }
+
+ public Project getProject() {
+ return project;
+ }
+
+ public Optional<Branch> getBranch() {
+ return Optional.ofNullable(branch);
+ }
+
+ public Optional<QualityGate> getQualityGate() {
+ return Optional.ofNullable(qualityGate);
+ }
+
+ public Optional<Date> getAnalysisDate() {
+ return Optional.ofNullable(date).map(Date::new);
+ }
+
+ public Map<String, String> getProperties() {
+ return properties;
+ }
+
+ @Override
+ public boolean equals(Object o) {
+ if (this == o) {
+ return true;
+ }
+ if (o == null || getClass() != o.getClass()) {
+ return false;
+ }
+ ProjectAnalysis that = (ProjectAnalysis) o;
+ return Objects.equals(ceTask, that.ceTask) &&
+ Objects.equals(project, that.project) &&
+ Objects.equals(branch, that.branch) &&
+ Objects.equals(qualityGate, that.qualityGate) &&
+ Objects.equals(date, that.date) &&
+ Objects.equals(properties, that.properties);
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(ceTask, project, branch, qualityGate, date, properties);
+ }
+
+ @Override
+ public String toString() {
+ return "ProjectAnalysis{" +
+ "ceTask=" + ceTask +
+ ", project=" + project +
+ ", branch=" + branch +
+ ", qualityGate=" + qualityGate +
+ ", date=" + date +
+ ", properties=" + properties +
+ '}';
+ }
+}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/webhook/QualityGate.java b/server/sonar-server/src/main/java/org/sonar/server/webhook/QualityGate.java
new file mode 100644
index 00000000000..2133842c6a8
--- /dev/null
+++ b/server/sonar-server/src/main/java/org/sonar/server/webhook/QualityGate.java
@@ -0,0 +1,211 @@
+/*
+ * 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.sonar.server.webhook;
+
+import com.google.common.collect.ImmutableSet;
+import java.util.Collection;
+import java.util.Objects;
+import java.util.Optional;
+import java.util.Set;
+import javax.annotation.Nullable;
+
+import static java.util.Objects.requireNonNull;
+
+public final class QualityGate {
+ private final String id;
+ private final String name;
+ private final Status status;
+ private final Set<Condition> conditions;
+
+ public QualityGate(String id, String name, Status status, Set<Condition> conditions) {
+ this.id = requireNonNull(id, "id can't be null");
+ this.name = requireNonNull(name, "name can't be null");
+ this.status = requireNonNull(status, "status can't be null");
+ this.conditions = ImmutableSet.copyOf(requireNonNull(conditions, "conditions can't be null"));
+ }
+
+ public String getId() {
+ return id;
+ }
+
+ public String getName() {
+ return name;
+ }
+
+ public Status getStatus() {
+ return status;
+ }
+
+ public Collection<Condition> getConditions() {
+ return conditions;
+ }
+
+ @Override
+ public boolean equals(Object o) {
+ if (this == o) {
+ return true;
+ }
+ if (o == null || getClass() != o.getClass()) {
+ return false;
+ }
+ QualityGate that = (QualityGate) o;
+ return Objects.equals(id, that.id) &&
+ Objects.equals(name, that.name) &&
+ status == that.status &&
+ Objects.equals(conditions, that.conditions);
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(id, name, status, conditions);
+ }
+
+ @Override
+ public String toString() {
+ return "QualityGate{" +
+ "id='" + id + '\'' +
+ ", name='" + name + '\'' +
+ ", status=" + status +
+ ", conditions=" + conditions +
+ '}';
+ }
+
+ public enum Status {
+ OK,
+ WARN,
+ ERROR
+ }
+
+ public static final class Condition {
+ private final EvaluationStatus status;
+ private final String metricKey;
+ private final Operator operator;
+ private final String errorThreshold;
+ private final String warnThreshold;
+ private final boolean onLeakPeriod;
+ private final String value;
+
+ public Condition(EvaluationStatus status, String metricKey, Operator operator,
+ @Nullable String errorThreshold, @Nullable String warnThreshold,
+ boolean onLeakPeriod, @Nullable String value) {
+ this.status = requireNonNull(status, "status can't be null");
+ this.metricKey = requireNonNull(metricKey, "metricKey can't be null");
+ this.operator = requireNonNull(operator, "operator can't be null");
+ this.errorThreshold = errorThreshold;
+ this.warnThreshold = warnThreshold;
+ this.onLeakPeriod = onLeakPeriod;
+ this.value = value;
+ }
+
+ public EvaluationStatus getStatus() {
+ return status;
+ }
+
+ public String getMetricKey() {
+ return metricKey;
+ }
+
+ public Operator getOperator() {
+ return operator;
+ }
+
+ public Optional<String> getErrorThreshold() {
+ return Optional.ofNullable(errorThreshold);
+ }
+
+ public Optional<String> getWarningThreshold() {
+ return Optional.ofNullable(warnThreshold);
+ }
+
+ public boolean isOnLeakPeriod() {
+ return onLeakPeriod;
+ }
+
+ public Optional<String> getValue() {
+ return Optional.ofNullable(value);
+ }
+
+ @Override
+ public boolean equals(Object o) {
+ if (this == o) {
+ return true;
+ }
+ if (o == null || getClass() != o.getClass()) {
+ return false;
+ }
+ Condition condition = (Condition) o;
+ return onLeakPeriod == condition.onLeakPeriod &&
+ status == condition.status &&
+ Objects.equals(metricKey, condition.metricKey) &&
+ operator == condition.operator &&
+ Objects.equals(errorThreshold, condition.errorThreshold) &&
+ Objects.equals(warnThreshold, condition.warnThreshold) &&
+ Objects.equals(value, condition.value);
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(status, metricKey, operator, errorThreshold, warnThreshold, onLeakPeriod, value);
+ }
+
+ @Override
+ public String toString() {
+ return "Condition{" +
+ "status=" + status +
+ ", metricKey='" + metricKey + '\'' +
+ ", operator=" + operator +
+ ", errorThreshold='" + errorThreshold + '\'' +
+ ", warnThreshold='" + warnThreshold + '\'' +
+ ", onLeakPeriod=" + onLeakPeriod +
+ ", value='" + value + '\'' +
+ '}';
+ }
+ }
+
+ /**
+ * Quality Gate condition operator.
+ */
+ public enum Operator {
+ EQUALS, NOT_EQUALS, GREATER_THAN, LESS_THAN
+ }
+
+ /**
+ * Quality gate condition evaluation status.
+ */
+ public enum EvaluationStatus {
+ /**
+ * No measure found or measure had no value. The condition has not been evaluated and therefor ignored in
+ * the computation of the Quality Gate status.
+ */
+ NO_VALUE,
+ /**
+ * Condition evaluated as OK, neither error nor warning thresholds have been reached.
+ */
+ OK,
+ /**
+ * Condition evaluated as WARN, only warning thresholds has been reached.
+ */
+ WARN,
+ /**
+ * Condition evaluated as ERROR, error thresholds has been reached (and most likely warning thresholds too).
+ */
+ ERROR
+ }
+}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/webhook/WebhookModule.java b/server/sonar-server/src/main/java/org/sonar/server/webhook/WebhookModule.java
index 6a5da0c597b..a33610c3dbb 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/webhook/WebhookModule.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/webhook/WebhookModule.java
@@ -27,6 +27,7 @@ public class WebhookModule extends Module {
add(
WebhookCallerImpl.class,
WebhookDeliveryStorage.class,
- WebHooksImpl.class);
+ WebHooksImpl.class,
+ WebhookPayloadFactoryImpl.class);
}
}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/computation/task/projectanalysis/webhook/WebhookPayloadFactory.java b/server/sonar-server/src/main/java/org/sonar/server/webhook/WebhookPayloadFactory.java
index 9f5891ae924..bd5b6a8138c 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/computation/task/projectanalysis/webhook/WebhookPayloadFactory.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/webhook/WebhookPayloadFactory.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.task.projectanalysis.webhook;
-
-import org.sonar.api.ce.posttask.PostProjectAnalysisTask;
-import org.sonar.server.webhook.WebhookPayload;
+package org.sonar.server.webhook;
@FunctionalInterface
public interface WebhookPayloadFactory {
- WebhookPayload create(PostProjectAnalysisTask.ProjectAnalysis analysis);
+ WebhookPayload create(ProjectAnalysis analysis);
}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/computation/task/projectanalysis/webhook/WebhookPayloadFactoryImpl.java b/server/sonar-server/src/main/java/org/sonar/server/webhook/WebhookPayloadFactoryImpl.java
index 4990f9b3fee..613a0e985f1 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/computation/task/projectanalysis/webhook/WebhookPayloadFactoryImpl.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/webhook/WebhookPayloadFactoryImpl.java
@@ -17,25 +17,18 @@
* 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.task.projectanalysis.webhook;
+package org.sonar.server.webhook;
import java.io.StringWriter;
import java.io.UnsupportedEncodingException;
import java.io.Writer;
import java.net.URLEncoder;
import java.util.Date;
-import javax.annotation.Nullable;
+import java.util.Map;
import org.sonar.api.ce.ComputeEngineSide;
-import org.sonar.api.ce.posttask.Branch;
-import org.sonar.api.ce.posttask.CeTask;
-import org.sonar.api.ce.posttask.PostProjectAnalysisTask;
-import org.sonar.api.ce.posttask.Project;
-import org.sonar.api.ce.posttask.QualityGate;
-import org.sonar.api.ce.posttask.ScannerContext;
import org.sonar.api.platform.Server;
import org.sonar.api.utils.System2;
import org.sonar.api.utils.text.JsonWriter;
-import org.sonar.server.webhook.WebhookPayload;
import static java.lang.String.format;
import static org.sonar.core.config.WebhookProperties.ANALYSIS_PROPERTY_PREFIX;
@@ -52,7 +45,7 @@ public class WebhookPayloadFactoryImpl implements WebhookPayloadFactory {
}
@Override
- public WebhookPayload create(PostProjectAnalysisTask.ProjectAnalysis analysis) {
+ public WebhookPayload create(ProjectAnalysis analysis) {
Writer string = new StringWriter();
try (JsonWriter writer = JsonWriter.of(string)) {
writer.beginObject();
@@ -61,8 +54,8 @@ public class WebhookPayloadFactoryImpl implements WebhookPayloadFactory {
writeDates(writer, analysis, system2);
writeProject(analysis, writer, analysis.getProject());
analysis.getBranch().ifPresent(b -> writeBranch(writer, analysis.getProject(), b));
- writeQualityGate(writer, analysis.getQualityGate());
- writeAnalysisProperties(writer, analysis.getScannerContext());
+ analysis.getQualityGate().ifPresent(qualityGate -> writeQualityGate(writer, qualityGate));
+ writeAnalysisProperties(writer, analysis.getProperties());
writer.endObject().close();
return new WebhookPayload(analysis.getProject().getKey(), string.toString());
}
@@ -72,12 +65,12 @@ public class WebhookPayloadFactoryImpl implements WebhookPayloadFactory {
writer.prop("serverUrl", server.getPublicRootUrl());
}
- private static void writeDates(JsonWriter writer, PostProjectAnalysisTask.ProjectAnalysis analysis, System2 system2) {
+ private static void writeDates(JsonWriter writer, ProjectAnalysis analysis, System2 system2) {
analysis.getAnalysisDate().ifPresent(date -> writer.propDateTime("analysedAt", date));
writer.propDateTime("changedAt", analysis.getAnalysisDate().orElseGet(() -> new Date(system2.now())));
}
- private void writeProject(PostProjectAnalysisTask.ProjectAnalysis analysis, JsonWriter writer, Project project) {
+ private void writeProject(ProjectAnalysis analysis, JsonWriter writer, Project project) {
writer
.name("project")
.beginObject()
@@ -87,11 +80,11 @@ public class WebhookPayloadFactoryImpl implements WebhookPayloadFactory {
.endObject();
}
- private static void writeAnalysisProperties(JsonWriter writer, ScannerContext scannerContext) {
+ private static void writeAnalysisProperties(JsonWriter writer, Map<String, String> properties) {
writer
.name("properties")
.beginObject();
- scannerContext.getProperties().entrySet()
+ properties.entrySet()
.stream()
.filter(prop -> prop.getKey().startsWith(ANALYSIS_PROPERTY_PREFIX))
.forEach(prop -> writer.prop(prop.getKey(), prop.getValue()));
@@ -132,34 +125,30 @@ public class WebhookPayloadFactoryImpl implements WebhookPayloadFactory {
}
}
- private static void writeQualityGate(JsonWriter writer, @Nullable QualityGate gate) {
- if (gate != null) {
+ private static void writeQualityGate(JsonWriter writer, QualityGate gate) {
+ writer
+ .name("qualityGate")
+ .beginObject()
+ .prop("name", gate.getName())
+ .prop("status", gate.getStatus().toString())
+ .name("conditions")
+ .beginArray();
+ for (QualityGate.Condition condition : gate.getConditions()) {
writer
- .name("qualityGate")
.beginObject()
- .prop("name", gate.getName())
- .prop("status", gate.getStatus().toString())
- .name("conditions")
- .beginArray();
- for (QualityGate.Condition condition : gate.getConditions()) {
- writer
- .beginObject()
- .prop("metric", condition.getMetricKey())
- .prop("operator", condition.getOperator().name());
- if (condition.getStatus() != QualityGate.EvaluationStatus.NO_VALUE) {
- writer.prop("value", condition.getValue());
- }
- writer
- .prop("status", condition.getStatus().name())
- .prop("onLeakPeriod", condition.isOnLeakPeriod())
- .prop("errorThreshold", condition.getErrorThreshold())
- .prop("warningThreshold", condition.getWarningThreshold())
- .endObject();
- }
+ .prop("metric", condition.getMetricKey())
+ .prop("operator", condition.getOperator().name());
+ condition.getValue().ifPresent(t -> writer.prop("value", t));
writer
- .endArray()
+ .prop("status", condition.getStatus().name())
+ .prop("onLeakPeriod", condition.isOnLeakPeriod())
+ .prop("errorThreshold", condition.getErrorThreshold().orElse(null))
+ .prop("warningThreshold", condition.getWarningThreshold().orElse(null))
.endObject();
}
+ writer
+ .endArray()
+ .endObject();
}
private static String encode(String toEncode) {
diff --git a/server/sonar-server/src/test/java/org/sonar/server/computation/task/projectanalysis/webhook/WebhookPostTaskTest.java b/server/sonar-server/src/test/java/org/sonar/server/computation/task/projectanalysis/webhook/WebhookPostTaskTest.java
index 600d7c11335..d398ae6f561 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/computation/task/projectanalysis/webhook/WebhookPostTaskTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/computation/task/projectanalysis/webhook/WebhookPostTaskTest.java
@@ -19,6 +19,7 @@
*/
package org.sonar.server.computation.task.projectanalysis.webhook;
+import java.util.Collections;
import java.util.Date;
import java.util.Random;
import java.util.function.Supplier;
@@ -26,13 +27,14 @@ import org.junit.Before;
import org.junit.Test;
import org.mockito.ArgumentCaptor;
import org.sonar.api.ce.posttask.CeTask;
-import org.sonar.api.ce.posttask.PostProjectAnalysisTask;
import org.sonar.api.ce.posttask.PostProjectAnalysisTaskTester;
import org.sonar.api.ce.posttask.Project;
import org.sonar.api.config.Configuration;
import org.sonar.server.computation.task.projectanalysis.component.ConfigurationRepository;
+import org.sonar.server.webhook.ProjectAnalysis;
import org.sonar.server.webhook.WebHooks;
import org.sonar.server.webhook.WebhookPayload;
+import org.sonar.server.webhook.WebhookPayloadFactory;
import static org.apache.commons.lang.RandomStringUtils.randomAlphanumeric;
import static org.assertj.core.api.Assertions.assertThat;
@@ -57,7 +59,7 @@ public class WebhookPostTaskTest {
@Before
public void wireMocks() throws Exception {
- when(payloadFactory.create(any(PostProjectAnalysisTask.ProjectAnalysis.class))).thenReturn(webhookPayload);
+ when(payloadFactory.create(any(ProjectAnalysis.class))).thenReturn(webhookPayload);
when(configurationRepository.getConfiguration()).thenReturn(configuration);
}
@@ -72,9 +74,10 @@ public class WebhookPostTaskTest {
.setStatus(CeTask.Status.values()[new Random().nextInt(CeTask.Status.values().length)])
.setId(randomAlphanumeric(6))
.build();
+ Date date = new Date();
- PostProjectAnalysisTask.ProjectAnalysis projectAnalysis = PostProjectAnalysisTaskTester.of(underTest)
- .at(new Date())
+ PostProjectAnalysisTaskTester.of(underTest)
+ .at(date)
.withCeTask(ceTask)
.withProject(project)
.withScannerContext(newScannerContextBuilder().build())
@@ -85,7 +88,14 @@ public class WebhookPostTaskTest {
assertThat(supplierCaptor.getValue().get()).isSameAs(webhookPayload);
- verify(payloadFactory).create(same(projectAnalysis));
+ verify(payloadFactory).create(new ProjectAnalysis(
+ new org.sonar.server.webhook.CeTask(ceTask.getId(),
+ org.sonar.server.webhook.CeTask.Status.valueOf(ceTask.getStatus().name())),
+ new org.sonar.server.webhook.Project(project.getUuid(), project.getKey(), project.getName()),
+ null,
+ null,
+ date.getTime(),
+ Collections.emptyMap()));
}
}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/webhook/BranchTest.java b/server/sonar-server/src/test/java/org/sonar/server/webhook/BranchTest.java
new file mode 100644
index 00000000000..af63c4ddfd7
--- /dev/null
+++ b/server/sonar-server/src/test/java/org/sonar/server/webhook/BranchTest.java
@@ -0,0 +1,55 @@
+/*
+ * 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.sonar.server.webhook;
+
+import java.util.Random;
+import org.junit.Rule;
+import org.junit.Test;
+import org.junit.rules.ExpectedException;
+
+import static org.assertj.core.api.Assertions.assertThat;
+
+public class BranchTest {
+ @Rule
+ public ExpectedException expectedException = ExpectedException.none();
+
+ private Branch underTest = new Branch(true, "b", Branch.Type.SHORT);
+
+ @Test
+ public void constructor_throws_NPE_if_type_is_null() {
+ expectedException.expect(NullPointerException.class);
+ expectedException.expectMessage("type can't be null");
+
+ new Branch(new Random().nextBoolean(), "s", null);
+ }
+
+ @Test
+ public void verify_getters() {
+ assertThat(underTest.isMain()).isTrue();
+ assertThat(underTest.getName()).contains("b");
+ assertThat(underTest.getType()).isEqualTo(Branch.Type.SHORT);
+
+
+ Branch underTestWithNull = new Branch(false, null, Branch.Type.LONG);
+ assertThat(underTestWithNull.isMain()).isFalse();
+ assertThat(underTestWithNull.getName()).isEmpty();
+ assertThat(underTestWithNull.getType()).isEqualTo(Branch.Type.LONG);
+ }
+}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/webhook/CeTaskTest.java b/server/sonar-server/src/test/java/org/sonar/server/webhook/CeTaskTest.java
new file mode 100644
index 00000000000..502bde02926
--- /dev/null
+++ b/server/sonar-server/src/test/java/org/sonar/server/webhook/CeTaskTest.java
@@ -0,0 +1,55 @@
+/*
+ * 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.sonar.server.webhook;
+
+import org.junit.Rule;
+import org.junit.Test;
+import org.junit.rules.ExpectedException;
+
+import static org.assertj.core.api.Assertions.assertThat;
+
+public class CeTaskTest {
+ @Rule
+ public ExpectedException expectedException = ExpectedException.none();
+
+ private CeTask underTest = new CeTask("A", CeTask.Status.SUCCESS);
+
+ @Test
+ public void constructor_throws_NPE_if_id_is_null() {
+ expectedException.expect(NullPointerException.class);
+ expectedException.expectMessage("id can't be null");
+
+ new CeTask(null, CeTask.Status.SUCCESS);
+ }
+
+ @Test
+ public void constructor_throws_NPE_if_status_is_null() {
+ expectedException.expect(NullPointerException.class);
+ expectedException.expectMessage("status can't be null");
+
+ new CeTask("B", null);
+ }
+
+ @Test
+ public void verify_getters() {
+ assertThat(underTest.getId()).isEqualTo("A");
+ assertThat(underTest.getStatus()).isEqualTo(CeTask.Status.SUCCESS);
+ }
+}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/webhook/ProjectAnalysisTest.java b/server/sonar-server/src/test/java/org/sonar/server/webhook/ProjectAnalysisTest.java
new file mode 100644
index 00000000000..1cad78c7c02
--- /dev/null
+++ b/server/sonar-server/src/test/java/org/sonar/server/webhook/ProjectAnalysisTest.java
@@ -0,0 +1,150 @@
+/*
+ * 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.sonar.server.webhook;
+
+import com.google.common.collect.ImmutableMap;
+import java.util.Date;
+import java.util.Map;
+import org.junit.Rule;
+import org.junit.Test;
+import org.junit.rules.ExpectedException;
+
+import static java.util.Collections.emptyMap;
+import static java.util.Collections.emptySet;
+import static org.assertj.core.api.Assertions.assertThat;
+
+public class ProjectAnalysisTest {
+ @Rule
+ public ExpectedException expectedException = ExpectedException.none();
+
+ private final CeTask ceTask = new CeTask("id", CeTask.Status.SUCCESS);
+ private final Project project = new Project("uuid", "key", "name");
+ private final Branch branch = new Branch(true, "name", Branch.Type.SHORT);
+ private final QualityGate qualityGate = new QualityGate("id", "name", QualityGate.Status.WARN, emptySet());
+ private final Map<String, String> properties = ImmutableMap.of("a", "b");
+ private ProjectAnalysis underTest = new ProjectAnalysis(ceTask,
+ project,
+ branch,
+ qualityGate,
+ 1L,
+ properties);
+
+ @Test
+ public void constructor_throws_NPE_if_ceTask_is_null() {
+
+ expectedException.expect(NullPointerException.class);
+ expectedException.expectMessage("ceTask can't be null");
+
+ new ProjectAnalysis(null,
+ project,
+ branch,
+ qualityGate,
+ 1L,
+ emptyMap());
+ }
+
+ @Test
+ public void constructor_throws_NPE_if_project_is_null() {
+
+ expectedException.expect(NullPointerException.class);
+ expectedException.expectMessage("project can't be null");
+
+ new ProjectAnalysis(ceTask,
+ null,
+ branch,
+ qualityGate,
+ 1L,
+ emptyMap());
+ }
+
+ @Test
+ public void constructor_throws_NPE_if_properties_is_null() {
+
+ expectedException.expect(NullPointerException.class);
+ expectedException.expectMessage("properties can't be null");
+
+ new ProjectAnalysis(ceTask,
+ project,
+ branch,
+ qualityGate,
+ 1L,
+ null);
+ }
+
+ @Test
+ public void verify_getters() {
+ assertThat(underTest.getCeTask()).isSameAs(ceTask);
+ assertThat(underTest.getProject()).isSameAs(project);
+ assertThat(underTest.getBranch().get()).isSameAs(branch);
+ assertThat(underTest.getQualityGate().get()).isSameAs(qualityGate);
+ assertThat(underTest.getAnalysisDate()).contains(new Date(1L));
+ assertThat(underTest.getProperties()).isEqualTo(properties);
+
+ ProjectAnalysis underTestWithNulls = new ProjectAnalysis(ceTask, project, null, null, null, emptyMap());
+ assertThat(underTestWithNulls.getBranch()).isEmpty();
+ assertThat(underTestWithNulls.getQualityGate()).isEmpty();
+ assertThat(underTestWithNulls.getAnalysisDate()).isEmpty();
+ assertThat(underTestWithNulls.getProperties()).isEmpty();
+ }
+
+ @Test
+ public void defines_equals_based_on_all_fields() {
+ assertThat(underTest).isEqualTo(underTest);
+ assertThat(underTest).isEqualTo(new ProjectAnalysis(ceTask, project, branch, qualityGate, 1L, properties));
+ assertThat(underTest).isNotEqualTo(null);
+ assertThat(underTest).isNotEqualTo(new Object());
+ assertThat(underTest).isNotEqualTo(new ProjectAnalysis(new CeTask("2", CeTask.Status.SUCCESS), project, branch, qualityGate, 1L, properties));
+ assertThat(underTest).isNotEqualTo(new ProjectAnalysis(ceTask, new Project("A", "B", "C"), branch, qualityGate, 1L, properties));
+ assertThat(underTest).isNotEqualTo(new ProjectAnalysis(ceTask, new Project("A", "B", "C"), branch, qualityGate, 1L, properties));
+ assertThat(underTest).isNotEqualTo(new ProjectAnalysis(ceTask, project, null, qualityGate, 1L, properties));
+ assertThat(underTest).isNotEqualTo(new ProjectAnalysis(ceTask, project, new Branch(false, "B", Branch.Type.SHORT), qualityGate, 1L, properties));
+ assertThat(underTest).isNotEqualTo(new ProjectAnalysis(ceTask, project, branch, null, 1L, properties));
+ assertThat(underTest).isNotEqualTo(new ProjectAnalysis(ceTask, project, branch, new QualityGate("A", "B", QualityGate.Status.WARN, emptySet()), 1L, properties));
+ assertThat(underTest).isNotEqualTo(new ProjectAnalysis(ceTask, project, branch, qualityGate, null, properties));
+ assertThat(underTest).isNotEqualTo(new ProjectAnalysis(ceTask, project, branch, qualityGate, 2L, properties));
+ assertThat(underTest).isNotEqualTo(new ProjectAnalysis(ceTask, project, branch, qualityGate, 1L, emptyMap()));
+ assertThat(underTest).isNotEqualTo(new ProjectAnalysis(ceTask, project, branch, qualityGate, 1L, ImmutableMap.of("A", "B")));
+ }
+
+ @Test
+ public void defines_hashcode_based_on_all_fields() {
+ assertThat(underTest.hashCode()).isEqualTo(underTest.hashCode());
+ assertThat(underTest.hashCode()).isEqualTo(new ProjectAnalysis(ceTask, project, branch, qualityGate, 1L, properties).hashCode());
+ assertThat(underTest.hashCode()).isNotEqualTo(new Object().hashCode());
+ assertThat(underTest.hashCode()).isNotEqualTo(new ProjectAnalysis(new CeTask("2", CeTask.Status.SUCCESS), project, branch, qualityGate, 1L, properties).hashCode());
+ assertThat(underTest.hashCode()).isNotEqualTo(new ProjectAnalysis(ceTask, new Project("A", "B", "C"), branch, qualityGate, 1L, properties).hashCode());
+ assertThat(underTest.hashCode()).isNotEqualTo(new ProjectAnalysis(ceTask, new Project("A", "B", "C"), branch, qualityGate, 1L, properties).hashCode());
+ assertThat(underTest.hashCode()).isNotEqualTo(new ProjectAnalysis(ceTask, project, null, qualityGate, 1L, properties).hashCode());
+ assertThat(underTest.hashCode()).isNotEqualTo(new ProjectAnalysis(ceTask, project, new Branch(false, "B", Branch.Type.SHORT), qualityGate, 1L, properties).hashCode());
+ assertThat(underTest.hashCode()).isNotEqualTo(new ProjectAnalysis(ceTask, project, branch, null, 1L, properties).hashCode());
+ assertThat(underTest.hashCode())
+ .isNotEqualTo(new ProjectAnalysis(ceTask, project, branch, new QualityGate("A", "B", QualityGate.Status.WARN, emptySet()), 1L, properties).hashCode());
+ assertThat(underTest.hashCode()).isNotEqualTo(new ProjectAnalysis(ceTask, project, branch, qualityGate, null, properties).hashCode());
+ assertThat(underTest.hashCode()).isNotEqualTo(new ProjectAnalysis(ceTask, project, branch, qualityGate, 2L, properties).hashCode());
+ assertThat(underTest.hashCode()).isNotEqualTo(new ProjectAnalysis(ceTask, project, branch, qualityGate, 1L, emptyMap()).hashCode());
+ assertThat(underTest.hashCode()).isNotEqualTo(new ProjectAnalysis(ceTask, project, branch, qualityGate, 1L, ImmutableMap.of("B", "C")).hashCode());
+ }
+
+ @Test
+ public void verify_toString() {
+ assertThat(underTest.toString()).isEqualTo(
+ "ProjectAnalysis{ceTask=CeTask{id='id', status=SUCCESS}, project=Project{uuid='uuid', key='key', name='name'}, branch=Branch{main=true, name='name', type=SHORT}, qualityGate=QualityGate{id='id', name='name', status=WARN, conditions=[]}, date=1, properties={a=b}}");
+ }
+}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/webhook/ProjectTest.java b/server/sonar-server/src/test/java/org/sonar/server/webhook/ProjectTest.java
new file mode 100644
index 00000000000..adf6036c1fd
--- /dev/null
+++ b/server/sonar-server/src/test/java/org/sonar/server/webhook/ProjectTest.java
@@ -0,0 +1,64 @@
+/*
+ * 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.sonar.server.webhook;
+
+import org.junit.Rule;
+import org.junit.Test;
+import org.junit.rules.ExpectedException;
+
+import static org.assertj.core.api.Assertions.assertThat;
+
+public class ProjectTest {
+ @Rule
+ public ExpectedException expectedException = ExpectedException.none();
+
+ private Project underTest = new Project("a", "b", "c");
+
+ @Test
+ public void constructor_throws_NPE_if_uuid_is_null() {
+ expectedException.expect(NullPointerException.class);
+ expectedException.expectMessage("uuid can't be null");
+
+ new Project(null, "b", "c");
+ }
+
+ @Test
+ public void constructor_throws_NPE_if_key_is_null() {
+ expectedException.expect(NullPointerException.class);
+ expectedException.expectMessage("key can't be null");
+
+ new Project("a", null, "c");
+ }
+
+ @Test
+ public void constructor_throws_NPE_if_name_is_null() {
+ expectedException.expect(NullPointerException.class);
+ expectedException.expectMessage("name can't be null");
+
+ new Project("a", "b", null);
+ }
+
+ @Test
+ public void verify_getters() {
+ assertThat(underTest.getUuid()).isEqualTo("a");
+ assertThat(underTest.getKey()).isEqualTo("b");
+ assertThat(underTest.getName()).isEqualTo("c");
+ }
+}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/webhook/QualityGateTest.java b/server/sonar-server/src/test/java/org/sonar/server/webhook/QualityGateTest.java
new file mode 100644
index 00000000000..b75c4673764
--- /dev/null
+++ b/server/sonar-server/src/test/java/org/sonar/server/webhook/QualityGateTest.java
@@ -0,0 +1,122 @@
+/*
+ * 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.sonar.server.webhook;
+
+import java.util.Random;
+import org.junit.Rule;
+import org.junit.Test;
+import org.junit.rules.ExpectedException;
+
+import static java.util.Collections.singleton;
+import static org.assertj.core.api.Assertions.assertThat;
+
+public class QualityGateTest {
+ @Rule
+ public ExpectedException expectedException = ExpectedException.none();
+
+ private final Random random = new Random();
+ private boolean onLeak = random.nextBoolean();
+ private QualityGate.EvaluationStatus randomEvaluationStatus = QualityGate.EvaluationStatus.values()[random.nextInt(QualityGate.EvaluationStatus.values().length)];
+ private QualityGate.Condition condition = new QualityGate.Condition(
+ randomEvaluationStatus, "k", QualityGate.Operator.GREATER_THAN, "l", "m", onLeak, "val");
+ private QualityGate.Status randomStatus = QualityGate.Status.values()[random.nextInt(QualityGate.Status.values().length)];
+ private QualityGate underTest = new QualityGate("i", "j", randomStatus, singleton(condition));
+
+ @Test
+ public void constructor_throws_NPE_if_id_is_null() {
+ expectedException.expect(NullPointerException.class);
+ expectedException.expectMessage("id can't be null");
+
+ new QualityGate(null, "j", QualityGate.Status.WARN, singleton(condition));
+ }
+
+ @Test
+ public void constructor_throws_NPE_if_name_is_null() {
+ expectedException.expect(NullPointerException.class);
+ expectedException.expectMessage("name can't be null");
+
+ new QualityGate("i", null, QualityGate.Status.WARN, singleton(condition));
+ }
+
+ @Test
+ public void constructor_throws_NPE_if_status_is_null() {
+ expectedException.expect(NullPointerException.class);
+ expectedException.expectMessage("status can't be null");
+
+ new QualityGate("i", "j", null, singleton(condition));
+ }
+
+ @Test
+ public void constructor_throws_NPE_if_conditions_is_null() {
+ expectedException.expect(NullPointerException.class);
+ expectedException.expectMessage("conditions can't be null");
+
+ new QualityGate("i", "j", QualityGate.Status.WARN, null);
+ }
+
+ @Test
+ public void condition_constructor_throws_NPE_if_evaluation_status_is_null() {
+ expectedException.expect(NullPointerException.class);
+ expectedException.expectMessage("status can't be null");
+
+ new QualityGate.Condition(null, "k", QualityGate.Operator.GREATER_THAN, "l", "m", onLeak, "val");
+ }
+
+ @Test
+ public void condition_constructor_throws_NPE_if_metricKey_is_null() {
+ expectedException.expect(NullPointerException.class);
+ expectedException.expectMessage("metricKey can't be null");
+
+ new QualityGate.Condition(randomEvaluationStatus, null, QualityGate.Operator.GREATER_THAN, "l", "m", onLeak, "val");
+ }
+
+ @Test
+ public void condition_constructor_throws_NPE_if_operator_status_is_null() {
+ expectedException.expect(NullPointerException.class);
+ expectedException.expectMessage("operator can't be null");
+
+ new QualityGate.Condition(randomEvaluationStatus, "k", null, "l", "m", onLeak, "val");
+ }
+
+ @Test
+ public void verify_getters() {
+ assertThat(underTest.getId()).isEqualTo("i");
+ assertThat(underTest.getName()).isEqualTo("j");
+ assertThat(underTest.getStatus()).isEqualTo(randomStatus);
+ assertThat(underTest.getConditions()).isEqualTo(singleton(condition));
+ }
+
+ @Test
+ public void verify_condition_getters() {
+ assertThat(condition.getStatus()).isEqualTo(randomEvaluationStatus);
+ assertThat(condition.getMetricKey()).isEqualTo("k");
+ assertThat(condition.getErrorThreshold()).contains("l");
+ assertThat(condition.getWarningThreshold()).contains("m");
+ assertThat(condition.isOnLeakPeriod()).isEqualTo(onLeak);
+ assertThat(condition.getValue()).contains("val");
+
+ QualityGate.Condition conditionWithNulls = new QualityGate.Condition(
+ randomEvaluationStatus, "k", QualityGate.Operator.GREATER_THAN, null, null, onLeak, null);
+
+ assertThat(conditionWithNulls.getErrorThreshold()).isEmpty();
+ assertThat(conditionWithNulls.getWarningThreshold()).isEmpty();
+ assertThat(conditionWithNulls.getValue()).isEmpty();
+ }
+}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/webhook/TestWebhookPayloadFactory.java b/server/sonar-server/src/test/java/org/sonar/server/webhook/TestWebhookPayloadFactory.java
deleted file mode 100644
index 603b8ee3861..00000000000
--- a/server/sonar-server/src/test/java/org/sonar/server/webhook/TestWebhookPayloadFactory.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.sonar.server.webhook;
-
-import org.sonar.api.ce.posttask.PostProjectAnalysisTask;
-import org.sonar.server.computation.task.projectanalysis.webhook.WebhookPayloadFactory;
-
-public class TestWebhookPayloadFactory implements WebhookPayloadFactory {
-
- private static final String FAKE_JSON = "{\"payload\": true}";
-
- @Override
- public WebhookPayload create(PostProjectAnalysisTask.ProjectAnalysis analysis) {
- return new WebhookPayload(analysis.getProject().getKey(), FAKE_JSON);
- }
-}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/webhook/WebhookModuleTest.java b/server/sonar-server/src/test/java/org/sonar/server/webhook/WebhookModuleTest.java
index 5a80a0b7459..6c29df54a4d 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/webhook/WebhookModuleTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/webhook/WebhookModuleTest.java
@@ -40,6 +40,6 @@ public class WebhookModuleTest {
underTest.configure(container);
- assertThat(container.size()).isEqualTo(3 + COMPONENTS_IN_EMPTY_COMPONENT_CONTAINER);
+ assertThat(container.size()).isEqualTo(4 + COMPONENTS_IN_EMPTY_COMPONENT_CONTAINER);
}
}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/computation/task/projectanalysis/webhook/WebhookPayloadFactoryImplTest.java b/server/sonar-server/src/test/java/org/sonar/server/webhook/WebhookPayloadFactoryImplTest.java
index e4c409f2019..56aea60cf48 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/computation/task/projectanalysis/webhook/WebhookPayloadFactoryImplTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/webhook/WebhookPayloadFactoryImplTest.java
@@ -17,35 +17,22 @@
* 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.task.projectanalysis.webhook;
+package org.sonar.server.webhook;
import com.google.common.collect.ImmutableMap;
-import java.util.Date;
import java.util.Map;
-import java.util.Optional;
import javax.annotation.Nullable;
import org.junit.Before;
import org.junit.Test;
-import org.sonar.api.ce.posttask.Branch;
-import org.sonar.api.ce.posttask.CeTask;
-import org.sonar.api.ce.posttask.PostProjectAnalysisTask;
-import org.sonar.api.ce.posttask.Project;
-import org.sonar.api.ce.posttask.QualityGate;
-import org.sonar.api.ce.posttask.ScannerContext;
import org.sonar.api.platform.Server;
import org.sonar.api.utils.System2;
-import org.sonar.server.computation.task.projectanalysis.api.posttask.BranchImpl;
-import org.sonar.server.webhook.WebhookPayload;
import static java.util.Collections.emptyMap;
+import static java.util.Collections.emptySet;
+import static java.util.Collections.singleton;
import static org.assertj.core.api.Assertions.assertThat;
import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.when;
-import static org.sonar.api.ce.posttask.PostProjectAnalysisTaskTester.newCeTaskBuilder;
-import static org.sonar.api.ce.posttask.PostProjectAnalysisTaskTester.newConditionBuilder;
-import static org.sonar.api.ce.posttask.PostProjectAnalysisTaskTester.newProjectBuilder;
-import static org.sonar.api.ce.posttask.PostProjectAnalysisTaskTester.newQualityGateBuilder;
-import static org.sonar.api.ce.posttask.PostProjectAnalysisTaskTester.newScannerContextBuilder;
import static org.sonar.test.JsonAssert.assertJson;
public class WebhookPayloadFactoryImplTest {
@@ -64,23 +51,11 @@ public class WebhookPayloadFactoryImplTest {
@Test
public void create_payload_for_successful_analysis() {
- CeTask task = newCeTaskBuilder()
- .setStatus(CeTask.Status.SUCCESS)
- .setId("#1")
- .build();
- QualityGate gate = newQualityGateBuilder()
- .setId("G1")
- .setName("Gate One")
- .setStatus(QualityGate.Status.WARN)
- .add(newConditionBuilder()
- .setMetricKey("coverage")
- .setOperator(QualityGate.Operator.GREATER_THAN)
- .setOnLeakPeriod(true)
- .setWarningThreshold("75.0")
- .setErrorThreshold("70.0")
- .build(QualityGate.EvaluationStatus.WARN, "74.0"))
- .build();
- PostProjectAnalysisTask.ProjectAnalysis analysis = newAnalysis(task, gate, null, 1_500_000_000_000L, emptyMap());
+ CeTask task = new CeTask("#1", CeTask.Status.SUCCESS);
+ QualityGate gate = new QualityGate("G1", "Gate One", QualityGate.Status.WARN, singleton(new QualityGate.Condition(
+ QualityGate.EvaluationStatus.WARN,
+ "coverage", QualityGate.Operator.GREATER_THAN, "70.0", "75.0", true, "74.0")));
+ ProjectAnalysis analysis = newAnalysis(task, gate, null, 1_500_000_000_000L, emptyMap());
WebhookPayload payload = underTest.create(analysis);
assertThat(payload.getProjectKey()).isEqualTo(PROJECT_KEY);
@@ -118,22 +93,10 @@ public class WebhookPayloadFactoryImplTest {
@Test
public void create_payload_with_gate_conditions_without_value() {
- CeTask task = newCeTaskBuilder()
- .setStatus(CeTask.Status.SUCCESS)
- .setId("#1")
- .build();
- QualityGate gate = newQualityGateBuilder()
- .setId("G1")
- .setName("Gate One")
- .setStatus(QualityGate.Status.WARN)
- .add(newConditionBuilder()
- .setMetricKey("coverage")
- .setOperator(QualityGate.Operator.GREATER_THAN)
- .setWarningThreshold("75.0")
- .setErrorThreshold("70.0")
- .buildNoValue())
- .build();
- PostProjectAnalysisTask.ProjectAnalysis analysis = newAnalysis(task, gate, null, 1_500_000_000_000L, emptyMap());
+ CeTask task = new CeTask("#1", CeTask.Status.SUCCESS);
+ QualityGate gate = new QualityGate("G1", "Gate One", QualityGate.Status.WARN, singleton(
+ new QualityGate.Condition(QualityGate.EvaluationStatus.NO_VALUE, "coverage", QualityGate.Operator.GREATER_THAN, "70.0", "75.0", false, null)));
+ ProjectAnalysis analysis = newAnalysis(task, gate, null, 1_500_000_000_000L, emptyMap());
WebhookPayload payload = underTest.create(analysis);
assertThat(payload.getProjectKey()).isEqualTo(PROJECT_KEY);
@@ -168,21 +131,14 @@ public class WebhookPayloadFactoryImplTest {
@Test
public void create_payload_with_analysis_properties() {
- CeTask task = newCeTaskBuilder()
- .setStatus(CeTask.Status.SUCCESS)
- .setId("#1")
- .build();
- QualityGate gate = newQualityGateBuilder()
- .setId("G1")
- .setName("Gate One")
- .setStatus(QualityGate.Status.WARN)
- .build();
+ CeTask task = new CeTask("#1", CeTask.Status.SUCCESS);
+ QualityGate gate = new QualityGate("G1", "Gate One", QualityGate.Status.WARN, emptySet());
Map<String, String> scannerProperties = ImmutableMap.of(
"sonar.analysis.revision", "ab45d24",
"sonar.analysis.buildNumber", "B123",
"not.prefixed.with.sonar.analysis", "should be ignored",
"ignored", "should be ignored too");
- PostProjectAnalysisTask.ProjectAnalysis analysis = newAnalysis(task, gate, null, 1_500_000_000_000L, scannerProperties);
+ ProjectAnalysis analysis = newAnalysis(task, gate, null, 1_500_000_000_000L, scannerProperties);
WebhookPayload payload = underTest.create(analysis);
assertJson(payload.getJson())
@@ -215,8 +171,8 @@ public class WebhookPayloadFactoryImplTest {
@Test
public void create_payload_for_failed_analysis() {
- CeTask ceTask = newCeTaskBuilder().setStatus(CeTask.Status.FAILED).setId("#1").build();
- PostProjectAnalysisTask.ProjectAnalysis analysis = newAnalysis(ceTask, null, null, 1_500_000_000_000L, emptyMap());
+ CeTask ceTask = new CeTask("#1", CeTask.Status.FAILED);
+ ProjectAnalysis analysis = newAnalysis(ceTask, null, null, 1_500_000_000_000L, emptyMap());
WebhookPayload payload = underTest.create(analysis);
@@ -239,8 +195,8 @@ public class WebhookPayloadFactoryImplTest {
@Test
public void create_payload_for_no_analysis_date() {
- CeTask ceTask = newCeTaskBuilder().setStatus(CeTask.Status.FAILED).setId("#1").build();
- PostProjectAnalysisTask.ProjectAnalysis analysis = newAnalysis(ceTask, null, null, null, emptyMap());
+ CeTask ceTask = new CeTask("#1", CeTask.Status.FAILED);
+ ProjectAnalysis analysis = newAnalysis(ceTask, null, null, null, emptyMap());
WebhookPayload payload = underTest.create(analysis);
@@ -262,11 +218,8 @@ public class WebhookPayloadFactoryImplTest {
@Test
public void create_payload_on_short_branch() {
- CeTask task = newCeTaskBuilder()
- .setStatus(CeTask.Status.SUCCESS)
- .setId("#1")
- .build();
- PostProjectAnalysisTask.ProjectAnalysis analysis = newAnalysis(task, null, new BranchImpl(false, "feature/foo", Branch.Type.SHORT), 1_500_000_000_000L, emptyMap());
+ CeTask task = new CeTask("#1", CeTask.Status.SUCCESS);
+ ProjectAnalysis analysis = newAnalysis(task, null, new Branch(false, "feature/foo", Branch.Type.SHORT), 1_500_000_000_000L, emptyMap());
WebhookPayload payload = underTest.create(analysis);
assertJson(payload.getJson())
@@ -282,11 +235,8 @@ public class WebhookPayloadFactoryImplTest {
@Test
public void create_payload_on_long_branch() {
- CeTask task = newCeTaskBuilder()
- .setStatus(CeTask.Status.SUCCESS)
- .setId("#1")
- .build();
- PostProjectAnalysisTask.ProjectAnalysis analysis = newAnalysis(task, null, new BranchImpl(false, "feature/foo", Branch.Type.LONG), 1_500_000_000_000L, emptyMap());
+ CeTask task = new CeTask("#1", CeTask.Status.SUCCESS);
+ ProjectAnalysis analysis = newAnalysis(task, null, new Branch(false, "feature/foo", Branch.Type.LONG), 1_500_000_000_000L, emptyMap());
WebhookPayload payload = underTest.create(analysis);
assertJson(payload.getJson())
@@ -302,11 +252,8 @@ public class WebhookPayloadFactoryImplTest {
@Test
public void create_payload_on_main_branch_without_name() {
- CeTask task = newCeTaskBuilder()
- .setStatus(CeTask.Status.SUCCESS)
- .setId("#1")
- .build();
- PostProjectAnalysisTask.ProjectAnalysis analysis = newAnalysis(task, null, new BranchImpl(true, null, Branch.Type.LONG), 1_500_000_000_000L, emptyMap());
+ CeTask task = new CeTask("#1", CeTask.Status.SUCCESS);
+ ProjectAnalysis analysis = newAnalysis(task, null, new Branch(true, null, Branch.Type.LONG), 1_500_000_000_000L, emptyMap());
WebhookPayload payload = underTest.create(analysis);
assertJson(payload.getJson())
@@ -319,50 +266,9 @@ public class WebhookPayloadFactoryImplTest {
"}");
}
- private static PostProjectAnalysisTask.ProjectAnalysis newAnalysis(CeTask task, @Nullable QualityGate gate,
+ private static ProjectAnalysis newAnalysis(CeTask task, @Nullable QualityGate gate,
@Nullable Branch branch, @Nullable Long analysisDate, Map<String, String> scannerProperties) {
- return new PostProjectAnalysisTask.ProjectAnalysis() {
- @Override
- public CeTask getCeTask() {
- return task;
- }
-
- @Override
- public Project getProject() {
- return newProjectBuilder()
- .setUuid("P1_UUID")
- .setKey(PROJECT_KEY)
- .setName("Project One")
- .build();
- }
-
- @Override
- public Optional<Branch> getBranch() {
- return Optional.ofNullable(branch);
- }
-
- @Override
- public QualityGate getQualityGate() {
- return gate;
- }
-
- @Override
- public Date getDate() {
- throw new UnsupportedOperationException();
- }
-
- @Override
- public Optional<Date> getAnalysisDate() {
- return Optional.ofNullable(analysisDate).map(Date::new);
- }
-
- @Override
- public ScannerContext getScannerContext() {
- return newScannerContextBuilder()
- .addProperties(scannerProperties)
- .build();
- }
- };
+ return new ProjectAnalysis(task, new Project("P1_UUID", PROJECT_KEY, "Project One"), branch, gate, analysisDate, scannerProperties);
}
}