Skip to content
This repository has been archived by the owner on Oct 13, 2024. It is now read-only.

Commit

Permalink
feat: track theme file md5 hash
Browse files Browse the repository at this point in the history
  • Loading branch information
ReenigneArcher committed Dec 16, 2023
1 parent eddfed8 commit 7dfbde6
Show file tree
Hide file tree
Showing 4 changed files with 130 additions and 19 deletions.
53 changes: 49 additions & 4 deletions Jellyfin.Plugin.Themerr.Tests/TestThemerrManager.cs
Original file line number Diff line number Diff line change
Expand Up @@ -60,6 +60,23 @@ private List<string> FixtureThemerrDbUrls()
return youtubeUrls;
}

[Fact]
[Trait("Category", "Unit")]
private void TestGetExistingThemerrDataValue()
{
var dummyJson = Path.Combine(
Directory.GetCurrentDirectory(),
"data",
"dummy.json");

Assert.Equal(
"dummy_value",
_themerrManager.GetExistingThemerrDataValue("dummy_key", dummyJson));
Assert.Equal(
"https://www.youtube.com/watch?v=E8nxMWr2sr4",
_themerrManager.GetExistingThemerrDataValue("youtube_theme_url", dummyJson));
}

[Fact]
[Trait("Category", "Unit")]
private void TestSaveMp3()
Expand Down Expand Up @@ -180,15 +197,15 @@ private void TestSaveMp3InvalidUrl()

[Fact]
[Trait("Category", "Unit")]
private void TestShouldSkipDownload()
private void TestContinueDownload()
{
var themePath = Path.Combine(
"theme.mp3");
var themerrDataPath = Path.Combine(
"themerr_data.json");

var shouldSkipDownload = _themerrManager.ShouldSkipDownload(themePath, themerrDataPath);
Assert.False(shouldSkipDownload, "ShouldSkipDownload returned True");
var check = _themerrManager.ContinueDownload(themePath, themerrDataPath);
Assert.True(check, "ContinueDownload returned False");
}

[Fact]
Expand Down Expand Up @@ -273,11 +290,16 @@ private void TestSaveThemerrData()
// set mock themerrDataPath using a random number
var mockThemerrDataPath = $"themerr_{new Random().Next()}.json";

var stubVideoPath = Path.Combine(
Directory.GetCurrentDirectory(),
"data",
"video_stub.mp4");

// loop over each themerrDbLink
foreach (var youtubeThemeUrl in FixtureYoutubeUrls())
{
// save themerr data
var fileExists = _themerrManager.SaveThemerrData(mockThemerrDataPath, youtubeThemeUrl);
var fileExists = _themerrManager.SaveThemerrData(stubVideoPath, mockThemerrDataPath, youtubeThemeUrl);
Assert.True(fileExists, $"SaveThemerrData did not return True for {youtubeThemeUrl}");

// check if file exists
Expand All @@ -293,4 +315,27 @@ private void TestSaveThemerrData()
$"youtubeThemeUrl {youtubeThemeUrl} does not match savedYoutubeThemeUrl {savedYoutubeThemeUrl}");
}
}

[Fact]
[Trait("Category", "Unit")]
private void TestGetMd5Hash()
{
var stubVideoPath = Path.Combine(
Directory.GetCurrentDirectory(),
"data",
"video_stub.mp4");

var expectedMd5HashFile = Path.Combine(
Directory.GetCurrentDirectory(),
"data",
"video_stub.mp4.md5");

// get expected md5 hash out of file
var expectedMd5Hash = File.ReadAllText(expectedMd5HashFile).Trim();

// get actual md5 hash
var actualMd5Hash = _themerrManager.GetMd5Hash(stubVideoPath);

Assert.Equal(expectedMd5Hash, actualMd5Hash);
}
}
5 changes: 5 additions & 0 deletions Jellyfin.Plugin.Themerr.Tests/data/dummy.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
{
"downloaded_timestamp": "2023-12-12T04:07:41.7659077Z",
"youtube_theme_url": "https://www.youtube.com/watch?v=E8nxMWr2sr4",
"dummy_key": "dummy_value"
}
Binary file not shown.
91 changes: 76 additions & 15 deletions Jellyfin.Plugin.Themerr/ThemerrManager.cs
Original file line number Diff line number Diff line change
Expand Up @@ -42,11 +42,12 @@ public ThemerrManager(ILibraryManager libraryManager, ILogger<ThemerrManager> lo
}

