Uważam, że odpowiedź @ AnujGupta jest poprawna. Jednak wycofanie może samo w sobie wywołać wyjątek, który należy złapać i obsługiwać:
from django.db import transaction, DatabaseError
try:
a.save()
except DatabaseError:
try:
transaction.rollback()
except transaction.TransactionManagementError:
# Log or handle otherwise
Jeśli okaże się, że przepisujesz ten kod w różnych save()
lokalizacjach, możesz wyodrębnić metodę:
import traceback
def try_rolling_back():
try:
transaction.rollback()
log.warning('rolled back') # example handling
except transaction.TransactionManagementError:
log.exception(traceback.format_exc()) # example handling
Na koniec możesz go upiększyć za pomocą dekoratora, który chroni metody wykorzystujące save()
:
from functools import wraps
def try_rolling_back_on_exception(fn):
@wraps(fn)
def wrapped(*args, **kwargs):
try:
return fn(*args, **kwargs)
except:
traceback.print_exc()
try_rolling_back()
return wrapped
@try_rolling_back_on_exception
def some_saving_method():
# ...
model.save()
# ...
Nawet jeśli implementujesz dekorator powyżej, nadal wygodnie jest zachować go try_rolling_back()
jako wyodrębnioną metodę na wypadek, gdybyś musiał użyć go ręcznie w przypadkach, w których wymagana jest specyficzna obsługa, a ogólna obsługa dekoratora nie jest wystarczająca.