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

io.cucumber.junit.platform.engine.CucumberBatchTestEngine Maven / Gradle / Ivy

The newest version!
package io.cucumber.junit.platform.engine;

import net.thucydides.model.environment.SystemEnvironmentVariables;
import net.thucydides.model.util.EnvironmentVariables;

import org.junit.jupiter.engine.config.DefaultJupiterConfiguration;
import org.junit.jupiter.engine.descriptor.JupiterEngineDescriptor;
import org.junit.jupiter.engine.discovery.DiscoverySelectorResolver;
import org.junit.platform.engine.ConfigurationParameters;
import org.junit.platform.engine.EngineDiscoveryRequest;
import org.junit.platform.engine.ExecutionRequest;
import org.junit.platform.engine.TestDescriptor;
import org.junit.platform.engine.UniqueId;
import org.junit.platform.engine.support.config.PrefixedConfigurationParameters;
import org.junit.platform.engine.support.hierarchical.ForkJoinPoolHierarchicalTestExecutorService;
import org.junit.platform.engine.support.hierarchical.HierarchicalTestEngine;
import org.junit.platform.engine.support.hierarchical.HierarchicalTestExecutorService;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.util.ArrayList;
import java.util.Comparator;
import java.util.List;
import java.util.Set;
import java.util.stream.Collectors;
import java.util.stream.IntStream;
import java.util.stream.Stream;

import static io.cucumber.core.options.Constants.FILTER_TAGS_PROPERTY_NAME;
import static io.cucumber.junit.platform.engine.Constants.PARALLEL_CONFIG_PREFIX;
import static io.cucumber.junit.platform.engine.Constants.PARALLEL_EXECUTION_ENABLED_PROPERTY_NAME;
import static net.thucydides.model.ThucydidesSystemProperty.SERENITY_BATCH_COUNT;
import static net.thucydides.model.ThucydidesSystemProperty.SERENITY_BATCH_NUMBER;
import static net.thucydides.model.ThucydidesSystemProperty.SERENITY_FORK_COUNT;
import static net.thucydides.model.ThucydidesSystemProperty.SERENITY_FORK_NUMBER;


public final class CucumberBatchTestEngine extends HierarchicalTestEngine {

    static final Logger LOGGER = LoggerFactory.getLogger(CucumberBatchTestEngine.class);

    @Override
    public String getId() {
        return "cucumber-batch";
    }

    @Override
    public TestDescriptor discover(EngineDiscoveryRequest discoveryRequest, UniqueId uniqueId) {
        CucumberEngineDescriptor engineDescriptor = new CucumberEngineDescriptor(uniqueId);
        DefaultJupiterConfiguration jupiterConfiguration = new DefaultJupiterConfiguration(null);
        JupiterEngineDescriptor dd = new JupiterEngineDescriptor(uniqueId, jupiterConfiguration);
        new DiscoverySelectorResolver().resolveSelectors(discoveryRequest, dd);
        return engineDescriptor;
    }

    @Override
    protected HierarchicalTestExecutorService createExecutorService(ExecutionRequest request) {
        ConfigurationParameters config = request.getConfigurationParameters();
        if (config.getBoolean(PARALLEL_EXECUTION_ENABLED_PROPERTY_NAME).orElse(false)) {
            return new ForkJoinPoolHierarchicalTestExecutorService(
                new PrefixedConfigurationParameters(config, PARALLEL_CONFIG_PREFIX));
        }

        if (!request.getRootTestDescriptor().getChildren().isEmpty()) {
            processRequestIfBatched(request);
        }

        return super.createExecutorService(request);
    }

    static void processRequestIfBatched(ExecutionRequest request) {
        //populate list
        String tagFilter = request.getConfigurationParameters().get(FILTER_TAGS_PROPERTY_NAME)
            .orElse(System.getProperty(FILTER_TAGS_PROPERTY_NAME));
        List scenarioList = request.getRootTestDescriptor().getChildren().stream()
            .map(TestDescriptor::getChildren)
            .flatMap(Set::stream)
            .map(WeightedTest::new)
            .collect(Collectors.toList());
        int total = scenarioList.size();
        List tagFilteredScenarioList = scenarioList.stream()
            .filter(scenario -> scenario.isTagMatchingFilter(tagFilter))
            .collect(Collectors.toList());
        LOGGER.info("Found {} scenarios in classpath, {} match(es) tag filter {}", total, tagFilteredScenarioList.size(), tagFilter);

        EnvironmentVariables envs = SystemEnvironmentVariables.currentEnvironmentVariables();
        int batchCount = envs.getPropertyAsInteger(SERENITY_BATCH_COUNT, 1);
        int batchNumber = envs.getPropertyAsInteger(SERENITY_BATCH_NUMBER, 1);
        int forkCount = envs.getPropertyAsInteger(SERENITY_FORK_COUNT, 1);
        int forkNumber = envs.getPropertyAsInteger(SERENITY_FORK_NUMBER, 1);

        LOGGER.info("Parameters: \n{}", request.getConfigurationParameters());
        LOGGER.info("Running partitioning for batch {} of {} and fork {} of {}", batchNumber,
                    batchCount, forkNumber, forkCount);

        List batch = getPartition(tagFilteredScenarioList, batchCount, batchNumber);
        List testToRun = getPartition(batch, forkCount, forkNumber);

        //prune and keep only test to run
        scenarioList.removeAll(testToRun);
        scenarioList.forEach(WeightedTest::removeFromHierarchy);

        LOGGER.info("Running {} of {} scenarios", testToRun.size(), total);
        LOGGER.info("Test to run: {}", testToRun);
        LOGGER.info("Root test descriptor has {} feature(s)",
                    request.getRootTestDescriptor().getChildren().size());
    }

    @Override
    protected CucumberEngineExecutionContext createExecutionContext(ExecutionRequest request) {
        return new CucumberEngineExecutionContext(request.getConfigurationParameters());
    }

    static List getPartition(List list, int partitions, int index) {
        if (partitions == 1 && index == 1) {
            return new ArrayList<>(list);
        }
        return getPartitionedTests(list, partitions).get(index - 1);
    }

    static List> getPartitionedTests(List list, int partitions) {
        List> result = Stream.generate(ArrayList::new)
            .limit(partitions)
            .collect(Collectors.toList());

        //sort all scenarios from large to small
        list.sort(Comparator.comparing(WeightedTest::getWeight).reversed());
        int[] weights = new int[partitions];

        for (WeightedTest test : list) {
            int minPartition = getMinPartition(weights);
            result.get(minPartition).add(test);
            weights[minPartition] += test.getWeight();
        }

        for (int i = 0; i < result.size(); i++) {
            LOGGER.info("{} of {}, weight = {}", i + 1, partitions,
                        result.get(i).stream().mapToInt(WeightedTest::getWeight).sum());
            LOGGER.info(print(result.get(i)));
        }
        return result;
    }

    private static String print(List list) {
        return list.stream().map(WeightedTest::toString).collect(Collectors.joining("\n"));
    }

    private static int getMinPartition(int[] weights) {
        return IntStream.range(0, weights.length)
            .boxed()
            .min(Comparator.comparingInt(i -> weights[i]))
            .orElse(-1);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy