Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
;;; GNU Guix --- Functional package management for GNU
|
2021-01-10 05:23:40 -05:00
|
|
|
|
;;; Copyright © 2013, 2014, 2015, 2016, 2017, 2018, 2019, 2020, 2021 Ludovic Courtès <ludo@gnu.org>
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
;;; Copyright © 2013 Nikita Karetnikov <nikita@karetnikov.org>
|
2016-06-30 15:01:06 -04:00
|
|
|
|
;;; Copyright © 2014, 2016 Alex Kost <alezost@gmail.com>
|
2015-03-03 02:09:30 -05:00
|
|
|
|
;;; Copyright © 2015 Mark H Weaver <mhw@netris.org>
|
2015-05-27 08:58:27 -04:00
|
|
|
|
;;; Copyright © 2015 Sou Bunnbu <iyzsong@gmail.com>
|
2019-01-15 07:03:48 -05:00
|
|
|
|
;;; Copyright © 2016, 2018, 2019 Ricardo Wurmus <rekado@elephly.net>
|
2016-11-02 01:48:11 -04:00
|
|
|
|
;;; Copyright © 2016 Chris Marusich <cmmarusich@gmail.com>
|
2017-03-12 07:53:59 -04:00
|
|
|
|
;;; Copyright © 2017 Huang Ying <huang.ying.caritas@gmail.com>
|
2017-04-05 04:09:22 -04:00
|
|
|
|
;;; Copyright © 2017 Maxim Cournoyer <maxim.cournoyer@gmail.com>
|
2019-06-29 01:50:15 -04:00
|
|
|
|
;;; Copyright © 2019 Kyle Meyer <kyle@kyleam.com>
|
2019-12-24 09:04:57 -05:00
|
|
|
|
;;; Copyright © 2019 Mathieu Othacehe <m.othacehe@gmail.com>
|
2020-02-18 04:42:07 -05:00
|
|
|
|
;;; Copyright © 2020 Danny Milosavljevic <dannym@scratchpost.org>
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
;;;
|
|
|
|
|
;;; This file is part of GNU Guix.
|
|
|
|
|
;;;
|
|
|
|
|
;;; GNU Guix is free software; you can redistribute it and/or modify it
|
|
|
|
|
;;; under the terms of the GNU General Public License as published by
|
|
|
|
|
;;; the Free Software Foundation; either version 3 of the License, or (at
|
|
|
|
|
;;; your option) any later version.
|
|
|
|
|
;;;
|
|
|
|
|
;;; GNU Guix is distributed in the hope that it will be useful, but
|
|
|
|
|
;;; WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
;;; GNU General Public License for more details.
|
|
|
|
|
;;;
|
|
|
|
|
;;; You should have received a copy of the GNU General Public License
|
|
|
|
|
;;; along with GNU Guix. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
|
|
(define-module (guix profiles)
|
2018-05-13 10:08:24 -04:00
|
|
|
|
#:use-module ((guix config) #:select (%state-directory))
|
2016-03-31 09:47:18 -04:00
|
|
|
|
#:use-module ((guix utils) #:hide (package-name->name+version))
|
|
|
|
|
#:use-module ((guix build utils)
|
2018-10-11 12:04:51 -04:00
|
|
|
|
#:select (package-name->name+version mkdir-p))
|
2020-07-24 17:28:11 -04:00
|
|
|
|
#:use-module ((guix diagnostics) #:select (&fix-hint))
|
2018-10-11 12:04:51 -04:00
|
|
|
|
#:use-module (guix i18n)
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
#:use-module (guix records)
|
|
|
|
|
#:use-module (guix packages)
|
Move search path specifications to (guix search-paths).
* guix/packages.scm (<search-path-specification>,
search-path-specification->sexp, sexp->search-path-specification):
Move to...
* guix/search-paths.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
* guix/build-system/cmake.scm, guix/build-system/glib-or-gtk.scm,
guix/build-system/gnu.scm, guix/build-system/haskell.scm,
guix/build-system/perl.scm, guix/build-system/python.scm,
guix/build-system/ruby.scm, guix/build-system/waf.scm,
guix/profiles.scm, guix/scripts/package.scm: Use it.
2015-05-04 16:11:37 -04:00
|
|
|
|
#:use-module (guix derivations)
|
|
|
|
|
#:use-module (guix search-paths)
|
2014-07-26 16:08:10 -04:00
|
|
|
|
#:use-module (guix gexp)
|
2017-12-15 16:16:18 -05:00
|
|
|
|
#:use-module (guix modules)
|
2014-08-23 12:41:14 -04:00
|
|
|
|
#:use-module (guix monads)
|
monads: Move '%store-monad' and related procedures where they belong.
This turns (guix monads) into a generic module for monads, and moves the
store monad and related monadic procedures in their corresponding
module.
* guix/monads.scm (store-return, store-bind, %store-monad, store-lift,
text-file, interned-file, package-file, package->derivation,
package->cross-derivation, origin->derivation, imported-modules,
compiled, modules, built-derivations, run-with-store): Move to...
* guix/store.scm (store-return, store-bind, %store-monad, store-lift,
text-file, interned-file): ... here.
(%guile-for-build): New variable.
(run-with-store): Moved from monads.scm. Remove default value for
#:guile-for-build.
* guix/packages.scm (default-guile): Export.
(set-guile-for-build): New procedure.
(package-file, package->derivation, package->cross-derivation,
origin->derivation): Moved from monads.scm.
* guix/derivations.scm (%guile-for-build): Remove.
(imported-modules): Rename to...
(%imported-modules): ... this.
(compiled-modules): Rename to...
(%compiled-modules): ... this.
(built-derivations, imported-modules, compiled-modules): New
procedures.
* gnu/services/avahi.scm, gnu/services/base.scm, gnu/services/dbus.scm,
gnu/services/dmd.scm, gnu/services/networking.scm,
gnu/services/ssh.scm, gnu/services/xorg.scm, gnu/system/install.scm,
gnu/system/linux-initrd.scm, gnu/system/shadow.scm, guix/download.scm,
guix/gexp.scm, guix/git-download.scm, guix/profiles.scm,
guix/svn-download.scm, tests/monads.scm: Adjust imports accordingly.
* guix/monad-repl.scm (default-guile-derivation): New procedure.
(store-monad-language, run-in-store): Use it.
* build-aux/hydra/gnu-system.scm (qemu-jobs): Add explicit
'set-guile-for-build' call.
* guix/scripts/archive.scm (derivation-from-expression): Likewise.
* guix/scripts/build.scm (options/resolve-packages): Likewise.
* guix/scripts/environment.scm (guix-environment): Likewise.
* guix/scripts/system.scm (guix-system): Likewise.
* doc/guix.texi (The Store Monad): Adjust module names accordingly.
2015-01-14 07:34:52 -05:00
|
|
|
|
#:use-module (guix store)
|
2017-06-07 03:51:55 -04:00
|
|
|
|
#:use-module (ice-9 vlist)
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
#:use-module (ice-9 match)
|
|
|
|
|
#:use-module (ice-9 regex)
|
|
|
|
|
#:use-module (ice-9 ftw)
|
2014-08-13 16:03:53 -04:00
|
|
|
|
#:use-module (ice-9 format)
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
#:use-module (srfi srfi-1)
|
|
|
|
|
#:use-module (srfi srfi-9)
|
2014-08-30 15:52:32 -04:00
|
|
|
|
#:use-module (srfi srfi-11)
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
#:use-module (srfi srfi-19)
|
|
|
|
|
#:use-module (srfi srfi-26)
|
2014-10-08 09:29:01 -04:00
|
|
|
|
#:use-module (srfi srfi-34)
|
|
|
|
|
#:use-module (srfi srfi-35)
|
|
|
|
|
#:export (&profile-error
|
|
|
|
|
profile-error?
|
|
|
|
|
profile-error-profile
|
|
|
|
|
&profile-not-found-error
|
|
|
|
|
profile-not-found-error?
|
2018-10-26 14:04:49 -04:00
|
|
|
|
&profile-collision-error
|
2017-06-07 03:51:55 -04:00
|
|
|
|
profile-collision-error?
|
|
|
|
|
profile-collision-error-entry
|
|
|
|
|
profile-collision-error-conflict
|
2014-10-08 09:29:01 -04:00
|
|
|
|
&missing-generation-error
|
|
|
|
|
missing-generation-error?
|
|
|
|
|
missing-generation-error-generation
|
2019-02-07 08:54:43 -05:00
|
|
|
|
&unmatched-pattern-error
|
|
|
|
|
unmatched-pattern-error?
|
|
|
|
|
unmatched-pattern-error-pattern
|
|
|
|
|
unmatched-pattern-error-manifest
|
2014-10-08 09:29:01 -04:00
|
|
|
|
|
|
|
|
|
manifest make-manifest
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
manifest?
|
|
|
|
|
manifest-entries
|
2017-06-07 03:51:55 -04:00
|
|
|
|
manifest-transitive-entries
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
|
|
|
|
|
<manifest-entry> ; FIXME: eventually make it internal
|
|
|
|
|
manifest-entry
|
|
|
|
|
manifest-entry?
|
|
|
|
|
manifest-entry-name
|
|
|
|
|
manifest-entry-version
|
|
|
|
|
manifest-entry-output
|
2014-07-26 16:08:10 -04:00
|
|
|
|
manifest-entry-item
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
manifest-entry-dependencies
|
2015-05-02 17:55:24 -04:00
|
|
|
|
manifest-entry-search-paths
|
2017-06-06 09:29:50 -04:00
|
|
|
|
manifest-entry-parent
|
2018-05-13 12:48:22 -04:00
|
|
|
|
manifest-entry-properties
|
2020-03-30 15:48:51 -04:00
|
|
|
|
lower-manifest-entry
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
|
2020-03-30 17:34:48 -04:00
|
|
|
|
manifest-entry=?
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
|
2013-11-01 18:11:17 -04:00
|
|
|
|
manifest-pattern
|
|
|
|
|
manifest-pattern?
|
2016-09-06 17:05:26 -04:00
|
|
|
|
manifest-pattern-name
|
|
|
|
|
manifest-pattern-version
|
|
|
|
|
manifest-pattern-output
|
2013-11-01 18:11:17 -04:00
|
|
|
|
|
2019-11-20 06:07:02 -05:00
|
|
|
|
concatenate-manifests
|
2019-12-29 10:19:56 -05:00
|
|
|
|
map-manifest-entries
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
manifest-remove
|
2014-08-12 04:32:16 -04:00
|
|
|
|
manifest-add
|
2014-09-02 15:12:59 -04:00
|
|
|
|
manifest-lookup
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
manifest-installed?
|
2013-11-01 18:11:17 -04:00
|
|
|
|
manifest-matching-entries
|
2018-07-09 04:44:36 -04:00
|
|
|
|
manifest-search-paths
|
2020-06-14 09:06:53 -04:00
|
|
|
|
check-for-collisions
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
|
2021-01-10 05:23:40 -05:00
|
|
|
|
manifest->code
|
|
|
|
|
|
2014-08-13 16:03:53 -04:00
|
|
|
|
manifest-transaction
|
|
|
|
|
manifest-transaction?
|
|
|
|
|
manifest-transaction-install
|
|
|
|
|
manifest-transaction-remove
|
2016-09-06 14:19:21 -04:00
|
|
|
|
manifest-transaction-install-entry
|
|
|
|
|
manifest-transaction-remove-pattern
|
|
|
|
|
manifest-transaction-null?
|
2017-06-26 16:23:11 -04:00
|
|
|
|
manifest-transaction-removal-candidate?
|
2014-08-13 16:03:53 -04:00
|
|
|
|
manifest-perform-transaction
|
2014-08-30 15:52:32 -04:00
|
|
|
|
manifest-transaction-effects
|
2014-08-13 16:03:53 -04:00
|
|
|
|
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
profile-manifest
|
2014-07-26 16:54:40 -04:00
|
|
|
|
package->manifest-entry
|
2015-05-18 07:51:56 -04:00
|
|
|
|
packages->manifest
|
2017-01-15 04:25:52 -05:00
|
|
|
|
ca-certificate-bundle
|
2015-04-15 16:44:51 -04:00
|
|
|
|
%default-profile-hooks
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
profile-derivation
|
2018-07-09 07:22:29 -04:00
|
|
|
|
profile-search-paths
|
2015-10-26 18:01:06 -04:00
|
|
|
|
|
2020-04-22 09:43:43 -04:00
|
|
|
|
profile
|
|
|
|
|
profile?
|
|
|
|
|
profile-name
|
|
|
|
|
profile-content
|
|
|
|
|
profile-hooks
|
|
|
|
|
profile-locales?
|
|
|
|
|
profile-allow-collisions?
|
|
|
|
|
profile-relative-symlinks?
|
|
|
|
|
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
generation-number
|
2019-04-06 17:05:27 -04:00
|
|
|
|
generation-profile
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
generation-numbers
|
2014-09-21 06:24:09 -04:00
|
|
|
|
profile-generations
|
2016-11-02 01:48:11 -04:00
|
|
|
|
relative-generation-spec->number
|
2014-10-10 09:56:59 -04:00
|
|
|
|
relative-generation
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
previous-generation-number
|
|
|
|
|
generation-time
|
2015-10-26 18:01:06 -04:00
|
|
|
|
generation-file-name
|
|
|
|
|
switch-to-generation
|
|
|
|
|
roll-back
|
2018-05-13 10:08:24 -04:00
|
|
|
|
delete-generation
|
|
|
|
|
|
|
|
|
|
%user-profile-directory
|
|
|
|
|
%profile-directory
|
|
|
|
|
%current-profile
|
2018-10-11 12:04:51 -04:00
|
|
|
|
ensure-profile-directory
|
2018-05-13 10:08:24 -04:00
|
|
|
|
canonicalize-profile
|
2020-02-18 04:42:07 -05:00
|
|
|
|
user-friendly-profile
|
|
|
|
|
|
|
|
|
|
linux-module-database))
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
|
|
|
|
|
;;; Commentary:
|
|
|
|
|
;;;
|
|
|
|
|
;;; Tools to create and manipulate profiles---i.e., the representation of a
|
|
|
|
|
;;; set of installed packages.
|
|
|
|
|
;;;
|
|
|
|
|
;;; Code:
|
|
|
|
|
|
2014-10-08 09:29:01 -04:00
|
|
|
|
|
|
|
|
|
;;;
|
|
|
|
|
;;; Condition types.
|
|
|
|
|
;;;
|
|
|
|
|
|
|
|
|
|
(define-condition-type &profile-error &error
|
|
|
|
|
profile-error?
|
|
|
|
|
(profile profile-error-profile))
|
|
|
|
|
|
|
|
|
|
(define-condition-type &profile-not-found-error &profile-error
|
|
|
|
|
profile-not-found-error?)
|
|
|
|
|
|
2017-06-07 03:51:55 -04:00
|
|
|
|
(define-condition-type &profile-collision-error &error
|
|
|
|
|
profile-collision-error?
|
|
|
|
|
(entry profile-collision-error-entry) ;<manifest-entry>
|
|
|
|
|
(conflict profile-collision-error-conflict)) ;<manifest-entry>
|
|
|
|
|
|
2019-02-07 08:54:43 -05:00
|
|
|
|
(define-condition-type &unmatched-pattern-error &error
|
|
|
|
|
unmatched-pattern-error?
|
|
|
|
|
(pattern unmatched-pattern-error-pattern) ;<manifest-pattern>
|
|
|
|
|
(manifest unmatched-pattern-error-manifest)) ;<manifest>
|
|
|
|
|
|
2014-10-08 09:29:01 -04:00
|
|
|
|
(define-condition-type &missing-generation-error &profile-error
|
|
|
|
|
missing-generation-error?
|
|
|
|
|
(generation missing-generation-error-generation))
|
|
|
|
|
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
|
|
|
|
|
;;;
|
|
|
|
|
;;; Manifests.
|
|
|
|
|
;;;
|
|
|
|
|
|
|
|
|
|
(define-record-type <manifest>
|
|
|
|
|
(manifest entries)
|
|
|
|
|
manifest?
|
|
|
|
|
(entries manifest-entries)) ; list of <manifest-entry>
|
|
|
|
|
|
|
|
|
|
;; Convenient alias, to avoid name clashes.
|
|
|
|
|
(define make-manifest manifest)
|
|
|
|
|
|
|
|
|
|
(define-record-type* <manifest-entry> manifest-entry
|
|
|
|
|
make-manifest-entry
|
|
|
|
|
manifest-entry?
|
|
|
|
|
(name manifest-entry-name) ; string
|
|
|
|
|
(version manifest-entry-version) ; string
|
|
|
|
|
(output manifest-entry-output ; string
|
|
|
|
|
(default "out"))
|
2018-05-02 05:01:56 -04:00
|
|
|
|
(item manifest-entry-item) ; package | file-like | store path
|
2017-06-06 08:01:12 -04:00
|
|
|
|
(dependencies manifest-entry-dependencies ; <manifest-entry>*
|
2015-05-02 17:55:24 -04:00
|
|
|
|
(default '()))
|
|
|
|
|
(search-paths manifest-entry-search-paths ; search-path-specification*
|
2017-06-06 09:29:50 -04:00
|
|
|
|
(default '()))
|
|
|
|
|
(parent manifest-entry-parent ; promise (#f | <manifest-entry>)
|
2018-05-13 12:48:22 -04:00
|
|
|
|
(default (delay #f)))
|
|
|
|
|
(properties manifest-entry-properties ; list of symbol/value pairs
|
|
|
|
|
(default '())))
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
|
2013-11-01 18:11:17 -04:00
|
|
|
|
(define-record-type* <manifest-pattern> manifest-pattern
|
|
|
|
|
make-manifest-pattern
|
|
|
|
|
manifest-pattern?
|
|
|
|
|
(name manifest-pattern-name) ; string
|
|
|
|
|
(version manifest-pattern-version ; string | #f
|
|
|
|
|
(default #f))
|
|
|
|
|
(output manifest-pattern-output ; string | #f
|
|
|
|
|
(default "out")))
|
|
|
|
|
|
2020-03-30 17:34:48 -04:00
|
|
|
|
(define (list=? = lst1 lst2)
|
|
|
|
|
"Return true if LST1 and LST2 have the same length and their elements are
|
|
|
|
|
pairwise equal per =."
|
|
|
|
|
(match lst1
|
|
|
|
|
(()
|
|
|
|
|
(null? lst2))
|
|
|
|
|
((head1 . tail1)
|
|
|
|
|
(match lst2
|
|
|
|
|
((head2 . tail2)
|
|
|
|
|
(and (= head1 head2) (list=? = tail1 tail2)))
|
|
|
|
|
(()
|
|
|
|
|
#f)))))
|
|
|
|
|
|
|
|
|
|
(define (manifest-entry=? entry1 entry2)
|
|
|
|
|
"Return true if ENTRY1 is equivalent to ENTRY2, ignoring their 'properties'
|
|
|
|
|
field."
|
|
|
|
|
(match entry1
|
|
|
|
|
(($ <manifest-entry> name1 version1 output1 item1 dependencies1 paths1)
|
|
|
|
|
(match entry2
|
|
|
|
|
(($ <manifest-entry> name2 version2 output2 item2 dependencies2 paths2)
|
|
|
|
|
(and (string=? name1 name2)
|
|
|
|
|
(string=? version1 version2)
|
|
|
|
|
(string=? output1 output2)
|
|
|
|
|
(equal? item1 item2) ;XXX: could be <package> vs. store item
|
|
|
|
|
(equal? paths1 paths2)
|
|
|
|
|
(list=? manifest-entry=? dependencies1 dependencies2)))))))
|
|
|
|
|
|
2017-06-21 05:58:39 -04:00
|
|
|
|
(define (manifest-transitive-entries manifest)
|
|
|
|
|
"Return the entries of MANIFEST along with their propagated inputs,
|
|
|
|
|
recursively."
|
|
|
|
|
(let loop ((entries (manifest-entries manifest))
|
|
|
|
|
(result '())
|
2020-06-14 08:51:02 -04:00
|
|
|
|
(visited vlist-null)) ;compare with 'manifest-entry=?'
|
2017-06-21 05:58:39 -04:00
|
|
|
|
(match entries
|
|
|
|
|
(()
|
|
|
|
|
(reverse result))
|
|
|
|
|
((head . tail)
|
2020-06-14 08:51:02 -04:00
|
|
|
|
(if (vhash-assoc head visited manifest-entry=?)
|
2017-06-21 05:58:39 -04:00
|
|
|
|
(loop tail result visited)
|
|
|
|
|
(loop (append (manifest-entry-dependencies head)
|
|
|
|
|
tail)
|
|
|
|
|
(cons head result)
|
2020-06-14 08:51:02 -04:00
|
|
|
|
(vhash-cons head #t visited)))))))
|
2017-06-21 05:58:39 -04:00
|
|
|
|
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
(define (profile-manifest profile)
|
|
|
|
|
"Return the PROFILE's manifest."
|
|
|
|
|
(let ((file (string-append profile "/manifest")))
|
|
|
|
|
(if (file-exists? file)
|
|
|
|
|
(call-with-input-file file read-manifest)
|
|
|
|
|
(manifest '()))))
|
|
|
|
|
|
2017-06-07 03:51:55 -04:00
|
|
|
|
(define (manifest-entry-lookup manifest)
|
|
|
|
|
"Return a lookup procedure for the entries of MANIFEST. The lookup
|
|
|
|
|
procedure takes two arguments: the entry name and output."
|
|
|
|
|
(define mapping
|
|
|
|
|
(let loop ((entries (manifest-entries manifest))
|
|
|
|
|
(mapping vlist-null))
|
|
|
|
|
(fold (lambda (entry result)
|
|
|
|
|
(vhash-cons (cons (manifest-entry-name entry)
|
|
|
|
|
(manifest-entry-output entry))
|
|
|
|
|
entry
|
|
|
|
|
(loop (manifest-entry-dependencies entry)
|
|
|
|
|
result)))
|
|
|
|
|
mapping
|
|
|
|
|
entries)))
|
|
|
|
|
|
|
|
|
|
(lambda (name output)
|
|
|
|
|
(match (vhash-assoc (cons name output) mapping)
|
|
|
|
|
((_ . entry) entry)
|
|
|
|
|
(#f #f))))
|
|
|
|
|
|
|
|
|
|
(define* (lower-manifest-entry entry system #:key target)
|
|
|
|
|
"Lower ENTRY for SYSTEM and TARGET such that its 'item' field is a store
|
|
|
|
|
file name."
|
2020-03-30 16:39:54 -04:00
|
|
|
|
(define (recurse entry)
|
|
|
|
|
(mapm/accumulate-builds (lambda (entry)
|
|
|
|
|
(lower-manifest-entry entry system
|
|
|
|
|
#:target target))
|
|
|
|
|
(manifest-entry-dependencies entry)))
|
|
|
|
|
|
2017-06-07 03:51:55 -04:00
|
|
|
|
(let ((item (manifest-entry-item entry)))
|
|
|
|
|
(if (string? item)
|
|
|
|
|
(with-monad %store-monad
|
|
|
|
|
(return entry))
|
|
|
|
|
(mlet %store-monad ((drv (lower-object item system
|
|
|
|
|
#:target target))
|
2020-03-30 16:39:54 -04:00
|
|
|
|
(dependencies (recurse entry))
|
2017-06-07 03:51:55 -04:00
|
|
|
|
(output -> (manifest-entry-output entry)))
|
|
|
|
|
(return (manifest-entry
|
|
|
|
|
(inherit entry)
|
2020-03-30 16:39:54 -04:00
|
|
|
|
(item (derivation->output-path drv output))
|
|
|
|
|
(dependencies dependencies)))))))
|
2017-06-07 03:51:55 -04:00
|
|
|
|
|
|
|
|
|
(define* (check-for-collisions manifest system #:key target)
|
|
|
|
|
"Check whether the entries of MANIFEST conflict with one another; raise a
|
|
|
|
|
'&profile-collision-error' when a conflict is encountered."
|
|
|
|
|
(define lookup
|
|
|
|
|
(manifest-entry-lookup manifest))
|
|
|
|
|
|
2020-03-25 07:45:12 -04:00
|
|
|
|
(define candidates
|
|
|
|
|
(filter-map (lambda (entry)
|
|
|
|
|
(let ((other (lookup (manifest-entry-name entry)
|
|
|
|
|
(manifest-entry-output entry))))
|
|
|
|
|
(and other (list entry other))))
|
|
|
|
|
(manifest-transitive-entries manifest)))
|
|
|
|
|
|
|
|
|
|
(define lower-pair
|
|
|
|
|
(match-lambda
|
|
|
|
|
((first second)
|
|
|
|
|
(mlet %store-monad ((first (lower-manifest-entry first system
|
|
|
|
|
#:target target))
|
|
|
|
|
(second (lower-manifest-entry second system
|
|
|
|
|
#:target target)))
|
|
|
|
|
(return (list first second))))))
|
|
|
|
|
|
|
|
|
|
;; Start by lowering CANDIDATES "in parallel".
|
|
|
|
|
(mlet %store-monad ((lst (mapm/accumulate-builds lower-pair candidates)))
|
2017-06-07 03:51:55 -04:00
|
|
|
|
(foldm %store-monad
|
2020-03-25 07:45:12 -04:00
|
|
|
|
(lambda (entries result)
|
|
|
|
|
(match entries
|
|
|
|
|
((first second)
|
|
|
|
|
(if (string=? (manifest-entry-item first)
|
|
|
|
|
(manifest-entry-item second))
|
|
|
|
|
(return result)
|
|
|
|
|
(raise (condition
|
|
|
|
|
(&profile-collision-error
|
|
|
|
|
(entry first)
|
|
|
|
|
(conflict second))))))))
|
2017-06-07 03:51:55 -04:00
|
|
|
|
#t
|
2020-03-25 07:45:12 -04:00
|
|
|
|
lst)))
|
2017-06-07 03:51:55 -04:00
|
|
|
|
|
2017-06-06 09:29:50 -04:00
|
|
|
|
(define* (package->manifest-entry package #:optional (output "out")
|
2018-09-04 04:56:14 -04:00
|
|
|
|
#:key (parent (delay #f))
|
|
|
|
|
(properties '()))
|
2016-07-26 12:18:53 -04:00
|
|
|
|
"Return a manifest entry for the OUTPUT of package PACKAGE."
|
2017-06-06 09:29:50 -04:00
|
|
|
|
;; For each dependency, keep a promise pointing to its "parent" entry.
|
|
|
|
|
(letrec* ((deps (map (match-lambda
|
|
|
|
|
((label package)
|
|
|
|
|
(package->manifest-entry package
|
|
|
|
|
#:parent (delay entry)))
|
|
|
|
|
((label package output)
|
|
|
|
|
(package->manifest-entry package output
|
|
|
|
|
#:parent (delay entry))))
|
|
|
|
|
(package-propagated-inputs package)))
|
|
|
|
|
(entry (manifest-entry
|
|
|
|
|
(name (package-name package))
|
|
|
|
|
(version (package-version package))
|
|
|
|
|
(output output)
|
|
|
|
|
(item package)
|
|
|
|
|
(dependencies (delete-duplicates deps))
|
|
|
|
|
(search-paths
|
|
|
|
|
(package-transitive-native-search-paths package))
|
2018-09-04 04:56:14 -04:00
|
|
|
|
(parent parent)
|
|
|
|
|
(properties properties))))
|
2017-06-06 09:29:50 -04:00
|
|
|
|
entry))
|
2014-07-26 16:54:40 -04:00
|
|
|
|
|
2015-05-18 07:51:56 -04:00
|
|
|
|
(define (packages->manifest packages)
|
|
|
|
|
"Return a list of manifest entries, one for each item listed in PACKAGES.
|
|
|
|
|
Elements of PACKAGES can be either package objects or package/string tuples
|
|
|
|
|
denoting a specific output of a package."
|
2018-09-18 04:21:28 -04:00
|
|
|
|
(define inferiors-loaded?
|
|
|
|
|
;; This hack allows us to provide seamless integration for inferior
|
|
|
|
|
;; packages while not having a hard dependency on (guix inferior).
|
|
|
|
|
(resolve-module '(guix inferior) #f #f #:ensure #f))
|
|
|
|
|
|
|
|
|
|
(define (inferior->entry)
|
|
|
|
|
(module-ref (resolve-interface '(guix inferior))
|
|
|
|
|
'inferior-package->manifest-entry))
|
|
|
|
|
|
2015-05-18 07:51:56 -04:00
|
|
|
|
(manifest
|
2020-12-05 11:20:10 -05:00
|
|
|
|
(delete-duplicates
|
|
|
|
|
(map (match-lambda
|
|
|
|
|
(((? package? package) output)
|
|
|
|
|
(package->manifest-entry package output))
|
|
|
|
|
((? package? package)
|
|
|
|
|
(package->manifest-entry package))
|
|
|
|
|
((thing output)
|
|
|
|
|
(if inferiors-loaded?
|
|
|
|
|
((inferior->entry) thing output)
|
|
|
|
|
(throw 'wrong-type-arg 'packages->manifest
|
|
|
|
|
"Wrong package object: ~S" (list thing) (list thing))))
|
|
|
|
|
(thing
|
|
|
|
|
(if inferiors-loaded?
|
|
|
|
|
((inferior->entry) thing)
|
|
|
|
|
(throw 'wrong-type-arg 'packages->manifest
|
|
|
|
|
"Wrong package object: ~S" (list thing) (list thing)))))
|
|
|
|
|
packages)
|
|
|
|
|
manifest-entry=?)))
|
2015-05-18 07:51:56 -04:00
|
|
|
|
|
2014-07-26 16:08:10 -04:00
|
|
|
|
(define (manifest->gexp manifest)
|
|
|
|
|
"Return a representation of MANIFEST as a gexp."
|
|
|
|
|
(define (entry->gexp entry)
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
(match entry
|
2015-05-02 17:55:24 -04:00
|
|
|
|
(($ <manifest-entry> name version output (? string? path)
|
2018-05-13 12:48:22 -04:00
|
|
|
|
(deps ...) (search-paths ...) _ (properties ...))
|
2015-05-02 17:55:24 -04:00
|
|
|
|
#~(#$name #$version #$output #$path
|
2017-06-06 08:01:12 -04:00
|
|
|
|
(propagated-inputs #$(map entry->gexp deps))
|
2015-05-02 17:55:24 -04:00
|
|
|
|
(search-paths #$(map search-path-specification->sexp
|
2018-05-13 12:48:22 -04:00
|
|
|
|
search-paths))
|
|
|
|
|
(properties . #$properties)))
|
2018-05-02 05:01:56 -04:00
|
|
|
|
(($ <manifest-entry> name version output package
|
2018-05-13 12:48:22 -04:00
|
|
|
|
(deps ...) (search-paths ...) _ (properties ...))
|
2014-07-26 16:08:10 -04:00
|
|
|
|
#~(#$name #$version #$output
|
2015-05-02 17:55:24 -04:00
|
|
|
|
(ungexp package (or output "out"))
|
2017-06-06 08:01:12 -04:00
|
|
|
|
(propagated-inputs #$(map entry->gexp deps))
|
2015-05-02 17:55:24 -04:00
|
|
|
|
(search-paths #$(map search-path-specification->sexp
|
2018-05-13 12:48:22 -04:00
|
|
|
|
search-paths))
|
|
|
|
|
(properties . #$properties)))))
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
|
|
|
|
|
(match manifest
|
|
|
|
|
(($ <manifest> (entries ...))
|
2017-06-06 08:01:12 -04:00
|
|
|
|
#~(manifest (version 3)
|
2014-07-26 16:08:10 -04:00
|
|
|
|
(packages #$(map entry->gexp entries))))))
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
|
2015-05-02 17:55:24 -04:00
|
|
|
|
(define (find-package name version)
|
|
|
|
|
"Return a package from the distro matching NAME and possibly VERSION. This
|
|
|
|
|
procedure is here for backward-compatibility and will eventually vanish."
|
|
|
|
|
(define find-best-packages-by-name ;break abstractions
|
|
|
|
|
(module-ref (resolve-interface '(gnu packages))
|
|
|
|
|
'find-best-packages-by-name))
|
|
|
|
|
|
|
|
|
|
;; Use 'find-best-packages-by-name' and not 'find-packages-by-name'; the
|
|
|
|
|
;; former traverses the module tree only once and then allows for efficient
|
|
|
|
|
;; access via a vhash.
|
|
|
|
|
(match (find-best-packages-by-name name version)
|
|
|
|
|
((p _ ...) p)
|
|
|
|
|
(_
|
|
|
|
|
(match (find-best-packages-by-name name #f)
|
|
|
|
|
((p _ ...) p)
|
|
|
|
|
(_ #f)))))
|
|
|
|
|
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
(define (sexp->manifest sexp)
|
|
|
|
|
"Parse SEXP as a manifest."
|
2015-05-02 17:55:24 -04:00
|
|
|
|
(define (infer-search-paths name version)
|
|
|
|
|
;; Infer the search path specifications for NAME-VERSION by looking up a
|
|
|
|
|
;; same-named package in the distro. Useful for the old manifest formats
|
|
|
|
|
;; that did not store search path info.
|
|
|
|
|
(let ((package (find-package name version)))
|
|
|
|
|
(if package
|
|
|
|
|
(package-native-search-paths package)
|
|
|
|
|
'())))
|
|
|
|
|
|
2017-06-06 09:29:50 -04:00
|
|
|
|
(define (infer-dependency item parent)
|
2017-06-06 08:01:12 -04:00
|
|
|
|
;; Return a <manifest-entry> for ITEM.
|
|
|
|
|
(let-values (((name version)
|
|
|
|
|
(package-name->name+version
|
|
|
|
|
(store-path-package-name item))))
|
|
|
|
|
(manifest-entry
|
|
|
|
|
(name name)
|
|
|
|
|
(version version)
|
2017-06-06 09:29:50 -04:00
|
|
|
|
(item item)
|
|
|
|
|
(parent parent))))
|
|
|
|
|
|
|
|
|
|
(define* (sexp->manifest-entry sexp #:optional (parent (delay #f)))
|
|
|
|
|
(match sexp
|
|
|
|
|
((name version output path
|
|
|
|
|
('propagated-inputs deps)
|
|
|
|
|
('search-paths search-paths)
|
|
|
|
|
extra-stuff ...)
|
|
|
|
|
;; For each of DEPS, keep a promise pointing to ENTRY.
|
|
|
|
|
(letrec* ((deps* (map (cut sexp->manifest-entry <> (delay entry))
|
|
|
|
|
deps))
|
|
|
|
|
(entry (manifest-entry
|
|
|
|
|
(name name)
|
|
|
|
|
(version version)
|
|
|
|
|
(output output)
|
|
|
|
|
(item path)
|
|
|
|
|
(dependencies deps*)
|
|
|
|
|
(search-paths (map sexp->search-path-specification
|
|
|
|
|
search-paths))
|
2018-05-13 12:48:22 -04:00
|
|
|
|
(parent parent)
|
|
|
|
|
(properties (or (assoc-ref extra-stuff 'properties)
|
|
|
|
|
'())))))
|
2017-06-06 09:29:50 -04:00
|
|
|
|
entry))))
|
2017-06-06 08:01:12 -04:00
|
|
|
|
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
(match sexp
|
|
|
|
|
(('manifest ('version 0)
|
|
|
|
|
('packages ((name version output path) ...)))
|
|
|
|
|
(manifest
|
|
|
|
|
(map (lambda (name version output path)
|
|
|
|
|
(manifest-entry
|
2017-06-06 08:01:12 -04:00
|
|
|
|
(name name)
|
|
|
|
|
(version version)
|
|
|
|
|
(output output)
|
|
|
|
|
(item path)
|
|
|
|
|
(search-paths (infer-search-paths name version))))
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
name version output path)))
|
|
|
|
|
|
|
|
|
|
;; Version 1 adds a list of propagated inputs to the
|
|
|
|
|
;; name/version/output/path tuples.
|
|
|
|
|
(('manifest ('version 1)
|
|
|
|
|
('packages ((name version output path deps) ...)))
|
|
|
|
|
(manifest
|
|
|
|
|
(map (lambda (name version output path deps)
|
2014-08-12 16:09:42 -04:00
|
|
|
|
;; Up to Guix 0.7 included, dependencies were listed as ("gmp"
|
|
|
|
|
;; "/gnu/store/...-gmp") for instance. Discard the 'label' in
|
|
|
|
|
;; such lists.
|
|
|
|
|
(let ((deps (match deps
|
|
|
|
|
(((labels directories) ...)
|
|
|
|
|
directories)
|
|
|
|
|
((directories ...)
|
|
|
|
|
directories))))
|
2017-06-06 09:29:50 -04:00
|
|
|
|
(letrec* ((deps* (map (cute infer-dependency <> (delay entry))
|
|
|
|
|
deps))
|
|
|
|
|
(entry (manifest-entry
|
|
|
|
|
(name name)
|
|
|
|
|
(version version)
|
|
|
|
|
(output output)
|
|
|
|
|
(item path)
|
|
|
|
|
(dependencies deps*)
|
|
|
|
|
(search-paths
|
|
|
|
|
(infer-search-paths name version)))))
|
|
|
|
|
entry)))
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
name version output path deps)))
|
|
|
|
|
|
2015-05-02 17:55:24 -04:00
|
|
|
|
;; Version 2 adds search paths and is slightly more verbose.
|
|
|
|
|
(('manifest ('version 2 minor-version ...)
|
|
|
|
|
('packages ((name version output path
|
|
|
|
|
('propagated-inputs deps)
|
|
|
|
|
('search-paths search-paths)
|
|
|
|
|
extra-stuff ...)
|
|
|
|
|
...)))
|
|
|
|
|
(manifest
|
|
|
|
|
(map (lambda (name version output path deps search-paths)
|
2017-06-06 09:29:50 -04:00
|
|
|
|
(letrec* ((deps* (map (cute infer-dependency <> (delay entry))
|
|
|
|
|
deps))
|
|
|
|
|
(entry (manifest-entry
|
|
|
|
|
(name name)
|
|
|
|
|
(version version)
|
|
|
|
|
(output output)
|
|
|
|
|
(item path)
|
|
|
|
|
(dependencies deps*)
|
|
|
|
|
(search-paths
|
|
|
|
|
(map sexp->search-path-specification
|
|
|
|
|
search-paths)))))
|
|
|
|
|
entry))
|
2015-05-02 17:55:24 -04:00
|
|
|
|
name version output path deps search-paths)))
|
2017-06-06 08:01:12 -04:00
|
|
|
|
|
|
|
|
|
;; Version 3 represents DEPS as full-blown manifest entries.
|
|
|
|
|
(('manifest ('version 3 minor-version ...)
|
|
|
|
|
('packages (entries ...)))
|
2017-06-06 09:29:50 -04:00
|
|
|
|
(manifest (map sexp->manifest-entry entries)))
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
(_
|
2015-05-04 16:41:31 -04:00
|
|
|
|
(raise (condition
|
|
|
|
|
(&message (message "unsupported manifest format")))))))
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
|
|
|
|
|
(define (read-manifest port)
|
|
|
|
|
"Return the packages listed in MANIFEST."
|
|
|
|
|
(sexp->manifest (read port)))
|
|
|
|
|
|
2019-11-20 06:07:02 -05:00
|
|
|
|
(define (concatenate-manifests lst)
|
|
|
|
|
"Concatenate the manifests listed in LST and return the resulting manifest."
|
|
|
|
|
(manifest (append-map manifest-entries lst)))
|
|
|
|
|
|
2019-12-29 10:19:56 -05:00
|
|
|
|
(define (map-manifest-entries proc manifest)
|
|
|
|
|
"Apply PROC to all the entries of MANIFEST and return a new manifest."
|
|
|
|
|
(make-manifest
|
|
|
|
|
(map proc (manifest-entries manifest))))
|
|
|
|
|
|
2013-11-01 18:11:17 -04:00
|
|
|
|
(define (entry-predicate pattern)
|
|
|
|
|
"Return a procedure that returns #t when passed a manifest entry that
|
|
|
|
|
matches NAME/OUTPUT/VERSION. OUTPUT and VERSION may be #f, in which case they
|
|
|
|
|
are ignored."
|
|
|
|
|
(match pattern
|
|
|
|
|
(($ <manifest-pattern> name version output)
|
|
|
|
|
(match-lambda
|
|
|
|
|
(($ <manifest-entry> entry-name entry-version entry-output)
|
|
|
|
|
(and (string=? entry-name name)
|
|
|
|
|
(or (not entry-output) (not output)
|
|
|
|
|
(string=? entry-output output))
|
|
|
|
|
(or (not version)
|
|
|
|
|
(string=? entry-version version))))))))
|
|
|
|
|
|
|
|
|
|
(define (manifest-remove manifest patterns)
|
|
|
|
|
"Remove entries for each of PATTERNS from MANIFEST. Each item in PATTERNS
|
|
|
|
|
must be a manifest-pattern."
|
|
|
|
|
(define (remove-entry pattern lst)
|
|
|
|
|
(remove (entry-predicate pattern) lst))
|
|
|
|
|
|
|
|
|
|
(make-manifest (fold remove-entry
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
(manifest-entries manifest)
|
2013-11-01 18:11:17 -04:00
|
|
|
|
patterns)))
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
|
2014-08-12 04:32:16 -04:00
|
|
|
|
(define (manifest-add manifest entries)
|
|
|
|
|
"Add a list of manifest ENTRIES to MANIFEST and return new manifest.
|
|
|
|
|
Remove MANIFEST entries that have the same name and output as ENTRIES."
|
|
|
|
|
(define (same-entry? entry name output)
|
|
|
|
|
(match entry
|
2018-03-01 05:37:36 -05:00
|
|
|
|
(($ <manifest-entry> entry-name _ entry-output _)
|
2014-08-12 04:32:16 -04:00
|
|
|
|
(and (equal? name entry-name)
|
|
|
|
|
(equal? output entry-output)))))
|
|
|
|
|
|
|
|
|
|
(make-manifest
|
2018-03-01 05:37:36 -05:00
|
|
|
|
(fold (lambda (entry result) ;XXX: quadratic
|
|
|
|
|
(match entry
|
|
|
|
|
(($ <manifest-entry> name _ out _)
|
|
|
|
|
(cons entry
|
|
|
|
|
(remove (cut same-entry? <> name out)
|
|
|
|
|
result)))))
|
|
|
|
|
(manifest-entries manifest)
|
|
|
|
|
entries)))
|
2014-08-12 04:32:16 -04:00
|
|
|
|
|
2014-09-02 15:12:59 -04:00
|
|
|
|
(define (manifest-lookup manifest pattern)
|
|
|
|
|
"Return the first item of MANIFEST that matches PATTERN, or #f if there is
|
|
|
|
|
no match.."
|
|
|
|
|
(find (entry-predicate pattern)
|
|
|
|
|
(manifest-entries manifest)))
|
|
|
|
|
|
2013-11-01 18:11:17 -04:00
|
|
|
|
(define (manifest-installed? manifest pattern)
|
|
|
|
|
"Return #t if MANIFEST has an entry matching PATTERN (a manifest-pattern),
|
|
|
|
|
#f otherwise."
|
2014-09-02 15:12:59 -04:00
|
|
|
|
(->bool (manifest-lookup manifest pattern)))
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
|
2013-11-01 18:11:17 -04:00
|
|
|
|
(define (manifest-matching-entries manifest patterns)
|
2019-02-07 08:54:43 -05:00
|
|
|
|
"Return all the entries of MANIFEST that match one of the PATTERNS. Raise
|
|
|
|
|
an '&unmatched-pattern-error' if none of the entries of MANIFEST matches one
|
|
|
|
|
of PATTERNS."
|
|
|
|
|
(fold-right (lambda (pattern matches)
|
|
|
|
|
(match (filter (entry-predicate pattern)
|
|
|
|
|
(manifest-entries manifest))
|
|
|
|
|
(()
|
|
|
|
|
(raise (condition
|
|
|
|
|
(&unmatched-pattern-error
|
|
|
|
|
(pattern pattern)
|
|
|
|
|
(manifest manifest)))))
|
|
|
|
|
(lst
|
|
|
|
|
(append lst matches))))
|
|
|
|
|
'()
|
|
|
|
|
patterns))
|
2013-11-01 18:11:17 -04:00
|
|
|
|
|
2018-07-09 04:44:36 -04:00
|
|
|
|
(define (manifest-search-paths manifest)
|
|
|
|
|
"Return the list of search path specifications that apply to MANIFEST,
|
|
|
|
|
including the search path specification for $PATH."
|
|
|
|
|
(delete-duplicates
|
|
|
|
|
(cons $PATH
|
|
|
|
|
(append-map manifest-entry-search-paths
|
|
|
|
|
(manifest-entries manifest)))))
|
|
|
|
|
|
2021-01-10 05:23:40 -05:00
|
|
|
|
(define* (manifest->code manifest
|
|
|
|
|
#:key (entry-package-version (const "")))
|
|
|
|
|
"Return an sexp representing code to build an approximate version of
|
|
|
|
|
MANIFEST; the code is wrapped in a top-level 'begin' form. Call
|
|
|
|
|
ENTRY-PACKAGE-VERSION to determine the version number to use in the spec for a
|
|
|
|
|
given entry; it can be set to 'manifest-entry-version' for fully-specified
|
|
|
|
|
version numbers, or to some other procedure to disambiguate versions for
|
|
|
|
|
packages for which several versions are available."
|
|
|
|
|
(define (entry-transformations entry)
|
|
|
|
|
;; Return the transformations that apply to ENTRY.
|
|
|
|
|
(assoc-ref (manifest-entry-properties entry) 'transformations))
|
|
|
|
|
|
|
|
|
|
(define transformation-procedures
|
|
|
|
|
;; List of transformation options/procedure name pairs.
|
|
|
|
|
(let loop ((entries (manifest-entries manifest))
|
|
|
|
|
(counter 1)
|
|
|
|
|
(result '()))
|
|
|
|
|
(match entries
|
|
|
|
|
(() result)
|
|
|
|
|
((entry . tail)
|
|
|
|
|
(match (entry-transformations entry)
|
|
|
|
|
(#f
|
|
|
|
|
(loop tail counter result))
|
|
|
|
|
(options
|
|
|
|
|
(if (assoc-ref result options)
|
|
|
|
|
(loop tail counter result)
|
|
|
|
|
(loop tail (+ 1 counter)
|
|
|
|
|
(alist-cons options
|
|
|
|
|
(string->symbol
|
|
|
|
|
(format #f "transform~a" counter))
|
|
|
|
|
result)))))))))
|
|
|
|
|
|
|
|
|
|
(define (qualified-name entry)
|
|
|
|
|
;; Return the name of ENTRY possibly with "@" followed by a version.
|
|
|
|
|
(match (entry-package-version entry)
|
|
|
|
|
("" (manifest-entry-name entry))
|
|
|
|
|
(version (string-append (manifest-entry-name entry)
|
|
|
|
|
"@" version))))
|
|
|
|
|
|
|
|
|
|
(if (null? transformation-procedures)
|
|
|
|
|
`(begin ;simplest case
|
|
|
|
|
(specifications->manifest
|
|
|
|
|
(list ,@(map (lambda (entry)
|
|
|
|
|
(match (manifest-entry-output entry)
|
|
|
|
|
("out" (qualified-name entry))
|
|
|
|
|
(output (string-append (qualified-name entry)
|
|
|
|
|
":" output))))
|
|
|
|
|
(manifest-entries manifest)))))
|
|
|
|
|
(let* ((transform (lambda (options exp)
|
|
|
|
|
(if (not options)
|
|
|
|
|
exp
|
|
|
|
|
(let ((proc (assoc-ref transformation-procedures
|
|
|
|
|
options)))
|
|
|
|
|
`(,proc ,exp))))))
|
|
|
|
|
`(begin ;transformations apply
|
|
|
|
|
(use-modules (guix transformations))
|
|
|
|
|
|
|
|
|
|
,@(map (match-lambda
|
|
|
|
|
((options . name)
|
|
|
|
|
`(define ,name
|
|
|
|
|
(options->transformation ',options))))
|
|
|
|
|
transformation-procedures)
|
|
|
|
|
|
|
|
|
|
(packages->manifest
|
|
|
|
|
(list ,@(map (lambda (entry)
|
|
|
|
|
(define options
|
|
|
|
|
(entry-transformations entry))
|
|
|
|
|
|
|
|
|
|
(define name
|
|
|
|
|
(qualified-name entry))
|
|
|
|
|
|
|
|
|
|
(match (manifest-entry-output entry)
|
|
|
|
|
("out"
|
|
|
|
|
(transform options
|
|
|
|
|
`(specification->package ,name)))
|
|
|
|
|
(output
|
|
|
|
|
`(list ,(transform
|
|
|
|
|
options
|
|
|
|
|
`(specification->package ,name))
|
|
|
|
|
,output))))
|
|
|
|
|
(manifest-entries manifest))))))))
|
|
|
|
|
|
2014-08-13 16:03:53 -04:00
|
|
|
|
|
|
|
|
|
;;;
|
|
|
|
|
;;; Manifest transactions.
|
|
|
|
|
;;;
|
|
|
|
|
|
|
|
|
|
(define-record-type* <manifest-transaction> manifest-transaction
|
|
|
|
|
make-manifest-transaction
|
|
|
|
|
manifest-transaction?
|
|
|
|
|
(install manifest-transaction-install ; list of <manifest-entry>
|
|
|
|
|
(default '()))
|
|
|
|
|
(remove manifest-transaction-remove ; list of <manifest-pattern>
|
|
|
|
|
(default '())))
|
|
|
|
|
|
2016-09-06 14:19:21 -04:00
|
|
|
|
(define (manifest-transaction-install-entry entry transaction)
|
|
|
|
|
"Augment TRANSACTION's set of installed packages with ENTRY, a
|
|
|
|
|
<manifest-entry>."
|
|
|
|
|
(manifest-transaction
|
|
|
|
|
(inherit transaction)
|
|
|
|
|
(install
|
|
|
|
|
(cons entry (manifest-transaction-install transaction)))))
|
|
|
|
|
|
|
|
|
|
(define (manifest-transaction-remove-pattern pattern transaction)
|
|
|
|
|
"Add PATTERN to TRANSACTION's list of packages to remove."
|
|
|
|
|
(manifest-transaction
|
|
|
|
|
(inherit transaction)
|
|
|
|
|
(remove
|
|
|
|
|
(cons pattern (manifest-transaction-remove transaction)))))
|
|
|
|
|
|
|
|
|
|
(define (manifest-transaction-null? transaction)
|
|
|
|
|
"Return true if TRANSACTION has no effect---i.e., it neither installs nor
|
|
|
|
|
remove software."
|
|
|
|
|
(match transaction
|
|
|
|
|
(($ <manifest-transaction> () ()) #t)
|
|
|
|
|
(($ <manifest-transaction> _ _) #f)))
|
|
|
|
|
|
2017-06-26 16:23:11 -04:00
|
|
|
|
(define (manifest-transaction-removal-candidate? entry transaction)
|
|
|
|
|
"Return true if ENTRY is a candidate for removal in TRANSACTION."
|
|
|
|
|
(any (lambda (pattern)
|
|
|
|
|
((entry-predicate pattern) entry))
|
|
|
|
|
(manifest-transaction-remove transaction)))
|
|
|
|
|
|
2014-08-30 15:52:32 -04:00
|
|
|
|
(define (manifest-transaction-effects manifest transaction)
|
2015-02-08 12:52:00 -05:00
|
|
|
|
"Compute the effect of applying TRANSACTION to MANIFEST. Return 4 values:
|
|
|
|
|
the list of packages that would be removed, installed, upgraded, or downgraded
|
|
|
|
|
when applying TRANSACTION to MANIFEST. Upgrades are represented as pairs
|
|
|
|
|
where the head is the entry being upgraded and the tail is the entry that will
|
|
|
|
|
replace it."
|
2014-08-30 15:52:32 -04:00
|
|
|
|
(define (manifest-entry->pattern entry)
|
|
|
|
|
(manifest-pattern
|
|
|
|
|
(name (manifest-entry-name entry))
|
|
|
|
|
(output (manifest-entry-output entry))))
|
2020-12-05 11:20:09 -05:00
|
|
|
|
(define manifest-entry-pair=?
|
|
|
|
|
(match-lambda*
|
|
|
|
|
(((m1a . m2a) (m1b . m2b))
|
|
|
|
|
(and (manifest-entry=? m1a m1b)
|
|
|
|
|
(manifest-entry=? m2a m2b)))
|
|
|
|
|
(_ #f)))
|
2014-08-30 15:52:32 -04:00
|
|
|
|
|
2015-02-08 12:52:00 -05:00
|
|
|
|
(let loop ((input (manifest-transaction-install transaction))
|
|
|
|
|
(install '())
|
|
|
|
|
(upgrade '())
|
|
|
|
|
(downgrade '()))
|
2014-08-30 15:52:32 -04:00
|
|
|
|
(match input
|
|
|
|
|
(()
|
|
|
|
|
(let ((remove (manifest-transaction-remove transaction)))
|
2020-12-05 11:20:09 -05:00
|
|
|
|
(values (delete-duplicates
|
|
|
|
|
(manifest-matching-entries manifest remove)
|
|
|
|
|
manifest-entry=?)
|
|
|
|
|
(delete-duplicates (reverse install) manifest-entry=?)
|
|
|
|
|
(delete-duplicates
|
|
|
|
|
(reverse upgrade)
|
|
|
|
|
manifest-entry-pair=?)
|
|
|
|
|
(delete-duplicates
|
|
|
|
|
(reverse downgrade)
|
|
|
|
|
manifest-entry-pair=?))))
|
2014-08-30 15:52:32 -04:00
|
|
|
|
((entry rest ...)
|
|
|
|
|
;; Check whether installing ENTRY corresponds to the installation of a
|
|
|
|
|
;; new package or to an upgrade.
|
|
|
|
|
|
|
|
|
|
;; XXX: When the exact same output directory is installed, we're not
|
|
|
|
|
;; really upgrading anything. Add a check for that case.
|
|
|
|
|
(let* ((pattern (manifest-entry->pattern entry))
|
2015-02-08 12:52:00 -05:00
|
|
|
|
(previous (manifest-lookup manifest pattern))
|
|
|
|
|
(newer? (and previous
|
2015-02-09 11:18:48 -05:00
|
|
|
|
(version>=? (manifest-entry-version entry)
|
|
|
|
|
(manifest-entry-version previous)))))
|
2014-08-30 15:52:32 -04:00
|
|
|
|
(loop rest
|
2014-09-02 15:12:59 -04:00
|
|
|
|
(if previous install (cons entry install))
|
2015-02-08 12:52:00 -05:00
|
|
|
|
(if (and previous newer?)
|
2014-09-02 15:12:59 -04:00
|
|
|
|
(alist-cons previous entry upgrade)
|
2015-02-08 12:52:00 -05:00
|
|
|
|
upgrade)
|
|
|
|
|
(if (and previous (not newer?))
|
|
|
|
|
(alist-cons previous entry downgrade)
|
|
|
|
|
downgrade)))))))
|
2014-08-30 15:52:32 -04:00
|
|
|
|
|
2014-08-13 16:03:53 -04:00
|
|
|
|
(define (manifest-perform-transaction manifest transaction)
|
2016-09-06 14:19:21 -04:00
|
|
|
|
"Perform TRANSACTION on MANIFEST and return the new manifest."
|
2014-08-13 16:03:53 -04:00
|
|
|
|
(let ((install (manifest-transaction-install transaction))
|
|
|
|
|
(remove (manifest-transaction-remove transaction)))
|
|
|
|
|
(manifest-add (manifest-remove manifest remove)
|
|
|
|
|
install)))
|
|
|
|
|
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
|
|
|
|
|
;;;
|
|
|
|
|
;;; Profiles.
|
|
|
|
|
;;;
|
|
|
|
|
|
2014-08-23 12:41:14 -04:00
|
|
|
|
(define (manifest-inputs manifest)
|
2015-03-15 16:51:34 -04:00
|
|
|
|
"Return a list of <gexp-input> objects for MANIFEST."
|
2017-06-06 08:01:12 -04:00
|
|
|
|
(define entry->input
|
|
|
|
|
(match-lambda
|
|
|
|
|
(($ <manifest-entry> name version output thing deps)
|
|
|
|
|
;; THING may be a package or a file name. In the latter case, assume
|
|
|
|
|
;; it's already valid.
|
|
|
|
|
(cons (gexp-input thing output)
|
|
|
|
|
(append-map entry->input deps)))))
|
|
|
|
|
|
|
|
|
|
(append-map entry->input (manifest-entries manifest)))
|
2014-08-23 12:41:14 -04:00
|
|
|
|
|
2016-09-22 16:25:12 -04:00
|
|
|
|
(define* (manifest-lookup-package manifest name #:optional version)
|
2016-04-30 02:52:30 -04:00
|
|
|
|
"Return as a monadic value the first package or store path referenced by
|
2016-09-22 16:25:12 -04:00
|
|
|
|
MANIFEST that is named NAME and optionally has the given VERSION prefix, or #f
|
|
|
|
|
if not found."
|
2016-04-30 02:52:30 -04:00
|
|
|
|
;; Return as a monadic value the package or store path referenced by the
|
|
|
|
|
;; manifest ENTRY, or #f if not referenced.
|
|
|
|
|
(define (entry-lookup-package entry)
|
|
|
|
|
(define (find-among-inputs inputs)
|
|
|
|
|
(find (lambda (input)
|
|
|
|
|
(and (package? input)
|
2016-09-22 16:25:12 -04:00
|
|
|
|
(equal? name (package-name input))
|
|
|
|
|
(if version
|
|
|
|
|
(string-prefix? version (package-version input))
|
|
|
|
|
#t)))
|
2016-04-30 02:52:30 -04:00
|
|
|
|
inputs))
|
|
|
|
|
(define (find-among-store-items items)
|
|
|
|
|
(find (lambda (item)
|
2016-10-30 02:21:15 -04:00
|
|
|
|
(let-values (((name* version*)
|
2016-09-22 16:25:12 -04:00
|
|
|
|
(package-name->name+version
|
|
|
|
|
(store-path-package-name item))))
|
2016-10-30 02:21:15 -04:00
|
|
|
|
(and (string=? name name*)
|
2016-09-22 16:25:12 -04:00
|
|
|
|
(if version
|
2016-10-30 02:21:15 -04:00
|
|
|
|
(string-prefix? version version*)
|
2016-09-22 16:25:12 -04:00
|
|
|
|
#t))))
|
2016-04-30 02:52:30 -04:00
|
|
|
|
items))
|
|
|
|
|
|
|
|
|
|
(with-monad %store-monad
|
|
|
|
|
(match (manifest-entry-item entry)
|
|
|
|
|
((? package? package)
|
2016-06-10 22:29:45 -04:00
|
|
|
|
(match (cons (list (package-name package) package)
|
|
|
|
|
(package-transitive-inputs package))
|
2016-04-30 02:52:30 -04:00
|
|
|
|
(((labels inputs . _) ...)
|
|
|
|
|
(return (find-among-inputs inputs)))))
|
|
|
|
|
((? string? item)
|
|
|
|
|
(mlet %store-monad ((refs (references* item)))
|
2018-05-02 05:01:56 -04:00
|
|
|
|
(return (find-among-store-items refs))))
|
|
|
|
|
(item
|
|
|
|
|
;; XXX: ITEM might be a 'computed-file' or anything like that, in
|
|
|
|
|
;; which case we don't know what to do. The fix may be to check
|
|
|
|
|
;; references once ITEM is compiled, as proposed at
|
|
|
|
|
;; <https://bugs.gnu.org/29927>.
|
|
|
|
|
(return #f)))))
|
2016-04-30 02:52:30 -04:00
|
|
|
|
|
|
|
|
|
(anym %store-monad
|
|
|
|
|
entry-lookup-package (manifest-entries manifest)))
|
|
|
|
|
|
2014-08-23 12:41:14 -04:00
|
|
|
|
(define (info-dir-file manifest)
|
|
|
|
|
"Return a derivation that builds the 'dir' file for all the entries of
|
|
|
|
|
MANIFEST."
|
2014-08-27 05:00:11 -04:00
|
|
|
|
(define texinfo ;lazy reference
|
|
|
|
|
(module-ref (resolve-interface '(gnu packages texinfo)) 'texinfo))
|
|
|
|
|
(define gzip ;lazy reference
|
|
|
|
|
(module-ref (resolve-interface '(gnu packages compression)) 'gzip))
|
2018-07-04 04:52:59 -04:00
|
|
|
|
(define glibc-utf8-locales ;lazy reference
|
|
|
|
|
(module-ref (resolve-interface '(gnu packages base)) 'glibc-utf8-locales))
|
2014-08-27 05:00:11 -04:00
|
|
|
|
|
2014-08-23 12:41:14 -04:00
|
|
|
|
(define build
|
2016-07-11 18:54:22 -04:00
|
|
|
|
(with-imported-modules '((guix build utils))
|
|
|
|
|
#~(begin
|
|
|
|
|
(use-modules (guix build utils)
|
|
|
|
|
(srfi srfi-1) (srfi srfi-26)
|
|
|
|
|
(ice-9 ftw))
|
|
|
|
|
|
|
|
|
|
(define (info-file? file)
|
|
|
|
|
(or (string-suffix? ".info" file)
|
|
|
|
|
(string-suffix? ".info.gz" file)))
|
|
|
|
|
|
|
|
|
|
(define (info-files top)
|
|
|
|
|
(let ((infodir (string-append top "/share/info")))
|
|
|
|
|
(map (cut string-append infodir "/" <>)
|
|
|
|
|
(or (scandir infodir info-file?) '()))))
|
|
|
|
|
|
2018-07-04 04:52:59 -04:00
|
|
|
|
(define (info-file-language file)
|
|
|
|
|
(let* ((base (if (string-suffix? ".gz" file)
|
|
|
|
|
(basename file ".info.gz")
|
|
|
|
|
(basename file ".info")))
|
|
|
|
|
(dot (string-rindex base #\.)))
|
|
|
|
|
(if dot
|
|
|
|
|
(string-drop base (+ 1 dot))
|
|
|
|
|
"en")))
|
|
|
|
|
|
2016-07-11 18:54:22 -04:00
|
|
|
|
(define (install-info info)
|
2018-07-04 04:52:59 -04:00
|
|
|
|
(let ((language (info-file-language info)))
|
|
|
|
|
;; We need to choose a valid locale for $LANGUAGE to be honored.
|
|
|
|
|
(setenv "LC_ALL" "en_US.utf8")
|
|
|
|
|
(setenv "LANGUAGE" language)
|
|
|
|
|
(zero?
|
|
|
|
|
(system* #+(file-append texinfo "/bin/install-info")
|
|
|
|
|
"--silent" info
|
|
|
|
|
(apply string-append #$output "/share/info/dir"
|
|
|
|
|
(if (string=? "en" language)
|
|
|
|
|
'("")
|
|
|
|
|
`("." ,language)))))))
|
|
|
|
|
|
|
|
|
|
(setenv "PATH" (string-append #+gzip "/bin")) ;for info.gz files
|
|
|
|
|
(setenv "GUIX_LOCPATH"
|
|
|
|
|
#+(file-append glibc-utf8-locales "/lib/locale"))
|
2016-07-11 18:54:22 -04:00
|
|
|
|
|
|
|
|
|
(mkdir-p (string-append #$output "/share/info"))
|
|
|
|
|
(exit (every install-info
|
|
|
|
|
(append-map info-files
|
|
|
|
|
'#$(manifest-inputs manifest)))))))
|
2014-08-23 12:41:14 -04:00
|
|
|
|
|
2015-04-15 16:44:51 -04:00
|
|
|
|
(gexp->derivation "info-dir" build
|
2015-09-24 16:13:11 -04:00
|
|
|
|
#:local-build? #t
|
2018-12-19 08:36:29 -05:00
|
|
|
|
#:substitutable? #f
|
|
|
|
|
#:properties
|
|
|
|
|
`((type . profile-hook)
|
|
|
|
|
(hook . info-dir))))
|
2014-08-23 12:41:14 -04:00
|
|
|
|
|
2015-04-04 16:51:13 -04:00
|
|
|
|
(define (ghc-package-cache-file manifest)
|
|
|
|
|
"Return a derivation that builds the GHC 'package.cache' file for all the
|
2015-04-15 16:44:51 -04:00
|
|
|
|
entries of MANIFEST, or #f if MANIFEST does not have any GHC packages."
|
2016-07-11 18:54:22 -04:00
|
|
|
|
(define ghc ;lazy reference
|
2015-04-04 16:51:13 -04:00
|
|
|
|
(module-ref (resolve-interface '(gnu packages haskell)) 'ghc))
|
|
|
|
|
|
|
|
|
|
(define build
|
2016-07-11 18:54:22 -04:00
|
|
|
|
(with-imported-modules '((guix build utils))
|
|
|
|
|
#~(begin
|
|
|
|
|
(use-modules (guix build utils)
|
|
|
|
|
(srfi srfi-1) (srfi srfi-26)
|
|
|
|
|
(ice-9 ftw))
|
|
|
|
|
|
|
|
|
|
(define ghc-name-version
|
|
|
|
|
(let* ((base (basename #+ghc)))
|
|
|
|
|
(string-drop base
|
|
|
|
|
(+ 1 (string-index base #\-)))))
|
|
|
|
|
|
|
|
|
|
(define db-subdir
|
|
|
|
|
(string-append "lib/" ghc-name-version "/package.conf.d"))
|
|
|
|
|
|
|
|
|
|
(define db-dir
|
|
|
|
|
(string-append #$output "/" db-subdir))
|
|
|
|
|
|
|
|
|
|
(define (conf-files top)
|
|
|
|
|
(let ((db (string-append top "/" db-subdir)))
|
|
|
|
|
(if (file-exists? db)
|
|
|
|
|
(find-files db "\\.conf$")
|
|
|
|
|
'())))
|
|
|
|
|
|
|
|
|
|
(define (copy-conf-file conf)
|
|
|
|
|
(let ((base (basename conf)))
|
|
|
|
|
(copy-file conf (string-append db-dir "/" base))))
|
|
|
|
|
|
|
|
|
|
(system* (string-append #+ghc "/bin/ghc-pkg") "init" db-dir)
|
|
|
|
|
(for-each copy-conf-file
|
|
|
|
|
(append-map conf-files
|
|
|
|
|
(delete-duplicates
|
|
|
|
|
'#$(manifest-inputs manifest))))
|
|
|
|
|
(let ((success
|
|
|
|
|
(zero?
|
|
|
|
|
(system* (string-append #+ghc "/bin/ghc-pkg") "recache"
|
|
|
|
|
(string-append "--package-db=" db-dir)))))
|
|
|
|
|
(for-each delete-file (find-files db-dir "\\.conf$"))
|
|
|
|
|
(exit success)))))
|
2015-04-04 16:51:13 -04:00
|
|
|
|
|
2015-05-26 17:38:27 -04:00
|
|
|
|
(with-monad %store-monad
|
|
|
|
|
;; Don't depend on GHC when there's nothing to do.
|
|
|
|
|
(if (any (cut string-prefix? "ghc" <>)
|
|
|
|
|
(map manifest-entry-name (manifest-entries manifest)))
|
|
|
|
|
(gexp->derivation "ghc-package-cache" build
|
2015-09-24 16:13:11 -04:00
|
|
|
|
#:local-build? #t
|
2018-12-19 08:36:29 -05:00
|
|
|
|
#:substitutable? #f
|
|
|
|
|
#:properties
|
|
|
|
|
`((type . profile-hook)
|
|
|
|
|
(hook . ghc-package-cache)))
|
2015-05-26 17:38:27 -04:00
|
|
|
|
(return #f))))
|
2015-04-04 16:51:13 -04:00
|
|
|
|
|
2015-03-03 02:09:30 -05:00
|
|
|
|
(define (ca-certificate-bundle manifest)
|
|
|
|
|
"Return a derivation that builds a single-file bundle containing the CA
|
|
|
|
|
certificates in the /etc/ssl/certs sub-directories of the packages in
|
|
|
|
|
MANIFEST. Single-file bundles are required by programs such as Git and Lynx."
|
|
|
|
|
;; See <http://lists.gnu.org/archive/html/guix-devel/2015-02/msg00429.html>
|
|
|
|
|
;; for a discussion.
|
|
|
|
|
|
|
|
|
|
(define glibc-utf8-locales ;lazy reference
|
|
|
|
|
(module-ref (resolve-interface '(gnu packages base)) 'glibc-utf8-locales))
|
|
|
|
|
|
|
|
|
|
(define build
|
2016-07-11 18:54:22 -04:00
|
|
|
|
(with-imported-modules '((guix build utils))
|
|
|
|
|
#~(begin
|
|
|
|
|
(use-modules (guix build utils)
|
|
|
|
|
(rnrs io ports)
|
|
|
|
|
(srfi srfi-1)
|
|
|
|
|
(srfi srfi-26)
|
|
|
|
|
(ice-9 ftw)
|
|
|
|
|
(ice-9 match))
|
|
|
|
|
|
|
|
|
|
(define (pem-file? file)
|
|
|
|
|
(string-suffix? ".pem" file))
|
|
|
|
|
|
|
|
|
|
(define (ca-files top)
|
|
|
|
|
(let ((cert-dir (string-append top "/etc/ssl/certs")))
|
|
|
|
|
(map (cut string-append cert-dir "/" <>)
|
|
|
|
|
(or (scandir cert-dir pem-file?) '()))))
|
|
|
|
|
|
|
|
|
|
(define (concatenate-files files result)
|
|
|
|
|
"Make RESULT the concatenation of all of FILES."
|
|
|
|
|
(define (dump file port)
|
|
|
|
|
(display (call-with-input-file file get-string-all)
|
|
|
|
|
port)
|
|
|
|
|
(newline port)) ;required, see <https://bugs.debian.org/635570>
|
|
|
|
|
|
|
|
|
|
(call-with-output-file result
|
|
|
|
|
(lambda (port)
|
|
|
|
|
(for-each (cut dump <> port) files))))
|
|
|
|
|
|
|
|
|
|
;; Some file names in the NSS certificates are UTF-8 encoded so
|
|
|
|
|
;; install a UTF-8 locale.
|
|
|
|
|
(setenv "LOCPATH"
|
|
|
|
|
(string-append #+glibc-utf8-locales "/lib/locale/"
|
2017-12-02 20:32:16 -05:00
|
|
|
|
#+(version-major+minor
|
|
|
|
|
(package-version glibc-utf8-locales))))
|
2016-07-11 18:54:22 -04:00
|
|
|
|
(setlocale LC_ALL "en_US.utf8")
|
|
|
|
|
|
|
|
|
|
(match (append-map ca-files '#$(manifest-inputs manifest))
|
|
|
|
|
(()
|
|
|
|
|
;; Since there are no CA files, just create an empty directory. Do
|
|
|
|
|
;; not create the etc/ssl/certs sub-directory, since that would
|
|
|
|
|
;; wrongfully lead to a message about 'SSL_CERT_DIR' needing to be
|
|
|
|
|
;; defined.
|
|
|
|
|
(mkdir #$output)
|
|
|
|
|
#t)
|
|
|
|
|
((ca-files ...)
|
|
|
|
|
(let ((result (string-append #$output "/etc/ssl/certs")))
|
|
|
|
|
(mkdir-p result)
|
|
|
|
|
(concatenate-files ca-files
|
|
|
|
|
(string-append result
|
|
|
|
|
"/ca-certificates.crt"))
|
|
|
|
|
#t))))))
|
2015-03-03 02:09:30 -05:00
|
|
|
|
|
2015-04-15 16:44:51 -04:00
|
|
|
|
(gexp->derivation "ca-certificate-bundle" build
|
2015-09-24 16:13:11 -04:00
|
|
|
|
#:local-build? #t
|
2018-12-19 08:36:29 -05:00
|
|
|
|
#:substitutable? #f
|
|
|
|
|
#:properties
|
|
|
|
|
`((type . profile-hook)
|
|
|
|
|
(hook . ca-certificate-bundle))))
|
2015-04-15 16:44:51 -04:00
|
|
|
|
|
2018-05-15 08:49:17 -04:00
|
|
|
|
(define (glib-schemas manifest)
|
|
|
|
|
"Return a derivation that unions all schemas from manifest entries and
|
|
|
|
|
creates the Glib 'gschemas.compiled' file."
|
|
|
|
|
(define glib ; lazy reference
|
|
|
|
|
(module-ref (resolve-interface '(gnu packages glib)) 'glib))
|
|
|
|
|
|
|
|
|
|
(mlet %store-monad ((%glib (manifest-lookup-package manifest "glib"))
|
|
|
|
|
;; XXX: Can't use glib-compile-schemas corresponding
|
|
|
|
|
;; to the glib referenced by 'manifest'. Because
|
|
|
|
|
;; '%glib' can be either a package or store path, and
|
|
|
|
|
;; there's no way to get the "bin" output for the later.
|
|
|
|
|
(glib-compile-schemas
|
|
|
|
|
-> #~(string-append #+glib:bin
|
|
|
|
|
"/bin/glib-compile-schemas")))
|
|
|
|
|
|
|
|
|
|
(define build
|
|
|
|
|
(with-imported-modules '((guix build utils)
|
|
|
|
|
(guix build union)
|
|
|
|
|
(guix build profiles)
|
|
|
|
|
(guix search-paths)
|
|
|
|
|
(guix records))
|
|
|
|
|
#~(begin
|
|
|
|
|
(use-modules (guix build utils)
|
|
|
|
|
(guix build union)
|
|
|
|
|
(guix build profiles)
|
|
|
|
|
(srfi srfi-26))
|
|
|
|
|
|
|
|
|
|
(let* ((destdir (string-append #$output "/share/glib-2.0/schemas"))
|
|
|
|
|
(schemadirs (filter file-exists?
|
|
|
|
|
(map (cut string-append <> "/share/glib-2.0/schemas")
|
|
|
|
|
'#$(manifest-inputs manifest)))))
|
|
|
|
|
|
|
|
|
|
;; Union all the schemas.
|
|
|
|
|
(mkdir-p (string-append #$output "/share/glib-2.0"))
|
|
|
|
|
(union-build destdir schemadirs
|
|
|
|
|
#:log-port (%make-void-port "w"))
|
|
|
|
|
|
|
|
|
|
(let ((dir destdir))
|
|
|
|
|
(when (file-is-directory? dir)
|
|
|
|
|
(ensure-writable-directory dir)
|
|
|
|
|
(invoke #+glib-compile-schemas
|
|
|
|
|
(string-append "--targetdir=" dir)
|
|
|
|
|
dir)))))))
|
|
|
|
|
|
|
|
|
|
;; Don't run the hook when there's nothing to do.
|
|
|
|
|
(if %glib
|
|
|
|
|
(gexp->derivation "glib-schemas" build
|
|
|
|
|
#:local-build? #t
|
2018-12-19 08:36:29 -05:00
|
|
|
|
#:substitutable? #f
|
|
|
|
|
#:properties
|
|
|
|
|
`((type . profile-hook)
|
|
|
|
|
(hook . glib-schemas)))
|
2018-05-15 08:49:17 -04:00
|
|
|
|
(return #f))))
|
|
|
|
|
|
2015-05-27 08:58:27 -04:00
|
|
|
|
(define (gtk-icon-themes manifest)
|
|
|
|
|
"Return a derivation that unions all icon themes from manifest entries and
|
|
|
|
|
creates the GTK+ 'icon-theme.cache' file for each theme."
|
2016-08-06 06:28:57 -04:00
|
|
|
|
(define gtk+ ; lazy reference
|
|
|
|
|
(module-ref (resolve-interface '(gnu packages gtk)) 'gtk+))
|
|
|
|
|
|
|
|
|
|
(mlet %store-monad ((%gtk+ (manifest-lookup-package manifest "gtk+"))
|
|
|
|
|
;; XXX: Can't use gtk-update-icon-cache corresponding
|
|
|
|
|
;; to the gtk+ referenced by 'manifest'. Because
|
|
|
|
|
;; '%gtk+' can be either a package or store path, and
|
|
|
|
|
;; there's no way to get the "bin" output for the later.
|
|
|
|
|
(gtk-update-icon-cache
|
|
|
|
|
-> #~(string-append #+gtk+:bin
|
|
|
|
|
"/bin/gtk-update-icon-cache")))
|
|
|
|
|
|
2015-05-27 08:58:27 -04:00
|
|
|
|
(define build
|
2016-07-11 18:54:22 -04:00
|
|
|
|
(with-imported-modules '((guix build utils)
|
|
|
|
|
(guix build union)
|
|
|
|
|
(guix build profiles)
|
|
|
|
|
(guix search-paths)
|
|
|
|
|
(guix records))
|
|
|
|
|
#~(begin
|
|
|
|
|
(use-modules (guix build utils)
|
|
|
|
|
(guix build union)
|
|
|
|
|
(guix build profiles)
|
|
|
|
|
(srfi srfi-26)
|
|
|
|
|
(ice-9 ftw))
|
|
|
|
|
|
|
|
|
|
(let* ((destdir (string-append #$output "/share/icons"))
|
|
|
|
|
(icondirs (filter file-exists?
|
|
|
|
|
(map (cut string-append <> "/share/icons")
|
2016-08-06 06:28:57 -04:00
|
|
|
|
'#$(manifest-inputs manifest)))))
|
2016-07-11 18:54:22 -04:00
|
|
|
|
|
|
|
|
|
;; Union all the icons.
|
|
|
|
|
(mkdir-p (string-append #$output "/share"))
|
|
|
|
|
(union-build destdir icondirs
|
|
|
|
|
#:log-port (%make-void-port "w"))
|
|
|
|
|
|
|
|
|
|
;; Update the 'icon-theme.cache' file for each icon theme.
|
|
|
|
|
(for-each
|
|
|
|
|
(lambda (theme)
|
|
|
|
|
(let ((dir (string-append destdir "/" theme)))
|
|
|
|
|
;; Occasionally DESTDIR contains plain files, such as
|
|
|
|
|
;; "abiword_48.png". Ignore these.
|
|
|
|
|
(when (file-is-directory? dir)
|
|
|
|
|
(ensure-writable-directory dir)
|
2016-08-06 06:28:57 -04:00
|
|
|
|
(system* #+gtk-update-icon-cache "-t" dir "--quiet"))))
|
2016-07-11 18:54:22 -04:00
|
|
|
|
(scandir destdir (negate (cut member <> '("." "..")))))))))
|
2015-05-27 08:58:27 -04:00
|
|
|
|
|
|
|
|
|
;; Don't run the hook when there's nothing to do.
|
2016-08-06 06:28:57 -04:00
|
|
|
|
(if %gtk+
|
2015-05-27 08:58:27 -04:00
|
|
|
|
(gexp->derivation "gtk-icon-themes" build
|
2015-09-24 16:13:11 -04:00
|
|
|
|
#:local-build? #t
|
2018-12-19 08:36:29 -05:00
|
|
|
|
#:substitutable? #f
|
|
|
|
|
#:properties
|
|
|
|
|
`((type . profile-hook)
|
|
|
|
|
(hook . gtk-icon-themes)))
|
2015-05-27 08:58:27 -04:00
|
|
|
|
(return #f))))
|
|
|
|
|
|
2016-09-22 16:27:06 -04:00
|
|
|
|
(define (gtk-im-modules manifest)
|
|
|
|
|
"Return a derivation that builds the cache files for input method modules
|
|
|
|
|
for both major versions of GTK+."
|
|
|
|
|
|
|
|
|
|
(mlet %store-monad ((gtk+ (manifest-lookup-package manifest "gtk+" "3"))
|
|
|
|
|
(gtk+-2 (manifest-lookup-package manifest "gtk+" "2")))
|
|
|
|
|
|
2017-02-08 07:10:46 -05:00
|
|
|
|
(define (build gtk gtk-version query)
|
2016-09-22 16:27:06 -04:00
|
|
|
|
(let ((major (string-take gtk-version 1)))
|
|
|
|
|
(with-imported-modules '((guix build utils)
|
|
|
|
|
(guix build union)
|
|
|
|
|
(guix build profiles)
|
|
|
|
|
(guix search-paths)
|
|
|
|
|
(guix records))
|
|
|
|
|
#~(begin
|
|
|
|
|
(use-modules (guix build utils)
|
|
|
|
|
(guix build union)
|
|
|
|
|
(guix build profiles)
|
|
|
|
|
(ice-9 popen)
|
|
|
|
|
(srfi srfi-1)
|
|
|
|
|
(srfi srfi-26))
|
|
|
|
|
|
|
|
|
|
(let* ((prefix (string-append "/lib/gtk-" #$major ".0/"
|
|
|
|
|
#$gtk-version))
|
|
|
|
|
(destdir (string-append #$output prefix))
|
|
|
|
|
(moddirs (cons (string-append #$gtk prefix "/immodules")
|
|
|
|
|
(filter file-exists?
|
|
|
|
|
(map (cut string-append <> prefix "/immodules")
|
|
|
|
|
'#$(manifest-inputs manifest)))))
|
|
|
|
|
(modules (append-map (cut find-files <> "\\.so$")
|
|
|
|
|
moddirs)))
|
|
|
|
|
|
|
|
|
|
;; Generate a new immodules cache file.
|
|
|
|
|
(mkdir-p (string-append #$output prefix))
|
2017-02-08 07:10:46 -05:00
|
|
|
|
(let ((pipe (apply open-pipe* OPEN_READ #$query modules))
|
2016-09-22 16:27:06 -04:00
|
|
|
|
(outfile (string-append #$output prefix
|
|
|
|
|
"/immodules-gtk" #$major ".cache")))
|
|
|
|
|
(dynamic-wind
|
|
|
|
|
(const #t)
|
|
|
|
|
(lambda ()
|
|
|
|
|
(call-with-output-file outfile
|
|
|
|
|
(lambda (out)
|
|
|
|
|
(while (not (eof-object? (peek-char pipe)))
|
|
|
|
|
(write-char (read-char pipe) out))))
|
|
|
|
|
#t)
|
|
|
|
|
(lambda ()
|
|
|
|
|
(close-pipe pipe)))))))))
|
|
|
|
|
|
|
|
|
|
;; Don't run the hook when there's nothing to do.
|
2017-02-08 07:10:46 -05:00
|
|
|
|
(let* ((pkg-gtk+ (module-ref ; lazy reference
|
|
|
|
|
(resolve-interface '(gnu packages gtk)) 'gtk+))
|
2019-12-14 13:47:57 -05:00
|
|
|
|
(pkg-gtk+2 (module-ref ; lazy reference
|
|
|
|
|
(resolve-interface '(gnu packages gtk)) 'gtk+-2))
|
2017-02-08 07:10:46 -05:00
|
|
|
|
(gexp #~(begin
|
|
|
|
|
#$(if gtk+
|
|
|
|
|
(build
|
|
|
|
|
gtk+ "3.0.0"
|
|
|
|
|
;; Use 'gtk-query-immodules-3.0' from the 'bin'
|
|
|
|
|
;; output of latest gtk+ package.
|
|
|
|
|
#~(string-append
|
|
|
|
|
#$pkg-gtk+:bin "/bin/gtk-query-immodules-3.0"))
|
|
|
|
|
#t)
|
|
|
|
|
#$(if gtk+-2
|
|
|
|
|
(build
|
|
|
|
|
gtk+-2 "2.10.0"
|
|
|
|
|
#~(string-append
|
2019-12-15 12:19:41 -05:00
|
|
|
|
#$pkg-gtk+2:bin "/bin/gtk-query-immodules-2.0"))
|
2017-02-08 07:10:46 -05:00
|
|
|
|
#t))))
|
2016-09-22 16:27:06 -04:00
|
|
|
|
(if (or gtk+ gtk+-2)
|
|
|
|
|
(gexp->derivation "gtk-im-modules" gexp
|
|
|
|
|
#:local-build? #t
|
2018-12-19 08:36:29 -05:00
|
|
|
|
#:substitutable? #f
|
|
|
|
|
#:properties
|
|
|
|
|
`((type . profile-hook)
|
|
|
|
|
(hook . gtk-im-modules)))
|
2016-09-22 16:27:06 -04:00
|
|
|
|
(return #f)))))
|
|
|
|
|
|
2020-02-18 04:42:07 -05:00
|
|
|
|
(define (linux-module-database manifest)
|
|
|
|
|
"Return a derivation that unites all the kernel modules of the manifest
|
|
|
|
|
and creates the dependency graph of all these kernel modules.
|
|
|
|
|
|
|
|
|
|
This is meant to be used as a profile hook."
|
2020-06-18 08:33:55 -04:00
|
|
|
|
(define kmod ; lazy reference
|
2020-02-18 04:42:07 -05:00
|
|
|
|
(module-ref (resolve-interface '(gnu packages linux)) 'kmod))
|
linux-libre: Support module compression.
This commit adds support for GZIP compression for linux-libre kernel
modules. The initrd modules are kept uncompressed as the initrd is already
compressed as a whole.
The linux-libre kernel also supports XZ compression, but as Guix does not have
any available bindings for now, and the compression time is far more
significant, GZIP seems to be a better option.
* gnu/build/linux-modules.scm (modinfo-section-contents): Use
'call-with-gzip-input-port' to read from a module file using '.gz' extension,
(strip-extension): new procedure,
(dot-ko): adapt to support compression,
(ensure-dot-ko): ditto,
(file-name->module-name): ditto,
(find-module-file): ditto,
(load-linux-module*): ditto,
(module-name->file-name/guess): ditto,
(module-name-lookup): ditto,
(write-module-name-database): ditto,
(write-module-alias-database): ditto,
(write-module-device-database): ditto.
* gnu/installer.scm (installer-program): Add "guile-zlib" to the extensions.
* gnu/machine/ssh.scm (machine-check-initrd-modules): Ditto.
* gnu/services.scm (activation-script): Ditto.
* gnu/services/base.scm (default-serial-port): Ditto,
(agetty-shepherd-service): ditto,
(udev-service-type): ditto.
* gnu/system/image.scm (gcrypt-sqlite3&co): Ditto.
* gnu/system/linux-initrd.scm (flat-linux-module-directory): Add "guile-zlib"
to the extensions and make sure that the initrd only contains
uncompressed module files.
* gnu/system/shadow.scm (account-shepherd-service): Add "guile-zlib" to the
extensions.
* guix/profiles.scm (linux-module-database): Ditto.
2020-07-05 06:23:21 -04:00
|
|
|
|
|
|
|
|
|
(define guile-zlib
|
|
|
|
|
(module-ref (resolve-interface '(gnu packages guile)) 'guile-zlib))
|
|
|
|
|
|
2020-02-18 04:42:07 -05:00
|
|
|
|
(define build
|
2020-06-18 08:33:55 -04:00
|
|
|
|
(with-imported-modules (source-module-closure
|
|
|
|
|
'((guix build utils)
|
2020-02-18 04:42:07 -05:00
|
|
|
|
(gnu build linux-modules)))
|
linux-libre: Support module compression.
This commit adds support for GZIP compression for linux-libre kernel
modules. The initrd modules are kept uncompressed as the initrd is already
compressed as a whole.
The linux-libre kernel also supports XZ compression, but as Guix does not have
any available bindings for now, and the compression time is far more
significant, GZIP seems to be a better option.
* gnu/build/linux-modules.scm (modinfo-section-contents): Use
'call-with-gzip-input-port' to read from a module file using '.gz' extension,
(strip-extension): new procedure,
(dot-ko): adapt to support compression,
(ensure-dot-ko): ditto,
(file-name->module-name): ditto,
(find-module-file): ditto,
(load-linux-module*): ditto,
(module-name->file-name/guess): ditto,
(module-name-lookup): ditto,
(write-module-name-database): ditto,
(write-module-alias-database): ditto,
(write-module-device-database): ditto.
* gnu/installer.scm (installer-program): Add "guile-zlib" to the extensions.
* gnu/machine/ssh.scm (machine-check-initrd-modules): Ditto.
* gnu/services.scm (activation-script): Ditto.
* gnu/services/base.scm (default-serial-port): Ditto,
(agetty-shepherd-service): ditto,
(udev-service-type): ditto.
* gnu/system/image.scm (gcrypt-sqlite3&co): Ditto.
* gnu/system/linux-initrd.scm (flat-linux-module-directory): Add "guile-zlib"
to the extensions and make sure that the initrd only contains
uncompressed module files.
* gnu/system/shadow.scm (account-shepherd-service): Add "guile-zlib" to the
extensions.
* guix/profiles.scm (linux-module-database): Ditto.
2020-07-05 06:23:21 -04:00
|
|
|
|
(with-extensions (list guile-zlib)
|
|
|
|
|
#~(begin
|
|
|
|
|
(use-modules (ice-9 ftw)
|
|
|
|
|
(ice-9 match)
|
|
|
|
|
(srfi srfi-1) ; append-map
|
|
|
|
|
(gnu build linux-modules))
|
|
|
|
|
|
|
|
|
|
(let* ((inputs '#$(manifest-inputs manifest))
|
|
|
|
|
(module-directories
|
|
|
|
|
(map (lambda (directory)
|
|
|
|
|
(string-append directory "/lib/modules"))
|
|
|
|
|
inputs))
|
|
|
|
|
(directory-entries
|
|
|
|
|
(lambda (directory)
|
|
|
|
|
(or (scandir directory
|
|
|
|
|
(lambda (basename)
|
|
|
|
|
(not (string-prefix? "." basename))))
|
|
|
|
|
'())))
|
|
|
|
|
;; Note: Should usually result in one entry.
|
|
|
|
|
(versions (delete-duplicates
|
|
|
|
|
(append-map directory-entries
|
|
|
|
|
module-directories))))
|
|
|
|
|
(match versions
|
|
|
|
|
((version)
|
|
|
|
|
(let ((old-path (getenv "PATH")))
|
|
|
|
|
(setenv "PATH" #+(file-append kmod "/bin"))
|
|
|
|
|
(make-linux-module-directory inputs version #$output)
|
|
|
|
|
(setenv "PATH" old-path)))
|
|
|
|
|
(()
|
|
|
|
|
;; Nothing here, maybe because this is a kernel with
|
|
|
|
|
;; CONFIG_MODULES=n.
|
|
|
|
|
(mkdir #$output))
|
|
|
|
|
(_ (error "Specified Linux kernel and Linux kernel modules
|
|
|
|
|
are not all of the same version"))))))))
|
2020-02-18 04:42:07 -05:00
|
|
|
|
(gexp->derivation "linux-module-database" build
|
|
|
|
|
#:local-build? #t
|
|
|
|
|
#:substitutable? #f
|
|
|
|
|
#:properties
|
|
|
|
|
`((type . profile-hook)
|
|
|
|
|
(hook . linux-module-database))))
|
|
|
|
|
|
2016-02-04 02:33:07 -05:00
|
|
|
|
(define (xdg-desktop-database manifest)
|
|
|
|
|
"Return a derivation that builds the @file{mimeinfo.cache} database from
|
|
|
|
|
desktop files. It's used to query what applications can handle a given
|
|
|
|
|
MIME type."
|
2017-07-03 11:11:13 -04:00
|
|
|
|
(define desktop-file-utils ; lazy reference
|
|
|
|
|
(module-ref (resolve-interface '(gnu packages freedesktop))
|
|
|
|
|
'desktop-file-utils))
|
|
|
|
|
|
|
|
|
|
(mlet %store-monad ((glib
|
2016-04-30 02:52:30 -04:00
|
|
|
|
(manifest-lookup-package
|
2017-07-03 11:11:13 -04:00
|
|
|
|
manifest "glib")))
|
2016-04-30 02:52:30 -04:00
|
|
|
|
(define build
|
2016-07-11 18:54:22 -04:00
|
|
|
|
(with-imported-modules '((guix build utils)
|
|
|
|
|
(guix build union))
|
|
|
|
|
#~(begin
|
|
|
|
|
(use-modules (srfi srfi-26)
|
|
|
|
|
(guix build utils)
|
|
|
|
|
(guix build union))
|
|
|
|
|
(let* ((destdir (string-append #$output "/share/applications"))
|
|
|
|
|
(appdirs (filter file-exists?
|
|
|
|
|
(map (cut string-append <>
|
|
|
|
|
"/share/applications")
|
|
|
|
|
'#$(manifest-inputs manifest))))
|
|
|
|
|
(update-desktop-database (string-append
|
|
|
|
|
#+desktop-file-utils
|
|
|
|
|
"/bin/update-desktop-database")))
|
|
|
|
|
(mkdir-p (string-append #$output "/share"))
|
|
|
|
|
(union-build destdir appdirs
|
|
|
|
|
#:log-port (%make-void-port "w"))
|
|
|
|
|
(exit (zero? (system* update-desktop-database destdir)))))))
|
2016-02-04 02:33:07 -05:00
|
|
|
|
|
2017-07-03 11:11:13 -04:00
|
|
|
|
;; Don't run the hook when 'glib' is not referenced.
|
|
|
|
|
(if glib
|
2016-04-30 02:52:30 -04:00
|
|
|
|
(gexp->derivation "xdg-desktop-database" build
|
|
|
|
|
#:local-build? #t
|
2018-12-19 08:36:29 -05:00
|
|
|
|
#:substitutable? #f
|
|
|
|
|
#:properties
|
|
|
|
|
`((type . profile-hook)
|
|
|
|
|
(hook . xdg-desktop-database)))
|
2016-04-30 02:52:30 -04:00
|
|
|
|
(return #f))))
|
2016-02-04 02:33:07 -05:00
|
|
|
|
|
2016-02-04 02:35:03 -05:00
|
|
|
|
(define (xdg-mime-database manifest)
|
|
|
|
|
"Return a derivation that builds the @file{mime.cache} database from manifest
|
|
|
|
|
entries. It's used to query the MIME type of a given file."
|
2016-08-11 08:59:16 -04:00
|
|
|
|
(define shared-mime-info ; lazy reference
|
|
|
|
|
(module-ref (resolve-interface '(gnu packages gnome)) 'shared-mime-info))
|
|
|
|
|
|
2020-11-02 08:25:15 -05:00
|
|
|
|
(mlet %store-monad ((glib (manifest-lookup-package manifest "glib")))
|
2016-04-30 02:52:30 -04:00
|
|
|
|
(define build
|
2016-07-11 18:54:22 -04:00
|
|
|
|
(with-imported-modules '((guix build utils)
|
|
|
|
|
(guix build union))
|
|
|
|
|
#~(begin
|
2020-11-02 08:25:15 -05:00
|
|
|
|
(use-modules (guix build utils)
|
|
|
|
|
(guix build union)
|
|
|
|
|
(srfi srfi-26)
|
|
|
|
|
(ice-9 match))
|
|
|
|
|
|
2016-07-11 18:54:22 -04:00
|
|
|
|
(let* ((datadir (string-append #$output "/share"))
|
|
|
|
|
(destdir (string-append datadir "/mime"))
|
|
|
|
|
(pkgdirs (filter file-exists?
|
|
|
|
|
(map (cut string-append <>
|
|
|
|
|
"/share/mime/packages")
|
2016-08-11 08:59:16 -04:00
|
|
|
|
(cons #+shared-mime-info
|
2020-11-02 08:25:15 -05:00
|
|
|
|
'#$(manifest-inputs manifest))))))
|
|
|
|
|
|
|
|
|
|
(match pkgdirs
|
|
|
|
|
((shared-mime-info)
|
|
|
|
|
;; PKGDIRS contains nothing but 'shared-mime-info', which
|
|
|
|
|
;; already contains its database, so nothing to do.
|
|
|
|
|
(mkdir-p datadir)
|
|
|
|
|
(symlink #$(file-append shared-mime-info "/share/mime")
|
|
|
|
|
destdir))
|
|
|
|
|
(_
|
|
|
|
|
;; PKGDIRS contains additional packages providing
|
|
|
|
|
;; 'share/mime/packages' (very few packages do so) so rebuild
|
|
|
|
|
;; the database. TODO: Find a way to avoid reprocessing
|
|
|
|
|
;; 'shared-mime-info', which is the most expensive one.
|
|
|
|
|
(mkdir-p destdir)
|
|
|
|
|
(union-build (string-append destdir "/packages") pkgdirs
|
|
|
|
|
#:log-port (%make-void-port "w"))
|
|
|
|
|
(setenv "XDG_DATA_HOME" datadir)
|
|
|
|
|
(invoke #+(file-append shared-mime-info
|
|
|
|
|
"/bin/update-mime-database")
|
|
|
|
|
destdir)))))))
|
2016-04-30 02:52:30 -04:00
|
|
|
|
|
2016-08-11 08:59:16 -04:00
|
|
|
|
;; Don't run the hook when there are no GLib based applications.
|
|
|
|
|
(if glib
|
2016-04-30 02:52:30 -04:00
|
|
|
|
(gexp->derivation "xdg-mime-database" build
|
|
|
|
|
#:local-build? #t
|
2018-12-19 08:36:29 -05:00
|
|
|
|
#:substitutable? #f
|
|
|
|
|
#:properties
|
|
|
|
|
`((type . profile-hook)
|
|
|
|
|
(hook . xdg-mime-database)))
|
2016-04-30 02:52:30 -04:00
|
|
|
|
(return #f))))
|
2016-02-04 02:35:03 -05:00
|
|
|
|
|
2017-03-12 07:53:59 -04:00
|
|
|
|
;; Several font packages may install font files into same directory, so
|
|
|
|
|
;; fonts.dir and fonts.scale file should be generated here, instead of in
|
|
|
|
|
;; packages.
|
2016-06-30 15:01:06 -04:00
|
|
|
|
(define (fonts-dir-file manifest)
|
|
|
|
|
"Return a derivation that builds the @file{fonts.dir} and @file{fonts.scale}
|
2017-03-12 07:53:59 -04:00
|
|
|
|
files for the fonts of the @var{manifest} entries."
|
2016-06-30 15:01:06 -04:00
|
|
|
|
(define mkfontscale
|
|
|
|
|
(module-ref (resolve-interface '(gnu packages xorg)) 'mkfontscale))
|
|
|
|
|
|
|
|
|
|
(define mkfontdir
|
|
|
|
|
(module-ref (resolve-interface '(gnu packages xorg)) 'mkfontdir))
|
|
|
|
|
|
|
|
|
|
(define build
|
|
|
|
|
#~(begin
|
|
|
|
|
(use-modules (srfi srfi-26)
|
|
|
|
|
(guix build utils)
|
|
|
|
|
(guix build union))
|
2017-03-12 07:53:59 -04:00
|
|
|
|
(let ((fonts-dirs (filter file-exists?
|
|
|
|
|
(map (cut string-append <>
|
|
|
|
|
"/share/fonts")
|
|
|
|
|
'#$(manifest-inputs manifest)))))
|
2016-06-30 15:01:06 -04:00
|
|
|
|
(mkdir #$output)
|
2017-03-12 07:53:59 -04:00
|
|
|
|
(if (null? fonts-dirs)
|
2016-06-30 15:01:06 -04:00
|
|
|
|
(exit #t)
|
2017-03-12 07:53:59 -04:00
|
|
|
|
(let* ((share-dir (string-append #$output "/share"))
|
|
|
|
|
(fonts-dir (string-append share-dir "/fonts"))
|
2016-06-30 15:01:06 -04:00
|
|
|
|
(mkfontscale (string-append #+mkfontscale
|
|
|
|
|
"/bin/mkfontscale"))
|
|
|
|
|
(mkfontdir (string-append #+mkfontdir
|
2017-03-12 07:53:59 -04:00
|
|
|
|
"/bin/mkfontdir"))
|
|
|
|
|
(empty-file? (lambda (filename)
|
|
|
|
|
(call-with-ascii-input-file filename
|
|
|
|
|
(lambda (p)
|
|
|
|
|
(eqv? #\0 (read-char p))))))
|
|
|
|
|
(fonts-dir-file "fonts.dir")
|
|
|
|
|
(fonts-scale-file "fonts.scale"))
|
|
|
|
|
(mkdir-p share-dir)
|
|
|
|
|
;; Create all sub-directories, because we may create fonts.dir
|
|
|
|
|
;; and fonts.scale files in the sub-directories.
|
|
|
|
|
(union-build fonts-dir fonts-dirs
|
|
|
|
|
#:log-port (%make-void-port "w")
|
|
|
|
|
#:create-all-directories? #t)
|
|
|
|
|
(let ((directories (find-files fonts-dir
|
|
|
|
|
(lambda (file stat)
|
|
|
|
|
(eq? 'directory (stat:type stat)))
|
|
|
|
|
#:directories? #t)))
|
|
|
|
|
(for-each (lambda (dir)
|
|
|
|
|
(with-directory-excursion dir
|
|
|
|
|
(when (file-exists? fonts-scale-file)
|
|
|
|
|
(delete-file fonts-scale-file))
|
|
|
|
|
(when (file-exists? fonts-dir-file)
|
|
|
|
|
(delete-file fonts-dir-file))
|
|
|
|
|
(unless (and (zero? (system* mkfontscale))
|
|
|
|
|
(zero? (system* mkfontdir)))
|
|
|
|
|
(exit #f))
|
2017-08-08 10:05:58 -04:00
|
|
|
|
(when (and (file-exists? fonts-scale-file)
|
|
|
|
|
(empty-file? fonts-scale-file))
|
2017-03-12 07:53:59 -04:00
|
|
|
|
(delete-file fonts-scale-file))
|
2017-08-08 10:05:58 -04:00
|
|
|
|
(when (and (file-exists? fonts-dir-file)
|
|
|
|
|
(empty-file? fonts-dir-file))
|
2017-03-12 07:53:59 -04:00
|
|
|
|
(delete-file fonts-dir-file))))
|
|
|
|
|
directories)))))))
|
2016-06-30 15:01:06 -04:00
|
|
|
|
|
|
|
|
|
(gexp->derivation "fonts-dir" build
|
|
|
|
|
#:modules '((guix build utils)
|
2017-03-12 07:53:59 -04:00
|
|
|
|
(guix build union)
|
|
|
|
|
(srfi srfi-26))
|
2016-06-30 15:01:06 -04:00
|
|
|
|
#:local-build? #t
|
2018-12-19 08:36:29 -05:00
|
|
|
|
#:substitutable? #f
|
|
|
|
|
#:properties
|
|
|
|
|
`((type . profile-hook)
|
|
|
|
|
(hook . fonts-dir))))
|
2016-06-30 15:01:06 -04:00
|
|
|
|
|
2017-04-05 04:09:22 -04:00
|
|
|
|
(define (manual-database manifest)
|
|
|
|
|
"Return a derivation that builds the manual page database (\"mandb\") for
|
|
|
|
|
the entries in MANIFEST."
|
2017-12-15 16:16:18 -05:00
|
|
|
|
(define gdbm-ffi
|
|
|
|
|
(module-ref (resolve-interface '(gnu packages guile))
|
|
|
|
|
'guile-gdbm-ffi))
|
|
|
|
|
|
Use "guile-zlib" and "guile-lzlib" instead of (guix config).
* Makefile.am (MODULES): Remove guix/zlib.scm and guix/lzlib.scm,
(SCM_TESTS): remove tests/zlib.scm, tests/lzlib.scm.
* build-aux/build-self.scm (make-config.scm): Remove unused %libz variable.
* configure.ac: Remove LIBZ and LIBLZ variables and check instead for
Guile-zlib and Guile-lzlib.
* doc/guix.texi ("Requirements"): Remove zlib requirement and add Guile-zlib
and Guile-lzlib instead.
* gnu/packages/package-management.scm (guix)[native-inputs]: Add "guile-zlib"
and "guile-lzlib",
[inputs]: remove "zlib" and "lzlib",
[propagated-inputs]: ditto,
[arguments]: add "guile-zlib" and "guile-lzlib" to Guile load path.
* guix/config.scm.in (%libz, %liblz): Remove them.
* guix/lzlib.scm: Remove it.
* guix/man-db.scm: Use (zlib) instead of (guix zlib).
* guix/profiles.scm (manual-database): Do not stub (guix config) in imported
modules list, instead add "guile-zlib" to the extension list.
* guix/scripts/publish.scm: Use (zlib) instead of (guix zlib) and (lzlib)
instead of (guix lzlib),
(string->compression-type, effective-compression): do not check for zlib and
lzlib availability.
* guix/scripts/substitute.scm (%compression-methods): Do not check for lzlib
availability.
* guix/self.scm (specification->package): Add "guile-zlib" and "guile-lzlib"
and remove "zlib" and "lzlib",
(compiled-guix): remove "zlib" and "lzlib" arguments and add guile-zlib and
guile-lzlib to the dependencies, also do not pass "zlib" and "lzlib" to
"make-config.scm" procedure,
(make-config.scm): remove "zlib" and "lzlib" arguments as well as %libz and
%liblz variables.
* guix/utils.scm (lzip-port): Use (lzlib) instead of (guix lzlib) and do not
check for lzlib availability.
* guix/zlib.scm: Remove it.
* m4/guix.m4 (GUIX_LIBZ_LIBDIR, GUIX_LIBLZ_FILE_NAME): Remove them.
* tests/lzlib.scm: Use (zlib) instead of (guix zlib) and (lzlib)
instead of (guix lzlib), and do not check for zlib and lzlib availability.
* tests/publish.scm: Ditto.
* tests/substitute.scm: Do not check for lzlib availability.
* tests/utils.scm: Ditto.
* tests/zlib.scm: Remove it.
2020-07-27 10:36:39 -04:00
|
|
|
|
(define guile-zlib
|
|
|
|
|
(module-ref (resolve-interface '(gnu packages guile)) 'guile-zlib))
|
2017-12-15 16:16:18 -05:00
|
|
|
|
|
|
|
|
|
(define modules
|
Use "guile-zlib" and "guile-lzlib" instead of (guix config).
* Makefile.am (MODULES): Remove guix/zlib.scm and guix/lzlib.scm,
(SCM_TESTS): remove tests/zlib.scm, tests/lzlib.scm.
* build-aux/build-self.scm (make-config.scm): Remove unused %libz variable.
* configure.ac: Remove LIBZ and LIBLZ variables and check instead for
Guile-zlib and Guile-lzlib.
* doc/guix.texi ("Requirements"): Remove zlib requirement and add Guile-zlib
and Guile-lzlib instead.
* gnu/packages/package-management.scm (guix)[native-inputs]: Add "guile-zlib"
and "guile-lzlib",
[inputs]: remove "zlib" and "lzlib",
[propagated-inputs]: ditto,
[arguments]: add "guile-zlib" and "guile-lzlib" to Guile load path.
* guix/config.scm.in (%libz, %liblz): Remove them.
* guix/lzlib.scm: Remove it.
* guix/man-db.scm: Use (zlib) instead of (guix zlib).
* guix/profiles.scm (manual-database): Do not stub (guix config) in imported
modules list, instead add "guile-zlib" to the extension list.
* guix/scripts/publish.scm: Use (zlib) instead of (guix zlib) and (lzlib)
instead of (guix lzlib),
(string->compression-type, effective-compression): do not check for zlib and
lzlib availability.
* guix/scripts/substitute.scm (%compression-methods): Do not check for lzlib
availability.
* guix/self.scm (specification->package): Add "guile-zlib" and "guile-lzlib"
and remove "zlib" and "lzlib",
(compiled-guix): remove "zlib" and "lzlib" arguments and add guile-zlib and
guile-lzlib to the dependencies, also do not pass "zlib" and "lzlib" to
"make-config.scm" procedure,
(make-config.scm): remove "zlib" and "lzlib" arguments as well as %libz and
%liblz variables.
* guix/utils.scm (lzip-port): Use (lzlib) instead of (guix lzlib) and do not
check for lzlib availability.
* guix/zlib.scm: Remove it.
* m4/guix.m4 (GUIX_LIBZ_LIBDIR, GUIX_LIBLZ_FILE_NAME): Remove them.
* tests/lzlib.scm: Use (zlib) instead of (guix zlib) and (lzlib)
instead of (guix lzlib), and do not check for zlib and lzlib availability.
* tests/publish.scm: Ditto.
* tests/substitute.scm: Do not check for lzlib availability.
* tests/utils.scm: Ditto.
* tests/zlib.scm: Remove it.
2020-07-27 10:36:39 -04:00
|
|
|
|
(delete '(guix config)
|
|
|
|
|
(source-module-closure `((guix build utils)
|
|
|
|
|
(guix man-db)))))
|
2017-04-05 04:09:22 -04:00
|
|
|
|
|
|
|
|
|
(define build
|
2017-12-15 16:16:18 -05:00
|
|
|
|
(with-imported-modules modules
|
Use "guile-zlib" and "guile-lzlib" instead of (guix config).
* Makefile.am (MODULES): Remove guix/zlib.scm and guix/lzlib.scm,
(SCM_TESTS): remove tests/zlib.scm, tests/lzlib.scm.
* build-aux/build-self.scm (make-config.scm): Remove unused %libz variable.
* configure.ac: Remove LIBZ and LIBLZ variables and check instead for
Guile-zlib and Guile-lzlib.
* doc/guix.texi ("Requirements"): Remove zlib requirement and add Guile-zlib
and Guile-lzlib instead.
* gnu/packages/package-management.scm (guix)[native-inputs]: Add "guile-zlib"
and "guile-lzlib",
[inputs]: remove "zlib" and "lzlib",
[propagated-inputs]: ditto,
[arguments]: add "guile-zlib" and "guile-lzlib" to Guile load path.
* guix/config.scm.in (%libz, %liblz): Remove them.
* guix/lzlib.scm: Remove it.
* guix/man-db.scm: Use (zlib) instead of (guix zlib).
* guix/profiles.scm (manual-database): Do not stub (guix config) in imported
modules list, instead add "guile-zlib" to the extension list.
* guix/scripts/publish.scm: Use (zlib) instead of (guix zlib) and (lzlib)
instead of (guix lzlib),
(string->compression-type, effective-compression): do not check for zlib and
lzlib availability.
* guix/scripts/substitute.scm (%compression-methods): Do not check for lzlib
availability.
* guix/self.scm (specification->package): Add "guile-zlib" and "guile-lzlib"
and remove "zlib" and "lzlib",
(compiled-guix): remove "zlib" and "lzlib" arguments and add guile-zlib and
guile-lzlib to the dependencies, also do not pass "zlib" and "lzlib" to
"make-config.scm" procedure,
(make-config.scm): remove "zlib" and "lzlib" arguments as well as %libz and
%liblz variables.
* guix/utils.scm (lzip-port): Use (lzlib) instead of (guix lzlib) and do not
check for lzlib availability.
* guix/zlib.scm: Remove it.
* m4/guix.m4 (GUIX_LIBZ_LIBDIR, GUIX_LIBLZ_FILE_NAME): Remove them.
* tests/lzlib.scm: Use (zlib) instead of (guix zlib) and (lzlib)
instead of (guix lzlib), and do not check for zlib and lzlib availability.
* tests/publish.scm: Ditto.
* tests/substitute.scm: Do not check for lzlib availability.
* tests/utils.scm: Ditto.
* tests/zlib.scm: Remove it.
2020-07-27 10:36:39 -04:00
|
|
|
|
(with-extensions (list gdbm-ffi ;for (guix man-db)
|
|
|
|
|
guile-zlib)
|
2018-05-28 16:00:11 -04:00
|
|
|
|
#~(begin
|
|
|
|
|
(use-modules (guix man-db)
|
|
|
|
|
(guix build utils)
|
2019-07-12 17:42:45 -04:00
|
|
|
|
(ice-9 threads)
|
2018-05-28 16:00:11 -04:00
|
|
|
|
(srfi srfi-1)
|
|
|
|
|
(srfi srfi-19))
|
|
|
|
|
|
2019-07-12 17:42:45 -04:00
|
|
|
|
(define (print-string msg)
|
|
|
|
|
(display msg)
|
|
|
|
|
(force-output))
|
|
|
|
|
|
|
|
|
|
(define-syntax-rule (print fmt args ...)
|
|
|
|
|
;; Build up the string and display it at once.
|
|
|
|
|
(print-string (format #f fmt args ...)))
|
|
|
|
|
|
|
|
|
|
(define (compute-entry directory count total)
|
|
|
|
|
(print "\r[~3d/~3d] building list of man-db entries..."
|
|
|
|
|
count total)
|
|
|
|
|
(let ((man (string-append directory "/share/man")))
|
|
|
|
|
(if (directory-exists? man)
|
|
|
|
|
(mandb-entries man)
|
|
|
|
|
'())))
|
|
|
|
|
|
2018-05-28 16:00:11 -04:00
|
|
|
|
(define (compute-entries)
|
2019-02-01 13:25:38 -05:00
|
|
|
|
;; This is the most expensive part (I/O and CPU, due to
|
|
|
|
|
;; decompression), so report progress as we traverse INPUTS.
|
2019-07-12 17:42:45 -04:00
|
|
|
|
;; Cap at 4 threads because we don't see any speedup beyond that
|
|
|
|
|
;; on an SSD laptop.
|
|
|
|
|
(let* ((inputs '#$(manifest-inputs manifest))
|
|
|
|
|
(total (length inputs))
|
|
|
|
|
(threads (min (parallel-job-count) 4)))
|
|
|
|
|
(concatenate
|
|
|
|
|
(n-par-map threads compute-entry inputs
|
|
|
|
|
(iota total 1)
|
|
|
|
|
(make-list total total)))))
|
2018-05-28 16:00:11 -04:00
|
|
|
|
|
|
|
|
|
(define man-directory
|
|
|
|
|
(string-append #$output "/share/man"))
|
|
|
|
|
|
|
|
|
|
(mkdir-p man-directory)
|
|
|
|
|
|
|
|
|
|
(format #t "Creating manual page database...~%")
|
|
|
|
|
(force-output)
|
|
|
|
|
(let* ((start (current-time))
|
|
|
|
|
(entries (compute-entries))
|
|
|
|
|
(_ (write-mandb-database (string-append man-directory
|
|
|
|
|
"/index.db")
|
|
|
|
|
entries))
|
|
|
|
|
(duration (time-difference (current-time) start)))
|
2019-02-01 13:25:38 -05:00
|
|
|
|
(newline)
|
2018-05-28 16:00:11 -04:00
|
|
|
|
(format #t "~a entries processed in ~,1f s~%"
|
|
|
|
|
(length entries)
|
|
|
|
|
(+ (time-second duration)
|
|
|
|
|
(* (time-nanosecond duration) (expt 10 -9))))
|
|
|
|
|
(force-output))))))
|
2017-04-05 04:09:22 -04:00
|
|
|
|
|
|
|
|
|
(gexp->derivation "manual-database" build
|
2017-12-15 16:16:18 -05:00
|
|
|
|
|
|
|
|
|
;; Work around GDBM 1.13 issue whereby uninitialized bytes
|
|
|
|
|
;; get written to disk:
|
|
|
|
|
;; <https://debbugs.gnu.org/cgi/bugreport.cgi?bug=29654#23>.
|
|
|
|
|
#:env-vars `(("MALLOC_PERTURB_" . "1"))
|
|
|
|
|
|
2020-04-30 18:40:23 -04:00
|
|
|
|
#:substitutable? #f
|
2018-12-19 08:36:29 -05:00
|
|
|
|
#:local-build? #t
|
|
|
|
|
#:properties
|
|
|
|
|
`((type . profile-hook)
|
|
|
|
|
(hook . manual-database))))
|
2017-04-05 04:09:22 -04:00
|
|
|
|
|
2019-01-15 07:03:48 -05:00
|
|
|
|
(define (texlive-configuration manifest)
|
|
|
|
|
"Return a derivation that builds a TeXlive configuration for the entries in
|
|
|
|
|
MANIFEST."
|
|
|
|
|
(define entry->texlive-input
|
|
|
|
|
(match-lambda
|
|
|
|
|
(($ <manifest-entry> name version output thing deps)
|
|
|
|
|
(if (string-prefix? "texlive-" name)
|
|
|
|
|
(cons (gexp-input thing output)
|
|
|
|
|
(append-map entry->texlive-input deps))
|
|
|
|
|
'()))))
|
|
|
|
|
(define build
|
|
|
|
|
(with-imported-modules '((guix build utils)
|
|
|
|
|
(guix build union))
|
|
|
|
|
#~(begin
|
|
|
|
|
(use-modules (guix build utils)
|
|
|
|
|
(guix build union))
|
|
|
|
|
|
|
|
|
|
;; Build a modifiable union of all texlive inputs. We do this so
|
|
|
|
|
;; that TeX live can resolve the parent and grandparent directories
|
|
|
|
|
;; correctly. There might be a more elegant way to accomplish this.
|
|
|
|
|
(union-build #$output
|
|
|
|
|
'#$(append-map entry->texlive-input
|
|
|
|
|
(manifest-entries manifest))
|
|
|
|
|
#:create-all-directories? #t
|
|
|
|
|
#:log-port (%make-void-port "w"))
|
2019-01-17 16:00:19 -05:00
|
|
|
|
(let ((texmf.cnf (string-append
|
|
|
|
|
#$output
|
|
|
|
|
"/share/texmf-dist/web2c/texmf.cnf")))
|
|
|
|
|
(when (file-exists? texmf.cnf)
|
|
|
|
|
(substitute* texmf.cnf
|
|
|
|
|
(("^TEXMFROOT = .*")
|
|
|
|
|
(string-append "TEXMFROOT = " #$output "/share\n"))
|
|
|
|
|
(("^TEXMF = .*")
|
|
|
|
|
"TEXMF = $TEXMFROOT/share/texmf-dist\n"))))
|
2019-01-15 07:03:48 -05:00
|
|
|
|
#t)))
|
|
|
|
|
|
|
|
|
|
(with-monad %store-monad
|
|
|
|
|
(if (any (cut string-prefix? "texlive-" <>)
|
|
|
|
|
(map manifest-entry-name (manifest-entries manifest)))
|
|
|
|
|
(gexp->derivation "texlive-configuration" build
|
|
|
|
|
#:substitutable? #f
|
|
|
|
|
#:local-build? #t
|
|
|
|
|
#:properties
|
|
|
|
|
`((type . profile-hook)
|
|
|
|
|
(hook . texlive-configuration)))
|
|
|
|
|
(return #f))))
|
|
|
|
|
|
2015-04-15 16:44:51 -04:00
|
|
|
|
(define %default-profile-hooks
|
|
|
|
|
;; This is the list of derivation-returning procedures that are called by
|
|
|
|
|
;; default when making a non-empty profile.
|
|
|
|
|
(list info-dir-file
|
2017-04-05 04:09:22 -04:00
|
|
|
|
manual-database
|
2016-06-30 15:01:06 -04:00
|
|
|
|
fonts-dir-file
|
2015-04-15 16:44:51 -04:00
|
|
|
|
ghc-package-cache-file
|
2015-05-27 08:58:27 -04:00
|
|
|
|
ca-certificate-bundle
|
2018-05-15 08:49:17 -04:00
|
|
|
|
glib-schemas
|
2016-02-04 02:33:07 -05:00
|
|
|
|
gtk-icon-themes
|
2016-09-22 16:27:06 -04:00
|
|
|
|
gtk-im-modules
|
2019-01-15 07:03:48 -05:00
|
|
|
|
texlive-configuration
|
2016-02-04 02:35:03 -05:00
|
|
|
|
xdg-desktop-database
|
|
|
|
|
xdg-mime-database))
|
2015-03-03 02:09:30 -05:00
|
|
|
|
|
|
|
|
|
(define* (profile-derivation manifest
|
|
|
|
|
#:key
|
2020-04-22 09:35:36 -04:00
|
|
|
|
(name "profile")
|
2016-02-12 13:12:18 -05:00
|
|
|
|
(hooks %default-profile-hooks)
|
2016-12-17 06:43:10 -05:00
|
|
|
|
(locales? #t)
|
2017-06-21 10:50:59 -04:00
|
|
|
|
(allow-collisions? #f)
|
2018-04-28 16:20:36 -04:00
|
|
|
|
(relative-symlinks? #f)
|
2017-03-17 16:48:40 -04:00
|
|
|
|
system target)
|
2014-08-23 12:41:14 -04:00
|
|
|
|
"Return a derivation that builds a profile (aka. 'user environment') with
|
2015-04-15 16:44:51 -04:00
|
|
|
|
the given MANIFEST. The profile includes additional derivations returned by
|
2016-12-17 06:43:10 -05:00
|
|
|
|
the monadic procedures listed in HOOKS--such as an Info 'dir' file, etc.
|
2017-06-21 10:50:59 -04:00
|
|
|
|
Unless ALLOW-COLLISIONS? is true, a '&profile-collision-error' is raised if
|
|
|
|
|
entries in MANIFEST collide (for instance if there are two same-name packages
|
|
|
|
|
with a different version number.)
|
2016-12-17 06:43:10 -05:00
|
|
|
|
|
|
|
|
|
When LOCALES? is true, the build is performed under a UTF-8 locale; this adds
|
2017-03-17 16:48:40 -04:00
|
|
|
|
a dependency on the 'glibc-utf8-locales' package.
|
|
|
|
|
|
2018-04-28 16:20:36 -04:00
|
|
|
|
When RELATIVE-SYMLINKS? is true, use relative file names for symlink targets.
|
|
|
|
|
This is one of the things to do for the result to be relocatable.
|
|
|
|
|
|
2017-03-17 16:48:40 -04:00
|
|
|
|
When TARGET is true, it must be a GNU triplet, and the packages in MANIFEST
|
|
|
|
|
are cross-built for TARGET."
|
2017-06-07 03:51:55 -04:00
|
|
|
|
(mlet* %store-monad ((system (if system
|
|
|
|
|
(return system)
|
|
|
|
|
(current-system)))
|
2019-12-24 09:04:57 -05:00
|
|
|
|
(target (if target
|
|
|
|
|
(return target)
|
|
|
|
|
(current-target-system)))
|
2017-06-21 10:50:59 -04:00
|
|
|
|
(ok? (if allow-collisions?
|
|
|
|
|
(return #t)
|
|
|
|
|
(check-for-collisions manifest system
|
|
|
|
|
#:target target)))
|
2017-06-07 03:51:55 -04:00
|
|
|
|
(extras (if (null? (manifest-entries manifest))
|
|
|
|
|
(return '())
|
2020-03-25 07:45:12 -04:00
|
|
|
|
(mapm/accumulate-builds (lambda (hook)
|
|
|
|
|
(hook manifest))
|
|
|
|
|
hooks))))
|
2014-08-23 12:41:14 -04:00
|
|
|
|
(define inputs
|
2015-05-27 09:03:21 -04:00
|
|
|
|
(append (filter-map (lambda (drv)
|
|
|
|
|
(and (derivation? drv)
|
|
|
|
|
(gexp-input drv)))
|
2015-05-26 17:38:27 -04:00
|
|
|
|
extras)
|
2015-03-03 02:09:30 -05:00
|
|
|
|
(manifest-inputs manifest)))
|
2014-08-23 12:41:14 -04:00
|
|
|
|
|
2016-12-16 12:01:08 -05:00
|
|
|
|
(define glibc-utf8-locales ;lazy reference
|
|
|
|
|
(module-ref (resolve-interface '(gnu packages base))
|
|
|
|
|
'glibc-utf8-locales))
|
|
|
|
|
|
2016-12-17 06:43:10 -05:00
|
|
|
|
(define set-utf8-locale
|
|
|
|
|
;; Some file names (e.g., in 'nss-certs') are UTF-8 encoded so
|
|
|
|
|
;; install a UTF-8 locale.
|
|
|
|
|
#~(begin
|
|
|
|
|
(setenv "LOCPATH"
|
|
|
|
|
#$(file-append glibc-utf8-locales "/lib/locale/"
|
2017-12-02 20:32:16 -05:00
|
|
|
|
(version-major+minor
|
|
|
|
|
(package-version glibc-utf8-locales))))
|
2016-12-17 06:43:10 -05:00
|
|
|
|
(setlocale LC_ALL "en_US.utf8")))
|
|
|
|
|
|
2014-08-23 12:41:14 -04:00
|
|
|
|
(define builder
|
2016-07-11 18:54:22 -04:00
|
|
|
|
(with-imported-modules '((guix build profiles)
|
|
|
|
|
(guix build union)
|
|
|
|
|
(guix build utils)
|
|
|
|
|
(guix search-paths)
|
|
|
|
|
(guix records))
|
|
|
|
|
#~(begin
|
|
|
|
|
(use-modules (guix build profiles)
|
|
|
|
|
(guix search-paths)
|
|
|
|
|
(srfi srfi-1))
|
|
|
|
|
|
2020-03-17 11:10:58 -04:00
|
|
|
|
(let ((line (cond-expand (guile-2.2 'line)
|
|
|
|
|
(else _IOLBF)))) ;Guile 2.0
|
|
|
|
|
(setvbuf (current-output-port) line)
|
|
|
|
|
(setvbuf (current-error-port) line))
|
2016-07-11 18:54:22 -04:00
|
|
|
|
|
2016-12-17 06:43:10 -05:00
|
|
|
|
#+(if locales? set-utf8-locale #t)
|
2016-12-16 12:01:08 -05:00
|
|
|
|
|
2016-07-11 18:54:22 -04:00
|
|
|
|
(define search-paths
|
|
|
|
|
;; Search paths of MANIFEST's packages, converted back to their
|
|
|
|
|
;; record form.
|
|
|
|
|
(map sexp->search-path-specification
|
|
|
|
|
(delete-duplicates
|
|
|
|
|
'#$(map search-path-specification->sexp
|
2018-07-09 04:44:36 -04:00
|
|
|
|
(manifest-search-paths manifest)))))
|
2016-07-11 18:54:22 -04:00
|
|
|
|
|
|
|
|
|
(build-profile #$output '#$inputs
|
2018-04-28 16:20:36 -04:00
|
|
|
|
#:symlink #$(if relative-symlinks?
|
|
|
|
|
#~symlink-relative
|
|
|
|
|
#~symlink)
|
2016-07-11 18:54:22 -04:00
|
|
|
|
#:manifest '#$(manifest->gexp manifest)
|
|
|
|
|
#:search-paths search-paths))))
|
2014-08-23 12:41:14 -04:00
|
|
|
|
|
2020-04-22 09:35:36 -04:00
|
|
|
|
(gexp->derivation name builder
|
2016-06-04 18:04:05 -04:00
|
|
|
|
#:system system
|
2017-03-17 16:48:40 -04:00
|
|
|
|
#:target target
|
2015-09-24 16:13:11 -04:00
|
|
|
|
|
2017-12-03 16:14:50 -05:00
|
|
|
|
;; Don't complain about _IO* on Guile 2.2.
|
|
|
|
|
#:env-vars '(("GUILE_WARN_DEPRECATED" . "no"))
|
|
|
|
|
|
2015-09-24 16:13:11 -04:00
|
|
|
|
;; Not worth offloading.
|
|
|
|
|
#:local-build? #t
|
|
|
|
|
|
|
|
|
|
;; Disable substitution because it would trigger a
|
|
|
|
|
;; connection to the substitute server, which is likely
|
|
|
|
|
;; to have no substitute to offer.
|
2020-03-26 07:25:37 -04:00
|
|
|
|
#:substitutable? #f
|
|
|
|
|
|
|
|
|
|
#:properties `((type . profile)
|
|
|
|
|
(profile
|
|
|
|
|
(count
|
|
|
|
|
. ,(length
|
|
|
|
|
(manifest-entries manifest))))))))
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
|
2020-04-22 09:43:43 -04:00
|
|
|
|
;; Declarative profile.
|
|
|
|
|
(define-record-type* <profile> profile make-profile
|
|
|
|
|
profile?
|
|
|
|
|
(name profile-name (default "profile")) ;string
|
|
|
|
|
(content profile-content) ;<manifest>
|
|
|
|
|
(hooks profile-hooks ;list of procedures
|
|
|
|
|
(default %default-profile-hooks))
|
|
|
|
|
(locales? profile-locales? ;Boolean
|
|
|
|
|
(default #t))
|
|
|
|
|
(allow-collisions? profile-allow-collisions? ;Boolean
|
|
|
|
|
(default #f))
|
|
|
|
|
(relative-symlinks? profile-relative-symlinks? ;Boolean
|
|
|
|
|
(default #f)))
|
|
|
|
|
|
|
|
|
|
(define-gexp-compiler (profile-compiler (profile <profile>) system target)
|
|
|
|
|
"Compile PROFILE to a derivation."
|
|
|
|
|
(match profile
|
|
|
|
|
(($ <profile> name manifest hooks
|
|
|
|
|
locales? allow-collisions? relative-symlinks?)
|
|
|
|
|
(profile-derivation manifest
|
|
|
|
|
#:name name
|
|
|
|
|
#:hooks hooks
|
|
|
|
|
#:locales? locales?
|
|
|
|
|
#:allow-collisions? allow-collisions?
|
|
|
|
|
#:relative-symlinks? relative-symlinks?
|
|
|
|
|
#:system system #:target target))))
|
|
|
|
|
|
2018-07-09 07:22:29 -04:00
|
|
|
|
(define* (profile-search-paths profile
|
|
|
|
|
#:optional (manifest (profile-manifest profile))
|
|
|
|
|
#:key (getenv (const #f)))
|
|
|
|
|
"Read the manifest of PROFILE and evaluate the values of search path
|
|
|
|
|
environment variables required by PROFILE; return a list of
|
|
|
|
|
specification/value pairs. If MANIFEST is not #f, it is assumed to be the
|
|
|
|
|
manifest of PROFILE, which avoids rereading it.
|
|
|
|
|
|
|
|
|
|
Use GETENV to determine the current settings and report only settings not
|
|
|
|
|
already effective."
|
|
|
|
|
(evaluate-search-paths (manifest-search-paths manifest)
|
|
|
|
|
(list profile) getenv))
|
|
|
|
|
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
(define (profile-regexp profile)
|
|
|
|
|
"Return a regular expression that matches PROFILE's name and number."
|
|
|
|
|
(make-regexp (string-append "^" (regexp-quote (basename profile))
|
|
|
|
|
"-([0-9]+)")))
|
|
|
|
|
|
|
|
|
|
(define (generation-number profile)
|
|
|
|
|
"Return PROFILE's number or 0. An absolute file name must be used."
|
|
|
|
|
(or (and=> (false-if-exception (regexp-exec (profile-regexp profile)
|
|
|
|
|
(basename (readlink profile))))
|
|
|
|
|
(compose string->number (cut match:substring <> 1)))
|
|
|
|
|
0))
|
|
|
|
|
|
2019-04-06 17:05:27 -04:00
|
|
|
|
(define %profile-generation-rx
|
|
|
|
|
;; Regexp that matches profile generation.
|
|
|
|
|
(make-regexp "(.*)-([0-9]+)-link$"))
|
|
|
|
|
|
|
|
|
|
(define (generation-profile file)
|
|
|
|
|
"If FILE is a profile generation GC root such as \"guix-profile-42-link\",
|
|
|
|
|
return its corresponding profile---e.g., \"guix-profile\". Otherwise return
|
|
|
|
|
#f."
|
|
|
|
|
(match (regexp-exec %profile-generation-rx file)
|
|
|
|
|
(#f #f)
|
|
|
|
|
(m (let ((profile (match:substring m 1)))
|
|
|
|
|
(and (file-exists? (string-append profile "/manifest"))
|
|
|
|
|
profile)))))
|
|
|
|
|
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
(define (generation-numbers profile)
|
|
|
|
|
"Return the sorted list of generation numbers of PROFILE, or '(0) if no
|
|
|
|
|
former profiles were found."
|
|
|
|
|
(match (scandir (dirname profile)
|
|
|
|
|
(cute regexp-exec (profile-regexp profile) <>))
|
|
|
|
|
(#f ; no profile directory
|
|
|
|
|
'(0))
|
|
|
|
|
(() ; no profiles
|
|
|
|
|
'(0))
|
|
|
|
|
((profiles ...) ; former profiles around
|
|
|
|
|
(sort (map (compose string->number
|
|
|
|
|
(cut match:substring <> 1)
|
|
|
|
|
(cute regexp-exec (profile-regexp profile) <>))
|
|
|
|
|
profiles)
|
|
|
|
|
<))))
|
|
|
|
|
|
2014-09-21 06:24:09 -04:00
|
|
|
|
(define (profile-generations profile)
|
|
|
|
|
"Return a list of PROFILE's generations."
|
|
|
|
|
(let ((generations (generation-numbers profile)))
|
|
|
|
|
(if (equal? generations '(0))
|
|
|
|
|
'()
|
|
|
|
|
generations)))
|
|
|
|
|
|
2016-11-02 01:48:11 -04:00
|
|
|
|
(define (relative-generation-spec->number profile spec)
|
|
|
|
|
"Return PROFILE's generation specified by SPEC, which is a string. The SPEC
|
|
|
|
|
may be a N, -N, or +N, where N is a number. If the spec is N, then the number
|
|
|
|
|
returned is N. If it is -N, then the number returned is the profile's current
|
|
|
|
|
generation number minus N. If it is +N, then the number returned is the
|
|
|
|
|
profile's current generation number plus N. Return #f if there is no such
|
|
|
|
|
generation."
|
|
|
|
|
(let ((number (string->number spec)))
|
|
|
|
|
(and number
|
|
|
|
|
(case (string-ref spec 0)
|
|
|
|
|
((#\+ #\-)
|
|
|
|
|
(relative-generation profile number))
|
|
|
|
|
(else (if (memv number (profile-generations profile))
|
|
|
|
|
number
|
|
|
|
|
#f))))))
|
|
|
|
|
|
|
|
|
|
|
2014-10-10 09:56:59 -04:00
|
|
|
|
(define* (relative-generation profile shift #:optional
|
|
|
|
|
(current (generation-number profile)))
|
|
|
|
|
"Return PROFILE's generation shifted from the CURRENT generation by SHIFT.
|
|
|
|
|
SHIFT is a positive or negative number.
|
|
|
|
|
Return #f if there is no such generation."
|
|
|
|
|
(let* ((abs-shift (abs shift))
|
|
|
|
|
(numbers (profile-generations profile))
|
|
|
|
|
(from-current (memq current
|
|
|
|
|
(if (negative? shift)
|
|
|
|
|
(reverse numbers)
|
|
|
|
|
numbers))))
|
|
|
|
|
(and from-current
|
|
|
|
|
(< abs-shift (length from-current))
|
|
|
|
|
(list-ref from-current abs-shift))))
|
|
|
|
|
|
|
|
|
|
(define* (previous-generation-number profile #:optional
|
|
|
|
|
(number (generation-number profile)))
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
"Return the number of the generation before generation NUMBER of
|
|
|
|
|
PROFILE, or 0 if none exists. It could be NUMBER - 1, but it's not the
|
|
|
|
|
case when generations have been deleted (there are \"holes\")."
|
2014-10-10 09:56:59 -04:00
|
|
|
|
(or (relative-generation profile -1 number)
|
|
|
|
|
0))
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
|
|
|
|
|
(define (generation-file-name profile generation)
|
|
|
|
|
"Return the file name for PROFILE's GENERATION."
|
|
|
|
|
(format #f "~a-~a-link" profile generation))
|
|
|
|
|
|
|
|
|
|
(define (generation-time profile number)
|
|
|
|
|
"Return the creation time of a generation in the UTC format."
|
|
|
|
|
(make-time time-utc 0
|
|
|
|
|
(stat:ctime (stat (generation-file-name profile number)))))
|
|
|
|
|
|
2015-10-26 18:01:06 -04:00
|
|
|
|
(define (link-to-empty-profile store generation)
|
|
|
|
|
"Link GENERATION, a string, to the empty profile. An error is raised if
|
|
|
|
|
that fails."
|
|
|
|
|
(let* ((drv (run-with-store store
|
2016-12-17 06:43:10 -05:00
|
|
|
|
(profile-derivation (manifest '())
|
|
|
|
|
#:locales? #f)))
|
2015-10-26 18:01:06 -04:00
|
|
|
|
(prof (derivation->output-path drv "out")))
|
|
|
|
|
(build-derivations store (list drv))
|
|
|
|
|
(switch-symlinks generation prof)))
|
|
|
|
|
|
|
|
|
|
(define (switch-to-generation profile number)
|
|
|
|
|
"Atomically switch PROFILE to the generation NUMBER. Return the number of
|
|
|
|
|
the generation that was current before switching."
|
|
|
|
|
(let ((current (generation-number profile))
|
|
|
|
|
(generation (generation-file-name profile number)))
|
|
|
|
|
(cond ((not (file-exists? profile))
|
|
|
|
|
(raise (condition (&profile-not-found-error
|
|
|
|
|
(profile profile)))))
|
|
|
|
|
((not (file-exists? generation))
|
|
|
|
|
(raise (condition (&missing-generation-error
|
|
|
|
|
(profile profile)
|
|
|
|
|
(generation number)))))
|
|
|
|
|
(else
|
2018-07-13 08:33:11 -04:00
|
|
|
|
(switch-symlinks profile (basename generation))
|
2015-10-26 18:01:06 -04:00
|
|
|
|
current))))
|
|
|
|
|
|
|
|
|
|
(define (switch-to-previous-generation profile)
|
|
|
|
|
"Atomically switch PROFILE to the previous generation. Return the former
|
|
|
|
|
generation number and the current one."
|
|
|
|
|
(let ((previous (previous-generation-number profile)))
|
|
|
|
|
(values (switch-to-generation profile previous)
|
|
|
|
|
previous)))
|
|
|
|
|
|
|
|
|
|
(define (roll-back store profile)
|
|
|
|
|
"Roll back to the previous generation of PROFILE. Return the number of the
|
|
|
|
|
generation that was current before switching and the new generation number."
|
|
|
|
|
(let* ((number (generation-number profile))
|
|
|
|
|
(previous-number (previous-generation-number profile number))
|
|
|
|
|
(previous-generation (generation-file-name profile previous-number)))
|
|
|
|
|
(cond ((not (file-exists? profile)) ;invalid profile
|
|
|
|
|
(raise (condition (&profile-not-found-error
|
|
|
|
|
(profile profile)))))
|
|
|
|
|
((zero? number) ;empty profile
|
|
|
|
|
(values number number))
|
|
|
|
|
((or (zero? previous-number) ;going to emptiness
|
|
|
|
|
(not (file-exists? previous-generation)))
|
|
|
|
|
(link-to-empty-profile store previous-generation)
|
|
|
|
|
(switch-to-previous-generation profile))
|
|
|
|
|
(else ;anything else
|
|
|
|
|
(switch-to-previous-generation profile)))))
|
|
|
|
|
|
|
|
|
|
(define (delete-generation store profile number)
|
|
|
|
|
"Delete generation with NUMBER from PROFILE. Return the file name of the
|
|
|
|
|
generation that has been deleted, or #f if nothing was done (for instance
|
|
|
|
|
because the NUMBER is zero.)"
|
|
|
|
|
(define (delete-and-return)
|
|
|
|
|
(let ((generation (generation-file-name profile number)))
|
|
|
|
|
(delete-file generation)
|
|
|
|
|
generation))
|
|
|
|
|
|
|
|
|
|
(let* ((current-number (generation-number profile))
|
|
|
|
|
(previous-number (previous-generation-number profile number))
|
|
|
|
|
(previous-generation (generation-file-name profile previous-number)))
|
|
|
|
|
(cond ((zero? number) #f) ;do not delete generation 0
|
|
|
|
|
((and (= number current-number)
|
|
|
|
|
(not (file-exists? previous-generation)))
|
|
|
|
|
(link-to-empty-profile store previous-generation)
|
|
|
|
|
(switch-to-previous-generation profile)
|
|
|
|
|
(delete-and-return))
|
|
|
|
|
((= number current-number)
|
|
|
|
|
(roll-back store profile)
|
|
|
|
|
(delete-and-return))
|
|
|
|
|
(else
|
|
|
|
|
(delete-and-return)))))
|
|
|
|
|
|
2018-05-13 10:08:24 -04:00
|
|
|
|
(define %user-profile-directory
|
|
|
|
|
(and=> (getenv "HOME")
|
|
|
|
|
(cut string-append <> "/.guix-profile")))
|
|
|
|
|
|
|
|
|
|
(define %profile-directory
|
|
|
|
|
(string-append %state-directory "/profiles/"
|
|
|
|
|
(or (and=> (or (getenv "USER")
|
2019-12-10 11:21:31 -05:00
|
|
|
|
(getenv "LOGNAME")
|
|
|
|
|
(false-if-exception
|
|
|
|
|
(passwd:name (getpwuid (getuid)))))
|
2018-05-13 10:08:24 -04:00
|
|
|
|
(cut string-append "per-user/" <>))
|
|
|
|
|
"default")))
|
|
|
|
|
|
|
|
|
|
(define %current-profile
|
|
|
|
|
;; Call it `guix-profile', not `profile', to allow Guix profiles to
|
|
|
|
|
;; coexist with Nix profiles.
|
|
|
|
|
(string-append %profile-directory "/guix-profile"))
|
|
|
|
|
|
2018-10-11 12:04:51 -04:00
|
|
|
|
(define (ensure-profile-directory)
|
2019-10-16 05:51:42 -04:00
|
|
|
|
"Attempt to create /…/profiles/per-user/$USER if needed. Nowadays this is
|
|
|
|
|
taken care of by the daemon."
|
2018-10-11 12:04:51 -04:00
|
|
|
|
(let ((s (stat %profile-directory #f)))
|
|
|
|
|
(unless (and s (eq? 'directory (stat:type s)))
|
|
|
|
|
(catch 'system-error
|
|
|
|
|
(lambda ()
|
|
|
|
|
(mkdir-p %profile-directory))
|
|
|
|
|
(lambda args
|
|
|
|
|
;; Often, we cannot create %PROFILE-DIRECTORY because its
|
|
|
|
|
;; parent directory is root-owned and we're running
|
|
|
|
|
;; unprivileged.
|
|
|
|
|
(raise (condition
|
|
|
|
|
(&message
|
|
|
|
|
(message
|
|
|
|
|
(format #f
|
|
|
|
|
(G_ "while creating directory `~a': ~a")
|
|
|
|
|
%profile-directory
|
|
|
|
|
(strerror (system-error-errno args)))))
|
|
|
|
|
(&fix-hint
|
|
|
|
|
(hint
|
|
|
|
|
(format #f (G_ "Please create the @file{~a} directory, \
|
|
|
|
|
with you as the owner.")
|
|
|
|
|
%profile-directory))))))))
|
|
|
|
|
|
|
|
|
|
;; Bail out if it's not owned by the user.
|
|
|
|
|
(unless (or (not s) (= (stat:uid s) (getuid)))
|
|
|
|
|
(raise (condition
|
|
|
|
|
(&message
|
|
|
|
|
(message
|
|
|
|
|
(format #f (G_ "directory `~a' is not owned by you")
|
|
|
|
|
%profile-directory)))
|
|
|
|
|
(&fix-hint
|
|
|
|
|
(hint
|
|
|
|
|
(format #f (G_ "Please change the owner of @file{~a} \
|
|
|
|
|
to user ~s.")
|
|
|
|
|
%profile-directory (or (getenv "USER")
|
|
|
|
|
(getenv "LOGNAME")
|
|
|
|
|
(getuid))))))))))
|
|
|
|
|
|
2018-05-13 10:08:24 -04:00
|
|
|
|
(define (canonicalize-profile profile)
|
2018-10-09 05:51:12 -04:00
|
|
|
|
"If PROFILE points to a profile in %PROFILE-DIRECTORY, return that.
|
|
|
|
|
Otherwise return PROFILE unchanged. The goal is to treat '-p ~/.guix-profile'
|
|
|
|
|
as if '-p' was omitted." ; see <http://bugs.gnu.org/17939>
|
|
|
|
|
;; Trim trailing slashes so 'readlink' can do its job.
|
2018-05-13 10:08:24 -04:00
|
|
|
|
(let ((profile (string-trim-right profile #\/)))
|
2018-10-09 05:51:12 -04:00
|
|
|
|
(catch 'system-error
|
|
|
|
|
(lambda ()
|
|
|
|
|
(let ((target (readlink profile)))
|
|
|
|
|
(if (string=? (dirname target) %profile-directory)
|
|
|
|
|
target
|
|
|
|
|
profile)))
|
|
|
|
|
(const profile))))
|
2018-05-13 10:08:24 -04:00
|
|
|
|
|
2018-10-11 18:12:00 -04:00
|
|
|
|
(define %known-shorthand-profiles
|
|
|
|
|
;; Known shorthand forms for profiles that the user manipulates.
|
|
|
|
|
(list (string-append (config-directory #:ensure? #f) "/current")
|
|
|
|
|
%user-profile-directory))
|
|
|
|
|
|
2018-05-13 10:08:24 -04:00
|
|
|
|
(define (user-friendly-profile profile)
|
2018-10-11 18:12:00 -04:00
|
|
|
|
"Return either ~/.guix-profile or ~/.config/guix/current if that's what
|
|
|
|
|
PROFILE refers to, directly or indirectly, or PROFILE."
|
|
|
|
|
(or (find (lambda (shorthand)
|
|
|
|
|
(and shorthand
|
|
|
|
|
(let ((target (false-if-exception
|
|
|
|
|
(readlink shorthand))))
|
|
|
|
|
(and target (string=? target profile)))))
|
|
|
|
|
%known-shorthand-profiles)
|
2018-05-13 10:08:24 -04:00
|
|
|
|
profile))
|
|
|
|
|
|
Add (guix profiles).
* guix/scripts/package.scm (show-what-to-remove/install): New procedure,
moved from...
(guix-package): ... here.
(<manifest>, make-manifest, <manifest-entry>,
profile-manifest, manifest->sexp, sexp->manifest, read-manifest,
write-manifest, remove-manifest-entry, manifest-remove,
manifest-installed?, manifest=?, profile-regexp, generation-numbers,
previous-generation-number, profile-derivation, generation-number,
generation-file-name, generation-time, lower-input): Move to...
* guix/profiles.scm: ... here. New file.
* Makefile.am (MODULES): Add it.
2013-11-01 11:31:45 -04:00
|
|
|
|
;;; profiles.scm ends here
|