v2.6.2
This commit is contained in:
xinsuiyuandong 2025-07-12 02:31:52 +08:00 committed by GitHub
parent da4797fd78
commit 2a2154a267
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194

270
main.go
View File

@ -6,6 +6,8 @@ import (
"log" "log"
"os" "os"
"os/signal" "os/signal"
"os/exec"
"strings"
"syscall" "syscall"
_ "unsafe" _ "unsafe"
@ -13,12 +15,10 @@ import (
"x-ui/database" "x-ui/database"
"x-ui/logger" "x-ui/logger"
"x-ui/sub" "x-ui/sub"
"x-ui/util/crypto"
"x-ui/web" "x-ui/web"
"x-ui/web/global" "x-ui/web/global"
"x-ui/web/service" "x-ui/web/service"
"github.com/joho/godotenv"
"github.com/op/go-logging" "github.com/op/go-logging"
) )
@ -40,11 +40,9 @@ func runWebServer() {
log.Fatalf("Unknown log level: %v", config.GetLogLevel()) log.Fatalf("Unknown log level: %v", config.GetLogLevel())
} }
godotenv.Load()
err := database.InitDB(config.GetDBPath()) err := database.InitDB(config.GetDBPath())
if err != nil { if err != nil {
log.Fatalf("Error initializing database: %v", err) log.Fatalf("Error initializing database(初始化数据库出错): %v", err)
} }
var server *web.Server var server *web.Server
@ -114,65 +112,119 @@ func runWebServer() {
func resetSetting() { func resetSetting() {
err := database.InitDB(config.GetDBPath()) err := database.InitDB(config.GetDBPath())
if err != nil { if err != nil {
fmt.Println("Failed to initialize database:", err) fmt.Println("Failed to initialize database(初始化数据库失败):", err)
return return
} }
settingService := service.SettingService{} settingService := service.SettingService{}
err = settingService.ResetSettings() err = settingService.ResetSettings()
if err != nil { if err != nil {
fmt.Println("Failed to reset settings:", err) fmt.Println("reset setting failed重置设置失败:", err)
} else { } else {
fmt.Println("Settings successfully reset.") fmt.Println("reset setting success---->>重置设置成功")
} }
} }
func showSetting(show bool) { func showSetting(show bool) {
// 执行 shell 命令获取 IPv4 地址
cmdIPv4 := exec.Command("sh", "-c", "curl -s4m8 ip.p3terx.com -k | sed -n 1p")
outputIPv4, err := cmdIPv4.Output()
if err != nil {
log.Fatal(err)
}
// 执行 shell 命令获取 IPv6 地址
cmdIPv6 := exec.Command("sh", "-c", "curl -s6m8 ip.p3terx.com -k | sed -n 1p")
outputIPv6, err := cmdIPv6.Output()
if err != nil {
log.Fatal(err)
}
// 去除命令输出中的换行符
ipv4 := strings.TrimSpace(string(outputIPv4))
ipv6 := strings.TrimSpace(string(outputIPv6))
// 定义转义字符,定义不同颜色的转义字符
const (
Reset = "\033[0m"
Red = "\033[31m"
Green = "\033[32m"
Yellow = "\033[33m"
)
if show { if show {
settingService := service.SettingService{} settingService := service.SettingService{}
port, err := settingService.GetPort() port, err := settingService.GetPort()
if err != nil { if err != nil {
fmt.Println("get current port failed, error info:", err) fmt.Println("get current port failed, error info(获取当前端口失败,错误信息):", err)
} }
webBasePath, err := settingService.GetBasePath() webBasePath, err := settingService.GetBasePath()
if err != nil { if err != nil {
fmt.Println("get webBasePath failed, error info:", err) fmt.Println("get webBasePath failed, error info获取访问路径失败错误信息:", err)
}
certFile, err := settingService.GetCertFile()
if err != nil {
fmt.Println("get cert file failed, error info:", err)
}
keyFile, err := settingService.GetKeyFile()
if err != nil {
fmt.Println("get key file failed, error info:", err)
} }
userService := service.UserService{} userService := service.UserService{}
userModel, err := userService.GetFirstUser() userModel, err := userService.GetFirstUser()
if err != nil { if err != nil {
fmt.Println("get current user info failed, error info:", err) fmt.Println("get current user info failed, error info获取当前用户信息失败错误信息:", err)
} }
if userModel.Username == "" || userModel.Password == "" { username := userModel.Username
fmt.Println("current username or password is empty") userpasswd := userModel.Password
if username == "" || userpasswd == "" {
fmt.Println("current username or password is empty--->>当前用户名或密码为空")
} }
fmt.Println("")
fmt.Println("current panel settings as follows:") fmt.Println(Yellow + "----->>>以下为面板重要信息,请自行记录保存<<<-----" + Reset)
if certFile == "" || keyFile == "" { fmt.Println(Green + "Current panel settings as follows (当前面板设置如下):" + Reset)
fmt.Println("Warning: Panel is not secure with SSL") fmt.Println("")
fmt.Println(Green + fmt.Sprintf("username用户名: %s", username) + Reset)
fmt.Println(Green + fmt.Sprintf("password密 码): %s", userpasswd) + Reset)
fmt.Println(Green + fmt.Sprintf("port端口号: %d", port) + Reset)
if webBasePath != "" {
fmt.Println(Green + fmt.Sprintf("webBasePath访问路径: %s", webBasePath) + Reset)
} else { } else {
fmt.Println("Panel is secure with SSL") fmt.Println("webBasePath is not set----->>未设置访问路径")
} }
fmt.Println("")
fmt.Println("--------------------------------------------------")
// 根据条件打印带颜色的字符串
if ipv4 != "" {
fmt.Println("")
formattedIPv4 := fmt.Sprintf("%s %s%s:%d%s" + Reset,
Green+"面板 IPv4 访问地址------>>",
Yellow+"http://",
ipv4,
port,
Yellow+webBasePath + Reset)
fmt.Println(formattedIPv4)
fmt.Println("")
}
hasDefaultCredential := func() bool { if ipv6 != "" {
return userModel.Username == "admin" && crypto.CheckPasswordHash(userModel.Password, "admin") fmt.Println("")
}() formattedIPv6 := fmt.Sprintf("%s %s[%s%s%s]:%d%s%s",
Green+"面板 IPv6 访问地址------>>", // 绿色的提示信息
fmt.Println("hasDefaultCredential:", hasDefaultCredential) Yellow+"http://", // 黄色的 http:// 部分
fmt.Println("port:", port) Yellow, // 黄色的[ 左方括号
fmt.Println("webBasePath:", webBasePath) ipv6, // IPv6 地址
Yellow, // 黄色的] 右方括号
port, // 端口号
Yellow+webBasePath, // 黄色的 Web 基础路径
Reset) // 重置颜色
fmt.Println(formattedIPv6)
fmt.Println("")
}
fmt.Println(Green + ">>>>>>>>注若您安装了证书请把IP换成您的域名用https方式登录" + Reset)
fmt.Println("")
fmt.Println("--------------------------------------------------")
fmt.Println("↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑")
fmt.Println(fmt.Sprintf("%s请确保 %s%d%s 端口已打开放行%s",Green, Red, port, Green, Reset))
fmt.Println("请自行确保此端口没有被其他程序占用")
fmt.Println(Green + "若要登录访问面板,请复制上面的地址到浏览器" + Reset)
fmt.Println("")
fmt.Println("--------------------------------------------------")
fmt.Println("")
} }
} }
@ -198,7 +250,7 @@ func updateTgbotEnableSts(status bool) {
func updateTgbotSetting(tgBotToken string, tgBotChatid string, tgBotRuntime string) { func updateTgbotSetting(tgBotToken string, tgBotChatid string, tgBotRuntime string) {
err := database.InitDB(config.GetDBPath()) err := database.InitDB(config.GetDBPath())
if err != nil { if err != nil {
fmt.Println("Error initializing database:", err) fmt.Println("Error initializing database(初始化数据库出错):", err)
return return
} }
@ -207,35 +259,35 @@ func updateTgbotSetting(tgBotToken string, tgBotChatid string, tgBotRuntime stri
if tgBotToken != "" { if tgBotToken != "" {
err := settingService.SetTgBotToken(tgBotToken) err := settingService.SetTgBotToken(tgBotToken)
if err != nil { if err != nil {
fmt.Printf("Error setting Telegram bot token: %v\n", err) fmt.Printf("Error setting Telegram bot token设置TG电报机器人令牌出错: %v\n", err)
return return
} }
logger.Info("Successfully updated Telegram bot token.") logger.Info("Successfully updated Telegram bot token----->>已成功更新TG电报机器人令牌")
} }
if tgBotRuntime != "" { if tgBotRuntime != "" {
err := settingService.SetTgbotRuntime(tgBotRuntime) err := settingService.SetTgbotRuntime(tgBotRuntime)
if err != nil { if err != nil {
fmt.Printf("Error setting Telegram bot runtime: %v\n", err) fmt.Printf("Error setting Telegram bot runtime设置TG电报机器人通知周期出错: %v\n", err)
return return
} }
logger.Infof("Successfully updated Telegram bot runtime to [%s].", tgBotRuntime) logger.Infof("Successfully updated Telegram bot runtime to已成功将TG电报机器人通知周期设置为 [%s].", tgBotRuntime)
} }
if tgBotChatid != "" { if tgBotChatid != "" {
err := settingService.SetTgBotChatId(tgBotChatid) err := settingService.SetTgBotChatId(tgBotChatid)
if err != nil { if err != nil {
fmt.Printf("Error setting Telegram bot chat ID: %v\n", err) fmt.Printf("Error setting Telegram bot chat ID设置TG电报机器人管理者聊天ID出错: %v\n", err)
return return
} }
logger.Info("Successfully updated Telegram bot chat ID.") logger.Info("Successfully updated Telegram bot chat ID----->>已成功更新TG电报机器人管理者聊天ID")
} }
} }
func updateSetting(port int, username string, password string, webBasePath string, listenIP string, resetTwoFactor bool) { func updateSetting(port int, username string, password string, webBasePath string) {
err := database.InitDB(config.GetDBPath()) err := database.InitDB(config.GetDBPath())
if err != nil { if err != nil {
fmt.Println("Database initialization failed:", err) fmt.Println("Database initialization failed(初始化数据库失败):", err)
return return
} }
@ -245,47 +297,27 @@ func updateSetting(port int, username string, password string, webBasePath strin
if port > 0 { if port > 0 {
err := settingService.SetPort(port) err := settingService.SetPort(port)
if err != nil { if err != nil {
fmt.Println("Failed to set port:", err) fmt.Println("Failed to set port(设置端口失败):", err)
} else { } else {
fmt.Printf("Port set successfully: %v\n", port) fmt.Printf("Port set successfully(端口设置成功): %v\n", port)
} }
} }
if username != "" || password != "" { if username != "" || password != "" {
err := userService.UpdateFirstUser(username, password) err := userService.UpdateFirstUser(username, password)
if err != nil { if err != nil {
fmt.Println("Failed to update username and password:", err) fmt.Println("Failed to update username and password(更新用户名和密码失败):", err)
} else { } else {
fmt.Println("Username and password updated successfully") fmt.Println("Username and password updated successfully------>>用户名和密码更新成功")
} }
} }
if webBasePath != "" { if webBasePath != "" {
err := settingService.SetBasePath(webBasePath) err := settingService.SetBasePath(webBasePath)
if err != nil { if err != nil {
fmt.Println("Failed to set base URI path:", err) fmt.Println("Failed to set base URI path(设置访问路径失败):", err)
} else { } else {
fmt.Println("Base URI path set successfully") fmt.Println("Base URI path set successfully------>>设置访问路径成功")
}
}
if resetTwoFactor {
err := settingService.SetTwoFactorEnable(false)
if err != nil {
fmt.Println("Failed to reset two-factor authentication:", err)
} else {
settingService.SetTwoFactorToken("")
fmt.Println("Two-factor authentication reset successfully")
}
}
if listenIP != "" {
err := settingService.SetListen(listenIP)
if err != nil {
fmt.Println("Failed to set listen IP:", err)
} else {
fmt.Printf("listen %v set successfully", listenIP)
} }
} }
} }
@ -301,50 +333,19 @@ func updateCert(publicKey string, privateKey string) {
settingService := service.SettingService{} settingService := service.SettingService{}
err = settingService.SetCertFile(publicKey) err = settingService.SetCertFile(publicKey)
if err != nil { if err != nil {
fmt.Println("set certificate public key failed:", err) fmt.Println("set certificate public key failed(设置证书公钥失败):", err)
} else { } else {
fmt.Println("set certificate public key success") fmt.Println("set certificate public key success--------->>设置证书公钥成功")
} }
err = settingService.SetKeyFile(privateKey) err = settingService.SetKeyFile(privateKey)
if err != nil { if err != nil {
fmt.Println("set certificate private key failed:", err) fmt.Println("set certificate private key failed(设置证书私钥失败):", err)
} else { } else {
fmt.Println("set certificate private key success") fmt.Println("set certificate private key success--------->>设置证书私钥成功")
} }
} else { } else {
fmt.Println("both public and private key should be entered.") fmt.Println("both public and private key should be entered.------>>必须同时输入证书公钥和私钥")
}
}
func GetCertificate(getCert bool) {
if getCert {
settingService := service.SettingService{}
certFile, err := settingService.GetCertFile()
if err != nil {
fmt.Println("get cert file failed, error info:", err)
}
keyFile, err := settingService.GetKeyFile()
if err != nil {
fmt.Println("get key file failed, error info:", err)
}
fmt.Println("cert:", certFile)
fmt.Println("key:", keyFile)
}
}
func GetListenIP(getListen bool) {
if getListen {
settingService := service.SettingService{}
ListenIP, err := settingService.GetListen()
if err != nil {
log.Printf("Failed to retrieve listen IP: %v", err)
return
}
fmt.Println("listenIP:", ListenIP)
} }
} }
@ -355,9 +356,40 @@ func migrateDb() {
if err != nil { if err != nil {
log.Fatal(err) log.Fatal(err)
} }
fmt.Println("Start migrating database...") fmt.Println("Start migrating database...---->>开始迁移数据库...")
inboundService.MigrateDB() inboundService.MigrateDB()
fmt.Println("Migration done!") fmt.Println("")
fmt.Println("Migration done!------------>>迁移完成!")
}
func removeSecret() {
userService := service.UserService{}
secretExists, err := userService.CheckSecretExistence()
if err != nil {
fmt.Println("Error checking secret existence:", err)
return
}
if !secretExists {
fmt.Println("No secret exists to remove.")
return
}
err = userService.RemoveUserSecret()
if err != nil {
fmt.Println("Error removing secret:", err)
return
}
settingService := service.SettingService{}
err = settingService.SetSecretStatus(false)
if err != nil {
fmt.Println("Error updating secret status:", err)
return
}
fmt.Println("Secret removed successfully.")
} }
func main() { func main() {
@ -376,8 +408,6 @@ func main() {
var username string var username string
var password string var password string
var webBasePath string var webBasePath string
var listenIP string
var getListen bool
var webCertFile string var webCertFile string
var webKeyFile string var webKeyFile string
var tgbottoken string var tgbottoken string
@ -386,18 +416,14 @@ func main() {
var tgbotRuntime string var tgbotRuntime string
var reset bool var reset bool
var show bool var show bool
var getCert bool var remove_secret bool
var resetTwoFactor bool
settingCmd.BoolVar(&reset, "reset", false, "Reset all settings") settingCmd.BoolVar(&reset, "reset", false, "Reset all settings")
settingCmd.BoolVar(&show, "show", false, "Display current settings") settingCmd.BoolVar(&show, "show", false, "Display current settings")
settingCmd.BoolVar(&remove_secret, "remove_secret", false, "Remove secret key")
settingCmd.IntVar(&port, "port", 0, "Set panel port number") settingCmd.IntVar(&port, "port", 0, "Set panel port number")
settingCmd.StringVar(&username, "username", "", "Set login username") settingCmd.StringVar(&username, "username", "", "Set login username")
settingCmd.StringVar(&password, "password", "", "Set login password") settingCmd.StringVar(&password, "password", "", "Set login password")
settingCmd.StringVar(&webBasePath, "webBasePath", "", "Set base path for Panel") settingCmd.StringVar(&webBasePath, "webBasePath", "", "Set base path for Panel")
settingCmd.StringVar(&listenIP, "listenIP", "", "set panel listenIP IP")
settingCmd.BoolVar(&resetTwoFactor, "resetTwoFactor", false, "Reset two-factor authentication settings")
settingCmd.BoolVar(&getListen, "getListen", false, "Display current panel listenIP IP")
settingCmd.BoolVar(&getCert, "getCert", false, "Display current certificate settings")
settingCmd.StringVar(&webCertFile, "webCert", "", "Set path to public key file for panel") settingCmd.StringVar(&webCertFile, "webCert", "", "Set path to public key file for panel")
settingCmd.StringVar(&webKeyFile, "webCertKey", "", "Set path to private key file for panel") settingCmd.StringVar(&webKeyFile, "webCertKey", "", "Set path to private key file for panel")
settingCmd.StringVar(&tgbottoken, "tgbottoken", "", "Set token for Telegram bot") settingCmd.StringVar(&tgbottoken, "tgbottoken", "", "Set token for Telegram bot")
@ -440,20 +466,17 @@ func main() {
if reset { if reset {
resetSetting() resetSetting()
} else { } else {
updateSetting(port, username, password, webBasePath, listenIP, resetTwoFactor) updateSetting(port, username, password, webBasePath)
} }
if show { if show {
showSetting(show) showSetting(show)
} }
if getListen {
GetListenIP(getListen)
}
if getCert {
GetCertificate(getCert)
}
if (tgbottoken != "") || (tgbotchatid != "") || (tgbotRuntime != "") { if (tgbottoken != "") || (tgbotchatid != "") || (tgbotRuntime != "") {
updateTgbotSetting(tgbottoken, tgbotchatid, tgbotRuntime) updateTgbotSetting(tgbottoken, tgbotchatid, tgbotRuntime)
} }
if remove_secret {
removeSecret()
}
if enabletgbot { if enabletgbot {
updateTgbotEnableSts(enabletgbot) updateTgbotEnableSts(enabletgbot)
} }
@ -468,8 +491,9 @@ func main() {
} else { } else {
updateCert(webCertFile, webKeyFile) updateCert(webCertFile, webKeyFile)
} }
default: default:
fmt.Println("Invalid subcommands") fmt.Println("Invalid subcommands----->>无效命令")
fmt.Println() fmt.Println()
runCmd.Usage() runCmd.Usage()
fmt.Println() fmt.Println()