Custom components
The Genesis Application Platform enables you to create modules for business logic.
Injecting dependencies
A key part of writing custom code is injecting the objects which your module or process relies on.
Genesis supports some of the key annotations from javax.annotation and javax.inject, defined in Java Platform Enterprise Edition, using Google Guice as the internal dependency injection mechanism. In addition, Genesis provides some of its own annotations.
@Module
All classes that need to be created on microservice start-up must be annotated with @Module
. This ensures the instance is created as singleton.
@ProviderOf
This is used to annotate a class that is responsible for acting as Factory for a specified type. Adding the @Singleton
annotation ensures that only one Factory class is created.
Example:
@Singleton
@ProviderOf(type = PriceFeed::class)
class PriceFeedProvider : Provider<PriceFeedProvider> {
override fun get(): PriceFeed {
return new PriceFeed()
}
}
@Inject
This is used to annotate a field or constructor to indicate to Genesis that it should inject an object of matching type during the dependency injection stage. These types can be provided with the aforementioned @Module
and @ProviderOf
annotations. You should use the Java @Inject
annotation for best practice.
@Named
This annotation is used to provide Genesis system definition properties as part of the dependency injection mechanism, and should be used alongside @Inject
.
@PostConstruct
The Genesis microservice runtime environment will call this only once after initialization of the object, including all injected beans.
@PreDestroy
The Genesis microservice runtime environment will call this only once, just before Genesis removes the object from the application context on JVM shutdown.
Example
The example below combines @Module
, @Named
and @Inject
annotations on a constructor and on a field, and it uses @PostConstruct
and @PreDestroy
.
@Module
class PriceFeed @Inject constructor(@Named("CONNECTION_URL") private val connectionUrl: String) {
@Inject
private lateinit var value: Value
@PostConstruct
fun init() {
// code would go here
}
@PreDestroy
fun cleanUp() {
// code would go here
}
}
Conditional annotations
Conditional on property
You can define a module as conditional, based on system definition properties. In the example below, the AeronDriverModule will only be instantiated if the MqLayer property is set to AERON:
@Module
@ConditionalOnProperty("MqLayer", conditionalValue = "AERON")
Conditional on class
Likewise, modules can be defined to be conditional on classes, so if the selected class has been instantiated (e.g. the previous example shown as AeronDriverModule), then our dependency injection mechanism will instantiate this class as well.
@Module
@ConditionalOnClass(AeronDriverModule::class)
Conditional on missing class
You can also add conditional modules on missing classes. This annotation is very helpful if you want to develop an “extension” mechanism as part of your application.
For example, you could define an interface. You could then define an @Module
implementing this interface with a conditional on that interface class being missing.
- If a class is defined as an
@Module
that implements the interface, that class will be selected (i.e. this class can be defined in an external jar and added to classpath using framework config). - If no class is defined as an
@Module
to implement that interface, then the@ConditionalOnMissingClass
module will be instantiated.
For example, imagine you want to create a “hook” into the Genesis message-handling logic, so it can be extended as desired in other applications. The interface and default class would look like this in Kotlin:
interface MessageTracer{
fun onMessageReceived(msg: Message)
}
@Module
@ConditionalOnMissingClass(MessageTracer::class)
class DefaultMessageTracer : MessageTracer{
override fun onMessageReceived(msg: Message) = println("This is a message $msg")
}
If an application requires custom logic, a new @Module
class could be defined in a separate jar file like this:
@Module
class ProductMessageTracer : MessageTracer{
override fun onMessageReceived(msg: Message) = println("Hi $msg")
}
If the package for this class is available in the classpath (<classpath>
section in processes.xml) and is also being scanned (i.e. <package>
section in processes.xml), ProductMessageTracer
will take precedence when starting the process and DefaultMessageTracer
will be ignored. Otherwise, DefaultMessageTracer
will be instantiated.
Conditional on property and missing class
Lastly, we have a combination of both “conditional on property” and “conditional on missing class“ annotations. Referring to the previous example, we could use both features at once like this:
@Module
@ConditionalOnPropertyAndMissingClass("MessageTracer", conditionalValue = "DEFAULT", MessageTracer::class)
class DefaultMessageTracer : MessageTracer{
override fun onMessageReceived(msg: Message) = println("This is a message $msg")
}
The process will crash on start up if the system definition value for MessageTracer
is set to anything other than DEFAULT and no implementation has been provided for MessageTracer
. This annotation can be used to enforce good practice.
Injectable properties from System Definition
Example of genesis-system-definition.kts
file
systemDefinition {
global {
item(name = "CONFIG_FILE_NAME", value = "/data/")
// other params omitted for simplicity
}
}
System Definition property being referenced in Java file
@Inject
public RequestReplyDefinitionReader(RxDb db,
@Named("CONFIG_FILE_NAME") String configFileName) throws GenesisConfigurationException {
this(db.getDictionary(), configFileName);
}