The Scenario
Recently we had a tricky defect raised where, under certain circumstances, the user-entered values were not being persisted. The page was a somewhat complex data entry form, with many fields, lots of validation, and a complex hierarchy of view model objects and controls/helpers.
The process
After reproducing the issue (which itself was difficult), the first thing we did was to attach the debugger and set a break point on the first line of the controller action.
Attach a debugger
Upon breaking into the debugger, we noticed the ComplexModel
was missing certain values that were entered into the form. After checking the Razor template, we couldn’t see any obvious issues, so we used IE and Chrome developer tools to inspect the running page. Again, no obvious issues or explanations.
Inspect the request variables
The next step was to check the Request.Form
variables. Strange… the correct values had been submitted, and they were structured and named correctly. Instead of trusting ASP.NET MVC entirely, we also attached Fiddler. Once again, the missing values appeared and looked correct! Still no explanation for the missing values within our ComplexModel
.
Debug the Framework
At this point, we guessed there was a problem in the Default Model Binder, however, we remained skeptical. The best option might be to debug into the DefaultModelBinder
, and try to figure out why it’s not binding the missing fields.
Unfortunately, we hit numerous issues at this step. We found some great suggestions on StackOverflow, which unfortunately didn’t work as hoped. Our developer PCs were locked down in an SOE which meant we didn’t have the necessary configuration and permissions to download from the source server and Debug the framework.
Instead we tried downloading the MVC Source and copy it into our solution. Unfortunately, the project configuration was again somewhat complex, with configuration files, addins, third part libraries, all using the strongly signed ASP.NET MVC assemblies. After a few hours of trouble, we decided to try another option.
Replace the DefaultModelBinder
We grabbed a copy of the DefaultModelBinder, copied it into our project, renamed it to DebugDefaultModelBinder
, fixed a few compile errors where it was referencing MVC internal classes and methods.
In our Application_Start method, we changed the application to use this new model binder: ModelBinders.Binders.DefaultBinder = new DebugDefaultModelBinder();
.
Create a simplified test case
It was incredibly time consuming and error prone to continue loading up the application, navigating to the correct form, filling out all the fields so client-side validation would succeed, attaching the debugger, then pressing submit.
Instead, we looked for ways to recreate the failing scenario in a unit test. We found an old post describing how to achieve this. Basically, create a unit test like this:
Diving into the DefaultModelBinder
Now that we have a relatively targeted test case, debugging the problem should be a lot easier. We step into our DebugDefaultModelBinder
and notice the following:
At this point, we realise that our Children
property is being skipped because the BindingContext.ValueProvider.ContainsPrefix
method returns false
!
Finding the real bug
The following test case now gets us closer to the real problem:
It fails! We seem to have found the bug, and head back to google where we find out more:
The bug has been reported a number of times:
- FormValueProvider skips form values
- And directly reported on codeplex 616 and 832 at least twice.
The bug has been fixed, but not yet released.
The person who eventually fixed the bug, commented:
If there’s something in the search boundary that starts with the same name as the collection prefix that we’re trying to find, the binary search would actually fail. For example, let’s say we have foo.a, foo.bM and foo.b[0]. Calling Array.BinarySearch will fail to find foo.b because it will land on foo.bM, then look at foo.a and finally failing to find the prefix which is actually present in the container (foo.b[0]). Here we’re doing another pass looking specifically for collection prefix.
A number of work-arounds exist:
- Simple: avoid having a child collection with the same
prefix
text that other fields begin with. - More complex: Fixing the Model Binding issue of ASP.NET MVC 4 and ASP.NET Web API