Я столкнулся со странной ситуацией, где последовательность запросов, зарегистрированных в Django и Postgres, отличается при использовании select_for_update()
внутри transaction.atomic()
.
В основном у меня есть ModelForm
, где я проверяю cleaned_data
на базу данных для дублирования запроса. А затем в режиме создания представления form_valid()
я сохраняю экземпляр. Чтобы выполнить операцию внутри одной транзакции, я переопределяю метод post()
и обертываю эти два вызова метода внутри transaction.atomic()
.
Здесь код для того, что я сказал выше:
# Form
class MenuForm(forms.ModelForm):
def __init__(self, *args, **kwargs):
user_id = kwargs.pop('user_id', None)
super(MenuForm, self).__init__(*args, **kwargs)
def clean(self):
cleaned_data = super(MenuForm, self).clean()
dish_name = cleaned_data.get('dish_name')
menus = Menu.objects.select_for_update().filter(user_id=self.user_id)
for menu in menus:
if menu.dish_name == dish_name:
self.add_error('dish_name', 'Dish already exists')
return cleaned_data
return cleaned_data
# CreateView
class MenuCreateView(CreateView):
form_class = MenuForm
def get_form_kwargs(self):
kwargs = super(MenuCreateView, self).get_form_kwargs()
kwargs.update({'user_id': self.request.session.get('user_id')})
return kwargs
def form_valid(self, form):
user = User.objects.get(id=self.request.session.get('user_id'))
form.instance.user = user
return super(MenuCreateView, self).form_valid(form)
def post(self, request, *args, **kwargs):
form = self.get_form()
with transaction.atomic():
if form.is_valid():
return self.form_valid(form)
else:
return self.form_invalid(form)
Теперь предположим, что я запускаю два запроса одновременно, чтобы создать меню с одним и тем же блюдом. Я ожидаю, что второй запрос потерпит неудачу. Но они оба проходят. Похоже, вторая транзакция не видит изменений, выполненных в предыдущей транзакции. Из-за этого общая сумма menus
остается неизменной как в транзакции, возвращаемой select_for_update()
.
Учитывая, что уровень изоляции по умолчанию Postgres равен READ COMMITTED
, я ожидаю, что изменения будут видимыми. Итак, я попытался выполнить регистрацию запросов, чтобы увидеть, что COMMIT; увольняется в нужное время. Здесь журнал запросов django и postgres:
Журнал Django:
SELECT "menu"."id", "menu"."dish_id", "menu"."dish_name" FROM "menu" WHERE ("menu"."dish_name" = "Test Dish") FOR UPDATE; args=("Test Dish")
INSERT INTO "menu" ("dish_id", "dish_name") VALUES (2, "Test Dish") RETURNING "menu"."id"; args=(2, "Test Dish")
SELECT "menu"."id", "menu"."dish_id", "menu"."dish_name" FROM "menu" WHERE ("menu"."dish_name" = "Test Dish") FOR UPDATE; args=("Test Dish")
INSERT INTO "menu" ("dish_id", "dish_name") VALUES (2, "Test Dish") RETURNING "menu"."id"; args=(2, "Test Dish")
Журнал Postgres:
<2016-03-18 17:55:46.176 IST 0 2/31 56ebf3ca.aac0>LOG: statement: SHOW default_transaction_isolation
<2016-03-18 17:55:46.177 IST 0 2/32 56ebf3ca.aac0>LOG: statement: SET TIME ZONE 'UTC'
<2016-03-18 17:55:46.178 IST 0 2/33 56ebf3ca.aac0>LOG: statement: SELECT t.oid, typarray
FROM pg_type t JOIN pg_namespace ns
ON typnamespace = ns.oid
WHERE typname = 'hstore';
<2016-03-18 17:55:46.182 IST 0 2/34 56ebf3ca.aac0>LOG: statement: BEGIN
<2016-03-18 17:55:46.301 IST 0 3/2 56ebf3ca.aac1>LOG: statement: SHOW default_transaction_isolation
<2016-03-18 17:55:46.302 IST 0 3/3 56ebf3ca.aac1>LOG: statement: SET TIME ZONE 'UTC'
<2016-03-18 17:55:46.302 IST 0 3/4 56ebf3ca.aac1>LOG: statement: SELECT t.oid, typarray
FROM pg_type t JOIN pg_namespace ns
ON typnamespace = ns.oid
WHERE typname = 'hstore';
<2016-03-18 17:55:46.312 IST 0 3/5 56ebf3ca.aac1>LOG: statement: BEGIN
<2016-03-18 17:55:46.963 IST 0 3/5 56ebf3ca.aac1>LOG: statement: SELECT "menu"."id", "menu"."dish_id", "menu"."dish_name" FROM "menu"
WHERE ("menu"."dish_name" = "Test Dish") FOR UPDATE
<2016-03-18 17:55:46.964 IST 0 2/34 56ebf3ca.aac0>LOG: statement: SELECT "menu"."id", "menu"."dish_id", "menu"."dish_name" FROM "menu"
WHERE ("menu"."dish_name" = "Test Dish") FOR UPDATE
<2016-03-18 17:55:47.040 IST 23712 3/5 56ebf3ca.aac1>LOG: statement: INSERT INTO "menu" ("dish_id", "dish_name") VALUES (2, "Test Dish")RETURNING "menu"."id"
<2016-03-18 17:55:47.061 IST 23712 3/5 56ebf3ca.aac1>LOG: statement: COMMIT
<2016-03-18 17:55:47.229 IST 23713 2/34 56ebf3ca.aac0>LOG: statement: INSERT INTO "menu" ("dish_id", "dish_name") VALUES (2, "Test Dish")RETURNING "menu"."id"
<2016-03-18 17:55:47.231 IST 23713 2/34 56ebf3ca.aac0>LOG: statement: COMMIT
Postgres.conf:
max_connections = 100
log_destination = 'stderr'
logging_collector = on
log_directory = 'pg_log'
log_line_prefix = '<%m %x %v %c>'
log_statement = 'all'
Как вы можете видеть, порядок запросов SELECT и INSERT не одинаковый в обоих журналах. Я не могу понять, почему это произойдет. Кроме того, если вы заметили, session_id для запросов SELECT в журнале Postgres отличается. Может ли это объяснить что-то здесь?
И если это ожидаемое поведение, как я могу решить основную проблему здесь? Избегайте одновременных запросов INSERT на основе существующей записи.
UPDATE
Я не упомянул, что фактическая логика для игнорирования дублированного меню основана не только на имени тарелки. Приведенный выше пример упрощен.
Учитывая модель меню как:
class Menu:
user_id = models.IntegerField()
dish = models.ForeignKey(Dish)
order_start_time = models.DateTimeField()
order_end_time = models.DateTimeField()
Фактическая логика выглядит следующим образом:
- Извлеките все меню с помощью
dish_name
из db. - Отметьте
order_start_time
иorder_end_time
для всех этих меню и посмотрите, не пересекаются ли какие-либо из них сorder_start_time
иorder_end_time
для нового меню. Если конфликт найден, избегайте добавления.
Итак, мы можем добавить два меню для блюда - d1
, имеющих окно заказа - [9am-10am]
и [2pm-3pm]
.