Plugins

Differences between revisions 5 and 15 (spanning 10 versions)
Revision 5 as of 2009-07-17 17:50:59
Size: 8619
Editor: bismuth
Comment:
Revision 15 as of 2009-08-21 13:00:18
Size: 9260
Editor: 65-78-0-53
Comment:
Deletions are marked like this. Additions are marked like this.
Line 7: Line 7:
 * The ability to stop going forward (say, a EULA page that requires an approval checkbox). Might as well through in stopping going back.  * The ability to prevent going forward (say, a EULA page that requires an approval checkbox). Might as well also allow preventing going back.
Line 14: Line 14:
 * Allow plugins to be pure Python (no required shell)
Line 32: Line 33:
It can install the UI files (.glade, .ui) anywhere it wants. The plugin will have UI classes like 'PageGtk' or 'PageKde' or 'PageNoninteractive' which will be instantiated by Ubiquity and passed to the filter class. If the right class for the current frontend isn't find, the plugin is ignored. For GUI frontends, a method {{{get_ui()}}} can be called to get the frontend-specific UI object (for GTK, a tuple of the top-level name and the glade xml object). The GTK frontend will load and connect the glade file itself. The plugin will have UI classes like 'PageGtk' or 'PageKde' or 'PageNoninteractive' which will be instantiated by Ubiquity and passed to the filter class. If the right class for the current frontend isn't find, the plugin is ignored. For GUI frontends, a method {{{get_ui()}}} can be called to get frontend-specific objects and info. This will return a dictionary object with keys that the frontend knows. (For GTK, 'widgets' containing a list of pages and 'optional_widgets').

