Skip to content

sentryco/SDUtil

Repository files navigation

Tests codebeat badge

SDUtil

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.

Features

  • 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, and getIndex. 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, and fetch.

  • 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.

Examples

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)")
}

Installation

.package(url: "https://github.com/sentryco/SDUtil", branch: "main")

Imporovments:

    1. 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.
    1. 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.
    1. 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.
    1. 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.
    1. Refactoring and Code Organization: Make variables and function names more consistent
    1. 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.
    1. 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.

Todo:

  • Add MetaDataError

About

Makes SwiftData easier to work with

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages