aegis-stack 0.2.0rc2__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.
- aegis/__init__.py +5 -0
- aegis/__main__.py +51 -0
- aegis/cli/__init__.py +6 -0
- aegis/cli/callbacks.py +114 -0
- aegis/cli/interactive.py +611 -0
- aegis/cli/utils.py +70 -0
- aegis/cli/validators.py +34 -0
- aegis/commands/__init__.py +6 -0
- aegis/commands/add.py +353 -0
- aegis/commands/add_service.py +332 -0
- aegis/commands/components.py +35 -0
- aegis/commands/init.py +370 -0
- aegis/commands/remove.py +227 -0
- aegis/commands/services.py +52 -0
- aegis/commands/update.py +252 -0
- aegis/commands/version.py +12 -0
- aegis/config/__init__.py +1 -0
- aegis/config/shared_files.py +136 -0
- aegis/core/CLAUDE.md +377 -0
- aegis/core/__init__.py +6 -0
- aegis/core/component_files.py +228 -0
- aegis/core/component_utils.py +220 -0
- aegis/core/components.py +127 -0
- aegis/core/copier_manager.py +315 -0
- aegis/core/copier_updater.py +475 -0
- aegis/core/dependency_resolver.py +119 -0
- aegis/core/manual_updater.py +554 -0
- aegis/core/post_gen_tasks.py +547 -0
- aegis/core/service_resolver.py +261 -0
- aegis/core/services.py +157 -0
- aegis/core/template_generator.py +266 -0
- aegis/core/version_compatibility.py +259 -0
- aegis/templates/CLAUDE.md +591 -0
- aegis/templates/cookiecutter-aegis-project/cookiecutter.json +39 -0
- aegis/templates/cookiecutter-aegis-project/hooks/post_gen_project.py +214 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/.dockerignore +71 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/.env.example.j2 +130 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/.gitignore +131 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/Dockerfile +53 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/Makefile +236 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/README.md.j2 +196 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/alembic/alembic.ini.j2 +111 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/alembic/env.py.j2 +91 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/alembic/script.py.mako +25 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/alembic/versions/001_initial_auth.py.j2 +51 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/__init__.py +5 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/cli/__init__.py +6 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/cli/ai.py.j2 +700 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/cli/ai_rendering.py +361 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/cli/auth.py.j2 +253 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/cli/health.py.j2 +419 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/cli/load_test.py.j2 +656 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/cli/main.py.j2 +65 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/cli/marko_terminal_renderer.py +489 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/cli/tasks.py.j2 +328 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/cli/{% if cookiecutter.include_scheduler == /"yes/" %}tasks.py{% endif %}" +340 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/backend/__init__.py +0 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/backend/api/__init__.py +0 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/backend/api/ai/__init__.py +8 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/backend/api/ai/router.py +329 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/backend/api/auth/__init__.py +1 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/backend/api/auth/router.py +64 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/backend/api/deps.py +58 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/backend/api/health.py +163 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/backend/api/models.py.j2 +280 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/backend/api/routing.py.j2 +32 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/backend/api/scheduler.py.j2 +121 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/backend/api/worker.py.j2 +478 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/backend/hooks.py +144 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/backend/main.py +31 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/backend/middleware/__init__.py +1 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/backend/middleware/cors.py +20 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/backend/shutdown/__init__.py +1 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/backend/shutdown/cleanup.py +14 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/backend/startup/__init__.py +1 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/backend/startup/component_health.py.j2 +418 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/backend/startup/database_init.py.j2 +83 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/frontend/__init__.py +5 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/frontend/controls/__init__.py +27 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/frontend/controls/table.py +78 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/frontend/controls/text.py +142 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/frontend/dashboard/cards/__init__.py.j2 +47 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/frontend/dashboard/cards/ai_card.py +287 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/frontend/dashboard/cards/auth_card.py +198 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/frontend/dashboard/cards/base_card.py +256 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/frontend/dashboard/cards/card_factory.py +227 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/frontend/dashboard/cards/card_utils.py +333 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/frontend/dashboard/cards/database_card.py +420 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/frontend/dashboard/cards/fastapi_card.py +328 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/frontend/dashboard/cards/flet_card.py +267 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/frontend/dashboard/cards/redis_card.py +322 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/frontend/dashboard/cards/scheduler_card.py +352 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/frontend/dashboard/cards/services_card.py +233 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/frontend/dashboard/cards/worker_card.py +684 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/frontend/main.py.j2 +653 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/frontend/theme.py +48 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/scheduler/__init__.py +1 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/scheduler/main.py.j2 +156 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/worker/CLAUDE.md.j2 +213 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/worker/__init__.py +6 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/worker/constants.py.j2 +30 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/worker/pools.py +97 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/worker/queues/__init__.py +1 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/worker/queues/load_test.py +55 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/worker/queues/media.py +49 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/worker/queues/system.py +44 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/worker/registry.py +139 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/worker/tasks/__init__.py +120 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/worker/tasks/load_tasks.py +507 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/worker/tasks/simple_system_tasks.py +33 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/components/worker/tasks/system_tasks.py +281 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/core/config.py.j2 +178 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/core/constants.py +58 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/core/db.py.j2 +176 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/core/log.py +92 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/core/security.py +62 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/entrypoints/__init__.py +1 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/entrypoints/webserver.py +40 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/entrypoints/{% if cookiecutter.include_scheduler == /"yes/" %}scheduler.py{% endif %}" +21 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/integrations/__init__.py +0 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/integrations/main.py +62 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/models/__init__.py +1 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/models/user.py +44 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/py.typed +0 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/services/__init__.py +1 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/services/ai/__init__.py +8 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/services/ai/config.py +130 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/services/ai/conversation.py +213 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/services/ai/health.py +96 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/services/ai/models.py +229 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/services/ai/providers.py +370 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/services/ai/service.py +388 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/services/auth/__init__.py +1 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/services/auth/auth_service.py +41 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/services/auth/health.py +164 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/services/auth/user_service.py +83 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/services/backend/middleware_inspector.py.j2 +223 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/services/backend/models.py.j2 +70 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/services/backend/route_inspector.py.j2 +155 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/services/load_test.py +679 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/services/load_test_models.py +266 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/services/scheduler/__init__.py.j2 +21 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/services/scheduler/models.py.j2 +119 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/services/scheduler/scheduled_task_manager.py.j2 +273 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/services/scheduler/task_monitor.py.j2 +189 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/services/shared/__init__.py +15 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/services/shared/models.py +26 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/services/system/__init__.py +52 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/services/system/alerts.py +94 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/services/system/backup.py.j2 +119 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/services/system/health.py.j2 +1333 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/services/system/models.py +243 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/app/services/system/ui.py +52 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/assets/aegis-manifesto-dark.png +0 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/assets/aegis-manifesto-square-backup.png +0 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/assets/aegis-manifesto.png +0 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/clean-validation/.dockerignore +71 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/clean-validation/.env.example.j2 +64 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/clean-validation/.gitignore +131 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/clean-validation/Dockerfile +53 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/clean-validation/Makefile +211 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/clean-validation/README.md.j2 +172 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/clean-validation/docker-compose.yml.j2 +78 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/clean-validation/mkdocs.yml.j2 +62 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/clean-validation/pyproject.toml.j2 +120 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/clean-validation/uv.lock +1673 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/docker-compose.yml.j2 +200 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/docs/api.md +191 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/docs/components/scheduler.md +0 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/docs/components/scheduler.md.j2 +621 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/docs/development.md +215 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/docs/health.md +240 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/docs/javascripts/mermaid-config.js +62 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/docs/stylesheets/mermaid.css +95 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/mkdocs.yml.j2 +62 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/pyproject.toml.j2 +131 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/scripts/entrypoint.sh +87 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/scripts/entrypoint.sh.j2 +93 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/scripts/gen_docs.py +16 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/tests/api/__init__.py +1 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/tests/api/test_auth_endpoints.py.j2 +307 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/tests/api/test_health_endpoints.py.j2 +262 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/tests/api/test_scheduler_endpoints.py.j2 +214 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/tests/api/test_worker_endpoints.py.j2 +165 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/tests/cli/test_ai_rendering.py +427 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/tests/cli/test_conversation_memory.py +465 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/tests/components/test_scheduler.py +43 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/tests/conftest.py.j2 +195 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/tests/services/__init__.py +1 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/tests/services/ai/__init__.py +1 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/tests/services/ai/conftest.py +78 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/tests/services/ai/test_health.py +157 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/tests/services/ai/test_models.py +164 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/tests/services/ai/test_service.py +198 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/tests/services/test_auth_integration.py.j2 +528 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/tests/services/test_component_integration.py.j2 +387 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/tests/services/test_conversation_persistence.py +342 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/tests/services/test_health_logic.py.j2 +663 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/tests/services/test_load_test_models.py +619 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/tests/services/test_load_test_service.py +603 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/tests/services/test_middleware_inspector.py.j2 +248 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/tests/services/test_scheduled_task_manager.py.j2 +292 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/tests/services/test_system_service.py +98 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/tests/services/test_worker_health_registration.py.j2 +257 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/tests/test_core.py +49 -0
- aegis/templates/cookiecutter-aegis-project/{{cookiecutter.project_slug}}/uv.lock +1673 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/.copier-answers.yml.jinja +21 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/.dockerignore +71 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/.env.example.jinja +130 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/.gitignore +131 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/Dockerfile +53 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/Makefile.jinja +236 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/README.md.jinja +196 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/alembic/alembic.ini.jinja +111 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/alembic/env.py.jinja +91 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/alembic/script.py.mako +25 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/alembic/versions/001_initial_auth.py.jinja +51 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/__init__.py.jinja +5 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/cli/__init__.py.jinja +6 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/cli/ai.py.jinja +700 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/cli/ai_rendering.py +360 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/cli/auth.py.jinja +253 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/cli/health.py.jinja +419 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/cli/load_test.py.jinja +656 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/cli/main.py.jinja +65 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/cli/marko_terminal_renderer.py +489 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/cli/tasks.py.jinja +328 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/backend/__init__.py +0 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/backend/api/__init__.py +0 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/backend/api/ai/__init__.py +8 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/backend/api/ai/router.py +329 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/backend/api/auth/__init__.py +1 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/backend/api/auth/router.py +64 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/backend/api/deps.py +58 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/backend/api/health.py.jinja +163 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/backend/api/models.py.jinja +280 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/backend/api/routing.py.jinja +32 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/backend/api/scheduler.py.jinja +121 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/backend/api/worker.py.jinja +478 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/backend/hooks.py +144 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/backend/main.py +31 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/backend/middleware/__init__.py +1 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/backend/middleware/cors.py +20 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/backend/shutdown/__init__.py +1 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/backend/shutdown/cleanup.py +14 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/backend/startup/__init__.py +1 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/backend/startup/component_health.py.jinja +418 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/backend/startup/database_init.py.jinja +83 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/frontend/__init__.py +5 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/frontend/controls/__init__.py +27 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/frontend/controls/table.py +78 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/frontend/controls/text.py +142 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/frontend/dashboard/cards/__init__.py.jinja +47 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/frontend/dashboard/cards/ai_card.py +287 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/frontend/dashboard/cards/auth_card.py +198 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/frontend/dashboard/cards/base_card.py +256 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/frontend/dashboard/cards/card_factory.py +227 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/frontend/dashboard/cards/card_utils.py +333 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/frontend/dashboard/cards/database_card.py +420 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/frontend/dashboard/cards/fastapi_card.py +328 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/frontend/dashboard/cards/flet_card.py +267 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/frontend/dashboard/cards/redis_card.py +322 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/frontend/dashboard/cards/scheduler_card.py +352 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/frontend/dashboard/cards/services_card.py +233 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/frontend/dashboard/cards/worker_card.py +684 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/frontend/main.py.jinja +653 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/frontend/theme.py +48 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/scheduler/__init__.py +1 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/scheduler/main.py.jinja +156 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/worker/CLAUDE.md.jinja +213 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/worker/__init__.py +6 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/worker/constants.py.jinja +30 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/worker/pools.py +97 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/worker/queues/__init__.py +1 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/worker/queues/load_test.py +55 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/worker/queues/media.py +49 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/worker/queues/system.py +44 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/worker/registry.py +139 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/worker/tasks/__init__.py +120 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/worker/tasks/load_tasks.py +507 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/worker/tasks/simple_system_tasks.py +33 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/components/worker/tasks/system_tasks.py +281 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/core/config.py.jinja +178 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/core/constants.py +58 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/core/db.py.jinja +176 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/core/log.py +92 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/core/security.py +62 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/entrypoints/__init__.py +1 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/entrypoints/scheduler.py.jinja +21 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/entrypoints/webserver.py +39 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/integrations/__init__.py +0 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/integrations/main.py +61 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/models/__init__.py +1 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/models/user.py +44 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/py.typed +0 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/services/__init__.py +1 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/services/ai/__init__.py +8 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/services/ai/config.py +130 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/services/ai/conversation.py +213 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/services/ai/health.py +96 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/services/ai/models.py +229 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/services/ai/providers.py.jinja +370 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/services/ai/service.py +387 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/services/auth/__init__.py +1 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/services/auth/auth_service.py +40 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/services/auth/health.py +162 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/services/auth/user_service.py +82 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/services/backend/middleware_inspector.py.jinja +223 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/services/backend/models.py.jinja +70 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/services/backend/route_inspector.py.jinja +155 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/services/load_test.py +678 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/services/load_test_models.py +265 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/services/scheduler/__init__.py.jinja +21 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/services/scheduler/models.py.jinja +119 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/services/scheduler/scheduled_task_manager.py.jinja +273 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/services/scheduler/task_monitor.py.jinja +189 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/services/shared/__init__.py +15 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/services/shared/models.py +26 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/services/system/__init__.py +52 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/services/system/alerts.py +94 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/services/system/backup.py.jinja +119 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/services/system/health.py.jinja +1333 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/services/system/models.py +243 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/app/services/system/ui.py +52 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/assets/.!57223!aegis-manifesto.png +0 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/assets/.!57224!aegis-manifesto-dark.png +0 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/assets/.!57225!aegis-manifesto-square-backup.png +0 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/assets/.!57533!aegis-manifesto.png +0 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/assets/.!57534!aegis-manifesto-dark.png +0 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/assets/.!57538!aegis-manifesto-square-backup.png +0 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/assets/.!57897!aegis-manifesto.png +0 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/assets/.!57898!aegis-manifesto-dark.png +0 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/assets/.!57904!aegis-manifesto-square-backup.png +0 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/assets/.!58315!aegis-manifesto.png +0 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/assets/.!58316!aegis-manifesto-dark.png +0 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/assets/.!58324!aegis-manifesto-square-backup.png +0 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/assets/.!58837!aegis-manifesto.png +0 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/assets/.!58838!aegis-manifesto-dark.png +0 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/assets/.!58849!aegis-manifesto-square-backup.png +0 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/assets/aegis-manifesto-dark.png +0 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/assets/aegis-manifesto-square-backup.png +0 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/assets/aegis-manifesto.png +0 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/clean-validation/.env.example.jinja +64 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/clean-validation/README.md.jinja +172 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/clean-validation/docker-compose.yml.jinja +78 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/clean-validation/mkdocs.yml.jinja +62 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/clean-validation/pyproject.toml.jinja +120 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/docker-compose.yml.jinja +200 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/docs/api.md.jinja +191 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/docs/components/scheduler.md +0 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/docs/components/scheduler.md.jinja +621 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/docs/development.md.jinja +215 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/docs/health.md.jinja +240 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/docs/javascripts/mermaid-config.js +62 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/docs/stylesheets/mermaid.css +95 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/mkdocs.yml.jinja +62 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/pyproject.toml.jinja +131 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/scripts/entrypoint.sh +87 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/scripts/entrypoint.sh.jinja +93 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/scripts/gen_docs.py +16 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/tests/api/__init__.py +1 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/tests/api/test_auth_endpoints.py.jinja +307 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/tests/api/test_health_endpoints.py.jinja +262 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/tests/api/test_scheduler_endpoints.py.jinja +214 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/tests/api/test_worker_endpoints.py.jinja +165 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/tests/cli/test_ai_rendering.py +427 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/tests/cli/test_conversation_memory.py +465 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/tests/components/test_scheduler.py +43 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/tests/conftest.py.jinja +195 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/tests/services/__init__.py +1 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/tests/services/ai/__init__.py +1 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/tests/services/ai/conftest.py +78 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/tests/services/ai/test_health.py +157 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/tests/services/ai/test_models.py +164 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/tests/services/ai/test_service.py +198 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/tests/services/test_auth_integration.py.jinja +528 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/tests/services/test_component_integration.py.jinja +387 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/tests/services/test_conversation_persistence.py +342 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/tests/services/test_health_logic.py.jinja +663 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/tests/services/test_load_test_models.py +619 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/tests/services/test_load_test_service.py +603 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/tests/services/test_middleware_inspector.py.jinja +248 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/tests/services/test_scheduled_task_manager.py.jinja +292 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/tests/services/test_system_service.py +98 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/tests/services/test_worker_health_registration.py.jinja +257 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/tests/test_core.py +49 -0
- aegis/templates/copier-aegis-project/{{ project_slug }}/uv.lock +1673 -0
- aegis_stack-0.2.0rc2.dist-info/METADATA +165 -0
- aegis_stack-0.2.0rc2.dist-info/RECORD +392 -0
- aegis_stack-0.2.0rc2.dist-info/WHEEL +4 -0
- aegis_stack-0.2.0rc2.dist-info/entry_points.txt +3 -0
- aegis_stack-0.2.0rc2.dist-info/licenses/LICENSE +21 -0
|
@@ -0,0 +1,419 @@
|
|
|
1
|
+
"""
|
|
2
|
+
Health monitoring CLI commands.
|
|
3
|
+
|
|
4
|
+
Command-line interface for system health checking and monitoring via API endpoints.
|
|
5
|
+
"""
|
|
6
|
+
|
|
7
|
+
import asyncio
|
|
8
|
+
import json
|
|
9
|
+
import sys
|
|
10
|
+
from typing import Any
|
|
11
|
+
|
|
12
|
+
import httpx
|
|
13
|
+
from rich.console import Console
|
|
14
|
+
from rich.panel import Panel
|
|
15
|
+
import typer
|
|
16
|
+
|
|
17
|
+
from app.core.config import settings
|
|
18
|
+
from app.core.constants import CLI, APIEndpoints, Defaults
|
|
19
|
+
from app.core.log import setup_logging
|
|
20
|
+
from app.services.system.models import (
|
|
21
|
+
ComponentStatusType,
|
|
22
|
+
DetailedHealthResponse,
|
|
23
|
+
HealthResponse,
|
|
24
|
+
)
|
|
25
|
+
from app.services.system.ui import get_status_color_name, get_status_icon
|
|
26
|
+
|
|
27
|
+
app = typer.Typer(name="health", help="System health monitoring commands")
|
|
28
|
+
console = Console()
|
|
29
|
+
|
|
30
|
+
|
|
31
|
+
def _get_status_icon_and_color(status: ComponentStatusType) -> tuple[str, str]:
|
|
32
|
+
"""Get the appropriate icon and color for a component status (shared mapping)."""
|
|
33
|
+
return get_status_icon(status), get_status_color_name(status)
|
|
34
|
+
|
|
35
|
+
|
|
36
|
+
def _is_scheduler_metadata(metadata: dict[str, Any]) -> bool:
|
|
37
|
+
"""Check if metadata contains scheduler-specific task information."""
|
|
38
|
+
return "total_tasks" in metadata and "upcoming_tasks" in metadata
|
|
39
|
+
|
|
40
|
+
|
|
41
|
+
def _display_scheduler_metadata(
|
|
42
|
+
metadata: dict[str, Any], base_indent: str, is_last: bool, detailed: bool = False
|
|
43
|
+
) -> None:
|
|
44
|
+
"""Display scheduler metadata in a structured, readable format."""
|
|
45
|
+
tree_indent = f"{base_indent} " if is_last else f"{base_indent}│ "
|
|
46
|
+
|
|
47
|
+
# Task statistics
|
|
48
|
+
total_tasks = metadata.get("total_tasks", 0)
|
|
49
|
+
active_tasks = metadata.get("active_tasks", 0)
|
|
50
|
+
paused_tasks = metadata.get("paused_tasks", 0)
|
|
51
|
+
|
|
52
|
+
console.print(f"{tree_indent}[dim]Task Statistics:[/dim]")
|
|
53
|
+
console.print(
|
|
54
|
+
f"{tree_indent} [dim]• Total: {total_tasks}, Active: {active_tasks}, "
|
|
55
|
+
f"Paused: {paused_tasks}[/dim]"
|
|
56
|
+
)
|
|
57
|
+
|
|
58
|
+
# Upcoming tasks
|
|
59
|
+
upcoming_tasks = metadata.get("upcoming_tasks", [])
|
|
60
|
+
if upcoming_tasks:
|
|
61
|
+
console.print(f"{tree_indent}[dim]Upcoming Tasks:[/dim]")
|
|
62
|
+
# In detailed mode, show all tasks. Otherwise show top 3
|
|
63
|
+
max_tasks = len(upcoming_tasks) if detailed else 3
|
|
64
|
+
for task in upcoming_tasks[:max_tasks]:
|
|
65
|
+
task_name = task.get("name", task.get("id", "Unknown"))
|
|
66
|
+
next_run = task.get("next_run", "Unknown")
|
|
67
|
+
|
|
68
|
+
# Format next run time more human-readable
|
|
69
|
+
if next_run and next_run != "Unknown":
|
|
70
|
+
try:
|
|
71
|
+
from datetime import datetime
|
|
72
|
+
|
|
73
|
+
if next_run.endswith("+00:00") or next_run.endswith("Z"):
|
|
74
|
+
dt = datetime.fromisoformat(next_run.replace("Z", "+00:00"))
|
|
75
|
+
formatted_time = dt.strftime("%Y-%m-%d %H:%M UTC")
|
|
76
|
+
else:
|
|
77
|
+
formatted_time = next_run
|
|
78
|
+
except Exception:
|
|
79
|
+
formatted_time = next_run
|
|
80
|
+
else:
|
|
81
|
+
formatted_time = "Unknown"
|
|
82
|
+
|
|
83
|
+
console.print(
|
|
84
|
+
f"{tree_indent} [dim]• {task_name} - Next: {formatted_time}[/dim]"
|
|
85
|
+
)
|
|
86
|
+
|
|
87
|
+
# Show "and X more..." if there are more tasks (only in non-detailed mode)
|
|
88
|
+
if not detailed and len(upcoming_tasks) > 3:
|
|
89
|
+
remaining = len(upcoming_tasks) - 3
|
|
90
|
+
task_word = "task" if remaining == 1 else "tasks"
|
|
91
|
+
console.print(
|
|
92
|
+
f"{tree_indent} [dim]• ... and {remaining} more {task_word}[/dim]"
|
|
93
|
+
)
|
|
94
|
+
else:
|
|
95
|
+
console.print(f"{tree_indent}[dim]No upcoming tasks scheduled[/dim]")
|
|
96
|
+
|
|
97
|
+
|
|
98
|
+
async def get_health_data(
|
|
99
|
+
endpoint: str = APIEndpoints.HEALTH_BASIC,
|
|
100
|
+
) -> HealthResponse | DetailedHealthResponse:
|
|
101
|
+
"""Get health data from the API endpoint with Pydantic validation."""
|
|
102
|
+
base_url = getattr(settings, "API_BASE_URL", "http://localhost:8000")
|
|
103
|
+
url = f"{base_url}{endpoint}"
|
|
104
|
+
|
|
105
|
+
timeout = httpx.Timeout(Defaults.API_TIMEOUT)
|
|
106
|
+
async with httpx.AsyncClient(timeout=timeout) as client:
|
|
107
|
+
try:
|
|
108
|
+
response = await client.get(url)
|
|
109
|
+
response.raise_for_status()
|
|
110
|
+
json_data = response.json()
|
|
111
|
+
|
|
112
|
+
# Validate response with appropriate Pydantic model
|
|
113
|
+
if endpoint == APIEndpoints.HEALTH_DETAILED:
|
|
114
|
+
return DetailedHealthResponse.model_validate(json_data)
|
|
115
|
+
else:
|
|
116
|
+
return HealthResponse.model_validate(json_data)
|
|
117
|
+
|
|
118
|
+
except httpx.ConnectError:
|
|
119
|
+
raise ConnectionError(
|
|
120
|
+
f"Cannot connect to API server at {base_url}. "
|
|
121
|
+
"Make sure the application is running with "
|
|
122
|
+
"'make server' or 'make run-dev'."
|
|
123
|
+
) from None
|
|
124
|
+
except httpx.TimeoutException:
|
|
125
|
+
raise TimeoutError(
|
|
126
|
+
f"API request to {url} timed out after {Defaults.API_TIMEOUT} seconds."
|
|
127
|
+
) from None
|
|
128
|
+
except httpx.HTTPStatusError as e:
|
|
129
|
+
# Handle structured error responses from health endpoint
|
|
130
|
+
if e.response.status_code == 503:
|
|
131
|
+
try:
|
|
132
|
+
error_data = e.response.json()
|
|
133
|
+
if "detail" in error_data and isinstance(
|
|
134
|
+
error_data["detail"], dict
|
|
135
|
+
):
|
|
136
|
+
detail = error_data["detail"]
|
|
137
|
+
message = detail.get("message", "System is unhealthy")
|
|
138
|
+
unhealthy_components = detail.get("unhealthy_components", [])
|
|
139
|
+
health_percentage = detail.get("health_percentage", 0)
|
|
140
|
+
|
|
141
|
+
error_msg = f"{message}"
|
|
142
|
+
if unhealthy_components:
|
|
143
|
+
components_str = ", ".join(unhealthy_components)
|
|
144
|
+
error_msg += f" (Unhealthy: {components_str})"
|
|
145
|
+
if health_percentage is not None:
|
|
146
|
+
error_msg += f" - Health: {health_percentage:.1f}%"
|
|
147
|
+
|
|
148
|
+
raise RuntimeError(error_msg) from None
|
|
149
|
+
except (ValueError, KeyError, TypeError):
|
|
150
|
+
# Fall back to generic error message if JSON parsing fails
|
|
151
|
+
pass
|
|
152
|
+
|
|
153
|
+
raise RuntimeError(
|
|
154
|
+
f"API returned error {e.response.status_code}: {e.response.text}"
|
|
155
|
+
) from None
|
|
156
|
+
|
|
157
|
+
|
|
158
|
+
async def is_system_healthy() -> bool:
|
|
159
|
+
"""Quick check if system is healthy via API."""
|
|
160
|
+
try:
|
|
161
|
+
health_data = await get_health_data(APIEndpoints.HEALTH_BASIC)
|
|
162
|
+
return health_data.healthy
|
|
163
|
+
except Exception:
|
|
164
|
+
return False
|
|
165
|
+
|
|
166
|
+
|
|
167
|
+
@app.command("status")
|
|
168
|
+
def health_status(
|
|
169
|
+
detailed: bool = typer.Option(
|
|
170
|
+
False, "--detailed", "-d", help="Show detailed component information"
|
|
171
|
+
),
|
|
172
|
+
json_output: bool = typer.Option(False, "--json", "-j", help="Output as JSON"),
|
|
173
|
+
) -> None:
|
|
174
|
+
"""Show system health status (always exits 0 for inspection).
|
|
175
|
+
|
|
176
|
+
Use --detailed for component metadata and response times."""
|
|
177
|
+
setup_logging()
|
|
178
|
+
|
|
179
|
+
try:
|
|
180
|
+
endpoint = (
|
|
181
|
+
APIEndpoints.HEALTH_DETAILED if detailed else APIEndpoints.HEALTH_BASIC
|
|
182
|
+
)
|
|
183
|
+
health_data = asyncio.run(get_health_data(endpoint))
|
|
184
|
+
|
|
185
|
+
if json_output:
|
|
186
|
+
print(json.dumps(health_data.model_dump(), indent=2))
|
|
187
|
+
else:
|
|
188
|
+
_display_health_status(health_data, detailed)
|
|
189
|
+
|
|
190
|
+
# Always exit 0 for status command (informational)
|
|
191
|
+
|
|
192
|
+
except Exception as e:
|
|
193
|
+
if json_output:
|
|
194
|
+
error_data = {"error": str(e), "status": "error"}
|
|
195
|
+
print(json.dumps(error_data, indent=2))
|
|
196
|
+
else:
|
|
197
|
+
console.print(f"[red]❌ Health status failed: {e}[/red]")
|
|
198
|
+
# Exit 1 only on actual errors (connection failures, etc), not unhealthy status
|
|
199
|
+
sys.exit(1)
|
|
200
|
+
|
|
201
|
+
|
|
202
|
+
@app.command("probe")
|
|
203
|
+
def health_probe() -> None:
|
|
204
|
+
"""Health probe for monitoring - exits 1 if unhealthy (like k8s probes)."""
|
|
205
|
+
setup_logging()
|
|
206
|
+
|
|
207
|
+
try:
|
|
208
|
+
healthy = asyncio.run(is_system_healthy())
|
|
209
|
+
|
|
210
|
+
if healthy:
|
|
211
|
+
console.print("[green]✅ System is healthy[/green]")
|
|
212
|
+
sys.exit(0)
|
|
213
|
+
else:
|
|
214
|
+
console.print("[red]❌ System is unhealthy[/red]")
|
|
215
|
+
sys.exit(1)
|
|
216
|
+
|
|
217
|
+
except Exception as e:
|
|
218
|
+
console.print(f"[red]💥 Health probe failed: {e}[/red]")
|
|
219
|
+
sys.exit(1)
|
|
220
|
+
|
|
221
|
+
|
|
222
|
+
def _display_sub_components(
|
|
223
|
+
sub_components: dict[str, Any], detailed: bool, level: int
|
|
224
|
+
) -> None:
|
|
225
|
+
"""Recursively display sub-components with appropriate tree indentation."""
|
|
226
|
+
sub_items = list(sub_components.items())
|
|
227
|
+
|
|
228
|
+
# Calculate tree indentation based on level
|
|
229
|
+
base_indent = " " * level
|
|
230
|
+
|
|
231
|
+
for i, (sub_name, sub_component) in enumerate(sub_items):
|
|
232
|
+
sub_icon, sub_color = _get_status_icon_and_color(sub_component.status)
|
|
233
|
+
|
|
234
|
+
# Tree connector: ├── for middle items, └── for last item
|
|
235
|
+
is_last = i == len(sub_items) - 1
|
|
236
|
+
tree_connector = f"{base_indent}└── " if is_last else f"{base_indent}├── "
|
|
237
|
+
|
|
238
|
+
sub_line = f"{tree_connector}[{sub_color}]{sub_icon} {sub_name}[/{sub_color}]"
|
|
239
|
+
if detailed and sub_component.response_time_ms is not None:
|
|
240
|
+
sub_line += f" ([dim]{sub_component.response_time_ms:.1f}ms[/dim])"
|
|
241
|
+
sub_line += f" {sub_component.message}"
|
|
242
|
+
console.print(sub_line)
|
|
243
|
+
|
|
244
|
+
# Recursively display sub-sub-components
|
|
245
|
+
if hasattr(sub_component, "sub_components") and sub_component.sub_components:
|
|
246
|
+
_display_sub_components(sub_component.sub_components, detailed, level + 1)
|
|
247
|
+
|
|
248
|
+
# Show metadata for sub-components if detailed and available
|
|
249
|
+
elif detailed and sub_component.metadata:
|
|
250
|
+
# Special handling for scheduler component
|
|
251
|
+
if sub_name == "scheduler" and _is_scheduler_metadata(
|
|
252
|
+
sub_component.metadata
|
|
253
|
+
):
|
|
254
|
+
_display_scheduler_metadata(
|
|
255
|
+
sub_component.metadata, base_indent, is_last, detailed
|
|
256
|
+
)
|
|
257
|
+
else:
|
|
258
|
+
# Generic metadata display for other components
|
|
259
|
+
metadata_str = json.dumps(sub_component.metadata, separators=(",", ":"))
|
|
260
|
+
max_length = CLI.MAX_METADATA_DISPLAY_LENGTH
|
|
261
|
+
if len(metadata_str) > max_length:
|
|
262
|
+
metadata_str = metadata_str[: max_length - 3] + "..."
|
|
263
|
+
# Adjust tree indent based on whether this is the last item
|
|
264
|
+
tree_indent = f"{base_indent} " if is_last else f"{base_indent}│ "
|
|
265
|
+
console.print(f"{tree_indent}[dim]({metadata_str})[/dim]")
|
|
266
|
+
|
|
267
|
+
|
|
268
|
+
def _display_health_status(
|
|
269
|
+
health_data: HealthResponse | DetailedHealthResponse, detailed: bool = False
|
|
270
|
+
) -> None:
|
|
271
|
+
"""Display health status with rich formatting."""
|
|
272
|
+
|
|
273
|
+
# Extract data from Pydantic model
|
|
274
|
+
overall_healthy = health_data.healthy
|
|
275
|
+
components = health_data.components
|
|
276
|
+
timestamp = health_data.timestamp
|
|
277
|
+
|
|
278
|
+
# Use health percentage from API response if available (DetailedHealthResponse)
|
|
279
|
+
# Otherwise calculate from top-level components (HealthResponse)
|
|
280
|
+
if hasattr(health_data, "health_percentage"):
|
|
281
|
+
health_percentage = health_data.health_percentage
|
|
282
|
+
# For detailed response, use the component counts from API
|
|
283
|
+
if hasattr(health_data, "healthy_components"):
|
|
284
|
+
if hasattr(health_data, "unhealthy_components"):
|
|
285
|
+
healthy_count = len(health_data.healthy_components)
|
|
286
|
+
total_count = len(health_data.healthy_components) + len(
|
|
287
|
+
health_data.unhealthy_components
|
|
288
|
+
)
|
|
289
|
+
else:
|
|
290
|
+
# HealthResponse doesn't have detailed component lists
|
|
291
|
+
healthy_count = 1 if health_data.healthy else 0
|
|
292
|
+
total_count = 1
|
|
293
|
+
else:
|
|
294
|
+
# Fallback for detailed response without component lists
|
|
295
|
+
healthy_count = sum(1 for comp in components.values() if comp.healthy)
|
|
296
|
+
total_count = len(components)
|
|
297
|
+
else:
|
|
298
|
+
# Basic health response - count main sub-components for better overview
|
|
299
|
+
if components and "aegis" in components:
|
|
300
|
+
aegis_component = components["aegis"]
|
|
301
|
+
if (
|
|
302
|
+
hasattr(aegis_component, "sub_components")
|
|
303
|
+
and aegis_component.sub_components
|
|
304
|
+
):
|
|
305
|
+
healthy_count = sum(
|
|
306
|
+
1
|
|
307
|
+
for comp in aegis_component.sub_components.values()
|
|
308
|
+
if comp.healthy
|
|
309
|
+
)
|
|
310
|
+
total_count = len(aegis_component.sub_components)
|
|
311
|
+
health_percentage = (
|
|
312
|
+
(healthy_count / total_count) * 100 if total_count > 0 else 100.0
|
|
313
|
+
)
|
|
314
|
+
else:
|
|
315
|
+
# Fallback to aegis component only
|
|
316
|
+
healthy_count = 1 if aegis_component.healthy else 0
|
|
317
|
+
total_count = 1
|
|
318
|
+
health_percentage = 100.0 if aegis_component.healthy else 0.0
|
|
319
|
+
else:
|
|
320
|
+
# No components or no aegis component
|
|
321
|
+
healthy_count = 0
|
|
322
|
+
total_count = 0
|
|
323
|
+
health_percentage = 0.0
|
|
324
|
+
|
|
325
|
+
overall_color = "green" if overall_healthy else "red"
|
|
326
|
+
overall_icon = "✅" if overall_healthy else "❌"
|
|
327
|
+
|
|
328
|
+
title = f"System Health - {overall_icon}"
|
|
329
|
+
if overall_healthy:
|
|
330
|
+
title += " Healthy"
|
|
331
|
+
else:
|
|
332
|
+
title += " Unhealthy"
|
|
333
|
+
|
|
334
|
+
panel_content = [
|
|
335
|
+
f"Overall Status: [bold {overall_color}]"
|
|
336
|
+
+ ("Healthy" if overall_healthy else "Unhealthy")
|
|
337
|
+
+ f"[/bold {overall_color}]",
|
|
338
|
+
f"Health Percentage: [bold]"
|
|
339
|
+
f"{health_percentage:.{CLI.HEALTH_PERCENTAGE_DECIMALS}f}%[/bold]",
|
|
340
|
+
f"Components: {healthy_count}/" + f"{total_count} healthy",
|
|
341
|
+
]
|
|
342
|
+
|
|
343
|
+
# Add service information if available (DetailedHealthResponse)
|
|
344
|
+
if hasattr(health_data, "has_services") and health_data.has_services:
|
|
345
|
+
healthy_services_count = len(health_data.healthy_services)
|
|
346
|
+
total_services_count = len(health_data.service_names)
|
|
347
|
+
if healthy_services_count == total_services_count:
|
|
348
|
+
service_status_color = "green"
|
|
349
|
+
elif healthy_services_count > 0:
|
|
350
|
+
service_status_color = "yellow"
|
|
351
|
+
else:
|
|
352
|
+
service_status_color = "red"
|
|
353
|
+
|
|
354
|
+
panel_content.append(
|
|
355
|
+
f"Services: [bold {service_status_color}]{healthy_services_count}/"
|
|
356
|
+
f"{total_services_count} healthy[/bold {service_status_color}]"
|
|
357
|
+
)
|
|
358
|
+
|
|
359
|
+
panel_content.append(f"Timestamp: {timestamp}")
|
|
360
|
+
|
|
361
|
+
console.print(
|
|
362
|
+
Panel("\n".join(panel_content), title=title, border_style=overall_color)
|
|
363
|
+
)
|
|
364
|
+
|
|
365
|
+
# Component and Service Tree Display
|
|
366
|
+
tree_title = "Component Tree"
|
|
367
|
+
if hasattr(health_data, "has_services") and health_data.has_services:
|
|
368
|
+
tree_title = "Component & Service Tree"
|
|
369
|
+
console.print(f"\n[bold magenta]{tree_title}:[/bold magenta]")
|
|
370
|
+
|
|
371
|
+
# Sort components: unhealthy first, then by name
|
|
372
|
+
sorted_components = sorted(components.items(), key=lambda x: (x[1].healthy, x[0]))
|
|
373
|
+
|
|
374
|
+
for name, component in sorted_components:
|
|
375
|
+
status_icon, status_color = _get_status_icon_and_color(component.status)
|
|
376
|
+
|
|
377
|
+
# Display main component
|
|
378
|
+
component_line = f"[{status_color}]{status_icon} {name}[/{status_color}]"
|
|
379
|
+
if detailed and component.response_time_ms is not None:
|
|
380
|
+
component_line += f" ([dim]{component.response_time_ms:.1f}ms[/dim])"
|
|
381
|
+
component_line += f" {component.message}"
|
|
382
|
+
console.print(component_line)
|
|
383
|
+
|
|
384
|
+
# Display sub-components with tree structure (recursive)
|
|
385
|
+
if hasattr(component, "sub_components") and component.sub_components:
|
|
386
|
+
_display_sub_components(component.sub_components, detailed, level=1)
|
|
387
|
+
|
|
388
|
+
# Show metadata for main components if detailed and available
|
|
389
|
+
elif detailed and component.metadata:
|
|
390
|
+
# Special handling for scheduler component
|
|
391
|
+
if name == "scheduler" and _is_scheduler_metadata(component.metadata):
|
|
392
|
+
_display_scheduler_metadata(component.metadata, "", True, detailed)
|
|
393
|
+
else:
|
|
394
|
+
# Generic metadata display for other components
|
|
395
|
+
metadata_str = json.dumps(component.metadata, separators=(",", ":"))
|
|
396
|
+
max_length = CLI.MAX_METADATA_DISPLAY_LENGTH
|
|
397
|
+
if len(metadata_str) > max_length:
|
|
398
|
+
metadata_str = metadata_str[: max_length - 3] + "..."
|
|
399
|
+
console.print(f" [dim]({metadata_str})[/dim]")
|
|
400
|
+
|
|
401
|
+
# System information (only in detailed mode)
|
|
402
|
+
if detailed and isinstance(health_data, DetailedHealthResponse):
|
|
403
|
+
system_info = health_data.system_info
|
|
404
|
+
if system_info:
|
|
405
|
+
sys_info_content = []
|
|
406
|
+
for key, value in system_info.items():
|
|
407
|
+
sys_info_content.append(f"{key.replace('_', ' ').title()}: {value}")
|
|
408
|
+
|
|
409
|
+
console.print(
|
|
410
|
+
Panel(
|
|
411
|
+
"\n".join(sys_info_content),
|
|
412
|
+
title="System Information",
|
|
413
|
+
border_style="blue",
|
|
414
|
+
)
|
|
415
|
+
)
|
|
416
|
+
|
|
417
|
+
|
|
418
|
+
if __name__ == "__main__":
|
|
419
|
+
app()
|