2015-06-10 13:39:54 +02:00
|
|
|
|
@node Contributing
|
|
|
|
|
@chapter Contributing
|
|
|
|
|
|
|
|
|
|
This project is a cooperative effort, and we need your help to make it
|
|
|
|
|
grow! Please get in touch with us on @email{guix-devel@@gnu.org} and
|
|
|
|
|
@code{#guix} on the Freenode IRC network. We welcome ideas, bug
|
|
|
|
|
reports, patches, and anything that may be helpful to the project. We
|
|
|
|
|
particularly welcome help on packaging (@pxref{Packaging Guidelines}).
|
|
|
|
|
|
|
|
|
|
@menu
|
|
|
|
|
* Building from Git:: The latest and greatest.
|
|
|
|
|
* Running Guix Before It Is Installed:: Hacker tricks.
|
|
|
|
|
* The Perfect Setup:: The right tools.
|
|
|
|
|
* Coding Style:: Hygiene of the contributor.
|
|
|
|
|
* Submitting Patches:: Share your work.
|
|
|
|
|
@end menu
|
|
|
|
|
|
|
|
|
|
@node Building from Git
|
|
|
|
|
@section Building from Git
|
|
|
|
|
|
|
|
|
|
If you want to hack Guix itself, it is recommended to use the latest
|
|
|
|
|
version from the Git repository. When building Guix from a checkout,
|
|
|
|
|
the following packages are required in addition to those mentioned in
|
|
|
|
|
the installation instructions (@pxref{Requirements}).
|
|
|
|
|
|
|
|
|
|
@itemize
|
|
|
|
|
@item @url{http://gnu.org/software/autoconf/, GNU Autoconf};
|
|
|
|
|
@item @url{http://gnu.org/software/automake/, GNU Automake};
|
|
|
|
|
@item @url{http://gnu.org/software/gettext/, GNU Gettext};
|
2015-09-13 17:00:53 +02:00
|
|
|
|
@item @url{http://gnu.org/software/texinfo/, GNU Texinfo};
|
2015-06-10 13:39:54 +02:00
|
|
|
|
@item @url{http://www.graphviz.org/, Graphviz};
|
|
|
|
|
@item @url{http://www.gnu.org/software/help2man/, GNU Help2man (optional)}.
|
|
|
|
|
@end itemize
|
|
|
|
|
|
2015-11-22 11:24:18 +01:00
|
|
|
|
The easiest way to set up a development environment for Guix is, of
|
|
|
|
|
course, by using Guix! The following command starts a new shell where
|
|
|
|
|
all the dependencies and appropriate environment variables are set up to
|
|
|
|
|
hack on Guix:
|
2015-06-10 13:39:54 +02:00
|
|
|
|
|
2015-11-22 11:24:18 +01:00
|
|
|
|
@example
|
|
|
|
|
guix environment guix
|
|
|
|
|
@end example
|
|
|
|
|
|
|
|
|
|
@xref{Invoking guix environment}, for more information on that command.
|
|
|
|
|
Extra dependencies can be added with @option{--ad-hoc}:
|
|
|
|
|
|
|
|
|
|
@example
|
|
|
|
|
guix environment guix --ad-hoc help2man git strace
|
|
|
|
|
@end example
|
|
|
|
|
|
|
|
|
|
Run @command{./bootstrap} to generate the build system infrastructure
|
|
|
|
|
using Autoconf and Automake. If you get an error like this one:
|
2015-06-10 13:39:54 +02:00
|
|
|
|
|
|
|
|
|
@example
|
|
|
|
|
configure.ac:46: error: possibly undefined macro: PKG_CHECK_MODULES
|
|
|
|
|
@end example
|
|
|
|
|
|
2015-11-22 11:24:18 +01:00
|
|
|
|
@noindent
|
2015-06-10 13:39:54 +02:00
|
|
|
|
it probably means that Autoconf couldn’t find @file{pkg.m4}, which is
|
2015-11-22 11:24:18 +01:00
|
|
|
|
provided by pkg-config. Make sure that @file{pkg.m4} is available. The
|
|
|
|
|
same holds for the @file{guile.m4} set of macros provided by Guile. For
|
|
|
|
|
instance, if you installed Automake in @file{/usr/local}, it wouldn’t
|
|
|
|
|
look for @file{.m4} files in @file{/usr/share}. In that case, you have
|
|
|
|
|
to invoke the following command:
|
2015-06-10 13:39:54 +02:00
|
|
|
|
|
|
|
|
|
@example
|
|
|
|
|
export ACLOCAL_PATH=/usr/share/aclocal
|
|
|
|
|
@end example
|
|
|
|
|
|
2015-11-21 21:43:19 +01:00
|
|
|
|
@xref{Macro Search Path,,, automake, The GNU Automake Manual}, for
|
2015-06-10 13:39:54 +02:00
|
|
|
|
more information.
|
|
|
|
|
|
|
|
|
|
Then, run @command{./configure} as usual.
|
|
|
|
|
|
|
|
|
|
Finally, you have to invoke @code{make check} to run tests. If anything
|
|
|
|
|
fails, take a look at installation instructions (@pxref{Installation})
|
|
|
|
|
or send a message to the @email{guix-devel@@gnu.org, mailing list}.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@node Running Guix Before It Is Installed
|
|
|
|
|
@section Running Guix Before It Is Installed
|
|
|
|
|
|
|
|
|
|
In order to keep a sane working environment, you will find it useful to
|
|
|
|
|
test the changes made in your local source tree checkout without
|
|
|
|
|
actually installing them. So that you can distinguish between your
|
|
|
|
|
``end-user'' hat and your ``motley'' costume.
|
|
|
|
|
|
|
|
|
|
To that end, all the command-line tools can be used even if you have not
|
|
|
|
|
run @code{make install}. To do that, prefix each command with
|
|
|
|
|
@command{./pre-inst-env} (the @file{pre-inst-env} script lives in the
|
|
|
|
|
top build tree of Guix), as in:
|
|
|
|
|
|
|
|
|
|
@example
|
|
|
|
|
$ sudo ./pre-inst-env guix-daemon --build-users-group=guixbuild
|
|
|
|
|
$ ./pre-inst-env guix build hello
|
|
|
|
|
@end example
|
|
|
|
|
|
|
|
|
|
@noindent
|
|
|
|
|
Similarly, for a Guile session using the Guix modules:
|
|
|
|
|
|
|
|
|
|
@example
|
|
|
|
|
$ ./pre-inst-env guile -c '(use-modules (guix utils)) (pk (%current-system))'
|
2015-10-26 18:44:18 +01:00
|
|
|
|
|
|
|
|
|
;;; ("x86_64-linux")
|
|
|
|
|
@end example
|
|
|
|
|
|
|
|
|
|
@noindent
|
|
|
|
|
@cindex REPL
|
|
|
|
|
@cindex read-eval-print loop
|
|
|
|
|
@dots{} and for a REPL (@pxref{Using Guile Interactively,,, guile, Guile
|
|
|
|
|
Reference Manual}):
|
|
|
|
|
|
|
|
|
|
@example
|
|
|
|
|
$ ./pre-inst-env guile
|
|
|
|
|
scheme@@(guile-user)> ,use(guix)
|
|
|
|
|
scheme@@(guile-user)> ,use(gnu)
|
|
|
|
|
scheme@@(guile-user)> (define snakes
|
|
|
|
|
(fold-packages
|
|
|
|
|
(lambda (package lst)
|
|
|
|
|
(if (string-prefix? "python"
|
|
|
|
|
(package-name package))
|
|
|
|
|
(cons package lst)
|
|
|
|
|
lst))
|
|
|
|
|
'()))
|
|
|
|
|
scheme@@(guile-user)> (length snakes)
|
|
|
|
|
$1 = 361
|
2015-06-10 13:39:54 +02:00
|
|
|
|
@end example
|
|
|
|
|
|
|
|
|
|
The @command{pre-inst-env} script sets up all the environment variables
|
|
|
|
|
necessary to support this, including @env{PATH} and @env{GUILE_LOAD_PATH}.
|
|
|
|
|
|
2015-11-02 16:34:16 +01:00
|
|
|
|
Note that @command{./pre-inst-env guix pull} does @emph{not} upgrade the
|
|
|
|
|
local source tree; it simply updates the @file{~/.config/guix/latest}
|
|
|
|
|
symlink (@pxref{Invoking guix pull}). Run @command{git pull} instead if
|
|
|
|
|
you want to upgrade your local source tree.
|
|
|
|
|
|
2015-06-10 13:39:54 +02:00
|
|
|
|
|
|
|
|
|
@node The Perfect Setup
|
|
|
|
|
@section The Perfect Setup
|
|
|
|
|
|
|
|
|
|
The Perfect Setup to hack on Guix is basically the perfect setup used
|
|
|
|
|
for Guile hacking (@pxref{Using Guile in Emacs,,, guile, Guile Reference
|
|
|
|
|
Manual}). First, you need more than an editor, you need
|
|
|
|
|
@url{http://www.gnu.org/software/emacs, Emacs}, empowered by the
|
|
|
|
|
wonderful @url{http://nongnu.org/geiser/, Geiser}.
|
|
|
|
|
|
|
|
|
|
Geiser allows for interactive and incremental development from within
|
|
|
|
|
Emacs: code compilation and evaluation from within buffers, access to
|
|
|
|
|
on-line documentation (docstrings), context-sensitive completion,
|
|
|
|
|
@kbd{M-.} to jump to an object definition, a REPL to try out your code,
|
|
|
|
|
and more (@pxref{Introduction,,, geiser, Geiser User Manual}). For
|
|
|
|
|
convenient Guix development, make sure to augment Guile’s load path so
|
|
|
|
|
that it finds source files from your checkout:
|
|
|
|
|
|
|
|
|
|
@lisp
|
|
|
|
|
;; @r{Assuming the Guix checkout is in ~/src/guix.}
|
2015-11-14 19:13:07 +01:00
|
|
|
|
(with-eval-after-load 'geiser-guile
|
|
|
|
|
(add-to-list 'geiser-guile-load-path "~/src/guix"))
|
2015-06-10 13:39:54 +02:00
|
|
|
|
@end lisp
|
|
|
|
|
|
|
|
|
|
To actually edit the code, Emacs already has a neat Scheme mode. But in
|
|
|
|
|
addition to that, you must not miss
|
|
|
|
|
@url{http://www.emacswiki.org/emacs/ParEdit, Paredit}. It provides
|
|
|
|
|
facilities to directly operate on the syntax tree, such as raising an
|
|
|
|
|
s-expression or wrapping it, swallowing or rejecting the following
|
|
|
|
|
s-expression, etc.
|
|
|
|
|
|
2015-07-24 17:33:14 +02:00
|
|
|
|
GNU Guix also comes with a minor mode that provides some additional
|
|
|
|
|
functionality for Scheme buffers (@pxref{Emacs Development}).
|
|
|
|
|
|
2015-06-10 13:39:54 +02:00
|
|
|
|
|
|
|
|
|
@node Coding Style
|
|
|
|
|
@section Coding Style
|
|
|
|
|
|
|
|
|
|
In general our code follows the GNU Coding Standards (@pxref{Top,,,
|
|
|
|
|
standards, GNU Coding Standards}). However, they do not say much about
|
|
|
|
|
Scheme, so here are some additional rules.
|
|
|
|
|
|
|
|
|
|
@menu
|
|
|
|
|
* Programming Paradigm:: How to compose your elements.
|
|
|
|
|
* Modules:: Where to store your code?
|
|
|
|
|
* Data Types and Pattern Matching:: Implementing data structures.
|
|
|
|
|
* Formatting Code:: Writing conventions.
|
|
|
|
|
@end menu
|
|
|
|
|
|
|
|
|
|
@node Programming Paradigm
|
|
|
|
|
@subsection Programming Paradigm
|
|
|
|
|
|
|
|
|
|
Scheme code in Guix is written in a purely functional style. One
|
|
|
|
|
exception is code that involves input/output, and procedures that
|
|
|
|
|
implement low-level concepts, such as the @code{memoize} procedure.
|
|
|
|
|
|
|
|
|
|
@node Modules
|
|
|
|
|
@subsection Modules
|
|
|
|
|
|
|
|
|
|
Guile modules that are meant to be used on the builder side must live in
|
|
|
|
|
the @code{(guix build @dots{})} name space. They must not refer to
|
|
|
|
|
other Guix or GNU modules. However, it is OK for a ``host-side'' module
|
|
|
|
|
to use a build-side module.
|
|
|
|
|
|
|
|
|
|
Modules that deal with the broader GNU system should be in the
|
|
|
|
|
@code{(gnu @dots{})} name space rather than @code{(guix @dots{})}.
|
|
|
|
|
|
|
|
|
|
@node Data Types and Pattern Matching
|
|
|
|
|
@subsection Data Types and Pattern Matching
|
|
|
|
|
|
|
|
|
|
The tendency in classical Lisp is to use lists to represent everything,
|
|
|
|
|
and then to browse them ``by hand'' using @code{car}, @code{cdr},
|
|
|
|
|
@code{cadr}, and co. There are several problems with that style,
|
|
|
|
|
notably the fact that it is hard to read, error-prone, and a hindrance
|
|
|
|
|
to proper type error reports.
|
|
|
|
|
|
|
|
|
|
Guix code should define appropriate data types (for instance, using
|
|
|
|
|
@code{define-record-type*}) rather than abuse lists. In addition, it
|
|
|
|
|
should use pattern matching, via Guile’s @code{(ice-9 match)} module,
|
|
|
|
|
especially when matching lists.
|
|
|
|
|
|
|
|
|
|
@node Formatting Code
|
|
|
|
|
@subsection Formatting Code
|
|
|
|
|
|
|
|
|
|
When writing Scheme code, we follow common wisdom among Scheme
|
|
|
|
|
programmers. In general, we follow the
|
|
|
|
|
@url{http://mumble.net/~campbell/scheme/style.txt, Riastradh's Lisp
|
|
|
|
|
Style Rules}. This document happens to describe the conventions mostly
|
|
|
|
|
used in Guile’s code too. It is very thoughtful and well written, so
|
|
|
|
|
please do read it.
|
|
|
|
|
|
|
|
|
|
Some special forms introduced in Guix, such as the @code{substitute*}
|
|
|
|
|
macro, have special indentation rules. These are defined in the
|
|
|
|
|
@file{.dir-locals.el} file, which Emacs automatically uses. If you do
|
|
|
|
|
not use Emacs, please make sure to let your editor know the rules.
|
|
|
|
|
|
|
|
|
|
We require all top-level procedures to carry a docstring. This
|
|
|
|
|
requirement can be relaxed for simple private procedures in the
|
|
|
|
|
@code{(guix build @dots{})} name space, though.
|
|
|
|
|
|
|
|
|
|
Procedures should not have more than four positional parameters. Use
|
|
|
|
|
keyword parameters for procedures that take more than four parameters.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@node Submitting Patches
|
|
|
|
|
@section Submitting Patches
|
|
|
|
|
|
|
|
|
|
Development is done using the Git distributed version control system.
|
|
|
|
|
Thus, access to the repository is not strictly necessary. We welcome
|
|
|
|
|
contributions in the form of patches as produced by @code{git
|
|
|
|
|
format-patch} sent to the @email{guix-devel@@gnu.org, mailing list}.
|
|
|
|
|
Please write commit logs in the ChangeLog format (@pxref{Change Logs,,,
|
|
|
|
|
standards, GNU Coding Standards}); you can check the commit history for
|
|
|
|
|
examples.
|
|
|
|
|
|
|
|
|
|
Before submitting a patch that adds or modifies a package definition,
|
2015-06-18 00:22:13 +02:00
|
|
|
|
please run through this check list:
|
|
|
|
|
|
|
|
|
|
@enumerate
|
2015-09-15 22:37:12 +02:00
|
|
|
|
@item
|
|
|
|
|
Take some time to provide an adequate synopsis and description for the
|
|
|
|
|
package. @xref{Synopses and Descriptions}, for some guidelines.
|
|
|
|
|
|
2015-06-18 00:22:13 +02:00
|
|
|
|
@item
|
|
|
|
|
Run @code{guix lint @var{package}}, where @var{package} is the
|
2015-06-10 13:39:54 +02:00
|
|
|
|
name of the new or modified package, and fix any errors it reports
|
2015-06-18 00:22:13 +02:00
|
|
|
|
(@pxref{Invoking guix lint}).
|
|
|
|
|
|
|
|
|
|
@item
|
|
|
|
|
Make sure the package builds on your platform, using @code{guix build
|
|
|
|
|
@var{package}}.
|
|
|
|
|
|
|
|
|
|
@item
|
|
|
|
|
Take a look at the profile reported by @command{guix size}
|
|
|
|
|
(@pxref{Invoking guix size}). This will allow you to notice references
|
|
|
|
|
to other packages unwillingly retained. It may also help determine
|
|
|
|
|
whether to split the package (@pxref{Packages with Multiple Outputs}),
|
|
|
|
|
and which optional dependencies should be used.
|
|
|
|
|
|
|
|
|
|
@item
|
|
|
|
|
For important changes, check that dependent package (if applicable) are
|
|
|
|
|
not affected by the change; @code{guix refresh --list-dependent
|
2015-06-10 13:39:54 +02:00
|
|
|
|
@var{package}} will help you do that (@pxref{Invoking guix refresh}).
|
|
|
|
|
|
2015-10-20 00:55:09 +02:00
|
|
|
|
@item
|
2015-12-08 23:27:53 +01:00
|
|
|
|
@cindex determinism, of build processes
|
|
|
|
|
@cindex reproducible builds, checking
|
2015-10-20 00:55:09 +02:00
|
|
|
|
Check whether the package's build process is deterministic. This
|
|
|
|
|
typically means checking whether an independent build of the package
|
|
|
|
|
yields the exact same result that you obtained, bit for bit.
|
|
|
|
|
|
2015-12-08 23:27:53 +01:00
|
|
|
|
A simple way to do that is by building the same package several times in
|
|
|
|
|
a row on your machine (@pxref{Invoking guix build}):
|
|
|
|
|
|
|
|
|
|
@example
|
|
|
|
|
guix build --rounds=2 my-package
|
|
|
|
|
@end example
|
|
|
|
|
|
|
|
|
|
This is enough to catch a class of common non-determinism issues, such
|
|
|
|
|
as timestamps or randomly-generated output in the build result.
|
|
|
|
|
|
|
|
|
|
Another option is to use @command{guix challenge} (@pxref{Invoking guix
|
|
|
|
|
challenge}). You may run it once the package has been committed and
|
|
|
|
|
built by @code{hydra.gnu.org} to check whether it obtains the same
|
|
|
|
|
result as you did. Better yet: Find another machine that can build it
|
|
|
|
|
and run @command{guix publish}. Since the remote build machine is
|
|
|
|
|
likely different from yours, this can catch non-determinism issues
|
|
|
|
|
related to the hardware---e.g., use of different instruction set
|
|
|
|
|
extensions---or to the operating system kernel---e.g., reliance on
|
|
|
|
|
@code{uname} or @file{/proc} files.
|
2015-10-20 00:55:09 +02:00
|
|
|
|
|
2015-06-18 00:22:13 +02:00
|
|
|
|
@end enumerate
|
|
|
|
|
|
2015-06-10 13:39:54 +02:00
|
|
|
|
When posting a patch to the mailing list, use @samp{[PATCH] @dots{}} as a
|
|
|
|
|
subject. You may use your email client or the @command{git send-mail}
|
|
|
|
|
command.
|