// 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. namespace WixToolset.Tools { using System; using System.Collections.Generic; using System.Globalization; using System.IO; using WixToolset.Data; using WixToolset.Extensibility; using WixToolset.Extensibility.Services; public class LightCommandLine { public LightCommandLine(IMessaging messaging) { this.Messaging = messaging; this.ShowLogo = true; this.Tidy = true; this.CubeFiles = new List(); this.SuppressIces = new List(); this.Ices = new List(); this.BindPaths = new List(); this.Files = new List(); this.LocalizationFiles = new List(); this.Variables = new Dictionary(); } public IMessaging Messaging { get; } public string PdbFile { get; private set; } public CompressionLevel? DefaultCompressionLevel { get; set; } public bool SuppressAclReset { get; private set; } public bool SuppressLayout { get; private set; } public bool SuppressWixPdb { get; private set; } public bool SuppressValidation { get; private set; } public string IntermediateFolder { get; private set; } public string OutputsFile { get; private set; } public string BuiltOutputsFile { get; private set; } public string WixprojectFile { get; private set; } public string ContentsFile { get; private set; } public List Ices { get; private set; } public string CabCachePath { get; private set; } public int CabbingThreadCount { get; private set; } public List CubeFiles { get; private set; } public List SuppressIces { get; private set; } public bool ShowLogo { get; private set; } public bool ShowHelp { get; private set; } public bool ShowPedanticMessages { get; private set; } public bool SuppressLocalization { get; private set; } public bool SuppressVersionCheck { get; private set; } public string[] Cultures { get; private set; } public string OutputFile { get; private set; } public bool OutputXml { get; private set; } public List BindPaths { get; private set; } public List Files { get; private set; } public List LocalizationFiles { get; private set; } public bool Tidy { get; private set; } public string UnreferencedSymbolsFile { get; private set; } public IDictionary Variables { get; private set; } /// /// Parse the commandline arguments. /// /// Commandline arguments. public string[] Parse(ICommandLineContext context) { var unprocessed = new List(); var extensions = context.ExtensionManager.Create(); foreach (var extension in extensions) { extension.PreParse(context); } var parser = context.Arguments.Parse(); while (!this.ShowHelp && String.IsNullOrEmpty(parser.ErrorArgument) && parser.TryGetNextSwitchOrArgument(out var arg)) { if (String.IsNullOrWhiteSpace(arg)) // skip blank arguments. { continue; } if (parser.IsSwitch(arg)) { var parameter = arg.Substring(1); if (parameter.Equals("b", StringComparison.Ordinal)) { var result = parser.GetNextArgumentOrError(arg); if (!String.IsNullOrEmpty(result)) { var bindPath = BindPath.Parse(result); this.BindPaths.Add(bindPath); } } else if (parameter.StartsWith("cultures:", StringComparison.Ordinal)) { string culturesString = arg.Substring(10).ToLower(CultureInfo.InvariantCulture); // When null is used treat it as if cultures wasn't specified. // This is needed for batching over the light task when using MSBuild which doesn't // support empty items if (culturesString.Equals("null", StringComparison.OrdinalIgnoreCase)) { this.Cultures = null; } else { this.Cultures = culturesString.Split(';', ','); for (int c = 0; c < this.Cultures.Length; ++c) { // Neutral is different from null. For neutral we still want to do WXL filtering. // Set the culture to the empty string = identifier for the invariant culture if (this.Cultures[c].Equals("neutral", StringComparison.OrdinalIgnoreCase)) { this.Cultures[c] = String.Empty; } } } } else if (parameter.StartsWith("dcl:", StringComparison.Ordinal)) { string defaultCompressionLevel = arg.Substring(5); if (String.IsNullOrEmpty(defaultCompressionLevel)) { break; } else if (Enum.TryParse(defaultCompressionLevel, true, out CompressionLevel compressionLevel)) { this.DefaultCompressionLevel = compressionLevel; } } else if (parameter.StartsWith("d", StringComparison.Ordinal)) { parameter = arg.Substring(2); string[] value = parameter.Split("=".ToCharArray(), 2); string preexisting; if (1 == value.Length) { this.Messaging.Write(ErrorMessages.ExpectedWixVariableValue(value[0])); } else if (this.Variables.TryGetValue(value[0], out preexisting)) { this.Messaging.Write(ErrorMessages.WixVariableCollision(null, value[0])); } else { this.Variables.Add(value[0], value[1]); } } else if (parameter.Equals("loc", StringComparison.Ordinal)) { parser.GetNextArgumentAsFilePathOrError(arg, "localization files", this.LocalizationFiles); } else if (parameter.Equals("nologo", StringComparison.Ordinal)) { this.ShowLogo = false; } else if (parameter.Equals("notidy", StringComparison.Ordinal)) { this.Tidy = false; } else if ("o" == parameter || "out" == parameter) { this.OutputFile = parser.GetNextArgumentAsFilePathOrError(arg); } else if (parameter.Equals("pedantic", StringComparison.Ordinal)) { this.ShowPedanticMessages = true; } else if (parameter.Equals("sloc", StringComparison.Ordinal)) { this.SuppressLocalization = true; } else if (parameter.Equals("usf", StringComparison.Ordinal)) { this.UnreferencedSymbolsFile = parser.GetNextArgumentAsDirectoryOrError(arg); } else if (parameter.Equals("xo", StringComparison.Ordinal)) { this.OutputXml = true; } else if (parameter.Equals("cc", StringComparison.Ordinal)) { this.CabCachePath = parser.GetNextArgumentAsDirectoryOrError(arg); } else if (parameter.Equals("ct", StringComparison.Ordinal)) { var result = parser.GetNextArgumentOrError(arg); if (!String.IsNullOrEmpty(result)) { if (!Int32.TryParse(result, out var ct) || 0 >= ct) { this.Messaging.Write(ErrorMessages.IllegalCabbingThreadCount(result)); parser.ErrorArgument = arg; break; } this.CabbingThreadCount = ct; this.Messaging.Write(VerboseMessages.SetCabbingThreadCount(this.CabbingThreadCount.ToString())); } } else if (parameter.Equals("cub", StringComparison.Ordinal)) { parser.GetNextArgumentAsFilePathOrError(arg, "static validation files", this.CubeFiles); } else if (parameter.StartsWith("ice:", StringComparison.Ordinal)) { this.Ices.Add(parameter.Substring(4)); } else if (parameter.Equals("intermediatefolder", StringComparison.OrdinalIgnoreCase)) { this.IntermediateFolder = parser.GetNextArgumentAsDirectoryOrError(arg); } else if (parameter.Equals("contentsfile", StringComparison.Ordinal)) { this.ContentsFile = parser.GetNextArgumentAsFilePathOrError(arg); } else if (parameter.Equals("outputsfile", StringComparison.Ordinal)) { this.OutputsFile = parser.GetNextArgumentAsFilePathOrError(arg); } else if (parameter.Equals("builtoutputsfile", StringComparison.Ordinal)) { this.BuiltOutputsFile = parser.GetNextArgumentAsFilePathOrError(arg); } else if (parameter.Equals("wixprojectfile", StringComparison.Ordinal)) { this.WixprojectFile = parser.GetNextArgumentAsFilePathOrError(arg); } else if (parameter.Equals("pdbout", StringComparison.Ordinal)) { this.PdbFile = parser.GetNextArgumentAsFilePathOrError(arg); } else if (parameter.StartsWith("sice:", StringComparison.Ordinal)) { this.SuppressIces.Add(parameter.Substring(5)); } else if (parameter.Equals("sl", StringComparison.Ordinal)) { this.SuppressLayout = true; } else if (parameter.Equals("spdb", StringComparison.Ordinal)) { this.SuppressWixPdb = true; } else if (parameter.Equals("sacl", StringComparison.Ordinal)) { this.SuppressAclReset = true; } else if (parameter.Equals("sval", StringComparison.Ordinal)) { this.SuppressValidation = true; } else if ("sv" == parameter) { this.SuppressVersionCheck = true; } else if (parameter.StartsWith("sw", StringComparison.Ordinal)) { string paramArg = parameter.Substring(2); if (0 == paramArg.Length) { this.Messaging.SuppressAllWarnings = true; } else { int suppressWarning = 0; if (!Int32.TryParse(paramArg, out suppressWarning) || 0 >= suppressWarning) { this.Messaging.Write(ErrorMessages.IllegalSuppressWarningId(paramArg)); } else { this.Messaging.SuppressWarningMessage(suppressWarning); } } } else if (parameter.StartsWith("wx", StringComparison.Ordinal)) { string paramArg = parameter.Substring(2); if (0 == paramArg.Length) { this.Messaging.WarningsAsError = true; } else { int elevateWarning = 0; if (!Int32.TryParse(paramArg, out elevateWarning) || 0 >= elevateWarning) { this.Messaging.Write(ErrorMessages.IllegalWarningIdAsError(paramArg)); } else { this.Messaging.ElevateWarningMessage(elevateWarning); } } } else if ("v" == parameter) { this.Messaging.ShowVerboseMessages = true; } else if ("?" == parameter || "help" == parameter) { this.ShowHelp = true; break; } else if (!this.TryParseCommandLineArgumentWithExtension(arg, parser, extensions)) { unprocessed.Add(arg); } } else if (!this.TryParseCommandLineArgumentWithExtension(arg, parser, extensions)) { unprocessed.Add(arg); } } return this.ParsePostExtensions(parser, unprocessed.ToArray()); } private string[] ParsePostExtensions(IParseCommandLine parser, string[] remaining) { var unprocessed = new List(); for (int i = 0; i < remaining.Length; ++i) { var arg = remaining[i]; if (parser.IsSwitch(arg)) { unprocessed.Add(arg); } else { parser.GetArgumentAsFilePathOrError(arg, "source files", this.Files); } } if (0 == this.Files.Count) { this.ShowHelp = true; } else if (String.IsNullOrEmpty(this.OutputFile)) { if (1 < this.Files.Count) { this.Messaging.Write(ErrorMessages.MustSpecifyOutputWithMoreThanOneInput()); } // After the linker tells us what the output type actually is, we'll change the ".wix" to the correct extension. this.OutputFile = Path.ChangeExtension(Path.GetFileName(this.Files[0]), ".wix"); // Add the directories of the input files as unnamed bind paths. foreach (string file in this.Files) { var bindPath = new BindPath(Path.GetDirectoryName(Path.GetFullPath(file))); this.BindPaths.Add(bindPath); } } if (!this.SuppressWixPdb && String.IsNullOrEmpty(this.PdbFile) && !String.IsNullOrEmpty(this.OutputFile)) { this.PdbFile = Path.ChangeExtension(this.OutputFile, ".wixpdb"); } return unprocessed.ToArray(); } private bool TryParseCommandLineArgumentWithExtension(string arg, IParseCommandLine parser, IEnumerable extensions) { foreach (var extension in extensions) { if (extension.TryParseArgument(parser, arg)) { return true; } } return false; } } }