Revolutionizing Core Data with SwiftData-inspired Concurrent Operations
Welcome to CoreDataEvolution, a library aimed at modernizing Core Data by incorporating the elegance and safety of SwiftData-style concurrency. This library is designed to simplify and enhance Core Data’s handling of multithreading, drawing inspiration from SwiftData's @ModelActor
feature, enabling efficient, safe, and scalable operations.
Don't miss out on the latest updates and excellent articles about Swift, SwiftUI, Core Data, and SwiftData. Subscribe to Fatbobman's Swift Weekly and receive weekly insights and valuable content directly to your inbox.
SwiftData introduced modern concurrency features like @ModelActor
, making it easier to handle concurrent data access with safety guaranteed by the compiler. However, SwiftData's platform requirements and limited maturity in certain areas have deterred many developers from adopting it. CoreDataEvolution bridges the gap, bringing SwiftData’s advanced design into the Core Data world for developers who are still reliant on Core Data.
- Core Data Reform: Achieving Elegant Concurrency Operations like SwiftData
- Practical SwiftData: Building SwiftUI Applications with Modern Approaches
- Concurrent Programming in SwiftData
-
Custom Executors for Core Data Actors
Using Swift 5.9's newSerialExecutor
andExecutorJob
protocols, CoreDataEvolution provides custom executors that ensure all operations on managed objects are performed on the appropriate thread associated with their managed object context. -
@NSModelActor Macro
The@NSModelActor
macro simplifies Core Data concurrency, mirroring SwiftData’s@ModelActor
macro. It generates the necessary boilerplate code to manage a Core Data stack within an actor, ensuring safe and efficient access to managed objects. -
NSMainModelActor Macro
NSMainModelActor
will provide the same functionality asNSModelActor
, but it will be used to declare a class that runs on the main thread. -
Elegant Actor-based Concurrency
CoreDataEvolution allows you to create actors with custom executors tied to Core Data contexts, ensuring that all operations within the actor are executed serially on the context’s thread.
Here’s how you can use CoreDataEvolution to manage concurrent Core Data operations with an actor:
import CoreDataEvolution
@NSModelActor
actor DataHandler {
func updateItem(identifier: NSManagedObjectID, timestamp: Date) throws {
guard let item = self[identifier, as: Item.self] else {
throw MyError.objectNotExist
}
item.timestamp = timestamp
try modelContext.save()
}
}
In this example, the @NSModelActor
macro simplifies the setup, automatically creating the required executor and Core Data stack inside the actor. Developers can then focus on their business logic without worrying about concurrency pitfalls.
This approach allows you to safely integrate modern Swift concurrency mechanisms into your existing Core Data stack, enhancing performance and code clarity.
You can disable the automatic generation of the constructor by using disableGenerateInit
:
@NSModelActor(disableGenerateInit: true)
public actor DataHandler {
let viewName: String
func createNemItem(_ timestamp: Date = .now, showThread: Bool = false) throws -> NSManagedObjectID {
let item = Item(context: modelContext)
item.timestamp = timestamp
try modelContext.save()
return item.objectID
}
init(container: NSPersistentContainer, viewName: String) {
modelContainer = container
self.viewName = viewName
let context = container.newBackgroundContext()
context.name = viewName
modelExecutor = .init(context: context)
}
}
NSMainModelActor will provide the same functionality as NSModelActor, but it will be used to declare a class that runs on the main thread:
@MainActor
@NSMainModelActor
final class DataHandler {
func updateItem(identifier: NSManagedObjectID, timestamp: Date) throws {
guard let item = self[identifier, as: Item.self] else {
throw MyError.objectNotExist
}
item.timestamp = timestamp
try modelContext.save()
}
}
You can add CoreDataEvolution to your project using Swift Package Manager by adding the following dependency to your Package.swift
file:
dependencies: [
.package(url: "https://github.com/fatbobman/CoreDataEvolution.git", .upToNextMajor(from: "0.3.0"))
]
Then, import the module into your Swift files:
import CoreDataEvolution
- iOS 17.0 / macOS 14.0
- Swift 6.0
Note: Due to system limitations, custom executors and SerialExecutor
are only available on iOS 17/macOS 14 and later.
We welcome contributions! Whether you want to report issues, propose new features, or contribute to the code, feel free to open issues or pull requests on the GitHub repository.
CoreDataEvolution is available under the MIT license. See the LICENSE file for more information.
Special thanks to the Swift community for their continuous support and contributions.