Makes SwiftData easier to work with
SDUtil is a library that enhances the functionality of SwiftData by providing easy access to metadata and reading the insert order of data entities.
-
Easy Access to Metadata: Read and write metadata to the database with simple syntax:
db.metaData["version"] = "1.0" let version = db.metaData["version"]
-
Work with Insert Index Easily: Iterate through the database based on insert order using functions like
getLast
,getFirst
, andgetIndex
. This removes the need for manual record-keeping. -
Range Support: Fetch data based on a range of indexes using
fetch(range: 0..<10)
. -
Background Contexts: Perform asynchronous operations on the database using
getBackgroundContext
. -
Utility Functions: Reset, remove, and assert that databases exist and are working properly.
-
Accessible CRUD Operations: Perform standard database operations such as
insert
,delete
,update
, andfetch
. -
Merge Contexts and Handle Conflicts: Merge changes from background contexts and handle potential conflicts gracefully.
-
Metadata Management: Manage metadata with features like versioning, audit trails, and synchronization mechanisms.
-
Error Handling: Improved error handling by throwing and managing errors effectively.
Inserting Data
To insert a new item into the database, you can use the insert
method from the ModelContext
extension. This method allows you to specify the item and whether the changes should be saved immediately.
let context = try Database().getContext()
let newItem = User(userName: "John", password: "abc123")
try? context.insertData(item: newItem, shouldSave: true)
Fetching Data
To fetch data using a FetchDescriptor
, you can use the readData
method from the ModelContext
extension. This method allows you to specify the type of data and any constraints or filters.
let context = try Database().getContext()
let predicate = #Predicate<User> { user in
user.userName == "John"
}
let descriptor = FetchDescriptor<User>(predicate: predicate)
let results = try context.readData(descriptor: descriptor)
print(results.first?.password)
Deleting Data
let context = try Database().getContext()
let userToDelete = try context.readFirst(descriptor: FetchDescriptor<User>())
if let user = userToDelete {
try context.delete(item: user, shouldSave: true)
}
Updating Data
let context = try Database().getContext()
if let user = try context.readFirst(descriptor: FetchDescriptor<User>()) {
user.password = "newPassword123"
try context.saveIfChanged()
}
Working with Background Contexts
let container = try Database().getContainer()
container.getBackgroundContext { context in
guard let context = context else { return }
let newItem = User(userName: "BackgroundUser", password: "backgroundPass")
context.insert(newItem)
do {
try context.saveIfChanged()
} catch {
print("Error saving in background context: \(error)")
}
}
Handling Metadata
let db = Database()
db.metaData["appVersion"] = "2.0"
if let appVersion = db.metaData["appVersion"] {
print("Current app version: \(appVersion)")
}
.package(url: "https://github.com/sentryco/SDUtil", branch: "main")
-
- Error Handling: the error handling could be improved by throwing and managing errors more effectively rather than just returning empty dictionaries or ignoring the errors.
-
- Metadata Management: The metadata functionality could be expanded to include more robust features such as versioning, audit trails, and synchronization mechanisms, especially if the metadata is critical for the application's functionality.
-
- Testing and Coverage: Increase the unit test coverage for critical components, especially those handling database operations and metadata management. This could help ensure stability and catch potential issues early. The TODO item in the README.md (startLine: 71, endLine: 74) about adding delete operations to unit tests is a good start.
-
- Documentation and Examples: Ensuring that all public APIs are well-documented and include examples can significantly improve the developer experience and ease of use. Consider adding more complex examples that showcase the full capabilities of the library, such as handling concurrent database operations or integrating with other systems.
-
- Refactoring and Code Organization: Make variables and function names more consistent
-
- Performance Optimization: Review and optimize database interactions, especially those that might be impacted by large datasets or complex queries. Profiling and benchmarking could be used to identify bottlenecks.
-
- Security Enhancements: Ensure that all data handling practices meet security best practices, particularly in how metadata is managed and accessed. This includes securing any sensitive information that might be stored in the metadata.
- Add MetaDataError