Вложенный пользовательский интерфейс, требующий вложенных транзакций в ActiveRecord

У меня есть диалог D1, который редактирует тип модели M1, и другой диалог D2, который редактирует тип модели M2. Одна из вещей, которые содержит M2, - это ссылка на M1, и поэтому для удобства пользователя D2 содержит кнопку, которая запускает D1, в дополнение к тому, что D1 и D2 доступны с верхнего уровня.

Каждый из D1 и D2 создает TransactionScope (с помощью TransactionMode.New) при входе изменяет объекты модели (M1 или M2), когда пользователь взаимодействует с диалоговым окном, и выполняет фиксацию / откат в зависимости от ситуации, когда пользователь нажимает кнопки OK/ Отмена.

Это прекрасно работает в изоляции, когда диалоги открываются с верхнего уровня.

Когда D1 открывается изнутри D2, ожидаемое поведение состоит в том, что нажатие OK в D1 должно немедленно сохранить в базе данных, а изменения в M1 должны стать видимыми для D2. (D2 может затем сохранить или отменить свои собственные изменения в M2, не влияя на M1.)

На самом деле происходит то, что изменения в M1, похоже, сохраняются в базе данных, когда D1 закрывается (и до того, как D2 закрывается), но D2 не может видеть изменения в M1 - возможно, потому что сеанс / транзакция в D2 считает, что он имеет современные объекты, поэтому не запрашивает базу данных, даже когда FindFirst и т. д. позвонить. Есть ли способ заставить его (не теряя изменений в M2)?

(Другое странное поведение заключается в том, что D1 и D2 подключаются к TransactionScope.OnCompleted, но это срабатывает только на D1, когда D2 закрывается для случая, когда D1 вызывается из D2.)

1 ответ

Решение

Хорошо, я думаю, что нашел что-то, что работает, но это немного уродливо, так что я все еще интересуюсь лучшими ответами.

Хитрость заключалась в том, чтобы модифицировать D1 таким образом, чтобы при его вызове во вложенном состоянии после совершения собственной транзакции (и, таким образом, возврата в область транзакции D2)Find объекты, которые он только что изменил и Refresh их.

И он может обнаружить, что это в этом вложенном сценарии, потому что OnCompleted не позвонили, когда это Disposeсделаю свою собственную сделку.

(Одна из проблем этого решения заключается в том, что оно, по-видимому, не будет работать для трехслойного сценария вложения, если только самый внутренний слой не сможет передать свой список измененных объектов вплоть до самого внешнего слоя, что становится по-настоящему уродливым. В моем случае мне понадобились только два слоя.)

Другие вопросы по тегам