我有一个简单的管道脚本:
#!groovy
@org.jenkinsci.plugins.workflow.libs.Library('Some@lib')
import com.cloudbees.groovy.cps.NonCPS
node() {
echo CheekyEnum.getByName('name1').getName()
}
enum CheekyEnum {
ENUM_1('name1', 'f1'),
ENUM_2('name2', 'f2')
String name
String field
CheekyEnum(String name, String field) {
this.name = name
this.field = field
}
static CheekyEnum getByName(String name) {
return values().find { it.name == name }
}
String getName() {
return name
}
}
当我运行它时,一切都正常,但如果方法getName()
有一点变化
@NonCPS
String getName() {
return name
}
我得到一个相当长的错误堆栈:
java.lang.StackOverflowError
at java.lang.ClassLoader.loadClass(ClassLoader.java:398)
at java.lang.ClassLoader.loadClass(ClassLoader.java:351)
at org.jenkinsci.plugins.scriptsecurity.sandbox.groovy.SandboxResolvingClassLoader.lambda$loadClass$0(SandboxResolvingClassLoader.java:51)
at org.jenkinsci.plugins.scriptsecurity.sandbox.groovy.SandboxResolvingClassLoader.lambda$load$2(SandboxResolvingClassLoader.java:85)
at com.github.benmanes.caffeine.cache.BoundedLocalCache.lambda$doComputeIfAbsent$14(BoundedLocalCache.java:2337)
at java.util.concurrent.ConcurrentHashMap.compute(ConcurrentHashMap.java:1892)
at com.github.benmanes.caffeine.cache.BoundedLocalCache.doComputeIfAbsent(BoundedLocalCache.java:2335)
at com.github.benmanes.caffeine.cache.BoundedLocalCache.computeIfAbsent(BoundedLocalCache.java:2318)
at com.github.benmanes.caffeine.cache.LocalCache.computeIfAbsent(LocalCache.java:111)
at com.github.benmanes.caffeine.cache.LocalManualCache.get(LocalManualCache.java:54)
at org.jenkinsci.plugins.scriptsecurity.sandbox.groovy.SandboxResolvingClassLoader.load(SandboxResolvingClassLoader.java:79)
...
为什么?@NonCPS
不就是将方法排除在CPS转换之外吗?
enum本身是可序列化类型。因此,您应该更好地为它创建一个包装器函数:
import com.cloudbees.groovy.cps.NonCPS
node() {
echo getName(CheekyEnum.getByName('name1'))
}
...
@NonCPS
String getName(CheekyEnum cheeky) {
return cheeky.name
}
相关的StackOverflowError可能是工作流cps插件中的错误/气味。请看一下它的技术设计
管道脚本可以用注释@NonCPS标记指定的方法。然后正常编译这些文件(沙箱安全检查除外(。
AFAICS您正在Groovy沙箱中运行。SandboxInterceptor可能正在生成此堆栈溢出。在沙箱之外运行应该也可以解决您的问题。
顺便说一句,你也可以阅读管道CPS方法不匹配,以更好地理解在非CPS转换的代码中可以调用什么。