微信公众号开发(一)

上一篇文章大致解读了官方文档给出的开发概述,本文正式开始开发步骤的记录。

1. 为了配合微信请求只能使用域名的要求,可以使用natapp搭建外网服务器,模拟域名访问,详细的步骤可参考文章:搭建外网传送门。主要就是配置一个免费隧道,并下载对应的natapp插件,按照免费隧道中的authtoken,配置config.ini文件放在natapp根目录下,双击启动即可。

 

 启动natapp见下列这样即说明配置成功,可通过域名访问

域名设置成功就可以进行公众号开发了.

step1 引包

<!--微信封装类-->
<dependency>
    <groupId>com.github.binarywang</groupId>
    <artifactId>weixin-java-mp</artifactId>
    <version>3.2.0</version>
</dependency>
<!--用于进行配置文件的注入-->
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-configuration-processor</artifactId>
    <optional>true</optional>
</dependency>

step2 微信相关配置信息的设置

server:
  port: 8803
# 测试公众号
hwy.wx.mp:
  configs[0]:
    appId: 你的公众号appid
    secret: 你的公众号appsecret
    token: 自定义设置一个token,会在公众号配置中使用,要保持一致
    aesKey: 公众号中的
    template1:
      alarm: 告警推送信息的模板id

step3 代码开发

①  微信公众号相关配置信息注入到WxMpProperties类中,支持多公众号的注入

package com.iris.wechat.config;

import lombok.Data;
import lombok.ToString;
import org.springframework.boot.context.properties.ConfigurationProperties;

import java.util.List;

/**
 * 注入微信公众号的配置信息
 */
@Data
@ConfigurationProperties(prefix = "hwy.wx.mp")
public class WxMpProperties {
    private List<MpConfig> configs;

    @Data
    @ToString
    public static class MpConfig {
        /**
         * 设置微信公众号的appid
         */
        private String appId;

        /**
         * 设置微信公众号的app secret
         */
        private String secret;

        /**
         * 设置微信公众号的token
         */
        private String token;

        /**
         * 设置微信公众号的EncodingAESKey
         */
        private String aesKey;
    }
}

② 注入配置文件对象,公众号常见事件的路由层WxMpConfiguration

package com.iris.wechat.config;

import com.google.common.collect.Maps;
import com.iris.wechat.handler.*;
import me.chanjar.weixin.common.api.WxConsts.EventType;
import me.chanjar.weixin.common.api.WxConsts.MenuButtonType;
import me.chanjar.weixin.common.api.WxConsts.XmlMsgType;
import me.chanjar.weixin.mp.api.WxMpInMemoryConfigStorage;
import me.chanjar.weixin.mp.api.WxMpMessageRouter;
import me.chanjar.weixin.mp.api.WxMpService;
import me.chanjar.weixin.mp.api.impl.WxMpServiceImpl;
import me.chanjar.weixin.mp.constant.WxMpEventConstants;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.context.properties.EnableConfigurationProperties;
import org.springframework.context.annotation.Configuration;

import javax.annotation.PostConstruct;
import java.util.Map;
import java.util.stream.Collectors;

/**
 * wechat mp configuration
 *
 * @author Binary Wang(https://github.com/binarywang)
 */
@Configuration
@EnableConfigurationProperties(WxMpProperties.class)
public class WxMpConfiguration {
    private LogHandler logHandler;
    private NullHandler nullHandler;
    private KfSessionHandler kfSessionHandler;
    private StoreCheckNotifyHandler storeCheckNotifyHandler;
    private LocationHandler locationHandler;
    private MenuHandler menuHandler;
    private MsgHandler msgHandler;
    private UnsubscribeHandler unsubscribeHandler;
    private SubscribeHandler subscribeHandler;

    private WxMpProperties properties;

    private static Map<String, WxMpMessageRouter> routers = Maps.newHashMap();
    private static Map<String, WxMpService> mpServices = Maps.newHashMap();

    @Autowired
    public WxMpConfiguration(LogHandler logHandler, NullHandler nullHandler, KfSessionHandler kfSessionHandler,
                             StoreCheckNotifyHandler storeCheckNotifyHandler, LocationHandler locationHandler,
                             MenuHandler menuHandler, MsgHandler msgHandler, UnsubscribeHandler unsubscribeHandler,
                             SubscribeHandler subscribeHandler, WxMpProperties properties) {
        this.logHandler = logHandler;
        this.nullHandler = nullHandler;
        this.kfSessionHandler = kfSessionHandler;
        this.storeCheckNotifyHandler = storeCheckNotifyHandler;
        this.locationHandler = locationHandler;
        this.menuHandler = menuHandler;
        this.msgHandler = msgHandler;
        this.unsubscribeHandler = unsubscribeHandler;
        this.subscribeHandler = subscribeHandler;
        this.properties = properties;
    }

