Web   ·   Wiki   ·   Activities   ·   Blog   ·   Lists   ·   Chat   ·   Meeting   ·   Bugs   ·   Git   ·   Translate   ·   Archive   ·   People   ·   Donate
summaryrefslogtreecommitdiffstats
path: root/activity.py
diff options
context:
space:
mode:
authorChris Ball <cjb@laptop.org>2007-08-16 20:01:06 (GMT)
committer Chris Ball <cjb@laptop.org>2007-08-16 20:01:06 (GMT)
commit7a592b7ec2384f6dcd54aeee61e592a82af79839 (patch)
tree476cda17dd7162954e0421b95c61eaa8fb208e08 /activity.py
Initial import. Plans for the future, from FIXMEs:
# FIXME: move to gtksourceview2 # FIXME: Need a sidebar to show different files. # FIXME: Need a scrollbar for the output window. # FIXME: We're losing an odd race here
Diffstat (limited to 'activity.py')
-rw-r--r--activity.py378
1 files changed, 378 insertions, 0 deletions
diff --git a/activity.py b/activity.py
new file mode 100644
index 0000000..4a33d36
--- /dev/null
+++ b/activity.py
@@ -0,0 +1,378 @@
+# Copyright 2007 Chris Ball, based on Collabora's "hellomesh" demo.
+#
+# This program is free software; you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation, either version 2 of the License, or
+# (at your option) any later version.
+#
+# This program 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 General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with this program; if not, write to the Free Software
+# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
+
+"""Pippy Activity: A simple Python programming activity ."""
+
+import gtksourceview # FIXME: move to gtksourceview2
+import gtk
+import logging
+import telepathy
+import telepathy.client
+import hippo
+import pango
+import vte
+from sys import executable
+
+from dbus import Interface
+from dbus.service import method, signal
+from dbus.gobject_service import ExportedGObject
+
+from sugar.activity.activity import Activity, ActivityToolbox
+from sugar.presence import presenceservice
+
+# will eventually be imported from sugar
+from tubeconn import TubeConnection
+
+SERVICE = "org.laptop.Pippy"
+IFACE = SERVICE
+PATH = "/org/laptop/Pippy"
+
+class PippyActivity(Activity):
+ """Pippy Activity as specified in activity.info"""
+ def __init__(self, handle):
+ """Set up the Pippy activity."""
+ Activity.__init__(self, handle)
+ self.set_title('Pippy Activity')
+ self._logger = logging.getLogger('pippy-activity')
+
+ # top toolbar with share and close buttons:
+ toolbox = ActivityToolbox(self)
+ self.set_toolbox(toolbox)
+ toolbox.show()
+
+ # Hippo Canvas:
+ # FIXME: Need a sidebar to show different files.
+ vbox = hippo.CanvasBox(spacing=4,
+ orientation=hippo.ORIENTATION_VERTICAL)
+
+ #self.main_panel = hippo.CanvasBox(spacing=4,
+ # orientation=hippo.ORIENTATION_VERTICAL)
+ #self.entry = gtk.Entry()
+ #self.main_panel.append(hippo.CanvasWidget(widget=self.entry))
+ #hbox.append(self.main_panel, hippo.PACK_EXPAND)
+
+ win = gtk.Window()
+ self.text_buffer = gtksourceview.SourceBuffer()
+ lm = gtksourceview.SourceLanguagesManager()
+
+ lang = lm.get_language_from_mime_type("text/x-python")
+ self.text_buffer.set_language(lang)
+ self.text_buffer.set_highlight(True)
+
+ self.text_view = gtksourceview.SourceView(self.text_buffer)
+ self.text_view.set_size_request(1200, 300)
+ self.text_view.set_editable(True)
+ self.text_view.set_cursor_visible(True)
+ self.text_view.set_show_line_numbers(True)
+ self.text_view.modify_font(pango.FontDescription("Monospace 12"))
+
+ # The GTK source view window
+ codesw = gtk.ScrolledWindow()
+ codesw.set_policy(gtk.POLICY_AUTOMATIC,
+ gtk.POLICY_AUTOMATIC)
+ codesw.set_shadow_type(gtk.SHADOW_IN)
+ codesw.add(self.text_view)
+ vbox.append(hippo.CanvasWidget(widget=codesw), hippo.PACK_EXPAND)
+
+ # The "go" button
+ gobutton = gtk.Button(label="Run!")
+ gobutton.add(self.text_view)
+ gobutton.connect('clicked', self.gobutton_cb)
+ gobutton.set_size_request(1200, 50)
+ vbox.append(hippo.CanvasWidget(widget=gobutton))
+
+ # The vte python window
+ self._vte = vte.Terminal()
+ self._vte.set_size(30, 5)
+ self._vte.set_size_request(200, 150)
+ self._vte.show()
+
+ # FIXME: Need a scrollbar for the output window.
+ #self._scrollbar = gtk.VScrollbar(self._vte.get_adjustment())
+ #self._scrollbar.show()
+ #self.pack_start(self._scrollbar, False, False, 0)
+
+ font = 'Monospace 12'
+ self._vte.set_font(pango.FontDescription(font))
+ self._vte.set_colors(gtk.gdk.color_parse ('#000000'),
+ gtk.gdk.color_parse ('#E7E7E7'),
+ [])
+
+ vbox.append(hippo.CanvasWidget(widget=self._vte), hippo.PACK_EXPAND)
+
+ canvas = hippo.Canvas()
+ canvas.set_root(vbox)
+ self.set_canvas(canvas)
+ self.show_all()
+
+ self.hellotube = None
+
+ # get the Presence Service
+ self.pservice = presenceservice.get_instance()
+ name, path = self.pservice.get_preferred_connection()
+ self.tp_conn_name = name
+ self.tp_conn_path = path
+ self.conn = telepathy.client.Connection(name, path)
+ self.initiating = None
+
+ self.connect('shared', self._shared_cb)
+
+ # Buddy object for you
+ owner = self.pservice.get_owner()
+ self.owner = owner
+
+ if self._shared_activity:
+ # we are joining the activity
+ self.connect('joined', self._joined_cb)
+ self._shared_activity.connect('buddy-joined',
+ self._buddy_joined_cb)
+ self._shared_activity.connect('buddy-left',
+ self._buddy_left_cb)
+ if self.get_shared():
+ # we've already joined
+ self._joined_cb()
+
+ def gobutton_cb(self, button):
+ #self._vte.reset(True, True)
+ self._vte.feed("\x1B[H\x1B[J")
+
+ # FIXME: We're losing an odd race here
+ # gtk.main_iteration(block=False)
+
+ start, end = self.text_buffer.get_bounds()
+ text = self.text_buffer.get_text(start, end)
+
+ file = open('/tmp/pippy.py', 'w', 0)
+ for line in text:
+ file.write(line)
+ file.close()
+
+ pid = self._vte.fork_command(sys.executable, ["python", "/tmp/pippy.py"])
+
+ def _shared_cb(self, activity):
+ self._logger.debug('My activity was shared')
+ self.initiating = True
+ self._setup()
+
+ for buddy in self._shared_activity.get_joined_buddies():
+ self._logger.debug('Buddy %s is already in the activity' %
+ buddy.props.nick)
+
+ self._shared_activity.connect('buddy-joined', self._buddy_joined_cb)
+ self._shared_activity.connect('buddy-left', self._buddy_left_cb)
+
+ self._logger.debug('This is my activity: making a tube...')
+ id = self.tubes_chan[telepathy.CHANNEL_TYPE_TUBES].OfferTube(
+ telepathy.TUBE_TYPE_DBUS, SERVICE, {})
+
+ # presence service should be tubes-aware and give us more help
+ # with this
+ def _setup(self):
+ if self._shared_activity is None:
+ self._logger.error('Failed to share or join activity')
+ return
+
+ bus_name, conn_path, channel_paths =\
+ self._shared_activity.get_channels()
+
+ # Work out what our room is called and whether we have Tubes already
+ room = None
+ tubes_chan = None
+ text_chan = None
+ for channel_path in channel_paths:
+ channel = telepathy.client.Channel(bus_name, channel_path)
+ htype, handle = channel.GetHandle()
+ if htype == telepathy.HANDLE_TYPE_ROOM:
+ self._logger.debug('Found our room: it has handle#%d "%s"',
+ handle, self.conn.InspectHandles(htype, [handle])[0])
+ room = handle
+ ctype = channel.GetChannelType()
+ if ctype == telepathy.CHANNEL_TYPE_TUBES:
+ self._logger.debug('Found our Tubes channel at %s', channel_path)
+ tubes_chan = channel
+ elif ctype == telepathy.CHANNEL_TYPE_TEXT:
+ self._logger.debug('Found our Text channel at %s', channel_path)
+ text_chan = channel
+
+ if room is None:
+ self._logger.error("Presence service didn't create a room")
+ return
+ if text_chan is None:
+ self._logger.error("Presence service didn't create a text channel")
+ return
+
+ # Make sure we have a Tubes channel - PS doesn't yet provide one
+ if tubes_chan is None:
+ self._logger.debug("Didn't find our Tubes channel, requesting one...")
+ tubes_chan = self.conn.request_channel(telepathy.CHANNEL_TYPE_TUBES,
+ telepathy.HANDLE_TYPE_ROOM, room, True)
+
+ self.tubes_chan = tubes_chan
+ self.text_chan = text_chan
+
+ tubes_chan[telepathy.CHANNEL_TYPE_TUBES].connect_to_signal('NewTube',
+ self._new_tube_cb)
+
+ def _list_tubes_reply_cb(self, tubes):
+ for tube_info in tubes:
+ self._new_tube_cb(*tube_info)
+
+ def _list_tubes_error_cb(self, e):
+ self._logger.error('ListTubes() failed: %s', e)
+
+ def _joined_cb(self, activity):
+ if not self._shared_activity:
+ return
+
+ # Find out who's already in the shared activity:
+ for buddy in self._shared_activity.get_joined_buddies():
+ self._logger.debug('Buddy %s is already in the activity' % buddy.props.nick)
+
+ self._logger.debug('Joined an existing shared activity')
+ self.initiating = False
+ self._setup()
+
+ self._logger.debug('This is not my activity: waiting for a tube...')
+ self.tubes_chan[telepathy.CHANNEL_TYPE_TUBES].ListTubes(
+ reply_handler=self._list_tubes_reply_cb,
+ error_handler=self._list_tubes_error_cb)
+
+ def _new_tube_cb(self, id, initiator, type, service, params, state):
+ self._logger.debug('New tube: ID=%d initator=%d type=%d service=%s '
+ 'params=%r state=%d', id, initiator, type, service,
+ params, state)
+
+ if (type == telepathy.TUBE_TYPE_DBUS and
+ service == SERVICE):
+ if state == telepathy.TUBE_STATE_LOCAL_PENDING:
+ self.tubes_chan[telepathy.CHANNEL_TYPE_TUBES].AcceptTube(id)
+
+ tube_conn = TubeConnection(self.conn,
+ self.tubes_chan[telepathy.CHANNEL_TYPE_TUBES],
+ id, group_iface=self.text_chan[telepathy.CHANNEL_INTERFACE_GROUP])
+ self.hellotube = HelloTube(tube_conn, self.initiating, self._get_buddy)
+
+ def _buddy_joined_cb (self, activity, buddy):
+ self._logger.debug('Buddy %s joined' % buddy.props.nick)
+
+ def _buddy_left_cb (self, activity, buddy):
+ self._logger.debug('Buddy %s left' % buddy.props.nick)
+
+ def _get_buddy(self, cs_handle):
+ """Get a Buddy from a channel specific handle."""
+ self._logger.debug('Trying to find owner of handle %u...', cs_handle)
+ group = self.text_chan[telepathy.CHANNEL_INTERFACE_GROUP]
+ my_csh = group.GetSelfHandle()
+ self._logger.debug('My handle in that group is %u', my_csh)
+ if my_csh == cs_handle:
+ handle = self.conn.GetSelfHandle()
+ self._logger.debug('CS handle %u belongs to me, %u', cs_handle, handle)
+ elif group.GetGroupFlags() & telepathy.CHANNEL_GROUP_FLAG_CHANNEL_SPECIFIC_HANDLES:
+ handle = group.GetHandleOwners([cs_handle])[0]
+ self._logger.debug('CS handle %u belongs to %u', cs_handle, handle)
+ else:
+ handle = cs_handle
+ logger.debug('non-CS handle %u belongs to itself', handle)
+
+ # XXX: deal with failure to get the handle owner
+ assert handle != 0
+
+ # XXX: we're assuming that we have Buddy objects for all contacts -
+ # this might break when the server becomes scalable.
+ return self.pservice.get_buddy_by_telepathy_handle(self.tp_conn_name,
+ self.tp_conn_path, handle)
+
+class HelloTube(ExportedGObject):
+ """The bit that talks over the TUBES!!!"""
+
+ def __init__(self, tube, is_initiator, get_buddy):
+ super(HelloTube, self).__init__(tube, PATH)
+ self._logger = logging.getLogger('pippy-activity.HelloTube')
+ self.tube = tube
+ self.is_initiator = is_initiator
+ self.entered = False # Have we set up the tube?
+ self.helloworld = False # Have we said Hello and received World?
+ self._get_buddy = get_buddy # Converts handle to Buddy object
+
+ self.ordered_bus_names = []
+
+ self.tube.watch_participants(self.participant_change_cb)
+
+ def participant_change_cb(self, added, removed):
+ self._logger.debug('Adding participants: %r' % added)
+ self._logger.debug('Removing participants: %r' % type(removed))
+
+ for handle, bus_name in added:
+ buddy = self._get_buddy(handle)
+ if buddy is not None:
+ self._logger.debug('Buddy %s was added' % buddy.props.nick)
+
+ for handle in removed:
+ buddy = self._get_buddy(handle)
+ if buddy is not None:
+ self._logger.debug('Buddy %s was removed' % buddy.props.nick)
+ try:
+ self.ordered_bus_names.remove(self.tube.participants[handle])
+ except ValueError:
+ # already absent
+ pass
+
+ if not self.entered:
+ #self.tube.add_signal_receiver(self.insert_cb, 'Insert', IFACE,
+ # path=PATH, sender_keyword='sender')
+ if self.is_initiator:
+ self._logger.debug("I'm initiating the tube, will "
+ "watch for hellos.")
+ self.add_hello_handler()
+ self.ordered_bus_names = [self.tube.get_unique_name()]
+ else:
+ self._logger.debug('Hello, everyone! What did I miss?')
+ self.Hello()
+ self.entered = True
+
+ @signal(dbus_interface=IFACE, signature='')
+ def Hello(self):
+ """Say Hello to whoever else is in the tube."""
+ self._logger.debug('I said Hello.')
+
+ @method(dbus_interface=IFACE, in_signature='as', out_signature='')
+ def World(self, bus_names):
+ """To be called on the incoming XO after they Hello."""
+ if not 1 or self.helloworld: # XXX remove 1
+ self._logger.debug('Somebody said World.')
+ self.ordered_bus_names = bus_names
+ # now I can World others
+ self.add_hello_handler()
+
+ #buddy = self._get_buddy(self.tube.bus_name_to_handle[bus_names[0]])
+ else:
+ self._logger.debug("I've already been welcomed, doing nothing")
+
+ def add_hello_handler(self):
+ self._logger.debug('Adding hello handler.')
+ self.tube.add_signal_receiver(self.hello_cb, 'Hello', IFACE,
+ path=PATH, sender_keyword='sender')
+
+ def hello_cb(self, sender=None):
+ """Somebody Helloed me. World them."""
+ self._logger.debug('Newcomer %s has joined', sender)
+ self.ordered_bus_names.append(sender)
+ self._logger.debug('Bus names are now: %r', self.ordered_bus_names)
+ self._logger.debug('Welcoming newcomer and sending them the game state')
+ self.tube.get_object(sender, PATH).World(self.ordered_bus_names,
+ dbus_interface=IFACE)
+
+