在Eclipse RCP项目中运行普通JUnit + Mockito时的SecurityException

时间:2015-02-18 09:28:08

标签: junit eclipse-plugin eclipse-rcp mockito osgi-fragment

我有一个带有多个插件的Eclipse RCP项目。我正在编写简单的JUnit测试(没有依赖于Eclipse / UI)作为被测插件的单独片段。

当使用Mockito并尝试从另一个插件模拟一个接口(导出正确;我可以在我的代码中使用该接口)时,我得到一个与类签名相关的SecurityException:

org.mockito.exceptions.base.MockitoException: 
Mockito cannot mock this class: interface ch.sbb.polar.client.communication.inf.service.IUserService
Mockito can only mock visible & non-final classes.
If you're not sure why you're getting this error, please report to the mailing list.
at org.mockito.internal.runners.JUnit45AndHigherRunnerImpl$1.withBefores(JUnit45AndHigherRunnerImpl.java:27)

[...]

Caused by: org.mockito.cglib.core.CodeGenerationException: java.lang.reflect.InvocationTargetException-->null
at org.mockito.cglib.core.AbstractClassGenerator.create(AbstractClassGenerator.java:238)

[...]

Caused by: java.lang.reflect.InvocationTargetException
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)

[...]

Caused by: java.lang.SecurityException: Signers of 'ch.sbb.polar.client.communication.inf.service.IUserService$$EnhancerByMockitoWithCGLIB$$a8bfe723' do not match signers of other classes in package

at java.lang.ClassLoader.checkPackageSigners(ClassLoader.java:361)

at java.lang.ClassLoader.defineClass(ClassLoader.java:295)

... 40 more

当我按照" JUnit插件测试"运行测试时,即使用OSGi环境,一切都按预期工作。但是我想因为速度而使用普通的JUnit执行;在被测课程中,我不需要OSGi环境。

有人知道这样做的方法吗?

2 个答案:

答案 0 :(得分:5)

正如评论中提到的,根本原因是Mockito的Eclipse Orbit包(我已添加到我的目标平台)已签名,并且由于底层CGLIB中的错误,您无法模拟未签名的类/与已签名的Mockito接口。

有关最详细的说明,请参阅https://code.google.com/p/mockito/issues/detail?id=393。该错误已在CGLIB head中修复,但已not yet appeared in a release。 Mockito只使用已发布的版本作为依赖项,因此该修复程序尚未在Mockito中,并且具有未知(对我而言)的时间轴,就像它将在何时一样。

解决方法:在单独的捆绑包中提供未签名的Mockito

解决方法是将Mockito JAR(及其依赖项)打包到自己的bundle中并导出必要的API包。

当使用Maven Tycho,JUnit,Hamcrest和Mockito时,我能够正常工作并正确解决所有依赖/类路径/类加载器问题的方法如下:

  • 使用pom.xml中的以下条目创建Maven模块:

    <packaging>eclipse-plugin</packaging>
    

    [...]

    <dependencies>
      <dependency>
        <groupId>org.mockito</groupId>
        <artifactId>mockito-core</artifactId>
        <version>1.10.19</version>
      </dependency>
    </dependencies>
    

    [...]

    <plugin>
      <groupId>org.apache.maven.plugins</groupId>
      <artifactId>maven-dependency-plugin</artifactId>
      <executions>
        <execution>
          <id>copy-test-libs</id>
          <goals>
            <goal>copy-dependencies</goal>
          </goals>
          <configuration>
            <outputDirectory>lib</outputDirectory>
            <stripVersion>true</stripVersion>
            <includeScope>runtime</includeScope>
          </configuration>
        </execution>
      </executions>
    </plugin>
    
  • 在MANIFEST.MF中使用以下条目:

    Bundle-ClassPath: lib/mockito-core.jar,
     lib/objenesis.jar
    Export-Package: org.mockito,
     org.mockito.runners
    Require-Bundle: org.junit;bundle-version="4.11.0";visibility:=reexport,
     org.hamcrest.library;bundle-version="1.3.0";visibility:=reexport,
     org.hamcrest.core;bundle-version="1.3.0";visibility:=reexport
    
  • 最后在您的单元测试片段中,将此新捆绑包添加为依赖项。

答案 1 :(得分:0)

我遇到了同样的问题,并且能够通过使用更新了Mockito 2.x的较新的Orbit存储库来解决它:

http://download.eclipse.org/tools/orbit/downloads/drops/R20181128170323/?d

此存储库包含uses Byte Buddy instead of CGLIB的Mockito 2.23.0。

在目标中,我只需从上面的Orbit存储库中提取mockito-core 2.23.0Byte Buddy Java Agent 1.9.0

<unit id="org.mockito" version="2.23.0.v20181106-1534"/>
<unit id="org.mockito.source" version="2.23.0.v20181106-1534"/>
<unit id="net.bytebuddy.byte-buddy-agent" version="1.9.0.v20181106-1534"/>
<unit id="net.bytebuddy.byte-buddy-agent.source" version="1.9.0.v20181106-1534"/>