搭建企业级微信公众号
微信公众平台:
https://mp.weixin.qq.com/cgi-bin/home?t=home/index&lang=zh_CN&token=795093844
微信公众号测试平台
https://mp.weixin.qq.com/debug/cgi-bin/sandbox?t=sandbox/login
微信公众平台密码 mayikt_2019@163.com 15527339672w.
微信公众平台环境搭建
https://mp.weixin.qq.com/wiki?t=resource/res_main&id=mp1421135319
画图演示原理
外网映射工具
Natapp网址ngrok
https://natapp.cn/
windows环境运行
运行 natapp -authtoken=a021a8fe913ea048
http://mtmayikt.natapp1.cc/wx/portal/wx5c43fde3c9733d9e
WxJava框架快速开发微信公众号
WxJava 微信公众号框架 https://github.com/Wechat-Group/WxJava
微服务电商项目引入WxJava框架
错误方式:
<dependency>
<groupId>com.github.binarywang</groupId>
<artifactId>weixin-java-mp</artifactId>
<version>3.3.0</version>
</dependency>
java.lang.NoSuchMethodError: com.thoughtworks.xstream.io.xml.XppDriver.(Lcom/thoughtworks/xstream/io/naming/NameCoder;)
原因是微信框架引入了xstream的版本为1.4.1 而springCloud中eureka-client也引入了xstream为了1.4.9从而版本有冲突。
正确方式:
<dependency>
<groupId>com.github.binarywang</groupId>
<artifactId>weixin-java-mp</artifactId>
<version>3.3.0</version>
<exclusions>
<exclusion>
<artifactId>xstream</artifactId>
<groupId>com.thoughtworks.xstream</groupId>
</exclusion>
</exclusions>
</dependency>
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>
spring-cloud-starter-netflix-eureka-client
</artifactId>
<exclusions>
<exclusion>
<artifactId>xstream</artifactId>
<groupId>com.thoughtworks.xstream</groupId>
</exclusion>
</exclusions>
</dependency>
<dependency>
<artifactId>xstream</artifactId>
<groupId>com.thoughtworks.xstream</groupId>
<version>1.4.10</version>
</dependency>
yml配置
logging:
level:
org.springframework.web: INFO
com.github.binarywang.demo.wx.mp: DEBUG
me.chanjar.weixin: DEBUG
wx:
mp:
configs:
- appId: wx6f8ce0ccadf3afff #(一个公众号的appid)
secret: cf2b0b4908a84d92b38b939a6fbabe09#(公众号的appsecret)
token: mayikt #(接口配置里的Token值)
mayikt:
weixin:
registration:
code:
###微信注册码消息
message: 您的注册码为:%s,请关注<a href="https://ke.qq.com/course/273548">腾讯课堂免费公开课</a>,欢迎观看97后架构师余老师的精品课程讲解。官方QQ群:<a href='https://jq.qq.com/?_wv=1027&k=5TVfAMF'>193086273</a>,期待你的加入,感谢!
###默认提示消息
default:
registration:
code:
message: 您的消息,我们已经收到,会及时回复给您的!
微信公众号开发案例
案例1
关注微信公众号,在公众号输入手机号码,对应返回验证码。
@Component
public class MsgHandler extends AbstractHandler {
// 用户发送手机验证码提示
@Value("${mayikt.weixin.registration.code.message}")
private String registrationCodeMessage;
// 默认用户发送验证码提示
@Value("${mayikt.weixin.default.registration.code.message}")
private String defaultRegistrationCodeMessage;
@Override
public WxMpXmlOutMessage handle(WxMpXmlMessage wxMessage, Map<String, Object> context, WxMpService weixinService,
WxSessionManager sessionManager) {
if (!wxMessage.getMsgType().equals(XmlMsgType.EVENT)) {
// TODO 可以选择将消息保存到本地
}
// 当用户输入关键词如“你好”,“客服”等,并且有客服在线时,把消息转发给在线客服
try {
if (StringUtils.startsWithAny(wxMessage.getContent(), "你好", "客服")
&& weixinService.getKefuService().kfOnlineList().getKfOnlineList().size() > 0) {
return WxMpXmlOutMessage.TRANSFER_CUSTOMER_SERVICE().fromUser(wxMessage.getToUser())
.toUser(wxMessage.getFromUser()).build();
}
} catch (WxErrorException e) {
e.printStackTrace();
}
// TODO 组装回复消息
// 1.验证关键字是否为手机号码类型
String fromMsg = wxMessage.getContent();
if (RegexUtils.checkMobile(fromMsg)) {
// 如果发送消息为手机号码类型,则发送短信验证码
int registCode = registCode();
String retContext = registrationCodeMessage.replaceAll("registrationCodeMessage", registCode + "");
return new TextBuilder().build(retContext, wxMessage, weixinService);
}
return new TextBuilder().build(defaultRegistrationCodeMessage, wxMessage, weixinService);
}
// 获取注册码
private int registCode() {
int registCode = (int) (Math.random() * 9000 + 1000);
return registCode;
}
}
注意:后期会将验证码存放在redis中。
微服务项目整合WxJava框架提供注册码接口
将实体类层抽取出来单独服务应用
创建meite-shop-api-entity模块
---- meite-shop-api-weixin-entity—实体类
规定统一微服务接口状态码
创建meite-shop-common-core工程
{“rtnCode”:500,“msg”:“系统错误!”,“data”:null}
{“rtnCode”:200,“msg”:“success”,“data”:{“appId”:“yushengjun”,“appName”:“mayikt”}}
BaseApiService
@Data
@Component
public class BaseApiService<T> {
public BaseResponse<T> setResultError(Integer code, String msg) {
return setResult(code, msg, null);
}
// 返回错误,可以传msg
public BaseResponse<T> setResultError(String msg) {
return setResult(Constants.HTTP_RES_CODE_500, msg, null);
}
// 返回成功,可以传data值
public BaseResponse<T> setResultSuccess(Object data) {
return setResult(Constants.HTTP_RES_CODE_200, Constants.HTTP_RES_CODE_200_VALUE, data);
}
// 返回成功,沒有data值
public BaseResponse<T> setResultSuccess() {
return setResult(Constants.HTTP_RES_CODE_200, Constants.HTTP_RES_CODE_200_VALUE, null);
}
// 返回成功,沒有data值
public BaseResponse<T> setResultSuccess(String msg) {
return setResult(Constants.HTTP_RES_CODE_200, msg, null);
}
// 通用封装
public BaseResponse<T> setResult(Integer code, String msg, Object data) {
return new BaseResponse(code, msg, data);
}
}
@Data
public class BaseResponse<T> {
private Integer rtnCode;
private String msg;
private T data;
public BaseResponse() {
}
public BaseResponse(Integer rtnCode, String msg, T data) {
super();
this.rtnCode = rtnCode;
this.msg = msg;
this.data = data;
}
}
public interface Constants {
// 响应请求成功
String HTTP_RES_CODE_200_VALUE = "success";
// 系统错误
String HTTP_RES_CODE_500_VALUE = "fial";
// 响应请求成功code
Integer HTTP_RES_CODE_200 = 200;
// 系统错误
Integer HTTP_RES_CODE_500 = 500;
// 未关联QQ账号
Integer HTTP_RES_CODE_201 = 201;
// 发送邮件
String MSG_EMAIL = "email";
// 会员token
String TOKEN_MEMBER = "TOKEN_MEMBER";
// 用户有效期 90天
Long TOKEN_MEMBER_TIME = (long) (60 * 60 * 24 * 90);
int COOKIE_TOKEN_MEMBER_TIME = (60 * 60 * 24 * 90);
// cookie 会员 totoken 名称
String COOKIE_MEMBER_TOKEN = "cookie_member_token";
// 微信code
String WEIXINCODE_KEY = "weixin.code";
// 微信注册码有效期30分钟
Long WEIXINCODE_TIMEOUT = 1800l;
}
@RestController
public class WeiXinAppServiceImpl extends BaseApiService<AppEntity> implements WeiXinAppService {
@Value("${mayikt.weixin.name}")
private String name;
public BaseResponse<AppEntity> getApp() {
// return setResultSuccess(new AppEntity("yushengjun", "mayikt"));
return setResultError("系统错误!");
}
}
微信服务提供注册验证码接口
微信服务引入 Maven依赖
<dependencies>
<dependency>
<groupId>com.mayikt</groupId>
<artifactId>meite-shop-service-api-weixin</artifactId>
<version>0.0.1-SNAPSHOT</version>
</dependency>
<dependency>
<groupId>com.github.binarywang</groupId>
<artifactId>weixin-java-mp</artifactId>
<version>3.3.0</version>
<exclusions>
<exclusion>
<artifactId>xstream</artifactId>
<groupId>com.thoughtworks.xstream</groupId>
</exclusion>
</exclusions>
</dependency>
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>
spring-cloud-starter-netflix-eureka-client
</artifactId>
<exclusions>
<exclusion>
<artifactId>xstream</artifactId>
<groupId>com.thoughtworks.xstream</groupId>
</exclusion>
</exclusions>
</dependency>
<dependency>
<artifactId>xstream</artifactId>
<groupId>com.thoughtworks.xstream</groupId>
<version>1.4.10</version>
</dependency>
</dependencies>
MsgHandler
/**
* 发送验证码消息
*/
@Value("${mayikt.weixin.registration.code.message}")
private String registrationCodeMessage;
/**
* 默认回复消息
*/
@Value("${mayikt.weixin.default.registration.code.message}")
private String defaultRegistrationCodeMessage;
@Autowired
private RedisUtil redisUtil;
@Override
public WxMpXmlOutMessage handle(WxMpXmlMessage wxMessage, Map<String, Object> context, WxMpService weixinService,
WxSessionManager sessionManager) {
if (!wxMessage.getMsgType().equals(XmlMsgType.EVENT)) {
// TODO 可以选择将消息保存到本地
}
// 当用户输入关键词如“你好”,“客服”等,并且有客服在线时,把消息转发给在线客服
try {
if (StringUtils.startsWithAny(wxMessage.getContent(), "你好", "客服")
&& weixinService.getKefuService().kfOnlineList().getKfOnlineList().size() > 0) {
return WxMpXmlOutMessage.TRANSFER_CUSTOMER_SERVICE().fromUser(wxMessage.getToUser())
.toUser(wxMessage.getFromUser()).build();
}
} catch (WxErrorException e) {
e.printStackTrace();
}
// 1.获取客户端发送的消息
String fromContent = wxMessage.getContent();
// 2.如果客户端发送消息为手机号码,则发送验证码
if (RegexUtils.checkMobile(fromContent)) {
// 3.生成随机四位注册码
int registCode = registCode();
String content = String.format(registrationCodeMessage, registCode);
// 4.将验证码存放在Redis中
redisUtil.setString(Constants.WEIXINCODE_KEY + fromContent, registCode + "", Constants.WEIXINCODE_TIMEOUT);
return new TextBuilder().build(content, wxMessage, weixinService);
}
return new TextBuilder().build(defaultRegistrationCodeMessage, wxMessage, weixinService);
}
// 获取注册码
private int registCode() {
int registCode = (int) (Math.random() * 9000 + 1000);
return registCode;
}
RedisUtil
@Component
public class RedisUtil {
@Autowired
private StringRedisTemplate stringRedisTemplate;
/**
* 存放string类型
*
* @param key
* key
* @param data
* 数据
* @param timeout
* 超时间
*/
public void setString(String key, String data, Long timeout) {
stringRedisTemplate.opsForValue().set(key, data);
if (timeout != null) {
stringRedisTemplate.expire(key, timeout, TimeUnit.SECONDS);
}
}
/**
* 存放string类型
*
* @param key
* key
* @param data
* 数据
*/
public void setString(String key, String data) {
setString(key, data, null);
}
/**
* 根据key查询string类型
*
* @param key
* @return
*/
public String getString(String key) {
String value = stringRedisTemplate.opsForValue().get(key);
return value;
}
/**
* 根据对应的key删除key
*
* @param key
*/
public void delKey(String key) {
stringRedisTemplate.delete(key);
}
}
RegexUtils正则表达式
public class RegexUtils {
/**
* 验证Email
*
* @param email
* email地址,格式:zhangsan@zuidaima.com,zhangsan@xxx.com.cn,
* xxx代表邮件服务商
* @return 验证成功返回true,验证失败返回false
*/
public static boolean checkEmail(String email) {
String regex = "\\w+@\\w+\\.[a-z]+(\\.[a-z]+)?";
return Pattern.matches(regex, email);
}
/**
* 验证身份证号码
*
* @param idCard
* 居民身份证号码15位或18位,最后一位可能是数字或字母
* @return 验证成功返回true,验证失败返回false
*/
public static boolean checkIdCard(String idCard) {
String regex = "[1-9]\\d{13,16}[a-zA-Z0-9]{1}";
return Pattern.matches(regex, idCard);
}
/**
* 验证手机号码(支持国际格式,+86135xxxx...(中国内地),+00852137xxxx...(中国香港))
*
* @param mobile
* 移动、联通、电信运营商的号码段
* <p>
* 移动的号段:134(0-8)、135、136、137、138、139、147(预计用于TD上网卡)
* 、150、151、152、157(TD专用)、158、159、187(未启用)、188(TD专用) 177 170 166
* 开头
* </p>
* <p>
* 联通的号段:130、131、132、155、156(世界风专用)、185(未启用)、186(3g)
* </p>
* <p>
* 电信的号段:133、153、180(未启用)、189
* </p>
* @return 验证成功返回true,验证失败返回false
*/
public static boolean checkMobile(String mobile) {
String regex = "^(13[0-9]|14[579]|15[0-3,5-9]|16[6]|17[0135678]|18[0-9]|19[89])\\d{8}$";
return Pattern.matches(regex, mobile);
}
/**
* 验证固定电话号码
*
* @param phone
* 电话号码,格式:国家(地区)电话代码 + 区号(城市代码) + 电话号码,如:+8602085588447
* <p>
* <b>国家(地区) 代码 :</b>标识电话号码的国家(地区)的标准国家(地区)代码。它包含从 0 到 9
* 的一位或多位数字, 数字之后是空格分隔的国家(地区)代码。
* </p>
* <p>
* <b>区号(城市代码):</b>这可能包含一个或多个从 0 到 9 的数字,地区或城市代码放在圆括号——
* 对不使用地区或城市代码的国家(地区),则省略该组件。
* </p>
* <p>
* <b>电话号码:</b>这包含从 0 到 9 的一个或多个数字
* </p>
* @return 验证成功返回true,验证失败返回false
*/
public static boolean checkPhone(String phone) {
String regex = "(\\+\\d+)?(\\d{3,4}\\-?)?\\d{7,8}$";
return Pattern.matches(regex, phone);
}
/**
* 验证整数(正整数和负整数)
*
* @param digit
* 一位或多位0-9之间的整数
* @return 验证成功返回true,验证失败返回false
*/
public static boolean checkDigit(String digit) {
String regex = "\\-?[1-9]\\d+";
return Pattern.matches(regex, digit);
}
/**
* 验证整数和浮点数(正负整数和正负浮点数)
*
* @param decimals
* 一位或多位0-9之间的浮点数,如:1.23,233.30
* @return 验证成功返回true,验证失败返回false
*/
public static boolean checkDecimals(String decimals) {
String regex = "\\-?[1-9]\\d+(\\.\\d+)?";
return Pattern.matches(regex, decimals);
}
/**
* 验证空白字符
*
* @param blankSpace
* 空白字符,包括:空格、\t、\n、\r、\f、\x0B
* @return 验证成功返回true,验证失败返回false
*/
public static boolean checkBlankSpace(String blankSpace) {
String regex = "\\s+";
return Pattern.matches(regex, blankSpace);
}
/**
* 验证中文
*
* @param chinese
* 中文字符
* @return 验证成功返回true,验证失败返回false
*/
public static boolean checkChinese(String chinese) {
String regex = "^[\u4E00-\u9FA5]+$";
return Pattern.matches(regex, chinese);
}
/**
* 验证日期(年月日)
*
* @param birthday
* 日期,格式:1992-09-03,或1992.09.03
* @return 验证成功返回true,验证失败返回false
*/
public static boolean checkBirthday(String birthday) {
String regex = "[1-9]{4}([-./])\\d{1,2}\\1\\d{1,2}";
return Pattern.matches(regex, birthday);
}
/**
* 验证URL地址
*
* @param url
* 格式:http://blog.csdn.net:80/xyang81/article/details/7705960? 或
* http://www.csdn.net:80
* @return 验证成功返回true,验证失败返回false
*/
public static boolean checkURL(String url) {
String regex = "(https?://(w{3}\\.)?)?\\w+\\.\\w+(\\.[a-zA-Z]+)*(:\\d{1,5})?(/\\w*)*(\\??(.+=.*)?(&.+=.*)?)?";
return Pattern.matches(regex, url);
}
/**
* <pre>
* 获取网址 URL 的一级域
* </pre>
*
* @param url
* @return
*/
public static String getDomain(String url) {
Pattern p = Pattern.compile("(?<=http://|\\.)[^.]*?\\.(com|cn|net|org|biz|info|cc|tv)",
Pattern.CASE_INSENSITIVE);
// 获取完整的域名
// Pattern
// p=Pattern.compile("[^//]*?\\.(com|cn|net|org|biz|info|cc|tv)",
// Pattern.CASE_INSENSITIVE);
Matcher matcher = p.matcher(url);
matcher.find();
return matcher.group();
}
/**
* 匹配中国邮政编码
*
* @param postcode
* 邮政编码
* @return 验证成功返回true,验证失败返回false
*/
public static boolean checkPostcode(String postcode) {
String regex = "[1-9]\\d{5}";
return Pattern.matches(regex, postcode);
}
/**
* 匹配IP地址(简单匹配,格式,如:192.168.1.1,127.0.0.1,没有匹配IP段的大小)
*
* @param ipAddress
* IPv4标准地址
* @return 验证成功返回true,验证失败返回false
*/
public static boolean checkIpAddress(String ipAddress) {
String regex = "[1-9](\\d{1,2})?\\.(0|([1-9](\\d{1,2})?))\\.(0|([1-9](\\d{1,2})?))\\.(0|([1-9](\\d{1,2})?))";
return Pattern.matches(regex, ipAddress);
}
}
YML配置
spring:
# application:
# name: app-mayikt-weixin
redis:
host: 192.168.212.240
port: 6379
pool:
max-idle: 100
min-idle: 1
max-active: 1000
max-wait: -1
logging:
level:
org.springframework.web: INFO
com.github.binarywang.demo.wx.mp: DEBUG
me.chanjar.weixin: DEBUG
wx:
mp:
configs:
- appId: wx5c43fde3c9733d9e #(一个公众号的appid)
secret: b8b217126c33a5fb7074927d5e72a81a #(公众号的appsecret)
token: mayikt #(接口配置里的Token值)
server:
port: 8200
mayikt:
weixin:
registration:
code:
###微信注册码消息
message: 您的注册码为:%s,请关注<a href="https://ke.qq.com/course/273548">腾讯课堂免费公开课</a>,欢迎观看97后架构师余老师的精品课程讲解。官方QQ群:<a href='https://jq.qq.com/?_wv=1027&k=5TVfAMF'>193086273</a>,期待你的加入,感谢!
###默认提示消息
default:
registration:
code:
message: 您的消息,我们已经收到,会及时回复给您的!
相关依赖
RedisMaven依赖
<!-- 集成redis -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>
commons-lang3Maven依赖
<dependency>
<groupId>org.apache.commons</groupId>
<artifactId>commons-lang3</artifactId>
</dependency>
将微信配置文件上传到阿波罗
1.阿波罗平台创建Namespace
2.application.properties
app.id=new_app-mayikt-weixin
apollo.meta=http://192.168.212.240:8080
apollo.bootstrap.enabled = true
apollo.bootstrap.namespaces = application,mayikt.weixin
https://github.com/ctripcorp/apollo/wiki/Apollo%E6%A0%B8%E5%BF%83%E6%A6%82%E5%BF%B5%E4%B9%8B%E2%80%9CNamespace%E2%80%9D
提供给会员服务接口注册码验证接口
@Api(tags = "微信注册码验证码接口")
public interface VerificaCodeService {
/**
* 功能说明:根据手机号码验证码token是否正确
*
* @return
*/
@ApiOperation(value = "根据手机号码验证码token是否正确")
@GetMapping("/verificaWeixinCode")
@ApiImplicitParams({
// @ApiImplicitParam(paramType="header",name="name",dataType="String",required=true,value="用户的姓名",defaultValue="zhaojigang"),
@ApiImplicitParam(paramType = "query", name = "phone", dataType = "String", required = true, value = "用户手机号码"),
@ApiImplicitParam(paramType = "query", name = "weixinCode", dataType = "String", required = true, value = "微信注册码") })
public BaseResponse<JSONObject> verificaWeixinCode(String phone, String weixinCode);
}
@RestController
public class VerificaCodeServiceImpl extends BaseApiService<JSONObject> implements VerificaCodeService {
@Autowired
private RedisUtil redisUtil;
@Override
public BaseResponse<JSONObject> verificaWeixinCode(String phone, String weixinCode) {
if (StringUtils.isEmpty(phone)) {
return setResultError("手机号码不能为空!");
}
if (StringUtils.isEmpty(weixinCode)) {
return setResultError("注册码不能为空!");
}
String code = redisUtil.getString(Constants.WEIXINCODE_KEY + phone);
if (StringUtils.isEmpty(code)) {
return setResultError("注册码已经过期,请重新发送验证码");
}
if (!code.equals(weixinCode)) {
return setResultError("注册码不正确");
}
return setResultSuccess("注册码验证码正确");
}
}
会员注册需求分析:
1、微信专注订阅号输入手机号码获取注册码
流程:使用微信事件通知直接返回注册码(将手机号码和对应的注册码存入在redis中)
手机号码作为rediskey,value表示注册码
3.用户注册调用会员服务接口,会员服务接口调用微信注册码验证接口传递:手机号码和注册码
4.微信服务接口需要提供根据手机号+注册码验证是否正确
微信框架中官方Demo例子中xstream 1.4.10
微信服务项目xstream1.4.9
如何解决Mmaven依赖Jar冲突问题
1.排除依赖jar,在强制引入需要的版本