2009-09-03 10 views
6

J'ai un service créé avec Preview 2 du kit de démarrage WCF REST, mais j'ai rencontré un problème avec la transmission de données de type XML dans les appels. Voici mon objet de demande:Utilisation de CDATA avec des kits de démarrage WCF REST

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

} 

Tout fonctionne bien jusqu'à ce que vous jetiez '' là-dedans. Y at-il un pour encapsuler la propriété ContentText dans un CDATA ou quelque chose de similaire?

Répondre

11

Marc Gravell a une solution here pour la sérialisation des sections CDATA.

J'ai copié le code ici pour la postérité.

mise à jour: l'exemple précédent n'a pas généré un schéma valide, le XmlSchemaProviderAttribute et la méthode d'accompagnement va générer « xs: string » qui fonctionne [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

conversion VB du CDataWrapper dans la réponse acceptée:

Imports System 
Imports System.IO 
Imports System.Runtime.Serialization 
Imports System.Xml 
Imports System.Xml.Schema 
Imports System.Xml.Serialization 
Imports System.ComponentModel 

Public Class CDataWrapper 
    Implements IXmlSerializable 

    'underlying value 
    Public Property Value As String 

    'Implicit to/from string 
    Public Shared Widening Operator CType(ByVal value As CDataWrapper) As String 
     If value Is Nothing Then 
      Return Nothing 
     Else 
      Return value.Value 
     End If 
    End Operator 

    Public Shared Widening Operator CType(value As String) As CDataWrapper 
     If value Is Nothing Then 
      Return Nothing 
     Else 
      Return New CDataWrapper() With {.Value = value} 
     End If 
    End Operator 


    Public Function GetSchema() As XmlSchema Implements IXmlSerializable.GetSchema 
     Return Nothing 
    End Function 

    ' <Node/> => "" 
    ' <Node></Node> => "" 
    ' <Node>Foo</Node> => "Foo" 
    ' <Node><![CDATA[Foo]]></Node> => "Foo" 
    Public Sub ReadXml(reader As XmlReader) Implements IXmlSerializable.ReadXml 
     If reader.IsEmptyElement Then 
      Me.Value = "" 
     Else 
      reader.Read() 

      Select Case reader.NodeType 
       Case XmlNodeType.EndElement 
        Me.Value = "" ' empty after all... 
       Case XmlNodeType.Text, XmlNodeType.CDATA 
        Me.Value = reader.ReadContentAsString() 
       Case Else 
        Throw New InvalidOperationException("Expected text/cdata") 
      End Select 
     End If 
    End Sub 

    ' "" => <Node/> 
    ' "Foo" => <Node><![CDATA[Foo]]></Node> 
    Public Sub WriteXml(writer As XmlWriter) Implements IXmlSerializable.WriteXml 
     If Not String.IsNullOrEmpty(Me.Value) Then 
      writer.WriteCData(Me.Value) 
     End If 
    End Sub 

    Public Overrides Function ToString() As String 
     Return Me.Value 
    End Function 
End Class 
+0

Bonnes choses. Merci Kevin! –

+0

Le code ci-dessus arrête l'analyse des propriétés après le CData. Ceci est un problème lorsque vous attendez plus d'un élément dans le fichier xml. Ajout du lecteur manquant.Read() à la fin de la balise else dans ReadXML résout ce problème. – Beejee

3

Au-dessus du code est manquant le fait que vous devez aller au-delà du contenu après l'avoir lu. Donc, cette classe en tant que telle ne fonctionnera pas avec une collection. Remplacez-le par ce qui suit, et vous pouvez maintenant conserver Collections de CDataWrapper.

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

Bien que ce soit un post plus ancien voici mes 2 ¢. J'ai résolu ce problème en définissant le membre de données comme XmlElement.

[DataMember(Name = "MyCData")] 
     public XmlElement MyCDataField { get; set; } 
Questions connexes