Skip to main content

An implementation of DMN in Python. DMN rules are read from an Excel workbook

Project description

pyDMNrules

An implementation DMN (Decision Model Notation) in Python, using the pySFeel and openxlrd modules.

DMN rules are read from an Excel workbook. Then data, matching the input variables in the DMN rules is passed to the decide() function. The returned data contains the decion.

The passed Excel workbook must contain the two tabs 'Glossary' and 'Decision'. Other tabs contain as many DMN rules tables as necessary.

The 'Glossary' tab must contain a table headed 'Glossary'. This table must contain three columns with the headings 'Variable', 'Business Concept' and 'Attribute'. The data in these three columns describe the inputs and outputs associated with the 'Decision'. 'Variable' can be any text and is used to pass data to and from pyDMNrules. 'Business Concept' and 'Attribute' must be valid S-FEEL names, but may not contain the dot, or period character. Items in the Glossary can be outputs of one decision table and inputs in the next decision table, which makes it easy to support complex business models.

Glossary
Variable Business Concept Attribute
Customer Customer sector
OrderSize Order orderSize
Delivery delivery
Discount Discount discount

The 'Decision' tab must contain a table headed 'Decision' with the headings 'Decisions' and 'Execute Decision Table'. The row contain a description and the name of a DMN rules table. pyDMNrules will execute each decision table in this order.

Decision
Decisions Execute Decision Table
Determine Discount Discount

The Decision table can be preceded by 'Input' columns. The headings of each 'Input' column must be a Variable from the Glossary. The cells under day input heading are input test cells which must contain expresions about the input Variable, and which must evalutate to True, or False, depending upon the value of the input Variable. pyDMNrules will evalutate all the input test cells before running the associate DMN rules table. If all of the input test cells do not evauate to True, then pyDMNrules skip this DMN decision table and moves on to the next row in the Decision table. These input cell are evaluated on the fly, which means that a preceeding DMN rules table can set or clear values in the Glossary which can effect which DMN rules tables are included, and which DMN rules tables are excluded, in the final decision.

The DMN rules table(s), listed under Execute Decision Table, must exist on other spreadsheets in the Excel workbook.

Decisions as Rows DMN rules tables have columns of inputs and columns of outputs, with a double line vertial border deliniating where input columns stop and output columns begin. The headings for the input and output columns must be Variables from the Glossary. The input test cells under the input heading must contain expressions about the input Variable, which will evalutate to True or False, depending upon the value of the input Variable. The values in the output results cells, under the output headings, are the values that will be assinged to the output Variable, if all the input cells evaluate to True on the same row. ExampleRows.xlsx is an example fo a Decision as Rows DMN table.

Decisions as Columns DMN rules tables have rows of inputs and rows of outputs, with a double line horizonal border deliniating where input rows stop and output rows begin. The headings on the left hand side of the decision table, for the input and output rows, must be Variables from the Glossary. The input test cells across the row must contain expressions about the input Variable, which will evalutate to True or False, depending upon the value of the input Variable. The values in the output cells, across the row, are the values that will be assinged to the output Variable, if all the input cells evaluate to True in the same column. ExampleColumns.xlsx is an example fo a Decision as Columns DMN table.

Decisions as Crostab DMN rules tables have one output heading at the top left of the table. The horizontal and vertical headings, which must be Variables from the Glossary, are the input headings. Under the horizontal headings and besides the vertical headings are the input test cells. The output results cells form the body of the table and are assigned to the output Variable when the input cells in the same row and same column, all evaluate to True. ExampleCrosstab.xlsx is an example fo a Decision as Crosstab DMN table.

String:

