RSS-каналы в ASP.NET MVC


113

Как вы порекомендуете обрабатывать RSS-каналы в ASP.NET MVC? Используете стороннюю библиотеку? Используете RSS в BCL? Просто создаете представление RSS, которое отображает XML? Или что-то совсем другое?


Используя RssToolkit, вам просто нужно иметь один файл .ashx в вашем проекте для создания RSS-канала. Затем вы можете переписать его URL-адрес на более удобный. Я думаю, что в этом подходе нет ничего против MVC.
Махди Тагизаде

Вот следующий пост, который развивает идею RssActionResult немного дальше с обобщенным классом результатов SyndicationAction, а также с условным фильтром получения 304 NotModified. 58bits.com/blog/…
Blue Waters

Я написал RssResult, который вы можете посмотреть, если хотите. Он должен соответствовать вашим требованиям http://www.wduffy.co.uk/blog/rssresult-aspnet-mvc-rss-actionresult/
WDuffy,

1
Просто чтобы обновить этот вопрос через 18 месяцев после последнего редактирования - казалось разумным спросить: «Изменились ли вещи с итерациями как для .net, так и для mvc, которые изменили бы наш подход к проблеме управления фидами»? Консенсус (через другой поток SO) таков: «Никаких фундаментальных изменений - это остается вашим лучшим набором альтернатив».
JustSteve

плохое предложение для ASP.NET MVC.
tugberk

Ответы:


64

Вот что я рекомендую:

  1. Создайте класс с именем RssResult, который наследуется от абстрактного базового класса ActionResult.
  2. Переопределите метод ExecuteResult.
  3. ExecuteResult имеет ControllerContext, переданный ему вызывающей стороной, и с его помощью вы можете получить данные и тип содержимого.
  4. После изменения типа контента на rss вы захотите сериализовать данные в RSS (используя свой собственный код или другую библиотеку) и написать в ответ.

  5. Создайте действие на контроллере, который вы хотите вернуть rss, и установите тип возвращаемого значения как RssResult. Возьмите данные из своей модели на основе того, что вы хотите вернуть.

  6. Тогда любой запрос к этому действию получит rss любых выбранных вами данных.

Это, вероятно, самый быстрый и многоразовый способ возврата rss в ответ на запрос в ASP.NET MVC.


10
У Гензельмана есть аналогичное решение (видео: начиная с 41 м), в котором он наследуется от FileResult. Таким образом вы можете вызвать конструктор вашего класса base("application/rss+xml")и избежать шагов 3 и 4. Он действительно переопределяет ExecuteResult, но это не жизненно важно. Он также сокращает множество типичных доморощенных кодов и использует функции 3.5+ SyndicateItem, SyndicateFeedи Rss20FeedFormatter.
patridge

@Dale: можно ли написать ответ, если вы хотите вывести его для частичного просмотра? Спасибо.
Кристиан

1
Обновленная ссылка на видео Гензельмана из моего предыдущего комментария.
patridge 02

150

Платформа .NET предоставляет классы, которые обрабатывают синдикацию: SyndicationFeed и т. Д. Итак, вместо того, чтобы выполнять рендеринг самостоятельно или использовать другую предложенную библиотеку RSS, почему бы не позволить структуре позаботиться об этом?

По сути, вам просто нужен следующий настраиваемый ActionResult, и вы готовы к работе:

public class RssActionResult : ActionResult
{
    public SyndicationFeed Feed { get; set; }

    public override void ExecuteResult(ControllerContext context)
    {
        context.HttpContext.Response.ContentType = "application/rss+xml";

        Rss20FeedFormatter rssFormatter = new Rss20FeedFormatter(Feed);
        using (XmlWriter writer = XmlWriter.Create(context.HttpContext.Response.Output))
        {
            rssFormatter.WriteTo(writer);
        }
    }
}

Теперь в действии вашего контроллера вы можете просто вернуть следующее:

return new RssActionResult() { Feed = myFeedInstance };

В моем блоге есть полный образец по адресу http://www.developerzen.com/2009/01/11/aspnet-mvc-rss-feed-action-result/


34

Я согласен с Хаакедом. В настоящее время я реализую свой сайт / блог, используя структуру MVC, и я выбрал простой подход создания нового представления для RSS:

<%@ Page ContentType="application/rss+xml" Language="C#" AutoEventWireup="true" CodeBehind="PostRSS.aspx.cs" Inherits="rr.web.Views.Blog.PostRSS" %><?xml version="1.0" encoding="utf-8"?>
<rss version="2.0">
<channel>
<title>ricky rosario's blog</title>
<link>http://<%= Request.Url.Host %></link>
<description>Blog RSS feed for rickyrosario.com</description>
<lastBuildDate><%= ViewData.Model.First().DatePublished.Value.ToUniversalTime().ToString("r") %></lastBuildDate>
<language>en-us</language>
<% foreach (Post p in ViewData.Model) { %>
    <item>
    <title><%= Html.Encode(p.Title) %></title>
    <link>http://<%= Request.Url.Host + Url.Action("ViewPostByName", new RouteValueDictionary(new { name = p.Name })) %></link>
    <guid>http://<%= Request.Url.Host + Url.Action("ViewPostByName", new RouteValueDictionary(new { name = p.Name })) %></guid>
    <pubDate><%= p.DatePublished.Value.ToUniversalTime().ToString("r") %></pubDate>
    <description><%= Html.Encode(p.Content) %></description>
    </item>
<% } %>
</channel>
</rss>

