Jenkins Groovy StackOverFlowError при доступе к свойству

Я пытаюсь написать небольшой DSL для работы, но у меня возникают java.lang.StackOverflowError ошибки при доступе к свойству класса.

Поэтому даже сложный сценарий не требуется. См. Следующий сценарий

class Komponente {

  String name

  Komponente(name) {
    this.name = name
  }

  String getName() {
    return this.name
  }  

  String toString() {
    return 'Klasse: Komponente (name: [' + this.name +'])'
  }
}

Komponente komponente = new Komponente('Testkomponente')

println 'Erstellte Komponente: ' + komponente.getName()

При запуске на веб-консоли Groovy он работает нормально, но при запуске в моем Jenkins я получаю:

FATAL: null
java.lang.StackOverflowError
  at java.lang.ClassLoader.findLoadedClass(ClassLoader.java:1038)
  at java.lang.ClassLoader.loadClass(ClassLoader.java:406)
  at groovy.lang.GroovyClassLoader.loadClass(GroovyClassLoader.java:677)
  at groovy.lang.GroovyClassLoader$InnerLoader.loadClass(GroovyClassLoader.java:425)
  at groovy.lang.GroovyClassLoader.loadClass(GroovyClassLoader.java:787)
  at groovy.lang.GroovyClassLoader.loadClass(GroovyClassLoader.java:775)
  at sun.reflect.GeneratedMethodAccessor316.invoke(Unknown Source)
  at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
  at java.lang.reflect.Method.invoke(Method.java:498)
  at org.codehaus.groovy.reflection.CachedMethod.invoke(CachedMethod.java:93)
  at groovy.lang.MetaMethod.doMethodInvoke(MetaMethod.java:325)
  at groovy.lang.MetaClassImpl.getProperty(MetaClassImpl.java:1850)
  at groovy.lang.MetaClassImpl.getProperty(MetaClassImpl.java:3758)
  at Komponente.getProperty(script)
  at org.codehaus.groovy.runtime.InvokerHelper.getProperty(InvokerHelper.java:174)
  at org.codehaus.groovy.runtime.ScriptBytecodeAdapter.getProperty(ScriptBytecodeAdapter.java:456)
  at org.kohsuke.groovy.sandbox.impl.Checker$6.call(Checker.java:290)
  at org.kohsuke.groovy.sandbox.GroovyInterceptor.onGetProperty(GroovyInterceptor.java:68)
  at org.jenkinsci.plugins.scriptsecurity.sandbox.groovy.SandboxInterceptor.onGetProperty(SandboxInterceptor.java:257)
  at org.kohsuke.groovy.sandbox.impl.Checker$6.call(Checker.java:288)
  at org.kohsuke.groovy.sandbox.impl.Checker.checkedGetProperty(Checker.java:292)
  at org.kohsuke.groovy.sandbox.impl.Checker$checkedGetProperty$2.callStatic(Unknown Source)
  at Komponente.getName(script:10)
  at sun.reflect.GeneratedMethodAccessor316.invoke(Unknown Source)
  at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
  at java.lang.reflect.Method.invoke(Method.java:498)
  at org.codehaus.groovy.reflection.CachedMethod.invoke(CachedMethod.java:93)
  at groovy.lang.MetaMethod.doMethodInvoke(MetaMethod.java:325)
  at groovy.lang.MetaClassImpl.getProperty(MetaClassImpl.java:1850)
  at groovy.lang.MetaClassImpl.getProperty(MetaClassImpl.java:3758)
  at Komponente.getProperty(script)
  at org.codehaus.groovy.runtime.InvokerHelper.getProperty(InvokerHelper.java:174)
  at org.codehaus.groovy.runtime.ScriptBytecodeAdapter.getProperty(ScriptBytecodeAdapter.java:456)
  at org.kohsuke.groovy.sandbox.impl.Checker$6.call(Checker.java:290)
  at org.kohsuke.groovy.sandbox.GroovyInterceptor.onGetProperty(GroovyInterceptor.java:68)
  at org.jenkinsci.plugins.scriptsecurity.sandbox.groovy.SandboxInterceptor.onGetProperty(SandboxInterceptor.java:257)
  at org.kohsuke.groovy.sandbox.impl.Checker$6.call(Checker.java:288)
  at org.kohsuke.groovy.sandbox.impl.Checker.checkedGetProperty(Checker.java:292)
  at org.kohsuke.groovy.sandbox.impl.Checker$checkedGetProperty$2.callStatic(Unknown Source)
  at Komponente.getName(script:10)
  ...

  (As you can see this repeates)

Как я могу получить доступ к свойству класса в сценарии Groovy, запущенном на Jenkins, без StackOverflowError исключения?

Моя система:

  • Версия Jenkins - 2.73.3
  • Плагин JobDSL 1.68
  • Модуль степлера Groovy org.kohsuke.stapler:stapler-groovy:1.250
  • Apache Groovy org.codehaus.groovy:groovy-all:2.4.11

Я уже немного искал это сообщение об ошибке, но все обнаруженные мной проблемы с Jenkins были исправлены и закрыты как минимум в 2.7.1, поэтому их следует включить в мою версию jenkins.


person bish    schedule 06.04.2018    source источник


Ответы (1)


Удалите getName() метод из вашего класса. Когда вы указываете поле класса, например String name, вы получаете методы String getName() и void setName(String name) из коробки.

