Handle Sqlite automatic rollbacks gracefully (#3996)

This commit is contained in:
zah 2022-10-05 01:40:46 +03:00 committed by GitHub
parent ce915c0a03
commit 576b999387
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23

View File

@ -400,6 +400,22 @@ template withManyWrites*(dbParam: BeaconChainDB, body: untyped) =
if commit: if commit:
expectDb db.db.exec("COMMIT TRANSACTION;") expectDb db.db.exec("COMMIT TRANSACTION;")
else: else:
# https://www.sqlite.org/lang_transaction.html
#
# For all of these errors, SQLite attempts to undo just the one statement
# it was working on and leave changes from prior statements within the same
# transaction intact and continue with the transaction. However, depending
# on the statement being evaluated and the point at which the error occurs,
# it might be necessary for SQLite to rollback and cancel the entire transaction.
# An application can tell which course of action SQLite took by using the
# sqlite3_get_autocommit() C-language interface.
#
# It is recommended that applications respond to the errors listed above by
# explicitly issuing a ROLLBACK command. If the transaction has already been
# rolled back automatically by the error response, then the ROLLBACK command
# will fail with an error, but no harm is caused by this.
#
if isInsideTransaction(db.db): # calls `sqlite3_get_autocommit`
expectDb db.db.exec("ROLLBACK TRANSACTION;") expectDb db.db.exec("ROLLBACK TRANSACTION;")
proc new*(T: type BeaconChainDB, proc new*(T: type BeaconChainDB,