diff --git a/src/main/kotlin/org/springframework/data/mapping/KPropertyPath.kt b/src/main/kotlin/org/springframework/data/mapping/KPropertyPath.kt index ec1cab4943..8566aab4f6 100644 --- a/src/main/kotlin/org/springframework/data/mapping/KPropertyPath.kt +++ b/src/main/kotlin/org/springframework/data/mapping/KPropertyPath.kt @@ -31,14 +31,29 @@ private class KPropertyPath( val child: KProperty1 ) : KProperty by child +/** + * Abstraction of a property path that consists of parent [KProperty], + * and child property [KProperty], where parent [parent] has an [Iterable] + * of children, so it represents 1-M mapping, not 1-1, like [KPropertyPath] + * + * @author Mikhail Polivakha + */ +internal class KIterablePropertyPath( + val parent: KProperty?>, + val child: KProperty1 +) : KProperty by child + /** * Recursively construct field name for a nested property. * @author Tjeu Kayim + * @author Mikhail Polivakha */ internal fun asString(property: KProperty<*>): String { return when (property) { is KPropertyPath<*, *> -> "${asString(property.parent)}.${property.child.name}" + is KIterablePropertyPath<*, *> -> + "${asString(property.parent)}.${property.child.name}" else -> property.name } } @@ -55,5 +70,32 @@ internal fun asString(property: KProperty<*>): String { * @author Yoann de Martino * @since 2.5 */ +@JvmName("div") operator fun KProperty.div(other: KProperty1): KProperty = KPropertyPath(this, other) + +/** + * Builds [KPropertyPath] from Property References. + * Refer to a nested property in an embeddable or association. + * + * Note, that this function is different from [div] above in the + * way that it represents a division operator overloading for + * child references, where parent to child reference relation is 1-M, not 1-1. + * It implies that parent has an [Iterable] or any liner [Collection] of children. + ** + * For example, referring to the field "addresses.street": + * ``` + * User::addresses / Author::street contains "Austin" + * ``` + * + * And the entities may look like this: + * ``` + * class User(val addresses: List
) + * + * class Address(val street: String) + * ``` + * @author Mikhail Polivakha + */ +@JvmName("divIterable") +operator fun KProperty?>.div(other: KProperty1): KProperty = + KIterablePropertyPath(this, other) diff --git a/src/test/kotlin/org/springframework/data/mapping/KPropertyPathTests.kt b/src/test/kotlin/org/springframework/data/mapping/KPropertyPathTests.kt index 18276ae9af..29f5f86dbb 100644 --- a/src/test/kotlin/org/springframework/data/mapping/KPropertyPathTests.kt +++ b/src/test/kotlin/org/springframework/data/mapping/KPropertyPathTests.kt @@ -24,6 +24,7 @@ import org.junit.Test * @author Tjeu Kayim * @author Yoann de Martino * @author Mark Paluch + * @author Mikhail Polivakha */ class KPropertyPathTests { @@ -43,6 +44,14 @@ class KPropertyPathTests { assertThat(property).isEqualTo("author.name") } + @Test // DATACMNS-3010 + fun `Convert from Iterable nested KProperty to field name`() { + + val property = (User::addresses / Address::street).toDotPath() + + assertThat(property).isEqualTo("addresses.street") + } + @Test // DATACMNS-1835 fun `Convert double nested KProperty to field name`() { @@ -106,4 +115,8 @@ class KPropertyPathTests { class Book(val title: String, val author: Author) class Author(val name: String) + + class User(val addresses: List
) + + class Address(val street: String) }