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 UserGroup UpdateUserGroup(UserGroup userGroupForUpdate, User owner) { //Аутентификация пользователя-владельца AuthenticateUser(owner); //Проверка - иммеет ли право владелец добавлять или изменять данные this.ChechIsAdmin(owner, typeof(UserGroup)); //Создание экземпляра контекста using (CFAPContext ctx = new CFAPContext()) { ctx.Configuration.ProxyCreationEnabled = false; try { ctx.Entry(userGroupForUpdate).State = EntityState.Modified; ctx.SaveChanges(DbConcurencyUpdateOptions.ClientPriority); } catch (DbEntityValidationException ex) { throw new FaultException <DataNotValidException>(new DataNotValidException(ex.EntityValidationErrors)); } catch (Exception ex) { throw new FaultException <DbException>(new DbException(ex)); } } return(userGroupForUpdate); }
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); }
public void CustomValidate(CFAPContext ctx) { List <DbEntityValidationResult> validationResults = new List <DbEntityValidationResult>(); var rateResult = ctx.Entry(this).GetValidationResult(); if (!rateResult.IsValid) { validationResults.Add(rateResult); } if (this.DateRate == default(DateTime)) { DbValidationError validationError = new DbValidationError(typeof(DateTime).ToString(), "Значениие не определено"); DbEntityValidationResult dbEntityValidationResult = new DbEntityValidationResult(ctx.Entry(this), new DbValidationError[] { validationError }); validationResults.Add(dbEntityValidationResult); } if (validationResults.Count > 0) { throw new DbEntityValidationException( "Ошибка при проверке данных. Данные могут остутствовать или указаны не верно. Проверте внесенные данные и повторите попытку" , validationResults); } }
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); }
public User UpdateUser(User userForUpdate, User owner) { //Аутентификация пользователя-владельца AuthenticateUser(owner); //Проверка - иммеет ли право владелец добавлять или изменять данные пользователей (User.CanAddNewUser) this.CheckCanChangeUsersData(owner, typeof(User)); if (userForUpdate.UserGroups == null || userForUpdate.UserGroups.Count == 0) { throw new FaultException <UserHasNotGroupsException>(new UserHasNotGroupsException(userForUpdate)); } if (userForUpdate.IsAccountable && userForUpdate.Accountable == null) { throw new FaultException <AccountableUserHasNotAccountableRefferenceException>(new AccountableUserHasNotAccountableRefferenceException(userForUpdate)); } //Создание экземпляра контекста using (CFAPContext ctx = new CFAPContext()) { ctx.Configuration.ProxyCreationEnabled = false; var oldPassword = (from u in ctx.Users where u.Id == userForUpdate.Id select u.Password).Single(); if (oldPassword != userForUpdate.Password) { userForUpdate.EncriptPassword(); } try { //Загрузка в контекст данных о группах пользователя //userForUpdate.LoadUserGroupsFromObject(ctx); userForUpdate.SetRelationships(ctx); //Изменение связей с группами если они изменились userForUpdate.ChangeUserGroups(ctx); ctx.Entry(userForUpdate).State = EntityState.Modified; ctx.SaveChanges(DbConcurencyUpdateOptions.ClientPriority); } catch (DbEntityValidationException ex) { throw new FaultException <DataNotValidException>(new DataNotValidException(ex.EntityValidationErrors)); } catch (Exception ex) { throw new FaultException <DbException>(new DbException(ex)); } } return(userForUpdate); }
public void ChangeSummaryReadOnlyStatus(bool onOff, Filter filter, User user) { 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)); } this.ChechIsAdmin(user, typeof(Summary)); List <Summary> summaries = this.GetSummary(user, filter).ToList(); using (CFAPContext ctx = new CFAPContext()) { try { foreach (var summary in summaries) { ctx.Summaries.Attach(summary); summary.ReadOnly = onOff; ctx.Entry(summary).State = EntityState.Modified; ctx.SaveChanges(DbConcurencyUpdateOptions.ClientPriority); } } catch (Exception ex) { throw new FaultException <DbException>(new DbException(ex)); } } }
private void CustomProperiesValidate <TProperty>(TProperty property, CFAPContext ctx, List <DbEntityValidationResult> validationResults) where TProperty : class { if (validationResults == null) { throw new ArgumentNullException("List<DbEntityValidationResult> validationResults педреан с неопределенным значением."); } if (property == null) { DbValidationError validationError = new DbValidationError(typeof(TProperty).ToString(), "Значениие не определено"); DbEntityValidationResult dbEntityValidationResult = new DbEntityValidationResult(ctx.Entry(this), new DbValidationError[] { validationError }); validationResults.Add(dbEntityValidationResult); return; } var propertyValidationResult = ctx.Entry(property).GetValidationResult(); if (!propertyValidationResult.IsValid) { validationResults.Add(propertyValidationResult); } }
public void CustomValidate(CFAPContext ctx) { List <DbEntityValidationResult> validationResults = new List <DbEntityValidationResult>(); var summaryResult = ctx.Entry(this).GetValidationResult(); if (!summaryResult.IsValid) { validationResults.Add(summaryResult); } CustomProperiesValidate <Accountable>(this.Accountable, ctx, validationResults); CustomProperiesValidate <BudgetItem>(this.BudgetItem, ctx, validationResults); CustomProperiesValidate <Project>(this.Project, ctx, validationResults); if (validationResults.Count > 0) { throw new DbEntityValidationException( "Ошибка при проверке данных. Данные могут остутствовать или указаны не верно. Проверте внесенные данные и повторите попытку" , validationResults); } }
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); }