@diff-review-system/drs 4.0.0-rc.4 → 4.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (315) hide show
  1. package/.pi/agents/describe/pr-describer.md +14 -0
  2. package/.pi/agents/review/unified-reviewer.md +31 -1
  3. package/.pi/agents/task/agents-md-updater.md +3 -1
  4. package/.pi/agents/task/review-issue-fixer.md +18 -1
  5. package/.pi/agents/visual/pr-explainer.md +205 -0
  6. package/.pi/workflows/github-pr-describe.yaml +10 -7
  7. package/.pi/workflows/github-pr-fix-review-issues-stacked.yaml +148 -0
  8. package/.pi/workflows/github-pr-post-comment.yaml +10 -10
  9. package/.pi/workflows/github-pr-review-post.yaml +19 -8
  10. package/.pi/workflows/github-pr-review.yaml +348 -7
  11. package/.pi/workflows/github-pr-show-changes.yaml +8 -8
  12. package/.pi/workflows/github-pr-update-agents-md-stacked.yaml +103 -0
  13. package/.pi/workflows/github-pr-visual-explain.yaml +35 -0
  14. package/.pi/workflows/gitlab-mr-describe.yaml +8 -5
  15. package/.pi/workflows/gitlab-mr-fix-review-issues-stacked.yaml +144 -0
  16. package/.pi/workflows/gitlab-mr-post-comment.yaml +8 -8
  17. package/.pi/workflows/gitlab-mr-review.yaml +348 -5
  18. package/.pi/workflows/gitlab-mr-show-changes.yaml +6 -6
  19. package/.pi/workflows/gitlab-mr-update-agents-md-stacked.yaml +100 -0
  20. package/.pi/workflows/gitlab-mr-visual-explain.yaml +33 -0
  21. package/.pi/workflows/local-fix-review-issues.yaml +82 -13
  22. package/.pi/workflows/local-review.yaml +9 -2
  23. package/.pi/workflows/local-update-agents-md.yaml +1 -1
  24. package/.pi/workflows/local-visual-explain.yaml +31 -0
  25. package/.pi/workflows/release-changelog-finalize.yaml +47 -0
  26. package/.pi/workflows/tag-changelog-update.yaml +4 -4
  27. package/README.md +91 -27
  28. package/dist/ci/runner.d.ts.map +1 -1
  29. package/dist/ci/runner.js +3 -1
  30. package/dist/ci/runner.js.map +1 -1
  31. package/dist/cli/index.js +48 -6
  32. package/dist/cli/index.js.map +1 -1
  33. package/dist/cli/run-agent.d.ts +2 -0
  34. package/dist/cli/run-agent.d.ts.map +1 -1
  35. package/dist/cli/run-agent.js +4 -0
  36. package/dist/cli/run-agent.js.map +1 -1
  37. package/dist/cli/workflow.d.ts +56 -2
  38. package/dist/cli/workflow.d.ts.map +1 -1
  39. package/dist/cli/workflow.js +2165 -85
  40. package/dist/cli/workflow.js.map +1 -1
  41. package/dist/github/client.d.ts +12 -0
  42. package/dist/github/client.d.ts.map +1 -1
  43. package/dist/github/client.js +27 -0
  44. package/dist/github/client.js.map +1 -1
  45. package/dist/github/platform-adapter.d.ts +6 -1
  46. package/dist/github/platform-adapter.d.ts.map +1 -1
  47. package/dist/github/platform-adapter.js +84 -8
  48. package/dist/github/platform-adapter.js.map +1 -1
  49. package/dist/gitlab/client.d.ts +11 -0
  50. package/dist/gitlab/client.d.ts.map +1 -1
  51. package/dist/gitlab/client.js +11 -0
  52. package/dist/gitlab/client.js.map +1 -1
  53. package/dist/gitlab/platform-adapter.d.ts +3 -1
  54. package/dist/gitlab/platform-adapter.d.ts.map +1 -1
  55. package/dist/gitlab/platform-adapter.js +32 -1
  56. package/dist/gitlab/platform-adapter.js.map +1 -1
  57. package/dist/lib/comment-formatter.d.ts +8 -0
  58. package/dist/lib/comment-formatter.d.ts.map +1 -1
  59. package/dist/lib/comment-formatter.js +12 -4
  60. package/dist/lib/comment-formatter.js.map +1 -1
  61. package/dist/lib/comment-poster.d.ts.map +1 -1
  62. package/dist/lib/comment-poster.js +28 -1
  63. package/dist/lib/comment-poster.js.map +1 -1
  64. package/dist/lib/config.d.ts +50 -11
  65. package/dist/lib/config.d.ts.map +1 -1
  66. package/dist/lib/config.js +163 -28
  67. package/dist/lib/config.js.map +1 -1
  68. package/dist/lib/context-compression.d.ts +10 -0
  69. package/dist/lib/context-compression.d.ts.map +1 -1
  70. package/dist/lib/context-compression.js +101 -13
  71. package/dist/lib/context-compression.js.map +1 -1
  72. package/dist/lib/context-loader.d.ts +2 -1
  73. package/dist/lib/context-loader.d.ts.map +1 -1
  74. package/dist/lib/context-loader.js +70 -1
  75. package/dist/lib/context-loader.js.map +1 -1
  76. package/dist/lib/describe-core.d.ts.map +1 -1
  77. package/dist/lib/describe-core.js +3 -2
  78. package/dist/lib/describe-core.js.map +1 -1
  79. package/dist/lib/diff-lines.d.ts +9 -0
  80. package/dist/lib/diff-lines.d.ts.map +1 -1
  81. package/dist/lib/diff-lines.js +17 -9
  82. package/dist/lib/diff-lines.js.map +1 -1
  83. package/dist/lib/exit.js +4 -4
  84. package/dist/lib/exit.js.map +1 -1
  85. package/dist/lib/html-artifact.d.ts +14 -0
  86. package/dist/lib/html-artifact.d.ts.map +1 -0
  87. package/dist/lib/html-artifact.js +59 -0
  88. package/dist/lib/html-artifact.js.map +1 -0
  89. package/dist/lib/issue-parser.js +3 -3
  90. package/dist/lib/issue-parser.js.map +1 -1
  91. package/dist/lib/json-output-schema.d.ts +70 -0
  92. package/dist/lib/json-output-schema.d.ts.map +1 -1
  93. package/dist/lib/json-output-schema.js +40 -0
  94. package/dist/lib/json-output-schema.js.map +1 -1
  95. package/dist/lib/platform-client.d.ts +26 -0
  96. package/dist/lib/platform-client.d.ts.map +1 -1
  97. package/dist/lib/review-artifact.d.ts +69 -0
  98. package/dist/lib/review-artifact.d.ts.map +1 -0
  99. package/dist/lib/review-artifact.js +171 -0
  100. package/dist/lib/review-artifact.js.map +1 -0
  101. package/dist/lib/review-core.d.ts +6 -4
  102. package/dist/lib/review-core.d.ts.map +1 -1
  103. package/dist/lib/review-core.js +71 -151
  104. package/dist/lib/review-core.js.map +1 -1
  105. package/dist/lib/review-orchestrator.d.ts +23 -0
  106. package/dist/lib/review-orchestrator.d.ts.map +1 -1
  107. package/dist/lib/review-orchestrator.js +20 -13
  108. package/dist/lib/review-orchestrator.js.map +1 -1
  109. package/dist/lib/review-usage.d.ts +4 -0
  110. package/dist/lib/review-usage.d.ts.map +1 -1
  111. package/dist/lib/review-usage.js +25 -0
  112. package/dist/lib/review-usage.js.map +1 -1
  113. package/dist/lib/trace-collector.d.ts +105 -0
  114. package/dist/lib/trace-collector.d.ts.map +1 -0
  115. package/dist/lib/trace-collector.js +255 -0
  116. package/dist/lib/trace-collector.js.map +1 -0
  117. package/dist/lib/trace-html.d.ts +3 -0
  118. package/dist/lib/trace-html.d.ts.map +1 -0
  119. package/dist/lib/trace-html.js +349 -0
  120. package/dist/lib/trace-html.js.map +1 -0
  121. package/dist/lib/workflow-artifacts.d.ts +54 -0
  122. package/dist/lib/workflow-artifacts.d.ts.map +1 -0
  123. package/dist/lib/workflow-artifacts.js +150 -0
  124. package/dist/lib/workflow-artifacts.js.map +1 -0
  125. package/dist/pi/sdk.d.ts.map +1 -1
  126. package/dist/pi/sdk.js +570 -6
  127. package/dist/pi/sdk.js.map +1 -1
  128. package/dist/runtime/agent-loader.js +2 -2
  129. package/dist/runtime/client.d.ts +2 -0
  130. package/dist/runtime/client.d.ts.map +1 -1
  131. package/dist/runtime/client.js +11 -5
  132. package/dist/runtime/client.js.map +1 -1
  133. package/package.json +21 -15
  134. package/.pi/workflows/github-pr-describe-post.yaml +0 -24
  135. package/.pi/workflows/gitlab-mr-describe-post.yaml +0 -22
  136. package/.pi/workflows/gitlab-mr-review-code-quality.yaml +0 -31
  137. package/.pi/workflows/gitlab-mr-review-post-code-quality.yaml +0 -40
  138. package/.pi/workflows/gitlab-mr-review-post.yaml +0 -30
  139. package/.pi/workflows/local-staged-review.yaml +0 -17
  140. package/dist/cli/run-agent.test.d.ts +0 -2
  141. package/dist/cli/run-agent.test.d.ts.map +0 -1
  142. package/dist/cli/run-agent.test.js +0 -204
  143. package/dist/cli/run-agent.test.js.map +0 -1
  144. package/dist/cli/workflow.test.d.ts +0 -2
  145. package/dist/cli/workflow.test.d.ts.map +0 -1
  146. package/dist/cli/workflow.test.js +0 -1410
  147. package/dist/cli/workflow.test.js.map +0 -1
  148. package/dist/github/client.test.d.ts +0 -2
  149. package/dist/github/client.test.d.ts.map +0 -1
  150. package/dist/github/client.test.js +0 -206
  151. package/dist/github/client.test.js.map +0 -1
  152. package/dist/github/platform-adapter.test.d.ts +0 -2
  153. package/dist/github/platform-adapter.test.d.ts.map +0 -1
  154. package/dist/github/platform-adapter.test.js +0 -40
  155. package/dist/github/platform-adapter.test.js.map +0 -1
  156. package/dist/gitlab/diff-parser.test.d.ts +0 -2
  157. package/dist/gitlab/diff-parser.test.d.ts.map +0 -1
  158. package/dist/gitlab/diff-parser.test.js +0 -315
  159. package/dist/gitlab/diff-parser.test.js.map +0 -1
  160. package/dist/gitlab/platform-adapter.test.d.ts +0 -2
  161. package/dist/gitlab/platform-adapter.test.d.ts.map +0 -1
  162. package/dist/gitlab/platform-adapter.test.js +0 -21
  163. package/dist/gitlab/platform-adapter.test.js.map +0 -1
  164. package/dist/index.test.d.ts +0 -2
  165. package/dist/index.test.d.ts.map +0 -1
  166. package/dist/index.test.js +0 -7
  167. package/dist/index.test.js.map +0 -1
  168. package/dist/lib/code-quality-report.test.d.ts +0 -2
  169. package/dist/lib/code-quality-report.test.d.ts.map +0 -1
  170. package/dist/lib/code-quality-report.test.js +0 -327
  171. package/dist/lib/code-quality-report.test.js.map +0 -1
  172. package/dist/lib/comment-formatter.test.d.ts +0 -2
  173. package/dist/lib/comment-formatter.test.d.ts.map +0 -1
  174. package/dist/lib/comment-formatter.test.js +0 -727
  175. package/dist/lib/comment-formatter.test.js.map +0 -1
  176. package/dist/lib/comment-manager.test.d.ts +0 -2
  177. package/dist/lib/comment-manager.test.d.ts.map +0 -1
  178. package/dist/lib/comment-manager.test.js +0 -680
  179. package/dist/lib/comment-manager.test.js.map +0 -1
  180. package/dist/lib/comment-poster.test.d.ts +0 -5
  181. package/dist/lib/comment-poster.test.d.ts.map +0 -1
  182. package/dist/lib/comment-poster.test.js +0 -255
  183. package/dist/lib/comment-poster.test.js.map +0 -1
  184. package/dist/lib/config-model-overrides.test.d.ts +0 -2
  185. package/dist/lib/config-model-overrides.test.d.ts.map +0 -1
  186. package/dist/lib/config-model-overrides.test.js +0 -218
  187. package/dist/lib/config-model-overrides.test.js.map +0 -1
  188. package/dist/lib/config.test.d.ts +0 -2
  189. package/dist/lib/config.test.d.ts.map +0 -1
  190. package/dist/lib/config.test.js +0 -353
  191. package/dist/lib/config.test.js.map +0 -1
  192. package/dist/lib/context-compression.test.d.ts +0 -2
  193. package/dist/lib/context-compression.test.d.ts.map +0 -1
  194. package/dist/lib/context-compression.test.js +0 -337
  195. package/dist/lib/context-compression.test.js.map +0 -1
  196. package/dist/lib/context-loader.test.d.ts +0 -2
  197. package/dist/lib/context-loader.test.d.ts.map +0 -1
  198. package/dist/lib/context-loader.test.js +0 -212
  199. package/dist/lib/context-loader.test.js.map +0 -1
  200. package/dist/lib/cursor-fix-link.test.d.ts +0 -2
  201. package/dist/lib/cursor-fix-link.test.d.ts.map +0 -1
  202. package/dist/lib/cursor-fix-link.test.js +0 -70
  203. package/dist/lib/cursor-fix-link.test.js.map +0 -1
  204. package/dist/lib/describe-core.test.d.ts +0 -2
  205. package/dist/lib/describe-core.test.d.ts.map +0 -1
  206. package/dist/lib/describe-core.test.js +0 -208
  207. package/dist/lib/describe-core.test.js.map +0 -1
  208. package/dist/lib/describe-output-path.test.d.ts +0 -2
  209. package/dist/lib/describe-output-path.test.d.ts.map +0 -1
  210. package/dist/lib/describe-output-path.test.js +0 -51
  211. package/dist/lib/describe-output-path.test.js.map +0 -1
  212. package/dist/lib/describe-parser.test.d.ts +0 -2
  213. package/dist/lib/describe-parser.test.d.ts.map +0 -1
  214. package/dist/lib/describe-parser.test.js +0 -282
  215. package/dist/lib/describe-parser.test.js.map +0 -1
  216. package/dist/lib/description-executor.test.d.ts +0 -2
  217. package/dist/lib/description-executor.test.d.ts.map +0 -1
  218. package/dist/lib/description-executor.test.js +0 -135
  219. package/dist/lib/description-executor.test.js.map +0 -1
  220. package/dist/lib/description-formatter.test.d.ts +0 -2
  221. package/dist/lib/description-formatter.test.d.ts.map +0 -1
  222. package/dist/lib/description-formatter.test.js +0 -57
  223. package/dist/lib/description-formatter.test.js.map +0 -1
  224. package/dist/lib/diff-lines.test.d.ts +0 -2
  225. package/dist/lib/diff-lines.test.d.ts.map +0 -1
  226. package/dist/lib/diff-lines.test.js +0 -13
  227. package/dist/lib/diff-lines.test.js.map +0 -1
  228. package/dist/lib/diff-parser.test.d.ts +0 -2
  229. package/dist/lib/diff-parser.test.d.ts.map +0 -1
  230. package/dist/lib/diff-parser.test.js +0 -335
  231. package/dist/lib/diff-parser.test.js.map +0 -1
  232. package/dist/lib/error-comment-poster.test.d.ts +0 -2
  233. package/dist/lib/error-comment-poster.test.d.ts.map +0 -1
  234. package/dist/lib/error-comment-poster.test.js +0 -128
  235. package/dist/lib/error-comment-poster.test.js.map +0 -1
  236. package/dist/lib/exit.test.d.ts +0 -2
  237. package/dist/lib/exit.test.d.ts.map +0 -1
  238. package/dist/lib/exit.test.js +0 -120
  239. package/dist/lib/exit.test.js.map +0 -1
  240. package/dist/lib/issue-parser.test.d.ts +0 -2
  241. package/dist/lib/issue-parser.test.d.ts.map +0 -1
  242. package/dist/lib/issue-parser.test.js +0 -281
  243. package/dist/lib/issue-parser.test.js.map +0 -1
  244. package/dist/lib/json-output-schema.test.d.ts +0 -2
  245. package/dist/lib/json-output-schema.test.d.ts.map +0 -1
  246. package/dist/lib/json-output-schema.test.js +0 -92
  247. package/dist/lib/json-output-schema.test.js.map +0 -1
  248. package/dist/lib/json-output.test.d.ts +0 -2
  249. package/dist/lib/json-output.test.d.ts.map +0 -1
  250. package/dist/lib/json-output.test.js +0 -141
  251. package/dist/lib/json-output.test.js.map +0 -1
  252. package/dist/lib/logger.test.d.ts +0 -2
  253. package/dist/lib/logger.test.d.ts.map +0 -1
  254. package/dist/lib/logger.test.js +0 -324
  255. package/dist/lib/logger.test.js.map +0 -1
  256. package/dist/lib/position-validator.test.d.ts +0 -2
  257. package/dist/lib/position-validator.test.d.ts.map +0 -1
  258. package/dist/lib/position-validator.test.js +0 -128
  259. package/dist/lib/position-validator.test.js.map +0 -1
  260. package/dist/lib/prompt-budget.test.d.ts +0 -2
  261. package/dist/lib/prompt-budget.test.d.ts.map +0 -1
  262. package/dist/lib/prompt-budget.test.js +0 -55
  263. package/dist/lib/prompt-budget.test.js.map +0 -1
  264. package/dist/lib/repository-validator.test.d.ts +0 -5
  265. package/dist/lib/repository-validator.test.d.ts.map +0 -1
  266. package/dist/lib/repository-validator.test.js +0 -341
  267. package/dist/lib/repository-validator.test.js.map +0 -1
  268. package/dist/lib/review-core.test.d.ts +0 -2
  269. package/dist/lib/review-core.test.d.ts.map +0 -1
  270. package/dist/lib/review-core.test.js +0 -614
  271. package/dist/lib/review-core.test.js.map +0 -1
  272. package/dist/lib/review-orchestrator.test.d.ts +0 -2
  273. package/dist/lib/review-orchestrator.test.d.ts.map +0 -1
  274. package/dist/lib/review-orchestrator.test.js +0 -552
  275. package/dist/lib/review-orchestrator.test.js.map +0 -1
  276. package/dist/lib/review-output-path.test.d.ts +0 -2
  277. package/dist/lib/review-output-path.test.d.ts.map +0 -1
  278. package/dist/lib/review-output-path.test.js +0 -83
  279. package/dist/lib/review-output-path.test.js.map +0 -1
  280. package/dist/lib/review-parser.test.d.ts +0 -2
  281. package/dist/lib/review-parser.test.d.ts.map +0 -1
  282. package/dist/lib/review-parser.test.js +0 -130
  283. package/dist/lib/review-parser.test.js.map +0 -1
  284. package/dist/lib/review-usage.test.d.ts +0 -2
  285. package/dist/lib/review-usage.test.d.ts.map +0 -1
  286. package/dist/lib/review-usage.test.js +0 -83
  287. package/dist/lib/review-usage.test.js.map +0 -1
  288. package/dist/lib/unified-review-executor.d.ts +0 -58
  289. package/dist/lib/unified-review-executor.d.ts.map +0 -1
  290. package/dist/lib/unified-review-executor.js +0 -201
  291. package/dist/lib/unified-review-executor.js.map +0 -1
  292. package/dist/lib/unified-review-executor.test.d.ts +0 -5
  293. package/dist/lib/unified-review-executor.test.d.ts.map +0 -1
  294. package/dist/lib/unified-review-executor.test.js +0 -472
  295. package/dist/lib/unified-review-executor.test.js.map +0 -1
  296. package/dist/lib/write-json-output.test.d.ts +0 -2
  297. package/dist/lib/write-json-output.test.d.ts.map +0 -1
  298. package/dist/lib/write-json-output.test.js +0 -259
  299. package/dist/lib/write-json-output.test.js.map +0 -1
  300. package/dist/pi/sdk.test.d.ts +0 -2
  301. package/dist/pi/sdk.test.d.ts.map +0 -1
  302. package/dist/pi/sdk.test.js +0 -488
  303. package/dist/pi/sdk.test.js.map +0 -1
  304. package/dist/runtime/agent-loader.test.d.ts +0 -2
  305. package/dist/runtime/agent-loader.test.d.ts.map +0 -1
  306. package/dist/runtime/agent-loader.test.js +0 -277
  307. package/dist/runtime/agent-loader.test.js.map +0 -1
  308. package/dist/runtime/client.test.d.ts +0 -2
  309. package/dist/runtime/client.test.d.ts.map +0 -1
  310. package/dist/runtime/client.test.js +0 -772
  311. package/dist/runtime/client.test.js.map +0 -1
  312. package/dist/runtime/path-config.test.d.ts +0 -2
  313. package/dist/runtime/path-config.test.d.ts.map +0 -1
  314. package/dist/runtime/path-config.test.js +0 -112
  315. package/dist/runtime/path-config.test.js.map +0 -1
