Spring Boot testing - Focus on your changes

Jan 31, 2022

Testing in Spring Boot

Time and time again we find ourselves in the situation of writing a new micro-service in SpringBoot. Part of the TDD process is to start writing tests first. As not all the times is possible to write the tests first, we will go back and add the tests after we wrote the code. As there are many reasons to not do that, I will not enumerate them here. Let’s focus on how to test what matters, versus testing what was already tested.

Imports omitted for code brevity.

Initial Java code

package com.georgeracu.blog.springboot.application.config;

public class CustomWebMvcConfigurer implements WebMvcConfigurer {
    
    @Value("${config.origins.allowed}")
    private String allowedOrigins;

    @Override
    public void addCorsMappings(CorsRegistry registry) {
        if(StringUtils.isNotEmpty(allowedOrigins)) {
            registry
                .addMapping("/**")
                .allowedOrigins(allowedOrigins.split(","));

        }
    }
}

Once our production code has been written, let’s see how a test could look like.

package com.georgeracu.blog.springboot.application.config;

class CustomWebMvcConfigurerTest {
    private CustomWebMvcConfigurer configurer;
    private TestCorsRegistry registry;

    @BeforeEach
    void setup() {
        configurer = new CustomWebMvcConfigurer();
        registry = new TestCorsRegistry();
    }

    @ParameterizedTest
    @MethodSource("argsForCorsProvider")
    void shouldAddCorsMappingsWhenEnabled(String allowedOrigins, int expectedSize) {
        // arrange
        ReflectionTestUtils.setField(configurer, "allowedOrigins", allowedOrigins);

        // act
        configurer.addCorsMappings(registry);

        // assert
        assertEquals(1, registry.getCorsConfigurations().size());
    }

    private static Stream<Arguments> argsForCorsProvider() {
        return Stream.of(Arguments.of("http://example.com", 1));
    }

    class TestCorsRegistry extends CorsRegistry {
        public Map<String, CorsConfiguration> getCorsConfiguration() {
            return super.getCorsConfiguration();
        }
    }
}

So far, we wrote a test that is testing that our CORS origins are added to the list of allowed origins. All nice and simple, but a few observations:

  • The test is testing that the CorsRegistry adds a mapping to the list of the allowed origins. This feels that it was tested already by the creators of the framework;
  • We are not focusing that we actually tested our custom code: adding several values to the list of the allowed origins;
  • The class TestCorsRegistry is not needed when there are powerful mocking frameworks that allow us to mock that object and assert that custom behavior has happened on the mock itself;
  • Field injection is in our way of actually writing simpler unit tests;

Lets’s see how this code would look like with a bit of refactoring.

package com.georgeracu.blog.springboot.application.config;

@Configuration
public class CustomWebMvcConfigurer implements WebMvcConfigurer {
    
    private String allowedOrigins;

    public CustomWebMvcConfigurer(@Value("${config.origins.allowed}") String allowedOrigins) {
        this.allowedOrigins = allowedOrigins;
    }

    @Override
    public void addCorsMappings(CorsRegistry registry) {
        if(StringUtils.isNotEmpty(allowedOrigins)) {
            registry
                .addMapping("/**")
                .allowedOrigins(allowedOrigins.split(","));

        }
    }
}
package com.georgeracu.blog.springboot.application.config;

class CustomWebMvcConfigurerTest {
    private CustomWebMvcConfigurer configurer;
    private CorsRegistry registry;
    private CorsRegistration corsRegistration;

    @BeforeEach
    void setup() {
        registry = Mockito.mock(CorsRegistry.class);
        corsRegistration = Mockito.mock(CorsRegistration.class);
    }

    @ParameterizedTest
    @MethodSource("argsForCorsProvider")
    void shouldAddCorsMappingsWhenEnabled(String allowedOrigins) {
        // arrange
        when(registry.addMapping("/**")).thenReturn(corsRegistration);
        configurer = new CustomWebMvcConfigurer(allowedOrigins);

        // act
        configurer.addCorsMappings(registry);

        // assert
        verify(registry).addMapping("/**");
        verify(corsRegistration).allowedOrigins(allowedOrigins);
    }

    private static Stream<Arguments> argsForCorsProvider() {
        return Stream.of(Arguments.of("http://example.com"));
    }
}

Tags: programmingspring bootjavatesting

Archives

  1. December 2024
  2. Keeping Software Simple to speed up Software Development
  3. October 2024
  4. The Kanban Café - A Story of Flow
  5. A Story on Accidental Complexity in Software Development
  6. February 2024
  7. Maximizing Software Development Productivity: The Power of Flow and Minimizing Interruptions
  8. December 2023
  9. Clean Code in Java: Writing Code that Speaks
  10. Clean Code in Java: A concise guide
  11. Understanding Value Objects in Java: A Brief Guide
  12. August 2023
  13. Must Have on Message Payload
  14. Centralised Management System For Message Schemas
  15. Consuming RabbitMQ Messages with Clojure: A Step-by-Step Tutorial with Tests
  16. January 2023
  17. Running a Spring Boot service with kubernetes
  18. December 2022
  19. Hosting a PWA with Jekyll and Github pages
  20. November 2022
  21. Global Day of Code Retreat
  22. Facilitating a mini Code Retreat
  23. October 2022
  24. The Curse of Optional
  25. September 2022
  26. Testing Spring Boot Microservices - Presentation
  27. March 2022
  28. TDD Workshop
  29. February 2022
  30. Value Objects in Java
  31. Efficient Java
  32. January 2022
  33. Spring Boot testing - Focus on your changes
  34. Product users - Personas
  35. December 2021
  36. Write code fit for testing
  37. November 2020
  38. Running a Spring Boot app with kubernetes
  39. September 2019
  40. Setup GPG on Mac and sign git repositories
  41. July 2019
  42. Running a Clojure Pedestal application on Raspberry Pi model B revision 2
  43. Clojure from zero to hero (part 3) - First endpoint
  44. Clojure from zero to hero (part 2) - A bit of syntax
  45. June 2019
  46. Clojure from zero to hero (1) - explaining project.clj
  47. Clojure from zero to hero (0) - creating a Pedestal app
  48. November 2017
  49. Introduction to Docker
  50. April 2015
  51. Git micro commits
  52. July 2014
  53. Google Glass Development - setup tools, environment and turn on debugging on Glass
  54. June 2013
  55. How To: Get the rendered HTML of a webpage with Python
  56. Set union of two lists in Python