Exploring CoreData - From Data Model Creation to Managed Object Instances

Published on

For every developer who uses Core Data, building a data model using Xcode’s Core Data Model Editor, creating a container, loading a data model, and ultimately creating managed object instances through a managed object context are common operations. But have you ever wondered about the internal mechanisms behind all of this? How does Core Data assist us in completing these tasks behind the scenes? This article will delve into the inner workings of Core Data in constructing managed object instances from a data model. By the end of this article, you will have a deeper understanding of Core Data’s workflow, allowing you to be more proficient in your development endeavors.

Preface

Recently, I have been writing an article about concurrent programming in SwiftData. The original plan was to discuss how SwiftData creates PersistentModel instances based on model declarations in the first part. I initially intended to explain it in a few paragraphs, but while writing, I realized that it couldn’t be easily expressed and needed to be a separate piece. As I started writing this article, I also realized that another article is needed to specifically explain the implementation process of the Core Data version. Thus, this article was born by chance.

In this article, we will not delve into every detail of building data models to creating managed object instances. We will primarily focus on two aspects: how Core Data converts a model file into a ManagedObjectModel, and how it extracts information from it to create managed object instances.

This text will discuss the data model file provided by the Core Data project template created in Xcode.

Building Core Data Data Model File with Xcode Model Editor

The model editor in Xcode provides us with a visual interface to define the data model of a Core Data application, including entities, properties, and other information. Using the model editor allows us to construct the data model in a more intuitive way.

When creating a new project with Core Data included, Xcode will automatically create a data model file called ProjectName.xcdatamodeld in the project. Alternatively, we can manually create a Core Data data model file in the project with a file extension of .xcdatamodeld.

image-20230918092422868.png

image-20230918092749973.png

Xcode stores all the information created by the developer in the model editor in xcdatamodeld.

Specifically, xcdatamodeld is a directory commonly referred to as a “Core Data Model Bundle”. It is a special bundle used to store and manage the data model information for Core Data. It contains one or more data model files (.xcdatamodel) as well as other information related to the data model. Xcode creates a separate VersionName.xcdatamodel bundle for each model version within the xcdatamodeld directory.

Now, open the content file in the xcdatamodel with a text editor, and you will see that all the model information of the current version is saved in XML format.

XML
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<model type="com.apple.IDECoreDataModeler.DataModel" documentVersion="1.0" lastSavedToolsVersion="1" systemVersion="11A491" minimumToolsVersion="Automatic" sourceLanguage="Swift" usedWithCloudKit="false" userDefinedModelVersionIdentifier="">
    <entity name="Item" representedClassName="Item" syncable="YES" codeGenerationType="class">
        <attribute name="timestamp" optional="YES" attributeType="Date" usesScalarValueType="NO"/>
    </entity>
    <elements>
        <element name="Item" positionX="-63" positionY="-18" width="128" height="44"/>
    </elements>
</model>

In this case, each entity element corresponds to an Entity, which contains numerous pieces of information such as entity name, corresponding subclass name, attributes, relationships, custom indices, and more. If we create a new Configuration or Fetch Request in the model editor, we can also find the corresponding information in the XML file. In Xcode 14, the visual relationship view has been removed. This relationship view played an important role in the model editor, allowing for a visual representation of the relationships between entities. With the removal of the visual relationship view, the information in the elements element has essentially become ineffective.

Xcode, when compiling a project, will include the .xcdatamodel directory as a momd bundle in the application’s resources. The .xcdatamodel bundle will be compiled into a binary file with the mom extension. This reduces the space occupied and improves loading speed. This is also why we need to set the extension to momd when loading the model file using code.

Developers should understand that the model file created through Xcode’s model editor is just a structured representation of the model, not a programmatic representation.

Generate the declaration of NSManagedObject subclass corresponding to the entity

In most cases, developers will declare a corresponding NSManagedObject subclass for the Entity. When Codegen is set to Class Definition or Category/Extension, Xcode will implicitly assist us in completing this task.

image-20230918143644990.png

When Codegen is set to Class Definition, Xcode will generate a separate NSManagedObject subclass that includes the definition of entity properties and methods. For example:

Swift
@objc(Item)
public class Item: NSManagedObject {}

extension Item {
    @nonobjc public class func fetchRequest() -> NSFetchRequest<Item> {
        return NSFetchRequest<Item>(entityName: "Item")
    }

    @NSManaged public var timestamp: Date?
}

extension Item : Identifiable {}

When Codegen is set to Category/Extension, Xcode will generate an extension that adds entity properties and methods to the default implementation of NSManagedObject. This helps avoid modifying the automatically generated code and maintains the maintainability of the code.

@NSManaged is an attribute modifier used to mark a property that is managed by Core Data. It informs the compiler that this property will have its relevant access methods automatically generated by Core Data and will be dynamically associated with the property on the managed object at runtime.

