/CacheManager

CacheManager is an open source caching abstraction layer for .NET written in C#. It supports various cache providers and implements many advanced features.

Primary LanguageC#Apache License 2.0Apache-2.0

CacheManager

CacheManager is an open source caching abstraction layer for .NET written in C#. It supports various cache providers and implements many advanced features.

The main goal of the CacheManager package is to make developer's life easier to handle even very complex caching scenarios.
With CacheManager it is possible to implement multiple layers of caching, e.g. in-process caching in front of a distributed cache, in just a few lines of code.

CacheManager is not just an interface to unify the programming model for various cache providers, which will make it very easy to change the caching strategy later on in a project. It also offers additional features, like cache synchronization, concurrent updates, serialization, events, performance counters... The developer can opt-in to those features only if needed.

Build Status

Build Server Status
Windows, KoreBuild Build status
Windows, MSBuild Build status
Linux, Mono Build Status

CacheManager Nuget Packages

Package Name .Net 4.0 .Net 4.5 DNX 4.5.1 Dotnet 5.2 Dotnet 5.4
[CacheManager.Core] Core.nuget x x x x x
[CacheManager.Serialization.Json] Json.nuget x x x x x
[CacheManager.StackExchange.Redis] Redis.nuget x x x - -
[CacheManager.SystemRuntimeCaching] SystemRuntimeCaching.nuget x x x - -
[CacheManager.Memcached] Memcached.nuget x x x - -
[CacheManager.Microsoft.Extensions.Configuration] Configuration.nuget - - x - x
[CacheManager.Microsoft.Extensions.Logging] Logging.nuget - - x - x
[CacheManager.Web] Web.nuget - x x - -
[CacheManager.Couchbase] Couchbase.nuget - x x - -

Beta Packages

Beta versions of the CacheManager packages are getting pushed to https://www.myget.org/gallery/cachemanager on each build. Add the following feed, if you want to play with the not yet released bits:

https://www.myget.org/F/cachemanager/api/v3/index.json

To find which check-in created which build, use this build history.

Documentation

Documentation can be found within the articles folder of the Cache Manager's repository and hosted on my website:

  • Getting Started Is a very basic introduction of how to install and use Cache Manager
  • Configuration Explains how to configure Cache Manager via code or configuration file
  • Features and Architecture A more advanced in depth introduction to all features of Cache Manager
  • Update Operations Explanation of why and when to use the update method instead of Put
  • Serialization Cache value serialization and configuration options explained in detail.
  • Cache Synchronization Use case for and explanation of the Cache Backplane feature.
  • Logging The logging abstraction and implementations explained

There is also from source generated html documentation available online.

Examples

