| | 1 | | using System; |
| | 2 | | using System.Collections.Concurrent; |
| | 3 | | using System.Collections.Generic; |
| | 4 | | using System.Diagnostics; |
| | 5 | | using System.Globalization; |
| | 6 | | using System.IO; |
| | 7 | | using System.Linq; |
| | 8 | | using System.Net.Http; |
| | 9 | | using System.Threading; |
| | 10 | | using System.Threading.Tasks; |
| | 11 | | using AsyncKeyedLock; |
| | 12 | | using Jellyfin.Data.Enums; |
| | 13 | | using Jellyfin.Database.Implementations.Enums; |
| | 14 | | using Jellyfin.LiveTv.Configuration; |
| | 15 | | using Jellyfin.LiveTv.IO; |
| | 16 | | using Jellyfin.LiveTv.Timers; |
| | 17 | | using MediaBrowser.Common.Configuration; |
| | 18 | | using MediaBrowser.Controller.Configuration; |
| | 19 | | using MediaBrowser.Controller.Dto; |
| | 20 | | using MediaBrowser.Controller.Entities; |
| | 21 | | using MediaBrowser.Controller.Entities.TV; |
| | 22 | | using MediaBrowser.Controller.Library; |
| | 23 | | using MediaBrowser.Controller.LiveTv; |
| | 24 | | using MediaBrowser.Controller.MediaEncoding; |
| | 25 | | using MediaBrowser.Controller.Providers; |
| | 26 | | using MediaBrowser.Model.Configuration; |
| | 27 | | using MediaBrowser.Model.Dto; |
| | 28 | | using MediaBrowser.Model.Entities; |
| | 29 | | using MediaBrowser.Model.IO; |
| | 30 | | using MediaBrowser.Model.LiveTv; |
| | 31 | | using MediaBrowser.Model.MediaInfo; |
| | 32 | | using MediaBrowser.Model.Providers; |
| | 33 | | using Microsoft.Extensions.Logging; |
| | 34 | |
|
| | 35 | | namespace Jellyfin.LiveTv.Recordings; |
| | 36 | |
|
| | 37 | | /// <inheritdoc cref="IRecordingsManager" /> |
| | 38 | | public sealed class RecordingsManager : IRecordingsManager, IDisposable |
| | 39 | | { |
| | 40 | | private readonly ILogger<RecordingsManager> _logger; |
| | 41 | | private readonly IServerConfigurationManager _config; |
| | 42 | | private readonly IHttpClientFactory _httpClientFactory; |
| | 43 | | private readonly IFileSystem _fileSystem; |
| | 44 | | private readonly ILibraryManager _libraryManager; |
| | 45 | | private readonly ILibraryMonitor _libraryMonitor; |
| | 46 | | private readonly IProviderManager _providerManager; |
| | 47 | | private readonly IMediaEncoder _mediaEncoder; |
| | 48 | | private readonly IMediaSourceManager _mediaSourceManager; |
| | 49 | | private readonly IStreamHelper _streamHelper; |
| | 50 | | private readonly TimerManager _timerManager; |
| | 51 | | private readonly SeriesTimerManager _seriesTimerManager; |
| | 52 | | private readonly RecordingsMetadataManager _recordingsMetadataManager; |
| | 53 | |
|
| 21 | 54 | | private readonly ConcurrentDictionary<string, ActiveRecordingInfo> _activeRecordings = new(StringComparer.OrdinalIgn |
| 21 | 55 | | private readonly AsyncNonKeyedLocker _recordingDeleteSemaphore = new(); |
| | 56 | | private bool _disposed; |
| | 57 | |
|
| | 58 | | /// <summary> |
| | 59 | | /// Initializes a new instance of the <see cref="RecordingsManager"/> class. |
| | 60 | | /// </summary> |
| | 61 | | /// <param name="logger">The <see cref="ILogger"/>.</param> |
| | 62 | | /// <param name="config">The <see cref="IServerConfigurationManager"/>.</param> |
| | 63 | | /// <param name="httpClientFactory">The <see cref="IHttpClientFactory"/>.</param> |
| | 64 | | /// <param name="fileSystem">The <see cref="IFileSystem"/>.</param> |
| | 65 | | /// <param name="libraryManager">The <see cref="ILibraryManager"/>.</param> |
| | 66 | | /// <param name="libraryMonitor">The <see cref="ILibraryMonitor"/>.</param> |
| | 67 | | /// <param name="providerManager">The <see cref="IProviderManager"/>.</param> |
| | 68 | | /// <param name="mediaEncoder">The <see cref="IMediaEncoder"/>.</param> |
| | 69 | | /// <param name="mediaSourceManager">The <see cref="IMediaSourceManager"/>.</param> |
| | 70 | | /// <param name="streamHelper">The <see cref="IStreamHelper"/>.</param> |
| | 71 | | /// <param name="timerManager">The <see cref="TimerManager"/>.</param> |
| | 72 | | /// <param name="seriesTimerManager">The <see cref="SeriesTimerManager"/>.</param> |
| | 73 | | /// <param name="recordingsMetadataManager">The <see cref="RecordingsMetadataManager"/>.</param> |
| | 74 | | public RecordingsManager( |
| | 75 | | ILogger<RecordingsManager> logger, |
| | 76 | | IServerConfigurationManager config, |
| | 77 | | IHttpClientFactory httpClientFactory, |
| | 78 | | IFileSystem fileSystem, |
| | 79 | | ILibraryManager libraryManager, |
| | 80 | | ILibraryMonitor libraryMonitor, |
| | 81 | | IProviderManager providerManager, |
| | 82 | | IMediaEncoder mediaEncoder, |
| | 83 | | IMediaSourceManager mediaSourceManager, |
| | 84 | | IStreamHelper streamHelper, |
| | 85 | | TimerManager timerManager, |
| | 86 | | SeriesTimerManager seriesTimerManager, |
| | 87 | | RecordingsMetadataManager recordingsMetadataManager) |
| | 88 | | { |
| 21 | 89 | | _logger = logger; |
| 21 | 90 | | _config = config; |
| 21 | 91 | | _httpClientFactory = httpClientFactory; |
| 21 | 92 | | _fileSystem = fileSystem; |
| 21 | 93 | | _libraryManager = libraryManager; |
| 21 | 94 | | _libraryMonitor = libraryMonitor; |
| 21 | 95 | | _providerManager = providerManager; |
| 21 | 96 | | _mediaEncoder = mediaEncoder; |
| 21 | 97 | | _mediaSourceManager = mediaSourceManager; |
| 21 | 98 | | _streamHelper = streamHelper; |
| 21 | 99 | | _timerManager = timerManager; |
| 21 | 100 | | _seriesTimerManager = seriesTimerManager; |
| 21 | 101 | | _recordingsMetadataManager = recordingsMetadataManager; |
| | 102 | |
|
| 21 | 103 | | _config.NamedConfigurationUpdated += OnNamedConfigurationUpdated; |
| 21 | 104 | | } |
| | 105 | |
|
| | 106 | | private string DefaultRecordingPath |
| | 107 | | { |
| | 108 | | get |
| | 109 | | { |
| 23 | 110 | | var path = _config.GetLiveTvConfiguration().RecordingPath; |
| | 111 | |
|
| 23 | 112 | | return string.IsNullOrWhiteSpace(path) |
| 23 | 113 | | ? Path.Combine(_config.CommonApplicationPaths.DataPath, "livetv", "recordings") |
| 23 | 114 | | : path; |
| | 115 | | } |
| | 116 | | } |
| | 117 | |
|
| | 118 | | /// <inheritdoc /> |
| | 119 | | public string? GetActiveRecordingPath(string id) |
| 0 | 120 | | => _activeRecordings.GetValueOrDefault(id)?.Path; |
| | 121 | |
|
| | 122 | | /// <inheritdoc /> |
| | 123 | | public ActiveRecordingInfo? GetActiveRecordingInfo(string path) |
| | 124 | | { |
| 9 | 125 | | if (string.IsNullOrWhiteSpace(path) || _activeRecordings.IsEmpty) |
| | 126 | | { |
| 9 | 127 | | return null; |
| | 128 | | } |
| | 129 | |
|
| 0 | 130 | | foreach (var (_, recordingInfo) in _activeRecordings) |
| | 131 | | { |
| 0 | 132 | | if (string.Equals(recordingInfo.Path, path, StringComparison.Ordinal) |
| 0 | 133 | | && !recordingInfo.CancellationTokenSource.IsCancellationRequested) |
| | 134 | | { |
| 0 | 135 | | return recordingInfo.Timer.Status == RecordingStatus.InProgress ? recordingInfo : null; |
| | 136 | | } |
| | 137 | | } |
| | 138 | |
|
| 0 | 139 | | return null; |
| 0 | 140 | | } |
| | 141 | |
|
| | 142 | | /// <inheritdoc /> |
| | 143 | | public IEnumerable<VirtualFolderInfo> GetRecordingFolders() |
| | 144 | | { |
| | 145 | | if (Directory.Exists(DefaultRecordingPath)) |
| | 146 | | { |
| | 147 | | yield return new VirtualFolderInfo |
| | 148 | | { |
| | 149 | | Locations = [DefaultRecordingPath], |
| | 150 | | Name = "Recordings" |
| | 151 | | }; |
| | 152 | | } |
| | 153 | |
|
| | 154 | | var customPath = _config.GetLiveTvConfiguration().MovieRecordingPath; |
| | 155 | | if (!string.IsNullOrWhiteSpace(customPath) |
| | 156 | | && !string.Equals(customPath, DefaultRecordingPath, StringComparison.OrdinalIgnoreCase) |
| | 157 | | && Directory.Exists(customPath)) |
| | 158 | | { |
| | 159 | | yield return new VirtualFolderInfo |
| | 160 | | { |
| | 161 | | Locations = [customPath], |
| | 162 | | Name = "Recorded Movies", |
| | 163 | | CollectionType = CollectionTypeOptions.movies |
| | 164 | | }; |
| | 165 | | } |
| | 166 | |
|
| | 167 | | customPath = _config.GetLiveTvConfiguration().SeriesRecordingPath; |
| | 168 | | if (!string.IsNullOrWhiteSpace(customPath) |
| | 169 | | && !string.Equals(customPath, DefaultRecordingPath, StringComparison.OrdinalIgnoreCase) |
| | 170 | | && Directory.Exists(customPath)) |
| | 171 | | { |
| | 172 | | yield return new VirtualFolderInfo |
| | 173 | | { |
| | 174 | | Locations = [customPath], |
| | 175 | | Name = "Recorded Shows", |
| | 176 | | CollectionType = CollectionTypeOptions.tvshows |
| | 177 | | }; |
| | 178 | | } |
| | 179 | | } |
| | 180 | |
|
| | 181 | | /// <inheritdoc /> |
| | 182 | | public async Task CreateRecordingFolders() |
| | 183 | | { |
| | 184 | | try |
| | 185 | | { |
| | 186 | | var recordingFolders = GetRecordingFolders().ToArray(); |
| | 187 | | var virtualFolders = _libraryManager.GetVirtualFolders(); |
| | 188 | |
|
| | 189 | | var allExistingPaths = virtualFolders.SelectMany(i => i.Locations).ToList(); |
| | 190 | |
|
| | 191 | | var pathsAdded = new List<string>(); |
| | 192 | |
|
| | 193 | | foreach (var recordingFolder in recordingFolders) |
| | 194 | | { |
| | 195 | | var pathsToCreate = recordingFolder.Locations |
| | 196 | | .Where(i => !allExistingPaths.Any(p => _fileSystem.AreEqual(p, i))) |
| | 197 | | .ToList(); |
| | 198 | |
|
| | 199 | | if (pathsToCreate.Count == 0) |
| | 200 | | { |
| | 201 | | continue; |
| | 202 | | } |
| | 203 | |
|
| | 204 | | var mediaPathInfos = pathsToCreate.Select(i => new MediaPathInfo(i)).ToArray(); |
| | 205 | | var libraryOptions = new LibraryOptions |
| | 206 | | { |
| | 207 | | PathInfos = mediaPathInfos |
| | 208 | | }; |
| | 209 | |
|
| | 210 | | try |
| | 211 | | { |
| | 212 | | await _libraryManager |
| | 213 | | .AddVirtualFolder(recordingFolder.Name, recordingFolder.CollectionType, libraryOptions, true) |
| | 214 | | .ConfigureAwait(false); |
| | 215 | | } |
| | 216 | | catch (Exception ex) |
| | 217 | | { |
| | 218 | | _logger.LogError(ex, "Error creating virtual folder"); |
| | 219 | | } |
| | 220 | |
|
| | 221 | | pathsAdded.AddRange(pathsToCreate); |
| | 222 | | } |
| | 223 | |
|
| | 224 | | var config = _config.GetLiveTvConfiguration(); |
| | 225 | |
|
| | 226 | | var pathsToRemove = config.MediaLocationsCreated |
| | 227 | | .Except(recordingFolders.SelectMany(i => i.Locations)) |
| | 228 | | .ToList(); |
| | 229 | |
|
| | 230 | | if (pathsAdded.Count > 0 || pathsToRemove.Count > 0) |
| | 231 | | { |
| | 232 | | pathsAdded.InsertRange(0, config.MediaLocationsCreated); |
| | 233 | | config.MediaLocationsCreated = pathsAdded.Except(pathsToRemove).Distinct(StringComparer.OrdinalIgnoreCas |
| | 234 | | _config.SaveConfiguration("livetv", config); |
| | 235 | | } |
| | 236 | |
|
| | 237 | | foreach (var path in pathsToRemove) |
| | 238 | | { |
| | 239 | | await RemovePathFromLibraryAsync(path).ConfigureAwait(false); |
| | 240 | | } |
| | 241 | | } |
| | 242 | | catch (Exception ex) |
| | 243 | | { |
| | 244 | | _logger.LogError(ex, "Error creating recording folders"); |
| | 245 | | } |
| | 246 | | } |
| | 247 | |
|
| | 248 | | private async Task RemovePathFromLibraryAsync(string path) |
| | 249 | | { |
| | 250 | | _logger.LogDebug("Removing path from library: {0}", path); |
| | 251 | |
|
| | 252 | | var requiresRefresh = false; |
| | 253 | | var virtualFolders = _libraryManager.GetVirtualFolders(); |
| | 254 | |
|
| | 255 | | foreach (var virtualFolder in virtualFolders) |
| | 256 | | { |
| | 257 | | if (!virtualFolder.Locations.Contains(path, StringComparer.OrdinalIgnoreCase)) |
| | 258 | | { |
| | 259 | | continue; |
| | 260 | | } |
| | 261 | |
|
| | 262 | | if (virtualFolder.Locations.Length == 1) |
| | 263 | | { |
| | 264 | | try |
| | 265 | | { |
| | 266 | | await _libraryManager.RemoveVirtualFolder(virtualFolder.Name, true).ConfigureAwait(false); |
| | 267 | | } |
| | 268 | | catch (Exception ex) |
| | 269 | | { |
| | 270 | | _logger.LogError(ex, "Error removing virtual folder"); |
| | 271 | | } |
| | 272 | | } |
| | 273 | | else |
| | 274 | | { |
| | 275 | | try |
| | 276 | | { |
| | 277 | | _libraryManager.RemoveMediaPath(virtualFolder.Name, path); |
| | 278 | | requiresRefresh = true; |
| | 279 | | } |
| | 280 | | catch (Exception ex) |
| | 281 | | { |
| | 282 | | _logger.LogError(ex, "Error removing media path"); |
| | 283 | | } |
| | 284 | | } |
| | 285 | | } |
| | 286 | |
|
| | 287 | | if (requiresRefresh) |
| | 288 | | { |
| | 289 | | await _libraryManager.ValidateMediaLibrary(new Progress<double>(), CancellationToken.None).ConfigureAwait(fa |
| | 290 | | } |
| | 291 | | } |
| | 292 | |
|
| | 293 | | /// <inheritdoc /> |
| | 294 | | public void CancelRecording(string timerId, TimerInfo? timer) |
| | 295 | | { |
| 0 | 296 | | if (_activeRecordings.TryGetValue(timerId, out var activeRecordingInfo)) |
| | 297 | | { |
| 0 | 298 | | activeRecordingInfo.Timer = timer; |
| 0 | 299 | | activeRecordingInfo.CancellationTokenSource.Cancel(); |
| | 300 | | } |
| 0 | 301 | | } |
| | 302 | |
|
| | 303 | | /// <inheritdoc /> |
| | 304 | | public async Task RecordStream(ActiveRecordingInfo recordingInfo, BaseItem channel, DateTime recordingEndDate) |
| | 305 | | { |
| | 306 | | ArgumentNullException.ThrowIfNull(recordingInfo); |
| | 307 | | ArgumentNullException.ThrowIfNull(channel); |
| | 308 | |
|
| | 309 | | var timer = recordingInfo.Timer; |
| | 310 | | var remoteMetadata = await FetchInternetMetadata(timer, CancellationToken.None).ConfigureAwait(false); |
| | 311 | | var recordingPath = GetRecordingPath(timer, remoteMetadata, out var seriesPath); |
| | 312 | |
|
| | 313 | | string? liveStreamId = null; |
| | 314 | | RecordingStatus recordingStatus; |
| | 315 | | try |
| | 316 | | { |
| | 317 | | var allMediaSources = await _mediaSourceManager |
| | 318 | | .GetPlaybackMediaSources(channel, null, true, false, CancellationToken.None).ConfigureAwait(false); |
| | 319 | |
|
| | 320 | | var mediaStreamInfo = allMediaSources[0]; |
| | 321 | | IDirectStreamProvider? directStreamProvider = null; |
| | 322 | | if (mediaStreamInfo.RequiresOpening) |
| | 323 | | { |
| | 324 | | var liveStreamResponse = await _mediaSourceManager.OpenLiveStreamInternal( |
| | 325 | | new LiveStreamRequest |
| | 326 | | { |
| | 327 | | ItemId = channel.Id, |
| | 328 | | OpenToken = mediaStreamInfo.OpenToken |
| | 329 | | }, |
| | 330 | | CancellationToken.None).ConfigureAwait(false); |
| | 331 | |
|
| | 332 | | mediaStreamInfo = liveStreamResponse.Item1.MediaSource; |
| | 333 | | liveStreamId = mediaStreamInfo.LiveStreamId; |
| | 334 | | directStreamProvider = liveStreamResponse.Item2; |
| | 335 | | } |
| | 336 | |
|
| | 337 | | using var recorder = GetRecorder(mediaStreamInfo); |
| | 338 | |
|
| | 339 | | recordingPath = recorder.GetOutputPath(mediaStreamInfo, recordingPath); |
| | 340 | | recordingPath = EnsureFileUnique(recordingPath, timer.Id); |
| | 341 | |
|
| | 342 | | _libraryMonitor.ReportFileSystemChangeBeginning(recordingPath); |
| | 343 | |
|
| | 344 | | var duration = recordingEndDate - DateTime.UtcNow; |
| | 345 | |
|
| | 346 | | _logger.LogInformation("Beginning recording. Will record for {Duration} minutes.", duration.TotalMinutes); |
| | 347 | | _logger.LogInformation("Writing file to: {Path}", recordingPath); |
| | 348 | |
|
| | 349 | | async void OnStarted() |
| | 350 | | { |
| | 351 | | recordingInfo.Path = recordingPath; |
| | 352 | | _activeRecordings.TryAdd(timer.Id, recordingInfo); |
| | 353 | |
|
| | 354 | | timer.Status = RecordingStatus.InProgress; |
| | 355 | | _timerManager.AddOrUpdate(timer, false); |
| | 356 | |
|
| | 357 | | await _recordingsMetadataManager.SaveRecordingMetadata(timer, recordingPath, seriesPath).ConfigureAwait( |
| | 358 | | await CreateRecordingFolders().ConfigureAwait(false); |
| | 359 | |
|
| | 360 | | TriggerRefresh(recordingPath); |
| | 361 | | await EnforceKeepUpTo(timer, seriesPath).ConfigureAwait(false); |
| | 362 | | } |
| | 363 | |
|
| | 364 | | await recorder.Record( |
| | 365 | | directStreamProvider, |
| | 366 | | mediaStreamInfo, |
| | 367 | | recordingPath, |
| | 368 | | duration, |
| | 369 | | OnStarted, |
| | 370 | | recordingInfo.CancellationTokenSource.Token).ConfigureAwait(false); |
| | 371 | |
|
| | 372 | | recordingStatus = RecordingStatus.Completed; |
| | 373 | | _logger.LogInformation("Recording completed: {RecordPath}", recordingPath); |
| | 374 | | } |
| | 375 | | catch (OperationCanceledException) |
| | 376 | | { |
| | 377 | | _logger.LogInformation("Recording stopped: {RecordPath}", recordingPath); |
| | 378 | | recordingStatus = RecordingStatus.Completed; |
| | 379 | | } |
| | 380 | | catch (Exception ex) |
| | 381 | | { |
| | 382 | | _logger.LogError(ex, "Error recording to {RecordPath}", recordingPath); |
| | 383 | | recordingStatus = RecordingStatus.Error; |
| | 384 | | } |
| | 385 | |
|
| | 386 | | if (!string.IsNullOrWhiteSpace(liveStreamId)) |
| | 387 | | { |
| | 388 | | try |
| | 389 | | { |
| | 390 | | await _mediaSourceManager.CloseLiveStream(liveStreamId).ConfigureAwait(false); |
| | 391 | | } |
| | 392 | | catch (Exception ex) |
| | 393 | | { |
| | 394 | | _logger.LogError(ex, "Error closing live stream"); |
| | 395 | | } |
| | 396 | | } |
| | 397 | |
|
| | 398 | | DeleteFileIfEmpty(recordingPath); |
| | 399 | | TriggerRefresh(recordingPath); |
| | 400 | | _libraryMonitor.ReportFileSystemChangeComplete(recordingPath, false); |
| | 401 | | _activeRecordings.TryRemove(timer.Id, out _); |
| | 402 | |
|
| | 403 | | if (recordingStatus != RecordingStatus.Completed && DateTime.UtcNow < timer.EndDate && timer.RetryCount < 10) |
| | 404 | | { |
| | 405 | | const int RetryIntervalSeconds = 60; |
| | 406 | | _logger.LogInformation("Retrying recording in {0} seconds.", RetryIntervalSeconds); |
| | 407 | |
|
| | 408 | | timer.Status = RecordingStatus.New; |
| | 409 | | timer.PrePaddingSeconds = 0; |
| | 410 | | timer.StartDate = DateTime.UtcNow.AddSeconds(RetryIntervalSeconds); |
| | 411 | | timer.RetryCount++; |
| | 412 | | _timerManager.AddOrUpdate(timer); |
| | 413 | | } |
| | 414 | | else if (File.Exists(recordingPath)) |
| | 415 | | { |
| | 416 | | timer.RecordingPath = recordingPath; |
| | 417 | | timer.Status = RecordingStatus.Completed; |
| | 418 | | _timerManager.AddOrUpdate(timer, false); |
| | 419 | | await PostProcessRecording(recordingPath).ConfigureAwait(false); |
| | 420 | | } |
| | 421 | | else |
| | 422 | | { |
| | 423 | | _timerManager.Delete(timer); |
| | 424 | | } |
| | 425 | | } |
| | 426 | |
|
| | 427 | | /// <inheritdoc /> |
| | 428 | | public void Dispose() |
| | 429 | | { |
| 21 | 430 | | if (_disposed) |
| | 431 | | { |
| 0 | 432 | | return; |
| | 433 | | } |
| | 434 | |
|
| 21 | 435 | | _recordingDeleteSemaphore.Dispose(); |
| | 436 | |
|
| 42 | 437 | | foreach (var pair in _activeRecordings.ToList()) |
| | 438 | | { |
| 0 | 439 | | pair.Value.CancellationTokenSource.Cancel(); |
| | 440 | | } |
| | 441 | |
|
| 21 | 442 | | _disposed = true; |
| 21 | 443 | | } |
| | 444 | |
|
| | 445 | | private async void OnNamedConfigurationUpdated(object? sender, ConfigurationUpdateEventArgs e) |
| | 446 | | { |
| | 447 | | if (string.Equals(e.Key, "livetv", StringComparison.OrdinalIgnoreCase)) |
| | 448 | | { |
| | 449 | | await CreateRecordingFolders().ConfigureAwait(false); |
| | 450 | | } |
| | 451 | | } |
| | 452 | |
|
| | 453 | | private async Task<RemoteSearchResult?> FetchInternetMetadata(TimerInfo timer, CancellationToken cancellationToken) |
| | 454 | | { |
| | 455 | | if (!timer.IsSeries || timer.SeriesProviderIds.Count == 0) |
| | 456 | | { |
| | 457 | | return null; |
| | 458 | | } |
| | 459 | |
|
| | 460 | | var query = new RemoteSearchQuery<SeriesInfo> |
| | 461 | | { |
| | 462 | | SearchInfo = new SeriesInfo |
| | 463 | | { |
| | 464 | | ProviderIds = timer.SeriesProviderIds, |
| | 465 | | Name = timer.Name, |
| | 466 | | MetadataCountryCode = _config.Configuration.MetadataCountryCode, |
| | 467 | | MetadataLanguage = _config.Configuration.PreferredMetadataLanguage |
| | 468 | | } |
| | 469 | | }; |
| | 470 | |
|
| | 471 | | var results = await _providerManager.GetRemoteSearchResults<Series, SeriesInfo>(query, cancellationToken).Config |
| | 472 | |
|
| | 473 | | return results.FirstOrDefault(); |
| | 474 | | } |
| | 475 | |
|
| | 476 | | private string GetRecordingPath(TimerInfo timer, RemoteSearchResult? metadata, out string? seriesPath) |
| | 477 | | { |
| 0 | 478 | | var recordingPath = DefaultRecordingPath; |
| 0 | 479 | | var config = _config.GetLiveTvConfiguration(); |
| 0 | 480 | | seriesPath = null; |
| | 481 | |
|
| 0 | 482 | | if (timer.IsProgramSeries) |
| | 483 | | { |
| 0 | 484 | | var customRecordingPath = config.SeriesRecordingPath; |
| 0 | 485 | | var allowSubfolder = true; |
| 0 | 486 | | if (!string.IsNullOrWhiteSpace(customRecordingPath)) |
| | 487 | | { |
| 0 | 488 | | allowSubfolder = string.Equals(customRecordingPath, recordingPath, StringComparison.OrdinalIgnoreCase); |
| 0 | 489 | | recordingPath = customRecordingPath; |
| | 490 | | } |
| | 491 | |
|
| 0 | 492 | | if (allowSubfolder && config.EnableRecordingSubfolders) |
| | 493 | | { |
| 0 | 494 | | recordingPath = Path.Combine(recordingPath, "Series"); |
| | 495 | | } |
| | 496 | |
|
| | 497 | | // trim trailing period from the folder name |
| 0 | 498 | | var folderName = _fileSystem.GetValidFilename(timer.Name).Trim().TrimEnd('.').Trim(); |
| | 499 | |
|
| 0 | 500 | | if (metadata is not null && metadata.ProductionYear.HasValue) |
| | 501 | | { |
| 0 | 502 | | folderName += " (" + metadata.ProductionYear.Value.ToString(CultureInfo.InvariantCulture) + ")"; |
| | 503 | | } |
| | 504 | |
|
| | 505 | | // Can't use the year here in the folder name because it is the year of the episode, not the series. |
| 0 | 506 | | recordingPath = Path.Combine(recordingPath, folderName); |
| | 507 | |
|
| 0 | 508 | | seriesPath = recordingPath; |
| | 509 | |
|
| 0 | 510 | | if (timer.SeasonNumber.HasValue) |
| | 511 | | { |
| 0 | 512 | | folderName = string.Format( |
| 0 | 513 | | CultureInfo.InvariantCulture, |
| 0 | 514 | | "Season {0}", |
| 0 | 515 | | timer.SeasonNumber.Value); |
| 0 | 516 | | recordingPath = Path.Combine(recordingPath, folderName); |
| | 517 | | } |
| | 518 | | } |
| 0 | 519 | | else if (timer.IsMovie) |
| | 520 | | { |
| 0 | 521 | | var customRecordingPath = config.MovieRecordingPath; |
| 0 | 522 | | var allowSubfolder = true; |
| 0 | 523 | | if (!string.IsNullOrWhiteSpace(customRecordingPath)) |
| | 524 | | { |
| 0 | 525 | | allowSubfolder = string.Equals(customRecordingPath, recordingPath, StringComparison.OrdinalIgnoreCase); |
| 0 | 526 | | recordingPath = customRecordingPath; |
| | 527 | | } |
| | 528 | |
|
| 0 | 529 | | if (allowSubfolder && config.EnableRecordingSubfolders) |
| | 530 | | { |
| 0 | 531 | | recordingPath = Path.Combine(recordingPath, "Movies"); |
| | 532 | | } |
| | 533 | |
|
| 0 | 534 | | var folderName = _fileSystem.GetValidFilename(timer.Name).Trim(); |
| 0 | 535 | | if (timer.ProductionYear.HasValue) |
| | 536 | | { |
| 0 | 537 | | folderName += " (" + timer.ProductionYear.Value.ToString(CultureInfo.InvariantCulture) + ")"; |
| | 538 | | } |
| | 539 | |
|
| | 540 | | // trim trailing period from the folder name |
| 0 | 541 | | folderName = folderName.TrimEnd('.').Trim(); |
| | 542 | |
|
| 0 | 543 | | recordingPath = Path.Combine(recordingPath, folderName); |
| | 544 | | } |
| 0 | 545 | | else if (timer.IsKids) |
| | 546 | | { |
| 0 | 547 | | if (config.EnableRecordingSubfolders) |
| | 548 | | { |
| 0 | 549 | | recordingPath = Path.Combine(recordingPath, "Kids"); |
| | 550 | | } |
| | 551 | |
|
| 0 | 552 | | var folderName = _fileSystem.GetValidFilename(timer.Name).Trim(); |
| 0 | 553 | | if (timer.ProductionYear.HasValue) |
| | 554 | | { |
| 0 | 555 | | folderName += " (" + timer.ProductionYear.Value.ToString(CultureInfo.InvariantCulture) + ")"; |
| | 556 | | } |
| | 557 | |
|
| | 558 | | // trim trailing period from the folder name |
| 0 | 559 | | folderName = folderName.TrimEnd('.').Trim(); |
| | 560 | |
|
| 0 | 561 | | recordingPath = Path.Combine(recordingPath, folderName); |
| | 562 | | } |
| 0 | 563 | | else if (timer.IsSports) |
| | 564 | | { |
| 0 | 565 | | if (config.EnableRecordingSubfolders) |
| | 566 | | { |
| 0 | 567 | | recordingPath = Path.Combine(recordingPath, "Sports"); |
| | 568 | | } |
| | 569 | |
|
| 0 | 570 | | recordingPath = Path.Combine(recordingPath, _fileSystem.GetValidFilename(timer.Name).Trim()); |
| | 571 | | } |
| | 572 | | else |
| | 573 | | { |
| 0 | 574 | | if (config.EnableRecordingSubfolders) |
| | 575 | | { |
| 0 | 576 | | recordingPath = Path.Combine(recordingPath, "Other"); |
| | 577 | | } |
| | 578 | |
|
| 0 | 579 | | recordingPath = Path.Combine(recordingPath, _fileSystem.GetValidFilename(timer.Name).Trim()); |
| | 580 | | } |
| | 581 | |
|
| 0 | 582 | | var recordingFileName = _fileSystem.GetValidFilename(RecordingHelper.GetRecordingName(timer)).Trim() + ".ts"; |
| | 583 | |
|
| 0 | 584 | | return Path.Combine(recordingPath, recordingFileName); |
| | 585 | | } |
| | 586 | |
|
| | 587 | | private void DeleteFileIfEmpty(string path) |
| | 588 | | { |
| 0 | 589 | | var file = _fileSystem.GetFileInfo(path); |
| | 590 | |
|
| 0 | 591 | | if (file.Exists && file.Length == 0) |
| | 592 | | { |
| | 593 | | try |
| | 594 | | { |
| 0 | 595 | | _fileSystem.DeleteFile(path); |
| 0 | 596 | | } |
| 0 | 597 | | catch (Exception ex) |
| | 598 | | { |
| 0 | 599 | | _logger.LogError(ex, "Error deleting 0-byte failed recording file {Path}", path); |
| 0 | 600 | | } |
| | 601 | | } |
| 0 | 602 | | } |
| | 603 | |
|
| | 604 | | private void TriggerRefresh(string path) |
| | 605 | | { |
| 0 | 606 | | _logger.LogInformation("Triggering refresh on {Path}", path); |
| | 607 | |
|
| 0 | 608 | | var item = GetAffectedBaseItem(Path.GetDirectoryName(path)); |
| 0 | 609 | | if (item is null) |
| | 610 | | { |
| 0 | 611 | | return; |
| | 612 | | } |
| | 613 | |
|
| 0 | 614 | | _logger.LogInformation("Refreshing recording parent {Path}", item.Path); |
| 0 | 615 | | _providerManager.QueueRefresh( |
| 0 | 616 | | item.Id, |
| 0 | 617 | | new MetadataRefreshOptions(new DirectoryService(_fileSystem)) |
| 0 | 618 | | { |
| 0 | 619 | | RefreshPaths = |
| 0 | 620 | | [ |
| 0 | 621 | | path, |
| 0 | 622 | | Path.GetDirectoryName(path), |
| 0 | 623 | | Path.GetDirectoryName(Path.GetDirectoryName(path)) |
| 0 | 624 | | ] |
| 0 | 625 | | }, |
| 0 | 626 | | RefreshPriority.High); |
| 0 | 627 | | } |
| | 628 | |
|
| | 629 | | private BaseItem? GetAffectedBaseItem(string? path) |
| | 630 | | { |
| 0 | 631 | | BaseItem? item = null; |
| 0 | 632 | | var parentPath = Path.GetDirectoryName(path); |
| 0 | 633 | | while (item is null && !string.IsNullOrEmpty(path)) |
| | 634 | | { |
| 0 | 635 | | item = _libraryManager.FindByPath(path, null); |
| 0 | 636 | | path = Path.GetDirectoryName(path); |
| | 637 | | } |
| | 638 | |
|
| 0 | 639 | | if (item is not null |
| 0 | 640 | | && item.GetType() == typeof(Folder) |
| 0 | 641 | | && string.Equals(item.Path, parentPath, StringComparison.OrdinalIgnoreCase)) |
| | 642 | | { |
| 0 | 643 | | var parentItem = item.GetParent(); |
| 0 | 644 | | if (parentItem is not null && parentItem is not AggregateFolder) |
| | 645 | | { |
| 0 | 646 | | item = parentItem; |
| | 647 | | } |
| | 648 | | } |
| | 649 | |
|
| 0 | 650 | | return item; |
| | 651 | | } |
| | 652 | |
|
| | 653 | | private async Task EnforceKeepUpTo(TimerInfo timer, string? seriesPath) |
| | 654 | | { |
| | 655 | | if (string.IsNullOrWhiteSpace(timer.SeriesTimerId) |
| | 656 | | || string.IsNullOrWhiteSpace(seriesPath)) |
| | 657 | | { |
| | 658 | | return; |
| | 659 | | } |
| | 660 | |
|
| | 661 | | var seriesTimerId = timer.SeriesTimerId; |
| | 662 | | var seriesTimer = _seriesTimerManager.GetAll() |
| | 663 | | .FirstOrDefault(i => string.Equals(i.Id, seriesTimerId, StringComparison.OrdinalIgnoreCase)); |
| | 664 | |
|
| | 665 | | if (seriesTimer is null || seriesTimer.KeepUpTo <= 0) |
| | 666 | | { |
| | 667 | | return; |
| | 668 | | } |
| | 669 | |
|
| | 670 | | if (_disposed) |
| | 671 | | { |
| | 672 | | return; |
| | 673 | | } |
| | 674 | |
|
| | 675 | | using (await _recordingDeleteSemaphore.LockAsync().ConfigureAwait(false)) |
| | 676 | | { |
| | 677 | | if (_disposed) |
| | 678 | | { |
| | 679 | | return; |
| | 680 | | } |
| | 681 | |
|
| | 682 | | var timersToDelete = _timerManager.GetAll() |
| | 683 | | .Where(timerInfo => timerInfo.Status == RecordingStatus.Completed |
| | 684 | | && !string.IsNullOrWhiteSpace(timerInfo.RecordingPath) |
| | 685 | | && string.Equals(timerInfo.SeriesTimerId, seriesTimerId, StringComparison.OrdinalIgnoreCase) |
| | 686 | | && File.Exists(timerInfo.RecordingPath)) |
| | 687 | | .OrderByDescending(i => i.EndDate) |
| | 688 | | .Skip(seriesTimer.KeepUpTo - 1) |
| | 689 | | .ToList(); |
| | 690 | |
|
| | 691 | | DeleteLibraryItemsForTimers(timersToDelete); |
| | 692 | |
|
| | 693 | | if (_libraryManager.FindByPath(seriesPath, true) is not Folder librarySeries) |
| | 694 | | { |
| | 695 | | return; |
| | 696 | | } |
| | 697 | |
|
| | 698 | | var episodesToDelete = librarySeries.GetItemList( |
| | 699 | | new InternalItemsQuery |
| | 700 | | { |
| | 701 | | OrderBy = [(ItemSortBy.DateCreated, SortOrder.Descending)], |
| | 702 | | IsVirtualItem = false, |
| | 703 | | IsFolder = false, |
| | 704 | | Recursive = true, |
| | 705 | | DtoOptions = new DtoOptions(true) |
| | 706 | | }) |
| | 707 | | .Where(i => i.IsFileProtocol && File.Exists(i.Path)) |
| | 708 | | .Skip(seriesTimer.KeepUpTo - 1); |
| | 709 | |
|
| | 710 | | foreach (var item in episodesToDelete) |
| | 711 | | { |
| | 712 | | try |
| | 713 | | { |
| | 714 | | _libraryManager.DeleteItem( |
| | 715 | | item, |
| | 716 | | new DeleteOptions |
| | 717 | | { |
| | 718 | | DeleteFileLocation = true |
| | 719 | | }, |
| | 720 | | true); |
| | 721 | | } |
| | 722 | | catch (Exception ex) |
| | 723 | | { |
| | 724 | | _logger.LogError(ex, "Error deleting item"); |
| | 725 | | } |
| | 726 | | } |
| | 727 | | } |
| | 728 | | } |
| | 729 | |
|
| | 730 | | private void DeleteLibraryItemsForTimers(List<TimerInfo> timers) |
| | 731 | | { |
| 0 | 732 | | foreach (var timer in timers) |
| | 733 | | { |
| 0 | 734 | | if (_disposed) |
| | 735 | | { |
| 0 | 736 | | return; |
| | 737 | | } |
| | 738 | |
|
| | 739 | | try |
| | 740 | | { |
| 0 | 741 | | DeleteLibraryItemForTimer(timer); |
| 0 | 742 | | } |
| 0 | 743 | | catch (Exception ex) |
| | 744 | | { |
| 0 | 745 | | _logger.LogError(ex, "Error deleting recording"); |
| 0 | 746 | | } |
| | 747 | | } |
| 0 | 748 | | } |
| | 749 | |
|
| | 750 | | private void DeleteLibraryItemForTimer(TimerInfo timer) |
| | 751 | | { |
| 0 | 752 | | var libraryItem = _libraryManager.FindByPath(timer.RecordingPath, false); |
| 0 | 753 | | if (libraryItem is not null) |
| | 754 | | { |
| 0 | 755 | | _libraryManager.DeleteItem( |
| 0 | 756 | | libraryItem, |
| 0 | 757 | | new DeleteOptions |
| 0 | 758 | | { |
| 0 | 759 | | DeleteFileLocation = true |
| 0 | 760 | | }, |
| 0 | 761 | | true); |
| | 762 | | } |
| 0 | 763 | | else if (File.Exists(timer.RecordingPath)) |
| | 764 | | { |
| 0 | 765 | | _fileSystem.DeleteFile(timer.RecordingPath); |
| | 766 | | } |
| | 767 | |
|
| 0 | 768 | | _timerManager.Delete(timer); |
| 0 | 769 | | } |
| | 770 | |
|
| | 771 | | private string EnsureFileUnique(string path, string timerId) |
| | 772 | | { |
| 0 | 773 | | var parent = Path.GetDirectoryName(path)!; |
| 0 | 774 | | var name = Path.GetFileNameWithoutExtension(path); |
| 0 | 775 | | var extension = Path.GetExtension(path); |
| | 776 | |
|
| 0 | 777 | | var index = 1; |
| 0 | 778 | | while (File.Exists(path) || _activeRecordings.Any(i |
| 0 | 779 | | => string.Equals(i.Value.Path, path, StringComparison.OrdinalIgnoreCase) |
| 0 | 780 | | && !string.Equals(i.Value.Timer.Id, timerId, StringComparison.OrdinalIgnoreCase))) |
| | 781 | | { |
| 0 | 782 | | name += " - " + index.ToString(CultureInfo.InvariantCulture); |
| | 783 | |
|
| 0 | 784 | | path = Path.ChangeExtension(Path.Combine(parent, name), extension); |
| 0 | 785 | | index++; |
| | 786 | | } |
| | 787 | |
|
| 0 | 788 | | return path; |
| | 789 | | } |
| | 790 | |
|
| | 791 | | private IRecorder GetRecorder(MediaSourceInfo mediaSource) |
| | 792 | | { |
| 0 | 793 | | if (mediaSource.RequiresLooping |
| 0 | 794 | | || !(mediaSource.Container ?? string.Empty).EndsWith("ts", StringComparison.OrdinalIgnoreCase) |
| 0 | 795 | | || (mediaSource.Protocol != MediaProtocol.File && mediaSource.Protocol != MediaProtocol.Http)) |
| | 796 | | { |
| 0 | 797 | | return new EncodedRecorder(_logger, _mediaEncoder, _config.ApplicationPaths, _config); |
| | 798 | | } |
| | 799 | |
|
| 0 | 800 | | return new DirectRecorder(_logger, _httpClientFactory, _streamHelper); |
| | 801 | | } |
| | 802 | |
|
| | 803 | | private async Task PostProcessRecording(string path) |
| | 804 | | { |
| | 805 | | var options = _config.GetLiveTvConfiguration(); |
| | 806 | | if (string.IsNullOrWhiteSpace(options.RecordingPostProcessor)) |
| | 807 | | { |
| | 808 | | return; |
| | 809 | | } |
| | 810 | |
|
| | 811 | | try |
| | 812 | | { |
| | 813 | | using var process = new Process(); |
| | 814 | | process.StartInfo = new ProcessStartInfo |
| | 815 | | { |
| | 816 | | Arguments = options.RecordingPostProcessorArguments |
| | 817 | | .Replace("{path}", path, StringComparison.OrdinalIgnoreCase), |
| | 818 | | CreateNoWindow = true, |
| | 819 | | ErrorDialog = false, |
| | 820 | | FileName = options.RecordingPostProcessor, |
| | 821 | | WindowStyle = ProcessWindowStyle.Hidden, |
| | 822 | | UseShellExecute = false |
| | 823 | | }; |
| | 824 | | process.EnableRaisingEvents = true; |
| | 825 | |
|
| | 826 | | _logger.LogInformation("Running recording post processor {0} {1}", process.StartInfo.FileName, process.Start |
| | 827 | |
|
| | 828 | | process.Start(); |
| | 829 | | await process.WaitForExitAsync(CancellationToken.None).ConfigureAwait(false); |
| | 830 | |
|
| | 831 | | _logger.LogInformation("Recording post-processing script completed with exit code {ExitCode}", process.ExitC |
| | 832 | | } |
| | 833 | | catch (Exception ex) |
| | 834 | | { |
| | 835 | | _logger.LogError(ex, "Error running recording post processor"); |
| | 836 | | } |
| | 837 | | } |
| | 838 | | } |