技术交流请加QQ群:Jeewx微信开发④【289709451】

微信公众号第三方平台的开放,让公众号运营者在面向垂直行业需求时,可以通过一键登录授权给第三方开发者,来完成相关的处理能力,方便快捷,那如何才能开发出一个公众号第三方平台供一键授权呢?本文以JAVA作为后台服务的实现语言,实现了微信第三方开放平台开发所需要的主要业务流程,并针对全网发布的检测做了相应的代码处理,以通过微信全网检测,可以接入任意的微信公众号。

根据微信第三方平台的审核需求,你需要在微信开放平台上注册第三方平台信息时,提供如下几个主要的服务:

1、 授权事件接收服务 ,对应填写的审核资料中授权事件接收URL,微信会将相关的授权事件信息推送到该REST服务上,推送的主要消息包括验证票据ComponentVerifyTicket和取消授权的公众号AuthorizerAppid,该服务需要对微信推送过来的该类消息立即做出回应并返回success内容,该服务事件的JAVA实现方式如下:

  /*** 授权事件接收* * @param request* @param response* @throws IOException* @throws AesException* @throws DocumentException*/@RequestMapping(value = "/open/event/authorize", method = RequestMethod.POST)@ResponseStatus(HttpStatus.NO_CONTENT)public void acceptAuthorizeEvent(HttpServletRequest request, HttpServletResponse response) throws IOException, AesException, DocumentException {WeixinOpenService.getInstance().processAuthorizeEvent(request);WeixinOpenService.getInstance().output(response, "success"); // 输出响应的内容。}

更具体的实现代码如下:

  /*** 处理授权事件的推送* * @param request* @throws IOException* @throws AesException* @throws DocumentException*/public void processAuthorizeEvent(HttpServletRequest request) throws IOException, DocumentException, AesException {String token = WeixinOpenService.TOKEN;String nonce = request.getParameter("nonce");String timestamp = request.getParameter("timestamp");String signature = request.getParameter("signature");String msgSignature = request.getParameter("msg_signature");if (!StringUtils.isNotBlank(msgSignature))return;// 微信推送给第三方开放平台的消息一定是加过密的,无消息加密无法解密消息boolean isValid = WechatCallbackServiceController.checkSignature(token, signature, timestamp, nonce);if (isValid) {StringBuilder sb = new StringBuilder();BufferedReader in = request.getReader();String line;while ((line = in.readLine()) != null) {sb.append(line);}String xml = sb.toString();String encodingAesKey = WeixinOpenService.ENCODINGAESKEY;// 第三方平台组件加密密钥String appId = getAuthorizerAppidFromXml(xml, "authorizationEvent");// 此时加密的xml数据中ToUserName是非加密的,解析xml获取即可WXBizMsgCrypt pc = new WXBizMsgCrypt(token, encodingAesKey, appId);xml = pc.decryptMsg(msgSignature, timestamp, nonce, xml, "AppId");processAuthorizationEvent(xml);}}

2、 公众号消息与事件接收服务 ,对应填写的审核资料中公众号消息与事件接收URL,微信会将粉丝发送给公众号的消息和事件推送到该REST服务上,微信公众平台要求该消息和事件接收服务在5秒内做出回应,如果5秒内微信公众平台得不到响应消息,粉丝将将收到提示公众号暂时服务提供服务的错误信息。对于需要对粉丝发送的消息走人工渠道做出响应的公众号来说,此时就需要首先接收下消息,将消息交给后来逻辑转人工处理,然后立即以空格消息响应微信公众平台,微信收到空格消息后就会知道该粉丝发送的消息已经被妥善处理,并对该响应不做任何处理,同时不会发起消息重新推送的重试。该服务的JAVA实现实现方式如下:

 /*** 处理微信推送过来的授权公众号的消息及事件* */public void processMessageAndEvent(HttpServletRequest request,String xml) throws IOException, AesException, DocumentException {String nonce = request.getParameter("nonce");String timestamp = request.getParameter("timestamp");String msgSignature = request.getParameter("msg_signature");String encodingAesKey = WeixinOpenService.ENCODINGAESKEY;String token = WeixinOpenService.TOKEN;WXBizMsgCrypt pc = new WXBizMsgCrypt(token, encodingAesKey, WeixinOpenService.COMPONENT_APPID);xml = pc.decryptMsg(msgSignature, timestamp, nonce, xml, "ToUserName");WechatCallbackServiceController.processMessage(xml);}

以上是开发微信第三方开发平台的主要服务代码,想要通过微信全网接入检测并成功发布,还有如下的工作的需要做:

  • 开发一个体验页,可以直接让审核人员体验,因为需要的是直接体验,所以访问该页面就不要有认证和权限控制之类的逻辑了,这个页面要求符合微信第三方平台基本的设计要求,本人简单实现了如下的页面格式是可以成功通过审核的,如下:
  • 针对微信全网检测的固定账号做出特定的响应,主要包括一个文本消息响应,一个事件消息响应和一个客服接口调用验证,微信全网检测要求测试的固定账号接收到以上消息后,分别做出如下的响应:接收到TESTCOMPONENT_MSG_TYPE_TEXT这样的文本消息立即回复给粉丝文本内容TESTCOMPONENT_MSG_TYPE_TEXT_callback;接收到事件消息,立即以文本内容的消息格式回复粉丝内容event + “from_callback”,其中event需要根据实际内容替换为具体事件类型;接收到QUERY_AUTH_CODE:query_auth_code  这样的文本消息,需要立即响应空字符串给微信,之后调用客服接口回复粉丝文本消息,内容为:$query_auth_code\$_from_api,其中query_auth_code需要替换为微信实际推送过来的数据。主要的JAVA后台实现代码如下:
 /*** 公众号消息与事件接收* * @param request* @param response* @throws DocumentException* @throws AesException* @throws IOException*/@RequestMapping(value = "/open/{appid}/callback", method = RequestMethod.POST)@ResponseStatus(HttpStatus.NO_CONTENT)public void acceptMessageAndEvent(HttpServletRequest request, HttpServletResponse response) throws IOException, AesException, DocumentException {String msgSignature = request.getParameter("msg_signature");if (!StringUtils.isNotBlank(msgSignature))return;// 微信推送给第三方开放平台的消息一定是加过密的,无消息加密无法解密消息StringBuilder sb = new StringBuilder();BufferedReader in = request.getReader();String line;while ((line = in.readLine()) != null) {sb.append(line);}in.close();String xml = sb.toString();Document doc = DocumentHelper.parseText(xml);Element rootElt = doc.getRootElement();String toUserName = rootElt.elementText("ToUserName");if (StringUtils.equalsIgnoreCase(toUserName, "gh_3c884a361561")) {// 微信全网测试账号WeixinWholeNetworkTestService.getInstance().checkWeixinAllNetworkCheck(request,response,xml);}else{WeixinOpenService.getInstance().processMessageAndEvent(request,xml);WeixinOpenService.getInstance().output(response, "");}}
  • 其中gh_3c884a361561这个账号是微信全网接入检测的固定账号,针对全网检测需要对该账号做特出响应,一旦全网接入检测通过,这部分的代码是可以去掉的,只有全网检测的时候才需要这部分代码。
 public void checkWeixinAllNetworkCheck(HttpServletRequest request, HttpServletResponse response,String xml) throws DocumentException, IOException, AesException{String nonce = request.getParameter("nonce");String timestamp = request.getParameter("timestamp");String msgSignature = request.getParameter("msg_signature");String encodingAesKey = WeixinOpenService.ENCODINGAESKEY;String token = WeixinOpenService.TOKEN;WXBizMsgCrypt pc = new WXBizMsgCrypt(token, encodingAesKey, WeixinOpenService.COMPONENT_APPID);xml = pc.decryptMsg(msgSignature, timestamp, nonce, xml, "ToUserName");Document doc = DocumentHelper.parseText(xml);Element rootElt = doc.getRootElement();String msgType = rootElt.elementText("MsgType");String toUserName = rootElt.elementText("ToUserName");String fromUserName = rootElt.elementText("FromUserName");switch (msgType) {case "event":String event = rootElt.elementText("Event");replyEventMessage(request,response,event,toUserName,fromUserName);break;case "text":String content = rootElt.elementText("Content");processTextMessage(request,response,content,toUserName,fromUserName);break;default:break;}}

根据消息或事件类型区分后,剩余的逻辑只需要处理成对应的回复内容即可,如下:

public void replyEventMessage(HttpServletRequest request, HttpServletResponse response, String event, String toUserName, String fromUserName) throws DocumentException, IOException {String content = event + "from_callback";replyTextMessage(request,response,content,toUserName,fromUserName);}public void processTextMessage(HttpServletRequest request, HttpServletResponse response,String content,String toUserName, String fromUserName) throws IOException, DocumentException{if("TESTCOMPONENT_MSG_TYPE_TEXT".equals(content)){String returnContent = content+"_callback";replyTextMessage(request,response,returnContent,toUserName,fromUserName);}else if(StringUtils.startsWithIgnoreCase(content, "QUERY_AUTH_CODE")){WeixinOpenService.getInstance().output(response, "");//接下来客服API再回复一次消息replyApiTextMessage(request,response,content.split(":")[1],fromUserName);}}public void replyApiTextMessage(HttpServletRequest request, HttpServletResponse response, String auth_code, String fromUserName) throws DocumentException, IOException {String authorization_code = auth_code;// 得到微信授权成功的消息后,应该立刻进行处理!!相关信息只会在首次授权的时候推送过来WeixinOpenData weixinOpenData = WeixinOpenService.getInstance().getWeixinOpenData(WeixinOpenService.COMPONENT_APPID);long accessTokenExpires = weixinOpenData.getAccessTokenExpires();String componentAccessToken = weixinOpenData.getComponentAccessToken();String componentVerifyTicket = weixinOpenData.getComponentVerifyTicket();JSONObject authorizationInfoJson;if (!this.isExpired(accessTokenExpires)) {authorizationInfoJson = WeixinOpenService.getInstance().apiQueryAuth(componentAccessToken, WeixinOpenService.COMPONENT_APPID, authorization_code);} else {JSONObject accessTokenJson = WeixinOpenService.getInstance().getComponentAccessToken(WeixinOpenService.COMPONENT_APPID, WeixinOpenService.COMPONENT_APPSECRET, componentVerifyTicket);componentAccessToken = accessTokenJson.getString("component_access_token");authorizationInfoJson = WeixinOpenService.getInstance().apiQueryAuth(componentAccessToken, WeixinOpenService.COMPONENT_APPID, authorization_code);}if (log.isDebugEnabled()) {log.debug("weixinopen callback authorizationInfo is " + authorizationInfoJson);}JSONObject infoJson = authorizationInfoJson.getJSONObject("authorization_info");String authorizer_access_token = infoJson.getString("authorizer_access_token");String url = "https://api.weixin.qq.com/cgi- ... ot%3B + authorizer_access_token;DefaultHttpClient client = new DefaultHttpClient();enableSSLDefaultHttpClient(client);HttpPost httpPost = new HttpPost(url);JSONObject message = processWechatTextMessage(client, httpPost, fromUserName, auth_code + "_from_api");if(log.isDebugEnabled()){log.debug("api reply messto to weixin whole network test respose = "+message);}}   public void replyTextMessage(HttpServletRequest request, HttpServletResponse response, String content, String toUserName, String fromUserName) throws DocumentException, IOException {Long createTime = Calendar.getInstance().getTimeInMillis() / 1000;StringBuffer sb = new StringBuffer();sb.append("");sb.append("");sb.append("");sb.append("" + createTime + "");sb.append("");sb.append("");sb.append("");String replyMsg = sb.toString();String returnvaleue = "";try {WXBizMsgCrypt pc = new WXBizMsgCrypt(WeixinOpenService.TOKEN, WeixinOpenService.ENCODINGAESKEY, WeixinOpenService.COMPONENT_APPID);returnvaleue = pc.encryptMsg(replyMsg, createTime.toString(), "easemob");} catch (AesException e) {log.error("auto reply to weixin whole network test occur exception = "+ e);e.printStackTrace();}if(log.isDebugEnabled()){log.debug("return weixin whole network test Text message is = "+returnvaleue);}WeixinOpenService.getInstance().output(response, returnvaleue);}

以上是微信第三方开放平台开发主要的业务流程,在实际开发中,还有两点需要特别注意:
1、微信要求第三方开放平台必须以密文方式接收消息;
2、在实际部署时,需要更换JAVA安全包相关的内容,否则将出现秘钥长度不够的异常,需要替换的文件包括JAVA_HOME/jre/lib/security/local_policy.jar和  JAVA_HOME/jre/lib/security/US_export_policy.jar这两个文件。

微信开放平台之公众号第三方平台开发及全网发布验证相关推荐

  1. 微信公众号开发之微信公众平台与公众号第三方平台区别

    微信公众号开发分为微信公众平台和公众号第三方平台. 首先需要一个认证服务号,然后在设置,基本配置里面配置公众号开发信息和服务器配置. 这是我们会得到开发者ID(AppID),开发者密码(AppSecr ...

  2. 微信开放平台 公众号第三方平台开发 教程一 平台介绍

    教程导航: 微信开放平台 公众号第三方平台开发 教程一 平台介绍 微信开放平台 公众号第三方平台开发 教程二 创建公众号第三方平台 微信开放平台 公众号第三方平台开发 教程三 一键登录授权给第三方平台 ...

  3. 微信开放平台公众号第三方平台开发 教程一 平台介绍

        微信现在火,火的如火如荼,给我们这些第三方的开发者带来了不少机会,相信现在有不少人在基于微信的公众平台在做二次开发,接下来会一系列的文章来介绍微信的另一种开发模式- 基于微信开发平台的公众号第 ...

  4. mysql 推送微信公众号_10分钟完成微信公众号第三方平台全网发布

    背景:在微信公众平台配置服务器URL时,使用了新浪云SAE自带的二级域名,提交时出现一个安全风险的警告,网上查了下,许多服务平台和团队也遇到同样的问题. 经过一番研究 - 为什么会有安全风险的警告? ...

  5. 微信公众号第三方开发之一创建微信公众号第三方平台

    首先声明,在接下来一系列公众号第三方开发教程中,核心原理是参照下面博主的源码: http://www.cnblogs.com/sujingnan/p/4397203.html 拓展业务需求的. 为什么 ...

  6. 微信开放平台 公众号第三方平台开发 教程三 一键登录授权给第三方平台

     更多微信技术交流,请加QQ群:289709451.287090836    公众号第三方平台的开放,是为了让公众号运营者,在面向垂直行业需求时,可以一键登录授权给第三方的公众号运营平台,通过第三方开 ...

  7. 公众号第三方平台和微信公众号平台的区别与开发步骤

    我们知道通过微信公众号平台的开放API可以实现用户个性制定制的功能,我们又知道使用开放API时必须知道AppID(应用ID)和AppSecret(应用密钥),而且还要配置URL(服务器地址).Toke ...

  8. 微赞config.修改php,微信公众号第三方平台 微赞WZ_V100.0版20170612整合包 整合人人商城V2新版+一键升级...

    php+mysql php版本5.3或者以上,OPENSSL必需开启,这是本程序与微信公众号通讯的需求. 我们建议您用云主机!Windows或者Linux皆可,windows主机不推荐用IIS环境,可 ...

  9. 微信公众号第三方平台开发PYTHON教程 PART 2

    github地址:cppfun@wechat-open-third-party-dev 微信公众号第三方平台开发python教程 Part 1 这一节肯定是在第一节的基础上,如果你没有看过第一节,可能 ...

最新文章

  1. Source Insight 基本使用(1)-使用Source Insight查看Android Framework 源码
  2. 清华大学朱文武团队夺冠AAAI 2021国际深度元学习挑战赛
  3. 常见的数据库端口及查询方法
  4. SAP Cloud for Customer Sales Lead明细页面视图的UI模型
  5. 安卓加载asset中的json文件_Android解析Asset目录下的json文件
  6. [转]wince中解析reg和bib文件的不同之处
  7. IOS之xib计算cell的高度
  8. NLP特征工程(待完善细节)
  9. mysql序列号生成软件_GitHub - spcent/seq: 基于mysql的序列号生成器
  10. [我叫以赏]Python获取B站UP主粉丝数
  11. 多个元素过渡---过渡模式
  12. 使用 WebSocket 实现一个网页版的聊天室(摸鱼更隐蔽)
  13. .glusterfs_如何在Ubuntu 20.04上使用GlusterFS创建冗余存储池
  14. Region Proposal by Guided Anchoring论文解读
  15. 自己DIY一个pinephone——debian与主线linux在红米2(msm8916)上的移植
  16. 如何使用Python抓取移动端APP的评论(小白篇)
  17. CSS中的长度单位和HTML5中多媒体标签的使用
  18. 编程小白学习编程的开始
  19. 计算机软件水平考试什么题型,计算机软考考什么内容
  20. 硬件工程师成长之路(11)——职业规划

热门文章

  1. 低代码助力生产管理:车间管理系统
  2. 黑马《linux基础编程》学习笔记(从16到20)
  3. 学计算机的3个奋斗目标,我的学习方法和奋斗目标
  4. 广告营销用户点击预测分析
  5. 华为云构建“好用的化工数字化”
  6. PTC贴片保险的测试
  7. 利用自带命令 手动杀毒
  8. 不稳定就是人生常态,要坚信未来一定会非常美好
  9. Pytorch深度学习(五):加载数据集以及mini-batch的使用
  10. SAP中物料单位更改处理案例