我正在使用OData构建Web API服务,并希望在服务中将方法公开为Action,如下所示. http://myServer/odata/myAction 我目前正在映射OData路由如下: Dim modelBuilder As ODataModelBuilder = New ODataConventionModelBui
http://myServer/odata/myAction
我目前正在映射OData路由如下:
Dim modelBuilder As ODataModelBuilder = New ODataConventionModelBuilder modelBuilder.EntitySet(Of Product)("Products") Dim myAction = modelBuilder.Action("myAction") myAction.Parameter(Of String)("Parameter1") myAction.Returns(Of Boolean)() Dim model As IEdmModel = modelBuilder.GetEdmModel config.Routes.MapODataRoute("ODataRoute", "odata", model)
This wonderful tutorial显示了如何将动作与这样的实体相关联:
http://myServer/odata/Products(1)/myAction
在本教程之后,我可以在使用以下行创建模型后,在ProductsController类中编写操作的方法:
Dim myAction = modelBuilder.Entity(Of Product).Action("myAction")
但是,如果我不想将操作与实体关联,那么我将在哪里编写操作方法?我需要编写一个DefaultController类吗?
我们目前没有开箱即用的支持,但它很容易自己做.下面的例子(这个漂亮的样本实际上来自Mike Wasson,尚未公开:-))------------------------------------------------------ // CreateMovie is a non-bindable action. // You invoke it from the service root: ~/odata/CreateMovie ActionConfiguration createMovie = modelBuilder.Action("CreateMovie"); createMovie.Parameter<string>("Title"); createMovie.ReturnsFromEntitySet<Movie>("Movies"); // Add a custom route convention for non-bindable actions. // (Web API does not have a built-in routing convention for non-bindable actions.) IList<IODataRoutingConvention> conventions = ODataRoutingConventions.CreateDefault(); conventions.Insert(0, new NonBindableActionRoutingConvention("NonBindableActions")); // Map the OData route. Microsoft.Data.Edm.IEdmModel model = modelBuilder.GetEdmModel(); config.Routes.MapODataRoute("ODataRoute", "odata", model, new DefaultODataPathHandler(), conventions); -------------------------------------------------------------- // Implements a routing convention for non-bindable actions. // The convention maps "MyAction" to Controller:MyAction() method, where the name of the controller // is specified in the constructor. public class NonBindableActionRoutingConvention : IODataRoutingConvention { private string _controllerName; public NonBindableActionRoutingConvention(string controllerName) { _controllerName = controllerName; } // Route all non-bindable actions to a single controller. public string SelectController(ODataPath odataPath, System.Net.Http.HttpRequestMessage request) { if (odataPath.PathTemplate == "~/action") { return _controllerName; } return null; } // Route the action to a method with the same name as the action. public string SelectAction(ODataPath odataPath, System.Web.Http.Controllers.HttpControllerContext controllerContext, ILookup<string, System.Web.Http.Controllers.HttpActionDescriptor> actionMap) { if (controllerContext.Request.Method == HttpMethod.Post) { if (odataPath.PathTemplate == "~/action") { ActionPathSegment actionSegment = odataPath.Segments.First() as ActionPathSegment; IEdmFunctionImport action = actionSegment.Action; if (!action.IsBindable && actionMap.Contains(action.Name)) { return action.Name; } } } return null; } } -------------------------------------------------- // Controller for handling non-bindable actions. [ODataFormatting] [ApiExplorerSettings(IgnoreApi = true)] public class NonBindableActionsController : ApiController { MoviesContext db = new MoviesContext(); [HttpPost] public Movie CreateMovie(ODataActionParameters parameters) { if (!ModelState.IsValid) { throw new HttpResponseException(HttpStatusCode.BadRequest); } string title = parameters["Title"] as string; Movie movie = new Movie() { Title = title }; db.Movies.Add(movie); db.SaveChanges(); return movie; } protected override void Dispose(bool disposing) { db.Dispose(); base.Dispose(disposing); } }