Beispiel #1
0
        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);
        }
Beispiel #2
0
        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);
        }
Beispiel #3
0
        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);
        }
Beispiel #4
0
        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);
            }
        }
Beispiel #5
0
        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);
        }
Beispiel #6
0
        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);
        }
Beispiel #7
0
        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));
                }
            }
        }
Beispiel #8
0
        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);
            }
        }
Beispiel #9
0
        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);
            }
        }
Beispiel #10
0
        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);
        }
Beispiel #11
0
        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);
        }