微信小程序获取用户openId并通过服务端向用户发送模板消息

2024-08-26 03:52

本文主要是介绍微信小程序获取用户openId并通过服务端向用户发送模板消息,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1.引言

注意:

1.标题中的服务端是自己研发的服务端,不是腾讯公司的服务端。

2.小程序的模板消息分为一次性订阅消息与长期订阅,一次性订阅就是每次在给用户发送消息之前都需要获得用户的同意(即用户订阅),长期性订阅是只需要用户同意一次,长期性订阅需要的小程序的分类为腾讯规定的服务种类(金融,公共服务,政务服务等),要求比较严格。本文所描述的为长期性订阅服务,默认用户已经订阅了此模板消息的通知。

3.此服务通知是代替短信的功能,是一个消息通知的形式,发短信是需要钱的,而服务通知不需要。

说说本篇文章诞生的业务环境,会议室预定系统,包含服务端,技术栈主要为Spring Boot+Spring Security+Spring AlibabaCloud+Nacos+MyBatis Plus+MySql;Web端,技术栈使用Vue3+Element UI;移动端,技术栈为uni-app+ts,用户通过用户名密码在Web端或者移动端登录进入系统后,进入会议室预定界面,填写会议室预定所需信息,保存后提交到服务端,服务端验证信息通过后,通过微信小程序的模板消息,调用腾讯开放的服务接口,通知对应的人员。

给对应人员发送服务通知,必须知道此人员在小程序中的唯一身份标识,如果是在小程序中,可以通过微信的wx.pluginLogin接口获取,如果在Web端中,如何通知了?

因为用户的openId在小程序中是唯一不变的,所以我们可以在用户使用小程序端输入用户名密码调用服务端登录接口时,获取其openId,并存储到用户信息表中,那么理论上就可以给系统中的任何用户发送模板消息的通知。所以,要给用户发送此订阅消息,用户必须登录过小程序,否则不能实现。

2.获取用户的openId并存储

在小程序中,通过wx.login获取code,此code连同用户名密码一起传递到服务端,服务端验证用户名密码后,调用小程序登录接口,拿到返回的openId并存储。

2.1 小程序端登录

小程序端登录先获取code,然后调用后端的登录接口。代码中区分了H5与小程序,H5是不需要调用腾讯的wx.login的。

1.小程序端关键代码
// 登录系统的form
const loginForm = reactive<WXProgramLoginReqVO>({username: '',password: '',clientId: import.meta.env.VITE_CLIENT_ID,code: '',
})const formRef = ref() // 表单
// 登录系统 一进系统就需要登录
const handleLogin = async () => {// #ifdef MP-WEIXINconst res = await wx.login()console.log(res.code)loginForm.code = res.code// #endif// 校验表单formRef.value.validate().then(async () => {let loginRes = {accessToken: '',refreshToken: '',}// #ifdef H5loginRes = await loginApi.login(loginForm)// #endif// #ifdef MP-WEIXINloginRes = await loginApi.wxProgramLogin(loginForm)// #endifsetAccessToken(loginRes.accessToken)setRefreshToken(loginRes.refreshToken)// 设置用户信息const userInfoRes = await permissionApi.getUserPermissionInfo()userStore.setUserInfo(userInfoRes)// 设置字典信息dictStore.setDictMap()// 直接跳转到首页uni.switchTab({ url: '/pages/index/index' })}).catch((err) => {console.log('登录表单错误信息:', err)})
}
/** 用户登录 h5 */
export const login = (data: LoginReqVO) => {return http.post({ url: '/auth/login', data })
}/** 用户登录 微信小程序 携带一个微信登录时返回的code */
export const wxProgramLogin = (data: WXProgramLoginReqVO) => {return http.post({ url: '/auth/wx-program/login', data })
}

代码中,H5与小程序端调用的登录接口为两个,其实可以简化为一个接口,后端通过有没有code的值来判断是不是需要调用腾讯的登录接口。

2.服务端关键代码