The plugin can install the UI files (.glade, .ui) anywhere it wants, because it is responsible for loading it
. The GTK and KDE frontends only want to see actual widgets.
Line 36: Line 39:
Can use its own debconf template. Ubiquity will do the translation for it, from its template. Plugins can use their own debconf template. Ubiquity will do the translation for it, from its template.
Line 40: Line 43:
Plugins can drop the filter in /usr/lib/ubiquity/plugins.d/ Plugins drop python page files in /usr/lib/ubiquity/plugins.d/
Line 42: Line 45:
=== How to add/remove pages === === How to order/add/remove pages ===
Line 44: Line 47:
There are four major options here:
 0. Require some third party to specify a hard list of pages. This could be done via preseeding the appropriate page list that Ubiquity uses (there's one for Ubiquity and one for OEM-config).
 0. Only allow one plugin (though it could still use multiple pages). This plugin would own the mechanism for specifying a page list (much like the above, but could be via a well-known file like /etc/ubiquity/pages).
 0. Have some syntax for saying a page should be before this page, but after this page. Sort of like how upstart lets your order jobs, but way simpler. Could reference either standard pages or other plugin pages. If Ubiquity can't figure out how to order the plugin, it just won't show it. Additionally would need to allow a plugin to list pages it wants to hide.
 0. Use filesystem ordering, like have plugins use plugins.d/10-mypage and 22-otherpage. The standard pages would need to have standard numbers. Which has all sorts of difficulties if we ever tried to add, remove, or reorder a standard page. All plugins would need to change. Could use a separate directory like plugins.hide.d/mypage to hide pages.
Plugins has a module-wide NAME field that specifies a name to use. The recommended NAME for pages would be similar to the module name that defines the page. For third-party plugins, using a namespace (e.g. 'foo-page' and 'foo-page2') is recommended.
Line 50: Line 49:
The first two options above are best suited for OEM installs, where one entity is crafting the experience. But the latter two options are slightly better for Ubuntu itself where it's easier for derivatives and packagers to plug into the existing system on a dynamic 'am-I-installed' basis. Seeds would do the rest of the job. Even here though, there is tight control over the experience. The first two options could still work.

I think option 3 above gives us the most flexibility for the least cost. The syntax can be dreadfully simple: only allow one 'after' ''or'' 'before' statement, and it would use a defined 'name' for the specified page (reducing complexity for things like the partitioner that have 2 UI files, but one module. So, for example:
To insert your page into the sequence of pages, use AFTER or BEFORE fields.
Line 54: Line 51:
AFTER='language' AFTER='goober-language'
BEFORE=['language', None]
Line 57: Line 55:
Pages can be hidden from a HIDE list (a non-list would also work): Pages can be hidden from a HIDDEN field (a list would also work):
Line 59: Line 57:
HIDE=['language', 'timezone'] HIDDEN='language'
Line 62: Line 60:
Plugins would have a NAME field that specifies a name to use. The standard pages would have similar values. The recommended NAME for pages would be the module name that defines the page.  * If neither AFTER or BEFORE are specified, the page will not be used, but may still specify other plugin options, like HIDDEN. (i.e. a page that only hides, and does not insert itself)
 * If AFTER or BEFORE is a list of names, the first component that Ubiquity recognizes is used.
 * If both AFTER and BEFORE are specified, Ubiquity will use AFTER if it recognizes it, else use BEFORE.
 * If Ubiquity still can't find a value that it recognizes, the page is ignored (and other plugin options like HIDDEN are '''not''' used -- it's likely this page is optional, pending installation of some package we don't have or was written for an older Ubiquity).
 * If AFTER's value is None or otherwise 'Python-false', it is inserted at the beginning of the page sequence.
 * If BEFORE's value is None or otherwise 'Python-false', it is inserted at the end of the page sequence.
Line 64: Line 67:
Ties (where two pages specify 'after language' for example) can be broken by file system ordering in plugins.d (ala option 4 above). Thus, Ubiquity just has to read those in in order, and everything just works. The number before the filename can be considered a priority then for ties. i.e. if both 10-mypage and 22-otherpage specify {{{AFTER=language}}}, 22 would be first, then 10.

 * If neither AFTER or BEFORE are specified, the page will not be used, but may still specify other plugin options, like HIDE.
 * If AFTER or BEFORE is a list, the first component that Ubiquity recognizes is used.
 * If both AFTER and BEFORE are specified, Ubiquity will use AFTER if it recognizes it, else use BEFORE.
 * If Ubiquity still can't find a value that it recognizes, the page is ignored (and other plugin options like HIDE are '''not''' used -- it's likely this page is optional, pending installation of some package we don't have or was written for an older Ubiquity).
 * This allows
Ties (where two pages specify 'after language' for example) are broken by file system ordering in plugins.d. Ubiquity reads the plugins.d files in order, and resolves names as they come in. The number before the filename can thus be considered a priority then for ties. i.e. if both 10-mypage and 22-otherpage specify {{{AFTER=language}}}, 22 would be first in the resulting sequence, then 10.
Line 78: Line 75:
It must provide a method called {{{get_ui(self)}}} that returns a tuple of (string, gtk.glade.XML), where the string specifies the name of the top-level page widget in the glade XML. It must provide a method called {{{get_ui(self)}}} that returns a dictionary with 'widgets' and optionally, 'optional_widgets' fields. 'widgets' is a list or singleton of GTK pages (usually just a single page).
Line 86: Line 83:
It must provide a method called {{{get_ui(self)}}} that returns a page widget. It must provide a method called {{{get_ui(self)}}} that returns a dictionary with 'widgets' and 'breadcrumb' fields. 'widgets' is a list or singleton of Qt pages (usually just a single page). 'breadcrumb' is a debconf template key for the list of stages. Setting 'breadcrumb' to None indicates that you don't want the breadcrumb list visible for your page.
Line 90: Line 87:
The controller object passed to Page frontend classes is a simple mechanism for talking to the main frontend. Right now the only methods allowed are: The controller object passed to Page UI classes is a simple mechanism for talking to the main frontend. Right now the only methods allowed are:
Line 92: Line 89:
 * allow_go_forward(bool) (page starts with forward allowed)
 * allow_go_backward(bool) (page starts with backward allowed)
 * dbfilter (lets you have access to your Page class)
 * oem_config, oem_user_config (flags for which mode we're in)
 * is_language_page (set this to True if you intend to change languages)
 * translate(self, lang=None, just_me=True, reget=False) (retranslate the UI)
 * allow_go_forward(self, bool) (page starts with forward allowed)
 * allow_go_backward(self, bool) (page starts with backward allowed)
 * go_forward(self) (clicks forward)
 * go_backward(self) (clicks back)
 * go_to_page(self, page) (jumps to the given page)
Line 95: Line 99:
This may or may not be the main frontend object (which already has these functions), depending on how paranoid we are when implementing it. === Applying the Page ===
Line 97: Line 101:
=== Open Questions ===
 * What about the apply() method and progress? Do we want to expose a string to show to user while Ubiquity is apply'ing the plugins? And allow progress reporting? Or do we just say 'plugins should apply quickly'?
If your plugin wants to perform install-time work, it can define a {{{Install}}} class that derives from the {{{InstallPlugin}}} class.

{{{InstallPlugin}}} is a {{{Plugin}}} and follows the same conventions -- it is a debconf filter that has {{{prepare}}} and {{{cleanup}}} methods. But additionally and more relevantly for us, it adds an {{{install}}} method.

{{{install}}} is passed a target directory (usually '/target' or '/'), a progress class (more on that below), and possibly some future arguments. It's recommended that plugins use *args and **kwargs to gracefully handle future extensions.

If {{{install}}} returns a Python-true value, it is considered an error (ala command line programs). The error value (a string or a number) will be printed to syslog.

==== Progress API ====

{{{install}}} is passed a progress class that can provide updates on what your plugin is doing. Currently, it only supports passing a text description. Each plugin takes up 1% on the install progress bar.

{{{
#!Python
def info(self, title):
    pass
}}}

==== Install Example ====

{{{
#!Python
def install(self, target, progress, *args, **kwargs):
    progress.info('ubiquity/install/timezone')
    return InstallPlugin.install(target, progress, *args, **kwargs)
}}}
Line 104: Line 132:
import gtk #!Python
Line 106: Line 134:
from ubiquity import Plugin from ubiquity.plugin import *
Line 114: Line 142:
        import gtk
Line 115: Line 144:
        self.gladexml = gtk.glade.XML('/usr/share/ubiquity-foo/foo.glade', 'foo')         self.builder = gtk.Builder()
        self.builder.
add_from_file('/usr/share/ubiquity-foo/foo.ui')
        se
lf.builder.connect_signals(self)
Line 117: Line 148:
        return ('foo', self.gladexml)         return self.builder.get_object('foo')
Line 121: Line 152:
        button = self.gladexml.get_widget('bar')         button = self.builder.get_object('bar')
Line 124: Line 155:
        button = self.gladexml.get_widget('bar')         button = self.builder.get_object('bar')
Line 131: Line 162:
        self.frontend.set_bar(bar)         self.ui.set_bar(bar)
Line 135: Line 166:
        bar = self.frontend.get_bar()         bar = self.ui.get_bar()
Line 139: Line 170:
    def apply(self):
        bar = self.db.get('foo/bar')
        if bar:
            os.system(['touch', '/etc/bar'])
        else:
            os.system(['rm', '/etc/bar'])
class Install(InstallPlugin):
    def install(self, target, progress, *args, **kwargs):
        progress.info('foo/applying')
        cmd = 'touch' if self.db.get('foo/bar') else 'rm'
        rv = os.system([cmd, os.path.join(target, '/etc/bar')])
        if rv:
            return rv
        return InstallPlugin.install(target, progress, *args, **kwargs)

Here, I describe a possible system for enabling drop-in plugins for Ubiquity and OEM-config.

Requirements

  • Insert a page: The ability to add a page anywhere in the page sequence
  • Hide a page: The ability to stop a standard page from showing up (the combination of inserting/hiding lets you replace a page)
  • The ability to prevent going forward (say, a EULA page that requires an approval checkbox). Might as well also allow preventing going back.
  • Let plugins be translatable
  • Let plugins execute code after user is done entering any info, to actually do what user asks
  • Let plugins offer gtk or kde UI (or debconf or noninteractive, etc if desired)

Wants

  • Reduce plugin's exposure to Ubiquity internals, to make them more change-proof.
  • Allow plugins to be pure Python (no required shell)

Design

What a plugin looks like

Bits of a standard gtk page:

  • A glade file
  • A debconf filter
  • Ask and apply scripts

With the above, by providing the above files, we can do everything a normal page can in a plugin. Assuming Ubiquity knows how to find the files.

The asking can be provided by Ubiquity, as long as the plugin gives us a list of debconf questions to ask. And the apply script could just be done by the filter by a special 'apply' method. Then all a plugin needs is a filter and a UI file.

We could additionally dumb down the python a plugin needs to provide by having them be 'filter lites' custom classes that didn't do everything a filter does. But the filter parent classes already provide good default methods. We should probably have a plugin parent class that sits between the debconf filter and the plugin, just to provide further default methods and any special configuration we need.

How the plugin will talk to its own UI

The plugin will have UI classes like 'PageGtk' or 'PageKde' or 'PageNoninteractive' which will be instantiated by Ubiquity and passed to the filter class. If the right class for the current frontend isn't find, the plugin is ignored. For GUI frontends, a method get_ui() can be called to get frontend-specific objects and info. This will return a dictionary object with keys that the frontend knows. (For GTK, 'widgets' containing a list of pages and 'optional_widgets').

The plugin can install the UI files (.glade, .ui) anywhere it wants, because it is responsible for loading it. The GTK and KDE frontends only want to see actual widgets.

How the plugin is translated

Plugins can use their own debconf template. Ubiquity will do the translation for it, from its template.

How to find the plugin

Plugins drop python page files in /usr/lib/ubiquity/plugins.d/

How to order/add/remove pages

Plugins has a module-wide NAME field that specifies a name to use. The recommended NAME for pages would be similar to the module name that defines the page. For third-party plugins, using a namespace (e.g. 'foo-page' and 'foo-page2') is recommended.

To insert your page into the sequence of pages, use AFTER or BEFORE fields.

AFTER='goober-language'
BEFORE=['language', None]

Pages can be hidden from a HIDDEN field (a list would also work):

HIDDEN='language'
  • If neither AFTER or BEFORE are specified, the page will not be used, but may still specify other plugin options, like HIDDEN. (i.e. a page that only hides, and does not insert itself)
  • If AFTER or BEFORE is a list of names, the first component that Ubiquity recognizes is used.
  • If both AFTER and BEFORE are specified, Ubiquity will use AFTER if it recognizes it, else use BEFORE.
  • If Ubiquity still can't find a value that it recognizes, the page is ignored (and other plugin options like HIDDEN are not used -- it's likely this page is optional, pending installation of some package we don't have or was written for an older Ubiquity).

  • If AFTER's value is None or otherwise 'Python-false', it is inserted at the beginning of the page sequence.
  • If BEFORE's value is None or otherwise 'Python-false', it is inserted at the end of the page sequence.

Ties (where two pages specify 'after language' for example) are broken by file system ordering in plugins.d. Ubiquity reads the plugins.d files in order, and resolves names as they come in. The number before the filename can thus be considered a priority then for ties. i.e. if both 10-mypage and 22-otherpage specify AFTER=language, 22 would be first in the resulting sequence, then 10.

PageGtk Design

The PageGtk class must provide an init that takes at least one argument. Additional keyword arguments are allowed, but not currently used.

This first argument is a 'controller' class, specified below. This is how PageGtk can provide feedback to the 'real' frontend.

It must provide a method called get_ui(self) that returns a dictionary with 'widgets' and optionally, 'optional_widgets' fields. 'widgets' is a list or singleton of GTK pages (usually just a single page).

PageKde Design

The PageKde class must provide an init that takes at least one argument. Additional keyword arguments are allowed, but not currently used.

This first argument is a 'controller' class, specified below. This is how PageKde can provide feedback to the 'real' frontend.

It must provide a method called get_ui(self) that returns a dictionary with 'widgets' and 'breadcrumb' fields. 'widgets' is a list or singleton of Qt pages (usually just a single page). 'breadcrumb' is a debconf template key for the list of stages. Setting 'breadcrumb' to None indicates that you don't want the breadcrumb list visible for your page.

Controller Design

The controller object passed to Page UI classes is a simple mechanism for talking to the main frontend. Right now the only methods allowed are:

  • dbfilter (lets you have access to your Page class)
  • oem_config, oem_user_config (flags for which mode we're in)
  • is_language_page (set this to True if you intend to change languages)
  • translate(self, lang=None, just_me=True, reget=False) (retranslate the UI)
  • allow_go_forward(self, bool) (page starts with forward allowed)
  • allow_go_backward(self, bool) (page starts with backward allowed)
  • go_forward(self) (clicks forward)
  • go_backward(self) (clicks back)
  • go_to_page(self, page) (jumps to the given page)

Applying the Page

If your plugin wants to perform install-time work, it can define a Install class that derives from the InstallPlugin class.

InstallPlugin is a Plugin and follows the same conventions -- it is a debconf filter that has prepare and cleanup methods. But additionally and more relevantly for us, it adds an install method.

install is passed a target directory (usually '/target' or '/'), a progress class (more on that below), and possibly some future arguments. It's recommended that plugins use *args and **kwargs to gracefully handle future extensions.

If install returns a Python-true value, it is considered an error (ala command line programs). The error value (a string or a number) will be printed to syslog.

Progress API

install is passed a progress class that can provide updates on what your plugin is doing. Currently, it only supports passing a text description. Each plugin takes up 1% on the install progress bar.

   1 def info(self, title):
   2     pass

Install Example

   1 def install(self, target, progress, *args, **kwargs):
   2     progress.info('ubiquity/install/timezone')
   3     return InstallPlugin.install(target, progress, *args, **kwargs)

Example

In /usr/lib/ubiquity/plugins.d/30-foo.py

   1 import os
   2 from ubiquity.plugin import *
   3 
   4 # Plugin settings
   5 NAME='foo'
   6 BEFORE='usersetup'
   7 
   8 class PageGtk:
   9     def __init__(self, controller):
  10         import gtk
  11         self.controller = controller
  12         self.builder = gtk.Builder()
  13         self.builder.add_from_file('/usr/share/ubiquity-foo/foo.ui')
  14         self.builder.connect_signals(self)
  15     def get_ui(self):
  16         return self.builder.get_object('foo')
  17 
  18     # Custom foo controls
  19     def get_bar(self):
  20         button = self.builder.get_object('bar')
  21         return button.get_active()
  22     def set_bar(self, on):
  23         button = self.builder.get_object('bar')
  24         button.set_active(on)
  25 
  26 class Page(Plugin):
  27     def prepare(self, unfiltered=False):
  28         # self.frontend is PageGtk above
  29         bar = self.db.get('foo/bar')
  30         self.ui.set_bar(bar)
  31         return ([], ['foo/bar'])
  32 
  33     def ok_handler(self):
  34         bar = self.ui.get_bar()
  35         self.preseed_bool('foo/bar', bar)
  36         Plugin.ok_handler(self)
  37 
  38 class Install(InstallPlugin):
  39     def install(self, target, progress, *args, **kwargs):
  40         progress.info('foo/applying')
  41         cmd = 'touch' if self.db.get('foo/bar') else 'rm'
  42         rv = os.system([cmd, os.path.join(target, '/etc/bar')])
  43         if rv:
  44             return rv
  45         return InstallPlugin.install(target, progress, *args, **kwargs)

Ubiquity/Plugins (last edited 2013-01-16 10:57:22 by fourdollars)