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

歡迎訪問(wèn) 生活随笔!

生活随笔

當(dāng)前位置: 首頁(yè) > 前端技术 > javascript >内容正文

javascript

在Spring MVC Web应用程序中添加社交登录:集成测试

發(fā)布時(shí)間:2023/12/3 javascript 48 豆豆
生活随笔 收集整理的這篇文章主要介紹了 在Spring MVC Web应用程序中添加社交登录:集成测试 小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

我已經(jīng)寫了關(guān)于為使用Spring Social 1.1.0的應(yīng)用程序編寫單元測(cè)試的挑戰(zhàn),并為此提供了一種解決方案 。

盡管單元測(cè)試很有價(jià)值,但是它并不能真正告訴我們我們的應(yīng)用程序是否正常運(yùn)行。

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

這篇博客文章可以幫助我們做到這一點(diǎn)。 在此博客文章中,我們將學(xué)習(xí)如何為示例應(yīng)用程序的注冊(cè)和登錄功能編寫集成測(cè)試。

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

  • 在Spring MVC Web應(yīng)用程序中添加社交登錄:配置描述了如何配置示例應(yīng)用程序。
  • 在Spring MVC Web應(yīng)用程序中添加社交登錄:注冊(cè)和登錄介紹了如何向示例應(yīng)用程序添加注冊(cè)和登錄功能。
  • 在Spring MVC Web應(yīng)用程序中添加社交登錄:單元測(cè)試描述了如何為示例應(yīng)用程序編寫單元測(cè)試。
  • Spring MVC測(cè)試教程描述了如何使用Spring MVC Test框架編寫單元測(cè)試和集成測(cè)試。
  • Spring Data JPA教程:集成測(cè)試描述了我們?nèi)绾螢镾pring Data JPA存儲(chǔ)庫(kù)編寫集成測(cè)試。 這篇博客文章幫助您了解如何使用Spring Test DBUnit和DbUnit編寫集成測(cè)試。
  • 使用Maven進(jìn)行集成測(cè)試介紹了如何使用Maven運(yùn)行集成測(cè)試和單元測(cè)試。 我們的示例應(yīng)用程序的構(gòu)建過(guò)程遵循此博客文章中描述的方法。

讓我們從對(duì)構(gòu)建過(guò)程的配置進(jìn)行一些更改開始。

配置我們的構(gòu)建過(guò)程

