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");
    }
  • 0
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

等一场春雨

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值