Merge branch 'master' of git+ssh://dev.laptop.org/git/sugar
This commit is contained in:
		
						commit
						aa71d354b2
					
				@ -104,7 +104,7 @@ class Buddy(object):
 | 
				
			|||||||
	"""Represents another person on the network and keeps track of the
 | 
						"""Represents another person on the network and keeps track of the
 | 
				
			||||||
	activities and resources they make available for sharing."""
 | 
						activities and resources they make available for sharing."""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	def __init__(self, bus_name, object_id, service):
 | 
						def __init__(self, bus_name, object_id, service, icon_cache):
 | 
				
			||||||
		if not bus_name:
 | 
							if not bus_name:
 | 
				
			||||||
			raise ValueError("DBus bus name must be valid")
 | 
								raise ValueError("DBus bus name must be valid")
 | 
				
			||||||
		if not object_id or type(object_id) != type(1):
 | 
							if not object_id or type(object_id) != type(1):
 | 
				
			||||||
@ -137,6 +137,8 @@ class Buddy(object):
 | 
				
			|||||||
		if service is not None:
 | 
							if service is not None:
 | 
				
			||||||
			self.add_service(service)
 | 
								self.add_service(service)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							self._icon_cache = icon_cache
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	def object_path(self):
 | 
						def object_path(self):
 | 
				
			||||||
		return dbus.ObjectPath(self._object_path)
 | 
							return dbus.ObjectPath(self._object_path)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -148,25 +150,37 @@ class Buddy(object):
 | 
				
			|||||||
		if result_status == network.RESULT_SUCCESS:
 | 
							if result_status == network.RESULT_SUCCESS:
 | 
				
			||||||
			if icon and len(icon):
 | 
								if icon and len(icon):
 | 
				
			||||||
				icon = base64.b64decode(icon)
 | 
									icon = base64.b64decode(icon)
 | 
				
			||||||
				logging.debug("Buddy icon for '%s' is size %d" % (self._nick_name, len(icon)))
 | 
					 | 
				
			||||||
				self._set_icon(icon)
 | 
									self._set_icon(icon)
 | 
				
			||||||
 | 
									self._icon_cache.add_icon(icon)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		if (result_status == network.RESULT_FAILED or not icon) and self._icon_tries < 3:
 | 
							if (result_status == network.RESULT_FAILED or not icon) and self._icon_tries < 3:
 | 
				
			||||||
			self._icon_tries = self._icon_tries + 1
 | 
								self._icon_tries = self._icon_tries + 1
 | 
				
			||||||
			logging.debug("Failed to retrieve buddy icon for '%s' on try %d of %d" % (self._nick_name, \
 | 
								logging.debug("Failed to retrieve buddy icon for '%s' on try %d of %d" % (self._nick_name, \
 | 
				
			||||||
					self._icon_tries, 3))
 | 
										self._icon_tries, 3))
 | 
				
			||||||
			gobject.timeout_add(1000, self._request_buddy_icon, service)
 | 
								gobject.timeout_add(1000, self._get_buddy_icon, service)
 | 
				
			||||||
		return False
 | 
							return False
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	def _request_buddy_icon(self, service):
 | 
						def _get_buddy_icon(self, service, retry=False):
 | 
				
			||||||
		"""Contact the buddy to retrieve the buddy icon."""
 | 
							"""Get the buddy's icon.  Check the cache first, if its
 | 
				
			||||||
 | 
							not there get the icon from the buddy over the network."""
 | 
				
			||||||
 | 
							if retry != True:
 | 
				
			||||||
 | 
								# Only hit the cache once
 | 
				
			||||||
 | 
								icon_hash = service.get_one_property('icon-hash')
 | 
				
			||||||
 | 
								if icon_hash is not None:
 | 
				
			||||||
 | 
									icon = self._icon_cache.get_icon(icon_hash)
 | 
				
			||||||
 | 
									if icon:
 | 
				
			||||||
 | 
										logging.debug("%s: icon cache hit for %s." % (self._nick_name, icon_hash))
 | 
				
			||||||
 | 
										self._set_icon(icon)
 | 
				
			||||||
 | 
										return False
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							logging.debug("%s: icon cache miss, adding icon to cache." % self._nick_name)
 | 
				
			||||||
		from sugar.p2p import Stream
 | 
							from sugar.p2p import Stream
 | 
				
			||||||
		buddy_stream = Stream.Stream.new_from_service(service, start_reader=False)
 | 
							buddy_stream = Stream.Stream.new_from_service(service, start_reader=False)
 | 
				
			||||||
		writer = buddy_stream.new_writer(service)
 | 
							writer = buddy_stream.new_writer(service)
 | 
				
			||||||
		success = writer.custom_request("get_buddy_icon", self._request_buddy_icon_cb, service)
 | 
							success = writer.custom_request("get_buddy_icon", self._request_buddy_icon_cb, service)
 | 
				
			||||||
		if not success:
 | 
							if not success:
 | 
				
			||||||
			del writer, buddy_stream
 | 
								del writer, buddy_stream
 | 
				
			||||||
			gobject.timeout_add(1000, self._request_buddy_icon, service)
 | 
								gobject.timeout_add(1000, self._get_buddy_icon, service, True)
 | 
				
			||||||
		return False
 | 
							return False
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	def _get_service_key(self, service):
 | 
						def _get_service_key(self, service):
 | 
				
			||||||
