io.github.theangrydev.yatspecfluent.Verification Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of yatspec-fluent Show documentation
Show all versions of yatspec-fluent Show documentation
A plugin for the yatspec acceptance test library used to write acceptance tests in a fluent way, reusing givens, whens and thens.
The newest version!
/*
* 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 java.lang.reflect.Field;
import java.util.ArrayList;
import java.util.List;
import static java.lang.String.format;
import static java.lang.reflect.Modifier.isFinal;
import static java.util.Arrays.stream;
@SuppressWarnings("PMD.TooManyMethods") // Splitting this up further would be too artificial
class Verification {
private enum Stage {
GIVEN,
WHEN,
THEN
}
private Stage stage = Stage.GIVEN;
private final List usedGivens = new ArrayList<>();
private final List> usedThenVerifications = new ArrayList<>();
private final List> usedThenAssertions = new ArrayList<>();
public void checkGivenIsAllowed(Given given) {
if (stage != Stage.GIVEN) {
throw new IllegalStateException("The 'given' steps must be specified before the 'when' and 'then' steps");
}
checkMutableInstanceHasNotAlreadyBeenUsed(given, usedGivens);
}
public void recordGiven(Given given) {
stage = Stage.GIVEN;
usedGivens.add(given);
}
public void checkWhenIsAllowed() {
if (stage != Stage.GIVEN) {
throw new IllegalStateException("There should only be one 'when', after the 'given' and before the 'then'");
}
}
public void recordWhen(When when, TestResult testResult) {
if (testResult == null) {
throw new IllegalStateException(format("'%s' test result was null", when));
}
stage = Stage.WHEN;
}
public void checkThenVerificationIsAllowed(ThenVerification thenVerification) {
checkThenIsAllowed();
checkMutableInstanceHasNotAlreadyBeenUsed(thenVerification, usedThenVerifications);
}
public void recordThenVerification(ThenVerification thenVerification) {
usedThenVerifications.add(thenVerification);
}
public void checkThenAssertionIsAllowed(ThenAssertion thenAssertion) {
checkThenIsAllowed();
checkMutableInstanceHasNotAlreadyBeenUsed(thenAssertion, usedThenAssertions);
usedThenAssertions.add(thenAssertion);
}
public void checkThenHasBeenUsed() {
if (stage != Stage.THEN) {
throw new IllegalStateException("Each test needs at least a 'when' and a 'then'");
}
}
private void checkThenIsAllowed() {
if (stage.compareTo(Stage.WHEN) < 0) {
throw new IllegalStateException("The 'then' steps should be after the 'when'");
}
stage = Stage.THEN;
}
private boolean appearsToBeMutable(Class> aClass) {
return stream(aClass.getDeclaredFields())
.mapToInt(Field::getModifiers)
.anyMatch(modifiers -> !isFinal(modifiers));
}
private void checkMutableInstanceHasNotAlreadyBeenUsed(T instance, List usedInstances) {
if (appearsToBeMutable(instance.getClass()) && usedInstances.contains(instance)) {
throw new IllegalStateException(format("This '%s' instance has been used once already. To avoid accidentally sharing state, use a new instance.", instance.getClass().getSimpleName()));
}
}
}