DDD Invariants Business Rules and Validation

前端 未结 2 838
春和景丽
春和景丽 2021-01-31 10:37

I am looking for advice on where to add validation rules for domain entities, and best practices for implementation. I did search and did not find what i was looking for, or i

相关标签:
2条回答
  • 2021-01-31 11:23

    When modeling your domain entity, it is best to consider real-world implications. Let's say you are dealing with a Employee entity.

    Employees need a name

    We know that in the real-world an employee must always have a name. It is impossible for an employee not to have a name. In other words, one cannot 'construct' an employee without specifying its name. So, use parameterised constructors! We also know that an employees name cannot change - so we prevent this from even happening by creating a private setter. Using the .NET type system to verify your employee is a very strong form of validation.

    public string Name { get; private set; }
    
    public Employee(string name)
    {
        Name = name;
    }
    

    Valid names have some rules

    Now it starts to get interesting. A name has certain rules. Let's just take the simplistic route and assume that a valid name is one which is not null or empty. In the code example above, the following business rule is not validated against. At this point, we can still currently create invalid employees! Let's prevent this from EVER occurring by amending our setter:

    public string Name
    {
        get
        {
            return name;
        }
        private set
        {
            if (String.IsNullOrWhiteSpace(value))
            {
                throw new ArgumentOutOfRangeException("value", "Employee name cannot be an empty value");
            }
    
            name = value;
        }
    }
    

    Personally I prefer to have this logic in the private setter than in the constructor. The setter is not completely invisible. The entity itself can still change it, and we need to ensure validity. Also, always throw exceptions!

    What about exposing some form of IsValid() method?

    Take the above Employee entity. Where and how would an IsValid() method work?

    Would you allow an invalid Employee to be created and then expect the developer to check it's validity with an IsValid() check? This is a weak design - before you know it, nameless Employees are going to be cruising around your system causing havoc.

    But perhaps you would like to expose the name validation logic?

    We don't want to catch exceptions for control flow. Exceptions are for catastrophic system failure. We also don't want to duplicate these validation rules in our codebase. So, perhaps exposing this validation logic isn't such a bad idea (but still not the greatest!).

    What you could do is provide a static IsValidName(string) method:

    public static bool IsValidName(string name)
    {
        return (String.IsNullOrWhiteSpace(value))
    }
    

    Our property would now change somewhat:

    public string Name
    {
        get
        {
            return name;
        }
        private set
        {
            if (!Employee.IsValidName(value))
            {
                throw new ArgumentOutOfRangeException("value", "Employee name cannot be an empty value");
            }
    
            name = value;
        }
    }
    

    But there is something fishy about this design...

    We now are starting to spawn validation methods for individual properties of our entity. If a property has all kinds of rules and behavior attached to it, perhaps this is a sign that we can create an value object for it!

    public PersonName : IEquatable<PersonName>
    {
        public string Name
        {
            get
            {
                return name;
            }
            private set
            {
                if (!PersonName.IsValid(value))
                {
                    throw new ArgumentOutOfRangeException("value", "Person name cannot be an empty value");
                }
    
                name = value;
            }
        }
    
        private PersonName(string name)
        {
            Name = name;
        }
    
        public static PersonName From(string name)
        {
            return new PersonName(name);
        }
    
        public static bool IsValid(string name)
        {
            return !String.IsNullOrWhiteSpace(value);
        }
    
        // Don't forget to override .Equals
    }
    

    Now our Employee entity can be simplified (I have excluded a null reference check):

    public Employee
    {
        public PersonName Name { get; private set; }
    
        public Employee(PersonName name)
        {
            Name = name;
        }
    }
    

    Our client code can now look something like this:

    if(PersonName.IsValid(name))
    {
        employee = new Employee(PersonName.From(name));
    }
    else
    {
        // Send a validation message to the user or something
    }
    

    So what have we done here?

    We have ensured that our domain model is always consistent. Extremely important. An invalid entity cannot be created. In addition, we have used value objects to provide further 'richness'. PersonName has given the client code more control and more power and has also simplified Employee.

    0 讨论(0)
  • 2021-01-31 11:32

    I built a library that can help you.

    https://github.com/mersocarlin/ffffd-validation

    0 讨论(0)
提交回复
热议问题