У меня есть два метода действий, которые противоречат друг другу. В принципе, я хочу иметь возможность перейти к одному и тому же представлению с использованием двух разных маршрутов: идентификатором элемента или именем элемента и его родителем (элементы могут иметь одно и то же имя для разных родителей). Для фильтрации списка можно использовать термин поиска.
Например...
Items/{action}/ParentName/ItemName
Items/{action}/1234-4321-1234-4321
Вот мои методы действий (есть также методы действий Remove)...
// Method #1
public ActionResult Assign(string parentName, string itemName) { 
    // Logic to retrieve item ID here...
    string itemId = ...;
    return RedirectToAction("Assign", "Items", new { itemId });
}
// Method #2
public ActionResult Assign(string itemId, string searchTerm, int? page) { ... }
И вот маршруты...
routes.MapRoute("AssignRemove",
                "Items/{action}/{itemId}",
                new { controller = "Items" }
                );
routes.MapRoute("AssignRemovePretty",
                "Items/{action}/{parentName}/{itemName}",
                new { controller = "Items" }
                );
Я понимаю, почему возникает ошибка, так как параметр page может быть нулевым, но я не могу найти оптимальный способ его решения. С чего начать? Я подумал о расширении подписи Method #1, чтобы включить параметры поиска и переместить логику в Method #2 на закрытый метод, который они оба вызывали, но я не верю, что это действительно устранит двусмысленность.
Любая помощь будет принята с благодарностью.
Фактическое решение (на основе ответа Леви)
Я добавил следующий класс...
public class RequireRouteValuesAttribute : ActionMethodSelectorAttribute {
    public RequireRouteValuesAttribute(string[] valueNames) {
        ValueNames = valueNames;
    }
    public override bool IsValidForRequest(ControllerContext controllerContext, MethodInfo methodInfo) {
        bool contains = false;
        foreach (var value in ValueNames) {
            contains = controllerContext.RequestContext.RouteData.Values.ContainsKey(value);
            if (!contains) break;
        }
        return contains;
    }
    public string[] ValueNames { get; private set; }
}
И затем оформлены методы действия...
[RequireRouteValues(new[] { "parentName", "itemName" })]
public ActionResult Assign(string parentName, string itemName) { ... }
[RequireRouteValues(new[] { "itemId" })]
public ActionResult Assign(string itemId) { ... }