-
Notifications
You must be signed in to change notification settings - Fork 66
Expand file tree
/
Copy pathagent.ts
More file actions
105 lines (97 loc) · 2.71 KB
/
agent.ts
File metadata and controls
105 lines (97 loc) · 2.71 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
import path from 'path';
import { query, type Query } from '@anthropic-ai/claude-agent-sdk';
import { getClaudeCodePlugins, isClaudeCodePluginProvider } from 'cli/ai/claude-code-plugins';
import {
ALLOWED_TOOLS,
STUDIO_ROOT,
createPathApprovalSession,
promptForApproval,
type AskUserQuestion,
} from 'cli/ai/security';
import { buildSystemPrompt } from 'cli/ai/system-prompt';
import { createStudioTools } from 'cli/ai/tools';
import type { AiProviderId } from 'cli/ai/providers';
export type { AskUserQuestion } from 'cli/ai/security';
export interface AiAgentConfig {
prompt: string;
provider?: AiProviderId;
siteContext?: string;
env?: Record< string, string >;
model?: AiModelId;
maxTurns?: number;
resume?: string;
onAskUser?: ( questions: AskUserQuestion[] ) => Promise< Record< string, string > >;
}
export const AI_MODELS = {
'claude-sonnet-4-6': 'Sonnet 4.6',
'claude-opus-4-6': 'Opus 4.6',
} as const;
export type AiModelId = keyof typeof AI_MODELS;
export const DEFAULT_MODEL: AiModelId = 'claude-sonnet-4-6';
const pathApprovalSession = createPathApprovalSession();
/**
* Start the AI agent and return the Query object.
* Caller can iterate messages with `for await` and call `interrupt()` to stop.
*/
export function startAiAgent( config: AiAgentConfig ): Query {
const {
prompt,
provider,
siteContext,
env,
model = DEFAULT_MODEL,
maxTurns = 50,
resume,
onAskUser,
} = config;
const resolvedEnv = env ?? { ...( process.env as Record< string, string > ) };
return query( {
prompt,
options: {
env: resolvedEnv,
systemPrompt: {
type: 'preset',
preset: 'claude_code',
append: buildSystemPrompt( siteContext ),
},
mcpServers: {
studio: createStudioTools(),
},
maxTurns,
cwd: STUDIO_ROOT,
tools: { type: 'preset', preset: 'claude_code' },
allowedTools: [ ...ALLOWED_TOOLS ],
permissionMode: 'default',
canUseTool: async ( toolName, input, metadata ) => {
if ( toolName === 'AskUserQuestion' && onAskUser ) {
const typedInput = input as {
questions?: AskUserQuestion[];
answers?: Record< string, string >;
};
const questions = ( typedInput.questions ?? [] ).map( ( q ) => ( {
...q,
allowFreeForm: true,
} ) );
const answers = await onAskUser( questions );
return {
behavior: 'allow' as const,
updatedInput: { ...input, answers },
};
}
return promptForApproval( {
toolName,
input,
metadata,
onAskUser,
pathApprovalSession,
} );
},
plugins: [
{ type: 'local' as const, path: path.resolve( import.meta.dirname, 'plugin' ) },
...( isClaudeCodePluginProvider( provider ) ? getClaudeCodePlugins() : [] ),
],
model,
resume,
},
} );
}