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

com.capitalone.dashboard.evaluator.FeatureTestResultEvaluator Maven / Gradle / Ivy

package com.capitalone.dashboard.evaluator;

import com.capitalone.dashboard.ApiSettings;
import com.capitalone.dashboard.misc.HygieiaException;
import com.capitalone.dashboard.model.AuditException;
import com.capitalone.dashboard.model.CollectorItem;
import com.capitalone.dashboard.model.CollectorType;
import com.capitalone.dashboard.model.Dashboard;
import com.capitalone.dashboard.model.TestCapability;
import com.capitalone.dashboard.model.TestResult;
import com.capitalone.dashboard.model.TestSuite;
import com.capitalone.dashboard.repository.CollectorItemRepository;
import com.capitalone.dashboard.repository.TestResultRepository;
import com.capitalone.dashboard.request.ArtifactAuditRequest;
import com.capitalone.dashboard.response.TestResultsAuditResponse;
import com.capitalone.dashboard.status.TestResultAuditStatus;
import org.apache.commons.collections.CollectionUtils;
import org.apache.commons.lang3.StringUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;

import java.util.ArrayList;
import java.util.Collection;
import java.util.Comparator;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.stream.Collectors;

@Component
public class FeatureTestResultEvaluator extends Evaluator{

    private final TestResultRepository testResultRepository;
    private static final Logger LOGGER = LoggerFactory.getLogger(FeatureTestResultEvaluator.class);

    public static final String FUNCTIONAL = "Functional";
    public static final String ARTIFACT_NAME = "artifactName";
    public static final String ARTIFACT_VERSION = "artifactVersion";
    public static final String IDENTIFIER_VERSION = "identifierVersion";
    private static final String SUCCESS_COUNT = "successCount";
    private static final String FAILURE_COUNT = "failureCount";
    private static final String SKIP_COUNT = "skippedCount";
    private static final String TOTAL_COUNT = "totalCount";
    private static final String TEST_CASE_SUCCESS_COUNT = "successTestCaseCount";
    private static final String TEST_CASE_FAILURE_COUNT = "failureTestCaseCount";
    private static final String TEST_CASE_SKIPPED_COUNT = "skippedTestCaseCount";
    private static final String TEST_CASE_UNKNOWN_COUNT = "unknownStatusTestCaseCount";
    private static final String TEST_CASE_TOTAL_COUNT = "totalTestCaseCount";

    private static final String PRIORITY_HIGH = "High";



    @Autowired
    public FeatureTestResultEvaluator(TestResultRepository testResultRepository, CollectorItemRepository collectorItemRepository) {
        this.testResultRepository = testResultRepository;
        this.collectorItemRepository = collectorItemRepository;
    }


    @Override
    public Collection evaluate(Dashboard dashboard, long beginDate, long endDate, Map data, String altIdentifier, String identifierName) throws AuditException {
        Map collItemOptions = new HashMap<>();

        if (StringUtils.isEmpty((String)data.get(IDENTIFIER_VERSION)) || StringUtils.isEmpty(identifierName)){
            throw new AuditException("identifierVersion or identifierName missing.", AuditException.MISSING_DETAILS);
        }

        collItemOptions.put(ARTIFACT_NAME, identifierName);
        collItemOptions.put(ARTIFACT_VERSION, data.get(IDENTIFIER_VERSION));

        CollectorItem testItem = getCollectorItemForIdentifierVersion(dashboard, collItemOptions);
        if (testItem == null) {
            throw new AuditException("No tests configured", AuditException.NO_COLLECTOR_ITEM_CONFIGURED);
        }


        Double featureTestThreshold;
        try{
            try{
                featureTestThreshold = Double.parseDouble((String)data.get("featureTestThreshold"));
            }catch(NumberFormatException | NullPointerException e){
                LOGGER.error("Could not parse double from featureTestThreshold. Setting to default value.");
                LOGGER.error(e.toString());
                featureTestThreshold = Double.parseDouble(settings.getFeatureTestResultThreshold());
            }
        }catch(Exception e){
            throw new AuditException("testingThreshold unavailable. Cannot perform audit.", AuditException.BAD_INPUT_DATA);
        }


        // Value needs to be final to be used in lambda
        Double threshold = featureTestThreshold;

        Collection testResultsAuditResponseCollection = new ArrayList<>();

        List testResultList = testResultRepository.findByCollectorItemIdAndTimestampIsBetweenOrderByTimestampDesc(testItem.getId(), beginDate, endDate);
        TestResultsAuditResponse testResultsAuditResponse = new TestResultsAuditResponse();
        Map auditEntity = new HashMap<>();
        auditEntity.putAll(testItem.getOptions());
        auditEntity.put("testingThreshold", threshold);
        testResultsAuditResponse.setAuditEntity(auditEntity);
        testResultsAuditResponse.setLastUpdated(testItem.getLastUpdated());
        testResultsAuditResponse.setCollectorItemId(testItem.getId());

        // If no test results, set status to TEST_RESULT_MISSING and return
        if (CollectionUtils.isEmpty(testResultList)){
            testResultsAuditResponse.addAuditStatus(TestResultAuditStatus.TEST_RESULT_MISSING);
            testResultsAuditResponseCollection.add(testResultsAuditResponse);
            return testResultsAuditResponseCollection;
        }


        testResultList.forEach(testResult -> testResultsAuditResponseCollection.add(getFeatureTestResultAudit(testResultsAuditResponse ,testResult, threshold)));


        return testResultsAuditResponseCollection;
    }

