@devo-bmad-custom/agent-orchestration 1.0.2 → 1.0.4
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 +97 -47
- 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,504 +1,504 @@
|
|
|
1
|
-
# Swift Testing Patterns Reference
|
|
2
|
-
|
|
3
|
-
## Contents
|
|
4
|
-
- Basic Tests and Traits
|
|
5
|
-
- Expectations and Requirements
|
|
6
|
-
- Suite Organization
|
|
7
|
-
- Parameterized Tests
|
|
8
|
-
- Confirmation and Known Issues
|
|
9
|
-
- Tags
|
|
10
|
-
- TestScoping and Test Organization
|
|
11
|
-
- Mocking and Test Doubles
|
|
12
|
-
- Testable Architecture
|
|
13
|
-
- Async and Concurrent Tests
|
|
14
|
-
- XCTest UI Tests — Page Object Pattern
|
|
15
|
-
- Performance Testing
|
|
16
|
-
- Snapshot Testing
|
|
17
|
-
- Test Attachments
|
|
18
|
-
- Exit Testing
|
|
19
|
-
- Test File Organization
|
|
20
|
-
- Common Mistakes and Review Checklist
|
|
21
|
-
|
|
22
|
-
## Basic Tests and Traits
|
|
23
|
-
|
|
24
|
-
```swift
|
|
25
|
-
import Testing
|
|
26
|
-
|
|
27
|
-
@Test("User can update their display name")
|
|
28
|
-
func updateDisplayName() {
|
|
29
|
-
var user = User(name: "Alice")
|
|
30
|
-
user.name = "Bob"
|
|
31
|
-
#expect(user.name == "Bob")
|
|
32
|
-
}
|
|
33
|
-
|
|
34
|
-
@Test(.tags(.validation, .email))
|
|
35
|
-
func validatesEmailFormat() { /* ... */ }
|
|
36
|
-
```
|
|
37
|
-
|
|
38
|
-
## Expectations and Requirements
|
|
39
|
-
|
|
40
|
-
```swift
|
|
41
|
-
#expect(result == 42)
|
|
42
|
-
#expect(name.isEmpty == false)
|
|
43
|
-
#expect(items.count > 0, "Items should not be empty")
|
|
44
|
-
|
|
45
|
-
// Error type checking
|
|
46
|
-
#expect(throws: ValidationError.self) {
|
|
47
|
-
try validate(email: "not-an-email")
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
// Specific error matching
|
|
51
|
-
#expect {
|
|
52
|
-
try validate(email: "")
|
|
53
|
-
} throws: { error in
|
|
54
|
-
guard let err = error as? ValidationError else { return false }
|
|
55
|
-
return err == .empty
|
|
56
|
-
}
|
|
57
|
-
|
|
58
|
-
// #require unwraps or fails the test
|
|
59
|
-
let user = try #require(await fetchUser(id: 1))
|
|
60
|
-
let first = try #require(items.first)
|
|
61
|
-
```
|
|
62
|
-
|
|
63
|
-
**Rule: Use `#require` when subsequent assertions depend on the value. Use `#expect` for independent checks.**
|
|
64
|
-
|
|
65
|
-
## Suite Organization
|
|
66
|
-
|
|
67
|
-
```swift
|
|
68
|
-
@Suite("User Authentication")
|
|
69
|
-
struct AuthTests {
|
|
70
|
-
let service: AuthService
|
|
71
|
-
let mockRepo: MockUserRepository
|
|
72
|
-
|
|
73
|
-
// init() replaces setUp() -- runs before each test
|
|
74
|
-
init() {
|
|
75
|
-
mockRepo = MockUserRepository()
|
|
76
|
-
service = AuthService(repository: mockRepo)
|
|
77
|
-
}
|
|
78
|
-
|
|
79
|
-
@Test func loginSucceeds() async throws {
|
|
80
|
-
let user = try await service.login(email: "test@test.com", password: "pass")
|
|
81
|
-
#expect(user.email == "test@test.com")
|
|
82
|
-
}
|
|
83
|
-
|
|
84
|
-
@Test func loginFailsWithBadPassword() async {
|
|
85
|
-
#expect(throws: AuthError.invalidCredentials) {
|
|
86
|
-
try await service.login(email: "test@test.com", password: "wrong")
|
|
87
|
-
}
|
|
88
|
-
}
|
|
89
|
-
}
|
|
90
|
-
```
|
|
91
|
-
|
|
92
|
-
Suites can nest for logical grouping:
|
|
93
|
-
|
|
94
|
-
```swift
|
|
95
|
-
@Suite("Payments")
|
|
96
|
-
struct PaymentTests {
|
|
97
|
-
@Suite("Subscriptions")
|
|
98
|
-
struct SubscriptionTests {
|
|
99
|
-
@Test func renewsAutomatically() { /* ... */ }
|
|
100
|
-
}
|
|
101
|
-
@Suite("One-Time")
|
|
102
|
-
struct OneTimeTests {
|
|
103
|
-
@Test func chargesCorrectAmount() { /* ... */ }
|
|
104
|
-
}
|
|
105
|
-
}
|
|
106
|
-
```
|
|
107
|
-
|
|
108
|
-
## Parameterized Tests
|
|
109
|
-
|
|
110
|
-
```swift
|
|
111
|
-
@Test("Email validation", arguments: [
|
|
112
|
-
("user@example.com", true),
|
|
113
|
-
("user@", false),
|
|
114
|
-
("@example.com", false),
|
|
115
|
-
("", false),
|
|
116
|
-
])
|
|
117
|
-
func validateEmail(email: String, isValid: Bool) {
|
|
118
|
-
#expect(EmailValidator.isValid(email) == isValid)
|
|
119
|
-
}
|
|
120
|
-
|
|
121
|
-
// From CaseIterable
|
|
122
|
-
@Test(arguments: Currency.allCases)
|
|
123
|
-
func currencyHasSymbol(currency: Currency) {
|
|
124
|
-
#expect(currency.symbol.isEmpty == false)
|
|
125
|
-
}
|
|
126
|
-
|
|
127
|
-
// Two collections: cartesian product
|
|
128
|
-
@Test(arguments: [1, 2, 3], ["a", "b"])
|
|
129
|
-
func combinations(number: Int, letter: String) {
|
|
130
|
-
#expect(number > 0)
|
|
131
|
-
}
|
|
132
|
-
|
|
133
|
-
// Use zip for 1:1 pairing
|
|
134
|
-
@Test(arguments: zip(["USD", "EUR"], ["$", "€"]))
|
|
135
|
-
func currencySymbols(code: String, symbol: String) {
|
|
136
|
-
#expect(Currency(code: code).symbol == symbol)
|
|
137
|
-
}
|
|
138
|
-
```
|
|
139
|
-
|
|
140
|
-
Each argument combination runs as an independent test case reported separately.
|
|
141
|
-
|
|
142
|
-
## Confirmation and Known Issues
|
|
143
|
-
|
|
144
|
-
### Confirmation (Async Event Testing)
|
|
145
|
-
|
|
146
|
-
```swift
|
|
147
|
-
// Basic confirmation -- event must fire exactly once
|
|
148
|
-
await confirmation("Received notification") { confirm in
|
|
149
|
-
let observer = NotificationCenter.default.addObserver(
|
|
150
|
-
forName: .userLoggedIn, object: nil, queue: .main
|
|
151
|
-
) { _ in confirm() }
|
|
152
|
-
await authService.login()
|
|
153
|
-
NotificationCenter.default.removeObserver(observer)
|
|
154
|
-
}
|
|
155
|
-
|
|
156
|
-
// Expected count -- event must fire exactly N times
|
|
157
|
-
await confirmation("Received 3 items", expectedCount: 3) { confirm in
|
|
158
|
-
processor.onItem = { _ in confirm() }
|
|
159
|
-
await processor.process(items)
|
|
160
|
-
}
|
|
161
|
-
```
|
|
162
|
-
|
|
163
|
-
### Known Issues
|
|
164
|
-
|
|
165
|
-
```swift
|
|
166
|
-
// Known failing test -- does not count as failure
|
|
167
|
-
withKnownIssue("Propane tank is empty") {
|
|
168
|
-
#expect(truck.grill.isHeating)
|
|
169
|
-
}
|
|
170
|
-
|
|
171
|
-
// Intermittent / flaky
|
|
172
|
-
withKnownIssue(isIntermittent: true) {
|
|
173
|
-
#expect(service.isReachable)
|
|
174
|
-
}
|
|
175
|
-
|
|
176
|
-
// Conditional
|
|
177
|
-
withKnownIssue {
|
|
178
|
-
#expect(foodTruck.grill.isHeating)
|
|
179
|
-
} when: {
|
|
180
|
-
!hasPropane
|
|
181
|
-
}
|
|
182
|
-
|
|
183
|
-
// Match specific issues only
|
|
184
|
-
try withKnownIssue {
|
|
185
|
-
let level = try #require(foodTruck.batteryLevel)
|
|
186
|
-
#expect(level >= 0.8)
|
|
187
|
-
} matching: { issue in
|
|
188
|
-
guard case .expectationFailed(let expectation) = issue.kind else { return false }
|
|
189
|
-
return expectation.isRequired
|
|
190
|
-
}
|
|
191
|
-
```
|
|
192
|
-
|
|
193
|
-
If no known issues are recorded, Swift Testing records a distinct issue notifying you the problem may be resolved.
|
|
194
|
-
|
|
195
|
-
## Tags
|
|
196
|
-
|
|
197
|
-
Tags must be declared as static members in an extension on `Tag`:
|
|
198
|
-
|
|
199
|
-
```swift
|
|
200
|
-
extension Tag {
|
|
201
|
-
@Tag static var critical: Self
|
|
202
|
-
@Tag static var slow: Self
|
|
203
|
-
@Tag static var networking: Self
|
|
204
|
-
@Tag static var validation: Self
|
|
205
|
-
}
|
|
206
|
-
|
|
207
|
-
@Test(.tags(.critical, .networking))
|
|
208
|
-
func apiCallReturnsData() async throws { /* ... */ }
|
|
209
|
-
```
|
|
210
|
-
|
|
211
|
-
Filter tests by tag in Xcode test plans or CLI (`swift test --filter tag:critical`).
|
|
212
|
-
|
|
213
|
-
## TestScoping and Test Organization
|
|
214
|
-
|
|
215
|
-
`TestScoping` consolidates per-test setup/teardown into reusable fixtures:
|
|
216
|
-
|
|
217
|
-
```swift
|
|
218
|
-
struct DatabaseFixture: TestScoping {
|
|
219
|
-
let db: TestDatabase
|
|
220
|
-
|
|
221
|
-
func provideScope(
|
|
222
|
-
for test: Test, testCase: Test.Case?,
|
|
223
|
-
performing body: @Sendable () async throws -> Void
|
|
224
|
-
) async throws {
|
|
225
|
-
let db = try await TestDatabase.create()
|
|
226
|
-
try await body()
|
|
227
|
-
try await db.destroy()
|
|
228
|
-
}
|
|
229
|
-
}
|
|
230
|
-
|
|
231
|
-
// Use with @Test trait
|
|
232
|
-
@Test(.tags(.database))
|
|
233
|
-
func insertsRecord() async throws {
|
|
234
|
-
// DatabaseFixture.provideScope wraps this test
|
|
235
|
-
}
|
|
236
|
-
```
|
|
237
|
-
|
|
238
|
-
## Mocking and Test Doubles
|
|
239
|
-
|
|
240
|
-
Define testable boundaries with protocols:
|
|
241
|
-
|
|
242
|
-
```swift
|
|
243
|
-
protocol UserRepository: Sendable {
|
|
244
|
-
func fetch(id: String) async throws -> User
|
|
245
|
-
func save(_ user: User) async throws
|
|
246
|
-
}
|
|
247
|
-
|
|
248
|
-
struct MockUserRepository: UserRepository {
|
|
249
|
-
var users: [String: User] = [:]
|
|
250
|
-
var fetchError: Error?
|
|
251
|
-
var savedUsers: [User] = []
|
|
252
|
-
|
|
253
|
-
func fetch(id: String) async throws -> User {
|
|
254
|
-
if let error = fetchError { throw error }
|
|
255
|
-
guard let user = users[id] else { throw NotFoundError() }
|
|
256
|
-
return user
|
|
257
|
-
}
|
|
258
|
-
|
|
259
|
-
mutating func save(_ user: User) async throws {
|
|
260
|
-
savedUsers.append(user)
|
|
261
|
-
users[user.id] = user
|
|
262
|
-
}
|
|
263
|
-
}
|
|
264
|
-
```
|
|
265
|
-
|
|
266
|
-
**Pattern:** Mocks conform to protocols, never subclass concrete types. Store call counts and arguments for verification.
|
|
267
|
-
|
|
268
|
-
## Testable Architecture
|
|
269
|
-
|
|
270
|
-
Inject dependencies through initializers for testability:
|
|
271
|
-
|
|
272
|
-
```swift
|
|
273
|
-
@Observable
|
|
274
|
-
class ProfileViewModel {
|
|
275
|
-
var user: User?
|
|
276
|
-
var error: Error?
|
|
277
|
-
private let repository: UserRepository
|
|
278
|
-
|
|
279
|
-
init(repository: UserRepository) {
|
|
280
|
-
self.repository = repository
|
|
281
|
-
}
|
|
282
|
-
|
|
283
|
-
func load() async {
|
|
284
|
-
do {
|
|
285
|
-
user = try await repository.fetch(id: "current")
|
|
286
|
-
} catch {
|
|
287
|
-
self.error = error
|
|
288
|
-
}
|
|
289
|
-
}
|
|
290
|
-
}
|
|
291
|
-
|
|
292
|
-
// Test with mock
|
|
293
|
-
@Test @MainActor func viewModelLoadsUser() async {
|
|
294
|
-
let mock = MockUserRepository(users: ["current": .preview])
|
|
295
|
-
let vm = ProfileViewModel(repository: mock)
|
|
296
|
-
await vm.load()
|
|
297
|
-
#expect(vm.user?.name == "Alice")
|
|
298
|
-
}
|
|
299
|
-
|
|
300
|
-
@Test @MainActor func viewModelHandlesError() async {
|
|
301
|
-
var mock = MockUserRepository()
|
|
302
|
-
mock.fetchError = URLError(.notConnectedToInternet)
|
|
303
|
-
let vm = ProfileViewModel(repository: mock)
|
|
304
|
-
await vm.load()
|
|
305
|
-
#expect(vm.user == nil)
|
|
306
|
-
#expect(vm.error != nil)
|
|
307
|
-
}
|
|
308
|
-
```
|
|
309
|
-
|
|
310
|
-
## Async and Concurrent Tests
|
|
311
|
-
|
|
312
|
-
```swift
|
|
313
|
-
@Test @MainActor func viewModelUpdatesOnMainActor() async {
|
|
314
|
-
let vm = ProfileViewModel(repository: MockUserRepository())
|
|
315
|
-
await vm.load()
|
|
316
|
-
#expect(vm.user != nil)
|
|
317
|
-
}
|
|
318
|
-
|
|
319
|
-
// Clock injection for time-dependent logic
|
|
320
|
-
@Test func debounceUsesCorrectDelay() async throws {
|
|
321
|
-
let clock = TestClock()
|
|
322
|
-
let debouncer = Debouncer(delay: .seconds(1), clock: clock)
|
|
323
|
-
debouncer.submit { /* action */ }
|
|
324
|
-
await clock.advance(by: .milliseconds(500))
|
|
325
|
-
#expect(!debouncer.hasExecuted)
|
|
326
|
-
await clock.advance(by: .milliseconds(500))
|
|
327
|
-
#expect(debouncer.hasExecuted)
|
|
328
|
-
}
|
|
329
|
-
|
|
330
|
-
// Error path testing
|
|
331
|
-
@Test func fetchThrowsOnNetworkError() async {
|
|
332
|
-
var mock = MockUserRepository()
|
|
333
|
-
mock.fetchError = URLError(.notConnectedToInternet)
|
|
334
|
-
#expect(throws: URLError.self) {
|
|
335
|
-
try await mock.fetch(id: "1")
|
|
336
|
-
}
|
|
337
|
-
}
|
|
338
|
-
```
|
|
339
|
-
|
|
340
|
-
## XCTest UI Tests — Page Object Pattern
|
|
341
|
-
|
|
342
|
-
Swift Testing does not support UI testing. Use XCTest with XCUITest for all UI tests.
|
|
343
|
-
|
|
344
|
-
```swift
|
|
345
|
-
class LoginUITests: XCTestCase {
|
|
346
|
-
let app = XCUIApplication()
|
|
347
|
-
|
|
348
|
-
override func setUpWithError() throws {
|
|
349
|
-
continueAfterFailure = false
|
|
350
|
-
app.launch()
|
|
351
|
-
}
|
|
352
|
-
|
|
353
|
-
func testLoginFlow() throws {
|
|
354
|
-
let loginPage = LoginPage(app: app)
|
|
355
|
-
let homePage = loginPage.login(email: "test@test.com", password: "password")
|
|
356
|
-
XCTAssertTrue(homePage.welcomeLabel.exists)
|
|
357
|
-
}
|
|
358
|
-
}
|
|
359
|
-
```
|
|
360
|
-
|
|
361
|
-
### Page Object Pattern
|
|
362
|
-
|
|
363
|
-
Encapsulate UI element queries in page objects for reusable, readable UI tests:
|
|
364
|
-
|
|
365
|
-
```swift
|
|
366
|
-
struct LoginPage {
|
|
367
|
-
let app: XCUIApplication
|
|
368
|
-
var emailField: XCUIElement { app.textFields["Email"] }
|
|
369
|
-
var passwordField: XCUIElement { app.secureTextFields["Password"] }
|
|
370
|
-
var signInButton: XCUIElement { app.buttons["Sign In"] }
|
|
371
|
-
|
|
372
|
-
@discardableResult
|
|
373
|
-
func login(email: String, password: String) -> HomePage {
|
|
374
|
-
emailField.tap(); emailField.typeText(email)
|
|
375
|
-
passwordField.tap(); passwordField.typeText(password)
|
|
376
|
-
signInButton.tap()
|
|
377
|
-
return HomePage(app: app)
|
|
378
|
-
}
|
|
379
|
-
}
|
|
380
|
-
|
|
381
|
-
struct HomePage {
|
|
382
|
-
let app: XCUIApplication
|
|
383
|
-
var welcomeLabel: XCUIElement { app.staticTexts["Welcome"] }
|
|
384
|
-
}
|
|
385
|
-
```
|
|
386
|
-
|
|
387
|
-
## Performance Testing
|
|
388
|
-
|
|
389
|
-
```swift
|
|
390
|
-
class FeedPerformanceTests: XCTestCase {
|
|
391
|
-
func testFeedParsingPerformance() throws {
|
|
392
|
-
let data = try loadFixture("large-feed.json")
|
|
393
|
-
let metrics: [XCTMetric] = [XCTClockMetric(), XCTMemoryMetric()]
|
|
394
|
-
measure(metrics: metrics) {
|
|
395
|
-
_ = try? FeedParser.parse(data)
|
|
396
|
-
}
|
|
397
|
-
}
|
|
398
|
-
}
|
|
399
|
-
```
|
|
400
|
-
|
|
401
|
-
Performance tests require XCTest — not available in Swift Testing.
|
|
402
|
-
|
|
403
|
-
## Snapshot Testing
|
|
404
|
-
|
|
405
|
-
Use swift-snapshot-testing (pointfreeco) for visual regression. Requires XCTest:
|
|
406
|
-
|
|
407
|
-
```swift
|
|
408
|
-
import SnapshotTesting
|
|
409
|
-
import XCTest
|
|
410
|
-
|
|
411
|
-
class ProfileViewSnapshotTests: XCTestCase {
|
|
412
|
-
func testProfileView() {
|
|
413
|
-
let view = ProfileView(user: .preview)
|
|
414
|
-
assertSnapshot(of: view, as: .image(layout: .device(config: .iPhone13)))
|
|
415
|
-
|
|
416
|
-
// Dark mode
|
|
417
|
-
assertSnapshot(of: view.environment(\.colorScheme, .dark),
|
|
418
|
-
as: .image(layout: .device(config: .iPhone13)), named: "dark")
|
|
419
|
-
|
|
420
|
-
// Large Dynamic Type
|
|
421
|
-
assertSnapshot(of: view.environment(\.dynamicTypeSize, .accessibility3),
|
|
422
|
-
as: .image(layout: .device(config: .iPhone13)), named: "largeText")
|
|
423
|
-
}
|
|
424
|
-
}
|
|
425
|
-
```
|
|
426
|
-
|
|
427
|
-
Always test Dark Mode and large Dynamic Type in snapshots.
|
|
428
|
-
|
|
429
|
-
## Test Attachments
|
|
430
|
-
|
|
431
|
-
Attach diagnostic data to test results for debugging failures:
|
|
432
|
-
|
|
433
|
-
```swift
|
|
434
|
-
@Test func generateReport() async throws {
|
|
435
|
-
let report = try generateReport()
|
|
436
|
-
// Attach the output for later inspection
|
|
437
|
-
Attachment(report.data, named: "report.json").record()
|
|
438
|
-
#expect(report.isValid)
|
|
439
|
-
}
|
|
440
|
-
|
|
441
|
-
// Attach from a file URL
|
|
442
|
-
@Test func processImage() async throws {
|
|
443
|
-
let output = try processImage()
|
|
444
|
-
try await Attachment(contentsOf: output.url, named: "result.png")
|
|
445
|
-
.record()
|
|
446
|
-
}
|
|
447
|
-
```
|
|
448
|
-
|
|
449
|
-
Attachments support any `Attachable` type and images via `AttachableAsImage`.
|
|
450
|
-
|
|
451
|
-
## Exit Testing
|
|
452
|
-
|
|
453
|
-
Test code that calls `exit()`, `fatalError()`, or `preconditionFailure()`:
|
|
454
|
-
|
|
455
|
-
```swift
|
|
456
|
-
@Test func invalidInputCausesExit() async {
|
|
457
|
-
await #expect(processExitsWith: .failure) {
|
|
458
|
-
processInvalidInput() // calls fatalError()
|
|
459
|
-
}
|
|
460
|
-
}
|
|
461
|
-
```
|
|
462
|
-
|
|
463
|
-
Exit testing runs the closure in a subprocess. The test passes if the process exits with the expected status.
|
|
464
|
-
|
|
465
|
-
## Test File Organization
|
|
466
|
-
|
|
467
|
-
```text
|
|
468
|
-
Tests/AppTests/ # Swift Testing (Models/, ViewModels/, Services/)
|
|
469
|
-
Tests/AppUITests/ # XCTest UI tests (Pages/, Flows/)
|
|
470
|
-
Tests/Fixtures/ # Test data (JSON, images)
|
|
471
|
-
Tests/Mocks/ # Shared mock implementations
|
|
472
|
-
```
|
|
473
|
-
|
|
474
|
-
Name test files `<TypeUnderTest>Tests.swift`. Describe behavior in function names: `fetchUserReturnsNilOnNetworkError()` not `testFetchUser()`. Name mocks `Mock<ProtocolName>`.
|
|
475
|
-
|
|
476
|
-
### What to Test
|
|
477
|
-
|
|
478
|
-
**Always test:** business logic, validation rules, state transitions in view models, error handling paths, edge cases (empty collections, nil, boundaries), async success and failure, Task cancellation.
|
|
479
|
-
|
|
480
|
-
**Skip:** SwiftUI view body layout (use snapshots), simple property forwarding, Apple framework behavior, private methods (test through public API).
|
|
481
|
-
|
|
482
|
-
## Common Mistakes and Review Checklist
|
|
483
|
-
|
|
484
|
-
1. **Testing implementation, not behavior.** Test what the code does, not how.
|
|
485
|
-
2. **No error path tests.** If a function can throw, test the throw path.
|
|
486
|
-
3. **Flaky async tests.** Use `confirmation` with expected counts, not `sleep` calls.
|
|
487
|
-
4. **Shared mutable state between tests.** Each test sets up its own state via `init()` in `@Suite`.
|
|
488
|
-
5. **Missing accessibility identifiers in UI tests.** XCUITest queries rely on them.
|
|
489
|
-
6. **Using `sleep` in tests.** Use `confirmation`, clock injection, or `withKnownIssue`.
|
|
490
|
-
7. **Not testing cancellation.** If code supports `Task` cancellation, verify it cancels cleanly.
|
|
491
|
-
8. **Mixing XCTest and Swift Testing in one file.** Keep them in separate files.
|
|
492
|
-
9. **Non-Sendable test helpers shared across tests.** Ensure test helper types are Sendable when shared across concurrent test cases.
|
|
493
|
-
|
|
494
|
-
### Review Checklist
|
|
495
|
-
|
|
496
|
-
- [ ] All new tests use Swift Testing (`@Test`, `#expect`), not XCTest assertions
|
|
497
|
-
- [ ] Test names describe behavior (`fetchUserReturnsNilOnNetworkError` not `testFetchUser`)
|
|
498
|
-
- [ ] Error paths have dedicated tests
|
|
499
|
-
- [ ] Async tests use `confirmation()`, not `Task.sleep`
|
|
500
|
-
- [ ] Parameterized tests used for repetitive variations
|
|
501
|
-
- [ ] Tags applied for filtering (`.critical`, `.slow`)
|
|
502
|
-
- [ ] Mocks conform to protocols, not subclass concrete types
|
|
503
|
-
- [ ] No shared mutable state between tests
|
|
504
|
-
- [ ] Cancellation tested for cancellable async operations
|
|
1
|
+
# Swift Testing Patterns Reference
|
|
2
|
+
|
|
3
|
+
## Contents
|
|
4
|
+
- Basic Tests and Traits
|
|
5
|
+
- Expectations and Requirements
|
|
6
|
+
- Suite Organization
|
|
7
|
+
- Parameterized Tests
|
|
8
|
+
- Confirmation and Known Issues
|
|
9
|
+
- Tags
|
|
10
|
+
- TestScoping and Test Organization
|
|
11
|
+
- Mocking and Test Doubles
|
|
12
|
+
- Testable Architecture
|
|
13
|
+
- Async and Concurrent Tests
|
|
14
|
+
- XCTest UI Tests — Page Object Pattern
|
|
15
|
+
- Performance Testing
|
|
16
|
+
- Snapshot Testing
|
|
17
|
+
- Test Attachments
|
|
18
|
+
- Exit Testing
|
|
19
|
+
- Test File Organization
|
|
20
|
+
- Common Mistakes and Review Checklist
|
|
21
|
+
|
|
22
|
+
## Basic Tests and Traits
|
|
23
|
+
|
|
24
|
+
```swift
|
|
25
|
+
import Testing
|
|
26
|
+
|
|
27
|
+
@Test("User can update their display name")
|
|
28
|
+
func updateDisplayName() {
|
|
29
|
+
var user = User(name: "Alice")
|
|
30
|
+
user.name = "Bob"
|
|
31
|
+
#expect(user.name == "Bob")
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
@Test(.tags(.validation, .email))
|
|
35
|
+
func validatesEmailFormat() { /* ... */ }
|
|
36
|
+
```
|
|
37
|
+
|
|
38
|
+
## Expectations and Requirements
|
|
39
|
+
|
|
40
|
+
```swift
|
|
41
|
+
#expect(result == 42)
|
|
42
|
+
#expect(name.isEmpty == false)
|
|
43
|
+
#expect(items.count > 0, "Items should not be empty")
|
|
44
|
+
|
|
45
|
+
// Error type checking
|
|
46
|
+
#expect(throws: ValidationError.self) {
|
|
47
|
+
try validate(email: "not-an-email")
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
// Specific error matching
|
|
51
|
+
#expect {
|
|
52
|
+
try validate(email: "")
|
|
53
|
+
} throws: { error in
|
|
54
|
+
guard let err = error as? ValidationError else { return false }
|
|
55
|
+
return err == .empty
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
// #require unwraps or fails the test
|
|
59
|
+
let user = try #require(await fetchUser(id: 1))
|
|
60
|
+
let first = try #require(items.first)
|
|
61
|
+
```
|
|
62
|
+
|
|
63
|
+
**Rule: Use `#require` when subsequent assertions depend on the value. Use `#expect` for independent checks.**
|
|
64
|
+
|
|
65
|
+
## Suite Organization
|
|
66
|
+
|
|
67
|
+
```swift
|
|
68
|
+
@Suite("User Authentication")
|
|
69
|
+
struct AuthTests {
|
|
70
|
+
let service: AuthService
|
|
71
|
+
let mockRepo: MockUserRepository
|
|
72
|
+
|
|
73
|
+
// init() replaces setUp() -- runs before each test
|
|
74
|
+
init() {
|
|
75
|
+
mockRepo = MockUserRepository()
|
|
76
|
+
service = AuthService(repository: mockRepo)
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
@Test func loginSucceeds() async throws {
|
|
80
|
+
let user = try await service.login(email: "test@test.com", password: "pass")
|
|
81
|
+
#expect(user.email == "test@test.com")
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
@Test func loginFailsWithBadPassword() async {
|
|
85
|
+
#expect(throws: AuthError.invalidCredentials) {
|
|
86
|
+
try await service.login(email: "test@test.com", password: "wrong")
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
```
|
|
91
|
+
|
|
92
|
+
Suites can nest for logical grouping:
|
|
93
|
+
|
|
94
|
+
```swift
|
|
95
|
+
@Suite("Payments")
|
|
96
|
+
struct PaymentTests {
|
|
97
|
+
@Suite("Subscriptions")
|
|
98
|
+
struct SubscriptionTests {
|
|
99
|
+
@Test func renewsAutomatically() { /* ... */ }
|
|
100
|
+
}
|
|
101
|
+
@Suite("One-Time")
|
|
102
|
+
struct OneTimeTests {
|
|
103
|
+
@Test func chargesCorrectAmount() { /* ... */ }
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
```
|
|
107
|
+
|
|
108
|
+
## Parameterized Tests
|
|
109
|
+
|
|
110
|
+
```swift
|
|
111
|
+
@Test("Email validation", arguments: [
|
|
112
|
+
("user@example.com", true),
|
|
113
|
+
("user@", false),
|
|
114
|
+
("@example.com", false),
|
|
115
|
+
("", false),
|
|
116
|
+
])
|
|
117
|
+
func validateEmail(email: String, isValid: Bool) {
|
|
118
|
+
#expect(EmailValidator.isValid(email) == isValid)
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
// From CaseIterable
|
|
122
|
+
@Test(arguments: Currency.allCases)
|
|
123
|
+
func currencyHasSymbol(currency: Currency) {
|
|
124
|
+
#expect(currency.symbol.isEmpty == false)
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
// Two collections: cartesian product
|
|
128
|
+
@Test(arguments: [1, 2, 3], ["a", "b"])
|
|
129
|
+
func combinations(number: Int, letter: String) {
|
|
130
|
+
#expect(number > 0)
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
// Use zip for 1:1 pairing
|
|
134
|
+
@Test(arguments: zip(["USD", "EUR"], ["$", "€"]))
|
|
135
|
+
func currencySymbols(code: String, symbol: String) {
|
|
136
|
+
#expect(Currency(code: code).symbol == symbol)
|
|
137
|
+
}
|
|
138
|
+
```
|
|
139
|
+
|
|
140
|
+
Each argument combination runs as an independent test case reported separately.
|
|
141
|
+
|
|
142
|
+
## Confirmation and Known Issues
|
|
143
|
+
|
|
144
|
+
### Confirmation (Async Event Testing)
|
|
145
|
+
|
|
146
|
+
```swift
|
|
147
|
+
// Basic confirmation -- event must fire exactly once
|
|
148
|
+
await confirmation("Received notification") { confirm in
|
|
149
|
+
let observer = NotificationCenter.default.addObserver(
|
|
150
|
+
forName: .userLoggedIn, object: nil, queue: .main
|
|
151
|
+
) { _ in confirm() }
|
|
152
|
+
await authService.login()
|
|
153
|
+
NotificationCenter.default.removeObserver(observer)
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
// Expected count -- event must fire exactly N times
|
|
157
|
+
await confirmation("Received 3 items", expectedCount: 3) { confirm in
|
|
158
|
+
processor.onItem = { _ in confirm() }
|
|
159
|
+
await processor.process(items)
|
|
160
|
+
}
|
|
161
|
+
```
|
|
162
|
+
|
|
163
|
+
### Known Issues
|
|
164
|
+
|
|
165
|
+
```swift
|
|
166
|
+
// Known failing test -- does not count as failure
|
|
167
|
+
withKnownIssue("Propane tank is empty") {
|
|
168
|
+
#expect(truck.grill.isHeating)
|
|
169
|
+
}
|
|
170
|
+
|
|
171
|
+
// Intermittent / flaky
|
|
172
|
+
withKnownIssue(isIntermittent: true) {
|
|
173
|
+
#expect(service.isReachable)
|
|
174
|
+
}
|
|
175
|
+
|
|
176
|
+
// Conditional
|
|
177
|
+
withKnownIssue {
|
|
178
|
+
#expect(foodTruck.grill.isHeating)
|
|
179
|
+
} when: {
|
|
180
|
+
!hasPropane
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
// Match specific issues only
|
|
184
|
+
try withKnownIssue {
|
|
185
|
+
let level = try #require(foodTruck.batteryLevel)
|
|
186
|
+
#expect(level >= 0.8)
|
|
187
|
+
} matching: { issue in
|
|
188
|
+
guard case .expectationFailed(let expectation) = issue.kind else { return false }
|
|
189
|
+
return expectation.isRequired
|
|
190
|
+
}
|
|
191
|
+
```
|
|
192
|
+
|
|
193
|
+
If no known issues are recorded, Swift Testing records a distinct issue notifying you the problem may be resolved.
|
|
194
|
+
|
|
195
|
+
## Tags
|
|
196
|
+
|
|
197
|
+
Tags must be declared as static members in an extension on `Tag`:
|
|
198
|
+
|
|
199
|
+
```swift
|
|
200
|
+
extension Tag {
|
|
201
|
+
@Tag static var critical: Self
|
|
202
|
+
@Tag static var slow: Self
|
|
203
|
+
@Tag static var networking: Self
|
|
204
|
+
@Tag static var validation: Self
|
|
205
|
+
}
|
|
206
|
+
|
|
207
|
+
@Test(.tags(.critical, .networking))
|
|
208
|
+
func apiCallReturnsData() async throws { /* ... */ }
|
|
209
|
+
```
|
|
210
|
+
|
|
211
|
+
Filter tests by tag in Xcode test plans or CLI (`swift test --filter tag:critical`).
|
|
212
|
+
|
|
213
|
+
## TestScoping and Test Organization
|
|
214
|
+
|
|
215
|
+
`TestScoping` consolidates per-test setup/teardown into reusable fixtures:
|
|
216
|
+
|
|
217
|
+
```swift
|
|
218
|
+
struct DatabaseFixture: TestScoping {
|
|
219
|
+
let db: TestDatabase
|
|
220
|
+
|
|
221
|
+
func provideScope(
|
|
222
|
+
for test: Test, testCase: Test.Case?,
|
|
223
|
+
performing body: @Sendable () async throws -> Void
|
|
224
|
+
) async throws {
|
|
225
|
+
let db = try await TestDatabase.create()
|
|
226
|
+
try await body()
|
|
227
|
+
try await db.destroy()
|
|
228
|
+
}
|
|
229
|
+
}
|
|
230
|
+
|
|
231
|
+
// Use with @Test trait
|
|
232
|
+
@Test(.tags(.database))
|
|
233
|
+
func insertsRecord() async throws {
|
|
234
|
+
// DatabaseFixture.provideScope wraps this test
|
|
235
|
+
}
|
|
236
|
+
```
|
|
237
|
+
|
|
238
|
+
## Mocking and Test Doubles
|
|
239
|
+
|
|
240
|
+
Define testable boundaries with protocols:
|
|
241
|
+
|
|
242
|
+
```swift
|
|
243
|
+
protocol UserRepository: Sendable {
|
|
244
|
+
func fetch(id: String) async throws -> User
|
|
245
|
+
func save(_ user: User) async throws
|
|
246
|
+
}
|
|
247
|
+
|
|
248
|
+
struct MockUserRepository: UserRepository {
|
|
249
|
+
var users: [String: User] = [:]
|
|
250
|
+
var fetchError: Error?
|
|
251
|
+
var savedUsers: [User] = []
|
|
252
|
+
|
|
253
|
+
func fetch(id: String) async throws -> User {
|
|
254
|
+
if let error = fetchError { throw error }
|
|
255
|
+
guard let user = users[id] else { throw NotFoundError() }
|
|
256
|
+
return user
|
|
257
|
+
}
|
|
258
|
+
|
|
259
|
+
mutating func save(_ user: User) async throws {
|
|
260
|
+
savedUsers.append(user)
|
|
261
|
+
users[user.id] = user
|
|
262
|
+
}
|
|
263
|
+
}
|
|
264
|
+
```
|
|
265
|
+
|
|
266
|
+
**Pattern:** Mocks conform to protocols, never subclass concrete types. Store call counts and arguments for verification.
|
|
267
|
+
|
|
268
|
+
## Testable Architecture
|
|
269
|
+
|
|
270
|
+
Inject dependencies through initializers for testability:
|
|
271
|
+
|
|
272
|
+
```swift
|
|
273
|
+
@Observable
|
|
274
|
+
class ProfileViewModel {
|
|
275
|
+
var user: User?
|
|
276
|
+
var error: Error?
|
|
277
|
+
private let repository: UserRepository
|
|
278
|
+
|
|
279
|
+
init(repository: UserRepository) {
|
|
280
|
+
self.repository = repository
|
|
281
|
+
}
|
|
282
|
+
|
|
283
|
+
func load() async {
|
|
284
|
+
do {
|
|
285
|
+
user = try await repository.fetch(id: "current")
|
|
286
|
+
} catch {
|
|
287
|
+
self.error = error
|
|
288
|
+
}
|
|
289
|
+
}
|
|
290
|
+
}
|
|
291
|
+
|
|
292
|
+
// Test with mock
|
|
293
|
+
@Test @MainActor func viewModelLoadsUser() async {
|
|
294
|
+
let mock = MockUserRepository(users: ["current": .preview])
|
|
295
|
+
let vm = ProfileViewModel(repository: mock)
|
|
296
|
+
await vm.load()
|
|
297
|
+
#expect(vm.user?.name == "Alice")
|
|
298
|
+
}
|
|
299
|
+
|
|
300
|
+
@Test @MainActor func viewModelHandlesError() async {
|
|
301
|
+
var mock = MockUserRepository()
|
|
302
|
+
mock.fetchError = URLError(.notConnectedToInternet)
|
|
303
|
+
let vm = ProfileViewModel(repository: mock)
|
|
304
|
+
await vm.load()
|
|
305
|
+
#expect(vm.user == nil)
|
|
306
|
+
#expect(vm.error != nil)
|
|
307
|
+
}
|
|
308
|
+
```
|
|
309
|
+
|
|
310
|
+
## Async and Concurrent Tests
|
|
311
|
+
|
|
312
|
+
```swift
|
|
313
|
+
@Test @MainActor func viewModelUpdatesOnMainActor() async {
|
|
314
|
+
let vm = ProfileViewModel(repository: MockUserRepository())
|
|
315
|
+
await vm.load()
|
|
316
|
+
#expect(vm.user != nil)
|
|
317
|
+
}
|
|
318
|
+
|
|
319
|
+
// Clock injection for time-dependent logic
|
|
320
|
+
@Test func debounceUsesCorrectDelay() async throws {
|
|
321
|
+
let clock = TestClock()
|
|
322
|
+
let debouncer = Debouncer(delay: .seconds(1), clock: clock)
|
|
323
|
+
debouncer.submit { /* action */ }
|
|
324
|
+
await clock.advance(by: .milliseconds(500))
|
|
325
|
+
#expect(!debouncer.hasExecuted)
|
|
326
|
+
await clock.advance(by: .milliseconds(500))
|
|
327
|
+
#expect(debouncer.hasExecuted)
|
|
328
|
+
}
|
|
329
|
+
|
|
330
|
+
// Error path testing
|
|
331
|
+
@Test func fetchThrowsOnNetworkError() async {
|
|
332
|
+
var mock = MockUserRepository()
|
|
333
|
+
mock.fetchError = URLError(.notConnectedToInternet)
|
|
334
|
+
#expect(throws: URLError.self) {
|
|
335
|
+
try await mock.fetch(id: "1")
|
|
336
|
+
}
|
|
337
|
+
}
|
|
338
|
+
```
|
|
339
|
+
|
|
340
|
+
## XCTest UI Tests — Page Object Pattern
|
|
341
|
+
|
|
342
|
+
Swift Testing does not support UI testing. Use XCTest with XCUITest for all UI tests.
|
|
343
|
+
|
|
344
|
+
```swift
|
|
345
|
+
class LoginUITests: XCTestCase {
|
|
346
|
+
let app = XCUIApplication()
|
|
347
|
+
|
|
348
|
+
override func setUpWithError() throws {
|
|
349
|
+
continueAfterFailure = false
|
|
350
|
+
app.launch()
|
|
351
|
+
}
|
|
352
|
+
|
|
353
|
+
func testLoginFlow() throws {
|
|
354
|
+
let loginPage = LoginPage(app: app)
|
|
355
|
+
let homePage = loginPage.login(email: "test@test.com", password: "password")
|
|
356
|
+
XCTAssertTrue(homePage.welcomeLabel.exists)
|
|
357
|
+
}
|
|
358
|
+
}
|
|
359
|
+
```
|
|
360
|
+
|
|
361
|
+
### Page Object Pattern
|
|
362
|
+
|
|
363
|
+
Encapsulate UI element queries in page objects for reusable, readable UI tests:
|
|
364
|
+
|
|
365
|
+
```swift
|
|
366
|
+
struct LoginPage {
|
|
367
|
+
let app: XCUIApplication
|
|
368
|
+
var emailField: XCUIElement { app.textFields["Email"] }
|
|
369
|
+
var passwordField: XCUIElement { app.secureTextFields["Password"] }
|
|
370
|
+
var signInButton: XCUIElement { app.buttons["Sign In"] }
|
|
371
|
+
|
|
372
|
+
@discardableResult
|
|
373
|
+
func login(email: String, password: String) -> HomePage {
|
|
374
|
+
emailField.tap(); emailField.typeText(email)
|
|
375
|
+
passwordField.tap(); passwordField.typeText(password)
|
|
376
|
+
signInButton.tap()
|
|
377
|
+
return HomePage(app: app)
|
|
378
|
+
}
|
|
379
|
+
}
|
|
380
|
+
|
|
381
|
+
struct HomePage {
|
|
382
|
+
let app: XCUIApplication
|
|
383
|
+
var welcomeLabel: XCUIElement { app.staticTexts["Welcome"] }
|
|
384
|
+
}
|
|
385
|
+
```
|
|
386
|
+
|
|
387
|
+
## Performance Testing
|
|
388
|
+
|
|
389
|
+
```swift
|
|
390
|
+
class FeedPerformanceTests: XCTestCase {
|
|
391
|
+
func testFeedParsingPerformance() throws {
|
|
392
|
+
let data = try loadFixture("large-feed.json")
|
|
393
|
+
let metrics: [XCTMetric] = [XCTClockMetric(), XCTMemoryMetric()]
|
|
394
|
+
measure(metrics: metrics) {
|
|
395
|
+
_ = try? FeedParser.parse(data)
|
|
396
|
+
}
|
|
397
|
+
}
|
|
398
|
+
}
|
|
399
|
+
```
|
|
400
|
+
|
|
401
|
+
Performance tests require XCTest — not available in Swift Testing.
|
|
402
|
+
|
|
403
|
+
## Snapshot Testing
|
|
404
|
+
|
|
405
|
+
Use swift-snapshot-testing (pointfreeco) for visual regression. Requires XCTest:
|
|
406
|
+
|
|
407
|
+
```swift
|
|
408
|
+
import SnapshotTesting
|
|
409
|
+
import XCTest
|
|
410
|
+
|
|
411
|
+
class ProfileViewSnapshotTests: XCTestCase {
|
|
412
|
+
func testProfileView() {
|
|
413
|
+
let view = ProfileView(user: .preview)
|
|
414
|
+
assertSnapshot(of: view, as: .image(layout: .device(config: .iPhone13)))
|
|
415
|
+
|
|
416
|
+
// Dark mode
|
|
417
|
+
assertSnapshot(of: view.environment(\.colorScheme, .dark),
|
|
418
|
+
as: .image(layout: .device(config: .iPhone13)), named: "dark")
|
|
419
|
+
|
|
420
|
+
// Large Dynamic Type
|
|
421
|
+
assertSnapshot(of: view.environment(\.dynamicTypeSize, .accessibility3),
|
|
422
|
+
as: .image(layout: .device(config: .iPhone13)), named: "largeText")
|
|
423
|
+
}
|
|
424
|
+
}
|
|
425
|
+
```
|
|
426
|
+
|
|
427
|
+
Always test Dark Mode and large Dynamic Type in snapshots.
|
|
428
|
+
|
|
429
|
+
## Test Attachments
|
|
430
|
+
|
|
431
|
+
Attach diagnostic data to test results for debugging failures:
|
|
432
|
+
|
|
433
|
+
```swift
|
|
434
|
+
@Test func generateReport() async throws {
|
|
435
|
+
let report = try generateReport()
|
|
436
|
+
// Attach the output for later inspection
|
|
437
|
+
Attachment(report.data, named: "report.json").record()
|
|
438
|
+
#expect(report.isValid)
|
|
439
|
+
}
|
|
440
|
+
|
|
441
|
+
// Attach from a file URL
|
|
442
|
+
@Test func processImage() async throws {
|
|
443
|
+
let output = try processImage()
|
|
444
|
+
try await Attachment(contentsOf: output.url, named: "result.png")
|
|
445
|
+
.record()
|
|
446
|
+
}
|
|
447
|
+
```
|
|
448
|
+
|
|
449
|
+
Attachments support any `Attachable` type and images via `AttachableAsImage`.
|
|
450
|
+
|
|
451
|
+
## Exit Testing
|
|
452
|
+
|
|
453
|
+
Test code that calls `exit()`, `fatalError()`, or `preconditionFailure()`:
|
|
454
|
+
|
|
455
|
+
```swift
|
|
456
|
+
@Test func invalidInputCausesExit() async {
|
|
457
|
+
await #expect(processExitsWith: .failure) {
|
|
458
|
+
processInvalidInput() // calls fatalError()
|
|
459
|
+
}
|
|
460
|
+
}
|
|
461
|
+
```
|
|
462
|
+
|
|
463
|
+
Exit testing runs the closure in a subprocess. The test passes if the process exits with the expected status.
|
|
464
|
+
|
|
465
|
+
## Test File Organization
|
|
466
|
+
|
|
467
|
+
```text
|
|
468
|
+
Tests/AppTests/ # Swift Testing (Models/, ViewModels/, Services/)
|
|
469
|
+
Tests/AppUITests/ # XCTest UI tests (Pages/, Flows/)
|
|
470
|
+
Tests/Fixtures/ # Test data (JSON, images)
|
|
471
|
+
Tests/Mocks/ # Shared mock implementations
|
|
472
|
+
```
|
|
473
|
+
|
|
474
|
+
Name test files `<TypeUnderTest>Tests.swift`. Describe behavior in function names: `fetchUserReturnsNilOnNetworkError()` not `testFetchUser()`. Name mocks `Mock<ProtocolName>`.
|
|
475
|
+
|
|
476
|
+
### What to Test
|
|
477
|
+
|
|
478
|
+
**Always test:** business logic, validation rules, state transitions in view models, error handling paths, edge cases (empty collections, nil, boundaries), async success and failure, Task cancellation.
|
|
479
|
+
|
|
480
|
+
**Skip:** SwiftUI view body layout (use snapshots), simple property forwarding, Apple framework behavior, private methods (test through public API).
|
|
481
|
+
|
|
482
|
+
## Common Mistakes and Review Checklist
|
|
483
|
+
|
|
484
|
+
1. **Testing implementation, not behavior.** Test what the code does, not how.
|
|
485
|
+
2. **No error path tests.** If a function can throw, test the throw path.
|
|
486
|
+
3. **Flaky async tests.** Use `confirmation` with expected counts, not `sleep` calls.
|
|
487
|
+
4. **Shared mutable state between tests.** Each test sets up its own state via `init()` in `@Suite`.
|
|
488
|
+
5. **Missing accessibility identifiers in UI tests.** XCUITest queries rely on them.
|
|
489
|
+
6. **Using `sleep` in tests.** Use `confirmation`, clock injection, or `withKnownIssue`.
|
|
490
|
+
7. **Not testing cancellation.** If code supports `Task` cancellation, verify it cancels cleanly.
|
|
491
|
+
8. **Mixing XCTest and Swift Testing in one file.** Keep them in separate files.
|
|
492
|
+
9. **Non-Sendable test helpers shared across tests.** Ensure test helper types are Sendable when shared across concurrent test cases.
|
|
493
|
+
|
|
494
|
+
### Review Checklist
|
|
495
|
+
|
|
496
|
+
- [ ] All new tests use Swift Testing (`@Test`, `#expect`), not XCTest assertions
|
|
497
|
+
- [ ] Test names describe behavior (`fetchUserReturnsNilOnNetworkError` not `testFetchUser`)
|
|
498
|
+
- [ ] Error paths have dedicated tests
|
|
499
|
+
- [ ] Async tests use `confirmation()`, not `Task.sleep`
|
|
500
|
+
- [ ] Parameterized tests used for repetitive variations
|
|
501
|
+
- [ ] Tags applied for filtering (`.critical`, `.slow`)
|
|
502
|
+
- [ ] Mocks conform to protocols, not subclass concrete types
|
|
503
|
+
- [ ] No shared mutable state between tests
|
|
504
|
+
- [ ] Cancellation tested for cancellable async operations
|