Skip to content

Commit d983c26

Browse files
Allow resolving the server URL via a property (#533)
* First cut * Allow resolving the server URL via a property Fixes #527 * Make Checkstyle happier
1 parent 824d797 commit d983c26

File tree

9 files changed

+198
-17
lines changed

9 files changed

+198
-17
lines changed

gradle/libs.versions.toml

+3
Original file line numberDiff line numberDiff line change
@@ -45,6 +45,7 @@ spock = "2.3-groovy-4.0"
4545
vertx-client = "4.5.4"
4646
amazon-awssdk-v1 = "1.12.667"
4747
amazon-awssdk-v2 = "2.24.11"
48+
jansi = "1.18"
4849

4950
# Managed versions appear in the BOM
5051
managed-testcontainers = "1.19.6"
@@ -110,6 +111,8 @@ amazon-awssdk-v1-sqs = { module = "com.amazonaws:aws-java-sdk-sqs", version.ref
110111
amazon-awssdk-v2-dynamodb = { module = "software.amazon.awssdk:dynamodb", version.ref = "amazon-awssdk-v2" }
111112
amazon-awssdk-v2-s3 = { module = "software.amazon.awssdk:s3", version.ref = "amazon-awssdk-v2" }
112113
amazon-awssdk-v2-sqs = { module = "software.amazon.awssdk:sqs", version.ref = "amazon-awssdk-v2" }
114+
115+
jansi = { module = "org.fusesource.jansi:jansi", version.ref = "jansi" }
113116
#
114117
# Managed dependencies appear in the BOM
115118
#
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
/*
2+
* Copyright 2017-2024 original authors
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* https://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
package io.micronaut.testresources.core;
17+
18+
import java.util.List;
19+
20+
/**
21+
* Interface for loading {@link TestResourcesResolver} instances.
22+
*
23+
* @author Álvaro Sánchez-Mariscal
24+
* @since 2.4.0
25+
*/
26+
public interface ResolverLoader {
27+
28+
List<TestResourcesResolver> getResolvers();
29+
}

test-resources-embedded/src/main/java/io/micronaut/testresources/embedded/TestResourcesResolverLoader.java

+3-1
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,7 @@
1717

1818
import io.micronaut.core.io.service.SoftServiceLoader;
1919
import io.micronaut.core.order.OrderUtil;
20+
import io.micronaut.testresources.core.ResolverLoader;
2021
import io.micronaut.testresources.core.TestResourcesResolver;
2122
import org.slf4j.Logger;
2223
import org.slf4j.LoggerFactory;
@@ -29,7 +30,7 @@
2930
* This class is responsible for loading {@link TestResourcesResolver} instances
3031
* via service loading and caching them.
3132
*/
32-
public final class TestResourcesResolverLoader {
33+
public final class TestResourcesResolverLoader implements ResolverLoader {
3334
private static final Logger LOGGER = LoggerFactory.getLogger(TestResourcesResolverLoader.class);
3435

3536
private final List<TestResourcesResolver> resolvers;
@@ -53,6 +54,7 @@ public TestResourcesResolverLoader() {
5354
}
5455
}
5556

57+
@Override
5658
public List<TestResourcesResolver> getResolvers() {
5759
return resolvers;
5860
}

test-resources-server/build.gradle

+9-9
Original file line numberDiff line numberDiff line change
@@ -17,23 +17,23 @@ client.
1717
dependencies {
1818
implementation(mn.micronaut.http.server.netty)
1919
implementation(mn.reactor)
20-
implementation(project(':micronaut-test-resources-core'))
21-
implementation(project(':micronaut-test-resources-embedded'))
22-
implementation(project(':micronaut-test-resources-testcontainers'))
20+
implementation(projects.micronautTestResourcesCore)
21+
implementation(projects.micronautTestResourcesEmbedded)
22+
implementation(projects.micronautTestResourcesTestcontainers)
2323
runtimeOnly(mnLogging.logback.classic)
2424
runtimeOnly(mn.micronaut.management)
2525
runtimeOnly(mnSerde.micronaut.serde.jackson)
2626

27-
testFixturesImplementation(project(':micronaut-test-resources-core'))
28-
testFixturesImplementation(project(':micronaut-test-resources-testcontainers'))
27+
testFixturesImplementation(projects.micronautTestResourcesCore)
28+
testFixturesImplementation(projects.micronautTestResourcesTestcontainers)
2929
testImplementation(mn.micronaut.http.client)
30-
testImplementation(project(':micronaut-test-resources-client'))
31-
testRuntimeOnly(project(':micronaut-test-resources-kafka'))
30+
testImplementation(projects.micronautTestResourcesClient)
31+
testRuntimeOnly(projects.micronautTestResourcesKafka)
3232

3333
// For logback conversion
3434
aotPlugins(mnLogging.logback.classic)
35-
aotPlugins("org.fusesource.jansi:jansi:1.18")
36-
aotPlugins(testFixtures(project(":micronaut-test-resources-server")))
35+
aotPlugins(libs.jansi)
36+
aotPlugins(testFixtures(projects.micronautTestResourcesServer))
3737

3838
// In order to support the control panel at runtime
3939
aotApplication projects.micronautTestResourcesControlPanel
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,49 @@
1+
/*
2+
* Copyright 2017-2024 original authors
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* https://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
package io.micronaut.testresources.server;
17+
18+
import io.micronaut.testresources.core.ResolverLoader;
19+
import io.micronaut.testresources.core.TestResourcesResolver;
20+
import io.micronaut.testresources.embedded.TestResourcesResolverLoader;
21+
import jakarta.inject.Singleton;
22+
23+
import java.util.ArrayList;
24+
import java.util.List;
25+
26+
/**
27+
* A {@link ResolverLoader} that loads {@link TestResourcesResolver} instances from both Java
28+
* service loading and the Micronaut application context.
29+
*
30+
* @author Álvaro Sánchez-Mariscal
31+
* @since 2.4.0
32+
*/
33+
@Singleton
34+
public class CompositeResolverLoader implements ResolverLoader {
35+
36+
private final List<TestResourcesResolver> resolvers;
37+
38+
public CompositeResolverLoader(List<InjectableTestResourcesResolver> resolverBeans) {
39+
var loader = TestResourcesResolverLoader.getInstance();
40+
this.resolvers = new ArrayList<>(resolverBeans.size() + loader.getResolvers().size());
41+
this.resolvers.addAll(resolverBeans);
42+
this.resolvers.addAll(loader.getResolvers());
43+
}
44+
45+
@Override
46+
public List<TestResourcesResolver> getResolvers() {
47+
return resolvers;
48+
}
49+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
/*
2+
* Copyright 2017-2024 original authors
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* https://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
package io.micronaut.testresources.server;
17+
18+
import io.micronaut.testresources.core.TestResourcesResolver;
19+
20+
/**
21+
* Marker interface for a {@link TestResourcesResolver} implementations that can be injected.
22+
*
23+
* @author Álvaro Sánchez-Mariscal
24+
* @since 2.4.0
25+
*/
26+
public interface InjectableTestResourcesResolver extends TestResourcesResolver {
27+
28+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,59 @@
1+
/*
2+
* Copyright 2017-2024 original authors
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* https://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
package io.micronaut.testresources.server;
17+
18+
import io.micronaut.core.annotation.Internal;
19+
import io.micronaut.runtime.server.EmbeddedServer;
20+
import io.micronaut.testresources.core.TestResourcesResolver;
21+
import jakarta.inject.Singleton;
22+
23+
import java.util.Collection;
24+
import java.util.List;
25+
import java.util.Map;
26+
import java.util.Optional;
27+
28+
/**
29+
* A {@link TestResourcesResolver} that resolves properties regarding the Test Resource Service itself,
30+
* such as <code>micronaut.test.resources.server.uri</code>.
31+
*
32+
* @author Álvaro Sánchez-Mariscal
33+
* @since 2.4.0
34+
*/
35+
@Internal
36+
@Singleton
37+
public class InternalTestResourcesServiceResolver implements InjectableTestResourcesResolver {
38+
39+
public static final String SERVER_URI = "micronaut.test.resources.server.uri";
40+
41+
private final EmbeddedServer server;
42+
43+
public InternalTestResourcesServiceResolver(EmbeddedServer server) {
44+
this.server = server;
45+
}
46+
47+
@Override
48+
public List<String> getResolvableProperties(Map<String, Collection<String>> propertyEntries, Map<String, Object> testResourcesConfig) {
49+
return List.of(SERVER_URI);
50+
}
51+
52+
@Override
53+
public Optional<String> resolve(String propertyName, Map<String, Object> properties, Map<String, Object> testResourcesConfig) {
54+
if (SERVER_URI.equals(propertyName)) {
55+
return Optional.of(server.getURI().toString());
56+
}
57+
return Optional.empty();
58+
}
59+
}

test-resources-server/src/main/java/io/micronaut/testresources/server/TestResourcesController.java

+6-4
Original file line numberDiff line numberDiff line change
@@ -23,10 +23,10 @@
2323
import io.micronaut.runtime.server.EmbeddedServer;
2424
import io.micronaut.scheduling.TaskExecutors;
2525
import io.micronaut.scheduling.annotation.ExecuteOn;
26+
import io.micronaut.testresources.core.ResolverLoader;
2627
import io.micronaut.testresources.core.TestResourcesResolutionException;
2728
import io.micronaut.testresources.core.TestResourcesResolver;
2829
import io.micronaut.testresources.core.ToggableTestResourcesResolver;
29-
import io.micronaut.testresources.embedded.TestResourcesResolverLoader;
3030
import io.micronaut.testresources.testcontainers.TestContainers;
3131
import org.slf4j.Logger;
3232
import org.slf4j.LoggerFactory;
@@ -50,22 +50,24 @@ public class TestResourcesController implements TestResourcesResolver {
5050
private static final Logger LOGGER = LoggerFactory.getLogger(TestResourcesController.class);
5151
private static final int MAX_STOP_TIMEOUT = 5000;
5252

53-
private final TestResourcesResolverLoader loader = TestResourcesResolverLoader.getInstance();
53+
private final ResolverLoader loader;
5454

5555
private final List<PropertyResolutionListener> propertyResolutionListeners;
5656
private final EmbeddedServer embeddedServer;
5757
private final ApplicationContext applicationContext;
5858

5959
public TestResourcesController(List<PropertyResolutionListener> propertyResolutionListeners,
6060
EmbeddedServer embeddedServer,
61-
ApplicationContext applicationContext) {
61+
ApplicationContext applicationContext,
62+
ResolverLoader loader) {
6263
this.propertyResolutionListeners = propertyResolutionListeners;
6364
this.embeddedServer = embeddedServer;
6465
this.applicationContext = applicationContext;
66+
this.loader = loader;
6567
}
6668

6769
/**
68-
* Lists all resolvable properties. Prefer {@link #getResolvableProperties()} to list
70+
* Lists all resolvable properties. Prefer {@link #getResolvableProperties(Map, Map)} to list
6971
* all properties which can be resolved for a particular configuration.
7072
*
7173
* @return the list of resolvable properties which do not depend on the application configuration

test-resources-server/src/test/groovy/io/micronaut/testresources/server/TestResourcesControllerTest.groovy

+12-3
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,7 @@ class TestResourcesControllerTest extends Specification {
2020

2121
def "verifies that server can instantiate container"() {
2222
expect:
23-
client.resolvableProperties ==~ ['kafka.bootstrap.servers', 'failing.message', 'failing.container']
23+
client.resolvableProperties ==~ ['kafka.bootstrap.servers', 'failing.message', 'failing.container', 'micronaut.test.resources.server.uri']
2424
client.listContainers().empty
2525

2626
when:
@@ -40,7 +40,7 @@ class TestResourcesControllerTest extends Specification {
4040

4141
def "can resolve a valid property after an error in a regular test resource"() {
4242
expect:
43-
client.resolvableProperties ==~ ['kafka.bootstrap.servers', 'failing.message', 'failing.container']
43+
client.resolvableProperties ==~ ['kafka.bootstrap.servers', 'failing.message', 'failing.container', 'micronaut.test.resources.server.uri']
4444
client.listContainers().empty
4545

4646
when: "resolution of a property fails"
@@ -66,7 +66,7 @@ class TestResourcesControllerTest extends Specification {
6666

6767
def "can resolve a valid property after an error in a container test resource"() {
6868
expect:
69-
client.resolvableProperties ==~ ['kafka.bootstrap.servers', 'failing.message', 'failing.container']
69+
client.resolvableProperties ==~ ['kafka.bootstrap.servers', 'failing.message', 'failing.container', 'micronaut.test.resources.server.uri']
7070
client.listContainers().empty
7171

7272
when: "resolution of a property which starts a container"
@@ -90,6 +90,15 @@ class TestResourcesControllerTest extends Specification {
9090
client.listContainers().empty
9191
}
9292

93+
def "can resolve the server URL"() {
94+
when:
95+
def url = client.resolve("micronaut.test.resources.server.uri", [:], [:])
96+
97+
then:
98+
url.isPresent()
99+
url.get().startsWith("http://localhost:")
100+
}
101+
93102
@Client("/")
94103
static interface DiagnosticsClient extends TestResourcesClient {
95104
@Get("/testcontainers")

0 commit comments

Comments
 (0)