How to Write Unit Test Code

Junit5 + mockito + Jacobo + H2 local database

Idea enhancement plugin

  • JUnitGenerator V2. 0 standard module for generating test cases
  • Create the allnewset object and set the default value for allnewset
  • The association mapping between mybatisx DAO and mapper is easy to view
  1. ########################################################################################
  2. ##
  3. ## Available variables:
  4. ## $entryList.methodList - List of method composites
  5. ## $entryList.privateMethodList - List of private method composites
  6. ## $entryList.fieldList - ArrayList of class scope field names
  7. ## $entryList.className - class name
  8. ## $entryList.packageName - package name
  9. ## $today - Todays date in MM/dd/yyyy format
  10. ##
  11. ## MethodComposite variables:
  12. ## $method.name - Method Name
  13. ## $method.signature - Full method signature in String form
  14. ## $method.reflectionCode - list of strings representing commented out reflection code to access method (Private Methods)
  15. ## $method.paramNames - List of Strings representing the method's parameters' names
  16. ## $method.paramClasses - List of Strings representing the method's parameters' classes
  17. ##
  18. ## You can configure the output class name using "testClass" variable below.
  19. ## Here are some examples:
  20. ## Test${entry.ClassName} - will produce TestSomeClass
  21. ## ${entry.className}Test - will produce SomeClassTest
  22. ##
  23. ########################################################################################
  24. ##
  25. ## title case
  26. #macro (cap $strIn)$strIn.valueOf($strIn.charAt(0)).toUpperCase()$strIn.substring(1)#end
  27. ## Initial lowercase custom down
  28. #macro (down $strIn)$strIn.valueOf($strIn.charAt(0)).toLowerCase()$strIn.substring(1)#end
  29. ## Iterate through the list and generate testcase for every entry.
  30. #foreach ($entry in $entryList)
  31. #set( $testClass="${entry.className}Test")
  32. ##
  33. /*
  34. * Licensed to the Apache Software Foundation (ASF) under one or more
  35. * contributor license agreements. See the NOTICE file distributed with
  36. * this work for additional information regarding copyright ownership.
  37. * The ASF licenses this file to You under the Apache License, Version 2.0
  38. * (the "License"); you may not use this file except in compliance with
  39. * the License. You may obtain a copy of the License at
  40. *
  41. * http://www.apache.org/licenses/LICENSE-2.0
  42. *
  43. * Unless required by applicable law or agreed to in writing, software
  44. * distributed under the License is distributed on an "AS IS" BASIS,
  45. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  46. * See the License for the specific language governing permissions and
  47. * limitations under the License.
  48. */
  49. package $entry.packageName;
  50. import org.junit.jupiter.api.AfterEach;
  51. import org.junit.jupiter.api.BeforeEach;
  52. import org.junit.jupiter.api.DisplayName;
  53. import org.junit.jupiter.api.Test;
  54. import org.springframework.beans.factory.annotation.Autowired;
  55. /**
  56. * ${entry.className} Tester
  57. */
  58. public class $testClass {
  59. @Autowired
  60. private ${entry.className} #down(${entry.className});
  61. @BeforeEach
  62. @DisplayName("Each unit test method is executed once before execution")
  63. public void before() throws Exception {
  64. }
  65. @AfterEach
  66. @DisplayName("Each unit test method is executed once before execution")
  67. public void after() throws Exception {
  68. }
  69. #foreach($method in $entry.methodList)
  70. @Test
  71. @DisplayName("Method description: ...")
  72. public void test#cap(${method.name})() throws Exception {
  73. //TODO: Test goes here...
  74. }
  75. #end
  76. #foreach($method in $entry.privateMethodList)
  77. @Test
  78. @DisplayName("Method description: ...")
  79. public void test#cap(${method.name})() throws Exception {
  80. //TODO: Test goes here...
  81. #foreach($string in $method.reflectionCode)
  82. $string
  83. #end
  84. }
  85. #end
  86. }
  87. #end

