mirror of
https://github.com/silenceper/wechat.git
synced 2026-02-04 21:02:25 +08:00
308 lines
10 KiB
Go
308 lines
10 KiB
Go
package auth
|
||
|
||
import (
|
||
context2 "context"
|
||
"encoding/json"
|
||
"fmt"
|
||
|
||
"github.com/silenceper/wechat/v2/miniprogram/context"
|
||
"github.com/silenceper/wechat/v2/util"
|
||
)
|
||
|
||
const (
|
||
// code2SessionURL 小程序登录
|
||
code2SessionURL = "https://api.weixin.qq.com/sns/jscode2session?appid=%s&secret=%s&js_code=%s&grant_type=authorization_code"
|
||
// checkEncryptedDataURL 检查加密信息
|
||
checkEncryptedDataURL = "https://api.weixin.qq.com/wxa/business/checkencryptedmsg?access_token=%s"
|
||
// getPhoneNumber 获取手机号
|
||
getPhoneNumber = "https://api.weixin.qq.com/wxa/business/getuserphonenumber?access_token=%s"
|
||
// checkSessionURL 检验登录态
|
||
checkSessionURL = "https://api.weixin.qq.com/wxa/checksession?access_token=%s&signature=%s&openid=%s&sig_method=hmac_sha256"
|
||
// resetUserSessionKeyURL 重置登录态
|
||
resetUserSessionKeyURL = "https://api.weixin.qq.com/wxa/resetusersessionkey?access_token=%s&signature=%s&openid=%s&sig_method=hmac_sha256"
|
||
// getPluginOpenPIDURL 获取插件用户openPID
|
||
getPluginOpenPIDURL = "https://api.weixin.qq.com/wxa/getpluginopenpid?access_token=%s"
|
||
// getPaidUnionIDURL 支付后获取 UnionID
|
||
getPaidUnionIDURL = "https://api.weixin.qq.com/wxa/getpaidunionid"
|
||
// getUserEncryptKeyURL 获取用户encryptKey
|
||
getUserEncryptKeyURL = "https://api.weixin.qq.com/wxa/business/getuserencryptkey?access_token=%s&signature=%s&openid=%s&sig_method=hmac_sha256"
|
||
)
|
||
|
||
// Auth 登录/用户信息
|
||
type Auth struct {
|
||
*context.Context
|
||
}
|
||
|
||
// NewAuth new auth
|
||
func NewAuth(ctx *context.Context) *Auth {
|
||
return &Auth{ctx}
|
||
}
|
||
|
||
// ResCode2Session 登录凭证校验的返回结果
|
||
type ResCode2Session struct {
|
||
util.CommonError
|
||
|
||
OpenID string `json:"openid"` // 用户唯一标识
|
||
SessionKey string `json:"session_key"` // 会话密钥
|
||
UnionID string `json:"unionid"` // 用户在开放平台的唯一标识符,在满足UnionID下发条件的情况下会返回
|
||
}
|
||
|
||
// RspCheckEncryptedData .
|
||
type RspCheckEncryptedData struct {
|
||
util.CommonError
|
||
|
||
Vaild bool `json:"vaild"` // 是否是合法的数据
|
||
CreateTime uint64 `json:"create_time"` // 加密数据生成的时间戳
|
||
}
|
||
|
||
// Code2Session 登录凭证校验。
|
||
func (auth *Auth) Code2Session(jsCode string) (result ResCode2Session, err error) {
|
||
return auth.Code2SessionContext(context2.Background(), jsCode)
|
||
}
|
||
|
||
// Code2SessionContext 登录凭证校验。
|
||
func (auth *Auth) Code2SessionContext(ctx context2.Context, jsCode string) (result ResCode2Session, err error) {
|
||
var response []byte
|
||
if response, err = util.HTTPGetContext(ctx, fmt.Sprintf(code2SessionURL, auth.AppID, auth.AppSecret, jsCode)); err != nil {
|
||
return
|
||
}
|
||
if err = json.Unmarshal(response, &result); err != nil {
|
||
return
|
||
}
|
||
if result.ErrCode != 0 {
|
||
err = fmt.Errorf("Code2Session error : errcode=%v , errmsg=%v", result.ErrCode, result.ErrMsg)
|
||
return
|
||
}
|
||
return
|
||
}
|
||
|
||
type (
|
||
// GetPaidUnionIDRequest 支付后获取UnionID请求
|
||
GetPaidUnionIDRequest struct {
|
||
OpenID string `json:"openid"`
|
||
TransactionID string `json:"transaction_id,omitempty"`
|
||
MchID string `json:"mch_id,omitempty"`
|
||
OutTradeNo string `json:"out_trade_no,omitempty"`
|
||
}
|
||
|
||
// GetPaidUnionIDResponse 支付后获取UnionID响应
|
||
GetPaidUnionIDResponse struct {
|
||
util.CommonError
|
||
UnionID string `json:"unionid"`
|
||
}
|
||
)
|
||
|
||
// GetPaidUnionID 用户支付完成后,获取该用户的 UnionId,无需用户授权
|
||
// see https://developers.weixin.qq.com/miniprogram/dev/OpenApiDoc/user-info/basic-info/getPaidUnionid.html
|
||
func (auth *Auth) GetPaidUnionID(req *GetPaidUnionIDRequest) (string, error) {
|
||
var (
|
||
accessToken string
|
||
err error
|
||
)
|
||
if accessToken, err = auth.GetAccessToken(); err != nil {
|
||
return "", err
|
||
}
|
||
var url string
|
||
if req.TransactionID != "" {
|
||
url = fmt.Sprintf("%s?access_token=%s&openid=%s&transaction_id=%s", getPaidUnionIDURL, accessToken, req.OpenID, req.TransactionID)
|
||
} else {
|
||
url = fmt.Sprintf("%s?access_token=%s&openid=%s&mch_id=%s&out_trade_no=%s", getPaidUnionIDURL, accessToken, req.OpenID, req.MchID, req.OutTradeNo)
|
||
}
|
||
var response []byte
|
||
if response, err = util.HTTPGet(url); err != nil {
|
||
return "", err
|
||
}
|
||
result := &GetPaidUnionIDResponse{}
|
||
err = util.DecodeWithError(response, result, "GetPaidUnionID")
|
||
return result.UnionID, err
|
||
}
|
||
|
||
// CheckEncryptedData .检查加密信息是否由微信生成(当前只支持手机号加密数据),只能检测最近3天生成的加密数据
|
||
func (auth *Auth) CheckEncryptedData(encryptedMsgHash string) (result RspCheckEncryptedData, err error) {
|
||
return auth.CheckEncryptedDataContext(context2.Background(), encryptedMsgHash)
|
||
}
|
||
|
||
// CheckEncryptedDataContext .检查加密信息是否由微信生成(当前只支持手机号加密数据),只能检测最近3天生成的加密数据
|
||
func (auth *Auth) CheckEncryptedDataContext(ctx context2.Context, encryptedMsgHash string) (result RspCheckEncryptedData, err error) {
|
||
var response []byte
|
||
var (
|
||
at string
|
||
)
|
||
if at, err = auth.GetAccessTokenContext(ctx); err != nil {
|
||
return
|
||
}
|
||
|
||
// 由于GetPhoneNumberContext需要传入JSON,所以HTTPPostContext入参改为[]byte
|
||
if response, err = util.HTTPPostContext(ctx, fmt.Sprintf(checkEncryptedDataURL, at), []byte("encrypted_msg_hash="+encryptedMsgHash), nil); err != nil {
|
||
return
|
||
}
|
||
if err = util.DecodeWithError(response, &result, "CheckEncryptedDataAuth"); err != nil {
|
||
return
|
||
}
|
||
return
|
||
}
|
||
|
||
// GetPhoneNumberResponse 新版获取用户手机号响应结构体
|
||
type GetPhoneNumberResponse struct {
|
||
util.CommonError
|
||
|
||
PhoneInfo PhoneInfo `json:"phone_info"`
|
||
}
|
||
|
||
// PhoneInfo 获取用户手机号内容
|
||
type PhoneInfo struct {
|
||
PhoneNumber string `json:"phoneNumber"` // 用户绑定的手机号
|
||
PurePhoneNumber string `json:"purePhoneNumber"` // 没有区号的手机号
|
||
CountryCode string `json:"countryCode"` // 区号
|
||
WaterMark struct {
|
||
Timestamp int64 `json:"timestamp"`
|
||
AppID string `json:"appid"`
|
||
} `json:"watermark"` // 数据水印
|
||
}
|
||
|
||
// GetPhoneNumberContext 小程序通过code获取用户手机号
|
||
func (auth *Auth) GetPhoneNumberContext(ctx context2.Context, code string) (*GetPhoneNumberResponse, error) {
|
||
var response []byte
|
||
var (
|
||
at string
|
||
err error
|
||
)
|
||
if at, err = auth.GetAccessTokenContext(ctx); err != nil {
|
||
return nil, err
|
||
}
|
||
body := map[string]interface{}{
|
||
"code": code,
|
||
}
|
||
|
||
bodyBytes, err := json.Marshal(body)
|
||
if err != nil {
|
||
return nil, err
|
||
}
|
||
|
||
header := map[string]string{"Content-Type": "application/json;charset=utf-8"}
|
||
if response, err = util.HTTPPostContext(ctx, fmt.Sprintf(getPhoneNumber, at), bodyBytes, header); err != nil {
|
||
return nil, err
|
||
}
|
||
|
||
var result GetPhoneNumberResponse
|
||
err = util.DecodeWithError(response, &result, "phonenumber.getPhoneNumber")
|
||
return &result, err
|
||
}
|
||
|
||
// GetPhoneNumber 小程序通过code获取用户手机号
|
||
func (auth *Auth) GetPhoneNumber(code string) (*GetPhoneNumberResponse, error) {
|
||
return auth.GetPhoneNumberContext(context2.Background(), code)
|
||
}
|
||
|
||
// CheckSession 检验登录态
|
||
// see https://developers.weixin.qq.com/miniprogram/dev/OpenApiDoc/user-login/checkSessionKey.html
|
||
func (auth *Auth) CheckSession(signature, openID string) error {
|
||
var (
|
||
accessToken string
|
||
err error
|
||
)
|
||
if accessToken, err = auth.GetAccessToken(); err != nil {
|
||
return err
|
||
}
|
||
var response []byte
|
||
if response, err = util.HTTPGet(fmt.Sprintf(checkSessionURL, accessToken, signature, openID)); err != nil {
|
||
return err
|
||
}
|
||
return util.DecodeWithCommonError(response, "CheckSession")
|
||
}
|
||
|
||
// ResetUserSessionKeyResponse 重置登录态响应
|
||
type ResetUserSessionKeyResponse struct {
|
||
util.CommonError
|
||
OpenID string `json:"openid"`
|
||
SessionKey string `json:"session_key"`
|
||
}
|
||
|
||
// ResetUserSessionKey 重置登录态
|
||
// see https://developers.weixin.qq.com/miniprogram/dev/OpenApiDoc/user-login/ResetUserSessionKey.html
|
||
func (auth *Auth) ResetUserSessionKey(signature, openID string) (*ResetUserSessionKeyResponse, error) {
|
||
var (
|
||
accessToken string
|
||
err error
|
||
)
|
||
if accessToken, err = auth.GetAccessToken(); err != nil {
|
||
return nil, err
|
||
}
|
||
var response []byte
|
||
if response, err = util.HTTPGet(fmt.Sprintf(resetUserSessionKeyURL, accessToken, signature, openID)); err != nil {
|
||
return nil, err
|
||
}
|
||
result := &ResetUserSessionKeyResponse{}
|
||
err = util.DecodeWithError(response, result, "ResetUserSessionKey")
|
||
return result, err
|
||
}
|
||
|
||
type (
|
||
// GetPluginOpenPIDRequest 获取插件用户openPID请求
|
||
GetPluginOpenPIDRequest struct {
|
||
Code string `json:"code"`
|
||
}
|
||
|
||
// GetPluginOpenPIDResponse 获取插件用户openPID响应
|
||
GetPluginOpenPIDResponse struct {
|
||
util.CommonError
|
||
OpenPID string `json:"openpid"`
|
||
}
|
||
)
|
||
|
||
// GetPluginOpenPID 获取插件用户openPID
|
||
// see https://developers.weixin.qq.com/miniprogram/dev/OpenApiDoc/user-info/basic-info/getPluginOpenPId.html
|
||
func (auth *Auth) GetPluginOpenPID(code string) (string, error) {
|
||
var (
|
||
accessToken string
|
||
err error
|
||
)
|
||
if accessToken, err = auth.GetAccessToken(); err != nil {
|
||
return "", err
|
||
}
|
||
req := &GetPluginOpenPIDRequest{
|
||
Code: code,
|
||
}
|
||
var response []byte
|
||
if response, err = util.PostJSON(fmt.Sprintf(getPluginOpenPIDURL, accessToken), req); err != nil {
|
||
return "", err
|
||
}
|
||
result := &GetPluginOpenPIDResponse{}
|
||
err = util.DecodeWithError(response, result, "GetPluginOpenPID")
|
||
return result.OpenPID, err
|
||
}
|
||
|
||
// GetUserEncryptKeyResponse 获取用户encryptKey响应
|
||
type GetUserEncryptKeyResponse struct {
|
||
util.CommonError
|
||
KeyInfoList []KeyInfo `json:"key_info_list"`
|
||
}
|
||
|
||
// KeyInfo 用户最近三次的加密key
|
||
type KeyInfo struct {
|
||
EncryptKey string `json:"encrypt_key"`
|
||
Version int64 `json:"version"`
|
||
ExpireIn int64 `json:"expire_in"`
|
||
Iv string `json:"iv"`
|
||
CreateTime int64 `json:"create_time"`
|
||
}
|
||
|
||
// GetUserEncryptKey 获取用户encryptKey
|
||
// see https://developers.weixin.qq.com/miniprogram/dev/OpenApiDoc/user-info/internet/getUserEncryptKey.html
|
||
func (auth *Auth) GetUserEncryptKey(signature, openID string) (*GetUserEncryptKeyResponse, error) {
|
||
var (
|
||
accessToken string
|
||
err error
|
||
)
|
||
if accessToken, err = auth.GetAccessToken(); err != nil {
|
||
return nil, err
|
||
}
|
||
var response []byte
|
||
if response, err = util.HTTPGet(fmt.Sprintf(getUserEncryptKeyURL, accessToken, signature, openID)); err != nil {
|
||
return nil, err
|
||
}
|
||
result := &GetUserEncryptKeyResponse{}
|
||
err = util.DecodeWithError(response, result, "GetUserEncryptKey")
|
||
return result, err
|
||
}
|