日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術(shù)文章
文章詳情頁

Spring Security實(shí)現(xiàn)微信公眾號(hào)網(wǎng)頁授權(quán)功能

瀏覽:209日期:2022-07-16 17:33:37

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

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

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

微信公眾號(hào)服務(wù)號(hào)

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

微信公眾號(hào)服務(wù)號(hào)只有企事業(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)穿透工具提供的虛擬域名配置到微信測試帳號(hào)的回調(diào)配置中

Spring Security實(shí)現(xiàn)微信公眾號(hào)網(wǎng)頁授權(quán)功能

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

Spring Security實(shí)現(xiàn)微信公眾號(hào)網(wǎng)頁授權(quán)功能

然后我們就可以開發(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)給公眾號(hào)后,公眾號(hào)可以獲取到一個(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

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

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

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,由于微信的不一樣所以我們針對(duì)性的定制,也就是改造OAuth2AuthorizationRequestRedirectFilter中的OAuth2AuthorizationRequestResolver。

自定義URL

因?yàn)镾pring Security會(huì)根據(jù)模板鏈接去組裝一個(gè)鏈接而不是我們填參數(shù)就行了,所以需要我們對(duì)構(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請求對(duì)象{@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ù)需要我們針對(duì)微信進(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_typeformParameters.add(OAuth2ParameterNames.GRANT_TYPE, authorizationCodeGrantRequest.getGrantType().getValue());// codeformParameters.add(OAuth2ParameterNames.CODE, authorizationExchange.getAuthorizationResponse().getCode());// 如果有redirect-uriString redirectUri = authorizationExchange.getAuthorizationRequest().getRedirectUri();if (redirectUri != null) { formParameters.add(OAuth2ParameterNames.REDIRECT_URI, redirectUri);}//appidformParameters.add('appid', clientRegistration.getClientId());//secretformParameters.add('secret', clientRegistration.getClientSecret());return formParameters; }

然后生成RestTemplate的請求對(duì)象RequestEntity:

@Override public RequestEntity<?> convert(OAuth2AuthorizationCodeGrantRequest authorizationCodeGrantRequest) {ClientRegistration clientRegistration = authorizationCodeGrantRequest.getClientRegistration();HttpHeaders headers = getTokenRequestHeaders(clientRegistration);String tokenUri = clientRegistration.getProviderDetails().getTokenUri();// 針對(duì)微信的定制 WECHAT_ID表示為微信公眾號(hào)專用的registrationIdif (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返回解析

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

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

首先增加Content-Type為text-plain的適配;其次因?yàn)镾pring Security接收token返回的對(duì)象要求必須顯式聲明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-plaintokenResponseHttpMessageConverter.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獲取用戶信息

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

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

@FunctionalInterfacepublic interface OAuth2UserService<R extends OAuth2UserRequest, U extends OAuth2User> {U loadUser(R userRequest) throws OAuth2AuthenticationException;}

所以我們針對(duì)性的實(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 */@Datapublic 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_tokenqueryParams.add(OAuth2ParameterNames.ACCESS_TOKEN, userRequest.getAccessToken().getTokenValue());// openidqueryParams.add(OPENID_KEY, String.valueOf(userRequest.getAdditionalParameters().get(OPENID_KEY)));// lang=zh_CNqueryParams.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');}

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

相關(guān)配置

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

spring: security: oauth2: client:registration: wechat: # 可以去試一下沙箱 # 公眾號(hào)服務(wù)號(hào) appid client-id: wxdf9033184b2xxx38e7f # 公眾號(hào)服務(wù)號(hào) 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_userinfoprovider: 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)于Spring Security中實(shí)現(xiàn)微信網(wǎng)頁授權(quán)的文章就介紹到這了,更多相關(guān)Spring Security微信網(wǎng)頁授權(quán)內(nèi)容請搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持好吧啦網(wǎng)!

標(biāo)簽: 微信
相關(guān)文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
波多视频一区| 欧美日韩三区| 日韩激情精品| 日本vs亚洲vs韩国一区三区二区| 在线 亚洲欧美在线综合一区| 国产综合亚洲精品一区二| 欧美日一区二区| 欧美日韩国产免费观看| 亚洲一区日韩在线| 久久亚洲欧洲| 亚洲精品伦理| 日韩高清一级| 国产日韩欧美一区在线| 国产精品久久久久久模特 | 亚洲欧美专区| 日韩一区二区三区在线看| 欧美影院视频| 欧美国产另类| 国产一区二区精品久| 亚洲成人一区在线观看| 日韩中文视频| 狠狠色狠狠色综合日日tαg| 日韩一区欧美二区| 日韩激情中文字幕| 久久精品二区亚洲w码| 日韩久久一区二区三区| 99视频精品全部免费在线视频| 国产精品99一区二区| 一区福利视频| 午夜电影一区| 精品国产三区在线| 亚洲网站视频| 亚洲一区有码| 国产精品超碰| 久久久9色精品国产一区二区三区| 欧美精选一区二区三区| 亚洲区国产区| 久久69成人| 亚洲大片在线| 日精品一区二区三区| 麻豆国产精品777777在线| 欧美好骚综合网| 亚洲女同中文字幕| 欧美日韩国产一区二区在线观看| 国产精品99视频| 婷婷综合在线| 日韩超碰人人爽人人做人人添| 国精品产品一区| 最新亚洲激情| 国产经典一区| 亚洲深夜影院| 国产激情久久| 亚洲高清激情| 欧美日韩一区二区三区四区在线观看| 91视频精品| 亚洲综合婷婷| sm捆绑调教国产免费网站在线观看| 久久久天天操| 欧美亚洲tv| 红桃视频国产精品| 麻豆精品视频在线观看免费| 欧美日韩日本国产亚洲在线| 国产精品嫩模av在线| 久久网站免费观看| 国产精品久久久免费| 久久国产精品久久久久久电车 | 日av在线不卡| 久久久久久夜| 日韩欧美美女在线观看| 日本精品不卡| 日韩国产一二三区| 亚洲va在线| 国产精品xxxav免费视频| 免费不卡中文字幕在线| 欧美激情综合| 天堂av在线一区| 日本精品不卡| 国产黄色一区| 亚洲精品第一| 亚洲先锋成人| 精品国产一区二| 午夜电影一区| 激情欧美一区二区三区| 韩国女主播一区二区三区| 中文字幕亚洲影视| 秋霞影院一区二区三区| 国产精品激情| 亚洲精品观看| 亚洲一区成人| 麻豆精品蜜桃| 精品不卡一区| 欧美性www| 中文字幕一区二区三区日韩精品 | 波多野结衣一区| 精品不卡一区| 国产经典一区| 国产日韩亚洲| 日韩国产一二三区| 午夜欧美视频| 私拍精品福利视频在线一区| 久久99精品久久久野外观看| 日本一不卡视频| 噜噜噜躁狠狠躁狠狠精品视频 | 人人精品久久| 视频一区中文字幕精品| 视频一区二区三区入口| 国精品一区二区| 欧美日韩精品免费观看视欧美高清免费大片 | 亚洲激情欧美| 91精品精品| 97人人精品| 国产一区二区三区天码| 国产精品1区在线| 欧美国产免费| 麻豆一区二区99久久久久| 国产免费播放一区二区| 日本不卡一区二区三区| 综合激情一区| 免费黄网站欧美| 玖玖精品视频| 午夜电影一区| 国产亚洲欧美日韩精品一区二区三区 | 国产毛片精品| 久久黄色影视| 国产欧美日韩在线观看视频 | 日韩精品视频网站| 日韩精品久久理论片| 日韩午夜视频在线| 日韩精品一区二区三区中文字幕| 亚洲精品一二| 日韩精品一区二区三区中文| 日韩精彩视频在线观看| 欧美一区成人| 久久99性xxx老妇胖精品| 免费在线亚洲欧美| 国产精品对白久久久久粗| 免费在线日韩av| 久久uomeier| 久久国产中文字幕| 欧美粗暴jizz性欧美20| 狠狠久久婷婷| 亚洲3区在线| 国产精品一区二区三区av麻| 精品一区视频| 亚洲成人不卡| 日韩午夜免费| 亚洲精品国产精品粉嫩| 国产欧美二区| av资源新版天堂在线| 99精品美女| 免费在线欧美视频| 欧美中文一区| 国产精品久久久久久久免费观看 | 亚洲午夜久久久久久尤物| 亚洲一区网站| 日韩高清二区| 91综合网人人| 亚洲在线免费| 欧美日韩18| 97国产成人高清在线观看| 免费观看不卡av| 日韩高清电影一区| 色一区二区三区四区| 欧美va天堂| 日韩国产欧美视频| а√天堂中文在线资源8| 久久美女精品| 中文字幕一区二区精品区| 欧美激情精品| 免费黄色成人| 91欧美精品| 欧洲av不卡| 亚州av一区| 国产成人精品亚洲线观看| 亚洲激情不卡| 国产日产精品一区二区三区四区的观看方式| 久久精品午夜| 亚洲欧洲午夜| 久久精品国产精品亚洲毛片| 黄色亚洲精品| 麻豆中文一区二区| 欧美在线资源| 国产欧美一区二区色老头| 99久久婷婷| 国产亚洲电影| 99久久婷婷| 国产精品欧美在线观看| 亚洲性色视频| 国产毛片精品久久| 国产精品99一区二区| 国产毛片精品久久| 欧美日韩国产在线观看网站 | 亚洲免费影视| 成人精品视频| 亚洲伊人精品酒店| 亚洲精品国产嫩草在线观看| 亚洲区国产区| 久久国产中文字幕| 国产精品**亚洲精品| 日韩影院精彩在线|