test-0

  1. Configure test class generation path

    Original configuration: ${sourcepath}/test/${package}/${filename}
    Modified configuration: ${sourcepath}/..//test/java/${PACKAGE}/${FILENAME}

    As shown in the figure:
    test-1

  2. Select class -> right click -> generate -> JUnit test to generate a test class

    test-2

    1. Unit test code directory It must be written in the following project directory: src/test/java. It is not allowed to write in the business code directory.
      Note: this directory will be skipped during source code compilation, while the unit test framework scans this directory by default. The test configuration file must be placed under the src/test/resources file
    1. The package name of the test class should be consistent with the package name of the tested class
      Example:
      Business class: src/main/java/org/apache/linkis/jobhistory/dao/JobDetailMapper.java
      Corresponding test class:src/main/java/org/apache/linkis/jobhistory/dao/JobDetailMapperTest java
    1. Naming and definition specification of test class: use test as the suffix of class name
      The test class is named as follows:
      Tested business + test, tested interface + test, tested class + test
    1. Specification for naming and defining test cases: use test as the prefix of method names The naming rule of test cases is: test + method name. Avoid using names that have no meaning in test1 and test2. Secondly, necessary function and method annotations are required.
    1. System is not allowed to be used in unit test Out for human flesh verification, or if judgment for verification (log can be used for Key log output). Assertion assert must be used for verification.
    1. Maintain the independence of unit testing. In order to ensure that unit tests are stable, reliable and easy to maintain, unit test cases must not call each other or rely on the order of execution. Counterexample: method2 needs to rely on the execution of method1 and take the execution result as the input of method2
    1. Unit tests must be repeatable and not affected by the external environment.
      Note: unit tests are usually put into continuous integration. Unit tests will be executed every time there is code check in. If the single test depends on the external environment (network, service, middleware, etc.), it is easy to lead to the unavailability of the continuous integration mechanism.
      Positive example: in order not to be affected by the external environment, it is required to change the relevant dependencies of the tested class into injection when designing the code, and inject a local (memory) implementation or mock implementation with a dependency injection framework such as spring during testing.
    1. Incremental code ensures that the unit test passes. Note: the new code must supplement the unit test. If the new code affects the original unit test, please correct it
    1. For unit testing, it is necessary to ensure that the test granularity is small enough to help accurately locate the problem. Single test granularity is generally at the method level (very few scenarios such as tool classes or enumeration classes can be at the class level).
      Note: only with small test granularity can we locate the error location as soon as possible. Single test is not responsible for checking cross class or cross system interaction logic, which is the field of integration testing.
  1. The result verification of all test cases must use the assertion pattern
  2. use Assertions.assertEquals
  3. Assertions.assertEquals(expectedJobDetail, actualJobDetail)
  4. The assertions assertion of junit5 is preferred, and the assertions of assertij are allowed in very few scenarios
  5. Comparison of objects before/after updating common scene databases
  6. Asserting the usingrecursive comparison pattern using assertj's assertThat
  7. Assertions.assertThat(actualObject).usingRecursiveComparison().isEqualTo(expectedObject);
Methoddescriptionremarks
Assertequalsjudge whether two objects or two original types are equal
Assertnotequalsjudge whether two objects or two original types are not equal
Asserttruejudge whether the given Boolean value is true
Assertfalsejudge whether the given Boolean value is false
AssertNulljudge whether the given object reference is null
AssertNotNulljudge whether the given object reference is not null
Assert allmultiple judgment logics are processed together. As long as one error is reported, the overall test will fail

Composite assertion The assertall method can process multiple judgment logics together. As long as one error is reported, the overall test will fail:

  1. @Test
  2. @DisplayName("assert all")
  3. public void all() {
  4. //Multiple judgments are executed together. Only when all judgments are passed can they be considered as passed
  5. assertAll("Math",
  6. () -> assertEquals(2, 1 + 1),
  7. () -> assertTrue(1 > 0)
  8. );
  9. }

Exception assertion

Assertions. The assertthrows method is used to test whether the executable instance throws an exception of the specified type when executing the execute method;
If the execute method does not throw an exception during execution, or the exception thrown is inconsistent with the expected type, the test will fail;
Example:

  1. @Test
  2. @DisplayName("Assertion of exception")
  3. void exceptionTesting() {
  4. // When the execute method is executed, if an exception is thrown and the type of the exception is the first parameter of assertthrows (here is arithmeticexception. Class)
  5. // The return value is an instance of an exception
  6. Exception exception = assertThrows(ArithmeticException.class, () -> Math.floorDiv(1,0));
  7. log.info("assertThrows pass,return instance:{}", exception.getMessage());
  8. }

Object instance equality assertion

  1. Is it the same object instance
  1. Use junitd's assertions assertEquals
  2. Assertions.assertEquals(expectedJobDetail, actualJobDetail)

Not the same instance, but whether the attribute values of the comparison instance are exactly equal
AssertJ

  1. Comparison of objects before/after updating common scene databases
  2. Asserting the usingrecursive comparison pattern using assertj's assertthat
  3. Assertions. assertThat(actualObject). usingRecursiveComparison(). isEqualTo(expectedObject);
  1. Assertion of set results such as list The size of the result set needs to be asserted Scope or specific size Each object in the result set needs assertion, which is recommended to be used in combination with the predicate of stream mode Example:
  1. ArrayList<JobRespProtocol> jobRespProtocolArrayList=service. batchChange(jobDetailReqBatchUpdate);
  2. //List is matched with the predicate of stream for assertion judgment
  3. Predicate<JobRespProtocol> statusPrecate = e -> e.getStatus()==0;
  4. assertEquals(2, jobRespProtocolArrayList.size());
  5. assertTrue(jobRespProtocolArrayList.stream(). anyMatch(statusPrecate));

Sometimes we just test some apis or service modules, where the service or dao returns null values for some methods by default, but if the logic includes the judgment or secondary value of the returned null object, it is to throw some exceptions

Example:

  1. PageInfo<UDFAddVo> pageInfo =
  2. udfService.getManagerPages(udfName, udfTypes, userName, curPage, pageSize);
  3. message = Message.ok();
  4. // The pageInfo here is null, and subsequent get methods will have exceptions
  5. message.data("infoList", pageInfo.getList());
  6. message.data("totalPage", pageInfo.getPages());
  7. message.data("total", pageInfo.getTotal());

