Skip to content

Development guide

JensTimmerman edited this page Dec 7, 2012 · 16 revisions

So you want to extend EasyBuild? Great idea!

There are three major parts to EasyBuild:

  • Framework with:

    • main.py: the main script which sets up the rest of the application and determines the build-order, this is called through the eb command which sets a correct Python environment before starting.
    • framework: the general EasyBlock, EasyConfig and Extension classes.
    • tools: General utilities used trought the framework code.
      • asyncprocess: Written by Josiah Carlson (http://code.activestate.com/recipes/440554/ )
      • build_log: wrapper around Python's logger (will be replaced by fancylooger soon
      • config: proxy for the Configuration
      • filetools: file management and running commands
      • module_generator: generates module-files
      • modules: interface to the module command
      • repository: interface to the configured source control system such as pySVN subversion or the filesystem.
      • toolchains: Contains implementations of toolchain components with their switches and flags. This inculdes the compiler, math libs, mpi implementations...
      • others...
  • EasyBlocks: Custom EasyBlock classes are grouped here.

  • EasyConfig: A (huge) list of example EasyConfigs

How to build a custom EasyBlock-class

First check if the available functionality is already available in EasyBuild. Some Generic EasyBlock classes are very flexible and might be sufficient for your needs. If not, go ahead and create a new class inheriting from EasyBlock or a Generic subclass or even an specific software package class (you can get a quick overview of all available classes using --dump-classes).

Each step in the build-process consists out of one or more methods which you can override. If you want to skip a step, just implement it with pass. See Easyconfig files for options for each step in the default implementation.

  1. Get the sources:
    • fetch_step: fetches files, either from local filesystem, or from a remote url. (We only support http and ftp for now, git/svn support is on the wishlist, contact us if you feel like adding it)
  2. Generate installation path and create build directory
    • check_readinesstep
    • gen_installdir
    • make_builddir
    • reset_changes: resets changes in the environment
  3. Unpack source in the build directory
  4. Apply patches
    • patch_step
  5. Configure
    • prepare_step: prepare toolkit (using self.tk.prepare)
    • configure_step
  6. Build
    • build_step
  7. Test (run make test or variants)
    • test_step
  8. Install
    • stage_install_step
    • make_installdir
    • install_step
  9. Extensions: build extensions (if any)
    • extensions_step
  10. Upload eb files to repository
    • post_instal_step
  11. Check installed files
    • sanity_check_step
  12. Cleanup
    • cleanup_step
  13. Create module
    • make_module_step

Specification-options

Some specification options are added in an EasyConfig, to see the available options, run eb --avail-easyconfig-params On EasyBlock this returns:

MANDATORY
---------
name:			Name of software
version:		Version of software
toolchain:		Name and version of toolchain
description:		A short description of the software
homepage:		The homepage of the software

EASYBLOCK-SPECIFIC
------------------
tar_config_opts:	Override tar settings as determined by configure.

TOOLCHAIN
---------
toolchainopts:		Extra options for compilers
onlytcmod:		Boolean/string to indicate if the toolchain should only load the environment with module (True) or also set all other variables (False) like compiler CC etc (if string: comma separated list of variables that will be ignored). (default: False)

BUILD
-----
easybuild_version:	EasyBuild-version this spec-file was written for
versionsuffix:		Additional suffix for software version (placed after toolchain name)
versionprefix:		Additional prefix for software version (placed before version and toolchain name)
runtest:		Indicates if a test should be run after make; should specify argument after make (for e.g.,"test" for make test) (default: None)
preconfigopts:		Extra options pre-passed to configure.
configopts:		Extra options passed to configure (default already has --prefix)
premakeopts:		Extra options pre-passed to build command.
makeopts:		Extra options passed to make (default already has -j X)
preinstallopts:		Extra prefix options for installation (default: nothing)
installopts:		Extra options for installation (default: nothing)
unpack_options:		Extra options for unpacking source (default: None)
stop:			Keyword to halt the buildprocess at certain points. Valid are ['cfg', 'source', 'patch', 'prepare', 'configure', 'make', 'install', 'test', 'postproc', 'cleanup', 'extensions']
skip:			Skip existing software (default: False)
parallel:		Degree of parallelism for e.g. make (default: based on the number of cores and restrictions in ulimit)
maxparallel:		Max degree of parallelism (default: None)
sources:		List of source files
source_urls:		List of URLs for source files
patches:		List of patches to apply
tests:			List of test-scripts to run after install. A test script should return a non-zero exit status to fail
sanity_check_paths:	List of files and directories to check (format: {'files':<list>, 'dirs':<list>}, default: {})
sanity_check_commands:	format: [(name, options)] e.g. [('gzip','-h')]. Using a non-tuple is equivalent to (name, '-h')

FILE-MANAGEMENT
---------------
start_dir:		Path to start the make in. If the path is absolute, use that path. If not, this is added to the guessed path.
keeppreviousinstall:	Boolean to keep the previous installation with identical name. (default: False) Experts only!
cleanupoldbuild:	Boolean to remove (True) or backup (False) the previous build directory with identical name or not. (default: True)
cleanupoldinstall:	Boolean to remove (True) or backup (False) the previous install directory with identical name or not. (default: True)
dontcreateinstalldir:	Boolean to create (False) or not create (True) the install directory (default: False)
keepsymlinks:		Boolean to determine whether symlinks are to be kept during copying or if the content of the files pointed to should be copied

DEPENDENCIES
------------
dependencies:		List of dependencies (default: [])
builddependencies:	List of build dependencies (default: [])
osdependencies:		OS dependencies that should be present on the system

LICENSE
-------
license_server:		License server for software
license_serverPort:	Port for license server
key:			Key for installing software
group:			Name of the user group for which the software should be available

EXTENSIONS
----------
exts_list:		List with extensions added to the base installation (default: [])
exts_defaultclass:	List of module for and name of the default extension class (default: None)
exts_filter:		Extension filter details: template for cmd and input to cmd (templates for name, version and src). (default: None)

MODULES
-------
modextravars:		Extra environment variables to be added to module file (default: {})
moduleclass:		Module class to be used for this software (default: base) (valid: ['base', 'compiler', 'lib'])
moduleforceunload:	Force unload of all modules when loading the extension (default: False)
moduleloadnoconflict:	Don't check for conflicts, unload other versions instead (default: False)

OTHER
-----
buildstats:		A list of dicts with build statistics

Adding specification-options

Sometimes you want to add extra options that should be configurable in the EasyConfig file.

EasyConfig files are read by the EasyConfig-class. Instead of overriding this class, we extend the extra_options function in EasyBlock to return list of your specification options. To do so, create the following extra_options method:

from easybuild.framework.easyconfig import CUSTOM
from easybuild.easyblocks.generic import ConfigureMake

class EB_MySoftware(ConfigureMake)

    def __init__(self, *args, **kwargs):
        ConfigureMake.__init__(self, *args, **kwargs)

    @staticmethod
    def extra_options():
        extra_vars = [
                      ('importdeps', [None, 'A list of modules to import when configuring', CUSTOM]),
                      ('config-key', [<default>, <description>, CUSTOM ]),
                      (..., [..., ..., CUSTOM]),
                     ]
        return ConfigureMake.extra_options(extra_vars)

Afterwards you can read the requested configuration using self.cfg['X'] in your easyblock. If a key was not set in the EasyConfig file, the default value will be returned.

Testing

Unittests should be added (if possible) for each added feature. You can run the unittests with python -m unittest easybuild.test.suite

Adding more unittests should be done by creating a new test module, adding a suite() method which returns a TestSuite object with all the testcases in it.

In easybuild/test/suite.py you should add it to the list of modules then, so it will be included when somebody runs the suite.

Regression test

To run a full regression test: first append the easybuild directory to the PYTHONPATH. Also, set your MODULEPATH to something which doesn't have dependencies installed. then run python easybuild/scripts/regtest.py. (see -h) for specific options.

output will be placed in the current directory in easybuild-test-TIMESTAMP. You can aggregate the results into a single xml file. by using the -a option.

Make sure to populate the easybuild_config.py with settings that make sense for the regression tester.

The final xml will contain JUnit-compatible xml. Per build there is either a failure (with reason). the buildstats for each application and also a summary in a top comment.

Clone this wiki locally