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

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 前端技术 > javascript >内容正文

javascript

集成spring mvc_向Spring MVC Web应用程序添加社交登录:集成测试

發布時間:2023/12/3 javascript 42 豆豆
生活随笔 收集整理的這篇文章主要介紹了 集成spring mvc_向Spring MVC Web应用程序添加社交登录:集成测试 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

集成spring mvc

我已經寫了關于為使用Spring Social 1.1.0的應用程序編寫單元測試的挑戰,并為此提供了一種解決方案 。

盡管單元測試很有價值,但它并不能真正告訴我們我們的應用程序是否正常運行。

這就是為什么我們必須為此編寫集成測試的原因

這篇博客文章可以幫助我們做到這一點。 在此博客文章中,我們將學習如何為示例應用程序的注冊和登錄功能編寫集成測試。

如果您尚未閱讀Spring Social教程的先前部分,建議您在閱讀本博客文章之前先閱讀它們。 以下描述了此博客文章的前提條件:

  • 在Spring MVC Web應用程序中添加社交登錄:配置描述了如何配置示例應用程序。
  • 在Spring MVC Web應用程序中添加社交登錄:注冊和登錄描述了如何向示例應用程序添加注冊和登錄功能。
  • 在Spring MVC Web應用程序中添加社交登錄:單元測試描述了如何為示例應用程序編寫單元測試。
  • Spring MVC測試教程描述了如何使用Spring MVC Test框架編寫單元測試和集成測試。
  • Spring Data JPA教程:集成測試描述了我們如何為Spring Data JPA存儲庫編寫集成測試。 這篇博客文章幫助您了解如何使用Spring Test DBUnit和DbUnit編寫集成測試。
  • 使用Maven進行集成測試介紹了如何使用Maven運行集成測試和單元測試。 我們的示例應用程序的構建過程遵循此博客文章中描述的方法。

讓我們從對構建過程的配置進行一些更改開始。

配置我們的構建過程

