/usr/lib/rhythmbox/plugins/rbzeitgeist/rbzeitgeist.py is in rhythmbox-plugin-zeitgeist 3.0.2-0ubuntu1.
This file is owned by root:root, with mode 0o644.
The actual contents of the file can be viewed below.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 | # -.- coding: utf-8 -.-
# Copyright © 2009 Markus Korn <thekorn@gmx.de>
# Copyright © 2010 Laszlo Pandy <laszlok2@gmail.com>
# Copyright © 2011 Michal Hruby <michal.mhr@gmail.com>
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2, or (at your option)
# any later version.
#
# The Rhythmbox authors hereby grant permission for non-GPL compatible
# GStreamer plugins to be used and distributed together with GStreamer
# and Rhythmbox. This permission is above and beyond the permissions granted
# by the GPL license by which Rhythmbox is covered. If you modify this code
# you may extend this exception to your version of the code, but you are not
# obligated to do so. If you do not wish to do so, delete this exception
# statement from your version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
import time
import rb
from gi.repository import GObject, Gio, GLib, Peas
from gi.repository import RB
from zeitgeist.client import ZeitgeistClient
from zeitgeist.datamodel import Event, Subject, Interpretation, Manifestation
try:
IFACE = ZeitgeistClient()
except RuntimeError as e:
print("Unable to connect to Zeitgeist, won't send events. Reason: '%s'" % e)
IFACE = None
class ZeitgeistPlugin(GObject.Object, Peas.Activatable):
__gtype_name__ = 'ZeitgeistPlugin'
object = GObject.property(type=GObject.Object)
def __init__(self):
GObject.Object.__init__(self)
def do_activate(self):
print("Loading Zeitgeist plugin...")
if IFACE is not None:
shell = self.object
shell_player = shell.props.shell_player
self.__psc_id = shell_player.connect("playing-song-changed", self.playing_song_changed)
backend_player = shell_player.props.player
self.__eos_id = backend_player.connect("eos", self.on_backend_eos)
self.__manual_switch = True
self.__current_song = None
if IFACE.get_version() >= [0, 3, 2, 999]:
IFACE.register_data_source("org.gnome.Rhythmbox,dataprovider", "Rhythmbox", "Play and organize your music collection",
[Event.new_for_values(actor="application://rhythmbox.desktop")])
@staticmethod
def get_song_info(db, entry):
# we don't want the PROP_MEDIA_TYPE, as it doesn't contain mimetype
# of the audio file itself
song = {
"album": entry.get_string(RB.RhythmDBPropType.ALBUM),
"artist": entry.get_string(RB.RhythmDBPropType.ARTIST),
"title": entry.get_string(RB.RhythmDBPropType.TITLE),
"location": entry.get_playback_uri(),
}
return song
def on_backend_eos(self, backend_player, stream_data, eos_early):
# EOS signal means that the song changed because the song is over.
# ie. the user did not explicitly change the song.
self.__manual_switch = False
def playing_song_changed(self, shell, entry):
if self.__current_song is not None:
self.send_to_zeitgeist_async(self.__current_song, Interpretation.LEAVE_EVENT)
if entry is not None:
self.send_to_zeitgeist_async(entry, Interpretation.ACCESS_EVENT)
self.__current_song = entry
GLib.idle_add(self.reset_manual_switch)
def reset_manual_switch(self):
"""
After the eos signal has fired, and after the zeitgeist events have
been sent asynchronously, reset the manual_switch variable.
"""
self.__manual_switch = True
def send_to_zeitgeist_async(self, entry, event_type):
"""
We do async here because the "eos" signal is fired
*after* the "playing-song-changed" signal.
We don't know if the song change was manual or automatic
until we get get the eos signal. If the mainloop goes to
idle, it means there are no more signals scheduled, so we
will have already received the eos if it was coming.
"""
shell = self.object
db = shell.props.db
GLib.idle_add(self.send_to_zeitgeist, db, entry, event_type)
def send_to_zeitgeist(self, db, entry, event_type):
song = self.get_song_info(db, entry)
if self.__manual_switch:
manifest = Manifestation.USER_ACTIVITY
else:
manifest = Manifestation.SCHEDULED_ACTIVITY
def file_info_complete(obj, res, user_data):
try:
fi = obj.query_info_finish(res)
except:
return
uri_mimetype = fi.get_content_type()
subject = Subject.new_for_values(
uri=song["location"],
interpretation=unicode(Interpretation.AUDIO),
manifestation=unicode(Manifestation.FILE_DATA_OBJECT),
origin=song["location"].rpartition("/")[0],
mimetype=uri_mimetype,
text=" - ".join([song["title"], song["artist"], song["album"]])
)
event = Event.new_for_values(
timestamp=int(time.time()*1000),
interpretation=unicode(event_type),
manifestation=unicode(manifest),
actor="application://rhythmbox.desktop",
subjects=[subject,]
)
IFACE.insert_event(event)
f = Gio.file_new_for_uri(song["location"])
f.query_info_async(Gio.FILE_ATTRIBUTE_STANDARD_CONTENT_TYPE, Gio.FileQueryInfoFlags.NONE, GLib.PRIORITY_DEFAULT, None, file_info_complete, None)
def do_deactivate(self):
print("Deactivating Zeitgeist plugin...")
if IFACE is not None:
shell = self.object
shell_player = shell.props.shell_player
shell_player.disconnect(self.__psc_id)
self.__psc_id = None
backend_player = shell_player.props.player
backend_player.disconnect(self.__eos_id)
self.__eos_id = None
self.__current_song = None
|