public ICrudOperations Update(DbConcurencyUpdateOptions concurencyUpdateOptions, User user) { if (user.IsAdmin == false) { throw new FaultException <NoRightsToChangeDataException>(new NoRightsToChangeDataException(user, this.GetType().Name)); } concurencyUpdateOptions = DbConcurencyUpdateOptions.ClientPriority; Period result = null; try { using (CFAPContext ctx = new CFAPContext()) { ctx.Configuration.ProxyCreationEnabled = false; ctx.Periods.Attach(this); ctx.Entry(this).State = System.Data.Entity.EntityState.Modified; ctx.SaveChanges(concurencyUpdateOptions); result = (from p in ctx.Periods where p.Month == this.Month && p.Year == this.Year select p).Single(); } } catch (DbEntityValidationException ex) { throw new FaultException <DataNotValidException>(new DataNotValidException(ex.EntityValidationErrors)); } catch (Exception ex) { throw new FaultException <DbException>(new DbException(ex)); } return(result); }
private Transport RunOperation(ICrudOperations entity, User user, DbConcurencyUpdateOptions concurencyUpdateOptions, CrudOperation operation, Filter filter) { Transport result = new Transport(); switch (operation) { case CrudOperation.Add: result.Single = entity.Add(concurencyUpdateOptions, user); break; case CrudOperation.Update: result.Single = entity.Update(concurencyUpdateOptions, user); break; case CrudOperation.Delete: result.Single = entity.Delete(concurencyUpdateOptions, user); break; case CrudOperation.Select: Type type = entity.GetType(); var ob = (ICrudOperations)Activator.CreateInstance(type); result.Collection = ob.Select(filter, user); break; } return(result); }
public int SaveChanges(DbConcurencyUpdateOptions concurrencyOption) { int result = 0; result = SaveChangesFullData(concurrencyOption); return(result); }
public ICrudOperations Add(DbConcurencyUpdateOptions concurencyUpdateOptions, User user) { concurencyUpdateOptions = DbConcurencyUpdateOptions.ClientPriority; if (user.IsAdmin == false) { throw new FaultException <NoRightsToChangeDataException>(new NoRightsToChangeDataException(user, this.GetType().Name)); } bool isExists = false; try { using (CFAPContext ctx = new CFAPContext()) { var existsPeriod = (from p in ctx.Periods where p.Month == this.Month && p.Year == this.Year select p).FirstOrDefault(); if (existsPeriod != null) { isExists = true; } } } catch (Exception ex) { throw new FaultException <DbException>(new DbException(ex)); } if (isExists) { return(Update(concurencyUpdateOptions, user)); } Period result = null; try { using (CFAPContext ctx = new CFAPContext()) { ctx.Configuration.ProxyCreationEnabled = false; ctx.Periods.Add(this); ctx.SaveChanges(concurencyUpdateOptions); result = (from p in ctx.Periods where p.Month == this.Month && p.Year == this.Year select p).Single(); } } catch (DbEntityValidationException ex) { throw new FaultException <DataNotValidException>(new DataNotValidException(ex.EntityValidationErrors)); } catch (Exception ex) { throw new FaultException <DbException>(new DbException(ex)); } return(result); }
public Rate UpdateRate(Rate rateToUpdate, User user, DbConcurencyUpdateOptions concurencyUpdateOption) { AuthenticateUser(user); this.ChechIsAdmin(user, typeof(Rate)); using (CFAPContext ctx = new CFAPContext()) { ctx.Configuration.ProxyCreationEnabled = false; try { ctx.Rates.Attach(rateToUpdate); var ratesToUpdateDbVersion = (Rate)ctx.Entry(rateToUpdate).GetDatabaseValues().ToObject(); if (ratesToUpdateDbVersion.ReadOnly) { throw new ReadOnlyException(); } ctx.Entry(rateToUpdate).State = EntityState.Modified; rateToUpdate.CustomValidate(ctx); ctx.SaveChanges(concurencyUpdateOption); } catch (ReadOnlyException) { throw new FaultException <TryChangeReadOnlyFiledException>(new TryChangeReadOnlyFiledException(typeof(Rate), rateToUpdate.Id, null, user)); } catch (DbUpdateConcurrencyException ex) { var currentValue = rateToUpdate; var dbValue = (Rate)ex.Entries.Single().GetDatabaseValues().ToObject(); ConcurrencyException <Rate> concurrencyException = new ConcurrencyException <Rate>(dbValue, currentValue); throw new FaultException <ConcurrencyException <Rate> >(concurrencyException); } catch (DbEntityValidationException ex) { throw new FaultException <DataNotValidException>(new DataNotValidException(ex.EntityValidationErrors)); } catch (Exception ex) { throw new FaultException <DbException>(new DbException(ex)); } } return(rateToUpdate); }
public BudgetItem UpdateBudgetItem(BudgetItem budgetItemToUpdate, User user, DbConcurencyUpdateOptions concurencyUpdateOption) { AuthenticateUser(user); this.ChechIsAdmin(user, typeof(BudgetItem)); using (CFAPContext ctx = new CFAPContext()) { ctx.Configuration.ProxyCreationEnabled = false; try { ctx.BudgetItems.Attach(budgetItemToUpdate); var budgetItemToUpdateDbVersion = (BudgetItem)ctx.Entry(budgetItemToUpdate).GetDatabaseValues().ToObject(); if (budgetItemToUpdateDbVersion.ReadOnly) { throw new ReadOnlyException(); } ctx.Entry(budgetItemToUpdate).State = EntityState.Modified; ctx.SaveChanges(concurencyUpdateOption); } catch (ReadOnlyException) { throw new FaultException <TryChangeReadOnlyFiledException>(new TryChangeReadOnlyFiledException(typeof(BudgetItem), budgetItemToUpdate.Id, budgetItemToUpdate.ItemName, user)); } catch (DbUpdateConcurrencyException ex) { var currentValue = budgetItemToUpdate; var dbValue = (BudgetItem)ex.Entries.Single().GetDatabaseValues().ToObject(); ConcurrencyException <BudgetItem> concurrencyException = new ConcurrencyException <BudgetItem>(dbValue, currentValue); throw new FaultException <ConcurrencyException <BudgetItem> >(concurrencyException); } catch (DbEntityValidationException ex) { throw new FaultException <DataNotValidException>(new DataNotValidException(ex.EntityValidationErrors)); } catch (Exception ex) { throw new FaultException <DbException>(new DbException(ex)); } } return(budgetItemToUpdate); }
//private void CalculateSummary() //{ // var s = Summaries.Local; // foreach (var summary in Summaries.Local) // { // if (this.Entry<Summary>(summary).State == EntityState.Added // || this.Entry<Summary>(summary).State == EntityState.Modified) // { // summary.SetSummaDollar(); // } // } //} private int SaveChangesFullData(DbConcurencyUpdateOptions concurrencyOption) { //CalculateSummary(); //Алгоритм изменен. Вызов метода больше не актуален. int result = 0; bool saveFailed; do { saveFailed = false; try { result = base.SaveChanges(); } catch (DbUpdateConcurrencyException ex) { saveFailed = true; var entry = ex.Entries.Single();//Перезагрузка исходного значения свойства сущности с БД для решения проблемы оптимистичного параллелизма. switch (concurrencyOption) { case DbConcurencyUpdateOptions.None: throw ex; //break; case DbConcurencyUpdateOptions.ClientPriority: entry.OriginalValues.SetValues(entry.GetDatabaseValues()); //Приоритет клиента break; case DbConcurencyUpdateOptions.DatabasePriority: entry.Reload(); //Приоритет БД break; } } } while (saveFailed); return(result); }
public Summary RemoveSummary(Summary summary, User user, DbConcurencyUpdateOptions concurencyUpdateOption) { AuthenticateUser(user); bool canReadSummary = false; foreach (var g in user.UserGroups) { if (g.CanReadAccountablesSummary) { canReadSummary = true; } } if (!canReadSummary) { throw new FaultException <AuthenticateFaultException>(new AuthenticateFaultException(user)); } if (summary.CheckPeriodIsUnlocked() == false) { throw new FaultException <PeriodIsLockedException>(new PeriodIsLockedException(summary.SummaryDate)); } if (concurencyUpdateOption == DbConcurencyUpdateOptions.DatabasePriority) { throw new FaultException <InvalidOperationException>(new InvalidOperationException("Работа опрации в режиме DbConcurencyUpdateOptions.DatabasePriority не имеет смысла.")); } Summary result = null; using (CFAPContext ctx = new CFAPContext()) { if (summary.UserLastChanged == null || summary.UserLastChanged.Id != user.Id) { summary.UserLastChanged = user; } summary.SetRelationships(ctx); //Когда сущность проходит через SaveChanges все ассоциации обнуляються. Так как с базы данных после удаления обькт получить невозможно - единственный способ сохранить его состояние через клонирование. result = (Summary)summary.Clone(); try { ctx.Summaries.Attach(summary); var summaryDbVersion = (Summary)ctx.Entry(summary).GetDatabaseValues().ToObject(); if (summaryDbVersion.ReadOnly) { throw new ReadOnlyException(); } ctx.Entry(summary).State = EntityState.Deleted; ctx.SaveChanges(concurencyUpdateOption); //Ссылочные свойства (связи в БД) больше не отслеживаютсья контекстом и подлежать востановлению только вручную, через явное указание Id и поиск в БД сущности связи по идентификатору. } catch (ReadOnlyException) { throw new FaultException <TryChangeReadOnlyFiledException>(new TryChangeReadOnlyFiledException(typeof(Summary), summary.Id, null, user)); } catch (DbUpdateConcurrencyException ex) { Summary dbSummary = (Summary)ex.Entries.Single().GetDatabaseValues().ToObject(); Summary currentSummary = result; //Метод Load не загрузит ссылочные свойства без конкретного указания Id. //Свойство коллекции UserGroups является связью многие-ко-многим. Экземпляр сущности не содержит ключи этих связей в виде элементарных типов данных. //dbSummary.UserGroups = result.UserGroups; dbSummary = LoadRelationshipsDbSummary(dbSummary); ConcurrencyException <Summary> concurrencyException = new ConcurrencyException <Summary>(dbSummary, currentSummary); throw new FaultException <ConcurrencyException <Summary> >(concurrencyException); } catch (NullReferenceException ex) { throw new FaultException <FiledDeletedException>(new FiledDeletedException(ex)); } catch (Exception ex) { throw new FaultException <DbException>(new DbException(ex)); } } return(result); }
public Summary UpdateSummary(Summary summary, User user, DbConcurencyUpdateOptions concurencyUpdateOption) { //Изменение значения поля ReadOnly осуществляеться другой операцией службы AuthenticateUser(user); bool canReadSummary = false; foreach (var g in user.UserGroups) { if (g.CanReadAccountablesSummary) { canReadSummary = true; } } if (!canReadSummary) { throw new FaultException <AuthenticateFaultException>(new AuthenticateFaultException(user)); } if (summary.CheckPeriodIsUnlocked() == false) { throw new FaultException <PeriodIsLockedException>(new PeriodIsLockedException(summary.SummaryDate)); } //Пользователь который изменил данные устанавливается по факту if (summary.UserLastChanged == null || summary.UserLastChanged.Id != user.Id) { summary.UserLastChanged = user; } summary.SetAutoValues(); Summary result = null; using (CFAPContext ctx = new CFAPContext()) { ctx.Configuration.ProxyCreationEnabled = false; try { //Группы пользователей уже существуют в сущности //Если, по какой то причине их нет - будет исключение валидации summary.SetRelationships(ctx); //В данном случае AddOrUpdate не сработает в выдаче исключения оптимистичного паралелизма. Он перезагружет сущности в контекс и формирует уже актуальное поле RowVersion ctx.Summaries.Attach(summary); var summaryDbVersion = (Summary)ctx.Entry(summary).GetDatabaseValues().ToObject(); if (summaryDbVersion.ReadOnly) { throw new ReadOnlyException(); } ctx.Entry(summary).State = EntityState.Modified; //Ручной запуск валидации необходм так как при модификации данных связи с сущностями помечены как EntityState.Unchanged //В итоге, без ручной валидации, при неверных данных или значениях null будет исключения при внесении данных в бд, а валидация EF ничего не заметит summary.CustomValidate(ctx); ctx.SaveChanges(concurencyUpdateOption); result = (from s in ctx.Summaries where s.Id == summary.Id select s).Single(); } catch (ReadOnlyException) { throw new FaultException <TryChangeReadOnlyFiledException>(new TryChangeReadOnlyFiledException(typeof(Summary), summary.Id, null, user)); } catch (DbUpdateConcurrencyException ex) { Summary dbSummary = (Summary)ex.Entries.Single().GetDatabaseValues().ToObject(); Summary currentSummary = summary; dbSummary = LoadRelationshipsDbSummary(dbSummary); ConcurrencyException <Summary> concurrencyException = new ConcurrencyException <Summary>(dbSummary, currentSummary); throw new FaultException <ConcurrencyException <Summary> >(concurrencyException); } catch (DbEntityValidationException ex) { throw new FaultException <DataNotValidException>(new DataNotValidException(ex.EntityValidationErrors)); } catch (NullReferenceException ex) { throw new FaultException <FiledDeletedException>(new FiledDeletedException(ex)); } catch (Exception ex) { throw new FaultException <DbException>(new DbException(ex)); } } return(result); }
public Transport MakeOperation(ICrudOperations entity, User user, DbConcurencyUpdateOptions concurencyUpdateOptions, CrudOperation operation, Filter filter) { AuthenticateUser(user); return(RunOperation(entity, user, concurencyUpdateOptions, operation, filter)); }
public ICrudOperations Delete(DbConcurencyUpdateOptions concurencyUpdateOptions, User user) { throw new NotImplementedException(); }