BaseSyndicationHandler.cs :  » Bloggers » SubText » Subtext » Framework » Syndication » C# / CSharp Open Source

Home
C# / CSharp Open Source
1.2.6.4 mono .net core
2.2.6.4 mono core
3.Aspect Oriented Frameworks
4.Bloggers
5.Build Systems
6.Business Application
7.Charting Reporting Tools
8.Chat Servers
9.Code Coverage Tools
10.Content Management Systems CMS
11.CRM ERP
12.Database
13.Development
14.Email
15.Forum
16.Game
17.GIS
18.GUI
19.IDEs
20.Installers Generators
21.Inversion of Control Dependency Injection
22.Issue Tracking
23.Logging Tools
24.Message
25.Mobile
26.Network Clients
27.Network Servers
28.Office
29.PDF
30.Persistence Frameworks
31.Portals
32.Profilers
33.Project Management
34.RSS RDF
35.Rule Engines
36.Script
37.Search Engines
38.Sound Audio
39.Source Control
40.SQL Clients
41.Template Engines
42.Testing
43.UML
44.Web Frameworks
45.Web Service
46.Web Testing
47.Wiki Engines
48.Windows Presentation Foundation
49.Workflows
50.XML Parsers
C# / C Sharp
C# / C Sharp by API
C# / CSharp Tutorial
C# / CSharp Open Source » Bloggers » SubText 
SubText » Subtext » Framework » Syndication » BaseSyndicationHandler.cs
#region Disclaimer/Info

///////////////////////////////////////////////////////////////////////////////////////////////////
// Subtext WebLog
// 
// Subtext is an open source weblog system that is a fork of the .TEXT
// weblog system.
//
// For updated news and information please visit http://subtextproject.com/
// Subtext is hosted at Google Code at http://code.google.com/p/subtext/
// The development mailing list is at subtext@googlegroups.com 
//
// This project is licensed under the BSD license.  See the License.txt file for more information.
///////////////////////////////////////////////////////////////////////////////////////////////////

#endregion

using System;
using System.Globalization;
using System.Net;
using System.Text;
using System.Web;
using System.Web.Security;
using Subtext.Framework.Security;
using Subtext.Framework.Syndication.Compression;
using Subtext.Framework.Web.Handlers;
using Subtext.Framework.Web;

namespace Subtext.Framework.Syndication{
    /// <summary>
    /// Abstract base class used to respond to requests for 
    /// syndicated feeds such as RSS and ATOM.
    /// </summary>
    public abstract class BaseSyndicationHandler : SubtextHttpHandler
    {
        const int HttpImUsed = 226;

        protected BaseSyndicationHandler(ISubtextContext subtextContext) : base(subtextContext)
        {
        }

        protected CachedFeed Feed { get; set; }

        // TODO: Why is the private setter not used?
        protected virtual bool RequiresAdminRole { get; private set; }
        // TODO: Why is the private setter not used?
        protected virtual bool RequiresHostAdminRole { get; private set; }

        protected HttpContextBase HttpContext
        {
            get { return SubtextContext.HttpContext; }
        }

        /// <summary>
        /// Returns the "If-Modified-Since" HTTP header.  This indicates 
        /// the last time the client requested data and is used to 
        /// determine whether new data is to be sent.
        /// </summary>
        /// <value></value>
        protected string LastModifiedHeader
        {
            get { return HttpContext.Request.Headers["If-Modified-Since"]; }
        }

        /// <summary>
        /// Returns the "If-None-Match" HTTP header.  This is used to indicate 
        /// a conditional GET and is used to implement RFC3229 with feeds 
        /// <see href="http://bobwyman.pubsub.com/main/2004/09/using_rfc3229_w.html"/>.
        /// </summary>
        /// <value></value>
        protected string IfNonMatchHeader
        {
            get { return HttpContext.Request.Headers["If-None-Match"]; }
        }

        /// <summary>
        /// Gets the Publish Date of the last feed item received by the client. 
        /// This is used to determine whether the client is up to date 
        /// or whether the client is ready to receive new feed items. 
        /// We will then send just the difference.
        /// </summary>
        /// <value></value>
        protected DateTime PublishDateOfLastFeedItemReceived
        {
            get
            {
                if(!string.IsNullOrEmpty(IfNonMatchHeader))
                {
                    try
                    {
                        return DateTime.Parse(IfNonMatchHeader, CultureInfo.InvariantCulture);
                    }
                    catch(FormatException)
                    {
                        //Swallow it.
                    }
                }
                return NullValue.NullDateTime;
            }
        }

