问题描述

我们需要在 Jenkins Pipeline 中使用 Selenium 自动化测试,该笔记将记录实现 Jenkins + Selenium 自动化测试的方法。

解决方法

我们所面对的问题,以及对应的解决方案已经在 #1 中说明,以下是我们最终的解决方案(有些无奈):
1)脱离 Jenkins 而实现自己的自动化脚本,通过退出状态反映自动化测试结果;
2)在 Jenkins Pipeline 中使用 sh 执行脚本,并获取退出状态;
3)通过自动化脚本的退出状态,判断自动化测试是否成功,并执行后续的操作及任务;

代码参考(我们只记录关键步骤):

pipeline {
    agent any
    stages {
        stage('build') {
            steps {
            	script {
            		def statusCode = sh returnStatus: true, script: "groovy selenium.groovy"
            		if (statusCode != 0) {
            			// 执行失败
            		}
            	}
            }
        }
    }
}

关于 sh 的用法可以参考 Shell Command 笔记,或者官方 Pipeline: Nodes and Processes/sh 文档。

常见问题汇总

java.lang.NoSuchMethodError: com.google.common.base.Preconditions.checkArgument(ZLjava/lang/String;Ljava/lang/Object;)V

java - NoSuchMethodError: com.google.common.base.Preconditions.checkArgument(ZLjava/lang/String;J)V - Stack Overflow
Maven Repository: org.seleniumhq.selenium » selenium-java » 3.141.59

问题描述:

在 Jenkins Pipeline 中,使用 Selenium Java 类库,在执行自动化测试程序时,返回错误:

// 程序代码
@Grab(group='org.seleniumhq.selenium', module='selenium-java', version='3.141.59')
...
WebDriver webDriver = new RemoteWebDriver(new URL(seleniumHub), desiredCapabilities);
...

// 错误信息
java.lang.NoSuchMethodError: com.google.common.base.Preconditions.checkArgument(ZLjava/lang/String;Ljava/lang/Object;)V
	at com.google.common.reflect.TypeCapture.capture(TypeCapture.java:32)
	at com.google.common.reflect.TypeToken.<init>(TypeToken.java:125)
	at org.openqa.selenium.json.Json$1.<init>(Json.java:33)
	at org.openqa.selenium.json.Json.<clinit>(Json.java:33)
	at org.openqa.selenium.remote.NewSessionPayload.create(NewSessionPayload.java:100)
	at org.openqa.selenium.remote.NewSessionPayload.create(NewSessionPayload.java:94)
	at org.openqa.selenium.remote.ProtocolHandshake.createSession(ProtocolHandshake.java:68)
	at org.openqa.selenium.remote.HttpCommandExecutor.execute(HttpCommandExecutor.java:136)
	at org.openqa.selenium.remote.RemoteWebDriver.execute(RemoteWebDriver.java:552)
	at org.openqa.selenium.remote.RemoteWebDriver.startSession(RemoteWebDriver.java:213)
	at org.openqa.selenium.remote.RemoteWebDriver.<init>(RemoteWebDriver.java:131)
	at org.openqa.selenium.remote.RemoteWebDriver.<init>(RemoteWebDriver.java:144)
	at sun.reflect.NativeConstructorAccessorImpl.newInstance0(Native Method)
	at sun.reflect.NativeConstructorAccessorImpl.newInstance(NativeConstructorAccessorImpl.java:62)
	at sun.reflect.DelegatingConstructorAccessorImpl.newInstance(DelegatingConstructorAccessorImpl.java:45)
	at java.lang.reflect.Constructor.newInstance(Constructor.java:423)
	at org.codehaus.groovy.reflection.CachedConstructor.invoke(CachedConstructor.java:83)
	at org.codehaus.groovy.runtime.callsite.ConstructorSite$ConstructorSiteNoUnwrapNoCoerce.callConstructor(ConstructorSite.java:105)
	at org.codehaus.groovy.runtime.callsite.CallSiteArray.defaultCallConstructor(CallSiteArray.java:60)
	at org.codehaus.groovy.runtime.callsite.AbstractCallSite.callConstructor(AbstractCallSite.java:235)
	at com.cloudbees.groovy.cps.sandbox.DefaultInvoker.constructorCall(DefaultInvoker.java:25)
...

问题原因:

1)Selenium Java 3.141.59 需要使用 com.google.guava.guava 25.0-jre 类库
2)但是 Jenkins 2.241 也使用该类库,但是在 Jenkins 中该类库的版本非常旧(com.google.guava/guava/11.0.1)
3)因此导致 Selenium Java 调用旧版本的 Guava 类库,程序无法正常执行

解决办法:

# 不可行 # 最开始我们在编辑器中开发自动化测试程序(Selenium Java),现在要集成到 Jenkins 中(即代码复制到 Pipeline 中执行)才出现这个问题。我们无法强制指定 com.google.guava/guava/11.0.1 类库,因为该类库在 Jenkins 中已经加载,因此即使使用 @Grab 指定也不会成功。我们不会也不想使用那些奇技淫巧,比如改变类加载方式,这只会给以后带来麻烦。

# 不可行 # 也无法通过 Docker 节点运行,因为节点执行 Pipeline 也要载入 Jenkins 运行环境,依旧会出现同样的问题。

# 不可行 # 那在 Jenkins 中有没有 Selenium 插件呢?现有 Selenium 插件,该插件只是使 Jenkins 集群中增加 Selenium 集群,并且在 Jenkins 中增加 Selenium 管理界面。并不能解决我们的问题。

# 方案一 # 我们开始在网上寻找解决方法,很多做法大同小异,常见的做法是 maven test 执行自动化测试,本质上将就是脱离 Jenkins 而在命令行执行自动化测试脚本。最后我们也采用了这种做法:使用 Groovy 脚本,然后在 Jenkins 中执行 groovy 命令

我们只能选择# 方案一 #,因为只有# 方案一 #是 我们唯一能想到的 且 可行的 方案。

 

参考文献

Integration of Jenkins with Selenium WebDriver: Step-by-Step Tutorial
Maven & Jenkins Integration with Selenium: Complete Tutorial
How to Setup and Run Selenium Tests in Jenkins Using Maven and JUnit | BlazeMeter