-
Notifications
You must be signed in to change notification settings - Fork 41.1k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
When using Kotlin, mutable super-class's properties are not bound when a sub-class declares a non-default primary constructor that does not cover the super-class's properties #44849
Comments
Could you please provide a complete sample application that works with earlier version and fails with the later one. Please either attach a zip file or provide a link to a GitHub project. |
Here is the Github project that indicates the problem: https://github.com/vominhtri231/PropertyNotMap |
I confirm it's a regression. The |
The change in behavior is due to 3.0 switching to implicit constructor binding. The Kotlin class has two constructors ( @vominhtri231, we'll see what we can do to get this working out of the box. In the meantime, there are a couple of ways in which you can work around the change in behavior. You can use @ConfigurationProperties(prefix = "my.test")
@Component
class MyProperties {
var customs: List<CustomA> = listOf()
var bases: List<Base> = listOf()
open class Base(
var type: String = ""
)
class CustomA @Autowired constructor(
var y: String = "",
var z: String = "",
) : Base()
} Alternatively, you can declare the class with only a default constructor: @ConfigurationProperties(prefix = "my.test")
@Component
class MyProperties {
var customs: List<CustomA> = listOf()
var bases: List<Base> = listOf()
open class Base(
var type: String = ""
)
class CustomA : Base() {
var y: String = ""
var z: String = ""
}
}
Thanks for the suggestion but this won't restore 2.x's behavior:
|
On further thought, this may be working as best it can given the concept of a primary constructor in Kotlin. If we started ignoring them and fell back to pure setter-based binding, it may cause a regression elsewhere by no longer calling the primary constructor. We don't describe in the documentation how primary constructors are treated and we probably should. Right now, I am leaning towards documenting the current behavior and recommending the used Let's see what the rest of the team thinks. |
Using |
@ConfigurationProperties(prefix = "my.test")
class MyProperties @Autowired constructor (
var y: String = "",
var z: String = ""
) : Base()
|
I have a spring boot - Kotlin project, which I am upgrading from Spring 5, Java 8 to Spring 6, Java 21
build.gradle.kts
MyProperties.kt
Run the application by
bootRun
task, with these env variables:I would expect that the
myProperties.customs[0].type
would returnvalue3
, but it would return an empty string instead.The problem only happens if the
kotlin("reflect")
dependency exists and does not occur in Spring 5, Java 8.It looks like the base class's properties are not bound at all. If I modify the
MyProperties
class as this, thetype
property will bind correctlyThe text was updated successfully, but these errors were encountered: