2020-01-16 16:20:01 -07:00
|
|
|
import logging
|
|
|
|
import pathlib
|
|
|
|
import time
|
|
|
|
|
2021-10-05 17:26:46 -04:00
|
|
|
from ..claims.test_claim_commands import ClaimTestCase
|
2020-01-16 16:20:01 -07:00
|
|
|
from lbry.conf import TranscodeConfig
|
|
|
|
from lbry.file_analysis import VideoFileAnalyzer
|
|
|
|
|
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
|
|
|
class MeasureTime:
|
|
|
|
def __init__(self, text):
|
|
|
|
print(text, end="...", flush=True)
|
|
|
|
|
|
|
|
def __enter__(self):
|
|
|
|
self.start = time.perf_counter()
|
|
|
|
|
|
|
|
def __exit__(self, exc_type, exc_val, exc_tb):
|
|
|
|
end = time.perf_counter()
|
|
|
|
print(f" done in {end - self.start:.6f}s", flush=True)
|
|
|
|
|
|
|
|
|
|
|
|
class TranscodeValidation(ClaimTestCase):
|
|
|
|
|
|
|
|
def make_name(self, name, extension=""):
|
|
|
|
path = pathlib.Path(self.video_file_name)
|
|
|
|
return path.parent / f"{path.stem}_{name}{extension or path.suffix}"
|
|
|
|
|
|
|
|
async def asyncSetUp(self):
|
|
|
|
await super().asyncSetUp()
|
|
|
|
self.conf = TranscodeConfig()
|
|
|
|
self.conf.volume_analysis_time = 0 # disable it as the test file isn't very good here
|
|
|
|
self.analyzer = VideoFileAnalyzer(self.conf)
|
2020-03-18 12:43:27 -06:00
|
|
|
self.assertTrue((await self.analyzer.status())["available"]) # ensure ffmpeg path detected
|
2020-01-16 16:20:01 -07:00
|
|
|
file_ogg = self.make_name("ogg", ".ogg")
|
2020-02-03 15:53:27 -07:00
|
|
|
self.video_file_ogg = str(file_ogg)
|
2020-01-16 16:20:01 -07:00
|
|
|
if not file_ogg.exists():
|
|
|
|
command = f'-i "{self.video_file_name}" -c:v libtheora -q:v 4 -c:a libvorbis -q:a 4 ' \
|
|
|
|
f'-c:s copy -c:d copy "{file_ogg}"'
|
|
|
|
with MeasureTime(f"Creating {file_ogg.name}"):
|
2020-03-18 12:43:27 -06:00
|
|
|
output, code = await self.analyzer._execute_ffmpeg(command)
|
2020-01-16 16:20:01 -07:00
|
|
|
self.assertEqual(code, 0, output)
|
|
|
|
|
|
|
|
file_webm = self.make_name("webm", ".webm")
|
2020-02-03 15:53:27 -07:00
|
|
|
self.video_file_webm = str(file_webm)
|
2020-01-16 16:20:01 -07:00
|
|
|
if not file_webm.exists():
|
|
|
|
command = f'-i "{self.video_file_name}" -c:v libvpx-vp9 -crf 36 -b:v 0 -cpu-used 2 ' \
|
|
|
|
f'-c:a libopus -b:a 128k -c:s copy -c:d copy "{file_webm}"'
|
|
|
|
with MeasureTime(f"Creating {file_webm.name}"):
|
2020-03-18 12:43:27 -06:00
|
|
|
output, code = await self.analyzer._execute_ffmpeg(command)
|
2020-01-16 16:20:01 -07:00
|
|
|
self.assertEqual(code, 0, output)
|
|
|
|
|
|
|
|
async def test_should_work(self):
|
2020-03-26 07:53:13 -06:00
|
|
|
new_file_name, _ = await self.analyzer.verify_or_repair(True, False, self.video_file_name)
|
2020-02-03 15:53:27 -07:00
|
|
|
self.assertEqual(self.video_file_name, new_file_name)
|
2020-03-26 07:53:13 -06:00
|
|
|
new_file_name, _ = await self.analyzer.verify_or_repair(True, False, self.video_file_ogg)
|
2020-02-03 15:53:27 -07:00
|
|
|
self.assertEqual(self.video_file_ogg, new_file_name)
|
2020-03-26 07:53:13 -06:00
|
|
|
new_file_name, spec = await self.analyzer.verify_or_repair(True, False, self.video_file_webm)
|
2020-02-03 15:53:27 -07:00
|
|
|
self.assertEqual(self.video_file_webm, new_file_name)
|
2020-03-26 07:53:13 -06:00
|
|
|
self.assertEqual(spec["width"], 1280)
|
|
|
|
self.assertEqual(spec["height"], 720)
|
2020-03-31 11:29:58 -04:00
|
|
|
self.assertEqual(spec["duration"], 16)
|
2020-01-16 16:20:01 -07:00
|
|
|
|
|
|
|
async def test_volume(self):
|
2020-02-03 22:06:15 -05:00
|
|
|
self.conf.volume_analysis_time = 200
|
|
|
|
with self.assertRaisesRegex(Exception, "lower than prime"):
|
|
|
|
await self.analyzer.verify_or_repair(True, False, self.video_file_name)
|
2020-01-16 16:20:01 -07:00
|
|
|
|
|
|
|
async def test_container(self):
|
|
|
|
file_name = self.make_name("bad_container", ".avi")
|
|
|
|
if not file_name.exists():
|
|
|
|
command = f'-i "{self.video_file_name}" -c copy -map 0 "{file_name}"'
|
|
|
|
with MeasureTime(f"Creating {file_name.name}"):
|
2020-03-18 12:43:27 -06:00
|
|
|
output, code = await self.analyzer._execute_ffmpeg(command)
|
2020-01-16 16:20:01 -07:00
|
|
|
self.assertEqual(code, 0, output)
|
|
|
|
|
|
|
|
with self.assertRaisesRegex(Exception, "Container format is not in the approved list"):
|
|
|
|
await self.analyzer.verify_or_repair(True, False, file_name)
|
|
|
|
|
2020-03-26 07:53:13 -06:00
|
|
|
fixed_file, _ = await self.analyzer.verify_or_repair(True, True, file_name)
|
2020-01-16 16:20:01 -07:00
|
|
|
pathlib.Path(fixed_file).unlink()
|
|
|
|
|
|
|
|
async def test_video_codec(self):
|
|
|
|
file_name = self.make_name("bad_video_codec_1")
|
|
|
|
if not file_name.exists():
|
|
|
|
command = f'-i "{self.video_file_name}" -c copy -map 0 -c:v libx265 -preset superfast "{file_name}"'
|
|
|
|
with MeasureTime(f"Creating {file_name.name}"):
|
2020-03-18 12:43:27 -06:00
|
|
|
output, code = await self.analyzer._execute_ffmpeg(command)
|
2020-01-16 16:20:01 -07:00
|
|
|
self.assertEqual(code, 0, output)
|
|
|
|
|
|
|
|
with self.assertRaisesRegex(Exception, "Video codec is not in the approved list"):
|
|
|
|
await self.analyzer.verify_or_repair(True, False, file_name)
|
|
|
|
with self.assertRaisesRegex(Exception, "faststart flag was not used"):
|
|
|
|
await self.analyzer.verify_or_repair(True, False, file_name)
|
|
|
|
|
2020-03-26 07:53:13 -06:00
|
|
|
fixed_file, _ = await self.analyzer.verify_or_repair(True, True, file_name)
|
2020-01-16 16:20:01 -07:00
|
|
|
pathlib.Path(fixed_file).unlink()
|
|
|
|
|
2020-03-04 15:53:46 -07:00
|
|
|
async def test_max_bit_rate(self):
|
|
|
|
self.conf.video_bitrate_maximum = 100
|
|
|
|
with self.assertRaisesRegex(Exception, "The bit rate is above the configured maximum"):
|
|
|
|
await self.analyzer.verify_or_repair(True, False, self.video_file_name)
|
|
|
|
|
2020-01-16 16:20:01 -07:00
|
|
|
async def test_video_format(self):
|
|
|
|
file_name = self.make_name("bad_video_format_1")
|
|
|
|
if not file_name.exists():
|
|
|
|
command = f'-i "{self.video_file_name}" -c copy -map 0 -c:v libx264 ' \
|
|
|
|
f'-vf format=yuv444p "{file_name}"'
|
|
|
|
with MeasureTime(f"Creating {file_name.name}"):
|
2020-03-18 12:43:27 -06:00
|
|
|
output, code = await self.analyzer._execute_ffmpeg(command)
|
2020-01-16 16:20:01 -07:00
|
|
|
self.assertEqual(code, 0, output)
|
|
|
|
|
|
|
|
with self.assertRaisesRegex(Exception, "pixel format does not match the approved"):
|
|
|
|
await self.analyzer.verify_or_repair(True, False, file_name)
|
|
|
|
|
2020-03-26 07:53:13 -06:00
|
|
|
fixed_file, _ = await self.analyzer.verify_or_repair(True, True, file_name)
|
2020-01-16 16:20:01 -07:00
|
|
|
pathlib.Path(fixed_file).unlink()
|
|
|
|
|
|
|
|
async def test_audio_codec(self):
|
|
|
|
file_name = self.make_name("bad_audio_codec_1", ".mkv")
|
|
|
|
if not file_name.exists():
|
|
|
|
command = f'-i "{self.video_file_name}" -c copy -map 0 -c:a pcm_s16le "{file_name}"'
|
|
|
|
with MeasureTime(f"Creating {file_name.name}"):
|
2020-03-18 12:43:27 -06:00
|
|
|
output, code = await self.analyzer._execute_ffmpeg(command)
|
2020-01-16 16:20:01 -07:00
|
|
|
self.assertEqual(code, 0, output)
|
|
|
|
|
|
|
|
with self.assertRaisesRegex(Exception, "Audio codec is not in the approved list"):
|
|
|
|
await self.analyzer.verify_or_repair(True, False, file_name)
|
|
|
|
|
2020-03-26 07:53:13 -06:00
|
|
|
fixed_file, _ = await self.analyzer.verify_or_repair(True, True, file_name)
|
2020-01-16 16:20:01 -07:00
|
|
|
pathlib.Path(fixed_file).unlink()
|
|
|
|
|
|
|
|
async def test_extension_choice(self):
|
|
|
|
|
2020-02-03 15:53:27 -07:00
|
|
|
scan_data = await self.analyzer._get_scan_data(True, self.video_file_name)
|
|
|
|
extension = self.analyzer._get_best_container_extension(scan_data, "")
|
|
|
|
self.assertEqual(extension, pathlib.Path(self.video_file_name).suffix[1:])
|
|
|
|
|
|
|
|
scan_data = await self.analyzer._get_scan_data(True, self.video_file_ogg)
|
|
|
|
extension = self.analyzer._get_best_container_extension(scan_data, "")
|
2020-02-11 12:23:19 -07:00
|
|
|
self.assertEqual(extension, "ogv")
|
2020-02-03 15:53:27 -07:00
|
|
|
|
|
|
|
scan_data = await self.analyzer._get_scan_data(True, self.video_file_webm)
|
|
|
|
extension = self.analyzer._get_best_container_extension(scan_data, "")
|
|
|
|
self.assertEqual(extension, "webm")
|
2020-01-16 16:20:01 -07:00
|
|
|
|
|
|
|
extension = self.analyzer._get_best_container_extension("", "libx264 -crf 23")
|
|
|
|
self.assertEqual("mp4", extension)
|
|
|
|
|
|
|
|
extension = self.analyzer._get_best_container_extension("", "libvpx-vp9 -crf 23")
|
|
|
|
self.assertEqual("webm", extension)
|
|
|
|
|
|
|
|
extension = self.analyzer._get_best_container_extension("", "libtheora")
|
2020-02-11 12:23:19 -07:00
|
|
|
self.assertEqual("ogv", extension)
|
2020-01-16 16:20:01 -07:00
|
|
|
|
|
|
|
async def test_no_ffmpeg(self):
|
2020-03-16 17:48:45 -06:00
|
|
|
self.conf.ffmpeg_path = "I don't really exist/"
|
|
|
|
self.analyzer._env_copy.pop("PATH", None)
|
2020-03-18 12:43:27 -06:00
|
|
|
await self.analyzer.status(reset=True)
|
2020-02-03 22:06:15 -05:00
|
|
|
with self.assertRaisesRegex(Exception, "Unable to locate"):
|
|
|
|
await self.analyzer.verify_or_repair(True, False, self.video_file_name)
|
2020-03-30 14:06:41 -04:00
|
|
|
|
|
|
|
async def test_dont_recheck_ffmpeg_installation(self):
|
|
|
|
|
|
|
|
call_count = 0
|
|
|
|
|
|
|
|
original = self.daemon._video_file_analyzer._verify_ffmpeg_installed
|
|
|
|
|
|
|
|
def _verify_ffmpeg_installed():
|
|
|
|
nonlocal call_count
|
|
|
|
call_count += 1
|
|
|
|
return original()
|
|
|
|
|
|
|
|
self.daemon._video_file_analyzer._verify_ffmpeg_installed = _verify_ffmpeg_installed
|
|
|
|
self.assertEqual(0, call_count)
|
|
|
|
await self.daemon.jsonrpc_status()
|
|
|
|
self.assertEqual(1, call_count)
|
|
|
|
# counter should not go up again
|
|
|
|
await self.daemon.jsonrpc_status()
|
|
|
|
self.assertEqual(1, call_count)
|
|
|
|
|
|
|
|
# this should force rechecking the installation
|
|
|
|
await self.daemon.jsonrpc_ffmpeg_find()
|
|
|
|
self.assertEqual(2, call_count)
|