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

groovy.com.cedarsolutions.gradle.CedarCucumberPluginConvention.groovy Maven / Gradle / Ivy

Go to download

Gradle plugins and other functionality for use with a standardized build process.

There is a newer version: 0.9.8
Show newest version
// vim: set ft=groovy ts=4 sw=4:
// * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
// *
// *              C E D A R
// *          S O L U T I O N S       "Software done right."
// *           S O F T W A R E
// *
// * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
// *
// * Copyright (c) 2013-2016 Kenneth J. Pronovici.
// * All rights reserved.
// *
// * This program is free software; you can redistribute it and/or
// * modify it under the terms of the Apache License, Version 2.0.
// * See LICENSE for more information about the licensing terms.
// *
// * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
// *
// * Author   : Kenneth J. Pronovici 
// * Language : Gradle (>= 2.5)
// * Project  : Common Gradle Build Functionality
// *
// * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
package com.cedarsolutions.gradle

import org.gradle.api.Project
import org.gradle.api.InvalidUserDataException
import org.apache.tools.ant.taskdefs.condition.Os
import groovy.json.JsonSlurper;
import org.boon.json.JsonSerializerFactory;

/**
 * Plugin convention for cedarCucumber.
 * @author Kenneth J. Pronovici 
 */
class CedarCucumberPluginConvention {

    /** The project tied to this convention. */
    private Project project;

    /** Create a convention tied to a project. */
    public CedarCucumberPluginConvention(Project project) {
        this.project = project
    }

    /**
     * Execute the cucumber tests, returning an ExecResult so caller can handle failures.
     * If you want the database to start over fresh, you need to reboot dev mode.
     * You can optionally provide either name or feature, but not both.
     * The final result, useful for continuous integration, goes into a file called "result.json".
     * @param mode      The mode to run in, either "single-pass", "first-pass" or "second-pass"
     * @param name      Specific name of a test, or a substring, as for the Cucumber --name option
     * @param feature   Path to a specific feature to execute, relative to the acceptance/cucumber directory
     * See: http://jeannotsweblog.blogspot.com/2013/02/cucumber-10-command-line.html
     */
    def execCucumber(String mode, String name, String feature) {
        def command = []
        def rerunEnabled = false
        def rerunFile = project.file(project.projectDir.canonicalPath + "/build/tmp/rerun.txt").canonicalPath
        def reportDir = project.file(project.projectDir.canonicalPath + "/build/reports/cucumber").canonicalPath
        def firstPass = project.file(reportDir + "/first-pass.json").canonicalPath
        def secondPass = project.file(reportDir + "/second-pass.json").canonicalPath
        def result = project.file(reportDir + "/result.json").canonicalPath
        def returnValue

        if ("single-pass".equals(mode)) {
            println("\r\nExecuting Cucumber acceptance tests, single-pass mode")
        } else if ("first-pass".equals(mode)) {
            println("\r\nExecuting Cucumber acceptance tests, pass 1 of 2")
        } else if ("second-pass".equals(mode)) {
            println("\r\nExecuting Cucumber acceptance tests, pass 2 of 2 (retry failed tests)")
        }

        if (project.cedarCucumber.isJRuby()) {
            // Work around problems with the 1.6 JVM and Selenium with JRuby.
            // Without this option set, Selenium hangs forever trying to open Firefox.
            // See: http://stackoverflow.com/questions/10983307/jruby-watir-is-hanging-when-launching-browser
            command += [ "-J-Djava.net.preferIPv4Stack=true", ]

            // Larger Cucumber test suites use more than the default 500 MB of memory in JRuby
            if (project.cedarCucumber.getJrubyCucumberMemory() != null) {
                command += [ "-J-Xmx" + project.cedarCucumber.getJrubyCucumberMemory(), ]
            }
        }

        command += [ project.cedarCucumber.getCucumberPath(), ]
        command += [ "--require", project.cedarCucumber.getRubySubdir(), ]

        if (project.cedarCucumber.getCucumberFormatter() != null) {
            command += [ "--format", project.cedarCucumber.getCucumberFormatter(), ]
        }

        project.file(reportDir).mkdirs()

        if ("single-pass".equals(mode)) {
            project.file(firstPass).delete()
            project.file(secondPass).delete()
            project.file(result).delete()
        } else if ("first-pass".equals(mode)) {
            rerunEnabled = true
            project.file(rerunFile).delete()
            project.file(firstPass).delete()
            project.file(secondPass).delete()
            project.file(result).delete()
        } else if ("second-pass".equals(mode)) {
            project.file(secondPass).delete()
            project.file(result).delete()
        }

        if ("second-pass".equals(mode)) {
            command += [ "@" + rerunFile, ]
            command += [ "--format", "json", "--out", secondPass, ]
        } else {
            command += [ "--format", "json", "--out", firstPass, ]
            if (name != null) {
                command += "--name"
                command += name.replaceAll('"', "")  // quotes cause problems, so just remove them and the test won't be found
                command += project.cedarCucumber.getFeaturesSubdir()
            } else if (feature != null) {
                command += project.cedarCucumber.getFeaturesSubdir() + "/" + feature
            } else {
                command += project.cedarCucumber.getFeaturesSubdir()
            }
        }

        if (rerunEnabled) {
            returnValue = project.exec {
                workingDir = project.cedarCucumber.getCucumberDir()
                ignoreExitValue = true
                environment RERUN_FILE: rerunFile
                executable = project.cedarCucumber.getRubyPath()
                args = command
            }
        } else {
            returnValue = project.exec {
                workingDir = project.cedarCucumber.getCucumberDir()
                ignoreExitValue = true
                executable = project.cedarCucumber.getRubyPath()
                args = command
            }
        }

        if ("single-pass".equals(mode)) {
            project.file(firstPass).renameTo(result)
        } else if ("second-pass".equals(mode)) {
            mergeJsonResults(firstPass, secondPass, result)
            project.file(rerunFile).delete()
            project.file(firstPass).delete()
            project.file(secondPass).delete()
        }

        return returnValue
    }

