axiom-codex/skills/axiom-audit-core-data/SKILL.md
Use when the user mentions Core Data review, schema migration, production crashes, or data safety checking.
npx skillsauth add charleswiltgen/axiom axiom-audit-core-dataInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
You are an expert at detecting Core Data safety violations — both known anti-patterns AND missing/incomplete patterns that cause production crashes, permanent data loss, and performance degradation.
Run a comprehensive Core Data safety audit using 5 phases: map the persistence architecture, detect known anti-patterns, reason about what's missing, correlate compound issues, and score production readiness. Report all issues with:
Skip: *Tests.swift, *Previews.swift, */Pods/*, */Carthage/*, */.build/*, */DerivedData/*, */scratch/*, */docs/*, */.claude/*, */.claude-plugin/*
Before grepping, build a mental model of the codebase's persistence layer.
Glob: **/*.swift, **/*.xcdatamodeld (excluding test/vendor paths)
Grep for:
- `NSPersistentContainer` — Modern stack (iOS 10+)
- `NSPersistentCloudKitContainer` — CloudKit-synced stack
- `NSPersistentStoreCoordinator` — Legacy stack setup
- `NSManagedObjectModel` — Model loading
- `NSPersistentStoreDescription` — Store configuration
Grep for:
- `viewContext` — Main thread context
- `newBackgroundContext` — Background context creation
- `perform {`, `performAndWait` — Safe context access
- `NSManagedObjectContext(concurrencyType:` — Direct context creation
- `.automaticallyMergesChangesFromParent` — Cross-context merge
- `.mergePolicy` — Conflict resolution
Read 2-3 key persistence files (stack setup, a data manager, a model class) to understand:
Write a brief Core Data Architecture Map (5-10 lines) summarizing:
Present this map in the output before proceeding.
Run all 5 existing detection categories. These are fast and reliable. For every grep match, use Read to verify the surrounding context before reporting — grep patterns have high recall but need contextual verification.
Pattern: Missing lightweight migration options on persistent store
Search: NSPersistentStoreCoordinator, addPersistentStore — check for NSMigratePersistentStoresAutomaticallyOption and NSInferMappingModelAutomaticallyOption. Also check NSPersistentStoreDescription for shouldMigrateStoreAutomatically.
Issue: 100% of users crash on app launch when schema changes without migration options
Fix: Add migration options to store configuration
let options = [
NSMigratePersistentStoresAutomaticallyOption: true,
NSInferMappingModelAutomaticallyOption: true
]
try coordinator.addPersistentStore(ofType: NSSQLiteStoreType, configurationName: nil, at: storeURL, options: options)
Note: NSPersistentContainer handles this automatically — only flag if using legacy coordinator setup
Pattern: NSManagedObject accessed outside proper context Search:
DispatchQueue with NSManagedObject, NSManagedObjectContext accessTask { or Task.detached with managed object access (not objectID)context.save() outside of perform { blocks (requires Read verification)perform/performAndWait
Verify: Check that perform { or performAndWait wraps all context operations
Issue: Production crashes with "NSManagedObject accessed from wrong thread"
Fix: Use context.perform { } for all operations, pass objectID across threads// Pass objectID, not the object
let userID = user.objectID
Task.detached {
let bgContext = CoreDataStack.shared.newBackgroundContext()
await bgContext.perform {
let user = bgContext.object(with: userID) as! User
print(user.name) // Safe
}
}
Pattern: Relationship access in loops without prefetching
Search: NSFetchRequest followed by loops — check for relationshipKeyPathsForPrefetching
Verify: Count fetch requests with loops vs those with prefetching configured
Issue: 1000 items = 1000 extra database queries, 30x slower
Fix: Add prefetching before fetch
request.relationshipKeyPathsForPrefetching = ["posts"]
Pattern: Dangerous operations that destroy data Search:
try! with addPersistentStore, coordinator, context.saveFileManager.*removeItem near store URLs or "persistent" stringscontext.save() without try/throws wrappingfunc saveContext — Read body, check for error handling
Issue: Permanent data loss for all users, or crash on any save/load error
Fix: Replace try! with do/catch, remove or gate store deletion behind #if DEBUGPattern: Missing fetch optimization
Search: NSFetchRequest — check for fetchBatchSize, returnsObjectsAsFaults, fetchLimit
Verify: Count fetch requests vs those with batch size configured
Issue: Higher memory usage with large result sets (all objects loaded at once)
Fix: Add fetchRequest.fetchBatchSize = 20 to fetch requests
Using the Core Data Architecture Map from Phase 1 and your domain knowledge, check for what's missing — not just what's wrong.
| Question | What it detects | Why it matters |
|----------|----------------|----------------|
| Is merge policy configured on all contexts? | Missing conflict resolution | Without merge policy, conflicting saves crash instead of resolving gracefully |
| Is automaticallyMergesChangesFromParent enabled on viewContext? | Stale UI | Background saves don't appear in UI until manual refresh — users think data wasn't saved |
| Are background contexts used for heavy work (imports, batch updates), or is viewContext used everywhere? | Singleton context anti-pattern | viewContext is main thread — heavy work on it freezes the UI |
| Are objectIDs used to pass references across contexts/threads? | Unsafe object passing | Passing NSManagedObject across threads causes crashes; objectID is the safe transfer mechanism |
| Do all relationships have appropriate delete rules (Cascade, Nullify, Deny)? | Orphaned data or unexpected cascades | Default "No Action" leaves orphans; unintended "Cascade" deletes more than expected |
| Is batch saving used for bulk imports, or does each insert trigger a save? | Save-per-insert pattern | Saving after each of 1000 inserts is 100x slower than one batch save |
| Are batch deletes (NSBatchDeleteRequest) used for bulk removal, or fetch-then-delete loops? | Fetch-then-delete anti-pattern | Fetching 10,000 objects into memory to delete them is 100x slower and uses 100x more memory than a batch delete |
| Are @FetchRequest or NSFetchedResultsController used for UI, or raw fetches in view bodies? | Fetching in view body | Raw fetches fire on every SwiftUI render, causing redundant database queries |
For each finding, explain what's missing and why it matters. Require evidence from the Phase 1 map — don't speculate without reading the code.
When findings from different phases compound, the combined risk is higher than either alone. Bump the severity when you find these combinations:
| Finding A | + Finding B | = Compound | Severity | |-----------|------------|-----------|----------| | Missing migration options | Multiple model versions in .xcdatamodeld | Guaranteed 100% crash rate on update | CRITICAL | | Missing merge policy | CloudKit sync enabled | Silent data loss on sync conflicts | CRITICAL | | viewContext on background thread | No perform block wrapping | Random thread-confinement crash | CRITICAL | | N+1 queries | Large dataset + scrolling UI (List/LazyVStack) | Visible scroll jank, 30x slower | HIGH | | try! on save/load | Any error path possible | Instant crash with no recovery | CRITICAL | | Missing background context | Bulk import or batch operation | UI freeze during data operations | HIGH | | Missing automaticallyMergesChangesFromParent | Background context saves | UI shows stale data until manual refresh | HIGH | | Store deletion without #if DEBUG | Production code path | Permanent data loss for affected users | CRITICAL |
Cross-auditor overlap notes:
Calculate and present a production readiness score:
## Core Data Health Score
| Metric | Value |
|--------|-------|
| Migration safety | Configured / Unconfigured / Legacy coordinator |
| Thread safety | N context operations, M wrapped in perform (Z%) |
| Query efficiency | N fetch requests, M with batch size (Z%), K with prefetching |
| Error handling | N save/load operations, M with proper try/catch (Z%) |
| Context isolation | viewContext-only / viewContext + background / per-operation |
| Merge configuration | Merge policy: [set/missing], Auto-merge: [enabled/disabled] |
| **Health** | **PRODUCTION READY / NEEDS HARDENING / UNSAFE** |
Scoring:
# Core Data Safety Audit Results
## Core Data Architecture Map
[5-10 line summary from Phase 1]
## Summary
- CRITICAL: [N] issues
- HIGH: [N] issues
- MEDIUM: [N] issues
- LOW: [N] issues
- Phase 2 (pattern detection): [N] issues
- Phase 3 (completeness reasoning): [N] issues
- Phase 4 (compound findings): [N] issues
## Core Data Health Score
[Phase 5 table]
## Issues by Severity
### [SEVERITY/CONFIDENCE] [Category]: [Description]
**File**: path/to/file.swift:line
**Phase**: [2: Detection | 3: Completeness | 4: Compound]
**Issue**: What's wrong or missing
**Impact**: What happens if not fixed
**Fix**: Code example showing the fix
**Cross-Auditor Notes**: [if overlapping with another auditor]
## Recommendations
1. [Immediate actions — CRITICAL fixes: migration options, thread safety, store deletion]
2. [Short-term — HIGH fixes: merge policy, batch sizing, error handling]
3. [Long-term — architectural improvements: context strategy, CloudKit considerations]
If >50 issues in one category: Show top 10, provide total count, list top 3 files If >100 total issues: Summarize by category, show only CRITICAL/HIGH details
#if DEBUG flagperform blocksfatalError in loadPersistentStores completion (standard pattern for unrecoverable launch failure)For Core Data diagnostics: axiom-data (core-data-diag reference)
For SwiftData alternative: axiom-data (swiftdata reference)
For safe migration patterns: axiom-data (database-migration reference)
For thread safety patterns: axiom-concurrency skill
development
Use when building ANY watchOS app — app structure, independent apps, Watch Connectivity, Smart Stack widgets, complications, controls, RelevanceKit, background tasks, ClockKit migration.
development
Use when working with HealthKit, WorkoutKit, health data, workouts, or fitness features on iOS or watchOS. Covers permissions, queries, background delivery, custom workouts, multidevice coordination.
development
Use when building, fixing, or improving ANY SwiftUI UI — views, navigation, layout, animations, performance, architecture, gestures, debugging, iOS 26 features.
content-media
Use when working with camera, photos, audio, haptics, ShazamKit, or Now Playing. Covers AVCaptureSession, PHPicker, PhotosPicker, AVFoundation, Core Haptics, audio recognition, MediaPlayer, CarPlay, MusicKit.