2
0
mirror of https://github.com/acepanel/helper.git synced 2026-02-04 01:47:16 +08:00
Files
helper/internal/system/executor.go

160 lines
3.9 KiB
Go

package system
import (
"bytes"
"context"
"errors"
"fmt"
"io"
"os/exec"
"strings"
"github.com/acepanel/helper/pkg/config"
)
// CommandResult 命令执行结果
type CommandResult struct {
ExitCode int
Stdout string
Stderr string
}
// VerboseCallback verbose 模式回调 (cmd, stdout, stderr, err)
type VerboseCallback func(cmd, stdout, stderr string, err error)
// Executor 命令执行器接口
type Executor interface {
// Run 执行命令并等待完成
Run(ctx context.Context, name string, args ...string) (*CommandResult, error)
// RunWithInput 执行命令并提供输入
RunWithInput(ctx context.Context, input string, name string, args ...string) (*CommandResult, error)
// RunStream 执行命令并流式输出
RunStream(ctx context.Context, stdout, stderr io.Writer, name string, args ...string) error
// SetVerboseCallback 设置 verbose 回调
SetVerboseCallback(cb VerboseCallback)
}
type executor struct {
verboseCallback VerboseCallback
}
// NewExecutor 创建执行器
func NewExecutor() Executor {
return &executor{}
}
func (e *executor) SetVerboseCallback(cb VerboseCallback) {
e.verboseCallback = cb
}
func (e *executor) logVerbose(name string, args []string, result *CommandResult, err error) {
cmdStr := name
if len(args) > 0 {
cmdStr += " " + strings.Join(args, " ")
}
// 写入日志文件
if config.Global.LogFile != "" {
config.Global.WriteLog("$ %s", cmdStr)
if result != nil {
if result.Stdout != "" {
config.Global.WriteLog("stdout: %s", strings.TrimSpace(result.Stdout))
}
if result.Stderr != "" {
config.Global.WriteLog("stderr: %s", strings.TrimSpace(result.Stderr))
}
}
if err != nil {
config.Global.WriteLog("error: %v", err)
}
}
// 回调 UI
if config.Global.Verbose && e.verboseCallback != nil {
stdout, stderr := "", ""
if result != nil {
stdout = result.Stdout
stderr = result.Stderr
}
e.verboseCallback(cmdStr, stdout, stderr, err)
}
}
func (e *executor) Run(ctx context.Context, name string, args ...string) (*CommandResult, error) {
cmd := exec.CommandContext(ctx, name, args...)
var stdout, stderr bytes.Buffer
cmd.Stdout = &stdout
cmd.Stderr = &stderr
err := cmd.Run()
result := &CommandResult{
Stdout: stdout.String(),
Stderr: stderr.String(),
}
if err != nil {
var exitErr *exec.ExitError
if errors.As(err, &exitErr) {
result.ExitCode = exitErr.ExitCode()
}
// 包含 stderr 信息到错误中
stderrStr := strings.TrimSpace(stderr.String())
if stderrStr != "" {
err = fmt.Errorf("%w: %s", err, stderrStr)
}
}
e.logVerbose(name, args, result, err)
return result, err
}
func (e *executor) RunWithInput(ctx context.Context, input string, name string, args ...string) (*CommandResult, error) {
cmd := exec.CommandContext(ctx, name, args...)
cmd.Stdin = bytes.NewBufferString(input)
var stdout, stderr bytes.Buffer
cmd.Stdout = &stdout
cmd.Stderr = &stderr
err := cmd.Run()
result := &CommandResult{
Stdout: stdout.String(),
Stderr: stderr.String(),
}
if err != nil {
var exitErr *exec.ExitError
if errors.As(err, &exitErr) {
result.ExitCode = exitErr.ExitCode()
}
// 包含 stderr 信息到错误中
stderrStr := strings.TrimSpace(stderr.String())
if stderrStr != "" {
err = fmt.Errorf("%w: %s", err, stderrStr)
}
}
e.logVerbose(name, args, result, err)
return result, err
}
func (e *executor) RunStream(ctx context.Context, stdoutW, stderrW io.Writer, name string, args ...string) error {
var stdoutBuf, stderrBuf bytes.Buffer
cmd := exec.CommandContext(ctx, name, args...)
cmd.Stdout = io.MultiWriter(stdoutW, &stdoutBuf)
cmd.Stderr = io.MultiWriter(stderrW, &stderrBuf)
err := cmd.Run()
result := &CommandResult{
Stdout: stdoutBuf.String(),
Stderr: stderrBuf.String(),
}
if err != nil {
var exitErr *exec.ExitError
if errors.As(err, &exitErr) {
result.ExitCode = exitErr.ExitCode()
}
}
e.logVerbose(name, args, result, err)
return err
}