Add property to migrate deprecated endoint IDs

Allow legacy actuator endpoint IDs that contain dots to be transparently
migrated to the new format. This update will allow Spring Cloud users
to proactively migrate from endpoints such as `hystrix.stream` to
`hystrixstream`.

Closes gh-18148
This commit is contained in:
Phillip Webb 2019-09-24 17:56:36 -07:00
parent 0a70e33009
commit 323a78c4b9
9 changed files with 102 additions and 8 deletions

View File

@ -62,7 +62,7 @@ abstract class AbstractEndpointCondition extends SpringBootCondition {
Class<? extends Annotation> annotationClass) { Class<? extends Annotation> annotationClass) {
Environment environment = context.getEnvironment(); Environment environment = context.getEnvironment();
AnnotationAttributes attributes = getEndpointAttributes(annotationClass, context, metadata); AnnotationAttributes attributes = getEndpointAttributes(annotationClass, context, metadata);
EndpointId id = EndpointId.of(attributes.getString("id")); EndpointId id = EndpointId.of(environment, attributes.getString("id"));
String key = "management.endpoint." + id.toLowerCaseString() + ".enabled"; String key = "management.endpoint." + id.toLowerCaseString() + ".enabled";
Boolean userDefinedEnabled = environment.getProperty(key, Boolean.class); Boolean userDefinedEnabled = environment.getProperty(key, Boolean.class);
if (userDefinedEnabled != null) { if (userDefinedEnabled != null) {

View File

@ -62,7 +62,7 @@ class OnAvailableEndpointCondition extends AbstractEndpointCondition {
} }
AnnotationAttributes attributes = getEndpointAttributes(ConditionalOnAvailableEndpoint.class, context, AnnotationAttributes attributes = getEndpointAttributes(ConditionalOnAvailableEndpoint.class, context,
metadata); metadata);
EndpointId id = EndpointId.of(attributes.getString("id")); EndpointId id = EndpointId.of(environment, attributes.getString("id"));
Set<ExposureInformation> exposureInformations = getExposureInformation(environment); Set<ExposureInformation> exposureInformations = getExposureInformation(environment);
for (ExposureInformation exposureInformation : exposureInformations) { for (ExposureInformation exposureInformation : exposureInformations) {
if (exposureInformation.isExposed(id)) { if (exposureInformation.isExposed(id)) {

View File

@ -24,6 +24,7 @@ import java.util.regex.Pattern;
import org.apache.commons.logging.Log; import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory; import org.apache.commons.logging.LogFactory;
import org.springframework.core.env.Environment;
import org.springframework.util.Assert; import org.springframework.util.Assert;
/** /**
@ -44,6 +45,8 @@ public final class EndpointId {
private static final Pattern WARNING_PATTERN = Pattern.compile("[\\.\\-]+"); private static final Pattern WARNING_PATTERN = Pattern.compile("[\\.\\-]+");
private static final String MIGRATE_LEGACY_NAMES_PROPRTY = "management.endpoints.migrate-legacy-ids";
private final String value; private final String value;
private final String lowerCaseValue; private final String lowerCaseValue;
@ -112,6 +115,27 @@ public final class EndpointId {
return new EndpointId(value); return new EndpointId(value);
} }
/**
* Factory method to create a new {@link EndpointId} of the specified value. This
* variant will respect the {@code management.endpoints.migrate-legacy-names} property
* if it has been set in the {@link Environment}.
* @param environment the Spring environment
* @param value the endpoint ID value
* @return an {@link EndpointId} instance
* @since 2.2.0
*/
public static EndpointId of(Environment environment, String value) {
Assert.notNull(environment, "Environment must not be null");
return new EndpointId(migrateLegacyId(environment, value));
}
private static String migrateLegacyId(Environment environment, String value) {
if (environment.getProperty(MIGRATE_LEGACY_NAMES_PROPRTY, Boolean.class, false)) {
return value.replace(".", "");
}
return value;
}
/** /**
* Factory method to create a new {@link EndpointId} from a property value. More * Factory method to create a new {@link EndpointId} from a property value. More
* lenient than {@link #of(String)} to allow for common "relaxed" property variants. * lenient than {@link #of(String)} to allow for common "relaxed" property variants.

View File

@ -47,6 +47,7 @@ import org.springframework.core.ResolvableType;
import org.springframework.core.annotation.MergedAnnotation; import org.springframework.core.annotation.MergedAnnotation;
import org.springframework.core.annotation.MergedAnnotations; import org.springframework.core.annotation.MergedAnnotations;
import org.springframework.core.annotation.MergedAnnotations.SearchStrategy; import org.springframework.core.annotation.MergedAnnotations.SearchStrategy;
import org.springframework.core.env.Environment;
import org.springframework.util.Assert; import org.springframework.util.Assert;
import org.springframework.util.CollectionUtils; import org.springframework.util.CollectionUtils;
import org.springframework.util.LinkedMultiValueMap; import org.springframework.util.LinkedMultiValueMap;
@ -140,7 +141,7 @@ public abstract class EndpointDiscoverer<E extends ExposableEndpoint<O>, O exten
private EndpointBean createEndpointBean(String beanName) { private EndpointBean createEndpointBean(String beanName) {
Object bean = this.applicationContext.getBean(beanName); Object bean = this.applicationContext.getBean(beanName);
return new EndpointBean(beanName, bean); return new EndpointBean(this.applicationContext.getEnvironment(), beanName, bean);
} }
private void addExtensionBeans(Collection<EndpointBean> endpointBeans) { private void addExtensionBeans(Collection<EndpointBean> endpointBeans) {
@ -159,7 +160,7 @@ public abstract class EndpointDiscoverer<E extends ExposableEndpoint<O>, O exten
private ExtensionBean createExtensionBean(String beanName) { private ExtensionBean createExtensionBean(String beanName) {
Object bean = this.applicationContext.getBean(beanName); Object bean = this.applicationContext.getBean(beanName);
return new ExtensionBean(beanName, bean); return new ExtensionBean(this.applicationContext.getEnvironment(), beanName, bean);
} }
private void addExtensionBean(EndpointBean endpointBean, ExtensionBean extensionBean) { private void addExtensionBean(EndpointBean endpointBean, ExtensionBean extensionBean) {
@ -401,7 +402,7 @@ public abstract class EndpointDiscoverer<E extends ExposableEndpoint<O>, O exten
private Set<ExtensionBean> extensions = new LinkedHashSet<>(); private Set<ExtensionBean> extensions = new LinkedHashSet<>();
EndpointBean(String beanName, Object bean) { EndpointBean(Environment environment, String beanName, Object bean) {
MergedAnnotation<Endpoint> annotation = MergedAnnotations MergedAnnotation<Endpoint> annotation = MergedAnnotations
.from(bean.getClass(), SearchStrategy.TYPE_HIERARCHY).get(Endpoint.class); .from(bean.getClass(), SearchStrategy.TYPE_HIERARCHY).get(Endpoint.class);
String id = annotation.getString("id"); String id = annotation.getString("id");
@ -409,7 +410,7 @@ public abstract class EndpointDiscoverer<E extends ExposableEndpoint<O>, O exten
() -> "No @Endpoint id attribute specified for " + bean.getClass().getName()); () -> "No @Endpoint id attribute specified for " + bean.getClass().getName());
this.beanName = beanName; this.beanName = beanName;
this.bean = bean; this.bean = bean;
this.id = EndpointId.of(id); this.id = EndpointId.of(environment, id);
this.enabledByDefault = annotation.getBoolean("enableByDefault"); this.enabledByDefault = annotation.getBoolean("enableByDefault");
this.filter = getFilter(this.bean.getClass()); this.filter = getFilter(this.bean.getClass());
} }
@ -462,7 +463,7 @@ public abstract class EndpointDiscoverer<E extends ExposableEndpoint<O>, O exten
private final Class<?> filter; private final Class<?> filter;
ExtensionBean(String beanName, Object bean) { ExtensionBean(Environment environment, String beanName, Object bean) {
this.bean = bean; this.bean = bean;
this.beanName = beanName; this.beanName = beanName;
MergedAnnotation<EndpointExtension> extensionAnnotation = MergedAnnotations MergedAnnotation<EndpointExtension> extensionAnnotation = MergedAnnotations
@ -472,7 +473,7 @@ public abstract class EndpointDiscoverer<E extends ExposableEndpoint<O>, O exten
.from(endpointType, SearchStrategy.TYPE_HIERARCHY).get(Endpoint.class); .from(endpointType, SearchStrategy.TYPE_HIERARCHY).get(Endpoint.class);
Assert.state(endpointAnnotation.isPresent(), Assert.state(endpointAnnotation.isPresent(),
() -> "Extension " + endpointType.getName() + " does not specify an endpoint"); () -> "Extension " + endpointType.getName() + " does not specify an endpoint");
this.endpointId = EndpointId.of(endpointAnnotation.getString("id")); this.endpointId = EndpointId.of(environment, endpointAnnotation.getString("id"));
this.filter = extensionAnnotation.getClass("filter"); this.filter = extensionAnnotation.getClass("filter");
} }

View File

@ -0,0 +1,10 @@
{
"properties": [
{
"name": "management.endpoints.migrate-legacy-ids",
"type": "java.lang.Boolean",
"description": "Whether to transparently migrate legacy endpoint IDs.",
"defaultValue": false
}
]
}

View File

@ -21,6 +21,7 @@ import org.junit.jupiter.api.extension.ExtendWith;
import org.springframework.boot.test.system.CapturedOutput; import org.springframework.boot.test.system.CapturedOutput;
import org.springframework.boot.test.system.OutputCaptureExtension; import org.springframework.boot.test.system.OutputCaptureExtension;
import org.springframework.mock.env.MockEnvironment;
import static org.assertj.core.api.Assertions.assertThat; import static org.assertj.core.api.Assertions.assertThat;
import static org.assertj.core.api.Assertions.assertThatIllegalArgumentException; import static org.assertj.core.api.Assertions.assertThatIllegalArgumentException;
@ -92,6 +93,16 @@ class EndpointIdTests {
.contains("Endpoint ID 'foo-bar' contains invalid characters, please migrate to a valid format"); .contains("Endpoint ID 'foo-bar' contains invalid characters, please migrate to a valid format");
} }
@Test
void ofWhenMigratingLegacyNameRemovesDots(CapturedOutput output) {
EndpointId.resetLoggedWarnings();
MockEnvironment environment = new MockEnvironment();
environment.setProperty("management.endpoints.migrate-legacy-ids", "true");
EndpointId endpointId = EndpointId.of(environment, "foo.bar");
assertThat(endpointId.toString()).isEqualTo("foobar");
assertThat(output).doesNotContain("contains invalid characters");
}
@Test @Test
void equalsAndHashCode() { void equalsAndHashCode() {
EndpointId one = EndpointId.of("foobar1"); EndpointId one = EndpointId.of("foobar1");

View File

@ -0,0 +1,35 @@
/*
* Copyright 2012-2019 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 smoketest.actuator;
import java.util.Collections;
import java.util.Map;
import org.springframework.boot.actuate.endpoint.annotation.Endpoint;
import org.springframework.boot.actuate.endpoint.annotation.ReadOperation;
import org.springframework.stereotype.Component;
@Component
@Endpoint(id = "lega.cy")
public class SampleLegacyEndpoint {
@ReadOperation
public Map<String, String> example() {
return Collections.singletonMap("legacy", "legacy");
}
}

View File

@ -24,3 +24,4 @@ management.endpoint.health.show-details=always
management.endpoint.health.group.ready.include=db,diskSpace management.endpoint.health.group.ready.include=db,diskSpace
management.endpoint.health.group.live.include=example,hello,db management.endpoint.health.group.live.include=example,hello,db
management.endpoint.health.group.live.show-details=never management.endpoint.health.group.live.show-details=never
management.endpoints.migrate-legacy-ids=true

View File

@ -36,6 +36,7 @@ import org.springframework.http.MediaType;
import org.springframework.http.ResponseEntity; import org.springframework.http.ResponseEntity;
import static org.assertj.core.api.Assertions.assertThat; import static org.assertj.core.api.Assertions.assertThat;
import static org.assertj.core.api.Assertions.entry;
/** /**
* Basic integration tests for service demo application. * Basic integration tests for service demo application.
@ -188,6 +189,17 @@ class SampleActuatorApplicationTests {
assertThat(beans).containsKey("spring.datasource-" + DataSourceProperties.class.getName()); assertThat(beans).containsKey("spring.datasource-" + DataSourceProperties.class.getName());
} }
@Test
void testLegacy() {
@SuppressWarnings("rawtypes")
ResponseEntity<Map> entity = this.restTemplate.withBasicAuth("user", getPassword())
.getForEntity("/actuator/legacy", Map.class);
assertThat(entity.getStatusCode()).isEqualTo(HttpStatus.OK);
@SuppressWarnings("unchecked")
Map<String, Object> body = entity.getBody();
assertThat(body).contains(entry("legacy", "legacy"));
}
private String getPassword() { private String getPassword() {
return "password"; return "password";
} }