2015-08-02 23:47:31 +00:00
using MediaBrowser.Common ;
2015-07-20 18:32:55 +00:00
using MediaBrowser.Common.Configuration ;
using MediaBrowser.Common.Net ;
2015-08-21 19:25:35 +00:00
using MediaBrowser.Common.Security ;
2015-08-22 19:46:55 +00:00
using MediaBrowser.Controller.Configuration ;
2015-07-20 18:32:55 +00:00
using MediaBrowser.Controller.Drawing ;
2015-08-22 19:46:55 +00:00
using MediaBrowser.Controller.FileOrganization ;
using MediaBrowser.Controller.Library ;
2015-07-20 18:32:55 +00:00
using MediaBrowser.Controller.LiveTv ;
2015-09-21 16:26:05 +00:00
using MediaBrowser.Controller.MediaEncoding ;
2015-08-22 19:46:55 +00:00
using MediaBrowser.Controller.Providers ;
2015-07-20 18:32:55 +00:00
using MediaBrowser.Model.Dto ;
2015-08-21 19:25:35 +00:00
using MediaBrowser.Model.Entities ;
2015-07-20 18:32:55 +00:00
using MediaBrowser.Model.Events ;
2015-08-22 19:46:55 +00:00
using MediaBrowser.Model.FileOrganization ;
2015-07-20 18:32:55 +00:00
using MediaBrowser.Model.LiveTv ;
using MediaBrowser.Model.Logging ;
using MediaBrowser.Model.Serialization ;
2015-08-22 19:46:55 +00:00
using MediaBrowser.Server.Implementations.FileOrganization ;
2015-07-20 18:32:55 +00:00
using System ;
using System.Collections.Concurrent ;
using System.Collections.Generic ;
2016-01-26 18:18:54 +00:00
using System.Globalization ;
2015-07-20 18:32:55 +00:00
using System.IO ;
using System.Linq ;
using System.Threading ;
using System.Threading.Tasks ;
2015-10-04 04:23:11 +00:00
using CommonIO ;
2016-05-04 20:50:47 +00:00
using MediaBrowser.Common.Events ;
2015-10-01 16:28:24 +00:00
using MediaBrowser.Common.Extensions ;
2016-05-04 20:50:47 +00:00
using MediaBrowser.Controller.Entities ;
using MediaBrowser.Controller.Entities.TV ;
2016-01-21 17:29:14 +00:00
using MediaBrowser.Controller.Power ;
2016-01-27 06:07:01 +00:00
using Microsoft.Win32 ;
2015-07-20 18:32:55 +00:00
namespace MediaBrowser.Server.Implementations.LiveTv.EmbyTV
{
2016-06-08 21:04:52 +00:00
public class EmbyTV : ILiveTvService , ISupportsNewTimerIds , IHasRegistrationInfo , IDisposable
2015-07-20 18:32:55 +00:00
{
2015-07-21 04:22:46 +00:00
private readonly IApplicationHost _appHpst ;
2015-07-20 18:32:55 +00:00
private readonly ILogger _logger ;
private readonly IHttpClient _httpClient ;
2015-08-22 19:46:55 +00:00
private readonly IServerConfigurationManager _config ;
2015-07-20 18:32:55 +00:00
private readonly IJsonSerializer _jsonSerializer ;
private readonly ItemDataProvider < SeriesTimerInfo > _seriesTimerProvider ;
private readonly TimerManager _timerProvider ;
2015-07-23 13:23:22 +00:00
private readonly LiveTvManager _liveTvManager ;
2015-08-16 18:54:25 +00:00
private readonly IFileSystem _fileSystem ;
2015-08-21 19:25:35 +00:00
private readonly ISecurityManager _security ;
2015-07-23 05:25:55 +00:00
2015-08-22 19:46:55 +00:00
private readonly ILibraryMonitor _libraryMonitor ;
private readonly ILibraryManager _libraryManager ;
private readonly IProviderManager _providerManager ;
private readonly IFileOrganizationService _organizationService ;
2015-09-21 16:26:05 +00:00
private readonly IMediaEncoder _mediaEncoder ;
2015-08-22 19:46:55 +00:00
2015-07-29 03:42:03 +00:00
public static EmbyTV Current ;
2016-05-04 20:50:47 +00:00
public event EventHandler DataSourceChanged ;
public event EventHandler < RecordingStatusChangedEventArgs > RecordingStatusChanged ;
private readonly ConcurrentDictionary < string , ActiveRecordingInfo > _activeRecordings =
new ConcurrentDictionary < string , ActiveRecordingInfo > ( StringComparer . OrdinalIgnoreCase ) ;
2016-01-21 17:29:14 +00:00
public EmbyTV ( IApplicationHost appHost , ILogger logger , IJsonSerializer jsonSerializer , IHttpClient httpClient , IServerConfigurationManager config , ILiveTvManager liveTvManager , IFileSystem fileSystem , ISecurityManager security , ILibraryManager libraryManager , ILibraryMonitor libraryMonitor , IProviderManager providerManager , IFileOrganizationService organizationService , IMediaEncoder mediaEncoder , IPowerManagement powerManagement )
2015-07-20 18:32:55 +00:00
{
2015-07-29 03:42:03 +00:00
Current = this ;
2015-07-21 04:22:46 +00:00
_appHpst = appHost ;
2015-07-20 18:32:55 +00:00
_logger = logger ;
_httpClient = httpClient ;
_config = config ;
2015-08-16 18:54:25 +00:00
_fileSystem = fileSystem ;
2015-08-21 19:25:35 +00:00
_security = security ;
2015-08-22 19:46:55 +00:00
_libraryManager = libraryManager ;
_libraryMonitor = libraryMonitor ;
_providerManager = providerManager ;
_organizationService = organizationService ;
2015-09-21 16:26:05 +00:00
_mediaEncoder = mediaEncoder ;
2015-07-23 13:23:22 +00:00
_liveTvManager = ( LiveTvManager ) liveTvManager ;
2015-07-20 18:32:55 +00:00
_jsonSerializer = jsonSerializer ;
2015-09-21 16:26:05 +00:00
_seriesTimerProvider = new SeriesTimerManager ( fileSystem , jsonSerializer , _logger , Path . Combine ( DataPath , "seriestimers" ) ) ;
2016-01-21 17:29:14 +00:00
_timerProvider = new TimerManager ( fileSystem , jsonSerializer , _logger , Path . Combine ( DataPath , "timers" ) , powerManagement , _logger ) ;
2015-07-20 18:32:55 +00:00
_timerProvider . TimerFired + = _timerProvider_TimerFired ;
2016-05-04 20:50:47 +00:00
_config . NamedConfigurationUpdated + = _config_NamedConfigurationUpdated ;
}
private void _config_NamedConfigurationUpdated ( object sender , ConfigurationUpdateEventArgs e )
{
if ( string . Equals ( e . Key , "livetv" , StringComparison . OrdinalIgnoreCase ) )
{
OnRecordingFoldersChanged ( ) ;
}
2015-07-20 18:32:55 +00:00
}
2015-07-29 03:42:03 +00:00
public void Start ( )
{
_timerProvider . RestartTimers ( ) ;
2016-01-27 06:07:01 +00:00
SystemEvents . PowerModeChanged + = SystemEvents_PowerModeChanged ;
2016-05-23 04:47:02 +00:00
CreateRecordingFolders ( ) ;
2016-05-04 20:50:47 +00:00
}
private void OnRecordingFoldersChanged ( )
{
CreateRecordingFolders ( ) ;
}
2016-05-22 17:07:30 +00:00
internal void CreateRecordingFolders ( )
{
try
{
CreateRecordingFoldersInternal ( ) ;
}
catch ( Exception ex )
{
_logger . ErrorException ( "Error creating recording folders" , ex ) ;
}
}
internal void CreateRecordingFoldersInternal ( )
2016-05-04 20:50:47 +00:00
{
var recordingFolders = GetRecordingFolders ( ) ;
var virtualFolders = _libraryManager . GetVirtualFolders ( )
. ToList ( ) ;
var allExistingPaths = virtualFolders . SelectMany ( i = > i . Locations ) . ToList ( ) ;
2016-05-20 15:57:07 +00:00
var pathsAdded = new List < string > ( ) ;
2016-05-04 20:50:47 +00:00
foreach ( var recordingFolder in recordingFolders )
{
var pathsToCreate = recordingFolder . Locations
. Where ( i = > ! allExistingPaths . Contains ( i , StringComparer . OrdinalIgnoreCase ) )
. ToList ( ) ;
if ( pathsToCreate . Count = = 0 )
{
continue ;
}
try
{
_libraryManager . AddVirtualFolder ( recordingFolder . Name , recordingFolder . CollectionType , pathsToCreate . ToArray ( ) , true ) ;
}
catch ( Exception ex )
{
_logger . ErrorException ( "Error creating virtual folder" , ex ) ;
}
2016-05-20 15:57:07 +00:00
pathsAdded . AddRange ( pathsToCreate ) ;
}
var config = GetConfiguration ( ) ;
var pathsToRemove = config . MediaLocationsCreated
. Except ( recordingFolders . SelectMany ( i = > i . Locations ) )
. ToList ( ) ;
if ( pathsAdded . Count > 0 | | pathsToRemove . Count > 0 )
{
pathsAdded . InsertRange ( 0 , config . MediaLocationsCreated ) ;
config . MediaLocationsCreated = pathsAdded . Except ( pathsToRemove ) . Distinct ( StringComparer . OrdinalIgnoreCase ) . ToArray ( ) ;
_config . SaveConfiguration ( "livetv" , config ) ;
}
foreach ( var path in pathsToRemove )
{
RemovePathFromLibrary ( path ) ;
2016-05-04 20:50:47 +00:00
}
}
private void RemovePathFromLibrary ( string path )
{
2016-05-20 15:57:07 +00:00
_logger . Debug ( "Removing path from library: {0}" , path ) ;
2016-05-04 20:50:47 +00:00
var requiresRefresh = false ;
var virtualFolders = _libraryManager . GetVirtualFolders ( )
. ToList ( ) ;
foreach ( var virtualFolder in virtualFolders )
{
if ( ! virtualFolder . Locations . Contains ( path , StringComparer . OrdinalIgnoreCase ) )
{
continue ;
}
if ( virtualFolder . Locations . Count = = 1 )
{
// remove entire virtual folder
try
{
_libraryManager . RemoveVirtualFolder ( virtualFolder . Name , true ) ;
}
catch ( Exception ex )
{
_logger . ErrorException ( "Error removing virtual folder" , ex ) ;
}
}
else
{
try
{
_libraryManager . RemoveMediaPath ( virtualFolder . Name , path ) ;
requiresRefresh = true ;
}
catch ( Exception ex )
{
_logger . ErrorException ( "Error removing media path" , ex ) ;
}
}
}
if ( requiresRefresh )
{
_libraryManager . ValidateMediaLibrary ( new Progress < Double > ( ) , CancellationToken . None ) ;
}
2016-01-27 06:07:01 +00:00
}
void SystemEvents_PowerModeChanged ( object sender , PowerModeChangedEventArgs e )
{
2016-01-27 16:58:08 +00:00
_logger . Info ( "Power mode changed to {0}" , e . Mode ) ;
2016-01-27 06:07:01 +00:00
if ( e . Mode = = PowerModes . Resume )
{
_timerProvider . RestartTimers ( ) ;
}
2015-07-29 03:42:03 +00:00
}
2015-07-20 18:32:55 +00:00
public string Name
{
get { return "Emby" ; }
}
public string DataPath
{
get { return Path . Combine ( _config . CommonApplicationPaths . DataPath , "livetv" ) ; }
}
2016-05-04 20:50:47 +00:00
private string DefaultRecordingPath
{
get
{
return Path . Combine ( DataPath , "recordings" ) ;
}
}
private string RecordingPath
{
get
{
var path = GetConfiguration ( ) . RecordingPath ;
return string . IsNullOrWhiteSpace ( path )
? DefaultRecordingPath
: path ;
}
}
2015-07-20 18:32:55 +00:00
public string HomePageUrl
{
get { return "http://emby.media" ; }
}
public async Task < LiveTvServiceStatusInfo > GetStatusInfoAsync ( CancellationToken cancellationToken )
{
var status = new LiveTvServiceStatusInfo ( ) ;
var list = new List < LiveTvTunerInfo > ( ) ;
2015-08-19 16:43:23 +00:00
foreach ( var hostInstance in _liveTvManager . TunerHosts )
2015-07-20 18:32:55 +00:00
{
2015-07-23 23:40:54 +00:00
try
2015-07-20 18:32:55 +00:00
{
2015-08-19 16:43:23 +00:00
var tuners = await hostInstance . GetTunerInfos ( cancellationToken ) . ConfigureAwait ( false ) ;
2015-07-20 18:32:55 +00:00
2015-07-23 23:40:54 +00:00
list . AddRange ( tuners ) ;
}
catch ( Exception ex )
{
_logger . ErrorException ( "Error getting tuners" , ex ) ;
2015-07-20 18:32:55 +00:00
}
}
status . Tuners = list ;
status . Status = LiveTvServiceStatus . Ok ;
2015-07-21 04:22:46 +00:00
status . Version = _appHpst . ApplicationVersion . ToString ( ) ;
status . IsVisible = false ;
2015-07-20 18:32:55 +00:00
return status ;
}
2015-09-21 16:26:05 +00:00
public async Task RefreshSeriesTimers ( CancellationToken cancellationToken , IProgress < double > progress )
{
2015-11-21 04:34:55 +00:00
var seriesTimers = await GetSeriesTimersAsync ( cancellationToken ) . ConfigureAwait ( false ) ;
2015-09-21 16:26:05 +00:00
List < ChannelInfo > channels = null ;
2015-11-21 04:34:55 +00:00
foreach ( var timer in seriesTimers )
2015-09-21 16:26:05 +00:00
{
List < ProgramInfo > epgData ;
if ( timer . RecordAnyChannel )
{
if ( channels = = null )
{
channels = ( await GetChannelsAsync ( true , CancellationToken . None ) . ConfigureAwait ( false ) ) . ToList ( ) ;
}
var channelIds = channels . Select ( i = > i . Id ) . ToList ( ) ;
epgData = GetEpgDataForChannels ( channelIds ) ;
}
else
{
epgData = GetEpgDataForChannel ( timer . ChannelId ) ;
}
2016-02-07 19:19:41 +00:00
await UpdateTimersForSeriesTimer ( epgData , timer , true ) . ConfigureAwait ( false ) ;
2015-09-21 16:26:05 +00:00
}
2015-11-21 04:34:55 +00:00
var timers = await GetTimersAsync ( cancellationToken ) . ConfigureAwait ( false ) ;
foreach ( var timer in timers . ToList ( ) )
{
if ( DateTime . UtcNow > timer . EndDate & & ! _activeRecordings . ContainsKey ( timer . Id ) )
{
_timerProvider . Delete ( timer ) ;
}
}
2015-09-21 16:26:05 +00:00
}
2015-08-16 18:54:25 +00:00
private List < ChannelInfo > _channelCache = null ;
private async Task < IEnumerable < ChannelInfo > > GetChannelsAsync ( bool enableCache , CancellationToken cancellationToken )
2015-07-20 18:32:55 +00:00
{
2015-08-16 18:54:25 +00:00
if ( enableCache & & _channelCache ! = null )
{
return _channelCache . ToList ( ) ;
}
2015-07-20 18:32:55 +00:00
var list = new List < ChannelInfo > ( ) ;
2015-08-19 16:43:23 +00:00
foreach ( var hostInstance in _liveTvManager . TunerHosts )
2015-07-20 18:32:55 +00:00
{
2015-07-23 23:40:54 +00:00
try
2015-07-20 18:32:55 +00:00
{
2015-08-19 16:43:23 +00:00
var channels = await hostInstance . GetChannels ( cancellationToken ) . ConfigureAwait ( false ) ;
2015-07-20 18:32:55 +00:00
2015-08-16 18:37:53 +00:00
list . AddRange ( channels ) ;
2015-07-23 23:40:54 +00:00
}
catch ( Exception ex )
{
_logger . ErrorException ( "Error getting channels" , ex ) ;
2015-07-20 18:32:55 +00:00
}
}
2016-02-24 19:06:26 +00:00
foreach ( var provider in GetListingProviders ( ) )
2015-07-23 05:25:55 +00:00
{
2016-02-24 19:06:26 +00:00
var enabledChannels = list
. Where ( i = > IsListingProviderEnabledForTuner ( provider . Item2 , i . TunerHostId ) )
. ToList ( ) ;
if ( enabledChannels . Count > 0 )
2015-07-23 05:25:55 +00:00
{
2015-08-02 23:47:31 +00:00
try
{
2016-03-16 21:30:49 +00:00
await provider . Item1 . AddMetadata ( provider . Item2 , enabledChannels , cancellationToken ) . ConfigureAwait ( false ) ;
2015-08-02 23:47:31 +00:00
}
catch ( NotSupportedException )
{
}
catch ( Exception ex )
{
_logger . ErrorException ( "Error adding metadata" , ex ) ;
}
2015-07-23 05:25:55 +00:00
}
}
2016-02-24 19:06:26 +00:00
2016-03-17 19:29:53 +00:00
_channelCache = list . ToList ( ) ;
2015-07-20 18:32:55 +00:00
return list ;
}
2016-06-09 16:13:25 +00:00
public async Task < List < ChannelInfo > > GetChannelsForListingsProvider ( ListingsProviderInfo listingsProvider , CancellationToken cancellationToken )
{
var list = new List < ChannelInfo > ( ) ;
foreach ( var hostInstance in _liveTvManager . TunerHosts )
{
try
{
var channels = await hostInstance . GetChannels ( cancellationToken ) . ConfigureAwait ( false ) ;
list . AddRange ( channels ) ;
}
catch ( Exception ex )
{
_logger . ErrorException ( "Error getting channels" , ex ) ;
}
}
return list
. Where ( i = > IsListingProviderEnabledForTuner ( listingsProvider , i . TunerHostId ) )
. ToList ( ) ;
}
2015-08-16 18:54:25 +00:00
public Task < IEnumerable < ChannelInfo > > GetChannelsAsync ( CancellationToken cancellationToken )
{
return GetChannelsAsync ( false , cancellationToken ) ;
}
2015-07-20 18:32:55 +00:00
public Task CancelSeriesTimerAsync ( string timerId , CancellationToken cancellationToken )
{
2016-02-01 00:57:40 +00:00
var timers = _timerProvider
. GetAll ( )
. Where ( i = > string . Equals ( i . SeriesTimerId , timerId , StringComparison . OrdinalIgnoreCase ) )
. ToList ( ) ;
2015-08-17 16:52:56 +00:00
foreach ( var timer in timers )
{
CancelTimerInternal ( timer . Id ) ;
}
2015-07-29 03:42:03 +00:00
var remove = _seriesTimerProvider . GetAll ( ) . FirstOrDefault ( r = > string . Equals ( r . Id , timerId , StringComparison . OrdinalIgnoreCase ) ) ;
2015-07-20 18:32:55 +00:00
if ( remove ! = null )
{
_seriesTimerProvider . Delete ( remove ) ;
}
return Task . FromResult ( true ) ;
}
private void CancelTimerInternal ( string timerId )
{
2015-07-29 03:42:03 +00:00
var remove = _timerProvider . GetAll ( ) . FirstOrDefault ( r = > string . Equals ( r . Id , timerId , StringComparison . OrdinalIgnoreCase ) ) ;
2015-07-20 18:32:55 +00:00
if ( remove ! = null )
{
_timerProvider . Delete ( remove ) ;
}
2016-03-01 04:24:42 +00:00
ActiveRecordingInfo activeRecordingInfo ;
2015-07-20 18:32:55 +00:00
2016-03-01 04:24:42 +00:00
if ( _activeRecordings . TryGetValue ( timerId , out activeRecordingInfo ) )
2015-07-20 18:32:55 +00:00
{
2016-03-01 04:24:42 +00:00
activeRecordingInfo . CancellationTokenSource . Cancel ( ) ;
2015-07-20 18:32:55 +00:00
}
}
public Task CancelTimerAsync ( string timerId , CancellationToken cancellationToken )
{
CancelTimerInternal ( timerId ) ;
return Task . FromResult ( true ) ;
}
2016-05-04 20:50:47 +00:00
public Task DeleteRecordingAsync ( string recordingId , CancellationToken cancellationToken )
2015-07-20 18:32:55 +00:00
{
2016-05-04 20:50:47 +00:00
return Task . FromResult ( true ) ;
2015-07-20 18:32:55 +00:00
}
public Task CreateTimerAsync ( TimerInfo info , CancellationToken cancellationToken )
2016-06-08 21:04:52 +00:00
{
return CreateTimer ( info , cancellationToken ) ;
}
2016-06-23 05:26:34 +00:00
public Task CreateSeriesTimerAsync ( SeriesTimerInfo info , CancellationToken cancellationToken )
2016-06-08 21:04:52 +00:00
{
return CreateSeriesTimer ( info , cancellationToken ) ;
}
public Task < string > CreateTimer ( TimerInfo info , CancellationToken cancellationToken )
2015-07-20 18:32:55 +00:00
{
info . Id = Guid . NewGuid ( ) . ToString ( "N" ) ;
_timerProvider . Add ( info ) ;
2016-06-08 21:04:52 +00:00
return Task . FromResult ( info . Id ) ;
2015-07-20 18:32:55 +00:00
}
2016-06-08 21:04:52 +00:00
public async Task < string > CreateSeriesTimer ( SeriesTimerInfo info , CancellationToken cancellationToken )
2015-07-20 18:32:55 +00:00
{
2015-07-29 17:16:00 +00:00
info . Id = Guid . NewGuid ( ) . ToString ( "N" ) ;
2015-07-20 18:32:55 +00:00
2015-08-16 22:03:22 +00:00
List < ProgramInfo > epgData ;
if ( info . RecordAnyChannel )
{
var channels = await GetChannelsAsync ( true , CancellationToken . None ) . ConfigureAwait ( false ) ;
var channelIds = channels . Select ( i = > i . Id ) . ToList ( ) ;
epgData = GetEpgDataForChannels ( channelIds ) ;
}
else
{
epgData = GetEpgDataForChannel ( info . ChannelId ) ;
}
// populate info.seriesID
var program = epgData . FirstOrDefault ( i = > string . Equals ( i . Id , info . ProgramId , StringComparison . OrdinalIgnoreCase ) ) ;
if ( program ! = null )
{
info . SeriesId = program . SeriesId ;
}
else
{
throw new InvalidOperationException ( "SeriesId for program not found" ) ;
}
2015-07-20 18:32:55 +00:00
_seriesTimerProvider . Add ( info ) ;
2016-01-20 03:48:37 +00:00
await UpdateTimersForSeriesTimer ( epgData , info , false ) . ConfigureAwait ( false ) ;
2016-06-08 21:04:52 +00:00
return info . Id ;
2015-07-20 18:32:55 +00:00
}
2015-08-15 21:58:52 +00:00
public async Task UpdateSeriesTimerAsync ( SeriesTimerInfo info , CancellationToken cancellationToken )
2015-07-20 18:32:55 +00:00
{
2016-01-20 03:48:37 +00:00
var instance = _seriesTimerProvider . GetAll ( ) . FirstOrDefault ( i = > string . Equals ( i . Id , info . Id , StringComparison . OrdinalIgnoreCase ) ) ;
if ( instance ! = null )
2015-08-16 22:03:22 +00:00
{
2016-01-20 03:48:37 +00:00
instance . ChannelId = info . ChannelId ;
instance . Days = info . Days ;
instance . EndDate = info . EndDate ;
instance . IsPostPaddingRequired = info . IsPostPaddingRequired ;
instance . IsPrePaddingRequired = info . IsPrePaddingRequired ;
instance . PostPaddingSeconds = info . PostPaddingSeconds ;
instance . PrePaddingSeconds = info . PrePaddingSeconds ;
instance . Priority = info . Priority ;
instance . RecordAnyChannel = info . RecordAnyChannel ;
instance . RecordAnyTime = info . RecordAnyTime ;
instance . RecordNewOnly = info . RecordNewOnly ;
instance . StartDate = info . StartDate ;
_seriesTimerProvider . Update ( instance ) ;
2015-08-16 22:03:22 +00:00
2016-01-20 03:48:37 +00:00
List < ProgramInfo > epgData ;
if ( instance . RecordAnyChannel )
{
var channels = await GetChannelsAsync ( true , CancellationToken . None ) . ConfigureAwait ( false ) ;
var channelIds = channels . Select ( i = > i . Id ) . ToList ( ) ;
epgData = GetEpgDataForChannels ( channelIds ) ;
}
else
{
epgData = GetEpgDataForChannel ( instance . ChannelId ) ;
}
await UpdateTimersForSeriesTimer ( epgData , instance , true ) . ConfigureAwait ( false ) ;
}
2015-07-20 18:32:55 +00:00
}
public Task UpdateTimerAsync ( TimerInfo info , CancellationToken cancellationToken )
{
_timerProvider . Update ( info ) ;
return Task . FromResult ( true ) ;
}
public Task < ImageStream > GetChannelImageAsync ( string channelId , CancellationToken cancellationToken )
{
throw new NotImplementedException ( ) ;
}
public Task < ImageStream > GetRecordingImageAsync ( string recordingId , CancellationToken cancellationToken )
{
throw new NotImplementedException ( ) ;
}
public Task < ImageStream > GetProgramImageAsync ( string programId , string channelId , CancellationToken cancellationToken )
{
throw new NotImplementedException ( ) ;
}
2015-10-11 00:39:30 +00:00
public async Task < IEnumerable < RecordingInfo > > GetRecordingsAsync ( CancellationToken cancellationToken )
2015-07-20 18:32:55 +00:00
{
2016-05-04 20:50:47 +00:00
return new List < RecordingInfo > ( ) ;
2015-07-20 18:32:55 +00:00
}
public Task < IEnumerable < TimerInfo > > GetTimersAsync ( CancellationToken cancellationToken )
{
return Task . FromResult ( ( IEnumerable < TimerInfo > ) _timerProvider . GetAll ( ) ) ;
}
public Task < SeriesTimerInfo > GetNewTimerDefaultsAsync ( CancellationToken cancellationToken , ProgramInfo program = null )
{
2015-08-24 02:08:20 +00:00
var config = GetConfiguration ( ) ;
2015-07-20 18:32:55 +00:00
var defaults = new SeriesTimerInfo ( )
{
2015-08-24 02:08:20 +00:00
PostPaddingSeconds = Math . Max ( config . PostPaddingSeconds , 0 ) ,
PrePaddingSeconds = Math . Max ( config . PrePaddingSeconds , 0 ) ,
2016-05-29 18:42:39 +00:00
RecordAnyChannel = true ,
RecordAnyTime = true ,
RecordNewOnly = false ,
Days = new List < DayOfWeek >
{
DayOfWeek . Sunday ,
DayOfWeek . Monday ,
DayOfWeek . Tuesday ,
DayOfWeek . Wednesday ,
DayOfWeek . Thursday ,
DayOfWeek . Friday ,
DayOfWeek . Saturday
}
2015-07-20 18:32:55 +00:00
} ;
2015-08-11 17:47:29 +00:00
if ( program ! = null )
{
defaults . SeriesId = program . SeriesId ;
defaults . ProgramId = program . Id ;
}
2015-07-20 18:32:55 +00:00
return Task . FromResult ( defaults ) ;
}
public Task < IEnumerable < SeriesTimerInfo > > GetSeriesTimersAsync ( CancellationToken cancellationToken )
{
return Task . FromResult ( ( IEnumerable < SeriesTimerInfo > ) _seriesTimerProvider . GetAll ( ) ) ;
}
2015-07-23 05:25:55 +00:00
public async Task < IEnumerable < ProgramInfo > > GetProgramsAsync ( string channelId , DateTime startDateUtc , DateTime endDateUtc , CancellationToken cancellationToken )
2015-09-10 18:28:22 +00:00
{
try
{
return await GetProgramsAsyncInternal ( channelId , startDateUtc , endDateUtc , cancellationToken ) . ConfigureAwait ( false ) ;
}
2015-10-25 18:34:31 +00:00
catch ( OperationCanceledException )
{
throw ;
}
2015-09-10 18:28:22 +00:00
catch ( Exception ex )
{
_logger . ErrorException ( "Error getting programs" , ex ) ;
return GetEpgDataForChannel ( channelId ) . Where ( i = > i . StartDate < = endDateUtc & & i . EndDate > = startDateUtc ) ;
}
}
2016-02-24 19:06:26 +00:00
private bool IsListingProviderEnabledForTuner ( ListingsProviderInfo info , string tunerHostId )
{
2016-03-16 04:14:38 +00:00
if ( info . EnableAllTuners )
{
return true ;
}
if ( string . IsNullOrWhiteSpace ( tunerHostId ) )
{
throw new ArgumentNullException ( "tunerHostId" ) ;
}
return info . EnabledTuners . Contains ( tunerHostId , StringComparer . OrdinalIgnoreCase ) ;
2016-02-24 19:06:26 +00:00
}
2015-09-10 18:28:22 +00:00
private async Task < IEnumerable < ProgramInfo > > GetProgramsAsyncInternal ( string channelId , DateTime startDateUtc , DateTime endDateUtc , CancellationToken cancellationToken )
2015-07-20 18:32:55 +00:00
{
2015-08-16 18:54:25 +00:00
var channels = await GetChannelsAsync ( true , cancellationToken ) . ConfigureAwait ( false ) ;
var channel = channels . First ( i = > string . Equals ( i . Id , channelId , StringComparison . OrdinalIgnoreCase ) ) ;
2015-07-23 05:25:55 +00:00
foreach ( var provider in GetListingProviders ( ) )
{
2016-02-24 19:06:26 +00:00
if ( ! IsListingProviderEnabledForTuner ( provider . Item2 , channel . TunerHostId ) )
{
2016-03-17 19:29:53 +00:00
_logger . Debug ( "Skipping getting programs for channel {0}-{1} from {2}-{3}, because it's not enabled for this tuner." , channel . Number , channel . Name , provider . Item1 . Name , provider . Item2 . ListingsId ? ? string . Empty ) ;
2016-02-24 19:06:26 +00:00
continue ;
}
2016-03-17 19:29:53 +00:00
_logger . Debug ( "Getting programs for channel {0}-{1} from {2}-{3}" , channel . Number , channel . Name , provider . Item1 . Name , provider . Item2 . ListingsId ? ? string . Empty ) ;
2016-03-23 19:06:36 +00:00
2016-06-06 18:22:42 +00:00
var channelMappings = GetChannelMappings ( provider . Item2 ) ;
var channelNumber = channel . Number ;
string mappedChannelNumber ;
if ( channelMappings . TryGetValue ( channelNumber , out mappedChannelNumber ) )
{
_logger . Debug ( "Found mapped channel on provider {0}. Tuner channel number: {1}, Mapped channel number: {2}" , provider . Item1 . Name , channelNumber , mappedChannelNumber ) ;
channelNumber = mappedChannelNumber ;
}
var programs = await provider . Item1 . GetProgramsAsync ( provider . Item2 , channelNumber , channel . Name , startDateUtc , endDateUtc , cancellationToken )
2015-07-23 05:25:55 +00:00
. ConfigureAwait ( false ) ;
2015-09-10 18:28:22 +00:00
2015-07-23 05:25:55 +00:00
var list = programs . ToList ( ) ;
2015-07-23 23:40:54 +00:00
// Replace the value that came from the provider with a normalized value
foreach ( var program in list )
{
program . ChannelId = channelId ;
}
2015-07-23 05:25:55 +00:00
if ( list . Count > 0 )
{
2015-07-29 03:42:03 +00:00
SaveEpgDataForChannel ( channelId , list ) ;
2015-07-23 05:25:55 +00:00
return list ;
}
}
return new List < ProgramInfo > ( ) ;
}
2016-06-06 18:22:42 +00:00
private Dictionary < string , string > GetChannelMappings ( ListingsProviderInfo info )
{
var dict = new Dictionary < string , string > ( StringComparer . OrdinalIgnoreCase ) ;
foreach ( var mapping in info . ChannelMappings )
{
dict [ mapping . Name ] = mapping . Value ;
}
return dict ;
}
2015-07-23 05:25:55 +00:00
private List < Tuple < IListingsProvider , ListingsProviderInfo > > GetListingProviders ( )
{
return GetConfiguration ( ) . ListingProviders
. Select ( i = >
{
2015-07-23 13:23:22 +00:00
var provider = _liveTvManager . ListingProviders . FirstOrDefault ( l = > string . Equals ( l . Type , i . Type , StringComparison . OrdinalIgnoreCase ) ) ;
2015-07-23 05:25:55 +00:00
return provider = = null ? null : new Tuple < IListingsProvider , ListingsProviderInfo > ( provider , i ) ;
} )
. Where ( i = > i ! = null )
. ToList ( ) ;
2015-07-20 18:32:55 +00:00
}
public Task < MediaSourceInfo > GetRecordingStream ( string recordingId , string streamId , CancellationToken cancellationToken )
{
throw new NotImplementedException ( ) ;
}
2015-07-23 23:40:54 +00:00
public async Task < MediaSourceInfo > GetChannelStream ( string channelId , string streamId , CancellationToken cancellationToken )
2015-07-20 18:32:55 +00:00
{
2015-07-23 23:40:54 +00:00
_logger . Info ( "Streaming Channel " + channelId ) ;
2015-10-16 18:11:11 +00:00
foreach ( var hostInstance in _liveTvManager . TunerHosts )
{
try
{
var result = await hostInstance . GetChannelStream ( channelId , streamId , cancellationToken ) . ConfigureAwait ( false ) ;
result . Item2 . Release ( ) ;
return result . Item1 ;
}
catch ( Exception e )
{
_logger . ErrorException ( "Error getting channel stream" , e ) ;
}
}
throw new ApplicationException ( "Tuner not found." ) ;
}
2016-04-26 02:16:46 +00:00
private async Task < Tuple < MediaSourceInfo , ITunerHost , SemaphoreSlim > > GetChannelStreamInternal ( string channelId , string streamId , CancellationToken cancellationToken )
2015-10-16 18:11:11 +00:00
{
_logger . Info ( "Streaming Channel " + channelId ) ;
2015-08-19 19:25:18 +00:00
foreach ( var hostInstance in _liveTvManager . TunerHosts )
2015-07-23 23:40:54 +00:00
{
try
{
2016-04-26 02:16:46 +00:00
var result = await hostInstance . GetChannelStream ( channelId , streamId , cancellationToken ) . ConfigureAwait ( false ) ;
return new Tuple < MediaSourceInfo , ITunerHost , SemaphoreSlim > ( result . Item1 , hostInstance , result . Item2 ) ;
2015-07-23 23:40:54 +00:00
}
2015-08-16 18:37:53 +00:00
catch ( Exception e )
2015-07-23 23:40:54 +00:00
{
2015-08-16 18:37:53 +00:00
_logger . ErrorException ( "Error getting channel stream" , e ) ;
2015-07-23 23:40:54 +00:00
}
}
throw new ApplicationException ( "Tuner not found." ) ;
2015-07-20 18:32:55 +00:00
}
2015-07-23 23:40:54 +00:00
public async Task < List < MediaSourceInfo > > GetChannelStreamMediaSources ( string channelId , CancellationToken cancellationToken )
2015-07-20 18:32:55 +00:00
{
2015-08-19 19:25:18 +00:00
foreach ( var hostInstance in _liveTvManager . TunerHosts )
2015-07-23 23:40:54 +00:00
{
2015-08-16 18:37:53 +00:00
try
2015-07-23 23:40:54 +00:00
{
2015-08-19 19:25:18 +00:00
var sources = await hostInstance . GetChannelStreamMediaSources ( channelId , cancellationToken ) . ConfigureAwait ( false ) ;
2015-07-23 23:40:54 +00:00
2015-08-16 18:37:53 +00:00
if ( sources . Count > 0 )
{
return sources ;
}
}
catch ( NotImplementedException )
{
2015-08-16 18:54:25 +00:00
2015-07-23 23:40:54 +00:00
}
}
2015-08-27 19:59:42 +00:00
throw new NotImplementedException ( ) ;
2015-07-20 18:32:55 +00:00
}
public Task < List < MediaSourceInfo > > GetRecordingStreamMediaSources ( string recordingId , CancellationToken cancellationToken )
{
throw new NotImplementedException ( ) ;
}
public Task CloseLiveStream ( string id , CancellationToken cancellationToken )
{
return Task . FromResult ( 0 ) ;
}
public Task RecordLiveStream ( string id , CancellationToken cancellationToken )
{
return Task . FromResult ( 0 ) ;
}
public Task ResetTuner ( string id , CancellationToken cancellationToken )
{
return Task . FromResult ( 0 ) ;
}
async void _timerProvider_TimerFired ( object sender , GenericEventArgs < TimerInfo > e )
{
2015-08-22 18:29:12 +00:00
var timer = e . Argument ;
2015-08-31 23:47:47 +00:00
_logger . Info ( "Recording timer fired." ) ;
2015-09-21 16:23:20 +00:00
2015-07-20 18:32:55 +00:00
try
{
2015-09-26 02:31:13 +00:00
var recordingEndDate = timer . EndDate . AddSeconds ( timer . PostPaddingSeconds ) ;
if ( recordingEndDate < = DateTime . UtcNow )
{
_logger . Warn ( "Recording timer fired for timer {0}, Id: {1}, but the program has already ended." , timer . Name , timer . Id ) ;
return ;
}
2016-03-01 04:24:42 +00:00
var activeRecordingInfo = new ActiveRecordingInfo
{
CancellationTokenSource = new CancellationTokenSource ( ) ,
TimerId = timer . Id
} ;
2015-07-20 18:32:55 +00:00
2016-03-01 04:24:42 +00:00
if ( _activeRecordings . TryAdd ( timer . Id , activeRecordingInfo ) )
2015-07-20 18:32:55 +00:00
{
2016-03-01 04:24:42 +00:00
await RecordStream ( timer , recordingEndDate , activeRecordingInfo , activeRecordingInfo . CancellationTokenSource . Token ) . ConfigureAwait ( false ) ;
2015-07-20 18:32:55 +00:00
}
2016-01-26 18:18:54 +00:00
else
{
_logger . Info ( "Skipping RecordStream because it's already in progress." ) ;
}
2015-07-20 18:32:55 +00:00
}
catch ( OperationCanceledException )
{
}
catch ( Exception ex )
{
_logger . ErrorException ( "Error recording stream" , ex ) ;
}
}
2016-05-04 20:50:47 +00:00
private string GetRecordingPath ( TimerInfo timer , ProgramInfo info )
2015-07-20 18:32:55 +00:00
{
var recordPath = RecordingPath ;
2016-05-04 20:50:47 +00:00
var config = GetConfiguration ( ) ;
2015-08-20 23:55:23 +00:00
2015-07-20 18:32:55 +00:00
if ( info . IsMovie )
{
2016-05-04 20:50:47 +00:00
var customRecordingPath = config . MovieRecordingPath ;
2016-05-09 03:13:38 +00:00
var allowSubfolder = true ;
if ( ! string . IsNullOrWhiteSpace ( customRecordingPath ) )
{
allowSubfolder = string . Equals ( customRecordingPath , recordPath , StringComparison . OrdinalIgnoreCase ) ;
recordPath = customRecordingPath ;
}
if ( allowSubfolder & & config . EnableRecordingSubfolders )
2016-05-04 20:50:47 +00:00
{
recordPath = Path . Combine ( recordPath , "Movies" ) ;
}
var folderName = _fileSystem . GetValidFilename ( info . Name ) . Trim ( ) ;
if ( info . ProductionYear . HasValue )
{
folderName + = " (" + info . ProductionYear . Value . ToString ( CultureInfo . InvariantCulture ) + ")" ;
}
recordPath = Path . Combine ( recordPath , folderName ) ;
2015-07-20 18:32:55 +00:00
}
2015-08-20 23:55:23 +00:00
else if ( info . IsSeries )
{
2016-05-04 20:50:47 +00:00
var customRecordingPath = config . SeriesRecordingPath ;
2016-05-09 03:13:38 +00:00
var allowSubfolder = true ;
if ( ! string . IsNullOrWhiteSpace ( customRecordingPath ) )
{
allowSubfolder = string . Equals ( customRecordingPath , recordPath , StringComparison . OrdinalIgnoreCase ) ;
recordPath = customRecordingPath ;
}
if ( allowSubfolder & & config . EnableRecordingSubfolders )
2016-05-04 20:50:47 +00:00
{
recordPath = Path . Combine ( recordPath , "Series" ) ;
}
var folderName = _fileSystem . GetValidFilename ( info . Name ) . Trim ( ) ;
var folderNameWithYear = folderName ;
if ( info . ProductionYear . HasValue )
{
folderNameWithYear + = " (" + info . ProductionYear . Value . ToString ( CultureInfo . InvariantCulture ) + ")" ;
}
if ( Directory . Exists ( Path . Combine ( recordPath , folderName ) ) )
{
recordPath = Path . Combine ( recordPath , folderName ) ;
}
else
{
recordPath = Path . Combine ( recordPath , folderNameWithYear ) ;
}
2016-05-03 04:17:57 +00:00
if ( info . SeasonNumber . HasValue )
{
2016-05-04 20:50:47 +00:00
folderName = string . Format ( "Season {0}" , info . SeasonNumber . Value . ToString ( CultureInfo . InvariantCulture ) ) ;
2016-05-03 04:17:57 +00:00
recordPath = Path . Combine ( recordPath , folderName ) ;
}
2015-08-20 23:55:23 +00:00
}
else if ( info . IsKids )
{
2016-05-04 20:50:47 +00:00
if ( config . EnableRecordingSubfolders )
{
recordPath = Path . Combine ( recordPath , "Kids" ) ;
}
var folderName = _fileSystem . GetValidFilename ( info . Name ) . Trim ( ) ;
if ( info . ProductionYear . HasValue )
{
folderName + = " (" + info . ProductionYear . Value . ToString ( CultureInfo . InvariantCulture ) + ")" ;
}
recordPath = Path . Combine ( recordPath , folderName ) ;
2015-08-20 23:55:23 +00:00
}
else if ( info . IsSports )
{
2016-05-04 20:50:47 +00:00
if ( config . EnableRecordingSubfolders )
{
recordPath = Path . Combine ( recordPath , "Sports" ) ;
}
recordPath = Path . Combine ( recordPath , _fileSystem . GetValidFilename ( info . Name ) . Trim ( ) ) ;
2015-08-20 23:55:23 +00:00
}
2015-07-20 18:32:55 +00:00
else
{
2016-05-04 20:50:47 +00:00
if ( config . EnableRecordingSubfolders )
{
recordPath = Path . Combine ( recordPath , "Other" ) ;
}
recordPath = Path . Combine ( recordPath , _fileSystem . GetValidFilename ( info . Name ) . Trim ( ) ) ;
2015-07-20 18:32:55 +00:00
}
2015-09-27 21:02:39 +00:00
var recordingFileName = _fileSystem . GetValidFilename ( RecordingHelper . GetRecordingName ( timer , info ) ) . Trim ( ) + ".ts" ;
2015-08-21 02:36:30 +00:00
2016-05-04 20:50:47 +00:00
return Path . Combine ( recordPath , recordingFileName ) ;
}
2015-07-20 18:32:55 +00:00
2016-05-04 20:50:47 +00:00
private async Task RecordStream ( TimerInfo timer , DateTime recordingEndDate , ActiveRecordingInfo activeRecordingInfo , CancellationToken cancellationToken )
{
if ( timer = = null )
{
throw new ArgumentNullException ( "timer" ) ;
}
2015-07-20 18:32:55 +00:00
2016-05-04 20:50:47 +00:00
ProgramInfo info = null ;
if ( string . IsNullOrWhiteSpace ( timer . ProgramId ) )
2015-07-20 18:32:55 +00:00
{
2016-05-04 20:50:47 +00:00
_logger . Info ( "Timer {0} has null programId" , timer . Id ) ;
}
else
{
info = GetProgramInfoFromCache ( timer . ChannelId , timer . ProgramId ) ;
}
if ( info = = null )
{
_logger . Info ( "Unable to find program with Id {0}. Will search using start date" , timer . ProgramId ) ;
info = GetProgramInfoFromCache ( timer . ChannelId , timer . StartDate ) ;
2015-07-20 18:32:55 +00:00
}
2016-05-04 20:50:47 +00:00
if ( info = = null )
{
throw new InvalidOperationException ( string . Format ( "Program with Id {0} not found" , timer . ProgramId ) ) ;
}
var recordPath = GetRecordingPath ( timer , info ) ;
var recordingStatus = RecordingStatus . New ;
2016-06-27 22:53:42 +00:00
var isResourceOpen = false ;
SemaphoreSlim semaphore = null ;
2016-05-04 20:50:47 +00:00
2015-07-20 18:32:55 +00:00
try
{
2016-02-05 03:44:44 +00:00
var result = await GetChannelStreamInternal ( timer . ChannelId , null , CancellationToken . None ) . ConfigureAwait ( false ) ;
2016-06-27 22:53:42 +00:00
isResourceOpen = true ;
semaphore = result . Item3 ;
2015-10-16 18:11:11 +00:00
var mediaStreamInfo = result . Item1 ;
2015-09-26 02:31:13 +00:00
2016-06-27 22:53:42 +00:00
// HDHR doesn't seem to release the tuner right away after first probing with ffmpeg
//await Task.Delay(3000, cancellationToken).ConfigureAwait(false);
2016-02-12 07:01:38 +00:00
2016-06-27 22:53:42 +00:00
var recorder = await GetRecorder ( ) . ConfigureAwait ( false ) ;
2015-10-16 18:11:11 +00:00
2016-06-27 22:53:42 +00:00
recordPath = recorder . GetOutputPath ( mediaStreamInfo , recordPath ) ;
recordPath = EnsureFileUnique ( recordPath , timer . Id ) ;
2016-02-22 16:00:17 +00:00
2016-06-27 22:53:42 +00:00
_libraryMonitor . ReportFileSystemChangeBeginning ( recordPath ) ;
_fileSystem . CreateDirectory ( Path . GetDirectoryName ( recordPath ) ) ;
activeRecordingInfo . Path = recordPath ;
2016-04-26 02:16:46 +00:00
2016-06-27 22:53:42 +00:00
var duration = recordingEndDate - DateTime . UtcNow ;
2015-10-16 18:11:11 +00:00
2016-06-27 22:53:42 +00:00
_logger . Info ( "Beginning recording. Will record for {0} minutes." , duration . TotalMinutes . ToString ( CultureInfo . InvariantCulture ) ) ;
2016-02-06 21:32:02 +00:00
2016-06-27 22:53:42 +00:00
_logger . Info ( "Writing file to path: " + recordPath ) ;
_logger . Info ( "Opening recording stream from tuner provider" ) ;
2016-02-06 21:32:02 +00:00
2016-06-27 22:53:42 +00:00
Action onStarted = ( ) = >
{
timer . Status = RecordingStatus . InProgress ;
_timerProvider . AddOrUpdate ( timer , false ) ;
2016-04-26 02:16:46 +00:00
2016-06-27 22:53:42 +00:00
result . Item3 . Release ( ) ;
isResourceOpen = false ;
} ;
2016-04-26 02:16:46 +00:00
2016-06-27 22:53:42 +00:00
var pathWithDuration = result . Item2 . ApplyDuration ( mediaStreamInfo . Path , duration ) ;
2015-09-26 02:31:13 +00:00
2016-06-27 22:53:42 +00:00
// If it supports supplying duration via url
if ( ! string . Equals ( pathWithDuration , mediaStreamInfo . Path , StringComparison . OrdinalIgnoreCase ) )
2015-07-20 18:32:55 +00:00
{
2016-06-27 22:53:42 +00:00
mediaStreamInfo . Path = pathWithDuration ;
mediaStreamInfo . RunTimeTicks = duration . Ticks ;
2015-07-20 18:32:55 +00:00
}
2016-06-27 22:53:42 +00:00
await recorder . Record ( mediaStreamInfo , recordPath , duration , onStarted , cancellationToken ) . ConfigureAwait ( false ) ;
recordingStatus = RecordingStatus . Completed ;
_logger . Info ( "Recording completed: {0}" , recordPath ) ;
2015-07-20 18:32:55 +00:00
}
catch ( OperationCanceledException )
{
2016-03-01 04:24:42 +00:00
_logger . Info ( "Recording stopped: {0}" , recordPath ) ;
2016-05-04 20:50:47 +00:00
recordingStatus = RecordingStatus . Completed ;
2015-07-20 18:32:55 +00:00
}
2015-08-05 03:43:54 +00:00
catch ( Exception ex )
2015-07-20 18:32:55 +00:00
{
2016-03-01 04:24:42 +00:00
_logger . ErrorException ( "Error recording to {0}" , ex , recordPath ) ;
2016-05-04 20:50:47 +00:00
recordingStatus = RecordingStatus . Error ;
2015-07-20 18:32:55 +00:00
}
2015-09-04 01:34:57 +00:00
finally
{
2016-06-27 22:53:42 +00:00
if ( isResourceOpen & & semaphore ! = null )
{
semaphore . Release ( ) ;
}
_libraryMonitor . ReportFileSystemChangeComplete ( recordPath , true ) ;
2016-03-01 04:24:42 +00:00
ActiveRecordingInfo removed ;
2015-09-04 01:34:57 +00:00
_activeRecordings . TryRemove ( timer . Id , out removed ) ;
}
2015-07-20 18:32:55 +00:00
2016-05-04 20:50:47 +00:00
if ( recordingStatus = = RecordingStatus . Completed )
2015-08-22 19:46:55 +00:00
{
2016-06-20 22:07:18 +00:00
timer . Status = RecordingStatus . Completed ;
2016-06-27 22:53:42 +00:00
_timerProvider . Delete ( timer ) ;
2016-06-20 22:07:18 +00:00
2016-05-04 20:50:47 +00:00
OnSuccessfulRecording ( info . IsSeries , recordPath ) ;
2015-09-03 01:40:47 +00:00
}
2015-09-04 01:34:57 +00:00
else if ( DateTime . UtcNow < timer . EndDate )
2015-09-03 01:40:47 +00:00
{
2015-09-04 01:34:57 +00:00
const int retryIntervalSeconds = 60 ;
_logger . Info ( "Retrying recording in {0} seconds." , retryIntervalSeconds ) ;
2015-09-03 01:40:47 +00:00
2016-06-20 22:07:18 +00:00
timer . Status = RecordingStatus . New ;
timer . StartDate = DateTime . UtcNow . AddSeconds ( retryIntervalSeconds ) ;
_timerProvider . AddOrUpdate ( timer ) ;
2015-09-04 01:34:57 +00:00
}
else
{
_timerProvider . Delete ( timer ) ;
2015-08-22 19:46:55 +00:00
}
}
2016-03-01 04:24:42 +00:00
private string EnsureFileUnique ( string path , string timerId )
2016-02-26 04:09:42 +00:00
{
var originalPath = path ;
var index = 1 ;
2016-03-01 04:24:42 +00:00
while ( FileExists ( path , timerId ) )
2016-02-26 04:09:42 +00:00
{
var parent = Path . GetDirectoryName ( originalPath ) ;
var name = Path . GetFileNameWithoutExtension ( originalPath ) ;
name + = "-" + index . ToString ( CultureInfo . InvariantCulture ) ;
path = Path . ChangeExtension ( Path . Combine ( parent , name ) , Path . GetExtension ( originalPath ) ) ;
index + + ;
}
return path ;
}
2016-03-01 04:24:42 +00:00
private bool FileExists ( string path , string timerId )
{
if ( _fileSystem . FileExists ( path ) )
{
return true ;
}
var hasRecordingAtPath = _activeRecordings . Values . ToList ( ) . Any ( i = > string . Equals ( i . Path , path , StringComparison . OrdinalIgnoreCase ) & & ! string . Equals ( i . TimerId , timerId , StringComparison . OrdinalIgnoreCase ) ) ;
if ( hasRecordingAtPath )
{
return true ;
}
return false ;
}
2016-02-12 07:01:38 +00:00
private async Task < IRecorder > GetRecorder ( )
{
2016-04-27 21:26:28 +00:00
var config = GetConfiguration ( ) ;
if ( config . EnableRecordingEncoding )
2016-02-12 07:01:38 +00:00
{
2016-02-12 18:29:28 +00:00
var regInfo = await _security . GetRegistrationStatus ( "embytvrecordingconversion" ) . ConfigureAwait ( false ) ;
2016-02-12 07:01:38 +00:00
if ( regInfo . IsValid )
{
2016-06-20 22:07:18 +00:00
return new EncodedRecorder ( _logger , _fileSystem , _mediaEncoder , _config . ApplicationPaths , _jsonSerializer , config , _httpClient ) ;
2016-02-12 07:01:38 +00:00
}
}
return new DirectRecorder ( _logger , _httpClient , _fileSystem ) ;
}
2016-05-04 20:50:47 +00:00
private async void OnSuccessfulRecording ( bool isSeries , string path )
2015-08-22 19:46:55 +00:00
{
if ( GetConfiguration ( ) . EnableAutoOrganize )
{
2016-05-04 20:50:47 +00:00
if ( isSeries )
2015-08-22 19:46:55 +00:00
{
try
{
2016-03-03 17:52:47 +00:00
// this is to account for the library monitor holding a lock for additional time after the change is complete.
// ideally this shouldn't be hard-coded
await Task . Delay ( 30000 ) . ConfigureAwait ( false ) ;
2015-08-22 19:46:55 +00:00
var organize = new EpisodeFileOrganizer ( _organizationService , _config , _fileSystem , _logger , _libraryManager , _libraryMonitor , _providerManager ) ;
2016-05-04 20:50:47 +00:00
var result = await organize . OrganizeEpisodeFile ( path , CancellationToken . None ) . ConfigureAwait ( false ) ;
2015-08-22 19:46:55 +00:00
}
catch ( Exception ex )
{
_logger . ErrorException ( "Error processing new recording" , ex ) ;
}
}
}
2015-07-20 18:32:55 +00:00
}
private ProgramInfo GetProgramInfoFromCache ( string channelId , string programId )
{
var epgData = GetEpgDataForChannel ( channelId ) ;
2015-08-18 04:22:45 +00:00
return epgData . FirstOrDefault ( p = > string . Equals ( p . Id , programId , StringComparison . OrdinalIgnoreCase ) ) ;
2015-07-20 18:32:55 +00:00
}
2016-02-06 21:32:02 +00:00
private ProgramInfo GetProgramInfoFromCache ( string channelId , DateTime startDateUtc )
{
var epgData = GetEpgDataForChannel ( channelId ) ;
var startDateTicks = startDateUtc . Ticks ;
// Find the first program that starts within 3 minutes
return epgData . FirstOrDefault ( p = > Math . Abs ( startDateTicks - p . StartDate . Ticks ) < = TimeSpan . FromMinutes ( 3 ) . Ticks ) ;
}
2015-07-20 18:32:55 +00:00
private LiveTvOptions GetConfiguration ( )
{
return _config . GetConfiguration < LiveTvOptions > ( "livetv" ) ;
}
2016-01-20 03:48:37 +00:00
private async Task UpdateTimersForSeriesTimer ( List < ProgramInfo > epgData , SeriesTimerInfo seriesTimer , bool deleteInvalidTimers )
2015-07-20 18:32:55 +00:00
{
2016-05-04 20:50:47 +00:00
var newTimers = GetTimersForSeries ( seriesTimer , epgData , true ) . ToList ( ) ;
2015-11-21 04:34:55 +00:00
2015-08-21 19:25:35 +00:00
var registration = await GetRegistrationInfo ( "seriesrecordings" ) . ConfigureAwait ( false ) ;
if ( registration . IsValid )
2015-07-20 18:32:55 +00:00
{
2015-08-21 19:25:35 +00:00
foreach ( var timer in newTimers )
{
_timerProvider . AddOrUpdate ( timer ) ;
}
2015-07-20 18:32:55 +00:00
}
2016-01-20 03:48:37 +00:00
if ( deleteInvalidTimers )
{
2016-05-04 20:50:47 +00:00
var allTimers = GetTimersForSeries ( seriesTimer , epgData , false )
2016-01-20 03:48:37 +00:00
. Select ( i = > i . Id )
. ToList ( ) ;
var deletes = _timerProvider . GetAll ( )
. Where ( i = > string . Equals ( i . SeriesTimerId , seriesTimer . Id , StringComparison . OrdinalIgnoreCase ) )
. Where ( i = > ! allTimers . Contains ( i . Id , StringComparer . OrdinalIgnoreCase ) & & i . StartDate > DateTime . UtcNow )
. ToList ( ) ;
foreach ( var timer in deletes )
{
await CancelTimerAsync ( timer . Id , CancellationToken . None ) . ConfigureAwait ( false ) ;
}
}
2015-07-20 18:32:55 +00:00
}
2016-05-04 20:50:47 +00:00
private IEnumerable < TimerInfo > GetTimersForSeries ( SeriesTimerInfo seriesTimer ,
IEnumerable < ProgramInfo > allPrograms ,
bool filterByCurrentRecordings )
2015-07-29 17:16:00 +00:00
{
2016-03-13 07:34:17 +00:00
if ( seriesTimer = = null )
{
throw new ArgumentNullException ( "seriesTimer" ) ;
}
if ( allPrograms = = null )
{
throw new ArgumentNullException ( "allPrograms" ) ;
}
2015-08-17 16:52:56 +00:00
// Exclude programs that have already ended
2016-01-20 03:48:37 +00:00
allPrograms = allPrograms . Where ( i = > i . EndDate > DateTime . UtcNow & & i . StartDate > DateTime . UtcNow ) ;
2015-08-17 16:52:56 +00:00
2015-07-29 17:16:00 +00:00
allPrograms = GetProgramsForSeries ( seriesTimer , allPrograms ) ;
2016-05-04 20:50:47 +00:00
if ( filterByCurrentRecordings )
{
allPrograms = allPrograms . Where ( i = > ! IsProgramAlreadyInLibrary ( i ) ) ;
}
2015-07-29 17:16:00 +00:00
return allPrograms . Select ( i = > RecordingHelper . CreateTimer ( i , seriesTimer ) ) ;
}
2016-05-04 20:50:47 +00:00
private bool IsProgramAlreadyInLibrary ( ProgramInfo program )
{
if ( ( program . EpisodeNumber . HasValue & & program . SeasonNumber . HasValue ) | | ! string . IsNullOrWhiteSpace ( program . EpisodeTitle ) )
{
var seriesIds = _libraryManager . GetItemIds ( new InternalItemsQuery
{
IncludeItemTypes = new [ ] { typeof ( Series ) . Name } ,
Name = program . Name
} ) . Select ( i = > i . ToString ( "N" ) ) . ToArray ( ) ;
if ( seriesIds . Length = = 0 )
{
return false ;
}
if ( program . EpisodeNumber . HasValue & & program . SeasonNumber . HasValue )
{
var result = _libraryManager . GetItemsResult ( new InternalItemsQuery
{
IncludeItemTypes = new [ ] { typeof ( Episode ) . Name } ,
ParentIndexNumber = program . SeasonNumber . Value ,
IndexNumber = program . EpisodeNumber . Value ,
2016-05-10 05:00:50 +00:00
AncestorIds = seriesIds ,
ExcludeLocationTypes = new [ ] { LocationType . Virtual }
2016-05-04 20:50:47 +00:00
} ) ;
if ( result . TotalRecordCount > 0 )
{
return true ;
}
}
if ( ! string . IsNullOrWhiteSpace ( program . EpisodeTitle ) )
{
var result = _libraryManager . GetItemsResult ( new InternalItemsQuery
{
IncludeItemTypes = new [ ] { typeof ( Episode ) . Name } ,
Name = program . EpisodeTitle ,
2016-05-10 05:00:50 +00:00
AncestorIds = seriesIds ,
ExcludeLocationTypes = new [ ] { LocationType . Virtual }
2016-05-04 20:50:47 +00:00
} ) ;
if ( result . TotalRecordCount > 0 )
{
return true ;
}
}
}
return false ;
}
2015-07-29 17:16:00 +00:00
private IEnumerable < ProgramInfo > GetProgramsForSeries ( SeriesTimerInfo seriesTimer , IEnumerable < ProgramInfo > allPrograms )
{
if ( ! seriesTimer . RecordAnyTime )
{
2015-12-07 15:41:22 +00:00
allPrograms = allPrograms . Where ( epg = > Math . Abs ( seriesTimer . StartDate . TimeOfDay . Ticks - epg . StartDate . TimeOfDay . Ticks ) < TimeSpan . FromMinutes ( 5 ) . Ticks ) ;
2016-05-29 18:42:39 +00:00
allPrograms = allPrograms . Where ( i = > seriesTimer . Days . Contains ( i . StartDate . ToLocalTime ( ) . DayOfWeek ) ) ;
2015-07-29 17:16:00 +00:00
}
if ( seriesTimer . RecordNewOnly )
{
2015-08-16 18:37:53 +00:00
allPrograms = allPrograms . Where ( epg = > ! epg . IsRepeat ) ;
2015-07-29 17:16:00 +00:00
}
if ( ! seriesTimer . RecordAnyChannel )
{
allPrograms = allPrograms . Where ( epg = > string . Equals ( epg . ChannelId , seriesTimer . ChannelId , StringComparison . OrdinalIgnoreCase ) ) ;
}
2015-08-16 18:37:53 +00:00
if ( string . IsNullOrWhiteSpace ( seriesTimer . SeriesId ) )
{
_logger . Error ( "seriesTimer.SeriesId is null. Cannot find programs for series" ) ;
return new List < ProgramInfo > ( ) ;
}
2015-08-15 21:58:52 +00:00
return allPrograms . Where ( i = > string . Equals ( i . SeriesId , seriesTimer . SeriesId , StringComparison . OrdinalIgnoreCase ) ) ;
2015-07-29 17:16:00 +00:00
}
2015-08-02 23:47:31 +00:00
2015-07-20 18:32:55 +00:00
private string GetChannelEpgCachePath ( string channelId )
{
2015-10-24 15:33:22 +00:00
return Path . Combine ( _config . CommonApplicationPaths . CachePath , "embytvepg" , channelId + ".json" ) ;
2015-07-20 18:32:55 +00:00
}
private readonly object _epgLock = new object ( ) ;
private void SaveEpgDataForChannel ( string channelId , List < ProgramInfo > epgData )
{
var path = GetChannelEpgCachePath ( channelId ) ;
2015-09-21 16:26:05 +00:00
_fileSystem . CreateDirectory ( Path . GetDirectoryName ( path ) ) ;
2015-07-20 18:32:55 +00:00
lock ( _epgLock )
{
_jsonSerializer . SerializeToFile ( epgData , path ) ;
}
}
private List < ProgramInfo > GetEpgDataForChannel ( string channelId )
{
try
{
lock ( _epgLock )
{
return _jsonSerializer . DeserializeFromFile < List < ProgramInfo > > ( GetChannelEpgCachePath ( channelId ) ) ;
}
}
catch
{
return new List < ProgramInfo > ( ) ;
}
}
2015-08-15 21:58:52 +00:00
private List < ProgramInfo > GetEpgDataForChannels ( List < string > channelIds )
2015-07-20 18:32:55 +00:00
{
2015-08-15 21:58:52 +00:00
return channelIds . SelectMany ( GetEpgDataForChannel ) . ToList ( ) ;
2015-07-20 18:32:55 +00:00
}
public void Dispose ( )
{
foreach ( var pair in _activeRecordings . ToList ( ) )
{
2016-03-01 04:24:42 +00:00
pair . Value . CancellationTokenSource . Cancel ( ) ;
2015-07-20 18:32:55 +00:00
}
}
2015-08-21 19:25:35 +00:00
public Task < MBRegistrationRecord > GetRegistrationInfo ( string feature )
{
if ( string . Equals ( feature , "seriesrecordings" , StringComparison . OrdinalIgnoreCase ) )
{
return _security . GetRegistrationStatus ( "embytvseriesrecordings" ) ;
}
return Task . FromResult ( new MBRegistrationRecord
{
IsValid = true ,
IsRegistered = true
} ) ;
}
2016-03-01 04:24:42 +00:00
2016-05-04 20:50:47 +00:00
public List < VirtualFolderInfo > GetRecordingFolders ( )
{
var list = new List < VirtualFolderInfo > ( ) ;
var defaultFolder = RecordingPath ;
var defaultName = "Recordings" ;
if ( Directory . Exists ( defaultFolder ) )
{
list . Add ( new VirtualFolderInfo
{
Locations = new List < string > { defaultFolder } ,
Name = defaultName
} ) ;
}
var customPath = GetConfiguration ( ) . MovieRecordingPath ;
if ( ( ! string . IsNullOrWhiteSpace ( customPath ) & & ! string . Equals ( customPath , defaultFolder , StringComparison . OrdinalIgnoreCase ) ) & & Directory . Exists ( customPath ) )
{
list . Add ( new VirtualFolderInfo
{
Locations = new List < string > { customPath } ,
Name = "Recorded Movies" ,
CollectionType = CollectionType . Movies
} ) ;
}
customPath = GetConfiguration ( ) . SeriesRecordingPath ;
if ( ( ! string . IsNullOrWhiteSpace ( customPath ) & & ! string . Equals ( customPath , defaultFolder , StringComparison . OrdinalIgnoreCase ) ) & & Directory . Exists ( customPath ) )
{
list . Add ( new VirtualFolderInfo
{
Locations = new List < string > { customPath } ,
Name = "Recorded Series" ,
CollectionType = CollectionType . TvShows
} ) ;
}
return list ;
}
2016-03-01 04:24:42 +00:00
class ActiveRecordingInfo
{
public string Path { get ; set ; }
public string TimerId { get ; set ; }
public CancellationTokenSource CancellationTokenSource { get ; set ; }
}
2015-07-20 18:32:55 +00:00
}
2015-09-21 16:26:05 +00:00
}