deserialize xml from azure response

时间:2015-09-14 16:09:02

标签: c# xml azure deserialization

I am looking to deserialize data and place it into a generic class from a response from Azure.

<ServiceResources xmlns:i="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://schemas.microsoft.com/windowsazure">
<ServiceResource>
    <Name>Airport1</Name>
    <Type>Microsoft.SqlAzure.FirewallRule</Type>
    <State>Normal</State>
    <SelfLink>https://management.core.windows.net:xxx/xxx/services/sqlservers/servers/xxx/firewallrules/Airport1</SelfLink>
    <ParentLink>https://management.core.windows.net:xxxx/services/sqlservers/servers/xxx</ParentLink>
    <StartIPAddress>000.000.000.000</StartIPAddress>
    <EndIPAddress>2000.000.000.000</EndIPAddress>
  </ServiceResource>

There are several objects I need to deserialze into my class.

[Serializable, XmlRoot(ElementName = "ServiceResource", Namespace = "http://schemas.microsoft.com/windowsazure/")]
public class ServiceResource
{
    [XmlElement("Name")]
    public string Name { get; set; }
    [XmlElement("Type")]
    public string Type { get; set; }
    [XmlElement("State")]
    public string State { get; set; }
    [XmlElement("SelfLink")]
    public string SelfLink { get; set; }
    [XmlElement("ParentLink")]
    public string ParentLink { get; set; }
    [XmlElement("StartIPAddress")]
    public string StartIPAddress { get; set; }
    [XmlElement("EndIPAddress")]
    public string EndIPAddress { get; set; }
} 

I have tried several different ventures into this and can't nail it. I have used the xmlSerializer but hit blocks on that.

using (var responseStreamReader = new StreamReader(webResponse.GetResponseStream()))
{
    XmlSerializer serializer = new XmlSerializer(typeof(ServiceResource));
    ServiceResource deserialized = (ServiceResource)serializer.Deserialize(responseStreamReader);

}

Any help would be gratefully accepted.

2 个答案:

答案 0 :(得分:3)

我假设您正在尝试反序列化整个对象图。给定xml的根节点old: Optional(<null>) new: Optional(ABC) 包含ServiceResources。你有两个选择,你可以将整个xml模仿成类并进行反序列化;或者只是获取ServiceResource的内部节点并反序列化该部分。

如果你使用第一个选项,那么你需要将ServiceResource存储在另一个具有映射集合属性的类中,其中XmlElement名称设置为&#34; ServiceResource&#34;,例如:

ServiceResource

有了这个,您应该能够直接反序列化。 Container类具有映射到ServiceResource元素的集合,该元素将加载服务资源的所有节点。注意:反序列化目标类型现在是&#34; ServiceResources&#34;不是内在类型&#34; ServiceResource&#34;

[XmlType(Namespace="http://schemas.microsoft.com/windowsazure")]
[XmlRoot(Namespace="http://schemas.microsoft.com/windowsazure")]        
public class ServiceResource
{
    public string Name { get; set; }

    public string Type { get; set; }

    public string State { get; set; }

    public string SelfLink { get; set; }

    public string ParentLink { get; set; }

    public string StartIPAddress { get; set; }

    public string EndIPAddress { get; set; }

}

[XmlType(Namespace="http://schemas.microsoft.com/windowsazure")]
[XmlRoot(Namespace="http://schemas.microsoft.com/windowsazure")]
public class ServiceResources
{
    [XmlElement("ServiceResource")]
    public List<ServiceResource> ServiceResource { get; set; }
}

答案 1 :(得分:2)

答案

Azure REST Api在XML中返回ServiceResource的列表。所以你需要将它封装到一个类中。这是一个例子。

[XmlRoot(
    ElementName = "ServiceResources",
    Namespace = "http://schemas.microsoft.com/windowsazure")]
public class ServiceResources
{
    public ServiceResources()
    {
        Items = new List<ServiceResource>();
    }

    [XmlElement("ServiceResource")]
    public List<ServiceResource> Items { get; set; }
}

public class ServiceResource
{
    [XmlElement("Name")]
    public string Name { get; set; }
    [XmlElement("Type")]
    public string Type { get; set; }
    [XmlElement("State")]
    public string State { get; set; }
    [XmlElement("SelfLink")]
    public string SelfLink { get; set; }
    [XmlElement("ParentLink")]
    public string ParentLink { get; set; }
    [XmlElement("StartIPAddress")]
    public string StartIPAddress { get; set; }
    [XmlElement("EndIPAddress")]
    public string EndIPAddress { get; set; }
}

使用这两个类,您现在可以执行以下操作。

var response = request.GetResponse();
var message = string.Empty;
using (var responseStreamReader = new StreamReader(response.GetResponseStream()))
{
    message = responseStreamReader.ReadToEnd();
}

