Java-最新微信第三方平台公众号授权

第三方平台api地址
https://developers.weixin.qq.com/doc/oplatform/openApi/OpenApiDoc/

预授权码获取之后调用接口获取授权方信息
https://developers.weixin.qq.com/doc/oplatform/Third-party_Platforms/2.0/api/ThirdParty/token/authorization_info.html#%E6%8E%88%E6%9D%83%E4%BF%A1%E6%81%AF%E8%AF%B4%E6%98%8E

1.微信开放平台编辑开发配置


image.png

2.授权事件接受配置>获取令牌>获取预授权码>查询预授权码

        <!--            网络请求-->
        <dependency>
            <groupId>com.github.lianjiatech</groupId>
            <artifactId>retrofit-spring-boot-starter</artifactId>
            <version>2.3.5</version>
        </dependency>
        <dependency>
            <groupId>com.squareup.retrofit2</groupId>
            <artifactId>converter-simplexml</artifactId>
            <version>2.9.0</version>
        </dependency>
@RetrofitClient(baseUrl = "https://api.weixin.qq.com/")
public interface WeChatApi {
    /**
     * 令牌
     * 文档地址 https://developers.weixin.qq.com/doc/oplatform/Third-party_Platforms/2.0/api/ThirdParty/token/component_access_token.html
     * 请求地址 https://api.weixin.qq.com/cgi-bin/component/api_component_token
     */
    @POST("cgi-bin/component/api_component_token")
    JSONObject getComponentAccessToken(@Body GetComponentAccessTokenParam param);

    /**
     * 预授权码
     * <p>
     * 文档地址 https://developers.weixin.qq.com/doc/oplatform/Third-party_Platforms/2.0/api/ThirdParty/token/pre_auth_code.html
     * 请求地址 https://api.weixin.qq.com/cgi-bin/component/api_create_preauthcode?component_access_token=COMPONENT_ACCESS_TOKEN
     */
    @POST("/cgi-bin/component/api_create_preauthcode")
    JSONObject getPreAuthCode(@Query("component_access_token") String token, @Body GetPreAuthCodeParam param);


    /**
     * 查询授权方接口
     * https://api.weixin.qq.com/cgi-bin/component/api_query_auth?component_access_token=COMPONENT_ACCESS_TOKEN
     */
    @POST("/cgi-bin/component/api_query_auth")
    JSONObject getApiQueryAuth(@Query("component_access_token") String accessToken,@Body GetApiQueryAuthParam param);
}

3.授权事件接收配置

    @ApiOperation("授权事件接收配置")
    @RequestMapping("/component_verify_ticket")
    @ResponseBody
    public String componentVerifyTicket(@RequestParam("timestamp") String timestamp, @RequestParam("nonce") String nonce, @RequestParam("msg_signature") String msgSignature,
                                        @RequestBody String postData) throws DocumentException, AesException {
        //https://sunuping.com/vt/api/wx/open/component_verify_ticket
        return this.memberWxService.componentVerifyTicket(postData, msgSignature, timestamp, nonce);
    }


@Override
    public String componentVerifyTicket(String data, String msgSignature, String timestamp, String nonce) throws DocumentException, AesException {
        if (log.isDebugEnabled()) {
            log.debug(data);
        }
        //这个类是微信官网提供的解密类,需要用到消息校验Token 消息加密Key和服务平台appid
        WXBizMsgCrypt pc = new WXBizMsgCrypt(WxConfigConstant.TOKEN, WxConfigConstant.KEY, WxConfigConstant.APPID);
        String xml = pc.decryptMsg(msgSignature, timestamp, nonce, data);
        Map<String, String> map = XmlTools.getMap(xml);
        final String d = map.get("ComponentVerifyTicket");
        if (log.isDebugEnabled()) {
            log.debug("获取微信验证票据:{}", d);
        }

        if (StringUtils.isNotBlank(d)) {
            //缓存11小时
            redisService.set(WxRedisKeyConstant.COMPONENT_VERIFY_TICKET, d, 39600);
        }
        return "success";
    }

