2020-01-16 14:33:44 +01:00
|
|
|
from __future__ import print_function
|
2014-03-25 19:34:50 +01:00
|
|
|
import datetime
|
2014-03-26 20:06:48 +01:00
|
|
|
import mutagen
|
|
|
|
import mock
|
2014-03-10 21:32:23 +01:00
|
|
|
from nose.tools import *
|
2020-01-21 08:13:42 +01:00
|
|
|
from airtime_analyzer.metadata_analyzer import MetadataAnalyzer
|
2014-03-10 21:32:23 +01:00
|
|
|
|
2021-05-27 16:23:02 +02:00
|
|
|
|
2014-03-10 21:32:23 +01:00
|
|
|
def setup():
|
|
|
|
pass
|
|
|
|
|
2021-05-27 16:23:02 +02:00
|
|
|
|
2014-03-10 21:32:23 +01:00
|
|
|
def teardown():
|
|
|
|
pass
|
|
|
|
|
2021-05-27 16:23:02 +02:00
|
|
|
|
2014-03-10 21:32:23 +01:00
|
|
|
def check_default_metadata(metadata):
|
2021-05-27 16:23:02 +02:00
|
|
|
assert metadata["track_title"] == "Test Title"
|
|
|
|
assert metadata["artist_name"] == "Test Artist"
|
|
|
|
assert metadata["album_title"] == "Test Album"
|
|
|
|
assert metadata["year"] == "1999"
|
|
|
|
assert metadata["genre"] == "Test Genre"
|
|
|
|
assert metadata["track_number"] == "1"
|
|
|
|
assert metadata["length"] == str(
|
|
|
|
datetime.timedelta(seconds=metadata["length_seconds"])
|
|
|
|
)
|
|
|
|
|
2014-03-10 21:32:23 +01:00
|
|
|
|
|
|
|
def test_mp3_mono():
|
2021-05-27 16:23:02 +02:00
|
|
|
metadata = MetadataAnalyzer.analyze(
|
|
|
|
"tests/test_data/44100Hz-16bit-mono.mp3", dict()
|
|
|
|
)
|
2014-03-10 21:32:23 +01:00
|
|
|
check_default_metadata(metadata)
|
2021-05-27 16:23:02 +02:00
|
|
|
assert metadata["channels"] == 1
|
|
|
|
assert metadata["bit_rate"] == 63998
|
|
|
|
assert abs(metadata["length_seconds"] - 3.9) < 0.1
|
|
|
|
assert metadata["mime"] == "audio/mp3" # Not unicode because MIMEs aren't.
|
|
|
|
assert metadata["track_total"] == "10" # MP3s can have a track_total
|
|
|
|
# Mutagen doesn't extract comments from mp3s it seems
|
|
|
|
|
2014-03-10 21:32:23 +01:00
|
|
|
|
|
|
|
def test_mp3_jointstereo():
|
2021-05-27 16:23:02 +02:00
|
|
|
metadata = MetadataAnalyzer.analyze(
|
|
|
|
"tests/test_data/44100Hz-16bit-jointstereo.mp3", dict()
|
|
|
|
)
|
2014-03-10 21:32:23 +01:00
|
|
|
check_default_metadata(metadata)
|
2021-05-27 16:23:02 +02:00
|
|
|
assert metadata["channels"] == 2
|
|
|
|
assert metadata["bit_rate"] == 127998
|
|
|
|
assert abs(metadata["length_seconds"] - 3.9) < 0.1
|
|
|
|
assert metadata["mime"] == "audio/mp3"
|
|
|
|
assert metadata["track_total"] == "10" # MP3s can have a track_total
|
|
|
|
|
2014-03-10 21:32:23 +01:00
|
|
|
|
|
|
|
def test_mp3_simplestereo():
|
2021-05-27 16:23:02 +02:00
|
|
|
metadata = MetadataAnalyzer.analyze(
|
|
|
|
"tests/test_data/44100Hz-16bit-simplestereo.mp3", dict()
|
|
|
|
)
|
2014-03-10 21:32:23 +01:00
|
|
|
check_default_metadata(metadata)
|
2021-05-27 16:23:02 +02:00
|
|
|
assert metadata["channels"] == 2
|
|
|
|
assert metadata["bit_rate"] == 127998
|
|
|
|
assert abs(metadata["length_seconds"] - 3.9) < 0.1
|
|
|
|
assert metadata["mime"] == "audio/mp3"
|
|
|
|
assert metadata["track_total"] == "10" # MP3s can have a track_total
|
|
|
|
|
2014-03-10 21:32:23 +01:00
|
|
|
|
|
|
|
def test_mp3_dualmono():
|
2021-05-27 16:23:02 +02:00
|
|
|
metadata = MetadataAnalyzer.analyze(
|
|
|
|
"tests/test_data/44100Hz-16bit-dualmono.mp3", dict()
|
|
|
|
)
|
2014-03-10 21:32:23 +01:00
|
|
|
check_default_metadata(metadata)
|
2021-05-27 16:23:02 +02:00
|
|
|
assert metadata["channels"] == 2
|
|
|
|
assert metadata["bit_rate"] == 127998
|
|
|
|
assert abs(metadata["length_seconds"] - 3.9) < 0.1
|
|
|
|
assert metadata["mime"] == "audio/mp3"
|
|
|
|
assert metadata["track_total"] == "10" # MP3s can have a track_total
|
2014-03-10 21:32:23 +01:00
|
|
|
|
|
|
|
|
|
|
|
def test_ogg_mono():
|
2021-05-27 16:23:02 +02:00
|
|
|
metadata = MetadataAnalyzer.analyze(
|
|
|
|
"tests/test_data/44100Hz-16bit-mono.ogg", dict()
|
|
|
|
)
|
2014-03-10 21:32:23 +01:00
|
|
|
check_default_metadata(metadata)
|
2021-05-27 16:23:02 +02:00
|
|
|
assert metadata["channels"] == 1
|
|
|
|
assert metadata["bit_rate"] == 80000
|
|
|
|
assert abs(metadata["length_seconds"] - 3.8) < 0.1
|
|
|
|
assert metadata["mime"] == "audio/vorbis"
|
|
|
|
assert metadata["comment"] == "Test Comment"
|
|
|
|
|
2014-03-10 21:32:23 +01:00
|
|
|
|
|
|
|
def test_ogg_stereo():
|
2021-05-27 16:23:02 +02:00
|
|
|
metadata = MetadataAnalyzer.analyze(
|
|
|
|
"tests/test_data/44100Hz-16bit-stereo.ogg", dict()
|
|
|
|
)
|
2014-03-10 21:32:23 +01:00
|
|
|
check_default_metadata(metadata)
|
2021-05-27 16:23:02 +02:00
|
|
|
assert metadata["channels"] == 2
|
|
|
|
assert metadata["bit_rate"] == 112000
|
|
|
|
assert abs(metadata["length_seconds"] - 3.8) < 0.1
|
|
|
|
assert metadata["mime"] == "audio/vorbis"
|
|
|
|
assert metadata["comment"] == "Test Comment"
|
|
|
|
|
2014-03-10 21:32:23 +01:00
|
|
|
|
2021-05-27 16:23:02 +02:00
|
|
|
""" faac and avconv can't seem to create a proper mono AAC file... ugh
|
2014-03-10 21:32:23 +01:00
|
|
|
def test_aac_mono():
|
2020-01-21 08:13:42 +01:00
|
|
|
metadata = MetadataAnalyzer.analyze('tests/test_data/44100Hz-16bit-mono.m4a')
|
2020-01-16 14:33:44 +01:00
|
|
|
print("Mono AAC metadata:")
|
|
|
|
print(metadata)
|
2014-03-10 21:32:23 +01:00
|
|
|
check_default_metadata(metadata)
|
|
|
|
assert metadata['channels'] == 1
|
|
|
|
assert metadata['bit_rate'] == 80000
|
2014-04-16 22:42:37 +02:00
|
|
|
assert abs(metadata['length_seconds'] - 3.8) < 0.1
|
2014-04-24 00:20:50 +02:00
|
|
|
assert metadata['mime'] == 'audio/mp4'
|
2020-01-21 08:13:42 +01:00
|
|
|
assert metadata['comment'] == 'Test Comment'
|
2021-05-27 16:23:02 +02:00
|
|
|
"""
|
|
|
|
|
2014-03-10 21:32:23 +01:00
|
|
|
|
|
|
|
def test_aac_stereo():
|
2021-05-27 16:23:02 +02:00
|
|
|
metadata = MetadataAnalyzer.analyze(
|
|
|
|
"tests/test_data/44100Hz-16bit-stereo.m4a", dict()
|
|
|
|
)
|
2014-03-10 21:32:23 +01:00
|
|
|
check_default_metadata(metadata)
|
2021-05-27 16:23:02 +02:00
|
|
|
assert metadata["channels"] == 2
|
|
|
|
assert metadata["bit_rate"] == 102619
|
|
|
|
assert abs(metadata["length_seconds"] - 3.8) < 0.1
|
|
|
|
assert metadata["mime"] == "audio/mp4"
|
|
|
|
assert metadata["comment"] == "Test Comment"
|
|
|
|
|
2014-03-10 21:32:23 +01:00
|
|
|
|
|
|
|
def test_mp3_utf8():
|
2021-05-27 16:23:02 +02:00
|
|
|
metadata = MetadataAnalyzer.analyze(
|
|
|
|
"tests/test_data/44100Hz-16bit-stereo-utf8.mp3", dict()
|
|
|
|
)
|
2014-03-10 21:32:23 +01:00
|
|
|
# Using a bunch of different UTF-8 codepages here. Test data is from:
|
|
|
|
# http://winrus.com/utf8-jap.htm
|
2021-05-27 16:23:02 +02:00
|
|
|
assert metadata["track_title"] == "アイウエオカキクケコサシスセソタチツテ"
|
|
|
|
assert metadata["artist_name"] == "てすと"
|
|
|
|
assert metadata["album_title"] == "Ä ä Ü ü ß"
|
|
|
|
assert metadata["year"] == "1999"
|
|
|
|
assert metadata["genre"] == "Я Б Г Д Ж Й"
|
|
|
|
assert metadata["track_number"] == "1"
|
|
|
|
assert metadata["channels"] == 2
|
|
|
|
assert metadata["bit_rate"] < 130000
|
|
|
|
assert metadata["bit_rate"] > 127000
|
|
|
|
assert abs(metadata["length_seconds"] - 3.9) < 0.1
|
|
|
|
assert metadata["mime"] == "audio/mp3"
|
|
|
|
assert metadata["track_total"] == "10" # MP3s can have a track_total
|
|
|
|
|
2014-03-10 21:32:23 +01:00
|
|
|
|
2014-12-11 00:44:35 +01:00
|
|
|
def test_invalid_wma():
|
2021-05-27 16:23:02 +02:00
|
|
|
metadata = MetadataAnalyzer.analyze(
|
|
|
|
"tests/test_data/44100Hz-16bit-stereo-invalid.wma", dict()
|
|
|
|
)
|
|
|
|
assert metadata["mime"] == "audio/x-ms-wma"
|
|
|
|
|
2014-12-11 00:44:35 +01:00
|
|
|
|
|
|
|
def test_wav_stereo():
|
2021-05-27 16:23:02 +02:00
|
|
|
metadata = MetadataAnalyzer.analyze(
|
|
|
|
"tests/test_data/44100Hz-16bit-stereo.wav", dict()
|
|
|
|
)
|
|
|
|
assert metadata["mime"] == "audio/x-wav"
|
|
|
|
assert abs(metadata["length_seconds"] - 3.9) < 0.1
|
|
|
|
assert metadata["channels"] == 2
|
|
|
|
assert metadata["sample_rate"] == 44100
|
2014-12-11 00:44:35 +01:00
|
|
|
|
|
|
|
|
2014-03-26 20:06:48 +01:00
|
|
|
# Make sure the parameter checking works
|
2020-01-21 08:13:42 +01:00
|
|
|
@raises(FileNotFoundError)
|
2014-03-26 20:06:48 +01:00
|
|
|
def test_move_wrong_string_param1():
|
2021-05-27 16:23:02 +02:00
|
|
|
not_unicode = "asdfasdf"
|
2014-03-26 20:06:48 +01:00
|
|
|
MetadataAnalyzer.analyze(not_unicode, dict())
|
|
|
|
|
2021-05-27 16:23:02 +02:00
|
|
|
|
2014-03-26 20:06:48 +01:00
|
|
|
@raises(TypeError)
|
|
|
|
def test_move_wrong_metadata_dict():
|
|
|
|
not_a_dict = list()
|
2021-05-27 16:23:02 +02:00
|
|
|
MetadataAnalyzer.analyze("asdfasdf", not_a_dict)
|
|
|
|
|
2014-03-26 20:06:48 +01:00
|
|
|
|
|
|
|
# Test an mp3 file where the number of channels is invalid or missing:
|
|
|
|
def test_mp3_bad_channels():
|
2021-05-27 16:23:02 +02:00
|
|
|
filename = "tests/test_data/44100Hz-16bit-mono.mp3"
|
|
|
|
"""
|
2014-03-26 20:06:48 +01:00
|
|
|
It'd be a pain in the ass to construct a real MP3 with an invalid number
|
|
|
|
of channels by hand because that value is stored in every MP3 frame in the file
|
2021-05-27 16:23:02 +02:00
|
|
|
"""
|
2014-03-26 20:06:48 +01:00
|
|
|
audio_file = mutagen.File(filename, easy=True)
|
|
|
|
audio_file.info.mode = 1777
|
2021-05-27 16:23:02 +02:00
|
|
|
with mock.patch("airtime_analyzer.metadata_analyzer.mutagen") as mock_mutagen:
|
2014-03-26 20:06:48 +01:00
|
|
|
mock_mutagen.File.return_value = audio_file
|
2021-05-27 16:23:02 +02:00
|
|
|
# mock_mutagen.side_effect = lambda *args, **kw: audio_file #File(*args, **kw)
|
2014-03-26 20:06:48 +01:00
|
|
|
|
|
|
|
metadata = MetadataAnalyzer.analyze(filename, dict())
|
|
|
|
check_default_metadata(metadata)
|
2021-05-27 16:23:02 +02:00
|
|
|
assert metadata["channels"] == 1
|
|
|
|
assert metadata["bit_rate"] == 63998
|
|
|
|
assert abs(metadata["length_seconds"] - 3.9) < 0.1
|
|
|
|
assert metadata["mime"] == "audio/mp3" # Not unicode because MIMEs aren't.
|
|
|
|
assert metadata["track_total"] == "10" # MP3s can have a track_total
|
|
|
|
# Mutagen doesn't extract comments from mp3s it seems
|
|
|
|
|
2014-04-16 22:42:37 +02:00
|
|
|
|
|
|
|
def test_unparsable_file():
|
2021-05-27 16:23:02 +02:00
|
|
|
MetadataAnalyzer.analyze("tests/test_data/unparsable.txt", dict())
|