AudioNormalizationPostScanTask.cs 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Diagnostics;
  4. using System.Globalization;
  5. using System.IO;
  6. using System.Linq;
  7. using System.Text.RegularExpressions;
  8. using System.Threading;
  9. using System.Threading.Tasks;
  10. using Jellyfin.Data.Enums;
  11. using MediaBrowser.Common.Configuration;
  12. using MediaBrowser.Controller.Entities;
  13. using MediaBrowser.Controller.Entities.Audio;
  14. using MediaBrowser.Controller.Library;
  15. using MediaBrowser.Controller.MediaEncoding;
  16. using MediaBrowser.Controller.Persistence;
  17. using MediaBrowser.Model.Globalization;
  18. using MediaBrowser.Model.Tasks;
  19. using Microsoft.Extensions.Logging;
  20. namespace Emby.Server.Implementations.ScheduledTasks.Tasks;
  21. /// <summary>
  22. /// The splashscreen post scan task.
  23. /// </summary>
  24. public partial class AudioNormalizationTask : IScheduledTask
  25. {
  26. private readonly IItemRepository _itemRepository;
  27. private readonly ILibraryManager _libraryManager;
  28. private readonly IMediaEncoder _mediaEncoder;
  29. private readonly IConfigurationManager _configurationManager;
  30. private readonly ILocalizationManager _localization;
  31. private readonly ILogger<AudioNormalizationTask> _logger;
  32. /// <summary>
  33. /// Initializes a new instance of the <see cref="AudioNormalizationTask"/> class.
  34. /// </summary>
  35. /// <param name="itemRepository">Instance of the <see cref="IItemRepository"/> interface.</param>
  36. /// <param name="libraryManager">Instance of the <see cref="ILibraryManager"/> interface.</param>
  37. /// <param name="mediaEncoder">Instance of the <see cref="IMediaEncoder"/> interface.</param>
  38. /// <param name="configurationManager">Instance of the <see cref="IConfigurationManager"/> interface.</param>
  39. /// <param name="localizationManager">Instance of the <see cref="ILocalizationManager"/> interface.</param>
  40. /// <param name="logger">Instance of the <see cref="ILogger{AudioNormalizationTask}"/> interface.</param>
  41. public AudioNormalizationTask(
  42. IItemRepository itemRepository,
  43. ILibraryManager libraryManager,
  44. IMediaEncoder mediaEncoder,
  45. IConfigurationManager configurationManager,
  46. ILocalizationManager localizationManager,
  47. ILogger<AudioNormalizationTask> logger)
  48. {
  49. _itemRepository = itemRepository;
  50. _libraryManager = libraryManager;
  51. _mediaEncoder = mediaEncoder;
  52. _configurationManager = configurationManager;
  53. _localization = localizationManager;
  54. _logger = logger;
  55. }
  56. /// <inheritdoc />
  57. public string Name => _localization.GetLocalizedString("TaskAudioNormalization");
  58. /// <inheritdoc />
  59. public string Description => _localization.GetLocalizedString("TaskAudioNormalizationDescription");
  60. /// <inheritdoc />
  61. public string Category => _localization.GetLocalizedString("TasksLibraryCategory");
  62. /// <inheritdoc />
  63. public string Key => "AudioNormalization";
  64. [GeneratedRegex(@"I:\s+(.*?)\s+LUFS")]
  65. private static partial Regex LUFSRegex();
  66. /// <inheritdoc />
  67. public async Task ExecuteAsync(IProgress<double> progress, CancellationToken cancellationToken)
  68. {
  69. foreach (var library in _libraryManager.RootFolder.Children)
  70. {
  71. var libraryOptions = _libraryManager.GetLibraryOptions(library);
  72. if (!libraryOptions.EnableLUFSScan)
  73. {
  74. continue;
  75. }
  76. // Album gain
  77. var albums = _libraryManager.GetItemList(new InternalItemsQuery
  78. {
  79. IncludeItemTypes = [BaseItemKind.MusicAlbum],
  80. Parent = library,
  81. Recursive = true
  82. });
  83. foreach (var a in albums)
  84. {
  85. if (a.NormalizationGain.HasValue || a.LUFS.HasValue)
  86. {
  87. continue;
  88. }
  89. var albumTracks = ((MusicAlbum)a).Tracks.Where(x => x.IsFileProtocol).ToList();
  90. if (albumTracks.Count == 0)
  91. {
  92. continue;
  93. }
  94. var tempFile = Path.Join(_configurationManager.GetTranscodePath(), Guid.NewGuid() + ".concat");
  95. var inputLines = albumTracks.Select(x => string.Format(CultureInfo.InvariantCulture, "file '{0}'", x.Path.Replace("'", @"'\''", StringComparison.Ordinal)));
  96. await File.WriteAllLinesAsync(tempFile, inputLines, cancellationToken).ConfigureAwait(false);
  97. a.LUFS = await CalculateLUFSAsync(
  98. string.Format(CultureInfo.InvariantCulture, "-f concat -safe 0 -i \"{0}\"", tempFile),
  99. cancellationToken).ConfigureAwait(false);
  100. File.Delete(tempFile);
  101. }
  102. _itemRepository.SaveItems(albums, cancellationToken);
  103. var tracks = _libraryManager.GetItemList(new InternalItemsQuery
  104. {
  105. MediaTypes = [MediaType.Audio],
  106. IncludeItemTypes = [BaseItemKind.Audio],
  107. Parent = library,
  108. Recursive = true
  109. });
  110. foreach (var t in tracks)
  111. {
  112. if (t.NormalizationGain.HasValue || t.LUFS.HasValue || !t.IsFileProtocol)
  113. {
  114. continue;
  115. }
  116. t.LUFS = await CalculateLUFSAsync(string.Format(CultureInfo.InvariantCulture, "-i \"{0}\"", t.Path.Replace("\"", "\\\"", StringComparison.Ordinal)), cancellationToken);
  117. }
  118. _itemRepository.SaveItems(tracks, cancellationToken);
  119. }
  120. }
  121. /// <inheritdoc />
  122. public IEnumerable<TaskTriggerInfo> GetDefaultTriggers()
  123. {
  124. return
  125. [
  126. new TaskTriggerInfo
  127. {
  128. Type = TaskTriggerInfo.TriggerInterval,
  129. IntervalTicks = TimeSpan.FromHours(24).Ticks
  130. }
  131. ];
  132. }
  133. private string EscapeFilename(string filename)
  134. => filename;
  135. private async Task<float?> CalculateLUFSAsync(string inputArgs, CancellationToken cancellationToken)
  136. {
  137. var args = $"-hide_banner {inputArgs} -af ebur128=framelog=verbose -f null -";
  138. using (var process = new Process()
  139. {
  140. StartInfo = new ProcessStartInfo
  141. {
  142. FileName = _mediaEncoder.EncoderPath,
  143. Arguments = args,
  144. RedirectStandardOutput = false,
  145. RedirectStandardError = true
  146. },
  147. })
  148. {
  149. try
  150. {
  151. _logger.LogDebug("Starting ffmpeg with arguments: {Arguments}", args);
  152. process.Start();
  153. }
  154. catch (Exception ex)
  155. {
  156. _logger.LogError(ex, "Error starting ffmpeg with arguments: {Arguments}", args);
  157. return null;
  158. }
  159. using var reader = process.StandardError;
  160. var output = await reader.ReadToEndAsync(cancellationToken).ConfigureAwait(false);
  161. cancellationToken.ThrowIfCancellationRequested();
  162. MatchCollection split = LUFSRegex().Matches(output);
  163. if (split.Count != 0)
  164. {
  165. return float.Parse(split[0].Groups[1].ValueSpan, CultureInfo.InvariantCulture.NumberFormat);
  166. }
  167. _logger.LogError("Failed to find LUFS value in output:\n{Output}", output);
  168. return null;
  169. }
  170. }
  171. }