Saturday, September 14, 2013

Oracle Business Rules 11g - Best Practices for Decision Tables

Oracle Business Rules is a lightweight and powerful rules engine that is part of the Oracle SOA Suite. It supports two fundamental ways of entering the rules:
  • an IF/THEN form which is the intrinsic form of rules in any production rule system
  • a decision table - a spreadsheet-like form that will appeal a business savvy user

While Oracle does a pretty good job on documenting the decision table features - Working with Decision Tables - there is much less information available on how to actually design the decision tables.

In this post, I want to share my experience with the decision tables and describe what I consider the best practices in using them. I am looking forward to your feedback in the comments section.
  
Have a Case
The first and foremost question you should be asking yourself before you create a decision table is, "is the decision table the right vehicle for this job?". Sometimes this may be a no-brainer - like when your customer gives you a spreadsheet that neatly translates into a decision table. Other times the choice may not be that obvious.

Let's have a look at an example. Imagine a company that grants a company car to an employee if any of the following applies:
  • The employee's job level is expert and the employee has been at least 3 years with the company
  • The employee's yearly salary is at least 65,000 USD and the employee was hired before Nov 1, 2008
  • The employee's job title is Sales Rep

It is questionable though whether a decision table is a good fit here. While it is still appealing thanks to its visually comprehensive layout, a solution with IF/THEN rules is going to be simpler and cleaner, especially if the number of rules increases in the future.
Now lets take a look at another imaginary company car policy:
  • All employees in the Sales department are entitled to a company car.
  • Employees in the Executive department are entitled to a company car if their job level is intermediate or expert.
  • Employees in the IT department are entitled to a company car if their job title is Architect and their job level is expert.

Even though this policy has the same number of  conditions as the first one, it yields much more concise and coherent decision table. Why is that? The first policy uses disconnected conditions which do not blend very well into a decision table. The second policy has conditions shared by most of the rules which differ only by the condition input values.

My advice is to consider the structure and size of your to-be-implemented rule set, then take a mental picture of the decision table solutions and compare it to the IF/THEN rule solution.

Only use a decision table if it gives you an advantage compared to an IF/THEN rule solution.

Be Accurate
I mentioned above that you may get a spreadsheet from your customer that translates neatly into a decision table. In reality though, it's rarely that simple. More often than not, the specification will contain a few stumbling blocks. I once received a spreadsheet from the customer that contained two overlapping conditions which were supposed to yield opposite results. The good news is, if you accurately translate your specification into a decision table, the logical flaws in the specification will pop up as conflicts or gaps. This way the implementation can actually help improve the specification - isn't it amazing? On top of that, the more your decision table resembles the specification, the easier it will be to implement any future specification changes. For all above reasons:

Make your decision table match your specification as closely as possible.

Be Complete
When you look at either of the above decision tables, you will notice that they capture only the cases when the company car privilege is granted to the employee. I did it on purpose to keep the tables as simple as possible. In my experience the real world specification also tends to come that way - as an enumeration of cases when something happens or applies - rather than as a full list of all eventualities. Yet it pays off to consider all possible cases - and there is a nifty tool built in the Rules Designer that does exactly that - the Gap Analysis.

  
The above picture shows the gap analysis for the decision table implementing the second company car policy. I like to see it as an inverse version of the original decision table - as it is covering all the cases when the company car is not granted. It is very useful to examine the gaps to verify the completeness of a decision table. For instance, the fact that the Finance department is missing from the rules in our example may be a hint that the specification is incomplete (unless the company really hates their finance folks).

You may also choose to have the gaps automatically filled in - this is how I created the following gapless decision table from the original one:


Even when I choose to allow gaps in a decision table, I still employ the Gap Analysis to check that no eventuality has been overlooked.

Gap Analysis is your friend - seek its advice whenever you create or change a decision table.

Divide and Conquer
I bet you've seen this before - a source code method or function in your favorite programming language that spans across hundreds of lines. This is the infamous Taller Than Me anti-pattern. Unfortunately, decision tables are not immune to a similar anti-pattern. It is created by the same evil code-supersizing forces, but as a decision table tends to grow wide as more rules are added into it, let me dub it "Wider Than The Sky".

If you have to scroll horizontally to see all of your decision table, it's probably a good idea to split the table into several smaller partitions. To illustrate this, let's imagine that our fictitious company determines salary raises for its employees based solely on their job title and the last performance rating:


