This document is an advanced user guide that shows how to use the more complex features of the Natural Rule Language (NRL). The Natural Rule Language is an open specification hosted on SourceForge, at http://nrl.sourceforge.net. Its principal purpose is to enable the expression of constraints over models in a restricted language that is reasonably close to English.
The web site contains some related documents, most importantly the language specification. It also contains a collection of example rule sets that can provide further insight into the language.
This document is under copyright ©2006-2010 by its authors:
You will know from the basic user guide or your experience with the NRL that NRL rules are written against a model, such as UML or XML Schema. In this advanced guide we will use the same example model as in the basic guide. If you require an explanation of the meaning, please refer to the model section in the basic user guide.
Let's now add some more advanced rules to constrain this model. The following rules are complete and correct and could be used in a project. They make use of various features that will be discussed below:
Model
"userguide.uml2"
Context:
Settlement
Validation Rule "v1"
code represents
settlementCurrency.ccyCode,
if code =
'GBP' or code = 'EUR' then
tax = 0.175 otherwise tax =
0
Context: Settlement
Validation Rule "v2"
code represents settlementCurrency.ccyCode,
type represents transaction.type, if the type
is equal to 'PURCHASE' then the code is
equal to 'GBP'
Context: Settlement ("settlement")
Validation Fragment "taxable"
settlement.settlementCurrency.ccyCode is one of
'GBP', 'EUR'
Context: Settlement
Validation Rule "p1"
If the Settlement is taxable then
netAmount = grossAmount * (1 + tax) else
netAmount = grossAmount
Context: Settlement
Validation Rule "o1"
transaction.date [is not a holiday]
When computing values or performing complex navigations along a model, it is sometimes useful to hold part of a rule in a variable, to prevent any repetition. It can also be useful to declare a variable that has a more meaningful name than a model attribute in a given context:
Context: Settlement
Validation Rule "v1"
code represents
settlementCurrency.ccyCode,
if code =
'GBP' or code = 'EUR' then
tax = 0.175 otherwise tax =
0
Here, in order to avoid having to type settlementCurrency.ccyCode twice, we have assigned it to a variable 'code'. We can then make use of that variable in a rule as normal. Note the following conventions:
Context: Settlement
Validation Rule "v2"
code represents settlementCurrency.ccyCode, type represents
transaction.type, if the type
is equal to 'PURCHASE' then the code is
equal to 'GBP'
It sometimes happens that we want to check that a condition holds for a particular context in multiple rules. In our example model, a transaction is taxable if it occurs with a currency of "GBP" or "EUR" and we may have several rules that hold only for taxable transactions. It would be tedious to have to write out the currency check over and over again - instead we define a validation fragment called "taxable", which we can then make use of in multiple rules:
Context:
Settlement ("settlement")
Validation Fragment "taxable"
settlement.settlementCurrency.ccyCode is one of
'GBP', 'EUR'
Context:
Settlement
Validation Rule "p1"
If the Settlement is taxable then
netAmount = grossAmount * (1 + tax) else
netAmount = grossAmount
A fragment has to be applied to some attribute or to the context. In this rule, it is being applied to the context, Settlement. The "is" keyword is optional - if it was part of the fragment name, we would omit it. Here is a summary of how to refer to fragment:
NRL Text | Comments |
---|---|
attribute is "property" | The attribute must be an attribute of the context, or refer to the context itself |
attribute has "property" | Alternative to "is" |
attribute "property" | If the property name contains "has" or "is" or a similar verb |
The Natural Rule Language, as a restricted language, will always be too limited to express some types of rules. Frequently seen examples that come to mind include:
Context:
Settlement
Validation Rule "o1"
transaction.date [is not a holiday]
In this rule we have referenced an operator "is not a holiday", in square brackets, and the date given before it is passed as a parameter. The operator will check the date for us.
At this stage, one wonders how the operators actually get implemented. We make a distinction here between rule writers, the people charged with writing rules - who could be business analysts or technical analysts - and implementors charged with an eventual implementation. This distinction is necessary and useful, because some rules are provided only for documentation and never actually get implemented.
The NRL specification does not state how operators like "is not a holiday" actually map to a target implementation. If NRL rules are translated to Java, they may be implemented as helper methods in some class, in a rule engine they may be scripted. The mapping implementation will hold further details in these cases.
The following is a summary of the permissible operator syntax:
NRL Text | Comments |
---|---|
attribute [applied operator] | For example, "date [is not a holiday]" |
[applied operator] attribute | For example, "[there is a] currency" |
attribute [operator] attribute | For example, "dateA [is in the same year as] dateB" |
In order to see some examples of complex rules, visit the NRL Home Page. Very interested rule writers may want to look at the specification, which covers the language in pain-staking detail and provides a full grammar.
©2006-2010 The Authors.