# Copyright (C) 2006-2007 Red Hat, Inc. # Copyright (C) 2007 One Laptop Per Child # # This library is free software; you can redistribute it and/or # modify it under the terms of the GNU Lesser General Public # License as published by the Free Software Foundation; either # version 2 of the License, or (at your option) any later version. # # This library 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 # Lesser General Public License for more details. # # You should have received a copy of the GNU Lesser General Public # License along with this library; if not, write to the # Free Software Foundation, Inc., 59 Temple Place - Suite 330, # Boston, MA 02111-1307, USA. import logging import dbus import gobject _ACTIVITY_REGISTRY_SERVICE_NAME = 'org.laptop.ActivityRegistry' _ACTIVITY_REGISTRY_IFACE = 'org.laptop.ActivityRegistry' _ACTIVITY_REGISTRY_PATH = '/org/laptop/ActivityRegistry' def _activity_info_from_dict(info_dict): if not info_dict: return None return ActivityInfo(info_dict['name'], info_dict['icon'], info_dict['bundle_id'], info_dict['version'], info_dict['path'], info_dict['show_launcher'], info_dict['command'], info_dict['favorite'], info_dict['installation_time'], info_dict['position_x'], info_dict['position_y']) class ActivityInfo(object): def __init__(self, name, icon, bundle_id, version, path, show_launcher, command, favorite, installation_time, position_x, position_y): self.name = name self.icon = icon self.bundle_id = bundle_id self.version = version self.path = path self.command = command self.show_launcher = show_launcher self.favorite = favorite self.installation_time = installation_time self.position = (position_x, position_y) class ActivityRegistry(gobject.GObject): __gsignals__ = { 'activity-added': (gobject.SIGNAL_RUN_FIRST, gobject.TYPE_NONE, ([gobject.TYPE_PYOBJECT])), 'activity-removed': (gobject.SIGNAL_RUN_FIRST, gobject.TYPE_NONE, ([gobject.TYPE_PYOBJECT])), 'activity-changed': (gobject.SIGNAL_RUN_FIRST, gobject.TYPE_NONE, ([gobject.TYPE_PYOBJECT])) } def __init__(self): gobject.GObject.__init__(self) bus = dbus.SessionBus() # NOTE: We need to follow_name_owner_changes here # because we can not connect to a signal unless # we follow the changes or we start the service # before we connect. Starting the service here # causes a major bottleneck during startup bus_object = bus.get_object(_ACTIVITY_REGISTRY_SERVICE_NAME, _ACTIVITY_REGISTRY_PATH, follow_name_owner_changes = True) self._registry = dbus.Interface(bus_object, _ACTIVITY_REGISTRY_IFACE) self._registry.connect_to_signal('ActivityAdded', self._activity_added_cb) self._registry.connect_to_signal('ActivityRemoved', self._activity_removed_cb) self._registry.connect_to_signal('ActivityChanged', self._activity_changed_cb) # Two caches fo saving some travel across dbus. self._service_name_to_activity_info = {} self._mime_type_to_activities = {} def _convert_info_list(self, info_list): result = [] for info_dict in info_list: result.append(_activity_info_from_dict(info_dict)) return result def get_activities(self): info_list = self._registry.GetActivities() return self._convert_info_list(info_list) def _get_activities_cb(self, reply_handler, info_list): result = [] for info_dict in info_list: result.append(_activity_info_from_dict(info_dict)) reply_handler(result) def _get_activities_error_cb(self, error_handler, e): if error_handler: error_handler(e) else: logging.error('Error getting activities async: %s' % str(e)) def get_activities_async(self, reply_handler=None, error_handler=None): if not reply_handler: logging.error('Function get_activities_async called' \ 'without a reply handler. Can not run.') return self._registry.GetActivities( reply_handler=lambda info_list: \ self._get_activities_cb(reply_handler, info_list), error_handler=lambda e: \ self._get_activities_error_cb(error_handler, e)) def get_activity(self, service_name): if self._service_name_to_activity_info.has_key(service_name): return self._service_name_to_activity_info[service_name] info_dict = self._registry.GetActivity(service_name) activity_info = _activity_info_from_dict(info_dict) self._service_name_to_activity_info[service_name] = activity_info return activity_info def find_activity(self, name): info_list = self._registry.FindActivity(name) return self._convert_info_list(info_list) def get_activities_for_type(self, mime_type): if self._mime_type_to_activities.has_key(mime_type): return self._mime_type_to_activities[mime_type] info_list = self._registry.GetActivitiesForType(mime_type) activities = self._convert_info_list(info_list) self._mime_type_to_activities[mime_type] = activities return activities def add_bundle(self, bundle_path): result = self._registry.AddBundle(bundle_path) # Need to invalidate here because get_activity could be called after # add_bundle and before we receive activity-added, causing a race. self._invalidate_cache() return result def _activity_added_cb(self, info_dict): logging.debug('ActivityRegistry._activity_added_cb: invalidating cache') self._invalidate_cache() self.emit('activity-added', _activity_info_from_dict(info_dict)) def _invalidate_cache(self): self._service_name_to_activity_info.clear() self._mime_type_to_activities.clear() def remove_bundle(self, bundle_path): self._invalidate_cache() return self._registry.RemoveBundle(bundle_path) def _activity_removed_cb(self, info_dict): logging.debug('ActivityRegistry._activity_removed_cb: flushing caches') self._invalidate_cache() self.emit('activity-removed', _activity_info_from_dict(info_dict)) def _activity_changed_cb(self, info_dict): logging.debug('ActivityRegistry._activity_changed_cb: flushing caches') self._invalidate_cache() self.emit('activity-changed', _activity_info_from_dict(info_dict)) def set_activity_favorite(self, bundle_id, version, favorite): self._registry.SetActivityFavorite(bundle_id, version, favorite) def set_activity_position(self, bundle_id, version, x, y): self._registry.SetActivityPosition(bundle_id, version, x, y) _registry = None def get_registry(): global _registry if not _registry: _registry = ActivityRegistry() return _registry