@atlascrew/apparatus 0.9.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 (291) hide show
  1. package/bin/apparatus.mjs +2 -0
  2. package/certs/server.crt +17 -0
  3. package/certs/server.key +28 -0
  4. package/dist/ai/client.js +104 -0
  5. package/dist/ai/client.js.map +1 -0
  6. package/dist/ai/personas.js +104 -0
  7. package/dist/ai/personas.js.map +1 -0
  8. package/dist/ai/redteam.js +1404 -0
  9. package/dist/ai/redteam.js.map +1 -0
  10. package/dist/ai/report-store.js +309 -0
  11. package/dist/ai/report-store.js.map +1 -0
  12. package/dist/app.js +525 -0
  13. package/dist/app.js.map +1 -0
  14. package/dist/attack-sim.js +69 -0
  15. package/dist/attack-sim.js.map +1 -0
  16. package/dist/attacker-tracker.js +276 -0
  17. package/dist/attacker-tracker.js.map +1 -0
  18. package/dist/blackhole.js +95 -0
  19. package/dist/blackhole.js.map +1 -0
  20. package/dist/chaos.js +88 -0
  21. package/dist/chaos.js.map +1 -0
  22. package/dist/cluster.js +462 -0
  23. package/dist/cluster.js.map +1 -0
  24. package/dist/config.js +61 -0
  25. package/dist/config.js.map +1 -0
  26. package/dist/deception.js +205 -0
  27. package/dist/deception.js.map +1 -0
  28. package/dist/demo-mode.js +109 -0
  29. package/dist/demo-mode.js.map +1 -0
  30. package/dist/dist-dashboard/assets/index-BsMhEnGu.js +648 -0
  31. package/dist/dist-dashboard/assets/index-CNOkYC_Q.css +10 -0
  32. package/dist/dist-dashboard/assets/index-CW2grvPC.js +648 -0
  33. package/dist/dist-dashboard/assets/logo/apparatus-favicon.svg +15 -0
  34. package/dist/dist-dashboard/assets/logo/apparatus-icon-dark.svg +24 -0
  35. package/dist/dist-dashboard/assets/logo/apparatus-icon-light.svg +24 -0
  36. package/dist/dist-dashboard/assets/logo/apparatus-logo-512.png +0 -0
  37. package/dist/dist-dashboard/assets/logo/apparatus-logo-dark.svg +18 -0
  38. package/dist/dist-dashboard/assets/logo/apparatus-logo.svg +17 -0
  39. package/dist/dist-dashboard/assets/logo/apple-touch-icon.png +0 -0
  40. package/dist/dist-dashboard/assets/logo/favicon-192.png +0 -0
  41. package/dist/dist-dashboard/assets/logo/favicon-32.png +0 -0
  42. package/dist/dist-dashboard/assets/logo/favicon.ico +0 -0
  43. package/dist/dist-dashboard/assets/logo/icon-192.png +0 -0
  44. package/dist/dist-dashboard/assets/logo/icon-512.png +0 -0
  45. package/dist/dist-dashboard/assets/logo/icon-light-512.png +0 -0
  46. package/dist/dist-dashboard/assets/react-vendor-DpRMSntD.js +1 -0
  47. package/dist/dist-dashboard/assets/router-DSc5pRwN.js +59 -0
  48. package/dist/dist-dashboard/docs-index.json +1577 -0
  49. package/dist/dist-dashboard/index.html +21 -0
  50. package/dist/dlp.js +40 -0
  51. package/dist/dlp.js.map +1 -0
  52. package/dist/drills.js +770 -0
  53. package/dist/drills.js.map +1 -0
  54. package/dist/echoHandler.js +113 -0
  55. package/dist/echoHandler.js.map +1 -0
  56. package/dist/escape/index.js +225 -0
  57. package/dist/escape/index.js.map +1 -0
  58. package/dist/escape/methods/dns.js +74 -0
  59. package/dist/escape/methods/dns.js.map +1 -0
  60. package/dist/escape/methods/http.js +81 -0
  61. package/dist/escape/methods/http.js.map +1 -0
  62. package/dist/escape/methods/icmp.js +36 -0
  63. package/dist/escape/methods/icmp.js.map +1 -0
  64. package/dist/escape/methods/tcp.js +38 -0
  65. package/dist/escape/methods/tcp.js.map +1 -0
  66. package/dist/escape/methods/udp.js +27 -0
  67. package/dist/escape/methods/udp.js.map +1 -0
  68. package/dist/escape/methods/websocket.js +37 -0
  69. package/dist/escape/methods/websocket.js.map +1 -0
  70. package/dist/forensics.js +111 -0
  71. package/dist/forensics.js.map +1 -0
  72. package/dist/generator.js +67 -0
  73. package/dist/generator.js.map +1 -0
  74. package/dist/ghosting.js +414 -0
  75. package/dist/ghosting.js.map +1 -0
  76. package/dist/graphql.js +44 -0
  77. package/dist/graphql.js.map +1 -0
  78. package/dist/history.js +40 -0
  79. package/dist/history.js.map +1 -0
  80. package/dist/imposter/creds.js +16 -0
  81. package/dist/imposter/creds.js.map +1 -0
  82. package/dist/imposter/index.js +44 -0
  83. package/dist/imposter/index.js.map +1 -0
  84. package/dist/imposter/providers/aws.js +103 -0
  85. package/dist/imposter/providers/aws.js.map +1 -0
  86. package/dist/imposter/providers/gcp.js +26 -0
  87. package/dist/imposter/providers/gcp.js.map +1 -0
  88. package/dist/index.js +53 -0
  89. package/dist/index.js.map +1 -0
  90. package/dist/infra-debug.js +68 -0
  91. package/dist/infra-debug.js.map +1 -0
  92. package/dist/jwt-debug.js +272 -0
  93. package/dist/jwt-debug.js.map +1 -0
  94. package/dist/kv.js +22 -0
  95. package/dist/kv.js.map +1 -0
  96. package/dist/lib/generators.js +43 -0
  97. package/dist/lib/generators.js.map +1 -0
  98. package/dist/lib/json.js +26 -0
  99. package/dist/lib/json.js.map +1 -0
  100. package/dist/logger.js +9 -0
  101. package/dist/logger.js.map +1 -0
  102. package/dist/metrics.js +20 -0
  103. package/dist/metrics.js.map +1 -0
  104. package/dist/mtd.js +30 -0
  105. package/dist/mtd.js.map +1 -0
  106. package/dist/oidc.js +69 -0
  107. package/dist/oidc.js.map +1 -0
  108. package/dist/persistence/cluster-state.js +47 -0
  109. package/dist/persistence/cluster-state.js.map +1 -0
  110. package/dist/persistence/deception-history.js +65 -0
  111. package/dist/persistence/deception-history.js.map +1 -0
  112. package/dist/persistence/drill-runs.js +138 -0
  113. package/dist/persistence/drill-runs.js.map +1 -0
  114. package/dist/persistence/request-history.js +41 -0
  115. package/dist/persistence/request-history.js.map +1 -0
  116. package/dist/persistence/scenario-catalog.js +73 -0
  117. package/dist/persistence/scenario-catalog.js.map +1 -0
  118. package/dist/persistence/status.js +51 -0
  119. package/dist/persistence/status.js.map +1 -0
  120. package/dist/persistence/tarpit-state.js +47 -0
  121. package/dist/persistence/tarpit-state.js.map +1 -0
  122. package/dist/persistence/webhook-store.js +69 -0
  123. package/dist/persistence/webhook-store.js.map +1 -0
  124. package/dist/proxy.js +28 -0
  125. package/dist/proxy.js.map +1 -0
  126. package/dist/ratelimit.js +32 -0
  127. package/dist/ratelimit.js.map +1 -0
  128. package/dist/redteam.js +442 -0
  129. package/dist/redteam.js.map +1 -0
  130. package/dist/scenarios.js +229 -0
  131. package/dist/scenarios.js.map +1 -0
  132. package/dist/scripting.js +30 -0
  133. package/dist/scripting.js.map +1 -0
  134. package/dist/self-healing.js +42 -0
  135. package/dist/self-healing.js.map +1 -0
  136. package/dist/sentinel.js +50 -0
  137. package/dist/sentinel.js.map +1 -0
  138. package/dist/server-bad-ssl.js +47 -0
  139. package/dist/server-bad-ssl.js.map +1 -0
  140. package/dist/server-grpc.js +66 -0
  141. package/dist/server-grpc.js.map +1 -0
  142. package/dist/server-http1.js +5 -0
  143. package/dist/server-http1.js.map +1 -0
  144. package/dist/server-http2.js +27 -0
  145. package/dist/server-http2.js.map +1 -0
  146. package/dist/server-icap.js +46 -0
  147. package/dist/server-icap.js.map +1 -0
  148. package/dist/server-l4.js +30 -0
  149. package/dist/server-l4.js.map +1 -0
  150. package/dist/server-mqtt.js +29 -0
  151. package/dist/server-mqtt.js.map +1 -0
  152. package/dist/server-protocols.js +18 -0
  153. package/dist/server-protocols.js.map +1 -0
  154. package/dist/server-redis.js +112 -0
  155. package/dist/server-redis.js.map +1 -0
  156. package/dist/server-smtp.js +66 -0
  157. package/dist/server-smtp.js.map +1 -0
  158. package/dist/server-syslog.js +23 -0
  159. package/dist/server-syslog.js.map +1 -0
  160. package/dist/server-ws.js +18 -0
  161. package/dist/server-ws.js.map +1 -0
  162. package/dist/sidecar/chaos/engine.js +41 -0
  163. package/dist/sidecar/chaos/engine.js.map +1 -0
  164. package/dist/sidecar/index.js +98 -0
  165. package/dist/sidecar/index.js.map +1 -0
  166. package/dist/simulator/dependency-graph.js +102 -0
  167. package/dist/simulator/dependency-graph.js.map +1 -0
  168. package/dist/simulator/supply-chain.js +67 -0
  169. package/dist/simulator/supply-chain.js.map +1 -0
  170. package/dist/sink.js +24 -0
  171. package/dist/sink.js.map +1 -0
  172. package/dist/sse-broadcast.js +105 -0
  173. package/dist/sse-broadcast.js.map +1 -0
  174. package/dist/swagger.js +309 -0
  175. package/dist/swagger.js.map +1 -0
  176. package/dist/sysinfo.js +36 -0
  177. package/dist/sysinfo.js.map +1 -0
  178. package/dist/tarpit.js +126 -0
  179. package/dist/tarpit.js.map +1 -0
  180. package/dist/tool-executor.js +315 -0
  181. package/dist/tool-executor.js.map +1 -0
  182. package/dist/tui/api-client.js +341 -0
  183. package/dist/tui/api-client.js.map +1 -0
  184. package/dist/tui/core/action-handler.js +302 -0
  185. package/dist/tui/core/action-handler.js.map +1 -0
  186. package/dist/tui/core/index.js +18 -0
  187. package/dist/tui/core/index.js.map +1 -0
  188. package/dist/tui/core/keyboard.js +329 -0
  189. package/dist/tui/core/keyboard.js.map +1 -0
  190. package/dist/tui/core/modal.js +397 -0
  191. package/dist/tui/core/modal.js.map +1 -0
  192. package/dist/tui/core/screen-manager.js +262 -0
  193. package/dist/tui/core/screen-manager.js.map +1 -0
  194. package/dist/tui/core/store.js +254 -0
  195. package/dist/tui/core/store.js.map +1 -0
  196. package/dist/tui/core/widget.js +167 -0
  197. package/dist/tui/core/widget.js.map +1 -0
  198. package/dist/tui/dashboard.js +649 -0
  199. package/dist/tui/dashboard.js.map +1 -0
  200. package/dist/tui/index.js +118 -0
  201. package/dist/tui/index.js.map +1 -0
  202. package/dist/tui/modals/add-rule-modal.js +190 -0
  203. package/dist/tui/modals/add-rule-modal.js.map +1 -0
  204. package/dist/tui/modals/dlp-output-modal.js +102 -0
  205. package/dist/tui/modals/dlp-output-modal.js.map +1 -0
  206. package/dist/tui/modals/dns-form-modal.js +26 -0
  207. package/dist/tui/modals/dns-form-modal.js.map +1 -0
  208. package/dist/tui/modals/ghost-config-modal.js +35 -0
  209. package/dist/tui/modals/ghost-config-modal.js.map +1 -0
  210. package/dist/tui/modals/har-results-modal.js +41 -0
  211. package/dist/tui/modals/har-results-modal.js.map +1 -0
  212. package/dist/tui/modals/index.js +15 -0
  213. package/dist/tui/modals/index.js.map +1 -0
  214. package/dist/tui/modals/jwt-decode-modal.js +45 -0
  215. package/dist/tui/modals/jwt-decode-modal.js.map +1 -0
  216. package/dist/tui/modals/jwt-mint-modal.js +70 -0
  217. package/dist/tui/modals/jwt-mint-modal.js.map +1 -0
  218. package/dist/tui/modals/ping-form-modal.js +19 -0
  219. package/dist/tui/modals/ping-form-modal.js.map +1 -0
  220. package/dist/tui/modals/redteam-results-modal.js +43 -0
  221. package/dist/tui/modals/redteam-results-modal.js.map +1 -0
  222. package/dist/tui/modals/scan-form-modal.js +26 -0
  223. package/dist/tui/modals/scan-form-modal.js.map +1 -0
  224. package/dist/tui/screens/defense-screen.js +281 -0
  225. package/dist/tui/screens/defense-screen.js.map +1 -0
  226. package/dist/tui/screens/forensics-screen.js +81 -0
  227. package/dist/tui/screens/forensics-screen.js.map +1 -0
  228. package/dist/tui/screens/index.js +140 -0
  229. package/dist/tui/screens/index.js.map +1 -0
  230. package/dist/tui/screens/system-screen.js +81 -0
  231. package/dist/tui/screens/system-screen.js.map +1 -0
  232. package/dist/tui/screens/testing-screen.js +429 -0
  233. package/dist/tui/screens/testing-screen.js.map +1 -0
  234. package/dist/tui/screens/traffic-screen.js +76 -0
  235. package/dist/tui/screens/traffic-screen.js.map +1 -0
  236. package/dist/tui/sse-client.js +130 -0
  237. package/dist/tui/sse-client.js.map +1 -0
  238. package/dist/tui/state/metrics-buffer.js +195 -0
  239. package/dist/tui/state/metrics-buffer.js.map +1 -0
  240. package/dist/tui/state/metrics-buffer.test.js +102 -0
  241. package/dist/tui/state/metrics-buffer.test.js.map +1 -0
  242. package/dist/tui/theme.js +136 -0
  243. package/dist/tui/theme.js.map +1 -0
  244. package/dist/tui/types.js +6 -0
  245. package/dist/tui/types.js.map +1 -0
  246. package/dist/tui/widgets/chaos-widget.js +152 -0
  247. package/dist/tui/widgets/chaos-widget.js.map +1 -0
  248. package/dist/tui/widgets/cluster-widget.js +156 -0
  249. package/dist/tui/widgets/cluster-widget.js.map +1 -0
  250. package/dist/tui/widgets/dlp-widget.js +161 -0
  251. package/dist/tui/widgets/dlp-widget.js.map +1 -0
  252. package/dist/tui/widgets/ghost-widget.js +169 -0
  253. package/dist/tui/widgets/ghost-widget.js.map +1 -0
  254. package/dist/tui/widgets/har-widget.js +173 -0
  255. package/dist/tui/widgets/har-widget.js.map +1 -0
  256. package/dist/tui/widgets/index.js +122 -0
  257. package/dist/tui/widgets/index.js.map +1 -0
  258. package/dist/tui/widgets/jwt-widget.js +177 -0
  259. package/dist/tui/widgets/jwt-widget.js.map +1 -0
  260. package/dist/tui/widgets/kv-widget.js +261 -0
  261. package/dist/tui/widgets/kv-widget.js.map +1 -0
  262. package/dist/tui/widgets/mtd-widget.js +181 -0
  263. package/dist/tui/widgets/mtd-widget.js.map +1 -0
  264. package/dist/tui/widgets/netdiag-widget.js +155 -0
  265. package/dist/tui/widgets/netdiag-widget.js.map +1 -0
  266. package/dist/tui/widgets/oidc-widget.js +162 -0
  267. package/dist/tui/widgets/oidc-widget.js.map +1 -0
  268. package/dist/tui/widgets/pcap-widget.js +239 -0
  269. package/dist/tui/widgets/pcap-widget.js.map +1 -0
  270. package/dist/tui/widgets/redteam-widget.js +155 -0
  271. package/dist/tui/widgets/redteam-widget.js.map +1 -0
  272. package/dist/tui/widgets/rps-gauge-widget.js +124 -0
  273. package/dist/tui/widgets/rps-gauge-widget.js.map +1 -0
  274. package/dist/tui/widgets/sentinel-widget.js +171 -0
  275. package/dist/tui/widgets/sentinel-widget.js.map +1 -0
  276. package/dist/tui/widgets/sparklines-widget.js +127 -0
  277. package/dist/tui/widgets/sparklines-widget.js.map +1 -0
  278. package/dist/tui/widgets/sysinfo-widget.js +197 -0
  279. package/dist/tui/widgets/sysinfo-widget.js.map +1 -0
  280. package/dist/tui/widgets/traffic-chart-widget.js +170 -0
  281. package/dist/tui/widgets/traffic-chart-widget.js.map +1 -0
  282. package/dist/tui/widgets/webhook-widget.js +259 -0
  283. package/dist/tui/widgets/webhook-widget.js.map +1 -0
  284. package/dist/utils/ip.js +18 -0
  285. package/dist/utils/ip.js.map +1 -0
  286. package/dist/victim/index.js +71 -0
  287. package/dist/victim/index.js.map +1 -0
  288. package/dist/webhook.js +88 -0
  289. package/dist/webhook.js.map +1 -0
  290. package/package.json +90 -0
  291. package/proto/echo.proto +19 -0
