- Split 25+ index.ts files into hook.ts + extracted modules - Rename all catch-all utils.ts/helpers.ts to domain-specific names - Split src/tools/lsp/ into ~15 focused modules - Split src/tools/delegate-task/ into ~18 focused modules - Separate shared types from implementation - 155 files changed, 60+ new files created - All typecheck clean, 61 tests pass
118 lines
4.8 KiB
TypeScript
118 lines
4.8 KiB
TypeScript
import type { PluginInput } from "@opencode-ai/plugin"
|
|
import { tool, type ToolDefinition } from "@opencode-ai/plugin/tool"
|
|
import { CLI_LANGUAGES } from "./constants"
|
|
import { runSg } from "./cli"
|
|
import { formatSearchResult, formatReplaceResult } from "./result-formatter"
|
|
import type { CliLanguage } from "./types"
|
|
|
|
async function showOutputToUser(context: unknown, output: string): Promise<void> {
|
|
const ctx = context as {
|
|
metadata?: (input: { metadata: { output: string } }) => void | Promise<void>
|
|
}
|
|
await ctx.metadata?.({ metadata: { output } })
|
|
}
|
|
|
|
function getEmptyResultHint(pattern: string, lang: CliLanguage): string | null {
|
|
const src = pattern.trim()
|
|
|
|
if (lang === "python") {
|
|
if (src.startsWith("class ") && src.endsWith(":")) {
|
|
const withoutColon = src.slice(0, -1)
|
|
return `Hint: Remove trailing colon. Try: "${withoutColon}"`
|
|
}
|
|
if ((src.startsWith("def ") || src.startsWith("async def ")) && src.endsWith(":")) {
|
|
const withoutColon = src.slice(0, -1)
|
|
return `Hint: Remove trailing colon. Try: "${withoutColon}"`
|
|
}
|
|
}
|
|
|
|
if (["javascript", "typescript", "tsx"].includes(lang)) {
|
|
if (/^(export\s+)?(async\s+)?function\s+\$[A-Z_]+\s*$/i.test(src)) {
|
|
return `Hint: Function patterns need params and body. Try "function $NAME($$$) { $$$ }"`
|
|
}
|
|
}
|
|
|
|
return null
|
|
}
|
|
|
|
export function createAstGrepTools(ctx: PluginInput): Record<string, ToolDefinition> {
|
|
const ast_grep_search: ToolDefinition = tool({
|
|
description:
|
|
"Search code patterns across filesystem using AST-aware matching. Supports 25 languages. " +
|
|
"Use meta-variables: $VAR (single node), $$$ (multiple nodes). " +
|
|
"IMPORTANT: Patterns must be complete AST nodes (valid code). " +
|
|
"For functions, include params and body: 'export async function $NAME($$$) { $$$ }' not 'export async function $NAME'. " +
|
|
"Examples: 'console.log($MSG)', 'def $FUNC($$$):', 'async function $NAME($$$)'",
|
|
args: {
|
|
pattern: tool.schema.string().describe("AST pattern with meta-variables ($VAR, $$$). Must be complete AST node."),
|
|
lang: tool.schema.enum(CLI_LANGUAGES).describe("Target language"),
|
|
paths: tool.schema.array(tool.schema.string()).optional().describe("Paths to search (default: ['.'])"),
|
|
globs: tool.schema.array(tool.schema.string()).optional().describe("Include/exclude globs (prefix ! to exclude)"),
|
|
context: tool.schema.number().optional().describe("Context lines around match"),
|
|
},
|
|
execute: async (args, context) => {
|
|
try {
|
|
const result = await runSg({
|
|
pattern: args.pattern,
|
|
lang: args.lang as CliLanguage,
|
|
paths: args.paths ?? [ctx.directory],
|
|
globs: args.globs,
|
|
context: args.context,
|
|
})
|
|
|
|
let output = formatSearchResult(result)
|
|
|
|
if (result.matches.length === 0 && !result.error) {
|
|
const hint = getEmptyResultHint(args.pattern, args.lang as CliLanguage)
|
|
if (hint) {
|
|
output += `\n\n${hint}`
|
|
}
|
|
}
|
|
|
|
await showOutputToUser(context, output)
|
|
return output
|
|
} catch (e) {
|
|
const output = `Error: ${e instanceof Error ? e.message : String(e)}`
|
|
await showOutputToUser(context, output)
|
|
return output
|
|
}
|
|
},
|
|
})
|
|
|
|
const ast_grep_replace: ToolDefinition = tool({
|
|
description:
|
|
"Replace code patterns across filesystem with AST-aware rewriting. " +
|
|
"Dry-run by default. Use meta-variables in rewrite to preserve matched content. " +
|
|
"Example: pattern='console.log($MSG)' rewrite='logger.info($MSG)'",
|
|
args: {
|
|
pattern: tool.schema.string().describe("AST pattern to match"),
|
|
rewrite: tool.schema.string().describe("Replacement pattern (can use $VAR from pattern)"),
|
|
lang: tool.schema.enum(CLI_LANGUAGES).describe("Target language"),
|
|
paths: tool.schema.array(tool.schema.string()).optional().describe("Paths to search"),
|
|
globs: tool.schema.array(tool.schema.string()).optional().describe("Include/exclude globs"),
|
|
dryRun: tool.schema.boolean().optional().describe("Preview changes without applying (default: true)"),
|
|
},
|
|
execute: async (args, context) => {
|
|
try {
|
|
const result = await runSg({
|
|
pattern: args.pattern,
|
|
rewrite: args.rewrite,
|
|
lang: args.lang as CliLanguage,
|
|
paths: args.paths ?? [ctx.directory],
|
|
globs: args.globs,
|
|
updateAll: args.dryRun === false,
|
|
})
|
|
const output = formatReplaceResult(result, args.dryRun !== false)
|
|
await showOutputToUser(context, output)
|
|
return output
|
|
} catch (e) {
|
|
const output = `Error: ${e instanceof Error ? e.message : String(e)}`
|
|
await showOutputToUser(context, output)
|
|
return output
|
|
}
|
|
},
|
|
})
|
|
|
|
return { ast_grep_search, ast_grep_replace }
|
|
}
|