Miguel A. Castro's Blog

# Saturday, August 07, 2010

My last posting demonstrating what I thought was a pretty cool technique for intercepting WCF operation calls so you can report them on a console; that is when you host on a console application for development.  Looking back at that technique, however useful to me, there was something that bugged me.  The behaviors I showed you how to write had the reporting-to-the-console part built right into them, and I wasn’t exactly satisfied that.  I still maintain the usefulness of reporting every operation call to my host console but wouldn’t it be a more powerful technique to report every operation call and allow the actual host application to do the reporting?  That way, whenever you finally deploy you services, you can track operation calls with any hosting application and do whatever you want with them.  In the case of a dev-console host, I can still report the calls to the console, but in the case of an IIS or NT-Service deployment I can chose to log them somewhere, even a custom event log.

So building on the same technique and code from the previous post, I changed the parameter inspector to raise an event instead of accessing the console.  Since the operation behavior is what instantiated and installed the parameter inspector, its job is to bubble the event up through another event.  In turn, the service behavior instantiated and installed the operation behavior on every operation so it also takes the information from the captured event and raises its own event.  Finally, the console host (in my case), which instantiates and installs the service behavior, captures its event and reports to the console.  The end-result in my case is the same, but the ability to simply capture an event from my hosting application gives me the ability to do whatever I want with the information.  Remember, each class raises an event to its parent which in turn raises its own event all the way up to the actual host.

If you haven’t read my last blog posting, I strongly suggest you go back and read it because I will not be re-explaining the details of the inspector and behavior classes, nor their installation details in this posting.

 

WcfOperationReportDiagram

 

The first I thing I did was create an event argument class called OperationInformationEventArgs which contains all the information about an operation that I want to send up the chain.  This class is used by all three of the other classes, the parameter inspector, the operation behavior, and the service behavior.

Here is the code for the event argument class:

public class OperationInformationEventArgs : EventArgs
{
    public OperationInformationEventArgs(string serviceName, string operationName, DateTime timeStamp)
    {
        ServiceName = serviceName;
        OperationName = operationName;
        TimeStamp = timeStamp;
    }

    public string ServiceName { get; set; }
    public string OperationName { get; set; }
    public DateTime TimeStamp { get; set; }
}

 

Next, I renamed the two behavior classes and the parameter inspector so their names were more general and did not imply console-reporting.  I called them OperationReportInspector, OperationReportOperationBehavior, and OperationReportServiceBehavior.  Also notice that I removed the word Attribute from their class name because I decided to no longer make them attribute capable.  Since I’m going to need to do some event wiring, using them in an attribute context would not have worked.

The parameter inspector class now takes the information that it was previously reporting directly to the console, and instead sticks it in the event argument and raises an event.

Here is the code for the new parameter inspector class:

public class OperationReportInspector : IParameterInspector
{
    public OperationReportInspector(string serviceName)
    {
        _ServiceName = serviceName;
    }

    public event EventHandler<OperationInformationEventArgs> ServiceOperationCalled;

    protected virtual void OnServiceOperationCalled(OperationInformationEventArgs e)
    {
        if (ServiceOperationCalled != null)
            ServiceOperationCalled(this, e);
    }

    string _ServiceName = string.Empty;

    void IParameterInspector.AfterCall(string operationName, object[] outputs, object returnValue, object correlationState)
    {
    }

    object IParameterInspector.BeforeCall(string operationName, object[] inputs)
    {
        //Console.WriteLine(string.Format("{0} - '{1}.{2}' operation called.", DateTime.Now.ToLongTimeString(), _ServiceName, operationName));
        OnServiceOperationCalled(new OperationInformationEventArgs(_ServiceName, operationName, DateTime.Now));

        return null;
    }
}

 

For comparison purposes, I left the old line that wrote out to the console in there and commented it out so you can see the difference.

Now I modified the operation behavior that would install the parameter inspector so just after it instantiates, it wires into the ServiceOperationCalled event.  This class also raises an event with the same information (with the same event arguments class in fact) in the method wired to the parameter inspector’s event.

Here is the code for the new operation behavior class:

public class OperationReportOperationBehavior : IOperationBehavior
{
    public event EventHandler<OperationInformationEventArgs> ServiceOperationCalled;

    protected virtual void OnServiceOperationCalled(OperationInformationEventArgs e)
    {
        if (ServiceOperationCalled != null)
            ServiceOperationCalled(this, e);
    }

