Betamax с Spring Boot выбрасывает незаконное использование не виртуальной функции вызова
Я пытаюсь настроить простой тест в Groovy со Споком с помощью Betamax:
class BetaMaxSpockTest extends Specification {
@Rule
public Recorder recorder = new Recorder()
@Betamax(tape = "some_tape")
def 'You shall pass'() {
expect:
true
}
}
Я также использую Spring Boot, поэтому у меня есть spring-boot-starter-parent в качестве моего родителя в pom.xml:
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>1.2.6.RELEASE</version>
</parent>
Когда я запускаю выше тест, я получаю эту ошибку:
java.lang.VerifyError: (class: co/freeside/betamax/proxy/jetty/BetamaxProxy, method: super$3$getBean signature: (Ljava/lang/Class;)Ljava/lang/Object;) Illegal use of nonvirtual function call
at java.lang.Class.forName(Class.java:264)
at co.freeside.betamax.proxy.jetty.ProxyServer.start(ProxyServer.groovy:47)
at co.freeside.betamax.Recorder.startProxy(Recorder.groovy:198)
at co.freeside.betamax.Recorder.withTape(Recorder.groovy:167)
at co.freeside.betamax.Recorder$1.evaluate(Recorder.groovy:185)
at org.spockframework.runtime.extension.builtin.MethodRuleInterceptor.intercept(MethodRuleInterceptor.java:40)
at org.spockframework.runtime.extension.MethodInvocation.proceed(MethodInvocation.java:84)
at org.spockframework.util.ReflectionUtil.invokeMethod(ReflectionUtil.java:138)
at org.spockframework.util.ReflectionUtil.invokeMethod(ReflectionUtil.java:138)
at org.spockframework.util.ReflectionUtil.invokeMethod
Без Spring Boot на пути все работает нормально. Похоже, проблема некоторых версий. У кого-нибудь была похожая проблема?
Мой полный pom.xml:
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>com.test</groupId>
<artifactId>test</artifactId>
<version>1.0-SNAPSHOT</version>
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>1.2.6.RELEASE</version>
</parent>
<properties>
<java.version>1.8</java.version>
<!--plugins versions-->
<maven.compiler.plugin.version>3.3</maven.compiler.plugin.version>
<!--settings-->
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
</properties>
<dependencies>
<!--Spring Boot-->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-jpa</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<!--Spock Testing-->
<dependency>
<groupId>org.spockframework</groupId>
<artifactId>spock-core</artifactId>
<scope>test</scope>
</dependency>
<!--Groovy Testing-->
<dependency>
<groupId>org.codehaus.groovy</groupId>
<artifactId>groovy-all</artifactId>
<scope>test</scope>
</dependency>
<!--Betamax Http Mocks-->
<dependency>
<groupId>co.freeside</groupId>
<artifactId>betamax</artifactId>
<version>1.1.2</version>
<scope>test</scope>
</dependency>
</dependencies>
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
</plugin>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-compiler-plugin</artifactId>
</plugin>
</plugins>
<pluginManagement>
<--...-->
</pluginManagement>
</build>
</project>
2 ответа
Я нашел решение. Проблема была в неправильных версиях Jetty и httpclient. BetaMax требует версий, показанных ниже, в то время как родительский pom Spring объявляет гораздо более новые версии.
Добавление этих двух строк к свойствам в моем pom.xml решило проблему:
<jetty.version>7.3.1.v20110307</jetty.version>
<httpclient.version>4.2.1</httpclient.version>
В любом случае, спасибо за вашу помощь!
Есть несколько вещей, которые вы можете сделать, чтобы это исправить. Во-первых, вы можете сделать соответствующий код @CompileStatic
(вероятно, Spec в вашем случае), который часто решает проблему, но это не всегда желательно (и не всегда работает).
Другая вещь, которую вы можете сделать, это добавить -Xverify:none
к параметрам командной строки Java (где бы вы ни указали их для вашей среды). Это отключит проверку, которая является причиной ошибки.
Из того, что я читал об этой проблеме, когда я впервые столкнулся с ней, они планируют отключить проверку по умолчанию в следующей версии Java. Это обычно происходит с Groovy или другими библиотеками, которые выполняют множество манипуляций с байт-кодом под оболочкой (asm - другое).