我有一个非常简单的抽象来处理可以回滚(在某种程度上)的一系列 IO 动作,即如果一个动作写入一个文件,那么回滚将删除这个文件,或者如果一个动作创建一个目录树,修剪这将是回滚等。
data IOAction = IOAction {
execute :: IO (),
rollback :: IO ()
}
executeAll :: [IOAction] -> IO ()
executeAll [] = return ()
executeAll (a : as) = do
execute a
executeAll as `catch` rollbackAndRethrow
where
rollbackAndRethrow :: SomeException -> IO ()
rollbackAndRethrow e = rollback a >> throw e
它几乎可以满足我的要求,但我有一种强烈的预感,即有更多可组合和更可靠(在异常处理的意义上)的方法来做到这一点。所以我的问题是我可以使用某个库中已知的 monad 转换器来实现相同的想法吗?
有类似的东西
writeFilesAtomically :: CanRollbackT IO ()
writeFilesAtomically = do
a1 <- (writeFile a str1) `orRollback` removeFile a
a2 <- (writeFile x str2) `orRollback` removeFile x
....
将比当前的解决方案更方便。