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

Support parameter injection for @EventListener methods #29833

Closed
joshlong opened this issue Jan 16, 2023 · 8 comments
Closed

Support parameter injection for @EventListener methods #29833

joshlong opened this issue Jan 16, 2023 · 8 comments
Labels
in: core Issues in core modules (aop, beans, core, context, expression) status: declined A suggestion or change that we don't feel we should currently apply

Comments

@joshlong
Copy link
Member

I use events a lot. They let me think In terms of clean separate of modules, in terms of events, not data/state.
I use ApplicationEvent and ApplicationListener a lot. There's at least one in every Spring Boot app I've ever written. In anything more serious than a demo, I use them a ton, as they help me keep my API surface area clean - I only export and work with types. This in turn makes it easier to move to things like CQRS or microservices. I love events and they're implemented well in Spring.

@EventListener 
public void applicationReady (ApplicationReadyEvent are)  throws Exception {
   System.out.println( "the application is ready!" ) ;
} 

The trouble is that I often need dependencies, so I end up creating a @Bean:

@Bean 
ApplicationListener<ApplicationReadyEvent> applicationReady (A a ) { 
 return event ->    System.out.println( "the application is ready? " + (a.isReady()) ) ;
}

This is a problem because
a) i have to use ApplicationEvent subclasses again (I like not having to use ApplicationEvent with the plain @EventListener
b) i am forced to handle exceptions because ApplicationListener doesn't have throws Throwable on its signature.

So, my dream use case is:

@EventListener 
public void applicationReady ( ApplicationReadyEvent are ,  A a)  throws Throwable { 
   System.out.println( "the application is ready? " + (a.isReady()) ) ;
}

I'd be willing to settle for having to declare @Autowired on the beans that are injected (A a) or , alternatively a new annotation to disambiguate (say, @ApplicationEvent on the instance of the event)

If I could have anything I wanted for my birthday, at the end of January, I'd also love to be able to have package private @EventListener-annotated methods that are not public :)

Thanks in advance for an amazing piece of software.

@spring-projects-issues spring-projects-issues added the status: waiting-for-triage An issue we've not yet triaged or decided on label Jan 16, 2023
@odrotbohm
Copy link
Member

Where exactly does the first method you show live? To become a listener, it has to live in a bean, which is subject for DI. In other words: wouldn't you just inject the needed dependency into the bean that declares the event listener method?

Also, @EventListener works like a charm on package protected methods.

@joshlong
Copy link
Member Author

good to know on the package protected methods! thanks

I often put these listeners in @Configuration classes. I often put them in the Spring Boot main class, but sometimes in package-specific configuration classes, in the same class as the Configuration class that exports the dependency, which creates a circular dependency, so I have to extract them out to another component, which means I have another full class, which again is what I'm trying to avoid. not everything in the world needs a class. Sometimes an event listener is just a function/method invocation.

@sbrannen
Copy link
Member

sbrannen commented Jan 18, 2023

in the same class as the Configuration class that exports the dependency, which creates a circular dependency, so I have to extract them out to another component, which means I have another full class,

What happens if you make the injection point @Lazy or use an ObjectProvider -- to avoid the circular dependency?

Also, regarding having dependencies injected as method arguments in an @EventListener method, doing so would require that the @Autowired dependencies be resolved for each method invocation of the @EventListener method (unless the resolved dependencies were somehow cached for each @EventListener method, but I'm not sure we would want to introduce a cache for that purpose). So it might not be feasible to provide such support. Instead, we recommend that dependencies of an @EventListener method be injected into the enclosing instance.

@bclozel bclozel changed the title please support injected parameters for @EventListener Support parameter injection for @EventListener methods Jan 18, 2023
@bclozel bclozel changed the title Support parameter injection for @EventListener methods Support parameter injection for @EventListener methods Jan 18, 2023
@bclozel bclozel added in: core Issues in core modules (aop, beans, core, context, expression) type: enhancement A general enhancement labels Jan 18, 2023
@joshlong
Copy link
Member Author

joshlong commented Jan 19, 2023

I think having a cache seems reasonable enough. Spring maintains a cache for beans. SIngletons are going to be the most common. That’ll be a quick constant time operation, right? If the beans are scoped, then of course that’ll require re-evaluation but that’s true for the much more verbose alternative of having fields in a containing class, too.

@snicoll
Copy link
Member

snicoll commented Oct 2, 2023

Thanks for the suggestion but I think it would be at odd with everything else we're doing. If you look at our method invocation mechanism, we do support injections of things that are related to the task at hand. The best example of this is @RequestMapping that provides a wide range of options based on the HTTP request/response. We don't offer a way to inject a bean in there and I think we should continue to do that for consistency at the very least.

@snicoll snicoll closed this as not planned Won't fix, can't repro, duplicate, stale Oct 2, 2023
@snicoll snicoll 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 type: enhancement A general enhancement labels Oct 2, 2023
@chkpnt
Copy link

chkpnt commented Nov 15, 2023

@joshlong If you do not want to inject the bean into your @Configuration class, you can request a bean from the ApplicationContext. Something like A a = ApplicationContextProvider.getApplicationContext().getBean(A.class).

@joshlong
Copy link
Member Author

Thank you for the suggestion

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
in: core Issues in core modules (aop, beans, core, context, expression) status: declined A suggestion or change that we don't feel we should currently apply
Projects
None yet
Development

No branches or pull requests

7 participants