带参数的微信小程序码服务端(golang)生成

官方接口

微信小程序码共有3个接口可以生成:

  1. 获取小程序码
    POST https://api.weixin.qq.com/wxa/getwxacode?access_token=ACCESS_TOKEN
    该接口用于获取小程序码,适用于需要的码数量较少的业务场景。通过该接口生成的小程序码,永久有效,有数量限制

    path参数指定扫码进入的页面,最大长度 1024 字节,不能为空可以带参数page/example?foo=bar

  2. 获取不限制的小程序码
    POST https://api.weixin.qq.com/wxa/getwxacodeunlimit?access_token=ACCESS_TOKEN
    该接口用于获取小程序码,适用于需要的码数量极多的业务场景。通过该接口生成的小程序码,永久有效,数量暂无限制

    page参数指定扫码进入的页面,非必要参数,不可以带参数page/example

  3. 获取小程序二维码
    POST https://api.weixin.qq.com/cgi-bin/wxaapp/createwxaqrcode?access_token=ACCESS_TOKEN
    获取小程序二维码,适用于需要的码数量较少的业务场景。通过该接口生成的小程序码,永久有效,有数量限制

    path参数指定扫码进入的页面,最大长度 128 字节,不能为空,可以带参数page/example?foo=bar

以上3个接口之间的差异主要在于参数以及可以生成的数量, 接口1和3整体差异比较少(path参数长度不同; 小程序码和小程序二维码),接口2生成数量不受限制,但是指定打开的页面的参数page不能带参数(可以考虑通过scene参数传递,我目前还没验证)

具体的差异和参数可以通过链接查看官方文档

Golang实现

我根据自己的需要,选择了接口1的形式来实现,其他的接口只是接口url和参数的差异,整体流程和调用方式都是一样的。

获取ACCESS_TOKEN

func requestToken(appid, secret string) (string, error) {
	u, err := url.Parse("https://api.weixin.qq.com/cgi-bin/token")
	if err != nil {
		log.Fatal(err)
	}
	paras := &url.Values{}
	//设置请求参数
	paras.Set("appid", appid)
	paras.Set("secret", secret)
	paras.Set("grant_type", "client_credential")
	u.RawQuery = paras.Encode()
	resp, err := http.Get(u.String())
	//关闭资源
	if resp != nil && resp.Body != nil {
		defer resp.Body.Close()
	}
	if err != nil {
		return "", errors.New("request token err :" + err.Error())
	}

	jMap := make(map[string]interface{})
	err = json.NewDecoder(resp.Body).Decode(&jMap)
	if err != nil {
		return "", errors.New("request token response json parse err :" + err.Error())
	}
	if jMap["errcode"] == nil || jMap["errcode"] == 0 {
		accessToken, _ := jMap["access_token"].(string)
		return accessToken, nil
	} else {
		//返回错误信息
		errcode := jMap["errcode"].(string)
		errmsg := jMap["errmsg"].(string)
		err = errors.New(errcode + ":" + errmsg)
		return "", err
	}
}

获取小程序码

func GetQRCode(id string) ([]byte, error) {
	//上面生成的access code 判断为空时重新请求
	if appAccessTokenForClient == "" {
		accessToken, err := requestToken(appId, appSecret)
		if err != nil {
			return nil, err
		}
		appAccessTokenForClient = accessToken
	}
	strUrl := fmt.Sprintf("https://api.weixin.qq.com/wxa/getwxacode?access_token=%s", appAccessTokenForClient)

	parm := make(map[string]string)
	parm["path"] = fmt.Sprintf("pages/index/index?id=%s", id)
	jsonStr, err := json.Marshal(parm)
	if err != nil {
		return nil, errors.New("json Marshal QRCode paramter err :" + err.Error())
	}
	req, err := http.NewRequest("POST", strUrl, bytes.NewBuffer([]byte(jsonStr)))
	if err != nil {
		return nil, errors.New("get QRCode err :" + err.Error())
	}

	req.Header.Set("Content-Type", "application/json")
	client := &http.Client{}
	resp, err := client.Do(req)
	if err != nil {
		return nil, errors.New("get QRCode err :" + err.Error())
	}

	defer resp.Body.Close()
	body, err := io.ReadAll(resp.Body)
	if err != nil {
		return nil, errors.New("get QRCode err :" + err.Error())
	}

	return body, nil
}

常见问题

整个过程主要在调试时比较费时间,微信接口返回的错误信息比较范,记录下主要遇到的几种问题

40159

{\"errcode\":40159,\"errmsg\":\"invalid length for path, or the data is not json string hint: [Id2VBa0339xa11]\"}

主要是检查传递的参数,特别是path对应的长度

我这边是因为从接口2修改为接口1,但是page参数名没有修改为path,导致系统判断path为空,一直提示此错误

40169

errCode: 40169 errMsg: "openapi.wxacode.getUnlimited:fail invalid length for scene, or the data is not json string rid: 62764ad3-09f60c12-58046222"

接口2中scene参数,最大32个可见字符,只支持数字,大小写英文以及部分特殊字符:!#$&'()*+,/:;=?@-._~,其它字符请自行编码为合法字符(因不支持%,中文无法使用 urlencode 处理,请使用其他编码方式)

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Golang中,可以使用微信小程序提供的解密算法来获取用户的手机号。首先,你需要获取到用户的session_key和加密的手机号数据。然后,使用Golang的crypto/aes包进行解密操作。 下面是一个示例代,演示了如何在Golang获取微信小程序用户的手机号: ```go package main import ( "crypto/aes" "crypto/cipher" "encoding/base64" "encoding/json" "fmt" ) type PhoneNumberData struct { PhoneNumber string `json:"phoneNumber"` PurePhoneNumber string `json:"purePhoneNumber"` CountryCode string `json:"countryCode"` Watermark struct { AppID string `json:"appid"` Timestamp int64 `json:"timestamp"` } `json:"watermark"` } func main() { encryptedData := "加密后的手机号数据" sessionKey := "用户的session_key" // 对session_key进行base64解 sessionKeyBytes, err := base64.StdEncoding.DecodeString(sessionKey) if err != nil { fmt.Println("解session_key失败:", err) return } // 对加密的手机号数据进行base64解 encryptedDataBytes, err := base64.StdEncoding.DecodeString(encryptedData) if err != nil { fmt.Println("解加密的手机号数据失败:", err) return } // 创建AES解密器 block, err := aes.NewCipher(sessionKeyBytes) if err != nil { fmt.Println("创建AES解密器失败:", err) return } // 解密数据 iv := sessionKeyBytes[:aes.BlockSize] mode := cipher.NewCBCDecrypter(block, iv) decryptedData := make([]byte, len(encryptedDataBytes)) mode.CryptBlocks(decryptedData, encryptedDataBytes) // 去除解密后的数据的补位字符 padding := int(decryptedData[len(decryptedData)-1]) phoneNumberData := decryptedData[:len(decryptedData)-padding] // 解析JSON数据 var data PhoneNumberData err = json.Unmarshal(phoneNumberData, &data) if err != nil { fmt.Println("解析JSON数据失败:", err) return } // 打印手机号 fmt.Println("手机号:", data.PhoneNumber) } ``` 请注意,上述代中的`encryptedData`和`sessionKey`需要替换为实际的加密数据和用户的session_key。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值