#region License Information
/* HeuristicLab
* Copyright (C) 2002-2019 Heuristic and Evolutionary Algorithms Laboratory (HEAL)
*
* This file is part of HeuristicLab.
*
* HeuristicLab is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* HeuristicLab is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with HeuristicLab. If not, see .
*/
#endregion
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Reflection;
using System.Threading.Tasks;
using HeuristicLab.Common;
using ICSharpCode.NRefactory.Documentation;
using ICSharpCode.NRefactory.TypeSystem;
namespace HeuristicLab.CodeEditor {
internal class AssemblyLoader {
private readonly HashSet assemblySet = new HashSet();
private readonly object locker = new object();
public void AddAssembly(Assembly assembly) {
var assemblies = new[] { assembly };
OnAssembliesLoading(assemblies);
var unresolvedAssembly = Load(assembly);
lock (locker)
assemblySet.Add(unresolvedAssembly);
OnInternalAssembliesLoaded(new[] { unresolvedAssembly });
OnAssembliesLoaded(assemblies);
}
public void AddAssemblies(IEnumerable assemblies) {
var a = assemblies.ToArray();
OnAssembliesLoading(a);
var ua = new IUnresolvedAssembly[a.Length];
Parallel.For(0, a.Length, i => ua[i] = Load(a[i]));
lock (locker)
foreach (var asm in ua)
assemblySet.Add(asm);
OnInternalAssembliesLoaded(ua);
OnAssembliesLoaded(a);
}
public async Task AddAssembliesAsync(IEnumerable assemblies) {
await Task.Run(() => AddAssemblies(assemblies));
}
public void RemoveAssembly(Assembly assembly) {
var assemblies = new[] { assembly };
OnAssembliesUnloading(assemblies);
IUnresolvedAssembly unresolvedAssembly;
lock (locker) {
unresolvedAssembly = assemblySet.SingleOrDefault(x => x.FullAssemblyName == assembly.FullName);
assemblySet.Remove(unresolvedAssembly);
}
OnInternalAssembliesUnloaded(new[] { unresolvedAssembly });
OnAssembliesUnloaded(assemblies);
}
#region Loading Helper
private IUnresolvedAssembly Load(Assembly assembly) {
var loader = new CecilLoader {
DocumentationProvider = GetXmlDocumentation(assembly.Location)
};
return loader.LoadAssemblyFile(assembly.Location);
}
#endregion
#region XML Documentation Helpers
private static readonly List xmlDocLookupDirectories = new List {
Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.ProgramFilesX86), @"Reference Assemblies\Microsoft\Framework\.NETFramework\v4.0"),
System.Runtime.InteropServices.RuntimeEnvironment.GetRuntimeDirectory()
};
private static IDocumentationProvider GetXmlDocumentation(string assemblyLocation) {
if (!File.Exists(assemblyLocation)) return null;
string xmlDocFileName = Path.GetFileNameWithoutExtension(assemblyLocation) + ".xml";
foreach (var dir in xmlDocLookupDirectories) {
string xmlDocFileLocation = Path.Combine(dir, xmlDocFileName);
if (File.Exists(xmlDocFileLocation)) return new XmlDocumentationProvider(xmlDocFileLocation);
}
return null;
}
#endregion
#region Events
public event EventHandler>> AssembliesLoading;
private void OnAssembliesLoading(IEnumerable args) {
var handler = AssembliesLoading;
if (handler != null) handler(this, new EventArgs>(args));
}
public event EventHandler>> AssembliesLoaded;
private void OnAssembliesLoaded(IEnumerable args) {
var handler = AssembliesLoaded;
if (handler != null) handler(this, new EventArgs>(args));
}
public event EventHandler>> InternalAssembliesLoaded;
private void OnInternalAssembliesLoaded(IEnumerable args) {
var handler = InternalAssembliesLoaded;
if (handler != null) handler(this, new EventArgs>(args));
}
public event EventHandler>> AssembliesUnloading;
private void OnAssembliesUnloading(IEnumerable args) {
var handler = AssembliesUnloading;
if (handler != null) handler(this, new EventArgs>(args));
}
public event EventHandler>> AssembliesUnloaded;
private void OnAssembliesUnloaded(IEnumerable args) {
var handler = AssembliesUnloaded;
if (handler != null) handler(this, new EventArgs>(args));
}
public event EventHandler>> InternalAssembliesUnloaded;
private void OnInternalAssembliesUnloaded(IEnumerable args) {
var handler = InternalAssembliesUnloaded;
if (handler != null) handler(this, new EventArgs>(args));
}
#endregion
}
}