Files
duplicati/Duplicati/Library/DynamicLoader/CompressionLoader.cs
Kenneth Skovhede 8e25c6e2ab Reduced dynamic scanning with static linking.
Cleaned up some `proj` files.
Added caching to options.
2024-04-26 14:32:41 +02:00

151 lines
6.8 KiB
C#

// Copyright (C) 2024, 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 System.Linq;
using Duplicati.Library.Compression;
using Duplicati.Library.Interface;
namespace Duplicati.Library.DynamicLoader
{
/// <summary>
/// Loads all compression modules dynamically and exposes a list of those
/// </summary>
public class CompressionLoader
{
/// <summary>
/// Implementation overrides specific to compression
/// </summary>
private class CompressionLoaderSub : DynamicLoader<ICompression>
{
/// <summary>
/// Returns the filename extension, which is also the key
/// </summary>
/// <param name="item">The item to load the key for</param>
/// <returns>The file extension used by the module</returns>
protected override string GetInterfaceKey(ICompression item)
{
return item.FilenameExtension;
}
/// <summary>
/// Returns the subfolders searched for compression modules
/// </summary>
protected override string[] Subfolders => ["compression"];
/// <summary>
/// The built-in modules
/// </summary>
protected override IEnumerable<ICompression> BuiltInModules => CompressionModules.BuiltInCompressionModules;
/// <summary>
/// Instanciates a specific compression module, given the file extension and options
/// </summary>
/// <param name="fileExtension">The file extension to create the instance for</param>
/// <param name="stream">The stream of the file used to compress/decompress contents</param>
/// <param name="mode">The mode for compression/decompression</param>
/// <param name="options">The options to pass to the instance constructor</param>
/// <returns>The instanciated encryption module or null if the file extension is not supported</returns>
public ICompression GetModule(string fileExtension, Stream stream, ArchiveMode mode, Dictionary<string, string> options)
{
if (string.IsNullOrEmpty(fileExtension))
throw new ArgumentNullException(nameof(fileExtension));
LoadInterfaces();
lock (m_lock)
{
if (m_interfaces.ContainsKey(fileExtension))
return (ICompression)Activator.CreateInstance(m_interfaces[fileExtension].GetType(), stream, mode, options);
else
return null;
}
}
/// <summary>
/// Gets the supported commands for a certain key
/// </summary>
/// <param name="key">The key to find commands for</param>
/// <returns>The supported commands or null if the key was not found</returns>
public IReadOnlyList<ICommandLineArgument> GetSupportedCommands(string key)
{
if (string.IsNullOrEmpty(key))
throw new ArgumentNullException(nameof(key));
LoadInterfaces();
lock (m_lock)
{
ICompression b;
if (m_interfaces.TryGetValue(key, out b) && b != null)
return GetSupportedCommandsCached(b).ToList();
else
return null;
}
}
}
/// <summary>
/// The static instance used to access compression module information
/// </summary>
private static readonly CompressionLoaderSub _compressionLoader = new CompressionLoaderSub();
#region Public static API
/// <summary>
/// Gets a list of loaded compression modules, the instances can be used to extract interface information, not used to interact with the module.
/// </summary>
public static ICompression[] Modules { get { return _compressionLoader.Interfaces; } }
/// <summary>
/// Gets a list of keys supported
/// </summary>
public static string[] Keys { get { return _compressionLoader.Keys; } }
/// <summary>
/// Gets the supported commands for a given compression module
/// </summary>
/// <param name="key">The compression module to find the commands for</param>
/// <returns>The supported commands or null if the key is not supported</returns>
public static IReadOnlyList<ICommandLineArgument> GetSupportedCommands(string key)
{
return _compressionLoader.GetSupportedCommands(key);
}
/// <summary>
/// Instanciates a specific compression module, given the file extension and options
/// </summary>
/// <param name="fileextension">The file extension to create the instance for</param>
/// <param name="stream">The stream of the file used to compress/decompress contents</param>
/// <param name="mode">The mode for compression/decompression</param>
/// <param name="options">The options to pass to the instance constructor</param>
/// <returns>The instanciated compression module or null if the file extension is not supported</returns>
public static ICompression GetModule(string fileextension, Stream stream, ArchiveMode mode, Dictionary<string, string> options)
{
return _compressionLoader.GetModule(fileextension, stream, mode, options);
}
#endregion
}
}