8

我在http://www.asp.net/web-api/overview/odata-support-in-aspnet-web-api/working-with-entity-relations使用 OData 示例项目。在 Get 中,我希望能够更改 EntitySetController 的 QueryOptions 中的过滤器:

public class ProductsController : EntitySetController<Product, int>
{
    ProductsContext _context = new ProductsContext();

    [Queryable(AllowedQueryOptions=AllowedQueryOptions.All)]
    public override IQueryable<Product> Get()
    {
        var products = QueryOptions.ApplyTo(_context.Products).Cast<Product>();
        return products.AsQueryable();
    }

我希望能够找到特别提到的属性。我可以通过解析this.QueryOptions.Filter.RawValue属性名称来做到这一点,但我无法更新它,RawValue因为它是只读的。FilterQueryOption但是,我可以从修改后创建另一个实例,RawValue但我无法将其分配给它,this.QueryOptions.Filter因为它也是只读的。

我想我可以调用新过滤器的ApplyTo传递它_context.Products,但是我需要单独调用like和ApplyTo的其他属性。还有比这更好的解决方案吗?QueryOptionsSkipOrderBy

更新

我尝试了以下方法:

    public override IQueryable<Product> Get()
    {
        IQueryable<Product> encryptedProducts = _context.Products;

        var filter = QueryOptions.Filter;
        if (filter != null && filter.RawValue.Contains("Name"))
        {
            var settings = new ODataQuerySettings();
            var originalFilter = filter.RawValue;
            var newFilter = ParseAndEncyptValue(originalFilter);
            filter = new FilterQueryOption(newFilter, QueryOptions.Context);
            encryptedProducts = filter.ApplyTo(encryptedProducts, settings).Cast<Product>();

            if (QueryOptions.OrderBy != null)
            {
                QueryOptions.OrderBy.ApplyTo<Product>(encryptedProducts);
            }
        }
        else
        {
            encryptedProducts = QueryOptions.ApplyTo(encryptedProducts).Cast<Product>();
        }

        var unencryptedProducts = encryptedProducts.Decrypt().ToList();

        return unencryptedProducts.AsQueryable();
    }

它似乎正在发挥作用。如果我设置断点,我可以在unencryptedProducts列表中看到我的产品,但是当方法返回时,我没有得到任何项目。我试着[Queryable(AllowedQueryOptions=AllowedQueryOptions.All)]重新装上,但没有效果。有什么想法为什么我没有收到物品?

更新 2

我发现我的查询被应用了两次,即使我没有使用该Queryable属性。这意味着即使我有要返回的项目列表正在使用未加密的值进行查询,因此没有返回任何值。

我尝试ODataController改用:

public class ODriversController : ODataController
{

    //[Authorize()]
    //[Queryable(AllowedQueryOptions = AllowedQueryOptions.All)]
    public IQueryable<Products> Get(ODataQueryOptions options)
    {

这有效!这是否表明存在错误EntitySetController

4

1 回答 1

4

您可能需要重新生成 ODataQueryOptions 来解决您的问题。假设你想修改添加 $orderby,你可以这样做:

string url = HttpContext.Current.Request.Url.AbsoluteUri;
url += "&$orderby=name";
var request = new HttpRequestMessage(HttpMethod.Get, url);
ODataModelBuilder modelBuilder = new ODataConventionModelBuilder();
modelBuilder.EntitySet<Product>("Product");
var options = new ODataQueryOptions<Product>(new ODataQueryContext(modelBuilder.GetEdmModel(), typeof(Product)), request);
于 2014-01-18T02:34:32.020 回答