Skip to content

global-soft-ba/decisionTable

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Decision Table

This library enables the creation and conversion of decision tables in Golang programming language. Inspired by the JBOSS Drools or Camunda DMN Suite. Decision tables are commonly used or rather visualized as a table in a UI to represent complex decisions in a more human-readable way. The library can be used as a data representation in Golang for such frontend/UI components.

Each table represents a complex system of rules. Such systems describe a decision for which input, which output is to be generated. The library allows:

  • The creation of such rules
  • Data-type dependent validation of rule expression
  • Converting into a rule-engine format (to finally execute rules)

Table Example

Image of Decision Table

Decision Table Builder (Code Example)

table, _ := decisionTable.NewDecisionTableBuilder().
    SetID("determineEmployee").
    SetName("Determine Employee").
    SetHitPolicy(hitPolicy.Unique).
    SetExpressionLanguage(expressionLanguage.SFEEL).
    SetStandard(standard.GRULE).
    AddInputField(field.Field{Name: "Claim.TypeOfClaim", Type: dataType.String}).
    AddInputField(field.Field{Name: "Claim.ExpenditureOfClaim", Type: dataType.Integer}).
    AddOutputField(field.Field{Name: "Employee.ResponsibleEmployee", Type: dataType.String}).
    AddOutputField(field.Field{Name: "Employee.FourEyesPrinciple", Type: dataType.Boolean}).
    AddRule(decisionTable.NewRuleBuilder().
        SetAnnotation("R1").
        AddInputEntry(`"Car Accident"`).
        AddInputEntry("<1000").
        AddOutputEntry(`"Müller"`).
        AddOutputEntry("false").
        Build(),
    ).
    AddRule(decisionTable.NewRuleBuilder().
        SetAnnotation("R2").
        AddInputEntry(`"Car Accident"`).
        AddInputEntry("[1000..10000]").
        AddOutputEntry(`"Schulz"`).
        AddOutputEntry("false").
        Build(),
    ).
    AddRule(decisionTable.NewRuleBuilder().
        SetAnnotation("R3").
        AddInputEntry("-").
        AddInputEntry(">=10000").
        AddOutputEntry("-").
        AddOutputEntry("true").
        Build(),
    ).
    Build()
    
rules, _ := table.Convert(standard.GRULE)
fmt.Print(rules)

We assume that the frontend will represent a decision table as a kind of table. In case, that a user changes something in the frontend table, we assume that the old decision table representation will be dropped and then rebuild from the new (frontend) table (which was manipulated by the user). By doing so, we don't need any "insert function" for rules/inputs/outputs etc and so it keeps the DecisionTable-Builder simple.

Hit Policies and Collect Operators

A decision table consists of several rules, typically represented as rows. When reading such a row we look at certain input values and deduct a certain result represented by output values.

When using the simplest hit policy "unique" (U), such rules do not overlap: only a single rule must match.

Now consider that we build a decision table with overlapping rules. In other words that means more than one rule may match a given set of input values. We then need one of the alternative hit policy indicators to unambiguously understand the decision logic according to which such rules are interpreted.

Single Decision Tables

Such tables either return the output of only one rule or aggregate the output of many rules into one result. The hit policies to be considered are

  • Unique: Rules do not overlap. Only a single rule can match.
  • First: Rules are evaluated from top to bottom. Rules may overlap, but only the first match counts.
  • Priority: Rule outputs are prioritized. Rules may overlap, but only the match with the highest output priority counts.
  • Any: Multiple matching rules must not make a difference: all matching rules must lead to the same output.
  • Collect: The output of all matching rules is aggregated by means of an operator:
    • Sum: Sums/Add all outputs of the matching rule’s distinct outputs.
    • Minimum: Take the smallest value of all the matching rule’s outputs.
    • Maximum: Take the largest value of all the matching rule’s outputs.
    • Number: Return the number of all the matching rule’s distinct outputs.

Multiple Result DecisionTable

Multiple result tables may return the output of multiple rules. The hit policies for such tables are:

  • Collect: All outputs of the matching rules are combined in an arbitrarily ordered list of all the output entries.
    • List: Collect all outputs of all matching rules in a list of output entries
  • Rule Order: All outputs of the matching rules are combined in a list of outputs ordered by the sequence of those rules in the decision table.
  • Output Order: All outputs of the matching rules are combined in a list of outputs ordered by their (decreasing) output priority.

Examples

Examples can be found here: Camunda Page

Supported Variable Types

The allowed types of input or output variables differs on the selected notation standard of a decision table:

Variable Types DMN Notation GRL Notation
Boolean X X
String X X
Integer X X
Float X
Long X
Double X
DateTime X X

More Details can be found here:

Supported Entry Languages

The expresion language defined the functions and comparisons of a single rule entry. It depends on the chosen table notations standard. So far we support the SFEEL Standard partially.

SFEEL INPUT EXPRESSIONS

Converter for Standards/Engines

The following converter are supported, so far: