mirror of
https://github.com/TeaOSLab/EdgeAdmin.git
synced 2025-11-03 20:40:26 +08:00
245 lines
6.8 KiB
Go
245 lines
6.8 KiB
Go
package index
|
|
|
|
import (
|
|
"fmt"
|
|
"github.com/TeaOSLab/EdgeAdmin/internal/configloaders"
|
|
teaconst "github.com/TeaOSLab/EdgeAdmin/internal/const"
|
|
"github.com/TeaOSLab/EdgeAdmin/internal/oplogs"
|
|
"github.com/TeaOSLab/EdgeAdmin/internal/rpc"
|
|
"github.com/TeaOSLab/EdgeAdmin/internal/setup"
|
|
"github.com/TeaOSLab/EdgeAdmin/internal/utils"
|
|
"github.com/TeaOSLab/EdgeAdmin/internal/web/actions/actionutils"
|
|
"github.com/TeaOSLab/EdgeAdmin/internal/web/actions/default/index/loginutils"
|
|
adminserverutils "github.com/TeaOSLab/EdgeAdmin/internal/web/actions/default/settings/server/admin-server-utils"
|
|
"github.com/TeaOSLab/EdgeAdmin/internal/web/helpers"
|
|
"github.com/TeaOSLab/EdgeCommon/pkg/configutils"
|
|
"github.com/TeaOSLab/EdgeCommon/pkg/iplibrary"
|
|
"github.com/TeaOSLab/EdgeCommon/pkg/langs"
|
|
"github.com/TeaOSLab/EdgeCommon/pkg/langs/codes"
|
|
"github.com/TeaOSLab/EdgeCommon/pkg/rpc/dao"
|
|
"github.com/TeaOSLab/EdgeCommon/pkg/rpc/pb"
|
|
"github.com/iwind/TeaGo/actions"
|
|
"github.com/iwind/TeaGo/lists"
|
|
"github.com/iwind/TeaGo/types"
|
|
stringutil "github.com/iwind/TeaGo/utils/string"
|
|
"net"
|
|
"time"
|
|
)
|
|
|
|
const regionDenyMessage = "当前软件系统暂时不为你所在的区域提供服务。"
|
|
|
|
type IndexAction struct {
|
|
actionutils.ParentAction
|
|
}
|
|
|
|
// 首页(登录页)
|
|
|
|
// TokenKey 加密用的密钥
|
|
var TokenKey = stringutil.Rand(32)
|
|
|
|
func (this *IndexAction) RunGet(params struct {
|
|
From string
|
|
|
|
Auth *helpers.UserShouldAuth
|
|
}) {
|
|
if !this.checkRegion() {
|
|
this.WriteString(regionDenyMessage)
|
|
return
|
|
}
|
|
|
|
// 是否自动从HTTP跳转到HTTPS
|
|
if this.Request.TLS == nil {
|
|
httpsPort, _ := adminserverutils.ReadServerHTTPS()
|
|
if httpsPort > 0 {
|
|
currentHost, _, err := net.SplitHostPort(this.Request.Host)
|
|
if err != nil {
|
|
currentHost = this.Request.Host
|
|
}
|
|
|
|
var newHost = configutils.QuoteIP(currentHost)
|
|
if httpsPort != 443 /** default https port **/ {
|
|
newHost += ":" + types.String(httpsPort)
|
|
}
|
|
|
|
// 如果没有前端反向代理,则跳转
|
|
if len(this.Request.Header.Get("X-Forwarded-For")) == 0 && len(this.Request.Header.Get("X-Real-Ip")) == 0 {
|
|
this.RedirectURL("https://" + newHost + this.Request.RequestURI)
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
// DEMO模式
|
|
this.Data["isDemo"] = teaconst.IsDemoMode
|
|
|
|
// 检查系统是否已经配置过
|
|
if !setup.IsConfigured() {
|
|
this.RedirectURL("/setup")
|
|
return
|
|
}
|
|
|
|
//// 是否新安装
|
|
if setup.IsNewInstalled() {
|
|
this.RedirectURL("/setup/confirm")
|
|
return
|
|
}
|
|
|
|
// 已登录跳转到dashboard
|
|
if params.Auth.IsUser() {
|
|
this.RedirectURL("/dashboard")
|
|
return
|
|
}
|
|
|
|
this.Data["isUser"] = false
|
|
this.Data["menu"] = "signIn"
|
|
|
|
var timestamp = fmt.Sprintf("%d", time.Now().Unix())
|
|
this.Data["token"] = stringutil.Md5(TokenKey+timestamp) + timestamp
|
|
this.Data["from"] = params.From
|
|
|
|
uiConfig, err := configloaders.LoadAdminUIConfig()
|
|
if err != nil {
|
|
this.ErrorPage(err)
|
|
return
|
|
}
|
|
this.Data["systemName"] = uiConfig.AdminSystemName
|
|
this.Data["showVersion"] = uiConfig.ShowVersion
|
|
if len(uiConfig.Version) > 0 {
|
|
this.Data["version"] = uiConfig.Version
|
|
} else {
|
|
this.Data["version"] = teaconst.Version
|
|
}
|
|
this.Data["faviconFileId"] = uiConfig.FaviconFileId
|
|
|
|
securityConfig, err := configloaders.LoadSecurityConfig()
|
|
if err != nil {
|
|
this.Data["rememberLogin"] = false
|
|
} else {
|
|
this.Data["rememberLogin"] = securityConfig.AllowRememberLogin
|
|
}
|
|
|
|
// 删除Cookie
|
|
loginutils.UnsetCookie(this.Object())
|
|
|
|
this.Show()
|
|
}
|
|
|
|
// RunPost 提交
|
|
func (this *IndexAction) RunPost(params struct {
|
|
Token string
|
|
Username string
|
|
Password string
|
|
OtpCode string
|
|
Remember bool
|
|
|
|
Must *actions.Must
|
|
Auth *helpers.UserShouldAuth
|
|
CSRF *actionutils.CSRF
|
|
}) {
|
|
if !this.checkRegion() {
|
|
this.Fail(regionDenyMessage)
|
|
return
|
|
}
|
|
|
|
params.Must.
|
|
Field("username", params.Username).
|
|
Require("请输入用户名").
|
|
Field("password", params.Password).
|
|
Require("请输入密码")
|
|
|
|
if params.Password == stringutil.Md5("") {
|
|
this.FailField("password", "请输入密码")
|
|
}
|
|
|
|
// 检查token
|
|
if len(params.Token) <= 32 {
|
|
this.Fail("请通过登录页面登录")
|
|
}
|
|
var timestampString = params.Token[32:]
|
|
if stringutil.Md5(TokenKey+timestampString) != params.Token[:32] {
|
|
this.FailField("refresh", "登录页面已过期,请刷新后重试")
|
|
}
|
|
var timestamp = types.Int64(timestampString)
|
|
if timestamp < time.Now().Unix()-1800 {
|
|
this.FailField("refresh", "登录页面已过期,请刷新后重试")
|
|
}
|
|
|
|
rpcClient, err := rpc.SharedRPC()
|
|
if err != nil {
|
|
this.Fail("服务器出了点小问题:" + err.Error())
|
|
return
|
|
}
|
|
resp, err := rpcClient.AdminRPC().LoginAdmin(rpcClient.Context(0), &pb.LoginAdminRequest{
|
|
Username: params.Username,
|
|
Password: params.Password,
|
|
})
|
|
|
|
if err != nil {
|
|
err = dao.SharedLogDAO.CreateAdminLog(rpcClient.Context(0), oplogs.LevelError, this.Request.URL.Path, langs.DefaultMessage(codes.AdminLogin_LogSystemError, err.Error()), loginutils.RemoteIP(&this.ActionObject), codes.AdminLogin_LogSystemError, []any{err.Error()})
|
|
if err != nil {
|
|
utils.PrintError(err)
|
|
}
|
|
|
|
actionutils.Fail(this, err)
|
|
return
|
|
}
|
|
|
|
if !resp.IsOk {
|
|
err = dao.SharedLogDAO.CreateAdminLog(rpcClient.Context(0), oplogs.LevelWarn, this.Request.URL.Path, langs.DefaultMessage(codes.AdminLogin_LogFailed, params.Username), loginutils.RemoteIP(&this.ActionObject), codes.AdminLogin_LogFailed, []any{params.Username})
|
|
if err != nil {
|
|
utils.PrintError(err)
|
|
}
|
|
|
|
this.Fail("请输入正确的用户名密码")
|
|
return
|
|
}
|
|
var adminId = resp.AdminId
|
|
|
|
// 检查是否支持OTP
|
|
checkOTPResp, err := this.RPC().AdminRPC().CheckAdminOTPWithUsername(this.AdminContext(), &pb.CheckAdminOTPWithUsernameRequest{Username: params.Username})
|
|
if err != nil {
|
|
this.ErrorPage(err)
|
|
return
|
|
}
|
|
var requireOTP = checkOTPResp.RequireOTP
|
|
this.Data["requireOTP"] = requireOTP
|
|
if requireOTP {
|
|
this.Data["remember"] = params.Remember
|
|
|
|
var sid = this.Session().Sid
|
|
this.Data["sid"] = sid
|
|
_, err = this.RPC().LoginSessionRPC().WriteLoginSessionValue(this.AdminContext(), &pb.WriteLoginSessionValueRequest{
|
|
Sid: sid + "_otp",
|
|
Key: "adminId",
|
|
Value: types.String(adminId),
|
|
})
|
|
if err != nil {
|
|
this.ErrorPage(err)
|
|
return
|
|
}
|
|
this.Success()
|
|
return
|
|
}
|
|
|
|
// 写入SESSION
|
|
params.Auth.StoreAdmin(adminId, params.Remember)
|
|
|
|
// 记录日志
|
|
err = dao.SharedLogDAO.CreateAdminLog(rpcClient.Context(adminId), oplogs.LevelInfo, this.Request.URL.Path, langs.DefaultMessage(codes.AdminLogin_LogSuccess, params.Username), loginutils.RemoteIP(&this.ActionObject), codes.AdminLogin_LogSuccess, []any{params.Username})
|
|
if err != nil {
|
|
utils.PrintError(err)
|
|
}
|
|
|
|
this.Success()
|
|
}
|
|
|
|
// 检查登录区域
|
|
func (this *IndexAction) checkRegion() bool {
|
|
var ip = loginutils.RemoteIP(&this.ActionObject)
|
|
var result = iplibrary.LookupIP(ip)
|
|
if result != nil && result.IsOk() && result.CountryId() > 0 && lists.ContainsInt64([]int64{9, 10}, result.CountryId()) {
|
|
return false
|
|
}
|
|
return true
|
|
}
|