();
-
- if (entry.Latitude.HasValue)
- {
- latitude = entry.Latitude;
- }
- else
- {
- if (dasBlogSettings.SiteConfiguration.EnableDefaultLatLongForNonGeoCodedPosts)
- {
- latitude = dasBlogSettings.SiteConfiguration.DefaultLatitude;
- }
- }
-
- if (entry.Longitude.HasValue)
- {
- longitude = entry.Longitude;
- }
- else
- {
- if (dasBlogSettings.SiteConfiguration.EnableDefaultLatLongForNonGeoCodedPosts)
- {
- longitude = dasBlogSettings.SiteConfiguration.DefaultLongitude;
- }
- }
-
- if (latitude.HasValue && longitude.HasValue)
- {
- XmlElement geoLoc = doc2.CreateElement("georss", "point", "http://www.georss.org/georss");
- geoLoc.InnerText = String.Format(CultureInfo.InvariantCulture, "{0:R} {1:R}", latitude, longitude);
- anyElements.Add(geoLoc);
- }
- }
-
- if (dasBlogSettings.SiteConfiguration.EnableComments)
- {
- if (entry.AllowComments)
- {
- XmlElement commentApi = doc2.CreateElement("wfw", "comment", "http://wellformedweb.org/CommentAPI/");
- commentApi.InnerText = dasBlogSettings.GetCommentViewUrl(dasBlogSettings.GeneratePostUrl(entry));
- anyElements.Add(commentApi);
- }
-
- XmlElement commentRss = doc2.CreateElement("wfw", "commentRss", "http://wellformedweb.org/CommentAPI/");
- commentRss.InnerText = dasBlogSettings.GetEntryCommentsRssUrl(entry.EntryId);
- anyElements.Add(commentRss);
-
- //for RSS conformance per FeedValidator.org
- int commentsCount = dataService.GetPublicCommentsFor(entry.EntryId).Count;
- if (commentsCount > 0)
- {
- XmlElement slashComments = doc2.CreateElement("slash", "comments", "http://purl.org/rss/1.0/modules/slash/");
- slashComments.InnerText = commentsCount.ToString();
- anyElements.Add(slashComments);
- }
- item.Comments = dasBlogSettings.GetCommentViewUrl(dasBlogSettings.GeneratePostUrl(entry));
- }
- item.Language = entry.Language;
-
- if (entry.Categories != null && entry.Categories.Length > 0)
- {
- if (item.Categories == null) item.Categories = new RssCategoryCollection();
-
- string[] cats = entry.Categories.Split(';');
- foreach (string c in cats)
- {
- RssCategory cat = new RssCategory();
- string cleanCat = c.Replace('|', '/');
- cat.Text = cleanCat;
- item.Categories.Add(cat);
- }
- }
- if (entry.Attachments.Count > 0)
- {
- // RSS currently supports only a single enclsoure so we return the first one
- item.Enclosure = new Enclosure();
- item.Enclosure.Url = entry.Attachments[0].Name;
- item.Enclosure.Type = entry.Attachments[0].Type;
- item.Enclosure.Length = entry.Attachments[0].Length.ToString();
- }
- item.PubDate = entry.CreatedUtc.ToString("R");
- if (ch.LastBuildDate == null || ch.LastBuildDate.Length == 0)
- {
- ch.LastBuildDate = item.PubDate;
- }
-
- if (!dasBlogSettings.SiteConfiguration.AlwaysIncludeContentInRSS &&
- entry.Description != null &&
- entry.Description.Trim().Length > 0)
- {
- item.Description = PreprocessItemContent(entry.EntryId, entry.Description);
-
- }
- else
- {
- if (dasBlogSettings.SiteConfiguration.HtmlTidyContent == false)
- {
- item.Description = "" + PreprocessItemContent(entry.EntryId, entry.Content) + "
";
- }
- else
- {
- item.Description = ContentFormatter.FormatContentAsHTML(PreprocessItemContent(entry.EntryId, entry.Content));
+ {
+ if (entry.IsPublic == false || entry.Syndicated == false)
+ {
+ continue;
+ }
+ var item = MapEntryToRssItem(entry, ch, true);
+ ch.Items.Add(item);
+ }
+
+ return documentRoot;
+ }
+ private RssItem MapEntryToRssItem(Entry entry, RssChannel ch = null, bool feedView = false)
+ {
+ var doc2 = new XmlDocument();
+ var anyElements = new List();
+ var item = new RssItem();
+ item.Id = entry.EntryId;
+ item.Title = entry.Title;
+ item.Guid = new DasBlog.Services.Rss.Rss20.Guid();
+ item.Guid.IsPermaLink = false;
+ item.Guid.Text = dasBlogSettings.GetPermaLinkUrl(entry.EntryId);
+ item.Link = dasBlogSettings.RelativeToRoot(dasBlogSettings.GeneratePostUrl(entry));
+ User user = dasBlogSettings.GetUserByEmail(entry.Author);
+
+ if (dasBlogSettings.SiteConfiguration.EnableTrackbackService)
+ {
+ XmlElement trackbackPing = doc2.CreateElement("trackback", "ping", "http://madskills.com/public/xml/rss/module/trackback/");
+ trackbackPing.InnerText = dasBlogSettings.GetTrackbackUrl(entry.EntryId);
+ anyElements.Add(trackbackPing);
+ }
+
+ if (dasBlogSettings.SiteConfiguration.EnablePingbackService)
+ {
+ XmlElement pingbackServer = doc2.CreateElement("pingback", "server", "http://madskills.com/public/xml/rss/module/pingback/");
+ pingbackServer.InnerText = dasBlogSettings.PingBackUrl;
+ anyElements.Add(pingbackServer);
+
+ XmlElement pingbackTarget = doc2.CreateElement("pingback", "target", "http://madskills.com/public/xml/rss/module/pingback/");
+ pingbackTarget.InnerText = dasBlogSettings.GetPermaLinkUrl(entry.EntryId);
+ anyElements.Add(pingbackTarget);
+ }
+
+ XmlElement dcCreator = doc2.CreateElement("dc", "creator", "http://purl.org/dc/elements/1.1/");
+ if (user != null)
+ {
+ dcCreator.InnerText = user.DisplayName;
+ }
+ anyElements.Add(dcCreator);
+
+ // Add GeoRSS if it exists.
+ if (dasBlogSettings.SiteConfiguration.EnableGeoRss)
+ {
+ var latitude = new Nullable();
+ var longitude = new Nullable();
+
+ if (entry.Latitude.HasValue)
+ {
+ latitude = entry.Latitude;
+ }
+ else
+ {
+ if (dasBlogSettings.SiteConfiguration.EnableDefaultLatLongForNonGeoCodedPosts)
+ {
+ latitude = dasBlogSettings.SiteConfiguration.DefaultLatitude;
+ }
+ }
+
+ if (entry.Longitude.HasValue)
+ {
+ longitude = entry.Longitude;
+ }
+ else
+ {
+ if (dasBlogSettings.SiteConfiguration.EnableDefaultLatLongForNonGeoCodedPosts)
+ {
+ longitude = dasBlogSettings.SiteConfiguration.DefaultLongitude;
+ }
+ }
+
+ if (latitude.HasValue && longitude.HasValue)
+ {
+ XmlElement geoLoc = doc2.CreateElement("georss", "point", "http://www.georss.org/georss");
+ geoLoc.InnerText = String.Format(CultureInfo.InvariantCulture, "{0:R} {1:R}", latitude, longitude);
+ anyElements.Add(geoLoc);
+ }
+ }
- try
- {
- string xhtml = ContentFormatter.FormatContentAsXHTML(PreprocessItemContent(entry.EntryId, entry.Content));
- doc2.LoadXml(xhtml);
- anyElements.Add((XmlElement)doc2.SelectSingleNode("//*[local-name() = 'body'][namespace-uri()='http://www.w3.org/1999/xhtml']"));
- }
- catch //(Exception ex)
- {
- //Debug.Write(ex.ToString());
- // absorb
- }
- }
- }
-
- item.anyElements = anyElements.ToArray();
- ch.Items.Add(item);
- }
-
- return documentRoot;
- }
+ if (dasBlogSettings.SiteConfiguration.EnableComments)
+ {
+ if (entry.AllowComments)
+ {
+ XmlElement commentApi = doc2.CreateElement("wfw", "comment", "http://wellformedweb.org/CommentAPI/");
+ commentApi.InnerText = dasBlogSettings.GetCommentViewUrl(dasBlogSettings.GeneratePostUrl(entry));
+ anyElements.Add(commentApi);
+ }
+
+ XmlElement commentRss = doc2.CreateElement("wfw", "commentRss", "http://wellformedweb.org/CommentAPI/");
+ commentRss.InnerText = dasBlogSettings.GetEntryCommentsRssUrl(entry.EntryId);
+ anyElements.Add(commentRss);
+
+ //for RSS conformance per FeedValidator.org
+ int commentsCount = dataService.GetPublicCommentsFor(entry.EntryId).Count;
+ if (commentsCount > 0)
+ {
+ XmlElement slashComments = doc2.CreateElement("slash", "comments", "http://purl.org/rss/1.0/modules/slash/");
+ slashComments.InnerText = commentsCount.ToString();
+ anyElements.Add(slashComments);
+ }
+ item.Comments = dasBlogSettings.GetCommentViewUrl(dasBlogSettings.GeneratePostUrl(entry));
+ }
+ item.Language = entry.Language;
+
+ if (entry.Categories != null && entry.Categories.Length > 0)
+ {
+ if (item.Categories == null) item.Categories = new RssCategoryCollection();
+
+ string[] cats = entry.Categories.Split(';');
+ foreach (string c in cats)
+ {
+ RssCategory cat = new RssCategory();
+ string cleanCat = c.Replace('|', '/');
+ cat.Text = cleanCat;
+ item.Categories.Add(cat);
+ }
+ }
+ if (entry.Attachments.Count > 0)
+ {
+ // RSS currently supports only a single enclsoure so we return the first one
+ item.Enclosure = new Enclosure();
+ item.Enclosure.Url = entry.Attachments[0].Name;
+ item.Enclosure.Type = entry.Attachments[0].Type;
+ item.Enclosure.Length = entry.Attachments[0].Length.ToString();
+ }
+ item.PubDate = entry.CreatedUtc.ToString("R");
+ if (ch != null && (ch.LastBuildDate == null || ch.LastBuildDate.Length == 0))
+ {
+ ch.LastBuildDate = item.PubDate;
+ }
+
+ if (feedView || !dasBlogSettings.SiteConfiguration.AlwaysIncludeContentInRSS &&
+ entry.Description != null &&
+ entry.Description.Trim().Length > 0)
+ {
+ if (feedView)
+ {
+ item.Description = PreprocessItemContent(entry.EntryId, entry.Description);
+ }
+ else
+ {
+ item.Description = entry.Description;
+ }
+ }
+ else
+ {
+ var content = (feedView ? PreprocessItemContent(entry.EntryId, entry.Content) : entry.Content);
+ if (!dasBlogSettings.SiteConfiguration.HtmlTidyContent && feedView)
+ {
+ item.Description = "" + content + "
";
+ }
+ else
+ {
+ item.Description = ContentFormatter.FormatContentAsHTML(!string.IsNullOrEmpty(entry.Description)?entry.Description:content);
+ try
+ {
+ string xhtml = ContentFormatter.FormatContentAsXHTML(System.Net.WebUtility.HtmlDecode(content));
+ doc2.LoadXml(xhtml);
+ anyElements.Add((XmlElement)doc2.SelectSingleNode("//*[local-name() = 'body'][namespace-uri()='http://www.w3.org/1999/xhtml']"));
+ }
+ catch //(Exception ex)
+ {
+ //Debug.Write(ex.ToString());
+ // absorb
+ }
+ }
+ }
+
+ item.anyElements = anyElements.ToArray();
+ return item;
+ }
- protected EntryCollection BuildEntries(string category, int maxDayCount, int maxEntryCount)
+ protected EntryCollection BuildEntries(string category, int maxDayCount, int maxEntryCount)
{
var entryList = new EntryCollection();
diff --git a/source/DasBlog.Web.UI/Config/IISUrlRewrite.Development.config b/source/DasBlog.Web.UI/Config/IISUrlRewrite.Development.config
index 13384168c..88439f0c9 100644
--- a/source/DasBlog.Web.UI/Config/IISUrlRewrite.Development.config
+++ b/source/DasBlog.Web.UI/Config/IISUrlRewrite.Development.config
@@ -57,7 +57,7 @@
-
+
diff --git a/source/DasBlog.Web.UI/Config/site.Development.config b/source/DasBlog.Web.UI/Config/site.Development.config
index e4dc4fb41..ca580195d 100644
--- a/source/DasBlog.Web.UI/Config/site.Development.config
+++ b/source/DasBlog.Web.UI/Config/site.Development.config
@@ -3,7 +3,7 @@
- https://localhost:5001/
+ http://localhost:5001/
@@ -139,6 +139,7 @@
false
false
false
+ false
true
false
false
@@ -159,6 +160,18 @@
+
+
+ eventGrid
+ https://{myns}.westeurope-1.eventgrid.azure.net/api/events
+
+
+ aeg-sas-key
+ xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx
+
+
+
+
true
true
false
diff --git a/source/DasBlog.Web.UI/Config/site.config b/source/DasBlog.Web.UI/Config/site.config
index b7682391b..99ff794f4 100644
--- a/source/DasBlog.Web.UI/Config/site.config
+++ b/source/DasBlog.Web.UI/Config/site.config
@@ -152,6 +152,7 @@
false
false
false
+ false
true
false
false
@@ -172,6 +173,18 @@
+
+
+ eventGrid
+ https://{myns}.westeurope-1.eventgrid.azure.net/api/events
+
+
+ aeg-sas-key
+ xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx
+
+
+
+
true
true
false
diff --git a/source/DasBlog.Web.UI/Controllers/FeedController.cs b/source/DasBlog.Web.UI/Controllers/FeedController.cs
index 3041a3b44..66da17540 100644
--- a/source/DasBlog.Web.UI/Controllers/FeedController.cs
+++ b/source/DasBlog.Web.UI/Controllers/FeedController.cs
@@ -45,7 +45,20 @@ public IActionResult Rss()
}
[Produces("text/xml")]
- [HttpGet("feed/rss/{category}"), HttpHead("feed/rss/{category}")]
+ [HttpGet("feed/rss/{id}"), HttpHead("feed/rss/{id}")]
+ public IActionResult RssItem(string id)
+ {
+ if (!memoryCache.TryGetValue(CACHEKEY_RSS+id, out RssItem rssItem))
+ {
+ rssItem = subscriptionManager.GetRssItem(id);
+
+ memoryCache.Set(CACHEKEY_RSS+id, rssItem, SiteCacheSettings());
+ }
+ return Ok(rssItem);
+ }
+
+ [Produces("text/xml")]
+ [HttpGet("feed/tags/{category}/rss"), HttpHead("feed/tags/{category}/rss")]
public IActionResult RssByCategory(string category)
{
if (!memoryCache.TryGetValue(CACHEKEY_RSS + "_" + category, out RssRoot rss))
@@ -120,7 +133,7 @@ public ActionResult PingBack()
return Ok();
}
- [HttpGet("feed/rss/comments/{entryid}"), HttpHead("feed/rss/comments/{entryid}")]
+ [HttpGet("feed/rss/{entryid}/comments"), HttpHead("feed/rss/{entryid}/comments")]
public ActionResult RssComments(string entryid)
{
return Ok();
diff --git a/source/DasBlog.Web.UI/Settings/DasBlogSettings.cs b/source/DasBlog.Web.UI/Settings/DasBlogSettings.cs
index e211c63bc..3fd8f0858 100644
--- a/source/DasBlog.Web.UI/Settings/DasBlogSettings.cs
+++ b/source/DasBlog.Web.UI/Settings/DasBlogSettings.cs
@@ -117,7 +117,7 @@ public string GetTrackbackUrl(string entryId)
public string GetEntryCommentsRssUrl(string entryId)
{
- return RelativeToRoot(RssUrl + "/comments/" + entryId);
+ return RelativeToRoot(RssUrl + $"/{entryId}/comments/");
}
public string GetCategoryViewUrl(string category)
@@ -132,7 +132,12 @@ public string GetCategoryViewUrlName(string category)
public string GetRssCategoryUrl(string category)
{
- return string.Empty;
+ return RelativeToRoot($"feed/tags/{category}/rss");
+ }
+
+ public string GetRssEntryUrl(string entryId)
+ {
+ return RelativeToRoot($"feed/rss/{entryId}");
}
public User GetUser(string userName)
@@ -323,5 +328,7 @@ public DateTime GetCreateTime(DateTime datetime)
return datetime;
}
+
+
}
}
diff --git a/source/newtelligence.DasBlog.Runtime/CloudEventsTarget.cs b/source/newtelligence.DasBlog.Runtime/CloudEventsTarget.cs
new file mode 100644
index 000000000..2330b35f7
--- /dev/null
+++ b/source/newtelligence.DasBlog.Runtime/CloudEventsTarget.cs
@@ -0,0 +1,143 @@
+#region Copyright (c) 2003, newtelligence AG. All rights reserved.
+/*
+// Copyright (c) 2003, newtelligence AG. (http://www.newtelligence.com)
+// Original BlogX Source Code: Copyright (c) 2003, Chris Anderson (http://simplegeek.com)
+// All rights reserved.
+//
+// Redistribution and use in source and binary forms, with or without modification, are permitted
+// provided that the following conditions are met:
+//
+// (1) Redistributions of source code must retain the above copyright notice, this list of
+// conditions and the following disclaimer.
+// (2) Redistributions in binary form must reproduce the above copyright notice, this list of
+// conditions and the following disclaimer in the documentation and/or other materials
+// provided with the distribution.
+// (3) Neither the name of the newtelligence AG nor the names of its contributors may be used
+// to endorse or promote products derived from this software without specific prior
+// written permission.
+//
+// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS
+// OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
+// AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
+// CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+// DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER
+// IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
+// OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+// -------------------------------------------------------------------------
+//
+// Original BlogX source code (c) 2003 by Chris Anderson (http://simplegeek.com)
+//
+// newtelligence is a registered trademark of newtelligence Aktiengesellschaft.
+//
+// For portions of this software, the some additional copyright notices may apply
+// which can either be found in the license.txt file included in the source distribution
+// or following this notice.
+//
+*/
+#endregion
+
+
+using System;
+using System.Collections.Generic;
+using System.Collections.ObjectModel;
+using System.Xml;
+using System.Xml.Serialization;
+
+
+namespace newtelligence.DasBlog.Runtime
+{
+
+ public class CloudEventsTargetHeader
+ {
+ public string Name { get; set; }
+ public string Value { get; set; }
+ }
+
+ [Serializable]
+ public class CloudEventsTarget
+ {
+ string profileName;
+ string uri;
+ CloudEventsTargetHeader[] headers;
+ CloudEventsTargetHeader[] queryArgs;
+
+ public CloudEventsTarget()
+ {
+ }
+
+ public CloudEventsTarget(string profileName, string uri, CloudEventsTargetHeader[] headers, CloudEventsTargetHeader[] queryArgs)
+ {
+ this.profileName = profileName;
+ this.uri = uri;
+ this.headers = headers;
+ this.queryArgs = queryArgs;
+ }
+
+ [XmlElement("ProfileName")]
+ public string ProfileName { get { return profileName; } set { profileName = value; } }
+ [XmlElement("Uri")]
+ public string Uri { get { return uri; } set { uri = value; } }
+ [XmlArray("QueryArgs")]
+ public CloudEventsTargetHeader[] QueryArgs { get { return queryArgs; } set { queryArgs = value; } }
+ [XmlArray("Headers")]
+ public CloudEventsTargetHeader[] Headers { get { return headers; } set { headers = value; } }
+
+ [XmlAnyElement]
+ public XmlElement[] anyElements;
+ [XmlAnyAttribute]
+ public XmlAttribute[] anyAttributes;
+ }
+
+ ///
+ /// A collection of elements of type CloudEventsSite
+ ///
+ [Serializable]
+ public class CloudEventsTargetCollection: Collection
+ {
+ ///
+ /// Initializes a new empty instance of the CloudEventsSiteCollection class.
+ ///
+ public CloudEventsTargetCollection()
+ :base()
+ {
+ // empty
+ }
+
+ ///
+ /// Initializes a new instance of the CloudEventsSiteCollection class, containing elements
+ /// copied from an array.
+ ///
+ ///
+ /// The array whose elements are to be added to the new CloudEventsSiteCollection.
+ ///
+ public CloudEventsTargetCollection(IList items)
+ :base()
+ {
+ if (items == null) {
+ throw new ArgumentNullException("items");
+ }
+
+ this.AddRange(items);
+ }
+
+ ///
+ /// Adds the elements of an array to the end of this CloudEventsSiteCollection.
+ ///
+ ///
+ /// The array whose elements are to be added to the end of this CloudEventsSiteCollection.
+ ///
+ public virtual void AddRange(IEnumerable items)
+ {
+ if (items == null)
+ {
+ throw new ArgumentNullException("items");
+ }
+
+ foreach (CloudEventsTarget item in items)
+ {
+ this.Add(item);
+ }
+ }
+ }
+}