diff --git a/EasyCaching.sln b/EasyCaching.sln
index 7d07ad0a..55b18b3b 100644
--- a/EasyCaching.sln
+++ b/EasyCaching.sln
@@ -1,4 +1,5 @@
-Microsoft Visual Studio Solution File, Format Version 12.00
+
+Microsoft Visual Studio Solution File, Format Version 12.00
# Visual Studio Version 17
VisualStudioVersion = 17.2.32616.157
MinimumVisualStudioVersion = 10.0.40219.1
@@ -72,11 +73,13 @@ Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "EasyCaching.Serialization.S
EndProject
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "EasyCaching.Bus.ConfluentKafka", "bus\EasyCaching.Bus.ConfluentKafka\EasyCaching.Bus.ConfluentKafka.csproj", "{F7FBADEB-D766-4595-949A-07104B52692C}"
EndProject
-Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "EasyCaching.Bus.Zookeeper", "bus\EasyCaching.Bus.Zookeeper\EasyCaching.Bus.Zookeeper.csproj", "{5E488583-391E-4E15-83C1-7301B4FE79AE}"
+Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "EasyCaching.Bus.Zookeeper", "bus\EasyCaching.Bus.Zookeeper\EasyCaching.Bus.Zookeeper.csproj", "{5E488583-391E-4E15-83C1-7301B4FE79AE}"
EndProject
-Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "EasyCaching.FasterKv", "src\EasyCaching.FasterKv\EasyCaching.FasterKv.csproj", "{7191E567-38DF-4879-82E1-73EC618AFCAC}"
+Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "EasyCaching.FasterKv", "src\EasyCaching.FasterKv\EasyCaching.FasterKv.csproj", "{7191E567-38DF-4879-82E1-73EC618AFCAC}"
EndProject
-Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "EasyCaching.Serialization.MemoryPack", "serialization\EasyCaching.Serialization.MemoryPack\EasyCaching.Serialization.MemoryPack.csproj", "{EEF22C21-F380-4980-B72C-F14488369333}"
+Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "EasyCaching.Serialization.MemoryPack", "serialization\EasyCaching.Serialization.MemoryPack\EasyCaching.Serialization.MemoryPack.csproj", "{EEF22C21-F380-4980-B72C-F14488369333}"
+EndProject
+Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "EasyCaching.Etcd", "src\EasyCaching.Etcd\EasyCaching.Etcd.csproj", "{984B7E9C-B8C3-4C01-BBC2-345AF28FCC11}"
EndProject
Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution
@@ -208,6 +211,10 @@ Global
{EEF22C21-F380-4980-B72C-F14488369333}.Debug|Any CPU.Build.0 = Debug|Any CPU
{EEF22C21-F380-4980-B72C-F14488369333}.Release|Any CPU.ActiveCfg = Release|Any CPU
{EEF22C21-F380-4980-B72C-F14488369333}.Release|Any CPU.Build.0 = Release|Any CPU
+ {984B7E9C-B8C3-4C01-BBC2-345AF28FCC11}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
+ {984B7E9C-B8C3-4C01-BBC2-345AF28FCC11}.Debug|Any CPU.Build.0 = Debug|Any CPU
+ {984B7E9C-B8C3-4C01-BBC2-345AF28FCC11}.Release|Any CPU.ActiveCfg = Release|Any CPU
+ {984B7E9C-B8C3-4C01-BBC2-345AF28FCC11}.Release|Any CPU.Build.0 = Release|Any CPU
EndGlobalSection
GlobalSection(SolutionProperties) = preSolution
HideSolutionNode = FALSE
@@ -244,6 +251,7 @@ Global
{5E488583-391E-4E15-83C1-7301B4FE79AE} = {B337509B-75F9-4851-821F-9BBE87C4E4BC}
{7191E567-38DF-4879-82E1-73EC618AFCAC} = {A0F5CC7E-155F-4726-8DEB-E966950B3FE9}
{EEF22C21-F380-4980-B72C-F14488369333} = {15070C49-A507-4844-BCFE-D319CFBC9A63}
+ {984B7E9C-B8C3-4C01-BBC2-345AF28FCC11} = {A0F5CC7E-155F-4726-8DEB-E966950B3FE9}
EndGlobalSection
GlobalSection(ExtensibilityGlobals) = postSolution
SolutionGuid = {63A57886-054B-476C-AAE1-8D7C8917682E}
diff --git a/README.md b/README.md
index 266053d8..97c71485 100644
--- a/README.md
+++ b/README.md
@@ -95,6 +95,7 @@ public class Startup
config.DBConfig.Endpoints.Add(new ServerEndPoint("127.0.0.1", 6379));
}, "redis1")
.WithMessagePack()//with messagepack serialization
+ .UseRedisLock()//with distributed lock
;
});
}
diff --git a/build/releasenotes.props b/build/releasenotes.props
index 5cba50e4..d1a12da0 100644
--- a/build/releasenotes.props
+++ b/build/releasenotes.props
@@ -72,5 +72,8 @@
1. Upgrading dependencies.
+
+ 1. Add etcd module.
+
diff --git a/build/version.props b/build/version.props
index 8cadff5e..d595021e 100644
--- a/build/version.props
+++ b/build/version.props
@@ -24,5 +24,6 @@
1.9.0
1.9.0
1.9.0
+ 1.9.0
diff --git a/bus/EasyCaching.Bus.Zookeeper/Configurations/ZookeeperOptionsExtension.cs b/bus/EasyCaching.Bus.Zookeeper/Configurations/ZookeeperOptionsExtension.cs
index 901ff238..6e4b957b 100644
--- a/bus/EasyCaching.Bus.Zookeeper/Configurations/ZookeeperOptionsExtension.cs
+++ b/bus/EasyCaching.Bus.Zookeeper/Configurations/ZookeeperOptionsExtension.cs
@@ -33,4 +33,4 @@ public void AddServices(IServiceCollection services)
services.AddSingleton();
}
}
-}
\ No newline at end of file
+}
diff --git a/docs/Etcd.md b/docs/Etcd.md
new file mode 100644
index 00000000..0d64c376
--- /dev/null
+++ b/docs/Etcd.md
@@ -0,0 +1,86 @@
+# DefaultEtcdCachingProvider
+
+Etcd is a hybrid memory and disk Kv Store, so it can support much larger data storage than memory.
+
+EasyCaching.Etcd is a lib that is based on **EasyCaching.Core** and **dotnet-etcd-core**.
+
+
+# How to use ?
+
+## 1. Install the package via Nuget
+
+```
+Install-Package EasyCaching.Etcd
+```
+
+## 2. Config in Startup class
+
+```csharp
+public class Startup
+{
+ //...
+
+ public void ConfigureServices(IServiceCollection services)
+ {
+ services.AddMvc();
+
+ services.AddEasyCaching(option =>
+ {
+ //use Etcd cache
+ option.UseEtcd(config =>
+ {
+ config.Address = "http://127.0.0.1:2379";
+ config.Timeout = 30000;
+ // Etcd must be set SerializerName
+ config.SerializerName = "msg";
+ })
+ .WithMessagePack("msg");
+ });
+ }
+}
+```
+
+### 3. Call the EasyCachingProvider
+
+The following code show how to use EasyCachingProvider in ASP.NET Core Web API.
+
+```csharp
+[Route("api/[controller]")]
+public class ValuesController : Controller
+{
+ private readonly IEasyCachingProvider _provider;
+
+ public ValuesController(IEasyCachingProvider provider)
+ {
+ this._provider = provider;
+ }
+
+ [HttpGet]
+ public string Get()
+ {
+ //Remove
+ _provider.Remove("demo");
+
+ //Set
+ _provider.Set("demo", "123", TimeSpan.FromMinutes(1));
+
+ //Get
+ var res = _provider.Get("demo", () => "456", TimeSpan.FromMinutes(1));
+
+ //Get without data retriever
+ var res = _provider.Get("demo");
+
+ //Remove Async
+ await _provider.RemoveAsync("demo");
+
+ //Set Async
+ await _provider.SetAsync("demo", "123", TimeSpan.FromMinutes(1));
+
+ //Get Async
+ var res = await _provider.GetAsync("demo",async () => await Task.FromResult("456"), TimeSpan.FromMinutes(1));
+
+ //Get without data retriever Async
+ var res = await _provider.GetAsync("demo");
+ }
+}
+```
diff --git a/docs/Hybrid.md b/docs/Hybrid.md
index 3552a2eb..fc1196fb 100644
--- a/docs/Hybrid.md
+++ b/docs/Hybrid.md
@@ -19,6 +19,7 @@ Install-Package EasyCaching.HybridCache
Install-Package EasyCaching.InMemory
Install-Package EasyCaching.Redis
Install-Package EasyCaching.Bus.Redis
+Install-Package EasyCaching.Serialization.Json
```
## 2. Config in Startup class
@@ -41,7 +42,7 @@ public class Startup
// distributed
option.UseRedis(config =>
{
- config.DBConfig.Endpoints.Add(new Core.Configurations.ServerEndPoint("127.0.0.1", 6379));
+ config.DBConfig.Endpoints.Add(new ServerEndPoint("127.0.0.1", 6379));
config.DBConfig.Database = 5;
config.SerializerName = "myjson";
}, "myredis");
diff --git a/mkdocs.yml b/mkdocs.yml
index ca4bb888..26e4965b 100644
--- a/mkdocs.yml
+++ b/mkdocs.yml
@@ -15,6 +15,7 @@ pages:
- Disk : Disk.md
- LiteDB : LiteDB.md
- FasterKv: FasterKv.md
+ - Etcd: Etcd.md
# - ProviderFactory : ProviderFactory.md
- Provider Factory:
- Overview : FactoryOverview.md
diff --git a/sample/EasyCaching.Demo.ConsoleApp/EasyCaching.Demo.ConsoleApp.csproj b/sample/EasyCaching.Demo.ConsoleApp/EasyCaching.Demo.ConsoleApp.csproj
index 61cbb702..f9d4a7a4 100644
--- a/sample/EasyCaching.Demo.ConsoleApp/EasyCaching.Demo.ConsoleApp.csproj
+++ b/sample/EasyCaching.Demo.ConsoleApp/EasyCaching.Demo.ConsoleApp.csproj
@@ -14,6 +14,7 @@
+
diff --git a/sample/EasyCaching.Demo.ConsoleApp/Program.cs b/sample/EasyCaching.Demo.ConsoleApp/Program.cs
index ccb3f2e1..4fc83d4c 100644
--- a/sample/EasyCaching.Demo.ConsoleApp/Program.cs
+++ b/sample/EasyCaching.Demo.ConsoleApp/Program.cs
@@ -6,10 +6,13 @@ namespace EasyCaching.Demo.ConsoleApp
using EasyCaching.Disk;
using EasyCaching.Serialization.SystemTextJson.Configurations;
using EasyCaching.SQLite;
+ using Google.Protobuf.WellKnownTypes;
using MemoryPack;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Options;
+ using Newtonsoft.Json;
using System;
+ using System.Collections.Generic;
using System.IO;
class Program
@@ -27,6 +30,17 @@ static void Main(string[] args)
option.UseInMemory("m1");
+ option.UseEtcd(options =>
+ {
+ options.Address = "http://127.0.0.1:2379";
+ options.Timeout = 30000;
+ options.SerializerName= "json";
+ }, "e1").WithJson(jsonSerializerSettingsConfigure: x =>
+ {
+ x.TypeNameHandling = Newtonsoft.Json.TypeNameHandling.None;
+ x.ReferenceLoopHandling = ReferenceLoopHandling.Ignore;
+ }, "json");
+
option.UseRedis((options) =>
{
options.SerializerName = "mempack";
@@ -43,17 +57,22 @@ static void Main(string[] args)
};
}, "s1");
- // option.WithJson(jsonSerializerSettingsConfigure: x =>
- // {
- // x.TypeNameHandling = Newtonsoft.Json.TypeNameHandling.None;
- // }, "json");
+ option.WithJson(jsonSerializerSettingsConfigure: x =>
+ {
+ x.TypeNameHandling = Newtonsoft.Json.TypeNameHandling.None;
+ x.ReferenceLoopHandling = ReferenceLoopHandling.Ignore;
+ }, "json");
option.UseDisk(cfg =>
{
cfg.DBConfig = new DiskDbOptions { BasePath = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "Cache") };
cfg.SerializerName = "msgpack";
}, "disk")
- .WithJson("json")
+ .WithJson(jsonSerializerSettingsConfigure: x =>
+ {
+ x.TypeNameHandling = Newtonsoft.Json.TypeNameHandling.None;
+ x.ReferenceLoopHandling = ReferenceLoopHandling.Ignore;
+ }, "json")
.WithSystemTextJson("sysjson")
.WithMessagePack("msgpack");
});
@@ -61,15 +80,15 @@ static void Main(string[] args)
IServiceProvider serviceProvider = services.BuildServiceProvider();
var factory = serviceProvider.GetService();
- // var redisCache = factory.GetCachingProvider("r1");
- //
- // redisCache.Set("rkey", new Product() { Name = "test" }, TimeSpan.FromSeconds(20));
- //
- // var redisAllKey = redisCache.GetAllKeysByPrefix("rkey");
- //
- // var redisVal = redisCache.Get("rkey");
- //
- // Console.WriteLine($"redis cache get value, {redisVal.HasValue} {redisVal.IsNull} {redisVal.Value}");
+ var redisCache = factory.GetCachingProvider("r1");
+
+ redisCache.Set("rkey", new Product() { Name = "test" }, TimeSpan.FromSeconds(20));
+
+ var redisAllKey = redisCache.GetAllKeysByPrefix("rkey");
+
+ var redisVal = redisCache.Get("rkey");
+
+ Console.WriteLine($"redis cache get value, {redisVal.HasValue} {redisVal.IsNull} {redisVal.Value}");
var prod = new Product()
{
@@ -104,6 +123,19 @@ static void Main(string[] args)
var diskVal = diskCache.Get("diskkey");
Console.WriteLine($"disk cache get value, {diskVal.HasValue} {diskVal.IsNull} {diskVal.Value} ");
+ //etcd cache
+ var etcdCache = factory.GetCachingProvider("e1");
+ var re11 = etcdCache.GetAllKeysByPrefix("emk");
+ var re12 = etcdCache.GetByPrefix("emk");
+ etcdCache.Set("emkey3", prod, TimeSpan.FromSeconds(2000));
+ var re13 = etcdCache.Get("emkey3");
+ var re14 = etcdCache.GetAll(new List()
+ {
+ "emkey3"
+ });
+ etcdCache.Remove("emkey3");
+ Console.WriteLine($"etcd cache get value, {re13.HasValue} {re13.IsNull} {re13.Value} ");
+
Console.ReadKey();
}
}
diff --git a/sample/EasyCaching.Demo.Providers/Controllers/ValuesController.cs b/sample/EasyCaching.Demo.Providers/Controllers/ValuesController.cs
index d7f8878a..dc4f24d7 100644
--- a/sample/EasyCaching.Demo.Providers/Controllers/ValuesController.cs
+++ b/sample/EasyCaching.Demo.Providers/Controllers/ValuesController.cs
@@ -8,7 +8,7 @@
[Route("api/[controller]")]
public class ValuesController : Controller
{
- //1. InMemory,Memcached,Redis,SQLite,FasterKv
+ //1. InMemory,Memcached,Redis,SQLite,FasterKv,Etcd
private readonly IEasyCachingProvider _provider;
public ValuesController(IEasyCachingProvider provider)
diff --git a/sample/EasyCaching.Demo.Providers/EasyCaching.Demo.Providers.csproj b/sample/EasyCaching.Demo.Providers/EasyCaching.Demo.Providers.csproj
index 116c4206..be344116 100644
--- a/sample/EasyCaching.Demo.Providers/EasyCaching.Demo.Providers.csproj
+++ b/sample/EasyCaching.Demo.Providers/EasyCaching.Demo.Providers.csproj
@@ -7,6 +7,8 @@
+
+
diff --git a/sample/EasyCaching.Demo.Providers/Startup.cs b/sample/EasyCaching.Demo.Providers/Startup.cs
index 5372e2ae..2fe48365 100644
--- a/sample/EasyCaching.Demo.Providers/Startup.cs
+++ b/sample/EasyCaching.Demo.Providers/Startup.cs
@@ -8,6 +8,7 @@
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Hosting;
using Microsoft.Extensions.Logging;
+ using Microsoft.Extensions.Options;
public class Startup
{
@@ -69,6 +70,16 @@ public void ConfigureServices(IServiceCollection services)
config.SerializerName = "msg";
})
.WithMessagePack("msg");
+
+ //use Etcd
+ option.UseEtcd(config =>
+ {
+ config.Address = "http://127.0.0.1:2379";
+ config.Timeout = 30000;
+ // Etcd must be set SerializerName
+ config.SerializerName = "json";
+ })
+ .WithMessagePack("msg2").WithJson("json");
});
}
diff --git a/src/EasyCaching.Core/Internal/CachingProviderType.cs b/src/EasyCaching.Core/Internal/CachingProviderType.cs
index ed1d8f27..d328e4a5 100644
--- a/src/EasyCaching.Core/Internal/CachingProviderType.cs
+++ b/src/EasyCaching.Core/Internal/CachingProviderType.cs
@@ -14,5 +14,6 @@ public enum CachingProviderType
Ext2,
LiteDB,
FasterKv,
+ Etcd
}
}
diff --git a/src/EasyCaching.Core/Internal/EasyCachingConstValue.cs b/src/EasyCaching.Core/Internal/EasyCachingConstValue.cs
index 342a0ae4..fc84be55 100644
--- a/src/EasyCaching.Core/Internal/EasyCachingConstValue.cs
+++ b/src/EasyCaching.Core/Internal/EasyCachingConstValue.cs
@@ -45,6 +45,11 @@ public class EasyCachingConstValue
///
public const string HybridSection = "easycaching:hybrid";
+ ///
+ /// The etcd section.
+ ///
+ public const string EtcdSection = "easycaching:etcd";
+
///
/// The redis bus section.
///
@@ -53,13 +58,13 @@ public class EasyCachingConstValue
///
/// The rabbitMQ Bus section.
///
- public const string RabbitMQBusSection = "easycaching:rabbitmqbus";
-
+ public const string RabbitMQBusSection = "easycaching:rabbitmqbus";
+
///
/// The kafka bus section.
///
- public const string KafkaBusSection = "easycaching:kafkabus";
-
+ public const string KafkaBusSection = "easycaching:kafkabus";
+
///
/// The zookeeper bus section.
///
@@ -110,6 +115,12 @@ public class EasyCachingConstValue
/// The default name of the LiteDB.
///
public const string DefaultLiteDBName = "DefaultLiteDB";
+
+ ///
+ /// The default name of the etcd.
+ ///
+ public const string DefaultEtcdName = "DefaultEtcd";
+
///
/// The LiteDB Bus section.
///
diff --git a/src/EasyCaching.Etcd/Configurations/EtcdCachingOptions.cs b/src/EasyCaching.Etcd/Configurations/EtcdCachingOptions.cs
new file mode 100644
index 00000000..e84b8755
--- /dev/null
+++ b/src/EasyCaching.Etcd/Configurations/EtcdCachingOptions.cs
@@ -0,0 +1,31 @@
+using EasyCaching.Core.Configurations;
+
+namespace EasyCaching.Etcd
+{
+ ///
+ /// EasyCaching options extensions of Etcd.
+ ///
+ public class EtcdCachingOptions : BaseProviderOptions
+ {
+ ///
+ /// Etcd address
+ /// cluster:like "https://localhost:23790,https://localhost:23791,https://localhost:23792"
+ ///
+ public string Address { get; set; }
+
+ ///
+ /// Etcd access UserName
+ ///
+ public string UserName { get; set; }
+
+ ///
+ /// Etcd access Pwd
+ ///
+ public string Password { get; set; }
+
+ ///
+ /// Etcd timeout with Milliseconds
+ ///
+ public long Timeout { get; set; } = 3000;
+ }
+}
\ No newline at end of file
diff --git a/src/EasyCaching.Etcd/Configurations/EtcdCachingOptionsExtensions.cs b/src/EasyCaching.Etcd/Configurations/EtcdCachingOptionsExtensions.cs
new file mode 100644
index 00000000..3a109aa3
--- /dev/null
+++ b/src/EasyCaching.Etcd/Configurations/EtcdCachingOptionsExtensions.cs
@@ -0,0 +1,65 @@
+using System;
+using EasyCaching.Core;
+using EasyCaching.Core.Configurations;
+using EasyCaching.Etcd;
+using Microsoft.Extensions.Configuration;
+// ReSharper disable CheckNamespace
+
+namespace Microsoft.Extensions.DependencyInjection;
+
+public static class EtcdCachingOptionsExtensions
+{
+ ///
+ /// Uses the Etcd provider (specify the config via hard code).
+ ///
+ /// Options.
+ /// Configure provider settings.
+ /// The name of this provider instance.
+ public static EasyCachingOptions UseEtcd(
+ this EasyCachingOptions options,
+ Action configure,
+ string name = EasyCachingConstValue.DefaultEtcdName
+ )
+ {
+ ArgumentCheck.NotNull(configure, nameof(configure));
+
+ options.RegisterExtension(new EtcdOptionsExtension(name, configure));
+ return options;
+ }
+
+ ///
+ /// Uses the Etcd provider (read config from configuration file).
+ ///
+ /// Options.
+ /// The configuration.
+ /// The name of this provider instance.
+ /// The section name in the configuration file.
+ public static EasyCachingOptions UseEtcd(
+ this EasyCachingOptions options,
+ IConfiguration configuration,
+ string name = EasyCachingConstValue.DefaultEtcdName,
+ string sectionName = EasyCachingConstValue.EtcdSection
+ )
+ {
+ var dbConfig = configuration.GetSection(sectionName);
+ var EtcdOptions = new EtcdCachingOptions();
+ dbConfig.Bind(EtcdOptions);
+
+ void Configure(EtcdCachingOptions x)
+ {
+ x.EnableLogging = EtcdOptions.EnableLogging;
+ x.MaxRdSecond = EtcdOptions.MaxRdSecond;
+ x.LockMs = EtcdOptions.LockMs;
+ x.SleepMs = EtcdOptions.SleepMs;
+ x.SerializerName = EtcdOptions.SerializerName;
+ x.CacheNulls = EtcdOptions.CacheNulls;
+ x.Address = EtcdOptions.Address;
+ x.UserName = EtcdOptions.UserName;
+ x.Password = EtcdOptions.Password;
+ x.Timeout= EtcdOptions.Timeout;
+ }
+
+ options.RegisterExtension(new EtcdOptionsExtension(name, Configure));
+ return options;
+ }
+}
\ No newline at end of file
diff --git a/src/EasyCaching.Etcd/Configurations/EtcdOptionsExtension.cs b/src/EasyCaching.Etcd/Configurations/EtcdOptionsExtension.cs
new file mode 100644
index 00000000..7eb8adc5
--- /dev/null
+++ b/src/EasyCaching.Etcd/Configurations/EtcdOptionsExtension.cs
@@ -0,0 +1,70 @@
+using System;
+using EasyCaching.Core;
+using EasyCaching.Core.Configurations;
+using EasyCaching.Core.Serialization;
+using Microsoft.Extensions.DependencyInjection;
+using Microsoft.Extensions.DependencyInjection.Extensions;
+using Microsoft.Extensions.Logging;
+using Microsoft.Extensions.Options;
+
+namespace EasyCaching.Etcd
+{
+ ///
+ /// Etcd options extension.
+ ///
+ internal sealed class EtcdOptionsExtension : IEasyCachingOptionsExtension
+ {
+ ///
+ /// The name.
+ ///
+ private readonly string _name;
+
+ ///
+ /// The configure.
+ ///
+ private readonly Action _configure;
+
+ ///
+ /// Initializes a new instance of the class.
+ ///
+ /// Name.
+ /// Configure.
+ public EtcdOptionsExtension(string name, Action configure)
+ {
+ _name = name;
+ _configure = configure;
+ }
+
+ ///
+ /// Adds the services.
+ ///
+ /// Services.
+ public void AddServices(IServiceCollection services)
+ {
+ services.AddOptions();
+
+ services.Configure(_name, _configure);
+
+ services.TryAddSingleton();
+
+ services.AddSingleton(x =>
+ {
+ var optionsMon = x.GetRequiredService>();
+ var options = optionsMon.Get(_name);
+ var factory = x.GetService();
+ var serializers = x.GetServices();
+ return new EtcdCaching(_name, options,serializers,factory);
+ });
+
+ services.AddSingleton(x =>
+ {
+ var mCache = x.GetServices();
+ var optionsMon = x.GetRequiredService>();
+ var options = optionsMon.Get(_name);
+ var factory = x.GetService();
+ var serializers = x.GetServices();
+ return new DefaultEtcdCachingProvider(_name,mCache, options, serializers, factory);
+ });
+ }
+ }
+}
diff --git a/src/EasyCaching.Etcd/DefaultEtcdCachingProvider.Async.cs b/src/EasyCaching.Etcd/DefaultEtcdCachingProvider.Async.cs
new file mode 100644
index 00000000..626ebba0
--- /dev/null
+++ b/src/EasyCaching.Etcd/DefaultEtcdCachingProvider.Async.cs
@@ -0,0 +1,398 @@
+namespace EasyCaching.Etcd
+{
+ using EasyCaching.Core;
+ using Microsoft.Extensions.Logging;
+ using System;
+ using System.Collections.Generic;
+ using System.Text;
+ using System.Threading;
+ using System.Threading.Tasks;
+
+ ///
+ /// MemoryCaching provider.
+ ///
+ public partial class DefaultEtcdCachingProvider : EasyCachingAbstractProvider
+ {
+ ///
+ /// Gets the specified cacheKey, dataRetriever and expiration async.
+ ///
+ /// The async.
+ /// Cache key.
+ /// Data retriever.
+ /// Expiration.
+ /// CancellationToken
+ /// The 1st type parameter.
+ public override async Task> BaseGetAsync(string cacheKey, Func> dataRetriever, TimeSpan expiration, CancellationToken cancellationToken = default)
+ {
+ ArgumentCheck.NotNullOrWhiteSpace(cacheKey, nameof(cacheKey));
+ ArgumentCheck.NotNegativeOrZero(expiration, nameof(expiration));
+
+ var result = await _cache.GetAsync(cacheKey);
+ if (result.HasValue)
+ {
+ if (_options.EnableLogging)
+ _logger?.LogInformation($"Cache Hit : cachekey = {cacheKey}");
+
+ CacheStats.OnHit();
+
+ return result;
+ }
+
+ CacheStats.OnMiss();
+
+ if (_options.EnableLogging)
+ _logger?.LogInformation($"Cache Missed : cachekey = {cacheKey}");
+
+ if (!await _cache.SetAsync($"{cacheKey}_Lock", "1", TimeSpan.FromMilliseconds(_options.LockMs)))
+ {
+ //wait for some ms
+ await Task.Delay(_options.SleepMs, cancellationToken);
+ return await GetAsync(cacheKey, dataRetriever, expiration);
+ }
+
+ try
+ {
+ var res = await dataRetriever();
+
+ if (res != null || _options.CacheNulls)
+ {
+ await SetAsync(cacheKey, res, expiration);
+ //remove mutex key
+ await _cache.DeleteAsync($"{cacheKey}_Lock");
+
+ return new CacheValue(res, true);
+ }
+ else
+ {
+ //remove mutex key
+ await _cache.DeleteAsync($"{cacheKey}_Lock");
+ return CacheValue.NoValue;
+ }
+ }
+ catch
+ {
+ //remove mutex key
+ await _cache.DeleteAsync($"{cacheKey}_Lock");
+ throw;
+ }
+ }
+
+ ///
+ /// Gets the specified cacheKey async.
+ ///
+ /// The async.
+ /// Cache key.
+ /// CancellationToken
+ /// The 1st type parameter.
+ public override async Task> BaseGetAsync(string cacheKey, CancellationToken cancellationToken = default)
+ {
+ ArgumentCheck.NotNullOrWhiteSpace(cacheKey, nameof(cacheKey));
+
+ var result = await _cache.GetAsync(cacheKey);
+
+ if (result.HasValue)
+ {
+ if (_options.EnableLogging)
+ _logger?.LogInformation($"Cache Hit : cachekey = {cacheKey}");
+
+ CacheStats.OnHit();
+
+ return result;
+ }
+ else
+ {
+ if (_options.EnableLogging)
+ _logger?.LogInformation($"Cache Missed : cachekey = {cacheKey}");
+
+ CacheStats.OnMiss();
+
+ return CacheValue.NoValue;
+ }
+ }
+
+ ///
+ /// Gets the count.
+ ///
+ /// The count.
+ /// Prefix.
+ /// CancellationToken
+ public override async Task BaseGetCountAsync(string prefix = "", CancellationToken cancellationToken = default)
+ {
+ var dicData = await _cache.GetAllAsync(prefix);
+ return dicData != null ? dicData.Count : 0;
+ }
+
+ ///
+ /// Gets the specified cacheKey async.
+ ///
+ /// The async.
+ /// Cache key.
+ /// Object Type.
+ /// CancellationToken
+ public override async Task