8

在 WCF 项目中使用 Rss20FeedFormatter 类时,我试图用一个<![CDATA[ ]]>部分来包装我的描述元素的内容。我发现无论我做什么,描述元素的 HTML 内容总是被编码,并且从未添加 CDATA 部分。在查看 Rss20FeedFormatter 的源代码后,我发现在构建 Summary 节点时,它基本上创建了一个新的 TextSyndicationContent 实例,该实例消除了之前指定的任何设置(我认为)。

我的代码

public class CDataSyndicationContent : TextSyndicationContent
{
    public CDataSyndicationContent(TextSyndicationContent content)
        : base(content)
    {
    }

    protected override void WriteContentsTo(System.Xml.XmlWriter writer)
    {
        writer.WriteCData(Text);
    }
}

...(以下代码应使用 CDATA 部分包装摘要)

SyndicationItem item = new SyndicationItem();
item.Title = new TextSyndicationContent(name);
item.Summary = new CDataSyndicationContent(
                   new TextSyndicationContent(
                         "<div>This is a test</div>", 
                         TextSyndicationContentKind.Html));

Rss20FeedFormatter 代码 (AFAIK,由于这个逻辑,上面的代码不起作用)

...
else if (reader.IsStartElement("description", ""))
   result.Summary = new TextSyndicationContent(reader.ReadElementString());
...

作为一种解决方法,我使用了 RSS20FeedFormatter 来构建 RSS,然后手动修补 RSS。例如:

        StringBuilder buffer = new StringBuilder();
        XmlTextWriter writer = new XmlTextWriter(new StringWriter(buffer));
        feedFormatter.WriteTo(writer ); // feedFormatter = RSS20FeedFormatter

        PostProcessOutputBuffer(buffer);
        WebOperationContext.Current.OutgoingResponse.ContentType = 
                                   "application/xml; charset=utf-8";
        return new MemoryStream(Encoding.UTF8.GetBytes(buffer.ToString()));      

...

    public void PostProcessOutputBuffer(StringBuilder buffer)
    {
        var xmlDoc = XDocument.Parse(buffer.ToString());
        foreach (var element in xmlDoc.Descendants("channel").First()
                                      .Descendants("item")
                                      .Descendants("description"))
        {
            VerifyCdataHtmlEncoding(buffer, element);
        }

        foreach (var element in xmlDoc.Descendants("channel").First()
                                      .Descendants("description"))
        {
            VerifyCdataHtmlEncoding(buffer, element);
        }

        buffer.Replace(" xmlns:a10=\"http://www.w3.org/2005/Atom\"",
                       " xmlns:atom=\"http://www.w3.org/2005/Atom\"");
        buffer.Replace("a10:", "atom:");
    }

    private static void VerifyCdataHtmlEncoding(StringBuilder buffer,
                                                XElement element)
    {
        if (!element.Value.Contains("<") || !element.Value.Contains(">"))
        {
            return;
        }

        var cdataValue = string.Format("<{0}><![CDATA[{1}]]></{2}>",
                                       element.Name,
                                       element.Value, 
                                       element.Name);
        buffer.Replace(element.ToString(), cdataValue);
    }

此解决方法的想法来自以下位置,我只是将其调整为使用 WCF 而不是 MVC。http://localhost:8732/Design_Time_Addresses/SyndicationServiceLibrary1/Feed1/

我只是想知道这是否只是 Rss20FeedFormatter 中的一个错误,还是设计使然?另外,如果有人有更好的解决方案,我很想听听!

4

3 回答 3

3

好吧@Page Brooks,我认为这更像是一个解决方案,而不是一个问题:)。谢谢!!!并回答您的问题(;)),是的,我绝对认为这是 Rss20FeedFormatter 中的一个错误(尽管我没有追到它),因为遇到了与您描述的完全相同的问题。

您的帖子中有一个“localhost:8732”推荐,但它在我的本地主机上不可用;)。我认为您的意思是将“PostProcessOutputBuffer”解决方法归功于这篇文章: http ://damieng.com/blog/2010/04/26/creating-rss-feeds-in-asp-net-mvc

或者实际上它不在这篇文章中,而是在大卫惠特尼的评论中,他后来在这里单独提出了一个要点: https ://gist.github.com/davidwhitney/1027181

感谢您根据我的需要提供此解决方法的调整,因为我也找到了解决方法,但仍在努力从 MVC 进行调整。现在我只需要调整您的解决方案,将 RSS 提要放到我正在使用的 .ashx 处理程序中的当前 Http 请求中。

基本上我猜你提到的使用 CDataSyndicationContent 的修复是从 2011 年 2 月开始的,假设你是从这篇文章中得到的(至少我做到了): SyndicationFeed: Content as CDATA?

由于 Rss20FeedFormatter 的代码更改为您在帖子中的内容,此修复程序在某些较新的 ASP.NET 版本或其他内容中停止工作。此代码更改也可能是对 MVC 框架中其他内容的改进,但对于那些使用 CDataSyndicationContent 修复的人来说,它肯定会导致错误!

于 2013-05-23T22:31:40.803 回答
1
string stylesheet = @"<xsl:stylesheet version=""1.0"" xmlns:xsl=""http://www.w3.org/1999/XSL/Transform""><xsl:output cdata-section-elements=""description"" method=""xml"" indent=""yes""/></xsl:stylesheet>";
XmlReader reader = XmlReader.Create(new StringReader(stylesheet));
XslCompiledTransform t = new XslCompiledTransform(true);
t.Load(reader);

using (MemoryStream ms = new MemoryStream())
{
    XmlWriter writer = XmlWriter.Create(ms, t.OutputSettings);
    rssFeed.WriteTo(writer);  // rssFeed is Rss20FeedFormatter 
    writer.Flush();
    ms.Position = 0;
    string niko = Encoding.UTF8.GetString(ms.ToArray());
}

我确定有人已经指出了这一点,但这是我使用的一个愚蠢的解决方法。t.OutputSettings 是 XmlWriterSettings 类型,其中 cdataSections 填充有单个 XmlQualifiedName“描述”。

希望它可以帮助别人。

于 2015-12-24T19:00:59.480 回答
0

我在别处找到了 Cdata 的代码

    public class CDataSyndicationContent : TextSyndicationContent
{
    public CDataSyndicationContent(TextSyndicationContent content)
        : base(content)
    {
    }

    protected override void WriteContentsTo(System.Xml.XmlWriter writer)
    {
        writer.WriteCData(Text);
    }
}

代码将其称为类似的东西:

item.Content = new Helpers.CDataSyndicationContent(new TextSyndicationContent("<span>TEST2</span>", TextSyndicationContentKind.Html));

但是没有调用“WriteContentsTo”函数。

我尝试了 Atom10FeedFormatter 而不是 Rss20FeedFormatter - 它奏效了!显然,这提供了 Atom 提要而不是传统的 RSS - 但值得一提。

输出代码为:

//var formatter = new Rss20FeedFormatter(feed);
    Atom10FeedFormatter formatter = new Atom10FeedFormatter(feed);
    using (var writer = XmlWriter.Create(response.Output, new XmlWriterSettings { Indent = true }))
    {
        formatter.WriteTo(writer);
    }
于 2015-08-18T11:35:49.037 回答