Sometimes, when you use EntLib (Microsoft Enterprise Library) validation block, you may realize, that the built-in set of validators in the EntLib does not satisfy all the your needs. Of course, you can create a custom validation, inheriting from the abstract Validator and ValidationAttribute classes. This is a good way to create a validation logic, that can be used in the different parts of your project. But what to do, if you need to access some business components inside of validator, which should be resolved via the dependency injection (DI) mechanism? Since the EntLib is responsible for instantiating validators instances, it doesn`t know anything about your registered dependency classes, and will not be able to resolve such dependencies. In this case you have two possible solutions:
- Access DI container via a static class, that has a reference to it;
- Use SelfValidationAttribute in the class you have to validate;
I do not recommend the first approach with a static class, because it makes your domain model dependent on the application infrastructure. From my point of view, the less classes know about dependency injection container - the better. The second approach is more prefferable, it is very suitable for the progressive domain models and view-models from the MV-VM pattern. All that you need to do is:
- Define in your class a public void method that takes a single parameter of type ValidationResults;
- Decorate this method with a SelfValidationAttribute;
- Decorate your class with a HasSelfValidationAttribute;
Let`s consider the following example. We have a CreateProductViewModel class that is used for representing a product creation form to user. Let`s assume, that we want to deny user to create a new product, if there is a product with the same name already exists in the store database. Moreover, we want to use a business component, which is responsible for the product-related operations. Since we have a large composite application, we use a DI mechanism to provide a loosely coupled architecture and increase testability. Below you can find the code of this sample class.
- [HasSelfValidation]
- public class CreateProductViewModel
- {
- private readonly IProductComponent
_productComponent;
- public CreateProductViewModel(IProductComponent productComponent)
- {
- _productComponent = productComponent;
- }
- [StringLengthValidator(1, 20)]
- public string ProductName { get; set; }
- [SelfValidation]
- public void
IsDuplicateProductName(ValidationResults results)
- {
- var product =
_productComponent.GetProductByName(ProductName);
- if (product != null)
- results.AddResult(new ValidationResult("Product with same name already exists", null,
"ProductName",
null, null));
- }
- }
* This source code was highlighted with Source Code Highlighter.
Here is the IsDuplicateProductName method that performs a validation logic by acquiring a product component. Product component is an instance of the class, which implements IProductComponent interface and most probably becomes resolved via the constructor dependency injection, but this is not so significant. You can get full validation results by invoking the code like this:
- var validator = ValidationFactory.
CreateValidatorFromAttributes<CreateProductViewModel>();
- ValidationResults results;
- validator.Validate(createProductViewModelInstance, results);
* This source code was highlighted with Source Code Highlighter.