| | 1 | | #nullable disable |
| | 2 | | #pragma warning disable SA1649 // File name should match first type name |
| | 3 | |
|
| | 4 | | using System; |
| | 5 | | using System.IO; |
| | 6 | | using System.Runtime.InteropServices; |
| | 7 | | using System.Threading; |
| | 8 | | using MediaBrowser.Common.Configuration; |
| | 9 | | using MediaBrowser.Model.Plugins; |
| | 10 | | using MediaBrowser.Model.Serialization; |
| | 11 | |
|
| | 12 | | namespace MediaBrowser.Common.Plugins |
| | 13 | | { |
| | 14 | | /// <summary> |
| | 15 | | /// Provides a common base class for all plugins. |
| | 16 | | /// </summary> |
| | 17 | | /// <typeparam name="TConfigurationType">The type of the T configuration type.</typeparam> |
| | 18 | | public abstract class BasePlugin<TConfigurationType> : BasePlugin, IHasPluginConfiguration |
| | 19 | | where TConfigurationType : BasePluginConfiguration |
| | 20 | | { |
| | 21 | | /// <summary> |
| | 22 | | /// The configuration sync lock. |
| | 23 | | /// </summary> |
| 147 | 24 | | private readonly Lock _configurationSyncLock = new(); |
| | 25 | |
|
| | 26 | | /// <summary> |
| | 27 | | /// The configuration save lock. |
| | 28 | | /// </summary> |
| 147 | 29 | | private readonly Lock _configurationSaveLock = new(); |
| | 30 | |
|
| | 31 | | /// <summary> |
| | 32 | | /// The configuration. |
| | 33 | | /// </summary> |
| | 34 | | private TConfigurationType _configuration; |
| | 35 | |
|
| | 36 | | /// <summary> |
| | 37 | | /// Initializes a new instance of the <see cref="BasePlugin{TConfigurationType}" /> class. |
| | 38 | | /// </summary> |
| | 39 | | /// <param name="applicationPaths">The application paths.</param> |
| | 40 | | /// <param name="xmlSerializer">The XML serializer.</param> |
| 147 | 41 | | protected BasePlugin(IApplicationPaths applicationPaths, IXmlSerializer xmlSerializer) |
| | 42 | | { |
| 147 | 43 | | ApplicationPaths = applicationPaths; |
| 147 | 44 | | XmlSerializer = xmlSerializer; |
| | 45 | |
|
| | 46 | | var assembly = GetType().Assembly; |
| 147 | 47 | | var assemblyName = assembly.GetName(); |
| 147 | 48 | | var assemblyFilePath = assembly.Location; |
| | 49 | |
|
| 147 | 50 | | var dataFolderPath = Path.Combine(ApplicationPaths.PluginsPath, Path.GetFileNameWithoutExtension(assemblyFil |
| 147 | 51 | | if (Version is not null && !Directory.Exists(dataFolderPath)) |
| | 52 | | { |
| | 53 | | // Try again with the version number appended to the folder name. |
| 0 | 54 | | dataFolderPath += "_" + Version; |
| | 55 | | } |
| | 56 | |
|
| 147 | 57 | | SetAttributes(assemblyFilePath, dataFolderPath, assemblyName.Version); |
| | 58 | |
|
| 147 | 59 | | var idAttributes = assembly.GetCustomAttributes(typeof(GuidAttribute), true); |
| 147 | 60 | | if (idAttributes.Length > 0) |
| | 61 | | { |
| 0 | 62 | | var attribute = (GuidAttribute)idAttributes[0]; |
| 0 | 63 | | var assemblyId = new Guid(attribute.Value); |
| | 64 | |
|
| 0 | 65 | | SetId(assemblyId); |
| | 66 | | } |
| 147 | 67 | | } |
| | 68 | |
|
| | 69 | | /// <summary> |
| | 70 | | /// Gets the application paths. |
| | 71 | | /// </summary> |
| | 72 | | /// <value>The application paths.</value> |
| | 73 | | protected IApplicationPaths ApplicationPaths { get; private set; } |
| | 74 | |
|
| | 75 | | /// <summary> |
| | 76 | | /// Gets the XML serializer. |
| | 77 | | /// </summary> |
| | 78 | | /// <value>The XML serializer.</value> |
| | 79 | | protected IXmlSerializer XmlSerializer { get; private set; } |
| | 80 | |
|
| | 81 | | /// <summary> |
| | 82 | | /// Gets the type of configuration this plugin uses. |
| | 83 | | /// </summary> |
| | 84 | | /// <value>The type of the configuration.</value> |
| 0 | 85 | | public Type ConfigurationType => typeof(TConfigurationType); |
| | 86 | |
|
| | 87 | | /// <summary> |
| | 88 | | /// Gets or sets the event handler that is triggered when this configuration changes. |
| | 89 | | /// </summary> |
| | 90 | | public EventHandler<BasePluginConfiguration> ConfigurationChanged { get; set; } |
| | 91 | |
|
| | 92 | | /// <summary> |
| | 93 | | /// Gets the name the assembly file. |
| | 94 | | /// </summary> |
| | 95 | | /// <value>The name of the assembly file.</value> |
| 2 | 96 | | protected string AssemblyFileName => Path.GetFileName(AssemblyFilePath); |
| | 97 | |
|
| | 98 | | /// <summary> |
| | 99 | | /// Gets or sets the plugin configuration. |
| | 100 | | /// </summary> |
| | 101 | | /// <value>The configuration.</value> |
| | 102 | | public TConfigurationType Configuration |
| | 103 | | { |
| | 104 | | get |
| | 105 | | { |
| | 106 | | // Lazy load |
| 105 | 107 | | if (_configuration is null) |
| 42 | 108 | | { |
| | 109 | | lock (_configurationSyncLock) |
| | 110 | | { |
| 42 | 111 | | _configuration ??= LoadConfiguration(); |
| 42 | 112 | | } |
| | 113 | | } |
| | 114 | |
|
| 105 | 115 | | return _configuration; |
| | 116 | | } |
| | 117 | |
|
| 0 | 118 | | protected set => _configuration = value; |
| | 119 | | } |
| | 120 | |
|
| | 121 | | /// <summary> |
| | 122 | | /// Gets the name of the configuration file. Subclasses should override. |
| | 123 | | /// </summary> |
| | 124 | | /// <value>The name of the configuration file.</value> |
| 2 | 125 | | public virtual string ConfigurationFileName => Path.ChangeExtension(AssemblyFileName, ".xml"); |
| | 126 | |
|
| | 127 | | /// <summary> |
| | 128 | | /// Gets the full path to the configuration file. |
| | 129 | | /// </summary> |
| | 130 | | /// <value>The configuration file path.</value> |
| 126 | 131 | | public string ConfigurationFilePath => Path.Combine(ApplicationPaths.PluginConfigurationsPath, ConfigurationFile |
| | 132 | |
|
| | 133 | | /// <summary> |
| | 134 | | /// Gets the plugin configuration. |
| | 135 | | /// </summary> |
| | 136 | | /// <value>The configuration.</value> |
| 0 | 137 | | BasePluginConfiguration IHasPluginConfiguration.Configuration => Configuration; |
| | 138 | |
|
| | 139 | | /// <summary> |
| | 140 | | /// Saves the current configuration to the file system. |
| | 141 | | /// </summary> |
| | 142 | | /// <param name="config">Configuration to save.</param> |
| | 143 | | public virtual void SaveConfiguration(TConfigurationType config) |
| 42 | 144 | | { |
| | 145 | | lock (_configurationSaveLock) |
| | 146 | | { |
| 42 | 147 | | var folder = Path.GetDirectoryName(ConfigurationFilePath); |
| 42 | 148 | | if (!Directory.Exists(folder)) |
| | 149 | | { |
| 21 | 150 | | Directory.CreateDirectory(folder); |
| | 151 | | } |
| | 152 | |
|
| 42 | 153 | | XmlSerializer.SerializeToFile(config, ConfigurationFilePath); |
| 42 | 154 | | } |
| 42 | 155 | | } |
| | 156 | |
|
| | 157 | | /// <summary> |
| | 158 | | /// Saves the current configuration to the file system. |
| | 159 | | /// </summary> |
| | 160 | | public virtual void SaveConfiguration() |
| | 161 | | { |
| 0 | 162 | | SaveConfiguration(Configuration); |
| 0 | 163 | | } |
| | 164 | |
|
| | 165 | | /// <inheritdoc /> |
| | 166 | | public virtual void UpdateConfiguration(BasePluginConfiguration configuration) |
| | 167 | | { |
| 0 | 168 | | ArgumentNullException.ThrowIfNull(configuration); |
| | 169 | |
|
| 0 | 170 | | Configuration = (TConfigurationType)configuration; |
| | 171 | |
|
| 0 | 172 | | SaveConfiguration(Configuration); |
| | 173 | |
|
| 0 | 174 | | ConfigurationChanged?.Invoke(this, configuration); |
| 0 | 175 | | } |
| | 176 | |
|
| | 177 | | /// <inheritdoc /> |
| | 178 | | public override PluginInfo GetPluginInfo() |
| | 179 | | { |
| 7 | 180 | | var info = base.GetPluginInfo(); |
| | 181 | |
|
| 7 | 182 | | info.ConfigurationFileName = ConfigurationFileName; |
| | 183 | |
|
| 7 | 184 | | return info; |
| | 185 | | } |
| | 186 | |
|
| | 187 | | private TConfigurationType LoadConfiguration() |
| | 188 | | { |
| 42 | 189 | | var path = ConfigurationFilePath; |
| | 190 | |
|
| | 191 | | try |
| | 192 | | { |
| 42 | 193 | | return (TConfigurationType)XmlSerializer.DeserializeFromFile(typeof(TConfigurationType), path); |
| | 194 | | } |
| 42 | 195 | | catch |
| | 196 | | { |
| 42 | 197 | | var config = Activator.CreateInstance<TConfigurationType>(); |
| 42 | 198 | | SaveConfiguration(config); |
| 42 | 199 | | return config; |
| | 200 | | } |
| 42 | 201 | | } |
| | 202 | | } |
| | 203 | | } |