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

com.facebook.presto.verifier.framework.AbstractVerification Maven / Gradle / Ivy

The newest version!
/*
 * 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 com.facebook.presto.verifier.framework;

import com.facebook.airlift.log.Logger;
import com.facebook.presto.jdbc.QueryStats;
import com.facebook.presto.sql.SqlFormatter;
import com.facebook.presto.sql.tree.Statement;
import com.facebook.presto.verifier.event.DeterminismAnalysisDetails;
import com.facebook.presto.verifier.event.QueryInfo;
import com.facebook.presto.verifier.event.VerifierQueryEvent;
import com.facebook.presto.verifier.event.VerifierQueryEvent.EventStatus;
import com.facebook.presto.verifier.prestoaction.PrestoAction;
import com.facebook.presto.verifier.prestoaction.PrestoAction.ResultSetConverter;
import com.facebook.presto.verifier.prestoaction.QueryAction;
import com.facebook.presto.verifier.prestoaction.QueryActionStats;
import com.facebook.presto.verifier.prestoaction.QueryActions;
import com.facebook.presto.verifier.prestoaction.SqlExceptionClassifier;
import com.facebook.presto.verifier.source.SnapshotQueryConsumer;
import com.google.common.collect.ImmutableList;
import com.google.common.util.concurrent.ListenableFuture;
import com.google.common.util.concurrent.ListeningExecutorService;

import java.util.List;
import java.util.Map;
import java.util.Optional;

import static com.facebook.airlift.concurrent.MoreFutures.getFutureValue;
import static com.facebook.presto.spi.StandardErrorCode.EXCEEDED_TIME_LIMIT;
import static com.facebook.presto.verifier.event.VerifierQueryEvent.EventStatus.FAILED;
import static com.facebook.presto.verifier.event.VerifierQueryEvent.EventStatus.FAILED_RESOLVED;
import static com.facebook.presto.verifier.event.VerifierQueryEvent.EventStatus.SKIPPED;
import static com.facebook.presto.verifier.event.VerifierQueryEvent.EventStatus.SUCCEEDED;
import static com.facebook.presto.verifier.framework.ClusterType.CONTROL;
import static com.facebook.presto.verifier.framework.ClusterType.TEST;
import static com.facebook.presto.verifier.framework.DataVerificationUtil.teardownSafely;
import static com.facebook.presto.verifier.framework.QueryStage.CONTROL_MAIN;
import static com.facebook.presto.verifier.framework.QueryStage.CONTROL_SETUP;
import static com.facebook.presto.verifier.framework.QueryStage.TEST_MAIN;
import static com.facebook.presto.verifier.framework.QueryStage.TEST_SETUP;
import static com.facebook.presto.verifier.framework.QueryState.FAILED_TO_SETUP;
import static com.facebook.presto.verifier.framework.QueryState.NOT_RUN;
import static com.facebook.presto.verifier.framework.QueryState.TIMED_OUT;
import static com.facebook.presto.verifier.framework.SkippedReason.CONTROL_QUERY_FAILED;
import static com.facebook.presto.verifier.framework.SkippedReason.CONTROL_QUERY_TIMED_OUT;
import static com.facebook.presto.verifier.framework.SkippedReason.CONTROL_SETUP_QUERY_FAILED;
import static com.facebook.presto.verifier.framework.SkippedReason.FAILED_BEFORE_CONTROL_QUERY;
import static com.facebook.presto.verifier.framework.SkippedReason.NON_DETERMINISTIC;
import static com.facebook.presto.verifier.framework.SkippedReason.VERIFIER_INTERNAL_ERROR;
import static com.facebook.presto.verifier.framework.VerifierConfig.QUERY_BANK_MODE;
import static com.facebook.presto.verifier.framework.VerifierUtil.callAndConsume;
import static com.facebook.presto.verifier.framework.VerifierUtil.runAndConsume;
import static com.facebook.presto.verifier.prestoaction.QueryActionStats.EMPTY_STATS;
import static com.google.common.base.Preconditions.checkArgument;
import static com.google.common.base.Preconditions.checkState;
import static com.google.common.base.Strings.nullToEmpty;
import static com.google.common.base.Throwables.getStackTraceAsString;
import static com.google.common.collect.ImmutableList.toImmutableList;
import static com.google.common.util.concurrent.Futures.immediateFuture;
import static java.lang.String.format;
import static java.util.Objects.requireNonNull;

public abstract class AbstractVerification
        implements Verification
{
    private static final Logger LOG = Logger.get(AbstractVerification.class);
    private static final String INTERNAL_ERROR = "VERIFIER_INTERNAL_ERROR";
    private static final String SNAPSHOT_DOES_NOT_EXIST = "SNAPSHOT_DOES_NOT_EXIST";

    private final QueryActions queryActions;
    private final SourceQuery sourceQuery;
    private final SqlExceptionClassifier exceptionClassifier;
    private final VerificationContext verificationContext;
    private final Optional> mainQueryResultSetConverter;
    private final ListeningExecutorService executor;

    private final String testId;
    private final boolean smartTeardown;
    private final int verificationResubmissionLimit;

    private final boolean setupOnMainClusters;
    private final boolean teardownOnMainClusters;
    protected final boolean skipControl;
    private final boolean skipChecksum;
    protected final String runningMode;
    protected final boolean saveSnapshot;
    protected final boolean isExplain;
    private final boolean concurrentControlAndTest;
    protected final SnapshotQueryConsumer snapshotQueryConsumer;
    protected final Map snapshotQueries;

    public AbstractVerification(
            QueryActions queryActions,
            SourceQuery sourceQuery,
            SqlExceptionClassifier exceptionClassifier,
            VerificationContext verificationContext,
            Optional> mainQueryResultSetConverter,
            VerifierConfig verifierConfig,
            ListeningExecutorService executor,
            SnapshotQueryConsumer snapshotQueryConsumer,
            Map snapshotQueries)
    {
        this.queryActions = requireNonNull(queryActions, "queryActions is null");
        this.sourceQuery = requireNonNull(sourceQuery, "sourceQuery is null");
        this.exceptionClassifier = requireNonNull(exceptionClassifier, "exceptionClassifier is null");
        this.verificationContext = requireNonNull(verificationContext, "verificationContext is null");
        this.mainQueryResultSetConverter = requireNonNull(mainQueryResultSetConverter, "mainQueryResultSetConverter is null");
        this.executor = requireNonNull(executor, "executor is null");
        this.snapshotQueryConsumer = requireNonNull(snapshotQueryConsumer, "snapshotQueryConsumer is null");
        this.snapshotQueries = requireNonNull(snapshotQueries, "snapshotQuerySupplier is null");

        this.testId = requireNonNull(verifierConfig.getTestId(), "testId is null");
        this.smartTeardown = verifierConfig.isSmartTeardown();
        this.verificationResubmissionLimit = verifierConfig.getVerificationResubmissionLimit();
        this.setupOnMainClusters = verifierConfig.isSetupOnMainClusters();
        this.teardownOnMainClusters = verifierConfig.isTeardownOnMainClusters();
        this.skipControl = verifierConfig.isSkipControl();
        this.skipChecksum = verifierConfig.isSkipChecksum();
        this.concurrentControlAndTest = verifierConfig.isConcurrentControlAndTest();
        this.runningMode = verifierConfig.getRunningMode();
        this.saveSnapshot = verifierConfig.isSaveSnapshot();
        this.isExplain = verifierConfig.isExplain();
    }

    protected abstract B getQueryRewrite(ClusterType clusterType);

    protected abstract R verify(
            B control,
            B test,
            Optional> controlQueryResult,
            Optional> testQueryResult,
            ChecksumQueryContext controlChecksumQueryContext,
            ChecksumQueryContext testChecksumQueryContext);

    protected abstract DeterminismAnalysisDetails analyzeDeterminism(B control, R matchResult);

    protected abstract Optional resolveFailure(
            Optional control,
            Optional test,
            QueryContext controlQueryContext,
            Optional matchResult,
            Optional throwable);

    protected void updateQueryInfo(QueryInfo.Builder queryInfo, Optional> queryResult) {}

    protected void updateQueryInfoWithQueryBundle(QueryInfo.Builder queryInfo, Optional queryBundle)
    {
        queryInfo.setQuery(queryBundle.map(B::getQuery).map(AbstractVerification::formatSql))
                .setSetupQueries(queryBundle.map(B::getSetupQueries).map(AbstractVerification::formatSqls))
                .setTeardownQueries(queryBundle.map(B::getTeardownQueries).map(AbstractVerification::formatSqls));
    }

    protected PrestoAction getHelperAction()
    {
        return queryActions.getHelperAction();
    }

    protected boolean isControlEnabled()
    {
        return !skipControl || saveSnapshot;
    }

    @Override
    public SourceQuery getSourceQuery()
    {
        return sourceQuery;
    }

    @Override
    public VerificationContext getVerificationContext()
    {
        return verificationContext;
    }

    @Override
    public VerificationResult run()
    {
        Optional control = Optional.empty();
        Optional test = Optional.empty();
        Optional> controlQueryResult = Optional.empty();
        Optional> testQueryResult = Optional.empty();
        QueryContext controlQueryContext = new QueryContext();
        QueryContext testQueryContext = new QueryContext();
        ChecksumQueryContext controlChecksumQueryContext = new ChecksumQueryContext();
        ChecksumQueryContext testChecksumQueryContext = new ChecksumQueryContext();
        Optional matchResult = Optional.empty();
        Optional determinismAnalysisDetails = Optional.empty();
        boolean controlReuseTable = false;
        boolean testReuseTable = false;

        Optional partialResult = Optional.empty();
        Optional throwable = Optional.empty();

        try {
            // Rewrite queries
            if (isControlEnabled()) {
                control = Optional.of(getQueryRewrite(CONTROL));
                controlReuseTable = control.isPresent() && control.get() instanceof QueryObjectBundle && ((QueryObjectBundle) control.get()).isReuseTable();
                if (controlReuseTable) {
                    controlQueryContext.setState(QueryState.REUSE);
                    controlQueryContext.setMainQueryStats(QueryActionStats.queryIdStats(sourceQuery.getQueryId(CONTROL).get()));
                }
            }

            test = Optional.of(getQueryRewrite(TEST));
            testReuseTable = test.isPresent() && test.get() instanceof QueryObjectBundle && ((QueryObjectBundle) test.get()).isReuseTable();
            if (testReuseTable) {
                testQueryContext.setState(QueryState.REUSE);
                testQueryContext.setMainQueryStats(QueryActionStats.queryIdStats(sourceQuery.getQueryId(TEST).get()));
            }

            // First run setup queries
            if (isControlEnabled() && !controlReuseTable) {
                QueryBundle controlQueryBundle = control.get();
                QueryAction controlSetupAction = setupOnMainClusters ? queryActions.getControlAction() : queryActions.getHelperAction();
                controlQueryBundle.getSetupQueries().forEach(query -> runAndConsume(
                        () -> controlSetupAction.execute(query, CONTROL_SETUP),
                        controlQueryContext::addSetupQuery,
                        controlQueryContext::setException));
            }
            if (!testReuseTable) {
                QueryBundle testQueryBundle = test.get();
                QueryAction testSetupAction = setupOnMainClusters ? queryActions.getTestAction() : queryActions.getHelperAction();
                testQueryBundle.getSetupQueries().forEach(query -> runAndConsume(
                        () -> testSetupAction.execute(query, TEST_SETUP),
                        testQueryContext::addSetupQuery,
                        testQueryContext::setException));
            }

            ListenableFuture>> controlQueryFuture = immediateFuture(Optional.empty());
            ListenableFuture>> testQueryFuture = immediateFuture(Optional.empty());
            // Start control query
            if (isControlEnabled() && !controlReuseTable) {
                QueryBundle controlQueryBundle = control.get();
                controlQueryFuture = executor.submit(() -> runMainQuery(controlQueryBundle.getQuery(), CONTROL, controlQueryContext));
            }
            if (!concurrentControlAndTest) {
                getFutureValue(controlQueryFuture);
            }

            // Run test queries
            if (!testReuseTable) {
                QueryBundle testQueryBundle = test.get();
                testQueryFuture = executor.submit(() -> runMainQuery(testQueryBundle.getQuery(), TEST, testQueryContext));
            }
            controlQueryResult = getFutureValue(controlQueryFuture);

            if (QUERY_BANK_MODE.equals(runningMode) && !saveSnapshot) {
                controlQueryContext.setState(QueryState.SUCCEEDED);
                controlQueryContext.setMainQueryStats(EMPTY_STATS);
            }
            else if (!skipControl || QUERY_BANK_MODE.equals(runningMode)) {
                // saveSnapshot or regular run with skipControl = false
                if (!controlReuseTable) {
                    controlQueryContext.setState(QueryState.SUCCEEDED);
                }
            }
            else {
                controlQueryContext.setState(NOT_RUN);
            }

            testQueryResult = getFutureValue(testQueryFuture);
            if (!testReuseTable) {
                testQueryContext.setState(QueryState.SUCCEEDED);
            }

            // Verify results
            if (QUERY_BANK_MODE.equals(runningMode) && !saveSnapshot && !skipChecksum) {
                // query-bank mode
                control = test;
                matchResult = Optional.of(verify(control.get(), test.get(), controlQueryResult, testQueryResult, controlChecksumQueryContext, testChecksumQueryContext));
            }
            else if ((isControlEnabled()) && !skipChecksum) {
                // regular mode or query-bank with saveSnapshot = true
                matchResult = Optional.of(verify(control.get(), test.get(), controlQueryResult, testQueryResult, controlChecksumQueryContext, testChecksumQueryContext));

                // Determinism analysis
                if (!QUERY_BANK_MODE.equals(runningMode)) {
                    if ((controlReuseTable || testReuseTable) && matchResult.get().isMismatchPossiblyCausedByReuseOutdatedTable() && verificationContext.getResubmissionCount() < verificationResubmissionLimit) {
                        return new VerificationResult(this, true, Optional.empty());
                    }
                    else if (matchResult.get().isMismatchPossiblyCausedByNonDeterminism()) {
                        determinismAnalysisDetails = Optional.of(analyzeDeterminism(control.get(), matchResult.get()));
                    }
                }
            }

            partialResult = Optional.of(concludeVerificationPartial(
                    control,
                    test,
                    controlQueryContext,
                    testQueryContext,
                    matchResult,
                    determinismAnalysisDetails,
                    Optional.empty()));
        }
        catch (Throwable t) {
            if (exceptionClassifier.shouldResubmit(t)
                    && verificationContext.getResubmissionCount() < verificationResubmissionLimit) {
                LOG.info("Error during verification: %s", t);
                return new VerificationResult(this, true, Optional.empty());
            }
            throwable = Optional.of(t);
            partialResult = Optional.of(concludeVerificationPartial(control, test, controlQueryContext, testQueryContext, matchResult, determinismAnalysisDetails, Optional.of(t)));
        }
        finally {
            if (!smartTeardown
                    || !ImmutableList.of(QueryState.SUCCEEDED, QueryState.REUSE).contains(testQueryContext.getState())
                    || (partialResult.isPresent() && partialResult.get().getStatus().equals(SUCCEEDED))) {
                QueryAction controlTeardownAction = teardownOnMainClusters ? queryActions.getControlAction() : queryActions.getHelperAction();
                QueryAction testTeardownAction = teardownOnMainClusters ? queryActions.getTestAction() : queryActions.getHelperAction();
                teardownSafely(controlTeardownAction, control, controlQueryContext::addTeardownQuery);
                teardownSafely(testTeardownAction, test, testQueryContext::addTeardownQuery);
            }
        }

        return concludeVerification(
                partialResult.get(),
                control,
                test,
                controlQueryResult,
                testQueryResult,
                controlQueryContext,
                testQueryContext,
                matchResult,
                controlChecksumQueryContext,
                testChecksumQueryContext,
                determinismAnalysisDetails,
                throwable);
    }

    private Optional> runMainQuery(Statement statement, ClusterType clusterType, QueryContext queryContext)
    {
        checkArgument(clusterType == CONTROL || clusterType == TEST, "Invalid ClusterType: %s", clusterType);

        if (mainQueryResultSetConverter.isPresent()) {
            return Optional.of(callAndConsume(
                    () -> ((PrestoAction) queryActions.getQueryAction(clusterType)).execute(
                            statement,
                            clusterType == CONTROL ? CONTROL_MAIN : TEST_MAIN,
                            mainQueryResultSetConverter.get()),
                    queryContext::setMainQueryStats,
                    queryContext::setException));
        }

        runAndConsume(
                () -> queryActions.getQueryAction(clusterType).execute(statement, clusterType == CONTROL ? CONTROL_MAIN : TEST_MAIN),
                queryContext::setMainQueryStats,
                queryContext::setException);
        return Optional.empty();
    }

    private EventStatus getEventStatus(
            Optional skippedReason,
            Optional resolveMessage,
            Optional matchResult,
            QueryContext controlQueryContext,
            QueryContext testQueryContext)
    {
        if (skippedReason.isPresent() || (matchResult.isPresent() && SNAPSHOT_DOES_NOT_EXIST.equals(matchResult.get().getMatchTypeName()))) {
            return SKIPPED;
        }
        if (resolveMessage.isPresent()) {
            return FAILED_RESOLVED;
        }

        if (skipControl) {
            if (ImmutableList.of(QueryState.SUCCEEDED, QueryState.REUSE).contains(testQueryContext.getState())) {
                return SUCCEEDED;
            }
        }
        else {
            if (skipChecksum) {
                if (ImmutableList.of(QueryState.SUCCEEDED, QueryState.REUSE).contains(testQueryContext.getState()) &&
                        ImmutableList.of(QueryState.SUCCEEDED, QueryState.REUSE).contains(controlQueryContext.getState())) {
                    return SUCCEEDED;
                }
            }
            else {
                if (matchResult.isPresent() && matchResult.get().isMatched()) {
                    return SUCCEEDED;
                }
            }
        }

        return FAILED;
    }

    private PartialVerificationResult concludeVerificationPartial(
            Optional control,
            Optional test,
            QueryContext controlQueryContext,
            QueryContext testQueryContext,
            Optional matchResult,
            Optional determinismAnalysisDetails,
            Optional throwable)
    {
        Optional skippedReason = getSkippedReason(throwable, controlQueryContext.getState(), determinismAnalysisDetails.map(DeterminismAnalysisDetails::getDeterminismAnalysis));
        Optional resolveMessage = resolveFailure(control, test, controlQueryContext, matchResult, throwable);
        EventStatus status = getEventStatus(skippedReason, resolveMessage, matchResult, controlQueryContext, testQueryContext);
        return new PartialVerificationResult(skippedReason, resolveMessage, status);
    }

    private VerificationResult concludeVerification(
            PartialVerificationResult partialResult,
            Optional control,
            Optional test,
            Optional> controlQueryResult,
            Optional> testQueryResult,
            QueryContext controlQueryContext,
            QueryContext testQueryContext,
            Optional matchResult,
            ChecksumQueryContext controlChecksumQueryContext,
            ChecksumQueryContext testChecksumQueryContext,
            Optional determinismAnalysisDetails,
            Optional throwable)
    {
        Optional errorCode = Optional.empty();
        Optional errorMessage = Optional.empty();
        if (partialResult.getStatus() != SUCCEEDED) {
            errorCode = Optional.ofNullable(throwable.map(t -> t instanceof QueryException ? ((QueryException) t).getErrorCodeName() : INTERNAL_ERROR)
                    .orElse(matchResult.map(MatchResult::getMatchTypeName).orElse(null)));
            errorMessage = Optional.of(constructErrorMessage(throwable, matchResult, controlQueryContext.getState(), testQueryContext.getState()));
        }

        VerifierQueryEvent event = new VerifierQueryEvent(
                sourceQuery.getSuite(),
                testId,
                sourceQuery.getName(),
                partialResult.getStatus(),
                matchResult.map(MatchResult::getDataType),
                matchResult.map(MatchResult::getMatchTypeName),
                partialResult.getSkippedReason(),
                determinismAnalysisDetails,
                partialResult.getResolveMessage(),
                skipControl ?
                        Optional.empty() :
                        Optional.of(buildQueryInfo(
                                sourceQuery.getControlConfiguration(),
                                sourceQuery.getQuery(CONTROL),
                                controlChecksumQueryContext,
                                control,
                                controlQueryContext,
                                controlQueryResult)),
                buildQueryInfo(
                        sourceQuery.getTestConfiguration(),
                        sourceQuery.getQuery(TEST),
                        testChecksumQueryContext,
                        test,
                        testQueryContext,
                        testQueryResult),
                errorCode,
                errorMessage,
                throwable.filter(QueryException.class::isInstance)
                        .map(QueryException.class::cast)
                        .map(QueryException::toQueryFailure),
                verificationContext.getQueryFailures(),
                verificationContext.getResubmissionCount());
        return new VerificationResult(this, false, Optional.of(event));
    }

    private QueryInfo buildQueryInfo(
            QueryConfiguration configuration,
            String originalQuery,
            ChecksumQueryContext checksumQueryContext,
            Optional queryBundle,
            QueryContext queryContext,
            Optional> queryResult)
    {
        QueryInfo.Builder queryInfo = QueryInfo.builder(configuration.getCatalog(), configuration.getSchema(), originalQuery, configuration.getSessionProperties())
                .setSetupQueryIds(queryContext.getSetupQueryIds())
                .setTeardownQueryIds(queryContext.getTeardownQueryIds())
                .setChecksumQueryId(checksumQueryContext.getChecksumQueryId())
                .setChecksumQuery(checksumQueryContext.getChecksumQuery())
                .setPartitionChecksumQueryId(checksumQueryContext.getPartitionChecksumQueryId())
                .setPartitionChecksumQuery(checksumQueryContext.getPartitionChecksumQuery())
                .setBucketChecksumQueryId(checksumQueryContext.getBucketChecksumQueryId())
                .setBucketChecksumQuery(checksumQueryContext.getBucketChecksumQuery())
                .setQueryActionStats(queryContext.getMainQueryStats());
        updateQueryInfoWithQueryBundle(queryInfo, queryBundle);
        updateQueryInfo(queryInfo, queryResult);
        return queryInfo.build();
    }

    protected static String formatSql(Statement statement)
    {
        return SqlFormatter.formatSql(statement, Optional.empty());
    }

    protected static String formatSql(Statement statement, Optional comment)
    {
        return comment.isPresent() ? formatSql(statement) + "\n-- " + comment.get() : formatSql(statement);
    }

    protected static List formatSqls(List statements)
    {
        return statements.stream()
                .map(AbstractVerification::formatSql)
                .collect(toImmutableList());
    }

    private Optional getSkippedReason(Optional throwable, QueryState controlState, Optional determinismAnalysis)
    {
        if (throwable.isPresent() && !(throwable.get() instanceof QueryException)) {
            return Optional.of(VERIFIER_INTERNAL_ERROR);
        }
        if (skipControl && !QUERY_BANK_MODE.equals(runningMode)) {
            return Optional.empty();
        }
        switch (controlState) {
            case FAILED:
                return Optional.of(CONTROL_QUERY_FAILED);
            case FAILED_TO_SETUP:
                return Optional.of(CONTROL_SETUP_QUERY_FAILED);
            case TIMED_OUT:
                return Optional.of(CONTROL_QUERY_TIMED_OUT);
            case NOT_RUN:
                return Optional.of(FAILED_BEFORE_CONTROL_QUERY);
        }
        if (determinismAnalysis.isPresent() && determinismAnalysis.get().isNonDeterministic()) {
            return Optional.of(NON_DETERMINISTIC);
        }
        return Optional.empty();
    }

    private static QueryState getFailingQueryState(QueryException queryException)
    {
        QueryStage queryStage = queryException.getQueryStage();
        checkArgument(
                queryStage.isSetup() || queryStage.isMain(),
                "Expect QueryStage SETUP or MAIN: %s",
                queryStage);

        if (queryStage.isSetup()) {
            return FAILED_TO_SETUP;
        }
        return queryException instanceof PrestoQueryException
                && ((PrestoQueryException) queryException).getErrorCode().equals(Optional.of(EXCEEDED_TIME_LIMIT)) ?
                TIMED_OUT :
                QueryState.FAILED;
    }

    private String constructErrorMessage(
            Optional throwable,
            Optional matchResult,
            QueryState controlState,
            QueryState testState)
    {
        StringBuilder message = new StringBuilder(format("Test state %s, Control state %s.%n%n", testState, controlState));
        if (throwable.isPresent()) {
            if (throwable.get() instanceof PrestoQueryException) {
                PrestoQueryException exception = (PrestoQueryException) throwable.get();
                message.append(exception.getQueryStage().name().replace("_", " "))
                        .append(" query failed on ")
                        .append(exception.getQueryStage().getTargetCluster())
                        .append(" cluster:\n")
                        .append(exception.getCause() == null ? nullToEmpty(exception.getMessage()) : getStackTraceAsString(exception.getCause()));
            }
            else {
                message.append(getStackTraceAsString(throwable.get()));
            }
        }
        matchResult.ifPresent(result -> message.append(result.getReport()));
        return message.toString();
    }

    protected static class QueryContext
    {
        private Optional mainQueryStats = Optional.empty();
        private Optional state = Optional.empty();
        private ImmutableList.Builder setupQueryIds = ImmutableList.builder();
        private ImmutableList.Builder teardownQueryIds = ImmutableList.builder();

        public Optional getMainQueryStats()
        {
            return mainQueryStats;
        }

        public void setMainQueryStats(QueryActionStats mainQueryStats)
        {
            checkState(!this.mainQueryStats.isPresent(), "mainQueryStats is already set", mainQueryStats);
            this.mainQueryStats = Optional.of(mainQueryStats);
        }

        public QueryState getState()
        {
            return state.orElse(NOT_RUN);
        }

        public void setState(QueryState state)
        {
            checkState(!this.state.isPresent(), "state is already set", state);
            this.state = Optional.of(state);
        }

        public void setException(QueryException e)
        {
            setState(getFailingQueryState(e));
        }

        public List getSetupQueryIds()
        {
            return setupQueryIds.build();
        }

        public void addSetupQuery(QueryActionStats queryActionStats)
        {
            queryActionStats.getQueryStats().map(QueryStats::getQueryId).ifPresent(setupQueryIds::add);
        }

        public List getTeardownQueryIds()
        {
            return teardownQueryIds.build();
        }

        public void addTeardownQuery(QueryActionStats queryActionStats)
        {
            queryActionStats.getQueryStats().map(QueryStats::getQueryId).ifPresent(teardownQueryIds::add);
        }
    }

    private static class PartialVerificationResult
    {
        private final Optional skippedReason;
        private final Optional resolveMessage;
        private final EventStatus status;

        public PartialVerificationResult(Optional skippedReason, Optional resolveMessage, EventStatus status)
        {
            this.skippedReason = requireNonNull(skippedReason, "skippedReason is null");
            this.resolveMessage = requireNonNull(resolveMessage, "resolveMessage is null");
            this.status = requireNonNull(status, "status is null");
        }

        public Optional getSkippedReason()
        {
            return skippedReason;
        }

        public Optional getResolveMessage()
        {
            return resolveMessage;
        }

        public EventStatus getStatus()
        {
            return status;
        }
    }
}