@ryanstark24/sfgraph-server 1.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (261) hide show
  1. package/dist/__tests__/context-cache.test.d.ts +2 -0
  2. package/dist/__tests__/context-cache.test.d.ts.map +1 -0
  3. package/dist/__tests__/context-cache.test.js +77 -0
  4. package/dist/__tests__/context-cache.test.js.map +1 -0
  5. package/dist/__tests__/context-default-factory.test.d.ts +2 -0
  6. package/dist/__tests__/context-default-factory.test.d.ts.map +1 -0
  7. package/dist/__tests__/context-default-factory.test.js +89 -0
  8. package/dist/__tests__/context-default-factory.test.js.map +1 -0
  9. package/dist/__tests__/network-egress.test.d.ts +2 -0
  10. package/dist/__tests__/network-egress.test.d.ts.map +1 -0
  11. package/dist/__tests__/network-egress.test.js +33 -0
  12. package/dist/__tests__/network-egress.test.js.map +1 -0
  13. package/dist/__tests__/shutdown.test.d.ts +2 -0
  14. package/dist/__tests__/shutdown.test.d.ts.map +1 -0
  15. package/dist/__tests__/shutdown.test.js +78 -0
  16. package/dist/__tests__/shutdown.test.js.map +1 -0
  17. package/dist/__tests__/tool-registry.test.d.ts +2 -0
  18. package/dist/__tests__/tool-registry.test.d.ts.map +1 -0
  19. package/dist/__tests__/tool-registry.test.js +21 -0
  20. package/dist/__tests__/tool-registry.test.js.map +1 -0
  21. package/dist/__tests__/zod-schema.test.d.ts +2 -0
  22. package/dist/__tests__/zod-schema.test.d.ts.map +1 -0
  23. package/dist/__tests__/zod-schema.test.js +15 -0
  24. package/dist/__tests__/zod-schema.test.js.map +1 -0
  25. package/dist/bin.d.ts +2 -0
  26. package/dist/bin.d.ts.map +1 -0
  27. package/dist/bin.js +14 -0
  28. package/dist/bin.js.map +1 -0
  29. package/dist/context.d.ts +29 -0
  30. package/dist/context.d.ts.map +1 -0
  31. package/dist/context.js +202 -0
  32. package/dist/context.js.map +1 -0
  33. package/dist/index.d.ts +8 -0
  34. package/dist/index.d.ts.map +1 -0
  35. package/dist/index.js +6 -0
  36. package/dist/index.js.map +1 -0
  37. package/dist/server.d.ts +31 -0
  38. package/dist/server.d.ts.map +1 -0
  39. package/dist/server.js +91 -0
  40. package/dist/server.js.map +1 -0
  41. package/dist/shutdown.d.ts +10 -0
  42. package/dist/shutdown.d.ts.map +1 -0
  43. package/dist/shutdown.js +43 -0
  44. package/dist/shutdown.js.map +1 -0
  45. package/dist/tool-registry.d.ts +31 -0
  46. package/dist/tool-registry.d.ts.map +1 -0
  47. package/dist/tool-registry.js +29 -0
  48. package/dist/tool-registry.js.map +1 -0
  49. package/dist/tools/__tests__/_fixture.d.ts +29 -0
  50. package/dist/tools/__tests__/_fixture.d.ts.map +1 -0
  51. package/dist/tools/__tests__/_fixture.js +59 -0
  52. package/dist/tools/__tests__/_fixture.js.map +1 -0
  53. package/dist/tools/__tests__/_runner.d.ts +10 -0
  54. package/dist/tools/__tests__/_runner.d.ts.map +1 -0
  55. package/dist/tools/__tests__/_runner.js +12 -0
  56. package/dist/tools/__tests__/_runner.js.map +1 -0
  57. package/dist/tools/__tests__/analyze_field.test.d.ts +2 -0
  58. package/dist/tools/__tests__/analyze_field.test.d.ts.map +1 -0
  59. package/dist/tools/__tests__/analyze_field.test.js +92 -0
  60. package/dist/tools/__tests__/analyze_field.test.js.map +1 -0
  61. package/dist/tools/__tests__/cross_layer_flow_map.test.d.ts +2 -0
  62. package/dist/tools/__tests__/cross_layer_flow_map.test.d.ts.map +1 -0
  63. package/dist/tools/__tests__/cross_layer_flow_map.test.js +111 -0
  64. package/dist/tools/__tests__/cross_layer_flow_map.test.js.map +1 -0
  65. package/dist/tools/__tests__/cross_org_diff.test.d.ts +2 -0
  66. package/dist/tools/__tests__/cross_org_diff.test.d.ts.map +1 -0
  67. package/dist/tools/__tests__/cross_org_diff.test.js +112 -0
  68. package/dist/tools/__tests__/cross_org_diff.test.js.map +1 -0
  69. package/dist/tools/__tests__/dead_code_audit.test.d.ts +2 -0
  70. package/dist/tools/__tests__/dead_code_audit.test.d.ts.map +1 -0
  71. package/dist/tools/__tests__/dead_code_audit.test.js +71 -0
  72. package/dist/tools/__tests__/dead_code_audit.test.js.map +1 -0
  73. package/dist/tools/__tests__/deployment_manifest_gen.test.d.ts +2 -0
  74. package/dist/tools/__tests__/deployment_manifest_gen.test.d.ts.map +1 -0
  75. package/dist/tools/__tests__/deployment_manifest_gen.test.js +143 -0
  76. package/dist/tools/__tests__/deployment_manifest_gen.test.js.map +1 -0
  77. package/dist/tools/__tests__/dispatcher.test.d.ts +2 -0
  78. package/dist/tools/__tests__/dispatcher.test.d.ts.map +1 -0
  79. package/dist/tools/__tests__/dispatcher.test.js +39 -0
  80. package/dist/tools/__tests__/dispatcher.test.js.map +1 -0
  81. package/dist/tools/__tests__/explain_code.test.d.ts +2 -0
  82. package/dist/tools/__tests__/explain_code.test.d.ts.map +1 -0
  83. package/dist/tools/__tests__/explain_code.test.js +88 -0
  84. package/dist/tools/__tests__/explain_code.test.js.map +1 -0
  85. package/dist/tools/__tests__/freshness_report.test.d.ts +2 -0
  86. package/dist/tools/__tests__/freshness_report.test.d.ts.map +1 -0
  87. package/dist/tools/__tests__/freshness_report.test.js +103 -0
  88. package/dist/tools/__tests__/freshness_report.test.js.map +1 -0
  89. package/dist/tools/__tests__/get_ingest_job.test.d.ts +2 -0
  90. package/dist/tools/__tests__/get_ingest_job.test.d.ts.map +1 -0
  91. package/dist/tools/__tests__/get_ingest_job.test.js +42 -0
  92. package/dist/tools/__tests__/get_ingest_job.test.js.map +1 -0
  93. package/dist/tools/__tests__/governor_risk_check.test.d.ts +2 -0
  94. package/dist/tools/__tests__/governor_risk_check.test.d.ts.map +1 -0
  95. package/dist/tools/__tests__/governor_risk_check.test.js +66 -0
  96. package/dist/tools/__tests__/governor_risk_check.test.js.map +1 -0
  97. package/dist/tools/__tests__/impact_from_git_diff.test.d.ts +2 -0
  98. package/dist/tools/__tests__/impact_from_git_diff.test.d.ts.map +1 -0
  99. package/dist/tools/__tests__/impact_from_git_diff.test.js +115 -0
  100. package/dist/tools/__tests__/impact_from_git_diff.test.js.map +1 -0
  101. package/dist/tools/__tests__/list_orgs.test.d.ts +2 -0
  102. package/dist/tools/__tests__/list_orgs.test.d.ts.map +1 -0
  103. package/dist/tools/__tests__/list_orgs.test.js +137 -0
  104. package/dist/tools/__tests__/list_orgs.test.js.map +1 -0
  105. package/dist/tools/__tests__/ping.test.d.ts +2 -0
  106. package/dist/tools/__tests__/ping.test.d.ts.map +1 -0
  107. package/dist/tools/__tests__/ping.test.js +10 -0
  108. package/dist/tools/__tests__/ping.test.js.map +1 -0
  109. package/dist/tools/__tests__/point_in_time_diff.test.d.ts +2 -0
  110. package/dist/tools/__tests__/point_in_time_diff.test.d.ts.map +1 -0
  111. package/dist/tools/__tests__/point_in_time_diff.test.js +64 -0
  112. package/dist/tools/__tests__/point_in_time_diff.test.js.map +1 -0
  113. package/dist/tools/__tests__/security_audit.test.d.ts +2 -0
  114. package/dist/tools/__tests__/security_audit.test.d.ts.map +1 -0
  115. package/dist/tools/__tests__/security_audit.test.js +99 -0
  116. package/dist/tools/__tests__/security_audit.test.js.map +1 -0
  117. package/dist/tools/__tests__/snapshot.test.d.ts +2 -0
  118. package/dist/tools/__tests__/snapshot.test.d.ts.map +1 -0
  119. package/dist/tools/__tests__/snapshot.test.js +54 -0
  120. package/dist/tools/__tests__/snapshot.test.js.map +1 -0
  121. package/dist/tools/__tests__/staleness_check.test.d.ts +2 -0
  122. package/dist/tools/__tests__/staleness_check.test.d.ts.map +1 -0
  123. package/dist/tools/__tests__/staleness_check.test.js +51 -0
  124. package/dist/tools/__tests__/staleness_check.test.js.map +1 -0
  125. package/dist/tools/__tests__/start_ingest_job.test.d.ts +2 -0
  126. package/dist/tools/__tests__/start_ingest_job.test.d.ts.map +1 -0
  127. package/dist/tools/__tests__/start_ingest_job.test.js +55 -0
  128. package/dist/tools/__tests__/start_ingest_job.test.js.map +1 -0
  129. package/dist/tools/__tests__/test_gap_intelligence.test.d.ts +2 -0
  130. package/dist/tools/__tests__/test_gap_intelligence.test.d.ts.map +1 -0
  131. package/dist/tools/__tests__/test_gap_intelligence.test.js +77 -0
  132. package/dist/tools/__tests__/test_gap_intelligence.test.js.map +1 -0
  133. package/dist/tools/__tests__/trace.test.d.ts +2 -0
  134. package/dist/tools/__tests__/trace.test.d.ts.map +1 -0
  135. package/dist/tools/__tests__/trace.test.js +130 -0
  136. package/dist/tools/__tests__/trace.test.js.map +1 -0
  137. package/dist/tools/__tests__/what_broke.test.d.ts +2 -0
  138. package/dist/tools/__tests__/what_broke.test.d.ts.map +1 -0
  139. package/dist/tools/__tests__/what_broke.test.js +60 -0
  140. package/dist/tools/__tests__/what_broke.test.js.map +1 -0
  141. package/dist/tools/__tests__/wip.test.d.ts +2 -0
  142. package/dist/tools/__tests__/wip.test.d.ts.map +1 -0
  143. package/dist/tools/__tests__/wip.test.js +173 -0
  144. package/dist/tools/__tests__/wip.test.js.map +1 -0
  145. package/dist/tools/_define.d.ts +11 -0
  146. package/dist/tools/_define.d.ts.map +1 -0
  147. package/dist/tools/_define.js +29 -0
  148. package/dist/tools/_define.js.map +1 -0
  149. package/dist/tools/_job-store.d.ts +19 -0
  150. package/dist/tools/_job-store.d.ts.map +1 -0
  151. package/dist/tools/_job-store.js +26 -0
  152. package/dist/tools/_job-store.js.map +1 -0
  153. package/dist/tools/_project-root.d.ts +20 -0
  154. package/dist/tools/_project-root.d.ts.map +1 -0
  155. package/dist/tools/_project-root.js +44 -0
  156. package/dist/tools/_project-root.js.map +1 -0
  157. package/dist/tools/analyze_field.d.ts +2 -0
  158. package/dist/tools/analyze_field.d.ts.map +1 -0
  159. package/dist/tools/analyze_field.js +77 -0
  160. package/dist/tools/analyze_field.js.map +1 -0
  161. package/dist/tools/cross_layer_flow_map.d.ts +2 -0
  162. package/dist/tools/cross_layer_flow_map.d.ts.map +1 -0
  163. package/dist/tools/cross_layer_flow_map.js +104 -0
  164. package/dist/tools/cross_layer_flow_map.js.map +1 -0
  165. package/dist/tools/cross_org_diff.d.ts +2 -0
  166. package/dist/tools/cross_org_diff.d.ts.map +1 -0
  167. package/dist/tools/cross_org_diff.js +59 -0
  168. package/dist/tools/cross_org_diff.js.map +1 -0
  169. package/dist/tools/dead_code_audit.d.ts +2 -0
  170. package/dist/tools/dead_code_audit.d.ts.map +1 -0
  171. package/dist/tools/dead_code_audit.js +67 -0
  172. package/dist/tools/dead_code_audit.js.map +1 -0
  173. package/dist/tools/deployment_manifest_gen.d.ts +2 -0
  174. package/dist/tools/deployment_manifest_gen.d.ts.map +1 -0
  175. package/dist/tools/deployment_manifest_gen.js +43 -0
  176. package/dist/tools/deployment_manifest_gen.js.map +1 -0
  177. package/dist/tools/explain_code.d.ts +2 -0
  178. package/dist/tools/explain_code.d.ts.map +1 -0
  179. package/dist/tools/explain_code.js +109 -0
  180. package/dist/tools/explain_code.js.map +1 -0
  181. package/dist/tools/freshness_report.d.ts +2 -0
  182. package/dist/tools/freshness_report.d.ts.map +1 -0
  183. package/dist/tools/freshness_report.js +66 -0
  184. package/dist/tools/freshness_report.js.map +1 -0
  185. package/dist/tools/get_ingest_job.d.ts +2 -0
  186. package/dist/tools/get_ingest_job.d.ts.map +1 -0
  187. package/dist/tools/get_ingest_job.js +24 -0
  188. package/dist/tools/get_ingest_job.js.map +1 -0
  189. package/dist/tools/governor_risk_check.d.ts +2 -0
  190. package/dist/tools/governor_risk_check.d.ts.map +1 -0
  191. package/dist/tools/governor_risk_check.js +50 -0
  192. package/dist/tools/governor_risk_check.js.map +1 -0
  193. package/dist/tools/impact_from_git_diff.d.ts +2 -0
  194. package/dist/tools/impact_from_git_diff.d.ts.map +1 -0
  195. package/dist/tools/impact_from_git_diff.js +67 -0
  196. package/dist/tools/impact_from_git_diff.js.map +1 -0
  197. package/dist/tools/index.d.ts +27 -0
  198. package/dist/tools/index.d.ts.map +1 -0
  199. package/dist/tools/index.js +28 -0
  200. package/dist/tools/index.js.map +1 -0
  201. package/dist/tools/list_orgs.d.ts +30 -0
  202. package/dist/tools/list_orgs.d.ts.map +1 -0
  203. package/dist/tools/list_orgs.js +302 -0
  204. package/dist/tools/list_orgs.js.map +1 -0
  205. package/dist/tools/ping.d.ts +3 -0
  206. package/dist/tools/ping.d.ts.map +1 -0
  207. package/dist/tools/ping.js +23 -0
  208. package/dist/tools/ping.js.map +1 -0
  209. package/dist/tools/point_in_time_diff.d.ts +2 -0
  210. package/dist/tools/point_in_time_diff.d.ts.map +1 -0
  211. package/dist/tools/point_in_time_diff.js +37 -0
  212. package/dist/tools/point_in_time_diff.js.map +1 -0
  213. package/dist/tools/security_audit.d.ts +2 -0
  214. package/dist/tools/security_audit.d.ts.map +1 -0
  215. package/dist/tools/security_audit.js +56 -0
  216. package/dist/tools/security_audit.js.map +1 -0
  217. package/dist/tools/snapshot_create.d.ts +2 -0
  218. package/dist/tools/snapshot_create.d.ts.map +1 -0
  219. package/dist/tools/snapshot_create.js +24 -0
  220. package/dist/tools/snapshot_create.js.map +1 -0
  221. package/dist/tools/snapshot_list.d.ts +2 -0
  222. package/dist/tools/snapshot_list.d.ts.map +1 -0
  223. package/dist/tools/snapshot_list.js +22 -0
  224. package/dist/tools/snapshot_list.js.map +1 -0
  225. package/dist/tools/staleness_check.d.ts +2 -0
  226. package/dist/tools/staleness_check.d.ts.map +1 -0
  227. package/dist/tools/staleness_check.js +50 -0
  228. package/dist/tools/staleness_check.js.map +1 -0
  229. package/dist/tools/start_ingest_job.d.ts +2 -0
  230. package/dist/tools/start_ingest_job.d.ts.map +1 -0
  231. package/dist/tools/start_ingest_job.js +83 -0
  232. package/dist/tools/start_ingest_job.js.map +1 -0
  233. package/dist/tools/test_gap_intelligence_from_git_diff.d.ts +2 -0
  234. package/dist/tools/test_gap_intelligence_from_git_diff.d.ts.map +1 -0
  235. package/dist/tools/test_gap_intelligence_from_git_diff.js +50 -0
  236. package/dist/tools/test_gap_intelligence_from_git_diff.js.map +1 -0
  237. package/dist/tools/trace_downstream.d.ts +2 -0
  238. package/dist/tools/trace_downstream.d.ts.map +1 -0
  239. package/dist/tools/trace_downstream.js +43 -0
  240. package/dist/tools/trace_downstream.js.map +1 -0
  241. package/dist/tools/trace_upstream.d.ts +2 -0
  242. package/dist/tools/trace_upstream.d.ts.map +1 -0
  243. package/dist/tools/trace_upstream.js +43 -0
  244. package/dist/tools/trace_upstream.js.map +1 -0
  245. package/dist/tools/what_broke.d.ts +2 -0
  246. package/dist/tools/what_broke.d.ts.map +1 -0
  247. package/dist/tools/what_broke.js +86 -0
  248. package/dist/tools/what_broke.js.map +1 -0
  249. package/dist/tools/wip_diff.d.ts +2 -0
  250. package/dist/tools/wip_diff.d.ts.map +1 -0
  251. package/dist/tools/wip_diff.js +55 -0
  252. package/dist/tools/wip_diff.js.map +1 -0
  253. package/dist/tools/wip_impact.d.ts +2 -0
  254. package/dist/tools/wip_impact.d.ts.map +1 -0
  255. package/dist/tools/wip_impact.js +55 -0
  256. package/dist/tools/wip_impact.js.map +1 -0
  257. package/dist/tools/wip_test_gap.d.ts +2 -0
  258. package/dist/tools/wip_test_gap.d.ts.map +1 -0
  259. package/dist/tools/wip_test_gap.js +50 -0
  260. package/dist/tools/wip_test_gap.js.map +1 -0
  261. package/package.json +61 -0
