package cmd import ( "bytes" "context" "fmt" "io" "os/exec" "strings" "time" "github.com/1Panel-dev/1Panel/backend/buserr" "github.com/1Panel-dev/1Panel/backend/constant" ) func Exec(cmdStr string) (string, error) { ctx, cancel := context.WithTimeout(context.Background(), 20*time.Second) defer cancel() cmd := exec.Command("bash", "-c", cmdStr) var stdout, stderr bytes.Buffer cmd.Stdout = &stdout cmd.Stderr = &stderr err := cmd.Run() if ctx.Err() == context.DeadlineExceeded { return "", buserr.New(constant.ErrCmdTimeout) } if err != nil { return handleErr(stdout, stderr, err) } return stdout.String(), nil } func handleErr(stdout, stderr bytes.Buffer, err error) (string, error) { errMsg := "" if len(stderr.String()) != 0 { errMsg = fmt.Sprintf("stderr: %s", stderr.String()) } if len(stdout.String()) != 0 { if len(errMsg) != 0 { errMsg = fmt.Sprintf("%s; stdout: %s", errMsg, stdout.String()) } else { errMsg = fmt.Sprintf("stdout: %s", stdout.String()) } } return errMsg, err } func ExecWithTimeOut(cmdStr string, timeout time.Duration) (string, error) { ctx, cancel := context.WithTimeout(context.Background(), timeout) defer cancel() cmd := exec.Command("bash", "-c", cmdStr) var stdout, stderr bytes.Buffer cmd.Stdout = &stdout cmd.Stderr = &stderr err := cmd.Run() if ctx.Err() == context.DeadlineExceeded { return "", buserr.New(constant.ErrCmdTimeout) } if err != nil { return handleErr(stdout, stderr, err) } return stdout.String(), nil } func ExecContainerScript(containerName, cmdStr string, timeout time.Duration) error { cmdStr = fmt.Sprintf("docker exec -i %s bash -c '%s'", containerName, cmdStr) out, err := ExecWithTimeOut(cmdStr, timeout) if err != nil { if out != "" { return fmt.Errorf("%s; err: %v", out, err) } return err } return nil } func ExecCronjobWithTimeOut(cmdStr string, workdir string, timeout time.Duration) (string, error) { ctx, cancel := context.WithTimeout(context.Background(), timeout) defer cancel() cmd := exec.Command("bash", "-c", cmdStr) cmd.Dir = workdir var stdout, stderr bytes.Buffer cmd.Stdout = &stdout cmd.Stderr = &stderr output := new(bytes.Buffer) cmd.Stdout = io.MultiWriter(output, cmd.Stdout) cmd.Stderr = io.MultiWriter(output, cmd.Stderr) err := cmd.Run() if ctx.Err() == context.DeadlineExceeded { return "", buserr.New(constant.ErrCmdTimeout) } if len(stderr.String()) != 0 { err = buserr.New(constant.ErrBashExecute) } return output.String(), err } func Execf(cmdStr string, a ...interface{}) (string, error) { cmd := exec.Command("bash", "-c", fmt.Sprintf(cmdStr, a...)) var stdout, stderr bytes.Buffer cmd.Stdout = &stdout cmd.Stderr = &stderr err := cmd.Run() if err != nil { return handleErr(stdout, stderr, err) } return stdout.String(), nil } func ExecWithCheck(name string, a ...string) (string, error) { cmd := exec.Command(name, a...) var stdout, stderr bytes.Buffer cmd.Stdout = &stdout cmd.Stderr = &stderr err := cmd.Run() if err != nil { return handleErr(stdout, stderr, err) } return stdout.String(), nil } func ExecScript(scriptPath, workDir string) (string, error) { ctx, cancel := context.WithTimeout(context.Background(), 10*time.Minute) defer cancel() cmd := exec.Command("bash", scriptPath) cmd.Dir = workDir var stdout, stderr bytes.Buffer cmd.Stdout = &stdout cmd.Stderr = &stderr err := cmd.Run() if ctx.Err() == context.DeadlineExceeded { return "", buserr.New(constant.ErrCmdTimeout) } if err != nil { return handleErr(stdout, stderr, err) } return stdout.String(), nil } func ExecCmd(cmdStr string) error { cmd := exec.Command("bash", "-c", cmdStr) output, err := cmd.CombinedOutput() if err != nil { return fmt.Errorf("error : %v, output: %s", err, output) } return nil } func ExecCmdWithDir(cmdStr, workDir string) error { cmd := exec.Command("bash", "-c", cmdStr) cmd.Dir = workDir output, err := cmd.CombinedOutput() if err != nil { return fmt.Errorf("error : %v, output: %s", err, output) } return nil } func CheckIllegal(args ...string) bool { if args == nil { return false } for _, arg := range args { if strings.Contains(arg, "&") || strings.Contains(arg, "|") || strings.Contains(arg, ";") || strings.Contains(arg, "$") || strings.Contains(arg, "'") || strings.Contains(arg, "`") || strings.Contains(arg, "(") || strings.Contains(arg, ")") || strings.Contains(arg, "\"") { return true } } return false } func HasNoPasswordSudo() bool { cmd2 := exec.Command("sudo", "-n", "ls") err2 := cmd2.Run() return err2 == nil } func SudoHandleCmd() string { cmd := exec.Command("sudo", "-n", "ls") if err := cmd.Run(); err == nil { return "sudo " } return "" } func Which(name string) bool { _, err := exec.LookPath(name) return err == nil }