Skip to content
This repository has been archived by the owner on Feb 23, 2023. It is now read-only.

@Component/@Bean with @ConfigurationProperties does not work with Kotlin #1679

Closed
Azbesciak opened this issue Aug 3, 2022 · 6 comments
Closed
Labels
status: declined A suggestion or change that we don't feel we should currently apply

Comments

@Azbesciak
Copy link

Hi, I tried to find the example, but failed.
I have build my app - in various place I use combination like @Component or @Bean, and @ConfigurationProperties on the class (I use kotlin.
In short, it is something like this

@Component
@ConfigurationProperties(prefix = "some.name")
data class SomeNameData(
	var value: String = "",
)

or (there without defining annotations on the class)

@Configuration
class MyConfig {
	@Bean
	@ConfigurationProperties(prefix = "some.name")
	fun myData() = SomeNameData()
}

data class SomeNameData(
	var value: String = "",
)

And for normal spring app it works fine.
But for spring-native I got following error

Error creating bean with name 'myData': Initialization of bean failed; nested 
exception is kotlin.reflect.jvm.internal.KotlinReflectionInternalError: Could not compute caller for function: public constructor SomeNameData(value: kotlin.String = ...) defined in com.example.demo.SomeNameData[DeserializedClassConstructorDescriptor@61ddb0ae] (member = null)
        at org.springframework.aot.beans.factory.InjectedConstructionResolver.resolve(InjectedConstructionResolver.java:88) ~[na:na]
        at org.springframework.aot.beans.factory.InjectedElementResolver.resolve(InjectedElementResolver.java:35) ~[com.example.demo.DemoApplicationKt:0.12.1]
        at org.springframework.aot.beans.factory.InjectedElementResolver.create(InjectedElementResolver.java:66) ~[com.example.demo.DemoApplicationKt:0.12.1]
        at org.springframework.aot.beans.factory.BeanDefinitionRegistrar$BeanInstanceContext.create(BeanDefinitionRegistrar.java:211) ~[na:na]
        at org.springframework.aot.ContextBootstrapInitializer.lambda$initialize$1(ContextBootstrapInitializer.java:108) ~[na:na]
        at org.springframework.aot.beans.factory.BeanDefinitionRegistrar$ThrowableFunction.apply(BeanDefinitionRegistrar.java:294) ~[com.example.demo.DemoApplicationKt:0.12.1]
        at org.springframework.aot.beans.factory.BeanDefinitionRegistrar.lambda$instanceSupplier$0(BeanDefinitionRegistrar.java:115) ~[na:na]
        at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.obtainFromSupplier(AbstractAutowireCapableBeanFactory.java:1249) ~[com.example.demo.DemoApplicationKt:5.3.22]
        at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.createBeanInstance(AbstractAutowireCapableBeanFactory.java:1191) ~[com.example.demo.DemoApplicationKt:5.3.22]
        at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.doCreateBean(AbstractAutowireCapableBeanFactory.java:582) ~[com.example.demo.DemoApplicationKt:5.3.22]
        at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.createBean(AbstractAutowireCapableBeanFactory.java:542) ~[com.example.demo.DemoApplicationKt:5.3.22]
        at org.springframework.beans.factory.support.AbstractBeanFactory.lambda$doGetBean$0(AbstractBeanFactory.java:335) ~[com.example.demo.DemoApplicationKt:5.3.22]
        at org.springframework.beans.factory.support.DefaultSingletonBeanRegistry.getSingleton(DefaultSingletonBeanRegistry.java:234) ~[com.example.demo.DemoApplicationKt:5.3.22]
        at org.springframework.beans.factory.support.AbstractBeanFactory.doGetBean(AbstractBeanFactory.java:333) ~[com.example.demo.DemoApplicationKt:5.3.22]
        at org.springframework.beans.factory.support.AbstractBeanFactory.getBean(AbstractBeanFactory.java:208) ~[com.example.demo.DemoApplicationKt:5.3.22]
        at org.springframework.beans.factory.support.DefaultListableBeanFactory.preInstantiateSingletons(DefaultListableBeanFactory.java:955) ~[com.example.demo.DemoApplicationKt:5.3.22]
        at org.springframework.context.support.AbstractApplicationContext.finishBeanFactoryInitialization(AbstractApplicationContext.java:918) ~[com.example.demo.DemoApplicationKt:5.3.22]
        at org.springframework.context.support.AbstractApplicationContext.refresh(AbstractApplicationContext.java:583) ~[com.example.demo.DemoApplicationKt:5.3.22]
        at org.springframework.boot.web.reactive.context.ReactiveWebServerApplicationContext.refresh(ReactiveWebServerApplicationContext.java:66) ~[com.example.demo.DemoApplicationKt:2.7.2]
        at org.springframework.boot.SpringApplication.refresh(SpringApplication.java:734) ~[com.example.demo.DemoApplicationKt:2.7.2]
        at org.springframework.boot.SpringApplication.refreshContext(SpringApplication.java:408) ~[com.example.demo.DemoApplicationKt:2.7.2]
        at org.springframework.boot.SpringApplication.run(SpringApplication.java:308) ~[com.example.demo.DemoApplicationKt:2.7.2]
        at org.springframework.boot.SpringApplication.run(SpringApplication.java:1306) ~[com.example.demo.DemoApplicationKt:2.7.2]
        at org.springframework.boot.SpringApplication.run(SpringApplication.java:1295) ~[com.example.demo.DemoApplicationKt:2.7.2]
        at com.example.demo.DemoApplicationKt.main(DemoApplication.kt:39) ~[na:na]
Caused by: org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'myData': Initialization of bean failed; nested exception is kotlin.reflect.jvm.internal.KotlinReflectionInternalError: Could not compute caller for function: public constructor SomeNameData(value: kotlin.String = ...) defined in com.exam
ple.demo.SomeNameData[DeserializedClassConstructorDescriptor@61ddb0ae] (member = null)
        at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.doCreateBean(AbstractAutowireCapableBeanFactory.java:628) ~[com.example.demo.DemoApplicationKt:5.3.22]
        at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.createBean(AbstractAutowireCapableBeanFactory.java:542) ~[com.example.demo.DemoApplicationKt:5.3.22]
        at org.springframework.beans.factory.support.AbstractBeanFactory.lambda$doGetBean$0(AbstractBeanFactory.java:335) ~[com.example.demo.DemoApplicationKt:5.3.22]
        at org.springframework.beans.factory.support.DefaultSingletonBeanRegistry.getSingleton(DefaultSingletonBeanRegistry.java:234) ~[com.example.demo.DemoApplicationKt:5.3.22]
        at org.springframework.beans.factory.support.AbstractBeanFactory.doGetBean(AbstractBeanFactory.java:333) ~[com.example.demo.DemoApplicationKt:5.3.22]
        at org.springframework.beans.factory.support.AbstractBeanFactory.getBean(AbstractBeanFactory.java:208) ~[com.example.demo.DemoApplicationKt:5.3.22]
        at org.springframework.beans.factory.config.DependencyDescriptor.resolveCandidate(DependencyDescriptor.java:276) ~[com.example.demo.DemoApplicationKt:5.3.22]
        at org.springframework.beans.factory.support.DefaultListableBeanFactory.doResolveDependency(DefaultListableBeanFactory.java:1391) ~[com.example.demo.DemoApplicationKt:5.3.22]
        at org.springframework.beans.factory.support.DefaultListableBeanFactory.resolveDependency(DefaultListableBeanFactory.java:1311) ~[com.example.demo.DemoApplicationKt:5.3.22]
        at org.springframework.aot.beans.factory.InjectedConstructionResolver.lambda$resolve$0(InjectedConstructionResolver.java:83) ~[na:na]
        at org.springframework.aot.beans.factory.InjectedConstructionResolver.resolveDependency(InjectedConstructionResolver.java:97) ~[na:na]
        at org.springframework.aot.beans.factory.InjectedConstructionResolver.resolve(InjectedConstructionResolver.java:83) ~[na:na]
        ... 24 common frames omitted
Caused by: kotlin.reflect.jvm.internal.KotlinReflectionInternalError: Could not compute caller for function: public constructor SomeNameData(value: kotlin.String = ...) defined in com.example.demo.SomeNameData[DeserializedClassConstructorDescriptor@61ddb0ae] (member = null)
        at kotlin.reflect.jvm.internal.KFunctionImpl.getCaller(KFunctionImpl.kt:61) ~[na:na]
        at kotlin.reflect.jvm.ReflectJvmMapping.getJavaConstructor(ReflectJvmMapping.kt:71) ~[na:na]
        at org.springframework.beans.BeanUtils$KotlinDelegate.findPrimaryConstructor(BeanUtils.java:853) ~[na:na]
        at org.springframework.beans.BeanUtils.findPrimaryConstructor(BeanUtils.java:283) ~[na:na]
        at org.springframework.boot.context.properties.ConfigurationPropertiesBindConstructorProvider.findConstructorBindingAnnotatedConstructor(ConfigurationPropertiesBindConstructorProvider.java:57) ~[na:na]
        at org.springframework.boot.context.properties.ConfigurationPropertiesBindConstructorProvider.getBindConstructor(ConfigurationPropertiesBindConstructorProvider.java:48) ~[na:na]
        at org.springframework.boot.context.properties.ConfigurationPropertiesBean$BindMethod.forType(ConfigurationPropertiesBean.java:317) ~[com.example.demo.DemoApplicationKt:2.7.2]
        at org.springframework.boot.context.properties.ConfigurationPropertiesBean.<init>(ConfigurationPropertiesBean.java:78) ~[na:na]
        at org.springframework.boot.context.properties.ConfigurationPropertiesBean.create(ConfigurationPropertiesBean.java:276) ~[na:na]
        at org.springframework.boot.context.properties.ConfigurationPropertiesBean.get(ConfigurationPropertiesBean.java:207) ~[na:na]
        at org.springframework.boot.context.properties.ConfigurationPropertiesBindingPostProcessor.postProcessBeforeInitialization(ConfigurationPropertiesBindingPostProcessor.java:78) ~[com.example.demo.DemoApplicationKt:2.7.2]
        at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.applyBeanPostProcessorsBeforeInitialization(AbstractAutowireCapableBeanFactory.java:440) ~[com.example.demo.DemoApplicationKt:5.3.22]
        at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.initializeBean(AbstractAutowireCapableBeanFactory.java:1796) ~[com.example.demo.DemoApplicationKt:5.3.22]
        at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.doCreateBean(AbstractAutowireCapableBeanFactory.java:620) ~[com.example.demo.DemoApplicationKt:5.3.22]
        ... 35 common frames omitted

With @ConfigurationPropertiesScan and @ConstructorBinding as in examples - it works, but IMHO the solution with @Bean is more configurable. Am I doing something wrong? Can something be done with that?

@Azbesciak Azbesciak changed the title @Component/@Bean with @ConfigurationProperties does not work @Component/@Bean with @ConfigurationProperties does not work with Kotlin Aug 3, 2022
@spring-projects-issues spring-projects-issues added the status: waiting-for-triage An issue we've not yet triaged or decided on label Aug 3, 2022
@mhalbritter
Copy link
Contributor

mhalbritter commented Aug 4, 2022

Hi! If you do a @EnableConfigurationProperties(SomeNameData) it works? It only doesn't work if you annotate it with @Component?

@Azbesciak
Copy link
Author

Azbesciak commented Aug 4, 2022

Yes, the following app works

@SpringBootApplication
@ConfigurationPropertiesScan
class DemoApplication

fun main(args: Array<String>) {
	runApplication<DemoApplication>(*args)
}

@ConstructorBinding
@ConfigurationProperties(prefix = "some.name")
data class SomeNameData(
	var value: String = "",
)

@RequestMapping("hello")
@RestController
class HelloController(private val name: SomeNameData) {
	@GetMapping
	fun getName() = Mono.just("HELLO ${name.value}")
}

ok, the question was slightly different... need to check
It does not

@SpringBootApplication
@ConfigurationPropertiesScan
class DemoApplication

fun main(args: Array<String>) {
	runApplication<DemoApplication>(*args)
}

@Configuration
@EnableConfigurationProperties
class Config

@Component
@ConfigurationProperties(prefix = "some.name")
data class SomeNameData(
	var value: String = "",
)

it also does not

@SpringBootApplication
class DemoApplication

fun main(args: Array<String>) {
	runApplication<DemoApplication>(*args)
}

@Configuration
@EnableConfigurationProperties
class Config {
	@Bean
	fun data() = SomeNameData()
}

@ConfigurationProperties(prefix = "some.name")
data class SomeNameData(
	var value: String = "",
)

@mhalbritter
Copy link
Contributor

As there's a workaround by not annotating your configuration properties with @Component, this is unlikely to be fixed in Spring native, as this is in minimal maintenance mode. We'll look out for that usecase in the upcoming Boot 3 / Spring Framework 6 though.

@mhalbritter mhalbritter closed this as not planned Won't fix, can't repro, duplicate, stale Aug 4, 2022
@mhalbritter mhalbritter added status: declined A suggestion or change that we don't feel we should currently apply and removed status: waiting-for-triage An issue we've not yet triaged or decided on labels Aug 4, 2022
@Azbesciak
Copy link
Author

But the problem is that it does not work only in spring native

@mhalbritter
Copy link
Contributor

mhalbritter commented Aug 4, 2022

Yes, i understand that. But the problem you described has a workaround, and we are working on the native compilation story for Boot 3 / Spring Framework 6. When Spring Boot 3 is released, you won't need spring-native anymore, it's built into Boot itself. And there we strife to support your usecase. You can read more here

@Azbesciak
Copy link
Author

That is what I needed to know, thank you :)

Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Labels
status: declined A suggestion or change that we don't feel we should currently apply
Development

No branches or pull requests

3 participants