claude-mpm 4.14.6__py3-none-any.whl → 4.21.0__py3-none-any.whl
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.
Potentially problematic release.
This version of claude-mpm might be problematic. Click here for more details.
- claude_mpm/VERSION +1 -1
- claude_mpm/agents/BASE_ENGINEER.md +286 -0
- claude_mpm/agents/BASE_PM.md +272 -23
- claude_mpm/agents/OUTPUT_STYLE.md +48 -3
- claude_mpm/agents/PM_INSTRUCTIONS.md +49 -0
- claude_mpm/agents/agent_loader.py +17 -5
- claude_mpm/agents/frontmatter_validator.py +284 -253
- claude_mpm/agents/templates/agentic-coder-optimizer.json +9 -2
- claude_mpm/agents/templates/api_qa.json +7 -1
- claude_mpm/agents/templates/clerk-ops.json +8 -1
- claude_mpm/agents/templates/code_analyzer.json +4 -1
- claude_mpm/agents/templates/dart_engineer.json +11 -1
- claude_mpm/agents/templates/data_engineer.json +11 -1
- claude_mpm/agents/templates/documentation.json +6 -1
- claude_mpm/agents/templates/engineer.json +18 -1
- claude_mpm/agents/templates/gcp_ops_agent.json +8 -1
- claude_mpm/agents/templates/golang_engineer.json +11 -1
- claude_mpm/agents/templates/java_engineer.json +12 -2
- claude_mpm/agents/templates/local_ops_agent.json +216 -37
- claude_mpm/agents/templates/nextjs_engineer.json +11 -1
- claude_mpm/agents/templates/ops.json +8 -1
- claude_mpm/agents/templates/php-engineer.json +20 -4
- claude_mpm/agents/templates/project_organizer.json +10 -3
- claude_mpm/agents/templates/prompt-engineer.json +5 -1
- claude_mpm/agents/templates/python_engineer.json +19 -4
- claude_mpm/agents/templates/qa.json +7 -1
- claude_mpm/agents/templates/react_engineer.json +11 -1
- claude_mpm/agents/templates/refactoring_engineer.json +8 -1
- claude_mpm/agents/templates/research.json +4 -1
- claude_mpm/agents/templates/ruby-engineer.json +11 -1
- claude_mpm/agents/templates/rust_engineer.json +23 -8
- claude_mpm/agents/templates/security.json +6 -1
- claude_mpm/agents/templates/svelte-engineer.json +225 -0
- claude_mpm/agents/templates/ticketing.json +6 -1
- claude_mpm/agents/templates/typescript_engineer.json +11 -1
- claude_mpm/agents/templates/vercel_ops_agent.json +8 -1
- claude_mpm/agents/templates/version_control.json +8 -1
- claude_mpm/agents/templates/web_qa.json +7 -1
- claude_mpm/agents/templates/web_ui.json +11 -1
- claude_mpm/cli/__init__.py +34 -740
- claude_mpm/cli/commands/__init__.py +2 -0
- claude_mpm/cli/commands/agent_manager.py +25 -12
- claude_mpm/cli/commands/agent_state_manager.py +186 -0
- claude_mpm/cli/commands/agents.py +204 -148
- claude_mpm/cli/commands/aggregate.py +7 -3
- claude_mpm/cli/commands/analyze.py +9 -4
- claude_mpm/cli/commands/analyze_code.py +7 -2
- claude_mpm/cli/commands/auto_configure.py +7 -9
- claude_mpm/cli/commands/config.py +47 -13
- claude_mpm/cli/commands/configure.py +294 -1788
- claude_mpm/cli/commands/configure_agent_display.py +261 -0
- claude_mpm/cli/commands/configure_behavior_manager.py +204 -0
- claude_mpm/cli/commands/configure_hook_manager.py +225 -0
- claude_mpm/cli/commands/configure_models.py +18 -0
- claude_mpm/cli/commands/configure_navigation.py +167 -0
- claude_mpm/cli/commands/configure_paths.py +104 -0
- claude_mpm/cli/commands/configure_persistence.py +254 -0
- claude_mpm/cli/commands/configure_startup_manager.py +646 -0
- claude_mpm/cli/commands/configure_template_editor.py +497 -0
- claude_mpm/cli/commands/configure_validators.py +73 -0
- claude_mpm/cli/commands/local_deploy.py +3 -2
- claude_mpm/cli/commands/memory.py +54 -20
- claude_mpm/cli/commands/mpm_init/__init__.py +73 -0
- claude_mpm/cli/commands/mpm_init/core.py +525 -0
- claude_mpm/cli/commands/mpm_init/display.py +341 -0
- claude_mpm/cli/commands/mpm_init/git_activity.py +427 -0
- claude_mpm/cli/commands/mpm_init/modes.py +397 -0
- claude_mpm/cli/commands/mpm_init/prompts.py +442 -0
- claude_mpm/cli/commands/mpm_init_cli.py +396 -0
- claude_mpm/cli/commands/mpm_init_handler.py +75 -4
- claude_mpm/cli/commands/skills.py +488 -0
- claude_mpm/cli/executor.py +204 -0
- claude_mpm/cli/helpers.py +105 -0
- claude_mpm/cli/interactive/__init__.py +3 -0
- claude_mpm/cli/interactive/skills_wizard.py +491 -0
- claude_mpm/cli/parsers/base_parser.py +7 -0
- claude_mpm/cli/parsers/mpm_init_parser.py +42 -0
- claude_mpm/cli/parsers/skills_parser.py +137 -0
- claude_mpm/cli/shared/output_formatters.py +28 -19
- claude_mpm/cli/startup.py +538 -0
- claude_mpm/commands/mpm-auto-configure.md +52 -0
- claude_mpm/commands/mpm-help.md +3 -0
- claude_mpm/commands/mpm-init.md +112 -6
- claude_mpm/commands/mpm-version.md +113 -0
- claude_mpm/commands/mpm.md +1 -0
- claude_mpm/config/agent_config.py +2 -2
- claude_mpm/constants.py +12 -0
- claude_mpm/core/base_service.py +13 -12
- claude_mpm/core/config.py +42 -0
- claude_mpm/core/enums.py +452 -0
- claude_mpm/core/factories.py +1 -1
- claude_mpm/core/instruction_reinforcement_hook.py +2 -1
- claude_mpm/core/interactive_session.py +6 -3
- claude_mpm/core/interfaces.py +56 -1
- claude_mpm/core/logging_config.py +6 -2
- claude_mpm/core/oneshot_session.py +8 -4
- claude_mpm/core/optimized_agent_loader.py +3 -3
- claude_mpm/core/output_style_manager.py +12 -192
- claude_mpm/core/service_registry.py +5 -1
- claude_mpm/core/types.py +2 -9
- claude_mpm/core/typing_utils.py +7 -6
- claude_mpm/dashboard/static/js/dashboard.js +0 -14
- claude_mpm/dashboard/templates/index.html +3 -41
- claude_mpm/hooks/__init__.py +8 -0
- claude_mpm/hooks/claude_hooks/response_tracking.py +35 -1
- claude_mpm/hooks/instruction_reinforcement.py +7 -2
- claude_mpm/hooks/session_resume_hook.py +121 -0
- claude_mpm/models/resume_log.py +340 -0
- claude_mpm/services/agents/auto_config_manager.py +10 -11
- claude_mpm/services/agents/deployment/agent_configuration_manager.py +1 -1
- claude_mpm/services/agents/deployment/agent_record_service.py +1 -1
- claude_mpm/services/agents/deployment/agent_validator.py +17 -1
- claude_mpm/services/agents/deployment/async_agent_deployment.py +1 -1
- claude_mpm/services/agents/deployment/interface_adapter.py +3 -2
- claude_mpm/services/agents/deployment/local_template_deployment.py +1 -1
- claude_mpm/services/agents/deployment/pipeline/steps/agent_processing_step.py +7 -6
- claude_mpm/services/agents/deployment/pipeline/steps/base_step.py +7 -16
- claude_mpm/services/agents/deployment/pipeline/steps/configuration_step.py +4 -3
- claude_mpm/services/agents/deployment/pipeline/steps/target_directory_step.py +5 -3
- claude_mpm/services/agents/deployment/pipeline/steps/validation_step.py +6 -5
- claude_mpm/services/agents/deployment/refactored_agent_deployment_service.py +9 -6
- claude_mpm/services/agents/deployment/validation/__init__.py +3 -1
- claude_mpm/services/agents/deployment/validation/validation_result.py +1 -9
- claude_mpm/services/agents/local_template_manager.py +1 -1
- claude_mpm/services/agents/memory/agent_memory_manager.py +5 -2
- claude_mpm/services/agents/recommender.py +47 -0
- claude_mpm/services/agents/registry/modification_tracker.py +5 -2
- claude_mpm/services/cli/resume_service.py +617 -0
- claude_mpm/services/cli/session_manager.py +87 -0
- claude_mpm/services/cli/session_pause_manager.py +504 -0
- claude_mpm/services/cli/session_resume_helper.py +372 -0
- claude_mpm/services/command_handler_service.py +11 -5
- claude_mpm/services/core/interfaces/process.py +6 -6
- claude_mpm/services/core/interfaces.py +56 -1
- claude_mpm/services/core/models/__init__.py +0 -2
- claude_mpm/services/core/models/agent_config.py +15 -28
- claude_mpm/services/core/models/health.py +1 -28
- claude_mpm/services/core/models/process.py +22 -41
- claude_mpm/services/core/path_resolver.py +1 -1
- claude_mpm/services/diagnostics/__init__.py +2 -2
- claude_mpm/services/diagnostics/checks/agent_check.py +25 -24
- claude_mpm/services/diagnostics/checks/claude_code_check.py +24 -23
- claude_mpm/services/diagnostics/checks/common_issues_check.py +25 -24
- claude_mpm/services/diagnostics/checks/configuration_check.py +24 -23
- claude_mpm/services/diagnostics/checks/filesystem_check.py +18 -17
- claude_mpm/services/diagnostics/checks/installation_check.py +30 -29
- claude_mpm/services/diagnostics/checks/instructions_check.py +20 -19
- claude_mpm/services/diagnostics/checks/mcp_check.py +50 -36
- claude_mpm/services/diagnostics/checks/mcp_services_check.py +36 -31
- claude_mpm/services/diagnostics/checks/monitor_check.py +23 -22
- claude_mpm/services/diagnostics/checks/startup_log_check.py +9 -8
- claude_mpm/services/diagnostics/diagnostic_runner.py +6 -5
- claude_mpm/services/diagnostics/doctor_reporter.py +28 -25
- claude_mpm/services/diagnostics/models.py +37 -21
- claude_mpm/services/infrastructure/monitoring/__init__.py +1 -1
- claude_mpm/services/infrastructure/monitoring/aggregator.py +12 -12
- claude_mpm/services/infrastructure/monitoring/base.py +5 -13
- claude_mpm/services/infrastructure/monitoring/network.py +7 -6
- claude_mpm/services/infrastructure/monitoring/process.py +13 -12
- claude_mpm/services/infrastructure/monitoring/resources.py +7 -6
- claude_mpm/services/infrastructure/monitoring/service.py +16 -15
- claude_mpm/services/infrastructure/resume_log_generator.py +439 -0
- claude_mpm/services/local_ops/__init__.py +5 -3
- claude_mpm/services/local_ops/crash_detector.py +1 -1
- claude_mpm/services/local_ops/health_checks/http_check.py +2 -1
- claude_mpm/services/local_ops/health_checks/process_check.py +2 -1
- claude_mpm/services/local_ops/health_checks/resource_check.py +2 -1
- claude_mpm/services/local_ops/health_manager.py +1 -1
- claude_mpm/services/local_ops/process_manager.py +12 -12
- claude_mpm/services/local_ops/restart_manager.py +1 -1
- claude_mpm/services/local_ops/state_manager.py +6 -5
- claude_mpm/services/local_ops/unified_manager.py +2 -2
- claude_mpm/services/mcp_config_manager.py +7 -126
- claude_mpm/services/mcp_gateway/auto_configure.py +31 -25
- claude_mpm/services/mcp_gateway/core/__init__.py +1 -2
- claude_mpm/services/mcp_gateway/core/base.py +18 -31
- claude_mpm/services/mcp_gateway/core/process_pool.py +19 -10
- claude_mpm/services/mcp_gateway/tools/external_mcp_services.py +97 -45
- claude_mpm/services/mcp_gateway/tools/health_check_tool.py +30 -28
- claude_mpm/services/memory_hook_service.py +4 -1
- claude_mpm/services/monitor/daemon_manager.py +3 -2
- claude_mpm/services/monitor/handlers/dashboard.py +2 -1
- claude_mpm/services/monitor/handlers/hooks.py +2 -1
- claude_mpm/services/monitor/management/lifecycle.py +3 -2
- claude_mpm/services/monitor/server.py +2 -1
- claude_mpm/services/session_management_service.py +3 -2
- claude_mpm/services/session_manager.py +205 -1
- claude_mpm/services/shared/async_service_base.py +16 -27
- claude_mpm/services/shared/lifecycle_service_base.py +1 -14
- claude_mpm/services/socketio/handlers/__init__.py +5 -2
- claude_mpm/services/socketio/handlers/hook.py +13 -2
- claude_mpm/services/socketio/handlers/registry.py +4 -2
- claude_mpm/services/socketio/server/main.py +10 -8
- claude_mpm/services/subprocess_launcher_service.py +14 -5
- claude_mpm/services/unified/analyzer_strategies/code_analyzer.py +8 -7
- claude_mpm/services/unified/analyzer_strategies/dependency_analyzer.py +6 -5
- claude_mpm/services/unified/analyzer_strategies/performance_analyzer.py +8 -7
- claude_mpm/services/unified/analyzer_strategies/security_analyzer.py +7 -6
- claude_mpm/services/unified/analyzer_strategies/structure_analyzer.py +5 -4
- claude_mpm/services/unified/config_strategies/validation_strategy.py +13 -9
- claude_mpm/services/unified/deployment_strategies/cloud_strategies.py +10 -3
- claude_mpm/services/unified/deployment_strategies/local.py +6 -5
- claude_mpm/services/unified/deployment_strategies/utils.py +6 -5
- claude_mpm/services/unified/deployment_strategies/vercel.py +7 -6
- claude_mpm/services/unified/interfaces.py +3 -1
- claude_mpm/services/unified/unified_analyzer.py +14 -10
- claude_mpm/services/unified/unified_config.py +2 -1
- claude_mpm/services/unified/unified_deployment.py +9 -4
- claude_mpm/services/version_service.py +104 -1
- claude_mpm/skills/__init__.py +42 -0
- claude_mpm/skills/agent_skills_injector.py +324 -0
- claude_mpm/skills/bundled/LICENSE_ATTRIBUTIONS.md +79 -0
- claude_mpm/skills/bundled/__init__.py +6 -0
- claude_mpm/skills/bundled/api-documentation.md +393 -0
- claude_mpm/skills/bundled/async-testing.md +571 -0
- claude_mpm/skills/bundled/code-review.md +143 -0
- claude_mpm/skills/bundled/collaboration/brainstorming/SKILL.md +79 -0
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/SKILL.md +178 -0
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/agent-prompts.md +577 -0
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/coordination-patterns.md +467 -0
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/examples.md +537 -0
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/troubleshooting.md +730 -0
- claude_mpm/skills/bundled/collaboration/requesting-code-review/SKILL.md +112 -0
- claude_mpm/skills/bundled/collaboration/requesting-code-review/references/code-reviewer-template.md +146 -0
- claude_mpm/skills/bundled/collaboration/requesting-code-review/references/review-examples.md +412 -0
- claude_mpm/skills/bundled/collaboration/writing-plans/SKILL.md +81 -0
- claude_mpm/skills/bundled/collaboration/writing-plans/references/best-practices.md +362 -0
- claude_mpm/skills/bundled/collaboration/writing-plans/references/plan-structure-templates.md +312 -0
- claude_mpm/skills/bundled/database-migration.md +199 -0
- claude_mpm/skills/bundled/debugging/root-cause-tracing/SKILL.md +152 -0
- claude_mpm/skills/bundled/debugging/root-cause-tracing/references/advanced-techniques.md +668 -0
- claude_mpm/skills/bundled/debugging/root-cause-tracing/references/examples.md +587 -0
- claude_mpm/skills/bundled/debugging/root-cause-tracing/references/integration.md +438 -0
- claude_mpm/skills/bundled/debugging/root-cause-tracing/references/tracing-techniques.md +391 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/CREATION-LOG.md +119 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/SKILL.md +148 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/references/anti-patterns.md +483 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/references/examples.md +452 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/references/troubleshooting.md +449 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/references/workflow.md +411 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/test-academic.md +14 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-1.md +58 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-2.md +68 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-3.md +69 -0
- claude_mpm/skills/bundled/debugging/verification-before-completion/SKILL.md +131 -0
- claude_mpm/skills/bundled/debugging/verification-before-completion/references/gate-function.md +325 -0
- claude_mpm/skills/bundled/debugging/verification-before-completion/references/integration-and-workflows.md +490 -0
- claude_mpm/skills/bundled/debugging/verification-before-completion/references/red-flags-and-failures.md +425 -0
- claude_mpm/skills/bundled/debugging/verification-before-completion/references/verification-patterns.md +499 -0
- claude_mpm/skills/bundled/docker-containerization.md +194 -0
- claude_mpm/skills/bundled/express-local-dev.md +1429 -0
- claude_mpm/skills/bundled/fastapi-local-dev.md +1199 -0
- claude_mpm/skills/bundled/git-workflow.md +414 -0
- claude_mpm/skills/bundled/imagemagick.md +204 -0
- claude_mpm/skills/bundled/json-data-handling.md +223 -0
- claude_mpm/skills/bundled/main/artifacts-builder/SKILL.md +86 -0
- claude_mpm/skills/bundled/main/internal-comms/SKILL.md +43 -0
- claude_mpm/skills/bundled/main/internal-comms/examples/3p-updates.md +47 -0
- claude_mpm/skills/bundled/main/internal-comms/examples/company-newsletter.md +65 -0
- claude_mpm/skills/bundled/main/internal-comms/examples/faq-answers.md +30 -0
- claude_mpm/skills/bundled/main/internal-comms/examples/general-comms.md +16 -0
- claude_mpm/skills/bundled/main/mcp-builder/SKILL.md +160 -0
- claude_mpm/skills/bundled/main/mcp-builder/reference/design_principles.md +412 -0
- claude_mpm/skills/bundled/main/mcp-builder/reference/evaluation.md +602 -0
- claude_mpm/skills/bundled/main/mcp-builder/reference/mcp_best_practices.md +915 -0
- claude_mpm/skills/bundled/main/mcp-builder/reference/node_mcp_server.md +916 -0
- claude_mpm/skills/bundled/main/mcp-builder/reference/python_mcp_server.md +752 -0
- claude_mpm/skills/bundled/main/mcp-builder/reference/workflow.md +1237 -0
- claude_mpm/skills/bundled/main/mcp-builder/scripts/connections.py +157 -0
- claude_mpm/skills/bundled/main/mcp-builder/scripts/evaluation.py +425 -0
- claude_mpm/skills/bundled/main/skill-creator/SKILL.md +189 -0
- claude_mpm/skills/bundled/main/skill-creator/references/best-practices.md +500 -0
- claude_mpm/skills/bundled/main/skill-creator/references/creation-workflow.md +464 -0
- claude_mpm/skills/bundled/main/skill-creator/references/examples.md +619 -0
- claude_mpm/skills/bundled/main/skill-creator/references/progressive-disclosure.md +437 -0
- claude_mpm/skills/bundled/main/skill-creator/references/skill-structure.md +231 -0
- claude_mpm/skills/bundled/main/skill-creator/scripts/init_skill.py +303 -0
- claude_mpm/skills/bundled/main/skill-creator/scripts/package_skill.py +113 -0
- claude_mpm/skills/bundled/main/skill-creator/scripts/quick_validate.py +72 -0
- claude_mpm/skills/bundled/nextjs-local-dev.md +807 -0
- claude_mpm/skills/bundled/pdf.md +141 -0
- claude_mpm/skills/bundled/performance-profiling.md +567 -0
- claude_mpm/skills/bundled/php/espocrm-development/SKILL.md +170 -0
- claude_mpm/skills/bundled/php/espocrm-development/references/architecture.md +602 -0
- claude_mpm/skills/bundled/php/espocrm-development/references/common-tasks.md +821 -0
- claude_mpm/skills/bundled/php/espocrm-development/references/development-workflow.md +742 -0
- claude_mpm/skills/bundled/php/espocrm-development/references/frontend-customization.md +726 -0
- claude_mpm/skills/bundled/php/espocrm-development/references/hooks-and-services.md +764 -0
- claude_mpm/skills/bundled/php/espocrm-development/references/testing-debugging.md +831 -0
- claude_mpm/skills/bundled/refactoring-patterns.md +180 -0
- claude_mpm/skills/bundled/rust/desktop-applications/SKILL.md +226 -0
- claude_mpm/skills/bundled/rust/desktop-applications/references/architecture-patterns.md +901 -0
- claude_mpm/skills/bundled/rust/desktop-applications/references/native-gui-frameworks.md +901 -0
- claude_mpm/skills/bundled/rust/desktop-applications/references/platform-integration.md +775 -0
- claude_mpm/skills/bundled/rust/desktop-applications/references/state-management.md +937 -0
- claude_mpm/skills/bundled/rust/desktop-applications/references/tauri-framework.md +770 -0
- claude_mpm/skills/bundled/rust/desktop-applications/references/testing-deployment.md +961 -0
- claude_mpm/skills/bundled/security-scanning.md +327 -0
- claude_mpm/skills/bundled/systematic-debugging.md +473 -0
- claude_mpm/skills/bundled/test-driven-development.md +378 -0
- claude_mpm/skills/bundled/testing/condition-based-waiting/SKILL.md +119 -0
- claude_mpm/skills/bundled/testing/condition-based-waiting/references/patterns-and-implementation.md +253 -0
- claude_mpm/skills/bundled/testing/test-driven-development/SKILL.md +145 -0
- claude_mpm/skills/bundled/testing/test-driven-development/references/anti-patterns.md +543 -0
- claude_mpm/skills/bundled/testing/test-driven-development/references/examples.md +741 -0
- claude_mpm/skills/bundled/testing/test-driven-development/references/integration.md +470 -0
- claude_mpm/skills/bundled/testing/test-driven-development/references/philosophy.md +458 -0
- claude_mpm/skills/bundled/testing/test-driven-development/references/workflow.md +639 -0
- claude_mpm/skills/bundled/testing/testing-anti-patterns/SKILL.md +140 -0
- claude_mpm/skills/bundled/testing/testing-anti-patterns/references/completeness-anti-patterns.md +572 -0
- claude_mpm/skills/bundled/testing/testing-anti-patterns/references/core-anti-patterns.md +411 -0
- claude_mpm/skills/bundled/testing/testing-anti-patterns/references/detection-guide.md +569 -0
- claude_mpm/skills/bundled/testing/testing-anti-patterns/references/tdd-connection.md +695 -0
- claude_mpm/skills/bundled/testing/webapp-testing/SKILL.md +184 -0
- claude_mpm/skills/bundled/testing/webapp-testing/decision-tree.md +459 -0
- claude_mpm/skills/bundled/testing/webapp-testing/examples/console_logging.py +35 -0
- claude_mpm/skills/bundled/testing/webapp-testing/examples/element_discovery.py +44 -0
- claude_mpm/skills/bundled/testing/webapp-testing/examples/static_html_automation.py +34 -0
- claude_mpm/skills/bundled/testing/webapp-testing/playwright-patterns.md +479 -0
- claude_mpm/skills/bundled/testing/webapp-testing/reconnaissance-pattern.md +687 -0
- claude_mpm/skills/bundled/testing/webapp-testing/scripts/with_server.py +129 -0
- claude_mpm/skills/bundled/testing/webapp-testing/server-management.md +758 -0
- claude_mpm/skills/bundled/testing/webapp-testing/troubleshooting.md +868 -0
- claude_mpm/skills/bundled/vite-local-dev.md +1061 -0
- claude_mpm/skills/bundled/web-performance-optimization.md +2305 -0
- claude_mpm/skills/bundled/xlsx.md +157 -0
- claude_mpm/skills/registry.py +286 -0
- claude_mpm/skills/skill_manager.py +310 -0
- claude_mpm/skills/skills_registry.py +348 -0
- claude_mpm/skills/skills_service.py +739 -0
- claude_mpm/tools/code_tree_analyzer.py +177 -141
- claude_mpm/tools/code_tree_events.py +4 -2
- claude_mpm/utils/agent_dependency_loader.py +2 -2
- {claude_mpm-4.14.6.dist-info → claude_mpm-4.21.0.dist-info}/METADATA +211 -33
- {claude_mpm-4.14.6.dist-info → claude_mpm-4.21.0.dist-info}/RECORD +339 -199
- claude_mpm/agents/INSTRUCTIONS_OLD_DEPRECATED.md +0 -602
- claude_mpm/cli/commands/mpm_init.py +0 -1994
- claude_mpm/dashboard/static/css/code-tree.css +0 -1639
- claude_mpm/dashboard/static/js/components/code-tree/tree-breadcrumb.js +0 -353
- claude_mpm/dashboard/static/js/components/code-tree/tree-constants.js +0 -235
- claude_mpm/dashboard/static/js/components/code-tree/tree-search.js +0 -409
- claude_mpm/dashboard/static/js/components/code-tree/tree-utils.js +0 -435
- claude_mpm/dashboard/static/js/components/code-tree.js +0 -5869
- claude_mpm/dashboard/static/js/components/code-viewer.js +0 -1386
- claude_mpm/hooks/claude_hooks/hook_handler_eventbus.py +0 -425
- claude_mpm/hooks/claude_hooks/hook_handler_original.py +0 -1041
- claude_mpm/hooks/claude_hooks/hook_handler_refactored.py +0 -347
- claude_mpm/services/agents/deployment/agent_lifecycle_manager_refactored.py +0 -575
- claude_mpm/services/project/analyzer_refactored.py +0 -450
- {claude_mpm-4.14.6.dist-info → claude_mpm-4.21.0.dist-info}/WHEEL +0 -0
- {claude_mpm-4.14.6.dist-info → claude_mpm-4.21.0.dist-info}/entry_points.txt +0 -0
- {claude_mpm-4.14.6.dist-info → claude_mpm-4.21.0.dist-info}/licenses/LICENSE +0 -0
- {claude_mpm-4.14.6.dist-info → claude_mpm-4.21.0.dist-info}/top_level.txt +0 -0
|
@@ -0,0 +1,378 @@
|
|
|
1
|
+
---
|
|
2
|
+
skill_id: test-driven-development
|
|
3
|
+
skill_version: 0.1.0
|
|
4
|
+
description: Comprehensive TDD patterns and practices for all programming languages, eliminating redundant testing guidance per agent.
|
|
5
|
+
updated_at: 2025-10-30T17:00:00Z
|
|
6
|
+
tags: [testing, tdd, best-practices, quality-assurance]
|
|
7
|
+
---
|
|
8
|
+
|
|
9
|
+
# Test-Driven Development (TDD)
|
|
10
|
+
|
|
11
|
+
Comprehensive TDD patterns and practices for all programming languages. This skill eliminates ~500-800 lines of redundant testing guidance per agent.
|
|
12
|
+
|
|
13
|
+
## When to Use
|
|
14
|
+
|
|
15
|
+
Apply TDD for:
|
|
16
|
+
- New feature implementation
|
|
17
|
+
- Bug fixes (test the bug first)
|
|
18
|
+
- Code refactoring (tests ensure behavior preservation)
|
|
19
|
+
- API development (test contracts)
|
|
20
|
+
- Complex business logic
|
|
21
|
+
|
|
22
|
+
## TDD Workflow (Red-Green-Refactor)
|
|
23
|
+
|
|
24
|
+
### 1. Red Phase: Write Failing Test
|
|
25
|
+
```
|
|
26
|
+
Write a test that:
|
|
27
|
+
- Describes the desired behavior
|
|
28
|
+
- Fails for the right reason (not due to syntax errors)
|
|
29
|
+
- Is focused on a single behavior
|
|
30
|
+
```
|
|
31
|
+
|
|
32
|
+
### 2. Green Phase: Make It Pass
|
|
33
|
+
```
|
|
34
|
+
Write the minimum code to:
|
|
35
|
+
- Pass the test
|
|
36
|
+
- Not introduce regressions
|
|
37
|
+
- Follow existing patterns
|
|
38
|
+
```
|
|
39
|
+
|
|
40
|
+
### 3. Refactor Phase: Improve Code
|
|
41
|
+
```
|
|
42
|
+
While keeping tests green:
|
|
43
|
+
- Remove duplication
|
|
44
|
+
- Improve naming
|
|
45
|
+
- Simplify logic
|
|
46
|
+
- Extract functions/classes
|
|
47
|
+
```
|
|
48
|
+
|
|
49
|
+
## Test Structure Patterns
|
|
50
|
+
|
|
51
|
+
### Arrange-Act-Assert (AAA)
|
|
52
|
+
```
|
|
53
|
+
// Arrange: Set up test data and conditions
|
|
54
|
+
const user = createTestUser({ role: 'admin' });
|
|
55
|
+
|
|
56
|
+
// Act: Perform the action being tested
|
|
57
|
+
const result = await authenticateUser(user);
|
|
58
|
+
|
|
59
|
+
// Assert: Verify the outcome
|
|
60
|
+
expect(result.isAuthenticated).toBe(true);
|
|
61
|
+
expect(result.permissions).toContain('admin');
|
|
62
|
+
```
|
|
63
|
+
|
|
64
|
+
### Given-When-Then (BDD Style)
|
|
65
|
+
```
|
|
66
|
+
Given: A user with admin privileges
|
|
67
|
+
When: They attempt to access protected resource
|
|
68
|
+
Then: Access is granted with appropriate permissions
|
|
69
|
+
```
|
|
70
|
+
|
|
71
|
+
## Test Naming Conventions
|
|
72
|
+
|
|
73
|
+
### Pattern: `test_should_<expected_behavior>_when_<condition>`
|
|
74
|
+
|
|
75
|
+
**Examples:**
|
|
76
|
+
- `test_should_return_user_when_id_exists()`
|
|
77
|
+
- `test_should_raise_error_when_user_not_found()`
|
|
78
|
+
- `test_should_validate_email_format_when_creating_account()`
|
|
79
|
+
|
|
80
|
+
### Language-Specific Conventions
|
|
81
|
+
|
|
82
|
+
**Python (pytest):**
|
|
83
|
+
```python
|
|
84
|
+
def test_should_calculate_total_when_items_added():
|
|
85
|
+
# Arrange
|
|
86
|
+
cart = ShoppingCart()
|
|
87
|
+
cart.add_item(Item("Book", 10.00))
|
|
88
|
+
cart.add_item(Item("Pen", 1.50))
|
|
89
|
+
|
|
90
|
+
# Act
|
|
91
|
+
total = cart.calculate_total()
|
|
92
|
+
|
|
93
|
+
# Assert
|
|
94
|
+
assert total == 11.50
|
|
95
|
+
```
|
|
96
|
+
|
|
97
|
+
**JavaScript (Jest):**
|
|
98
|
+
```javascript
|
|
99
|
+
describe('ShoppingCart', () => {
|
|
100
|
+
test('should calculate total when items added', () => {
|
|
101
|
+
const cart = new ShoppingCart();
|
|
102
|
+
cart.addItem({ name: 'Book', price: 10.00 });
|
|
103
|
+
cart.addItem({ name: 'Pen', price: 1.50 });
|
|
104
|
+
|
|
105
|
+
const total = cart.calculateTotal();
|
|
106
|
+
|
|
107
|
+
expect(total).toBe(11.50);
|
|
108
|
+
});
|
|
109
|
+
});
|
|
110
|
+
```
|
|
111
|
+
|
|
112
|
+
**Go:**
|
|
113
|
+
```go
|
|
114
|
+
func TestShouldCalculateTotalWhenItemsAdded(t *testing.T) {
|
|
115
|
+
// Arrange
|
|
116
|
+
cart := NewShoppingCart()
|
|
117
|
+
cart.AddItem(Item{Name: "Book", Price: 10.00})
|
|
118
|
+
cart.AddItem(Item{Name: "Pen", Price: 1.50})
|
|
119
|
+
|
|
120
|
+
// Act
|
|
121
|
+
total := cart.CalculateTotal()
|
|
122
|
+
|
|
123
|
+
// Assert
|
|
124
|
+
if total != 11.50 {
|
|
125
|
+
t.Errorf("Expected 11.50, got %f", total)
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
```
|
|
129
|
+
|
|
130
|
+
## Test Types and Scope
|
|
131
|
+
|
|
132
|
+
### Unit Tests
|
|
133
|
+
- **Scope:** Single function/method
|
|
134
|
+
- **Dependencies:** Mocked
|
|
135
|
+
- **Speed:** Fast (< 10ms per test)
|
|
136
|
+
- **Coverage:** 80%+ of code paths
|
|
137
|
+
|
|
138
|
+
### Integration Tests
|
|
139
|
+
- **Scope:** Multiple components
|
|
140
|
+
- **Dependencies:** Real or test doubles
|
|
141
|
+
- **Speed:** Moderate (< 1s per test)
|
|
142
|
+
- **Coverage:** Critical paths and interfaces
|
|
143
|
+
|
|
144
|
+
### End-to-End Tests
|
|
145
|
+
- **Scope:** Full user workflows
|
|
146
|
+
- **Dependencies:** Real (in test environment)
|
|
147
|
+
- **Speed:** Slow (seconds to minutes)
|
|
148
|
+
- **Coverage:** Core user journeys
|
|
149
|
+
|
|
150
|
+
## Mocking and Test Doubles
|
|
151
|
+
|
|
152
|
+
### When to Mock
|
|
153
|
+
- External APIs and services
|
|
154
|
+
- Database operations (for unit tests)
|
|
155
|
+
- File system operations
|
|
156
|
+
- Time-dependent operations
|
|
157
|
+
- Random number generation
|
|
158
|
+
|
|
159
|
+
### Mock Types
|
|
160
|
+
|
|
161
|
+
**Stub:** Returns predefined data
|
|
162
|
+
```python
|
|
163
|
+
def get_user_stub(user_id):
|
|
164
|
+
return User(id=user_id, name="Test User")
|
|
165
|
+
```
|
|
166
|
+
|
|
167
|
+
**Mock:** Verifies interactions
|
|
168
|
+
```python
|
|
169
|
+
mock_service = Mock()
|
|
170
|
+
service.process_payment(payment_data)
|
|
171
|
+
mock_service.process_payment.assert_called_once_with(payment_data)
|
|
172
|
+
```
|
|
173
|
+
|
|
174
|
+
**Fake:** Working implementation (simplified)
|
|
175
|
+
```python
|
|
176
|
+
class FakeDatabase:
|
|
177
|
+
def __init__(self):
|
|
178
|
+
self.data = {}
|
|
179
|
+
|
|
180
|
+
def save(self, key, value):
|
|
181
|
+
self.data[key] = value
|
|
182
|
+
|
|
183
|
+
def get(self, key):
|
|
184
|
+
return self.data.get(key)
|
|
185
|
+
```
|
|
186
|
+
|
|
187
|
+
## Test Coverage Guidelines
|
|
188
|
+
|
|
189
|
+
### Target Coverage Levels
|
|
190
|
+
- **Critical paths:** 100%
|
|
191
|
+
- **Business logic:** 95%+
|
|
192
|
+
- **Overall project:** 80%+
|
|
193
|
+
- **UI components:** 70%+
|
|
194
|
+
|
|
195
|
+
### What to Test
|
|
196
|
+
- ✅ Business logic and algorithms
|
|
197
|
+
- ✅ Edge cases and boundary conditions
|
|
198
|
+
- ✅ Error handling and validation
|
|
199
|
+
- ✅ State transitions
|
|
200
|
+
- ✅ Public APIs and interfaces
|
|
201
|
+
|
|
202
|
+
### What NOT to Test
|
|
203
|
+
- ❌ Framework internals
|
|
204
|
+
- ❌ Third-party libraries
|
|
205
|
+
- ❌ Trivial getters/setters
|
|
206
|
+
- ❌ Generated code
|
|
207
|
+
- ❌ Configuration files
|
|
208
|
+
|
|
209
|
+
## Testing Best Practices
|
|
210
|
+
|
|
211
|
+
### 1. One Assertion Per Test (When Possible)
|
|
212
|
+
```python
|
|
213
|
+
# Good: Focused test
|
|
214
|
+
def test_should_validate_email_format():
|
|
215
|
+
assert is_valid_email("user@example.com") is True
|
|
216
|
+
|
|
217
|
+
# Avoid: Multiple unrelated assertions
|
|
218
|
+
def test_validation():
|
|
219
|
+
assert is_valid_email("user@example.com") is True
|
|
220
|
+
assert is_valid_phone("123-456-7890") is True # Different concept
|
|
221
|
+
```
|
|
222
|
+
|
|
223
|
+
### 2. Test Independence
|
|
224
|
+
```python
|
|
225
|
+
# Good: Each test is self-contained
|
|
226
|
+
def test_user_creation():
|
|
227
|
+
user = create_user("test@example.com")
|
|
228
|
+
assert user.email == "test@example.com"
|
|
229
|
+
|
|
230
|
+
# Avoid: Tests depending on execution order
|
|
231
|
+
shared_user = None
|
|
232
|
+
|
|
233
|
+
def test_create_user():
|
|
234
|
+
global shared_user
|
|
235
|
+
shared_user = create_user("test@example.com")
|
|
236
|
+
|
|
237
|
+
def test_update_user(): # Depends on previous test
|
|
238
|
+
shared_user.name = "Updated"
|
|
239
|
+
```
|
|
240
|
+
|
|
241
|
+
### 3. Descriptive Test Failures
|
|
242
|
+
```python
|
|
243
|
+
# Good: Clear failure message
|
|
244
|
+
assert result.status == 200, f"Expected 200, got {result.status}: {result.body}"
|
|
245
|
+
|
|
246
|
+
# Avoid: Unclear failure
|
|
247
|
+
assert result.status == 200
|
|
248
|
+
```
|
|
249
|
+
|
|
250
|
+
### 4. Test Data Builders
|
|
251
|
+
```python
|
|
252
|
+
# Good: Reusable test data creation
|
|
253
|
+
def create_test_user(**overrides):
|
|
254
|
+
defaults = {
|
|
255
|
+
'email': 'test@example.com',
|
|
256
|
+
'name': 'Test User',
|
|
257
|
+
'role': 'user'
|
|
258
|
+
}
|
|
259
|
+
return User(**{**defaults, **overrides})
|
|
260
|
+
|
|
261
|
+
# Usage
|
|
262
|
+
admin = create_test_user(role='admin')
|
|
263
|
+
guest = create_test_user(email='guest@example.com')
|
|
264
|
+
```
|
|
265
|
+
|
|
266
|
+
## Testing Anti-Patterns to Avoid
|
|
267
|
+
|
|
268
|
+
### ❌ Testing Implementation Details
|
|
269
|
+
```python
|
|
270
|
+
# Bad: Tests internal structure
|
|
271
|
+
def test_user_storage():
|
|
272
|
+
user = User("test@example.com")
|
|
273
|
+
assert user._internal_cache is not None # Implementation detail
|
|
274
|
+
```
|
|
275
|
+
|
|
276
|
+
### ❌ Fragile Tests
|
|
277
|
+
```python
|
|
278
|
+
# Bad: Breaks with harmless changes
|
|
279
|
+
assert user.to_json() == '{"name":"John","email":"john@example.com"}'
|
|
280
|
+
|
|
281
|
+
# Good: Tests behavior, not format
|
|
282
|
+
data = json.loads(user.to_json())
|
|
283
|
+
assert data['name'] == "John"
|
|
284
|
+
assert data['email'] == "john@example.com"
|
|
285
|
+
```
|
|
286
|
+
|
|
287
|
+
### ❌ Slow Tests in Unit Test Suite
|
|
288
|
+
```python
|
|
289
|
+
# Bad: Real HTTP calls in unit tests
|
|
290
|
+
def test_api_integration():
|
|
291
|
+
response = requests.get("https://api.example.com/users") # Slow!
|
|
292
|
+
assert response.status_code == 200
|
|
293
|
+
```
|
|
294
|
+
|
|
295
|
+
### ❌ Testing Everything Through UI
|
|
296
|
+
```python
|
|
297
|
+
# Bad: Testing business logic through UI
|
|
298
|
+
def test_calculation():
|
|
299
|
+
browser.click("#input1")
|
|
300
|
+
browser.type("5")
|
|
301
|
+
browser.click("#input2")
|
|
302
|
+
browser.type("3")
|
|
303
|
+
browser.click("#calculate")
|
|
304
|
+
assert browser.find("#result").text == "8"
|
|
305
|
+
|
|
306
|
+
# Good: Test logic directly
|
|
307
|
+
def test_calculation():
|
|
308
|
+
assert calculate(5, 3) == 8
|
|
309
|
+
```
|
|
310
|
+
|
|
311
|
+
## Quick Reference by Language
|
|
312
|
+
|
|
313
|
+
### Python (pytest)
|
|
314
|
+
```python
|
|
315
|
+
# Setup/Teardown
|
|
316
|
+
@pytest.fixture
|
|
317
|
+
def database():
|
|
318
|
+
db = create_test_database()
|
|
319
|
+
yield db
|
|
320
|
+
db.cleanup()
|
|
321
|
+
|
|
322
|
+
# Parametrized tests
|
|
323
|
+
@pytest.mark.parametrize("input,expected", [
|
|
324
|
+
("user@example.com", True),
|
|
325
|
+
("invalid-email", False),
|
|
326
|
+
])
|
|
327
|
+
def test_email_validation(input, expected):
|
|
328
|
+
assert is_valid_email(input) == expected
|
|
329
|
+
```
|
|
330
|
+
|
|
331
|
+
### JavaScript (Jest)
|
|
332
|
+
```javascript
|
|
333
|
+
// Setup/Teardown
|
|
334
|
+
beforeEach(() => {
|
|
335
|
+
database = createTestDatabase();
|
|
336
|
+
});
|
|
337
|
+
|
|
338
|
+
afterEach(() => {
|
|
339
|
+
database.cleanup();
|
|
340
|
+
});
|
|
341
|
+
|
|
342
|
+
// Async tests
|
|
343
|
+
test('should fetch user data', async () => {
|
|
344
|
+
const user = await fetchUser(1);
|
|
345
|
+
expect(user.name).toBe('John');
|
|
346
|
+
});
|
|
347
|
+
```
|
|
348
|
+
|
|
349
|
+
### Go
|
|
350
|
+
```go
|
|
351
|
+
// Table-driven tests
|
|
352
|
+
func TestEmailValidation(t *testing.T) {
|
|
353
|
+
tests := []struct {
|
|
354
|
+
input string
|
|
355
|
+
expected bool
|
|
356
|
+
}{
|
|
357
|
+
{"user@example.com", true},
|
|
358
|
+
{"invalid-email", false},
|
|
359
|
+
}
|
|
360
|
+
|
|
361
|
+
for _, tt := range tests {
|
|
362
|
+
result := IsValidEmail(tt.input)
|
|
363
|
+
if result != tt.expected {
|
|
364
|
+
t.Errorf("IsValidEmail(%s) = %v, want %v",
|
|
365
|
+
tt.input, result, tt.expected)
|
|
366
|
+
}
|
|
367
|
+
}
|
|
368
|
+
}
|
|
369
|
+
```
|
|
370
|
+
|
|
371
|
+
## TDD Benefits Realized
|
|
372
|
+
|
|
373
|
+
- **Design Improvement:** Tests drive better API design
|
|
374
|
+
- **Documentation:** Tests serve as executable documentation
|
|
375
|
+
- **Confidence:** Refactoring becomes safe
|
|
376
|
+
- **Debugging:** Tests isolate issues quickly
|
|
377
|
+
- **Coverage:** Ensures comprehensive test coverage
|
|
378
|
+
- **Regression Prevention:** Catches bugs before deployment
|
|
@@ -0,0 +1,119 @@
|
|
|
1
|
+
---
|
|
2
|
+
name: Condition-Based Waiting
|
|
3
|
+
description: Replace arbitrary timeouts with condition polling for reliable async tests
|
|
4
|
+
when_to_use: when tests have race conditions, timing dependencies, or inconsistent pass/fail behavior
|
|
5
|
+
version: 1.1.0
|
|
6
|
+
languages: all
|
|
7
|
+
progressive_disclosure:
|
|
8
|
+
entry_point:
|
|
9
|
+
summary: "Replace arbitrary timeouts with condition polling for reliable async tests"
|
|
10
|
+
when_to_use: "Tests with setTimeout/sleep, flaky tests, or timing-dependent async operations"
|
|
11
|
+
quick_start: |
|
|
12
|
+
1. Identify arbitrary delays in tests (setTimeout, sleep, time.sleep())
|
|
13
|
+
2. Replace with condition-based waiting (waitFor pattern)
|
|
14
|
+
3. Use domain-specific helpers for common scenarios
|
|
15
|
+
See @example.ts for complete working implementation
|
|
16
|
+
core_pattern: |
|
|
17
|
+
// ❌ Guessing at timing
|
|
18
|
+
await new Promise(r => setTimeout(r, 50));
|
|
19
|
+
|
|
20
|
+
// ✅ Waiting for condition
|
|
21
|
+
await waitFor(() => getResult() !== undefined);
|
|
22
|
+
references:
|
|
23
|
+
- path: references/patterns-and-implementation.md
|
|
24
|
+
purpose: Detailed waiting patterns, implementation guide, and common mistakes
|
|
25
|
+
when_to_read: When implementing waitFor or debugging timing issues
|
|
26
|
+
---
|
|
27
|
+
|
|
28
|
+
# Condition-Based Waiting
|
|
29
|
+
|
|
30
|
+
## Overview
|
|
31
|
+
|
|
32
|
+
Flaky tests often guess at timing with arbitrary delays. This creates race conditions where tests pass on fast machines but fail under load or in CI.
|
|
33
|
+
|
|
34
|
+
**Core principle:** Wait for the actual condition you care about, not a guess about how long it takes.
|
|
35
|
+
|
|
36
|
+
## When to Use
|
|
37
|
+
|
|
38
|
+
```dot
|
|
39
|
+
digraph when_to_use {
|
|
40
|
+
"Test uses setTimeout/sleep?" [shape=diamond];
|
|
41
|
+
"Testing timing behavior?" [shape=diamond];
|
|
42
|
+
"Document WHY timeout needed" [shape=box];
|
|
43
|
+
"Use condition-based waiting" [shape=box];
|
|
44
|
+
|
|
45
|
+
"Test uses setTimeout/sleep?" -> "Testing timing behavior?" [label="yes"];
|
|
46
|
+
"Testing timing behavior?" -> "Document WHY timeout needed" [label="yes"];
|
|
47
|
+
"Testing timing behavior?" -> "Use condition-based waiting" [label="no"];
|
|
48
|
+
}
|
|
49
|
+
```
|
|
50
|
+
|
|
51
|
+
**Use when:**
|
|
52
|
+
- Tests have arbitrary delays (`setTimeout`, `sleep`, `time.sleep()`)
|
|
53
|
+
- Tests are flaky (pass sometimes, fail under load)
|
|
54
|
+
- Tests timeout when run in parallel
|
|
55
|
+
- Waiting for async operations to complete
|
|
56
|
+
|
|
57
|
+
**Don't use when:**
|
|
58
|
+
- Testing actual timing behavior (debounce, throttle intervals)
|
|
59
|
+
- Always document WHY if using arbitrary timeout
|
|
60
|
+
|
|
61
|
+
## Core Pattern
|
|
62
|
+
|
|
63
|
+
```typescript
|
|
64
|
+
// ❌ BEFORE: Guessing at timing
|
|
65
|
+
await new Promise(r => setTimeout(r, 50));
|
|
66
|
+
const result = getResult();
|
|
67
|
+
expect(result).toBeDefined();
|
|
68
|
+
|
|
69
|
+
// ✅ AFTER: Waiting for condition
|
|
70
|
+
await waitFor(() => getResult() !== undefined);
|
|
71
|
+
const result = getResult();
|
|
72
|
+
expect(result).toBeDefined();
|
|
73
|
+
```
|
|
74
|
+
|
|
75
|
+
## Quick Patterns
|
|
76
|
+
|
|
77
|
+
| Scenario | Pattern |
|
|
78
|
+
|----------|---------|
|
|
79
|
+
| Wait for event | `waitFor(() => events.find(e => e.type === 'DONE'))` |
|
|
80
|
+
| Wait for state | `waitFor(() => machine.state === 'ready')` |
|
|
81
|
+
| Wait for count | `waitFor(() => items.length >= 5)` |
|
|
82
|
+
| Wait for file | `waitFor(() => fs.existsSync(path))` |
|
|
83
|
+
| Complex condition | `waitFor(() => obj.ready && obj.value > 10)` |
|
|
84
|
+
|
|
85
|
+
## Implementation
|
|
86
|
+
|
|
87
|
+
Generic polling function:
|
|
88
|
+
```typescript
|
|
89
|
+
async function waitFor<T>(
|
|
90
|
+
condition: () => T | undefined | null | false,
|
|
91
|
+
description: string,
|
|
92
|
+
timeoutMs = 5000
|
|
93
|
+
): Promise<T> {
|
|
94
|
+
const startTime = Date.now();
|
|
95
|
+
|
|
96
|
+
while (true) {
|
|
97
|
+
const result = condition();
|
|
98
|
+
if (result) return result;
|
|
99
|
+
|
|
100
|
+
if (Date.now() - startTime > timeoutMs) {
|
|
101
|
+
throw new Error(`Timeout waiting for ${description} after ${timeoutMs}ms`);
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
await new Promise(r => setTimeout(r, 10)); // Poll every 10ms
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
```
|
|
108
|
+
|
|
109
|
+
See @example.ts for complete implementation with domain-specific helpers (`waitForEvent`, `waitForEventCount`, `waitForEventMatch`).
|
|
110
|
+
|
|
111
|
+
For detailed patterns, implementation guide, and common mistakes, see @references/patterns-and-implementation.md
|
|
112
|
+
|
|
113
|
+
## Real-World Impact
|
|
114
|
+
|
|
115
|
+
From debugging session (2025-10-03):
|
|
116
|
+
- Fixed 15 flaky tests across 3 files
|
|
117
|
+
- Pass rate: 60% → 100%
|
|
118
|
+
- Execution time: 40% faster
|
|
119
|
+
- No more race conditions
|