package/dist/pi/sdk.js CHANGED
@@ -1,8 +1,15 @@
1
1
  import { randomUUID } from 'crypto';
2
+ import { spawn } from 'child_process';
2
3
  import { isAbsolute, join, resolve } from 'path';
3
4
  import { Type } from '@sinclair/typebox';
4
- import { AuthStorage, createAgentSession, DefaultResourceLoader, getAgentDir, ModelRegistry, SettingsManager, SessionManager, } from '@mariozechner/pi-coding-agent';
5
+ import { AuthStorage, createAgentSession, DefaultResourceLoader, getAgentDir, ModelRegistry, SettingsManager, SessionManager, } from '@earendil-works/pi-coding-agent';
5
6
  import { writeJsonOutput } from '../lib/write-json-output.js';
7
+ import { writeArtifactOutput } from '../lib/html-artifact.js';
8
+ import { isReviewArtifactPayload } from '../lib/review-artifact.js';
9
+ import { resolveWithinWorkingDir } from '../lib/path-utils.js';
10
+ const DEFAULT_GIT_DIFF_MAX_BYTES = 120_000;
11
+ const HARD_GIT_DIFF_MAX_BYTES = 500_000;
12
+ const GIT_DIFF_STDERR_MAX_BYTES = 16_384;
6
13
  function asRecord(value) {
7
14
  return value && typeof value === 'object' && !Array.isArray(value)
8
15
  ? value
@@ -148,6 +155,162 @@ function asPositiveInt(value) {
148
155
  const rounded = Math.round(value);
149
156
  return rounded > 0 ? rounded : undefined;
150
157
  }
158
+ function formatUnknownError(error) {
159
+ if (error instanceof Error) {
160
+ return error.message;
161
+ }
162
+ if (typeof error === 'string') {
163
+ return error;
164
+ }
165
+ try {
166
+ return JSON.stringify(error);
167
+ }
168
+ catch {
169
+ return 'Unknown error';
170
+ }
171
+ }
172
+ function normalizeGitDiffPath(file) {
173
+ const trimmed = file.trim();
174
+ if (!trimmed) {
175
+ throw new Error('git_diff requires a non-empty file path');
176
+ }
177
+ if (trimmed.includes('\0')) {
178
+ throw new Error('git_diff file path must not contain NUL bytes');
179
+ }
180
+ if (isAbsolute(trimmed)) {
181
+ throw new Error('git_diff only accepts repository-relative file paths');
182
+ }
183
+ if (trimmed.split(/[\\/]+/).includes('..')) {
184
+ throw new Error('git_diff file path must stay inside the repository');
185
+ }
186
+ const resolved = resolve('/', trimmed);
187
+ const relativePath = resolved.slice(1);
188
+ if (!relativePath || relativePath === '..' || relativePath.startsWith('../')) {
189
+ throw new Error('git_diff file path must stay inside the repository');
190
+ }
191
+ return relativePath;
192
+ }
193
+ function normalizeGitRev(value, label) {
194
+ const trimmed = value?.trim();
195
+ if (!trimmed) {
196
+ return undefined;
197
+ }
198
+ if (trimmed.startsWith('-') || trimmed.includes('\0') || /\s/.test(trimmed)) {
199
+ throw new Error(`git_diff ${label} revision is not allowed`);
200
+ }
201
+ return trimmed;
202
+ }
203
+ function normalizeGitDiffMaxBytes(maxBytes) {
204
+ if (!maxBytes || !Number.isFinite(maxBytes)) {
205
+ return DEFAULT_GIT_DIFF_MAX_BYTES;
206
+ }
207
+ return Math.max(1, Math.min(Math.floor(maxBytes), HARD_GIT_DIFF_MAX_BYTES));
208
+ }
209
+ function appendCappedChunk(chunks, chunk, maxBytes) {
210
+ const currentBytes = chunks.reduce((sum, existing) => sum + existing.length, 0);
211
+ const remaining = maxBytes - currentBytes;
212
+ if (remaining <= 0) {
213
+ return chunk.length > 0;
214
+ }
215
+ if (chunk.length <= remaining) {
216
+ chunks.push(chunk);
217
+ return false;
218
+ }
219
+ chunks.push(chunk.subarray(0, remaining));
220
+ return true;
221
+ }
222
+ function runGitDiff(workingDir, args, maxBytes) {
223
+ return new Promise((resolvePromise, reject) => {
224
+ const child = spawn('git', ['diff', '--no-ext-diff', '-M', ...args], {
225
+ cwd: workingDir,
226
+ stdio: ['ignore', 'pipe', 'pipe'],
227
+ });
228
+ const stdoutChunks = [];
229
+ const stderrChunks = [];
230
+ let truncated = false;
231
+ child.stdout.on('data', (chunk) => {
232
+ truncated = appendCappedChunk(stdoutChunks, chunk, maxBytes) || truncated;
233
+ });
234
+ child.stderr.on('data', (chunk) => {
235
+ appendCappedChunk(stderrChunks, chunk, GIT_DIFF_STDERR_MAX_BYTES);
236
+ });
237
+ child.on('error', reject);
238
+ child.on('close', (code) => {
239
+ const stderr = Buffer.concat(stderrChunks).toString('utf8').trim();
240
+ if (code !== 0) {
241
+ reject(new Error(stderr || `git diff exited with code ${code ?? 'unknown'}`));
242
+ return;
243
+ }
244
+ resolvePromise({
245
+ stdout: Buffer.concat(stdoutChunks).toString('utf8'),
246
+ truncated,
247
+ });
248
+ });
249
+ });
250
+ }
251
+ function runGitNameStatus(workingDir, args) {
252
+ return new Promise((resolvePromise, reject) => {
253
+ const child = spawn('git', ['diff', '--name-status', '-M', ...args], {
254
+ cwd: workingDir,
255
+ stdio: ['ignore', 'pipe', 'pipe'],
256
+ });
257
+ const stdoutChunks = [];
258
+ const stderrChunks = [];
259
+ child.stdout.on('data', (chunk) => {
260
+ stdoutChunks.push(chunk);
261
+ });
262
+ child.stderr.on('data', (chunk) => {
263
+ appendCappedChunk(stderrChunks, chunk, GIT_DIFF_STDERR_MAX_BYTES);
264
+ });
265
+ child.on('error', reject);
266
+ child.on('close', (code) => {
267
+ const stderr = Buffer.concat(stderrChunks).toString('utf8').trim();
268
+ if (code !== 0) {
269
+ reject(new Error(stderr || `git diff --name-status exited with code ${code ?? 'unknown'}`));
270
+ return;
271
+ }
272
+ resolvePromise(Buffer.concat(stdoutChunks).toString('utf8'));
273
+ });
274
+ });
275
+ }
276
+ function extractGitDiffMetadata(diff) {
277
+ const lines = diff.split('\n');
278
+ const oldPath = lines
279
+ .find((line) => line.startsWith('rename from '))
280
+ ?.slice('rename from '.length);
281
+ const newPath = lines.find((line) => line.startsWith('rename to '))?.slice('rename to '.length);
282
+ return {
283
+ binary: lines.some((line) => line.startsWith('Binary files ') || line.startsWith('GIT binary patch')),
284
+ deleted: lines.some((line) => line.startsWith('deleted file mode')),
285
+ renamed: oldPath !== undefined || newPath !== undefined,
286
+ oldPath,
287
+ newPath,
288
+ empty: diff.trim().length === 0,
289
+ };
290
+ }
291
+ function mergeGitNameStatusMetadata(metadata, nameStatus, file) {
292
+ for (const line of nameStatus.split('\n')) {
293
+ const parts = line.split('\t');
294
+ const status = parts[0];
295
+ if (!status)
296
+ continue;
297
+ if (status.startsWith('R') && parts[2] === file) {
298
+ return {
299
+ ...metadata,
300
+ renamed: true,
301
+ oldPath: parts[1],
302
+ newPath: parts[2],
303
+ };
304
+ }
305
+ if (status === 'D' && parts[1] === file) {
306
+ return {
307
+ ...metadata,
308
+ deleted: true,
309
+ };
310
+ }
311
+ }
312
+ return metadata;
313
+ }
151
314
  function normalizeAgentSkills(value) {
152
315
  const normalized = {};
153
316
  for (const [agentName, skills] of Object.entries(value)) {
@@ -158,6 +321,188 @@ function normalizeAgentSkills(value) {
158
321
  }
159
322
  return normalized;
160
323
  }
324
+ function parseFixChecks(value) {
325
+ if (!Array.isArray(value)) {
326
+ return undefined;
327
+ }
328
+ const checks = [];
329
+ for (const entry of value) {
330
+ const record = asRecord(entry);
331
+ const name = asString(record.name);
332
+ const command = asString(record.command);
333
+ if (!name || !command) {
334
+ continue;
335
+ }
336
+ const check = { name, command };
337
+ const matchPaths = asStringArray(record.matchPaths);
338
+ if (matchPaths && matchPaths.length > 0) {
339
+ check.matchPaths = matchPaths;
340
+ }
341
+ const timeoutMs = asNumber(record.timeoutMs);
342
+ if (timeoutMs !== undefined) {
343
+ check.timeoutMs = timeoutMs;
344
+ }
345
+ checks.push(check);
346
+ }
347
+ return checks.length > 0 ? checks : undefined;
348
+ }
349
+ const DEFAULT_CHECK_TIMEOUT_MS = 120_000;
350
+ const HARD_CHECK_TIMEOUT_MS = 600_000;
351
+ const CHECK_OUTPUT_MAX_BYTES = 64_000;
352
+ function normalizeCheckTimeout(timeoutMs) {
353
+ if (!timeoutMs || !Number.isFinite(timeoutMs)) {
354
+ return DEFAULT_CHECK_TIMEOUT_MS;
355
+ }
356
+ return Math.max(1_000, Math.min(Math.floor(timeoutMs), HARD_CHECK_TIMEOUT_MS));
357
+ }
358
+ function globToRegex(pattern) {
359
+ let regex = '';
360
+ let i = 0;
361
+ while (i < pattern.length) {
362
+ const char = pattern[i];
363
+ if (char === '*' && pattern[i + 1] === '*') {
364
+ regex += '.*';
365
+ i += 2;
366
+ if (pattern[i] === '/') {
367
+ i++;
368
+ }
369
+ }
370
+ else if (char === '*') {
371
+ regex += '[^/]*';
372
+ i++;
373
+ }
374
+ else if (char === '?') {
375
+ regex += '[^/]';
376
+ i++;
377
+ }
378
+ else if ('.+^${}()|[]\\'.includes(char)) {
379
+ regex += '\\' + char;
380
+ i++;
381
+ }
382
+ else {
383
+ regex += char;
384
+ i++;
385
+ }
386
+ }
387
+ return new RegExp('^' + regex + '$');
388
+ }
389
+ function fileMatchesGlobs(filePath, patterns) {
390
+ if (!patterns || patterns.length === 0) {
391
+ return true;
392
+ }
393
+ return patterns.some((pattern) => globToRegex(pattern).test(filePath));
394
+ }
395
+ async function getChangedFiles(workingDir) {
396
+ return new Promise((resolvePromise) => {
397
+ const child = spawn('git', ['status', '--porcelain', '--no-renames'], {
398
+ cwd: workingDir,
399
+ stdio: ['ignore', 'pipe', 'pipe'],
400
+ });
401
+ const stdoutChunks = [];
402
+ child.stdout.on('data', (chunk) => stdoutChunks.push(chunk));
403
+ child.on('error', () => resolvePromise([]));
404
+ child.on('close', () => {
405
+ const output = Buffer.concat(stdoutChunks).toString('utf8').trim();
406
+ resolvePromise(output
407
+ ? output
408
+ .split('\n')
409
+ .map((line) => line.slice(3).trim())
410
+ .filter(Boolean)
411
+ : []);
412
+ });
413
+ });
414
+ }
415
+ function runCheckCommand(workingDir, check) {
416
+ return new Promise((resolvePromise) => {
417
+ const timeoutMs = normalizeCheckTimeout(check.timeoutMs);
418
+ const child = spawn(check.command, {
419
+ cwd: workingDir,
420
+ shell: true,
421
+ stdio: ['ignore', 'pipe', 'pipe'],
422
+ });
423
+ const stdoutChunks = [];
424
+ const stderrChunks = [];
425
+ let truncated = false;
426
+ let timedOut = false;
427
+ const startTime = Date.now();
428
+ const timer = setTimeout(() => {
429
+ timedOut = true;
430
+ child.kill('SIGTERM');
431
+ }, timeoutMs);
432
+ child.stdout.on('data', (chunk) => {
433
+ truncated = appendCappedChunk(stdoutChunks, chunk, CHECK_OUTPUT_MAX_BYTES) || truncated;
434
+ });
435
+ child.stderr.on('data', (chunk) => {
436
+ appendCappedChunk(stderrChunks, chunk, CHECK_OUTPUT_MAX_BYTES);
437
+ });
438
+ child.on('error', () => {
439
+ clearTimeout(timer);
440
+ resolvePromise({
441
+ name: check.name,
442
+ command: check.command,
443
+ exitCode: null,
444
+ stdout: '',
445
+ stderr: `Failed to spawn: ${check.command}`,
446
+ truncated: false,
447
+ skipped: false,
448
+ durationMs: Date.now() - startTime,
449
+ });
450
+ });
451
+ child.on('close', (code) => {
452
+ clearTimeout(timer);
453
+ resolvePromise({
454
+ name: check.name,
455
+ command: check.command,
456
+ exitCode: timedOut ? null : code,
457
+ stdout: Buffer.concat(stdoutChunks).toString('utf8'),
458
+ stderr: (timedOut ? `Timed out after ${timeoutMs}ms\n` : '') +
459
+ Buffer.concat(stderrChunks).toString('utf8'),
460
+ truncated,
461
+ skipped: false,
462
+ durationMs: Date.now() - startTime,
463
+ });
464
+ });
465
+ });
466
+ }
467
+ function findingIdParamProvided(params) {
468
+ return typeof params.findingId === 'string' && params.findingId.trim().length > 0;
469
+ }
470
+ function extractFindingFromArtifact(raw, findingId) {
471
+ const envelope = raw;
472
+ const payload = envelope?.payload ?? raw;
473
+ if (!isReviewArtifactPayload(payload)) {
474
+ return undefined;
475
+ }
476
+ return payload.findings.find((f) => f.id === findingId);
477
+ }
478
+ function buildArtifactManifest(raw, artifactPath) {
479
+ const envelope = raw;
480
+ const payload = envelope?.payload ?? raw;
481
+ if (!isReviewArtifactPayload(payload)) {
482
+ return { artifactPath, ok: false, totalFindings: 0, actionableFindings: 0, findings: [] };
483
+ }
484
+ const findings = payload.findings.map((f) => ({
485
+ id: f.id,
486
+ severity: f.issue.severity,
487
+ state: f.state,
488
+ disposition: f.disposition,
489
+ file: f.issue.file,
490
+ line: f.issue.line,
491
+ title: f.issue.title,
492
+ }));
493
+ const actionableFindings = findings.filter((f) => f.state === 'open' &&
494
+ (f.disposition === 'still_open' ||
495
+ f.disposition === 'partial' ||
496
+ f.disposition === 'regression')).length;
497
+ return {
498
+ artifactPath,
499
+ ok: true,
500
+ reviewId: payload.reviewId,
501
+ totalFindings: findings.length,
502
+ actionableFindings,
503
+ findings,
504
+ };
505
+ }
161
506
  function normalizeSkillPath(cwd, skillPath) {
162
507
  return isAbsolute(skillPath) ? resolve(skillPath) : resolve(cwd, skillPath);
163
508
  }
@@ -255,6 +600,8 @@ class PiSessionRuntime {
255
600
  skillSearchPaths: asStringArray(config.skillSearchPaths),
256
601
  agentSkills: normalizeAgentSkills(asRecord(config.agentSkills)),
257
602
  thinkingLevel: asString(config.thinkingLevel),
603
+ fixChecks: parseFixChecks(config.fixChecks),
604
+ traceCollector: config.traceCollector,
258
605
  retry: asRecord(config.retry),
259
606
  };
260
607
  this.authStorage = AuthStorage.create();
@@ -416,9 +763,9 @@ class PiSessionRuntime {
416
763
  resolveAgentSkills(agentName) {
417
764
  return this.runtimeConfig.agentSkills?.[agentName] ?? [];
418
765
  }
419
- resolveCustomTools(workingDir) {
766
+ resolveCustomTools(workingDir, agentTools) {
420
767
  const customTools = [];
421
- if (this.isToolEnabled('write_json_output', true)) {
768
+ if (this.isToolEnabled('write_json_output', true, agentTools)) {
422
769
  customTools.push({
423
770
  name: 'write_json_output',
424
771
  label: 'write_json_output',
@@ -444,6 +791,210 @@ class PiSessionRuntime {
444
791
  },
445
792
  });
446
793
  }
794
+ if (this.isToolEnabled('write_artifact_output', false, agentTools)) {
795
+ customTools.push({
796
+ name: 'write_artifact_output',
797
+ label: 'write_artifact_output',
798
+ description: 'Validate and write a self-contained HTML artifact for DRS agents.',
799
+ parameters: Type.Object({
800
+ outputPath: Type.String({ minLength: 1 }),
801
+ content: Type.String({ minLength: 1 }),
802
+ }),
803
+ execute: async (_toolCallId, params) => {
804
+ const pointer = await writeArtifactOutput({
805
+ outputPath: params.outputPath,
806
+ content: params.content,
807
+ workingDir,
808
+ });
809
+ return {
810
+ content: [{ type: 'text', text: JSON.stringify(pointer) }],
811
+ details: pointer,
812
+ };
813
+ },
814
+ });
815
+ }
816
+ if (this.isToolEnabled('git_diff', false, agentTools)) {
817
+ customTools.push({
818
+ name: 'git_diff',
819
+ label: 'git_diff',
820
+ description: 'Read a unified git diff for one repository-relative file path.',
821
+ parameters: Type.Object({
822
+ file: Type.String({ minLength: 1 }),
823
+ base: Type.Optional(Type.String()),
824
+ head: Type.Optional(Type.String()),
825
+ maxBytes: Type.Optional(Type.Number({ minimum: 1, maximum: HARD_GIT_DIFF_MAX_BYTES })),
826
+ }),
827
+ execute: async (_toolCallId, params) => {
828
+ const file = normalizeGitDiffPath(params.file);
829
+ const base = normalizeGitRev(params.base, 'base');
830
+ const head = normalizeGitRev(params.head, 'head');
831
+ const maxBytes = normalizeGitDiffMaxBytes(params.maxBytes);
832
+ const range = base && head ? [`${base}...${head}`] : base ? [base] : [];
833
+ let diff = '';
834
+ let truncated = false;
835
+ let error;
836
+ try {
837
+ const result = await runGitDiff(workingDir, [...range, '--', file], maxBytes);
838
+ diff = result.stdout;
839
+ truncated = result.truncated;
840
+ }
841
+ catch (caught) {
842
+ error = caught instanceof Error ? caught.message : String(caught);
843
+ }
844
+ let metadata = extractGitDiffMetadata(diff);
845
+ if (error === undefined && range.length > 0) {
846
+ try {
847
+ const nameStatus = await runGitNameStatus(workingDir, range);
848
+ metadata = mergeGitNameStatusMetadata(metadata, nameStatus, file);
849
+ }
850
+ catch {
851
+ // Diff text remains authoritative; name-status only enriches metadata.
852
+ }
853
+ }
854
+ const details = {
855
+ file,
856
+ base,
857
+ head,
858
+ ok: error === undefined,
859
+ error,
860
+ truncated,
861
+ bytes: Buffer.byteLength(diff, 'utf8'),
862
+ metadata,
863
+ diff,
864
+ };
865
+ return {
866
+ content: [{ type: 'text', text: JSON.stringify(details) }],
867
+ details,
868
+ };
869
+ },
870
+ });
871
+ }
872
+ if (this.isToolEnabled('read_artifact', false, agentTools)) {
873
+ customTools.push({
874
+ name: 'read_artifact',
875
+ label: 'read_artifact',
876
+ description: 'Read a DRS review artifact from a file path. ' +
877
+ 'Without findingId: returns a compact manifest of all findings (id, severity, state, disposition, file, line, title). ' +
878
+ 'With findingId: returns the full finding detail including issue, verification rationale, and fingerprint.',
879
+ parameters: Type.Object({
880
+ artifactPath: Type.String({ minLength: 1 }),
881
+ findingId: Type.Optional(Type.String({ minLength: 1 })),
882
+ }),
883
+ execute: async (_toolCallId, params) => {
884
+ const { readFile } = await import('fs/promises');
885
+ try {
886
+ const fullPath = resolveWithinWorkingDir(workingDir, params.artifactPath, 'read');
887
+ const raw = JSON.parse(await readFile(fullPath, 'utf-8'));
888
+ if (findingIdParamProvided(params)) {
889
+ const finding = extractFindingFromArtifact(raw, params.findingId);
890
+ if (!finding) {
891
+ const details = {
892
+ artifactPath: params.artifactPath,
893
+ findingId: params.findingId,
894
+ ok: false,
895
+ error: `Finding "${params.findingId}" not found in artifact`,
896
+ };
897
+ return {
898
+ content: [{ type: 'text', text: JSON.stringify(details) }],
899
+ details,
900
+ };
901
+ }
902
+ const details = {
903
+ artifactPath: params.artifactPath,
904
+ findingId: params.findingId,
905
+ ok: true,
906
+ finding,
907
+ };
908
+ return {
909
+ content: [{ type: 'text', text: JSON.stringify(details) }],
910
+ details,
911
+ };
912
+ }
913
+ const manifest = buildArtifactManifest(raw, params.artifactPath);
914
+ return {
915
+ content: [{ type: 'text', text: JSON.stringify(manifest) }],
916
+ details: manifest,
917
+ };
918
+ }
919
+ catch (caught) {
920
+ const error = formatUnknownError(caught);
921
+ const details = {
922
+ artifactPath: params.artifactPath,
923
+ findingId: params.findingId,
924
+ ok: false,
925
+ error,
926
+ };
927
+ return {
928
+ content: [{ type: 'text', text: JSON.stringify(details) }],
929
+ details,
930
+ };
931
+ }
932
+ },
933
+ });
934
+ }
935
+ if (this.isToolEnabled('drs_check', false, agentTools) && this.runtimeConfig.fixChecks) {
936
+ const configuredChecks = this.runtimeConfig.fixChecks;
937
+ customTools.push({
938
+ name: 'drs_check',
939
+ label: 'drs_check',
940
+ description: 'Run configured DRS fix checks (type-check, lint, tests, etc.). ' +
941
+ 'Without name: runs all applicable checks (filtered by matchPaths against changed files). ' +
942
+ 'With name: runs a single named check. ' +
943
+ 'Returns exit code, stdout, and stderr for each check.',
944
+ parameters: Type.Object({
945
+ name: Type.Optional(Type.String({ minLength: 1 })),
946
+ }),
947
+ execute: async (_toolCallId, params) => {
948
+ const changedFiles = await getChangedFiles(workingDir);
949
+ let checksToRun;
950
+ if (params.name) {
951
+ const match = configuredChecks.find((c) => c.name === params.name);
952
+ if (!match) {
953
+ const details = {
954
+ ok: false,
955
+ error: `No check named "${params.name}" is configured`,
956
+ availableChecks: configuredChecks.map((c) => c.name),
957
+ };
958
+ return {
959
+ content: [{ type: 'text', text: JSON.stringify(details) }],
960
+ details,
961
+ };
962
+ }
963
+ checksToRun = [match];
964
+ }
965
+ else {
966
+ checksToRun = configuredChecks.filter((check) => !check.matchPaths ||
967
+ check.matchPaths.length === 0 ||
968
+ changedFiles.some((f) => fileMatchesGlobs(f, check.matchPaths)));
969
+ }
970
+ if (checksToRun.length === 0) {
971
+ const details = {
972
+ ok: true,
973
+ checks: [],
974
+ changedFiles,
975
+ skipped: 'No applicable checks for changed files',
976
+ };
977
+ return {
978
+ content: [{ type: 'text', text: JSON.stringify(details) }],
979
+ details,
980
+ };
981
+ }
982
+ const results = [];
983
+ for (const check of checksToRun) {
984
+ results.push(await runCheckCommand(workingDir, check));
985
+ }
986
+ const details = {
987
+ ok: results.every((r) => r.exitCode === 0),
988
+ checks: results,
989
+ changedFiles,
990
+ };
991
+ return {
992
+ content: [{ type: 'text', text: JSON.stringify(details) }],
993
+ details,
994
+ };
995
+ },
996
+ });
997
+ }
447
998
  return customTools;
448
999
  }
449
1000
  async createAgentSession(cwd, agentName) {
@@ -487,6 +1038,11 @@ class PiSessionRuntime {
487
1038
  },
488
1039
  })
489
1040
  : undefined;
1041
+ const customTools = this.resolveCustomTools(cwd, settings.tools);
1042
+ const tools = [
1043
+ ...this.resolveTools(cwd, settings.tools),
1044
+ ...customTools.map((tool) => tool.name),
1045
+ ];
490
1046
  const { session } = await createAgentSession({
491
1047
  cwd,
492
1048
  authStorage: this.authStorage,
@@ -495,8 +1051,8 @@ class PiSessionRuntime {
495
1051
  resourceLoader,
496
1052
  sessionManager: SessionManager.inMemory(),
497
1053
  settingsManager,
498
- tools: this.resolveTools(cwd, settings.tools),
499
- customTools: this.resolveCustomTools(cwd),
1054
+ tools,
1055
+ customTools,
500
1056
  thinkingLevel: this.runtimeConfig.thinkingLevel,
501
1057
  });
502
1058
  return session;
@@ -534,6 +1090,9 @@ class PiSessionRuntime {
534
1090
  record.session = await this.createAgentSession(cwd, input.body.agent);
535
1091
  record.agent = input.body.agent;
536
1092
  record.cwd = cwd;
1093
+ if (this.runtimeConfig.traceCollector) {
1094
+ this.runtimeConfig.traceCollector.attachSession(record.session, record.id);
1095
+ }
537
1096
  }
538
1097
  record.error = undefined;
539
1098
  await record.session.prompt(promptText);
@@ -543,6 +1102,11 @@ class PiSessionRuntime {
543
1102
  record.error = error;
544
1103
  return { ok: false };
545
1104
  }
1105
+ finally {
1106
+ if (this.runtimeConfig.traceCollector && record.session) {
1107
+ this.runtimeConfig.traceCollector.finalizeSession(record.id);
1108
+ }
1109
+ }
546
1110
  }
547
1111
  readMessages(input) {
548
1112
  const record = this.getSessionRecord(input.path.id);
@@ -563,7 +1127,7 @@ class PiSessionRuntime {
563
1127
  id: `${record.id}-error`,
564
1128
  role: 'assistant',
565
1129
  time: { completed: Date.now() },
566
- error: record.error instanceof Error ? record.error.message : String(record.error),
1130
+ error: formatUnknownError(record.error),
567
1131
  },
568
1132
  parts: [{ text: '' }],
569
1133
  });