160 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			160 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using Dalamud.Plugin;
 | |
| using MareSynchronos.Services.Mediator;
 | |
| using Microsoft.Extensions.Hosting;
 | |
| using Microsoft.Extensions.Logging;
 | |
| using CapturedPluginState = (string InternalName, System.Version Version, bool IsLoaded);
 | |
| 
 | |
| namespace MareSynchronos.Services;
 | |
| 
 | |
| /* Parts of this code from ECommons DalamudReflector
 | |
| 
 | |
| MIT License
 | |
| 
 | |
| Copyright (c) 2023 NightmareXIV
 | |
| 
 | |
| 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.
 | |
| 
 | |
| */
 | |
| 
 | |
| public class PluginWatcherService : MediatorSubscriberBase, IHostedService
 | |
| {
 | |
|     private readonly IDalamudPluginInterface _pluginInterface;
 | |
| 
 | |
|     private CapturedPluginState[] _prevInstalledPluginState = [];
 | |
| 
 | |
| #pragma warning disable
 | |
|     private static bool ExposedPluginsEqual(IEnumerable<IExposedPlugin> plugins, IEnumerable<CapturedPluginState> other)
 | |
|     {
 | |
|         if (plugins.Count() != other.Count()) return false;
 | |
|         var enumeratorOriginal = plugins.GetEnumerator();
 | |
|         var enumeratorOther = other.GetEnumerator();
 | |
|         while (true)
 | |
|         {
 | |
|             var move1 = enumeratorOriginal.MoveNext();
 | |
|             var move2 = enumeratorOther.MoveNext();
 | |
|             if (move1 != move2) return false;
 | |
|             if (move1 == false) return true;
 | |
|             if (enumeratorOriginal.Current.IsLoaded != enumeratorOther.Current.IsLoaded) return false;
 | |
|             if (enumeratorOriginal.Current.Version != enumeratorOther.Current.Version) return false;
 | |
|             if (enumeratorOriginal.Current.InternalName != enumeratorOther.Current.InternalName) return false;
 | |
|         }
 | |
|     }
 | |
| #pragma warning restore
 | |
| 
 | |
|     public PluginWatcherService(ILogger<PluginWatcherService> logger, IDalamudPluginInterface pluginInterface, MareMediator mediator) : base(logger, mediator)
 | |
|     {
 | |
|         _pluginInterface = pluginInterface;
 | |
| 
 | |
|         Mediator.Subscribe<PriorityFrameworkUpdateMessage>(this, (_) =>
 | |
|         {
 | |
|             try
 | |
|             {
 | |
|                 Update();
 | |
|             }
 | |
|             catch (Exception e)
 | |
|             {
 | |
|                 Logger.LogError(e, "PluginWatcherService exception");
 | |
|             }
 | |
|         });
 | |
| 
 | |
|         // Continue scanning plugins during gpose as well
 | |
|         Mediator.Subscribe<CutsceneFrameworkUpdateMessage>(this, (_) =>
 | |
|         {
 | |
|             try
 | |
|             {
 | |
|                 Update();
 | |
|             }
 | |
|             catch (Exception e)
 | |
|             {
 | |
|                 Logger.LogError(e, "PluginWatcherService exception");
 | |
|             }
 | |
|         });
 | |
| 
 | |
|         Update(publish: false);
 | |
|     }
 | |
| 
 | |
|     public Task StartAsync(CancellationToken cancellationToken)
 | |
|     {
 | |
|         return Task.CompletedTask;
 | |
|     }
 | |
| 
 | |
|     public Task StopAsync(CancellationToken cancellationToken)
 | |
|     {
 | |
|         Mediator.UnsubscribeAll(this);
 | |
|         return Task.CompletedTask;
 | |
|     }
 | |
| 
 | |
|     public static PluginChangeMessage? GetInitialPluginState(IDalamudPluginInterface pi, string internalName)
 | |
|     {
 | |
|         try
 | |
|         {
 | |
|             var plugin = pi.InstalledPlugins.Where(p => p.InternalName.Equals(internalName, StringComparison.Ordinal))
 | |
|                 .OrderBy(p => (!p.IsLoaded, p.Version))
 | |
|                 .FirstOrDefault();
 | |
| 
 | |
|             if (plugin == null)
 | |
|                 return null;
 | |
| 
 | |
|             return new PluginChangeMessage(plugin.InternalName, plugin.Version, plugin.IsLoaded);
 | |
|         }
 | |
|         catch
 | |
|         {
 | |
|             return null;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     private void Update(bool publish = true)
 | |
|     {
 | |
|         if (!ExposedPluginsEqual(_pluginInterface.InstalledPlugins, _prevInstalledPluginState))
 | |
|         {
 | |
|             var state = _pluginInterface.InstalledPlugins.Select(x => new CapturedPluginState(x.InternalName, x.Version, x.IsLoaded)).ToArray();
 | |
| 
 | |
|             // The same plugin can be installed multiple times -- InternalName is not unique
 | |
| 
 | |
|             var oldDict = _prevInstalledPluginState.Where(x => x.InternalName.Length > 0)
 | |
|                 .GroupBy(x => x.InternalName, StringComparer.Ordinal)
 | |
|                 .ToDictionary(x => x.Key, StringComparer.Ordinal);
 | |
| 
 | |
|             var newDict = state.Where(x => x.InternalName.Length > 0)
 | |
|                 .GroupBy(x => x.InternalName, StringComparer.Ordinal)
 | |
|                 .ToDictionary(x => x.Key, StringComparer.Ordinal);
 | |
| 
 | |
|             _prevInstalledPluginState = state;
 | |
| 
 | |
|             foreach (var internalName in newDict.Keys.Except(oldDict.Keys, StringComparer.Ordinal))
 | |
|             {
 | |
|                 var p = newDict[internalName].OrderBy(p => (!p.IsLoaded, p.Version)).First();
 | |
|                 if (publish) Mediator.Publish(new PluginChangeMessage(internalName, p.Version, p.IsLoaded));
 | |
|             }
 | |
| 
 | |
|             foreach (var internalName in oldDict.Keys.Except(newDict.Keys, StringComparer.Ordinal))
 | |
|             {
 | |
|                 var p = oldDict[internalName].OrderBy(p => (!p.IsLoaded, p.Version)).First();
 | |
|                 if (publish) Mediator.Publish(new PluginChangeMessage(p.InternalName, p.Version, IsLoaded: false));
 | |
|             }
 | |
| 
 | |
|             foreach (var changedGroup in newDict.Where(p => oldDict.TryGetValue(p.Key, out var old) && !old.SequenceEqual(p.Value)))
 | |
|             {
 | |
|                 var internalName = changedGroup.Value.First().InternalName;
 | |
|                 var p = newDict[internalName].OrderBy(p => (!p.IsLoaded, p.Version)).First();
 | |
|                 if (publish) Mediator.Publish(new PluginChangeMessage(p.InternalName, p.Version, p.IsLoaded));
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| } | 
