mirror of https://github.com/ErsatzTV/ErsatzTV.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
195 lines
7.5 KiB
195 lines
7.5 KiB
using System; |
|
using System.Collections.Generic; |
|
using System.IO; |
|
using System.Linq; |
|
using System.Threading.Tasks; |
|
using ErsatzTV.Core.Domain; |
|
using ErsatzTV.Core.Errors; |
|
using ErsatzTV.Core.Interfaces.Images; |
|
using ErsatzTV.Core.Interfaces.Metadata; |
|
using ErsatzTV.Core.Interfaces.Repositories; |
|
using ErsatzTV.Core.Interfaces.Search; |
|
using LanguageExt; |
|
using MediatR; |
|
using Microsoft.Extensions.Logging; |
|
using static LanguageExt.Prelude; |
|
using Unit = LanguageExt.Unit; |
|
|
|
namespace ErsatzTV.Core.Metadata |
|
{ |
|
public class SongFolderScanner : LocalFolderScanner, ISongFolderScanner |
|
{ |
|
private readonly ILocalFileSystem _localFileSystem; |
|
private readonly ILocalMetadataProvider _localMetadataProvider; |
|
private readonly IMediator _mediator; |
|
private readonly ISearchIndex _searchIndex; |
|
private readonly ISearchRepository _searchRepository; |
|
private readonly ISongRepository _songRepository; |
|
private readonly ILibraryRepository _libraryRepository; |
|
private readonly ILogger<SongFolderScanner> _logger; |
|
|
|
public SongFolderScanner( |
|
ILocalFileSystem localFileSystem, |
|
ILocalStatisticsProvider localStatisticsProvider, |
|
ILocalMetadataProvider localMetadataProvider, |
|
IMetadataRepository metadataRepository, |
|
IImageCache imageCache, |
|
IMediator mediator, |
|
ISearchIndex searchIndex, |
|
ISearchRepository searchRepository, |
|
ISongRepository songRepository, |
|
ILibraryRepository libraryRepository, |
|
ILogger<SongFolderScanner> logger) : base( |
|
localFileSystem, |
|
localStatisticsProvider, |
|
metadataRepository, |
|
imageCache, |
|
logger) |
|
{ |
|
_localFileSystem = localFileSystem; |
|
_localMetadataProvider = localMetadataProvider; |
|
_mediator = mediator; |
|
_searchIndex = searchIndex; |
|
_searchRepository = searchRepository; |
|
_songRepository = songRepository; |
|
_libraryRepository = libraryRepository; |
|
_logger = logger; |
|
} |
|
|
|
public async Task<Either<BaseError, Unit>> ScanFolder( |
|
LibraryPath libraryPath, |
|
string ffprobePath, |
|
decimal progressMin, |
|
decimal progressMax) |
|
{ |
|
decimal progressSpread = progressMax - progressMin; |
|
|
|
if (!_localFileSystem.IsLibraryPathAccessible(libraryPath)) |
|
{ |
|
return new MediaSourceInaccessible(); |
|
} |
|
|
|
var foldersCompleted = 0; |
|
|
|
var folderQueue = new Queue<string>(); |
|
foreach (string folder in _localFileSystem.ListSubdirectories(libraryPath.Path) |
|
.Filter(ShouldIncludeFolder) |
|
.OrderBy(identity)) |
|
{ |
|
folderQueue.Enqueue(folder); |
|
} |
|
|
|
while (folderQueue.Count > 0) |
|
{ |
|
decimal percentCompletion = (decimal)foldersCompleted / (foldersCompleted + folderQueue.Count); |
|
await _mediator.Publish( |
|
new LibraryScanProgress(libraryPath.LibraryId, progressMin + percentCompletion * progressSpread)); |
|
|
|
string songFolder = folderQueue.Dequeue(); |
|
foldersCompleted++; |
|
|
|
var filesForEtag = _localFileSystem.ListFiles(songFolder).ToList(); |
|
|
|
var allFiles = filesForEtag |
|
.Filter(f => AudioFileExtensions.Contains(Path.GetExtension(f))) |
|
.Filter(f => !Path.GetFileName(f).StartsWith("._")) |
|
.ToList(); |
|
|
|
foreach (string subdirectory in _localFileSystem.ListSubdirectories(songFolder) |
|
.Filter(ShouldIncludeFolder) |
|
.OrderBy(identity)) |
|
{ |
|
folderQueue.Enqueue(subdirectory); |
|
} |
|
|
|
string etag = FolderEtag.Calculate(songFolder, _localFileSystem); |
|
Option<LibraryFolder> knownFolder = libraryPath.LibraryFolders |
|
.Filter(f => f.Path == songFolder) |
|
.HeadOrNone(); |
|
|
|
// skip folder if etag matches |
|
if (!allFiles.Any() || await knownFolder.Map(f => f.Etag ?? string.Empty).IfNoneAsync(string.Empty) == etag) |
|
{ |
|
continue; |
|
} |
|
|
|
_logger.LogDebug( |
|
"UPDATE: Etag has changed for folder {Folder}", |
|
songFolder); |
|
|
|
foreach (string file in allFiles.OrderBy(identity)) |
|
{ |
|
Either<BaseError, MediaItemScanResult<Song>> maybeSong = await _songRepository |
|
.GetOrAdd(libraryPath, file) |
|
.BindT(video => UpdateStatistics(video, ffprobePath)) |
|
.BindT(UpdateMetadata); |
|
|
|
await maybeSong.Match( |
|
async result => |
|
{ |
|
if (result.IsAdded) |
|
{ |
|
await _searchIndex.AddItems(_searchRepository, new List<MediaItem> { result.Item }); |
|
} |
|
else if (result.IsUpdated) |
|
{ |
|
await _searchIndex.UpdateItems(_searchRepository, new List<MediaItem> { result.Item }); |
|
} |
|
|
|
await _libraryRepository.SetEtag(libraryPath, knownFolder, songFolder, etag); |
|
}, |
|
error => |
|
{ |
|
_logger.LogWarning("Error processing song at {Path}: {Error}", file, error.Value); |
|
return Task.CompletedTask; |
|
}); |
|
} |
|
} |
|
|
|
foreach (string path in await _songRepository.FindSongPaths(libraryPath)) |
|
{ |
|
if (!_localFileSystem.FileExists(path)) |
|
{ |
|
_logger.LogInformation("Removing missing song at {Path}", path); |
|
List<int> songIds = await _songRepository.DeleteByPath(libraryPath, path); |
|
await _searchIndex.RemoveItems(songIds); |
|
} |
|
else if (Path.GetFileName(path).StartsWith("._")) |
|
{ |
|
_logger.LogInformation("Removing dot underscore file at {Path}", path); |
|
List<int> songIds = await _songRepository.DeleteByPath(libraryPath, path); |
|
await _searchIndex.RemoveItems(songIds); |
|
} |
|
} |
|
|
|
_searchIndex.Commit(); |
|
return Unit.Default; |
|
} |
|
|
|
private async Task<Either<BaseError, MediaItemScanResult<Song>>> UpdateMetadata( |
|
MediaItemScanResult<Song> result) |
|
{ |
|
try |
|
{ |
|
Song song = result.Item; |
|
if (!Optional(song.SongMetadata).Flatten().Any()) |
|
{ |
|
song.SongMetadata ??= new List<SongMetadata>(); |
|
|
|
string path = song.MediaVersions.Head().MediaFiles.Head().Path; |
|
_logger.LogDebug("Refreshing {Attribute} for {Path}", "Fallback Metadata", path); |
|
if (await _localMetadataProvider.RefreshFallbackMetadata(song)) |
|
{ |
|
result.IsUpdated = true; |
|
} |
|
} |
|
|
|
return result; |
|
} |
|
catch (Exception ex) |
|
{ |
|
return BaseError.New(ex.ToString()); |
|
} |
|
} |
|
} |
|
}
|
|
|