[转]Support Composite Key in ASP.NET Web API OData


本文摘自PHP中文网,作者PHP中文网,侵删。

本文转自:

he default EntitySetController doesn't support composite keys. So if you have composite key models, you need some additional work. Here is an example about how to do that.

The model is simple:

1

2

3

4

5

6

7

8

9

10

<code>public class Person

{

    [Key]

    public string FirstName { get; set; }

    [Key]

    public string LastName { get; set; }

 

    public int Age { get; set; }

}

</code>

The odata url for this model will look like:

1

2

<code>GET http://localhost:33051/People(FirstName='Kate',LastName='Jones') HTTP/1.1

</code>

And we want to have strong typed parameters in web api actions to this URL.

1

2

<code>    public Person Get([FromODataUri] string firstName, [FromODataUri] string lastName)

</code>

Note that the FromODataUri model binder attribute is used to parse from odata uri representation to clr type. In odata, string value is "'xxx'" and we want it to be "xxx".

In order to make the route to work, you can add a custom routing convention to parse the key path. Here is a sample implementation:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

<code>public class CompositeKeyRoutingConvention : EntityRoutingConvention

{

    public override string SelectAction(System.Web.Http.OData.Routing.ODataPath odataPath, System.Web.Http.Controllers.HttpControllerContext controllerContext, ILookup<string, System.Web.Http.Controllers.HttpActionDescriptor> actionMap)

    {

        var action = base.SelectAction(odataPath, controllerContext, actionMap);

        if (action != null)

        {

            var routeValues = controllerContext.RouteData.Values;

            if (routeValues.ContainsKey(ODataRouteConstants.Key))

            {

                var keyRaw = routeValues[ODataRouteConstants.Key] as string;

                IEnumerable<string> compoundKeyPairs = keyRaw.Split(',');

                if (compoundKeyPairs == null || compoundKeyPairs.Count() == 0)

                {

                    return action;

                }

 

                foreach (var compoundKeyPair in compoundKeyPairs)

                {

                    string[] pair = compoundKeyPair.Split('=');

                    if (pair == null || pair.Length != 2)

                    {

                        continue;

                    }

                    var keyName = pair[0].Trim();

                    var keyValue = pair[1].Trim();

 

                    routeValues.Add(keyName, keyValue);

                }

            }

        }

 

        return action;

    }

}

</code>

The convention is inherited from EntityRoutingConvention, which is the default convetion to handle entity key. By calling base.SelectAction, it will add the full key path into routeValues. The new convention will check if it contains "," and seperate it into multiple keys and set each of them into routeValues. So when web api select actions, it will use those values to determine which action to choose. If there is no "," found, it behaves same as base convetion.

To register the convetion, you need to set it when mapping odata route:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

<code>public static class WebApiConfig

{

    public static void Register(HttpConfiguration config)

    {

        config.EnableQuerySupport();

 

        var mb = new ODataConventionModelBuilder(config);

        mb.EntitySet<Person>("People");

 

        var conventions = ODataRoutingConventions.CreateDefault();

        conventions.Insert(0, new CompositeKeyRoutingConvention());

 

        config.Routes.MapODataRoute(

            routeName: "OData",

            routePrefix: null,

            model: mb.GetEdmModel(),

            pathHandler: new DefaultODataPathHandler(),

            routingConventions: conventions);

    }

}

</code>

Register the route at the postion 0 is to make it be executed before other default routing convetions. So the default EntityRoutingConvetion won't be executed before it. After that, you should be able to get routing work.

Then, how to build url for composite keys? 
You don't need to do that for odata links include edit link and self link when using ODataConventionModelBuilder. It will automatically identify composite keys and build the uri for you.

However, you need to build the link for location header. Here is a sample code from PeopleController.cs to handle post request:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

<code>public HttpResponseMessage PostPerson(Person person)

{

    if (ModelState.IsValid)

    {

        _repo.UpdateOrAdd(person);

 

        HttpResponseMessage response = Request.CreateResponse(HttpStatusCode.Created, person);

        string key = string.Format(

            "{0}={1},{2}={3}",

            "FirstName", ODataUriUtils.ConvertToUriLiteral(person.FirstName, Microsoft.Data.OData.ODataVersion.V3),

            "LastName", ODataUriUtils.ConvertToUriLiteral(person.LastName, Microsoft.Data.OData.ODataVersion.V3));

        response.Headers.Location = new Uri(Url.ODataLink(

            new EntitySetPathSegment("People"),

            new KeyValuePathSegment(key)));

        return response;

    }

    else

    {

        return Request.CreateResponse(HttpStatusCode.BadRequest);

    }

}

</code>

Hope it helps.

 

以上就是[转]Support Composite Key in ASP.NET Web API OData的详细内容!

相关阅读 >>

asp.net core中关于webapi几种版本控制对比详解(图)

极客学院asp.net视频教程的资料推荐

asp.net mvc 对用户输入的字符串做trim处理的方法实例

asp.net中有关config文件的读写功能讲解

两个需要在asp.net程序中值得注意的地方

asp.net性能监控及其优化入门

asp.net之exceptionfilter过滤器

基于calendar实现blog日历的实例详解

分享asp.net mvc4如何实现通过id更新表单内容的实例分析

asp.net的简单定义与介绍

更多相关阅读请进入《Composite》频道 >>




打赏

取消

感谢您的支持,我会继续努力的!

扫码支持
扫码打赏,您说多少就多少

打开支付宝扫一扫,即可进行扫码打赏哦

分享从这里开始,精彩与您同在

评论

管理员已关闭评论功能...