#region License Information /* HeuristicLab * Copyright (C) 2002-2010 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.Runtime.CompilerServices; using System.Runtime.Serialization; using System.Runtime.Serialization.Formatters.Binary; using System.Threading; namespace HeuristicLab.Clients.Hive.SlaveCore { public class PluginCache { private static object locker = new object(); private const string lastUsedFileName = "lastUsed.dat"; //maximum number of days after which a plugin gets deleted if not used private const int maxAge = 5; private static PluginCache instance = null; public string PluginCacheDir { get; set; } public string PluginTempBaseDir { get; set; } private List cachedPluginsGuids = new List(); public static PluginCache Instance { get { if (instance == null) instance = new PluginCache(); return instance; } } public PluginCache() { PluginCacheDir = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "PluginCache"); PluginTempBaseDir = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "PluginTemp"); DoUpdateRun(); } private string GetFilenameFromPath(string path) { string[] dirParts = path.Split(Path.DirectorySeparatorChar); if (dirParts.Length > 0) { string fileGuid = dirParts[dirParts.Length - 1]; return fileGuid; } else return ""; } private void DoUpdateRun() { if (!Directory.Exists(PluginCacheDir)) { Directory.CreateDirectory(PluginCacheDir); } foreach (string dir in Directory.EnumerateDirectories(PluginCacheDir)) { cachedPluginsGuids.Add(Guid.Parse(GetFilenameFromPath(dir))); // Todo: cleaner solution to getFilenameFromPath } } [MethodImpl(MethodImplOptions.Synchronized)] public void CopyPluginsForJob(List requests, Guid jobId, out string configFileName) { lock (locker) { configFileName = string.Empty; String targetDir = Path.Combine(PluginTempBaseDir, jobId.ToString()); if (Directory.Exists(targetDir)) { Directory.Delete(targetDir, true); } Directory.CreateDirectory(targetDir); foreach (Plugin requestedPlugin in requests) { var filePaths = GetPluginFilePaths(requestedPlugin.Id); foreach (string filePath in filePaths) { File.Copy(filePath, Path.Combine(targetDir, Path.GetFileName(filePath))); } if (requestedPlugin.Name == "Configuration") { configFileName = Path.Combine(targetDir, Path.GetFileName(filePaths.SingleOrDefault())); // configuration plugin consists only of 1 file (usually the "HeuristicLab X.X.exe.config") } } // copy files from PluginInfrastructure (which are not declared in any plugins) string baseDir = Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location); CopyFile(baseDir, targetDir, "HeuristicLab.PluginInfrastructure-3.3.dll"); CopyFile(baseDir, targetDir, "ICSharpCode.SharpZipLib.dll"); CopyFile(baseDir, targetDir, "ICSharpCode.SharpZipLib License.txt"); // copy slave plugins, otherwise its not possible to register the UnhandledException handler to the appdomain CopyFile(baseDir, targetDir, "HeuristicLab.Clients.Hive.SlaveCore-3.4.dll"); CopyFile(baseDir, targetDir, "HeuristicLab.Clients.Hive-3.4.dll"); //CopyFile(baseDir, targetDir, "HeuristicLab.Services.Hive.Common-3.4.dll"); CopyFile(baseDir, targetDir, "HeuristicLab.Hive-3.4.dll"); CopyFile(baseDir, targetDir, "HeuristicLab.Clients.Common-3.3.dll"); } } private void CopyFile(string baseDir, string targetDir, string fileName) { if (!File.Exists(Path.Combine(targetDir, fileName))) File.Copy(Path.Combine(baseDir, fileName), Path.Combine(targetDir, fileName)); } [MethodImpl(MethodImplOptions.Synchronized)] internal void PreparePlugins(Job myJob, out string configFileName) { lock (locker) { SlaveClientCom.Instance.ClientCom.LogMessage("Fetching plugins for job"); IEnumerable pluginDescriptions = WcfService.Instance.GetPlugins(); List requiredPlugins = new List(); foreach (Guid pluginId in myJob.PluginsNeededIds) { Plugin pl = pluginDescriptions.FirstOrDefault(p => p.Id == pluginId); if (pl != null) requiredPlugins.Add(pl); } List localPlugins = new List(); List missingPlugins = new List(); List missingGuids = new List(); bool found = false; foreach (Plugin info in requiredPlugins) { foreach (Guid cachedPlugin in cachedPluginsGuids) { if (info.Id == cachedPlugin) { localPlugins.Add(new Plugin() { Id = cachedPlugin, Name = info.Name, Version = info.Version }); found = true; break; } } if (!found) { SlaveClientCom.Instance.ClientCom.LogMessage("Plugin NOT found " + info.Name + ", " + info.Version); missingPlugins.Add(info); missingGuids.Add(info.Id); } found = false; } SlaveClientCom.Instance.ClientCom.LogMessage("First run - Update the plugins in the cache"); localPlugins.AddRange(missingPlugins); IEnumerable pluginDatas = WcfService.Instance.GetPluginDatas(missingGuids); foreach (PluginData pluginData in pluginDatas) { string pluginDir = Path.Combine(PluginCacheDir, pluginData.PluginId.ToString()); //put all files belonging to a plugin in the same directory if (!Directory.Exists(pluginDir)) { DirectoryInfo di = Directory.CreateDirectory(pluginDir); } File.WriteAllBytes(Path.Combine(pluginDir, Path.GetFileName(pluginData.FileName)), pluginData.Data); } DoUpdateRun(); CopyPluginsForJob(requiredPlugins, myJob.Id, out configFileName); } } /// /// Returns a list of files which belong to a plugin from the plugincache /// private IEnumerable GetPluginFilePaths(Guid pluginId) { string pluginPath = Path.Combine(PluginCacheDir, pluginId.ToString()); if (Directory.Exists(pluginPath)) { WriteDateLastUsed(pluginPath); foreach (string filePath in Directory.GetFiles(pluginPath)) { string fn = Path.GetFileName(filePath); if (fn != lastUsedFileName) yield return filePath; } } } /// /// creates a file in path with the current date; /// this can later be used to find plugins which are outdated /// private void WriteDateLastUsed(string path) { FileStream fs = new FileStream(Path.Combine(path, lastUsedFileName), FileMode.Create); BinaryFormatter formatter = new BinaryFormatter(); try { formatter.Serialize(fs, DateTime.Now); } catch (SerializationException) { //rethrow... throw; } finally { fs.Close(); } } /// /// checks the pluginCacheDirectory and deletes plugin folders which are not used anymore /// /// private void CleanPluginCache() { FileStream fs = null; DateTime luDate; bool changed = false; if (Directory.Exists(PluginCacheDir)) { foreach (string dir in Directory.EnumerateDirectories(PluginCacheDir)) { try { fs = new FileStream(Path.Combine(dir, lastUsedFileName), FileMode.Open); BinaryFormatter formatter = new BinaryFormatter(); luDate = (DateTime)formatter.Deserialize(fs); fs.Close(); if (luDate.AddDays(maxAge) < DateTime.Now) { Directory.Delete(dir, true); changed = true; } } catch (SerializationException) { fs.Close(); throw; } catch (System.IO.FileNotFoundException) { //nerver used Directory.Delete(dir, true); changed = true; } catch (Exception) { throw; } } } if (changed) DoUpdateRun(); } internal void DeletePluginsForJob(Guid id) { try { SlaveClientCom.Instance.ClientCom.LogMessage("unloading..."); int tries = 5; while (tries > 0) { try { Directory.Delete(Path.Combine(PluginTempBaseDir, id.ToString()), true); tries = 0; } catch (Exception e) { Thread.Sleep(1000); tries--; if (tries == 0) throw;// TODO: don't know what do do } } } catch (Exception ex) { SlaveClientCom.Instance.ClientCom.LogMessage("failed while unloading " + id + " with exception " + ex); } CleanPluginCache(); } } }