在 MessageDrivenBean 中将事务标记为仅回滚,而不重试消息

发布于 2024-12-25 13:46:09 字数 1137 浏览 0 评论 0原文

我有以下 MessageDrivenBean:

@MessageDriven(mappedName = "jms/...", activationConfig = {
    @ActivationConfigProperty(propertyName = "destinationType", propertyValue = "javax.jms.Queue"),
    @ActivationConfigProperty(propertyName = "endpointExceptionRedeliveryAttempts", propertyValue = "5"),
    @ActivationConfigProperty(propertyName = "endpointExceptionRedeliveryInterval", propertyValue = "1000")
})
public class MyMessageListener implements MessageListener {

    @Resource
    private MessageDrivenContext context;

    @Override
    @TransactionAttribute(REQUIRED)
    public void onMessage(Message message) {            
    }
}

onMessage 方法中,我做了一些处理,这在这里并不重要。在某些情况下,我想放弃处理。在这种情况下,我想将事务标记为仅回滚,并且不想重试处理。

至少在 Glassfish 3.1.1 上,如果我在 MessageDrivenContext 上调用 context.setRollbackOnly() 并结束处理而不引发异常,则消息会以非常、间隔很短。所以这对我来说是没有选择的。

如果我抛出 RuntimeException ,消息将按照我指定的方式重新传递。重发5次,间隔1秒。但就我而言,我根本不想重试处理。

如果我只是简单地结束处理而不引发任何异常,并且不调用 context.setRollbackOnly(),那么事务就会如人们所期望的那样被提交。但我需要回滚事务,因为我可能已经更改了一些 JPA 实体。

我的问题是:如何将事务标记为仅回滚而不触发消息重新传递?

I have the following MessageDrivenBean:

@MessageDriven(mappedName = "jms/...", activationConfig = {
    @ActivationConfigProperty(propertyName = "destinationType", propertyValue = "javax.jms.Queue"),
    @ActivationConfigProperty(propertyName = "endpointExceptionRedeliveryAttempts", propertyValue = "5"),
    @ActivationConfigProperty(propertyName = "endpointExceptionRedeliveryInterval", propertyValue = "1000")
})
public class MyMessageListener implements MessageListener {

    @Resource
    private MessageDrivenContext context;

    @Override
    @TransactionAttribute(REQUIRED)
    public void onMessage(Message message) {            
    }
}

Inside the onMessage method, I do some processing which is not important here. In certain situations, I want to give up on the processing. In that situations I want to mark the transaction as rollback-only and don't want to retry the processing.

At least on Glassfish 3.1.1, if I call context.setRollbackOnly() on the MessageDrivenContext and end the processing without throwing an exception, the message is redelivered infinitely in very, very short intervals. So that is no option for me.

If I throw a RuntimeException the message is redelivered as I have specified it. It is redelivered 5 times with 1 second interval. But in my case, I don't want to retry the processing at all.

If I simply end the processing without throwing any exception and without calling context.setRollbackOnly(), the transaction is commited, as one would expect. But I need to rollback the transaction because I may have already altered some JPA entities.

The question I have is: How can I mark the transaction as rollback-only without triggering a message redelivery?

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(1

岁月打碎记忆 2025-01-01 13:46:09

我看到两个选项:

  1. 使用 @TransactionAttribute(TransactionAttributeType.NOT_SUPPORTED) 注释 bean,使用 bean 管理事务并且不使用 AUTO_ACKNOWLEDGE(在 MDB 中显式确认消息)。如果您的 MDB 是较大 XA 事务的一部分,则不是一个选择,您将没有机会影响全局事务结果。
  2. 检查消息上的 getJMSRedelivered() 标志,然后简单地丢弃重新传递的消息。这里的问题是,在重新交付时,您不知道是否应该忽略重新交付(您说过在某些情况下您确实想重试处理)——您必须临时存储(在数据库或缓存中)ID处理失败的消息,以便您知道是否可以安全地忽略它们。

I see two options:

  1. Annotate the bean with @TransactionAttribute(TransactionAttributeType.NOT_SUPPORTED), use bean-managed transaction and no AUTO_ACKNOWLEDGE (acknowledge the message explicitly in the MDB). Not an option if your MDB is a part of a larger XA transaction, you won't have a chance to influence the global transaction result.
  2. Examine the getJMSRedelivered() flag on message and simply discard the redelivered message. The trouble here is, upon redelivery you don't know if it's a redelivery that you should ignore (you said in some cases you do want to retry processing) — you'd have to store temporarily (in a database, or cache) ids of messages that failed processing, so you know if you may ignore them safely.
~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文