    /** Verify that all of the required components have been installed in order to run Cucumber. */
    def verifyCucumberInstall() {
        project.logger.lifecycle("Using Cucumber from: " + project.cedarCucumber.getRubyInstallDir())
        verifyRuby()
        verifyGem()
        verifyCucumber()
        verifyGemVersions()
        project.logger.lifecycle("Cucumber install is ok.")
    }

    /** Install Cucumber, including Ruby and all of the other required dependencies.  */
    def installCucumber() {
        if (project.cedarCucumber.getRubyInstallDir() != "tools/cucumber") {
            project.logger.error("Project is configured to use Ruby from: " + project.cedarCucumber.getRubyInstallDir())
            throw new InvalidUserDataException("Before running installing Cucumber, re-configure project to use tools/cucumber")
        } else {
            // Only install if it's not there.  That way, we can always use installCucumber from continuous integration.
            if (!project.file(project.cedarCucumber.getRubyInstallDir()).exists()) {
                installJRuby(project.cedarCucumber.getJRubyDownloadUrl())
                installGem("selenium-webdriver", project.cedarCucumber.getSeleniumVersion())
                installGem("rspec", project.cedarCucumber.getRspecVersion())
                installGem("mime-types", project.cedarCucumber.getMimeTypesVersion())
                installGem("capybara", project.cedarCucumber.getCapybaraVersion())
                installGem("cucumber", project.cedarCucumber.getCucumberVersion())
                installGem("headless", project.cedarCucumber.getHeadlessVersion())
                verifyCucumberInstall()
                project.logger.lifecycle("All Cucumber tooling has been installed.")
            }
        }
    }

    /** Uninstall Cucumber, removing the install directory. */
    def uninstallCucumber() {
        if (project.file("tools").exists()) {
            if (project.file("tools/cucumber").exists()) {
                project.file("tools/cucumber").deleteDir()
                project.logger.lifecycle("All Cucumber tooling has been uninstalled.")
            }

            project.ant.delete(includeemptydirs : "true", quiet : "true") {
                fileset(dir: "tools", excludes : "**/*")  // remove tools only if empty
            }
        }
    }

    /** Verify that the Ruby interpreter is available. */
    private void verifyRuby() {
        if (!isRubyAvailable()) {
            project.logger.error("Ruby interpreter not available: " + project.cedarCucumber.getRubyPath())
            project.logger.error("You must either run the installCucumber task or install Ruby and Cucumber manually.")
            throw new InvalidUserDataException("Ruby interpreter not available: " + project.cedarCucumber.getRubyPath())
        }
    }

    /** Verify that the Ruby 'gem' tool is available. */
    private void verifyGem() {
        if (!isGemAvailable()) {
            project.logger.error("Ruby 'gem' tool not available: " + project.cedarCucumber.getGemPath())
            project.logger.error("You must either run the installCucumber task or install Ruby and Cucumber manually.")
            throw new InvalidUserDataException("Ruby 'gem' tool not available: " + project.cedarCucumber.getGemPath())
        }
    }

    /** Verify that the Cucumber tool is available. */
    private void verifyCucumber() {
        if (!isCucumberAvailable()) {
            project.logger.error("Cucumber tool not available: " + project.cedarCucumber.getCucumberPath())
            project.logger.error("You must either run the installCucumber task or install Ruby and Cucumber manually.")
            throw new InvalidUserDataException("Cucumber tool not available: " + project.cedarCucumber.getCucumberPath())
        }
    }

