Monday, February 06, 2012
 
   
 
Welcome to my site

First let me say thanks for stopping by my site. My name is David Hanson-Graville and I am a IT consultant working in the UK. Let me make it clear, I am passionate about technology and specifically .net and its various forms. I've programmed in a range of langages, but I can say, I am now at my happiest when coding with c#. I hope my blog is an enjoyable & educational read and please feel free to email me at David.Hanson@OnTheBlog.net if you have any questions. 

Defensive Coding Minimize
Location: BlogsOnTheBlog    
Posted by: David Hanson Fri, 05 Jun 2009 06:33:18 GMT
How often have you seen code like this?
 
private void AllocateRoles(RoleChangeInfo roleChangeInfo)
{
            foreach (Role role in roleChangeInfo.Roles)
            {
                if (roleChangeInfo.Person.Age > 21 && roleChangeInfo.Person is Senior)
                    RoleChangeService.AddRole(rolechangeInfo.Person, role);
            }
      }
 
 
If we deconstruct this code we can see that the method AllocateRoles should received an instance of a RoleChangeInfo which is a container for all the information that is required to add the roles to the appropriate person. If we take a look at the implementation of this type we can see that it’s fairly simplistic.  It holds the ID of the Role and a RoleTaken DateTime.
 
public class Role
{
    public int ID { get; set; }
    public DateTime? RoleTaken { get; set; }
}
 
The body of the AllocateRoles method consists of a simple ForEach iteration around the Roles collection and with each role we check to make sure the person is older than 21 and they are of type senior. If both of these conditions are true then we call the RoleChangeService.AddRole method which will perform the work and update the RoleTaken.
 
I have come across this kind of code many times in the past and it is a good example of where a developer has made a number of assumptions and have not thought about making the AllocateRoles method defensive.  So before we jump into making this code defensive let’s take a look at some of the possible issues with this code.  Can you see them?
 
Developer Assumptions
·         That roleChangeInfo will not be null
·         That the Roles collection will be initialised
·         That each role within the Roles collection will not be null
·         That the roleChangeInfo.Person will not be null
·         That it is ok to call this method when the Person is not a senior
·         That it is ok to call this method with when the person is younger than 21
·         That it is ok to call this method and no roles are present.
·         That it is ok to exit this method without checking to see if our Role.RoleTaken has been set.
·         None of the Roles.RoleTaken stamps have already been set.
 
From the list of items above you get the idea.  The code in AllocateRoles takes the path of least resistance and should invalid data be passed in no specific exceptions will be raised. This can result in two possible outcomes. Firstly, the code crashes with a null reference exception and we spend hours trying to find the cause of the issue or the code executes a scenario that is considered invalid. This can lead to data corruption and a large headache trying to figure out where it’s happening.  
 
Enter Defensive coding – When we apply defensive coding techniques to this method we are trying to attain 2 possible outcomes. Firstly we wish to identify all possible errors within our method and throw exceptions if any of them are met. Secondly, we want to ensure that no business functionality will be executed accidently and that the results of our code is sound. Given this, let’s take another stab at our AllocateRoles method.
 
private void AllocateRoles(RoleChangeInfo roleChangeInfo)
{
      //CHECK: that we have a valid rolechangeInfo and that we have a Person and that the person
      //is of type Senior and that the roles collection is not null.
if (roleChangeInfo != null && roleChangeInfo.Person != null
    && roleChangeInfo.Person.GetType() == typeof(Senior)
    && roleChangeInfo.Roles != null)
{
         // CHECK:  that we actuall have roles.
    if (roleChangeInfo.Roles.Count == 0)
        throw new ApplicationException("You must specify at least one role.");
 
    foreach (var role in roleChangeInfo.Roles)
    {
  // CHECK:  we dont have a null role.
        if (role == null)
            throw new NullReferenceException("Role cannot be null");
  // CHECK:  that the Role Taken has not been set.
        if (role.RoleTaken != null)
            throw new ApplicationException("Role has already been taken.");
  // CHECK:  that the Persons ages is not less than 21.
        if (roleChangeInfo.Person.Age < 21)
            throw new ApplicationException("You cannot allocate roles to seniors under 21");
 
        //DO OUR WORK HERE
        RoleChangeService.AddRole(rolechangeInfo.Person,role)
 
    }
 
    // CHECK:  all roles have a taken date.
    if (roleChangeInfo.Roles.Exists(role => role.RoleTaken == null))
        throw new ApplicationException("Role taken date has not been set.");
}
}
 
