public Task <ResponseBase> Handle(HttpInformation context, DeleteCommand command)
        {
            SapphireDbContext           db       = GetContext(command.ContextName);
            KeyValuePair <Type, string> property = db.GetType().GetDbSetType(command.CollectionName);

            if (property.Key != null)
            {
                try
                {
                    object[] primaryKeys = property.Key.GetPrimaryKeyValues(db, command.PrimaryKeys);
                    object   value       = db.Find(property.Key, primaryKeys);

                    if (!property.Key.CanRemove(context, value, serviceProvider))
                    {
                        return(Task.FromResult(command.CreateExceptionResponse <DeleteResponse>(
                                                   "The user is not authorized for this action.")));
                    }

                    if (value != null)
                    {
                        property.Key.ExecuteHookMethods <RemoveEventAttribute>(ModelStoreEventAttributeBase.EventType.Before, value, context, serviceProvider);

                        db.Remove(value);

                        property.Key.ExecuteHookMethods <RemoveEventAttribute>(ModelStoreEventAttributeBase.EventType.BeforeSave, value, context, serviceProvider);

                        db.SaveChanges();

                        property.Key.ExecuteHookMethods <RemoveEventAttribute>(ModelStoreEventAttributeBase.EventType.After, value, context, serviceProvider);

                        return(Task.FromResult <ResponseBase>(new DeleteResponse()
                        {
                            ReferenceId = command.ReferenceId
                        }));
                    }

                    return(Task.FromResult(command.CreateExceptionResponse <DeleteResponse>("The value was not found.")));
                }
                catch (Exception ex)
                {
                    return(Task.FromResult(command.CreateExceptionResponse <DeleteResponse>(ex)));
                }
            }

            return(Task.FromResult(command.CreateExceptionResponse <DeleteResponse>("No set for collection was found.")));
        }
        public async Task <ResponseBase> Handle(HttpInformation context, DeleteRangeCommand command,
                                                ExecutionContext executionContext)
        {
            SapphireDbContext           db       = GetContext(command.ContextName);
            KeyValuePair <Type, string> property = db.GetType().GetDbSetType(command.CollectionName);

            if (property.Key == null)
            {
                throw new CollectionNotFoundException(command.ContextName, command.CollectionName);
            }

            if (property.Key.GetModelAttributesInfo().DisableDeleteAttribute != null)
            {
                throw new OperationDisabledException("Delete", command.ContextName, command.CollectionName);
            }

            List <object> removedValues = new List <object>();

            DeleteRangeResponse response;
            bool updateRejected;

            do
            {
                updateRejected = false;

                response = new DeleteRangeResponse
                {
                    ReferenceId = command.ReferenceId,
                    Results     = command.Values.Select(valuePrimaryKeys =>
                    {
                        object[] primaryKeys = property.Key.GetPrimaryKeyValues(db, valuePrimaryKeys);
                        object value         = db.Find(property.Key, primaryKeys);

                        if (value == null)
                        {
                            throw new ValueNotFoundException(command.ContextName, command.CollectionName, primaryKeys);
                        }

                        if (value is SapphireOfflineEntity valueOfflineEntity &&
                            valuePrimaryKeys.TryGetValue("modifiedOn", out JValue modifiedOn))
                        {
                            DateTimeOffset commandModifiedOn = modifiedOn.ToObject <DateTimeOffset>();

                            if (!valueOfflineEntity.ModifiedOn.EqualWithTolerance(commandModifiedOn,
                                                                                  db.Database.ProviderName))
                            {
                                throw new DeleteRejectedException(command.ContextName, command.CollectionName, primaryKeys);
                            }
                        }

                        if (!property.Key.CanDelete(context, value, serviceProvider))
                        {
                            throw new UnauthorizedException("The user is not authorized for this action");
                        }

                        int insteadOfExecuteCount = property.Key.ExecuteHookMethods <DeleteEventAttribute>(
                            ModelStoreEventAttributeBase.EventType.InsteadOf,
                            value, null, context, serviceProvider, db);

                        if (insteadOfExecuteCount > 0)
                        {
                            return(new DeleteResponse());
                        }

                        property.Key.ExecuteHookMethods <DeleteEventAttribute>(
                            ModelStoreEventAttributeBase.EventType.Before, value, null, context, serviceProvider, db);

                        db.Remove(value);

                        property.Key.ExecuteHookMethods <DeleteEventAttribute>(
                            ModelStoreEventAttributeBase.EventType.BeforeSave, value, null, context,
                            serviceProvider, db);

                        removedValues.Add(value);

                        return(new DeleteResponse()
                        {
                            Value = value,
                            ReferenceId = command.ReferenceId
                        });
                    }).ToList()
                };

                try
                {
                    db.SaveChanges();
                }
                catch (DbUpdateConcurrencyException)
                {
                    foreach (EntityEntry entityEntry in db.ChangeTracker.Entries())
                    {
                        await entityEntry.ReloadAsync();
                    }

                    updateRejected = true;
                }
            } while (updateRejected);

            foreach (object value in removedValues)
            {
                property.Key.ExecuteHookMethods <DeleteEventAttribute>(
                    ModelStoreEventAttributeBase.EventType.After, value, null, context, serviceProvider, db);
            }

            return(response);
        }
    }
