java - Cobertura 导致 ClassNotFoundException

标签 java maven maven-cobertura-plugin

当使用带有 surefire 和 Cobertura 的 JUnit-Tests 使用 maven 构建我的项目以获得测试覆盖率时,通常,一切正常。但是当我最近添加了一个可能被某些测试抛出(并被排除)的异常时,maven 总是告诉我:

[ERROR] Failed to execute goal org.apache.maven.plugins:maven-surefire-plugin:2.18.1:test (default-test) on project backend-server: Execution default-test of goal org.apache.maven.plugins:maven-surefire-plugin:2.18.1:test failed: There was an error in the forked process
[ERROR] java.lang.NoClassDefFoundError: de/unileipzig/irpsim/backend/simulation/TimerowTooShortException
[ERROR] at java.lang.Class.getDeclaredMethods0(Native Method)
[ERROR] at java.lang.Class.privateGetDeclaredMethods(Class.java:2701)
[ERROR] at java.lang.Class.privateGetMethodRecursive(Class.java:3048)
[ERROR] at java.lang.Class.getMethod0(Class.java:3018)
[ERROR] at java.lang.Class.getMethod(Class.java:1784)
[ERROR] at org.apache.maven.surefire.util.ReflectionUtils.tryGetMethod(ReflectionUtils.java:57)
[ERROR] at org.apache.maven.surefire.common.junit3.JUnit3TestChecker.isSuiteOnly(JUnit3TestChecker.java:64)
[ERROR] at org.apache.maven.surefire.common.junit3.JUnit3TestChecker.isValidJUnit3Test(JUnit3TestChecker.java:59)
[ERROR] at org.apache.maven.surefire.common.junit3.JUnit3TestChecker.accept(JUnit3TestChecker.java:54)
[ERROR] at org.apache.maven.surefire.common.junit4.JUnit4TestChecker.accept(JUnit4TestChecker.java:52)
[ERROR] at org.apache.maven.surefire.util.DefaultScanResult.applyFilter(DefaultScanResult.java:97)
[ERROR] at org.apache.maven.surefire.junit4.JUnit4Provider.scanClassPath(JUnit4Provider.java:222)
[ERROR] at org.apache.maven.surefire.junit4.JUnit4Provider.invoke(JUnit4Provider.java:107)
[ERROR] at org.apache.maven.surefire.booter.ForkedBooter.invokeProviderInSameClassLoader(ForkedBooter.java:203)
[ERROR] at org.apache.maven.surefire.booter.ForkedBooter.runSuitesInProcess(ForkedBooter.java:155)
[ERROR] at org.apache.maven.surefire.booter.ForkedBooter.main(ForkedBooter.java:103)
[ERROR] Caused by: java.lang.ClassNotFoundException: de.unileipzig.irpsim.backend.simulation.TimerowTooShortException
[ERROR] at java.net.URLClassLoader.findClass(URLClassLoader.java:381)
[ERROR] at java.lang.ClassLoader.loadClass(ClassLoader.java:424)
[ERROR] at sun.misc.Launcher$AppClassLoader.loadClass(Launcher.java:331)
[ERROR] at java.lang.ClassLoader.loadClass(ClassLoader.java:357)
[ERROR] ... 16 more
[ERROR] -> [Help 1]

不幸的是,使用 -X 或类似的东西运行没有帮助,同时也在查看 http://maven.apache.org/surefire/maven-surefire-plugin/examples/class-loading.htmlIntermittent NoClassDefFoundError when running a maven/surefire build in jenkins没有为这个问题提供任何有用的提示。

我的 surefire-plugin 定义如下:

<plugin>
    <groupId>org.apache.maven.plugins</groupId>
    <artifactId>maven-surefire-plugin</artifactId>
    <version>2.18.1</version>
    <configuration>
        <forkedProcessTimeoutInSeconds>900</forkedProcessTimeoutInSeconds>
        <argLine>-Djava.library.path=${nativelib.directory}</argLine>
        <excludes>
            <exclude>**/DatabaseTest.java</exclude>
        </excludes>
    </configuration>
</plugin>

当停用 fork 时,一切开始正常工作,但不包括 native 库,因此这不是一个选项。此外,当我排除所有使用 TimerowTooShortException 的测试时,一切正常。

经过一些尝试,我发现即使使用 -Dcobertura.skip=true 运行测试,cobertura 也会导致问题。 Cobertura在插件中定义如下:

<plugin>
    <groupId>org.codehaus.mojo</groupId>
    <artifactId>cobertura-maven-plugin</artifactId>
    <version>2.6</version>
    <configuration>
        <formats>
            <format>xml</format>
        </formats>
    </configuration>
    <dependencies>
        <dependency>
            <groupId>org.ow2.asm</groupId>
            <artifactId>asm</artifactId>
            <version>5.0.3</version>
        </dependency>
    </dependencies>
    <executions>
        <execution>
            <phase>package</phase>
            <goals>
                <goal>cobertura</goal>
            </goals>
        </execution>
    </executions>
</plugin>

不幸的是,在将 surefire 和 cobertura 与预期异常一起使用时,我找不到有关此特定问题的任何信息。有人知道为什么在使用 cobertura 时会发生这种情况吗?对此有解决方法吗?

最佳答案

我在多模块项目中遇到了类似的错误。每当我添加 -Dcobertura.skip=true 时,我都会收到 NoClassDefFound 和 ClassNotFound 异常。

不幸的是,我不知道为什么。我的猜测是,并非每个 cobertura 目标都采用该参数。

但是,我使用配置文件“解决”了这个问题。也就是说,我将 cobertura 插件从默认构建配置移至 cobertura 配置文件。

<profiles>
   <profile>
       <id>cobertura-run</id>
       <properties>
            <maven.test.skip>false</maven.test.skip>
       </properties>
       <build>
           <plugins>
                <plugin>
                    <groupId>org.codehaus.mojo</groupId>
                    <artifactId>cobertura-maven-plugin</artifactId>
                    <version>2.7</version>
                    <configuration>
                        <formats>
                            <format>xml</format>
                        </formats>
                    </configuration>
                    <executions>
                        <execution>
                            <phase>package</phase>
                            <goals>
                                <goal>cobertura</goal>
                            </goals>
                        </execution>
                    </executions>
                </plugin>
           </plugins>
       </build>
   </profile>
</profiles>

每当我想让 cobertura 运行时,我都会使用 -P cobertura-run 而不是在命令行中禁用或启用 cobertura。

关于java - Cobertura 导致 ClassNotFoundException,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29566281/

相关文章:

java - 如何打印我的 Java 对象而不得到 "SomeType@2f92e0f4"?

java - mvn clean : PKIX path building failed: sun. security.provider.certpath.SunCertPathBuilderException

maven - Gradle中的Maven配置

java - 使用 Java 8 的 cobertura-maven-plugin

java - Cobertura 检查 Travis CI 失败

java - Java RMI 和 RPC 有什么区别?

c# - 为什么 Java 没有方法委托(delegate)?

java - Dropwizard 不接受 pojo 对象的 @PathParam

maven - Proguard 与 JDK 10/JDK 11 和 maven pugin