@ -210,8 +224,7 @@ class Buddy(object):
 | 
				
			|||||||
			# A buddy isn't valid until its official presence
 | 
								# A buddy isn't valid until its official presence
 | 
				
			||||||
			# service has been found and resolved
 | 
								# service has been found and resolved
 | 
				
			||||||
			self._valid = True
 | 
								self._valid = True
 | 
				
			||||||
			logging.debug('Requesting buddy icon %s' % self._nick_name)
 | 
								self._get_buddy_icon(service)
 | 
				
			||||||
			self._request_buddy_icon(service)
 | 
					 | 
				
			||||||
			self._color = service.get_one_property(_BUDDY_KEY_COLOR)
 | 
								self._color = service.get_one_property(_BUDDY_KEY_COLOR)
 | 
				
			||||||
			if self._color:
 | 
								if self._color:
 | 
				
			||||||
				self._dbus_helper.PropertyChanged([_BUDDY_KEY_COLOR])
 | 
									self._dbus_helper.PropertyChanged([_BUDDY_KEY_COLOR])
 | 
				
			||||||
@ -343,8 +356,8 @@ class Buddy(object):
 | 
				
			|||||||
class Owner(Buddy):
 | 
					class Owner(Buddy):
 | 
				
			||||||
	"""Class representing the owner of the machine.  This is the client
 | 
						"""Class representing the owner of the machine.  This is the client
 | 
				
			||||||
	portion of the Owner, paired with the server portion in Owner.py."""
 | 
						portion of the Owner, paired with the server portion in Owner.py."""
 | 
				
			||||||
	def __init__(self, ps, bus_name, object_id):
 | 
						def __init__(self, ps, bus_name, object_id, icon_cache):
 | 
				
			||||||
		Buddy.__init__(self, bus_name, object_id, None)
 | 
							Buddy.__init__(self, bus_name, object_id, None, icon_cache)
 | 
				
			||||||
		self._nick_name = env.get_nick_name()
 | 
							self._nick_name = env.get_nick_name()
 | 
				
			||||||
		self._color = env.get_color()
 | 
							self._color = env.get_color()
 | 
				
			||||||
		self._ps = ps
 | 
							self._ps = ps
 | 
				
			||||||
 | 
				
			|||||||
							
								
								
									
										61
									
								
								services/presence/BuddyIconCache.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										61
									
								
								services/presence/BuddyIconCache.py
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,61 @@
 | 
				
			|||||||
 | 
					import os, time, md5
 | 
				
			||||||
 | 
					from sugar import env
 | 
				
			||||||
 | 
					from sugar import util
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class BuddyIconCache(object):
 | 
				
			||||||
 | 
						"""Caches icons on disk and finds them based on md5 hash."""
 | 
				
			||||||
 | 
						def __init__(self):
 | 
				
			||||||
 | 
							ppath = env.get_profile_path()
 | 
				
			||||||
 | 
							self._cachepath = os.path.join(ppath, "cache", "buddy-icons")
 | 
				
			||||||
 | 
							if not os.path.exists(self._cachepath):
 | 
				
			||||||
 | 
								os.makedirs(self._cachepath)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							self._cache = {}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							# Read all cached icons and their sums
 | 
				
			||||||
 | 
							for fname in os.listdir(self._cachepath):
 | 
				
			||||||
 | 
								m = md5.new()
 | 
				
			||||||
 | 
								data = self._get_icon_data(fname)
 | 
				
			||||||
 | 
								if len(data) == 0:
 | 
				
			||||||
 | 
									continue
 | 
				
			||||||
 | 
								m.update(data)
 | 
				
			||||||
 | 
								printable_hash = util.printable_hash(m.digest())
 | 
				
			||||||
 | 
								self._cache[printable_hash] = fname
 | 
				
			||||||
 | 
								del m
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						def _get_icon_data(self, fname):
 | 
				
			||||||
 | 
							fd = open(os.path.join(self._cachepath, fname), "r")
 | 
				
			||||||
 | 
							data = fd.read()
 | 
				
			||||||
 | 
							fd.close()
 | 
				
			||||||
 | 
							del fd
 | 
				
			||||||
 | 
							return data
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						def get_icon(self, printable_hash):
 | 
				
			||||||
 | 
							if type(printable_hash) != type(u""):
 | 
				
			||||||
 | 
								raise RuntimeError("printable_hash must be a unicode string.")
 | 
				
			||||||
 | 
							try:
 | 
				
			||||||
 | 
								fname = self._cache[printable_hash]
 | 
				
			||||||
 | 
								return self._get_icon_data(fname)
 | 
				
			||||||
 | 
							except KeyError:
 | 
				
			||||||
 | 
								pass
 | 
				
			||||||
 | 
							return None
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						def add_icon(self, icon_data):
 | 
				
			||||||
 | 
							if len(icon_data) == 0:
 | 
				
			||||||
 | 
								return
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							m = md5.new()
 | 
				
			||||||
 | 
							m.update(icon_data)
 | 
				
			||||||
 | 
							printable_hash = util.printable_hash(m.digest())
 | 
				
			||||||
 | 
							if self._cache.has_key(printable_hash):
 | 
				
			||||||
 | 
								del m
 | 
				
			||||||
 | 
								return
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							# Write the icon to disk and add an entry to our cache for it
 | 
				
			||||||
 | 
							m.update(time.asctime())
 | 
				
			||||||
 | 
							fname = util.printable_hash(m.digest())
 | 
				
			||||||
 | 
							fd = open(os.path.join(self._cachepath, fname), "w")
 | 
				
			||||||
 | 
							fd.write(icon_data)
 | 
				
			||||||
 | 
							fd.close()
 | 
				
			||||||
 | 
							self._cache[printable_hash] = fname
 | 
				
			||||||
 | 
							del m
 | 
				
			||||||
