mirror of
				https://github.com/TeaOSLab/EdgeAdmin.git
				synced 2025-11-04 13:10:26 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			170 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			170 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
package index
 | 
						|
 | 
						|
import (
 | 
						|
	"encoding/json"
 | 
						|
	"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/helpers"
 | 
						|
	"github.com/TeaOSLab/EdgeCommon/pkg/rpc/dao"
 | 
						|
	"github.com/TeaOSLab/EdgeCommon/pkg/rpc/pb"
 | 
						|
	"github.com/iwind/TeaGo/actions"
 | 
						|
	"github.com/iwind/TeaGo/maps"
 | 
						|
	"github.com/iwind/TeaGo/types"
 | 
						|
	stringutil "github.com/iwind/TeaGo/utils/string"
 | 
						|
	"github.com/xlzd/gotp"
 | 
						|
	"time"
 | 
						|
)
 | 
						|
 | 
						|
type IndexAction struct {
 | 
						|
	actionutils.ParentAction
 | 
						|
}
 | 
						|
 | 
						|
// 首页(登录页)
 | 
						|
 | 
						|
var TokenSalt = stringutil.Rand(32)
 | 
						|
 | 
						|
func (this *IndexAction) RunGet(params struct {
 | 
						|
	From string
 | 
						|
 | 
						|
	Auth *helpers.UserShouldAuth
 | 
						|
}) {
 | 
						|
	// DEMO模式
 | 
						|
	this.Data["isDemo"] = teaconst.IsDemoMode
 | 
						|
 | 
						|
	// 检查系统是否已经配置过
 | 
						|
	if !setup.IsConfigured() {
 | 
						|
		this.RedirectURL("/setup")
 | 
						|
		return
 | 
						|
	}
 | 
						|
 | 
						|
	// 已登录跳转到dashboard
 | 
						|
	if params.Auth.IsUser() {
 | 
						|
		this.RedirectURL("/dashboard")
 | 
						|
		return
 | 
						|
	}
 | 
						|
 | 
						|
	this.Data["isUser"] = false
 | 
						|
	this.Data["menu"] = "signIn"
 | 
						|
 | 
						|
	timestamp := fmt.Sprintf("%d", time.Now().Unix())
 | 
						|
	this.Data["token"] = stringutil.Md5(TokenSalt+timestamp) + timestamp
 | 
						|
	this.Data["from"] = params.From
 | 
						|
 | 
						|
	config, err := configloaders.LoadAdminUIConfig()
 | 
						|
	if err != nil {
 | 
						|
		this.ErrorPage(err)
 | 
						|
		return
 | 
						|
	}
 | 
						|
	this.Data["systemName"] = config.AdminSystemName
 | 
						|
	this.Data["showVersion"] = config.ShowVersion
 | 
						|
	if len(config.Version) > 0 {
 | 
						|
		this.Data["version"] = config.Version
 | 
						|
	} else {
 | 
						|
		this.Data["version"] = teaconst.Version
 | 
						|
	}
 | 
						|
	this.Data["faviconFileId"] = config.FaviconFileId
 | 
						|
 | 
						|
	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
 | 
						|
}) {
 | 
						|
	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("请通过登录页面登录")
 | 
						|
	}
 | 
						|
	timestampString := params.Token[32:]
 | 
						|
	if stringutil.Md5(TokenSalt+timestampString) != params.Token[:32] {
 | 
						|
		this.FailField("refresh", "登录页面已过期,请刷新后重试")
 | 
						|
	}
 | 
						|
	timestamp := types.Int64(timestampString)
 | 
						|
	if timestamp < time.Now().Unix()-1800 {
 | 
						|
		this.FailField("refresh", "登录页面已过期,请刷新后重试")
 | 
						|
	}
 | 
						|
 | 
						|
	rpcClient, err := rpc.SharedRPC()
 | 
						|
	if err != nil {
 | 
						|
		this.Fail("服务器出了点小问题:" + err.Error())
 | 
						|
	}
 | 
						|
	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, "登录时发生系统错误:"+err.Error(), this.RequestRemoteIP())
 | 
						|
		if err != nil {
 | 
						|
			utils.PrintError(err)
 | 
						|
		}
 | 
						|
 | 
						|
		actionutils.Fail(this, err)
 | 
						|
	}
 | 
						|
 | 
						|
	if !resp.IsOk {
 | 
						|
		err = dao.SharedLogDAO.CreateAdminLog(rpcClient.Context(0), oplogs.LevelWarn, this.Request.URL.Path, "登录失败,用户名:"+params.Username, this.RequestRemoteIP())
 | 
						|
		if err != nil {
 | 
						|
			utils.PrintError(err)
 | 
						|
		}
 | 
						|
 | 
						|
		this.Fail("请输入正确的用户名密码")
 | 
						|
	}
 | 
						|
 | 
						|
	// 检查OTP
 | 
						|
	otpLoginResp, err := this.RPC().LoginRPC().FindEnabledLogin(this.AdminContext(), &pb.FindEnabledLoginRequest{
 | 
						|
		AdminId: resp.AdminId,
 | 
						|
		Type:    "otp",
 | 
						|
	})
 | 
						|
	if err != nil {
 | 
						|
		this.ErrorPage(err)
 | 
						|
		return
 | 
						|
	}
 | 
						|
	if otpLoginResp.Login != nil && otpLoginResp.Login.IsOn {
 | 
						|
		loginParams := maps.Map{}
 | 
						|
		err = json.Unmarshal(otpLoginResp.Login.ParamsJSON, &loginParams)
 | 
						|
		if err != nil {
 | 
						|
			this.ErrorPage(err)
 | 
						|
			return
 | 
						|
		}
 | 
						|
		secret := loginParams.GetString("secret")
 | 
						|
		if gotp.NewDefaultTOTP(secret).Now() != params.OtpCode {
 | 
						|
			this.Fail("请输入正确的OTP动态密码")
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	adminId := resp.AdminId
 | 
						|
	params.Auth.StoreAdmin(adminId, params.Remember)
 | 
						|
 | 
						|
	// 记录日志
 | 
						|
	err = dao.SharedLogDAO.CreateAdminLog(rpcClient.Context(adminId), oplogs.LevelInfo, this.Request.URL.Path, "成功登录系统,用户名:"+params.Username, this.RequestRemoteIP())
 | 
						|
	if err != nil {
 | 
						|
		utils.PrintError(err)
 | 
						|
	}
 | 
						|
 | 
						|
	this.Success()
 | 
						|
}
 |