Skip to content

Commit

Permalink
Merge pull request #1386 from pprkut/sync
Browse files Browse the repository at this point in the history
Plugin to sync metadata from other applications.
  • Loading branch information
sampsyo committed Apr 18, 2015
2 parents e14b1d7 + 0ecc560 commit 59b2e41
Show file tree
Hide file tree
Showing 5 changed files with 218 additions and 0 deletions.
88 changes: 88 additions & 0 deletions beetsplug/metasync/__init__.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,88 @@
# This file is part of beets.
# Copyright 2015, Heinz Wiesinger.
#
# 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.

"""Synchronize information from music player libraries
"""

from beets import ui, logging
from beets.plugins import BeetsPlugin
from beets.dbcore import types
from beets.library import DateType
from sys import modules
import inspect

# Loggers.
log = logging.getLogger('beets.metasync')


class MetaSyncPlugin(BeetsPlugin):

item_types = {
'amarok_rating': types.INTEGER,
'amarok_score': types.FLOAT,
'amarok_uid': types.STRING,
'amarok_playcount': types.INTEGER,
'amarok_firstplayed': DateType(),
'amarok_lastplayed': DateType()
}

def __init__(self):
super(MetaSyncPlugin, self).__init__()

def commands(self):
cmd = ui.Subcommand('metasync',
help='update metadata from music player libraries')
cmd.parser.add_option('-p', '--pretend', action='store_true',
help='show all changes but do nothing')
cmd.parser.add_option('-s', '--source', action='store_false',
default=self.config['source'].as_str_seq(),
help="select specific sources to import from")
cmd.parser.add_format_option()
cmd.func = self.func
return [cmd]

def func(self, lib, opts, args):
"""Command handler for the metasync function.
"""
pretend = opts.pretend
source = opts.source
query = ui.decargs(args)

sources = {}

for player in source:
__import__('beetsplug.metasync', fromlist=[str(player)])

module = 'beetsplug.metasync.' + player

if module not in modules.keys():
log.error(u'Unknown metadata source \'' + player + '\'')
continue

classes = inspect.getmembers(modules[module], inspect.isclass)

for entry in classes:
if entry[0].lower() == player:
sources[player] = entry[1]()
else:
continue

for item in lib.items(query):
for player in sources.values():
player.get_data(item)

changed = ui.show_model_changes(item)

if changed and not pretend:
item.store()
81 changes: 81 additions & 0 deletions beetsplug/metasync/amarok.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,81 @@
# This file is part of beets.
# Copyright 2015, Heinz Wiesinger.
#
# 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.

"""Synchronize information from amarok's library via dbus
"""

from os.path import basename
from datetime import datetime
from time import mktime
from beets.util import displayable_path
from xml.sax.saxutils import escape
import dbus


class Amarok(object):

queryXML = u'<query version="1.0"> \
<filters> \
<and><include field="filename" value="%s" /></and> \
</filters> \
</query>'

def __init__(self):
self.collection = \
dbus.SessionBus().get_object('org.kde.amarok', '/Collection')

def get_data(self, item):
path = displayable_path(item.path)

# amarok unfortunately doesn't allow searching for the full path, only
# for the patch relative to the mount point. But the full path is part
# of the result set. So query for the filename and then try to match
# the correct item from the results we get back
results = self.collection.Query(self.queryXML % escape(basename(path)))
for result in results:
if result['xesam:url'] != path:
continue

item.amarok_rating = result['xesam:userRating']
item.amarok_score = result['xesam:autoRating']
item.amarok_playcount = result['xesam:useCount']
item.amarok_uid = \
result['xesam:id'].replace('amarok-sqltrackuid://', '')

if result['xesam:firstUsed'][0][0] != 0:
# These dates are stored as timestamps in amarok's db, but
# exposed over dbus as fixed integers in the current timezone.
first_played = datetime(
result['xesam:firstUsed'][0][0],
result['xesam:firstUsed'][0][1],
result['xesam:firstUsed'][0][2],
result['xesam:firstUsed'][1][0],
result['xesam:firstUsed'][1][1],
result['xesam:firstUsed'][1][2]
)

if result['xesam:lastUsed'][0][0] != 0:
last_played = datetime(
result['xesam:lastUsed'][0][0],
result['xesam:lastUsed'][0][1],
result['xesam:lastUsed'][0][2],
result['xesam:lastUsed'][1][0],
result['xesam:lastUsed'][1][1],
result['xesam:lastUsed'][1][2]
)
else:
last_played = first_played

item.amarok_firstplayed = mktime(first_played.timetuple())
item.amarok_lastplayed = mktime(last_played.timetuple())
2 changes: 2 additions & 0 deletions docs/plugins/index.rst
Original file line number Diff line number Diff line change
Expand Up @@ -56,6 +56,7 @@ Each plugin has its own set of options that can be defined in a section bearing
lyrics
mbcollection
mbsync
metasync
missing
mpdstats
mpdupdate
Expand Down Expand Up @@ -104,6 +105,7 @@ Metadata
* :doc:`lastimport`: Collect play counts from Last.fm.
* :doc:`lyrics`: Automatically fetch song lyrics.
* :doc:`mbsync`: Fetch updated metadata from MusicBrainz
* :doc:`metasync`: Fetch metadata from local or remote sources
* :doc:`mpdstats`: Connect to `MPD`_ and update the beets library with play
statistics (last_played, play_count, skip_count, rating).
* :doc:`replaygain`: Calculate volume normalization for players that support it.
Expand Down
45 changes: 45 additions & 0 deletions docs/plugins/metasync.rst
Original file line number Diff line number Diff line change
@@ -0,0 +1,45 @@
MetaSync Plugin
===============

This plugin provides the ``metasync`` command, which lets you fetch certain
metadata from other local or remote sources, for example your favorite audio
player.

Currently we support the following list of metadata sources:
- **amarok**: This syncs rating, score, first played, last played, playcount and uid from amarok.


Installing Dependencies
-----------------------

Fetching metadata from amarok requires the dbus-python library.

There are packages for most major linux distributions, or you can download the
library from its _website.

_website: http://dbus.freedesktop.org/releases/dbus-python/


Configuration
-------------

To configure the plugin, make a ``metasync:`` section in your configuration
file. The available options are:

- **source**: A list of sources to fetch metadata from.
Default: empty


Usage
-----

Enable the ``metasync`` plugin in your configuration (see
:ref:`using-plugins`) then run ``beet metasync QUERY`` to fetch updated
metadata from the configured list of sources.

The command has a few command-line options:

* To preview the changes that would be made without applying them, use the
``-p`` (``--pretend``) flag.
* To specify a temporary source to fetch metadata from, use the ``-s``
(``--source``) flag.
2 changes: 2 additions & 0 deletions setup.py
Original file line number Diff line number Diff line change
Expand Up @@ -76,6 +76,7 @@ def build_manpages():
'beetsplug.bpd',
'beetsplug.web',
'beetsplug.lastgenre',
'beetsplug.metasync',
],
entry_points={
'console_scripts': [
Expand Down Expand Up @@ -117,6 +118,7 @@ def build_manpages():
'web': ['flask', 'flask-cors'],
'import': ['rarfile'],
'thumbnails': ['pathlib', 'pyxdg'],
'metasync': ['dbus-python'],
},
# Non-Python/non-PyPI plugin dependencies:
# replaygain: mp3gain || aacgain
Expand Down

0 comments on commit 59b2e41

Please sign in to comment.