@ -10,6 +10,7 @@ sugar_PYTHON =				\
 | 
				
			|||||||
	__init__.py			\
 | 
						__init__.py			\
 | 
				
			||||||
	Activity.py			\
 | 
						Activity.py			\
 | 
				
			||||||
	Buddy.py			\
 | 
						Buddy.py			\
 | 
				
			||||||
 | 
						BuddyIconCache.py	\
 | 
				
			||||||
	PresenceService.py	\
 | 
						PresenceService.py	\
 | 
				
			||||||
	Service.py
 | 
						Service.py
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
				
			|||||||
@ -5,6 +5,7 @@ import Activity
 | 
				
			|||||||
import random
 | 
					import random
 | 
				
			||||||
import logging
 | 
					import logging
 | 
				
			||||||
from sugar import util
 | 
					from sugar import util
 | 
				
			||||||
 | 
					import BuddyIconCache
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
_SA_UNRESOLVED = 0
 | 
					_SA_UNRESOLVED = 0
 | 
				
			||||||
@ -302,9 +303,11 @@ class PresenceService(object):
 | 
				
			|||||||
		self._bus_name = dbus.service.BusName(_PRESENCE_SERVICE, bus=self._session_bus)		
 | 
							self._bus_name = dbus.service.BusName(_PRESENCE_SERVICE, bus=self._session_bus)		
 | 
				
			||||||
		self._dbus_helper = PresenceServiceDBusHelper(self, self._bus_name)
 | 
							self._dbus_helper = PresenceServiceDBusHelper(self, self._bus_name)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							self._icon_cache = BuddyIconCache.BuddyIconCache()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		# Our owner object
 | 
							# Our owner object
 | 
				
			||||||
		objid = self._get_next_object_id()
 | 
							objid = self._get_next_object_id()
 | 
				
			||||||
		self._owner = Buddy.Owner(self, self._bus_name, objid)
 | 
							self._owner = Buddy.Owner(self, self._bus_name, objid, self._icon_cache)
 | 
				
			||||||
		self._buddies[self._owner.get_name()] = self._owner
 | 
							self._buddies[self._owner.get_name()] = self._owner
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		self._started = False
 | 
							self._started = False
 | 
				
			||||||
