solokit 0.1.1__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.
- solokit/__init__.py +10 -0
- solokit/__version__.py +3 -0
- solokit/cli.py +374 -0
- solokit/core/__init__.py +1 -0
- solokit/core/cache.py +102 -0
- solokit/core/command_runner.py +278 -0
- solokit/core/config.py +453 -0
- solokit/core/config_validator.py +204 -0
- solokit/core/constants.py +291 -0
- solokit/core/error_formatter.py +279 -0
- solokit/core/error_handlers.py +346 -0
- solokit/core/exceptions.py +1567 -0
- solokit/core/file_ops.py +309 -0
- solokit/core/logging_config.py +166 -0
- solokit/core/output.py +99 -0
- solokit/core/performance.py +57 -0
- solokit/core/protocols.py +141 -0
- solokit/core/types.py +312 -0
- solokit/deployment/__init__.py +1 -0
- solokit/deployment/executor.py +411 -0
- solokit/git/__init__.py +1 -0
- solokit/git/integration.py +619 -0
- solokit/init/__init__.py +41 -0
- solokit/init/claude_commands_installer.py +87 -0
- solokit/init/dependency_installer.py +313 -0
- solokit/init/docs_structure.py +90 -0
- solokit/init/env_generator.py +160 -0
- solokit/init/environment_validator.py +334 -0
- solokit/init/git_hooks_installer.py +71 -0
- solokit/init/git_setup.py +188 -0
- solokit/init/gitignore_updater.py +195 -0
- solokit/init/initial_commit.py +145 -0
- solokit/init/initial_scans.py +109 -0
- solokit/init/orchestrator.py +246 -0
- solokit/init/readme_generator.py +207 -0
- solokit/init/session_structure.py +239 -0
- solokit/init/template_installer.py +424 -0
- solokit/learning/__init__.py +1 -0
- solokit/learning/archiver.py +115 -0
- solokit/learning/categorizer.py +126 -0
- solokit/learning/curator.py +428 -0
- solokit/learning/extractor.py +352 -0
- solokit/learning/reporter.py +351 -0
- solokit/learning/repository.py +254 -0
- solokit/learning/similarity.py +342 -0
- solokit/learning/validator.py +144 -0
- solokit/project/__init__.py +1 -0
- solokit/project/init.py +1162 -0
- solokit/project/stack.py +436 -0
- solokit/project/sync_plugin.py +438 -0
- solokit/project/tree.py +375 -0
- solokit/quality/__init__.py +1 -0
- solokit/quality/api_validator.py +424 -0
- solokit/quality/checkers/__init__.py +25 -0
- solokit/quality/checkers/base.py +114 -0
- solokit/quality/checkers/context7.py +221 -0
- solokit/quality/checkers/custom.py +162 -0
- solokit/quality/checkers/deployment.py +323 -0
- solokit/quality/checkers/documentation.py +179 -0
- solokit/quality/checkers/formatting.py +161 -0
- solokit/quality/checkers/integration.py +394 -0
- solokit/quality/checkers/linting.py +159 -0
- solokit/quality/checkers/security.py +261 -0
- solokit/quality/checkers/spec_completeness.py +127 -0
- solokit/quality/checkers/tests.py +184 -0
- solokit/quality/env_validator.py +306 -0
- solokit/quality/gates.py +655 -0
- solokit/quality/reporters/__init__.py +10 -0
- solokit/quality/reporters/base.py +25 -0
- solokit/quality/reporters/console.py +98 -0
- solokit/quality/reporters/json_reporter.py +34 -0
- solokit/quality/results.py +98 -0
- solokit/session/__init__.py +1 -0
- solokit/session/briefing/__init__.py +245 -0
- solokit/session/briefing/documentation_loader.py +53 -0
- solokit/session/briefing/formatter.py +476 -0
- solokit/session/briefing/git_context.py +282 -0
- solokit/session/briefing/learning_loader.py +212 -0
- solokit/session/briefing/milestone_builder.py +78 -0
- solokit/session/briefing/orchestrator.py +137 -0
- solokit/session/briefing/stack_detector.py +51 -0
- solokit/session/briefing/tree_generator.py +52 -0
- solokit/session/briefing/work_item_loader.py +209 -0
- solokit/session/briefing.py +353 -0
- solokit/session/complete.py +1188 -0
- solokit/session/status.py +246 -0
- solokit/session/validate.py +452 -0
- solokit/templates/.claude/commands/end.md +109 -0
- solokit/templates/.claude/commands/init.md +159 -0
- solokit/templates/.claude/commands/learn-curate.md +88 -0
- solokit/templates/.claude/commands/learn-search.md +62 -0
- solokit/templates/.claude/commands/learn-show.md +69 -0
- solokit/templates/.claude/commands/learn.md +136 -0
- solokit/templates/.claude/commands/start.md +114 -0
- solokit/templates/.claude/commands/status.md +22 -0
- solokit/templates/.claude/commands/validate.md +27 -0
- solokit/templates/.claude/commands/work-delete.md +119 -0
- solokit/templates/.claude/commands/work-graph.md +139 -0
- solokit/templates/.claude/commands/work-list.md +26 -0
- solokit/templates/.claude/commands/work-new.md +114 -0
- solokit/templates/.claude/commands/work-next.md +25 -0
- solokit/templates/.claude/commands/work-show.md +24 -0
- solokit/templates/.claude/commands/work-update.md +141 -0
- solokit/templates/CHANGELOG.md +17 -0
- solokit/templates/WORK_ITEM_TYPES.md +141 -0
- solokit/templates/__init__.py +1 -0
- solokit/templates/bug_spec.md +217 -0
- solokit/templates/config.schema.json +150 -0
- solokit/templates/dashboard_refine/base/.gitignore +36 -0
- solokit/templates/dashboard_refine/base/app/(dashboard)/layout.tsx +22 -0
- solokit/templates/dashboard_refine/base/app/(dashboard)/page.tsx +68 -0
- solokit/templates/dashboard_refine/base/app/(dashboard)/users/page.tsx +77 -0
- solokit/templates/dashboard_refine/base/app/globals.css +60 -0
- solokit/templates/dashboard_refine/base/app/layout.tsx +23 -0
- solokit/templates/dashboard_refine/base/app/page.tsx +9 -0
- solokit/templates/dashboard_refine/base/components/client-refine-wrapper.tsx +21 -0
- solokit/templates/dashboard_refine/base/components/layout/header.tsx +44 -0
- solokit/templates/dashboard_refine/base/components/layout/sidebar.tsx +82 -0
- solokit/templates/dashboard_refine/base/components/ui/button.tsx +53 -0
- solokit/templates/dashboard_refine/base/components/ui/card.tsx +78 -0
- solokit/templates/dashboard_refine/base/components/ui/table.tsx +116 -0
- solokit/templates/dashboard_refine/base/components.json +16 -0
- solokit/templates/dashboard_refine/base/lib/refine.tsx +65 -0
- solokit/templates/dashboard_refine/base/lib/utils.ts +13 -0
- solokit/templates/dashboard_refine/base/next.config.ts +10 -0
- solokit/templates/dashboard_refine/base/package.json.template +40 -0
- solokit/templates/dashboard_refine/base/postcss.config.mjs +8 -0
- solokit/templates/dashboard_refine/base/providers/refine-provider.tsx +26 -0
- solokit/templates/dashboard_refine/base/tailwind.config.ts +57 -0
- solokit/templates/dashboard_refine/base/tsconfig.json +27 -0
- solokit/templates/dashboard_refine/docker/Dockerfile +57 -0
- solokit/templates/dashboard_refine/docker/docker-compose.prod.yml +31 -0
- solokit/templates/dashboard_refine/docker/docker-compose.yml +21 -0
- solokit/templates/dashboard_refine/tier-1-essential/.eslintrc.json +7 -0
- solokit/templates/dashboard_refine/tier-1-essential/jest.config.ts +17 -0
- solokit/templates/dashboard_refine/tier-1-essential/jest.setup.ts +1 -0
- solokit/templates/dashboard_refine/tier-1-essential/package.json.tier1.template +57 -0
- solokit/templates/dashboard_refine/tier-1-essential/tests/setup.ts +26 -0
- solokit/templates/dashboard_refine/tier-1-essential/tests/unit/example.test.tsx +73 -0
- solokit/templates/dashboard_refine/tier-2-standard/package.json.tier2.template +62 -0
- solokit/templates/dashboard_refine/tier-3-comprehensive/eslint.config.mjs +22 -0
- solokit/templates/dashboard_refine/tier-3-comprehensive/package.json.tier3.template +79 -0
- solokit/templates/dashboard_refine/tier-3-comprehensive/playwright.config.ts +66 -0
- solokit/templates/dashboard_refine/tier-3-comprehensive/stryker.conf.json +38 -0
- solokit/templates/dashboard_refine/tier-3-comprehensive/tests/e2e/dashboard.spec.ts +88 -0
- solokit/templates/dashboard_refine/tier-3-comprehensive/tests/e2e/user-management.spec.ts +102 -0
- solokit/templates/dashboard_refine/tier-3-comprehensive/tests/integration/dashboard.test.tsx +90 -0
- solokit/templates/dashboard_refine/tier-3-comprehensive/type-coverage.json +16 -0
- solokit/templates/dashboard_refine/tier-4-production/instrumentation.ts +9 -0
- solokit/templates/dashboard_refine/tier-4-production/k6/dashboard-load-test.js +70 -0
- solokit/templates/dashboard_refine/tier-4-production/next.config.ts +46 -0
- solokit/templates/dashboard_refine/tier-4-production/package.json.tier4.template +89 -0
- solokit/templates/dashboard_refine/tier-4-production/sentry.client.config.ts +26 -0
- solokit/templates/dashboard_refine/tier-4-production/sentry.edge.config.ts +11 -0
- solokit/templates/dashboard_refine/tier-4-production/sentry.server.config.ts +11 -0
- solokit/templates/deployment_spec.md +500 -0
- solokit/templates/feature_spec.md +248 -0
- solokit/templates/fullstack_nextjs/base/.gitignore +36 -0
- solokit/templates/fullstack_nextjs/base/app/api/example/route.ts +65 -0
- solokit/templates/fullstack_nextjs/base/app/globals.css +27 -0
- solokit/templates/fullstack_nextjs/base/app/layout.tsx +20 -0
- solokit/templates/fullstack_nextjs/base/app/page.tsx +32 -0
- solokit/templates/fullstack_nextjs/base/components/example-component.tsx +20 -0
- solokit/templates/fullstack_nextjs/base/lib/prisma.ts +17 -0
- solokit/templates/fullstack_nextjs/base/lib/utils.ts +13 -0
- solokit/templates/fullstack_nextjs/base/lib/validations.ts +20 -0
- solokit/templates/fullstack_nextjs/base/next.config.ts +7 -0
- solokit/templates/fullstack_nextjs/base/package.json.template +32 -0
- solokit/templates/fullstack_nextjs/base/postcss.config.mjs +8 -0
- solokit/templates/fullstack_nextjs/base/prisma/schema.prisma +21 -0
- solokit/templates/fullstack_nextjs/base/tailwind.config.ts +19 -0
- solokit/templates/fullstack_nextjs/base/tsconfig.json +27 -0
- solokit/templates/fullstack_nextjs/docker/Dockerfile +60 -0
- solokit/templates/fullstack_nextjs/docker/docker-compose.prod.yml +57 -0
- solokit/templates/fullstack_nextjs/docker/docker-compose.yml +47 -0
- solokit/templates/fullstack_nextjs/tier-1-essential/.eslintrc.json +7 -0
- solokit/templates/fullstack_nextjs/tier-1-essential/jest.config.ts +17 -0
- solokit/templates/fullstack_nextjs/tier-1-essential/jest.setup.ts +1 -0
- solokit/templates/fullstack_nextjs/tier-1-essential/package.json.tier1.template +48 -0
- solokit/templates/fullstack_nextjs/tier-1-essential/tests/api/example.test.ts +88 -0
- solokit/templates/fullstack_nextjs/tier-1-essential/tests/setup.ts +22 -0
- solokit/templates/fullstack_nextjs/tier-1-essential/tests/unit/example.test.tsx +22 -0
- solokit/templates/fullstack_nextjs/tier-2-standard/package.json.tier2.template +52 -0
- solokit/templates/fullstack_nextjs/tier-3-comprehensive/eslint.config.mjs +39 -0
- solokit/templates/fullstack_nextjs/tier-3-comprehensive/package.json.tier3.template +68 -0
- solokit/templates/fullstack_nextjs/tier-3-comprehensive/playwright.config.ts +66 -0
- solokit/templates/fullstack_nextjs/tier-3-comprehensive/stryker.conf.json +33 -0
- solokit/templates/fullstack_nextjs/tier-3-comprehensive/tests/e2e/flow.spec.ts +59 -0
- solokit/templates/fullstack_nextjs/tier-3-comprehensive/tests/integration/api.test.ts +165 -0
- solokit/templates/fullstack_nextjs/tier-3-comprehensive/type-coverage.json +12 -0
- solokit/templates/fullstack_nextjs/tier-4-production/instrumentation.ts +9 -0
- solokit/templates/fullstack_nextjs/tier-4-production/k6/load-test.js +45 -0
- solokit/templates/fullstack_nextjs/tier-4-production/next.config.ts +46 -0
- solokit/templates/fullstack_nextjs/tier-4-production/package.json.tier4.template +77 -0
- solokit/templates/fullstack_nextjs/tier-4-production/sentry.client.config.ts +26 -0
- solokit/templates/fullstack_nextjs/tier-4-production/sentry.edge.config.ts +11 -0
- solokit/templates/fullstack_nextjs/tier-4-production/sentry.server.config.ts +11 -0
- solokit/templates/git-hooks/prepare-commit-msg +24 -0
- solokit/templates/integration_test_spec.md +363 -0
- solokit/templates/learnings.json +15 -0
- solokit/templates/ml_ai_fastapi/base/.gitignore +104 -0
- solokit/templates/ml_ai_fastapi/base/alembic/env.py +96 -0
- solokit/templates/ml_ai_fastapi/base/alembic.ini +114 -0
- solokit/templates/ml_ai_fastapi/base/pyproject.toml.template +91 -0
- solokit/templates/ml_ai_fastapi/base/requirements.txt.template +28 -0
- solokit/templates/ml_ai_fastapi/base/src/__init__.py +5 -0
- solokit/templates/ml_ai_fastapi/base/src/api/__init__.py +3 -0
- solokit/templates/ml_ai_fastapi/base/src/api/dependencies.py +20 -0
- solokit/templates/ml_ai_fastapi/base/src/api/routes/__init__.py +3 -0
- solokit/templates/ml_ai_fastapi/base/src/api/routes/example.py +134 -0
- solokit/templates/ml_ai_fastapi/base/src/api/routes/health.py +66 -0
- solokit/templates/ml_ai_fastapi/base/src/core/__init__.py +3 -0
- solokit/templates/ml_ai_fastapi/base/src/core/config.py +64 -0
- solokit/templates/ml_ai_fastapi/base/src/core/database.py +50 -0
- solokit/templates/ml_ai_fastapi/base/src/main.py +64 -0
- solokit/templates/ml_ai_fastapi/base/src/models/__init__.py +7 -0
- solokit/templates/ml_ai_fastapi/base/src/models/example.py +61 -0
- solokit/templates/ml_ai_fastapi/base/src/services/__init__.py +3 -0
- solokit/templates/ml_ai_fastapi/base/src/services/example.py +115 -0
- solokit/templates/ml_ai_fastapi/docker/Dockerfile +59 -0
- solokit/templates/ml_ai_fastapi/docker/docker-compose.prod.yml +112 -0
- solokit/templates/ml_ai_fastapi/docker/docker-compose.yml +77 -0
- solokit/templates/ml_ai_fastapi/tier-1-essential/pyproject.toml.tier1.template +112 -0
- solokit/templates/ml_ai_fastapi/tier-1-essential/pyrightconfig.json +41 -0
- solokit/templates/ml_ai_fastapi/tier-1-essential/pytest.ini +69 -0
- solokit/templates/ml_ai_fastapi/tier-1-essential/requirements-dev.txt +17 -0
- solokit/templates/ml_ai_fastapi/tier-1-essential/ruff.toml +81 -0
- solokit/templates/ml_ai_fastapi/tier-1-essential/tests/__init__.py +3 -0
- solokit/templates/ml_ai_fastapi/tier-1-essential/tests/conftest.py +72 -0
- solokit/templates/ml_ai_fastapi/tier-1-essential/tests/test_main.py +49 -0
- solokit/templates/ml_ai_fastapi/tier-1-essential/tests/unit/__init__.py +3 -0
- solokit/templates/ml_ai_fastapi/tier-1-essential/tests/unit/test_example.py +113 -0
- solokit/templates/ml_ai_fastapi/tier-2-standard/pyproject.toml.tier2.template +130 -0
- solokit/templates/ml_ai_fastapi/tier-3-comprehensive/locustfile.py +99 -0
- solokit/templates/ml_ai_fastapi/tier-3-comprehensive/mutmut_config.py +53 -0
- solokit/templates/ml_ai_fastapi/tier-3-comprehensive/pyproject.toml.tier3.template +150 -0
- solokit/templates/ml_ai_fastapi/tier-3-comprehensive/tests/integration/__init__.py +3 -0
- solokit/templates/ml_ai_fastapi/tier-3-comprehensive/tests/integration/conftest.py +74 -0
- solokit/templates/ml_ai_fastapi/tier-3-comprehensive/tests/integration/test_api.py +131 -0
- solokit/templates/ml_ai_fastapi/tier-4-production/pyproject.toml.tier4.template +162 -0
- solokit/templates/ml_ai_fastapi/tier-4-production/requirements-prod.txt +25 -0
- solokit/templates/ml_ai_fastapi/tier-4-production/src/api/routes/metrics.py +19 -0
- solokit/templates/ml_ai_fastapi/tier-4-production/src/core/logging.py +74 -0
- solokit/templates/ml_ai_fastapi/tier-4-production/src/core/monitoring.py +68 -0
- solokit/templates/ml_ai_fastapi/tier-4-production/src/core/sentry.py +66 -0
- solokit/templates/ml_ai_fastapi/tier-4-production/src/middleware/__init__.py +3 -0
- solokit/templates/ml_ai_fastapi/tier-4-production/src/middleware/logging.py +79 -0
- solokit/templates/ml_ai_fastapi/tier-4-production/src/middleware/tracing.py +60 -0
- solokit/templates/refactor_spec.md +287 -0
- solokit/templates/saas_t3/base/.gitignore +36 -0
- solokit/templates/saas_t3/base/app/api/trpc/[trpc]/route.ts +33 -0
- solokit/templates/saas_t3/base/app/globals.css +27 -0
- solokit/templates/saas_t3/base/app/layout.tsx +23 -0
- solokit/templates/saas_t3/base/app/page.tsx +31 -0
- solokit/templates/saas_t3/base/lib/api.tsx +77 -0
- solokit/templates/saas_t3/base/lib/utils.ts +13 -0
- solokit/templates/saas_t3/base/next.config.ts +7 -0
- solokit/templates/saas_t3/base/package.json.template +38 -0
- solokit/templates/saas_t3/base/postcss.config.mjs +8 -0
- solokit/templates/saas_t3/base/prisma/schema.prisma +20 -0
- solokit/templates/saas_t3/base/server/api/root.ts +19 -0
- solokit/templates/saas_t3/base/server/api/routers/example.ts +28 -0
- solokit/templates/saas_t3/base/server/api/trpc.ts +52 -0
- solokit/templates/saas_t3/base/server/db.ts +17 -0
- solokit/templates/saas_t3/base/tailwind.config.ts +19 -0
- solokit/templates/saas_t3/base/tsconfig.json +27 -0
- solokit/templates/saas_t3/docker/Dockerfile +60 -0
- solokit/templates/saas_t3/docker/docker-compose.prod.yml +59 -0
- solokit/templates/saas_t3/docker/docker-compose.yml +49 -0
- solokit/templates/saas_t3/tier-1-essential/.eslintrc.json +7 -0
- solokit/templates/saas_t3/tier-1-essential/jest.config.ts +17 -0
- solokit/templates/saas_t3/tier-1-essential/jest.setup.ts +1 -0
- solokit/templates/saas_t3/tier-1-essential/package.json.tier1.template +54 -0
- solokit/templates/saas_t3/tier-1-essential/tests/setup.ts +22 -0
- solokit/templates/saas_t3/tier-1-essential/tests/unit/example.test.tsx +24 -0
- solokit/templates/saas_t3/tier-2-standard/package.json.tier2.template +58 -0
- solokit/templates/saas_t3/tier-3-comprehensive/eslint.config.mjs +39 -0
- solokit/templates/saas_t3/tier-3-comprehensive/package.json.tier3.template +74 -0
- solokit/templates/saas_t3/tier-3-comprehensive/playwright.config.ts +66 -0
- solokit/templates/saas_t3/tier-3-comprehensive/stryker.conf.json +34 -0
- solokit/templates/saas_t3/tier-3-comprehensive/tests/e2e/home.spec.ts +41 -0
- solokit/templates/saas_t3/tier-3-comprehensive/tests/integration/api.test.ts +44 -0
- solokit/templates/saas_t3/tier-3-comprehensive/type-coverage.json +12 -0
- solokit/templates/saas_t3/tier-4-production/instrumentation.ts +9 -0
- solokit/templates/saas_t3/tier-4-production/k6/load-test.js +51 -0
- solokit/templates/saas_t3/tier-4-production/next.config.ts +46 -0
- solokit/templates/saas_t3/tier-4-production/package.json.tier4.template +83 -0
- solokit/templates/saas_t3/tier-4-production/sentry.client.config.ts +26 -0
- solokit/templates/saas_t3/tier-4-production/sentry.edge.config.ts +11 -0
- solokit/templates/saas_t3/tier-4-production/sentry.server.config.ts +11 -0
- solokit/templates/saas_t3/tier-4-production/vercel.json +37 -0
- solokit/templates/security_spec.md +287 -0
- solokit/templates/stack-versions.yaml +617 -0
- solokit/templates/status_update.json +6 -0
- solokit/templates/template-registry.json +257 -0
- solokit/templates/work_items.json +11 -0
- solokit/testing/__init__.py +1 -0
- solokit/testing/integration_runner.py +550 -0
- solokit/testing/performance.py +637 -0
- solokit/visualization/__init__.py +1 -0
- solokit/visualization/dependency_graph.py +788 -0
- solokit/work_items/__init__.py +1 -0
- solokit/work_items/creator.py +217 -0
- solokit/work_items/delete.py +264 -0
- solokit/work_items/get_dependencies.py +185 -0
- solokit/work_items/get_dependents.py +113 -0
- solokit/work_items/get_metadata.py +121 -0
- solokit/work_items/get_next_recommendations.py +133 -0
- solokit/work_items/manager.py +235 -0
- solokit/work_items/milestones.py +137 -0
- solokit/work_items/query.py +376 -0
- solokit/work_items/repository.py +267 -0
- solokit/work_items/scheduler.py +184 -0
- solokit/work_items/spec_parser.py +838 -0
- solokit/work_items/spec_validator.py +493 -0
- solokit/work_items/updater.py +157 -0
- solokit/work_items/validator.py +205 -0
- solokit-0.1.1.dist-info/METADATA +640 -0
- solokit-0.1.1.dist-info/RECORD +323 -0
- solokit-0.1.1.dist-info/WHEEL +5 -0
- solokit-0.1.1.dist-info/entry_points.txt +2 -0
- solokit-0.1.1.dist-info/licenses/LICENSE +21 -0
- solokit-0.1.1.dist-info/top_level.txt +1 -0
|
@@ -0,0 +1,278 @@
|
|
|
1
|
+
"""Centralized command execution with consistent error handling.
|
|
2
|
+
|
|
3
|
+
This module provides a unified interface for running subprocess commands
|
|
4
|
+
with standardized timeout handling, error handling, logging, and retry logic.
|
|
5
|
+
"""
|
|
6
|
+
|
|
7
|
+
import json
|
|
8
|
+
import logging
|
|
9
|
+
import subprocess
|
|
10
|
+
import time
|
|
11
|
+
from dataclasses import dataclass
|
|
12
|
+
from pathlib import Path
|
|
13
|
+
from typing import Any, Optional, Union
|
|
14
|
+
|
|
15
|
+
from solokit.core.error_handlers import log_errors
|
|
16
|
+
from solokit.core.exceptions import (
|
|
17
|
+
CommandExecutionError,
|
|
18
|
+
TimeoutError,
|
|
19
|
+
)
|
|
20
|
+
|
|
21
|
+
logger = logging.getLogger(__name__)
|
|
22
|
+
|
|
23
|
+
|
|
24
|
+
@dataclass
|
|
25
|
+
class CommandResult:
|
|
26
|
+
"""Result of a command execution."""
|
|
27
|
+
|
|
28
|
+
returncode: int
|
|
29
|
+
stdout: str
|
|
30
|
+
stderr: str
|
|
31
|
+
command: list[str]
|
|
32
|
+
duration_seconds: float
|
|
33
|
+
timed_out: bool = False
|
|
34
|
+
|
|
35
|
+
@property
|
|
36
|
+
def success(self) -> bool:
|
|
37
|
+
"""Whether command succeeded."""
|
|
38
|
+
return self.returncode == 0 and not self.timed_out
|
|
39
|
+
|
|
40
|
+
@property
|
|
41
|
+
def output(self) -> str:
|
|
42
|
+
"""Get stdout or stderr if stdout empty."""
|
|
43
|
+
return self.stdout.strip() if self.stdout else self.stderr.strip()
|
|
44
|
+
|
|
45
|
+
|
|
46
|
+
class CommandRunner:
|
|
47
|
+
"""Centralized command execution with consistent error handling."""
|
|
48
|
+
|
|
49
|
+
DEFAULT_TIMEOUT = 30 # seconds
|
|
50
|
+
|
|
51
|
+
def __init__(
|
|
52
|
+
self,
|
|
53
|
+
default_timeout: float = DEFAULT_TIMEOUT,
|
|
54
|
+
working_dir: Optional[Path] = None,
|
|
55
|
+
raise_on_error: bool = False,
|
|
56
|
+
):
|
|
57
|
+
"""Initialize command runner.
|
|
58
|
+
|
|
59
|
+
Args:
|
|
60
|
+
default_timeout: Default timeout in seconds
|
|
61
|
+
working_dir: Working directory for commands (None = current)
|
|
62
|
+
raise_on_error: Whether to raise exception on non-zero exit
|
|
63
|
+
"""
|
|
64
|
+
self.default_timeout = default_timeout
|
|
65
|
+
self.working_dir = working_dir
|
|
66
|
+
self.raise_on_error = raise_on_error
|
|
67
|
+
|
|
68
|
+
@log_errors()
|
|
69
|
+
def run(
|
|
70
|
+
self,
|
|
71
|
+
command: Union[str, list[str]],
|
|
72
|
+
timeout: Optional[float] = None,
|
|
73
|
+
check: Optional[bool] = None,
|
|
74
|
+
working_dir: Optional[Path] = None,
|
|
75
|
+
retry_count: int = 0,
|
|
76
|
+
retry_delay: float = 1.0,
|
|
77
|
+
env: Optional[dict] = None,
|
|
78
|
+
) -> CommandResult:
|
|
79
|
+
"""Run a command with consistent error handling.
|
|
80
|
+
|
|
81
|
+
Args:
|
|
82
|
+
command: Command to run (string or list)
|
|
83
|
+
timeout: Timeout in seconds (None = use default)
|
|
84
|
+
check: Raise exception on non-zero exit (None = use instance setting)
|
|
85
|
+
working_dir: Working directory (None = use instance setting)
|
|
86
|
+
retry_count: Number of retries on failure
|
|
87
|
+
retry_delay: Delay between retries in seconds
|
|
88
|
+
env: Environment variables
|
|
89
|
+
|
|
90
|
+
Returns:
|
|
91
|
+
CommandResult with output and status
|
|
92
|
+
|
|
93
|
+
Raises:
|
|
94
|
+
CommandExecutionError: If check=True and command fails
|
|
95
|
+
"""
|
|
96
|
+
if isinstance(command, str):
|
|
97
|
+
command = command.split()
|
|
98
|
+
|
|
99
|
+
timeout = timeout if timeout is not None else self.default_timeout
|
|
100
|
+
check = check if check is not None else self.raise_on_error
|
|
101
|
+
cwd = working_dir or self.working_dir
|
|
102
|
+
|
|
103
|
+
attempt = 0
|
|
104
|
+
max_attempts = retry_count + 1
|
|
105
|
+
|
|
106
|
+
while attempt < max_attempts:
|
|
107
|
+
try:
|
|
108
|
+
start_time = time.time()
|
|
109
|
+
|
|
110
|
+
logger.debug(
|
|
111
|
+
f"Running command: {' '.join(command)} "
|
|
112
|
+
f"(timeout={timeout}s, cwd={cwd}, attempt={attempt + 1}/{max_attempts})"
|
|
113
|
+
)
|
|
114
|
+
|
|
115
|
+
result = subprocess.run(
|
|
116
|
+
command,
|
|
117
|
+
capture_output=True,
|
|
118
|
+
text=True,
|
|
119
|
+
timeout=timeout,
|
|
120
|
+
check=False, # We handle errors ourselves
|
|
121
|
+
cwd=cwd,
|
|
122
|
+
env=env,
|
|
123
|
+
)
|
|
124
|
+
|
|
125
|
+
duration = time.time() - start_time
|
|
126
|
+
|
|
127
|
+
cmd_result = CommandResult(
|
|
128
|
+
returncode=result.returncode,
|
|
129
|
+
stdout=result.stdout,
|
|
130
|
+
stderr=result.stderr,
|
|
131
|
+
command=command,
|
|
132
|
+
duration_seconds=duration,
|
|
133
|
+
)
|
|
134
|
+
|
|
135
|
+
if cmd_result.success:
|
|
136
|
+
logger.debug(f"Command succeeded in {duration:.2f}s")
|
|
137
|
+
return cmd_result
|
|
138
|
+
|
|
139
|
+
# Command failed
|
|
140
|
+
logger.warning(
|
|
141
|
+
f"Command failed with exit code {result.returncode}: "
|
|
142
|
+
f"{' '.join(command)}\nstderr: {result.stderr[:200]}"
|
|
143
|
+
)
|
|
144
|
+
|
|
145
|
+
if check:
|
|
146
|
+
raise CommandExecutionError(
|
|
147
|
+
command=" ".join(command),
|
|
148
|
+
returncode=result.returncode,
|
|
149
|
+
stderr=result.stderr,
|
|
150
|
+
stdout=result.stdout,
|
|
151
|
+
)
|
|
152
|
+
|
|
153
|
+
# Retry if configured
|
|
154
|
+
if attempt < max_attempts - 1:
|
|
155
|
+
logger.info(f"Retrying in {retry_delay}s...")
|
|
156
|
+
time.sleep(retry_delay)
|
|
157
|
+
attempt += 1
|
|
158
|
+
continue
|
|
159
|
+
|
|
160
|
+
return cmd_result
|
|
161
|
+
|
|
162
|
+
except subprocess.TimeoutExpired as e:
|
|
163
|
+
duration = time.time() - start_time
|
|
164
|
+
|
|
165
|
+
logger.error(f"Command timed out after {timeout}s: {' '.join(command)}")
|
|
166
|
+
|
|
167
|
+
stdout_str = e.stdout.decode() if isinstance(e.stdout, bytes) else (e.stdout or "")
|
|
168
|
+
stderr_str = e.stderr.decode() if isinstance(e.stderr, bytes) else (e.stderr or "")
|
|
169
|
+
|
|
170
|
+
cmd_result = CommandResult(
|
|
171
|
+
returncode=-1,
|
|
172
|
+
stdout=stdout_str,
|
|
173
|
+
stderr=stderr_str,
|
|
174
|
+
command=command,
|
|
175
|
+
duration_seconds=duration,
|
|
176
|
+
timed_out=True,
|
|
177
|
+
)
|
|
178
|
+
|
|
179
|
+
if check:
|
|
180
|
+
raise TimeoutError(
|
|
181
|
+
operation=" ".join(command),
|
|
182
|
+
timeout_seconds=int(timeout),
|
|
183
|
+
context={
|
|
184
|
+
"stdout": e.stdout or "",
|
|
185
|
+
"stderr": e.stderr or "",
|
|
186
|
+
},
|
|
187
|
+
) from e
|
|
188
|
+
|
|
189
|
+
# Retry if configured
|
|
190
|
+
if attempt < max_attempts - 1:
|
|
191
|
+
logger.info(f"Retrying in {retry_delay}s...")
|
|
192
|
+
time.sleep(retry_delay)
|
|
193
|
+
attempt += 1
|
|
194
|
+
continue
|
|
195
|
+
|
|
196
|
+
return cmd_result
|
|
197
|
+
|
|
198
|
+
except (CommandExecutionError, TimeoutError):
|
|
199
|
+
# Re-raise our standardized errors as-is (don't catch and re-wrap)
|
|
200
|
+
raise
|
|
201
|
+
except Exception as e:
|
|
202
|
+
logger.error(f"Unexpected error running command: {e}")
|
|
203
|
+
|
|
204
|
+
if check:
|
|
205
|
+
raise CommandExecutionError(
|
|
206
|
+
command=" ".join(command),
|
|
207
|
+
returncode=-1,
|
|
208
|
+
stderr=str(e),
|
|
209
|
+
stdout="",
|
|
210
|
+
) from e
|
|
211
|
+
|
|
212
|
+
# Don't retry on unexpected errors
|
|
213
|
+
return CommandResult(
|
|
214
|
+
returncode=-1,
|
|
215
|
+
stdout="",
|
|
216
|
+
stderr=str(e),
|
|
217
|
+
command=command,
|
|
218
|
+
duration_seconds=time.time() - start_time,
|
|
219
|
+
)
|
|
220
|
+
|
|
221
|
+
# Should never reach here
|
|
222
|
+
raise RuntimeError("Retry logic error")
|
|
223
|
+
|
|
224
|
+
@log_errors()
|
|
225
|
+
def run_json(self, command: Union[str, list[str]], **kwargs: Any) -> Optional[dict[str, Any]]:
|
|
226
|
+
"""Run command and parse JSON output.
|
|
227
|
+
|
|
228
|
+
Args:
|
|
229
|
+
command: Command to run
|
|
230
|
+
**kwargs: Additional arguments passed to run()
|
|
231
|
+
|
|
232
|
+
Returns:
|
|
233
|
+
Parsed JSON dict or None if parse fails
|
|
234
|
+
"""
|
|
235
|
+
result = self.run(command, **kwargs)
|
|
236
|
+
if not result.success:
|
|
237
|
+
return None
|
|
238
|
+
|
|
239
|
+
try:
|
|
240
|
+
parsed: dict[str, Any] = json.loads(result.stdout)
|
|
241
|
+
return parsed
|
|
242
|
+
except json.JSONDecodeError as e:
|
|
243
|
+
logger.error(f"Failed to parse JSON output: {e}")
|
|
244
|
+
return None
|
|
245
|
+
|
|
246
|
+
@log_errors()
|
|
247
|
+
def run_lines(self, command: Union[str, list[str]], **kwargs: Any) -> list[str]:
|
|
248
|
+
"""Run command and return output as lines.
|
|
249
|
+
|
|
250
|
+
Args:
|
|
251
|
+
command: Command to run
|
|
252
|
+
**kwargs: Additional arguments passed to run()
|
|
253
|
+
|
|
254
|
+
Returns:
|
|
255
|
+
List of non-empty lines
|
|
256
|
+
"""
|
|
257
|
+
result = self.run(command, **kwargs)
|
|
258
|
+
if not result.success:
|
|
259
|
+
return []
|
|
260
|
+
|
|
261
|
+
return [line.strip() for line in result.stdout.split("\n") if line.strip()]
|
|
262
|
+
|
|
263
|
+
|
|
264
|
+
# Global instance for convenience
|
|
265
|
+
_default_runner = CommandRunner()
|
|
266
|
+
|
|
267
|
+
|
|
268
|
+
def run_command(command: Union[str, list[str]], **kwargs: Any) -> CommandResult:
|
|
269
|
+
"""Convenience function to run command with default runner.
|
|
270
|
+
|
|
271
|
+
Args:
|
|
272
|
+
command: Command to run
|
|
273
|
+
**kwargs: Additional arguments passed to run()
|
|
274
|
+
|
|
275
|
+
Returns:
|
|
276
|
+
Command result
|
|
277
|
+
"""
|
|
278
|
+
return _default_runner.run(command, **kwargs)
|