@defmac with-sqlite-transaction db body@dots{}
Like @code{progn} (@pxref{Sequencing}), but executes @var{body} with a
-transaction held, and commits the transaction at the end.
+transaction held, and commits the transaction at the end if @var{body}
+completes normally. If @var{body} signals an error, or committing the
+transaction fails, the changes in @var{db} performed by @var{body} are
+rolled back. The macro returns the value of @var{body} if it
+completes normally and commit succeeds.
@end defmac
@defun sqlite-pragma db pragma
\f
* Incompatible Lisp Changes in Emacs 29.2
++++
+** 'with-sqlite-transaction' rolls back changes if its BODY fails.
+If the BODY of the macro signals an error, or committing the results
+of the transaction fails, the changes will now be rolled back.
+
\f
* Lisp Changes in Emacs 29.2
;;; Code:
(defmacro with-sqlite-transaction (db &rest body)
- "Execute BODY while holding a transaction for DB."
+ "Execute BODY while holding a transaction for DB.
+If BODY completes normally, commit the changes and return
+the value of BODY.
+If BODY signals an error, or transaction commit fails, roll
+back the transaction changes."
(declare (indent 1) (debug (form body)))
(let ((db-var (gensym))
- (func-var (gensym)))
+ (func-var (gensym))
+ (res-var (gensym))
+ (commit-var (gensym)))
`(let ((,db-var ,db)
- (,func-var (lambda () ,@body)))
+ (,func-var (lambda () ,@body))
+ ,res-var ,commit-var)
(if (sqlite-available-p)
(unwind-protect
(progn
(sqlite-transaction ,db-var)
- (funcall ,func-var))
- (sqlite-commit ,db-var))
- (funcall ,func-var)))))
+ (setq ,res-var (funcall ,func-var))
+ (setq ,commit-var (sqlite-commit ,db-var))
+ ,res-var)
+ (or ,commit-var (sqlite-rollback ,db-var))))
+ (funcall ,func-var))))
(provide 'sqlite)