mirror of
https://github.com/beetbox/beets.git
synced 2025-12-06 08:39:17 +01:00
Add autobpm plugin
This plugin uses librosa to automatically calculate the BPM for a track. It is based on the keyfinder plugin, and rounds the BPM to an int. Co-authored-by: Adrian Sampson <adrian@radbox.org>
This commit is contained in:
parent
7736ae7634
commit
7961cf3aaa
5 changed files with 117 additions and 3 deletions
80
beetsplug/autobpm.py
Normal file
80
beetsplug/autobpm.py
Normal file
|
|
@ -0,0 +1,80 @@
|
||||||
|
# This file is part of beets.
|
||||||
|
#
|
||||||
|
# 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.
|
||||||
|
|
||||||
|
"""Uses Librosa to calculate the `bpm` field.
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
from beets import ui
|
||||||
|
from beets import util
|
||||||
|
from beets.plugins import BeetsPlugin
|
||||||
|
|
||||||
|
from librosa import load, beat
|
||||||
|
from soundfile import LibsndfileError
|
||||||
|
|
||||||
|
|
||||||
|
class AutoBPMPlugin(BeetsPlugin):
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
super().__init__()
|
||||||
|
self.config.add({
|
||||||
|
'auto': True,
|
||||||
|
'overwrite': False,
|
||||||
|
})
|
||||||
|
|
||||||
|
if self.config['auto'].get(bool):
|
||||||
|
self.import_stages = [self.imported]
|
||||||
|
|
||||||
|
def commands(self):
|
||||||
|
cmd = ui.Subcommand('autobpm',
|
||||||
|
help='detect and add bpm from audio using Librosa')
|
||||||
|
cmd.func = self.command
|
||||||
|
return [cmd]
|
||||||
|
|
||||||
|
def command(self, lib, opts, args):
|
||||||
|
self.calculate_bpm(lib.items(ui.decargs(args)),
|
||||||
|
write=ui.should_write())
|
||||||
|
|
||||||
|
def imported(self, session, task):
|
||||||
|
self.calculate_bpm(task.imported_items())
|
||||||
|
|
||||||
|
def calculate_bpm(self, items, write=False):
|
||||||
|
overwrite = self.config['overwrite'].get(bool)
|
||||||
|
|
||||||
|
for item in items:
|
||||||
|
if item['bpm']:
|
||||||
|
self._log.info('found bpm {0} for {1}',
|
||||||
|
item['bpm'], util.displayable_path(item.path))
|
||||||
|
if not overwrite:
|
||||||
|
continue
|
||||||
|
|
||||||
|
try:
|
||||||
|
y, sr = load(util.syspath(item.path), res_type='kaiser_fast')
|
||||||
|
except LibsndfileError as exc:
|
||||||
|
self._log.error('LibsndfileError: failed to load {0} {1}',
|
||||||
|
util.displayable_path(item.path), exc)
|
||||||
|
continue
|
||||||
|
except ValueError as exc:
|
||||||
|
self._log.error('ValueError: failed to load {0} {1}',
|
||||||
|
util.displayable_path(item.path), exc)
|
||||||
|
continue
|
||||||
|
|
||||||
|
tempo, _ = beat.beat_track(y=y, sr=sr)
|
||||||
|
bpm = round(tempo)
|
||||||
|
item['bpm'] = bpm
|
||||||
|
self._log.info('added computed bpm {0} for {1}',
|
||||||
|
bpm, util.displayable_path(item.path))
|
||||||
|
|
||||||
|
if write:
|
||||||
|
item.try_write()
|
||||||
|
item.store()
|
||||||
|
|
@ -122,6 +122,9 @@ New features:
|
||||||
:bug:`2786`
|
:bug:`2786`
|
||||||
* Add support for ``artists`` and ``albumartists`` multi-valued tags.
|
* Add support for ``artists`` and ``albumartists`` multi-valued tags.
|
||||||
:bug:`505`
|
:bug:`505`
|
||||||
|
* :doc:`/plugins/autobpm`: Add the `autobpm` plugin which uses Librosa to
|
||||||
|
calculate the BPM of the audio.
|
||||||
|
:bug:`3856`
|
||||||
|
|
||||||
Bug fixes:
|
Bug fixes:
|
||||||
|
|
||||||
|
|
|
||||||
25
docs/plugins/autobpm.rst
Normal file
25
docs/plugins/autobpm.rst
Normal file
|
|
@ -0,0 +1,25 @@
|
||||||
|
AutoBPM Plugin
|
||||||
|
==============
|
||||||
|
|
||||||
|
The `autobpm` plugin uses the `Librosa`_ library to calculate the BPM
|
||||||
|
of a track from its audio data and store it in the `bpm` field of your
|
||||||
|
database. It does so automatically when importing music or through
|
||||||
|
the ``beet autobpm [QUERY]`` command.
|
||||||
|
|
||||||
|
To use the ``autobpm`` plugin, enable it in your configuration (see
|
||||||
|
:ref:`using-plugins`).
|
||||||
|
|
||||||
|
Configuration
|
||||||
|
-------------
|
||||||
|
|
||||||
|
To configure the plugin, make a ``autobpm:`` section in your
|
||||||
|
configuration file. The available options are:
|
||||||
|
|
||||||
|
- **auto**: Analyze every file on import.
|
||||||
|
Otherwise, you need to use the ``beet autobpm`` command explicitly.
|
||||||
|
Default: ``yes``
|
||||||
|
- **overwrite**: Calculate a BPM even for files that already have a
|
||||||
|
`bpm` value.
|
||||||
|
Default: ``no``.
|
||||||
|
|
||||||
|
.. _Librosa: https://github.com/librosa/librosa/
|
||||||
|
|
@ -63,6 +63,7 @@ following to your configuration::
|
||||||
acousticbrainz
|
acousticbrainz
|
||||||
albumtypes
|
albumtypes
|
||||||
aura
|
aura
|
||||||
|
autobpm
|
||||||
badfiles
|
badfiles
|
||||||
bareasc
|
bareasc
|
||||||
beatport
|
beatport
|
||||||
|
|
@ -164,6 +165,9 @@ Metadata
|
||||||
:doc:`acousticbrainz <acousticbrainz>`
|
:doc:`acousticbrainz <acousticbrainz>`
|
||||||
Fetch various AcousticBrainz metadata
|
Fetch various AcousticBrainz metadata
|
||||||
|
|
||||||
|
:doc:`autobpm <autobpm>`
|
||||||
|
Use `Librosa`_ to calculate the BPM from the audio.
|
||||||
|
|
||||||
:doc:`bpm <bpm>`
|
:doc:`bpm <bpm>`
|
||||||
Measure tempo using keystrokes.
|
Measure tempo using keystrokes.
|
||||||
|
|
||||||
|
|
@ -222,6 +226,7 @@ Metadata
|
||||||
:doc:`zero <zero>`
|
:doc:`zero <zero>`
|
||||||
Nullify fields by pattern or unconditionally.
|
Nullify fields by pattern or unconditionally.
|
||||||
|
|
||||||
|
.. _Librosa: https://github.com/librosa/librosa/
|
||||||
.. _KeyFinder: http://www.ibrahimshaath.co.uk/keyfinder/
|
.. _KeyFinder: http://www.ibrahimshaath.co.uk/keyfinder/
|
||||||
.. _streaming_extractor_music: https://acousticbrainz.org/download
|
.. _streaming_extractor_music: https://acousticbrainz.org/download
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -32,6 +32,7 @@ per-file-ignores =
|
||||||
./extra/release.py:D
|
./extra/release.py:D
|
||||||
./beetsplug/duplicates.py:D
|
./beetsplug/duplicates.py:D
|
||||||
./beetsplug/bpm.py:D
|
./beetsplug/bpm.py:D
|
||||||
|
./beetsplug/autobpm.py:D
|
||||||
./beetsplug/convert.py:D
|
./beetsplug/convert.py:D
|
||||||
./beetsplug/info.py:D
|
./beetsplug/info.py:D
|
||||||
./beetsplug/parentwork.py:D
|
./beetsplug/parentwork.py:D
|
||||||
|
|
|
||||||
Loading…
Reference in a new issue