CollectionBinderTests Class — spring-boot Architecture
Architecture documentation for the CollectionBinderTests class in CollectionBinderTests.java from the spring-boot codebase.
Entity Profile
Relationship Graph
Source Code
core/spring-boot/src/test/java/org/springframework/boot/context/properties/bind/CollectionBinderTests.java lines 51–662
class CollectionBinderTests {
private static final Bindable<List<Integer>> INTEGER_LIST = Bindable.listOf(Integer.class);
private static final Bindable<List<String>> STRING_LIST = Bindable.listOf(String.class);
private static final Bindable<Set<String>> STRING_SET = Bindable.setOf(String.class);
private final List<ConfigurationPropertySource> sources = new ArrayList<>();
private Binder binder = new Binder(this.sources);
@Test
void bindToCollectionShouldReturnPopulatedCollection() {
MockConfigurationPropertySource source = new MockConfigurationPropertySource();
source.put("foo[0]", "1");
source.put("foo[1]", "2");
source.put("foo[2]", "3");
this.sources.add(source);
List<Integer> result = this.binder.bind("foo", INTEGER_LIST).get();
assertThat(result).containsExactly(1, 2, 3);
}
@Test
void bindToSetShouldReturnPopulatedCollection() {
MockConfigurationPropertySource source = new MockConfigurationPropertySource();
source.put("foo[0]", "a");
source.put("foo[1]", "b");
source.put("foo[2]", "c");
this.sources.add(source);
Set<String> result = this.binder.bind("foo", STRING_SET).get();
assertThat(result).containsExactly("a", "b", "c");
}
@Test
void bindToCollectionWhenNestedShouldReturnPopulatedCollection() {
MockConfigurationPropertySource source = new MockConfigurationPropertySource();
source.put("foo[0][0]", "1");
source.put("foo[0][1]", "2");
source.put("foo[1][0]", "3");
source.put("foo[1][1]", "4");
this.sources.add(source);
Bindable<List<List<Integer>>> target = Bindable
.of(ResolvableType.forClassWithGenerics(List.class, INTEGER_LIST.getType()));
List<List<Integer>> result = this.binder.bind("foo", target).get();
assertThat(result).hasSize(2);
assertThat(result.get(0)).containsExactly(1, 2);
assertThat(result.get(1)).containsExactly(3, 4);
}
@Test
void bindToCollectionWhenNotInOrderShouldReturnPopulatedCollection() {
MockConfigurationPropertySource source = new MockConfigurationPropertySource();
source.put("foo[1]", "2");
source.put("foo[0]", "1");
source.put("foo[2]", "3");
this.sources.add(source);
List<Integer> result = this.binder.bind("foo", INTEGER_LIST).get();
assertThat(result).containsExactly(1, 2, 3);
}
@Test
void bindToCollectionWhenNonSequentialShouldThrowException() {
MockConfigurationPropertySource source = new MockConfigurationPropertySource();
source.put("foo[0]", "2");
source.put("foo[1]", "1");
source.put("foo[3]", "3");
this.sources.add(source);
assertThatExceptionOfType(BindException.class).isThrownBy(() -> this.binder.bind("foo", INTEGER_LIST))
.satisfies((ex) -> {
Throwable cause = ex.getCause();
assertThat(cause).isNotNull();
Set<ConfigurationProperty> unbound = ((UnboundConfigurationPropertiesException) cause)
.getUnboundProperties();
assertThat(unbound).singleElement().satisfies((property) -> {
assertThat(property.getName()).hasToString("foo[3]");
assertThat(property.getValue()).isEqualTo("3");
});
});
}
@Test
void bindToCollectionWhenNonKnownIndexedChildNotBoundThrowsException() {
// gh-45994
MockConfigurationPropertySource source = new MockConfigurationPropertySource();
source.put("foo[0].first", "Spring");
source.put("foo[0].last", "Boot");
source.put("foo[1].missing", "bad");
this.sources.add(source);
assertThatExceptionOfType(BindException.class)
.isThrownBy(() -> this.binder.bind("foo", Bindable.listOf(Name.class)))
.satisfies((ex) -> {
Throwable cause = ex.getCause();
assertThat(cause).isNotNull();
Set<ConfigurationProperty> unbound = ((UnboundConfigurationPropertiesException) cause)
.getUnboundProperties();
assertThat(unbound).singleElement().satisfies((property) -> {
assertThat(property.getName()).hasToString("foo[1].missing");
assertThat(property.getValue()).isEqualTo("bad");
});
});
}
@Test
void bindToNestedCollectionWhenNonKnownIndexed() {
// gh-46039
MockConfigurationPropertySource source = new MockConfigurationPropertySource();
source.put("foo[0].items[0]", "a");
source.put("foo[0].items[1]", "b");
source.put("foo[0].string", "test");
this.sources.add(source);
List<ExampleCollectionBean> list = this.binder.bind("foo", Bindable.listOf(ExampleCollectionBean.class)).get();
assertThat(list).singleElement().satisfies((bean) -> {
assertThat(bean.getItems()).containsExactly("a", "b", "d");
assertThat(bean.getString()).isEqualTo("test");
});
}
@Test
void bindToNonScalarCollectionWhenNonSequentialShouldThrowException() {
MockConfigurationPropertySource source = new MockConfigurationPropertySource();
source.put("foo[0].value", "1");
source.put("foo[1].value", "2");
source.put("foo[4].value", "4");
this.sources.add(source);
Bindable<List<JavaBean>> target = Bindable.listOf(JavaBean.class);
assertThatExceptionOfType(BindException.class).isThrownBy(() -> this.binder.bind("foo", target))
.satisfies((ex) -> {
Throwable cause = ex.getCause();
assertThat(cause).isNotNull();
Set<ConfigurationProperty> unbound = ((UnboundConfigurationPropertiesException) cause)
.getUnboundProperties();
assertThat(unbound).singleElement().satisfies((property) -> {
assertThat(property.getName()).hasToString("foo[4].value");
assertThat(property.getValue()).isEqualTo("4");
});
});
}
@Test
void bindToCollectionWhenNonIterableShouldReturnPopulatedCollection() {
MockConfigurationPropertySource source = new MockConfigurationPropertySource();
source.put("foo[1]", "2");
source.put("foo[0]", "1");
source.put("foo[2]", "3");
this.sources.add(source.nonIterable());
List<Integer> result = this.binder.bind("foo", INTEGER_LIST).get();
assertThat(result).containsExactly(1, 2, 3);
}
@Test
void bindToCollectionWhenMultipleSourceShouldOnlyUseFirst() {
MockConfigurationPropertySource source1 = new MockConfigurationPropertySource();
source1.put("bar", "baz");
this.sources.add(source1);
MockConfigurationPropertySource source2 = new MockConfigurationPropertySource();
source2.put("foo[0]", "1");
source2.put("foo[1]", "2");
this.sources.add(source2);
MockConfigurationPropertySource source3 = new MockConfigurationPropertySource();
source3.put("foo[0]", "7");
source3.put("foo[1]", "8");
source3.put("foo[2]", "9");
this.sources.add(source3);
List<Integer> result = this.binder.bind("foo", INTEGER_LIST).get();
assertThat(result).containsExactly(1, 2);
}
@Test
void bindToCollectionWhenHasExistingCollectionShouldReplaceAllContents() {
this.sources.add(new MockConfigurationPropertySource("foo[0]", "1"));
List<Integer> existing = new LinkedList<>();
existing.add(1000);
existing.add(1001);
List<Integer> result = this.binder.bind("foo", INTEGER_LIST.withExistingValue(existing)).get();
assertThat(result).isExactlyInstanceOf(LinkedList.class);
assertThat(result).containsExactly(1);
}
@Test
void bindToCollectionWhenHasExistingCollectionButNoValueShouldReturnUnbound() {
this.sources.add(new MockConfigurationPropertySource("faf[0]", "1"));
List<Integer> existing = new LinkedList<>();
existing.add(1000);
BindResult<List<Integer>> result = this.binder.bind("foo", INTEGER_LIST.withExistingValue(existing));
assertThat(result.isBound()).isFalse();
}
@Test
void bindToCollectionShouldRespectCollectionType() {
this.sources.add(new MockConfigurationPropertySource("foo[0]", "1"));
ResolvableType type = ResolvableType.forClassWithGenerics(LinkedList.class, Integer.class);
Object defaultList = this.binder.bind("foo", INTEGER_LIST).get();
Object customList = this.binder.bind("foo", Bindable.of(type)).get();
assertThat(customList).isExactlyInstanceOf(LinkedList.class).isNotInstanceOf(defaultList.getClass());
}
@Test
void bindToCollectionWhenNoValueShouldReturnUnbound() {
this.sources.add(new MockConfigurationPropertySource("faf.bar", "1"));
BindResult<List<Integer>> result = this.binder.bind("foo", INTEGER_LIST);
assertThat(result.isBound()).isFalse();
}
@Test
void bindToCollectionWhenCommaListShouldReturnPopulatedCollection() {
this.sources.add(new MockConfigurationPropertySource("foo", "1,2,3"));
List<Integer> result = this.binder.bind("foo", INTEGER_LIST).get();
assertThat(result).containsExactly(1, 2, 3);
}
@Test
void bindToCollectionWhenCommaListWithPlaceholdersShouldReturnPopulatedCollection() {
StandardEnvironment environment = new StandardEnvironment();
TestPropertySourceUtils.addInlinedPropertiesToEnvironment(environment, "bar=1,2,3");
this.binder = new Binder(this.sources, new PropertySourcesPlaceholdersResolver(environment));
this.sources.add(new MockConfigurationPropertySource("foo", "${bar}"));
List<Integer> result = this.binder.bind("foo", INTEGER_LIST).get();
assertThat(result).containsExactly(1, 2, 3);
}
@Test
void bindToCollectionWhenCommaListAndIndexedShouldOnlyUseFirst() {
MockConfigurationPropertySource source1 = new MockConfigurationPropertySource();
source1.put("foo", "1,2");
this.sources.add(source1);
MockConfigurationPropertySource source2 = new MockConfigurationPropertySource();
source2.put("foo[0]", "2");
source2.put("foo[1]", "3");
List<Integer> result = this.binder.bind("foo", INTEGER_LIST).get();
assertThat(result).containsExactly(1, 2);
}
@Test
void bindToCollectionWhenIndexedAndCommaListShouldOnlyUseFirst() {
MockConfigurationPropertySource source1 = new MockConfigurationPropertySource();
source1.put("foo[0]", "1");
source1.put("foo[1]", "2");
this.sources.add(source1);
MockConfigurationPropertySource source2 = new MockConfigurationPropertySource();
source2.put("foo", "2,3");
List<Integer> result = this.binder.bind("foo", INTEGER_LIST).get();
assertThat(result).containsExactly(1, 2);
}
@Test
void bindToCollectionWhenItemContainsCommasShouldReturnPopulatedCollection() {
MockConfigurationPropertySource source = new MockConfigurationPropertySource();
source.put("foo[0]", "1,2");
source.put("foo[1]", "3");
this.sources.add(source);
List<String> result = this.binder.bind("foo", STRING_LIST).get();
assertThat(result).containsExactly("1,2", "3");
}
@Test
void bindToCollectionWhenEmptyStringShouldReturnEmptyCollection() {
MockConfigurationPropertySource source = new MockConfigurationPropertySource();
source.put("foo", "");
this.sources.add(source);
List<String> result = this.binder.bind("foo", STRING_LIST).get();
assertThat(result).isEmpty();
}
@Test
void bindToNonScalarCollectionShouldReturnPopulatedCollection() {
MockConfigurationPropertySource source = new MockConfigurationPropertySource();
source.put("foo[0].value", "a");
source.put("foo[1].value", "b");
source.put("foo[2].value", "c");
this.sources.add(source);
Bindable<List<JavaBean>> target = Bindable.listOf(JavaBean.class);
List<JavaBean> result = this.binder.bind("foo", target).get();
assertThat(result).hasSize(3);
List<String> values = result.stream().map(JavaBean::getValue).toList();
assertThat(values).containsExactly("a", "b", "c");
}
@Test
void bindToImmutableCollectionShouldReturnPopulatedCollection() {
MockConfigurationPropertySource source = new MockConfigurationPropertySource();
source.put("foo.values", "a,b,c");
this.sources.add(source);
Set<String> result = this.binder.bind("foo.values", STRING_SET.withExistingValue(Collections.emptySet())).get();
assertThat(result).hasSize(3);
}
@Test
void bindToCollectionShouldAlsoCallSetterIfPresent() {
MockConfigurationPropertySource source = new MockConfigurationPropertySource();
source.put("foo.items", "a,b,c");
this.sources.add(source);
ExampleCollectionBean result = this.binder.bind("foo", ExampleCollectionBean.class).get();
assertThat(result.getItems()).hasSize(4);
assertThat(result.getItems()).containsExactly("a", "b", "c", "d");
}
@Test
void bindToCollectionWithNoDefaultConstructor() {
MockConfigurationPropertySource source = new MockConfigurationPropertySource();
source.put("foo.items", "a,b,c,c");
this.sources.add(source);
ExampleCustomNoDefaultConstructorBean result = this.binder
.bind("foo", ExampleCustomNoDefaultConstructorBean.class)
.get();
assertThat(result.getItems()).hasSize(4);
assertThat(result.getItems()).containsExactly("a", "b", "c", "c");
}
@Test
void bindToCollectionWithDefaultConstructor() {
// gh-12322
MockConfigurationPropertySource source = new MockConfigurationPropertySource();
source.put("foo.items", "a,b,c,c");
this.sources.add(source);
ExampleCustomWithDefaultConstructorBean result = this.binder
.bind("foo", ExampleCustomWithDefaultConstructorBean.class)
.get();
assertThat(result.getItems()).hasSize(4);
assertThat(result.getItems()).containsExactly("a", "b", "c", "c");
}
@Test
void bindToListShouldAllowDuplicateValues() {
MockConfigurationPropertySource source = new MockConfigurationPropertySource();
source.put("foo.items", "a,b,c,c");
this.sources.add(source);
ExampleCollectionBean result = this.binder.bind("foo", ExampleCollectionBean.class).get();
assertThat(result.getItems()).hasSize(5);
assertThat(result.getItems()).containsExactly("a", "b", "c", "c", "d");
}
@Test
void bindToSetShouldNotAllowDuplicateValues() {
MockConfigurationPropertySource source = new MockConfigurationPropertySource();
source.put("foo.items-set", "a,b,c,c");
this.sources.add(source);
ExampleCollectionBean result = this.binder.bind("foo", ExampleCollectionBean.class).get();
assertThat(result.getItemsSet()).hasSize(3);
assertThat(result.getItemsSet()).containsExactly("a", "b", "c");
}
@Test
void bindToBeanWithNestedCollectionShouldPopulateCollection() {
MockConfigurationPropertySource source = new MockConfigurationPropertySource();
source.put("foo.value", "one");
source.put("foo.foos[0].value", "two");
source.put("foo.foos[1].value", "three");
this.sources.add(source);
Bindable<BeanWithNestedCollection> target = Bindable.of(BeanWithNestedCollection.class);
BeanWithNestedCollection foo = this.binder.bind("foo", target).get();
assertThat(foo.getValue()).isEqualTo("one");
List<BeanWithNestedCollection> foos = foo.getFoos();
assertThat(foos).isNotNull();
assertThat(foos.get(0).getValue()).isEqualTo("two");
assertThat(foos.get(1).getValue()).isEqualTo("three");
}
@Test
void bindToNestedCollectionWhenEmptyStringShouldReturnEmptyCollection() {
MockConfigurationPropertySource source = new MockConfigurationPropertySource();
source.put("foo.value", "one");
source.put("foo.foos", "");
this.sources.add(source);
Bindable<BeanWithNestedCollection> target = Bindable.of(BeanWithNestedCollection.class);
BeanWithNestedCollection foo = this.binder.bind("foo", target).get();
assertThat(foo.getValue()).isEqualTo("one");
assertThat(foo.getFoos()).isEmpty();
}
@Test
void bindToCollectionShouldUsePropertyEditor() {
// gh-12166
MockConfigurationPropertySource source = new MockConfigurationPropertySource();
source.put("foo[0]", "java.lang.RuntimeException");
source.put("foo[1]", "java.lang.IllegalStateException");
this.sources.add(source);
assertThat(this.binder.bind("foo", Bindable.listOf(Class.class)).get()).containsExactly(RuntimeException.class,
IllegalStateException.class);
}
@Test
void bindToCollectionWhenStringShouldUsePropertyEditor() {
// gh-12166
MockConfigurationPropertySource source = new MockConfigurationPropertySource();
source.put("foo", "java.lang.RuntimeException,java.lang.IllegalStateException");
this.sources.add(source);
assertThat(this.binder.bind("foo", Bindable.listOf(Class.class)).get()).containsExactly(RuntimeException.class,
IllegalStateException.class);
}
@Test
void bindToBeanWithNestedCollectionAndNonIterableSourceShouldNotFail() {
// gh-10702
MockConfigurationPropertySource source = new MockConfigurationPropertySource();
this.sources.add(source.nonIterable());
Bindable<BeanWithNestedCollection> target = Bindable.of(BeanWithNestedCollection.class);
this.binder.bind("foo", target);
}
@Test
void bindToBeanWithClonedArray() {
MockConfigurationPropertySource source = new MockConfigurationPropertySource();
source.put("foo.bar[0]", "hello");
this.sources.add(source);
Bindable<ClonedArrayBean> target = Bindable.of(ClonedArrayBean.class);
ClonedArrayBean bean = this.binder.bind("foo", target).get();
assertThat(bean.getBar()).containsExactly("hello");
}
@Test
void bindToBeanWithExceptionInGetterForExistingValue() {
MockConfigurationPropertySource source = new MockConfigurationPropertySource();
source.put("foo.values", "a,b,c");
this.sources.add(source);
BeanWithGetterException result = this.binder.bind("foo", Bindable.of(BeanWithGetterException.class)).get();
assertThat(result.getValues()).containsExactly("a", "b", "c");
}
@Test
void bindToBeanWithEnumSetCollection() {
MockConfigurationPropertySource source = new MockConfigurationPropertySource();
source.put("foo.values[0]", "foo-bar,bar-baz");
this.sources.add(source);
BeanWithEnumSetCollection result = this.binder.bind("foo", Bindable.of(BeanWithEnumSetCollection.class)).get();
List<EnumSet<ExampleEnum>> values = result.getValues();
assertThat(values).isNotNull();
assertThat(values.get(0)).containsExactly(ExampleEnum.FOO_BAR, ExampleEnum.BAR_BAZ);
}
@Test
void bindToWellFormedSystemEnvironmentVariableProperty() {
// gh-46184
Map<String, Object> map = new LinkedHashMap<>();
map.put("FOO_THENAMES_0_FIRST", "spring");
map.put("FOO_THENAMES_0_LAST", "boot");
map.put("FOO_THENAMES_1_FIRST", "binding");
map.put("FOO_THENAMES_1_LAST", "test");
SystemEnvironmentPropertySource propertySource = new SystemEnvironmentPropertySource(
StandardEnvironment.SYSTEM_ENVIRONMENT_PROPERTY_SOURCE_NAME, map);
this.sources.add(ConfigurationPropertySource.from(propertySource));
BeanWithCamelCaseNameList result = this.binder.bind("foo", BeanWithCamelCaseNameList.class).get();
assertThat(result.theNames()).containsExactly(new Name("spring", "boot"), new Name("binding", "test"));
}
@Test
void bindToLegacySystemEnvironmentVariableProperty() {
// gh-46184
Map<String, Object> map = new LinkedHashMap<>();
map.put("FOO_THE_NAMES_0_FIRST", "spring");
map.put("FOO_THE_NAMES_0_LAST", "boot");
map.put("FOO_THE_NAMES_1_FIRST", "binding");
map.put("FOO_THE_NAMES_1_LAST", "test");
SystemEnvironmentPropertySource propertySource = new SystemEnvironmentPropertySource(
StandardEnvironment.SYSTEM_ENVIRONMENT_PROPERTY_SOURCE_NAME, map);
this.sources.add(ConfigurationPropertySource.from(propertySource));
BeanWithCamelCaseNameList result = this.binder.bind("foo", BeanWithCamelCaseNameList.class).get();
assertThat(result.theNames()).containsExactly(new Name("spring", "boot"), new Name("binding", "test"));
}
static class ExampleCollectionBean {
private final List<String> items = new ArrayList<>();
private Set<String> itemsSet = new LinkedHashSet<>();
private @Nullable String string;
List<String> getItems() {
return this.items;
}
void setItems(List<String> items) {
this.items.add("d");
}
Set<String> getItemsSet() {
return this.itemsSet;
}
void setItemsSet(Set<String> itemsSet) {
this.itemsSet = itemsSet;
}
@Nullable String getString() {
return this.string;
}
void setString(@Nullable String string) {
this.string = string;
}
}
static class ExampleCustomNoDefaultConstructorBean {
private MyCustomNoDefaultConstructorList items = new MyCustomNoDefaultConstructorList(
Collections.singletonList("foo"));
MyCustomNoDefaultConstructorList getItems() {
return this.items;
}
void setItems(MyCustomNoDefaultConstructorList items) {
this.items = items;
}
}
static class MyCustomNoDefaultConstructorList extends ArrayList<String> {
MyCustomNoDefaultConstructorList(List<String> items) {
addAll(items);
}
}
static class ExampleCustomWithDefaultConstructorBean {
private final MyCustomWithDefaultConstructorList items = new MyCustomWithDefaultConstructorList();
MyCustomWithDefaultConstructorList getItems() {
return this.items;
}
void setItems(MyCustomWithDefaultConstructorList items) {
this.items.clear();
this.items.addAll(items);
}
}
static class MyCustomWithDefaultConstructorList extends ArrayList<String> {
}
static class BeanWithNestedCollection {
private @Nullable String value;
private @Nullable List<BeanWithNestedCollection> foos;
@Nullable List<BeanWithNestedCollection> getFoos() {
return this.foos;
}
void setFoos(@Nullable List<BeanWithNestedCollection> foos) {
this.foos = foos;
}
@Nullable String getValue() {
return this.value;
}
void setValue(@Nullable String value) {
this.value = value;
}
}
static class ClonedArrayBean {
private String @Nullable [] bar;
String @Nullable [] getBar() {
return (this.bar != null) ? this.bar.clone() : null;
}
void setBar(String @Nullable [] bar) {
this.bar = bar;
}
}
static class BeanWithGetterException {
private @Nullable List<String> values;
void setValues(@Nullable List<String> values) {
this.values = values;
}
@Nullable List<String> getValues() {
return Collections.unmodifiableList(this.values);
}
}
static class BeanWithEnumSetCollection {
private @Nullable List<EnumSet<ExampleEnum>> values;
void setValues(@Nullable List<EnumSet<ExampleEnum>> values) {
this.values = values;
}
@Nullable List<EnumSet<ExampleEnum>> getValues() {
return this.values;
}
}
record BeanWithCamelCaseNameList(List<Name> theNames) {
}
record Name(String first, String last) {
}
}
Domain
Source
Analyze Your Own Codebase
Get architecture documentation, dependency graphs, and domain analysis for your codebase in minutes.
Try Supermodel Free