    void IOperationBehavior.AddBindingParameters(OperationDescription operationDescription, BindingParameterCollection bindingParameters)
    {
    }

    void IOperationBehavior.ApplyClientBehavior(OperationDescription operationDescription, ClientOperation clientOperation)
    {
    }

    void IOperationBehavior.ApplyDispatchBehavior(OperationDescription operationDescription, DispatchOperation dispatchOperation)
    {
        string serviceName = dispatchOperation.Parent.Type.Name;
        OperationReportInspector inspector = new OperationReportInspector(serviceName);
        inspector.ServiceOperationCalled += inspector_ServiceOperationCalled;

        dispatchOperation.ParameterInspectors.Add(inspector);
    }

    void IOperationBehavior.Validate(OperationDescription operationDescription)
    {
    }

    void inspector_ServiceOperationCalled(object sender, OperationInformationEventArgs e)
    {
        OnServiceOperationCalled(e);
    }
}

 

Finally, I modified the service behavior class so just after it instantiates the operation behavior, it wires its event to a method where its own event is raised; again with the same event argument information.

Here is the code for the new service behavior class:

public class OperationReportServiceBehavior : IServiceBehavior
{
    public event EventHandler<OperationInformationEventArgs> ServiceOperationCalled;

    protected virtual void OnServiceOperationCalled(OperationInformationEventArgs e)
    {
        if (ServiceOperationCalled != null)
            ServiceOperationCalled(this, e);
    }
        
    void IServiceBehavior.AddBindingParameters(ServiceDescription serviceDescription, ServiceHostBase serviceHostBase, System.Collections.ObjectModel.Collection<ServiceEndpoint> endpoints, BindingParameterCollection bindingParameters)
    {
    }

    void IServiceBehavior.ApplyDispatchBehavior(ServiceDescription serviceDescription, ServiceHostBase serviceHostBase)
    {
        foreach (ServiceEndpoint endpoint in serviceDescription.Endpoints)
            foreach (OperationDescription operation in endpoint.Contract.Operations)
            {
                OperationReportOperationBehavior behavior = new OperationReportOperationBehavior();
                behavior.ServiceOperationCalled += behavior_ServiceOperationCalled;

                operation.Behaviors.Add(behavior);
            }
    }

    void IServiceBehavior.Validate(ServiceDescription serviceDescription, ServiceHostBase serviceHostBase)
    {
    }

    void behavior_ServiceOperationCalled(object sender, OperationInformationEventArgs e)
    {
        OnServiceOperationCalled(e);
    }
}

 

So now, all that is left to do is to change the actual console host application so when it instantiates and installs the service behavior, it wires into its ServiceOperationCalled event, then displays the information to the console using the wired method.

Here’s the new code for the console host:

OperationReportServiceBehavior behavior = serviceHost.Host.Description.Behaviors.Find<OperationReportServiceBehavior>();

if (behavior == null)
{
    behavior = new OperationReportServiceBehavior();
     behavior.ServiceOperationCalled += behavior_ServiceOperationCalled;

     serviceHost.Host.Description.Behaviors.Add(behavior);
}

void behavior_ServiceOperationCalled(object sender, OperationInformationEventArgs e)
{
    Console.WriteLine(string.Format("{0} - '{1}.{2}' operation called.",
        e.TimeStamp.ToLongTimeString(), e.ServiceName, e.OperationName));
}

 

As I said, the end-result is the same as in my previous posting but now the hosting application can chose to do what it wants with the contents of the event argument.  In my humble opinion, what was a cool technique is now a cooler technique.

 

Until next time…

Saturday, August 07, 2010 7:22:06 PM (Eastern Daylight Time, UTC-04:00)  #    Comments [1] - - Follow me on Twitter

# Tuesday, July 27, 2010

Note: Using a Console app for hosting should be done for development only !

Now that the disclaimer is out of the way, I currently doing some development work for a customer and I had the need to know every time any of my services receives a call.  Since I do my development using a console application for service hosting, it was easy enough to write out to the console from each service operation.  Of course, putting code in every method was possible but not only would it be time consuming, to turn it off later would be a pain.  The answer needed to be something that gets hit on every operation.  Since WCF is essentially one giant inversion of control container, I knew there was a point I could tap into before any operation call.

Enter behaviors !

This turned out to be a multi-step process but once it’s in my library of WCF stuff, I can reuse it and it actually turned out to be quite cool.  I’ll explain each step and the reasons for the step.

