@prsense/workflows 0.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 (169) hide show
  1. package/LICENSE +201 -0
  2. package/dist/contract/ResolvedConfig.d.ts +37 -0
  3. package/dist/contract/ResolvedConfig.d.ts.map +1 -0
  4. package/dist/contract/ResolvedConfig.js +3 -0
  5. package/dist/contract/ResolvedConfig.js.map +1 -0
  6. package/dist/doctor/adaptCapability.d.ts +11 -0
  7. package/dist/doctor/adaptCapability.d.ts.map +1 -0
  8. package/dist/doctor/adaptCapability.js +35 -0
  9. package/dist/doctor/adaptCapability.js.map +1 -0
  10. package/dist/doctor/buildCapabilityContext.d.ts +3 -0
  11. package/dist/doctor/buildCapabilityContext.d.ts.map +1 -0
  12. package/dist/doctor/buildCapabilityContext.js +15 -0
  13. package/dist/doctor/buildCapabilityContext.js.map +1 -0
  14. package/dist/doctor/checks/checkEnvConfig.d.ts +3 -0
  15. package/dist/doctor/checks/checkEnvConfig.d.ts.map +1 -0
  16. package/dist/doctor/checks/checkEnvConfig.js +18 -0
  17. package/dist/doctor/checks/checkEnvConfig.js.map +1 -0
  18. package/dist/doctor/checks/checkLLM.d.ts +3 -0
  19. package/dist/doctor/checks/checkLLM.d.ts.map +1 -0
  20. package/dist/doctor/checks/checkLLM.js +46 -0
  21. package/dist/doctor/checks/checkLLM.js.map +1 -0
  22. package/dist/doctor/checks/checkRepository.d.ts +3 -0
  23. package/dist/doctor/checks/checkRepository.d.ts.map +1 -0
  24. package/dist/doctor/checks/checkRepository.js +18 -0
  25. package/dist/doctor/checks/checkRepository.js.map +1 -0
  26. package/dist/doctor/checks/checkUserConfig.d.ts +3 -0
  27. package/dist/doctor/checks/checkUserConfig.d.ts.map +1 -0
  28. package/dist/doctor/checks/checkUserConfig.js +16 -0
  29. package/dist/doctor/checks/checkUserConfig.js.map +1 -0
  30. package/dist/doctor/doctorWorkflow.d.ts +3 -0
  31. package/dist/doctor/doctorWorkflow.d.ts.map +1 -0
  32. package/dist/doctor/doctorWorkflow.js +26 -0
  33. package/dist/doctor/doctorWorkflow.js.map +1 -0
  34. package/dist/doctor/types.d.ts +26 -0
  35. package/dist/doctor/types.d.ts.map +1 -0
  36. package/dist/doctor/types.js +2 -0
  37. package/dist/doctor/types.js.map +1 -0
  38. package/dist/doctor/workflow.d.ts +8 -0
  39. package/dist/doctor/workflow.d.ts.map +1 -0
  40. package/dist/doctor/workflow.js +66 -0
  41. package/dist/doctor/workflow.js.map +1 -0
  42. package/dist/index/debug.d.ts +16 -0
  43. package/dist/index/debug.d.ts.map +1 -0
  44. package/dist/index/debug.js +2 -0
  45. package/dist/index/debug.js.map +1 -0
  46. package/dist/index/deps.d.ts +8 -0
  47. package/dist/index/deps.d.ts.map +1 -0
  48. package/dist/index/deps.js +2 -0
  49. package/dist/index/deps.js.map +1 -0
  50. package/dist/index/events.d.ts +15 -0
  51. package/dist/index/events.d.ts.map +1 -0
  52. package/dist/index/events.js +2 -0
  53. package/dist/index/events.js.map +1 -0
  54. package/dist/index/helper.d.ts +2 -0
  55. package/dist/index/helper.d.ts.map +1 -0
  56. package/dist/index/helper.js +15 -0
  57. package/dist/index/helper.js.map +1 -0
  58. package/dist/index/index.d.ts +4 -0
  59. package/dist/index/index.d.ts.map +1 -0
  60. package/dist/index/index.js +4 -0
  61. package/dist/index/index.js.map +1 -0
  62. package/dist/index/indexWorkflow.d.ts +13 -0
  63. package/dist/index/indexWorkflow.d.ts.map +1 -0
  64. package/dist/index/indexWorkflow.js +301 -0
  65. package/dist/index/indexWorkflow.js.map +1 -0
  66. package/dist/index/listIndexedRepositories.d.ts +4 -0
  67. package/dist/index/listIndexedRepositories.d.ts.map +1 -0
  68. package/dist/index/listIndexedRepositories.js +6 -0
  69. package/dist/index/listIndexedRepositories.js.map +1 -0
  70. package/dist/index/ports.d.ts +6 -0
  71. package/dist/index/ports.d.ts.map +1 -0
  72. package/dist/index/ports.js +3 -0
  73. package/dist/index/ports.js.map +1 -0
  74. package/dist/index/types.d.ts +8 -0
  75. package/dist/index/types.d.ts.map +1 -0
  76. package/dist/index/types.js +3 -0
  77. package/dist/index/types.js.map +1 -0
  78. package/dist/index.d.ts +9 -0
  79. package/dist/index.d.ts.map +1 -0
  80. package/dist/index.js +9 -0
  81. package/dist/index.js.map +1 -0
  82. package/dist/review/buildDiffEmbeddingQuery.d.ts +8 -0
  83. package/dist/review/buildDiffEmbeddingQuery.d.ts.map +1 -0
  84. package/dist/review/buildDiffEmbeddingQuery.js +43 -0
  85. package/dist/review/buildDiffEmbeddingQuery.js.map +1 -0
  86. package/dist/review/dedupeSignals.d.ts +3 -0
  87. package/dist/review/dedupeSignals.d.ts.map +1 -0
  88. package/dist/review/dedupeSignals.js +11 -0
  89. package/dist/review/dedupeSignals.js.map +1 -0
  90. package/dist/review/extractJson.d.ts +2 -0
  91. package/dist/review/extractJson.d.ts.map +1 -0
  92. package/dist/review/extractJson.js +15 -0
  93. package/dist/review/extractJson.js.map +1 -0
  94. package/dist/review/index.d.ts +4 -0
  95. package/dist/review/index.d.ts.map +1 -0
  96. package/dist/review/index.js +4 -0
  97. package/dist/review/index.js.map +1 -0
  98. package/dist/review/input/ReviewInput.d.ts +18 -0
  99. package/dist/review/input/ReviewInput.d.ts.map +1 -0
  100. package/dist/review/input/ReviewInput.js +3 -0
  101. package/dist/review/input/ReviewInput.js.map +1 -0
  102. package/dist/review/normalizeSignal.d.ts +3 -0
  103. package/dist/review/normalizeSignal.d.ts.map +1 -0
  104. package/dist/review/normalizeSignal.js +31 -0
  105. package/dist/review/normalizeSignal.js.map +1 -0
  106. package/dist/review/ports.d.ts +5 -0
  107. package/dist/review/ports.d.ts.map +1 -0
  108. package/dist/review/ports.js +3 -0
  109. package/dist/review/ports.js.map +1 -0
  110. package/dist/review/retrieveContext.d.ts +12 -0
  111. package/dist/review/retrieveContext.d.ts.map +1 -0
  112. package/dist/review/retrieveContext.js +76 -0
  113. package/dist/review/retrieveContext.js.map +1 -0
  114. package/dist/review/reviewWorkflow.d.ts +11 -0
  115. package/dist/review/reviewWorkflow.d.ts.map +1 -0
  116. package/dist/review/reviewWorkflow.js +286 -0
  117. package/dist/review/reviewWorkflow.js.map +1 -0
  118. package/dist/review/types.d.ts +14 -0
  119. package/dist/review/types.d.ts.map +1 -0
  120. package/dist/review/types.js +3 -0
  121. package/dist/review/types.js.map +1 -0
  122. package/dist/review/validateReviewOutput.d.ts +4 -0
  123. package/dist/review/validateReviewOutput.d.ts.map +1 -0
  124. package/dist/review/validateReviewOutput.js +10 -0
  125. package/dist/review/validateReviewOutput.js.map +1 -0
  126. package/dist/setup/setupWorkflow.d.ts +9 -0
  127. package/dist/setup/setupWorkflow.d.ts.map +1 -0
  128. package/dist/setup/setupWorkflow.js +81 -0
  129. package/dist/setup/setupWorkflow.js.map +1 -0
  130. package/dist/setup/types.d.ts +16 -0
  131. package/dist/setup/types.d.ts.map +1 -0
  132. package/dist/setup/types.js +2 -0
  133. package/dist/setup/types.js.map +1 -0
  134. package/dist/types/checks.d.ts +17 -0
  135. package/dist/types/checks.d.ts.map +1 -0
  136. package/dist/types/checks.js +2 -0
  137. package/dist/types/checks.js.map +1 -0
  138. package/dist/types/core.d.ts +17 -0
  139. package/dist/types/core.d.ts.map +1 -0
  140. package/dist/types/core.js +2 -0
  141. package/dist/types/core.js.map +1 -0
  142. package/dist/types/deps.d.ts +9 -0
  143. package/dist/types/deps.d.ts.map +1 -0
  144. package/dist/types/deps.js +2 -0
  145. package/dist/types/deps.js.map +1 -0
  146. package/package.json +30 -0
  147. package/src/doctor/workflow.ts +99 -0
  148. package/src/index/index.ts +3 -0
  149. package/src/index/indexWorkflow.ts +411 -0
  150. package/src/index/listIndexedRepositories.ts +11 -0
  151. package/src/index/ports.ts +8 -0
  152. package/src/index/types.ts +11 -0
  153. package/src/index.ts +13 -0
  154. package/src/review/buildDiffEmbeddingQuery.ts +66 -0
  155. package/src/review/dedupeSignals.ts +10 -0
  156. package/src/review/extractJson.ts +17 -0
  157. package/src/review/index.ts +3 -0
  158. package/src/review/input/ReviewInput.ts +22 -0
  159. package/src/review/normalizeSignal.ts +34 -0
  160. package/src/review/ports.ts +7 -0
  161. package/src/review/retrieveContext.ts +105 -0
  162. package/src/review/reviewWorkflow.ts +366 -0
  163. package/src/review/types.ts +18 -0
  164. package/src/review/validateReviewOutput.ts +13 -0
  165. package/src/setup/setupWorkflow.ts +110 -0
  166. package/src/setup/types.ts +9 -0
  167. package/src/types/core.ts +18 -0
  168. package/tsconfig.json +11 -0
  169. package/tsconfig.tsbuildinfo +1 -0
