Miguel A. Castro's Blog

# 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

# Monday, May 24, 2010

WCF 4.0 incorporates the features previously only obtainable via the REST Starter kit for .NET 3.5.  Among the features included is the ability to cache REST requests using a similar technique to caching ASP.NET pages.

The 4.0 version of the System.ServiceModel.Web assembly includes an attribute called AspNetCacheProfile which takes a string as an argument.  The attribute is to be used on a service contract operation as follows:

[OperationContract]
[WebGet(UriTemplate = "/{name}/getGreeting.xml",
        BodyStyle = WebMessageBodyStyle.Bare,
        RequestFormat = WebMessageFormat.Xml,
        ResponseFormat = WebMessageFormat.Xml)]
[AspNetCacheProfile("CacheFor10Seconds")]
string GetGreetingXml(string name);

 

The string points to a cache profile to be defined in the config section as follows:


    
        
            
        
    

 

As you can see, this is pretty much identical to setting up cache profiles for ASPX pages.  The effect is pretty much what you would expect, the caching of a REST request.  This can be proven and tested by implementing the above contract as such:

public string GetGreetingXml(string name)
{
    return string.Format("Hello {0}.  The time is {1}", name, DateTime.Now.ToLongTimeString());
}

 

Now you can refresh the request and you’ll see the same time for 10 seconds before it changes; once again, the same effect as in caching an ASPX page.

So what did I learn the hard way you ask?  Well, common sense dictates that the request go through ASP.NET in order to use its caching capability.  This not only means that you have to host in IIS or WAS, but also that you need to turn on ASP.NET compatibility mode.  This shifts the processing of the REST request over to an HTTP handler that also sets an HTTP Context for you, something not normally done with conventional IIS hosting for REST services.  To turn this on, you must enable it in the section under the section using the aspNetCompatibilityEnabled attribute.  Not only that, but you have to “allow” it on the service using the AspNetCompatibilityRequirements attribute.

This part was not a surprise for me since I was already familiar with this technique and the added scope your service receives because it.  I’m referring to the entire HTTP Context including session, application, cache, etc.  Keep in mind that this is not for everything and should be used in very specific scenarios as it ties the service to the hosting style quite tightly.

What drove me crazy is something I’m surprised I did not notice in the past and that is the fact that your REST requests will not work while using the Visual Studio development server.  Meaning you need to create an application in IIS and run it that way.  No big deal but it drove me a little crazy since the meta data exposure worked fine, and so did the new HTML Help pages (another new feature in WCF 4.0 For REST).  But the second I tried to request one of my URLs, it bombed out.  Once I set up my application under IIS, everything was fine.

I have to give credit where it’s due and thank Jon Flanders.  I found the answer in his book Restful .NET from O’Reilly press.

Until next time…

Monday, May 24, 2010 4:50:14 PM (Eastern Daylight Time, UTC-04:00)  #    Comments [0] - - Follow me on Twitter
Dev Stuff | WCF
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
<July 2010>
SunMonTueWedThuFriSat
27282930123
45678910
11121314151617
18192021222324
25262728293031
1234567
Statistics
Total Posts: 19
This Year: 19
This Month: 2
This Week: 0
Comments: 20