mirror of https://github.com/1Panel-dev/1Panel
337 lines
11 KiB
Go
337 lines
11 KiB
Go
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
|
|
}
|