As you can see the method implementation above now performs a number of checks on the input data and where appropriate throws exceptions if the state is invalid.  At this point we could consider this method defensive enough for our purposes. However by implementing numerous checks on the state of the data we have lost readability of our business logic.  Secondly the checks are scattered throughout the method and the very first check still falls through and can exit with no code being run. In most cases I may not mind this but others I may want to ensure fall through does not occur. The result is that I would have to break apart this check into multiple If’s. Not ideal.
 
Enter Arrange, Act, Assert -  If you are familiar with unit testing you may have come this pattern when creating your tests. If you haven’t I suggest you have a quick read here. But the basics of this pattern are that you arrange your code in a way so that the Act you are performing can execute with the desired effect and that after the work is done you can assert any failures.  Once you get the hang of it it certainly makes writing tests easier. Given this, I want to refactor the method so that a reusable pattern can be established when performing defensive coding. If we look at our pattern is follows the following workflow.
 
Assert (Input) >> Act >> Assert (Output)
 
Given this pattern and I what I want to achieve, I needed to create a simple utility class that would allowing me to perform daisy chaining of conditions which can then be executed at a given point. The Defensive methods such as IsNull would return instances of themselves allowing semantics such as the following.
 
Defensive.CurrentMethod
    .IsNotNull(test, "Message")
    .IsInstanceOfType(new object(), typeof(object), "Type does not match")
    .IsTrue(() => 1 == 2, "1 wont equal 2")
    .Check();
 
Using this new utility class we can now go back to our AllocateRoles method and refactor to meet AAA pattern we use in our tests. Below is the final defensive method.
 
private void AllocateRoles(RoleChangeInfo roleChangeInfo)
{
    //ASSERT Inputs
    Defensive.CurrentMethod
        .IsNotNull(roleChangeInfo, "Cannot call AllocateRoles without a roleChangeInfo")
        .IsNotNull(roleChangeInfo.Person, "You must have a valid Person reference")
        .AnyNull(roleChangeInfo.Roles, "You must specify roles")
        .IsInstanceOfType(roleChangeInfo.Person, typeof(Senior), "You can only apply roles to senior people")
        .IsTrue(() => roleChangeInfo.Person.Age > 21, "You cannot allocate roles to seniors under 21")
        .IsTrue(() => roleChangeInfo.Roles.Count > 0, "You must specify at least one role.")
        .Check();
 
    //ACT
    roleChangeInfo.Roles.ForEach(role => RoleChangeService.AddRole(rolechangeInfo.Person, role));
 
    //ASSERT Outputs
    Defensive.CurrentMethod
        .IsTrue(() => !roleChangeInfo.Roles.Exists(role => role.RoleTaken == null), "Role allocation failed.")
        .Check();
}
 
You can see that the AAA syntax coupled with the daisy chaining allows for our defensive checks to be grouped together at the top and bottom of our method. This avoids diluting our business code with numerous checks. Defensive coding is a powerful tool in our arsenal when it comes to increasing the resilence of our application code.
Permalink |  Trackback

Comments (2)   Add Comment
Re: Defensive Coding    By Kane on Sun, 13 Dec 2009 09:39:46 GMT
Hey dude, you got any code for the "Defensive" static class?

Re: Defensive Coding    By host on Mon, 14 Dec 2009 09:47:21 GMT
Posted for you.


