[前言]
当我将我的 api 从 .netcore2.1 升级到 .netcore3.1 时,我发现我需要大幅更改我的实体框架代码。
我没有意识到我的远程数据库中的存储过程中有一个错误,这在我的 .netcore2.1 代码中没有表达出来。
我遇到的错误消息并没有将我指向检查存储过程内部的方向。如果搜索错误消息对某人有帮助,我会留下我的问题的最终形式。
[问题]
我有以下代码来调用返回整数的存储过程
该代码在服务器端单元测试中工作,但在通过 api 或通过 SwaggerUI 调用时失败
var obj = connect.Ints.FromSqlInterpolated<intDto>(@$"Set NOCOUNT ON
declare @num int
exec @num = spGetDefaultID {userName}
select @num as num").ToList();
id = (int)obj.First().num;
connect 是我的 DbContext 包含
public DbSet<intDto> Ints { get; set; } // I actually don't want a table
和
public class intDto
{
public int num { get; set; }
}
我按照建议创建了一个虚拟视图,这样我就不会在我的数据库中得到不需要的表。
并且有
protected override void OnModelCreating(ModelBuilder modelBuilder)
{
base.OnModelCreating(modelBuilder);
modelBuilder.Entity<intDto>).HasNoKey().ToView("view_name_that_doesnt_exist");
调用堆栈是
System.InvalidOperationException: The required column 'num' was not present in the results of a 'FromSql' operation.
at Microsoft.EntityFrameworkCore.Query.Internal.BufferedDataReader.BufferedDataRecord.InitializeFields()
at Microsoft.EntityFrameworkCore.Query.Internal.BufferedDataReader.BufferedDataRecord.Initialize(DbDataReader reader, IReadOnlyList`1 columns)
at Microsoft.EntityFrameworkCore.Query.Internal.BufferedDataReader.Initialize(IReadOnlyList`1 columns)
at Microsoft.EntityFrameworkCore.Storage.RelationalCommand.ExecuteReader(RelationalCommandParameterObject parameterObject)
at Microsoft.EntityFrameworkCore.Query.Internal.QueryingEnumerable`1.Enumerator.InitializeReader(DbContext _, Boolean result)
at Microsoft.EntityFrameworkCore.Storage.ExecutionStrategy.ExecuteImplementation[TState,TResult](Func`3 operation, Func`3 verifySucceeded, TState state)
at Microsoft.EntityFrameworkCore.Storage.ExecutionStrategy.Execute[TState,TResult](TState state, Func`3 operation, Func`3 verifySucceeded)
at Microsoft.EntityFrameworkCore.Query.Internal.QueryingEnumerable`1.Enumerator.MoveNext()
at System.Collections.Generic.List`1..ctor(IEnumerable`1 collection)
at System.Linq.Enumerable.ToList[TSource](IEnumerable`1 source)
at MyAPi.Job_Management.JobDataRead.GetCartIdForUser(ApiDbContext connect, String userName)
注意我已经大量编辑了这个问题,因为有一次我让 api 工作,但不再工作。
我发现我必须使用 .ToList() 否则出现错误
System.InvalidOperationException:FromSqlRaw 或 FromSqlInterpolated 是使用不可组合的 SQL 调用的,并且查询在其上构成。考虑
AsEnumerable
在 FromSqlRaw 或 FromSqlInterpolated 方法之后调用以在客户端执行组合。
[更新]
我弹出以下代码并没有失败
var obj = db.Ints.FromSqlInterpolated(@$"Set NOCOUNT on
select count(*) as num from people
where email like {name} ").ToList();
所以下一步是尝试制作一个新的存储过程
或者,也许它是数据库版本..
有趣的是单元测试失败,除非我有
Microsoft.EntityFrameworkCore.Relational.dll
在正在测试的项目中引用。即使我没有明确使用它。
这种情况下的错误消息是
System.MissingMethodException: 'Method not found:'System.Linq.IQueriable '1<!!O>
Microsoft.EntityFrameworkCore.RelationalQueryableExtensions.
FromSqlInterpolated(Microsoft.EntityFrameworkCOre.DbSet'1<!!0>,
System.FormattableString)'
我尝试将我的代码更改为
var obj = connect.Ints.FromSqlInterpolated<intDto>(@$"declare @num int
exec @num = spGetDefaultID {userName}").ToList();
但是我的单元测试出错了
The underlying reader doesn't have as many fields as expected
[更新]
远程存储过程使用结果集返回数据。
create PROCEDURE [dbo].[spGetDefaultID]
@email varchar(300)
AS
BEGIN
SET NOCOUNT ON;
declare @cartID int
select @CartID= id from people where email = @email /* simplified */
select @CartID /* this line was not present in my local database */
return @CartID
END