Click here to Skip to main content
15,886,362 members
Articles / Database Development / NoSQL

RavenDB - An Introduction

,
Rate me:
Please Sign up or sign in to vote.
4.87/5 (38 votes)
28 Apr 2010CPOL7 min read 263.1K   2.7K   112  
An introduction to RavenDB - a new open source .NET document database using .NET 4.0 and VS 2010
using System.Collections.Generic;
using System.Linq;
using System.Net;
using Raven.Client.Document;
using Raven.Client.Shard.ShardStrategy;
using Raven.Client.Shard.ShardStrategy.ShardResolution;
using System;

namespace Raven.Client.Shard
{
	public class ShardedDocumentSession : IDocumentSession
	{
		public void Clear()
		{
			foreach (var shardSession in shardSessions)
			{
				shardSession.Clear();
			}
		}

        public bool UseOptimisticConcurrency
        {
            get
            {
                return shardSessions.All(x => x.UseOptimisticConcurrency);
            }
            set
            {
                foreach (var shardSession in shardSessions)
                {
                    shardSession.UseOptimisticConcurrency = value;
                }
            }
        }

	    public event Action<object> Stored;

        public ShardedDocumentSession(IShardStrategy shardStrategy, params IDocumentSession[] shardSessions)
		{
            this.shardStrategy = shardStrategy;
            this.shardSessions = shardSessions;

            foreach (var shardSession in shardSessions)
            {
                shardSession.Stored += Stored;
            }
		}

		private readonly IShardStrategy shardStrategy;
		private readonly IDocumentSession[] shardSessions;

		public T Load<T>(string id)
		{
            var shardIds = shardStrategy.ShardResolutionStrategy.SelectShardIdsFromData(ShardResolutionStrategyData.BuildFrom(typeof(T))) ?? new string[] { };

            IDocumentSession[] shardsToUse = shardSessions.Where(x => shardIds.Contains(x.StoreIdentifier)).ToArray();

            //default to all sessions if none found to use
            if (shardIds.Count == 0)
                shardsToUse = shardSessions;

            //if we can narrow down to single shard, explicitly call it
            if (shardsToUse.Length == 1)
            {
                return shardsToUse[0].Load<T>(id);
            }
			var results = shardStrategy.ShardAccessStrategy.Apply(shardsToUse, x =>
			{
				try
				{
					return new[] {x.Load<T>(id)};
				}
				catch (WebException e)
				{
					var httpWebResponse = e.Response as HttpWebResponse; // we ignore 404, it is expected
					if (httpWebResponse == null || httpWebResponse.StatusCode != HttpStatusCode.NotFound)
						throw;
					return null;
				}
			});

			return results
				.Where(x => ReferenceEquals(null, x) == false)
				.FirstOrDefault();
		}

	    public T[] Load<T>(params string[] ids)
	    {
	        return shardStrategy.ShardAccessStrategy.Apply(shardSessions, sessions => sessions.Load<T>(ids)).ToArray();
	    }

	    public void Delete<T>(T entity)
	    {
            if(ReferenceEquals(entity,null))
                throw new ArgumentNullException("entity");

            var shardIds = shardStrategy.ShardSelectionStrategy.SelectShardIdForExistingObject(entity);

	        var shardToUse =
	            shardSessions.Where(x => shardIds.Contains(x.StoreIdentifier)).FirstOrDefault();

            if(shardToUse == null)
                throw new InvalidOperationException("Could not find shard id for: " + entity);

            shardToUse.Delete(entity);
	    }

	    private IDocumentSession GetSingleShardSession(string shardId)
        {
			var shardSession = shardSessions.FirstOrDefault(x => x.StoreIdentifier == shardId);
            if (shardSession == null) 
				throw new ApplicationException("Can't find a shard with identifier: " + shardId);
            return shardSession;
        }

        private void SingleShardAction<T>(T entity, Action<IDocumentSession> action)
        {
            string shardId = shardStrategy.ShardSelectionStrategy.SelectShardIdForNewObject(entity);
            if (String.IsNullOrEmpty(shardId))
				throw new ApplicationException("Can't find a shard to use for entity: " + entity);

            var shardSession = GetSingleShardSession(shardId);

            action(shardSession);
        }

		public void Store<T>(T entity)
		{
            SingleShardAction(entity, shard => shard.Store(entity));
		}

		public void Evict<T>(T entity)
		{
			SingleShardAction(entity, session => session.Evict(entity));
		}

		/// <summary>
		/// Note that while we can assume a transaction for a single shard, cross shard transactions will NOT work.
		/// </summary>
		public void SaveChanges()
		{
			foreach (var shardSession in shardSessions)
			{
				shardSession.SaveChanges();
			}
        }

        public IDocumentQuery<T> Query<T>(string indexName)
		{
        	return new ShardedDocumentQuery<T>(indexName, shardSessions);
        }

		public string StoreIdentifier
		{
			get
			{
				return "ShardedSession";
			}
		}

        #region IDisposable Members

        public void Dispose()
        {
            foreach (var shardSession in shardSessions)
                shardSession.Dispose();

            //dereference all event listeners
            Stored = null;
        }

        #endregion

	    public void Commit(Guid txId)
	    {
	        shardStrategy.ShardAccessStrategy.Apply(shardSessions, session =>
	        {
                session.Commit(txId);
	            return new List<int>();
	        });
	    }

	    public void Rollback(Guid txId)
	    {
	        shardStrategy.ShardAccessStrategy.Apply(shardSessions, session =>
	        {
	            session.Rollback(txId);
                return new List<int>();
	        });
	    }
	}
}

By viewing downloads associated with this article you agree to the Terms of Service and the article's licence.

If a file you wish to view isn't highlighted, and is a text file (not binary), please let us know and we'll add colourisation support for it.

License

This article, along with any associated source code and files, is licensed under The Code Project Open License (CPOL)


Written By
United States United States
I've been a software developer since 1996 and have enjoyed C# since 2003. I have a Bachelor's degree in Computer Science and for some reason, a Master's degree in Business Administration. I currently do software development contracting/consulting.

Written By
Israel Israel
This member has not yet provided a Biography. Assume it's interesting and varied, and probably something to do with programming.

Comments and Discussions