gsd-pi 2.80.0-dev.cf9433f56 → 2.80.0-dev.d4fc28e6b

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 (237) hide show
  1. package/dist/cli.js +0 -19
  2. package/dist/resources/.managed-resources-content-hash +1 -1
  3. package/dist/resources/extensions/claude-code-cli/stream-adapter.js +29 -0
  4. package/dist/resources/extensions/gsd/auto/loop.js +71 -8
  5. package/dist/resources/extensions/gsd/auto/phases.js +150 -94
  6. package/dist/resources/extensions/gsd/auto/resolve.js +12 -0
  7. package/dist/resources/extensions/gsd/auto/run-unit.js +10 -30
  8. package/dist/resources/extensions/gsd/auto/session.js +8 -0
  9. package/dist/resources/extensions/gsd/auto/workflow-dispatch-claim.js +33 -1
  10. package/dist/resources/extensions/gsd/auto/workflow-worker-heartbeat.js +9 -1
  11. package/dist/resources/extensions/gsd/auto-direct-dispatch.js +5 -32
  12. package/dist/resources/extensions/gsd/auto-dispatch.js +16 -0
  13. package/dist/resources/extensions/gsd/auto-post-unit.js +17 -4
  14. package/dist/resources/extensions/gsd/auto-prompts.js +90 -15
  15. package/dist/resources/extensions/gsd/auto-start.js +197 -6
  16. package/dist/resources/extensions/gsd/auto-worktree.js +111 -1
  17. package/dist/resources/extensions/gsd/auto.js +18 -22
  18. package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +86 -19
  19. package/dist/resources/extensions/gsd/bootstrap/db-tools.js +49 -36
  20. package/dist/resources/extensions/gsd/bootstrap/dynamic-tools.js +15 -5
  21. package/dist/resources/extensions/gsd/bootstrap/exec-tools.js +9 -3
  22. package/dist/resources/extensions/gsd/bootstrap/journal-tools.js +7 -1
  23. package/dist/resources/extensions/gsd/bootstrap/memory-tools.js +9 -3
  24. package/dist/resources/extensions/gsd/bootstrap/query-tools.js +8 -2
  25. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +298 -54
  26. package/dist/resources/extensions/gsd/bootstrap/system-context.js +82 -23
  27. package/dist/resources/extensions/gsd/clean-root-preflight.js +24 -6
  28. package/dist/resources/extensions/gsd/commands-handlers.js +23 -9
  29. package/dist/resources/extensions/gsd/db/unit-dispatches.js +53 -0
  30. package/dist/resources/extensions/gsd/ecosystem/gsd-extension-api.js +2 -0
  31. package/dist/resources/extensions/gsd/guided-flow.js +47 -28
  32. package/dist/resources/extensions/gsd/native-git-bridge.js +32 -8
  33. package/dist/resources/extensions/gsd/orphan-stash-audit.js +101 -0
  34. package/dist/resources/extensions/gsd/parallel-orchestrator.js +13 -3
  35. package/dist/resources/extensions/gsd/pre-execution-checks.js +15 -0
  36. package/dist/resources/extensions/gsd/prompts/complete-milestone.md +2 -0
  37. package/dist/resources/extensions/gsd/prompts/complete-slice.md +1 -1
  38. package/dist/resources/extensions/gsd/prompts/execute-task.md +4 -2
  39. package/dist/resources/extensions/gsd/prompts/plan-slice.md +1 -1
  40. package/dist/resources/extensions/gsd/prompts/replan-slice.md +2 -2
  41. package/dist/resources/extensions/gsd/workflow-protocol.js +131 -0
  42. package/dist/resources/extensions/gsd/worktree-resolver.js +35 -4
  43. package/dist/tsconfig.extensions.tsbuildinfo +1 -1
  44. package/dist/web/standalone/.next/BUILD_ID +1 -1
  45. package/dist/web/standalone/.next/app-path-routes-manifest.json +12 -12
  46. package/dist/web/standalone/.next/build-manifest.json +2 -2
  47. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  48. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  49. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  50. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  51. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  52. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  53. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  54. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  55. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  56. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  57. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  58. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  59. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  60. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  61. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  62. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  63. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  64. package/dist/web/standalone/.next/server/app/index.html +1 -1
  65. package/dist/web/standalone/.next/server/app/index.rsc +1 -1
  66. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  67. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
  68. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  69. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  70. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  71. package/dist/web/standalone/.next/server/app-paths-manifest.json +12 -12
  72. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  73. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  74. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  75. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  76. package/dist/welcome-screen.d.ts +2 -0
  77. package/dist/welcome-screen.js +9 -7
  78. package/package.json +1 -1
  79. package/packages/pi-agent-core/dist/agent-loop.d.ts.map +1 -1
  80. package/packages/pi-agent-core/dist/agent-loop.js +4 -1
  81. package/packages/pi-agent-core/dist/agent-loop.js.map +1 -1
  82. package/packages/pi-agent-core/dist/agent.d.ts +5 -0
  83. package/packages/pi-agent-core/dist/agent.d.ts.map +1 -1
  84. package/packages/pi-agent-core/dist/agent.js +2 -0
  85. package/packages/pi-agent-core/dist/agent.js.map +1 -1
  86. package/packages/pi-agent-core/dist/index.d.ts +1 -0
  87. package/packages/pi-agent-core/dist/index.d.ts.map +1 -1
  88. package/packages/pi-agent-core/dist/index.js +2 -0
  89. package/packages/pi-agent-core/dist/index.js.map +1 -1
  90. package/packages/pi-agent-core/dist/token-audit.d.ts +47 -0
  91. package/packages/pi-agent-core/dist/token-audit.d.ts.map +1 -0
  92. package/packages/pi-agent-core/dist/token-audit.js +221 -0
  93. package/packages/pi-agent-core/dist/token-audit.js.map +1 -0
  94. package/packages/pi-agent-core/dist/types.d.ts +9 -0
  95. package/packages/pi-agent-core/dist/types.d.ts.map +1 -1
  96. package/packages/pi-agent-core/dist/types.js.map +1 -1
  97. package/packages/pi-agent-core/src/agent-loop.test.ts +128 -0
  98. package/packages/pi-agent-core/src/agent-loop.ts +4 -1
  99. package/packages/pi-agent-core/src/agent.ts +8 -0
  100. package/packages/pi-agent-core/src/index.ts +2 -0
  101. package/packages/pi-agent-core/src/token-audit.test.ts +189 -0
  102. package/packages/pi-agent-core/src/token-audit.ts +287 -0
  103. package/packages/pi-agent-core/src/types.ts +14 -0
  104. package/packages/pi-agent-core/tsconfig.tsbuildinfo +1 -1
  105. package/packages/pi-coding-agent/dist/core/agent-session-tool-refresh.test.js +18 -0
  106. package/packages/pi-coding-agent/dist/core/agent-session-tool-refresh.test.js.map +1 -1
  107. package/packages/pi-coding-agent/dist/core/agent-session.d.ts +12 -0
  108. package/packages/pi-coding-agent/dist/core/agent-session.d.ts.map +1 -1
  109. package/packages/pi-coding-agent/dist/core/agent-session.js +36 -7
  110. package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
  111. package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts.map +1 -1
  112. package/packages/pi-coding-agent/dist/core/extensions/loader.js +8 -0
  113. package/packages/pi-coding-agent/dist/core/extensions/loader.js.map +1 -1
  114. package/packages/pi-coding-agent/dist/core/extensions/runner.d.ts +2 -0
  115. package/packages/pi-coding-agent/dist/core/extensions/runner.d.ts.map +1 -1
  116. package/packages/pi-coding-agent/dist/core/extensions/runner.js +3 -6
  117. package/packages/pi-coding-agent/dist/core/extensions/runner.js.map +1 -1
  118. package/packages/pi-coding-agent/dist/core/extensions/runner.test.js +3 -3
  119. package/packages/pi-coding-agent/dist/core/extensions/runner.test.js.map +1 -1
  120. package/packages/pi-coding-agent/dist/core/extensions/types.d.ts +32 -1
  121. package/packages/pi-coding-agent/dist/core/extensions/types.d.ts.map +1 -1
  122. package/packages/pi-coding-agent/dist/core/extensions/types.js.map +1 -1
  123. package/packages/pi-coding-agent/dist/core/hooks-runner.test.js +2 -0
  124. package/packages/pi-coding-agent/dist/core/hooks-runner.test.js.map +1 -1
  125. package/packages/pi-coding-agent/dist/core/sdk-tool-filter.test.d.ts +2 -0
  126. package/packages/pi-coding-agent/dist/core/sdk-tool-filter.test.d.ts.map +1 -0
  127. package/packages/pi-coding-agent/dist/core/sdk-tool-filter.test.js +46 -0
  128. package/packages/pi-coding-agent/dist/core/sdk-tool-filter.test.js.map +1 -0
  129. package/packages/pi-coding-agent/dist/core/sdk.d.ts +10 -2
  130. package/packages/pi-coding-agent/dist/core/sdk.d.ts.map +1 -1
  131. package/packages/pi-coding-agent/dist/core/sdk.js +74 -2
  132. package/packages/pi-coding-agent/dist/core/sdk.js.map +1 -1
  133. package/packages/pi-coding-agent/dist/core/skill-tool.test.js +22 -0
  134. package/packages/pi-coding-agent/dist/core/skill-tool.test.js.map +1 -1
  135. package/packages/pi-coding-agent/dist/core/system-prompt.d.ts +6 -7
  136. package/packages/pi-coding-agent/dist/core/system-prompt.d.ts.map +1 -1
  137. package/packages/pi-coding-agent/dist/core/system-prompt.js +2 -3
  138. package/packages/pi-coding-agent/dist/core/system-prompt.js.map +1 -1
  139. package/packages/pi-coding-agent/src/core/agent-session-tool-refresh.test.ts +25 -0
  140. package/packages/pi-coding-agent/src/core/agent-session.ts +40 -7
  141. package/packages/pi-coding-agent/src/core/extensions/loader.ts +10 -0
  142. package/packages/pi-coding-agent/src/core/extensions/runner.test.ts +3 -3
  143. package/packages/pi-coding-agent/src/core/extensions/runner.ts +5 -5
  144. package/packages/pi-coding-agent/src/core/extensions/types.ts +35 -1
  145. package/packages/pi-coding-agent/src/core/hooks-runner.test.ts +2 -0
  146. package/packages/pi-coding-agent/src/core/sdk-tool-filter.test.ts +60 -0
  147. package/packages/pi-coding-agent/src/core/sdk.ts +85 -3
  148. package/packages/pi-coding-agent/src/core/skill-tool.test.ts +28 -0
  149. package/packages/pi-coding-agent/src/core/system-prompt.ts +8 -10
  150. package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
  151. package/src/resources/extensions/claude-code-cli/stream-adapter.ts +30 -0
  152. package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +26 -0
  153. package/src/resources/extensions/gsd/auto/loop-deps.ts +2 -2
  154. package/src/resources/extensions/gsd/auto/loop.ts +84 -8
  155. package/src/resources/extensions/gsd/auto/phases.ts +218 -154
  156. package/src/resources/extensions/gsd/auto/resolve.ts +19 -0
  157. package/src/resources/extensions/gsd/auto/run-unit.ts +10 -29
  158. package/src/resources/extensions/gsd/auto/session.ts +8 -0
  159. package/src/resources/extensions/gsd/auto/workflow-dispatch-claim.ts +63 -1
  160. package/src/resources/extensions/gsd/auto/workflow-worker-heartbeat.ts +14 -1
  161. package/src/resources/extensions/gsd/auto-direct-dispatch.ts +8 -34
  162. package/src/resources/extensions/gsd/auto-dispatch.ts +16 -0
  163. package/src/resources/extensions/gsd/auto-post-unit.ts +18 -4
  164. package/src/resources/extensions/gsd/auto-prompts.ts +95 -14
  165. package/src/resources/extensions/gsd/auto-start.ts +230 -9
  166. package/src/resources/extensions/gsd/auto-worktree.ts +123 -0
  167. package/src/resources/extensions/gsd/auto.ts +18 -18
  168. package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +100 -18
  169. package/src/resources/extensions/gsd/bootstrap/db-tools.ts +50 -36
  170. package/src/resources/extensions/gsd/bootstrap/dynamic-tools.ts +16 -5
  171. package/src/resources/extensions/gsd/bootstrap/exec-tools.ts +10 -3
  172. package/src/resources/extensions/gsd/bootstrap/journal-tools.ts +8 -1
  173. package/src/resources/extensions/gsd/bootstrap/memory-tools.ts +10 -3
  174. package/src/resources/extensions/gsd/bootstrap/query-tools.ts +9 -2
  175. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +347 -54
  176. package/src/resources/extensions/gsd/bootstrap/system-context.ts +90 -22
  177. package/src/resources/extensions/gsd/clean-root-preflight.ts +32 -7
  178. package/src/resources/extensions/gsd/commands-handlers.ts +34 -15
  179. package/src/resources/extensions/gsd/db/unit-dispatches.ts +66 -0
  180. package/src/resources/extensions/gsd/ecosystem/gsd-extension-api.ts +3 -0
  181. package/src/resources/extensions/gsd/guided-flow.ts +52 -35
  182. package/src/resources/extensions/gsd/native-git-bridge.ts +39 -6
  183. package/src/resources/extensions/gsd/orphan-stash-audit.ts +117 -0
  184. package/src/resources/extensions/gsd/parallel-orchestrator.ts +13 -3
  185. package/src/resources/extensions/gsd/pre-execution-checks.ts +16 -0
  186. package/src/resources/extensions/gsd/prompts/complete-milestone.md +2 -0
  187. package/src/resources/extensions/gsd/prompts/complete-slice.md +1 -1
  188. package/src/resources/extensions/gsd/prompts/execute-task.md +4 -2
  189. package/src/resources/extensions/gsd/prompts/plan-slice.md +1 -1
  190. package/src/resources/extensions/gsd/prompts/replan-slice.md +2 -2
  191. package/src/resources/extensions/gsd/tests/artifact-retry-cap.test.ts +2 -2
  192. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +361 -10
  193. package/src/resources/extensions/gsd/tests/auto-wrapup-inflight-guard.test.ts +168 -6
  194. package/src/resources/extensions/gsd/tests/clean-root-preflight.test.ts +15 -6
  195. package/src/resources/extensions/gsd/tests/complete-milestone-excerpt.test.ts +31 -0
  196. package/src/resources/extensions/gsd/tests/complete-slice-composer.test.ts +3 -2
  197. package/src/resources/extensions/gsd/tests/context-store.test.ts +7 -1
  198. package/src/resources/extensions/gsd/tests/custom-engine-loop-integration.test.ts +5 -1
  199. package/src/resources/extensions/gsd/tests/execute-task-rendering.test.ts +5 -2
  200. package/src/resources/extensions/gsd/tests/fast-forward-reused-milestone-branch.test.ts +219 -0
  201. package/src/resources/extensions/gsd/tests/finalize-survivor-branch.test.ts +132 -0
  202. package/src/resources/extensions/gsd/tests/isolation-none-branch-guard.test.ts +6 -3
  203. package/src/resources/extensions/gsd/tests/journal-integration.test.ts +5 -1
  204. package/src/resources/extensions/gsd/tests/journal-query-tool.test.ts +32 -0
  205. package/src/resources/extensions/gsd/tests/knowledge.test.ts +47 -0
  206. package/src/resources/extensions/gsd/tests/merge-conflict-stops-loop.test.ts +1 -0
  207. package/src/resources/extensions/gsd/tests/milestone-merge-stash-restore.test.ts +242 -0
  208. package/src/resources/extensions/gsd/tests/native-git-bridge-exec-fallback.test.ts +34 -2
  209. package/src/resources/extensions/gsd/tests/originalbase-path-comparison.test.ts +3 -0
  210. package/src/resources/extensions/gsd/tests/orphan-merge-bootstrap.test.ts +133 -0
  211. package/src/resources/extensions/gsd/tests/orphan-stash-audit.test.ts +201 -0
  212. package/src/resources/extensions/gsd/tests/parallel-orchestrator-fast-forward.test.ts +113 -0
  213. package/src/resources/extensions/gsd/tests/pre-execution-checks.test.ts +7 -5
  214. package/src/resources/extensions/gsd/tests/prompt-duplication-cuts.test.ts +230 -0
  215. package/src/resources/extensions/gsd/tests/query-tools-db-open.test.ts +3 -3
  216. package/src/resources/extensions/gsd/tests/restore-tools-after-discuss.test.ts +38 -17
  217. package/src/resources/extensions/gsd/tests/select-resumable-milestone.test.ts +96 -0
  218. package/src/resources/extensions/gsd/tests/session-start-footer.test.ts +77 -0
  219. package/src/resources/extensions/gsd/tests/session-switch-abort-misclassification.test.ts +166 -0
  220. package/src/resources/extensions/gsd/tests/state-corruption-2945.test.ts +1 -0
  221. package/src/resources/extensions/gsd/tests/system-context-memory.test.ts +112 -0
  222. package/src/resources/extensions/gsd/tests/system-context-message-routing.test.ts +7 -9
  223. package/src/resources/extensions/gsd/tests/token-tool-gating.test.ts +291 -0
  224. package/src/resources/extensions/gsd/tests/unit-dispatches.test.ts +50 -1
  225. package/src/resources/extensions/gsd/tests/unstructured-continue-context-injection.test.ts +5 -4
  226. package/src/resources/extensions/gsd/tests/workflow-dispatch-claim.test.ts +142 -0
  227. package/src/resources/extensions/gsd/tests/workflow-protocol-excerpt.test.ts +99 -0
  228. package/src/resources/extensions/gsd/tests/workflow-worker-heartbeat.test.ts +32 -1
  229. package/src/resources/extensions/gsd/tests/worktree-journal-events.test.ts +1 -0
  230. package/src/resources/extensions/gsd/tests/worktree-path-injection.test.ts +22 -19
  231. package/src/resources/extensions/gsd/tests/worktree-project-root-degrade.test.ts +66 -0
  232. package/src/resources/extensions/gsd/tests/worktree-resolver.test.ts +104 -3
  233. package/src/resources/extensions/gsd/workflow-protocol.ts +160 -0
  234. package/src/resources/extensions/gsd/worktree-resolver.ts +49 -4
  235. package/src/resources/extensions/gsd/tests/phases-merge-error-stops-auto.test.ts +0 -97
  236. /package/dist/web/standalone/.next/static/{-5nHJWzSdG-WkPMul_khA → cWaxzf-sdbSSbbwYu8q7a}/_buildManifest.js +0 -0
  237. /package/dist/web/standalone/.next/static/{-5nHJWzSdG-WkPMul_khA → cWaxzf-sdbSSbbwYu8q7a}/_ssgManifest.js +0 -0