Features in Version: 0.8.x

  • One common interface for handling different caching technologies: ICache<T>
  • Configurable by
    • Code with many different paths and a fluent configuration builder
    • Microsoft.Extensions.Configuration
    • App/Web.config
    • See configuration docs
  • Support for different cache providers
  • Serialization can now be configured. Serialization is only needed in distributed caches. The default implementation uses binary serialization. The Serialization.Json packages provides a Newtonsoft.Json based alternative.
  • Update values with lock or transaction for distributed caches. The interfaced provides a simple update method which internally ensures you work with the latest version. And CacheManager handles version conflicts for you.
  • Logging CacheManager comes with an extensible logging API ([see samples] (https://github.com/MichaCo/CacheManager/blob/master/samples/CacheManager.Examples/Program.cs#L31)).
    • All standard cache operations are logged
    • Based on log levels more or less information will be logged (try Trace and Debug)
    • Current concrete implementation is based on the ASP.NET Core logging. Other implementation of CacheManager's ILoggerFactory might follow.
  • Strongly typed cache interface.
  • Multiple layers By having multiple cache handles managed by CacheManager, you can easily implement layered caches. For example, an in process cache in front of your distributed cache, to make read access faster. CacheManager will synchronize those layers for you.
    • Put and Add operations will always be executed on all cache handles registered on the manager.
    • On Get, there are different configuration options defined by CacheUpdateMode, if the item was available in one cache handle:
      • None: No update across the cache handles on Get
      • Up: Updates the handles "above"
      • All: Updates/Adds the item to all handles
  • Expiration: It is possible to configure the expiration per cache handle within the manager or per cache item. The following are the supported expiration modes:
    • Sliding expiration: On cache hit, the cache item expiration timeout will be extended by the configured amount.
    • Absolute expiration: The cache item will expire after the configured timeout.
  • Cache Regions: Even if some cache systems do not support or implement cache regions, the CacheManager implements the mechanism. This can be used to for example group elements and remove all of them at once.
  • Statistics: Counters for all kind of cache actions.
  • Performance Counters: To be able to inspect certain numbers with perfmon, CacheManager supports performance counters per instance of the manager and per cache handle.
  • Event System: CacheManager triggers events for common cache actions: OnGet, OnAdd, OnPut, OnRemove, OnClear, OnClearRegion
  • System.Web.OutputCache implementation to use CacheManager as OutputCache provider which makes the OutputCache extremely flexible, for example by using a distributed cache like Redis across many web servers.
  • Cache clients synchronization
    • Implemented with the Redis pub/sub feature
    • (Other implementations without Redis might be an option for a later version)
  • Supports .Net 4.0, .Net 4.5, ASP.NET DNX 4.5.1 and Core

Benchmarks

Benchmarks have been implemented with BenchmarkDotNet.

All CacheManager instances used in the benchmarks have only one cache handle configured, either the Dictionary, System.Runtime or Redis handle.

We are using the same configuration for all benchmarks and running two jobs each, one for x86 and one for x64. Regarding the different platforms, the conclusion is obviously that x64 is always faster than the x86 platform, but x64 consumes slightly more memory of course.

BenchmarkDotNet=v0.9.1.0
OS=Microsoft Windows NT 6.2.9200.0
Processor=Intel(R) Core(TM) i7-6700 CPU @ 3.40GHz, ProcessorCount=8
Frequency=3328117 ticks, Resolution=300.4702 ns
HostCLR=MS.NET 4.0.30319.42000, Arch=32-bit RELEASE

Type=PutWithRegionSingleBenchmark  Mode=Throughput  Platform=X64  
Jit=RyuJit  LaunchCount=1  WarmupCount=2  
TargetCount=100  

Add

Adding one item per run

Redis will be a lot slower in this scenario because CacheManager waits for the response to be able to return the bool value if the key has been added or not. In general, it is good to see how fast the Dictionary handle is compared to the System.Runtime one. One thing you cannot see here is that also the memory footprint of the Dictionary handle is much lower.

Method Platform Median StdDev Scaled
Dictionary X64 1.7254 us 0.0511 us 1.00
Dictionary X86 1.9563 us 0.0399 us 1.00
Runtime |      X64 |  4.9839 us | 0.0778 us |   2.89 |
Runtime |      X86 |  7.0324 us | 0.2012 us |   3.59 |
  Redis |      X64 | 56.6671 us | 1.7202 us |  32.84 |
  Redis |      X86 | 58.0775 us | 0.9517 us |  29.69 |      

Adding one item per run with using region

Method Platform Median StdDev Scaled
Dictionary X64 1.8094 us 0.0386 us 1.00
Dictionary X86 2.5029 us 0.1179 us 1.00
Runtime |      X64 |  6.6934 us | 0.1275 us |   3.70 |
Runtime |      X86 |  9.2334 us | 0.1637 us |   3.69 |
  Redis |      X64 | 58.5355 us | 1.7054 us |  32.35 |
  Redis |      X86 | 61.0272 us | 1.4178 us |  24.38 |      

Put

Put 1 item per run Redis is as fast as the other handles in this scenario because CacheManager uses fire and forget for those operations. For Put it doesn't matter to know if the item has been added or updated...

Method Platform Median StdDev Scaled
Dictionary X64 1.6802 us 0.0235 us 1.00
Dictionary X86 1.9445 us 0.0341 us 1.00
Runtime |      X64 | 4.4431 us | 0.0651 us |   2.64 |
Runtime |      X86 | 6.5231 us | 0.1063 us |   3.35 |
  Redis |      X64 | 2.6869 us | 0.0934 us |   1.60 |
  Redis |      X86 | 3.5490 us | 0.0848 us |   1.83 |

Put 1 item per run with region

Method Platform Median StdDev Scaled
Dictionary X64 1.7401 us 0.0365 us 1.00
Dictionary X86 2.4589 us 0.1022 us 1.00
Runtime |      X64 | 6.1772 us | 0.3683 us |   3.55 |
Runtime |      X86 | 9.9298 us | 0.5574 us |   4.04 |
  Redis |      X64 | 3.0807 us | 0.0906 us |   1.77 |
  Redis |      X86 | 3.6385 us | 0.2123 us |   1.48 |

Get

Get 1 item per run With Get operations we can clearly see how much faster an in-memory cache is, compared to the distributed variant. That's why it makes so much sense to use CacheManager with a first and secondary cache layer.

Method Platform Median StdDev Scaled
Dictionary X64 169.8708 ns 4.6186 ns 1.00
Dictionary X86 540.0879 ns 8.9363 ns 1.00
Runtime |      X64 |    310.4025 ns |     8.2928 ns |   1.83 |
Runtime |      X86 |  1,030.5911 ns |     8.0532 ns |   1.91 |
  Redis |      X64 | 56,917.3537 ns | 2,035.4765 ns | 335.06 |
  Redis |      X86 | 59,519.5459 ns | 1,527.9613 ns | 110.20 |

Get 1 item per run with region

Method Platform Median StdDev Scaled
Dictionary X64 234.9676 ns 7.9172 ns 1.00
Dictionary X86 600.0429 ns 13.7090 ns 1.00
Runtime |      X64 |    497.4116 ns |    15.6998 ns |   2.12 |
Runtime |      X86 |  1,191.6452 ns |    19.8484 ns |   1.99 |
  Redis |      X64 | 57,257.9868 ns | 1,705.0148 ns | 243.68 |
  Redis |      X86 | 61,789.3944 ns | 1,775.6064 ns | 102.97 |

[TOC]