在数据库外部实现事务的模式

发布于 2024-07-13 16:15:33 字数 272 浏览 10 评论 0原文

我必须发送电子邮件、写入文件并调用网络服务。 为了保持一致性,所有步骤都必须发生。 如果任何步骤引发异常或错误,则必须回滚所有步骤。

在我开始使用自己的对象 ACID 引擎之前,是否有任何普遍接受的模式用于在对象级别实现 ACID 语义?

更好的是,是否有任何现有的库可以用于 .NET 平台?

编辑:我知道发送电子邮件无法撤消,但无法连接到 SMTP 服务器会导致整个事务终止。 另外,我希望它可以扩展以供将来的操作使用。

I have to send an email, write to a file, and call a web service. To maintain consistency, all steps must happen. If any step throws an exception or errors out, all steps must be rolled back.

Before I go rolling my own object ACID engine, are there any commonly accepted patterns for implementing ACID semantics at the object level?

Better yet, are there any existing libraries I can use for the the .NET platform?

Edit: I know sending an email can't be undone, but failing to connect to the SMTP server is cause to kill the whole transaction. Also, I'd like this to be extensible for use with future actions.

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

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

发布评论

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

评论(7

一袭白衣梦中忆 2024-07-20 16:15:33

我上次看到这样的事情已经是几年前的事了。 我记得的一点是它使用命令模式并将每个命令对象存储在队列中。 我认为这是一个 LIFO 堆栈。

因此,如果“事务”失败,引擎将弹出命令对象,撤消该命令,然后销毁该命令对象。 重复直到堆栈为空。 如果“事务”成功,堆栈将被清除。

不幸的是,我不记得更多了。

CSLA.NET 实现了类似的撤消堆栈。 这是我能立即想到的唯一一个包含代码的示例。

The last time I saw something like this was several years ago. The little bit that I remember about it is that it was using the command pattern and storing each command object in a queue. I think it was a LIFO stack.

So if the "transaction" failed, the engine would pop off a command object, undo the command, then destroy the command object. Repeat until the stack was empty. The stack got cleared if the "transaction" was successful.

Unfortunately, I don't remember more than that.

CSLA.NET implements a similar undo stack. That's the only example with code that I can think off the top of my head.

疯狂的代价 2024-07-20 16:15:33

Windows Workflow Foundation 有一个补偿(使用复合活动)的概念ACID 语义可能不合适。当然,它也支持 ACID 事务。

一个好问题是为什么要费心
赔偿? 不是一大酸吗
事务自动回滚
同样好? ACID 事务是
当操作发生时最合适
在同一数据库内或在
相同的信息系统。 也是
运营结束时最合适
迅速地。 当不同的公司和
涉及服务,定义
ACID 语义方面的过程
往往具有挑战性。 为了它成为
隔离且耐用,你必须保持
不同公司的所有资源
在任务期间锁定。
这往往是不合理的,
尤其是当任务很长时。 为了它
为了保持一致和原子性,你需要
特别补偿代码。

Windows Workflow Foundation has a concept of compensation (using Composite activity) when ACID semantics might not be appropriate..Off course, it has support for ACID transactions as well.

A good question is why bother with
compensation? Isn’t one big ACID
transaction with automatic rollback
just as good? An ACID transaction is
most appropriate when operations occur
within the same database or within the
same information system. It is also
most appropriate when operations end
quickly. When different companies and
services are involved, defining the
process in terms of the ACID semantics
is often challenging. For it to be
isolated and durable, you have to keep
all resources of different companies
locked for the duration of the task.
This is frequently unreasonable,
especially if the task is long. For it
to be consistent and atomic, you need
ad hoc compensation code.

宫墨修音 2024-07-20 16:15:33

不严重依赖外部库的最简单技术是流行。 通过使用序列化定期检查点来拍摄状态快照,然后通过序列化针对数据的每个副作用操作的足够信息来维护日志,以便稍后重复。 如果出现问题,请重新加载最近的检查点,然后重新应用该点之后写入的所有日志记录。

对于更复杂的东西,请尝试软件事务内存。 用当前的语言实现可能有点笨拙,但功能非常强大,并且还可以为您提供一些额外的并发技术。

对于访问 Web 服务或发送电子邮件等不可逆转的操作,您需要使用补偿事务:进行另一个 Web 服务调用以取消或更新前一个的结果,或者可能发送另一封电子邮件,通知收件人事情没有按预期进行。

The simplest technique without relying heavily on an external library is prevalence. Periodically checkpoint by using serialization to take a snapshot of your state, then maintain a journal by serializing enough information on every side-effectful operation against your data to repeat it later. If something blows up, reload the most recent checkpoint, then re-apply all journal records written after that point.

For something more sophisticated, try software transactional memory. It may be somewhat clumsy to implement in current languages, but is quite powerful and may give you some additional concurrency techniques as well.

For irreversible operations like accessing a Web service or sending an email, you'll need to use compensating transactions: make another Web service call to cancel or update the results of the previous one, or perhaps send another email advising the recipient that things didn't work as intended.

悸初 2024-07-20 16:15:33

由于您无法取消发送电子邮件,并且写入文件相对便宜,因此我只会按正确的顺序执行这些操作:

  1. 尝试写入文件/写入文件。 如果不成功,则停止,否则继续:
  2. 调用 Web 服务。 如果不成功,请删除该文件并停止,否则继续:
  3. 发送电子邮件 - 电子邮件无论如何都是异步的,因此您永远不会真正知道它是否已发送,因为大多数电子邮件服务器都设置为重试几个如果发生错误,并且即使成功,您也永远不会收到电子邮件已发送完毕的确认信息,需要几天的时间。

Since you can't un-send an e-mail, and it's relatively inexpensive to write a file, I'd just do those things in the proper order:

  1. Try to write the file/write the file. If unssuccessful, stop, otherwise continue to:
  2. Call the web service. If unsuccessful, delete the file and stop, otherwise continue to:
  3. Send e-mail -- email is asynchronous anyhow, so you'd never really know if it was sent or not since most e-mail servers are set to retry for a couple of days if an error occurs and you never get back an acknowledgment that the e-mail went through even if it was successful.
浪菊怪哟 2024-07-20 16:15:33

一种想法是使用 JMS 作为“引擎”,您可以利用 JMS 事务(它可以加入现有事务,例如数据库事务)。 这开始导致异步事件驱动的架构,这可能是一件好事,除非我们谈论的是简单的应用程序 - 在这种情况下,您可能不需要问这个问题。

简单的帐户创建就是一个例子。 为此,您希望将帐户信息保存到数据库,并向用户发送一封电子邮件以进行激活 - 但由于显而易见的原因,您希望它们处于同一事务中。

您不应该将电子邮件发送代码放入事务中,因为即使您可以发送电子邮件 - 数据库事务提交也可能会由于某种原因而失败。
您也不应该将电子邮件发送放在事务之外(提交后),因为电子邮件发送可能会失败,从而导致孤立帐户。

因此,要在这种情况下使用 JMS,请将 JMS 发送代码放入数据库事务中,并让它加入该事务。 保证您的消息传递。 另一端有一些东西消耗队列发送电子邮件。 如果电子邮件发送失败,最好的选择是记录/发出警报 - JMS 将回滚并将消息放回到队列中以供以后使用。 即,一旦您希望解决问题,就尝试重新发送电子邮件。

关键是——数据库记录一致并且电子邮件最终发送。

One idea is to use JMS as the 'engine' and you can utilize JMS transactions (which can join existing transactions e.g. DB transaction). This starts to lead towards an async event-driven architecture which is probably a good thing unless we are talking simple apps - in which case you probably don't need to be asking the question.

An example of this would be simple account creation. For this you want to persist account information to the DB and also send the user an email for activation - but you want them in the same transaction for obvious reasons.

You should not put email sending code within the transaction because even though you may send the email - db transaction commit may fail for one reason or another.
You also should not put email sending outside of the transaction (after commit) because email sending may fail leading to an orphan account.

So to use JMS in this scenario - put JMS sending code within the DB transaction and have it join that transaction. You are guaranteed message delivery. On the other end have something consuming the queue sending emails out. In case of email send failure, best option is to log/raise an alert - JMS will roll back and put message back into the queue for later consumption. i.e. to try and resend email once you have hopefully fixed whatever the issue is.

The critical thing is - DB record is consistent and email is eventually sent.

醉酒的小男人 2024-07-20 16:15:33

两个想法:

Two thoughts:

太阳哥哥 2024-07-20 16:15:33

此外,最近还发布了一个实验项目 STM .NET。 该项目向 C# 添加了事务内存。 它实际上修改了 CLR 以支持这一点。

Also, A experimental project STM .NET was recently released. This projects adds transaction memory to C#. It actually modifies the CLR to support this.

~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文