Merge branch 'master' of git+ssh://dev.laptop.org/git/sugar
Conflicts: NEWS
This commit is contained in:
commit
7e45c5446d
4
NEWS
4
NEWS
@ -1,4 +1,8 @@
|
|||||||
* Activity launching now timeout after 120 seconds. (marco)
|
* Activity launching now timeout after 120 seconds. (marco)
|
||||||
|
* Add timeout arg to sugar.datastore.Datastore. (tomeu)
|
||||||
|
* Turn off logging by default. Logs may be re-enabled on a per-module basis
|
||||||
|
by adding environment variables like SHELL_DEBUG and RECORD_DEBUG to
|
||||||
|
the sugar environment
|
||||||
|
|
||||||
Snapshot 088c7612e3
|
Snapshot 088c7612e3
|
||||||
|
|
||||||
|
@ -19,13 +19,13 @@ from hashlib import sha1
|
|||||||
|
|
||||||
import dbus
|
import dbus
|
||||||
from gtk import VBox, Label, TreeView, Expander, ListStore, CellRendererText,\
|
from gtk import VBox, Label, TreeView, Expander, ListStore, CellRendererText,\
|
||||||
ScrolledWindow, CellRendererToggle
|
ScrolledWindow, CellRendererToggle, TextView, VPaned
|
||||||
from gobject import timeout_add
|
from gobject import timeout_add
|
||||||
|
|
||||||
|
|
||||||
logger = logging.getLogger('ps_watcher')
|
logger = logging.getLogger('ps_watcher')
|
||||||
logging.basicConfig(filename='/tmp/ps_watcher.log')
|
#logging.basicConfig(filename='/tmp/ps_watcher.log')
|
||||||
logging.getLogger().setLevel(1)
|
#logging.getLogger().setLevel(1)
|
||||||
|
|
||||||
|
|
||||||
PS_NAME = 'org.laptop.Sugar.Presence'
|
PS_NAME = 'org.laptop.Sugar.Presence'
|
||||||
@ -42,6 +42,9 @@ ACT_COL_ID = 3
|
|||||||
ACT_COL_COLOR = 4
|
ACT_COL_COLOR = 4
|
||||||
ACT_COL_TYPE = 5
|
ACT_COL_TYPE = 5
|
||||||
ACT_COL_NAME = 6
|
ACT_COL_NAME = 6
|
||||||
|
ACT_COL_CONN = 7
|
||||||
|
ACT_COL_CHANNELS = 8
|
||||||
|
ACT_COL_BUDDIES = 9
|
||||||
BUDDY_COL_PATH = 0
|
BUDDY_COL_PATH = 0
|
||||||
BUDDY_COL_WEIGHT = 1
|
BUDDY_COL_WEIGHT = 1
|
||||||
BUDDY_COL_STRIKE = 2
|
BUDDY_COL_STRIKE = 2
|
||||||
@ -51,6 +54,8 @@ BUDDY_COL_COLOR = 5
|
|||||||
BUDDY_COL_IP4 = 6
|
BUDDY_COL_IP4 = 6
|
||||||
BUDDY_COL_CUR_ACT = 7
|
BUDDY_COL_CUR_ACT = 7
|
||||||
BUDDY_COL_KEY_ID = 8
|
BUDDY_COL_KEY_ID = 8
|
||||||
|
BUDDY_COL_ACTIVITIES = 9
|
||||||
|
BUDDY_COL_HANDLES = 10
|
||||||
|
|
||||||
|
|
||||||
class ActivityWatcher(object):
|
class ActivityWatcher(object):
|
||||||
@ -70,6 +75,9 @@ class ActivityWatcher(object):
|
|||||||
self.color = '?'
|
self.color = '?'
|
||||||
self.type = '?'
|
self.type = '?'
|
||||||
self.name = '?'
|
self.name = '?'
|
||||||
|
self.conn = '?'
|
||||||
|
self.channels = None
|
||||||
|
self.buddies = None
|
||||||
|
|
||||||
self.iter = self.ps_watcher.add_activity(self)
|
self.iter = self.ps_watcher.add_activity(self)
|
||||||
|
|
||||||
@ -85,12 +93,94 @@ class ActivityWatcher(object):
|
|||||||
self.iface.GetName(reply_handler=self._on_get_name_success,
|
self.iface.GetName(reply_handler=self._on_get_name_success,
|
||||||
error_handler=self._on_get_name_failure)
|
error_handler=self._on_get_name_failure)
|
||||||
|
|
||||||
|
self.iface.connect_to_signal('NewChannel', self._on_new_channel)
|
||||||
|
self.iface.GetChannels(reply_handler=self._on_get_channels_success,
|
||||||
|
error_handler=self._on_get_channels_failure)
|
||||||
|
|
||||||
|
self.iface.connect_to_signal('BuddyJoined', self._on_buddy_joined)
|
||||||
|
self.iface.connect_to_signal('BuddyLeft', self._on_buddy_left)
|
||||||
|
self.iface.GetJoinedBuddies(reply_handler=self._on_get_buddies_success,
|
||||||
|
error_handler=self._on_get_buddies_failure)
|
||||||
|
|
||||||
|
def _on_buddy_joined(self, buddy):
|
||||||
|
if self.buddies is None:
|
||||||
|
return
|
||||||
|
if buddy.startswith('/org/laptop/Sugar/Presence/Buddies/'):
|
||||||
|
buddy = '.../' + buddy[35:]
|
||||||
|
self.ps_watcher.log('INFO: Activity %s emitted BuddyJoined("%s") '
|
||||||
|
'or mentioned the buddy in GetJoinedBuddies',
|
||||||
|
self.object_path, buddy)
|
||||||
|
self.buddies.append(buddy)
|
||||||
|
self.ps_watcher.activities_list_store.set(self.iter, ACT_COL_BUDDIES,
|
||||||
|
' '.join(self.buddies))
|
||||||
|
|
||||||
|
def _on_buddy_left(self, buddy):
|
||||||
|
if self.buddies is None:
|
||||||
|
return
|
||||||
|
if buddy.startswith('/org/laptop/Sugar/Presence/Buddies/'):
|
||||||
|
buddy = '.../' + buddy[35:]
|
||||||
|
self.ps_watcher.log('INFO: Activity %s emitted BuddyLeft("%s")',
|
||||||
|
self.object_path, buddy)
|
||||||
|
try:
|
||||||
|
self.buddies.remove(buddy)
|
||||||
|
except ValueError:
|
||||||
|
pass
|
||||||
|
self.ps_watcher.activities_list_store.set(self.iter, ACT_COL_BUDDIES,
|
||||||
|
' '.join(self.buddies))
|
||||||
|
|
||||||
|
def _on_get_buddies_success(self, buddies):
|
||||||
|
self.buddies = []
|
||||||
|
for buddy in buddies:
|
||||||
|
self._on_buddy_joined(buddy)
|
||||||
|
|
||||||
|
def _on_get_buddies_failure(self, e):
|
||||||
|
self.log('ERROR: <Activity %s>.GetJoinedBuddies(): %s',
|
||||||
|
self.object_path, e)
|
||||||
|
self.ps_watcher.activities_list_store.set(self.iter, ACT_COL_BUDDIES,
|
||||||
|
'!')
|
||||||
|
|
||||||
|
def _on_new_channel(self, channel):
|
||||||
|
if self.channels is None:
|
||||||
|
return
|
||||||
|
if channel.startswith(self.full_conn):
|
||||||
|
channel = '...' + channel[len(self.full_conn):]
|
||||||
|
self.ps_watcher.log('INFO: Activity %s emitted NewChannel("%s") '
|
||||||
|
'or mentioned the channel in GetChannels()',
|
||||||
|
self.object_path, channel)
|
||||||
|
self.channels.append(channel)
|
||||||
|
# FIXME: listen for Telepathy Closed signal!
|
||||||
|
self.ps_watcher.activities_list_store.set(self.iter, ACT_COL_CHANNELS,
|
||||||
|
' '.join(self.channels))
|
||||||
|
|
||||||
|
def _on_get_channels_success(self, service, conn, channels):
|
||||||
|
self.full_conn = conn
|
||||||
|
if conn.startswith('/org/freedesktop/Telepathy/Connection/'):
|
||||||
|
self.conn = '.../' + conn[38:]
|
||||||
|
else:
|
||||||
|
self.conn = conn
|
||||||
|
self.ps_watcher.activities_list_store.set(self.iter, ACT_COL_CONN,
|
||||||
|
self.conn)
|
||||||
|
self.channels = []
|
||||||
|
for channel in channels:
|
||||||
|
self._on_new_channel(channel)
|
||||||
|
self.ps_watcher.activities_list_store.set(self.iter, ACT_COL_CHANNELS,
|
||||||
|
' '.join(self.channels))
|
||||||
|
|
||||||
|
def _on_get_channels_failure(self, e):
|
||||||
|
self.ps_watcher.log('ERROR: <Activity %s>.GetChannels(): %s',
|
||||||
|
self.object_path, e)
|
||||||
|
self.ps_watcher.activities_list_store.set(self.iter, ACT_COL_CONN,
|
||||||
|
'!')
|
||||||
|
self.ps_watcher.activities_list_store.set(self.iter, ACT_COL_CHANNELS,
|
||||||
|
'!')
|
||||||
|
|
||||||
def _on_get_id_success(self, ident):
|
def _on_get_id_success(self, ident):
|
||||||
self.id = ident
|
self.id = ident
|
||||||
self.ps_watcher.activities_list_store.set(self.iter, ACT_COL_ID, ident)
|
self.ps_watcher.activities_list_store.set(self.iter, ACT_COL_ID, ident)
|
||||||
|
|
||||||
def _on_get_id_failure(self, e):
|
def _on_get_id_failure(self, e):
|
||||||
logger.warning('<Activity %s>.GetId(): %s', self.object_path, e)
|
self.ps_watcher.log('ERROR: <Activity %s>.GetId(): %s',
|
||||||
|
self.object_path, e)
|
||||||
self.ps_watcher.activities_list_store.set(self.iter, ACT_COL_ID,
|
self.ps_watcher.activities_list_store.set(self.iter, ACT_COL_ID,
|
||||||
'!')
|
'!')
|
||||||
|
|
||||||
@ -100,7 +190,8 @@ class ActivityWatcher(object):
|
|||||||
color)
|
color)
|
||||||
|
|
||||||
def _on_get_color_failure(self, e):
|
def _on_get_color_failure(self, e):
|
||||||
logger.warning('<Activity %s>.GetColor(): %s', self.object_path, e)
|
self.ps_watcher.log('ERROR: <Activity %s>.GetColor(): %s',
|
||||||
|
self.object_path, e)
|
||||||
self.ps_watcher.activities_list_store.set(self.iter, ACT_COL_COLOR,
|
self.ps_watcher.activities_list_store.set(self.iter, ACT_COL_COLOR,
|
||||||
'!')
|
'!')
|
||||||
|
|
||||||
@ -110,7 +201,8 @@ class ActivityWatcher(object):
|
|||||||
type_)
|
type_)
|
||||||
|
|
||||||
def _on_get_type_failure(self, e):
|
def _on_get_type_failure(self, e):
|
||||||
logger.warning('<Activity %s>.GetType(): %s', self.object_path, e)
|
self.ps_watcher.log('ERROR: <Activity %s>.GetType(): %s',
|
||||||
|
self.object_path, e)
|
||||||
self.ps_watcher.activities_list_store.set(self.iter, ACT_COL_TYPE,
|
self.ps_watcher.activities_list_store.set(self.iter, ACT_COL_TYPE,
|
||||||
'!')
|
'!')
|
||||||
|
|
||||||
@ -120,7 +212,8 @@ class ActivityWatcher(object):
|
|||||||
name)
|
name)
|
||||||
|
|
||||||
def _on_get_name_failure(self, e):
|
def _on_get_name_failure(self, e):
|
||||||
logger.warning('<Activity %s>.GetName(): %s', self.object_path, e)
|
self.ps_watcher.log('ERROR: <Activity %s>.GetName(): %s',
|
||||||
|
self.object_path, e)
|
||||||
self.ps_watcher.activities_list_store.set(self.iter, ACT_COL_NAME,
|
self.ps_watcher.activities_list_store.set(self.iter, ACT_COL_NAME,
|
||||||
'!')
|
'!')
|
||||||
|
|
||||||
@ -160,41 +253,168 @@ class BuddyWatcher(object):
|
|||||||
self.ipv4 = '?'
|
self.ipv4 = '?'
|
||||||
self.cur_act = '?'
|
self.cur_act = '?'
|
||||||
self.keyid = '?'
|
self.keyid = '?'
|
||||||
|
self.activities = None
|
||||||
|
self.handles = None
|
||||||
|
|
||||||
self.iter = self.ps_watcher.add_buddy(self)
|
self.iter = self.ps_watcher.add_buddy(self)
|
||||||
|
|
||||||
|
self.iface.connect_to_signal('PropertyChanged', self._on_props_changed,
|
||||||
|
byte_arrays=True)
|
||||||
|
self.ps_watcher.log('Calling <Buddy %s>.GetProperties()', object_path)
|
||||||
self.iface.GetProperties(reply_handler=self._on_get_props_success,
|
self.iface.GetProperties(reply_handler=self._on_get_props_success,
|
||||||
error_handler=self._on_get_props_failure,
|
error_handler=self._on_get_props_failure,
|
||||||
byte_arrays=True)
|
byte_arrays=True)
|
||||||
|
|
||||||
|
self.iface.connect_to_signal('JoinedActivity', self._on_joined)
|
||||||
|
self.iface.connect_to_signal('LeftActivity', self._on_left)
|
||||||
|
self.ps_watcher.log('Calling <Buddy %s>.GetJoinedActivities()',
|
||||||
|
object_path)
|
||||||
|
self.iface.GetJoinedActivities(reply_handler=self._on_get_acts_success,
|
||||||
|
error_handler=self._on_get_acts_failure)
|
||||||
|
|
||||||
|
self.iface.connect_to_signal('TelepathyHandleAdded',
|
||||||
|
self._on_handle_added)
|
||||||
|
self.iface.connect_to_signal('TelepathyHandleRemoved',
|
||||||
|
self._on_handle_removed)
|
||||||
|
self.ps_watcher.log('Calling <Buddy %s>.GetTelepathyHandles()',
|
||||||
|
object_path)
|
||||||
|
self.iface.GetTelepathyHandles(
|
||||||
|
reply_handler=self._on_get_handles_success,
|
||||||
|
error_handler=self._on_get_handles_failure)
|
||||||
|
|
||||||
|
def _on_handle_added(self, service, conn, handle):
|
||||||
|
if self.handles is None:
|
||||||
|
return
|
||||||
|
self.ps_watcher.log('INFO: Buddy %s emitted Telepathy HandleAdded('
|
||||||
|
'"%s", "%s", %u) or mentioned the handle in '
|
||||||
|
'GetTelepathyHandles()',
|
||||||
|
self.object_path, service, conn, handle)
|
||||||
|
if conn.startswith('/org/freedesktop/Telepathy/Connection/'):
|
||||||
|
conn = '.../' + conn[38:]
|
||||||
|
self.handles.append('%u@%s' % (handle, conn))
|
||||||
|
self.ps_watcher.buddies_list_store.set(self.iter,
|
||||||
|
BUDDY_COL_HANDLES,
|
||||||
|
' '.join(self.handles))
|
||||||
|
|
||||||
|
def _on_handle_removed(self, service, conn, handle):
|
||||||
|
if self.handles is None:
|
||||||
|
return
|
||||||
|
if conn.startswith('/org/freedesktop/Telepathy/Connection/'):
|
||||||
|
conn = '.../' + conn[38:]
|
||||||
|
self.ps_watcher.log('INFO: Buddy %s emitted HandleRemoved("%s", '
|
||||||
|
'"%s", %u)', self.object_path, service, conn,
|
||||||
|
handle)
|
||||||
|
try:
|
||||||
|
self.handles.remove('%u@%s' % (handle, conn))
|
||||||
|
except ValueError:
|
||||||
|
pass
|
||||||
|
self.ps_watcher.buddies_list_store.set(self.iter, BUDDY_COL_HANDLES,
|
||||||
|
' '.join(self.handles))
|
||||||
|
|
||||||
|
def _on_get_handles_success(self, handles):
|
||||||
|
self.handles = []
|
||||||
|
for service, conn, handle in handles:
|
||||||
|
self._on_handle_added(service, conn, handle)
|
||||||
|
|
||||||
|
def _on_get_handles_failure(self, e):
|
||||||
|
self.ps_watcher.log('ERROR: <Buddy %s>.GetTelepathyHandles(): %s',
|
||||||
|
self.object_path, e)
|
||||||
|
self.ps_watcher.buddies_list_store.set(self.iter, BUDDY_COL_HANDLES,
|
||||||
|
'!')
|
||||||
|
|
||||||
|
def _on_joined(self, act):
|
||||||
|
if self.activities is None:
|
||||||
|
return
|
||||||
|
if act.startswith('/org/laptop/Sugar/Presence/Activities/'):
|
||||||
|
act = '.../' + act[38:]
|
||||||
|
self.ps_watcher.log('INFO: Buddy %s emitted ActivityJoined("%s") '
|
||||||
|
'or mentioned it in GetJoinedActivities()',
|
||||||
|
self.object_path, act)
|
||||||
|
self.activities.append(act)
|
||||||
|
self.ps_watcher.buddies_list_store.set(self.iter,
|
||||||
|
BUDDY_COL_ACTIVITIES,
|
||||||
|
' '.join(self.activities))
|
||||||
|
|
||||||
|
def _on_left(self, act):
|
||||||
|
if self.activities is None:
|
||||||
|
return
|
||||||
|
if act.startswith('/org/laptop/Sugar/Presence/Activities/'):
|
||||||
|
act = '.../' + act[38:]
|
||||||
|
self.ps_watcher.log('INFO: Buddy %s emitted ActivityLeft("%s")',
|
||||||
|
self.object_path, act)
|
||||||
|
try:
|
||||||
|
self.activities.remove(act)
|
||||||
|
except ValueError:
|
||||||
|
pass
|
||||||
|
self.ps_watcher.buddies_list_store.set(self.iter, BUDDY_COL_ACTIVITIES,
|
||||||
|
' '.join(self.activities))
|
||||||
|
|
||||||
|
def _on_get_acts_success(self, activities):
|
||||||
|
self.activities = []
|
||||||
|
for act in activities:
|
||||||
|
self._on_joined(act)
|
||||||
|
|
||||||
|
def _on_get_acts_failure(self, e):
|
||||||
|
self.ps_watcher.log('ERROR: <Buddy %s>.GetJoinedActivities(): %s',
|
||||||
|
self.object_path, e)
|
||||||
|
self.ps_watcher.buddies_list_store.set(self.iter, BUDDY_COL_ACTIVITIES,
|
||||||
|
'!')
|
||||||
|
|
||||||
|
def _on_props_changed(self, props):
|
||||||
|
try:
|
||||||
|
logger.debug('PropertyChanged(%s, %s)', self, props)
|
||||||
|
self.ps_watcher.log('INFO: <Buddy %s> emitted PropertyChanged(%r)',
|
||||||
|
self.object_path, props)
|
||||||
|
self._props_changed(props)
|
||||||
|
except Exception, e:
|
||||||
|
self.ps_watcher.log('INTERNAL ERROR: %s', e)
|
||||||
|
|
||||||
def _on_get_props_success(self, props):
|
def _on_get_props_success(self, props):
|
||||||
# ignore key for now
|
try:
|
||||||
|
logger.debug('GetProperties(%s, %s)', self, props)
|
||||||
|
self.ps_watcher.log('INFO: <Buddy %s>.GetProperties() -> %r',
|
||||||
|
self.object_path, props)
|
||||||
|
self._props_changed(props)
|
||||||
|
except Exception, e:
|
||||||
|
self.ps_watcher.log('INTERNAL ERROR: %s', e)
|
||||||
|
|
||||||
|
def _props_changed(self, props):
|
||||||
|
logger.debug('Begin _props_changed')
|
||||||
|
if 'nick' in props:
|
||||||
self.nick = props.get('nick', '?')
|
self.nick = props.get('nick', '?')
|
||||||
self.owner = props.get('owner', False)
|
|
||||||
self.color = props.get('color', '?')
|
|
||||||
self.ipv4 = props.get('ip4-address', '?')
|
|
||||||
self.ipv4 = props.get('ip4-address', '?')
|
|
||||||
self.cur_act = props.get('current-activity', '?')
|
|
||||||
key = props.get('key', None)
|
|
||||||
if key is not None:
|
|
||||||
self.keyid = sha1(key).hexdigest()[:8] + '...'
|
|
||||||
else:
|
|
||||||
self.keyid = '?'
|
|
||||||
self.ps_watcher.buddies_list_store.set(self.iter, BUDDY_COL_NICK,
|
self.ps_watcher.buddies_list_store.set(self.iter, BUDDY_COL_NICK,
|
||||||
self.nick)
|
self.nick)
|
||||||
|
if 'owner' in props:
|
||||||
|
self.owner = bool(props.get('owner', False))
|
||||||
self.ps_watcher.buddies_list_store.set(self.iter, BUDDY_COL_OWNER,
|
self.ps_watcher.buddies_list_store.set(self.iter, BUDDY_COL_OWNER,
|
||||||
self.owner)
|
self.owner)
|
||||||
|
if 'color' in props:
|
||||||
|
self.color = props.get('color', '?')
|
||||||
self.ps_watcher.buddies_list_store.set(self.iter, BUDDY_COL_COLOR,
|
self.ps_watcher.buddies_list_store.set(self.iter, BUDDY_COL_COLOR,
|
||||||
self.color)
|
self.color)
|
||||||
|
if 'ip4-address' in props:
|
||||||
|
self.ipv4 = props.get('ip4-address', '?')
|
||||||
self.ps_watcher.buddies_list_store.set(self.iter, BUDDY_COL_IP4,
|
self.ps_watcher.buddies_list_store.set(self.iter, BUDDY_COL_IP4,
|
||||||
self.ipv4)
|
self.ipv4)
|
||||||
|
if 'current-activity' in props:
|
||||||
|
self.cur_act = props.get('current-activity', '?')
|
||||||
self.ps_watcher.buddies_list_store.set(self.iter, BUDDY_COL_CUR_ACT,
|
self.ps_watcher.buddies_list_store.set(self.iter, BUDDY_COL_CUR_ACT,
|
||||||
self.cur_act)
|
self.cur_act)
|
||||||
|
if 'key' in props:
|
||||||
|
key = props.get('key', None)
|
||||||
|
if key:
|
||||||
|
self.keyid = '%d bytes, sha1 %s' % (len(key),
|
||||||
|
sha1(key).hexdigest())
|
||||||
|
else:
|
||||||
|
# could be '' (present, empty value) or None (absent). Either way:
|
||||||
|
self.keyid = '?'
|
||||||
self.ps_watcher.buddies_list_store.set(self.iter, BUDDY_COL_KEY_ID,
|
self.ps_watcher.buddies_list_store.set(self.iter, BUDDY_COL_KEY_ID,
|
||||||
self.keyid)
|
self.keyid)
|
||||||
|
logger.debug('End _props_changed')
|
||||||
|
|
||||||
def _on_get_props_failure(self, e):
|
def _on_get_props_failure(self, e):
|
||||||
logger.warning('<Buddy %s>.GetProperties(): %s', self.object_path, e)
|
self.ps_watcher.log('ERROR: <Buddy %s>.GetProperties(): %s',
|
||||||
|
self.object_path, e)
|
||||||
self.ps_watcher.buddies_list_store.set(self.iter, BUDDY_COL_NICK, '!')
|
self.ps_watcher.buddies_list_store.set(self.iter, BUDDY_COL_NICK, '!')
|
||||||
self.ps_watcher.buddies_list_store.set(self.iter, BUDDY_COL_OWNER,
|
self.ps_watcher.buddies_list_store.set(self.iter, BUDDY_COL_OWNER,
|
||||||
False)
|
False)
|
||||||
@ -205,7 +425,6 @@ class BuddyWatcher(object):
|
|||||||
self.ps_watcher.buddies_list_store.set(self.iter, BUDDY_COL_KEY_ID,
|
self.ps_watcher.buddies_list_store.set(self.iter, BUDDY_COL_KEY_ID,
|
||||||
'!')
|
'!')
|
||||||
|
|
||||||
|
|
||||||
def _finish_appearing(self):
|
def _finish_appearing(self):
|
||||||
self.appearing = False
|
self.appearing = False
|
||||||
self.ps_watcher.buddies_list_store.set(self.iter, BUDDY_COL_WEIGHT,
|
self.ps_watcher.buddies_list_store.set(self.iter, BUDDY_COL_WEIGHT,
|
||||||
@ -225,16 +444,14 @@ class BuddyWatcher(object):
|
|||||||
|
|
||||||
class PresenceServiceWatcher(VBox):
|
class PresenceServiceWatcher(VBox):
|
||||||
|
|
||||||
def __init__(self, bus, unique_name):
|
def __init__(self, bus, unique_name, log):
|
||||||
VBox.__init__(self)
|
VBox.__init__(self)
|
||||||
|
|
||||||
logger.debug('Starting up PresenceServiceWatcher...')
|
|
||||||
self.bus = bus
|
self.bus = bus
|
||||||
self.unique_name = unique_name
|
self.unique_name = unique_name
|
||||||
self.proxy = bus.get_object(unique_name, PS_PATH)
|
self.proxy = bus.get_object(unique_name, PS_PATH)
|
||||||
self.iface = dbus.Interface(self.proxy, PS_IFACE)
|
self.iface = dbus.Interface(self.proxy, PS_IFACE)
|
||||||
|
self.log = log
|
||||||
logger.debug('Starting up PresenceServiceWatcher (2)...')
|
|
||||||
|
|
||||||
self.activities = None
|
self.activities = None
|
||||||
self.iface.connect_to_signal('ActivityAppeared',
|
self.iface.connect_to_signal('ActivityAppeared',
|
||||||
@ -260,6 +477,9 @@ class PresenceServiceWatcher(VBox):
|
|||||||
str, # color
|
str, # color
|
||||||
str, # type
|
str, # type
|
||||||
str, # name
|
str, # name
|
||||||
|
str, # conn
|
||||||
|
str, # channels
|
||||||
|
str, # buddies
|
||||||
)
|
)
|
||||||
|
|
||||||
self.pack_start(Label('Activities:'), False, False)
|
self.pack_start(Label('Activities:'), False, False)
|
||||||
@ -290,6 +510,19 @@ class PresenceServiceWatcher(VBox):
|
|||||||
strikethrough=ACT_COL_STRIKE)
|
strikethrough=ACT_COL_STRIKE)
|
||||||
c.set_resizable(True)
|
c.set_resizable(True)
|
||||||
c.set_sort_column_id(ACT_COL_NAME)
|
c.set_sort_column_id(ACT_COL_NAME)
|
||||||
|
c = self.activities_list.insert_column_with_attributes(5, 'Connection',
|
||||||
|
CellRendererText(), text=ACT_COL_CONN, weight=ACT_COL_WEIGHT,
|
||||||
|
strikethrough=ACT_COL_STRIKE)
|
||||||
|
c.set_resizable(True)
|
||||||
|
c.set_sort_column_id(ACT_COL_CONN)
|
||||||
|
c = self.activities_list.insert_column_with_attributes(6, 'Channels',
|
||||||
|
CellRendererText(), text=ACT_COL_CHANNELS, weight=ACT_COL_WEIGHT,
|
||||||
|
strikethrough=ACT_COL_STRIKE)
|
||||||
|
c.set_resizable(True)
|
||||||
|
c = self.activities_list.insert_column_with_attributes(7, 'Buddies',
|
||||||
|
CellRendererText(), text=ACT_COL_BUDDIES, weight=ACT_COL_WEIGHT,
|
||||||
|
strikethrough=ACT_COL_STRIKE)
|
||||||
|
c.set_resizable(True)
|
||||||
|
|
||||||
scroller = ScrolledWindow()
|
scroller = ScrolledWindow()
|
||||||
scroller.add(self.activities_list)
|
scroller.add(self.activities_list)
|
||||||
@ -297,7 +530,7 @@ class PresenceServiceWatcher(VBox):
|
|||||||
|
|
||||||
# keep this in sync with the BUDDY_COL_ constants
|
# keep this in sync with the BUDDY_COL_ constants
|
||||||
self.buddies_list_store = ListStore(str, int, bool, str, bool,
|
self.buddies_list_store = ListStore(str, int, bool, str, bool,
|
||||||
str, str, str, str)
|
str, str, str, str, str, str)
|
||||||
|
|
||||||
self.pack_start(Label('Buddies:'), False, False)
|
self.pack_start(Label('Buddies:'), False, False)
|
||||||
self.buddies_list = TreeView(self.buddies_list_store)
|
self.buddies_list = TreeView(self.buddies_list_store)
|
||||||
@ -306,7 +539,7 @@ class PresenceServiceWatcher(VBox):
|
|||||||
weight=BUDDY_COL_WEIGHT, strikethrough=BUDDY_COL_STRIKE)
|
weight=BUDDY_COL_WEIGHT, strikethrough=BUDDY_COL_STRIKE)
|
||||||
c.set_resizable(True)
|
c.set_resizable(True)
|
||||||
c.set_sort_column_id(BUDDY_COL_PATH)
|
c.set_sort_column_id(BUDDY_COL_PATH)
|
||||||
c = self.buddies_list.insert_column_with_attributes(1, 'Key ID',
|
c = self.buddies_list.insert_column_with_attributes(1, 'Pubkey',
|
||||||
CellRendererText(), text=BUDDY_COL_KEY_ID,
|
CellRendererText(), text=BUDDY_COL_KEY_ID,
|
||||||
weight=BUDDY_COL_WEIGHT, strikethrough=BUDDY_COL_STRIKE)
|
weight=BUDDY_COL_WEIGHT, strikethrough=BUDDY_COL_STRIKE)
|
||||||
c.set_resizable(True)
|
c.set_resizable(True)
|
||||||
@ -333,6 +566,16 @@ class PresenceServiceWatcher(VBox):
|
|||||||
weight=BUDDY_COL_WEIGHT, strikethrough=BUDDY_COL_STRIKE)
|
weight=BUDDY_COL_WEIGHT, strikethrough=BUDDY_COL_STRIKE)
|
||||||
c.set_resizable(True)
|
c.set_resizable(True)
|
||||||
c.set_sort_column_id(BUDDY_COL_CUR_ACT)
|
c.set_sort_column_id(BUDDY_COL_CUR_ACT)
|
||||||
|
c = self.buddies_list.insert_column_with_attributes(7, 'Activities',
|
||||||
|
CellRendererText(), text=BUDDY_COL_ACTIVITIES,
|
||||||
|
weight=BUDDY_COL_WEIGHT, strikethrough=BUDDY_COL_STRIKE)
|
||||||
|
c.set_resizable(True)
|
||||||
|
c.set_sort_column_id(BUDDY_COL_ACTIVITIES)
|
||||||
|
c = self.buddies_list.insert_column_with_attributes(8, 'Handles',
|
||||||
|
CellRendererText(), text=BUDDY_COL_HANDLES,
|
||||||
|
weight=BUDDY_COL_WEIGHT, strikethrough=BUDDY_COL_STRIKE)
|
||||||
|
c.set_resizable(True)
|
||||||
|
c.set_sort_column_id(BUDDY_COL_HANDLES)
|
||||||
|
|
||||||
scroller = ScrolledWindow()
|
scroller = ScrolledWindow()
|
||||||
scroller.add(self.buddies_list)
|
scroller.add(self.buddies_list)
|
||||||
@ -344,20 +587,20 @@ class PresenceServiceWatcher(VBox):
|
|||||||
self._on_private_invitation)
|
self._on_private_invitation)
|
||||||
|
|
||||||
def _on_get_activities_success(self, paths):
|
def _on_get_activities_success(self, paths):
|
||||||
logger.debug('PS GetActivities() returned %r', paths)
|
self.log('INFO: PS GetActivities() returned %r', paths)
|
||||||
self.activities = {}
|
self.activities = {}
|
||||||
for path in paths:
|
for path in paths:
|
||||||
self.activities[path] = ActivityWatcher(self, path)
|
self.activities[path] = ActivityWatcher(self, path)
|
||||||
|
|
||||||
def _on_get_activities_failure(self, e):
|
def _on_get_activities_failure(self, e):
|
||||||
logger.warning('PS GetActivities() failed with %s', e)
|
self.log('ERROR: PS GetActivities() failed with %s', e)
|
||||||
|
|
||||||
def add_activity(self, act):
|
def add_activity(self, act):
|
||||||
path = act.object_path
|
path = act.object_path
|
||||||
if path.startswith('/org/laptop/Sugar/Presence/Activities/'):
|
if path.startswith('/org/laptop/Sugar/Presence/Activities/'):
|
||||||
path = '.../' + path[38:]
|
path = '.../' + path[38:]
|
||||||
return self.activities_list_store.append((path, 700, False,
|
return self.activities_list_store.append((path, 700, False,
|
||||||
act.id, act.color, act.type, act.name))
|
act.id, act.color, act.type, act.name, act.conn, '?', '?'))
|
||||||
|
|
||||||
def remove_activity(self, act):
|
def remove_activity(self, act):
|
||||||
self.activities.pop(act.object_path, None)
|
self.activities.pop(act.object_path, None)
|
||||||
@ -366,43 +609,44 @@ class PresenceServiceWatcher(VBox):
|
|||||||
def _on_activity_appeared(self, path):
|
def _on_activity_appeared(self, path):
|
||||||
if self.activities is None:
|
if self.activities is None:
|
||||||
return
|
return
|
||||||
logger.debug('PS emitted ActivityAppeared("%s")', path)
|
self.log('INFO: PS emitted ActivityAppeared("%s")', path)
|
||||||
self.activities[path] = ActivityWatcher(self, path)
|
self.activities[path] = ActivityWatcher(self, path)
|
||||||
|
|
||||||
def _on_activity_disappeared(self, path):
|
def _on_activity_disappeared(self, path):
|
||||||
if self.activities is None:
|
if self.activities is None:
|
||||||
return
|
return
|
||||||
logger.debug('PS emitted ActivityDisappeared("%s")', path)
|
self.log('INFO: PS emitted ActivityDisappeared("%s")', path)
|
||||||
act = self.activities.get(path)
|
act = self.activities.get(path)
|
||||||
if act is None:
|
if act is None:
|
||||||
logger.warning('Trying to remove activity "%s" which is already '
|
self.log('WARNING: Trying to remove activity "%s" which is '
|
||||||
'absent', path)
|
'already absent', path)
|
||||||
else:
|
else:
|
||||||
# we don't remove the activity straight away, just cross it out
|
# we don't remove the activity straight away, just cross it out
|
||||||
act.disappear()
|
act.disappear()
|
||||||
|
|
||||||
def _on_activity_invitation(self, path):
|
def _on_activity_invitation(self, path):
|
||||||
logger.debug('PS emitted ActivityInvitation("%s")', path)
|
self.log('INFO: PS emitted ActivityInvitation("%s")', path)
|
||||||
|
|
||||||
def _on_private_invitation(self, bus_name, conn, channel):
|
def _on_private_invitation(self, bus_name, conn, channel):
|
||||||
logger.debug('PS emitted PrivateInvitation("%s", "%s", "%s")',
|
self.log('INFO: PS emitted PrivateInvitation("%s", "%s", "%s")',
|
||||||
bus_name, conn, channel)
|
bus_name, conn, channel)
|
||||||
|
|
||||||
def _on_get_buddies_success(self, paths):
|
def _on_get_buddies_success(self, paths):
|
||||||
logger.debug('PS GetBuddies() returned %r', paths)
|
self.log('INFO: PS GetBuddies() returned %r', paths)
|
||||||
self.buddies = {}
|
self.buddies = {}
|
||||||
for path in paths:
|
for path in paths:
|
||||||
self.buddies[path] = BuddyWatcher(self, path)
|
self.buddies[path] = BuddyWatcher(self, path)
|
||||||
|
|
||||||
def _on_get_buddies_failure(self, e):
|
def _on_get_buddies_failure(self, e):
|
||||||
logger.warning('PS GetBuddies() failed with %s', e)
|
self.log('ERROR: PS GetBuddies() failed with %s', e)
|
||||||
|
|
||||||
def add_buddy(self, b):
|
def add_buddy(self, b):
|
||||||
path = b.object_path
|
path = b.object_path
|
||||||
if path.startswith('/org/laptop/Sugar/Presence/Buddies/'):
|
if path.startswith('/org/laptop/Sugar/Presence/Buddies/'):
|
||||||
path = '.../' + path[35:]
|
path = '.../' + path[35:]
|
||||||
return self.buddies_list_store.append((path, 700, False,
|
return self.buddies_list_store.append((path, 700, False,
|
||||||
b.nick, b.owner, b.color, b.ipv4, b.cur_act, b.keyid))
|
b.nick, b.owner, b.color, b.ipv4, b.cur_act, b.keyid,
|
||||||
|
'?', '?'))
|
||||||
|
|
||||||
def remove_buddy(self, b):
|
def remove_buddy(self, b):
|
||||||
self.buddies.pop(b.object_path, None)
|
self.buddies.pop(b.object_path, None)
|
||||||
@ -411,16 +655,16 @@ class PresenceServiceWatcher(VBox):
|
|||||||
def _on_buddy_appeared(self, path):
|
def _on_buddy_appeared(self, path):
|
||||||
if self.buddies is None:
|
if self.buddies is None:
|
||||||
return
|
return
|
||||||
logger.debug('PS emitted BuddyAppeared("%s")', path)
|
self.log('INFO: PS emitted BuddyAppeared("%s")', path)
|
||||||
self.buddies[path] = BuddyWatcher(self, path)
|
self.buddies[path] = BuddyWatcher(self, path)
|
||||||
|
|
||||||
def _on_buddy_disappeared(self, path):
|
def _on_buddy_disappeared(self, path):
|
||||||
if self.buddies is None:
|
if self.buddies is None:
|
||||||
return
|
return
|
||||||
logger.debug('PS emitted BuddyDisappeared("%s")', path)
|
self.log('INFO: PS emitted BuddyDisappeared("%s")', path)
|
||||||
b = self.buddies.get(path)
|
b = self.buddies.get(path)
|
||||||
if b is None:
|
if b is None:
|
||||||
logger.warning('Trying to remove buddy "%s" which is already '
|
self.log('ERROR: Trying to remove buddy "%s" which is already '
|
||||||
'absent', path)
|
'absent', path)
|
||||||
else:
|
else:
|
||||||
# we don't remove the activity straight away, just cross it out
|
# we don't remove the activity straight away, just cross it out
|
||||||
@ -434,31 +678,50 @@ class PresenceServiceNameWatcher(VBox):
|
|||||||
|
|
||||||
self.bus = bus
|
self.bus = bus
|
||||||
|
|
||||||
|
logger.debug('Running...')
|
||||||
self.label = Label('Looking for Presence Service...')
|
self.label = Label('Looking for Presence Service...')
|
||||||
bus.watch_name_owner(PS_NAME, self.on_name_owner_change)
|
self.errors = ListStore(str)
|
||||||
|
|
||||||
|
errors_tree = TreeView(model=self.errors)
|
||||||
|
errors_tree.insert_column_with_attributes(0, 'Log', CellRendererText(),
|
||||||
|
text=0)
|
||||||
|
scroller = ScrolledWindow()
|
||||||
|
scroller.add(errors_tree)
|
||||||
|
|
||||||
|
self.paned = VPaned()
|
||||||
|
self.paned.pack1(scroller)
|
||||||
|
|
||||||
self.pack_start(self.label, False, False)
|
self.pack_start(self.label, False, False)
|
||||||
self.ps_watcher = None
|
self.pack_end(self.paned)
|
||||||
|
|
||||||
|
bus.watch_name_owner(PS_NAME, self.on_name_owner_change)
|
||||||
|
self.ps_watcher = Label('-')
|
||||||
|
self.paned.pack2(self.ps_watcher)
|
||||||
|
|
||||||
self.show_all()
|
self.show_all()
|
||||||
|
|
||||||
|
def log(self, format, *args):
|
||||||
|
self.errors.append((format % args,))
|
||||||
|
|
||||||
def on_name_owner_change(self, owner):
|
def on_name_owner_change(self, owner):
|
||||||
try:
|
try:
|
||||||
if owner:
|
if owner:
|
||||||
self.label.set_text('Presence Service running: unique name %s'
|
self.label.set_text('Presence Service running: unique name %s'
|
||||||
% owner)
|
% owner)
|
||||||
if self.ps_watcher is not None:
|
if self.ps_watcher is not None:
|
||||||
self.remove(self.ps_watcher)
|
self.paned.remove(self.ps_watcher)
|
||||||
self.ps_watcher = PresenceServiceWatcher(self.bus, owner)
|
self.ps_watcher = PresenceServiceWatcher(self.bus, owner,
|
||||||
self.pack_start(self.ps_watcher)
|
self.log)
|
||||||
|
self.paned.pack2(self.ps_watcher)
|
||||||
self.show_all()
|
self.show_all()
|
||||||
else:
|
else:
|
||||||
self.label.set_text('Presence Service not running')
|
self.label.set_text('Presence Service not running')
|
||||||
if self.ps_watcher is not None:
|
if self.ps_watcher is not None:
|
||||||
self.remove(self.ps_watcher)
|
self.paned.remove(self.ps_watcher)
|
||||||
self.ps_watcher = None
|
self.ps_watcher = Label('-')
|
||||||
|
self.paned.pack2(self.ps_watcher)
|
||||||
except Exception, e:
|
except Exception, e:
|
||||||
logger.warning('%s', e)
|
self.log('ERROR: %s', e)
|
||||||
|
|
||||||
|
|
||||||
class Interface(object):
|
class Interface(object):
|
||||||
|
@ -172,7 +172,7 @@ def create():
|
|||||||
metadata['mtime'] = metadata['ctime']
|
metadata['mtime'] = metadata['ctime']
|
||||||
return DSObject(object_id=None, metadata=metadata, file_path=None)
|
return DSObject(object_id=None, metadata=metadata, file_path=None)
|
||||||
|
|
||||||
def write(ds_object, update_mtime=True, reply_handler=None, error_handler=None):
|
def write(ds_object, update_mtime=True, reply_handler=None, error_handler=None, timeout=-1):
|
||||||
logging.debug('datastore.write')
|
logging.debug('datastore.write')
|
||||||
|
|
||||||
properties = ds_object.metadata.get_dictionary().copy()
|
properties = ds_object.metadata.get_dictionary().copy()
|
||||||
@ -185,7 +185,8 @@ def write(ds_object, update_mtime=True, reply_handler=None, error_handler=None):
|
|||||||
properties,
|
properties,
|
||||||
ds_object.file_path,
|
ds_object.file_path,
|
||||||
reply_handler=reply_handler,
|
reply_handler=reply_handler,
|
||||||
error_handler=error_handler)
|
error_handler=error_handler,
|
||||||
|
timeout=timeout)
|
||||||
else:
|
else:
|
||||||
ds_object.object_id = dbus_helpers.create(properties,
|
ds_object.object_id = dbus_helpers.create(properties,
|
||||||
ds_object.file_path)
|
ds_object.file_path)
|
||||||
|
@ -37,12 +37,13 @@ def create(properties, filename):
|
|||||||
logging.debug('dbus_helpers.create: ' + object_id)
|
logging.debug('dbus_helpers.create: ' + object_id)
|
||||||
return object_id
|
return object_id
|
||||||
|
|
||||||
def update(uid, properties, filename, reply_handler=None, error_handler=None):
|
def update(uid, properties, filename, reply_handler=None, error_handler=None, timeout=-1):
|
||||||
logging.debug('dbus_helpers.update: %s, %s, %s' % (uid, filename, properties))
|
logging.debug('dbus_helpers.update: %s, %s, %s' % (uid, filename, properties))
|
||||||
if reply_handler and error_handler:
|
if reply_handler and error_handler:
|
||||||
_data_store.update(uid, dbus.Dictionary(properties), filename,
|
_data_store.update(uid, dbus.Dictionary(properties), filename,
|
||||||
reply_handler=reply_handler,
|
reply_handler=reply_handler,
|
||||||
error_handler=error_handler)
|
error_handler=error_handler,
|
||||||
|
timeout=timeout)
|
||||||
else:
|
else:
|
||||||
_data_store.update(uid, dbus.Dictionary(properties), filename)
|
_data_store.update(uid, dbus.Dictionary(properties), filename)
|
||||||
|
|
||||||
|
@ -103,6 +103,15 @@ def _get_logs_dir():
|
|||||||
return logs_dir
|
return logs_dir
|
||||||
|
|
||||||
def start(module_id):
|
def start(module_id):
|
||||||
|
# Only log if logging is set up for the activity
|
||||||
|
module_key = module_id.upper() + "_DEBUG"
|
||||||
|
emulator = False
|
||||||
|
if os.environ.has_key("SUGAR_EMULATOR"):
|
||||||
|
if os.environ["SUGAR_EMULATOR"] == "yes":
|
||||||
|
emulator = True
|
||||||
|
if not os.environ.has_key(module_key) and not emulator:
|
||||||
|
return
|
||||||
|
|
||||||
log_writer = LogWriter(module_id)
|
log_writer = LogWriter(module_id)
|
||||||
|
|
||||||
root_logger = logging.getLogger('')
|
root_logger = logging.getLogger('')
|
||||||
|
Loading…
Reference in New Issue
Block a user