mirror of
https://github.com/httprunner/httprunner.git
synced 2026-05-12 10:59:42 +08:00
164 lines
4.2 KiB
Go
164 lines
4.2 KiB
Go
package myexec
|
|
|
|
import (
|
|
"fmt"
|
|
"os"
|
|
"os/exec"
|
|
"path/filepath"
|
|
"strings"
|
|
|
|
"github.com/pkg/errors"
|
|
"github.com/rs/zerolog/log"
|
|
|
|
"github.com/httprunner/httprunner/v4/hrp/internal/code"
|
|
"github.com/httprunner/httprunner/v4/hrp/internal/env"
|
|
)
|
|
|
|
var python3Executable string = "python3" // system default python3
|
|
|
|
func isPython3(python string) bool {
|
|
out, err := Command(python, "--version").Output()
|
|
if err != nil {
|
|
return false
|
|
}
|
|
if strings.HasPrefix(string(out), "Python 3") {
|
|
return true
|
|
}
|
|
return false
|
|
}
|
|
|
|
// EnsurePython3Venv ensures python3 venv with specified packages
|
|
// venv should be directory path of target venv
|
|
func EnsurePython3Venv(venv string, packages ...string) (python3 string, err error) {
|
|
// priority: specified > $HOME/.hrp/venv
|
|
if venv == "" {
|
|
home, err := os.UserHomeDir()
|
|
if err != nil {
|
|
return "", errors.Wrap(err, "get user home dir failed")
|
|
}
|
|
venv = filepath.Join(home, ".hrp", "venv")
|
|
}
|
|
python3, err = ensurePython3Venv(venv, packages...)
|
|
if err != nil {
|
|
return "", errors.Wrap(code.InvalidPython3Venv, err.Error())
|
|
}
|
|
python3Executable = python3
|
|
log.Info().Str("Python3Executable", python3Executable).Msg("set python3 executable path")
|
|
return python3, nil
|
|
}
|
|
|
|
func ExecPython3Command(cmdName string, args ...string) error {
|
|
args = append([]string{"-m", cmdName}, args...)
|
|
return RunCommand(python3Executable, args...)
|
|
}
|
|
|
|
func AssertPythonPackage(python3 string, pkgName, pkgVersion string) error {
|
|
out, err := exec.Command(
|
|
python3, "-c", fmt.Sprintf("import %s; print(%s.__version__)", pkgName, pkgName),
|
|
).Output()
|
|
if err != nil {
|
|
return fmt.Errorf("python package %s not found", pkgName)
|
|
}
|
|
|
|
// do not check version if pkgVersion is empty
|
|
if pkgVersion == "" {
|
|
log.Info().Str("name", pkgName).Msg("python package is ready")
|
|
return nil
|
|
}
|
|
|
|
// check package version equality
|
|
version := strings.TrimSpace(string(out))
|
|
if strings.TrimLeft(version, "v") != strings.TrimLeft(pkgVersion, "v") {
|
|
return fmt.Errorf("python package %s version %s not matched, please upgrade to %s",
|
|
pkgName, version, pkgVersion)
|
|
}
|
|
|
|
log.Info().Str("name", pkgName).Str("version", pkgVersion).Msg("python package is ready")
|
|
return nil
|
|
}
|
|
|
|
func InstallPythonPackage(python3 string, pkg string) (err error) {
|
|
var pkgName, pkgVersion string
|
|
if strings.Contains(pkg, "==") {
|
|
// funppy==0.5.0
|
|
pkgInfo := strings.Split(pkg, "==")
|
|
pkgName = pkgInfo[0]
|
|
pkgVersion = pkgInfo[1]
|
|
} else {
|
|
// funppy
|
|
pkgName = pkg
|
|
}
|
|
|
|
// check if package installed and version matched
|
|
err = AssertPythonPackage(python3, pkgName, pkgVersion)
|
|
if err == nil {
|
|
return nil
|
|
}
|
|
|
|
// check if pip available
|
|
err = RunCommand(python3, "-m", "pip", "--version")
|
|
if err != nil {
|
|
log.Warn().Msg("pip is not available")
|
|
return errors.Wrap(err, "pip is not available")
|
|
}
|
|
|
|
log.Info().Str("pkgName", pkgName).Str("pkgVersion", pkgVersion).Msg("installing python package")
|
|
|
|
// install package
|
|
pypiIndexURL := env.PYPI_INDEX_URL
|
|
if pypiIndexURL == "" {
|
|
pypiIndexURL = "https://pypi.org/simple" // default
|
|
}
|
|
err = RunCommand(python3, "-m", "pip", "install", "--upgrade", pkg,
|
|
"--index-url", pypiIndexURL,
|
|
"--quiet", "--disable-pip-version-check")
|
|
if err != nil {
|
|
return errors.Wrap(err, "pip install package failed")
|
|
}
|
|
|
|
return AssertPythonPackage(python3, pkgName, pkgVersion)
|
|
}
|
|
|
|
func RunCommand(cmdName string, args ...string) error {
|
|
cmd := Command(cmdName, args...)
|
|
log.Info().Str("cmd", cmd.String()).Msg("exec command")
|
|
|
|
// add cmd dir path to $PATH
|
|
if cmdDir := filepath.Dir(cmdName); cmdDir != "" {
|
|
path := fmt.Sprintf("%s:%s", cmdDir, env.PATH)
|
|
if err := os.Setenv("PATH", path); err != nil {
|
|
log.Error().Err(err).Msg("set env $PATH failed")
|
|
return err
|
|
}
|
|
}
|
|
|
|
// print output with colors
|
|
cmd.Stdout = os.Stdout
|
|
cmd.Stderr = os.Stderr
|
|
|
|
err := cmd.Run()
|
|
if err != nil {
|
|
log.Error().Err(err).Msg("exec command failed")
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func ExecCommandInDir(cmd *exec.Cmd, dir string) error {
|
|
log.Info().Str("cmd", cmd.String()).Str("dir", dir).Msg("exec command")
|
|
cmd.Dir = dir
|
|
|
|
// print output with colors
|
|
cmd.Stdout = os.Stdout
|
|
cmd.Stderr = os.Stderr
|
|
|
|
err := cmd.Run()
|
|
if err != nil {
|
|
log.Error().Err(err).Msg("exec command failed")
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|