WELD-001408: การอ้างอิงที่ไม่เป็นที่พอใจสำหรับ Type Logger ที่มี qualifiers @Default ที่จุดฉีด
ฉันประสบปัญหานี้ขณะทดสอบแอปพลิเคชัน JavaEE8 กับ Arquillian: WELD-001408: การอ้างอิงที่ไม่เป็นที่พอใจสำหรับประเภท Logger ที่มีคุณสมบัติ @Default ที่จุดฉีด [BackedAnnotatedField] @Inject private academy.learnprogramming.services.TodoService.LOG
ฉันมีข้อผิดพลาดเดียวกันสำหรับ EntityManager ที่ถูกฉีดเข้าไป
นี่คือชั้นเรียนของฉัน:
TodoService.class (คลาสที่ฉันต้องทดสอบ)
package academy.learnprogramming.services;
import academy.learnprogramming.entities.Todo;
import org.apache.log4j.Logger;
import javax.annotation.PostConstruct;
import javax.inject.Inject;
import javax.persistence.EntityManager;
import javax.transaction.Transactional;
import java.util.List;
@Transactional
public class TodoService {
@Inject
private Logger LOG = Logger.getLogger(TodoService.class);
@Inject
EntityManager entityManager;
public Todo createTodo(Todo todo) {
//Persist into db
entityManager.persist(todo);
return todo;
}
public Todo updateTodo(Todo todo) {
entityManager.merge(todo);
return todo;
}
@PostConstruct
private void init() {
LOG.info("Bean TodoService created");
}
public Todo findTodoById(Long id) {
return entityManager.find(Todo.class, id);
}
public List<Todo> getTodos(){
return entityManager.createQuery("SELECT t from Todo t", Todo.class).getResultList();
}
}
TodoServiceTest.class (คลาสที่ใช้สำหรับการทดสอบ)
package academy.learnprogramming.services;
import academy.learnprogramming.config.Producers;
import academy.learnprogramming.entities.Todo;
import org.jboss.arquillian.container.test.api.Deployment;
import org.jboss.arquillian.junit.Arquillian;
import org.jboss.shrinkwrap.api.ShrinkWrap;
import org.jboss.shrinkwrap.api.asset.EmptyAsset;
import org.jboss.shrinkwrap.api.spec.JavaArchive;
import org.junit.After;
import org.junit.Before;
import org.junit.Test;
import org.junit.runner.RunWith;
import static org.junit.Assert.*;
@RunWith(Arquillian.class)
public class TodoServiceTest {
@Deployment
public static JavaArchive createDeployment() {
return ShrinkWrap.create(JavaArchive.class)
.addClasses(TodoService.class, Todo.class, Producers.class)
.addAsResource("persistence.xml", "META-INF/persistence.xml")
.addAsManifestResource(EmptyAsset.INSTANCE, "beans.xml");
}
@Before
public void setUp() throws Exception {
}
@After
public void tearDown() throws Exception {
}
@Test
public void createTodo() {
}
@Test
public void updateTodo() {
}
@Test
public void findTodoById() {
}
@Test
public void getTodos() {
}
}
Producers.class (คลาสที่ใช้เพื่อเพิ่มไลบรารีของบุคคลที่สามไปยัง CDI)
package academy.learnprogramming.config;
import org.apache.log4j.Logger;
import org.dozer.DozerBeanMapper;
import javax.enterprise.context.RequestScoped;
import javax.enterprise.inject.Produces;
import javax.enterprise.inject.spi.InjectionPoint;
import javax.persistence.EntityManager;
import javax.persistence.PersistenceContext;
import java.util.ArrayList;
import java.util.List;
//If producers becomes complitated, split it in many classes, each for a single producer
public class Producers {
@Produces
@PersistenceContext
EntityManager entityManager;
@Produces
public Logger produceLogger(InjectionPoint injectionPoint) {
return Logger.getLogger(injectionPoint.getMember().getDeclaringClass().getName());
}
//Auto mapper
@Produces
public DozerBeanMapper produceDozenBeanMapper() {
DozerBeanMapper mapper = new DozerBeanMapper();
List<String> mappingFiles = new ArrayList();
mappingFiles.add("dozerJdk8Converters.xml");
mapper.setMappingFiles(mappingFiles);
return mapper;
}
}
เมื่อค้นหาไปรอบ ๆ ฉันเข้าใจว่าปัญหาที่อาจเกิดขึ้นอาจเป็น:
- bean.xml อาจทำให้เกิดปัญหาได้หากคุณไม่ใส่ bean-discovery-mode = "all" เพราะจะไม่สแกนหาถั่วที่ไม่ใส่คำอธิบายประกอบ แต่ไฟล์ beans.xml ของฉันมี bean-discovery-mode = "all"
- Logger และ EntityManager ที่ฉีดเข้าไปในคลาส TodoService ไม่สามารถมี Producer ได้ แต่ฉันมีคลาส Producers
- ผู้คนสามารถทำผิดพลาดได้ในระหว่างการนำเข้าคลาส Logger สำหรับคลาส TodoService, TodoServiceTest และ Producers โดยมีตัวอย่างเช่น import org.apache.log4j.Logger ในคลาสเดียวและ java util.logging.Logger สำหรับคลาสอื่น แต่ฉันมีคลาสทั้งหมดที่มีการนำเข้า org.apache.log4j.Logger
ถ้าฉันลบคำอธิบายประกอบ @Inject และฉันใช้คนตัดไม้จากชั้นเรียนมันก็ใช้ได้
ฉันจะลองทำอะไรได้อีก? ขอบคุณมากสำหรับเวลาของคุณ
คำตอบ
ลองใช้ WebArchive หากคุณใช้ไฟล์ WAR
@Deployment
public static WebArchive createDeployment() {
return ShrinkWrap.create(WebArchive.class)
.addClasses(TodoService.class, Todo.class, Producers.class)
.addAsResource("persistence.xml", "META-INF/persistence.xml")
.addAsManifestResource(EmptyAsset.INSTANCE, "beans.xml");
}
เอาLogger.getLogger(TodoService.class);
ผู้ผลิตมันจะสร้างอินสแตนซ์คนตัดไม้
@RequestScoped
@Transactional
public class TodoService {
@Inject
private Logger log;
@Inject
EntityManager entityManager;
public Todo createTodo(Todo todo) {
//Persist into db
entityManager.persist(todo);
return todo;
}
public Todo updateTodo(Todo todo) {
entityManager.merge(todo);
return todo;
}
@PostConstruct
private void init() {
log.info("Bean TodoService created");
}
public Todo findTodoById(Long id) {
return entityManager.find(Todo.class, id);
}
public List<Todo> getTodos(){
return entityManager.createQuery("SELECT t from Todo t", Todo.class).getResultList();
}