4.获取第三方api授权token

    @Override
    public String getComponentAccessToken() {
        String componentAccessToken = (String) redisService.get(WxRedisKeyConstant.COMPONENT_ACCESS_TOKEN);
        if (StringUtils.isBlank(componentAccessToken)) {
            final String ticket = Optional.ofNullable((String) redisService.get(WxRedisKeyConstant.COMPONENT_VERIFY_TICKET)).orElseThrow(() -> new ErrorException("验证凭据获取失败"));
            JSONObject tokenJson = Optional.ofNullable(weChatApi.getComponentAccessToken(new GetComponentAccessTokenParam(WxConfigConstant.APPID, WxConfigConstant.APP_SECRET, ticket)))
                                           //
                                           .orElseThrow(() -> new ErrorException("令牌获取失败"));
            if (log.isDebugEnabled()) {
                log.debug(tokenJson.toJSONString());
            }
            componentAccessToken = Optional.ofNullable(tokenJson.getString("component_access_token")).orElseThrow(() -> new ErrorException("令牌数据为空"));
            //缓存1小时50分钟
            redisService.set(WxRedisKeyConstant.COMPONENT_ACCESS_TOKEN, componentAccessToken, tokenJson.getIntValue("expires_in"));
        }
        return componentAccessToken;
    }

5.获取预授权码

@Override
    public String getPreAuthCode() {
        final String componentAccessToken = this.getComponentAccessToken();

        String code = (String) this.redisService.get(WxRedisKeyConstant.PRE_AUTH_CODE);
        if (StringUtils.isBlank(code)) {
            JSONObject preAuthCodeJson = Optional.ofNullable(weChatApi.getPreAuthCode(componentAccessToken, new GetPreAuthCodeParam(WxConfigConstant.APPID)))
                                                 //
                                                 .orElseThrow(() -> new ErrorException("获取预授权码失败"));
            if (log.isDebugEnabled()) {
                log.debug(preAuthCodeJson.toJSONString());
            }
            code = Optional.ofNullable(preAuthCodeJson.getString("pre_auth_code")).orElseThrow(() -> new ErrorException("获取预授权码失败"));
            this.redisService.set(WxRedisKeyConstant.PRE_AUTH_CODE, code, preAuthCodeJson.getIntValue("expires_in"));

        }
        return this.generatePreAuthCodeUrl(code);
    }

    private String generatePreAuthCodeUrl(String code) {
        long mid = StpUtil.getLoginIdAsLong();
        //https://mp.weixin.qq.com/cgi-bin/componentloginpage?component_appid=xxxx&pre_auth_code=xxxxx&redirect_uri=xxxx&auth_type=xxx
        return "https://mp.weixin.qq.com/cgi-bin/componentloginpage?component_appid=" + WxConfigConstant.APPID + "&pre_auth_code=" + code
                //授权完回去的页面 授权回去的域名要去配置的域名一样
                + "&redirect_uri=https://xxxx/x/bind_wx_public?mid=" + mid + "&auth_type=1";
    }

6.拉取授权方信息

    @GetMapping("/bind_wx_public")
    @ApiOperation("绑定微信公众号")
    public void bindWxPublic(@RequestParam("auth_code") String authCode, @RequestParam("expires_in") Integer expiresIn, @RequestParam("mid") Long mid) throws IOException {
        this.memberWxService.bindWxPublic(authCode, expiresIn, mid);
    }

@Override
    public void bindWxPublic(String authCode, Integer expiresIn, Long mid) throws IOException {
        log.debug("授权码:{},失效时间/秒:{}", authCode, expiresIn);
        final String componentAccessToken = this.getComponentAccessToken();
        JSONObject apiQueryAuthJson = this.weChatApi.getApiQueryAuth(componentAccessToken, new GetApiQueryAuthParam(WxConfigConstant.APPID, authCode));
        log.debug("授权方信息:{}", apiQueryAuthJson.toJSONString());
        JSONObject authorizationInfo = apiQueryAuthJson.getJSONObject("authorization_info");
        //授权方 appid
        String authorizerAppid = authorizationInfo.getString("authorizer_appid");
        //接口调用令牌
        String authorizerAccessToken = authorizationInfo.getString("authorizer_access_token");
        //authorizer_access_token 的有效期
        int authorizerAccessTokenExpiresIn = authorizationInfo.getIntValue("expires_in");
        //刷新令牌
        String authorizerRefreshToken = authorizationInfo.getString("authorizer_refresh_token");
        //授权给开发者的权限集列表
        JSONArray funcInfoJsonArr = authorizationInfo.getJSONArray("func_info");

       //业务处理完后 重定向到某个页面
        response.sendRedirect("https://xxxxx");
    }
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,921评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,635评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,393评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,836评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,833评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,685评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,043评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,694评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,671评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,670评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,779评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,424评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,027评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,984评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,214评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,108评论 2 351
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,517评论 2 343

推荐阅读更多精彩内容