Orchard 1.8 - сохранение записи с уникальным полем

В своей миграции я определил один из столбцов таблицы как уникальный. Если я пытаюсь сохранить запись, которая нарушает это ограничение уникальности, я получаю исключение:

A duplicate value cannot be inserted into a unique index.

Чтобы сохранить запись, я использую методы создания и обновления IRepository, потому что она не имеет соответствующей части. После получения вышеупомянутого исключения я ловлю его и затем запускаю AddModelError, чтобы остановить процедуру сохранения. Neverthless Orchard дает мне еще одно исключение после этого:

[AssertionFailure: null id in MyModule.Models.MyRecord entry (don't flush the Session after an exception occurs)]
   NHibernate.Event.Default.DefaultFlushEntityEventListener.CheckId(Object obj, IEntityPersister persister, Object id, EntityMode entityMode) +267
   NHibernate.Event.Default.DefaultFlushEntityEventListener.GetValues(Object entity, EntityEntry entry, EntityMode entityMode, Boolean mightBeDirty, ISessionImplementor session) +95
   NHibernate.Event.Default.DefaultFlushEntityEventListener.OnFlushEntity(FlushEntityEvent event) +139
   NHibernate.Event.Default.AbstractFlushingEventListener.FlushEntities(FlushEvent event) +448
   NHibernate.Event.Default.AbstractFlushingEventListener.FlushEverythingToExecutions(FlushEvent event) +283
   NHibernate.Event.Default.DefaultAutoFlushEventListener.OnAutoFlush(AutoFlushEvent event) +84
   NHibernate.Impl.SessionImpl.AutoFlushIfRequired(ISet`1 querySpaces) +471
   NHibernate.Impl.SessionImpl.List(IQueryExpression queryExpression, QueryParameters queryParameters, IList results) +477
   NHibernate.Impl.AbstractSessionImpl.List(IQueryExpression queryExpression, QueryParameters parameters) +223
   NHibernate.Impl.ExpressionQueryImpl.List() +189
   NHibernate.Linq.DefaultQueryProvider.ExecuteQuery(NhLinqExpression nhLinqExpression, IQuery query, NhLinqExpression nhQuery) +61
   NHibernate.Linq.DefaultQueryProvider.Execute(Expression expression) +262
   NHibernate.Linq.DefaultQueryProvider.Execute(Expression expression) +26
   Remotion.Linq.QueryableBase`1.GetEnumerator() +83
   System.Linq.Enumerable.ToDictionary(IEnumerable`1 source, Func`2 keySelector, Func`2 elementSelector, IEqualityComparer`1 comparer) +173
   System.Linq.Enumerable.ToDictionary(IEnumerable`1 source, Func`2 keySelector) +125
   Orchard.Data.Migration.DataMigrationManager.GetFeaturesThatNeedUpdate() +311
   Orchard.Modules.Data.Migration.<GetNotifications>d__6.MoveNext() +219
   System.Linq.<SelectManyIterator>d__14`2.MoveNext() +507
   System.Collections.Generic.List`1..ctor(IEnumerable`1 collection) +536
   System.Linq.Enumerable.ToList(IEnumerable`1 source) +80
   Orchard.UI.Admin.Notification.NotificationManager.GetNotifications() +151
   Orchard.UI.Admin.Notification.AdminNotificationFilter.OnResultExecuting(ResultExecutingContext filterContext) +380
   System.Web.Mvc.ControllerActionInvoker.InvokeActionResultFilterRecursive(IList`1 filters, Int32 filterIndex, ResultExecutingContext preContext, ControllerContext controllerContext, ActionResult actionResult) +245
   System.Web.Mvc.ControllerActionInvoker.InvokeActionResultFilterRecursive(IList`1 filters, Int32 filterIndex, ResultExecutingContext preContext, ControllerContext controllerContext, ActionResult actionResult) +890
   System.Web.Mvc.ControllerActionInvoker.InvokeActionResultFilterRecursive(IList`1 filters, Int32 filterIndex, ResultExecutingContext preContext, ControllerContext controllerContext, ActionResult actionResult) +890
   System.Web.Mvc.ControllerActionInvoker.InvokeActionResultFilterRecursive(IList`1 filters, Int32 filterIndex, ResultExecutingContext preContext, ControllerContext controllerContext, ActionResult actionResult) +890
   System.Web.Mvc.ControllerActionInvoker.InvokeActionResultFilterRecursive(IList`1 filters, Int32 filterIndex, ResultExecutingContext preContext, ControllerContext controllerContext, ActionResult actionResult) +890
   System.Web.Mvc.ControllerActionInvoker.InvokeActionResultWithFilters(ControllerContext controllerContext, IList`1 filters, ActionResult actionResult) +97
   System.Web.Mvc.Async.<>c__DisplayClass21.<BeginInvokeAction>b__1e(IAsyncResult asyncResult) +241
   System.Web.Mvc.Controller.<BeginExecuteCore>b__1d(IAsyncResult asyncResult, ExecuteCoreState innerState) +29
   System.Web.Mvc.Async.WrappedAsyncVoid`1.CallEndDelegate(IAsyncResult asyncResult) +111
   System.Web.Mvc.Controller.EndExecuteCore(IAsyncResult asyncResult) +53
   System.Web.Mvc.Async.WrappedAsyncVoid`1.CallEndDelegate(IAsyncResult asyncResult) +19
   System.Web.Mvc.MvcHandler.<BeginProcessRequest>b__5(IAsyncResult asyncResult, ProcessRequestState innerState) +51
   System.Web.Mvc.Async.WrappedAsyncVoid`1.CallEndDelegate(IAsyncResult asyncResult) +111
   Orchard.Mvc.Routes.HttpAsyncHandler.EndProcessRequest(IAsyncResult result) +95
   System.Web.CallHandlerExecutionStep.System.Web.HttpApplication.IExecutionStep.Execute() +606
   System.Web.HttpApplication.ExecuteStep(IExecutionStep step, Boolean& completedSynchronously) +288

Если я правильно понимаю, что является причиной этого, Orchard автоматически пытается сбросить запись в базу данных, даже если я добавляю ошибку модели. Я пытался изменить объем транзакции, но это не сработало. Есть ли способ решить эту проблему?

2 ответа

Решение

Вы не можете использовать Unique ограничение при работе с записями части контента. Причина в том, что они создаются рано, когда еще не было передано никаких данных (все свойства пустые / нулевые). Это происходит внутри ContentManager, Следовательно, все свойства таких записей должны быть обнуляемыми и неуникальными.

Но вы можете использовать уникальные / не нулевые ограничения для классов, которые сопоставляются с записями, которые вы создаете и используете через IRepository непосредственно.

Если это для ContentPart, и у вас есть способ создать уникальное значение для вашего поля в.NET, то вы всегда можете использовать событие OnInitializing в вашем ContentHandler с вашей стороны.

Например, это работает для уникального поля Sku в БД:

public class ProductPartHandler: ContentHandler
{
    public ProductPartHandler(IRepository<ProductPartRecord> repository)
    {
        Filters.Add(StorageFilter.For(repository));
        OnInitializing<ProductPart>((context, part) =>
        {
            part.Sku = Guid.NewGuid().ToString();
        });
    }
}

Пока вы можете создать гарантированное уникальное значение некоторого вида, поддерживаемое вашим полем, вы всегда можете установить его в обработчике, чтобы иметь дело с уникальным ограничением, а затем обновить его с реальным желаемым значением.

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