    /** Verify the versions of some specific gems. */
    private void verifyGemVersions() {
        if (project.cedarCucumber.getSeleniumVersion() != null) {
            def seleniumVersion = getGemVersion("selenium-webdriver")
            if (seleniumVersion != project.cedarCucumber.getSeleniumVersion()) {
                def version = project.cedarCucumber.getSeleniumVersion()
                project.logger.warn("Cucumber tests might not work due to version mismatch; expected Selenium ${version}, but got: " + seleniumVersion)
            }
        }

        if (project.cedarCucumber.getRspecVersion() != null) {
            def rspecVersion = getGemVersion("rspec")
            if (rspecVersion != project.cedarCucumber.getRspecVersion()) {
                def version = project.cedarCucumber.getRspecVersion()
                project.logger.warn("Cucumber tests might not work due to version mismatch; expected Rspec ${version}, but got: " + rspecVersion)
            }
        }

        if (project.cedarCucumber.getCapybaraVersion() != null) {
            def capybaraVersion = getGemVersion("capybara")
            if (capybaraVersion != project.cedarCucumber.getCapybaraVersion()) {
                def version = project.cedarCucumber.getCapybaraVersion()
                project.logger.warn("Cucumber tests might not work due to version mismatch; expected Capybara ${version}, but got: " + capybaraVersion)
            }
        }

        if (project.cedarCucumber.getCucumberVersion() != null) {
            def cucumberVersion = getGemVersion("cucumber")
            if (cucumberVersion != project.cedarCucumber.getCucumberVersion()) {
                def version = project.cedarCucumber.getCucumberVersion()
                project.logger.warn("Cucumber tests might not work due to version mismatch; expected Cucumber ${version}, but got: " + cucumberVersion)
            } 
        }

        if (project.cedarCucumber.getHeadlessVersion() != null) {
            def headlessVersion = getGemVersion("headless")
            if (headlessVersion != project.cedarCucumber.getHeadlessVersion()) {
                def version = project.cedarCucumber.getHeadlessVersion()
                project.logger.warn("Cucumber tests might not work due to version mismatch; expected Headless ${version}, but got: " + headlessVersion)
            }
        }
    }

    /** Check whether the Ruby interpreter is available. */
    private boolean isRubyAvailable() {
        return isCommandAvailable(project.cedarCucumber.getRubyPath(), [ "--version", ])
    }

    /** Check whether the Ruby 'gem' tool is available. */
    private boolean isGemAvailable() {
        return isCommandAvailable(project.cedarCucumber.getRubyPath(), [ project.cedarCucumber.getGemPath(), "--version", ])
    }

    /** Check whether the Cucumber tool is available. */
    private boolean isCucumberAvailable() {
        return isCommandAvailable(project.cedarCucumber.getRubyPath(), [ project.cedarCucumber.getCucumberPath(), "--version", ])
    }

    /** Get the installed version of a Ruby gem. */
    private String getGemVersion(gem) {
        try {
            def stdout = new ByteArrayOutputStream()
            def stderr = new ByteArrayOutputStream()

            def result = project.exec {
                standardOutput = stdout
                errorOutput = stderr
                executable = project.cedarCucumber.getRubyPath()
                args = [ project.cedarCucumber.getGemPath(), "list", gem, ]
            }

            def contents = stdout.toString()
            def regex = ~/(?s)(^${gem} [(])([.0-9]*)([,)].*$)/       // for a string like "selenium-webdriver (2.37.0)"
            def matcher = regex.matcher(contents)
            if (matcher.matches()) {
                return matcher.group(2)
            } 
        } catch (Exception e) { }

        project.logger.error("Error checking version for Ruby gem: " + gem)
        project.logger.error("You must either run the installCucumber task or install Ruby and Cucumber manually.")
        throw new InvalidUserDataException("Error checking version for Ruby gem: " + gem)
    }

    /** 
     * Verify whether a command is available by executing it.
     * @param command     The executable to check
     * @param arguments   Arguments to pass to the command
     * @return True if the command could be executed, false otherwise.
     */
    private boolean isCommandAvailable(command, arguments) {
        try {
            def devnull = new ByteArrayOutputStream()

            def result = project.exec {
                standardOutput = devnull
                errorOutput = devnull
                executable = command
                args = arguments
            }

            return result.getExitValue() == 0
        } catch (Exception e) {
            return false
        }
    }

