WebApi:ApiExplorer和Custom ModelBinders

时间:2015-06-05 16:28:01

标签: asp.net-web-api custom-model-binder asp.net-mvc-apiexplorer

我的大多数api路线都是这样分段的:

/api/{segment}/MyEntity(即“/ api / SegmentA / MyEntity”)

我已经定义了一个ModelBinder,它将字符串转换为Segment对象,如下所示:

class SegmentModelBinder : IModelBinder
{
    public bool BindModel(HttpActionContext actionContext, ModelBindingContext bindingContext)
    {
        var value = bindingContext.ValueProvider.GetValue(bindingContext.ModelName);
        if (value == null || String.IsNullOrEmpty(value.AttemptedValue))
            return false;

        bindingContext.Model = **logic to find segment object from value.AttemptedValue**;
        return true;
    }
}

配置为:

GlobalConfiguration.Configuration.BindParameter(typeof(Segment), new SegmentModelBinder());

所以我的路线看起来像这样:

public class MyEntityController : BaseController
{
    [HttpGet, Route("api/{segment}/MyEntity")]
    public IEnumerable<MyEntity> Get(Segment segment)
    {
        ...
    }
}

问题是,我现在正在尝试为这些Api调用生成文档,并且ApiExplorer完全被这些路由混淆并忽略它们。

我如何告诉它,对于这些路由,当它看到Segment类型的参数时,它实际上只是路由中的一个字符串?

1 个答案:

答案 0 :(得分:0)

从使用ModelBinder切换到TypeConverter解决了问题:

[TypeConverter(typeof(MyEntityConverter))]
public class MyEntity
{....

-

public class MyEntityConverter : TypeConverter
{
    public override bool CanConvertFrom(ITypeDescriptorContext context, Type sourceType)
    {
        if (sourceType == typeof(string))
            return true;
        return base.CanConvertFrom(context, sourceType);
    }

    public override object ConvertFrom(ITypeDescriptorContext context, System.Globalization.CultureInfo culture, object value)
    {
        var key = value as string;
        if (!String.IsNullOrEmpty(key))
            return **Find Entity**;

        return base.ConvertFrom(context, culture, value);
    }
}

编辑:

如果你在调用中返回这个实体,你也需要这个,否则newtonsoft json序列化器会将类序列化为类型名称:

    public override bool CanConvertTo(ITypeDescriptorContext context, Type destinationType)
    {
        return false;
    }