@posthog/agent 1.22.0 → 1.24.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.
- package/CLAUDE.md +3 -3
- package/README.md +3 -3
- package/dist/index.d.ts +11 -11
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +3 -3
- package/dist/src/adapters/claude/claude-adapter.d.ts +3 -3
- package/dist/src/adapters/claude/claude-adapter.d.ts.map +1 -1
- package/dist/src/adapters/claude/claude-adapter.js +156 -111
- package/dist/src/adapters/claude/claude-adapter.js.map +1 -1
- package/dist/src/adapters/claude/tool-mapper.d.ts +1 -1
- package/dist/src/adapters/claude/tool-mapper.d.ts.map +1 -1
- package/dist/src/adapters/claude/tool-mapper.js.map +1 -1
- package/dist/src/adapters/types.d.ts +1 -1
- package/dist/src/adapters/types.d.ts.map +1 -1
- package/dist/src/agent.d.ts +7 -7
- package/dist/src/agent.d.ts.map +1 -1
- package/dist/src/agent.js +143 -85
- package/dist/src/agent.js.map +1 -1
- package/dist/src/agents/execution.js.map +1 -1
- package/dist/src/agents/planning.js.map +1 -1
- package/dist/src/agents/research.js.map +1 -1
- package/dist/src/file-manager.d.ts +4 -4
- package/dist/src/file-manager.d.ts.map +1 -1
- package/dist/src/file-manager.js +59 -58
- package/dist/src/file-manager.js.map +1 -1
- package/dist/src/git-manager.d.ts +1 -1
- package/dist/src/git-manager.d.ts.map +1 -1
- package/dist/src/git-manager.js +93 -69
- package/dist/src/git-manager.js.map +1 -1
- package/dist/src/posthog-api.d.ts +2 -3
- package/dist/src/posthog-api.d.ts.map +1 -1
- package/dist/src/posthog-api.js +22 -22
- package/dist/src/posthog-api.js.map +1 -1
- package/dist/src/prompt-builder.d.ts +3 -3
- package/dist/src/prompt-builder.d.ts.map +1 -1
- package/dist/src/prompt-builder.js +123 -93
- package/dist/src/prompt-builder.js.map +1 -1
- package/dist/src/task-manager.d.ts +4 -4
- package/dist/src/task-manager.d.ts.map +1 -1
- package/dist/src/task-manager.js +19 -18
- package/dist/src/task-manager.js.map +1 -1
- package/dist/src/task-progress-reporter.d.ts +3 -4
- package/dist/src/task-progress-reporter.d.ts.map +1 -1
- package/dist/src/task-progress-reporter.js +59 -54
- package/dist/src/task-progress-reporter.js.map +1 -1
- package/dist/src/template-manager.d.ts +1 -1
- package/dist/src/template-manager.d.ts.map +1 -1
- package/dist/src/template-manager.js +30 -28
- package/dist/src/template-manager.js.map +1 -1
- package/dist/src/todo-manager.d.ts +3 -3
- package/dist/src/todo-manager.d.ts.map +1 -1
- package/dist/src/todo-manager.js +29 -24
- package/dist/src/todo-manager.js.map +1 -1
- package/dist/src/tools/registry.d.ts +1 -1
- package/dist/src/tools/registry.js +60 -60
- package/dist/src/tools/registry.js.map +1 -1
- package/dist/src/tools/types.d.ts +31 -31
- package/dist/src/types.d.ts +33 -33
- package/dist/src/types.d.ts.map +1 -1
- package/dist/src/types.js.map +1 -1
- package/dist/src/utils/logger.d.ts +4 -4
- package/dist/src/utils/logger.d.ts.map +1 -1
- package/dist/src/utils/logger.js +8 -8
- package/dist/src/utils/logger.js.map +1 -1
- package/dist/src/workflow/config.d.ts +1 -1
- package/dist/src/workflow/config.d.ts.map +1 -1
- package/dist/src/workflow/config.js +18 -18
- package/dist/src/workflow/config.js.map +1 -1
- package/dist/src/workflow/steps/build.d.ts +1 -1
- package/dist/src/workflow/steps/build.d.ts.map +1 -1
- package/dist/src/workflow/steps/build.js +46 -38
- package/dist/src/workflow/steps/build.js.map +1 -1
- package/dist/src/workflow/steps/finalize.d.ts +1 -1
- package/dist/src/workflow/steps/finalize.d.ts.map +1 -1
- package/dist/src/workflow/steps/finalize.js +54 -48
- package/dist/src/workflow/steps/finalize.js.map +1 -1
- package/dist/src/workflow/steps/plan.d.ts +1 -1
- package/dist/src/workflow/steps/plan.d.ts.map +1 -1
- package/dist/src/workflow/steps/plan.js +58 -46
- package/dist/src/workflow/steps/plan.js.map +1 -1
- package/dist/src/workflow/steps/research.d.ts +1 -1
- package/dist/src/workflow/steps/research.d.ts.map +1 -1
- package/dist/src/workflow/steps/research.js +68 -56
- package/dist/src/workflow/steps/research.js.map +1 -1
- package/dist/src/workflow/types.d.ts +12 -12
- package/dist/src/workflow/types.d.ts.map +1 -1
- package/dist/src/workflow/utils.d.ts +1 -1
- package/dist/src/workflow/utils.d.ts.map +1 -1
- package/dist/src/workflow/utils.js +7 -4
- package/dist/src/workflow/utils.js.map +1 -1
- package/package.json +8 -8
- package/src/adapters/claude/claude-adapter.ts +220 -168
- package/src/adapters/claude/tool-mapper.ts +2 -2
- package/src/adapters/types.ts +1 -1
- package/src/agent.ts +579 -444
- package/src/agents/execution.ts +1 -1
- package/src/agents/planning.ts +1 -1
- package/src/agents/research.ts +0 -1
- package/src/file-manager.ts +64 -63
- package/src/git-manager.ts +152 -87
- package/src/posthog-api.ts +122 -82
- package/src/prompt-builder.ts +180 -135
- package/src/task-manager.ts +38 -30
- package/src/task-progress-reporter.ts +70 -59
- package/src/template-manager.ts +98 -45
- package/src/todo-manager.ts +35 -30
- package/src/tools/registry.ts +62 -62
- package/src/tools/types.ts +36 -36
- package/src/types.ts +93 -71
- package/src/utils/logger.ts +62 -56
- package/src/workflow/config.ts +48 -48
- package/src/workflow/steps/build.ts +122 -113
- package/src/workflow/steps/finalize.ts +214 -182
- package/src/workflow/steps/plan.ts +151 -131
- package/src/workflow/steps/research.ts +205 -186
- package/src/workflow/types.ts +38 -36
- package/src/workflow/utils.ts +37 -34
- package/LICENSE +0 -33
|
@@ -1,35 +1,41 @@
|
|
|
1
1
|
import { query } from '@anthropic-ai/claude-agent-sdk';
|
|
2
2
|
import { PLANNING_SYSTEM_PROMPT } from '../../agents/planning.js';
|
|
3
|
-
import { finalizeStepGitActions } from '../utils.js';
|
|
4
3
|
import { TodoManager } from '../../todo-manager.js';
|
|
4
|
+
import { finalizeStepGitActions } from '../utils.js';
|
|
5
5
|
|
|
6
6
|
const planStep = async ({ step, context }) => {
|
|
7
7
|
const { task, cwd, isCloudMode, options, logger, fileManager, gitManager, promptBuilder, adapter, mcpServers, emitEvent, } = context;
|
|
8
|
-
const stepLogger = logger.child(
|
|
8
|
+
const stepLogger = logger.child("PlanStep");
|
|
9
9
|
const existingPlan = await fileManager.readPlan(task.id);
|
|
10
10
|
if (existingPlan) {
|
|
11
|
-
stepLogger.info(
|
|
12
|
-
return { status:
|
|
11
|
+
stepLogger.info("Plan already exists, skipping step", { taskId: task.id });
|
|
12
|
+
return { status: "skipped" };
|
|
13
13
|
}
|
|
14
14
|
const researchData = await fileManager.readResearch(task.id);
|
|
15
15
|
if (researchData?.questions && !researchData.answered) {
|
|
16
|
-
stepLogger.info(
|
|
17
|
-
|
|
18
|
-
|
|
16
|
+
stepLogger.info("Waiting for answered research questions", {
|
|
17
|
+
taskId: task.id,
|
|
18
|
+
});
|
|
19
|
+
emitEvent(adapter.createStatusEvent("phase_complete", {
|
|
20
|
+
phase: "research_questions",
|
|
21
|
+
}));
|
|
22
|
+
return { status: "skipped", halt: true };
|
|
19
23
|
}
|
|
20
|
-
stepLogger.info(
|
|
21
|
-
emitEvent(adapter.createStatusEvent(
|
|
22
|
-
let researchContext =
|
|
24
|
+
stepLogger.info("Starting planning phase", { taskId: task.id });
|
|
25
|
+
emitEvent(adapter.createStatusEvent("phase_start", { phase: "planning" }));
|
|
26
|
+
let researchContext = "";
|
|
23
27
|
if (researchData) {
|
|
24
28
|
researchContext += `## Research Context\n\n${researchData.context}\n\n`;
|
|
25
29
|
if (researchData.keyFiles.length > 0) {
|
|
26
|
-
researchContext += `**Key Files:**\n${researchData.keyFiles.map(f => `- ${f}`).join(
|
|
30
|
+
researchContext += `**Key Files:**\n${researchData.keyFiles.map((f) => `- ${f}`).join("\n")}\n\n`;
|
|
27
31
|
}
|
|
28
32
|
if (researchData.blockers && researchData.blockers.length > 0) {
|
|
29
|
-
researchContext += `**Considerations:**\n${researchData.blockers.map(b => `- ${b}`).join(
|
|
33
|
+
researchContext += `**Considerations:**\n${researchData.blockers.map((b) => `- ${b}`).join("\n")}\n\n`;
|
|
30
34
|
}
|
|
31
35
|
// Add answered questions if they exist
|
|
32
|
-
if (researchData.questions &&
|
|
36
|
+
if (researchData.questions &&
|
|
37
|
+
researchData.answers &&
|
|
38
|
+
researchData.answered) {
|
|
33
39
|
researchContext += `## Implementation Decisions\n\n`;
|
|
34
40
|
for (const question of researchData.questions) {
|
|
35
41
|
const answer = researchData.answers.find((a) => a.questionId === question.id);
|
|
@@ -52,21 +58,21 @@ const planStep = async ({ step, context }) => {
|
|
|
52
58
|
const baseOptions = {
|
|
53
59
|
model: step.model,
|
|
54
60
|
cwd,
|
|
55
|
-
permissionMode:
|
|
56
|
-
settingSources: [
|
|
61
|
+
permissionMode: "plan",
|
|
62
|
+
settingSources: ["local"],
|
|
57
63
|
mcpServers,
|
|
58
64
|
// Allow research tools: read-only operations, web search, MCP resources, and ExitPlanMode
|
|
59
65
|
allowedTools: [
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
66
|
+
"Read",
|
|
67
|
+
"Glob",
|
|
68
|
+
"Grep",
|
|
69
|
+
"WebFetch",
|
|
70
|
+
"WebSearch",
|
|
71
|
+
"ListMcpResources",
|
|
72
|
+
"ReadMcpResource",
|
|
73
|
+
"ExitPlanMode",
|
|
74
|
+
"TodoWrite",
|
|
75
|
+
"BashOutput",
|
|
70
76
|
],
|
|
71
77
|
};
|
|
72
78
|
const response = query({
|
|
@@ -74,40 +80,46 @@ const planStep = async ({ step, context }) => {
|
|
|
74
80
|
options: { ...baseOptions, ...(options.queryOverrides || {}) },
|
|
75
81
|
});
|
|
76
82
|
const todoManager = new TodoManager(fileManager, stepLogger);
|
|
77
|
-
let planContent =
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
83
|
+
let planContent = "";
|
|
84
|
+
try {
|
|
85
|
+
for await (const message of response) {
|
|
86
|
+
emitEvent(adapter.createRawSDKEvent(message));
|
|
87
|
+
const transformedEvents = adapter.transform(message);
|
|
88
|
+
for (const event of transformedEvents) {
|
|
89
|
+
emitEvent(event);
|
|
90
|
+
}
|
|
91
|
+
const todoList = await todoManager.checkAndPersistFromMessage(message, task.id);
|
|
92
|
+
if (todoList) {
|
|
93
|
+
emitEvent(adapter.createArtifactEvent("todos", todoList));
|
|
94
|
+
}
|
|
95
|
+
// Extract text content for plan
|
|
96
|
+
if (message.type === "assistant" && message.message?.content) {
|
|
97
|
+
for (const block of message.message.content) {
|
|
98
|
+
if (block.type === "text" && block.text) {
|
|
99
|
+
planContent += `${block.text}\n`;
|
|
100
|
+
}
|
|
93
101
|
}
|
|
94
102
|
}
|
|
95
103
|
}
|
|
96
104
|
}
|
|
105
|
+
catch (error) {
|
|
106
|
+
stepLogger.error("Error during plan step query", error);
|
|
107
|
+
throw error;
|
|
108
|
+
}
|
|
97
109
|
if (planContent.trim()) {
|
|
98
110
|
await fileManager.writePlan(task.id, planContent.trim());
|
|
99
|
-
stepLogger.info(
|
|
111
|
+
stepLogger.info("Plan completed", { taskId: task.id });
|
|
100
112
|
}
|
|
101
113
|
await gitManager.addAllPostHogFiles();
|
|
102
114
|
await finalizeStepGitActions(context, step, {
|
|
103
115
|
commitMessage: `Planning phase for ${task.title}`,
|
|
104
116
|
});
|
|
105
117
|
if (!isCloudMode) {
|
|
106
|
-
emitEvent(adapter.createStatusEvent(
|
|
107
|
-
return { status:
|
|
118
|
+
emitEvent(adapter.createStatusEvent("phase_complete", { phase: "planning" }));
|
|
119
|
+
return { status: "completed", halt: true };
|
|
108
120
|
}
|
|
109
|
-
emitEvent(adapter.createStatusEvent(
|
|
110
|
-
return { status:
|
|
121
|
+
emitEvent(adapter.createStatusEvent("phase_complete", { phase: "planning" }));
|
|
122
|
+
return { status: "completed" };
|
|
111
123
|
};
|
|
112
124
|
|
|
113
125
|
export { planStep };
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"plan.js","sources":["../../../../src/workflow/steps/plan.ts"],"sourcesContent":["import { query } from
|
|
1
|
+
{"version":3,"file":"plan.js","sources":["../../../../src/workflow/steps/plan.ts"],"sourcesContent":["import { query } from \"@anthropic-ai/claude-agent-sdk\";\nimport { PLANNING_SYSTEM_PROMPT } from \"../../agents/planning.js\";\nimport { TodoManager } from \"../../todo-manager.js\";\nimport type { WorkflowStepRunner } from \"../types.js\";\nimport { finalizeStepGitActions } from \"../utils.js\";\n\nexport const planStep: WorkflowStepRunner = async ({ step, context }) => {\n const {\n task,\n cwd,\n isCloudMode,\n options,\n logger,\n fileManager,\n gitManager,\n promptBuilder,\n adapter,\n mcpServers,\n emitEvent,\n } = context;\n\n const stepLogger = logger.child(\"PlanStep\");\n\n const existingPlan = await fileManager.readPlan(task.id);\n if (existingPlan) {\n stepLogger.info(\"Plan already exists, skipping step\", { taskId: task.id });\n return { status: \"skipped\" };\n }\n\n const researchData = await fileManager.readResearch(task.id);\n if (researchData?.questions && !researchData.answered) {\n stepLogger.info(\"Waiting for answered research questions\", {\n taskId: task.id,\n });\n emitEvent(\n adapter.createStatusEvent(\"phase_complete\", {\n phase: \"research_questions\",\n }),\n );\n return { status: \"skipped\", halt: true };\n }\n\n stepLogger.info(\"Starting planning phase\", { taskId: task.id });\n emitEvent(adapter.createStatusEvent(\"phase_start\", { phase: \"planning\" }));\n let researchContext = \"\";\n if (researchData) {\n researchContext += `## Research Context\\n\\n${researchData.context}\\n\\n`;\n if (researchData.keyFiles.length > 0) {\n researchContext += `**Key Files:**\\n${researchData.keyFiles.map((f) => `- ${f}`).join(\"\\n\")}\\n\\n`;\n }\n if (researchData.blockers && researchData.blockers.length > 0) {\n researchContext += `**Considerations:**\\n${researchData.blockers.map((b) => `- ${b}`).join(\"\\n\")}\\n\\n`;\n }\n\n // Add answered questions if they exist\n if (\n researchData.questions &&\n researchData.answers &&\n researchData.answered\n ) {\n researchContext += `## Implementation Decisions\\n\\n`;\n for (const question of researchData.questions) {\n const answer = researchData.answers.find(\n (a) => a.questionId === question.id,\n );\n\n researchContext += `### ${question.question}\\n\\n`;\n if (answer) {\n researchContext += `**Selected:** ${answer.selectedOption}\\n`;\n if (answer.customInput) {\n researchContext += `**Details:** ${answer.customInput}\\n`;\n }\n } else {\n researchContext += `**Selected:** Not answered\\n`;\n }\n researchContext += `\\n`;\n }\n }\n }\n\n const planningPrompt = await promptBuilder.buildPlanningPrompt(task, cwd);\n const fullPrompt = `${PLANNING_SYSTEM_PROMPT}\\n\\n${planningPrompt}\\n\\n${researchContext}`;\n\n const baseOptions: Record<string, unknown> = {\n model: step.model,\n cwd,\n permissionMode: \"plan\",\n settingSources: [\"local\"],\n mcpServers,\n // Allow research tools: read-only operations, web search, MCP resources, and ExitPlanMode\n allowedTools: [\n \"Read\",\n \"Glob\",\n \"Grep\",\n \"WebFetch\",\n \"WebSearch\",\n \"ListMcpResources\",\n \"ReadMcpResource\",\n \"ExitPlanMode\",\n \"TodoWrite\",\n \"BashOutput\",\n ],\n };\n\n const response = query({\n prompt: fullPrompt,\n options: { ...baseOptions, ...(options.queryOverrides || {}) },\n });\n\n const todoManager = new TodoManager(fileManager, stepLogger);\n\n let planContent = \"\";\n try {\n for await (const message of response) {\n emitEvent(adapter.createRawSDKEvent(message));\n const transformedEvents = adapter.transform(message);\n for (const event of transformedEvents) {\n emitEvent(event);\n }\n\n const todoList = await todoManager.checkAndPersistFromMessage(\n message,\n task.id,\n );\n if (todoList) {\n emitEvent(adapter.createArtifactEvent(\"todos\", todoList));\n }\n\n // Extract text content for plan\n if (message.type === \"assistant\" && message.message?.content) {\n for (const block of message.message.content) {\n if (block.type === \"text\" && block.text) {\n planContent += `${block.text}\\n`;\n }\n }\n }\n }\n } catch (error) {\n stepLogger.error(\"Error during plan step query\", error);\n throw error;\n }\n\n if (planContent.trim()) {\n await fileManager.writePlan(task.id, planContent.trim());\n stepLogger.info(\"Plan completed\", { taskId: task.id });\n }\n\n await gitManager.addAllPostHogFiles();\n await finalizeStepGitActions(context, step, {\n commitMessage: `Planning phase for ${task.title}`,\n });\n\n if (!isCloudMode) {\n emitEvent(\n adapter.createStatusEvent(\"phase_complete\", { phase: \"planning\" }),\n );\n return { status: \"completed\", halt: true };\n }\n\n emitEvent(adapter.createStatusEvent(\"phase_complete\", { phase: \"planning\" }));\n return { status: \"completed\" };\n};\n"],"names":[],"mappings":"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"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"research.d.ts","sourceRoot":"","sources":["../../../../src/workflow/steps/research.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"research.d.ts","sourceRoot":"","sources":["../../../../src/workflow/steps/research.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,aAAa,CAAC;AAGtD,eAAO,MAAM,YAAY,EAAE,kBAuN1B,CAAC"}
|
|
@@ -4,80 +4,90 @@ import { finalizeStepGitActions } from '../utils.js';
|
|
|
4
4
|
|
|
5
5
|
const researchStep = async ({ step, context }) => {
|
|
6
6
|
const { task, cwd, isCloudMode, options, logger, fileManager, gitManager, promptBuilder, adapter, mcpServers, emitEvent, } = context;
|
|
7
|
-
const stepLogger = logger.child(
|
|
7
|
+
const stepLogger = logger.child("ResearchStep");
|
|
8
8
|
const existingResearch = await fileManager.readResearch(task.id);
|
|
9
9
|
if (existingResearch) {
|
|
10
|
-
stepLogger.info(
|
|
10
|
+
stepLogger.info("Research already exists", {
|
|
11
|
+
taskId: task.id,
|
|
12
|
+
hasQuestions: !!existingResearch.questions,
|
|
13
|
+
answered: existingResearch.answered,
|
|
14
|
+
});
|
|
11
15
|
// If there are unanswered questions, re-emit them so UI can prompt user
|
|
12
16
|
if (existingResearch.questions && !existingResearch.answered) {
|
|
13
|
-
stepLogger.info(
|
|
17
|
+
stepLogger.info("Re-emitting unanswered research questions", {
|
|
14
18
|
taskId: task.id,
|
|
15
|
-
questionCount: existingResearch.questions.length
|
|
19
|
+
questionCount: existingResearch.questions.length,
|
|
16
20
|
});
|
|
17
21
|
emitEvent({
|
|
18
|
-
type:
|
|
22
|
+
type: "artifact",
|
|
19
23
|
ts: Date.now(),
|
|
20
|
-
kind:
|
|
24
|
+
kind: "research_questions",
|
|
21
25
|
content: existingResearch.questions,
|
|
22
26
|
});
|
|
23
27
|
// In local mode, halt to allow user to answer
|
|
24
28
|
if (!isCloudMode) {
|
|
25
|
-
emitEvent(adapter.createStatusEvent(
|
|
26
|
-
return { status:
|
|
29
|
+
emitEvent(adapter.createStatusEvent("phase_complete", { phase: "research" }));
|
|
30
|
+
return { status: "skipped", halt: true };
|
|
27
31
|
}
|
|
28
32
|
}
|
|
29
|
-
return { status:
|
|
33
|
+
return { status: "skipped" };
|
|
30
34
|
}
|
|
31
|
-
stepLogger.info(
|
|
32
|
-
emitEvent(adapter.createStatusEvent(
|
|
35
|
+
stepLogger.info("Starting research phase", { taskId: task.id });
|
|
36
|
+
emitEvent(adapter.createStatusEvent("phase_start", { phase: "research" }));
|
|
33
37
|
const researchPrompt = await promptBuilder.buildResearchPrompt(task, cwd);
|
|
34
38
|
const fullPrompt = `${RESEARCH_SYSTEM_PROMPT}\n\n${researchPrompt}`;
|
|
35
39
|
const baseOptions = {
|
|
36
40
|
model: step.model,
|
|
37
41
|
cwd,
|
|
38
|
-
permissionMode:
|
|
39
|
-
settingSources: [
|
|
42
|
+
permissionMode: "plan",
|
|
43
|
+
settingSources: ["local"],
|
|
40
44
|
mcpServers,
|
|
41
45
|
// Allow research tools: read-only operations, web search, and MCP resources
|
|
42
46
|
allowedTools: [
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
47
|
+
"Read",
|
|
48
|
+
"Glob",
|
|
49
|
+
"Grep",
|
|
50
|
+
"WebFetch",
|
|
51
|
+
"WebSearch",
|
|
52
|
+
"ListMcpResources",
|
|
53
|
+
"ReadMcpResource",
|
|
54
|
+
"TodoWrite",
|
|
55
|
+
"BashOutput",
|
|
52
56
|
],
|
|
53
57
|
};
|
|
54
58
|
const response = query({
|
|
55
59
|
prompt: fullPrompt,
|
|
56
60
|
options: { ...baseOptions, ...(options.queryOverrides || {}) },
|
|
57
61
|
});
|
|
58
|
-
let jsonContent =
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
62
|
+
let jsonContent = "";
|
|
63
|
+
try {
|
|
64
|
+
for await (const message of response) {
|
|
65
|
+
emitEvent(adapter.createRawSDKEvent(message));
|
|
66
|
+
const transformedEvents = adapter.transform(message);
|
|
67
|
+
for (const event of transformedEvents) {
|
|
68
|
+
emitEvent(event);
|
|
69
|
+
}
|
|
70
|
+
if (message.type === "assistant" && message.message?.content) {
|
|
71
|
+
for (const c of message.message.content) {
|
|
72
|
+
if (c.type === "text" && c.text) {
|
|
73
|
+
jsonContent += c.text;
|
|
74
|
+
}
|
|
69
75
|
}
|
|
70
76
|
}
|
|
71
77
|
}
|
|
72
78
|
}
|
|
79
|
+
catch (error) {
|
|
80
|
+
stepLogger.error("Error during research step query", error);
|
|
81
|
+
throw error;
|
|
82
|
+
}
|
|
73
83
|
if (!jsonContent.trim()) {
|
|
74
|
-
stepLogger.error(
|
|
84
|
+
stepLogger.error("No JSON output from research agent", { taskId: task.id });
|
|
75
85
|
emitEvent({
|
|
76
|
-
type:
|
|
86
|
+
type: "error",
|
|
77
87
|
ts: Date.now(),
|
|
78
|
-
message:
|
|
88
|
+
message: "Research agent returned no output",
|
|
79
89
|
});
|
|
80
|
-
return { status:
|
|
90
|
+
return { status: "completed", halt: true };
|
|
81
91
|
}
|
|
82
92
|
// Parse JSON response
|
|
83
93
|
let evaluation;
|
|
@@ -85,27 +95,27 @@ const researchStep = async ({ step, context }) => {
|
|
|
85
95
|
// Extract JSON from potential markdown code blocks or other wrapping
|
|
86
96
|
const jsonMatch = jsonContent.match(/\{[\s\S]*\}/);
|
|
87
97
|
if (!jsonMatch) {
|
|
88
|
-
throw new Error(
|
|
98
|
+
throw new Error("No JSON object found in response");
|
|
89
99
|
}
|
|
90
100
|
evaluation = JSON.parse(jsonMatch[0]);
|
|
91
|
-
stepLogger.info(
|
|
101
|
+
stepLogger.info("Parsed research evaluation", {
|
|
92
102
|
taskId: task.id,
|
|
93
103
|
score: evaluation.actionabilityScore,
|
|
94
104
|
hasQuestions: !!evaluation.questions,
|
|
95
105
|
});
|
|
96
106
|
}
|
|
97
107
|
catch (error) {
|
|
98
|
-
stepLogger.error(
|
|
108
|
+
stepLogger.error("Failed to parse research JSON", {
|
|
99
109
|
taskId: task.id,
|
|
100
110
|
error: error instanceof Error ? error.message : String(error),
|
|
101
111
|
content: jsonContent.substring(0, 500),
|
|
102
112
|
});
|
|
103
113
|
emitEvent({
|
|
104
|
-
type:
|
|
114
|
+
type: "error",
|
|
105
115
|
ts: Date.now(),
|
|
106
116
|
message: `Failed to parse research JSON: ${error instanceof Error ? error.message : String(error)}`,
|
|
107
117
|
});
|
|
108
|
-
return { status:
|
|
118
|
+
return { status: "completed", halt: true };
|
|
109
119
|
}
|
|
110
120
|
// Add answered/answers fields to evaluation
|
|
111
121
|
if (evaluation.questions && evaluation.questions.length > 0) {
|
|
@@ -114,15 +124,15 @@ const researchStep = async ({ step, context }) => {
|
|
|
114
124
|
}
|
|
115
125
|
// Always write research.json
|
|
116
126
|
await fileManager.writeResearch(task.id, evaluation);
|
|
117
|
-
stepLogger.info(
|
|
127
|
+
stepLogger.info("Research evaluation written", {
|
|
118
128
|
taskId: task.id,
|
|
119
129
|
score: evaluation.actionabilityScore,
|
|
120
130
|
hasQuestions: !!evaluation.questions,
|
|
121
131
|
});
|
|
122
132
|
emitEvent({
|
|
123
|
-
type:
|
|
133
|
+
type: "artifact",
|
|
124
134
|
ts: Date.now(),
|
|
125
|
-
kind:
|
|
135
|
+
kind: "research_evaluation",
|
|
126
136
|
content: evaluation,
|
|
127
137
|
});
|
|
128
138
|
await gitManager.addAllPostHogFiles();
|
|
@@ -130,40 +140,42 @@ const researchStep = async ({ step, context }) => {
|
|
|
130
140
|
commitMessage: `Research phase for ${task.title}`,
|
|
131
141
|
});
|
|
132
142
|
// Log whether questions need answering
|
|
133
|
-
if (evaluation.actionabilityScore < 0.7 &&
|
|
134
|
-
|
|
143
|
+
if (evaluation.actionabilityScore < 0.7 &&
|
|
144
|
+
evaluation.questions &&
|
|
145
|
+
evaluation.questions.length > 0) {
|
|
146
|
+
stepLogger.info("Actionability score below threshold, questions needed", {
|
|
135
147
|
taskId: task.id,
|
|
136
148
|
score: evaluation.actionabilityScore,
|
|
137
149
|
questionCount: evaluation.questions.length,
|
|
138
150
|
});
|
|
139
151
|
emitEvent({
|
|
140
|
-
type:
|
|
152
|
+
type: "artifact",
|
|
141
153
|
ts: Date.now(),
|
|
142
|
-
kind:
|
|
154
|
+
kind: "research_questions",
|
|
143
155
|
content: evaluation.questions,
|
|
144
156
|
});
|
|
145
157
|
}
|
|
146
158
|
else {
|
|
147
|
-
stepLogger.info(
|
|
159
|
+
stepLogger.info("Actionability score acceptable, proceeding to planning", {
|
|
148
160
|
taskId: task.id,
|
|
149
161
|
score: evaluation.actionabilityScore,
|
|
150
162
|
});
|
|
151
163
|
}
|
|
152
164
|
// In local mode, always halt after research for user review
|
|
153
165
|
if (!isCloudMode) {
|
|
154
|
-
emitEvent(adapter.createStatusEvent(
|
|
155
|
-
return { status:
|
|
166
|
+
emitEvent(adapter.createStatusEvent("phase_complete", { phase: "research" }));
|
|
167
|
+
return { status: "completed", halt: true };
|
|
156
168
|
}
|
|
157
169
|
// In cloud mode, check if questions need answering
|
|
158
170
|
const researchData = await fileManager.readResearch(task.id);
|
|
159
171
|
if (researchData?.questions && !researchData.answered) {
|
|
160
172
|
// Questions need answering - halt for user input in cloud mode too
|
|
161
|
-
emitEvent(adapter.createStatusEvent(
|
|
162
|
-
return { status:
|
|
173
|
+
emitEvent(adapter.createStatusEvent("phase_complete", { phase: "research" }));
|
|
174
|
+
return { status: "completed", halt: true };
|
|
163
175
|
}
|
|
164
176
|
// No questions or questions already answered - proceed to planning
|
|
165
|
-
emitEvent(adapter.createStatusEvent(
|
|
166
|
-
return { status:
|
|
177
|
+
emitEvent(adapter.createStatusEvent("phase_complete", { phase: "research" }));
|
|
178
|
+
return { status: "completed" };
|
|
167
179
|
};
|
|
168
180
|
|
|
169
181
|
export { researchStep };
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"research.js","sources":["../../../../src/workflow/steps/research.ts"],"sourcesContent":["import { query } from '@anthropic-ai/claude-agent-sdk';\nimport { RESEARCH_SYSTEM_PROMPT } from '../../agents/research.js';\nimport type { WorkflowStepRunner } from '../types.js';\nimport type { ResearchEvaluation } from '../../types.js';\nimport { finalizeStepGitActions } from '../utils.js';\n\nexport const researchStep: WorkflowStepRunner = async ({ step, context }) => {\n const {\n task,\n cwd,\n isCloudMode,\n options,\n logger,\n fileManager,\n gitManager,\n promptBuilder,\n adapter,\n mcpServers,\n emitEvent,\n } = context;\n\n const stepLogger = logger.child('ResearchStep');\n\n const existingResearch = await fileManager.readResearch(task.id);\n if (existingResearch) {\n stepLogger.info('Research already exists', { taskId: task.id, hasQuestions: !!existingResearch.questions, answered: existingResearch.answered });\n \n // If there are unanswered questions, re-emit them so UI can prompt user\n if (existingResearch.questions && !existingResearch.answered) {\n stepLogger.info('Re-emitting unanswered research questions', { \n taskId: task.id,\n questionCount: existingResearch.questions.length \n });\n \n emitEvent({\n type: 'artifact',\n ts: Date.now(),\n kind: 'research_questions',\n content: existingResearch.questions,\n });\n \n // In local mode, halt to allow user to answer\n if (!isCloudMode) {\n emitEvent(adapter.createStatusEvent('phase_complete', { phase: 'research' }));\n return { status: 'skipped', halt: true };\n }\n }\n \n return { status: 'skipped' };\n }\n\n stepLogger.info('Starting research phase', { taskId: task.id });\n emitEvent(adapter.createStatusEvent('phase_start', { phase: 'research' }));\n\n const researchPrompt = await promptBuilder.buildResearchPrompt(task, cwd);\n const fullPrompt = `${RESEARCH_SYSTEM_PROMPT}\\n\\n${researchPrompt}`;\n\n const baseOptions: Record<string, any> = {\n model: step.model,\n cwd,\n permissionMode: 'plan',\n settingSources: ['local'],\n mcpServers,\n // Allow research tools: read-only operations, web search, and MCP resources\n allowedTools: [\n 'Read',\n 'Glob',\n 'Grep',\n 'WebFetch',\n 'WebSearch',\n 'ListMcpResources',\n 'ReadMcpResource',\n 'TodoWrite',\n 'BashOutput',\n ],\n };\n\n const response = query({\n prompt: fullPrompt,\n options: { ...baseOptions, ...(options.queryOverrides || {}) },\n });\n\n let jsonContent = '';\n for await (const message of response) {\n emitEvent(adapter.createRawSDKEvent(message));\n const transformedEvents = adapter.transform(message);\n for (const event of transformedEvents) {\n emitEvent(event);\n }\n if (message.type === 'assistant' && message.message?.content) {\n for (const c of message.message.content) {\n if (c.type === 'text' && c.text) {\n jsonContent += c.text;\n }\n }\n }\n }\n\n if (!jsonContent.trim()) {\n stepLogger.error('No JSON output from research agent', { taskId: task.id });\n emitEvent({\n type: 'error',\n ts: Date.now(),\n message: 'Research agent returned no output',\n });\n return { status: 'completed', halt: true };\n }\n\n // Parse JSON response\n let evaluation: ResearchEvaluation;\n try {\n // Extract JSON from potential markdown code blocks or other wrapping\n const jsonMatch = jsonContent.match(/\\{[\\s\\S]*\\}/);\n if (!jsonMatch) {\n throw new Error('No JSON object found in response');\n }\n evaluation = JSON.parse(jsonMatch[0]);\n stepLogger.info('Parsed research evaluation', {\n taskId: task.id,\n score: evaluation.actionabilityScore,\n hasQuestions: !!evaluation.questions,\n });\n } catch (error) {\n stepLogger.error('Failed to parse research JSON', {\n taskId: task.id,\n error: error instanceof Error ? error.message : String(error),\n content: jsonContent.substring(0, 500),\n });\n emitEvent({\n type: 'error',\n ts: Date.now(),\n message: `Failed to parse research JSON: ${\n error instanceof Error ? error.message : String(error)\n }`,\n });\n return { status: 'completed', halt: true };\n }\n\n // Add answered/answers fields to evaluation\n if (evaluation.questions && evaluation.questions.length > 0) {\n evaluation.answered = false;\n evaluation.answers = undefined;\n }\n\n // Always write research.json\n await fileManager.writeResearch(task.id, evaluation);\n stepLogger.info('Research evaluation written', {\n taskId: task.id,\n score: evaluation.actionabilityScore,\n hasQuestions: !!evaluation.questions,\n });\n\n emitEvent({\n type: 'artifact',\n ts: Date.now(),\n kind: 'research_evaluation',\n content: evaluation,\n });\n\n await gitManager.addAllPostHogFiles();\n await finalizeStepGitActions(context, step, {\n commitMessage: `Research phase for ${task.title}`,\n });\n\n // Log whether questions need answering\n if (evaluation.actionabilityScore < 0.7 && evaluation.questions && evaluation.questions.length > 0) {\n stepLogger.info('Actionability score below threshold, questions needed', {\n taskId: task.id,\n score: evaluation.actionabilityScore,\n questionCount: evaluation.questions.length,\n });\n \n emitEvent({\n type: 'artifact',\n ts: Date.now(),\n kind: 'research_questions',\n content: evaluation.questions,\n });\n } else {\n stepLogger.info('Actionability score acceptable, proceeding to planning', {\n taskId: task.id,\n score: evaluation.actionabilityScore,\n });\n }\n\n // In local mode, always halt after research for user review\n if (!isCloudMode) {\n emitEvent(adapter.createStatusEvent('phase_complete', { phase: 'research' }));\n return { status: 'completed', halt: true };\n }\n\n // In cloud mode, check if questions need answering\n const researchData = await fileManager.readResearch(task.id);\n if (researchData?.questions && !researchData.answered) {\n // Questions need answering - halt for user input in cloud mode too\n emitEvent(adapter.createStatusEvent('phase_complete', { phase: 'research' }));\n return { status: 'completed', halt: true };\n }\n\n // No questions or questions already answered - proceed to planning\n emitEvent(adapter.createStatusEvent('phase_complete', { phase: 'research' }));\n return { status: 'completed' };\n};\n"],"names":[],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"research.js","sources":["../../../../src/workflow/steps/research.ts"],"sourcesContent":["import { query } from \"@anthropic-ai/claude-agent-sdk\";\nimport { RESEARCH_SYSTEM_PROMPT } from \"../../agents/research.js\";\nimport type { ResearchEvaluation } from \"../../types.js\";\nimport type { WorkflowStepRunner } from \"../types.js\";\nimport { finalizeStepGitActions } from \"../utils.js\";\n\nexport const researchStep: WorkflowStepRunner = async ({ step, context }) => {\n const {\n task,\n cwd,\n isCloudMode,\n options,\n logger,\n fileManager,\n gitManager,\n promptBuilder,\n adapter,\n mcpServers,\n emitEvent,\n } = context;\n\n const stepLogger = logger.child(\"ResearchStep\");\n\n const existingResearch = await fileManager.readResearch(task.id);\n if (existingResearch) {\n stepLogger.info(\"Research already exists\", {\n taskId: task.id,\n hasQuestions: !!existingResearch.questions,\n answered: existingResearch.answered,\n });\n\n // If there are unanswered questions, re-emit them so UI can prompt user\n if (existingResearch.questions && !existingResearch.answered) {\n stepLogger.info(\"Re-emitting unanswered research questions\", {\n taskId: task.id,\n questionCount: existingResearch.questions.length,\n });\n\n emitEvent({\n type: \"artifact\",\n ts: Date.now(),\n kind: \"research_questions\",\n content: existingResearch.questions,\n });\n\n // In local mode, halt to allow user to answer\n if (!isCloudMode) {\n emitEvent(\n adapter.createStatusEvent(\"phase_complete\", { phase: \"research\" }),\n );\n return { status: \"skipped\", halt: true };\n }\n }\n\n return { status: \"skipped\" };\n }\n\n stepLogger.info(\"Starting research phase\", { taskId: task.id });\n emitEvent(adapter.createStatusEvent(\"phase_start\", { phase: \"research\" }));\n\n const researchPrompt = await promptBuilder.buildResearchPrompt(task, cwd);\n const fullPrompt = `${RESEARCH_SYSTEM_PROMPT}\\n\\n${researchPrompt}`;\n\n const baseOptions: Record<string, unknown> = {\n model: step.model,\n cwd,\n permissionMode: \"plan\",\n settingSources: [\"local\"],\n mcpServers,\n // Allow research tools: read-only operations, web search, and MCP resources\n allowedTools: [\n \"Read\",\n \"Glob\",\n \"Grep\",\n \"WebFetch\",\n \"WebSearch\",\n \"ListMcpResources\",\n \"ReadMcpResource\",\n \"TodoWrite\",\n \"BashOutput\",\n ],\n };\n\n const response = query({\n prompt: fullPrompt,\n options: { ...baseOptions, ...(options.queryOverrides || {}) },\n });\n\n let jsonContent = \"\";\n try {\n for await (const message of response) {\n emitEvent(adapter.createRawSDKEvent(message));\n const transformedEvents = adapter.transform(message);\n for (const event of transformedEvents) {\n emitEvent(event);\n }\n if (message.type === \"assistant\" && message.message?.content) {\n for (const c of message.message.content) {\n if (c.type === \"text\" && c.text) {\n jsonContent += c.text;\n }\n }\n }\n }\n } catch (error) {\n stepLogger.error(\"Error during research step query\", error);\n throw error;\n }\n\n if (!jsonContent.trim()) {\n stepLogger.error(\"No JSON output from research agent\", { taskId: task.id });\n emitEvent({\n type: \"error\",\n ts: Date.now(),\n message: \"Research agent returned no output\",\n });\n return { status: \"completed\", halt: true };\n }\n\n // Parse JSON response\n let evaluation: ResearchEvaluation;\n try {\n // Extract JSON from potential markdown code blocks or other wrapping\n const jsonMatch = jsonContent.match(/\\{[\\s\\S]*\\}/);\n if (!jsonMatch) {\n throw new Error(\"No JSON object found in response\");\n }\n evaluation = JSON.parse(jsonMatch[0]);\n stepLogger.info(\"Parsed research evaluation\", {\n taskId: task.id,\n score: evaluation.actionabilityScore,\n hasQuestions: !!evaluation.questions,\n });\n } catch (error) {\n stepLogger.error(\"Failed to parse research JSON\", {\n taskId: task.id,\n error: error instanceof Error ? error.message : String(error),\n content: jsonContent.substring(0, 500),\n });\n emitEvent({\n type: \"error\",\n ts: Date.now(),\n message: `Failed to parse research JSON: ${\n error instanceof Error ? error.message : String(error)\n }`,\n });\n return { status: \"completed\", halt: true };\n }\n\n // Add answered/answers fields to evaluation\n if (evaluation.questions && evaluation.questions.length > 0) {\n evaluation.answered = false;\n evaluation.answers = undefined;\n }\n\n // Always write research.json\n await fileManager.writeResearch(task.id, evaluation);\n stepLogger.info(\"Research evaluation written\", {\n taskId: task.id,\n score: evaluation.actionabilityScore,\n hasQuestions: !!evaluation.questions,\n });\n\n emitEvent({\n type: \"artifact\",\n ts: Date.now(),\n kind: \"research_evaluation\",\n content: evaluation,\n });\n\n await gitManager.addAllPostHogFiles();\n await finalizeStepGitActions(context, step, {\n commitMessage: `Research phase for ${task.title}`,\n });\n\n // Log whether questions need answering\n if (\n evaluation.actionabilityScore < 0.7 &&\n evaluation.questions &&\n evaluation.questions.length > 0\n ) {\n stepLogger.info(\"Actionability score below threshold, questions needed\", {\n taskId: task.id,\n score: evaluation.actionabilityScore,\n questionCount: evaluation.questions.length,\n });\n\n emitEvent({\n type: \"artifact\",\n ts: Date.now(),\n kind: \"research_questions\",\n content: evaluation.questions,\n });\n } else {\n stepLogger.info(\"Actionability score acceptable, proceeding to planning\", {\n taskId: task.id,\n score: evaluation.actionabilityScore,\n });\n }\n\n // In local mode, always halt after research for user review\n if (!isCloudMode) {\n emitEvent(\n adapter.createStatusEvent(\"phase_complete\", { phase: \"research\" }),\n );\n return { status: \"completed\", halt: true };\n }\n\n // In cloud mode, check if questions need answering\n const researchData = await fileManager.readResearch(task.id);\n if (researchData?.questions && !researchData.answered) {\n // Questions need answering - halt for user input in cloud mode too\n emitEvent(\n adapter.createStatusEvent(\"phase_complete\", { phase: \"research\" }),\n );\n return { status: \"completed\", halt: true };\n }\n\n // No questions or questions already answered - proceed to planning\n emitEvent(adapter.createStatusEvent(\"phase_complete\", { phase: \"research\" }));\n return { status: \"completed\" };\n};\n"],"names":[],"mappings":"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"}
|
|
@@ -1,11 +1,11 @@
|
|
|
1
|
-
import type {
|
|
2
|
-
import type {
|
|
3
|
-
import type {
|
|
4
|
-
import type {
|
|
5
|
-
import type { PromptBuilder } from
|
|
6
|
-
import type { TaskProgressReporter } from
|
|
7
|
-
import type {
|
|
8
|
-
import type {
|
|
1
|
+
import type { ProviderAdapter } from "../adapters/types.js";
|
|
2
|
+
import type { PostHogFileManager } from "../file-manager.js";
|
|
3
|
+
import type { GitManager } from "../git-manager.js";
|
|
4
|
+
import type { PostHogAPIClient } from "../posthog-api.js";
|
|
5
|
+
import type { PromptBuilder } from "../prompt-builder.js";
|
|
6
|
+
import type { TaskProgressReporter } from "../task-progress-reporter.js";
|
|
7
|
+
import type { PermissionMode, Task, TaskExecutionOptions } from "../types.js";
|
|
8
|
+
import type { Logger } from "../utils/logger.js";
|
|
9
9
|
export interface WorkflowRuntime {
|
|
10
10
|
task: Task;
|
|
11
11
|
taskSlug: string;
|
|
@@ -18,10 +18,10 @@ export interface WorkflowRuntime {
|
|
|
18
18
|
promptBuilder: PromptBuilder;
|
|
19
19
|
progressReporter: TaskProgressReporter;
|
|
20
20
|
adapter: ProviderAdapter;
|
|
21
|
-
mcpServers?: Record<string,
|
|
21
|
+
mcpServers?: Record<string, unknown>;
|
|
22
22
|
posthogAPI?: PostHogAPIClient;
|
|
23
|
-
emitEvent: (event:
|
|
24
|
-
stepResults: Record<string,
|
|
23
|
+
emitEvent: (event: unknown) => void;
|
|
24
|
+
stepResults: Record<string, unknown>;
|
|
25
25
|
}
|
|
26
26
|
export interface WorkflowStepDefinition {
|
|
27
27
|
id: string;
|
|
@@ -38,7 +38,7 @@ export interface WorkflowStepRuntime {
|
|
|
38
38
|
context: WorkflowRuntime;
|
|
39
39
|
}
|
|
40
40
|
export interface WorkflowStepResult {
|
|
41
|
-
status:
|
|
41
|
+
status: "completed" | "skipped";
|
|
42
42
|
halt?: boolean;
|
|
43
43
|
}
|
|
44
44
|
export type WorkflowStepRunner = (runtime: WorkflowStepRuntime) => Promise<WorkflowStepResult>;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../../src/workflow/types.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,
|
|
1
|
+
{"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../../src/workflow/types.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,sBAAsB,CAAC;AAC5D,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,oBAAoB,CAAC;AAC7D,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,mBAAmB,CAAC;AACpD,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,mBAAmB,CAAC;AAC1D,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,sBAAsB,CAAC;AAC1D,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,8BAA8B,CAAC;AACzE,OAAO,KAAK,EAAE,cAAc,EAAE,IAAI,EAAE,oBAAoB,EAAE,MAAM,aAAa,CAAC;AAC9E,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,oBAAoB,CAAC;AAEjD,MAAM,WAAW,eAAe;IAC9B,IAAI,EAAE,IAAI,CAAC;IACX,QAAQ,EAAE,MAAM,CAAC;IACjB,GAAG,EAAE,MAAM,CAAC;IACZ,WAAW,EAAE,OAAO,CAAC;IACrB,OAAO,EAAE,oBAAoB,CAAC;IAC9B,MAAM,EAAE,MAAM,CAAC;IACf,WAAW,EAAE,kBAAkB,CAAC;IAChC,UAAU,EAAE,UAAU,CAAC;IACvB,aAAa,EAAE,aAAa,CAAC;IAC7B,gBAAgB,EAAE,oBAAoB,CAAC;IACvC,OAAO,EAAE,eAAe,CAAC;IACzB,UAAU,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IACrC,UAAU,CAAC,EAAE,gBAAgB,CAAC;IAC9B,SAAS,EAAE,CAAC,KAAK,EAAE,OAAO,KAAK,IAAI,CAAC;IACpC,WAAW,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;CACtC;AAED,MAAM,WAAW,sBAAsB;IACrC,EAAE,EAAE,MAAM,CAAC;IACX,IAAI,EAAE,MAAM,CAAC;IACb,KAAK,EAAE,MAAM,CAAC;IACd,KAAK,EAAE,MAAM,CAAC;IACd,cAAc,CAAC,EAAE,cAAc,GAAG,MAAM,CAAC;IACzC,MAAM,CAAC,EAAE,OAAO,CAAC;IACjB,IAAI,CAAC,EAAE,OAAO,CAAC;IACf,GAAG,EAAE,kBAAkB,CAAC;CACzB;AAED,MAAM,WAAW,mBAAmB;IAClC,IAAI,EAAE,sBAAsB,CAAC;IAC7B,OAAO,EAAE,eAAe,CAAC;CAC1B;AAED,MAAM,WAAW,kBAAkB;IACjC,MAAM,EAAE,WAAW,GAAG,SAAS,CAAC;IAChC,IAAI,CAAC,EAAE,OAAO,CAAC;CAChB;AAED,MAAM,MAAM,kBAAkB,GAAG,CAC/B,OAAO,EAAE,mBAAmB,KACzB,OAAO,CAAC,kBAAkB,CAAC,CAAC;AAEjC,MAAM,MAAM,kBAAkB,GAAG,sBAAsB,EAAE,CAAC"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"utils.d.ts","sourceRoot":"","sources":["../../../src/workflow/utils.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAE,sBAAsB,EAAE,MAAM,YAAY,CAAC;AAE1E,UAAU,kBAAkB;
|
|
1
|
+
{"version":3,"file":"utils.d.ts","sourceRoot":"","sources":["../../../src/workflow/utils.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAE,sBAAsB,EAAE,MAAM,YAAY,CAAC;AAE1E,UAAU,kBAAkB;IAC1B,aAAa,EAAE,MAAM,CAAC;IACtB,gBAAgB,CAAC,EAAE,OAAO,CAAC;CAC5B;AAED;;;GAGG;AACH,wBAAsB,sBAAsB,CAC1C,OAAO,EAAE,eAAe,EACxB,IAAI,EAAE,sBAAsB,EAC5B,OAAO,EAAE,kBAAkB,GAC1B,OAAO,CAAC,OAAO,CAAC,CAqClB"}
|