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

nl.hsac.fitnesse.fixture.fit.ServiceAndCheckMapColumnFixture Maven / Gradle / Ivy

There is a newer version: 5.3.18
Show newest version
package nl.hsac.fitnesse.fixture.fit;

import nl.hsac.fitnesse.fixture.Environment;

import java.util.Map;

/**
 * Base class for fixtures that call a service and then another (possibly with retry).
 * The time to wait (in ms) between calls to the check service is configured with the first argument to the fixture
 * (i.e. column in the header row, next to the fixture class name),
 * the maximum number of times to call the check service is configured using the second.
 *
 * If no check is required, just don't call getRawCheckResponse(), since a call to that method will be the trigger
 * to invoke the check.
 * @param  class expected as response to original call.
 * @param  class expected as response to check call (if any).
 */
public abstract class ServiceAndCheckMapColumnFixture extends MapColumnFixture {
    private final Class responseClass;
    private final Class checkResponseClass;
    private final Environment env = Environment.getInstance();
    private Response response;
    private CheckResponse checkResponse;
    private boolean checkCalled;
    private int tryCount;
    private int waitTime;
    private int maxTries;
    private long responseTime;
    private long checkTime;

    public ServiceAndCheckMapColumnFixture(Class aResponseClass, Class aCheckResponseClass) {
        checkResponseClass = aCheckResponseClass;
        responseClass = aResponseClass;
    }

    public Class getResponseClass() {
        return responseClass;
    }

    public Class getCheckResponseClass() {
        return checkResponseClass;
    }

    @Override
    public void execute() {
        super.execute();
        long startTime = currentTimeMillis();
        try {
            setRawResponse(callService());
        } finally {
            responseTime = currentTimeMillis() - startTime;
        }
    }

    /**
     * Calls service to obtain response.
     * @return response from service.
     */
    protected abstract Response callService();

    @Override
    public void reset() {
        super.reset();
        response = getEnvironment().createInstance(responseClass);
        checkResponse = getEnvironment().createInstance(checkResponseClass);
        checkCalled = false;
        tryCount = 0;
        responseTime = 0;
        checkTime = 0;
    }

    /**
     * @return raw response
     */
    public Response getRawResponse() {
        return response;
    }

    /**
     * @param aResponse the response to set (only for unit tests)
     */
    public void setRawResponse(Response aResponse) {
        response = aResponse;
    }

    /**
     * Trigger to actually call check.
     * @return raw check response
     */
    public CheckResponse getRawCheckResponse() {
        if (!checkCalled) {
            setupMaxTries();
            setupWaitTime();

            addResultsToValuesForCheck(getCurrentRowValues());
            long startTime = currentTimeMillis();
            try {
                executeCheckWithRetry();
            } finally {
                checkTime = currentTimeMillis() - startTime;
            }
        }
        return checkResponse;
    }

    protected void setupMaxTries() {
        maxTries = parseIntArg(0, 1);
    }

    protected void setupWaitTime() {
        waitTime = parseIntArg(1, 0);
    }

    private long currentTimeMillis() {
        return System.currentTimeMillis();
    }

    /**
     * @param aCheckResponse the checkResponse to set
     */
    public void setRawCheckResponse(CheckResponse aCheckResponse) {
        checkCalled = true;
        checkResponse = aCheckResponse;
    }

    /**
     * Allows subclasses to add more values before check template is processed.
     * This allows, for instance, results of original call to be available
     * to the checkTemplate.
     * @param values map to put extra key/values in.
     */
    protected void addResultsToValuesForCheck(Map values) {
    }

    /**
     * Calls {@link #executeCheckCall()} one or more times (until either
     * the call returned no mismatches or {@link #tryCount} == {@link #maxTries}).
     * Before each call to {@link #executeCheckCall()} {@link #executeCheckCall()}
     * will be called.
     */
    protected void executeCheckWithRetry() {
        tryCount++;
        executeCheckCall();
        if (tryCount < maxTries && isRetryDesirable()) {
            // not changed yet, try again
            executeCheckWithRetry();
        } else {
            checkExecutionCompleted();
        }
    }

    /**
     * Will be invoked when check call is completed (either done or
     * maxRetries reached).
     */
    protected void checkExecutionCompleted() {
    }

    /**
     * Determine whether response from last call to check service indicate another call is needed, or not.
     * The last check call's response is available using {@link #getRawCheckResponse()}
     * @return does check response warrant retry, or not.
     */
    protected boolean isRetryDesirable() {
        return false;
    }

    /**
     * Performs check, after waiting for change to complete.
     */
    protected void executeCheckCall() {
        waitForChangeToComplete();
        setRawCheckResponse(callCheckService());
    }

    /**
     * To be called from executeCheckCall() before making actual call.
     */
    protected void waitForChangeToComplete() {
        if (waitTime > 0) {
            try {
                Thread.sleep(waitTime);
            } catch (InterruptedException e) {
                throw new RuntimeException("Waiting for change to complete interrupted", e);
            }
        }
    }

    /**
     * Calls check service.
     * @return response from check service.
     */
    protected abstract CheckResponse callCheckService();

    /**
     * @return the environment instance
     */
    protected Environment getEnvironment() {
        return env;
    }

    public int getWaitTime() {
        return waitTime;
    }

    public void setWaitTime(int aWaitTime) {
        waitTime = aWaitTime;
    }

    public int getMaxTries() {
        return maxTries;
    }

    public void setMaxTries(int aMaxTries) {
        maxTries = aMaxTries;
    }

    /**
     * @return number of times the check template was called.
     */
    public int tryCount() {
        return tryCount;
    }

    /**
     * @return response time in ms for call to service.
     */
    public long responseTime() {
        return responseTime;
    }

    /**
     * @return response time in ms for call to checkService.
     */
    public long checkTime() {
        return checkTime;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy