qualia-framework 2.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +50 -0
- package/bin/cli.js +519 -0
- package/framework/agents/architecture-strategist.md +53 -0
- package/framework/agents/backend-agent.md +150 -0
- package/framework/agents/code-simplicity-reviewer.md +86 -0
- package/framework/agents/frontend-agent.md +111 -0
- package/framework/agents/kieran-typescript-reviewer.md +96 -0
- package/framework/agents/performance-oracle.md +111 -0
- package/framework/agents/qualia-codebase-mapper.md +760 -0
- package/framework/agents/qualia-debugger.md +1203 -0
- package/framework/agents/qualia-executor.md +881 -0
- package/framework/agents/qualia-integration-checker.md +423 -0
- package/framework/agents/qualia-phase-researcher.md +453 -0
- package/framework/agents/qualia-plan-checker.md +699 -0
- package/framework/agents/qualia-planner.md +1241 -0
- package/framework/agents/qualia-project-researcher.md +602 -0
- package/framework/agents/qualia-research-synthesizer.md +236 -0
- package/framework/agents/qualia-roadmapper.md +605 -0
- package/framework/agents/qualia-verifier.md +685 -0
- package/framework/agents/team-orchestrator.md +228 -0
- package/framework/agents/teams/full-stack-team.md +48 -0
- package/framework/agents/teams/optimize-team.md +53 -0
- package/framework/agents/teams/review-team.md +62 -0
- package/framework/agents/teams/ship-team.md +86 -0
- package/framework/agents/test-agent.md +182 -0
- package/framework/askpass.sh +2 -0
- package/framework/commands/design.md +53 -0
- package/framework/commands/quick-db.md +22 -0
- package/framework/config/retention.json +35 -0
- package/framework/core/PRINCIPLES.md +77 -0
- package/framework/hooks/auto-format.sh +45 -0
- package/framework/hooks/block-env-edit.sh +42 -0
- package/framework/hooks/branch-guard.sh +46 -0
- package/framework/hooks/confirm-delete.sh +56 -0
- package/framework/hooks/migration-validate.sh +68 -0
- package/framework/hooks/notification-speak.sh +15 -0
- package/framework/hooks/pre-commit.sh +80 -0
- package/framework/hooks/pre-compact.sh +55 -0
- package/framework/hooks/pre-deploy-gate.sh +151 -0
- package/framework/hooks/qualia-colors.sh +32 -0
- package/framework/hooks/retention-cleanup.sh +43 -0
- package/framework/hooks/save-session-state.sh +153 -0
- package/framework/hooks/session-context-loader.sh +28 -0
- package/framework/hooks/session-learn.sh +30 -0
- package/framework/knowledge/claudecode-bible.md +1384 -0
- package/framework/knowledge/client-prefs.md +22 -0
- package/framework/knowledge/common-fixes.md +25 -0
- package/framework/knowledge/deployment-map.md +35 -0
- package/framework/knowledge/email-signature.html +1 -0
- package/framework/knowledge/employees.md +8 -0
- package/framework/knowledge/learned-patterns.md +51 -0
- package/framework/knowledge/optimization-research-2026.md +137 -0
- package/framework/knowledge/qualia-context.md +67 -0
- package/framework/knowledge/supabase-patterns.md +50 -0
- package/framework/knowledge/voice-agent-patterns.md +46 -0
- package/framework/qualia-engine/VERSION +1 -0
- package/framework/qualia-engine/bin/qualia-tools.js +2160 -0
- package/framework/qualia-engine/bin/qualia-tools.test.js +1054 -0
- package/framework/qualia-engine/references/checkpoints.md +775 -0
- package/framework/qualia-engine/references/continuation-format.md +249 -0
- package/framework/qualia-engine/references/decimal-phase-calculation.md +65 -0
- package/framework/qualia-engine/references/design-quality.md +56 -0
- package/framework/qualia-engine/references/git-integration.md +254 -0
- package/framework/qualia-engine/references/git-planning-commit.md +50 -0
- package/framework/qualia-engine/references/model-profile-resolution.md +32 -0
- package/framework/qualia-engine/references/model-profiles.md +73 -0
- package/framework/qualia-engine/references/phase-argument-parsing.md +61 -0
- package/framework/qualia-engine/references/planning-config.md +195 -0
- package/framework/qualia-engine/references/questioning.md +141 -0
- package/framework/qualia-engine/references/tdd.md +263 -0
- package/framework/qualia-engine/references/ui-brand.md +160 -0
- package/framework/qualia-engine/references/verification-patterns.md +612 -0
- package/framework/qualia-engine/templates/DEBUG.md +159 -0
- package/framework/qualia-engine/templates/DESIGN.md +81 -0
- package/framework/qualia-engine/templates/UAT.md +247 -0
- package/framework/qualia-engine/templates/codebase/architecture.md +255 -0
- package/framework/qualia-engine/templates/codebase/concerns.md +310 -0
- package/framework/qualia-engine/templates/codebase/conventions.md +307 -0
- package/framework/qualia-engine/templates/codebase/integrations.md +280 -0
- package/framework/qualia-engine/templates/codebase/stack.md +186 -0
- package/framework/qualia-engine/templates/codebase/structure.md +285 -0
- package/framework/qualia-engine/templates/codebase/testing.md +480 -0
- package/framework/qualia-engine/templates/config.json +35 -0
- package/framework/qualia-engine/templates/context.md +283 -0
- package/framework/qualia-engine/templates/continue-here.md +78 -0
- package/framework/qualia-engine/templates/debug-subagent-prompt.md +91 -0
- package/framework/qualia-engine/templates/discovery.md +146 -0
- package/framework/qualia-engine/templates/milestone-archive.md +123 -0
- package/framework/qualia-engine/templates/milestone.md +115 -0
- package/framework/qualia-engine/templates/phase-prompt.md +567 -0
- package/framework/qualia-engine/templates/planner-subagent-prompt.md +117 -0
- package/framework/qualia-engine/templates/project.md +184 -0
- package/framework/qualia-engine/templates/projects/ai-agent.md +156 -0
- package/framework/qualia-engine/templates/projects/mobile-app.md +181 -0
- package/framework/qualia-engine/templates/projects/voice-agent.md +134 -0
- package/framework/qualia-engine/templates/projects/website.md +137 -0
- package/framework/qualia-engine/templates/requirements.md +231 -0
- package/framework/qualia-engine/templates/research-project/ARCHITECTURE.md +204 -0
- package/framework/qualia-engine/templates/research-project/FEATURES.md +147 -0
- package/framework/qualia-engine/templates/research-project/PITFALLS.md +200 -0
- package/framework/qualia-engine/templates/research-project/STACK.md +120 -0
- package/framework/qualia-engine/templates/research-project/SUMMARY.md +170 -0
- package/framework/qualia-engine/templates/research.md +552 -0
- package/framework/qualia-engine/templates/roadmap.md +202 -0
- package/framework/qualia-engine/templates/state.md +176 -0
- package/framework/qualia-engine/templates/summary-complex.md +59 -0
- package/framework/qualia-engine/templates/summary-minimal.md +41 -0
- package/framework/qualia-engine/templates/summary-standard.md +48 -0
- package/framework/qualia-engine/templates/summary.md +246 -0
- package/framework/qualia-engine/templates/user-setup.md +311 -0
- package/framework/qualia-engine/templates/verification-report.md +322 -0
- package/framework/qualia-engine/workflows/add-phase.md +179 -0
- package/framework/qualia-engine/workflows/add-todo.md +157 -0
- package/framework/qualia-engine/workflows/audit-milestone.md +241 -0
- package/framework/qualia-engine/workflows/check-todos.md +176 -0
- package/framework/qualia-engine/workflows/complete-milestone.md +858 -0
- package/framework/qualia-engine/workflows/diagnose-issues.md +219 -0
- package/framework/qualia-engine/workflows/discovery-phase.md +289 -0
- package/framework/qualia-engine/workflows/discuss-phase.md +534 -0
- package/framework/qualia-engine/workflows/execute-phase.md +559 -0
- package/framework/qualia-engine/workflows/execute-plan.md +438 -0
- package/framework/qualia-engine/workflows/help.md +470 -0
- package/framework/qualia-engine/workflows/insert-phase.md +220 -0
- package/framework/qualia-engine/workflows/list-phase-assumptions.md +178 -0
- package/framework/qualia-engine/workflows/map-codebase.md +327 -0
- package/framework/qualia-engine/workflows/new-milestone.md +363 -0
- package/framework/qualia-engine/workflows/new-project.md +1037 -0
- package/framework/qualia-engine/workflows/pause-work.md +122 -0
- package/framework/qualia-engine/workflows/plan-milestone-gaps.md +256 -0
- package/framework/qualia-engine/workflows/plan-phase.md +422 -0
- package/framework/qualia-engine/workflows/progress.md +354 -0
- package/framework/qualia-engine/workflows/quick.md +252 -0
- package/framework/qualia-engine/workflows/remove-phase.md +326 -0
- package/framework/qualia-engine/workflows/research-phase.md +74 -0
- package/framework/qualia-engine/workflows/resume-project.md +306 -0
- package/framework/qualia-engine/workflows/set-profile.md +80 -0
- package/framework/qualia-engine/workflows/settings.md +145 -0
- package/framework/qualia-engine/workflows/transition.md +556 -0
- package/framework/qualia-engine/workflows/update.md +197 -0
- package/framework/qualia-engine/workflows/verify-phase.md +195 -0
- package/framework/qualia-engine/workflows/verify-work.md +625 -0
- package/framework/rules/context7.md +11 -0
- package/framework/rules/deployment.md +29 -0
- package/framework/rules/frontend.md +33 -0
- package/framework/rules/security.md +12 -0
- package/framework/rules/speed.md +20 -0
- package/framework/scripts/__pycache__/say.cpython-314.pyc +0 -0
- package/framework/scripts/apply-retention.sh +120 -0
- package/framework/scripts/bootstrap-pop-os.sh +354 -0
- package/framework/scripts/claude-voice +13 -0
- package/framework/scripts/cleanup.sh +131 -0
- package/framework/scripts/cowork-mode.sh +141 -0
- package/framework/scripts/generate-project-claude-md.sh +153 -0
- package/framework/scripts/load-test-webhook.js +172 -0
- package/framework/scripts/say.py +236 -0
- package/framework/scripts/showcase-video-recorder/ffmpeg-builder.js +167 -0
- package/framework/scripts/showcase-video-recorder/playwright-helpers.js +216 -0
- package/framework/scripts/speak.py +55 -0
- package/framework/scripts/speak.sh +18 -0
- package/framework/scripts/status.sh +138 -0
- package/framework/scripts/sync-to-framework.sh +65 -0
- package/framework/scripts/voice-hotkey.py +227 -0
- package/framework/scripts/voice-input.sh +51 -0
- package/framework/skills/animate/SKILL.md +202 -0
- package/framework/skills/bolder/SKILL.md +144 -0
- package/framework/skills/browser-qa/SKILL.md +536 -0
- package/framework/skills/clarify/SKILL.md +179 -0
- package/framework/skills/colorize/SKILL.md +170 -0
- package/framework/skills/critique/SKILL.md +126 -0
- package/framework/skills/deep-research/SKILL.md +271 -0
- package/framework/skills/delight/SKILL.md +329 -0
- package/framework/skills/deploy/SKILL.md +261 -0
- package/framework/skills/deploy-verify/SKILL.md +377 -0
- package/framework/skills/deploy-verify/scripts/canary-check.sh +206 -0
- package/framework/skills/deploy-verify/scripts/check-console-errors.js +147 -0
- package/framework/skills/deploy-verify/scripts/check-cwv.js +139 -0
- package/framework/skills/deploy-verify/scripts/project-detect.sh +84 -0
- package/framework/skills/deploy-verify/scripts/verify.sh +548 -0
- package/framework/skills/design-quieter/SKILL.md +130 -0
- package/framework/skills/distill/SKILL.md +149 -0
- package/framework/skills/docs-lookup/SKILL.md +78 -0
- package/framework/skills/fcm-notifications/SKILL.md +125 -0
- package/framework/skills/financial-ledger/SKILL.md +1039 -0
- package/framework/skills/frontend-master/NOTICE.md +4 -0
- package/framework/skills/frontend-master/SKILL.md +127 -0
- package/framework/skills/frontend-master/reference/color-and-contrast.md +132 -0
- package/framework/skills/frontend-master/reference/interaction-design.md +123 -0
- package/framework/skills/frontend-master/reference/motion-design.md +99 -0
- package/framework/skills/frontend-master/reference/responsive-design.md +114 -0
- package/framework/skills/frontend-master/reference/spatial-design.md +100 -0
- package/framework/skills/frontend-master/reference/typography.md +131 -0
- package/framework/skills/frontend-master/reference/ux-writing.md +107 -0
- package/framework/skills/harden/SKILL.md +357 -0
- package/framework/skills/i18n-rtl/SKILL.md +752 -0
- package/framework/skills/learn/SKILL.md +71 -0
- package/framework/skills/memory/SKILL.md +50 -0
- package/framework/skills/mobile-expo/SKILL.md +864 -0
- package/framework/skills/mobile-expo/references/store-checklist.md +550 -0
- package/framework/skills/nestjs-backend/README.md +73 -0
- package/framework/skills/nestjs-backend/SKILL.md +446 -0
- package/framework/skills/nestjs-backend/references/templates.md +1173 -0
- package/framework/skills/normalize/SKILL.md +79 -0
- package/framework/skills/onboard/SKILL.md +242 -0
- package/framework/skills/polish/SKILL.md +209 -0
- package/framework/skills/pr/SKILL.md +66 -0
- package/framework/skills/qualia/SKILL.md +153 -0
- package/framework/skills/qualia-add-todo/SKILL.md +68 -0
- package/framework/skills/qualia-audit-milestone/SKILL.md +92 -0
- package/framework/skills/qualia-check-todos/SKILL.md +55 -0
- package/framework/skills/qualia-complete-milestone/SKILL.md +108 -0
- package/framework/skills/qualia-debug/SKILL.md +149 -0
- package/framework/skills/qualia-design/SKILL.md +203 -0
- package/framework/skills/qualia-discuss-phase/SKILL.md +72 -0
- package/framework/skills/qualia-execute-phase/SKILL.md +86 -0
- package/framework/skills/qualia-help/SKILL.md +67 -0
- package/framework/skills/qualia-idk/SKILL.md +352 -0
- package/framework/skills/qualia-list-phase-assumptions/SKILL.md +67 -0
- package/framework/skills/qualia-new-milestone/SKILL.md +72 -0
- package/framework/skills/qualia-new-project/SKILL.md +92 -0
- package/framework/skills/qualia-optimize/SKILL.md +417 -0
- package/framework/skills/qualia-pause-work/SKILL.md +96 -0
- package/framework/skills/qualia-plan-milestone-gaps/SKILL.md +57 -0
- package/framework/skills/qualia-plan-phase/SKILL.md +101 -0
- package/framework/skills/qualia-progress/SKILL.md +53 -0
- package/framework/skills/qualia-quick/SKILL.md +89 -0
- package/framework/skills/qualia-research-phase/SKILL.md +88 -0
- package/framework/skills/qualia-resume-work/SKILL.md +62 -0
- package/framework/skills/qualia-review/SKILL.md +263 -0
- package/framework/skills/qualia-start/SKILL.md +182 -0
- package/framework/skills/qualia-verify-work/SKILL.md +105 -0
- package/framework/skills/qualia-workflow/SKILL.md +130 -0
- package/framework/skills/rag/SKILL.md +750 -0
- package/framework/skills/responsive/SKILL.md +231 -0
- package/framework/skills/retro/SKILL.md +284 -0
- package/framework/skills/sakani-conventions/SKILL.md +136 -0
- package/framework/skills/sakani-conventions/evals/evals.json +23 -0
- package/framework/skills/sakani-conventions/references/entities.md +365 -0
- package/framework/skills/sakani-conventions/references/error-codes.md +95 -0
- package/framework/skills/seo-master/SKILL.md +490 -0
- package/framework/skills/seo-master/references/checklist.md +199 -0
- package/framework/skills/seo-master/references/structured-data.md +609 -0
- package/framework/skills/ship/SKILL.md +202 -0
- package/framework/skills/stack-researcher/SKILL.md +215 -0
- package/framework/skills/status/SKILL.md +154 -0
- package/framework/skills/status/scripts/health-check.sh +562 -0
- package/framework/skills/subscription-payments/SKILL.md +250 -0
- package/framework/skills/supabase/SKILL.md +973 -0
- package/framework/skills/supabase/references/templates.md +159 -0
- package/framework/skills/team/SKILL.md +67 -0
- package/framework/skills/test-runner/SKILL.md +202 -0
- package/framework/skills/voice-agent/SKILL.md +407 -0
- package/framework/skills/zoho-workflow/SKILL.md +51 -0
- package/framework/statusline-command.sh +117 -0
- package/package.json +24 -0
- package/profiles/fawzi.json +16 -0
- package/profiles/hasan.json +16 -0
- package/profiles/moayad.json +16 -0
- package/templates/CLAUDE-owner.md +52 -0
- package/templates/CLAUDE.md.hbs +58 -0
- package/templates/env.claude.template +12 -0
- package/templates/settings.json +141 -0
|
@@ -0,0 +1,562 @@
|
|
|
1
|
+
#!/bin/bash
|
|
2
|
+
# health-check.sh — Fleet-wide health check for all Qualia projects
|
|
3
|
+
#
|
|
4
|
+
# Usage:
|
|
5
|
+
# health-check.sh # Check all active/maintenance projects
|
|
6
|
+
# health-check.sh --quick # HTTP status only (fast)
|
|
7
|
+
# health-check.sh --project vero # Deep check on one project
|
|
8
|
+
# health-check.sh --all # Alias for default (cron-friendly)
|
|
9
|
+
#
|
|
10
|
+
# Output: Colored when interactive, plain text when piped/redirected.
|
|
11
|
+
|
|
12
|
+
set -uo pipefail
|
|
13
|
+
|
|
14
|
+
# ─── Detect terminal vs pipe (for cron compatibility) ─────────────────
|
|
15
|
+
if [[ -t 1 ]]; then
|
|
16
|
+
C_RESET="\033[0m"; C_GREEN="\033[32m"; C_YELLOW="\033[33m"
|
|
17
|
+
C_RED="\033[31m"; C_DIM="\033[2m"; C_BOLD="\033[1m"
|
|
18
|
+
else
|
|
19
|
+
C_RESET=""; C_GREEN=""; C_YELLOW=""
|
|
20
|
+
C_RED=""; C_DIM=""; C_BOLD=""
|
|
21
|
+
fi
|
|
22
|
+
|
|
23
|
+
ICON_PASS="✓"; ICON_WARN="⚠"; ICON_FAIL="✗"; ICON_SKIP="○"
|
|
24
|
+
TIMEOUT=10
|
|
25
|
+
CONTEXT_FILE="$HOME/.claude/knowledge/qualia-context.md"
|
|
26
|
+
TMPDIR="/tmp/qualia-health-$$"
|
|
27
|
+
mkdir -p "$TMPDIR"
|
|
28
|
+
trap 'rm -rf "$TMPDIR"' EXIT
|
|
29
|
+
|
|
30
|
+
# ─── Parse args ───────────────────────────────────────────────────────
|
|
31
|
+
MODE="default"
|
|
32
|
+
TARGET_PROJECT=""
|
|
33
|
+
|
|
34
|
+
while [[ $# -gt 0 ]]; do
|
|
35
|
+
case "$1" in
|
|
36
|
+
--quick) MODE="quick"; shift ;;
|
|
37
|
+
--project) MODE="deep"; TARGET_PROJECT="$2"; shift 2 ;;
|
|
38
|
+
--all) MODE="default"; shift ;;
|
|
39
|
+
--help) echo "Usage: health-check.sh [--quick|--project <name>|--all]"; exit 0 ;;
|
|
40
|
+
*) echo "Unknown option: $1"; exit 1 ;;
|
|
41
|
+
esac
|
|
42
|
+
done
|
|
43
|
+
|
|
44
|
+
# ─── Read qualia-context.md ───────────────────────────────────────────
|
|
45
|
+
if [[ ! -f "$CONTEXT_FILE" ]]; then
|
|
46
|
+
echo "ERROR: Cannot find $CONTEXT_FILE"
|
|
47
|
+
exit 1
|
|
48
|
+
fi
|
|
49
|
+
|
|
50
|
+
# Known URL overrides (projects whose URL != {name}.vercel.app)
|
|
51
|
+
declare -A URL_OVERRIDES=(
|
|
52
|
+
["qualia"]="qualia-erp.vercel.app"
|
|
53
|
+
)
|
|
54
|
+
|
|
55
|
+
# Parse project inventory from qualia-context.md
|
|
56
|
+
# Format: name|status|supabase_ref|hosting|category
|
|
57
|
+
parse_projects() {
|
|
58
|
+
local in_section="" category=""
|
|
59
|
+
|
|
60
|
+
while IFS= read -r line; do
|
|
61
|
+
# Detect section headers
|
|
62
|
+
if [[ "$line" =~ "### Websites" ]]; then category="WEBSITES"; in_section="table"; continue; fi
|
|
63
|
+
if [[ "$line" =~ "### AI Agents" ]]; then category="AI_AGENTS"; in_section="table"; continue; fi
|
|
64
|
+
if [[ "$line" =~ "### Voice" ]]; then category="VOICE"; in_section="table"; continue; fi
|
|
65
|
+
if [[ "$line" =~ "### ERP" ]]; then
|
|
66
|
+
# ERP is a special single-project section
|
|
67
|
+
echo "qualia|ACTIVE||vercel|ERP"
|
|
68
|
+
in_section=""
|
|
69
|
+
continue
|
|
70
|
+
fi
|
|
71
|
+
if [[ "$line" =~ ^"##" ]] && [[ ! "$line" =~ "###" ]]; then in_section=""; continue; fi
|
|
72
|
+
|
|
73
|
+
# Parse table rows
|
|
74
|
+
if [[ "$in_section" == "table" ]] && [[ "$line" =~ ^\| ]] && [[ ! "$line" =~ ^\|[-] ]] && [[ ! "$line" =~ "Project" ]]; then
|
|
75
|
+
# Extract fields from markdown table row
|
|
76
|
+
local name status supabase_ref hosting
|
|
77
|
+
|
|
78
|
+
if [[ "$category" == "WEBSITES" ]]; then
|
|
79
|
+
name=$(echo "$line" | awk -F'|' '{gsub(/^[ \t]+|[ \t]+$/, "", $2); print $2}')
|
|
80
|
+
status=$(echo "$line" | awk -F'|' '{gsub(/^[ \t]+|[ \t]+$/, "", $3); print $3}')
|
|
81
|
+
supabase_ref=$(echo "$line" | awk -F'|' '{gsub(/^[ \t]+|[ \t]+$/, "", $4); gsub(/`/, "", $4); print $4}')
|
|
82
|
+
hosting=$(echo "$line" | awk -F'|' '{gsub(/^[ \t]+|[ \t]+$/, "", $5); print $5}')
|
|
83
|
+
elif [[ "$category" == "AI_AGENTS" ]]; then
|
|
84
|
+
name=$(echo "$line" | awk -F'|' '{gsub(/^[ \t]+|[ \t]+$/, "", $2); print $2}')
|
|
85
|
+
status=$(echo "$line" | awk -F'|' '{gsub(/^[ \t]+|[ \t]+$/, "", $3); print $3}')
|
|
86
|
+
supabase_ref=$(echo "$line" | awk -F'|' '{gsub(/^[ \t]+|[ \t]+$/, "", $4); gsub(/`/, "", $4); print $4}')
|
|
87
|
+
hosting="vercel"
|
|
88
|
+
elif [[ "$category" == "VOICE" ]]; then
|
|
89
|
+
name=$(echo "$line" | awk -F'|' '{gsub(/^[ \t]+|[ \t]+$/, "", $2); print $2}')
|
|
90
|
+
status=$(echo "$line" | awk -F'|' '{gsub(/^[ \t]+|[ \t]+$/, "", $3); print $3}')
|
|
91
|
+
hosting=$(echo "$line" | awk -F'|' '{gsub(/^[ \t]+|[ \t]+$/, "", $5); print $5}')
|
|
92
|
+
supabase_ref=$(echo "$line" | awk -F'|' '{gsub(/^[ \t]+|[ \t]+$/, "", $6); gsub(/`/, "", $6); print $6}')
|
|
93
|
+
fi
|
|
94
|
+
|
|
95
|
+
# Clean up
|
|
96
|
+
[[ "$supabase_ref" == "—" || "$supabase_ref" == "planned" ]] && supabase_ref=""
|
|
97
|
+
[[ "$hosting" == "—" ]] && hosting=""
|
|
98
|
+
|
|
99
|
+
echo "${name}|${status}|${supabase_ref}|${hosting}|${category}"
|
|
100
|
+
fi
|
|
101
|
+
done < "$CONTEXT_FILE"
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
# ─── URL resolution ──────────────────────────────────────────────────
|
|
105
|
+
resolve_url() {
|
|
106
|
+
local name="$1" hosting="$2"
|
|
107
|
+
# Check overrides first
|
|
108
|
+
if [[ -n "${URL_OVERRIDES[$name]:-}" ]]; then
|
|
109
|
+
echo "https://${URL_OVERRIDES[$name]}"
|
|
110
|
+
return
|
|
111
|
+
fi
|
|
112
|
+
# Cloudflare projects don't follow vercel.app convention
|
|
113
|
+
if [[ "$hosting" =~ [Cc]loudflare ]]; then
|
|
114
|
+
# For Cloudflare projects, we'd need to know the domain
|
|
115
|
+
# armenius is the only one — check its worker URL
|
|
116
|
+
echo "https://${name}.workers.dev"
|
|
117
|
+
return
|
|
118
|
+
fi
|
|
119
|
+
echo "https://${name}.vercel.app"
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
# ─── Individual check functions ───────────────────────────────────────
|
|
123
|
+
|
|
124
|
+
check_http() {
|
|
125
|
+
local url="$1"
|
|
126
|
+
local code time_total
|
|
127
|
+
code=$(curl -sL -o /dev/null -w "%{http_code}" --max-time "$TIMEOUT" "$url" 2>/dev/null || echo "000")
|
|
128
|
+
time_total=$(curl -sL -o /dev/null -w "%{time_total}" --max-time "$TIMEOUT" "$url" 2>/dev/null || echo "0")
|
|
129
|
+
local ms
|
|
130
|
+
ms=$(echo "$time_total * 1000" | bc 2>/dev/null | cut -d. -f1)
|
|
131
|
+
ms=${ms:-0}
|
|
132
|
+
echo "${code}|${ms}"
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
check_ssl() {
|
|
136
|
+
local domain="$1"
|
|
137
|
+
local cert_output expiry_date days_left
|
|
138
|
+
|
|
139
|
+
cert_output=$(echo | openssl s_client -servername "$domain" -connect "$domain:443" 2>/dev/null)
|
|
140
|
+
if [[ -z "$cert_output" ]]; then
|
|
141
|
+
echo "FAIL|0|Connection failed"
|
|
142
|
+
return
|
|
143
|
+
fi
|
|
144
|
+
|
|
145
|
+
expiry_date=$(echo "$cert_output" | openssl x509 -noout -enddate 2>/dev/null | cut -d= -f2)
|
|
146
|
+
if [[ -z "$expiry_date" ]]; then
|
|
147
|
+
echo "FAIL|0|Cannot parse cert"
|
|
148
|
+
return
|
|
149
|
+
fi
|
|
150
|
+
|
|
151
|
+
local expiry_epoch now_epoch
|
|
152
|
+
expiry_epoch=$(date -d "$expiry_date" +%s 2>/dev/null)
|
|
153
|
+
now_epoch=$(date +%s)
|
|
154
|
+
days_left=$(( (expiry_epoch - now_epoch) / 86400 ))
|
|
155
|
+
|
|
156
|
+
if ((days_left < 7)); then
|
|
157
|
+
echo "FAIL|${days_left}|Expires in ${days_left}d"
|
|
158
|
+
elif ((days_left < 30)); then
|
|
159
|
+
echo "WARN|${days_left}|Expires in ${days_left}d"
|
|
160
|
+
else
|
|
161
|
+
echo "PASS|${days_left}|${days_left}d"
|
|
162
|
+
fi
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
check_supabase() {
|
|
166
|
+
local ref="$1"
|
|
167
|
+
if [[ -z "$ref" ]]; then
|
|
168
|
+
echo "SKIP|0|No ref"
|
|
169
|
+
return
|
|
170
|
+
fi
|
|
171
|
+
|
|
172
|
+
local start_ns end_ns status_code
|
|
173
|
+
start_ns=$(date +%s%N)
|
|
174
|
+
status_code=$(curl -sL -o /dev/null -w "%{http_code}" --max-time "$TIMEOUT" \
|
|
175
|
+
"https://${ref}.supabase.co/rest/v1/" \
|
|
176
|
+
-H "apikey: placeholder" 2>/dev/null || echo "000")
|
|
177
|
+
end_ns=$(date +%s%N)
|
|
178
|
+
local db_ms=$(( (end_ns - start_ns) / 1000000 ))
|
|
179
|
+
|
|
180
|
+
if [[ "$status_code" =~ ^(200|401|403)$ ]]; then
|
|
181
|
+
if ((db_ms > 500)); then
|
|
182
|
+
echo "WARN|${db_ms}|Slow (${db_ms}ms)"
|
|
183
|
+
else
|
|
184
|
+
echo "PASS|${db_ms}|healthy"
|
|
185
|
+
fi
|
|
186
|
+
elif [[ "$status_code" == "000" ]]; then
|
|
187
|
+
echo "FAIL|0|Unreachable"
|
|
188
|
+
else
|
|
189
|
+
echo "WARN|${db_ms}|HTTP ${status_code}"
|
|
190
|
+
fi
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
# ─── Check a single project (runs as background process) ─────────────
|
|
194
|
+
check_project() {
|
|
195
|
+
local name="$1" status="$2" supabase_ref="$3" hosting="$4" category="$5"
|
|
196
|
+
local url result_file
|
|
197
|
+
|
|
198
|
+
url=$(resolve_url "$name" "$hosting")
|
|
199
|
+
result_file="$TMPDIR/${name}.result"
|
|
200
|
+
local domain
|
|
201
|
+
domain=$(echo "$url" | sed -E 's|https?://([^/:]+).*|\1|')
|
|
202
|
+
|
|
203
|
+
# HTTP check (always)
|
|
204
|
+
local http_result http_code http_ms
|
|
205
|
+
http_result=$(check_http "$url")
|
|
206
|
+
http_code=$(echo "$http_result" | cut -d'|' -f1)
|
|
207
|
+
http_ms=$(echo "$http_result" | cut -d'|' -f2)
|
|
208
|
+
|
|
209
|
+
# Determine overall HTTP status
|
|
210
|
+
local http_status http_detail
|
|
211
|
+
if [[ "$http_code" == "200" ]]; then
|
|
212
|
+
if ((http_ms < 1000)); then
|
|
213
|
+
http_status="PASS"
|
|
214
|
+
elif ((http_ms < 3000)); then
|
|
215
|
+
http_status="WARN"
|
|
216
|
+
else
|
|
217
|
+
http_status="FAIL"
|
|
218
|
+
fi
|
|
219
|
+
http_detail="${http_code} OK (${http_ms}ms)"
|
|
220
|
+
elif [[ "$http_code" =~ ^3[0-9]{2}$ ]]; then
|
|
221
|
+
http_status="WARN"
|
|
222
|
+
http_detail="${http_code} Redirect (${http_ms}ms)"
|
|
223
|
+
elif [[ "$http_code" == "000" ]]; then
|
|
224
|
+
http_status="FAIL"
|
|
225
|
+
http_detail="Timeout/DNS error"
|
|
226
|
+
else
|
|
227
|
+
http_status="FAIL"
|
|
228
|
+
http_detail="HTTP ${http_code}"
|
|
229
|
+
fi
|
|
230
|
+
|
|
231
|
+
local ssl_summary="" db_summary=""
|
|
232
|
+
|
|
233
|
+
if [[ "$MODE" != "quick" ]]; then
|
|
234
|
+
# SSL check
|
|
235
|
+
local ssl_result ssl_status ssl_days
|
|
236
|
+
ssl_result=$(check_ssl "$domain")
|
|
237
|
+
ssl_status=$(echo "$ssl_result" | cut -d'|' -f1)
|
|
238
|
+
ssl_days=$(echo "$ssl_result" | cut -d'|' -f2)
|
|
239
|
+
|
|
240
|
+
if [[ "$ssl_status" == "PASS" ]]; then
|
|
241
|
+
ssl_summary="SSL: ${ssl_days}d"
|
|
242
|
+
elif [[ "$ssl_status" == "WARN" ]]; then
|
|
243
|
+
ssl_summary="SSL: ${ssl_days}d ⚠"
|
|
244
|
+
elif [[ "$ssl_status" == "FAIL" ]]; then
|
|
245
|
+
ssl_summary="SSL: FAIL"
|
|
246
|
+
fi
|
|
247
|
+
|
|
248
|
+
# Supabase check
|
|
249
|
+
local db_result db_status db_detail
|
|
250
|
+
db_result=$(check_supabase "$supabase_ref")
|
|
251
|
+
db_status=$(echo "$db_result" | cut -d'|' -f1)
|
|
252
|
+
db_detail=$(echo "$db_result" | cut -d'|' -f3)
|
|
253
|
+
|
|
254
|
+
if [[ "$db_status" == "PASS" ]]; then
|
|
255
|
+
db_summary="DB: ${db_detail}"
|
|
256
|
+
elif [[ "$db_status" == "WARN" ]]; then
|
|
257
|
+
db_summary="DB: ${db_detail} ⚠"
|
|
258
|
+
elif [[ "$db_status" == "FAIL" ]]; then
|
|
259
|
+
db_summary="DB: ${db_detail}"
|
|
260
|
+
elif [[ "$db_status" == "SKIP" ]]; then
|
|
261
|
+
db_summary=""
|
|
262
|
+
fi
|
|
263
|
+
fi
|
|
264
|
+
|
|
265
|
+
# Determine overall project status (worst of all checks)
|
|
266
|
+
local overall="PASS"
|
|
267
|
+
if [[ "$http_status" == "FAIL" ]] || [[ "${ssl_status:-PASS}" == "FAIL" ]] || [[ "${db_status:-PASS}" == "FAIL" ]]; then
|
|
268
|
+
overall="FAIL"
|
|
269
|
+
elif [[ "$http_status" == "WARN" ]] || [[ "${ssl_status:-PASS}" == "WARN" ]] || [[ "${db_status:-PASS}" == "WARN" ]]; then
|
|
270
|
+
overall="WARN"
|
|
271
|
+
fi
|
|
272
|
+
|
|
273
|
+
# Build extras string
|
|
274
|
+
local extras=""
|
|
275
|
+
[[ -n "$ssl_summary" ]] && extras=" ${ssl_summary}"
|
|
276
|
+
[[ -n "$db_summary" ]] && extras="${extras} ${db_summary}"
|
|
277
|
+
|
|
278
|
+
# Write result to file (category|overall|name|http_detail|extras|issues)
|
|
279
|
+
local issue=""
|
|
280
|
+
if [[ "$overall" == "FAIL" ]]; then
|
|
281
|
+
if [[ "$http_code" == "000" ]]; then
|
|
282
|
+
issue="FAIL|${name}|Unreachable (timeout/DNS)|Check: curl -sL ${url} ; Check Vercel dashboard"
|
|
283
|
+
elif [[ "$http_code" =~ ^5 ]]; then
|
|
284
|
+
issue="FAIL|${name}|HTTP ${http_code}|Check: vercel logs ${url} ; Check Supabase: is project paused?"
|
|
285
|
+
elif [[ "${db_status:-PASS}" == "FAIL" ]]; then
|
|
286
|
+
issue="FAIL|${name}|Supabase unreachable|Check Supabase dashboard for ref ${supabase_ref}"
|
|
287
|
+
elif [[ "${ssl_status:-PASS}" == "FAIL" ]]; then
|
|
288
|
+
issue="FAIL|${name}|SSL certificate issue|Check: vercel certs ls ; vercel certs issue ${domain}"
|
|
289
|
+
fi
|
|
290
|
+
elif [[ "$overall" == "WARN" ]]; then
|
|
291
|
+
if ((http_ms > 1000)) 2>/dev/null; then
|
|
292
|
+
issue="WARN|${name}|Response time ${http_ms}ms (> 1s)|Check: vercel logs ${url}"
|
|
293
|
+
fi
|
|
294
|
+
if [[ "${ssl_status:-PASS}" == "WARN" ]]; then
|
|
295
|
+
local ssl_msg="SSL expires in ${ssl_days} days"
|
|
296
|
+
if [[ "$hosting" =~ [Cc]loudflare ]]; then
|
|
297
|
+
ssl_msg="${ssl_msg} (Cloudflare — check auto-renewal)"
|
|
298
|
+
else
|
|
299
|
+
ssl_msg="${ssl_msg} (Vercel — should auto-renew)"
|
|
300
|
+
fi
|
|
301
|
+
issue="${issue:+${issue};}WARN|${name}|${ssl_msg}|Check Vercel/Cloudflare dashboard"
|
|
302
|
+
fi
|
|
303
|
+
if [[ "${db_status:-PASS}" == "WARN" ]]; then
|
|
304
|
+
issue="${issue:+${issue};}WARN|${name}|Supabase: ${db_detail}|Check: supabase inspect db outliers --project-ref ${supabase_ref}"
|
|
305
|
+
fi
|
|
306
|
+
fi
|
|
307
|
+
|
|
308
|
+
echo "${category}|${overall}|${name}|${http_detail}|${extras}|${issue}" > "$result_file"
|
|
309
|
+
}
|
|
310
|
+
|
|
311
|
+
# ─── Deep check for single project ───────────────────────────────────
|
|
312
|
+
deep_check() {
|
|
313
|
+
local name="$1" supabase_ref="$2" hosting="$3"
|
|
314
|
+
local url
|
|
315
|
+
url=$(resolve_url "$name" "$hosting")
|
|
316
|
+
local domain
|
|
317
|
+
domain=$(echo "$url" | sed -E 's|https?://([^/:]+).*|\1|')
|
|
318
|
+
|
|
319
|
+
echo ""
|
|
320
|
+
echo -e "${C_BOLD}══════════════════════════════════════════════════════${C_RESET}"
|
|
321
|
+
echo -e " ${C_BOLD}DEEP CHECK${C_RESET} — ${name}"
|
|
322
|
+
echo -e " URL: ${url}"
|
|
323
|
+
echo -e " Timestamp: $(date -u '+%Y-%m-%d %H:%M:%S UTC')"
|
|
324
|
+
[[ -n "$supabase_ref" ]] && echo -e " Supabase: ${supabase_ref}"
|
|
325
|
+
echo -e "${C_BOLD}══════════════════════════════════════════════════════${C_RESET}"
|
|
326
|
+
echo ""
|
|
327
|
+
|
|
328
|
+
# HTTP check
|
|
329
|
+
local http_result http_code http_ms
|
|
330
|
+
http_result=$(check_http "$url")
|
|
331
|
+
http_code=$(echo "$http_result" | cut -d'|' -f1)
|
|
332
|
+
http_ms=$(echo "$http_result" | cut -d'|' -f2)
|
|
333
|
+
if [[ "$http_code" == "200" ]] && ((http_ms < 1000)); then
|
|
334
|
+
echo -e " ${C_GREEN}${ICON_PASS} PASS${C_RESET} HTTP Status ${http_code} OK (${http_ms}ms)"
|
|
335
|
+
elif [[ "$http_code" == "200" ]]; then
|
|
336
|
+
echo -e " ${C_YELLOW}${ICON_WARN} WARN${C_RESET} HTTP Status ${http_code} OK but slow (${http_ms}ms)"
|
|
337
|
+
else
|
|
338
|
+
echo -e " ${C_RED}${ICON_FAIL} FAIL${C_RESET} HTTP Status ${http_code} (${http_ms}ms)"
|
|
339
|
+
fi
|
|
340
|
+
|
|
341
|
+
# Also check /api/health
|
|
342
|
+
local health_code
|
|
343
|
+
health_code=$(curl -sL -o /dev/null -w "%{http_code}" --max-time "$TIMEOUT" "$url/api/health" 2>/dev/null || echo "000")
|
|
344
|
+
if [[ "$health_code" == "200" ]]; then
|
|
345
|
+
echo -e " ${C_GREEN}${ICON_PASS} PASS${C_RESET} /api/health ${health_code} OK"
|
|
346
|
+
elif [[ "$health_code" == "404" ]]; then
|
|
347
|
+
echo -e " ${C_DIM}${ICON_SKIP} SKIP${C_RESET} /api/health Not found (no health endpoint)"
|
|
348
|
+
else
|
|
349
|
+
echo -e " ${C_YELLOW}${ICON_WARN} WARN${C_RESET} /api/health ${health_code}"
|
|
350
|
+
fi
|
|
351
|
+
|
|
352
|
+
# SSL
|
|
353
|
+
local ssl_result ssl_status ssl_days ssl_detail
|
|
354
|
+
ssl_result=$(check_ssl "$domain")
|
|
355
|
+
ssl_status=$(echo "$ssl_result" | cut -d'|' -f1)
|
|
356
|
+
ssl_days=$(echo "$ssl_result" | cut -d'|' -f2)
|
|
357
|
+
ssl_detail=$(echo "$ssl_result" | cut -d'|' -f3)
|
|
358
|
+
if [[ "$ssl_status" == "PASS" ]]; then
|
|
359
|
+
echo -e " ${C_GREEN}${ICON_PASS} PASS${C_RESET} SSL Certificate Valid, expires in ${ssl_days} days"
|
|
360
|
+
elif [[ "$ssl_status" == "WARN" ]]; then
|
|
361
|
+
echo -e " ${C_YELLOW}${ICON_WARN} WARN${C_RESET} SSL Certificate ${ssl_detail}"
|
|
362
|
+
else
|
|
363
|
+
echo -e " ${C_RED}${ICON_FAIL} FAIL${C_RESET} SSL Certificate ${ssl_detail}"
|
|
364
|
+
fi
|
|
365
|
+
|
|
366
|
+
# Supabase health
|
|
367
|
+
if [[ -n "$supabase_ref" ]]; then
|
|
368
|
+
local db_result db_status db_ms db_detail
|
|
369
|
+
db_result=$(check_supabase "$supabase_ref")
|
|
370
|
+
db_status=$(echo "$db_result" | cut -d'|' -f1)
|
|
371
|
+
db_ms=$(echo "$db_result" | cut -d'|' -f2)
|
|
372
|
+
db_detail=$(echo "$db_result" | cut -d'|' -f3)
|
|
373
|
+
if [[ "$db_status" == "PASS" ]]; then
|
|
374
|
+
echo -e " ${C_GREEN}${ICON_PASS} PASS${C_RESET} Supabase REST Reachable (${db_ms}ms)"
|
|
375
|
+
elif [[ "$db_status" == "WARN" ]]; then
|
|
376
|
+
echo -e " ${C_YELLOW}${ICON_WARN} WARN${C_RESET} Supabase REST ${db_detail}"
|
|
377
|
+
else
|
|
378
|
+
echo -e " ${C_RED}${ICON_FAIL} FAIL${C_RESET} Supabase REST ${db_detail}"
|
|
379
|
+
fi
|
|
380
|
+
|
|
381
|
+
# Supabase auth health
|
|
382
|
+
local auth_response auth_status
|
|
383
|
+
auth_response=$(curl -s --max-time "$TIMEOUT" "https://${supabase_ref}.supabase.co/auth/v1/health" 2>/dev/null || echo "")
|
|
384
|
+
auth_status=$(echo "$auth_response" | jq -r '.status' 2>/dev/null || echo "")
|
|
385
|
+
if [[ "$auth_status" == "ok" ]]; then
|
|
386
|
+
echo -e " ${C_GREEN}${ICON_PASS} PASS${C_RESET} Supabase Auth Healthy"
|
|
387
|
+
elif [[ -n "$auth_response" ]]; then
|
|
388
|
+
echo -e " ${C_YELLOW}${ICON_WARN} WARN${C_RESET} Supabase Auth Status: ${auth_status:-unknown}"
|
|
389
|
+
else
|
|
390
|
+
echo -e " ${C_RED}${ICON_FAIL} FAIL${C_RESET} Supabase Auth Unreachable"
|
|
391
|
+
fi
|
|
392
|
+
|
|
393
|
+
# Edge functions (if supabase CLI is available and project is linked)
|
|
394
|
+
if command -v supabase &>/dev/null; then
|
|
395
|
+
local func_count
|
|
396
|
+
func_count=$(supabase functions list --project-ref "$supabase_ref" 2>/dev/null | grep -c "^│" || echo "0")
|
|
397
|
+
if ((func_count > 0)); then
|
|
398
|
+
echo -e " ${C_GREEN}${ICON_PASS} PASS${C_RESET} Edge Functions ${func_count} deployed"
|
|
399
|
+
else
|
|
400
|
+
echo -e " ${C_DIM}${ICON_SKIP} SKIP${C_RESET} Edge Functions None deployed"
|
|
401
|
+
fi
|
|
402
|
+
fi
|
|
403
|
+
else
|
|
404
|
+
echo -e " ${C_DIM}${ICON_SKIP} SKIP${C_RESET} Supabase No Supabase ref"
|
|
405
|
+
fi
|
|
406
|
+
|
|
407
|
+
# Vercel deployment info (if vercel CLI available)
|
|
408
|
+
if command -v vercel &>/dev/null; then
|
|
409
|
+
# Try to get recent deployments
|
|
410
|
+
local deploy_info
|
|
411
|
+
deploy_info=$(vercel ls "$url" --limit 1 2>/dev/null | tail -1 || echo "")
|
|
412
|
+
if [[ -n "$deploy_info" ]] && [[ ! "$deploy_info" =~ "Error" ]]; then
|
|
413
|
+
echo -e " ${C_GREEN}${ICON_PASS} PASS${C_RESET} Last Deploy ${deploy_info}"
|
|
414
|
+
else
|
|
415
|
+
echo -e " ${C_DIM}${ICON_SKIP} SKIP${C_RESET} Last Deploy Could not fetch (not linked?)"
|
|
416
|
+
fi
|
|
417
|
+
fi
|
|
418
|
+
|
|
419
|
+
echo ""
|
|
420
|
+
echo -e "${C_BOLD}──────────────────────────────────────────────────────${C_RESET}"
|
|
421
|
+
}
|
|
422
|
+
|
|
423
|
+
# ═══════════════════════════════════════════════════════════════════════
|
|
424
|
+
# Main
|
|
425
|
+
# ═══════════════════════════════════════════════════════════════════════
|
|
426
|
+
|
|
427
|
+
START_TIME=$(date +%s%N)
|
|
428
|
+
TIMESTAMP=$(date -u '+%Y-%m-%d %H:%M UTC')
|
|
429
|
+
|
|
430
|
+
# Parse all projects from context
|
|
431
|
+
mapfile -t ALL_PROJECTS < <(parse_projects)
|
|
432
|
+
|
|
433
|
+
# Filter based on mode
|
|
434
|
+
if [[ "$MODE" == "deep" ]]; then
|
|
435
|
+
# Find the target project
|
|
436
|
+
found=false
|
|
437
|
+
for proj in "${ALL_PROJECTS[@]}"; do
|
|
438
|
+
IFS='|' read -r name status supabase_ref hosting category <<< "$proj"
|
|
439
|
+
if [[ "${name,,}" == "${TARGET_PROJECT,,}" ]]; then
|
|
440
|
+
deep_check "$name" "$supabase_ref" "$hosting"
|
|
441
|
+
found=true
|
|
442
|
+
break
|
|
443
|
+
fi
|
|
444
|
+
done
|
|
445
|
+
if ! $found; then
|
|
446
|
+
echo "ERROR: Project '${TARGET_PROJECT}' not found in qualia-context.md"
|
|
447
|
+
echo "Available projects:"
|
|
448
|
+
for proj in "${ALL_PROJECTS[@]}"; do
|
|
449
|
+
IFS='|' read -r name status _ _ _ <<< "$proj"
|
|
450
|
+
echo " ${name} (${status})"
|
|
451
|
+
done
|
|
452
|
+
exit 1
|
|
453
|
+
fi
|
|
454
|
+
exit 0
|
|
455
|
+
fi
|
|
456
|
+
|
|
457
|
+
# Default/quick mode: check all active/maintenance/live projects
|
|
458
|
+
CHECKABLE=()
|
|
459
|
+
for proj in "${ALL_PROJECTS[@]}"; do
|
|
460
|
+
IFS='|' read -r name status supabase_ref hosting category <<< "$proj"
|
|
461
|
+
# Only check ACTIVE, MAINTENANCE, LIVE, BUILT projects
|
|
462
|
+
case "${status^^}" in
|
|
463
|
+
ACTIVE|MAINTENANCE|LIVE|BUILT) CHECKABLE+=("$proj") ;;
|
|
464
|
+
esac
|
|
465
|
+
done
|
|
466
|
+
|
|
467
|
+
echo ""
|
|
468
|
+
echo -e "${C_BOLD}════════════════════════════════════════════════════${C_RESET}"
|
|
469
|
+
echo -e " ${C_BOLD}QUALIA FLEET HEALTH${C_RESET} — ${TIMESTAMP}"
|
|
470
|
+
echo -e " Checking: ${#CHECKABLE[@]} active/maintenance projects"
|
|
471
|
+
[[ "$MODE" == "quick" ]] && echo -e " Mode: ${C_YELLOW}QUICK${C_RESET} (HTTP only)"
|
|
472
|
+
echo -e "${C_BOLD}════════════════════════════════════════════════════${C_RESET}"
|
|
473
|
+
|
|
474
|
+
# Launch all checks in parallel
|
|
475
|
+
PIDS=()
|
|
476
|
+
for proj in "${CHECKABLE[@]}"; do
|
|
477
|
+
IFS='|' read -r name status supabase_ref hosting category <<< "$proj"
|
|
478
|
+
check_project "$name" "$status" "$supabase_ref" "$hosting" "$category" &
|
|
479
|
+
PIDS+=($!)
|
|
480
|
+
done
|
|
481
|
+
|
|
482
|
+
# Wait for all background checks to complete
|
|
483
|
+
for pid in "${PIDS[@]}"; do
|
|
484
|
+
wait "$pid" 2>/dev/null || true
|
|
485
|
+
done
|
|
486
|
+
|
|
487
|
+
# Collect and display results grouped by category
|
|
488
|
+
PASS_COUNT=0
|
|
489
|
+
WARN_COUNT=0
|
|
490
|
+
FAIL_COUNT=0
|
|
491
|
+
CURRENT_CAT=""
|
|
492
|
+
|
|
493
|
+
# Sort results into a single file (avoids subshell from pipe)
|
|
494
|
+
for result_file in "$TMPDIR"/*.result; do
|
|
495
|
+
[[ -f "$result_file" ]] || continue
|
|
496
|
+
cat "$result_file"
|
|
497
|
+
done | sort -t'|' -k1,1 -k3,3 > "$TMPDIR/sorted-results.txt"
|
|
498
|
+
|
|
499
|
+
# Display results (no pipe — runs in current shell so counters persist)
|
|
500
|
+
while IFS='|' read -r category overall name http_detail extras issues; do
|
|
501
|
+
# Print category header if changed
|
|
502
|
+
if [[ "$category" != "$CURRENT_CAT" ]]; then
|
|
503
|
+
CURRENT_CAT="$category"
|
|
504
|
+
display_cat="${category//_/ }"
|
|
505
|
+
echo ""
|
|
506
|
+
echo -e " ${C_DIM}${display_cat}${C_RESET}"
|
|
507
|
+
fi
|
|
508
|
+
|
|
509
|
+
# Print project line
|
|
510
|
+
padded_name=$(printf "%-18s" "$name")
|
|
511
|
+
color=""; icon=""
|
|
512
|
+
case "$overall" in
|
|
513
|
+
PASS) icon="$ICON_PASS"; color="$C_GREEN"; ((PASS_COUNT++)) || true ;;
|
|
514
|
+
WARN) icon="$ICON_WARN"; color="$C_YELLOW"; ((WARN_COUNT++)) || true ;;
|
|
515
|
+
FAIL) icon="$ICON_FAIL"; color="$C_RED"; ((FAIL_COUNT++)) || true ;;
|
|
516
|
+
esac
|
|
517
|
+
|
|
518
|
+
echo -e " ${color}${icon} ${overall}${C_RESET} ${color}${padded_name}${C_RESET} ${http_detail}${extras}"
|
|
519
|
+
|
|
520
|
+
# Collect issues
|
|
521
|
+
if [[ -n "$issues" ]]; then
|
|
522
|
+
echo "$issues" >> "$TMPDIR/all-issues.txt"
|
|
523
|
+
fi
|
|
524
|
+
done < "$TMPDIR/sorted-results.txt"
|
|
525
|
+
|
|
526
|
+
END_TIME=$(date +%s%N)
|
|
527
|
+
ELAPSED_MS=$(( (END_TIME - START_TIME) / 1000000 ))
|
|
528
|
+
ELAPSED_S=$(echo "scale=1; $ELAPSED_MS / 1000" | bc 2>/dev/null || echo "$((ELAPSED_MS / 1000))")
|
|
529
|
+
|
|
530
|
+
echo ""
|
|
531
|
+
echo -e "${C_BOLD}────────────────────────────────────────────────────${C_RESET}"
|
|
532
|
+
echo -e " RESULT: ${C_GREEN}${PASS_COUNT} PASS${C_RESET} · ${C_YELLOW}${WARN_COUNT} WARN${C_RESET} · ${C_RED}${FAIL_COUNT} FAIL${C_RESET}"
|
|
533
|
+
echo -e " Total check time: ${ELAPSED_S}s"
|
|
534
|
+
echo -e "${C_BOLD}────────────────────────────────────────────────────${C_RESET}"
|
|
535
|
+
|
|
536
|
+
# Print issues if any
|
|
537
|
+
if [[ -f "$TMPDIR/all-issues.txt" ]]; then
|
|
538
|
+
echo ""
|
|
539
|
+
echo -e " ${C_BOLD}ISSUES FOUND:${C_RESET}"
|
|
540
|
+
echo ""
|
|
541
|
+
|
|
542
|
+
while IFS=';' read -ra issue_list; do
|
|
543
|
+
for issue_entry in "${issue_list[@]}"; do
|
|
544
|
+
IFS='|' read -r severity iname description action <<< "$issue_entry"
|
|
545
|
+
local icolor
|
|
546
|
+
if [[ "$severity" == "FAIL" ]]; then
|
|
547
|
+
icolor="$C_RED"
|
|
548
|
+
else
|
|
549
|
+
icolor="$C_YELLOW"
|
|
550
|
+
fi
|
|
551
|
+
echo -e " ${icolor}${severity}${C_RESET} ${iname} — ${description}"
|
|
552
|
+
echo -e " ${C_DIM}→ ${action}${C_RESET}"
|
|
553
|
+
echo ""
|
|
554
|
+
done
|
|
555
|
+
done < "$TMPDIR/all-issues.txt"
|
|
556
|
+
fi
|
|
557
|
+
|
|
558
|
+
# Exit code
|
|
559
|
+
if ((FAIL_COUNT > 0)); then
|
|
560
|
+
exit 1
|
|
561
|
+
fi
|
|
562
|
+
exit 0
|