Step 1 – Write a custom Parameter Inspector

The first class you need to write is one that implements the IParameterInspector which is in the System.ServiceModel.Dispatcher namespace.  This class will later be installed in such a way that it gets hit on every operation call.  The operation name is one of the arguments you have access to, as well as all the operation arguments which I don’t need in this case.

The methods in this implementation are BeforeCall and AfterCall, and it is the BeforeCall that I’m interested in.  The code I’ll place in this method very simply outputs something out to the console using Console.Writeline.

Unfortunately the name of the service is not passed into this method so I need to get that into this class using a custom constructor; you’ll see how I’ll get it to the class later.

Here’s the code for my ConsoleReportInspector class:

 

public class ConsoleReportInspector : IParameterInspector
{
    public ConsoleReportInspector(string serviceName)
    {
        _ServiceName = serviceName;
    }

    string _ServiceName = string.Empty;

    void IParameterInspector.AfterCall(string operationName, object[] outputs, object returnValue, object correlationState)
    {
    }

    object IParameterInspector.BeforeCall(string operationName, object[] inputs)
    {
        Console.WriteLine(string.Format("{0} - '{1}.{2}' operation called.", DateTime.Now.ToLongTimeString(), _ServiceName, operationName));

        return null;
    }
}


 

Step 2 – Install the class using an Operation Behavior

In order to install the parameter inspector, I’ll need to write a custom operation behavior class.  This is a class that implements the IOperationBehavior interface.  You need to remove the throw to NotImplementedException from all the methods; and the method of concern here is ApplyDispatchBehavior.

The name of the service being called is accessible through the Parent property of the dispatchOperation argument.  I need to instantiate my ConsoleReportInspector class and send the service name into the constructor I added earlier.  Afterward, I just need to add my parameter inspector class to the list of parameter inspectors accessible in this behavior.

I also inherited this class from the System.Attribute class so I can use it as an operation behavior attribute on individual operations if I wanted.  This is just for flexibility since I plan on hooking this into a service so it affects all operations.

Here’s the code for my ConsoleReportOperationBehaviorAttribute class:

 

[AttributeUsage(AttributeTargets.Method)]
public class ConsoleReportOperationBehaviorAttribute : Attribute, IOperationBehavior
{
    void IOperationBehavior.AddBindingParameters(OperationDescription operationDescription, BindingParameterCollection bindingParameters)
    {
    }

    void IOperationBehavior.ApplyClientBehavior(OperationDescription operationDescription, ClientOperation clientOperation)
    {
    }

    void IOperationBehavior.ApplyDispatchBehavior(OperationDescription operationDescription, DispatchOperation dispatchOperation)
    {
        string serviceName = dispatchOperation.Parent.Type.Name;
        dispatchOperation.ParameterInspectors.Add(new ConsoleReportInspector(serviceName));
    }

    void IOperationBehavior.Validate(OperationDescription operationDescription)
    {
    }
}


As-is, I can simply decorate a service operation with [ConsoleReportOperationBehavior] and if hosted on a console application, you’ll see output to the console when the decorated operation is called.

 

Step 3 – Install the operation behavior on all operations using a Service Behavior

Next I need to write a service behavior.  This is a behavior that gets installed when the service loads (the host is opened).  The job of this behavior is to install my operation behavior on all the service operations.  A service behavior is a class that implement the IServiceBehavior interface.  Like before, I’ll remove the exception calls and once again the method I’m interested in is ApplyDispatchBehavior.

I’m going to iterate through all the endpoints of this service, which are accessible through the serviceDescription argument, and for each endpoint I will iterate through the operations of the contract of each endpoint.  To each operation I will add an instance of my ConsoleReportOperationBehaviorAttribute class.

This class too I will inherit from Attribute so I can use it to decorate a service in the case that I wanted to take this approach at some time. 

Here’s the code for my ConsoleReportServiceBehaviorAttribute class:

 

[AttributeUsage(AttributeTargets.Class)]
public class ConsoleReportServiceBehaviorAttribute : Attribute, IServiceBehavior
{
    void IServiceBehavior.AddBindingParameters(ServiceDescription serviceDescription, ServiceHostBase serviceHostBase, System.Collections.ObjectModel.Collection<ServiceEndpoint> endpoints, BindingParameterCollection bindingParameters)
    {
    }