Example of mock simulation data:

  1. PageInfo<UDFAddVo> pageInfo = new PageInfo<>();
  2. pageInfo.setList(new ArrayList<>());
  3. pageInfo.setPages(10);
  4. pageInfo.setTotal(100);
  5. // For udfService.getManagerPages method passes parameters arbitrarily, and the simulation returns the pageInfo object
  6. // With this simulation data, the above example will not have exceptions when executing the get method
  7. Mockito.when(
  8. udfService.getManagerPages(
  9. Mockito.anyString(),
  10. Mockito.anyCollection(),
  11. Mockito.anyString(),
  12. Mockito.anyInt(),
  13. Mockito.anyInt()))
  14. .thenReturn(pageInfo);

It can be roughly classified according to the major functions of the class

-The controller of the HTTP service provided by the controller cooperates with mockmvc for unit testing -Service layer of service business logic code -Dao and Dao layer of database operation -Util tool function class is a common function tool -Exception class is a custom exception class -Enum class -Entity class is used for DB interaction and parameter VO object and other entity classes processed by methods (if there are other user-defined functions besides normal get set, unit test is required)

Using mockmvc

It mainly verifies the requestmethod method of interface request, basic parameters and expected return results.
Main scenarios: scenarios with and without unnecessary parameters are abnormal

  1. @Test
  2. public void testList() throws Exception {
  3. //Bring unnecessary parameters
  4. MultiValueMap<String, String> paramsMap = new LinkedMultiValueMap<>();
  5. paramsMap.add("startDate", String.valueOf(System.currentTimeMillis()));
  6. MvcResult mvcResult = mockMvc.perform(get("/jobhistory/list")
  7. .params(paramsMap))
  8. .andExpect(status().isOk())
  9. .andExpect(content().contentType(MediaType.APPLICATION_JSON))
  10. .andReturn();
  11. Message res = JsonUtils.jackson().readValue(mvcResult.getResponse().getContentAsString(), Message.class);
  12. assertEquals(res.getStatus(), MessageStatus.SUCCESS());
  13. logger.info(mvcResult.getResponse().getContentAsString());
  14. //Without unnecessary parameters
  15. mvcResult = mockMvc.perform(get("/jobhistory/list"))
  16. .andExpect(status().isOk())
  17. .andExpect(content().contentType(MediaType.APPLICATION_JSON))
  18. .andReturn();
  19. res = JsonUtils.jackson().readValue(mvcResult.getResponse().getContentAsString(), Message.class);
  20. assertEquals(res.getStatus(), MessageStatus.SUCCESS());
  21. logger.info(mvcResult.getResponse().getContentAsString());
  22. }

//todo

Use H2 database, application. In the configuration file In properties, you need to configure the basic information of H2 database and the relevant path information of mybatis

  1. #h2 database configuration
  2. spring.datasource.driver-class-name=org.h2.Driver
  3. # Script to connect database
  4. spring.datasource.url=jdbc:h2:mem:test;MODE=MySQL;DB_CLOSE_DELAY=-1;DATABASE_TO_LOWER=true
  5. #Script to initialize database tables
  6. spring.datasource.schema=classpath:create.sql
  7. #Script to initialize data for database tables
  8. spring.datasource.data=classpath:data.sql
  9. spring.datasource.username=sa
  10. spring.datasource.password=
  11. spring.datasource.hikari.connection-test-query=select 1
  12. spring.datasource.hikari.minimum-idle=5
  13. spring.datasource.hikari.auto-commit=true
  14. spring.datasource.hikari.validation-timeout=3000
  15. spring.datasource.hikari.pool-name=linkis-test
  16. spring.datasource.hikari.maximum-pool-size=50
  17. spring.datasource.hikari.connection-timeout=30000
  18. spring.datasource.hikari.idle-timeout=600000
  19. spring.datasource.hikari.leak-detection-threshold=0
  20. spring.datasource.hikari.initialization-fail-timeout=1
  21. #配置mybatis-plus的mapper信息 因为使用的是mybatis-plus,使用mybatis-plus
  22. mybatis-plus.mapper-locations=classpath:org/apache/linkis/jobhistory/dao/impl/JobDetailMapper.xml,classpath:org/apache/linkis/jobhistory/dao/impl/JobHistoryMapper.xml
  23. mybatis-plus.type-aliases-package=org.apache.linkis.jobhistory.entity
  24. mybatis-plus.configuration.log-impl=org.apache.ibatis.logging.stdout.StdOutImpl

List is configured with predicate of stream to make assertion judgment and write specification

  1. Use @Transactional and @Rollback to realize data rollback and avoid data pollution
  2. Each DaoTest should have a public method for creating and initializing data (or the way of importing data CSV) to prepare data. For related queries, updates, deletions and other operations, the public method should be called first to prepare data
  3. Create test data. If an attribute value is a self increasing ID, it should not be assigned
  4. The test data created shall be consistent with the actual sample data as far as possible
  5. When updating the data test, if the field allows, please prefix it with ‘modify original value’