    protected TestResultsAuditResponse getFeatureTestResultAudit(TestResultsAuditResponse baseTestResultsAuditResponse, TestResult testResult, Double threshold){
        TestResultsAuditResponse testResultsAuditResponse = new TestResultsAuditResponse();
        testResultsAuditResponse.setAuditEntity(baseTestResultsAuditResponse.getAuditEntity());
        testResultsAuditResponse.setLastUpdated(baseTestResultsAuditResponse.getLastUpdated());
        testResultsAuditResponse.setCollectorItemId(baseTestResultsAuditResponse.getCollectorItemId());
        testResultsAuditResponse.setBuildArtifact(testResult.getBuildArtifact());
        testResultsAuditResponse.setLastExecutionTime(testResult.getStartTime());
        testResultsAuditResponse.setType(testResult.getType().toString());
        testResultsAuditResponse.setFeatureTestResult(getFeatureTestResult(testResult));

        List testCapabilities = testResult.getTestCapabilities().stream().collect(Collectors.toList());
        testResultsAuditResponse = updateTestResultAuditStatuses(testCapabilities, testResultsAuditResponse, threshold);

        // Clearing for readability in response
        for(TestCapability test: testCapabilities){
            test.setTestSuites(null);
        }
        testResultsAuditResponse.setTestCapabilities(testCapabilities);
        return testResultsAuditResponse;
    }

    /**
     * Updates test result audit statuses based on percentage of test cases (not suites) that exceed the threshold.
     * @param testCapabilities List of testCapability attached to the testResult being processed
     * @param testResultsAuditResponse the audit response that the audit statuses will be added to
     * @param threshold testingThreshold from Bladerunner
     * @return TestResultsAuditResponse updated with statuses
     */
    private TestResultsAuditResponse updateTestResultAuditStatuses(List testCapabilities, TestResultsAuditResponse testResultsAuditResponse, double threshold) {
        if(isAllTestCasesSkipped(testCapabilities)){
            testResultsAuditResponse.addAuditStatus(TestResultAuditStatus.TEST_RESULT_SKIPPED);
            return testResultsAuditResponse;
        }

        double testCasePassPercent = this.getTestCasePassPercent(testCapabilities);

        if (testCasePassPercent < threshold) {
            testResultsAuditResponse.addAuditStatus(TestResultAuditStatus.TEST_RESULT_AUDIT_FAIL);
        } else {
            testResultsAuditResponse.addAuditStatus(TestResultAuditStatus.TEST_RESULT_AUDIT_OK);
        }

        return testResultsAuditResponse;
    }



    /**
     * Get test case pass percent for a given set of test capabilities.
     * @param testCapabilities List of TestCapability attached to testResult being processed
     * @return testCasePassPercent Double representing percentage of test cases that passed for given List of test capabilities
     */
    private double getTestCasePassPercent(List testCapabilities) {
        try{
            double testCaseSuccessCount = testCapabilities.stream().mapToDouble(testCapability ->
                    testCapability.getTestSuites().parallelStream().mapToDouble(TestSuite::getSuccessTestCaseCount).sum()
            ).sum();
            double totalTestCaseCount = testCapabilities.stream().mapToDouble(testCapability ->
                    testCapability.getTestSuites().parallelStream().mapToDouble(TestSuite::getTotalTestCaseCount).sum()
            ).sum();
            double testCaseSkipCount = testCapabilities.stream().mapToDouble(testCapability ->
                    testCapability.getTestSuites().parallelStream().mapToDouble(TestSuite::getSkippedTestCaseCount).sum()
            ).sum();
            double testCaseUnkownCount = testCapabilities.stream().mapToDouble(testCapability ->
                    testCapability.getTestSuites().parallelStream().mapToDouble(TestSuite::getUnknownStatusCount).sum()
            ).sum();

            if(totalTestCaseCount == 0 || (testCaseSkipCount + testCaseUnkownCount) == totalTestCaseCount){
                return 100.0;
            }

            return (testCaseSuccessCount/(totalTestCaseCount - testCaseSkipCount - testCaseUnkownCount)) * 100;

        }catch(Exception e){
            LOGGER.error("Could not get 'testCasePassPercent', setting to 0.0%");
            return 0.0;
        }
    }

