
io.github.theangrydev.yatspecfluent.FluentTest Maven / Gradle / Ivy
/*
* Copyright 2016 Liam Williams .
*
* This file is part of yatspec-fluent.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package io.github.theangrydev.yatspecfluent;
import com.googlecode.yatspec.state.givenwhenthen.TestState;
import com.googlecode.yatspec.state.givenwhenthen.WithTestState;
import org.junit.Rule;
import org.junit.rules.TestWatcher;
import org.junit.runner.Description;
import static java.lang.String.format;
/**
* Use this as the base class for your acceptance tests.
*
* @param The type of test result produced by the {@link When}
*/
@SuppressWarnings("PMD.TooManyMethods") // Maybe I will refactor this one day...
public abstract class FluentTest implements WithTestState, WriteOnlyTestItems {
private final TestState state = new TestState();
private Stage stage = Stage.GIVEN;
private TestResult testResult;
private enum Stage {
GIVEN,
WHEN,
THEN
}
@Rule
public TestWatcher makeSureThenIsUsed = new TestWatcher() {
@Override
protected void succeeded(Description description) {
if (stage != Stage.THEN) {
throw new IllegalStateException("Each test needs at least a 'when' and a 'then'");
}
}
};
/**
* You should aim to never access the state directly, but you might need to (e.g. global shared state).
* Call {@link #addToGivens(String, Object)} when possible or make use of the {@link WriteOnlyTestItems} interface.
* Call {@link #addToCapturedInputsAndOutputs(String, Object)} when possible or make use of the {@link WriteOnlyTestItems} interface.
*/
@Override
public TestState testState() {
return state;
}
/**
* Same as {@link #given(Given)}.
*
* Prime the given immediately.
*
* @param given The first given in the acceptance test, which should be built up inside the brackets
*/
public void and(Given given) {
given(given);
}
/**
* Prime the given immediately.
*
* @param given The first given in the acceptance test, which should be built up inside the brackets
*/
public void given(Given given) {
if (stage != Stage.GIVEN) {
throw new IllegalStateException("The 'given' steps must be specified before the 'when' and 'then' steps");
}
stage = Stage.GIVEN;
given.prime();
}
/**
* Invoke the system under test and store the {@link TestResult} ready for the assertions.
*
* @param when The system under test, which should be built up inside the brackets
* @param The type of {@link When}
*/
public > void when(T when) {
if (stage != Stage.GIVEN) {
throw new IllegalStateException("There should only be one 'when', after the 'given' and before the 'then'");
}
testResult = when.execute();
if (testResult == null) {
throw new IllegalStateException(format("'%s' test result was null", when));
}
stage = Stage.WHEN;
}
/**
* Adapt the 'when' to a 'given'. This is a common pattern when e.g. calling an endpoint that changes some state in the database.
* This is the equivalent of {@link #given(Given)}.
*
* @param when The 'when' to adapt to a 'given'
*/
public void given(When when) {
given((Given) when::execute);
}
/**
* Same as {@link #given(When)}.
*
* Adapt the 'when' to a 'given'. This is a common pattern when e.g. calling an endpoint that changes some state in the database.
* This is the equivalent of {@link #given(Given)}.
*
* @param when The 'when' to adapt to a 'given'
*/
public void and(When when) {
given(when);
}
/**
* Perform an assertion. Assertions should be chained outside the brackets.
*
* @param thenAssertion A {@link ThenAssertion} that will produce a {@link Then} given the stored {@link TestResult}
* @param The type of fluent assertions that will be performed
* @return The fluent assertions instance
*/
public Then then(ThenAssertion thenAssertion) {
checkThenIsPossible();
return thenAssertion.then(testResult);
}
/**
* Same as {@link #then(ThenAssertion)}.
*
* Perform an assertion. Assertions should be chained outside the brackets.
*
* @param thenAssertion A {@link ThenAssertion} that will produce a {@link Then} given the stored {@link TestResult}
* @param The type of fluent assertions that will be performed
* @return The fluent assertions instance
*/
public Then and(ThenAssertion thenAssertion) {
return then(thenAssertion);
}
/**
* Same as {@link #then(ThenVerification)}.
*
* Perform a verification, which should be built up inside the brackets.
*
* @param thenVerification A {@link ThenVerification}, which should be built up inside the brackets
*/
public void and(ThenVerification thenVerification) {
then(thenVerification);
}
/**
* Perform a verification, which should be built up inside the brackets.
*
* @param thenVerification A {@link ThenVerification}, which should be built up inside the brackets
*/
public void then(ThenVerification thenVerification) {
checkThenIsPossible();
thenVerification.verify(testResult);
}
@Override
public void addToGivens(String key, Object instance) {
testState().interestingGivens.add(key, instance);
}
@Override
public void addToCapturedInputsAndOutputs(String key, Object instance) {
testState().capturedInputAndOutputs.add(key, instance);
}
private void checkThenIsPossible() {
if (stage.compareTo(Stage.WHEN) < 0) {
throw new IllegalStateException("The 'then' steps should be after the 'when'");
}
stage = Stage.THEN;
}
}