aboutsummaryrefslogtreecommitdiff
path: root/MediaBrowser.Server.Implementations
diff options
context:
space:
mode:
authorLuke Pulverenti <luke.pulverenti@gmail.com>2014-01-18 23:25:01 -0500
committerLuke Pulverenti <luke.pulverenti@gmail.com>2014-01-18 23:25:01 -0500
commitd2cae4012853bb6457554516f06e5bbf11121b8d (patch)
tree7e3dbc416fc99266310635dd85b0ce7962054807 /MediaBrowser.Server.Implementations
parent18a9720857ec9e6cdd580b3812bb72cf864d2f20 (diff)
rework news downloading
Diffstat (limited to 'MediaBrowser.Server.Implementations')
-rw-r--r--MediaBrowser.Server.Implementations/LiveTv/LiveTvManager.cs26
-rw-r--r--MediaBrowser.Server.Implementations/MediaBrowser.Server.Implementations.csproj1
-rw-r--r--MediaBrowser.Server.Implementations/News/NewsEntryPoint.cs127
-rw-r--r--MediaBrowser.Server.Implementations/News/NewsService.cs108
4 files changed, 176 insertions, 86 deletions
diff --git a/MediaBrowser.Server.Implementations/LiveTv/LiveTvManager.cs b/MediaBrowser.Server.Implementations/LiveTv/LiveTvManager.cs
index 8577e7510..f62efd9da 100644
--- a/MediaBrowser.Server.Implementations/LiveTv/LiveTvManager.cs
+++ b/MediaBrowser.Server.Implementations/LiveTv/LiveTvManager.cs
@@ -229,6 +229,12 @@ namespace MediaBrowser.Server.Implementations.LiveTv
return result;
}
+ catch (Exception ex)
+ {
+ _logger.ErrorException("Error getting recording stream", ex);
+
+ throw;
+ }
finally
{
_liveStreamSemaphore.Release();
@@ -245,6 +251,8 @@ namespace MediaBrowser.Server.Implementations.LiveTv
var channel = GetInternalChannel(id);
+ _logger.Info("Opening channel stream from {0}, external channel Id: {1}", service.Name, channel.ChannelInfo.Id);
+
var result = await service.GetChannelStream(channel.ChannelInfo.Id, cancellationToken).ConfigureAwait(false);
if (!string.IsNullOrEmpty(result.Id))
@@ -254,6 +262,12 @@ namespace MediaBrowser.Server.Implementations.LiveTv
return result;
}
+ catch (Exception ex)
+ {
+ _logger.ErrorException("Error getting channel stream", ex);
+
+ throw;
+ }
finally
{
_liveStreamSemaphore.Release();
@@ -1261,9 +1275,19 @@ namespace MediaBrowser.Server.Implementations.LiveTv
{
await _liveStreamSemaphore.WaitAsync(cancellationToken).ConfigureAwait(false);
+ var service = ActiveService;
+
+ _logger.Info("Closing live stream from {0}, stream Id: {1}", service.Name, id);
+
try
{
- await ActiveService.CloseLiveStream(id, cancellationToken).ConfigureAwait(false);
+ await service.CloseLiveStream(id, cancellationToken).ConfigureAwait(false);
+ }
+ catch (Exception ex)
+ {
+ _logger.ErrorException("Error closing live stream", ex);
+
+ throw;
}
finally
{
diff --git a/MediaBrowser.Server.Implementations/MediaBrowser.Server.Implementations.csproj b/MediaBrowser.Server.Implementations/MediaBrowser.Server.Implementations.csproj
index 907301699..855183b97 100644
--- a/MediaBrowser.Server.Implementations/MediaBrowser.Server.Implementations.csproj
+++ b/MediaBrowser.Server.Implementations/MediaBrowser.Server.Implementations.csproj
@@ -173,6 +173,7 @@
<Compile Include="LiveTv\RefreshChannelsScheduledTask.cs" />
<Compile Include="Localization\LocalizationManager.cs" />
<Compile Include="MediaEncoder\MediaEncoder.cs" />
+ <Compile Include="News\NewsEntryPoint.cs" />
<Compile Include="News\NewsService.cs" />
<Compile Include="Persistence\SqliteChapterRepository.cs" />
<Compile Include="Persistence\SqliteExtensions.cs" />
diff --git a/MediaBrowser.Server.Implementations/News/NewsEntryPoint.cs b/MediaBrowser.Server.Implementations/News/NewsEntryPoint.cs
new file mode 100644
index 000000000..bb6ae5a90
--- /dev/null
+++ b/MediaBrowser.Server.Implementations/News/NewsEntryPoint.cs
@@ -0,0 +1,127 @@
+using MediaBrowser.Common.Configuration;
+using MediaBrowser.Common.Extensions;
+using MediaBrowser.Common.IO;
+using MediaBrowser.Common.Net;
+using MediaBrowser.Controller.Plugins;
+using MediaBrowser.Model.Logging;
+using MediaBrowser.Model.News;
+using MediaBrowser.Model.Serialization;
+using System;
+using System.Collections.Generic;
+using System.IO;
+using System.Linq;
+using System.Threading;
+using System.Threading.Tasks;
+using System.Xml;
+
+namespace MediaBrowser.Server.Implementations.News
+{
+ public class NewsEntryPoint : IServerEntryPoint
+ {
+ private Timer _timer;
+ private readonly IHttpClient _httpClient;
+ private readonly IApplicationPaths _appPaths;
+ private readonly IFileSystem _fileSystem;
+ private readonly ILogger _logger;
+ private readonly IJsonSerializer _json;
+
+ private readonly TimeSpan _frequency = TimeSpan.FromHours(24);
+
+ public NewsEntryPoint(IHttpClient httpClient, IApplicationPaths appPaths, IFileSystem fileSystem, ILogger logger, IJsonSerializer json)
+ {
+ _httpClient = httpClient;
+ _appPaths = appPaths;
+ _fileSystem = fileSystem;
+ _logger = logger;
+ _json = json;
+ }
+
+ public void Run()
+ {
+ _timer = new Timer(OnTimerFired, null, TimeSpan.FromMilliseconds(500), _frequency);
+ }
+
+ /// <summary>
+ /// Called when [timer fired].
+ /// </summary>
+ /// <param name="state">The state.</param>
+ private async void OnTimerFired(object state)
+ {
+ var path = Path.Combine(_appPaths.CachePath, "news.json");
+
+ try
+ {
+ await DownloadNews(path).ConfigureAwait(false);
+ }
+ catch (Exception ex)
+ {
+ _logger.ErrorException("Error downloading news", ex);
+ }
+ }
+
+ private async Task DownloadNews(string path)
+ {
+ var requestOptions = new HttpRequestOptions
+ {
+ Url = "http://mediabrowser3.com/community/index.php?/blog/rss/1-media-browser-developers-blog",
+ Progress = new Progress<double>()
+ };
+
+ using (var stream = await _httpClient.Get(requestOptions).ConfigureAwait(false))
+ {
+ var doc = new XmlDocument();
+ doc.Load(stream);
+
+ var news = ParseRssItems(doc).ToList();
+
+ _json.SerializeToFile(news, path);
+ }
+ }
+
+ private IEnumerable<NewsItem> ParseRssItems(XmlDocument xmlDoc)
+ {
+ var nodes = xmlDoc.SelectNodes("rss/channel/item");
+
+ if (nodes != null)
+ {
+ foreach (XmlNode node in nodes)
+ {
+ var newsItem = new NewsItem();
+
+ newsItem.Title = ParseDocElements(node, "title");
+
+ newsItem.DescriptionHtml = ParseDocElements(node, "description");
+ newsItem.Description = newsItem.DescriptionHtml.StripHtml();
+
+ newsItem.Link = ParseDocElements(node, "link");
+
+ var date = ParseDocElements(node, "pubDate");
+ DateTime parsedDate;
+
+ if (DateTime.TryParse(date, out parsedDate))
+ {
+ newsItem.Date = parsedDate;
+ }
+
+ yield return newsItem;
+ }
+ }
+ }
+
+ private string ParseDocElements(XmlNode parent, string xPath)
+ {
+ var node = parent.SelectSingleNode(xPath);
+
+ return node != null ? node.InnerText : string.Empty;
+ }
+
+ public void Dispose()
+ {
+ if (_timer != null)
+ {
+ _timer.Dispose();
+ _timer = null;
+ }
+ }
+ }
+}
diff --git a/MediaBrowser.Server.Implementations/News/NewsService.cs b/MediaBrowser.Server.Implementations/News/NewsService.cs
index 9849d538a..34da857dd 100644
--- a/MediaBrowser.Server.Implementations/News/NewsService.cs
+++ b/MediaBrowser.Server.Implementations/News/NewsService.cs
@@ -1,38 +1,46 @@
using MediaBrowser.Common.Configuration;
-using MediaBrowser.Common.IO;
-using MediaBrowser.Common.Net;
using MediaBrowser.Controller.News;
using MediaBrowser.Model.News;
using MediaBrowser.Model.Querying;
-using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
-using System.Threading.Tasks;
-using System.Xml;
+using MediaBrowser.Model.Serialization;
namespace MediaBrowser.Server.Implementations.News
{
public class NewsService : INewsService
{
private readonly IApplicationPaths _appPaths;
- private readonly IFileSystem _fileSystem;
- private readonly IHttpClient _httpClient;
+ private readonly IJsonSerializer _json;
- public NewsService(IApplicationPaths appPaths, IFileSystem fileSystem, IHttpClient httpClient)
+ public NewsService(IApplicationPaths appPaths, IJsonSerializer json)
{
_appPaths = appPaths;
- _fileSystem = fileSystem;
- _httpClient = httpClient;
+ _json = json;
}
- public async Task<QueryResult<NewsItem>> GetProductNews(NewsQuery query)
+ public QueryResult<NewsItem> GetProductNews(NewsQuery query)
{
- var path = Path.Combine(_appPaths.CachePath, "news.xml");
+ try
+ {
+ return GetProductNewsInternal(query);
+ }
+ catch (FileNotFoundException)
+ {
+ // No biggie
+ return new QueryResult<NewsItem>
+ {
+ Items = new NewsItem[] { }
+ };
+ }
+ }
- await EnsureNewsFile(path).ConfigureAwait(false);
+ private QueryResult<NewsItem> GetProductNewsInternal(NewsQuery query)
+ {
+ var path = Path.Combine(_appPaths.CachePath, "news.json");
- var items = GetNewsItems(path);
+ var items = GetNewsItems(path).OrderByDescending(i => i.Date);
var itemsArray = items.ToArray();
var count = itemsArray.Length;
@@ -56,77 +64,7 @@ namespace MediaBrowser.Server.Implementations.News
private IEnumerable<NewsItem> GetNewsItems(string path)
{
- var xmlDoc = new XmlDocument();
-
- xmlDoc.Load(path);
-
- return ParseRssItems(xmlDoc);
- }
-
- private IEnumerable<NewsItem> ParseRssItems(XmlDocument xmlDoc)
- {
- var nodes = xmlDoc.SelectNodes("rss/channel/item");
-
- if (nodes == null)
- {
- yield return null;
- }
-
- foreach (XmlNode node in nodes)
- {
- var newsItem = new NewsItem();
-
- newsItem.Title = ParseDocElements(node, "title");
-
- newsItem.Description = ParseDocElements(node, "description");
-
- newsItem.Link = ParseDocElements(node, "link");
-
- var date = ParseDocElements(node, "pubDate");
- DateTime parsedDate;
-
- if (DateTime.TryParse(date, out parsedDate))
- {
- newsItem.Date = parsedDate;
- }
-
- yield return newsItem;
- }
- }
-
- private string ParseDocElements(XmlNode parent, string xPath)
- {
- var node = parent.SelectSingleNode(xPath);
-
- return node != null ? node.InnerText : "Unresolvable";
- }
-
-
- /// <summary>
- /// Ensures the news file.
- /// </summary>
- /// <param name="path">The path.</param>
- /// <returns>Task.</returns>
- private async Task EnsureNewsFile(string path)
- {
- var info = _fileSystem.GetFileSystemInfo(path);
-
- if (!info.Exists || (DateTime.UtcNow - _fileSystem.GetLastWriteTimeUtc(info)).TotalHours > 12)
- {
- var requestOptions = new HttpRequestOptions
- {
- Url = "http://mediabrowser3.com/community/index.php?/blog/rss/1-media-browser-developers-blog",
- Progress = new Progress<double>()
- };
-
- using (var stream = await _httpClient.Get(requestOptions).ConfigureAwait(false))
- {
- using (var fileStream = _fileSystem.GetFileStream(path, FileMode.Create, FileAccess.Write, FileShare.Read, true))
- {
- await stream.CopyToAsync(fileStream).ConfigureAwait(false);
- }
- }
- }
+ return _json.DeserializeFromFile<List<NewsItem>>(path);
}
}
}