Skip to content

Commit

Permalink
More comprehensive registration/release of config in some tests
Browse files Browse the repository at this point in the history
If someone calls ConfigProvider.getConfig() out of the blue in a test
that doesn't use any Quarkus*Test extension, this will indirectly call
QuarkusConfigFactory and leak config in two ways:

1. In QuarkusConfigFactory#config
2. In SmallRyeConfigProviderResolver, registering config for the TCCL,
   which in such a case is most likely the system CL.

Thus, a well-behaved test should call QuarkusConfigFactory.setConfig(null)
to clean up all that, no just SmallRyeConfigProviderResolver.releaseConfig().

Similarly, tests that register configuration explicitly can just call
QuarkusConfigFactory.setConfig(config) at the beginning and
QuarkusConfigFactory.setConfig(null) at the end,
which will properly simulate how a real Quarkus application behaves,
and should cover all edge cases involving multiple classloaders,
properly cleaning up everything at the end of the test.
  • Loading branch information
yrodiere committed Oct 25, 2023
1 parent c02cfda commit 02e2966
Show file tree
Hide file tree
Showing 7 changed files with 14 additions and 69 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,6 @@

import org.eclipse.microprofile.config.Config;
import org.eclipse.microprofile.config.ConfigProvider;
import org.eclipse.microprofile.config.spi.ConfigProviderResolver;
import org.junit.jupiter.api.AfterAll;
import org.junit.jupiter.api.BeforeAll;
import org.junit.jupiter.api.Test;
Expand All @@ -21,19 +20,12 @@
*/
public class ApplicationYamlTest {

static volatile SmallRyeConfig config;

@BeforeAll
public static void doBefore() {
SmallRyeConfigBuilder builder = new SmallRyeConfigBuilder();
builder.addDefaultSources().addDiscoveredConverters().addDiscoveredSources();
QuarkusConfigFactory.setConfig(config = builder.build());
Config conf = ConfigProvider.getConfig();
if (conf != config) {
ConfigProviderResolver cpr = ConfigProviderResolver.instance();
cpr.releaseConfig(conf);
ConfigProvider.getConfig();
}
SmallRyeConfig config = builder.build();
QuarkusConfigFactory.setConfig(config);
System.out.println(System.getProperty("user.dir"));
}

Expand All @@ -47,8 +39,6 @@ public void testBasicApplicationYaml() {

@AfterAll
public static void doAfter() {
ConfigProviderResolver cpr = ConfigProviderResolver.instance();
cpr.releaseConfig(config);
config = null;
QuarkusConfigFactory.setConfig(null);
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -8,29 +8,28 @@

import org.eclipse.microprofile.config.Config;
import org.eclipse.microprofile.config.ConfigProvider;
import org.eclipse.microprofile.config.spi.ConfigBuilder;
import org.eclipse.microprofile.config.spi.ConfigProviderResolver;
import org.eclipse.microprofile.rest.client.ext.QueryParamStyle;
import org.junit.jupiter.api.AfterEach;
import org.junit.jupiter.api.BeforeAll;
import org.junit.jupiter.api.Test;

import io.quarkus.runtime.configuration.QuarkusConfigFactory;
import io.smallrye.config.PropertiesConfigSource;
import io.smallrye.config.SmallRyeConfigBuilder;
import io.smallrye.config.SmallRyeConfigProviderResolver;

public class RestClientConfigTest {

private static ClassLoader classLoader;
private static ConfigProviderResolver configProviderResolver;
private static SmallRyeConfigProviderResolver configProviderResolver;

@BeforeAll
public static void initConfig() {
classLoader = Thread.currentThread().getContextClassLoader();
configProviderResolver = ConfigProviderResolver.instance();
configProviderResolver = (SmallRyeConfigProviderResolver) SmallRyeConfigProviderResolver.instance();
}

@AfterEach
public void releaseConfig() {
configProviderResolver.releaseConfig(configProviderResolver.getConfig());
QuarkusConfigFactory.setConfig(null);
}

@Test
Expand Down Expand Up @@ -76,10 +75,10 @@ private void verifyConfig(RestClientConfig config) {
}

private static void setupMPConfig() throws IOException {
ConfigBuilder configBuilder = configProviderResolver.getBuilder();
SmallRyeConfigBuilder configBuilder = configProviderResolver.getBuilder();
URL propertyFile = RestClientConfigTest.class.getClassLoader().getResource("application.properties");
assertThat(propertyFile).isNotNull();
configBuilder.withSources(new PropertiesConfigSource(propertyFile));
configProviderResolver.registerConfig(configBuilder.build(), classLoader);
QuarkusConfigFactory.setConfig(configBuilder.build());
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -21,13 +21,10 @@
import jakarta.ws.rs.client.ClientResponseContext;
import jakarta.ws.rs.client.ClientResponseFilter;

import org.eclipse.microprofile.config.Config;
import org.eclipse.microprofile.config.spi.ConfigProviderResolver;
import org.eclipse.microprofile.rest.client.RestClientBuilder;
import org.eclipse.microprofile.rest.client.ext.QueryParamStyle;
import org.eclipse.microprofile.rest.client.inject.RegisterRestClient;
import org.junit.jupiter.api.AfterAll;
import org.junit.jupiter.api.AfterEach;
import org.junit.jupiter.api.BeforeAll;
import org.junit.jupiter.api.Test;
import org.mockito.Mockito;
Expand All @@ -42,7 +39,6 @@ public class RestClientBaseTest {

private static Path truststorePath;
private static Path keystorePath;
private static Config createdConfig;

@BeforeAll
public static void beforeAll() throws IOException, KeyStoreException, CertificateException, NoSuchAlgorithmException {
Expand All @@ -65,14 +61,6 @@ public static void beforeAll() throws IOException, KeyStoreException, Certificat
}
}

@AfterEach
public void afterEach() {
if (createdConfig != null) {
ConfigProviderResolver.instance().releaseConfig(createdConfig);
createdConfig = null;
}
}

@AfterAll
public static void afterAll() {
if (truststorePath != null) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -16,12 +16,9 @@
import jakarta.ws.rs.client.ClientResponseContext;
import jakarta.ws.rs.client.ClientResponseFilter;

import org.eclipse.microprofile.config.Config;
import org.eclipse.microprofile.config.spi.ConfigProviderResolver;
import org.eclipse.microprofile.rest.client.ext.QueryParamStyle;
import org.eclipse.microprofile.rest.client.inject.RegisterRestClient;
import org.jboss.resteasy.reactive.client.api.QuarkusRestClientProperties;
import org.junit.jupiter.api.AfterEach;
import org.junit.jupiter.api.BeforeAll;
import org.junit.jupiter.api.Test;
import org.junit.jupiter.api.io.TempDir;
Expand All @@ -42,7 +39,6 @@ public class RestClientCDIDelegateBuilderTest {
static File tempDir;
private static File truststoreFile;
private static File keystoreFile;
private static Config createdConfig;

@BeforeAll
public static void beforeAll() throws IOException, KeyStoreException, CertificateException, NoSuchAlgorithmException {
Expand All @@ -60,14 +56,6 @@ public static void beforeAll() throws IOException, KeyStoreException, Certificat
keystore.store(new FileOutputStream(keystoreFile), KEYSTORE_PASSWORD.toCharArray());
}

@AfterEach
public void afterEach() {
if (createdConfig != null) {
ConfigProviderResolver.instance().releaseConfig(createdConfig);
createdConfig = null;
}
}

@Test
public void testClientSpecificConfigs() {
// given
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -11,9 +11,6 @@
import jakarta.inject.Inject;
import jakarta.validation.constraints.Pattern;

import org.eclipse.microprofile.config.Config;
import org.eclipse.microprofile.config.ConfigProvider;
import org.eclipse.microprofile.config.spi.ConfigProviderResolver;
import org.junit.jupiter.api.AfterAll;
import org.junit.jupiter.api.BeforeAll;
import org.junit.jupiter.api.Test;
Expand Down Expand Up @@ -88,19 +85,12 @@ public void validateMapping(Class<?> mappingClass, String prefix, Object mapping
}
});
QuarkusConfigFactory.setConfig(smallryeConfig = builder.build());
Config conf = ConfigProvider.getConfig();
if (conf != smallryeConfig) {
ConfigProviderResolver cpr = ConfigProviderResolver.instance();
cpr.releaseConfig(conf);
ConfigProvider.getConfig();
}
suppressedConfigValidatorExceptions = new HashMap<>();
}

@AfterAll
public static void doAfter() {
ConfigProviderResolver cpr = ConfigProviderResolver.instance();
cpr.releaseConfig(smallryeConfig);
QuarkusConfigFactory.setConfig(null);
smallryeConfig = null;
suppressedConfigValidatorExceptions = null;
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,6 @@
import java.util.regex.Pattern;

import org.eclipse.microprofile.config.Config;
import org.eclipse.microprofile.config.spi.ConfigProviderResolver;

import io.quarkus.runtime.LaunchMode;
import io.quarkus.runtime.configuration.ConfigUtils;
Expand All @@ -41,14 +40,6 @@ private LauncherUtil() {
public static Config installAndGetSomeConfig() {
final SmallRyeConfig config = ConfigUtils.configBuilder(false, LaunchMode.NORMAL).build();
QuarkusConfigFactory.setConfig(config);
final ConfigProviderResolver cpr = ConfigProviderResolver.instance();
try {
final Config installed = cpr.getConfig();
if (installed != config) {
cpr.releaseConfig(installed);
}
} catch (IllegalStateException ignored) {
}
return config;
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,7 @@
import org.jboss.jandex.DotName;
import org.jboss.jandex.IndexView;

import io.smallrye.config.SmallRyeConfigProviderResolver;
import io.quarkus.runtime.configuration.QuarkusConfigFactory;

public class TestResourceManager implements Closeable {

Expand Down Expand Up @@ -180,8 +180,7 @@ public void close() {
}
}
try {
((SmallRyeConfigProviderResolver) SmallRyeConfigProviderResolver.instance())
.releaseConfig(Thread.currentThread().getContextClassLoader());
QuarkusConfigFactory.setConfig(null);
} catch (Throwable ignored) {
}
configProperties.clear();
Expand Down

0 comments on commit 02e2966

Please sign in to comment.