        /// <summary>
        /// Gets a value indicating whether use delta encoding within this request.
        /// </summary>
        /// <value>
        ///   <c>true</c> if use delta encoding; otherwise, <c>false</c>.
        /// </value>
        protected bool UseDeltaEncoding
        {
            get { return Blog.RFC3229DeltaEncodingEnabled && AcceptDeltaEncoding; }
        }

        /// <summary>
        /// Gets the syndication writer.
        /// </summary>
        /// <returns></returns>
        protected abstract BaseSyndicationWriter SyndicationWriter { get; }

        /// <summary>
        /// Returns the "Accept-Encoding" value from the HTTP Request header. 
        /// This is a list of encodings that may be sent to the browser.
        /// </summary>
        /// <remarks>
        /// Specifically we're looking for gzip.
        /// </remarks>
        /// <value></value>
        protected string AcceptEncoding
        {
            get
            {
                string header = HttpContext.Request.Headers["Accept-Encoding"];
                if(header != null)
                {
                    return header;
                }
                return string.Empty;
            }
        }

        /// <summary>
        /// Gets the accept IM header from the request.
        /// </summary>
        /// <value></value>
        protected string AcceptImHeader
        {
            get
            {
                string header = HttpContext.Request.Headers["A-IM"];
                if(header != null)
                {
                    return header;
                }
                return string.Empty;
            }
        }

        /// <summary>
        /// Gets a value indicating whether the client accepts 
        /// <see href="http://bobwyman.pubsub.com/main/2004/09/using_rfc3229_w.html">RFC3229 Feed Delta 
        /// Encoding</see>. 
        /// </summary>
        /// <value>
        ///   <c>true</c> if [accepts delta encoding]; otherwise, <c>false</c>.
        /// </value>
        protected bool AcceptDeltaEncoding
        {
            get { return AcceptImHeader.IndexOf("feed") >= 0; }
        }

        /// <summary>
        /// Gets a value indicating whether the client accepts gzip compression.
        /// </summary>
        /// <value>
        ///   <c>true</c> if accepts gzip compression; otherwise, <c>false</c>.
        /// </value>
        protected bool AcceptGzipCompression
        {
            get
            {
                return AcceptEncoding.IndexOf("gzip") >= 0 ||
                       AcceptImHeader.IndexOf("gzip") >= 0;
            }
        }

        /// <summary>
        /// Returns true if the feed is the main feed.  False for category feeds and comment feeds.
        /// </summary>
        protected abstract bool IsMainfeed { get; }

        /// <summary>
        /// Compares the requesting clients <see cref="LastModifiedHeader"/> against 
        /// the date the feed was last updated.  If the feed hasn't been updated, then 
        /// it sends a 304 HTTP header indicating such.
        /// </summary>
        /// <returns></returns>
        protected virtual bool IsLocalCacheOk()
        {
            string dt = LastModifiedHeader;
            if(dt != null)
            {
                try
                {
                    DateTime feedDt = DateTime.Parse(dt, CultureInfo.InvariantCulture);
                    DateTime lastUpdated = ConvertLastUpdatedDate(Blog.LastUpdated);
                    TimeSpan ts = feedDt - lastUpdated;

                    //We need to allow some margin of error.
                    return Math.Abs(ts.TotalMilliseconds) <= 500;
                }
                catch(FormatException)
                {
                    //TODO: Review
                    //swallow it for now.
                    //Some browsers send a funky last modified header.
                    //We don't want to throw an exception in those cases.
                }
            }
            return false;
        }

        /// <summary>
        /// Returns whether or not the http cache is OK.
        /// </summary>
        /// <returns></returns>
        protected virtual bool IsHttpCacheOk()
        {
            if(HttpContext.Cache == null)
            {
                Feed = null;
                return false;
            }

            Feed = HttpContext.Cache[CacheKey(PublishDateOfLastFeedItemReceived)] as CachedFeed;
            if(Feed == null)
            {
                return false;
            }
            return Feed.LastModified == ConvertLastUpdatedDate(Blog.LastUpdated);
        }

        /// <summary>
        /// Send the HTTP status code 304 to the response this instance.
        /// </summary>
        private void Send304()
        {
            HttpContext.Response.StatusCode = 304;
        }

        /// <summary>
        /// Convert a date to the server time.
        /// </summary>
        protected DateTime ConvertLastUpdatedDate(DateTime dateTime)
        {
            return Blog.TimeZone.ToServerDateTime(dateTime);
        }