    public static Map<String, WxMpMessageRouter> getRouters() {
        return routers;
    }
    
    @PostConstruct
    public void init() {
      services();
    }

    public static Map<String, WxMpService> getMpServices() {
        return mpServices;
    }

    public Object services() {
        mpServices = this.properties.getConfigs()
            .stream()
            .map(a -> {
                WxMpInMemoryConfigStorage configStorage = new WxMpInMemoryConfigStorage();
                configStorage.setAppId(a.getAppId());
                configStorage.setSecret(a.getSecret());
                configStorage.setToken(a.getToken());
                configStorage.setAesKey(a.getAesKey());
                WxMpService service = new WxMpServiceImpl();
                service.setWxMpConfigStorage(configStorage);
                routers.put(a.getAppId(), this.newRouter(service));
                return service;
            }).collect(Collectors.toMap(s -> s.getWxMpConfigStorage().getAppId(), a -> a));
        return Boolean.TRUE;
    }

    private WxMpMessageRouter newRouter(WxMpService wxMpService) {
        final WxMpMessageRouter newRouter = new WxMpMessageRouter(wxMpService);

        // 记录所有事件的日志 (异步执行)
        newRouter.rule().handler(this.logHandler).next();

        // 接收客服会话管理事件
        newRouter.rule().async(false).msgType(XmlMsgType.EVENT)
            .event(WxMpEventConstants.CustomerService.KF_CREATE_SESSION)
            .handler(this.kfSessionHandler).end();
        newRouter.rule().async(false).msgType(XmlMsgType.EVENT)
            .event(WxMpEventConstants.CustomerService.KF_CLOSE_SESSION)
            .handler(this.kfSessionHandler)
            .end();
        newRouter.rule().async(false).msgType(XmlMsgType.EVENT)
            .event(WxMpEventConstants.CustomerService.KF_SWITCH_SESSION)
            .handler(this.kfSessionHandler).end();

        // 门店审核事件
        newRouter.rule().async(false).msgType(XmlMsgType.EVENT)
            .event(WxMpEventConstants.POI_CHECK_NOTIFY)
            .handler(this.storeCheckNotifyHandler).end();

        // 自定义菜单事件
        newRouter.rule().async(false).msgType(XmlMsgType.EVENT)
            .event(MenuButtonType.CLICK).handler(this.menuHandler).end();

        // 点击菜单连接事件
        newRouter.rule().async(false).msgType(XmlMsgType.EVENT)
            .event(MenuButtonType.VIEW).handler(this.nullHandler).end();

        // 关注事件
        newRouter.rule().async(false).msgType(XmlMsgType.EVENT)
            .event(EventType.SUBSCRIBE).handler(this.subscribeHandler)
            .end();

        // 取消关注事件
        newRouter.rule().async(false).msgType(XmlMsgType.EVENT)
            .event(EventType.UNSUBSCRIBE)
            .handler(this.unsubscribeHandler).end();

        // 上报地理位置事件
        newRouter.rule().async(false).msgType(XmlMsgType.EVENT)
            .event(EventType.LOCATION).handler(this.locationHandler)
            .end();

        // 接收地理位置消息
        newRouter.rule().async(false).msgType(XmlMsgType.LOCATION)
            .handler(this.locationHandler).end();

        // 扫码事件
        newRouter.rule().async(false).msgType(XmlMsgType.EVENT)
            .event(EventType.SCAN).handler(this.nullHandler).end();

        // 默认
        newRouter.rule().async(false).handler(this.msgHandler).end();

        return newRouter;
    }
}

③ 根据WxMpConfiguration类中的handler完成各个handler的开发,根据自己的业务场景作相应的开发,

 

④  完成验证接口的开发,分两个接口,接口的路径一致只是方法不同,一个get方法:用于用户界面上域名接口验证,一个post方法:用于关注,取关,自定义菜单等事件的触发。

package com.iris.wechat.controller;

import com.iris.wechat.config.WxMpConfiguration;
import com.iris.wechat.log.XlyLogger;
import me.chanjar.weixin.mp.api.WxMpService;
import me.chanjar.weixin.mp.bean.message.WxMpXmlMessage;
import me.chanjar.weixin.mp.bean.message.WxMpXmlOutMessage;
import org.apache.commons.lang3.StringUtils;
import org.slf4j.Logger;
import org.springframework.web.bind.annotation.*;

import javax.servlet.http.HttpServletRequest;

/**
 * 与微信交互的API文件
 * @Date 2019-11-22 16:41:00
 * @author muruan.lt
 * 配合华为云公众号绑定
 */
