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

相关文章

C#如何在Excel文档中获取分页信息

《C#如何在Excel文档中获取分页信息》在日常工作中,我们经常需要处理大量的Excel数据,本文将深入探讨如何利用Spire.XLSfor.NET,高效准确地获取Excel文档中的分页信息,包括水平... 目录理解Excel中的分页机制借助 Spire.XLS for .NET 获取分页信息为什么选择 S

springboot3.x使用@NacosValue无法获取配置信息的解决过程

《springboot3.x使用@NacosValue无法获取配置信息的解决过程》在SpringBoot3.x中升级Nacos依赖后,使用@NacosValue无法动态获取配置,通过引入SpringC... 目录一、python问题描述二、解决方案总结一、问题描述springboot从2android.x

springboot的controller中如何获取applicatim.yml的配置值

《springboot的controller中如何获取applicatim.yml的配置值》本文介绍了在SpringBoot的Controller中获取application.yml配置值的四种方式,... 目录1. 使用@Value注解(最常用)application.yml 配置Controller 中

Java利用Spire.Doc for Java实现在模板的基础上创建Word文档

《Java利用Spire.DocforJava实现在模板的基础上创建Word文档》在日常开发中,我们经常需要根据特定数据动态生成Word文档,本文将深入探讨如何利用强大的Java库Spire.Do... 目录1. Spire.Doc for Java 库介绍与安装特点与优势Maven 依赖配置2. 通过替换

SpringBoot+Vue3整合SSE实现实时消息推送功能

《SpringBoot+Vue3整合SSE实现实时消息推送功能》在日常开发中,我们经常需要实现实时消息推送的功能,这篇文章将基于SpringBoot和Vue3来简单实现一个入门级的例子,下面小编就和大... 目录前言先大概介绍下SSE后端实现(SpringBoot)前端实现(vue3)1. 数据类型定义2.

golang实现nacos获取配置和服务注册-支持集群详解

《golang实现nacos获取配置和服务注册-支持集群详解》文章介绍了如何在Go语言中使用Nacos获取配置和服务注册,支持集群初始化,客户端结构体中的IpAddresses可以配置多个地址,新客户... 目录golang nacos获取配置和服务注册-支持集群初始化客户端可选参数配置new一个客户端 支

Mysql 驱动程序的程序小结

《Mysql驱动程序的程序小结》MySQL驱动程序是连接应用程序与MySQL数据库的重要组件,根据不同的编程语言和应用场景,MySQL提供了多种驱动程序,下面就来详细的了解一下驱动程序,感兴趣的可以... 目录一、mysql 驱动程序的概念二、常见的 MySQL 驱动程序1. MySQL Connector

Python实现Word文档自动化的操作大全(批量生成、模板填充与内容修改)

《Python实现Word文档自动化的操作大全(批量生成、模板填充与内容修改)》在职场中,Word文档是公认的好伙伴,但你有没有被它折磨过?批量生成合同、制作报告以及发放证书/通知等等,这些重复、低效... 目录重复性文档制作,手动填充模板,效率低下还易错1.python-docx入门:Word文档的“瑞士

Python版本信息获取方法详解与实战

《Python版本信息获取方法详解与实战》在Python开发中,获取Python版本号是调试、兼容性检查和版本控制的重要基础操作,本文详细介绍了如何使用sys和platform模块获取Python的主... 目录1. python版本号获取基础2. 使用sys模块获取版本信息2.1 sys模块概述2.1.1

使用Java填充Word模板的操作指南

《使用Java填充Word模板的操作指南》本文介绍了Java填充Word模板的实现方法,包括文本、列表和复选框的填充,首先通过Word域功能设置模板变量,然后使用poi-tl、aspose-words... 目录前言一、设置word模板普通字段列表字段复选框二、代码1. 引入POM2. 模板放入项目3.代码