    void IServiceBehavior.ApplyDispatchBehavior(ServiceDescription serviceDescription, ServiceHostBase serviceHostBase)
    {
        foreach (ServiceEndpoint endpoint in serviceDescription.Endpoints)
            foreach (OperationDescription operation in endpoint.Contract.Operations)
                operation.Behaviors.Add(new ConsoleReportOperationBehaviorAttribute());
    }

    void IServiceBehavior.Validate(ServiceDescription serviceDescription, ServiceHostBase serviceHostBase)
    {
    }
}

 

Step 4 – Apply the service behavior to any service I want programmatically at the host

Whatever approach you take to hosting, all you need to do is access each instance of ServiceHost and add the service behavior to the list of behaviors in the host.  I have my own techniques for hosting which include some custom declarative stuff to allow me to turn hosting on and off at will through config.  I also like to separate the accumulation of my ServiceHost instances from the actual application that will do the hosting.  This way I can move it to a Windows Service when I’m ready to go to production.

Here’s the code that installs the behavior:

 

 

ConsoleReportServiceBehaviorAttribute behavior = serviceHost.Host.Description.Behaviors.Find<ConsoleReportServiceBehaviorAttribute>();
if (behavior == null)
    host.Description.Behaviors.Add(new ConsoleReportServiceBehaviorAttribute());

 

The host variable is an instance of ServiceHost.  I perform the code above in between a config check so I can toggle it off at will.

The end result is quite cool since when turned on, every call to any of your services will be shown on the console.

 

Until next time…

Tuesday, July 27, 2010 10:57:11 PM (Eastern Daylight Time, UTC-04:00)  #    Comments [1] - - Follow me on Twitter
Dev Stuff | WCF
# Saturday, July 03, 2010

The following text was originall written by Rachel Appel, Developer Evangelist for the NY region:


On July 7th, 2010 at 7pm in Manhattan the NYC area developer community will be holding an event to benefit NYC-area .NET software developer Wendy Friedlander.

Wendy is a 30 year old software agilista from Long Island. She's a strong WPF developer and a firm believer in the agile method of development including pair programming and TDD. Wendy is wife and mother of a beautiful girl named Kaylee who will be 2 in August of this year. In August of 2009 Wendy learned that she had a rare and aggressive pediatric cancer called aveolar rhabdomyosarcoma. Her treatment consists of high dose chemotherapy and radiation. She has had to leave her job, and her husband has been forced into part time work in order to care for their daughter.
Come and be part of something special!

We are very pleased to be able to announce that the evening will include a unique presentation by noted .NET luminary and author Charles Petzold on an obscure (but interesting) chapter in the history of computing with appearances (in chronological order) by Cicero, Galileo, Kepler, Newton, Fourier, Charles Darwin, Lord Kelvin, Madame Curie, and Albert Einstein.

In addition, Miguel Castro has volunteered to provide a very special talk for the attendees where he promises to demonstrate some of his own pet software development projects for us!

