4.9.2. Web 集成测试

最低版本7.1

Web 集成测试运行在 Web 客户端 block 的 Spring 容器中。测试容器独立于中间件工作,因为框架会自动为所有中间件服务创建桩代码。测试基础设施由 com.haulmont.cuba.web.testsupport 及其内部包的下列类组成:

  • TestContainer - Spring 容器的包装器,用来作为项目特定容器的基类。

  • TestServiceProxy - 为中间件服务提供默认的桩代码。该类可以用来注册为特定用例 mock 的服务,参考其 mock() 静态方法。

  • DataServiceProxy - DataManager 的默认桩代码。其包含一个 commit() 方法的实现,能模拟真正的数据存储的行为:能让新实体 detach,增加实体版本,等等。加载方法返回 null 和空集合。

  • TestUiEnvironment - 提供一组方法用来配置和获取 TestContainer。该类的实例在测试中需要作为 JUnit 5 的扩展来使用。

  • TestEntityFactory - 测试中为方便创建实体实例的工厂。可以通过 TestContainer 获取工厂。

尽管框架为服务提供了默认桩代码,但是在测试中也许需要自己创建服务的 mock。要创建 mock,可以使用任何 mocking 框架,通过添加其为依赖即可,如上节所说。服务的 mock 均使用 TestServiceProxy.mock() 方法注册。

Web 集成测试容器示例

web 模块创建 test 目录。然后在 test 目录合适的包内创建项目的测试容器类:

  1. package com.company.demo;
  2. import com.haulmont.cuba.web.testsupport.TestContainer;
  3. import java.util.Arrays;
  4. public class DemoWebTestContainer extends TestContainer {
  5. public DemoWebTestContainer() {
  6. appComponents = Arrays.asList(
  7. "com.haulmont.cuba"
  8. // add CUBA add-ons and custom app components here
  9. );
  10. appPropertiesFiles = Arrays.asList(
  11. // List the files defined in your web.xml
  12. // in appPropertiesConfig context parameter of the web module
  13. "com/company/demo/web-app.properties",
  14. // Add this file which is located in CUBA and defines some properties
  15. // specifically for test environment. You can replace it with your own
  16. // or add another one in the end.
  17. "com/haulmont/cuba/web/testsupport/test-web-app.properties"
  18. );
  19. }
  20. public static class Common extends DemoWebTestContainer {
  21. // A common singleton instance of the test container which is initialized once for all tests
  22. public static final DemoWebTestContainer.Common INSTANCE = new DemoWebTestContainer.Common();
  23. private static volatile boolean initialized;
  24. private Common() {
  25. }
  26. @Override
  27. public void before() throws Throwable {
  28. if (!initialized) {
  29. super.before();
  30. initialized = true;
  31. }
  32. setupContext();
  33. }
  34. @Override
  35. public void after() {
  36. cleanupContext();
  37. // never stops - do not call super
  38. }
  39. }
  40. }

UI 界面测试示例

下面是 Web 集成测试的示例,在一些用户操作之后检查了编辑实体的状态。

  1. package com.company.demo.customer;
  2. import com.company.demo.DemoWebTestContainer;
  3. import com.company.demo.entity.Customer;
  4. import com.company.demo.web.screens.customer.CustomerEdit;
  5. import com.haulmont.cuba.gui.Screens;
  6. import com.haulmont.cuba.gui.components.Button;
  7. import com.haulmont.cuba.gui.screen.OpenMode;
  8. import com.haulmont.cuba.web.app.main.MainScreen;
  9. import com.haulmont.cuba.web.testsupport.TestEntityFactory;
  10. import com.haulmont.cuba.web.testsupport.TestEntityState;
  11. import com.haulmont.cuba.web.testsupport.TestUiEnvironment;
  12. import org.junit.jupiter.api.BeforeEach;
  13. import org.junit.jupiter.api.Test;
  14. import org.junit.jupiter.api.extension.RegisterExtension;
  15. import java.util.Collections;
  16. import static org.junit.jupiter.api.Assertions.assertEquals;
  17. import static org.junit.jupiter.api.Assertions.assertNull;
  18. public class CustomerEditInteractionTest {
  19. @RegisterExtension
  20. TestUiEnvironment environment =
  21. new TestUiEnvironment(DemoWebTestContainer.Common.INSTANCE).withUserLogin("admin"); (1)
  22. private Customer customer;
  23. @BeforeEach
  24. public void setUp() throws Exception {
  25. TestEntityFactory<Customer> customersFactory =
  26. environment.getContainer().getEntityFactory(Customer.class, TestEntityState.NEW);
  27. customer = customersFactory.create(Collections.emptyMap()); (2)
  28. }
  29. @Test
  30. public void testGenerateName() {
  31. Screens screens = environment.getScreens(); (3)
  32. screens.create(MainScreen.class, OpenMode.ROOT).show(); (4)
  33. CustomerEdit customerEdit = screens.create(CustomerEdit.class); (5)
  34. customerEdit.setEntityToEdit(customer);
  35. customerEdit.show();
  36. assertNull(customerEdit.getEditedEntity().getName());
  37. Button generateBtn = (Button) customerEdit.getWindow().getComponent("generateBtn"); (6)
  38. customerEdit.onGenerateBtnClick(new Button.ClickEvent(generateBtn)); (7)
  39. assertEquals("Generated name", customerEdit.getEditedEntity().getName());
  40. }
  41. }
