2006-10-15 01:08:44 +02:00
|
|
|
# Copyright (C) 2006, Red Hat, Inc.
|
|
|
|
#
|
|
|
|
# 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.
|
|
|
|
|
2006-08-09 18:29:33 +02:00
|
|
|
import os
|
2006-08-11 17:05:06 +02:00
|
|
|
import logging
|
2006-08-09 18:29:33 +02:00
|
|
|
|
2006-05-12 08:34:20 +02:00
|
|
|
import dbus
|
|
|
|
import dbus.service
|
2006-07-15 12:31:06 +02:00
|
|
|
import gtk
|
|
|
|
import gobject
|
2006-12-24 02:51:37 +01:00
|
|
|
import datetime
|
2006-05-12 08:34:20 +02:00
|
|
|
|
2006-12-18 14:24:28 +01:00
|
|
|
from sugar.presence import PresenceService
|
2006-12-24 02:51:37 +01:00
|
|
|
from sugar.datastore import datastore
|
2006-09-04 21:34:54 +02:00
|
|
|
from sugar import activity
|
2006-12-18 14:05:34 +01:00
|
|
|
from sugar import env
|
2006-07-09 17:37:54 +02:00
|
|
|
import sugar.util
|
2006-07-06 16:06:07 +02:00
|
|
|
|
2006-08-09 12:57:42 +02:00
|
|
|
ACTIVITY_SERVICE_NAME = "org.laptop.Activity"
|
|
|
|
ACTIVITY_SERVICE_PATH = "/org/laptop/Activity"
|
|
|
|
ACTIVITY_INTERFACE = "org.laptop.Activity"
|
2006-06-02 20:52:20 +02:00
|
|
|
|
2006-08-09 18:29:33 +02:00
|
|
|
def get_service_name(xid):
|
2006-12-04 20:12:24 +01:00
|
|
|
return ACTIVITY_SERVICE_NAME + '%d' % xid
|
2006-08-09 18:29:33 +02:00
|
|
|
|
|
|
|
def get_object_path(xid):
|
2006-12-04 20:12:24 +01:00
|
|
|
return ACTIVITY_SERVICE_PATH + "/%s" % xid
|
2006-08-09 18:29:33 +02:00
|
|
|
|
2006-06-02 20:52:20 +02:00
|
|
|
class ActivityDbusService(dbus.service.Object):
|
2006-12-04 20:12:24 +01:00
|
|
|
"""Base dbus service object that each Activity uses to export dbus methods.
|
|
|
|
|
|
|
|
The dbus service is separate from the actual Activity object so that we can
|
|
|
|
tightly control what stuff passes through the dbus python bindings."""
|
|
|
|
|
2006-12-18 14:24:28 +01:00
|
|
|
def __init__(self, activity):
|
|
|
|
xid = activity.window.xid
|
|
|
|
bus = dbus.SessionBus()
|
|
|
|
bus_name = dbus.service.BusName(get_service_name(xid), bus=bus)
|
|
|
|
dbus.service.Object.__init__(self, bus_name, get_object_path(xid))
|
|
|
|
|
2006-12-04 20:12:24 +01:00
|
|
|
self._activity = activity
|
2006-12-18 14:24:28 +01:00
|
|
|
self._pservice = PresenceService.get_instance()
|
2006-12-04 20:12:24 +01:00
|
|
|
|
|
|
|
@dbus.service.method(ACTIVITY_INTERFACE)
|
2006-12-18 14:24:28 +01:00
|
|
|
def start(self):
|
|
|
|
"""Start the activity."""
|
|
|
|
self._activity.start()
|
2006-12-04 20:12:24 +01:00
|
|
|
|
|
|
|
@dbus.service.method(ACTIVITY_INTERFACE)
|
|
|
|
def join(self, activity_ps_path):
|
2006-12-18 14:24:28 +01:00
|
|
|
"""Join the activity specified by its presence service path."""
|
2006-12-04 20:12:24 +01:00
|
|
|
activity_ps = self._pservice.get(activity_ps_path)
|
|
|
|
return self._activity.join(activity_ps)
|
|
|
|
|
2006-12-18 14:24:28 +01:00
|
|
|
@dbus.service.method(ACTIVITY_INTERFACE)
|
|
|
|
def share(self):
|
|
|
|
"""Called by the shell to request the activity to share itself on the network."""
|
|
|
|
self._activity.share()
|
|
|
|
|
2006-12-04 20:12:24 +01:00
|
|
|
@dbus.service.method(ACTIVITY_INTERFACE)
|
|
|
|
def get_id(self):
|
|
|
|
"""Get the activity identifier"""
|
|
|
|
return self._activity.get_id()
|
|
|
|
|
|
|
|
@dbus.service.method(ACTIVITY_INTERFACE)
|
|
|
|
def get_type(self):
|
|
|
|
"""Get the activity type"""
|
|
|
|
return self._activity.get_type()
|
|
|
|
|
|
|
|
@dbus.service.method(ACTIVITY_INTERFACE)
|
|
|
|
def get_shared(self):
|
|
|
|
"""Returns True if the activity is shared on the mesh."""
|
|
|
|
return self._activity.get_shared()
|
|
|
|
|
|
|
|
@dbus.service.method(ACTIVITY_INTERFACE,
|
2006-12-21 12:37:02 +01:00
|
|
|
in_signature="sas", out_signature="b")
|
2006-12-04 20:12:24 +01:00
|
|
|
def execute(self, command, args):
|
2006-12-21 12:37:02 +01:00
|
|
|
return self._activity.execute(command, args)
|
2006-08-09 15:53:10 +02:00
|
|
|
|
2006-07-08 15:47:51 +02:00
|
|
|
class Activity(gtk.Window):
|
2006-12-04 20:12:24 +01:00
|
|
|
"""Base Activity class that all other Activities derive from."""
|
|
|
|
|
|
|
|
def __init__(self):
|
|
|
|
gtk.Window.__init__(self)
|
|
|
|
|
2006-12-24 02:51:37 +01:00
|
|
|
self.connect('destroy', self._destroy_cb)
|
|
|
|
self.connect('notify::title', self._title_changed_cb)
|
2006-12-04 20:12:24 +01:00
|
|
|
|
|
|
|
self._shared = False
|
|
|
|
self._activity_id = None
|
|
|
|
self._service = None
|
2006-12-24 02:51:37 +01:00
|
|
|
self._journal_object = None
|
2006-12-18 14:24:28 +01:00
|
|
|
self._pservice = PresenceService.get_instance()
|
2006-12-04 20:12:24 +01:00
|
|
|
|
2006-12-20 00:53:27 +01:00
|
|
|
self.realize()
|
2006-12-04 20:12:24 +01:00
|
|
|
|
|
|
|
group = gtk.Window()
|
|
|
|
group.realize()
|
|
|
|
self.window.set_group(group.window)
|
|
|
|
|
2006-12-18 14:24:28 +01:00
|
|
|
self._bus = ActivityDbusService(self)
|
2006-12-04 20:12:24 +01:00
|
|
|
|
2006-12-18 14:24:28 +01:00
|
|
|
def start(self):
|
|
|
|
"""Start the activity."""
|
2006-12-20 13:24:37 +01:00
|
|
|
if self._activity_id != None:
|
|
|
|
logging.warning('The activity has been already started.')
|
|
|
|
return
|
|
|
|
|
2006-12-18 14:24:28 +01:00
|
|
|
self._activity_id = sugar.util.unique_id()
|
2006-12-24 02:51:37 +01:00
|
|
|
|
|
|
|
ds = datastore.get_instance()
|
2006-12-24 02:59:20 +01:00
|
|
|
#self._journal_object = ds.create('', {}, self._activity_id)
|
|
|
|
#
|
|
|
|
#date = datetime.datetime.now()
|
|
|
|
#self._journal_jobject.set_properties({'date' : date,
|
|
|
|
# 'title' : self.get_title()})
|
2006-12-24 02:51:37 +01:00
|
|
|
|
2006-12-20 00:53:27 +01:00
|
|
|
self.present()
|
2006-12-04 20:12:24 +01:00
|
|
|
|
2006-12-24 02:51:37 +01:00
|
|
|
def get_journal_object(self):
|
|
|
|
"""Returns the journal object associated with the activity."""
|
|
|
|
return self._journal_object
|
|
|
|
|
2006-12-04 20:12:24 +01:00
|
|
|
def get_type(self):
|
|
|
|
"""Gets the activity type."""
|
2006-12-18 13:56:41 +01:00
|
|
|
return env.get_bundle_service_name()
|
2006-12-04 20:12:24 +01:00
|
|
|
|
|
|
|
def get_default_type(self):
|
2006-12-18 14:05:34 +01:00
|
|
|
"""Gets the type of the default activity network service"""
|
2006-12-20 13:43:54 +01:00
|
|
|
return env.get_bundle_default_type()
|
2006-12-04 20:12:24 +01:00
|
|
|
|
|
|
|
def get_shared(self):
|
|
|
|
"""Returns TRUE if the activity is shared on the mesh."""
|
|
|
|
return self._shared
|
|
|
|
|
|
|
|
def get_id(self):
|
|
|
|
"""Get the unique activity identifier."""
|
|
|
|
return self._activity_id
|
|
|
|
|
|
|
|
def join(self, activity_ps):
|
|
|
|
"""Join an activity shared on the network."""
|
2006-12-20 13:24:37 +01:00
|
|
|
if self._activity_id != None:
|
|
|
|
logging.warning('The activity has been already started.')
|
|
|
|
return
|
|
|
|
|
2006-12-04 20:12:24 +01:00
|
|
|
self._shared = True
|
|
|
|
self._activity_id = activity_ps.get_id()
|
|
|
|
|
|
|
|
# Publish the default service, it's a copy of
|
|
|
|
# one of those we found on the network.
|
2006-12-18 13:56:41 +01:00
|
|
|
default_type = self.get_default_type()
|
|
|
|
services = activity_ps.get_services_of_type(default_type)
|
2006-12-04 20:12:24 +01:00
|
|
|
if len(services) > 0:
|
|
|
|
service = services[0]
|
|
|
|
addr = service.get_address()
|
|
|
|
port = service.get_port()
|
|
|
|
properties = service.get_published_values()
|
|
|
|
self._service = self._pservice.share_activity(
|
2006-12-18 13:56:41 +01:00
|
|
|
self, default_type, properties, addr, port)
|
2006-12-04 20:12:24 +01:00
|
|
|
else:
|
|
|
|
logging.error('Cannot join the activity')
|
|
|
|
|
2006-12-24 02:51:37 +01:00
|
|
|
ds = datastore.get_instance()
|
|
|
|
self._journal_object = ds.get_activity_object(self._activity_id)
|
|
|
|
|
2006-12-20 00:53:27 +01:00
|
|
|
self.present()
|
|
|
|
|
2006-12-04 20:12:24 +01:00
|
|
|
def share(self):
|
|
|
|
"""Share the activity on the network."""
|
|
|
|
logging.debug('Share activity %s on the network.' % self.get_id())
|
|
|
|
|
2006-12-18 13:56:41 +01:00
|
|
|
default_type = self.get_default_type()
|
|
|
|
self._service = self._pservice.share_activity(self, default_type)
|
2006-12-04 20:12:24 +01:00
|
|
|
self._shared = True
|
|
|
|
|
|
|
|
def execute(self, command, args):
|
|
|
|
"""Execute the given command with args"""
|
2006-12-21 12:37:02 +01:00
|
|
|
return False
|
2006-12-04 20:12:24 +01:00
|
|
|
|
2006-12-24 02:51:37 +01:00
|
|
|
def _destroy_cb(self, window):
|
2006-12-04 20:12:24 +01:00
|
|
|
if self._bus:
|
|
|
|
del self._bus
|
|
|
|
self._bus = None
|
|
|
|
if self._service:
|
|
|
|
self._pservice.unregister_service(self._service)
|
2006-12-24 02:51:37 +01:00
|
|
|
|
|
|
|
def _title_changed_cb(self, window, spec):
|
|
|
|
jobject = self.get_journal_object()
|
|
|
|
if jobject:
|
|
|
|
jobject.set_properties({'title' : self.props.title})
|