
Swift Actor Persistence
Implement thread-safe Swift persistence with actors plus file-backed JSON cache for offline-first iOS or macOS features.
Overview
swift-actor-persistence is an agent skill for the Build phase that implements thread-safe Swift persistence with actors and file-backed JSON storage.
Install
npx skills add https://github.com/affaan-m/everything-claude-code --skill swift-actor-persistenceWhat is this skill?
- Generic actor LocalRepository for Codable & Identifiable models with String IDs
- In-memory cache synchronized to a JSON file on save and delete
- Compiler-enforced isolation—no manual locks or DispatchQueue juggling
- Synchronous initial load from disk before actor isolation activates
- Offline-first pattern suitable for indie mobile apps and small SaaS clients
Adoption & trust: 4.5k installs on skills.sh; 210k GitHub stars; 3/3 security scanners passed (skills.sh audits).
What problem does it solve?
Your Swift app shares mutable state across tasks and you are fighting crashes or corrupt JSON from unsynchronized reads and writes.
Who is it for?
Solo iOS or macOS builders adding local JSON persistence or offline caches with Swift Concurrency already in the project.
Skip if: Greenfield Android or React Native apps, large relational schemas needing Core Data migrations, or server-side Swift on Linux without the same client patterns.
When should I use this skill?
Building a data persistence layer in Swift 5.5+, needing thread-safe shared mutable state, eliminating locks/DispatchQueues, or building offline-first apps with local storage.
What do I get? / Deliverables
You get an actor-isolated repository pattern with cache-plus-file persistence that eliminates data races without manual synchronization.
- Actor-based LocalRepository implementation
- File-backed JSON persist/load helpers
- Thread-safe public async API for save, delete, and fetch
Recommended Skills
Journey fit
Local repository and storage layers are built during implementation, not during idea or launch distribution work. Persistence, caching, and serialized mutation belong with backend/data access patterns even in a SwiftUI app.
How it compares
Swift-concurrency persistence template—not a cross-platform SQLite ORM skill or a Core Data code generator.
Common Questions / FAQ
Who is swift-actor-persistence for?
Indie Swift developers building offline-first or local-cache features who want actor-isolated APIs instead of lock-based repositories.
When should I use swift-actor-persistence?
During Build backend/mobile data layers when implementing save/load for Codable models, replacing unsafe shared singletons, or designing file-backed caches before TestFlight.
Is swift-actor-persistence safe to install?
It is reference code for your repo; review the Security Audits panel on this page and audit any filesystem paths and data you persist on device.
SKILL.md
READMESKILL.md - Swift Actor Persistence
# Swift Actors for Thread-Safe Persistence Patterns for building thread-safe data persistence layers using Swift actors. Combines in-memory caching with file-backed storage, leveraging the actor model to eliminate data races at compile time. ## When to Activate - Building a data persistence layer in Swift 5.5+ - Need thread-safe access to shared mutable state - Want to eliminate manual synchronization (locks, DispatchQueues) - Building offline-first apps with local storage ## Core Pattern ### Actor-Based Repository The actor model guarantees serialized access — no data races, enforced by the compiler. ```swift public actor LocalRepository<T: Codable & Identifiable> where T.ID == String { private var cache: [String: T] = [:] private let fileURL: URL public init(directory: URL = .documentsDirectory, filename: String = "data.json") { self.fileURL = directory.appendingPathComponent(filename) // Synchronous load during init (actor isolation not yet active) self.cache = Self.loadSynchronously(from: fileURL) } // MARK: - Public API public func save(_ item: T) throws { cache[item.id] = item try persistToFile() } public func delete(_ id: String) throws { cache[id] = nil try persistToFile() } public func find(by id: String) -> T? { cache[id] } public func loadAll() -> [T] { Array(cache.values) } // MARK: - Private private func persistToFile() throws { let data = try JSONEncoder().encode(Array(cache.values)) try data.write(to: fileURL, options: .atomic) } private static func loadSynchronously(from url: URL) -> [String: T] { guard let data = try? Data(contentsOf: url), let items = try? JSONDecoder().decode([T].self, from: data) else { return [:] } return Dictionary(uniqueKeysWithValues: items.map { ($0.id, $0) }) } } ``` ### Usage All calls are automatically async due to actor isolation: ```swift let repository = LocalRepository<Question>() // Read — fast O(1) lookup from in-memory cache let question = await repository.find(by: "q-001") let allQuestions = await repository.loadAll() // Write — updates cache and persists to file atomically try await repository.save(newQuestion) try await repository.delete("q-001") ``` ### Combining with @Observable ViewModel ```swift @Observable final class QuestionListViewModel { private(set) var questions: [Question] = [] private let repository: LocalRepository<Question> init(repository: LocalRepository<Question> = LocalRepository()) { self.repository = repository } func load() async { questions = await repository.loadAll() } func add(_ question: Question) async throws { try await repository.save(question) questions = await repository.loadAll() } } ``` ## Key Design Decisions | Decision | Rationale | |----------|-----------| | Actor (not class + lock) | Compiler-enforced thread safety, no manual synchronization | | In-memory cache + file persistence | Fast reads from cache, durable writes to disk | | Synchronous init loading | Avoids async initialization complexity | | Dictionary keyed by ID | O(1) lookups by identifier | | Generic over `Codable & Identifiable` | Reusable across any model type | | Atomic file writes (`.atomic`) | Prevents partial writes on crash | ## Best Practices - **Use `Sendable` types** for all data crossing actor boundaries - **Keep the actor's public API minimal** — only expose domain operations, not persistence details - **Use `.atomic` writes** to prevent data corruption if the app crashes mid-write - **Load synchronously in `init`** — async initializers add complexity with minimal benefit fo