1- 定义带共享容器和带有 admin 的用户会话存根的测试环境。
2- 创建 new 状态的实体实例。
3- 从环境获取 Screens 基础设施对象。
4- 打开主界面,打开应用程序界面必须的步骤。
5- 创建、初始化并打开实体编辑界面。
6- 获取 Button 组件。
7- 创建一个点击事件,并以调用控制器方法的方式响应点击操作。

测试在界面加载数据的示例

下面是一个 web 集成测试的示例,检查加载数据的正确性。

  1. package com.company.demo.customer;
  2. import com.company.demo.DemoWebTestContainer;
  3. import com.company.demo.entity.Customer;
  4. import com.company.demo.web.screens.customer.CustomerEdit;
  5. import com.haulmont.cuba.core.app.DataService;
  6. import com.haulmont.cuba.core.entity.Entity;
  7. import com.haulmont.cuba.core.global.LoadContext;
  8. import com.haulmont.cuba.gui.Screens;
  9. import com.haulmont.cuba.gui.model.InstanceContainer;
  10. import com.haulmont.cuba.gui.screen.OpenMode;
  11. import com.haulmont.cuba.gui.screen.UiControllerUtils;
  12. import com.haulmont.cuba.web.app.main.MainScreen;
  13. import com.haulmont.cuba.web.testsupport.TestEntityFactory;
  14. import com.haulmont.cuba.web.testsupport.TestEntityState;
  15. import com.haulmont.cuba.web.testsupport.TestUiEnvironment;
  16. import com.haulmont.cuba.web.testsupport.proxy.TestServiceProxy;
  17. import mockit.Delegate;
  18. import mockit.Expectations;
  19. import mockit.Mocked;
  20. import org.junit.jupiter.api.AfterEach;
  21. import org.junit.jupiter.api.BeforeEach;
  22. import org.junit.jupiter.api.Test;
  23. import org.junit.jupiter.api.extension.RegisterExtension;
  24. import static org.junit.jupiter.api.Assertions.assertEquals;
  25. public class CustomerEditLoadDataTest {
  26. @RegisterExtension
  27. TestUiEnvironment environment =
  28. new TestUiEnvironment(DemoWebTestContainer.Common.INSTANCE).withUserLogin("admin"); (1)
  29. @Mocked
  30. private DataService dataService; (1)
  31. private Customer customer;
  32. @BeforeEach
  33. public void setUp() throws Exception {
  34. new Expectations() {{ (2)
  35. dataService.load((LoadContext<? extends Entity>) any);
  36. result = new Delegate() {
  37. Entity load(LoadContext lc) {
  38. if ("demo_Customer".equals(lc.getEntityMetaClass())) {
  39. return customer;
  40. } else
  41. return null;
  42. }
  43. };
  44. }};
  45. TestServiceProxy.mock(DataService.class, dataService); (3)
  46. TestEntityFactory<Customer> customersFactory =
  47. environment.getContainer().getEntityFactory(Customer.class, TestEntityState.DETACHED);
  48. customer = customersFactory.create(
  49. "name", "Homer", "email", "homer@simpson.com"); (4)
  50. }
  51. @AfterEach
  52. public void tearDown() throws Exception {
  53. TestServiceProxy.clear(); (5)
  54. }
  55. @Test
  56. public void testLoadData() {
  57. Screens screens = environment.getScreens();
  58. screens.create(MainScreen.class, OpenMode.ROOT).show();
  59. CustomerEdit customerEdit = screens.create(CustomerEdit.class);
  60. customerEdit.setEntityToEdit(customer);
  61. customerEdit.show();
  62. InstanceContainer customerDc = UiControllerUtils.getScreenData(customerEdit).getContainer("customerDc"); (6)
  63. assertEquals(customer, customerDc.getItem());
  64. }
  65. }
1- 使用 JMockit framework 定义数据服务 mock。
2- 定义 mock 行为。
3- 注册 mock。
4- 创建 detached 状态的实体实例。
5- 测试完成后移除 mock。
6- 获取数据容器。