- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我们将 Microsoft ASP.NET MVC OData WebAPI 用于我们的 Web 服务。由于围绕层次结构 ID 的一些数据架构问题(超出本次对话的范围),我们的一些 GET 操作必须使用 ODataQueryOptions 并手动操作表达式以添加额外的限制。我们这样做(为清楚起见,删除了错误处理代码并调用了内联的其他方法):
public IQueryable<Person> Get(ODataQueryOptions<Person> oDataQueryOptions)
{
IQueryable<Person> result;
IQueryable<Person> dataSet = context.Persons;
var tempQuery = oDataQueryOptions.ApplyTo(dataSet).Cast<Person>();
var modifier = new HierarchyNodeExpressionVisitor(GetDescendantsOfNode, GetAncestorsOfNode);
var expression = modifier.ModifyHierarchyNodeExpression(tempQuery.Expression);
result = context.Persons.Provider.CreateQuery<Person>(expression);
return result;
}
这一段时间以来一直运行良好,但我们一直热切地等待选择和扩展,以便我们可以更好地控制从我们的服务接收的数据。星期一,我们将我们的开发环境更新为 WebApi OData 5.0.0-rc1 并开始选择和扩展工作,但我们不能将它用于这些使用 ODataQueryOptions 的服务。我们只能将它用于我们的其他服务。如果我们使用 $select
和/或 $expand
查询上面的代码,我们会得到以下错误:
"message": "The 'ObjectContent`1' type failed to serialize the response body for content type 'application/json; charset=utf-8'.",
"type": "System.InvalidOperationException",
"stacktrace": "",
"internalexception":
{
"message": "Unable to cast the type 'System.Web.Http.OData.Query.Expressions.SelectAllAndExpand`1' to type 'OurCompany.Domains.Data.Models.Person'. LINQ to Entities only supports casting EDM primitive or enumeration types.",
"type": "System.NotSupportedException",
"stacktrace": " at System.Data.Objects.ELinq.ExpressionConverter.ValidateAndAdjustCastTypes(TypeUsage toType, TypeUsage fromType, Type toClrType, Type fromClrType) at System.Data.Objects.ELinq.ExpressionConverter.GetCastTargetType(TypeUsage fromType, Type toClrType, Type fromClrType, Boolean preserveCastForDateTime) at System.Data.Objects.ELinq.ExpressionConverter.CreateCastExpression(DbExpression source, Type toClrType, Type fromClrType) at System.Data.Objects.ELinq.ExpressionConverter.MethodCallTranslator.CastMethodTranslator.Translate(ExpressionConverter parent, MethodCallExpression call) at System.Data.Objects.ELinq.ExpressionConverter.MethodCallTranslator.TypedTranslate(ExpressionConverter parent, MethodCallExpression linq) at System.Data.Objects.ELinq.ExpressionConverter.Convert() at System.Data.Objects.ELinq.ELinqQueryState.GetExecutionPlan(Nullable`1 forMergeOption) at System.Data.Objects.ObjectQuery`1.GetResults(Nullable`1 forMergeOption) at System.Data.Objects.ObjectQuery`1.System.Collections.Generic.IEnumerable.GetEnumerator() at System.Data.Entity.Infrastructure.DbQuery`1.System.Collections.IEnumerable.GetEnumerator() at System.Web.Http.OData.Formatter.Serialization.ODataFeedSerializer.WriteFeed(IEnumerable enumerable, IEdmTypeReference feedType, ODataWriter writer, ODataSerializerContext writeContext) at System.Web.Http.OData.Formatter.ODataMediaTypeFormatter.WriteToStream(Type type, Object value, Stream writeStream, HttpContent content, HttpContentHeaders contentHeaders) at System.Web.Http.OData.Formatter.ODataMediaTypeFormatter.WriteToStreamAsync(Type type, Object value, Stream writeStream, HttpContent content, TransportContext transportContext) --- End of stack trace from previous location where exception was thrown --- at System.Runtime.CompilerServices.TaskAwaiter.ThrowForNonSuccess(Task task) at System.Runtime.CompilerServices.TaskAwaiter.HandleNonSuccessAndDebuggerNotification(Task task) at System.Web.Http.WebHost.HttpControllerHandler.d__10.MoveNext()"
}
我做了一些谷歌搜索并偶然发现了 this和 this ,但都没有帮助。似乎没有人在做我们正在做的事情并尝试使用选择和展开。我们如何解决这个问题?我在这里不知所措......
最佳答案
问题出在这行代码,
var tempQuery = oDataQueryOptions.ApplyTo(dataSet).Cast<Person>();
Cast 一次无效$select
和 $expand
应用结果不再是一个人。这将是一个 Wrapper<Person>
仅包含客户要求的属性。您可能需要修改您的 HierarchyNodeExpressionVisitor
考虑到这一点。
此外,尝试将您的操作更改为此以处理结果可能不是 IQueryable<Person>
的事实。不再。
public IHttpActionResult Get(ODataQueryOptions<Person> oDataQueryOptions)
{
IQueryable result;
IQueryable<Person> dataSet = context.Persons;
IQueryable tempQuery = oDataQueryOptions.ApplyTo(dataSet);
var modifier = new HierarchyNodeExpressionVisitor(GetDescendantsOfNode, GetAncestorsOfNode);
var expression = modifier.ModifyHierarchyNodeExpression(tempQuery.Expression);
result = context.Persons.Provider.CreateQuery(expression);
return Ok(result, result.GetType());
}
private IHttpActionResult Ok(object content, Type type)
{
Type resultType = typeof(OkNegotiatedContentResult<>).MakeGenericType(type);
return Activator.CreateInstance(resultType, content, this) as IHttpActionResult;
}
关于c# - $select 和 $expand 中断 ODataQueryOptions——如何修复?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18879779/
我有一个可用的(简化的)ODataController,方法如下。 public class MyTypeController : ODataController { [HttpGet] [E
在尝试进行一些测试驱动开发时,我创建了最基本的可构建方法: public class NoteService : INoteService { public IEnumerable GetNo
我有一个常规 Controller ,它具有 odata 功能: [Route("[controller]")] public class ChannelsController : Controlle
我是为 WebAPI 编写测试用例的新手。我曾经看到过类似的问题,但没有得到回答,但我想知道如果我的 API 有 ODataQueryOptions,我将如何测试它们。作为参数的一部分。见下文: pu
我正在尝试通过 EF Core 和 OData(7.1.0) 实现对 Sql Server 的查询。 操作方法如下: [HttpGet] public IEnumerable Get(ODataQue
我需要能够从 ODataQueryOptions 进行转换至 RestRequest为了能够发出带有指定过滤器的 RestRequest,并创建了以下辅助类: public static class
我正在实现一个 Web API 接口(interface)来支持一些相当复杂的查询来运行它,并且遇到了最大请求 URI 长度的问题。 我的 Web API 方法的定义如下所示(使用 Automappe
更新 投票here on User Voice解决歧义。 我编写了一个继承于 ODataController 的 OData WebAPI Controller 。 . public class Ma
最近我们决定使用包 Microsoft.AspNetCore.OData为我们的服务解决方案。这些服务具有 ODataQueryOptions 参数,并使用它来过滤它们提供的数据。为了对此进行单元测试
我有一个 Web API 项目,在没有 OData 支持的情况下使用了很多年,只有标准的 URL 参数。 我现在希望向该 API 添加 OData 支持,但由于该 API 不是基于可查询模型构建的,因
我正在构建一个 OData v.4 Web 服务,该服务必须公开从另一个第 3 方 Web 源检索的数据,因此数据与 LINQ 世界中的任何内容都不相似,即:没有 IQueryable,没有上下文,没
我正在构建一个 OData v.4 Web 服务,该服务必须公开从另一个第 3 方 Web 源检索的数据,因此数据与 LINQ 世界中的任何内容都不相似,即:没有 IQueryable,没有上下文,没
有 2 个类,Foo 和 Bar。在 Foo 对象中嵌套了一个 Bar 对象。 public class Foo { public Guid FooId { get; set; } p
问题开始here ,我无法调用 $select或 $expand在 results . 给定: var results = options.ApplyTo(_uow.Repos
我们将 Microsoft ASP.NET MVC OData WebAPI 用于我们的 Web 服务。由于围绕层次结构 ID 的一些数据架构问题(超出本次对话的范围),我们的一些 GET 操作必须使
我在 OData v4 中使用 ApiController(不是 ODataController),其中有一个将 ODataQueryOptions 作为参数的 Get 操作,如下所示: public
我尝试了使用 Breeze 和 API Controller 、使用过滤器(部分使用自定义对象,另一部分使用 ODataQueryOptions)加载项目列表的不同方法,但结果证明它们都不是真正成功的
我想重建我的上一个项目。过去,我没有使用任何 Web API。我可以使用 ODataQueryOptions 来执行 $filter、$orderby 、$top 、$skip 操作吗对于我自己的 h
我正在将我们现有的 API 迁移到 .net 5,我面临单元测试迁移的问题。我想保留测试和测试 Controller ,但我找不到处理 ODataQueryOptions 的方法,因为此类已更改,我再
我正在将我们现有的 API 迁移到 .net 5,我面临单元测试迁移的问题。我想保留测试和测试 Controller ,但我找不到处理 ODataQueryOptions 的方法,因为此类已更改,我再
我是一名优秀的程序员,十分优秀!