Bulk Inserting data using Entity Framework
Using tools like Entity Framework makes life far easier for a developer. Recently I blogged about how using them is what makes .Net Core one of the best platforms for prototype development, but the benefits don’t end there. They are also great from a security perspective by cutting a lot of risk around SQL injection attacks just by avoiding easy mistakes when using regular ADO.NET.
However, they do have some downsides, a main one being that they are particularly slow when it comes to doing bulk inserts to a database.
For example, assume you have an application which regularly receives an xml import file consisting of 200,000 records and each one either needs to be an insert of an update into the db. You’ll quickly learn that looping through the whole lot and then calling save changes results in a process taking an extremely long time to run, it may even just timeout. You then decide to get rid of that long save changes line by breaking it up into blocks of 500 and call save changes for each of those. That may save the timeout issue, but it still results in a process potentially lasting around an hour.
The problem is that this is a scenario Entity Framework or EF.Core just weren’t designed to handle. As a solution you could opt to drop Entity Framework altogether and revert to something like a native SQL Bulk Insert command, but what if you need to be doing some processing in code on the record before the import happens? What if you have one of those classic not quite always valid XML, XML files which would cause SQLs Bulk Insert to fail.
The solution is to use an open source extension called EFCore.BulkExtensions.
EFCore.BulkExtensions
EFCore.BulkExtensions is a set of extension methods to Entity Framework that provide the functionality to do bulk inserts. You can add it to your project using NuGet and you’ll find the project on GitHub here https://github.com/borisdj/EFCore.BulkExtensions
Usage is also very simple to do. Let’s assume you have some existing tradition EF code that loops through a collection and for each one create a new db item and adds it to the db:
1public void DoImport(List<foo> collection)2{3 foreach (var item in collection)4 {5 Jobs job = new Jobs();67 job.DateAdded = DateTime.UtcNow;8 job.Name = item.Name;9 job.Location = item.Location;1011 await dbContext.Jobs.AddAsync(job);12 }1314 await dbContext.SaveChangesAsync();15}
Rather than adding each item to the Entity Framework db context, you instead create a list of those objects and then call a BulkInsert function with them on your db context.
1public void DoImport(List<foo> collection)2{3 List<Jobs> importJobs4 foreach (var item in collection)5 {6 Jobs job = new Jobs();78 job.DateAdded = DateTime.UtcNow;9 job.Name = item.Name;10 job.Location = item.Location;1112 importJobs.Add(job);13 }1415 await dbContext.BulkInsert(importJobs);16}
If also works for updates, but rather than creating a new item, first retrieve it form the db and then at the end call BulkInsertOrUpdate with the list.
1await dbContext.BulkInsertOrUpdate(importJobs);
From my experience doing this took my import process that would run for over an hour down to something which would complete in a few minutes.