diff --git a/spring-cloud-kubernetes-client-autoconfig/src/main/java/org/springframework/cloud/kubernetes/client/KubernetesClientPodUtils.java b/spring-cloud-kubernetes-client-autoconfig/src/main/java/org/springframework/cloud/kubernetes/client/KubernetesClientPodUtils.java index a87fd61409..c1edcf87b7 100644 --- a/spring-cloud-kubernetes-client-autoconfig/src/main/java/org/springframework/cloud/kubernetes/client/KubernetesClientPodUtils.java +++ b/spring-cloud-kubernetes-client-autoconfig/src/main/java/org/springframework/cloud/kubernetes/client/KubernetesClientPodUtils.java @@ -109,6 +109,9 @@ private V1Pod internalGetPod() { } catch (Throwable t) { if (failFast) { + if (t instanceof ApiException apiException) { + LOG.error("error reading pod : " + apiException.getResponseBody()); + } throw new RuntimeException(t); } if (t instanceof ApiException apiException) { diff --git a/spring-cloud-kubernetes-commons/src/main/java/org/springframework/cloud/kubernetes/commons/config/ConfigUtils.java b/spring-cloud-kubernetes-commons/src/main/java/org/springframework/cloud/kubernetes/commons/config/ConfigUtils.java index 70f6b1ce96..9efc8ba77f 100644 --- a/spring-cloud-kubernetes-commons/src/main/java/org/springframework/cloud/kubernetes/commons/config/ConfigUtils.java +++ b/spring-cloud-kubernetes-commons/src/main/java/org/springframework/cloud/kubernetes/commons/config/ConfigUtils.java @@ -23,7 +23,10 @@ import java.util.LinkedHashSet; import java.util.List; import java.util.Map; +import java.util.Optional; import java.util.Set; +import java.util.function.BiPredicate; +import java.util.function.BooleanSupplier; import java.util.function.Function; import java.util.function.Supplier; import java.util.stream.Collectors; @@ -51,6 +54,12 @@ public final class ConfigUtils { private static final Log LOG = LogFactory.getLog(ConfigUtils.class); + // sourceName (configmap or secret name) ends with : "-dev.yaml" or the like. + private static final BiPredicate ENDS_WITH_PROFILE_AND_EXTENSION = (sourceName, + activeProfile) -> sourceName.endsWith("-" + activeProfile + ".yml") + || sourceName.endsWith("-" + activeProfile + ".yaml") + || sourceName.endsWith("-" + activeProfile + ".properties"); + private ConfigUtils() { } @@ -205,19 +214,13 @@ public static MultipleSourcesContainer processNamedData(List sourceName.endsWith("-" + activeProfile) - || "default".equals(activeProfile)); - if (includeDefaultProfileData || containsActiveProfile - || containsDataWithProfile(rawData, environment.getActiveProfiles())) { + /* + * In some cases we want to include properties from the default profile + * along with any active profiles In these cases includeDefaultProfileData + * will be true If includeDefaultProfileData is false then we want to make + * sure that we only return properties from any active profiles + */ + if (processSource(includeDefaultProfileData, environment, sourceName, rawData)) { data.putAll(SourceDataEntriesProcessor.processAllEntries(rawData == null ? Map.of() : rawData, environment, includeDefaultProfileData)); } @@ -227,13 +230,34 @@ public static MultipleSourcesContainer processNamedData(List sourceRawData) { + List activeProfiles = Arrays.stream(environment.getActiveProfiles()).toList(); + + boolean emptyActiveProfiles = activeProfiles.isEmpty(); + + boolean profileBasedSourceName = activeProfiles.stream() + .anyMatch(activeProfile -> sourceName.endsWith("-" + activeProfile)); + + boolean defaultProfilePresent = activeProfiles.contains("default"); + + return includeDefaultProfileData || emptyActiveProfiles || profileBasedSourceName || defaultProfilePresent + || rawDataContainsProfileBasedSource(activeProfiles, sourceRawData).getAsBoolean(); + } + /* - * In the case there the data contains yaml or properties files we need to check their - * names to see if they contain any active profiles. + * this one is not inlined into 'processSource' because other filters, that come + * before it, might have already resolved to 'true', so no need to compute it at all. + * + * This method is supposed to answer the question if raw data that a certain source + * (configmap or secret) has entries that are themselves profile based + * yaml/yml/properties. For example: 'account-k8s.yaml' or the like. */ - private static boolean containsDataWithProfile(Map rawData, String[] activeProfiles) { - return rawData.keySet().stream().anyMatch(key -> Arrays.stream(activeProfiles) - .anyMatch(activeProfile -> key.contains("-" + activeProfile) || "default".equals(activeProfile))); + static BooleanSupplier rawDataContainsProfileBasedSource(List activeProfiles, + Map sourceRawData) { + return () -> Optional.ofNullable(sourceRawData).orElse(Map.of()).keySet().stream() + .anyMatch(keyName -> activeProfiles.stream() + .anyMatch(activeProfile -> ENDS_WITH_PROFILE_AND_EXTENSION.test(keyName, activeProfile))); } /** diff --git a/spring-cloud-kubernetes-commons/src/main/java/org/springframework/cloud/kubernetes/commons/config/SourceDataEntriesProcessor.java b/spring-cloud-kubernetes-commons/src/main/java/org/springframework/cloud/kubernetes/commons/config/SourceDataEntriesProcessor.java index a73c8508f2..d54cec8764 100644 --- a/spring-cloud-kubernetes-commons/src/main/java/org/springframework/cloud/kubernetes/commons/config/SourceDataEntriesProcessor.java +++ b/spring-cloud-kubernetes-commons/src/main/java/org/springframework/cloud/kubernetes/commons/config/SourceDataEntriesProcessor.java @@ -24,6 +24,7 @@ import java.util.List; import java.util.Map; import java.util.Set; +import java.util.function.Predicate; import org.apache.commons.logging.Log; import org.apache.commons.logging.LogFactory; @@ -47,6 +48,9 @@ public class SourceDataEntriesProcessor extends MapPropertySource { private static final Log LOG = LogFactory.getLog(SourceDataEntriesProcessor.class); + private static Predicate ENDS_IN_EXTENSION = x -> x.endsWith(".yml") || x.endsWith(".yaml") + || x.endsWith(".properties"); + public SourceDataEntriesProcessor(SourceData sourceData) { super(sourceData.sourceName(), sourceData.sourceData()); } @@ -92,7 +96,7 @@ static List> sorted(Map input, Environ * 3. then plain properties * */ - static List> sorted(Map input, Environment environment, + static List> sorted(Map rawData, Environment environment, boolean includeDefaultProfileData) { record WeightedEntry(Map.Entry entry, int weight) { @@ -124,16 +128,17 @@ record WeightedEntry(Map.Entry entry, int weight) { int current = orderedFileNames.size() - 1; List weightedEntries = new ArrayList<>(); - if (input.entrySet().stream().noneMatch(entry -> entry.getKey().endsWith(".yml") - || entry.getKey().endsWith(".yaml") || entry.getKey().endsWith(".properties"))) { - for (Map.Entry entry : input.entrySet()) { + boolean plainEntriesOnly = rawData.keySet().stream().noneMatch(ENDS_IN_EXTENSION); + + if (plainEntriesOnly) { + for (Map.Entry entry : rawData.entrySet()) { weightedEntries.add(new WeightedEntry(entry, ++current)); } } else { - for (Map.Entry entry : input.entrySet()) { + for (Map.Entry entry : rawData.entrySet()) { String key = entry.getKey(); - if (key.endsWith(".yml") || key.endsWith(".yaml") || key.endsWith(".properties")) { + if (ENDS_IN_EXTENSION.test(key)) { String withoutExtension = key.split("\\.", 2)[0]; int index = orderedFileNames.indexOf(withoutExtension); if (index >= 0) { diff --git a/spring-cloud-kubernetes-commons/src/test/java/org/springframework/cloud/kubernetes/commons/config/ConfigUtilsProcessSourceTests.java b/spring-cloud-kubernetes-commons/src/test/java/org/springframework/cloud/kubernetes/commons/config/ConfigUtilsProcessSourceTests.java new file mode 100644 index 0000000000..90a14ec7d1 --- /dev/null +++ b/spring-cloud-kubernetes-commons/src/test/java/org/springframework/cloud/kubernetes/commons/config/ConfigUtilsProcessSourceTests.java @@ -0,0 +1,481 @@ +/* + * Copyright 2013-2024 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.cloud.kubernetes.commons.config; + +import java.util.LinkedHashSet; +import java.util.List; +import java.util.Map; + +import org.junit.jupiter.api.Assertions; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.ExtendWith; + +import org.springframework.boot.test.system.CapturedOutput; +import org.springframework.boot.test.system.OutputCaptureExtension; +import org.springframework.core.env.Environment; +import org.springframework.mock.env.MockEnvironment; + +/** + * Class that is supposed to test only ConfigUtils::processSource and + * ConfigUtils::processNamedData. + * + * @author wind57 + */ +@ExtendWith(OutputCaptureExtension.class) +class ConfigUtilsProcessSourceTests { + + /** + *
+	 *		- includeDefaultProfileData         = true
+	 *		- emptyActiveProfiles               = does not matter
+	 *		- profileBasedSourceName            = does not matter
+	 * 		- defaultProfilePresent             = does not matter
+	 * 		- rawDataContainsProfileBasedSource = does not matter
+	 * 
+ * + * Since 'includeDefaultProfileData=true', all other arguments are irrelevant and + * method must return 'true'. + */ + @Test + void testProcessSourceOne() { + boolean includeDefaultProfileData = true; + Environment environment = new MockEnvironment(); + String sourceName = "account"; + Map sourceRawData = Map.of(); + + boolean result = ConfigUtils.processSource(includeDefaultProfileData, environment, sourceName, sourceRawData); + Assertions.assertTrue(result); + } + + /** + * this case is not very "interesting" because 'includeDefaultProfileData = true' + * which denotes a request not from config server; and such cases are tested in + * various other tests, before we fixed: + * https://github.com/spring-cloud/spring-cloud-kubernetes/pull/1600 + */ + @Test + void testProcessNamedDataOne() { + List strippedSources = List + .of(new StrippedSourceContainer(Map.of(), "configmap-a", Map.of("one", "1"))); + Environment environment = new MockEnvironment(); + LinkedHashSet sourceNames = new LinkedHashSet<>(List.of("configmap-a")); + String namespace = "namespace-a"; + boolean decode = false; + boolean includeDefaultProfileData = true; + + MultipleSourcesContainer result = ConfigUtils.processNamedData(strippedSources, environment, sourceNames, + namespace, decode, includeDefaultProfileData); + Assertions.assertNotNull(result); + Assertions.assertEquals(result.names().toString(), "[configmap-a]"); + Assertions.assertEquals(result.data(), Map.of("one", "1")); + } + + /** + *
+	 *		- includeDefaultProfileData         = false
+	 * 		- emptyActiveProfiles               = false
+	 *		- profileBasedSourceName            = false
+	 * 		- defaultProfilePresent             = true
+	 * 		- rawDataContainsProfileBasedSource = does not matter
+	 * 
+ * + * Since 'defaultProfilePresent=true', this method must return 'true'. + */ + @Test + void testProcessSourceTwo() { + boolean includeDefaultProfileData = false; + MockEnvironment environment = new MockEnvironment(); + environment.setActiveProfiles("default"); + String sourceName = "account"; + Map sourceRawData = Map.of(); + + boolean result = ConfigUtils.processSource(includeDefaultProfileData, environment, sourceName, sourceRawData); + Assertions.assertTrue(result); + } + + /** + *
+	 *		- request is coming from config server
+	 *		- activeProfile = ['default']
+	 *		- sourceName = 'account'
+	 * 
+ * + * As such the above will generate: + * + *
+	 *     - includeDefaultProfileData         = false
+	 * 	   - emptyActiveProfiles               = false
+	 * 	   - profileBasedSourceName            = false
+	 * 	   - defaultProfilePresent             = true
+	 * 	   - rawDataContainsProfileBasedSource = does not matter
+	 * 
+ * + * In this case, three types of properties will be read from the source: + * + *
+	 *     - all simple properties
+	 *     - all nested ones (yaml/yml/properties themselves) that match "${SOURCE_NAME}.{EXTENSION}"
+	 *       (in our case 'account.properties')
+	 *     - all nested ones (yaml/yml/properties themselves) that match "${SOURCE_NAME}-${ACTIVE_PROFILE}.{EXTENSION}"
+	 *       (in our case 'account-default.properties')
+	 *     - there are strict sorting rules if both of the above are matched
+	 * 
+ */ + @Test + void testProcessNamedDataTwo(CapturedOutput output) { + // @formatter:off + Map sourceRawData = Map.of( + "one", "1", + "two", "2", + "account-default.properties", "five=5", + "account.properties", "one=11\nthree=3", + "account-k8s.properties", "one=22\nfour=4" + ); + // @formatter:on + String sourceName = "account"; + List strippedSources = List + .of(new StrippedSourceContainer(Map.of(), sourceName, sourceRawData)); + MockEnvironment environment = new MockEnvironment().withProperty("spring.application.name", sourceName); + environment.setActiveProfiles("default"); + LinkedHashSet sourceNames = new LinkedHashSet<>(List.of(sourceName)); + String namespace = "namespace-a"; + boolean decode = false; + boolean includeDefaultProfileData = false; + + MultipleSourcesContainer result = ConfigUtils.processNamedData(strippedSources, environment, sourceNames, + namespace, decode, includeDefaultProfileData); + Assertions.assertNotNull(result); + Assertions.assertEquals(result.names().toString(), "[account]"); + + /** + *
+		 * there are some things to see here:
+		 *
+		 * 		1. 'three=3' is present in the result, which means we have read 'account.properties'
+		 *		2. 'five=5' is present in the result, which means we have read 'account-default.properties'
+		 *		3. even if we have read 'account.properties', we have 'one=1' (and not 'one=11'),
+		 *			since simple properties override the ones from yml/yaml/properties
+		 *		4. 'four=4' is not present in the result, because we do not read 'account-k8s.properties',
+		 *			since 'k8s' is not an active profile.
+		 * 
+ */ + Assertions.assertEquals(result.data(), Map.of("one", "1", "two", "2", "three", "3", "five", "5")); + Assertions.assertTrue(output.getOut().contains("entry : account-k8s.properties will be skipped")); + } + + /** + *
+	 *		- includeDefaultProfileData         = false
+	 * 		- emptyActiveProfiles               = false
+	 * 		- profileBasedSourceName            = true
+	 * 		- defaultProfilePresent             = false
+	 * 		- rawDataContainsProfileBasedSource = does not matter
+	 * 
+ * + * Since 'profileBasedSourceName=true', this method must return 'true'. + */ + @Test + void testProcessSourceThree() { + boolean includeDefaultProfileData = false; + MockEnvironment environment = new MockEnvironment(); + environment.setActiveProfiles("default"); + String sourceName = "account-default"; + Map sourceRawData = Map.of(); + + boolean result = ConfigUtils.processSource(includeDefaultProfileData, environment, sourceName, sourceRawData); + Assertions.assertTrue(result); + } + + /** + *
+	 *		- request is coming from config server
+	 *		- activeProfile = ['default']
+	 *		- sourceName = 'account-default'
+	 * 
+ * + * As such the above will generate: + * + *
+	 *     - includeDefaultProfileData         = false
+	 * 	   - emptyActiveProfiles               = false
+	 * 	   - profileBasedSourceName            = true
+	 * 	   - defaultProfilePresent             = does not matter
+	 * 	   - rawDataContainsProfileBasedSource = does not matter
+	 * 
+ * + * In this case, three types of properties will be read from the source: + * + *
+	 *     - all simple properties
+	 *     - all nested ones (yaml/yml/properties themselves) that match "${SOURCE_NAME}.{EXTENSION}"
+	 *       (in our case 'account.properties')
+	 *     - all nested ones (yaml/yml/properties themselves) that match "${SOURCE_NAME}-${ACTIVE_PROFILE}.{EXTENSION}"
+	 *       (in our case 'account-default.properties')
+	 *     - there are strict sorting rules if both of the above are matched
+	 * 
+ */ + @Test + void testProcessNamedDataThree(CapturedOutput output) { + // @formatter:off + Map sourceRawData = Map.of( + "one", "1", + "two", "2", + "account.properties", "one=11\nthree=3", + "account-default.properties", "one=111\nfive=5", + "account-k8s.properties", "one=22\nfour=4" + ); + // @formatter:on + String sourceName = "account-default"; + List strippedSources = List + .of(new StrippedSourceContainer(Map.of(), sourceName, sourceRawData)); + MockEnvironment environment = new MockEnvironment().withProperty("spring.application.name", "account"); + environment.setActiveProfiles("default"); + LinkedHashSet sourceNames = new LinkedHashSet<>(List.of(sourceName)); + String namespace = "namespace-a"; + boolean decode = false; + boolean includeDefaultProfileData = false; + + MultipleSourcesContainer result = ConfigUtils.processNamedData(strippedSources, environment, sourceNames, + namespace, decode, includeDefaultProfileData); + Assertions.assertNotNull(result); + Assertions.assertEquals(result.names().toString(), "[account-default]"); + + /** + *
+		 * there are some things to see here:
+		 *
+		 *		1. 'one=1' is present in the result, which means we have read simple properties.
+		 *		2. 'two-2' is present in the result, which means we have read simple properties.
+		 *		3. even if we have read 'account.properties', we have 'one=1' (and not 'one=11'),
+		 *			since simple properties override the ones from yml/yaml/properties
+		 *		4. 'three=3' is present in the result, which means we have read 'account.properties'.
+		 *		5. 'five=5' is present in the result, which means we have read 'account-default.properties'
+		 *			(but we don't have 'one=111', instead : 'one=1')
+		 *		6. we do not have 'four=4' since we do not read 'account-k8s.properties'
+		 * 
+ */ + Assertions.assertEquals(result.data(), Map.of("one", "1", "two", "2", "three", "3", "five", "5")); + Assertions.assertTrue(output.getOut().contains("entry : account-k8s.properties will be skipped")); + + } + + /** + *
+	 *		- includeDefaultProfileData         = false
+	 * 		- emptyActiveProfiles               = false
+	 *		- profileBasedSourceName            = false
+	 *		- defaultProfilePresent             = false
+	 *		- rawDataContainsProfileBasedSource = false
+	 * 
+ * + */ + @Test + void testProcessSourceFour() { + boolean includeDefaultProfileData = false; + MockEnvironment environment = new MockEnvironment(); + environment.setActiveProfiles("k8s"); + String sourceName = "account"; + Map sourceRawData = Map.of("one", "1"); + + boolean result = ConfigUtils.processSource(includeDefaultProfileData, environment, sourceName, sourceRawData); + Assertions.assertFalse(result); + } + + /** + *
+	 *		- includeDefaultProfileData         = false
+	 * 		- emptyActiveProfiles               = false
+	 * 		- profileBasedSourceName            = false
+	 * 		- defaultProfilePresent             = false
+	 * 		- rawDataContainsProfileBasedSource = true
+	 * 
+ * + */ + @Test + void testProcessSourceFive() { + boolean includeDefaultProfileData = false; + MockEnvironment environment = new MockEnvironment(); + environment.setActiveProfiles("k8s"); + String sourceName = "account"; + Map sourceRawData = Map.of("one", "1", "account-k8s.properties", "one=11"); + + boolean result = ConfigUtils.processSource(includeDefaultProfileData, environment, sourceName, sourceRawData); + Assertions.assertTrue(result); + } + + /** + *
+	 *		- request is coming from config server
+	 *		- activeProfile = ['k8s']
+	 *		- sourceName = 'account'
+	 *	    - rawData inside the source has an entry : 'account-k8s.properties'
+	 * 
+ * + * As such the above will generate: + * + *
+	 *     - includeDefaultProfileData         = false
+	 * 	   - emptyActiveProfiles               = false
+	 * 	   - profileBasedSourceName            = false
+	 * 	   - defaultProfilePresent             = false
+	 * 	   - rawDataContainsProfileBasedSource = true
+	 * 
+ * + * In this case, only one type of source data will be read + * + *
+	 *     - "${SOURCE_NAME}-${ACTIVE_PROFILE}.{EXTENSION}"
+	 * 
+ */ + @Test + void testProcessNamedDataFive(CapturedOutput output) { + // @formatter:off + Map sourceRawData = Map.of( + "one", "1", + "account.properties", "one=11\ntwo=2", + "account-default.properties", "one=111\nthree=3", + "account-k8s.properties", "one=1111\nfour=4" + ); + // @formatter:on + String sourceName = "account"; + List strippedSources = List + .of(new StrippedSourceContainer(Map.of(), sourceName, sourceRawData)); + MockEnvironment environment = new MockEnvironment().withProperty("spring.application.name", sourceName); + environment.setActiveProfiles("k8s"); + LinkedHashSet sourceNames = new LinkedHashSet<>(List.of(sourceName)); + String namespace = "namespace-a"; + boolean decode = false; + boolean includeDefaultProfileData = false; + + MultipleSourcesContainer result = ConfigUtils.processNamedData(strippedSources, environment, sourceNames, + namespace, decode, includeDefaultProfileData); + Assertions.assertNotNull(result); + Assertions.assertEquals(result.names().toString(), "[account]"); + + /** + *
+		 * 		- we only read from 'account-k8s.properties'
+		 * 
+ */ + Assertions.assertEquals(result.data(), Map.of("one", "1111", "four", "4")); + Assertions.assertTrue(output.getOut().contains("entry : account.properties will be skipped")); + Assertions.assertTrue(output.getOut().contains("entry : account-default.properties will be skipped")); + + } + + /** + *
+	 *		- includeDefaultProfileData         = false
+	 *		- emptyActiveProfiles               = false
+	 *		- profileBasedSourceName            = true
+	 * 		- defaultProfilePresent             = does not matter
+	 * 		- rawDataContainsProfileBasedSource = does not matter
+	 * 
+ * + */ + @Test + void testProcessSourceSix() { + boolean includeDefaultProfileData = false; + MockEnvironment environment = new MockEnvironment(); + environment.setActiveProfiles("k8s"); + String sourceName = "account-k8s"; + Map sourceRawData = Map.of("one", "1", "account-k8s.properties", "one=11"); + + boolean result = ConfigUtils.processSource(includeDefaultProfileData, environment, sourceName, sourceRawData); + Assertions.assertTrue(result); + } + + /** + *
+	 *		- request is coming from config server
+	 *		- activeProfile = ['k8s']
+	 *		- sourceName = 'account-k8s'
+	 * 
+ * + * As such the above will generate: + * + *
+	 *     - includeDefaultProfileData         = false
+	 * 	   - emptyActiveProfiles               = false
+	 * 	   - profileBasedSourceName            = true
+	 * 	   - defaultProfilePresent             = does not matter
+	 * 	   - rawDataContainsProfileBasedSource = does not matter
+	 * 
+ * + * In this case, a few types of data will be read: + * + *
+	 *     - all simple properties
+	 *     - all nested ones (yaml/yml/properties themselves) that match "${SOURCE_NAME}.{EXTENSION}"
+	 *       (in our case 'account.properties')
+	 *     - all nested ones (yaml/yml/properties themselves) that match "${SOURCE_NAME}-${ACTIVE_PROFILE}.{EXTENSION}"
+	 *       (in our case 'account-k8s.properties')
+	 *     - there are strict sorting rules if both of the above are matched
+	 * 
+ */ + @Test + void testProcessNamedDataSix(CapturedOutput output) { + // @formatter:off + Map sourceRawData = Map.of( + "one", "1", + "two", "2", + "account.properties", "one=11\nthree=3", + "account-default.properties", "one=111\nfour=4", + "account-k8s.properties", "one=1111\nfive=5", + "account-prod.properties", "six=6" + ); + // @formatter:on + String sourceName = "account-k8s"; + List strippedSources = List + .of(new StrippedSourceContainer(Map.of(), sourceName, sourceRawData)); + MockEnvironment environment = new MockEnvironment().withProperty("spring.application.name", "account"); + environment.setActiveProfiles("k8s"); + LinkedHashSet sourceNames = new LinkedHashSet<>(List.of(sourceName)); + String namespace = "namespace-a"; + boolean decode = false; + boolean includeDefaultProfileData = false; + + MultipleSourcesContainer result = ConfigUtils.processNamedData(strippedSources, environment, sourceNames, + namespace, decode, includeDefaultProfileData); + Assertions.assertNotNull(result); + Assertions.assertEquals(result.names().toString(), "[account-k8s]"); + + /** + *
+		 * there are some things to see here:
+		 *
+		 * 		1. 'one=1' is not present in the result, we are not supposed to read simple properties.
+		 *		2. 'two-2' is not present in the result, we are not supposed to read simple properties.
+		 *		3. even if we have read 'account.properties', we have 'one=1' (and not 'one=11'),
+		 *			since simple properties override the ones from yml/yaml/properties
+		 *		4. 'three=3' is not present in the result, we are not supposed to read '${SPRING>APPLICATION.NAME}'
+		 *			properties.
+		 *		5. 'four=4' is not present in the result, which means we have not read 'account-default.properties'
+		 *			(but we don't have 'one=111', instead : 'one=1')
+		 *		6. we do not have 'three=3' since we do not read 'account-default.properties'
+		 *		7. we do not have 'six=6' since we do not read 'account-prod.properties'
+		 *			(because 'prod' is not an active profile)
+		 * 
+ */ + Assertions.assertEquals(result.data(), Map.of("one", "1111", "five", "5")); + Assertions.assertTrue(output.getOut().contains("entry : account-prod.properties will be skipped")); + Assertions.assertTrue(output.getOut().contains("entry : account.properties will be skipped")); + Assertions.assertTrue(output.getOut().contains("entry : account-default.properties will be skipped")); + + } + +} diff --git a/spring-cloud-kubernetes-commons/src/test/java/org/springframework/cloud/kubernetes/commons/config/ConfigUtilsRawDataContainsProfileBasedSourceTests.java b/spring-cloud-kubernetes-commons/src/test/java/org/springframework/cloud/kubernetes/commons/config/ConfigUtilsRawDataContainsProfileBasedSourceTests.java new file mode 100644 index 0000000000..4458864b06 --- /dev/null +++ b/spring-cloud-kubernetes-commons/src/test/java/org/springframework/cloud/kubernetes/commons/config/ConfigUtilsRawDataContainsProfileBasedSourceTests.java @@ -0,0 +1,95 @@ +/* + * Copyright 2013-2024 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.cloud.kubernetes.commons.config; + +import java.util.List; +import java.util.Map; + +import org.junit.jupiter.api.Assertions; +import org.junit.jupiter.api.Test; + +/** + * Class that is supposed to test only ConfigUtils::rawDataContainsProfileBasedSource + * + * @author wind57 + */ +class ConfigUtilsRawDataContainsProfileBasedSourceTests { + + @Test + void nullSourceRawData() { + List activeProfiles = List.of(); + Map rawData = null; + + boolean result = ConfigUtils.rawDataContainsProfileBasedSource(activeProfiles, rawData).getAsBoolean(); + Assertions.assertFalse(result); + } + + @Test + void sourceRawDataPresentEmptyActiveProfiles() { + List activeProfiles = List.of(); + Map rawData = Map.of("account-k8s.yaml", "value"); + + boolean result = ConfigUtils.rawDataContainsProfileBasedSource(activeProfiles, rawData).getAsBoolean(); + Assertions.assertFalse(result); + } + + @Test + void plainValuesOnly() { + List activeProfiles = List.of("k8s"); + Map rawData = Map.of("account", "value"); + + boolean result = ConfigUtils.rawDataContainsProfileBasedSource(activeProfiles, rawData).getAsBoolean(); + Assertions.assertFalse(result); + } + + @Test + void noMatchInActiveProfiles() { + List activeProfiles = List.of("k8s"); + Map rawData = Map.of("account-dev.yml", "value"); + + boolean result = ConfigUtils.rawDataContainsProfileBasedSource(activeProfiles, rawData).getAsBoolean(); + Assertions.assertFalse(result); + } + + @Test + void matchInActiveProfilesWithYml() { + List activeProfiles = List.of("dev"); + Map rawData = Map.of("account-dev.yml", "value"); + + boolean result = ConfigUtils.rawDataContainsProfileBasedSource(activeProfiles, rawData).getAsBoolean(); + Assertions.assertTrue(result); + } + + @Test + void matchInActiveProfilesWithYaml() { + List activeProfiles = List.of("dev", "k8s"); + Map rawData = Map.of("account-dev.yaml", "value"); + + boolean result = ConfigUtils.rawDataContainsProfileBasedSource(activeProfiles, rawData).getAsBoolean(); + Assertions.assertTrue(result); + } + + @Test + void matchInActiveProfilesWithProperties() { + List activeProfiles = List.of("dev", "k8s"); + Map rawData = Map.of("account-dev.properties", "value"); + + boolean result = ConfigUtils.rawDataContainsProfileBasedSource(activeProfiles, rawData).getAsBoolean(); + Assertions.assertTrue(result); + } + +} diff --git a/spring-cloud-kubernetes-fabric8-loadbalancer/src/main/java/org/springframework/cloud/kubernetes/fabric8/loadbalancer/Fabric8ServiceInstanceMapper.java b/spring-cloud-kubernetes-fabric8-loadbalancer/src/main/java/org/springframework/cloud/kubernetes/fabric8/loadbalancer/Fabric8ServiceInstanceMapper.java index 99de7559dd..8f2692070d 100644 --- a/spring-cloud-kubernetes-fabric8-loadbalancer/src/main/java/org/springframework/cloud/kubernetes/fabric8/loadbalancer/Fabric8ServiceInstanceMapper.java +++ b/spring-cloud-kubernetes-fabric8-loadbalancer/src/main/java/org/springframework/cloud/kubernetes/fabric8/loadbalancer/Fabric8ServiceInstanceMapper.java @@ -18,12 +18,13 @@ import java.util.List; import java.util.Map; +import java.util.Objects; import java.util.Optional; import io.fabric8.kubernetes.api.model.ObjectMeta; import io.fabric8.kubernetes.api.model.Service; import io.fabric8.kubernetes.api.model.ServicePort; -import io.fabric8.kubernetes.client.utils.Utils; +import org.apache.commons.logging.LogFactory; import org.springframework.cloud.kubernetes.commons.discovery.DefaultKubernetesServiceInstance; import org.springframework.cloud.kubernetes.commons.discovery.DiscoveryClientUtils; @@ -35,6 +36,8 @@ import org.springframework.cloud.kubernetes.commons.loadbalancer.KubernetesLoadBalancerProperties; import org.springframework.cloud.kubernetes.commons.loadbalancer.KubernetesServiceInstanceMapper; import org.springframework.cloud.kubernetes.fabric8.Fabric8Utils; +import org.springframework.core.log.LogAccessor; +import org.springframework.util.StringUtils; import static org.springframework.cloud.kubernetes.commons.discovery.ServicePortSecureResolver.Input; @@ -45,6 +48,10 @@ */ public class Fabric8ServiceInstanceMapper implements KubernetesServiceInstanceMapper { + private static final String PORT_NAME_PROPERTY = "'spring.cloud.kubernetes.loadbalancer.portName'"; + + private static final LogAccessor LOG = new LogAccessor(LogFactory.getLog(Fabric8ServiceInstanceMapper.class)); + /** * empty on purpose, load balancer implementation does not need them. */ @@ -65,28 +72,48 @@ public class Fabric8ServiceInstanceMapper implements KubernetesServiceInstanceMa @Override public KubernetesServiceInstance map(Service service) { - ObjectMeta meta = service.getMetadata(); + ObjectMeta metadata = service.getMetadata(); List ports = service.getSpec().getPorts(); - ServicePort port = null; + ServicePort port; + + if (ports.isEmpty()) { + LOG.warn(() -> "service : " + metadata.getName() + " does not have any ServicePort(s)," + + " will not consider it for load balancing"); + return null; + } + if (ports.size() == 1) { + LOG.debug(() -> "single ServicePort found, will use it as-is " + "(without checking " + PORT_NAME_PROPERTY + + ")"); port = ports.get(0); } - else if (ports.size() > 1 && Utils.isNotNullOrEmpty(properties.getPortName())) { - Optional optPort = ports.stream().filter(it -> properties.getPortName().endsWith(it.getName())) - .findAny(); - if (optPort.isPresent()) { - port = optPort.get(); + else { + String portNameFromProperties = properties.getPortName(); + if (StringUtils.hasText(portNameFromProperties)) { + Optional optionalPort = ports.stream() + .filter(x -> Objects.equals(x.getName(), portNameFromProperties)).findAny(); + if (optionalPort.isPresent()) { + LOG.debug(() -> "found port name that matches : " + portNameFromProperties); + port = optionalPort.get(); + } + else { + logWarning(portNameFromProperties); + port = ports.get(0); + } + } + else { + LOG.warn(() -> PORT_NAME_PROPERTY + " is not set, as such will not consider service with name : " + + metadata.getName()); + return null; } } - if (port == null) { - return null; - } + String host = KubernetesServiceInstanceMapper.createHost(service.getMetadata().getName(), service.getMetadata().getNamespace(), properties.getClusterDomain()); boolean secure = secure(port, service); - return new DefaultKubernetesServiceInstance(meta.getUid(), meta.getName(), host, port.getPort(), + return new DefaultKubernetesServiceInstance(metadata.getUid(), metadata.getName(), host, port.getPort(), serviceMetadata(service), secure); } @@ -102,4 +129,9 @@ boolean secure(ServicePort port, Service service) { return resolver.resolve(input); } + private void logWarning(String portNameFromProperties) { + LOG.warn(() -> "Did not find a port name that is equal to the value " + portNameFromProperties); + LOG.warn(() -> "Will return 'first' port found, which is non-deterministic"); + } + } diff --git a/spring-cloud-kubernetes-fabric8-loadbalancer/src/test/java/org/springframework/cloud/kubernetes/fabric8/loadbalancer/Fabric8ServiceInstanceMapperTests.java b/spring-cloud-kubernetes-fabric8-loadbalancer/src/test/java/org/springframework/cloud/kubernetes/fabric8/loadbalancer/Fabric8ServiceInstanceMapperTests.java index 06015a01ba..04ea207dfc 100644 --- a/spring-cloud-kubernetes-fabric8-loadbalancer/src/test/java/org/springframework/cloud/kubernetes/fabric8/loadbalancer/Fabric8ServiceInstanceMapperTests.java +++ b/spring-cloud-kubernetes-fabric8-loadbalancer/src/test/java/org/springframework/cloud/kubernetes/fabric8/loadbalancer/Fabric8ServiceInstanceMapperTests.java @@ -28,11 +28,15 @@ import io.fabric8.kubernetes.api.model.ServicePortBuilder; import org.junit.jupiter.api.Assertions; import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.ExtendWith; +import org.springframework.boot.test.system.CapturedOutput; +import org.springframework.boot.test.system.OutputCaptureExtension; import org.springframework.cloud.kubernetes.commons.discovery.KubernetesDiscoveryProperties; import org.springframework.cloud.kubernetes.commons.discovery.KubernetesServiceInstance; import org.springframework.cloud.kubernetes.commons.loadbalancer.KubernetesLoadBalancerProperties; +@ExtendWith(OutputCaptureExtension.class) class Fabric8ServiceInstanceMapperTests { @Test @@ -134,6 +138,159 @@ void serviceMetadataTest() { Assertions.assertEquals(result.get("two"), "2"); } + /** + *
+	 *     service has no ServicePorts
+	 * 
+ */ + @Test + void testMapEmptyPorts(CapturedOutput output) { + KubernetesLoadBalancerProperties loadBalancerProperties = new KubernetesLoadBalancerProperties(); + KubernetesDiscoveryProperties discoveryProperties = new KubernetesDiscoveryProperties(true, false, Set.of(), + true, 60, false, null, Set.of(), Map.of(), null, KubernetesDiscoveryProperties.Metadata.DEFAULT, 0, + true); + + List ports = List.of(); + Service service = buildService("test", "test-namespace", "abc", ports, Map.of(), Map.of()); + KubernetesServiceInstance result = new Fabric8ServiceInstanceMapper(loadBalancerProperties, discoveryProperties) + .map(service); + + Assertions.assertNull(result); + Assertions.assertTrue(output.getOut() + .contains("service : test does not have any ServicePort(s), will not consider it for load balancing")); + + } + + /** + *
+	 *     service has a single ServicePort, and its name matches
+	 *     'spring.cloud.kubernetes.loadbalancer.portName'
+	 * 
+ */ + @Test + void testSinglePortsMatchesProperty(CapturedOutput output) { + KubernetesLoadBalancerProperties loadBalancerProperties = new KubernetesLoadBalancerProperties(); + loadBalancerProperties.setPortName("my-port-name"); + KubernetesDiscoveryProperties discoveryProperties = new KubernetesDiscoveryProperties(true, false, Set.of(), + true, 60, false, null, Set.of(), Map.of(), null, KubernetesDiscoveryProperties.Metadata.DEFAULT, 0, + true); + + List ports = List.of(new ServicePortBuilder().withPort(8080).withName("my-port-name").build()); + Service service = buildService("test", "test-namespace", "abc", ports, Map.of(), Map.of()); + KubernetesServiceInstance result = new Fabric8ServiceInstanceMapper(loadBalancerProperties, discoveryProperties) + .map(service); + + Assertions.assertNotNull(result); + Assertions.assertTrue(output.getOut().contains( + "single ServicePort found, will use it as-is (without checking 'spring.cloud.kubernetes.loadbalancer.portName')")); + + } + + /** + *
+	 *     service has a single ServicePort, and its name does not match
+	 *     'spring.cloud.kubernetes.loadbalancer.portName'.
+	 *
+	 *     in this case, service is still considered, because we don't care
+	 *     about the property name when there is a single service port.
+	 * 
+ */ + @Test + void testSinglePortDoesNotMatchProperty(CapturedOutput output) { + KubernetesLoadBalancerProperties loadBalancerProperties = new KubernetesLoadBalancerProperties(); + loadBalancerProperties.setPortName("my-different-port-name"); + KubernetesDiscoveryProperties discoveryProperties = new KubernetesDiscoveryProperties(true, false, Set.of(), + true, 60, false, null, Set.of(), Map.of(), null, KubernetesDiscoveryProperties.Metadata.DEFAULT, 0, + true); + + List ports = List.of(new ServicePortBuilder().withPort(8080).withName("my-port-name").build()); + Service service = buildService("test", "test-namespace", "abc", ports, Map.of(), Map.of()); + KubernetesServiceInstance result = new Fabric8ServiceInstanceMapper(loadBalancerProperties, discoveryProperties) + .map(service); + + Assertions.assertNotNull(result); + Assertions.assertTrue(output.getOut().contains( + "single ServicePort found, will use it as-is (without checking 'spring.cloud.kubernetes.loadbalancer.portName')")); + + } + + /** + *
+	 *     service has multiple ServicePorts, and 'spring.cloud.kubernetes.loadbalancer.portName' is empty.
+	 *     in this case, service will be skipped.
+	 * 
+ */ + @Test + void testMultiplePortsWithoutPortNameProperty(CapturedOutput output) { + KubernetesLoadBalancerProperties loadBalancerProperties = new KubernetesLoadBalancerProperties(); + loadBalancerProperties.setPortName(""); + KubernetesDiscoveryProperties discoveryProperties = new KubernetesDiscoveryProperties(true, false, Set.of(), + true, 60, false, null, Set.of(), Map.of(), null, KubernetesDiscoveryProperties.Metadata.DEFAULT, 0, + true); + + List ports = List.of(new ServicePortBuilder().withPort(8080).withName("one").build(), + new ServicePortBuilder().withPort(8081).withName("two").build()); + Service service = buildService("test", "test-namespace", "abc", ports, Map.of(), Map.of()); + KubernetesServiceInstance result = new Fabric8ServiceInstanceMapper(loadBalancerProperties, discoveryProperties) + .map(service); + + Assertions.assertNull(result); + Assertions.assertTrue(output.getOut().contains( + "'spring.cloud.kubernetes.loadbalancer.portName' is not set, as such will not consider service with name : test")); + + } + + /** + *
+	 *     service has multiple ServicePorts, and 'spring.cloud.kubernetes.loadbalancer.portName' is empty.
+	 *     in this case, service will be skipped.
+	 * 
+ */ + @Test + void testMultiplePortsWithPortNamePropertyMatch(CapturedOutput output) { + KubernetesLoadBalancerProperties loadBalancerProperties = new KubernetesLoadBalancerProperties(); + loadBalancerProperties.setPortName("one"); + KubernetesDiscoveryProperties discoveryProperties = new KubernetesDiscoveryProperties(true, false, Set.of(), + true, 60, false, null, Set.of(), Map.of(), null, KubernetesDiscoveryProperties.Metadata.DEFAULT, 0, + true); + + List ports = List.of(new ServicePortBuilder().withPort(8080).withName("one").build(), + new ServicePortBuilder().withPort(8081).withName("two").build()); + Service service = buildService("test", "test-namespace", "abc", ports, Map.of(), Map.of()); + KubernetesServiceInstance result = new Fabric8ServiceInstanceMapper(loadBalancerProperties, discoveryProperties) + .map(service); + + Assertions.assertNotNull(result); + Assertions.assertTrue(output.getOut().contains("found port name that matches : one")); + + } + + /** + *
+	 *     service has multiple ServicePorts, and 'spring.cloud.kubernetes.loadbalancer.portName' is empty.
+	 *     in this case, service will be skipped.
+	 * 
+ */ + @Test + void testMultiplePortsWithPortNamePropertyNoMatch(CapturedOutput output) { + KubernetesLoadBalancerProperties loadBalancerProperties = new KubernetesLoadBalancerProperties(); + loadBalancerProperties.setPortName("three"); + KubernetesDiscoveryProperties discoveryProperties = new KubernetesDiscoveryProperties(true, false, Set.of(), + true, 60, false, null, Set.of(), Map.of(), null, KubernetesDiscoveryProperties.Metadata.DEFAULT, 0, + true); + + List ports = List.of(new ServicePortBuilder().withPort(8080).withName("one").build(), + new ServicePortBuilder().withPort(8081).withName("two").build()); + Service service = buildService("test", "test-namespace", "abc", ports, Map.of(), Map.of()); + KubernetesServiceInstance result = new Fabric8ServiceInstanceMapper(loadBalancerProperties, discoveryProperties) + .map(service); + + Assertions.assertNotNull(result); + Assertions.assertTrue(output.getOut().contains("Did not find a port name that is equal to the value three")); + Assertions.assertTrue(output.getOut().contains("Will return 'first' port found, which is non-deterministic")); + Assertions.assertTrue(result.getPort() == 8081 || result.getPort() == 8080); + } + private Service buildService(String name, String namespace, String uid, int port, String portName, Map labels) { ServicePort servicePort = new ServicePortBuilder().withPort(port).withName(portName).build();