Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Support @TestPropertySource at method level [SPR-14378] #18951

Open
spring-projects-issues opened this issue Jun 17, 2016 · 24 comments
Open

Support @TestPropertySource at method level [SPR-14378] #18951

spring-projects-issues opened this issue Jun 17, 2016 · 24 comments
Labels
in: test Issues in the test module type: enhancement A general enhancement

Comments

@spring-projects-issues
Copy link
Collaborator

spring-projects-issues commented Jun 17, 2016

Justin Knowles opened SPR-14378 and commented

@TestPropertySource allows you to set test-suite / class-level properties, but I want test-level properties so I can test drive classes with @ConditionalOnProperty and @Value(${var:default}) type annotations for correct behavior.

I have considered creating separate test suites for each test, but that seems unnatural. I currently manually manage the context within the test to validate the behavior.


Issue Links:

8 votes, 14 watchers

@spring-projects-issues
Copy link
Collaborator Author

Seth Wingert commented

This would be really useful for me as well. I'm doing a lot of custom @AutoConfigure and managing different properties between tests is really tricky since @TestPropertySource only works on the class level.

@spring-projects-issues
Copy link
Collaborator Author

vijay parashar commented

Any timeline for fixing this issue.

@membersound
Copy link

Any plans on this? It would be very useful to not having to create a separate class each time an application.properties parameter should change for testing.

@RonaldFindling
Copy link

We would love to see this aswell

@drodionov
Copy link

It would be great to have this feature!

@srferron
Copy link

Yes, it´s needed

@ben-pearson
Copy link

Yep, 4 years on, I want this too!

@jhoeller jhoeller modified the milestones: 5.x Backlog, General Backlog Aug 24, 2020
@aarowman
Copy link

Any updates on this? Would really make testing simpler on conditional properties...

@bruno-oliveira
Copy link

Bump

@joel-regen
Copy link

I need this too!!

@blorp-goes-the-derp
Copy link

Also could use this feature!

@ahoehma
Copy link

ahoehma commented Jun 6, 2021

I have a working implementation for this.

@vinay36999
Copy link

Most developers resort to ReflectionTestUtils.setField(...) to work around this limitation, if the need happens to be to set just couple of properties.

@pieterdb4477
Copy link

This would be a great addition, indeed.

@rahul-gupta-1525
Copy link

good to have this in spring, looking forward to this

@bbortt
Copy link

bbortt commented Sep 6, 2021

@ahoehma

I have a working implementation for this.

would you mind sharing it? 😉

@bfrggit
Copy link

bfrggit commented Oct 27, 2021

Should be very helpful when testing some component with different configurations.

@EarthCitizen
Copy link

This would be BEYOND incredible. No more nested static test classes just to have different properties for 1 or 2 tests!!!

@EarthCitizen
Copy link

@sbrannen Any chance of getting this on the next minor version bump?

@sbrannen
Copy link
Member

@sbrannen Any chance of getting this on the next minor version bump?

The change would be too large for a point release. If we implement this, it would come in 6.x.

@romerorsp
Copy link

It Will be great when you guys release that change in 6.x! I'd needed it now, but will figure another way of achieving the same goal.

@sergey-morenets
Copy link

It Will be great when you guys release that change in 6.x! I'd needed it now, but will figure another way of achieving the same goal.

Hi @romerorsp

You can easily achieve this functionality if you use Junit 5. So you can just extract all the methods into inner class:

@SpringJUnitConfig(AppConfig.class)
public class ServerTest {

	@Nested
	@TestPropertySource(properties = "db.port=7000")
	public class ServerLoadConfiguration {

@martinwunderlich-celonis

any updates on this? would be a great feature to have.

@ahoehma
Copy link

ahoehma commented Sep 10, 2024

Let me share what I have here:

import static com.google.common.base.Preconditions.checkNotNull;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
import java.lang.reflect.Method;
import java.util.List;
import java.util.Properties;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.cloud.context.config.annotation.RefreshScope;
import org.springframework.cloud.context.refresh.ContextRefresher;
import org.springframework.cloud.context.scope.refresh.RefreshScopeRefreshedEvent;
import org.springframework.core.env.ConfigurableEnvironment;
import org.springframework.core.env.PropertiesPropertySource;
import org.springframework.test.annotation.DirtiesContext;
import org.springframework.test.annotation.DirtiesContext.ClassMode;
import org.springframework.test.annotation.DirtiesContext.MethodMode;
import org.springframework.test.context.TestContext;
import org.springframework.test.context.support.AbstractTestExecutionListener;

import com.google.common.collect.ArrayListMultimap;
import com.google.common.collect.Lists;
import com.google.common.collect.Multimap;

/**
 * Note: the whole idea of test-method-specific properties is ONLY working if the consuming bean is initialized AFTER spring was able to add the property source to the context. Any other combination
 * of early initialized bean, constructor using properties etc. will not work. Even if you put a {@link RefreshScope} on such beans. The problem is how spring is calling this listener here. Its
 * sometimes simply too late. Maybe later if we really need/have refresh-able apps (consuming {@link RefreshScopeRefreshedEvent} etc.) then this can work.
 *
 * @author Andreas Höhmann
 */
public final class TestPropertiesPerMethodHandler extends AbstractTestExecutionListener {

