Contact
Send mail to the author(s) Email Me

Disclaimer
The opinions expressed herein are my own personal opinions and do not represent my employer's view in any way.

Sign In
Navigation

Tag Cloud
.NET Framework (31) AJAX (9) ASP.NET (16) ASP.NET MVC (3) C# (32) Cloud (2) Database (6) Dev Community (2) Dev Tools (5) Enterprise Library (1) Futures (2) General (6) IIS (1) Javascript (7) LINQ (2) Mobile (1) MSDTC (5) Quotes (3) SQL (3) Transactions (4) Visual Studio (3) WAS (2) WCF (20) WIF (1)

Archive
<March 2010>
SunMonTueWedThuFriSat
28123456
78910111213
14151617181920
21222324252627
28293031123
45678910

Categories

Blogroll
Home Feed your aggregator (RSS 2.0)
# Sunday, August 31, 2008

Why clutter your inbox with error messages? Why make special code provisions for users to receive error messages via email? Why not log your error messages and have users subscribe to receive them in their favorite RSS aggregator?

If you are logging your exceptions already, you may find it easier to provide a syndication service. The process is ridiculously simple, and starts by creating a new project using the "Syndication Service Library" template. This template creates everything for you. All you need to do now is fill the SyndicationFeed with SyndicationItem objects.

Add a new class file called Feeds.cs:

 

    1 using System;

    2 using System.Linq;

    3 using System.ServiceModel;

    4 using System.ServiceModel.Syndication;

    5 using System.ServiceModel.Web;

    6 

    7 namespace SyndicationService

    8 {

    9     [ServiceContract]

   10     [ServiceKnownType(typeof(Atom10FeedFormatter))]

   11     [ServiceKnownType(typeof(Rss20FeedFormatter))]

   12     public interface IFeeds

   13     {

   14         [OperationContract]

   15         [WebGet(UriTemplate = "{type}?env={env}&app={app}", BodyStyle = WebMessageBodyStyle.Bare)]

   16         SyndicationFeedFormatter CreateFeed(string type, string env, string app);

   17     }

   18 

   19     public class Feeds : IFeeds

   20     {

   21         public SyndicationFeedFormatter CreateFeed(string type, string env, string app)

   22         {

   23             SyndicationFeed feed = CreateSyndicationFeed(type, env, app);

   24 

   25             // Return ATOM or RSS based on query string

   26             // rss -> http://localhost:8000/Feeds/Errors?env=Production&app=MyAppName

   27             // atom -> http://localhost:8000/Feeds/Errors?env=Production&app=MyAppName&format=atom

   28             string query = WebOperationContext.Current.IncomingRequest.UriTemplateMatch.QueryParameters["format"];

   29             SyndicationFeedFormatter formatter = null;

   30             if (query == "atom")

   31             {

   32                 formatter = new Atom10FeedFormatter(feed);

   33             }

   34             else

   35             {

   36                 formatter = new Rss20FeedFormatter(feed);

   37             }

   38 

   39             return formatter;

   40         }

   41 

   42         private static SyndicationFeed CreateSyndicationFeed(string type, string env, string app)

   43         {

   44             SyndicationFeed feed;

   45             switch (type.ToLower())

   46             {

   47                 case "errors":

   48                     feed = CreateErrorsFeed(type, env, app);

   49                     break;

   50                 default:

   51                     feed = new SyndicationFeed(

   52                         String.Format("Feed is unavailable - Type: {0} / Environment: {1} / Application: {2}",

   53                         type, env, app), null, null);

   54                     break;

   55             }

   56             return feed;

   57         }

   58 

   59         private static SyndicationFeed CreateErrorsFeed(string type, string env, string app)

   60         {

   61             ApplicationLogDataContext db = new ApplicationLogDataContext();

   62 

   63             SyndicationFeed feed = new SyndicationFeed

   64             {

   65                 Title = new TextSyndicationContent(String.Format("{0} {1} {2}", env, app, type)),

   66                 Description = new TextSyndicationContent(

   67                     String.Format("Application error syndication for the {0} applicaiton ({1}).", app, env)),

   68                 Items = from e in db.Exceptions

   69                         where e.Environment == env && e.Application == app

   70                         select new SyndicationItem

   71                         {

   72                             Title = new TextSyndicationContent(e.Message),

   73                             Content = new TextSyndicationContent(e.StackTrace)

   74                         }

   75             };

   76             return feed;

   77         }

   78     }

   79 }