        /// <summary>
        /// Processs the feed. Responds to the incoming request with the 
        /// contents of the feed.
        /// </summary>
        protected virtual void ProcessFeed()
        {
            if(RedirectToFeedBurnerIfNecessary())
            {
                return;
            }

            // Checks Last Modified Header.
            if(IsLocalCacheOk())
            {
                Send304();
                return;
            }

            // Checks our cache against last modified header.
            if(!IsHttpCacheOk())
            {
                Feed = BuildFeed();
                if(Feed != null)
                {
                    if(UseDeltaEncoding && Feed.ClientHasAllFeedItems)
                    {
                        Send304();
                        return;
                    }
                    Cache(Feed);
                }
            }

            WriteFeed();
        }

        /// <summary>
        /// Returns the key used to cache this feed.
        /// </summary>
        /// <param name="dateLastViewedFeedItemPublished">Date last viewed feed item published.</param>
        /// <returns></returns>
        protected abstract string CacheKey(DateTime dateLastViewedFeedItemPublished);

        protected abstract void Cache(CachedFeed feed);

        protected virtual CachedFeed BuildFeed()
        {
            var feed = new CachedFeed {LastModified = ConvertLastUpdatedDate(Blog.LastUpdated)};
            BaseSyndicationWriter writer = SyndicationWriter;
            feed.Xml = writer.Xml;
            feed.ClientHasAllFeedItems = writer.ClientHasAllFeedItems;
            feed.Etag = writer.DateLastViewedFeedItemPublished.ToString(CultureInfo.InvariantCulture);
            feed.LatestFeedItemPublishDate = writer.DateLastViewedFeedItemPublished;

            return feed;
        }

        /// <summary>
        /// Writes the feed to the response.
        /// </summary>
        protected virtual void WriteFeed()
        {
            string encoding = null;

            if(Feed != null)
            {
                if(Blog.UseSyndicationCompression && AcceptGzipCompression)
                {
                    // We're GZip Encoding!
                    SyndicationCompressionFilter filter = SyndicationCompressionHelper.GetFilterForScheme(
                        AcceptEncoding, HttpContext.Response.Filter);

                    if(filter != null)
                    {
                        encoding = filter.ContentEncoding;
                        HttpContext.Response.Filter = filter.Filter;
                    }
                }

                if(encoding == null)
                {
                    HttpContext.Response.ContentEncoding = Encoding.UTF8;
                }

                HttpContext.Response.ContentType = "text/xml";
                HttpContext.Response.Cache.SetCacheability(HttpCacheability.Public);
                HttpContext.Response.Cache.SetLastModified(Feed.LastModified);
                HttpContext.Response.Cache.SetETag(Feed.Etag);

                if(AcceptGzipCompression)
                {
                    HttpContext.Response.AddHeader("IM", "feed, gzip");
                }
                else
                {
                    HttpContext.Response.AddHeader("IM", "feed");
                }
                if(UseDeltaEncoding)
                {
                    HttpContext.Response.StatusCode = HttpImUsed; //IM Used
                }
                else
                {
                    HttpContext.Response.StatusCode = (int)HttpStatusCode.OK;
                }

                HttpContext.Response.Write(Feed.Xml);
            }
        }

        /// <summary>
        /// Processs the request and sends the feed to the response.
        /// </summary>
        public override void ProcessRequest()
        {
            if((RequiresAdminRole && !SecurityHelper.IsAdmin) || (RequiresHostAdminRole && !SecurityHelper.IsHostAdmin))
            {
                FormsAuthentication.RedirectToLoginPage();
                return;
            }
            ProcessFeed();
        }

        // Adapted from DasBlog
        private bool RedirectToFeedBurnerIfNecessary()
        {
            //If we are using FeedBurner, only allow them to get our feed...
            if(!String.IsNullOrEmpty(Blog.RssProxyUrl))
            {
                string userAgent = HttpContext.Request.UserAgent;
                if(!String.IsNullOrEmpty(userAgent))
                {
                    // If they aren't FeedBurner and they aren't asking for a category or comment rss, redirect them!
                    if(!userAgent.StartsWith("FeedBurner") && IsMainfeed)
                    {
                        HttpContext.Response.RedirectPermanent(SubtextContext.UrlHelper.RssProxyUrl(SubtextContext.Blog).ToString());
                        return true;
                    }
                }
            }
            return false;
        }
    }
}
www.java2v.com | Contact Us
Copyright 2009 - 12 Demo Source and Support. All rights reserved.
All other trademarks are property of their respective owners.