Для получения дополнительной информации посетите (бесстыдный плагин) http://rickyrosario.com/blog/creating-an-rss-feed-in-asp-net-mvc


4
для использования с Razor: @model PageModel @ {Response.ContentType = "application / rss + xml"; } <? xml version = "1.0" encoding = "UTF-8"?>
Энтони Джонстон

2
Какие накладные расходы? вы имеете в виду тот факт, что вы пишете меньше кода, чтобы сделать то же самое более читаемым?
Пол

12

Еще один безумный подход, но имеющий свое преимущество, заключается в использовании обычного представления .aspx для отображения RSS. В вашем методе действия просто установите соответствующий тип контента. Одним из преимуществ этого подхода является то, что легко понять, что отображается и как добавлять пользовательские элементы, такие как геолокация.

Опять же, другие перечисленные подходы могут быть лучше, просто я их не использовал. ;)


3
@Haacked: В мире полно недействительных RSS XML, созданных системой шаблонов. Пожалуйста, не добавляйте беспорядка! Рики, кодировка HTML! = Кодировка XML.
Брэд Уилсон,

Ниже представлена ​​документация по Html Encode из MSDN:> Благодаря текущим деталям реализации эту функцию можно использовать как функцию xmlEncode. В настоящее время все именованные сущности, используемые этой функцией, также являются предопределенными именованными сущностями xml. Они <> "& закодированы как & lt; & gt; & quot; и & amp;. Остальные объекты закодированы в десятичной системе, например & # 160 ;. http://msdn.microsoft.com/en-us/library/73z22y6h.aspx
Ricky

Как при этом убедиться, что XML действителен? Было бы неплохо, если бы рендеринг представления был отделен от входящего веб-запроса, чтобы сделать возможным представление XML или шаблоны электронной почты, такие как done ruby ​​on rails.
Paco

Вместо использования механизма просмотра вы можете создать RssResult, производный от ActionResult. Мы делаем это с помощью JsonResult, который сериализует объект в JSON. В вашем случае вы найдете сериализатор (я думаю, что он есть в WCF), который сериализируется в RSS.
Haacked

8

Я получил это от Эрана Кампфа и видеоролика Скотта Хансельмана (забыл ссылку), поэтому он немного отличается от некоторых других сообщений здесь, но, надеюсь, полезен и готов для копирования и вставки в качестве примера RSS-канала.

Из моего блога

Эран Кампф

using System;
using System.Collections.Generic;
using System.ServiceModel.Syndication;
using System.Web;
using System.Web.Mvc;
using System.Xml;

namespace MVC3JavaScript_3_2012.Rss
{
    public class RssFeed : FileResult
    {
        private Uri _currentUrl;
        private readonly string _title;
        private readonly string _description;
        private readonly List<SyndicationItem> _items;

        public RssFeed(string contentType, string title, string description, List<SyndicationItem> items)
            : base(contentType)
        {
            _title = title;
            _description = description;
            _items = items;
        }

        protected override void WriteFile(HttpResponseBase response)
        {
            var feed = new SyndicationFeed(title: this._title, description: _description, feedAlternateLink: _currentUrl,
                                           items: this._items);
            var formatter = new Rss20FeedFormatter(feed);
            using (var writer = XmlWriter.Create(response.Output))
            {
                formatter.WriteTo(writer);
            }
        }

        public override void ExecuteResult(ControllerContext context)
        {
            _currentUrl = context.RequestContext.HttpContext.Request.Url;
            base.ExecuteResult(context);
        }
    }
}

И код контроллера ....

    [HttpGet]
public ActionResult RssFeed()
{
    var items = new List<SyndicationItem>();
    for (int i = 0; i < 20; i++)
    {
        var item = new SyndicationItem()
        {
            Id = Guid.NewGuid().ToString(),
            Title = SyndicationContent.CreatePlaintextContent(String.Format("My Title {0}", Guid.NewGuid())),
            Content = SyndicationContent.CreateHtmlContent("Content The stuff."),
            PublishDate = DateTime.Now
        };
        item.Links.Add(SyndicationLink.CreateAlternateLink(new Uri("http://www.google.com")));//Nothing alternate about it. It is the MAIN link for the item.
        items.Add(item);
    }

    return new RssFeed(title: "Greatness",
                       items: items,
                       contentType: "application/rss+xml",
                       description: String.Format("Sooper Dooper {0}", Guid.NewGuid()));

}
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.