From 9e10bed03862a5a9076ab59fba8181f282ce7921 Mon Sep 17 00:00:00 2001 From: ReenigneArcher <42013603+ReenigneArcher@users.noreply.github.com> Date: Thu, 11 Jan 2024 21:49:35 -0500 Subject: [PATCH] feat: add themes for tv shows (#223) --- .../FixtureJellyfinServer.cs | 133 ++++++-- .../TestThemerrManager.cs | 296 +++++++++++++----- .../Api/ThemerrController.cs | 36 +-- .../Configuration/configPage.html | 10 +- .../ScheduledTasks/ThemerrTasks.cs | 4 +- Jellyfin.Plugin.Themerr/ThemerrManager.cs | 162 ++++++---- docs/Doxyfile | 2 +- docs/source/about/usage.rst | 8 +- docs/source/conf.py | 20 +- docs/source/contributing/testing.rst | 3 - 10 files changed, 472 insertions(+), 202 deletions(-) diff --git a/Jellyfin.Plugin.Themerr.Tests/FixtureJellyfinServer.cs b/Jellyfin.Plugin.Themerr.Tests/FixtureJellyfinServer.cs index 2a76c94..25ab6a6 100644 --- a/Jellyfin.Plugin.Themerr.Tests/FixtureJellyfinServer.cs +++ b/Jellyfin.Plugin.Themerr.Tests/FixtureJellyfinServer.cs @@ -1,22 +1,25 @@ +using MediaBrowser.Controller.Entities; +using MediaBrowser.Controller.Entities.Audio; +using MediaBrowser.Controller.Entities.Movies; +using MediaBrowser.Controller.Entities.TV; using MetadataProvider = MediaBrowser.Model.Entities.MetadataProvider; -using Movie = MediaBrowser.Controller.Entities.Movies.Movie; namespace Jellyfin.Plugin.Themerr.Tests; /// -/// This class is used as a fixture for the Jellyfin server with mock movies +/// This class is used as a fixture for the Jellyfin server with mock media items (movies and series). /// public class FixtureJellyfinServer { /// - /// Mock movies to use for testing + /// Mock media items to use for testing /// - /// List containing mock objects. - public static List MockMovies() + /// List containing mock objects. + public static List MockItems() { - return new List + return new List { - new() + new Movie { Name = "Elephants Dream", ProductionYear = 2006, @@ -26,7 +29,7 @@ public static List MockMovies() { MetadataProvider.Tmdb.ToString(), "9761"}, } }, - new() + new Movie { Name = "Sita Sings the Blues", ProductionYear = 2008, @@ -36,7 +39,7 @@ public static List MockMovies() { MetadataProvider.Tmdb.ToString(), "20529"}, } }, - new() + new Movie { Name = "Big Buck Bunny", ProductionYear = 2008, @@ -46,7 +49,7 @@ public static List MockMovies() { MetadataProvider.Tmdb.ToString(), "10378"}, } }, - new() + new Movie { Name = "Sintel", ProductionYear = 2010, @@ -56,18 +59,38 @@ public static List MockMovies() { MetadataProvider.Tmdb.ToString(), "45745"}, } }, + new Series + { + Name = "Game of Thrones", + ProductionYear = 2011, + ProviderIds = new Dictionary + { + { MetadataProvider.Imdb.ToString(), "tt0944947"}, + { MetadataProvider.Tmdb.ToString(), "1399"}, + } + }, + new Series + { + Name = "The 100", + ProductionYear = 2014, + ProviderIds = new Dictionary + { + { MetadataProvider.Imdb.ToString(), "tt2661044"}, + { MetadataProvider.Tmdb.ToString(), "48866"}, + } + } }; } /// - /// Mock movies without an associated theme in ThemerrDB to use for testing + /// Mock items without an associated theme in ThemerrDB to use for testing /// - /// List containing mock objects. - public static List MockMovies2() + /// List containing mock objects. + public static List MockItems2() { - return new List + return new List { - new() + new Movie { Name = "Themerr Test Movie", ProductionYear = 1970, @@ -77,17 +100,46 @@ public static List MockMovies2() { MetadataProvider.Tmdb.ToString(), "0"}, } }, + new Series + { + Name = "Themerr Test Show", + ProductionYear = 1970, + ProviderIds = new Dictionary + { + { MetadataProvider.Imdb.ToString(), "tt0000000"}, + { MetadataProvider.Tmdb.ToString(), "0"}, + } + }, }; } /// - /// Create mock movies from stub video + /// Mock items which are not supported by Themerr + /// + /// List containing mock objects. + public static List UnsupportedMockItems() + { + return new List + { + new MusicAlbum + { + Name = "Themerr Test Album", + }, + new MusicArtist + { + Name = "Themerr Test Artist", + }, + }; + } + + /// + /// Create mock items from stub video /// [Fact] [Trait("Category", "Init")] - private void CreateMockMovies() + private void CreateMockItems() { - var mockMovies = MockMovies(); + var mockItems = MockItems(); // get the stub video path based on the directory of this file var stubVideoPath = Path.Combine( @@ -97,29 +149,50 @@ private void CreateMockMovies() Assert.True(File.Exists(stubVideoPath), "Could not find ./data/video_stub.mp4"); - foreach (var movie in mockMovies) + foreach (var item in mockItems) { // copy the ./data/video_stub.mp4 to the movie folder "movie.Name (movie.ProductionYear)" - var movieFolder = Path.Combine( + var itemFolder = Path.Combine( "themerr_jellyfin_tests", - $"{movie.Name} ({movie.ProductionYear})"); + $"{item.Name} ({item.ProductionYear})"); // create the movie folder - Directory.CreateDirectory(movieFolder); + Directory.CreateDirectory(itemFolder); - // copy the video_stub.mp4 to the movie folder, renaming it based on movie name - var movieVideoPath = Path.Combine( - movieFolder, - $"{movie.Name} ({movie.ProductionYear}).mp4"); + string? itemVideoPath = null; + if (item is Movie) + { + // copy the video_stub.mp4 to the movie folder, renaming it based on movie name + itemVideoPath = Path.Combine( + itemFolder, + $"{item.Name} ({item.ProductionYear}).mp4"); + } + else if (item is Series) + { + // season folder + var seasonFolder = Path.Combine( + itemFolder, + "Season 01"); + Directory.CreateDirectory(seasonFolder); + + // copy the video_stub.mp4 to the season folder, renaming it based on series name + itemVideoPath = Path.Combine( + seasonFolder, + $"{item.Name} ({item.ProductionYear}) - S01E01 - Episode Name.mp4"); + } + else + { + Assert.Fail($"Unknown item type: {item.GetType()}"); + } // if file does not exist - if (!File.Exists(movieVideoPath)) + if (!File.Exists(itemVideoPath)) { - // copy the stub video to the movie folder - File.Copy(stubVideoPath, movieVideoPath); + // copy the stub video to the item folder + File.Copy(stubVideoPath, itemVideoPath); } - Assert.True(File.Exists(movieVideoPath), $"Could not find {movieVideoPath}"); + Assert.True(File.Exists(itemVideoPath), $"Could not find {itemVideoPath}"); } } } diff --git a/Jellyfin.Plugin.Themerr.Tests/TestThemerrManager.cs b/Jellyfin.Plugin.Themerr.Tests/TestThemerrManager.cs index 7dd351d..9285f31 100644 --- a/Jellyfin.Plugin.Themerr.Tests/TestThemerrManager.cs +++ b/Jellyfin.Plugin.Themerr.Tests/TestThemerrManager.cs @@ -1,3 +1,5 @@ +using MediaBrowser.Controller.Entities.Movies; +using MediaBrowser.Controller.Entities.TV; using MediaBrowser.Controller.Library; using Microsoft.Extensions.Logging; using Moq; @@ -183,52 +185,70 @@ private void TestSaveMp3InvalidUrl() [Fact] [Trait("Category", "Unit")] - private void TestGetMoviesFromLibrary() + private void TestGetTmdbItemsFromLibrary() { - var movies = _themerrManager.GetMoviesFromLibrary(); + var items = _themerrManager.GetTmdbItemsFromLibrary(); - // movies list should be empty - Assert.Empty(movies); + // items list should be empty + Assert.Empty(items); - // todo: test with actual movies + // todo: test with actual items } - // todo: fix this test - // [Fact] - // [Trait("Category", "Unit")] - // private void TestProcessMovieTheme() - // { - // // get fixture movies - // var mockMovies = FixtureJellyfinServer.MockMovies(); - // - // Assert.True(mockMovies.Count > 0, "mockMovies.Count is not greater than 0"); - // - // foreach (var movie in mockMovies) - // { - // // get the movie theme - // _themerrManager.ProcessMovieTheme(movie); - // - // Assert.True(File.Exists(_themerrManager.GetThemePath(movie)), $"File {_themerrManager.GetThemePath(movie)} does not exist"); - // } - // } + [Fact] + [Trait("Category", "Unit")] + private void TestProcessItemTheme() + { + // get fixture movies + var mockItems = FixtureJellyfinServer.MockItems(); + + Assert.True(mockItems.Count > 0, "mockItems.Count is not greater than 0"); + + foreach (var item in mockItems) + { + // get the item theme + _themerrManager.ProcessItemTheme(item); + + Assert.True(File.Exists(_themerrManager.GetThemePath(item)), $"File {_themerrManager.GetThemePath(item)} does not exist"); + + // cleanup and delete the file + File.Delete(_themerrManager.GetThemePath(item)); + } + } + + [Fact] + [Trait("Category", "Unit")] + private void TestProcessItemThemeUnsupportedType() + { + // get fixture items + var mockItems = FixtureJellyfinServer.UnsupportedMockItems(); + + foreach (var item in mockItems) + { + // get the item theme + _themerrManager.ProcessItemTheme(item); + + Assert.False(File.Exists(_themerrManager.GetThemePath(item)), $"File {_themerrManager.GetThemePath(item)} exists"); + } + } [Fact] [Trait("Category", "Unit")] private void TestGetTmdbId() { - // get fixture movies - var mockMovies = FixtureJellyfinServer.MockMovies(); + // get fixture items + var mockItems = FixtureJellyfinServer.MockItems(); - foreach (var movie in mockMovies) + foreach (var item in mockItems) { - // get the movie theme - var tmdbId = _themerrManager.GetTmdbId(movie); + // get the item theme + var tmdbId = _themerrManager.GetTmdbId(item); // ensure tmdbId is not empty Assert.NotEmpty(tmdbId); - // ensure tmdbId is the same as the one in the movie fixture - Assert.Equal(movie.ProviderIds[MetadataProvider.Tmdb.ToString()], tmdbId); + // ensure tmdbId is the same as the one in the item fixture + Assert.Equal(item.ProviderIds[MetadataProvider.Tmdb.ToString()], tmdbId); } } @@ -237,19 +257,19 @@ private void TestGetTmdbId() // [Trait("Category", "Unit")] // private void TestGetThemeProvider() // { - // // get fixture movies - // var mockMovies = FixtureJellyfinServer.MockMovies(); + // // get fixture items + // var mockItems = FixtureJellyfinServer.MockItems(); // - // foreach (var movie in mockMovies) + // foreach (var item in mockItems) // { - // // get the movie theme - // var themeProvider = _themerrManager.GetThemeProvider(movie); + // // get the item theme + // var themeProvider = _themerrManager.GetThemeProvider(item); // // // ensure themeProvider null // Assert.Null(themeProvider); // } // - // // todo: test with actual movies + // // todo: test with actual items // } [Fact] @@ -334,53 +354,104 @@ private void TestContinueDownload() [Trait("Category", "Unit")] private void TestGetThemePath() { - // get fixture movies - var mockMovies = FixtureJellyfinServer.MockMovies(); + // get fixture items + var mockItems = FixtureJellyfinServer.MockItems(); - Assert.True(mockMovies.Count > 0, "mockMovies.Count is not greater than 0"); + Assert.True(mockItems.Count > 0, "mockItems.Count is not greater than 0"); - foreach (var movie in mockMovies) + foreach (var item in mockItems) { - // get the movie theme - var themePath = _themerrManager.GetThemePath(movie); + // get the item theme + var themePath = _themerrManager.GetThemePath(item); // ensure path ends with theme.mp3 Assert.EndsWith("theme.mp3", themePath); } } + [Fact] + [Trait("Category", "Unit")] + private void TestGetThemePathUnsupportedType() + { + // get fixture items + var mockItems = FixtureJellyfinServer.UnsupportedMockItems(); + + Assert.True(mockItems.Count > 0, "mockItems.Count is not greater than 0"); + + foreach (var item in mockItems) + { + // get the item theme + var themePath = _themerrManager.GetThemePath(item); + + // ensure path is null + Assert.Null(themePath); + } + } + [Fact] [Trait("Category", "Unit")] private void TestGetThemerrDataPath() { - // get fixture movies - var mockMovies = FixtureJellyfinServer.MockMovies(); + // get fixture items + var mockItems = FixtureJellyfinServer.MockItems(); - Assert.True(mockMovies.Count > 0, "mockMovies.Count is not greater than 0"); + Assert.True(mockItems.Count > 0, "mockItems.Count is not greater than 0"); - foreach (var movie in mockMovies) + foreach (var item in mockItems) { - // get the movie theme - var themerrDataPath = _themerrManager.GetThemerrDataPath(movie); + // get the item theme + var themerrDataPath = _themerrManager.GetThemerrDataPath(item); // ensure path ends with theme.mp3 Assert.EndsWith("themerr.json", themerrDataPath); } } + [Fact] + [Trait("Category", "Unit")] + private void TestGetThemerrDataPathUnsupportedType() + { + // get fixture items + var mockItems = FixtureJellyfinServer.UnsupportedMockItems(); + + Assert.True(mockItems.Count > 0, "mockItems.Count is not greater than 0"); + + foreach (var item in mockItems) + { + // get the item theme + var themerrDataPath = _themerrManager.GetThemerrDataPath(item); + + // ensure path is null + Assert.Null(themerrDataPath); + } + } + [Fact] [Trait("Category", "Unit")] private void TestCreateThemerrDbLink() { - // get fixture movies - var mockMovies = FixtureJellyfinServer.MockMovies(); + // get fixture items + var mockItems = FixtureJellyfinServer.MockItems(); - Assert.True(mockMovies.Count > 0, "mockMovies.Count is not greater than 0"); + Assert.True(mockItems.Count > 0, "mockItems.Count is not greater than 0"); - foreach (var movie in mockMovies) + foreach (var item in mockItems) { - var tmdbId = movie.ProviderIds[MetadataProvider.Tmdb.ToString()]; - var themerrDbUrl = _themerrManager.CreateThemerrDbLink(tmdbId); + var dbType = item switch + { + Movie _ => "movies", + Series _ => "tv_shows", + _ => null + }; + + // return if dbType is null + if (string.IsNullOrEmpty(dbType)) + { + Assert.Fail($"Unknown item type: {item.GetType()}"); + } + + var tmdbId = item.ProviderIds[MetadataProvider.Tmdb.ToString()]; + var themerrDbUrl = _themerrManager.CreateThemerrDbLink(tmdbId, dbType); TestLogger.Info($"themerrDbLink: {themerrDbUrl}"); @@ -392,20 +463,33 @@ private void TestCreateThemerrDbLink() [Trait("Category", "Unit")] private void TestGetYoutubeThemeUrl() { - // get fixture movies - var mockMovies = FixtureJellyfinServer.MockMovies(); + // get fixture items + var mockItems = FixtureJellyfinServer.MockItems(); - Assert.True(mockMovies.Count > 0, "mockMovies.Count is not greater than 0"); + Assert.True(mockItems.Count > 0, "mockItems.Count is not greater than 0"); - // loop over each movie - foreach (var movie in mockMovies) + // loop over each item + foreach (var item in mockItems) { + var dbType = item switch + { + Movie _ => "movies", + Series _ => "tv_shows", + _ => null + }; + + // return if dbType is null + if (string.IsNullOrEmpty(dbType)) + { + Assert.Fail($"Unknown item type: {item.GetType()}"); + } + // get themerrDbUrl - var tmdbId = _themerrManager.GetTmdbId(movie); - var themerrDbLink = _themerrManager.CreateThemerrDbLink(tmdbId); + var tmdbId = _themerrManager.GetTmdbId(item); + var themerrDbLink = _themerrManager.CreateThemerrDbLink(tmdbId, dbType); // get the new youtube theme url - var youtubeThemeUrl = _themerrManager.GetYoutubeThemeUrl(themerrDbLink, movie); + var youtubeThemeUrl = _themerrManager.GetYoutubeThemeUrl(themerrDbLink, item); // log TestLogger.Info($"youtubeThemeUrl: {youtubeThemeUrl}"); @@ -418,20 +502,33 @@ private void TestGetYoutubeThemeUrl() [Trait("Category", "Unit")] private void TestGetYoutubeThemeUrlExceptions() { - // get fixture movies - var mockMovies = FixtureJellyfinServer.MockMovies2(); + // get fixture items + var mockItems = FixtureJellyfinServer.MockItems2(); - Assert.True(mockMovies.Count > 0, "mockMovies.Count is not greater than 0"); + Assert.True(mockItems.Count > 0, "mockItems.Count is not greater than 0"); - // loop over each movie - foreach (var movie in mockMovies) + // loop over each item + foreach (var item in mockItems) { + var dbType = item switch + { + Movie _ => "movies", + Series _ => "tv_shows", + _ => null + }; + + // return if dbType is null + if (string.IsNullOrEmpty(dbType)) + { + Assert.Fail($"Unknown item type: {item.GetType()}"); + } + // get themerrDbUrl - var tmdbId = _themerrManager.GetTmdbId(movie); - var themerrDbLink = _themerrManager.CreateThemerrDbLink(tmdbId); + var tmdbId = _themerrManager.GetTmdbId(item); + var themerrDbLink = _themerrManager.CreateThemerrDbLink(tmdbId, dbType); // get the new youtube theme url - var youtubeThemeUrl = _themerrManager.GetYoutubeThemeUrl(themerrDbLink, movie); + var youtubeThemeUrl = _themerrManager.GetYoutubeThemeUrl(themerrDbLink, item); Assert.Empty(youtubeThemeUrl); } @@ -441,22 +538,42 @@ private void TestGetYoutubeThemeUrlExceptions() [Trait("Category", "Unit")] private void TestGetIssueUrl() { - // get fixture movies - var mockMovies = FixtureJellyfinServer.MockMovies(); + // get fixture items + var mockItems = FixtureJellyfinServer.MockItems(); - Assert.True(mockMovies.Count > 0, "mockMovies.Count is not greater than 0"); + Assert.True(mockItems.Count > 0, "mockItems.Count is not greater than 0"); - // loop over each movie - foreach (var movie in mockMovies) + // loop over each item + foreach (var item in mockItems) { + var issueType = item switch + { + Movie _ => "MOVIE", + Series _ => "TV SHOW", + _ => null + }; + + var tmdbEndpoint = item switch + { + Movie _ => "movie", + Series _ => "tv", + _ => null + }; + + // return if dbType is null + if (string.IsNullOrEmpty(issueType) || string.IsNullOrEmpty(tmdbEndpoint)) + { + Assert.Fail($"Unknown item type: {item.GetType()}"); + } + // parts of expected url - var tmdbId = _themerrManager.GetTmdbId(movie); - var encodedName = movie.Name.Replace(" ", "%20"); - var year = movie.ProductionYear; - var expectedUrl = $"https://github.com/LizardByte/ThemerrDB/issues/new?assignees=&labels=request-theme&template=theme.yml&title=[MOVIE]:%20{encodedName}%20({year})&database_url=https://www.themoviedb.org/movie/{tmdbId}"; + var tmdbId = _themerrManager.GetTmdbId(item); + var encodedName = item.Name.Replace(" ", "%20"); + var year = item.ProductionYear; + var expectedUrl = $"https://github.com/LizardByte/ThemerrDB/issues/new?assignees=&labels=request-theme&template=theme.yml&title=[{issueType}]:%20{encodedName}%20({year})&database_url=https://www.themoviedb.org/{tmdbEndpoint}/{tmdbId}"; // get the new youtube theme url - var issueUrl = _themerrManager.GetIssueUrl(movie); + var issueUrl = _themerrManager.GetIssueUrl(item); Assert.NotEmpty(issueUrl); @@ -468,6 +585,25 @@ private void TestGetIssueUrl() } } + [Fact] + [Trait("Category", "Unit")] + private void TestGetIssueUrlUnsupportedType() + { + // get fixture items + var mockItems = FixtureJellyfinServer.UnsupportedMockItems(); + + Assert.True(mockItems.Count > 0, "mockItems.Count is not greater than 0"); + + // loop over each item + foreach (var item in mockItems) + { + // get the new youtube theme url + var issueUrl = _themerrManager.GetIssueUrl(item); + + Assert.Null(issueUrl); + } + } + [Fact] [Trait("Category", "Unit")] private void TestSaveThemerrData() diff --git a/Jellyfin.Plugin.Themerr/Api/ThemerrController.cs b/Jellyfin.Plugin.Themerr/Api/ThemerrController.cs index 64eace8..a7b21b9 100644 --- a/Jellyfin.Plugin.Themerr/Api/ThemerrController.cs +++ b/Jellyfin.Plugin.Themerr/Api/ThemerrController.cs @@ -3,7 +3,6 @@ using System.Linq; using System.Net.Mime; using System.Threading.Tasks; -using MediaBrowser.Controller.Entities.Movies; using MediaBrowser.Controller.Library; using Microsoft.AspNetCore.Authorization; using Microsoft.AspNetCore.Http; @@ -49,7 +48,7 @@ public ThemerrController( [ProducesResponseType(StatusCodes.Status204NoContent)] public async Task TriggerUpdateRequest() { - _logger.LogInformation("Updating Movie Theme Songs"); + _logger.LogInformation("Updating Theme Songs"); await _themerrManager.UpdateAll(); _logger.LogInformation("Completed"); } @@ -57,11 +56,11 @@ public async Task TriggerUpdateRequest() /// /// Get the data required to populate the progress dashboard. /// - /// Loop over all Jellyfin libraries and movies, creating a json object with the following structure: + /// Loop over all Jellyfin libraries and supported items, creating a json object with the following structure: /// { - /// "items": [Movies], - /// "media_count": Movies.Count, - /// "media_percent_complete": ThemedMovies.Count / Movies.Count * 100, + /// "items": [BaseItems], + /// "media_count": BaseItems.Count, + /// "media_percent_complete": ThemedItems.Count / BaseItems.Count * 100, /// } /// /// JSON object containing progress data. @@ -75,29 +74,30 @@ public ActionResult GetProgress() var mediaWithThemes = 0; var mediaPercentComplete = 0; - var movies = _themerrManager.GetMoviesFromLibrary(); + var items = _themerrManager.GetTmdbItemsFromLibrary(); - // sort movies by name, then year - var enumerable = movies.OrderBy(m => m.Name).ThenBy(m => m.ProductionYear); + // sort items by name, then year + var enumerable = items.OrderBy(i => i.Name).ThenBy(i => i.ProductionYear); - foreach (var movie in enumerable) + foreach (var item in enumerable) { - var year = movie.ProductionYear; - var issueUrl = _themerrManager.GetIssueUrl(movie); - var themeProvider = _themerrManager.GetThemeProvider(movie); - var item = new + var year = item.ProductionYear; + var issueUrl = _themerrManager.GetIssueUrl(item); + var themeProvider = _themerrManager.GetThemeProvider(item); + var tmpItem = new { - name = movie.Name, - id = movie.Id, + name = item.Name, + id = item.Id, issue_url = issueUrl, theme_provider = themeProvider, + type = item.GetType().Name, // Movie, Series, etc. year = year }; - tmpItems.Add(item); + tmpItems.Add(tmpItem); mediaCount++; - var themeSongs = movie.GetThemeSongs(); + var themeSongs = item.GetThemeSongs(); if (themeSongs.Count > 0) { mediaWithThemes++; diff --git a/Jellyfin.Plugin.Themerr/Configuration/configPage.html b/Jellyfin.Plugin.Themerr/Configuration/configPage.html index 3ca35a0..8907203 100644 --- a/Jellyfin.Plugin.Themerr/Configuration/configPage.html +++ b/Jellyfin.Plugin.Themerr/Configuration/configPage.html @@ -64,7 +64,7 @@

Themerr



- +
@@ -204,7 +204,7 @@

Themerr

let tableHeaderRow = document.createElement('tr') tableHeader.appendChild(tableHeaderRow) - let columns = ['Title', 'Year', 'Contribute', 'Status'] + let columns = ['Title', 'Year', 'Type', 'Contribute', 'Status'] // create table header columns for (let column in columns) { @@ -216,10 +216,12 @@

Themerr

// loop over items for (let item in response["items"]) { + console.log(LogPrefix + "------------------") console.log(LogPrefix + "item: " + item) console.log(LogPrefix + "name: " + response["items"][item]["name"]) console.log(LogPrefix + "id: " + response["items"][item]["id"]) console.log(LogPrefix + "issue_url: " + response["items"][item]["issue_url"]) + console.log(LogPrefix + "type: " + response["items"][item]["type"]) console.log(LogPrefix + "theme_provider: " + response["items"][item]["theme_provider"]) console.log(LogPrefix + "year: " + response["items"][item]["year"]) @@ -235,6 +237,10 @@

Themerr

let tableColumnYear = document.createElement('td') tableColumnYear.innerHTML = response["items"][item]["year"] tableRow.appendChild(tableColumnYear) + + let tableColumnType = document.createElement('td') + tableColumnType.innerHTML = response["items"][item]["type"] + tableRow.appendChild(tableColumnType) let contributeButton = document.createElement('a') contributeButton.setAttribute('is', 'emby-linkbutton') diff --git a/Jellyfin.Plugin.Themerr/ScheduledTasks/ThemerrTasks.cs b/Jellyfin.Plugin.Themerr/ScheduledTasks/ThemerrTasks.cs index c51946d..2f267e7 100644 --- a/Jellyfin.Plugin.Themerr/ScheduledTasks/ThemerrTasks.cs +++ b/Jellyfin.Plugin.Themerr/ScheduledTasks/ThemerrTasks.cs @@ -40,7 +40,7 @@ public ThemerrTasks(ILibraryManager libraryManager, ILogger logg /// /// Gets the description of the task. /// - public string Description => "Scans all libraries to download Movie Theme Songs"; + public string Description => "Scans all libraries to download supported Theme Songs"; /// /// Gets the category of the task. @@ -55,7 +55,7 @@ public ThemerrTasks(ILibraryManager libraryManager, ILogger logg /// A representing the asynchronous operation. public async Task ExecuteAsync(IProgress progress, CancellationToken cancellationToken) { - _logger.LogInformation("Starting plugin, Downloading Movie Theme Songs..."); + _logger.LogInformation("Starting plugin, Downloading supported Theme Songs..."); await _themerrManager.UpdateAll(); _logger.LogInformation("All theme songs downloaded"); } diff --git a/Jellyfin.Plugin.Themerr/ThemerrManager.cs b/Jellyfin.Plugin.Themerr/ThemerrManager.cs index 681dbe6..3d8d6bc 100644 --- a/Jellyfin.Plugin.Themerr/ThemerrManager.cs +++ b/Jellyfin.Plugin.Themerr/ThemerrManager.cs @@ -7,6 +7,7 @@ using Jellyfin.Data.Enums; using MediaBrowser.Controller.Entities; using MediaBrowser.Controller.Entities.Movies; +using MediaBrowser.Controller.Entities.TV; using MediaBrowser.Controller.Library; using MediaBrowser.Controller.Plugins; using MediaBrowser.Model.Entities; @@ -15,9 +16,6 @@ using YoutubeExplode; using YoutubeExplode.Videos.Streams; -// TODO: Add support for TV shows -// using MediaBrowser.Controller.Entities.TV; - namespace Jellyfin.Plugin.Themerr { /// @@ -94,59 +92,71 @@ public bool SaveMp3(string destination, string videoUrl) } /// - /// Get all movies from the library that have a tmdb id. + /// Get all supported items from the library that have a tmdb id. /// - /// List of . - public IEnumerable GetMoviesFromLibrary() + /// List of objects. + public IEnumerable GetTmdbItemsFromLibrary() { - var movies = _libraryManager.GetItemList(new InternalItemsQuery + var items = _libraryManager.GetItemList(new InternalItemsQuery { - IncludeItemTypes = new[] {BaseItemKind.Movie}, + IncludeItemTypes = new[] + { + BaseItemKind.Movie, + BaseItemKind.Series + }, IsVirtualItem = false, Recursive = true, HasTmdbId = true }); - var movieList = new List(); - if (movies == null || movies.Count == 0) + var itemList = new List(); + if (items == null || items.Count == 0) { - return movieList; + return itemList; } - foreach (var movie in movies) - { - if (movie is Movie m) - { - movieList.Add(m); - } - } + itemList.AddRange(items.Where(item => item is Movie or Series)); - return movieList; + return itemList; } /// - /// Enumerate through all movies in the library and downloads their theme songs as required. + /// Enumerate through all supported items in the library and downloads their theme songs as required. /// /// A representing the asynchronous operation. public Task UpdateAll() { - var movies = GetMoviesFromLibrary(); - foreach (var movie in movies) + var items = GetTmdbItemsFromLibrary(); + foreach (var item in items) { - ProcessMovieTheme(movie); + ProcessItemTheme(item); } return Task.CompletedTask; } /// - /// Download the theme song for a movie if it doesn't already exist. + /// Download the theme song for a media item if it doesn't already exist. /// - /// The Jellyfin movie object. - public void ProcessMovieTheme(Movie movie) + /// The Jellyfin media object. + public void ProcessItemTheme(BaseItem item) { - var themePath = GetThemePath(movie); - var themerrDataPath = GetThemerrDataPath(movie); + // get themerrDB database type, used to create the themerrdb url + var dbType = item switch + { + Movie _ => "movies", + Series _ => "tv_shows", + _ => null + }; + + // return if dbType is null + if (string.IsNullOrEmpty(dbType)) + { + return; + } + + var themePath = GetThemePath(item); + var themerrDataPath = GetThemerrDataPath(item); if (!ContinueDownload(themePath, themerrDataPath)) { @@ -156,12 +166,12 @@ public void ProcessMovieTheme(Movie movie) var existingYoutubeThemeUrl = GetExistingThemerrDataValue("youtube_theme_url", themerrDataPath); // get tmdb id - var tmdbId = GetTmdbId(movie); + var tmdbId = GetTmdbId(item); // create themerrdb url - var themerrDbUrl = CreateThemerrDbLink(tmdbId); + var themerrDbUrl = CreateThemerrDbLink(tmdbId, dbType); - var youtubeThemeUrl = GetYoutubeThemeUrl(themerrDbUrl, movie); + var youtubeThemeUrl = GetYoutubeThemeUrl(themerrDbUrl, item); // skip if no youtube theme url in ThemerrDB or // if the youtube themes match AND the theme_md5 is unknown @@ -184,35 +194,35 @@ public void ProcessMovieTheme(Movie movie) return; } - movie.RefreshMetadata(CancellationToken.None); + item.RefreshMetadata(CancellationToken.None); } /// - /// Get TMDB id from a movie. + /// Get TMDB id from an item. /// - /// The Jellyfin movie object. + /// The Jellyfin media object. /// TMDB id. - public string GetTmdbId(Movie movie) + public string GetTmdbId(BaseItem item) { - var tmdbId = movie.GetProviderId(MetadataProvider.Tmdb); + var tmdbId = item.GetProviderId(MetadataProvider.Tmdb); return tmdbId; } /// /// Get the theme provider. /// - /// The Jellyfin movie object. + /// The Jellyfin media object. /// The theme provider. - public string GetThemeProvider(Movie movie) + public string GetThemeProvider(BaseItem item) { // check if item has a theme song - var themeSongs = movie.GetThemeSongs(); + var themeSongs = item.GetThemeSongs(); if (themeSongs == null || themeSongs.Count == 0) { return null; } - var themerrDataPath = GetThemerrDataPath(movie); + var themerrDataPath = GetThemerrDataPath(item); var themerrHash = GetExistingThemerrDataValue("theme_md5", themerrDataPath); var themeHash = GetMd5Hash(themeSongs[0].Path); @@ -267,40 +277,51 @@ public bool ContinueDownload(string themePath, string themerrDataPath) /// /// Get the path to the theme song. /// - /// The Jellyfin movie object. + /// The Jellyfin media object. /// The path to the theme song. - public string GetThemePath(Movie movie) + public string GetThemePath(BaseItem item) { - return $"{movie.ContainingFolderPath}/theme.mp3"; + return item switch + { + Movie movie => System.IO.Path.Join(movie.ContainingFolderPath, "theme.mp3"), + Series series => System.IO.Path.Join(series.Path, "theme.mp3"), + _ => null + }; } /// /// Get the path to the themerr data file. /// - /// The Jellyfin movie object. + /// The Jellyfin media object. /// The path to the themerr data file. - public string GetThemerrDataPath(Movie movie) + public string GetThemerrDataPath(BaseItem item) { - return $"{movie.ContainingFolderPath}/themerr.json"; + return item switch + { + Movie movie => System.IO.Path.Join(movie.ContainingFolderPath, "themerr.json"), + Series series => System.IO.Path.Join(series.Path, "themerr.json"), + _ => null + }; } /// /// Create a link to the themerr database. /// /// The tmdb id. + /// The database type. /// The themerr database link. - public string CreateThemerrDbLink(string tmdbId) + public string CreateThemerrDbLink(string tmdbId, string dbType) { - return $"https://app.lizardbyte.dev/ThemerrDB/movies/themoviedb/{tmdbId}.json"; + return $"https://app.lizardbyte.dev/ThemerrDB/{dbType}/themoviedb/{tmdbId}.json"; } /// /// Get the YouTube theme url from the themerr database. /// /// The themerr database url. - /// The Jellyfin movie object. + /// The Jellyfin media object. /// The YouTube theme url. - public string GetYoutubeThemeUrl(string themerrDbUrl, Movie movie) + public string GetYoutubeThemeUrl(string themerrDbUrl, BaseItem item) { var client = new HttpClient(); @@ -313,9 +334,9 @@ public string GetYoutubeThemeUrl(string themerrDbUrl, Movie movie) catch (Exception) { _logger.LogWarning( - "Missing from ThemerrDB: {MovieTitle}, contribute:\n {IssueUrl}", - movie.Name, - GetIssueUrl(movie)); + "Missing from ThemerrDB: {ItemTitle}, contribute:\n {IssueUrl}", + item.Name, + GetIssueUrl(item)); return string.Empty; } } @@ -325,18 +346,37 @@ public string GetYoutubeThemeUrl(string themerrDbUrl, Movie movie) /// /// This url can be used to easily add/edit theme songs in ThemerrDB. /// - /// The Jellyfin movie object. + /// The Jellyfin media object. /// The ThemerrDB issue url. - public string GetIssueUrl(Movie movie) + public string GetIssueUrl(BaseItem item) { + string issueBaseTitle = item switch + { + Movie _ => "MOVIE", + Series _ => "TV SHOW", + _ => null + }; + + string tmdbEndpoint = item switch + { + Movie _ => "movie", + Series _ => "tv", + _ => null + }; + + // return if either is null + if (string.IsNullOrEmpty(issueBaseTitle) || string.IsNullOrEmpty(tmdbEndpoint)) + { + return null; + } + // url components - const string issueBase = - "https://github.com/LizardByte/ThemerrDB/issues/new?assignees=&labels=request-theme&template=theme.yml&title=[MOVIE]:%20"; - const string databaseBase = "https://www.themoviedb.org/movie/"; + string issueBase = $"https://github.com/LizardByte/ThemerrDB/issues/new?assignees=&labels=request-theme&template=theme.yml&title=[{issueBaseTitle}]:%20"; + string databaseBase = $"https://www.themoviedb.org/{tmdbEndpoint}/"; - var urlEncodedName = movie.Name.Replace(" ", "%20"); - var year = movie.ProductionYear; - var tmdbId = GetTmdbId(movie); + var urlEncodedName = item.Name.Replace(" ", "%20"); + var year = item.ProductionYear; + var tmdbId = GetTmdbId(item); var issueUrl = $"{issueBase}{urlEncodedName}%20({year})&database_url={databaseBase}{tmdbId}"; return issueUrl; diff --git a/docs/Doxyfile b/docs/Doxyfile index 6d1d8c4..d8b0a4e 100644 --- a/docs/Doxyfile +++ b/docs/Doxyfile @@ -54,7 +54,7 @@ PROJECT_NUMBER = # for a project that appears at the top of each page and should give viewer a # quick idea about the purpose of the project. Keep the description short. -PROJECT_BRIEF = "Plugin for Jellyfin that adds theme songs to movies using ThemerrDB." +PROJECT_BRIEF = "Plugin for Jellyfin that adds theme songs to movies and tv shows using ThemerrDB." # With the PROJECT_LOGO tag one can specify a logo or an icon that is included # in the documentation. The maximum height of the logo should not exceed 55 diff --git a/docs/source/about/usage.rst b/docs/source/about/usage.rst index db587fd..8758746 100644 --- a/docs/source/about/usage.rst +++ b/docs/source/about/usage.rst @@ -11,10 +11,12 @@ Enable Themes #. Select `Display` from the user section. #. Within the `Library` section, ensure `Theme songs` is enabled. -Movie Directory Structure -------------------------- +Directory Structure +------------------- -.. Attention:: Jellyfin requires movies to be stored in separate subdirectories, with each movie in its own folder. +.. Attention:: Jellyfin requires your media to be stored in separate subdirectories, with each movie/show in its + own folder. See `Movies `__ + or `TV Shows `__ for more information. Task Activation --------------- diff --git a/docs/source/conf.py b/docs/source/conf.py index ef7a53f..df97880 100644 --- a/docs/source/conf.py +++ b/docs/source/conf.py @@ -118,9 +118,15 @@ 'Jellyfin.Controller.MediaBrowser.Model.Tasks': ( 'https://github.com/jellyfin/jellyfin/blob/v10.8.13/MediaBrowser.Model/Tasks/%s.cs', ), + 'Jellyfin.Controller.MediaBrowser.Controller.Entities': ( + 'https://github.com/jellyfin/jellyfin/blob/v10.8.13/MediaBrowser.Controller/Entities/%s.cs', + ), 'Jellyfin.Controller.MediaBrowser.Controller.Entities.Movies': ( 'https://github.com/jellyfin/jellyfin/blob/v10.8.13/MediaBrowser.Controller/Entities/Movies/%s.cs', ), + 'Jellyfin.Controller.MediaBrowser.Controller.Entities.TV': ( + 'https://github.com/jellyfin/jellyfin/blob/v10.8.13/MediaBrowser.Controller/Entities/TV/%s.cs', + ), 'Jellyfin.Controller.MediaBrowser.Controller.Entities.Library': ( 'https://github.com/jellyfin/jellyfin/blob/v10.8.13/MediaBrowser.Controller/Library/%s.cs', ), @@ -209,15 +215,25 @@ 'TaskTriggerInfo', ], }, + 'Jellyfin.Controller.MediaBrowser.Controller.Entities': { + '': [ + 'BaseItem', + ], + }, + 'Jellyfin.Controller.MediaBrowser.Controller.Entities.Library': { + '': [ + 'ILibraryManager', + ], + }, 'Jellyfin.Controller.MediaBrowser.Controller.Entities.Movies': { '': [ 'BoxSet', 'Movie', ], }, - 'Jellyfin.Controller.MediaBrowser.Controller.Entities.Library': { + 'Jellyfin.Controller.MediaBrowser.Controller.Entities.TV': { '': [ - 'ILibraryManager', + 'Series', ], }, 'Jellyfin.Controller.MediaBrowser.Controller.Plugins': { diff --git a/docs/source/contributing/testing.rst b/docs/source/contributing/testing.rst index be00be6..6ba52ad 100644 --- a/docs/source/contributing/testing.rst +++ b/docs/source/contributing/testing.rst @@ -21,9 +21,6 @@ Themerr-jellyfin uses `Sphinx `__ for doc other required python dependencies are included in the `./docs/requirements.txt` file. Python is required to build sphinx docs. Installation and setup of python will not be covered here. -.. todo:: - Add documentation within C# code to be included in sphinx docs. - The config file for Sphinx is ``docs/source/conf.py``. This is already included in the root of the repo and should not be modified.