mirror of
https://github.com/bufanyun/hotgo.git
synced 2025-01-24 03:29:05 +08:00
157 lines
4.3 KiB
Go
157 lines
4.3 KiB
Go
// Package common
|
|
// @Link https://github.com/bufanyun/hotgo
|
|
// @Copyright Copyright (c) 2023 HotGo CLI
|
|
// @Author Ms <133814250@qq.com>
|
|
// @License https://github.com/bufanyun/hotgo/blob/master/LICENSE
|
|
package common
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"github.com/gogf/gf/v2/errors/gerror"
|
|
"github.com/gogf/gf/v2/frame/g"
|
|
"github.com/gogf/gf/v2/net/ghttp"
|
|
"github.com/gogf/gf/v2/os/gcron"
|
|
"github.com/gogf/gf/v2/os/gctx"
|
|
"github.com/gogf/gf/v2/os/gtime"
|
|
"github.com/gogf/gf/v2/util/gmeta"
|
|
"hotgo/api/admin/common"
|
|
"hotgo/internal/consts"
|
|
"hotgo/internal/library/cache"
|
|
"hotgo/internal/library/contexts"
|
|
"hotgo/internal/library/response"
|
|
"hotgo/internal/library/token"
|
|
"hotgo/internal/library/wechat"
|
|
"hotgo/internal/model/input/commonin"
|
|
"hotgo/internal/service"
|
|
"time"
|
|
)
|
|
|
|
type sCommonWechat struct {
|
|
temp map[string]*AuthorizeCallState
|
|
}
|
|
|
|
// AuthorizeCallState 微信授权回调参数
|
|
type AuthorizeCallState struct {
|
|
State string `json:"state" dc:"state"`
|
|
MemberId int64 `json:"memberId" dc:"管理员ID"`
|
|
Type string `json:"type" dc:"授权类型"`
|
|
SyncRedirect string `json:"syncRedirect" dc:"同步跳转地址"`
|
|
Tick *gtime.Time `json:"tick" dc:"标记授权时间"`
|
|
}
|
|
|
|
func NewCommonWechat() *sCommonWechat {
|
|
return &sCommonWechat{
|
|
temp: make(map[string]*AuthorizeCallState),
|
|
}
|
|
}
|
|
|
|
func init() {
|
|
serv := NewCommonWechat()
|
|
service.RegisterCommonWechat(serv)
|
|
_, _ = gcron.Add(gctx.New(), "@every 300s", serv.CleanTempMap, "WechatCleanTempMap")
|
|
}
|
|
|
|
// Authorize 微信用户授权
|
|
func (s *sCommonWechat) Authorize(ctx context.Context, in *commonin.WechatAuthorizeInp) (res *commonin.WechatAuthorizeModel, err error) {
|
|
basic, err := service.SysConfig().GetBasic(ctx)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
var (
|
|
request = g.RequestFromCtx(ctx)
|
|
prefix = g.Cfg().MustGet(ctx, "router.admin.prefix", "/admin").String()
|
|
path = gmeta.Get(common.WechatAuthorizeCallReq{}, "path").String()
|
|
redirectUri = basic.Domain + prefix + path
|
|
memberId = contexts.GetUserId(ctx)
|
|
state = s.GetCacheKey(in.Type, token.GetAuthKey(token.GetAuthorization(request)))
|
|
scope string
|
|
)
|
|
|
|
if memberId <= 0 {
|
|
err = gerror.New("获取用户信息失败!")
|
|
return
|
|
}
|
|
|
|
switch in.Type {
|
|
case consts.WechatAuthorizeOpenId: // 设置openid
|
|
scope = consts.WechatScopeBase
|
|
case consts.WechatAuthorizeBindLogin: // 绑定微信登录
|
|
scope = consts.WechatScopeUserinfo
|
|
default:
|
|
err = gerror.New("无效的授权方式!")
|
|
return
|
|
}
|
|
|
|
url, err := wechat.GetOauthURL(ctx, redirectUri, scope, state)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
s.temp[state] = &AuthorizeCallState{
|
|
State: state,
|
|
MemberId: memberId,
|
|
Type: in.Type,
|
|
SyncRedirect: in.SyncRedirect,
|
|
Tick: gtime.Now(),
|
|
}
|
|
response.Redirect(g.RequestFromCtx(ctx), url)
|
|
return
|
|
}
|
|
|
|
func (s *sCommonWechat) AuthorizeCall(ctx context.Context, in *commonin.WechatAuthorizeCallInp) (res *commonin.WechatAuthorizeCallModel, err error) {
|
|
data, ok := s.temp[in.State]
|
|
if !ok || data == nil {
|
|
err = gerror.New("授权无效或已过期,请重试")
|
|
return
|
|
}
|
|
|
|
defer delete(s.temp, in.State)
|
|
|
|
tk, err := wechat.GetUserAccessToken(ctx, in.Code)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
switch data.Type {
|
|
case consts.WechatAuthorizeOpenId: // 设置openid
|
|
_ = cache.Instance().Set(ctx, data.State, tk.OpenID, time.Hour*24*7)
|
|
case consts.WechatAuthorizeBindLogin: // 绑定微信登录
|
|
// ...
|
|
default:
|
|
err = gerror.New("无效的授权方式!")
|
|
return
|
|
}
|
|
|
|
response.Redirect(g.RequestFromCtx(ctx), data.SyncRedirect)
|
|
return
|
|
}
|
|
|
|
// GetOpenId 从缓存中获取临时openid
|
|
func (s *sCommonWechat) GetOpenId(ctx context.Context) (openId string, err error) {
|
|
request := ghttp.RequestFromCtx(ctx)
|
|
key := s.GetCacheKey(consts.WechatAuthorizeOpenId, token.GetAuthKey(token.GetAuthorization(request)))
|
|
date, err := cache.Instance().Get(ctx, key)
|
|
if err != nil {
|
|
err = gerror.Newf("GetOpenId err:%+v", err.Error())
|
|
return
|
|
}
|
|
openId = date.String()
|
|
return
|
|
}
|
|
|
|
func (s *sCommonWechat) GetCacheKey(typ, ak string) string {
|
|
return fmt.Sprintf("%v:%v", typ, ak)
|
|
}
|
|
|
|
// CleanTempMap 清理临时map
|
|
func (s *sCommonWechat) CleanTempMap(ctx context.Context) {
|
|
t := gtime.Now().Add(time.Second * 600)
|
|
for _, state := range s.temp {
|
|
if state.Tick.After(t) {
|
|
delete(s.temp, state.State)
|
|
}
|
|
}
|
|
}
|