@RestController
@RequestMapping("/wx/{appid}")
public class WechatController {

    private static Logger log = XlyLogger.get();

    // 接口配置信息调用接口(GET)
    @GetMapping(produces = "text/plain;charset=utf-8")
    public String doGet(@PathVariable String appid, HttpServletRequest request) {
        // 微信加密签名
        String signature = request.getParameter("signature");
        // 时间戳
        String timestamp = request.getParameter("timestamp");
        // 随机数
        String nonce = request.getParameter("nonce");
        // 随机字符串
        String echostr = request.getParameter("echostr");
        log.info("\n接收到来自微信服务器的认证消息:[{}, {}, {}, {}]", signature, timestamp, nonce, echostr);
        if (StringUtils.isAnyBlank(signature, timestamp, nonce, echostr)) {
            log.info("signature " + signature + "  timestamp " + timestamp + " nonce " + nonce
                    + " echostr " + echostr);
            throw new IllegalArgumentException("请求参数非法,请核实!");
        }

        final WxMpService wxService = WxMpConfiguration.getMpServices().get(appid);
        if (wxService == null) {
            throw new IllegalArgumentException(String.format("未找到对应appid=[%d]的配置,请核实!", appid));
        }

        // 通过检验signature对请求进行校验,若校验成功则原样返回echostr,表示接入成功,否则接入失败
        if (wxService.checkSignature(timestamp, nonce, signature)) {
            log.info("weixin get success...."+echostr);
            return echostr;
        }else {
            log.error("weixin get failed....");
            return "weixin get failed....";
        }
    }

    // 关注,取关,客服,菜单等调用接口(POST)
    @PostMapping(produces = "application/xml; charset=UTF-8")
    public String doPost(HttpServletRequest request, @PathVariable String appid, @RequestBody String requestBody) {
        log.debug("weixin login get...");
        // 获取微信公众号传输过来的code,通过code可获取access_token,进而获取用户信息
        String code = request.getParameter("code");
        // 微信加密签名
        String signature = request.getParameter("signature");
        // 时间戳
        String timestamp = request.getParameter("timestamp");
        // 随机数
        String nonce = request.getParameter("nonce");
        // openid
        String openid = request.getParameter("openid");
        // encType--可空
        String encType = request.getParameter("encType");
        // msgSignature--可空
        String msgSignature = request.getParameter("msgSignature");

        final WxMpService wxService = WxMpConfiguration.getMpServices().get(appid);
        if (!wxService.checkSignature(timestamp, nonce, signature)) {
            log.info(
                    "\n接收微信请求:[openid=[{}], [signature=[{}], encType=[{}], msgSignature=[{}],"
                            + " timestamp=[{}], nonce=[{}], requestBody=[\n{}\n] ",
                    openid, signature, encType, msgSignature, timestamp, nonce, requestBody);
            throw new IllegalArgumentException("非法请求,可能属于伪造的请求!");
        }

        String out = null;
        if (StringUtils.isBlank(encType)) {
            // 明文传输的消息
            WxMpXmlMessage inMessage = WxMpXmlMessage.fromXml(requestBody);
            WxMpXmlOutMessage outMessage = this.route(inMessage, appid);
            if (outMessage == null) {
                return "";
            }
            out = outMessage.toXml();
        } else if ("aes".equalsIgnoreCase(encType)) {
            // aes加密的消息
            WxMpXmlMessage inMessage = WxMpXmlMessage.fromEncryptedXml(requestBody,
                    wxService.getWxMpConfigStorage(), timestamp, nonce, msgSignature);
            log.debug("\n消息解密后内容为:\n{} ", inMessage.toString());
            WxMpXmlOutMessage outMessage = this.route(inMessage, appid);
            if (outMessage == null) {
                return "";
            }
            out = outMessage.toEncryptedXml(wxService.getWxMpConfigStorage());
        }

        log.debug("\n组装回复信息:{}", out);
        return out;
    }

    private WxMpXmlOutMessage route(WxMpXmlMessage message, String appid) {
        try {
            return WxMpConfiguration.getRouters().get(appid).route(message);
        } catch (Exception e) {
            log.error("路由消息时出现异常!", e);
        }
        return null;
    }

}

其中doGet方法用于接口配置信息的调用接口,doPost方法用于关注,取关等事件的触发。

开发阶段若服务器,域名已经准备好,则可将代码打包上线,进行下一步开发调试,由于上线的代码并不是很方便调试故而可使用上文中natapp产生的域名代替,进行本地调试,详细配置如下,

①  接口配置信息修改:URL是验证接口,Token是自己定义的,务必与服务器配置文件中的token一致

登陆公众号测试账号,找到如下位置,

 

