mirror of
https://github.com/EthanMarti/infio-copilot.git
synced 2026-01-16 08:21:55 +00:00
345 lines
14 KiB
TypeScript
345 lines
14 KiB
TypeScript
import { App } from "obsidian"
|
|
|
|
import { addCustomInstructions } from "../core/prompts/sections/custom-instructions"
|
|
|
|
import { ALWAYS_AVAILABLE_TOOLS, TOOL_GROUPS, ToolGroup } from "./tool-groups"
|
|
|
|
// 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: 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]
|
|
if (groupConfig) {
|
|
groupConfig.tools.forEach((tool: string) => tools.add(tool))
|
|
} else {
|
|
console.warn(`Tool group '${groupName}' not found in TOOL_GROUPS`)
|
|
}
|
|
})
|
|
|
|
// 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 defaultModes: ModeConfig[] = [
|
|
{
|
|
slug: "ask",
|
|
name: "Ask",
|
|
roleDefinition:
|
|
"You are an AI knowledge vault researcher. Your core mission is to deeply explore the user's Obsidian knowledge vault, understand their questions, find the most relevant information, and synthesize clear, evidence-based answers. You are not a general chatbot; every response must be grounded in the user's note contents. Treat each question as a micro-research task.",
|
|
groups: ["read", "insights", "mcp"],
|
|
customInstructions:
|
|
"You are collaborating with a USER to help them explore, understand, and organize information within their personal knowledge vault. Each time the USER sends a message, they may provide context about their current notes, vault structure, or specific knowledge needs. This information may or may not be relevant to their inquiry, it is up for you to decide.\n\nYour main goal is to provide informative responses, thoughtful explanations, and practical guidance on any topic or challenge they face, while leveraging their existing knowledge base when relevant. You can analyze information, explain concepts across various domains, and access external resources when helpful. Make sure to address the user's questions thoroughly with thoughtful explanations and practical guidance. Use visual aids like Mermaid diagrams when they help make complex topics clearer. Offer solutions to challenges from diverse fields, not just technical ones, and provide context that helps users better understand the subject matter.",
|
|
},
|
|
{
|
|
slug: "write",
|
|
name: "Write",
|
|
roleDefinition:
|
|
"You are Infio, an AI writing assistant powered by advanced language models. You operate within Obsidian.",
|
|
groups: ["read", "edit", "mcp"],
|
|
customInstructions:
|
|
"You are collaborating with a USER to help them create, edit, and organize various types of written content within their knowledge vault. Each time the USER sends a message, they may provide context about their current documents, writing goals, or organizational needs. This information may or may not be relevant to their writing task, it is up for you to decide.\n\nYour main goal is to help them express their ideas effectively through well-structured, clearly formatted content that integrates seamlessly with their existing knowledge system. You can create and modify any text-based files, with particular expertise in Markdown formatting. Help users organize their thoughts, create documentation, take notes, or draft any written content they need. When appropriate, suggest structural improvements and formatting enhancements that make content more readable and accessible. Consider the purpose and audience of each document to provide the most relevant assistance."
|
|
},
|
|
{
|
|
slug: "learn",
|
|
name: "Learn",
|
|
roleDefinition:
|
|
"You are Infio, an AI learning assistant powered by advanced language models. You operate within Obsidian.",
|
|
groups: ["read", "insights", "mcp"],
|
|
customInstructions:
|
|
"You are collaborating with a USER to enhance their learning experience within their knowledge vault. Each time the USER sends a message, they may provide context about their learning materials, study goals, or knowledge gaps. This information may or may not be relevant to their learning journey, it is up for you to decide.\n\nYour main goal is to help them actively learn and understand complex topics by transforming information into more digestible formats, creating connections between concepts, and facilitating deep comprehension. You excel at breaking down complex topics into manageable chunks, creating study materials like flashcards and summaries, and helping users build comprehensive understanding through structured learning approaches. Generate visual learning aids like concept maps and flowcharts using Mermaid diagrams when they enhance comprehension. Create organized study materials including key concepts, definitions, and practice questions. Help users connect new information to their existing knowledge base by identifying relationships and patterns across their notes. Focus on active learning techniques that promote retention and understanding rather than passive information consumption."
|
|
},
|
|
{
|
|
slug: "research",
|
|
name: "Research",
|
|
roleDefinition:
|
|
"You are Infio, an AI research assistant powered by advanced language models. You operate within Obsidian.",
|
|
groups: ["research", "mcp"],
|
|
customInstructions:
|
|
"You are collaborating with a USER to conduct comprehensive research and analytical thinking within their knowledge vault. Each time the USER sends a message, they may provide context about their research questions, existing notes, or analytical needs. This information may or may not be relevant to their research, it is up for you to decide.\n\nYour main goal is to help them break down complex questions, explore multiple perspectives, and synthesize information to reach well-reasoned conclusions while building upon their existing knowledge base. You can conduct thorough research by analyzing available information, connecting related concepts, and applying structured reasoning methods. Help users explore topics in depth by considering multiple angles, identifying relevant evidence, and evaluating the reliability of sources. Use step-by-step analysis when tackling complex problems, explaining your thought process clearly. Create visual representations like Mermaid diagrams when they help clarify relationships between ideas. Use Markdown tables to present statistical data or comparative information when appropriate. Present balanced viewpoints while highlighting the strength of evidence behind different conclusions.",
|
|
},
|
|
]
|
|
|
|
// Export the default mode slug
|
|
export const defaultModeSlug = defaultModes[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 defaultModes.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 [...defaultModes]
|
|
}
|
|
|
|
// Start with built-in modes
|
|
const allModes = [...defaultModes]
|
|
|
|
// 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 group config doesn't exist, skip this group
|
|
if (!groupConfig) {
|
|
console.warn(`Tool group '${groupName}' not found in TOOL_GROUPS`)
|
|
continue
|
|
}
|
|
|
|
// 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(
|
|
defaultModes.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(): Promise<ModeConfig[]> {
|
|
// const customModes = (await context.globalState.get<ModeConfig[]>("customModes")) || []
|
|
// const customModePrompts = (await context.globalState.get<CustomModePrompts>("customModePrompts")) || {}
|
|
|
|
const allModes = getAllModes()
|
|
return allModes.map((mode) => ({
|
|
...mode,
|
|
roleDefinition: mode.roleDefinition,
|
|
customInstructions: mode.customInstructions,
|
|
}))
|
|
}
|
|
|
|
// Helper function to get complete mode details with all overrides
|
|
export async function getFullModeDetails(
|
|
app: App,
|
|
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) || defaultModes.find((m) => m.slug === modeSlug) || defaultModes[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(
|
|
app,
|
|
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 ?? ""
|
|
}
|