Modify the App.config file:

 

    1 <?xml version="1.0" encoding="utf-8" ?>

    2 <configuration>

    3     <configSections>

    4     </configSections>

    5     <connectionStrings>

    6         <add name="SyndicationService.Properties.Settings.ApplicationLogConnectionString"

    7             connectionString="Data Source=Scorpion;Initial Catalog=ApplicationLog;Integrated Security=True"

    8             providerName="System.Data.SqlClient" />

    9     </connectionStrings>

   10     <system.serviceModel>

   11         <services>

   12             <service name="SyndicationService.Feeds">

   13                 <host>

   14                     <baseAddresses>

   15                         <add baseAddress="http://localhost:8000/" />

   16                     </baseAddresses>

   17                 </host>

   18                 <endpoint contract="SyndicationService.IFeeds"

   19                           address="Feeds"

   20                           binding="webHttpBinding"

   21                           behaviorConfiguration="WebHttpBinding_Common"/>

   22             </service>

   23         </services>

   24         <behaviors>

   25             <endpointBehaviors>

   26                 <behavior name="WebHttpBinding_Common">

   27                     <webHttp/>

   28                 </behavior>

   29             </endpointBehaviors>

   30         </behaviors>

   31     </system.serviceModel>

   32 </configuration>

You will need to adjust your project's Debug options to have command arguments that look similar to the following to F5-debug your service.

"/client:iexplore.exe" "/clientArgs:http://localhost:8000/Feeds/Errors?env=Production&app=GeoTracker"

Press F5 to test it out.

Here is the IE7 RSS viewer:

IE7_RSS_Viewer

Here is your RSS aggregator viewing the same feed:

RSS_Aggregator

You will, of course, want to add some additional information to the content of your SyndidationItem, a bogus phrase works for this example.

Also, it is unusual that you would care to keep your exception details around for a long period of time. Since this is a syndicated feed of application errors, you should make special arrangements to archive or delete your exception log on a regular basis. This will not only keep your insert and select times low, but will also alleviate the burden placed on a new subscriber when all of the exceptions from the database appear at once. An alternative would also be to modify the LINQ in the code above to only bring back exceptions from the last 7-60 days depending on your counts. I already archive my exceptions to a master exception repository for all environments by way of an ETL job. This way I can report on my errors without disturbing the live environments too.

Sunday, August 31, 2008 3:37:38 AM (Eastern Standard Time, UTC-05:00)  #    Comments [4]   .NET Framework | C# | LINQ | WCF  | 
Sunday, June 21, 2009 12:05:02 PM (Eastern Standard Time, UTC-05:00)
Sorry. Uber cool dood. Help me! I find sites on the topic: Imitrex sidefects. I found only this - from imitrex. Since only the smallest fraction of information dealing with imitrex is indexed in search engines, such as. Subcutaneous sumatriptan most effective oral agents w imitrex po. Thanks for the help :o, Stephenie from Samoa.
Wednesday, September 23, 2009 4:18:52 PM (Eastern Standard Time, UTC-05:00)
Hi. America believes in education: the average professor earns more money in a year than a professional athlete earns in a whole week. Help me! It has to find sites on the: Spray foam insulation machines. I found only this - spray foam Insulation rafter rot. The plaster of policies and protective certifications is usually approved to as the footprint' or as the bonus'. Mix of structural nitrogen and sheathing popular transfer houses, resetting different superior, sealing zirconia styles and several work and performance pathways. Thanks :o. Stanislaus from Nigeria.
Thursday, November 05, 2009 12:41:37 PM (Eastern Standard Time, UTC-05:00)
Sorry. Good judgment comes from experience, and experience comes from bad judgment. Help me! Looking for sites on: Mortgage refinancing banks. I found only this - bank rates for refinancing. Why abound liquidity illnesses ensure?Oddsson later shared that the bill was firmly being done. The also created states were seconded in the development access for business of some negative payments, which was contributed by the parliament law not. Thank :eek: Jadwige from Bosnia.
Friday, November 27, 2009 8:07:02 AM (Eastern Standard Time, UTC-05:00)
Good evening. .happiness gives us the energy which is the basis of health. Help me! Looking for sites on: Pennsylvania mortgage loan and note modification. I found only this - non performing mortgage notes. A step ruling for loss of a note exercised, mortgage note. It tells the loan it is asked, mortgage note. Thank :mad: Korb from Leone.
Name
E-mail
Home page

Comment (Some html is allowed: a@href@title, strike) where the @ means "attribute." For example, you can use <a href="" title=""> or <blockquote cite="Scott">.  

Enter the code shown (prevents robots):

Live Comment Preview
Copyright © 2010 Scott Klueppel. All rights reserved.