Huh? You can't see a thing? That's right - the table has grown too wide! Arguably the best refactoring  in this case is to have a dedicated table for every job title - here are two examples (check out the project from Subversion if you want to see more):



 Avoid creating decision tables that are Wider Than The Sky.

Be Consistent
The bucketsets used in decision tables can be defined either locally or globally. It may be tempting to quickly hack a local bucketset with just few values that you need (for example if a condition needs to check for two specific departments only), however in my experience you are almost always better off with a global bucketset that contains a complete list of values. The reasons are threefold:
  • Gap Analysis is less reliable with partial bucketsets.
  • It is very confusing when multiple local bucketsets with different sets of values are defined for the same entity.
  • If you create a local bucketset, you will face rework if you need to add another condition for the same entity later.

 Avoid local bucketsets unless you have a really good reason to use them.

Control the Conflicts
If a decision table has overlapping rules that yield different results it will be marked as a conflict. Note that the decision table from the very first example has overlapping rules but there is no conflict as the corresponding actions do not differ.

This is the reason why you should avoid unnecessary stuff in your decision table actions. For many years I believed that it is a good practice to add a print action to debug which rule was actually fired. Now I don't think much about it as it is:
  • Creating unnecessary conflicts
  • Obscuring the business meaning of the rules
  • Completely redundant - the rules that fired can always be tracked by other means (the server audit trail, test rule  function or explicitly by calling RL.watch.rules) 

If you do have a genuine conflict though, it's time to check your specification. If you modeled the rules closely to the specification, as I advocated in the Be Accurate section, then actually the specification itself must have a conflict. Your next course of action then depends on how the specification is fixed:
  • A business user may fix the rules in the specification which in turn means a refactoring of the decision table.
  • A business user may prioritize the conflicting rules - which is best implemented as manual conflict resolution.
    • As a special case, a business user may implicitly assume that a more specific rule has higher priority than a less specific one. That's actually quite reasonable and it is supported by the Rules Designer as the auto override conflict policy. You just need to understand your business' assumptions or even better, let them articulate the assumptions explicitly.

For details on manual and auto override conflict resolution see Understanding Decision Table Conflict Analysis.

Avoid conflicts in a decision table by keeping the action part uniform for overlapping rules. For genuine conflicts let the specification drive the decision between conflict resolution and refactoring the decision table.

Conclusion
I hope you find the above useful. If you have already developed your own best practices I'd be happy if you leave a comment and share how they compare with these. If you want to try the examples, you can get the project from Subversion or download it as a 7-zip archive.

6 comments:

  1. Nice clarifying blog on when to use what kind of rules type. I especially like the "wider than the sky" refactoring rule. You find violations on this rule everywhere on modern polyglot oriented platforms ... for example, think of large, unreadable, monolithic BPEL or BPMN processes. Often business users want to stay on Excel spreadsheets, therefore a automatic import or better roundtrip approach would be helpful (should be on the feature list of Oracle). Again, great blog!

    ReplyDelete
  2. Excellent article. Thanks for sharing.

    ReplyDelete
  3. Hi Dusan,

    Nice article. I was just wondering how to call another decisionfunction in an action as part of a decision table. In your example, you split the SalaryRaise Decision table in multiple decision tables.....what I can't figure out though is how to go from one dec. table to the other one. I was hoping that I could see this in you example but in there I just see the complete one and the splitted ones....but nowhere a call.

    I'm trying this now by using something like
    RL.list.get(MY-SUB-DecisionFunction(Request),1).toString()

    but this gives me the error: The RL function run, step or runUntilHalt was invoked from a rule action at line.....

    Do you know perhaps how to do this. I also cannot find this anywhere in de Oracle Documentation.

    Thanks in advance and best regards,
    Hugo

    ReplyDelete
    Replies
    1. Hugo, to go from one decision table to another one you need to do the same thing as when you want to go from one rule to another one: in the first rule (or a decision table action) assert a new fact that is used in the IF part of the second rule (or in the condition part of a second decision table).

      This kind of rule chaining is commonplace in real world rule sets that are more complex than the examples in my blog post.

      Delete
    2. Hi Dusan,
      Thanks for the input. That was just the little boost in the right direction to solve it. I got it to work now. Thanks a bunch!

      Best regards,
      Hugo

      Delete
  4. Very good explanation, sometimes we foget to create easy code.

    We have to remember is that our job is to simplify complex real world rules into comprensible ones.

    ReplyDelete