Package Notes
This chapter contains information about how to use and maintain the Nix
expressions for a number of specific packages, such as the Linux kernel or
X.org.
Linux kernel
The Nix expressions to build the Linux kernel are in
pkgs/os-specific/linux/kernel.
The function that builds the kernel has an argument
kernelPatches which should be a list of {name,
patch, extraConfig} attribute sets, where name
is the name of the patch (which is included in the kernel’s
meta.description attribute), patch is
the patch itself (possibly compressed), and extraConfig
(optional) is a string specifying extra options to be concatenated to the
kernel configuration file (.config).
The kernel derivation exports an attribute features
specifying whether optional functionality is or isn’t enabled. This is
used in NixOS to implement kernel-specific behaviour. For instance, if the
kernel has the iwlwifi feature (i.e. has built-in support
for Intel wireless chipsets), then NixOS doesn’t have to build the
external iwlwifi package:
modulesTree = [kernel]
++ pkgs.lib.optional (!kernel.features ? iwlwifi) kernelPackages.iwlwifi
++ ...;
How to add a new (major) version of the Linux kernel to Nixpkgs:
Copy the old Nix expression (e.g. linux-2.6.21.nix)
to the new one (e.g. linux-2.6.22.nix) and update
it.
Add the new kernel to all-packages.nix (e.g., create
an attribute kernel_2_6_22).
Now we’re going to update the kernel configuration. First unpack the
kernel. Then for each supported platform (i686,
x86_64, uml) do the following:
Make an copy from the old config (e.g.
config-2.6.21-i686-smp) to the new one (e.g.
config-2.6.22-i686-smp).
Copy the config file for this platform (e.g.
config-2.6.22-i686-smp) to
.config in the kernel source tree.
Run make oldconfig
ARCH={i386,x86_64,um} and answer
all questions. (For the uml configuration, also add
SHELL=bash.) Make sure to keep the configuration
consistent between platforms (i.e. don’t enable some feature on
i686 and disable it on x86_64).
If needed you can also run make menuconfig:
$ nix-env -i ncurses
$ export NIX_CFLAGS_LINK=-lncurses
$ make menuconfig ARCH=arch
Copy .config over the new config file (e.g.
config-2.6.22-i686-smp).
Test building the kernel: nix-build -A kernel_2_6_22.
If it compiles, ship it! For extra credit, try booting NixOS with it.
It may be that the new kernel requires updating the external kernel
modules and kernel-dependent packages listed in the
linuxPackagesFor function in
all-packages.nix (such as the NVIDIA drivers, AUFS,
etc.). If the updated packages aren’t backwards compatible with older
kernels, you may need to keep the older versions around.
X.org
The Nix expressions for the X.org packages reside in
pkgs/servers/x11/xorg/default.nix. This file is
automatically generated from lists of tarballs in an X.org release. As such
it should not be modified directly; rather, you should modify the lists, the
generator script or the file
pkgs/servers/x11/xorg/overrides.nix, in which you can
override or add to the derivations produced by the generator.
The generator is invoked as follows:
$ cd pkgs/servers/x11/xorg
$ cat tarballs-7.5.list extra.list old.list \
| perl ./generate-expr-from-tarballs.pl
For each of the tarballs in the .list files, the script
downloads it, unpacks it, and searches its configure.ac
and *.pc.in files for dependencies. This information is
used to generate default.nix. The generator caches
downloaded tarballs between runs. Pay close attention to the NOT
FOUND: name messages at the end of the
run, since they may indicate missing dependencies. (Some might be optional
dependencies, however.)
A file like tarballs-7.5.list contains all tarballs in
a X.org release. It can be generated like this:
$ export i="mirror://xorg/X11R7.4/src/everything/"
$ cat $(PRINT_PATH=1 nix-prefetch-url $i | tail -n 1) \
| perl -e 'while (<>) { if (/(href|HREF)="([^"]*.bz2)"/) { print "$ENV{'i'}$2\n"; }; }' \
| sort > tarballs-7.4.list
extra.list contains libraries that aren’t part of
X.org proper, but are closely related to it, such as
libxcb. old.list contains some
packages that were removed from X.org, but are still needed by some people
or by other packages (such as imake).
If the expression for a package requires derivation attributes that the
generator cannot figure out automatically (say, patches
or a postInstall hook), you should modify
pkgs/servers/x11/xorg/overrides.nix.
Eclipse
The Nix expressions related to the Eclipse platform and IDE are in
pkgs/applications/editors/eclipse.
Nixpkgs provides a number of packages that will install Eclipse in its
various forms. These range from the bare-bones Eclipse Platform to the more
fully featured Eclipse SDK or Scala-IDE packages and multiple version are
often available. It is possible to list available Eclipse packages by
issuing the command:
$ nix-env -f '<nixpkgs>' -qaP -A eclipses --description
Once an Eclipse variant is installed it can be run using the
eclipse command, as expected. From within Eclipse it is
then possible to install plugins in the usual manner by either manually
specifying an Eclipse update site or by installing the Marketplace Client
plugin and using it to discover and install other plugins. This installation
method provides an Eclipse installation that closely resemble a manually
installed Eclipse.
If you prefer to install plugins in a more declarative manner then Nixpkgs
also offer a number of Eclipse plugins that can be installed in an
Eclipse environment. This type of environment is
created using the function eclipseWithPlugins found
inside the nixpkgs.eclipses attribute set. This function
takes as argument { eclipse, plugins ? [], jvmArgs ? [] }
where eclipse is a one of the Eclipse packages described
above, plugins is a list of plugin derivations, and
jvmArgs is a list of arguments given to the JVM running
the Eclipse. For example, say you wish to install the latest Eclipse
Platform with the popular Eclipse Color Theme plugin and also allow Eclipse
to use more RAM. You could then add
packageOverrides = pkgs: {
myEclipse = with pkgs.eclipses; eclipseWithPlugins {
eclipse = eclipse-platform;
jvmArgs = [ "-Xmx2048m" ];
plugins = [ plugins.color-theme ];
};
}
to your Nixpkgs configuration
(~/.config/nixpkgs/config.nix) and install it by
running nix-env -f '<nixpkgs>' -iA myEclipse and
afterward run Eclipse as usual. It is possible to find out which plugins are
available for installation using eclipseWithPlugins by
running
$ nix-env -f '<nixpkgs>' -qaP -A eclipses.plugins --description
If there is a need to install plugins that are not available in Nixpkgs then
it may be possible to define these plugins outside Nixpkgs using the
buildEclipseUpdateSite and
buildEclipsePlugin functions found in the
nixpkgs.eclipses.plugins attribute set. Use the
buildEclipseUpdateSite function to install a plugin
distributed as an Eclipse update site. This function takes { name,
src } as argument where src indicates the
Eclipse update site archive. All Eclipse features and plugins within the
downloaded update site will be installed. When an update site archive is not
available then the buildEclipsePlugin function can be
used to install a plugin that consists of a pair of feature and plugin JARs.
This function takes an argument { name, srcFeature, srcPlugin
} where srcFeature and
srcPlugin are the feature and plugin JARs, respectively.
Expanding the previous example with two plugins using the above functions we
have
packageOverrides = pkgs: {
myEclipse = with pkgs.eclipses; eclipseWithPlugins {
eclipse = eclipse-platform;
jvmArgs = [ "-Xmx2048m" ];
plugins = [
plugins.color-theme
(plugins.buildEclipsePlugin {
name = "myplugin1-1.0";
srcFeature = fetchurl {
url = "http://…/features/myplugin1.jar";
sha256 = "123…";
};
srcPlugin = fetchurl {
url = "http://…/plugins/myplugin1.jar";
sha256 = "123…";
};
});
(plugins.buildEclipseUpdateSite {
name = "myplugin2-1.0";
src = fetchurl {
stripRoot = false;
url = "http://…/myplugin2.zip";
sha256 = "123…";
};
});
];
};
}
Elm
To start a development environment do nix-shell -p elmPackages.elm elmPackages.elm-format
To update Elm compiler, see
nixpkgs/pkgs/development/compilers/elm/README.md.
To package Elm applications,
read about
elm2nix.
Interactive shell helpers
Some packages provide the shell integration to be more useful. But unlike
other systems, nix doesn't have a standard share directory location. This is
why a bunch PACKAGE-share scripts are shipped that print
the location of the corresponding shared folder. Current list of such
packages is as following:
autojump: autojump-sharefzf: fzf-share
E.g. autojump can then used in the .bashrc like this:
source "$(autojump-share)/autojump.bash"
Weechat
Weechat can be configured to include your choice of plugins, reducing its
closure size from the default configuration which includes all available
plugins. To make use of this functionality, install an expression that
overrides its configuration such as
weechat.override {configure = {availablePlugins, ...}: {
plugins = with availablePlugins; [ python perl ];
}
}
If the configure function returns an attrset without the
plugins attribute, availablePlugins
will be used automatically.
The plugins currently available are python,
perl, ruby, guile,
tcl and lua.
The python and perl plugins allows the addition of extra libraries. For
instance, the inotify.py script in weechat-scripts
requires D-Bus or libnotify, and the fish.py script
requires pycrypto. To use these scripts, use the plugin's
withPackages attribute:
weechat.override { configure = {availablePlugins, ...}: {
plugins = with availablePlugins; [
(python.withPackages (ps: with ps; [ pycrypto python-dbus ]))
];
};
}
In order to also keep all default plugins installed, it is possible to use
the following method:
weechat.override { configure = { availablePlugins, ... }: {
plugins = builtins.attrValues (availablePlugins // {
python = availablePlugins.python.withPackages (ps: with ps; [ pycrypto python-dbus ]);
});
}; }
WeeChat allows to set defaults on startup using the
--run-command. The configure method
can be used to pass commands to the program:
weechat.override {
configure = { availablePlugins, ... }: {
init = ''
/set foo bar
/server add freenode chat.freenode.org
'';
};
}
Further values can be added to the list of commands when running
weechat --run-command "your-commands".
Additionally it's possible to specify scripts to be loaded when starting
weechat. These will be loaded before the commands from
init:
weechat.override {
configure = { availablePlugins, ... }: {
scripts = with pkgs.weechatScripts; [
weechat-xmpp weechat-matrix-bridge wee-slack
];
init = ''
/set plugins.var.python.jabber.key "val"
'':
};
}
In nixpkgs there's a subpackage which contains
derivations for WeeChat scripts. Such derivations expect a
passthru.scripts attribute which contains a list of all
scripts inside the store path. Furthermore all scripts have to live in
$out/share. An exemplary derivation looks like this:
{ stdenv, fetchurl }:
stdenv.mkDerivation {
name = "exemplary-weechat-script";
src = fetchurl {
url = "https://scripts.tld/your-scripts.tar.gz";
sha256 = "...";
};
passthru.scripts = [ "foo.py" "bar.lua" ];
installPhase = ''
mkdir $out/share
cp foo.py $out/share
cp bar.lua $out/share
'';
}ibus-engines.typing-booster
This package is an ibus-based completion method to speed up typing.
Activating the engine
IBus needs to be configured accordingly to activate
typing-booster. The configuration depends on the desktop
manager in use. For detailed instructions, please refer to the
upstream
docs.
On NixOS you need to explicitly enable ibus with given
engines before customizing your desktop to use
typing-booster. This can be achieved using the
ibus module:
{ pkgs, ... }: {
i18n.inputMethod = {
enabled = "ibus";
ibus.engines = with pkgs.ibus-engines; [ typing-booster ];
};
}Using custom hunspell dictionaries
The IBus engine is based on hunspell to support
completion in many languages. By default the dictionaries
de-de, en-us,
es-es, it-it,
sv-se and sv-fi are in use. To add
another dictionary, the package can be overridden like this:
ibus-engines.typing-booster.override {
langs = [ "de-at" "en-gb" ];
}Note: each language passed to langs must be an
attribute name in pkgs.hunspellDicts.Built-in emoji picker
The ibus-engines.typing-booster package contains a
program named emoji-picker. To display all emojis
correctly, a special font such as noto-fonts-emoji is
needed:
On NixOS it can be installed using the following expression:
{ pkgs, ... }: {
fonts.fonts = with pkgs; [ noto-fonts-emoji ];
}Nginx
Nginx is a
reverse proxy and lightweight webserver.
ETags on static files served from the Nix store
HTTP has a couple different mechanisms for caching to prevent
clients from having to download the same content repeatedly
if a resource has not changed since the last time it was requested.
When nginx is used as a server for static files, it implements
the caching mechanism based on the
Last-Modified
response header automatically; unfortunately, it works by using
filesystem timestamps to determine the value of the
Last-Modified header. This doesn't give the
desired behavior when the file is in the Nix store, because all
file timestamps are set to 0 (for reasons related to build
reproducibility).
Fortunately, HTTP supports an alternative (and more effective)
caching mechanism: the
ETag
response header. The value of the ETag header
specifies some identifier for the particular content that the
server is sending (e.g. a hash). When a client makes a second
request for the same resource, it sends that value back in an
If-None-Match header. If the ETag value is
unchanged, then the server does not need to resend the content.
As of NixOS 19.09, the nginx package in Nixpkgs is patched such
that when nginx serves a file out of /nix/store,
the hash in the store path is used as the ETag
header in the HTTP response, thus providing proper caching functionality.
This happens automatically; you do not need to do modify any
configuration to get this behavior.