Thursday, July 30, 2009

Validate All SubmitChanges

Source Code

2009-08-24 - I have updated the code to incude an extension method for EntitySet addition called ValidatePriorToAddOnSubmit that will only .Add the entity if it passes. Also I have updated my sample code to show additional flows for validating when attaching to EntitySet or attaching a new entity as the EntityRef.

Unfortunately LINQ to SQL does not provide a way to get an accumulated list of errors after the SubmitChanges event. The default behavior is to not commit the transaction if there are ANY validation exceptions.

I have written a few extension methods to get around this behavior. With these extension methods updates, inserts, deletes, and adds will be cancelled if the entity does not pass OnValidate. This allows you to collect all of the invalid entities and yet still allow others to get processed.





I found this pattern very helpful when writing import routines and also grids that have all rows in edit mode. I'm sure there are many other places where this can be helpful. Here is a snippet using xml to deserialize and then persist only valid entities to the database.



Wednesday, July 29, 2009

Complex DataAnnotations Validation Patterns Part 1

Source Code

2009-09-01 - I have updated the code to allow validated properties to be NonPublic so that you can implement the Unidirectional serialization property on the DataContext and HIDE the circular references but still validate them.

This page shows how to enhance the DataAnnotations functionality in the System.ComponentModel namespace and implement it in a LINQ to SQL ORM project observing a layered architecture. All the examples utilize the Northwind database.

Generate your .dbml and create partial class .vb files for all of your entities that need validation. We will be utilizing the System.ComponentModel.DataAnnotations.MetadataTypeAttribute on these classes to help with the fact that we cannot properly decorate the LINQ to SQL designer file classes.



An underlying issue with the ValidationAttributes is that the IsValid function takes just one parameter. This makes it impossible to easily do complex validations on the entity. In order to solve this problem we create an interface called IComplexValidationAttribute with a public member that will contain the System.Reflection.PropertyInfo for the property that is getting validated. This now allows us to know more about the property including wether or not its Nullable.

Here is a Validate function that accepts the entity itself and returns an List(Of ValidationException). It checks to see if the attribute is a IComplexValidationAttribute and if it is instead of passing the value of the property it passes the object itself and assigns the public PropertyInfo member. This allows us much more flexability when creating our own custom ValidationAttributes.



I have created a IComplexValidationAttribute called RequiredAssociation for help dealing with LINQ to SQL required foreign keys. It looks at the UnderlyingValue of the associated entity and also the storage column for the entity. If either of them are Nothing, 0, String.Empty, or Guid.Empty then the IsValid call returns false.

Another IComplexValidationAttribute called RequiredProperty should take the place of the regular Required attribute if a property allows NULL in the database but because of updates or changes in the business logic should be required. This attribute understands nullables, numerics, guids, and strings.

Now all you have to do is hookup to LINQ to SQLs partial subroutine for OnValidate.



Soon to follow part 2.