9.3.  API for YaST2 installation proposal

9.3.1. Motivation

After five releases, YaST2 is now smart enough to make reasonable proposals for (near) every installation setting, thus it is no longer necessary to ask the user that many questions during installation: Most users simply hit the [next] button anyway.

Hence, YaST2 now collects all the individual proposals from its submodules and presents them for confirmation right away. The user can change each individual setting, but he is no longer required to go through all the steps just to change some simple things. The only that (currently) really has to be queried is the installation language - this cannot reasonably be guessed (yet?).

The new YaST2 installation includes the following steps:

  • (Minimal) hardware probing - no user interaction required

  • Language selection - user picks installation language

  • Installation proposal - very much like the old installation summary just before the real installation started, only this time the user CAN change settings by clicking into the summary (or via a separate menu as a fallback).

  • Create / format partitions according to proposal / user selection - no user interaction required

  • Install software packages from CD / DVD / other installation media

After this, all that is remained left is basic system configuration like:

  • X11

  • Network interface(s)

  • Network services

  • Additional hardware (printer, sound card, scanner, ...)

9.3.2. Overview

YaST2 installation modules should cooperate with the main program in a consistent API. General usage:

  • inst_proposal (main program) creates empty dialog with RichText widget

  • inst_proposal calls each sub-module in turn to make proposal

  • user may choose to change individual settings (i.e., clicks on a hyperlink)

  • inst_proposal starts that module's sub-workflow which runs independently. After this, inst_proposal tells all subsequent (all?) modules to check their states and return whether a change of their proposal is necessary after the user interaction.

  • main program calls each sub-module to write the settings to the system

9.3.3. The Dispatcher Interface

Each submodule provides a function dispatcher that can be called with 'CallFunction()'. The function to be called is passed as a parameter to this dispatcher. Parameters to the function are passed as another parameter in a map. The result of each call is a map, the contents of which depend on the function called.

The reason for this additional overhead is to provide extensibility and reusability for the installation workflow: A list of submodules to be called is read from file. This requires, however, that no explicit 'mod::func()' calls are used in 'inst_proposal.ycp'. Rather, the list contains the name of the submodule. Since each submodule is required to provide an identical API, this is sufficient.

Example 9.2. Proposal Example

Proposal is to call

  • input_devices (keyboard, mouse)

  • partitioning

  • software_selection

  • boot_loader

  • timezone

inst_proposal calls

	map result = CallFunction (input_devices( "MakeProposal", $[ "force_reset"     : false,
	"language_changed": false ] ) );
	map result = CallFunction (partitioning ( "MakeProposal", $[ "force_reset"	  : false,
	"language_changed": false ] ) );
	...

      

If the user clicks on the hyperlink on "input_devices" in the proposal display, inst_proposal calls:

	
	map result = CallFunction (input_devices( "AskUser", $[ "has_next": true ] ) );
      

9.3.4. API functions

[Note]Note

If any parameter is marked as "optional", it should only be specified if it contains a meaningful value. Don't add it with a 'nil' value.

The dispatcher provides the following functions:

9.3.5. Dummy Proposal

      /**
 * Module:		proposal_dummy.ycp
 *
 * $Id: dummy_proposal.ycp,v 1.1 2004/02/27 02:37:39 nashif Exp $
 *
 * Author:		Stefan Hundhammer <sh@suse.de>
 *
 * Purpose:		Proposal function dispatcher - dummy version.
 *			Use this as a template for other proposal dispatchers.
 *			Don't forget to replace all fixed values with real values!
 *
 *			See also file proposal-API.txt for details.
 */
{
    textdomain "installation";

    string func  = (string) WFM::Args(0);
    map    param = (map) WFM::Args(1);
    map    ret   = $[];

    if ( func == "MakeProposal" )
    {
	boolean force_reset      = param["force_reset"     ]:false;
	boolean language_changed = param["language_changed"]:false;

	// call some function that makes a proposal here:
	//
	// DummyMod::MakeProposal( force_reset );

	// Fill return map

	ret =
	    $[
	      "raw_proposal" :  [
				 "proposal item #1",
				 "proposal item #2",
				 "proposal item #3"
				],
	      "warning"		: "This is just a dummy proposal!",
	      "warning_level"	: `blocker
	    ];
    }
    else if ( func == "AskUser" )
    {
	boolean has_next = param["has_next"]:false;

	// call some function that displays a user dialog
	// or a sequence of dialogs here:
	//
	// sequence = DummyMod::AskUser( has_next );


	// Fill return map

	ret =
	    $[
	      "workflow_sequence":	`next
	    ];
    }
    else if ( func == "Description" )
    {
	// Fill return map.
	//
	// Static values do just nicely here, no need to call a function.

	ret =
	    $[
	      // this is a heading
	      "rich_text_title"	:	_( "Dummy"  ),
	      // this is a menu entry
	      "menu_title"	:	_( "&Dummy" ),
	      "id"		:	"dummy_stuff"
	    ];
    }
    else if ( func == "Write" )
    {
	// Fill return map.
	//

	ret =
	    $[
	      "success" : true
	    ];
    }

    return ret;
}