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

Match most specific mapping prefix when matching environment variables #1267

Merged
merged 1 commit into from
Dec 11, 2024
Merged
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Match most specific mapping prefix when matching environment variables
  • Loading branch information
radcortez committed Dec 11, 2024
commit ac93b31d9bd545efa5a9987d8078f48d54f3d672
Original file line number Diff line number Diff line change
@@ -10,6 +10,7 @@
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.Comparator;
import java.util.HashMap;
import java.util.HashSet;
import java.util.IdentityHashMap;
@@ -70,7 +71,11 @@ public Map<String, Map<String, Set<String>>> get() {

this.config = config;
this.names = new ConfigMappingNames(names);
matchPropertiesWithEnv(roots);
Set<String> mappingsPrefixes = new HashSet<>();
for (Set<String> mappingPrefixes : roots.values()) {
mappingsPrefixes.addAll(mappingPrefixes);
}
matchPropertiesWithEnv(mappingsPrefixes);
for (Map.Entry<Class<?>, Set<String>> mapping : roots.entrySet()) {
Map<String, ConfigMappingObject> mappingObjects = new HashMap<>();
for (String rootPath : mapping.getValue()) {
@@ -172,19 +177,36 @@ Map<Class<?>, Map<String, ConfigMappingObject>> getRootsMap() {
return roots;
}

private void matchPropertiesWithEnv(final Map<Class<?>, Set<String>> roots) {
private void matchPropertiesWithEnv(final Set<String> mappingsPrefixes) {
// TODO - We shouldn't be mutating the EnvSource.
// We should do the calculation when creating the EnvSource, but right now mappings and sources are not well integrated.

Set<String> rootPaths = new HashSet<>();
for (Set<String> paths : roots.values()) {
rootPaths.addAll(paths);
}
boolean all = rootPaths.contains("");
List<String> prefixes = new ArrayList<>(mappingsPrefixes);
// Sort by number of segments to match the most specific ones first
prefixes.sort(new Comparator<String>() {
@Override
public int compare(final String o1, final String o2) {
int segmentsO1 = 0;
for (int i = 0; i < o1.length(); i++) {
if (o1.charAt(i) == '.') {
segmentsO1++;
}
}

int segmentsO2 = 0;
for (int i = 0; i < o2.length(); i++) {
if (o2.charAt(i) == '.') {
segmentsO2++;
}
}
return Integer.compare(segmentsO2, segmentsO1);
}
});
boolean all = prefixes.contains("");
StringBuilder sb = new StringBuilder();

for (ConfigSource configSource : config.getConfigSources(EnvConfigSource.class)) {
if (roots.isEmpty()) {
if (prefixes.isEmpty()) {
break;
}

@@ -201,7 +223,7 @@ private void matchPropertiesWithEnv(final Map<Class<?>, Set<String>> roots) {

String matchedRoot = null;
if (!all) {
for (String rootPath : rootPaths) {
for (String rootPath : prefixes) {
if (StringUtil.isInPath(rootPath, activeEnvProperty)) {
matchedRoot = rootPath;
break;
Original file line number Diff line number Diff line change
@@ -590,6 +590,86 @@ interface ServiceDiscoveryConfiguration {
}
}

@Test
void mappingMapsWithEnvMultiplePrefixes() {
SmallRyeConfig config = new SmallRyeConfigBuilder()
.withSources(new EnvConfigSource(Map.of(
"PREFIX_COMPOSED_SERVICE_SERVICE_DISCOVERY_ADDRESS", "from-env"), 300))
.withMapping(SimplePrefix.class)
.withMapping(ComposedPrefix.class)
.build();

ComposedPrefix mapping = config.getConfigMapping(ComposedPrefix.class);
assertEquals("from-env", mapping.serviceConfiguration().get("service").serviceDiscovery().params().get("address"));
}

@ConfigMapping(prefix = "prefix")
interface SimplePrefix {
Optional<String> value();
}

@ConfigMapping(prefix = "prefix.composed")
interface ComposedPrefix {
@WithParentName
Map<String, ServiceConfiguration> serviceConfiguration();

interface ServiceConfiguration {
ServiceDiscoveryConfiguration serviceDiscovery();

interface ServiceDiscoveryConfiguration {
@WithParentName
Map<String, String> params();
}
}
}

@Test
void mappingsMapsWithEnvSplit() {
SmallRyeConfig config = new SmallRyeConfigBuilder()
.withSources(new EnvConfigSource(Map.of(
"SERVICES_SERVICE_SERVICE_DISCOVERY_ADDRESS", "from-env",
"SERVICES_SERVICE_ANOTHER_DISCOVERY_ADDRESS", "from-env"), 300))
.withMapping(ServicesOne.class)
.withMapping(ServicesTwo.class)
.build();

ServicesOne servicesOne = config.getConfigMapping(ServicesOne.class);
ServicesTwo servicesTwo = config.getConfigMapping(ServicesTwo.class);

assertEquals("from-env", servicesOne.serviceConfiguration().get("service").serviceDiscovery().params().get("address"));
assertEquals("from-env", servicesTwo.serviceConfiguration().get("service").anotherDiscovery().params().get("address"));
}

@ConfigMapping(prefix = "services")
interface ServicesOne {
@WithParentName
Map<String, ServiceConfiguration> serviceConfiguration();

interface ServiceConfiguration {
ServiceDiscoveryConfiguration serviceDiscovery();

interface ServiceDiscoveryConfiguration {
@WithParentName
Map<String, String> params();
}
}
}

@ConfigMapping(prefix = "services")
interface ServicesTwo {
@WithParentName
Map<String, ServiceConfiguration> serviceConfiguration();

interface ServiceConfiguration {
ServiceDiscoveryConfiguration anotherDiscovery();

interface ServiceDiscoveryConfiguration {
@WithParentName
Map<String, String> params();
}
}
}

private static boolean envSourceEquals(String name, String lookup) {
return BOOLEAN_CONVERTER.convert(new EnvConfigSource(Map.of(name, "true"), 100).getValue(lookup));
}