duplicati/Duplicati/Library/Encryption/AESEncryption.cs
Kenneth Skovhede 9464caf622
Feature/update license 2025 (#5851)
* Fixed some minor whitespace issues

* Updated all copyright to 2025
2025-01-07 09:40:39 +01:00

164 lines
6.8 KiB
C#

// Copyright (C) 2025, The Duplicati Team
// https://duplicati.com, hello@duplicati.com
//
// Permission is hereby granted, free of charge, to any person obtaining a
// copy of this software and associated documentation files (the "Software"),
// to deal in the Software without restriction, including without limitation
// the rights to use, copy, modify, merge, publish, distribute, sublicense,
// and/or sell copies of the Software, and to permit persons to whom the
// Software is furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
// DEALINGS IN THE SOFTWARE.
using System;
using System.Collections.Generic;
using System.IO;
using Duplicati.Library.Interface;
namespace Duplicati.Library.Encryption
{
/// <summary>
/// Implements AES encryption
/// </summary>
public class AESEncryption : EncryptionBase
{
/// <summary>
/// The key used to encrypt the data
/// </summary>
private readonly string m_key;
/// <summary>
/// The cached value for size overhead
/// </summary>
private static long m_cachedsizeoverhead = -1;
/// <summary>
/// Cached set of options for minimal header
/// </summary>
private static readonly SharpAESCrypt.EncryptionOptions m_minimalHeaderOptions = new(InsertCreatedByIdentifier: false, InsertTimeStamp: false, InsertPlaceholder: false);
/// <summary>
/// Cached set of options for decryption
/// </summary>
private static readonly SharpAESCrypt.DecryptionOptions m_decryptionOptions = new(IgnorePaddingBytes: Environment.GetEnvironmentVariable("AES_IGNORE_PADDING_BYTES") == "1");
/// <summary>
/// Options to use for encryption
/// </summary>
private readonly SharpAESCrypt.EncryptionOptions m_encryptionOptions;
/// <summary>
/// Default constructor, used to read file extension and supported commands
/// </summary>
public AESEncryption()
{
}
/// <summary>
/// Constructs a new AES encryption/decyption instance
/// </summary>
/// <param name="passphrase">The passphrase to use</param>
/// <param name="minimalheader">Flag controlling if the encryption is done with a minimal header</param>
public AESEncryption(string passphrase, bool minimalheader)
{
if (string.IsNullOrEmpty(passphrase))
throw new ArgumentException(Strings.AESEncryption.EmptyKeyError, nameof(passphrase));
m_key = passphrase;
m_encryptionOptions = minimalheader ? m_minimalHeaderOptions : default;
}
/// <summary>
/// Constructs a new AES encryption/decyption instance
/// </summary>
public AESEncryption(string passphrase, Dictionary<string, string> options)
: this(passphrase, false)
{
}
#region IEncryption Members
/// <summary>
/// The extension that the encryption implementation adds to the filename
/// </summary>
/// <value>The filename extension.</value>
public override string FilenameExtension { get { return "aes"; } }
/// <summary>
/// A localized description of the encryption module
/// </summary>
/// <value>The description.</value>
public override string Description { get { return string.Format(Strings.AESEncryption.Description_v2); } }
/// <summary>
/// A localized string describing the encryption module with a friendly name
/// </summary>
/// <value>The display name.</value>
public override string DisplayName { get { return Strings.AESEncryption.DisplayName; } }
/// <summary>
/// Dispose the specified disposing.
/// </summary>
/// <param name="disposing">If set to <c>true</c> disposing.</param>
protected override void Dispose(bool disposing) { }
/// <summary>
/// Returns the size in bytes of the overhead that will be added to a file of the given size when encrypted
/// </summary>
/// <param name="filesize">The size of the file to encrypt</param>
/// <returns>The size of the overhead in bytes</returns>
public override long SizeOverhead(long filesize)
{
if (m_cachedsizeoverhead != -1)
return m_cachedsizeoverhead;
//If we use 1, we trigger the blocksize.
//As the AES algorithm does not alter the size,
// the results are the same as for the real size,
// but a single byte encryption is much faster.
return m_cachedsizeoverhead = base.SizeOverhead(1);
}
/// <summary>
/// Encrypts the stream
/// </summary>
/// <param name="input">The target stream</param>
/// <returns>An encrypted stream that can be written to</returns>
public override Stream Encrypt(Stream input)
=> new SharpAESCrypt.EncryptingStream(m_key, input, m_encryptionOptions);
/// <summary>
/// Decrypts the stream to the output stream
/// </summary>
/// <param name="input">The encrypted stream</param>
/// <returns>The unencrypted stream</returns>
public override Stream Decrypt(Stream input)
=> new SharpAESCrypt.DecryptingStream(m_key, input, m_decryptionOptions);
/// <summary>
/// Gets a list of supported commandline arguments
/// </summary>
/// <value>The supported commands.</value>
public override IList<ICommandLineArgument> SupportedCommands
=> new List<ICommandLineArgument>([
new CommandLineArgument(
"aes-set-threadlevel",
CommandLineArgument.ArgumentType.String,
Strings.AESEncryption.AessetthreadlevelShort,
Strings.AESEncryption.AessetthreadlevelLong,
"",
null,
null,
Strings.AESEncryption.AessetthreadlevelDeprecated
),
]);
#endregion
}
}