Free cookie consent management tool by TermsFeed Policy Generator

source: branches/HeuristicLab.Hive-3.4/sources/HeuristicLab.Clients.Hive.Slave/3.4/PluginCache.cs @ 5787

Last change on this file since 5787 was 5787, checked in by cneumuel, 13 years ago

#1233

  • made deleting and creating directories for PluginTemp more robust
File size: 10.4 KB
Line 
1#region License Information
2/* HeuristicLab
3 * Copyright (C) 2002-2010 Heuristic and Evolutionary Algorithms Laboratory (HEAL)
4 *
5 * This file is part of HeuristicLab.
6 *
7 * HeuristicLab is free software: you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published by
9 * the Free Software Foundation, either version 3 of the License, or
10 * (at your option) any later version.
11 *
12 * HeuristicLab is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15 * GNU General Public License for more details.
16 *
17 * You should have received a copy of the GNU General Public License
18 * along with HeuristicLab. If not, see <http://www.gnu.org/licenses/>.
19 */
20#endregion
21
22using System;
23using System.Collections.Generic;
24using System.IO;
25using System.Linq;
26using System.Reflection;
27using System.Runtime.CompilerServices;
28using System.Runtime.Serialization;
29using System.Runtime.Serialization.Formatters.Binary;
30using System.Threading;
31
32namespace HeuristicLab.Clients.Hive.SlaveCore {
33  public class PluginCache {
34    private static object locker = new object();
35    private const string lastUsedFileName = "lastUsed.dat";
36
37    //maximum number of days after which a plugin gets deleted if not used
38    private const int maxAge = 5;
39
40    private static PluginCache instance = null;
41
42    public string PluginCacheDir { get; set; }
43    public string PluginTempBaseDir { get; set; }
44
45    private List<Guid> cachedPluginsGuids = new List<Guid>();
46
47    public static PluginCache Instance {
48      get {
49        if (instance == null)
50          instance = new PluginCache();
51        return instance;
52      }
53    }
54
55    public PluginCache() {
56      PluginCacheDir = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "PluginCache");
57      PluginTempBaseDir = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "PluginTemp");
58
59      DoUpdateRun();
60    }
61
62    private string GetFilenameFromPath(string path) {
63      string[] dirParts = path.Split(Path.DirectorySeparatorChar);
64      if (dirParts.Length > 0) {
65        string fileGuid = dirParts[dirParts.Length - 1];
66        return fileGuid;
67      } else
68        return "";
69    }
70
71    private void DoUpdateRun() {
72      SafelyCreateDirectory(PluginCacheDir);
73      foreach (string dir in Directory.EnumerateDirectories(PluginCacheDir)) {
74        cachedPluginsGuids.Add(Guid.Parse(GetFilenameFromPath(dir))); // Todo: cleaner solution to getFilenameFromPath
75      }
76    }
77
78    [MethodImpl(MethodImplOptions.Synchronized)]
79    public void CopyPluginsForJob(List<Plugin> requests, Guid jobId, out string configFileName) {
80      lock (locker) {
81        configFileName = string.Empty;
82        String targetDir = Path.Combine(PluginTempBaseDir, jobId.ToString());
83
84        RecreateDirectory(targetDir);
85
86        foreach (Plugin requestedPlugin in requests) {
87          var filePaths = GetPluginFilePaths(requestedPlugin.Id);
88          foreach (string filePath in filePaths) {
89            File.Copy(filePath, Path.Combine(targetDir, Path.GetFileName(filePath)));
90          }
91
92          if (requestedPlugin.Name == "Configuration") {
93            configFileName = Path.Combine(targetDir, Path.GetFileName(filePaths.SingleOrDefault())); // configuration plugin consists only of 1 file (usually the "HeuristicLab X.X.exe.config")
94          }
95        }
96
97        // copy files from PluginInfrastructure (which are not declared in any plugins)
98        string baseDir = Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location);
99        CopyFile(baseDir, targetDir, "HeuristicLab.PluginInfrastructure-3.3.dll");
100        CopyFile(baseDir, targetDir, "ICSharpCode.SharpZipLib.dll");
101        CopyFile(baseDir, targetDir, "ICSharpCode.SharpZipLib License.txt");
102
103        // copy slave plugins, otherwise its not possible to register the UnhandledException handler to the appdomain       
104        CopyFile(baseDir, targetDir, "HeuristicLab.Clients.Hive.SlaveCore-3.4.dll");
105        CopyFile(baseDir, targetDir, "HeuristicLab.Clients.Hive-3.4.dll");
106        //CopyFile(baseDir, targetDir, "HeuristicLab.Services.Hive.Common-3.4.dll");
107        CopyFile(baseDir, targetDir, "HeuristicLab.Hive-3.4.dll");
108        CopyFile(baseDir, targetDir, "HeuristicLab.Clients.Common-3.3.dll");
109      }
110    }
111
112    private static DirectoryInfo RecreateDirectory(String targetDir) {
113      var di = new DirectoryInfo(targetDir);
114      if (di.Exists) Directory.Delete(targetDir, true);
115      di.Refresh();
116      while (di.Exists) {
117        Thread.Sleep(50);
118        di.Refresh();
119      }
120      return SafelyCreateDirectory(targetDir);
121    }
122
123    private static DirectoryInfo SafelyCreateDirectory(String targetDir) {
124      var di = new DirectoryInfo(targetDir);
125      if (!di.Exists) {
126        di = Directory.CreateDirectory(targetDir);
127        while (!di.Exists) {
128          Thread.Sleep(50);
129          di.Refresh();
130        }
131      }
132      return di;
133    }
134
135    private void CopyFile(string baseDir, string targetDir, string fileName) {
136      if (!File.Exists(Path.Combine(targetDir, fileName))) File.Copy(Path.Combine(baseDir, fileName), Path.Combine(targetDir, fileName));
137    }
138
139    [MethodImpl(MethodImplOptions.Synchronized)]
140    internal void PreparePlugins(Job myJob, out string configFileName) {
141      lock (locker) {
142        SlaveClientCom.Instance.ClientCom.LogMessage("Fetching plugins for job");
143
144        IEnumerable<Plugin> pluginDescriptions = WcfService.Instance.GetPlugins();
145        List<Plugin> requiredPlugins = new List<Plugin>();
146        foreach (Guid pluginId in myJob.PluginsNeededIds) {
147          Plugin pl = pluginDescriptions.FirstOrDefault(p => p.Id == pluginId);
148          if (pl != null)
149            requiredPlugins.Add(pl);
150        }
151
152        List<Plugin> localPlugins = new List<Plugin>();
153        List<Plugin> missingPlugins = new List<Plugin>();
154        List<Guid> missingGuids = new List<Guid>();
155        bool found = false;
156
157        foreach (Plugin info in requiredPlugins) {
158          foreach (Guid cachedPlugin in cachedPluginsGuids) {
159            if (info.Id == cachedPlugin) {
160              localPlugins.Add(new Plugin() { Id = cachedPlugin, Name = info.Name, Version = info.Version });
161              found = true;
162
163              break;
164            }
165          }
166          if (!found) {
167            SlaveClientCom.Instance.ClientCom.LogMessage("Plugin NOT found " + info.Name + ", " + info.Version);
168            missingPlugins.Add(info);
169            missingGuids.Add(info.Id);
170          }
171          found = false;
172        }
173
174        SlaveClientCom.Instance.ClientCom.LogMessage("First run - Update the plugins in the cache");
175        localPlugins.AddRange(missingPlugins);
176
177        IEnumerable<PluginData> pluginDatas = WcfService.Instance.GetPluginDatas(missingGuids);
178
179        foreach (PluginData pluginData in pluginDatas) {
180          string pluginDir = Path.Combine(PluginCacheDir, pluginData.PluginId.ToString());
181
182          //put all files belonging to a plugin in the same directory
183          SafelyCreateDirectory(pluginDir);
184          File.WriteAllBytes(Path.Combine(pluginDir, Path.GetFileName(pluginData.FileName)), pluginData.Data);
185        }
186
187        DoUpdateRun();
188        CopyPluginsForJob(requiredPlugins, myJob.Id, out configFileName);
189      }
190    }
191
192    /// <summary>
193    /// Returns a list of files which belong to a plugin from the plugincache
194    /// </summary>
195    private IEnumerable<string> GetPluginFilePaths(Guid pluginId) {
196      string pluginPath = Path.Combine(PluginCacheDir, pluginId.ToString());
197
198      if (Directory.Exists(pluginPath)) {
199        WriteDateLastUsed(pluginPath);
200        foreach (string filePath in Directory.GetFiles(pluginPath)) {
201          string fn = Path.GetFileName(filePath);
202          if (fn != lastUsedFileName)
203            yield return filePath;
204        }
205      }
206    }
207
208    /// <summary>
209    /// creates a file in path with the current date;
210    /// this can later be used to find plugins which are outdated
211    /// </summary>   
212    private void WriteDateLastUsed(string path) {
213      FileStream fs = new FileStream(Path.Combine(path, lastUsedFileName), FileMode.Create);
214      BinaryFormatter formatter = new BinaryFormatter();
215
216      try {
217        formatter.Serialize(fs, DateTime.Now);
218      }
219      catch (SerializationException) {
220        //rethrow...
221        throw;
222      }
223      finally {
224        fs.Close();
225      }
226    }
227
228    /// <summary>
229    /// checks the pluginCacheDirectory and deletes plugin folders which are not used anymore
230    /// </summary>
231    /// <param name="path"></param>
232    private void CleanPluginCache() {
233      FileStream fs = null;
234      DateTime luDate;
235      bool changed = false;
236
237      if (Directory.Exists(PluginCacheDir)) {
238        foreach (string dir in Directory.EnumerateDirectories(PluginCacheDir)) {
239          try {
240            fs = new FileStream(Path.Combine(dir, lastUsedFileName), FileMode.Open);
241            BinaryFormatter formatter = new BinaryFormatter();
242            luDate = (DateTime)formatter.Deserialize(fs);
243            fs.Close();
244
245            if (luDate.AddDays(maxAge) < DateTime.Now) {
246              Directory.Delete(dir, true);
247              changed = true;
248            }
249          }
250          catch (SerializationException) {
251            fs.Close();
252            throw;
253          }
254          catch (System.IO.FileNotFoundException) {
255            //nerver used
256            Directory.Delete(dir, true);
257            changed = true;
258          }
259          catch (Exception) {
260            throw;
261          }
262        }
263      }
264      if (changed)
265        DoUpdateRun();
266    }
267
268
269    internal void DeletePluginsForJob(Guid id) {
270      try {
271        SlaveClientCom.Instance.ClientCom.LogMessage("unloading...");
272        int tries = 5;
273        while (tries > 0) {
274          try {
275            Directory.Delete(Path.Combine(PluginTempBaseDir, id.ToString()), true);
276            tries = 0;
277          }
278          catch (Exception e) {
279            Thread.Sleep(1000);
280            tries--;
281            if (tries == 0) throw;// TODO: don't know what do do
282          }
283        }
284      }
285      catch (Exception ex) {
286        SlaveClientCom.Instance.ClientCom.LogMessage("failed while unloading " + id + " with exception " + ex);
287      }
288      CleanPluginCache();
289    }
290  }
291}
Note: See TracBrowser for help on using the repository browser.