EDM模型根上的ASP.NET Web API OData操作

时间:2013-04-03 21:56:29

标签: vb.net asp.net-web-api action odata

我正在使用OData构建Web API服务,并希望在服务中将方法公开为Action,如下所示。

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类吗?

1 个答案:

答案 0 :(得分:8)

我们目前没有开箱即用的支持,但它很容易自己做。下面的示例(这个很好的样本实际上来自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);
    }
}