@sonde/packs 0.1.0 → 0.1.1

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 (359) hide show
  1. package/.turbo/turbo-build.log +6 -0
  2. package/.turbo/turbo-test.log +814 -0
  3. package/.turbo/turbo-typecheck.log +4 -0
  4. package/CHANGELOG.md +10 -0
  5. package/dist/index.d.ts +16 -1
  6. package/dist/index.d.ts.map +1 -1
  7. package/dist/index.js +40 -2
  8. package/dist/index.js.map +1 -1
  9. package/dist/integrations/citrix.d.ts +13 -0
  10. package/dist/integrations/citrix.d.ts.map +1 -0
  11. package/dist/integrations/citrix.js +420 -0
  12. package/dist/integrations/citrix.js.map +1 -0
  13. package/dist/integrations/citrix.test.d.ts +2 -0
  14. package/dist/integrations/citrix.test.d.ts.map +1 -0
  15. package/dist/integrations/citrix.test.js +464 -0
  16. package/dist/integrations/citrix.test.js.map +1 -0
  17. package/dist/integrations/graph.d.ts +9 -0
  18. package/dist/integrations/graph.d.ts.map +1 -0
  19. package/dist/integrations/graph.js +290 -0
  20. package/dist/integrations/graph.js.map +1 -0
  21. package/dist/integrations/graph.test.d.ts +2 -0
  22. package/dist/integrations/graph.test.d.ts.map +1 -0
  23. package/dist/integrations/graph.test.js +356 -0
  24. package/dist/integrations/graph.test.js.map +1 -0
  25. package/dist/integrations/httpbin.d.ts +3 -0
  26. package/dist/integrations/httpbin.d.ts.map +1 -0
  27. package/dist/integrations/httpbin.js +70 -0
  28. package/dist/integrations/httpbin.js.map +1 -0
  29. package/dist/integrations/nutanix.d.ts +18 -0
  30. package/dist/integrations/nutanix.d.ts.map +1 -0
  31. package/dist/integrations/nutanix.js +1121 -0
  32. package/dist/integrations/nutanix.js.map +1 -0
  33. package/dist/integrations/nutanix.test.d.ts +2 -0
  34. package/dist/integrations/nutanix.test.d.ts.map +1 -0
  35. package/dist/integrations/nutanix.test.js +978 -0
  36. package/dist/integrations/nutanix.test.js.map +1 -0
  37. package/dist/integrations/proxmox.d.ts +12 -0
  38. package/dist/integrations/proxmox.d.ts.map +1 -0
  39. package/dist/integrations/proxmox.js +733 -0
  40. package/dist/integrations/proxmox.js.map +1 -0
  41. package/dist/integrations/proxmox.test.d.ts +2 -0
  42. package/dist/integrations/proxmox.test.d.ts.map +1 -0
  43. package/dist/integrations/proxmox.test.js +697 -0
  44. package/dist/integrations/proxmox.test.js.map +1 -0
  45. package/dist/integrations/servicenow.d.ts +3 -0
  46. package/dist/integrations/servicenow.d.ts.map +1 -0
  47. package/dist/integrations/servicenow.js +257 -0
  48. package/dist/integrations/servicenow.js.map +1 -0
  49. package/dist/integrations/servicenow.test.d.ts +2 -0
  50. package/dist/integrations/servicenow.test.d.ts.map +1 -0
  51. package/dist/integrations/servicenow.test.js +217 -0
  52. package/dist/integrations/servicenow.test.js.map +1 -0
  53. package/dist/integrations/splunk.d.ts +9 -0
  54. package/dist/integrations/splunk.d.ts.map +1 -0
  55. package/dist/integrations/splunk.js +242 -0
  56. package/dist/integrations/splunk.js.map +1 -0
  57. package/dist/integrations/splunk.test.d.ts +2 -0
  58. package/dist/integrations/splunk.test.d.ts.map +1 -0
  59. package/dist/integrations/splunk.test.js +323 -0
  60. package/dist/integrations/splunk.test.js.map +1 -0
  61. package/dist/mysql/index.d.ts +3 -0
  62. package/dist/mysql/index.d.ts.map +1 -0
  63. package/dist/mysql/index.js +13 -0
  64. package/dist/mysql/index.js.map +1 -0
  65. package/dist/mysql/manifest.d.ts +3 -0
  66. package/dist/mysql/manifest.d.ts.map +1 -0
  67. package/dist/mysql/manifest.js +69 -0
  68. package/dist/mysql/manifest.js.map +1 -0
  69. package/dist/mysql/probes/databases-list.d.ts +13 -0
  70. package/dist/mysql/probes/databases-list.d.ts.map +1 -0
  71. package/dist/mysql/probes/databases-list.js +31 -0
  72. package/dist/mysql/probes/databases-list.js.map +1 -0
  73. package/dist/mysql/probes/databases-list.test.d.ts +2 -0
  74. package/dist/mysql/probes/databases-list.test.d.ts.map +1 -0
  75. package/dist/mysql/probes/databases-list.test.js +54 -0
  76. package/dist/mysql/probes/databases-list.test.js.map +1 -0
  77. package/dist/mysql/probes/processlist.d.ts +18 -0
  78. package/dist/mysql/probes/processlist.d.ts.map +1 -0
  79. package/dist/mysql/probes/processlist.js +36 -0
  80. package/dist/mysql/probes/processlist.js.map +1 -0
  81. package/dist/mysql/probes/processlist.test.d.ts +2 -0
  82. package/dist/mysql/probes/processlist.test.d.ts.map +1 -0
  83. package/dist/mysql/probes/processlist.test.js +41 -0
  84. package/dist/mysql/probes/processlist.test.js.map +1 -0
  85. package/dist/mysql/probes/status.d.ts +14 -0
  86. package/dist/mysql/probes/status.d.ts.map +1 -0
  87. package/dist/mysql/probes/status.js +40 -0
  88. package/dist/mysql/probes/status.js.map +1 -0
  89. package/dist/mysql/probes/status.test.d.ts +2 -0
  90. package/dist/mysql/probes/status.test.d.ts.map +1 -0
  91. package/dist/mysql/probes/status.test.js +43 -0
  92. package/dist/mysql/probes/status.test.js.map +1 -0
  93. package/dist/nginx/index.d.ts +3 -0
  94. package/dist/nginx/index.d.ts.map +1 -0
  95. package/dist/nginx/index.js +13 -0
  96. package/dist/nginx/index.js.map +1 -0
  97. package/dist/nginx/manifest.d.ts +3 -0
  98. package/dist/nginx/manifest.d.ts.map +1 -0
  99. package/dist/nginx/manifest.js +68 -0
  100. package/dist/nginx/manifest.js.map +1 -0
  101. package/dist/nginx/probes/access-log-tail.d.ts +9 -0
  102. package/dist/nginx/probes/access-log-tail.d.ts.map +1 -0
  103. package/dist/nginx/probes/access-log-tail.js +14 -0
  104. package/dist/nginx/probes/access-log-tail.js.map +1 -0
  105. package/dist/nginx/probes/access-log-tail.test.d.ts +2 -0
  106. package/dist/nginx/probes/access-log-tail.test.d.ts.map +1 -0
  107. package/dist/nginx/probes/access-log-tail.test.js +40 -0
  108. package/dist/nginx/probes/access-log-tail.test.js.map +1 -0
  109. package/dist/nginx/probes/config-test.d.ts +8 -0
  110. package/dist/nginx/probes/config-test.d.ts.map +1 -0
  111. package/dist/nginx/probes/config-test.js +18 -0
  112. package/dist/nginx/probes/config-test.js.map +1 -0
  113. package/dist/nginx/probes/config-test.test.d.ts +2 -0
  114. package/dist/nginx/probes/config-test.test.d.ts.map +1 -0
  115. package/dist/nginx/probes/config-test.test.js +35 -0
  116. package/dist/nginx/probes/config-test.test.js.map +1 -0
  117. package/dist/nginx/probes/error-log-tail.d.ts +9 -0
  118. package/dist/nginx/probes/error-log-tail.d.ts.map +1 -0
  119. package/dist/nginx/probes/error-log-tail.js +14 -0
  120. package/dist/nginx/probes/error-log-tail.js.map +1 -0
  121. package/dist/nginx/probes/error-log-tail.test.d.ts +2 -0
  122. package/dist/nginx/probes/error-log-tail.test.d.ts.map +1 -0
  123. package/dist/nginx/probes/error-log-tail.test.js +34 -0
  124. package/dist/nginx/probes/error-log-tail.test.js.map +1 -0
  125. package/dist/postgres/index.d.ts +3 -0
  126. package/dist/postgres/index.d.ts.map +1 -0
  127. package/dist/postgres/index.js +13 -0
  128. package/dist/postgres/index.js.map +1 -0
  129. package/dist/postgres/manifest.d.ts +3 -0
  130. package/dist/postgres/manifest.d.ts.map +1 -0
  131. package/dist/postgres/manifest.js +90 -0
  132. package/dist/postgres/manifest.js.map +1 -0
  133. package/dist/postgres/probes/connections-active.d.ts +17 -0
  134. package/dist/postgres/probes/connections-active.d.ts.map +1 -0
  135. package/dist/postgres/probes/connections-active.js +37 -0
  136. package/dist/postgres/probes/connections-active.js.map +1 -0
  137. package/dist/postgres/probes/connections-active.test.d.ts +2 -0
  138. package/dist/postgres/probes/connections-active.test.d.ts.map +1 -0
  139. package/dist/postgres/probes/connections-active.test.js +36 -0
  140. package/dist/postgres/probes/connections-active.test.js.map +1 -0
  141. package/dist/postgres/probes/databases-list.d.ts +14 -0
  142. package/dist/postgres/probes/databases-list.d.ts.map +1 -0
  143. package/dist/postgres/probes/databases-list.js +34 -0
  144. package/dist/postgres/probes/databases-list.js.map +1 -0
  145. package/dist/postgres/probes/databases-list.test.d.ts +2 -0
  146. package/dist/postgres/probes/databases-list.test.d.ts.map +1 -0
  147. package/dist/postgres/probes/databases-list.test.js +49 -0
  148. package/dist/postgres/probes/databases-list.test.js.map +1 -0
  149. package/dist/postgres/probes/query-slow.d.ts +17 -0
  150. package/dist/postgres/probes/query-slow.d.ts.map +1 -0
  151. package/dist/postgres/probes/query-slow.js +37 -0
  152. package/dist/postgres/probes/query-slow.js.map +1 -0
  153. package/dist/postgres/probes/query-slow.test.d.ts +2 -0
  154. package/dist/postgres/probes/query-slow.test.d.ts.map +1 -0
  155. package/dist/postgres/probes/query-slow.test.js +30 -0
  156. package/dist/postgres/probes/query-slow.test.js.map +1 -0
  157. package/dist/proxmox/index.d.ts +3 -0
  158. package/dist/proxmox/index.d.ts.map +1 -0
  159. package/dist/proxmox/index.js +23 -0
  160. package/dist/proxmox/index.js.map +1 -0
  161. package/dist/proxmox/manifest.d.ts +3 -0
  162. package/dist/proxmox/manifest.d.ts.map +1 -0
  163. package/dist/proxmox/manifest.js +75 -0
  164. package/dist/proxmox/manifest.js.map +1 -0
  165. package/dist/proxmox/probes/ceph-status.d.ts +36 -0
  166. package/dist/proxmox/probes/ceph-status.d.ts.map +1 -0
  167. package/dist/proxmox/probes/ceph-status.js +71 -0
  168. package/dist/proxmox/probes/ceph-status.js.map +1 -0
  169. package/dist/proxmox/probes/ceph-status.test.d.ts +2 -0
  170. package/dist/proxmox/probes/ceph-status.test.d.ts.map +1 -0
  171. package/dist/proxmox/probes/ceph-status.test.js +115 -0
  172. package/dist/proxmox/probes/ceph-status.test.js.map +1 -0
  173. package/dist/proxmox/probes/cluster-config.d.ts +31 -0
  174. package/dist/proxmox/probes/cluster-config.d.ts.map +1 -0
  175. package/dist/proxmox/probes/cluster-config.js +72 -0
  176. package/dist/proxmox/probes/cluster-config.js.map +1 -0
  177. package/dist/proxmox/probes/cluster-config.test.d.ts +2 -0
  178. package/dist/proxmox/probes/cluster-config.test.d.ts.map +1 -0
  179. package/dist/proxmox/probes/cluster-config.test.js +107 -0
  180. package/dist/proxmox/probes/cluster-config.test.js.map +1 -0
  181. package/dist/proxmox/probes/ha-status.d.ts +18 -0
  182. package/dist/proxmox/probes/ha-status.d.ts.map +1 -0
  183. package/dist/proxmox/probes/ha-status.js +38 -0
  184. package/dist/proxmox/probes/ha-status.js.map +1 -0
  185. package/dist/proxmox/probes/ha-status.test.d.ts +2 -0
  186. package/dist/proxmox/probes/ha-status.test.d.ts.map +1 -0
  187. package/dist/proxmox/probes/ha-status.test.js +66 -0
  188. package/dist/proxmox/probes/ha-status.test.js.map +1 -0
  189. package/dist/proxmox/probes/lvm.d.ts +35 -0
  190. package/dist/proxmox/probes/lvm.d.ts.map +1 -0
  191. package/dist/proxmox/probes/lvm.js +75 -0
  192. package/dist/proxmox/probes/lvm.js.map +1 -0
  193. package/dist/proxmox/probes/lvm.test.d.ts +2 -0
  194. package/dist/proxmox/probes/lvm.test.d.ts.map +1 -0
  195. package/dist/proxmox/probes/lvm.test.js +128 -0
  196. package/dist/proxmox/probes/lvm.test.js.map +1 -0
  197. package/dist/proxmox/probes/lxc-config.d.ts +29 -0
  198. package/dist/proxmox/probes/lxc-config.d.ts.map +1 -0
  199. package/dist/proxmox/probes/lxc-config.js +67 -0
  200. package/dist/proxmox/probes/lxc-config.js.map +1 -0
  201. package/dist/proxmox/probes/lxc-config.test.d.ts +2 -0
  202. package/dist/proxmox/probes/lxc-config.test.d.ts.map +1 -0
  203. package/dist/proxmox/probes/lxc-config.test.js +77 -0
  204. package/dist/proxmox/probes/lxc-config.test.js.map +1 -0
  205. package/dist/proxmox/probes/lxc-list.d.ts +20 -0
  206. package/dist/proxmox/probes/lxc-list.d.ts.map +1 -0
  207. package/dist/proxmox/probes/lxc-list.js +49 -0
  208. package/dist/proxmox/probes/lxc-list.js.map +1 -0
  209. package/dist/proxmox/probes/lxc-list.test.d.ts +2 -0
  210. package/dist/proxmox/probes/lxc-list.test.d.ts.map +1 -0
  211. package/dist/proxmox/probes/lxc-list.test.js +51 -0
  212. package/dist/proxmox/probes/lxc-list.test.js.map +1 -0
  213. package/dist/proxmox/probes/vm-config.d.ts +21 -0
  214. package/dist/proxmox/probes/vm-config.d.ts.map +1 -0
  215. package/dist/proxmox/probes/vm-config.js +58 -0
  216. package/dist/proxmox/probes/vm-config.js.map +1 -0
  217. package/dist/proxmox/probes/vm-config.test.d.ts +2 -0
  218. package/dist/proxmox/probes/vm-config.test.d.ts.map +1 -0
  219. package/dist/proxmox/probes/vm-config.test.js +80 -0
  220. package/dist/proxmox/probes/vm-config.test.js.map +1 -0
  221. package/dist/proxmox/probes/vm-locks.d.ts +16 -0
  222. package/dist/proxmox/probes/vm-locks.d.ts.map +1 -0
  223. package/dist/proxmox/probes/vm-locks.js +35 -0
  224. package/dist/proxmox/probes/vm-locks.js.map +1 -0
  225. package/dist/proxmox/probes/vm-locks.test.d.ts +2 -0
  226. package/dist/proxmox/probes/vm-locks.test.d.ts.map +1 -0
  227. package/dist/proxmox/probes/vm-locks.test.js +54 -0
  228. package/dist/proxmox/probes/vm-locks.test.js.map +1 -0
  229. package/dist/redis/index.d.ts +3 -0
  230. package/dist/redis/index.d.ts.map +1 -0
  231. package/dist/redis/index.js +13 -0
  232. package/dist/redis/index.js.map +1 -0
  233. package/dist/redis/manifest.d.ts +3 -0
  234. package/dist/redis/manifest.d.ts.map +1 -0
  235. package/dist/redis/manifest.js +51 -0
  236. package/dist/redis/manifest.js.map +1 -0
  237. package/dist/redis/probes/info.d.ts +15 -0
  238. package/dist/redis/probes/info.d.ts.map +1 -0
  239. package/dist/redis/probes/info.js +32 -0
  240. package/dist/redis/probes/info.js.map +1 -0
  241. package/dist/redis/probes/info.test.d.ts +2 -0
  242. package/dist/redis/probes/info.test.d.ts.map +1 -0
  243. package/dist/redis/probes/info.test.js +64 -0
  244. package/dist/redis/probes/info.test.js.map +1 -0
  245. package/dist/redis/probes/keys-count.d.ts +13 -0
  246. package/dist/redis/probes/keys-count.d.ts.map +1 -0
  247. package/dist/redis/probes/keys-count.js +24 -0
  248. package/dist/redis/probes/keys-count.js.map +1 -0
  249. package/dist/redis/probes/keys-count.test.d.ts +2 -0
  250. package/dist/redis/probes/keys-count.test.d.ts.map +1 -0
  251. package/dist/redis/probes/keys-count.test.js +37 -0
  252. package/dist/redis/probes/keys-count.test.js.map +1 -0
  253. package/dist/redis/probes/memory-usage.d.ts +16 -0
  254. package/dist/redis/probes/memory-usage.d.ts.map +1 -0
  255. package/dist/redis/probes/memory-usage.js +31 -0
  256. package/dist/redis/probes/memory-usage.js.map +1 -0
  257. package/dist/redis/probes/memory-usage.test.d.ts +2 -0
  258. package/dist/redis/probes/memory-usage.test.d.ts.map +1 -0
  259. package/dist/redis/probes/memory-usage.test.js +48 -0
  260. package/dist/redis/probes/memory-usage.test.js.map +1 -0
  261. package/dist/runbooks/nutanix.d.ts +3 -0
  262. package/dist/runbooks/nutanix.d.ts.map +1 -0
  263. package/dist/runbooks/nutanix.js +619 -0
  264. package/dist/runbooks/nutanix.js.map +1 -0
  265. package/dist/runbooks/nutanix.test.d.ts +2 -0
  266. package/dist/runbooks/nutanix.test.d.ts.map +1 -0
  267. package/dist/runbooks/nutanix.test.js +971 -0
  268. package/dist/runbooks/nutanix.test.js.map +1 -0
  269. package/dist/runbooks/proxmox.d.ts +3 -0
  270. package/dist/runbooks/proxmox.d.ts.map +1 -0
  271. package/dist/runbooks/proxmox.js +451 -0
  272. package/dist/runbooks/proxmox.js.map +1 -0
  273. package/dist/runbooks/proxmox.test.d.ts +2 -0
  274. package/dist/runbooks/proxmox.test.d.ts.map +1 -0
  275. package/dist/runbooks/proxmox.test.js +700 -0
  276. package/dist/runbooks/proxmox.test.js.map +1 -0
  277. package/dist/signatures.d.ts +2 -0
  278. package/dist/signatures.d.ts.map +1 -0
  279. package/dist/signatures.js +2 -0
  280. package/dist/signatures.js.map +1 -0
  281. package/dist/types.d.ts +53 -0
  282. package/dist/types.d.ts.map +1 -1
  283. package/dist/validation.d.ts +6 -1
  284. package/dist/validation.d.ts.map +1 -1
  285. package/dist/validation.js +10 -1
  286. package/dist/validation.js.map +1 -1
  287. package/package.json +1 -1
  288. package/src/index.ts +60 -6
  289. package/src/integrations/citrix.test.ts +592 -0
  290. package/src/integrations/citrix.ts +557 -0
  291. package/src/integrations/graph.test.ts +478 -0
  292. package/src/integrations/graph.ts +413 -0
  293. package/src/integrations/httpbin.ts +72 -0
  294. package/src/integrations/nutanix.test.ts +1508 -0
  295. package/src/integrations/nutanix.ts +1460 -0
  296. package/src/integrations/proxmox.test.ts +1020 -0
  297. package/src/integrations/proxmox.ts +989 -0
  298. package/src/integrations/servicenow.test.ts +314 -0
  299. package/src/integrations/servicenow.ts +285 -0
  300. package/src/integrations/splunk.test.ts +440 -0
  301. package/src/integrations/splunk.ts +356 -0
  302. package/src/mysql/index.ts +14 -0
  303. package/src/mysql/manifest.ts +70 -0
  304. package/src/mysql/probes/databases-list.test.ts +62 -0
  305. package/src/mysql/probes/databases-list.ts +45 -0
  306. package/src/mysql/probes/processlist.test.ts +47 -0
  307. package/src/mysql/probes/processlist.ts +55 -0
  308. package/src/mysql/probes/status.test.ts +50 -0
  309. package/src/mysql/probes/status.ts +56 -0
  310. package/src/nginx/index.ts +14 -0
  311. package/src/nginx/manifest.ts +69 -0
  312. package/src/nginx/probes/access-log-tail.test.ts +51 -0
  313. package/src/nginx/probes/access-log-tail.ts +23 -0
  314. package/src/nginx/probes/config-test.test.ts +47 -0
  315. package/src/nginx/probes/config-test.ts +24 -0
  316. package/src/nginx/probes/error-log-tail.test.ts +44 -0
  317. package/src/nginx/probes/error-log-tail.ts +23 -0
  318. package/src/postgres/index.ts +14 -0
  319. package/src/postgres/manifest.ts +91 -0
  320. package/src/postgres/probes/connections-active.test.ts +42 -0
  321. package/src/postgres/probes/connections-active.ts +55 -0
  322. package/src/postgres/probes/databases-list.test.ts +57 -0
  323. package/src/postgres/probes/databases-list.ts +49 -0
  324. package/src/postgres/probes/query-slow.test.ts +37 -0
  325. package/src/postgres/probes/query-slow.ts +55 -0
  326. package/src/proxmox/index.ts +24 -0
  327. package/src/proxmox/manifest.ts +76 -0
  328. package/src/proxmox/probes/ceph-status.test.ts +126 -0
  329. package/src/proxmox/probes/ceph-status.ts +116 -0
  330. package/src/proxmox/probes/cluster-config.test.ts +118 -0
  331. package/src/proxmox/probes/cluster-config.ts +97 -0
  332. package/src/proxmox/probes/ha-status.test.ts +76 -0
  333. package/src/proxmox/probes/ha-status.ts +56 -0
  334. package/src/proxmox/probes/lvm.test.ts +140 -0
  335. package/src/proxmox/probes/lvm.ts +121 -0
  336. package/src/proxmox/probes/lxc-config.test.ts +89 -0
  337. package/src/proxmox/probes/lxc-config.ts +90 -0
  338. package/src/proxmox/probes/lxc-list.test.ts +60 -0
  339. package/src/proxmox/probes/lxc-list.ts +67 -0
  340. package/src/proxmox/probes/vm-config.test.ts +93 -0
  341. package/src/proxmox/probes/vm-config.ts +77 -0
  342. package/src/proxmox/probes/vm-locks.test.ts +63 -0
  343. package/src/proxmox/probes/vm-locks.ts +49 -0
  344. package/src/redis/index.ts +14 -0
  345. package/src/redis/manifest.ts +52 -0
  346. package/src/redis/probes/info.test.ts +73 -0
  347. package/src/redis/probes/info.ts +46 -0
  348. package/src/redis/probes/keys-count.test.ts +44 -0
  349. package/src/redis/probes/keys-count.ts +38 -0
  350. package/src/redis/probes/memory-usage.test.ts +54 -0
  351. package/src/redis/probes/memory-usage.ts +46 -0
  352. package/src/runbooks/nutanix.test.ts +1138 -0
  353. package/src/runbooks/nutanix.ts +941 -0
  354. package/src/runbooks/proxmox.test.ts +838 -0
  355. package/src/runbooks/proxmox.ts +626 -0
  356. package/src/signatures.ts +1 -0
  357. package/src/types.ts +62 -0
  358. package/src/validation.ts +21 -1
  359. package/tsconfig.tsbuildinfo +1 -0