/// <summary>
/// Get the existing youtube theme url from the themerr data file if it exists.
/// Get a value from the themerr data file if it exists.
/// </summary>
/// <param name="key">The key to search for.</param>
/// <param name="themerrDataPath">The path to the themerr data file.</param>
/// <returns>The existing YouTube theme url if it exists, empty string otherwise.</returns>
public static string GetExistingYoutubeThemeUrl(string themerrDataPath)
/// <returns>The value of the key if it exists, empty string otherwise.</returns>
public string GetExistingThemerrDataValue(string key, string themerrDataPath)
{
if (!System.IO.File.Exists(themerrDataPath))
{
Expand All @@ -55,7 +56,7 @@ public static string GetExistingYoutubeThemeUrl(string themerrDataPath)

var jsonString = System.IO.File.ReadAllText(themerrDataPath);
dynamic jsonData = JsonConvert.DeserializeObject(jsonString);
return jsonData?.youtube_theme_url;
return jsonData?[key];
}

/// <summary>
Expand Down Expand Up @@ -132,42 +133,83 @@ public void ProcessMovieTheme(Movie movie)
var themePath = GetThemePath(movie);
var themerrDataPath = GetThemerrDataPath(movie);

if (ShouldSkipDownload(themePath, themerrDataPath))
if (!ContinueDownload(themePath, themerrDataPath))
{
return;
}

var existingYoutubeThemeUrl = GetExistingYoutubeThemeUrl(themerrDataPath);
var existingYoutubeThemeUrl = GetExistingThemerrDataValue("youtube_theme_url", themerrDataPath);

Check warning on line 141 in Jellyfin.Plugin.Themerr/ThemerrManager.cs

View check run for this annotation

Codecov / codecov/patch

Jellyfin.Plugin.Themerr/ThemerrManager.cs#L141

Added line #L141 was not covered by tests

// get tmdb id
var tmdbId = movie.GetProviderId(MetadataProvider.Tmdb);

// create themerrdb url
var themerrDbLink = CreateThemerrDbLink(tmdbId);
var themerrDbUrl = CreateThemerrDbLink(tmdbId);

Check warning on line 147 in Jellyfin.Plugin.Themerr/ThemerrManager.cs

View check run for this annotation

Codecov / codecov/patch

Jellyfin.Plugin.Themerr/ThemerrManager.cs#L147

Added line #L147 was not covered by tests

var youtubeThemeUrl = GetYoutubeThemeUrl(themerrDbLink, movieTitle);
var youtubeThemeUrl = GetYoutubeThemeUrl(themerrDbUrl, movieTitle);

Check warning on line 149 in Jellyfin.Plugin.Themerr/ThemerrManager.cs

View check run for this annotation

Codecov / codecov/patch

Jellyfin.Plugin.Themerr/ThemerrManager.cs#L149

Added line #L149 was not covered by tests

if (string.IsNullOrEmpty(youtubeThemeUrl) || youtubeThemeUrl == existingYoutubeThemeUrl)
{
return;
}

SaveMp3(themePath, youtubeThemeUrl);
SaveThemerrData(themerrDataPath, youtubeThemeUrl);
var successMp3 = SaveMp3(themePath, youtubeThemeUrl);

Check warning on line 156 in Jellyfin.Plugin.Themerr/ThemerrManager.cs

View check run for this annotation

Codecov / codecov/patch

Jellyfin.Plugin.Themerr/ThemerrManager.cs#L156

Added line #L156 was not covered by tests
if (!successMp3)
{
return;

Check warning on line 159 in Jellyfin.Plugin.Themerr/ThemerrManager.cs

View check run for this annotation

Codecov / codecov/patch

Jellyfin.Plugin.Themerr/ThemerrManager.cs#L158-L159

Added lines #L158 - L159 were not covered by tests
}

var successThemerrData = SaveThemerrData(themePath, themerrDataPath, youtubeThemeUrl);

Check warning on line 162 in Jellyfin.Plugin.Themerr/ThemerrManager.cs

View check run for this annotation

Codecov / codecov/patch

Jellyfin.Plugin.Themerr/ThemerrManager.cs#L162

Added line #L162 was not covered by tests
if (!successThemerrData)
{
return;

Check warning on line 165 in Jellyfin.Plugin.Themerr/ThemerrManager.cs

View check run for this annotation

Codecov / codecov/patch

Jellyfin.Plugin.Themerr/ThemerrManager.cs#L164-L165

Added lines #L164 - L165 were not covered by tests
}

movie.RefreshMetadata(CancellationToken.None);
}

