mirror of
https://github.com/acepanel/helper.git
synced 2026-02-03 20:20:43 +08:00
160 lines
3.9 KiB
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
|
|
}
|