package service import ( "bytes" "fmt" "github.com/1Panel-dev/1Panel/backend/app/dto/request" "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, rebuild bool) { 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) } } if rebuild && runtime.ID > 0 { websites, _ := websiteRepo.GetBy(websiteRepo.WithRuntimeID(runtime.ID)) if len(websites) > 0 { installService := NewIAppInstalledService() installMap := make(map[uint]string) for _, website := range websites { if website.AppInstallID > 0 { installMap[website.AppInstallID] = website.PrimaryDomain } } for installID, domain := range installMap { go func(installID uint, domain string) { global.LOG.Infof("rebuild php runtime [%s] domain [%s]", runtime.Name, domain) if err := installService.Operate(request.AppInstalledOperate{ InstallId: installID, Operate: constant.Rebuild, }); err != nil { global.LOG.Errorf("rebuild php runtime [%s] domain [%s] error %v", runtime.Name, domain, err) } }(installID, domain) } } } } _ = runtimeRepo.Save(runtime) } func handleParams(image, runtimeType, runtimeDir, source 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, ",") } } params["CONTAINER_PACKAGE_URL"] = source } 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 }