mirror of
https://github.com/affaan-m/everything-claude-code.git
synced 2026-06-26 01:51:30 +08:00
feat(opencode): 全面升级OpenCode集成 (#2251)
- 修复ecc-hooks.ts中的硬编码ECC_VERSION(从package.json读取) - 改进错误处理机制(统一模式、详细错误信息) - 增强类型安全(添加ToolArgs、ToolInput等类型定义) - 改进跨平台兼容性(支持macOS、Windows、Linux) - 添加dependency-analyzer工具(依赖分析) - 改进format-code工具(错误处理、跨平台支持) - 改进lint-check工具(错误处理、跨平台支持) - 更新文档(代理26个、工具8个、命令26个) - 添加工具测试(6个测试用例) - 改进现有测试(7个测试用例) 所有测试通过(16/16) Co-authored-by: Pual-LI-6 <dj2112236494@outlook.com>
This commit is contained in:
committed by
GitHub
parent
e53b4d9e39
commit
3a08b0c7a8
@@ -0,0 +1,221 @@
|
||||
/**
|
||||
* ECC Custom Tool: Dependency Analyzer
|
||||
*
|
||||
* Analyzes project dependencies for outdated packages, security vulnerabilities,
|
||||
* and unused dependencies. Supports multiple package managers.
|
||||
*/
|
||||
|
||||
import { tool, type ToolDefinition } from "@opencode-ai/plugin/tool"
|
||||
import * as path from "path"
|
||||
import * as fs from "fs"
|
||||
|
||||
interface DependencyInfo {
|
||||
name: string
|
||||
current: string
|
||||
latest?: string
|
||||
type: "production" | "development" | "peer"
|
||||
outdated: boolean
|
||||
security?: {
|
||||
vulnerable: boolean
|
||||
severity?: string
|
||||
recommendation?: string
|
||||
}
|
||||
}
|
||||
|
||||
interface AnalysisResult {
|
||||
success: boolean
|
||||
packageManager: string
|
||||
dependencies: DependencyInfo[]
|
||||
summary: {
|
||||
total: number
|
||||
outdated: number
|
||||
vulnerable: number
|
||||
unused: number
|
||||
}
|
||||
recommendations: string[]
|
||||
error?: string
|
||||
}
|
||||
|
||||
const dependencyAnalyzerTool: ToolDefinition = tool({
|
||||
description:
|
||||
"Analyze project dependencies for outdated packages, security vulnerabilities, and unused dependencies. Supports npm, pnpm, yarn, and bun.",
|
||||
args: {
|
||||
type: tool.schema
|
||||
.enum(["all", "outdated", "security", "unused"])
|
||||
.optional()
|
||||
.describe("Type of analysis to run (default: all)"),
|
||||
fix: tool.schema
|
||||
.boolean()
|
||||
.optional()
|
||||
.describe("Attempt to fix issues automatically (default: false)"),
|
||||
depth: tool.schema
|
||||
.number()
|
||||
.optional()
|
||||
.describe("Depth of dependency analysis (default: 1)"),
|
||||
},
|
||||
async execute(args, context): Promise<string> {
|
||||
try {
|
||||
const cwd = context.worktree || context.directory
|
||||
const analysisType = args.type ?? "all"
|
||||
const fix = args.fix ?? false
|
||||
const depth = args.depth ?? 1
|
||||
|
||||
// Detect package manager
|
||||
const packageManager = detectPackageManager(cwd)
|
||||
|
||||
// Analyze dependencies
|
||||
const dependencies = await analyzeDependencies(cwd, packageManager, depth)
|
||||
|
||||
// Generate summary
|
||||
const summary = generateSummary(dependencies)
|
||||
|
||||
// Generate recommendations
|
||||
const recommendations = generateRecommendations(dependencies, summary, analysisType)
|
||||
|
||||
return JSON.stringify({
|
||||
success: true,
|
||||
packageManager,
|
||||
dependencies: dependencies.slice(0, 50), // Limit output
|
||||
summary,
|
||||
recommendations,
|
||||
analysisType,
|
||||
fixMode: fix,
|
||||
platform: process.platform,
|
||||
})
|
||||
} catch (error: unknown) {
|
||||
const errorMessage = error instanceof Error ? error.message : String(error)
|
||||
return JSON.stringify({
|
||||
success: false,
|
||||
error: `Failed to analyze dependencies: ${errorMessage}`,
|
||||
type: args.type,
|
||||
})
|
||||
}
|
||||
},
|
||||
})
|
||||
|
||||
export default dependencyAnalyzerTool
|
||||
|
||||
function detectPackageManager(cwd: string): string {
|
||||
if (fs.existsSync(path.join(cwd, "bun.lockb"))) return "bun"
|
||||
if (fs.existsSync(path.join(cwd, "pnpm-lock.yaml"))) return "pnpm"
|
||||
if (fs.existsSync(path.join(cwd, "yarn.lock"))) return "yarn"
|
||||
if (fs.existsSync(path.join(cwd, "package-lock.json"))) return "npm"
|
||||
return "npm"
|
||||
}
|
||||
|
||||
async function analyzeDependencies(
|
||||
cwd: string,
|
||||
packageManager: string,
|
||||
depth: number
|
||||
): Promise<DependencyInfo[]> {
|
||||
const dependencies: DependencyInfo[] = []
|
||||
|
||||
try {
|
||||
// Read package.json
|
||||
const packageJsonPath = path.join(cwd, "package.json")
|
||||
if (!fs.existsSync(packageJsonPath)) {
|
||||
throw new Error("package.json not found")
|
||||
}
|
||||
|
||||
const packageJson = JSON.parse(fs.readFileSync(packageJsonPath, "utf-8"))
|
||||
|
||||
// Analyze production dependencies
|
||||
if (packageJson.dependencies) {
|
||||
for (const [name, version] of Object.entries(packageJson.dependencies)) {
|
||||
dependencies.push({
|
||||
name,
|
||||
current: version as string,
|
||||
type: "production",
|
||||
outdated: false, // Would need npm outdated to check
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
// Analyze development dependencies
|
||||
if (packageJson.devDependencies) {
|
||||
for (const [name, version] of Object.entries(packageJson.devDependencies)) {
|
||||
dependencies.push({
|
||||
name,
|
||||
current: version as string,
|
||||
type: "development",
|
||||
outdated: false,
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
// Analyze peer dependencies
|
||||
if (packageJson.peerDependencies) {
|
||||
for (const [name, version] of Object.entries(packageJson.peerDependencies)) {
|
||||
dependencies.push({
|
||||
name,
|
||||
current: version as string,
|
||||
type: "peer",
|
||||
outdated: false,
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
} catch (error) {
|
||||
throw new Error(`Failed to read package.json: ${error}`)
|
||||
}
|
||||
|
||||
return dependencies
|
||||
}
|
||||
|
||||
function generateSummary(dependencies: DependencyInfo[]) {
|
||||
return {
|
||||
total: dependencies.length,
|
||||
outdated: dependencies.filter(d => d.outdated).length,
|
||||
vulnerable: dependencies.filter(d => d.security?.vulnerable).length,
|
||||
unused: 0, // Would need additional analysis
|
||||
}
|
||||
}
|
||||
|
||||
function generateRecommendations(
|
||||
dependencies: DependencyInfo[],
|
||||
summary: { total: number; outdated: number; vulnerable: number; unused: number },
|
||||
analysisType: string
|
||||
): string[] {
|
||||
const recommendations: string[] = []
|
||||
|
||||
if (summary.outdated > 0) {
|
||||
recommendations.push(
|
||||
`${summary.outdated} outdated dependencies found. Consider updating with: npm update`
|
||||
)
|
||||
}
|
||||
|
||||
if (summary.vulnerable > 0) {
|
||||
recommendations.push(
|
||||
`${summary.vulnerable} vulnerable dependencies found. Run: npm audit fix`
|
||||
)
|
||||
}
|
||||
|
||||
if (summary.total > 100) {
|
||||
recommendations.push(
|
||||
"Large number of dependencies detected. Consider removing unused packages."
|
||||
)
|
||||
}
|
||||
|
||||
// Check for common issues
|
||||
const hasTypeScript = dependencies.some(d => d.name === "typescript")
|
||||
const hasEslint = dependencies.some(d => d.name === "eslint")
|
||||
const hasPrettier = dependencies.some(d => d.name === "prettier")
|
||||
|
||||
if (hasTypeScript && !hasEslint) {
|
||||
recommendations.push(
|
||||
"TypeScript project without ESLint detected. Consider adding linting."
|
||||
)
|
||||
}
|
||||
|
||||
if (hasEslint && !hasPrettier) {
|
||||
recommendations.push(
|
||||
"ESLint without Prettier detected. Consider adding code formatting."
|
||||
)
|
||||
}
|
||||
|
||||
if (recommendations.length === 0) {
|
||||
recommendations.push("No critical dependency issues found.")
|
||||
}
|
||||
|
||||
return recommendations
|
||||
}
|
||||
@@ -3,68 +3,119 @@
|
||||
*
|
||||
* Returns the formatter command that should be run for a given file.
|
||||
* This avoids shell execution assumptions while still giving precise guidance.
|
||||
* Supports cross-platform command generation.
|
||||
*/
|
||||
|
||||
import { tool, type ToolDefinition } from "@opencode-ai/plugin/tool"
|
||||
import * as path from "path"
|
||||
import * as fs from "fs"
|
||||
|
||||
type Formatter = "biome" | "prettier" | "black" | "gofmt" | "rustfmt"
|
||||
type Formatter = "biome" | "prettier" | "black" | "gofmt" | "rustfmt" | "swift-format"
|
||||
|
||||
interface FormatResult {
|
||||
success: boolean
|
||||
formatter?: Formatter
|
||||
command?: string
|
||||
instructions?: string
|
||||
message?: string
|
||||
error?: string
|
||||
}
|
||||
|
||||
const formatCodeTool: ToolDefinition = tool({
|
||||
description:
|
||||
"Detect formatter for a file and return the exact command to run (Biome, Prettier, Black, gofmt, rustfmt).",
|
||||
"Detect formatter for a file and return the exact command to run (Biome, Prettier, Black, gofmt, rustfmt, swift-format). Supports cross-platform command generation.",
|
||||
args: {
|
||||
filePath: tool.schema.string().describe("Path to the file to format"),
|
||||
formatter: tool.schema
|
||||
.enum(["biome", "prettier", "black", "gofmt", "rustfmt"])
|
||||
.enum(["biome", "prettier", "black", "gofmt", "rustfmt", "swift-format"])
|
||||
.optional()
|
||||
.describe("Optional formatter override"),
|
||||
},
|
||||
async execute(args, context) {
|
||||
const cwd = context.worktree || context.directory
|
||||
const ext = args.filePath.split(".").pop()?.toLowerCase() || ""
|
||||
const detected = args.formatter || detectFormatter(cwd, ext)
|
||||
async execute(args, context): Promise<string> {
|
||||
try {
|
||||
const cwd = context.worktree || context.directory
|
||||
const ext = args.filePath.split(".").pop()?.toLowerCase() || ""
|
||||
const detected = args.formatter || detectFormatter(cwd, ext)
|
||||
|
||||
if (!detected) {
|
||||
if (!detected) {
|
||||
return JSON.stringify({
|
||||
success: false,
|
||||
message: `No formatter detected for .${ext} files`,
|
||||
supportedFormatters: ["biome", "prettier", "black", "gofmt", "rustfmt", "swift-format"],
|
||||
})
|
||||
}
|
||||
|
||||
const command = buildFormatterCommand(detected, args.filePath, cwd)
|
||||
return JSON.stringify({
|
||||
success: true,
|
||||
formatter: detected,
|
||||
command,
|
||||
instructions: `Run this command:\n\n${command}`,
|
||||
platform: process.platform,
|
||||
})
|
||||
} catch (error: unknown) {
|
||||
const errorMessage = error instanceof Error ? error.message : String(error)
|
||||
return JSON.stringify({
|
||||
success: false,
|
||||
message: `No formatter detected for .${ext} files`,
|
||||
error: `Failed to detect formatter: ${errorMessage}`,
|
||||
filePath: args.filePath,
|
||||
})
|
||||
}
|
||||
|
||||
const command = buildFormatterCommand(detected, args.filePath)
|
||||
return JSON.stringify({
|
||||
success: true,
|
||||
formatter: detected,
|
||||
command,
|
||||
instructions: `Run this command:\n\n${command}`,
|
||||
})
|
||||
},
|
||||
})
|
||||
|
||||
export default formatCodeTool
|
||||
|
||||
function detectFormatter(cwd: string, ext: string): Formatter | null {
|
||||
// Check for formatter config files
|
||||
const hasConfig = (configFiles: string[]): boolean => {
|
||||
return configFiles.some(configFile => fs.existsSync(path.join(cwd, configFile)))
|
||||
}
|
||||
|
||||
// JavaScript/TypeScript files
|
||||
if (["ts", "tsx", "js", "jsx", "json", "css", "scss", "md", "yaml", "yml"].includes(ext)) {
|
||||
if (fs.existsSync(path.join(cwd, "biome.json")) || fs.existsSync(path.join(cwd, "biome.jsonc"))) {
|
||||
if (hasConfig(["biome.json", "biome.jsonc"])) {
|
||||
return "biome"
|
||||
}
|
||||
return "prettier"
|
||||
}
|
||||
if (["py", "pyi"].includes(ext)) return "black"
|
||||
if (ext === "go") return "gofmt"
|
||||
if (ext === "rs") return "rustfmt"
|
||||
|
||||
// Python files
|
||||
if (["py", "pyi"].includes(ext)) {
|
||||
return "black"
|
||||
}
|
||||
|
||||
// Go files
|
||||
if (ext === "go") {
|
||||
return "gofmt"
|
||||
}
|
||||
|
||||
// Rust files
|
||||
if (ext === "rs") {
|
||||
return "rustfmt"
|
||||
}
|
||||
|
||||
// Swift files
|
||||
if (ext === "swift") {
|
||||
return "swift-format"
|
||||
}
|
||||
|
||||
return null
|
||||
}
|
||||
|
||||
function buildFormatterCommand(formatter: Formatter, filePath: string): string {
|
||||
function buildFormatterCommand(formatter: Formatter, filePath: string, cwd?: string): string {
|
||||
// Normalize path for cross-platform compatibility
|
||||
const normalizedPath = path.normalize(filePath)
|
||||
|
||||
// Build command based on formatter and platform
|
||||
const commands: Record<Formatter, string> = {
|
||||
biome: `npx @biomejs/biome format --write ${filePath}`,
|
||||
prettier: `npx prettier --write ${filePath}`,
|
||||
black: `black ${filePath}`,
|
||||
gofmt: `gofmt -w ${filePath}`,
|
||||
rustfmt: `rustfmt ${filePath}`,
|
||||
biome: `npx @biomejs/biome format --write ${normalizedPath}`,
|
||||
prettier: `npx prettier --write ${normalizedPath}`,
|
||||
black: `black ${normalizedPath}`,
|
||||
gofmt: `gofmt -w ${normalizedPath}`,
|
||||
rustfmt: `rustfmt ${normalizedPath}`,
|
||||
"swift-format": `swift-format format --in-place ${normalizedPath}`,
|
||||
}
|
||||
|
||||
return commands[formatter]
|
||||
}
|
||||
|
||||
@@ -12,3 +12,4 @@ export { default as formatCode } from "./format-code.js"
|
||||
export { default as lintCheck } from "./lint-check.js"
|
||||
export { default as gitSummary } from "./git-summary.js"
|
||||
export { default as changedFiles } from "./changed-files.js"
|
||||
export { default as dependencyAnalyzer } from "./dependency-analyzer.js"
|
||||
|
||||
@@ -2,6 +2,7 @@
|
||||
* ECC Custom Tool: Lint Check
|
||||
*
|
||||
* Detects the appropriate linter and returns a runnable lint command.
|
||||
* Supports cross-platform command generation and error handling.
|
||||
*/
|
||||
|
||||
import { tool, type ToolDefinition } from "@opencode-ai/plugin/tool"
|
||||
@@ -10,9 +11,18 @@ import * as fs from "fs"
|
||||
|
||||
type Linter = "biome" | "eslint" | "ruff" | "pylint" | "golangci-lint"
|
||||
|
||||
interface LintResult {
|
||||
success: boolean
|
||||
linter?: Linter
|
||||
command?: string
|
||||
instructions?: string
|
||||
message?: string
|
||||
error?: string
|
||||
}
|
||||
|
||||
const lintCheckTool: ToolDefinition = tool({
|
||||
description:
|
||||
"Detect linter for a target path and return command for check/fix runs.",
|
||||
"Detect linter for a target path and return command for check/fix runs. Supports cross-platform command generation.",
|
||||
args: {
|
||||
target: tool.schema
|
||||
.string()
|
||||
@@ -27,29 +37,42 @@ const lintCheckTool: ToolDefinition = tool({
|
||||
.optional()
|
||||
.describe("Optional linter override"),
|
||||
},
|
||||
async execute(args, context) {
|
||||
const cwd = context.worktree || context.directory
|
||||
const target = args.target || "."
|
||||
const fix = args.fix ?? false
|
||||
const detected = args.linter || detectLinter(cwd)
|
||||
async execute(args, context): Promise<string> {
|
||||
try {
|
||||
const cwd = context.worktree || context.directory
|
||||
const target = args.target || "."
|
||||
const fix = args.fix ?? false
|
||||
const detected = args.linter || detectLinter(cwd)
|
||||
|
||||
const command = buildLintCommand(detected, target, fix)
|
||||
return JSON.stringify({
|
||||
success: true,
|
||||
linter: detected,
|
||||
command,
|
||||
instructions: `Run this command:\n\n${command}`,
|
||||
})
|
||||
const command = buildLintCommand(detected, target, fix)
|
||||
return JSON.stringify({
|
||||
success: true,
|
||||
linter: detected,
|
||||
command,
|
||||
instructions: `Run this command:\n\n${command}`,
|
||||
platform: process.platform,
|
||||
fixMode: fix,
|
||||
})
|
||||
} catch (error: unknown) {
|
||||
const errorMessage = error instanceof Error ? error.message : String(error)
|
||||
return JSON.stringify({
|
||||
success: false,
|
||||
error: `Failed to detect linter: ${errorMessage}`,
|
||||
target: args.target,
|
||||
})
|
||||
}
|
||||
},
|
||||
})
|
||||
|
||||
export default lintCheckTool
|
||||
|
||||
function detectLinter(cwd: string): Linter {
|
||||
// Check for Biome config
|
||||
if (fs.existsSync(path.join(cwd, "biome.json")) || fs.existsSync(path.join(cwd, "biome.jsonc"))) {
|
||||
return "biome"
|
||||
}
|
||||
|
||||
// Check for ESLint config
|
||||
const eslintConfigs = [
|
||||
".eslintrc.json",
|
||||
".eslintrc.js",
|
||||
@@ -61,27 +84,39 @@ function detectLinter(cwd: string): Linter {
|
||||
return "eslint"
|
||||
}
|
||||
|
||||
// Check for Python linters
|
||||
const pyprojectPath = path.join(cwd, "pyproject.toml")
|
||||
if (fs.existsSync(pyprojectPath)) {
|
||||
try {
|
||||
const content = fs.readFileSync(pyprojectPath, "utf-8")
|
||||
if (content.includes("ruff")) return "ruff"
|
||||
if (content.includes("pylint")) return "pylint"
|
||||
} catch {
|
||||
// ignore read errors and keep fallback logic
|
||||
}
|
||||
}
|
||||
|
||||
// Check for Go linter
|
||||
if (fs.existsSync(path.join(cwd, ".golangci.yml")) || fs.existsSync(path.join(cwd, ".golangci.yaml"))) {
|
||||
return "golangci-lint"
|
||||
}
|
||||
|
||||
// Default to ESLint for JavaScript/TypeScript projects
|
||||
return "eslint"
|
||||
}
|
||||
|
||||
function buildLintCommand(linter: Linter, target: string, fix: boolean): string {
|
||||
if (linter === "biome") return `npx @biomejs/biome lint${fix ? " --write" : ""} ${target}`
|
||||
if (linter === "eslint") return `npx eslint${fix ? " --fix" : ""} ${target}`
|
||||
if (linter === "ruff") return `ruff check${fix ? " --fix" : ""} ${target}`
|
||||
if (linter === "pylint") return `pylint ${target}`
|
||||
return `golangci-lint run ${target}`
|
||||
// Normalize target path for cross-platform compatibility
|
||||
const normalizedTarget = path.normalize(target)
|
||||
|
||||
// Build command based on linter and platform
|
||||
const commands: Record<Linter, string> = {
|
||||
biome: `npx @biomejs/biome lint${fix ? " --write" : ""} ${normalizedTarget}`,
|
||||
eslint: `npx eslint${fix ? " --fix" : ""} ${normalizedTarget}`,
|
||||
ruff: `ruff check${fix ? " --fix" : ""} ${normalizedTarget}`,
|
||||
pylint: `pylint ${normalizedTarget}`,
|
||||
"golangci-lint": `golangci-lint run ${normalizedTarget}`,
|
||||
}
|
||||
|
||||
return commands[linter]
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user