Имейте в виду, что среда выполнения сценария Jenkins Groovy немного отличается от обычного Groovy (например, в консоли Groovy) - Jenkins использует среда выполнения groovy-cps.

В вашем случае следующий класс org.kohsuke.groovy.sandbox.impl.Checker вызвал ошибки - согласно вашей трассировке стека, вызывающей метод Komponente.getName() triggered Komponente.getProperty() через следующую цепочку выполнения:

  at Komponente.getProperty(script)
  at org.codehaus.groovy.runtime.InvokerHelper.getProperty(InvokerHelper.java:174)
  at org.codehaus.groovy.runtime.ScriptBytecodeAdapter.getProperty(ScriptBytecodeAdapter.java:456)
  at org.kohsuke.groovy.sandbox.impl.Checker$6.call(Checker.java:290)
  at org.kohsuke.groovy.sandbox.GroovyInterceptor.onGetProperty(GroovyInterceptor.java:68)
  at org.jenkinsci.plugins.scriptsecurity.sandbox.groovy.SandboxInterceptor.onGetProperty(SandboxInterceptor.java:257)
  at org.kohsuke.groovy.sandbox.impl.Checker$6.call(Checker.java:288)
  at org.kohsuke.groovy.sandbox.impl.Checker.checkedGetProperty(Checker.java:292)
  at org.kohsuke.groovy.sandbox.impl.Checker$checkedGetProperty$2.callStatic(Unknown Source)
  at Komponente.getName(script:10)

И Komponente.getProperty() сработал Komponente.getName(), и вы попадаете в бесконечный цикл, который вызвал StackOverflowError.

Похожая проблема возникла, когда я скопировал ваш класс в свой тестовый сценарий конвейера, вот трассировка стека, которую я получил:

Started by user admin
[Pipeline] End of Pipeline
java.lang.StackOverflowError: Excessively nested closures/functions at Komponente.getName(WorkflowScript:10) - look for unbounded recursion - call depth: 1025
    at com.cloudbees.groovy.cps.impl.CpsFunction.invoke(CpsFunction.java:28)
    at com.cloudbees.groovy.cps.impl.CpsCallableInvocation.invoke(CpsCallableInvocation.java:40)
    at com.cloudbees.groovy.cps.impl.ContinuationGroup.methodCall(ContinuationGroup.java:62)
    at com.cloudbees.groovy.cps.impl.FunctionCallBlock$ContinuationImpl.dispatchOrArg(FunctionCallBlock.java:109)
    at com.cloudbees.groovy.cps.impl.FunctionCallBlock$ContinuationImpl.fixName(FunctionCallBlock.java:77)
    at sun.reflect.GeneratedMethodAccessor106.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.ConstantBlock.eval(ConstantBlock.java:21)
    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:122)
    at org.codehaus.groovy.runtime.GroovyCategorySupport.use(GroovyCategorySupport.java:261)
    at com.cloudbees.groovy.cps.Continuable.run0(Continuable.java:163)
    at org.jenkinsci.plugins.workflow.cps.SandboxContinuable.access$001(SandboxContinuable.java:19)
    at org.jenkinsci.plugins.workflow.cps.SandboxContinuable$1.call(SandboxContinuable.java:35)
    at org.jenkinsci.plugins.workflow.cps.SandboxContinuable$1.call(SandboxContinuable.java:32)
    at org.jenkinsci.plugins.scriptsecurity.sandbox.groovy.GroovySandbox.runInSandbox(GroovySandbox.java:108)
    at org.jenkinsci.plugins.workflow.cps.SandboxContinuable.run0(SandboxContinuable.java:32)
    at org.jenkinsci.plugins.workflow.cps.CpsThread.runNextChunk(CpsThread.java:174)
    at org.jenkinsci.plugins.workflow.cps.CpsThreadGroup.run(CpsThreadGroup.java:331)
    at org.jenkinsci.plugins.workflow.cps.CpsThreadGroup.access$200(CpsThreadGroup.java:82)
    at org.jenkinsci.plugins.workflow.cps.CpsThreadGroup$2.call(CpsThreadGroup.java:243)
    at org.jenkinsci.plugins.workflow.cps.CpsThreadGroup$2.call(CpsThreadGroup.java:231)
    at org.jenkinsci.plugins.workflow.cps.CpsVmExecutorService$2.call(CpsVmExecutorService.java:64)
    at java.util.concurrent.FutureTask.run(FutureTask.java:266)
    at hudson.remoting.SingleLaneExecutorService$1.run(SingleLaneExecutorService.java:112)
    at jenkins.util.ContextResettingExecutorService$1.run(ContextResettingExecutorService.java:28)
    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:1142)
    at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:617)
    at java.lang.Thread.run(Thread.java:745)
Finished: FAILURE

Это не так просто, как у вас, но вызвано тем же. Когда я удалил String getName() метод, он работал как положено. Это должно сработать и для вас. Надеюсь, это поможет.

person Szymon Stepniak    schedule 06.04.2018
comment
Вау - я читал в руководствах по стилю, что вам не нужно создавать геттеры / сеттеры, но поскольку я из java, и я впервые пишу Groovy, мне было проще - и поскольку это работало в веб-консоли Я не думал, что это может быть проблемой. - person bish; 06.04.2018