Is it possible to skip a scenario with Cucumber-JVM at run-time

后端 未结 7 547
面向向阳花
面向向阳花 2020-12-30 12:54

I want to add a tag @skiponchrome to a scenario, this should skip the scenario when running a Selenium test with the Chrome browser. The reason to-do this is because some sc

相关标签:
7条回答
  • 2020-12-30 13:23

    If you're using Maven, you could read use a browser profile and then set the appropriate ~ exclude tags there?

    Unless you're asking how to run this from command line, in which case you tag the scenario with @skipchrome and then when you run cucumber set the cucumber options to tags = {"~@skipchrome"}

    0 讨论(0)
  • 2020-12-30 13:25

    I've implemented a customized junit runner as below. The idea is to add tags during runtime.

    So say for a scenario we need new users, we tag the scenarios as "@requires_new_user". Then if we run our test in an environment (say production environment which dose not allow you to register new user easily), then we will figure out that we are not able to get new user. Then the ""not @requires_new_user" will be added to cucumber options to skip the scenario.

    This is the most clean solution I can imagine now.

    public class WebuiCucumberRunner extends ParentRunner<FeatureRunner> {
        private final JUnitReporter jUnitReporter;
        private final List<FeatureRunner> children = new ArrayList<FeatureRunner>();
        private final Runtime runtime;
        private final Formatter formatter;
    
        /**
         * Constructor called by JUnit.
         *
         * @param clazz the class with the @RunWith annotation.
         * @throws java.io.IOException                         if there is a problem
         * @throws org.junit.runners.model.InitializationError if there is another problem
         */
        public WebuiCucumberRunner(Class clazz) throws InitializationError, IOException {
            super(clazz);
            ClassLoader classLoader = clazz.getClassLoader();
            Assertions.assertNoCucumberAnnotatedMethods(clazz);
    
            RuntimeOptionsFactory runtimeOptionsFactory = new RuntimeOptionsFactory(clazz);
            RuntimeOptions runtimeOptions = runtimeOptionsFactory.create();
    
            addTagFiltersAsPerTestRuntimeEnvironment(runtimeOptions);
    
            ResourceLoader resourceLoader = new MultiLoader(classLoader);
            runtime = createRuntime(resourceLoader, classLoader, runtimeOptions);
            formatter = runtimeOptions.formatter(classLoader);
            final JUnitOptions junitOptions = new JUnitOptions(runtimeOptions.getJunitOptions());
            final List<CucumberFeature> cucumberFeatures = runtimeOptions.cucumberFeatures(resourceLoader, runtime.getEventBus());
            jUnitReporter = new JUnitReporter(runtime.getEventBus(), runtimeOptions.isStrict(), junitOptions);
            addChildren(cucumberFeatures);
        }
    
        private void addTagFiltersAsPerTestRuntimeEnvironment(RuntimeOptions runtimeOptions) 
        {
            String channel = Configuration.TENANT_NAME.getValue().toLowerCase();
            runtimeOptions.getTagFilters().add("@" + channel);
    
            if (!TestEnvironment.getEnvironment().isNewUserAvailable()) {
                runtimeOptions.getTagFilters().add("not @requires_new_user");
            }
    
        }
    ...
    }
    

    Or you can extends the official Cucumber Junit test runner cucumber.api.junit.Cucumber and override method

        /**
         * Create the Runtime. Can be overridden to customize the runtime or backend.
         *
         * @param resourceLoader used to load resources
         * @param classLoader    used to load classes
         * @param runtimeOptions configuration
         * @return a new runtime
         * @throws InitializationError if a JUnit error occurred
         * @throws IOException         if a class or resource could not be loaded
         * @deprecated Neither the runtime nor the backend or any of the classes involved in their construction are part of
         * the public API. As such they should not be  exposed. The recommended way to observe the cucumber process is to
         * listen to events by using a plugin. For example the JSONFormatter.
         */
        @Deprecated
        protected Runtime createRuntime(ResourceLoader resourceLoader, ClassLoader classLoader,
                                        RuntimeOptions runtimeOptions) throws InitializationError, IOException {
            ClassFinder classFinder = new ResourceLoaderClassFinder(resourceLoader, classLoader);
            return new Runtime(resourceLoader, classFinder, classLoader, runtimeOptions);
        }
    

    You can manipulate runtimeOptions here as you wish. But the method is marked as deprecated, so use it with caution.

    0 讨论(0)
  • 2020-12-30 13:35

    I really prefer to be explicit about which tests are being run, by having separate run configurations defined for each environment. I also like to keep the number of tags I use to a minimum, to keep the number of configurations manageable.

    I don't think it's possible to achieve what you want with tags alone. You would need to write a custom jUnit test runner to use in place of @RunWith(Cucumber.class). Take a look at the Cucumber implementation to see how things work. You would need to alter the RuntimeOptions created by the RuntimeOptionsFactory to include/exclude tags depending on the browser, or other runtime condition.

    Alternatively, you could consider writing a small script which invokes your test suite, building up a list of tags to include/exclude dynamically, depending on the environment you're running in. I would consider this to be a more maintainable, cleaner solution.

    0 讨论(0)
  • 2020-12-30 13:37

    I realized that this is a late update to an already answered question, but I want to add one more option directly supported by cucumber-jvm:

    @Before //(cucumber one)
    public void setup(){
        Assume.assumeTrue(weAreInPreProductionEnvironment);
    }
    

    "and the scenario will be marked as ignored (but the test will pass) if weAreInPreProductionEnvironment is false."

    You will need to add

    import org.junit.Assume;
    

    The major difference with the accepted answer is that JUnit assume failures behave just like pending

    Important Because of a bug fix you will need cucumber-jvm release 1.2.5 which as of this writing is the latest. For example, the above will generate a failure instead of a pending in cucumber-java8-1.2.3.jar

    0 讨论(0)
  • 2020-12-30 13:40

    I too had the same challenge, where in I need to skip a scenario from running based on a flag which I obtain from the application dynamically in run-time, which tells whether the feature to be tested is enabled on the application or not..

    so this is how I wrote my logic in the scenarios file, where we have the glue code for each step.

    I have used a unique tag '@Feature-01AXX' to mark my scenarios that need to be run only when that feature(code) is available on the application.

    so for every scenario, the tag '@Feature-01XX' is checked first, if its present then the check for the availability of the feature is made, only then the scenario will be picked for running. Else it will be merely skipped, and Junit will not mark this as failure, instead it will me marked as Pass. So the final result if these tests did not run due to the un-availability of the feature will be pass, that's cool...

    @Before
    public void before(final Scenario scenario) throws Exception {
        /*
            my other pre-setup tasks for each scenario.
        */
    
        // get all the scenario tags from the scenario head.
        final ArrayList<String> scenarioTags = new ArrayList<>();
        scenarioTags.addAll(scenario.getSourceTagNames());
    
        // check if the feature is enabled on the appliance, so that the tests can be run.
        if (checkForSkipScenario(scenarioTags)) {
            throw new AssumptionViolatedException("The feature 'Feature-01AXX' is not enabled on this appliance, so skipping");
        }
    }
    
    private boolean checkForSkipScenario(final ArrayList<String> scenarioTags) {
        // I use a tag "@Feature-01AXX" on the scenarios which needs to be run when the feature is enabled on the appliance/application
        if (scenarioTags.contains("@Feature-01AXX") && !isTheFeatureEnabled()) { // if feature is not enabled, then we need to skip the scenario.
            return true;
        }
        return false;
    }
    
    private boolean isTheFeatureEnabled(){
        /*
            my logic to check if the feature is available/enabled on the application.
            in my case its an REST api call, I parse the JSON and check if the feature is enabled.
            if it is enabled return 'true', else return 'false'
        */
    }
    
    0 讨论(0)
  • 2020-12-30 13:40

    If you wish simply to temporarily skip a scenario (for example, while writing the scenarios), you can comment it out (ctrl+/ in Eclipse or Intellij).

    0 讨论(0)
提交回复
热议问题