Skip to content
New issue

Have a question about this project? # for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “#”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? # to your account

spring-rabbit-test: rabbitListenerAnnotationProcessor has already been defined #914

Closed
liias opened this issue Feb 22, 2019 · 10 comments · Fixed by #915
Closed

spring-rabbit-test: rabbitListenerAnnotationProcessor has already been defined #914

liias opened this issue Feb 22, 2019 · 10 comments · Fixed by #915
Assignees
Milestone

Comments

@liias
Copy link

liias commented Feb 22, 2019

I am using:
spring boot 2.1.2.RELEASE
spring-rabbit-2.1.3.RELEASE
spring-rabbit-test-2.1.3.RELEASE

When running tests I get an error:

The bean 'org.springframework.amqp.rabbit.config.internalRabbitListenerAnnotationProcessor', defined in class path resource [org/springframework/amqp/rabbit/test/RabbitListenerTestBootstrap.class], could not be registered. A bean with that name has already been defined in class path resource [org/springframework/amqp/rabbit/annotation/RabbitBootstrapConfiguration.class] and overriding is disabled.

Spring suggests to use "spring.main.allow-bean-definition-overriding=true" to override the default behaviour and it does work. However, I find it a bit bad solution to change global spring configuration.

There is also possibility that I am doing something wrong.

@liias
Copy link
Author

liias commented Feb 22, 2019

Just looked into how RabbitBootstrapConfiguration gets loaded in my tests.

Using @RabbitListenerTest does two things:
Enables @EnableRabbit (which does @Import(RabbitBootstrapConfiguration.class))
Adds @Import(RabbitListenerTestBootstrap.class)

Now, I can't do @Import(RabbitListenerTestBootstrap.class) instead of @RabbitListenerTest, because
rabbit-test checks the annotation specifically in org.springframework.amqp.rabbit.test.RabbitListenerTestHarness#RabbitListenerTestHarness.

And even if I could, there are probably some parts of RabbitBootstrapConfiguration I do want to have?

Things to try:

  1. annotate
    org.springframework.amqp.rabbit.test.RabbitListenerTestBootstrap#rabbitListenerAnnotationProcessor with @Primary (I need to test it first, this still means overriding, so I assume same error). Update: nope, this doesnt work.

Related spring topic: spring-projects/spring-boot#15216

@garyrussell
Copy link
Contributor

I am not sure why you opened an issue here; this has nothing to do with this project.

Can you explain what you think we can do about it?

@liias
Copy link
Author

liias commented Feb 22, 2019

Hey.

So, I don't know the project well to state facts, so here are my guesses:
You could take away @EnableRabbit from @RabbitListenerTest, so RabbitBootstrapConfiguration is not loaded in. In order to get other beans from RabbitBootstrapConfiguration, the RabbitBootstrapConfiguration could be split up or you could redefine all beans in RabbitListenerTestBootstrap, so RabbitListenerTestBootstrap would be a full replacement for RabbitBootstrapConfiguration.

In that case user can use spring profiles (i.e test vs not test) to choose what bean configurations they choose.

The goal would be making the @RabbitListenerTest work with the spring boot default config spring.main.allow-bean-definition-overriding=false.

Does that make any sense?

@garyrussell
Copy link
Contributor

It makes sense for Boot users but it will break every application that uses @RabbitListenerTest but not Spring Boot.

I don't know if it's possible to make it conditional on whether Boot is present on the class path, but it could be difficult because we'd end up with circular dependencies.

@liias
Copy link
Author

liias commented Feb 22, 2019

If nothing else, then this issue will at least appear in search results for people starting a new project with the test lib.

So you are saying spring.main.allow-bean-definition-overriding=false is very much a spring-boot default, and won't make it into spring proper as default value?

@garyrussell
Copy link
Contributor

The problem is that if we, say, unconditionally set spring.main.allow-bean-definition-overriding=true in the test framework, it could mask the very issue that the Boot team is trying to avoid by making it false by default.

What is the difficulty of adding a TestPropertySource to your test case?

@RabbitListenerTest
@TestPropertySource(properties = "spring.main.allow-bean-definition-overriding=true")

@liias
Copy link
Author

liias commented Feb 22, 2019

@TestPropertySource certainly works. But I would personally just enable it in my application, because if just enabling it when running tests you could inadvertently get errors when actually running application (because tests passed).

