318 lines
11 KiB
Python
318 lines
11 KiB
Python
# Copyright (C) 2007, 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.
|
|
|
|
"""Metadata description of a given application/activity"""
|
|
|
|
import logging
|
|
import locale
|
|
import os
|
|
import zipfile
|
|
from ConfigParser import ConfigParser
|
|
import StringIO
|
|
import tempfile
|
|
|
|
import dbus
|
|
|
|
from sugar import env
|
|
from sugar import activity
|
|
|
|
_PYTHON_FACTORY='sugar-activity-factory'
|
|
|
|
_DBUS_SHELL_SERVICE = "org.laptop.Shell"
|
|
_DBUS_SHELL_PATH = "/org/laptop/Shell"
|
|
_DBUS_ACTIVITY_REGISTRY_IFACE = "org.laptop.Shell.ActivityRegistry"
|
|
|
|
class AlreadyInstalledException(Exception): pass
|
|
class NotInstalledException(Exception): pass
|
|
class InvalidPathException(Exception): pass
|
|
class ZipExtractException(Exception): pass
|
|
class RegistrationException(Exception): pass
|
|
|
|
class Bundle:
|
|
"""Metadata description of a given application/activity
|
|
|
|
The metadata is normally read from an activity.info file,
|
|
which is an INI-style configuration file read using the
|
|
standard Python ConfigParser module.
|
|
|
|
The format reference for the Bundle definition files is
|
|
available for further reference:
|
|
|
|
http://wiki.laptop.org/go/Activity_bundles
|
|
"""
|
|
def __init__(self, path):
|
|
self._init_with_path(path)
|
|
|
|
def _init_with_path(self, path):
|
|
self._name = None
|
|
self._icon = None
|
|
self._service_name = None
|
|
self._mime_types = None
|
|
self._show_launcher = True
|
|
self._valid = True
|
|
self._path = path
|
|
self._activity_version = 0
|
|
|
|
info_file = self._get_info_file()
|
|
if info_file:
|
|
self._parse_info(info_file)
|
|
else:
|
|
self._valid = False
|
|
|
|
linfo_file = self._get_linfo_file()
|
|
if linfo_file:
|
|
self._parse_linfo(linfo_file)
|
|
|
|
def _get_info_file(self):
|
|
info_file = None
|
|
|
|
if os.path.isdir(self._path):
|
|
info_path = os.path.join(self._path, 'activity', 'activity.info')
|
|
if os.path.isfile(info_path):
|
|
info_file = open(info_path)
|
|
else:
|
|
zip_file = zipfile.ZipFile(self._path)
|
|
file_names = zip_file.namelist()
|
|
root_dir = self._get_bundle_root_dir(file_names)
|
|
info_path = os.path.join(root_dir, 'activity', 'activity.info')
|
|
if info_path in file_names:
|
|
info_data = zip_file.read(info_path)
|
|
info_file = StringIO.StringIO(info_data)
|
|
zip_file.close()
|
|
|
|
return info_file
|
|
|
|
def _parse_info(self, info_file):
|
|
cp = ConfigParser()
|
|
cp.readfp(info_file)
|
|
|
|
section = 'Activity'
|
|
|
|
if cp.has_option(section, 'service_name'):
|
|
self._service_name = cp.get(section, 'service_name')
|
|
else:
|
|
self._valid = False
|
|
logging.error('%s must specify a service name' % self._path)
|
|
|
|
if cp.has_option(section, 'name'):
|
|
self._name = cp.get(section, 'name')
|
|
else:
|
|
self._valid = False
|
|
logging.error('%s must specify a name' % self._path)
|
|
|
|
if cp.has_option(section, 'class'):
|
|
self._class = cp.get(section, 'class')
|
|
self._exec = '%s --bundle-path="%s"' % (
|
|
env.get_bin_path(_PYTHON_FACTORY), self._path)
|
|
elif cp.has_option(section, 'exec'):
|
|
self._class = None
|
|
cmdline = cp.get(section, 'exec')
|
|
cmdline = os.path.join(self._path, cmdline)
|
|
cmdline = cmdline.replace('$SUGAR_BUNDLE_PATH', self._path)
|
|
cmdline = os.path.expandvars(cmdline)
|
|
self._exec = cmdline
|
|
else:
|
|
self._exec = None
|
|
self._valid = False
|
|
logging.error('%s must specify exec or class' % self._path)
|
|
|
|
if cp.has_option(section, 'mime_types'):
|
|
mime_list = cp.get(section, 'mime_types')
|
|
self._mime_types = mime_list.strip(';').split(';')
|
|
|
|
if cp.has_option(section, 'show_launcher'):
|
|
if cp.get(section, 'show_launcher') == 'no':
|
|
self._show_launcher = False
|
|
|
|
if cp.has_option(section, 'icon'):
|
|
self._icon = cp.get(section, 'icon')
|
|
|
|
if cp.has_option(section, 'activity_version'):
|
|
self._activity_version = int(cp.get(section, 'activity_version'))
|
|
|
|
def _parse_linfo(self, linfo_file):
|
|
cp = ConfigParser()
|
|
cp.readfp(linfo_file)
|
|
|
|
section = 'Activity'
|
|
|
|
if cp.has_option(section, 'name'):
|
|
self._name = cp.get(section, 'name')
|
|
|
|
def _get_linfo_file(self):
|
|
linfo_file = None
|
|
|
|
lang = locale.getdefaultlocale()[0]
|
|
if not lang:
|
|
return None
|
|
|
|
if os.path.isdir(self._path):
|
|
linfo_path = os.path.join(self.get_locale_path(), lang, 'activity.linfo')
|
|
if not os.path.isfile(linfo_path):
|
|
linfo_path = os.path.join(self.get_locale_path(), lang[:2], 'activity.linfo')
|
|
if os.path.isfile(linfo_path):
|
|
linfo_file = open(linfo_path)
|
|
else:
|
|
zip_file = zipfile.ZipFile(self._path)
|
|
file_names = zip_file.namelist()
|
|
root_dir = self._get_bundle_root_dir(file_names)
|
|
linfo_path = os.path.join(root_dir, 'locale', lang, 'activity.linfo')
|
|
if not linfo_path in file_names:
|
|
linfo_path = os.path.join(root_dir, 'locale', lang[:2], 'activity.linfo')
|
|
if linfo_path in zip_file.namelist():
|
|
linfo_data = zip_file.read(linfo_path)
|
|
linfo_file = StringIO.StringIO(linfo_data)
|
|
|
|
zip_file.close()
|
|
|
|
return linfo_file
|
|
|
|
def is_valid(self):
|
|
return self._valid
|
|
|
|
def get_locale_path(self):
|
|
"""Get the locale path inside the activity bundle."""
|
|
return os.path.join(self._path, 'locale')
|
|
|
|
def get_icons_path(self):
|
|
"""Get the icons path inside the activity bundle."""
|
|
return os.path.join(self._path, 'icons')
|
|
|
|
def get_path(self):
|
|
"""Get the activity bundle path."""
|
|
return self._path
|
|
|
|
def get_name(self):
|
|
"""Get the activity user visible name."""
|
|
return self._name
|
|
|
|
def get_service_name(self):
|
|
"""Get the activity service name"""
|
|
return self._service_name
|
|
|
|
def get_icon(self):
|
|
"""Get the activity icon name"""
|
|
if os.path.isdir(self._path):
|
|
activity_path = os.path.join(self._path, 'activity')
|
|
return os.path.join(activity_path, self._icon + '.svg')
|
|
else:
|
|
zip_file = zipfile.ZipFile(self._path)
|
|
file_names = zip_file.namelist()
|
|
root_dir = self._get_bundle_root_dir(file_names)
|
|
icon_path = os.path.join(root_dir, 'activity', self._icon + '.svg')
|
|
print icon_path
|
|
print file_names
|
|
if icon_path in file_names:
|
|
icon_data = zip_file.read(icon_path)
|
|
temp_file, temp_file_path = tempfile.mkstemp(self._icon)
|
|
os.write(temp_file, icon_data)
|
|
os.close(temp_file)
|
|
return temp_file_path
|
|
else:
|
|
return None
|
|
|
|
def get_activity_version(self):
|
|
"""Get the activity version"""
|
|
return self._activity_version
|
|
|
|
def get_exec(self):
|
|
"""Get the command to execute to launch the activity factory"""
|
|
return self._exec
|
|
|
|
def get_class(self):
|
|
"""Get the main Activity class"""
|
|
return self._class
|
|
|
|
def get_mime_types(self):
|
|
"""Get the MIME types supported by the activity"""
|
|
return self._mime_types
|
|
|
|
def get_show_launcher(self):
|
|
"""Get whether there should be a visible launcher for the activity"""
|
|
return self._show_launcher
|
|
|
|
def is_installed(self):
|
|
if self._valid and activity.get_registry().get_activity(self._service_name):
|
|
return True
|
|
else:
|
|
return False
|
|
|
|
def _get_bundle_root_dir(self, file_names):
|
|
"""
|
|
We check here that all the files in the .xo are inside one only dir
|
|
(bundle_root_dir).
|
|
"""
|
|
bundle_root_dir = None
|
|
for file_name in file_names:
|
|
if not bundle_root_dir:
|
|
bundle_root_dir = file_name.split('/')[0]
|
|
if not bundle_root_dir.endswith('.activity'):
|
|
raise 'Incorrect bundle.'
|
|
else:
|
|
if not file_name.startswith(bundle_root_dir):
|
|
raise 'Incorrect bundle.'
|
|
|
|
return bundle_root_dir
|
|
|
|
def install(self):
|
|
if self.is_installed():
|
|
raise AlreadyInstalledException
|
|
|
|
ext = os.path.splitext(self._path)[1]
|
|
if not os.path.isfile(self._path):
|
|
raise InvalidPathException
|
|
|
|
bundle_dir = env.get_user_activities_path()
|
|
if not os.path.isdir(bundle_dir):
|
|
os.mkdir(bundle_dir)
|
|
|
|
zip_file = zipfile.ZipFile(self._path)
|
|
file_names = zip_file.namelist()
|
|
bundle_root_dir = self._get_bundle_root_dir(file_names)
|
|
bundle_path = os.path.join(bundle_dir, bundle_root_dir)
|
|
|
|
if os.spawnlp(os.P_WAIT, 'unzip', 'unzip', self._path, '-d', bundle_dir):
|
|
raise ZipExtractException
|
|
|
|
self._init_with_path(bundle_path)
|
|
|
|
bus = dbus.SessionBus()
|
|
proxy_obj = bus.get_object(_DBUS_SHELL_SERVICE, _DBUS_SHELL_PATH)
|
|
dbus_service = dbus.Interface(proxy_obj, _DBUS_ACTIVITY_REGISTRY_IFACE)
|
|
if not dbus_service.AddBundle(bundle_path):
|
|
raise RegistrationException
|
|
|
|
def deinstall(self):
|
|
if not self.is_installed():
|
|
raise NotInstalledException
|
|
|
|
ext = os.path.splitext(self._path)[1]
|
|
if not os.path.isfile(self._path) or ext != '.activity':
|
|
raise InvalidPathException
|
|
|
|
for root, dirs, files in os.walk(self._path, topdown=False):
|
|
for name in files:
|
|
os.remove(os.path.join(root, name))
|
|
for name in dirs:
|
|
os.rmdir(os.path.join(root, name))
|
|
os.rmdir(self._path)
|
|
|
|
self._init_with_path(None)
|
|
|
|
# TODO: notify shell
|
|
|