Your name:
Title:
Comment:
Security Code
Enter the code shown above in the box below
Add Comment   Cancel 
Tweets Minimize
Twitter / LordHanson
  1. LordHanson: #southeastern have once again proved their rolling stock can reach new lows.

    Published Wed, 01 Feb 2012 08:09:05 +0000 by
  2. LordHanson: Checked in at The Cards http://t.co/LANfHukb

    Published Sun, 29 Jan 2012 11:04:51 +0000 by
  3. LordHanson: @BillGates Run for president Bill.

    Published Fri, 27 Jan 2012 08:41:35 +0000 by
  4. LordHanson: @swhelband Again!

    Published Fri, 27 Jan 2012 08:39:44 +0000 by
  5. LordHanson: The update for the Nokia Lumia recently has done wonders for battery life! Good job #Nokia #windowsphone

    Published Fri, 27 Jan 2012 08:36:27 +0000 by
  6. LordHanson: Checked in at Victoria Station http://t.co/L0BJ5smd

    Published Thu, 26 Jan 2012 08:35:41 +0000 by
  7. LordHanson: @pdl_uk VAT No: 924 5933 08 Shoulder again! Dang!

    Published Wed, 25 Jan 2012 21:47:45 +0000 by
  8. LordHanson: @tommyjmquinn I think that would be easier. Next Thursday ok?

    Published Wed, 25 Jan 2012 21:04:46 +0000 by
  9. LordHanson: @tommyjmquinn London bridge doable?

    Published Wed, 25 Jan 2012 20:32:34 +0000 by
  10. LordHanson: @tommyjmquinn so where's the meet?

    Published Wed, 25 Jan 2012 19:04:02 +0000 by
  11. LordHanson: @tommyjmquinn your choice mate. Somewhere easy to get to from Bankside. :-D

    Published Tue, 24 Jan 2012 22:01:20 +0000 by
  12. LordHanson: @tommyjmquinn so Darius, Justin and me confirmed. Thursday good for you? Waiting to hear from Sal.

    Published Tue, 24 Jan 2012 21:47:21 +0000 by
  13. LordHanson: @mark_mann which is illegal I thought!

    Published Tue, 24 Jan 2012 21:46:17 +0000 by
  14. LordHanson: Details on Windows Phone 8 confirms NT kernel http://t.co/5Qg1MILl

    Published Tue, 24 Jan 2012 21:34:11 +0000 by
  15. LordHanson: But next target for framework is #winrt. Need to see if my dependencies like DI, RX, ReactiveUi etc will work. Hmm

    Published Mon, 23 Jan 2012 08:33:16 +0000 by
  16. LordHanson: @pdl_uk hey Phil, how's marathon training going?

    Published Mon, 23 Jan 2012 08:31:37 +0000 by
  17. LordHanson: So I now have a framework for apps which targets .net, Silverlight and windows phone. Thankyou project linker!

    Published Mon, 23 Jan 2012 08:28:08 +0000 by
  18. LordHanson: For some reason dropped twitter for a month. Can't say I missed it really. Maybe I need to broaden my follow list!

    Published Mon, 23 Jan 2012 08:24:44 +0000 by
  19. LordHanson: Soo much hype over SIRI when it came out yet I never see anyone use it and don't know anybody who does either. #apple #sooverhyped

    Published Mon, 23 Jan 2012 08:23:18 +0000 by
  20. LordHanson: #southeastern customer satisfaction survey given to me today. This should be fun! Bit wait......no extremely poor option! Just very poor.

    Published Mon, 23 Jan 2012 08:20:09 +0000 by
Print  
Archive Minimize
Print  
Contact me Minimize
Print  
Microsoft Certs Minimize







Print  
Silverlight News Minimize
Silverlight - Google News
  1. Windows Phone 8 - Silverlight Apps Are Legacy - iProgrammer

    Published Fri, 03 Feb 2012 13:03:27 GMT by
  2. Super Bowl Streaming Fail - StreamingMedia.com

    Published Mon, 06 Feb 2012 05:59:33 GMT by
  3. Delphi Developers Rejoice at Silverlight, FireMonkey and VCL Coming Together ... - San Francisco Chronicle (press release)

    Published Tue, 31 Jan 2012 17:34:58 GMT by
  4. WP7 Upgrades and WP8 - Silverlight or C++ - iProgrammer

    Published Tue, 31 Jan 2012 17:21:58 GMT by
  5. Watch The 2012 Super Bowl Online - SportsGrid

    Published Sun, 05 Feb 2012 23:15:21 GMT by
  6. US viewers can watch Super Bowl on Mac, iOS - Macworld

    Published Sun, 05 Feb 2012 20:22:31 GMT by
  7. Hydra 4 Sharpens Its Teeth, Breathes New Fire - Dr. Dobb's

    Published Sun, 05 Feb 2012 17:25:01 GMT by
  8. Will Silverlight live or die? Microsoft won't say - InfoWorld

    Published Fri, 27 Jan 2012 11:00:46 GMT by
  9. Cablevision Flips Live TV To Laptops With Microsoft Silverlight - Multichannel News

    Published Fri, 27 Jan 2012 17:24:53 GMT by
  10. Do SharePoint & Silverlight Have a Future Together? - CMSWire

    Published Mon, 16 Jan 2012 17:29:27 GMT by
Print