@ -423,7 +426,7 @@ class PresenceService(object):
 | 
				
			|||||||
		except KeyError:
 | 
							except KeyError:
 | 
				
			||||||
			source_addr = service.get_source_address()
 | 
								source_addr = service.get_source_address()
 | 
				
			||||||
			objid = self._get_next_object_id()
 | 
								objid = self._get_next_object_id()
 | 
				
			||||||
			buddy = Buddy.Buddy(self._bus_name, objid, service)
 | 
								buddy = Buddy.Buddy(self._bus_name, objid, service, self._icon_cache)
 | 
				
			||||||
			self._buddies[name] = buddy
 | 
								self._buddies[name] = buddy
 | 
				
			||||||
			self._dbus_helper.ServiceAppeared(service.object_path())
 | 
								self._dbus_helper.ServiceAppeared(service.object_path())
 | 
				
			||||||
		if not buddy_was_valid and buddy.is_valid():
 | 
							if not buddy_was_valid and buddy.is_valid():
 | 
				
			||||||
 | 
				
			|||||||
@ -8,6 +8,7 @@ from sugar import env
 | 
				
			|||||||
import logging
 | 
					import logging
 | 
				
			||||||
from sugar.p2p import Stream
 | 
					from sugar.p2p import Stream
 | 
				
			||||||
from sugar.presence import PresenceService
 | 
					from sugar.presence import PresenceService
 | 
				
			||||||
 | 
					from sugar import util
 | 
				
			||||||
from model.Invites import Invites
 | 
					from model.Invites import Invites
 | 
				
			||||||
import dbus
 | 
					import dbus
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -24,11 +25,17 @@ class ShellOwner(object):
 | 
				
			|||||||
		user_dir = profile.get_path()
 | 
							user_dir = profile.get_path()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		self._icon = None
 | 
							self._icon = None
 | 
				
			||||||
 | 
							self._icon_hash = ""
 | 
				
			||||||
		for fname in os.listdir(user_dir):
 | 
							for fname in os.listdir(user_dir):
 | 
				
			||||||
			if not fname.startswith("buddy-icon."):
 | 
								if not fname.startswith("buddy-icon."):
 | 
				
			||||||
				continue
 | 
									continue
 | 
				
			||||||
			fd = open(os.path.join(user_dir, fname), "r")
 | 
								fd = open(os.path.join(user_dir, fname), "r")
 | 
				
			||||||
			self._icon = fd.read()
 | 
								self._icon = fd.read()
 | 
				
			||||||
 | 
								if self._icon:
 | 
				
			||||||
 | 
									# Get the icon's hash
 | 
				
			||||||
 | 
									import md5, binascii
 | 
				
			||||||
 | 
									digest = md5.new(self._icon).digest()
 | 
				
			||||||
 | 
									self._icon_hash = util.printable_hash(digest)
 | 
				
			||||||
			fd.close()
 | 
								fd.close()
 | 
				
			||||||
			break
 | 
								break
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -49,7 +56,7 @@ class ShellOwner(object):
 | 
				
			|||||||
	def announce(self):
 | 
						def announce(self):
 | 
				
			||||||
		# Create and announce our presence
 | 
							# Create and announce our presence
 | 
				
			||||||
		color = conf.get_profile().get_color()
 | 
							color = conf.get_profile().get_color()
 | 
				
			||||||
		props = {'color':color.to_string()}
 | 
							props = {'color': color.to_string(), 'icon-hash': self._icon_hash}
 | 
				
			||||||
		self._service = self._pservice.register_service(self._nick,
 | 
							self._service = self._pservice.register_service(self._nick,
 | 
				
			||||||
				PRESENCE_SERVICE_TYPE, properties=props)
 | 
									PRESENCE_SERVICE_TYPE, properties=props)
 | 
				
			||||||
		logging.debug("Owner '%s' using port %d" % (self._nick, self._service.get_port()))
 | 
							logging.debug("Owner '%s' using port %d" % (self._nick, self._service.get_port()))
 | 
				
			||||||
 | 
				
			|||||||
		Loading…
	
		Reference in New Issue
	
	Block a user