Использование .with на WorkflowScript в конвейере Jenkins - PullRequest
1 голос
/ 28 октября 2019

У меня есть некоторые общие части моих конвейерных сценариев Jenkins, извлеченных для вспомогательных классов в конвейерной библиотеке. Например, каждая работа должна использовать timestamps и ansiColor, поэтому давайте напишем это:

class Foo implements Serializable {
  WorkflowScript wfs

  Foo(WorkflowScript wfs) {
    this.wfs = wfs
  }

  def foo(Closure body) {
    this.wfs.timestamps {
      this.wfs.ansiColor {
        body()
      }
    }
  }
}

Конвейеры могут использовать этот класс следующим образом:

def f = new Foo(this)
f.foo { echo "Hello World!" }

Это работает, нопрефикс каждого шага конвейера с this.wfs в классе помощника добавляет много шума. Я подумывал написать вместо этого:

class Foo implements Serializable {
  WorkflowScript wfs

  Foo(WorkflowScript wfs) {
    this.wfs = wfs
  }

  def foo(Closure body) {
    this.wfs.with {
      timestamps {
        ansiColor {
          body()
        }
      }
    }
  }
}

Таким образом, однако, задание не выполняется со следующей ошибкой:

hudson.remoting.ProxyException: groovy.lang.MissingMethodException: No signature of method: Foo.timestamps() is applicable for argument types: (org.jenkinsci.plugins.workflow.cps.CpsClosure2) values: [org.jenkinsci.plugins.workflow.cps.CpsClosure2@279d65ed]
    at org.codehaus.groovy.runtime.ScriptBytecodeAdapter.unwrap(ScriptBytecodeAdapter.java:58)
    at org.codehaus.groovy.runtime.ScriptBytecodeAdapter.unwrap(ScriptBytecodeAdapter.java:64)
    at org.codehaus.groovy.runtime.callsite.PogoMetaClassSite.call(PogoMetaClassSite.java:54)
    at org.codehaus.groovy.runtime.callsite.CallSiteArray.defaultCall(CallSiteArray.java:48)
    at org.codehaus.groovy.runtime.callsite.AbstractCallSite.call(AbstractCallSite.java:113)
    at org.kohsuke.groovy.sandbox.impl.Checker$1.call(Checker.java:160)
    at org.kohsuke.groovy.sandbox.GroovyInterceptor.onMethodCall(GroovyInterceptor.java:23)
    at org.jenkinsci.plugins.scriptsecurity.sandbox.groovy.SandboxInterceptor.onMethodCall(SandboxInterceptor.java:157)
    at org.jenkinsci.plugins.scriptsecurity.sandbox.groovy.SandboxInterceptor.onMethodCall(SandboxInterceptor.java:142)
    at org.kohsuke.groovy.sandbox.impl.Checker$1.call(Checker.java:158)
    at org.kohsuke.groovy.sandbox.impl.Checker.checkedCall(Checker.java:162)
    at com.cloudbees.groovy.cps.sandbox.SandboxInvoker.methodCall(SandboxInvoker.java:17)
    at Foo.foo(WorkflowScript:146)
    at com.cloudbees.groovy.cps.CpsDefaultGroovyMethods.with(CpsDefaultGroovyMethods:242)
    at Foo.foo(WorkflowScript:145)
    at WorkflowScript.run(WorkflowScript:159)
    at ___cps.transform___(Native Method)
    at com.cloudbees.groovy.cps.impl.ContinuationGroup.methodCall(ContinuationGroup.java:84)
    at com.cloudbees.groovy.cps.impl.FunctionCallBlock$ContinuationImpl.dispatchOrArg(FunctionCallBlock.java:113)
    at com.cloudbees.groovy.cps.impl.FunctionCallBlock$ContinuationImpl.fixArg(FunctionCallBlock.java:83)
    at sun.reflect.GeneratedMethodAccessor188.invoke(Unknown Source)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:498)
    at com.cloudbees.groovy.cps.impl.ContinuationPtr$ContinuationImpl.receive(ContinuationPtr.java:72)
    at com.cloudbees.groovy.cps.impl.ClosureBlock.eval(ClosureBlock.java:46)
    at com.cloudbees.groovy.cps.Next.step(Next.java:83)
    at com.cloudbees.groovy.cps.Continuable$1.call(Continuable.java:174)
    at com.cloudbees.groovy.cps.Continuable$1.call(Continuable.java:163)
    at org.codehaus.groovy.runtime.GroovyCategorySupport$ThreadCategoryInfo.use(GroovyCategorySupport.java:129)
    at org.codehaus.groovy.runtime.GroovyCategorySupport.use(GroovyCategorySupport.java:268)
    at com.cloudbees.groovy.cps.Continuable.run0(Continuable.java:163)
    at org.jenkinsci.plugins.workflow.cps.SandboxContinuable.access$001(SandboxContinuable.java:18)
    at org.jenkinsci.plugins.workflow.cps.SandboxContinuable.run0(SandboxContinuable.java:51)
    at org.jenkinsci.plugins.workflow.cps.CpsThread.runNextChunk(CpsThread.java:186)
    at org.jenkinsci.plugins.workflow.cps.CpsThreadGroup.run(CpsThreadGroup.java:370)
    at org.jenkinsci.plugins.workflow.cps.CpsThreadGroup.access$200(CpsThreadGroup.java:93)
    at org.jenkinsci.plugins.workflow.cps.CpsThreadGroup$2.call(CpsThreadGroup.java:282)
    at org.jenkinsci.plugins.workflow.cps.CpsThreadGroup$2.call(CpsThreadGroup.java:270)
    at org.jenkinsci.plugins.workflow.cps.CpsVmExecutorService$2.call(CpsVmExecutorService.java:66)
    at java.util.concurrent.FutureTask.run(FutureTask.java:266)
    at hudson.remoting.SingleLaneExecutorService$1.run(SingleLaneExecutorService.java:131)
    at jenkins.util.ContextResettingExecutorService$1.run(ContextResettingExecutorService.java:28)
    at jenkins.security.ImpersonatingExecutorService$1.run(ImpersonatingExecutorService.java:59)
    at java.util.concurrent.Executors$RunnableAdapter.call(Executors.java:511)
    at java.util.concurrent.FutureTask.run(FutureTask.java:266)
    at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149)
    at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624)
    at java.lang.Thread.run(Thread.java:748)