@@ -0,0 +1,700 @@
1
+ import { describe, expect, it } from 'vitest';
2
+ import { proxmoxDiagnosticRunbooks } from './proxmox.js';
3
+ // --- Test helpers ---
4
+ function getHandler(category) {
5
+ const def = proxmoxDiagnosticRunbooks.find((r) => r.category === category);
6
+ if (!def)
7
+ throw new Error(`No runbook for category "${category}"`);
8
+ return def.handler;
9
+ }
10
+ function mockResult(probe, data, status = 'success', error) {
11
+ return { probe, status, data, durationMs: 10, error };
12
+ }
13
+ /** Create a mock runProbe that returns canned data keyed by probe name */
14
+ function createMockRunProbe(responses) {
15
+ return async (probe, _params, _agent) => {
16
+ const key = probe;
17
+ const response = responses[key];
18
+ if (!response) {
19
+ return mockResult(probe, undefined, 'error', `No mock for probe ${probe}`);
20
+ }
21
+ return mockResult(probe, response.data, response.status ?? 'success', response.error);
22
+ };
23
+ }
24
+ const defaultContext = { connectedAgents: [] };
25
+ // --- Healthy cluster data fixtures ---
26
+ const healthyClusterStatus = {
27
+ clusterName: 'prod',
28
+ quorate: true,
29
+ nodes: [
30
+ { name: 'pve01', online: true },
31
+ { name: 'pve02', online: true },
32
+ ],
33
+ warnings: [],
34
+ };
35
+ const healthyHaStatus = {
36
+ managerStatus: 'active',
37
+ resources: [{ sid: 'vm:100', state: 'started', node: 'pve01', type: 'vm' }],
38
+ warnings: [],
39
+ };
40
+ const healthyVmStatus = {
41
+ vmid: 100,
42
+ name: 'web-01',
43
+ status: 'running',
44
+ node: 'pve01',
45
+ type: 'qemu',
46
+ uptime: 86400,
47
+ cpu: 0.1,
48
+ mem: 2e9,
49
+ maxmem: 4e9,
50
+ lock: null,
51
+ hastate: 'managed',
52
+ warnings: [],
53
+ };
54
+ const healthyVmConfig = {
55
+ vmid: 100,
56
+ node: 'pve01',
57
+ config: {},
58
+ disks: [{ key: 'scsi0', storage: 'ceph-pool', format: 'raw', size: '32G' }],
59
+ warnings: [],
60
+ };
61
+ const healthyNodeStorage = {
62
+ storages: [
63
+ {
64
+ storage: 'local',
65
+ type: 'dir',
66
+ total: 100e9,
67
+ used: 40e9,
68
+ avail: 60e9,
69
+ shared: false,
70
+ enabled: true,
71
+ active: true,
72
+ },
73
+ {
74
+ storage: 'local-lvm',
75
+ type: 'lvmthin',
76
+ total: 500e9,
77
+ used: 100e9,
78
+ avail: 400e9,
79
+ shared: false,
80
+ enabled: true,
81
+ active: true,
82
+ },
83
+ {
84
+ storage: 'ceph-pool',
85
+ type: 'rbd',
86
+ total: 1e12,
87
+ used: 200e9,
88
+ avail: 800e9,
89
+ shared: true,
90
+ enabled: true,
91
+ active: true,
92
+ },
93
+ ],
94
+ warnings: [],
95
+ };
96
+ const emptyTasks = { tasks: [], warnings: [] };
97
+ // =============================================================================
98
+ // proxmox-vm-health tests
99
+ // =============================================================================
100
+ describe('proxmox-vm-health', () => {
101
+ const handler = getHandler('proxmox-vm');
102
+ it('healthy QEMU VM — all probes good', async () => {
103
+ const runProbe = createMockRunProbe({
104
+ 'proxmox.cluster.status': { data: healthyClusterStatus },
105
+ 'proxmox.cluster.ha.status': { data: healthyHaStatus },
106
+ 'proxmox.vm.status': { data: healthyVmStatus },
107
+ 'proxmox.cluster.tasks': { data: emptyTasks },
108
+ 'proxmox.vm.config': { data: healthyVmConfig },
109
+ 'proxmox.node.storage': { data: healthyNodeStorage },
110
+ });
111
+ const result = await handler({ vmid: 100 }, runProbe, defaultContext);
112
+ expect(result.category).toBe('proxmox-vm');
113
+ expect(result.findings).toHaveLength(1);
114
+ expect(result.findings[0]?.severity).toBe('info');
115
+ expect(result.findings[0]?.title).toContain('VM 100 is healthy');
116
+ expect(result.summary.probesRun).toBeGreaterThanOrEqual(6);
117
+ });
118
+ it('healthy LXC container — routes to lxc.config', async () => {
119
+ const lxcVmStatus = { ...healthyVmStatus, vmid: 200, type: 'lxc', name: 'ct-db' };
120
+ const lxcConfig = {
121
+ vmid: 200,
122
+ node: 'pve01',
123
+ config: {},
124
+ rootfs: { storage: 'ceph-pool', size: '8G' },
125
+ mountpoints: [],
126
+ warnings: [],
127
+ };
128
+ const haNoResources = { ...healthyHaStatus, resources: [] };
129
+ const calls = [];
130
+ const runProbe = async (probe, _params, _agent) => {
131
+ calls.push(probe);
132
+ const responses = {
133
+ 'proxmox.cluster.status': healthyClusterStatus,
134
+ 'proxmox.cluster.ha.status': haNoResources,
135
+ 'proxmox.vm.status': lxcVmStatus,
136
+ 'proxmox.cluster.tasks': emptyTasks,
137
+ 'proxmox.lxc.config': lxcConfig,
138
+ 'proxmox.node.storage': healthyNodeStorage,
139
+ };
140
+ return mockResult(probe, responses[probe]);
141
+ };
142
+ const result = await handler({ vmid: 200 }, runProbe, defaultContext);
143
+ expect(calls).toContain('proxmox.lxc.config');
144
+ expect(calls).not.toContain('proxmox.vm.config');
145
+ expect(result.findings.some((f) => f.severity === 'info')).toBe(true);
146
+ });
147
+ it('HA error state — critical finding with ha-manager remediation', async () => {
148
+ const haError = {
149
+ ...healthyHaStatus,
150
+ resources: [{ sid: 'vm:100', state: 'error', node: 'pve01', type: 'vm' }],
151
+ };
152
+ const runProbe = createMockRunProbe({
153
+ 'proxmox.cluster.status': { data: healthyClusterStatus },
154
+ 'proxmox.cluster.ha.status': { data: haError },
155
+ 'proxmox.vm.status': { data: healthyVmStatus },
156
+ 'proxmox.cluster.tasks': { data: emptyTasks },
157
+ 'proxmox.vm.config': { data: healthyVmConfig },
158
+ 'proxmox.node.storage': { data: healthyNodeStorage },
159
+ });
160
+ const result = await handler({ vmid: 100 }, runProbe, defaultContext);
161
+ const critical = result.findings.filter((f) => f.severity === 'critical');
162
+ expect(critical.length).toBeGreaterThanOrEqual(1);
163
+ const haFinding = critical.find((f) => f.title.includes('HA error'));
164
+ expect(haFinding).toBeDefined();
165
+ expect(haFinding?.remediation).toContain('ha-manager set vm:100 --state disabled');
166
+ });
167
+ it('VM locked — warning with unlock remediation', async () => {
168
+ const lockedVm = { ...healthyVmStatus, lock: 'backup' };
169
+ const haNoResources = { ...healthyHaStatus, resources: [] };
170
+ const runProbe = createMockRunProbe({
171
+ 'proxmox.cluster.status': { data: healthyClusterStatus },
172
+ 'proxmox.cluster.ha.status': { data: haNoResources },
173
+ 'proxmox.vm.status': { data: lockedVm },
174
+ 'proxmox.cluster.tasks': { data: emptyTasks },
175
+ 'proxmox.vm.config': { data: healthyVmConfig },
176
+ 'proxmox.node.storage': { data: healthyNodeStorage },
177
+ });
178
+ const result = await handler({ vmid: 100 }, runProbe, defaultContext);
179
+ const warning = result.findings.find((f) => f.title.includes('stale lock'));
180
+ expect(warning).toBeDefined();
181
+ expect(warning?.severity).toBe('warning');
182
+ expect(warning?.remediation).toContain('qm unlock 100');
183
+ });
184
+ it('local storage + HA — critical finding with qm move-disk', async () => {
185
+ const localConfig = {
186
+ ...healthyVmConfig,
187
+ disks: [{ key: 'scsi0', storage: 'local-lvm', format: 'raw', size: '32G' }],
188
+ };
189
+ const runProbe = createMockRunProbe({
190
+ 'proxmox.cluster.status': { data: healthyClusterStatus },
191
+ 'proxmox.cluster.ha.status': { data: healthyHaStatus },
192
+ 'proxmox.vm.status': { data: healthyVmStatus },
193
+ 'proxmox.cluster.tasks': { data: emptyTasks },
194
+ 'proxmox.vm.config': { data: localConfig },
195
+ 'proxmox.node.storage': { data: healthyNodeStorage },
196
+ });
197
+ const result = await handler({ vmid: 100 }, runProbe, defaultContext);
198
+ const critical = result.findings.find((f) => f.title.includes('local storage but is HA-managed'));
199
+ expect(critical).toBeDefined();
200
+ expect(critical?.severity).toBe('critical');
201
+ expect(critical?.remediation).toContain('qm move-disk 100 scsi0 ceph-pool --delete');
202
+ });
203
+ it('storage inaccessible — critical finding', async () => {
204
+ const inactiveStorage = {
205
+ storages: [
206
+ ...healthyNodeStorage.storages.map((s) => s.storage === 'ceph-pool' ? { ...s, active: false } : s),
207
+ ],
208
+ warnings: [],
209
+ };
210
+ const runProbe = createMockRunProbe({
211
+ 'proxmox.cluster.status': { data: healthyClusterStatus },
212
+ 'proxmox.cluster.ha.status': { data: healthyHaStatus },
213
+ 'proxmox.vm.status': { data: healthyVmStatus },
214
+ 'proxmox.cluster.tasks': { data: emptyTasks },
215
+ 'proxmox.vm.config': { data: healthyVmConfig },
216
+ 'proxmox.node.storage': { data: inactiveStorage },
217
+ });
218
+ const result = await handler({ vmid: 100 }, runProbe, defaultContext);
219
+ const critical = result.findings.find((f) => f.title.includes('not accessible'));
220
+ expect(critical).toBeDefined();
221
+ expect(critical?.severity).toBe('critical');
222
+ expect(critical?.remediation).toContain('LVM/NFS');
223
+ });
224
+ it('failed recent tasks — warning finding', async () => {
225
+ const failedTasks = {
226
+ tasks: [
227
+ {
228
+ type: 'qmstart',
229
+ status: 'ERROR: start failed',
230
+ node: 'pve01',
231
+ starttime: 1000,
232
+ endtime: 1010,
233
+ },
234
+ ],
235
+ warnings: [],
236
+ };
237
+ const haNoResources = { ...healthyHaStatus, resources: [] };
238
+ const runProbe = createMockRunProbe({
239
+ 'proxmox.cluster.status': { data: healthyClusterStatus },
240
+ 'proxmox.cluster.ha.status': { data: haNoResources },
241
+ 'proxmox.vm.status': { data: healthyVmStatus },
242
+ 'proxmox.cluster.tasks': { data: failedTasks },
243
+ 'proxmox.vm.config': { data: healthyVmConfig },
244
+ 'proxmox.node.storage': { data: healthyNodeStorage },
245
+ });
246
+ const result = await handler({ vmid: 100 }, runProbe, defaultContext);
247
+ const warning = result.findings.find((f) => f.title.includes('task failures'));
248
+ expect(warning).toBeDefined();
249
+ expect(warning?.severity).toBe('warning');
250
+ });
251
+ it('VM not found — handles error gracefully', async () => {
252
+ const runProbe = createMockRunProbe({
253
+ 'proxmox.cluster.status': { data: healthyClusterStatus },
254
+ 'proxmox.cluster.ha.status': { data: healthyHaStatus },
255
+ 'proxmox.vm.status': { status: 'error', error: 'VM/container 999 not found' },
256
+ 'proxmox.cluster.tasks': { data: emptyTasks },
257
+ });
258
+ const result = await handler({ vmid: 999 }, runProbe, defaultContext);
259
+ expect(result.findings).toHaveLength(1);
260
+ expect(result.findings[0]?.severity).toBe('critical');
261
+ expect(result.findings[0]?.title).toContain('not found');
262
+ });
263
+ it('agent available — calls agent probe when node matches', async () => {
264
+ const haNoResources = { ...healthyHaStatus, resources: [] };
265
+ const calls = [];
266
+ const runProbe = async (probe, _params, agent) => {
267
+ calls.push({ probe, agent });
268
+ const responses = {
269
+ 'proxmox.cluster.status': healthyClusterStatus,
270
+ 'proxmox.cluster.ha.status': haNoResources,
271
+ 'proxmox.vm.status': healthyVmStatus,
272
+ 'proxmox.cluster.tasks': emptyTasks,
273
+ 'proxmox.vm.config': healthyVmConfig,
274
+ 'proxmox.node.storage': healthyNodeStorage,
275
+ 'proxmox-node.local.lvm': { volumeGroups: [], warnings: [] },
276
+ };
277
+ return mockResult(probe, responses[probe]);
278
+ };
279
+ const ctx = { connectedAgents: ['pve01'] };
280
+ await handler({ vmid: 100 }, runProbe, ctx);
281
+ const agentCall = calls.find((c) => c.probe === 'proxmox-node.local.lvm');
282
+ expect(agentCall).toBeDefined();
283
+ expect(agentCall?.agent).toBe('pve01');
284
+ });
285
+ it('agent unavailable — skips agent probe silently', async () => {
286
+ const haNoResources = { ...healthyHaStatus, resources: [] };
287
+ const calls = [];
288
+ const runProbe = async (probe) => {
289
+ calls.push(probe);
290
+ const responses = {
291
+ 'proxmox.cluster.status': healthyClusterStatus,
292
+ 'proxmox.cluster.ha.status': haNoResources,
293
+ 'proxmox.vm.status': healthyVmStatus,
294
+ 'proxmox.cluster.tasks': emptyTasks,
295
+ 'proxmox.vm.config': healthyVmConfig,
296
+ 'proxmox.node.storage': healthyNodeStorage,
297
+ };
298
+ return mockResult(probe, responses[probe]);
299
+ };
300
+ await handler({ vmid: 100 }, runProbe, defaultContext);
301
+ expect(calls).not.toContain('proxmox-node.local.lvm');
302
+ });
303
+ it('VM stopped — warning finding', async () => {
304
+ const stoppedVm = { ...healthyVmStatus, status: 'stopped' };
305
+ const haNoResources = { ...healthyHaStatus, resources: [] };
306
+ const runProbe = createMockRunProbe({
307
+ 'proxmox.cluster.status': { data: healthyClusterStatus },
308
+ 'proxmox.cluster.ha.status': { data: haNoResources },
309
+ 'proxmox.vm.status': { data: stoppedVm },
310
+ 'proxmox.cluster.tasks': { data: emptyTasks },
311
+ 'proxmox.vm.config': { data: healthyVmConfig },
312
+ 'proxmox.node.storage': { data: healthyNodeStorage },
313
+ });
314
+ const result = await handler({ vmid: 100 }, runProbe, defaultContext);
315
+ const warning = result.findings.find((f) => f.title.includes('stopped'));
316
+ expect(warning).toBeDefined();
317
+ expect(warning?.severity).toBe('warning');
318
+ });
319
+ it('quorum lost — critical finding', async () => {
320
+ const noQuorum = { ...healthyClusterStatus, quorate: false };
321
+ const haNoResources = { ...healthyHaStatus, resources: [] };
322
+ const runProbe = createMockRunProbe({
323
+ 'proxmox.cluster.status': { data: noQuorum },
324
+ 'proxmox.cluster.ha.status': { data: haNoResources },
325
+ 'proxmox.vm.status': { data: healthyVmStatus },
326
+ 'proxmox.cluster.tasks': { data: emptyTasks },
327
+ 'proxmox.vm.config': { data: healthyVmConfig },
328
+ 'proxmox.node.storage': { data: healthyNodeStorage },
329
+ });
330
+ const result = await handler({ vmid: 100 }, runProbe, defaultContext);
331
+ const critical = result.findings.find((f) => f.title.includes('quorum'));
332
+ expect(critical).toBeDefined();
333
+ expect(critical?.severity).toBe('critical');
334
+ });
335
+ });
336
+ // =============================================================================
337
+ // proxmox-cluster-health tests
338
+ // =============================================================================
339
+ describe('proxmox-cluster-health', () => {
340
+ const handler = getHandler('proxmox-cluster');
341
+ const healthyNodesList = {
342
+ nodes: [
343
+ { node: 'pve01', status: 'online', cpu: 0.25, mem: 4e9, maxmem: 16e9 },
344
+ { node: 'pve02', status: 'online', cpu: 0.3, mem: 6e9, maxmem: 16e9 },
345
+ ],
346
+ warnings: [],
347
+ };
348
+ it('healthy cluster — no issues', async () => {
349
+ const runProbe = createMockRunProbe({
350
+ 'proxmox.cluster.status': { data: healthyClusterStatus },
351
+ 'proxmox.nodes.list': { data: healthyNodesList },
352
+ 'proxmox.cluster.ha.status': { data: { ...healthyHaStatus, resources: [] } },
353
+ 'proxmox.cluster.tasks': { data: emptyTasks },
354
+ 'proxmox.node.storage': { data: healthyNodeStorage },
355
+ });
356
+ const result = await handler({}, runProbe, defaultContext);
357
+ expect(result.category).toBe('proxmox-cluster');
358
+ // All findings should be info or no critical/warning
359
+ const issues = result.findings.filter((f) => f.severity !== 'info');
360
+ expect(issues).toHaveLength(0);
361
+ expect(result.summary.summaryText).toContain('2 nodes online');
362
+ });
363
+ it('offline node — critical finding', async () => {
364
+ const nodeWithOffline = {
365
+ nodes: [
366
+ { node: 'pve01', status: 'online', cpu: 0.25, mem: 4e9, maxmem: 16e9 },
367
+ { node: 'pve02', status: 'offline', cpu: 0, mem: 0, maxmem: 16e9 },
368
+ ],
369
+ warnings: [],
370
+ };
371
+ const runProbe = createMockRunProbe({
372
+ 'proxmox.cluster.status': { data: healthyClusterStatus },
373
+ 'proxmox.nodes.list': { data: nodeWithOffline },
374
+ 'proxmox.cluster.ha.status': { data: { ...healthyHaStatus, resources: [] } },
375
+ 'proxmox.cluster.tasks': { data: emptyTasks },
376
+ 'proxmox.node.storage': { data: healthyNodeStorage },
377
+ });
378
+ const result = await handler({}, runProbe, defaultContext);
379
+ const critical = result.findings.find((f) => f.title.includes('pve02') && f.title.includes('offline'));
380
+ expect(critical).toBeDefined();
381
+ expect(critical?.severity).toBe('critical');
382
+ });
383
+ it('high CPU — warning per node', async () => {
384
+ const highCpuNodes = {
385
+ nodes: [{ node: 'pve01', status: 'online', cpu: 0.95, mem: 4e9, maxmem: 16e9 }],
386
+ warnings: [],
387
+ };
388
+ const runProbe = createMockRunProbe({
389
+ 'proxmox.cluster.status': { data: healthyClusterStatus },
390
+ 'proxmox.nodes.list': { data: highCpuNodes },
391
+ 'proxmox.cluster.ha.status': { data: { ...healthyHaStatus, resources: [] } },
392
+ 'proxmox.cluster.tasks': { data: emptyTasks },
393
+ 'proxmox.node.storage': { data: healthyNodeStorage },
394
+ });
395
+ const result = await handler({}, runProbe, defaultContext);
396
+ const warning = result.findings.find((f) => f.title.includes('CPU') && f.title.includes('95%'));
397
+ expect(warning).toBeDefined();
398
+ expect(warning?.severity).toBe('warning');
399
+ });
400
+ it('high memory — warning per node', async () => {
401
+ const highMemNodes = {
402
+ nodes: [{ node: 'pve01', status: 'online', cpu: 0.2, mem: 15e9, maxmem: 16e9 }],
403
+ warnings: [],
404
+ };
405
+ const runProbe = createMockRunProbe({
406
+ 'proxmox.cluster.status': { data: healthyClusterStatus },
407
+ 'proxmox.nodes.list': { data: highMemNodes },
408
+ 'proxmox.cluster.ha.status': { data: { ...healthyHaStatus, resources: [] } },
409
+ 'proxmox.cluster.tasks': { data: emptyTasks },
410
+ 'proxmox.node.storage': { data: healthyNodeStorage },
411
+ });
412
+ const result = await handler({}, runProbe, defaultContext);
413
+ const warning = result.findings.find((f) => f.title.includes('memory') && f.title.includes('94%'));
414
+ expect(warning).toBeDefined();
415
+ expect(warning?.severity).toBe('warning');
416
+ });
417
+ it('HA resources in error — critical with remediation', async () => {
418
+ const haError = {
419
+ resources: [{ sid: 'vm:100', state: 'error', node: 'pve01' }],
420
+ warnings: [],
421
+ };
422
+ const runProbe = createMockRunProbe({
423
+ 'proxmox.cluster.status': { data: healthyClusterStatus },
424
+ 'proxmox.nodes.list': { data: healthyNodesList },
425
+ 'proxmox.cluster.ha.status': { data: haError },
426
+ 'proxmox.cluster.tasks': { data: emptyTasks },
427
+ 'proxmox.node.storage': { data: healthyNodeStorage },
428
+ });
429
+ const result = await handler({}, runProbe, defaultContext);
430
+ const critical = result.findings.find((f) => f.title.includes('vm:100'));
431
+ expect(critical).toBeDefined();
432
+ expect(critical?.severity).toBe('critical');
433
+ expect(critical?.remediation).toContain('ha-manager set vm:100 --state disabled');
434
+ });
435
+ it('storage > 85% — warning per storage', async () => {
436
+ const fullStorage = {
437
+ storages: [
438
+ {
439
+ storage: 'local',
440
+ type: 'dir',
441
+ total: 100e9,
442
+ used: 90e9,
443
+ avail: 10e9,
444
+ shared: false,
445
+ enabled: true,
446
+ active: true,
447
+ },
448
+ ],
449
+ warnings: [],
450
+ };
451
+ const runProbe = createMockRunProbe({
452
+ 'proxmox.cluster.status': { data: healthyClusterStatus },
453
+ 'proxmox.nodes.list': {
454
+ data: {
455
+ nodes: [{ node: 'pve01', status: 'online', cpu: 0.1, mem: 4e9, maxmem: 16e9 }],
456
+ warnings: [],
457
+ },
458
+ },
459
+ 'proxmox.cluster.ha.status': { data: { resources: [], warnings: [] } },
460
+ 'proxmox.cluster.tasks': { data: emptyTasks },
461
+ 'proxmox.node.storage': { data: fullStorage },
462
+ });
463
+ const result = await handler({}, runProbe, defaultContext);
464
+ const warning = result.findings.find((f) => f.title.includes('local') && f.title.includes('90%'));
465
+ expect(warning).toBeDefined();
466
+ expect(warning?.severity).toBe('warning');
467
+ });
468
+ it('failed tasks in last 24h — warning per task', async () => {
469
+ const now = Math.floor(Date.now() / 1000);
470
+ const recentFailedTasks = {
471
+ tasks: [
472
+ {
473
+ type: 'qmstart',
474
+ status: 'ERROR: failed',
475
+ node: 'pve01',
476
+ starttime: now - 3600,
477
+ endtime: now - 3500,
478
+ },
479
+ ],
480
+ warnings: [],
481
+ };
482
+ const runProbe = createMockRunProbe({
483
+ 'proxmox.cluster.status': { data: healthyClusterStatus },
484
+ 'proxmox.nodes.list': { data: healthyNodesList },
485
+ 'proxmox.cluster.ha.status': { data: { resources: [], warnings: [] } },
486
+ 'proxmox.cluster.tasks': { data: recentFailedTasks },
487
+ 'proxmox.node.storage': { data: healthyNodeStorage },
488
+ });
489
+ const result = await handler({}, runProbe, defaultContext);
490
+ const warning = result.findings.find((f) => f.title.includes('Failed task'));
491
+ expect(warning).toBeDefined();
492
+ expect(warning?.severity).toBe('warning');
493
+ });
494
+ it('per-node storage iteration — probes run for each node', async () => {
495
+ const calls = [];
496
+ const runProbe = async (probe, params) => {
497
+ const key = params?.node ? `${probe}:${params.node}` : probe;
498
+ calls.push(key);
499
+ const responses = {
500
+ 'proxmox.cluster.status': healthyClusterStatus,
501
+ 'proxmox.nodes.list': healthyNodesList,
502
+ 'proxmox.cluster.ha.status': { resources: [], warnings: [] },
503
+ 'proxmox.cluster.tasks': emptyTasks,
504
+ 'proxmox.node.storage': healthyNodeStorage,
505
+ };
506
+ return mockResult(probe, responses[probe]);
507
+ };
508
+ await handler({}, runProbe, defaultContext);
509
+ // Should have storage calls for both pve01 and pve02
510
+ expect(calls.filter((c) => c.startsWith('proxmox.node.storage'))).toHaveLength(2);
511
+ });
512
+ });
513
+ // =============================================================================
514
+ // proxmox-storage-audit tests
515
+ // =============================================================================
516
+ describe('proxmox-storage-audit', () => {
517
+ const handler = getHandler('proxmox-storage');
518
+ it('no HA VMs — info finding', async () => {
519
+ const noHa = { resources: [], warnings: [] };
520
+ const resources = {
521
+ resources: [{ vmid: 100, name: 'web-01', node: 'pve01', type: 'qemu', status: 'running' }],
522
+ };
523
+ const runProbe = createMockRunProbe({
524
+ 'proxmox.cluster.resources': { data: resources },
525
+ 'proxmox.cluster.ha.status': { data: noHa },
526
+ });
527
+ const result = await handler({}, runProbe, defaultContext);
528
+ expect(result.findings).toHaveLength(1);
529
+ expect(result.findings[0]?.severity).toBe('info');
530
+ expect(result.findings[0]?.title).toContain('No HA-managed VMs');
531
+ expect(result.summary.summaryText).toContain('0 HA-managed VMs');
532
+ });
533
+ it('all shared storage — info finding', async () => {
534
+ const resources = {
535
+ resources: [{ vmid: 100, name: 'web-01', node: 'pve01', type: 'qemu', status: 'running' }],
536
+ };
537
+ const haStatus = {
538
+ resources: [{ sid: 'vm:100', state: 'started', node: 'pve01' }],
539
+ warnings: [],
540
+ };
541
+ const sharedConfig = {
542
+ vmid: 100,
543
+ node: 'pve01',
544
+ config: {},
545
+ disks: [{ key: 'scsi0', storage: 'ceph-pool', format: 'raw', size: '32G' }],
546
+ warnings: [],
547
+ };
548
+ const runProbe = createMockRunProbe({
549
+ 'proxmox.cluster.resources': { data: resources },
550
+ 'proxmox.cluster.ha.status': { data: haStatus },
551
+ 'proxmox.vm.config': { data: sharedConfig },
552
+ 'proxmox.node.storage': { data: healthyNodeStorage },
553
+ });
554
+ const result = await handler({}, runProbe, defaultContext);
555
+ expect(result.findings).toHaveLength(1);
556
+ expect(result.findings[0]?.severity).toBe('info');
557
+ expect(result.findings[0]?.title).toContain('shared storage');
558
+ });
559
+ it('local storage on HA QEMU VM — critical with qm move-disk', async () => {
560
+ const resources = {
561
+ resources: [{ vmid: 100, name: 'web-01', node: 'pve01', type: 'qemu', status: 'running' }],
562
+ };
563
+ const haStatus = {
564
+ resources: [{ sid: 'vm:100', state: 'started', node: 'pve01' }],
565
+ warnings: [],
566
+ };
567
+ const localConfig = {
568
+ vmid: 100,
569
+ node: 'pve01',
570
+ config: {},
571
+ disks: [{ key: 'scsi0', storage: 'local-lvm', format: 'raw', size: '32G' }],
572
+ warnings: [],
573
+ };
574
+ const runProbe = createMockRunProbe({
575
+ 'proxmox.cluster.resources': { data: resources },
576
+ 'proxmox.cluster.ha.status': { data: haStatus },
577
+ 'proxmox.vm.config': { data: localConfig },
578
+ 'proxmox.node.storage': { data: healthyNodeStorage },
579
+ });
580
+ const result = await handler({}, runProbe, defaultContext);
581
+ const critical = result.findings.find((f) => f.severity === 'critical');
582
+ expect(critical).toBeDefined();
583
+ expect(critical?.title).toContain('local storage');
584
+ expect(critical?.remediation).toContain('qm move-disk 100 scsi0 ceph-pool --delete');
585
+ });
586
+ it('local storage on HA LXC container — critical (checks rootfs)', async () => {
587
+ const resources = {
588
+ resources: [{ vmid: 200, name: 'ct-db', node: 'pve01', type: 'lxc', status: 'running' }],
589
+ };
590
+ const haStatus = {
591
+ resources: [{ sid: 'ct:200', state: 'started', node: 'pve01' }],
592
+ warnings: [],
593
+ };
594
+ const lxcConfig = {
595
+ vmid: 200,
596
+ node: 'pve01',
597
+ config: {},
598
+ rootfs: { storage: 'local-lvm', size: '8G' },
599
+ mountpoints: [{ key: 'mp0', storage: 'local-lvm', mountpoint: '/data', size: '20G' }],
600
+ warnings: [],
601
+ };
602
+ const runProbe = createMockRunProbe({
603
+ 'proxmox.cluster.resources': { data: resources },
604
+ 'proxmox.cluster.ha.status': { data: haStatus },
605
+ 'proxmox.lxc.config': { data: lxcConfig },
606
+ 'proxmox.node.storage': { data: healthyNodeStorage },
607
+ });
608
+ const result = await handler({}, runProbe, defaultContext);
609
+ const criticals = result.findings.filter((f) => f.severity === 'critical');
610
+ // Should flag both rootfs and mp0
611
+ expect(criticals.length).toBeGreaterThanOrEqual(2);
612
+ expect(criticals.some((f) => f.title.includes('rootfs'))).toBe(true);
613
+ expect(criticals.some((f) => f.title.includes('mp0'))).toBe(true);
614
+ });
615
+ it('mixed — only flags local ones', async () => {
616
+ const resources = {
617
+ resources: [
618
+ { vmid: 100, name: 'shared-vm', node: 'pve01', type: 'qemu', status: 'running' },
619
+ { vmid: 101, name: 'local-vm', node: 'pve01', type: 'qemu', status: 'running' },
620
+ ],
621
+ };
622
+ const haStatus = {
623
+ resources: [
624
+ { sid: 'vm:100', state: 'started', node: 'pve01' },
625
+ { sid: 'vm:101', state: 'started', node: 'pve01' },
626
+ ],
627
+ warnings: [],
628
+ };
629
+ const calls = [];
630
+ const runProbe = async (probe, params) => {
631
+ calls.push({ probe, params });
632
+ if (probe === 'proxmox.cluster.resources')
633
+ return mockResult(probe, resources);
634
+ if (probe === 'proxmox.cluster.ha.status')
635
+ return mockResult(probe, haStatus);
636
+ if (probe === 'proxmox.node.storage')
637
+ return mockResult(probe, healthyNodeStorage);
638
+ if (probe === 'proxmox.vm.config') {
639
+ const vmid = params?.vmid;
640
+ if (vmid === 100) {
641
+ return mockResult(probe, {
642
+ disks: [{ key: 'scsi0', storage: 'ceph-pool', format: 'raw', size: '32G' }],
643
+ });
644
+ }
645
+ return mockResult(probe, {
646
+ disks: [{ key: 'scsi0', storage: 'local-lvm', format: 'raw', size: '32G' }],
647
+ });
648
+ }
649
+ return mockResult(probe, undefined, 'error', 'Unknown probe');
650
+ };
651
+ const result = await handler({}, runProbe, defaultContext);
652
+ const criticals = result.findings.filter((f) => f.severity === 'critical');
653
+ expect(criticals).toHaveLength(1);
654
+ expect(criticals[0]?.title).toContain('101');
655
+ });
656
+ it('suggests shared storage name from available storages', async () => {
657
+ const resources = {
658
+ resources: [{ vmid: 100, name: 'web-01', node: 'pve01', type: 'qemu', status: 'running' }],
659
+ };
660
+ const haStatus = {
661
+ resources: [{ sid: 'vm:100', state: 'started', node: 'pve01' }],
662
+ warnings: [],
663
+ };
664
+ const localConfig = {
665
+ disks: [{ key: 'scsi0', storage: 'local-lvm', format: 'raw', size: '32G' }],
666
+ };
667
+ const runProbe = createMockRunProbe({
668
+ 'proxmox.cluster.resources': { data: resources },
669
+ 'proxmox.cluster.ha.status': { data: haStatus },
670
+ 'proxmox.vm.config': { data: localConfig },
671
+ 'proxmox.node.storage': { data: healthyNodeStorage },
672
+ });
673
+ const result = await handler({}, runProbe, defaultContext);
674
+ const critical = result.findings.find((f) => f.severity === 'critical');
675
+ expect(critical?.remediation).toContain('ceph-pool');
676
+ });
677
+ });
678
+ // =============================================================================
679
+ // Definitions
680
+ // =============================================================================
681
+ describe('proxmoxDiagnosticRunbooks definitions', () => {
682
+ it('exports 3 runbook definitions', () => {
683
+ expect(proxmoxDiagnosticRunbooks).toHaveLength(3);
684
+ });
685
+ it('has correct categories', () => {
686
+ const categories = proxmoxDiagnosticRunbooks.map((r) => r.category);
687
+ expect(categories).toContain('proxmox-vm');
688
+ expect(categories).toContain('proxmox-cluster');
689
+ expect(categories).toContain('proxmox-storage');
690
+ });
691
+ it('proxmox-vm requires vmid param', () => {
692
+ const def = proxmoxDiagnosticRunbooks.find((r) => r.category === 'proxmox-vm');
693
+ expect(def?.params?.vmid?.required).toBe(true);
694
+ });
695
+ it('proxmox-storage has no required params', () => {
696
+ const def = proxmoxDiagnosticRunbooks.find((r) => r.category === 'proxmox-storage');
697
+ expect(def?.params).toBeUndefined();
698
+ });
699
+ });
700
+ //# sourceMappingURL=proxmox.test.js.map