Skip to content
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

SpyBean does not necessary spy on the primary bean when there are 2 instance #7621

Closed
lsiu opened this issue Dec 12, 2016 · 5 comments
Closed
Labels
type: bug A general bug
Milestone

Comments

@lsiu
Copy link

lsiu commented Dec 12, 2016

When there are 2 beans and one marked as @Primary, in normal auto-wiring, the Primary bean is used. However, in case of test code autowired via @SpyBean, this is not necessary the case. This cause unexpected behavior when using the spybean to mock responses on methods.

I have created a sample project which illustrate this. See the test class and the same code

Looking more into the details. It appear which bean is spied on depends on the order the beans are returned from the method getExistingBeans in MockitoPostProcessor (the last bean return ends up being the one is spied on, since the loop in registerSpies essential overwrite previous spy registered). I would expect we always spy on the bean marked with @Primary similar to the normal auto-wiring behavior.

@spring-projects-issues spring-projects-issues added the status: waiting-for-triage An issue we've not yet triaged label Dec 12, 2016
@philwebb philwebb added type: bug A general bug and removed status: waiting-for-triage An issue we've not yet triaged labels Dec 12, 2016
@philwebb philwebb added this to the 1.4.3 milestone Dec 12, 2016
@philwebb
Copy link
Member

Thanks for the sample project. There's certainly an issue here but I'm not sure that it's directly related to @Primary. The MockitoPostProcessor should throw an exception because there is more than one candidate bean, however, the field is null. It appears that a new DefinitionsParser is being created and the field mappings are being lost.

@lsiu
Copy link
Author

lsiu commented Dec 13, 2016

Throwing an error is not an idea solution, since in production code, we do allow multiple beans and resolve using the @Primary bean annotation.

Resolving the Bean to mock or spy or should be same as how it is done in the production code. Is it possible to just reuse the same code? For example, I see DefaultListableBeanFactory.determineAutowireCandidate is use to resolve when there are multiple autowire candidate.

I have not fully investigate this yet, but I would think instead of spring-boot-test having its own logic to "pick" the bean to mock/spy on, it should delegate to the BeanFactory to resolve it.

@philwebb philwebb added the for: team-attention An issue we'd like other members of the team to review label Dec 13, 2016
@philwebb
Copy link
Member

@lsiu I see your point, but it's more than just picking the bean. It also needs to update the bean definition in the context. Let me have a think about the options.

@lsiu
Copy link
Author

lsiu commented Dec 14, 2016

@philwebb - I have created 2 tests that illustrate the problem and a proposed fix with the above commit (9f5c777). It may not be the best place to fix it, but it does fix the issue and doesn't break the other tests.

@philwebb
Copy link
Member

Thanks @lsiu! We discussed this a little on our regular team call and we think using the @Primary bean as the spy source is the best option.

@philwebb philwebb removed the for: team-attention An issue we'd like other members of the team to review label Dec 16, 2016
sbrannen added a commit to spring-projects/spring-framework that referenced this issue Oct 30, 2024
Spring Boot has honored @⁠Primary for selecting which candidate bean
@⁠MockBean and @⁠SpyBean should mock or spy since Spring Boot 1.4.3;
however, the support for @⁠Primary was not ported from Spring Boot to
Spring Framework's new Bean Overrides feature in the TestContext
framework.

To address that, this commit introduces support for @⁠Primary for
selecting bean overrides -- for example, for annotations such as
@⁠TestBean, @⁠MockitoBean, and @⁠MockitoSpyBean.

See spring-projects/spring-boot#7621
Closes gh-33819
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
type: bug A general bug
Projects
None yet
Development

No branches or pull requests

3 participants