Esempio n. 3
0
        public async Task <ResponseBase> Handle(HttpInformation context, DeleteRangeCommand command)
        {
            SapphireDbContext           db       = GetContext(command.ContextName);
            KeyValuePair <Type, string> property = db.GetType().GetDbSetType(command.CollectionName);

            if (property.Key != null)
            {
                try
                {
                    List <object> removedValues = new List <object>();

                    DeleteRangeResponse response;
                    bool updateRejected;

                    do
                    {
                        updateRejected = false;

                        response = new DeleteRangeResponse
                        {
                            ReferenceId = command.ReferenceId,
                            Results     = command.Values.Select(valuePrimaryKeys =>
                            {
                                object[] primaryKeys = property.Key.GetPrimaryKeyValues(db, valuePrimaryKeys);
                                object value         = db.Find(property.Key, primaryKeys);

                                if (value != null)
                                {
                                    if (value is SapphireOfflineEntity valueOfflineEntity &&
                                        valuePrimaryKeys.TryGetValue("modifiedOn", out JValue modifiedOn))
                                    {
                                        DateTime commandModifiedOn = modifiedOn.ToObject <DateTime>();

                                        if (valueOfflineEntity.ModifiedOn.RoundToMilliseconds() !=
                                            commandModifiedOn.RoundToMilliseconds())
                                        {
                                            return((DeleteResponse)command.CreateExceptionResponse <DeleteResponse>(
                                                       "Deletion rejected. The object state has changed."));
                                        }
                                    }

                                    if (!property.Key.CanRemove(context, value, serviceProvider))
                                    {
                                        return((DeleteResponse)command.CreateExceptionResponse <DeleteResponse>(
                                                   "The user is not authorized for this action."));
                                    }

                                    property.Key.ExecuteHookMethods <RemoveEventAttribute>(
                                        ModelStoreEventAttributeBase.EventType.Before, value, context, serviceProvider);

                                    db.Remove(value);

                                    property.Key.ExecuteHookMethods <RemoveEventAttribute>(
                                        ModelStoreEventAttributeBase.EventType.BeforeSave, value, context,
                                        serviceProvider);

                                    removedValues.Add(value);

                                    return(new DeleteResponse()
                                    {
                                        Value = value,
                                        ReferenceId = command.ReferenceId
                                    });
                                }

                                return((DeleteResponse)command.CreateExceptionResponse <DeleteResponse>(
                                           "The value was not found."));
                            }).ToList()
                        };

                        try
                        {
                            db.SaveChanges();
                        }
                        catch (DbUpdateConcurrencyException)
                        {
                            foreach (EntityEntry entityEntry in db.ChangeTracker.Entries())
                            {
                                await entityEntry.ReloadAsync();
                            }

                            updateRejected = true;
                        }
                    } while (updateRejected);

                    foreach (object value in removedValues)
                    {
                        property.Key.ExecuteHookMethods <RemoveEventAttribute>(
                            ModelStoreEventAttributeBase.EventType.After, value, context, serviceProvider);
                    }

                    return(response);
                }
                catch (Exception ex)
                {
                    return(command.CreateExceptionResponse <DeleteResponse>(ex));
                }
            }

            return(command.CreateExceptionResponse <DeleteResponse>("No set for collection was found."));
        }