All Downloads are FREE. Search and download functionalities are using the official Maven repository.

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; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy