Skip to content

Commit 85cac43

Browse files
authored
fix: add DB migration & endpoint to download missing TMDB People (#1203)
Added a new database migration and endpoint to download missing TMDB people in case the database gets into a bad state for whatever reason (though most likely because it failed to completely download one or more TMDB shows/movies). Fixes #1202
1 parent 9808915 commit 85cac43

File tree

6 files changed

+83
-3
lines changed

6 files changed

+83
-3
lines changed

Shoko.Server/API/v3/Controllers/ActionController.cs

+10
Original file line numberDiff line numberDiff line change
@@ -192,6 +192,16 @@ public ActionResult UpdateAllTmdbShows()
192192
return Ok();
193193
}
194194

195+
/// <summary>
196+
/// Download any missing TMDB People.
197+
/// </summary>
198+
[HttpGet("DownloadMissingTmdbPeople")]
199+
public ActionResult DownloadMissingTmdbPeople()
200+
{
201+
Task.Factory.StartNew(() => _tmdbService.RepairMissingPeople());
202+
return Ok();
203+
}
204+
195205
/// <summary>
196206
/// Purge all unused TMDB Shows that are not linked to any AniDB anime.
197207
/// </summary>

Shoko.Server/Databases/DatabaseFixes.cs

+6
Original file line numberDiff line numberDiff line change
@@ -837,4 +837,10 @@ public static void ClearQuartzQueue()
837837
var queueHandler = Utils.ServiceContainer.GetRequiredService<QueueHandler>();
838838
queueHandler.Clear().ConfigureAwait(false).GetAwaiter().GetResult();
839839
}
840+
841+
public static void RepairMissingTMDBPersons()
842+
{
843+
var service = Utils.ServiceContainer.GetRequiredService<TmdbMetadataService>();
844+
service.RepairMissingPeople().ConfigureAwait(false).GetAwaiter().GetResult();
845+
}
840846
}

Shoko.Server/Databases/MySQL.cs