@@ -0,0 +1,189 @@
1
+ // Project/App: GSD-2
2
+ // File Purpose: Tests for provider-boundary token payload audit helpers.
3
+
4
+ import assert from "node:assert/strict";
5
+ import test from "node:test";
6
+ import { Type } from "@sinclair/typebox";
7
+ import type { Context } from "@gsd/pi-ai";
8
+ import type { AgentMessage } from "./types.js";
9
+ import {
10
+ buildProviderPayloadAuditSummary,
11
+ buildTokenAuditSummary,
12
+ maybeLogProviderPayloadAudit,
13
+ maybeLogTokenAudit,
14
+ } from "./token-audit.js";
15
+
16
+ test("buildTokenAuditSummary reports payload section sizes without content fields", () => {
17
+ const context: Context = {
18
+ systemPrompt: "system prompt",
19
+ tools: [
20
+ {
21
+ name: "read",
22
+ description: "Read a file",
23
+ parameters: Type.Object({ path: Type.String() }),
24
+ },
25
+ {
26
+ name: "large_tool",
27
+ description: "Large schema".repeat(20),
28
+ parameters: Type.Object({ value: Type.String() }),
29
+ },
30
+ ],
31
+ messages: [
32
+ { role: "user", content: [{ type: "text", text: "hello" }], timestamp: 1 },
33
+ {
34
+ role: "toolResult",
35
+ toolCallId: "call-1",
36
+ toolName: "read",
37
+ content: [{ type: "text", text: "tool output" }],
38
+ isError: false,
39
+ timestamp: 2,
40
+ },
41
+ { role: "user", content: [{ type: "image", data: "abc123", mimeType: "image/png" }], timestamp: 3 },
42
+ ],
43
+ };
44
+ const sourceMessages = [
45
+ ...context.messages,
46
+ {
47
+ role: "custom",
48
+ customType: "gsd-memory",
49
+ content: "memory block",
50
+ display: false,
51
+ timestamp: 4,
52
+ } as AgentMessage,
53
+ ];
54
+
55
+ const summary = buildTokenAuditSummary(context, sourceMessages);
56
+
57
+ assert.equal(summary.systemChars, "system prompt".length);
58
+ assert.equal(summary.toolCount, 2);
59
+ assert.equal(summary.messageCount, 3);
60
+ assert.equal(summary.toolResultChars, "tool output".length);
61
+ assert.equal(summary.imageCount, 1);
62
+ assert.ok(summary.toolSchemaChars > 0);
63
+ assert.ok(summary.customMessageChars > 0);
64
+ assert.ok(summary.estimatedInputTokens > 0);
65
+ assert.deepEqual(
66
+ summary.largestMessages.map((message) => Object.keys(message).sort()),
67
+ summary.largestMessages.map(() => ["chars", "index", "role", "type"]),
68
+ );
69
+ assert.equal(summary.largestTools[0].name, "large_tool");
70
+ assert.deepEqual(
71
+ summary.largestTools.map((tool) => Object.keys(tool).sort()),
72
+ summary.largestTools.map(() => ["chars", "name"]),
73
+ );
74
+ assert.deepEqual(summary.largestCustomMessages, [
75
+ { index: 3, role: "custom", customType: "gsd-memory", chars: summary.largestCustomMessages[0].chars },
76
+ ]);
77
+ assert.ok(!JSON.stringify(summary).includes("tool output"));
78
+ assert.ok(!JSON.stringify(summary).includes("memory block"));
79
+ });
80
+
81
+ test("maybeLogTokenAudit is opt-in and emits metadata only", () => {
82
+ const original = process.env.PI_TOKEN_AUDIT;
83
+ const originalWrite = process.stderr.write;
84
+ let written = "";
85
+ process.stderr.write = ((chunk: string | Uint8Array) => {
86
+ written += chunk.toString();
87
+ return true;
88
+ }) as typeof process.stderr.write;
89
+
90
+ try {
91
+ delete process.env.PI_TOKEN_AUDIT;
92
+ maybeLogTokenAudit({ messages: [{ role: "user", content: "secret prompt", timestamp: 1 }] }, []);
93
+ assert.equal(written, "");
94
+
95
+ process.env.PI_TOKEN_AUDIT = "1";
96
+ maybeLogTokenAudit({ systemPrompt: "hidden system", messages: [{ role: "user", content: "secret prompt", timestamp: 1 }] }, []);
97
+ assert.match(written, /"type":"token_audit"/);
98
+ assert.doesNotMatch(written, /secret prompt/);
99
+ assert.doesNotMatch(written, /hidden system/);
100
+ } finally {
101
+ process.stderr.write = originalWrite;
102
+ if (original === undefined) delete process.env.PI_TOKEN_AUDIT;
103
+ else process.env.PI_TOKEN_AUDIT = original;
104
+ }
105
+ });
106
+
107
+ test("provider payload audit summarizes post-hook payload without raw content", () => {
108
+ const payload = {
109
+ system: "secret system content",
110
+ tools: [{
111
+ type: "function",
112
+ function: {
113
+ name: "read",
114
+ description: "secret tool description",
115
+ parameters: { type: "object" },
116
+ },
117
+ }],
118
+ messages: [
119
+ { role: "user", content: "secret user content" },
120
+ { role: "assistant", content: [{ type: "text", text: "secret assistant content" }] },
121
+ ],
122
+ };
123
+
124
+ const summary = buildProviderPayloadAuditSummary(payload);
125
+
126
+ assert.equal(summary.messageCount, 2);
127
+ assert.equal(summary.toolCount, 1);
128
+ assert.ok(summary.payloadChars > 0);
129
+ assert.ok(summary.toolSchemaChars > 0);
130
+ assert.deepEqual(summary.largestTools.map((tool) => tool.name), ["read"]);
131
+ assert.equal(JSON.stringify(summary).includes("secret"), false);
132
+ });
133
+
134
+ test("provider payload audit recognizes Gemini and Bedrock payload shapes", () => {
135
+ const gemini = buildProviderPayloadAuditSummary({
136
+ contents: [{ role: "user", parts: [{ text: "hidden gemini prompt" }] }],
137
+ config: {
138
+ tools: [{
139
+ functionDeclarations: [
140
+ { name: "gsd_exec", description: "hidden declaration", parameters: { type: "object" } },
141
+ ],
142
+ }],
143
+ },
144
+ });
145
+ const bedrock = buildProviderPayloadAuditSummary({
146
+ messages: [{ role: "user", content: [{ text: "hidden bedrock prompt" }] }],
147
+ toolConfig: {
148
+ tools: [
149
+ { toolSpec: { name: "gsd_resume", description: "hidden tool", inputSchema: { json: {} } } },
150
+ ],
151
+ },
152
+ });
153
+
154
+ assert.equal(gemini.messageCount, 1);
155
+ assert.equal(gemini.toolCount, 1);
156
+ assert.deepEqual(gemini.largestTools.map((tool) => tool.name), ["gsd_exec"]);
157
+ assert.equal(JSON.stringify(gemini).includes("hidden"), false);
158
+
159
+ assert.equal(bedrock.messageCount, 1);
160
+ assert.equal(bedrock.toolCount, 1);
161
+ assert.deepEqual(bedrock.largestTools.map((tool) => tool.name), ["gsd_resume"]);
162
+ assert.equal(JSON.stringify(bedrock).includes("hidden"), false);
163
+ });
164
+
165
+ test("provider payload audit logging is metadata-only", () => {
166
+ const original = process.env.PI_TOKEN_AUDIT;
167
+ const originalWrite = process.stderr.write;
168
+ let written = "";
169
+ process.env.PI_TOKEN_AUDIT = "1";
170
+ process.stderr.write = ((chunk: string | Uint8Array) => {
171
+ written += chunk.toString();
172
+ return true;
173
+ }) as typeof process.stderr.write;
174
+
175
+ try {
176
+ maybeLogProviderPayloadAudit({
177
+ messages: [{ role: "user", content: "raw prompt text must not log" }],
178
+ tools: [{ name: "bash", description: "raw tool description must not log" }],
179
+ }, "after");
180
+ assert.match(written, /"type":"token_audit_provider_payload"/);
181
+ assert.match(written, /"phase":"after"/);
182
+ assert.doesNotMatch(written, /raw prompt text/);
183
+ assert.doesNotMatch(written, /raw tool description/);
184
+ } finally {
185
+ process.stderr.write = originalWrite;
186
+ if (original === undefined) delete process.env.PI_TOKEN_AUDIT;
187
+ else process.env.PI_TOKEN_AUDIT = original;
188
+ }
189
+ });
@@ -0,0 +1,287 @@
1
+ // Project/App: GSD-2
2
+ // File Purpose: Provider-boundary token payload audit helpers.
3
+
4
+ import type { Context, ImageContent, Message, TextContent, Tool } from "@gsd/pi-ai";
5
+ import type { AgentMessage } from "./types.js";
6
+
7
+ const CHARS_PER_TOKEN = 4;
8
+ const LARGEST_MESSAGE_LIMIT = 5;
9
+ const LARGEST_TOOL_LIMIT = 10;
10
+ const LARGEST_CUSTOM_MESSAGE_LIMIT = 5;
11
+
12
+ export interface TokenAuditMessageSummary {
13
+ index: number;
14
+ role: string;
15
+ type: string;
16
+ chars: number;
17
+ }
18
+
19
+ export interface TokenAuditToolSummary {
20
+ name: string;
21
+ chars: number;
22
+ }
23
+
24
+ export interface TokenAuditCustomMessageSummary {
25
+ index: number;
26
+ role: string;
27
+ customType?: string;
28
+ chars: number;
29
+ }
30
+
31
+ export interface TokenAuditSummary {
32
+ systemChars: number;
33
+ toolSchemaChars: number;
34
+ messageCharsByRole: Record<string, number>;
35
+ toolResultChars: number;
36
+ customMessageChars: number;
37
+ imageCount: number;
38
+ estimatedInputTokens: number;
39
+ messageCount: number;
40
+ toolCount: number;
41
+ largestMessages: TokenAuditMessageSummary[];
42
+ largestTools: TokenAuditToolSummary[];
43
+ largestCustomMessages: TokenAuditCustomMessageSummary[];
44
+ }
45
+
46
+ export interface ProviderPayloadAuditSummary {
47
+ payloadChars: number;
48
+ messageCharsByRole: Record<string, number>;
49
+ toolSchemaChars: number;
50
+ imageCount: number;
51
+ messageCount: number;
52
+ toolCount: number;
53
+ largestMessages: TokenAuditMessageSummary[];
54
+ largestTools: TokenAuditToolSummary[];
55
+ }
56
+
57
+ export function buildTokenAuditSummary(context: Context, sourceMessages: AgentMessage[]): TokenAuditSummary {
58
+ const systemChars = context.systemPrompt?.length ?? 0;
59
+ const toolSummaries = summarizeTools(context.tools ?? []);
60
+ const toolSchemaChars = toolSummaries.reduce((sum, tool) => sum + tool.chars, 0);
61
+ const messageSummaries = context.messages.map((message, index) => summarizeMessage(message, index));
62
+ const messageCharsByRole: Record<string, number> = {};
63
+ let toolResultChars = 0;
64
+ let imageCount = 0;
65
+
66
+ for (const summary of messageSummaries) {
67
+ messageCharsByRole[summary.role] = (messageCharsByRole[summary.role] ?? 0) + summary.chars;
68
+ }
69
+
70
+ for (const message of context.messages) {
71
+ imageCount += countImagesInMessage(message);
72
+ if (message.role === "toolResult") {
73
+ toolResultChars += countContentChars(message.content);
74
+ }
75
+ }
76
+
77
+ const customMessageSummaries = summarizeCustomMessages(sourceMessages);
78
+ const customMessageChars = customMessageSummaries.reduce((sum, message) => sum + message.chars, 0);
79
+
80
+ const totalChars =
81
+ systemChars +
82
+ toolSchemaChars +
83
+ messageSummaries.reduce((sum, message) => sum + message.chars, 0);
84
+
85
+ return {
86
+ systemChars,
87
+ toolSchemaChars,
88
+ messageCharsByRole,
89
+ toolResultChars,
90
+ customMessageChars,
91
+ imageCount,
92
+ estimatedInputTokens: Math.ceil(totalChars / CHARS_PER_TOKEN),
93
+ messageCount: context.messages.length,
94
+ toolCount: context.tools?.length ?? 0,
95
+ largestMessages: [...messageSummaries].sort((a, b) => b.chars - a.chars).slice(0, LARGEST_MESSAGE_LIMIT),
96
+ largestTools: [...toolSummaries].sort((a, b) => b.chars - a.chars).slice(0, LARGEST_TOOL_LIMIT),
97
+ largestCustomMessages: [...customMessageSummaries]
98
+ .sort((a, b) => b.chars - a.chars)
99
+ .slice(0, LARGEST_CUSTOM_MESSAGE_LIMIT),
100
+ };
101
+ }
102
+
103
+ export function maybeLogTokenAudit(context: Context, sourceMessages: AgentMessage[]): void {
104
+ if (process.env.PI_TOKEN_AUDIT !== "1") return;
105
+ const summary = buildTokenAuditSummary(context, sourceMessages);
106
+ process.stderr.write(`${JSON.stringify({ type: "token_audit", summary })}\n`);
107
+ }
108
+
109
+ export function buildProviderPayloadAuditSummary(payload: unknown): ProviderPayloadAuditSummary {
110
+ const record = asRecord(payload);
111
+ const messages = extractProviderMessages(record);
112
+ const tools = extractProviderTools(record);
113
+ const messageSummaries = messages.map((message, index) => summarizeProviderMessage(message, index));
114
+ const toolSummaries = tools.map((tool) => summarizeProviderTool(tool));
115
+ const messageCharsByRole: Record<string, number> = {};
116
+ let imageCount = 0;
117
+
118
+ for (const summary of messageSummaries) {
119
+ messageCharsByRole[summary.role] = (messageCharsByRole[summary.role] ?? 0) + summary.chars;
120
+ }
121
+ for (const message of messages) {
122
+ imageCount += countImagesInValue(message);
123
+ }
124
+
125
+ return {
126
+ payloadChars: safeJsonLength(payload),
127
+ messageCharsByRole,
128
+ toolSchemaChars: toolSummaries.reduce((sum, tool) => sum + tool.chars, 0),
129
+ imageCount,
130
+ messageCount: messages.length,
131
+ toolCount: tools.length,
132
+ largestMessages: [...messageSummaries].sort((a, b) => b.chars - a.chars).slice(0, LARGEST_MESSAGE_LIMIT),
133
+ largestTools: [...toolSummaries].sort((a, b) => b.chars - a.chars).slice(0, LARGEST_TOOL_LIMIT),
134
+ };
135
+ }
136
+
137
+ export function maybeLogProviderPayloadAudit(payload: unknown, phase: string): void {
138
+ if (process.env.PI_TOKEN_AUDIT !== "1") return;
139
+ const summary = buildProviderPayloadAuditSummary(payload);
140
+ process.stderr.write(`${JSON.stringify({ type: "token_audit_provider_payload", phase, summary })}\n`);
141
+ }
142
+
143
+ function summarizeMessage(message: Message, index: number): TokenAuditMessageSummary {
144
+ return {
145
+ index,
146
+ role: message.role,
147
+ type: messageType(message),
148
+ chars: safeJsonLength(message),
149
+ };
150
+ }
151
+
152
+ function messageType(message: Message): string {
153
+ if (message.role === "assistant") {
154
+ const types = new Set(message.content.map((content) => content.type));
155
+ return types.size > 0 ? Array.from(types).join("+") : "assistant";
156
+ }
157
+ if (message.role === "toolResult") return message.toolName || "toolResult";
158
+ if (typeof message.content === "string") return "text";
159
+ const types = new Set(message.content.map((content) => content.type));
160
+ return types.size > 0 ? Array.from(types).join("+") : "user";
161
+ }
162
+
163
+ function summarizeTools(tools: Tool[]): TokenAuditToolSummary[] {
164
+ return tools.map((tool) => ({
165
+ name: tool.name,
166
+ chars: safeJsonLength(tool),
167
+ }));
168
+ }
169
+
170
+ function summarizeCustomMessages(messages: AgentMessage[]): TokenAuditCustomMessageSummary[] {
171
+ return messages.flatMap((message, index) => {
172
+ if (!isCustomOrInjectedMessage(message)) return [];
173
+ const customType = (message as { customType?: unknown }).customType;
174
+ return [{
175
+ index,
176
+ role: String((message as { role?: unknown }).role ?? "unknown"),
177
+ customType: typeof customType === "string" ? customType : undefined,
178
+ chars: safeJsonLength(message),
179
+ }];
180
+ });
181
+ }
182
+
183
+ function summarizeProviderMessage(message: unknown, index: number): TokenAuditMessageSummary {
184
+ const record = asRecord(message);
185
+ const role = typeof record?.role === "string" ? record.role : "unknown";
186
+ const type = typeof record?.type === "string" ? record.type : role;
187
+ return { index, role, type, chars: safeJsonLength(message) };
188
+ }
189
+
190
+ function summarizeProviderTool(tool: unknown): TokenAuditToolSummary {
191
+ const record = asRecord(tool);
192
+ const name =
193
+ typeof record?.name === "string"
194
+ ? record.name
195
+ : typeof asRecord(record?.function)?.name === "string"
196
+ ? String(asRecord(record?.function)?.name)
197
+ : typeof asRecord(record?.toolSpec)?.name === "string"
198
+ ? String(asRecord(record?.toolSpec)?.name)
199
+ : "unknown";
200
+ return { name, chars: safeJsonLength(tool) };
201
+ }
202
+
203
+ function extractProviderMessages(record: Record<string, unknown> | null): unknown[] {
204
+ const candidates = [
205
+ getArrayField(record, "messages"),
206
+ getArrayField(record, "input"),
207
+ getArrayField(record, "contents"),
208
+ ].filter((value): value is unknown[] => value !== null);
209
+ if (candidates.length > 0) return candidates[0];
210
+ const input = record?.input;
211
+ return typeof input === "string" ? [{ role: "input", content: input }] : [];
212
+ }
213
+
214
+ function extractProviderTools(record: Record<string, unknown> | null): unknown[] {
215
+ const direct = getArrayField(record, "tools")
216
+ ?? getNestedArrayField(record, ["config", "tools"])
217
+ ?? getNestedArrayField(record, ["toolConfig", "tools"])
218
+ ?? getNestedArrayField(record, ["tool_config", "tools"])
219
+ ?? [];
220
+ return direct.flatMap(expandProviderToolEntry);
221
+ }
222
+
223
+ function expandProviderToolEntry(tool: unknown): unknown[] {
224
+ const record = asRecord(tool);
225
+ const functionDeclarations =
226
+ getArrayField(record, "functionDeclarations")
227
+ ?? getArrayField(record, "function_declarations");
228
+ if (functionDeclarations) return functionDeclarations;
229
+ return [tool];
230
+ }
231
+
232
+ function countImagesInMessage(message: Message): number {
233
+ if (!("content" in message) || !Array.isArray(message.content)) return 0;
234
+ return message.content.filter((content): content is ImageContent => content.type === "image").length;
235
+ }
236
+
237
+ function countImagesInValue(value: unknown): number {
238
+ if (!value || typeof value !== "object") return 0;
239
+ if (Array.isArray(value)) {
240
+ return value.reduce((sum, item) => sum + countImagesInValue(item), 0);
241
+ }
242
+ const record = value as Record<string, unknown>;
243
+ let count = record.type === "image" || record.type === "input_image" ? 1 : 0;
244
+ for (const nested of Object.values(record)) {
245
+ count += countImagesInValue(nested);
246
+ }
247
+ return count;
248
+ }
249
+
250
+ function countContentChars(content: string | (TextContent | ImageContent)[]): number {
251
+ if (typeof content === "string") return content.length;
252
+ return content.reduce((sum, block) => {
253
+ if (block.type === "text") return sum + block.text.length;
254
+ if (block.type === "image") return sum + block.data.length;
255
+ return sum;
256
+ }, 0);
257
+ }
258
+
259
+ function isCustomOrInjectedMessage(message: AgentMessage): boolean {
260
+ const role = (message as { role?: unknown }).role;
261
+ return role === "custom" || role === "bashExecution" || role === "branchSummary" || role === "compactionSummary";
262
+ }
263
+
264
+ function asRecord(value: unknown): Record<string, unknown> | null {
265
+ return value && typeof value === "object" && !Array.isArray(value) ? value as Record<string, unknown> : null;
266
+ }
267
+
268
+ function getArrayField(record: Record<string, unknown> | null, key: string): unknown[] | null {
269
+ const value = record?.[key];
270
+ return Array.isArray(value) ? value : null;
271
+ }
272
+
273
+ function getNestedArrayField(record: Record<string, unknown> | null, path: string[]): unknown[] | null {
274
+ let current: unknown = record;
275
+ for (const key of path) {
276
+ current = asRecord(current)?.[key];
277
+ }
278
+ return Array.isArray(current) ? current : null;
279
+ }
280
+
281
+ function safeJsonLength(value: unknown): number {
282
+ try {
283
+ return JSON.stringify(value)?.length ?? 0;
284
+ } catch {
285
+ return 0;
286
+ }
287
+ }
@@ -134,6 +134,20 @@ export interface AgentLoopConfig extends SimpleStreamOptions {
134
134
  */
135
135
  transformContext?: (messages: AgentMessage[], signal?: AbortSignal) => Promise<AgentMessage[]>;
136
136
 
137
+ /**
138
+ * Optional final tool filter applied immediately before each provider call.
139
+ *
140
+ * Use this for runtime policy that depends on the fully active tool set.
141
+ * The returned list is what token audit and the provider request both see.
142
+ * Receives the post-transform AgentMessage context so policy can scope
143
+ * request-local custom messages without inspecting provider payload text.
144
+ */
145
+ filterTools?: (
146
+ tools: AgentTool<any>[],
147
+ signal?: AbortSignal,
148
+ messages?: AgentMessage[],
149
+ ) => AgentTool<any>[] | Promise<AgentTool<any>[]>;
150
+
137
151
  /**
138
152
  * Resolves an API key dynamically for each LLM call.
139
153
  *