    /**
     * Builds feature test result data map
     */
    protected HashMap getFeatureTestResult(TestResult testResult) {
        HashMap featureTestResultMap = new HashMap<>();

        //If an exception occurs an empty map will be returned.
        try{
            featureTestResultMap.put(SUCCESS_COUNT, testResult.getSuccessCount());
            featureTestResultMap.put(FAILURE_COUNT, testResult.getFailureCount());
            featureTestResultMap.put(SKIP_COUNT, testResult.getSkippedCount());
            featureTestResultMap.put(TOTAL_COUNT,testResult.getTotalCount());

            Collection testCapabilities = testResult.getTestCapabilities();
            int totalTestCaseCount = testCapabilities.stream().mapToInt(testCapability ->
                    testCapability.getTestSuites().parallelStream().mapToInt(TestSuite::getTotalTestCaseCount).sum()).sum();
            int testCaseSuccessCount = testCapabilities.stream().mapToInt(testCapability ->
                    testCapability.getTestSuites().parallelStream().mapToInt(TestSuite::getSuccessTestCaseCount).sum()).sum();
            int testCaseFailureCount = testCapabilities.stream().mapToInt(testCapability ->
                    testCapability.getTestSuites().parallelStream().mapToInt(TestSuite::getFailedTestCaseCount).sum()).sum();
            int testCaseSkippedCount = testCapabilities.stream().mapToInt(testCapability ->
                    testCapability.getTestSuites().parallelStream().mapToInt(TestSuite::getSkippedTestCaseCount).sum()).sum();
            int testCaseUnknownCount = testCapabilities.stream().mapToInt(testCapability ->
                    testCapability.getTestSuites().parallelStream().mapToInt(TestSuite::getUnknownStatusCount).sum()).sum();

            featureTestResultMap.put(TEST_CASE_TOTAL_COUNT, totalTestCaseCount);
            featureTestResultMap.put(TEST_CASE_SUCCESS_COUNT, testCaseSuccessCount);
            featureTestResultMap.put(TEST_CASE_FAILURE_COUNT, testCaseFailureCount);
            featureTestResultMap.put(TEST_CASE_SKIPPED_COUNT, testCaseSkippedCount);
            featureTestResultMap.put(TEST_CASE_UNKNOWN_COUNT, testCaseUnknownCount);

        }catch(Exception e){
            LOGGER.error("Exception occurred while processing testResult " + testResult.getDescription(), e);
        }


        return featureTestResultMap;
    }

    /**
     * Gets the collector item of type TEST associated with the given artifact name/version
     * @return testCollectorItem
     */
    protected CollectorItem getCollectorItemForIdentifierVersion(Dashboard dashboard, Map collItemOptions) {
        List testItems = getCollectorItems(dashboard, CollectorType.Test, FUNCTIONAL);
        // Sort so that the most recent results are audited
        testItems.sort(Comparator.comparing(CollectorItem::getUpsertTime).reversed());
        for(CollectorItem testItem : testItems){
            if(isEqualsIdentifierName(testItem, (String) collItemOptions.get(ARTIFACT_NAME)) && isEqualsIdentifierVersion(testItem, (String) collItemOptions.get(ARTIFACT_VERSION))){
               CollectorItem testCollItem = testItem;
               return testCollItem;
            }
        }
        LOGGER.warn("No feature test collector items found for this artifact name/version combination.");
        return null;
    }


    private boolean isEqualsIdentifierName(CollectorItem testItem, String identifierName) {
        return (Objects.nonNull(identifierName) && Objects.nonNull(testItem.getOptions())) ? identifierName.equalsIgnoreCase((String)testItem.getOptions().get(ARTIFACT_NAME)) : false;
    }

    private boolean isEqualsIdentifierVersion(CollectorItem testItem, String identifierVersion){
        return (Objects.nonNull(identifierVersion) && Objects.nonNull(testItem.getOptions())) ? identifierVersion.equalsIgnoreCase((String)testItem.getOptions().get(ARTIFACT_VERSION)) : false;
    }


    @Override
    public Collection evaluateNextGen(ArtifactAuditRequest artifactAuditRequest, Dashboard dashboard, long beginDate, long endDate, Map data) throws AuditException {
        return null;
    }

    @Override
    public TestResultsAuditResponse evaluate(CollectorItem collectorItem, long beginDate, long endDate, Map data) throws AuditException, HygieiaException {
        return null;
    }


    public void setSettings(ApiSettings settings) {
        this.settings = settings;
    }

    /**
     * Check if all the test cases are skipped
     * @param testCapabilities set of testCapabilities attached to the testResult being reviewed
     * @return boolean: true if all tests were skipped, false otherwise
     */
    public boolean isAllTestCasesSkipped(List testCapabilities) {
        try{
            int totalTestCaseCount = testCapabilities.stream().mapToInt(testCapability ->
                    testCapability.getTestSuites().parallelStream().mapToInt(TestSuite::getTotalTestCaseCount).sum()
            ).sum();
            int testCaseSkippedCount = testCapabilities.stream().mapToInt(testCapability ->
                    testCapability.getTestSuites().parallelStream().mapToInt(TestSuite::getSkippedTestCaseCount).sum()
            ).sum();

            boolean isSkippedHighPriority = settings.getTestResultSkippedPriority().equalsIgnoreCase(PRIORITY_HIGH);

            if ((testCaseSkippedCount >= totalTestCaseCount) && isSkippedHighPriority){
                return true;
            }
            return false;
        }catch(Exception e){
            LOGGER.warn("Error during isAllTestCasesSkipped call", e);
            return false;

        }
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy