语言版本:go-1.17.3
import (
"bytes"
"crypto/aes"
"crypto/cipher"
"encoding/base64"
"encoding/json"
"my.test/server/global"
"my.test/server/model/modelSystem"
"io/ioutil"
"log"
"net/http"
"strings"
)
type UserService struct{}
// LoginByVx 使用vx授权的携带手机信息的code登陆
func (userService *UserService)LoginByVx(vxCode, iv, ecpData string)(ds *modelSystem.SysUser,err error){
sb := strings.Builder{}
sb.WriteString("https://api.weixin.qq.com/sns/jscode2session?")
sb.WriteString("appid="+global.GvaConfig.VX.AppId) // 小程序appid
sb.WriteString("&secret="+global.GvaConfig.VX.Secret) // 小程序加密值
sb.WriteString("&js_code="+vxCode)
sb.WriteString("&grant_type=authorization_code")
url := sb.String()
resp, err := http.Get(url)
if err != nil{
log.Println(err)
return
}
defer resp.Body.Close()
s, _ := ioutil.ReadAll(resp.Body)
res := make(map[string]string)
_ = json.Unmarshal(s, &res)
keyByte, _ := base64.StdEncoding.DecodeString(res["session_key"])
ivByte, _ := base64.StdEncoding.DecodeString(iv)
ciphertext, _ := base64.StdEncoding.DecodeString(ecpData)
plaintext := make([]byte, len(ciphertext))
block, err := aes.NewCipher(keyByte)
if err != nil {
log.Println(err)
return
}
mode := cipher.NewCBCDecrypter(block, ivByte)
mode.CryptBlocks(plaintext, ciphertext)
plaintext = PKCS7UnPadding(plaintext)
//plaintext = []byte("{\"phoneNumber\":\"13*97******\",\"purePhoneNumber\":\"13*97******\",\"countryCode\":\"86\",\"watermark\":{\"timestamp\":165692***,\"appid\":\"wx**********\"}}")
//log.Println("return:", string(plaintext))
var rbkMap map[string]string
_ = json.Unmarshal(plaintext,&rbkMap)
ds = &modelSystem.SysUser{}
ds.Phone = rbkMap["phoneNumber"]
return
}
// 两个工具函数
func PKCS7Padding(ciphertext []byte) []byte {
padding := aes.BlockSize - len(ciphertext)%aes.BlockSize
padtext := bytes.Repeat([]byte{byte(padding)}, padding)
return append(ciphertext, padtext...)
}
func PKCS7UnPadding(plantText []byte) []byte {
length := len(plantText)
unPadding := int(plantText[length-1])
return plantText[:(length - unPadding)]
}