我尝试在Dictionary
个任务之间发送/分享Hashtable
或MSBuild
。
我有以下两个自定义任务,Get
生成一个Hashtable和Set
应该使用它。
Get.cs
public class Get : Task
{
[Output]
public Hashtable Output { get; set; }
public override bool Execute()
{
Output = new Hashtable();
return true;
}
}
Set.cs
public class Set : Task
{
[Required]
public Hashtable Output { get; set; }
public override bool Execute()
{
var items = Output.Cast<DictionaryEntry>().ToDictionary(d => d.Key.ToString(), d => d.Value.ToString());
foreach(var item in items)
{
//Do Something
}
return true;
}
}
上述类可以很好地构建到Assembly.dll
然后我在以下构建目标脚本中使用Assembly.dll
来调用Get和Set自定义任务:
MyTarget.targets
<?xml version="1.0" encoding="utf-8"?>
<Project xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
<UsingTask TaskName="Get" AssemblyFile=".\Assembly.dll"/>
<UsingTask TaskName="Set" AssemblyFile=".\Assembly.dll"/>
<Target Name="Get">
<Get>
<Output TaskParameter="Output" ItemName="Output" />
</Get>
<Set Output=@(Output) />
</Target>
</Project>
当我使用上述目标MSBuild构建项目时,显示以下错误:
&#34; System.Collections.Hashtable&#34; &#34;输出&#34;的类型&#34; Get&#34;的参数MSBuild不支持任务
如何在属性中使用Hashtable或Dictionary来执行自定义MSBuild任务?
答案 0 :(得分:3)
可以进出任务的参数仅限于ITaskItem
或ITaskItem
的数组。
所以你的属性应该从
改变public Hashtable Output { get; set; }
到
public ITaskItem[] Output { get; set; }
以匹配该要求。
接下来,您需要一个实现ITaskItem
的实现类。这允许您处理您的哈希集或字典。我为你添加了添加,但最小的KeyValue类看起来像这样:
public class KeyValue: ITaskItem
{
string _spec = String.Empty;
public KeyValue(string key, string value)
{
_spec = key;
metadata.Add("value", value);
}
Dictionary<string,string> metadata = new Dictionary<string,string>();
public string ItemSpec
{
get {return _spec;}
set {}
}
public ICollection MetadataNames
{
get {return metadata.Keys;}
}
public int MetadataCount
{
get {return metadata.Keys.Count;}
}
public string GetMetadata(string metadataName)
{
return metadata[metadataName];
}
public void SetMetadata(string metadataName, string metadataValue)
{
metadata[metadataName] = metadataValue;
}
public void RemoveMetadata(string metadataName)
{
}
public void CopyMetadataTo(ITaskItem destinationItem)
{
}
public IDictionary CloneCustomMetadata()
{
return metadata;
}
}
如果它是在平面MSBuild脚本中完成的,那么这个类将生成和看起来如此的项目:
<Item Include="key">
<value>some value</value>
</Item>
接下来,您可以调整Set和Get Task来使用这个新类KeyValue:
public class Set : Task
{
TaskLoggingHelper log;
public Set() {
log = new TaskLoggingHelper(this);
}
[Required]
public ITaskItem[] Output { get; set; }
public override bool Execute()
{
log.LogMessage("start set");
foreach(var item in Output)
{
log.LogMessage(String.Format("Set sees key {0} with value {1}.",item.ItemSpec, item.GetMetadata("value")));
}
log.LogMessage("end set");
return true;
}
}
public class Get : Task
{
// notice this property no longer is called Output
// as that gave me errors as the property is reserved
[Output]
public ITaskItem[] Result { get; set; }
public override bool Execute()
{
// convert a Dictionary or Hashset to an array of ITaskItems
// by creating instances of the class KeyValue.
// I use a simple list here, I leave it as an exercise to do the other colletions
Result = new List<ITaskItem> { new KeyValue("bar", "bar-val"), new KeyValue("foo","foo val") }.ToArray();
return true;
}
}
我用来测试上面代码的构建文件:
<Project xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
<UsingTask TaskName="Get" AssemblyFile=".\cb.dll"/>
<UsingTask TaskName="Set" AssemblyFile=".\cb.dll"/>
<Target Name="Get">
<Get>
<Output TaskParameter="Result" ItemName="GetResult" />
</Get>
<!-- lets see what we've got -->
<Message Importance="high" Text="key: @(GetResult) :: value: %(value)" />
<Set Output="@(GetResult)">
</Set>
</Target>
</Project>
运行时结果为:
Build started 24-12-2017 21:26:17.
Project "C:\Prj\bld\test.build" on node 1 (default targets).
Get:
key: bar :: value: bar-val
key: foo :: value: foo val
start set
Set sees key bar with value bar-val.
Set sees key foo with value foo val.
end set
Done Building Project "C:\Prj\bld\test.build" (default targets).
Build succeeded.
0 Warning(s)
0 Error(s)