2012-06-28 01:24:34 +02:00
|
|
|
|
;;; Guix --- Nix package management from Guile. -*- coding: utf-8 -*-
|
|
|
|
|
;;; Copyright (C) 2012 Ludovic Courtès <ludo@gnu.org>
|
|
|
|
|
;;;
|
|
|
|
|
;;; This file is part of Guix.
|
|
|
|
|
;;;
|
|
|
|
|
;;; 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.
|
|
|
|
|
;;;
|
|
|
|
|
;;; 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 Guix. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
|
|
(define-module (guix packages)
|
|
|
|
|
#:use-module (guix utils)
|
|
|
|
|
#:use-module (guix store)
|
|
|
|
|
#:use-module (guix build-system)
|
|
|
|
|
#:use-module (ice-9 match)
|
2012-07-01 17:32:03 +02:00
|
|
|
|
#:use-module (srfi srfi-1)
|
2012-09-01 19:21:06 +02:00
|
|
|
|
#:use-module (srfi srfi-9 gnu)
|
2012-08-23 23:09:13 +02:00
|
|
|
|
#:use-module (srfi srfi-34)
|
|
|
|
|
#:use-module (srfi srfi-35)
|
2012-09-01 19:21:06 +02:00
|
|
|
|
#:export (origin
|
2012-07-01 17:32:03 +02:00
|
|
|
|
origin?
|
|
|
|
|
origin-uri
|
|
|
|
|
origin-method
|
|
|
|
|
origin-sha256
|
|
|
|
|
origin-file-name
|
2012-07-01 17:32:03 +02:00
|
|
|
|
base32
|
2012-06-28 01:24:34 +02:00
|
|
|
|
|
|
|
|
|
package
|
|
|
|
|
package?
|
|
|
|
|
package-name
|
|
|
|
|
package-version
|
2012-09-05 19:01:47 +02:00
|
|
|
|
package-full-name
|
2012-06-28 01:24:34 +02:00
|
|
|
|
package-source
|
|
|
|
|
package-build-system
|
|
|
|
|
package-arguments
|
|
|
|
|
package-inputs
|
|
|
|
|
package-native-inputs
|
2012-07-01 17:32:03 +02:00
|
|
|
|
package-propagated-inputs
|
2012-06-28 01:24:34 +02:00
|
|
|
|
package-outputs
|
|
|
|
|
package-search-paths
|
|
|
|
|
package-description
|
|
|
|
|
package-long-description
|
|
|
|
|
package-license
|
|
|
|
|
package-platforms
|
|
|
|
|
package-maintainers
|
2012-07-01 17:32:03 +02:00
|
|
|
|
package-properties
|
2012-06-28 23:15:24 +02:00
|
|
|
|
package-location
|
2012-06-28 01:24:34 +02:00
|
|
|
|
|
2012-07-07 20:14:20 +02:00
|
|
|
|
package-transitive-inputs
|
2012-09-01 19:21:06 +02:00
|
|
|
|
package-transitive-propagated-inputs
|
2012-06-28 01:24:34 +02:00
|
|
|
|
package-source-derivation
|
|
|
|
|
package-derivation
|
2012-08-23 23:09:13 +02:00
|
|
|
|
package-cross-derivation
|
|
|
|
|
|
|
|
|
|
&package-error
|
|
|
|
|
package-error-package
|
|
|
|
|
&package-input-error
|
|
|
|
|
package-error-invalid-input))
|
2012-06-28 01:24:34 +02:00
|
|
|
|
|
|
|
|
|
;;; Commentary:
|
|
|
|
|
;;;
|
|
|
|
|
;;; This module provides a high-level mechanism to define packages in a
|
|
|
|
|
;;; Guix-based distribution.
|
|
|
|
|
;;;
|
|
|
|
|
;;; Code:
|
|
|
|
|
|
2012-07-01 17:32:03 +02:00
|
|
|
|
;; The source of a package, such as a tarball URL and fetcher---called
|
|
|
|
|
;; "origin" to avoid name clash with `package-source', `source', etc.
|
|
|
|
|
(define-record-type* <origin>
|
|
|
|
|
origin make-origin
|
|
|
|
|
origin?
|
|
|
|
|
(uri origin-uri) ; string
|
|
|
|
|
(method origin-method) ; symbol
|
|
|
|
|
(sha256 origin-sha256) ; bytevector
|
|
|
|
|
(file-name origin-file-name (default #f))) ; optional file name
|
2012-06-28 01:24:34 +02:00
|
|
|
|
|
2012-07-01 17:32:03 +02:00
|
|
|
|
(define-syntax base32
|
|
|
|
|
(lambda (s)
|
|
|
|
|
"Return the bytevector corresponding to the given Nix-base32
|
|
|
|
|
representation."
|
|
|
|
|
(syntax-case s ()
|
|
|
|
|
((_ str)
|
|
|
|
|
(string? (syntax->datum #'str))
|
|
|
|
|
(with-syntax ((bv (nix-base32-string->bytevector
|
|
|
|
|
(syntax->datum #'str))))
|
|
|
|
|
#''bv)))))
|
|
|
|
|
|
2012-06-28 23:15:24 +02:00
|
|
|
|
;; A package.
|
2012-08-23 23:09:13 +02:00
|
|
|
|
|
2012-06-28 01:24:34 +02:00
|
|
|
|
(define-record-type* <package>
|
|
|
|
|
package make-package
|
|
|
|
|
package?
|
|
|
|
|
(name package-name) ; string
|
|
|
|
|
(version package-version) ; string
|
2012-07-01 17:32:03 +02:00
|
|
|
|
(source package-source) ; <origin> instance
|
2012-06-28 01:24:34 +02:00
|
|
|
|
(build-system package-build-system) ; build system
|
2012-06-29 00:28:57 +02:00
|
|
|
|
(arguments package-arguments ; arguments for the build method
|
|
|
|
|
(default '()))
|
2012-07-01 17:32:03 +02:00
|
|
|
|
|
2012-06-28 01:24:34 +02:00
|
|
|
|
(inputs package-inputs ; input packages or derivations
|
|
|
|
|
(default '()))
|
2012-07-01 17:32:03 +02:00
|
|
|
|
(propagated-inputs package-propagated-inputs ; same, but propagated
|
|
|
|
|
(default '()))
|
2012-06-28 01:24:34 +02:00
|
|
|
|
(native-inputs package-native-inputs ; native input packages/derivations
|
|
|
|
|
(default '()))
|
2012-07-01 17:32:03 +02:00
|
|
|
|
(self-native-input? package-self-native-input? ; whether to use itself as
|
|
|
|
|
; a native input when cross-
|
|
|
|
|
(default #f)) ; compiling
|
2012-07-01 17:32:03 +02:00
|
|
|
|
|
2012-06-28 01:24:34 +02:00
|
|
|
|
(outputs package-outputs ; list of strings
|
|
|
|
|
(default '("out")))
|
|
|
|
|
(search-paths package-search-paths ; list of (ENV-VAR (DIRS ...))
|
|
|
|
|
(default '())) ; tuples; see
|
|
|
|
|
; `set-path-environment-variable'
|
|
|
|
|
; (aka. "setup-hook")
|
|
|
|
|
|
|
|
|
|
(description package-description) ; one-line description
|
|
|
|
|
(long-description package-long-description) ; one or two paragraphs
|
|
|
|
|
(license package-license (default '()))
|
2012-06-28 23:44:43 +02:00
|
|
|
|
(home-page package-home-page)
|
2012-06-28 01:24:34 +02:00
|
|
|
|
(platforms package-platforms (default '()))
|
2012-06-28 23:15:24 +02:00
|
|
|
|
(maintainers package-maintainers (default '()))
|
2012-06-28 23:44:43 +02:00
|
|
|
|
|
2012-07-01 17:32:03 +02:00
|
|
|
|
(properties package-properties (default '())) ; alist for anything else
|
|
|
|
|
|
2012-06-28 23:15:24 +02:00
|
|
|
|
(location package-location
|
|
|
|
|
(default (and=> (current-source-location)
|
|
|
|
|
source-properties->location))))
|
2012-06-28 01:24:34 +02:00
|
|
|
|
|
2012-09-01 19:21:06 +02:00
|
|
|
|
(set-record-type-printer! <package>
|
|
|
|
|
(lambda (package port)
|
|
|
|
|
(let ((loc (package-location package))
|
|
|
|
|
(format simple-format))
|
|
|
|
|
(format port "#<package ~a-~a ~a:~a ~a>"
|
|
|
|
|
(package-name package)
|
|
|
|
|
(package-version package)
|
|
|
|
|
(location-file loc)
|
|
|
|
|
(location-line loc)
|
|
|
|
|
(number->string (object-address
|
|
|
|
|
package)
|
|
|
|
|
16)))))
|
|
|
|
|
|
2012-08-23 23:09:13 +02:00
|
|
|
|
|
|
|
|
|
;; Error conditions.
|
|
|
|
|
|
|
|
|
|
(define-condition-type &package-error &error
|
|
|
|
|
package-error?
|
|
|
|
|
(package package-error-package))
|
|
|
|
|
|
|
|
|
|
(define-condition-type &package-input-error &package-error
|
|
|
|
|
package-input-error?
|
|
|
|
|
(input package-error-invalid-input))
|
|
|
|
|
|
|
|
|
|
|
2012-09-05 19:01:47 +02:00
|
|
|
|
(define (package-full-name package)
|
|
|
|
|
"Return the full name of PACKAGE--i.e., `NAME-VERSION'."
|
|
|
|
|
(string-append (package-name package) "-" (package-version package)))
|
|
|
|
|
|
2012-06-28 01:24:34 +02:00
|
|
|
|
(define (package-source-derivation store source)
|
|
|
|
|
"Return the derivation path for SOURCE, a package source."
|
|
|
|
|
(match source
|
2012-07-01 17:32:03 +02:00
|
|
|
|
(($ <origin> uri method sha256 name)
|
2012-06-28 01:24:34 +02:00
|
|
|
|
(method store uri 'sha256 sha256 name))))
|
|
|
|
|
|
2012-09-01 19:21:06 +02:00
|
|
|
|
(define (transitive-inputs inputs)
|
|
|
|
|
(let loop ((inputs inputs)
|
2012-07-07 20:14:20 +02:00
|
|
|
|
(result '()))
|
|
|
|
|
(match inputs
|
|
|
|
|
(()
|
|
|
|
|
(delete-duplicates (reverse result))) ; XXX: efficiency
|
|
|
|
|
(((and i (name (? package? p) sub ...)) rest ...)
|
|
|
|
|
(let ((t (map (match-lambda
|
|
|
|
|
((dep-name derivation ...)
|
|
|
|
|
(cons (string-append name "/" dep-name)
|
|
|
|
|
derivation)))
|
|
|
|
|
(package-propagated-inputs p))))
|
|
|
|
|
(loop (append t rest)
|
|
|
|
|
(append t (cons i result)))))
|
|
|
|
|
((input rest ...)
|
|
|
|
|
(loop rest (cons input result))))))
|
|
|
|
|
|
2012-09-01 19:21:06 +02:00
|
|
|
|
(define (package-transitive-inputs package)
|
|
|
|
|
"Return the transitive inputs of PACKAGE---i.e., its direct inputs along
|
|
|
|
|
with their propagated inputs, recursively."
|
|
|
|
|
(transitive-inputs (append (package-native-inputs package)
|
|
|
|
|
(package-inputs package)
|
|
|
|
|
(package-propagated-inputs package))))
|
|
|
|
|
|
|
|
|
|
(define (package-transitive-propagated-inputs package)
|
|
|
|
|
"Return the propagated inputs of PACKAGE, and their propagated inputs,
|
|
|
|
|
recursively."
|
|
|
|
|
(transitive-inputs (package-propagated-inputs package)))
|
|
|
|
|
|
2012-09-01 19:21:06 +02:00
|
|
|
|
|
|
|
|
|
;;;
|
|
|
|
|
;;; Package derivations.
|
|
|
|
|
;;;
|
|
|
|
|
|
|
|
|
|
(define %derivation-cache
|
|
|
|
|
;; Package to derivation-path mapping.
|
2012-10-08 22:07:19 +02:00
|
|
|
|
(make-weak-key-hash-table 100))
|
2012-09-01 19:21:06 +02:00
|
|
|
|
|
|
|
|
|
(define (cache package system drv)
|
|
|
|
|
"Memoize DRV as the derivation of PACKAGE on SYSTEM."
|
2012-10-08 22:07:19 +02:00
|
|
|
|
|
|
|
|
|
;; Use `hashq-set!' instead of `hash-set!' because `hash' returns the
|
|
|
|
|
;; same value for all structs (as of Guile 2.0.6), and because pointer
|
|
|
|
|
;; equality is sufficient in practice.
|
|
|
|
|
(hashq-set! %derivation-cache package `((,system . ,drv)))
|
2012-09-01 19:21:06 +02:00
|
|
|
|
drv)
|
|
|
|
|
|
|
|
|
|
(define (cached-derivation package system)
|
|
|
|
|
"Return the cached derivation path of PACKAGE for SYSTEM, or #f."
|
2012-10-08 22:07:19 +02:00
|
|
|
|
(match (hashq-ref %derivation-cache package)
|
|
|
|
|
((alist ...)
|
|
|
|
|
(assoc-ref alist system))
|
|
|
|
|
(#f #f)))
|
2012-09-01 19:21:06 +02:00
|
|
|
|
|
2012-06-28 01:24:34 +02:00
|
|
|
|
(define* (package-derivation store package
|
|
|
|
|
#:optional (system (%current-system)))
|
|
|
|
|
"Return the derivation of PACKAGE for SYSTEM."
|
2012-10-21 16:57:50 +02:00
|
|
|
|
(define (intern file)
|
|
|
|
|
;; Add FILE to the store. Set the `recursive?' bit to #t, so that
|
|
|
|
|
;; file permissions are preserved.
|
|
|
|
|
(add-to-store store (basename file)
|
|
|
|
|
#t #t "sha256" file))
|
|
|
|
|
|
|
|
|
|
(define expand-input
|
|
|
|
|
;; Expand the given input tuple such that it contains only
|
|
|
|
|
;; references to derivation paths or store paths.
|
|
|
|
|
(match-lambda
|
|
|
|
|
(((? string? name) (? package? package))
|
|
|
|
|
(list name (package-derivation store package)))
|
|
|
|
|
(((? string? name) (? package? package)
|
|
|
|
|
(? string? sub-drv))
|
|
|
|
|
(list name (package-derivation store package)
|
|
|
|
|
sub-drv))
|
|
|
|
|
(((? string? name)
|
|
|
|
|
(and (? string?) (? derivation-path?) drv))
|
|
|
|
|
(list name drv))
|
|
|
|
|
(((? string? name)
|
|
|
|
|
(and (? string?) (? file-exists? file)))
|
|
|
|
|
;; Add FILE to the store. When FILE is in the sub-directory of a
|
|
|
|
|
;; store path, it needs to be added anyway, so it can be used as a
|
|
|
|
|
;; source.
|
|
|
|
|
(list name (intern file)))
|
|
|
|
|
(((? string? name) (? origin? source))
|
|
|
|
|
(list name (package-source-derivation store source)))
|
|
|
|
|
((and i ((? string? name) (? procedure? proc) sub-drv ...))
|
|
|
|
|
;; This form allows PROC to make a SYSTEM-dependent choice.
|
|
|
|
|
|
|
|
|
|
;; XXX: Currently PROC must return a .drv, a store path, a local
|
|
|
|
|
;; file name, or an <origin>. If it were allowed to return a
|
|
|
|
|
;; package, then `transitive-inputs' and co. would need to be
|
|
|
|
|
;; adjusted.
|
|
|
|
|
(let ((input (proc system)))
|
|
|
|
|
(if (or (string? input) (origin? input))
|
|
|
|
|
(expand-input (cons* name input sub-drv))
|
|
|
|
|
(raise (condition (&package-input-error
|
|
|
|
|
(package package)
|
|
|
|
|
(input i)))))))
|
|
|
|
|
(x
|
|
|
|
|
(raise (condition (&package-input-error
|
|
|
|
|
(package package)
|
|
|
|
|
(input x)))))))
|
|
|
|
|
|
2012-09-01 19:21:06 +02:00
|
|
|
|
(or (cached-derivation package system)
|
2012-10-08 21:57:14 +02:00
|
|
|
|
|
|
|
|
|
;; Compute the derivation and cache the result. Caching is
|
|
|
|
|
;; important because some derivations, such as the implicit inputs
|
|
|
|
|
;; of the GNU build system, will be queried many, many times in a
|
|
|
|
|
;; row.
|
|
|
|
|
(cache
|
|
|
|
|
package system
|
|
|
|
|
(match package
|
|
|
|
|
(($ <package> name version source (= build-system-builder builder)
|
|
|
|
|
args inputs propagated-inputs native-inputs self-native-input?
|
|
|
|
|
outputs)
|
|
|
|
|
;; TODO: For `search-paths', add a builder prologue that calls
|
|
|
|
|
;; `set-path-environment-variable'.
|
2012-10-21 16:57:50 +02:00
|
|
|
|
(let ((inputs (map expand-input
|
2012-10-08 21:57:14 +02:00
|
|
|
|
(package-transitive-inputs package))))
|
|
|
|
|
|
|
|
|
|
(apply builder
|
|
|
|
|
store (package-full-name package)
|
|
|
|
|
(and source (package-source-derivation store source))
|
|
|
|
|
inputs
|
|
|
|
|
#:outputs outputs #:system system
|
|
|
|
|
(if (procedure? args)
|
|
|
|
|
(args system)
|
|
|
|
|
args))))))))
|
2012-06-28 01:24:34 +02:00
|
|
|
|
|
|
|
|
|
(define* (package-cross-derivation store package)
|
|
|
|
|
;; TODO
|
|
|
|
|
#f)
|