If you want to write rules in a Java program you have lot of choices. You can use a third party library like Drools or use the JAVA built-in JSR-94 reference implementation. In WSO2 Carbon, there is a component that abstract the behaviour of different rule engine and give you a unified API. Currently it has plugged into Drools and JAVA built-in JSR-94 implementation.
The rule component in WSO2 Carbon platform mainly used by WSO2 ESB product to mediate messages according to the given business rules. But the component is written to facilitate any requirement of using business rules in WSO2 Carbon platform. I had such a requirement in past few days and manage to use the rule component easily with the help of the component author, indika@wso2.com. So I thought it is worth sharing my experience in here.
Here You will be preparing the following stuff.
1. Rule configuration –
We can use this to provide the information about the rule implementation we are going to use, the rules (You can write rules inline or provide a reference to an external file) and the input and output adapter information.
<configuration xmlns="http://www.wso2.org/products/rule/drools">
<executionSet uri="simpleItemRuleXML">
<source key="file:src/test/resources/rules/simple-rules.drl"/>
<!-- <source>
<x><![CDATA[
rule InvokeABC
// rules inbuilt to the rule conf
end
]]>
</x>
</source> -->
<creation>
<property name="source" value="drl"/>
</creation>
</executionSet>
<session type="stateless"/>
<input name="facts" type="itemData" key="dataContext"></input>
<output name="results" type="itemData" key="dataContext"></output>
</configuration>
2. The Rules –
You can write rules inline in the above configuration or put it in a file and refer it from a key which can be refered from the ResourceHelper (described below).
import java.util.Calendar;
rule YearEndDiscount
when
$item : org.test.pojo.SimpleItem(price > 100 )
then
Calendar calendar = Calendar.getInstance();
if (calendar.get(Calendar.MONTH) == Calendar.JANUARY) {
$item.setPrice($item.getPrice() * 80/100);
}
end
3. Data Context –
The context object that can be used to feed and retrieve data from and to rule engine. Here is the data context for my application.
public class SimpleDataContext {
public List<NameValuePair> getInput() {
// in reality the data will be retrieve from a database or some datasource
List<NameValuePair> itemPairList = new ArrayList<NameValuePair>();
SimpleItem item1 = new SimpleItem();
item1.setName("item1");
item1.setPrice(50);
itemPairList.add(new NameValuePair(item1.getName(), item1));
SimpleItem item2 = new SimpleItem();
item2.setName("item2");
item2.setPrice(120);
itemPairList.add(new NameValuePair(item2.getName(), item2));
SimpleItem item3 = new SimpleItem();
item3.setName("item3");
item3.setPrice(130);
itemPairList.add(new NameValuePair(item3.getName(), item3));
return itemPairList;
}
public void setResult(Object result) {
if (!(result instanceof SimpleItem)) {
System.out.println("it is not a SimpleItem");
}
SimpleItem item = (SimpleItem)result;
System.out.println("Item: " + item.getName() + ", Price: " + item.getPrice());
}
}
And the Item I’m going to manipulate using rule is a simple bean like this,
public class SimpleItem {
String name;
int price;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getPrice() {
return price;
}
public void setPrice(int price) {
this.price = price;
}
}
4. Data Adapter
You have to adapt the input and output with the rule engine. Mostly here you only have to wrap the data context. The advantage of having the data adapter is, a data adapter always associated with a input/output type. So in the rule configuration I can provide the type for the input and output. If you see my rule configuration above, you see the input/output type is marked as “ItemData”. Here is my custom data adapter that is associated with the “itemData” type.
public class SimpleDataAdapter implements
ResourceAdapter, InputAdaptable, OutputAdaptable {
// the type associated with the adapter
private final static String TYPE = "itemData";
public String getType() {
return TYPE;
}
public Object adaptInput(ResourceDescription resourceDescription, Object tobeadapted) {
if (!(tobeadapted instanceof SimpleDataContext)) {
return null;
}
SimpleDataContext dataContext = (SimpleDataContext)tobeadapted;
return dataContext.getInput();
}
public boolean adaptOutput(ResourceDescription description,
Object value,
Object context,
ResourceHelper resourceHelper) {
if (!(context instanceof SimpleDataContext)) {
return false;
}
((SimpleDataContext)context).setResult(value);
return true;
}
public boolean canAdapt(ResourceDescription description, Object ouptput) {
String key = description.getKey();
return key != null && !"".equals(key);
}
}
5. Resource Helper
Resource Helper will map the keys refered from the configuration to JAVA objects. This is mostly used in mediation rule configurations which can extract the message data using a key or an xpath. In this example, we don’t have much keys refering from the configuration only the rule file and the data context.
public class SimpleResourceHelper extends ResourceHelper {
public ReturnValue findByKey(String key, Object source, Object defaultValue) {
if (!(source instanceof SimpleDataContext)) {
return new ReturnValue(defaultValue);
}
SimpleDataContext dataContext = (SimpleDataContext)source;
if (key.startsWith("file:")) {
String filename = key.substring("file:".length());
try {
BufferedInputStream in = new BufferedInputStream(new FileInputStream(filename));
return new ReturnValue(in);
} catch (Exception e) {
return new ReturnValue(defaultValue);
}
}
if (key.startsWith("dataContext")) {
return new ReturnValue(dataContext);
}
return new ReturnValue(defaultValue);
}
// there are few more methods to be implemented, which can just leave not implemented for this example
}
}
That is all the accessories. Now you will be able to write the rule engine execution code.
File ruleConfigFile = new File(ruleConfigFilename);
XMLStreamReader parser = XMLInputFactory.newInstance().createXMLStreamReader(new FileInputStream(ruleConfigFile));
//create the builder
StAXOMBuilder builder = new StAXOMBuilder(parser);
//get the root element (in this case the envelope)
OMElement ruleConfig = builder.getDocumentElement();
EngineConfiguration configuration =
new EngineConfigurationFactory().create(ruleConfig, new AXIOMXPathFactory());
EngineController
engineController = new EngineController(configuration, new SimpleResourceHelper());
final ResourceAdapterFactory factory = ResourceAdapterFactory.getInstance();
ResourceAdapter adapter = new SimpleDataAdapter();
String adapterType = adapter.getType();
if (!factory.containsResourceAdapter(adapterType)) {
factory.addResourceAdapter(adapter);
}
SimpleDataContext simpleContext = new SimpleDataContext();
if (!engineController.isInitialized()) {
engineController.init(simpleContext);
}
if (engineController.isInitialized()) {
engineController.execute(simpleContext, simpleContext);
}