2beb067cd29ed48d48be79ce8ad62b7264bd667c
[mono.git] / mcs / class / Microsoft.Build.Tasks / Microsoft.Build.Tasks / AssemblyResolver.cs
1 //
2 // AssemblyResolver.cs
3 //
4 // Author:
5 //   Marek Sieradzki (marek.sieradzki@gmail.com)
6 // 
7 // (C) 2006 Marek Sieradzki
8 //
9 // Permission is hereby granted, free of charge, to any person obtaining
10 // a copy of this software and associated documentation files (the
11 // "Software"), to deal in the Software without restriction, including
12 // without limitation the rights to use, copy, modify, merge, publish,
13 // distribute, sublicense, and/or sell copies of the Software, and to
14 // permit persons to whom the Software is furnished to do so, subject to
15 // the following conditions:
16 //
17 // The above copyright notice and this permission notice shall be
18 // included in all copies or substantial portions of the Software.
19 //
20 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
21 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
22 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
23 // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
24 // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
25 // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
26 // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
27
28 #if NET_2_0
29
30 using System;
31 using System.Collections.Generic;
32 using System.IO;
33 using System.Reflection;
34 using System.Security;
35 using Microsoft.Build.Framework;
36 using Microsoft.Build.Utilities;
37
38 namespace Microsoft.Build.Tasks {
39         internal class AssemblyResolver {
40
41                 // name -> (version -> assemblypath)
42                 Dictionary <string, Dictionary <Version, string>> gac;
43                 TaskLoggingHelper log;
44
45                 public AssemblyResolver ()
46                 {
47                         gac = new Dictionary <string, Dictionary <Version, string>> ();
48
49                         GatherGacAssemblies ();
50                 }
51
52                 string GetGacPath ()
53                 {
54                         // NOTE: code from mcs/tools/gacutil/driver.cs
55                         PropertyInfo gac = typeof (System.Environment).GetProperty ("GacPath", BindingFlags.Static | BindingFlags.NonPublic);
56
57                         if (gac == null)
58                                 return null;
59
60                         MethodInfo get_gac = gac.GetGetMethod (true);
61                         return (string) get_gac.Invoke (null, null);
62                 }
63
64                 void GatherGacAssemblies ()
65                 {
66                         string gac_path = GetGacPath ();
67                         if (gac_path == null)
68                                 throw new InvalidOperationException ("XBuild must be run on Mono runtime");
69
70                         Version version;
71                         DirectoryInfo version_info, assembly_info;
72
73                         foreach (string assembly_name in Directory.GetDirectories (gac_path)) {
74                                 assembly_info = new DirectoryInfo (assembly_name);
75                                 foreach (string version_token in Directory.GetDirectories (assembly_name)) {
76                                         foreach (string file in Directory.GetFiles (version_token, "*.dll")) {
77                                                 version_info = new DirectoryInfo (version_token);
78                                                 version = new Version (version_info.Name.Split (
79                                                         new char [] {'_'}, StringSplitOptions.RemoveEmptyEntries) [0]);
80
81                                                 if (!gac.ContainsKey (assembly_info.Name))
82                                                         gac.Add (assembly_info.Name, new Dictionary <Version, string> ());
83                                                 gac [assembly_info.Name].Add (version, file);
84                                         }
85                                 }
86                         }
87                 }
88
89                 public string ResolveAssemblyReference (ITaskItem reference)
90                 {
91                         AssemblyName name = null;
92                         string resolved = null;
93
94                         name = new AssemblyName (reference.ItemSpec);
95
96                         if (reference.GetMetadata ("HintPath") != String.Empty) {
97
98                                 bool specificVersion;
99
100                                 if (reference.GetMetadata ("SpecificVersion") != String.Empty) {
101                                         specificVersion = Boolean.Parse (reference.GetMetadata ("SpecificVersion"));
102                                 } else {
103                                         specificVersion = IsStrongNamed (name);
104                                 }
105
106                                 resolved = ResolveHintPathReference (name, reference.GetMetadata ("HintPath"), specificVersion);
107                         }
108                         
109                         if (resolved == null)
110                                 resolved = ResolveGacReference (name, gac);
111
112                         return resolved;
113                 }
114
115                 string ResolveGacReference (AssemblyName name, Dictionary <string, Dictionary <Version, string>> dic)
116                 {
117                         // FIXME: deal with SpecificVersion=False
118
119                         if (!dic.ContainsKey (name.Name))
120                                 return null;
121
122                         if (name.Version != null) {
123                                 if (!dic [name.Name].ContainsKey (name.Version))
124                                         return null;
125                                 else
126                                         return dic [name.Name] [name.Version];
127                         }
128
129                         Version [] versions = new Version [dic [name.Name].Keys.Count];
130                         dic [name.Name].Keys.CopyTo (versions, 0);
131                         Array.Sort (versions, (IComparer <Version>) null);
132                         Version highest = versions [versions.Length - 1];
133                         return dic [name.Name] [highest];
134                 }
135
136                 string ResolveHintPathReference (AssemblyName name, string hintpath, bool specificVersion)
137                 {
138                         AssemblyName found;
139                         string ret = null;
140
141                         if (!File.Exists (hintpath))
142                                 log.LogMessage (MessageImportance.Low, "HintPath {0} does not exist.", hintpath);
143
144                         try {
145                                 found = AssemblyName.GetAssemblyName (hintpath);
146                                 if (AssemblyNamesCompatible (name, found, specificVersion))
147                                         ret = hintpath;
148                                 else
149                                         log.LogMessage (MessageImportance.Low, "Assembly names are not compatible.");
150                         } catch {
151                         }
152
153                         return ret;
154                 }
155
156                 static bool AssemblyNamesCompatible (AssemblyName a, AssemblyName b, bool specificVersion)
157                 {
158                         if (a.Name != b.Name)
159                                 return false;
160
161                         if (a.CultureInfo != null && a.CultureInfo != b.CultureInfo)
162                                 return false;
163
164                         if (specificVersion && a.Version != null && a.Version > b.Version)
165                                 return false;
166
167                         byte [] a_bytes = a.GetPublicKeyToken ();
168                         byte [] b_bytes = b.GetPublicKeyToken ();
169
170                         if (specificVersion) {
171                                 if (a_bytes == null || a_bytes.Length == 0)
172                                         return false;
173                                 if (b_bytes == null || b_bytes.Length == 0)
174                                         return false;
175
176                                 for (int i = 0; i < a_bytes.Length; i++)
177                                         if (a_bytes [i] != b_bytes [i])
178                                                 return false;
179                         }
180
181                         return true;
182                 }
183
184                 static bool IsStrongNamed (AssemblyName name)
185                 {
186                         return (name.Version != null && name.GetPublicKeyToken ().Length != 0);
187                 }
188
189                 public TaskLoggingHelper Log {
190                         set { log = value; }
191                 }
192         }
193 }
194
195 #endif