One solution is to build the js rules directly from the DataAnnotations - the following snippet of code does this and demonstrates loading both built in DataAnnotations and custom annotations, it generates a module that can be loaded using requirejs to provide rule data for jQuery validation.
A set of rules is generated for every object in the namespace suplied in "ModelNamespace" that has DataAnnotations.
The following example is best used in builds build from the command line - the reason being that the assembly cannot be unloaded following inspection (I have not been able to come up with a solution for this short of building a full plugin mechanism for loading the assembly to be inspected). My workaround for this is to only generate the js for release builds using the following task configuration:
<Target Name="AfterBuild" Condition="$(Configuration) == 'Release'"> <CreateJsFromAnnotations ModelAssembly="bin\WebInterface.dll" ModelNamespace="WebInterface.Models" OutputFile="Scripts\common\generated\ModelMetadata.js" /> </Target>
Sample output would be something along the lines of:
define( [], function () { return { "folderValidation": { "folderapp": { "locationparsing": "PathOnly", "regex": "^[^\\s]*$", "maxlength": 256, "required": true }, "folderproxy": { "locationparsing": "PathOnly", "regex": "^[^\\s]*$", "maxlength": 256, "required": true } } }; } );
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
using System; | |
using System.Collections.Generic; | |
using System.ComponentModel.DataAnnotations; | |
using System.IO; | |
using System.Linq; | |
using System.Reflection; | |
using System.Text; | |
using Microsoft.Build.Framework; | |
using Microsoft.Build.Utilities; | |
using Newtonsoft.Json; | |
using System.Dynamic; | |
using Newtonsoft.Json.Converters; | |
namespace BuildTasks | |
{ | |
public class CreateRulesFromAnnotations : Task | |
{ | |
// wrapper js so the rules slot into requirejs | |
const string ModuleBegin = @"define( | |
[], | |
function () { | |
return "; | |
const string ModuleEnd = @"; | |
} | |
);"; | |
// Build Task properties to be set in the build configuration | |
[Microsoft.Build.Framework.Required] | |
public string ModelAssembly { get; set; } | |
[Microsoft.Build.Framework.Required] | |
public string ModelNamespace { get; set; } | |
[Microsoft.Build.Framework.Required] | |
public string OutputFile { get; set; } | |
public override bool Execute() | |
{ | |
try | |
{ | |
Log.LogMessage(MessageImportance.High, "Building JS rules from {0}", ModelNamespace); | |
var jsonRoot = new ExpandoObject() as IDictionary<string, object>; | |
foreach(var modelType in Assembly.LoadFrom(ModelAssembly).GetTypes().Where(t=>t.Namespace == ModelNamespace)) | |
{ | |
Log.LogMessage(MessageImportance.Low, "Processing {0}", modelType.FullName); | |
var jsonPropRules = new ExpandoObject() as IDictionary<string, object>; | |
foreach (var prop in modelType.GetProperties()) | |
{ | |
var jsonRules = new ExpandoObject() as IDictionary<string, object>; | |
foreach (var validationAttribute in prop.GetCustomAttributes(typeof(ValidationAttribute), false)) | |
{ | |
switch (validationAttribute.GetType().Name) | |
{ | |
case "StringLengthAttribute": // a standard DataAnnotation | |
jsonRules.Add("maxlength", ((StringLengthAttribute)validationAttribute).MaximumLength); | |
break; | |
case "RequiredAttribute": // a standard DataAnnotation | |
jsonRules.Add("required", true); | |
break; | |
case "LocationAttribute": // a custome attribute defined in the target dll | |
jsonRules.Add("locationparsing", validationAttribute.GetType().InvokeMember("LocationType", BindingFlags.GetProperty,null,validationAttribute,null)); | |
jsonRules.Add("regex", validationAttribute.GetType().InvokeMember("Pattern", BindingFlags.GetProperty, null, validationAttribute, null)); | |
jsonRules.Add("maxlength", validationAttribute.GetType().InvokeMember("MaxLength", BindingFlags.GetProperty, null, validationAttribute, null)); | |
break; | |
case "RegularExpressionAttribute": | |
jsonRules.Add("regex", ((RegularExpressionAttribute) validationAttribute).Pattern); | |
break; | |
} | |
} | |
if (jsonRules.Count > 0) | |
{ | |
var jsonPropName = string.Format("{0}{1}", modelType.Name, prop.Name).ToLower(); | |
jsonPropRules.Add(jsonPropName, jsonRules); | |
} | |
} | |
if (jsonPropRules.Count > 0) | |
{ | |
Log.LogMessage(MessageImportance.Normal, "Rules created for {0}", modelType.Name); | |
var jsonRootName = string.Format("{0}Validation", modelType.Name.ToLower()); | |
jsonRoot.Add(jsonRootName, jsonPropRules); | |
} | |
} | |
var settings = new JsonSerializerSettings | |
{ | |
Formatting = Formatting.Indented | |
}; | |
settings.Converters.Add(new StringEnumConverter()); | |
var json = new StringBuilder(ModuleBegin); | |
json.Append(JsonConvert.SerializeObject(jsonRoot, settings)); | |
json.Append(ModuleEnd); | |
File.WriteAllText(OutputFile, json.ToString()); | |
return true; | |
} | |
catch (Exception ex) | |
{ | |
Log.LogError(ex.ToString()); | |
return false; | |
} | |
} | |
} | |
} |