vaspera 2.7.0 → 2.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 (321) hide show
  1. package/CHANGELOG.md +72 -0
  2. package/README.md +111 -7
  3. package/dist/__tests__/agents/adversary/tactics/api.test.d.ts +5 -0
  4. package/dist/__tests__/agents/adversary/tactics/api.test.d.ts.map +1 -0
  5. package/dist/__tests__/agents/adversary/tactics/api.test.js +369 -0
  6. package/dist/__tests__/agents/adversary/tactics/api.test.js.map +1 -0
  7. package/dist/__tests__/agents/adversary/tactics/llm.test.d.ts +5 -0
  8. package/dist/__tests__/agents/adversary/tactics/llm.test.d.ts.map +1 -0
  9. package/dist/__tests__/agents/adversary/tactics/llm.test.js +409 -0
  10. package/dist/__tests__/agents/adversary/tactics/llm.test.js.map +1 -0
  11. package/dist/__tests__/agents/adversary/tactics/registry.test.d.ts +7 -0
  12. package/dist/__tests__/agents/adversary/tactics/registry.test.d.ts.map +1 -0
  13. package/dist/__tests__/agents/adversary/tactics/registry.test.js +74 -0
  14. package/dist/__tests__/agents/adversary/tactics/registry.test.js.map +1 -0
  15. package/dist/__tests__/agents/adversary/tactics/web-app.test.d.ts +7 -0
  16. package/dist/__tests__/agents/adversary/tactics/web-app.test.d.ts.map +1 -0
  17. package/dist/__tests__/agents/adversary/tactics/web-app.test.js +374 -0
  18. package/dist/__tests__/agents/adversary/tactics/web-app.test.js.map +1 -0
  19. package/dist/__tests__/compliance-bundle.test.d.ts +9 -0
  20. package/dist/__tests__/compliance-bundle.test.d.ts.map +1 -0
  21. package/dist/__tests__/compliance-bundle.test.js +344 -0
  22. package/dist/__tests__/compliance-bundle.test.js.map +1 -0
  23. package/dist/__tests__/healthcare-compliance.test.d.ts +9 -0
  24. package/dist/__tests__/healthcare-compliance.test.d.ts.map +1 -0
  25. package/dist/__tests__/healthcare-compliance.test.js +233 -0
  26. package/dist/__tests__/healthcare-compliance.test.js.map +1 -0
  27. package/dist/action/diff-mode.d.ts +124 -8
  28. package/dist/action/diff-mode.d.ts.map +1 -1
  29. package/dist/action/diff-mode.js +384 -65
  30. package/dist/action/diff-mode.js.map +1 -1
  31. package/dist/action/diff-mode.test.js +3 -3
  32. package/dist/action/diff-mode.test.js.map +1 -1
  33. package/dist/action/pr-comment.test.js +1 -0
  34. package/dist/action/pr-comment.test.js.map +1 -1
  35. package/dist/action/sarif-upload.test.js +1 -0
  36. package/dist/action/sarif-upload.test.js.map +1 -1
  37. package/dist/agents/adversary/config.d.ts +113 -0
  38. package/dist/agents/adversary/config.d.ts.map +1 -0
  39. package/dist/agents/adversary/config.js +391 -0
  40. package/dist/agents/adversary/config.js.map +1 -0
  41. package/dist/agents/adversary/index.d.ts +41 -0
  42. package/dist/agents/adversary/index.d.ts.map +1 -0
  43. package/dist/agents/adversary/index.js +838 -0
  44. package/dist/agents/adversary/index.js.map +1 -0
  45. package/dist/agents/adversary/reporting/compliance-mapper.d.ts +108 -0
  46. package/dist/agents/adversary/reporting/compliance-mapper.d.ts.map +1 -0
  47. package/dist/agents/adversary/reporting/compliance-mapper.js +391 -0
  48. package/dist/agents/adversary/reporting/compliance-mapper.js.map +1 -0
  49. package/dist/agents/adversary/reporting/index.d.ts +10 -0
  50. package/dist/agents/adversary/reporting/index.d.ts.map +1 -0
  51. package/dist/agents/adversary/reporting/index.js +10 -0
  52. package/dist/agents/adversary/reporting/index.js.map +1 -0
  53. package/dist/agents/adversary/reporting/poc-generator.d.ts +44 -0
  54. package/dist/agents/adversary/reporting/poc-generator.d.ts.map +1 -0
  55. package/dist/agents/adversary/reporting/poc-generator.js +308 -0
  56. package/dist/agents/adversary/reporting/poc-generator.js.map +1 -0
  57. package/dist/agents/adversary/tactics/api.d.ts +13 -0
  58. package/dist/agents/adversary/tactics/api.d.ts.map +1 -0
  59. package/dist/agents/adversary/tactics/api.js +815 -0
  60. package/dist/agents/adversary/tactics/api.js.map +1 -0
  61. package/dist/agents/adversary/tactics/auth.d.ts +13 -0
  62. package/dist/agents/adversary/tactics/auth.d.ts.map +1 -0
  63. package/dist/agents/adversary/tactics/auth.js +676 -0
  64. package/dist/agents/adversary/tactics/auth.js.map +1 -0
  65. package/dist/agents/adversary/tactics/index.d.ts +129 -0
  66. package/dist/agents/adversary/tactics/index.d.ts.map +1 -0
  67. package/dist/agents/adversary/tactics/index.js +199 -0
  68. package/dist/agents/adversary/tactics/index.js.map +1 -0
  69. package/dist/agents/adversary/tactics/infra.d.ts +13 -0
  70. package/dist/agents/adversary/tactics/infra.d.ts.map +1 -0
  71. package/dist/agents/adversary/tactics/infra.js +827 -0
  72. package/dist/agents/adversary/tactics/infra.js.map +1 -0
  73. package/dist/agents/adversary/tactics/injection.d.ts +12 -0
  74. package/dist/agents/adversary/tactics/injection.d.ts.map +1 -0
  75. package/dist/agents/adversary/tactics/injection.js +549 -0
  76. package/dist/agents/adversary/tactics/injection.js.map +1 -0
  77. package/dist/agents/adversary/tactics/llm.d.ts +13 -0
  78. package/dist/agents/adversary/tactics/llm.d.ts.map +1 -0
  79. package/dist/agents/adversary/tactics/llm.js +767 -0
  80. package/dist/agents/adversary/tactics/llm.js.map +1 -0
  81. package/dist/agents/adversary/tactics/web-app.d.ts +13 -0
  82. package/dist/agents/adversary/tactics/web-app.d.ts.map +1 -0
  83. package/dist/agents/adversary/tactics/web-app.js +717 -0
  84. package/dist/agents/adversary/tactics/web-app.js.map +1 -0
  85. package/dist/agents/adversary/types.d.ts +407 -0
  86. package/dist/agents/adversary/types.d.ts.map +1 -0
  87. package/dist/agents/adversary/types.js +12 -0
  88. package/dist/agents/adversary/types.js.map +1 -0
  89. package/dist/agents/index.d.ts +1 -0
  90. package/dist/agents/index.d.ts.map +1 -1
  91. package/dist/agents/index.js +2 -0
  92. package/dist/agents/index.js.map +1 -1
  93. package/dist/agents/zero-day-hunter.d.ts +1 -1
  94. package/dist/agents/zero-day-hunter.d.ts.map +1 -1
  95. package/dist/analysis/data-flow.d.ts +154 -0
  96. package/dist/analysis/data-flow.d.ts.map +1 -0
  97. package/dist/analysis/data-flow.js +393 -0
  98. package/dist/analysis/data-flow.js.map +1 -0
  99. package/dist/analysis/index.d.ts +9 -0
  100. package/dist/analysis/index.d.ts.map +1 -0
  101. package/dist/analysis/index.js +9 -0
  102. package/dist/analysis/index.js.map +1 -0
  103. package/dist/badge-service/index.d.ts +144 -0
  104. package/dist/badge-service/index.d.ts.map +1 -0
  105. package/dist/badge-service/index.js +206 -0
  106. package/dist/badge-service/index.js.map +1 -0
  107. package/dist/certification/consensus.test.js +2 -0
  108. package/dist/certification/consensus.test.js.map +1 -1
  109. package/dist/certification/store.d.ts.map +1 -1
  110. package/dist/certification/store.js +4 -0
  111. package/dist/certification/store.js.map +1 -1
  112. package/dist/certification/types.d.ts +3 -3
  113. package/dist/certification/types.d.ts.map +1 -1
  114. package/dist/certification/types.js +2 -0
  115. package/dist/certification/types.js.map +1 -1
  116. package/dist/commands/certification/certify.d.ts.map +1 -1
  117. package/dist/commands/certification/certify.js +18 -4
  118. package/dist/commands/certification/certify.js.map +1 -1
  119. package/dist/compliance/attestation.d.ts +39 -0
  120. package/dist/compliance/attestation.d.ts.map +1 -0
  121. package/dist/compliance/attestation.js +364 -0
  122. package/dist/compliance/attestation.js.map +1 -0
  123. package/dist/compliance/cfr42-part2.d.ts +42 -0
  124. package/dist/compliance/cfr42-part2.d.ts.map +1 -0
  125. package/dist/compliance/cfr42-part2.js +408 -0
  126. package/dist/compliance/cfr42-part2.js.map +1 -0
  127. package/dist/compliance/compliance-bundle.d.ts +100 -0
  128. package/dist/compliance/compliance-bundle.d.ts.map +1 -0
  129. package/dist/compliance/compliance-bundle.js +210 -0
  130. package/dist/compliance/compliance-bundle.js.map +1 -0
  131. package/dist/compliance/healthcare-bundle.d.ts +68 -0
  132. package/dist/compliance/healthcare-bundle.d.ts.map +1 -0
  133. package/dist/compliance/healthcare-bundle.js +104 -0
  134. package/dist/compliance/healthcare-bundle.js.map +1 -0
  135. package/dist/compliance/hipaa.d.ts.map +1 -1
  136. package/dist/compliance/hipaa.js +14 -11
  137. package/dist/compliance/hipaa.js.map +1 -1
  138. package/dist/compliance/index.d.ts +10 -2
  139. package/dist/compliance/index.d.ts.map +1 -1
  140. package/dist/compliance/index.js +9 -3
  141. package/dist/compliance/index.js.map +1 -1
  142. package/dist/compliance/mapper.d.ts.map +1 -1
  143. package/dist/compliance/mapper.js +3 -17
  144. package/dist/compliance/mapper.js.map +1 -1
  145. package/dist/compliance/nist-800-53.d.ts +22 -6
  146. package/dist/compliance/nist-800-53.d.ts.map +1 -1
  147. package/dist/compliance/nist-800-53.js +264 -272
  148. package/dist/compliance/nist-800-53.js.map +1 -1
  149. package/dist/compliance/report.d.ts +31 -2
  150. package/dist/compliance/report.d.ts.map +1 -1
  151. package/dist/compliance/report.js +255 -4
  152. package/dist/compliance/report.js.map +1 -1
  153. package/dist/compliance/types.d.ts +1 -1
  154. package/dist/compliance/types.d.ts.map +1 -1
  155. package/dist/config/flags.d.ts +12 -12
  156. package/dist/cost/index.d.ts +1 -1
  157. package/dist/cost/index.d.ts.map +1 -1
  158. package/dist/cost/index.js +1 -1
  159. package/dist/cost/index.js.map +1 -1
  160. package/dist/cost/tracker.d.ts +64 -0
  161. package/dist/cost/tracker.d.ts.map +1 -1
  162. package/dist/cost/tracker.js +165 -0
  163. package/dist/cost/tracker.js.map +1 -1
  164. package/dist/eval/fixtures/healthcare/audit-gaps.d.ts +28 -0
  165. package/dist/eval/fixtures/healthcare/audit-gaps.d.ts.map +1 -0
  166. package/dist/eval/fixtures/healthcare/audit-gaps.js +90 -0
  167. package/dist/eval/fixtures/healthcare/audit-gaps.js.map +1 -0
  168. package/dist/eval/fixtures/healthcare/consent-bypass.d.ts +31 -0
  169. package/dist/eval/fixtures/healthcare/consent-bypass.d.ts.map +1 -0
  170. package/dist/eval/fixtures/healthcare/consent-bypass.js +61 -0
  171. package/dist/eval/fixtures/healthcare/consent-bypass.js.map +1 -0
  172. package/dist/eval/fixtures/healthcare/phi-in-logs.d.ts +24 -0
  173. package/dist/eval/fixtures/healthcare/phi-in-logs.d.ts.map +1 -0
  174. package/dist/eval/fixtures/healthcare/phi-in-logs.js +41 -0
  175. package/dist/eval/fixtures/healthcare/phi-in-logs.js.map +1 -0
  176. package/dist/evidence/collector.d.ts +21 -0
  177. package/dist/evidence/collector.d.ts.map +1 -0
  178. package/dist/evidence/collector.js +340 -0
  179. package/dist/evidence/collector.js.map +1 -0
  180. package/dist/evidence/index.d.ts +11 -0
  181. package/dist/evidence/index.d.ts.map +1 -0
  182. package/dist/evidence/index.js +12 -0
  183. package/dist/evidence/index.js.map +1 -0
  184. package/dist/evidence/store.d.ts +39 -0
  185. package/dist/evidence/store.d.ts.map +1 -0
  186. package/dist/evidence/store.js +173 -0
  187. package/dist/evidence/store.js.map +1 -0
  188. package/dist/evidence/types.d.ts +175 -0
  189. package/dist/evidence/types.d.ts.map +1 -0
  190. package/dist/evidence/types.js +9 -0
  191. package/dist/evidence/types.js.map +1 -0
  192. package/dist/exporters/checkmarx.d.ts +18 -0
  193. package/dist/exporters/checkmarx.d.ts.map +1 -0
  194. package/dist/exporters/checkmarx.js +203 -0
  195. package/dist/exporters/checkmarx.js.map +1 -0
  196. package/dist/exporters/index.d.ts +22 -0
  197. package/dist/exporters/index.d.ts.map +1 -0
  198. package/dist/exporters/index.js +41 -0
  199. package/dist/exporters/index.js.map +1 -0
  200. package/dist/exporters/snyk.d.ts +18 -0
  201. package/dist/exporters/snyk.d.ts.map +1 -0
  202. package/dist/exporters/snyk.js +119 -0
  203. package/dist/exporters/snyk.js.map +1 -0
  204. package/dist/exporters/sonarqube.d.ts +18 -0
  205. package/dist/exporters/sonarqube.d.ts.map +1 -0
  206. package/dist/exporters/sonarqube.js +125 -0
  207. package/dist/exporters/sonarqube.js.map +1 -0
  208. package/dist/exporters/types.d.ts +190 -0
  209. package/dist/exporters/types.d.ts.map +1 -0
  210. package/dist/exporters/types.js +9 -0
  211. package/dist/exporters/types.js.map +1 -0
  212. package/dist/frontier/index.d.ts +12 -0
  213. package/dist/frontier/index.d.ts.map +1 -0
  214. package/dist/frontier/index.js +12 -0
  215. package/dist/frontier/index.js.map +1 -0
  216. package/dist/frontier/orchestrator.d.ts +73 -0
  217. package/dist/frontier/orchestrator.d.ts.map +1 -0
  218. package/dist/frontier/orchestrator.js +312 -0
  219. package/dist/frontier/orchestrator.js.map +1 -0
  220. package/dist/frontier/providers/stub.d.ts +32 -0
  221. package/dist/frontier/providers/stub.d.ts.map +1 -0
  222. package/dist/frontier/providers/stub.js +66 -0
  223. package/dist/frontier/providers/stub.js.map +1 -0
  224. package/dist/frontier/types.d.ts +318 -0
  225. package/dist/frontier/types.d.ts.map +1 -0
  226. package/dist/frontier/types.js +27 -0
  227. package/dist/frontier/types.js.map +1 -0
  228. package/dist/history/index.d.ts +13 -0
  229. package/dist/history/index.d.ts.map +1 -0
  230. package/dist/history/index.js +15 -0
  231. package/dist/history/index.js.map +1 -0
  232. package/dist/history/store.d.ts +74 -0
  233. package/dist/history/store.d.ts.map +1 -0
  234. package/dist/history/store.js +399 -0
  235. package/dist/history/store.js.map +1 -0
  236. package/dist/history/types.d.ts +282 -0
  237. package/dist/history/types.d.ts.map +1 -0
  238. package/dist/history/types.js +41 -0
  239. package/dist/history/types.js.map +1 -0
  240. package/dist/history/verify.d.ts +44 -0
  241. package/dist/history/verify.d.ts.map +1 -0
  242. package/dist/history/verify.js +230 -0
  243. package/dist/history/verify.js.map +1 -0
  244. package/dist/index.d.ts.map +1 -1
  245. package/dist/index.js +431 -18
  246. package/dist/index.js.map +1 -1
  247. package/dist/multimodel/index.d.ts +1 -0
  248. package/dist/multimodel/index.d.ts.map +1 -1
  249. package/dist/multimodel/index.js +2 -0
  250. package/dist/multimodel/index.js.map +1 -1
  251. package/dist/multimodel/leaderboard.d.ts +116 -0
  252. package/dist/multimodel/leaderboard.d.ts.map +1 -0
  253. package/dist/multimodel/leaderboard.js +262 -0
  254. package/dist/multimodel/leaderboard.js.map +1 -0
  255. package/dist/observability/otel.d.ts.map +1 -1
  256. package/dist/observability/otel.js +1 -3
  257. package/dist/observability/otel.js.map +1 -1
  258. package/dist/plugins/loader.js +1 -1
  259. package/dist/plugins/loader.js.map +1 -1
  260. package/dist/sbom/provenance.test.js +2 -2
  261. package/dist/sbom/provenance.test.js.map +1 -1
  262. package/dist/scanners/agent/agent-chain-analysis.d.ts +152 -0
  263. package/dist/scanners/agent/agent-chain-analysis.d.ts.map +1 -0
  264. package/dist/scanners/agent/agent-chain-analysis.js +438 -0
  265. package/dist/scanners/agent/agent-chain-analysis.js.map +1 -0
  266. package/dist/scanners/agent/manifest-audit.d.ts.map +1 -1
  267. package/dist/scanners/agent/manifest-audit.js +30 -18
  268. package/dist/scanners/agent/manifest-audit.js.map +1 -1
  269. package/dist/scanners/agent/payloads/index.d.ts +2 -1
  270. package/dist/scanners/agent/payloads/index.d.ts.map +1 -1
  271. package/dist/scanners/agent/payloads/index.js +25 -6
  272. package/dist/scanners/agent/payloads/index.js.map +1 -1
  273. package/dist/scanners/agent/prompt-injection-fuzzer.d.ts.map +1 -1
  274. package/dist/scanners/agent/prompt-injection-fuzzer.js +14 -0
  275. package/dist/scanners/agent/prompt-injection-fuzzer.js.map +1 -1
  276. package/dist/scanners/agent/types.d.ts +5 -5
  277. package/dist/scanners/agent/types.d.ts.map +1 -1
  278. package/dist/scanners/agent/types.js.map +1 -1
  279. package/dist/scanners/cache.d.ts +156 -0
  280. package/dist/scanners/cache.d.ts.map +1 -0
  281. package/dist/scanners/cache.js +462 -0
  282. package/dist/scanners/cache.js.map +1 -0
  283. package/dist/scanners/dependencies.d.ts.map +1 -1
  284. package/dist/scanners/dependencies.js +5 -6
  285. package/dist/scanners/dependencies.js.map +1 -1
  286. package/dist/scanners/gosec.d.ts.map +1 -1
  287. package/dist/scanners/gosec.js +47 -9
  288. package/dist/scanners/gosec.js.map +1 -1
  289. package/dist/scanners/healthcare.d.ts +29 -0
  290. package/dist/scanners/healthcare.d.ts.map +1 -0
  291. package/dist/scanners/healthcare.js +526 -0
  292. package/dist/scanners/healthcare.js.map +1 -0
  293. package/dist/scanners/index.d.ts +1 -0
  294. package/dist/scanners/index.d.ts.map +1 -1
  295. package/dist/scanners/index.js +33 -0
  296. package/dist/scanners/index.js.map +1 -1
  297. package/dist/scanners/index.test.js +6 -6
  298. package/dist/scanners/index.test.js.map +1 -1
  299. package/dist/scanners/secrets.js +4 -4
  300. package/dist/scanners/secrets.js.map +1 -1
  301. package/dist/scanners/semgrep.js +5 -5
  302. package/dist/scanners/semgrep.js.map +1 -1
  303. package/dist/scanners/types.d.ts +1 -1
  304. package/dist/scanners/types.d.ts.map +1 -1
  305. package/dist/scanners/types.js +1 -0
  306. package/dist/scanners/types.js.map +1 -1
  307. package/dist/scanners/typescript.test.js +1 -1
  308. package/dist/scanners/typescript.test.js.map +1 -1
  309. package/dist/telemetry/index.d.ts +10 -0
  310. package/dist/telemetry/index.d.ts.map +1 -0
  311. package/dist/telemetry/index.js +10 -0
  312. package/dist/telemetry/index.js.map +1 -0
  313. package/dist/telemetry/registry.d.ts +178 -0
  314. package/dist/telemetry/registry.d.ts.map +1 -0
  315. package/dist/telemetry/registry.js +297 -0
  316. package/dist/telemetry/registry.js.map +1 -0
  317. package/dist/telemetry/usage.d.ts +197 -0
  318. package/dist/telemetry/usage.d.ts.map +1 -0
  319. package/dist/telemetry/usage.js +244 -0
  320. package/dist/telemetry/usage.js.map +1 -0
  321. package/package.json +11 -2
