Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Broken code for saving a graph of entities #4863

Open
litoj opened this issue Nov 2, 2024 · 7 comments
Open

Broken code for saving a graph of entities #4863

litoj opened this issue Nov 2, 2024 · 7 comments

Comments

@litoj
Copy link

litoj commented Nov 2, 2024

Type of issue

Code doesn't work

Description

I am trying to save a graph just like in the example shown. But there are multiple issues with the example:

  • whatever I do, it throws an error when saving the Post, because the executed INSERT has still BlogId=0
  • providing the Blog to the Post didn't help
  • using Sync/Async processing didn't change anything
  • the Post has many objects that are expected to be not null, yet aren't required to be set in constructor - this generates compilator warnings

Page URL

https://learn.microsoft.com/en-us/ef/core/saving/related-data#adding-a-graph-of-new-entities

Content source URL

https://github.com/dotnet/EntityFramework.Docs/blob/main/entity-framework/core/saving/related-data.md

Document Version Independent Id

c7517e7d-f011-a24c-53da-913771913998

Article author

@ajcvickers

@litoj litoj changed the title Incorrect documentation for saving a graph of entities Broken code documentation for saving a graph of entities Nov 2, 2024
@litoj litoj changed the title Broken code documentation for saving a graph of entities Broken code for saving a graph of entities Nov 2, 2024
@timdeschryver
Copy link
Contributor

I tried to reproduce this, and I initially received the following error

Microsoft.Data.SqlClient.SqlException (0x80131904): Cannot insert the value NULL into column 'Content', table 'EFDocs.dbo.Posts'; column does not allow nulls. UPDATE fails.

This occurs because I created a new project with Nullable reference types enabled. Reading the comment "this generates compilator warnings", I expect this is also the case for you.

To fix the example, either remove the <Nullable>enable</Nullable> config in your csproj file, or set the content of the posts.

using (var context = new BloggingContext())
{
    var blog = new Blog
    {
        Url = "http://blogs.msdn.com/dotnet",
        Posts = new List<Post>
        {
            new Post { Title = "Intro to C#", Content = string.Empty},
            new Post { Title = "Intro to VB.NET", Content = ""},
            new Post { Title = "Intro to F#", Content = "Intro to F#"},
        }
    };

    context.Blogs.Add(blog);
    context.SaveChanges();
}

If wanted, I can update the docs by assigning a value to the content properties.

@litoj
Copy link
Author

litoj commented Nov 6, 2024

The primary issue is rather that there is no clarification what happens with the post.BlogId - does it just magically figure out it is from the object above? Also that part in particular did not work for me - I wasn't able to save it like that and instead it always threw an error about breaking FK constraint - precisely because the posts didn't get the BlogId set.

@timdeschryver
Copy link
Contributor

timdeschryver commented Nov 6, 2024

@litoj it's normal that the entity has id 0, during the insert it should receive the appropriate id.
For example, with Sql Server the Id column will be the Primary Key, which is auto incremented ([BlogId] [int] IDENTITY(1,1) NOT NULL)

@litoj
Copy link
Author

litoj commented Nov 6, 2024

Yes, but how does the post.BlogId get set? Because in my tests it didn't and that's why I got the errors.

@timdeschryver
Copy link
Contributor

Sorry @litoj , I don't know - normally, that should also work automatically.

@litoj
Copy link
Author

litoj commented Nov 24, 2024

I resolved to using a workaround by saving each entity manually. After getting the parent entity ID by saving it, I only then added the child collection to it and then it worked.

After finding out only the root entity needs to get saved separately, I simplified the code to this:

// duplicate `entity` while duplicating all of its children (.Bundles and .Bundles.Slots)
// ...

        await DbContext.TestTemplate.AddAsync(entity);
        var bundles = entity.Bundles;
        entity.Bundles = [];
        await DbContext.SaveChangesAsync();
        entity.Bundles = bundles;
        await DbContext.SaveChangesAsync();

@litoj
Copy link
Author

litoj commented Jan 16, 2025

In my previous comment, I actually am using this feature successfully, but only for the grandchildren (Bundles->Slots). It seems that it works fine for one layer of children, but doesn't cascade for multiple layers (Template->Bundles->Slots).

Is that maybe something that is planned to enable?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

2 participants