billbogaiv/hybrid-model-binding

Combine with ApiController attribute

artyom-p opened this issue · 6 comments

Hi, I'm using .NET core 2.2 and hybrid binding works fine without ApiControllerAttribute.
I'm trying to bind a model which consists of values from route and body, and when attribute is applied, only body binding works. Is it possible to combine them together?

Is it possible to combine them together?

Yes. This is the main feature of the library. Depending on your particular use-case, the body-binding process might be overriding bound-data from the route. If you post a sample of your controller-action and model, I might be able to provide better insight into what's happening.

So, i was testing it on a newly created .net core 2.2 webapi project.

Here's the Startup's ConfigureServices method

     public void ConfigureServices(IServiceCollection services)
     {
         services.AddMvc()
              .SetCompatibilityVersion(CompatibilityVersion.Version_2_2)
             .AddHybridModelBinder();
     }

and here's a controller and a model

    [Route("api/[controller]")]
    [ApiController]
    public class ValuesController : ControllerBase
    {
        // GET api/values
        [HttpGet("{Id}")]
        public ActionResult<bool> Get(Test model)
        {
            return true;
        }
    }

    public class Test
    {
        [From(Source.Route)]
        public int Id { get; set; }

        [From(Source.Body)]
        public string Value { get; set; }
    }

It would be great if it worked together with attribute, because attribute gives a few useful behaviors for controller.

Quick fix is to add the [FromHybrid] attribute to the controller-action parameter:

public ActionResult<bool> Get([FromHybrid]Test model)
{
    return true;
}

Here's what I discovered which explains why your version isn't working:

  • Applying API syntax to the controller automatically assigns the parameter's model to have a BindingSource == "Body".
  • When using .AddHybridModelBinder(), this adds a Convention to look for controller-actions which have one complex parameter (i.e. not a string or int) and no currently-assigned BindingSource and assign it with a BindingSource == "Hybrid".

I'm going to update the behavior of the convention to also look at the action-parameter's attributes and apply the hybrid binding source if no other binding-related attributes exist. This will cover cases like the current API behavior along with not overriding a user's intent to use some other non-hybrid binding for a particular parameter.

Wow, great!

When https://www.nuget.org/packages/HybridModelBinding/0.11.0 becomes available, you should be able to remove the [FromHybrid]-attribute.

New behavior looks at the parameter's attributes and applies the hybrid binder only if no other IBindingSourceMetadata-derived (i.e. [FromBody]) attribute is applied.

Works great! Thank you!