qualia-framework 2.6.0 → 3.2.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.
Files changed (321) hide show
  1. package/CLAUDE.md +64 -0
  2. package/README.md +103 -30
  3. package/agents/builder.md +110 -0
  4. package/agents/planner.md +134 -0
  5. package/agents/qa-browser.md +186 -0
  6. package/agents/verifier.md +221 -0
  7. package/bin/cli.js +336 -531
  8. package/bin/install.js +570 -0
  9. package/bin/qualia-ui.js +299 -0
  10. package/bin/state.js +630 -0
  11. package/bin/statusline.js +252 -0
  12. package/guide.md +63 -0
  13. package/hooks/auto-update.js +139 -0
  14. package/hooks/branch-guard.js +47 -0
  15. package/hooks/migration-guard.js +60 -0
  16. package/hooks/pre-compact.js +32 -0
  17. package/hooks/pre-deploy-gate.js +110 -0
  18. package/hooks/pre-push.js +33 -0
  19. package/hooks/session-start.js +170 -0
  20. package/package.json +29 -20
  21. package/rules/design-reference.md +179 -0
  22. package/rules/frontend.md +126 -0
  23. package/skills/qualia/SKILL.md +87 -0
  24. package/skills/qualia-build/SKILL.md +97 -0
  25. package/skills/qualia-debug/SKILL.md +87 -0
  26. package/skills/qualia-design/SKILL.md +93 -0
  27. package/skills/qualia-handoff/SKILL.md +66 -0
  28. package/skills/qualia-idk/SKILL.md +8 -0
  29. package/skills/qualia-learn/SKILL.md +88 -0
  30. package/skills/qualia-new/SKILL.md +323 -0
  31. package/{framework/skills → skills}/qualia-optimize/SKILL.md +1 -1
  32. package/skills/qualia-pause/SKILL.md +63 -0
  33. package/skills/qualia-plan/SKILL.md +101 -0
  34. package/skills/qualia-polish/SKILL.md +157 -0
  35. package/skills/qualia-quick/SKILL.md +37 -0
  36. package/skills/qualia-report/SKILL.md +105 -0
  37. package/skills/qualia-resume/SKILL.md +49 -0
  38. package/skills/qualia-review/SKILL.md +76 -0
  39. package/skills/qualia-ship/SKILL.md +90 -0
  40. package/skills/qualia-skill-new/SKILL.md +167 -0
  41. package/skills/qualia-task/SKILL.md +91 -0
  42. package/skills/qualia-verify/SKILL.md +113 -0
  43. package/templates/DESIGN.md +137 -0
  44. package/templates/plan.md +28 -0
  45. package/templates/project.md +22 -0
  46. package/templates/state.md +27 -0
  47. package/templates/tracking.json +20 -0
  48. package/tests/bin.test.sh +673 -0
  49. package/tests/hooks.test.sh +315 -0
  50. package/tests/state.test.sh +535 -0
  51. package/tests/statusline.test.sh +243 -0
  52. package/bin/collect-metrics.sh +0 -62
  53. package/framework/.claudeignore +0 -51
  54. package/framework/CLAUDE.md +0 -51
  55. package/framework/MCP_SETUP.md +0 -229
  56. package/framework/agents/architecture-strategist.md +0 -53
  57. package/framework/agents/backend-agent.md +0 -150
  58. package/framework/agents/code-simplicity-reviewer.md +0 -86
  59. package/framework/agents/frontend-agent.md +0 -111
  60. package/framework/agents/kieran-typescript-reviewer.md +0 -96
  61. package/framework/agents/performance-oracle.md +0 -111
  62. package/framework/agents/qualia-codebase-mapper.md +0 -761
  63. package/framework/agents/qualia-debugger.md +0 -1204
  64. package/framework/agents/qualia-executor.md +0 -882
  65. package/framework/agents/qualia-integration-checker.md +0 -424
  66. package/framework/agents/qualia-phase-researcher.md +0 -457
  67. package/framework/agents/qualia-plan-checker.md +0 -700
  68. package/framework/agents/qualia-planner.md +0 -1245
  69. package/framework/agents/qualia-project-researcher.md +0 -603
  70. package/framework/agents/qualia-research-synthesizer.md +0 -200
  71. package/framework/agents/qualia-roadmapper.md +0 -606
  72. package/framework/agents/qualia-verifier.md +0 -686
  73. package/framework/agents/red-team-qa.md +0 -130
  74. package/framework/agents/security-auditor.md +0 -72
  75. package/framework/agents/team-orchestrator.md +0 -229
  76. package/framework/agents/teams/framework-audit-team.md +0 -66
  77. package/framework/agents/teams/full-stack-team.md +0 -48
  78. package/framework/agents/teams/optimize-team.md +0 -53
  79. package/framework/agents/teams/review-team.md +0 -70
  80. package/framework/agents/teams/ship-team.md +0 -86
  81. package/framework/agents/test-agent.md +0 -182
  82. package/framework/hooks/auto-format.sh +0 -54
  83. package/framework/hooks/block-env-edit.sh +0 -42
  84. package/framework/hooks/branch-guard.sh +0 -43
  85. package/framework/hooks/confirm-delete.sh +0 -59
  86. package/framework/hooks/migration-validate.sh +0 -77
  87. package/framework/hooks/notification-speak.sh +0 -16
  88. package/framework/hooks/pre-commit.sh +0 -100
  89. package/framework/hooks/pre-compact.sh +0 -56
  90. package/framework/hooks/pre-deploy-gate.sh +0 -160
  91. package/framework/hooks/qualia-colors.sh +0 -32
  92. package/framework/hooks/retention-cleanup.sh +0 -62
  93. package/framework/hooks/save-session-state.sh +0 -185
  94. package/framework/hooks/session-context-loader.sh +0 -96
  95. package/framework/hooks/session-learn.sh +0 -32
  96. package/framework/hooks/skill-announce.sh +0 -123
  97. package/framework/hooks/tool-error-announce.sh +0 -27
  98. package/framework/install.ps1 +0 -323
  99. package/framework/install.sh +0 -313
  100. package/framework/qualia-framework/VERSION +0 -1
  101. package/framework/qualia-framework/assets/qualia-logo.png +0 -0
  102. package/framework/qualia-framework/bin/collect-metrics.sh +0 -67
  103. package/framework/qualia-framework/bin/generate-report-docx.py +0 -429
  104. package/framework/qualia-framework/bin/qualia-tools.js +0 -2201
  105. package/framework/qualia-framework/bin/qualia-tools.test.js +0 -1054
  106. package/framework/qualia-framework/references/checkpoints.md +0 -775
  107. package/framework/qualia-framework/references/completion-checklists.md +0 -359
  108. package/framework/qualia-framework/references/continuation-format.md +0 -249
  109. package/framework/qualia-framework/references/continuation-prompt.md +0 -97
  110. package/framework/qualia-framework/references/decimal-phase-calculation.md +0 -65
  111. package/framework/qualia-framework/references/design-quality.md +0 -56
  112. package/framework/qualia-framework/references/employee-guide.md +0 -167
  113. package/framework/qualia-framework/references/git-integration.md +0 -254
  114. package/framework/qualia-framework/references/git-planning-commit.md +0 -50
  115. package/framework/qualia-framework/references/model-profile-resolution.md +0 -32
  116. package/framework/qualia-framework/references/model-profiles.md +0 -73
  117. package/framework/qualia-framework/references/phase-argument-parsing.md +0 -61
  118. package/framework/qualia-framework/references/planning-config.md +0 -195
  119. package/framework/qualia-framework/references/questioning.md +0 -141
  120. package/framework/qualia-framework/references/tdd.md +0 -263
  121. package/framework/qualia-framework/references/ui-brand.md +0 -160
  122. package/framework/qualia-framework/references/verification-patterns.md +0 -612
  123. package/framework/qualia-framework/templates/DEBUG.md +0 -159
  124. package/framework/qualia-framework/templates/DESIGN.md +0 -81
  125. package/framework/qualia-framework/templates/UAT.md +0 -247
  126. package/framework/qualia-framework/templates/codebase/architecture.md +0 -255
  127. package/framework/qualia-framework/templates/codebase/concerns.md +0 -310
  128. package/framework/qualia-framework/templates/codebase/conventions.md +0 -307
  129. package/framework/qualia-framework/templates/codebase/integrations.md +0 -280
  130. package/framework/qualia-framework/templates/codebase/stack.md +0 -186
  131. package/framework/qualia-framework/templates/codebase/structure.md +0 -285
  132. package/framework/qualia-framework/templates/codebase/testing.md +0 -480
  133. package/framework/qualia-framework/templates/config.json +0 -35
  134. package/framework/qualia-framework/templates/context.md +0 -283
  135. package/framework/qualia-framework/templates/continue-here.md +0 -78
  136. package/framework/qualia-framework/templates/debug-subagent-prompt.md +0 -91
  137. package/framework/qualia-framework/templates/discovery.md +0 -146
  138. package/framework/qualia-framework/templates/lab-notes.md +0 -16
  139. package/framework/qualia-framework/templates/milestone-archive.md +0 -123
  140. package/framework/qualia-framework/templates/milestone.md +0 -115
  141. package/framework/qualia-framework/templates/phase-prompt.md +0 -567
  142. package/framework/qualia-framework/templates/planner-subagent-prompt.md +0 -117
  143. package/framework/qualia-framework/templates/project.md +0 -184
  144. package/framework/qualia-framework/templates/projects/ai-agent.md +0 -156
  145. package/framework/qualia-framework/templates/projects/mobile-app.md +0 -181
  146. package/framework/qualia-framework/templates/projects/voice-agent.md +0 -134
  147. package/framework/qualia-framework/templates/projects/website.md +0 -137
  148. package/framework/qualia-framework/templates/requirements.md +0 -231
  149. package/framework/qualia-framework/templates/research-project/ARCHITECTURE.md +0 -204
  150. package/framework/qualia-framework/templates/research-project/FEATURES.md +0 -147
  151. package/framework/qualia-framework/templates/research-project/PITFALLS.md +0 -200
  152. package/framework/qualia-framework/templates/research-project/STACK.md +0 -120
  153. package/framework/qualia-framework/templates/research-project/SUMMARY.md +0 -170
  154. package/framework/qualia-framework/templates/research.md +0 -552
  155. package/framework/qualia-framework/templates/roadmap.md +0 -206
  156. package/framework/qualia-framework/templates/state.md +0 -179
  157. package/framework/qualia-framework/templates/summary-complex.md +0 -59
  158. package/framework/qualia-framework/templates/summary-minimal.md +0 -41
  159. package/framework/qualia-framework/templates/summary-standard.md +0 -48
  160. package/framework/qualia-framework/templates/summary.md +0 -246
  161. package/framework/qualia-framework/templates/user-setup.md +0 -311
  162. package/framework/qualia-framework/templates/verification-report.md +0 -322
  163. package/framework/qualia-framework/workflows/add-phase.md +0 -179
  164. package/framework/qualia-framework/workflows/add-todo.md +0 -157
  165. package/framework/qualia-framework/workflows/audit-milestone.md +0 -241
  166. package/framework/qualia-framework/workflows/check-todos.md +0 -176
  167. package/framework/qualia-framework/workflows/complete-milestone.md +0 -858
  168. package/framework/qualia-framework/workflows/diagnose-issues.md +0 -219
  169. package/framework/qualia-framework/workflows/discovery-phase.md +0 -289
  170. package/framework/qualia-framework/workflows/discuss-phase.md +0 -534
  171. package/framework/qualia-framework/workflows/execute-phase.md +0 -559
  172. package/framework/qualia-framework/workflows/execute-plan.md +0 -438
  173. package/framework/qualia-framework/workflows/help.md +0 -470
  174. package/framework/qualia-framework/workflows/insert-phase.md +0 -220
  175. package/framework/qualia-framework/workflows/list-phase-assumptions.md +0 -178
  176. package/framework/qualia-framework/workflows/map-codebase.md +0 -327
  177. package/framework/qualia-framework/workflows/new-milestone.md +0 -363
  178. package/framework/qualia-framework/workflows/new-project.md +0 -982
  179. package/framework/qualia-framework/workflows/pause-work.md +0 -122
  180. package/framework/qualia-framework/workflows/plan-milestone-gaps.md +0 -256
  181. package/framework/qualia-framework/workflows/plan-phase.md +0 -422
  182. package/framework/qualia-framework/workflows/progress.md +0 -389
  183. package/framework/qualia-framework/workflows/quick.md +0 -252
  184. package/framework/qualia-framework/workflows/remove-phase.md +0 -326
  185. package/framework/qualia-framework/workflows/research-phase.md +0 -74
  186. package/framework/qualia-framework/workflows/resume-project.md +0 -306
  187. package/framework/qualia-framework/workflows/set-profile.md +0 -80
  188. package/framework/qualia-framework/workflows/settings.md +0 -145
  189. package/framework/qualia-framework/workflows/transition.md +0 -556
  190. package/framework/qualia-framework/workflows/update.md +0 -197
  191. package/framework/qualia-framework/workflows/verify-phase.md +0 -195
  192. package/framework/qualia-framework/workflows/verify-work.md +0 -625
  193. package/framework/rules/context7.md +0 -14
  194. package/framework/rules/frontend.md +0 -33
  195. package/framework/rules/speed.md +0 -23
  196. package/framework/scripts/__pycache__/say.cpython-314.pyc +0 -0
  197. package/framework/scripts/apply-retention.sh +0 -120
  198. package/framework/scripts/bootstrap-pop-os.sh +0 -354
  199. package/framework/scripts/claude-voice +0 -13
  200. package/framework/scripts/cleanup.sh +0 -131
  201. package/framework/scripts/cowork-mode.sh +0 -141
  202. package/framework/scripts/generate-project-claude-md.sh +0 -153
  203. package/framework/scripts/load-test-webhook.js +0 -172
  204. package/framework/scripts/say.py +0 -236
  205. package/framework/scripts/showcase-video-recorder/ffmpeg-builder.js +0 -167
  206. package/framework/scripts/showcase-video-recorder/playwright-helpers.js +0 -216
  207. package/framework/scripts/speak.py +0 -55
  208. package/framework/scripts/speak.sh +0 -18
  209. package/framework/scripts/status.sh +0 -138
  210. package/framework/scripts/sync-to-framework.sh +0 -65
  211. package/framework/scripts/voice-hotkey.py +0 -227
  212. package/framework/scripts/voice-input.sh +0 -51
  213. package/framework/skills/animate/SKILL.md +0 -202
  214. package/framework/skills/bolder/SKILL.md +0 -144
  215. package/framework/skills/browser-qa/SKILL.md +0 -536
  216. package/framework/skills/clarify/SKILL.md +0 -179
  217. package/framework/skills/client-handoff/SKILL.md +0 -135
  218. package/framework/skills/collab-onboard/SKILL.md +0 -111
  219. package/framework/skills/colorize/SKILL.md +0 -170
  220. package/framework/skills/critique/SKILL.md +0 -126
  221. package/framework/skills/deep-research/SKILL.md +0 -240
  222. package/framework/skills/delight/SKILL.md +0 -329
  223. package/framework/skills/deploy/SKILL.md +0 -261
  224. package/framework/skills/deploy-verify/SKILL.md +0 -377
  225. package/framework/skills/deploy-verify/scripts/canary-check.sh +0 -206
  226. package/framework/skills/deploy-verify/scripts/check-console-errors.js +0 -147
  227. package/framework/skills/deploy-verify/scripts/check-cwv.js +0 -139
  228. package/framework/skills/deploy-verify/scripts/project-detect.sh +0 -84
  229. package/framework/skills/deploy-verify/scripts/verify.sh +0 -548
  230. package/framework/skills/design-quieter/SKILL.md +0 -130
  231. package/framework/skills/distill/SKILL.md +0 -149
  232. package/framework/skills/docs-lookup/SKILL.md +0 -79
  233. package/framework/skills/fcm-notifications/SKILL.md +0 -125
  234. package/framework/skills/financial-ledger/SKILL.md +0 -1039
  235. package/framework/skills/frontend-master/NOTICE.md +0 -4
  236. package/framework/skills/frontend-master/SKILL.md +0 -127
  237. package/framework/skills/frontend-master/reference/color-and-contrast.md +0 -132
  238. package/framework/skills/frontend-master/reference/interaction-design.md +0 -123
  239. package/framework/skills/frontend-master/reference/motion-design.md +0 -99
  240. package/framework/skills/frontend-master/reference/responsive-design.md +0 -114
  241. package/framework/skills/frontend-master/reference/spatial-design.md +0 -100
  242. package/framework/skills/frontend-master/reference/typography.md +0 -131
  243. package/framework/skills/frontend-master/reference/ux-writing.md +0 -107
  244. package/framework/skills/harden/SKILL.md +0 -357
  245. package/framework/skills/i18n-rtl/SKILL.md +0 -752
  246. package/framework/skills/learn/SKILL.md +0 -95
  247. package/framework/skills/memory/SKILL.md +0 -50
  248. package/framework/skills/mobile-expo/SKILL.md +0 -977
  249. package/framework/skills/mobile-expo/references/store-checklist.md +0 -550
  250. package/framework/skills/nestjs-backend/README.md +0 -73
  251. package/framework/skills/nestjs-backend/SKILL.md +0 -446
  252. package/framework/skills/nestjs-backend/references/templates.md +0 -1173
  253. package/framework/skills/normalize/SKILL.md +0 -79
  254. package/framework/skills/onboard/SKILL.md +0 -242
  255. package/framework/skills/openrouter-agent/SKILL.md +0 -922
  256. package/framework/skills/polish/SKILL.md +0 -209
  257. package/framework/skills/pr/SKILL.md +0 -66
  258. package/framework/skills/qualia/SKILL.md +0 -199
  259. package/framework/skills/qualia-add-todo/SKILL.md +0 -68
  260. package/framework/skills/qualia-audit-milestone/SKILL.md +0 -95
  261. package/framework/skills/qualia-check-todos/SKILL.md +0 -55
  262. package/framework/skills/qualia-complete-milestone/SKILL.md +0 -134
  263. package/framework/skills/qualia-debug/SKILL.md +0 -149
  264. package/framework/skills/qualia-design/SKILL.md +0 -203
  265. package/framework/skills/qualia-discuss-phase/SKILL.md +0 -72
  266. package/framework/skills/qualia-evolve/SKILL.md +0 -200
  267. package/framework/skills/qualia-execute-phase/SKILL.md +0 -89
  268. package/framework/skills/qualia-framework-audit/SKILL.md +0 -604
  269. package/framework/skills/qualia-guide/SKILL.md +0 -32
  270. package/framework/skills/qualia-help/SKILL.md +0 -114
  271. package/framework/skills/qualia-idk/SKILL.md +0 -352
  272. package/framework/skills/qualia-list-phase-assumptions/SKILL.md +0 -67
  273. package/framework/skills/qualia-new-milestone/SKILL.md +0 -72
  274. package/framework/skills/qualia-new-project/SKILL.md +0 -232
  275. package/framework/skills/qualia-pause-work/SKILL.md +0 -96
  276. package/framework/skills/qualia-plan-milestone-gaps/SKILL.md +0 -57
  277. package/framework/skills/qualia-plan-phase/SKILL.md +0 -104
  278. package/framework/skills/qualia-production-check/SKILL.md +0 -0
  279. package/framework/skills/qualia-progress/SKILL.md +0 -53
  280. package/framework/skills/qualia-quick/SKILL.md +0 -89
  281. package/framework/skills/qualia-report/SKILL.md +0 -166
  282. package/framework/skills/qualia-research-phase/SKILL.md +0 -88
  283. package/framework/skills/qualia-resume-work/SKILL.md +0 -62
  284. package/framework/skills/qualia-review/SKILL.md +0 -263
  285. package/framework/skills/qualia-start/SKILL.md +0 -161
  286. package/framework/skills/qualia-verify-work/SKILL.md +0 -132
  287. package/framework/skills/rag/SKILL.md +0 -750
  288. package/framework/skills/responsive/SKILL.md +0 -231
  289. package/framework/skills/retro/SKILL.md +0 -284
  290. package/framework/skills/sakani-conventions/SKILL.md +0 -136
  291. package/framework/skills/sakani-conventions/evals/evals.json +0 -23
  292. package/framework/skills/sakani-conventions/references/entities.md +0 -365
  293. package/framework/skills/sakani-conventions/references/error-codes.md +0 -95
  294. package/framework/skills/seo-master/SKILL.md +0 -490
  295. package/framework/skills/seo-master/references/checklist.md +0 -199
  296. package/framework/skills/seo-master/references/structured-data.md +0 -609
  297. package/framework/skills/ship/SKILL.md +0 -239
  298. package/framework/skills/stack-researcher/SKILL.md +0 -215
  299. package/framework/skills/status/SKILL.md +0 -154
  300. package/framework/skills/status/scripts/health-check.sh +0 -562
  301. package/framework/skills/subscription-payments/SKILL.md +0 -250
  302. package/framework/skills/supabase/SKILL.md +0 -973
  303. package/framework/skills/supabase/references/templates.md +0 -159
  304. package/framework/skills/team/SKILL.md +0 -67
  305. package/framework/skills/test-runner/SKILL.md +0 -202
  306. package/framework/skills/voice-agent/SKILL.md +0 -1312
  307. package/framework/skills/zoho-workflow/SKILL.md +0 -51
  308. package/framework/statusline-command.sh +0 -117
  309. package/framework/teams/default/inboxes/plan-04.json +0 -9
  310. package/framework/teams/review-team.md +0 -75
  311. package/framework/teams/ship-team.md +0 -86
  312. package/profiles/fawzi.json +0 -16
  313. package/profiles/hasan.json +0 -16
  314. package/profiles/moayad.json +0 -16
  315. package/templates/CLAUDE-owner.md +0 -52
  316. package/templates/CLAUDE.md.hbs +0 -58
  317. package/templates/env.claude.template +0 -12
  318. package/templates/settings.json +0 -172
  319. package/uninstall.sh +0 -90
  320. /package/{framework/rules → rules}/deployment.md +0 -0
  321. /package/{framework/rules → rules}/security.md +0 -0
@@ -1,562 +0,0 @@
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