<kbd id="afajh"><form id="afajh"></form></kbd>
<strong id="afajh"><dl id="afajh"></dl></strong>
    <del id="afajh"><form id="afajh"></form></del>
        1. <th id="afajh"><progress id="afajh"></progress></th>
          <b id="afajh"><abbr id="afajh"></abbr></b>
          <th id="afajh"><progress id="afajh"></progress></th>

          Spring Security實(shí)戰(zhàn)干貨:集成微信公眾號OAuth2.0授權(quán)

          共 22593字,需瀏覽 46分鐘

           ·

          2021-08-20 01:56

          微信生態(tài)提供了微信支付、微信優(yōu)惠券、微信H5紅包、微信紅包封面等等促銷工具來幫助我們的應(yīng)用拉新?;?。但是這些福利要想正確地發(fā)放到用戶的手里就必須拿到用戶特定的(微信應(yīng)用)微信標(biāo)識(shí)openid甚至是用戶的微信用戶信息。如果用戶在微信客戶端中訪問我們第三方網(wǎng)頁,公眾號可以通過微信網(wǎng)頁授權(quán)機(jī)制,來獲取用戶基本信息,進(jìn)而實(shí)現(xiàn)業(yè)務(wù)邏輯。今天就結(jié)合Spring Security來實(shí)現(xiàn)一下微信公眾號網(wǎng)頁授權(quán)。

          環(huán)境準(zhǔn)備

          在開始之前我們需要準(zhǔn)備好微信網(wǎng)頁開發(fā)的環(huán)境。

          微信公眾號服務(wù)號

          請注意,一定是微信公眾號服務(wù)號,只有服務(wù)號才提供這樣的能力。像胖哥的這樣公眾號雖然也是認(rèn)證過的公眾號,但是只能發(fā)發(fā)文章并不具備提供服務(wù)的能力。但是微信公眾平臺(tái)提供了沙盒功能來模擬服務(wù)號,可以降低開發(fā)難度,你可以到微信公眾號測試賬號頁面申請,申請成功后別忘了關(guān)注測試公眾號。

          ?

          微信公眾號服務(wù)號只有企事業(yè)單位、政府機(jī)關(guān)才能開通。

          內(nèi)網(wǎng)穿透

          因?yàn)槲⑿欧?wù)器需要回調(diào)開發(fā)者提供的回調(diào)接口,為了能夠本地調(diào)試,內(nèi)網(wǎng)穿透工具也是必須的。啟動(dòng)內(nèi)網(wǎng)穿透后,需要把內(nèi)網(wǎng)穿透工具提供的虛擬域名配置到微信測試帳號的回調(diào)配置中

          點(diǎn)擊修改配置測試賬戶回調(diào)域名

          打開后只需要填寫域名,不要帶協(xié)議頭。例如回調(diào)是https://felord.cn/wechat/callback,只能填寫成這樣:

          然后我們就可以開發(fā)了。

          OAuth2.0客戶端集成

          ?

          基于 Spring Security 5.x

          微信網(wǎng)頁授權(quán)的文檔在網(wǎng)頁授權(quán),這里不再贅述。我們只聊聊如何結(jié)合Spring Security的事。微信網(wǎng)頁授權(quán)是通過OAuth2.0機(jī)制實(shí)現(xiàn)的,在用戶授權(quán)給公眾號后,公眾號可以獲取到一個(gè)網(wǎng)頁授權(quán)特有的接口調(diào)用憑證(網(wǎng)頁授權(quán)access_token),通過網(wǎng)頁授權(quán)獲得的access_token可以進(jìn)行授權(quán)后接口調(diào)用,如獲取用戶的基本信息。

          我們需要引入Spring Security提供的OAuth2.0相關(guān)的模塊:

                 <dependency>
                      <groupId>org.springframework.boot</groupId>
                      <artifactId>spring-boot-starter-security</artifactId>
                  </dependency>
                  <dependency>
                      <groupId>org.springframework.boot</groupId>
                      <artifactId>spring-boot-starter-oauth2-client</artifactId>
                  </dependency>

          ?

          由于我們需要獲取用戶的微信信息,所以要用到OAuth2.0 Login;如果你用不到用戶信息可以選擇OAuth2.0 Client。

          微信網(wǎng)頁授權(quán)流程

          接著按照微信提供的流程來結(jié)合Spring Security。

          獲取授權(quán)碼code

          微信網(wǎng)頁授權(quán)使用的是OAuth2.0的授權(quán)碼模式。我們先來看如何獲取授權(quán)碼。

          https://open.weixin.qq.com/connect/oauth2/authorize?appid=APPID&redirect_uri=REDIRECT_URI&response_type=code&scope=SCOPE&state=STATE#wechat_redirect 

          這是微信獲取codeOAuth2.0端點(diǎn)模板,這不是一個(gè)純粹的OAuth2.0協(xié)議。微信做了一些參數(shù)上的變動(dòng)。這里原生的client_id被替換成了appid,而且末尾還要加#wechat_redirect。這無疑增加了集成的難度。

          這里先放一放,我們目標(biāo)轉(zhuǎn)向Spring Securitycode獲取流程。

          Spring Security會(huì)提供一個(gè)模版鏈接:

          {baseUrl}/oauth2/authorization/{registrationId}

          當(dāng)使用該鏈接請求OAuth2.0客戶端時(shí)會(huì)被OAuth2AuthorizationRequestRedirectFilter攔截。機(jī)制這里不講了,在我個(gè)人博客felord.cn中的Spring Security 實(shí)戰(zhàn)干貨:客戶端OAuth2授權(quán)請求的入口一文中有詳細(xì)闡述。

          攔截之后會(huì)根據(jù)配置組裝獲取授權(quán)碼的請求URL,由于微信的不一樣所以我們針對性的定制,也就是改造OAuth2AuthorizationRequestRedirectFilter中的OAuth2AuthorizationRequestResolver。

          自定義URL

          因?yàn)镾pring Security會(huì)根據(jù)模板鏈接去組裝一個(gè)鏈接而不是我們填參數(shù)就行了,所以需要我們對構(gòu)建URL的處理器進(jìn)行自定義。

          /**
           * 兼容微信的oauth2 端點(diǎn).
           *
           * @author n1
           * @since 2021 /8/11 17:04
           */

          public class WechatOAuth2AuthRequestBuilderCustomizer {
             private static final String WECHAT_ID= "wechat";

              /**
               * Customize.
               *
               * @param builder the builder
               */

              public static void customize(OAuth2AuthorizationRequest.Builder builder) {
                 String regId = (String) builder.build()
                         .getAttributes()
                         .get(OAuth2ParameterNames.REGISTRATION_ID);
                 if (WECHAT_ID.equals(regId)){
                     builder.authorizationRequestUri(WechatOAuth2RequestUriBuilderCustomizer::customize);
                 }
              }

              /**
               * 定制微信OAuth2請求URI
               *
               * @author n1
               * @since 2021 /8/11 15:31
               */

              private static class WechatOAuth2RequestUriBuilderCustomizer {

                  /**
                   * 默認(rèn)情況下Spring Security會(huì)生成授權(quán)鏈接:
                   * {@code https://open.weixin.qq.com/connect/oauth2/authorize?response_type=code
                   * &client_id=wxdf9033184b238e7f
                   * &scope=snsapi_userinfo
                   * &state=5NDiQTMa9ykk7SNQ5-OIJDbIy9RLaEVzv3mdlj8TjuE%3D
                   * &redirect_uri=https%3A%2F%2Fmovingsale-h5-test.nashitianxia.com}
                   * 缺少了微信協(xié)議要求的{@code #wechat_redirect},同時(shí) {@code client_id}應(yīng)該替換為{@code app_id}
                   *
                   * @param builder the builder
                   * @return the uri
                   */

                  public static URI customize(UriBuilder builder) {
                      String reqUri = builder.build().toString()
                              .replaceAll("client_id=""appid=")
                              .concat("#wechat_redirect");
                      return URI.create(reqUri);
                  }
              }
          }
          配置解析器

          把上面?zhèn)€性化改造的邏輯配置到OAuth2AuthorizationRequestResolver:

          /**
           * 用來從{@link javax.servlet.http.HttpServletRequest}中檢索Oauth2需要的參數(shù)并封裝成OAuth2請求對象{@link OAuth2AuthorizationRequest}
           *
           * @param clientRegistrationRepository the client registration repository
           * @return DefaultOAuth2AuthorizationRequestResolver
           */

          private OAuth2AuthorizationRequestResolver oAuth2AuthorizationRequestResolver(ClientRegistrationRepository clientRegistrationRepository) {
              DefaultOAuth2AuthorizationRequestResolver resolver = new DefaultOAuth2AuthorizationRequestResolver(clientRegistrationRepository,
                      OAuth2AuthorizationRequestRedirectFilter.DEFAULT_AUTHORIZATION_REQUEST_BASE_URI);
              resolver.setAuthorizationRequestCustomizer(WechatOAuth2AuthRequestBuilderCustomizer::customize);
              return resolver;
          }
          配置到Spring Security

          適配好的OAuth2AuthorizationRequestResolver配置到HttpSecurity,偽代碼:

              httpSecurity.oauth2Login()
                          //  定制化授權(quán)端點(diǎn)的參數(shù)封裝
                          .authorizationEndpoint().authorizationRequestResolver(authorizationRequestResolver)

          通過code換取網(wǎng)頁授權(quán)access_token

          接下來第二步是用code去換token。

          構(gòu)建請求參數(shù)

          這是微信網(wǎng)頁授權(quán)獲取access_token的模板:

          GET https://api.weixin.qq.com/sns/oauth2/refresh_token?appid=APPID&grant_type=refresh_token&refresh_token=REFRESH_TOKEN

          其中前半段https://api.weixin.qq.com/sns/oauth2/refresh_token可以通過配置OAuth2.0的token-uri來指定;后半段參數(shù)需要我們針對微信進(jìn)行定制。Spring Security中定制token-uri的工具由OAuth2AuthorizationCodeGrantRequestEntityConverter這個(gè)轉(zhuǎn)換器負(fù)責(zé),這里需要來改造一下。

          我們先拼接參數(shù):

              private MultiValueMap<String, String> buildWechatQueryParameters(OAuth2AuthorizationCodeGrantRequest authorizationCodeGrantRequest) {
                  // 獲取微信的客戶端配置
                  ClientRegistration clientRegistration = authorizationCodeGrantRequest.getClientRegistration();
                  OAuth2AuthorizationExchange authorizationExchange = authorizationCodeGrantRequest.getAuthorizationExchange();
                  MultiValueMap<String, String> formParameters = new LinkedMultiValueMap<>();
                  // grant_type
                  formParameters.add(OAuth2ParameterNames.GRANT_TYPE, authorizationCodeGrantRequest.getGrantType().getValue());
                  // code
                  formParameters.add(OAuth2ParameterNames.CODE, authorizationExchange.getAuthorizationResponse().getCode());
                  // 如果有redirect-uri
                  String redirectUri = authorizationExchange.getAuthorizationRequest().getRedirectUri();
                  if (redirectUri != null) {
                      formParameters.add(OAuth2ParameterNames.REDIRECT_URI, redirectUri);
                  }
                  //appid
                  formParameters.add("appid", clientRegistration.getClientId());
                  //secret
                  formParameters.add("secret", clientRegistration.getClientSecret());
                  return formParameters;
              }

          然后生成RestTemplate的請求對象RequestEntity:

              @Override
              public RequestEntity<?> convert(OAuth2AuthorizationCodeGrantRequest authorizationCodeGrantRequest) {
                  ClientRegistration clientRegistration = authorizationCodeGrantRequest.getClientRegistration();
                  HttpHeaders headers = getTokenRequestHeaders(clientRegistration);


                  String tokenUri = clientRegistration.getProviderDetails().getTokenUri();
                  // 針對微信的定制  WECHAT_ID表示為微信公眾號專用的registrationId
                  if (WECHAT_ID.equals(clientRegistration.getRegistrationId())) {
                      MultiValueMap<String, String> queryParameters = this.buildWechatQueryParameters(authorizationCodeGrantRequest);
                      URI uri = UriComponentsBuilder.fromUriString(tokenUri).queryParams(queryParameters).build().toUri();
                      return RequestEntity.get(uri).headers(headers).build();
                  }
                  // 其它 客戶端
                  MultiValueMap<String, String> formParameters = this.buildFormParameters(authorizationCodeGrantRequest);
                  URI uri = UriComponentsBuilder.fromUriString(tokenUri).build()
                          .toUri();
                  return new RequestEntity<>(formParameters, headers, HttpMethod.POST, uri);
              }

          這樣兼容性就改造好了。

          兼容token返回解析

          微信公眾號授權(quán)token-uri的返回值雖然文檔說是個(gè)json,可它喵的Content-Typetext-plain。如果是application/jsonSpring Security就直接接收了。你說微信坑不坑?我們只能再寫個(gè)適配來正確的反序列化微信接口的返回值。

          Spring Security 中對token-uri的返回值的解析轉(zhuǎn)換由OAuth2AccessTokenResponseClient中的OAuth2AccessTokenResponseHttpMessageConverter負(fù)責(zé)。

          首先增加Content-Typetext-plain的適配;其次因?yàn)?strong style="color: rgb(53, 179, 120);">Spring Security接收token返回的對象要求必須顯式聲明tokenType,而微信返回的響應(yīng)體中沒有,我們一律指定為OAuth2AccessToken.TokenType.BEARER即可兼容。代碼比較簡單就不放了,有興趣可以去看我給的DEMO。

          配置到Spring Security

          先配置好我們上面兩個(gè)步驟的請求客戶端:

              /**
               * 調(diào)用token-uri去請求授權(quán)服務(wù)器獲取token的OAuth2 Http 客戶端
               *
               * @return OAuth2AccessTokenResponseClient
               */

              private OAuth2AccessTokenResponseClient<OAuth2AuthorizationCodeGrantRequest> accessTokenResponseClient() {
                  DefaultAuthorizationCodeTokenResponseClient tokenResponseClient = new DefaultAuthorizationCodeTokenResponseClient();
                  tokenResponseClient.setRequestEntityConverter(new WechatOAuth2AuthorizationCodeGrantRequestEntityConverter());

                  OAuth2AccessTokenResponseHttpMessageConverter tokenResponseHttpMessageConverter = new OAuth2AccessTokenResponseHttpMessageConverter();
                  // 微信返回的content-type 是 text-plain
                  tokenResponseHttpMessageConverter.setSupportedMediaTypes(Arrays.asList(MediaType.APPLICATION_JSON,
                          MediaType.TEXT_PLAIN,
                          new MediaType("application""*+json")));
                  // 兼容微信解析
                  tokenResponseHttpMessageConverter.setTokenResponseConverter(new WechatMapOAuth2AccessTokenResponseConverter());

                  RestTemplate restTemplate = new RestTemplate(
                          Arrays.asList(new FormHttpMessageConverter(),
                                  tokenResponseHttpMessageConverter
                          ));

                  restTemplate.setErrorHandler(new OAuth2ErrorResponseErrorHandler());
                  tokenResponseClient.setRestOperations(restTemplate);
                  return tokenResponseClient;
              }

          再把請求客戶端配置到HttpSecurity

             // 獲取token端點(diǎn)配置  比如根據(jù)code 獲取 token               
          httpSecurity.oauth2Login()
             .tokenEndpoint().accessTokenResponseClient(accessTokenResponseClient)

          根據(jù)token獲取用戶信息

          ?

          微信公眾號網(wǎng)頁授權(quán)獲取用戶信息需要scope包含snsapi_userinfo。

          Spring Security中定義了一個(gè)OAuth2.0獲取用戶信息的抽象接口:

          @FunctionalInterface
          public interface OAuth2UserService<R extends OAuth2UserRequestU extends OAuth2User{

           loadUser(R userRequest) throws OAuth2AuthenticationException;

          }

          所以我們針對性的實(shí)現(xiàn)即可,需要實(shí)現(xiàn)三個(gè)相關(guān)概念。

          OAuth2UserRequest

          OAuth2UserRequest是請求user-info-uri的入?yún)?shí)體,包含了三大塊屬性:

          • ClientRegistration 微信OAuth2.0客戶端配置
          • OAuth2AccessToken 從token-uri獲取的access_token的抽象實(shí)體
          • additionalParameters 一些token-uri返回的額外參數(shù),比如openid就可以從這里面取得

          根據(jù)微信獲取用戶信息的端點(diǎn)API這個(gè)能滿足需要,不過需要注意的是。如果使用的是 OAuth2.0 Client 就無法從additionalParameters獲取openid等額外參數(shù)。

          OAuth2User

          這個(gè)用來封裝微信用戶信息,細(xì)節(jié)看下面的注釋:

          /**
           * 微信授權(quán)的OAuth2User用戶信息
           *
           * @author n1
           * @since 2021/8/12 17:37
           */

          @Data
          public class WechatOAuth2User implements OAuth2User {
              private String openid;
              private String nickname;
              private Integer sex;
              private String province;
              private String city;
              private String country;
              private String headimgurl;
              private List<String> privilege;
              private String unionid;


              @Override
              public Map<String, Object> getAttributes() {
                  // 原本返回前端token 但是微信給的token比較敏感 所以不返回
                  return Collections.emptyMap();
              }

              @Override
              public Collection<? extends GrantedAuthority> getAuthorities() {
                  // 這里放scopes 或者其它你業(yè)務(wù)邏輯相關(guān)的用戶權(quán)限集 目前沒有什么用
                  return null;
              }

              @Override
              public String getName() {
                  // 用戶唯一標(biāo)識(shí)比較合適,這個(gè)不能為空啊,如果你能保證unionid不為空,也是不錯(cuò)的選擇。
                  return openid;
              }
          }

          ?

          注意: getName()一定不能返回null。

          OAuth2UserService

          參數(shù)OAuth2UserRequest和返回值OAuth2User都準(zhǔn)備好了,就剩下去請求微信服務(wù)器了。借鑒請求token-uri的實(shí)現(xiàn),還是一個(gè)RestTemplate調(diào)用,核心就這幾行:

          LinkedMultiValueMap<String, String> queryParams = new LinkedMultiValueMap<>();
          // access_token
          queryParams.add(OAuth2ParameterNames.ACCESS_TOKEN, userRequest.getAccessToken().getTokenValue());
          // openid
          queryParams.add(OPENID_KEY, String.valueOf(userRequest.getAdditionalParameters().get(OPENID_KEY)));
          // lang=zh_CN
          queryParams.add(LANG_KEY, DEFAULT_LANG);
          // 構(gòu)建 user-info-uri端點(diǎn)
          URI userInfoEndpoint = UriComponentsBuilder.fromUriString(userInfoUri).queryParams(queryParams).build().toUri();
          // 請求
          return this.restOperations.exchange(userInfoEndpoint, HttpMethod.GET, null, OAUTH2_USER_OBJECT);
          配置到Spring Security
          // 獲取用戶信息端點(diǎn)配置  根據(jù)accessToken獲取用戶基本信息
          httpSecurity.oauth2Login()
                .userInfoEndpoint().userService(oAuth2UserService);

          這里補(bǔ)充一下,寫一個(gè)授權(quán)成功后跳轉(zhuǎn)的接口并配置為授權(quán)登錄成功后的跳轉(zhuǎn)的url。

          // 默認(rèn)跳轉(zhuǎn)到 /  如果沒有會(huì) 404 所以弄個(gè)了接口
          httpSecurity.oauth2Login().defaultSuccessUrl("/weixin/h5/redirect")

          在這個(gè)接口里可以通過@RegisteredOAuth2AuthorizedClient@AuthenticationPrincipal分別拿到認(rèn)證客戶端的信息和用戶信息。

          @GetMapping("/h5/redirect")
          public void sendRedirect(HttpServletResponse response,
                                   @RegisteredOAuth2AuthorizedClient("wechat")
           OAuth2AuthorizedClient authorizedClient,
                                   @AuthenticationPrincipal WechatOAuth2User principal) throws IOException 
          {
              //todo 你可以再這里模擬一些授權(quán)后的業(yè)務(wù)邏輯 比如用戶靜默注冊 等等

              // 當(dāng)前認(rèn)證的客戶端 token 不要暴露給前臺(tái)
              OAuth2AccessToken accessToken = authorizedClient.getAccessToken();
              System.out.println("accessToken = " + accessToken);
              // 當(dāng)前用戶的userinfo
              System.out.println("principal = " + principal);
              response.sendRedirect("https://felord.cn");
          }

          到此微信公眾號授權(quán)就集成到Spring Security中了。

          相關(guān)配置

          application.yaml相關(guān)的配置:

          spring:
            security:
              oauth2:
                client:
                  registration:
                    wechat:
                      # 可以去試一下沙箱
                      # 公眾號服務(wù)號 appid
                      client-id: wxdf9033184b2xxx38e7f
                      # 公眾號服務(wù)號 secret
                      client-secret: bf1306baaa0dxxxxxxb15eb02d68df5
                      # oauth2 login 用 '{baseUrl}/login/oauth2/code/{registrationId}' 會(huì)自動(dòng)解析
                      # oauth2 client 寫你業(yè)務(wù)的鏈接即可
                      redirect-uri:  '{baseUrl}/login/oauth2/code/{registrationId}'
                      authorization-grant-type: authorization_code
                      scope: snsapi_userinfo
                  provider:
                    wechat:
                      authorization-uri: https://open.weixin.qq.com/connect/oauth2/authorize
                      token-uri: https://api.weixin.qq.com/sns/oauth2/access_token
                      user-info-uri: https://api.weixin.qq.com/sns/userinfo

          相關(guān)的DEMO可以通過微信公眾號:碼農(nóng)小胖哥, 私信回復(fù)wechatoauth2獲取。


          往期推薦

          Spring官宣新家族成員:Spring Authorization Server!

          我們自嘲的“碼農(nóng)”身份被官方實(shí)錘了!

          YYDS 的 IDEA插件,沒裝上的安排起來!

          混合辦公時(shí)代來了?攜程試點(diǎn)每周兩天居家辦公,76%的員工主動(dòng)報(bào)名!

          騰訊年度性愛報(bào)告發(fā)布,最后一條數(shù)據(jù)羞愧了...


          瀏覽 44
          點(diǎn)贊
          評論
          收藏
          分享

          手機(jī)掃一掃分享

          分享
          舉報(bào)
          評論
          圖片
          表情
          推薦
          點(diǎn)贊
          評論
          收藏
          分享

          手機(jī)掃一掃分享

          分享
          舉報(bào)
          <kbd id="afajh"><form id="afajh"></form></kbd>
          <strong id="afajh"><dl id="afajh"></dl></strong>
            <del id="afajh"><form id="afajh"></form></del>
                1. <th id="afajh"><progress id="afajh"></progress></th>
                  <b id="afajh"><abbr id="afajh"></abbr></b>
                  <th id="afajh"><progress id="afajh"></progress></th>
                  丁香五月天在线视频 | 黑人猛操白人嫩逼 | 国产一级A黄色片在线 | 欧美,日韩,另类,日韩 | 唐嫣一区二区三区在线 |