In addition to the opportunity to see these unique talks, every attendee will be given a free unlimited 30-day subscription to TekPub (http://www.tekpub.com), the video screencasting site with hours and hours of screencasts on topics including Silverlight, Entity Framework, NHibernate, JQuery, Ruby-on-Rails, developing for the iPhone, and developing for Android. Everyone who attends the event is guaranteed this free subscription just for their cost of attending.

Other SWAG that will be raffled off to attendees includes multiple software licenses for the following products:
* JetBrains Resharper, DotTrace, and DotCover!
* Developer Express CodeRush, Refactor! Pro, and DXperience Suite (every control they make!)
* Telerik RAD Controls!
* TypeMock Isolator!
* Visual Studio MSDN Ultimate i-year Subscriptions!
* ...and more!

Specific details for this event including location, RSVP links, and more can be found at the site:
http://www.DevsForWendy.com

For more details about the event, please see this comprehensive blog post by Steve Bohlen:
http://unhandled-exceptions.com/blog/index.php/2010/06/30/devsforwendy/

Attendance is $75 per person with all proceeds going to Wendy and her family. Since dinner will consist of pizza and soda, you can be assured that more than $65 of that attendance fee will go to benefit the Friedlanders directly!

Even if you cannot attend the benefit event itself, please consider using the [Donate] link on the site (http://devsforwendy.com/donate.html) to make a contribution directly to Wendy and her family. Any amount you may be able to spare would be very much appreciated -- whether $1, $5, $10, $50, or more it all adds up and it all goes to a great cause!

If there are any questions, please do not hesitate to email either Sara Chipps (sarajchipps@gmail.com) or Steve Bohlen (sbohlen@gmail.com) directly.
Otherwise, please consider either attending the event, making a direct donation, or both!

Saturday, July 03, 2010 5:18:06 PM (Eastern Daylight Time, UTC-04:00)  #    Comments [0] - - Follow me on Twitter
Speaking Events
# Monday, June 21, 2010

Wow, what a day!

The first ever WCF Firestarter event was yesterday at the New York City Microsoft offices and it was awesome.  The agenda was simple and to-the-point.

I started the day with the first two sessions, introducing everyone to WCF an then segueing them into more advanced binding and behavior configurations by showing the most commonly used configuration scenarios.  Seems that transaction handling was the more popular one there.  After lunch, one of our local DEs, Peter Laudati took the stage to talk about WCF REST capabilities.  After which, Don Demsak introduced everyone to WCF Data Services, rounding out the capability set of this technology.  I came back on at the end of the day to talk about the new WCF 4 stuff and it seems that routing capabilities was the highlight on that one.

  Peter and Don did an awesome job, as always.  As for my performance, I was happy but I’ll let others blog about what they thought.  All in all, I was comfortable with the outcome and I think the crowd was pleased.  We had about 200 attendees in the room and another 108 on LiveMeeting.  The event brought in people from around the country and world I think.  I put out a call for listeners to send me an email telling me where they are and immediately I received one from a listener in Southern Australia.  Kudos goes out to Coral who was listening in the middle of the night, down under.

 

 

When I took a poll at the beginning as to how many WCF users I had in the room, the count was extremely low (about 10) so I think this was definitely the target audience.

Thank you to Peter for getting Microsoft behind this event and for handling all the logistics, and thanks for letting me spearhead the agenda and content for the event.  I had a great time doing it.  O’Reilly was awesome and sent me 5 copies of Michele’s book and 5 of Juval’s to give away.  Also thanks to Developer Express and Telerik for donating product licenses for the event.  Both those companies can always be counted on to support these kind of events and their giveaways were the first ones to disappear.  Also, we had a tech out in Redmond with us every step of the way, monitoring the live feed and standing by in case of problems and we couldn’t have been more grateful to Erik Ostrowski for his help and for being there at 3am his time, to help us out.

As a goof, I have away a couple of “What Would Miguel Do?” t-shirts that a client of mine made for me.  I couldn’t believe it when one of the guys put it on and wore it for the entire event.  It was certainly flattering a very cool gesture.

 

 

 

I think that Danny, the DE down in the Philly market wants to put this event on in his territory so I guess that’s where we’ll be taking it next.

Thanks again to Microsoft, the local DEs, Don, and all the attendees and listener for help making it a great event.

Until next time…

Monday, June 21, 2010 1:42:35 PM (Eastern Daylight Time, UTC-04:00)  #    Comments [2] - - Follow me on Twitter
Speaking Events
# Monday, May 31, 2010

book-cps-smI just zoomed through Scott Berkun’s book, “Confessions of a Public Speaker”, which was recommended to me by my friend Don Demsak.  I can’t recommend this book enough to those of you who like me are already public speakers, as well as to those of you who want to break into speaking.  Scott delivers his point in an extremely entertaining way while actually sticking to it; which is exactly what the book preaches to do when speaking.  The book is written in a first-person form and like the author, I agree this is a great way of keeping a reader engaged.  I found it gratifying that a lot of points he recommends in his book are techniques I use today, but more importantly I found so many where I plan to improve that I’m actually excited about my upcoming Tech-Ed presentation and already have ideas for bettering my delivery using stuff I learned here.

The book is a quick read, not only because it’s only a couple of hundred pages but because I found it so entertaining and interesting, it was hard to put down.  Check this one out cause you won’t regret it.

Until next time…

Monday, May 31, 2010 11:23:36 AM (Eastern Daylight Time, UTC-04:00)  #    Comments [2] - - Follow me on Twitter

Search
Me & My Flair

Read all about me here.
Download my Resume here.

Check out where I am here.
 
Click on logos above for profiles.
Archive
<August 2010>
SunMonTueWedThuFriSat
25262728293031
1234567
891011121314
15161718192021
22232425262728
2930311234
Statistics
Total Posts: 36
This Year: 1
This Month: 1
This Week: 0
Comments: 80