Decision Model Notation specifies that strings must be enclosed in double quotes ("), as in "this is a string", or be italicized. pyDMNrules does not look for italicized text, but it does try to interpret uniquivical strings as string. So, HIGH,DECLINE are both strings; they are not number, nor dates, nor periods. pyDMNrules also recognizes entries from the Glossary, so Patient.age is not a string if 'Patient' is a Business Concept in the Glossary and 'age' is an attribute associated with the 'Patient' Business Concept in the Glossary. When any expression containing Patient.age is evaluated, Patient.age will be replaced with the current value associated with that Glossary entry. However, strings that may be ambiguous, must be enclosed in double quotes ("). Any string containing spaces is ambiguous and must enclosed in double quotes ("). All strings can be enclosed in double quotes, so HIGH,DECLINE and "HIGH","DECLINE" are equivalent.

USAGE:

import pyDMNrules
dmnRules = pyDMNrules.DMN()
status = dmnRules.load('ExampleHPV.xlsx')
if 'errors' in status:
    print('ExampleHPV.xlsx has errors', status['errors'])
    sys.exit(0)
else:
    print('ExampleHPV.xlsx loaded')

data = {}
data['Participant Age'] = 36
data['In Test of Cure'] = True
data['Hysterectomy Flag'] = False
data['Cancer Flag'] = False
data['HPV-V'] = 'V0'
data['Current Participant Risk Category'] = 'low'
print('Testing',repr(data))
(status, newData) = dmnRules.decide(data)
print('Decision',repr(newData))
if 'errors' in status:
    print('With errors', status['errors'])

For Single Hit Policy DMN rules tables newData will be a dictionary of the decision. For Multi Hit Policy DMN rules tables newData will be a list of dictionaries, each being the decision for the matching rule. The keys to each decision dictionary are

  • 'Result' - for a Single Hit Policy DMN rules table this will be the decision dictionary - a dictionary where the keys are every Variable from the Glossary and the matching the values are the final assigned value for each Variable after the decision was made. For a Multi Hit Policy DMN rules table this will be a list of decision dictionaries, one for each matched rule.
  • 'Excuted Rule' - for a Single Hit Policy DMN rules table this will be a tuple of Decision Table name and Rule number for the matching rule in that Decision Table. For a Multi Hit Policy DMN rules table this will be the a list of tuples, being the Decision Table name and Rule number for each matching rule.
  • 'DecisionAnnotations'(optional) - list of tuples (heading, value) of the annotations from the Decision table.
  • 'RuleAnnotations'(optional) - for a Single Hit Policy DMN rules table, this well be a list of tuples (heading, value) of the annotations for the matching rule, if there were any annotations for the matching rule. For Multi Hit Policy DMN rules table this will be a list of the lists of any annotations on any matching rule, where an empty list meant that the associated matching rule had no annotations.

If the Decision table contains multiple rows (multiple DMN rules tables run sequentially in order to make the decision) then the returned 'newData' will be a list of dictionaries, containing the keys 'Result', 'Excuted Rule', 'DecisionAnnotations'(optional) and 'RuleAnnotations'(optional), being one list entry for each DMN rules table used from the Decision table. The final enty in the list is the final decision. All other entries are the intermediate states involved in making the final decision.

$ python3 HPV.py
ExampleHPV.xlsx loaded
Testing {'Participant Age': 36, 'In Test of Cure': True, 'Hysterectomy Flag': False, 'Cancer Flag': False, 'HPV-V': 'V0', 'Current Participant Risk Category': 'low'}
Decision {'Immune Deficient Flag': None, 'Hysterectomy Flag': False, 'Cancer Flag': False, 'In Test of Cure': True, 'Participant Age': 36.0, 'Current Participant Risk Category': 'low', 'HPV-V': 'V0', 'Cyto-S': None, 'Cyto-E': None, 'Cyto-O': None, 'Collection Method': None, 'Test Risk Code': 'L', 'New Participant Risk Category': 'low', 'Participant Care Pathway': 'toBeDetermined', 'Next Rule': 'CervicalRisk2', 'Execute Rules': [('FirstTestOfCervicalRisk', '20')]}

Testing

pyDMNrules reserves the spreadsheet name 'Test' which can be used for storing test data. The function test() read the test data, passes it through the decide() function and assembles the results which are returned to the caller.

The 'Test' spreadsheet must contain unit test data tables. Each unit test data table must be labeled with a Business Concept from the Glossary. The heading of the table must be Variables from the Glossary associated with the Business Concept. The data, in each column, must be valid input data for the associated Variable. Each row is considered a unit test data set.

pyDMNrules also searches for a special table which must be called 'DMNrulesTests' which must exist. The 'DMNrulesTests' table has input columns followed by output columns, with a double vertical line border between them. The headings for the input columns must be Business Concepts from the Glossary. The data in input columns must be one based indexes into the matching unit test data table. The headings for the output columns must be Variables from the Glossary. The data in output columns will be valid data for the matching Variable and are the expected values returned by the decide() function.

The test() function will process each row in the 'DMNrulesTests' table, getting data from the unit test data tables and building a data{} dictionary. The test() function then passes this data{} dictionary to the decide() function. The returned newData{} is then compared to the output data for the matching test in the 'DMNrulesTests' table. The data{}, newData{} and a list of mismatched is then appended to the list of results, which is eventually passed back to the caller of the test() function.

The returned list is a list of dictionaries, one for each test in the 'DMNrulesTests' table. The keys to this dictionary are

  • 'Test ID' - the one based index into the 'DMNrulesTests' table which identifies which test which was run
  • 'TestAnnotations'(optional) - the list of annotation for this test - not present if no annotations were present in 'DMNrulesTests'
  • 'data' - the dictionary of assembed data passed to the decide() function
  • 'newData' - the decision dictionary returned by the decide() function [see above]
  • 'DataAnnotations'(optional) - the list of annotations from the unit test data tables, for the unit test sets used in this test - not present if no annotations were persent in any of the unit test data tables for the selected unit test sets.
  • 'Mismatches'(optional) - the list of mismatch reports, one for each 'DMNrulesTests' table output value that did not match the value returned from the decide() function - not present if all the data returned from the decide() function matched the values in the 'DMNrulesTest' table.

Note

If an output should be the value of an input, then you can use the Variable from the Glossary. However, if the output is a manuipulation of an input, then you will need to use the internal name for that variable, being the Business Concept and the Attibute concateneted with the period character. That is 'Patient Age' is valid, but 'Patient Age + 5' is not. Instead you will need to use the syntax 'Patient.age + 5'

USAGE:

import pyDMNrules
dmnRules = pyDMNrules.DMN()
(testStatus, results) = dmnRules.test()
for test in range(len(results)):
    if 'Mismatches' not in results[test]:
        print('Test ID', results[test]['Test ID'], 'passed')
    else:
        print('Test ID', results[test]['Test ID'], 'failed')

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

pyDMNrules-0.0.2.tar.gz (18.4 kB view hashes)

Uploaded Source

Built Distribution

pyDMNrules-0.0.2-py3-none-any.whl (29.4 kB view hashes)

Uploaded Python 3

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page