@@ -0,0 +1,83 @@
1
+ import { validateOrgIdentifier } from "@ryanstark24/sfgraph-shared";
2
+ import { defineTool, z } from "./_define.js";
3
+ // Reject all C0 control characters (0x00-0x1F) and DEL (0x7F). Built from
4
+ // a code-point check so the source file stays clean of literal control bytes.
5
+ function hasControlChar(s) {
6
+ for (let i = 0; i < s.length; i++) {
7
+ const c = s.charCodeAt(i);
8
+ if (c < 0x20 || c === 0x7f)
9
+ return true;
10
+ }
11
+ return false;
12
+ }
13
+ const sourceSchema = z.discriminatedUnion("type", [
14
+ z.object({
15
+ type: z.literal("live-org"),
16
+ alias: z
17
+ .string()
18
+ .min(1)
19
+ .refine((s) => {
20
+ try {
21
+ validateOrgIdentifier(s);
22
+ return true;
23
+ }
24
+ catch {
25
+ return false;
26
+ }
27
+ }, { message: "alias must be a Salesforce 15/18-char id or a safe alias" }),
28
+ }),
29
+ z.object({
30
+ type: z.literal("filesystem"),
31
+ // Reject control chars / newlines outright. The CLI command we render
32
+ // is meant to be copy-pasted; a path containing a newline or a NUL
33
+ // could splice a second shell command into the agent's clipboard.
34
+ path: z
35
+ .string()
36
+ .min(1)
37
+ .refine((p) => !hasControlChar(p), { message: "path contains control characters" }),
38
+ }),
39
+ ]);
40
+ const inputSchema = z.object({
41
+ source: sourceSchema,
42
+ mode: z.enum(["full", "incremental", "auto"]).default("auto"),
43
+ });
44
+ /**
45
+ * POSIX single-quote shell escape. Wraps the value in `'...'`, replacing
46
+ * embedded single quotes with `'\''`. Safe to copy-paste into bash/zsh.
47
+ */
48
+ function shellQuotePosix(value) {
49
+ return `'${value.replace(/'/g, "'\\''")}'`;
50
+ }
51
+ // Stays in the registry so agents can `tools/list` and discover the right
52
+ // out-of-band call to make, but refuses to enqueue. The previous version
53
+ // returned a fake queued record that would never advance — agents would
54
+ // poll `get_ingest_job` forever waiting for a worker that doesn't exist.
55
+ defineTool({
56
+ name: "start_ingest_job",
57
+ description: "RETURNS INSTRUCTIONS ONLY. The MCP server does NOT run ingest workers. To actually ingest, run `sfgraph ingest --org <alias>` in a shell. Calling this tool now returns the exact shell command to run rather than enqueueing a phantom job.",
58
+ inputSchema,
59
+ async execute(input) {
60
+ const aliasHint = input.source.type === "live-org" ? input.source.alias : null;
61
+ const cmd = input.source.type === "live-org"
62
+ ? // Alias already validated by validateOrgIdentifier; safe to interpolate.
63
+ `sfgraph ingest --org ${input.source.alias}${input.mode !== "auto" ? ` --mode ${input.mode}` : ""}`
64
+ : // Path may contain spaces / shell metacharacters — quote it.
65
+ `sfgraph ingest --from-fs ${shellQuotePosix(input.source.path)}`;
66
+ return {
67
+ summary: "ingest must run out-of-band",
68
+ markdown: [
69
+ "> **The MCP server cannot start ingests itself.** Run the command below in a shell.",
70
+ "",
71
+ "```bash",
72
+ cmd,
73
+ "```",
74
+ "",
75
+ aliasHint
76
+ ? `When it completes, MCP tools like \`trace_upstream\` / \`what_broke\` against \`${aliasHint}\` will see the new data.`
77
+ : "When it completes, MCP tools will see the new data on next invocation.",
78
+ ].join("\n"),
79
+ data: { executed: false, run_this_command: cmd },
80
+ };
81
+ },
82
+ });
83
+ //# sourceMappingURL=start_ingest_job.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"start_ingest_job.js","sourceRoot":"","sources":["../../src/tools/start_ingest_job.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=test_gap_intelligence_from_git_diff.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"test_gap_intelligence_from_git_diff.d.ts","sourceRoot":"","sources":["../../src/tools/test_gap_intelligence_from_git_diff.ts"],"names":[],"mappings":""}
@@ -0,0 +1,50 @@
1
+ import { analyze } from "@ryanstark24/sfgraph-core";
2
+ import { asQualifiedName } from "@ryanstark24/sfgraph-shared";
3
+ import { getToolContext } from "../context.js";
4
+ import { defineTool, z } from "./_define.js";
5
+ const inputSchema = z.object({
6
+ org: z.string().min(1),
7
+ diff: z.string().min(1),
8
+ });
9
+ defineTool({
10
+ name: "test_gap_intelligence_from_git_diff",
11
+ description: "List dependents of a git diff that have no test coverage.",
12
+ inputSchema,
13
+ async execute(input) {
14
+ const ctx = await getToolContext({ orgId: input.org });
15
+ const files = analyze.parseUnifiedDiff(input.diff);
16
+ const seedQnames = [];
17
+ for (const f of files) {
18
+ const qn = analyze.pathToQualifiedName(f.path);
19
+ if (qn)
20
+ seedQnames.push(qn);
21
+ }
22
+ const gaps = [];
23
+ const covered = [];
24
+ const seen = new Set();
25
+ for (const qn of seedQnames) {
26
+ const dep = analyze.findDependents(ctx.graphStore, ctx.orgId, asQualifiedName(qn), 3);
27
+ for (const n of dep.nodes) {
28
+ if (seen.has(n.qualifiedName))
29
+ continue;
30
+ seen.add(n.qualifiedName);
31
+ const hasTest = analyze.hasTestCoverage(ctx.graphStore, ctx.orgId, asQualifiedName(n.qualifiedName));
32
+ if (hasTest)
33
+ covered.push(n.qualifiedName);
34
+ else
35
+ gaps.push(n.qualifiedName);
36
+ }
37
+ }
38
+ const md = [
39
+ `**Test gaps:** ${gaps.length}`,
40
+ "",
41
+ ...gaps.slice(0, 50).map((g) => `- \`${g}\``),
42
+ ].join("\n");
43
+ return {
44
+ summary: `${gaps.length} dependents lack tests`,
45
+ markdown: md,
46
+ data: { gaps, covered },
47
+ };
48
+ },
49
+ });
50
+ //# sourceMappingURL=test_gap_intelligence_from_git_diff.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"test_gap_intelligence_from_git_diff.js","sourceRoot":"","sources":["../../src/tools/test_gap_intelligence_from_git_diff.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,2BAA2B,CAAC;AACpD,OAAO,EAAE,eAAe,EAAE,MAAM,6BAA6B,CAAC;AAC9D,OAAO,EAAE,cAAc,EAAE,MAAM,eAAe,CAAC;AAC/C,OAAO,EAAE,UAAU,EAAE,CAAC,EAAE,MAAM,cAAc,CAAC;AAE7C,MAAM,WAAW,GAAG,CAAC,CAAC,MAAM,CAAC;IAC3B,GAAG,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC;IACtB,IAAI,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC;CACxB,CAAC,CAAC;AAEH,UAAU,CAAC;IACT,IAAI,EAAE,qCAAqC;IAC3C,WAAW,EAAE,2DAA2D;IACxE,WAAW;IACX,KAAK,CAAC,OAAO,CAAC,KAAK;QACjB,MAAM,GAAG,GAAG,MAAM,cAAc,CAAC,EAAE,KAAK,EAAE,KAAK,CAAC,GAAG,EAAE,CAAC,CAAC;QACvD,MAAM,KAAK,GAAG,OAAO,CAAC,gBAAgB,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QACnD,MAAM,UAAU,GAAa,EAAE,CAAC;QAChC,KAAK,MAAM,CAAC,IAAI,KAAK,EAAE,CAAC;YACtB,MAAM,EAAE,GAAG,OAAO,CAAC,mBAAmB,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;YAC/C,IAAI,EAAE;gBAAE,UAAU,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAC9B,CAAC;QACD,MAAM,IAAI,GAAa,EAAE,CAAC;QAC1B,MAAM,OAAO,GAAa,EAAE,CAAC;QAC7B,MAAM,IAAI,GAAG,IAAI,GAAG,EAAU,CAAC;QAC/B,KAAK,MAAM,EAAE,IAAI,UAAU,EAAE,CAAC;YAC5B,MAAM,GAAG,GAAG,OAAO,CAAC,cAAc,CAAC,GAAG,CAAC,UAAU,EAAE,GAAG,CAAC,KAAK,EAAE,eAAe,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC;YACtF,KAAK,MAAM,CAAC,IAAI,GAAG,CAAC,KAAK,EAAE,CAAC;gBAC1B,IAAI,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,aAAa,CAAC;oBAAE,SAAS;gBACxC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,aAAa,CAAC,CAAC;gBAC1B,MAAM,OAAO,GAAG,OAAO,CAAC,eAAe,CACrC,GAAG,CAAC,UAAU,EACd,GAAG,CAAC,KAAK,EACT,eAAe,CAAC,CAAC,CAAC,aAAa,CAAC,CACjC,CAAC;gBACF,IAAI,OAAO;oBAAE,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,aAAa,CAAC,CAAC;;oBACtC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,aAAa,CAAC,CAAC;YAClC,CAAC;QACH,CAAC;QACD,MAAM,EAAE,GAAG;YACT,kBAAkB,IAAI,CAAC,MAAM,EAAE;YAC/B,EAAE;YACF,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC;SAC9C,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACb,OAAO;YACL,OAAO,EAAE,GAAG,IAAI,CAAC,MAAM,wBAAwB;YAC/C,QAAQ,EAAE,EAAE;YACZ,IAAI,EAAE,EAAE,IAAI,EAAE,OAAO,EAAE;SACxB,CAAC;IACJ,CAAC;CACF,CAAC,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=trace_downstream.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"trace_downstream.d.ts","sourceRoot":"","sources":["../../src/tools/trace_downstream.ts"],"names":[],"mappings":""}
@@ -0,0 +1,43 @@
1
+ import { analyze, render } from "@ryanstark24/sfgraph-core";
2
+ import { asQualifiedName } from "@ryanstark24/sfgraph-shared";
3
+ import { getToolContext } from "../context.js";
4
+ import { defineTool, z } from "./_define.js";
5
+ const inputSchema = z.object({
6
+ org: z.string().min(1),
7
+ qname: z.string().min(1),
8
+ depth: z.number().int().min(1).max(5).default(3),
9
+ });
10
+ defineTool({
11
+ name: "trace_downstream",
12
+ description: "USE THIS for any 'what does X depend on' / 'show dependencies' / 'what does this method call' question about a Salesforce metadata node (ApexClass, LWC, Flow, CustomField, etc.). Forward-edge graph N hops out.",
13
+ inputSchema,
14
+ async execute(input) {
15
+ const ctx = await getToolContext({ orgId: input.org });
16
+ const qname = asQualifiedName(input.qname);
17
+ const r = analyze.findDependencies(ctx.graphStore, ctx.orgId, qname, input.depth);
18
+ const nodes = [
19
+ { qualifiedName: qname, label: "Target" },
20
+ ...r.nodes.map((n) => ({ qualifiedName: n.qualifiedName, label: n.label })),
21
+ ];
22
+ const mermaid = render.renderDependencyGraph({
23
+ nodes,
24
+ edges: r.edges.map((e) => ({
25
+ srcQualifiedName: e.srcQualifiedName,
26
+ dstQualifiedName: e.dstQualifiedName,
27
+ relType: e.relType,
28
+ })),
29
+ title: "downstream",
30
+ });
31
+ const truncated = r.truncated === true;
32
+ const mdLines = ["```mermaid", mermaid, "```"];
33
+ if (truncated) {
34
+ mdLines.push("", "_truncated_ — traversal hit the node cap; deeper downstream paths were not explored. Lower `depth` or pick a narrower `qname`.");
35
+ }
36
+ return {
37
+ summary: `${r.nodes.length} downstream nodes${truncated ? " (truncated)" : ""}`,
38
+ markdown: mdLines.join("\n"),
39
+ data: { nodes: r.nodes, edges: r.edges, truncated },
40
+ };
41
+ },
42
+ });
43
+ //# sourceMappingURL=trace_downstream.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"trace_downstream.js","sourceRoot":"","sources":["../../src/tools/trace_downstream.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,EAAE,MAAM,2BAA2B,CAAC;AAC5D,OAAO,EAAE,eAAe,EAAE,MAAM,6BAA6B,CAAC;AAC9D,OAAO,EAAE,cAAc,EAAE,MAAM,eAAe,CAAC;AAC/C,OAAO,EAAE,UAAU,EAAE,CAAC,EAAE,MAAM,cAAc,CAAC;AAE7C,MAAM,WAAW,GAAG,CAAC,CAAC,MAAM,CAAC;IAC3B,GAAG,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC;IACtB,KAAK,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC;IACxB,KAAK,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC;CACjD,CAAC,CAAC;AAEH,UAAU,CAAC;IACT,IAAI,EAAE,kBAAkB;IACxB,WAAW,EACT,mNAAmN;IACrN,WAAW;IACX,KAAK,CAAC,OAAO,CAAC,KAAK;QACjB,MAAM,GAAG,GAAG,MAAM,cAAc,CAAC,EAAE,KAAK,EAAE,KAAK,CAAC,GAAG,EAAE,CAAC,CAAC;QACvD,MAAM,KAAK,GAAG,eAAe,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;QAC3C,MAAM,CAAC,GAAG,OAAO,CAAC,gBAAgB,CAAC,GAAG,CAAC,UAAU,EAAE,GAAG,CAAC,KAAK,EAAE,KAAK,EAAE,KAAK,CAAC,KAAK,CAAC,CAAC;QAClF,MAAM,KAAK,GAAG;YACZ,EAAE,aAAa,EAAE,KAAK,EAAE,KAAK,EAAE,QAAQ,EAAE;YACzC,GAAG,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,EAAE,aAAa,EAAE,CAAC,CAAC,aAAa,EAAE,KAAK,EAAE,CAAC,CAAC,KAAK,EAAE,CAAC,CAAC;SAC5E,CAAC;QACF,MAAM,OAAO,GAAG,MAAM,CAAC,qBAAqB,CAAC;YAC3C,KAAK;YACL,KAAK,EAAE,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;gBACzB,gBAAgB,EAAE,CAAC,CAAC,gBAAgB;gBACpC,gBAAgB,EAAE,CAAC,CAAC,gBAAgB;gBACpC,OAAO,EAAE,CAAC,CAAC,OAAO;aACnB,CAAC,CAAC;YACH,KAAK,EAAE,YAAY;SACpB,CAAC,CAAC;QACH,MAAM,SAAS,GAAG,CAAC,CAAC,SAAS,KAAK,IAAI,CAAC;QACvC,MAAM,OAAO,GAAG,CAAC,YAAY,EAAE,OAAO,EAAE,KAAK,CAAC,CAAC;QAC/C,IAAI,SAAS,EAAE,CAAC;YACd,OAAO,CAAC,IAAI,CACV,EAAE,EACF,gIAAgI,CACjI,CAAC;QACJ,CAAC;QACD,OAAO;YACL,OAAO,EAAE,GAAG,CAAC,CAAC,KAAK,CAAC,MAAM,oBAAoB,SAAS,CAAC,CAAC,CAAC,cAAc,CAAC,CAAC,CAAC,EAAE,EAAE;YAC/E,QAAQ,EAAE,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC;YAC5B,IAAI,EAAE,EAAE,KAAK,EAAE,CAAC,CAAC,KAAK,EAAE,KAAK,EAAE,CAAC,CAAC,KAAK,EAAE,SAAS,EAAE;SACpD,CAAC;IACJ,CAAC;CACF,CAAC,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=trace_upstream.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"trace_upstream.d.ts","sourceRoot":"","sources":["../../src/tools/trace_upstream.ts"],"names":[],"mappings":""}
@@ -0,0 +1,43 @@
1
+ import { analyze, render } from "@ryanstark24/sfgraph-core";
2
+ import { asQualifiedName } from "@ryanstark24/sfgraph-shared";
3
+ import { getToolContext } from "../context.js";
4
+ import { defineTool, z } from "./_define.js";
5
+ const inputSchema = z.object({
6
+ org: z.string().min(1),
7
+ qname: z.string().min(1),
8
+ depth: z.number().int().min(1).max(5).default(3),
9
+ });
10
+ defineTool({
11
+ name: "trace_upstream",
12
+ description: "USE THIS for any 'what depends on X' / 'who uses this' / 'find callers of' / 'who references this method' question about a Salesforce metadata node. Returns every node that depends on the target, N hops back.",
13
+ inputSchema,
14
+ async execute(input) {
15
+ const ctx = await getToolContext({ orgId: input.org });
16
+ const qname = asQualifiedName(input.qname);
17
+ const r = analyze.findDependents(ctx.graphStore, ctx.orgId, qname, input.depth);
18
+ const nodes = [
19
+ { qualifiedName: qname, label: "Target" },
20
+ ...r.nodes.map((n) => ({ qualifiedName: n.qualifiedName, label: n.label })),
21
+ ];
22
+ const mermaid = render.renderDependencyGraph({
23
+ nodes,
24
+ edges: r.edges.map((e) => ({
25
+ srcQualifiedName: e.srcQualifiedName,
26
+ dstQualifiedName: e.dstQualifiedName,
27
+ relType: e.relType,
28
+ })),
29
+ title: "upstream",
30
+ });
31
+ const truncated = r.truncated === true;
32
+ const mdLines = ["```mermaid", mermaid, "```"];
33
+ if (truncated) {
34
+ mdLines.push("", "_truncated_ — traversal hit the node cap; deeper upstream paths were not explored. Lower `depth` or pick a narrower `qname`.");
35
+ }
36
+ return {
37
+ summary: `${r.nodes.length} upstream nodes${truncated ? " (truncated)" : ""}`,
38
+ markdown: mdLines.join("\n"),
39
+ data: { nodes: r.nodes, edges: r.edges, truncated },
40
+ };
41
+ },
42
+ });
43
+ //# sourceMappingURL=trace_upstream.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"trace_upstream.js","sourceRoot":"","sources":["../../src/tools/trace_upstream.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,EAAE,MAAM,2BAA2B,CAAC;AAC5D,OAAO,EAAE,eAAe,EAAE,MAAM,6BAA6B,CAAC;AAC9D,OAAO,EAAE,cAAc,EAAE,MAAM,eAAe,CAAC;AAC/C,OAAO,EAAE,UAAU,EAAE,CAAC,EAAE,MAAM,cAAc,CAAC;AAE7C,MAAM,WAAW,GAAG,CAAC,CAAC,MAAM,CAAC;IAC3B,GAAG,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC;IACtB,KAAK,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC;IACxB,KAAK,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC;CACjD,CAAC,CAAC;AAEH,UAAU,CAAC;IACT,IAAI,EAAE,gBAAgB;IACtB,WAAW,EACT,kNAAkN;IACpN,WAAW;IACX,KAAK,CAAC,OAAO,CAAC,KAAK;QACjB,MAAM,GAAG,GAAG,MAAM,cAAc,CAAC,EAAE,KAAK,EAAE,KAAK,CAAC,GAAG,EAAE,CAAC,CAAC;QACvD,MAAM,KAAK,GAAG,eAAe,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;QAC3C,MAAM,CAAC,GAAG,OAAO,CAAC,cAAc,CAAC,GAAG,CAAC,UAAU,EAAE,GAAG,CAAC,KAAK,EAAE,KAAK,EAAE,KAAK,CAAC,KAAK,CAAC,CAAC;QAChF,MAAM,KAAK,GAAG;YACZ,EAAE,aAAa,EAAE,KAAK,EAAE,KAAK,EAAE,QAAQ,EAAE;YACzC,GAAG,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,EAAE,aAAa,EAAE,CAAC,CAAC,aAAa,EAAE,KAAK,EAAE,CAAC,CAAC,KAAK,EAAE,CAAC,CAAC;SAC5E,CAAC;QACF,MAAM,OAAO,GAAG,MAAM,CAAC,qBAAqB,CAAC;YAC3C,KAAK;YACL,KAAK,EAAE,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;gBACzB,gBAAgB,EAAE,CAAC,CAAC,gBAAgB;gBACpC,gBAAgB,EAAE,CAAC,CAAC,gBAAgB;gBACpC,OAAO,EAAE,CAAC,CAAC,OAAO;aACnB,CAAC,CAAC;YACH,KAAK,EAAE,UAAU;SAClB,CAAC,CAAC;QACH,MAAM,SAAS,GAAG,CAAC,CAAC,SAAS,KAAK,IAAI,CAAC;QACvC,MAAM,OAAO,GAAG,CAAC,YAAY,EAAE,OAAO,EAAE,KAAK,CAAC,CAAC;QAC/C,IAAI,SAAS,EAAE,CAAC;YACd,OAAO,CAAC,IAAI,CACV,EAAE,EACF,8HAA8H,CAC/H,CAAC;QACJ,CAAC;QACD,OAAO;YACL,OAAO,EAAE,GAAG,CAAC,CAAC,KAAK,CAAC,MAAM,kBAAkB,SAAS,CAAC,CAAC,CAAC,cAAc,CAAC,CAAC,CAAC,EAAE,EAAE;YAC7E,QAAQ,EAAE,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC;YAC5B,IAAI,EAAE,EAAE,KAAK,EAAE,CAAC,CAAC,KAAK,EAAE,KAAK,EAAE,CAAC,CAAC,KAAK,EAAE,SAAS,EAAE;SACpD,CAAC;IACJ,CAAC;CACF,CAAC,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=what_broke.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"what_broke.d.ts","sourceRoot":"","sources":["../../src/tools/what_broke.ts"],"names":[],"mappings":""}
@@ -0,0 +1,86 @@
1
+ import { analyze, render } from "@ryanstark24/sfgraph-core";
2
+ import { asQualifiedName } from "@ryanstark24/sfgraph-shared";
3
+ import { getToolContext } from "../context.js";
4
+ import { defineTool, z } from "./_define.js";
5
+ const inputSchema = z.object({
6
+ org: z.string().min(1),
7
+ since: z.string().optional(),
8
+ });
9
+ defineTool({
10
+ name: "what_broke",
11
+ description: "USE THIS for any 'what broke' / 'post-deploy regression' / 'what changed since the deploy' question about a Salesforce org. Compares current graph against latest pre-sync snapshot, identifies changed nodes, buckets dependents as at-risk (no tests) or covered.",
12
+ inputSchema,
13
+ async execute(input) {
14
+ const ctx = await getToolContext({ orgId: input.org });
15
+ let fromId = input.since;
16
+ if (!fromId) {
17
+ const snaps = ctx.snapshotStore.listSnapshots(ctx.orgId);
18
+ const auto = snaps.find((s) => s.isAuto || s.label.startsWith("pre-sync"));
19
+ if (!auto) {
20
+ return {
21
+ summary: "no baseline snapshot",
22
+ markdown: "> no auto snapshot found; pass `since` explicitly",
23
+ data: { atRisk: [], covered: [] },
24
+ };
25
+ }
26
+ fromId = auto.id;
27
+ }
28
+ const nodeDiff = ctx.snapshotStore.diffNodes(ctx.orgId, fromId, "current");
29
+ const changedQnames = [
30
+ ...nodeDiff.changed.map((c) => c.after.qualifiedName),
31
+ ...nodeDiff.removed.map((r) => r.qualifiedName),
32
+ ];
33
+ if (changedQnames.length === 0) {
34
+ return {
35
+ summary: "no changes",
36
+ markdown: "_no changes since snapshot_",
37
+ data: { atRisk: [], covered: [] },
38
+ };
39
+ }
40
+ const atRisk = [];
41
+ const covered = [];
42
+ const allNodes = new Map();
43
+ const allEdges = [];
44
+ for (const qn of changedQnames) {
45
+ const qname = asQualifiedName(qn);
46
+ allNodes.set(qn, { qualifiedName: qn, label: "Changed" });
47
+ const dep = analyze.findDependents(ctx.graphStore, ctx.orgId, qname, 3);
48
+ for (const n of dep.nodes) {
49
+ allNodes.set(n.qualifiedName, { qualifiedName: n.qualifiedName, label: n.label });
50
+ // Skip nodes that are themselves test classes (they reach the target via IS_TEST_FOR chain).
51
+ const isTestClass = ctx.graphStore.listEdgesFrom(ctx.orgId, asQualifiedName(n.qualifiedName), "IS_TEST_FOR").length > 0;
52
+ if (isTestClass)
53
+ continue;
54
+ const hasTest = analyze.hasTestCoverage(ctx.graphStore, ctx.orgId, asQualifiedName(n.qualifiedName));
55
+ if (hasTest)
56
+ covered.push(n.qualifiedName);
57
+ else
58
+ atRisk.push(n.qualifiedName);
59
+ }
60
+ for (const e of dep.edges) {
61
+ allEdges.push({
62
+ srcQualifiedName: e.srcQualifiedName,
63
+ dstQualifiedName: e.dstQualifiedName,
64
+ });
65
+ }
66
+ }
67
+ const mermaid = render.renderDependencyGraph({
68
+ nodes: Array.from(allNodes.values()),
69
+ edges: allEdges,
70
+ title: "what_broke",
71
+ });
72
+ const md = [
73
+ `Changed: ${changedQnames.length}, at risk: ${atRisk.length}, covered: ${covered.length}`,
74
+ "",
75
+ "```mermaid",
76
+ mermaid,
77
+ "```",
78
+ ].join("\n");
79
+ return {
80
+ summary: `${atRisk.length} dependents at risk`,
81
+ markdown: md,
82
+ data: { changed: changedQnames, atRisk, covered },
83
+ };
84
+ },
85
+ });
86
+ //# sourceMappingURL=what_broke.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"what_broke.js","sourceRoot":"","sources":["../../src/tools/what_broke.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=wip_diff.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"wip_diff.d.ts","sourceRoot":"","sources":["../../src/tools/wip_diff.ts"],"names":[],"mappings":""}
@@ -0,0 +1,55 @@
1
+ import { analyze } from "@ryanstark24/sfgraph-core";
2
+ import { ConfigError, readWorkspace } from "@ryanstark24/sfgraph-shared";
3
+ import { getToolContext } from "../context.js";
4
+ import { defineTool, z } from "./_define.js";
5
+ import { resolveWipProjectRoot } from "./_project-root.js";
6
+ const inputSchema = z.object({
7
+ project_root: z.string().min(1).optional(),
8
+ org: z.string().min(1).optional(),
9
+ mode: z.enum(["changed-only", "full-folder"]).default("changed-only"),
10
+ });
11
+ defineTool({
12
+ name: "wip_diff",
13
+ description: "USE THIS for any 'show me what is different between my local source and the org' / 'list local-only / org-only metadata' question. Returns just the added/changed/removed sets — no dependent fan-out. Faster than wip_impact when you only need the diff.",
14
+ inputSchema,
15
+ async execute(input) {
16
+ const projectRoot = resolveWipProjectRoot(input.project_root ?? process.cwd());
17
+ let orgArg = input.org;
18
+ if (!orgArg) {
19
+ const ws = await readWorkspace(projectRoot);
20
+ if (ws?.orgId) {
21
+ orgArg = ws.orgId;
22
+ }
23
+ else {
24
+ throw new ConfigError("wip_diff: no `org` provided and no workspace binding found. Run `sfgraph link --org <alias>` first.");
25
+ }
26
+ }
27
+ const ctx = await getToolContext(orgArg ? { orgId: orgArg } : {});
28
+ const result = await analyze.analyzeLocalImpact({
29
+ graphStore: ctx.graphStore,
30
+ orgId: ctx.orgId,
31
+ projectRoot,
32
+ mode: input.mode,
33
+ });
34
+ const md = [
35
+ `Added (${result.addedQnames.length}):`,
36
+ ...result.addedQnames.map((q) => ` + ${q}`),
37
+ "",
38
+ `Changed (${result.changedQnames.length}):`,
39
+ ...result.changedQnames.map((q) => ` ~ ${q}`),
40
+ "",
41
+ `Removed (${result.removedQnames.length}):`,
42
+ ...result.removedQnames.map((q) => ` - ${q}`),
43
+ ].join("\n");
44
+ return {
45
+ summary: `WIP diff: +${result.addedQnames.length} ~${result.changedQnames.length} -${result.removedQnames.length}`,
46
+ markdown: md,
47
+ data: {
48
+ added: result.addedQnames,
49
+ changed: result.changedQnames,
50
+ removed: result.removedQnames,
51
+ },
52
+ };
53
+ },
54
+ });
55
+ //# sourceMappingURL=wip_diff.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"wip_diff.js","sourceRoot":"","sources":["../../src/tools/wip_diff.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,2BAA2B,CAAC;AACpD,OAAO,EAAE,WAAW,EAAE,aAAa,EAAE,MAAM,6BAA6B,CAAC;AACzE,OAAO,EAAE,cAAc,EAAE,MAAM,eAAe,CAAC;AAC/C,OAAO,EAAE,UAAU,EAAE,CAAC,EAAE,MAAM,cAAc,CAAC;AAC7C,OAAO,EAAE,qBAAqB,EAAE,MAAM,oBAAoB,CAAC;AAE3D,MAAM,WAAW,GAAG,CAAC,CAAC,MAAM,CAAC;IAC3B,YAAY,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,QAAQ,EAAE;IAC1C,GAAG,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,QAAQ,EAAE;IACjC,IAAI,EAAE,CAAC,CAAC,IAAI,CAAC,CAAC,cAAc,EAAE,aAAa,CAAC,CAAC,CAAC,OAAO,CAAC,cAAc,CAAC;CACtE,CAAC,CAAC;AAEH,UAAU,CAAC;IACT,IAAI,EAAE,UAAU;IAChB,WAAW,EACT,4PAA4P;IAC9P,WAAW;IACX,KAAK,CAAC,OAAO,CAAC,KAAK;QACjB,MAAM,WAAW,GAAG,qBAAqB,CAAC,KAAK,CAAC,YAAY,IAAI,OAAO,CAAC,GAAG,EAAE,CAAC,CAAC;QAC/E,IAAI,MAAM,GAAG,KAAK,CAAC,GAAG,CAAC;QACvB,IAAI,CAAC,MAAM,EAAE,CAAC;YACZ,MAAM,EAAE,GAAG,MAAM,aAAa,CAAC,WAAW,CAAC,CAAC;YAC5C,IAAI,EAAE,EAAE,KAAK,EAAE,CAAC;gBACd,MAAM,GAAG,EAAE,CAAC,KAAK,CAAC;YACpB,CAAC;iBAAM,CAAC;gBACN,MAAM,IAAI,WAAW,CACnB,qGAAqG,CACtG,CAAC;YACJ,CAAC;QACH,CAAC;QACD,MAAM,GAAG,GAAG,MAAM,cAAc,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;QAClE,MAAM,MAAM,GAAG,MAAM,OAAO,CAAC,kBAAkB,CAAC;YAC9C,UAAU,EAAE,GAAG,CAAC,UAAU;YAC1B,KAAK,EAAE,GAAG,CAAC,KAAK;YAChB,WAAW;YACX,IAAI,EAAE,KAAK,CAAC,IAAI;SACjB,CAAC,CAAC;QACH,MAAM,EAAE,GAAG;YACT,UAAU,MAAM,CAAC,WAAW,CAAC,MAAM,IAAI;YACvC,GAAG,MAAM,CAAC,WAAW,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,OAAO,CAAC,EAAE,CAAC;YAC5C,EAAE;YACF,YAAY,MAAM,CAAC,aAAa,CAAC,MAAM,IAAI;YAC3C,GAAG,MAAM,CAAC,aAAa,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,OAAO,CAAC,EAAE,CAAC;YAC9C,EAAE;YACF,YAAY,MAAM,CAAC,aAAa,CAAC,MAAM,IAAI;YAC3C,GAAG,MAAM,CAAC,aAAa,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,OAAO,CAAC,EAAE,CAAC;SAC/C,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACb,OAAO;YACL,OAAO,EAAE,cAAc,MAAM,CAAC,WAAW,CAAC,MAAM,KAAK,MAAM,CAAC,aAAa,CAAC,MAAM,KAAK,MAAM,CAAC,aAAa,CAAC,MAAM,EAAE;YAClH,QAAQ,EAAE,EAAE;YACZ,IAAI,EAAE;gBACJ,KAAK,EAAE,MAAM,CAAC,WAAW;gBACzB,OAAO,EAAE,MAAM,CAAC,aAAa;gBAC7B,OAAO,EAAE,MAAM,CAAC,aAAa;aAC9B;SACF,CAAC;IACJ,CAAC;CACF,CAAC,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=wip_impact.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"wip_impact.d.ts","sourceRoot":"","sources":["../../src/tools/wip_impact.ts"],"names":[],"mappings":""}
@@ -0,0 +1,55 @@
1
+ import { analyze } from "@ryanstark24/sfgraph-core";
2
+ import { ConfigError, readWorkspace } from "@ryanstark24/sfgraph-shared";
3
+ import { getToolContext } from "../context.js";
4
+ import { defineTool, z } from "./_define.js";
5
+ import { resolveWipProjectRoot } from "./_project-root.js";
6
+ const inputSchema = z.object({
7
+ project_root: z.string().min(1).optional(),
8
+ org: z.string().min(1).optional(),
9
+ depth: z.number().int().min(1).max(5).default(3),
10
+ mode: z.enum(["changed-only", "full-folder"]).default("changed-only"),
11
+ });
12
+ defineTool({
13
+ name: "wip_impact",
14
+ description: "USE THIS for any 'what does this branch break' / 'dry-run my local changes' / 'what would happen if I deployed this' / 'impact of my uncommitted changes' question. Parses the local sfdx-source tree (force-app/) and overlays it on the org's persisted graph WITHOUT writing to it. Returns changed/added/removed qnames + N-hop dependent BFS + Mermaid. The local-equivalent of impact_from_git_diff for uncommitted work.",
15
+ inputSchema,
16
+ async execute(input) {
17
+ const projectRoot = resolveWipProjectRoot(input.project_root ?? process.cwd());
18
+ let orgArg = input.org;
19
+ if (!orgArg) {
20
+ const ws = await readWorkspace(projectRoot);
21
+ if (ws?.orgId) {
22
+ orgArg = ws.orgId;
23
+ }
24
+ else {
25
+ throw new ConfigError("wip_impact: no `org` provided and no workspace binding found. Run `sfgraph link --org <alias>` first.");
26
+ }
27
+ }
28
+ const ctx = await getToolContext(orgArg ? { orgId: orgArg } : {});
29
+ const result = await analyze.analyzeLocalImpact({
30
+ graphStore: ctx.graphStore,
31
+ orgId: ctx.orgId,
32
+ projectRoot,
33
+ depth: input.depth,
34
+ mode: input.mode,
35
+ });
36
+ const md = [
37
+ `Changed: ${result.changedQnames.length}, Added: ${result.addedQnames.length}, Removed: ${result.removedQnames.length}, Dependents: ${result.dependents.length}`,
38
+ "",
39
+ "```mermaid",
40
+ result.mermaid,
41
+ "```",
42
+ ].join("\n");
43
+ return {
44
+ summary: `WIP impact: ${result.changedQnames.length} changed, ${result.addedQnames.length} added, ${result.dependents.length} dependents`,
45
+ markdown: md,
46
+ data: {
47
+ changed: result.changedQnames,
48
+ added: result.addedQnames,
49
+ removed: result.removedQnames,
50
+ dependents: result.dependents,
51
+ },
52
+ };
53
+ },
54
+ });
55
+ //# sourceMappingURL=wip_impact.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"wip_impact.js","sourceRoot":"","sources":["../../src/tools/wip_impact.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,2BAA2B,CAAC;AACpD,OAAO,EAAE,WAAW,EAAE,aAAa,EAAE,MAAM,6BAA6B,CAAC;AACzE,OAAO,EAAE,cAAc,EAAE,MAAM,eAAe,CAAC;AAC/C,OAAO,EAAE,UAAU,EAAE,CAAC,EAAE,MAAM,cAAc,CAAC;AAC7C,OAAO,EAAE,qBAAqB,EAAE,MAAM,oBAAoB,CAAC;AAE3D,MAAM,WAAW,GAAG,CAAC,CAAC,MAAM,CAAC;IAC3B,YAAY,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,QAAQ,EAAE;IAC1C,GAAG,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,QAAQ,EAAE;IACjC,KAAK,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC;IAChD,IAAI,EAAE,CAAC,CAAC,IAAI,CAAC,CAAC,cAAc,EAAE,aAAa,CAAC,CAAC,CAAC,OAAO,CAAC,cAAc,CAAC;CACtE,CAAC,CAAC;AAEH,UAAU,CAAC;IACT,IAAI,EAAE,YAAY;IAClB,WAAW,EACT,iaAAia;IACna,WAAW;IACX,KAAK,CAAC,OAAO,CAAC,KAAK;QACjB,MAAM,WAAW,GAAG,qBAAqB,CAAC,KAAK,CAAC,YAAY,IAAI,OAAO,CAAC,GAAG,EAAE,CAAC,CAAC;QAC/E,IAAI,MAAM,GAAG,KAAK,CAAC,GAAG,CAAC;QACvB,IAAI,CAAC,MAAM,EAAE,CAAC;YACZ,MAAM,EAAE,GAAG,MAAM,aAAa,CAAC,WAAW,CAAC,CAAC;YAC5C,IAAI,EAAE,EAAE,KAAK,EAAE,CAAC;gBACd,MAAM,GAAG,EAAE,CAAC,KAAK,CAAC;YACpB,CAAC;iBAAM,CAAC;gBACN,MAAM,IAAI,WAAW,CACnB,uGAAuG,CACxG,CAAC;YACJ,CAAC;QACH,CAAC;QACD,MAAM,GAAG,GAAG,MAAM,cAAc,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;QAClE,MAAM,MAAM,GAAG,MAAM,OAAO,CAAC,kBAAkB,CAAC;YAC9C,UAAU,EAAE,GAAG,CAAC,UAAU;YAC1B,KAAK,EAAE,GAAG,CAAC,KAAK;YAChB,WAAW;YACX,KAAK,EAAE,KAAK,CAAC,KAAK;YAClB,IAAI,EAAE,KAAK,CAAC,IAAI;SACjB,CAAC,CAAC;QACH,MAAM,EAAE,GAAG;YACT,YAAY,MAAM,CAAC,aAAa,CAAC,MAAM,YAAY,MAAM,CAAC,WAAW,CAAC,MAAM,cAAc,MAAM,CAAC,aAAa,CAAC,MAAM,iBAAiB,MAAM,CAAC,UAAU,CAAC,MAAM,EAAE;YAChK,EAAE;YACF,YAAY;YACZ,MAAM,CAAC,OAAO;YACd,KAAK;SACN,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACb,OAAO;YACL,OAAO,EAAE,eAAe,MAAM,CAAC,aAAa,CAAC,MAAM,aAAa,MAAM,CAAC,WAAW,CAAC,MAAM,WAAW,MAAM,CAAC,UAAU,CAAC,MAAM,aAAa;YACzI,QAAQ,EAAE,EAAE;YACZ,IAAI,EAAE;gBACJ,OAAO,EAAE,MAAM,CAAC,aAAa;gBAC7B,KAAK,EAAE,MAAM,CAAC,WAAW;gBACzB,OAAO,EAAE,MAAM,CAAC,aAAa;gBAC7B,UAAU,EAAE,MAAM,CAAC,UAAU;aAC9B;SACF,CAAC;IACJ,CAAC;CACF,CAAC,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=wip_test_gap.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"wip_test_gap.d.ts","sourceRoot":"","sources":["../../src/tools/wip_test_gap.ts"],"names":[],"mappings":""}
@@ -0,0 +1,50 @@
1
+ import { analyze } from "@ryanstark24/sfgraph-core";
2
+ import { ConfigError, readWorkspace } from "@ryanstark24/sfgraph-shared";
3
+ import { getToolContext } from "../context.js";
4
+ import { defineTool, z } from "./_define.js";
5
+ import { resolveWipProjectRoot } from "./_project-root.js";
6
+ const inputSchema = z.object({
7
+ project_root: z.string().min(1).optional(),
8
+ org: z.string().min(1).optional(),
9
+ depth: z.number().int().min(1).max(5).default(3),
10
+ mode: z.enum(["changed-only", "full-folder"]).default("changed-only"),
11
+ });
12
+ defineTool({
13
+ name: "wip_test_gap",
14
+ description: "USE THIS for any 'what tests am I missing for these local changes' / 'will my uncommitted changes have test coverage gaps' question. Runs wip_impact internally then filters dependents to those without IS_TEST_FOR coverage.",
15
+ inputSchema,
16
+ async execute(input) {
17
+ const projectRoot = resolveWipProjectRoot(input.project_root ?? process.cwd());
18
+ let orgArg = input.org;
19
+ if (!orgArg) {
20
+ const ws = await readWorkspace(projectRoot);
21
+ if (ws?.orgId) {
22
+ orgArg = ws.orgId;
23
+ }
24
+ else {
25
+ throw new ConfigError("wip_test_gap: no `org` provided and no workspace binding found. Run `sfgraph link --org <alias>` first.");
26
+ }
27
+ }
28
+ const ctx = await getToolContext(orgArg ? { orgId: orgArg } : {});
29
+ const result = await analyze.analyzeLocalImpact({
30
+ graphStore: ctx.graphStore,
31
+ orgId: ctx.orgId,
32
+ projectRoot,
33
+ depth: input.depth,
34
+ mode: input.mode,
35
+ });
36
+ const uncovered = result.dependents.filter((d) => !d.coveredByTest);
37
+ const md = uncovered.length === 0
38
+ ? "No uncovered dependents found."
39
+ : [
40
+ `Uncovered dependents (${uncovered.length}):`,
41
+ ...uncovered.map((d) => `- ${d.qname} (${d.label}, depth=${d.depth}${d.viaRelType ? `, via=${d.viaRelType}` : ""})`),
42
+ ].join("\n");
43
+ return {
44
+ summary: `${uncovered.length} uncovered dependents from WIP changes`,
45
+ markdown: md,
46
+ data: { uncovered },
47
+ };
48
+ },
49
+ });
50
+ //# sourceMappingURL=wip_test_gap.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"wip_test_gap.js","sourceRoot":"","sources":["../../src/tools/wip_test_gap.ts"],"names":[],"mappings":"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"}