# -*- coding: utf-8 -*- # ## plugins/plugin_installer/plugin_installer.py ## ## Copyright (C) 2010-2011 Denis Fomin ## Copyright (C) 2011 Yann Leboulanger ## ## This file is part of Gajim. ## ## Gajim 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; version 3 only. ## ## Gajim 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 Gajim. If not, see . ## import gtk import pango import gobject import ftplib import io import threading import ConfigParser import os import fnmatch import sys from common import gajim from plugins import GajimPlugin from plugins.helpers import log_calls, log from dialogs import WarningDialog, HigDialog, YesNoDialog from plugins.gui import GajimPluginConfigDialog def convert_version_to_list(version_str): version_list = version_str.split('.') l = [] while len(version_list): l.append(int(version_list.pop(0))) return l class PluginInstaller(GajimPlugin): @log_calls('PluginInstallerPlugin') def init(self): self.description = _('Install and upgrade plugins from ftp') self.config_dialog = PluginInstallerPluginConfigDialog(self) self.config_default_values = {'ftp_server': ('ftp.gajim.org', '')} self.window = None self.progressbar = None self.available_plugins_model = None self.upgrading = False # True when opened from upgrade popup dialog @log_calls('PluginInstallerPlugin') def activate(self): self.pl_menuitem = gajim.interface.roster.xml.get_object( 'plugins_menuitem') self.id_ = self.pl_menuitem.connect_after('activate', self.on_activate) if 'plugins' in gajim.interface.instances: self.on_activate(None) gobject.timeout_add_seconds(30, self.check_update) @log_calls('PluginInstallerPlugin') def warn_update(self, plugins): def open_update(dummy): self.upgrading = True self.pl_menuitem.activate() nb = gajim.interface.instances['plugins'].plugins_notebook gobject.idle_add(nb.set_current_page, 1) if plugins: plugins_str = '\n'.join(plugins) YesNoDialog(_('Plugins updates'), _('Some updates are available for' ' your installer plugins. Do you want to update those plugins:' '\n%s') % plugins_str, on_response_yes=open_update) @log_calls('PluginInstallerPlugin') def check_update(self): def _run(): try: to_update = [] con = ftplib.FTP_TLS(ftp.server) con.login() con.prot_p() con.cwd('plugins') plugins_dirs = con.nlst() for dir_ in plugins_dirs: try: con.retrbinary('RETR %s/manifest.ini' % dir_, ftp.handleDownload) except Exception, error: if str(error).startswith('550'): continue ftp.config.readfp(io.BytesIO(ftp.buffer_.getvalue())) local_version = ftp.get_plugin_version(ftp.config.get( 'info', 'name')) if local_version: local = convert_version_to_list(local_version) remote = convert_version_to_list(ftp.config.get('info', 'version')) if remote > local: to_update.append(ftp.config.get('info', 'name')) con.quit() gobject.idle_add(self.warn_update, to_update) except Exception, e: log.debug('Ftp error when check updates: %s' % str(e)) ftp = Ftp(self) ftp.run = _run ftp.start() @log_calls('PluginInstallerPlugin') def deactivate(self): self.pl_menuitem.disconnect(self.id_) if hasattr(self, 'page_num'): self.notebook.remove_page(self.page_num) self.notebook.set_current_page(0) if hasattr(self, 'ftp'): del self.ftp def on_activate(self, widget): if 'plugins' not in gajim.interface.instances: return self.installed_plugins_model = gajim.interface.instances[ 'plugins'].installed_plugins_model self.notebook = gajim.interface.instances['plugins'].plugins_notebook self.id_n = self.notebook.connect('switch-page', self.on_notebook_switch_page) self.window = gajim.interface.instances['plugins'].window self.window.connect('destroy', self.on_win_destroy) self.GTK_BUILDER_FILE_PATH = self.local_file_path('config_dialog.ui') self.xml = gtk.Builder() self.xml.set_translation_domain('gajim_plugins') self.xml.add_objects_from_file(self.GTK_BUILDER_FILE_PATH, ['hpaned2']) hpaned = self.xml.get_object('hpaned2') self.page_num = self.notebook.append_page(hpaned, gtk.Label(_('Available'))) widgets_to_extract = ('plugin_name_label1', 'available_treeview', 'progressbar', 'inslall_upgrade_button', 'plugin_authors_label1', 'plugin_authors_label1', 'plugin_homepage_linkbutton1', 'plugin_description_textview1') for widget_name in widgets_to_extract: setattr(self, widget_name, self.xml.get_object(widget_name)) attr_list = pango.AttrList() attr_list.insert(pango.AttrWeight(pango.WEIGHT_BOLD, 0, -1)) self.plugin_name_label1.set_attributes(attr_list) self.available_plugins_model = gtk.ListStore(gobject.TYPE_PYOBJECT, gobject.TYPE_STRING, gobject.TYPE_STRING, gobject.TYPE_STRING, gobject.TYPE_BOOLEAN, gobject.TYPE_PYOBJECT, gobject.TYPE_PYOBJECT, gobject.TYPE_PYOBJECT) self.available_treeview.set_model(self.available_plugins_model) self.progressbar.set_property('no-show-all', True) renderer = gtk.CellRendererText() col = gtk.TreeViewColumn(_('Plugin'), renderer, text=1) col.set_resizable(True) col.set_property('expand', True) col.set_sizing(gtk.TREE_VIEW_COLUMN_GROW_ONLY) self.available_treeview.append_column(col) col = gtk.TreeViewColumn(_('Installed\nversion'), renderer, text=2) self.available_treeview.append_column(col) col = gtk.TreeViewColumn(_('Available\nversion'), renderer, text=3) col.set_property('expand', False) self.available_treeview.append_column(col) renderer = gtk.CellRendererToggle() renderer.set_property('activatable', True) renderer.connect('toggled', self.available_plugins_toggled_cb) col = gtk.TreeViewColumn(_('Install /\nUpgrade'), renderer, active=4) self.available_treeview.append_column(col) if gobject.signal_lookup('error_signal', self.window) is 0: gobject.signal_new('error_signal', self.window, gobject.SIGNAL_RUN_LAST, gobject.TYPE_STRING, (gobject.TYPE_STRING,)) gobject.signal_new('plugin_downloaded', self.window, gobject.SIGNAL_RUN_LAST, gobject.TYPE_STRING, (gobject.TYPE_PYOBJECT,)) self.window.connect('error_signal', self.on_some_ftp_error) self.window.connect('plugin_downloaded', self.on_plugin_downloaded) selection = self.available_treeview.get_selection() selection.connect('changed', self.available_plugins_treeview_selection_changed) selection.set_mode(gtk.SELECTION_SINGLE) self._clear_available_plugin_info() self.xml.connect_signals(self) self.window.show_all() def on_win_destroy(self, widget): if hasattr(self, 'ftp'): del self.ftp def available_plugins_toggled_cb(self, cell, path): is_active = self.available_plugins_model[path][4] self.available_plugins_model[path][4] = not is_active dir_list = [] for i in xrange(len(self.available_plugins_model)): if self.available_plugins_model[i][4]: dir_list.append(self.available_plugins_model[i][0]) if not dir_list: self.inslall_upgrade_button.set_property('sensitive', False) else: self.inslall_upgrade_button.set_property('sensitive', True) def on_notebook_switch_page(self, widget, page, page_num): if not hasattr(self, 'ftp') and self.page_num == page_num: self.available_plugins_model.clear() self.progressbar.show() self.ftp = Ftp(self) self.ftp.remote_dirs = None self.ftp.upgrading = True self.ftp.start() def on_inslall_upgrade_clicked(self, widget): self.inslall_upgrade_button.set_property('sensitive', False) dir_list = [] for i in xrange(len(self.available_plugins_model)): if self.available_plugins_model[i][4]: dir_list.append(self.available_plugins_model[i][0]) ftp = Ftp(self) ftp.remote_dirs = dir_list ftp.start() def on_some_ftp_error(self, widget, error_text): for i in xrange(len(self.available_plugins_model)): self.available_plugins_model[i][4] = False self.progressbar.hide() WarningDialog(_('Ftp error'), error_text, self.window) def on_plugin_downloaded(self, widget, plugin_dirs): for _dir in plugin_dirs: is_active = False plugins = None plugin_dir = os.path.join(gajim.PLUGINS_DIRS[1], _dir) plugin = gajim.plugin_manager.get_plugin_by_path(plugin_dir) if plugin: if plugin.active and plugin.name != self.name: is_active = True gobject.idle_add(gajim.plugin_manager.deactivate_plugin, plugin) gajim.plugin_manager.plugins.remove(plugin) model = self.installed_plugins_model for row in xrange(len(model)): if plugin == model[row][0]: model.remove(model.get_iter((row, 0))) break plugins = self.scan_dir_for_plugin(plugin_dir) if not plugins: continue gajim.plugin_manager.add_plugin(plugins[0]) plugin = gajim.plugin_manager.plugins[-1] for row in xrange(len(self.available_plugins_model)): if plugin.name == self.available_plugins_model[row][1]: self.available_plugins_model[row][2] = plugin.version self.available_plugins_model[row][4] = False continue if is_active and plugin.name != self.name: gobject.idle_add(gajim.plugin_manager.activate_plugin, plugin) if plugin.name != 'Plugin Installer': self.installed_plugins_model.append([plugin, plugin.name, is_active]) dialog = HigDialog(None, gtk.MESSAGE_INFO, gtk.BUTTONS_OK, '', _('All selected plugins downloaded')) dialog.set_modal(False) dialog.set_transient_for(self.window) dialog.popup() def available_plugins_treeview_selection_changed(self, treeview_selection): model, iter = treeview_selection.get_selected() if iter: self.plugin_name_label1.set_text(model.get_value(iter, 1)) self.plugin_authors_label1.set_text(model.get_value(iter, 6)) self.plugin_homepage_linkbutton1.set_uri(model.get_value(iter, 7)) self.plugin_homepage_linkbutton1.set_label(model.get_value(iter, 7)) label = self.plugin_homepage_linkbutton1.get_children()[0] label.set_ellipsize(pango.ELLIPSIZE_END) self.plugin_homepage_linkbutton1.set_property('sensitive', True) desc_textbuffer = self.plugin_description_textview1.get_buffer() desc_textbuffer.set_text(_(model.get_value(iter, 5))) self.plugin_description_textview1.set_property('sensitive', True) else: self._clear_available_plugin_info() def _clear_available_plugin_info(self): self.plugin_name_label1.set_text('') self.plugin_authors_label1.set_text('') self.plugin_homepage_linkbutton1.set_uri('') self.plugin_homepage_linkbutton1.set_label('') self.plugin_homepage_linkbutton1.set_property('sensitive', False) desc_textbuffer = self.plugin_description_textview1.get_buffer() desc_textbuffer.set_text('') self.plugin_description_textview1.set_property('sensitive', False) def scan_dir_for_plugin(self, path): plugins_found = [] conf = ConfigParser.ConfigParser() fields = ('name', 'short_name', 'version', 'description', 'authors', 'homepage') if not os.path.isdir(path): return plugins_found dir_list = os.listdir(path) dir_, mod = os.path.split(path) sys.path.insert(0, dir_) manifest_path = os.path.join(path, 'manifest.ini') if not os.path.isfile(manifest_path): return plugins_found for elem_name in dir_list: file_path = os.path.join(path, elem_name) module = None if os.path.isfile(file_path) and fnmatch.fnmatch(file_path, '*.py'): module_name = os.path.splitext(elem_name)[0] if module_name == '__init__': continue try: module = __import__('%s.%s' % (mod, module_name)) except ValueError, value_error: pass except ImportError, import_error: pass except AttributeError, attribute_error: pass if module is None: continue for module_attr_name in [attr_name for attr_name in dir(module) if not (attr_name.startswith('__') or attr_name.endswith('__'))]: module_attr = getattr(module, module_attr_name) try: if not issubclass(module_attr, GajimPlugin) or \ module_attr is GajimPlugin: continue module_attr.__path__ = os.path.abspath(os.path.dirname( file_path)) # read metadata from manifest.ini conf.readfp(open(manifest_path, 'r')) for option in fields: if conf.get('info', option) is '': raise ConfigParser.NoOptionError, 'field empty' setattr(module_attr, option, conf.get('info', option)) conf.remove_section('info') plugins_found.append(module_attr) except TypeError, type_error: pass except ConfigParser.NoOptionError, type_error: # all fields are required pass return plugins_found class Ftp(threading.Thread): def __init__(self, plugin): super(Ftp, self).__init__() self.plugin = plugin self.window = plugin.window self.server = plugin.config['ftp_server'] self.progressbar = plugin.progressbar self.model = plugin.available_plugins_model self.config = ConfigParser.ConfigParser() self.buffer_ = io.BytesIO() self.remote_dirs = None self.append_to_model = True self.upgrading = False def model_append(self, row): self.model.append(row) return False def progressbar_pulse(self): self.progressbar.pulse() return True def get_plugin_version(self, plugin_name): for plugin in gajim.plugin_manager.plugins: if plugin.name == plugin_name: return plugin.version def run(self): try: gobject.idle_add(self.progressbar.set_text, _('Connecting to server')) self.ftp = ftplib.FTP_TLS(self.server) self.ftp.login() self.ftp.prot_p() self.ftp.cwd('plugins') if not self.remote_dirs: self.plugins_dirs = self.ftp.nlst() progress_step = 1.0 / len(self.plugins_dirs) gobject.idle_add(self.progressbar.set_text, _('Scan files on the server')) for dir_ in self.plugins_dirs: fract = self.progressbar.get_fraction() + progress_step gobject.idle_add(self.progressbar.set_fraction, fract) gobject.idle_add(self.progressbar.set_text, _('Read "%s"') % dir_) try: self.ftp.retrbinary('RETR %s/manifest.ini' % dir_, self.handleDownload) except Exception, error: if str(error).startswith('550'): continue self.config.readfp(io.BytesIO(self.buffer_.getvalue())) local_version = self.get_plugin_version( self.config.get('info', 'name')) upgrade = False if self.upgrading and local_version: local = convert_version_to_list(local_version) remote = convert_version_to_list(self.config.get('info', 'version')) if remote > local: upgrade = True gobject.idle_add( self.plugin.inslall_upgrade_button.set_property, 'sensitive', True) gobject.idle_add(self.model_append, [dir_, self.config.get('info', 'name'), local_version, self.config.get('info', 'version'), upgrade, self.config.get('info', 'description'), self.config.get('info', 'authors'), self.config.get('info', 'homepage'), ]) self.plugins_dirs = None self.ftp.quit() gobject.idle_add(self.progressbar.set_fraction, 0) if self.remote_dirs: self.download_plugin() gobject.idle_add(self.progressbar.hide) except Exception, e: self.window.emit('error_signal', str(e)) def handleDownload(self, block): self.buffer_.write(block) def download_plugin(self): gobject.idle_add(self.progressbar.show) self.pulse = gobject.timeout_add(150, self.progressbar_pulse) gobject.idle_add(self.progressbar.set_text, _('Create a list of files')) for remote_dir in self.remote_dirs: def nlstr(dir_, subdir=None): if subdir: dir_ = dir_ + '/' + subdir list_ = self.ftp.nlst(dir_) for i in list_: name = i.split('/')[-1] if '.' not in name: try: if i == self.ftp.nlst(i)[0]: files.append(i[1:]) del dirs[i[1:]] except Exception, e: # empty dir or file continue dirs.append(i[1:]) subdirs = name nlstr(dir_, subdirs) else: files.append(i[1:]) dirs, files = [], [] nlstr('/plugins/' + remote_dir) if not os.path.isdir(gajim.PLUGINS_DIRS[1]): os.mkdir(gajim.PLUGINS_DIRS[1]) local_dir = ld = os.path.join(gajim.PLUGINS_DIRS[1], remote_dir) if not os.path.isdir(local_dir): os.mkdir(local_dir) local_dir = os.path.split(gajim.PLUGINS_DIRS[1])[0] # creating dirs for dir_ in dirs: try: os.mkdir(os.path.join(local_dir, dir_)) except OSError, e: if str(e).startswith('[Errno 17]'): continue raise # downloading files for filename in files: gobject.idle_add(self.progressbar.set_text, _('Downloading "%s"') % filename) full_filename = os.path.join(local_dir, filename) try: self.ftp.retrbinary('RETR /%s' % filename, open(full_filename, 'wb').write) #full_filename.close() except ftplib.error_perm: print 'ERROR: cannot read file "%s"' % filename os.unlink(filename) self.ftp.quit() gobject.idle_add(self.window.emit, 'plugin_downloaded', self.remote_dirs) gobject.source_remove(self.pulse) class PluginInstallerPluginConfigDialog(GajimPluginConfigDialog): def init(self): self.GTK_BUILDER_FILE_PATH = self.plugin.local_file_path( 'config_dialog.ui') self.xml = gtk.Builder() self.xml.set_translation_domain('gajim_plugins') self.xml.add_objects_from_file(self.GTK_BUILDER_FILE_PATH, ['hbox111']) hbox = self.xml.get_object('hbox111') self.child.pack_start(hbox) self.xml.connect_signals(self) self.connect('hide', self.on_hide) def on_run(self): widget = self.xml.get_object('ftp_server') widget.set_text(str(self.plugin.config['ftp_server'])) def on_hide(self, widget): widget = self.xml.get_object('ftp_server') self.plugin.config['ftp_server'] = widget.get_text()