mirror of
https://github.com/beetbox/beets.git
synced 2025-12-06 08:39:17 +01:00
393 lines
12 KiB
Python
393 lines
12 KiB
Python
# This file is part of beets.
|
|
# Copyright 2016, Thomas Scholtes
|
|
#
|
|
# Permission is hereby granted, free of charge, to any person obtaining
|
|
# a copy of this software and associated documentation files (the
|
|
# "Software"), to deal in the Software without restriction, including
|
|
# without limitation the rights to use, copy, modify, merge, publish,
|
|
# distribute, sublicense, and/or sell copies of the Software, and to
|
|
# permit persons to whom the Software is furnished to do so, subject to
|
|
# the following conditions:
|
|
#
|
|
# The above copyright notice and this permission notice shall be
|
|
# included in all copies or substantial portions of the Software.
|
|
|
|
|
|
import unittest
|
|
from typing import ClassVar
|
|
|
|
from mediafile import MediaFile
|
|
|
|
from beets import config
|
|
from beets.test.helper import (
|
|
AsIsImporterMixin,
|
|
ImportTestCase,
|
|
PluginMixin,
|
|
has_program,
|
|
)
|
|
from beetsplug.replaygain import (
|
|
FatalGstreamerPluginReplayGainError,
|
|
GStreamerBackend,
|
|
)
|
|
|
|
try:
|
|
import gi
|
|
|
|
gi.require_version("Gst", "1.0")
|
|
GST_AVAILABLE = True
|
|
except (ImportError, ValueError):
|
|
GST_AVAILABLE = False
|
|
|
|
if any(has_program(cmd, ["-v"]) for cmd in ["mp3gain", "aacgain"]):
|
|
GAIN_PROG_AVAILABLE = True
|
|
else:
|
|
GAIN_PROG_AVAILABLE = False
|
|
|
|
FFMPEG_AVAILABLE = has_program("ffmpeg", ["-version"])
|
|
|
|
|
|
def reset_replaygain(item):
|
|
item["rg_track_peak"] = None
|
|
item["rg_track_gain"] = None
|
|
item["rg_album_gain"] = None
|
|
item["rg_album_gain"] = None
|
|
item["r128_track_gain"] = None
|
|
item["r128_album_gain"] = None
|
|
item.write()
|
|
item.store()
|
|
|
|
|
|
class ReplayGainTestCase(PluginMixin, ImportTestCase):
|
|
db_on_disk = True
|
|
plugin = "replaygain"
|
|
preload_plugin = False
|
|
|
|
backend: ClassVar[str]
|
|
|
|
def setUp(self):
|
|
# Implemented by Mixins, see above. This may decide to skip the test.
|
|
self.test_backend()
|
|
|
|
super().setUp()
|
|
self.config["replaygain"]["backend"] = self.backend
|
|
|
|
self.load_plugins()
|
|
|
|
|
|
class ThreadedImportMixin:
|
|
def setUp(self):
|
|
super().setUp()
|
|
self.config["threaded"] = True
|
|
|
|
|
|
class GstBackendMixin:
|
|
backend = "gstreamer"
|
|
has_r128_support = True
|
|
|
|
def test_backend(self):
|
|
"""Check whether the backend actually has all required functionality."""
|
|
try:
|
|
# Check if required plugins can be loaded by instantiating a
|
|
# GStreamerBackend (via its .__init__).
|
|
config["replaygain"]["targetlevel"] = 89
|
|
GStreamerBackend(config["replaygain"], None)
|
|
except FatalGstreamerPluginReplayGainError as e:
|
|
# Skip the test if plugins could not be loaded.
|
|
self.skipTest(str(e))
|
|
|
|
|
|
class CmdBackendMixin:
|
|
backend = "command"
|
|
has_r128_support = False
|
|
|
|
def test_backend(self):
|
|
"""Check whether the backend actually has all required functionality."""
|
|
pass
|
|
|
|
|
|
class FfmpegBackendMixin:
|
|
backend = "ffmpeg"
|
|
has_r128_support = True
|
|
|
|
def test_backend(self):
|
|
"""Check whether the backend actually has all required functionality."""
|
|
pass
|
|
|
|
|
|
class ReplayGainCliTest:
|
|
FNAME: str
|
|
|
|
def _add_album(self, *args, **kwargs):
|
|
# Use a file with non-zero volume (most test assets are total silence)
|
|
album = self.add_album_fixture(*args, fname=self.FNAME, **kwargs)
|
|
for item in album.items():
|
|
reset_replaygain(item)
|
|
|
|
return album
|
|
|
|
def test_cli_saves_track_gain(self):
|
|
self._add_album(2)
|
|
|
|
for item in self.lib.items():
|
|
assert item.rg_track_peak is None
|
|
assert item.rg_track_gain is None
|
|
mediafile = MediaFile(item.path)
|
|
assert mediafile.rg_track_peak is None
|
|
assert mediafile.rg_track_gain is None
|
|
|
|
self.run_command("replaygain")
|
|
|
|
# Skip the test if rg_track_peak and rg_track gain is None, assuming
|
|
# that it could only happen if the decoder plugins are missing.
|
|
if all(
|
|
i.rg_track_peak is None and i.rg_track_gain is None
|
|
for i in self.lib.items()
|
|
):
|
|
self.skipTest("decoder plugins could not be loaded.")
|
|
|
|
for item in self.lib.items():
|
|
assert item.rg_track_peak is not None
|
|
assert item.rg_track_gain is not None
|
|
mediafile = MediaFile(item.path)
|
|
self.assertAlmostEqual(
|
|
mediafile.rg_track_peak, item.rg_track_peak, places=6
|
|
)
|
|
self.assertAlmostEqual(
|
|
mediafile.rg_track_gain, item.rg_track_gain, places=2
|
|
)
|
|
|
|
def test_cli_skips_calculated_tracks(self):
|
|
album_rg = self._add_album(1)
|
|
item_rg = album_rg.items()[0]
|
|
|
|
if self.has_r128_support:
|
|
album_r128 = self._add_album(1, ext="opus")
|
|
item_r128 = album_r128.items()[0]
|
|
|
|
self.run_command("replaygain")
|
|
|
|
item_rg.load()
|
|
assert item_rg.rg_track_gain is not None
|
|
assert item_rg.rg_track_peak is not None
|
|
assert item_rg.r128_track_gain is None
|
|
|
|
item_rg.rg_track_gain += 1.0
|
|
item_rg.rg_track_peak += 1.0
|
|
item_rg.store()
|
|
rg_track_gain = item_rg.rg_track_gain
|
|
rg_track_peak = item_rg.rg_track_peak
|
|
|
|
if self.has_r128_support:
|
|
item_r128.load()
|
|
assert item_r128.r128_track_gain is not None
|
|
assert item_r128.rg_track_gain is None
|
|
assert item_r128.rg_track_peak is None
|
|
|
|
item_r128.r128_track_gain += 1.0
|
|
item_r128.store()
|
|
r128_track_gain = item_r128.r128_track_gain
|
|
|
|
self.run_command("replaygain")
|
|
|
|
item_rg.load()
|
|
self.assertEqual(item_rg.rg_track_gain, rg_track_gain)
|
|
self.assertEqual(item_rg.rg_track_peak, rg_track_peak)
|
|
|
|
if self.has_r128_support:
|
|
item_r128.load()
|
|
self.assertEqual(item_r128.r128_track_gain, r128_track_gain)
|
|
|
|
def test_cli_does_not_skip_wrong_tag_type(self):
|
|
"""Check that items that have tags of the wrong type won't be skipped."""
|
|
if not self.has_r128_support:
|
|
# This test is a lot less interesting if the backend cannot write
|
|
# both tag types.
|
|
self.skipTest(
|
|
"r128 tags for opus not supported on backend {}".format(
|
|
self.backend
|
|
)
|
|
)
|
|
|
|
album_rg = self._add_album(1)
|
|
item_rg = album_rg.items()[0]
|
|
|
|
album_r128 = self._add_album(1, ext="opus")
|
|
item_r128 = album_r128.items()[0]
|
|
|
|
item_rg.r128_track_gain = 0.0
|
|
item_rg.store()
|
|
|
|
item_r128.rg_track_gain = 0.0
|
|
item_r128.rg_track_peak = 42.0
|
|
item_r128.store()
|
|
|
|
self.run_command("replaygain")
|
|
item_rg.load()
|
|
item_r128.load()
|
|
|
|
assert item_rg.rg_track_gain is not None
|
|
assert item_rg.rg_track_peak is not None
|
|
# FIXME: Should the plugin null this field?
|
|
# assert item_rg.r128_track_gain is None
|
|
|
|
assert item_r128.r128_track_gain is not None
|
|
# FIXME: Should the plugin null these fields?
|
|
# assert item_r128.rg_track_gain is None
|
|
# assert item_r128.rg_track_peak is None
|
|
|
|
def test_cli_saves_album_gain_to_file(self):
|
|
self._add_album(2)
|
|
|
|
for item in self.lib.items():
|
|
mediafile = MediaFile(item.path)
|
|
assert mediafile.rg_album_peak is None
|
|
assert mediafile.rg_album_gain is None
|
|
|
|
self.run_command("replaygain", "-a")
|
|
|
|
peaks = []
|
|
gains = []
|
|
for item in self.lib.items():
|
|
mediafile = MediaFile(item.path)
|
|
peaks.append(mediafile.rg_album_peak)
|
|
gains.append(mediafile.rg_album_gain)
|
|
|
|
# Make sure they are all the same
|
|
self.assertEqual(max(peaks), min(peaks))
|
|
self.assertEqual(max(gains), min(gains))
|
|
|
|
self.assertNotEqual(max(gains), 0.0)
|
|
self.assertNotEqual(max(peaks), 0.0)
|
|
|
|
def test_cli_writes_only_r128_tags(self):
|
|
if not self.has_r128_support:
|
|
self.skipTest(
|
|
"r128 tags for opus not supported on backend {}".format(
|
|
self.backend
|
|
)
|
|
)
|
|
|
|
album = self._add_album(2, ext="opus")
|
|
|
|
self.run_command("replaygain", "-a")
|
|
|
|
for item in album.items():
|
|
mediafile = MediaFile(item.path)
|
|
# does not write REPLAYGAIN_* tags
|
|
assert mediafile.rg_track_gain is None
|
|
assert mediafile.rg_album_gain is None
|
|
# writes R128_* tags
|
|
assert mediafile.r128_track_gain is not None
|
|
assert mediafile.r128_album_gain is not None
|
|
|
|
def test_targetlevel_has_effect(self):
|
|
album = self._add_album(1)
|
|
item = album.items()[0]
|
|
|
|
def analyse(target_level):
|
|
self.config["replaygain"]["targetlevel"] = target_level
|
|
self.run_command("replaygain", "-f")
|
|
item.load()
|
|
return item.rg_track_gain
|
|
|
|
gain_relative_to_84 = analyse(84)
|
|
gain_relative_to_89 = analyse(89)
|
|
|
|
self.assertNotEqual(gain_relative_to_84, gain_relative_to_89)
|
|
|
|
def test_r128_targetlevel_has_effect(self):
|
|
if not self.has_r128_support:
|
|
self.skipTest(
|
|
"r128 tags for opus not supported on backend {}".format(
|
|
self.backend
|
|
)
|
|
)
|
|
|
|
album = self._add_album(1, ext="opus")
|
|
item = album.items()[0]
|
|
|
|
def analyse(target_level):
|
|
self.config["replaygain"]["r128_targetlevel"] = target_level
|
|
self.run_command("replaygain", "-f")
|
|
item.load()
|
|
return item.r128_track_gain
|
|
|
|
gain_relative_to_84 = analyse(84)
|
|
gain_relative_to_89 = analyse(89)
|
|
|
|
self.assertNotEqual(gain_relative_to_84, gain_relative_to_89)
|
|
|
|
def test_per_disc(self):
|
|
# Use the per_disc option and add a little more concurrency.
|
|
album = self._add_album(track_count=4, disc_count=3)
|
|
self.config["replaygain"]["per_disc"] = True
|
|
self.run_command("replaygain", "-a")
|
|
|
|
# FIXME: Add fixtures with known track/album gain (within a suitable
|
|
# tolerance) so that we can actually check per-disc operation here.
|
|
for item in album.items():
|
|
assert item.rg_track_gain is not None
|
|
assert item.rg_album_gain is not None
|
|
|
|
|
|
@unittest.skipIf(not GST_AVAILABLE, "gstreamer cannot be found")
|
|
class ReplayGainGstCliTest(
|
|
ReplayGainCliTest, ReplayGainTestCase, GstBackendMixin
|
|
):
|
|
FNAME = "full" # file contains only silence
|
|
|
|
|
|
@unittest.skipIf(not GAIN_PROG_AVAILABLE, "no *gain command found")
|
|
class ReplayGainCmdCliTest(
|
|
ReplayGainCliTest, ReplayGainTestCase, CmdBackendMixin
|
|
):
|
|
FNAME = "full" # file contains only silence
|
|
|
|
|
|
@unittest.skipIf(not FFMPEG_AVAILABLE, "ffmpeg cannot be found")
|
|
class ReplayGainFfmpegCliTest(
|
|
ReplayGainCliTest, ReplayGainTestCase, FfmpegBackendMixin
|
|
):
|
|
FNAME = "full" # file contains only silence
|
|
|
|
|
|
@unittest.skipIf(not FFMPEG_AVAILABLE, "ffmpeg cannot be found")
|
|
class ReplayGainFfmpegNoiseCliTest(
|
|
ReplayGainCliTest, ReplayGainTestCase, FfmpegBackendMixin
|
|
):
|
|
FNAME = "whitenoise"
|
|
|
|
|
|
class ImportTest(AsIsImporterMixin):
|
|
def test_import_converted(self):
|
|
self.run_asis_importer()
|
|
for item in self.lib.items():
|
|
# FIXME: Add fixtures with known track/album gain (within a
|
|
# suitable tolerance) so that we can actually check correct
|
|
# operation here.
|
|
assert item.rg_track_gain is not None
|
|
assert item.rg_album_gain is not None
|
|
|
|
|
|
@unittest.skipIf(not GST_AVAILABLE, "gstreamer cannot be found")
|
|
class ReplayGainGstImportTest(ImportTest, ReplayGainTestCase, GstBackendMixin):
|
|
pass
|
|
|
|
|
|
@unittest.skipIf(not GAIN_PROG_AVAILABLE, "no *gain command found")
|
|
class ReplayGainCmdImportTest(ImportTest, ReplayGainTestCase, CmdBackendMixin):
|
|
pass
|
|
|
|
|
|
@unittest.skipIf(not FFMPEG_AVAILABLE, "ffmpeg cannot be found")
|
|
class ReplayGainFfmpegImportTest(
|
|
ImportTest, ReplayGainTestCase, FfmpegBackendMixin
|
|
):
|
|
pass
|
|
|
|
|
|
@unittest.skipIf(not FFMPEG_AVAILABLE, "ffmpeg cannot be found")
|
|
class ReplayGainFfmpegThreadedImportTest(
|
|
ThreadedImportMixin, ImportTest, ReplayGainTestCase, FfmpegBackendMixin
|
|
):
|
|
pass
|