Python DSL for setting up business intelligence rules
Project description
business-rule-engine
As a software system grows in complexity and usage, it can become burdensome if every change to the logic/behavior of the system also requires you to write and deploy new code. The goal of this business rules engine is to provide a simple interface allowing anyone to capture new rules and logic defining the behavior of a system, and a way to then process those rules on the backend.
You might, for example, find this is a useful way for analysts to define marketing logic around when certain customers or items are eligible for a discount, or to automate emails after users enter a certain state or go through a particular sequence of events.
Usage
1. Define your variables
Variables represent values in your system, usually the value of some particular object. You create rules by setting threshold conditions such that when a variable is computed that triggers the condition, some action is taken.
params = {
'products_in_stock': 10
}
2. Define custom functions
def order_more(items_to_order):
print("you ordered {} new items".format(items_to_order))
return items_to_order
3. Write the rules
Rules use standard Python expression syntax. The when block supports multi-line expressions with and/or — lines are joined and evaluated as a single Python expression. Each then line is an action executed in order.
rules = """
rule "order new items"
when
products_in_stock < 20
then
order_more(50)
end
"""
4. Create the parser and execute
from business_rule_engine import RuleParser
parser = RuleParser()
parser.register_function(order_more)
parser.parsestr(rules)
result = parser.execute(params)
if result:
print("A rule was triggered")
Multiple conditions and multiple actions
The when block uses standard Python boolean syntax. Multiple lines are joined into a single expression, so and/or work exactly as in Python. Each line in the then block is a separate action executed in order.
rules = """
rule "standard reorder"
when
products_in_stock < 20
and margin > 0.3
then
order_more(50)
notify_purchasing()
end
rule "urgent reorder"
when
products_in_stock < 5
or products_reserved > 100
then
order_more(200)
notify_manager()
end
"""
Custom functions
You can register your own functions to use in conditions and actions:
from business_rule_engine import RuleParser
def is_even(num):
return (num % 2) == 0
params = {
'number': 10
}
rules = """
rule "check even number"
when
is_even(number) == True
then
print("is even")
end
"""
parser = RuleParser()
parser.register_function(is_even)
parser.register_function(print)
parser.parsestr(rules)
parser.execute(params)
You can also register a function under a different name for use in rules:
parser.register_function(is_even, "even") # use as even(number) in rules
## Rule options
### Priority
Rules with a higher priority are evaluated first. The default priority is `0`. Rules with equal priority are evaluated in the order they were added.
Priority can be set on the rule line or as a separate keyword:
rule "urgent reorder" priority 10 when products_in_stock < 5 then order_more(200) end
rule "standard reorder" priority 5 when products_in_stock < 20 then order_more(50) end
Priority can also be set when using `add_rule()`:
```python
parser.add_rule("urgent reorder", "products_in_stock < 5", "order_more(200)", priority=10)
Description
Rules can carry a human-readable description:
rule "standard reorder"
description "Triggers a standard reorder when stock falls below 20 units"
when
products_in_stock < 20
then
order_more(50)
end
Allowing non-boolean conditions
By default, the parser raises ConditionReturnValueError if a when expression does not evaluate to a bool. Set condition_requires_bool=False to accept any truthy/falsy value instead:
parser = RuleParser(condition_requires_bool=False)
This can also be set per rule:
rule = Rule("my rule", condition_requires_bool=False)
Enabling and disabling rules
Rules can be disabled at runtime without removing them from the parser:
parser.rules["standard reorder"].enabled = False
Disabled rules are skipped during execute() and do not appear in the execution results.
Removing and counting rules
Check whether a rule exists and how many rules are loaded:
"standard reorder" in parser # True / False
len(parser) # number of registered rules
Remove a single rule or all rules at once:
parser.remove_rule("standard reorder") # raises KeyError if not found
parser.clear_rules() # removes all rules
Managing custom functions
Remove a previously registered function or clear all of them:
RuleParser.unregister_function("order_more") # raises KeyError if not found
RuleParser.clear_functions() # removes all custom functions
Note: CUSTOM_FUNCTIONS is shared across all parser instances, so clear_functions() affects every instance.
Processing all matching rules
By default, execute() stops after the first rule whose condition is satisfied (stop_on_first_trigger=True). Set it to False to evaluate every enabled rule regardless:
result = parser.execute(params, stop_on_first_trigger=False)
for r in result.results:
if r.triggered:
print(f"{r.rule_name}: {r.action_result}")
This is useful when multiple independent rules may apply to the same input.
Loading rules from a file
Use parsefile() to load rules directly from a file:
parser = RuleParser()
parser.register_function(order_more)
parser.parsefile("rules/reorder.rules")
parser.execute(params)
Accessing execution results
execute() returns an ExecutionResult object that behaves like a bool but also gives you access to the result of each rule:
from business_rule_engine import RuleParser
def order_more(items_to_order):
return "you ordered {} new items".format(items_to_order)
def notify_purchasing():
return "purchasing notified"
rules = """
rule "standard reorder"
when
products_in_stock < 20
then
order_more(50)
notify_purchasing()
end
"""
parser = RuleParser()
parser.register_function(order_more)
parser.register_function(notify_purchasing)
parser.parsestr(rules)
result = parser.execute({'products_in_stock': 10})
for r in result.results:
if r.triggered:
print(f"{r.rule_name}: {r.action_result}")
# → standard reorder: ['you ordered 50 new items', 'purchasing notified']
Each entry in result.results is a RuleResult with these fields:
| Field | Type | Description |
|---|---|---|
rule_name |
str |
Name of the rule |
triggered |
bool |
Whether all conditions evaluated to True |
condition_result |
bool |
Result of the condition evaluation |
action_result |
list[object] |
Return values of each action, or [] if not triggered |
Handle missing rule parameters
If a required argument is missing, the rule engine raises a MissingArgumentError.
For cases where you work with incomplete data, you can provide a default value:
params = {}
parser = RuleParser()
parser.register_function(order_more)
parser.parsestr(rules)
parser.execute(params, set_default_arg=True, default_arg=0)
More control of the RuleParser
If you need full control over rule execution, you can iterate over the parser and execute each rule individually:
from business_rule_engine import RuleParser
from business_rule_engine.exceptions import MissingArgumentError
def order_more(items_to_order):
return "you ordered {} new items".format(items_to_order)
rules = """
rule "order new items"
when
products_in_stock < 20
then
order_more(50)
end
"""
params = {'products_in_stock': 10}
parser = RuleParser()
parser.register_function(order_more)
parser.parsestr(rules)
for rule in parser:
try:
condition_result, action_results = rule.execute(params)
if rule.status:
print(action_results[0])
break
except MissingArgumentError:
pass
Error Handling
from business_rule_engine import RuleParser
from business_rule_engine.exceptions import MissingArgumentError
def order_more(items_to_order):
return "you ordered {} new items".format(items_to_order)
# intentional typo
params = {
'produtcs_in_stock': 30
}
rules = """
rule "order new items"
when
products_in_stock < 20
then
order_more(50)
end
"""
parser = RuleParser()
parser.register_function(order_more)
parser.parsestr(rules)
try:
result = parser.execute(params)
if not result:
print("No conditions matched")
except MissingArgumentError as e:
print(e)
Debug
To debug the rules processing, use the logging module:
import sys
import logging
logging.basicConfig(stream=sys.stdout, level=logging.DEBUG)
Migration Guide
From 0.x to 1.x
Version 1.0 is a major rewrite with several breaking changes.
Rule expression syntax
Rules no longer use Excel-style syntax. All expressions are plain Python:
| 0.x (Excel syntax) | 1.x (Python syntax) |
|---|---|
AND(a, b) |
a and b |
OR(a, b) |
a or b |
NOT(a) |
not a |
a = b |
a == b |
Multiple when lines
Multiple when lines are no longer treated as independent conditions joined with AND.
They are concatenated into a single Python expression, so you must write the operators explicitly:
# 0.x — implicit AND between lines
when
products_in_stock < 20
margin > 0.3
# 1.x — explicit operator required
when
products_in_stock < 20
and margin > 0.3
action_result is now a list
Each then line is a separate action. action_result is now list[object] instead of a single value:
# 0.x
condition_result, action_result = rule.execute(params)
print(action_result) # single value
# 1.x
condition_result, action_results = rule.execute(params)
print(action_results[0]) # first action result
print(action_results) # all action results
Renamed and restructured exceptions
The exception base class and one exception were renamed for consistency with Python naming conventions:
| 0.x | 1.x |
|---|---|
RuleParserException |
RuleParserError |
DuplicateRuleName |
DuplicateRuleNameError |
A new exception DuplicateThenError (subclass of RuleParserSyntaxError) is raised when a rule block contains more than one then section.
Boolean parameters are now keyword-only
All boolean and optional parameters must be passed as keyword arguments:
# 0.x
parser = RuleParser(False)
rule = Rule("name", False, 10, False)
parser.execute(params, False)
# 1.x
parser = RuleParser(condition_requires_bool=False)
rule = Rule("name", condition_requires_bool=False, priority=10, enabled=False)
parser.execute(params, stop_on_first_trigger=False)
This also applies to add_rule():
# 0.x
parser.add_rule("name", "cond", "action", 10, False, "desc")
# 1.x
parser.add_rule("name", "cond", "action", priority=10, enabled=False, description="desc")
CUSTOM_FUNCTIONS is now a class-level dict
In 0.x, functions were registered as a list of strings. In 1.x, CUSTOM_FUNCTIONS is a
dict[str, Callable] (name → callable) and is shared across all parser instances.
Use register_function() to add functions — do not modify CUSTOM_FUNCTIONS directly.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file business_rule_engine-1.0.0.tar.gz.
File metadata
- Download URL: business_rule_engine-1.0.0.tar.gz
- Upload date:
- Size: 11.3 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
8e445106063f466e6d49e385ad0fe9b0f530703e1a2aeb5ed5ffe814b7739c84
|
|
| MD5 |
f512fbb2dcc4dc8442e275378be9baba
|
|
| BLAKE2b-256 |
daec18a3a7bb794b39f727ff109a3692e1758fefc45d672b30a48cf917b65071
|
Provenance
The following attestation bundles were made for business_rule_engine-1.0.0.tar.gz:
Publisher:
python-publish.yml on manfred-kaiser/business-rule-engine
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
business_rule_engine-1.0.0.tar.gz -
Subject digest:
8e445106063f466e6d49e385ad0fe9b0f530703e1a2aeb5ed5ffe814b7739c84 - Sigstore transparency entry: 1894543419
- Sigstore integration time:
-
Permalink:
manfred-kaiser/business-rule-engine@cff925739059bb15d7633721521bac61e3fe2ce2 -
Branch / Tag:
refs/tags/1.0.0 - Owner: https://github.com/manfred-kaiser
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
python-publish.yml@cff925739059bb15d7633721521bac61e3fe2ce2 -
Trigger Event:
release
-
Statement type:
File details
Details for the file business_rule_engine-1.0.0-py3-none-any.whl.
File metadata
- Download URL: business_rule_engine-1.0.0-py3-none-any.whl
- Upload date:
- Size: 13.4 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
df58576ac9edba265639dd4ba016c22f65c0585602a9b69205e983614ef36cfa
|
|
| MD5 |
8fe46b6198d28f4f5d207f81c7864bea
|
|
| BLAKE2b-256 |
8752e36e032c6802b66196ae055cd2d34dad452beb29c26ef8ad76fc90132c25
|
Provenance
The following attestation bundles were made for business_rule_engine-1.0.0-py3-none-any.whl:
Publisher:
python-publish.yml on manfred-kaiser/business-rule-engine
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
business_rule_engine-1.0.0-py3-none-any.whl -
Subject digest:
df58576ac9edba265639dd4ba016c22f65c0585602a9b69205e983614ef36cfa - Sigstore transparency entry: 1894543501
- Sigstore integration time:
-
Permalink:
manfred-kaiser/business-rule-engine@cff925739059bb15d7633721521bac61e3fe2ce2 -
Branch / Tag:
refs/tags/1.0.0 - Owner: https://github.com/manfred-kaiser
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
python-publish.yml@cff925739059bb15d7633721521bac61e3fe2ce2 -
Trigger Event:
release
-
Statement type: