Browse Source

added some tests (mainly for symuid.Track)

Fabian Peter Hammerle 5 years ago
parent
commit
27e23534b2

+ 17 - 0
symuid/__init__.py

@@ -13,6 +13,10 @@ def _timestamp_to_utc_dt(ts):
         .replace(tzinfo=dt.timezone.utc)
 
 
+def _utc_dt_now():
+    return dt.datetime.utcnow().replace(tzinfo=dt.timezone.utc)
+
+
 class PlayCount:
 
     def __init__(self, player, library_id, register_dt, count):
@@ -24,6 +28,18 @@ class PlayCount:
         assert isinstance(count, int) and count >= 0, count
         self.count = count
 
+    def __eq__(self, other):
+        return type(self) == type(other) and vars(self) == vars(other)
+
+    def __hash__(self):
+        attrs_sorted = sorted(vars(self).items(), key=lambda p: p[0]) 
+        return hash(tuple(v for k, v in attrs_sorted))
+
+    def __repr__(self):
+        return 'PlayCount({})'.format(', '.join(
+            '{}={!r}'.format(k, v) for k, v in vars(self).items(),
+        ))
+
 
 class Track:
 
@@ -31,6 +47,7 @@ class Track:
 
     def __init__(self, path):
         mutagen_file = mutagen.File(filename=path)
+        # TODO support mp3 files without ID3 headers
         if mutagen_file is None:
             raise NotImplementedError(path)
         elif isinstance(mutagen_file.tags, mutagen.id3.ID3):

+ 0 - 1
symuid/tag_interface.py

@@ -21,7 +21,6 @@ class ID3(_mutagen):
     _UFID_OWNER_ID = 'symuid'
 
     def __init__(self, mutagen_file):
-        assert mutagen_file.tags, mutagen_file
         assert isinstance(mutagen_file.tags, mutagen.id3.ID3), \
             mutagen_file.tags
         self._mutagen_file = mutagen_file

+ 8 - 0
tests/test_functions.py

@@ -3,9 +3,17 @@ import pytest
 import symuid
 
 import datetime as dt
+import pytz
+
 
 @pytest.mark.parametrize(('ts', 'expected_dt'), [
     (1528795204, dt.datetime(2018, 6, 12, 9, 20, 4, tzinfo=dt.timezone.utc)),
 ])
 def test__timestamp_to_utc_dt(ts, expected_dt):
     assert expected_dt == symuid._timestamp_to_utc_dt(ts)
+
+
+def test__utc_dt_now():
+    dt = symuid._utc_dt_now()
+    pytz_dt = dt.now(pytz.timezone('UTC'))
+    assert abs((dt - pytz_dt).total_seconds()) < 5

+ 153 - 0
tests/test_track.py

@@ -0,0 +1,153 @@
+import pytest
+
+import symuid
+
+import datetime as dt
+import mutagen
+import os
+import shutil
+
+# TODO test aac / m4a itunes tags
+
+TRACKS_DIR_PATH = os.path.join(os.path.dirname(__file__), 'tracks')
+
+
+def utc_dt(h=0):
+    return dt.datetime(2018, 9, 26, h, tzinfo=dt.timezone.utc)
+
+
+@pytest.fixture
+def empty_id3_track(tmpdir):
+    p = tmpdir.join('empty.mp3').strpath
+    shutil.copyfile(
+        src=os.path.join(TRACKS_DIR_PATH, 'id3v2.4-empty.mp3'),
+        dst=p,
+    )
+    return symuid.Track(p)
+
+
+@pytest.mark.parametrize('path', [
+    os.path.join(TRACKS_DIR_PATH, 'id3v2.4-typical.mp3'),
+    os.path.join(TRACKS_DIR_PATH, 'id3v2.4-empty.mp3'),
+])
+def test_init(path):
+    symuid.Track(path)
+
+
+@pytest.mark.parametrize(('player', 'library_id', 'register_dt', 'count'), [
+    ('pytest', 'lib', dt.datetime(2018, 9, 26, 21, 16, 20, tzinfo=dt.timezone.utc), 7),
+])
+def test_register_play_count(empty_id3_track, player, library_id, register_dt, count):
+    empty_id3_track.register_play_count(
+        player=player,
+        library_id=library_id,
+        register_dt=register_dt,
+        play_count=count,
+    )
+    tags = mutagen.File(empty_id3_track.path).tags
+    assert len(tags) == 1
+    expected_desc = 'symuid:pcnt:{}:{}:{}'.format(
+        player, library_id, int(register_dt.timestamp()),
+    )
+    tag = tags['TXXX:' + expected_desc]
+    assert tag.desc == expected_desc
+    assert tag.text == [str(count)]
+
+
+@pytest.mark.parametrize(('player', 'library_id', 'register_dt', 'count'), [
+    ('pytest', 'lib', utc_dt(), 7),
+])
+def test_register_play_count(empty_id3_track, player, library_id, register_dt, count):
+    empty_id3_track.register_play_count(
+        player=player,
+        library_id=library_id,
+        register_dt=register_dt,
+        play_count=count,
+    )
+    tags = mutagen.File(empty_id3_track.path).tags
+    assert len(tags) == 1
+    expected_desc = 'symuid:pcnt:{}:{}:{}'.format(
+        player, library_id, int(register_dt.timestamp()),
+    )
+    tag = tags['TXXX:' + expected_desc]
+    assert tag.desc == expected_desc
+    assert tag.text == [str(count)]
+
+
+@pytest.mark.parametrize(('expected_counts'), [
+    [],
+    [symuid.PlayCount('player', 'lib', utc_dt(), 3)],
+    [symuid.PlayCount('a', '0', utc_dt(0), 1),
+     symuid.PlayCount('b', '1', utc_dt(1), 2)],
+    [symuid.PlayCount('a', '0', utc_dt(0), 1),
+     symuid.PlayCount('a', '2', utc_dt(1), 2),
+     symuid.PlayCount('b', '3', utc_dt(3), 3)],
+])
+def test__get_play_counts_all(empty_id3_track, expected_counts):
+    for pc in expected_counts:
+        empty_id3_track.register_play_count(
+            player=pc.player,
+            library_id=pc.library_id,
+            register_dt=pc.register_dt,
+            play_count=pc.count,
+        )
+    assert expected_counts == list(empty_id3_track._get_play_counts())
+
+
+def test__get_play_counts_filtered(empty_id3_track):
+    counts = [symuid.PlayCount('a', '0', utc_dt(0), 1),
+              symuid.PlayCount('a', '0', utc_dt(1), 2),
+              symuid.PlayCount('a', '1', utc_dt(0), 3),
+              symuid.PlayCount('b', '2', utc_dt(1), 4)]
+    for pc in counts:
+        empty_id3_track.register_play_count(
+            player=pc.player,
+            library_id=pc.library_id,
+            register_dt=pc.register_dt,
+            play_count=pc.count,
+        )
+    assert set(empty_id3_track._get_play_counts(player='a')) \
+        == set(filter(lambda pc: pc.player == 'a', counts))
+    assert set(empty_id3_track._get_play_counts(player='b')) \
+        == set(filter(lambda pc: pc.player == 'b', counts))
+    assert set(empty_id3_track._get_play_counts(player='a', library_id='0')) \
+        == set(filter(lambda pc: pc.library_id == '0', counts))
+    assert set(empty_id3_track._get_play_counts(player='a', library_id='2')) \
+        == set()
+
+
+def test__get_latest_play_counts(empty_id3_track):
+    counts = [symuid.PlayCount('a', '0', utc_dt(0), 1),
+              symuid.PlayCount('a', '0', utc_dt(1), 2),
+              symuid.PlayCount('a', '1', utc_dt(0), 3),
+              symuid.PlayCount('a', '1', utc_dt(2), 4),
+              symuid.PlayCount('b', '2', utc_dt(3), 5)]
+    for pc in counts:
+        empty_id3_track.register_play_count(
+            player=pc.player,
+            library_id=pc.library_id,
+            register_dt=pc.register_dt,
+            play_count=pc.count,
+        )
+    assert set(empty_id3_track._get_latest_play_counts()) \
+        == set([counts[1], counts[3], counts[4]])
+    assert set(empty_id3_track._get_latest_play_counts(player='a')) \
+        == set([counts[1], counts[3]])
+    assert set(empty_id3_track._get_latest_play_counts(player='a', library_id='0')) \
+        == set([counts[1]])
+
+
+def test_get_play_count_sum(empty_id3_track):
+    counts = [symuid.PlayCount('a', '0', utc_dt(0), 1),
+              symuid.PlayCount('a', '0', utc_dt(1), 2),
+              symuid.PlayCount('a', '1', utc_dt(0), 3),
+              symuid.PlayCount('a', '1', utc_dt(2), 4),
+              symuid.PlayCount('b', '2', utc_dt(3), 5)]
+    for pc in counts:
+        empty_id3_track.register_play_count(
+            player=pc.player,
+            library_id=pc.library_id,
+            register_dt=pc.register_dt,
+            play_count=pc.count,
+        )
+    assert 2 + 4 + 5 == empty_id3_track.get_play_count_sum()

BIN
tests/tracks/id3v2.4-empty.mp3


BIN
tests/tracks/id3v2.4-typical.mp3


BIN
tests/tracks/no-tags.mp3