gpt4 book ai didi

c# - IdentityDbContext.SaveChanges 在 Claims.Remove 或 Claims.Clear 后失败

转载 作者:行者123 更新时间:2023-11-30 22:03:45 26 4
gpt4 key购买 nike

我直接使用 AspNetUserClaims 表,因为它主要属于我的 ASP.NET 应用程序,但也被 Windows 应用程序使用。 ASP.NET 应用程序使用 Google 进行外部登录,而 Windows 应用程序定期轮询 Google 帐户并不时刷新访问 token 。因此,现在我正在该 Windows 应用程序下工作。

尝试从 AspNetUserClaims 表中删除声明时出现异常。它一直告诉我“需要 UserId 字段。”

如果我使用 dc.Configuration.ValidateOnSaveEnabled = false 禁用验证,我看到它尝试执行 UPDATE 查询(将所有值作为 NULL 或默认值传递)而不是 DELETE。无论我试图删除哪个特定声明,即使我使用 Clear 方法将它们全部删除,它都会失败。

该代码还显示了所有用户的所有当前可用声明(这就是为什么使用 foreach 而不是 dc.Users.Where 的原因)但我出于调试目的删除了这些部分。

var dc = new IdentityDbContext();
foreach (var u in dc.Users.ToList())
{
if (u.Email == "something@gmail.com")
{
u.Claims.Clear();
}
}
dc.SaveChanges();

这仅涉及删除声明。我使用 Claims.Add 或更改 Claim.ClaimValue 更新现有声明的值成功添加了新声明。可能是什么原因?

实际上,我可以在不删除声明的情况下生活,因为我的 Windows 应用程序只需要更新声明(使用刷新 token 更新访问 token )。但是我很好奇为什么删除不起作用。

好吧,我找到了一种使用以下代码删除 token 的方法:

var dc = new IdentityDbContext();
var manager = new UserManager<IdentityUser>(new UserStore<IdentityUser>(dc));
var user = manager.FindByEmail("something@gmail.com");
Claim c = manager.GetClaims(user.Id).Where(c => c.Type == "GoogleAccessToken").FirstOrDefault();
if (c != null)
{
manager.RemoveClaim(user.Id, c);
}

但是仍然不清楚为什么没有使用 UserManager 的原始代码不能删除声明。

最佳答案

这些是 Entity Framework 的怪癖。
当你说

user.Claims.Remove(claim);
dbContext.SaveChanges();

这只会将声明对象的外键重置为空,但这是不允许的。 EF 不知道更好,因此没有 DELETE陈述。

如果要删除 EF 中的对象,请执行以下操作:

dbContext.UserClaims.Remove(claim);

但是你需要映射IDbSet<UserClaim> UserClaims给你的dbContext ,因为它默认不可用。

this article 中阅读有关删除对象的 EF 方式的更多信息

关于c# - IdentityDbContext.SaveChanges 在 Claims.Remove 或 Claims.Clear 后失败,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25822538/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com