Calling all developers! We invite you to provide your input on Feature Experimentation by completing this brief survey.

 

enba
Feb 15, 2010
  5908
(3 votes)

Virtual Roles and Visitor Segmentation the diferent implementation

This post is inspired by very interesting series of posts by Allan Thraen about how to use virtual roles for personalization and segmentation.For reference you can (and should) read here and here. Also, recently I’ve got hooked on .NET Reactive Extensions so I decided to see if I can implement Alans ScoreRole and PageViewRole to “observe” pages they are interested into and do their work.

Code for the VirtualRolesSample pack is available on codeplex.

Virtual roles use provider model and have no knowledge of httpcontext so to be able to associate page visit to a certain role Alan implemented a plugin in a VirtualRolesInit class that basically subscribes on page.Load event and hadles all the logic for all profiles there. I had a goal to separate this event handling into single role provider so that each role provider would be responsible of it’s own event.

I have done a couple of changes to the VirtualRolesInit  class to be able to acomplish this.

First, I removed the PlugInAttribute interface and implemented instead the new IInitializableModule which is the new initialization API used in EPiServer CMS 6(Magnus Stråle has several posts about it). This means that my changes will not work with previous versinos of EPiServer but it is quite easy to port this code back to the plugin mechanism.

Second, I implemented IObservable<PageBase> on VirtualRolesInit so that role classes can observe on it. This allowes me to remove all the logic form the initialization and page.Load event handler.

using System;
using System.Collections.Generic;
using EPiServer.Framework;
using EPiServer.Framework.Initialization;


namespace EPiServer.Research.VirtualRoles
{
    [ModuleDependency(typeof(EPiServer.Web.InitializationModule))]
    public class VirtualRolesInit : IInitializableModule, IObservable<PageBase>
    {
        private bool _initialized;
        private static List<IObserver<PageBase>> _observers = new List<IObserver<PageBase>>();
        private static readonly object _syncRoot = new object();
       
        #region IInitializableModule Members

        public void Initialize(InitializationEngine context)
        {
            if (!_initialized)
            {
                PageBase.PageSetup += new PageSetupEventHandler(PageBase_PageSetup);
                _initialized = true;
            }
        }

        public void Uninitialize(InitializationEngine context){}

        public void Preload(string[] parameters){}

        #endregion

        #region IObservable<PageBase> Members

        public IDisposable Subscribe(IObserver<PageBase> observer)
        {
            lock (_syncRoot)
            {
                _observers.Add(observer);
            }
            return new ObserverReference(observer);
        }

        #endregion

        void PageBase_PageSetup(PageBase sender, PageSetupEventArgs e)
        {
            sender.Load += new EventHandler(Page_Load);
        }

        void Page_Load(object sender, EventArgs e)
        {
            if (sender is PageBase)
            {
                lock (_syncRoot)
                {
                    foreach (var observer in _observers)
                    {
                        observer.OnNext(sender as PageBase);
                    }
                }
            }
        }
    }
}
So now we are able to observe this class from a single role. In the initialize method of the ScoreRole I subscribe to the VirtualRolesInit and handle all the pages that it sends to me.
public override void Initialize(string name, System.Collections.Specialized.NameValueCollection config)
{
      //.... initialization code for Score role here
       VirtualRolesInit vri = new VirtualRolesInit();
       vri.Subscribe((page) => HandleScore(page));            
 }

 private void HandleScore(PageBase page)
 { 
      if (page != null && page.CurrentPage != null && page.CurrentPage[ScoreName] != null && page.CurrentPage[ScoreName] is int)
      {
           int score = (int)page.CurrentPage[ScoreName];
            EPiServerProfile.Current.SetPropertyValue(ScoreName, ((int)EPiServerProfile.Current[ScoreName]) + score);
            EPiServerProfile.Current.Save();
       }
 }

The same happeds for PageViewRole:

public override void Initialize(string name, System.Collections.Specialized.NameValueCollection config)
{
    // ...... initialization code for pageViewRole here
    VirtualRolesInit vri = new VirtualRolesInit();
    vri.Subscribe(page => HandlePage(page));
}

private void HandlePage(PageBase page)
{
    if (page.Session["PageViews"] == null)
    {
        page.Session.Add("PageViews", new List<string>());
    }

    var lst = page.Session["PageViews"] as List<string>;

    if (!lst.Contains(page.CurrentPageLink.ToString()))
    {
        lst.Add(page.CurrentPageLink.ToString());
    }
}
Thats it. The only clarification I still own to the careful readers of this post is that ObserverReference that subscribe method returns.

Since the IObservable<T> Subscribe method returns an IDisposable object I implemented a type that has one property of name observer and that will give you a referfence of your Observer object.You can of course create your own object as needed.  Here is the code:

class ObserverReference : IDisposable
{
    private IObserver<PageBase> _observer;
    public ObserverReference(IObserver<PageBase> observer)
    {
        _observer = observer;
    }

    public IObserver<PageBase> Observer
    {
        get
        {
            return _observer;
        }
    }
    #region IDisposable Members

    public void Dispose()
    {
        _observer = null;
    }
    #endregion
}

Try it. It does work on my machine. ;)

Feb 15, 2010

Comments

Sep 21, 2010 10:33 AM

Looking really good, Enes. Could you perhaps put your changes in a new branch on the codeplex project for virtual roles? That would be awesome!

Sep 21, 2010 10:33 AM

Thanks Allan I'll do it gladly, I guess I need some kind of invite to the project ... or do I?

Please login to comment.
Latest blogs
Level Up with Optimizely's Newly Relaunched Certifications!

We're thrilled to announce the relaunch of our Optimizely Certifications—designed to help partners, customers, and developers redefine what it mean...

Satata Satez | Jan 14, 2025

Introducing AI Assistance for DBLocalizationProvider

The LocalizationProvider for Optimizely has long been a powerful tool for enhancing the localization capabilities of Optimizely CMS. Designed to ma...

Luc Gosso (MVP) | Jan 14, 2025 | Syndicated blog

Order tabs with drag and drop - Blazor

I have started to play around a little with Blazor and the best way to learn is to reimplement some old stuff for CMS12. So I took a look at my old...

Per Nergård | Jan 14, 2025

Product Recommendations - Common Pitfalls

With the added freedom and flexibility that the release of the self-service widgets feature for Product Recommendations provides you as...

Dylan Walker | Jan 14, 2025

My blog is now running using Optimizely CMS!

It's official! You are currently reading this post on my shiny new Optimizely CMS website.  In the past weeks, I have been quite busy crunching eve...

David Drouin-Prince | Jan 12, 2025 | Syndicated blog

Developer meetup - Manchester, 23rd January

Yes, it's that time of year again where tradition dictates that people reflect on the year gone by and brace themselves for the year ahead, and wha...

Paul Gruffydd | Jan 9, 2025