From a2c598a58b6f3accc04e1064a8e71d5c3817cd9f Mon Sep 17 00:00:00 2001 From: Vincent Latombe Date: Mon, 25 Nov 2024 14:03:14 +0100 Subject: [PATCH 1/3] Provide Hamcrest matchers as alternatives to built-in JenkinsRule Plays nicely with Awaitability as well as in combination with other matchers. --- src/main/java/jenkins/test/RunMatchers.java | 119 ++++++++++++++++++ .../org/jvnet/hudson/test/JenkinsRule.java | 6 + .../java/jenkins/test/RunMatchersTest.java | 71 +++++++++++ 3 files changed, 196 insertions(+) create mode 100644 src/main/java/jenkins/test/RunMatchers.java create mode 100644 src/test/java/jenkins/test/RunMatchersTest.java diff --git a/src/main/java/jenkins/test/RunMatchers.java b/src/main/java/jenkins/test/RunMatchers.java new file mode 100644 index 000000000..55ad11c62 --- /dev/null +++ b/src/main/java/jenkins/test/RunMatchers.java @@ -0,0 +1,119 @@ +/* + * The MIT License + * + * Copyright 2024 CloudBees, Inc. + * + * Permission is hereby granted, free of charge, to any person obtaining a copy + * of this software and associated documentation files (the "Software"), to deal + * in the Software without restriction, including without limitation the rights + * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + * copies of the Software, and to permit persons to whom the Software is + * furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in + * all copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + * THE SOFTWARE. + */ +package jenkins.test; + +import edu.umd.cs.findbugs.annotations.NonNull; +import hudson.model.Result; +import hudson.model.Run; +import java.io.IOException; +import org.hamcrest.Description; +import org.hamcrest.Matcher; +import org.hamcrest.TypeSafeMatcher; +import org.jvnet.hudson.test.JenkinsRule; + +/** + * Matchers for {@link Run} objects. + */ +public final class RunMatchers { + private RunMatchers() {} + + /** + * Creates a matcher checking whether a build is successful. + */ + public static Matcher> isSuccessful() { + return new RunResultMatcher(Result.SUCCESS); + } + + /** + * Creates a matcher checking whether a build has a specific outcome. + */ + public static Matcher> hasStatus(Result result) { + return new RunResultMatcher(result); + } + + /** + * Creates a matcher checking whether build logs contain a specific message. + * @param message the expected message + */ + public static Matcher> logContains(String message) { + return new RunLogMatcher(message); + } + + private static class RunResultMatcher extends TypeSafeMatcher> { + @NonNull + private final Result expectedResult; + + public RunResultMatcher(@NonNull Result expectedResult) { + this.expectedResult = expectedResult; + } + + @Override + public void describeTo(Description description) { + description.appendText("a build with result " + expectedResult); + } + + @Override + protected boolean matchesSafely(Run run) { + return run.getResult() == expectedResult; + } + + @Override + protected void describeMismatchSafely(Run item, Description mismatchDescription) { + mismatchDescription.appendText("was ").appendValue(item.getResult()); + } + } + + private static class RunLogMatcher extends TypeSafeMatcher> { + @NonNull + private final String message; + + private RunLogMatcher(@NonNull String message) { + this.message = message; + } + + @Override + protected boolean matchesSafely(Run run) { + try { + return JenkinsRule.getLog(run).contains(message); + } catch (IOException x) { + return false; + } + } + + @Override + protected void describeMismatchSafely(Run item, Description mismatchDescription) { + mismatchDescription.appendText("was \n"); + try { + mismatchDescription.appendText(JenkinsRule.getLog(item)); + } catch (IOException e) { + mismatchDescription.appendText(""); + } + } + + @Override + public void describeTo(Description description) { + description.appendText("log containing ").appendValue(message); + } + } +} diff --git a/src/main/java/org/jvnet/hudson/test/JenkinsRule.java b/src/main/java/org/jvnet/hudson/test/JenkinsRule.java index 996cee769..6698ab04a 100644 --- a/src/main/java/org/jvnet/hudson/test/JenkinsRule.java +++ b/src/main/java/org/jvnet/hudson/test/JenkinsRule.java @@ -1509,6 +1509,8 @@ public C configRoundtrip(C cloud) throws Exception { /** * Asserts that the outcome of the build is a specific outcome. + *

+ * Consider {@link jenkins.test.RunMatchers#hasStatus(Result)} as an alternative. */ public R assertBuildStatus(Result status, R r) throws Exception { if (status == r.getResult()) { @@ -1583,6 +1585,8 @@ public FreeStyleBuild buildAndAssertSuccess(@NonNull FreeStyleProject job) throw /** * Asserts that the console output of the build contains the given substring. + *

+ * Consider {@link jenkins.test.RunMatchers#logContains(String)} as an alternative. */ public void assertLogContains(String substring, Run run) throws IOException { assertThat(getLog(run), containsString(substring)); @@ -1590,6 +1594,8 @@ public void assertLogContains(String substring, Run run) throws IOException { /** * Asserts that the console output of the build does not contain the given substring. + *

+ * Consider {@link org.hamcrest.Matchers#not} and {@link jenkins.test.RunMatchers#logContains(String)} as an alternative. */ public void assertLogNotContains(String substring, Run run) throws IOException { assertThat(getLog(run), not(containsString(substring))); diff --git a/src/test/java/jenkins/test/RunMatchersTest.java b/src/test/java/jenkins/test/RunMatchersTest.java new file mode 100644 index 000000000..b46ab54ce --- /dev/null +++ b/src/test/java/jenkins/test/RunMatchersTest.java @@ -0,0 +1,71 @@ +/* + * The MIT License + * + * Copyright 2024 CloudBees, Inc. + * + * Permission is hereby granted, free of charge, to any person obtaining a copy + * of this software and associated documentation files (the "Software"), to deal + * in the Software without restriction, including without limitation the rights + * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + * copies of the Software, and to permit persons to whom the Software is + * furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in + * all copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + * THE SOFTWARE. + */ +package jenkins.test; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.allOf; +import static org.hamcrest.Matchers.not; +import static jenkins.test.RunMatchers.logContains; +import static jenkins.test.RunMatchers.hasStatus; +import static jenkins.test.RunMatchers.isSuccessful; + +import hudson.Functions; +import hudson.model.Result; +import hudson.tasks.BatchFile; +import hudson.tasks.Shell; +import org.junit.Rule; +import org.junit.Test; +import org.jvnet.hudson.test.FailureBuilder; +import org.jvnet.hudson.test.JenkinsRule; +import org.jvnet.hudson.test.SleepBuilder; + +public class RunMatchersTest { + @Rule + public JenkinsRule j = new JenkinsRule(); + + @Test + public void buildSuccessful() throws Exception { + var p = j.createFreeStyleProject(); + p.getBuildersList().add(new SleepBuilder(1000)); + var b = p.scheduleBuild2(0).waitForStart(); + assertThat(j.waitForCompletion(b), isSuccessful()); + } + + @Test + public void buildFailure() throws Exception { + var p = j.createFreeStyleProject(); + p.getBuildersList().add(new FailureBuilder()); + var b = p.scheduleBuild2(0).waitForStart(); + assertThat(j.waitForCompletion(b), hasStatus(Result.FAILURE)); + } + + @Test + public void assertThatLogContains() throws Exception { + var p = j.createFreeStyleProject(); + p.getBuildersList().add(Functions.isWindows() ? new BatchFile("echo hello") : new Shell("echo hello")); + var b = p.scheduleBuild2(0).get(); + System.out.println(b.getDisplayName() + " completed"); + assertThat(b, allOf(logContains("echo hello"), not(logContains("echo bye")))); + } +} From b46b3de0df0c1601da599620c671d4fc2d017ab1 Mon Sep 17 00:00:00 2001 From: Vincent Latombe Date: Tue, 26 Nov 2024 09:01:17 +0100 Subject: [PATCH 2/3] Add completed() matcher to check whether a build has completed. --- src/main/java/jenkins/test/RunMatchers.java | 19 +++++++++++++++++++ .../java/jenkins/test/RunMatchersTest.java | 3 ++- 2 files changed, 21 insertions(+), 1 deletion(-) diff --git a/src/main/java/jenkins/test/RunMatchers.java b/src/main/java/jenkins/test/RunMatchers.java index 55ad11c62..9b2793d33 100644 --- a/src/main/java/jenkins/test/RunMatchers.java +++ b/src/main/java/jenkins/test/RunMatchers.java @@ -60,6 +60,13 @@ public static Matcher> logContains(String message) { return new RunLogMatcher(message); } + /** + * Creates a matcher checking whether a build has completed. + */ + public static Matcher> completed() { + return new CompletedRunMatcher(); + } + private static class RunResultMatcher extends TypeSafeMatcher> { @NonNull private final Result expectedResult; @@ -116,4 +123,16 @@ public void describeTo(Description description) { description.appendText("log containing ").appendValue(message); } } + + private static class CompletedRunMatcher extends TypeSafeMatcher> { + @Override + protected boolean matchesSafely(Run run) { + return !run.isLogUpdated(); + } + + @Override + public void describeTo(Description description) { + description.appendText("a completed build"); + } + } } diff --git a/src/test/java/jenkins/test/RunMatchersTest.java b/src/test/java/jenkins/test/RunMatchersTest.java index b46ab54ce..30c00ae32 100644 --- a/src/test/java/jenkins/test/RunMatchersTest.java +++ b/src/test/java/jenkins/test/RunMatchersTest.java @@ -23,6 +23,7 @@ */ package jenkins.test; +import static jenkins.test.RunMatchers.completed; import static org.hamcrest.MatcherAssert.assertThat; import static org.hamcrest.Matchers.allOf; import static org.hamcrest.Matchers.not; @@ -49,7 +50,7 @@ public void buildSuccessful() throws Exception { var p = j.createFreeStyleProject(); p.getBuildersList().add(new SleepBuilder(1000)); var b = p.scheduleBuild2(0).waitForStart(); - assertThat(j.waitForCompletion(b), isSuccessful()); + assertThat(j.waitForCompletion(b), allOf(completed(), isSuccessful())); } @Test From 8bd3f6c11ecdb6fbc8ebccc8cfd9b7de9e8b0567 Mon Sep 17 00:00:00 2001 From: Vincent Latombe Date: Wed, 27 Nov 2024 09:36:40 +0100 Subject: [PATCH 3/3] Mention matcher alternative when using Awaitibility --- src/main/java/org/jvnet/hudson/test/JenkinsRule.java | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/src/main/java/org/jvnet/hudson/test/JenkinsRule.java b/src/main/java/org/jvnet/hudson/test/JenkinsRule.java index 6698ab04a..7a55f4f5b 100644 --- a/src/main/java/org/jvnet/hudson/test/JenkinsRule.java +++ b/src/main/java/org/jvnet/hudson/test/JenkinsRule.java @@ -1619,6 +1619,9 @@ public static String getLog(Run run) throws IOException { /** * Waits for a build to complete. * Useful in conjunction with {@link BuildWatcher}. + *

+ * As an alternative, if using Awaitibility, you can use {@code await().until(() -> r, RunMatchers.completed());} + * * @return the same build, once done * @since 1.607 */ @@ -1635,7 +1638,6 @@ public > R waitForCompletion(R r) throws InterruptedException } } } - // Could be using com.jayway.awaitility:awaitility but it seems like overkill here. while (r.isLogUpdated()) { Thread.sleep(100); }