Files
BitSrunLoginGo/pkg/srun/srun.go
2024-01-01 17:56:06 +08:00

144 lines
3.0 KiB
Go

package srun
import (
"encoding/json"
"errors"
"fmt"
log "github.com/sirupsen/logrus"
"net/http"
)
type Conf struct {
//调用 API 时直接访问 https URL
Https bool
//登录参数,不可缺省
LoginInfo LoginInfo
Client *http.Client
CustomHeader map[string]interface{}
}
func New(conf *Conf) *Srun {
srun := &Srun{
LoginInfo: conf.LoginInfo,
}
srun.Api.Init(&ApiConfig{
Https: conf.Https,
Domain: conf.LoginInfo.Form.Domain,
Client: conf.Client,
CustomHeader: conf.CustomHeader,
})
return srun
}
type Srun struct {
//登录参数,不可缺省
LoginInfo LoginInfo
Api Api
}
func (c Srun) LoginStatus() (online bool, ip string, err error) {
res, err := c.Api.GetUserInfo()
if err != nil {
return false, "", err
}
errRes, ok := res["error"]
if !ok {
return false, "", ErrResultCannotFound
}
ipInterface, ok := res["client_ip"]
if !ok {
ipInterface, ok = res["online_ip"]
if !ok {
return false, "", ErrResultCannotFound
}
}
ip = ipInterface.(string)
online = errRes.(string) == "ok"
return
}
func (c Srun) DoLogin(clientIP string) error {
log.Debugln("正在获取 Token")
if c.LoginInfo.Form.UserType != "" {
c.LoginInfo.Form.Username += "@" + c.LoginInfo.Form.UserType
}
res, err := c.Api.GetChallenge(c.LoginInfo.Form.Username, clientIP)
if err != nil {
return err
}
token, ok := res["challenge"]
if !ok {
return ErrResultCannotFound
}
tokenStr := token.(string)
log.Debugln("token: ", tokenStr)
log.Debugln("发送登录请求")
info, err := json.Marshal(map[string]string{
"username": c.LoginInfo.Form.Username,
"password": c.LoginInfo.Form.Password,
"ip": clientIP,
"acid": c.LoginInfo.Meta.Acid,
"enc_ver": c.LoginInfo.Meta.Enc,
})
if err != nil {
return err
}
var infoPrefix string
if c.LoginInfo.Meta.InfoPrefix != "" {
infoPrefix = fmt.Sprintf("{%s}", c.LoginInfo.Meta.InfoPrefix)
}
EncryptedInfo := infoPrefix + Base64(XEncode(string(info), tokenStr))
Md5Str := Md5(tokenStr)
EncryptedMd5 := "{MD5}" + Md5Str
EncryptedChkstr := Sha1(
tokenStr + c.LoginInfo.Form.Username + tokenStr + Md5Str +
tokenStr + c.LoginInfo.Meta.Acid + tokenStr + clientIP +
tokenStr + c.LoginInfo.Meta.N + tokenStr + c.LoginInfo.Meta.Type +
tokenStr + EncryptedInfo,
)
var doubleStack string
if c.LoginInfo.Meta.DoubleStack {
doubleStack = "1"
} else {
doubleStack = "0"
}
res, err = c.Api.Login(&LoginRequest{
Username: c.LoginInfo.Form.Username,
Password: EncryptedMd5,
AcID: c.LoginInfo.Meta.Acid,
IP: clientIP,
Info: EncryptedInfo,
ChkSum: EncryptedChkstr,
N: c.LoginInfo.Meta.N,
Type: c.LoginInfo.Meta.Type,
OS: c.LoginInfo.Meta.OS,
Name: c.LoginInfo.Meta.Name,
DoubleStack: doubleStack,
})
if err != nil {
return err
}
var result interface{}
result, ok = res["error"]
if !ok {
return ErrResultCannotFound
}
LoginResult := result.(string)
if LoginResult != "ok" {
return errors.New(LoginResult)
}
return nil
}