  private static final Logger LOGGER = LoggerFactory.getLogger(TestPropertiesPerMethodHandler.class);

  private final Multimap<Method, PropertiesPropertySource> propertySources = ArrayListMultimap.create();

  @Target({
      ElementType.TYPE,
      ElementType.METHOD
  })
  @Retention(RetentionPolicy.RUNTIME)
  @DirtiesContext(methodMode = MethodMode.AFTER_METHOD, classMode = ClassMode.AFTER_CLASS)
  public @interface TestProperties {
    @Target({
        ElementType.TYPE,
        ElementType.METHOD
    })
    @Retention(RetentionPolicy.RUNTIME)
    @DirtiesContext(methodMode = MethodMode.AFTER_METHOD, classMode = ClassMode.AFTER_CLASS)
    public @interface TestProperty {
      String name();
      String value();
    }
    TestProperty[] value();
  }

  private static List<PropertiesPropertySource> getTestProperties(final Method method) {
    checkNotNull(method, "Method must not be null");
    final List<PropertiesPropertySource> result = Lists.newArrayList();
    {
      final TestProperties[] allTestEnvironmentProperties = method.getAnnotationsByType(TestProperties.class);
      if (allTestEnvironmentProperties != null) {
        int c = 0;
        for (final TestProperties testEnvironmentProperties : allTestEnvironmentProperties) {
          final Properties properties = new Properties();
          for (final TestProperty testEnvironmentProperty : testEnvironmentProperties.value()) {
            properties.put(testEnvironmentProperty.name(), testEnvironmentProperty.value());
          }
          if (!properties.isEmpty()) {
            result.add(new PropertiesPropertySource(method.getName().toLowerCase() + "-" + c, properties));
            c++;
          }
        }
      }
    }
    {
      final TestProperty[] allTestEnvironmentProperties = method.getAnnotationsByType(TestProperty.class);
      if (allTestEnvironmentProperties != null) {
        final Properties properties = new Properties();
        for (final TestProperty testEnvironmentProperty : allTestEnvironmentProperties) {
          properties.put(testEnvironmentProperty.name(), testEnvironmentProperty.value());
        }
        if (!properties.isEmpty()) {
          result.add(new PropertiesPropertySource(method.getName().toLowerCase(), properties));
        }
      }
    }
    return result;
  }

  @Override
  public void beforeTestMethod(final TestContext testContext) throws Exception {
    final Method testMethod = testContext.getTestMethod();
    boolean needRefresh = false;
    for (final PropertiesPropertySource propertiesPropertySource : getTestProperties(testMethod)) {
      LOGGER.info("Register additional property source '{}' with '{}' for method '{}'", propertiesPropertySource.getName(),
          propertiesPropertySource.getSource(), testMethod);
      ((ConfigurableEnvironment) testContext.getApplicationContext().getEnvironment())
          .getPropertySources()
          .addFirst(propertiesPropertySource);
      propertySources.put(testMethod, propertiesPropertySource);
      needRefresh = true;
    }
    if (needRefresh) {
      testContext.getApplicationContext().getBean(ContextRefresher.class).refresh();
    }
  }

  @Override
  public void afterTestMethod(final TestContext testContext) throws Exception {
    final Method testMethod = testContext.getTestMethod();
    boolean needRefresh = false;
    for (final PropertiesPropertySource propertiesPropertySource : propertySources.removeAll(testMethod)) {
      LOGGER.info("Unregister additional property source '{}' with '{}' for method '{}'", propertiesPropertySource.getName(),
          propertiesPropertySource.getSource(), testMethod);
      ((ConfigurableEnvironment) testContext.getApplicationContext().getEnvironment())
          .getPropertySources()
          .remove(propertiesPropertySource.getName());
      needRefresh = true;
      if (needRefresh) {
        testContext.getApplicationContext().getBean(ContextRefresher.class).refresh();
      }
    }
  }
}

Example:

import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.boot.test.context.SpringBootTest.WebEnvironment;
import org.junit.jupiter.api.Test;
import TestPropertiesPerMethodHandler.TestProperties;
import TestPropertiesPerMethodHandler.TestProperties.TestProperty;

@SpringBootTest(webEnvironment = WebEnvironment.NONE)
@TestPropertySource(
  properties = {
      "foo=false",
      "logging.level.TestPropertiesPerMethodHandler=DEBUG",
  })
@TestExecutionListeners(
  listeners = {
      TestPropertiesPerMethodHandler.class,
  }, mergeMode = TestExecutionListeners.MergeMode.MERGE_WITH_DEFAULTS)
class MyTest {

  @Test
  @TestProperties({
      @TestProperty(name = "foo", value = "true"), // will override global "foo" 
      @TestProperty(name = "bar", value = "false")
  })
  void test() {
   ... testing
  }
}

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
in: test Issues in the test module type: enhancement A general enhancement
Projects
None yet
Development

No branches or pull requests