224 lines
5.7 KiB
Go
224 lines
5.7 KiB
Go
package impl
|
|
|
|
import (
|
|
"encoding/gob"
|
|
"encoding/json"
|
|
"errors"
|
|
"github.com/gin-gonic/gin"
|
|
"github.com/mssola/useragent"
|
|
"gorm.io/gorm"
|
|
"schisandra-cloud-album/common/constant"
|
|
"schisandra-cloud-album/common/redis"
|
|
"schisandra-cloud-album/dao/impl"
|
|
"schisandra-cloud-album/global"
|
|
"schisandra-cloud-album/model"
|
|
"schisandra-cloud-album/utils"
|
|
"sync"
|
|
"time"
|
|
)
|
|
|
|
var userDao = impl.UserDaoImpl{}
|
|
|
|
type UserServiceImpl struct{}
|
|
|
|
var mu = &sync.Mutex{}
|
|
|
|
// ResponseData 返回数据
|
|
type ResponseData struct {
|
|
AccessToken string `json:"access_token"`
|
|
RefreshToken string `json:"refresh_token"`
|
|
ExpiresAt int64 `json:"expires_at"`
|
|
UID *string `json:"uid"`
|
|
}
|
|
|
|
func (res ResponseData) MarshalBinary() ([]byte, error) {
|
|
return json.Marshal(res)
|
|
}
|
|
|
|
func (res ResponseData) UnmarshalBinary(data []byte) error {
|
|
return json.Unmarshal(data, &res)
|
|
}
|
|
|
|
// GetUserListService 返回用户列表
|
|
func (UserServiceImpl) GetUserListService() []*model.ScaAuthUser {
|
|
return userDao.GetUserList()
|
|
}
|
|
|
|
// QueryUserByUsernameService 根据用户名查询用户
|
|
func (UserServiceImpl) QueryUserByUsernameService(username string) model.ScaAuthUser {
|
|
return userDao.QueryUserByUsername(username)
|
|
}
|
|
|
|
// QueryUserByUuidService 根据uid查询用户
|
|
func (UserServiceImpl) QueryUserByUuidService(uid *string) model.ScaAuthUser {
|
|
user, err := userDao.QueryUserByUuid(uid)
|
|
if err != nil {
|
|
return model.ScaAuthUser{}
|
|
}
|
|
return user
|
|
}
|
|
|
|
// DeleteUserService 根据uid删除用户
|
|
func (UserServiceImpl) DeleteUserService(uid string) error {
|
|
return userDao.DeleteUser(uid)
|
|
}
|
|
|
|
// QueryUserByPhoneService 根据手机号查询用户
|
|
func (UserServiceImpl) QueryUserByPhoneService(phone string) model.ScaAuthUser {
|
|
return userDao.QueryUserByPhone(phone)
|
|
}
|
|
|
|
// QueryUserByEmailService 根据邮箱查询用户
|
|
func (UserServiceImpl) QueryUserByEmailService(email string) model.ScaAuthUser {
|
|
return userDao.QueryUserByEmail(email)
|
|
}
|
|
|
|
// AddUserService 新增用户
|
|
func (UserServiceImpl) AddUserService(user model.ScaAuthUser) (*model.ScaAuthUser, error) {
|
|
return userDao.AddUser(user)
|
|
}
|
|
|
|
// UpdateUserService 更新用户信息
|
|
func (UserServiceImpl) UpdateUserService(phone, encrypt string) error {
|
|
return userDao.UpdateUser(phone, encrypt)
|
|
}
|
|
|
|
// RefreshTokenService 刷新用户token
|
|
func (UserServiceImpl) RefreshTokenService(refreshToken string) (*ResponseData, bool) {
|
|
parseRefreshToken, isUpd, err := utils.ParseRefreshToken(refreshToken)
|
|
if err != nil || !isUpd {
|
|
global.LOG.Errorln(err)
|
|
return nil, false
|
|
}
|
|
accessTokenString, err := utils.GenerateAccessToken(utils.AccessJWTPayload{UserID: parseRefreshToken.UserID})
|
|
if err != nil {
|
|
return nil, false
|
|
}
|
|
tokenKey := constant.UserLoginTokenRedisKey + *parseRefreshToken.UserID
|
|
token, err := redis.Get(tokenKey).Result()
|
|
if err != nil || token == "" {
|
|
global.LOG.Errorln(err)
|
|
return nil, false
|
|
}
|
|
data := ResponseData{
|
|
AccessToken: accessTokenString,
|
|
RefreshToken: refreshToken,
|
|
UID: parseRefreshToken.UserID,
|
|
}
|
|
if err = redis.Set(tokenKey, data, time.Hour*24*7).Err(); err != nil {
|
|
global.LOG.Errorln(err)
|
|
return nil, false
|
|
}
|
|
return &data, true
|
|
}
|
|
|
|
// HandelUserLogin 处理用户登录
|
|
func (UserServiceImpl) HandelUserLogin(user model.ScaAuthUser, autoLogin bool, c *gin.Context) (*ResponseData, bool) {
|
|
// 检查 user.UID 是否为 nil
|
|
if user.UID == nil {
|
|
return nil, false
|
|
}
|
|
if !GetUserLoginDevice(user, c) {
|
|
return nil, false
|
|
}
|
|
accessToken, err := utils.GenerateAccessToken(utils.AccessJWTPayload{UserID: user.UID})
|
|
if err != nil {
|
|
return nil, false
|
|
}
|
|
var days time.Duration
|
|
if autoLogin {
|
|
days = 7 * 24 * time.Hour
|
|
} else {
|
|
days = time.Minute * 30
|
|
}
|
|
|
|
refreshToken, expiresAt := utils.GenerateRefreshToken(utils.RefreshJWTPayload{UserID: user.UID}, days)
|
|
data := ResponseData{
|
|
AccessToken: accessToken,
|
|
RefreshToken: refreshToken,
|
|
ExpiresAt: expiresAt,
|
|
UID: user.UID,
|
|
}
|
|
|
|
err = redis.Set(constant.UserLoginTokenRedisKey+*user.UID, data, days).Err()
|
|
if err != nil {
|
|
return nil, false
|
|
}
|
|
gob.Register(ResponseData{})
|
|
err = utils.SetSession(c, constant.SessionKey, data)
|
|
if err != nil {
|
|
return nil, false
|
|
}
|
|
return &data, true
|
|
}
|
|
|
|
// GetUserLoginDevice 获取用户登录设备
|
|
func GetUserLoginDevice(user model.ScaAuthUser, c *gin.Context) bool {
|
|
|
|
// 检查user.UID是否为空
|
|
if user.UID == nil {
|
|
global.LOG.Errorln("user.UID is nil")
|
|
return false
|
|
}
|
|
userAgent := c.GetHeader("User-Agent")
|
|
if userAgent == "" {
|
|
global.LOG.Errorln("user-agent is empty")
|
|
return false
|
|
}
|
|
ua := useragent.New(userAgent)
|
|
|
|
ip := utils.GetClientIP(c)
|
|
location, err := global.IP2Location.SearchByStr(ip)
|
|
if err != nil {
|
|
global.LOG.Errorln(err)
|
|
return false
|
|
}
|
|
location = utils.RemoveZeroAndAdjust(location)
|
|
|
|
isBot := ua.Bot()
|
|
browser, browserVersion := ua.Browser()
|
|
os := ua.OS()
|
|
mobile := ua.Mobile()
|
|
mozilla := ua.Mozilla()
|
|
platform := ua.Platform()
|
|
engine, engineVersion := ua.Engine()
|
|
|
|
device := model.ScaAuthUserDevice{
|
|
UserID: user.UID,
|
|
IP: &ip,
|
|
Location: &location,
|
|
Agent: userAgent,
|
|
Browser: &browser,
|
|
BrowserVersion: &browserVersion,
|
|
OperatingSystem: &os,
|
|
Mobile: &mobile,
|
|
Bot: &isBot,
|
|
Mozilla: &mozilla,
|
|
Platform: &platform,
|
|
EngineName: &engine,
|
|
EngineVersion: &engineVersion,
|
|
}
|
|
|
|
mu.Lock()
|
|
defer mu.Unlock()
|
|
|
|
userDevice, err := userDeviceDao.GetUserDeviceByUIDIPAgent(*user.UID, ip, userAgent)
|
|
if err != nil && errors.Is(err, gorm.ErrRecordNotFound) {
|
|
err = userDeviceDao.AddUserDevice(&device)
|
|
if err != nil {
|
|
global.LOG.Errorln(err)
|
|
return false
|
|
}
|
|
} else if err != nil {
|
|
global.LOG.Errorln(err)
|
|
return false
|
|
} else {
|
|
err := userDeviceDao.UpdateUserDevice(userDevice.ID, &device)
|
|
if err != nil {
|
|
global.LOG.Errorln(err)
|
|
return false
|
|
}
|
|
}
|
|
return true
|
|
}
|