'How to get navigation properties with [Owned] attribute Tracked in Entity Framework Core
I got EF tracking turned off by default on my project.
I got a DB model class UserModel:
public class UserModel : BaseAdminDbModel
{
[Key]
public int UserId { get; set; }
[NotMapped]
public string UserName { get; set; }
[Required]
public RightsModel R { get; set; }
public BranchUserModel()
{
R = new RightsModel();
}
[ForeignKey("UserId")]
public UserDataModel User { get; set; }
}
RightsModel:
[Owned]
public class RightsModel
{
public bool? RightOne { get; set; }
public bool? RightTwo { get; set; }
public bool? RightN { get; set; }
}
I want to add Owned navigation properties to be tracked before saving db context so every change in RightModel
in UserModel
will be saved to database (currently its ignoring those fields)
I want to have somewhat like this:
_dbContext.Entry(model).State = EntityState.Modified;
foreach(var navigationRightProperty in _dbContext.Entry(model).Navigations)
{
navigationRightProperty.Load();
}
Result:
Changes to RightModel will be also tracked with the main UserModel
entity.
Solution 1:[1]
After some time I came up to solve this issue. Drakk L solution is good if you have tracking turned on by default and want to rely on real objects.
I wanted more abstract way to achieve this:
var navigationOwnedProps = _dbContext
.Entry(model)
.Navigations
.Where(e => e.Metadata
.ClrType.CustomAttributes.Any(c => c.AttributeType ==
typeof(OwnedAttribute))).Select(e => e.CurrentValue)
.ToList();
foreach (var nav in navigationOwnedProps)
{
_dbContext.Entry(nav).State = EntityState.Modified;
}
Solution 2:[2]
Model tracking is enabled by default for EF Core. So if you use the Include method on R
property while making your request and making changes to it, then you call the SaveChanges
method all changes will be saved in the database
var id = 1;
var model = _dbContext.UserModel.Include(x => x.R ).FirstOrDefalt(x => x.Id == id)
model.R.RightOne = true;
_dbContext.SaveChanges();
Also you can change your constructor for UserModel
like
public UserModel()
{
this.R = new HashSet<RightsModel>();
}
Sources
This article follows the attribution requirements of Stack Overflow and is licensed under CC BY-SA 3.0.
Source: Stack Overflow
Solution | Source |
---|---|
Solution 1 | Hantick |
Solution 2 | Darkk L |