Developers can also choose to manually create these codes or use Xcode to generate them explicitly. Manually creating codes can express the attribute types more accurately and provide higher flexibility. Using Xcode to generate codes can save the workload of manual coding, especially when there are many attributes or complex model structures. Regardless of the chosen method, generating a subclass declaration that conforms to NSManagedObject allows developers to safely and conveniently access the managed properties of the managed object. By overriding certain methods of the subclass (e.g., willSave), certain operations can be tailored to specific entities.

Swift
extension Item {
    public override func willSave() {
        super.willSave()
        // do something
    }
}

Although it is possible to gain the aforementioned benefits, it is not necessary to create a corresponding subclass of NSManagedObject for entity declaration. This is because Core Data also provides a lightweight way to access and manipulate managed objects using the NSManagedObject itself for property access and manipulation.

Swift
// item:Item
let timestamp = item.timestamp
// object is a NSManagedObject instance create by Item Entity description
let timestamp = object.value(forKey: "timestamp") // trigger KVO
let timestamp = object..primitiveValue(forKey: "timestamp") // not trigger KVO

In the example above, item.timestamp is achieved by declaring a corresponding subclass of NSManagedObject for the entity Item, called Item. On the other hand, object.value(forKey:) and object.primitiveValue(forKey:) are methods to access properties through the NSManagedObject object itself. It is important to note that the value(forKey:) method triggers Key-Value Observing (KVO), while the primitiveValue(forKey:) method does not trigger KVO.

To some extent, we can consider @NSManaged as a mechanism similar to Swift’s computed properties. Using the value(forKey:) and setValue(_:forKey:) methods, we can read and set the underlying value of a managed object. This allows us to perform custom logic operations on properties when needed, such as data format conversion, data validation, and so on.

Load Data Model, Create Container

Ever since Core Data introduced NSPersistentContainer, developers rarely need to explicitly read the data model file and create the data model in their code (NSManagedObjectModel), unless there are specific circumstances.

Swift
let container = NSPersistentContainer(name: "ModelEditorDemo")

However, understanding the work behind creating a container in Core Data is still very helpful for later understanding the process of creating managed object instances.

Swift
// Load the data model file and create NSManagedObjectModel
guard let url = Bundle.main.url(forResource: "ModelEditorDemo", withExtension: "momd"),
      let dataModel = NSManagedObjectModel(contentsOf: url) else {
     fatalError("Failed to load the data model file")
}

// Create the persistent store coordinator
let coordinator = NSPersistentStoreCoordinator(managedObjectModel: dataModel)

// Get the configuration from the data model
let configuration = dataModel.configurations.first!

// Create the URL for the persistent store
let storeURL = URL.applicationDirectory.appending(path: "store.sqlite")

// Create or load the persistent store with the specified configuration
guard let store = try? coordinator.addPersistentStore(type: .sqlite, configuration: configuration,at: storeURL,options: nil) else {
    fatalError("Failed to create persistent store: \(error)")
}

// Create a main queue NSMangedObjectContext
let viewContext = NSManagedObjectContext(.mainQueue)
// Link context to coordinator
viewContext.persistentStoreCoordinator = coordinator

The general process is as follows:

  1. Obtain the URL of the data model file (momd).
  2. Create an NSManagedObjectModel instance using the URL.
  3. Create an NSPersistentStoreCoordinator instance using the NSManagedObjectModel instance.
  4. Add a persistent store to the NSPersistentStoreCoordinator instance.
  5. Create a main thread managed object context.
  6. Associate the context with the NSPersistentStoreCoordinator instance.

In this case, when using the data model file URL to create an NSManagedObjectModel instance, Core Data first converts the descriptions in the model file into programmatic expressions for entities, and then uses these programmatic expressions to create the NSManagedObjectModel instance. This conversion process allows us to create and manipulate data models programmatically, not just limited to using the visual editor.

Describe entities programmatically and create data model instances

In addition to using the data model editor for visual operations, Core Data provides a way to express entities and create data models programmatically.

The following code demonstrates the process of describing the Item entity and creating a data model in a programmatic way.

Swift
func createModel() -> NSManagedObjectModel {
    let itemEntityDescription = NSEntityDescription()
    // Entity Name
    itemEntityDescription.name = "Item"
    // NSManagedObject SubClass Name
    itemEntityDescription.managedObjectClassName = "Item"
    // Descriptor timestamp attribute
    let timestampAttribute = NSAttributeDescription()
    // Attribute Name
    timestampAttribute.name = "timestamp"
    // Is Optional
    timestampAttribute.isOptional = true
    // Attribute Type
    if #available(macOS 12.0, iOS 15.0, tvOS 15.0, watchOS 8.0, *) {
        timestampAttribute.type = .date
    } else {
        timestampAttribute.attributeType = .dateAttributeType
    }
    // Add timestamp to Item
    itemEntityDescription.properties.append(timestampAttribute)
    // Create a empty NSManagedObject
    let model = NSManagedObjectModel()
    // Add Item Entity into model
    model.entities = [itemEntityDescription]
    return model
}

