[转]Composite Keys With WebApi OData

本文转自:http://chris.eldredge.io/blog/2014/04/24/composite-keys/

In our basic configuration we told the model builder that our entity has a composite key comprised of an ID and a version:

12345678910

登录后复制

public void MapDataServiceRoutes(HttpConfiguration config){    var builder = new ODataConventionModelBuilder();    var entity = builder.EntitySet<ODataPackage>("Packages");    entity.EntityType.HasKey(pkg => pkg.Id);    entity.EntityType.HasKey(pkg => pkg.Version);    // snip}

登录后复制

This is enough for our OData feed to render edit and self links for each individual entity in a form like:

http://localhost/odata/Packages(Id='Sample',Version='1.0.0')

登录后复制

But if we navigate to this URL, instead of getting just this one entity by key, we get back the entire entity set.

To get the correct behavior, first we need an override on our PackagesODataController that gets an individual entity instance by key:

1234567891011121314151617181920

登录后复制

public class PackagesODataController : ODataController{    public IMirroringPackageRepository Repository { get; set; }    public IQueryable<ODataPackage> Get()    {        return Repository.GetPackages().Select(p => p.ToODataPackage()).AsQueryable();    }    public IHttpActionResult Get(        [FromODataUri] string id,        [FromODataUri] string version)    {        var package = Repository.FindPackage(id, version);        return package == null          ? (IHttpActionResult)NotFound()          : Ok(package.ToODataPackage());    }}

登录后复制

However, out of the box WebApi OData doesn’t know how to bind composite key parameters to an action such as this, since the key is comprised of multiple values.

We can fix this by creating a new routing convention that binds the stuff inside the parenthesis to our route data map:

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152

登录后复制

public class CompositeKeyRoutingConvention : IODataRoutingConvention{    private readonly EntityRoutingConvention entityRoutingConvention =        new EntityRoutingConvention();    public virtual string SelectController(        ODataPath odataPath,        HttpRequestMessage request)    {        return entityRoutingConvention          .SelectController(odataPath, request);    }    public virtual string SelectAction(        ODataPath odataPath,        HttpControllerContext controllerContext,        ILookup<string, HttpActionDescriptor> actionMap)    {        var action = entityRoutingConvention            .SelectAction(odataPath, controllerContext, actionMap);        if (action == null)        {            return null;        }        var routeValues = controllerContext.RouteData.Values;        object value;        if (!routeValues.TryGetValue(ODataRouteConstants.Key,          out value))            {              return action;            }        var compoundKeyPairs = ((string)value).Split(',');        if (!compoundKeyPairs.Any())        {            return null;        }        var keyValues = compoundKeyPairs            .Select(kv => kv.Split('='))            .Select(kv =>              new KeyValuePair<string, object>(kv[0], kv[1]));        routeValues.AddRange(keyValues);        return action;    }}

登录后复制

This class decorates a standard EntityRoutingConvention and splits the raw key portion of the URI into key/value pairs and adds them all to the routeValues dictionary.

Once this is done the standard action resolution kicks in and finds the correct action overload to invoke.

This routing convention was adapted from the WebApi ODataCompositeKeySampleproject.

Here we see another difference between WebApi OData and WCF Data Services. In WCF Data Services, the framework handles generating a query that selects a single instance from an IQueryable. This limits our ability to customize how finding an instance by key is done. In WebApi OData, we have to explicitly define an overload that gets an entity instance by key, giving us more control over how the query is executed.

This distinction might not matter for most projects, but in the case of NuGet.Lucene.Web, it enables a mirror-on-demand capability where a local feed can fetch a package from another server on the fly, add it to the local repository, then send it back to the client as if it was always there in the first place.

To customize this in WCF Data Services required significant back flips.

 

Series Index

IntroductionBasic WebApi ODataComposite KeysDefault Streams

 

以上就是[转]Composite Keys With WebApi OData的详细内容,更多请关注【创想鸟】其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至253000106@qq.com举报,一经查实,本站将立刻删除。

发布者:PHP中文网,转转请注明出处:https://www.chuangxiangniao.com/p/2493972.html

(0)
上一篇 2025年3月5日 01:42:44
下一篇 2025年3月5日 01:43:04

AD推荐 黄金广告位招租... 更多推荐

相关推荐

发表回复

登录后才能评论