2015-07-24 16:49:57 +02:00
|
|
|
|
;;; GNU Guix --- Functional package management for GNU
|
2016-05-06 14:54:33 +02:00
|
|
|
|
;;; Copyright © 2015, 2016 Ricardo Wurmus <rekado@elephly.net>
|
2016-04-14 21:40:20 +02:00
|
|
|
|
;;; Copyright © 2015, 2016 Ludovic Courtès <ludo@gnu.org>
|
2015-07-24 16:49:57 +02: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 import cran)
|
|
|
|
|
#:use-module (ice-9 match)
|
|
|
|
|
#:use-module (ice-9 regex)
|
2015-12-03 16:12:09 +01:00
|
|
|
|
#:use-module ((ice-9 rdelim) #:select (read-string))
|
2015-07-24 16:49:57 +02:00
|
|
|
|
#:use-module (srfi srfi-1)
|
2015-10-21 14:36:14 +02:00
|
|
|
|
#:use-module (srfi srfi-26)
|
2015-07-24 16:49:57 +02:00
|
|
|
|
#:use-module (guix http-client)
|
|
|
|
|
#:use-module (guix hash)
|
|
|
|
|
#:use-module (guix store)
|
|
|
|
|
#:use-module (guix base32)
|
|
|
|
|
#:use-module ((guix download) #:select (download-to-store))
|
|
|
|
|
#:use-module (guix import utils)
|
2015-12-16 14:45:28 +01:00
|
|
|
|
#:use-module ((guix build-system r) #:select (cran-uri bioconductor-uri))
|
2015-10-21 14:36:14 +02:00
|
|
|
|
#:use-module (guix upstream)
|
|
|
|
|
#:use-module (guix packages)
|
|
|
|
|
#:export (cran->guix-package
|
2015-12-16 14:45:28 +01:00
|
|
|
|
bioconductor->guix-package
|
|
|
|
|
%cran-updater
|
|
|
|
|
%bioconductor-updater))
|
2015-07-24 16:49:57 +02:00
|
|
|
|
|
|
|
|
|
;;; Commentary:
|
|
|
|
|
;;;
|
|
|
|
|
;;; Generate a package declaration template for the latest version of an R
|
2015-12-03 16:12:09 +01:00
|
|
|
|
;;; package on CRAN, using the DESCRIPTION file downloaded from
|
2015-07-24 16:49:57 +02:00
|
|
|
|
;;; cran.r-project.org.
|
|
|
|
|
;;;
|
|
|
|
|
;;; Code:
|
|
|
|
|
|
|
|
|
|
(define string->license
|
|
|
|
|
(match-lambda
|
|
|
|
|
("AGPL-3" 'agpl3+)
|
|
|
|
|
("Artistic-2.0" 'artistic2.0)
|
|
|
|
|
("Apache License 2.0" 'asl2.0)
|
|
|
|
|
("BSD_2_clause" 'bsd-2)
|
|
|
|
|
("BSD_3_clause" 'bsd-3)
|
2015-12-03 15:00:43 +01:00
|
|
|
|
("GPL" (list 'gpl2+ 'gpl3+))
|
|
|
|
|
("GPL (>= 2)" 'gpl2+)
|
|
|
|
|
("GPL (>= 3)" 'gpl3+)
|
2015-07-24 16:49:57 +02:00
|
|
|
|
("GPL-2" 'gpl2+)
|
|
|
|
|
("GPL-3" 'gpl3+)
|
|
|
|
|
("LGPL-2" 'lgpl2.0+)
|
|
|
|
|
("LGPL-2.1" 'lgpl2.1+)
|
|
|
|
|
("LGPL-3" 'lgpl3+)
|
2015-12-03 15:00:43 +01:00
|
|
|
|
("LGPL (>= 2)" 'lgpl2.0+)
|
|
|
|
|
("LGPL (>= 3)" 'lgpl3+)
|
2015-07-24 16:49:57 +02:00
|
|
|
|
("MIT" 'x11)
|
2015-12-03 15:00:43 +01:00
|
|
|
|
("MIT + file LICENSE" 'x11)
|
2015-07-24 16:49:57 +02:00
|
|
|
|
((x) (string->license x))
|
|
|
|
|
((lst ...) `(list ,@(map string->license lst)))
|
|
|
|
|
(_ #f)))
|
|
|
|
|
|
2015-12-03 16:12:09 +01:00
|
|
|
|
|
|
|
|
|
(define (description->alist description)
|
|
|
|
|
"Convert a DESCRIPTION string into an alist."
|
|
|
|
|
(let ((lines (string-split description #\newline))
|
|
|
|
|
(parse (lambda (line acc)
|
|
|
|
|
(if (string-null? line) acc
|
|
|
|
|
;; Keys usually start with a capital letter and end with
|
|
|
|
|
;; ":". There are some exceptions, unfortunately (such
|
|
|
|
|
;; as "biocViews"). There are no blanks in a key.
|
|
|
|
|
(if (string-match "^[A-Za-z][^ :]+:( |\n|$)" line)
|
|
|
|
|
;; New key/value pair
|
|
|
|
|
(let* ((pos (string-index line #\:))
|
|
|
|
|
(key (string-take line pos))
|
|
|
|
|
(value (string-drop line (+ 1 pos))))
|
|
|
|
|
(cons (cons key
|
|
|
|
|
(string-trim-both value))
|
|
|
|
|
acc))
|
|
|
|
|
;; This is a continuation of the previous pair
|
|
|
|
|
(match-let ((((key . value) . rest) acc))
|
|
|
|
|
(cons (cons key (string-join
|
|
|
|
|
(list value
|
|
|
|
|
(string-trim-both line))))
|
|
|
|
|
rest)))))))
|
|
|
|
|
(fold parse '() lines)))
|
|
|
|
|
|
2015-07-24 16:49:57 +02:00
|
|
|
|
(define (format-inputs names)
|
|
|
|
|
"Generate a sorted list of package inputs from a list of package NAMES."
|
|
|
|
|
(map (lambda (name)
|
|
|
|
|
(list name (list 'unquote (string->symbol name))))
|
|
|
|
|
(sort names string-ci<?)))
|
|
|
|
|
|
|
|
|
|
(define* (maybe-inputs package-inputs #:optional (type 'inputs))
|
|
|
|
|
"Given a list of PACKAGE-INPUTS, tries to generate the TYPE field of a
|
|
|
|
|
package definition."
|
|
|
|
|
(match package-inputs
|
|
|
|
|
(()
|
|
|
|
|
'())
|
|
|
|
|
((package-inputs ...)
|
|
|
|
|
`((,type (,'quasiquote ,(format-inputs package-inputs)))))))
|
|
|
|
|
|
|
|
|
|
(define %cran-url "http://cran.r-project.org/web/packages/")
|
2015-12-16 14:45:28 +01:00
|
|
|
|
(define %bioconductor-url "http://bioconductor.org/packages/")
|
|
|
|
|
|
2016-10-24 00:02:40 +02:00
|
|
|
|
;; The latest Bioconductor release is 3.4. Bioconductor packages should be
|
2015-12-16 14:45:28 +01:00
|
|
|
|
;; updated together.
|
|
|
|
|
(define %bioconductor-svn-url
|
|
|
|
|
(string-append "https://readonly:readonly@"
|
2016-10-24 00:02:40 +02:00
|
|
|
|
"hedgehog.fhcrc.org/bioconductor/branches/RELEASE_3_4/"
|
2015-12-16 14:45:28 +01:00
|
|
|
|
"madman/Rpacks/"))
|
|
|
|
|
|
2015-07-24 16:49:57 +02:00
|
|
|
|
|
2015-12-16 14:28:43 +01:00
|
|
|
|
(define (fetch-description base-url name)
|
2015-12-03 16:12:09 +01:00
|
|
|
|
"Return an alist of the contents of the DESCRIPTION file for the R package
|
|
|
|
|
NAME, or #f on failure. NAME is case-sensitive."
|
2015-07-24 16:49:57 +02:00
|
|
|
|
;; This API always returns the latest release of the module.
|
2015-12-16 14:28:43 +01:00
|
|
|
|
(let ((url (string-append base-url name "/DESCRIPTION")))
|
2015-12-03 16:12:09 +01:00
|
|
|
|
(description->alist (read-string (http-fetch url)))))
|
|
|
|
|
|
|
|
|
|
(define (listify meta field)
|
|
|
|
|
"Look up FIELD in the alist META. If FIELD contains a comma-separated
|
|
|
|
|
string, turn it into a list and strip off parenthetic expressions. Return the
|
|
|
|
|
empty list when the FIELD cannot be found."
|
|
|
|
|
(let ((value (assoc-ref meta field)))
|
|
|
|
|
(if (not value)
|
|
|
|
|
'()
|
|
|
|
|
;; Strip off parentheses
|
|
|
|
|
(let ((items (string-split (regexp-substitute/global
|
|
|
|
|
#f "( *\\([^\\)]+\\)) *"
|
|
|
|
|
value 'pre 'post)
|
|
|
|
|
#\,)))
|
2015-12-16 14:29:38 +01:00
|
|
|
|
(remove (lambda (item)
|
|
|
|
|
(or (string-null? item)
|
|
|
|
|
;; When there is whitespace inside of items it is
|
|
|
|
|
;; probably because this was not an actual list to
|
|
|
|
|
;; begin with.
|
|
|
|
|
(string-any char-set:whitespace item)))
|
2015-12-03 16:12:09 +01:00
|
|
|
|
(map string-trim-both items))))))
|
|
|
|
|
|
2015-12-16 14:45:28 +01:00
|
|
|
|
(define (description->package repository meta)
|
|
|
|
|
"Return the `package' s-expression for an R package published on REPOSITORY
|
|
|
|
|
from the alist META, which was derived from the R package's DESCRIPTION file."
|
2015-07-24 16:49:57 +02:00
|
|
|
|
(define (guix-name name)
|
|
|
|
|
(if (string-prefix? "r-" name)
|
|
|
|
|
(string-downcase name)
|
|
|
|
|
(string-append "r-" (string-downcase name))))
|
|
|
|
|
|
2015-12-16 14:45:28 +01:00
|
|
|
|
(let* ((base-url (case repository
|
|
|
|
|
((cran) %cran-url)
|
|
|
|
|
((bioconductor) %bioconductor-url)))
|
|
|
|
|
(uri-helper (case repository
|
|
|
|
|
((cran) cran-uri)
|
|
|
|
|
((bioconductor) bioconductor-uri)))
|
|
|
|
|
(name (assoc-ref meta "Package"))
|
2015-12-03 16:12:09 +01:00
|
|
|
|
(synopsis (assoc-ref meta "Title"))
|
|
|
|
|
(version (assoc-ref meta "Version"))
|
|
|
|
|
(license (string->license (assoc-ref meta "License")))
|
|
|
|
|
;; Some packages have multiple home pages. Some have none.
|
|
|
|
|
(home-page (match (listify meta "URL")
|
|
|
|
|
((url rest ...) url)
|
2015-12-16 14:45:28 +01:00
|
|
|
|
(_ (string-append base-url name))))
|
|
|
|
|
(source-url (match (uri-helper name version)
|
2015-12-03 16:12:09 +01:00
|
|
|
|
((url rest ...) url)
|
2015-12-16 14:45:28 +01:00
|
|
|
|
((? string? url) url)
|
2015-12-03 16:12:09 +01:00
|
|
|
|
(_ #f)))
|
|
|
|
|
(tarball (with-store store (download-to-store store source-url)))
|
|
|
|
|
(sysdepends (map string-downcase (listify meta "SystemRequirements")))
|
|
|
|
|
(propagate (map guix-name (lset-union equal?
|
|
|
|
|
(listify meta "Imports")
|
|
|
|
|
(listify meta "LinkingTo")
|
|
|
|
|
(delete "R"
|
|
|
|
|
(listify meta "Depends"))))))
|
|
|
|
|
`(package
|
|
|
|
|
(name ,(guix-name name))
|
|
|
|
|
(version ,version)
|
|
|
|
|
(source (origin
|
|
|
|
|
(method url-fetch)
|
2015-12-16 14:45:28 +01:00
|
|
|
|
(uri (,(procedure-name uri-helper) ,name version))
|
2015-12-03 16:12:09 +01:00
|
|
|
|
(sha256
|
|
|
|
|
(base32
|
|
|
|
|
,(bytevector->nix-base32-string (file-sha256 tarball))))))
|
2015-12-16 14:45:28 +01:00
|
|
|
|
,@(if (not (equal? (string-append "r-" name)
|
|
|
|
|
(guix-name name)))
|
|
|
|
|
`((properties ,`(,'quasiquote ((,'upstream-name . ,name)))))
|
|
|
|
|
'())
|
2015-12-03 16:12:09 +01:00
|
|
|
|
(build-system r-build-system)
|
|
|
|
|
,@(maybe-inputs sysdepends)
|
|
|
|
|
,@(maybe-inputs propagate 'propagated-inputs)
|
|
|
|
|
(home-page ,(if (string-null? home-page)
|
2015-12-16 14:45:28 +01:00
|
|
|
|
(string-append base-url name)
|
2015-12-03 16:12:09 +01:00
|
|
|
|
home-page))
|
|
|
|
|
(synopsis ,synopsis)
|
|
|
|
|
(description ,(beautify-description (assoc-ref meta "Description")))
|
|
|
|
|
(license ,license))))
|
2015-07-24 16:49:57 +02:00
|
|
|
|
|
2015-12-16 14:45:28 +01:00
|
|
|
|
(define* (cran->guix-package package-name #:optional (repo 'cran))
|
|
|
|
|
"Fetch the metadata for PACKAGE-NAME from REPO and return the `package'
|
|
|
|
|
s-expression corresponding to that package, or #f on failure."
|
|
|
|
|
(let* ((url (case repo
|
|
|
|
|
((cran) %cran-url)
|
|
|
|
|
((bioconductor) %bioconductor-svn-url)))
|
|
|
|
|
(module-meta (fetch-description url package-name)))
|
|
|
|
|
(and=> module-meta (cut description->package repo <>))))
|
2015-10-21 14:36:14 +02:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
;;;
|
|
|
|
|
;;; Updater.
|
|
|
|
|
;;;
|
|
|
|
|
|
2015-12-16 14:22:17 +01:00
|
|
|
|
(define (package->upstream-name package)
|
|
|
|
|
"Return the upstream name of the PACKAGE."
|
|
|
|
|
(let* ((properties (package-properties package))
|
|
|
|
|
(upstream-name (and=> properties
|
|
|
|
|
(cut assoc-ref <> 'upstream-name))))
|
|
|
|
|
(if upstream-name
|
|
|
|
|
upstream-name
|
|
|
|
|
(match (package-source package)
|
|
|
|
|
((? origin? origin)
|
|
|
|
|
(match (origin-uri origin)
|
2016-03-22 15:12:30 +01:00
|
|
|
|
((or (? string? url) (url _ ...))
|
2015-12-16 14:22:17 +01:00
|
|
|
|
(let ((end (string-rindex url #\_))
|
|
|
|
|
(start (string-rindex url #\/)))
|
|
|
|
|
;; The URL ends on
|
|
|
|
|
;; (string-append "/" name "_" version ".tar.gz")
|
|
|
|
|
(substring url start end)))
|
|
|
|
|
(_ #f)))
|
|
|
|
|
(_ #f)))))
|
|
|
|
|
|
2015-12-16 14:45:28 +01:00
|
|
|
|
(define (latest-cran-release package)
|
2015-10-21 14:36:14 +02:00
|
|
|
|
"Return an <upstream-source> for the latest release of PACKAGE."
|
2015-12-03 16:12:09 +01:00
|
|
|
|
|
2015-12-16 14:22:17 +01:00
|
|
|
|
(define upstream-name
|
2016-04-14 21:40:20 +02:00
|
|
|
|
(package->upstream-name package))
|
2015-12-03 16:12:09 +01:00
|
|
|
|
|
|
|
|
|
(define meta
|
2015-12-16 14:28:43 +01:00
|
|
|
|
(fetch-description %cran-url upstream-name))
|
2015-12-03 16:12:09 +01:00
|
|
|
|
|
|
|
|
|
(and meta
|
|
|
|
|
(let ((version (assoc-ref meta "Version")))
|
|
|
|
|
;; CRAN does not provide signatures.
|
|
|
|
|
(upstream-source
|
2016-04-14 21:40:20 +02:00
|
|
|
|
(package (package-name package))
|
2015-12-03 16:12:09 +01:00
|
|
|
|
(version version)
|
2015-12-16 14:22:17 +01:00
|
|
|
|
(urls (cran-uri upstream-name version))))))
|
2015-10-21 14:36:14 +02:00
|
|
|
|
|
2015-12-16 14:45:28 +01:00
|
|
|
|
(define (latest-bioconductor-release package)
|
|
|
|
|
"Return an <upstream-source> for the latest release of PACKAGE."
|
|
|
|
|
|
|
|
|
|
(define upstream-name
|
2016-04-14 21:40:20 +02:00
|
|
|
|
(package->upstream-name package))
|
2015-12-16 14:45:28 +01:00
|
|
|
|
|
|
|
|
|
(define meta
|
|
|
|
|
(fetch-description %bioconductor-svn-url upstream-name))
|
|
|
|
|
|
|
|
|
|
(and meta
|
|
|
|
|
(let ((version (assoc-ref meta "Version")))
|
|
|
|
|
;; Bioconductor does not provide signatures.
|
|
|
|
|
(upstream-source
|
2016-04-14 21:40:20 +02:00
|
|
|
|
(package (package-name package))
|
2015-12-16 14:45:28 +01:00
|
|
|
|
(version version)
|
2016-05-10 14:18:44 +02:00
|
|
|
|
(urls (list (bioconductor-uri upstream-name version)))))))
|
2015-12-16 14:45:28 +01:00
|
|
|
|
|
2015-10-21 14:36:14 +02:00
|
|
|
|
(define (cran-package? package)
|
|
|
|
|
"Return true if PACKAGE is an R package from CRAN."
|
2015-12-16 14:45:28 +01:00
|
|
|
|
(and (string-prefix? "r-" (package-name package))
|
|
|
|
|
(match (and=> (package-source package) origin-uri)
|
|
|
|
|
((? string? uri)
|
|
|
|
|
(string-prefix? "mirror://cran" uri))
|
|
|
|
|
((? list? uris)
|
|
|
|
|
(any (cut string-prefix? "mirror://cran" <>) uris))
|
|
|
|
|
(_ #f))))
|
|
|
|
|
|
|
|
|
|
(define (bioconductor-package? package)
|
|
|
|
|
"Return true if PACKAGE is an R package from Bioconductor."
|
|
|
|
|
(and (string-prefix? "r-" (package-name package))
|
|
|
|
|
(match (and=> (package-source package) origin-uri)
|
|
|
|
|
((? string? uri)
|
|
|
|
|
(string-prefix? "http://bioconductor.org" uri))
|
|
|
|
|
((? list? uris)
|
|
|
|
|
(any (cut string-prefix? "http://bioconductor.org" <>) uris))
|
|
|
|
|
(_ #f))))
|
2015-10-21 14:36:14 +02:00
|
|
|
|
|
|
|
|
|
(define %cran-updater
|
2015-10-26 19:24:53 +01:00
|
|
|
|
(upstream-updater
|
|
|
|
|
(name 'cran)
|
|
|
|
|
(description "Updater for CRAN packages")
|
|
|
|
|
(pred cran-package?)
|
2015-12-16 14:45:28 +01:00
|
|
|
|
(latest latest-cran-release)))
|
|
|
|
|
|
|
|
|
|
(define %bioconductor-updater
|
|
|
|
|
(upstream-updater
|
|
|
|
|
(name 'bioconductor)
|
|
|
|
|
(description "Updater for Bioconductor packages")
|
|
|
|
|
(pred bioconductor-package?)
|
|
|
|
|
(latest latest-bioconductor-release)))
|
2015-10-21 14:36:14 +02:00
|
|
|
|
|
|
|
|
|
;;; cran.scm ends here
|