2012-03-06 1 views
7
public static string GetXml(Dictionary<string, object> parameters) 
{ 
    return XamlWriter.Save(parameters); 
} 

выше оператор возвращает NotSupportedException.TFS 2010: Почему это не представляется возможным десериализации словарь <строка, объект> с XamlWriter.Save, когда я могу использовать XamlReader для десериализации

Странно, что я могу использовать XamlReader для сериализации словаря.

public static Dictionary<string, object> GetParameters(IBuildDetail buildDetail, string buildDefinition) 
{ 
    var tfsProject = buildDetail.BuildDefinition.TeamProject; 
    var buildServer = buildDetail.BuildServer; 
    var buildDef = buildServer.GetBuildDefinition(tfsProject, buildDefinition); 
    using (var stringReader = new StringReader(buildDef.ProcessParameters)) 
    { 
     using (var xmlTextReader = new XmlTextReader(stringReader)) 
     { 
      return (Dictionary<string, object>) XamlReader.Load(xmlTextReader); 
     } 
    } 
} 

XML-:

<?xml version="1.0" ?> 
<Dictionary x:TypeArguments="x:String, x:Object" xmlns="clr-namespace:System.Collections.Generic;assembly=mscorlib" xmlns:mtbwa="clr-namespace:Microsoft.TeamFoundation.Build.Workflow.Activities;assembly=Microsoft.TeamFoundation.Build.Workflow" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"> 

    <mtbwa:BuildSettings ProjectsToBuild="$/XXX/Product/Company.XXX.Common/Company.XXX.Common.Basic/Company.XXX.Common.Basic.csproj" x:Key="BuildSettings"> 

    <mtbwa:BuildSettings.PlatformConfigurations> 

     <mtbwa:PlatformConfigurationList Capacity="4"> 

     <mtbwa:PlatformConfiguration Configuration="DEMO" Platform="AnyCPU"/> 

     <mtbwa:PlatformConfiguration Configuration="Release" Platform="AnyCPU"/> 

     </mtbwa:PlatformConfigurationList> 

    </mtbwa:BuildSettings.PlatformConfigurations> 

    </mtbwa:BuildSettings> 

    <mtbwa:TestSpecList Capacity="0" x:Key="TestSpecs"/> 

    <x:String x:Key="BuildNumberFormat"> 
    $(BuildDefinitionName) 6.0.0$(Rev:.r) 
    </x:String> 

    <mtbwa:CodeAnalysisOption x:Key="RunCodeAnalysis"> 
    Never 
    </mtbwa:CodeAnalysisOption> 

    <mtbwa:AgentSettings MaxWaitTime="00:15:00" TagComparison="MatchExactly" Tags="" x:Key="AgentSettings"/> 

    <x:Boolean x:Key="AssociateChangesetsAndWorkItems"> 
    False 
    </x:Boolean> 

    <x:Boolean x:Key="CreateWorkItem"> 
    False 
    </x:Boolean> 

    <x:Boolean x:Key="PerformTestImpactAnalysis"> 
    False 
    </x:Boolean> 

    <x:Boolean x:Key="CreateLabel"> 
    False 
    </x:Boolean> 

    <x:Boolean x:Key="DisableTests"> 
    True 
    </x:Boolean> 

    <x:Boolean x:Key="DoCheckinAssemblyInfoFiles"> 
    True 
    </x:Boolean> 

    <x:String x:Key="AssemblyVersionPattern"> 
    6.0.0.0 
    </x:String> 

    <x:String x:Key="AssemblyFileVersionPattern"> 
    6.0.0.B 
    </x:String> 

    <x:Boolean x:Key="UseObfuscation"> 
    True 
    </x:Boolean> 

    <x:String x:Key="ObfuscatorFilePath"> 
    C:\Program Files (x86)\LogicNP Software\Crypto Obfuscator For .Net 2011 R3\co.exe 
    </x:String> 

    <x:String x:Key="ObfuscatorProjectFile"> 
    $/XXX/Product/BuildProcess/Company.XXX.ZZZ.obproj 
    </x:String> 

    <x:String x:Key="ProjectPath"> 
    $/XXX/Product/Company.XXX.Common 
    </x:String> 

</Dictionary> 

ответ

2

Я использовал JetBrains dotPeek, чтобы узнать, как Microsoft является сериализации и десериализации Dictionary<string, object> для TFS 2010 Workflow.

Для десериализации они используют:Microsoft.TeamFoundation.Build.Workflow.WorkflowHelpers.DeserializeProcessParameters(string parameterValues)

Для Сериализация они используют:WorkflowHelpers.SerializeProcessParameters(IDictionary<string, object> parameterValues)

Монтаж: Microsoft.TeamFoundation.Build.Workflow.dll

Моя первоначальная проблема заключалась в том, что Я попытался добавить значения IBuildRequest.ProcessParameters перед очередью сборки.

private static void SetProcessParametersForSubBuild(IBuildRequest buildRequest, Dictionary<string, object> processParametersForSubBuild, IBuildDefinition buildDefinition) 
{ 
    var subBuildProcessParameters = WorkflowHelpers.DeserializeProcessParameters(buildDefinition.ProcessParameters); 

    if (processParametersForSubBuild.Any()) 
    { 
     foreach (var processParameter in processParametersForSubBuild) 
     { 
      if (subBuildProcessParameters.ContainsKey(processParameter.Key)) 
      { 
       subBuildProcessParameters[processParameter.Key] = processParameter.Value; 
      } 
      else 
      { 
       subBuildProcessParameters.Add(processParameter.Key, processParameter.Value); 
      } 
     } 
     buildRequest.ProcessParameters = WorkflowHelpers.SerializeProcessParameters(subBuildProcessParameters); 
    } 
} 

Очередь сборки:

вар queuedBuild = buildServer.QueueBuild (buildRequest);

1

это необходимость использовать XamlReader/XamlWriter? Если нет - посмотрите на DataContractSerializer:

public static class SerializationExtensions 
{ 
    public static string Serialize<T>(this T obj) 
    { 
     var serializer = new DataContractSerializer(obj.GetType()); 
     using (var writer = new StringWriter()) 
     using (var stm = new XmlTextWriter(writer)) 
     { 
      serializer.WriteObject(stm, obj); 
      return writer.ToString(); 
     } 
    } 
    public static T Deserialize<T>(this string serialized) 
    { 
     var serializer = new DataContractSerializer(typeof(T)); 
     using (var reader = new StringReader(serialized)) 
     using (var stm = new XmlTextReader(reader)) 
     { 
      return (T)serializer.ReadObject(stm); 
     } 
    } 
} 

(кредиты https://stackoverflow.com/a/5941122/1246870)

+0

Нет, это не обязательно, но это необходимо, чтобы получить ту же схему. Использование DataContractSerializer изменяет схему. – Rookian