| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131 | using System;using System.Collections.Generic;using System.Diagnostics;using System.Linq;using Jellyfin.Database.Implementations;using Jellyfin.Database.Implementations.Entities;using Jellyfin.Extensions;using MediaBrowser.Controller;using MediaBrowser.Controller.Configuration;using MediaBrowser.Controller.Entities;using MediaBrowser.Controller.Entities.Audio;using MediaBrowser.Controller.Library;using MediaBrowser.Model.IO;using Microsoft.EntityFrameworkCore;using Microsoft.Extensions.Logging;namespace Jellyfin.Server.Migrations.Routines;/// <summary>/// Migration to re-read creation dates for library items with internal metadata paths./// </summary>[JellyfinMigration("2025-04-20T23:00:00", nameof(RefreshInternalDateModified))]public class RefreshInternalDateModified : IDatabaseMigrationRoutine{    private readonly ILogger<RefreshInternalDateModified> _logger;    private readonly IDbContextFactory<JellyfinDbContext> _dbProvider;    private readonly IFileSystem _fileSystem;    private readonly IServerApplicationHost _applicationHost;    private readonly bool _useFileCreationTimeForDateAdded;    private IReadOnlyList<string> _internalTypes = [         typeof(Genre).FullName!,         typeof(MusicGenre).FullName!,         typeof(MusicArtist).FullName!,         typeof(People).FullName!,         typeof(Studio).FullName!    ];    private IReadOnlyList<string> _internalPaths;    /// <summary>    /// Initializes a new instance of the <see cref="RefreshInternalDateModified"/> class.    /// </summary>    /// <param name="applicationHost">Instance of the <see cref="IServerApplicationHost"/> interface.</param>    /// <param name="applicationPaths">Instance of the <see cref="IServerApplicationPaths"/> interface.</param>    /// <param name="configurationManager">Instance of the <see cref="IServerConfigurationManager"/> interface.</param>    /// <param name="dbProvider">Instance of the <see cref="IDbContextFactory{JellyfinDbContext}"/> interface.</param>    /// <param name="logger">The logger.</param>    /// <param name="fileSystem">Instance of the <see cref="IFileSystem"/> interface.</param>    public RefreshInternalDateModified(        IServerApplicationHost applicationHost,        IServerApplicationPaths applicationPaths,        IServerConfigurationManager configurationManager,        IDbContextFactory<JellyfinDbContext> dbProvider,        ILogger<RefreshInternalDateModified> logger,        IFileSystem fileSystem)    {        _dbProvider = dbProvider;        _logger = logger;        _fileSystem = fileSystem;        _applicationHost = applicationHost;        _internalPaths = [            applicationPaths.ArtistsPath,            applicationPaths.GenrePath,            applicationPaths.MusicGenrePath,            applicationPaths.StudioPath,            applicationPaths.PeoplePath        ];        _useFileCreationTimeForDateAdded = configurationManager.GetMetadataConfiguration().UseFileCreationTimeForDateAdded;    }    /// <inheritdoc />    public void Perform()    {        const int Limit = 5000;        int itemCount = 0, offset = 0;        var sw = Stopwatch.StartNew();        using var context = _dbProvider.CreateDbContext();        var records = context.BaseItems.Count(b => _internalTypes.Contains(b.Type));        _logger.LogInformation("Checking if {Count} potentially internal items require refreshed DateModified", records);        do        {            var results = context.BaseItems                            .Where(b => _internalTypes.Contains(b.Type))                            .OrderBy(e => e.Id)                            .Skip(offset)                            .Take(Limit)                            .ToList();            foreach (var item in results)            {                var itemPath = item.Path;                if (itemPath is not null)                {                    var realPath = _applicationHost.ExpandVirtualPath(item.Path);                    if (_internalPaths.Any(path => realPath.StartsWith(path, StringComparison.Ordinal)))                    {                        var writeTime = _fileSystem.GetLastWriteTimeUtc(realPath);                        var itemModificationTime = item.DateModified;                        if (writeTime != itemModificationTime)                        {                            _logger.LogDebug("Reset file modification date: Old: {Old} - New: {New} - Path: {Path}", itemModificationTime, writeTime, realPath);                            item.DateModified = writeTime;                            if (_useFileCreationTimeForDateAdded)                            {                                item.DateCreated = _fileSystem.GetCreationTimeUtc(realPath);                            }                            itemCount++;                        }                    }                }            }            offset += Limit;            if (offset > records)            {                offset = records;            }            _logger.LogInformation("Checked: {Count} - Refreshed: {Items} - Time: {Time}", offset, itemCount, sw.Elapsed);        } while (offset < records);        context.SaveChanges();        _logger.LogInformation("Refreshed DateModified for {Count} items in {Time}", itemCount, sw.Elapsed);    }}
 |