The above code corresponds almost one-to-one with the operations we do in the model editor. However, when there are numerous properties or complex relationships, visual operations are more efficient and convenient. Through visual operations, we can intuitively add, edit, and delete entities, properties, and relationships in the graphical interface without the need to manually write a large amount of code. This makes the creation and maintenance of data models easier and faster.

Now we can use this code snippet to replace the previous operation of creating NSManagedObjectModel through a data model file.

Swift
// Create data model by programming way
let dataModel = createModel()

// Create persistent store coordinator by dataModel
let coordinator = NSPersistentStoreCoordinator(managedObjectModel: dataModel)

Even though visual editing is more efficient, programmatic expression provides developers with a broader space for describing data models, allowing custom description methods to be mapped to programmable expressions that Core Data can accept. This flexibility enables developers to better meet specific business needs. Additionally, the programming approach can provide more type safety and compile-time checks, reducing the possibility of errors occurring at runtime.

Creating Managed Object Instances

Core Data is an object graph management framework that allows us to work with persistent data in an object-oriented manner. We build data models to define the structure of our data and perform operations on it using managed object instances.

There are two common ways to obtain managed object instances:

  • By setting predicates and using NSFetchRequest, Core Data returns managed objects that match the specified conditions.
  • By directly calling the constructor of the NSManagedObject subclass that corresponds to the entity, we can create managed object instances.

Developers often use the following approach to create managed object instances:

Swift
let item = Item(context: viewContext)
item.timestamp = .now
try? viewContext.save()

However, init(context:) requires us to first create a managed object context (NSManagedObjectContext). In fact, in Core Data, we can completely create managed object instances without a context.

Swift
let item = Item(entity: Item.entity(), insertInto: nil)
item.timestamp = .now
viewContext.insert(item)
try? viewContext.save()

In fact, the init(entity:, insertInto:) constructor is the designated initializer of NSManagedObject, while init(context:) is its convenience initializer. The key to creating a managed object instance is not whether there is a managed object context, but rather informing NSManagedObject which EntityDescription the instance corresponds to.

It is important to note that when we use Item.entity() to obtain the EntityDescription corresponding to Item, we need to ensure that the NSManagedObjectModel has been loaded by NSPersistentStoreCoordinator.

Swift
let coordinator = NSPersistentStoreCoordinator(managedObjectModel: dataModel)

In Core Data, after NSPersistentStoreCoordinator is created, the data model is saved in a location accessible to internal elements for retrieval. The Item.entity() method retrieves the EntityDescription corresponding to Item. If we did not use a data model containing Item when creating NSPersistentStoreCoordinator, or if we did not create NSPersistentStoreCoordinator at all, calling Item.entity() will result in Core Data throwing the following error:

Swift
CoreData: error: No NSEntityDescriptions in any model claim the NSManagedObject subclass 'Item' so +entity is confused.  Have you loaded your NSManagedObjectModel yet ?

This does not mean that we do not have other ways to bypass the limitations of NSPersistentStoreCoordinator.

Swift
guard let url = Bundle.main.url(forResource: "ModelEditorDemo", withExtension: "momd"),
      let dataModel = NSManagedObjectModel(contentsOf: url) else {
     fatalError("Failed to load the data model file")
}

let entityDescription = dataModel.entitiesByName["Item"]!
let item = Item(entity: entityDescription, insertInto: nil)

By directly obtaining the corresponding EntityDescription from NSManagedObjectModel, developers can have the ability to create managed object instances with only an instance of NSManagedObjectModel. This is particularly useful in certain situations where only manipulating the data model without dealing with the managed object context is needed.

Read the article ”How to Preview a SwiftUI View with Core Data Elements in Xcode” to see how this method is applied in SwiftUI previews.

As mentioned earlier, developers do not necessarily have to create instances of managed object subclasses. By using the correct EntityDescription, we can create NSManagedObject instances that can achieve the same effect in many scenarios.

Swift
let item = NSManagedObject(entity: Item.entity(), insertInto: nil)
item.setValue(Date.now, forKey: "timestamp")
viewContext.insert(item)
try? viewContext.save()

Finally

In this article, we discuss several different methods for building data models and creating managed object instances in Core Data, some of which may not be common. Some readers may find these methods confusing, but even without understanding them, it will not affect our ability to proficiently use Core Data. However, the purpose of this article is to introduce these uncommon methods to readers, as in the following articles, we will explore “How SwiftData creates PersistentModel instances based on model declarations”. At that time, we will see how the SwiftData development team utilizes the content introduced in this article and Swift’s new features to build a persistent framework that is in line with the new era.

Get weekly handpicked updates on Swift and SwiftUI!