/// <summary>
/// Check if the theme song should be downloaded.
///
/// If theme.mp3 exists and themerr.json doesn't exist, then skip to avoid overwriting user supplied themes.
/// Various checks are performed to determine if the theme song should be downloaded.
/// </summary>
/// <param name="themePath">The path to the theme song.</param>
/// <param name="themerrDataPath">The path to the themerr data file.</param>
/// <returns>True if the theme song should NOT be downloaded, false otherwise.</returns>
public bool ShouldSkipDownload(string themePath, string themerrDataPath)
/// <returns>True to continue with downloaded, false otherwise.</returns>
public bool ContinueDownload(string themePath, string themerrDataPath)
{
return System.IO.File.Exists(themePath) && !System.IO.File.Exists(themerrDataPath);
if (!System.IO.File.Exists(themePath) || !System.IO.File.Exists(themerrDataPath))
{
// neither file exists, so don't skip
return true;
}

if (!System.IO.File.Exists(themePath) && System.IO.File.Exists(themerrDataPath))
{

Check warning on line 188 in Jellyfin.Plugin.Themerr/ThemerrManager.cs

View check run for this annotation

Codecov / codecov/patch

Jellyfin.Plugin.Themerr/ThemerrManager.cs#L188

Added line #L188 was not covered by tests
// the theme is missing, so delete the themerr data file
System.IO.File.Delete(themerrDataPath);
}

Check warning on line 191 in Jellyfin.Plugin.Themerr/ThemerrManager.cs

View check run for this annotation

Codecov / codecov/patch

Jellyfin.Plugin.Themerr/ThemerrManager.cs#L190-L191

Added lines #L190 - L191 were not covered by tests

// return System.IO.File.Exists(themePath) && !System.IO.File.Exists(themerrDataPath);
if (System.IO.File.Exists(themePath) && !System.IO.File.Exists(themerrDataPath))
{

Check warning on line 195 in Jellyfin.Plugin.Themerr/ThemerrManager.cs

View check run for this annotation

Codecov / codecov/patch

Jellyfin.Plugin.Themerr/ThemerrManager.cs#L195

Added line #L195 was not covered by tests
// the theme is user supplied, so don't overwrite it
return false;

Check warning on line 197 in Jellyfin.Plugin.Themerr/ThemerrManager.cs

View check run for this annotation

Codecov / codecov/patch

Jellyfin.Plugin.Themerr/ThemerrManager.cs#L197

Added line #L197 was not covered by tests
}

var existingThemeMd5 = GetExistingThemerrDataValue("theme_md5", themerrDataPath);

Check warning on line 200 in Jellyfin.Plugin.Themerr/ThemerrManager.cs

View check run for this annotation

Codecov / codecov/patch

Jellyfin.Plugin.Themerr/ThemerrManager.cs#L200

Added line #L200 was not covered by tests

// if existing theme md5 is empty, don't skip
if (string.IsNullOrEmpty(existingThemeMd5))
{
return true;

Check warning on line 205 in Jellyfin.Plugin.Themerr/ThemerrManager.cs

View check run for this annotation

Codecov / codecov/patch

Jellyfin.Plugin.Themerr/ThemerrManager.cs#L204-L205

Added lines #L204 - L205 were not covered by tests
}

// check if the theme hash matches what is in the themerr data file
var themeMd5 = GetMd5Hash(themePath);

Check warning on line 209 in Jellyfin.Plugin.Themerr/ThemerrManager.cs

View check run for this annotation

Codecov / codecov/patch

Jellyfin.Plugin.Themerr/ThemerrManager.cs#L209

Added line #L209 was not covered by tests

// if hashes match, theme is supplied by themerr, otherwise it is user supplied
return themeMd5 != existingThemeMd5;

Check warning on line 212 in Jellyfin.Plugin.Themerr/ThemerrManager.cs

View check run for this annotation

Codecov / codecov/patch

Jellyfin.Plugin.Themerr/ThemerrManager.cs#L212

Added line #L212 was not covered by tests
}

/// <summary>
Expand Down Expand Up @@ -226,15 +268,17 @@ public string GetYoutubeThemeUrl(string themerrDbUrl, string movieTitle)
/// <summary>
/// Save the themerr data file.
/// </summary>
/// <param name="themePath">The path to the theme song.</param>
/// <param name="themerrDataPath">The path to the themerr data file.</param>
/// <param name="youtubeThemeUrl">The YouTube theme url.</param>
/// <returns>True if the file was saved successfully, false otherwise.</returns>
public bool SaveThemerrData(string themerrDataPath, string youtubeThemeUrl)
public bool SaveThemerrData(string themePath, string themerrDataPath, string youtubeThemeUrl)
{
var success = false;
var themerrData = new
{
downloaded_timestamp = DateTime.UtcNow,
theme_md5 = GetMd5Hash(themePath),
youtube_theme_url = youtubeThemeUrl
};
try
Expand All @@ -250,6 +294,23 @@ public bool SaveThemerrData(string themerrDataPath, string youtubeThemeUrl)
return success && WaitForFile(themerrDataPath, 10000);
}

/// <summary>
/// Get the MD5 hash of a file.
/// </summary>
/// <param name="filePath">The file path.</param>
/// <returns>The MD5 hash of the file.</returns>
public string GetMd5Hash(string filePath)
{
using (var md5 = System.Security.Cryptography.MD5.Create())
{
using (var stream = System.IO.File.OpenRead(filePath))
{
var hash = md5.ComputeHash(stream);
return BitConverter.ToString(hash).Replace("-", string.Empty).ToLowerInvariant();
}
}
}

/// <summary>
/// Wait for file to exist on disk and is not locked by another process.
/// </summary>
Expand Down

0 comments on commit 7dfbde6

Please sign in to comment.