日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程语言 > java >内容正文

java

使用Java EE和OIDC构建Java REST API

發布時間:2023/12/3 java 37 豆豆
生活随笔 收集整理的這篇文章主要介紹了 使用Java EE和OIDC构建Java REST API 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

“我喜歡編寫身份驗證和授權代碼。” ?從來沒有Java開發人員。 厭倦了一次又一次地建立相同的登錄屏幕? 嘗試使用Okta API進行托管身份驗證,授權和多因素身份驗證。

Java EE允許您使用JAX-RS和JPA快速輕松地構建Java REST API。 Java EE是保護傘標準規范,它描述了許多Java技術,包括EJB,JPA,JAX-RS和許多其他技術。 它最初旨在允許Java應用程序服務器之間的可移植性,并在2000年代初期蓬勃發展。 那時,應用服務器非常流行,并且由許多知名公司(例如IBM,BEA和Sun)提供。 JBoss是一家新興公司,它破壞了現狀,并表明可以將Java EE應用程序服務器開發為一個開源項目,并免費提供它。 JBoss在2006年被RedHat收購。

在2000年代初期,Java開發人員使用servlet和EJB來開發其服務器應用程序。 Hibernate和Spring分別于2002年和2004年問世。 兩種技術都對各地的Java開發人員產生了巨大的影響,這表明他們可以編寫沒有EJB的分布式健壯應用程序。 Hibernate的POJO模型最終被用作JPA標準,并且對EJB的影響也很大。

快進到2018年,Java EE肯定不像以前那樣! 現在,它主要是POJO和注釋,并且使用起來更簡單。

為什么要使用Java EE而不是Spring Boot構建Java REST API?

Spring Boot是Java生態系統中我最喜歡的技術之一。 它極大地減少了Spring應用程序中必需的配置,并使得僅用幾行代碼即可生成REST API。 但是,最近有一些不使用Spring Boot的開發人員提出了很多API安全性問題。 其中一些甚至沒有使用Spring!

基于這個原因,我認為構建一個Java REST API(使用Java EE)很有趣,該API與我過去開發的Spring Boot REST API相同。 即,我的Bootiful Angular和Bootiful React帖子中的“啤酒” API。

使用Java EE構建Java REST API

首先,我在Twitter上詢問了我的網絡,是否存在諸如start.spring.io之類的Java EE快速入門。 我收到了一些建議,并開始進行一些研究。 David Blevins建議我看一下tomee-jaxrs-starter-project ,所以我從那里開始。 我還研究了Roberto Cortez推薦的TomEE Maven原型 。

我喜歡jaxrs-starter項目,因為它展示了如何使用JAX-RS創建REST API。 TomEE Maven原型也很有用,特別是因為它展示了如何使用JPA,H2和JSF。 我將兩者結合起來,創建了自己的最小啟動器,可用于在TomEE上實現安全的Java EE API。 您不必在這些示例中使用TomEE,但我尚未在其他實現上對其進行測試。

如果您在其他應用服務器上使用了這些示例,請告訴我,我將更新此博客文章。

在這些示例中,我將使用Java 8和Java EE 7.0以及TomEE 7.1.0。 TomEE 7.x是EE 7兼容版本; 有一個TomEE 8.x分支用于EE8兼容性工作,但尚無發行版本。 我希望您也安裝了Apache Maven 。

首先,將我們的Java EE REST API存儲庫克隆到您的硬盤驅動器,然后運行它:

git clone https://github.com/oktadeveloper/okta-java-ee-rest-api-example.git javaee-rest-api cd javaee-rest-api mvn package tomee:run

導航到http:// localhost:8080并添加新啤酒。

單擊添加 ,您應該看到成功消息。

單擊查看存在的啤酒查看啤酒的完整列表。


您還可以在http://localhost:8080/good-beers查看系統中的優質啤酒列表。 以下是使用HTTPie時的輸出。

$ http :8080/good-beers HTTP/1.1 200 Content-Type: application/json Date: Wed, 29 Aug 2018 21:58:23 GMT Server: Apache TomEE Transfer-Encoding: chunked[{"id": 101,"name": "Kentucky Brunch Brand Stout"},{"id": 102,"name": "Marshmallow Handjee"},{"id": 103,"name": "Barrel-Aged Abraxas"},{"id": 104,"name": "Heady Topper"},{"id": 108,"name": "White Rascal"} ]

使用Java EE構建REST API

我向您展示了該應用程序可以做什么,但是我還沒有談論它是如何構建的。 它有一些XML配置文件,但我將跳過其中的大多數。 目錄結構如下所示:

$ tree . . ├── LICENSE ├── README.md ├── pom.xml └── src├── main│ ├── java│ │ └── com│ │ └── okta│ │ └── developer│ │ ├── Beer.java│ │ ├── BeerBean.java│ │ ├── BeerResource.java│ │ ├── BeerService.java│ │ └── StartupBean.java│ ├── resources│ │ └── META-INF│ │ └── persistence.xml│ └── webapp│ ├── WEB-INF│ │ ├── beans.xml│ │ └── faces-config.xml│ ├── beer.xhtml│ ├── index.jsp│ └── result.xhtml└── test└── resources└── arquillian.xml12 directories, 16 files

最重要的XML文件是pom.xml ,它定義了依賴關系,并允許您運行TomEE Maven插件。 它非常簡短,可愛,只有一個依賴項和一個插件。

<?xml version="1.0" encoding="UTF-8"?> <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd"><modelVersion>4.0.0</modelVersion><groupId>com.okta.developer</groupId><artifactId>java-ee-rest-api</artifactId><version>1.0-SNAPSHOT</version><packaging>war</packaging><name>Java EE Webapp with JAX-RS API</name><url>http://developer.okta.com</url><properties><project.build.sourceEncoding>UTF-8</project.build.sourceEncoding><project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding><maven.compiler.target>1.8</maven.compiler.target><maven.compiler.source>1.8</maven.compiler.source><failOnMissingWebXml>false</failOnMissingWebXml><javaee-api.version>7.0</javaee-api.version><tomee.version>7.1.0</tomee.version></properties><dependencies><dependency><groupId>javax</groupId><artifactId>javaee-api</artifactId><version>${javaee-api.version}</version><scope>provided</scope></dependency></dependencies><build><plugins><plugin><groupId>org.apache.tomee.maven</groupId><artifactId>tomee-maven-plugin</artifactId><version>${tomee.version}</version><configuration><context>ROOT</context></configuration></plugin></plugins></build> </project>

主要實體是Beer.java 。

package com.okta.developer;import javax.persistence.Entity; import javax.persistence.GeneratedValue; import javax.persistence.GenerationType; import javax.persistence.Id;@Entity public class Beer {@Id@GeneratedValue(strategy = GenerationType.AUTO)private int id;private String name;public Beer() {}public Beer(String name) {this.name = name;}public int getId() {return id;}public void setId(int id) {this.id = id;}public String getName() {return name;}public void setName(String beerName) {this.name = beerName;}@Overridepublic String toString() {return "Beer{" +"id=" + id +", name='" + name + '\'' +'}';} }

數據庫(aka,數據源)在src/main/resources/META-INF/persistence.xml 。

<?xml version="1.0" encoding="UTF-8"?> <persistence version="2.0" xmlns="http://java.sun.com/xml/ns/persistence"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://java.sun.com/xml/ns/persistence http://java.sun.com/xml/ns/persistence/persistence_2_0.xsd"><persistence-unit name="beer-pu" transaction-type="JTA"><jta-data-source>beerDatabase</jta-data-source><class>com.okta.developer.Beer</class><properties><property name="openjpa.jdbc.SynchronizeMappings" value="buildSchema(ForeignKeys=true)"/></properties></persistence-unit> </persistence>

BeerService.java類使用JPA的EntityManager處理該實體的讀取并將其保存到數據庫中。

package com.okta.developer;import javax.ejb.Stateless; import javax.persistence.EntityManager; import javax.persistence.PersistenceContext; import javax.persistence.Query; import javax.persistence.criteria.CriteriaQuery; import java.util.List;@Stateless public class BeerService {@PersistenceContext(unitName = "beer-pu")private EntityManager entityManager;public void addBeer(Beer beer) {entityManager.persist(beer);}public List<Beer> getAllBeers() {CriteriaQuery<Beer> cq = entityManager.getCriteriaBuilder().createQuery(Beer.class);cq.select(cq.from(Beer.class));return entityManager.createQuery(cq).getResultList();}public void clear() {Query removeAll = entityManager.createQuery("delete from Beer");removeAll.executeUpdate();} }

有一個StartupBean.java ,用于在啟動時填充數據庫,并在關閉時清除數據庫。

