Skip to content
This repository has been archived by the owner on Nov 27, 2018. It is now read-only.

Commit

Permalink
Fix for issue 85 - Dictionary types should return null on key not found
Browse files Browse the repository at this point in the history
This change makes RouteValueDictionary a full IDictionary implementation
instead of a subclass of Dictionary.

Followed the patterns used in the old implementation, namely preserving
the struct-returning behavior of Keys/Values/GetEnumerator.
  • Loading branch information
rynowak committed Jul 25, 2014
1 parent 3eb6c22 commit ae65001
Show file tree
Hide file tree
Showing 5 changed files with 215 additions and 22 deletions.
190 changes: 185 additions & 5 deletions src/Microsoft.AspNet.Routing/RouteValueDictionary.cs
Original file line number Diff line number Diff line change
Expand Up @@ -2,21 +2,48 @@
// Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information.

using System;
using System.Collections;
using System.Collections.Generic;
using System.Linq;
using System.Reflection;

namespace Microsoft.AspNet.Routing
{
public class RouteValueDictionary : Dictionary<string, object>
/// <summary>
/// An <see cref="IDictionary{string, object}"/> type for route values.
/// </summary>
public class RouteValueDictionary : IDictionary<string, object>
{
private readonly Dictionary<string, object> _dictionary;

/// <summary>
/// Creates an empty RouteValueDictionary.
/// </summary>
public RouteValueDictionary()
: base(StringComparer.OrdinalIgnoreCase)
{
_dictionary = new Dictionary<string, object>(StringComparer.OrdinalIgnoreCase);
}

/// <summary>
/// Creates a RouteValueDictionary initialized with the provided input values.
/// </summary>
/// <param name="values">Input values to copy into the dictionary.</param>
public RouteValueDictionary([NotNull] IDictionary<string, object> values)
{
_dictionary = new Dictionary<string, object>(values, StringComparer.OrdinalIgnoreCase);
}

/// <summary>
/// Creates a RouteValueDictionary initialized with the provided input values.
/// </summary>
/// <param name="values">Input values to copy into the dictionary.</param>
/// <remarks>
/// The input parameter is interpreted as a set of key-value-pairs where the property names
/// are keys, and property values are the values, and copied into the dictionary. Only public
/// instance non-index properties are considered.
/// </remarks>
public RouteValueDictionary(object obj)
: base(StringComparer.OrdinalIgnoreCase)
: this()
{
if (obj != null)
{
Expand Down Expand Up @@ -46,9 +73,162 @@ public RouteValueDictionary(object obj)
}
}

public RouteValueDictionary(IDictionary<string, object> other)
: base(other, StringComparer.OrdinalIgnoreCase)
/// <inheritdoc />
public object this[[NotNull] string key]
{
get
{
object value;
_dictionary.TryGetValue(key, out value);
return value;
}

set
{
_dictionary[key] = value;
}
}

/// <summary>
/// Gets the comparer for this dictionary.
/// </summary>
/// <remarks>
/// This will always be a reference to <see cref="StringComparer.OrdinalIgnoreCase"/>
/// </remarks>
public IEqualityComparer<string> Comparer
{
get
{
return _dictionary.Comparer;
}
}

/// <inheritdoc />
public int Count
{
get
{
return _dictionary.Count;
}
}

/// <inheritdoc />
bool ICollection<KeyValuePair<string, object>>.IsReadOnly
{
get
{
return ((ICollection<KeyValuePair<string, object>>)_dictionary).IsReadOnly;
}
}

/// <inheritdoc />
public Dictionary<string, object>.KeyCollection Keys
{
get
{
return _dictionary.Keys;
}
}

/// <inheritdoc />
ICollection<string> IDictionary<string, object>.Keys
{
get
{
return _dictionary.Keys;
}
}

/// <inheritdoc />
public Dictionary<string, object>.ValueCollection Values
{
get
{
return _dictionary.Values;
}
}

/// <inheritdoc />
ICollection<object> IDictionary<string, object>.Values
{
get
{
return _dictionary.Values;
}
}

/// <inheritdoc />
void ICollection<KeyValuePair<string, object>>.Add(KeyValuePair<string, object> item)
{
((ICollection<KeyValuePair<string, object>>)_dictionary).Add(item);
}

/// <inheritdoc />
public void Add([NotNull] string key, object value)
{
_dictionary.Add(key, value);
}

/// <inheritdoc />
public void Clear()
{
_dictionary.Clear();
}

/// <inheritdoc />
bool ICollection<KeyValuePair<string, object>>.Contains(KeyValuePair<string, object> item)
{
return ((ICollection<KeyValuePair<string, object>>)_dictionary).Contains(item);
}

/// <inheritdoc />
public bool ContainsKey([NotNull] string key)
{
return _dictionary.ContainsKey(key);
}

/// <inheritdoc />
void ICollection<KeyValuePair<string, object>>.CopyTo(
[NotNull] KeyValuePair<string, object>[] array,
int arrayIndex)
{
((ICollection<KeyValuePair<string, object>>)_dictionary).CopyTo(array, arrayIndex);
}

/// <inheritdoc />
public Dictionary<string, object>.Enumerator GetEnumerator()
{
return _dictionary.GetEnumerator();
}

/// <inheritdoc />
IEnumerator<KeyValuePair<string, object>> IEnumerable<KeyValuePair<string, object>>.GetEnumerator()
{
return _dictionary.GetEnumerator();
}

/// <inheritdoc />
IEnumerator IEnumerable.GetEnumerator()
{
return _dictionary.GetEnumerator();
}

/// <inheritdoc />
bool ICollection<KeyValuePair<string, object>>.Remove(KeyValuePair<string, object> item)
{
return ((ICollection<KeyValuePair<string, object>>)_dictionary).Remove(item);
}

/// <inheritdoc />
public bool Remove([NotNull] string key)
{
return _dictionary.Remove(key);
}

/// <inheritdoc />
public bool TryGetValue([NotNull] string key, out object value)
{
return _dictionary.TryGetValue(key, out value);
}
}
}
15 changes: 7 additions & 8 deletions src/Microsoft.AspNet.Routing/Template/TemplateBinder.cs
Original file line number Diff line number Diff line change
Expand Up @@ -147,7 +147,7 @@ public TemplateValuesResult GetValues(IDictionary<string, object> ambientValues,

// Add any ambient values that don't match parameters - they need to be visible to constraints
// but they will ignored by link generation.
var combinedValues = new Dictionary<string, object>(context.AcceptedValues, StringComparer.OrdinalIgnoreCase);
var combinedValues = new RouteValueDictionary(context.AcceptedValues);
if (ambientValues != null)
{
foreach (var kvp in ambientValues)
Expand Down Expand Up @@ -292,7 +292,6 @@ private TemplatePart GetParameter(string name)
return null;
}


/// <summary>
/// Compares two objects for equality as parts of a case-insensitive path.
/// </summary>
Expand Down Expand Up @@ -342,8 +341,8 @@ private class TemplateBindingContext
{
private readonly IDictionary<string, object> _defaults;

private readonly Dictionary<string, object> _acceptedValues;
private readonly Dictionary<string, object> _filters;
private readonly RouteValueDictionary _acceptedValues;
private readonly RouteValueDictionary _filters;

public TemplateBindingContext(IDictionary<string, object> defaults, IDictionary<string, object> values)
{
Expand All @@ -354,20 +353,20 @@ public TemplateBindingContext(IDictionary<string, object> defaults, IDictionary<

_defaults = defaults;

_acceptedValues = new Dictionary<string, object>(StringComparer.OrdinalIgnoreCase);
_acceptedValues = new RouteValueDictionary();

if (_defaults != null)
{
_filters = new Dictionary<string, object>(defaults, StringComparer.OrdinalIgnoreCase);
_filters = new RouteValueDictionary(defaults);
}
}

public Dictionary<string, object> AcceptedValues
public RouteValueDictionary AcceptedValues
{
get { return _acceptedValues; }
}

public Dictionary<string, object> Filters
public RouteValueDictionary Filters
{
get { return _filters; }
}
Expand Down
6 changes: 3 additions & 3 deletions src/Microsoft.AspNet.Routing/Template/TemplateMatcher.cs
Original file line number Diff line number Diff line change
Expand Up @@ -32,10 +32,10 @@ public IDictionary<string, object> Match(string requestPath, IDictionary<string,

if (defaults == null)
{
defaults = new Dictionary<string, object>(StringComparer.OrdinalIgnoreCase);
defaults = new RouteValueDictionary();
}

var values = new Dictionary<string, object>(StringComparer.OrdinalIgnoreCase);
var values = new RouteValueDictionary();

for (var i = 0; i < requestSegments.Length; i++)
{
Expand Down Expand Up @@ -175,7 +175,7 @@ public IDictionary<string, object> Match(string requestPath, IDictionary<string,
private bool MatchComplexSegment(TemplateSegment routeSegment,
string requestSegment,
IDictionary<string, object> defaults,
Dictionary<string, object> values)
RouteValueDictionary values)
{
Contract.Assert(routeSegment != null);
Contract.Assert(routeSegment.Parts.Count > 1);
Expand Down
2 changes: 1 addition & 1 deletion src/Microsoft.AspNet.Routing/Template/TemplateRoute.cs
Original file line number Diff line number Diff line change
Expand Up @@ -42,7 +42,7 @@ public TemplateRoute([NotNull] IRouter target,
_target = target;
_routeTemplate = routeTemplate ?? string.Empty;
Name = routeName;
_defaults = defaults ?? new Dictionary<string, object>(StringComparer.OrdinalIgnoreCase);
_defaults = defaults ?? new RouteValueDictionary();
_constraints = RouteConstraintBuilder.BuildConstraints(constraints, _routeTemplate) ??
new Dictionary<string, IRouteConstraint>();

Expand Down
24 changes: 19 additions & 5 deletions test/Microsoft.AspNet.Routing.Tests/Template/TemplateRouteTests.cs
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,7 @@ public class TemplateRouteTests

// PathString in HttpAbstractions guarantees a leading slash - so no value in testing other cases.
[Fact]
public async void Match_Success_LeadingSlash()
public async Task Match_Success_LeadingSlash()
{
// Arrange
var route = CreateRoute("{controller}/{action}");
Expand All @@ -40,7 +40,7 @@ public async void Match_Success_LeadingSlash()
}

[Fact]
public async void Match_Success_RootUrl()
public async Task Match_Success_RootUrl()
{
// Arrange
var route = CreateRoute("");
Expand All @@ -55,7 +55,7 @@ public async void Match_Success_RootUrl()
}

[Fact]
public async void Match_Success_Defaults()
public async Task Match_Success_Defaults()
{
// Arrange
var route = CreateRoute("{controller}/{action}", new { action = "Index" });
Expand All @@ -72,7 +72,7 @@ public async void Match_Success_Defaults()
}

[Fact]
public async void Match_Fails()
public async Task Match_Fails()
{
// Arrange
var route = CreateRoute("{controller}/{action}");
Expand All @@ -86,7 +86,7 @@ public async void Match_Fails()
}

[Fact]
public async void Match_RejectedByHandler()
public async Task Match_RejectedByHandler()
{
// Arrange
var route = CreateRoute("{controller}", accept: false);
Expand All @@ -102,6 +102,20 @@ public async void Match_RejectedByHandler()
Assert.NotNull(context.RouteData.Values);
}

[Fact]
public async Task Match_RouteValuesDoesntThrowOnKeyNotFound()
{
// Arrange
var route = CreateRoute("{controller}/{action}");
var context = CreateRouteContext("/Home/Index");

// Act
await route.RouteAsync(context);

// Assert
Assert.Null(context.RouteData.Values["1controller"]);
}

private static RouteContext CreateRouteContext(string requestPath)
{
var request = new Mock<HttpRequest>(MockBehavior.Strict);
Expand Down

0 comments on commit ae65001

Please sign in to comment.