1 /**
2  * Visual Studio components
3  *
4  * License:
5  *   This Source Code Form is subject to the terms of
6  *   the Mozilla Public License, v. 2.0. If a copy of
7  *   the MPL was not distributed with this file, You
8  *   can obtain one at http://mozilla.org/MPL/2.0/.
9  *
10  * Authors:
11  *   Vladimir Panteleev <ae@cy.md>
12  */
13 
14 module ae.sys.install.vs;
15 
16 import std.algorithm;
17 import std.array;
18 import std.exception;
19 import std.file;
20 import std.path;
21 import std.process;
22 import std.range;
23 import std.string;
24 
25 import ae.sys.archive;
26 import ae.sys.file;
27 import ae.sys.install.wix;
28 import ae.sys.net;
29 import ae.utils.json;
30 import ae.utils.meta : singleton, I;
31 import ae.utils.xmllite;
32 
33 public import ae.sys.install.common;
34 
35 /// Installs Visual Studio components.
36 class VisualStudioInstaller
37 {
38 	this(int year, string edition, int webInstaller, string versionName)
39 	{
40 		this.year = year;
41 		this.edition = edition;
42 		this.webInstaller = webInstaller;
43 		this.versionName = versionName;
44 	} ///
45 
46 	/// Installs a Visual Studio component.
47 	class VisualStudioComponentInstaller : Installer
48 	{
49 		string packageName; ///
50 
51 	protected:
52 		@property override string name() { return "Visual Studio %d %s (%s)".format(year, edition, packageName); }
53 		@property override string subdirectory() { return "vs%s-%s".format(year, edition.toLower()); }
54 
55 		@property override string[] binPaths() { return modelBinPaths(null); }
56 
57 		@property private string packageMarkerFile() { return ".ae-sys-install-vs/" ~ packageName ~ ".installed"; }
58 
59 		@property override bool installedLocally()
60 		{
61 			if (directory.buildPath("packages.json").exists)
62 			{
63 				log("Old-style installation detected, deleting...");
64 				rmdirRecurse(directory);
65 			}
66 			return directory.buildPath(packageMarkerFile).exists();
67 		}
68 
69 	private:
70 		this() {}
71 		this(string packageName) { this.packageName = packageName; }
72 
73 		static string msdl(int n)
74 		{
75 			return "http://go.microsoft.com/fwlink/?LinkId=%d&clcid=0x409"
76 				.format(n)
77 				.I!resolveRedirect()
78 				.I!save();
79 		}
80 
81 		public static void _decompileMSITo(string msi, string target)
82 		{
83 			wixInstaller.require();
84 			auto status = spawnProcess(["dark", msi, "-o", target]).wait();
85 			enforce(status == 0, "dark failed");
86 		}
87 		public static string _setExtensionWXS(string fn) { return fn.setExtension(".wxs"); }
88 		alias decompileMSI = withTarget!(_setExtensionWXS, cachedAction!(_decompileMSITo, "Decompiling %s to %s..."));
89 
90 		static void installWXS(string wxs, string target)
91 		{
92 			log("Installing %s to %s...".format(wxs, target));
93 
94 			auto wxsDoc = wxs
95 				.readText()
96 				.xmlParse();
97 
98 			string[string] disks;
99 			foreach (media; wxsDoc["Wix"]["Product"].findChildren("Media"))
100 				disks[media.attributes["Id"]] = media
101 					.attributes["Cabinet"]
102 					.absolutePath(wxs.dirName.absolutePath())
103 					.relativePath()
104 					.I!unpack();
105 
106 			void processTag(XmlNode node, string dir)
107 			{
108 				switch (node.tag)
109 				{
110 					case "Directory":
111 					{
112 						auto id = node.attributes["Id"];
113 						switch (id)
114 						{
115 							case "TARGETDIR":
116 								dir = target;
117 								break;
118 							case "ProgramFilesFolder":
119 								dir = dir.buildPath("Program Files (x86)");
120 								break;
121 							case "SystemFolder":
122 								dir = dir.buildPath("windows", "system32");
123 								break;
124 							case "System64Folder":
125 								dir = dir.buildPath("windows", "system64");
126 								break;
127 							default:
128 								if ("Name" in node.attributes)
129 									dir = dir.buildPath(node.attributes["Name"]);
130 								break;
131 						}
132 						break;
133 					}
134 					case "File":
135 					{
136 						auto src = node.attributes["Source"];
137 						enforce(src.startsWith(`SourceDir\File\`));
138 						src = src[`SourceDir\File\`.length .. $];
139 						auto disk = disks[node.attributes["DiskId"]];
140 						src = disk.buildPath(src);
141 						auto dst = dir.buildPath(node.attributes["Name"]);
142 						if (dst.exists)
143 							break;
144 						//log(src ~ " -> " ~ dst);
145 						ensurePathExists(dst);
146 						src.hardLink(dst);
147 						break;
148 					}
149 					default:
150 						break;
151 				}
152 
153 				foreach (child; node.children)
154 					processTag(child, dir);
155 			}
156 
157 			processTag(wxsDoc, null);
158 		}
159 
160 		XmlNode getManifest()
161 		{
162 			if (!manifestCache)
163 				manifestCache =
164 					webInstaller
165 					.I!msdl()
166 					.I!unpack()
167 					.buildPath("0")
168 					.readText()
169 					.xmlParse()
170 					["BurnManifest"];
171 			return manifestCache;
172 		}
173 
174 		void installPackageImpl(string target)
175 		{
176 			windowsOnly();
177 
178 			bool seenPackage;
179 
180 			auto manifest = getManifest();
181 
182 			string[] payloadIDs;
183 			foreach (node; manifest["Chain"].findChildren("MsiPackage"))
184 				if (node.attributes["Id"] == packageName)
185 				{
186 					foreach (payload; node.findChildren("PayloadRef"))
187 						payloadIDs ~= payload.attributes["Id"];
188 					seenPackage = true;
189 				}
190 
191 			enforce(seenPackage, "Unknown package: " ~ packageName);
192 
193 			string[][string] files;
194 			foreach (node; manifest.findChildren("Payload"))
195 				if (payloadIDs.canFind(node.attributes["Id"]))
196 				{
197 					auto path =
198 						node
199 						.attributes["FilePath"]
200 						.prependPath("%s-payloads".format(subdirectory));
201 
202 					auto url = node.attributes["DownloadUrl"];
203 					urlDigests[url] = node.attributes["Hash"].toLower();
204 					files[path.extension.toLower()] ~= url.I!saveAs(path);
205 				}
206 
207 			foreach (cab; files[".cab"])
208 				cab
209 				.I!unpack();
210 
211 			foreach (msi; files[".msi"])
212 				msi
213 				.I!decompileMSI()
214 				.I!installWXS(target);
215 
216 			auto marker = target.buildPath(packageMarkerFile);
217 			marker.ensurePathExists();
218 			marker.touch();
219 		}
220 
221 		void getAllMSIs()
222 		{
223 			auto manifest = getManifest();
224 
225 			string[] payloadIDs;
226 			foreach (node; manifest["Chain"].findChildren("MsiPackage"))
227 				foreach (payload; node.findChildren("PayloadRef"))
228 					payloadIDs ~= payload.attributes["Id"];
229 
230 			foreach (node; manifest.findChildren("Payload"))
231 			{
232 				auto path =
233 					node
234 					.attributes["FilePath"]
235 					.prependPath("%s-payloads".format(subdirectory));
236 
237 				if (path.extension.toLower() == ".msi")
238 				{
239 					auto url = node.attributes["DownloadUrl"];
240 					urlDigests[url] = node.attributes["Hash"].toLower();
241 
242 					url
243 					.I!saveAs(path)
244 					.I!decompileMSI();
245 				}
246 			}
247 		}
248 
249 	protected:
250 		override void atomicInstallImpl()
251 		{
252 			windowsOnly();
253 			auto target = directory ~ "." ~ packageName;
254 			void installPackageImplProxy(string target) { installPackageImpl(target); } // https://issues.dlang.org/show_bug.cgi?id=14580
255 			atomic!installPackageImplProxy(target);
256 			if (!directory.exists)
257 				directory.mkdir();
258 			target.atomicMoveInto(directory);
259 			assert(installedLocally);
260 		}
261 
262 		static this()
263 		{
264 			urlDigests["http://download.microsoft.com/download/7/2/E/72E0F986-D247-4289-B9DC-C4FB07374894/wdexpress_full.exe"] = "8a4c07fa11b20b85126988e7eaf792924b319ae0";
265 			urlDigests["http://download.microsoft.com/download/7/1/B/71BA74D8-B9A0-4E6C-9159-A8335D54437E/vs_community.exe"  ] = "51e5f04fc4648bde3c8276703bf7251216e4ceaf";
266 		}
267 	}
268 
269 	int year; /// Version/year.
270 	int webInstaller; /// Microsoft download number for the web installer.
271 	string edition; /// Edition variant (e.g. "Express").
272 	string versionName; /// Numeric version (e.g. "12.0").
273 
274 	/// Returns the paths to the "bin" directory for the given model.
275 	/// Model is x86 (null), amd64, or x86_amd64
276 	string[] modelBinPaths(string model)
277 	{
278 		string[] result = [
279 			`windows\system32`,
280 			`Program Files (x86)\MSBuild\` ~ versionName ~ `\Bin`,
281 		];
282 
283 		if (!model || model == "x86")
284 		{
285 			result ~= [
286 				`Program Files (x86)\Microsoft Visual Studio ` ~ versionName ~ `\VC\bin`,
287 			];
288 		}
289 		else
290 		if (model == "amd64")
291 		{
292 			result ~= [
293 				`Program Files (x86)\Microsoft Visual Studio ` ~ versionName ~ `\VC\bin\amd64`,
294 			];
295 		}
296 		else
297 		if (model == "x86_amd64")
298 		{
299 			// Binaries which target amd64 are under x86_amd64, but there is only one copy of DLLs
300 			// under bin. Therefore, add the bin directory too, after the x86_amd64 directory.
301 			result ~= [
302 				`Program Files (x86)\Microsoft Visual Studio ` ~ versionName ~ `\VC\bin\x86_amd64`,
303 				`Program Files (x86)\Microsoft Visual Studio ` ~ versionName ~ `\VC\bin`,
304 			];
305 		}
306 
307 		return result;
308 	}
309 
310 	/// Constructs a component installer for the given package.
311 	VisualStudioComponentInstaller opIndex(string name)
312 	{
313 		return new VisualStudioComponentInstaller(name);
314 	}
315 
316 	/// Decompile all MSI files.
317 	/// Useful for finding the name of the package which contains the file you want.
318 	void getAllMSIs()
319 	{
320 		(new VisualStudioComponentInstaller()).getAllMSIs();
321 	}
322 
323 	/// The full installation directory.
324 	@property string directory() { return (new VisualStudioComponentInstaller()).directory; }
325 
326 private:
327 	XmlNode manifestCache;
328 }
329 
330 deprecated alias vs2013 = vs2013express;
331 
332 alias vs2013express   = singleton!(VisualStudioInstaller, 2013, "Express"  , 320697, "12.0"); /// Visual Studio 2013 Express Edition
333 alias vs2013community = singleton!(VisualStudioInstaller, 2013, "Community", 517284, "12.0"); /// Visual Studio 2013 Community Edition