123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241 |
- ;;; GNU Guix --- Functional package management for GNU
- ;;; Copyright © 2017, 2020 Ricardo Wurmus <rekado@elephly.net>
- ;;; Copyright © 2021 Ludovic Courtès <ludo@gnu.org>
- ;;;
- ;;; 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 import print)
- #:use-module (guix base32)
- #:use-module (guix utils)
- #:use-module (guix licenses)
- #:use-module (guix packages)
- #:use-module (guix search-paths)
- #:use-module (guix build-system)
- #:use-module (gnu packages)
- #:use-module (srfi srfi-1)
- #:use-module (srfi srfi-26)
- #:use-module (guix import utils)
- #:use-module (ice-9 control)
- #:use-module (ice-9 match)
- #:export (package->code))
- (define (redundant-input-labels? inputs)
- "Return #t if input labels in the INPUTS list are redundant."
- (every (match-lambda
- ((label (? package? package) . _)
- (string=? label (package-name package)))
- (_ #f))
- inputs))
- (define (package->code package)
- "Return an S-expression representing the source code that produces PACKAGE
- when evaluated."
- ;; The module in which the package PKG is defined
- (define (package-module-name pkg)
- (map string->symbol
- (string-split (string-drop-right
- (location-file (package-location pkg)) 4)
- #\/)))
- ;; Return the first candidate variable name that is bound to VAL.
- (define (variable-name val mod)
- (match (let/ec return
- (module-for-each (lambda (sym var)
- (if (eq? val (variable-ref var))
- (return sym)
- #f))
- (resolve-interface mod)))
- ((? symbol? sym) sym)
- (_ #f)))
- ;; Print either license variable name or the code for a license object
- (define (license->code lic)
- (let ((var (variable-name lic '(guix licenses))))
- (if var
- (symbol-append 'license: var)
- `(license
- ,(license-name lic)
- ,(license-uri lic)
- ,(license-comment lic)))))
- (define (search-path-specification->code spec)
- `(search-path-specification
- (variable ,(search-path-specification-variable spec))
- (files (list ,@(search-path-specification-files spec)))
- (separator ,(search-path-specification-separator spec))
- (file-type (quote ,(search-path-specification-file-type spec)))
- (file-pattern ,(search-path-specification-file-pattern spec))))
- (define (factorized-uri-code uri version)
- (match (factorize-uri uri version)
- ((? string? uri) uri)
- ((factorized ...) `(string-append ,@factorized))))
- (define (source->code source version)
- (let ((uri (origin-uri source))
- (method (origin-method source))
- (hash (origin-hash source))
- (file-name (origin-file-name source))
- (patches (origin-patches source)))
- `(origin
- ;; Since 'procedure-name' returns the procedure name within the
- ;; module where it's defined, not its public name. Thus, try hard to
- ;; find its public name and use 'procedure-name' as a last resort.
- (method ,(or (any (lambda (module)
- (variable-name method module))
- '((guix download)
- (guix git-download)
- (guix hg-download)
- (guix svn-download)))
- (procedure-name method)))
- (uri ,(if version
- (match uri
- ((? string? uri)
- (factorized-uri-code uri version))
- ((lst ...)
- `(list
- ,@(map (cut factorized-uri-code <> version) uri))))
- uri))
- ,(if (equal? (content-hash-algorithm hash) 'sha256)
- `(sha256 (base32 ,(bytevector->nix-base32-string
- (content-hash-value hash))))
- `(hash (content-hash ,(bytevector->nix-base32-string
- (content-hash-value hash))
- ,(content-hash-algorithm hash))))
- ;; FIXME: in order to be able to throw away the directory prefix,
- ;; we just assume that the patch files can be found with
- ;; "search-patches".
- ,@(cond ((null? patches)
- '())
- ((every string? patches)
- `((patches (search-patches ,@(map basename patches)))))
- (else
- `((patches (list ,@(map (match-lambda
- ((? string? file)
- `(search-patch ,file))
- ((? origin? origin)
- (source->code origin #f)))
- patches)))))))))
- (define (variable-reference module name)
- ;; FIXME: using '@ certainly isn't pretty, but it avoids having to import
- ;; the individual package modules.
- (list '@ module name))
- (define (object->code obj quoted?)
- (match obj
- ((? package? package)
- (let* ((module (package-module-name package))
- (name (variable-name package module)))
- (if quoted?
- (list 'unquote (variable-reference module name))
- (variable-reference module name))))
- ((? origin? origin)
- (let ((code (source->code origin #f)))
- (if quoted?
- (list 'unquote code)
- code)))
- ((lst ...)
- (let ((lst (map (cut object->code <> #t) lst)))
- (if quoted?
- lst
- (list 'quasiquote lst))))
- (obj
- obj)))
- (define (inputs->code inputs)
- (if (redundant-input-labels? inputs)
- `(list ,@(map (match-lambda ;no need for input labels ("new style")
- ((_ package)
- (let* ((module (package-module-name package))
- (name (variable-name package module)))
- (variable-reference module name)))
- ((_ package output)
- (let* ((module (package-module-name package))
- (name (variable-name package module)))
- (list 'quasiquote
- (list
- (list 'unquote
- (variable-reference module name))
- output)))))
- inputs))
- (list 'quasiquote ;preserve input labels (deprecated)
- (object->code inputs #t))))
- (let ((name (package-name package))
- (version (package-version package))
- (source (package-source package))
- (build-system (package-build-system package))
- (arguments (package-arguments package))
- (inputs (package-inputs package))
- (propagated-inputs (package-propagated-inputs package))
- (native-inputs (package-native-inputs package))
- (outputs (package-outputs package))
- (native-search-paths (package-native-search-paths package))
- (search-paths (package-search-paths package))
- (replacement (package-replacement package))
- (synopsis (package-synopsis package))
- (description (package-description package))
- (license (package-license package))
- (home-page (package-home-page package))
- (supported-systems (package-supported-systems package))
- (properties (package-properties package)))
- `(define-public ,(string->symbol name)
- (package
- (name ,name)
- (version ,version)
- (source ,(source->code source version))
- ,@(match properties
- (() '())
- (_ `((properties ,properties))))
- ,@(if replacement
- `((replacement ,replacement))
- '())
- (build-system (@ (guix build-system ,(build-system-name build-system))
- ,(symbol-append (build-system-name build-system)
- '-build-system)))
- ,@(match arguments
- (() '())
- (_ `((arguments
- ,(list 'quasiquote (object->code arguments #t))))))
- ,@(match outputs
- (("out") '())
- (outs `((outputs (list ,@outs)))))
- ,@(match native-inputs
- (() '())
- (pkgs `((native-inputs ,(inputs->code pkgs)))))
- ,@(match inputs
- (() '())
- (pkgs `((inputs ,(inputs->code pkgs)))))
- ,@(match propagated-inputs
- (() '())
- (pkgs `((propagated-inputs ,(inputs->code pkgs)))))
- ,@(if (lset= string=? supported-systems %supported-systems)
- '()
- `((supported-systems (list ,@supported-systems))))
- ,@(match (map search-path-specification->code native-search-paths)
- (() '())
- (paths `((native-search-paths (list ,@paths)))))
- ,@(match (map search-path-specification->code search-paths)
- (() '())
- (paths `((search-paths (list ,@paths)))))
- (home-page ,home-page)
- (synopsis ,synopsis)
- (description ,description)
- (license ,(if (list? license)
- `(list ,@(map license->code license))
- (license->code license)))))))
|