5,447,640 members and growing! (20,276 online)
Email Password   helpLost your password?
Languages » C# » General     Intermediate License: The Code Project Open License (CPOL)

Using a Rules Engine to Separate Business Rules from the Application

By Jeff Bramlett

Using a Rules Engine to separate Business rules from the application
C# 1.0, C# 2.0, C#Windows, .NET, .NET 1.1, .NET 2.0, Win2K, WinXP, ASP.NET, Visual Studio, Dev

Posted: 10 Aug 2006
Updated: 29 Aug 2006
Views: 28,972
Bookmarked: 34 times
Announcements
Want a new Job?



Search    
Advanced Search
Sitemap
12 votes for this Article.
Popularity: 3.64 Rating: 3.37 out of 5
3 votes, 25.0%
1
2 votes, 16.7%
2
1 vote, 8.3%
3
3 votes, 25.0%
4
3 votes, 25.0%
5
Demo Application

Introduction

I believe that a de-coupling of business logic from an application makes for more robust and maintainable code. If done correctly, the logic for the application can be maintained by "non-programming" people using their terminology and methodology. The application can continue to use the business logic as it changes over time and under changing circumstances because the application does not contain the business logic, but only uses it. The location of the business logic can then be centrally located and maintained, or customized and sent for specific reasons for specific installations.

This demo Visual Studio solution is a very simple example (kind of like using an elephant gun to kill a flea). I prepared this example to be easily understood, but have enough complexity to illustrate the capability.

The Rule Engine is freely available for all uses here (CodeProject article).

An on-line example can be seen here.

In order to use a Rules Engine for any automation, I found some requirements:

  1. The Rules must be easy to make
  2. The Rule maker must be able to easily define how rules affect each other.
  3. The Rules Engine must be deterministic. There can be one and only one answer to any question.
  4. The Rules Engine must provide event listening so that the application can react to changes in the Rules Engine
  5. The Rules Engine must be able to respond to changes from the application.

The Rules Engine used in the demo meets these requirements.

Background

This demo is a simple invoice creation system. This invoice is for calculating varying quantities of widgets. In addition to calculating prices, the business rules must apply quantity discounts, which are different for each widget type. The weight of the total widgets is calculated and an appropriate shipper is selected and each shipper has a different rate. There is a discount applied for each customer based on the current balance that s/he has with the Widget Barn.

The order for calculating the invoice is:

  1. Multiply the quantity of each Widget to get the total price
  2. Multiply the quantity of each Widget to get the total weight
  3. Compute the discount for the quantities of each widget (each widget has its own quantity discount rate)
  4. Subtract the discount for each widget from its total price
  5. Sum the total prices of the widgets minus the discount amounts
  6. Compute the customer discount based on the balance owed and subtract it
  7. Sum the total weights of the widgets ordered
  8. Determine which shipper to use based on the total weight
  9. Calculate the shipping costs
  10. Sum the total price minus discounts to make the invoice total

Using the Code

The demo uses an XML file for defining rules. The rules engine does not require it but it is available and it works good. The XML rules file can be loaded from a Web site or from a local file. Both are provided for the demo application. The first thing is to open a rules file:

private void menuRulesWebFile_Click(object sender, EventArgs e)
{
    JaxlabReader reader = new JaxlabReader(homeWebfile, "");
    rulesEngine = reader.NewRulesEngine;

    cbxCustomers.SelectedIndex = 0;
}

Once loaded, the Rules engine will execute upon changes defined in any triggers, so all I have to do is change values to initiate the changes.

rulesEngine.SetVariableValue("SilverQty", qtySilver.Text);

Or I can "run" rules at any time like this:

rulesEngine.RunRule("CalculateInvoice");

I simply set up a method to get the values I need for the invoice and write it to the textbox. The Rules Engine does everything in between.