+2-1
Original file line numberDiff line numberDiff line change
@@ -27,7 +27,7 @@ namespace Shoko.Server.Databases;
2727
public class MySQL : BaseDatabase<MySqlConnection>
2828
{
2929
public override string Name { get; } = "MySQL";
30-
public override int RequiredVersion { get; } = 139;
30+
public override int RequiredVersion { get; } = 140;
3131

3232
private List<DatabaseCommand> createVersionTable = new()
3333
{
@@ -849,6 +849,7 @@ public class MySQL : BaseDatabase<MySqlConnection>
849849
new(139, 10, "ALTER TABLE Trakt_Show ADD COLUMN TmdbShowID INT NULL;"),
850850
new(139, 11, DatabaseFixes.CleanupAfterRemovingTvDB),
851851
new(139, 12, DatabaseFixes.ClearQuartzQueue),
852+
new(140, 1, DatabaseFixes.RepairMissingTMDBPersons)
852853
};
853854

854855
private DatabaseCommand linuxTableVersionsFix = new("RENAME TABLE versions TO Versions;");

Shoko.Server/Databases/SQLServer.cs

+2-1
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,7 @@ namespace Shoko.Server.Databases;
2828
public class SQLServer : BaseDatabase<SqlConnection>
2929
{
3030
public override string Name { get; } = "SQLServer";
31-
public override int RequiredVersion { get; } = 131;
31+
public override int RequiredVersion { get; } = 132;
3232

3333
public override void BackupDatabase(string fullfilename)
3434
{
@@ -779,6 +779,7 @@ public override bool HasVersionsTable()
779779
new DatabaseCommand(131, 10, "ALTER TABLE Trakt_Show ADD TmdbShowID INT NULL;"),
780780
new DatabaseCommand(131, 11, DatabaseFixes.CleanupAfterRemovingTvDB),
781781
new DatabaseCommand(131, 12, DatabaseFixes.ClearQuartzQueue),
782+
new DatabaseCommand(132, 1, DatabaseFixes.RepairMissingTMDBPersons)
782783
};
783784

784785
private static void AlterImdbMovieIDType()

Shoko.Server/Databases/SQLite.cs

+2-1
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,7 @@ public class SQLite : BaseDatabase<SqliteConnection>
2828
{
2929
public override string Name => "SQLite";
3030

31-
public override int RequiredVersion => 123;
31+
public override int RequiredVersion => 124;
3232

3333
public override void BackupDatabase(string fullfilename)
3434
{
@@ -774,6 +774,7 @@ public override void CreateDatabase()
774774
new(123, 10, "ALTER TABLE Trakt_Show ADD COLUMN TmdbShowID INTEGER NULL;"),
775775
new(123, 11, DatabaseFixes.CleanupAfterRemovingTvDB),
776776
new(123, 12, DatabaseFixes.ClearQuartzQueue),
777+
new(124, 1, DatabaseFixes.RepairMissingTMDBPersons)
777778
};
778779

779780
private static Tuple<bool, string> MigrateRenamers(object connection)

Shoko.Server/Providers/TMDB/TmdbMetadataService.cs

+61
Original file line numberDiff line numberDiff line change
@@ -2031,6 +2031,39 @@ private void PurgeCompany(int companyId, bool removeImageFiles = true)
20312031

20322032
#region People
20332033

2034+
public async Task RepairMissingPeople()
2035+
{
2036+
var missingIds = new HashSet<int>();
2037+
var updateCount = 0;
2038+
var skippedCount = 0;
2039+
2040+
var peopleIds = _tmdbPeople.GetAll().Select(person => person.TmdbPersonID).ToHashSet();
2041+
2042+
foreach (var person in _tmdbEpisodeCast.GetAll())
2043+
if (!peopleIds.Contains(person.TmdbPersonID)) missingIds.Add(person.TmdbPersonID);
2044+
foreach (var person in _tmdbEpisodeCrew.GetAll())
2045+
if (!peopleIds.Contains(person.TmdbPersonID)) missingIds.Add(person.TmdbPersonID);
2046+
2047+
foreach (var person in _tmdbMovieCast.GetAll())
2048+
if (!peopleIds.Contains(person.TmdbPersonID)) missingIds.Add(person.TmdbPersonID);
2049+
foreach (var person in _tmdbMovieCrew.GetAll())
2050+
if (!peopleIds.Contains(person.TmdbPersonID)) missingIds.Add(person.TmdbPersonID);
2051+
2052+
_logger.LogDebug("Found {@Count} unique missing TMDB People for Episode & Movie staff", missingIds.Count);
2053+
2054+
foreach (var personId in missingIds)
2055+
{
2056+
var (_, updated) = await UpdatePerson(personId, forceRefresh: true);
2057+
if (updated)
2058+
updateCount++;
2059+
else
2060+
skippedCount++;
2061+
}
2062+
2063+
_logger.LogInformation("Updated missing TMDB People: Found/Updated/Skipped {@Found}/{@Updated}/{@Skipped}",
2064+
missingIds.Count, updateCount, skippedCount);
2065+
}
2066+
20342067
public async Task<(bool added, bool updated)> UpdatePerson(int personId, bool forceRefresh = false, bool downloadImages = false)
20352068
{
20362069
using (await GetLockForEntity(ForeignEntityType.Person, personId, "metadata & images", "Update").ConfigureAwait(false))
@@ -2048,6 +2081,12 @@ private void PurgeCompany(int companyId, bool removeImageFiles = true)
20482081
methods |= PersonMethods.Images;
20492082
var newlyAdded = tmdbPerson.TMDB_PersonID is 0;
20502083
var person = await UseClient(c => c.GetPersonAsync(personId, methods), $"Get person {personId}");
2084+
if (person is null)
2085+
{
2086+
_logger.LogDebug("Unable to update staff; Scheduling refresh of related links. (Person={PersonId})", personId);
2087+
await ScheduleUpdateMoviesAndShowsByPerson(personId);
2088+
return (false, false);
2089+
}
20512090
var updated = tmdbPerson.Populate(person);
20522091
if (updated)
20532092
{
@@ -2062,6 +2101,28 @@ private void PurgeCompany(int companyId, bool removeImageFiles = true)
20622101
}
20632102
}
20642103

2104+
private async Task ScheduleUpdateMoviesAndShowsByPerson(int tmdbPersonId)
2105+
{
2106+
var showIds = new HashSet<int>();
2107+
var movieIds = new HashSet<int>();
2108+
2109+
foreach (var staff in _tmdbEpisodeCast.GetByTmdbPersonID(tmdbPersonId))
2110+
showIds.Add(staff.TmdbShowID);
2111+
foreach (var staff in _tmdbEpisodeCrew.GetByTmdbPersonID(tmdbPersonId))
2112+
showIds.Add(staff.TmdbShowID);
2113+
2114+
foreach (var staff in _tmdbMovieCast.GetByTmdbPersonID(tmdbPersonId))
2115+
movieIds.Add(staff.TmdbMovieID);
2116+
foreach (var staff in _tmdbMovieCrew.GetByTmdbPersonID(tmdbPersonId))
2117+
movieIds.Add(staff.TmdbMovieID);
2118+
2119+
foreach (var showId in showIds)
2120+
await ScheduleUpdateOfShow(showId, downloadCrewAndCast: true, forceRefresh: true);
2121+
2122+
foreach (var movieId in movieIds)
2123+
await ScheduleUpdateOfMovie(movieId, downloadCrewAndCast: true, forceRefresh: true);
2124+
}
2125+
20652126
private async Task DownloadPersonImages(int personId, ProfileImages images, bool forceDownload = false)
20662127
{
20672128
var settings = _settingsProvider.GetSettings();

0 commit comments

Comments
 (0)