2014-08-10 22:13:17 +00:00
|
|
|
|
using MediaBrowser.Common.Events;
|
|
|
|
|
using MediaBrowser.Common.Extensions;
|
2014-05-17 04:24:10 +00:00
|
|
|
|
using MediaBrowser.Common.IO;
|
2014-05-07 02:28:19 +00:00
|
|
|
|
using MediaBrowser.Controller.Entities;
|
|
|
|
|
using MediaBrowser.Controller.Entities.Movies;
|
|
|
|
|
using MediaBrowser.Controller.Entities.TV;
|
|
|
|
|
using MediaBrowser.Controller.Library;
|
2014-05-17 04:24:10 +00:00
|
|
|
|
using MediaBrowser.Controller.Persistence;
|
2014-05-11 22:38:10 +00:00
|
|
|
|
using MediaBrowser.Controller.Providers;
|
2014-05-07 02:28:19 +00:00
|
|
|
|
using MediaBrowser.Controller.Subtitles;
|
|
|
|
|
using MediaBrowser.Model.Entities;
|
|
|
|
|
using MediaBrowser.Model.Logging;
|
|
|
|
|
using MediaBrowser.Model.Providers;
|
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.IO;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using System.Threading;
|
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
|
|
|
|
|
namespace MediaBrowser.Providers.Subtitles
|
|
|
|
|
{
|
|
|
|
|
public class SubtitleManager : ISubtitleManager
|
|
|
|
|
{
|
|
|
|
|
private ISubtitleProvider[] _subtitleProviders;
|
|
|
|
|
private readonly ILogger _logger;
|
|
|
|
|
private readonly IFileSystem _fileSystem;
|
|
|
|
|
private readonly ILibraryMonitor _monitor;
|
2014-05-17 04:24:10 +00:00
|
|
|
|
private readonly ILibraryManager _libraryManager;
|
|
|
|
|
private readonly IItemRepository _itemRepo;
|
2014-05-07 02:28:19 +00:00
|
|
|
|
|
2014-08-10 22:13:17 +00:00
|
|
|
|
public event EventHandler<SubtitleDownloadEventArgs> SubtitlesDownloaded;
|
|
|
|
|
public event EventHandler<SubtitleDownloadFailureEventArgs> SubtitleDownloadFailure;
|
|
|
|
|
|
2014-05-17 04:24:10 +00:00
|
|
|
|
public SubtitleManager(ILogger logger, IFileSystem fileSystem, ILibraryMonitor monitor, ILibraryManager libraryManager, IItemRepository itemRepo)
|
2014-05-07 02:28:19 +00:00
|
|
|
|
{
|
|
|
|
|
_logger = logger;
|
|
|
|
|
_fileSystem = fileSystem;
|
|
|
|
|
_monitor = monitor;
|
2014-05-17 04:24:10 +00:00
|
|
|
|
_libraryManager = libraryManager;
|
|
|
|
|
_itemRepo = itemRepo;
|
2014-05-07 02:28:19 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void AddParts(IEnumerable<ISubtitleProvider> subtitleProviders)
|
|
|
|
|
{
|
|
|
|
|
_subtitleProviders = subtitleProviders.ToArray();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async Task<IEnumerable<RemoteSubtitleInfo>> SearchSubtitles(SubtitleSearchRequest request, CancellationToken cancellationToken)
|
|
|
|
|
{
|
2014-05-17 04:24:10 +00:00
|
|
|
|
var contentType = request.ContentType;
|
2014-05-07 02:28:19 +00:00
|
|
|
|
var providers = _subtitleProviders
|
2014-05-17 04:24:10 +00:00
|
|
|
|
.Where(i => i.SupportedMediaTypes.Contains(contentType))
|
2014-05-07 02:28:19 +00:00
|
|
|
|
.ToList();
|
|
|
|
|
|
2014-05-17 04:24:10 +00:00
|
|
|
|
// If not searching all, search one at a time until something is found
|
|
|
|
|
if (!request.SearchAllProviders)
|
|
|
|
|
{
|
|
|
|
|
foreach (var provider in providers)
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var searchResults = await provider.Search(request, cancellationToken).ConfigureAwait(false);
|
|
|
|
|
|
|
|
|
|
var list = searchResults.ToList();
|
|
|
|
|
|
|
|
|
|
if (list.Count > 0)
|
|
|
|
|
{
|
|
|
|
|
Normalize(list);
|
|
|
|
|
return list;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
{
|
|
|
|
|
_logger.ErrorException("Error downloading subtitles from {0}", ex, provider.Name);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return new List<RemoteSubtitleInfo>();
|
|
|
|
|
}
|
|
|
|
|
|
2014-05-07 02:28:19 +00:00
|
|
|
|
var tasks = providers.Select(async i =>
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
2014-05-17 04:24:10 +00:00
|
|
|
|
var searchResults = await i.Search(request, cancellationToken).ConfigureAwait(false);
|
|
|
|
|
|
|
|
|
|
var list = searchResults.ToList();
|
|
|
|
|
Normalize(list);
|
|
|
|
|
return list;
|
2014-05-07 02:28:19 +00:00
|
|
|
|
}
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
{
|
|
|
|
|
_logger.ErrorException("Error downloading subtitles from {0}", ex, i.Name);
|
|
|
|
|
return new List<RemoteSubtitleInfo>();
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
var results = await Task.WhenAll(tasks).ConfigureAwait(false);
|
|
|
|
|
|
|
|
|
|
return results.SelectMany(i => i);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async Task DownloadSubtitles(Video video,
|
|
|
|
|
string subtitleId,
|
|
|
|
|
CancellationToken cancellationToken)
|
|
|
|
|
{
|
2014-08-10 22:13:17 +00:00
|
|
|
|
var parts = subtitleId.Split(new[] { '_' }, 2);
|
|
|
|
|
var provider = GetProvider(parts.First());
|
2014-05-07 02:28:19 +00:00
|
|
|
|
|
2014-08-10 22:13:17 +00:00
|
|
|
|
try
|
2014-05-07 02:28:19 +00:00
|
|
|
|
{
|
2014-08-10 22:13:17 +00:00
|
|
|
|
var response = await GetRemoteSubtitles(subtitleId, cancellationToken).ConfigureAwait(false);
|
2014-05-17 04:24:10 +00:00
|
|
|
|
|
2014-08-10 22:13:17 +00:00
|
|
|
|
using (var stream = response.Stream)
|
2014-05-17 04:24:10 +00:00
|
|
|
|
{
|
2014-08-10 22:13:17 +00:00
|
|
|
|
var savePath = Path.Combine(Path.GetDirectoryName(video.Path),
|
|
|
|
|
_fileSystem.GetFileNameWithoutExtension(video.Path) + "." + response.Language.ToLower());
|
2014-05-17 04:24:10 +00:00
|
|
|
|
|
2014-08-10 22:13:17 +00:00
|
|
|
|
if (response.IsForced)
|
|
|
|
|
{
|
|
|
|
|
savePath += ".forced";
|
|
|
|
|
}
|
2014-05-07 02:28:19 +00:00
|
|
|
|
|
2014-08-10 22:13:17 +00:00
|
|
|
|
savePath += "." + response.Format.ToLower();
|
2014-05-07 02:28:19 +00:00
|
|
|
|
|
2014-08-10 22:13:17 +00:00
|
|
|
|
_logger.Info("Saving subtitles to {0}", savePath);
|
2014-05-07 02:28:19 +00:00
|
|
|
|
|
2014-08-10 22:13:17 +00:00
|
|
|
|
_monitor.ReportFileSystemChangeBeginning(savePath);
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
using (var fs = _fileSystem.GetFileStream(savePath, FileMode.Create, FileAccess.Write, FileShare.Read, true))
|
|
|
|
|
{
|
|
|
|
|
await stream.CopyToAsync(fs).ConfigureAwait(false);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
EventHelper.FireEventIfNotNull(SubtitlesDownloaded, this, new SubtitleDownloadEventArgs
|
|
|
|
|
{
|
|
|
|
|
Item = video,
|
|
|
|
|
Format = response.Format,
|
|
|
|
|
Language = response.Language,
|
|
|
|
|
IsForced = response.IsForced,
|
|
|
|
|
Provider = provider.Name
|
|
|
|
|
|
|
|
|
|
}, _logger);
|
|
|
|
|
}
|
|
|
|
|
finally
|
2014-05-07 02:28:19 +00:00
|
|
|
|
{
|
2014-08-10 22:13:17 +00:00
|
|
|
|
_monitor.ReportFileSystemChangeComplete(savePath, false);
|
2014-05-07 02:28:19 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2014-08-10 22:13:17 +00:00
|
|
|
|
}
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
{
|
|
|
|
|
EventHelper.FireEventIfNotNull(SubtitleDownloadFailure, this, new SubtitleDownloadFailureEventArgs
|
2014-05-07 02:28:19 +00:00
|
|
|
|
{
|
2014-08-10 22:13:17 +00:00
|
|
|
|
Item = video,
|
|
|
|
|
Exception = ex,
|
|
|
|
|
Provider = provider.Name
|
|
|
|
|
|
|
|
|
|
}, _logger);
|
|
|
|
|
|
|
|
|
|
throw;
|
2014-05-07 02:28:19 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Task<IEnumerable<RemoteSubtitleInfo>> SearchSubtitles(Video video, string language, CancellationToken cancellationToken)
|
|
|
|
|
{
|
|
|
|
|
if (video.LocationType != LocationType.FileSystem ||
|
|
|
|
|
video.VideoType != VideoType.VideoFile)
|
|
|
|
|
{
|
|
|
|
|
return Task.FromResult<IEnumerable<RemoteSubtitleInfo>>(new List<RemoteSubtitleInfo>());
|
|
|
|
|
}
|
|
|
|
|
|
2014-05-11 22:38:10 +00:00
|
|
|
|
VideoContentType mediaType;
|
2014-05-07 02:28:19 +00:00
|
|
|
|
|
|
|
|
|
if (video is Episode)
|
|
|
|
|
{
|
2014-05-11 22:38:10 +00:00
|
|
|
|
mediaType = VideoContentType.Episode;
|
2014-05-07 02:28:19 +00:00
|
|
|
|
}
|
|
|
|
|
else if (video is Movie)
|
|
|
|
|
{
|
2014-05-11 22:38:10 +00:00
|
|
|
|
mediaType = VideoContentType.Movie;
|
2014-05-07 02:28:19 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
// These are the only supported types
|
|
|
|
|
return Task.FromResult<IEnumerable<RemoteSubtitleInfo>>(new List<RemoteSubtitleInfo>());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var request = new SubtitleSearchRequest
|
|
|
|
|
{
|
|
|
|
|
ContentType = mediaType,
|
|
|
|
|
IndexNumber = video.IndexNumber,
|
|
|
|
|
Language = language,
|
|
|
|
|
MediaPath = video.Path,
|
|
|
|
|
Name = video.Name,
|
|
|
|
|
ParentIndexNumber = video.ParentIndexNumber,
|
|
|
|
|
ProductionYear = video.ProductionYear,
|
2014-05-11 22:38:10 +00:00
|
|
|
|
ProviderIds = video.ProviderIds,
|
|
|
|
|
RuntimeTicks = video.RunTimeTicks
|
2014-05-07 02:28:19 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
var episode = video as Episode;
|
|
|
|
|
|
|
|
|
|
if (episode != null)
|
|
|
|
|
{
|
|
|
|
|
request.IndexNumberEnd = episode.IndexNumberEnd;
|
|
|
|
|
request.SeriesName = episode.SeriesName;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return SearchSubtitles(request, cancellationToken);
|
|
|
|
|
}
|
2014-05-17 04:24:10 +00:00
|
|
|
|
|
|
|
|
|
private void Normalize(IEnumerable<RemoteSubtitleInfo> subtitles)
|
|
|
|
|
{
|
|
|
|
|
foreach (var sub in subtitles)
|
|
|
|
|
{
|
|
|
|
|
sub.Id = GetProviderId(sub.ProviderName) + "_" + sub.Id;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private string GetProviderId(string name)
|
|
|
|
|
{
|
|
|
|
|
return name.ToLower().GetMD5().ToString("N");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private ISubtitleProvider GetProvider(string id)
|
|
|
|
|
{
|
|
|
|
|
return _subtitleProviders.First(i => string.Equals(id, GetProviderId(i.Name)));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Task DeleteSubtitles(string itemId, int index)
|
|
|
|
|
{
|
|
|
|
|
var stream = _itemRepo.GetMediaStreams(new MediaStreamQuery
|
|
|
|
|
{
|
|
|
|
|
Index = index,
|
|
|
|
|
ItemId = new Guid(itemId),
|
|
|
|
|
Type = MediaStreamType.Subtitle
|
|
|
|
|
|
|
|
|
|
}).First();
|
|
|
|
|
|
|
|
|
|
var path = stream.Path;
|
|
|
|
|
_monitor.ReportFileSystemChangeBeginning(path);
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
2015-01-13 03:46:44 +00:00
|
|
|
|
_fileSystem.DeleteFile(path);
|
2014-05-17 04:24:10 +00:00
|
|
|
|
}
|
|
|
|
|
finally
|
|
|
|
|
{
|
|
|
|
|
_monitor.ReportFileSystemChangeComplete(path, false);
|
|
|
|
|
}
|
|
|
|
|
|
2014-10-25 18:32:58 +00:00
|
|
|
|
return _libraryManager.GetItemById(itemId).RefreshMetadata(new MetadataRefreshOptions(new DirectoryService())
|
2014-05-17 04:24:10 +00:00
|
|
|
|
{
|
|
|
|
|
ImageRefreshMode = ImageRefreshMode.ValidationOnly,
|
|
|
|
|
MetadataRefreshMode = MetadataRefreshMode.ValidationOnly
|
|
|
|
|
|
|
|
|
|
}, CancellationToken.None);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Task<SubtitleResponse> GetRemoteSubtitles(string id, CancellationToken cancellationToken)
|
|
|
|
|
{
|
|
|
|
|
var parts = id.Split(new[] { '_' }, 2);
|
|
|
|
|
|
|
|
|
|
var provider = GetProvider(parts.First());
|
|
|
|
|
id = parts.Last();
|
|
|
|
|
|
|
|
|
|
return provider.GetSubtitles(id, cancellationToken);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public IEnumerable<SubtitleProviderInfo> GetProviders(string itemId)
|
|
|
|
|
{
|
|
|
|
|
var video = _libraryManager.GetItemById(itemId) as Video;
|
|
|
|
|
VideoContentType mediaType;
|
|
|
|
|
|
|
|
|
|
if (video is Episode)
|
|
|
|
|
{
|
|
|
|
|
mediaType = VideoContentType.Episode;
|
|
|
|
|
}
|
|
|
|
|
else if (video is Movie)
|
|
|
|
|
{
|
|
|
|
|
mediaType = VideoContentType.Movie;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
// These are the only supported types
|
|
|
|
|
return new List<SubtitleProviderInfo>();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var providers = _subtitleProviders
|
|
|
|
|
.Where(i => i.SupportedMediaTypes.Contains(mediaType))
|
|
|
|
|
.ToList();
|
|
|
|
|
|
|
|
|
|
return providers.Select(i => new SubtitleProviderInfo
|
|
|
|
|
{
|
|
|
|
|
Name = i.Name,
|
|
|
|
|
Id = GetProviderId(i.Name)
|
|
|
|
|
});
|
|
|
|
|
}
|
2014-08-10 22:13:17 +00:00
|
|
|
|
|
2014-05-07 02:28:19 +00:00
|
|
|
|
}
|
|
|
|
|
}
|