    /** Install JRuby from a particular URL, expected to be a .tar.gz file. */
    private void installJRuby(url) {
        try {
            project.logger.lifecycle("Downloading JRuby from: " + url)
            project.file("build/tmp/jruby").deleteDir()
            project.file("build/tmp/jruby").mkdirs()
            project.ant.get(src: url, dest: "build/tmp/jruby/jruby.tar.gz")
            if (!project.file("build/tmp/jruby/jruby.tar.gz").exists()) {
                project.logger.error("jruby.tar.gz does not exist?")
                throw new InvalidUserDataException("Error downloading JRuby, file not retrieved.")
            }

            project.logger.lifecycle("Installing JRuby...")
            if (isWindows()) {
                project.ant.untar(src: "build/tmp/jruby/jruby.tar.gz", dest: "build/tmp/jruby", compression: "gzip")
            } else {
                // The Ant task doesn't preserve permissions, so use tar on other platforms
                def stdout = new ByteArrayOutputStream()
                def stderr = new ByteArrayOutputStream()
                def result = project.exec {
                    ignoreExitValue = true
                    workingDir = "build/tmp/jruby"
                    standardOutput = stdout
                    errorOutput = stderr
                    executable = "tar"
                    args = [ "zxvf", "jruby.tar.gz", ]
                }
            }

            project.file("tools").mkdirs()
            project.file("build/tmp/jruby").eachDir() { dir ->
                dir.renameTo("tools/cucumber") // there should be only one directory, like jruby-1.7.6
            }
        } catch (Exception e) { 
            throw new InvalidUserDataException("Error installing JRuby: " + e.getMessage(), e);
        } finally {
            project.file("build/tmp/jruby").deleteDir()
        }
    }

    /** Install a named gem, optionally specifying a version. */
    private void installGem(gem, version) {
        try {
            def arguments = [ project.cedarCucumber.getGemPath(), "install", gem, ]
            if (version != null) {
                arguments += [ "-v", version ]
            }

            if (version != null) {
                project.logger.lifecycle("Installing gem: " + gem + ", version " + version)
            } else {
                project.logger.lifecycle("Installing gem: " + gem + ", latest version")
            }
            
            def output = new ByteArrayOutputStream()
            def result = project.exec {
                ignoreExitValue = true
                standardOutput = output
                errorOutput = output
                executable = project.cedarCucumber.getRubyPath()
                args = arguments
            }

            if (result.getExitValue() != 0) {
                println(output.toString())
                project.logger.lifecycle("Note: gem install mirrors can sometimes be unavailable; you may want to try again later.")
                result.assertNormalExitValue()
            }
        } catch (Exception e) { 
            throw new InvalidUserDataException("Error installing gem: " + e.getMessage(), e);
        }
    }

    /** 
      * Merge JSON Cucumber results from two files on disk, using second pass results to overlay first pass results. 
      * This way, if a test fails in the first pass and succeeds in the second pass, we report success and not failure.
      */
    def mergeJsonResults(def firstPass, def secondPass, def result) {
        def firstPassMap = generateFeatureMap(firstPass);
        def secondPassMap = generateFeatureMap(secondPass);

        for (feature in firstPassMap) {
            for (scenario in feature.value.scenarioMap) {
                if (secondPassMap.containsKey(feature.value.id)) {
                    if (secondPassMap.get(feature.value.id).get("scenarioMap").containsKey(scenario.value.id)) {
                        feature.value.scenarioMap.put(scenario.value.id, secondPassMap.get(feature.value.id).get("scenarioMap").get(scenario.value.id))
                    }
                }
            }
        }

        def features = generateFeaturesList(firstPassMap)

        // Unfortunately, in the version of Groovy with Gradle v1.12, the
        // built-in JSON builder doesn't properly escape embedded quotes in map
        // keys.  This results in invalid JSON.  So, I've had to find an
        // alternative, and I've chosen to serialize with Boon instead.
        project.file(result).withWriter { writer ->
            def serializer = new JsonSerializerFactory().create();
            writer << serializer.serialize(features).toString();
        }
    }

    /** Parse Cucumber results and generate a more map-based structure that's easier to work with when merging results. */
    def generateFeatureMap(def json) {
        def featureMap = new LinkedHashMap()

        def jsonSlurper = new JsonSlurper()
        for (feature in jsonSlurper.parseText(project.file(json).text)) {
            featureMap.put(feature.id, feature)

            def scenarioMap = new LinkedHashMap();
            for (scenario in feature.elements) {
                scenarioMap.put(scenario.id, scenario)
            }

            featureMap.get(feature.id).remove("elements")
            featureMap.get(feature.id).put("scenarioMap", scenarioMap)
        }

        return featureMap
    }

    /** Turn our map-based structure back into a list that matches the Cucumber results format when merging results. */
    def generateFeaturesList(def map) {
        def features = new ArrayList()

        for (feature in map) {
            def elements = new ArrayList()

            for (scenario in feature.value.scenarioMap) {
                elements.add(scenario.value)
            }

            feature.value.remove("scenarioMap")
            feature.value.put("elements", elements);

            features.add(feature.value)
        }

        return features
    }

    /** Quick check to see whether the current platform is Windows. */
    private boolean isWindows() {
        return Os.isFamily(Os.FAMILY_WINDOWS);
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy