亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > .NET > 正文

ASP.NET Core 數據保護(Data Protection 集群場景)下篇

2024-07-10 13:31:22
字體:
來源:轉載
供稿:網友

前言 

接【中篇】 ,在有一些場景下,我們需要對 ASP.NET Core 的加密方法進行擴展,來適應我們的需求,這個時候就需要使用到了一些 Core 提供的高級的功能。 

本文還列舉了在集群場景下,有時候我們需要實現自己的一些方法來對Data Protection進行分布式配置。 

加密擴展 

IAuthenticatedEncryptor IAuthenticatedEncryptorDescriptor 

IAuthenticatedEncryptor是 Data Protection 在構建其密碼加密系統中的一個基礎的接口。
 一般情況下一個key 對應一個IAuthenticatedEncryptor,IAuthenticatedEncryptor封裝了加密操作中需要使用到的秘鑰材料和必要的加密算法信息等。 

下面是IAuthenticatedEncryptor接口提供的兩個 api方法:
Decrypt(ArraySegment<byte> ciphertext, ArraySegment<byte> additionalAuthenticatedData) : byte[]
Encrypt(ArraySegment<byte> plaintext, ArraySegment<byte> additionalAuthenticatedData) : byte[]

其中接口中的參數additionalAuthenticatedData表示在構建加密的時候提供的一些附屬信息。 

IAuthenticatedEncryptorDescriptor接口提供了一個創建包含類型信息IAuthenticatedEncryptor實例方法。

CreateEncryptorInstance() : IAuthenticatedEncryptor
ExportToXml() : XmlSerializedDescriptorInfo

密鑰管理擴展

在密鑰系統管理中,提供了一個基礎的接口IKey,它包含以下屬性: 

Activation
creation
expiration dates
Revocation status
Key identifier (a GUID)

IKey還提供了一個創建IAuthenticatedEncryptor實例的方法CreateEncryptorInstance。 

IKeyManager接口提供了一系列用來操作Key的方法,包括存儲,檢索操作等。他提供的高級操作有:

 •創建一個Key 并且持久存儲
 •從存儲庫中獲取所有的 Key
 •撤銷保存到存儲中的一個或多個鍵

XmlKeyManager
通常情況下,開發人員不需要去實現IKeyManager來自定義一個 KeyManager。我們可以使用系統默認提供的XmlKeyManager類。 

XMLKeyManager是一個具體實現IKeyManager的類,它提供了一些非常有用的方法。

 public sealed class XmlKeyManager : IKeyManager, IInternalXmlKeyManager{ public XmlKeyManager(IXmlRepository repository, IAuthenticatedEncryptorConfiguration configuration, IServiceProvider services); public IKey CreateNewKey(DateTimeOffset activationDate, DateTimeOffset expirationDate); public IReadOnlyCollection<IKey> GetAllKeys(); public CancellationToken GetCacheExpirationToken(); public void RevokeAllKeys(DateTimeOffset revocationDate, string reason = null); public void RevokeKey(Guid keyId, string reason = null);} 

•IAuthenticatedEncryptorConfiguration 主要是規定新 Key 使用的算法。
•IXmlRepository 主要控制 Key 在哪里持久化存儲。

IXmlRepository

IXmlRepository接口主要提供了持久化以及檢索XML的方法,它只要提供了兩個API:
 •GetAllElements() : IReadOnlyCollection
 •StoreElement(XElement element, string friendlyName) 

我們可以通過實現IXmlRepository接口的StoreElement方法來定義data protection xml的存儲位置。 

GetAllElements來檢索所有存在的加密的xml文件。 

接口部分寫到這里吧,因為這一篇我想把重點放到下面,更多接口的介紹大家還是去官方文檔看吧~ 

集群場景

上面的API估計看著有點枯燥,那我們就來看看我們需要在集群場景下借助于Data Protection來做點什么吧。 

就像我在【上篇】總結中末尾提到的,在做分布式集群的時候,Data Protection的一些機制我們需要知道,因為如果不了解這些可能會給你的部署帶來一些麻煩,下面我們就來看看吧。 

