An Entity (let’s say a UserEntity) has rigid rules for it’s properties, and it can exist in 2 states – persisted (which means it has an id) and pre-persisted (which means it does not have an id yet).
According to the answer to this question about how to handle required properties, a “real” UserEntity should only ever be created with an id passed in to its constructor.
However, when I need to create a new UserEntity from information sent by the browser, I need to be able to validate the information before persisting into the db.
In the past, I would simply create a blank UserEntity (without an id), set the new properties, and the validate it – but, in this new, more secure way of thinking about Entities, I shouldn’t ever create a new UserEntity without its id.
I don’t want to create TWO places that know how to validate my UserEntity’s properties, because if they ever change (and they will) it would be double the code to update and double the chances for bugs.
How do I efficiently centralize the validation knowledge of my entity’s properties?
Note
One idea I had is reflected in this question, in which I consider storing the non-state properties like email, password and name in a standardized value object that would know about the rules for its properties that different services, like the Controller, Validator, and Repo, or Mapper could use.
that’s what factories are for. to the factory method you pass only the data that is required to enforce the real invariants of UserEntity (take some time to figure out what are your real invariants of UserEntity and you’d better do it with your domain experts).
in the factory method you create a new Id and pass it to the UserEntity constructor.
In this stage i don’t think it is that bad to discard the instance if the validation inside the constructor fails. in the worst case – you’ve lost an id… it’s not a case that suppose to happen quite often – most of the time the data should be validated in the client.
Of course another option is that in the factory method you first validate the parameters and only then create a new Id and pass it to the UserEntity constructor.
itzik saban