2 // MSBuild.cs: Task that can run .*proj files
5 // Marek Sieradzki (marek.sieradzki@gmail.com)
7 // (C) 2005 Marek Sieradzki
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:
17 // The above copyright notice and this permission notice shall be
18 // included in all copies or substantial portions of the Software.
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.
31 using System.Collections;
32 using System.Collections.Generic;
34 using Microsoft.Build.BuildEngine;
35 using Microsoft.Build.Framework;
36 using Microsoft.Build.Utilities;
38 namespace Microsoft.Build.Tasks {
41 public class MSBuild : TaskExtension {
43 ITaskItem [] projects;
46 bool runEachTargetSeparately;
47 bool stopOnFirstFailure;
49 ITaskItem [] targetOutputs;
56 public override bool Execute ()
58 if (projects.Length == 0)
63 stopOnFirstFailure = false;
64 List <ITaskItem > outputItems = new List <ITaskItem> ();
65 string currentDirectory = Environment.CurrentDirectory;
68 var global_properties = SplitPropertiesToDictionary ();
70 Log.LogMessage (MessageImportance.Low, "Global Properties:");
71 if (global_properties != null)
72 foreach (KeyValuePair<string, string> pair in global_properties)
73 Log.LogMessage (MessageImportance.Low, "\t{0} = {1}", pair.Key, pair.Value);
75 foreach (ITaskItem project in projects) {
76 filename = project.GetMetadata ("FullPath");
77 if (!File.Exists (filename)) {
78 Log.LogError ("Could not find the project file '{0}'", filename);
79 if (stopOnFirstFailure)
85 Directory.SetCurrentDirectory (Path.GetDirectoryName (filename));
86 outputs = new Hashtable ();
89 // Order of precedence:
90 // ToolsVersion property, %(Project.ToolsVersion)
91 string tv = ToolsVersion;
92 if (String.IsNullOrEmpty (tv))
93 // metadata on the Project item
94 tv = project.GetMetadata ("ToolsVersion");
96 if (!String.IsNullOrEmpty (tv) && Engine.GlobalEngine.Toolsets [tv] == null)
97 throw new UnknownToolsVersionException (tv);
99 result = BuildEngine2.BuildProjectFile (filename, targets, global_properties, outputs, tv);
100 } catch (InvalidProjectFileException e) {
101 Log.LogError ("Error building project {0}: {1}", filename, e.Message);
106 foreach (DictionaryEntry de in outputs) {
107 ITaskItem [] array = (ITaskItem []) de.Value;
108 foreach (ITaskItem item in array) {
110 ITaskItem new_item = new TaskItem (item);
112 // copy the metadata from original @project to here
113 // CopyMetadataTo does _not_ overwrite
114 project.CopyMetadataTo (new_item);
116 outputItems.Add (new_item);
118 //FIXME: Correctly rebase output paths to be relative to the
121 // File.Copy (item.ItemSpec, Path.Combine (currentDirectory, item.ItemSpec), true);
125 if (stopOnFirstFailure)
129 Directory.SetCurrentDirectory (currentDirectory);
133 targetOutputs = outputItems.ToArray ();
135 Directory.SetCurrentDirectory (currentDirectory);
139 void ThrowIfInvalidToolsVersion (string toolsVersion)
141 if (!String.IsNullOrEmpty (toolsVersion) && Engine.GlobalEngine.Toolsets [toolsVersion] == null)
142 throw new UnknownToolsVersionException (toolsVersion);
146 public ITaskItem [] Projects {
147 get { return projects; }
148 set { projects = value; }
152 public string [] Properties {
153 get { return properties; }
154 set { properties = value; }
157 public bool RebaseOutputs {
158 get { return rebaseOutputs; }
159 set { rebaseOutputs = value; }
163 public bool RunEachTargetSeparately {
164 get { return runEachTargetSeparately; }
165 set { runEachTargetSeparately = value; }
168 public bool StopOnFirstFailure {
169 get { return stopOnFirstFailure; }
170 set { stopOnFirstFailure = value; }
174 public ITaskItem [] TargetOutputs {
175 get { return targetOutputs; }
178 public string [] Targets {
179 get { return targets; }
180 set { targets = value; }
183 public bool BuildInParallel {
184 get { return buildInParallel; }
185 set { buildInParallel = value; }
188 public string ToolsVersion {
192 SortedDictionary<string, string> SplitPropertiesToDictionary ()
194 if (properties == null)
197 var global_properties = new SortedDictionary<string, string> ();
198 foreach (string kvpair in properties) {
199 if (String.IsNullOrEmpty (kvpair))
202 string [] parts = kvpair.Trim ().Split (new char [] {'='}, 2);
203 if (parts.Length != 2) {
204 Log.LogWarning ("Invalid key/value pairs ({0}) in Properties, ignoring.", kvpair);
208 global_properties.Add (parts [0], parts [1]);
211 return global_properties;