@@ -0,0 +1 @@
1
+ {"version":3,"file":"buildDiffEmbeddingQuery.d.ts","sourceRoot":"","sources":["../../src/review/buildDiffEmbeddingQuery.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,eAAe,CAAC;AAEjD,wBAAgB,uBAAuB,CAAC,MAAM,EAAE;IAC9C,IAAI,EAAE,WAAW,CAAC;IAClB,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,QAAQ,CAAC,EAAE,MAAM,CAAC;CACnB,GAAG,MAAM,CAyDT"}
@@ -0,0 +1,43 @@
1
+ export function buildDiffEmbeddingQuery(params) {
2
+ const { diff, title, description, maxChars = 4000 } = params;
3
+ const parts = [];
4
+ // -------------------------------------------------
5
+ // PR Intent
6
+ // -------------------------------------------------
7
+ if (title) {
8
+ parts.push(`PR Title: ${title}`);
9
+ }
10
+ if (description) {
11
+ parts.push(`PR Description: ${description}`);
12
+ }
13
+ // -------------------------------------------------
14
+ // Changed Files (strong retrieval anchor)
15
+ // -------------------------------------------------
16
+ parts.push("Changed Files:");
17
+ for (const file of diff.files.slice(0, 50)) {
18
+ parts.push(file.path);
19
+ }
20
+ // -------------------------------------------------
21
+ // Meaningful Code Changes
22
+ // -------------------------------------------------
23
+ parts.push("\nChanged Code:");
24
+ for (const file of diff.files) {
25
+ parts.push(`File: ${file.path}`);
26
+ const lines = file.patch.split("\n");
27
+ for (const line of lines) {
28
+ // Only meaningful changes
29
+ if (line.startsWith("+") && !line.startsWith("+++")) {
30
+ parts.push(line.slice(1));
31
+ }
32
+ if (line.startsWith("-") && !line.startsWith("---")) {
33
+ parts.push(line.slice(1));
34
+ }
35
+ }
36
+ }
37
+ let query = parts.join("\n");
38
+ if (query.length > maxChars) {
39
+ query = query.slice(0, maxChars);
40
+ }
41
+ return query;
42
+ }
43
+ //# sourceMappingURL=buildDiffEmbeddingQuery.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"buildDiffEmbeddingQuery.js","sourceRoot":"","sources":["../../src/review/buildDiffEmbeddingQuery.ts"],"names":[],"mappings":"AAGA,MAAM,UAAU,uBAAuB,CAAC,MAKvC;IACC,MAAM,EAAE,IAAI,EAAE,KAAK,EAAE,WAAW,EAAE,QAAQ,GAAG,IAAI,EAAE,GAAG,MAAM,CAAC;IAE7D,MAAM,KAAK,GAAa,EAAE,CAAC;IAE3B,oDAAoD;IACpD,YAAY;IACZ,oDAAoD;IAEpD,IAAI,KAAK,EAAE,CAAC;QACV,KAAK,CAAC,IAAI,CAAC,aAAa,KAAK,EAAE,CAAC,CAAC;IACnC,CAAC;IAED,IAAI,WAAW,EAAE,CAAC;QAChB,KAAK,CAAC,IAAI,CAAC,mBAAmB,WAAW,EAAE,CAAC,CAAC;IAC/C,CAAC;IAED,oDAAoD;IACpD,0CAA0C;IAC1C,oDAAoD;IAEpD,KAAK,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC;IAE7B,KAAK,MAAM,IAAI,IAAI,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,EAAE,CAAC;QAC3C,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACxB,CAAC;IAED,oDAAoD;IACpD,0BAA0B;IAC1B,oDAAoD;IAEpD,KAAK,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;IAE9B,KAAK,MAAM,IAAI,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC;QAC9B,KAAK,CAAC,IAAI,CAAC,SAAS,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC;QAEjC,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QAErC,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;YACzB,0BAA0B;YAC1B,IAAI,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,EAAE,CAAC;gBACpD,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC;YAC5B,CAAC;YAED,IAAI,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,EAAE,CAAC;gBACpD,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC;YAC5B,CAAC;QACH,CAAC;IACH,CAAC;IAED,IAAI,KAAK,GAAG,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAE7B,IAAI,KAAK,CAAC,MAAM,GAAG,QAAQ,EAAE,CAAC;QAC5B,KAAK,GAAG,KAAK,CAAC,KAAK,CAAC,CAAC,EAAE,QAAQ,CAAC,CAAC;IACnC,CAAC;IAED,OAAO,KAAK,CAAC;AACf,CAAC"}
@@ -0,0 +1,3 @@
1
+ import { ReviewSignal } from "@prsense/core";
2
+ export declare function dedupeSignals(signals: ReviewSignal[]): ReviewSignal[];
3
+ //# sourceMappingURL=dedupeSignals.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"dedupeSignals.d.ts","sourceRoot":"","sources":["../../src/review/dedupeSignals.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,eAAe,CAAC;AAC7C,wBAAgB,aAAa,CAAC,OAAO,EAAE,YAAY,EAAE,GAAG,YAAY,EAAE,CAQrE"}
@@ -0,0 +1,11 @@
1
+ export function dedupeSignals(signals) {
2
+ const seen = new Set();
3
+ return signals.filter((s) => {
4
+ const key = `${s.file}:${s.lineStart}:${s.message}`;
5
+ if (seen.has(key))
6
+ return false;
7
+ seen.add(key);
8
+ return true;
9
+ });
10
+ }
11
+ //# sourceMappingURL=dedupeSignals.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"dedupeSignals.js","sourceRoot":"","sources":["../../src/review/dedupeSignals.ts"],"names":[],"mappings":"AACA,MAAM,UAAU,aAAa,CAAC,OAAuB;IACnD,MAAM,IAAI,GAAG,IAAI,GAAG,EAAU,CAAC;IAC/B,OAAO,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE;QAC1B,MAAM,GAAG,GAAG,GAAG,CAAC,CAAC,IAAI,IAAI,CAAC,CAAC,SAAS,IAAI,CAAC,CAAC,OAAO,EAAE,CAAC;QACpD,IAAI,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC;YAAE,OAAO,KAAK,CAAC;QAChC,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;QACd,OAAO,IAAI,CAAC;IACd,CAAC,CAAC,CAAC;AACL,CAAC"}
@@ -0,0 +1,2 @@
1
+ export declare function extractJson(text: string): string;
2
+ //# sourceMappingURL=extractJson.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"extractJson.d.ts","sourceRoot":"","sources":["../../src/review/extractJson.ts"],"names":[],"mappings":"AAAA,wBAAgB,WAAW,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM,CAgBhD"}
@@ -0,0 +1,15 @@
1
+ export function extractJson(text) {
2
+ // Remove markdown fences if present
3
+ const fenced = text.match(/```(?:json)?\s*([\s\S]*?)```/i);
4
+ if (fenced && fenced[1]) {
5
+ return fenced[1].trim();
6
+ }
7
+ // Otherwise attempt brace extraction
8
+ const first = text.indexOf("{");
9
+ const last = text.lastIndexOf("}");
10
+ if (first !== -1 && last !== -1) {
11
+ return text.slice(first, last + 1);
12
+ }
13
+ return text;
14
+ }
15
+ //# sourceMappingURL=extractJson.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"extractJson.js","sourceRoot":"","sources":["../../src/review/extractJson.ts"],"names":[],"mappings":"AAAA,MAAM,UAAU,WAAW,CAAC,IAAY;IACtC,oCAAoC;IACpC,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,+BAA+B,CAAC,CAAC;IAC3D,IAAI,MAAM,IAAI,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC;QACxB,OAAO,MAAM,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC;IAC1B,CAAC;IAED,qCAAqC;IACrC,MAAM,KAAK,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;IAChC,MAAM,IAAI,GAAG,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,CAAC;IAEnC,IAAI,KAAK,KAAK,CAAC,CAAC,IAAI,IAAI,KAAK,CAAC,CAAC,EAAE,CAAC;QAChC,OAAO,IAAI,CAAC,KAAK,CAAC,KAAK,EAAE,IAAI,GAAG,CAAC,CAAC,CAAC;IACrC,CAAC;IAED,OAAO,IAAI,CAAC;AACd,CAAC"}
@@ -0,0 +1,4 @@
1
+ export * from "./retrieveContext.js";
2
+ export * from "./reviewWorkflow.js";
3
+ export * from "./validateReviewOutput.js";
4
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/review/index.ts"],"names":[],"mappings":"AAAA,cAAc,sBAAsB,CAAC;AACrC,cAAc,qBAAqB,CAAC;AACpC,cAAc,2BAA2B,CAAC"}
@@ -0,0 +1,4 @@
1
+ export * from "./retrieveContext.js";
2
+ export * from "./reviewWorkflow.js";
3
+ export * from "./validateReviewOutput.js";
4
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/review/index.ts"],"names":[],"mappings":"AAAA,cAAc,sBAAsB,CAAC;AACrC,cAAc,qBAAqB,CAAC;AACpC,cAAc,2BAA2B,CAAC"}
@@ -0,0 +1,18 @@
1
+ export type ReviewInput = {
2
+ repo: {
3
+ owner: string;
4
+ name: string;
5
+ };
6
+ /** Raw unified diff text */
7
+ diffText: string;
8
+ /** Optional hints from adapter */
9
+ repoRoot?: string;
10
+ baseBranch?: string;
11
+ /** Human-authored metadata */
12
+ metadata?: {
13
+ title?: string;
14
+ description?: string;
15
+ author?: string;
16
+ };
17
+ };
18
+ //# sourceMappingURL=ReviewInput.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ReviewInput.d.ts","sourceRoot":"","sources":["../../../src/review/input/ReviewInput.ts"],"names":[],"mappings":"AAEA,MAAM,MAAM,WAAW,GAAG;IACxB,IAAI,EAAE;QACJ,KAAK,EAAE,MAAM,CAAC;QACd,IAAI,EAAE,MAAM,CAAC;KACd,CAAC;IAEF,4BAA4B;IAC5B,QAAQ,EAAE,MAAM,CAAC;IAEjB,kCAAkC;IAClC,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,UAAU,CAAC,EAAE,MAAM,CAAC;IAEpB,8BAA8B;IAC9B,QAAQ,CAAC,EAAE;QACT,KAAK,CAAC,EAAE,MAAM,CAAC;QACf,WAAW,CAAC,EAAE,MAAM,CAAC;QACrB,MAAM,CAAC,EAAE,MAAM,CAAC;KACjB,CAAC;CACH,CAAC"}
@@ -0,0 +1,3 @@
1
+ // packages/core/input/ReviewInput.ts
2
+ export {};
3
+ //# sourceMappingURL=ReviewInput.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ReviewInput.js","sourceRoot":"","sources":["../../../src/review/input/ReviewInput.ts"],"names":[],"mappings":"AAAA,qCAAqC"}
@@ -0,0 +1,3 @@
1
+ import type { ReviewSignal } from "@prsense/core";
2
+ export declare function normalizeSignal(raw: any, index: number): ReviewSignal | null;
3
+ //# sourceMappingURL=normalizeSignal.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"normalizeSignal.d.ts","sourceRoot":"","sources":["../../src/review/normalizeSignal.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,eAAe,CAAC;AAKlD,wBAAgB,eAAe,CAAC,GAAG,EAAE,GAAG,EAAE,KAAK,EAAE,MAAM,GAAG,YAAY,GAAG,IAAI,CA4B5E"}
@@ -0,0 +1,31 @@
1
+ const allowedTypes = ["bug", "risk", "test", "style"];
2
+ const allowedSeverity = ["low", "medium", "high"];
3
+ export function normalizeSignal(raw, index) {
4
+ if (!raw || typeof raw !== "object")
5
+ return null;
6
+ if (!allowedTypes.includes(raw.type))
7
+ return null;
8
+ if (!allowedSeverity.includes(raw.severity))
9
+ return null;
10
+ const confidence = typeof raw.confidence === "number"
11
+ ? Math.max(0, Math.min(1, raw.confidence))
12
+ : 0.5;
13
+ if (!raw.file || typeof raw.file !== "string")
14
+ return null;
15
+ if (!raw.message || typeof raw.message !== "string")
16
+ return null;
17
+ return {
18
+ id: `signal-${index}`,
19
+ type: raw.type,
20
+ severity: raw.severity,
21
+ confidence,
22
+ file: raw.file,
23
+ lineStart: typeof raw.lineStart === "number" ? raw.lineStart : undefined,
24
+ lineEnd: typeof raw.lineEnd === "number" ? raw.lineEnd : undefined,
25
+ message: raw.message,
26
+ rationale: typeof raw.rationale === "string" ? raw.rationale : undefined,
27
+ suggestedFix: typeof raw.suggestedFix === "string" ? raw.suggestedFix : undefined,
28
+ source: "llm",
29
+ };
30
+ }
31
+ //# sourceMappingURL=normalizeSignal.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"normalizeSignal.js","sourceRoot":"","sources":["../../src/review/normalizeSignal.ts"],"names":[],"mappings":"AAEA,MAAM,YAAY,GAAG,CAAC,KAAK,EAAE,MAAM,EAAE,MAAM,EAAE,OAAO,CAAU,CAAC;AAC/D,MAAM,eAAe,GAAG,CAAC,KAAK,EAAE,QAAQ,EAAE,MAAM,CAAU,CAAC;AAE3D,MAAM,UAAU,eAAe,CAAC,GAAQ,EAAE,KAAa;IACrD,IAAI,CAAC,GAAG,IAAI,OAAO,GAAG,KAAK,QAAQ;QAAE,OAAO,IAAI,CAAC;IAEjD,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,GAAG,CAAC,IAAI,CAAC;QAAE,OAAO,IAAI,CAAC;IAClD,IAAI,CAAC,eAAe,CAAC,QAAQ,CAAC,GAAG,CAAC,QAAQ,CAAC;QAAE,OAAO,IAAI,CAAC;IAEzD,MAAM,UAAU,GACd,OAAO,GAAG,CAAC,UAAU,KAAK,QAAQ;QAChC,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,GAAG,CAAC,UAAU,CAAC,CAAC;QAC1C,CAAC,CAAC,GAAG,CAAC;IAEV,IAAI,CAAC,GAAG,CAAC,IAAI,IAAI,OAAO,GAAG,CAAC,IAAI,KAAK,QAAQ;QAAE,OAAO,IAAI,CAAC;IAC3D,IAAI,CAAC,GAAG,CAAC,OAAO,IAAI,OAAO,GAAG,CAAC,OAAO,KAAK,QAAQ;QAAE,OAAO,IAAI,CAAC;IAEjE,OAAO;QACL,EAAE,EAAE,UAAU,KAAK,EAAE;QACrB,IAAI,EAAE,GAAG,CAAC,IAAI;QACd,QAAQ,EAAE,GAAG,CAAC,QAAQ;QACtB,UAAU;QACV,IAAI,EAAE,GAAG,CAAC,IAAI;QACd,SAAS,EAAE,OAAO,GAAG,CAAC,SAAS,KAAK,QAAQ,CAAC,CAAC,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC,CAAC,SAAS;QACxE,OAAO,EAAE,OAAO,GAAG,CAAC,OAAO,KAAK,QAAQ,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,SAAS;QAClE,OAAO,EAAE,GAAG,CAAC,OAAO;QACpB,SAAS,EAAE,OAAO,GAAG,CAAC,SAAS,KAAK,QAAQ,CAAC,CAAC,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC,CAAC,SAAS;QACxE,YAAY,EACV,OAAO,GAAG,CAAC,YAAY,KAAK,QAAQ,CAAC,CAAC,CAAC,GAAG,CAAC,YAAY,CAAC,CAAC,CAAC,SAAS;QACrE,MAAM,EAAE,KAAK;KACd,CAAC;AACJ,CAAC"}
@@ -0,0 +1,5 @@
1
+ import type { ContextChunk, UnifiedDiff, ReviewSignal } from "@prsense/core";
2
+ export interface ReviewSignalCompiler {
3
+ compile(diff: UnifiedDiff, context: ContextChunk[]): Promise<ReviewSignal[]>;
4
+ }
5
+ //# sourceMappingURL=ports.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ports.d.ts","sourceRoot":"","sources":["../../src/review/ports.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,YAAY,EAAE,WAAW,EAAE,YAAY,EAAE,MAAM,eAAe,CAAC;AAE7E,MAAM,WAAW,oBAAoB;IACnC,OAAO,CAAC,IAAI,EAAE,WAAW,EAAE,OAAO,EAAE,YAAY,EAAE,GAAG,OAAO,CAAC,YAAY,EAAE,CAAC,CAAC;CAC9E"}
@@ -0,0 +1,3 @@
1
+ // packages/workflows/src/review/ports.ts
2
+ export {};
3
+ //# sourceMappingURL=ports.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ports.js","sourceRoot":"","sources":["../../src/review/ports.ts"],"names":[],"mappings":"AAAA,yCAAyC"}
@@ -0,0 +1,12 @@
1
+ import type { RetrievedContext, EventBus } from "@prsense/core";
2
+ import type { ResolvedConfig } from "@prsense/config";
3
+ export declare function retrieveContext(params: {
4
+ config: ResolvedConfig;
5
+ query: string;
6
+ repoProvider: string;
7
+ repoName: string;
8
+ repoRef?: string;
9
+ limit: number;
10
+ eventBus?: EventBus;
11
+ }): Promise<RetrievedContext>;
12
+ //# sourceMappingURL=retrieveContext.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"retrieveContext.d.ts","sourceRoot":"","sources":["../../src/review/retrieveContext.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,gBAAgB,EAAE,QAAQ,EAAgB,MAAM,eAAe,CAAC;AAE9E,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,iBAAiB,CAAC;AAOtD,wBAAsB,eAAe,CAAC,MAAM,EAAE;IAC5C,MAAM,EAAE,cAAc,CAAC;IACvB,KAAK,EAAE,MAAM,CAAC;IACd,YAAY,EAAE,MAAM,CAAC;IACrB,QAAQ,EAAE,MAAM,CAAC;IACjB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,KAAK,EAAE,MAAM,CAAC;IACd,QAAQ,CAAC,EAAE,QAAQ,CAAC;CACrB,GAAG,OAAO,CAAC,gBAAgB,CAAC,CAqF5B"}
@@ -0,0 +1,76 @@
1
+ // packages/workflows/src/review/retrieveContext.ts
2
+ import { CoreEvents } from "@prsense/core";
3
+ import { PostgresRagChunkRepository } from "@prsense/context";
4
+ import { createOpenAiEmbeddingClient, createOllamaEmbeddingClient, } from "@prsense/llm";
5
+ export async function retrieveContext(params) {
6
+ const { config, query, repoProvider, repoName, repoRef, limit } = params;
7
+ // -------------------------------------------------
8
+ // Create embedding client
9
+ // -------------------------------------------------
10
+ const embeddingClient = config.embeddings.provider === "openai"
11
+ ? createOpenAiEmbeddingClient({
12
+ apiKey: process.env.OPENAI_API_KEY,
13
+ model: config.embeddings.model,
14
+ })
15
+ : createOllamaEmbeddingClient({
16
+ model: config.embeddings.model,
17
+ });
18
+ const [queryEmbedding] = await embeddingClient.embed([query]);
19
+ params.eventBus?.emit(CoreEvents.WorkflowReviewContextEmbeddingGenerated, {
20
+ dimension: queryEmbedding?.length,
21
+ });
22
+ if (!queryEmbedding) {
23
+ throw new Error("Failed to generate query embedding");
24
+ }
25
+ // -------------------------------------------------
26
+ // Query RAG store
27
+ // -------------------------------------------------
28
+ const repository = new PostgresRagChunkRepository(config.database.url);
29
+ const rows = await repository.searchNearest({
30
+ repoProvider,
31
+ repoName,
32
+ ...(repoRef ? { repoRef } : {}),
33
+ embedding: queryEmbedding,
34
+ limit,
35
+ });
36
+ params.eventBus?.emit(CoreEvents.WorkflowReviewContextRetrieved, {
37
+ chunks: rows.length,
38
+ repoProvider,
39
+ repoName,
40
+ minDistance: rows[0]?.distance,
41
+ maxDistance: rows[rows.length - 1]?.distance,
42
+ });
43
+ // -------------------------------------------------
44
+ // Map to domain objects
45
+ // -------------------------------------------------
46
+ const chunks = rows.map((row) => {
47
+ const chunk = {
48
+ id: row.id,
49
+ source: {
50
+ kind: "file",
51
+ path: row.path,
52
+ },
53
+ content: row.content,
54
+ metadata: {
55
+ path: row.path,
56
+ ...(row.lineStart != null ? { lineStart: row.lineStart } : {}),
57
+ ...(row.lineEnd != null ? { lineEnd: row.lineEnd } : {}),
58
+ ...(row.language != null ? { language: row.language } : {}),
59
+ },
60
+ };
61
+ params.eventBus?.emit(CoreEvents.WorkflowReviewContextChunkRetrieved, {
62
+ source: chunk.source,
63
+ metadata: chunk.metadata,
64
+ distance: row.distance,
65
+ });
66
+ return chunk;
67
+ });
68
+ return {
69
+ chunks,
70
+ stats: {
71
+ totalChunks: rows.length,
72
+ truncated: rows.length === limit,
73
+ },
74
+ };
75
+ }
76
+ //# sourceMappingURL=retrieveContext.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"retrieveContext.js","sourceRoot":"","sources":["../../src/review/retrieveContext.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,11 @@
1
+ import { EventBus } from "@prsense/core";
2
+ import type { DiffProvider } from "@prsense/core";
3
+ import type { ResolvedConfig, CredentialContext } from "@prsense/config";
4
+ import type { ReviewWorkflowResult } from "./types.js";
5
+ export declare function runReviewWorkflow({ config, credentials, diffProvider, eventBus, }: {
6
+ config: ResolvedConfig;
7
+ credentials: CredentialContext;
8
+ diffProvider: DiffProvider;
9
+ eventBus: EventBus;
10
+ }): Promise<ReviewWorkflowResult>;
11
+ //# sourceMappingURL=reviewWorkflow.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"reviewWorkflow.d.ts","sourceRoot":"","sources":["../../src/review/reviewWorkflow.ts"],"names":[],"mappings":"AAEA,OAAO,EAAc,QAAQ,EAAqB,MAAM,eAAe,CAAC;AACxE,OAAO,KAAK,EAAgB,YAAY,EAAE,MAAM,eAAe,CAAC;AAChE,OAAO,KAAK,EAAE,cAAc,EAAE,iBAAiB,EAAE,MAAM,iBAAiB,CAAC;AAUzE,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,YAAY,CAAC;AAQvD,wBAAsB,iBAAiB,CAAC,EACtC,MAAM,EACN,WAAW,EACX,YAAY,EACZ,QAAQ,GACT,EAAE;IACD,MAAM,EAAE,cAAc,CAAC;IACvB,WAAW,EAAE,iBAAiB,CAAC;IAC/B,YAAY,EAAE,YAAY,CAAC;IAC3B,QAAQ,EAAE,QAAQ,CAAC;CACpB,GAAG,OAAO,CAAC,oBAAoB,CAAC,CA6UhC"}
@@ -0,0 +1,286 @@
1
+ // packages/workflows/src/review/reviewWorkflow.ts
2
+ import { CoreEvents, buildReviewPrompt } from "@prsense/core";
3
+ import { PostgresIndexMetadataRepository } from "@prsense/context";
4
+ import { createOpenAiClient, createOllamaClient, createGoogleClient, createAnthropicClient, } from "@prsense/llm";
5
+ import { validateReviewOutput } from "./validateReviewOutput.js";
6
+ import { buildDiffEmbeddingQuery } from "./buildDiffEmbeddingQuery.js";
7
+ import { dedupeSignals } from "./dedupeSignals.js";
8
+ import { retrieveContext } from "./retrieveContext.js";
9
+ import { normalizeSignal } from "./normalizeSignal.js";
10
+ import { extractJson } from "./extractJson.js";
11
+ export async function runReviewWorkflow({ config, credentials, diffProvider, eventBus, }) {
12
+ eventBus.emit(CoreEvents.WorkflowReviewStarted);
13
+ try {
14
+ // -------------------------------------------------
15
+ // Load diff from provider
16
+ // -------------------------------------------------
17
+ const { diff, revision, repositoryIdentity, metadata } = await diffProvider.load();
18
+ const metadataRepository = new PostgresIndexMetadataRepository(config.database.url);
19
+ const storedMetadata = await metadataRepository.load(repositoryIdentity.provider, repositoryIdentity.id);
20
+ let contextualReviewAvailable = false;
21
+ if (storedMetadata) {
22
+ const embeddingMatches = storedMetadata.embedding.provider === config.embeddings.provider &&
23
+ storedMetadata.embedding.model === config.embeddings.model;
24
+ if (embeddingMatches) {
25
+ contextualReviewAvailable = true;
26
+ if (storedMetadata.revision.commitSha !== revision) {
27
+ eventBus.emit(CoreEvents.WorkflowReviewIndexOutdated, {
28
+ indexedCommit: storedMetadata.revision.commitSha,
29
+ currentCommit: revision,
30
+ });
31
+ }
32
+ }
33
+ }
34
+ if (!storedMetadata) {
35
+ eventBus.emit(CoreEvents.WorkflowReviewContextUnavailable);
36
+ }
37
+ else if (!contextualReviewAvailable) {
38
+ eventBus.emit(CoreEvents.WorkflowReviewIndexOutdated, {
39
+ indexedCommit: storedMetadata.revision.commitSha,
40
+ currentCommit: revision,
41
+ });
42
+ }
43
+ else {
44
+ eventBus.emit(CoreEvents.WorkflowReviewContextAvailable);
45
+ }
46
+ if (diff.files.length === 0) {
47
+ eventBus.emit(CoreEvents.WorkflowReviewFinished);
48
+ return {
49
+ outcome: "success",
50
+ payload: { signals: [] },
51
+ };
52
+ }
53
+ // -------------------------------------------------
54
+ // Retrieve contextual chunks (RAG) if available
55
+ // -------------------------------------------------
56
+ let contextText = "";
57
+ if (contextualReviewAvailable) {
58
+ const retrievalQuery = buildDiffEmbeddingQuery({
59
+ diff,
60
+ ...(metadata?.title ? { title: metadata.title } : {}),
61
+ ...(metadata?.description ? { description: metadata.description } : {}),
62
+ });
63
+ eventBus.emit(CoreEvents.WorkflowReviewContextQueryBuilt, {
64
+ preview: retrievalQuery.slice(0, 500),
65
+ });
66
+ const retrieved = await retrieveContext({
67
+ config,
68
+ query: retrievalQuery,
69
+ repoProvider: repositoryIdentity.provider,
70
+ repoName: repositoryIdentity.id,
71
+ limit: config.context.maxChunks,
72
+ eventBus,
73
+ });
74
+ // context size guard
75
+ const MAX_CONTEXT_CHARS = 20000;
76
+ let accumulated = "";
77
+ for (const chunk of retrieved.chunks) {
78
+ if (accumulated.length + chunk.content.length > MAX_CONTEXT_CHARS)
79
+ break;
80
+ accumulated += chunk.content + "\n\n";
81
+ }
82
+ contextText = accumulated;
83
+ eventBus.emit(CoreEvents.WorkflowReviewContextRetrieved, {
84
+ chunks: retrieved.stats.totalChunks,
85
+ truncated: retrieved.stats.truncated,
86
+ contextChars: contextText.length,
87
+ });
88
+ }
89
+ // -------------------------------------------------
90
+ // Create LLM client
91
+ // -------------------------------------------------
92
+ let llmClient;
93
+ switch (config.llm.provider) {
94
+ case "openai": {
95
+ const apiKey = credentials.openai?.apiKey;
96
+ if (!apiKey) {
97
+ throw new Error("OpenAI credentials missing");
98
+ }
99
+ llmClient = createOpenAiClient({
100
+ apiKey,
101
+ model: config.llm.model,
102
+ temperature: config.llm.temperature,
103
+ });
104
+ break;
105
+ }
106
+ case "google": {
107
+ const apiKey = credentials.google?.apiKey;
108
+ if (!apiKey) {
109
+ throw new Error("Google credentials missing");
110
+ }
111
+ llmClient = createGoogleClient({
112
+ apiKey,
113
+ model: config.llm.model,
114
+ temperature: config.llm.temperature,
115
+ });
116
+ break;
117
+ }
118
+ case "anthropic": {
119
+ const apiKey = credentials.anthropic?.apiKey;
120
+ if (!apiKey) {
121
+ throw new Error("Anthropic credentials missing");
122
+ }
123
+ llmClient = createAnthropicClient({
124
+ apiKey,
125
+ model: config.llm.model,
126
+ temperature: config.llm.temperature,
127
+ });
128
+ break;
129
+ }
130
+ case "ollama":
131
+ default:
132
+ llmClient = createOllamaClient({
133
+ model: config.llm.model,
134
+ temperature: config.llm.temperature,
135
+ });
136
+ }
137
+ // -------------------------------------------------
138
+ // Generate review
139
+ // -------------------------------------------------
140
+ const allSignals = [];
141
+ let totalUsage;
142
+ for (const file of diff.files) {
143
+ // Skip empty or trivial patches
144
+ if (!file.patch || file.patch.length < 40) {
145
+ continue;
146
+ }
147
+ // skip binary/non code files
148
+ if (file.path.endsWith(".png") ||
149
+ file.path.endsWith(".jpg") ||
150
+ file.path.endsWith(".svg") ||
151
+ file.path.endsWith(".lock")) {
152
+ continue;
153
+ }
154
+ eventBus.emit(CoreEvents.WorkflowReviewFileStarted, {
155
+ file: file.path,
156
+ });
157
+ const fileDiff = {
158
+ files: [file],
159
+ };
160
+ const prompt = buildReviewPrompt({
161
+ diff: fileDiff,
162
+ context: contextText,
163
+ ...(metadata ?? {}),
164
+ });
165
+ eventBus.emit(CoreEvents.WorkflowReviewPromptBuilt, {
166
+ model: config.llm.model,
167
+ provider: config.llm.provider,
168
+ promptChars: JSON.stringify(prompt).length,
169
+ preview: JSON.stringify(prompt).slice(0, 2000),
170
+ });
171
+ eventBus.emit(CoreEvents.WorkflowReviewLlmRequestStarted);
172
+ const start = Date.now();
173
+ const response = await llmClient.generate({ prompt });
174
+ eventBus.emit(CoreEvents.WorkflowReviewLlmResponseReceived, {
175
+ outputChars: response.text.length,
176
+ usage: response.usage,
177
+ durationMs: Date.now() - start,
178
+ });
179
+ eventBus.emit(CoreEvents.WorkflowReviewLlmRawResponse, {
180
+ preview: response.text.slice(0, 1000),
181
+ fullLength: response.text.length,
182
+ });
183
+ if (response.usage) {
184
+ if (!totalUsage) {
185
+ totalUsage = { ...response.usage };
186
+ }
187
+ else {
188
+ totalUsage.promptTokens += response.usage.promptTokens;
189
+ totalUsage.completionTokens += response.usage.completionTokens;
190
+ totalUsage.totalTokens += response.usage.totalTokens;
191
+ }
192
+ }
193
+ const cleaned = extractJson(response.text);
194
+ const trimmed = cleaned.trim();
195
+ if (!(trimmed.endsWith("}") || trimmed.endsWith("]}"))) {
196
+ throw new Error("LLM response truncated");
197
+ }
198
+ let parsed;
199
+ try {
200
+ parsed = JSON.parse(cleaned);
201
+ }
202
+ catch {
203
+ eventBus.emit(CoreEvents.WorkflowReviewInvalidJson, {
204
+ rawResponsePreview: response.text.slice(0, 2000),
205
+ });
206
+ throw new Error("LLM returned invalid JSON");
207
+ }
208
+ let validated;
209
+ try {
210
+ validated = validateReviewOutput(parsed);
211
+ }
212
+ catch {
213
+ const correctionPrompt = `
214
+ The previous output did not match the required schema.
215
+
216
+ Convert the following into valid JSON matching this schema:
217
+
218
+ {
219
+ "signals": [
220
+ {
221
+ "type": "bug" | "risk" | "test" | "style",
222
+ "severity": "low" | "medium" | "high",
223
+ "confidence": number,
224
+ "file": string,
225
+ "lineStart": number | null,
226
+ "lineEnd": number | null,
227
+ "message": string,
228
+ "rationale": string | null,
229
+ "suggestedFix": string | null
230
+ }
231
+ ]
232
+ }
233
+
234
+ Output only valid JSON.
235
+
236
+ Previous output:
237
+ ${cleaned}
238
+ `;
239
+ const retry = await llmClient.generate({
240
+ prompt: {
241
+ system: "You are correcting malformed JSON output.",
242
+ user: correctionPrompt,
243
+ },
244
+ });
245
+ const retryCleaned = extractJson(retry.text);
246
+ parsed = JSON.parse(retryCleaned);
247
+ validated = validateReviewOutput(parsed);
248
+ }
249
+ allSignals.push(...validated.signals);
250
+ }
251
+ // -------------------------------------------------
252
+ // Normalize + filter signals
253
+ // -------------------------------------------------
254
+ const rawSignals = allSignals;
255
+ const normalized = rawSignals
256
+ .map(normalizeSignal)
257
+ .filter((s) => s != null);
258
+ const thresholded = normalized.filter((s) => s.confidence >= config.review.confidenceThreshold);
259
+ const signals = dedupeSignals(thresholded);
260
+ eventBus.emit(CoreEvents.SignalCompiled, {
261
+ count: signals.length,
262
+ });
263
+ eventBus.emit(CoreEvents.WorkflowReviewFinished);
264
+ if (totalUsage) {
265
+ return {
266
+ outcome: "success",
267
+ payload: { signals, usage: totalUsage },
268
+ };
269
+ }
270
+ return {
271
+ outcome: "success",
272
+ payload: { signals },
273
+ };
274
+ }
275
+ catch (err) {
276
+ const message = err instanceof Error ? err.message : String(err);
277
+ eventBus.emit(CoreEvents.WorkflowReviewFailed, {
278
+ error: message,
279
+ });
280
+ return {
281
+ outcome: "failure",
282
+ payload: { signals: [] },
283
+ };
284
+ }
285
+ }
286
+ //# sourceMappingURL=reviewWorkflow.js.map