Skip to content

Commit 8f26241

Browse files
committed
Implement multi-tag genres field
1 parent 80bc539 commit 8f26241

File tree

4 files changed

+35
-4
lines changed

4 files changed

+35
-4
lines changed

beets/autotag/hooks.py

+4
Original file line numberDiff line numberDiff line change
@@ -100,6 +100,7 @@ def __init__(
100100
country: str | None = None,
101101
style: str | None = None,
102102
genre: str | None = None,
103+
genres: str | None = None,
103104
albumstatus: str | None = None,
104105
media: str | None = None,
105106
albumdisambig: str | None = None,
@@ -143,6 +144,7 @@ def __init__(
143144
self.country = country
144145
self.style = style
145146
self.genre = genre
147+
self.genres = genres
146148
self.albumstatus = albumstatus
147149
self.media = media
148150
self.albumdisambig = albumdisambig
@@ -212,6 +214,7 @@ def __init__(
212214
bpm: str | None = None,
213215
initial_key: str | None = None,
214216
genre: str | None = None,
217+
genres: str | None = None,
215218
album: str | None = None,
216219
**kwargs,
217220
):
@@ -246,6 +249,7 @@ def __init__(
246249
self.bpm = bpm
247250
self.initial_key = initial_key
248251
self.genre = genre
252+
self.genres = genres
249253
self.album = album
250254
self.update(kwargs)
251255

beets/library.py

+3
Original file line numberDiff line numberDiff line change
@@ -534,6 +534,7 @@ class Item(LibModel):
534534
"albumartist_credit": types.STRING,
535535
"albumartists_credit": types.MULTI_VALUE_DSV,
536536
"genre": types.STRING,
537+
"genres": types.MULTI_VALUE_DSV,
537538
"style": types.STRING,
538539
"discogs_albumid": types.INTEGER,
539540
"discogs_artistid": types.INTEGER,
@@ -1182,6 +1183,7 @@ class Album(LibModel):
11821183
"albumartists_credit": types.MULTI_VALUE_DSV,
11831184
"album": types.STRING,
11841185
"genre": types.STRING,
1186+
"genres": types.MULTI_VALUE_DSV,
11851187
"style": types.STRING,
11861188
"discogs_albumid": types.INTEGER,
11871189
"discogs_artistid": types.INTEGER,
@@ -1238,6 +1240,7 @@ class Album(LibModel):
12381240
"albumartists_credit",
12391241
"album",
12401242
"genre",
1243+
"genres",
12411244
"style",
12421245
"discogs_albumid",
12431246
"discogs_artistid",

docs/changelog.rst

+2
Original file line numberDiff line numberDiff line change
@@ -118,6 +118,8 @@ New features:
118118
* Beets now uses ``platformdirs`` to determine the default music directory.
119119
This location varies between systems -- for example, users can configure it
120120
on Unix systems via ``user-dirs.dirs(5)``.
121+
* New multi-valued ``genres`` tag. This change brings up the ``genres`` tag to the same state as the ``*artists*`` multi-valued tags (see :bug:`4743` for details).
122+
:bug:`5426`
121123

122124
Bug fixes:
123125

test/test_library.py

+26-4
Original file line numberDiff line numberDiff line change
@@ -710,13 +710,13 @@ def test_if_def_false_complete(self):
710710
self._assert_dest(b"/base/not_played")
711711

712712
def test_first(self):
713-
self.i.genres = "Pop; Rock; Classical Crossover"
714-
self._setf("%first{$genres}")
713+
self.i.semicolon_delimited_field = "Pop; Rock; Classical Crossover"
714+
self._setf("%first{$semicolon_delimited_field}")
715715
self._assert_dest(b"/base/Pop")
716716

717717
def test_first_skip(self):
718-
self.i.genres = "Pop; Rock; Classical Crossover"
719-
self._setf("%first{$genres,1,2}")
718+
self.i.semicolon_delimited_field = "Pop; Rock; Classical Crossover"
719+
self._setf("%first{$semicolon_delimited_field,1,2}")
720720
self._assert_dest(b"/base/Classical Crossover")
721721

722722
def test_first_different_sep(self):
@@ -1308,6 +1308,28 @@ def test_write_date_field(self):
13081308
item.write()
13091309
assert MediaFile(syspath(item.path)).year == clean_year
13101310

1311+
def test_write_multi_genres(self):
1312+
item = self.add_item_fixture(genre="old genre")
1313+
item.write(
1314+
tags={"genres": ["g1", "g2"]},
1315+
)
1316+
1317+
# Ensure it reads all genres
1318+
assert MediaFile(syspath(item.path)).genres == ["g1", "g2"]
1319+
1320+
# Ensure reading single genre outputs the first of the genres
1321+
assert MediaFile(syspath(item.path)).genre == "g1"
1322+
1323+
def test_write_multi_genres_both_single_and_multi(self):
1324+
item = self.add_item_fixture(genre="old genre 1")
1325+
item.write(
1326+
tags={"genre": "single genre", "genres": ["multi genre"]},
1327+
)
1328+
1329+
# Ensure the multi takes precedence
1330+
assert MediaFile(syspath(item.path)).genre == "multi genre"
1331+
assert MediaFile(syspath(item.path)).genres == ["multi genre"]
1332+
13111333

13121334
class ItemReadTest(unittest.TestCase):
13131335
def test_unreadable_raise_read_error(self):

0 commit comments

Comments
 (0)