2009-09-03 8 views
6

Ich habe einen Dienst erstellt, der Vorschau 2 des WCF REST Starterkits verwendet, aber ich habe ein Problem mit der Weitergabe von XML-Daten in den Aufrufen. Hier ist meine Anfrage Objekt:Verwenden von CDATA mit WCF-REST-Starterkits

[DataContract(Namespace = "")] 
public class ServiceRequest 
{ 
    [DataMember] 
    public string ContentText { get; set; } 
    [DataMember] 
    public string ApiKey { get; set; } 

} 

Alles funktioniert gut, bis Sie '' dort hineinwerfen. Gibt es eine Kapselung der ContentText-Eigenschaft in einer CDATA oder etwas Ähnliches?

Antwort

11

Marc Gravell hat eine Lösung here für die Serialisierung von CDATA-Abschnitten.

Ich habe den Code hier für die Nachwelt kopiert.

aktualisiert: das vorherige Beispiel kein gültiges Schema generiert haben, die XmlSchemaProviderAttribute und begleitende Methode generieren "xs: string", die [more...]

using System; 
using System.IO; 
using System.Runtime.Serialization; 
using System.Xml; 
using System.Xml.Serialization; 
using System.ComponentModel; 

[XmlSchemaProvider("GenerateSchema")] 
public sealed class CDataWrapper : IXmlSerializable 
{ 
    // implicit to/from string 
    public static implicit operator string(CDataWrapper value) 
    { 
    return value == null ? null : value.Value; 
    } 

    public static implicit operator CDataWrapper(string value) 
    { 
    return value == null ? null : new CDataWrapper { Value = value }; 
    } 

    public System.Xml.Schema.XmlSchema GetSchema() 
    { 
    return null; 
    } 

    // return "xs:string" as the type in scheme generation 
    public static XmlQualifiedName GenerateSchema(XmlSchemaSet xs) 
    { 
     return XmlSchemaType.GetBuiltInSimpleType(XmlTypeCode.String).QualifiedName; 
    } 

    // "" => <Node/> 
    // "Foo" => <Node><![CDATA[Foo]]></Node> 
    public void WriteXml(XmlWriter writer) 
    { 
    if (!string.IsNullOrEmpty(Value)) 
    { 
     writer.WriteCData(Value); 
    } 
    } 

    // <Node/> => "" 
    // <Node></Node> => "" 
    // <Node>Foo</Node> => "Foo" 
    // <Node><![CDATA[Foo]]></Node> => "Foo" 
    public void ReadXml(XmlReader reader) 
    { 
    if (reader.IsEmptyElement) 
    { 
     Value = ""; 
    } 
    else 
    { 
     reader.Read(); 

     switch (reader.NodeType) 
     { 
     case XmlNodeType.EndElement: 
      Value = ""; // empty after all... 
      break; 
     case XmlNodeType.Text: 
     case XmlNodeType.CDATA: 
      Value = reader.ReadContentAsString(); 
      break; 
     default: 
      throw new InvalidOperationException("Expected text/cdata"); 
     } 
    } 
    } 

    // underlying value 
    public string Value { get; set; } 
    public override string ToString() 
    { 
    return Value; 
    } 
} 

// example usage 
[DataContract(Namespace="http://myobjects/")] 
public sealed class MyType 
{ 
    public string SomeValue { get; set; } 
    [DataMember(Name = "SomeValue", EmitDefaultValue = false)] 
    private CDataWrapper SomeValueCData 
    { 
    get { return SomeValue; } 
    set { SomeValue = value; } 
    } 

    public string EmptyTest { get; set; } 
    [DataMember(Name = "EmptyTest", EmitDefaultValue = false)] 
    private CDataWrapper EmptyTestCData 
    { 
    get { return EmptyTest; } 
    set { EmptyTest = value; } 
    } 

    public string NullTest { get; set; } 
    [DataMember(Name = "NullTest", EmitDefaultValue = false)] 
    private CDataWrapper NullTestCData 
    { 
    get { return NullTest ; } 
    set { NullTest = value; } 
    } 
} 

// test rig 
static class Program 
{ 
    static void Main() 
    { 
    DataContractSerializer dcs = new DataContractSerializer(typeof(MyType)); 

    StringWriter writer = new StringWriter(); 
    using (XmlWriter xw = XmlWriter.Create(writer)) 
    { 
     MyType foo = new MyType 
     { 
     SomeValue = @"&<t\d", 
     NullTest = null, 
     EmptyTest = "" 
     }; 

     ShowObject("Original", foo); 

     dcs.WriteObject(xw, foo); 
     xw.Close(); 
    } 

    string xml = writer.ToString(); 
    ShowObject("Xml", xml); 

    StringReader reader = new StringReader(xml); 
    using (XmlReader xr = XmlReader.Create(reader)) 
    { 
     MyType bar = (MyType) dcs.ReadObject(xr); 
     ShowObject("Recreated", bar); 
    } 
    } 

    static void ShowObject(string caption, object obj) 
    { 
    Console.WriteLine(); 
    Console.WriteLine("** {0} **", caption); 
    Console.WriteLine(); 

    if (obj == null) 
    { 
     Console.WriteLine("(null)"); 
    } 
    else if (obj is string) 
    { 
     Console.WriteLine((string)obj); 
    } 
    else 
    { 
     foreach (PropertyDescriptor prop in TypeDescriptor.GetProperties(obj)) 
     { 
     Console.WriteLine("{0}:\t{1}", prop.Name, prop.GetValue(obj) ?? "(null)"); 
     } 
    } 
    } 
} 
3

VB Umwandlung des CDataWrapper in der akzeptierte Antwort funktioniert:

+0

Gute Sachen. Danke Kevin! –

+0

Der obige Code stoppt die Analyse der Eigenschaften nach dem CData. Dies ist ein Problem, wenn Sie mehr als ein Element in der XML erwarten. Hinzufügen des fehlenden readers.Read() am Ende des else-Tags in ReadXML behebt dieses Problem. – Beejee

3

Über dem Code fehlt die Tatsache, dass Sie nach dem Lesen über den Inhalt hinausgehen müssen. Diese Klasse funktioniert also nicht mit einer Sammlung.

Ändern Sie es in den folgenden, und Sie können jetzt Sammlungen von CDataWrapper halten.

Value = reader.ReadContentAsString(); 
reader.Read(); 
0

Obwohl dies ein älterer Beitrag hier ist meine 2 ¢. Ich habe dieses Problem behoben, indem ich das Datenelement als XmlElement definiert habe.