Merge branch '2.7.x'

Closes gh-31880
This commit is contained in:
Phillip Webb 2022-07-26 18:16:44 +01:00
commit 94ca5b5b53
2 changed files with 38 additions and 18 deletions

View File

@ -1,5 +1,5 @@
/* /*
* Copyright 2012-2021 the original author or authors. * Copyright 2012-2022 the original author or authors.
* *
* Licensed under the Apache License, Version 2.0 (the "License"); * Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License. * you may not use this file except in compliance with the License.
@ -18,10 +18,10 @@ package org.springframework.boot.actuate.health;
import java.util.Collections; import java.util.Collections;
import java.util.Iterator; import java.util.Iterator;
import java.util.LinkedHashMap;
import java.util.Map; import java.util.Map;
import java.util.Map.Entry; import java.util.Map.Entry;
import java.util.function.Function; import java.util.function.Function;
import java.util.stream.Collectors;
import org.springframework.util.Assert; import org.springframework.util.Assert;
@ -31,34 +31,36 @@ import org.springframework.util.Assert;
* @param <V> the value type * @param <V> the value type
* @param <C> the contributor type * @param <C> the contributor type
* @author Phillip Webb * @author Phillip Webb
* @author Guirong Hu
* @see CompositeHealthContributorMapAdapter * @see CompositeHealthContributorMapAdapter
* @see CompositeReactiveHealthContributorMapAdapter * @see CompositeReactiveHealthContributorMapAdapter
*/ */
abstract class NamedContributorsMapAdapter<V, C> implements NamedContributors<C> { abstract class NamedContributorsMapAdapter<V, C> implements NamedContributors<C> {
private final Map<String, V> map; private final Map<String, C> map;
private final Function<V, ? extends C> valueAdapter;
NamedContributorsMapAdapter(Map<String, V> map, Function<V, ? extends C> valueAdapter) { NamedContributorsMapAdapter(Map<String, V> map, Function<V, ? extends C> valueAdapter) {
Assert.notNull(map, "Map must not be null"); Assert.notNull(map, "Map must not be null");
Assert.notNull(valueAdapter, "ValueAdapter must not be null"); Assert.notNull(valueAdapter, "ValueAdapter must not be null");
map.keySet().forEach(this::validateKey); map.keySet().forEach(this::validateKey);
map.values().stream().map(valueAdapter) this.map = Collections.unmodifiableMap(map.entrySet().stream()
.forEach((value) -> Assert.notNull(value, "Map must not contain null values")); .collect(Collectors.toMap(Entry::getKey, (entry) -> adapt(entry.getValue(), valueAdapter))));
this.map = Collections.unmodifiableMap(new LinkedHashMap<>(map));
this.valueAdapter = valueAdapter;
} }
private void validateKey(String value) { private void validateKey(String value) {
Assert.notNull(value, "Map must not contain null keys"); Assert.notNull(value, "Map must not contain null keys");
Assert.isTrue(!value.contains("/"), "Map keys must not contain a '/'"); Assert.isTrue(!value.contains("/"), "Map keys must not contain a '/'");
}
private C adapt(V value, Function<V, ? extends C> valueAdapter) {
C contributor = (value != null) ? valueAdapter.apply(value) : null;
Assert.notNull(contributor, "Map must not contain null values");
return contributor;
} }
@Override @Override
public Iterator<NamedContributor<C>> iterator() { public Iterator<NamedContributor<C>> iterator() {
Iterator<Entry<String, V>> iterator = this.map.entrySet().iterator(); Iterator<Entry<String, C>> iterator = this.map.entrySet().iterator();
return new Iterator<NamedContributor<C>>() { return new Iterator<NamedContributor<C>>() {
@Override @Override
@ -68,8 +70,8 @@ abstract class NamedContributorsMapAdapter<V, C> implements NamedContributors<C>
@Override @Override
public NamedContributor<C> next() { public NamedContributor<C> next() {
Entry<String, V> entry = iterator.next(); Entry<String, C> entry = iterator.next();
return NamedContributor.of(entry.getKey(), adapt(entry.getValue())); return NamedContributor.of(entry.getKey(), entry.getValue());
} }
}; };
@ -77,11 +79,7 @@ abstract class NamedContributorsMapAdapter<V, C> implements NamedContributors<C>
@Override @Override
public C getContributor(String name) { public C getContributor(String name) {
return adapt(this.map.get(name)); return this.map.get(name);
}
private C adapt(V value) {
return (value != null) ? this.valueAdapter.apply(value) : null;
} }
} }

View File

@ -1,5 +1,5 @@
/* /*
* Copyright 2012-2021 the original author or authors. * Copyright 2012-2022 the original author or authors.
* *
* Licensed under the Apache License, Version 2.0 (the "License"); * Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License. * you may not use this file except in compliance with the License.
@ -20,6 +20,7 @@ import java.util.Collections;
import java.util.Iterator; import java.util.Iterator;
import java.util.LinkedHashMap; import java.util.LinkedHashMap;
import java.util.Map; import java.util.Map;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.function.Function; import java.util.function.Function;
import org.junit.jupiter.api.Test; import org.junit.jupiter.api.Test;
@ -31,6 +32,7 @@ import static org.assertj.core.api.Assertions.assertThatIllegalArgumentException
* Tests for {@link NamedContributorsMapAdapter}. * Tests for {@link NamedContributorsMapAdapter}.
* *
* @author Phillip Webb * @author Phillip Webb
* @author Guirong Hu
*/ */
class NamedContributorsMapAdapterTests { class NamedContributorsMapAdapterTests {
@ -92,6 +94,21 @@ class NamedContributorsMapAdapterTests {
assertThat(adapter.getContributor("two")).isEqualTo("owt"); assertThat(adapter.getContributor("two")).isEqualTo("owt");
} }
@Test
void getContributorCallsAdaptersOnlyOnce() {
Map<String, String> map = new LinkedHashMap<>();
map.put("one", "one");
map.put("two", "two");
int callCount = map.size();
AtomicInteger counter = new AtomicInteger(0);
TestNamedContributorsMapAdapter<String> adapter = new TestNamedContributorsMapAdapter<>(map,
(name) -> count(name, counter));
assertThat(adapter.getContributor("one")).isEqualTo("eno");
assertThat(counter.get()).isEqualTo(callCount);
assertThat(adapter.getContributor("two")).isEqualTo("owt");
assertThat(counter.get()).isEqualTo(callCount);
}
@Test @Test
void getContributorWhenNotInMapReturnsNull() { void getContributorWhenNotInMapReturnsNull() {
TestNamedContributorsMapAdapter<String> adapter = createAdapter(); TestNamedContributorsMapAdapter<String> adapter = createAdapter();
@ -106,6 +123,11 @@ class NamedContributorsMapAdapterTests {
return adapter; return adapter;
} }
private String count(CharSequence charSequence, AtomicInteger counter) {
counter.incrementAndGet();
return reverse(charSequence);
}
private String reverse(CharSequence charSequence) { private String reverse(CharSequence charSequence) {
return new StringBuilder(charSequence).reverse().toString(); return new StringBuilder(charSequence).reverse().toString();
} }