Merge branch 'master' of git+ssh://dev.laptop.org/git/sugar

Conflicts:

	NEWS
This commit is contained in:
Marco Pesenti Gritti 2007-07-26 10:08:47 +02:00
commit 7e45c5446d
5 changed files with 346 additions and 68 deletions

4
NEWS
View File

@ -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

View File

@ -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):

View File

@ -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)

View File

@ -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)

View File

@ -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('')