package com.okta.developer;import javax.annotation.PostConstruct; import javax.annotation.PreDestroy; import javax.ejb.Singleton; import javax.ejb.Startup; import javax.inject.Inject; import java.util.stream.Stream;@Singleton @Startup public class StartupBean {private final BeerService beerService;@Injectpublic StartupBean(BeerService beerService) {this.beerService = beerService;}@PostConstructprivate void startup() {// Top beers from https://www.beeradvocate.com/lists/top/Stream.of("Kentucky Brunch Brand Stout", "Marshmallow Handjee", "Barrel-Aged Abraxas", "Heady Topper","Budweiser", "Coors Light", "PBR").forEach(name ->beerService.addBeer(new Beer(name)));beerService.getAllBeers().forEach(System.out::println);}@PreDestroyprivate void shutdown() {beerService.clear();} }

這三個類構成了應用程序的基礎,此外還有一個BeerResource.java類,它使用JAX-RS公開/good-beers端點。

package com.okta.developer;import javax.ejb.Lock; import javax.ejb.Singleton; import javax.inject.Inject; import javax.ws.rs.GET; import javax.ws.rs.Path; import javax.ws.rs.Produces; import java.util.List; import java.util.stream.Collectors;import static javax.ejb.LockType.READ; import static javax.ws.rs.core.MediaType.APPLICATION_JSON;@Lock(READ) @Singleton @Path("/good-beers") public class BeerResource {private final BeerService beerService;@Injectpublic BeerResource(BeerService beerService) {this.beerService = beerService;}@GET@Produces({APPLICATION_JSON})public List<Beer> getGoodBeers() {return beerService.getAllBeers().stream().filter(this::isGreat).collect(Collectors.toList());}private boolean isGreat(Beer beer) {return !beer.getName().equals("Budweiser") &&!beer.getName().equals("Coors Light") &&!beer.getName().equals("PBR");} }

最后,有一個BeerBean.java類用作JSF的托管bean。

package com.okta.developer;import javax.enterprise.context.RequestScoped; import javax.inject.Inject; import javax.inject.Named; import java.util.List;@Named @RequestScoped public class BeerBean {@Injectprivate BeerService beerService;private List<Beer> beersAvailable;private String name;public String getName() {return name;}public void setName(String name) {this.name = name;}public List<Beer> getBeersAvailable() {return beersAvailable;}public void setBeersAvailable(List<Beer> beersAvailable) {this.beersAvailable = beersAvailable;}public String fetchBeers() {beersAvailable = beerService.getAllBeers();return "success";}public String add() {Beer beer = new Beer();beer.setName(name);beerService.addBeer(beer);return "success";} }

您現在擁有了使用Java EE構建的REST API! 但是,這并不安全。 在以下各節中,我將向您展示如何使用Okta的Java JWT驗證程序,Spring Security和Pac4j對其進行保護。

使用Okta將OIDC安全性添加到Java REST API

您將需要在Okta中創建OIDC應用程序,以驗證將要實施的安全配置。 要使此操作毫不費力,您可以使用Okta的OIDC API。 在Okta,我們的目標是使身份管理比您以往更加輕松,安全和可擴展。 Okta是一項云服務,允許開發人員創建,編輯和安全地存儲用戶帳戶和用戶帳戶數據,并將它們與一個或多個應用程序連接。 我們的API使您能夠:

  • 驗證和授權用戶
  • 存儲有關您的用戶的數據
  • 執行基于密碼的社交登錄
  • 通過多因素身份驗證保護您的應用程序
  • 以及更多! 查看我們的產品文檔

你賣了嗎 立即注冊一個永久免費的開發者帳戶 ! 完成后,請完成以下步驟以創建OIDC應用程序。

  • 登錄到您在developer.okta.com上的開發者帳戶。
  • 導航到應用程序 ,然后單擊添加應用程序
  • 選擇“ Web” ,然后單擊“ 下一步”
  • 為應用程序命名(例如Java EE Secure API ),然后添加以下內容作為登錄重定向URI:
    • http://localhost:3000/implicit/callback
    • http://localhost:8080/login/oauth2/code/okta
    • http://localhost:8080/callback?client_name=OidcClient
  • 單擊完成 ,然后編輯項目并啟用“隱式(混合)”作為授予類型(允許ID和訪問令牌),然后單擊保存
  • 使用JWT Verifier保護Java REST API

    要從Okta驗證JWT,您需要將Okta Java JWT Verifier添加到pom.xml 。

    <properties>...<okta-jwt.version>0.3.0</okta-jwt.version> </properties><dependencies>...<dependency><groupId>com.okta.jwt</groupId><artifactId>okta-jwt-verifier</artifactId><version>${okta-jwt.version}</version></dependency> </dependencies>

    然后創建一個JwtFilter.java (在src/main/java/com/okta/developer目錄中)。 該過濾器查找其中包含訪問令牌的authorization標頭。 如果存在,它將對其進行驗證并打印出用戶的sub ,也就是他們的電子郵件地址。 如果不存在或有效,則返回拒絕訪問狀態。

    確保使用您創建的應用中的設置替換{yourOktaDomain}和{clientId} 。

    package com.okta.developer;import com.nimbusds.oauth2.sdk.ParseException; import com.okta.jwt.JoseException; import com.okta.jwt.Jwt; import com.okta.jwt.JwtHelper; import com.okta.jwt.JwtVerifier;import javax.servlet.*; import javax.servlet.annotation.WebFilter; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException;@WebFilter(filterName = "jwtFilter", urlPatterns = "/*") public class JwtFilter implements Filter {private JwtVerifier jwtVerifier;@Overridepublic void init(FilterConfig filterConfig) {try {jwtVerifier = new JwtHelper().setIssuerUrl("https://{yourOktaDomain}/oauth2/default").setClientId("{yourClientId}").build();} catch (IOException | ParseException e) {System.err.print("Configuring JWT Verifier failed!");e.printStackTrace();}}@Overridepublic void doFilter(ServletRequest servletRequest, ServletResponse servletResponse,FilterChain chain) throws IOException, ServletException {HttpServletRequest request = (HttpServletRequest) servletRequest;HttpServletResponse response = (HttpServletResponse) servletResponse;System.out.println("In JwtFilter, path: " + request.getRequestURI());// Get access token from authorization headerString authHeader = request.getHeader("authorization");if (authHeader == null) {response.sendError(HttpServletResponse.SC_UNAUTHORIZED, "Access denied.");return;} else {String accessToken = authHeader.substring(authHeader.indexOf("Bearer ") + 7);try {Jwt jwt = jwtVerifier.decodeAccessToken(accessToken);System.out.println("Hello, " + jwt.getClaims().get("sub"));} catch (JoseException e) {e.printStackTrace();response.sendError(HttpServletResponse.SC_UNAUTHORIZED, "Access denied.");return;}}chain.doFilter(request, response);}@Overridepublic void destroy() {} }

    為確保此過濾器正常工作,請重新啟動您的應用并運行:

    mvn package tomee:run

    如果在瀏覽器中導航到http://localhost:8080/good-beers ,則會看到拒絕訪問錯誤。

    為了證明它可以與有效的JWT一起使用,您可以克隆我的Bootiful React項目,并運行其UI:

    git clone -b okta https://github.com/oktadeveloper/spring-boot-react-example.git bootiful-react cd bootiful-react/client npm install

    編輯此項目的client/src/App.tsx文件,并更改issuer和clientId以匹配您的應用程序。

    const config = {issuer: 'https://{yourOktaDomain}/oauth2/default',redirectUri: window.location.origin + '/implicit/callback',clientId: '{yourClientId}' };

    然后啟動它:

    npm start

    然后,您應該能夠使用創建帳戶所用的憑據登錄http://localhost:3000 。 但是,由于CORS錯誤(在瀏覽器的開發人員控制臺中),您將無法從API加載任何啤酒。

    Failed to load http://localhost:8080/good-beers: Response to preflight request doesn't pass access control check: No 'Access-Control-Allow-Origin' header is present on the requested resource. Origin 'http://localhost:3000' is therefore not allowed access.

    提示:如果看到401并且沒有CORS錯誤,則可能意味著您的客戶ID不匹配。

    要解決此CORS錯誤,請在JwtFilter.java類旁邊添加一個CorsFilter.java 。 下面的過濾器將允許OPTIONS請求,并向后發送訪問控制標頭,以允許任何起源,GET方法和任何標頭。 我建議您在生產中使這些設置更加具體。

    package com.okta.developer;import javax.servlet.*; import javax.servlet.annotation.WebFilter; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException;@WebFilter(filterName = "corsFilter") public class CorsFilter implements Filter {@Overridepublic void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain chain)throws IOException, ServletException {HttpServletRequest request = (HttpServletRequest) servletRequest;HttpServletResponse response = (HttpServletResponse) servletResponse;System.out.println("In CorsFilter, method: " + request.getMethod());// Authorize (allow) all domains to consume the contentresponse.addHeader("Access-Control-Allow-Origin", "http://localhost:3000");response.addHeader("Access-Control-Allow-Methods", "GET");response.addHeader("Access-Control-Allow-Headers", "*");// For HTTP OPTIONS verb/method reply with ACCEPTED status code -- per CORS handshakeif (request.getMethod().equals("OPTIONS")) {response.setStatus(HttpServletResponse.SC_ACCEPTED);return;}// pass the request along the filter chainchain.doFilter(request, response);}@Overridepublic void init(FilterConfig config) {}@Overridepublic void destroy() {} }

    您添加的兩個過濾器都使用@WebFilter進行注冊。 這是一個方便的注釋,但不提供任何過濾器排序功能。 要解決此丟失的功能,請修改JwtFilter ,使其@WebFilter中沒有urlPattern 。

    @WebFilter(filterName = "jwtFilter")

    然后創建一個src/main/webapp/WEB-INF/web.xml文件,并使用以下XML進行填充。 這些過濾器映射可確保CorsFilter處理CorsFilter 。

    <?xml version="1.0" encoding="UTF-8"?> <web-app version="3.1"xmlns="http://xmlns.jcp.org/xml/ns/javaee"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd"><filter-mapping><filter-name>corsFilter</filter-name><url-pattern>/*</url-pattern></filter-mapping><filter-mapping><filter-name>jwtFilter</filter-name><url-pattern>/*</url-pattern></filter-mapping> </web-app>

    重新啟動Java API,現在一切正常!

    在控制臺中,您應該看到類似于我的消息:

    In CorsFilter, method: OPTIONS In CorsFilter, method: GET In JwtFilter, path: /good-beers Hello, demo@okta.com

    使用Okta的JWT驗證程序過濾器是實現資源服務器的一種簡單方法(采用OAuth 2.0命名法)。 但是,它不向您提供有關該用戶的任何信息。 JwtVerifier接口的確有一個decodeIdToken(String idToken, String nonce)方法,但是您必須從客戶端傳遞ID令牌才能使用它。

    在接下來的兩節中,我將向您展示如何使用Spring Security和Pac4j來實現類似的安全性。 作為獎勵,我將向您展示如何提示用戶登錄(當他們嘗試直接訪問API時)并獲取用戶的信息。

    通過Spring Security保護Java REST API

    Spring Security是我在Javaland中最喜歡的框架之一。 在顯示如何使用Spring Security時,此博客上的大多數示例都使用Spring Boot。 我將使用最新版本– 5.1.0.RC2 –因此本教程將保持最新狀態。

    還原更改以添加JWT Verifier,或直接刪除web.xml繼續。

    修改您的pom.xml使其具有Spring Security所需的依賴關系。 您還需要添加Spring的快照存儲庫以獲取候選版本。

    <properties>...<spring-security.version>5.1.0.RC2</spring-security.version><spring.version>5.1.0.RC3</spring.version><jackson.version>2.9.6</jackson.version> </properties><dependencyManagement><dependencies><dependency><groupId>org.springframework</groupId><artifactId>spring-framework-bom</artifactId><version>${spring.version}</version><scope>import</scope><type>pom</type></dependency><dependency><groupId>org.springframework.security</groupId><artifactId>spring-security-bom</artifactId><version>${spring-security.version}</version><scope>import</scope><type>pom</type></dependency></dependencies> </dependencyManagement><dependencies>...<dependency><groupId>org.springframework</groupId><artifactId>spring-webmvc</artifactId></dependency><dependency><groupId>org.springframework.security</groupId><artifactId>spring-security-web</artifactId></dependency><dependency><groupId>org.springframework.security</groupId><artifactId>spring-security-config</artifactId></dependency><dependency><groupId>org.springframework.security</groupId><artifactId>spring-security-oauth2-client</artifactId></dependency><dependency><groupId>org.springframework.security</groupId><artifactId>spring-security-oauth2-resource-server</artifactId></dependency><dependency><groupId>org.springframework.security</groupId><artifactId>spring-security-oauth2-jose</artifactId></dependency><dependency><groupId>com.fasterxml.jackson.core</groupId><artifactId>jackson-core</artifactId><version>${jackson.version}</version></dependency><dependency><groupId>com.fasterxml.jackson.core</groupId><artifactId>jackson-databind</artifactId><version>${jackson.version}</version></dependency> </dependencies><pluginRepositories><pluginRepository><id>spring-snapshots</id><name>Spring Snapshots</name><url>https://repo.spring.io/libs-snapshot</url><snapshots><enabled>true</enabled></snapshots></pluginRepository> </pluginRepositories> <repositories><repository><id>spring-snapshots</id><name>Spring Snapshot</name><url>https://repo.spring.io/libs-snapshot</url></repository> </repositories>

    在src/main/java/com/okta/developer創建一個SecurityWebApplicationInitializer.java類:

    package com.okta.developer;import org.springframework.security.web.context.*;public class SecurityWebApplicationInitializerextends AbstractSecurityWebApplicationInitializer {public SecurityWebApplicationInitializer() {super(SecurityConfiguration.class);} }

    在同一目錄中創建一個SecurityConfiguration.java類。 此類使用Spring Security 5的oauth2Login()并向Spring Security注冊您的Okta應用程序。

    package com.okta.developer;import org.springframework.beans.factory.annotation.Autowired; import org.springframework.beans.factory.annotation.Value; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.PropertySource; import org.springframework.security.config.annotation.web.builders.HttpSecurity; import org.springframework.security.config.annotation.web.configuration.EnableWebSecurity; import org.springframework.security.config.annotation.web.configuration.WebSecurityConfigurerAdapter; import org.springframework.security.config.http.SessionCreationPolicy; import org.springframework.security.oauth2.client.InMemoryOAuth2AuthorizedClientService; import org.springframework.security.oauth2.client.OAuth2AuthorizedClientService; import org.springframework.security.oauth2.client.registration.ClientRegistration; import org.springframework.security.oauth2.client.registration.ClientRegistrationRepository; import org.springframework.security.oauth2.client.registration.ClientRegistrations; import org.springframework.security.oauth2.client.registration.InMemoryClientRegistrationRepository; import org.springframework.security.web.csrf.CookieCsrfTokenRepository;@Configuration @EnableWebSecurity @PropertySource("classpath:application.properties") public class SecurityConfiguration extends WebSecurityConfigurerAdapter {private final String clientSecret;private final String clientId;private final String issuerUri;@Autowiredpublic SecurityConfiguration(@Value("${okta.issuer-uri}") String issuerUri,@Value("${okta.client-id}") String clientId,@Value("${okta.client-secret}") String clientSecret) {this.issuerUri = issuerUri;this.clientId = clientId;this.clientSecret = clientSecret;}@Overrideprotected void configure(HttpSecurity http) throws Exception {http.sessionManagement().sessionCreationPolicy(SessionCreationPolicy.ALWAYS).and().csrf().csrfTokenRepository(CookieCsrfTokenRepository.withHttpOnlyFalse()).and().authorizeRequests().anyRequest().authenticated().and().oauth2Login();}@Beanpublic OAuth2AuthorizedClientService authorizedClientService() {return new InMemoryOAuth2AuthorizedClientService(clientRegistrationRepository());}@Beanpublic ClientRegistrationRepository clientRegistrationRepository() {List<ClientRegistration> registrations = clients.stream().map(this::getRegistration).filter(Objects::nonNull).collect(Collectors.toList());return new InMemoryClientRegistrationRepository(registrations);}@Beanpublic ClientRegistrationRepository clientRegistrationRepository() {ClientRegistration okta = getRegistration();return new InMemoryClientRegistrationRepository(okta);}ClientRegistrations.fromOidcIssuerLocation(Objects.requireNonNull(issuerUri)).registrationId("okta").clientId(clientId).clientSecret(clientSecret).build(); }

    創建src/main/resources/application.properties并用Okta OIDC應用設置進行填充。

    okta.client-id={clientId} okta.client-secret={clientSecret} okta.issuer-uri=https://{yourOktaDomain}/oauth2/default

    感謝Baeldung提供有關Spring Security 5 OAuth的出色文檔 。

    因為啟用了CSRF,所以必須在任何<h:form>標記內添加以下隱藏字段以保護CSRF。 我將以下內容添加到src/main/webapp/beer.xhtml和result.xhtml 。

    <input type="hidden" value="${_csrf.token}" name="${_csrf.parameterName}"/>

    重新啟動您的API( mvn clean package tomee:run )并導航到http://localhost:8080/good-beers 。 您應該重定向到Okta進行登錄。

    輸入有效的憑證,您應該在瀏覽器中看到JSON。 JSON Viewer Chrome插件提供了美觀的JSON。

    要求用戶登錄以查看您的API數據很方便,但是最好將其作為React UI示例的資源服務器。 OAuth 2.0資源服務器支持是Spring Security 5.1.0 RC1中的新增功能,因此我將向您展示如何使用它。

    用以下代碼替換SecurityConfiguration.java的configure()方法,該代碼啟用CORS并設置資源服務器。

    @Override protected void configure(HttpSecurity http) throws Exception {http.sessionManagement().sessionCreationPolicy(SessionCreationPolicy.ALWAYS).and().csrf().csrfTokenRepository(CookieCsrfTokenRepository.withHttpOnlyFalse()).and().cors().and().authorizeRequests().anyRequest().authenticated().and().oauth2Login().and().oauth2ResourceServer().jwt(); }@Bean JwtDecoder jwtDecoder() {return JwtDecoders.fromOidcIssuerLocation(this.issuerUri); }@Bean CorsConfigurationSource corsConfigurationSource() {CorsConfiguration configuration = new CorsConfiguration();configuration.setAllowCredentials(true);configuration.setAllowedOrigins(Collections.singletonList("http://localhost:3000"));configuration.setAllowedMethods(Collections.singletonList("GET"));configuration.setAllowedHeaders(Collections.singletonList("*"));UrlBasedCorsConfigurationSource source = new UrlBasedCorsConfigurationSource();source.registerCorsConfiguration("/**", configuration);return source; }

    進行這些更改之后,重新啟動您的API并確認您的React UI可以與之對話。 很漂亮吧?

    Spring Security的用戶信息

    Spring Security與Servlet API集成在一起,因此您可以使用以下方法來獲取當前用戶的信息。

    • HttpServletRequest.getRemoteUser()
    • HttpServletRequest.getUserPrincipal()

    擁有Principal ,您可以獲取有關用戶的詳細信息,包括其角色(又名,權限)。

    OAuth2Authentication authentication = (OAuth2Authentication) principal; Map<String, Object> user = (Map<String, Object>) authentication.getUserAuthentication().getDetails();

    請參閱Spring Security的Servlet API集成文檔以獲取更多信息。

    使用Pac4j鎖定Java REST API

    我想向您展示的確保Java REST API安全的最后一種技術是使用Pac4j,特別是j2e-pac4j 。

    恢復您的更改以添加Spring Security。

    git reset --hard HEAD

    編輯pom.xml以添加完成本節所需的Pac4j庫。

    <properties>...<pac4j-j2e.version>4.0.0</pac4j-j2e.version><pac4j.version>3.0.0</pac4j.version> </properties><dependencies>...<dependency><groupId>org.pac4j</groupId><artifactId>j2e-pac4j</artifactId><version>${pac4j-j2e.version}</version></dependency><dependency><groupId>org.pac4j</groupId><artifactId>pac4j-oidc</artifactId><version>${pac4j.version}</version></dependency><dependency><groupId>org.pac4j</groupId><artifactId>pac4j-http</artifactId><version>${pac4j.version}</version></dependency><dependency><groupId>org.pac4j</groupId><artifactId>pac4j-jwt</artifactId><version>${pac4j.version}</version></dependency> </dependencies>

    就像創建JWT Verifier一樣,創建一個src/main/java/com/okta/developer/CorsFilter.java 。

    package com.okta.developer;import javax.servlet.*; import javax.servlet.annotation.WebFilter; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException;@WebFilter(filterName = "corsFilter") public class CorsFilter implements Filter {@Overridepublic void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain chain)throws IOException, ServletException {HttpServletRequest request = (HttpServletRequest) servletRequest;HttpServletResponse response = (HttpServletResponse) servletResponse;System.out.println("In CorsFilter, method: " + request.getMethod());// Authorize (allow) all domains to consume the contentresponse.addHeader("Access-Control-Allow-Origin", "http://localhost:3000");response.addHeader("Access-Control-Allow-Methods", "GET");response.addHeader("Access-Control-Allow-Headers", "*");// For HTTP OPTIONS verb/method reply with ACCEPTED status code -- per CORS handshakeif (request.getMethod().equals("OPTIONS")) {response.setStatus(HttpServletResponse.SC_ACCEPTED);return;}// pass the request along the filter chainchain.doFilter(request, response);}@Overridepublic void init(FilterConfig config) {}@Overridepublic void destroy() {} }

    在同一程序包中創建一個SecurityConfigFactory.java 。 將客戶端ID,密鑰和域占位符替換為與OIDC應用程序匹配的占位符。

    package com.okta.developer;import com.fasterxml.jackson.databind.ObjectMapper; import org.pac4j.core.client.Clients; import org.pac4j.core.client.direct.AnonymousClient; import org.pac4j.core.config.Config; import org.pac4j.core.config.ConfigFactory; import org.pac4j.core.credentials.TokenCredentials; import org.pac4j.core.profile.CommonProfile; import org.pac4j.http.client.direct.HeaderClient; import org.pac4j.jwt.config.signature.RSASignatureConfiguration; import org.pac4j.jwt.credentials.authenticator.JwtAuthenticator; import org.pac4j.jwt.util.JWKHelper; import org.pac4j.oidc.client.OidcClient; import org.pac4j.oidc.config.OidcConfiguration; import org.pac4j.oidc.profile.OidcProfile;import java.io.IOException; import java.net.URL; import java.security.KeyPair; import java.util.ArrayList; import java.util.List; import java.util.Map;public class SecurityConfigFactory implements ConfigFactory {private final JwtAuthenticator jwtAuthenticator = new JwtAuthenticator();private final ObjectMapper mapper = new ObjectMapper();@Overridepublic Config build(final Object... parameters) {System.out.print("Building Security configuration...\n");final OidcConfiguration oidcConfiguration = new OidcConfiguration();oidcConfiguration.setClientId("{yourClientId}");oidcConfiguration.setSecret("{yourClientSecret}");oidcConfiguration.setDiscoveryURI("https://{yourOktaDomain}/oauth2/default/.well-known/openid-configuration");oidcConfiguration.setUseNonce(true);final OidcClient<OidcProfile, OidcConfiguration> oidcClient = new OidcClient<>(oidcConfiguration);oidcClient.setAuthorizationGenerator((ctx, profile) -> {profile.addRole("ROLE_USER");return profile;});HeaderClient headerClient = new HeaderClient("Authorization", "Bearer ", (credentials, ctx) -> {String token = ((TokenCredentials) credentials).getToken();if (token != null) {try {// Get JWKURL keysUrl = new URL("https://{yourOktaDomain}/oauth2/default/v1/keys");Map map = mapper.readValue(keysUrl, Map.class);List keys = (ArrayList) map.get("keys");String json = mapper.writeValueAsString(keys.get(0));// Build key pair and validate tokenKeyPair rsaKeyPair = JWKHelper.buildRSAKeyPairFromJwk(json);jwtAuthenticator.addSignatureConfiguration(new RSASignatureConfiguration(rsaKeyPair));CommonProfile profile = jwtAuthenticator.validateToken(token);credentials.setUserProfile(profile);System.out.println("Hello, " + profile.getId());} catch (IOException e) {System.err.println("Failed to validate Bearer token: " + e.getMessage());e.printStackTrace();}}});final Clients clients = new Clients("http://localhost:8080/callback",oidcClient, headerClient, new AnonymousClient());return new Config(clients);} }

    如果oidcClient的代碼中的oidcClient嘗試直接訪問您的API,將使用戶登錄Okta。 headerClient設置了資源服務器,該資源服務器根據用戶的訪問令牌對用戶進行授權。

    創建src/main/webapp/WEB-INF/web.xml來映射CorsFilter以及Pac4j的CallbackFilter和SecurityFilter 。 您可以看到SecurityFilter通過其configFactory init-param鏈接到SecurityConfigFactory類。

    <?xml version="1.0" encoding="UTF-8"?> <web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd"version="3.1"><display-name>javaee-pac4j-demo</display-name><absolute-ordering/><filter-mapping><filter-name>corsFilter</filter-name><url-pattern>/*</url-pattern></filter-mapping><filter><filter-name>callbackFilter</filter-name><filter-class>org.pac4j.j2e.filter.CallbackFilter</filter-class><init-param><param-name>defaultUrl</param-name><param-value>/</param-value></init-param><init-param><param-name>renewSession</param-name><param-value>true</param-value></init-param><init-param><param-name>multiProfile</param-name><param-value>true</param-value></init-param></filter><filter-mapping><filter-name>callbackFilter</filter-name><url-pattern>/callback</url-pattern><dispatcher>REQUEST</dispatcher></filter-mapping><filter><filter-name>OidcFilter</filter-name><filter-class>org.pac4j.j2e.filter.SecurityFilter</filter-class><init-param><param-name>configFactory</param-name><param-value>com.okta.developer.SecurityConfigFactory</param-value></init-param><init-param><param-name>clients</param-name><param-value>oidcClient,headerClient</param-value></init-param><init-param><param-name>authorizers</param-name><param-value>securityHeaders</param-value></init-param></filter><filter-mapping><filter-name>OidcFilter</filter-name><url-pattern>/*</url-pattern><dispatcher>REQUEST</dispatcher><dispatcher>FORWARD</dispatcher></filter-mapping> </web-app>

    為了更好地可視化用戶信息,您需要創建更多文件。 這些與JSF相關的文件是從j2e-pac4j-cdi-demo復制的。

    注意:我試圖在TomEE上運行j2e-pac4j-cdi-demo (沒有web.xml ),但是失敗并出現錯誤: Filters cannot be added to context [] as the context has been initialised ,因此無法將Filters cannot be added to context [] as the context has been initialised 。 當使用Payara Maven插件時,它確實起作用。

    創建src/main/java/com/okta/developer/ProfileView.java ,這是一個JSF托管的bean,用于收集用戶的信息。

    package com.okta.developer;import org.pac4j.core.context.WebContext; import org.pac4j.core.profile.ProfileManager; import org.slf4j.Logger; import org.slf4j.LoggerFactory;import javax.annotation.PostConstruct; import javax.enterprise.context.RequestScoped; import javax.inject.Inject; import javax.inject.Named; import java.util.List;/*** Managed bean which exposes the pac4j profile manager.** JSF views such as facelets can reference this to view the contents of profiles.** @author Phillip Ross*/ @Named @RequestScoped public class ProfileView {/** The static logger instance. */private static final Logger logger = LoggerFactory.getLogger(ProfileView.class);/** The pac4j web context. */@Injectprivate WebContext webContext;/** The pac4j profile manager. */@Injectprivate ProfileManager profileManager;/** Simple no-args constructor. */public ProfileView() {}/*** Gets the first profile (if it exists) contained in the profile manager.** @return a list of pac4j profiles*/public Object getProfile() {return profileManager.get(true).orElse(null); // It's fine to return a null reference if there is no value present.}/*** Gets the profiles contained in the profile manager.** @return a list of pac4j profiles*/public List getProfiles() {return profileManager.getAll(true);}/** Simply prints some debugging information post-construction. */@PostConstructpublic void init() {logger.debug("webContext is null? {}", (webContext == null));logger.debug("profileManager is null? {}", (profileManager == null));} }

    將src/main/webapp/oidc/index.xhtml為JSF模板。

    <ui:composition xmlns="http://www.w3.org/1999/xhtml"xmlns:h="http://java.sun.com/jsf/html"xmlns:f="http://java.sun.com/jsf/core"xmlns:ui="http://java.sun.com/jsf/facelets"template="/WEB-INF/template.xhtml"><ui:define name="title">Pac4J Java EE Demo - Protected Area</ui:define><ui:define name="content"><div class="ui-g"><div class="ui-g-12"><div class="ui-container"><h1>Protected Area</h1><p><h:link value="Back" outcome="/index"/></p></div><ui:include src="/WEB-INF/facelets/includes/pac4j-profiles-list.xhtml"/></div></div></ui:define> </ui:composition>

    創建pac4j-profiles-list.xhtml文件,該文件包含在WEB-INF/facelets/includes 。

    <ui:composition xmlns="http://www.w3.org/1999/xhtml"xmlns:h="http://java.sun.com/jsf/html"xmlns:f="http://java.sun.com/jsf/core"xmlns:ui="http://java.sun.com/jsf/facelets"><div class="ui-container"><p>Found <h:outputText value="#{profileView.profiles.size()}"/> profiles.</p><h:panelGroup layout="block" rendered="#{profileView.profiles.size() > 0}"><p>First profile: <h:outputText value="#{profileView.profile}"/></p></h:panelGroup></div><h:panelGroup layout="block" rendered="#{not empty profileView.profile}"><h2>Profile Details</h2><p><h:outputText value="Id: #{profileView.profile.id}"/></p><p><h:outputText value="Type Id: #{profileView.profile.typedId}"/></p><p><h:outputText value="Remembered: #{profileView.profile.remembered}"/></p><h3>Attributes (<h:outputText value="#{profileView.profile.attributes.size()}"/>)</h3><h:panelGroup layout="block" rendered="#{profileView.profile.attributes.size() > 0}"><ul><ui:repeat value="#{profileView.profile.attributes.keySet().toArray()}" var="attributeName"><li><h:outputText value="#{attributeName}"/>: <h:outputText value="#{profileView.profile.attributes.get(attributeName)}"/> </li></ui:repeat></ul></h:panelGroup><h3>Roles (<h:outputText value="#{profileView.profile.roles.size()}"/>)</h3><h:panelGroup layout="block" rendered="#{profileView.profile.roles.size() > 0}"><ul><ui:repeat value="#{profileView.profile.roles.toArray()}" var="role"><li><h:outputText value="#{role}"/></li></ui:repeat></ul></h:panelGroup><h3>Permissions (<h:outputText value="#{profileView.profile.permissions.size()}"/>)</h3><h:panelGroup layout="block" rendered="#{profileView.profile.permissions.size() > 0}"><ul><ui:repeat value="#{profileView.profile.permissions.toArray()}" var="permission"><li><h:outputText value="#{permission}"/></li></ui:repeat></ul></h:panelGroup></h:panelGroup> </ui:composition>

    oidc/index.xhtml模板使用WEB-INF/template.xhtml ,因此您也需要創建它。

    <!DOCTYPE html> <html xmlns="http://www.w3.org/1999/xhtml"xmlns:h="http://java.sun.com/jsf/html"xmlns:f="http://java.sun.com/jsf/core"xmlns:ui="http://java.sun.com/jsf/facelets"><h:head><f:facet name="first"><meta http-equiv="X-UA-Compatible" content="IE=edge" /><meta http-equiv="Content-Type" content="text/html; charset=UTF-8" /><meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=0"/><meta name="apple-mobile-web-app-capable" content="yes" /></f:facet><title><ui:insert name="title">Pac4J Java EE Demo</ui:insert></title><ui:insert name="head"/></h:head><h:body styleClass="main-body"><div class="layout-wrapper"><div class="layout-main"><ui:insert name="content"/></div></div></h:body> </html>

    添加這些文件后,重建項目并重新啟動TomEE。

    mvn clean package tomee:run

    導航到http://localhost:8080/oidc/index.jsf ,您將被重定向到Okta進行登錄。 如果您初次嘗試無法解決問題,請重新啟動瀏覽器并使用隱身窗口。 您應該看到用戶的個人資料信息。

    在http://localhost:3000嘗試您的React客戶端; 它也應該工作!

    如果您想知道為什么不堆疊圖像,那是因為我將React應用程序的BeerList.tsx的啤酒清單的JSX更改為內聯。

    <h2>Beer List</h2> {beers.map((beer: Beer) =><span key={beer.id} style={{float: 'left', marginRight: '10px', marginLeft: '10px'}}>{beer.name}<br/><GiphyImage name={beer.name}/></span> )}

    雅加達EE呢?

    您可能已經聽說Java EE已經成為開源的(類似于Java SE的OpenJDK ),其新名稱為Jakarta EE 。 David Blevins是一個很好的朋友,并且積極參與Java EE / Jakarta EE。 有關證明,請參閱他的Twitter傳記:Apache TomEE,OpenEJB和Geronimo項目的創始人。 Apache,JCP EC,EE4J PMC,Jakarta EE WG,MicroProfile和Eclipse Board的成員。 首席執行官@Tomitribe 。

    我問戴維何時會發布可用的Jakarta EE。

    David:目前的主要重點是創建與Java EE 8兼容的Jakarta EE版本。我們希望在今年年底之前將其發布。 發布之后,我們將開始開發Jakarta EE 9并根據需要進行迭代。

    Jakarta EE有一個工作組來決定平臺的方向。

    了解有關安全REST API,Java EE,Jakarta EE和OIDC的更多信息

    我希望您喜歡這個游覽,向您展示了如何使用JWT和OIDC構建和保護Java EE REST API。 如果您想查看每個完成部分的源代碼,我將它們放在GitHub repo的分支中。 您可以使用以下命令克隆不同的實現:

    git clone -b jwt-verifier https://github.com/oktadeveloper/okta-java-ee-rest-api-example.git git clone -b spring-security https://github.com/oktadeveloper/okta-java-ee-rest-api-example.git git clone -b pac4j https://github.com/oktadeveloper/okta-java-ee-rest-api-example.git

    如前所述,我們在此博客上獲得的大多數Java教程都展示了如何使用Spring Boot。 如果您有興趣學習Spring Boot,這里有一些我寫的教程將向您展示要點。

    • Spring Boot,OAuth 2.0和Okta入門
    • 使用React和Spring Boot構建一個簡單的CRUD應用
    • 使用Angular 7.0和Spring Boot 2.1構建基本的CRUD應用

    如果您是OIDC的新手,建議您查看以下文章:

    • Spring Security 5.0和OIDC入門
    • 身份,聲明和令牌– OpenID Connect入門,第1部分,共3部分
    • 行動中的OIDC – OpenID Connect入門,第2部分,共3部分
    • 令牌中有什么? – OpenID Connect入門,第3部分,共3部分

    有關Java REST API和TomEE的更多信息,我建議以下來源:

    • David Blevins –解構REST安全,迭代2018
    • Antonio Goncalves –使用JWT保護JAX-RS端點
    • TomEE:使用Systemd運行

    如果您到目前為止已經做到了,我懷疑您可能對以后的博客文章感興趣。 在Twitter上關注我和我的整個團隊 , 在Facebook上關注我們,或者查看我們的YouTube頻道 。 如有疑問,請在下面發表評論,或將其發布到我們的開發者論壇 。

    “我喜歡編寫身份驗證和授權代碼。” ?從來沒有Java開發人員。 厭倦了一次又一次地建立相同的登錄屏幕? 嘗試使用Okta API進行托管身份驗證,授權和多因素身份驗證。

    使用Java EE和OIDC構建Java REST API最初于2018年9月12日發布在Okta開發人員博客上。

    翻譯自: https://www.javacodegeeks.com/2018/10/build-java-rest-api-java-ee-oidc.html

    總結

    以上是生活随笔為你收集整理的使用Java EE和OIDC构建Java REST API的全部內容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

    精品天堂av | 999久久a精品合区久久久 | 午夜在线免费视频 | 亚洲精品高清在线观看 | 久久久国产精品人人片99精片欧美一 | 天天艹日日干 | 伊人首页| 中文字幕丝袜美腿 | 久草在线免费在线观看 | 超碰国产在线播放 | 91| 国产午夜精品一区二区三区 | 欧美激情另类 | www.成人sex| 色综合久久久久久中文网 | 夜夜爽夜夜操 | 911久久| 国产免费一区二区三区最新6 | 亚洲欧洲日韩在线观看 | 日韩一级精品 | 国产精品成人一区二区 | 9999精品| 九九99靖品| 日韩在线 一区二区 | 一级黄色片在线免费看 | 97人人超碰在线 | 成人黄色国产 | 国产在线播放观看 | 91看片在线 | 狠狠干美女 | 91久久丝袜国产露脸动漫 | 亚洲日本在线一区 | 久精品在线观看 | 亚洲专区中文字幕 | 天天操天天操天天操天天操 | 日日夜夜天天久久 | 国产精品久久久久久久久久新婚 | 日韩超碰在线 | 在线看片日韩 | 日韩欧美视频免费观看 | 精品国产乱子伦一区二区 | 超碰电影在线观看 | 国产精品18久久久久久久久久久久 | 久草在线久 | 福利电影一区二区 | 玖玖视频精品 | 国产精品福利在线观看 | 色中色综合 | 在线观看中文字幕第一页 | 在线观看日韩一区 | 97视频在线观看免费 | 久久夜色精品国产欧美乱极品 | 久久中文字幕视频 | 成年人免费在线观看 | 欧美在线观看视频 | 在线看v片| 国产精品久久亚洲 | 成人免费在线网 | 欧美成人xxxx | 亚洲精品成人 | 国产精品18久久久久白浆 | 天堂网av在线 | 人人天天夜夜 | 亚洲综合在线一区二区三区 | 久久五月婷婷丁香 | 精品国产电影 | 免费影视大全推荐 | 久久精品女人毛片国产 | 婷婷色狠狠 | 在线观看视频黄 | av字幕在线 | 久久视频免费观看 | 午夜精品久久久久 | 国产一区精品在线 | 亚洲国产精品电影 | 麻豆av电影| 在线视频一区二区 | 女人18毛片a级毛片一区二区 | 欧美在线观看禁18 | 国产做a爱一级久久 | 成人av一二三区 | 成人免费网视频 | 中文字幕婷婷 | 中文av网 | 国产成人综合图片 | 精品一区二区在线观看 | 午夜在线免费观看 | 日韩天天干 | 九七在线视频 | 国产日产精品一区二区三区四区 | 亚洲激情 欧美激情 | 亚洲欧美日韩精品一区二区 | 精品久久久久久久久久久久久久久久久久 | 808电影免费观看三年 | 国产日本三级 | 国产69精品久久99不卡的观看体验 | 欧美日韩性生活 | 人人爽久久久噜噜噜电影 | 久久久久久97三级 | 免费看十八岁美女 | 久草网视频 | 亚洲精品一区二区网址 | 国产视频在线观看一区二区 | 91免费日韩| 国产精品久久嫩一区二区免费 | 国产色视频网站2 | 国产精品淫片 | 麻豆视频免费播放 | 视频国产| 欧美日韩视频一区二区三区 | 欧美日韩在线免费观看 | 免费黄a大片 | 丝袜精品视频 | 狠狠狠狠狠狠狠干 | 97天堂| 色综合天天色 | 国产精品成人免费精品自在线观看 | 在线黄色国产 | 久久综合狠狠综合久久综合88 | 国产成人精品综合 | 97人人艹| 国产精品免费一区二区三区 | 高清中文字幕av | 成年人在线看视频 | 激情 一区二区 | 99中文字幕在线观看 | 免费观看9x视频网站在线观看 | 91九色porny蝌蚪主页 | 精品亚洲欧美无人区乱码 | 久久另类小说 | 久久手机精品视频 | 亚洲国产中文字幕在线观看 | 日本一区二区三区视频在线播放 | 中文字幕一区二区三区乱码在线 | 久久精品99国产精品酒店日本 | 激情深爱.com | 中文av日韩 | 人人爽人人片 | 最近更新的中文字幕 | 国产成人精品999 | 欧美一二区在线 | 96久久欧美麻豆网站 | 欧美日韩69 | 黄色电影小说 | 成人国产精品一区 | 久久久久久久久久久高潮一区二区 | 亚洲精品视频网址 | 日韩中午字幕 | 久久免费在线观看 | 黄色的片子 | 日韩在线观看电影 | 国产一区二区精品在线 | 黄色三级视频片 | 久久色视频 | 热久精品 | 欧美久久九九 | 欧美性护士 | 欧美日韩高清一区二区三区 | 蜜臀一区二区三区精品免费视频 | 日批在线看| 激情五月婷婷综合 | 久久久久久久久久久久久久免费看 | 日韩在线视频二区 | 国产精品视频免费 | 久久精品99国产精品亚洲最刺激 | 国产成人61精品免费看片 | 日韩成人免费在线观看 | 特级xxxxx欧美 | 一区二区久久久久 | 人人超碰人人 | 亚洲污视频 | 国产精品一区二区美女视频免费看 | 97超碰福利久久精品 | 懂色av懂色av粉嫩av分享吧 | 国产精品美女久久久久久网站 | 美女激情影院 | 成人aⅴ视频 | 五月婷婷六月丁香 | 全久久久久久久久久久电影 | 国产精品久久久久久影院 | 97色噜噜 | 精品电影一区二区 | 国产中年夫妇高潮精品视频 | 国内精品久久久久影院一蜜桃 | 视频一区二区在线观看 | 国产一级片一区二区三区 | 久久婷婷开心 | 日韩中文字幕亚洲一区二区va在线 | 精品欧美一区二区在线观看 | 五月开心综合 | 久久精品资源 | 欧美日韩国产一区二区三区在线观看 | 九九热免费在线视频 | 麻豆观看 | 成年人app网址 | 日本在线观看视频一区 | 美女网站在线观看 | 久久人人爽视频 | 日韩欧美网站 | 国产精品久久久一区二区三区网站 | 久久久久久毛片精品免费不卡 | 热久在线 | 免费午夜av| 涩涩成人在线 | 日韩免费观看视频 | 精品国产色 | 91精品影视 | 国语麻豆 | 久久 在线| 久久男人免费视频 | 久久精品综合 | 免费高清在线视频一区· | 成人一区二区三区在线观看 | 午夜精品视频福利 | 久久久在线视频 | 国产在线污 | 中文字幕在线第一页 | 开心激情五月婷婷 | 欧美另类交在线观看 | 在线观看欧美成人 | 中文字幕字幕中文 | 日韩精品免费一区二区 | 日韩性网站 | 国产精品久久久久影院 | 91av视频在线免费观看 | www.com久久| 亚洲午夜剧场 | 91精品中文字幕 | 久久精品999| 91自拍成人 | 国内精品99| 视频在线在亚洲 | 欧美一级黄色视屏 | 4438全国亚洲精品观看视频 | 久久综合九色九九 | 久久亚洲精品国产亚洲老地址 | 久久男人中文字幕资源站 | 91精品国产99久久久久久久 | 日日干天天 | 少妇bbw揉bbb欧美 | 免费三级影片 | 91超碰在线播放 | 国产精品大尺度 | 在线视频日韩一区 | 久久久国产精品人人片99精片欧美一 | 九九免费在线视频 | 中文字幕一区二区三区在线观看 | 91视频国产高清 | 国产精品在线看 | 色综合婷婷久久 | 男女全黄一级一级高潮免费看 | a在线一区 | 国产精品国产三级国产不产一地 | 黄色影院在线免费观看 | 97超碰人人澡人人爱学生 | 99在线精品视频 | 欧美精品一区二区在线观看 | 国产精品女人网站 | 在线99热| 国产精品视频永久免费播放 | 久久综合毛片 | 国产91欧美 | 97免费在线观看视频 | www.久久色 | 久久精品网站免费观看 | 在线观看完整版免费 | 91精品国产自产在线观看永久 | 国产热re99久久6国产精品 | 久久精品在线视频 | 国产又粗又猛又黄又爽的视频 | 91香蕉视频在线 | 日本中文字幕高清 | 91免费看黄色 | 日韩黄色免费 | 免费视频久久 | 久久精品日产第一区二区三区乱码 | 欧美a级一区二区 | 日韩精品一区二区在线观看视频 | 91福利视频久久久久 | 国产精品成人一区二区三区吃奶 | 在线观看中文字幕 | 久爱精品在线 | 国产一区二区在线播放视频 | 国产免费人成xvideos视频 | 国产精品成人av电影 | 97国产大学生情侣白嫩酒店 | 激情六月婷婷久久 | 欧美一二区视频 | 色综合久久久久综合体 | 九九热1 | 91色吧| 免费在线成人av | 美女黄频在线观看 | 91福利影院在线观看 | 超碰99在线 | 国产午夜精品一区二区三区 | 精品国产123 | 97热久久免费频精品99 | 久久精品国产亚洲a | 91精品国自产在线观看 | 黄色a视频 | 国产免费观看久久 | 中文字幕人成一区 | 国产精品一区二区久久精品爱微奶 | 欧美日韩国产一二三区 | 免费黄a大片 | 男女靠逼app| 丁香五月网久久综合 | 麻花天美星空视频 | 91私密视频 | 久久久精品久久日韩一区综合 | 国产97视频 | 欧美精品久久久久久久久久 | 色射爱| 亚洲久草视频 | 麻豆视频免费入口 | 444av| 色一级片 | 久久久久久久久久久高潮一区二区 | 久久黄色美女 | 国产在线观看99 | av中文字幕不卡 | 亚洲精品18p | 99热手机在线观看 | 国产在线精品国自产拍影院 | 久久综合狠狠综合 | 亚洲人天堂 | 四虎成人精品永久免费av九九 | 99久久久国产精品免费99 | 久久不见久久见免费影院 | 一二区av| 五月综合色婷婷 | 24小时日本在线www免费的 | 中文字幕中文字幕 | 麻豆精品视频在线 | 国产一级高清视频 | 91久久久久久国产精品 | 亚洲成人二区 | 国产超碰在线观看 | 日韩v欧美v日本v亚洲v国产v | av色图天堂网 | 国产成人精品一区二区三区网站观看 | 久久视频一区二区 | 欧美精彩视频在线观看 | 麻豆精品91| 五月天亚洲激情 | 色www精品视频在线观看 | 色天天久久 | 黄色一级大片在线免费看国产一 | 欧美孕妇视频 | 国产成人av福利 | 日本特黄一级片 | 在线成人免费电影 | 99热在线观看 | 99久在线精品99re8热视频 | 久久久久久国产精品 | 中午字幕在线观看 | 国产精品免费看久久久8精臀av | 国产精品精品国产婷婷这里av | 91精彩在线视频 | 天天插狠狠干 | 狠狠做深爱婷婷综合一区 | 奇米网777| 99久久国产免费免费 | 久久99精品一区二区三区三区 | 麻豆视传媒官网免费观看 | 日韩电影精品一区 | 激情偷乱人伦小说视频在线观看 | 亚洲一区二区三区在线看 | 国产一区二区不卡视频 | aaawww| 97免费视频在线播放 | 日韩丝袜 | 一级性视频 | 91精品伦理 | 在线小视频 | 日韩午夜视频在线观看 | 天天操天天色天天射 | 国产成人黄色网址 | 国产午夜三级一区二区三 | 中文字幕文字幕一区二区 | 中文字幕在线观看你懂的 | 欧美一区二区日韩一区二区 | 色视频网站免费观看 | 伊人天天狠天天添日日拍 | 99视频精品视频高清免费 | 国产精品99精品 | 91精品国产综合久久福利 | 中文字幕在线看视频 | 久久久久久久久免费视频 | 在线观看久草 | 丁香电影小说免费视频观看 | 欧美va天堂在线电影 | 精品色999 | 亚洲激情综合网 | a天堂在线看 | 亚洲精品日韩一区二区电影 | 国产系列在线观看 | 99这里精品| 成人国产亚洲 | 91香蕉久久 | 最近免费中文字幕mv在线视频3 | 亚州av网站大全 | 国产在线观看高清视频 | 国产黄色在线看 | 视频一区二区在线 | 久久精品电影网 | 国产一级免费视频 | 日本在线视频网址 | 韩国av一区二区三区在线观看 | 国产高清在线免费视频 | 久久新 | 久久久视频在线 | 五月天婷婷在线观看视频 | 一本一道久久a久久精品蜜桃 | 欧美精品一区二区免费 | 2017狠狠干| 亚洲成人高清在线 | 国产一级片在线播放 | 91视频一8mav | 久久综合久久综合久久 | 在线观看蜜桃视频 | 五月婷丁香 | 日韩国产精品一区 | 国产一区成人在线 | 激情文学综合丁香 | 在线日本看片免费人成视久网 | 日韩成人免费观看 | 亚洲91精品在线观看 | 久久国产精品视频免费看 | 国产亚洲精品久久 | 亚洲乱码中文字幕综合 | 久久综合毛片 | 日本精品久久久久 | 日韩经典一区二区三区 | 国产一区二区在线免费观看 | 日韩二区在线观看 | 在线www色 | 欧美日韩免费观看一区二区三区 | 国产精品夜夜夜一区二区三区尤 | 在线观看中文字幕亚洲 | 日韩精品无码一区二区三区 | 精品一区二区电影 | 日精品在线观看 | 中文字幕视频在线播放 | 九九精品视频在线观看 | 永久免费观看视频 | 亚洲四虎 | 免费观看第二部31集 | 国产中文字幕av | 99精品在线观看 | 久久久久北条麻妃免费看 | 超碰夜夜 | 成人一级影视 | 国产美女精品 | 欧美激情在线网站 | 91免费高清在线观看 | 久久亚洲综合色 | 91麻豆精品91久久久久同性 | 亚洲涩涩网 | 日韩精品在线视频免费观看 | 国产精品嫩草在线 | 新av在线 | 久久久99精品免费观看app | 久久ww| 91毛片在线观看 | 人人澡人人爽欧一区 | 久久免费激情视频 | 99精品国产一区二区三区麻豆 | 永久免费精品视频网站 | 一区二区三区免费在线 | 日韩免费视频线观看 | 国产亚洲精品成人av久久影院 | 88av色| 999国内精品永久免费视频 | 亚洲国内精品在线 | 久久久精品综合 | 丰满少妇一级片 | 成人18视频 | 久久高清视频免费 | 日日操日日插 | 国产九九九视频 | 国产精品a久久 | 亚洲国产成人精品久久 | 免费高清在线观看电视网站 | 亚洲精品视频免费在线观看 | 射射射综合网 | 亚洲精品高清在线 | 精品欧美一区二区三区久久久 | 综合久久综合久久 | 国产亚洲成人网 | 在线色亚洲| 国产女教师精品久久av | 久久免费av电影 | 日韩理论在线视频 | 久草在线观看资源 | 成人久久久精品国产乱码一区二区 | 久久久久欧美精品999 | 欧美一级性生活视频 | 精品在线观看一区二区 | 制服丝袜亚洲 | 久久一区国产 | 麻豆激情电影 | 国产精品久久久久永久免费观看 | 免费看久久久 | 色综合亚洲精品激情狠狠 | 成人cosplay福利网站 | 射综合网 | 国产亚洲成av人片在线观看桃 | 天天干天天操天天搞 | 玖玖精品在线 | 免费a网站 | 欧美性色综合 | 亚洲香蕉视频 | 精品久久久免费视频 | 500部大龄熟乱视频 欧美日本三级 | 精品999在线观看 | 激情亚洲综合在线 | 天天天天天天干 | 国产网红在线观看 | 欧美日韩二三区 | 最近日韩中文字幕中文 | 在线观看一区二区视频 | 亚洲国产精品传媒在线观看 | www.国产在线观看 | 97成人资源 | 国产伦精品一区二区三区无广告 | 一区在线免费观看 | 久久免费精品 | 久久欧洲视频 | 亚洲精品国偷自产在线99热 | 亚洲一本视频 | 国产xvideos免费视频播放 | 99国产在线视频 | 亚洲黄色成人 | 免费在线国产黄色 | 成人av在线影院 | 亚洲国产欧美在线人成大黄瓜 | 久久亚洲热 | 久久精品视频在线 | 91色在线观看视频 | 欧美污在线观看 | 中文字幕在线一区观看 | 在线观看av片 | 亚洲精品av中文字幕在线在线 | 亚洲理论影院 | 91视频-88av| 天天激情综合网 | 国产色婷婷 | 99精品视频免费在线观看 | 超碰在线1| 日韩一级片网址 | 91在线日本 | 91热在线 | 久久这里只有精品9 | 四虎5151久久欧美毛片 | 国产精品自拍av | 国产午夜影院 | 精品一二三区视频 | 欧美一级片免费在线观看 | 色欲综合视频天天天 | 国产一区二区三区在线免费观看 | 毛片网在线 | 五月在线视频 | 日韩影片在线观看 | 午夜久久久久久久久久影院 | 日韩电影一区二区三区在线观看 | 99久e精品热线免费 99国产精品久久久久久久久久 | 91亚洲综合 | 五月婷婷在线观看 | 免费高清男女打扑克视频 | 久久艹欧美| 91精品对白一区国产伦 | 国产精品资源网 | 国产精品九九热 | 国产精品黄色影片导航在线观看 | 亚洲一区黄色 | 一区二区三区中文字幕在线观看 | 精品国精品自拍自在线 | 色网站免费在线看 | 二区三区在线视频 | 久热精品国产 | 中文字幕一区二区三区视频 | 日韩av电影网站在线观看 | 伊人五月天 | 超碰免费av | 2018精品视频 | 91精品国自产在线观看 | 国产一在线精品一区在线观看 | 久久视频网址 | 亚洲欧洲日韩 | 亚洲人人网 | 五月天天色 | 在线观看网站av | 久久精品国产一区 | 婷婷在线资源 | 成人h视频| 热久在线| 国产高清在线看 | 99精品黄色片免费大全 | 久久精品国产免费 | 夜夜躁日日躁狠狠久久av | 五月婷婷黄色网 | 日韩xxxxxxxxx| 国产一区视频在线观看免费 | 六月丁香在线观看 | 国产拍揄自揄精品视频麻豆 | 在线观看播放av | 日韩av高清 | 手机看片99 | 国产一线二线三线在线观看 | 欧美亚洲免费在线一区 | 国产精品综合在线 | 久久国产精品久久精品 | 免费观看91视频大全 | 亚洲视频在线视频 | 久久免费观看少妇a级毛片 久久久久成人免费 | 国产精品免费在线播放 | 精品在线免费观看 | 成人黄大片视频在线观看 | 麻豆视传媒官网免费观看 | av网站有哪些 | 91精品国产92久久久久 | 日韩手机在线观看 | 亚洲国产精品va在线看黑人动漫 | av在线看片 | 天堂av免费在线 | 91av片 | 在线观看免费91 | 免费在线播放av电影 | 天天操夜夜曰 | 在线观看蜜桃视频 | 999热视频 | 97精品欧美91久久久久久 | av综合站 | 亚洲一区欧美精品 | 在线免费观看黄色小说 | 麻豆一区在线观看 | 欧美精品久久久久久久久老牛影院 | 国产99在线免费 | 久久精品99国产精品亚洲最刺激 | 色综合中文综合网 | 五月婷在线| 色噜噜狠狠狠狠色综合 | 久久天天躁夜夜躁狠狠85麻豆 | 天天色天天爱天天射综合 | 成人亚洲精品国产www | 麻豆国产露脸在线观看 | 99热这里只有精品免费 | 欧美一级大片在线观看 | 成人免费视频播放 | 天天干天天操天天干 | 久久精品国产一区 | 婷婷国产v亚洲v欧美久久 | 免费成人在线电影 | 亚洲精品视频在线播放 | 深爱开心激情网 | 国产精品wwwwww | 久日精品 | 国产999精品久久久久久麻豆 | 日本性久久 | 亚洲成人精品影院 | 乱子伦av| 国产中文视 | 在线a视频| 国产一区二区在线播放 | 人人看97 | 欧美不卡视频在线 | 丰满少妇在线观看资源站 | 一区精品久久 | 黄色国产高清 | 欧美激情精品久久 | 日韩黄色免费看 | 黄色免费在线视频 | 日韩视频免费观看高清完整版在线 | 中文国产字幕 | 五月天婷亚洲天综合网精品偷 | 99久久国产免费,99久久国产免费大片 | 精品一区二区影视 | 美女免费网站 | 精品国产一区二区三区久久久久久 | 美女久久久久久久 | 国产黄影院色大全免费 | www在线观看视频 | 国产无套精品久久久久久 | 久久国产精品成人免费浪潮 | 免费99精品国产自在在线 | 96亚洲精品久久 | 日韩视频一区二区三区在线播放免费观看 | 日本久久久精品视频 | 99国产精品视频免费观看一公开 | 人人爽久久久噜噜噜电影 | 999久久久久久久久久久 | 成人免费 在线播放 | 欧美一级电影在线观看 | 亚洲国产成人精品电影在线观看 | 国产一区二区精 | 99国产一区 | 国产麻豆电影在线观看 | 国产亚洲情侣一区二区无 | 韩国中文三级 | 国产高清久久久久 | 欧美在线观看视频 | 亚洲黄色影院 | 二区三区av| 四虎5151久久欧美毛片 | 黄色.com| 中文字幕成人av | 中文字幕乱码日本亚洲一区二区 | 国产网红在线观看 | 九九视频网站 | 久久亚洲综合国产精品99麻豆的功能介绍 | 国产精品高潮呻吟久久久久 | 波多在线视频 | av在线一级| wwwwww色| 午夜在线看片 | 国产精品九色 | 亚洲国产无 | 成人欧美一区二区三区黑人麻豆 | 日日干天天插 | 亚洲视频在线观看网站 | 国产精品免费久久久久影院仙踪林 | 亚洲成人资源在线观看 | 免费h精品视频在线播放 | 久久久久久久久久免费 | 亚洲天堂网在线播放 | 久久涩涩网站 | 国产高清绿奴videos | 国产伦理久久精品久久久久_ | 探花视频网站 | 一级黄色片在线免费看 | 玖玖精品视频 | 香蕉视频在线网站 | 在线免费视频a | 国产一区在线观看免费 | 国产高清av免费在线观看 | 丁香花在线视频观看免费 | 亚洲欧洲国产精品 | 久青草国产在线 | 亚洲草视频 | 国产最新视频在线 | 亚洲乱码国产乱码精品天美传媒 | 午夜美女福利直播 | 欧美日韩大片在线观看 | 国产高清中文字幕 | 我要色综合天天 | 免费观看www小视频的软件 | 国产资源精品 | 国产糖心vlog在线观看 | 婷婷干五月 | 日韩久久精品一区二区三区 | 国产成人一区二区在线观看 | 精品久久一区 | 国产精品免费一区二区三区在线观看 | 国产美女视频免费 | 久久er99热精品一区二区 | 免费视频三区 | 国产特级毛片aaaaaaa高清 | 成人毛片在线观看视频 | 国内精品久久久久久久影视简单 | 97精品国产97久久久久久久久久久久 | 国产美女永久免费 | 亚洲免费专区 | 色噜噜噜噜 | 黄色网在线播放 | 亚洲人片在线观看 | 亚洲精品午夜一区人人爽 | 中文字幕av在线播放 | 国产无遮挡又黄又爽在线观看 | 九九久久成人 | 日韩欧美精品一区二区三区经典 | 国产69精品久久久久99尤 | 精品成人a区在线观看 | 毛片精品免费在线观看 | 日韩理论电影在线观看 | 超碰97国产| 99精品免费久久久久久久久日本 | 国产丝袜一区二区三区 | 中文字幕色站 | 尤物97国产精品久久精品国产 | 日韩三区在线观看 | 日本在线观看一区二区三区 | 国产精品欧美 | 中文字幕一区二 | 国产精品18久久久久久久网站 | 久久艹综合 | www.久久免费 | 久久黄色片 | 国产一区不卡在线 | 蜜臀av免费一区二区三区 | 国产精品毛片一区二区 | 亚洲精品视频在线播放 | 91麻豆精品国产自产在线游戏 | 免费三级a | 久草免费色站 | 久久综合视频网 | 婷婷六月天丁香 | 日韩视频免费播放 | 久久精品视频99 | 91天堂在线观看 | 午夜视频在线观看一区二区三区 | 国产欧美最新羞羞视频在线观看 | 激情视频在线观看网址 | 黄色的网站免费看 | 欧美老少交 | 欧美一级xxxx | 在线观看成人小视频 | 免费久久久久久久 | 激情深爱.com | 国内视频| 国产精品99久久久久的智能播放 | 久久国产经典视频 | 亚洲电影成人 | 国产高清av| 91丨九色丨国产女 | 久久成视频 | 国产午夜一级毛片 | 久久综合久久鬼 | 66av99精品福利视频在线 | 99久久日韩精品免费热麻豆美女 | 欧美另类一二三四区 | 久久在线播放 | 亚洲第一区精品 | 久久九九影视网 | 国产精品久久久久永久免费观看 | 久久激情视频 | 久久免费国产 | 波多野结衣一区二区三区中文字幕 | 日韩狠狠操 | 久草视频在线资源站 | 欧美激情第八页 | 日韩在线免费小视频 | 中文字幕a在线 | 在线中文字母电影观看 | 五月婷婷影院 | 美女国产免费 | 狠狠操综合 | 人人射人人射 | 欧美精品久久久久久久久久久 | 91精品久久久久久综合乱菊 | 国产亚洲成人网 | 人人干人人做 | 麻豆视频免费播放 | 欧洲色综合 | 国产精品久久久久久久久久免费 | 视频在线亚洲 | 国产高清99 | 久久精品在线免费观看 | 欧美日韩性视频在线 | 色综合久久久久综合 | www.超碰97.com | 99久热在线精品视频成人一区 | 高清精品视频 | 91九色蝌蚪视频在线 | 成年人免费av网站 | 精品一区精品二区高清 | 91精品视频网站 | 久久激五月天综合精品 | 特黄色大片 | 狠狠操影视 | 日韩av综合网站 | 97超碰人人澡人人 | 国产r级在线观看 | 色99导航 | 免费在线观看一区二区三区 | www亚洲精品 | 永久av免费在线观看 | 白丝av在线 | 精品久久久久久亚洲 | 91视频免费观看 | 日韩日韩日韩日韩 | 青青看片 | 亚洲欧洲在线视频 | 久久久国产一区二区三区四区小说 | 久久区二区 | 九九免费在线视频 | 日韩综合精品 | 亚洲aⅴ在线 | 国产成人精品在线播放 | 九九精品久久久 | 国产一二三四在线观看视频 | 成人免费av电影 | 日批视频在线观看免费 | 五月天激情综合 | 免费日韩一区 | 伊在线视频 | 九九日韩| 在线天堂8√ | 精品一区二区在线看 | 九色琪琪久久综合网天天 | 99精品欧美一区二区 | 国产色女 | 2019国产精品 | 天堂中文在线播放 | 欧美极品少妇xbxb性爽爽视频 | 亚洲电影成人 | 日韩美女免费线视频 | 日韩精品视频一二三 | 亚洲欧美视频网站 | 国内精品久久天天躁人人爽 | 极品久久久久 | 国产伦精品一区二区三区照片91 | 欧美激情综合色综合啪啪五月 | 91污在线观看 | 国产高清不卡av | 久久精品一区八戒影视 | 91精品免费视频 | 日韩av在线免费看 | 精品国产乱码久久久久久久 | www最近高清中文国语在线观看 | 成年人精品| 亚洲乱码精品 | 国产一级片在线播放 | 狠狠操操操 | 国产麻豆精品久久一二三 | 91九色自拍| 99精品视频免费看 | 国产午夜激情视频 | 黄色三级av | 婷婷国产v亚洲v欧美久久 | 欧美va日韩va| 久久国产a| 91网页版在线观看 | 欧美专区国产专区 | 国产亚洲日| 欧美性爽爽 | 国产在线一线 | 一区二区 不卡 | 日本中文字幕在线看 | 日韩欧美一区二区在线播放 | 91片网 | 探花视频在线观看免费版 | 日本精品久久久久久 | www91在线观看 | 久草在线免费在线观看 | 草在线| 啪啪精品 | 久久久首页 | 国产一级免费在线 | 国产精品一区二区三区电影 | 天天摸日日摸人人看 | 国产精品剧情 | 色多多污污在线观看 | 成人午夜精品久久久久久久3d | 欧美中文字幕久久 | 在线观看理论 | 久久成人国产 | 91九色综合 | 日韩精品一区二区三区外面 | 国产成人一区二区三区在线观看 | 亚洲在线资源 | 成人免费视频网 | 欧美性极品xxxx做受 | 中文字幕一区二区三区四区视频 | 亚洲综合视频在线播放 | 国产韩国日本高清视频 | 超碰成人av | 国产精品亚州 | 亚洲综合色视频 | 天天操天天射天天 | www四虎影院 | 久草剧场 | 免费看网站在线 | 中文字幕永久免费 | 亚洲激情中文 | 欧美日韩国产区 | 99久久精品国产网站 | 最新国产视频 | 99视频精品视频高清免费 | 久久欧洲视频 | 国产精品日韩精品 | 91精品国产三级a在线观看 | 久久超级碰| 日本黄色大片儿 | 国产精美视频 | 久草在线视频首页 | 日韩剧| 国产亚洲激情视频在线 | 日韩一区在线播放 | 综合久久综合久久 | 欧美午夜激情网 | 国产精品18久久久久久久久 | 黄色1级大片 | 亚洲精品女 | 伊人春色电影网 | 中文一区在线观看 | 少妇搡bbbb搡bbb搡忠贞 | 中文字幕999| 国产又粗又猛又黄视频 | 欧美日本不卡视频 | 精油按摩av | 欧美黑人xxxx猛性大交 | av在线免费在线观看 | 日韩四虎| 久久九九免费视频 | a级国产乱理伦片在线观看 亚洲3级 | 免费又黄又爽 |