@devo-bmad-custom/agent-orchestration 1.0.1 → 1.0.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/lib/installer.js +44 -11
- 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/bmad-track-compact.md +1 -1
- package/src/.claude/commands/bmad-track-extended.md +1 -1
- package/src/.claude/commands/bmad-track-large.md +1 -1
- package/src/.claude/commands/bmad-track-medium.md +1 -1
- package/src/.claude/commands/bmad-track-nano.md +1 -1
- package/src/.claude/commands/bmad-track-rv.md +1 -1
- package/src/.claude/commands/bmad-track-small.md +1 -1
- package/src/.claude/commands/master-orchestrator.md +15 -0
- package/src/_memory/config.yaml +11 -11
- package/src/_memory/master-orchestrator-sidecar/instructions.md +85 -32
- package/src/_memory/skills/nimbalyst-tracking/SKILL.md +103 -103
- package/src/_memory/skills/writing-skills/SKILL.md +655 -655
- package/src/bmb/agents/agent-builder.md +59 -59
- package/src/bmb/agents/module-builder.md +60 -60
- package/src/bmb/agents/workflow-builder.md +61 -61
- package/src/bmb/config.yaml +12 -12
- package/src/bmb/module-help.csv +13 -13
- package/src/bmb/workflows/agent/data/agent-architecture.md +258 -258
- package/src/bmb/workflows/agent/data/agent-compilation.md +185 -185
- package/src/bmb/workflows/agent/data/agent-menu-patterns.md +189 -189
- package/src/bmb/workflows/agent/data/agent-metadata.md +133 -133
- package/src/bmb/workflows/agent/data/agent-validation.md +111 -111
- package/src/bmb/workflows/agent/data/brainstorm-context.md +96 -96
- package/src/bmb/workflows/agent/data/communication-presets.csv +61 -61
- package/src/bmb/workflows/agent/data/critical-actions.md +75 -75
- package/src/bmb/workflows/agent/data/persona-properties.md +252 -252
- package/src/bmb/workflows/agent/data/principles-crafting.md +142 -142
- package/src/bmb/workflows/agent/data/reference/module-examples/architect.md +68 -68
- package/src/bmb/workflows/agent/data/reference/with-sidecar/journal-keeper/journal-keeper-sidecar/entries/yy-mm-dd-entry-template.md +16 -16
- package/src/bmb/workflows/agent/data/understanding-agent-types.md +126 -126
- package/src/bmb/workflows/agent/steps-c/step-01-brainstorm.md +129 -129
- package/src/bmb/workflows/agent/steps-c/step-02-discovery.md +170 -170
- package/src/bmb/workflows/agent/steps-c/step-03-sidecar-metadata.md +309 -309
- package/src/bmb/workflows/agent/steps-c/step-04-persona.md +213 -213
- package/src/bmb/workflows/agent/steps-c/step-05-commands-menu.md +179 -179
- package/src/bmb/workflows/agent/steps-c/step-06-activation.md +278 -278
- package/src/bmb/workflows/agent/steps-c/step-07-build-agent.md +316 -316
- package/src/bmb/workflows/agent/steps-c/step-08-celebrate.md +247 -247
- package/src/bmb/workflows/agent/steps-e/e-01-load-existing.md +221 -221
- package/src/bmb/workflows/agent/steps-e/e-02-discover-edits.md +195 -195
- package/src/bmb/workflows/agent/steps-e/e-04-sidecar-metadata.md +126 -126
- package/src/bmb/workflows/agent/steps-e/e-05-persona.md +135 -135
- package/src/bmb/workflows/agent/steps-e/e-06-commands-menu.md +123 -123
- package/src/bmb/workflows/agent/steps-e/e-07-activation.md +124 -124
- package/src/bmb/workflows/agent/steps-e/e-08-edit-agent.md +197 -197
- package/src/bmb/workflows/agent/steps-e/e-09-celebrate.md +155 -155
- package/src/bmb/workflows/agent/steps-v/v-01-load-review.md +137 -137
- package/src/bmb/workflows/agent/steps-v/v-02a-validate-metadata.md +116 -116
- package/src/bmb/workflows/agent/steps-v/v-02b-validate-persona.md +124 -124
- package/src/bmb/workflows/agent/steps-v/v-02c-validate-menu.md +127 -127
- package/src/bmb/workflows/agent/steps-v/v-02d-validate-structure.md +134 -134
- package/src/bmb/workflows/agent/steps-v/v-02e-validate-sidecar.md +134 -134
- package/src/bmb/workflows/agent/steps-v/v-03-summary.md +104 -104
- package/src/bmb/workflows/agent/templates/agent-plan.template.md +5 -5
- package/src/bmb/workflows/agent/templates/agent-template.md +89 -89
- package/src/bmb/workflows/agent/workflow-create-agent.md +72 -72
- package/src/bmb/workflows/agent/workflow-edit-agent.md +75 -75
- package/src/bmb/workflows/agent/workflow-validate-agent.md +73 -73
- package/src/bmb/workflows/module/data/agent-architecture.md +179 -179
- package/src/bmb/workflows/module/data/agent-spec-template.md +79 -79
- package/src/bmb/workflows/module/data/module-standards.md +263 -263
- package/src/bmb/workflows/module/data/module-yaml-conventions.md +392 -392
- package/src/bmb/workflows/module/module-help-generate.md +254 -254
- package/src/bmb/workflows/module/steps-b/step-01-welcome.md +148 -148
- package/src/bmb/workflows/module/steps-b/step-02-spark.md +141 -141
- package/src/bmb/workflows/module/steps-b/step-03-module-type.md +149 -149
- package/src/bmb/workflows/module/steps-b/step-04-vision.md +83 -83
- package/src/bmb/workflows/module/steps-b/step-05-identity.md +97 -97
- package/src/bmb/workflows/module/steps-b/step-06-users.md +86 -86
- package/src/bmb/workflows/module/steps-b/step-07-value.md +76 -76
- package/src/bmb/workflows/module/steps-b/step-08-agents.md +97 -97
- package/src/bmb/workflows/module/steps-b/step-09-workflows.md +83 -83
- package/src/bmb/workflows/module/steps-b/step-10-tools.md +91 -91
- package/src/bmb/workflows/module/steps-b/step-11-scenarios.md +84 -84
- package/src/bmb/workflows/module/steps-b/step-12-creative.md +95 -95
- package/src/bmb/workflows/module/steps-b/step-13-review.md +105 -105
- package/src/bmb/workflows/module/steps-b/step-14-finalize.md +117 -117
- package/src/bmb/workflows/module/steps-c/step-01-load-brief.md +179 -179
- package/src/bmb/workflows/module/steps-c/step-01b-continue.md +82 -82
- package/src/bmb/workflows/module/steps-c/step-02-structure.md +105 -105
- package/src/bmb/workflows/module/steps-c/step-03-config.md +119 -119
- package/src/bmb/workflows/module/steps-c/step-04-agents.md +168 -168
- package/src/bmb/workflows/module/steps-c/step-05-workflows.md +184 -184
- package/src/bmb/workflows/module/steps-c/step-06-docs.md +401 -401
- package/src/bmb/workflows/module/steps-c/step-07-complete.md +152 -152
- package/src/bmb/workflows/module/steps-e/step-01-load-target.md +81 -81
- package/src/bmb/workflows/module/steps-e/step-02-select-edit.md +77 -77
- package/src/bmb/workflows/module/steps-e/step-03-apply-edit.md +77 -77
- package/src/bmb/workflows/module/steps-e/step-04-review.md +80 -80
- package/src/bmb/workflows/module/steps-e/step-05-confirm.md +75 -75
- package/src/bmb/workflows/module/steps-v/step-01-load-target.md +96 -96
- package/src/bmb/workflows/module/steps-v/step-02-file-structure.md +93 -93
- package/src/bmb/workflows/module/steps-v/step-03-module-yaml.md +99 -99
- package/src/bmb/workflows/module/steps-v/step-04-agent-specs.md +152 -152
- package/src/bmb/workflows/module/steps-v/step-05-workflow-specs.md +152 -152
- package/src/bmb/workflows/module/steps-v/step-06-documentation.md +143 -143
- package/src/bmb/workflows/module/steps-v/step-07-installation.md +102 -102
- package/src/bmb/workflows/module/steps-v/step-08-report.md +197 -197
- package/src/bmb/workflows/module/templates/brief-template.md +154 -154
- package/src/bmb/workflows/module/templates/workflow-spec-template.md +96 -96
- package/src/bmb/workflows/module/workflow-create-module-brief.md +71 -71
- package/src/bmb/workflows/module/workflow-create-module.md +86 -86
- package/src/bmb/workflows/module/workflow-edit-module.md +66 -66
- package/src/bmb/workflows/module/workflow-validate-module.md +66 -66
- package/src/bmb/workflows/workflow/data/architecture.md +150 -150
- package/src/bmb/workflows/workflow/data/common-workflow-tools.csv +19 -19
- package/src/bmb/workflows/workflow/data/csv-data-file-standards.md +53 -53
- package/src/bmb/workflows/workflow/data/frontmatter-standards.md +184 -184
- package/src/bmb/workflows/workflow/data/input-discovery-standards.md +191 -191
- package/src/bmb/workflows/workflow/data/intent-vs-prescriptive-spectrum.md +44 -44
- package/src/bmb/workflows/workflow/data/menu-handling-standards.md +133 -133
- package/src/bmb/workflows/workflow/data/output-format-standards.md +135 -135
- package/src/bmb/workflows/workflow/data/step-file-rules.md +235 -235
- package/src/bmb/workflows/workflow/data/step-type-patterns.md +257 -257
- package/src/bmb/workflows/workflow/data/subprocess-optimization-patterns.md +188 -188
- package/src/bmb/workflows/workflow/data/trimodal-workflow-structure.md +164 -164
- package/src/bmb/workflows/workflow/data/workflow-chaining-standards.md +222 -222
- package/src/bmb/workflows/workflow/data/workflow-examples.md +232 -232
- package/src/bmb/workflows/workflow/data/workflow-type-criteria.md +134 -134
- package/src/bmb/workflows/workflow/steps-c/step-00-conversion.md +263 -263
- package/src/bmb/workflows/workflow/steps-c/step-01-discovery.md +194 -194
- package/src/bmb/workflows/workflow/steps-c/step-01b-continuation.md +3 -3
- package/src/bmb/workflows/workflow/steps-c/step-02-classification.md +270 -270
- package/src/bmb/workflows/workflow/steps-c/step-03-requirements.md +283 -283
- package/src/bmb/workflows/workflow/steps-c/step-04-tools.md +282 -282
- package/src/bmb/workflows/workflow/steps-c/step-05-plan-review.md +243 -243
- package/src/bmb/workflows/workflow/steps-c/step-06-design.md +330 -330
- package/src/bmb/workflows/workflow/steps-c/step-07-foundation.md +239 -239
- package/src/bmb/workflows/workflow/steps-c/step-08-build-step-01.md +379 -379
- package/src/bmb/workflows/workflow/steps-c/step-09-build-next-step.md +350 -350
- package/src/bmb/workflows/workflow/steps-c/step-10-confirmation.md +322 -322
- package/src/bmb/workflows/workflow/steps-c/step-11-completion.md +191 -191
- package/src/bmb/workflows/workflow/steps-e/step-e-01-assess-workflow.md +237 -237
- package/src/bmb/workflows/workflow/steps-e/step-e-02-discover-edits.md +251 -251
- package/src/bmb/workflows/workflow/steps-e/step-e-03-fix-validation.md +254 -254
- package/src/bmb/workflows/workflow/steps-e/step-e-04-direct-edit.md +277 -277
- package/src/bmb/workflows/workflow/steps-e/step-e-05-apply-edit.md +154 -154
- package/src/bmb/workflows/workflow/steps-e/step-e-06-validate-after.md +190 -190
- package/src/bmb/workflows/workflow/steps-e/step-e-07-complete.md +206 -206
- package/src/bmb/workflows/workflow/steps-v/step-01-validate-max-mode.md +109 -109
- package/src/bmb/workflows/workflow/steps-v/step-01-validate.md +221 -221
- package/src/bmb/workflows/workflow/steps-v/step-01b-structure.md +152 -152
- package/src/bmb/workflows/workflow/steps-v/step-02-frontmatter-validation.md +199 -199
- package/src/bmb/workflows/workflow/steps-v/step-02b-path-violations.md +265 -265
- package/src/bmb/workflows/workflow/steps-v/step-03-menu-validation.md +164 -164
- package/src/bmb/workflows/workflow/steps-v/step-04-step-type-validation.md +211 -211
- package/src/bmb/workflows/workflow/steps-v/step-05-output-format-validation.md +200 -200
- package/src/bmb/workflows/workflow/steps-v/step-06-validation-design-check.md +195 -195
- package/src/bmb/workflows/workflow/steps-v/step-07-instruction-style-check.md +209 -209
- package/src/bmb/workflows/workflow/steps-v/step-08-collaborative-experience-check.md +199 -199
- package/src/bmb/workflows/workflow/steps-v/step-08b-subprocess-optimization.md +179 -179
- package/src/bmb/workflows/workflow/steps-v/step-09-cohesive-review.md +186 -186
- package/src/bmb/workflows/workflow/steps-v/step-10-report-complete.md +154 -154
- package/src/bmb/workflows/workflow/steps-v/step-11-plan-validation.md +237 -237
- package/src/bmb/workflows/workflow/templates/minimal-output-template.md +11 -11
- package/src/bmb/workflows/workflow/templates/step-01-init-continuable-template.md +241 -241
- package/src/bmb/workflows/workflow/templates/step-1b-template.md +224 -224
- package/src/bmb/workflows/workflow/templates/step-template.md +294 -294
- package/src/bmb/workflows/workflow/templates/workflow-template.md +102 -102
- package/src/bmb/workflows/workflow/workflow-create-workflow.md +79 -79
- package/src/bmb/workflows/workflow/workflow-edit-workflow.md +65 -65
- package/src/bmb/workflows/workflow/workflow-rework-workflow.md +65 -65
- package/src/bmb/workflows/workflow/workflow-validate-max-parallel-workflow.md +66 -66
- package/src/bmb/workflows/workflow/workflow-validate-workflow.md +65 -65
- package/src/bmm/agents/analyst.md +104 -104
- package/src/bmm/agents/dev.md +100 -100
- package/src/bmm/agents/qa.md +100 -90
- package/src/bmm/agents/review-agent.md +1 -1
- 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/agents/master-orchestrator.md +3 -3
- 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,443 +1,443 @@
|
|
|
1
|
-
# Business Logic Security Reference
|
|
2
|
-
|
|
3
|
-
## Overview
|
|
4
|
-
|
|
5
|
-
Business logic vulnerabilities occur when the application's logic can be manipulated to achieve unintended outcomes. Unlike technical vulnerabilities, these flaws exploit legitimate functionality in unexpected ways.
|
|
6
|
-
|
|
7
|
-
## Common Vulnerability Types
|
|
8
|
-
|
|
9
|
-
### 1. Race Conditions
|
|
10
|
-
|
|
11
|
-
#### Time-of-Check to Time-of-Use (TOCTOU)
|
|
12
|
-
|
|
13
|
-
```python
|
|
14
|
-
# VULNERABLE: Race condition in balance check
|
|
15
|
-
def transfer(from_account, to_account, amount):
|
|
16
|
-
if from_account.balance >= amount: # Check
|
|
17
|
-
time.sleep(0.1) # Simulating processing delay
|
|
18
|
-
from_account.balance -= amount # Use
|
|
19
|
-
to_account.balance += amount
|
|
20
|
-
|
|
21
|
-
# Attack: Two concurrent transfers can overdraft
|
|
22
|
-
|
|
23
|
-
# SAFE: Atomic operation with locking
|
|
24
|
-
from threading import Lock
|
|
25
|
-
|
|
26
|
-
account_locks = {}
|
|
27
|
-
|
|
28
|
-
def transfer(from_account, to_account, amount):
|
|
29
|
-
# Acquire locks in consistent order to prevent deadlock
|
|
30
|
-
locks = sorted([from_account.id, to_account.id])
|
|
31
|
-
with account_locks[locks[0]], account_locks[locks[1]]:
|
|
32
|
-
if from_account.balance >= amount:
|
|
33
|
-
from_account.balance -= amount
|
|
34
|
-
to_account.balance += amount
|
|
35
|
-
return True
|
|
36
|
-
return False
|
|
37
|
-
```
|
|
38
|
-
|
|
39
|
-
#### Database-Level Locking
|
|
40
|
-
|
|
41
|
-
```python
|
|
42
|
-
# SAFE: Database transaction with SELECT FOR UPDATE
|
|
43
|
-
from django.db import transaction
|
|
44
|
-
|
|
45
|
-
@transaction.atomic
|
|
46
|
-
def transfer(from_account_id, to_account_id, amount):
|
|
47
|
-
from_account = Account.objects.select_for_update().get(id=from_account_id)
|
|
48
|
-
to_account = Account.objects.select_for_update().get(id=to_account_id)
|
|
49
|
-
|
|
50
|
-
if from_account.balance >= amount:
|
|
51
|
-
from_account.balance -= amount
|
|
52
|
-
to_account.balance += amount
|
|
53
|
-
from_account.save()
|
|
54
|
-
to_account.save()
|
|
55
|
-
return True
|
|
56
|
-
return False
|
|
57
|
-
```
|
|
58
|
-
|
|
59
|
-
### 2. Workflow Bypass
|
|
60
|
-
|
|
61
|
-
```python
|
|
62
|
-
# VULNERABLE: Multi-step process without server-side tracking
|
|
63
|
-
# Step 1: /verify-email
|
|
64
|
-
# Step 2: /set-password
|
|
65
|
-
# Step 3: /complete-registration
|
|
66
|
-
|
|
67
|
-
# Attacker skips to Step 3
|
|
68
|
-
|
|
69
|
-
# SAFE: Server-side state machine
|
|
70
|
-
class RegistrationFlow:
|
|
71
|
-
STATES = ['email_pending', 'email_verified', 'password_set', 'complete']
|
|
72
|
-
|
|
73
|
-
def __init__(self, user_id):
|
|
74
|
-
self.state = self.get_state(user_id)
|
|
75
|
-
|
|
76
|
-
def verify_email(self, token):
|
|
77
|
-
if self.state != 'email_pending':
|
|
78
|
-
raise InvalidStateError("Email verification not pending")
|
|
79
|
-
# Verify token...
|
|
80
|
-
self.set_state('email_verified')
|
|
81
|
-
|
|
82
|
-
def set_password(self, password):
|
|
83
|
-
if self.state != 'email_verified':
|
|
84
|
-
raise InvalidStateError("Email not verified")
|
|
85
|
-
# Set password...
|
|
86
|
-
self.set_state('password_set')
|
|
87
|
-
|
|
88
|
-
def complete(self):
|
|
89
|
-
if self.state != 'password_set':
|
|
90
|
-
raise InvalidStateError("Password not set")
|
|
91
|
-
# Complete registration...
|
|
92
|
-
self.set_state('complete')
|
|
93
|
-
```
|
|
94
|
-
|
|
95
|
-
### 3. Numeric Manipulation
|
|
96
|
-
|
|
97
|
-
#### Integer Overflow
|
|
98
|
-
|
|
99
|
-
```python
|
|
100
|
-
# VULNERABLE: Integer overflow in quantity
|
|
101
|
-
def calculate_total(quantity, price):
|
|
102
|
-
return quantity * price
|
|
103
|
-
|
|
104
|
-
# Attack: quantity = -1 results in negative price (refund)
|
|
105
|
-
|
|
106
|
-
# SAFE: Validate numeric ranges
|
|
107
|
-
def calculate_total(quantity, price):
|
|
108
|
-
if quantity <= 0 or quantity > MAX_QUANTITY:
|
|
109
|
-
raise ValueError("Invalid quantity")
|
|
110
|
-
if price <= 0:
|
|
111
|
-
raise ValueError("Invalid price")
|
|
112
|
-
return quantity * price
|
|
113
|
-
```
|
|
114
|
-
|
|
115
|
-
#### Floating Point Issues
|
|
116
|
-
|
|
117
|
-
```python
|
|
118
|
-
# VULNERABLE: Floating point precision loss
|
|
119
|
-
total = 0.0
|
|
120
|
-
for item in items:
|
|
121
|
-
total += item.price * item.quantity
|
|
122
|
-
|
|
123
|
-
# 0.1 + 0.2 = 0.30000000000000004
|
|
124
|
-
|
|
125
|
-
# SAFE: Use Decimal for financial calculations
|
|
126
|
-
from decimal import Decimal, ROUND_HALF_UP
|
|
127
|
-
|
|
128
|
-
total = Decimal('0')
|
|
129
|
-
for item in items:
|
|
130
|
-
total += Decimal(str(item.price)) * item.quantity
|
|
131
|
-
|
|
132
|
-
# Round properly
|
|
133
|
-
total = total.quantize(Decimal('.01'), rounding=ROUND_HALF_UP)
|
|
134
|
-
```
|
|
135
|
-
|
|
136
|
-
### 4. Price/Discount Manipulation
|
|
137
|
-
|
|
138
|
-
```python
|
|
139
|
-
# VULNERABLE: Trust client-submitted price
|
|
140
|
-
@app.route('/checkout', methods=['POST'])
|
|
141
|
-
def checkout():
|
|
142
|
-
price = request.json['price'] # Client can set any price!
|
|
143
|
-
process_payment(price)
|
|
144
|
-
|
|
145
|
-
# SAFE: Calculate price server-side
|
|
146
|
-
@app.route('/checkout', methods=['POST'])
|
|
147
|
-
def checkout():
|
|
148
|
-
cart = get_cart(current_user.id)
|
|
149
|
-
price = calculate_total(cart) # Always server-calculated
|
|
150
|
-
process_payment(price)
|
|
151
|
-
```
|
|
152
|
-
|
|
153
|
-
```python
|
|
154
|
-
# VULNERABLE: Stackable discounts without limits
|
|
155
|
-
def apply_discounts(cart, discount_codes):
|
|
156
|
-
for code in discount_codes:
|
|
157
|
-
discount = get_discount(code)
|
|
158
|
-
cart.total -= discount.amount
|
|
159
|
-
|
|
160
|
-
# Attack: Apply same code multiple times, negative total
|
|
161
|
-
|
|
162
|
-
# SAFE: Limit discount application
|
|
163
|
-
def apply_discounts(cart, discount_codes):
|
|
164
|
-
# Remove duplicates
|
|
165
|
-
unique_codes = set(discount_codes)
|
|
166
|
-
|
|
167
|
-
total_discount = Decimal('0')
|
|
168
|
-
for code in unique_codes:
|
|
169
|
-
if is_code_used(cart.user_id, code):
|
|
170
|
-
continue # Code already used
|
|
171
|
-
discount = get_discount(code)
|
|
172
|
-
total_discount += discount.amount
|
|
173
|
-
mark_code_used(cart.user_id, code)
|
|
174
|
-
|
|
175
|
-
# Cap discount at total
|
|
176
|
-
max_discount = cart.subtotal * Decimal('0.5') # Max 50% off
|
|
177
|
-
final_discount = min(total_discount, max_discount)
|
|
178
|
-
cart.total -= final_discount
|
|
179
|
-
```
|
|
180
|
-
|
|
181
|
-
### 5. Inventory/Resource Exhaustion
|
|
182
|
-
|
|
183
|
-
```python
|
|
184
|
-
# VULNERABLE: No reservation during checkout
|
|
185
|
-
def checkout(cart):
|
|
186
|
-
for item in cart.items:
|
|
187
|
-
if get_stock(item.product_id) >= item.quantity:
|
|
188
|
-
# Stock available
|
|
189
|
-
pass
|
|
190
|
-
# Processing takes time...
|
|
191
|
-
process_payment()
|
|
192
|
-
for item in cart.items:
|
|
193
|
-
reduce_stock(item.product_id, item.quantity) # May oversell
|
|
194
|
-
|
|
195
|
-
# SAFE: Reserve inventory atomically
|
|
196
|
-
@transaction.atomic
|
|
197
|
-
def checkout(cart):
|
|
198
|
-
for item in cart.items:
|
|
199
|
-
product = Product.objects.select_for_update().get(id=item.product_id)
|
|
200
|
-
if product.stock < item.quantity:
|
|
201
|
-
raise InsufficientStock(product.name)
|
|
202
|
-
product.stock -= item.quantity # Reserve immediately
|
|
203
|
-
product.save()
|
|
204
|
-
|
|
205
|
-
# If payment fails, transaction rolls back
|
|
206
|
-
process_payment()
|
|
207
|
-
```
|
|
208
|
-
|
|
209
|
-
### 6. Time-Based Attacks
|
|
210
|
-
|
|
211
|
-
```python
|
|
212
|
-
# VULNERABLE: Expired coupon still usable with timing attack
|
|
213
|
-
def apply_coupon(code):
|
|
214
|
-
coupon = Coupon.objects.get(code=code)
|
|
215
|
-
if coupon.expiry > datetime.now():
|
|
216
|
-
return coupon.discount
|
|
217
|
-
raise CouponExpired()
|
|
218
|
-
|
|
219
|
-
# SAFE: Use database time, not application time
|
|
220
|
-
from django.db.models.functions import Now
|
|
221
|
-
|
|
222
|
-
def apply_coupon(code):
|
|
223
|
-
coupon = Coupon.objects.annotate(
|
|
224
|
-
is_valid=Q(expiry__gt=Now())
|
|
225
|
-
).get(code=code)
|
|
226
|
-
|
|
227
|
-
if not coupon.is_valid:
|
|
228
|
-
raise CouponExpired()
|
|
229
|
-
return coupon.discount
|
|
230
|
-
```
|
|
231
|
-
|
|
232
|
-
### 7. Parameter Tampering
|
|
233
|
-
|
|
234
|
-
```python
|
|
235
|
-
# VULNERABLE: Trust hidden form fields
|
|
236
|
-
# HTML: <input type="hidden" name="user_id" value="123">
|
|
237
|
-
|
|
238
|
-
@app.route('/update-profile', methods=['POST'])
|
|
239
|
-
def update_profile():
|
|
240
|
-
user_id = request.form['user_id'] # Attacker can change this!
|
|
241
|
-
User.query.get(user_id).update(...)
|
|
242
|
-
|
|
243
|
-
# SAFE: Use session-based user identification
|
|
244
|
-
@app.route('/update-profile', methods=['POST'])
|
|
245
|
-
def update_profile():
|
|
246
|
-
user_id = current_user.id # From authenticated session
|
|
247
|
-
User.query.get(user_id).update(...)
|
|
248
|
-
```
|
|
249
|
-
|
|
250
|
-
---
|
|
251
|
-
|
|
252
|
-
## Detection Patterns
|
|
253
|
-
|
|
254
|
-
### State Machine Validation
|
|
255
|
-
|
|
256
|
-
```python
|
|
257
|
-
class OrderStateMachine:
|
|
258
|
-
VALID_TRANSITIONS = {
|
|
259
|
-
'draft': ['submitted'],
|
|
260
|
-
'submitted': ['approved', 'rejected'],
|
|
261
|
-
'approved': ['shipped'],
|
|
262
|
-
'shipped': ['delivered', 'returned'],
|
|
263
|
-
'delivered': ['returned'],
|
|
264
|
-
'rejected': [],
|
|
265
|
-
'returned': ['refunded'],
|
|
266
|
-
'refunded': []
|
|
267
|
-
}
|
|
268
|
-
|
|
269
|
-
def transition(self, order, new_state):
|
|
270
|
-
current = order.state
|
|
271
|
-
if new_state not in self.VALID_TRANSITIONS.get(current, []):
|
|
272
|
-
raise InvalidTransition(f"Cannot go from {current} to {new_state}")
|
|
273
|
-
order.state = new_state
|
|
274
|
-
log_state_change(order, current, new_state)
|
|
275
|
-
```
|
|
276
|
-
|
|
277
|
-
### Idempotency
|
|
278
|
-
|
|
279
|
-
```python
|
|
280
|
-
# SAFE: Idempotent operations with idempotency keys
|
|
281
|
-
import hashlib
|
|
282
|
-
|
|
283
|
-
def process_request(request_data, idempotency_key):
|
|
284
|
-
# Check if request was already processed
|
|
285
|
-
existing = ProcessedRequest.query.filter_by(key=idempotency_key).first()
|
|
286
|
-
if existing:
|
|
287
|
-
return existing.response # Return cached response
|
|
288
|
-
|
|
289
|
-
# Process request
|
|
290
|
-
result = do_processing(request_data)
|
|
291
|
-
|
|
292
|
-
# Store for future duplicate requests
|
|
293
|
-
ProcessedRequest.create(key=idempotency_key, response=result)
|
|
294
|
-
return result
|
|
295
|
-
```
|
|
296
|
-
|
|
297
|
-
### Rate Limiting Business Actions
|
|
298
|
-
|
|
299
|
-
```python
|
|
300
|
-
# Limit business-critical actions
|
|
301
|
-
from functools import wraps
|
|
302
|
-
import time
|
|
303
|
-
|
|
304
|
-
def rate_limit_action(action_name, limit, window):
|
|
305
|
-
def decorator(f):
|
|
306
|
-
@wraps(f)
|
|
307
|
-
def wrapper(*args, **kwargs):
|
|
308
|
-
user_id = current_user.id
|
|
309
|
-
key = f"action:{action_name}:{user_id}"
|
|
310
|
-
|
|
311
|
-
count = redis.incr(key)
|
|
312
|
-
if count == 1:
|
|
313
|
-
redis.expire(key, window)
|
|
314
|
-
|
|
315
|
-
if count > limit:
|
|
316
|
-
raise RateLimitExceeded(f"Too many {action_name} attempts")
|
|
317
|
-
|
|
318
|
-
return f(*args, **kwargs)
|
|
319
|
-
return wrapper
|
|
320
|
-
return decorator
|
|
321
|
-
|
|
322
|
-
@rate_limit_action('password_reset', limit=3, window=3600)
|
|
323
|
-
def request_password_reset(email):
|
|
324
|
-
pass
|
|
325
|
-
|
|
326
|
-
@rate_limit_action('transfer', limit=10, window=86400)
|
|
327
|
-
def transfer_funds(from_account, to_account, amount):
|
|
328
|
-
pass
|
|
329
|
-
```
|
|
330
|
-
|
|
331
|
-
---
|
|
332
|
-
|
|
333
|
-
## Validation Patterns
|
|
334
|
-
|
|
335
|
-
### Server-Side Calculation
|
|
336
|
-
|
|
337
|
-
```python
|
|
338
|
-
# Always recalculate on server
|
|
339
|
-
def calculate_order_total(order):
|
|
340
|
-
subtotal = Decimal('0')
|
|
341
|
-
for item in order.items:
|
|
342
|
-
# Get current price from database, not from request
|
|
343
|
-
product = Product.query.get(item.product_id)
|
|
344
|
-
subtotal += product.price * item.quantity
|
|
345
|
-
|
|
346
|
-
# Apply tax
|
|
347
|
-
tax = subtotal * get_tax_rate(order.shipping_address)
|
|
348
|
-
|
|
349
|
-
# Apply discounts (validated server-side)
|
|
350
|
-
discount = calculate_discounts(order, order.discount_codes)
|
|
351
|
-
|
|
352
|
-
# Calculate total
|
|
353
|
-
total = subtotal + tax - discount
|
|
354
|
-
|
|
355
|
-
# Sanity checks
|
|
356
|
-
if total < Decimal('0'):
|
|
357
|
-
raise InvalidOrderError("Negative total")
|
|
358
|
-
if discount > subtotal:
|
|
359
|
-
raise InvalidOrderError("Discount exceeds subtotal")
|
|
360
|
-
|
|
361
|
-
return {
|
|
362
|
-
'subtotal': subtotal,
|
|
363
|
-
'tax': tax,
|
|
364
|
-
'discount': discount,
|
|
365
|
-
'total': total
|
|
366
|
-
}
|
|
367
|
-
```
|
|
368
|
-
|
|
369
|
-
### Business Rule Enforcement
|
|
370
|
-
|
|
371
|
-
```python
|
|
372
|
-
class TransferValidator:
|
|
373
|
-
def validate(self, transfer):
|
|
374
|
-
errors = []
|
|
375
|
-
|
|
376
|
-
# Check transfer limits
|
|
377
|
-
if transfer.amount > MAX_SINGLE_TRANSFER:
|
|
378
|
-
errors.append("Exceeds single transfer limit")
|
|
379
|
-
|
|
380
|
-
# Check daily limits
|
|
381
|
-
daily_total = get_daily_transfer_total(transfer.from_account)
|
|
382
|
-
if daily_total + transfer.amount > DAILY_LIMIT:
|
|
383
|
-
errors.append("Exceeds daily transfer limit")
|
|
384
|
-
|
|
385
|
-
# Check velocity (unusual number of transfers)
|
|
386
|
-
recent_count = get_recent_transfer_count(transfer.from_account, hours=1)
|
|
387
|
-
if recent_count > MAX_TRANSFERS_PER_HOUR:
|
|
388
|
-
errors.append("Too many transfers in short period")
|
|
389
|
-
|
|
390
|
-
# Check for unusual patterns
|
|
391
|
-
if is_unusual_recipient(transfer.from_account, transfer.to_account):
|
|
392
|
-
errors.append("Unusual recipient - requires verification")
|
|
393
|
-
|
|
394
|
-
if errors:
|
|
395
|
-
raise ValidationError(errors)
|
|
396
|
-
```
|
|
397
|
-
|
|
398
|
-
---
|
|
399
|
-
|
|
400
|
-
## Grep Patterns for Detection
|
|
401
|
-
|
|
402
|
-
```bash
|
|
403
|
-
# Race condition indicators
|
|
404
|
-
grep -rn "sleep\|time\.sleep\|Thread\|async" --include="*.py"
|
|
405
|
-
grep -rn "balance\|inventory\|stock" --include="*.py" | grep -v "select_for_update\|lock"
|
|
406
|
-
|
|
407
|
-
# Price/amount from request
|
|
408
|
-
grep -rn "request\.\w*\[.*price\|request\.\w*\[.*amount\|request\.\w*\[.*total" --include="*.py"
|
|
409
|
-
|
|
410
|
-
# Missing validation
|
|
411
|
-
grep -rn "def checkout\|def purchase\|def transfer" --include="*.py"
|
|
412
|
-
|
|
413
|
-
# Floating point for money
|
|
414
|
-
grep -rn "float.*price\|float.*amount\|float.*balance" --include="*.py"
|
|
415
|
-
```
|
|
416
|
-
|
|
417
|
-
---
|
|
418
|
-
|
|
419
|
-
## Testing Checklist
|
|
420
|
-
|
|
421
|
-
- [ ] Race conditions tested (concurrent requests)
|
|
422
|
-
- [ ] Workflow steps enforced server-side
|
|
423
|
-
- [ ] State transitions validated
|
|
424
|
-
- [ ] Prices/totals calculated server-side
|
|
425
|
-
- [ ] Discount limits enforced
|
|
426
|
-
- [ ] Inventory checked and reserved atomically
|
|
427
|
-
- [ ] Integer overflow/underflow prevented
|
|
428
|
-
- [ ] Decimal used for financial calculations
|
|
429
|
-
- [ ] Time-based logic uses server/database time
|
|
430
|
-
- [ ] Hidden field values not trusted
|
|
431
|
-
- [ ] Idempotency keys for critical operations
|
|
432
|
-
- [ ] Rate limits on business-critical actions
|
|
433
|
-
- [ ] Unusual patterns detected and flagged
|
|
434
|
-
|
|
435
|
-
---
|
|
436
|
-
|
|
437
|
-
## References
|
|
438
|
-
|
|
439
|
-
- [OWASP Business Logic Testing](https://owasp.org/www-project-web-security-testing-guide/latest/4-Web_Application_Security_Testing/10-Business_Logic_Testing/)
|
|
440
|
-
- [CWE-362: Race Condition](https://cwe.mitre.org/data/definitions/362.html)
|
|
441
|
-
- [CWE-367: TOCTOU Race Condition](https://cwe.mitre.org/data/definitions/367.html)
|
|
442
|
-
- [CWE-190: Integer Overflow](https://cwe.mitre.org/data/definitions/190.html)
|
|
443
|
-
- [CWE-840: Business Logic Errors](https://cwe.mitre.org/data/definitions/840.html)
|
|
1
|
+
# Business Logic Security Reference
|
|
2
|
+
|
|
3
|
+
## Overview
|
|
4
|
+
|
|
5
|
+
Business logic vulnerabilities occur when the application's logic can be manipulated to achieve unintended outcomes. Unlike technical vulnerabilities, these flaws exploit legitimate functionality in unexpected ways.
|
|
6
|
+
|
|
7
|
+
## Common Vulnerability Types
|
|
8
|
+
|
|
9
|
+
### 1. Race Conditions
|
|
10
|
+
|
|
11
|
+
#### Time-of-Check to Time-of-Use (TOCTOU)
|
|
12
|
+
|
|
13
|
+
```python
|
|
14
|
+
# VULNERABLE: Race condition in balance check
|
|
15
|
+
def transfer(from_account, to_account, amount):
|
|
16
|
+
if from_account.balance >= amount: # Check
|
|
17
|
+
time.sleep(0.1) # Simulating processing delay
|
|
18
|
+
from_account.balance -= amount # Use
|
|
19
|
+
to_account.balance += amount
|
|
20
|
+
|
|
21
|
+
# Attack: Two concurrent transfers can overdraft
|
|
22
|
+
|
|
23
|
+
# SAFE: Atomic operation with locking
|
|
24
|
+
from threading import Lock
|
|
25
|
+
|
|
26
|
+
account_locks = {}
|
|
27
|
+
|
|
28
|
+
def transfer(from_account, to_account, amount):
|
|
29
|
+
# Acquire locks in consistent order to prevent deadlock
|
|
30
|
+
locks = sorted([from_account.id, to_account.id])
|
|
31
|
+
with account_locks[locks[0]], account_locks[locks[1]]:
|
|
32
|
+
if from_account.balance >= amount:
|
|
33
|
+
from_account.balance -= amount
|
|
34
|
+
to_account.balance += amount
|
|
35
|
+
return True
|
|
36
|
+
return False
|
|
37
|
+
```
|
|
38
|
+
|
|
39
|
+
#### Database-Level Locking
|
|
40
|
+
|
|
41
|
+
```python
|
|
42
|
+
# SAFE: Database transaction with SELECT FOR UPDATE
|
|
43
|
+
from django.db import transaction
|
|
44
|
+
|
|
45
|
+
@transaction.atomic
|
|
46
|
+
def transfer(from_account_id, to_account_id, amount):
|
|
47
|
+
from_account = Account.objects.select_for_update().get(id=from_account_id)
|
|
48
|
+
to_account = Account.objects.select_for_update().get(id=to_account_id)
|
|
49
|
+
|
|
50
|
+
if from_account.balance >= amount:
|
|
51
|
+
from_account.balance -= amount
|
|
52
|
+
to_account.balance += amount
|
|
53
|
+
from_account.save()
|
|
54
|
+
to_account.save()
|
|
55
|
+
return True
|
|
56
|
+
return False
|
|
57
|
+
```
|
|
58
|
+
|
|
59
|
+
### 2. Workflow Bypass
|
|
60
|
+
|
|
61
|
+
```python
|
|
62
|
+
# VULNERABLE: Multi-step process without server-side tracking
|
|
63
|
+
# Step 1: /verify-email
|
|
64
|
+
# Step 2: /set-password
|
|
65
|
+
# Step 3: /complete-registration
|
|
66
|
+
|
|
67
|
+
# Attacker skips to Step 3
|
|
68
|
+
|
|
69
|
+
# SAFE: Server-side state machine
|
|
70
|
+
class RegistrationFlow:
|
|
71
|
+
STATES = ['email_pending', 'email_verified', 'password_set', 'complete']
|
|
72
|
+
|
|
73
|
+
def __init__(self, user_id):
|
|
74
|
+
self.state = self.get_state(user_id)
|
|
75
|
+
|
|
76
|
+
def verify_email(self, token):
|
|
77
|
+
if self.state != 'email_pending':
|
|
78
|
+
raise InvalidStateError("Email verification not pending")
|
|
79
|
+
# Verify token...
|
|
80
|
+
self.set_state('email_verified')
|
|
81
|
+
|
|
82
|
+
def set_password(self, password):
|
|
83
|
+
if self.state != 'email_verified':
|
|
84
|
+
raise InvalidStateError("Email not verified")
|
|
85
|
+
# Set password...
|
|
86
|
+
self.set_state('password_set')
|
|
87
|
+
|
|
88
|
+
def complete(self):
|
|
89
|
+
if self.state != 'password_set':
|
|
90
|
+
raise InvalidStateError("Password not set")
|
|
91
|
+
# Complete registration...
|
|
92
|
+
self.set_state('complete')
|
|
93
|
+
```
|
|
94
|
+
|
|
95
|
+
### 3. Numeric Manipulation
|
|
96
|
+
|
|
97
|
+
#### Integer Overflow
|
|
98
|
+
|
|
99
|
+
```python
|
|
100
|
+
# VULNERABLE: Integer overflow in quantity
|
|
101
|
+
def calculate_total(quantity, price):
|
|
102
|
+
return quantity * price
|
|
103
|
+
|
|
104
|
+
# Attack: quantity = -1 results in negative price (refund)
|
|
105
|
+
|
|
106
|
+
# SAFE: Validate numeric ranges
|
|
107
|
+
def calculate_total(quantity, price):
|
|
108
|
+
if quantity <= 0 or quantity > MAX_QUANTITY:
|
|
109
|
+
raise ValueError("Invalid quantity")
|
|
110
|
+
if price <= 0:
|
|
111
|
+
raise ValueError("Invalid price")
|
|
112
|
+
return quantity * price
|
|
113
|
+
```
|
|
114
|
+
|
|
115
|
+
#### Floating Point Issues
|
|
116
|
+
|
|
117
|
+
```python
|
|
118
|
+
# VULNERABLE: Floating point precision loss
|
|
119
|
+
total = 0.0
|
|
120
|
+
for item in items:
|
|
121
|
+
total += item.price * item.quantity
|
|
122
|
+
|
|
123
|
+
# 0.1 + 0.2 = 0.30000000000000004
|
|
124
|
+
|
|
125
|
+
# SAFE: Use Decimal for financial calculations
|
|
126
|
+
from decimal import Decimal, ROUND_HALF_UP
|
|
127
|
+
|
|
128
|
+
total = Decimal('0')
|
|
129
|
+
for item in items:
|
|
130
|
+
total += Decimal(str(item.price)) * item.quantity
|
|
131
|
+
|
|
132
|
+
# Round properly
|
|
133
|
+
total = total.quantize(Decimal('.01'), rounding=ROUND_HALF_UP)
|
|
134
|
+
```
|
|
135
|
+
|
|
136
|
+
### 4. Price/Discount Manipulation
|
|
137
|
+
|
|
138
|
+
```python
|
|
139
|
+
# VULNERABLE: Trust client-submitted price
|
|
140
|
+
@app.route('/checkout', methods=['POST'])
|
|
141
|
+
def checkout():
|
|
142
|
+
price = request.json['price'] # Client can set any price!
|
|
143
|
+
process_payment(price)
|
|
144
|
+
|
|
145
|
+
# SAFE: Calculate price server-side
|
|
146
|
+
@app.route('/checkout', methods=['POST'])
|
|
147
|
+
def checkout():
|
|
148
|
+
cart = get_cart(current_user.id)
|
|
149
|
+
price = calculate_total(cart) # Always server-calculated
|
|
150
|
+
process_payment(price)
|
|
151
|
+
```
|
|
152
|
+
|
|
153
|
+
```python
|
|
154
|
+
# VULNERABLE: Stackable discounts without limits
|
|
155
|
+
def apply_discounts(cart, discount_codes):
|
|
156
|
+
for code in discount_codes:
|
|
157
|
+
discount = get_discount(code)
|
|
158
|
+
cart.total -= discount.amount
|
|
159
|
+
|
|
160
|
+
# Attack: Apply same code multiple times, negative total
|
|
161
|
+
|
|
162
|
+
# SAFE: Limit discount application
|
|
163
|
+
def apply_discounts(cart, discount_codes):
|
|
164
|
+
# Remove duplicates
|
|
165
|
+
unique_codes = set(discount_codes)
|
|
166
|
+
|
|
167
|
+
total_discount = Decimal('0')
|
|
168
|
+
for code in unique_codes:
|
|
169
|
+
if is_code_used(cart.user_id, code):
|
|
170
|
+
continue # Code already used
|
|
171
|
+
discount = get_discount(code)
|
|
172
|
+
total_discount += discount.amount
|
|
173
|
+
mark_code_used(cart.user_id, code)
|
|
174
|
+
|
|
175
|
+
# Cap discount at total
|
|
176
|
+
max_discount = cart.subtotal * Decimal('0.5') # Max 50% off
|
|
177
|
+
final_discount = min(total_discount, max_discount)
|
|
178
|
+
cart.total -= final_discount
|
|
179
|
+
```
|
|
180
|
+
|
|
181
|
+
### 5. Inventory/Resource Exhaustion
|
|
182
|
+
|
|
183
|
+
```python
|
|
184
|
+
# VULNERABLE: No reservation during checkout
|
|
185
|
+
def checkout(cart):
|
|
186
|
+
for item in cart.items:
|
|
187
|
+
if get_stock(item.product_id) >= item.quantity:
|
|
188
|
+
# Stock available
|
|
189
|
+
pass
|
|
190
|
+
# Processing takes time...
|
|
191
|
+
process_payment()
|
|
192
|
+
for item in cart.items:
|
|
193
|
+
reduce_stock(item.product_id, item.quantity) # May oversell
|
|
194
|
+
|
|
195
|
+
# SAFE: Reserve inventory atomically
|
|
196
|
+
@transaction.atomic
|
|
197
|
+
def checkout(cart):
|
|
198
|
+
for item in cart.items:
|
|
199
|
+
product = Product.objects.select_for_update().get(id=item.product_id)
|
|
200
|
+
if product.stock < item.quantity:
|
|
201
|
+
raise InsufficientStock(product.name)
|
|
202
|
+
product.stock -= item.quantity # Reserve immediately
|
|
203
|
+
product.save()
|
|
204
|
+
|
|
205
|
+
# If payment fails, transaction rolls back
|
|
206
|
+
process_payment()
|
|
207
|
+
```
|
|
208
|
+
|
|
209
|
+
### 6. Time-Based Attacks
|
|
210
|
+
|
|
211
|
+
```python
|
|
212
|
+
# VULNERABLE: Expired coupon still usable with timing attack
|
|
213
|
+
def apply_coupon(code):
|
|
214
|
+
coupon = Coupon.objects.get(code=code)
|
|
215
|
+
if coupon.expiry > datetime.now():
|
|
216
|
+
return coupon.discount
|
|
217
|
+
raise CouponExpired()
|
|
218
|
+
|
|
219
|
+
# SAFE: Use database time, not application time
|
|
220
|
+
from django.db.models.functions import Now
|
|
221
|
+
|
|
222
|
+
def apply_coupon(code):
|
|
223
|
+
coupon = Coupon.objects.annotate(
|
|
224
|
+
is_valid=Q(expiry__gt=Now())
|
|
225
|
+
).get(code=code)
|
|
226
|
+
|
|
227
|
+
if not coupon.is_valid:
|
|
228
|
+
raise CouponExpired()
|
|
229
|
+
return coupon.discount
|
|
230
|
+
```
|
|
231
|
+
|
|
232
|
+
### 7. Parameter Tampering
|
|
233
|
+
|
|
234
|
+
```python
|
|
235
|
+
# VULNERABLE: Trust hidden form fields
|
|
236
|
+
# HTML: <input type="hidden" name="user_id" value="123">
|
|
237
|
+
|
|
238
|
+
@app.route('/update-profile', methods=['POST'])
|
|
239
|
+
def update_profile():
|
|
240
|
+
user_id = request.form['user_id'] # Attacker can change this!
|
|
241
|
+
User.query.get(user_id).update(...)
|
|
242
|
+
|
|
243
|
+
# SAFE: Use session-based user identification
|
|
244
|
+
@app.route('/update-profile', methods=['POST'])
|
|
245
|
+
def update_profile():
|
|
246
|
+
user_id = current_user.id # From authenticated session
|
|
247
|
+
User.query.get(user_id).update(...)
|
|
248
|
+
```
|
|
249
|
+
|
|
250
|
+
---
|
|
251
|
+
|
|
252
|
+
## Detection Patterns
|
|
253
|
+
|
|
254
|
+
### State Machine Validation
|
|
255
|
+
|
|
256
|
+
```python
|
|
257
|
+
class OrderStateMachine:
|
|
258
|
+
VALID_TRANSITIONS = {
|
|
259
|
+
'draft': ['submitted'],
|
|
260
|
+
'submitted': ['approved', 'rejected'],
|
|
261
|
+
'approved': ['shipped'],
|
|
262
|
+
'shipped': ['delivered', 'returned'],
|
|
263
|
+
'delivered': ['returned'],
|
|
264
|
+
'rejected': [],
|
|
265
|
+
'returned': ['refunded'],
|
|
266
|
+
'refunded': []
|
|
267
|
+
}
|
|
268
|
+
|
|
269
|
+
def transition(self, order, new_state):
|
|
270
|
+
current = order.state
|
|
271
|
+
if new_state not in self.VALID_TRANSITIONS.get(current, []):
|
|
272
|
+
raise InvalidTransition(f"Cannot go from {current} to {new_state}")
|
|
273
|
+
order.state = new_state
|
|
274
|
+
log_state_change(order, current, new_state)
|
|
275
|
+
```
|
|
276
|
+
|
|
277
|
+
### Idempotency
|
|
278
|
+
|
|
279
|
+
```python
|
|
280
|
+
# SAFE: Idempotent operations with idempotency keys
|
|
281
|
+
import hashlib
|
|
282
|
+
|
|
283
|
+
def process_request(request_data, idempotency_key):
|
|
284
|
+
# Check if request was already processed
|
|
285
|
+
existing = ProcessedRequest.query.filter_by(key=idempotency_key).first()
|
|
286
|
+
if existing:
|
|
287
|
+
return existing.response # Return cached response
|
|
288
|
+
|
|
289
|
+
# Process request
|
|
290
|
+
result = do_processing(request_data)
|
|
291
|
+
|
|
292
|
+
# Store for future duplicate requests
|
|
293
|
+
ProcessedRequest.create(key=idempotency_key, response=result)
|
|
294
|
+
return result
|
|
295
|
+
```
|
|
296
|
+
|
|
297
|
+
### Rate Limiting Business Actions
|
|
298
|
+
|
|
299
|
+
```python
|
|
300
|
+
# Limit business-critical actions
|
|
301
|
+
from functools import wraps
|
|
302
|
+
import time
|
|
303
|
+
|
|
304
|
+
def rate_limit_action(action_name, limit, window):
|
|
305
|
+
def decorator(f):
|
|
306
|
+
@wraps(f)
|
|
307
|
+
def wrapper(*args, **kwargs):
|
|
308
|
+
user_id = current_user.id
|
|
309
|
+
key = f"action:{action_name}:{user_id}"
|
|
310
|
+
|
|
311
|
+
count = redis.incr(key)
|
|
312
|
+
if count == 1:
|
|
313
|
+
redis.expire(key, window)
|
|
314
|
+
|
|
315
|
+
if count > limit:
|
|
316
|
+
raise RateLimitExceeded(f"Too many {action_name} attempts")
|
|
317
|
+
|
|
318
|
+
return f(*args, **kwargs)
|
|
319
|
+
return wrapper
|
|
320
|
+
return decorator
|
|
321
|
+
|
|
322
|
+
@rate_limit_action('password_reset', limit=3, window=3600)
|
|
323
|
+
def request_password_reset(email):
|
|
324
|
+
pass
|
|
325
|
+
|
|
326
|
+
@rate_limit_action('transfer', limit=10, window=86400)
|
|
327
|
+
def transfer_funds(from_account, to_account, amount):
|
|
328
|
+
pass
|
|
329
|
+
```
|
|
330
|
+
|
|
331
|
+
---
|
|
332
|
+
|
|
333
|
+
## Validation Patterns
|
|
334
|
+
|
|
335
|
+
### Server-Side Calculation
|
|
336
|
+
|
|
337
|
+
```python
|
|
338
|
+
# Always recalculate on server
|
|
339
|
+
def calculate_order_total(order):
|
|
340
|
+
subtotal = Decimal('0')
|
|
341
|
+
for item in order.items:
|
|
342
|
+
# Get current price from database, not from request
|
|
343
|
+
product = Product.query.get(item.product_id)
|
|
344
|
+
subtotal += product.price * item.quantity
|
|
345
|
+
|
|
346
|
+
# Apply tax
|
|
347
|
+
tax = subtotal * get_tax_rate(order.shipping_address)
|
|
348
|
+
|
|
349
|
+
# Apply discounts (validated server-side)
|
|
350
|
+
discount = calculate_discounts(order, order.discount_codes)
|
|
351
|
+
|
|
352
|
+
# Calculate total
|
|
353
|
+
total = subtotal + tax - discount
|
|
354
|
+
|
|
355
|
+
# Sanity checks
|
|
356
|
+
if total < Decimal('0'):
|
|
357
|
+
raise InvalidOrderError("Negative total")
|
|
358
|
+
if discount > subtotal:
|
|
359
|
+
raise InvalidOrderError("Discount exceeds subtotal")
|
|
360
|
+
|
|
361
|
+
return {
|
|
362
|
+
'subtotal': subtotal,
|
|
363
|
+
'tax': tax,
|
|
364
|
+
'discount': discount,
|
|
365
|
+
'total': total
|
|
366
|
+
}
|
|
367
|
+
```
|
|
368
|
+
|
|
369
|
+
### Business Rule Enforcement
|
|
370
|
+
|
|
371
|
+
```python
|
|
372
|
+
class TransferValidator:
|
|
373
|
+
def validate(self, transfer):
|
|
374
|
+
errors = []
|
|
375
|
+
|
|
376
|
+
# Check transfer limits
|
|
377
|
+
if transfer.amount > MAX_SINGLE_TRANSFER:
|
|
378
|
+
errors.append("Exceeds single transfer limit")
|
|
379
|
+
|
|
380
|
+
# Check daily limits
|
|
381
|
+
daily_total = get_daily_transfer_total(transfer.from_account)
|
|
382
|
+
if daily_total + transfer.amount > DAILY_LIMIT:
|
|
383
|
+
errors.append("Exceeds daily transfer limit")
|
|
384
|
+
|
|
385
|
+
# Check velocity (unusual number of transfers)
|
|
386
|
+
recent_count = get_recent_transfer_count(transfer.from_account, hours=1)
|
|
387
|
+
if recent_count > MAX_TRANSFERS_PER_HOUR:
|
|
388
|
+
errors.append("Too many transfers in short period")
|
|
389
|
+
|
|
390
|
+
# Check for unusual patterns
|
|
391
|
+
if is_unusual_recipient(transfer.from_account, transfer.to_account):
|
|
392
|
+
errors.append("Unusual recipient - requires verification")
|
|
393
|
+
|
|
394
|
+
if errors:
|
|
395
|
+
raise ValidationError(errors)
|
|
396
|
+
```
|
|
397
|
+
|
|
398
|
+
---
|
|
399
|
+
|
|
400
|
+
## Grep Patterns for Detection
|
|
401
|
+
|
|
402
|
+
```bash
|
|
403
|
+
# Race condition indicators
|
|
404
|
+
grep -rn "sleep\|time\.sleep\|Thread\|async" --include="*.py"
|
|
405
|
+
grep -rn "balance\|inventory\|stock" --include="*.py" | grep -v "select_for_update\|lock"
|
|
406
|
+
|
|
407
|
+
# Price/amount from request
|
|
408
|
+
grep -rn "request\.\w*\[.*price\|request\.\w*\[.*amount\|request\.\w*\[.*total" --include="*.py"
|
|
409
|
+
|
|
410
|
+
# Missing validation
|
|
411
|
+
grep -rn "def checkout\|def purchase\|def transfer" --include="*.py"
|
|
412
|
+
|
|
413
|
+
# Floating point for money
|
|
414
|
+
grep -rn "float.*price\|float.*amount\|float.*balance" --include="*.py"
|
|
415
|
+
```
|
|
416
|
+
|
|
417
|
+
---
|
|
418
|
+
|
|
419
|
+
## Testing Checklist
|
|
420
|
+
|
|
421
|
+
- [ ] Race conditions tested (concurrent requests)
|
|
422
|
+
- [ ] Workflow steps enforced server-side
|
|
423
|
+
- [ ] State transitions validated
|
|
424
|
+
- [ ] Prices/totals calculated server-side
|
|
425
|
+
- [ ] Discount limits enforced
|
|
426
|
+
- [ ] Inventory checked and reserved atomically
|
|
427
|
+
- [ ] Integer overflow/underflow prevented
|
|
428
|
+
- [ ] Decimal used for financial calculations
|
|
429
|
+
- [ ] Time-based logic uses server/database time
|
|
430
|
+
- [ ] Hidden field values not trusted
|
|
431
|
+
- [ ] Idempotency keys for critical operations
|
|
432
|
+
- [ ] Rate limits on business-critical actions
|
|
433
|
+
- [ ] Unusual patterns detected and flagged
|
|
434
|
+
|
|
435
|
+
---
|
|
436
|
+
|
|
437
|
+
## References
|
|
438
|
+
|
|
439
|
+
- [OWASP Business Logic Testing](https://owasp.org/www-project-web-security-testing-guide/latest/4-Web_Application_Security_Testing/10-Business_Logic_Testing/)
|
|
440
|
+
- [CWE-362: Race Condition](https://cwe.mitre.org/data/definitions/362.html)
|
|
441
|
+
- [CWE-367: TOCTOU Race Condition](https://cwe.mitre.org/data/definitions/367.html)
|
|
442
|
+
- [CWE-190: Integer Overflow](https://cwe.mitre.org/data/definitions/190.html)
|
|
443
|
+
- [CWE-840: Business Logic Errors](https://cwe.mitre.org/data/definitions/840.html)
|