| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328 |
- import * as vscode from "vscode"
- import { TOOL_GROUPS, ToolGroup, ALWAYS_AVAILABLE_TOOLS } from "./tool-groups"
- import { addCustomInstructions } from "../core/prompts/sections/custom-instructions"
- // Mode types
- export type Mode = string
- // Group options type
- export type GroupOptions = {
- fileRegex?: string // Regular expression pattern
- description?: string // Human-readable description of the pattern
- }
- // Group entry can be either a string or tuple with options
- export type GroupEntry = ToolGroup | readonly [ToolGroup, GroupOptions]
- // Mode configuration type
- export type ModeConfig = {
- slug: string
- name: string
- roleDefinition: string
- customInstructions?: string
- groups: readonly GroupEntry[] // Now supports both simple strings and tuples with options
- source?: "global" | "project" // Where this mode was loaded from
- }
- // Mode-specific prompts only
- export type PromptComponent = {
- roleDefinition?: string
- customInstructions?: string
- }
- export type CustomModePrompts = {
- [key: string]: PromptComponent | undefined
- }
- // Helper to extract group name regardless of format
- export function getGroupName(group: GroupEntry): ToolGroup {
- if (typeof group === "string") {
- return group
- }
- return group[0]
- }
- // Helper to get group options if they exist
- function getGroupOptions(group: GroupEntry): GroupOptions | undefined {
- return Array.isArray(group) ? group[1] : undefined
- }
- // Helper to check if a file path matches a regex pattern
- export function doesFileMatchRegex(filePath: string, pattern: string): boolean {
- try {
- const regex = new RegExp(pattern)
- return regex.test(filePath)
- } catch (error) {
- console.error(`Invalid regex pattern: ${pattern}`, error)
- return false
- }
- }
- // Helper to get all tools for a mode
- export function getToolsForMode(groups: readonly GroupEntry[]): string[] {
- const tools = new Set<string>()
- // Add tools from each group
- groups.forEach((group) => {
- const groupName = getGroupName(group)
- const groupConfig = TOOL_GROUPS[groupName]
- groupConfig.tools.forEach((tool: string) => tools.add(tool))
- })
- // Always add required tools
- ALWAYS_AVAILABLE_TOOLS.forEach((tool) => tools.add(tool))
- return Array.from(tools)
- }
- // Main modes configuration as an ordered array
- export const modes: readonly ModeConfig[] = [
- {
- slug: "code",
- name: "Code",
- roleDefinition:
- "You are Roo, a highly skilled software engineer with extensive knowledge in many programming languages, frameworks, design patterns, and best practices.",
- groups: ["read", "edit", "browser", "command", "mcp"],
- },
- {
- slug: "architect",
- name: "Architect",
- roleDefinition:
- "You are Roo, an experienced technical leader who is inquisitive and an excellent planner. Your goal is to gather information and get context to create a detailed plan for accomplishing the user's task, which the user will review and approve before they switch into another mode to implement the solution.",
- groups: ["read", ["edit", { fileRegex: "\\.md$", description: "Markdown files only" }], "browser", "mcp"],
- customInstructions:
- "1. Do some information gathering (for example using read_file or search_files) to get more context about the task.\n\n2. You should also ask the user clarifying questions to get a better understanding of the task.\n\n3. Once you've gained more context about the user's request, you should create a detailed plan for how to accomplish the task. Include Mermaid diagrams if they help make your plan clearer.\n\n4. Ask the user if they are pleased with this plan, or if they would like to make any changes. Think of this as a brainstorming session where you can discuss the task and plan the best way to accomplish it.\n\n5. Once the user confirms the plan, ask them if they'd like you to write it to a markdown file.\n\n6. Use the switch_mode tool to request that the user switch to another mode to implement the solution.",
- },
- {
- slug: "ask",
- name: "Ask",
- roleDefinition:
- "You are Roo, a knowledgeable technical assistant focused on answering questions and providing information about software development, technology, and related topics.",
- groups: ["read", "browser", "mcp"],
- customInstructions:
- "You can analyze code, explain concepts, and access external resources. Make sure to answer the user's questions and don't rush to switch to implementing code. Include Mermaid diagrams if they help make your response clearer.",
- },
- {
- slug: "debug",
- name: "Debug",
- roleDefinition:
- "You are Roo, an expert software debugger specializing in systematic problem diagnosis and resolution.",
- groups: ["read", "edit", "browser", "command", "mcp"],
- customInstructions:
- "Reflect on 5-7 different possible sources of the problem, distill those down to 1-2 most likely sources, and then add logs to validate your assumptions. Explicitly ask the user to confirm the diagnosis before fixing the problem.",
- },
- ] as const
- // Export the default mode slug
- export const defaultModeSlug = modes[0].slug
- // Helper functions
- export function getModeBySlug(slug: string, customModes?: ModeConfig[]): ModeConfig | undefined {
- // Check custom modes first
- const customMode = customModes?.find((mode) => mode.slug === slug)
- if (customMode) {
- return customMode
- }
- // Then check built-in modes
- return modes.find((mode) => mode.slug === slug)
- }
- export function getModeConfig(slug: string, customModes?: ModeConfig[]): ModeConfig {
- const mode = getModeBySlug(slug, customModes)
- if (!mode) {
- throw new Error(`No mode found for slug: ${slug}`)
- }
- return mode
- }
- // Get all available modes, with custom modes overriding built-in modes
- export function getAllModes(customModes?: ModeConfig[]): ModeConfig[] {
- if (!customModes?.length) {
- return [...modes]
- }
- // Start with built-in modes
- const allModes = [...modes]
- // Process custom modes
- customModes.forEach((customMode) => {
- const index = allModes.findIndex((mode) => mode.slug === customMode.slug)
- if (index !== -1) {
- // Override existing mode
- allModes[index] = customMode
- } else {
- // Add new mode
- allModes.push(customMode)
- }
- })
- return allModes
- }
- // Check if a mode is custom or an override
- export function isCustomMode(slug: string, customModes?: ModeConfig[]): boolean {
- return !!customModes?.some((mode) => mode.slug === slug)
- }
- // Custom error class for file restrictions
- export class FileRestrictionError extends Error {
- constructor(mode: string, pattern: string, description: string | undefined, filePath: string) {
- super(
- `This mode (${mode}) can only edit files matching pattern: ${pattern}${description ? ` (${description})` : ""}. Got: ${filePath}`,
- )
- this.name = "FileRestrictionError"
- }
- }
- export function isToolAllowedForMode(
- tool: string,
- modeSlug: string,
- customModes: ModeConfig[],
- toolRequirements?: Record<string, boolean>,
- toolParams?: Record<string, any>, // All tool parameters
- experiments?: Record<string, boolean>,
- ): boolean {
- // Always allow these tools
- if (ALWAYS_AVAILABLE_TOOLS.includes(tool as any)) {
- return true
- }
- if (experiments && tool in experiments) {
- if (!experiments[tool]) {
- return false
- }
- }
- // Check tool requirements if any exist
- if (toolRequirements && tool in toolRequirements) {
- if (!toolRequirements[tool]) {
- return false
- }
- }
- const mode = getModeBySlug(modeSlug, customModes)
- if (!mode) {
- return false
- }
- // Check if tool is in any of the mode's groups and respects any group options
- for (const group of mode.groups) {
- const groupName = getGroupName(group)
- const options = getGroupOptions(group)
- const groupConfig = TOOL_GROUPS[groupName]
- // If the tool isn't in this group's tools, continue to next group
- if (!groupConfig.tools.includes(tool)) {
- continue
- }
- // If there are no options, allow the tool
- if (!options) {
- return true
- }
- // For the edit group, check file regex if specified
- if (groupName === "edit" && options.fileRegex) {
- const filePath = toolParams?.path
- if (
- filePath &&
- (toolParams.diff || toolParams.content || toolParams.operations) &&
- !doesFileMatchRegex(filePath, options.fileRegex)
- ) {
- throw new FileRestrictionError(mode.name, options.fileRegex, options.description, filePath)
- }
- }
- return true
- }
- return false
- }
- // Create the mode-specific default prompts
- export const defaultPrompts: Readonly<CustomModePrompts> = Object.freeze(
- Object.fromEntries(
- modes.map((mode) => [
- mode.slug,
- {
- roleDefinition: mode.roleDefinition,
- customInstructions: mode.customInstructions,
- },
- ]),
- ),
- )
- // Helper function to get all modes with their prompt overrides from extension state
- export async function getAllModesWithPrompts(context: vscode.ExtensionContext): Promise<ModeConfig[]> {
- const customModes = (await context.globalState.get<ModeConfig[]>("customModes")) || []
- const customModePrompts = (await context.globalState.get<CustomModePrompts>("customModePrompts")) || {}
- const allModes = getAllModes(customModes)
- return allModes.map((mode) => ({
- ...mode,
- roleDefinition: customModePrompts[mode.slug]?.roleDefinition ?? mode.roleDefinition,
- customInstructions: customModePrompts[mode.slug]?.customInstructions ?? mode.customInstructions,
- }))
- }
- // Helper function to get complete mode details with all overrides
- export async function getFullModeDetails(
- modeSlug: string,
- customModes?: ModeConfig[],
- customModePrompts?: CustomModePrompts,
- options?: {
- cwd?: string
- globalCustomInstructions?: string
- preferredLanguage?: string
- },
- ): Promise<ModeConfig> {
- // First get the base mode config from custom modes or built-in modes
- const baseMode = getModeBySlug(modeSlug, customModes) || modes.find((m) => m.slug === modeSlug) || modes[0]
- // Check for any prompt component overrides
- const promptComponent = customModePrompts?.[modeSlug]
- // Get the base custom instructions
- const baseCustomInstructions = promptComponent?.customInstructions || baseMode.customInstructions || ""
- // If we have cwd, load and combine all custom instructions
- let fullCustomInstructions = baseCustomInstructions
- if (options?.cwd) {
- fullCustomInstructions = await addCustomInstructions(
- baseCustomInstructions,
- options.globalCustomInstructions || "",
- options.cwd,
- modeSlug,
- { preferredLanguage: options.preferredLanguage },
- )
- }
- // Return mode with any overrides applied
- return {
- ...baseMode,
- roleDefinition: promptComponent?.roleDefinition || baseMode.roleDefinition,
- customInstructions: fullCustomInstructions,
- }
- }
- // Helper function to safely get role definition
- export function getRoleDefinition(modeSlug: string, customModes?: ModeConfig[]): string {
- const mode = getModeBySlug(modeSlug, customModes)
- if (!mode) {
- console.warn(`No mode found for slug: ${modeSlug}`)
- return ""
- }
- return mode.roleDefinition
- }
- // Helper function to safely get custom instructions
- export function getCustomInstructions(modeSlug: string, customModes?: ModeConfig[]): string {
- const mode = getModeBySlug(modeSlug, customModes)
- if (!mode) {
- console.warn(`No mode found for slug: ${modeSlug}`)
- return ""
- }
- return mode.customInstructions ?? ""
- }
|