0

最近ひっかかりました。コード生成を担当するライブラリの一部を実行するために、いくつかの単体テストを作成しています。このコードは構成オブジェクトの入力を受け取り、CodeModel を使用してコードを記述し、これらのクラスを String にダンプし、Java 6 コンパイラ API を使用してその場で使用可能なクラスにコンパイルします。

私がヒットしている障害は、生成されたコードでテストしているメソッドの 1 つであり、最終メソッド (Android バンドル) を持つクラスを使用し、使用時に例外をスローします (「java.lang.RuntimeException: Stub!」)。したがって、これを回避するために、Ive は PowerMock を使用して最終的なメソッドをモックしました。ただし、この場合、コンパイラ API は NPE をスローします。これは、PowerMock が舞台裏で使用している機能によるものだと思いますが、よくわかりません。

例外は次のとおりです。

at org.androidtransfuse.gen.classloader.MemoryFileManager.<init>(MemoryFileManager.java:11)
at org.androidtransfuse.gen.classloader.MemoryClassLoader.<init>(MemoryClassLoader.java:16)
at org.androidtransfuse.gen.ParcelableGeneratorTest.setup(ParcelableGeneratorTest.java:67)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
at org.junit.internal.runners.MethodRoadie.runBefores(MethodRoadie.java:129)
at org.junit.internal.runners.MethodRoadie.runBeforesThenTestThenAfters(MethodRoadie.java:93)
at org.powermock.modules.junit4.internal.impl.PowerMockJUnit44RunnerDelegateImpl$PowerMockJUnit44MethodRunner.executeTest(PowerMockJUnit44RunnerDelegateImpl.java:296)

MemoryFileManager.java:

class MemoryFileManager extends ForwardingJavaFileManager<JavaFileManager> {
    public final Map<String, Output> map = new HashMap<String, Output>();

    MemoryFileManager(JavaCompiler compiler) {
        super(compiler.getStandardFileManager(null, null, null));
    }

    @Override
    public Output getJavaFileForOutput
            (Location location, String name, JavaFileObject.Kind kind, FileObject source) {
        Output mc = new Output(name, kind);
        this.map.put(name, mc);
        return mc;
    }
}

MemoryClassLoader.java (MemoryFileManager を使用):

public class MemoryClassLoader extends ClassLoader {
    private final JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
    private final MemoryFileManager manager = new MemoryFileManager(this.compiler);

    public void add(String classname, String filecontent) {
        add(Collections.singletonMap(classname, filecontent));
    }

    public void add(Map<String, String> map) {
        List<Source> list = new ArrayList<Source>();
        for (Map.Entry<String, String> entry : map.entrySet()) {
            list.add(new Source(entry.getKey(), JavaFileObject.Kind.SOURCE, entry.getValue()));
        }
        this.compiler.getTask(null, this.manager, null, null, null, list).call();
    }

    @Override
    protected Class<?> findClass(String name) throws ClassNotFoundException {
        synchronized (this.manager) {
            Output mc = this.manager.map.remove(name);
            if (mc != null) {
                byte[] array = mc.toByteArray();
                return defineClass(name, array, 0, array.length);
            }
        }
        return super.findClass(name);
    }
}

そして、この問題を示す非常に基本的なテスト:

@RunWith(PowerMockRunner.class)
@PrepareForTest(Parcel.class)
public class ExampleTest {

    private static final String TEST_VALUE = "test value";

    String classToCompile =
            "import android.os.Parcel;\n" +
            "\n" +
            "public class ClassToCompile\n" +
            "{\n" +
            "\n" +
            "    private Parcel parcel;\n" +
            "}";

    @Test
    public void test() throws ClassNotFoundException, IOException, NoSuchMethodException, InvocationTargetException, IllegalAccessException, InstantiationException {
        MemoryClassLoader classLoader = new MemoryClassLoader();
        classLoader.add("ClassToCompile", classToCompile);
        classLoader.loadClass("ClassToCompile");
    }
}
4

1 に答える 1

0

Powermock Google Group の Johanがこれに答えてくれましたが、ここにも結果を追加すると思いました。

私にとってうまくいったのは、 @RunWith(PowerMockRunner.class) アプローチから、ここで説明されている Java エージェント アプローチへの変更です。

これには、 @RunWith アノテーションを削除し、Maven POM に以下を追加する必要がありました。

<plugin>
    <groupId>org.apache.maven.plugins</groupId>
    <artifactId>maven-surefire-plugin</artifactId>
    <configuration>
        <argLine>
            -javaagent:${settings.localRepository}/org/powermock/powermock-module-javaagent/1.4.12/powermock-module-javaagent-1.4.12.jar
        </argLine>
    </configuration>
</plugin>

そして、IDE で単体テストを実行するときは、追加する必要がありました

-javaagent: <jarpath>/powermock-module-javaagent-1.4.12.jar

実行構成に。

于 2012-05-24T14:38:12.543 に答える