Ответ 1
Посмотрите django-reversion
. Он обеспечивает контроль версий для моделей Django. Может быть легко добавлен в существующий проект.
В нем не используется подход "текущего" указателя. Вместо этого он сериализует объект каждый раз при его сохранении и сохраняет его в отдельной модели Version
с общим внешним ключом, указывающим на этот объект. (Поля отношений по умолчанию сериализуются как первичные ключи.) Кроме того, он позволяет гибко группировать Version
в Revision
.
Итак, вы можете сделать что-то подобное:
- Когда пользователь загружает CSV, просто сохраняйте изменения как обычно, но добавьте
@revision.create_on_success
decorator в функцию, которая выполняет импорт, чтобы любые изменения в записи, сделанные этой функцией, сохранялись в одной версии. - Когда пользователь нажимает "Отменить", вы просто возвращаете последнюю версию.
Вот как это можно сделать:
@revision.create_on_success
def import_csv(request, csv):
# Old versions of all objects save()d here will
# belong to single revision.
def undo_last_csv_import(request):
# First, get latest revision saved by this user.
# (Assuming you create revisions only when user imports a CSV
# and do not version control other data.)
revision = Revision.objects.filter(user=request.user)\
.order_by('-date_created')[0]
# And revert it, delete=True means we want to delete
# any newly added records as well
revision.revert(delete=True)
Он основан на том, что вы создаете ревизии только тогда, когда пользователь импортирует CSV. Это означает, что если вы планируете также управлять версиями других данных, вам необходимо реализовать какой-то флаг, с помощью которого вы можете получить записи, затронутые последним импортом. Затем вы можете получить запись по этому флагу, получить последнюю сохраненную версию и вернуть всю версию, к которой принадлежит эта версия. Вот так:
def undo_last_csv_import(request):
some_record = Record.objects.by_user(request.user).from_the_last_import()[0]
latest_saved_version_of_some_record = Version.objects.get_for_date(
some_record,
datetime.now(), # The latest saved Version at the moment.
)
# Revert all versions that belong to the same revision
# as the version we got above.
latest_saved_version_of_some_record.revision.revert()
Это не красивое решение, там, безусловно, есть способы сделать это лучше с этим приложением. Я рекомендую взглянуть на код, чтобы лучше понять, как работает django-reversion
- очень хорошо документировано, не удалось найти функцию без docstring. ^ _ ^ Д
(Документация также хороша, но для меня это немного вводит в заблуждение, то есть они пишут Version.objects.get_for_date(your_model, date)
, где ваш_модель на самом деле является экземпляром модели.)
Обновление: django-reversion активно поддерживается, поэтому не стоит больше полагаться на код и лучше проверить их wiki о том, как управлять версиями и версиями за пределами администратора django. Например, комментарии к версии уже поддерживаются, что может немного упростить.