我們必須對(duì)構(gòu)建過(guò)程的配置進(jìn)行以下更改:

  • 我們已經(jīng)配置了一個(gè)本地Maven存儲(chǔ)庫(kù),并將Spring Test DbUnit 1.1.1快照二進(jìn)制文件添加到該存儲(chǔ)庫(kù)。
  • 我們必須將所需的測(cè)試依賴項(xiàng)添加到我們的POM文件中。
  • 我們必須將Liquibase變更集文件添加到classpath。
  • 讓我們找出如何進(jìn)行這些更改。

    將Spring Test DBUnit快照二進(jìn)制文件添加到本地Maven存儲(chǔ)庫(kù)

    由于Spring Test DBUnit的穩(wěn)定版本與Spring Framework 4不兼容 ,因此我們必須在集成測(cè)試中使用構(gòu)建快照。

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

  • 從Github克隆Spring Test DBUnit存儲(chǔ)庫(kù)并創(chuàng)建快照二進(jìn)制文件。
  • 創(chuàng)建etc / mavenrepo目錄。 該目錄是我們本地的Maven存儲(chǔ)庫(kù)。
  • 將創(chuàng)建的jar文件復(fù)制到etc / mavenrepo / com / github / springtestdbunit / 1.1.1-SNAPSHOT目錄。
  • 將jar文件復(fù)制到本地Maven存儲(chǔ)庫(kù)后,我們必須在pom.xml文件中配置本地存儲(chǔ)庫(kù)的位置。 我們可以通過(guò)將以下存儲(chǔ)庫(kù)聲明添加到我們的POM文件中來(lái)做到這一點(diǎn):

    <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獲取所需的測(cè)試依賴項(xiàng)

    通過(guò)將以下依賴項(xiàng)聲明添加到我們的POM文件中,我們可以獲得所需的測(cè)試依賴項(xiàng):

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

    pom.xml文件的相關(guān)部分如下所示:

    <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變更集添加到類路徑

    通常,我們應(yīng)該讓Hibernate創(chuàng)建用于集成測(cè)試的數(shù)據(jù)庫(kù)。 但是,僅當(dāng)在我們的域模型中配置了每個(gè)數(shù)據(jù)庫(kù)表時(shí),此方法才有效。

    現(xiàn)在不是這種情況。 示例應(yīng)用程序的數(shù)據(jù)庫(kù)具有一個(gè)UserConnection表,該表未在示例應(yīng)用程序的域模型中配置。 這就是為什么我們需要在運(yùn)行集成測(cè)試之前找到另一種方法來(lái)創(chuàng)建UserConnection表。

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

    我們可以通過(guò)使用Build Helper Maven插件來(lái)實(shí)現(xiàn) 。 我們可以按照以下步驟將Liquibase變更集添加到類路徑中:

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

    <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>

    如果要獲取有關(guān)使用Builder Helper Maven插件的用法的更多信息,可以查看以下網(wǎng)頁(yè):

    • 與Maven的集成測(cè)試
    • Builder Helper Maven插件

    現(xiàn)在,我們已經(jīng)完成了構(gòu)建過(guò)程的配置。 讓我們找出如何配置集成測(cè)試。

    配置我們的集成測(cè)試

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

  • 修改Liquibase更改日志文件。
  • 在調(diào)用我們的測(cè)試用例之前,配置應(yīng)用程序上下文以運(yùn)行Liquibase變更集。
  • 創(chuàng)建一個(gè)自定義DbUnit數(shù)據(jù)集加載器。
  • 配置集成測(cè)試用例
  • 讓我們繼續(xù)前進(jìn),仔細(xì)看看每個(gè)步驟。

    修改Liquibase變更日志

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

  • db-0.0.1.sql文件創(chuàng)建UserConnection表,該表用于將用戶與已使用的社交登錄提供程序的連接持久化。
  • db-0.0.2.sql文件創(chuàng)建了user_accounts表,其中包含示例應(yīng)用程序的用戶帳戶。
  • 因?yàn)槲覀冎幌脒\(yùn)行第一個(gè)變更集,所以我們必須對(duì)Liquibase變更日志文件進(jìn)行一些修改。 更具體地說(shuō),我們必須使用Liquibase上下文來(lái)指定

  • 當(dāng)創(chuàng)建示例應(yīng)用程序的數(shù)據(jù)庫(kù)時(shí),將執(zhí)行哪些變更集。
  • 當(dāng)我們運(yùn)行集成測(cè)試時(shí),將執(zhí)行哪些變更集。
  • 通過(guò)執(zhí)行以下步驟,我們可以實(shí)現(xiàn)我們的目標(biāo):

  • 指定當(dāng)Liquibase上下文是db或integrationtest時(shí),將執(zhí)行db-0.0.1.sql changeset文件。
  • 指定當(dāng)Liquibase上下文為db時(shí)執(zhí)行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>

    在運(yùn)行集成測(cè)試之前執(zhí)行Liquibase變更集

    通過(guò)在加載應(yīng)用程序上下文時(shí)執(zhí)行集成測(cè)試,我們可以在運(yùn)行集成測(cè)試之前執(zhí)行Liquibase變更集。 我們可以按照以下步驟進(jìn)行操作:

  • 創(chuàng)建一個(gè)IntegrationTestContext類,并使用@Configuration注釋對(duì)其進(jìn)行注釋。
  • 將DataSource字段添加到創(chuàng)建的類中,并使用@Autowired批注對(duì)其進(jìn)行批注。
  • 將liquibase()方法添加到類中,并使用@Bean注釋對(duì)其進(jìn)行注釋。 此方法配置SpringLiquibase bean,該bean在加載應(yīng)用程序上下文時(shí)執(zhí)行l(wèi)iquibase變更集。
  • 通過(guò)執(zhí)行以下步驟來(lái)實(shí)現(xiàn)liquibase()方法:
  • 創(chuàng)建一個(gè)新的SpringLiquibase對(duì)象。
  • 配置創(chuàng)建的對(duì)象使用的數(shù)據(jù)源。
  • 配置Liquibase更改日志的位置。
  • 將Liquibase上下文設(shè)置為“ integrationtest”。
  • 返回創(chuàng)建的對(duì)象。
  • 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;} }

    創(chuàng)建一個(gè)自定義DataSetLoader類

    包含不同用戶帳戶信息的DbUnit數(shù)據(jù)集如下所示:

    <?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>

    我們可以從該數(shù)據(jù)集中看到兩件事:

  • 使用社交登錄創(chuàng)建用戶帳戶的用戶沒(méi)有密碼。
  • 使用常規(guī)注冊(cè)創(chuàng)建其用戶帳戶的用戶具有密碼,但沒(méi)有登錄提供者。
  • 這是一個(gè)問(wèn)題,因?yàn)槲覀兪褂盟^的平面XML數(shù)據(jù)集 ,而默認(rèn)的DbUnit數(shù)據(jù)集加載器無(wú)法處理這種情況 。 當(dāng)然,我們可以開始使用標(biāo)準(zhǔn)XML數(shù)據(jù)集,但就我的口味而言,其語(yǔ)法有點(diǎn)過(guò)于冗長(zhǎng)。 這就是為什么我們必須創(chuàng)建一個(gè)可以處理這種情況的自定義數(shù)據(jù)集加載器的原因。

    我們可以按照以下步驟創(chuàng)建自定義數(shù)據(jù)集加載器:

  • 創(chuàng)建一個(gè)ColumnSensingFlatXMLDataSetLoader類,該類擴(kuò)展了AbstractDataSetLoader類。
  • 覆蓋createDataSet()方法并通過(guò)以下步驟實(shí)現(xiàn)它:
  • 創(chuàng)建一個(gè)新的FlatXmlDataSetBuilder對(duì)象。
  • 啟用列感測(cè)。 列檢測(cè)意味著DbUnit從數(shù)據(jù)集文件中讀取整個(gè)數(shù)據(jù)集,并在從數(shù)據(jù)集中找到新列時(shí)添加新列。 這樣可以確保將每一列的值正確插入數(shù)據(jù)庫(kù)中。
  • 創(chuàng)建一個(gè)新的IDataSet對(duì)象并返回創(chuàng)建的對(duì)象。
  • 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();}} }

    但是,創(chuàng)建自定義數(shù)據(jù)集加載器類還不夠。 加載數(shù)據(jù)集時(shí),我們?nèi)匀槐仨毰渲脺y(cè)試以使用此類。 我們可以通過(guò)使用@DbUnitConfiguration批注注釋測(cè)試類并將其dataSetLoader屬性的值設(shè)置為ColumnSensingFlatXMLDataSetLoader.class來(lái)實(shí)現(xiàn)此目的 。

    讓我們繼續(xù)看這是如何完成的。

    配置我們的集成測(cè)試

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

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

    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();} }

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

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

    現(xiàn)在,我們已經(jīng)了解了如何配置集成測(cè)試。 讓我們繼續(xù)并創(chuàng)建一些在集成測(cè)試中使用的測(cè)試實(shí)用程序類。

    創(chuàng)建測(cè)試實(shí)用程序類

    接下來(lái),我們將創(chuàng)建在集成測(cè)試中使用的三個(gè)實(shí)用程序類:

  • 我們將創(chuàng)建IntegrationTestConstants類,其中包含多個(gè)集成測(cè)試中使用的常量。
  • 我們將創(chuàng)建用于為集成測(cè)試創(chuàng)建ProviderSignInAttempt對(duì)象的類。
  • 我們將創(chuàng)建一個(gè)測(cè)試數(shù)據(jù)構(gòu)建器類,該類用于創(chuàng)建CsrfToken對(duì)象。
  • 讓我們找出為什么我們必須創(chuàng)建這些類以及如何創(chuàng)建它們。

    創(chuàng)建IntegrationTestConstants類

    當(dāng)我們編寫集成(或單元)測(cè)試時(shí),有時(shí)我們需要在許多測(cè)試類中使用相同的信息。 將這些信息復(fù)制到所有測(cè)試類是一個(gè)壞主意,因?yàn)檫@會(huì)使我們的測(cè)試難以維護(hù)和理解。 相反,我們應(yīng)該將此信息放在一個(gè)類中,并在需要時(shí)從該類中獲取。

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

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

    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;}} }

    創(chuàng)建ProviderSignInAttempt對(duì)象

    在為示例應(yīng)用程序編寫單元測(cè)試時(shí),我們快速瀏覽了ProviderSignInUtils類,并意識(shí)到我們必須找到一種創(chuàng)建ProviderSignInAttempt對(duì)象的方法。

    我們通過(guò)創(chuàng)建在單元測(cè)試中使用的存根類來(lái)解決該問(wèn)題。 這個(gè)存根類使我們可以配置返回的Connection <?>對(duì)象,并驗(yàn)證特定的連接是否“持久化到數(shù)據(jù)庫(kù)”。 但是,我們的存根類未持久連接到使用的數(shù)據(jù)庫(kù)。 而是將用戶的用戶ID存儲(chǔ)到Set對(duì)象。

    因?yàn)楝F(xiàn)在我們想將連接數(shù)據(jù)持久化到數(shù)據(jù)庫(kù),所以我們必須對(duì)存根類進(jìn)行更改。 我們可以通過(guò)對(duì)TestProviderSignInAttempt對(duì)象進(jìn)行以下更改來(lái)進(jìn)行這些更改:

  • 將一個(gè)專用的usersConnectionRepositorySet字段添加到TestProviderSignInAttempt類。 該字段的類型為布爾值 ,其默認(rèn)值為false。 該字段描述了我們是否可以持久連接到使用的數(shù)據(jù)存儲(chǔ)。
  • 將一個(gè)新的構(gòu)造函數(shù)參數(shù)添加到TestProviderSignInAttempt類的構(gòu)造函數(shù)中。 此參數(shù)的類型為UsersConnectionRepository ,用于持久連接到使用的數(shù)據(jù)存儲(chǔ)。
  • 通過(guò)執(zhí)行以下步驟來(lái)實(shí)現(xiàn)構(gòu)造函數(shù):
  • 調(diào)用超類的構(gòu)造函數(shù),并將Connection <?>和UsersConnectionRepository對(duì)象作為構(gòu)造函數(shù)參數(shù)傳遞。
  • 存儲(chǔ)對(duì)Connection <?>對(duì)象的引用,該引用作為連接字段的構(gòu)造函數(shù)參數(shù)給出。
  • 如果作為構(gòu)造函數(shù)參數(shù)給出的UsersConnectionRepository對(duì)象不為null,則將usersConnectionRepositoryField的值設(shè)置為true。
  • 通過(guò)執(zhí)行以下步驟來(lái)實(shí)現(xiàn)addConnection()方法:
  • 將作為方法參數(shù)給出的用戶ID添加到連接 Set中 。
  • 如果在創(chuàng)建新的TestProviderSignInAttempt對(duì)象時(shí)設(shè)置了UsersConnectionRepository對(duì)象,請(qǐng)調(diào)用ProviderSignInAttempt類的addConnection()方法,并將用戶ID作為方法參數(shù)傳遞。
  • 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;} }

    因?yàn)槲覀兺ㄟ^(guò)使用TestProviderSignInAttemptBuilder來(lái)構(gòu)建新的TestProviderSignInAttempt對(duì)象,所以我們也必須對(duì)該類進(jìn)行更改。 我們可以按照以下步驟進(jìn)行這些更改:

  • 將私有的usersConnectionRepository字段添加到TestProviderSignInAttemptBuilder類,并將其類型設(shè)置為UsersConnectionRepository 。
  • 向類添加一個(gè)usersConnectionRepository()方法。 將對(duì) UsersConnectionRepository對(duì)象的引用設(shè)置為usersConnectionRepository字段,并返回對(duì)構(gòu)建器對(duì)象的引用。
  • 修改build()方法的最后一行,并使用我們之前創(chuàng)建的新構(gòu)造函數(shù)創(chuàng)建一個(gè)新的TestProviderSignInAttempt對(duì)象。
  • 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);} }

    創(chuàng)建CsrfToken對(duì)象

    因?yàn)槲覀兊氖纠龖?yīng)用程序使用了Spring Security 3.2提供的CSRF保護(hù),所以我們必須找出一種在集成測(cè)試中創(chuàng)建有效CSRF令牌的方法。 CsrfToken接口聲明了提供有關(guān)預(yù)期CSRF令牌信息的方法。 該接口具有一個(gè)稱為DefaultCsrfToken的實(shí)現(xiàn)。

    換句話說(shuō),我們必須找出一種創(chuàng)建新的DefaultCsrfToken對(duì)象的方法。 DefaultCsrfToken類只有一個(gè)構(gòu)造函數(shù) ,在集成測(cè)試中創(chuàng)建新的DefaultCsrfToken對(duì)象時(shí),我們當(dāng)然可以使用它。 問(wèn)題是這不是很可讀。

    相反,我們將創(chuàng)建一個(gè)測(cè)試數(shù)據(jù)構(gòu)建器類 , 該類提供了用于創(chuàng)建新CsrfToken對(duì)象的流利API 。 我們可以按照以下步驟創(chuàng)建此類:

  • 創(chuàng)建一個(gè)CsrfTokenBuilder類。
  • 將私有的headerName字段添加到創(chuàng)建的類。
  • 將私有requestParameterName字段添加到創(chuàng)建的類中。
  • 將私有tokenValue字段添加到創(chuàng)建的類。
  • 向創(chuàng)建的類添加發(fā)布構(gòu)造函數(shù)。
  • 添加用于設(shè)置headerName , requestParameterName和tokenValue字段的字段值的方法。
  • 向所創(chuàng)建的類中添加一個(gè)build()方法,并將其返回類型設(shè)置為CsrfToken 。 通過(guò)執(zhí)行以下步驟來(lái)實(shí)現(xiàn)此方法:
  • 創(chuàng)建一個(gè)新的DefaultCsrfToken對(duì)象,并提供CSRF令牌標(biāo)頭的名稱,CSRF令牌請(qǐng)求參數(shù)的名稱以及CSRF令牌的值作為構(gòu)造函數(shù)參數(shù)。
  • 返回創(chuàng)建的對(duì)象。
  • 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);} }

    我們可以使用以下代碼創(chuàng)建新的CsrfToken對(duì)象:

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

    現(xiàn)在,我們已經(jīng)創(chuàng)建了必需的測(cè)試實(shí)用程序類。 讓我們繼續(xù)并開始為示例應(yīng)用程序編寫集成測(cè)試。

    編寫集成測(cè)試

    我們終于準(zhǔn)備好為示例應(yīng)用程序編寫一些集成測(cè)試。 我們將編寫以下集成測(cè)試:

    • 我們將編寫集成測(cè)試,以確保表單登錄正常工作。
    • 我們將編寫集成測(cè)試,以驗(yàn)證使用社交登錄時(shí)注冊(cè)是否正常運(yùn)行。

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

    向Spring Security提供有效的CSRF令牌

    之前我們了解了如何在集成測(cè)試中創(chuàng)建CsrfToken對(duì)象。 但是,我們?nèi)匀槐仨氄页鲆环N將這些CSRF令牌提供給Spring Security的方法。

    現(xiàn)在是時(shí)候仔細(xì)研究一下Spring Security處理CSRF令牌的方式了。

    CsrfTokenRepository接口聲明生成,保存和加載CSRF令牌所需的方法。 該接口的默認(rèn)實(shí)現(xiàn)是HttpSessionCsrfTokenRepository類,該類將CSRF令牌存儲(chǔ)到HTTP會(huì)話。

    我們需要找到以下問(wèn)題的答案:

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

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

    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. }

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

  • 使用我們的測(cè)試數(shù)據(jù)構(gòu)建器創(chuàng)建一個(gè)新的CsrfToken對(duì)象。
  • 發(fā)送CSRF令牌的值作為請(qǐng)求參數(shù)。
  • 將創(chuàng)建的DefaultCsrfToken對(duì)象存儲(chǔ)到HTTP會(huì)話中,以便HttpSessionCsrfTokenRepository找到它。
  • 我們的虛擬測(cè)試的源代碼如下所示:

    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.} }

    理論足夠了。 現(xiàn)在,我們準(zhǔn)備為我們的應(yīng)用程序編寫一些集成測(cè)試。 首先,將集成編寫到示例應(yīng)用程序的登錄功能中。

    編寫登錄功能測(cè)試

    現(xiàn)在該為示例應(yīng)用程序的登錄功能編寫集成測(cè)試了。 我們將為此編寫以下集成測(cè)試:

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

    <?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>

    讓我們開始吧。

    測(cè)試1:登錄成功

    我們可以按照以下步驟編寫第一個(gè)集成測(cè)試:

  • 使用@DatabaseSetup注釋對(duì)測(cè)試類進(jìn)行注釋,并配置數(shù)據(jù)集,該數(shù)據(jù)集用于在調(diào)用集成測(cè)試之前將數(shù)據(jù)庫(kù)初始化為已知狀態(tài)。
  • 創(chuàng)建一個(gè)新的CsrfToken對(duì)象。
  • 請(qǐng)按照以下步驟將POST請(qǐng)求發(fā)送到url'/ login / authenticate':
  • 設(shè)置用戶名和密碼請(qǐng)求參數(shù)的值。 使用正確的密碼。
  • 將CSRF令牌的值設(shè)置為請(qǐng)求。
  • 將創(chuàng)建的CsrfToken設(shè)置為session。
  • 確保返回HTTP狀態(tài)代碼302。
  • 驗(yàn)證請(qǐng)求已重定向到URL“ /”。
  • 我們的集成測(cè)試的源代碼如下所示:

    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("/"));} }

    測(cè)試2:登錄失敗

    我們可以按照以下步驟編寫第二個(gè)集成測(cè)試:

  • 使用@DatabaseSetup注釋對(duì)測(cè)試類進(jìn)行注釋,并配置數(shù)據(jù)集,該數(shù)據(jù)集用于在調(diào)用集成測(cè)試之前將數(shù)據(jù)庫(kù)初始化為已知狀態(tài)。
  • 創(chuàng)建一個(gè)新的CsrfToken對(duì)象。
  • 請(qǐng)按照以下步驟將POST請(qǐng)求發(fā)送到url'/ login / authenticate':
  • 設(shè)置用戶名和密碼請(qǐng)求參數(shù)的值。 使用錯(cuò)誤的密碼。
  • 將CSRF令牌的值設(shè)置為請(qǐng)求作為請(qǐng)求參數(shù)。
  • 將創(chuàng)建的CsrfToken對(duì)象設(shè)置為session。
  • 確保返回HTTP狀態(tài)代碼302。
  • 驗(yàn)證請(qǐng)求是否重定向到URL'/ login?error = bad_credentials'。
  • 我們的集成測(cè)試的源代碼如下所示:

    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"));} }

    為注冊(cè)功能編寫測(cè)試

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

  • 我們將編寫一個(gè)集成測(cè)試,以確保當(dāng)用戶使用社交登錄創(chuàng)建新用戶帳戶時(shí),我們的應(yīng)用程序正常運(yùn)行,但是提交的注冊(cè)表格的驗(yàn)證失敗。
  • 我們將編寫一個(gè)集成測(cè)試,該測(cè)試將通過(guò)使用社交登錄和從數(shù)據(jù)庫(kù)中找到的電子郵件地址來(lái)驗(yàn)證用戶創(chuàng)建新用戶帳戶時(shí),一切是否正常運(yùn)行。
  • 我們將編寫一個(gè)集成測(cè)試,以確保可以通過(guò)使用社交登錄來(lái)創(chuàng)建新的用戶帳戶。
  • 讓我們開始吧。

    測(cè)試1:驗(yàn)證失敗

    我們可以按照以下步驟編寫第一個(gè)集成測(cè)試:

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

    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"));} }

    我們的集成測(cè)試使用一個(gè)名為no-users.xml的DbUnit數(shù)據(jù)集文件,該文件如下所示:

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

    測(cè)試2:從數(shù)據(jù)庫(kù)中找到電子郵件地址

    我們可以按照以下步驟編寫第二個(gè)集成測(cè)試:

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

    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"));} }

    此集成測(cè)試使用一個(gè)名為users.xml的DbUnit數(shù)據(jù)集,該數(shù)據(jù)集如下所示:

    <?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>

    測(cè)試3:注冊(cè)成功

    我們可以按照以下步驟編寫第三項(xiàng)集成測(cè)試:

  • 將UsersConnectionRepository字段添加到測(cè)試類,并使用@Autowired批注對(duì)其進(jìn)行批注。
  • 使用@DatabaseSetup批注注釋測(cè)試方法,并配置數(shù)據(jù)集,該數(shù)據(jù)集用于在運(yùn)行集成測(cè)試之前將數(shù)據(jù)庫(kù)初始化為已知狀態(tài)。
  • 創(chuàng)建一個(gè)新的TestProviderSignInAttempt對(duì)象。 記住要設(shè)置使用的UsersConnectionRepository對(duì)象。
  • 創(chuàng)建一個(gè)新的RegistrationForm對(duì)象,并設(shè)置其email , firstName , lastName和signInProvider字段的值。
  • 創(chuàng)建一個(gè)新的CsrfToken對(duì)象。
  • 請(qǐng)按照以下步驟將POST請(qǐng)求發(fā)送到url'/ user / register':
  • 將請(qǐng)求的內(nèi)容類型設(shè)置為“ application / x-www-form-urlencoded”。
  • 將表單對(duì)象轉(zhuǎn)換為url編碼的字節(jié),并將其設(shè)置為請(qǐng)求的正文。
  • 將創(chuàng)建的TestProviderSignInAttempt對(duì)象設(shè)置為session。
  • 將CSRF令牌的值設(shè)置為請(qǐng)求作為請(qǐng)求參數(shù)。
  • 將創(chuàng)建的CsrfToken對(duì)象設(shè)置為session。
  • 將創(chuàng)建的表單對(duì)象設(shè)置為session。
  • 確保返回HTTP請(qǐng)求狀態(tài)302。
  • 驗(yàn)證請(qǐng)求是否重定向到URL“ /”。 這還可以確保創(chuàng)建的用戶已登錄,因?yàn)槟涿脩魺o(wú)法訪問(wèn)該URL。
  • 使用@ExpectedDatabase批注對(duì)測(cè)試方法進(jìn)行批注,并確保將新用戶帳戶保存到數(shù)據(jù)庫(kù)中,并且與使用的社交媒體提供者的連接得以持久。
  • 我們的集成測(cè)試的源代碼如下所示:

    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("/"));} }

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

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

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

    <?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>

    摘要

    現(xiàn)在,我們已經(jīng)了解了如何為使用Spring Social 1.1.0的常規(guī)Spring MVC應(yīng)用程序編寫集成測(cè)試。 本教程教會(huì)了我們很多東西,但是這兩件事是本博客文章的主要課程:

    • 我們了解了如何通過(guò)創(chuàng)建ProviderSignInAttempt對(duì)象并在集成測(cè)試中使用它們來(lái)“模擬”社交登錄。
    • 我們學(xué)習(xí)了如何創(chuàng)建CSRF令牌并將創(chuàng)建的令牌提供給Spring Security。

    讓我們花點(diǎn)時(shí)間來(lái)分析此博客文章中描述的方法的優(yōu)缺點(diǎn):

    優(yōu)點(diǎn):

    • 我們可以編寫集成測(cè)試,而無(wú)需使用外部社交登錄提供程序。 這使我們的測(cè)試不那么脆弱,更易于維護(hù)。
    • Spring Social( ProviderSignInAttempt )和Spring Security CSRF保護(hù)( CsrfToken )的實(shí)現(xiàn)細(xì)節(jié)被“隱藏”以測(cè)試數(shù)據(jù)構(gòu)建器類。 這使我們的測(cè)試更具可讀性,更易于維護(hù)。

    缺點(diǎn):

    • 本教程沒(méi)有描述我們?nèi)绾尉帉懮缃坏卿浖蓽y(cè)試(使用社交登錄提供程序登錄)。 我試圖找出一種無(wú)需使用外部登錄提供程序即可編寫這些測(cè)試的方法,但我只是用光了時(shí)間(這似乎很復(fù)雜,我想發(fā)布此博客文章)。

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

    我將寫一個(gè)類似的教程,描述將來(lái)如何在社會(huì)支持的REST API中添加社交登錄。 同時(shí),您可能需要閱讀本教程的其他部分 。

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

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

    總結(jié)

    以上是生活随笔為你收集整理的在Spring MVC Web应用程序中添加社交登录:集成测试的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。

    如果覺(jué)得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。

    www五月天com | 国产精品亚洲人在线观看 | 久久伦理电影网 | a成人v在线 | 在线观看91av | 国产丝袜美腿在线 | 国产中年夫妇高潮精品视频 | 2022中文字幕在线观看 | 亚洲四虎影院 | 美女露久久 | 成人久久久久久久久久 | 色av男人的天堂免费在线 | 亚洲国产一区在线观看 | 亚洲精品电影在线 | 婷婷在线观看视频 | 欧美在线aa | 国产日韩在线播放 | 欧美日韩在线精品一区二区 | 婷婷午夜激情 | 一区二区三区四区五区在线视频 | 五月天丁香视频 | 999日韩 | 成人中心免费视频 | 久久精品国产精品亚洲 | 草久久久久 | 天天综合色天天综合 | 婷婷色中文 | 日韩欧美在线高清 | 狠狠干.com| 中文字幕在线看视频国产 | 亚av在线 | 久久久久久黄色 | 天天搞天天| 亚洲国产精品第一区二区 | 免费在线观看不卡av | 久久精品影片 | 亚洲成色777777在线观看影院 | 久久久久久激情 | 久草资源免费 | 日韩高清不卡一区二区三区 | 欧美性大战 | 丁香花在线观看视频在线 | 在线v| 国产精品 久久 | 一区二区三区在线观看中文字幕 | 成年人电影毛片 | 亚洲另类xxxx | www.色爱| 成人在线视频网 | 日韩最新av在线 | 不卡av在线 | 高清国产在线一区 | 久久资源在线 | 国产精品入口a级 | 日批视频在线播放 | 91精品啪在线观看国产线免费 | 日日婷婷夜日日天干 | 在线观看免费一级片 | 西西www4444大胆视频 | 精品欧美一区二区精品久久 | 亚洲国产中文字幕在线观看 | 婷久久| 中文字幕在线看片 | 精品在线二区 | 天天骚夜夜操 | 九九爱免费视频 | 亚洲最新av | www婷婷| 国产精品色婷婷视频 | 香蕉精品视频在线观看 | 香蕉视频在线看 | 成人av亚洲 | 三级黄色免费片 | 亚洲国产wwwccc36天堂 | 一二三精品视频 | 久久 在线 | 国产一二三四在线视频 | 超碰国产在线 | 一区二区网 | 亚洲国产精品99久久久久久久久 | 99久久精品国产亚洲 | 欧美成人精品欧美一级乱黄 | 欧美一区二区三区在线 | 成人黄色在线视频 | 久久久精品久久日韩一区综合 | 免费人人干 | 激情九九| 国产色女人 | 亚洲精品av中文字幕在线在线 | 99视频在线看 | 国产欧美最新羞羞视频在线观看 | 精品国产片 | 亚洲成av人影院 | 成人精品一区二区三区电影免费 | 日韩精品视频免费专区在线播放 | 日日爽视频 | www.日本色| 97成人精品 | 亚洲精品在线资源 | 日韩av在线不卡 | 狠狠躁夜夜a产精品视频 | 五月天婷婷狠狠 | av中文字幕电影 | 少妇视频在线播放 | 久久激情视频 久久 | 日韩在线视频观看 | 欧美久久影院 | 成人久久久久久久久久 | 91系列在线观看 | 欧美午夜精品久久久久久孕妇 | 欧美俄罗斯性视频 | 91成人网页版 | 中文字幕一区二区三区乱码不卡 | 午夜久久久久久久 | 色婷婷亚洲综合 | 久久婷婷网| 亚洲免费av在线播放 | 成人免费一区二区三区在线观看 | 天天射综合网视频 | 国产一区二区三区四区大秀 | 久久99久久99精品免费看小说 | 天天鲁天天干天天射 | 五月婷婷操 | 午夜久久福利 | 深爱开心激情网 | 日韩久久久久久久 | 丁香五月亚洲综合在线 | 黄网站大全 | 日本中文字幕电影在线免费观看 | 中文字幕国产 | a级国产乱理论片在线观看 特级毛片在线观看 | 婷婷色在线资源 | 精品一区91 | 久久久亚洲麻豆日韩精品一区三区 | 永久免费视频国产 | 91porny九色91啦中文 | 九九视频在线 | 成人久久久精品国产乱码一区二区 | 狠狠地日| av福利在线免费观看 | 久久综合导航 | 天天se天天cao天天干 | 免费观看9x视频网站在线观看 | 成年人免费av网站 | 日韩av女优视频 | 97视频播放| 国产高清不卡在线 | 亚洲婷婷综合色高清在线 | 午夜久久久久久久久 | 久久精品这里都是精品 | 国产69久久久欧美一级 | 日韩一区二区久久 | 精品超碰| 久久久久久久久久久久久影院 | 国产一区二区三区免费在线观看 | 国产免费成人 | 五月情婷婷 | 在线 成人| 国产999精品久久久 免费a网站 | 免费看三级黄色片 | 91精品国自产在线偷拍蜜桃 | 国产伦理一区 | 女人18精品一区二区三区 | 亚洲黄在线观看 | 免费电影一区二区三区 | 国产亚洲一区 | 欧美一区二区在线看 | 91精品国产综合久久福利 | 亚洲桃花综合 | 天天五月天色 | 欧美在线观看视频一区二区 | 免费手机黄色网址 | 91在线视频一区 | 国产a级片免费观看 | 免费看的国产视频网站 | 久久久免费观看完整版 | 午夜精品影院 | 久久综合给合久久狠狠色 | 国产一级精品绿帽视频 | 麻豆va一区二区三区久久浪 | 99操视频 | 99re国产| 久久久久久影视 | 国产成人区 | 亚洲精品在线视频播放 | aaa亚洲精品一二三区 | 国产馆在线播放 | 欧美超碰在线 | 精品久久久久久电影 | 亚洲免费不卡 | 久久99亚洲精品久久 | 久久久人| 日韩av一区在线观看 | 中文字幕在线免费看线人 | 成人影片在线免费观看 | 狠狠色丁香九九婷婷综合五月 | 狠狠躁日日躁夜夜躁av | 欧美一区二区三区激情视频 | 亚洲黄色成人 | 在线视频一二三 | 国产福利精品一区二区 | 97视频精品 | 国产在线美女 | 97人人添人澡人人爽超碰动图 | 国产在线精品一区二区三区 | 亚洲男男gaygay无套同网址 | 亚洲精品视频网站在线观看 | 国产视频一区二区在线 | www色av| 乱男乱女www7788 | 国产黄色网 | 福利电影一区二区 | 亚洲精品久久久蜜桃直播 | 操操操影院 | 色视频网页 | 美女免费视频网站 | 综合伊人久久 | 日韩伦理一区二区三区av在线 | japanesefreesexvideo高潮 | 国产91在线观 | 超碰人人av | 亚洲2019精品 | 精品国模一区二区三区 | 丁香六月在线 | 久久一视频 | 国产精品美女久久久久久久久 | 久草精品视频在线看网站免费 | av中文国产 | 久久99精品波多结衣一区 | 狠狠干成人综合网 | 国产精品专区h在线观看 | 国产视频69| 777xxx欧美| 久久精品亚洲精品国产欧美 | 国产日韩中文字幕在线 | 欧美视屏一区二区 | 国产免费嫩草影院 | 亚洲a网| 国产成人区 | 久久国产精品第一页 | 青青久视频 | 亚洲国产精品成人精品 | 国产无套视频 | 久久国产精品影视 | 一区二区国产精品 | 色婷婷狠狠五月综合天色拍 | 中文字幕一二三区 | 日韩免费高清在线观看 | 三级小视频在线观看 | 成人av一区二区兰花在线播放 | 国产高清av在线播放 | 色综合久久久久 | 久久久国产精品久久久 | 欧美黄网站 | 日韩欧美高清 | 亚洲欧洲成人精品av97 | av一区二区三区在线播放 | 伊人久久电影网 | 日韩中文字幕在线观看 | 91视频免费 | 亚洲动漫在线观看 | 久久久精品一区二区 | 经典三级一区 | 欧美精品久久久久久久久久白贞 | 国产成人在线精品 | 久久婷婷精品视频 | 亚洲在线观看av | 一区二区三区久久精品 | 中文字幕在线观看完整 | 91精品国产成 | 黄色软件网站在线观看 | av电影免费在线看 | 偷拍福利视频一区二区三区 | 狠狠色丁香婷婷综合基地 | 色五丁香| 日韩中文字幕免费在线观看 | 中文字幕4 | 天天射,天天干 | 美女精品在线 | 亚洲视频免费在线观看 | 91在线小视频 | 久久精品久久99 | 97在线播放视频 | 精品视频一区在线 | 麻豆一区二区 | 日韩精品免费在线观看视频 | 免费影视大全推荐 | 91在线小视频| 在线中文字幕av观看 | 精品久久久久一区二区国产 | 999成人免费视频 | 国产原创中文在线 | 九色视频自拍 | 国产午夜精品一区二区三区欧美 | 亚洲国产中文字幕在线观看 | 久章草在线观看 | 97国产大学生情侣酒店的特点 | 处女av在线 | www.黄色在线 | 久久99精品热在线观看 | 久久99久久99精品免观看粉嫩 | 九九免费视频 | 国产精品视频区 | 国产亚洲精品久久久久秋 | 国产精品高潮呻吟久久久久 | 在线国产一区二区 | 黄色三级在线观看 | 亚洲国产字幕 | 天天草网站 | 午夜国产一区二区 | 日韩理论片中文字幕 | 国产黄色一级片在线 | 91人人在线 | 91香蕉视频 mp4 | 欧美日韩性视频在线 | 久久久国产一区二区三区 | 欧美极品xxx| 久久久国产高清 | 天天色综合1 | 天天插夜夜操 | 91在线中文 | 黄色aa久久 | 欧美一级片免费观看 | 成人国产在线 | 美女网站在线观看 | 国产亚洲人 | 国产色综合天天综合网 | 日韩中文字幕免费看 | 国内精品久久久久久久影视简单 | 久久久69| 精品视频在线播放 | 国产亚洲在线视频 | 国产无遮挡猛进猛出免费软件 | 丁香 久久 综合 | 日日夜夜天天干 | 久久久久久久久久久高潮一区二区 | 久久这里只有精品视频99 | 久久国产美女 | 久久艹影院 | 日韩女同一区二区三区在线观看 | 亚洲桃花综合 | 日韩在线观看网站 | 最新国产一区二区三区 | 成人午夜久久 | 久草网站在线 | 国产高清 不卡 | 欧美一区中文字幕 | 日韩av在线免费看 | 四虎国产精品免费观看视频优播 | 国产精品麻豆果冻传媒在线播放 | 久草av在线播放 | 9久久精品| 久久人人爽人人爽人人 | 91精品蜜桃 | 婷婷久久一区二区三区 | www.97色.com| 91免费视频网站在线观看 | 超碰伊人网 | 91九色视频在线观看 | 久久久黄色免费网站 | 久久女同性恋中文字幕 | 综合精品在线 | 黄色片免费在线 | 97视频在线看 | 在线影院 国内精品 | 丁香av在线 | 欧美日韩一区二区久久 | 国产亚洲日本 | 亚洲精品视频大全 | 天天操天天草 | 国产成人一区二区精品非洲 | 国产精品永久在线观看 | 中文在线免费看视频 | 久久久久久片 | 激情偷乱人伦小说视频在线观看 | 国产不卡在线播放 | 四虎永久免费网站 | 久久曰视频 | 天天射综合 | 黄色激情网址 | 久色婷婷 | 超碰人人超碰 | 国产亚洲字幕 | 久久草在线精品 | 国内精品久久久久久久影视简单 | 国产精品毛片一区二区 | 深爱婷婷网| 国产精品视频久久 | 久久精品免费看 | 色欧美综合 | 成人免费在线看片 | 日韩三区在线 | 亚洲视频免费视频 | 国产亚洲在线观看 | 欧美激情一区不卡 | 成人午夜性影院 | 成人播放器 | 日韩在线观看三区 | 91免费在线播放 | 国产精品日韩在线观看 | 国产精品6999成人免费视频 | 国产99免费 | 久久成人综合 | 免费在线视频一区二区 | 99精品热视频只有精品10 | 91福利社在线观看 | 在线观看免费av网站 | 婷婷色站 | 国产日产精品一区二区三区四区 | 亚洲久久视频 | 国产一区二区久久久 | 日本在线观看视频一区 | 亚洲精品乱码久久久久久久久久 | 18av在线视频 | 国产精品一区二区在线观看免费 | 在线成人免费电影 | 国产麻豆精品传媒av国产下载 | 欧美亚洲久久 | 国产亚洲精品久久久网站好莱 | 亚洲人成人在线 | 99在线精品视频在线观看 | 91视频在线观看大全 | 中文字幕二区三区 | 黄色app网站在线观看 | 99久久精品免费看国产免费软件 | 亚洲综合欧美激情 | 欧美资源在线观看 | 国产精品麻豆视频 | 精品一区二区视频 | 人人干,人人爽 | 操老逼免费视频 | 国产精品久久久久久久久蜜臀 | 免费情趣视频 | 日本天天色 | 国产亚洲精品女人久久久久久 | 欧美日韩在线精品 | 又黄又刺激视频 | 不卡av在线 | 九九免费在线观看 | 欧美激情综合五月色丁香 | 亚洲精品一区二区18漫画 | 欧美韩国在线 | 日韩电影一区二区在线观看 | 欧美一级性生活视频 | 国产在线观看xxx | 国产精品久久久久一区二区 | 亚洲成人软件 | 91在线porny国产在线看 | 91久久国产露脸精品国产闺蜜 | 免费视频成人 | 91在线免费观看国产 | 丁香婷婷综合色啪 | 六月丁香激情网 | 久草在线资源网 | 国产精品免费观看网站 | 高清精品久久 | 亚洲久草视频 | 91麻豆视频 | av黄色在线| 少妇性xxx | 亚洲第一区在线观看 | 亚洲免费激情 | 日韩中文字幕电影 | 在线观看一区二区视频 | 色婷婷国产精品一区在线观看 | 欧美有色 | 香蕉视频久久 | 欧美日韩国产网站 | 天天操天天操 | 成人av直播 | 九九99靖品 | 黄色三级久久 | 国产精品乱码一区二区视频 | 日韩国产欧美在线播放 | 最新日韩视频 | 成人看片 | 日日夜夜精品视频天天综合网 | 一本一道久久a久久精品 | 在线免费色 | 欧美伦理一区 | 成人欧美一区二区三区黑人麻豆 | 国产精品久久久久久久久久东京 | 免费日p视频 | 97色视频在线 | 精品二区视频 | 久久av伊人 | 黄色的片子 | 日日干夜夜骑 | 精品成人网 | 久久久久国产精品厨房 | 六月激情久久 | 91高清免费在线观看 | 91看片在线免费观看 | 国产成人久久久久 | 国产日韩欧美综合在线 | 免费久久久 | 久久草视频 | 玖玖视频在线 | 国产中文字幕视频在线观看 | 色吊丝在线永久观看最新版本 | 国产精品久久伊人 | 五月天亚洲综合小说网 | 日日噜噜噜噜夜夜爽亚洲精品 | 国产精品久久久久久久久大全 | 免费一区在线 | 亚洲欧美国内爽妇网 | 日韩h在线观看 | 日韩精品免费一区二区三区 | 国产精品自拍av | 射久久 | 久久久久综合 | av成人免费网站 | 五月婷婷导航 | 九九国产精品视频 | 色综合久久综合中文综合网 | 国产精品久久久久久久久久ktv | 欧美视频在线二区 | 激情欧美在线观看 | 久久一区国产 | 婷婷色中文 | 成人在线免费视频 | 国产精品久久久久久久av大片 | 成人污视频在线观看 | 国产手机精品视频 | 欧美国产一区在线 | 久久夜色精品国产欧美乱极品 | 综合网色 | 久久精品免视看 | 片网站| 西西www444| 激情五月婷婷综合网 | 国产专区精品视频 | 国产不卡av在线播放 | 免费在线观看午夜视频 | 国产精品原创在线 | 深夜国产福利 | 日韩中文字幕免费在线观看 | 国产999精品视频 | 操久在线 | www.亚洲黄 | 美女网站免费福利视频 | 日韩精品国产一区 | 91爱在线| 国产美女久久 | 天天曰夜夜爽 | 国产在线更新 | 精品自拍sae8—视频 | 久久久综合色 | 亚洲香蕉在线观看 | 国产 日韩 在线 亚洲 字幕 中文 | 国产成人精品av在线观 | 亚洲一区美女视频在线观看免费 | 久久99精品国产99久久6尤 | 成人黄色电影免费观看 | 日韩精品免费在线观看 | 国产在线观看91 | 中文av字幕在线观看 | 亚洲视频一区二区三区在线观看 | 在线视频一区二区 | 欧美在一区 | 女人18精品一区二区三区 | 国产成人在线观看 | 欧美先锋影音 | 国产一区二区三区免费在线 | 免费在线精品视频 | 中文字幕一区二区三 | 免费日韩av电影 | 干天天 | 日韩不卡高清视频 | 色综合久久五月 | 香蕉影院在线 | 精品国产伦一区二区三区观看说明 | 视频一区在线播放 | www.xxx.性狂虐| 亚洲电影av在线 | 一级黄视频 | 成人午夜剧场在线观看 | 精品久久综合 | 欧美日韩国产高清视频 | 精品国产一区二区三区久久久蜜臀 | 日日夜夜综合网 | 亚洲女在线 | 成年人免费电影在线观看 | 欧美激情综合色综合啪啪五月 | 色视频在线观看免费 | 在线观看精品视频 | 欧美日本高清视频 | 在线观看视频色 | 97视频人人澡人人爽 | 永久免费毛片 | 日韩av成人免费看 | a国产精品| 999国产在线 | www91在线| 久久精品国产成人精品 | 国产精品成人一区二区三区吃奶 | 免费精品视频在线观看 | 日韩免费播放 | 欧美韩日精品 | 久久久亚洲麻豆日韩精品一区三区 | 日韩精品免费 | a久久免费视频 | 少妇bbw搡bbbb搡bbb| 久久综合中文色婷婷 | 欧美日韩高清一区 | 午夜影院在线观看18 | 欧美日韩高清一区二区 国产亚洲免费看 | 久久免费视频在线 | 丝袜制服综合网 | 91精品视频观看 | 色妞色视频一区二区三区四区 | 日本xxxxav| 亚洲天堂毛片 | 久久综合九色综合欧美就去吻 | 欧美日韩国产精品一区 | 综合铜03 | 日韩动漫免费观看高清完整版在线观看 | 色香蕉网| www.色午夜 | 亚洲精品国产综合久久 | 麻豆视传媒官网免费观看 | 中文字幕在线观看视频一区二区三区 | 五月亚洲 | 亚洲精选在线 | 日韩动态视频 | 最新av网址在线观看 | 97视频人人免费看 | 男女视频久久久 | 一性一交视频 | 蜜桃视频成人在线观看 | 欧美日韩后| 97视频免费在线观看 | 亚洲国产av精品毛片鲁大师 | av免费片| 久久久黄色av | 欧美夫妻生活视频 | av综合在线观看 | 亚洲综合小说 | 91在线看黄| 久久精品国产久精国产 | 夜夜婷婷 | 欧美日韩精品免费观看 | 在线免费色 | 国产午夜精品一区二区三区在线观看 | 精品毛片久久久久久 | 天天干夜夜干 | 国产99视频在线观看 | 精品久久久网 | 毛片基地黄久久久久久天堂 | 在线免费观看黄色大片 | 国产又粗又猛又黄视频 | 麻豆 free xxxx movies hd | a√资源在线 | 中文字幕乱在线伦视频中文字幕乱码在线 | 免费观看第二部31集 | 久久国语露脸国产精品电影 | 欧美性生活大片 | 国产一区在线观看免费 | 99在线视频免费观看 | 天天插天天干天天操 | 久久久久久免费视频 | 国内偷拍精品视频 | 日韩av免费一区二区 | 精品免费视频. | 国产剧情一区二区 | av黄免费看 | 亚洲一片黄 | 日本三级人妇 | 91视频在线免费下载 | 免费在线观看av | 久久国产午夜精品理论片最新版本 | 久久成人国产精品一区二区 | 色综合激情久久 | 91精品久久久久久 | 国内精品视频免费 | 免费看国产视频 | 五月天综合网 | 免费人成在线观看网站 | 玖玖在线播放 | 亚洲www天堂com | 国产精品第52页 | 亚洲成人家庭影院 | 99久久99久久| a视频免费在线观看 | 日日操天天操狠狠操 | 色婷婷综合久久久中文字幕 | 欧美国产日韩在线观看 | 日韩欧美在线一区 | 国产 欧美 日产久久 | 超碰在线97观看 | 色偷偷网站视频 | 国产高清在线免费 | 91在线免费看片 | av网址在线播放 | 免费成人在线网站 | 丁香五月网久久综合 | 精品视频资源站 | 九九九电影免费看 | 91麻豆文化传媒在线观看 | 亚洲 欧美 国产 va在线影院 | 玖玖精品在线 | 五月开心激情 | 97电影在线| 国产91全国探花系列在线播放 | 日韩激情网 | 99综合电影在线视频 | 摸bbb搡bbb搡bbbb | 亚洲午夜精品久久久久久久久 | 午夜av大片 | 国产亚洲精品久久久久久久久久久久 | 91人人在线 | 午夜丁香视频在线观看 | 亚洲午夜精品在线观看 | 伊人宗合网 | 国产亚洲精品久久久久久久久久 | 日韩av免费观看网站 | 欧美日韩一区二区免费在线观看 | 久久高清av | 久久毛片高清国产 | 婷婷色网站 | 国产在线小视频 | 国产免费xvideos视频入口 | 中文字幕av一区二区三区四区 | 亚洲 欧洲av | 国产精品自产拍在线观看桃花 | 国产69精品久久app免费版 | 免费人成在线观看网站 | 日韩亚洲国产精品 | 亚洲国产精品一区二区尤物区 | 超碰日韩| 久久久精品| 综合久久综合久久 | 亚洲激情六月 | 91视频中文字幕 | 成人亚洲综合 | 人成在线免费视频 | 精品一二| 成人影音在线 | 69久久久久久久 | 99国产精品 | 日韩欧美久久 | 激情欧美一区二区免费视频 | 有码中文字幕在线观看 | 欧美一区二区三区特黄 | 免费观看国产精品 | 亚洲综合色丁香婷婷六月图片 | 丁香婷婷综合五月 | 人人澡人 | 黄色a大片 | 在线视频精品播放 | 久久狠狠一本精品综合网 | 日韩av高清在线观看 | 手机看片99 | 九九视频免费观看视频精品 | 午夜美女影院 | 久久精品五月 | 国产剧在线观看片 | 免费av电影网站 | 在线午夜| 中国一级特黄毛片大片久久 | 麻豆传媒视频在线免费观看 | 亚洲国产免费网站 | 97av在线视频 | 国产精品久久网 | 黄色www | 国产成人三级在线观看 | 国产理论影院 | 欧美国产亚洲精品久久久8v | 99精品视频免费观看视频 | 中文字幕在线观看网址 | 国产二级视频 | 在线免费观看麻豆视频 | 日韩精品视频在线免费观看 | 亚洲激情六月 | 国产精品免费视频一区二区 | 97在线观看免费视频 | 国产精品三级视频 | 黄色毛片电影 | 99久久日韩精品免费热麻豆美女 | 四虎国产精品免费观看视频优播 | 久久官网 | 色欧美88888久久久久久影院 | 欧美男男激情videos | 国产黄免费在线观看 | 91精彩在线视频 | 成人小视频在线观看免费 | 日韩在线观看你懂的 | 91正在播放 | 亚洲欧美综合精品久久成人 | 在线免费观看麻豆 | 97成人在线观看视频 | 黄色国产精品 | 国产精品久久久久永久免费 | 国产精品亚洲片在线播放 | 日本黄色a级大片 | 99riav1国产精品视频 | 日日干精品 | 91视频免费网址 | 公开超碰在线 | 国产精品久久一 | 日韩在线大片 | 99在线观看免费视频精品观看 | 国产在线a视频 | 免费av片在线 | 天天干夜夜爱 | 国产系列精品av | 国产精品一区二区在线观看 | 成人亚洲综合 | 欧美日韩网站 | 91黄视频在线观看 | 玖玖视频网 | 色妞色视频一区二区三区四区 | 国产男女爽爽爽免费视频 | 国产va饥渴难耐女保洁员在线观看 | 超碰在线色 | 日日干影院 | 色偷偷888欧美精品久久久 | 亚洲国产经典视频 | 久久久久免费电影 | 日韩欧美视频一区二区 | 一区 二区 精品 | 国产精品不卡 | 免费看国产一级片 | 久久国产精品久久w女人spa | 97精品国产一二三产区 | 在线国产中文字幕 | 视频在线观看入口黄最新永久免费国产 | 成人av电影在线 | 天天操比 | 99久久日韩精品视频免费在线观看 | 狠狠婷婷| 日韩免费精品 | 欧美激情视频一二三区 | 亚洲成年人在线播放 | 日韩av手机在线看 | 久久婷婷久久 | 国产区精品视频 | 在线播放 一区 | 深夜免费网站 | 九九九视频在线 | 国产伦理剧 | 一区二区在线影院 | 看黄色.com| 日韩在线免费高清视频 | www·22com天天操 | 天天干天天拍天天操 | 精品国产伦一区二区三区免费 | 国产色女人 | 午夜精品一区二区国产 | 免费v片| 国产小视频在线免费观看视频 | 成人免费毛片aaaaaa片 | 在线观看一级 | 亚洲黄色影院 | 日韩av播放在线 | 一区二区三区中文字幕在线观看 | 欧美性生活免费看 | 国产视频999| 亚洲精品国偷拍自产在线观看蜜桃 | 精品女同一区二区三区在线观看 | 成人免费视频播放 | 在线免费日韩 | 狠狠色噜噜狠狠 | 黄色免费大片 | 久久久亚洲成人 | 精品国产一区二区久久 | av免费在线看网站 | 日韩精品久久久免费观看夜色 | 国产一区在线不卡 | 中文字幕在线观看一区二区 | 色九九在线 | 欧美精品久久久久久久久久 | 国产亚洲免费的视频看 | 国产中文字幕在线 | 成年人在线电影 | 9在线观看免费高清完整版在线观看明 | 怡红院av久久久久久久 | 五月婷婷六月丁香激情 | 日韩二区三区 | 女人18毛片90分钟 | 狠狠色网 | 91精品国产乱码久久桃 | 久久国产精品成人免费浪潮 | 国产一区二区在线播放 | 亚洲精品玖玖玖av在线看 | 国产黄色一级片在线 | 日韩中文在线电影 | 正在播放国产精品 | 精品国产一区在线观看 | 久久久久久久久久久免费 | 天天夜夜狠狠操 | 精品一区二区三区久久久 | 国产中文字幕在线免费观看 | 国产在线日韩 | 精品视频免费 | 久久精品国产亚洲a | 国内精品久久久 | 中文字幕专区高清在线观看 | av资源免费在线观看 | 色久av | 久久视频精品 | 超碰免费在线公开 | 国产精品一区二区三区在线播放 | 国产欧美综合视频 | 999视频在线播放 | 91超级碰碰 | 国产99久久久国产精品成人免费 | 一二区av| 日本久久综合网 | 久久国产麻豆 | 国产九九热视频 | 国产高清在线一区 | 91亚洲综合| 99久久久免费视频 | 中文字幕视频免费观看 | 天天操天天谢 | 99久久精品免费看国产免费软件 | 国产成人久久av免费高清密臂 | 久福利 | 久久官网 | 97精品国产手机 | 国产一区二区三区免费在线 | 99精品国产福利在线观看免费 | 四虎在线观看精品视频 | 99精品成人| 国产精品18久久久久vr手机版特色 | 2020天天干夜夜爽 | 欧美成人黄 | 在线视频专区 | 色在线最新 | 在线观看亚洲精品视频 | 人人狠| 久免费视频 | 97超碰人人澡人人爱学生 | 国产精品久久久久久久久久妇女 | 狠狠综合 | 亚洲天堂毛片 | 97人人澡人人添人人爽超碰 | 久久久久久久久免费 | 黄色免费国产 | 超碰97国产在线 | 色婷婷狠狠18 | 在线综合色 | 成人午夜免费福利 | 国产高清av免费在线观看 | 天天草天天色 | 91经典在线 | 91.麻豆视频| 国产精品麻豆免费版 | 亚洲一区视频在线播放 | 欧美一级大片在线观看 | 久久视频网 | 欧美a级在线播放 | 久草com| 日韩欧美一区二区三区视频 | 欧美视频在线观看免费网址 | 又污又黄的网站 | 亚州中文av | 国产黄a三级三级三级三级三级 | 天天玩夜夜操 | 97超碰人人澡人人 | 日韩免费一级a毛片在线播放一级 | 国产精品久久久久久久久久久久午 | 黄色小说免费观看 | 精品国产免费看 | 97电院网手机版 | 久久99国产精品二区护士 | 亚洲欧美激情精品一区二区 | 丁香激情综合久久伊人久久 | 中文字幕在线影院 | 在线免费试看 | 99精品欧美一区二区三区黑人哦 | 国产日韩精品一区二区在线观看播放 | 久久九九久久 | 五月天久久婷 | 亚洲无吗av| 国产精品资源在线 | 国产一区在线免费观看 | 免费黄色在线网址 | 中文字幕人成乱码在线观看 | 在线精品亚洲一区二区 | 91人人网| 日本中文字幕视频 | 久要激情网 | 午夜狠狠干| 精品主播网红福利资源观看 | 日日夜夜精品免费 | 92国产精品久久久久首页 | av高清影院 | 国产小视频在线看 | 亚洲五月婷婷 | 激情电影影院 | 黄色免费在线看 | 精品久久久久久久久亚洲 | 在线小视频你懂得 | 欧美精品免费在线 | 日韩精品一区二区在线观看 | 国产美女视频免费观看的网站 |