package hook import ( "encoding/base64" "encoding/json" "fmt" "os" "path" "sort" "strings" "github.com/1Panel-dev/1Panel/backend/app/dto" "github.com/1Panel-dev/1Panel/backend/app/model" "github.com/1Panel-dev/1Panel/backend/app/repo" "github.com/1Panel-dev/1Panel/backend/constant" "github.com/1Panel-dev/1Panel/backend/global" "github.com/1Panel-dev/1Panel/backend/utils/cmd" "github.com/1Panel-dev/1Panel/backend/utils/common" "github.com/1Panel-dev/1Panel/backend/utils/encrypt" "github.com/1Panel-dev/1Panel/backend/utils/files" "github.com/1Panel-dev/1Panel/backend/utils/xpack" ) func Init() { settingRepo := repo.NewISettingRepo() portSetting, err := settingRepo.Get(settingRepo.WithByKey("ServerPort")) if err != nil { global.LOG.Errorf("load service port from setting failed, err: %v", err) } global.CONF.System.Port = portSetting.Value ipv6Setting, err := settingRepo.Get(settingRepo.WithByKey("Ipv6")) if err != nil { global.LOG.Errorf("load ipv6 status from setting failed, err: %v", err) } global.CONF.System.Ipv6 = ipv6Setting.Value bindAddressSetting, err := settingRepo.Get(settingRepo.WithByKey("BindAddress")) if err != nil { global.LOG.Errorf("load bind address from setting failed, err: %v", err) } global.CONF.System.BindAddress = bindAddressSetting.Value sslSetting, err := settingRepo.Get(settingRepo.WithByKey("SSL")) if err != nil { global.LOG.Errorf("load service ssl from setting failed, err: %v", err) } global.CONF.System.SSL = sslSetting.Value OneDriveID, err := settingRepo.Get(settingRepo.WithByKey("OneDriveID")) if err != nil { global.LOG.Errorf("load onedrive info from setting failed, err: %v", err) } idItem, _ := base64.StdEncoding.DecodeString(OneDriveID.Value) global.CONF.System.OneDriveID = string(idItem) OneDriveSc, err := settingRepo.Get(settingRepo.WithByKey("OneDriveSc")) if err != nil { global.LOG.Errorf("load onedrive info from setting failed, err: %v", err) } scItem, _ := base64.StdEncoding.DecodeString(OneDriveSc.Value) global.CONF.System.OneDriveSc = string(scItem) if _, err := settingRepo.Get(settingRepo.WithByKey("SystemStatus")); err != nil { _ = settingRepo.Create("SystemStatus", "Free") } if err := settingRepo.Update("SystemStatus", "Free"); err != nil { global.LOG.Fatalf("init service before start failed, err: %v", err) } apiInterfaceStatusSetting, err := settingRepo.Get(settingRepo.WithByKey("ApiInterfaceStatus")) if err != nil { global.LOG.Errorf("load service api interface from setting failed, err: %v", err) } global.CONF.System.ApiInterfaceStatus = apiInterfaceStatusSetting.Value if apiInterfaceStatusSetting.Value == "enable" { apiKeySetting, err := settingRepo.Get(settingRepo.WithByKey("ApiKey")) if err != nil { global.LOG.Errorf("load service api key from setting failed, err: %v", err) } global.CONF.System.ApiKey = apiKeySetting.Value ipWhiteListSetting, err := settingRepo.Get(settingRepo.WithByKey("IpWhiteList")) if err != nil { global.LOG.Errorf("load service ip white list from setting failed, err: %v", err) } global.CONF.System.IpWhiteList = ipWhiteListSetting.Value } handleUserInfo(global.CONF.System.ChangeUserInfo, settingRepo) handleCronjobStatus() handleSnapStatus() loadLocalDir() initDir() go initLang() } func handleSnapStatus() { msgFailed := "the task was interrupted due to the restart of the 1panel service" _ = global.DB.Model(&model.Snapshot{}).Where("status = ?", "OnSaveData"). Updates(map[string]interface{}{"status": constant.StatusSuccess}).Error _ = global.DB.Model(&model.Snapshot{}).Where("status = ?", constant.StatusWaiting). Updates(map[string]interface{}{ "status": constant.StatusFailed, "message": msgFailed, }).Error _ = global.DB.Model(&model.Snapshot{}).Where("recover_status = ?", constant.StatusWaiting). Updates(map[string]interface{}{ "recover_status": constant.StatusFailed, "recover_message": msgFailed, }).Error _ = global.DB.Model(&model.Snapshot{}).Where("rollback_status = ?", constant.StatusWaiting). Updates(map[string]interface{}{ "rollback_status": constant.StatusFailed, "rollback_message": msgFailed, }).Error snapRepo := repo.NewISnapshotRepo() status, _ := snapRepo.GetStatusList() for _, item := range status { updates := make(map[string]interface{}) if item.Panel == constant.StatusRunning { updates["panel"] = constant.StatusFailed } if item.PanelInfo == constant.StatusRunning { updates["panel_info"] = constant.StatusFailed } if item.DaemonJson == constant.StatusRunning { updates["daemon_json"] = constant.StatusFailed } if item.AppData == constant.StatusRunning { updates["app_data"] = constant.StatusFailed } if item.PanelData == constant.StatusRunning { updates["panel_data"] = constant.StatusFailed } if item.BackupData == constant.StatusRunning { updates["backup_data"] = constant.StatusFailed } if item.Compress == constant.StatusRunning { updates["compress"] = constant.StatusFailed } if item.Upload == constant.StatusUploading { updates["upload"] = constant.StatusFailed } if len(updates) != 0 { _ = snapRepo.UpdateStatus(item.ID, updates) } } } func handleCronjobStatus() { var jobRecords []model.JobRecords _ = global.DB.Where("status = ?", constant.StatusWaiting).Find(&jobRecords).Error for _, record := range jobRecords { err := global.DB.Model(&model.JobRecords{}).Where("status = ?", constant.StatusWaiting). Updates(map[string]interface{}{ "status": constant.StatusFailed, "message": "the task was interrupted due to the restart of the 1panel service", }).Error if err != nil { global.LOG.Errorf("Failed to update job ID: %v, Error:%v", record.ID, err) continue } var cronjob *model.Cronjob _ = global.DB.Where("id = ?", record.CronjobID).First(&cronjob).Error handleCronJobAlert(cronjob) } } func loadLocalDir() { var backup model.BackupAccount _ = global.DB.Where("type = ?", "LOCAL").First(&backup).Error if backup.ID == 0 { global.LOG.Errorf("no such backup account `%s` in db", "LOCAL") return } varMap := make(map[string]interface{}) if err := json.Unmarshal([]byte(backup.Vars), &varMap); err != nil { global.LOG.Errorf("json unmarshal backup.Vars: %v failed, err: %v", backup.Vars, err) return } if _, ok := varMap["dir"]; !ok { global.LOG.Error("load local backup dir failed") return } baseDir, ok := varMap["dir"].(string) if ok { if _, err := os.Stat(baseDir); err != nil && os.IsNotExist(err) { if err = os.MkdirAll(baseDir, os.ModePerm); err != nil { global.LOG.Errorf("mkdir %s failed, err: %v", baseDir, err) return } } global.CONF.System.Backup = baseDir return } global.LOG.Errorf("error type dir: %T", varMap["dir"]) } func handleUserInfo(tags string, settingRepo repo.ISettingRepo) { if len(tags) == 0 { return } if tags == "all" { if err := settingRepo.Update("UserName", common.RandStrAndNum(10)); err != nil { global.LOG.Fatalf("init username before start failed, err: %v", err) } pass, _ := encrypt.StringEncrypt(common.RandStrAndNum(10)) if err := settingRepo.Update("Password", pass); err != nil { global.LOG.Fatalf("init password before start failed, err: %v", err) } if err := settingRepo.Update("SecurityEntrance", common.RandStrAndNum(10)); err != nil { global.LOG.Fatalf("init entrance before start failed, err: %v", err) } return } if strings.Contains(global.CONF.System.ChangeUserInfo, "username") { if err := settingRepo.Update("UserName", common.RandStrAndNum(10)); err != nil { global.LOG.Fatalf("init username before start failed, err: %v", err) } } if strings.Contains(global.CONF.System.ChangeUserInfo, "password") { pass, _ := encrypt.StringEncrypt(common.RandStrAndNum(10)) if err := settingRepo.Update("Password", pass); err != nil { global.LOG.Fatalf("init password before start failed, err: %v", err) } } if strings.Contains(global.CONF.System.ChangeUserInfo, "entrance") { if err := settingRepo.Update("SecurityEntrance", common.RandStrAndNum(10)); err != nil { global.LOG.Fatalf("init entrance before start failed, err: %v", err) } } sudo := cmd.SudoHandleCmd() _, _ = cmd.Execf("%s sed -i '/CHANGE_USER_INFO=%v/d' /usr/local/bin/1pctl", sudo, global.CONF.System.ChangeUserInfo) } func initDir() { composePath := path.Join(global.CONF.System.BaseDir, "1panel/docker/compose/") if _, err := os.Stat(composePath); err != nil && os.IsNotExist(err) { if err = os.MkdirAll(composePath, os.ModePerm); err != nil { global.LOG.Errorf("mkdir %s failed, err: %v", composePath, err) return } } } func handleCronJobAlert(cronjob *model.Cronjob) { if cronjob.Type == "snapshot" { return } pushAlert := dto.PushAlert{ TaskName: cronjob.Name, AlertType: cronjob.Type, EntryID: cronjob.ID, Param: cronjob.Type, } err := xpack.PushAlert(pushAlert) if err != nil { global.LOG.Errorf("cronjob alert push failed, err: %v", err) return } } func initLang() { fileOp := files.NewFileOp() upgradePath := path.Join(global.CONF.System.BaseDir, "1panel/tmp/upgrade") if fileOp.Stat("/usr/local/bin/lang/zh.sh") { return } tmpPath, err := loadRestorePath(upgradePath) files, _ := os.ReadDir(path.Join(upgradePath, tmpPath, "downloads")) if len(files) == 0 { tmpPath = "no such file" } else { for _, item := range files { if item.IsDir() && strings.HasPrefix(item.Name(), "1panel-") { tmpPath = path.Join(upgradePath, tmpPath, "downloads", item.Name(), "lang") break } } } if err != nil || tmpPath == "no such file" || !fileOp.Stat(tmpPath) { path := fmt.Sprintf("%s/language/lang.tar.gz", global.CONF.System.RepoUrl) if err := fileOp.DownloadFileWithProxy(path, "/usr/local/bin/lang.tar.gz"); err != nil { global.LOG.Errorf("download lang.tar.gz failed, err: %v", err) return } if !fileOp.Stat("/usr/local/bin/lang.tar.gz") { global.LOG.Errorf("download lang.tar.gz failed, no such file, err: %v", err) return } std, err := cmd.Execf("tar zxvfC %s %s", "/usr/local/bin/lang.tar.gz", "/usr/local/bin/") if err != nil { fmt.Printf("decompress lang.tar.gz failed, std: %s, err: %v", std, err) return } _ = os.Remove("/usr/local/bin/lang.tar.gz") global.LOG.Info("init lang for 1pctl successful") return } std, err := cmd.Execf("cp -r %s %s", tmpPath, "/usr/local/bin/") if err != nil { fmt.Printf("load lang from package failed, std: %s, err: %v", std, err) return } global.LOG.Info("init lang for 1pctl successful") } func loadRestorePath(upgradeDir string) (string, error) { if _, err := os.Stat(upgradeDir); err != nil && os.IsNotExist(err) { return "no such file", nil } files, err := os.ReadDir(upgradeDir) if err != nil { return "", err } var folders []string for _, file := range files { if file.IsDir() { folders = append(folders, file.Name()) } } if len(folders) == 0 { return "no such file", nil } sort.Slice(folders, func(i, j int) bool { return folders[i] > folders[j] }) return folders[0], nil }