From 4899076996778efd0a0f49e6446ed8e9c4362798 Mon Sep 17 00:00:00 2001 From: Tim Quinn Date: Thu, 7 Nov 2024 16:38:33 -0600 Subject: [PATCH] 3.x: Support built-in health check config at `health.checks` (retaining now-deprecated `helidon.health` for compatibility) (#9337) * Add support for built-in health check config at health.checks while maintaining bw compat at helidon.health Signed-off-by: Tim Quinn --- .../helidon/config/mp/DeprecatedMpConfig.java | 74 +++++++++++++++++++ docs/se/health.adoc | 9 +-- health/health-checks/pom.xml | 4 + .../health/checks/DiskSpaceHealthCheck.java | 69 ++++++++++++----- .../helidon/health/checks/HealthChecks.java | 10 ++- .../health/checks/HeapMemoryHealthCheck.java | 40 +++++++--- .../src/main/java/module-info.java | 3 +- .../io/helidon/health/checks/ConfigTest.java | 24 +++++- .../src/test/resources/testConfig.yaml | 16 +++- microprofile/health/pom.xml | 5 ++ .../microprofile/health/TestConfigPrefix.java | 43 +++++++++++ .../health/TestDeprecatedConfigPrefix.java | 43 +++++++++++ .../microprofile/health/TestUtils.java | 53 +++++++++++++ 13 files changed, 347 insertions(+), 46 deletions(-) create mode 100644 config/config-mp/src/main/java/io/helidon/config/mp/DeprecatedMpConfig.java create mode 100644 microprofile/health/src/test/java/io/helidon/microprofile/health/TestConfigPrefix.java create mode 100644 microprofile/health/src/test/java/io/helidon/microprofile/health/TestDeprecatedConfigPrefix.java create mode 100644 microprofile/health/src/test/java/io/helidon/microprofile/health/TestUtils.java diff --git a/config/config-mp/src/main/java/io/helidon/config/mp/DeprecatedMpConfig.java b/config/config-mp/src/main/java/io/helidon/config/mp/DeprecatedMpConfig.java new file mode 100644 index 00000000000..f20978cb7df --- /dev/null +++ b/config/config-mp/src/main/java/io/helidon/config/mp/DeprecatedMpConfig.java @@ -0,0 +1,74 @@ +/* + * Copyright (c) 2024 Oracle and/or its affiliates. + * + * 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 + * + * http://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 io.helidon.config.mp; + +import java.util.Optional; +import java.util.logging.Logger; + +import org.eclipse.microprofile.config.Config; + +/** + * A utility class to handle MicroProfile configuration properties that should no longer be used. + *

+ * For one major release, the property is retrieved through this class, to warn about the usage. + * In next major release, the deprecated property is removed (as is use of this class). + *

+ * This class is closely patterned after {@code io.helidon.config.DeprecatedConfig} and works whether the MP config object + * is proxied by CDI or not. That other class can be used in conjunction with + * {@link io.helidon.config.mp.MpConfig#toHelidonConfig(org.eclipse.microprofile.config.Config)} for MP + * config objects that are not injected but that does work for MP config objects proxied by CDI. This one does work for + * those use cases. + */ +public final class DeprecatedMpConfig { + private static final Logger LOGGER = Logger.getLogger(DeprecatedMpConfig.class.getName()); + + private DeprecatedMpConfig() { + } + + /** + * Get a value from config, attempting to read both the keys. + * Warning is logged if either the current key is not defined, or both the keys are defined. + * + * @param config configuration instance + * @param type type of the retrieved value + * @param currentKey key that should be used + * @param deprecatedKey key that should not be used + * @param type of the retrieved value + * @return config value of the current key if exists, or the deprecated key if it does not, an empty {@code Optional} + * otherwise + */ + public static Optional getConfigValue(Config config, Class type, String currentKey, String deprecatedKey) { + Optional deprecatedConfig = config.getOptionalValue(deprecatedKey, type); + Optional currentConfig = config.getOptionalValue(currentKey, type); + + if (deprecatedConfig.isPresent()) { + if (currentConfig.isPresent()) { + LOGGER.warning("You are using both a deprecated configuration and a current one. " + + "Deprecated key: \"" + deprecatedKey + "\", " + + "current key: \"" + currentKey + "\", " + + "only the current key will be used, and deprecated will be ignored."); + return currentConfig; + } else { + LOGGER.warning("You are using a deprecated configuration key. " + + "Deprecated key: \"" + deprecatedKey + "\", " + + "current key: \"" + currentKey + "\"."); + return deprecatedConfig; + } + } else { + return currentConfig; + } + } +} diff --git a/docs/se/health.adoc b/docs/se/health.adoc index 8c1b9257c8b..c1aa351eca5 100644 --- a/docs/se/health.adoc +++ b/docs/se/health.adoc @@ -23,6 +23,7 @@ :rootdir: {docdir}/.. include::{rootdir}/includes/se.adoc[] +:built-in-health-check-config-prefix: health.checks == Contents @@ -216,16 +217,14 @@ common health check statuses: |available disk space |`diskSpace` | link:{health-javadoc-base-url}/io/helidon/health/checks/DiskSpaceHealthCheck.html[`DiskSpaceHealthCheck`] -|`helidon.healthCheck.diskSpace.thresholdPercent` + -+ -`helidon.healthCheck.diskSpace.path` +|`{built-in-health-check-config-prefix}.diskSpace.thresholdPercent` + +`{built-in-health-check-config-prefix}.diskSpace.path` | `99.999` + -+ `/` |available heap memory | `heapMemory` | link:{health-javadoc-base-url}/io/helidon/health/checks/HeapMemoryHealthCheck.html[`HeapMemoryHealthCheck`] -|`helidon.healthCheck.heapMemory.thresholdPercent` +|`{built-in-health-check-config-prefix}.heapMemory.thresholdPercent` |`98` |======= diff --git a/health/health-checks/pom.xml b/health/health-checks/pom.xml index 29df9fc767a..f9d37f2fe9f 100644 --- a/health/health-checks/pom.xml +++ b/health/health-checks/pom.xml @@ -49,6 +49,10 @@ io.helidon.config helidon-config + + io.helidon.config + helidon-config-mp + jakarta.enterprise diff --git a/health/health-checks/src/main/java/io/helidon/health/checks/DiskSpaceHealthCheck.java b/health/health-checks/src/main/java/io/helidon/health/checks/DiskSpaceHealthCheck.java index 7194c0521f2..7fecf2a9c50 100644 --- a/health/health-checks/src/main/java/io/helidon/health/checks/DiskSpaceHealthCheck.java +++ b/health/health-checks/src/main/java/io/helidon/health/checks/DiskSpaceHealthCheck.java @@ -16,7 +16,6 @@ package io.helidon.health.checks; -import java.io.File; import java.io.IOException; import java.nio.file.FileStore; import java.nio.file.Files; @@ -24,14 +23,15 @@ import java.nio.file.Paths; import java.util.Formatter; import java.util.Locale; +import java.util.function.Consumer; import io.helidon.config.Config; +import io.helidon.config.mp.DeprecatedMpConfig; import io.helidon.health.HealthCheckException; import io.helidon.health.common.BuiltInHealthCheck; import jakarta.enterprise.context.ApplicationScoped; import jakarta.inject.Inject; -import org.eclipse.microprofile.config.inject.ConfigProperty; import org.eclipse.microprofile.health.HealthCheck; import org.eclipse.microprofile.health.HealthCheckResponse; import org.eclipse.microprofile.health.Liveness; @@ -43,7 +43,8 @@ *

* By default, this health check has a threshold of 100%, meaning that it will never fail the threshold check. * Also, by default, it will check the root path {@code /}. These defaults can be modified using the - * {@value CONFIG_KEY_PATH} property (default {@value DEFAULT_PATH}), and the {@value CONFIG_KEY_THRESHOLD_PERCENT} + * {@value CURRENT_CONFIG_KEY_PATH} property (default {@value DEFAULT_PATH}), and the + * {@value CURRENT_CONFIG_KEY_THRESHOLD_PERCENT} * property (default {@value DEFAULT_THRESHOLD}, virtually 100). The threshold should be set to a percent, such as 50 for 50% or * 99 for 99%. If disk usage * exceeds this threshold, then the health check will fail. @@ -69,7 +70,7 @@ public class DiskSpaceHealthCheck implements HealthCheck { * directory as application path), use * {@link io.helidon.health.checks.DiskSpaceHealthCheck.Builder#path(java.nio.file.Path)}. * When running within a MicroProfile server, you can configure path using a configuration key - * {@value #CONFIG_KEY_PATH} + * {@value #CURRENT_CONFIG_KEY_PATH} * Defaults to {@value} */ public static final String DEFAULT_PATH = "."; @@ -82,19 +83,34 @@ public class DiskSpaceHealthCheck implements HealthCheck { static final String CONFIG_KEY_DISKSPACE_PREFIX = "diskSpace"; static final String CONFIG_KEY_PATH_SUFFIX = "path"; - static final String CONFIG_KEY_THRESHOLD_PERCENT_SUFFIX = "thresholdPercent"; - /** * Full configuration key for path, when configured through MicroProfile config. + * + * @deprecated The value will change to {@value CURRENT_CONFIG_KEY_PATH} in a future release */ - public static final String CONFIG_KEY_PATH = HealthChecks.CONFIG_KEY_HEALTH_PREFIX + @Deprecated(since = "3.2.11") + public static final String CONFIG_KEY_PATH = HealthChecks.DEPRECATED_CONFIG_KEY_BUILT_IN_HEALTH_CHECKS_PREFIX + "." + CONFIG_KEY_DISKSPACE_PREFIX + "." + CONFIG_KEY_PATH_SUFFIX; - + static final String CONFIG_KEY_THRESHOLD_PERCENT_SUFFIX = "thresholdPercent"; /** * Full configuration key for threshold percent, when configured through Microprofile config. + * + * @deprecated The value will change to {@value CURRENT_CONFIG_KEY_THRESHOLD_PERCENT} in future release */ - public static final String CONFIG_KEY_THRESHOLD_PERCENT = HealthChecks.CONFIG_KEY_HEALTH_PREFIX + @Deprecated(since = "3.2.11") + public static final String CONFIG_KEY_THRESHOLD_PERCENT = HealthChecks.DEPRECATED_CONFIG_KEY_BUILT_IN_HEALTH_CHECKS_PREFIX + + "." + CONFIG_KEY_DISKSPACE_PREFIX + + "." + CONFIG_KEY_THRESHOLD_PERCENT_SUFFIX; + + // The following two constants are used in the Javadoc to nudge users toward using the config key prefix "health.checks" + // rather than the obsolete "helidon.health". Because the original public constants above have always referred to the + // now-deprecated config prefixes, those values are unchanged to preserve backward compatibility. + private static final String CURRENT_CONFIG_KEY_PATH = HealthChecks.CONFIG_KEY_BUILT_IN_HEALTH_CHECKS_PREFIX + + "." + CONFIG_KEY_DISKSPACE_PREFIX + + "." + CONFIG_KEY_PATH_SUFFIX; + + private static final String CURRENT_CONFIG_KEY_THRESHOLD_PERCENT = HealthChecks.CONFIG_KEY_BUILT_IN_HEALTH_CHECKS_PREFIX + "." + CONFIG_KEY_DISKSPACE_PREFIX + "." + CONFIG_KEY_THRESHOLD_PERCENT_SUFFIX; @@ -114,16 +130,8 @@ public class DiskSpaceHealthCheck implements HealthCheck { } @Inject - DiskSpaceHealthCheck( - @ConfigProperty(name = CONFIG_KEY_PATH, defaultValue = DEFAULT_PATH) File path, - @ConfigProperty(name = CONFIG_KEY_THRESHOLD_PERCENT, defaultValue = "99.999") double thresholdPercent - ) { - try { - this.fileStore = Files.getFileStore(path.toPath()); - } catch (IOException e) { - throw new HealthCheckException("Failed to obtain file store for path " + path.getAbsolutePath(), e); - } - this.thresholdPercent = thresholdPercent; + DiskSpaceHealthCheck(org.eclipse.microprofile.config.Config mpConfig) { + this(builder().update(applyConfig(mpConfig))); } private DiskSpaceHealthCheck(Builder builder) { @@ -174,6 +182,29 @@ public static DiskSpaceHealthCheck create() { return builder().build(); } + private static Consumer applyConfig(org.eclipse.microprofile.config.Config mpConfig) { + return builder -> { + DeprecatedMpConfig.getConfigValue(mpConfig, + Path.class, + configKey(HealthChecks.CONFIG_KEY_BUILT_IN_HEALTH_CHECKS_PREFIX, + CONFIG_KEY_PATH_SUFFIX), + configKey(HealthChecks.DEPRECATED_CONFIG_KEY_BUILT_IN_HEALTH_CHECKS_PREFIX, + CONFIG_KEY_PATH_SUFFIX)) + .ifPresent(builder::path); + DeprecatedMpConfig.getConfigValue(mpConfig, + Double.class, + configKey(HealthChecks.CONFIG_KEY_BUILT_IN_HEALTH_CHECKS_PREFIX, + CONFIG_KEY_THRESHOLD_PERCENT_SUFFIX), + configKey(HealthChecks.DEPRECATED_CONFIG_KEY_BUILT_IN_HEALTH_CHECKS_PREFIX, + CONFIG_KEY_THRESHOLD_PERCENT_SUFFIX)) + .ifPresent(builder::thresholdPercent); + }; + } + + private static String configKey(String prefix, String suffix) { + return prefix + "." + CONFIG_KEY_DISKSPACE_PREFIX + "." + suffix; + } + @Override public HealthCheckResponse call() { long diskFreeInBytes; diff --git a/health/health-checks/src/main/java/io/helidon/health/checks/HealthChecks.java b/health/health-checks/src/main/java/io/helidon/health/checks/HealthChecks.java index 87493c2fc14..d9e99bb9684 100644 --- a/health/health-checks/src/main/java/io/helidon/health/checks/HealthChecks.java +++ b/health/health-checks/src/main/java/io/helidon/health/checks/HealthChecks.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2018, 2021 Oracle and/or its affiliates. + * Copyright (c) 2018, 2024 Oracle and/or its affiliates. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -31,7 +31,8 @@ */ public final class HealthChecks { - static final String CONFIG_KEY_HEALTH_PREFIX = "helidon.health"; + static final String CONFIG_KEY_BUILT_IN_HEALTH_CHECKS_PREFIX = "health.checks"; + static final String DEPRECATED_CONFIG_KEY_BUILT_IN_HEALTH_CHECKS_PREFIX = "helidon.health"; private HealthChecks() { } @@ -114,9 +115,10 @@ public static HealthCheck[] healthChecks() { } /** - * Built-in health checks, set up using "helidon.health" configuration. + * Built-in health checks, set up using configuration at {@value CONFIG_KEY_BUILT_IN_HEALTH_CHECKS_PREFIX} or the deprecated + * {@value DEPRECATED_CONFIG_KEY_BUILT_IN_HEALTH_CHECKS_PREFIX}. * - * @param config configuration rooted at "helidon.health" + * @param config configuration at the node containing health checks config * @return built-in health checks, set up using the provided configuration * @see io.helidon.health.HealthSupport.Builder#addLiveness(org.eclipse.microprofile.health.HealthCheck...) */ diff --git a/health/health-checks/src/main/java/io/helidon/health/checks/HeapMemoryHealthCheck.java b/health/health-checks/src/main/java/io/helidon/health/checks/HeapMemoryHealthCheck.java index dedecaafa60..603dc4295a7 100644 --- a/health/health-checks/src/main/java/io/helidon/health/checks/HeapMemoryHealthCheck.java +++ b/health/health-checks/src/main/java/io/helidon/health/checks/HeapMemoryHealthCheck.java @@ -19,12 +19,11 @@ import java.util.Formatter; import java.util.Locale; -import io.helidon.config.Config; +import io.helidon.config.mp.DeprecatedMpConfig; import io.helidon.health.common.BuiltInHealthCheck; import jakarta.enterprise.context.ApplicationScoped; import jakarta.inject.Inject; -import org.eclipse.microprofile.config.inject.ConfigProperty; import org.eclipse.microprofile.health.HealthCheck; import org.eclipse.microprofile.health.HealthCheckResponse; import org.eclipse.microprofile.health.Liveness; @@ -36,7 +35,7 @@ * By default, this health check has a threshold of {@value DEFAULT_THRESHOLD} ({@value DEFAULT_THRESHOLD}%). * If heap usage exceeds this level, then the server * is considered to be unhealthy. This default can be modified using the - * {@value CONFIG_KEY_THRESHOLD_PERCENT} property. The threshold should be set as a percent, such as + * {@value CURRENT_CONFIG_KEY_THRESHOLD_PERCENT} property. The threshold should be set as a percent, such as * 50 for 50% or 99 for 99%. *

*

@@ -62,8 +61,18 @@ public class HeapMemoryHealthCheck implements HealthCheck { /** * Config property key for heap memory threshold. + * + * @deprecated The value will change to {@value #CURRENT_CONFIG_KEY_THRESHOLD_PERCENT} in a future release */ - public static final String CONFIG_KEY_THRESHOLD_PERCENT = HealthChecks.CONFIG_KEY_HEALTH_PREFIX + @Deprecated(since = "3.2.11") + public static final String CONFIG_KEY_THRESHOLD_PERCENT = HealthChecks.DEPRECATED_CONFIG_KEY_BUILT_IN_HEALTH_CHECKS_PREFIX + + "." + CONFIG_KEY_HEAP_PREFIX + + "." + CONFIG_KEY_THRESHOLD_PERCENT_SUFFIX; + + // The following constant is used in the Javadoc to nudge users toward using the current config key prefix "health.checks" + // rather than the deprecated "helidon.health". The public constant above has always used the now-deprecated prefix so that + // value is unchanged to preserve backward compatibility. + private static final String CURRENT_CONFIG_KEY_THRESHOLD_PERCENT = HealthChecks.CONFIG_KEY_BUILT_IN_HEALTH_CHECKS_PREFIX + "." + CONFIG_KEY_HEAP_PREFIX + "." + CONFIG_KEY_THRESHOLD_PERCENT_SUFFIX; @@ -72,11 +81,18 @@ public class HeapMemoryHealthCheck implements HealthCheck { // this will be ignored if not within CDI @Inject - HeapMemoryHealthCheck( - Runtime runtime, - @ConfigProperty(name = CONFIG_KEY_THRESHOLD_PERCENT, defaultValue = "98") double threshold) { - this.thresholdPercent = threshold; + HeapMemoryHealthCheck(Runtime runtime, org.eclipse.microprofile.config.Config mpConfig) { + // Cannot use + this(runtime, DeprecatedMpConfig.getConfigValue(mpConfig, + Double.class, + thresholdPercentKey(HealthChecks.CONFIG_KEY_BUILT_IN_HEALTH_CHECKS_PREFIX), + thresholdPercentKey(HealthChecks.DEPRECATED_CONFIG_KEY_BUILT_IN_HEALTH_CHECKS_PREFIX)) + .orElse(DEFAULT_THRESHOLD)); + } + + HeapMemoryHealthCheck(Runtime runtime, double thresholdPercent) { this.rt = runtime; + this.thresholdPercent = thresholdPercent; } private HeapMemoryHealthCheck(Builder builder) { @@ -104,6 +120,12 @@ public static HeapMemoryHealthCheck create() { return builder().build(); } + private static String thresholdPercentKey(String prefix) { + return prefix + "." + + CONFIG_KEY_HEAP_PREFIX + "." + + CONFIG_KEY_THRESHOLD_PERCENT_SUFFIX; + } + @Override public HealthCheckResponse call() { //Formatter ensures that returned delimiter will be always the same @@ -174,7 +196,7 @@ public Builder thresholdPercent(double threshold) { * @param config {@code Config} node for heap memory * @return updated builder instance */ - public Builder config(Config config) { + public Builder config(io.helidon.config.Config config) { config.get(CONFIG_KEY_THRESHOLD_PERCENT_SUFFIX) .asDouble() .ifPresent(this::thresholdPercent); diff --git a/health/health-checks/src/main/java/module-info.java b/health/health-checks/src/main/java/module-info.java index bc7a53253fc..575bc592b33 100644 --- a/health/health-checks/src/main/java/module-info.java +++ b/health/health-checks/src/main/java/module-info.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2018, 2021 Oracle and/or its affiliates. + * Copyright (c) 2018, 2024 Oracle and/or its affiliates. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -26,6 +26,7 @@ requires io.helidon.common; requires io.helidon.config; + requires io.helidon.config.mp; requires io.helidon.health; requires io.helidon.health.common; requires static microprofile.config.api; diff --git a/health/health-checks/src/test/java/io/helidon/health/checks/ConfigTest.java b/health/health-checks/src/test/java/io/helidon/health/checks/ConfigTest.java index ff7d1054ced..39015adb15c 100644 --- a/health/health-checks/src/test/java/io/helidon/health/checks/ConfigTest.java +++ b/health/health-checks/src/test/java/io/helidon/health/checks/ConfigTest.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2021 Oracle and/or its affiliates. + * Copyright (c) 2021, 2024 Oracle and/or its affiliates. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -23,6 +23,7 @@ import io.helidon.common.http.MediaType; import io.helidon.config.Config; import io.helidon.config.ConfigSources; +import io.helidon.config.DeprecatedConfig; import io.helidon.health.HealthSupport; import io.helidon.media.jsonp.JsonpSupport; import io.helidon.webclient.WebClient; @@ -69,7 +70,7 @@ static void shutdownServer(WebServer server) { @Test void bothFail() throws InterruptedException, ExecutionException, TimeoutException { - JsonObject health = runWithConfig("bothFail", Http.Status.SERVICE_UNAVAILABLE_503.code()); + JsonObject health = runWithConfig("bothFailWithDeprecatedPrefix.helidon.health", Http.Status.SERVICE_UNAVAILABLE_503.code()); JsonObject diskSpace = getLivenessCheck(health, "diskSpace"); assertThat("Disk space liveness return data", diskSpace, is(notNullValue())); @@ -83,7 +84,7 @@ void bothFail() throws InterruptedException, ExecutionException, TimeoutExceptio @Test void bothPass() throws InterruptedException, ExecutionException, TimeoutException { - JsonObject health = runWithConfig("bothPass", Http.Status.OK_200.code()); + JsonObject health = runWithConfig("bothPassWithDeprecatedPrefix.helidon.health", Http.Status.OK_200.code()); JsonObject diskSpace = getLivenessCheck(health, "diskSpace"); assertThat("Disk space liveness return data", diskSpace, is(notNullValue())); @@ -95,10 +96,25 @@ void bothPass() throws InterruptedException, ExecutionException, TimeoutExceptio assertThat("Heap memory liveness check status", heapMemory.getString("status"), is("UP")); } + @Test + void testWithoutHelidonPrefix() throws ExecutionException, InterruptedException, TimeoutException { + JsonObject health = runWithConfig("bothFailWithoutDeprecatedHelidonPrefix.health.checks", + Http.Status.SERVICE_UNAVAILABLE_503.code()); + JsonObject diskSpace = getLivenessCheck(health, "diskSpace"); + + assertThat("Disk space liveness return data", diskSpace, is(notNullValue())); + assertThat("Disk space liveness check status", diskSpace.getString("status"), is("DOWN")); + + JsonObject heapMemory = getLivenessCheck(health, "heapMemory"); + + assertThat("Heap memory liveness return data", heapMemory, is(notNullValue())); + assertThat("Heap memory liveness check status", heapMemory.getString("status"), is("DOWN")); + } + private JsonObject runWithConfig(String configKey, int expectedStatus) throws InterruptedException, ExecutionException, TimeoutException { HealthSupport healthSupport = HealthSupport.builder() - .addLiveness(HealthChecks.healthChecks(testConfig.get(configKey + ".helidon.health"))) + .addLiveness(HealthChecks.healthChecks(testConfig.get(configKey))) .build(); WebServer webServer = null; try { diff --git a/health/health-checks/src/test/resources/testConfig.yaml b/health/health-checks/src/test/resources/testConfig.yaml index b32513a91d4..46847147070 100644 --- a/health/health-checks/src/test/resources/testConfig.yaml +++ b/health/health-checks/src/test/resources/testConfig.yaml @@ -1,5 +1,5 @@ # -# Copyright (c) 2021 Oracle and/or its affiliates. +# Copyright (c) 2021, 2024 Oracle and/or its affiliates. # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. @@ -13,17 +13,25 @@ # See the License for the specific language governing permissions and # limitations under the License. # -bothFail: +bothFailWithDeprecatedPrefix: helidon: health: diskSpace: thresholdPercent: 0.0 heapMemory: thresholdPercent: 0.0 -bothPass: +bothPassWithDeprecatedPrefix: helidon: health: diskSpace: thresholdPercent: 98.1 heapMemory: - thresholdPercent: 98.1 \ No newline at end of file + thresholdPercent: 98.1 +# Following are set to zero to avoid the defaults and force a DOWN state the test can easily detect. +bothFailWithoutDeprecatedHelidonPrefix: + health: + checks: + diskSpace: + thresholdPercent: 0.0 + heapMemory: + thresholdPercent: 0.0 \ No newline at end of file diff --git a/microprofile/health/pom.xml b/microprofile/health/pom.xml index 182a9ec38a4..8f2afa5a44f 100644 --- a/microprofile/health/pom.xml +++ b/microprofile/health/pom.xml @@ -60,6 +60,11 @@ io.helidon.service-common helidon-service-common-rest-cdi + + io.helidon.health + helidon-health-checks + test + org.junit.jupiter junit-jupiter-api diff --git a/microprofile/health/src/test/java/io/helidon/microprofile/health/TestConfigPrefix.java b/microprofile/health/src/test/java/io/helidon/microprofile/health/TestConfigPrefix.java new file mode 100644 index 00000000000..5b548729f76 --- /dev/null +++ b/microprofile/health/src/test/java/io/helidon/microprofile/health/TestConfigPrefix.java @@ -0,0 +1,43 @@ +/* + * Copyright (c) 2024 Oracle and/or its affiliates. + * + * 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 + * + * http://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 io.helidon.microprofile.health; + +import io.helidon.microprofile.tests.junit5.AddConfig; +import io.helidon.microprofile.tests.junit5.HelidonTest; + +import jakarta.inject.Inject; +import jakarta.ws.rs.client.WebTarget; +import org.junit.jupiter.api.Test; + +/** + * Tests the config prefix. + *

+ * We specify 0 values to force the statuses to DOWN. The default config settings cause the checks to return UP, so if we + * detect DOWN statuses we know the config has been found and applied correctly. + */ +@HelidonTest +@AddConfig(key = "health.checks.diskSpace.thresholdPercent", value = "0.0") +@AddConfig(key = "health.checks.heapMemory.thresholdPercent", value = "0.0") +class TestConfigPrefix { + + @Inject + private WebTarget webTarget; + + @Test + void testConfig() { + TestUtils.checkForFailure(webTarget); + } +} diff --git a/microprofile/health/src/test/java/io/helidon/microprofile/health/TestDeprecatedConfigPrefix.java b/microprofile/health/src/test/java/io/helidon/microprofile/health/TestDeprecatedConfigPrefix.java new file mode 100644 index 00000000000..9ca5d09a0d3 --- /dev/null +++ b/microprofile/health/src/test/java/io/helidon/microprofile/health/TestDeprecatedConfigPrefix.java @@ -0,0 +1,43 @@ +/* + * Copyright (c) 2024 Oracle and/or its affiliates. + * + * 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 + * + * http://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 io.helidon.microprofile.health; + +import io.helidon.microprofile.tests.junit5.AddConfig; +import io.helidon.microprofile.tests.junit5.HelidonTest; + +import jakarta.inject.Inject; +import jakarta.ws.rs.client.WebTarget; +import org.junit.jupiter.api.Test; + +/** + * Tests the deprecated config prefix. + *

+ * We specify 0 values to force the statuses to DOWN. The default config settings cause the checks to return UP, so if we + * detect DOWN statuses we know the config has been found and applied correctly. + */ +@HelidonTest +@AddConfig(key = "helidon.health.diskSpace.thresholdPercent", value = "0.0") +@AddConfig(key = "helidon.health.heapMemory.thresholdPercent", value = "0.0") +class TestDeprecatedConfigPrefix { + + @Inject + private WebTarget webTarget; + + @Test + void testConfig() { + TestUtils.checkForFailure(webTarget); + } +} diff --git a/microprofile/health/src/test/java/io/helidon/microprofile/health/TestUtils.java b/microprofile/health/src/test/java/io/helidon/microprofile/health/TestUtils.java new file mode 100644 index 00000000000..f46976a335c --- /dev/null +++ b/microprofile/health/src/test/java/io/helidon/microprofile/health/TestUtils.java @@ -0,0 +1,53 @@ +/* + * Copyright (c) 2024 Oracle and/or its affiliates. + * + * 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 + * + * http://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 io.helidon.microprofile.health; + +import io.helidon.common.http.Http; + +import jakarta.json.JsonObject; +import jakarta.json.JsonValue; +import jakarta.ws.rs.client.WebTarget; +import jakarta.ws.rs.core.Response; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.is; +import static org.hamcrest.Matchers.notNullValue; + +class TestUtils { + static JsonObject getLivenessCheck(JsonObject health, String checkName) { + return health.getJsonArray("checks").stream() + .map(JsonValue::asJsonObject) + .filter(jo -> jo.getString("name").equals(checkName)) + .findFirst() + .orElse(null); + } + + static void checkForFailure(WebTarget webTarget) { + Response response = webTarget.path("/health") + .request() + .get(); + + assertThat("Health endpoint status", response.getStatus(), is(Http.Status.SERVICE_UNAVAILABLE_503.code())); + + JsonObject healthJson = response.readEntity(JsonObject.class); + JsonObject diskSpace = TestUtils.getLivenessCheck(healthJson, "diskSpace"); + JsonObject heapMemory = TestUtils.getLivenessCheck(healthJson, "heapMemory"); + assertThat("Disk space liveness return data", diskSpace, is(notNullValue())); + assertThat("Disk space liveness check status", diskSpace.getString("status"), is("DOWN")); + assertThat("Heap memory liveness return data", heapMemory, is(notNullValue())); + assertThat("Heap memory liveness check status", heapMemory.getString("status"), is("DOWN")); + } +}