I'm trying to define the service bean name using a property placeholder value. But getting error saying no bean found for the specific name. I got to know that the issue is with reading the property value, because while hardcoding the value it's working. Please help as I need to read the value from property file. Code snippet below:
application.properties
event.testRequest=TEST_REQUEST
Service Class
@Service("${event.testRequest}") // This is not working, getting "No bean named 'TEST_REQUEST' available" error
// @Service("TEST_REQUEST") // This is working
public class TestRequestExecutor extends DefaultExecutionService {
...
}
Also, to confirm the property value is reading correctly I tried using @Value("${event.testRequest}") private String value
where I'm getting the value "TEST_REQUEST" as expected. Not sure how to use that with @Service annotation.
Edit: To elaborate the need of externalising the service bean name, I'm using factory pattern in getting the implementation based on the event name (event names such as Event1, Event2..). If there's a change in the event name, the change will be just on property file rather than the Service bean name which is using the property placeholder.
@RestController
public class RequestProcessController {
@Autowired
private ExecutorFactory executorFactory;
..
ExecutionService executionService = executorFactory.getExecutionService(request.getEventType());
executionService.executeRequest(request);
..
}
@Component
public class ExecutorFactory {
private BeanFactory beanFactory;
public ExecutionService getExecutionService(String eventType) {
return beanFactory.getBean(eventType, DefaultExecutionService.class);
}
Here DefaultExecutionService
has different implementations like below..
@Service("${event.first}")
public class Event1Executor extends DefaultExecutionService {..}
..
@Service("${event.second}")
public class Event2Executor extends DefaultExecutionService {..}
event.first = Event1
event.second = Event2
So basically in future if Event1 name is updated to EventOne, I just need to update the property file, not the service class.
Any help much appreciated! Thanks!
Ok, Now its clear.
I think you can achieve such a behavior by changing the implementation:
You don't need to work with bean factory inside
ExecutorFactory
, instead consider creating the following implementation:Now the creation of such a map is something that is tricky and requires a different approach:
Don't use property resolution in your implementations of the executor service, instead go with some way of "static identification", it can be another annotation or maybe qualifier or even static bean name. In this example I'll go with a qualifier based approach since it seems to me the easiest to show/implement.
Then you can create an
ExecutorFactory
from Java Configuration class, that's why I haven't put the@Component
/@Service
annotation on theExecutorFactory
class at the beginning of the answer.First of all, I use here a feature of spring that allows to inject map of string to your interface (in this case
DefaultExecutorService
). Spring will inject a map of:Then I access the configuration that is supposed to support a method for getting all the events. This can be implemented in different ways, probably the most natural way of doing this is using
@ConfigurationProperties
annotation and mapping the event map ofapplication.yaml
to the Map in Java. You can read this article for technical details.As a side note, although I've used
@Configuration
approach because It looks more clear to me, it's possible to use@Service
on theExecutorFactory
, but then the similar logic that I've shown will be a part of the executor factory (constructor or post-construct method), you still can inject map of bean names to actual beans and configuration properties to the constructor, its up to you to decide