private void MakeInvoice()
{
    List<STRING> lines = new List<STRING>();
    lines.Add("Invoice Demo");
    lines.Add("");

    lines.Add("Customer Name: " + rulesEngine.GetVariable("CustomerType").StringValue);
    lines.Add("Customer Balance: " + 
              rulesEngine.GetVariable("CustomerBalance").DoubleValue.ToString("c"));

    lines.Add("");
    lines.Add("Quantity of Gold Widgets: " + 
              rulesEngine.GetVariable("GoldQty").StringValue + 
              "\t Price: " + rulesEngine.GetVariable
              ("GoldPrice").DoubleValue.ToString("c") + 
              "\t Total Price: " + rulesEngine.GetVariable
              ("GoldTotalPrice").DoubleValue.ToString("c") + 
              "\t Discount: " + rulesEngine.GetVariable("GoldQtyDiscount").StringValue);

    lines.Add("Quantity of Silver Widgets: " + 
              rulesEngine.GetVariable("SilverQty").StringValue +
              "\t Price: " + rulesEngine.GetVariable
              ("SilverPrice").DoubleValue.ToString("c") +
              "\t Total Price: " + rulesEngine.GetVariable
              ("SilverTotalPrice").DoubleValue.ToString("c") +
              "\t Discount: " + rulesEngine.GetVariable
              ("SilverQtyDiscount").StringValue);

    lines.Add("Quantity of Star Widgets: " + 
              rulesEngine.GetVariable("StarQty").StringValue +
              "\t Price: " + rulesEngine.GetVariable=
              ("StarPrice").DoubleValue.ToString("c") +
              "\t Total Price: " + rulesEngine.GetVariable
              ("StarTotalPrice").DoubleValue.ToString("c") +
              "\t Discount: " + rulesEngine.GetVariable("StarQtyDiscount").StringValue);

    lines.Add("");
    lines.Add("\t\t\t\t\t\t         Subtotal: \t" + 
              rulesEngine.GetVariable("PriceSubTotal").DoubleValue.ToString("c"));
    lines.Add("\t\t\t\t\t\t         Discount: \t" + 
              rulesEngine.GetVariable
                  ("CustomerTypeDiscountAmount").DoubleValue.ToString("c"));
    lines.Add("\t\t\t\t\t\t         Shipping: \t" + 
              rulesEngine.GetVariable("ShippingTotal").DoubleValue.ToString("c"));
    lines.Add("\t\t\t\t\t\t              Tax: \t" + 
              rulesEngine.GetVariable("SalesTaxTotal").DoubleValue.ToString("c"));
    lines.Add("\t\t\t\t\t\t          Shipper: \t" + 
              rulesEngine.GetVariable("ShipperName").StringValue);

    lines.Add("");
    lines.Add("");
    lines.Add("\t\t\t\t\t\t            Total: \t" + 
              rulesEngine.GetVariable("InvoiceTotal").DoubleValue.ToString("c"));

    txtBox.Lines = lines.ToArray();
}

Points of Interest

The event handling with the Rules engine made this easy to implement.

License

This article, along with any associated source code and files, is licensed under The Code Project Open License (CPOL)

About the Author

Jeff Bramlett


Website: http://www.devbyjeffbramlett.com
Occupation: Web Developer
Location: United States United States

Other popular C# articles:

Article Top
Sign Up to vote for this article
You must Sign In to use this message board.
FAQ FAQ Noise ToleranceSearch Search Messages 
 Layout  Per page   
 Msgs 1 to 14 of 14 (Total in Forum: 14) (Refresh)FirstPrevNext
Subject  Author Date 
QuestionRuleEngine1memberLearnTech23:29 28 Dec '07  
Questionneeds Help on rule enginememberLearnTech18:31 28 Dec '07  
QuestionWWF (windows workflow framework) ?membergogac8:18 10 Aug '06  
AnswerRe: WWF (windows workflow framework) ?memberJeffBramlett8:42 10 Aug '06  
AnswerRe: WWF (windows workflow framework) ?memberGrav-Vt11:21 20 Aug '06  
GeneralRe: WWF (windows workflow framework) ?membergogac13:44 20 Aug '06  
AnswerRe: WWF (windows workflow framework) ?memberDanilo Mendez16:45 5 Apr '07  
GeneralExamplesmemberDustin Metzgar7:11 10 Aug '06  
GeneralRe: ExamplesmemberJeffBramlett7:49 10 Aug '06  
GeneralRe: ExamplesmemberDustin Metzgar8:12 10 Aug '06  
GeneralRe: ExamplesmemberJeffBramlett8:34 10 Aug '06  
GeneralWhere's the sourcememberJuergen Posny6:09 10 Aug '06  
GeneralRe: Where's the sourcememberJeffBramlett7:08 10 Aug '06  
GeneralRe: Where's the sourcememberJeffBramlett10:10 29 Aug '06  

General General    News News    Question Question    Answer Answer    Joke Joke    Rant Rant    Admin Admin   

PermaLink | Privacy | Terms of Use
Last Updated: 29 Aug 2006
Editor: Deeksha Shenoy
Copyright 2006 by Jeff Bramlett
Everything else Copyright © CodeProject, 1999-2008
Web17 | Advertise on the Code Project