服务端接受到请求后,首先验证用户名密码是否正确,验证通过后,调用腾讯的小程序登录接口,获取对应的openId,并存储。

public LoginRespVO wxLogin(WXProgramLoginReqVO reqVO) {// 验证用户名密码UserResponseDTO user = authenticate(reqVO.getUsername(), reqVO.getPassword());if (user.getOpenId() == null){// 说明是第一次登录 需要更新其openId// 登录微信 获取 openidWxProgramLoginResDTO res = wxApi.wxProgramLogin(reqVO.getCode()).getCheckedData();String openId = res.getOpenid();// 更新用户的openIduserApi.updateUserOpenId(reqVO.getUsername(), openId);}//创建tokenreturn createTokenAfterLoginSuccess(user.getId(), user.getNickname(),reqVO.getUsername(),  reqVO.getClientId(), LoginLogTypeEnum.LOGIN_USERNAME);}private UserResponseDTO authenticate(String adminName, String password) {// 校验账号是否存在UserResponseDTO user = userApi.getUserByLoginInfo(adminName, password).getCheckedData();if (user == null) {throw exception(AUTH_LOGIN_BAD_CREDENTIALS);}return user;}
public WxProgramLoginResDTO wxProgramLogin(String code) {WxProgramLoginResVO wxProgramLoginRes = wxProgramWebClient.loginByCode(code).getCheckedData();WxProgramLoginResDTO dto = new WxProgramLoginResDTO();dto.setOpenid(wxProgramLoginRes.getOpenid());return dto;}
public BaseResponse<WxProgramLoginResVO> loginByCode(String code){String res = webClient.buildWebClient().get().uri(wxProgramIdentityProperties.getUrl() + "/sns/jscode2session"+ "?grant_type=authorization_code"+ "&appid=" + wxProgramIdentityProperties.getAppId()+ "&secret=" + wxProgramIdentityProperties.getSecret()+ "&js_code=" + code).header(HttpHeaders.CONTENT_TYPE, MediaType.APPLICATION_JSON_VALUE).exchangeToMono(response -> response.bodyToMono(String.class)).block();BaseResponse<WxProgramLoginResVO> stringBaseResponse = parseResponse(res, WxProgramLoginResVO.class);if (stringBaseResponse.isError()){log.error("登录失败:{}", stringBaseResponse.getMessage());}return stringBaseResponse;}

3.获取接口调用凭据

获取小程序全局唯一后台接口调用凭据,token有效期为7200s。在调用一切小程序端接口时,都需要使用此凭据,此凭据两小时之内都有效,两小时后需要重新申请。

暂时想法为,接口调用时,获取access_token,获取后判断是否过期,如果过期,重新申请access_token,并更新存储,返回最新的获取access_token。

public AccessTokenDO getAccessToken() {AccessTokenDO accessTokenDO = accessTokenService.getAccessTokenByCode(TOKEN_CODE);if (accessTokenDO == null){// 第一次 需要创建accessTokenDO = createToken();}// 判断accessToken是否过期 如果过期 需要更新if (DateUtils.isExpired(accessTokenDO.getExpireTime())) {WxProgramGetAccessTokenResVO accessTokenRes = wxProgramWebClient.getAccessToken().getCheckedData();accessTokenDO.setValue(accessTokenRes.getAccess_token());// 提前5分钟过期 然后刷新tokenaccessTokenDO.setExpireTime(LocalDateTime.now().plusSeconds(Integer.parseInt(accessTokenRes.getExpires_in()) - 300));}// 更新updateToken(accessTokenDO);return accessTokenDO;}
public BaseResponse<WxProgramGetAccessTokenResVO> getAccessToken(){String res = webClient.buildWebClient().get().uri(wxProgramIdentityProperties.getUrl() + "/cgi-bin/token"+ "?grant_type=client_credential"+ "&appid=" + wxProgramIdentityProperties.getAppId()+ "&secret=" + wxProgramIdentityProperties.getSecret()).header(HttpHeaders.CONTENT_TYPE, MediaType.APPLICATION_JSON_VALUE).exchangeToMono(response -> response.bodyToMono(String.class)).block();BaseResponse<WxProgramGetAccessTokenResVO> stringBaseResponse = parseResponse(res, WxProgramGetAccessTokenResVO.class);if (stringBaseResponse.isError()){log.error("获取token失败:{}", stringBaseResponse.getMessage());}return stringBaseResponse;}

4.发送订阅模板消息

首先在小程序基础功能中的订阅消息中申请一个模板,获取此模板的ID以及模板的详细内容。

根据模板内容,调用发送订阅消息接口,把对应的参数以及内容的body传递过去。

public void sendBookingMeetingMsg(WxSendBookingMeetingMsgReqDTO meetingMsgReqDTO) {WxProgramSendTemplateMsgReqVO reqVO = new WxProgramSendTemplateMsgReqVO();MessageTemplateDO messageTemplate = messageTemplateService.getMessageTemplateByCode(OFFICE_TEMPLATE_CODE);reqVO.setTemplate_id(messageTemplate.getTemplateId());reqVO.setPage(messageTemplate.getPage());reqVO.setTouser(meetingMsgReqDTO.getOpenid());reqVO.setMiniprogram_state("developer");reqVO.setLang("zh_CN");// 构造data// 会议内容JSONObject meetingName = new JSONObject();meetingName.putOpt("value", meetingMsgReqDTO.getMeetingName());// 会议室JSONObject meetingRoom = new JSONObject();meetingRoom.putOpt("value", meetingMsgReqDTO.getMeetingRoomName());// 会议时间JSONObject meetingTime = new JSONObject();meetingTime.putOpt("value", meetingMsgReqDTO.getMeetingTime());// 会议开始时间JSONObject meetingStartTime = new JSONObject();meetingStartTime.putOpt("value", meetingMsgReqDTO.getMeetingStartTime());// 会议申请人JSONObject meetingApplyUserName = new JSONObject();meetingApplyUserName.putOpt("value", meetingMsgReqDTO.getMeetingApplyUserName());JSONObject data = new JSONObject();data.putOpt("thing4", meetingName);data.putOpt("thing1", meetingRoom);data.putOpt("character_string2", meetingTime);data.putOpt("time6", meetingStartTime);data.putOpt("thing3", meetingApplyUserName);reqVO.setData(data);BaseResponse<WxProgramSendTemplateMsgResVO> templateMsgRes= wxProgramWebClient.sendTemplateMsg(getAccessToken().getValue(), reqVO);log.info(templateMsgRes.getCheckedData().toString());}
/*** 发送消息模板* @param accessToken 凭据* @param wxProgramSendTemplateMsgReqVO 消息内容* @return 是否成功*/public BaseResponse<WxProgramSendTemplateMsgResVO> sendTemplateMsg(String accessToken,WxProgramSendTemplateMsgReqVO wxProgramSendTemplateMsgReqVO){String res = webClient.buildWebClient().post().uri(wxProgramIdentityProperties.getUrl() + "/cgi-bin/message/subscribe/send"+ "?access_token=" + accessToken).bodyValue(wxProgramSendTemplateMsgReqVO).header(HttpHeaders.CONTENT_TYPE, MediaType.APPLICATION_JSON_VALUE).exchangeToMono(response -> response.bodyToMono(String.class)).block();BaseResponse<WxProgramSendTemplateMsgResVO> stringBaseResponse = parseResponse(res, WxProgramSendTemplateMsgResVO.class);if (stringBaseResponse.isError()){log.error("登录失败:{}", stringBaseResponse.getMessage());}return stringBaseResponse;}

5.错误调试

在调用发送订阅消息接口时,总不是一帆风顺的,期间遇到了很多错误,错误通过返回的错误码可以看出一个大概,错误具体如何造成的,可以调用对应的接口来获取具体的信息。

6.写在最后

本篇文章写了使用小程序的订阅模板消息,给对应用户发送服务通知。本人设想是利用此服务通知作为一个消息的承接载体,替代短信通知,节省成本。

本篇文章的代码只给了一些片段,可能有些地方看起来有些吃力,如果有需要解释或者有指教的地方,欢迎留言或者私信,感谢大家的支持。

这篇关于微信小程序获取用户openId并通过服务端向用户发送模板消息的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1107477

相关文章

Django中使用SMTP实现邮件发送功能

《Django中使用SMTP实现邮件发送功能》在Django中使用SMTP发送邮件是一个常见的需求,通常用于发送用户注册确认邮件、密码重置邮件等,下面我们来看看如何在Django中配置S... 目录1. 配置 Django 项目以使用 SMTP2. 创建 Django 应用3. 添加应用到项目设置4. 创建

python获取当前文件和目录路径的方法详解

《python获取当前文件和目录路径的方法详解》:本文主要介绍Python中获取当前文件路径和目录的方法,包括使用__file__关键字、os.path.abspath、os.path.realp... 目录1、获取当前文件路径2、获取当前文件所在目录3、os.path.abspath和os.path.re

SpringBoot 自定义消息转换器使用详解

《SpringBoot自定义消息转换器使用详解》本文详细介绍了SpringBoot消息转换器的知识,并通过案例操作演示了如何进行自定义消息转换器的定制开发和使用,感兴趣的朋友一起看看吧... 目录一、前言二、SpringBoot 内容协商介绍2.1 什么是内容协商2.2 内容协商机制深入理解2.2.1 内容

Java子线程无法获取Attributes的解决方法(最新推荐)

《Java子线程无法获取Attributes的解决方法(最新推荐)》在Java多线程编程中,子线程无法直接获取主线程设置的Attributes是一个常见问题,本文探讨了这一问题的原因,并提供了两种解决... 目录一、问题原因二、解决方案1. 直接传递数据2. 使用ThreadLocal(适用于线程独立数据)

SpringBoot实现websocket服务端及客户端的详细过程

《SpringBoot实现websocket服务端及客户端的详细过程》文章介绍了WebSocket通信过程、服务端和客户端的实现,以及可能遇到的问题及解决方案,感兴趣的朋友一起看看吧... 目录一、WebSocket通信过程二、服务端实现1.pom文件添加依赖2.启用Springboot对WebSocket

W外链微信推广短连接怎么做?

制作微信推广链接的难点分析 一、内容创作难度 制作微信推广链接时,首先需要创作有吸引力的内容。这不仅要求内容本身有趣、有价值,还要能够激起人们的分享欲望。对于许多企业和个人来说,尤其是那些缺乏创意和写作能力的人来说,这是制作微信推广链接的一大难点。 二、精准定位难度 微信用户群体庞大,不同用户的需求和兴趣各异。因此,制作推广链接时需要精准定位目标受众,以便更有效地吸引他们点击并分享链接

poj3468(线段树成段更新模板题)

题意:包括两个操作:1、将[a.b]上的数字加上v;2、查询区间[a,b]上的和 下面的介绍是下解题思路: 首先介绍  lazy-tag思想:用一个变量记录每一个线段树节点的变化值,当这部分线段的一致性被破坏我们就将这个变化值传递给子区间,大大增加了线段树的效率。 比如现在需要对[a,b]区间值进行加c操作,那么就从根节点[1,n]开始调用update函数进行操作,如果刚好执行到一个子节点,

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟&nbsp;开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚&nbsp;第一站:海量资源,应有尽有 走进“智听

poj 1258 Agri-Net(最小生成树模板代码)

感觉用这题来当模板更适合。 题意就是给你邻接矩阵求最小生成树啦。~ prim代码:效率很高。172k...0ms。 #include<stdio.h>#include<algorithm>using namespace std;const int MaxN = 101;const int INF = 0x3f3f3f3f;int g[MaxN][MaxN];int n