在做集群的時,我們必須知道并且明白關于 ASP.NET Core Data Protection 的三個東西:

1、程序識別者

“Application discriminator”,它是用來標識應用程序的唯一性。
 為什么需要這個東西呢?因為在集群環境中,如果不被具體的硬件機器環境所限制,就要排除運行機器的一些差異,就需要抽象出來一些特定的標識,來標識應用程序本身并且使用該標識來區分不同的應用程序。這個時候,我們可以指定ApplicationDiscriminator。 

在services.AddDataProtection(DataProtectionOptions option)的時候,ApplicationDiscriminator可以作為參數傳遞,來看一下代碼:

 public void ConfigureServices(IServiceCollection services) { services.AddDataProtection(); services.AddDataProtection(DataProtectionOptions option);}//===========擴展方法如下:public static class DataProtectionServiceCollectionExtensions{ public static IDataProtectionBuilder AddDataProtection(this IServiceCollection services);  //具有可傳遞參數的重載,在集群環境中需要使用此項配置 public static IDataProtectionBuilder AddDataProtection(this IServiceCollection services, Action<DataProtectionOptions> setupAction);}// DataProtectionOptions 屬性:public class DataProtectionOptions{ public string ApplicationDiscriminator { get; set; }} 

可以看到這個擴展返回的是一個IDataProtectionBuilder,在IDataProtectionBuilder還有一個擴展方法叫 SetApplicationName ,這個擴展方法在內部還是修改的ApplicationDiscriminator的值。也就說以下寫法是等價的:

services.AddDataProtection(x => x.ApplicationDiscriminator = "my_app_sample_identity");

services.AddDataProtection().SetApplicationName("my_app_sample_identity"); 

也就是說集群環境下同一應用程序他們需要設定為相同的值(ApplicationName or ApplicationDiscriminator)。 

2、主加密鍵

