From 73016ae2fd59df949aaca62f4e5e3b2714859f76 Mon Sep 17 00:00:00 2001 From: Michael Hoennig Date: Thu, 5 Dec 2024 16:48:26 +0100 Subject: [PATCH] implement /actuator/custom endpoint for linked metrics endpoints --- .../config/CustomActuatorEndpoint.java | 42 ++++++++++++++++++ src/main/resources/application.yml | 2 +- .../CustomActuatorEndpointAcceptanceTest.java | 43 +++++++++++++++++++ src/test/resources/application.yml | 2 +- 4 files changed, 87 insertions(+), 2 deletions(-) create mode 100644 src/main/java/net/hostsharing/hsadminng/config/CustomActuatorEndpoint.java create mode 100644 src/test/java/net/hostsharing/hsadminng/config/CustomActuatorEndpointAcceptanceTest.java diff --git a/src/main/java/net/hostsharing/hsadminng/config/CustomActuatorEndpoint.java b/src/main/java/net/hostsharing/hsadminng/config/CustomActuatorEndpoint.java new file mode 100644 index 00000000..4c0a4f5b --- /dev/null +++ b/src/main/java/net/hostsharing/hsadminng/config/CustomActuatorEndpoint.java @@ -0,0 +1,42 @@ +package net.hostsharing.hsadminng.config; + +import lombok.Getter; +import org.springframework.boot.actuate.endpoint.annotation.Endpoint; +import org.springframework.boot.actuate.endpoint.annotation.ReadOperation; +import org.springframework.stereotype.Component; +import org.springframework.web.client.RestTemplate; +import org.springframework.web.servlet.support.ServletUriComponentsBuilder; + +import java.util.List; + +@Component +@Endpoint(id="custom") +public class CustomActuatorEndpoint { + + private final RestTemplate restTemplate = new RestTemplate(); + + @ReadOperation + public String getMetricsLinks() { + final String baseUrl = ServletUriComponentsBuilder.fromCurrentContextPath().build().toUriString(); + final var metricsEndpoint = baseUrl + "/actuator/metrics"; + + final var response = restTemplate.getForObject(metricsEndpoint, ActuatorMetricsEndpointResource.class); + + if (response == null || response.getNames() == null) { + throw new IllegalStateException("no metrics available"); + } + return generateJsonLinksToMetricEndpoints(response, metricsEndpoint); + } + + private static String generateJsonLinksToMetricEndpoints(final ActuatorMetricsEndpointResource response, final String metricsEndpoint) { + final var links = response.getNames().stream() + .map(name -> "\"" + name + "\": \"" + metricsEndpoint + "/" + name + "\"") + .toList(); + return "{\n" + String.join(",\n", links) + "\n}"; + } + + @Getter + private static class ActuatorMetricsEndpointResource { + private List names; + } +} diff --git a/src/main/resources/application.yml b/src/main/resources/application.yml index 27020234..d747b9e2 100644 --- a/src/main/resources/application.yml +++ b/src/main/resources/application.yml @@ -8,7 +8,7 @@ management: endpoints: web: exposure: - include: info, health, metrics + include: info, health, metrics, custom observations: annotations: enabled: true diff --git a/src/test/java/net/hostsharing/hsadminng/config/CustomActuatorEndpointAcceptanceTest.java b/src/test/java/net/hostsharing/hsadminng/config/CustomActuatorEndpointAcceptanceTest.java new file mode 100644 index 00000000..8dabfa6c --- /dev/null +++ b/src/test/java/net/hostsharing/hsadminng/config/CustomActuatorEndpointAcceptanceTest.java @@ -0,0 +1,43 @@ +package net.hostsharing.hsadminng.config; + +import io.restassured.RestAssured; +import net.hostsharing.hsadminng.HsadminNgApplication; +import net.hostsharing.hsadminng.rbac.test.JpaAttempt; +import net.hostsharing.hsadminng.test.DisableSecurityConfig; +import org.junit.jupiter.api.Test; +import org.springframework.boot.test.context.SpringBootTest; +import org.springframework.boot.test.web.server.LocalManagementPort; +import org.springframework.test.context.ActiveProfiles; + +import static net.hostsharing.hsadminng.rbac.test.JsonMatcher.lenientlyEquals; + +@SpringBootTest( + webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT, + classes = { HsadminNgApplication.class, DisableSecurityConfig.class, JpaAttempt.class } +) +@ActiveProfiles("test") +class CustomActuatorEndpointAcceptanceTest { + + @LocalManagementPort + private Integer managementPort; + + @Test + void shouldListMetricLinks() { + RestAssured // @formatter:off + .given() + .port(managementPort) + .when() + .get("http://localhost/actuator/custom") + .then().log().all().assertThat() + .statusCode(200) + .contentType("application/vnd.spring-boot.actuator.v3+json") + .body("", lenientlyEquals(""" + { + "application.ready.time": "http://localhost:%{managementPort}/actuator/metrics/application.ready.time", + "application.started.time": "http://localhost:%{managementPort}/actuator/metrics/application.started.time" + } + """.replace("%{managementPort}", managementPort.toString()))); + // @formatter:on + } + +} diff --git a/src/test/resources/application.yml b/src/test/resources/application.yml index a365daf3..871997c4 100644 --- a/src/test/resources/application.yml +++ b/src/test/resources/application.yml @@ -6,7 +6,7 @@ management: endpoints: web: exposure: - include: info, health, metrics + include: info, health, metrics, custom spring: sql: