如何使用jax-rs子资源定位器处理持久化上下文(EntityManager)?

时间:2011-04-20 18:12:24

标签: java rest jpa jax-rs

我在我的应用程序中使用jax-rs restful web服务和子资源定位器。但是,在将entityManager传递给子资源后,我无法在此子资源中保留任何新对象。

然而,entityManager让我可以查询数据。

这是我的主要资源:

@Path("/registrations")
@Stateless
public class RegistrationsResource {

    @Context
    private UriInfo context;

    @PersistenceContext(unitName="pctx")
    private EntityManager em;

    public RegistrationsResource() {
    }

    //POST method ommited

    @Path("{regKey}")
    public RegistrationResource getRegistrationResource(@PathParam("regKey")
    String regKey) {
        return RegistrationResource.getInstance(regKey, em);
    }

}

这是我的子资源:

public class RegistrationResource {

    private String regKey;
    private EntityManager em;

    private RegistrationResource(String regKey, EntityManager em) {
        this.regKey = regKey;
        this.em = em;
    }

    @Path("securityQuestion")
    @GET
    public String getQuestion() {
        return "iamahuman"+regKey;
    }

    @Path("securityQuestion")
    @POST
    public void postSecurityAnswer(String answer) {
        if(!answer.equals("iamahuman"+regKey)){
            throw new WebApplicationException(Status.BAD_REQUEST);
        }

        //Getting this information works properly
        List<RegistrationEntity> result = em.createNamedQuery("getRegistrationByKey")
            .setParameter("regKey", regKey).getResultList();

        switch(result.size()){
            case 0 :
                throw new WebApplicationException(Status.NOT_FOUND);
            case 1:
                break;
            default:
                throw new WebApplicationException(Status.INTERNAL_SERVER_ERROR);
            }

            RegistrationEntity reg = result.get(0);
            UserEntity newUser = new UserEntity();

            newUser.setHashedPassword(reg.getPwHash(), reg.getSalt());
            newUser.setUsername(reg.getUsername());
            newUser.setName(reg.getName());
            newUser.setSurname(reg.getSurname());

            //CRASHES HERE
            em.persist(newUser);
    }
}

正如您所看到的,它从数据库中获取注册对象,创建新用户进行注册并尝试保留它。但是,em.persist(newUser)抛出TransactionRequiredException。

我的问题是:我应该如何将EntityManager传递给子资源,以便它可以正确地保留新对象?

2 个答案:

答案 0 :(得分:5)

很抱歉再次挖掘这个问题,但我建议如下:

  • 将子资源注释为@Stateless EJB
  • 将@EJB注入成员字段放入父资源类中,如下所示:
        @EJB private RegistrationResource registrationResource;
    
  • 在“getRegistrationResource()”中,不要调用子资源的构造函数,而是返回注入的EJB引用:
    public RegistrationResource getRegistrationResource() {
        return this.registrationResource;
    }

但是,要使其工作,您不能将“@PathParam”作为构造函数参数传递。您必须通过“@Context”或其他@Path声明在子资源中单独访问它 这使您能够以与父资源完全相同的方式在子资源中注入EntityManager,您无需将其传递。

答案 1 :(得分:3)

可能为时已晚,但无论如何...... 当您返回子资源时,您将“留下”无状态bean。当容器管理事务时,从RegistrationsResource返回时提交事务。

Jersey将构建您的子资源,但它不是无状态bean,因此您不会拥有容器管理的事务。因此例外。

我建议你把你的业务逻辑放在一个新的类中,然后你创建一个无状态的bean。在这里,您可以执行所有数据库内容,然后始终在容器管理事务中处理。