2007-02-06 21:25:36 +00:00
|
|
|
|
<chapter xmlns="http://docbook.org/ns/docbook"
|
|
|
|
|
xmlns:xlink="http://www.w3.org/1999/xlink">
|
|
|
|
|
|
|
|
|
|
<title>Development</title>
|
|
|
|
|
|
|
|
|
|
<para>This chapter has some random notes on hacking on
|
|
|
|
|
NixOS.</para>
|
|
|
|
|
|
|
|
|
|
|
2011-05-01 11:24:37 +01:00
|
|
|
|
<!--===============================================================-->
|
|
|
|
|
|
2009-01-25 15:49:27 +00:00
|
|
|
|
<section>
|
|
|
|
|
|
|
|
|
|
<title>Extending NixOS</title>
|
|
|
|
|
|
2011-05-01 11:24:37 +01:00
|
|
|
|
<para>NixOS is based on a modular system for declarative configuration.
|
|
|
|
|
This system combines multiple <emphasis>modules</emphasis> to produce one
|
|
|
|
|
configuration. One of the module which compose your computer
|
|
|
|
|
configuration is <filename>/etc/nixos/configuration.nix</filename>. Other
|
|
|
|
|
modules are available under NixOS <filename>modules</filename>
|
|
|
|
|
directory</para>
|
|
|
|
|
|
|
|
|
|
<para>A module is a file which handles one specific part of the
|
|
|
|
|
configuration. This part of the configuration could correspond to an
|
|
|
|
|
hardware, a service, network settings, or preferences. A module
|
|
|
|
|
configuration does not have to handle everything from scratch, it can base
|
|
|
|
|
its configuration on other configurations provided by other modules. Thus
|
|
|
|
|
a module can <emphasis>define</emphasis> options to setup its
|
|
|
|
|
configuration, and it can also <emphasis>declare</emphasis> options to be
|
|
|
|
|
fed by other modules.</para>
|
|
|
|
|
|
|
|
|
|
<!-- module syntax -->
|
|
|
|
|
|
|
|
|
|
<para xml:id="para-module-syn">A module is a file which contains a Nix
|
|
|
|
|
expression. This expression should be either an expression which gets
|
|
|
|
|
evaluated into an attribute set or a function which returns an attribute
|
|
|
|
|
set.</para>
|
|
|
|
|
|
|
|
|
|
<para>When the expression is a function, it should expect only one argument
|
|
|
|
|
which is an attribute set containing an attribute
|
|
|
|
|
named <varname>config</varname> and another attribute
|
|
|
|
|
named <varname>pkgs</varname>. The <varname>config</varname> attribute
|
|
|
|
|
contains the result of the merge of all modules. This attribute is
|
|
|
|
|
evaluated lazily, such as any Nix expression. For more details on how
|
|
|
|
|
options are merged, see the details in <xref linkend="para-opt-decl"/>.
|
|
|
|
|
The <varname>pkgs</varname> attribute
|
|
|
|
|
contains <emphasis>nixpkgs</emphasis> attribute set of packages. This
|
|
|
|
|
attribute is necessary for declaring options.</para>
|
|
|
|
|
|
|
|
|
|
<example xml:id='module-syntax'><title>Usual module content</title>
|
|
|
|
|
<programlisting>
|
|
|
|
|
{config, pkgs, ...}: <co xml:id='module-syntax-1' />
|
2009-01-25 15:49:27 +00:00
|
|
|
|
|
2011-05-01 11:24:37 +01:00
|
|
|
|
{
|
|
|
|
|
imports = [
|
|
|
|
|
<co xml:id='module-syntax-2' />
|
|
|
|
|
];
|
2009-01-25 15:49:27 +00:00
|
|
|
|
|
2011-05-01 11:24:37 +01:00
|
|
|
|
options = {
|
|
|
|
|
<co xml:id='module-syntax-3' />
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
config = {
|
|
|
|
|
<co xml:id='module-syntax-4' />
|
|
|
|
|
};
|
|
|
|
|
}</programlisting>
|
|
|
|
|
</example>
|
|
|
|
|
|
|
|
|
|
<para><xref linkend='module-syntax' /> Illustrates
|
|
|
|
|
a <emphasis>module</emphasis> skeleton.
|
|
|
|
|
|
|
|
|
|
<calloutlist>
|
|
|
|
|
<callout arearefs='module-syntax-1'>
|
|
|
|
|
<para>This line makes the current Nix expression a function. This
|
|
|
|
|
line can be omitted if there is no reference to <varname>pkgs</varname>
|
|
|
|
|
and <varname>config</varname> inside the module.</para>
|
|
|
|
|
</callout>
|
|
|
|
|
|
|
|
|
|
<callout arearefs='module-syntax-2'>
|
|
|
|
|
<para>This list is used to enumerate path to other modules which are
|
|
|
|
|
declaring options used by the current module. In NixOS, default modules
|
|
|
|
|
are listed in the file <filename>modules/module-list.nix</filename>.
|
|
|
|
|
The default modules don't need to be added in the import list.</para>
|
|
|
|
|
</callout>
|
|
|
|
|
|
|
|
|
|
<callout arearefs='module-syntax-3'>
|
|
|
|
|
<para>This attribute set contains an attribute set of <emphasis>option
|
|
|
|
|
declaration</emphasis>.</para>
|
|
|
|
|
</callout>
|
|
|
|
|
|
|
|
|
|
<callout arearefs='module-syntax-4'>
|
|
|
|
|
<para>This attribute set contains an attribute set of <emphasis>option
|
|
|
|
|
definitions</emphasis>. If the module does not have any imported
|
|
|
|
|
modules or any option declarations, then this attribute set can be used
|
|
|
|
|
in place of its parent attribute set. This is a common case for simple
|
|
|
|
|
modules such
|
|
|
|
|
as <filename>/etc/nixos/configuration.nix</filename>.</para>
|
|
|
|
|
</callout>
|
|
|
|
|
</calloutlist>
|
|
|
|
|
|
|
|
|
|
</para>
|
|
|
|
|
|
|
|
|
|
<!-- option definitions -->
|
|
|
|
|
|
|
|
|
|
<para xml:id="para-opt-def">A module defines a configuration which would be
|
|
|
|
|
interpreted by other modules. To define a configuration, a module needs
|
|
|
|
|
to provide option definitions. An option definition is a simple
|
|
|
|
|
attribute assignment.</para>
|
|
|
|
|
|
|
|
|
|
<para>Option definitions are made in a declarative manner. Without
|
|
|
|
|
properties, options will always be defined with the same value. To
|
|
|
|
|
introduce more flexibility in the system, option definitions are guarded
|
|
|
|
|
by <emphasis>properties</emphasis>.</para>
|
|
|
|
|
|
|
|
|
|
<para>Properties are means to introduce conditional values inside option
|
|
|
|
|
definitions. This conditional values can be distinguished in two
|
|
|
|
|
categories. The condition which are local to the current configuration
|
|
|
|
|
and conditions which are dependent on others configurations. Local
|
|
|
|
|
properties are <varname>mkIf</varname>, <varname>mkAlways</varname>
|
|
|
|
|
and <varname>mkAssert</varname>. Global properties
|
|
|
|
|
are <varname>mkOverride</varname>, <varname>mkDefault</varname>
|
|
|
|
|
and <varname>mkOrder</varname>.</para>
|
|
|
|
|
|
|
|
|
|
<para><varname>mkIf</varname> is used to remove the option definitions which
|
|
|
|
|
are below it if the condition is evaluated to
|
|
|
|
|
false. <varname>mkAssert</varname> expects the condition to be evaluated
|
|
|
|
|
to true otherwise it raises an error message. <varname>mkAlways</varname>
|
|
|
|
|
is used to ignore all the <varname>mkIf</varname>
|
|
|
|
|
and <varname>mkAssert</varname> which have been made
|
|
|
|
|
previously. <varname>mkAlways</varname> and <varname>mkAssert</varname>
|
|
|
|
|
are often used together to set an option value and to ensure that it has
|
|
|
|
|
not been masked by another one.</para>
|
|
|
|
|
|
|
|
|
|
<para><varname>mkOverride</varname> is used to mask previous definitions if
|
|
|
|
|
the current value has a lower mask number. The mask value is 100 (default)
|
|
|
|
|
for any option definition which does not use this property.
|
|
|
|
|
Thus, <varname>mkDefault</varname> is just a short-cut with a higher mask
|
|
|
|
|
(1000) than the default mask value. This means that a module can set an
|
|
|
|
|
option definition as a preference, and still let another module defining
|
|
|
|
|
it with a different value without using any property.</para>
|
|
|
|
|
|
|
|
|
|
<para><varname>mkOrder</varname> is used to sort definitions based on the
|
|
|
|
|
rank number. The rank number will sort all options definitions before
|
|
|
|
|
giving the sorted list of option definition to the merge function defined
|
|
|
|
|
in the option declaration. A lower rank will move the definition to the
|
|
|
|
|
beginning and a higher rank will move the option toward the end. The
|
|
|
|
|
default rank is 100.</para>
|
|
|
|
|
|
|
|
|
|
<!-- option declarations -->
|
|
|
|
|
|
|
|
|
|
<para xml:id="para-opt-decl">A module may declare options which are used by
|
|
|
|
|
other module to change the configuration provided by the current module.
|
|
|
|
|
Changes to the option definitions are made with properties which are using
|
|
|
|
|
values extracted from the result of the merge of all modules
|
|
|
|
|
(the <varname>config</varname> argument).</para>
|
|
|
|
|
|
|
|
|
|
<para>The <varname>config</varname> argument reproduce the same hierarchy of
|
|
|
|
|
all options declared in all modules. For each option, the result of the
|
|
|
|
|
option is available, it is either the default value or the merge of all
|
|
|
|
|
definitions of the option.</para>
|
|
|
|
|
|
|
|
|
|
<para>Options are declared with the
|
|
|
|
|
function <varname>pkgs.lib.mkOption</varname>. This function expects an
|
|
|
|
|
attribute set which at least provides a description. A default value, an
|
|
|
|
|
example, a type, a merge function and a post-process function can be
|
|
|
|
|
added.</para>
|
|
|
|
|
|
|
|
|
|
<para>Types are used to provide a merge strategy for options and to ensure
|
|
|
|
|
the type of each option definitions. They are defined
|
|
|
|
|
in <varname>pkgs.lib.types</varname>.</para>
|
|
|
|
|
|
|
|
|
|
<para>The merge function expects a list of option definitions and merge
|
|
|
|
|
them to obtain one result of the same type.</para>
|
|
|
|
|
|
|
|
|
|
<para>The post-process function (named <varname>apply</varname>) takes the
|
|
|
|
|
result of the merge or of the default value, and produce an output which
|
|
|
|
|
could have a different type than the type expected by the option.</para>
|
|
|
|
|
|
|
|
|
|
<!-- end -->
|
|
|
|
|
|
|
|
|
|
<example xml:id='locate-example'><title>Locate Module Example</title>
|
2009-01-25 15:49:27 +00:00
|
|
|
|
<programlisting>
|
2011-05-01 11:24:37 +01:00
|
|
|
|
{config, pkgs, ...}:
|
2009-01-25 15:49:27 +00:00
|
|
|
|
|
2011-05-01 11:24:37 +01:00
|
|
|
|
with pkgs.lib;
|
2009-01-25 15:49:27 +00:00
|
|
|
|
|
2011-05-01 11:24:37 +01:00
|
|
|
|
let
|
|
|
|
|
cfg = config.services.locate;
|
2009-01-25 15:49:27 +00:00
|
|
|
|
locatedb = "/var/cache/locatedb";
|
|
|
|
|
logfile = "/var/log/updatedb";
|
2011-05-01 11:24:37 +01:00
|
|
|
|
cmd =''root updatedb --localuser=nobody --output=${locatedb} > ${logfile}'';
|
|
|
|
|
|
|
|
|
|
mkCheck = x:
|
|
|
|
|
mkIf cfg.enable (
|
|
|
|
|
mkAssert config.services.cron.enable ''
|
|
|
|
|
The cron daemon is not enabled, required by services.locate.enable.
|
|
|
|
|
''
|
|
|
|
|
x
|
|
|
|
|
)
|
2009-01-25 15:49:27 +00:00
|
|
|
|
in
|
|
|
|
|
|
2009-09-13 23:13:19 +01:00
|
|
|
|
{
|
2011-05-01 11:24:37 +01:00
|
|
|
|
imports = [
|
|
|
|
|
/etc/nixos/nixos/modules/services/scheduling/cron.nix
|
2009-01-25 15:49:27 +00:00
|
|
|
|
];
|
|
|
|
|
|
2011-05-01 11:24:37 +01:00
|
|
|
|
options = {
|
2009-09-13 23:13:19 +01:00
|
|
|
|
services.locate = {
|
|
|
|
|
enable = mkOption {
|
|
|
|
|
default = false;
|
|
|
|
|
example = true;
|
2011-05-01 11:24:37 +01:00
|
|
|
|
type = with types; bool;
|
2009-09-13 23:13:19 +01:00
|
|
|
|
description = ''
|
|
|
|
|
If enabled, NixOS will periodically update the database of
|
|
|
|
|
files used by the <command>locate</command> command.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
period = mkOption {
|
|
|
|
|
default = "15 02 * * *";
|
2011-05-01 11:24:37 +01:00
|
|
|
|
type = with types; uniq string;
|
2009-09-13 23:13:19 +01:00
|
|
|
|
description = ''
|
|
|
|
|
This option defines (in the format used by cron) when the
|
|
|
|
|
locate database is updated.
|
|
|
|
|
The default is to update at 02:15 (at night) every day.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
};
|
|
|
|
|
};
|
|
|
|
|
|
2011-05-01 11:24:37 +01:00
|
|
|
|
config = mkCheck {
|
2009-09-13 23:13:19 +01:00
|
|
|
|
services.cron = {
|
2011-05-01 11:24:37 +01:00
|
|
|
|
enable = mkAlways cfg.enable;
|
|
|
|
|
systemCronJobs = "${cfg.period} root ${cmd}";
|
2009-01-25 15:49:27 +00:00
|
|
|
|
};
|
|
|
|
|
};
|
|
|
|
|
}</programlisting>
|
|
|
|
|
</example>
|
|
|
|
|
|
2011-05-01 11:24:37 +01:00
|
|
|
|
<para><xref linkend='locate-example' /> illustrates a module which handles
|
|
|
|
|
the regular update of the database which index all files on the file
|
|
|
|
|
system. This modules has option definitions to rely on the cron service
|
|
|
|
|
to run the command at predefined dates. In addition, this modules
|
|
|
|
|
provides option declarations to enable the indexing and to use different
|
|
|
|
|
period of time to run the indexing. Properties are used to prevent
|
|
|
|
|
ambiguous definitions of option (enable locate service and disable cron
|
|
|
|
|
services) and to ensure that no options would be defined if the locate
|
|
|
|
|
service is not enabled.</para>
|
2009-01-25 15:49:27 +00:00
|
|
|
|
|
2011-05-01 11:24:37 +01:00
|
|
|
|
</section>
|
2009-01-25 15:49:27 +00:00
|
|
|
|
|
|
|
|
|
|
2011-05-01 11:24:37 +01:00
|
|
|
|
<!--===============================================================-->
|
2009-01-25 15:49:27 +00:00
|
|
|
|
|
2007-02-06 21:25:36 +00:00
|
|
|
|
<section>
|
|
|
|
|
|
|
|
|
|
<title>Building specific parts of NixOS</title>
|
|
|
|
|
|
|
|
|
|
<para>
|
|
|
|
|
|
|
|
|
|
<screen>
|
2009-09-13 23:13:19 +01:00
|
|
|
|
$ nix-build /etc/nixos/nixos -A <replaceable>attr</replaceable></screen>
|
2007-02-06 21:25:36 +00:00
|
|
|
|
|
|
|
|
|
where <replaceable>attr</replaceable> is an attribute in
|
2008-02-08 22:43:59 +00:00
|
|
|
|
<filename>/etc/nixos/nixos/default.nix</filename>. Attributes of interest include:
|
|
|
|
|
|
|
|
|
|
<variablelist>
|
|
|
|
|
|
|
|
|
|
<varlistentry>
|
2009-09-13 23:13:19 +01:00
|
|
|
|
<term><varname>config</varname></term>
|
|
|
|
|
<listitem><para>The computer configuration generated from
|
2009-09-14 07:56:00 +01:00
|
|
|
|
the <envar>NIXOS_CONFIG</envar> environment variable (default
|
2009-09-13 23:13:19 +01:00
|
|
|
|
is <filename>/etc/nixos/configuration.nix</filename>) with the NixOS
|
|
|
|
|
default set of modules.</para></listitem>
|
2008-02-08 22:43:59 +00:00
|
|
|
|
</varlistentry>
|
|
|
|
|
|
|
|
|
|
<varlistentry>
|
2009-09-13 23:13:19 +01:00
|
|
|
|
<term><varname>system</varname></term>
|
|
|
|
|
<listitem><para>The derivation which build your computer system. It is
|
|
|
|
|
built by the command <command>nixos-rebuild
|
|
|
|
|
build</command></para></listitem>
|
2008-02-08 22:43:59 +00:00
|
|
|
|
</varlistentry>
|
|
|
|
|
|
|
|
|
|
<varlistentry>
|
2009-09-13 23:13:19 +01:00
|
|
|
|
<term><varname>vm</varname></term>
|
|
|
|
|
<listitem><para>The derivation which build your computer system inside a
|
|
|
|
|
virtual machine. It is built by the command <command>nixos-rebuild
|
|
|
|
|
build-vm</command></para></listitem>
|
2008-02-08 22:43:59 +00:00
|
|
|
|
</varlistentry>
|
2009-09-13 23:13:19 +01:00
|
|
|
|
</variablelist>
|
2008-02-08 22:43:59 +00:00
|
|
|
|
|
2009-09-13 23:13:19 +01:00
|
|
|
|
</para>
|
2008-02-08 22:43:59 +00:00
|
|
|
|
|
2009-09-13 23:13:19 +01:00
|
|
|
|
<para>
|
|
|
|
|
Most parts of NixOS can be build through the <varname>config</varname>
|
|
|
|
|
attribute set. This attribute set allows you to have a view of the merged
|
|
|
|
|
option definitions and all its derivations. Important derivations are store
|
|
|
|
|
inside the option <option>system.build</option> and can be listed with the
|
|
|
|
|
command <command>nix-instantiate --xml --eval-only /etc/nixos/nixos -A
|
|
|
|
|
config.system.build</command>
|
2009-10-05 14:17:45 +01:00
|
|
|
|
</para>
|
2008-02-08 22:43:59 +00:00
|
|
|
|
|
2009-09-13 23:13:19 +01:00
|
|
|
|
</section>
|
2008-02-08 22:43:59 +00:00
|
|
|
|
|
2009-09-13 23:13:19 +01:00
|
|
|
|
|
2011-05-01 11:24:37 +01:00
|
|
|
|
<!--===============================================================-->
|
|
|
|
|
|
2009-09-13 23:13:19 +01:00
|
|
|
|
<section>
|
|
|
|
|
|
|
|
|
|
<title>Building your own NixOS CD</title>
|
|
|
|
|
|
|
|
|
|
<para>Building a NixOS CD is as easy as configuring your own computer. The
|
|
|
|
|
idea is to use another module which will replace
|
|
|
|
|
your <filename>configuration.nix</filename> to configure the system that
|
|
|
|
|
would be install on the CD.</para>
|
|
|
|
|
|
|
|
|
|
<para>Default CD/DVD configurations are available
|
|
|
|
|
inside <filename>nixos/modules/installer/cd-dvd</filename>. To build them
|
2009-09-14 07:56:00 +01:00
|
|
|
|
you have to set <envar>NIXOS_CONFIG</envar> before
|
2009-09-13 23:13:19 +01:00
|
|
|
|
running <command>nix-build</command> to build the ISO.
|
|
|
|
|
|
|
|
|
|
<screen>
|
|
|
|
|
$ export NIXOS_CONFIG=/etc/nixos/nixos/modules/installer/cd-dvd/installation-cd-minimal.nix
|
|
|
|
|
$ nix-build /etc/nixos/nixos -A config.system.build.isoImage</screen>
|
2008-02-08 22:43:59 +00:00
|
|
|
|
|
|
|
|
|
</para>
|
2007-02-06 21:25:36 +00:00
|
|
|
|
|
2009-09-13 23:13:19 +01:00
|
|
|
|
<para>Before burning your CD/DVD, you can check the content of the image by mounting anywhere like
|
|
|
|
|
suggested by the following command:
|
2007-02-06 21:25:36 +00:00
|
|
|
|
|
2009-09-13 23:13:19 +01:00
|
|
|
|
<screen>
|
|
|
|
|
$ mount -o loop -t iso9660 ./result/iso/cd.iso /mnt/iso</screen>
|
|
|
|
|
|
|
|
|
|
</para>
|
|
|
|
|
|
|
|
|
|
</section>
|
2007-02-06 21:25:36 +00:00
|
|
|
|
|
|
|
|
|
<section>
|
|
|
|
|
|
|
|
|
|
<title>Testing the installer</title>
|
|
|
|
|
|
2009-09-13 23:13:19 +01:00
|
|
|
|
|
2007-02-06 21:25:36 +00:00
|
|
|
|
<para>Building, burning, and booting from an installation CD is rather
|
|
|
|
|
tedious, so here is a quick way to see if the installer works
|
|
|
|
|
properly:
|
|
|
|
|
|
|
|
|
|
<screen>
|
2009-09-13 23:13:19 +01:00
|
|
|
|
$ export NIXOS_CONFIG=/etc/nixos/nixos/modules/installer/cd-dvd/installation-cd-minimal.nix
|
|
|
|
|
$ nix-build /etc/nixos/nixos -A config.system.build.nixosInstall
|
2007-02-06 21:25:36 +00:00
|
|
|
|
$ dd if=/dev/zero of=diskimage seek=2G count=0 bs=1
|
|
|
|
|
$ yes | mke2fs -j diskimage
|
|
|
|
|
$ mount -o loop diskimage /mnt
|
|
|
|
|
$ ./result/bin/nixos-install</screen>
|
|
|
|
|
|
|
|
|
|
</para>
|
|
|
|
|
|
|
|
|
|
</section>
|
|
|
|
|
|
|
|
|
|
|
2011-05-01 11:24:37 +01:00
|
|
|
|
<!--===============================================================-->
|
|
|
|
|
|
2008-08-26 13:48:27 +01:00
|
|
|
|
<section>
|
|
|
|
|
|
|
|
|
|
<title>Testing the <literal>initrd</literal></title>
|
|
|
|
|
|
|
|
|
|
<para>A quick way to test whether the kernel and the initial ramdisk
|
|
|
|
|
boot correctly is to use QEMU’s <option>-kernel</option> and
|
|
|
|
|
<option>-initrd</option> options:
|
|
|
|
|
|
|
|
|
|
<screen>
|
2009-09-13 23:13:19 +01:00
|
|
|
|
$ nix-build /etc/nixos/nixos -A config.system.build.initialRamdisk -o initrd
|
|
|
|
|
$ nix-build /etc/nixos/nixos -A config.system.build.kernel -o kernel
|
2010-02-18 11:35:39 +00:00
|
|
|
|
$ qemu-system-x86_64 -kernel ./kernel/bzImage -initrd ./initrd/initrd -hda /dev/null
|
2008-08-26 13:48:27 +01:00
|
|
|
|
</screen>
|
|
|
|
|
|
|
|
|
|
</para>
|
|
|
|
|
|
|
|
|
|
</section>
|
|
|
|
|
|
2011-03-08 20:55:43 +00:00
|
|
|
|
<section>
|
|
|
|
|
|
|
|
|
|
<title>Whole-system testing using virtual machines</title>
|
|
|
|
|
|
|
|
|
|
<para>
|
|
|
|
|
Complete NixOS GNU/Linux systems can be tested in virtual machines
|
|
|
|
|
(VMs). This makes it possible to test a system upgrade or
|
|
|
|
|
configuration change before rebooting into it, using the
|
|
|
|
|
<command>nixos-rebuild build-vm</command> or
|
|
|
|
|
<command>nixos-rebuild build-vm-with-bootloader</command> command.
|
|
|
|
|
</para>
|
|
|
|
|
|
|
|
|
|
<para>
|
|
|
|
|
<!-- The following is adapted from
|
|
|
|
|
http://wiki.nixos.org/wiki/NixOS_VM_tests, by Eelco Dolstra. -->
|
|
|
|
|
|
|
|
|
|
The <filename>tests/</filename> directory in the NixOS source tree
|
|
|
|
|
contains several <emphasis>whole-system unit tests</emphasis>.
|
2011-03-08 21:54:18 +00:00
|
|
|
|
These tests can be run<footnote><para>NixOS tests can be run both from
|
2011-03-08 20:55:43 +00:00
|
|
|
|
NixOS and from a non-NixOS GNU/Linux distribution, provided the
|
2011-03-08 21:54:18 +00:00
|
|
|
|
Nix package manager is installed.</para></footnote> from the NixOS
|
2011-03-08 20:55:43 +00:00
|
|
|
|
source tree as follows:
|
|
|
|
|
|
|
|
|
|
<screen>
|
|
|
|
|
$ nix-build tests/ -A nfs.test
|
|
|
|
|
</screen>
|
|
|
|
|
|
|
|
|
|
This performs an automated test of the NFS client and server
|
|
|
|
|
functionality in the Linux kernel, including file locking
|
|
|
|
|
semantics (e.g., whether locks are maintained across server
|
|
|
|
|
crashes). It will first build or download all the dependencies of
|
|
|
|
|
the test (e.g., all packages needed to run a NixOS VM). The test
|
|
|
|
|
is defined in <link
|
|
|
|
|
xlink:href="https://svn.nixos.org/repos/nix/nixos/trunk/tests/nfs.nix">
|
|
|
|
|
<filename>tests/nfs.nix</filename></link>. If the test succeeds,
|
|
|
|
|
<command>nix-build</command> will place a symlink
|
|
|
|
|
<filename>./result</filename> in the current directory pointing at
|
|
|
|
|
the location in the Nix store of the test results (e.g.,
|
|
|
|
|
screenshots, test reports, and so on). In particular, a
|
|
|
|
|
pretty-printed log of the test is written to
|
2011-03-09 10:38:18 +00:00
|
|
|
|
<filename>log.html</filename>, which can be viewed using a web
|
|
|
|
|
browser like this:
|
2011-03-08 20:55:43 +00:00
|
|
|
|
|
|
|
|
|
<screen>
|
|
|
|
|
$ icecat result/log.html
|
|
|
|
|
</screen>
|
|
|
|
|
</para>
|
|
|
|
|
|
|
|
|
|
<para>
|
|
|
|
|
It is also possible to run the test environment interactively,
|
|
|
|
|
allowing you to experiment with the VMs. For example:
|
|
|
|
|
|
|
|
|
|
<screen>
|
|
|
|
|
$ nix-build tests/ -A nfs.driver
|
|
|
|
|
$ ./result/bin/nixos-run-vms
|
|
|
|
|
</screen>
|
|
|
|
|
|
|
|
|
|
The script <command>nixos-run-vms</command> starts the three
|
|
|
|
|
virtual machines defined in the NFS test using QEMU/KVM. The root
|
|
|
|
|
file system of the VMs is created on the fly and kept across VM
|
|
|
|
|
restarts in
|
2011-03-09 10:38:18 +00:00
|
|
|
|
<filename>./</filename><varname>hostname</varname><filename>.qcow2</filename>.
|
2011-03-08 20:55:43 +00:00
|
|
|
|
</para>
|
|
|
|
|
|
|
|
|
|
<para>
|
|
|
|
|
Finally, the test itself can be run interactively. This is
|
|
|
|
|
particularly useful when developing or debugging a test:
|
|
|
|
|
|
|
|
|
|
<screen>
|
|
|
|
|
$ nix-build tests/ -A nfs.driver
|
|
|
|
|
$ ./result/bin/nixos-test-driver
|
|
|
|
|
starting VDE switch for network 1
|
2011-03-08 21:14:21 +00:00
|
|
|
|
>
|
2011-03-08 20:55:43 +00:00
|
|
|
|
</screen>
|
|
|
|
|
|
|
|
|
|
Perl statements can now be typed in to start or manipulate the
|
|
|
|
|
VMs:
|
|
|
|
|
|
|
|
|
|
<screen>
|
2011-03-08 21:14:21 +00:00
|
|
|
|
> startAll;
|
2011-03-08 20:55:43 +00:00
|
|
|
|
(the VMs start booting)
|
2011-03-08 21:14:21 +00:00
|
|
|
|
> $server->waitForJob("nfs-kernel-nfsd");
|
2011-03-08 21:31:00 +00:00
|
|
|
|
> $client1->succeed("flock -x /data/lock -c 'sleep 100000' &");
|
2011-03-08 21:14:21 +00:00
|
|
|
|
> $client2->fail("flock -n -s /data/lock true");
|
|
|
|
|
> $client1->shutdown;
|
2011-03-08 20:55:43 +00:00
|
|
|
|
(this releases client1's lock)
|
2011-03-08 21:14:21 +00:00
|
|
|
|
> $client2->succeed("flock -n -s /data/lock true");
|
2011-03-08 20:55:43 +00:00
|
|
|
|
</screen>
|
|
|
|
|
|
|
|
|
|
The function <command>testScript</command> executes the entire
|
|
|
|
|
test script and drops you back into the test driver command line
|
|
|
|
|
upon its completion. This allows you to inspect the state of the
|
|
|
|
|
VMs after the test (e.g. to debug the test script).
|
|
|
|
|
</para>
|
|
|
|
|
|
|
|
|
|
<para>
|
|
|
|
|
This and other tests are continuously run on <link
|
|
|
|
|
xlink:href="http://hydra.nixos.org/jobset/nixos/trunk/with-status">the
|
|
|
|
|
Hydra instance at <literal>nixos.org</literal></link>, which
|
|
|
|
|
allows developers to be notified of any regressions introduced by
|
|
|
|
|
a NixOS or Nixpkgs change.
|
|
|
|
|
</para>
|
|
|
|
|
|
2011-06-15 15:12:45 +01:00
|
|
|
|
<para>
|
|
|
|
|
The actual Nix programming interface to VM testing is in NixOS,
|
|
|
|
|
under <link
|
|
|
|
|
xlink:href="https://svn.nixos.org/repos/nix/nixos/trunk/lib/testing.nix">
|
|
|
|
|
<filename>lib/testing.nix</filename></link>. This file defines a
|
|
|
|
|
function which takes an attribute set containing a
|
|
|
|
|
<literal>nixpkgs</literal> attribute (the path to a Nixpkgs
|
|
|
|
|
checkout), and a <literal>system</literal> attribute (the system
|
|
|
|
|
type). It returns an attribute set containing several utility
|
|
|
|
|
functions, among which the main entry point is
|
|
|
|
|
<literal>makeTest</literal>.
|
|
|
|
|
</para>
|
|
|
|
|
|
|
|
|
|
<para>
|
|
|
|
|
The <literal>makeTest</literal> function takes a function similar to
|
|
|
|
|
that found in <link
|
|
|
|
|
xlink:href="https://svn.nixos.org/repos/nix/nixos/trunk/tests/nfs.nix">
|
|
|
|
|
<filename>tests/nfs.nix</filename></link> (discussed above). It
|
|
|
|
|
returns an attribute set containing (among others):
|
|
|
|
|
|
|
|
|
|
<variablelist>
|
|
|
|
|
|
|
|
|
|
<varlistentry>
|
|
|
|
|
<term><varname>test</varname></term>
|
2011-06-15 15:46:06 +01:00
|
|
|
|
<listitem><para>A derivation containing the test log as an HTML file,
|
2011-06-15 15:12:45 +01:00
|
|
|
|
as seen above, suitable for presentation in the Hydra continuous
|
2011-06-15 15:46:06 +01:00
|
|
|
|
build system.</para></listitem>
|
2011-06-15 15:12:45 +01:00
|
|
|
|
</varlistentry>
|
|
|
|
|
|
|
|
|
|
<varlistentry>
|
|
|
|
|
<term><varname>report</varname></term>
|
2011-06-15 15:46:06 +01:00
|
|
|
|
<listitem><para>A derivation containing a code coverage report, with
|
|
|
|
|
meta-data suitable for Hydra.</para></listitem>
|
2011-06-15 15:12:45 +01:00
|
|
|
|
</varlistentry>
|
|
|
|
|
|
|
|
|
|
<varlistentry>
|
|
|
|
|
<term><varname>driver</varname></term>
|
2011-06-15 15:46:06 +01:00
|
|
|
|
<listitem><para>A derivation containing scripts to run the VM test or
|
|
|
|
|
interact with the VM network interactively, as seen above.</para>
|
2011-06-15 15:12:45 +01:00
|
|
|
|
</listitem>
|
|
|
|
|
</varlistentry>
|
|
|
|
|
|
|
|
|
|
</variablelist>
|
|
|
|
|
</para>
|
|
|
|
|
|
|
|
|
|
|
2011-03-08 20:55:43 +00:00
|
|
|
|
</section>
|
2008-08-26 13:48:27 +01:00
|
|
|
|
|
2007-02-06 21:25:36 +00:00
|
|
|
|
</chapter>
|