// 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
{
///
/// Loads all compression modules dynamically and exposes a list of those
///
public class CompressionLoader
{
///
/// Implementation overrides specific to compression
///
private class CompressionLoaderSub : DynamicLoader
{
///
/// Returns the filename extension, which is also the key
///
/// The item to load the key for
/// The file extension used by the module
protected override string GetInterfaceKey(ICompression item)
{
return item.FilenameExtension;
}
///
/// Returns the subfolders searched for compression modules
///
protected override string[] Subfolders => ["compression"];
///
/// The built-in modules
///
protected override IEnumerable BuiltInModules => CompressionModules.BuiltInCompressionModules;
///
/// Instanciates a specific compression module, given the file extension and options
///
/// The file extension to create the instance for
/// The stream of the file used to compress/decompress contents
/// The mode for compression/decompression
/// The options to pass to the instance constructor
/// The instanciated encryption module or null if the file extension is not supported
public ICompression GetModule(string fileExtension, Stream stream, ArchiveMode mode, Dictionary 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;
}
}
///
/// Gets the supported commands for a certain key
///
/// The key to find commands for
/// The supported commands or null if the key was not found
public IReadOnlyList 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;
}
}
}
///
/// The static instance used to access compression module information
///
private static readonly CompressionLoaderSub _compressionLoader = new CompressionLoaderSub();
#region Public static API
///
/// Gets a list of loaded compression modules, the instances can be used to extract interface information, not used to interact with the module.
///
public static ICompression[] Modules { get { return _compressionLoader.Interfaces; } }
///
/// Gets a list of keys supported
///
public static string[] Keys { get { return _compressionLoader.Keys; } }
///
/// Gets the supported commands for a given compression module
///
/// The compression module to find the commands for
/// The supported commands or null if the key is not supported
public static IReadOnlyList GetSupportedCommands(string key)
{
return _compressionLoader.GetSupportedCommands(key);
}
///
/// Instanciates a specific compression module, given the file extension and options
///
/// The file extension to create the instance for
/// The stream of the file used to compress/decompress contents
/// The mode for compression/decompression
/// The options to pass to the instance constructor
/// The instanciated compression module or null if the file extension is not supported
public static ICompression GetModule(string fileextension, Stream stream, ArchiveMode mode, Dictionary options)
{
return _compressionLoader.GetModule(fileextension, stream, mode, options);
}
#endregion
}
}