Skip to content

Latest commit

 

History

History
168 lines (130 loc) · 4.37 KB

README.md

File metadata and controls

168 lines (130 loc) · 4.37 KB

pyrules

Python Rules Engine

This is a first go at implementing a generic Rule Engine in Python. Its a working solution, but it is not ready for large-scale or even small-scale production use. Use at your own risk. Contributions welcome.

What you can do with it

See sample.py for a complete sample.

Basic example

  1. Define simple rules in plain Python
class CalculateBasicFare(Rule):
    def should_trigger(self, context):
        return True
    def perform(self, context):
        context.fare = context.distance * 20
        return context.fare
        
class CalculateWeekendFare(Rule):
    def should_trigger(self, context):
        return context.weekend 
    def perform(self, context):
        context.fare = context.fare * 1.2
        return context.fare 

Or even simpler, using a LambdaRule:

class LambdaRuleSample(LambdaRule):
    condition = lambda self, context: not context.weekend
    action = lambda self, context: { 'fare' : context.fare * 4 }
  1. Define a ruleset
ruleset = (
  CalculateBasicFare(),
  CalculateWeekendFare(),
)
  1. Define the context

The rule context is the set of data the rules will operate on. Within a rule, the context is available as the context variable.

context = RuleContext()
context.distance = 10
context.weekend = 0
context.sunday = 1
  1. Execute the rules

This will execute the rules defined in the ruleset. The results will be stored in the context (it is up to the rules to set the respective context attribute).

engine = RuleEngine()
context = engine.execute(ruleset, context)
print "fare", context.fare
print "executed", context._executed
=>
fare 300.0
executed [('CalculateBasicFare', 200), ('TableRuleset.1', 200), ('TableRuleset.2', 300.0), ('TableRuleset', True)]

Advanced features

SequencedRuleset

A SequencedRuleset ensures that rules are executed in sequence:

ruleset = (
   SequencedRuleset([CalculateBasicFare(), CalculateWeekendFare()]),
)

TableRuleset

A TableRuleset is defined by a list of rules given in the following form. Note that the list of conditions (if) are must all be met for the rule to trigger (logical AND). If the rule triggers, each action i ( then) is executed and its result stored in the corresponding ith context variable (target).

manyrules = TableRuleset([
  { 'if': ['not weekend'],
    'then' : ['fare * 1'],
    'target' : ['fare'],
  },
  { 'if': ['not sunday'],
    'then' : ['fare * 1.5'],
    'target' : ['fare'],
  }
])

Add manyrules to the ruleset, just as you would with any other Rule instance:

ruleset = (CalculateBasicFare(),
           CalculateWeekendFare(),
           manyrules,
)

Natural language rules

TableRuleset offers a translation feature that can translate natural-language rules (somewhat) into executable rules:

translations=[
  ('the weather', 'context.weather'),
  ('is', '=='),
  ('nice', '"nice"'),
  ('Fare', 'context.fare'),
  ('weekend', 'context.weekend'),  
  ('not', 'not'),
  ('on Sunday', 'context.sunday'),
]

manyrules = TableRuleset([
  { 'if': ['not weekend', 'the weather is nice'],
    'then' : ['Fare * 1'],
    'target' : ['Fare'],
  },
  { 'if': ['on Sunday', 'the weather is nice'],
    'then' : ['Fare * 1.5'],
    'target' : ['Fare'],
  }
], translations=translations)

Execute:

context = RuleContext()
context.distance = 10
context.weekend = 0
context.weather = 'nice'
context.sunday = 1
ruleset = (CalculateBasicFare(),
           CalculateWeekendFare(),
           manyrules,
)
=> 
fare 300.0
executed [('CalculateBasicFare', 200), ('TableRuleset.1', 200), ('TableRuleset.2', 300.0), ('TableRuleset', True)]

A word of caution

TableRuleset rules are executed by Python's eval function, which is considered unsafe. This may become a problem if you allow users to edit their own rules by inserting arbitrary text (=> code) in the if, then or target sections of a rule in TableRuleset

So is pyrules unsafe by conclusion? No! Any Rule instance other than TableRuleset is just pure Python code -- no eval magic applied.

How to contribute

All contributes are welcome! Please have a look at the list of issues. If you find a bug, please open a new issue. If you contribute code or fixes, please create a pull-request. Thanks.