aboutsummaryrefslogtreecommitdiff
path: root/src/light
diff options
context:
space:
mode:
authorRob Mensching <rob@firegiant.com>2017-12-22 15:53:01 -0800
committerRob Mensching <rob@firegiant.com>2017-12-22 15:53:01 -0800
commitecf3a0cca5a424a91ab98557d963d2535963d582 (patch)
tree06355e906e5c404480dc6eac342b9b4d2ec9d122 /src/light
parentdc9f4c329e6f55ce7595970463e0caf148096f4b (diff)
downloadwix-ecf3a0cca5a424a91ab98557d963d2535963d582.tar.gz
wix-ecf3a0cca5a424a91ab98557d963d2535963d582.tar.bz2
wix-ecf3a0cca5a424a91ab98557d963d2535963d582.zip
Reintroduce binder extensions and light.exe for binding .wixouts
Diffstat (limited to 'src/light')
-rw-r--r--src/light/App.icobin0 -> 1078 bytes
-rw-r--r--src/light/AssemblyInfo.cs9
-rw-r--r--src/light/LightCommandLine.cs485
-rw-r--r--src/light/LightStrings.Designer.cs104
-rw-r--r--src/light/LightStrings.resx174
-rw-r--r--src/light/app.config9
-rw-r--r--src/light/light.cs595
-rw-r--r--src/light/light.csproj21
8 files changed, 1397 insertions, 0 deletions
diff --git a/src/light/App.ico b/src/light/App.ico
new file mode 100644
index 00000000..3a5525fd
--- /dev/null
+++ b/src/light/App.ico
Binary files differ
diff --git a/src/light/AssemblyInfo.cs b/src/light/AssemblyInfo.cs
new file mode 100644
index 00000000..ab2fc0ab
--- /dev/null
+++ b/src/light/AssemblyInfo.cs
@@ -0,0 +1,9 @@
1// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information.
2
3using System;
4using System.Reflection;
5using System.Runtime.CompilerServices;
6using System.Runtime.InteropServices;
7
8[assembly: AssemblyCulture("")]
9[assembly: ComVisible(false)]
diff --git a/src/light/LightCommandLine.cs b/src/light/LightCommandLine.cs
new file mode 100644
index 00000000..9a90b9ce
--- /dev/null
+++ b/src/light/LightCommandLine.cs
@@ -0,0 +1,485 @@
1// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information.
2
3namespace WixToolset.Tools
4{
5 using System;
6 using System.Collections.Generic;
7 using System.Globalization;
8 using System.IO;
9 using WixToolset.Core.CommandLine;
10 using WixToolset.Data;
11 using WixToolset.Extensibility.Services;
12
13 public class LightCommandLine
14 {
15 public LightCommandLine(IMessaging messaging)
16 {
17 this.Messaging = messaging;
18 this.ShowLogo = true;
19 this.Tidy = true;
20
21 this.CubeFiles = new List<string>();
22 this.SuppressIces = new List<string>();
23 this.Ices = new List<string>();
24 this.BindPaths = new List<BindPath>();
25 this.Extensions = new List<string>();
26 this.Files = new List<string>();
27 this.LocalizationFiles = new List<string>();
28 this.Variables = new Dictionary<string, string>();
29 }
30
31 public IMessaging Messaging { get; }
32
33 public string PdbFile { get; private set; }
34
35 public CompressionLevel? DefaultCompressionLevel { get; set; }
36
37 public bool SuppressAclReset { get; private set; }
38
39 public bool SuppressLayout { get; private set; }
40
41 public bool SuppressWixPdb { get; private set; }
42
43 public bool SuppressValidation { get; private set; }
44
45 public string IntermediateFolder { get; private set; }
46
47 public string OutputsFile { get; private set; }
48
49 public string BuiltOutputsFile { get; private set; }
50
51 public string WixprojectFile { get; private set; }
52
53 public string ContentsFile { get; private set; }
54
55 public List<string> Ices { get; private set; }
56
57 public string CabCachePath { get; private set; }
58
59 public int CabbingThreadCount { get; private set; }
60
61 public List<string> CubeFiles { get; private set; }
62
63 public List<string> SuppressIces { get; private set; }
64
65 public bool ShowLogo { get; private set; }
66
67 public bool ShowHelp { get; private set; }
68
69 public bool ShowPedanticMessages { get; private set; }
70
71 public bool SuppressLocalization { get; private set; }
72
73 public bool SuppressVersionCheck { get; private set; }
74
75 public string[] Cultures { get; private set; }
76
77 public string OutputFile { get; private set; }
78
79 public bool OutputXml { get; private set; }
80
81 public List<BindPath> BindPaths { get; private set; }
82
83 public List<string> Extensions { get; private set; }
84
85 public List<string> Files { get; private set; }
86
87 public List<string> LocalizationFiles { get; private set; }
88
89 public bool Tidy { get; private set; }
90
91 public string UnreferencedSymbolsFile { get; private set; }
92
93 public IDictionary<string, string> Variables { get; private set; }
94
95 /// <summary>
96 /// Parse the commandline arguments.
97 /// </summary>
98 /// <param name="args">Commandline arguments.</param>
99 public string[] Parse(string[] args)
100 {
101 List<string> unprocessed = new List<string>();
102
103 for (int i = 0; i < args.Length; ++i)
104 {
105 string arg = args[i];
106 if (String.IsNullOrEmpty(arg)) // skip blank arguments
107 {
108 continue;
109 }
110
111 if (1 == arg.Length) // treat '-' and '@' as filenames when by themselves.
112 {
113 unprocessed.Add(arg);
114 }
115 else if ('-' == arg[0] || '/' == arg[0])
116 {
117 string parameter = arg.Substring(1);
118 if (parameter.Equals("b", StringComparison.Ordinal))
119 {
120 if (!CommandLineHelper.IsValidArg(args, ++i))
121 {
122 break;
123 }
124
125 var bindPath = BindPath.Parse(args[i]);
126
127 this.BindPaths.Add(bindPath);
128 }
129 else if (parameter.StartsWith("cultures:", StringComparison.Ordinal))
130 {
131 string culturesString = arg.Substring(10).ToLower(CultureInfo.InvariantCulture);
132
133 // When null is used treat it as if cultures wasn't specified.
134 // This is needed for batching over the light task when using MSBuild which doesn't
135 // support empty items
136 if (culturesString.Equals("null", StringComparison.OrdinalIgnoreCase))
137 {
138 this.Cultures = null;
139 }
140 else
141 {
142 this.Cultures = culturesString.Split(';', ',');
143
144 for (int c = 0; c < this.Cultures.Length; ++c)
145 {
146 // Neutral is different from null. For neutral we still want to do WXL filtering.
147 // Set the culture to the empty string = identifier for the invariant culture
148 if (this.Cultures[c].Equals("neutral", StringComparison.OrdinalIgnoreCase))
149 {
150 this.Cultures[c] = String.Empty;
151 }
152 }
153 }
154 }
155 else if (parameter.StartsWith("dcl:", StringComparison.Ordinal))
156 {
157 string defaultCompressionLevel = arg.Substring(5);
158
159 if (String.IsNullOrEmpty(defaultCompressionLevel))
160 {
161 break;
162 }
163 else if (Enum.TryParse(defaultCompressionLevel, true, out CompressionLevel compressionLevel))
164 {
165 this.DefaultCompressionLevel = compressionLevel;
166 }
167 }
168 else if (parameter.StartsWith("d", StringComparison.Ordinal))
169 {
170 parameter = arg.Substring(2);
171 string[] value = parameter.Split("=".ToCharArray(), 2);
172
173 string preexisting;
174 if (1 == value.Length)
175 {
176 this.Messaging.Write(ErrorMessages.ExpectedWixVariableValue(value[0]));
177 }
178 else if (this.Variables.TryGetValue(value[0], out preexisting))
179 {
180 this.Messaging.Write(ErrorMessages.WixVariableCollision(null, value[0]));
181 }
182 else
183 {
184 this.Variables.Add(value[0], value[1]);
185 }
186 }
187 else if (parameter.Equals("ext", StringComparison.Ordinal))
188 {
189 if (!CommandLineHelper.IsValidArg(args, ++i))
190 {
191 this.Messaging.Write(ErrorMessages.TypeSpecificationForExtensionRequired("-ext"));
192 break;
193 }
194
195 this.Extensions.Add(args[i]);
196 }
197 else if (parameter.Equals("loc", StringComparison.Ordinal))
198 {
199 string locFile = CommandLineHelper.GetFile(parameter, this.Messaging, args, ++i);
200 if (String.IsNullOrEmpty(locFile))
201 {
202 break;
203 }
204
205 this.LocalizationFiles.Add(locFile);
206 }
207 else if (parameter.Equals("nologo", StringComparison.Ordinal))
208 {
209 this.ShowLogo = false;
210 }
211 else if (parameter.Equals("notidy", StringComparison.Ordinal))
212 {
213 this.Tidy = false;
214 }
215 else if ("o" == parameter || "out" == parameter)
216 {
217 this.OutputFile = CommandLineHelper.GetFile(parameter, this.Messaging, args, ++i);
218 if (String.IsNullOrEmpty(this.OutputFile))
219 {
220 break;
221 }
222 }
223 else if (parameter.Equals("pedantic", StringComparison.Ordinal))
224 {
225 this.ShowPedanticMessages = true;
226 }
227 else if (parameter.Equals("sloc", StringComparison.Ordinal))
228 {
229 this.SuppressLocalization = true;
230 }
231 else if (parameter.Equals("usf", StringComparison.Ordinal))
232 {
233 this.UnreferencedSymbolsFile = CommandLineHelper.GetFile(parameter, this.Messaging, args, ++i);
234
235 if (String.IsNullOrEmpty(this.UnreferencedSymbolsFile))
236 {
237 break;
238 }
239 }
240 else if (parameter.Equals("xo", StringComparison.Ordinal))
241 {
242 this.OutputXml = true;
243 }
244 else if (parameter.Equals("cc", StringComparison.Ordinal))
245 {
246 this.CabCachePath = CommandLineHelper.GetDirectory(parameter, this.Messaging, args, ++i);
247
248 if (String.IsNullOrEmpty(this.CabCachePath))
249 {
250 break;
251 }
252 }
253 else if (parameter.Equals("ct", StringComparison.Ordinal))
254 {
255 if (!CommandLineHelper.IsValidArg(args, ++i))
256 {
257 this.Messaging.Write(ErrorMessages.IllegalCabbingThreadCount(String.Empty));
258 break;
259 }
260
261 int ct = 0;
262 if (!Int32.TryParse(args[i], out ct) || 0 >= ct)
263 {
264 this.Messaging.Write(ErrorMessages.IllegalCabbingThreadCount(args[i]));
265 break;
266 }
267
268 this.CabbingThreadCount = ct;
269 this.Messaging.Write(VerboseMessages.SetCabbingThreadCount(this.CabbingThreadCount.ToString()));
270 }
271 else if (parameter.Equals("cub", StringComparison.Ordinal))
272 {
273 string cubeFile = CommandLineHelper.GetFile(parameter, this.Messaging, args, ++i);
274
275 if (String.IsNullOrEmpty(cubeFile))
276 {
277 break;
278 }
279
280 this.CubeFiles.Add(cubeFile);
281 }
282 else if (parameter.StartsWith("ice:", StringComparison.Ordinal))
283 {
284 this.Ices.Add(parameter.Substring(4));
285 }
286 else if (parameter.Equals("intermediatefolder", StringComparison.OrdinalIgnoreCase))
287 {
288 this.IntermediateFolder = CommandLineHelper.GetDirectory(parameter, this.Messaging, args, ++i);
289
290 if (String.IsNullOrEmpty(this.IntermediateFolder))
291 {
292 break;
293 }
294 }
295 else if (parameter.Equals("contentsfile", StringComparison.Ordinal))
296 {
297 this.ContentsFile = CommandLineHelper.GetFile(parameter, this.Messaging, args, ++i);
298
299 if (String.IsNullOrEmpty(this.ContentsFile))
300 {
301 break;
302 }
303 }
304 else if (parameter.Equals("outputsfile", StringComparison.Ordinal))
305 {
306 this.OutputsFile = CommandLineHelper.GetFile(parameter, this.Messaging, args, ++i);
307
308 if (String.IsNullOrEmpty(this.OutputsFile))
309 {
310 break;
311 }
312 }
313 else if (parameter.Equals("builtoutputsfile", StringComparison.Ordinal))
314 {
315 this.BuiltOutputsFile = CommandLineHelper.GetFile(parameter, this.Messaging, args, ++i);
316
317 if (String.IsNullOrEmpty(this.BuiltOutputsFile))
318 {
319 break;
320 }
321 }
322 else if (parameter.Equals("wixprojectfile", StringComparison.Ordinal))
323 {
324 this.WixprojectFile = CommandLineHelper.GetFile(parameter, this.Messaging, args, ++i);
325
326 if (String.IsNullOrEmpty(this.WixprojectFile))
327 {
328 break;
329 }
330 }
331 else if (parameter.Equals("pdbout", StringComparison.Ordinal))
332 {
333 this.PdbFile = CommandLineHelper.GetFile(parameter, this.Messaging, args, ++i);
334
335 if (String.IsNullOrEmpty(this.PdbFile))
336 {
337 break;
338 }
339 }
340 else if (parameter.StartsWith("sice:", StringComparison.Ordinal))
341 {
342 this.SuppressIces.Add(parameter.Substring(5));
343 }
344 else if (parameter.Equals("sl", StringComparison.Ordinal))
345 {
346 this.SuppressLayout = true;
347 }
348 else if (parameter.Equals("spdb", StringComparison.Ordinal))
349 {
350 this.SuppressWixPdb = true;
351 }
352 else if (parameter.Equals("sacl", StringComparison.Ordinal))
353 {
354 this.SuppressAclReset = true;
355 }
356 else if (parameter.Equals("sval", StringComparison.Ordinal))
357 {
358 this.SuppressValidation = true;
359 }
360 else if ("sv" == parameter)
361 {
362 this.SuppressVersionCheck = true;
363 }
364 else if (parameter.StartsWith("sw", StringComparison.Ordinal))
365 {
366 string paramArg = parameter.Substring(2);
367 if (0 == paramArg.Length)
368 {
369 this.Messaging.SuppressAllWarnings = true;
370 }
371 else
372 {
373 int suppressWarning = 0;
374 if (!Int32.TryParse(paramArg, out suppressWarning) || 0 >= suppressWarning)
375 {
376 this.Messaging.Write(ErrorMessages.IllegalSuppressWarningId(paramArg));
377 }
378 else
379 {
380 this.Messaging.SuppressWarningMessage(suppressWarning);
381 }
382 }
383 }
384 else if (parameter.StartsWith("wx", StringComparison.Ordinal))
385 {
386 string paramArg = parameter.Substring(2);
387 if (0 == paramArg.Length)
388 {
389 this.Messaging.WarningsAsError = true;
390 }
391 else
392 {
393 int elevateWarning = 0;
394 if (!Int32.TryParse(paramArg, out elevateWarning) || 0 >= elevateWarning)
395 {
396 this.Messaging.Write(ErrorMessages.IllegalWarningIdAsError(paramArg));
397 }
398 else
399 {
400 this.Messaging.ElevateWarningMessage(elevateWarning);
401 }
402 }
403 }
404 else if ("v" == parameter)
405 {
406 this.Messaging.ShowVerboseMessages = true;
407 }
408 else if ("?" == parameter || "help" == parameter)
409 {
410 this.ShowHelp = true;
411 break;
412 }
413 else
414 {
415 unprocessed.Add(arg);
416 }
417 }
418 else if ('@' == arg[0])
419 {
420 string[] parsedArgs = CommandLineResponseFile.Parse(arg.Substring(1));
421 string[] unparsedArgs = this.Parse(parsedArgs);
422 unprocessed.AddRange(unparsedArgs);
423 }
424 else
425 {
426 unprocessed.Add(arg);
427 }
428 }
429
430 return unprocessed.ToArray();
431 }
432
433 public string[] ParsePostExtensions(string[] remaining)
434 {
435 List<string> unprocessed = new List<string>();
436
437 for (int i = 0; i < remaining.Length; ++i)
438 {
439 string arg = remaining[i];
440 if (String.IsNullOrEmpty(arg)) // skip blank arguments
441 {
442 continue;
443 }
444
445 if (1 < arg.Length && ('-' == arg[0] || '/' == arg[0]))
446 {
447 unprocessed.Add(arg);
448 }
449 else
450 {
451 this.Files.AddRange(CommandLineHelper.GetFiles(arg, "Source"));
452 }
453 }
454
455 if (0 == this.Files.Count)
456 {
457 this.ShowHelp = true;
458 }
459 else if (String.IsNullOrEmpty(this.OutputFile))
460 {
461 if (1 < this.Files.Count)
462 {
463 this.Messaging.Write(ErrorMessages.MustSpecifyOutputWithMoreThanOneInput());
464 }
465
466 // After the linker tells us what the output type actually is, we'll change the ".wix" to the correct extension.
467 this.OutputFile = Path.ChangeExtension(Path.GetFileName(this.Files[0]), ".wix");
468
469 // Add the directories of the input files as unnamed bind paths.
470 foreach (string file in this.Files)
471 {
472 BindPath bindPath = new BindPath(Path.GetDirectoryName(Path.GetFullPath(file)));
473 this.BindPaths.Add(bindPath);
474 }
475 }
476
477 if (!this.SuppressWixPdb && String.IsNullOrEmpty(this.PdbFile) && !String.IsNullOrEmpty(this.OutputFile))
478 {
479 this.PdbFile = Path.ChangeExtension(this.OutputFile, ".wixpdb");
480 }
481
482 return unprocessed.ToArray();
483 }
484 }
485}
diff --git a/src/light/LightStrings.Designer.cs b/src/light/LightStrings.Designer.cs
new file mode 100644
index 00000000..50e271fd
--- /dev/null
+++ b/src/light/LightStrings.Designer.cs
@@ -0,0 +1,104 @@
1// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information.
2
3namespace WixToolset.Tools {
4 using System;
5
6
7 /// <summary>
8 /// A strongly-typed resource class, for looking up localized strings, etc.
9 /// </summary>
10 // This class was auto-generated by the StronglyTypedResourceBuilder
11 // class via a tool like ResGen or Visual Studio.
12 // To add or remove a member, edit your .ResX file then rerun ResGen
13 // with the /str option, or rebuild your VS project.
14 [global::System.CodeDom.Compiler.GeneratedCodeAttribute("System.Resources.Tools.StronglyTypedResourceBuilder", "4.0.0.0")]
15 [global::System.Diagnostics.DebuggerNonUserCodeAttribute()]
16 [global::System.Runtime.CompilerServices.CompilerGeneratedAttribute()]
17 internal class LightStrings {
18
19 private static global::System.Resources.ResourceManager resourceMan;
20
21 private static global::System.Globalization.CultureInfo resourceCulture;
22
23 [global::System.Diagnostics.CodeAnalysis.SuppressMessageAttribute("Microsoft.Performance", "CA1811:AvoidUncalledPrivateCode")]
24 internal LightStrings() {
25 }
26
27 /// <summary>
28 /// Returns the cached ResourceManager instance used by this class.
29 /// </summary>
30 [global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState.Advanced)]
31 internal static global::System.Resources.ResourceManager ResourceManager {
32 get {
33 if (object.ReferenceEquals(resourceMan, null)) {
34 global::System.Resources.ResourceManager temp = new global::System.Resources.ResourceManager("WixToolset.Tools.LightStrings", typeof(LightStrings).Assembly);
35 resourceMan = temp;
36 }
37 return resourceMan;
38 }
39 }
40
41 /// <summary>
42 /// Overrides the current thread's CurrentUICulture property for all
43 /// resource lookups using this strongly typed resource class.
44 /// </summary>
45 [global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState.Advanced)]
46 internal static global::System.Globalization.CultureInfo Culture {
47 get {
48 return resourceCulture;
49 }
50 set {
51 resourceCulture = value;
52 }
53 }
54
55 /// <summary>
56 /// Looks up a localized string similar to -b &lt;path&gt; specify a binder path to locate all files
57 /// (default: current directory)
58 /// prefix the path with &apos;name=&apos; where &apos;name&apos; is the name of your
59 /// named bindpath.
60 /// -cc &lt;path&gt; path to cache built cabinets (will not be deleted after linking)
61 /// -ct &lt;N&gt; number of threads to use when creating cabinets
62 /// (default: %NUMBER_OF_PROCESSORS%)
63 /// -cub &lt;file.cub&gt; additional .cub file containing ICEs to run
64 /// -dcl:level set default cabinet compression l [rest of string was truncated]&quot;;.
65 /// </summary>
66 internal static string CommandLineArguments {
67 get {
68 return ResourceManager.GetString("CommandLineArguments", resourceCulture);
69 }
70 }
71
72 /// <summary>
73 /// Looks up a localized string similar to The -bf (bind files) option is only applicable with the -xo option..
74 /// </summary>
75 internal static string EXP_BindFileOptionNotApplicable {
76 get {
77 return ResourceManager.GetString("EXP_BindFileOptionNotApplicable", resourceCulture);
78 }
79 }
80
81 /// <summary>
82 /// Looks up a localized string similar to Cannot link object files (.wixobj) files with an output file (.wixout).
83 /// </summary>
84 internal static string EXP_CannotLinkObjFilesWithOutpuFile {
85 get {
86 return ResourceManager.GetString("EXP_CannotLinkObjFilesWithOutpuFile", resourceCulture);
87 }
88 }
89
90 /// <summary>
91 /// Looks up a localized string similar to usage: light.exe [-?] [-b bindPath] [-nologo] [-out outputFile] objectFile [objectFile ...] [@responseFile]
92 ///
93 ///{0}
94 ///
95 ///Environment variables:
96 /// WIX_TEMP overrides the temporary directory used for cab creation, msm exploding, ....
97 /// </summary>
98 internal static string HelpMessage {
99 get {
100 return ResourceManager.GetString("HelpMessage", resourceCulture);
101 }
102 }
103 }
104}
diff --git a/src/light/LightStrings.resx b/src/light/LightStrings.resx
new file mode 100644
index 00000000..3f586a5d
--- /dev/null
+++ b/src/light/LightStrings.resx
@@ -0,0 +1,174 @@
1<?xml version="1.0" encoding="utf-8"?>
2<root>
3 <!--
4 Microsoft ResX Schema
5
6 Version 2.0
7
8 The primary goals of this format is to allow a simple XML format
9 that is mostly human readable. The generation and parsing of the
10 various data types are done through the TypeConverter classes
11 associated with the data types.
12
13 Example:
14
15 ... ado.net/XML headers & schema ...
16 <resheader name="resmimetype">text/microsoft-resx</resheader>
17 <resheader name="version">2.0</resheader>
18 <resheader name="reader">System.Resources.ResXResourceReader, System.Windows.Forms, ...</resheader>
19 <resheader name="writer">System.Resources.ResXResourceWriter, System.Windows.Forms, ...</resheader>
20 <data name="Name1"><value>this is my long string</value><comment>this is a comment</comment></data>
21 <data name="Color1" type="System.Drawing.Color, System.Drawing">Blue</data>
22 <data name="Bitmap1" mimetype="application/x-microsoft.net.object.binary.base64">
23 <value>[base64 mime encoded serialized .NET Framework object]</value>
24 </data>
25 <data name="Icon1" type="System.Drawing.Icon, System.Drawing" mimetype="application/x-microsoft.net.object.bytearray.base64">
26 <value>[base64 mime encoded string representing a byte array form of the .NET Framework object]</value>
27 <comment>This is a comment</comment>
28 </data>
29
30 There are any number of "resheader" rows that contain simple
31 name/value pairs.
32
33 Each data row contains a name, and value. The row also contains a
34 type or mimetype. Type corresponds to a .NET class that support
35 text/value conversion through the TypeConverter architecture.
36 Classes that don't support this are serialized and stored with the
37 mimetype set.
38
39 The mimetype is used for serialized objects, and tells the
40 ResXResourceReader how to depersist the object. This is currently not
41 extensible. For a given mimetype the value must be set accordingly:
42
43 Note - application/x-microsoft.net.object.binary.base64 is the format
44 that the ResXResourceWriter will generate, however the reader can
45 read any of the formats listed below.
46
47 mimetype: application/x-microsoft.net.object.binary.base64
48 value : The object must be serialized with
49 : System.Runtime.Serialization.Formatters.Binary.BinaryFormatter
50 : and then encoded with base64 encoding.
51
52 mimetype: application/x-microsoft.net.object.soap.base64
53 value : The object must be serialized with
54 : System.Runtime.Serialization.Formatters.Soap.SoapFormatter
55 : and then encoded with base64 encoding.
56
57 mimetype: application/x-microsoft.net.object.bytearray.base64
58 value : The object must be serialized into a byte array
59 : using a System.ComponentModel.TypeConverter
60 : and then encoded with base64 encoding.
61 -->
62 <xsd:schema id="root" xmlns="" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:msdata="urn:schemas-microsoft-com:xml-msdata">
63 <xsd:import namespace="http://www.w3.org/XML/1998/namespace" />
64 <xsd:element name="root" msdata:IsDataSet="true">
65 <xsd:complexType>
66 <xsd:choice maxOccurs="unbounded">
67 <xsd:element name="metadata">
68 <xsd:complexType>
69 <xsd:sequence>
70 <xsd:element name="value" type="xsd:string" minOccurs="0" />
71 </xsd:sequence>
72 <xsd:attribute name="name" use="required" type="xsd:string" />
73 <xsd:attribute name="type" type="xsd:string" />
74 <xsd:attribute name="mimetype" type="xsd:string" />
75 <xsd:attribute ref="xml:space" />
76 </xsd:complexType>
77 </xsd:element>
78 <xsd:element name="assembly">
79 <xsd:complexType>
80 <xsd:attribute name="alias" type="xsd:string" />
81 <xsd:attribute name="name" type="xsd:string" />
82 </xsd:complexType>
83 </xsd:element>
84 <xsd:element name="data">
85 <xsd:complexType>
86 <xsd:sequence>
87 <xsd:element name="value" type="xsd:string" minOccurs="0" msdata:Ordinal="1" />
88 <xsd:element name="comment" type="xsd:string" minOccurs="0" msdata:Ordinal="2" />
89 </xsd:sequence>
90 <xsd:attribute name="name" type="xsd:string" use="required" msdata:Ordinal="1" />
91 <xsd:attribute name="type" type="xsd:string" msdata:Ordinal="3" />
92 <xsd:attribute name="mimetype" type="xsd:string" msdata:Ordinal="4" />
93 <xsd:attribute ref="xml:space" />
94 </xsd:complexType>
95 </xsd:element>
96 <xsd:element name="resheader">
97 <xsd:complexType>
98 <xsd:sequence>
99 <xsd:element name="value" type="xsd:string" minOccurs="0" msdata:Ordinal="1" />
100 </xsd:sequence>
101 <xsd:attribute name="name" type="xsd:string" use="required" />
102 </xsd:complexType>
103 </xsd:element>
104 </xsd:choice>
105 </xsd:complexType>
106 </xsd:element>
107 </xsd:schema>
108 <resheader name="resmimetype">
109 <value>text/microsoft-resx</value>
110 </resheader>
111 <resheader name="version">
112 <value>2.0</value>
113 </resheader>
114 <resheader name="reader">
115 <value>System.Resources.ResXResourceReader, System.Windows.Forms, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
116 </resheader>
117 <resheader name="writer">
118 <value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
119 </resheader>
120 <data name="CommandLineArguments" xml:space="preserve">
121 <value> -b &lt;path&gt; specify a binder path to locate all files
122 (default: current directory)
123 prefix the path with 'name=' where 'name' is the name of your
124 named bindpath.
125 -cc &lt;path&gt; path to cache built cabinets (will not be deleted after linking)
126 -ct &lt;N&gt; number of threads to use when creating cabinets
127 (default: %NUMBER_OF_PROCESSORS%)
128 -cub &lt;file.cub&gt; additional .cub file containing ICEs to run
129 -dcl:level set default cabinet compression level
130 (low, medium, high, none, mszip; mszip default)
131 -eav exact assembly versions (breaks .NET 1.1 RTM compatibility)
132 -ice:&lt;ICE&gt; run a specific internal consistency evaluator (ICE)
133 -pdbout &lt;output.wixpdb&gt; save the WixPdb to a specific file
134 (default: same name as output with wixpdb extension)
135 -reusecab reuse cabinets from cabinet cache
136 -sacl suppress resetting ACLs
137 (useful when laying out image to a network share)
138 -sice:&lt;ICE&gt; suppress an internal consistency evaluator (ICE)
139 -sl suppress layout
140 -spdb suppress outputting the WixPdb
141 -sval suppress MSI/MSM validation
142 -cultures:&lt;cultures&gt; semicolon or comma delimited list of localized
143 string cultures to load from .wxl files and libraries.
144 Precedence of cultures is from left to right.
145 -d&lt;name&gt;[=&lt;value&gt;] define a wix variable, with or without a value.
146 -ext &lt;extension&gt; extension assembly or "class, assembly"
147 -loc &lt;loc.wxl&gt; read localization strings from .wxl file
148 -nologo skip printing light logo information
149 -notidy do not delete temporary files (useful for debugging)
150 -o[ut] specify output file (default: write to current directory)
151 -pedantic show pedantic messages
152 -sloc suppress localization
153 -sw[N] suppress all warnings or a specific message ID
154 (example: -sw1009 -sw1103)
155 -usf &lt;output.xml&gt; unreferenced symbols file
156 -v verbose output
157 -wx[N] treat all warnings or a specific message ID as an error
158 (example: -wx1009 -wx1103)
159 -xo output wixout format instead of MSI format
160 -? | -help this help information</value>
161 </data>
162 <data name="EXP_CannotLinkObjFilesWithOutpuFile" xml:space="preserve">
163 <value>Cannot link object files (.wixobj) files with an output file (.wixout)</value>
164 </data>
165 <data name="HelpMessage" xml:space="preserve">
166 <value> usage: light.exe [-?] [-b bindPath] [-nologo] [-out outputFile] objectFile [objectFile ...] [@responseFile]
167
168{0}
169
170Environment variables:
171 WIX_TEMP overrides the temporary directory used for cab creation, msm exploding, ...</value>
172 <comment>{0} is replaced by a list of light's arguments.</comment>
173 </data>
174</root>
diff --git a/src/light/app.config b/src/light/app.config
new file mode 100644
index 00000000..71c529fb
--- /dev/null
+++ b/src/light/app.config
@@ -0,0 +1,9 @@
1<?xml version="1.0" encoding="utf-8" ?>
2<!-- Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. -->
3
4
5<configuration>
6 <runtime>
7 <loadFromRemoteSources enabled="true"/>
8 </runtime>
9</configuration>
diff --git a/src/light/light.cs b/src/light/light.cs
new file mode 100644
index 00000000..c0967caa
--- /dev/null
+++ b/src/light/light.cs
@@ -0,0 +1,595 @@
1// Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information.
2
3namespace WixToolset.Tools
4{
5 using System;
6 using System.Collections.Generic;
7 using System.Globalization;
8 using System.IO;
9 using System.Linq;
10 using System.Runtime.InteropServices;
11 using System.Text;
12 using System.Threading;
13 using WixToolset.Core;
14 using WixToolset.Data;
15 using WixToolset.Data.Bind;
16 using WixToolset.Extensibility;
17 using WixToolset.Extensibility.Services;
18
19 /// <summary>
20 /// The main entry point for light.
21 /// </summary>
22 public sealed class Light
23 {
24 LightCommandLine commandLine;
25 private IEnumerable<IExtensionData> extensionData;
26 //private IEnumerable<IBinderExtension> binderExtensions;
27 //private IEnumerable<IBinderFileManager> fileManagers;
28
29 /// <summary>
30 /// The main entry point for light.
31 /// </summary>
32 /// <param name="args">Commandline arguments for the application.</param>
33 /// <returns>Returns the application error code.</returns>
34 [MTAThread]
35 public static int Main(string[] args)
36 {
37 var serviceProvider = new WixToolsetServiceProvider();
38
39 var listener = new ConsoleMessageListener("WIX", "light.exe");
40
41 Light light = new Light();
42 return light.Run(serviceProvider, listener, args);
43 }
44
45 /// <summary>
46 /// Main running method for the application.
47 /// </summary>
48 /// <param name="args">Commandline arguments to the application.</param>
49 /// <returns>Returns the application error code.</returns>
50 public int Run(IServiceProvider serviceProvider, IMessageListener listener, string[] args)
51 {
52 var messaging = serviceProvider.GetService<IMessaging>();
53 messaging.SetListener(listener);
54
55 try
56 {
57 var unparsed = this.ParseCommandLineAndLoadExtensions(serviceProvider, messaging, args);
58
59 if (!messaging.EncounteredError)
60 {
61 if (this.commandLine.ShowLogo)
62 {
63 AppCommon.DisplayToolHeader();
64 }
65
66 if (this.commandLine.ShowHelp)
67 {
68 PrintHelp();
69 AppCommon.DisplayToolFooter();
70 }
71 else
72 {
73 foreach (string arg in unparsed)
74 {
75 messaging.Write(WarningMessages.UnsupportedCommandLineArgument(arg));
76 }
77
78 this.Bind(serviceProvider, messaging);
79 }
80 }
81 }
82 catch (WixException we)
83 {
84 messaging.Write(we.Error);
85 }
86 catch (Exception e)
87 {
88 messaging.Write(ErrorMessages.UnexpectedException(e.Message, e.GetType().ToString(), e.StackTrace));
89 if (e is NullReferenceException || e is SEHException)
90 {
91 throw;
92 }
93 }
94
95 return messaging.LastErrorNumber;
96 }
97
98 /// <summary>
99 /// Parse command line and load all the extensions.
100 /// </summary>
101 /// <param name="args">Command line arguments to be parsed.</param>
102 private IEnumerable<string> ParseCommandLineAndLoadExtensions(IServiceProvider serviceProvider, IMessaging messaging, string[] args)
103 {
104 this.commandLine = new LightCommandLine(messaging);
105
106 string[] unprocessed = this.commandLine.Parse(args);
107 if (messaging.EncounteredError)
108 {
109 return unprocessed;
110 }
111
112 // Load extensions.
113 var extensionManager = CreateExtensionManagerWithStandardBackends(serviceProvider);
114 foreach (string extension in this.commandLine.Extensions)
115 {
116 extensionManager.Load(extension);
117 }
118
119 // Extension data command line processing.
120 var context = serviceProvider.GetService<ICommandLineContext>();
121 context.Arguments = null;
122 context.ExtensionManager = extensionManager;
123 context.Messaging = messaging;
124 context.ParsedArguments = args;
125
126 var commandLineExtensions = extensionManager.Create<IExtensionCommandLine>();
127 foreach (var extension in commandLineExtensions)
128 {
129 extension.PreParse(context);
130 }
131
132 // Process unproccessed arguments.
133 List<string> actuallyUnprocessed = new List<string>();
134 foreach (var arg in unprocessed)
135 {
136 if (!this.TryParseCommandLineArgumentWithExtension(arg, commandLineExtensions))
137 {
138 actuallyUnprocessed.Add(arg);
139 }
140 }
141
142 return this.commandLine.ParsePostExtensions(actuallyUnprocessed.ToArray());
143 }
144
145 private void Bind(IServiceProvider serviceProvider, IMessaging messaging)
146 {
147 var output = this.LoadWixout(messaging);
148
149 if (messaging.EncounteredError)
150 {
151 return;
152 }
153
154 var intermediateFolder = this.commandLine.IntermediateFolder;
155 if (String.IsNullOrEmpty(intermediateFolder))
156 {
157 intermediateFolder = Path.Combine(Path.GetTempPath(), Path.GetRandomFileName());
158 }
159
160 var localizations = this.LoadLocalizationFiles(messaging, this.commandLine.LocalizationFiles);
161
162 if (messaging.EncounteredError)
163 {
164 return;
165 }
166
167 ResolveResult resolveResult;
168 {
169 var resolver = new Resolver(serviceProvider);
170 resolver.BindPaths = this.commandLine.BindPaths;
171 resolver.IntermediateFolder = intermediateFolder;
172 resolver.IntermediateRepresentation = output;
173 resolver.Localizations = localizations;
174
175 resolveResult = resolver.Execute();
176 }
177
178 if (messaging.EncounteredError)
179 {
180 return;
181 }
182
183 BindResult bindResult;
184 {
185 var binder = new Binder(serviceProvider);
186 binder.CabbingThreadCount = this.commandLine.CabbingThreadCount;
187 binder.CabCachePath = this.commandLine.CabCachePath;
188 binder.Codepage = resolveResult.Codepage;
189 binder.DefaultCompressionLevel = this.commandLine.DefaultCompressionLevel;
190 binder.DelayedFields = resolveResult.DelayedFields;
191 binder.ExpectedEmbeddedFiles = resolveResult.ExpectedEmbeddedFiles;
192 binder.Ices = this.commandLine.Ices;
193 binder.IntermediateFolder = intermediateFolder;
194 binder.IntermediateRepresentation = resolveResult.IntermediateRepresentation;
195 binder.OutputPath = this.commandLine.OutputFile;
196 binder.OutputPdbPath = Path.ChangeExtension(this.commandLine.OutputFile, ".wixpdb");
197 binder.SuppressIces = this.commandLine.SuppressIces;
198 binder.SuppressValidation = this.commandLine.SuppressValidation;
199
200 bindResult = binder.Execute();
201 }
202
203 if (messaging.EncounteredError)
204 {
205 return;
206 }
207
208 {
209 var layout = new Layout(serviceProvider);
210 layout.FileTransfers = bindResult.FileTransfers;
211 layout.ContentFilePaths = bindResult.ContentFilePaths;
212 layout.ContentsFile = this.commandLine.ContentsFile;
213 layout.OutputsFile = this.commandLine.OutputsFile;
214 layout.BuiltOutputsFile = this.commandLine.BuiltOutputsFile;
215 layout.SuppressAclReset = this.commandLine.SuppressAclReset;
216
217 layout.Execute();
218 }
219 }
220
221 private void Run(IMessaging messaging)
222 {
223#if false
224 // Initialize the variable resolver from the command line.
225 WixVariableResolver wixVariableResolver = new WixVariableResolver();
226 foreach (var wixVar in this.commandLine.Variables)
227 {
228 wixVariableResolver.AddVariable(wixVar.Key, wixVar.Value);
229 }
230
231 // Initialize the linker from the command line.
232 Linker linker = new Linker();
233 linker.UnreferencedSymbolsFile = this.commandLine.UnreferencedSymbolsFile;
234 linker.ShowPedanticMessages = this.commandLine.ShowPedanticMessages;
235 linker.WixVariableResolver = wixVariableResolver;
236
237 foreach (IExtensionData data in this.extensionData)
238 {
239 linker.AddExtensionData(data);
240 }
241
242 // Initialize the binder from the command line.
243 WixToolset.Binder binder = new WixToolset.Binder();
244 binder.CabCachePath = this.commandLine.CabCachePath;
245 binder.ContentsFile = this.commandLine.ContentsFile;
246 binder.BuiltOutputsFile = this.commandLine.BuiltOutputsFile;
247 binder.OutputsFile = this.commandLine.OutputsFile;
248 binder.WixprojectFile = this.commandLine.WixprojectFile;
249 binder.BindPaths.AddRange(this.commandLine.BindPaths);
250 binder.CabbingThreadCount = this.commandLine.CabbingThreadCount;
251 if (this.commandLine.DefaultCompressionLevel.HasValue)
252 {
253 binder.DefaultCompressionLevel = this.commandLine.DefaultCompressionLevel.Value;
254 }
255 binder.Ices.AddRange(this.commandLine.Ices);
256 binder.SuppressIces.AddRange(this.commandLine.SuppressIces);
257 binder.SuppressAclReset = this.commandLine.SuppressAclReset;
258 binder.SuppressLayout = this.commandLine.SuppressLayout;
259 binder.SuppressValidation = this.commandLine.SuppressValidation;
260 binder.PdbFile = this.commandLine.SuppressWixPdb ? null : this.commandLine.PdbFile;
261 binder.TempFilesLocation = AppCommon.GetTempLocation();
262 binder.WixVariableResolver = wixVariableResolver;
263
264 foreach (IBinderExtension extension in this.binderExtensions)
265 {
266 binder.AddExtension(extension);
267 }
268
269 foreach (IBinderFileManager fileManager in this.fileManagers)
270 {
271 binder.AddExtension(fileManager);
272 }
273
274 // Initialize the localizer.
275 Localizer localizer = this.InitializeLocalization(linker.TableDefinitions);
276 if (messaging.EncounteredError)
277 {
278 return;
279 }
280
281 wixVariableResolver.Localizer = localizer;
282 linker.Localizer = localizer;
283 binder.Localizer = localizer;
284
285 // Loop through all the believed object files.
286 List<Section> sections = new List<Section>();
287 Output output = null;
288 foreach (string inputFile in this.commandLine.Files)
289 {
290 string inputFileFullPath = Path.GetFullPath(inputFile);
291 FileFormat format = FileStructure.GuessFileFormatFromExtension(Path.GetExtension(inputFileFullPath));
292 bool retry;
293 do
294 {
295 retry = false;
296
297 try
298 {
299 switch (format)
300 {
301 case FileFormat.Wixobj:
302 Intermediate intermediate = Intermediate.Load(inputFileFullPath, linker.TableDefinitions, this.commandLine.SuppressVersionCheck);
303 sections.AddRange(intermediate.Sections);
304 break;
305
306 case FileFormat.Wixlib:
307 Library library = Library.Load(inputFileFullPath, linker.TableDefinitions, this.commandLine.SuppressVersionCheck);
308 AddLibraryLocalizationsToLocalizer(library, this.commandLine.Cultures, localizer);
309 sections.AddRange(library.Sections);
310 break;
311
312 default:
313 output = Output.Load(inputFileFullPath, this.commandLine.SuppressVersionCheck);
314 break;
315 }
316 }
317 catch (WixUnexpectedFileFormatException e)
318 {
319 format = e.FileFormat;
320 retry = (FileFormat.Wixobj == format || FileFormat.Wixlib == format || FileFormat.Wixout == format); // .wixobj, .wixout and .wixout are supported by light.
321 if (!retry)
322 {
323 messaging.OnMessage(e.Error);
324 }
325 }
326 } while (retry);
327 }
328
329 // Stop processing if any errors were found loading object files.
330 if (messaging.EncounteredError)
331 {
332 return;
333 }
334
335 // and now for the fun part
336 if (null == output)
337 {
338 OutputType expectedOutputType = OutputType.Unknown;
339 if (!String.IsNullOrEmpty(this.commandLine.OutputFile))
340 {
341 expectedOutputType = Output.GetOutputType(Path.GetExtension(this.commandLine.OutputFile));
342 }
343
344 output = linker.Link(sections, expectedOutputType);
345
346 // If an error occurred during linking, stop processing.
347 if (null == output)
348 {
349 return;
350 }
351 }
352 else if (0 != sections.Count)
353 {
354 throw new InvalidOperationException(LightStrings.EXP_CannotLinkObjFilesWithOutpuFile);
355 }
356
357 bool tidy = true; // clean up after ourselves by default.
358 try
359 {
360 // only output the xml if its a patch build or user specfied to only output wixout
361 string outputFile = this.commandLine.OutputFile;
362 string outputExtension = Path.GetExtension(outputFile);
363 if (this.commandLine.OutputXml || OutputType.Patch == output.Type)
364 {
365 if (String.IsNullOrEmpty(outputExtension) || outputExtension.Equals(".wix", StringComparison.Ordinal))
366 {
367 outputExtension = (OutputType.Patch == output.Type) ? ".wixmsp" : ".wixout";
368 outputFile = Path.ChangeExtension(outputFile, outputExtension);
369 }
370
371 output.Save(outputFile);
372 }
373 else // finish creating the MSI/MSM
374 {
375 if (String.IsNullOrEmpty(outputExtension) || outputExtension.Equals(".wix", StringComparison.Ordinal))
376 {
377 outputExtension = Output.GetExtension(output.Type);
378 outputFile = Path.ChangeExtension(outputFile, outputExtension);
379 }
380
381 binder.Bind(output, outputFile);
382 }
383 }
384 catch (WixException we) // keep files around for debugging IDT issues.
385 {
386 if (we is WixInvalidIdtException)
387 {
388 tidy = false;
389 }
390
391 throw;
392 }
393 catch (Exception) // keep files around for debugging unexpected exceptions.
394 {
395 tidy = false;
396 throw;
397 }
398 finally
399 {
400 if (null != binder)
401 {
402 binder.Cleanup(tidy);
403 }
404 }
405
406 return;
407#endif
408 }
409
410#if false
411 private Localizer InitializeLocalization(TableDefinitionCollection tableDefinitions)
412 {
413 Localizer localizer = null;
414
415 // Instantiate the localizer and load any localization files.
416 if (!this.commandLine.SuppressLocalization || 0 < this.commandLine.LocalizationFiles.Count || null != this.commandLine.Cultures || !this.commandLine.OutputXml)
417 {
418 List<Localization> localizations = new List<Localization>();
419
420 // Load each localization file.
421 foreach (string localizationFile in this.commandLine.LocalizationFiles)
422 {
423 Localization localization = Localizer.ParseLocalizationFile(localizationFile, tableDefinitions);
424 if (null != localization)
425 {
426 localizations.Add(localization);
427 }
428 }
429
430 localizer = new Localizer();
431 if (null != this.commandLine.Cultures)
432 {
433 // Alocalizations in order specified in cultures.
434 foreach (string culture in this.commandLine.Cultures)
435 {
436 foreach (Localization localization in localizations)
437 {
438 if (culture.Equals(localization.Culture, StringComparison.OrdinalIgnoreCase))
439 {
440 localizer.AddLocalization(localization);
441 }
442 }
443 }
444 }
445 else // no cultures specified, so try neutral culture and if none of those add all loc files.
446 {
447 bool neutralFound = false;
448 foreach (Localization localization in localizations)
449 {
450 if (String.IsNullOrEmpty(localization.Culture))
451 {
452 // If a neutral wxl was provided use it.
453 localizer.AddLocalization(localization);
454 neutralFound = true;
455 }
456 }
457
458 if (!neutralFound)
459 {
460 // No cultures were specified and no neutral wxl are available, include all of the loc files.
461 foreach (Localization localization in localizations)
462 {
463 localizer.AddLocalization(localization);
464 }
465 }
466 }
467
468 // Load localizations provided by extensions with data.
469 foreach (IExtensionData data in this.extensionData)
470 {
471 Library library = data.GetLibrary(tableDefinitions);
472 if (null != library)
473 {
474 // Load the extension's default culture if it provides one and no cultures were specified.
475 string[] extensionCultures = this.commandLine.Cultures;
476 if (null == extensionCultures && null != data.DefaultCulture)
477 {
478 extensionCultures = new string[] { data.DefaultCulture };
479 }
480
481 AddLibraryLocalizationsToLocalizer(library, extensionCultures, localizer);
482 }
483 }
484 }
485
486 return localizer;
487 }
488
489 private void AddLibraryLocalizationsToLocalizer(Library library, string[] cultures, Localizer localizer)
490 {
491 foreach (Localization localization in library.GetLocalizations(cultures))
492 {
493 localizer.AddLocalization(localization);
494 }
495 }
496#endif
497
498 private bool TryParseCommandLineArgumentWithExtension(string arg, IEnumerable<IExtensionCommandLine> extensions)
499 {
500 foreach (var extension in extensions)
501 {
502 // TODO: decide what to do with "IParseCommandLine" argument.
503 if (extension.TryParseArgument(null, arg))
504 {
505 return true;
506 }
507 }
508
509 return false;
510 }
511
512 private IEnumerable<Localization> LoadLocalizationFiles(IMessaging messaging, IEnumerable<string> locFiles)
513 {
514 foreach (var loc in locFiles)
515 {
516 var localization = Localizer.ParseLocalizationFile(messaging, loc);
517
518 yield return localization;
519 }
520 }
521
522 private Intermediate LoadWixout(IMessaging messaging)
523 {
524 var path = this.commandLine.Files.Single();
525
526 return Intermediate.Load(path);
527 }
528
529 private static IExtensionManager CreateExtensionManagerWithStandardBackends(IServiceProvider serviceProvider)
530 {
531 var extensionManager = serviceProvider.GetService<IExtensionManager>();
532
533 foreach (var type in new[] { typeof(WixToolset.Core.Burn.WixToolsetStandardBackend), typeof(WixToolset.Core.WindowsInstaller.WixToolsetStandardBackend) })
534 {
535 extensionManager.Add(type.Assembly);
536 }
537
538 return extensionManager;
539 }
540
541 private static void PrintHelp()
542 {
543 string lightArgs = LightStrings.CommandLineArguments;
544
545 Console.WriteLine(String.Format(LightStrings.HelpMessage, lightArgs));
546 }
547
548 private class ConsoleMessageListener : IMessageListener
549 {
550 public ConsoleMessageListener(string shortName, string longName)
551 {
552 this.ShortAppName = shortName;
553 this.LongAppName = longName;
554
555 PrepareConsoleForLocalization();
556 }
557
558 public string LongAppName { get; }
559
560 public string ShortAppName { get; }
561
562 public void Write(Message message)
563 {
564 var filename = message.SourceLineNumbers?.FileName ?? this.LongAppName;
565 var line = message.SourceLineNumbers?.LineNumber ?? -1;
566 var type = message.Level.ToString().ToLowerInvariant();
567 var output = message.Level >= MessageLevel.Warning ? Console.Out : Console.Error;
568
569 if (line > 0)
570 {
571 filename = String.Concat(filename, "(", line, ")");
572 }
573
574 output.WriteLine("{0} : {1} {2}{3:0000}: {4}", filename, type, this.ShortAppName, message.Id, message.ToString());
575 }
576
577 public void Write(string message)
578 {
579 Console.Out.WriteLine(message);
580 }
581
582 private static void PrepareConsoleForLocalization()
583 {
584 Thread.CurrentThread.CurrentUICulture = CultureInfo.CurrentUICulture.GetConsoleFallbackUICulture();
585
586 if (Console.OutputEncoding.CodePage != Encoding.UTF8.CodePage &&
587 Console.OutputEncoding.CodePage != Thread.CurrentThread.CurrentUICulture.TextInfo.OEMCodePage &&
588 Console.OutputEncoding.CodePage != Thread.CurrentThread.CurrentUICulture.TextInfo.ANSICodePage)
589 {
590 Thread.CurrentThread.CurrentUICulture = new CultureInfo("en-US");
591 }
592 }
593 }
594 }
595}
diff --git a/src/light/light.csproj b/src/light/light.csproj
new file mode 100644
index 00000000..20e10b11
--- /dev/null
+++ b/src/light/light.csproj
@@ -0,0 +1,21 @@
1<?xml version="1.0" encoding="utf-8"?>
2<!-- Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. -->
3
4<Project Sdk="Microsoft.NET.Sdk">
5 <PropertyGroup>
6 <TargetFramework>net461</TargetFramework>
7 <OutputType>Exe</OutputType>
8 <Description>Linker</Description>
9 <Title>WiX Toolset Linker</Title>
10 </PropertyGroup>
11
12 <ItemGroup>
13 <ProjectReference Include="..\WixToolset.Core\WixToolset.Core.csproj" />
14 <ProjectReference Include="..\WixToolset.Core.Burn\WixToolset.Core.Burn.csproj" />
15 <ProjectReference Include="..\WixToolset.Core.WindowsInstaller\WixToolset.Core.WindowsInstaller.csproj" />
16 </ItemGroup>
17
18 <ItemGroup>
19 <PackageReference Include="Nerdbank.GitVersioning" Version="2.0.41" PrivateAssets="all" />
20 </ItemGroup>
21</Project>