autofac和webapi 2 ModelBinder调用两次 - 第二次使用空bindingContext.ModelName

时间:2016-06-24 11:24:11

标签: c# asp.net-web-api2 autofac imodelbinder

我有以下配置:

builder.RegisterWebApiModelBinderProvider();
RegisterModelBinder<TypeModelBinder, object>(config, builder).InstancePerLifetimeScope();

使用以下方法:

private static IRegistrationBuilder<TBinder, ConcreteReflectionActivatorData, SingleRegistrationStyle> RegisterModelBinder<TBinder, TTarget>(HttpConfiguration config, ContainerBuilder builder)
{
    var targetType = typeof(TTarget);
    var regBuilder = builder.RegisterType<TBinder>()
        .WithParameter("validateAllProperties", true)
        .AsModelBinderForTypes(targetType);
    config.ParameterBindingRules.Add(targetType, p => p.BindWithModelBinding());
    return regBuilder;
}

我还没有以任何方式标记我的控制器。

但签名的一个例子是:

[Route("data/{type}")]
public IHttpActionResult Post(string type, object data)

活页夹:

public class TypeModelBinder : IModelBinder
{
    public TypeModelBinder(ITypeResolver resolver, ITypeSerializer serializer, IContextETagExtractor eTagExtractor, bool validateAllProperties)
        : base(resolver, serializer, eTagExtractor, validateAllProperties)
    {
    }

    public override bool BindModel(HttpActionContext actionContext, ModelBindingContext bindingContext)
    {
        var typeInfo = default(RuntimeTypeInfo);
        var data = default(T);
        try
        {
            typeInfo = Resolver.Resolve(typeof(T).Name);
            data = this.BindData(actionContext);

            Validate(typeInfo.Type, data, bindingContext);
            if (this.IsConcurrentMethod(actionContext.Request.Method) && typeInfo.IsVersioned)
            {
                ValidateAndSetConcurrencyToken(typeInfo.Type, data, actionContext, bindingContext);
            }
        }
        catch (ArgumentException ae)
        {
            bindingContext.ModelState.AddModelError("Body", ae.Message);
        }
        catch (JsonSerializationException jse)
        {
            bindingContext.ModelState.AddModelError("Body", jse.Message);
        }
        catch (FormatException fe)
        {
            var errMsg = string.Format("{0} {1}", Constants.MODEL_ERROR_PREFIX_IF_MATCH_INVALID, fe.Message);
            bindingContext.ModelState.AddModelError(Constants.MODEL_ERROR_KEY_IF_MATCH, errMsg);
        }
        catch (Exception e)
        {
            bindingContext.ModelState.AddModelError("Body", e.Message);
        }

        bindingContext.Model = data;
        return bindingContext.ModelState.IsValid;
    }
}

我编写了一个测试来检查当我传递一个无效对象时TypeModelBinder中的模型绑定失败。

但是我注意到以下行为。第一次通过它就像我期望的那样,bindingContext.ModelName设置为"data"

但是我注意到,当绑定失败时,它会立即重新输入绑定器的BindModel方法,这次将bindingContext.ModelName设置为""然后添加另一个模型错误的副本添加并且owin堆栈中的某些内容会引发500 ...

在连接我的模型装订器时,有什么明显的错误吗?

0 个答案:

没有答案