@@ -0,0 +1,124 @@
1
+ /**
2
+ * RPS Gauge Widget
3
+ *
4
+ * Displays requests per second using a blessed-contrib gauge
5
+ * with color-coded thresholds (green/yellow/red)
6
+ */
7
+ import contrib from 'blessed-contrib';
8
+ import { BaseWidget } from '../core/widget.js';
9
+ /**
10
+ * RPS Gauge Widget showing real-time requests per second
11
+ */
12
+ export class RPSGaugeWidget extends BaseWidget {
13
+ gauge;
14
+ thresholds;
15
+ rpsHistory = [];
16
+ lastRequestCount = 0;
17
+ lastUpdateTime = Date.now();
18
+ element;
19
+ constructor(options) {
20
+ super({
21
+ id: options.id,
22
+ config: options.config,
23
+ subscriptions: ['requestCount'],
24
+ });
25
+ this.thresholds = options.thresholds ?? {
26
+ warning: 20,
27
+ critical: 50,
28
+ };
29
+ // Create the gauge element
30
+ // Note: blessed-contrib types are incomplete, using type casts through unknown
31
+ this.gauge = contrib.gauge({
32
+ label: options.config.label ?? ' RPS ',
33
+ stroke: 'green',
34
+ fill: 'white',
35
+ border: {
36
+ type: 'line',
37
+ fg: options.config.style?.border?.fg ?? 'cyan',
38
+ },
39
+ style: {
40
+ fg: options.config.style?.fg ?? 'white',
41
+ bg: options.config.style?.bg,
42
+ },
43
+ tags: true,
44
+ showLabel: true,
45
+ });
46
+ this.element = this.gauge;
47
+ }
48
+ render(state) {
49
+ if (!this.mounted)
50
+ return;
51
+ // Calculate RPS from request count delta
52
+ const now = Date.now();
53
+ const timeDelta = (now - this.lastUpdateTime) / 1000; // seconds
54
+ const requestDelta = state.requestCount - this.lastRequestCount;
55
+ let currentRPS = 0;
56
+ if (timeDelta > 0 && requestDelta >= 0) {
57
+ currentRPS = requestDelta / timeDelta;
58
+ }
59
+ // Update history
60
+ this.rpsHistory.push(currentRPS);
61
+ if (this.rpsHistory.length > 10) {
62
+ this.rpsHistory.shift();
63
+ }
64
+ // Calculate smoothed RPS (average of last 3 samples)
65
+ const recentSamples = this.rpsHistory.slice(-3);
66
+ const smoothedRPS = recentSamples.reduce((sum, val) => sum + val, 0) / recentSamples.length;
67
+ // Update last values
68
+ this.lastRequestCount = state.requestCount;
69
+ this.lastUpdateTime = now;
70
+ // Determine color based on thresholds
71
+ let stroke = 'green';
72
+ if (smoothedRPS >= this.thresholds.critical) {
73
+ stroke = 'red';
74
+ }
75
+ else if (smoothedRPS >= this.thresholds.warning) {
76
+ stroke = 'yellow';
77
+ }
78
+ // Calculate percentage (0-100 based on critical threshold as max)
79
+ const maxValue = this.thresholds.critical * 2; // Show up to 2x critical
80
+ const percent = Math.min(100, (smoothedRPS / maxValue) * 100);
81
+ // Update gauge - cast to any due to incomplete blessed-contrib types
82
+ this.gauge.setData([
83
+ {
84
+ percent: Math.round(percent),
85
+ stroke,
86
+ },
87
+ ]);
88
+ // Update label with current RPS
89
+ this.gauge.setLabel(` RPS: ${smoothedRPS.toFixed(1)} `);
90
+ this.requestRender();
91
+ }
92
+ /**
93
+ * Get current RPS value
94
+ */
95
+ getCurrentRPS() {
96
+ if (this.rpsHistory.length === 0)
97
+ return 0;
98
+ const recent = this.rpsHistory.slice(-3);
99
+ return recent.reduce((sum, val) => sum + val, 0) / recent.length;
100
+ }
101
+ /**
102
+ * Reset the gauge
103
+ */
104
+ reset() {
105
+ this.rpsHistory = [];
106
+ this.lastRequestCount = 0;
107
+ this.lastUpdateTime = Date.now();
108
+ this.gauge.setData([{ percent: 0, stroke: 'green' }]);
109
+ }
110
+ onUnmount() {
111
+ this.reset();
112
+ }
113
+ }
114
+ /**
115
+ * Create RPS gauge widget
116
+ */
117
+ export function createRPSGaugeWidget(id, config, options) {
118
+ return new RPSGaugeWidget({
119
+ id,
120
+ config,
121
+ thresholds: options?.thresholds,
122
+ });
123
+ }
124
+ //# sourceMappingURL=rps-gauge-widget.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"rps-gauge-widget.js","sourceRoot":"","sources":["../../../src/tui/widgets/rps-gauge-widget.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAGH,OAAO,OAAO,MAAM,iBAAiB,CAAC;AACtC,OAAO,EAAE,UAAU,EAAqB,MAAM,mBAAmB,CAAC;AAYlE;;GAEG;AACH,MAAM,OAAO,cAAe,SAAQ,UAA0B;IACpD,KAAK,CAA+B;IAC3B,UAAU,CAAwC;IAC3D,UAAU,GAAa,EAAE,CAAC;IAC1B,gBAAgB,GAAG,CAAC,CAAC;IACrB,cAAc,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;IAE3B,OAAO,CAA6B;IAE7C,YAAY,OAAwB;QAClC,KAAK,CAAC;YACJ,EAAE,EAAE,OAAO,CAAC,EAAE;YACd,MAAM,EAAE,OAAO,CAAC,MAAM;YACtB,aAAa,EAAE,CAAC,cAAc,CAAC;SAChC,CAAC,CAAC;QAEH,IAAI,CAAC,UAAU,GAAG,OAAO,CAAC,UAAU,IAAI;YACtC,OAAO,EAAE,EAAE;YACX,QAAQ,EAAE,EAAE;SACb,CAAC;QAEF,2BAA2B;QAC3B,+EAA+E;QAC/E,IAAI,CAAC,KAAK,GAAG,OAAO,CAAC,KAAK,CAAC;YACzB,KAAK,EAAE,OAAO,CAAC,MAAM,CAAC,KAAK,IAAI,OAAO;YACtC,MAAM,EAAE,OAAO;YACf,IAAI,EAAE,OAAO;YACb,MAAM,EAAE;gBACN,IAAI,EAAE,MAAM;gBACZ,EAAE,EAAE,OAAO,CAAC,MAAM,CAAC,KAAK,EAAE,MAAM,EAAE,EAAE,IAAI,MAAM;aACV;YACtC,KAAK,EAAE;gBACL,EAAE,EAAE,OAAO,CAAC,MAAM,CAAC,KAAK,EAAE,EAAE,IAAI,OAAO;gBACvC,EAAE,EAAE,OAAO,CAAC,MAAM,CAAC,KAAK,EAAE,EAAE;aACS;YACvC,IAAI,EAAE,IAAI;YACV,SAAS,EAAE,IAAI;SACgB,CAAC,CAAC;QAEnC,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,KAA8C,CAAC;IACrE,CAAC;IAED,MAAM,CAAC,KAAqB;QAC1B,IAAI,CAAC,IAAI,CAAC,OAAO;YAAE,OAAO;QAE1B,yCAAyC;QACzC,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QACvB,MAAM,SAAS,GAAG,CAAC,GAAG,GAAG,IAAI,CAAC,cAAc,CAAC,GAAG,IAAI,CAAC,CAAC,UAAU;QAChE,MAAM,YAAY,GAAG,KAAK,CAAC,YAAY,GAAG,IAAI,CAAC,gBAAgB,CAAC;QAEhE,IAAI,UAAU,GAAG,CAAC,CAAC;QACnB,IAAI,SAAS,GAAG,CAAC,IAAI,YAAY,IAAI,CAAC,EAAE,CAAC;YACvC,UAAU,GAAG,YAAY,GAAG,SAAS,CAAC;QACxC,CAAC;QAED,iBAAiB;QACjB,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QACjC,IAAI,IAAI,CAAC,UAAU,CAAC,MAAM,GAAG,EAAE,EAAE,CAAC;YAChC,IAAI,CAAC,UAAU,CAAC,KAAK,EAAE,CAAC;QAC1B,CAAC;QAED,qDAAqD;QACrD,MAAM,aAAa,GAAG,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC;QAChD,MAAM,WAAW,GACf,aAAa,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,GAAG,EAAE,EAAE,CAAC,GAAG,GAAG,GAAG,EAAE,CAAC,CAAC,GAAG,aAAa,CAAC,MAAM,CAAC;QAE1E,qBAAqB;QACrB,IAAI,CAAC,gBAAgB,GAAG,KAAK,CAAC,YAAY,CAAC;QAC3C,IAAI,CAAC,cAAc,GAAG,GAAG,CAAC;QAE1B,sCAAsC;QACtC,IAAI,MAAM,GAAG,OAAO,CAAC;QACrB,IAAI,WAAW,IAAI,IAAI,CAAC,UAAU,CAAC,QAAQ,EAAE,CAAC;YAC5C,MAAM,GAAG,KAAK,CAAC;QACjB,CAAC;aAAM,IAAI,WAAW,IAAI,IAAI,CAAC,UAAU,CAAC,OAAO,EAAE,CAAC;YAClD,MAAM,GAAG,QAAQ,CAAC;QACpB,CAAC;QAED,kEAAkE;QAClE,MAAM,QAAQ,GAAG,IAAI,CAAC,UAAU,CAAC,QAAQ,GAAG,CAAC,CAAC,CAAC,yBAAyB;QACxE,MAAM,OAAO,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,WAAW,GAAG,QAAQ,CAAC,GAAG,GAAG,CAAC,CAAC;QAE9D,qEAAqE;QACpE,IAAI,CAAC,KAAyD,CAAC,OAAO,CAAC;YACtE;gBACE,OAAO,EAAE,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC;gBAC5B,MAAM;aACP;SACF,CAAC,CAAC;QAEH,gCAAgC;QAChC,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,SAAS,WAAW,CAAC,OAAO,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC;QAExD,IAAI,CAAC,aAAa,EAAE,CAAC;IACvB,CAAC;IAED;;OAEG;IACH,aAAa;QACX,IAAI,IAAI,CAAC,UAAU,CAAC,MAAM,KAAK,CAAC;YAAE,OAAO,CAAC,CAAC;QAC3C,MAAM,MAAM,GAAG,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC;QACzC,OAAO,MAAM,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,GAAG,EAAE,EAAE,CAAC,GAAG,GAAG,GAAG,EAAE,CAAC,CAAC,GAAG,MAAM,CAAC,MAAM,CAAC;IACnE,CAAC;IAED;;OAEG;IACH,KAAK;QACH,IAAI,CAAC,UAAU,GAAG,EAAE,CAAC;QACrB,IAAI,CAAC,gBAAgB,GAAG,CAAC,CAAC;QAC1B,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QAChC,IAAI,CAAC,KAAyD,CAAC,OAAO,CAAC,CAAC,EAAE,OAAO,EAAE,CAAC,EAAE,MAAM,EAAE,OAAO,EAAE,CAAC,CAAC,CAAC;IAC7G,CAAC;IAES,SAAS;QACjB,IAAI,CAAC,KAAK,EAAE,CAAC;IACf,CAAC;CACF;AAED;;GAEG;AACH,MAAM,UAAU,oBAAoB,CAClC,EAAU,EACV,MAAoB,EACpB,OAAgE;IAEhE,OAAO,IAAI,cAAc,CAAC;QACxB,EAAE;QACF,MAAM;QACN,UAAU,EAAE,OAAO,EAAE,UAAU;KAChC,CAAC,CAAC;AACL,CAAC"}
@@ -0,0 +1,171 @@
1
+ /**
2
+ * Active Shield (Sentinel) Widget
3
+ * Displays and manages virtual patching rules
4
+ */
5
+ import blessed from 'blessed';
6
+ import { BaseWidget, createBoxOptions } from '../core/widget.js';
7
+ import { colors } from '../theme.js';
8
+ export class SentinelWidget extends BaseWidget {
9
+ element;
10
+ rulesTable;
11
+ rules = [];
12
+ selectedIndex = 0;
13
+ onAddRuleCallback;
14
+ onDeleteRuleCallback;
15
+ constructor(id, config) {
16
+ super({ id, config, focusable: true });
17
+ // Create container box
18
+ this.element = blessed.box({
19
+ ...createBoxOptions(config),
20
+ label: ' Active Shield Rules (A:add D:delete) ',
21
+ style: {
22
+ ...createBoxOptions(config).style,
23
+ border: { fg: colors.primary },
24
+ },
25
+ });
26
+ // Create table for rules
27
+ this.rulesTable = blessed.table({
28
+ parent: this.element,
29
+ top: 0,
30
+ left: 0,
31
+ width: '100%',
32
+ height: '100%',
33
+ tags: true,
34
+ keys: true,
35
+ vi: true,
36
+ mouse: true,
37
+ style: {
38
+ header: { fg: colors.accent, bold: true },
39
+ cell: { fg: 'white' },
40
+ selected: { bg: colors.primary, fg: 'black', bold: true },
41
+ },
42
+ columnSpacing: 2,
43
+ columnWidth: [20, 40, 10, 8],
44
+ });
45
+ this.setupKeyHandlers();
46
+ }
47
+ setupKeyHandlers() {
48
+ this.rulesTable.key(['up', 'k'], () => {
49
+ if (this.rules.length > 0) {
50
+ this.selectedIndex = Math.max(0, this.selectedIndex - 1);
51
+ this.updateDisplay();
52
+ }
53
+ });
54
+ this.rulesTable.key(['down', 'j'], () => {
55
+ if (this.rules.length > 0) {
56
+ this.selectedIndex = Math.min(this.rules.length - 1, this.selectedIndex + 1);
57
+ this.updateDisplay();
58
+ }
59
+ });
60
+ this.rulesTable.key(['a', 'A'], () => {
61
+ this.onAddRuleCallback?.();
62
+ });
63
+ this.rulesTable.key(['d', 'D'], () => {
64
+ if (this.rules.length > 0 && this.rules[this.selectedIndex]) {
65
+ this.onDeleteRuleCallback?.(this.rules[this.selectedIndex].id);
66
+ }
67
+ });
68
+ this.rulesTable.key(['enter'], () => {
69
+ if (this.rules.length > 0 && this.rules[this.selectedIndex]) {
70
+ this.showRuleDetails(this.rules[this.selectedIndex]);
71
+ }
72
+ });
73
+ }
74
+ showRuleDetails(rule) {
75
+ if (!this.screen || !this.store)
76
+ return;
77
+ const modal = blessed.box({
78
+ parent: this.screen,
79
+ top: 'center',
80
+ left: 'center',
81
+ width: '70%',
82
+ height: '60%',
83
+ label: ` Rule Details: ${rule.id} `,
84
+ border: { type: 'line' },
85
+ style: {
86
+ border: { fg: colors.accent },
87
+ fg: 'white',
88
+ bg: 'black',
89
+ },
90
+ tags: true,
91
+ keys: true,
92
+ scrollable: true,
93
+ alwaysScroll: true,
94
+ });
95
+ const content = [
96
+ `{bold}ID:{/bold} ${rule.id}`,
97
+ `{bold}Pattern:{/bold} ${rule.pattern}`,
98
+ `{bold}Action:{/bold} {${rule.action === 'block' ? 'red' : 'yellow'}-fg}${rule.action.toUpperCase()}{/${rule.action === 'block' ? 'red' : 'yellow'}-fg}`,
99
+ `{bold}Enabled:{/bold} ${rule.enabled ? '{green-fg}YES{/green-fg}' : '{red-fg}NO{/red-fg}'}`,
100
+ `{bold}Created At:{/bold} ${rule.createdAt || 'Unknown'}`,
101
+ `{bold}Match Count:{/bold} ${rule.matchCount || 0}`,
102
+ '',
103
+ '{gray-fg}Press Esc to close{/gray-fg}',
104
+ ].join('\n');
105
+ modal.setContent(content);
106
+ modal.key(['escape', 'q'], () => {
107
+ this.screen?.remove(modal);
108
+ this.requestRender();
109
+ });
110
+ modal.focus();
111
+ this.requestRender();
112
+ }
113
+ /**
114
+ * Set the add rule callback
115
+ */
116
+ onAddRule(callback) {
117
+ this.onAddRuleCallback = callback;
118
+ }
119
+ /**
120
+ * Set the delete rule callback
121
+ */
122
+ onDeleteRule(callback) {
123
+ this.onDeleteRuleCallback = callback;
124
+ }
125
+ /**
126
+ * Update rules from external source
127
+ */
128
+ setRules(rules) {
129
+ this.rules = rules;
130
+ this.selectedIndex = Math.min(this.selectedIndex, Math.max(0, rules.length - 1));
131
+ this.updateDisplay();
132
+ }
133
+ render(state) {
134
+ // This widget is updated externally via setRules
135
+ // No state subscription needed for now
136
+ }
137
+ updateDisplay() {
138
+ const headers = ['ID', 'Pattern', 'Action', 'Enabled'];
139
+ const rows = [headers];
140
+ if (this.rules.length === 0) {
141
+ rows.push(['{gray-fg}No rules configured{/gray-fg}', '', '', '']);
142
+ }
143
+ else {
144
+ this.rules.forEach((rule, index) => {
145
+ const isSelected = index === this.selectedIndex;
146
+ const prefix = isSelected ? '{bold}' : '';
147
+ const suffix = isSelected ? '{/bold}' : '';
148
+ const actionColor = rule.action === 'block' ? 'red' : 'yellow';
149
+ const enabledColor = rule.enabled ? 'green' : 'gray';
150
+ rows.push([
151
+ `${prefix}${rule.id.substring(0, 18)}${suffix}`,
152
+ `${prefix}${rule.pattern.substring(0, 38)}${suffix}`,
153
+ `${prefix}{${actionColor}-fg}${rule.action}{/${actionColor}-fg}${suffix}`,
154
+ `${prefix}{${enabledColor}-fg}${rule.enabled ? '✓' : '✗'}{/${enabledColor}-fg}${suffix}`,
155
+ ]);
156
+ });
157
+ }
158
+ this.rulesTable.setData(rows);
159
+ this.requestRender();
160
+ }
161
+ onFocus() {
162
+ this.element.style.border = { fg: colors.accent };
163
+ this.rulesTable.focus();
164
+ this.requestRender();
165
+ }
166
+ onBlur() {
167
+ this.element.style.border = { fg: colors.primary };
168
+ this.requestRender();
169
+ }
170
+ }
171
+ //# sourceMappingURL=sentinel-widget.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sentinel-widget.js","sourceRoot":"","sources":["../../../src/tui/widgets/sentinel-widget.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,127 @@
1
+ /**
2
+ * Sparklines Widget
3
+ *
4
+ * Displays historical trends for RPS, blocks, and latency
5
+ * using blessed-contrib sparkline charts
6
+ */
7
+ import contrib from 'blessed-contrib';
8
+ import { BaseWidget } from '../core/widget.js';
9
+ import { MetricsBuffer } from '../state/metrics-buffer.js';
10
+ /**
11
+ * Sparklines Widget showing multiple metric trends
12
+ */
13
+ export class SparklinesWidget extends BaseWidget {
14
+ sparkline;
15
+ buffers = new Map();
16
+ metrics;
17
+ lastRequestCount = 0;
18
+ lastDeceptionCount = 0;
19
+ lastUpdateTime = Date.now();
20
+ element;
21
+ constructor(options) {
22
+ super({
23
+ id: options.id,
24
+ config: options.config,
25
+ subscriptions: ['requestCount', 'deceptionCount', 'health'],
26
+ });
27
+ this.metrics = options.metrics ?? ['RPS', 'Blocks', 'Latency'];
28
+ // Initialize metric buffers
29
+ const bufferSize = Math.min(50, options.timeWindow ?? 50);
30
+ for (const metric of this.metrics) {
31
+ this.buffers.set(metric, new MetricsBuffer({ maxSize: bufferSize }));
32
+ }
33
+ // Create the sparkline element
34
+ // Note: blessed-contrib types are incomplete, using type casts through unknown
35
+ this.sparkline = contrib.sparkline({
36
+ label: options.config.label ?? ' Traffic Trends ',
37
+ tags: true,
38
+ border: {
39
+ type: 'line',
40
+ fg: options.config.style?.border?.fg ?? 'cyan',
41
+ },
42
+ style: {
43
+ fg: 'green',
44
+ bg: options.config.style?.bg,
45
+ },
46
+ });
47
+ this.element = this.sparkline;
48
+ }
49
+ render(state) {
50
+ if (!this.mounted)
51
+ return;
52
+ // Calculate current metrics
53
+ const now = Date.now();
54
+ const timeDelta = (now - this.lastUpdateTime) / 1000;
55
+ // Calculate RPS
56
+ let rps = 0;
57
+ if (timeDelta > 0) {
58
+ const requestDelta = state.requestCount - this.lastRequestCount;
59
+ rps = requestDelta / timeDelta;
60
+ }
61
+ // Calculate blocks per second (deception events)
62
+ let blocks = 0;
63
+ if (timeDelta > 0) {
64
+ const deceptionDelta = state.deceptionCount - this.lastDeceptionCount;
65
+ blocks = deceptionDelta / timeDelta;
66
+ }
67
+ // Get latency from health
68
+ const latency = state.health?.lag_ms ?? 0;
69
+ // Update buffers
70
+ this.buffers.get('RPS')?.add(rps);
71
+ this.buffers.get('Blocks')?.add(blocks);
72
+ this.buffers.get('Latency')?.add(latency);
73
+ // Update last values
74
+ this.lastRequestCount = state.requestCount;
75
+ this.lastDeceptionCount = state.deceptionCount;
76
+ this.lastUpdateTime = now;
77
+ // Get the first metric for display (typically RPS)
78
+ const primaryMetric = this.buffers.get(this.metrics[0]);
79
+ if (primaryMetric) {
80
+ const data = primaryMetric.getValues();
81
+ this.sparkline.setData([this.metrics[0]], [data.length > 0 ? data : [0]]);
82
+ }
83
+ this.requestRender();
84
+ }
85
+ /**
86
+ * Get values for a specific metric
87
+ */
88
+ getMetricValues(metric) {
89
+ return this.buffers.get(metric)?.getValues() ?? [];
90
+ }
91
+ /**
92
+ * Get latest value for a metric
93
+ */
94
+ getLatestValue(metric) {
95
+ return this.buffers.get(metric)?.getLatest() ?? 0;
96
+ }
97
+ /**
98
+ * Get average value for a metric
99
+ */
100
+ getAverageValue(metric) {
101
+ return this.buffers.get(metric)?.getAverage() ?? 0;
102
+ }
103
+ /**
104
+ * Clear all metric buffers
105
+ */
106
+ clear() {
107
+ this.buffers.forEach((buffer) => buffer.clear());
108
+ this.lastRequestCount = 0;
109
+ this.lastDeceptionCount = 0;
110
+ this.lastUpdateTime = Date.now();
111
+ }
112
+ onUnmount() {
113
+ this.clear();
114
+ }
115
+ }
116
+ /**
117
+ * Create sparklines widget
118
+ */
119
+ export function createSparklinesWidget(id, config, options) {
120
+ return new SparklinesWidget({
121
+ id,
122
+ config,
123
+ metrics: options?.metrics,
124
+ timeWindow: options?.timeWindow,
125
+ });
126
+ }
127
+ //# sourceMappingURL=sparklines-widget.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sparklines-widget.js","sourceRoot":"","sources":["../../../src/tui/widgets/sparklines-widget.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,197 @@
1
+ /**
2
+ * System Info Widget
3
+ *
4
+ * Displays memory usage, CPU usage, environment variables, and network interfaces.
5
+ */
6
+ import blessed from 'blessed';
7
+ import { BaseWidget, createBoxOptions } from '../core/widget.js';
8
+ import { colors } from '../theme.js';
9
+ /**
10
+ * System Info Widget
11
+ */
12
+ export class SysInfoWidget extends BaseWidget {
13
+ element;
14
+ systemData = null;
15
+ envExpanded = false;
16
+ constructor(id, config) {
17
+ super({
18
+ id,
19
+ config,
20
+ subscriptions: [],
21
+ focusable: true,
22
+ });
23
+ this.element = blessed.box({
24
+ ...createBoxOptions(config),
25
+ label: config.label ?? ' System Information ',
26
+ keys: true,
27
+ vi: true,
28
+ scrollable: true,
29
+ alwaysScroll: true,
30
+ scrollbar: {
31
+ ch: '█',
32
+ track: { bg: 'gray' },
33
+ style: { bg: colors.accent },
34
+ },
35
+ style: {
36
+ ...createBoxOptions(config).style,
37
+ border: { fg: colors.accent },
38
+ },
39
+ });
40
+ }
41
+ onMount() {
42
+ // Fetch system data on mount
43
+ this.fetchSystemData();
44
+ }
45
+ render(state) {
46
+ if (!this.systemData) {
47
+ this.element.setContent('\n {gray-fg}Loading system data...{/gray-fg}');
48
+ this.requestRender();
49
+ return;
50
+ }
51
+ const lines = [];
52
+ const sys = this.systemData;
53
+ // Platform info
54
+ lines.push(`{bold}Platform:{/bold} ${sys.platform}`);
55
+ lines.push(`{bold}Node Version:{/bold} ${sys.nodeVersion}`);
56
+ lines.push(`{bold}Uptime:{/bold} ${this.formatUptime(sys.uptime)}`);
57
+ lines.push('');
58
+ // Memory
59
+ const memBar = this.createProgressBar(sys.memory.percentUsed, 30);
60
+ const memColor = this.getUsageColor(sys.memory.percentUsed);
61
+ lines.push(`{bold}Memory:{/bold} {${memColor}}${sys.memory.percentUsed.toFixed(1)}%{/${memColor}}`);
62
+ lines.push(` ${memBar}`);
63
+ lines.push(` ${this.formatBytes(sys.memory.used)} / ${this.formatBytes(sys.memory.total)} (${this.formatBytes(sys.memory.free)} free)`);
64
+ lines.push('');
65
+ // CPU
66
+ const cpuBar = this.createProgressBar(sys.cpu.usage, 30);
67
+ const cpuColor = this.getUsageColor(sys.cpu.usage);
68
+ lines.push(`{bold}CPU:{/bold} {${cpuColor}}${sys.cpu.usage.toFixed(1)}%{/${cpuColor}}`);
69
+ lines.push(` ${cpuBar}`);
70
+ lines.push(` ${sys.cpu.model} (${sys.cpu.cores} cores)`);
71
+ lines.push('');
72
+ // Network interfaces
73
+ lines.push(`{bold}Network Interfaces:{/bold}`);
74
+ const externalIfaces = sys.network.filter((iface) => !iface.internal);
75
+ if (externalIfaces.length === 0) {
76
+ lines.push(' {gray-fg}No external interfaces{/gray-fg}');
77
+ }
78
+ else {
79
+ for (const iface of externalIfaces) {
80
+ lines.push(` {cyan-fg}${iface.name}{/cyan-fg}: ${iface.address} (${iface.family})`);
81
+ }
82
+ }
83
+ lines.push('');
84
+ // Environment variables (collapsed by default)
85
+ const envCount = Object.keys(sys.environment).length;
86
+ lines.push(`{bold}Environment Variables:{/bold} (${envCount}) ${this.envExpanded ? '{yellow-fg}[EXPANDED]{/yellow-fg}' : '{gray-fg}[COLLAPSED]{/gray-fg}'}`);
87
+ if (this.envExpanded) {
88
+ const envEntries = Object.entries(sys.environment).sort(([a], [b]) => a.localeCompare(b));
89
+ for (const [key, value] of envEntries) {
90
+ lines.push(` {cyan-fg}${key}{/cyan-fg} = ${value}`);
91
+ }
92
+ }
93
+ else {
94
+ lines.push(' {gray-fg}Press e to expand{/gray-fg}');
95
+ }
96
+ lines.push('');
97
+ lines.push('{gray-fg}Press r to refresh, e to toggle environment{/gray-fg}');
98
+ this.element.setContent(lines.join('\n'));
99
+ this.requestRender();
100
+ }
101
+ handleKey(key) {
102
+ if (key === 'r') {
103
+ this.fetchSystemData();
104
+ return true;
105
+ }
106
+ if (key === 'e') {
107
+ this.envExpanded = !this.envExpanded;
108
+ this.render(this.getState());
109
+ return true;
110
+ }
111
+ return false;
112
+ }
113
+ async fetchSystemData() {
114
+ try {
115
+ // Mock data for now - will be replaced with real API call
116
+ this.systemData = {
117
+ memory: {
118
+ total: 16 * 1024 * 1024 * 1024, // 16GB
119
+ free: 4 * 1024 * 1024 * 1024, // 4GB
120
+ used: 12 * 1024 * 1024 * 1024, // 12GB
121
+ percentUsed: 75,
122
+ },
123
+ cpu: {
124
+ model: 'Intel Core i7-9750H',
125
+ cores: 12,
126
+ usage: 32.5,
127
+ },
128
+ environment: {
129
+ NODE_ENV: 'production',
130
+ PORT_HTTP1: '8080',
131
+ PORT_HTTP2: '8443',
132
+ RISK_SERVER_URL: 'http://localhost:4100',
133
+ SENSOR_ID: 'apparatus-prod-1',
134
+ LOG_LEVEL: 'info',
135
+ H2C: 'false',
136
+ CORS: 'true',
137
+ },
138
+ network: [
139
+ { name: 'eth0', address: '10.0.1.5', family: 'IPv4', internal: false },
140
+ { name: 'eth0', address: 'fe80::1', family: 'IPv6', internal: false },
141
+ { name: 'lo', address: '127.0.0.1', family: 'IPv4', internal: true },
142
+ ],
143
+ uptime: 345678, // ~4 days
144
+ platform: 'linux',
145
+ nodeVersion: 'v20.11.0',
146
+ };
147
+ this.render(this.getState());
148
+ }
149
+ catch (error) {
150
+ this.element.setContent(`\n {red-fg}Error loading system data:{/red-fg}\n ${error instanceof Error ? error.message : String(error)}`);
151
+ this.requestRender();
152
+ }
153
+ }
154
+ createProgressBar(percent, width) {
155
+ const filled = Math.round((percent / 100) * width);
156
+ const empty = width - filled;
157
+ const color = this.getUsageColor(percent);
158
+ return `{${color}}${'█'.repeat(filled)}{/${color}}${'░'.repeat(empty)}`;
159
+ }
160
+ getUsageColor(percent) {
161
+ if (percent >= 90)
162
+ return 'red-fg';
163
+ if (percent >= 75)
164
+ return 'yellow-fg';
165
+ return 'green-fg';
166
+ }
167
+ formatBytes(bytes) {
168
+ const units = ['B', 'KB', 'MB', 'GB', 'TB'];
169
+ let value = bytes;
170
+ let unitIndex = 0;
171
+ while (value >= 1024 && unitIndex < units.length - 1) {
172
+ value /= 1024;
173
+ unitIndex++;
174
+ }
175
+ return `${value.toFixed(1)} ${units[unitIndex]}`;
176
+ }
177
+ formatUptime(seconds) {
178
+ const days = Math.floor(seconds / 86400);
179
+ const hours = Math.floor((seconds % 86400) / 3600);
180
+ const minutes = Math.floor((seconds % 3600) / 60);
181
+ const parts = [];
182
+ if (days > 0)
183
+ parts.push(`${days}d`);
184
+ if (hours > 0)
185
+ parts.push(`${hours}h`);
186
+ if (minutes > 0 || parts.length === 0)
187
+ parts.push(`${minutes}m`);
188
+ return parts.join(' ');
189
+ }
190
+ }
191
+ /**
192
+ * Factory function for widget registry
193
+ */
194
+ export function createSysInfoWidget(id, config) {
195
+ return new SysInfoWidget(id, config);
196
+ }
197
+ //# sourceMappingURL=sysinfo-widget.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sysinfo-widget.js","sourceRoot":"","sources":["../../../src/tui/widgets/sysinfo-widget.ts"],"names":[],"mappings":"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"}