“Master encryption key”,主要是用來加密解密的,包括一客戶端服務器在請求的過程中的一些會話數據,狀態等。有幾個可選項可以配置,比如使用證書或者是windows DPAPI或者注冊表等。如果是非windows平臺,注冊表和Windows DPAPI就不能用了。

 public void ConfigureServices(IServiceCollection services) { services.AddDataProtection()  //windows dpaip 作為主加密鍵 .ProtectKeysWithDpapi()  //如果是 windows 8+ 或者windows server2012+ 可以使用此選項(基于Windows DPAPI-NG) .ProtectKeysWithDpapiNG("SID={current account SID}", DpapiNGProtectionDescriptorFlags.None)  //如果是 windows 8+ 或者windows server2012+ 可以使用此選項(基于證書) .ProtectKeysWithDpapiNG("CERTIFICATE=HashId:3BCE558E2AD3E0E34A7743EAB5AEA2A9BD2575A0", DpapiNGProtectionDescriptorFlags.None)  //使用證書作為主加密鍵,目前只有widnows支持,linux還不支持。 .ProtectKeysWithCertificate();} 

如果在集群環境中,他們需要具有配置相同的主加密鍵。 

3、加密后存儲位置

在【上篇】的時候說過,默認情況下Data Protection會生成 xml 文件用來存儲session或者是狀態的密鑰文件。這些文件用來加密或者解密session等狀態數據。 

就是上篇中說的那個私鑰存儲位置:

1、如果程序寄宿在 Microsoft Azure下,存儲在“%HOME%/ASP.NET/DataProtection-Keys” 文件夾。
 2、如果程序寄宿在IIS下,它被保存在HKLM注冊表的ACLed特殊注冊表鍵,并且只有工作進程可以訪問,它使用windows的DPAPI加密。
 3、如果當前用戶可用,即win10或者win7中,它存儲在“%LOCALAPPDATA%/ASP.NET/DataProtection-Keys”文件夾,同樣使用的windows的DPAPI加密。
 4、如果這些都不符合,那么也就是私鑰是沒有被持久化的,也就是說當進程關閉的時候,生成的私鑰就丟失了。 

集群環境下:
 最簡單的方式是通過文件共享、DPAPI或者注冊表,也就是說把加密過后的xml文件都存儲在相同的地方。為什么說最簡單,因為系統已經給封裝好了,不需要寫多余的代碼了,但是要保證文件共享相關的端口是開放的。如下:

 public void ConfigureServices(IServiceCollection services) { services.AddDataProtection() //windows、Linux、macOS 下可以使用此種方式 保存到文件系統 .PersistKeysToFileSystem(new System.IO.DirectoryInfo("C://share_keys//")) //windows 下可以使用此種方式 保存到注冊表 .PersistKeysToRegistry(Microsoft.Win32.RegistryKey.FromHandle(null)) } 

你也可以自己擴展方法來自己定義一些存儲,比如使用數據庫或者Redis等。 

不過通常情況下,如果在linux上部署的話,都是需要擴展的。下面來看一下我們想要用redis存儲,該怎么做呢? 

如何擴展加密鍵集合的存儲位置?

首先,定義個針對IXmlRepository接口的 redis 實現類RedisXmlRepository.cs:

 public class RedisXmlRepository : IXmlRepository, IDisposable{ public static readonly string RedisHashKey = "DataProtectionXmlRepository";  private IConnectionMultiplexer _connection;  private bool _disposed = false;  public RedisXmlRepository(string connectionString, ILogger<RedisXmlRepository> logger)  : this(ConnectionMultiplexer.Connect(connectionString), logger) { }  public RedisXmlRepository(IConnectionMultiplexer connection, ILogger<RedisXmlRepository> logger) {  if (connection == null)  {   throw new ArgumentNullException(nameof(connection));  }   if (logger == null)  {   throw new ArgumentNullException(nameof(logger));  }   this._connection = connection;  this.Logger = logger;   var configuration = Regex.Replace(this._connection.Configuration, @"password/s*=/s*[^,]*", "password=****", RegexOptions.IgnoreCase);  this.Logger.LogDebug("Storing data protection keys in Redis: {RedisConfiguration}", configuration); }  public ILogger<RedisXmlRepository> Logger { get; private set; }  public void Dispose() {  this.Dispose(true); } public IReadOnlyCollection<XElement> GetAllElements() {  var database = this._connection.GetDatabase();  var hash = database.HashGetAll(RedisHashKey);  var elements = new List<XElement>();   if (hash == null || hash.Length == 0)  {   return elements.AsReadOnly();  }   foreach (var item in hash.ToStringDictionary())  {   elements.Add(XElement.Parse(item.Value));  }   this.Logger.LogDebug("Read {XmlElementCount} XML elements from Redis.", elements.Count);  return elements.AsReadOnly(); }  public void StoreElement(XElement element, string friendlyName) {  if (element == null)  {   throw new ArgumentNullException(nameof(element));  }   if (string.IsNullOrEmpty(friendlyName))  {   friendlyName = Guid.NewGuid().ToString();  }   this.Logger.LogDebug("Storing XML element with friendly name {XmlElementFriendlyName}.", friendlyName);   this._connection.GetDatabase().HashSet(RedisHashKey, friendlyName, element.ToString()); } protected virtual void Dispose(bool disposing) {  if (!this._disposed)  {   if (disposing)   {    if (this._connection != null)    {     this._connection.Close();     this._connection.Dispose();    }   }    this._connection = null;   this._disposed = true;  } }} 

然后任意一個擴展類中先定義一個擴展方法:

 public static IDataProtectionBuilder PersistKeysToRedis(this IDataProtectionBuilder builder, string redisConnectionString){ if (builder == null) {  throw new ArgumentNullException(nameof(builder)); } if (redisConnectionString == null) {  throw new ArgumentNullException(nameof(redisConnectionString)); } if (redisConnectionString.Length == 0) {  throw new ArgumentException("Redis connection string may not be empty.", nameof(redisConnectionString)); }  //因為在services.AddDataProtection()的時候,已經注入了IXmlRepository,所以應該先移除掉 //此處應該封裝成為一個方法來調用,為了讀者好理解,我就直接寫了 for (int i = builder.Services.Count - 1; i >= 0; i--) {  if (builder.Services[i]?.ServiceType == descriptor.ServiceType)  {   builder.Services.RemoveAt(i);  } }  var descriptor = ServiceDescriptor.Singleton<IXmlRepository>(services => new RedisXmlRepository(redisConnectionString, services.GetRequiredService<ILogger<RedisXmlRepository>>()))    builder.Services.Add(descriptor);    return builder.Use();} 

最終Services中關于DataProtection是這樣的:

 public void ConfigureServices(IServiceCollection services) { services.AddDataProtection()  // ================以下是唯一標識==============  //設置應用程序唯一標識 .SetApplicationName("my_app_sample_identity");   // =============以下是主加密鍵===============  //windows dpaip 作為主加密鍵 .ProtectKeysWithDpapi()  //如果是 windows 8+ 或者windows server2012+ 可以使用此選項(基于Windows DPAPI-NG) .ProtectKeysWithDpapiNG("SID={current account SID}", DpapiNGProtectionDescriptorFlags.None)  //如果是 windows 8+ 或者windows server2012+ 可以使用此選項(基于證書) .ProtectKeysWithDpapiNG("CERTIFICATE=HashId:3BCE558E2AD3E0E34A7743EAB5AEA2A9BD2575A0", DpapiNGProtectionDescriptorFlags.None)  //使用證書作為主加密鍵,目前只有widnows支持,linux還不支持。 .ProtectKeysWithCertificate();   // ==============以下是存儲位置=================  //windows、Linux、macOS 下可以使用此種方式 保存到文件系統 .PersistKeysToFileSystem(new System.IO.DirectoryInfo("C://share_keys//"))  //windows 下可以使用此種方式 保存到注冊表 .PersistKeysToRegistry(Microsoft.Win32.RegistryKey.FromHandle(null))    // 存儲到redis .PersistKeysToRedis(Configuration.Section["RedisConnection"])} 

在上面的配置中,我把所有可以使用的配置都列出來了哦,實際項目中應該視實際情況選擇。 

總結

關于ASP.NET Core Data Protection 系列終于寫完了,其實這這部分花了蠻多時間的,對于Data Protection來說我也是一個循循漸進的學習過程,希望能幫助到一些人。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持VeVb武林網。


注:相關教程知識閱讀請移步到ASP.NET教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
欧美在线激情视频| 欧美日韩亚洲精品内裤| 亚洲91精品在线| 黄色成人av网| 美女黄色丝袜一区| 久久亚洲精品中文字幕冲田杏梨| 亚洲国产成人精品久久久国产成人一区| 国产精品一区二区三区久久久| 欧美专区在线视频| 欧美成人精品在线观看| 精品亚洲夜色av98在线观看| 亚洲a中文字幕| 亚洲aⅴ男人的天堂在线观看| 国产精品爽黄69天堂a| 欧美野外猛男的大粗鳮| 亚洲国产精品嫩草影院久久| 亚洲精品国产精品国自产观看浪潮| 国产精品高清网站| 一区二区成人av| 日韩在线观看你懂的| 国产xxx69麻豆国语对白| 92福利视频午夜1000合集在线观看| 在线视频免费一区二区| 欧美精品电影在线| 日韩在线中文字幕| 中文日韩在线视频| 一本一本久久a久久精品牛牛影视| 国产亚洲aⅴaaaaaa毛片| 久久香蕉国产线看观看av| 亚洲韩国日本中文字幕| 亚州欧美日韩中文视频| 亚洲热线99精品视频| 在线观看免费高清视频97| 久久中文字幕在线| 久久亚洲精品一区二区| 日韩成人av在线| 精品国产一区久久久| www亚洲欧美| 91麻豆国产语对白在线观看| 亚洲激情小视频| 亚洲国产精品人人爽夜夜爽| 久久精品国产v日韩v亚洲| 欧美日韩免费一区| 久久国产精品亚洲| 夜夜嗨av一区二区三区四区| 国产精品一区av| 91在线观看免费观看| 精品国产鲁一鲁一区二区张丽| 51午夜精品视频| 国产剧情久久久久久| 久久精品国产一区二区三区| 91精品国产91久久久久久不卡| 亚洲精品黄网在线观看| 日本aⅴ大伊香蕉精品视频| 91免费在线视频| 成人有码视频在线播放| 欧美大片欧美激情性色a∨久久| 在线观看欧美日韩| 成人午夜激情免费视频| 欧美性猛交xxxxx水多| 麻豆乱码国产一区二区三区| 日本aⅴ大伊香蕉精品视频| 国产亚洲精品91在线| 国产精品白嫩初高中害羞小美女| 国产日韩欧美日韩大片| 欧美小视频在线观看| 欧美床上激情在线观看| 国产精品久久久久久久9999| 4438全国亚洲精品在线观看视频| 亚洲国产婷婷香蕉久久久久久| 伊人伊成久久人综合网站| 色妞色视频一区二区三区四区| 国产日韩欧美一二三区| 久久男人的天堂| 亚洲成人激情在线| 亚洲在线观看视频网站| 日韩精品免费视频| 日韩中文字幕av| 亚洲自拍偷拍福利| 日韩大片在线观看视频| 久久久成人精品视频| 91色琪琪电影亚洲精品久久| 国产精品久久久久9999| 美日韩丰满少妇在线观看| 91精品国产91久久久久久吃药| 九九精品视频在线| 国产精品福利网站| 国产精品自拍偷拍| 57pao成人国产永久免费| 久久久久久久久亚洲| 久久久这里只有精品视频| 久久大大胆人体| 亚洲丁香婷深爱综合| 久久99久久99精品免观看粉嫩| 亚洲另类欧美自拍| 亚洲一区二区久久| 国产mv免费观看入口亚洲| 亚洲美女激情视频| 欧美日韩一区二区免费在线观看| 中文字幕亚洲国产| 精品久久久国产精品999| 日韩人体视频一二区| 在线a欧美视频| 亚洲一二在线观看| 91性高湖久久久久久久久_久久99| 2019精品视频| 欧美激情精品久久久久久免费印度| 亚洲已满18点击进入在线看片| 成人信息集中地欧美| 97精品国产91久久久久久| 欧美大片欧美激情性色a∨久久| 欧美xxxx18性欧美| 国产精品专区一| 日本19禁啪啪免费观看www| 国产精品高潮呻吟视频| 92裸体在线视频网站| 好吊成人免视频| 欧美日韩国产一区中文午夜| 国产精品露脸av在线| 欧美人与物videos| 亚洲成人av中文字幕| 最新国产成人av网站网址麻豆| 在线成人中文字幕| 亚洲日本中文字幕| 国产午夜精品理论片a级探花| 福利一区视频在线观看| 亚洲图片制服诱惑| www.亚洲人.com| 国产日韩亚洲欧美| 欧美激情一级精品国产| 欧美福利视频在线| 国产成人精品日本亚洲专区61| 欧美视频在线看| 91久久久久久久久久| 国产欧美日韩视频| 亚洲色图色老头| 97精品国产97久久久久久| 欧美日韩国产色| 久久中文字幕视频| 日韩欧美在线视频| 中文字幕日韩在线观看| 亚洲精品成人久久电影| 欧美理论电影在线观看| 久久精品成人一区二区三区| 美女啪啪无遮挡免费久久网站| 永久555www成人免费| 欧美视频在线观看免费网址| 国产一区欧美二区三区| 久久亚洲精品小早川怜子66| 亚洲天堂开心观看| 日本午夜在线亚洲.国产| 91高清视频在线免费观看| 中文字幕av一区二区三区谷原希美| 日韩av在线天堂网| 法国裸体一区二区| 国产成人一区二区在线| 亚洲三级 欧美三级| 国语自产精品视频在线看抢先版图片| 欧美在线激情视频| 日韩久久精品成人| 久久视频国产精品免费视频在线| 91久久国产婷婷一区二区| 九九热这里只有精品6| 国产视频999|