Что не так с использованием .with на WorkflowScript? С другими классами это работает нормально. Есть ли обходной путь?

1 Ответ

0 голосов
/ 28 октября 2019

Если все, что вы хотите сделать - это обернуть конвейер, вы можете добиться этого на шаге общей библиотеки. Предположим, вы создаете Foo.groovy:

def call(Closure body) {
    pipeline {
        agent any
        stages {
            stage("Initialize") {
                steps {
                    timestamps {
                        ansiColor {
                            echo "Hello world!"
                            body()
                        }
                    }
                }
            }
        }
    }
}

Тогда вы можете использовать Foo вместо pipeline:

library "your-shared-lib"
Foo {
  echo "123"
}

Этот пример ограничен одним этапом, однако, если вы хотитебольше этапов вам придется использовать сценарий конвейера. В Foo.groovy вам нужно поместить body() в блок script, а затем вы можете добавить все, что вы хотите:

library "your-shared-lib"
Foo {
  stage("1") {
    echo "Only scripted pipeline here"
  }
  stage("2") {
    // If you plan on using more than one node it's better to
    // change the agent in Foo.groovy to none
    node("label") {
    }
  }
}

Недостатки этого метода в том, что вы вынуждены использоватьсценарий конвейера, и все ваши этапы будут вложены в родительский этап Foo (но Blue Ocean достаточно умен, чтобы отобразить этапы, следующие за родительским этапом.)

...