将上图中appID,appsecret信息添加到配置文件中,并启动服务。

 

 

 点击“修改“,将"接口配置信息"接口填在URL处,将上图中的token填在下图的Token处。其中URL的格式为natapp生成的外网访问地址+/wx/{appid},如下图所示

 

 

 点击提交,在本地如下方法中打断点,发现点击提交会进入下面方法,若没有进入则说明配置URL,token,appId,secret等出现错误,若进入该方法则说明配置没有问题。

// 接口配置信息调用接口(GET)
    @GetMapping(produces = "text/plain;charset=utf-8")
    public String doGet(@PathVariable String appid, HttpServletRequest request) {
        // 微信加密签名
        String signature = request.getParameter("signature");
        // 时间戳
        String timestamp = request.getParameter("timestamp");
        // 随机数
        String nonce = request.getParameter("nonce");
        // 随机字符串
        String echostr = request.getParameter("echostr");
        log.info("\n接收到来自微信服务器的认证消息:[{}, {}, {}, {}]", signature, timestamp, nonce, echostr);
        if (StringUtils.isAnyBlank(signature, timestamp, nonce, echostr)) {
            log.info("signature " + signature + "  timestamp " + timestamp + " nonce " + nonce
                    + " echostr " + echostr);
            throw new IllegalArgumentException("请求参数非法,请核实!");
        }

        final WxMpService wxService = WxMpConfiguration.getMpServices().get(appid);
        if (wxService == null) {
            throw new IllegalArgumentException(String.format("未找到对应appid=[%d]的配置,请核实!", appid));
        }

        // 通过检验signature对请求进行校验,若校验成功则原样返回echostr,表示接入成功,否则接入失败
        if (wxService.checkSignature(timestamp, nonce, signature)) {
            log.info("weixin get success...."+echostr);
            return echostr;
        }else {
            log.error("weixin get failed....");
            return "weixin get failed....";
        }
    }

 ②  配置JS接口安全域名:点击修改将外网访问地址填好提交就可以了。

 

 ③ 检查关注公众号,取消关注是否生效

 

 首先在下面方法上打断点,再扫码关注,会进入该方法,同样操作取关也会进入该方法,再需要自己根据实际需求进行业务代码的开发。

// 关注,取关,客服,菜单等调用接口(POST)
    @PostMapping(produces = "application/xml; charset=UTF-8")
    public String doPost(HttpServletRequest request, @PathVariable String appid, @RequestBody String requestBody) {
        log.debug("weixin login get...");
        // 获取微信公众号传输过来的code,通过code可获取access_token,进而获取用户信息
        String code = request.getParameter("code");
        // 微信加密签名
        String signature = request.getParameter("signature");
        // 时间戳
        String timestamp = request.getParameter("timestamp");
        // 随机数
        String nonce = request.getParameter("nonce");
        // openid
        String openid = request.getParameter("openid");
        // encType--可空
        String encType = request.getParameter("encType");
        // msgSignature--可空
        String msgSignature = request.getParameter("msgSignature");

        final WxMpService wxService = WxMpConfiguration.getMpServices().get(appid);
        if (!wxService.checkSignature(timestamp, nonce, signature)) {
            log.info(
                    "\n接收微信请求:[openid=[{}], [signature=[{}], encType=[{}], msgSignature=[{}],"
                            + " timestamp=[{}], nonce=[{}], requestBody=[\n{}\n] ",
                    openid, signature, encType, msgSignature, timestamp, nonce, requestBody);
            throw new IllegalArgumentException("非法请求,可能属于伪造的请求!");
        }

        String out = null;
        if (StringUtils.isBlank(encType)) {
            // 明文传输的消息
            WxMpXmlMessage inMessage = WxMpXmlMessage.fromXml(requestBody);
            WxMpXmlOutMessage outMessage = this.route(inMessage, appid);
            if (outMessage == null) {
                return "";
            }
            out = outMessage.toXml();
        } else if ("aes".equalsIgnoreCase(encType)) {
            // aes加密的消息
            WxMpXmlMessage inMessage = WxMpXmlMessage.fromEncryptedXml(requestBody,
                    wxService.getWxMpConfigStorage(), timestamp, nonce, msgSignature);
            log.debug("\n消息解密后内容为:\n{} ", inMessage.toString());
            WxMpXmlOutMessage outMessage = this.route(inMessage, appid);
            if (outMessage == null) {
                return "";
            }
            out = outMessage.toEncryptedXml(wxService.getWxMpConfigStorage());
        }

        log.debug("\n组装回复信息:{}", out);
        return out;
    }

 

posted @ 2019-12-01 17:03  八月的狮子  阅读(736)  评论(0编辑  收藏  举报