In conclusion, I thought spring is advertising to not override beans now and this library (which is part of spring umbrella) does do that, so wanted to let you know.

@garyrussell
Copy link
Contributor

Good point; we'll take a look to see if there is something we can do to avoid this.

@artembilan artembilan self-assigned this Feb 22, 2019
@artembilan artembilan added this to the 2.2.0.M1 milestone Feb 22, 2019
artembilan added a commit to artembilan/spring-amqp that referenced this issue Feb 22, 2019
Fixes spring-projects#914

* Rework `RabbitBootstrapConfiguration` into the `ImportBeanDefinitionRegistrar`
and check for bean definitions presence it is going to register.
This way an override from the `RabbitListenerTestBootstrap` is going to
have a precedence and its `RabbitListenerTestHarness` won't allow a
regular `RabbitListenerAnnotationBeanPostProcessor` to be registered
* Rework `TestRabbitTemplate` to gather all the listener container
from the `ContextRefreshedEvent` instead of the `SmartInitializingSingleton`.
Looks like Spring doesn't care about their order, therefore we need to
be sure that `RabbitListenerAnnotationBeanPostProcessor` has populated
its `registry` with containers before.
* Demonstrate that `allowBeanDefinitionOverriding = false` works well
now in the `ExampleRabbitListenerCaptureTest`
* Optimize `checkTestConfigs` and `updateCopyrights` Gradle tasks for
their `inputs` and `outputs` for better `UP-TO-DATE` handling

**Cherry-pick to 2.1.x without `build.gradle` changes**
garyrussell pushed a commit that referenced this issue Feb 22, 2019
* GH-914: Honor AllowBeanDefOverriding = false

Fixes #914

* Rework `RabbitBootstrapConfiguration` into the `ImportBeanDefinitionRegistrar`
and check for bean definitions presence it is going to register.
This way an override from the `RabbitListenerTestBootstrap` is going to
have a precedence and its `RabbitListenerTestHarness` won't allow a
regular `RabbitListenerAnnotationBeanPostProcessor` to be registered
* Rework `TestRabbitTemplate` to gather all the listener container
from the `ContextRefreshedEvent` instead of the `SmartInitializingSingleton`.
Looks like Spring doesn't care about their order, therefore we need to
be sure that `RabbitListenerAnnotationBeanPostProcessor` has populated
its `registry` with containers before.
* Demonstrate that `allowBeanDefinitionOverriding = false` works well
now in the `ExampleRabbitListenerCaptureTest`
* Optimize `checkTestConfigs` and `updateCopyrights` Gradle tasks for
their `inputs` and `outputs` for better `UP-TO-DATE` handling

**Cherry-pick to 2.1.x without `build.gradle` changes**

* * Fix typo in the `ExampleRabbitListenerCaptureTest`
garyrussell pushed a commit that referenced this issue Feb 22, 2019
* GH-914: Honor AllowBeanDefOverriding = false

Fixes #914

* Rework `RabbitBootstrapConfiguration` into the `ImportBeanDefinitionRegistrar`
and check for bean definitions presence it is going to register.
This way an override from the `RabbitListenerTestBootstrap` is going to
have a precedence and its `RabbitListenerTestHarness` won't allow a
regular `RabbitListenerAnnotationBeanPostProcessor` to be registered
* Rework `TestRabbitTemplate` to gather all the listener container
from the `ContextRefreshedEvent` instead of the `SmartInitializingSingleton`.
Looks like Spring doesn't care about their order, therefore we need to
be sure that `RabbitListenerAnnotationBeanPostProcessor` has populated
its `registry` with containers before.
* Demonstrate that `allowBeanDefinitionOverriding = false` works well
now in the `ExampleRabbitListenerCaptureTest`
* Optimize `checkTestConfigs` and `updateCopyrights` Gradle tasks for
their `inputs` and `outputs` for better `UP-TO-DATE` handling

**Cherry-pick to 2.1.x without `build.gradle` changes**

* * Fix typo in the `ExampleRabbitListenerCaptureTest`
@garyrussell
Copy link
Contributor

Fixed in 2.1.5.BUILD-SNAPSHOT - thanks for persevering!

@liias
Copy link
Author

liias commented Feb 22, 2019

This was the fastest issue to fix time I've seen for third party library. High five to both of you.

# for free to join this conversation on GitHub. Already have an account? # to comment
Projects
None yet
Development

Successfully merging a pull request may close this issue.

3 participants