我們必須對構建過程的配置進行以下更改:

  • 我們已經配置了一個本地Maven存儲庫,并將Spring Test DbUnit 1.1.1快照二進制文件添加到該存儲庫。
  • 我們必須將所需的測試依賴項添加到我們的POM文件中。
  • 我們必須將Liquibase變更集文件添加到classpath。
  • 讓我們找出如何進行這些更改。

    將Spring Test DBUnit快照二進制文件添加到本地Maven存儲庫

    由于Spring Test DBUnit的穩定版本與Spring Framework 4不兼容 ,因此我們必須在集成測試中使用構建快照。

    我們可以按照以下步驟將Spring Test DBUnit快照添加到本地Maven存儲庫:

  • 從Github克隆Spring Test DBUnit存儲庫并創建快照二進制文件。
  • 創建etc / mavenrepo目錄。 該目錄是我們本地的Maven存儲庫。
  • 將創建的jar文件復制到etc / mavenrepo / com / github / springtestdbunit / 1.1.1-SNAPSHOT目錄。
  • 將jar文件復制到本地Maven存儲庫后,我們必須在pom.xml文件中配置本地存儲庫的位置。 我們可以通過將以下存儲庫聲明添加到我們的POM文件中來做到這一點:

    <repositories><!-- Other repositories are omitted for the sake of clarity --><repository><id>local-repository</id><name>local repository</name><url>file://${project.basedir}/etc/mavenrepo</url></repository> </repositories>

    使用Maven獲取所需的測試依賴項

    通過將以下依賴項聲明添加到我們的POM文件中,我們可以獲得所需的測試依賴項:

    • Spring測試DBUnit (版本1.1.1-SNAPSHOT)。 我們使用Spring Test DBUnit將Spring Test框架與DbUnit庫集成。
    • DbUnit (版本2.4.9)。 在每次集成測試之前,我們使用DbUnit將數據庫初始化為已知狀態,并驗證數據庫的內容是否與預期數據匹配。
    • liquibase-core (版本3.1.1)。 加載集成測試的應用程序上下文時,我們使用Liquibase創建一些數據庫表。

    pom.xml文件的相關部分如下所示:

    <dependency><groupId>com.github.springtestdbunit</groupId><artifactId>spring-test-dbunit</artifactId><version>1.1.1-SNAPSHOT</version><scope>test</scope> </dependency> <dependency><groupId>org.dbunit</groupId><artifactId>dbunit</artifactId><version>2.4.9</version><scope>test</scope> </dependency> <dependency><groupId>org.liquibase</groupId><artifactId>liquibase-core</artifactId><version>3.1.1</version><scope>test</scope> </dependency>

    將Liquibase變更集添加到類路徑

    通常,我們應該讓Hibernate創建用于集成測試的數據庫。 但是,僅當在我們的域模型中配置了每個數據庫表時,此方法才有效。

    現在不是這種情況。 示例應用程序的數據庫具有一個UserConnection表,該表未在示例應用程序的域模型中配置。 這就是為什么我們需要在運行集成測試之前找到另一種方法來創建UserConnection表。

    為此,我們可以使用Liquibase庫的Spring集成,但這意味著我們必須將Liquibase變更集添加到類路徑中。

    我們可以通過使用Build Helper Maven插件來實現 。 我們可以按照以下步驟將Liquibase變更集添加到類路徑中:

  • 確保在generate-test-resources生命周期階段調用了Builder Helper Maven插件的add-test-resource目標。
  • 配置插件以將etc / db目錄添加到類路徑(此目錄包含所需的文件)。
  • 插件配置的相關部分如下所示:

    <plugin><groupId>org.codehaus.mojo</groupId><artifactId>build-helper-maven-plugin</artifactId><version>1.7</version><executions><!-- Other executions are omitted for the sake of clarity --><execution><id>add-integration-test-resources</id><!-- Run this execution in the generate-test-sources lifecycle phase --><phase>generate-test-resources</phase><goals><!-- Invoke the add-test-resource goal of this plugin --><goal>add-test-resource</goal></goals><configuration><resources><!-- Other resources are omitted for the sake of clarity --><!-- Add the directory which contains Liquibase change sets to classpath --><resource><directory>etc/db</directory></resource></resources></configuration></execution></executions> </plugin>

    如果要獲取有關使用Builder Helper Maven插件的用法的更多信息,可以查看以下網頁:

    • 與Maven的集成測試
    • Builder Helper Maven插件

    現在,我們已經完成了構建過程的配置。 讓我們找出如何配置集成測試。

    配置我們的集成測試

    我們可以按照以下步驟配置集成測試:

  • 修改Liquibase更改日志文件。
  • 在調用我們的測試用例之前,配置應用程序上下文以運行Liquibase變更集。
  • 創建一個自定義DbUnit數據集加載器。
  • 配置集成測試用例
  • 讓我們繼續前進,仔細看看每個步驟。

    修改Liquibase變更日志

    我們的示例應用程序有兩個Liquibase變更集,可從etc / db / schema目錄中找到。 這些變更集是:

  • db-0.0.1.sql文件創建UserConnection表,該表用于將用戶與已使用的社交登錄提供程序的連接持久化。
  • db-0.0.2.sql文件創建user_accounts表,該表包含我們示例應用程序的用戶帳戶。
  • 因為我們只想運行第一個變更集,所以我們必須對Liquibase變更日志文件進行一些修改。 更具體地說,我們必須使用Liquibase上下文來指定

  • 當創建示例應用程序的數據庫時,將執行哪些變更集。
  • 當我們運行集成測試時,將執行哪些變更集。
  • 通過執行以下步驟,我們可以實現我們的目標:

  • 指定當Liquibase上下文是db或integrationtest時,將執行db-0.0.1.sql changeset文件。
  • 指定當Liquibase上下文為db時執行db-0.0.2.sql changeset文件。
  • 我們的Liquibase changelog文件如下所示:

    <?xml version="1.0" encoding="UTF-8"?> <databaseChangeLogxmlns="http://www.liquibase.org/xml/ns/dbchangelog"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://www.liquibase.org/xml/ns/dbchangelog http://www.liquibase.org/xml/ns/dbchangelog/dbchangelog-3.1.xsd"><!-- Run this change set when the database is created and integration tests are run --><changeSet id="0.0.1" author="Petri" context="db,integrationtest"><sqlFile path="schema/db-0.0.1.sql" /></changeSet><!-- Run this change set when the database is created --><changeSet id="0.0.2" author="Petri" context="db"><sqlFile path="schema/db-0.0.2.sql" /></changeSet> </databaseChangeLog>

    在運行集成測試之前執行Liquibase變更集

    通過在加載應用程序上下文時執行集成測試,我們可以在運行集成測試之前執行Liquibase變更集。 我們可以按照以下步驟進行操作:

  • 創建一個IntegrationTestContext類,并使用@Configuration注釋對其進行注釋。
  • 將DataSource字段添加到創建的類中,并使用@Autowired注釋對其進行注釋。
  • 將liquibase()方法添加到類中,并使用@Bean注釋對其進行注釋。 此方法配置SpringLiquibase bean,該bean在加載應用程序上下文時執行liquibase變更集。
  • 通過執行以下步驟來實現liquibase()方法:
  • 創建一個新的SpringLiquibase對象。
  • 配置創建的對象使用的數據源。
  • 配置Liquibase更改日志的位置。
  • 將Liquibase上下文設置為“ integrationtest”。
  • 返回創建的對象。
  • IntegrationTestContext類的源代碼如下所示:

    import liquibase.integration.spring.SpringLiquibase; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration;import javax.sql.DataSource;@Configuration public class IntegrationTestContext {@Autowiredprivate DataSource dataSource;@Beanpublic SpringLiquibase liquibase() {SpringLiquibase liquibase = new SpringLiquibase();liquibase.setDataSource(dataSource);liquibase.setChangeLog("classpath:changelog.xml");liquibase.setContexts("integrationtest");return liquibase;} }

    創建一個自定義DataSetLoader類

    包含不同用戶帳戶信息的DbUnit數據集如下所示:

    <?xml version='1.0' encoding='UTF-8'?> <dataset><user_accounts id="1"creation_time="2014-02-20 11:13:28"email="facebook@socialuser.com"first_name="Facebook"last_name="User"modification_time="2014-02-20 11:13:28"role="ROLE_USER"sign_in_provider="FACEBOOK"version="0"/><user_accounts id="2"creation_time="2014-02-20 11:13:28"email="twitter@socialuser.com"first_name="Twitter"last_name="User"modification_time="2014-02-20 11:13:28"role="ROLE_USER"sign_in_provider="TWITTER"version="0"/><user_accounts id="3"creation_time="2014-02-20 11:13:28"email="registered@user.com"first_name="RegisteredUser"last_name="User"modification_time="2014-02-20 11:13:28"password="$2a$10$PFSfOaC2IFPG.1HjO05KoODRVSdESQ5q7ek4IyzVfTf.VWlKDa/.e"role="ROLE_USER"version="0"/><UserConnection/> </dataset>

    我們可以從該數據集中看到兩件事:

  • 使用社交登錄創建用戶帳戶的用戶沒有密碼。
  • 通過常規注冊創建其用戶帳戶的用戶具有密碼,但沒有登錄提供者。
  • 這是一個問題,因為我們使用所謂的平面XML數據集 ,并且默認的DbUnit數據集加載器無法處理這種情況 。 當然,我們可以開始使用標準XML數據集,但就我的口味而言,其語法有點過于冗長。 這就是為什么我們必須創建一個可以處理這種情況的自定義數據集加載器的原因。

    我們可以按照以下步驟創建自定義數據集加載器:

  • 創建一個ColumnSensingFlatXMLDataSetLoader類,該類擴展了AbstractDataSetLoader類。
  • 重寫createDataSet()方法并通過以下步驟實現它:
  • 創建一個新的FlatXmlDataSetBuilder對象。
  • 啟用列感測。 列檢測意味著DbUnit從數據集文件中讀取整個數據集,并在從數據集中找到新列時添加新列。 這樣可以確保將每一列的值正確插入數據庫中。
  • 創建一個新的IDataSet對象并返回創建的對象。
  • ColumnSensingFlatXMLDataSetLoader類的源代碼如下所示:

    import com.github.springtestdbunit.dataset.AbstractDataSetLoader; import org.dbunit.dataset.IDataSet; import org.dbunit.dataset.xml.FlatXmlDataSetBuilder; import org.springframework.core.io.Resource;import java.io.InputStream;public class ColumnSensingFlatXMLDataSetLoader extends AbstractDataSetLoader {@Overrideprotected IDataSet createDataSet(Resource resource) throws Exception {FlatXmlDataSetBuilder builder = new FlatXmlDataSetBuilder();builder.setColumnSensing(true);InputStream inputStream = resource.getInputStream();try {return builder.build(inputStream);} finally {inputStream.close();}} }

    但是,創建自定義數據集加載器類還不夠。 加載數據集時,我們仍然必須配置測試以使用此類。 我們可以通過使用@DbUnitConfiguration批注注釋測試類并將其dataSetLoader屬性的值設置為ColumnSensingFlatXMLDataSetLoader.class來實現 。

    讓我們繼續看這是如何完成的。

    配置我們的集成測試

    我們可以按照以下步驟配置集成測試:

  • 確保測試由Spring SpringJUnit4ClassRunner執行。 我們可以通過使用@RunWith注釋對測試類進行注釋并將其值設置為SpringJUnit4ClassRunner.class來實現 。
  • 通過使用@ContextConfiguration批注注釋測試類來加載應用程序上下文,并配置使用的應用程序上下文配置類或文件。
  • 用@WebAppConfiguration批注注釋測試類。 這可以確保為集成測試加載的應用程序上下文是WebApplicationContext 。
  • 用@TestExecutionListeners注釋為類添加注釋,并傳遞標準的Spring偵聽器和DBUnitTestExecutionListener作為其值。 DBUnitTestExecutionListener確保Spring處理從測試類中找到的DbUnit批注。
  • 通過使用@DbUnitConfiguration批注注釋測試類,將測試類配置為使用我們的自定義數據集加載器。 將其dataSetLoader屬性的值設置為ColumnSensingFlatXMLDataSetLoader.class 。
  • 將FilterChainProxy字段添加到測試類,并使用@Autowired批注對該字段進行批注。
  • 將WebApplicationContext字段添加到測試類,并使用@Autowired批注對該字段進行批注。
  • 將MockMvc字段添加到測試類。
  • 在測試類中添加一個setUp()方法,并使用@Before注釋對該方法進行注釋,以確保在每個測試方法之前都調用此方法。
  • 通過使用MockMvcBuilders類,實現setUp()方法并創建一個新的MockMvc對象。
  • 空測試類的源代碼如下所示:

    import com.github.springtestdbunit.DbUnitTestExecutionListener; import org.junit.Before; import org.junit.Test; import org.junit.runner.RunWith; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.security.web.FilterChainProxy; import org.springframework.test.context.ContextConfiguration; import org.springframework.test.context.TestExecutionListeners; import org.springframework.test.context.junit4.SpringJUnit4ClassRunner; import org.springframework.test.context.support.DependencyInjectionTestExecutionListener; import org.springframework.test.context.support.DirtiesContextTestExecutionListener; import org.springframework.test.context.transaction.TransactionalTestExecutionListener; import org.springframework.test.context.web.WebAppConfiguration; import org.springframework.test.web.servlet.MockMvc; import org.springframework.test.web.servlet.setup.MockMvcBuilders; import org.springframework.web.context.WebApplicationContext;@RunWith(SpringJUnit4ClassRunner.class) @ContextConfiguration(classes = {ExampleApplicationContext.class, IntegrationTestContext.class}) //@ContextConfiguration(locations = {"classpath:exampleApplicationContext.xml"}) @WebAppConfiguration @TestExecutionListeners({ DependencyInjectionTestExecutionListener.class,DirtiesContextTestExecutionListener.class,TransactionalTestExecutionListener.class,DbUnitTestExecutionListener.class }) @DbUnitConfiguration(dataSetLoader = ColumnSensingFlatXMLDataSetLoader.class) public class ITTest {@Autowiredprivate FilterChainProxy springSecurityFilterChain;@Autowiredprivate WebApplicationContext webApplicationContext;private MockMvc mockMvc;@Beforepublic void setUp() {mockMvc = MockMvcBuilders.webAppContextSetup(webApplicationContext).addFilter(springSecurityFilterChain).build();} }

    如果您需要有關我們的集成測試的配置的更多信息,建議您閱讀以下博客文章:

    • Spring MVC控制器的單元測試:配置介紹了如何配置Spring MVC Test框架。 本教程討論了單元測試,但仍應進一步闡明該問題。
    • Spring Data JPA教程:集成測試描述了如何為Spring Data JPA存儲庫編寫集成測試。 如果您想看一下Spring Test DBUnit的配置,這篇博客文章可能有助于您了解它。
    • Spring MVC應用程序的集成測試:安全性描述了如何編寫Spring MVC應用程序的安全性測試。 本教程基于Spring Security 3.1,但是它仍然可以幫助您了解如何編寫這些測試。

    現在,我們已經了解了如何配置集成測試。 讓我們繼續并創建一些在集成測試中使用的測試實用程序類。

    創建測試實用程序類

    接下來,我們將創建在集成測試中使用的三個實用程序類:

  • 我們將創建IntegrationTestConstants類,其中包含多個集成測試中使用的常量。
  • 我們將創建用于為集成測試創建ProviderSignInAttempt對象的類。
  • 我們將創建一個測試數據構建器類,該類用于創建CsrfToken對象。
  • 讓我們找出為什么我們必須創建這些類以及如何創建它們。

    創建IntegrationTestConstants類

    當我們編寫集成(或單元)測試時,有時我們需要在許多測試類中使用相同的信息。 將這些信息復制到所有測試類是一個壞主意,因為這會使我們的測試難以維護和理解。 相反,我們應該將此信息放在一個類中,并在需要時從該類中獲取。

    IntegrationTestConstants類包含以下信息,這些信息可在多個測試類中使用:

    • 它具有與Spring Security 3.2的CSRF保護相關的常數。 這些常量包括:包含CSRF令牌的HTTP標頭的名稱,包含CSRF令牌的值的請求參數的名稱,包含CsrfToken對象的會話屬性的名稱以及CSRF令牌的值。
    • 它包含User枚舉,該枚舉指定了我們的集成測試中使用的用戶。 每個用戶都有一個用戶名和密碼(這不是必需的)。 該枚舉的信息用于兩個目的:
    • 用于指定登錄用戶。 當我們對受保護的功能(需要某種授權的功能)進行集成測試時,這很有用。
    • 在為登錄功能編寫集成測試時,我們需要指定嘗試登錄該應用程序的用戶的用戶名和密碼。

    IntegrationTestConstants類的源代碼如下所示:

    import org.springframework.security.web.csrf.HttpSessionCsrfTokenRepository;public class IntegrationTestConstants {public static final String CSRF_TOKEN_HEADER_NAME = "X-CSRF-TOKEN";public static final String CSRF_TOKEN_REQUEST_PARAM_NAME = "_csrf";public static final String CSRF_TOKEN_SESSION_ATTRIBUTE_NAME = HttpSessionCsrfTokenRepository.class.getName().concat(".CSRF_TOKEN");public static final String CSRF_TOKEN_VALUE = "f416e226-bebc-401d-a1ed-f10f47aa9c56";public enum User {FACEBOOK_USER("facebook@socialuser.com", null),REGISTERED_USER("registered@user.com", "password"),TWITTER_USER("twitter@socialuser.com", null);private String password;private String username;private User(String username, String password) {this.password = password;this.username = username;}public String getPassword() {return password;}public String getUsername() {return username;}} }

    創建ProviderSignInAttempt對象

    在為示例應用程序編寫單元測試時,我們快速瀏覽了ProviderSignInUtils類,并意識到我們必須找到一種創建ProviderSignInAttempt對象的方法。

    我們通過創建在單元測試中使用的存根類來解決該問題。 這個存根類使我們可以配置返回的Connection <?>對象,并驗證特定的連接是否“持久化到數據庫”。 但是,我們的存根類未持久連接到使用的數據庫。 而是將用戶的用戶ID存儲到Set對象。

    因為現在我們想將連接數據持久化到數據庫,所以我們必須對存根類進行更改。 我們可以通過對TestProviderSignInAttempt對象進行以下更改來進行這些更改:

  • 將一個專用的usersConnectionRepositorySet字段添加到TestProviderSignInAttempt類。 該字段的類型為布爾值 ,其默認值為false。 該字段描述了我們是否可以持久連接到使用的數據存儲。
  • 將一個新的構造函數參數添加到TestProviderSignInAttempt類的構造函數中。 此參數的類型為UsersConnectionRepository ,用于持久連接到使用的數據存儲。
  • 通過執行以下步驟來實現構造函數:
  • 調用超類的構造函數,并將Connection <?>和UsersConnectionRepository對象作為構造函數參數傳遞。
  • 存儲對Connection <?>對象的引用,該引用作為連接字段的構造函數參數給出。
  • 如果作為構造函數參數給出的UsersConnectionRepository對象不為null,則將usersConnectionRepositoryField的值設置為true。
  • 通過執行以下步驟來實現addConnection()方法:
  • 將作為方法參數給出的用戶ID添加到連接 Set中 。
  • 如果在創建新的TestProviderSignInAttempt對象時設置了UsersConnectionRepository對象,請調用ProviderSignInAttempt類的addConnection()方法,并將用戶ID作為方法參數傳遞。
  • TestProviderSignInAttempt類的源代碼如下所示(修改的部分突出顯示):

    import org.springframework.social.connect.Connection; import org.springframework.social.connect.UsersConnectionRepository;import java.util.HashSet; import java.util.Set;public class TestProviderSignInAttempt extends ProviderSignInAttempt {private Connection<?> connection;private Set<String> connections = new HashSet<>();private boolean usersConnectionRepositorySet = false;public TestProviderSignInAttempt(Connection<?> connection, UsersConnectionRepository usersConnectionRepository) {super(connection, null, usersConnectionRepository);this.connection = connection;if (usersConnectionRepository != null) {this.usersConnectionRepositorySet = true;}}@Overridepublic Connection<?> getConnection() {return connection;}@Overridevoid addConnection(String userId) {connections.add(userId);if (usersConnectionRepositorySet) {super.addConnection(userId);}}public Set<String> getConnections() {return connections;} }

    因為我們使用TestProviderSignInAttemptBuilder來構建新的TestProviderSignInAttempt對象,所以我們也必須對該類進行更改。 我們可以按照以下步驟進行這些更改:

  • 將私有的usersConnectionRepository字段添加到TestProviderSignInAttemptBuilder類,并將其類型設置為UsersConnectionRepository 。
  • 將usersConnectionRepository()方法添加到該類。 將對 UsersConnectionRepository對象的引用設置為usersConnectionRepository字段,并返回對構建器對象的引用。
  • 修改build()方法的最后一行,并使用我們之前創建的新構造函數創建一個新的TestProviderSignInAttempt對象。
  • TestProviderSignInAttemptBuilder類的源代碼如下所示(修改的部分突出顯示):

    import org.springframework.social.connect.*; import org.springframework.social.connect.web.TestProviderSignInAttempt;public class TestProviderSignInAttemptBuilder {private String accessToken;private String displayName;private String email;private Long expireTime;private String firstName;private String imageUrl;private String lastName;private String profileUrl;private String providerId;private String providerUserId;private String refreshToken;private String secret;private UsersConnectionRepository usersConnectionRepository;public TestProviderSignInAttemptBuilder() {}public TestProviderSignInAttemptBuilder accessToken(String accessToken) {this.accessToken = accessToken;return this;}public TestProviderSignInAttemptBuilder connectionData() {return this;}public TestProviderSignInAttemptBuilder displayName(String displayName) {this.displayName = displayName;return this;}public TestProviderSignInAttemptBuilder email(String email) {this.email = email;return this;}public TestProviderSignInAttemptBuilder expireTime(Long expireTime) {this.expireTime = expireTime;return this;}public TestProviderSignInAttemptBuilder firstName(String firstName) {this.firstName = firstName;return this;}public TestProviderSignInAttemptBuilder imageUrl(String imageUrl) {this.imageUrl = imageUrl;return this;}public TestProviderSignInAttemptBuilder lastName(String lastName) {this.lastName = lastName;return this;}public TestProviderSignInAttemptBuilder profileUrl(String profileUrl) {this.profileUrl = profileUrl;return this;}public TestProviderSignInAttemptBuilder providerId(String providerId) {this.providerId = providerId;return this;}public TestProviderSignInAttemptBuilder providerUserId(String providerUserId) {this.providerUserId = providerUserId;return this;}public TestProviderSignInAttemptBuilder refreshToken(String refreshToken) {this.refreshToken = refreshToken;return this;}public TestProviderSignInAttemptBuilder secret(String secret) {this.secret = secret;return this;}public TestProviderSignInAttemptBuilder usersConnectionRepository(UsersConnectionRepository usersConnectionRepository) {this.usersConnectionRepository = usersConnectionRepository;return this;}public TestProviderSignInAttemptBuilder userProfile() {return this;}public TestProviderSignInAttempt build() {ConnectionData connectionData = new ConnectionData(providerId,providerUserId,displayName,profileUrl,imageUrl,accessToken,secret,refreshToken,expireTime);UserProfile userProfile = new UserProfileBuilder().setEmail(email).setFirstName(firstName).setLastName(lastName).build();Connection connection = new TestConnection(connectionData, userProfile);return new TestProviderSignInAttempt(connection, usersConnectionRepository);} }

    創建CsrfToken對象

    因為我們的示例應用程序使用了Spring Security 3.2提供的CSRF保護,所以我們必須找出一種在集成測試中創建有效CSRF令牌的方法。 CsrfToken接口聲明了提供有關預期CSRF令牌信息的方法。 該接口具有一個稱為DefaultCsrfToken的實現。

    換句話說,我們必須找出一種創建新的DefaultCsrfToken對象的方法。 DefaultCsrfToken類只有一個構造函數 ,我們可以在集成測試中創建新的DefaultCsrfToken對象時使用它。 問題是這不是很可讀。

    相反,我們將創建一個測試數據構建器類 , 該類提供了用于創建新CsrfToken對象的流利API 。 我們可以按照以下步驟創建此類:

  • 創建一個CsrfTokenBuilder類。
  • 在創建的類中添加一個私有的headerName字段。
  • 將私有requestParameterName字段添加到創建的類。
  • 將私有tokenValue字段添加到創建的類。
  • 向創建的類添加發布構造函數。
  • 添加用于設置headerName , requestParameterName和tokenValue字段的字段值的方法。
  • 向所創建的類中添加一個build()方法,并將其返回類型設置為CsrfToken 。 通過執行以下步驟來實現此方法:
  • 創建一個新的DefaultCsrfToken對象,并提供CSRF令牌頭的名稱,CSRF令牌請求參數的名稱以及CSRF令牌的值作為構造函數參數。
  • 返回創建的對象。
  • CsrfTokenBuilder類的源代碼如下所示:

    import org.springframework.security.web.csrf.CsrfToken; import org.springframework.security.web.csrf.DefaultCsrfToken;public class CsrfTokenBuilder {private String headerName;private String requestParameterName;private String tokenValue;public CsrfTokenBuilder() {}public CsrfTokenBuilder headerName(String headerName) {this.headerName = headerName;return this;}public CsrfTokenBuilder requestParameterName(String requestParameterName) {this.requestParameterName = requestParameterName;return this;}public CsrfTokenBuilder tokenValue(String tokenValue) {this.tokenValue = tokenValue;return this;}public CsrfToken build() {return new DefaultCsrfToken(headerName, requestParameterName, tokenValue);} }

    我們可以使用以下代碼創建新的CsrfToken對象:

    CsrfToken csrfToken = new CsrfTokenBuilder().headerName(IntegrationTestConstants.CSRF_TOKEN_HEADER_NAME).requestParameterName(IntegrationTestConstants.CSRF_TOKEN_REQUEST_PARAM_NAME).tokenValue(IntegrationTestConstants.CSRF_TOKEN_VALUE).build();

    現在,我們已經創建了必需的測試實用程序類。 讓我們繼續并開始為示例應用程序編寫集成測試。

    編寫集成測試

    我們終于準備好為示例應用程序編寫一些集成測試。 我們將編寫以下集成測試:

    • 我們將編寫集成測試,以確保表單登錄正常工作。
    • 我們將編寫集成測試,以驗證使用社交登錄時注冊是否正常運行。

    但是在開始編寫這些集成測試之前,我們將學習如何為Spring Security提供有效的CSRF令牌。

    向Spring Security提供有效的CSRF令牌

    之前我們了解了如何在集成測試中創建CsrfToken對象。 但是,我們仍然必須找出一種將這些CSRF令牌提供給Spring Security的方法。

    現在是時候仔細研究一下Spring Security處理CSRF令牌的方式了。

    CsrfTokenRepository接口聲明生成,保存和加載CSRF令牌所需的方法。 該接口的默認實現是HttpSessionCsrfTokenRepository類,該類將CSRF令牌存儲到HTTP會話。

    我們需要找到以下問題的答案:

    • CSRF令牌如何保存到HTTP會話?
    • 如何從HTTP會話加載CSRF令牌?

    通過查看HttpSessionCsrfTokenRepository類的源代碼,我們可以找到這些問題的答案。 HttpSessionCsrfTokenRepository類的相關部分如下所示:

    import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import javax.servlet.http.HttpSession;public final class HttpSessionCsrfTokenRepository implements CsrfTokenRepository {private static final String DEFAULT_CSRF_TOKEN_ATTR_NAME = HttpSessionCsrfTokenRepository.class.getName().concat(".CSRF_TOKEN");private String sessionAttributeName = DEFAULT_CSRF_TOKEN_ATTR_NAME;public void saveToken(CsrfToken token, HttpServletRequest request,HttpServletResponse response) {if (token == null) {HttpSession session = request.getSession(false);if (session != null) {session.removeAttribute(sessionAttributeName);}} else {HttpSession session = request.getSession();session.setAttribute(sessionAttributeName, token);}}public CsrfToken loadToken(HttpServletRequest request) {HttpSession session = request.getSession(false);if (session == null) {return null;}return (CsrfToken) session.getAttribute(sessionAttributeName);}//Other methods are omitted. }

    現在很清楚,CSRF令牌作為CsrfToken對象存儲到HTTP會話,并且通過使用sessionAttributeName屬性的值重試和存儲這些對象。 這意味著,如果我們想向Spring Security提供有效的CSRF令牌,則必須遵循以下步驟:

  • 通過使用我們的測試數據生成器創建一個新的CsrfToken對象。
  • 發送CSRF令牌的值作為請求參數。
  • 將創建的DefaultCsrfToken對象存儲到HTTP會話中,以便HttpSessionCsrfTokenRepository找到它。
  • 我們的虛擬測試的源代碼如下所示:

    import org.junit.Test; import org.springframework.security.web.csrf.CsrfToken; import org.springframework.security.web.csrf.DefaultCsrfToken; import org.springframework.test.web.servlet.MockMvc;import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.post;public class ITCSRFTest {private MockMvc mockMvc;@Testpublic void test() throws Exception {//1. Create a new CSRF tokenCsrfToken csrfToken = new CsrfTokenBuilder().headerName(IntegrationTestConstants.CSRF_TOKEN_HEADER_NAME).requestParameterName(IntegrationTestConstants.CSRF_TOKEN_REQUEST_PARAM_NAME).tokenValue(IntegrationTestConstants.CSRF_TOKEN_VALUE).build();mockMvc.perform(post("/login/authenticate")//2. Send the value of the CSRF token as request parameter.param(IntegrationTestConstants.CSRF_TOKEN_REQUEST_PARAM_NAME, IntegrationTestConstants.CSRF_TOKEN_VALUE)//3. Set the created CsrfToken object to session so that the CsrfTokenRepository finds it.sessionAttr(IntegrationTestConstants.CSRF_TOKEN_SESSION_ATTRIBUTE_NAME, csrfToken))//Add assertions here.} }

    理論足夠了。 現在,我們準備為我們的應用程序編寫一些集成測試。 首先,將集成編寫到示例應用程序的登錄功能中。

    編寫登錄功能測試

    現在該為示例應用程序的登錄功能編寫集成測試了。 我們將為此編寫以下集成測試:

  • 我們將編寫一個集成測試,以確保登錄成功后一切正常。
  • 我們將編寫一個集成測試,以確保登錄失敗時一切正常。
  • 這兩個集成測試都使用相同的DbUnit數據集文件( users.xml )將數據庫初始化為已知狀態,其內容如下所示:

    <?xml version='1.0' encoding='UTF-8'?> <dataset><user_accounts id="1"creation_time="2014-02-20 11:13:28"email="facebook@socialuser.com"first_name="Facebook"last_name="User"modification_time="2014-02-20 11:13:28"role="ROLE_USER"sign_in_provider="FACEBOOK"version="0"/><user_accounts id="2"creation_time="2014-02-20 11:13:28"email="twitter@socialuser.com"first_name="Twitter"last_name="User"modification_time="2014-02-20 11:13:28"role="ROLE_USER"sign_in_provider="TWITTER"version="0"/><user_accounts id="3"creation_time="2014-02-20 11:13:28"email="registered@user.com"first_name="RegisteredUser"last_name="User"modification_time="2014-02-20 11:13:28"password="$2a$10$PFSfOaC2IFPG.1HjO05KoODRVSdESQ5q7ek4IyzVfTf.VWlKDa/.e"role="ROLE_USER"version="0"/><UserConnection/> </dataset>

    讓我們開始吧。

    測試1:登錄成功

    我們可以按照以下步驟編寫第一個集成測試:

  • 使用@DatabaseSetup批注對測試類進行批注,并配置用于在調用集成測試之前將數據庫初始化為已知狀態的數據集。
  • 創建一個新的CsrfToken對象。
  • 請按照以下步驟將POST請求發送到url'/ login / authenticate':
  • 設置用戶名和密碼請求參數的值。 使用正確的密碼。
  • 將CSRF令牌的值設置為請求。
  • 將創建的CsrfToken設置為session。
  • 確保返回HTTP狀態代碼302。
  • 驗證請求是否重定向到URL“ /”。
  • 我們的集成測試的源代碼如下所示:

    import com.github.springtestdbunit.DbUnitTestExecutionListener; import com.github.springtestdbunit.annotation.DatabaseSetup; import com.github.springtestdbunit.annotation.DbUnitConfiguration; import org.junit.Test; import org.junit.runner.RunWith; import org.springframework.security.web.csrf.CsrfToken; import org.springframework.test.context.ContextConfiguration; import org.springframework.test.context.TestExecutionListeners; import org.springframework.test.context.junit4.SpringJUnit4ClassRunner; import org.springframework.test.context.support.DependencyInjectionTestExecutionListener; import org.springframework.test.context.support.DirtiesContextTestExecutionListener; import org.springframework.test.context.transaction.TransactionalTestExecutionListener; import org.springframework.test.context.web.WebAppConfiguration; import org.springframework.test.web.servlet.MockMvc;import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.post; import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.redirectedUrl; import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status;@RunWith(SpringJUnit4ClassRunner.class) @ContextConfiguration(classes = {ExampleApplicationContext.class, IntegrationTestContext.class}) //@ContextConfiguration(locations = {"classpath:exampleApplicationContext.xml"}) @WebAppConfiguration @TestExecutionListeners({ DependencyInjectionTestExecutionListener.class,DirtiesContextTestExecutionListener.class,TransactionalTestExecutionListener.class,DbUnitTestExecutionListener.class }) @DbUnitConfiguration(dataSetLoader = ColumnSensingFlatXMLDataSetLoader.class) @DatabaseSetup("/net/petrikainulainen/spring/social/signinmvc/user/users.xml") public class ITFormLoginTest {private static final String REQUEST_PARAM_PASSWORD = "password";private static final String REQUEST_PARAM_USERNAME = "username";//Some fields are omitted for the sake of clarityprivate MockMvc mockMvc;//The setUp() method is omitted for the sake of clarify.@Testpublic void login_CredentialsAreCorrect_ShouldRedirectUserToFrontPage() throws Exception {CsrfToken csrfToken = new CsrfTokenBuilder().headerName(IntegrationTestConstants.CSRF_TOKEN_HEADER_NAME).requestParameterName(IntegrationTestConstants.CSRF_TOKEN_REQUEST_PARAM_NAME).tokenValue(IntegrationTestConstants.CSRF_TOKEN_VALUE).build();mockMvc.perform(post("/login/authenticate").param(REQUEST_PARAM_USERNAME, IntegrationTestConstants.User.REGISTERED_USER.getUsername()).param(REQUEST_PARAM_PASSWORD, IntegrationTestConstants.User.REGISTERED_USER.getPassword()).param(IntegrationTestConstants.CSRF_TOKEN_REQUEST_PARAM_NAME, IntegrationTestConstants.CSRF_TOKEN_VALUE).sessionAttr(IntegrationTestConstants.CSRF_TOKEN_SESSION_ATTRIBUTE_NAME, csrfToken)).andExpect(status().isMovedTemporarily()).andExpect(redirectedUrl("/"));} }

    測試2:登錄失敗

    我們可以按照以下步驟編寫第二個集成測試:

  • 使用@DatabaseSetup批注對測試類進行批注,并配置用于在調用集成測試之前將數據庫初始化為已知狀態的數據集。
  • 創建一個新的CsrfToken對象。
  • 請按照以下步驟將POST請求發送到url'/ login / authenticate':
  • 設置用戶名和密碼請求參數的值。 使用錯誤的密碼。
  • 將CSRF令牌的值設置為請求作為請求參數。
  • 將創建的CsrfToken對象設置為session。
  • 確保返回HTTP狀態代碼302。
  • 驗證請求是否重定向到URL'/ login?error = bad_credentials'。
  • 我們的集成測試的源代碼如下所示:

    import com.github.springtestdbunit.DbUnitTestExecutionListener; import com.github.springtestdbunit.annotation.DatabaseSetup; import com.github.springtestdbunit.annotation.DbUnitConfiguration; import org.junit.Test; import org.junit.runner.RunWith; import org.springframework.security.web.csrf.CsrfToken; import org.springframework.test.context.ContextConfiguration; import org.springframework.test.context.TestExecutionListeners; import org.springframework.test.context.junit4.SpringJUnit4ClassRunner; import org.springframework.test.context.support.DependencyInjectionTestExecutionListener; import org.springframework.test.context.support.DirtiesContextTestExecutionListener; import org.springframework.test.context.transaction.TransactionalTestExecutionListener; import org.springframework.test.context.web.WebAppConfiguration; import org.springframework.test.web.servlet.MockMvc;import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.post; import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.redirectedUrl; import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status;@RunWith(SpringJUnit4ClassRunner.class) @ContextConfiguration(classes = {ExampleApplicationContext.class, IntegrationTestContext.class}) //@ContextConfiguration(locations = {"classpath:exampleApplicationContext.xml"}) @WebAppConfiguration @TestExecutionListeners({ DependencyInjectionTestExecutionListener.class,DirtiesContextTestExecutionListener.class,TransactionalTestExecutionListener.class,DbUnitTestExecutionListener.class }) @DbUnitConfiguration(dataSetLoader = ColumnSensingFlatXMLDataSetLoader.class) @DatabaseSetup("/net/petrikainulainen/spring/social/signinmvc/user/users.xml") public class ITFormLoginTest {private static final String REQUEST_PARAM_PASSWORD = "password";private static final String REQUEST_PARAM_USERNAME = "username";//Some fields are omitted for the sake of clarityprivate MockMvc mockMvc;//The setUp() method is omitted for the sake of clarify.@Testpublic void login_InvalidPassword_ShouldRedirectUserToLoginForm() throws Exception {CsrfToken csrfToken = new CsrfTokenBuilder().headerName(IntegrationTestConstants.CSRF_TOKEN_HEADER_NAME).requestParameterName(IntegrationTestConstants.CSRF_TOKEN_REQUEST_PARAM_NAME).tokenValue(IntegrationTestConstants.CSRF_TOKEN_VALUE).build();mockMvc.perform(post("/login/authenticate").param(REQUEST_PARAM_USERNAME, IntegrationTestConstants.User.REGISTERED_USER.getUsername()).param(REQUEST_PARAM_PASSWORD, "invalidPassword").param(IntegrationTestConstants.CSRF_TOKEN_REQUEST_PARAM_NAME, IntegrationTestConstants.CSRF_TOKEN_VALUE).sessionAttr(IntegrationTestConstants.CSRF_TOKEN_SESSION_ATTRIBUTE_NAME, csrfToken)).andExpect(status().isMovedTemporarily()).andExpect(redirectedUrl("/login?error=bad_credentials"));} }

    為注冊功能編寫測試

    我們將為注冊功能編寫以下集成測試:

  • 我們將編寫一個集成測試,以確保當用戶使用社交登錄創建新用戶帳戶時,我們的應用程序正常運行,但提交的注冊表格的驗證失敗。
  • 我們將編寫一個集成測試,該測試通過使用社交登錄名和從數據庫中找到的電子郵件地址來驗證用戶創建新用戶帳戶時,一切是否正常運行。
  • 我們將編寫一個集成測試,以確保可以通過使用社交登錄來創建新的用戶帳戶。
  • 讓我們開始吧。

    測試1:驗證失敗

    我們可以按照以下步驟編寫第一個集成測試:

  • 將UsersConnectionRepository字段添加到測試類,并使用@Autowired批注對其進行批注。
  • 使用@DatabaseSetup批注注釋測試方法,并配置數據集,該數據集用于在運行集成測試之前將數據庫初始化為已知狀態。
  • 創建一個新的TestProviderSignInAttempt對象。 記住要設置使用的UsersConnectionRepository對象。
  • 創建一個新的RegistrationForm對象,并設置其signInProvider字段的值。
  • 創建一個新的CsrfToken對象。
  • 請按照以下步驟將POST請求發送到url'/ user / register':
  • 將請求的內容類型設置為“ application / x-www-form-urlencoded”。
  • 將表單對象轉換為url編碼的字節,并將其設置為請求的正文。
  • 將創建的TestProviderSignInAttempt對象設置為session。
  • 將CSRF令牌的值設置為請求作為請求參數。
  • 將創建的CsrfToken對象設置為session。
  • 將創建的表單對象設置為session。
  • 確保返回HTTP請求狀態200。
  • 確保渲染視圖的名稱為“ user / registrationForm”。
  • 驗證請求是否轉發到URL'/WEB-INF/jsp/user/registrationForm.jsp'。
  • 驗證名為“ user”的模型屬性的字段正確。
  • 確保名為'user'的模型屬性在email , firstName和lastName字段中存在字段錯誤。
  • 使用@ExpectedDatabase批注注釋測試方法,并確保未將新用戶帳戶保存到數據庫(使用用于初始化數據庫的相同數據集)。
  • 我們的集成測試的源代碼如下所示:

    import com.github.springtestdbunit.DbUnitTestExecutionListener; import com.github.springtestdbunit.annotation.DatabaseSetup; import com.github.springtestdbunit.annotation.DbUnitConfiguration; import com.github.springtestdbunit.annotation.ExpectedDatabase; import com.github.springtestdbunit.assertion.DatabaseAssertionMode; import org.junit.Test; import org.junit.runner.RunWith; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.http.MediaType; import org.springframework.security.web.csrf.CsrfToken; import org.springframework.social.connect.UsersConnectionRepository; import org.springframework.social.connect.support.TestProviderSignInAttemptBuilder; import org.springframework.social.connect.web.ProviderSignInAttempt; import org.springframework.social.connect.web.TestProviderSignInAttempt; import org.springframework.test.context.ContextConfiguration; import org.springframework.test.context.TestExecutionListeners; import org.springframework.test.context.junit4.SpringJUnit4ClassRunner; import org.springframework.test.context.support.DependencyInjectionTestExecutionListener; import org.springframework.test.context.support.DirtiesContextTestExecutionListener; import org.springframework.test.context.transaction.TransactionalTestExecutionListener; import org.springframework.test.context.web.WebAppConfiguration; import org.springframework.test.web.servlet.MockMvc;import static net.petrikainulainen.spring.social.signinmvc.user.controller.TestProviderSignInAttemptAssert.assertThatSignIn; import static org.hamcrest.CoreMatchers.allOf; import static org.hamcrest.Matchers.*; import static org.hamcrest.Matchers.is; import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.post; import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.*;@RunWith(SpringJUnit4ClassRunner.class) @ContextConfiguration(classes = {ExampleApplicationContext.class, IntegrationTestContext.class}) //@ContextConfiguration(locations = {"classpath:exampleApplicationContext.xml"}) @WebAppConfiguration @TestExecutionListeners({ DependencyInjectionTestExecutionListener.class,DirtiesContextTestExecutionListener.class,TransactionalTestExecutionListener.class,DbUnitTestExecutionListener.class }) @DbUnitConfiguration(dataSetLoader = ColumnSensingFlatXMLDataSetLoader.class) public class ITRegistrationControllerTest {@Autowiredprivate UsersConnectionRepository usersConnectionRepository;//Some fields are omitted for the sake of clarity.private MockMvc mockMvc;//The setUp() is omitted for the sake of clarity.@Test@DatabaseSetup("no-users.xml")@ExpectedDatabase(value="no-users.xml", assertionMode = DatabaseAssertionMode.NON_STRICT)public void registerUserAccount_SocialSignInAndEmptyForm_ShouldRenderRegistrationFormWithValidationErrors() throws Exception {TestProviderSignInAttempt socialSignIn = new TestProviderSignInAttemptBuilder().connectionData().accessToken("accessToken").displayName("John Smith").expireTime(100000L).imageUrl("https://www.twitter.com/images/johnsmith.jpg").profileUrl("https://www.twitter.com/johnsmith").providerId("twitter").providerUserId("johnsmith").refreshToken("refreshToken").secret("secret").usersConnectionRepository(usersConnectionRepository).userProfile().email("john.smith@gmail.com").firstName("John").lastName("Smith").build();RegistrationForm userAccountData = new RegistrationFormBuilder().signInProvider(SocialMediaService.TWITTER).build();CsrfToken csrfToken = new CsrfTokenBuilder().headerName(IntegrationTestConstants.CSRF_TOKEN_HEADER_NAME).requestParameterName(IntegrationTestConstants.CSRF_TOKEN_REQUEST_PARAM_NAME).tokenValue(IntegrationTestConstants.CSRF_TOKEN_VALUE).build();mockMvc.perform(post("/user/register").contentType(MediaType.APPLICATION_FORM_URLENCODED).content(TestUtil.convertObjectToFormUrlEncodedBytes(userAccountData)).sessionAttr(ProviderSignInAttempt.SESSION_ATTRIBUTE, socialSignIn).param(IntegrationTestConstants.CSRF_TOKEN_REQUEST_PARAM_NAME, IntegrationTestConstants.CSRF_TOKEN_VALUE).sessionAttr(IntegrationTestConstants.CSRF_TOKEN_SESSION_ATTRIBUTE_NAME, csrfToken).sessionAttr("user", userAccountData)).andExpect(status().isOk()).andExpect(view().name("user/registrationForm")).andExpect(forwardedUrl("/WEB-INF/jsp/user/registrationForm.jsp")).andExpect(model().attribute("user", allOf(hasProperty("email", isEmptyOrNullString()),hasProperty("firstName", isEmptyOrNullString()),hasProperty("lastName", isEmptyOrNullString()),hasProperty("password", isEmptyOrNullString()),hasProperty("passwordVerification", isEmptyOrNullString()),hasProperty("signInProvider", is(SocialMediaService.TWITTER))))).andExpect(model().attributeHasFieldErrors("user", "email", "firstName", "lastName"));} }

    我們的集成測試使用一個名為no-users.xml的DbUnit數據集文件,該文件如下所示:

    <?xml version='1.0' encoding='UTF-8'?> <dataset><user_accounts/><UserConnection/> </dataset>

    測試2:從數據庫中找到電子郵件地址

    我們可以按照以下步驟編寫第二個集成測試:

  • 將UsersConnectionRepository字段添加到測試類,并使用@Autowired批注對其進行批注。
  • 使用@DatabaseSetup批注注釋測試方法,并配置數據集,該數據集用于在運行集成測試之前將數據庫初始化為已知狀態。
  • 創建一個新的TestProviderSignInAttempt對象。 記住要設置使用的UsersConnectionRepository對象。
  • 創建一個新的RegistrationForm對象,并設置其email , firstName , lastName和signInProvider字段的值。 使用現有的電子郵件地址。
  • 創建一個新的CsrfToken對象。
  • 請按照以下步驟將POST請求發送到url'/ user / register':
  • 將請求的內容類型設置為“ application / x-www-form-urlencoded”。
  • 將表單對象轉換為url編碼的字節,并將其設置為請求的正文。
  • 將創建的TestProviderSignInAttempt對象設置為session。
  • 將CSRF令牌的值設置為請求作為請求參數。
  • 將創建的CsrfToken對象設置為session。
  • 將創建的表單對象設置為session。
  • 確保返回HTTP請求狀態200。
  • 確保渲染視圖的名稱為“ user / registrationForm”。
  • 驗證請求是否轉發到URL'/WEB-INF/jsp/user/registrationForm.jsp'。
  • 驗證名為“ user”的模型屬性的字段正確。
  • 確保名為“用戶”的模型屬性在電子郵件字段中存在字段錯誤。
  • 使用@ExpectedDatabase批注注釋測試方法,并確保未將新用戶帳戶保存到數據庫(使用用于初始化數據庫的相同數據集)。
  • 我們的集成測試的源代碼如下所示:

    import com.github.springtestdbunit.DbUnitTestExecutionListener; import com.github.springtestdbunit.annotation.DatabaseSetup; import com.github.springtestdbunit.annotation.DbUnitConfiguration; import com.github.springtestdbunit.annotation.ExpectedDatabase; import com.github.springtestdbunit.assertion.DatabaseAssertionMode; import org.junit.Test; import org.junit.runner.RunWith; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.http.MediaType; import org.springframework.security.web.csrf.CsrfToken; import org.springframework.social.connect.UsersConnectionRepository; import org.springframework.social.connect.support.TestProviderSignInAttemptBuilder; import org.springframework.social.connect.web.ProviderSignInAttempt; import org.springframework.social.connect.web.TestProviderSignInAttempt; import org.springframework.test.context.ContextConfiguration; import org.springframework.test.context.TestExecutionListeners; import org.springframework.test.context.junit4.SpringJUnit4ClassRunner; import org.springframework.test.context.support.DependencyInjectionTestExecutionListener; import org.springframework.test.context.support.DirtiesContextTestExecutionListener; import org.springframework.test.context.transaction.TransactionalTestExecutionListener; import org.springframework.test.context.web.WebAppConfiguration; import org.springframework.test.web.servlet.MockMvc;import static org.hamcrest.CoreMatchers.allOf; import static org.hamcrest.Matchers.*; import static org.hamcrest.Matchers.is; import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.post; import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.*;@RunWith(SpringJUnit4ClassRunner.class) @ContextConfiguration(classes = {ExampleApplicationContext.class, IntegrationTestContext.class}) //@ContextConfiguration(locations = {"classpath:exampleApplicationContext.xml"}) @WebAppConfiguration @TestExecutionListeners({ DependencyInjectionTestExecutionListener.class,DirtiesContextTestExecutionListener.class,TransactionalTestExecutionListener.class,DbUnitTestExecutionListener.class }) @DbUnitConfiguration(dataSetLoader = ColumnSensingFlatXMLDataSetLoader.class) public class ITRegistrationControllerTest {@Autowiredprivate UsersConnectionRepository usersConnectionRepository;//Some fields are omitted for the sake of clarity.private MockMvc mockMvc;//The setUp() is omitted for the sake of clarity.@Test@DatabaseSetup("/net/petrikainulainen/spring/social/signinmvc/user/users.xml")@ExpectedDatabase(value = "/net/petrikainulainen/spring/social/signinmvc/user/users.xml", assertionMode = DatabaseAssertionMode.NON_STRICT)public void registerUserAccount_SocialSignInAndEmailExist_ShouldRenderRegistrationFormWithFieldError() throws Exception {TestProviderSignInAttempt socialSignIn = new TestProviderSignInAttemptBuilder().connectionData().accessToken("accessToken").displayName("John Smith").expireTime(100000L).imageUrl("https://www.twitter.com/images/johnsmith.jpg").profileUrl("https://www.twitter.com/johnsmith").providerId("twitter").providerUserId("johnsmith").refreshToken("refreshToken").secret("secret").usersConnectionRepository(usersConnectionRepository).userProfile().email(IntegrationTestConstants.User.REGISTERED_USER.getUsername()).firstName("John").lastName("Smith").build();RegistrationForm userAccountData = new RegistrationFormBuilder().email(IntegrationTestConstants.User.REGISTERED_USER.getUsername()).firstName("John").lastName("Smith").signInProvider(SocialMediaService.TWITTER).build();CsrfToken csrfToken = new CsrfTokenBuilder().headerName(IntegrationTestConstants.CSRF_TOKEN_HEADER_NAME).requestParameterName(IntegrationTestConstants.CSRF_TOKEN_REQUEST_PARAM_NAME).tokenValue(IntegrationTestConstants.CSRF_TOKEN_VALUE).build();mockMvc.perform(post("/user/register").contentType(MediaType.APPLICATION_FORM_URLENCODED).content(TestUtil.convertObjectToFormUrlEncodedBytes(userAccountData)).sessionAttr(ProviderSignInAttempt.SESSION_ATTRIBUTE, socialSignIn).param(IntegrationTestConstants.CSRF_TOKEN_REQUEST_PARAM_NAME, IntegrationTestConstants.CSRF_TOKEN_VALUE).sessionAttr(IntegrationTestConstants.CSRF_TOKEN_SESSION_ATTRIBUTE_NAME, csrfToken).sessionAttr("user", userAccountData)).andExpect(status().isOk()).andExpect(view().name("user/registrationForm")).andExpect(forwardedUrl("/WEB-INF/jsp/user/registrationForm.jsp")).andExpect(model().attribute("user", allOf(hasProperty("email", is(IntegrationTestConstants.User.REGISTERED_USER.getUsername())),hasProperty("firstName", is("John")),hasProperty("lastName", is("Smith")),hasProperty("password", isEmptyOrNullString()),hasProperty("passwordVerification", isEmptyOrNullString()),hasProperty("signInProvider", is(SocialMediaService.TWITTER))))).andExpect(model().attributeHasFieldErrors("user", "email"));} }

    該集成測試使用一個名為users.xml的DbUnit數據集,該數據集如下所示:

    <?xml version='1.0' encoding='UTF-8'?> <dataset><user_accounts id="1" creation_time="2014-02-20 11:13:28" email="facebook@socialuser.com" first_name="Facebook" last_name="User" modification_time="2014-02-20 11:13:28" role="ROLE_USER" sign_in_provider="FACEBOOK" version="0"/><user_accounts id="2" creation_time="2014-02-20 11:13:28" email="twitter@socialuser.com" first_name="Twitter" last_name="User" modification_time="2014-02-20 11:13:28" role="ROLE_USER" sign_in_provider="TWITTER" version="0"/><user_accounts id="3" creation_time="2014-02-20 11:13:28" email="registered@user.com" first_name="RegisteredUser" last_name="User" modification_time="2014-02-20 11:13:28" password="$2a$10$PFSfOaC2IFPG.1HjO05KoODRVSdESQ5q7ek4IyzVfTf.VWlKDa/.e" role="ROLE_USER" version="0"/><UserConnection/> </dataset>

    測試3:注冊成功

    我們可以按照以下步驟編寫第三項集成測試:

  • 將UsersConnectionRepository字段添加到測試類,并使用@Autowired批注對其進行批注。
  • 使用@DatabaseSetup批注注釋測試方法,并配置數據集,該數據集用于在運行集成測試之前將數據庫初始化為已知狀態。
  • 創建一個新的TestProviderSignInAttempt對象。 記住要設置使用的UsersConnectionRepository對象。
  • 創建一個新的RegistrationForm對象,并設置其email , firstName , lastName和signInProvider字段的值。
  • 創建一個新的CsrfToken對象。
  • 請按照以下步驟將POST請求發送到url'/ user / register':
  • 將請求的內容類型設置為“ application / x-www-form-urlencoded”。
  • 將表單對象轉換為url編碼的字節,并將其設置為請求的正文。
  • 將創建的TestProviderSignInAttempt對象設置為session。
  • 將CSRF令牌的值設置為請求作為請求參數。
  • 將創建的CsrfToken對象設置為session。
  • 將創建的表單對象設置為session。
  • 確保返回HTTP請求狀態302。
  • 驗證請求是否重定向到URL'/'。 這還可以確保創建的用戶已登錄,因為匿名用戶無法訪問該URL。
  • 使用@ExpectedDatabase批注對測試方法進行批注,并確保將新用戶帳戶保存到數據庫中,并且與使用的社交媒體提供商的連接得以持久。
  • 我們的集成測試的源代碼如下所示:

    import com.github.springtestdbunit.DbUnitTestExecutionListener; import com.github.springtestdbunit.annotation.DatabaseSetup; import com.github.springtestdbunit.annotation.DbUnitConfiguration; import com.github.springtestdbunit.annotation.ExpectedDatabase; import com.github.springtestdbunit.assertion.DatabaseAssertionMode; import org.junit.Test; import org.junit.runner.RunWith; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.http.MediaType; import org.springframework.security.web.csrf.CsrfToken; import org.springframework.social.connect.UsersConnectionRepository; import org.springframework.social.connect.support.TestProviderSignInAttemptBuilder; import org.springframework.social.connect.web.ProviderSignInAttempt; import org.springframework.social.connect.web.TestProviderSignInAttempt; import org.springframework.test.context.ContextConfiguration; import org.springframework.test.context.TestExecutionListeners; import org.springframework.test.context.junit4.SpringJUnit4ClassRunner; import org.springframework.test.context.support.DependencyInjectionTestExecutionListener; import org.springframework.test.context.support.DirtiesContextTestExecutionListener; import org.springframework.test.context.transaction.TransactionalTestExecutionListener; import org.springframework.test.context.web.WebAppConfiguration; import org.springframework.test.web.servlet.MockMvc;import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.post; import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.*;@RunWith(SpringJUnit4ClassRunner.class) @ContextConfiguration(classes = {ExampleApplicationContext.class, IntegrationTestContext.class}) //@ContextConfiguration(locations = {"classpath:exampleApplicationContext.xml"}) @WebAppConfiguration @TestExecutionListeners({ DependencyInjectionTestExecutionListener.class,DirtiesContextTestExecutionListener.class,TransactionalTestExecutionListener.class,DbUnitTestExecutionListener.class }) @DbUnitConfiguration(dataSetLoader = ColumnSensingFlatXMLDataSetLoader.class) public class ITRegistrationControllerTest2 {@Autowiredprivate UsersConnectionRepository usersConnectionRepository;//Some fields are omitted for the sake of clarity.private MockMvc mockMvc;//The setUp() is omitted for the sake of clarity.@Test@DatabaseSetup("no-users.xml")@ExpectedDatabase(value="register-social-user-expected.xml", assertionMode = DatabaseAssertionMode.NON_STRICT)public void registerUserAccount_SocialSignIn_ShouldCreateNewUserAccountAndRenderHomePage() throws Exception {TestProviderSignInAttempt socialSignIn = new TestProviderSignInAttemptBuilder().connectionData().accessToken("accessToken").displayName("John Smith").expireTime(100000L).imageUrl("https://www.twitter.com/images/johnsmith.jpg").profileUrl("https://www.twitter.com/johnsmith").providerId("twitter").providerUserId("johnsmith").refreshToken("refreshToken").secret("secret").usersConnectionRepository(usersConnectionRepository).userProfile().email("john.smith@gmail.com").firstName("John").lastName("Smith").build();RegistrationForm userAccountData = new RegistrationFormBuilder().email("john.smith@gmail.com").firstName("John").lastName("Smith").signInProvider(SocialMediaService.TWITTER).build();CsrfToken csrfToken = new CsrfTokenBuilder().headerName(IntegrationTestConstants.CSRF_TOKEN_HEADER_NAME).requestParameterName(IntegrationTestConstants.CSRF_TOKEN_REQUEST_PARAM_NAME).tokenValue(IntegrationTestConstants.CSRF_TOKEN_VALUE).build();mockMvc.perform(post("/user/register").contentType(MediaType.APPLICATION_FORM_URLENCODED).content(TestUtil.convertObjectToFormUrlEncodedBytes(userAccountData)).sessionAttr(ProviderSignInAttempt.SESSION_ATTRIBUTE, socialSignIn).param(IntegrationTestConstants.CSRF_TOKEN_REQUEST_PARAM_NAME, IntegrationTestConstants.CSRF_TOKEN_VALUE).sessionAttr(IntegrationTestConstants.CSRF_TOKEN_SESSION_ATTRIBUTE_NAME, csrfToken).sessionAttr("user", userAccountData)).andExpect(status().isMovedTemporarily()).andExpect(redirectedUrl("/"));} }

    用于將數據庫初始化為已知狀態的數據集( no-users.xml )如下所示:

    <?xml version='1.0' encoding='UTF-8'?> <dataset><user_accounts/><UserConnection/> </dataset>

    名為register-social-user-expected.xml的DbUnit數據集用于驗證是否成功創建了用戶帳戶,并且與使用的社交登錄提供者的連接已持久保存到數據庫中。 它看起來如下:

    <?xml version='1.0' encoding='UTF-8'?> <dataset><user_accounts email="john.smith@gmail.com" first_name="John" last_name="Smith" role="ROLE_USER" sign_in_provider="TWITTER"version="0"/><UserConnection userId="john.smith@gmail.com"providerId="twitter"providerUserId="johnsmith"rank="1"displayName="John Smith"profileUrl="https://www.twitter.com/johnsmith"imageUrl="https://www.twitter.com/images/johnsmith.jpg"accessToken="accessToken"secret="secret"refreshToken="refreshToken"expireTime="100000"/> </dataset>

    摘要

    現在,我們已經了解了如何為使用Spring Social 1.1.0的常規Spring MVC應用程序編寫集成測試。 本教程教會了我們很多東西,但是這兩件事是本博客文章的主要課程:

    • 我們了解了如何通過創建ProviderSignInAttempt對象并在集成測試中使用它們來“模擬”社交登錄。
    • 我們學習了如何創建CSRF令牌并將創建的令牌提供給Spring Security。

    讓我們花點時間來分析此博客文章中描述的方法的優缺點:

    優點:

    • 我們可以編寫集成測試,而無需使用外部社交登錄提供程序。 這使我們的測試不那么脆弱,更易于維護。
    • Spring Social( ProviderSignInAttempt )和Spring Security CSRF保護( CsrfToken )的實現細節被“隱藏”以測試數據構建器類。 這使我們的測試更具可讀性,更易于維護。

    缺點:

    • 本教程沒有描述我們如何編寫社交登錄集成測試(使用社交登錄提供程序登錄)。 我試圖找出一種無需使用外部登錄提供程序即可編寫這些測試的方法,但我只是用光了時間(這似乎很復雜,我想發布此博客文章)。

    這篇博客文章結束了我的“向Spring MVC應用程序添加社交登錄”教程。

    我將寫一個類似的教程,描述如何在將來將社交登錄添加到基于Spring的REST API中。 同時,您可能需要閱讀本教程的其他部分 。

    • 您可以從Github獲得此博客文章的示例應用程序。

    翻譯自: https://www.javacodegeeks.com/2014/03/adding-social-sign-in-to-a-spring-mvc-web-application-integration-testing.html

    集成spring mvc

    總結

    以上是生活随笔為你收集整理的集成spring mvc_向Spring MVC Web应用程序添加社交登录:集成测试的全部內容,希望文章能夠幫你解決所遇到的問題。

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

    99久久网站 | 日韩精品最新在线观看 | 亚州人成在线播放 | 黄色毛片视频免费观看中文 | 在线a视频| 精品亚洲免费视频 | 久久久久欧美精品999 | 亚洲国产精品va在线看黑人 | 国产69久久| 国产色婷婷在线 | 国产精国产精品 | 中文字幕 欧美性 | 久久国产精品一二三区 | 国产原创91 | 免费国产亚洲视频 | 开心激情婷婷 | 午夜视频在线观看一区二区三区 | 91成人黄色 | 国产福利网站 | 国产精品热 | 久久久久国产a免费观看rela | 五月婷婷中文 | 久草免费电影 | 国产麻豆剧果冻传媒视频播放量 | 在线观看韩国av | 国产精品一区二区在线免费观看 | 日韩欧美精品一区二区三区经典 | 欧美精品做受xxx性少妇 | 亚洲免费a| 成人中文字幕av | av电影在线观看完整版一区二区 | 欧美aaa一级 | 国产一级在线 | 天堂av影院 | 午夜少妇一区二区三区 | 成人xxxx| 草久在线视频 | 久久久久欧美精品999 | 久久视频免费在线 | 色婷婷国产 | 成人av在线看 | 日韩最新在线 | 欧美激情另类 | 黄色一级免费网站 | 天天天在线综合网 | 国产成人精品综合久久久 | 日韩系列在线 | 亚洲天堂精品视频 | 国产91精品一区二区绿帽 | 极品国产91在线网站 | 国产精品久久久久av福利动漫 | 小草av在线播放 | 欧美五月婷婷 | 亚洲va男人天堂 | 最新一区二区三区 | 欧美日韩亚洲一 | 国产 成人 久久 | 亚洲一区不卡视频 | 狠狠操操操 | 91九色视频观看 | 中文字幕精 | 日韩免费高清 | 国产成a人亚洲精v品在线观看 | 国产成人一区二区三区免费看 | 中文字幕在线观看完整版 | 国产精品久久久久久婷婷天堂 | 五月婷影院 | 色婷婷狠| 中文av在线免费观看 | 国产精品久久久一区二区三区网站 | 在线观看色网站 | 在线国产一区二区 | 久久精品国产免费看久久精品 | 亚洲三级视频 | 国产精品va在线 | 久久夜色网 | 6080yy精品一区二区三区 | 亚洲欧美国产日韩在线观看 | 国产精品国内免费一区二区三区 | 国产一区视频导航 | 精品国产一区二区三区四区vr | 日韩一区二区三区不卡 | 免费观看一区二区三区视频 | 黄色在线看网站 | 国产首页 | 91.精品高清在线观看 | 在线观看国产高清视频 | 毛片a级片 | 久久69av| 国内外激情视频 | 中文字幕丝袜制服 | 婷婷激情5月天 | 麻豆视频国产 | 97高清免费视频 | 国产精品一区二区视频 | 国内精品久久久久国产 | 成人一区二区三区中文字幕 | 亚洲午夜精品久久久久久久久 | 中文字幕成人在线 | 国产色就色 | 色av婷婷| 超碰在线日本 | 国产精品色视频 | 香蕉影院在线 | 日韩免费成人 | 在线观看国产一区二区 | 国产手机在线观看视频 | 中文字幕超清在线免费 | 97超碰在线久草超碰在线观看 | 亚洲精品国偷拍自产在线观看 | 国产 日韩 在线 亚洲 字幕 中文 | 成人av免费播放 | 深夜免费网站 | 日日夜夜精品免费观看 | 久久综合射 | 三级黄色在线 | 九九久久国产 | 中文字幕在线日亚洲9 | 国产精品欧美久久 | 黄色av电影免费观看 | 黄色av电影一级片 | 成人资源在线观看 | 国产精品免费观看在线 | 伊人网av | 最近中文字幕高清字幕免费mv | 色欲综合视频天天天 | 日日草天天草 | 亚洲欧美视屏 | 亚洲黄色免费在线 | 人人草在线视频 | 91桃色在线免费观看 | 久久福利国产 | 99久久精品电影 | 亚洲激精日韩激精欧美精品 | 色久av| 国产精品久久久久久久久软件 | 欧美黄色成人 | 日日摸日日添夜夜爽97 | 色网站在线 | 丁香视频 | 99国产精品久久久久久久久久 | 最新中文字幕在线资源 | 九色视频网址 | 精品a在线| 国产高清视频免费观看 | 西西444www大胆高清图片 | 黄色三级免费观看 | 五月婷婷六月丁香在线观看 | 激情丁香5月 | 91chinesexxx | 狠狠色丁香久久婷婷综合_中 | 色婷婷狠狠五月综合天色拍 | 免费看黄网站在线 | 久草国产视频 | 激情欧美日韩一区二区 | 精品免费视频123区 午夜久久成人 | 午夜久久影视 | 日本最新高清不卡中文字幕 | 亚洲精品福利视频 | 91热视频在线观看 | av免费看av| 99国产精品 | 亚洲视频 中文字幕 | 国产中文在线播放 | 精品国产亚洲在线 | 欧美亚洲精品一区 | 国产亚洲精品久久久久久网站 | 手机av永久免费 | 91丨porny丨九色 | 国产精品孕妇 | 日韩videos| 久久刺激视频 | av一区二区三区在线观看 | 精品成人a区在线观看 | 免费成人结看片 | 成人免费观看在线视频 | 一区在线观看 | 精品一区91 | 精品亚洲欧美无人区乱码 | 中文字幕三区 | 97国产情侣爱久久免费观看 | 91麻豆视频 | 中文不卡视频在线 | 午夜 免费 | 久久国产精品视频 | 精品国产1区二区 | 天天射天天爱天天干 | 美女免费网站 | 久久综合色影院 | 亚洲国产中文字幕在线观看 | 啪啪动态视频 | 国产精品精品国产 | 色久网 | 亚洲精品乱码久久久久久蜜桃91 | 一区二区精品在线视频 | 午夜精品久久久久久久久久 | 国产天天综合 | 久久99久久精品国产 | 欧美黄在线 | 97超级碰碰碰碰久久久久 | 黄色三级免费看 | 久久大视频 | 婷婷丁香九月 | 天天色天天操综合网 | 在线观看 亚洲 | 91精品一区二区三区久久久久久 | 久久视频这里有精品 | 亚洲aⅴ乱码精品成人区 | 麻豆传媒精品 | 福利视频入口 | 色视频网站免费观看 | 五月天久久婷婷 | 黄色中文字幕在线 | 国产精品久久 | 久久激情电影 | 亚洲国产精久久久久久久 | 久久免费视频2 | 欧美日韩国产综合一区二区 | av片子在线观看 | 国产视频精品免费 | 国产黄色片在线免费观看 | 在线观看黄| 久久99国产视频 | 在线观看精品一区 | 亚洲天堂va | 日本三级吹潮在线 | 国产裸体永久免费视频网站 | 亚洲免费成人 | 色网免费观看 | 国产黄色免费 | 激情图片久久 | 91国内产香蕉 | 久久99精品久久只有精品 | 国产精品美女久久久 | 日韩在线二区 | 在线观看不卡视频 | 日韩网站在线免费观看 | 国产精品国内免费一区二区三区 | 在线香蕉视频 | 热久久视久久精品18亚洲精品 | 久久综合视频网 | 天天做日日做天天爽视频免费 | 免费亚洲精品 | 久草资源在线观看 | 国产一区在线精品 | 豆豆色资源网xfplay | 亚洲国产精品va在线看黑人 | 国产黄网在线 | 欧美精品一区二区蜜臀亚洲 | 成人在线超碰 | 九九99| 久久精品一| 在线国产一区 | 国产精品mm | 国产黄色片一级三级 | 亚洲精品国产品国语在线 | 97在线观看视频国产 | 国产麻豆电影在线观看 | 91亚洲视频在线观看 | 日韩av电影免费在线观看 | 亚洲国产精品va在线 | 麻豆视频在线免费观看 | 亚洲不卡av一区二区三区 | 在线观看国产永久免费视频 | 欧美日韩免费一区二区三区 | 色播五月激情五月 | 国产91成人在在线播放 | 久久久久亚洲国产 | 久久艹99 | 操久在线 | 绯色av一区 | 国产91国语对白在线 | 国产成人久久精品 | 黄色免费观看网址 | 91麻豆视频 | 日本一区二区高清不卡 | 97影视| 98超碰在线观看 | 日韩精品第1页 | 日本韩国精品一区二区在线观看 | 亚洲日日射| 黄色亚洲| 高清不卡一区二区在线 | 超碰人人干人人 | 久久精品视频观看 | 五月天久久久久 | 欧美精品久久久久久 | 奇人奇案qvod | 黄色成人在线 | 人人爽人人爽人人爽学生一级 | 日韩欧美电影在线观看 | 四月婷婷在线观看 | 最新日本中文字幕 | 国产中文字幕大全 | 黄色一区三区 | 狠狠操.com | 99精品观看 | 国产中文字幕亚洲 | 激情影院在线观看 | 视频在线国产 | 国产特级毛片 | 日韩国产在线观看 | 亚洲精品国产高清 | 久久激情网站 | 亚洲精品影院在线观看 | 色国产精品 | 国产精品理论在线观看 | 亚洲乱码中文字幕综合 | 亚洲精品资源在线观看 | 亚洲精品乱码久久久久久蜜桃动漫 | 麻豆视频免费在线 | 国产在线观看免 | 99色人 | 婷婷久久一区二区三区 | 日本h视频在线观看 | 高潮毛片无遮挡高清免费 | 人人澡澡人人 | 久久高清 | 国产视频丨精品|在线观看 国产精品久久久久久久久久久久午夜 | 亚洲综合在线五月天 | 91成人免费在线 | 久久曰视频 | 色婷在线 | 深爱激情亚洲 | 中文字幕亚洲精品日韩 | 日韩久久精品一区二区 | 人人干在线观看 | 国产精品黄 | 日本成人中文字幕在线观看 | 五月综合在线观看 | 超碰在线97免费 | 91精品毛片| 久久综合九色欧美综合狠狠 | 日韩欧美一区二区在线 | www免费看| 国产亚洲精品成人av久久影院 | 六月丁香激情网 | 一区二区三区高清 | 国产一级黄色av | 国产精品s色 | 国产精品九九久久99视频 | 伊人五月天综合 | 国内成人av | 99久久精品久久久久久清纯 | 亚洲精品乱码久久久久久蜜桃不爽 | 成 人 黄 色 免费播放 | 中文字幕乱码在线播放 | 天天操天天干天天爱 | 国产二级视频 | www.久久婷婷 | 日韩精品久久久久久久电影竹菊 | 亚洲午夜精品电影 | 手机在线中文字幕 | av在线电影播放 | 香蕉视频免费看 | 天天操欧美| 91av视屏 | 国产剧情一区在线 | 欧美另类美少妇69xxxx | 伊人永久| 91看片成人 | 色噜噜狠狠色综合中国 | 黄色aaa毛片 | av中文字幕在线播放 | www.色爱| 在线国产欧美 | 超碰在线观看99 | 狠狠干中文字幕 | 综合色影院| 国产精品久久久久久一区二区 | 深爱激情五月综合 | 欧美成年人在线视频 | 久久开心激情 | 中文字幕观看在线 | 日日天天狠狠 | 国产日韩精品一区二区在线观看播放 | 成人午夜剧场在线观看 | 在线黄色观看 | 日韩激情精品 | 天天激情站 | 四虎国产精品永久在线国在线 | 99综合影院在线 | 免费试看一区 | 国产主播大尺度精品福利免费 | 亚洲人在线| 精品久久1 | 天堂网av在线 | 国产精品一区二区中文字幕 | 国产精品三级视频 | 天天操天天干天天操天天干 | 久草精品视频 | 在线 欧美 日韩 | 最新国产精品视频 | 成人啪啪18免费游戏链接 | 97超碰总站| 91成人在线视频 | 欧美激情第八页 | 国产在线高清视频 | 亚洲国产欧洲综合997久久, | 日韩91在线 | 国产精品免费观看久久 | 香蕉视频在线看 | www天天干 | 69国产在线观看 | 天天人人 | 国产精品人人做人人爽人人添 | 国产综合精品一区二区三区 | 一区二区三区观看 | 五月开心网| 人人藻人人澡人人爽 | 久久综合九色九九 | 视频在线播放国产 | 97成人在线 | 国产精品欧美久久久久天天影视 | 91电影福利 | 中文av在线播放 | 99欧美精品 | 91九色国产在线 | 亚洲激情小视频 | 99热官网| 一区二区网 | 91香蕉视频黄色 | 日韩 在线a | 91欧美国产 | 天天操操操操操操 | 精品国产中文字幕 | 日本激情动作片免费看 | 2019中文字幕网站 | 国产精品成人av电影 | 日本性xxx| 国产91在线免费视频 | 国内精品久久久久久久久久清纯 | 东方av在| 操操日日| 亚洲精品资源在线观看 | 亚洲人成在线观看 | 高清免费在线视频 | 国产精品不卡视频 | 99热超碰| 日韩久久午夜一级啪啪 | 91精品影视 | 五月婷婷在线综合 | 久久狠狠亚洲综合 | 四虎视频 | 天天操天天干天天综合网 | 欧美亚洲一级片 | 欧美日韩视频在线一区 | 国产精品亚洲a | 亚洲国产精品成人精品 | 国产美女久久 | 亚洲精品视频免费 | 激情久久伊人 | 婷婷久久五月天 | 精品在线一区二区 | 女人魂免费观看 | 国产福利专区 | aaa毛片视频| 手机在线中文字幕 | 丁香花在线视频观看免费 | 成人一级片视频 | 欧美精品在线观看 | 成全在线视频免费观看 | 久久久久 | 日韩欧美高清免费 | 毛片一区二区 | 久久高清片 | 国产视频一区精品 | 五月婷激情 | 久久久久久久久久久久久久免费看 | 国产a视频免费观看 | 久久乱码卡一卡2卡三卡四 五月婷婷久 | 亚洲视频 中文字幕 | 国产99免费视频 | 特级西西人体444是什么意思 | 91精品国产91久久久久 | 一本色道久久综合亚洲二区三区 | 色播五月婷婷 | 婷婷九月丁香 | 免费在线观看日韩 | 国产精品porn| 天天做天天爱天天爽综合网 | 国产视频一区二区在线 | 亚洲精品视频免费在线观看 | 久草免费电影 | 国模精品一区二区三区 | 蜜臀av在线一区二区三区 | 国产成人精品久 | 亚洲妇女av| 亚洲欧美在线综合 | 在线成人免费电影 | 人人搞人人干 | 国产精品久久久久久久久久ktv | www色com| 久久久久国产精品午夜一区 | 综合激情av | 99爱精品在线| 玖玖玖影院 | 国产精品剧情在线亚洲 | 国内成人精品视频 | 精品一区二区三区四区在线 | 麻豆一级视频 | 91精品久久久久久综合五月天 | 欧美极品少妇xbxb性爽爽视频 | 日本狠狠色 | 又紧又大又爽精品一区二区 | 二区中文字幕 | 夜夜操网 | 欧美日韩在线播放 | 黄色软件在线观看免费 | 九九免费在线看完整版 | 视频国产精品 | 精品国产一二三四区 | 中文字幕资源网在线观看 | 国产精品国产三级国产不产一地 | 99理论片| 伊人精品在线 | 国产中文伊人 | 在线观看视频精品 | 国产成人在线播放 | 中文不卡视频 | 色婷婷综合视频在线观看 | 在线观看成人网 | 精品国模一区二区 | 99精品在这里| 国内外激情视频 | 久久国色夜色精品国产 | 久久视频6 | 中文字幕 国产视频 | 日韩视频在线不卡 | 国产精品视频线看 | 97超碰伊人 | 免费网站色 | 在线播放第一页 | 夜夜夜影院 | 丁香花在线观看免费完整版视频 | 香蕉视频在线免费 | 日韩欧美v| 成人在线电影观看 | 日日综合 | 成人国产精品久久久 | 国产成年免费视频 | 免费人人干 | 国模视频一区二区三区 | 黄色大片中国 | 久久不射电影网 | 一区二区三区不卡在线 | av播放在线 | 最近免费中文字幕 | 99国产在线视频 | a黄在线观看 | 国产精品自产拍在线观看 | 成人亚洲精品久久久久 | av噜噜噜在线播放 | 色综合久久88色综合天天人守婷 | 97超碰伊人 | 久久综合婷婷国产二区高清 | 亚洲成av人片在线观看www | 97视频免费在线 | 人人爽夜夜爽 | 韩国av一区二区 | 99这里精品| 欧美久久久久久久久中文字幕 | www国产一区 | 2018好看的中文在线观看 | 天天综合精品 | 人人爽夜夜爽 | 国产精品久久久久免费观看 | 97国产在线观看 | 中文字幕国产一区二区 | 精品视频97 | 久青草影院| 免费视频 三区 | 婷婷去俺也去六月色 | 国产一级特黄电影 | 欧美最爽乱淫视频播放 | 男女拍拍免费视频 | 麻花天美星空视频 | 麻豆视频国产精品 | 国产视频精品久久 | 成人免费视频视频在线观看 免费 | 日韩国产在线观看 | 天天操天天爱天天干 | 日韩av一区二区三区四区 | 婷婷丁香激情 | 国产精品毛片一区二区在线看 | 在线亚洲欧美日韩 | 99在线免费视频 | 中文字幕在线免费看线人 | 中文字幕在线观看完整版 | 超碰在线观看97 | 久久久久久久亚洲精品 | 国产精品6 | 操操操天天操 | 在线国产激情视频 | 天天操天天曰 | 日韩高清免费电影 | 伊人天天狠天天添日日拍 | 香蕉影视在线观看 | 久久久久一区二区三区 | 亚洲精品乱码久久久久v最新版 | 最近字幕在线观看第一季 | 99热最新在线 | 亚洲乱码久久 | 日韩视频免费 | 日韩精品欧美精品 | 三级毛片视频 | 正在播放国产一区 | 欧美精品国产综合久久 | 日韩狠狠操| 日韩性片 | 国产毛片久久久 | 国产剧情一区在线 | 久久精品视频免费观看 | 在线精品播放 | 草久在线观看视频 | 91激情在线视频 | 香蕉在线视频观看 | 蜜臀av免费一区二区三区 | 欧美久久久久久久久久久 | 色www免费视频| 久久精品一区二区三区四区 | 久要激情网 | 成年人免费看的视频 | 色网av| 黄色片网站 | 亚洲精品自拍视频在线观看 | 成人一级免费电影 | 丁香九月婷婷 | 国产69久久久欧美一级 | 夜夜骑首页 | 国产精品一区二区三区四区在线观看 | 精品一区二区免费在线观看 | 亚洲天天摸日日摸天天欢 | a天堂最新版中文在线地址 久久99久久精品国产 | 国产成人av一区二区三区在线观看 | 免费看片成年人 | 久久久久综合精品福利啪啪 | 欧美日韩国产mv | 国产高清专区 | 久久手机免费观看 | 国产视频二区三区 | 色激情五月 | 亚洲国产精品第一区二区 | 欧美精品久久人人躁人人爽 | 国产在线 一区二区三区 | 欧美成年人在线观看 | 友田真希x88av | 久久久久久久国产精品 | 日韩精品偷拍 | 亚洲视频免费在线 | 欧美精品色 | 国产午夜三级一区二区三 | 少妇自拍av | 国内精品久久久久久久 | 国产精品成人一区二区三区吃奶 | 亚洲视频一级 | 国产精品女人久久久久久 | 91大神精品视频在线观看 | 狠狠操综合网 | 亚洲综合在线发布 | 一级c片 | 成年人看片网站 | 91精品免费在线 | 97在线看片| 亚洲国产中文在线 | 国产小视频精品 | 在线播放 日韩专区 | 在线亚洲欧美日韩 | 91福利视频免费观看 | 日韩中文字幕国产 | 亚州精品国产 | 日韩精品中文字幕在线观看 | 亚洲国产成人在线播放 | 亚洲电影av在线 | 国产成人久久精品一区二区三区 | 国产涩涩网站 | 蜜臀一区二区三区精品免费视频 | 欧美成人按摩 | 成人午夜网 | 国产婷婷久久 | 日韩精品第1页 | 国产在线观看免费观看 | 国产成人一二三 | 国产在线精品区 | 色综合久久久久综合体 | 国产视频一区在线免费观看 | 99色网站 | 久久久精品国产一区二区 | 精品久久国产一区 | 免费在线观看av片 | 久热爱| www色av| 日韩在线国产 | 在线看的av网站 | 久久人人爽 | 久久人人爽爽人人爽人人片av | 日韩欧美国产精品 | 日韩网站一区二区 | 在线观看成人网 | 欧美超碰在线 | 日韩区视频 | 天天干天天操天天操 | 色大片免费看 | 久久久久免费 | 91在线播放综合 | 免费高清在线观看电视网站 | 日本特黄特色aaa大片免费 | 久久久精品免费看 | 在线99视频| 999成人精品| 人人澡人摸人人添学生av | 亚洲精品国产成人 | 又黄又爽又无遮挡的视频 | 91香蕉视频污在线 | 午夜av大片 | 又黄又刺激的视频 | 五月天av在线 | 日本在线成人 | 视频1区2区| 天天操天天弄 | 午夜精品一区二区三区在线观看 | 久久精品2| 亚洲精品中文字幕在线观看 | 欧美视频在线二区 | 安徽妇搡bbbb搡bbbb | 久久伊人热 | 日韩中文字幕在线看 | 成人毛片一区二区三区 | 2022久久国产露脸精品国产 | 狠狠狠狠狠狠 | 国产精品日韩久久久久 | 蜜臀av性久久久久av蜜臀三区 | av色图天堂网 | 久久深夜福利免费观看 | 亚洲精品国产精品国产 | 国产又粗又长的视频 | 国产剧情在线一区 | a视频在线观看免费 | 一级性生活片 | 99免费看片 | 国产成人精品一区二区三区福利 | 日韩精品一区电影 | 久久免费一级片 | 夜夜天天干 | 97精品国产97久久久久久免费 | 久久av在线播放 | 国产精品久久久影视 | 色网站在线| av免费在线观看网站 | 美女黄频免费 | 亚洲电影第一页av | 正在播放亚洲精品 | 午夜影视剧场 | 奇米777777 | 国产精品精品国产婷婷这里av | 久草在线视频免费资源观看 | 美女在线免费观看视频 | 成人在线免费看视频 | 亚洲精品视频在线观看免费视频 | 精品国产欧美 | 亚洲欧美精品在线 | 国产一区在线不卡 | 久久精品中文 | 免费看的黄色网 | 啪啪免费观看网站 | www五月天婷婷 | 黄污视频网站大全 | 亚洲午夜精品福利 | 成人午夜电影免费在线观看 | 午夜在线看 | 国产在线观看免 | 亚洲三级性片 | 中文字幕在线专区 | www操操操 | 香蕉久草在线 | 国产精品综合在线观看 | 女人魂免费观看 | 五月天激情视频 | 黄色网www | 91精品成人久久 | 99爱视频在线观看 | 精品国产美女在线 | 久草香蕉在线视频 | 国产九色在线播放九色 | 中文字幕国产精品一区二区 | 中文字幕激情 | 日日草夜夜操 | 97超碰香蕉| 久久国产亚洲精品 | 久久色在线观看 | 日韩精品一区二区在线视频 | 91最新地址永久入口 | 一区二区中文字幕在线播放 | 国产欧美精品一区二区三区四区 | 日韩二三区 | 久草网站在线观看 | 精品国产精品国产偷麻豆 | 91欧美在线| 婷婷色视频| 久久亚洲电影 | 国产精品视频地址 | 九九有精品 | 天天曰夜夜操 | 日韩免费观看一区二区三区 | 911亚洲精品第一 | 久久天天拍 | 亚洲一区二区三区四区精品 | 色资源中文字幕 | 成人午夜电影在线观看 | 超碰免费在线公开 | 中文字幕亚洲欧美日韩2019 | 就色干综合 | 成人av中文字幕 | 69国产精品视频免费观看 | 日韩资源在线播放 | 日本久久电影网 | 91精品天码美女少妇 | 中文字幕免费看 | 九九免费视频 | 亚洲国产精彩中文乱码av | 在线视频日韩精品 | 国产精品99精品久久免费 | 久草综合在线观看 | 韩国av免费在线观看 | 中文字幕在线色 | 欧美日韩中文在线视频 | 三级av免费观看 | 2017狠狠干 | 婷婷六月丁 | 91九色国产蝌蚪 | 国产成人精品综合久久久久99 | 亚洲成人精品影院 | 91av视频导航| 日本中文乱码卡一卡二新区 | 香蕉97视频观看在线观看 | 国产中文视 | 亚洲欧洲在线视频 | 99爱国产精品 | 国产香蕉在线 | 国产美女视频网站 | 免费视频一二三区 | 天天激情综合 | 日韩久久在线 | 欧美精品三级 | 色黄久久久久久 | av先锋中文字幕 | 欧美精品久久久久久久久久丰满 | 91精品国产自产在线观看 | 午夜在线看片 | 91九色九色| 久久第四色 | 亚洲成熟女人毛片在线 | 99精品国产一区二区 | 国产群p| 国内精品久久久久久久影视麻豆 | 国产精品福利在线观看 | 国产成人精品一区二区在线 | 人人玩人人添人人澡超碰 | 激情五月色播五月 | 日本久热 | 亚洲视频六区 | 日韩久久精品一区二区三区 | 亚洲国产中文字幕 | 久久视频免费在线观看 | 亚洲专区免费观看 | av超碰在线观看 | 日日操操操 | 17婷婷久久www | 91av小视频 | 国产精品 9999| 美女免费黄网站 | 日本三级中文字幕在线观看 | 欧美日韩高清在线一区 | 日韩二区三区在线 | 久久视频免费看 | 免费毛片aaaaaa | 久久久久女人精品毛片 | 天天干天天摸 | 亚洲首页| 精品欧美乱码久久久久久 | 99精品国产在热久久下载 | 亚洲综合欧美精品电影 | 西西444www| 91正在播放 | 91精品国自产在线观看欧美 | 91免费观看| 超碰免费久久 | 视频在线精品 | 色橹橹欧美在线观看视频高清 | 日韩免费在线网站 | 色香天天| av黄免费看 | 久久久久成人精品 | 男女拍拍免费视频 | 一区二区三区免费在线观看视频 | 日韩欧美精品在线观看 | 日韩电影在线观看一区二区三区 | 国产精品免费观看视频 | 国产高清视频网 | 亚洲精品小视频在线观看 | 99视频精品免费观看, | 国产高清日韩欧美 | 96视频在线 | 成 人 黄 色 视频免费播放 | 一区二区久久久久 | 黄色国产高清 | 成年人在线电影 | 看毛片网站 | 久草网在线观看 | 黄色一级大片在线免费看国产一 | 国产香蕉视频在线播放 | av成人动漫 | 国产又粗又猛又爽 | 久久99精品久久久久久久久久久久 | 免费精品视频 | 天天玩天天操天天射 | 在线草 | 国产日韩在线播放 | 国产精品成久久久久三级 | 夜夜躁狠狠躁日日躁视频黑人 | 在线看的毛片 | 久久久福利影院 | 午夜免费福利视频 | 日本 在线 视频 中文 有码 | 免费精品国产 | 五月天av在线 | 国产精品久久在线观看 | 福利视频一区二区 | 国产精品va在线观看入 | 韩国一区在线 | 亚洲精品高清在线 | 久久av中文字幕片 | 亚洲japanese制服美女 | 日韩久久久 | 日本久久免费电影 | 狠狠色丁香婷婷综合欧美 | 国产精品入口麻豆www | 日韩久久久久久久久久 | 毛片永久新网址首页 | www视频免费在线观看 | 日日射av | 日日婷婷夜日日天干 | av黄色免费看 | 中文字幕a∨在线乱码免费看 | av一区二区在线观看中文字幕 | 综合黄色网 | 久久精品香蕉视频 | 国产精品 中文在线 | 美女黄久久 | 国产裸体视频bbbbb | 999久久久| 天天射天天干天天操 | 99精品视频免费观看 | 亚洲 欧美 变态 国产 另类 | 久久免费视频在线观看6 | 亚洲综合激情小说 | 国内小视频在线观看 | 激情网五月婷婷 | 国产麻豆精品95视频 | 91自拍成人 | 成人aaa毛片 | 国际精品久久久久 | 国产精品久久久久久久久久久久久久 | 日韩精品久久久久 | 久碰视频在线观看 | 久久久久久久久久电影 | 亚洲2019精品 | 91免费视频黄 | 国产一区二区精品91 | 黄色福利网 | 中文字幕免费高清 | 久久av影视| 久久免费电影 | 国产高清在线免费视频 | 成年人在线电影 | 国产精品久久久精品 | 欧美色图视频一区 | 天天色天天射天天综合网 | 精品久久久99 | 亚洲五月婷 | 国产精品成人一区二区三区吃奶 | 91在线看视频免费 | 亚州精品一二三区 | 欧美另类一二三四区 | 久久久国产精品免费 | 999久久a精品合区久久久 | 手机版av在线 | 久久久天天操 | 国产成人av网址 | 中文字幕国产在线 | 日韩精品欧美专区 | 99在线精品观看 | 天天色天天操天天爽 | www·22com天天操 | 综合伊人久久 | 人人看人人艹 | 97超碰在线免费 | 国产98色在线 | 日韩 |