package util import com.lesfurets.jenkins.unit.BasePipelineTest import org.junit.rules.TestRule import org.junit.runner.Description import org.junit.runners.model.Statement class JenkinsShellCallRule implements TestRule { enum Type { PLAIN, REGEX } class Command { final Type type final String script Command(Type type, String script) { this.type = type this.script = script } String toString() { return "${type} : ${script}" } @Override public int hashCode() { return type.hashCode() * script.hashCode() } @Override public boolean equals(Object obj) { if (obj == null || !obj instanceof Command) return false; Command other = (Command) obj; return type == other.type && script == other.script } } final BasePipelineTest testInstance List shell = [] Map returnValues = [:] List failingCommands = [] JenkinsShellCallRule(BasePipelineTest testInstance) { this.testInstance = testInstance } def setReturnValue(script, value) { setReturnValue(Type.PLAIN, script, value) } def setReturnValue(type, script, value) { returnValues[new Command(type, script)] = value } def failExecution(type, script) { failingCommands.add(new Command(type, script)) } @Override Statement apply(Statement base, Description description) { return statement(base) } private Statement statement(final Statement base) { return new Statement() { @Override void evaluate() throws Throwable { testInstance.helper.registerAllowedMethod("sh", [String.class], { command -> def unifiedScript = unify(command) shell.add(unifiedScript) for (Command failingCommand: failingCommands){ if(failingCommand.type == Type.REGEX && unifiedScript =~ failingCommand.script) { throw new Exception("Script execution failed!") break } else if(failingCommand.type == Type.PLAIN && unifiedScript.equals(failingCommand.script)) { throw new Exception("Script execution failed!") break } } }) testInstance.helper.registerAllowedMethod("sh", [Map.class], { m -> shell.add(m.script.replaceAll(/\s+/," ").trim()) def unifiedScript = unify(m.script) for (Command failingCommand: failingCommands){ if(failingCommand.type == Type.REGEX && unifiedScript =~ failingCommand.script) { throw new Exception("Script execution failed!") break } else if(failingCommand.type == Type.PLAIN && unifiedScript.equals(failingCommand.script)) { throw new Exception("Script execution failed!") break } } if (m.returnStdout || m.returnStatus) { def result = null for(def e : returnValues.entrySet()) { if(e.key.type == Type.REGEX && unifiedScript =~ e.key.script) { result = e.value break } else if(e.key.type == Type.PLAIN && unifiedScript.equals(e.key.script)) { result = e.value break } } if(result instanceof Closure) result = result() if (!result && m.returnStatus) result = 0 return result } }) base.evaluate() } } } private static String unify(String s) { s.replaceAll(/\s+/," ").trim() } }