微信小程序获取用户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

相关文章

Python实现微信自动锁定工具

《Python实现微信自动锁定工具》在数字化办公时代,微信已成为职场沟通的重要工具,但临时离开时忘记锁屏可能导致敏感信息泄露,下面我们就来看看如何使用Python打造一个微信自动锁定工具吧... 目录引言:当微信隐私遇到自动化守护效果展示核心功能全景图技术亮点深度解析1. 无操作检测引擎2. 微信路径智能获

利用Python打造一个Excel记账模板

《利用Python打造一个Excel记账模板》这篇文章主要为大家详细介绍了如何使用Python打造一个超实用的Excel记账模板,可以帮助大家高效管理财务,迈向财富自由之路,感兴趣的小伙伴快跟随小编一... 目录设置预算百分比超支标红预警记账模板功能介绍基础记账预算管理可视化分析摸鱼时间理财法碎片时间利用财

如何在 Spring Boot 中实现 FreeMarker 模板

《如何在SpringBoot中实现FreeMarker模板》FreeMarker是一种功能强大、轻量级的模板引擎,用于在Java应用中生成动态文本输出(如HTML、XML、邮件内容等),本文... 目录什么是 FreeMarker 模板?在 Spring Boot 中实现 FreeMarker 模板1. 环

SpringBoot整合mybatisPlus实现批量插入并获取ID详解

《SpringBoot整合mybatisPlus实现批量插入并获取ID详解》这篇文章主要为大家详细介绍了SpringBoot如何整合mybatisPlus实现批量插入并获取ID,文中的示例代码讲解详细... 目录【1】saveBATch(一万条数据总耗时:2478ms)【2】集合方式foreach(一万条数

python获取网页表格的多种方法汇总

《python获取网页表格的多种方法汇总》我们在网页上看到很多的表格,如果要获取里面的数据或者转化成其他格式,就需要将表格获取下来并进行整理,在Python中,获取网页表格的方法有多种,下面就跟随小编... 目录1. 使用Pandas的read_html2. 使用BeautifulSoup和pandas3.

SpringBoot UserAgentUtils获取用户浏览器的用法

《SpringBootUserAgentUtils获取用户浏览器的用法》UserAgentUtils是于处理用户代理(User-Agent)字符串的工具类,一般用于解析和处理浏览器、操作系统以及设备... 目录介绍效果图依赖封装客户端工具封装IP工具实体类获取设备信息入库介绍UserAgentUtils

Mysql用户授权(GRANT)语法及示例解读

《Mysql用户授权(GRANT)语法及示例解读》:本文主要介绍Mysql用户授权(GRANT)语法及示例,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录mysql用户授权(GRANT)语法授予用户权限语法GRANT语句中的<权限类型>的使用WITH GRANT

C# foreach 循环中获取索引的实现方式

《C#foreach循环中获取索引的实现方式》:本文主要介绍C#foreach循环中获取索引的实现方式,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录一、手动维护索引变量二、LINQ Select + 元组解构三、扩展方法封装索引四、使用 for 循环替代

Redis消息队列实现异步秒杀功能

《Redis消息队列实现异步秒杀功能》在高并发场景下,为了提高秒杀业务的性能,可将部分工作交给Redis处理,并通过异步方式执行,Redis提供了多种数据结构来实现消息队列,总结三种,本文详细介绍Re... 目录1 Redis消息队列1.1 List 结构1.2 Pub/Sub 模式1.3 Stream 结

将Java程序打包成EXE文件的实现方式

《将Java程序打包成EXE文件的实现方式》:本文主要介绍将Java程序打包成EXE文件的实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录如何将Java程序编程打包成EXE文件1.准备Java程序2.生成JAR包3.选择并安装打包工具4.配置Launch4