Проверка уникального имени FluentValidation с использованием базы данных

У меня есть модель Category с полем Name, и имя каждой категории должно быть уникальным. Я сделал проверку, и она работает, когда я пытаюсь создать новую категорию, но у меня возникают проблемы при попытке ее редактирования. На данный момент он просто проверяет, существует ли имя, и, конечно же, оно существует, когда я пытаюсь отредактировать ту же категорию.

Модель

[Validator(typeof(CategoryValidator))]
public class Category
{
    public int ID { get; set; }
    public string Name { get; set; }
    virtual public ICollection<Image> Images { get; set; }
}

public class CategoryValidator : AbstractValidator<Category>
{
    public CategoryValidator()
    {
        RuleFor(x => x.Name).NotEmpty().WithMessage("Category name is required.").Must(UniqueName).WithMessage("This category name already exists.");
    }

    private bool UniqueName(string name)
    {
        ProjecteDataContext _db = new ProjecteDataContext();
        var category = _db.Categories.Where(x => x.Name.ToLower() == name.ToLower()).SingleOrDefault();

        if (category == null) return true;
        return false;
    }
}

Как вы можете видеть, у меня есть функция UniqueName(string name), но как я могу передать в нее идентификатор или всю модель, чтобы я мог проверить, тот же ли это идентификатор, что и модель, которую я пытаюсь отредактировать, тогда она пройдет. Как я мог передать что-то вроде UniqueName(string name, int? id)? Я обнаружил FluentValidation только сегодня и не могу понять.


person Stan    schedule 15.11.2012    source источник


Ответы (1)


Проверка предикатов (она же Must) имеет перегруженную версию, которая принимает предикат с двумя параметрами (проверенный объект и стоимость имущества). В вашем случае предикат будет иметь тип Func<Category, string, bool>. Поэтому просто добавьте параметр Category к вашему уникальному методу проверки имени:

private bool UniqueName(Category category, string name)
{
        ProjecteDataContext _db = new ProjecteDataContext();
        var dbCategory = _db.Categories
                            .Where(x => x.Name.ToLower() == name.ToLower())
                            .SingleOrDefault();

        if (dbCategory == null) 
            return true;

        return dbCategory.ID == category.ID;
}
person Sergey Berezovskiy    schedule 15.11.2012
comment
примечание: вы должны избавиться от DbContext, как только закончите с ним, поэтому с помощью (var _db = new ProjecteDataContext()) - person Hamid Sadeghian; 14.11.2016
comment
Можно ли внедрить DbContext? - person Ish Thomas; 26.03.2020
comment
Короткий ответ @IshThomas: да, вы можете ввести контекст в валидатор. Рассмотрите возможность поиска специально для внедрения в валидаторы, и вы должны найти ответы, связанные с этим. В документации по FluentValidation также упоминается, как это сделать. - person julealgon; 16.04.2020