@devo-bmad-custom/agent-orchestration 1.0.2 → 1.0.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/lib/installer.js +33 -0
- package/package.json +1 -1
- package/src/.agents/skills/audit-website/README.md +20 -20
- package/src/.agents/skills/audit-website/SKILL.md +470 -470
- package/src/.agents/skills/audit-website/agents/openai.yaml +6 -6
- package/src/.agents/skills/audit-website/assets/icon-small.svg +41 -41
- package/src/.agents/skills/audit-website/references/OUTPUT-FORMAT.md +250 -250
- package/src/.agents/skills/clean-code-standards/SKILL.md +104 -104
- package/src/.agents/skills/excalidraw-dark-standard/SKILL.md +281 -281
- package/src/.agents/skills/frontend-responsive-design-standards/SKILL.md +434 -434
- package/src/.agents/skills/java-fundamentals/SKILL.md +116 -116
- package/src/.agents/skills/java-performance/SKILL.md +119 -119
- package/src/.agents/skills/next-best-practices/SKILL.md +153 -153
- package/src/.agents/skills/next-best-practices/async-patterns.md +87 -87
- package/src/.agents/skills/next-best-practices/bundling.md +180 -180
- package/src/.agents/skills/next-best-practices/data-patterns.md +297 -297
- package/src/.agents/skills/next-best-practices/debug-tricks.md +105 -105
- package/src/.agents/skills/next-best-practices/directives.md +73 -73
- package/src/.agents/skills/next-best-practices/error-handling.md +227 -227
- package/src/.agents/skills/next-best-practices/file-conventions.md +140 -140
- package/src/.agents/skills/next-best-practices/font.md +245 -245
- package/src/.agents/skills/next-best-practices/functions.md +108 -108
- package/src/.agents/skills/next-best-practices/hydration-error.md +91 -91
- package/src/.agents/skills/next-best-practices/image.md +173 -173
- package/src/.agents/skills/next-best-practices/metadata.md +301 -301
- package/src/.agents/skills/next-best-practices/parallel-routes.md +287 -287
- package/src/.agents/skills/next-best-practices/route-handlers.md +146 -146
- package/src/.agents/skills/next-best-practices/rsc-boundaries.md +159 -159
- package/src/.agents/skills/next-best-practices/runtime-selection.md +39 -39
- package/src/.agents/skills/next-best-practices/scripts.md +141 -141
- package/src/.agents/skills/next-best-practices/self-hosting.md +371 -371
- package/src/.agents/skills/next-best-practices/suspense-boundaries.md +67 -67
- package/src/.agents/skills/nextjs-app-router-patterns/SKILL.md +537 -537
- package/src/.agents/skills/postgresql-optimization/SKILL.md +404 -404
- package/src/.agents/skills/python-backend/SKILL.md +153 -153
- package/src/.agents/skills/python-fundamentals/SKILL.md +234 -234
- package/src/.agents/skills/python-performance/SKILL.md +404 -404
- package/src/.agents/skills/react-expert/SKILL.md +335 -335
- package/src/.agents/skills/redis-best-practices/SKILL.md +438 -438
- package/src/.agents/skills/security-best-practices/SKILL.md +288 -288
- package/src/.agents/skills/security-review/LICENSE +22 -22
- package/src/.agents/skills/security-review/SKILL.md +312 -312
- package/src/.agents/skills/security-review/infrastructure/docker.md +432 -432
- package/src/.agents/skills/security-review/languages/javascript.md +388 -388
- package/src/.agents/skills/security-review/languages/python.md +363 -363
- package/src/.agents/skills/security-review/references/api-security.md +519 -519
- package/src/.agents/skills/security-review/references/authentication.md +353 -353
- package/src/.agents/skills/security-review/references/authorization.md +372 -372
- package/src/.agents/skills/security-review/references/business-logic.md +443 -443
- package/src/.agents/skills/security-review/references/cryptography.md +329 -329
- package/src/.agents/skills/security-review/references/csrf.md +398 -398
- package/src/.agents/skills/security-review/references/data-protection.md +378 -378
- package/src/.agents/skills/security-review/references/deserialization.md +410 -410
- package/src/.agents/skills/security-review/references/error-handling.md +436 -436
- package/src/.agents/skills/security-review/references/file-security.md +457 -457
- package/src/.agents/skills/security-review/references/injection.md +259 -259
- package/src/.agents/skills/security-review/references/logging.md +433 -433
- package/src/.agents/skills/security-review/references/misconfiguration.md +435 -435
- package/src/.agents/skills/security-review/references/modern-threats.md +475 -475
- package/src/.agents/skills/security-review/references/ssrf.md +415 -415
- package/src/.agents/skills/security-review/references/supply-chain.md +405 -405
- package/src/.agents/skills/security-review/references/xss.md +336 -336
- package/src/.agents/skills/subagent-driven-development/SKILL.md +275 -275
- package/src/.agents/skills/subagent-driven-development/code-quality-reviewer-prompt.md +26 -26
- package/src/.agents/skills/subagent-driven-development/implementer-prompt.md +113 -113
- package/src/.agents/skills/subagent-driven-development/spec-reviewer-prompt.md +61 -61
- package/src/.agents/skills/systematic-debugging/CREATION-LOG.md +119 -119
- package/src/.agents/skills/systematic-debugging/SKILL.md +296 -296
- package/src/.agents/skills/systematic-debugging/condition-based-waiting-example.ts +158 -158
- package/src/.agents/skills/systematic-debugging/condition-based-waiting.md +115 -115
- package/src/.agents/skills/systematic-debugging/defense-in-depth.md +122 -122
- package/src/.agents/skills/systematic-debugging/root-cause-tracing.md +169 -169
- package/src/.agents/skills/systematic-debugging/test-academic.md +14 -14
- package/src/.agents/skills/systematic-debugging/test-pressure-1.md +58 -58
- package/src/.agents/skills/systematic-debugging/test-pressure-2.md +68 -68
- package/src/.agents/skills/systematic-debugging/test-pressure-3.md +69 -69
- package/src/.agents/skills/typescript-best-practices/SKILL.md +373 -373
- package/src/.agents/skills/ui-ux-pro-custom/SKILL.md +348 -348
- package/src/.agents/skills/ui-ux-pro-custom/data/charts.csv +26 -26
- package/src/.agents/skills/ui-ux-pro-custom/data/colors.csv +97 -97
- package/src/.agents/skills/ui-ux-pro-custom/data/icons.csv +101 -101
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/SKILL.md +106 -106
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/accessibility.md +475 -475
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/animation.md +466 -466
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/composition-locals.md +231 -231
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/deprecated-patterns.md +323 -323
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/lists-scrolling.md +400 -400
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/modifiers.md +331 -331
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/navigation.md +416 -416
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/performance.md +446 -446
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/side-effects.md +516 -516
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/source-code/foundation-source.md +13327 -13327
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/source-code/material3-source.md +19097 -19097
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/source-code/navigation-source.md +2947 -2947
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/source-code/runtime-source.md +11316 -11316
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/source-code/ui-source.md +7896 -7896
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/state-management.md +377 -377
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/styles-experimental.md +470 -470
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/theming-material3.md +349 -349
- package/src/.agents/skills/ui-ux-pro-custom/data/jetpack-compose-expert-skill/references/view-composition.md +595 -595
- package/src/.agents/skills/ui-ux-pro-custom/data/landing.csv +31 -31
- package/src/.agents/skills/ui-ux-pro-custom/data/mobile-ui-layout.md +654 -654
- package/src/.agents/skills/ui-ux-pro-custom/data/products.csv +96 -96
- package/src/.agents/skills/ui-ux-pro-custom/data/react-performance.csv +45 -45
- package/src/.agents/skills/ui-ux-pro-custom/data/stacks/astro.csv +54 -54
- package/src/.agents/skills/ui-ux-pro-custom/data/stacks/flutter.csv +53 -53
- package/src/.agents/skills/ui-ux-pro-custom/data/stacks/html-tailwind.csv +56 -56
- package/src/.agents/skills/ui-ux-pro-custom/data/stacks/jetpack-compose.csv +53 -53
- package/src/.agents/skills/ui-ux-pro-custom/data/stacks/nextjs.csv +53 -53
- package/src/.agents/skills/ui-ux-pro-custom/data/stacks/nuxt-ui.csv +51 -51
- package/src/.agents/skills/ui-ux-pro-custom/data/stacks/nuxtjs.csv +59 -59
- package/src/.agents/skills/ui-ux-pro-custom/data/stacks/react-native.csv +56 -56
- package/src/.agents/skills/ui-ux-pro-custom/data/stacks/react.csv +54 -54
- package/src/.agents/skills/ui-ux-pro-custom/data/stacks/shadcn.csv +61 -61
- package/src/.agents/skills/ui-ux-pro-custom/data/stacks/svelte.csv +54 -54
- package/src/.agents/skills/ui-ux-pro-custom/data/stacks/swiftui.csv +51 -51
- package/src/.agents/skills/ui-ux-pro-custom/data/stacks/vue.csv +50 -50
- package/src/.agents/skills/ui-ux-pro-custom/data/styles.csv +68 -68
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/alarmkit/SKILL.md +438 -438
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/alarmkit/references/alarmkit-patterns.md +584 -584
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/app-clips/SKILL.md +436 -436
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/app-intents/SKILL.md +489 -489
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/app-intents/references/appintents-advanced.md +1076 -1076
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/app-store-review/SKILL.md +340 -340
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/app-store-review/references/privacy-manifest.md +90 -90
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/app-store-review/references/review-checklists.md +106 -106
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/apple-on-device-ai/SKILL.md +500 -500
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/apple-on-device-ai/references/coreml-conversion.md +425 -425
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/apple-on-device-ai/references/coreml-optimization.md +344 -344
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/apple-on-device-ai/references/foundation-models.md +508 -508
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/apple-on-device-ai/references/mlx-swift.md +285 -285
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/authentication/SKILL.md +496 -496
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/authentication/references/keychain-biometric.md +211 -211
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/background-processing/SKILL.md +499 -499
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/background-processing/references/background-task-patterns.md +390 -390
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/callkit-voip/SKILL.md +461 -461
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/callkit-voip/references/callkit-patterns.md +425 -425
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/cloudkit-sync/SKILL.md +492 -492
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/cloudkit-sync/references/cloudkit-patterns.md +461 -461
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/codable-patterns/SKILL.md +467 -467
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/contacts-framework/SKILL.md +425 -425
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/contacts-framework/references/contacts-patterns.md +409 -409
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/core-bluetooth/SKILL.md +491 -491
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/core-bluetooth/references/ble-patterns.md +435 -435
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/core-motion/SKILL.md +388 -388
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/core-motion/references/motion-patterns.md +405 -405
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/core-nfc/SKILL.md +495 -495
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/core-nfc/references/nfc-patterns.md +420 -420
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/coreml/SKILL.md +459 -459
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/coreml/references/coreml-swift-integration.md +765 -765
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/debugging-instruments/SKILL.md +422 -422
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/debugging-instruments/references/instruments-guide.md +387 -387
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/debugging-instruments/references/lldb-patterns.md +298 -298
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/device-integrity/SKILL.md +477 -477
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/energykit/SKILL.md +460 -460
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/energykit/references/energykit-patterns.md +541 -541
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/eventkit-calendar/SKILL.md +483 -483
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/eventkit-calendar/references/eventkit-patterns.md +326 -326
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/healthkit/SKILL.md +498 -498
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/healthkit/references/healthkit-patterns.md +602 -602
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/homekit-matter/SKILL.md +496 -496
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/homekit-matter/references/matter-commissioning.md +455 -455
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/ios-accessibility/SKILL.md +301 -301
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/ios-accessibility/references/a11y-patterns.md +140 -140
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/ios-localization/SKILL.md +418 -418
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/ios-localization/references/formatstyle-locale.md +627 -627
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/ios-localization/references/string-catalogs.md +462 -462
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/ios-networking/SKILL.md +441 -441
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/ios-networking/references/background-websocket.md +862 -862
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/ios-networking/references/lightweight-clients.md +93 -93
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/ios-networking/references/network-framework.md +563 -563
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/ios-networking/references/urlsession-patterns.md +1116 -1116
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/ios-security/SKILL.md +496 -496
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/ios-security/references/app-review-guidelines.md +174 -174
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/ios-security/references/cryptokit-advanced.md +296 -296
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/ios-security/references/file-storage-patterns.md +354 -354
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/ios-security/references/privacy-manifest.md +117 -117
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/live-activities/SKILL.md +500 -500
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/live-activities/references/live-activity-patterns.md +868 -868
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/mapkit-location/SKILL.md +485 -485
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/mapkit-location/references/corelocation-patterns.md +730 -730
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/mapkit-location/references/mapkit-patterns.md +748 -748
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/metrickit-diagnostics/SKILL.md +479 -479
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/musickit-audio/SKILL.md +395 -395
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/musickit-audio/references/musickit-patterns.md +363 -363
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/natural-language/SKILL.md +412 -412
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/natural-language/references/translation-patterns.md +311 -311
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/passkit-wallet/SKILL.md +398 -398
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/passkit-wallet/references/wallet-passes.md +254 -254
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/pencilkit-drawing/SKILL.md +387 -387
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/pencilkit-drawing/references/paperkit-integration.md +376 -376
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/pencilkit-drawing/references/pencilkit-patterns.md +302 -302
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/permissionkit/SKILL.md +446 -446
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/permissionkit/references/permissionkit-patterns.md +435 -435
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/photos-camera-media/SKILL.md +500 -500
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/photos-camera-media/references/av-playback.md +701 -701
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/photos-camera-media/references/camera-capture.md +774 -774
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/photos-camera-media/references/image-loading-caching.md +869 -869
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/photos-camera-media/references/photospicker-patterns.md +597 -597
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/push-notifications/SKILL.md +500 -500
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/push-notifications/references/notification-patterns.md +677 -677
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/push-notifications/references/rich-notifications.md +745 -745
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/realitykit-ar/SKILL.md +479 -479
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/realitykit-ar/references/realitykit-patterns.md +480 -480
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/shareplay-activities/SKILL.md +483 -483
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/shareplay-activities/references/shareplay-patterns.md +544 -544
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/speech-recognition/SKILL.md +485 -485
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/storekit/SKILL.md +478 -478
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/storekit/references/app-review-guidelines.md +58 -58
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/storekit/references/storekit-advanced.md +755 -755
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swift-charts/SKILL.md +487 -487
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swift-charts/references/charts-patterns.md +895 -895
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swift-concurrency/SKILL.md +408 -408
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swift-concurrency/references/approachable-concurrency.md +80 -80
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swift-concurrency/references/swift-6-2-concurrency.md +233 -233
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swift-concurrency/references/swiftui-concurrency.md +187 -187
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swift-concurrency/references/synchronization-primitives.md +341 -341
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swift-language/SKILL.md +498 -498
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swift-language/references/swift-patterns-extended.md +505 -505
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swift-testing/SKILL.md +467 -467
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swift-testing/references/testing-patterns.md +504 -504
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftdata/SKILL.md +334 -334
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftdata/references/core-data-coexistence.md +504 -504
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftdata/references/swiftdata-advanced.md +975 -975
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftdata/references/swiftdata-queries.md +675 -675
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-animation/SKILL.md +481 -481
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-animation/references/animation-advanced.md +804 -804
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-animation/references/core-animation-bridge.md +553 -553
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-gestures/SKILL.md +450 -450
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-gestures/references/gesture-patterns.md +425 -425
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-layout-components/SKILL.md +336 -336
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-layout-components/references/form.md +97 -97
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-layout-components/references/grids.md +69 -69
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-layout-components/references/list.md +99 -99
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-layout-components/references/scrollview.md +147 -147
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-liquid-glass/SKILL.md +325 -325
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-liquid-glass/references/liquid-glass.md +387 -387
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-navigation/SKILL.md +262 -262
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-navigation/references/deeplinks.md +207 -207
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-navigation/references/navigationstack.md +177 -177
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-navigation/references/sheets.md +169 -169
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-navigation/references/tabview.md +178 -178
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-patterns/SKILL.md +381 -381
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-patterns/references/architecture-patterns.md +486 -486
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-patterns/references/deprecated-migration.md +1097 -1097
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-patterns/references/design-polish.md +780 -780
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-patterns/references/platform-and-sharing.md +696 -696
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-performance/SKILL.md +491 -491
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-performance/references/demystify-swiftui-performance-wwdc23.md +46 -46
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-performance/references/optimizing-swiftui-performance-instruments.md +29 -29
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-performance/references/understanding-hangs-in-your-app.md +33 -33
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-performance/references/understanding-improving-swiftui-performance.md +52 -52
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-uikit-interop/SKILL.md +428 -428
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-uikit-interop/references/hosting-migration.md +534 -534
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/swiftui-uikit-interop/references/representable-recipes.md +1133 -1133
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/tipkit/SKILL.md +494 -494
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/tipkit/references/tipkit-patterns.md +782 -782
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/vision-framework/SKILL.md +475 -475
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/vision-framework/references/vision-requests.md +736 -736
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/vision-framework/references/visionkit-scanner.md +738 -738
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/weatherkit/SKILL.md +410 -410
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/weatherkit/references/weatherkit-patterns.md +567 -567
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/widgetkit/SKILL.md +497 -497
- package/src/.agents/skills/ui-ux-pro-custom/data/swift-ios-skills/widgetkit/references/widgetkit-advanced.md +871 -871
- package/src/.agents/skills/ui-ux-pro-custom/data/typography.csv +57 -57
- package/src/.agents/skills/ui-ux-pro-custom/data/ui-reasoning.csv +101 -101
- package/src/.agents/skills/ui-ux-pro-custom/data/ux-guidelines.csv +99 -99
- package/src/.agents/skills/ui-ux-pro-custom/data/web-interface.csv +31 -31
- package/src/.agents/skills/ui-ux-pro-custom/scripts/core.py +253 -253
- package/src/.agents/skills/ui-ux-pro-custom/scripts/design_system.py +1067 -1067
- package/src/.agents/skills/ui-ux-pro-custom/scripts/search.py +114 -114
- package/src/.agents/skills/ux-audit/SKILL.md +150 -150
- package/src/.agents/skills/websocket-engineer/SKILL.md +168 -168
- package/src/.agents/skills/websocket-engineer/references/alternatives.md +391 -391
- package/src/.agents/skills/websocket-engineer/references/patterns.md +400 -400
- package/src/.agents/skills/websocket-engineer/references/protocol.md +195 -195
- package/src/.agents/skills/websocket-engineer/references/scaling.md +333 -333
- package/src/.agents/skills/websocket-engineer/references/security.md +474 -474
- package/src/.agents/skills/writing-skills/SKILL.md +655 -655
- package/src/.agents/skills/writing-skills/anthropic-best-practices.md +1150 -1150
- package/src/.agents/skills/writing-skills/examples/CLAUDE_MD_TESTING.md +189 -189
- package/src/.agents/skills/writing-skills/graphviz-conventions.dot +171 -171
- package/src/.agents/skills/writing-skills/persuasion-principles.md +187 -187
- package/src/.agents/skills/writing-skills/render-graphs.js +168 -168
- package/src/.agents/skills/writing-skills/testing-skills-with-subagents.md +384 -384
- package/src/.claude/commands/master-orchestrator.md +15 -0
- package/src/_memory/config.yaml +11 -11
- package/src/_memory/master-orchestrator-sidecar/instructions.md +85 -32
- package/src/_memory/skills/nimbalyst-tracking/SKILL.md +103 -103
- package/src/_memory/skills/writing-skills/SKILL.md +655 -655
- package/src/bmb/agents/agent-builder.md +59 -59
- package/src/bmb/agents/module-builder.md +60 -60
- package/src/bmb/agents/workflow-builder.md +61 -61
- package/src/bmb/config.yaml +12 -12
- package/src/bmb/module-help.csv +13 -13
- package/src/bmb/workflows/agent/data/agent-architecture.md +258 -258
- package/src/bmb/workflows/agent/data/agent-compilation.md +185 -185
- package/src/bmb/workflows/agent/data/agent-menu-patterns.md +189 -189
- package/src/bmb/workflows/agent/data/agent-metadata.md +133 -133
- package/src/bmb/workflows/agent/data/agent-validation.md +111 -111
- package/src/bmb/workflows/agent/data/brainstorm-context.md +96 -96
- package/src/bmb/workflows/agent/data/communication-presets.csv +61 -61
- package/src/bmb/workflows/agent/data/critical-actions.md +75 -75
- package/src/bmb/workflows/agent/data/persona-properties.md +252 -252
- package/src/bmb/workflows/agent/data/principles-crafting.md +142 -142
- package/src/bmb/workflows/agent/data/reference/module-examples/architect.md +68 -68
- package/src/bmb/workflows/agent/data/reference/with-sidecar/journal-keeper/journal-keeper-sidecar/entries/yy-mm-dd-entry-template.md +16 -16
- package/src/bmb/workflows/agent/data/understanding-agent-types.md +126 -126
- package/src/bmb/workflows/agent/steps-c/step-01-brainstorm.md +129 -129
- package/src/bmb/workflows/agent/steps-c/step-02-discovery.md +170 -170
- package/src/bmb/workflows/agent/steps-c/step-03-sidecar-metadata.md +309 -309
- package/src/bmb/workflows/agent/steps-c/step-04-persona.md +213 -213
- package/src/bmb/workflows/agent/steps-c/step-05-commands-menu.md +179 -179
- package/src/bmb/workflows/agent/steps-c/step-06-activation.md +278 -278
- package/src/bmb/workflows/agent/steps-c/step-07-build-agent.md +316 -316
- package/src/bmb/workflows/agent/steps-c/step-08-celebrate.md +247 -247
- package/src/bmb/workflows/agent/steps-e/e-01-load-existing.md +221 -221
- package/src/bmb/workflows/agent/steps-e/e-02-discover-edits.md +195 -195
- package/src/bmb/workflows/agent/steps-e/e-04-sidecar-metadata.md +126 -126
- package/src/bmb/workflows/agent/steps-e/e-05-persona.md +135 -135
- package/src/bmb/workflows/agent/steps-e/e-06-commands-menu.md +123 -123
- package/src/bmb/workflows/agent/steps-e/e-07-activation.md +124 -124
- package/src/bmb/workflows/agent/steps-e/e-08-edit-agent.md +197 -197
- package/src/bmb/workflows/agent/steps-e/e-09-celebrate.md +155 -155
- package/src/bmb/workflows/agent/steps-v/v-01-load-review.md +137 -137
- package/src/bmb/workflows/agent/steps-v/v-02a-validate-metadata.md +116 -116
- package/src/bmb/workflows/agent/steps-v/v-02b-validate-persona.md +124 -124
- package/src/bmb/workflows/agent/steps-v/v-02c-validate-menu.md +127 -127
- package/src/bmb/workflows/agent/steps-v/v-02d-validate-structure.md +134 -134
- package/src/bmb/workflows/agent/steps-v/v-02e-validate-sidecar.md +134 -134
- package/src/bmb/workflows/agent/steps-v/v-03-summary.md +104 -104
- package/src/bmb/workflows/agent/templates/agent-plan.template.md +5 -5
- package/src/bmb/workflows/agent/templates/agent-template.md +89 -89
- package/src/bmb/workflows/agent/workflow-create-agent.md +72 -72
- package/src/bmb/workflows/agent/workflow-edit-agent.md +75 -75
- package/src/bmb/workflows/agent/workflow-validate-agent.md +73 -73
- package/src/bmb/workflows/module/data/agent-architecture.md +179 -179
- package/src/bmb/workflows/module/data/agent-spec-template.md +79 -79
- package/src/bmb/workflows/module/data/module-standards.md +263 -263
- package/src/bmb/workflows/module/data/module-yaml-conventions.md +392 -392
- package/src/bmb/workflows/module/module-help-generate.md +254 -254
- package/src/bmb/workflows/module/steps-b/step-01-welcome.md +148 -148
- package/src/bmb/workflows/module/steps-b/step-02-spark.md +141 -141
- package/src/bmb/workflows/module/steps-b/step-03-module-type.md +149 -149
- package/src/bmb/workflows/module/steps-b/step-04-vision.md +83 -83
- package/src/bmb/workflows/module/steps-b/step-05-identity.md +97 -97
- package/src/bmb/workflows/module/steps-b/step-06-users.md +86 -86
- package/src/bmb/workflows/module/steps-b/step-07-value.md +76 -76
- package/src/bmb/workflows/module/steps-b/step-08-agents.md +97 -97
- package/src/bmb/workflows/module/steps-b/step-09-workflows.md +83 -83
- package/src/bmb/workflows/module/steps-b/step-10-tools.md +91 -91
- package/src/bmb/workflows/module/steps-b/step-11-scenarios.md +84 -84
- package/src/bmb/workflows/module/steps-b/step-12-creative.md +95 -95
- package/src/bmb/workflows/module/steps-b/step-13-review.md +105 -105
- package/src/bmb/workflows/module/steps-b/step-14-finalize.md +117 -117
- package/src/bmb/workflows/module/steps-c/step-01-load-brief.md +179 -179
- package/src/bmb/workflows/module/steps-c/step-01b-continue.md +82 -82
- package/src/bmb/workflows/module/steps-c/step-02-structure.md +105 -105
- package/src/bmb/workflows/module/steps-c/step-03-config.md +119 -119
- package/src/bmb/workflows/module/steps-c/step-04-agents.md +168 -168
- package/src/bmb/workflows/module/steps-c/step-05-workflows.md +184 -184
- package/src/bmb/workflows/module/steps-c/step-06-docs.md +401 -401
- package/src/bmb/workflows/module/steps-c/step-07-complete.md +152 -152
- package/src/bmb/workflows/module/steps-e/step-01-load-target.md +81 -81
- package/src/bmb/workflows/module/steps-e/step-02-select-edit.md +77 -77
- package/src/bmb/workflows/module/steps-e/step-03-apply-edit.md +77 -77
- package/src/bmb/workflows/module/steps-e/step-04-review.md +80 -80
- package/src/bmb/workflows/module/steps-e/step-05-confirm.md +75 -75
- package/src/bmb/workflows/module/steps-v/step-01-load-target.md +96 -96
- package/src/bmb/workflows/module/steps-v/step-02-file-structure.md +93 -93
- package/src/bmb/workflows/module/steps-v/step-03-module-yaml.md +99 -99
- package/src/bmb/workflows/module/steps-v/step-04-agent-specs.md +152 -152
- package/src/bmb/workflows/module/steps-v/step-05-workflow-specs.md +152 -152
- package/src/bmb/workflows/module/steps-v/step-06-documentation.md +143 -143
- package/src/bmb/workflows/module/steps-v/step-07-installation.md +102 -102
- package/src/bmb/workflows/module/steps-v/step-08-report.md +197 -197
- package/src/bmb/workflows/module/templates/brief-template.md +154 -154
- package/src/bmb/workflows/module/templates/workflow-spec-template.md +96 -96
- package/src/bmb/workflows/module/workflow-create-module-brief.md +71 -71
- package/src/bmb/workflows/module/workflow-create-module.md +86 -86
- package/src/bmb/workflows/module/workflow-edit-module.md +66 -66
- package/src/bmb/workflows/module/workflow-validate-module.md +66 -66
- package/src/bmb/workflows/workflow/data/architecture.md +150 -150
- package/src/bmb/workflows/workflow/data/common-workflow-tools.csv +19 -19
- package/src/bmb/workflows/workflow/data/csv-data-file-standards.md +53 -53
- package/src/bmb/workflows/workflow/data/frontmatter-standards.md +184 -184
- package/src/bmb/workflows/workflow/data/input-discovery-standards.md +191 -191
- package/src/bmb/workflows/workflow/data/intent-vs-prescriptive-spectrum.md +44 -44
- package/src/bmb/workflows/workflow/data/menu-handling-standards.md +133 -133
- package/src/bmb/workflows/workflow/data/output-format-standards.md +135 -135
- package/src/bmb/workflows/workflow/data/step-file-rules.md +235 -235
- package/src/bmb/workflows/workflow/data/step-type-patterns.md +257 -257
- package/src/bmb/workflows/workflow/data/subprocess-optimization-patterns.md +188 -188
- package/src/bmb/workflows/workflow/data/trimodal-workflow-structure.md +164 -164
- package/src/bmb/workflows/workflow/data/workflow-chaining-standards.md +222 -222
- package/src/bmb/workflows/workflow/data/workflow-examples.md +232 -232
- package/src/bmb/workflows/workflow/data/workflow-type-criteria.md +134 -134
- package/src/bmb/workflows/workflow/steps-c/step-00-conversion.md +263 -263
- package/src/bmb/workflows/workflow/steps-c/step-01-discovery.md +194 -194
- package/src/bmb/workflows/workflow/steps-c/step-01b-continuation.md +3 -3
- package/src/bmb/workflows/workflow/steps-c/step-02-classification.md +270 -270
- package/src/bmb/workflows/workflow/steps-c/step-03-requirements.md +283 -283
- package/src/bmb/workflows/workflow/steps-c/step-04-tools.md +282 -282
- package/src/bmb/workflows/workflow/steps-c/step-05-plan-review.md +243 -243
- package/src/bmb/workflows/workflow/steps-c/step-06-design.md +330 -330
- package/src/bmb/workflows/workflow/steps-c/step-07-foundation.md +239 -239
- package/src/bmb/workflows/workflow/steps-c/step-08-build-step-01.md +379 -379
- package/src/bmb/workflows/workflow/steps-c/step-09-build-next-step.md +350 -350
- package/src/bmb/workflows/workflow/steps-c/step-10-confirmation.md +322 -322
- package/src/bmb/workflows/workflow/steps-c/step-11-completion.md +191 -191
- package/src/bmb/workflows/workflow/steps-e/step-e-01-assess-workflow.md +237 -237
- package/src/bmb/workflows/workflow/steps-e/step-e-02-discover-edits.md +251 -251
- package/src/bmb/workflows/workflow/steps-e/step-e-03-fix-validation.md +254 -254
- package/src/bmb/workflows/workflow/steps-e/step-e-04-direct-edit.md +277 -277
- package/src/bmb/workflows/workflow/steps-e/step-e-05-apply-edit.md +154 -154
- package/src/bmb/workflows/workflow/steps-e/step-e-06-validate-after.md +190 -190
- package/src/bmb/workflows/workflow/steps-e/step-e-07-complete.md +206 -206
- package/src/bmb/workflows/workflow/steps-v/step-01-validate-max-mode.md +109 -109
- package/src/bmb/workflows/workflow/steps-v/step-01-validate.md +221 -221
- package/src/bmb/workflows/workflow/steps-v/step-01b-structure.md +152 -152
- package/src/bmb/workflows/workflow/steps-v/step-02-frontmatter-validation.md +199 -199
- package/src/bmb/workflows/workflow/steps-v/step-02b-path-violations.md +265 -265
- package/src/bmb/workflows/workflow/steps-v/step-03-menu-validation.md +164 -164
- package/src/bmb/workflows/workflow/steps-v/step-04-step-type-validation.md +211 -211
- package/src/bmb/workflows/workflow/steps-v/step-05-output-format-validation.md +200 -200
- package/src/bmb/workflows/workflow/steps-v/step-06-validation-design-check.md +195 -195
- package/src/bmb/workflows/workflow/steps-v/step-07-instruction-style-check.md +209 -209
- package/src/bmb/workflows/workflow/steps-v/step-08-collaborative-experience-check.md +199 -199
- package/src/bmb/workflows/workflow/steps-v/step-08b-subprocess-optimization.md +179 -179
- package/src/bmb/workflows/workflow/steps-v/step-09-cohesive-review.md +186 -186
- package/src/bmb/workflows/workflow/steps-v/step-10-report-complete.md +154 -154
- package/src/bmb/workflows/workflow/steps-v/step-11-plan-validation.md +237 -237
- package/src/bmb/workflows/workflow/templates/minimal-output-template.md +11 -11
- package/src/bmb/workflows/workflow/templates/step-01-init-continuable-template.md +241 -241
- package/src/bmb/workflows/workflow/templates/step-1b-template.md +224 -224
- package/src/bmb/workflows/workflow/templates/step-template.md +294 -294
- package/src/bmb/workflows/workflow/templates/workflow-template.md +102 -102
- package/src/bmb/workflows/workflow/workflow-create-workflow.md +79 -79
- package/src/bmb/workflows/workflow/workflow-edit-workflow.md +65 -65
- package/src/bmb/workflows/workflow/workflow-rework-workflow.md +65 -65
- package/src/bmb/workflows/workflow/workflow-validate-max-parallel-workflow.md +66 -66
- package/src/bmb/workflows/workflow/workflow-validate-workflow.md +65 -65
- package/src/bmm/agents/analyst.md +104 -104
- package/src/bmm/agents/dev.md +100 -100
- package/src/bmm/agents/qa.md +100 -90
- package/src/bmm/agents/tech-writer/tech-writer.md +94 -94
- package/src/bmm/module-help.csv +31 -31
- package/src/bmm/workflows/1-analysis/create-product-brief/steps/step-01-init.md +115 -115
- package/src/bmm/workflows/1-analysis/create-product-brief/steps/step-01b-continue.md +107 -107
- package/src/bmm/workflows/1-analysis/create-product-brief/steps/step-02-vision.md +141 -141
- package/src/bmm/workflows/1-analysis/create-product-brief/steps/step-03-users.md +144 -144
- package/src/bmm/workflows/1-analysis/create-product-brief/steps/step-04-metrics.md +147 -147
- package/src/bmm/workflows/1-analysis/create-product-brief/steps/step-05-scope.md +161 -161
- package/src/bmm/workflows/1-analysis/create-product-brief/steps/step-06-complete.md +99 -99
- package/src/bmm/workflows/1-analysis/create-product-brief/workflow.md +57 -57
- package/src/bmm/workflows/1-analysis/research/domain-steps/step-01-init.md +87 -87
- package/src/bmm/workflows/1-analysis/research/domain-steps/step-02-domain-analysis.md +156 -156
- package/src/bmm/workflows/1-analysis/research/domain-steps/step-03-competitive-landscape.md +165 -165
- package/src/bmm/workflows/1-analysis/research/domain-steps/step-04-regulatory-focus.md +140 -140
- package/src/bmm/workflows/1-analysis/research/domain-steps/step-05-technical-trends.md +152 -152
- package/src/bmm/workflows/1-analysis/research/domain-steps/step-06-research-synthesis.md +345 -345
- package/src/bmm/workflows/1-analysis/research/market-steps/step-01-init.md +92 -92
- package/src/bmm/workflows/1-analysis/research/market-steps/step-02-customer-behavior.md +164 -164
- package/src/bmm/workflows/1-analysis/research/market-steps/step-03-customer-pain-points.md +174 -174
- package/src/bmm/workflows/1-analysis/research/market-steps/step-04-customer-decisions.md +184 -184
- package/src/bmm/workflows/1-analysis/research/market-steps/step-05-competitive-analysis.md +105 -105
- package/src/bmm/workflows/1-analysis/research/market-steps/step-06-research-completion.md +360 -360
- package/src/bmm/workflows/1-analysis/research/technical-steps/step-01-init.md +87 -87
- package/src/bmm/workflows/1-analysis/research/technical-steps/step-02-technical-overview.md +165 -165
- package/src/bmm/workflows/1-analysis/research/technical-steps/step-03-integration-patterns.md +174 -174
- package/src/bmm/workflows/1-analysis/research/technical-steps/step-04-architectural-patterns.md +141 -141
- package/src/bmm/workflows/1-analysis/research/technical-steps/step-05-implementation-research.md +159 -159
- package/src/bmm/workflows/1-analysis/research/technical-steps/step-06-research-synthesis.md +387 -387
- package/src/bmm/workflows/1-analysis/research/workflow-domain-research.md +54 -54
- package/src/bmm/workflows/1-analysis/research/workflow-market-research.md +54 -54
- package/src/bmm/workflows/1-analysis/research/workflow-technical-research.md +54 -54
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-c/step-01b-continue.md +100 -100
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-c/step-02-discovery.md +160 -160
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-c/step-02b-vision.md +88 -88
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-c/step-02c-executive-summary.md +99 -99
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-c/step-03-success.md +169 -169
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-c/step-04-journeys.md +156 -156
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-c/step-05-domain.md +136 -136
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-c/step-06-innovation.md +176 -176
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-c/step-07-project-type.md +184 -184
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-c/step-08-scoping.md +174 -174
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-c/step-09-functional.md +175 -175
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-c/step-10-nonfunctional.md +189 -189
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-c/step-11-polish.md +162 -162
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-c/step-12-complete.md +79 -79
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-e/step-e-01-discovery.md +183 -183
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-e/step-e-01b-legacy-conversion.md +149 -149
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-e/step-e-02-review.md +187 -187
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-e/step-e-03-edit.md +192 -192
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-e/step-e-04-complete.md +108 -108
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-v/step-v-01-discovery.md +166 -166
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-v/step-v-02-format-detection.md +131 -131
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-v/step-v-02b-parity-check.md +150 -150
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-v/step-v-03-density-validation.md +118 -118
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-v/step-v-04-brief-coverage-validation.md +155 -155
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-v/step-v-05-measurability-validation.md +170 -170
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-v/step-v-06-traceability-validation.md +158 -158
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-v/step-v-07-implementation-leakage-validation.md +147 -147
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-v/step-v-08-domain-compliance-validation.md +182 -182
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-v/step-v-09-project-type-validation.md +202 -202
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-v/step-v-10-smart-validation.md +148 -148
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-v/step-v-11-holistic-quality-validation.md +201 -201
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-v/step-v-12-completeness-validation.md +179 -179
- package/src/bmm/workflows/2-plan-workflows/create-prd/steps-v/step-v-13-report-complete.md +164 -164
- package/src/bmm/workflows/2-plan-workflows/create-prd/workflow-create-prd.md +65 -65
- package/src/bmm/workflows/2-plan-workflows/create-prd/workflow-edit-prd.md +65 -65
- package/src/bmm/workflows/2-plan-workflows/create-prd/workflow-validate-prd.md +63 -63
- package/src/bmm/workflows/2-plan-workflows/create-ux-design/steps/step-01b-continue.md +63 -63
- package/src/bmm/workflows/2-plan-workflows/create-ux-design/steps/step-02-discovery.md +106 -106
- package/src/bmm/workflows/2-plan-workflows/create-ux-design/steps/step-03-core-experience.md +111 -111
- package/src/bmm/workflows/2-plan-workflows/create-ux-design/steps/step-04-emotional-response.md +115 -115
- package/src/bmm/workflows/2-plan-workflows/create-ux-design/steps/step-05-inspiration.md +127 -127
- package/src/bmm/workflows/2-plan-workflows/create-ux-design/steps/step-06-design-system.md +167 -167
- package/src/bmm/workflows/2-plan-workflows/create-ux-design/steps/step-07-defining-experience.md +143 -143
- package/src/bmm/workflows/2-plan-workflows/create-ux-design/steps/step-08-visual-foundation.md +118 -118
- package/src/bmm/workflows/2-plan-workflows/create-ux-design/steps/step-09-design-directions.md +154 -154
- package/src/bmm/workflows/2-plan-workflows/create-ux-design/steps/step-10-user-journeys.md +136 -136
- package/src/bmm/workflows/2-plan-workflows/create-ux-design/steps/step-11-component-strategy.md +165 -165
- package/src/bmm/workflows/2-plan-workflows/create-ux-design/steps/step-12-ux-patterns.md +135 -135
- package/src/bmm/workflows/2-plan-workflows/create-ux-design/steps/step-13-responsive-accessibility.md +192 -192
- package/src/bmm/workflows/2-plan-workflows/create-ux-design/steps/step-14-complete.md +101 -101
- package/src/bmm/workflows/2-plan-workflows/create-ux-design/workflow.md +45 -45
- package/src/bmm/workflows/3-solutioning/check-implementation-readiness/steps/step-01-document-discovery.md +185 -185
- package/src/bmm/workflows/3-solutioning/check-implementation-readiness/steps/step-02-prd-analysis.md +129 -129
- package/src/bmm/workflows/3-solutioning/check-implementation-readiness/steps/step-03-epic-coverage-validation.md +130 -130
- package/src/bmm/workflows/3-solutioning/check-implementation-readiness/steps/step-04-ux-alignment.md +93 -93
- package/src/bmm/workflows/3-solutioning/check-implementation-readiness/steps/step-05-epic-quality-review.md +196 -196
- package/src/bmm/workflows/3-solutioning/check-implementation-readiness/steps/step-06-final-assessment.md +129 -129
- package/src/bmm/workflows/3-solutioning/check-implementation-readiness/workflow.md +54 -54
- package/src/bmm/workflows/3-solutioning/create-architecture/steps/step-01b-continue.md +82 -82
- package/src/bmm/workflows/3-solutioning/create-architecture/steps/step-02-context.md +106 -106
- package/src/bmm/workflows/3-solutioning/create-architecture/steps/step-03-starter.md +138 -138
- package/src/bmm/workflows/3-solutioning/create-architecture/steps/step-04-decisions.md +129 -129
- package/src/bmm/workflows/3-solutioning/create-architecture/steps/step-05-patterns.md +166 -166
- package/src/bmm/workflows/3-solutioning/create-architecture/steps/step-06-structure.md +186 -186
- package/src/bmm/workflows/3-solutioning/create-architecture/steps/step-07-validation.md +163 -163
- package/src/bmm/workflows/3-solutioning/create-architecture/steps/step-08-complete.md +38 -38
- package/src/bmm/workflows/3-solutioning/create-architecture/workflow.md +49 -49
- package/src/bmm/workflows/3-solutioning/create-epics-and-stories/steps/step-02-design-epics.md +124 -124
- package/src/bmm/workflows/3-solutioning/create-epics-and-stories/steps/step-03-create-stories.md +122 -122
- package/src/bmm/workflows/3-solutioning/create-epics-and-stories/steps/step-04-final-validation.md +84 -84
- package/src/bmm/workflows/3-solutioning/create-epics-and-stories/workflow.md +58 -58
- package/src/bmm/workflows/4-implementation/code-review/workflow.yaml +43 -43
- package/src/bmm/workflows/4-implementation/correct-course/workflow.yaml +53 -53
- package/src/bmm/workflows/4-implementation/create-story/checklist.md +159 -159
- package/src/bmm/workflows/4-implementation/create-story/template.md +79 -79
- package/src/bmm/workflows/4-implementation/create-story/workflow.yaml +52 -52
- package/src/bmm/workflows/4-implementation/dev-story/workflow.yaml +20 -20
- package/src/bmm/workflows/4-implementation/retrospective/workflow.yaml +52 -52
- package/src/bmm/workflows/4-implementation/sprint-planning/workflow.yaml +52 -52
- package/src/bmm/workflows/4-implementation/sprint-status/workflow.yaml +25 -25
- package/src/bmm/workflows/bmad-quick-flow/quick-dev/steps/step-01-mode-detection.md +158 -158
- package/src/bmm/workflows/bmad-quick-flow/quick-dev/steps/step-02-context-gathering.md +122 -122
- package/src/bmm/workflows/bmad-quick-flow/quick-dev/steps/step-03-execute.md +93 -93
- package/src/bmm/workflows/bmad-quick-flow/quick-dev/steps/step-04-self-check.md +93 -93
- package/src/bmm/workflows/bmad-quick-flow/quick-dev/steps/step-05-adversarial-review.md +87 -87
- package/src/bmm/workflows/bmad-quick-flow/quick-dev/steps/step-06-resolve-findings.md +146 -146
- package/src/bmm/workflows/bmad-quick-flow/quick-dev/workflow.md +50 -50
- package/src/bmm/workflows/bmad-quick-flow/quick-spec/steps/step-02-investigate.md +152 -152
- package/src/bmm/workflows/bmad-quick-flow/quick-spec/steps/step-03-generate.md +123 -123
- package/src/bmm/workflows/bmad-quick-flow/quick-spec/steps/step-04-review.md +201 -201
- package/src/bmm/workflows/bmad-quick-flow/quick-spec/workflow.md +79 -79
- package/src/bmm/workflows/document-project/workflow.yaml +22 -22
- package/src/bmm/workflows/generate-project-context/steps/step-01-discover.md +184 -184
- package/src/bmm/workflows/generate-project-context/steps/step-02-generate.md +322 -322
- package/src/bmm/workflows/generate-project-context/steps/step-03-complete.md +235 -235
- package/src/bmm/workflows/generate-project-context/workflow.md +49 -49
- package/src/bmm/workflows/qa/automate/workflow.yaml +233 -233
- package/src/bmm/workflows/qa-generate-e2e-tests/workflow.yaml +42 -42
- package/src/core/config.yaml +9 -9
- package/src/core/module-help.csv +10 -10
- package/src/core/scripts/generate-loop-report.py +72 -72
- package/src/core/tasks/editorial-review-prose.xml +101 -101
- package/src/core/tasks/editorial-review-structure.xml +207 -207
- package/src/core/tasks/help.md +86 -86
- package/src/core/tasks/index-docs.xml +64 -64
- package/src/core/tasks/review-adversarial-general.xml +66 -66
- package/src/core/tasks/review-adversarial-loop.xml +46 -46
- package/src/core/tasks/review-edge-case-hunter.xml +63 -63
- package/src/core/tasks/review-party-loop.xml +46 -46
- package/src/core/tasks/shard-doc.xml +107 -107
- package/src/core/tasks/workflow.xml +235 -235
- package/src/core/templates/review-loop-report.html +88 -88
- package/src/core/templates/review-loop-report.md +5 -5
- package/src/core/workflows/advanced-elicitation/workflow.xml +117 -117
- package/src/core/workflows/brainstorming/steps/step-01-session-setup.md +212 -212
- package/src/core/workflows/brainstorming/steps/step-01b-continue.md +122 -122
- package/src/core/workflows/brainstorming/steps/step-02a-user-selected.md +225 -225
- package/src/core/workflows/brainstorming/steps/step-02b-ai-recommended.md +237 -237
- package/src/core/workflows/brainstorming/steps/step-02c-random-selection.md +209 -209
- package/src/core/workflows/brainstorming/steps/step-02d-progressive-flow.md +264 -264
- package/src/core/workflows/brainstorming/steps/step-02e-deep-dive.md +68 -68
- package/src/core/workflows/brainstorming/steps/step-03-technique-execution.md +403 -403
- package/src/core/workflows/brainstorming/steps/step-04-idea-organization.md +303 -303
- package/src/core/workflows/brainstorming/workflow.md +60 -60
- package/src/core/workflows/extract-trackers/workflow.md +45 -45
- package/src/core/workflows/party-mode/steps/step-01-agent-loading.md +142 -142
- package/src/core/workflows/party-mode/workflow.md +194 -194
- package/src/docs/dev/tmux/actions_popup.py +291 -291
- package/src/docs/dev/tmux/tmux-setup.md +62 -1
|
@@ -1,492 +1,492 @@
|
|
|
1
|
-
---
|
|
2
|
-
name: cloudkit-sync
|
|
3
|
-
description: "Implement, review, or improve CloudKit and iCloud sync in iOS/macOS apps. Use when working with CKContainer, CKRecord, CKQuery, CKSubscription, CKSyncEngine, CKShare, NSUbiquitousKeyValueStore, or iCloud Drive file coordination; when syncing SwiftData models via ModelConfiguration with cloudKitDatabase; when handling CKError codes for conflict resolution, network failures, or quota limits; or when checking iCloud account status before performing sync operations."
|
|
4
|
-
---
|
|
5
|
-
|
|
6
|
-
# CloudKit and iCloud Sync
|
|
7
|
-
|
|
8
|
-
Sync data across devices using CloudKit, iCloud key-value storage, and iCloud
|
|
9
|
-
Drive. Covers container setup, record CRUD, queries, subscriptions, CKSyncEngine,
|
|
10
|
-
SwiftData integration, conflict resolution, and error handling. Targets iOS 26+
|
|
11
|
-
with Swift 6.2; older availability noted where relevant.
|
|
12
|
-
|
|
13
|
-
## Contents
|
|
14
|
-
|
|
15
|
-
- [Container and Database Setup](#container-and-database-setup)
|
|
16
|
-
- [CKRecord CRUD](#ckrecord-crud)
|
|
17
|
-
- [CKQuery](#ckquery)
|
|
18
|
-
- [CKSubscription](#cksubscription)
|
|
19
|
-
- [CKSyncEngine (iOS 17+)](#cksyncengine-ios-17)
|
|
20
|
-
- [SwiftData + CloudKit](#swiftdata--cloudkit)
|
|
21
|
-
- [NSUbiquitousKeyValueStore](#nsubiquitouskeyvaluestore)
|
|
22
|
-
- [iCloud Drive File Sync](#icloud-drive-file-sync)
|
|
23
|
-
- [Account Status and Error Handling](#account-status-and-error-handling)
|
|
24
|
-
- [Conflict Resolution](#conflict-resolution)
|
|
25
|
-
- [Common Mistakes](#common-mistakes)
|
|
26
|
-
- [Review Checklist](#review-checklist)
|
|
27
|
-
- [References](#references)
|
|
28
|
-
|
|
29
|
-
## Container and Database Setup
|
|
30
|
-
|
|
31
|
-
Enable iCloud + CloudKit in Signing & Capabilities. A container provides
|
|
32
|
-
three databases:
|
|
33
|
-
|
|
34
|
-
| Database | Scope | Requires iCloud | Storage Quota |
|
|
35
|
-
|----------|-------|-----------------|---------------|
|
|
36
|
-
| Public | All users | Read: No, Write: Yes | App quota |
|
|
37
|
-
| Private | Current user | Yes | User quota |
|
|
38
|
-
| Shared | Shared records | Yes | Owner quota |
|
|
39
|
-
|
|
40
|
-
```swift
|
|
41
|
-
import CloudKit
|
|
42
|
-
|
|
43
|
-
let container = CKContainer.default()
|
|
44
|
-
// Or named: CKContainer(identifier: "iCloud.com.example.app")
|
|
45
|
-
|
|
46
|
-
let publicDB = container.publicCloudDatabase
|
|
47
|
-
let privateDB = container.privateCloudDatabase
|
|
48
|
-
let sharedDB = container.sharedCloudDatabase
|
|
49
|
-
```
|
|
50
|
-
|
|
51
|
-
## CKRecord CRUD
|
|
52
|
-
|
|
53
|
-
Records are key-value pairs. Max 1 MB per record (excluding CKAsset data).
|
|
54
|
-
|
|
55
|
-
```swift
|
|
56
|
-
// CREATE
|
|
57
|
-
let record = CKRecord(recordType: "Note")
|
|
58
|
-
record["title"] = "Meeting Notes" as CKRecordValue
|
|
59
|
-
record["body"] = "Discussed Q3 roadmap" as CKRecordValue
|
|
60
|
-
record["createdAt"] = Date() as CKRecordValue
|
|
61
|
-
record["tags"] = ["work", "planning"] as CKRecordValue
|
|
62
|
-
let saved = try await privateDB.save(record)
|
|
63
|
-
|
|
64
|
-
// FETCH by ID
|
|
65
|
-
let recordID = CKRecord.ID(recordName: "unique-id-123")
|
|
66
|
-
let fetched = try await privateDB.record(for: recordID)
|
|
67
|
-
|
|
68
|
-
// UPDATE -- fetch first, modify, then save
|
|
69
|
-
fetched["title"] = "Updated Title" as CKRecordValue
|
|
70
|
-
let updated = try await privateDB.save(fetched)
|
|
71
|
-
|
|
72
|
-
// DELETE
|
|
73
|
-
try await privateDB.deleteRecord(withID: recordID)
|
|
74
|
-
```
|
|
75
|
-
|
|
76
|
-
### Custom Record Zones (Private/Shared Only)
|
|
77
|
-
|
|
78
|
-
Custom zones support atomic commits, change tracking, and sharing.
|
|
79
|
-
|
|
80
|
-
```swift
|
|
81
|
-
let zoneID = CKRecordZone.ID(zoneName: "NotesZone")
|
|
82
|
-
let zone = CKRecordZone(zoneID: zoneID)
|
|
83
|
-
try await privateDB.save(zone)
|
|
84
|
-
|
|
85
|
-
let recordID = CKRecord.ID(recordName: UUID().uuidString, zoneID: zoneID)
|
|
86
|
-
let record = CKRecord(recordType: "Note", recordID: recordID)
|
|
87
|
-
```
|
|
88
|
-
|
|
89
|
-
## CKQuery
|
|
90
|
-
|
|
91
|
-
Query records with NSPredicate. Supported: `==`, `!=`, `<`, `>`, `<=`, `>=`,
|
|
92
|
-
`BEGINSWITH`, `CONTAINS`, `IN`, `AND`, `NOT`, `BETWEEN`,
|
|
93
|
-
`distanceToLocation:fromLocation:`.
|
|
94
|
-
|
|
95
|
-
```swift
|
|
96
|
-
let predicate = NSPredicate(format: "title BEGINSWITH %@", "Meeting")
|
|
97
|
-
let query = CKQuery(recordType: "Note", predicate: predicate)
|
|
98
|
-
query.sortDescriptors = [NSSortDescriptor(key: "createdAt", ascending: false)]
|
|
99
|
-
|
|
100
|
-
let (results, _) = try await privateDB.records(matching: query)
|
|
101
|
-
for (_, result) in results {
|
|
102
|
-
let record = try result.get()
|
|
103
|
-
print(record["title"] as? String ?? "")
|
|
104
|
-
}
|
|
105
|
-
|
|
106
|
-
// Fetch all records of a type
|
|
107
|
-
let allQuery = CKQuery(recordType: "Note", predicate: NSPredicate(value: true))
|
|
108
|
-
|
|
109
|
-
// Full-text search across string fields
|
|
110
|
-
let searchQuery = CKQuery(
|
|
111
|
-
recordType: "Note",
|
|
112
|
-
predicate: NSPredicate(format: "self CONTAINS %@", "roadmap")
|
|
113
|
-
)
|
|
114
|
-
|
|
115
|
-
// Compound predicate
|
|
116
|
-
let compound = NSCompoundPredicate(andPredicateWithSubpredicates: [
|
|
117
|
-
NSPredicate(format: "createdAt > %@", cutoffDate as NSDate),
|
|
118
|
-
NSPredicate(format: "tags CONTAINS %@", "work")
|
|
119
|
-
])
|
|
120
|
-
```
|
|
121
|
-
|
|
122
|
-
## CKSubscription
|
|
123
|
-
|
|
124
|
-
Subscriptions trigger push notifications when records change server-side.
|
|
125
|
-
CloudKit auto-enables APNs -- no explicit push entitlement needed.
|
|
126
|
-
|
|
127
|
-
```swift
|
|
128
|
-
// Query subscription -- fires when matching records change
|
|
129
|
-
let subscription = CKQuerySubscription(
|
|
130
|
-
recordType: "Note",
|
|
131
|
-
predicate: NSPredicate(format: "tags CONTAINS %@", "urgent"),
|
|
132
|
-
subscriptionID: "urgent-notes",
|
|
133
|
-
options: [.firesOnRecordCreation, .firesOnRecordUpdate]
|
|
134
|
-
)
|
|
135
|
-
let notifInfo = CKSubscription.NotificationInfo()
|
|
136
|
-
notifInfo.shouldSendContentAvailable = true // silent push
|
|
137
|
-
subscription.notificationInfo = notifInfo
|
|
138
|
-
try await privateDB.save(subscription)
|
|
139
|
-
|
|
140
|
-
// Database subscription -- fires on any database change
|
|
141
|
-
let dbSub = CKDatabaseSubscription(subscriptionID: "private-db-changes")
|
|
142
|
-
dbSub.notificationInfo = notifInfo
|
|
143
|
-
try await privateDB.save(dbSub)
|
|
144
|
-
|
|
145
|
-
// Record zone subscription -- fires on changes within a zone
|
|
146
|
-
let zoneSub = CKRecordZoneSubscription(
|
|
147
|
-
zoneID: CKRecordZone.ID(zoneName: "NotesZone"),
|
|
148
|
-
subscriptionID: "notes-zone-changes"
|
|
149
|
-
)
|
|
150
|
-
zoneSub.notificationInfo = notifInfo
|
|
151
|
-
try await privateDB.save(zoneSub)
|
|
152
|
-
```
|
|
153
|
-
|
|
154
|
-
Handle in AppDelegate:
|
|
155
|
-
|
|
156
|
-
```swift
|
|
157
|
-
func application(
|
|
158
|
-
_ application: UIApplication,
|
|
159
|
-
didReceiveRemoteNotification userInfo: [AnyHashable: Any]
|
|
160
|
-
) async -> UIBackgroundFetchResult {
|
|
161
|
-
let notification = CKNotification(fromRemoteNotificationDictionary: userInfo)
|
|
162
|
-
guard notification?.subscriptionID == "private-db-changes" else { return .noData }
|
|
163
|
-
// Fetch changes using CKSyncEngine or CKFetchRecordZoneChangesOperation
|
|
164
|
-
return .newData
|
|
165
|
-
}
|
|
166
|
-
```
|
|
167
|
-
|
|
168
|
-
## CKSyncEngine (iOS 17+)
|
|
169
|
-
|
|
170
|
-
`CKSyncEngine` is the recommended sync approach. It handles scheduling,
|
|
171
|
-
transient error retries, change tokens, and push notifications automatically.
|
|
172
|
-
Works with private and shared databases only.
|
|
173
|
-
|
|
174
|
-
```swift
|
|
175
|
-
import CloudKit
|
|
176
|
-
|
|
177
|
-
final class SyncManager: CKSyncEngineDelegate {
|
|
178
|
-
let syncEngine: CKSyncEngine
|
|
179
|
-
|
|
180
|
-
init(container: CKContainer = .default()) {
|
|
181
|
-
let config = CKSyncEngine.Configuration(
|
|
182
|
-
database: container.privateCloudDatabase,
|
|
183
|
-
stateSerialization: Self.loadState(),
|
|
184
|
-
delegate: self
|
|
185
|
-
)
|
|
186
|
-
self.syncEngine = CKSyncEngine(config)
|
|
187
|
-
}
|
|
188
|
-
|
|
189
|
-
func handleEvent(_ event: CKSyncEngine.Event, syncEngine: CKSyncEngine) {
|
|
190
|
-
switch event {
|
|
191
|
-
case .stateUpdate(let update):
|
|
192
|
-
Self.saveState(update.stateSerialization)
|
|
193
|
-
case .accountChange(let change):
|
|
194
|
-
handleAccountChange(change)
|
|
195
|
-
case .fetchedRecordZoneChanges(let changes):
|
|
196
|
-
for mod in changes.modifications { processRemoteRecord(mod.record) }
|
|
197
|
-
for del in changes.deletions { processRemoteDeletion(del.recordID) }
|
|
198
|
-
case .sentRecordZoneChanges(let sent):
|
|
199
|
-
for saved in sent.savedRecords { markSynced(saved) }
|
|
200
|
-
for fail in sent.failedRecordSaves { handleSaveFailure(fail) }
|
|
201
|
-
default: break
|
|
202
|
-
}
|
|
203
|
-
}
|
|
204
|
-
|
|
205
|
-
func nextRecordZoneChangeBatch(
|
|
206
|
-
_ context: CKSyncEngine.SendChangesContext,
|
|
207
|
-
syncEngine: CKSyncEngine
|
|
208
|
-
) -> CKSyncEngine.RecordZoneChangeBatch? {
|
|
209
|
-
let pending = syncEngine.state.pendingRecordZoneChanges
|
|
210
|
-
return CKSyncEngine.RecordZoneChangeBatch(
|
|
211
|
-
pendingChanges: Array(pending)
|
|
212
|
-
) { recordID in self.recordToSend(for: recordID) }
|
|
213
|
-
}
|
|
214
|
-
}
|
|
215
|
-
|
|
216
|
-
// Schedule changes
|
|
217
|
-
let zoneID = CKRecordZone.ID(zoneName: "NotesZone")
|
|
218
|
-
let recordID = CKRecord.ID(recordName: noteID, zoneID: zoneID)
|
|
219
|
-
syncEngine.state.add(pendingRecordZoneChanges: [.saveRecord(recordID)])
|
|
220
|
-
|
|
221
|
-
// Trigger immediate sync (pull-to-refresh)
|
|
222
|
-
try await syncEngine.fetchChanges()
|
|
223
|
-
try await syncEngine.sendChanges()
|
|
224
|
-
```
|
|
225
|
-
|
|
226
|
-
**Key point**: persist `stateSerialization` across launches; the engine needs it
|
|
227
|
-
to resume from the correct change token.
|
|
228
|
-
|
|
229
|
-
## SwiftData + CloudKit
|
|
230
|
-
|
|
231
|
-
`ModelConfiguration` supports CloudKit sync. CloudKit models must use optional
|
|
232
|
-
properties and avoid unique constraints.
|
|
233
|
-
|
|
234
|
-
```swift
|
|
235
|
-
import SwiftData
|
|
236
|
-
|
|
237
|
-
@Model
|
|
238
|
-
class Note {
|
|
239
|
-
var title: String
|
|
240
|
-
var body: String?
|
|
241
|
-
var createdAt: Date?
|
|
242
|
-
@Attribute(.externalStorage) var imageData: Data?
|
|
243
|
-
|
|
244
|
-
init(title: String, body: String? = nil) {
|
|
245
|
-
self.title = title
|
|
246
|
-
self.body = body
|
|
247
|
-
self.createdAt = Date()
|
|
248
|
-
}
|
|
249
|
-
}
|
|
250
|
-
|
|
251
|
-
let config = ModelConfiguration(
|
|
252
|
-
"Notes",
|
|
253
|
-
cloudKitDatabase: .private("iCloud.com.example.app")
|
|
254
|
-
)
|
|
255
|
-
let container = try ModelContainer(for: Note.self, configurations: config)
|
|
256
|
-
```
|
|
257
|
-
|
|
258
|
-
**CloudKit model rules**: use optionals for all non-String properties; avoid
|
|
259
|
-
`#Unique`; keep models flat; use `@Attribute(.externalStorage)` for large data;
|
|
260
|
-
avoid complex relationship graphs.
|
|
261
|
-
|
|
262
|
-
## NSUbiquitousKeyValueStore
|
|
263
|
-
|
|
264
|
-
Simple key-value sync. Max 1024 keys, 1 MB total, 1 MB per value. Stores
|
|
265
|
-
locally when iCloud is unavailable.
|
|
266
|
-
|
|
267
|
-
```swift
|
|
268
|
-
let kvStore = NSUbiquitousKeyValueStore.default
|
|
269
|
-
|
|
270
|
-
// Write
|
|
271
|
-
kvStore.set("dark", forKey: "theme")
|
|
272
|
-
kvStore.set(14.0, forKey: "fontSize")
|
|
273
|
-
kvStore.set(true, forKey: "notificationsEnabled")
|
|
274
|
-
kvStore.synchronize()
|
|
275
|
-
|
|
276
|
-
// Read
|
|
277
|
-
let theme = kvStore.string(forKey: "theme") ?? "system"
|
|
278
|
-
|
|
279
|
-
// Observe external changes
|
|
280
|
-
NotificationCenter.default.addObserver(
|
|
281
|
-
forName: NSUbiquitousKeyValueStore.didChangeExternallyNotification,
|
|
282
|
-
object: kvStore, queue: .main
|
|
283
|
-
) { notification in
|
|
284
|
-
guard let userInfo = notification.userInfo,
|
|
285
|
-
let reason = userInfo[NSUbiquitousKeyValueStoreChangeReasonKey] as? Int,
|
|
286
|
-
let keys = userInfo[NSUbiquitousKeyValueStoreChangedKeysKey] as? [String]
|
|
287
|
-
else { return }
|
|
288
|
-
|
|
289
|
-
switch reason {
|
|
290
|
-
case NSUbiquitousKeyValueStoreServerChange:
|
|
291
|
-
for key in keys { applyRemoteChange(key: key) }
|
|
292
|
-
case NSUbiquitousKeyValueStoreInitialSyncChange:
|
|
293
|
-
reloadAllSettings()
|
|
294
|
-
case NSUbiquitousKeyValueStoreQuotaViolationChange:
|
|
295
|
-
handleQuotaExceeded()
|
|
296
|
-
default: break
|
|
297
|
-
}
|
|
298
|
-
}
|
|
299
|
-
```
|
|
300
|
-
|
|
301
|
-
## iCloud Drive File Sync
|
|
302
|
-
|
|
303
|
-
Use `FileManager` ubiquity APIs for document-level sync.
|
|
304
|
-
|
|
305
|
-
```swift
|
|
306
|
-
guard let ubiquityURL = FileManager.default.url(
|
|
307
|
-
forUbiquityContainerIdentifier: "iCloud.com.example.app"
|
|
308
|
-
) else { return } // iCloud not available
|
|
309
|
-
|
|
310
|
-
let docsURL = ubiquityURL.appendingPathComponent("Documents")
|
|
311
|
-
let cloudURL = docsURL.appendingPathComponent("report.pdf")
|
|
312
|
-
try FileManager.default.setUbiquitous(true, itemAt: localURL, destinationURL: cloudURL)
|
|
313
|
-
|
|
314
|
-
// Monitor iCloud files
|
|
315
|
-
let query = NSMetadataQuery()
|
|
316
|
-
query.predicate = NSPredicate(format: "%K LIKE '*.pdf'", NSMetadataItemFSNameKey)
|
|
317
|
-
query.searchScopes = [NSMetadataQueryUbiquitousDocumentsScope]
|
|
318
|
-
NotificationCenter.default.addObserver(
|
|
319
|
-
forName: .NSMetadataQueryDidFinishGathering, object: query, queue: .main
|
|
320
|
-
) { _ in
|
|
321
|
-
query.disableUpdates()
|
|
322
|
-
for item in query.results as? [NSMetadataItem] ?? [] {
|
|
323
|
-
let name = item.value(forAttribute: NSMetadataItemFSNameKey) as? String
|
|
324
|
-
let status = item.value(
|
|
325
|
-
forAttribute: NSMetadataUbiquitousItemDownloadingStatusKey) as? String
|
|
326
|
-
}
|
|
327
|
-
query.enableUpdates()
|
|
328
|
-
}
|
|
329
|
-
query.start()
|
|
330
|
-
```
|
|
331
|
-
|
|
332
|
-
## Account Status and Error Handling
|
|
333
|
-
|
|
334
|
-
Always check account status before sync. Listen for `.CKAccountChanged`.
|
|
335
|
-
|
|
336
|
-
```swift
|
|
337
|
-
func checkiCloudStatus() async throws -> CKAccountStatus {
|
|
338
|
-
let status = try await CKContainer.default().accountStatus()
|
|
339
|
-
switch status {
|
|
340
|
-
case .available: return status
|
|
341
|
-
case .noAccount: throw SyncError.noiCloudAccount
|
|
342
|
-
case .restricted: throw SyncError.restricted
|
|
343
|
-
case .temporarilyUnavailable: throw SyncError.temporarilyUnavailable
|
|
344
|
-
case .couldNotDetermine: throw SyncError.unknown
|
|
345
|
-
@unknown default: throw SyncError.unknown
|
|
346
|
-
}
|
|
347
|
-
}
|
|
348
|
-
```
|
|
349
|
-
|
|
350
|
-
### CKError Handling
|
|
351
|
-
|
|
352
|
-
| Error Code | Strategy |
|
|
353
|
-
|-----------|----------|
|
|
354
|
-
| `.networkFailure`, `.networkUnavailable` | Queue for retry when network returns |
|
|
355
|
-
| `.serverRecordChanged` | Three-way merge (see Conflict Resolution) |
|
|
356
|
-
| `.requestRateLimited`, `.zoneBusy`, `.serviceUnavailable` | Retry after `retryAfterSeconds` |
|
|
357
|
-
| `.quotaExceeded` | Notify user; reduce data usage |
|
|
358
|
-
| `.notAuthenticated` | Prompt iCloud sign-in |
|
|
359
|
-
| `.partialFailure` | Inspect `partialErrorsByItemID` per item |
|
|
360
|
-
| `.changeTokenExpired` | Reset token, refetch all changes |
|
|
361
|
-
| `.userDeletedZone` | Recreate zone and re-upload data |
|
|
362
|
-
|
|
363
|
-
```swift
|
|
364
|
-
func handleCloudKitError(_ error: Error) {
|
|
365
|
-
guard let ckError = error as? CKError else { return }
|
|
366
|
-
switch ckError.code {
|
|
367
|
-
case .networkFailure, .networkUnavailable:
|
|
368
|
-
scheduleRetryWhenOnline()
|
|
369
|
-
case .serverRecordChanged:
|
|
370
|
-
resolveConflict(ckError)
|
|
371
|
-
case .requestRateLimited, .zoneBusy, .serviceUnavailable:
|
|
372
|
-
let delay = ckError.retryAfterSeconds ?? 3.0
|
|
373
|
-
scheduleRetry(after: delay)
|
|
374
|
-
case .quotaExceeded:
|
|
375
|
-
notifyUserStorageFull()
|
|
376
|
-
case .partialFailure:
|
|
377
|
-
if let partial = ckError.partialErrorsByItemID {
|
|
378
|
-
for (_, itemError) in partial { handleCloudKitError(itemError) }
|
|
379
|
-
}
|
|
380
|
-
case .changeTokenExpired:
|
|
381
|
-
resetChangeToken()
|
|
382
|
-
case .userDeletedZone:
|
|
383
|
-
recreateZoneAndResync()
|
|
384
|
-
default: logError(ckError)
|
|
385
|
-
}
|
|
386
|
-
}
|
|
387
|
-
```
|
|
388
|
-
|
|
389
|
-
## Conflict Resolution
|
|
390
|
-
|
|
391
|
-
When saving a record that changed server-side, CloudKit returns
|
|
392
|
-
`.serverRecordChanged` with three record versions. Always merge into
|
|
393
|
-
`serverRecord` -- it has the correct change tag.
|
|
394
|
-
|
|
395
|
-
```swift
|
|
396
|
-
func resolveConflict(_ error: CKError) {
|
|
397
|
-
guard error.code == .serverRecordChanged,
|
|
398
|
-
let ancestor = error.ancestorRecord,
|
|
399
|
-
let client = error.clientRecord,
|
|
400
|
-
let server = error.serverRecord
|
|
401
|
-
else { return }
|
|
402
|
-
|
|
403
|
-
// Merge client changes into server record
|
|
404
|
-
for key in client.changedKeys() {
|
|
405
|
-
if server[key] == ancestor[key] {
|
|
406
|
-
server[key] = client[key] // Server unchanged, use client
|
|
407
|
-
} else if client[key] == ancestor[key] {
|
|
408
|
-
// Client unchanged, keep server (already there)
|
|
409
|
-
} else {
|
|
410
|
-
server[key] = mergeValues( // Both changed, custom merge
|
|
411
|
-
ancestor: ancestor[key], client: client[key], server: server[key])
|
|
412
|
-
}
|
|
413
|
-
}
|
|
414
|
-
|
|
415
|
-
Task { try await CKContainer.default().privateCloudDatabase.save(server) }
|
|
416
|
-
}
|
|
417
|
-
```
|
|
418
|
-
|
|
419
|
-
## Common Mistakes
|
|
420
|
-
|
|
421
|
-
**DON'T:** Perform sync operations without checking account status.
|
|
422
|
-
**DO:** Check `CKContainer.accountStatus()` first; handle `.noAccount`.
|
|
423
|
-
```swift
|
|
424
|
-
// WRONG
|
|
425
|
-
try await privateDB.save(record)
|
|
426
|
-
// CORRECT
|
|
427
|
-
guard try await CKContainer.default().accountStatus() == .available
|
|
428
|
-
else { throw SyncError.noiCloudAccount }
|
|
429
|
-
try await privateDB.save(record)
|
|
430
|
-
```
|
|
431
|
-
|
|
432
|
-
**DON'T:** Ignore `.serverRecordChanged` errors.
|
|
433
|
-
**DO:** Implement three-way merge using ancestor, client, and server records.
|
|
434
|
-
|
|
435
|
-
**DON'T:** Store user-specific data in the public database.
|
|
436
|
-
**DO:** Use private database for personal data; public only for app-wide content.
|
|
437
|
-
|
|
438
|
-
**DON'T:** Assume data is available immediately after save.
|
|
439
|
-
**DO:** Update local cache optimistically and reconcile on fetch.
|
|
440
|
-
|
|
441
|
-
**DON'T:** Poll for changes on a timer.
|
|
442
|
-
**DO:** Use `CKDatabaseSubscription` or `CKSyncEngine` for push-based sync.
|
|
443
|
-
```swift
|
|
444
|
-
// WRONG
|
|
445
|
-
Timer.scheduledTimer(withTimeInterval: 30, repeats: true) { _ in fetchAll() }
|
|
446
|
-
// CORRECT
|
|
447
|
-
let sub = CKDatabaseSubscription(subscriptionID: "db-changes")
|
|
448
|
-
sub.notificationInfo = CKSubscription.NotificationInfo()
|
|
449
|
-
sub.notificationInfo?.shouldSendContentAvailable = true
|
|
450
|
-
try await privateDB.save(sub)
|
|
451
|
-
```
|
|
452
|
-
|
|
453
|
-
**DON'T:** Retry immediately on rate limiting.
|
|
454
|
-
**DO:** Use `CKError.retryAfterSeconds` to wait the required duration.
|
|
455
|
-
|
|
456
|
-
**DON'T:** Merge conflict changes into `clientRecord`.
|
|
457
|
-
**DO:** Always merge into `serverRecord` -- it has the correct change tag.
|
|
458
|
-
|
|
459
|
-
**DON'T:** Pass nil change token on every fetch.
|
|
460
|
-
**DO:** Persist change tokens to disk and supply them on subsequent fetches.
|
|
461
|
-
|
|
462
|
-
## Review Checklist
|
|
463
|
-
|
|
464
|
-
- [ ] iCloud + CloudKit capability enabled in Signing & Capabilities
|
|
465
|
-
- [ ] Account status checked before sync; `.noAccount` handled gracefully
|
|
466
|
-
- [ ] Private database used for user data; public only for shared content
|
|
467
|
-
- [ ] `CKError.serverRecordChanged` handled with three-way merge into `serverRecord`
|
|
468
|
-
- [ ] Network failures queued for retry; `retryAfterSeconds` respected
|
|
469
|
-
- [ ] `CKDatabaseSubscription` or `CKSyncEngine` used for push-based sync
|
|
470
|
-
- [ ] Change tokens persisted to disk; `changeTokenExpired` resets and refetches
|
|
471
|
-
- [ ] `.partialFailure` errors inspected per-item via `partialErrorsByItemID`
|
|
472
|
-
- [ ] `.userDeletedZone` handled by recreating zone and resyncing
|
|
473
|
-
- [ ] SwiftData CloudKit models use optionals, no `#Unique`, `.externalStorage` for large data
|
|
474
|
-
- [ ] `NSUbiquitousKeyValueStore.didChangeExternallyNotification` observed
|
|
475
|
-
- [ ] Sensitive data uses `encryptedValues` on CKRecord (not plain fields)
|
|
476
|
-
- [ ] `CKSyncEngine` state serialization persisted across launches (iOS 17+)
|
|
477
|
-
|
|
478
|
-
## References
|
|
479
|
-
|
|
480
|
-
- See `references/cloudkit-patterns.md` for CKFetchRecordZoneChangesOperation
|
|
481
|
-
incremental sync, CKShare collaboration, record zone management, CKAsset
|
|
482
|
-
file storage, batch operations, and CloudKit Dashboard usage.
|
|
483
|
-
- [CloudKit Framework](https://sosumi.ai/documentation/cloudkit)
|
|
484
|
-
- [CKContainer](https://sosumi.ai/documentation/cloudkit/ckcontainer)
|
|
485
|
-
- [CKRecord](https://sosumi.ai/documentation/cloudkit/ckrecord)
|
|
486
|
-
- [CKQuery](https://sosumi.ai/documentation/cloudkit/ckquery)
|
|
487
|
-
- [CKSubscription](https://sosumi.ai/documentation/cloudkit/cksubscription)
|
|
488
|
-
- [CKSyncEngine](https://sosumi.ai/documentation/cloudkit/cksyncengine)
|
|
489
|
-
- [CKShare](https://sosumi.ai/documentation/cloudkit/ckshare)
|
|
490
|
-
- [CKError](https://sosumi.ai/documentation/cloudkit/ckerror)
|
|
491
|
-
- [NSUbiquitousKeyValueStore](https://sosumi.ai/documentation/foundation/nsubiquitouskeyvaluestore)
|
|
492
|
-
- [CKAsset](https://sosumi.ai/documentation/cloudkit/ckasset)
|
|
1
|
+
---
|
|
2
|
+
name: cloudkit-sync
|
|
3
|
+
description: "Implement, review, or improve CloudKit and iCloud sync in iOS/macOS apps. Use when working with CKContainer, CKRecord, CKQuery, CKSubscription, CKSyncEngine, CKShare, NSUbiquitousKeyValueStore, or iCloud Drive file coordination; when syncing SwiftData models via ModelConfiguration with cloudKitDatabase; when handling CKError codes for conflict resolution, network failures, or quota limits; or when checking iCloud account status before performing sync operations."
|
|
4
|
+
---
|
|
5
|
+
|
|
6
|
+
# CloudKit and iCloud Sync
|
|
7
|
+
|
|
8
|
+
Sync data across devices using CloudKit, iCloud key-value storage, and iCloud
|
|
9
|
+
Drive. Covers container setup, record CRUD, queries, subscriptions, CKSyncEngine,
|
|
10
|
+
SwiftData integration, conflict resolution, and error handling. Targets iOS 26+
|
|
11
|
+
with Swift 6.2; older availability noted where relevant.
|
|
12
|
+
|
|
13
|
+
## Contents
|
|
14
|
+
|
|
15
|
+
- [Container and Database Setup](#container-and-database-setup)
|
|
16
|
+
- [CKRecord CRUD](#ckrecord-crud)
|
|
17
|
+
- [CKQuery](#ckquery)
|
|
18
|
+
- [CKSubscription](#cksubscription)
|
|
19
|
+
- [CKSyncEngine (iOS 17+)](#cksyncengine-ios-17)
|
|
20
|
+
- [SwiftData + CloudKit](#swiftdata--cloudkit)
|
|
21
|
+
- [NSUbiquitousKeyValueStore](#nsubiquitouskeyvaluestore)
|
|
22
|
+
- [iCloud Drive File Sync](#icloud-drive-file-sync)
|
|
23
|
+
- [Account Status and Error Handling](#account-status-and-error-handling)
|
|
24
|
+
- [Conflict Resolution](#conflict-resolution)
|
|
25
|
+
- [Common Mistakes](#common-mistakes)
|
|
26
|
+
- [Review Checklist](#review-checklist)
|
|
27
|
+
- [References](#references)
|
|
28
|
+
|
|
29
|
+
## Container and Database Setup
|
|
30
|
+
|
|
31
|
+
Enable iCloud + CloudKit in Signing & Capabilities. A container provides
|
|
32
|
+
three databases:
|
|
33
|
+
|
|
34
|
+
| Database | Scope | Requires iCloud | Storage Quota |
|
|
35
|
+
|----------|-------|-----------------|---------------|
|
|
36
|
+
| Public | All users | Read: No, Write: Yes | App quota |
|
|
37
|
+
| Private | Current user | Yes | User quota |
|
|
38
|
+
| Shared | Shared records | Yes | Owner quota |
|
|
39
|
+
|
|
40
|
+
```swift
|
|
41
|
+
import CloudKit
|
|
42
|
+
|
|
43
|
+
let container = CKContainer.default()
|
|
44
|
+
// Or named: CKContainer(identifier: "iCloud.com.example.app")
|
|
45
|
+
|
|
46
|
+
let publicDB = container.publicCloudDatabase
|
|
47
|
+
let privateDB = container.privateCloudDatabase
|
|
48
|
+
let sharedDB = container.sharedCloudDatabase
|
|
49
|
+
```
|
|
50
|
+
|
|
51
|
+
## CKRecord CRUD
|
|
52
|
+
|
|
53
|
+
Records are key-value pairs. Max 1 MB per record (excluding CKAsset data).
|
|
54
|
+
|
|
55
|
+
```swift
|
|
56
|
+
// CREATE
|
|
57
|
+
let record = CKRecord(recordType: "Note")
|
|
58
|
+
record["title"] = "Meeting Notes" as CKRecordValue
|
|
59
|
+
record["body"] = "Discussed Q3 roadmap" as CKRecordValue
|
|
60
|
+
record["createdAt"] = Date() as CKRecordValue
|
|
61
|
+
record["tags"] = ["work", "planning"] as CKRecordValue
|
|
62
|
+
let saved = try await privateDB.save(record)
|
|
63
|
+
|
|
64
|
+
// FETCH by ID
|
|
65
|
+
let recordID = CKRecord.ID(recordName: "unique-id-123")
|
|
66
|
+
let fetched = try await privateDB.record(for: recordID)
|
|
67
|
+
|
|
68
|
+
// UPDATE -- fetch first, modify, then save
|
|
69
|
+
fetched["title"] = "Updated Title" as CKRecordValue
|
|
70
|
+
let updated = try await privateDB.save(fetched)
|
|
71
|
+
|
|
72
|
+
// DELETE
|
|
73
|
+
try await privateDB.deleteRecord(withID: recordID)
|
|
74
|
+
```
|
|
75
|
+
|
|
76
|
+
### Custom Record Zones (Private/Shared Only)
|
|
77
|
+
|
|
78
|
+
Custom zones support atomic commits, change tracking, and sharing.
|
|
79
|
+
|
|
80
|
+
```swift
|
|
81
|
+
let zoneID = CKRecordZone.ID(zoneName: "NotesZone")
|
|
82
|
+
let zone = CKRecordZone(zoneID: zoneID)
|
|
83
|
+
try await privateDB.save(zone)
|
|
84
|
+
|
|
85
|
+
let recordID = CKRecord.ID(recordName: UUID().uuidString, zoneID: zoneID)
|
|
86
|
+
let record = CKRecord(recordType: "Note", recordID: recordID)
|
|
87
|
+
```
|
|
88
|
+
|
|
89
|
+
## CKQuery
|
|
90
|
+
|
|
91
|
+
Query records with NSPredicate. Supported: `==`, `!=`, `<`, `>`, `<=`, `>=`,
|
|
92
|
+
`BEGINSWITH`, `CONTAINS`, `IN`, `AND`, `NOT`, `BETWEEN`,
|
|
93
|
+
`distanceToLocation:fromLocation:`.
|
|
94
|
+
|
|
95
|
+
```swift
|
|
96
|
+
let predicate = NSPredicate(format: "title BEGINSWITH %@", "Meeting")
|
|
97
|
+
let query = CKQuery(recordType: "Note", predicate: predicate)
|
|
98
|
+
query.sortDescriptors = [NSSortDescriptor(key: "createdAt", ascending: false)]
|
|
99
|
+
|
|
100
|
+
let (results, _) = try await privateDB.records(matching: query)
|
|
101
|
+
for (_, result) in results {
|
|
102
|
+
let record = try result.get()
|
|
103
|
+
print(record["title"] as? String ?? "")
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
// Fetch all records of a type
|
|
107
|
+
let allQuery = CKQuery(recordType: "Note", predicate: NSPredicate(value: true))
|
|
108
|
+
|
|
109
|
+
// Full-text search across string fields
|
|
110
|
+
let searchQuery = CKQuery(
|
|
111
|
+
recordType: "Note",
|
|
112
|
+
predicate: NSPredicate(format: "self CONTAINS %@", "roadmap")
|
|
113
|
+
)
|
|
114
|
+
|
|
115
|
+
// Compound predicate
|
|
116
|
+
let compound = NSCompoundPredicate(andPredicateWithSubpredicates: [
|
|
117
|
+
NSPredicate(format: "createdAt > %@", cutoffDate as NSDate),
|
|
118
|
+
NSPredicate(format: "tags CONTAINS %@", "work")
|
|
119
|
+
])
|
|
120
|
+
```
|
|
121
|
+
|
|
122
|
+
## CKSubscription
|
|
123
|
+
|
|
124
|
+
Subscriptions trigger push notifications when records change server-side.
|
|
125
|
+
CloudKit auto-enables APNs -- no explicit push entitlement needed.
|
|
126
|
+
|
|
127
|
+
```swift
|
|
128
|
+
// Query subscription -- fires when matching records change
|
|
129
|
+
let subscription = CKQuerySubscription(
|
|
130
|
+
recordType: "Note",
|
|
131
|
+
predicate: NSPredicate(format: "tags CONTAINS %@", "urgent"),
|
|
132
|
+
subscriptionID: "urgent-notes",
|
|
133
|
+
options: [.firesOnRecordCreation, .firesOnRecordUpdate]
|
|
134
|
+
)
|
|
135
|
+
let notifInfo = CKSubscription.NotificationInfo()
|
|
136
|
+
notifInfo.shouldSendContentAvailable = true // silent push
|
|
137
|
+
subscription.notificationInfo = notifInfo
|
|
138
|
+
try await privateDB.save(subscription)
|
|
139
|
+
|
|
140
|
+
// Database subscription -- fires on any database change
|
|
141
|
+
let dbSub = CKDatabaseSubscription(subscriptionID: "private-db-changes")
|
|
142
|
+
dbSub.notificationInfo = notifInfo
|
|
143
|
+
try await privateDB.save(dbSub)
|
|
144
|
+
|
|
145
|
+
// Record zone subscription -- fires on changes within a zone
|
|
146
|
+
let zoneSub = CKRecordZoneSubscription(
|
|
147
|
+
zoneID: CKRecordZone.ID(zoneName: "NotesZone"),
|
|
148
|
+
subscriptionID: "notes-zone-changes"
|
|
149
|
+
)
|
|
150
|
+
zoneSub.notificationInfo = notifInfo
|
|
151
|
+
try await privateDB.save(zoneSub)
|
|
152
|
+
```
|
|
153
|
+
|
|
154
|
+
Handle in AppDelegate:
|
|
155
|
+
|
|
156
|
+
```swift
|
|
157
|
+
func application(
|
|
158
|
+
_ application: UIApplication,
|
|
159
|
+
didReceiveRemoteNotification userInfo: [AnyHashable: Any]
|
|
160
|
+
) async -> UIBackgroundFetchResult {
|
|
161
|
+
let notification = CKNotification(fromRemoteNotificationDictionary: userInfo)
|
|
162
|
+
guard notification?.subscriptionID == "private-db-changes" else { return .noData }
|
|
163
|
+
// Fetch changes using CKSyncEngine or CKFetchRecordZoneChangesOperation
|
|
164
|
+
return .newData
|
|
165
|
+
}
|
|
166
|
+
```
|
|
167
|
+
|
|
168
|
+
## CKSyncEngine (iOS 17+)
|
|
169
|
+
|
|
170
|
+
`CKSyncEngine` is the recommended sync approach. It handles scheduling,
|
|
171
|
+
transient error retries, change tokens, and push notifications automatically.
|
|
172
|
+
Works with private and shared databases only.
|
|
173
|
+
|
|
174
|
+
```swift
|
|
175
|
+
import CloudKit
|
|
176
|
+
|
|
177
|
+
final class SyncManager: CKSyncEngineDelegate {
|
|
178
|
+
let syncEngine: CKSyncEngine
|
|
179
|
+
|
|
180
|
+
init(container: CKContainer = .default()) {
|
|
181
|
+
let config = CKSyncEngine.Configuration(
|
|
182
|
+
database: container.privateCloudDatabase,
|
|
183
|
+
stateSerialization: Self.loadState(),
|
|
184
|
+
delegate: self
|
|
185
|
+
)
|
|
186
|
+
self.syncEngine = CKSyncEngine(config)
|
|
187
|
+
}
|
|
188
|
+
|
|
189
|
+
func handleEvent(_ event: CKSyncEngine.Event, syncEngine: CKSyncEngine) {
|
|
190
|
+
switch event {
|
|
191
|
+
case .stateUpdate(let update):
|
|
192
|
+
Self.saveState(update.stateSerialization)
|
|
193
|
+
case .accountChange(let change):
|
|
194
|
+
handleAccountChange(change)
|
|
195
|
+
case .fetchedRecordZoneChanges(let changes):
|
|
196
|
+
for mod in changes.modifications { processRemoteRecord(mod.record) }
|
|
197
|
+
for del in changes.deletions { processRemoteDeletion(del.recordID) }
|
|
198
|
+
case .sentRecordZoneChanges(let sent):
|
|
199
|
+
for saved in sent.savedRecords { markSynced(saved) }
|
|
200
|
+
for fail in sent.failedRecordSaves { handleSaveFailure(fail) }
|
|
201
|
+
default: break
|
|
202
|
+
}
|
|
203
|
+
}
|
|
204
|
+
|
|
205
|
+
func nextRecordZoneChangeBatch(
|
|
206
|
+
_ context: CKSyncEngine.SendChangesContext,
|
|
207
|
+
syncEngine: CKSyncEngine
|
|
208
|
+
) -> CKSyncEngine.RecordZoneChangeBatch? {
|
|
209
|
+
let pending = syncEngine.state.pendingRecordZoneChanges
|
|
210
|
+
return CKSyncEngine.RecordZoneChangeBatch(
|
|
211
|
+
pendingChanges: Array(pending)
|
|
212
|
+
) { recordID in self.recordToSend(for: recordID) }
|
|
213
|
+
}
|
|
214
|
+
}
|
|
215
|
+
|
|
216
|
+
// Schedule changes
|
|
217
|
+
let zoneID = CKRecordZone.ID(zoneName: "NotesZone")
|
|
218
|
+
let recordID = CKRecord.ID(recordName: noteID, zoneID: zoneID)
|
|
219
|
+
syncEngine.state.add(pendingRecordZoneChanges: [.saveRecord(recordID)])
|
|
220
|
+
|
|
221
|
+
// Trigger immediate sync (pull-to-refresh)
|
|
222
|
+
try await syncEngine.fetchChanges()
|
|
223
|
+
try await syncEngine.sendChanges()
|
|
224
|
+
```
|
|
225
|
+
|
|
226
|
+
**Key point**: persist `stateSerialization` across launches; the engine needs it
|
|
227
|
+
to resume from the correct change token.
|
|
228
|
+
|
|
229
|
+
## SwiftData + CloudKit
|
|
230
|
+
|
|
231
|
+
`ModelConfiguration` supports CloudKit sync. CloudKit models must use optional
|
|
232
|
+
properties and avoid unique constraints.
|
|
233
|
+
|
|
234
|
+
```swift
|
|
235
|
+
import SwiftData
|
|
236
|
+
|
|
237
|
+
@Model
|
|
238
|
+
class Note {
|
|
239
|
+
var title: String
|
|
240
|
+
var body: String?
|
|
241
|
+
var createdAt: Date?
|
|
242
|
+
@Attribute(.externalStorage) var imageData: Data?
|
|
243
|
+
|
|
244
|
+
init(title: String, body: String? = nil) {
|
|
245
|
+
self.title = title
|
|
246
|
+
self.body = body
|
|
247
|
+
self.createdAt = Date()
|
|
248
|
+
}
|
|
249
|
+
}
|
|
250
|
+
|
|
251
|
+
let config = ModelConfiguration(
|
|
252
|
+
"Notes",
|
|
253
|
+
cloudKitDatabase: .private("iCloud.com.example.app")
|
|
254
|
+
)
|
|
255
|
+
let container = try ModelContainer(for: Note.self, configurations: config)
|
|
256
|
+
```
|
|
257
|
+
|
|
258
|
+
**CloudKit model rules**: use optionals for all non-String properties; avoid
|
|
259
|
+
`#Unique`; keep models flat; use `@Attribute(.externalStorage)` for large data;
|
|
260
|
+
avoid complex relationship graphs.
|
|
261
|
+
|
|
262
|
+
## NSUbiquitousKeyValueStore
|
|
263
|
+
|
|
264
|
+
Simple key-value sync. Max 1024 keys, 1 MB total, 1 MB per value. Stores
|
|
265
|
+
locally when iCloud is unavailable.
|
|
266
|
+
|
|
267
|
+
```swift
|
|
268
|
+
let kvStore = NSUbiquitousKeyValueStore.default
|
|
269
|
+
|
|
270
|
+
// Write
|
|
271
|
+
kvStore.set("dark", forKey: "theme")
|
|
272
|
+
kvStore.set(14.0, forKey: "fontSize")
|
|
273
|
+
kvStore.set(true, forKey: "notificationsEnabled")
|
|
274
|
+
kvStore.synchronize()
|
|
275
|
+
|
|
276
|
+
// Read
|
|
277
|
+
let theme = kvStore.string(forKey: "theme") ?? "system"
|
|
278
|
+
|
|
279
|
+
// Observe external changes
|
|
280
|
+
NotificationCenter.default.addObserver(
|
|
281
|
+
forName: NSUbiquitousKeyValueStore.didChangeExternallyNotification,
|
|
282
|
+
object: kvStore, queue: .main
|
|
283
|
+
) { notification in
|
|
284
|
+
guard let userInfo = notification.userInfo,
|
|
285
|
+
let reason = userInfo[NSUbiquitousKeyValueStoreChangeReasonKey] as? Int,
|
|
286
|
+
let keys = userInfo[NSUbiquitousKeyValueStoreChangedKeysKey] as? [String]
|
|
287
|
+
else { return }
|
|
288
|
+
|
|
289
|
+
switch reason {
|
|
290
|
+
case NSUbiquitousKeyValueStoreServerChange:
|
|
291
|
+
for key in keys { applyRemoteChange(key: key) }
|
|
292
|
+
case NSUbiquitousKeyValueStoreInitialSyncChange:
|
|
293
|
+
reloadAllSettings()
|
|
294
|
+
case NSUbiquitousKeyValueStoreQuotaViolationChange:
|
|
295
|
+
handleQuotaExceeded()
|
|
296
|
+
default: break
|
|
297
|
+
}
|
|
298
|
+
}
|
|
299
|
+
```
|
|
300
|
+
|
|
301
|
+
## iCloud Drive File Sync
|
|
302
|
+
|
|
303
|
+
Use `FileManager` ubiquity APIs for document-level sync.
|
|
304
|
+
|
|
305
|
+
```swift
|
|
306
|
+
guard let ubiquityURL = FileManager.default.url(
|
|
307
|
+
forUbiquityContainerIdentifier: "iCloud.com.example.app"
|
|
308
|
+
) else { return } // iCloud not available
|
|
309
|
+
|
|
310
|
+
let docsURL = ubiquityURL.appendingPathComponent("Documents")
|
|
311
|
+
let cloudURL = docsURL.appendingPathComponent("report.pdf")
|
|
312
|
+
try FileManager.default.setUbiquitous(true, itemAt: localURL, destinationURL: cloudURL)
|
|
313
|
+
|
|
314
|
+
// Monitor iCloud files
|
|
315
|
+
let query = NSMetadataQuery()
|
|
316
|
+
query.predicate = NSPredicate(format: "%K LIKE '*.pdf'", NSMetadataItemFSNameKey)
|
|
317
|
+
query.searchScopes = [NSMetadataQueryUbiquitousDocumentsScope]
|
|
318
|
+
NotificationCenter.default.addObserver(
|
|
319
|
+
forName: .NSMetadataQueryDidFinishGathering, object: query, queue: .main
|
|
320
|
+
) { _ in
|
|
321
|
+
query.disableUpdates()
|
|
322
|
+
for item in query.results as? [NSMetadataItem] ?? [] {
|
|
323
|
+
let name = item.value(forAttribute: NSMetadataItemFSNameKey) as? String
|
|
324
|
+
let status = item.value(
|
|
325
|
+
forAttribute: NSMetadataUbiquitousItemDownloadingStatusKey) as? String
|
|
326
|
+
}
|
|
327
|
+
query.enableUpdates()
|
|
328
|
+
}
|
|
329
|
+
query.start()
|
|
330
|
+
```
|
|
331
|
+
|
|
332
|
+
## Account Status and Error Handling
|
|
333
|
+
|
|
334
|
+
Always check account status before sync. Listen for `.CKAccountChanged`.
|
|
335
|
+
|
|
336
|
+
```swift
|
|
337
|
+
func checkiCloudStatus() async throws -> CKAccountStatus {
|
|
338
|
+
let status = try await CKContainer.default().accountStatus()
|
|
339
|
+
switch status {
|
|
340
|
+
case .available: return status
|
|
341
|
+
case .noAccount: throw SyncError.noiCloudAccount
|
|
342
|
+
case .restricted: throw SyncError.restricted
|
|
343
|
+
case .temporarilyUnavailable: throw SyncError.temporarilyUnavailable
|
|
344
|
+
case .couldNotDetermine: throw SyncError.unknown
|
|
345
|
+
@unknown default: throw SyncError.unknown
|
|
346
|
+
}
|
|
347
|
+
}
|
|
348
|
+
```
|
|
349
|
+
|
|
350
|
+
### CKError Handling
|
|
351
|
+
|
|
352
|
+
| Error Code | Strategy |
|
|
353
|
+
|-----------|----------|
|
|
354
|
+
| `.networkFailure`, `.networkUnavailable` | Queue for retry when network returns |
|
|
355
|
+
| `.serverRecordChanged` | Three-way merge (see Conflict Resolution) |
|
|
356
|
+
| `.requestRateLimited`, `.zoneBusy`, `.serviceUnavailable` | Retry after `retryAfterSeconds` |
|
|
357
|
+
| `.quotaExceeded` | Notify user; reduce data usage |
|
|
358
|
+
| `.notAuthenticated` | Prompt iCloud sign-in |
|
|
359
|
+
| `.partialFailure` | Inspect `partialErrorsByItemID` per item |
|
|
360
|
+
| `.changeTokenExpired` | Reset token, refetch all changes |
|
|
361
|
+
| `.userDeletedZone` | Recreate zone and re-upload data |
|
|
362
|
+
|
|
363
|
+
```swift
|
|
364
|
+
func handleCloudKitError(_ error: Error) {
|
|
365
|
+
guard let ckError = error as? CKError else { return }
|
|
366
|
+
switch ckError.code {
|
|
367
|
+
case .networkFailure, .networkUnavailable:
|
|
368
|
+
scheduleRetryWhenOnline()
|
|
369
|
+
case .serverRecordChanged:
|
|
370
|
+
resolveConflict(ckError)
|
|
371
|
+
case .requestRateLimited, .zoneBusy, .serviceUnavailable:
|
|
372
|
+
let delay = ckError.retryAfterSeconds ?? 3.0
|
|
373
|
+
scheduleRetry(after: delay)
|
|
374
|
+
case .quotaExceeded:
|
|
375
|
+
notifyUserStorageFull()
|
|
376
|
+
case .partialFailure:
|
|
377
|
+
if let partial = ckError.partialErrorsByItemID {
|
|
378
|
+
for (_, itemError) in partial { handleCloudKitError(itemError) }
|
|
379
|
+
}
|
|
380
|
+
case .changeTokenExpired:
|
|
381
|
+
resetChangeToken()
|
|
382
|
+
case .userDeletedZone:
|
|
383
|
+
recreateZoneAndResync()
|
|
384
|
+
default: logError(ckError)
|
|
385
|
+
}
|
|
386
|
+
}
|
|
387
|
+
```
|
|
388
|
+
|
|
389
|
+
## Conflict Resolution
|
|
390
|
+
|
|
391
|
+
When saving a record that changed server-side, CloudKit returns
|
|
392
|
+
`.serverRecordChanged` with three record versions. Always merge into
|
|
393
|
+
`serverRecord` -- it has the correct change tag.
|
|
394
|
+
|
|
395
|
+
```swift
|
|
396
|
+
func resolveConflict(_ error: CKError) {
|
|
397
|
+
guard error.code == .serverRecordChanged,
|
|
398
|
+
let ancestor = error.ancestorRecord,
|
|
399
|
+
let client = error.clientRecord,
|
|
400
|
+
let server = error.serverRecord
|
|
401
|
+
else { return }
|
|
402
|
+
|
|
403
|
+
// Merge client changes into server record
|
|
404
|
+
for key in client.changedKeys() {
|
|
405
|
+
if server[key] == ancestor[key] {
|
|
406
|
+
server[key] = client[key] // Server unchanged, use client
|
|
407
|
+
} else if client[key] == ancestor[key] {
|
|
408
|
+
// Client unchanged, keep server (already there)
|
|
409
|
+
} else {
|
|
410
|
+
server[key] = mergeValues( // Both changed, custom merge
|
|
411
|
+
ancestor: ancestor[key], client: client[key], server: server[key])
|
|
412
|
+
}
|
|
413
|
+
}
|
|
414
|
+
|
|
415
|
+
Task { try await CKContainer.default().privateCloudDatabase.save(server) }
|
|
416
|
+
}
|
|
417
|
+
```
|
|
418
|
+
|
|
419
|
+
## Common Mistakes
|
|
420
|
+
|
|
421
|
+
**DON'T:** Perform sync operations without checking account status.
|
|
422
|
+
**DO:** Check `CKContainer.accountStatus()` first; handle `.noAccount`.
|
|
423
|
+
```swift
|
|
424
|
+
// WRONG
|
|
425
|
+
try await privateDB.save(record)
|
|
426
|
+
// CORRECT
|
|
427
|
+
guard try await CKContainer.default().accountStatus() == .available
|
|
428
|
+
else { throw SyncError.noiCloudAccount }
|
|
429
|
+
try await privateDB.save(record)
|
|
430
|
+
```
|
|
431
|
+
|
|
432
|
+
**DON'T:** Ignore `.serverRecordChanged` errors.
|
|
433
|
+
**DO:** Implement three-way merge using ancestor, client, and server records.
|
|
434
|
+
|
|
435
|
+
**DON'T:** Store user-specific data in the public database.
|
|
436
|
+
**DO:** Use private database for personal data; public only for app-wide content.
|
|
437
|
+
|
|
438
|
+
**DON'T:** Assume data is available immediately after save.
|
|
439
|
+
**DO:** Update local cache optimistically and reconcile on fetch.
|
|
440
|
+
|
|
441
|
+
**DON'T:** Poll for changes on a timer.
|
|
442
|
+
**DO:** Use `CKDatabaseSubscription` or `CKSyncEngine` for push-based sync.
|
|
443
|
+
```swift
|
|
444
|
+
// WRONG
|
|
445
|
+
Timer.scheduledTimer(withTimeInterval: 30, repeats: true) { _ in fetchAll() }
|
|
446
|
+
// CORRECT
|
|
447
|
+
let sub = CKDatabaseSubscription(subscriptionID: "db-changes")
|
|
448
|
+
sub.notificationInfo = CKSubscription.NotificationInfo()
|
|
449
|
+
sub.notificationInfo?.shouldSendContentAvailable = true
|
|
450
|
+
try await privateDB.save(sub)
|
|
451
|
+
```
|
|
452
|
+
|
|
453
|
+
**DON'T:** Retry immediately on rate limiting.
|
|
454
|
+
**DO:** Use `CKError.retryAfterSeconds` to wait the required duration.
|
|
455
|
+
|
|
456
|
+
**DON'T:** Merge conflict changes into `clientRecord`.
|
|
457
|
+
**DO:** Always merge into `serverRecord` -- it has the correct change tag.
|
|
458
|
+
|
|
459
|
+
**DON'T:** Pass nil change token on every fetch.
|
|
460
|
+
**DO:** Persist change tokens to disk and supply them on subsequent fetches.
|
|
461
|
+
|
|
462
|
+
## Review Checklist
|
|
463
|
+
|
|
464
|
+
- [ ] iCloud + CloudKit capability enabled in Signing & Capabilities
|
|
465
|
+
- [ ] Account status checked before sync; `.noAccount` handled gracefully
|
|
466
|
+
- [ ] Private database used for user data; public only for shared content
|
|
467
|
+
- [ ] `CKError.serverRecordChanged` handled with three-way merge into `serverRecord`
|
|
468
|
+
- [ ] Network failures queued for retry; `retryAfterSeconds` respected
|
|
469
|
+
- [ ] `CKDatabaseSubscription` or `CKSyncEngine` used for push-based sync
|
|
470
|
+
- [ ] Change tokens persisted to disk; `changeTokenExpired` resets and refetches
|
|
471
|
+
- [ ] `.partialFailure` errors inspected per-item via `partialErrorsByItemID`
|
|
472
|
+
- [ ] `.userDeletedZone` handled by recreating zone and resyncing
|
|
473
|
+
- [ ] SwiftData CloudKit models use optionals, no `#Unique`, `.externalStorage` for large data
|
|
474
|
+
- [ ] `NSUbiquitousKeyValueStore.didChangeExternallyNotification` observed
|
|
475
|
+
- [ ] Sensitive data uses `encryptedValues` on CKRecord (not plain fields)
|
|
476
|
+
- [ ] `CKSyncEngine` state serialization persisted across launches (iOS 17+)
|
|
477
|
+
|
|
478
|
+
## References
|
|
479
|
+
|
|
480
|
+
- See `references/cloudkit-patterns.md` for CKFetchRecordZoneChangesOperation
|
|
481
|
+
incremental sync, CKShare collaboration, record zone management, CKAsset
|
|
482
|
+
file storage, batch operations, and CloudKit Dashboard usage.
|
|
483
|
+
- [CloudKit Framework](https://sosumi.ai/documentation/cloudkit)
|
|
484
|
+
- [CKContainer](https://sosumi.ai/documentation/cloudkit/ckcontainer)
|
|
485
|
+
- [CKRecord](https://sosumi.ai/documentation/cloudkit/ckrecord)
|
|
486
|
+
- [CKQuery](https://sosumi.ai/documentation/cloudkit/ckquery)
|
|
487
|
+
- [CKSubscription](https://sosumi.ai/documentation/cloudkit/cksubscription)
|
|
488
|
+
- [CKSyncEngine](https://sosumi.ai/documentation/cloudkit/cksyncengine)
|
|
489
|
+
- [CKShare](https://sosumi.ai/documentation/cloudkit/ckshare)
|
|
490
|
+
- [CKError](https://sosumi.ai/documentation/cloudkit/ckerror)
|
|
491
|
+
- [NSUbiquitousKeyValueStore](https://sosumi.ai/documentation/foundation/nsubiquitouskeyvaluestore)
|
|
492
|
+
- [CKAsset](https://sosumi.ai/documentation/cloudkit/ckasset)
|