package service import ( "bytes" "fmt" "github.com/1Panel-dev/1Panel/backend/app/model" "github.com/1Panel-dev/1Panel/backend/buserr" "github.com/1Panel-dev/1Panel/backend/constant" "github.com/1Panel-dev/1Panel/backend/global" "github.com/1Panel-dev/1Panel/backend/utils/docker" "github.com/1Panel-dev/1Panel/backend/utils/files" "github.com/subosito/gotenv" "io" "os" "os/exec" "path" "strings" ) func buildRuntime(runtime *model.Runtime, oldImageID string) { runtimePath := path.Join(constant.RuntimeDir, runtime.Type, runtime.Name) composePath := path.Join(runtimePath, "docker-compose.yml") logPath := path.Join(runtimePath, "build.log") logFile, err := os.OpenFile(logPath, os.O_CREATE|os.O_WRONLY|os.O_TRUNC, 0666) if err != nil { fmt.Println("Failed to open log file:", err) return } defer func() { _ = logFile.Close() }() cmd := exec.Command("docker-compose", "-f", composePath, "build") multiWriterStdout := io.MultiWriter(os.Stdout, logFile) cmd.Stdout = multiWriterStdout var stderrBuf bytes.Buffer multiWriterStderr := io.MultiWriter(&stderrBuf, logFile, os.Stderr) cmd.Stderr = multiWriterStderr err = cmd.Run() if err != nil { runtime.Status = constant.RuntimeError runtime.Message = buserr.New(constant.ErrImageBuildErr).Error() + ":" + stderrBuf.String() } else { runtime.Status = constant.RuntimeNormal runtime.Message = "" if oldImageID != "" { client, err := docker.NewClient() if err == nil { newImageID, err := client.GetImageIDByName(runtime.Image) if err == nil && newImageID != oldImageID { global.LOG.Infof("delete imageID [%s] ", oldImageID) if err := client.DeleteImage(oldImageID); err != nil { global.LOG.Errorf("delete imageID [%s] error %v", oldImageID, err) } else { global.LOG.Infof("delete old image success") } } } else { global.LOG.Errorf("delete imageID [%s] error %v", oldImageID, err) } } } _ = runtimeRepo.Save(runtime) } func handleParams(image, runtimeType, runtimeDir string, params map[string]interface{}) (composeContent []byte, envContent []byte, forms []byte, err error) { fileOp := files.NewFileOp() composeContent, err = fileOp.GetContent(path.Join(runtimeDir, "docker-compose.yml")) if err != nil { return } env, err := gotenv.Read(path.Join(runtimeDir, ".env")) if err != nil { return } forms, err = fileOp.GetContent(path.Join(runtimeDir, "config.json")) if err != nil { return } params["IMAGE_NAME"] = image if runtimeType == constant.RuntimePHP { if extends, ok := params["PHP_EXTENSIONS"]; ok { if extendsArray, ok := extends.([]interface{}); ok { strArray := make([]string, len(extendsArray)) for i, v := range extendsArray { strArray[i] = strings.ToLower(fmt.Sprintf("%v", v)) } params["PHP_EXTENSIONS"] = strings.Join(strArray, ",") } } } newMap := make(map[string]string) handleMap(params, newMap) for k, v := range newMap { env[k] = v } envStr, err := gotenv.Marshal(env) if err != nil { return } if err = gotenv.Write(env, path.Join(runtimeDir, ".env")); err != nil { return } envContent = []byte(envStr) return }