This example is meant to show an OpenLMIS 3.x Stock Management Validator Extension Module at work.
- Docker 1.11+
- Fork/clone this repository from GitHub.
git clone https://github.com/OpenLMIS/openlmis-stockmanagement-validator-extension.git
- Fork/clone
openlmis-stockmanagement
repository from GitHub.
git clone https://github.com/OpenLMIS/openlmis-stockmanagement.git
- To assemble the outputs of project and create jar file run
docker-compose -f docker-compose.yml run builder
. - Edit configuration file
extensions.properties
fromopenlmis-example-extensions
repository to use your defined extension. - Run builder for
openlmis-example-extensions
and build image. - Run
openlmis-ref-distro
usingdocker-compose.openlmis-stockmanagement-validator-extension.yml
and check if your changes has been applied.
OpenLMIS allows extending or overriding certain behavior of service using extension points and extension modules. Every independent service can expose extension points that an extension module may utilize to extend its behavior. Extension point is a Java interface placed in the service. Every extension point has its default implementation that can be overridden. Extension modules can contain custom implementation of one or more extension points from main service.
Decision about which implementation should be used is made based on the configuration file extensions.properties
.
This configuration file specifies which modules (implementations) should be used for the Service's extension points.
In openlmis-example-extensions repository, there is an example of one such configuration file that specifies that
a NoneValidator
module should be used for the extension point AdjustmentReasonValidator
.
#Example extensions configuration
AdjustmentReasonValidator=NoneValidator
FreeTextValidator=ReasonFreeTextValidator
UnpackKitValidator=NoKitsValidator
The extension point AdjustmentReasonValidator
is an ID defined by the interface AdjustmentReasonValidator.java
,
while the extension module NoneValidator
is an implementation of that interface whose name is a Spring Bean
defined in NoneValidator.java
Configuration file lives in independent service repository. Every extension module should be deployed as JAR. Example extension module and configuration file is published in the repository openlmis-stockmanagement-validator-extension.
Following classes are example of extension points usage:
- AdjustmentReasonValidator.java - sample extension point, that has Id defined in ExtensionPointId class.
- DefaultAdjustmentReasonValidator.java - default implementation of that interface, it has
@Component
annotation that contains its Id. - NoneValidator.java - class extending AdjustmentReasonValidator interface from openlmis-stockmanagement repository. It has
@Component
annotation that contains its Id. - ExtensionManager.java - class that has getExtensionByPointId method. It returns implementation of an extension class that is defined in configuration file for extension point with given Id.
The extension points' and extension modules' IDs should be unique and in UpperCamelCase. A situation where two extension modules have the same ID leads to undeterministic behavior - it is not possible to predict which bean will be used.
- Extension points should be descriptive of the behavior that may be changed. For example "RequisitionOrderQuantityCalculation" instead of "OrderQuantity".
- Extension Modules should describe the behaviour that is implemented, and the extension point that is being used. For example "RequisitionOrderQuantityCalculationAMC" and "RequisitionOrderQuantityCalculationISA".