@@ -0,0 +1,399 @@
1
+ /**
2
+ * History Store
3
+ *
4
+ * Append-only storage for certification and scan history.
5
+ * Uses JSONL format for efficient streaming reads/writes.
6
+ *
7
+ * @module history/store
8
+ */
9
+ import { appendFile, mkdir, access, stat } from "fs/promises";
10
+ import { join } from "path";
11
+ import { createReadStream } from "fs";
12
+ import { createInterface } from "readline";
13
+ import { randomUUID, createHash } from "crypto";
14
+ import { logger } from "../logger.js";
15
+ const HISTORY_DIR = ".vaspera";
16
+ const HISTORY_FILE = "history.jsonl";
17
+ /** Genesis hash for the first entry in a chain */
18
+ const GENESIS_HASH = "0000000000000000000000000000000000000000000000000000000000000000";
19
+ /**
20
+ * Calculate SHA-256 hash of an entry (excluding the integrity field)
21
+ */
22
+ export function calculateEntryHash(entry) {
23
+ // Create a copy without the integrity field for hashing
24
+ const { integrity: _integrity, ...entryWithoutIntegrity } = entry;
25
+ const serialized = JSON.stringify(entryWithoutIntegrity, Object.keys(entryWithoutIntegrity).sort());
26
+ return createHash("sha256").update(serialized, "utf8").digest("hex");
27
+ }
28
+ /**
29
+ * Get the hash of the last entry in the history file
30
+ */
31
+ export async function getLastEntryHash(projectPath) {
32
+ const historyFile = join(projectPath, HISTORY_DIR, HISTORY_FILE);
33
+ try {
34
+ await access(historyFile);
35
+ }
36
+ catch {
37
+ return GENESIS_HASH;
38
+ }
39
+ // Read the file and get the last valid entry
40
+ const fileStream = createReadStream(historyFile);
41
+ const rl = createInterface({
42
+ input: fileStream,
43
+ crlfDelay: Infinity,
44
+ });
45
+ let lastEntry = null;
46
+ for await (const line of rl) {
47
+ if (!line.trim())
48
+ continue;
49
+ try {
50
+ lastEntry = JSON.parse(line);
51
+ }
52
+ catch {
53
+ // Skip malformed lines
54
+ }
55
+ }
56
+ if (!lastEntry) {
57
+ return GENESIS_HASH;
58
+ }
59
+ // Return the stored hash if available, otherwise calculate it
60
+ if (lastEntry.integrity?.hash) {
61
+ return lastEntry.integrity.hash;
62
+ }
63
+ // For legacy entries without integrity, calculate the hash
64
+ return calculateEntryHash(lastEntry);
65
+ }
66
+ /**
67
+ * Ensure history file exists
68
+ */
69
+ async function ensureHistoryFile(projectPath) {
70
+ const historyDir = join(projectPath, HISTORY_DIR);
71
+ const historyFile = join(historyDir, HISTORY_FILE);
72
+ try {
73
+ await access(historyDir);
74
+ }
75
+ catch {
76
+ await mkdir(historyDir, { recursive: true });
77
+ }
78
+ return historyFile;
79
+ }
80
+ /**
81
+ * Append a history entry
82
+ */
83
+ export async function appendHistoryEntry(projectPath, entry, options = {}) {
84
+ const historyFile = await ensureHistoryFile(projectPath);
85
+ const { actor, enableIntegrity = true } = options;
86
+ // Build base entry
87
+ const baseEntry = {
88
+ ...entry,
89
+ id: randomUUID().slice(0, 8),
90
+ timestamp: new Date().toISOString(),
91
+ projectPath,
92
+ ...(actor && { actor }),
93
+ };
94
+ // Add integrity proof if enabled
95
+ if (enableIntegrity) {
96
+ const previousHash = await getLastEntryHash(projectPath);
97
+ const hash = calculateEntryHash(baseEntry);
98
+ const integrity = {
99
+ hash,
100
+ previousHash,
101
+ };
102
+ // Note: Sigstore signing would be added here if options.sign is true
103
+ // This requires the signBlob function from src/sbom/signing.ts
104
+ // For now, we just set up the hash chain
105
+ baseEntry.integrity = integrity;
106
+ }
107
+ const line = JSON.stringify(baseEntry) + "\n";
108
+ await appendFile(historyFile, line, "utf-8");
109
+ logger.debug("history.entry_appended", {
110
+ type: entry.type,
111
+ id: baseEntry.id,
112
+ hasIntegrity: !!baseEntry.integrity,
113
+ hasActor: !!baseEntry.actor,
114
+ });
115
+ return baseEntry;
116
+ }
117
+ /**
118
+ * Query history entries with filtering and pagination
119
+ */
120
+ export async function queryHistory(projectPath, options = {}) {
121
+ const historyFile = join(projectPath, HISTORY_DIR, HISTORY_FILE);
122
+ const { type, certificationId, startDate, endDate, limit = 100, offset = 0, order = "desc", } = options;
123
+ const entries = [];
124
+ try {
125
+ await access(historyFile);
126
+ }
127
+ catch {
128
+ // History file doesn't exist yet
129
+ return { entries: [], total: 0, hasMore: false };
130
+ }
131
+ // Stream read the JSONL file
132
+ const fileStream = createReadStream(historyFile);
133
+ const rl = createInterface({
134
+ input: fileStream,
135
+ crlfDelay: Infinity,
136
+ });
137
+ const allMatching = [];
138
+ for await (const line of rl) {
139
+ if (!line.trim())
140
+ continue;
141
+ try {
142
+ const entry = JSON.parse(line);
143
+ // Apply filters
144
+ if (type) {
145
+ const types = Array.isArray(type) ? type : [type];
146
+ if (!types.includes(entry.type))
147
+ continue;
148
+ }
149
+ if (certificationId && entry.certificationId !== certificationId)
150
+ continue;
151
+ if (startDate && entry.timestamp < startDate)
152
+ continue;
153
+ if (endDate && entry.timestamp > endDate)
154
+ continue;
155
+ allMatching.push(entry);
156
+ }
157
+ catch {
158
+ // Skip malformed lines
159
+ logger.warn("history.malformed_line", { line: line.slice(0, 100) });
160
+ }
161
+ }
162
+ // Sort
163
+ allMatching.sort((a, b) => {
164
+ const cmp = a.timestamp.localeCompare(b.timestamp);
165
+ return order === "desc" ? -cmp : cmp;
166
+ });
167
+ // Paginate
168
+ const total = allMatching.length;
169
+ const paginated = allMatching.slice(offset, offset + limit);
170
+ return {
171
+ entries: paginated,
172
+ total,
173
+ hasMore: offset + limit < total,
174
+ };
175
+ }
176
+ /**
177
+ * Get period boundaries for trend calculation
178
+ */
179
+ function getPeriodBoundaries(startDate, endDate, period) {
180
+ const boundaries = [];
181
+ let current = new Date(startDate);
182
+ while (current < endDate) {
183
+ let periodStart = new Date(current);
184
+ let periodEnd;
185
+ let label;
186
+ switch (period) {
187
+ case "day":
188
+ periodEnd = new Date(current);
189
+ periodEnd.setDate(periodEnd.getDate() + 1);
190
+ label = current.toISOString().slice(0, 10);
191
+ break;
192
+ case "week":
193
+ periodEnd = new Date(current);
194
+ periodEnd.setDate(periodEnd.getDate() + 7);
195
+ label = `W${getWeekNumber(current)}-${current.getFullYear()}`;
196
+ break;
197
+ case "month":
198
+ periodEnd = new Date(current);
199
+ periodEnd.setMonth(periodEnd.getMonth() + 1);
200
+ label = `${current.getFullYear()}-${String(current.getMonth() + 1).padStart(2, "0")}`;
201
+ break;
202
+ case "quarter":
203
+ periodEnd = new Date(current);
204
+ periodEnd.setMonth(periodEnd.getMonth() + 3);
205
+ label = `Q${Math.floor(current.getMonth() / 3) + 1}-${current.getFullYear()}`;
206
+ break;
207
+ case "year":
208
+ periodEnd = new Date(current);
209
+ periodEnd.setFullYear(periodEnd.getFullYear() + 1);
210
+ label = String(current.getFullYear());
211
+ break;
212
+ }
213
+ boundaries.push({
214
+ start: periodStart,
215
+ end: periodEnd > endDate ? endDate : periodEnd,
216
+ label,
217
+ });
218
+ current = periodEnd;
219
+ }
220
+ return boundaries;
221
+ }
222
+ /**
223
+ * Get ISO week number
224
+ */
225
+ function getWeekNumber(date) {
226
+ const d = new Date(Date.UTC(date.getFullYear(), date.getMonth(), date.getDate()));
227
+ const dayNum = d.getUTCDay() || 7;
228
+ d.setUTCDate(d.getUTCDate() + 4 - dayNum);
229
+ const yearStart = new Date(Date.UTC(d.getUTCFullYear(), 0, 1));
230
+ return Math.ceil((((d.getTime() - yearStart.getTime()) / 86400000) + 1) / 7);
231
+ }
232
+ /**
233
+ * Calculate trends from history data
234
+ */
235
+ export async function calculateTrends(projectPath, options = {}) {
236
+ const { period = "week", startDate = new Date(Date.now() - 90 * 24 * 60 * 60 * 1000).toISOString(), // 90 days ago
237
+ endDate = new Date().toISOString(), } = options;
238
+ // Get all relevant history entries
239
+ const result = await queryHistory(projectPath, {
240
+ startDate,
241
+ endDate,
242
+ limit: 10000,
243
+ order: "asc",
244
+ });
245
+ const boundaries = getPeriodBoundaries(new Date(startDate), new Date(endDate), period);
246
+ const dataPoints = [];
247
+ const severities = ["critical", "high", "medium", "low", "info"];
248
+ for (const boundary of boundaries) {
249
+ const periodEntries = result.entries.filter((e) => {
250
+ const entryDate = new Date(e.timestamp);
251
+ return entryDate >= boundary.start && entryDate < boundary.end;
252
+ });
253
+ const certifications = periodEntries.filter((e) => e.type === "certification_completed");
254
+ const scans = periodEntries.filter((e) => e.type === "scan_completed");
255
+ const findings = periodEntries.filter((e) => e.type === "finding_submitted");
256
+ const fixed = periodEntries.filter((e) => e.type === "finding_fixed");
257
+ const bySeverity = {
258
+ critical: 0,
259
+ high: 0,
260
+ medium: 0,
261
+ low: 0,
262
+ info: 0,
263
+ };
264
+ for (const finding of findings) {
265
+ bySeverity[finding.severity] = (bySeverity[finding.severity] || 0) + 1;
266
+ }
267
+ const avgScore = certifications.length > 0
268
+ ? certifications.reduce((sum, c) => sum + c.score, 0) / certifications.length
269
+ : 0;
270
+ dataPoints.push({
271
+ period: boundary.label,
272
+ startDate: boundary.start.toISOString(),
273
+ endDate: boundary.end.toISOString(),
274
+ certificationCount: certifications.length,
275
+ scanCount: scans.length,
276
+ findingsTotal: findings.length,
277
+ findingsFixed: fixed.length,
278
+ avgScore: Math.round(avgScore * 100) / 100,
279
+ bySeverity,
280
+ });
281
+ }
282
+ // Calculate summary
283
+ const totalCertifications = dataPoints.reduce((sum, dp) => sum + dp.certificationCount, 0);
284
+ const totalScans = dataPoints.reduce((sum, dp) => sum + dp.scanCount, 0);
285
+ const totalFindings = dataPoints.reduce((sum, dp) => sum + dp.findingsTotal, 0);
286
+ const totalFixed = dataPoints.reduce((sum, dp) => sum + dp.findingsFixed, 0);
287
+ const scoredPoints = dataPoints.filter((dp) => dp.avgScore > 0);
288
+ const avgScore = scoredPoints.length > 0
289
+ ? scoredPoints.reduce((sum, dp) => sum + dp.avgScore, 0) / scoredPoints.length
290
+ : 0;
291
+ // Calculate change (compare first half to second half)
292
+ const midpoint = Math.floor(dataPoints.length / 2);
293
+ const firstHalf = dataPoints.slice(0, midpoint);
294
+ const secondHalf = dataPoints.slice(midpoint);
295
+ const firstHalfScore = firstHalf.filter(dp => dp.avgScore > 0).reduce((sum, dp) => sum + dp.avgScore, 0) / (firstHalf.filter(dp => dp.avgScore > 0).length || 1);
296
+ const secondHalfScore = secondHalf.filter(dp => dp.avgScore > 0).reduce((sum, dp) => sum + dp.avgScore, 0) / (secondHalf.filter(dp => dp.avgScore > 0).length || 1);
297
+ const firstHalfFindings = firstHalf.reduce((sum, dp) => sum + dp.findingsTotal, 0);
298
+ const secondHalfFindings = secondHalf.reduce((sum, dp) => sum + dp.findingsTotal, 0);
299
+ return {
300
+ projectPath,
301
+ period,
302
+ dataPoints,
303
+ summary: {
304
+ totalCertifications,
305
+ totalScans,
306
+ totalFindings,
307
+ totalFixed,
308
+ avgScore: Math.round(avgScore * 100) / 100,
309
+ scoreChange: Math.round((secondHalfScore - firstHalfScore) * 100) / 100,
310
+ findingsChange: secondHalfFindings - firstHalfFindings,
311
+ },
312
+ };
313
+ }
314
+ /**
315
+ * Get recent activity summary
316
+ */
317
+ export async function getRecentActivity(projectPath, days = 7) {
318
+ const startDate = new Date(Date.now() - days * 24 * 60 * 60 * 1000).toISOString();
319
+ const result = await queryHistory(projectPath, {
320
+ startDate,
321
+ limit: 1000,
322
+ order: "desc",
323
+ });
324
+ const certifications = result.entries.filter((e) => e.type === "certification_completed");
325
+ const scans = result.entries.filter((e) => e.type === "scan_completed");
326
+ const findings = result.entries.filter((e) => e.type === "finding_submitted");
327
+ const fixed = result.entries.filter((e) => e.type === "finding_fixed");
328
+ const latest = certifications[0];
329
+ return {
330
+ certifications: certifications.length,
331
+ scans: scans.length,
332
+ findings: findings.length,
333
+ fixed: fixed.length,
334
+ latestCertification: latest
335
+ ? {
336
+ id: latest.certificationId,
337
+ score: latest.score,
338
+ level: latest.level,
339
+ timestamp: latest.timestamp,
340
+ }
341
+ : undefined,
342
+ };
343
+ }
344
+ /**
345
+ * Format trends as markdown
346
+ */
347
+ export function formatTrendsAsMarkdown(analysis) {
348
+ const lines = [
349
+ `# Security Trends Report`,
350
+ "",
351
+ `**Project**: ${analysis.projectPath}`,
352
+ `**Period**: ${analysis.period}`,
353
+ `**Data Points**: ${analysis.dataPoints.length}`,
354
+ "",
355
+ "## Summary",
356
+ "",
357
+ `| Metric | Value |`,
358
+ `|--------|-------|`,
359
+ `| Total Certifications | ${analysis.summary.totalCertifications} |`,
360
+ `| Total Scans | ${analysis.summary.totalScans} |`,
361
+ `| Total Findings | ${analysis.summary.totalFindings} |`,
362
+ `| Findings Fixed | ${analysis.summary.totalFixed} |`,
363
+ `| Average Score | ${analysis.summary.avgScore} |`,
364
+ `| Score Change | ${analysis.summary.scoreChange >= 0 ? "+" : ""}${analysis.summary.scoreChange} |`,
365
+ `| Findings Change | ${analysis.summary.findingsChange >= 0 ? "+" : ""}${analysis.summary.findingsChange} |`,
366
+ "",
367
+ "## Trend Data",
368
+ "",
369
+ "| Period | Certs | Scans | Findings | Fixed | Avg Score | Critical | High |",
370
+ "|--------|-------|-------|----------|-------|-----------|----------|------|",
371
+ ];
372
+ for (const dp of analysis.dataPoints) {
373
+ lines.push(`| ${dp.period} | ${dp.certificationCount} | ${dp.scanCount} | ${dp.findingsTotal} | ${dp.findingsFixed} | ${dp.avgScore || "-"} | ${dp.bySeverity.critical} | ${dp.bySeverity.high} |`);
374
+ }
375
+ return lines.join("\n");
376
+ }
377
+ /**
378
+ * Get history file stats
379
+ */
380
+ export async function getHistoryStats(projectPath) {
381
+ const historyFile = join(projectPath, HISTORY_DIR, HISTORY_FILE);
382
+ try {
383
+ await access(historyFile);
384
+ }
385
+ catch {
386
+ return { exists: false, entries: 0, sizeBytes: 0 };
387
+ }
388
+ const stats = await stat(historyFile);
389
+ const result = await queryHistory(projectPath, { limit: 1, order: "asc" });
390
+ const newest = await queryHistory(projectPath, { limit: 1, order: "desc" });
391
+ return {
392
+ exists: true,
393
+ entries: result.total,
394
+ sizeBytes: stats.size,
395
+ oldestEntry: result.entries[0]?.timestamp,
396
+ newestEntry: newest.entries[0]?.timestamp,
397
+ };
398
+ }
399
+ //# sourceMappingURL=store.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"store.js","sourceRoot":"","sources":["../../src/history/store.ts"],"names":[],"mappings":"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"}