Skip to content

Commit 43bf626

Browse files
feature/secure-sensitive-data (#47)
* Added ProtectedData to provide in-memory encryption for sensitive data such as private keys and secrets. * Added xmldoc.
1 parent 156cfbd commit 43bf626

File tree

5 files changed

+113
-11
lines changed

5 files changed

+113
-11
lines changed

OnixLabs.Security.Cryptography/PrivateKey.cs

+14-3
Original file line numberDiff line numberDiff line change
@@ -19,11 +19,22 @@ namespace OnixLabs.Security.Cryptography;
1919
/// <summary>
2020
/// Represents a cryptographic private key.
2121
/// </summary>
22-
/// <param name="keyData">The underlying key data of the cryptographic private key.</param>
23-
public abstract partial class PrivateKey(ReadOnlySpan<byte> keyData) : ICryptoPrimitive<PrivateKey>
22+
public abstract partial class PrivateKey : ICryptoPrimitive<PrivateKey>
2423
{
24+
private readonly ProtectedData protectedData = new();
25+
private readonly byte[] encryptedKeyData;
26+
27+
/// <summary>
28+
/// Initializes a new instance of the <see cref="PrivateKey"/> class.
29+
/// </summary>
30+
/// <param name="keyData">The underlying key data of the cryptographic private key.</param>
31+
protected PrivateKey(ReadOnlySpan<byte> keyData)
32+
{
33+
encryptedKeyData = protectedData.Encrypt(keyData.ToArray());
34+
}
35+
2536
/// <summary>
2637
/// Gets the cryptographic private key data.
2738
/// </summary>
28-
protected byte[] KeyData { get; } = keyData.ToArray();
39+
protected byte[] KeyData => protectedData.Decrypt(encryptedKeyData);
2940
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,80 @@
1+
// Copyright 2020 ONIXLabs
2+
//
3+
// Licensed under the Apache License, Version 2.0 (the "License");
4+
// you may not use this file except in compliance with the License.
5+
// You may obtain a copy of the License at
6+
//
7+
// http://www.apache.org/licenses/LICENSE-2.0
8+
//
9+
// Unless required by applicable law or agreed to in writing, software
10+
// distributed under the License is distributed on an "AS IS" BASIS,
11+
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12+
// See the License for the specific language governing permissions and
13+
// limitations under the License.
14+
15+
using System.IO;
16+
using System.Security.Cryptography;
17+
using Aes = System.Security.Cryptography.Aes;
18+
19+
namespace OnixLabs.Security.Cryptography;
20+
21+
/// <summary>
22+
/// Represents an in-memory data protection mechanism for sensitive, long-lived cryptographic data.
23+
/// </summary>
24+
internal sealed class ProtectedData
25+
{
26+
private readonly byte[] key = Salt.CreateNonZero(32).ToByteArray();
27+
private readonly byte[] iv = Salt.CreateNonZero(16).ToByteArray();
28+
29+
/// <summary>
30+
/// Encrypted the specified data.
31+
/// </summary>
32+
/// <param name="data">The data to encrypt.</param>
33+
/// <returns>Returns the encrypted data.</returns>
34+
public byte[] Encrypt(byte[] data)
35+
{
36+
Require(data.Length > 0, "Data must not be empty.", nameof(data));
37+
38+
using Aes algorithm = Aes.Create();
39+
40+
algorithm.Key = key;
41+
algorithm.IV = iv;
42+
algorithm.Padding = PaddingMode.PKCS7;
43+
44+
ICryptoTransform transform = algorithm.CreateEncryptor(algorithm.Key, algorithm.IV);
45+
46+
using MemoryStream memoryStream = new();
47+
using CryptoStream cryptoStream = new(memoryStream, transform, CryptoStreamMode.Write);
48+
49+
cryptoStream.Write(data, 0, data.Length);
50+
cryptoStream.FlushFinalBlock();
51+
52+
return memoryStream.ToArray();
53+
}
54+
55+
/// <summary>
56+
/// Decrypts the specified data.
57+
/// </summary>
58+
/// <param name="data">The data to decrypt.</param>
59+
/// <returns>Returns the decrypted data.</returns>
60+
public byte[] Decrypt(byte[] data)
61+
{
62+
Require(data.Length > 0, "Data must not be empty.", nameof(data));
63+
64+
using Aes algorithm = Aes.Create();
65+
66+
algorithm.Key = key;
67+
algorithm.IV = iv;
68+
algorithm.Padding = PaddingMode.PKCS7;
69+
70+
ICryptoTransform transform = algorithm.CreateDecryptor(algorithm.Key, algorithm.IV);
71+
72+
using MemoryStream memoryStream = new(data);
73+
using CryptoStream cryptoStream = new(memoryStream, transform, CryptoStreamMode.Read);
74+
using MemoryStream resultStream = new();
75+
76+
cryptoStream.CopyTo(resultStream);
77+
78+
return resultStream.ToArray();
79+
}
80+
}

OnixLabs.Security.Cryptography/Secret.Equatable.cs

+2-2
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,7 @@ public readonly partial struct Secret
2424
/// </summary>
2525
/// <param name="other">An object to compare with the current object.</param>
2626
/// <returns>Returns <see langword="true"/> if the current object is equal to the other parameter; otherwise, <see langword="false"/>.</returns>
27-
public bool Equals(Secret other) => value.SequenceEqual(other.value);
27+
public bool Equals(Secret other) => hash == other.hash;
2828

2929
/// <summary>
3030
/// Checks for equality between the current instance and another object.
@@ -37,7 +37,7 @@ public readonly partial struct Secret
3737
/// Serves as a hash code function for the current instance.
3838
/// </summary>
3939
/// <returns>Returns a hash code for the current instance.</returns>
40-
public override int GetHashCode() => value.GetContentHashCode();
40+
public override int GetHashCode() => hash.GetHashCode();
4141

4242
/// <summary>
4343
/// Performs an equality comparison between two object instances.

OnixLabs.Security.Cryptography/Secret.To.cs

+2-3
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,6 @@
1313
// limitations under the License.
1414

1515
using System;
16-
using OnixLabs.Core;
1716

1817
namespace OnixLabs.Security.Cryptography;
1918

@@ -23,11 +22,11 @@ public readonly partial struct Secret
2322
/// Gets the underlying <see cref="T:Byte[]"/> representation of the current <see cref="DigitalSignature"/> instance.
2423
/// </summary>
2524
/// <returns>Return the underlying <see cref="T:Byte[]"/> representation of the current <see cref="DigitalSignature"/> instance.</returns>
26-
public byte[] ToByteArray() => value.Copy();
25+
public byte[] ToByteArray() => protectedData.Decrypt(encryptedKeyData);
2726

2827
/// <summary>
2928
/// Returns a <see cref="string"/> that represents the current object.
3029
/// </summary>
3130
/// <returns>Returns a <see cref="string"/> that represents the current object.</returns>
32-
public override string ToString() => Convert.ToHexString(value).ToLower();
31+
public override string ToString() => Convert.ToHexString(protectedData.Decrypt(encryptedKeyData)).ToLower();
3332
}

OnixLabs.Security.Cryptography/Secret.cs

+15-3
Original file line numberDiff line numberDiff line change
@@ -13,14 +13,26 @@
1313
// limitations under the License.
1414

1515
using System;
16+
using System.Security.Cryptography;
1617

1718
namespace OnixLabs.Security.Cryptography;
1819

1920
/// <summary>
2021
/// Represents a cryptographic secret.
2122
/// </summary>
22-
/// <param name="value">The underlying value of the cryptographic secret.</param>
23-
public readonly partial struct Secret(ReadOnlySpan<byte> value) : ICryptoPrimitive<Secret>
23+
public readonly partial struct Secret : ICryptoPrimitive<Secret>
2424
{
25-
private readonly byte[] value = value.ToArray();
25+
private readonly ProtectedData protectedData = new();
26+
private readonly byte[] encryptedKeyData;
27+
private readonly Hash hash;
28+
29+
/// <summary>
30+
/// Initializes a new instance of the <see cref="Secret"/> struct.
31+
/// </summary>
32+
/// <param name="value">The underlying value of the cryptographic secret.</param>
33+
public Secret(ReadOnlySpan<byte> value)
34+
{
35+
encryptedKeyData = protectedData.Encrypt(value.ToArray());
36+
hash = Hash.Compute(SHA256.Create(), value.ToArray());
37+
}
2638
}

0 commit comments

Comments
 (0)