1

私はMapStructを使用してマッパーを生成し、エンティティをDTOに、またはその逆に変換する作業を支援するMavenプロジェクトを持っています。このマッパーは、maven の generate-sources フェーズで生成され、target/generated-sources および target/AppName/WEB-INF/classes フォルダーに格納されます。

たとえば、私はこのマッパーを持っています

@Mapper
public interface RuleMapper {

    RuleDto ruletoDto(Rule rule);

    //other cool stuf
}

CDI を使用するように MapStruct を構成したので、次のように生成されます。

@Generated(
value = "org.mapstruct.ap.MappingProcessor",
date = "2016-12-19T23:19:36-0200",
comments = "version: 1.1.0.CR1, compiler: javac, environment: Java 1.8.0_112"
)
@Singleton
@Named
public class RuleMapperImpl implements RuleMapper {

    @Override
    public RuleDto ruletoDto(Rule rule) {

        ruleDto ruleDto = new ruleDto();

        if ( rule != null ) {
            ruleDto.setIdRule( rule.getIdRule() );
        }

        return ruleDto;
    }
}

Wildflyサーバーで実行すると完全に機能します。問題は、このクラスをjunitテストしようとしていることです。このために、以下に示すようにカスタムランナーを実装しました。

import org.junit.runners.BlockJUnit4ClassRunner;
import org.junit.runners.model.InitializationError;

public class WeldJUnit4Runner extends BlockJUnit4ClassRunner {

    public WeldJUnit4Runner(Class<Object> clazz) throws InitializationError {
        super(clazz);
    }

    @Override
    protected Object createTest() throws Exception {
        final Class<?> test = getTestClass().getJavaClass();
        return WeldContext.INSTANCE.getBean(test);
    }

}

と:

import org.jboss.weld.environment.se.Weld;
import org.jboss.weld.environment.se.WeldContainer;

public class WeldContext {

    public static final WeldContext INSTANCE = new WeldContext();

    private final Weld weld;
    private final WeldContainer container;

    private WeldContext() {
        this.weld = new Weld();
        this.container = weld.initialize();
        Runtime.getRuntime().addShutdownHook(new Thread() {
            @Override
            public void run() {
                weld.shutdown();
            }
        });
    }

    public <T> T getBean(Class<T> type) {
        return container.instance().select(type).get();
    }

}

これらの実装はhereから取得されました。

最後に、テスト:

@RunWith(WeldJUnit4Runner.class)
public class RuleMapperTest {

    @Inject
    private RuleMapper ruleMapper;

    @Test
    public void coolTestName() {
        Assert.assertTrue(Boolean.TRUE);
    }
}

実行しようとすると、これがコンソール出力です。

log4j:WARN ロガー (org.jboss.logging) のアペンダーが見つかりませんでした。log4j:WARN log4j システムを適切に初期化してください。log4j:WARN 詳細については、http: //logging.apache.org/log4j/1.2/faq.html#noconfigを参照してください。

ログに関する警告、および次の例外:

java.lang.ExceptionInInitializerError の br.com.treinoos.common.cdi.WeldJUnit4Runner.createTest(WeldJUnit4Runner.java:15) の org.junit.runners.BlockJUnit4ClassRunner$1.runReflectiveCall(BlockJUnit4ClassRunner.java:266) の org.junit.internal .runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12) で org.junit.runners.BlockJUnit4ClassRunner.methodBlock(BlockJUnit4ClassRunner.java:263) で org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:78) でorg.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:57) org.junit.runners.ParentRunner$3.run(ParentRunner.java:290) org.junit.runners.ParentRunner$1.schedule(ParentRunner.java: 71) org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288) で org.junit.runners.ParentRunner.access$000(ParentRunner.java:58) at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:268) at org.junit.runners.ParentRunner.run(ParentRunner.java:363) at org.eclipse.jdt .internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.java:86) at org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38) at org.eclipse.jdt.internal.junit org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:675) の .runner.RemoteTestRunner.runTests(RemoteTestRunner.java:459) org.eclipse.jdt.internal.junit.runner.RemoteTestRunner の.run(RemoteTestRunner.java:382) at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:192) 原因: org.jboss.weld.exceptions.DeploymentException: WELD-001408:インジェクション ポイント [BackedAnnotatedField] で修飾子 @Default を持つ型 RuleMapper の満たされていない依存関係 [BackedAnnotatedField] @Inject private treinoos.mappers.RuleMapperTest.ruleMapper(RuleMapperTest.java:0) org.jboss.weld.bootstrap.Validator.validateInjectionPointForDeploymentProblems(Validator.java:359) org.jboss.weld.bootstrap.Validator.validateInjectionPoint(Validator.java: 281) org.jboss.weld.bootstrap.Validator.validateGeneralBean(Validator.java:134) で org.jboss.weld.bootstrap.Validator.validateRIBean(Validator.java:155) で org.jboss.weld.bootstrap.Validator .validateBean(Validator.java:518) at org.jboss.weld.bootstrap.ConcurrentValidator$1.doWork(ConcurrentValidator.java:68) org.jboss.weld.bootstrap.ConcurrentValidator$1.doWork(ConcurrentValidator.java:66) で org.jboss.weld.executor.IterativeWorkerTaskFactory$1.call(IterativeWorkerTaskFactory.java:63) で org.jboss.weld.executor .IterativeWorkerTaskFactory$1.call(IterativeWorkerTaskFactory.java:56) で java.util.concurrent.FutureTask.run(FutureTask.java:266) で java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1142) で java.util .concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:617) at java.lang.Thread.run(Thread.java:745)call(IterativeWorkerTaskFactory.java:56) で java.util.concurrent.FutureTask.run(FutureTask.java:266) で java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1142) で java.util.concurrent.ThreadPoolExecutor で$Worker.run(ThreadPoolExecutor.java:617) at java.lang.Thread.run(Thread.java:745)call(IterativeWorkerTaskFactory.java:56) で java.util.concurrent.FutureTask.run(FutureTask.java:266) で java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1142) で java.util.concurrent.ThreadPoolExecutor で$Worker.run(ThreadPoolExecutor.java:617) at java.lang.Thread.run(Thread.java:745)

同様に、Weld は生成されたクラスを検索できませんでした。beans.xmlはすでにの下に作成されていsrc/test/resources/META-INF/beans.xmlます:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://xmlns.jcp.org/xml/ns/javaee"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee 
                  http://xmlns.jcp.org/xml/ns/javaee/beans_1_1.xsd"
    version="1.1" bean-discovery-mode="all">
</beans>

誰でもこの問題の解決策を教えてもらえますか? 私はすでに似たようなものを検索しましたが、成功しませんでした。

4

1 に答える 1

0

ここにあなたの問題の完全な説明と、私が書いたものがそれを修正する理由があります.

Maven には、少なくとも 2 つのクラスローダーがあります。テスト クラスパスとメイン クラスパスには、それぞれ独自のクラスローダーがあります。依存関係の構造に応じて、他のものを持つことができます。この方法で実行すると、CDI は各クラスローダーを個別の Bean アーカイブとして識別します。 src/main/webapp明示的にWARファイル用です。そこにはBeanアーカイブはbeans.xmlありません。に1つ追加src/main/resourcesします。この問題は、溶接をインスタンス化する方法に固有のものです。

これを正しく行うプロジェクトは他にもあります - CDI-unitArquillian、特に Weld Embedded コンテナです。これらのいずれかを使用する場合、これは問題になりません。

于 2016-12-21T23:55:16.860 に答える