var textReader = new StringReader(message);
var serializer = new XmlSerializer(typeof(ServiceResources));
var serviceResources = 
    serializer.Deserialize(textReader) as ServiceResources;

演示控制台应用

using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Net;
using System.Security.Cryptography.X509Certificates;
using System.Text;
using System.Xml;
using System.Xml.Serialization;

namespace DeserializeAzureXmlResponse
{
    class Program
    {
        private static string certificateThumbprint = "19DAED4D4ABBE0D400DC33A6D99D00D7BBB24472";
        private static string subscriptionId = "14929cfc-3501-48cf-a5c9-b24a7daaf694";
        static string sqlServerName = "mvp2015";

        static string managementUri = "https://management.core.windows.net";
        static string sqlServerApi = "services/sqlservers/servers";
        static string firewallRules = "firewallrules";

        static void Main(string[] args)
        {
            var restUri = CreateRestUri();
            var clientCert = GetX509FromPersonalStore();

            var request = (HttpWebRequest)HttpWebRequest.Create(restUri);
            request.Headers.Add("x-ms-version", "2012-03-01");
            request.ClientCertificates.Add(clientCert);

            var response = request.GetResponse();
            var message = string.Empty;
            using (var responseStreamReader = new StreamReader(response.GetResponseStream()))
            {
                message = responseStreamReader.ReadToEnd();
            }

            var textReader = new StringReader(message);
            var serializer = new XmlSerializer(typeof(ServiceResources));
            var serviceResources = serializer.Deserialize(textReader) as ServiceResources;
            foreach (var sr in serviceResources.Items)
            {
                Console.WriteLine("Name".PadRight(20) + sr.Name);
                Console.WriteLine("Type".PadRight(20) + sr.Type);
                Console.WriteLine("State".PadRight(20) + sr.State);
                Console.WriteLine("SelfLink".PadRight(20) + sr.SelfLink);
                Console.WriteLine("ParentLink".PadRight(20) + sr.ParentLink);
                Console.WriteLine("StartIP".PadRight(20) + sr.StartIPAddress);
                Console.WriteLine("EndIP".PadRight(20) + sr.EndIPAddress);
                Console.WriteLine("+++++++++++");
            }
            Console.ReadLine();
        }

        static Uri CreateRestUri()
        {
            // https://management.core.windows.net/{subscriptionID}/services/sqlservers/servers/{server}/firewallrules/
            var builder = new StringBuilder();
            builder.Append(managementUri + "/");
            builder.Append(subscriptionId + "/");
            builder.Append(sqlServerApi + "/");
            builder.Append(sqlServerName + "/");
            builder.Append(firewallRules + "/");
            var uri = new Uri(builder.ToString());
            return uri;
        }

        static X509Certificate GetX509FromPersonalStore()
        {
            // To view the personal store, press `Win + R` and then type `certmgr.msc`
            var store = new X509Store(StoreLocation.CurrentUser);
            store.Open(OpenFlags.ReadOnly);
            var certificates = store.Certificates.Find(X509FindType.FindByThumbprint, certificateThumbprint, true);
            var certificate = certificates[0];
            store.Close();
            return certificate;
        }
    }

    [XmlRoot(
        ElementName = "ServiceResources",
        Namespace = "http://schemas.microsoft.com/windowsazure")]
    public class ServiceResources
    {
        public ServiceResources()
        {
            Items = new List<ServiceResource>();
        }

        [XmlElement("ServiceResource")]
        public List<ServiceResource> Items { get; set; }
    }

    public class ServiceResource
    {
        [XmlElement("Name")]
        public string Name { get; set; }
        [XmlElement("Type")]
        public string Type { get; set; }
        [XmlElement("State")]
        public string State { get; set; }
        [XmlElement("SelfLink")]
        public string SelfLink { get; set; }
        [XmlElement("ParentLink")]
        public string ParentLink { get; set; }
        [XmlElement("StartIPAddress")]
        public string StartIPAddress { get; set; }
        [XmlElement("EndIPAddress")]
        public string EndIPAddress { get; set; }
    }
}

输出

Name                My-House
Type                Microsoft.SqlAzure.FirewallRule
State               Normal
SelfLink            https://management.core.windows.net/14929cfc-35
ParentLink          https://management.core.windows.net/14929cfc-35
StartIP             123.435.234.643
EndIP               123.435.234.643
+++++++++++
Name                AllowAllWindowsAzureIps
Type                Microsoft.SqlAzure.FirewallRule
State               Normal
SelfLink            https://management.core.windows.net/14929cfc-35
ParentLink          https://management.core.windows.net/14929cfc-35
StartIP             0.0.0.0
EndIP               0.0.0.0
+++++++++++

另见

Is it possible to deserialize XML into List<T>?

List Firewall Rules