2015-06-16 10:50:06 +02:00
|
|
|
|
;;; GNU Guix --- Functional package management for GNU
|
|
|
|
|
;;; Copyright © 2015 Federico Beffa <beffa@fbengineering.ch>
|
2016-04-14 21:40:20 +02:00
|
|
|
|
;;; Copyright © 2015, 2016 Ludovic Courtès <ludo@gnu.org>
|
2015-06-16 10:50:06 +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 elpa)
|
|
|
|
|
#:use-module (ice-9 match)
|
|
|
|
|
#:use-module (ice-9 rdelim)
|
2015-10-21 12:12:59 +02:00
|
|
|
|
#:use-module (web uri)
|
2015-06-16 10:50:06 +02:00
|
|
|
|
#:use-module (srfi srfi-1)
|
|
|
|
|
#:use-module (srfi srfi-9)
|
|
|
|
|
#:use-module (srfi srfi-9 gnu)
|
|
|
|
|
#:use-module (srfi srfi-11)
|
|
|
|
|
#:use-module (srfi srfi-26)
|
|
|
|
|
#:use-module ((guix download) #:select (download-to-store))
|
|
|
|
|
#:use-module (guix import utils)
|
2015-10-21 12:12:59 +02:00
|
|
|
|
#:use-module (guix http-client)
|
2015-06-16 10:50:06 +02:00
|
|
|
|
#:use-module (guix store)
|
|
|
|
|
#:use-module (guix ui)
|
|
|
|
|
#:use-module (guix hash)
|
|
|
|
|
#:use-module (guix base32)
|
2015-10-21 12:25:06 +02:00
|
|
|
|
#:use-module (guix upstream)
|
|
|
|
|
#:use-module (guix packages)
|
utils: Move combinators to (guix combinators).
* guix/utils.scm (compile-time-value, memoize, fold2)
(fold-tree, fold-tree-leaves): Move to...
* guix/combinators: ... here. New file.
* tests/utils.scm ("fold2, 1 list", "fold2, 2 lists")
(fold-tree tests): Move to...
* tests/combinators.scm: ... here. New file.
* Makefile.am (MODULES, SCM_TESTS): Add them.
* gnu/packages.scm, gnu/packages/bootstrap.scm,
gnu/services/herd.scm, guix/build-system/gnu.scm,
guix/build-system/python.scm, guix/derivations.scm,
guix/gnu-maintenance.scm, guix/import/elpa.scm,
guix/scripts/archive.scm, guix/scripts/build.scm,
guix/scripts/graph.scm, guix/scripts/lint.scm,
guix/scripts/size.scm, guix/scripts/substitute.scm,
guix/serialization.scm, guix/store.scm, guix/ui.scm: Adjust imports
accordingly.
2016-05-04 17:35:47 +02:00
|
|
|
|
#:use-module ((guix combinators) #:select (memoize))
|
|
|
|
|
#:use-module ((guix utils) #:select (call-with-temporary-output-file))
|
2015-10-21 12:25:06 +02:00
|
|
|
|
#:export (elpa->guix-package
|
|
|
|
|
%elpa-updater))
|
2015-06-16 10:50:06 +02:00
|
|
|
|
|
|
|
|
|
(define (elpa-dependencies->names deps)
|
|
|
|
|
"Convert DEPS, a list of symbol/version pairs à la ELPA, to a list of
|
|
|
|
|
package names as strings"
|
|
|
|
|
(match deps
|
|
|
|
|
(((names _ ...) ...)
|
|
|
|
|
(map symbol->string names))))
|
|
|
|
|
|
|
|
|
|
(define emacs-standard-library?
|
|
|
|
|
(let ((libs '("emacs" "cl-lib")))
|
|
|
|
|
(lambda (lib)
|
|
|
|
|
"Return true if LIB is part of Emacs itself. The check is not
|
|
|
|
|
exhaustive and only attempts to recognize a subset of packages which in the
|
|
|
|
|
past were distributed separately from Emacs."
|
|
|
|
|
(member lib libs))))
|
|
|
|
|
|
|
|
|
|
(define (filter-dependencies names)
|
|
|
|
|
"Remove the package names included with Emacs from the list of
|
|
|
|
|
NAMES (strings)."
|
2015-07-22 15:26:12 +02:00
|
|
|
|
(filter (compose not emacs-standard-library?) names))
|
2015-06-16 10:50:06 +02:00
|
|
|
|
|
|
|
|
|
(define (elpa-name->package-name name)
|
|
|
|
|
"Given the NAME of an Emacs package, return the corresponding Guix name."
|
|
|
|
|
(let ((package-name-prefix "emacs-"))
|
|
|
|
|
(if (string-prefix? package-name-prefix name)
|
|
|
|
|
(string-downcase name)
|
|
|
|
|
(string-append package-name-prefix (string-downcase name)))))
|
|
|
|
|
|
|
|
|
|
(define* (elpa-url #:optional (repo 'gnu))
|
|
|
|
|
"Retrun the URL of REPO."
|
|
|
|
|
(let ((elpa-archives
|
|
|
|
|
'((gnu . "http://elpa.gnu.org/packages")
|
|
|
|
|
(melpa-stable . "http://stable.melpa.org/packages")
|
|
|
|
|
(melpa . "http://melpa.org/packages"))))
|
|
|
|
|
(assq-ref elpa-archives repo)))
|
|
|
|
|
|
|
|
|
|
(define* (elpa-fetch-archive #:optional (repo 'gnu))
|
|
|
|
|
"Retrive the archive with the list of packages available from REPO."
|
|
|
|
|
(let ((url (and=> (elpa-url repo)
|
|
|
|
|
(cut string-append <> "/archive-contents"))))
|
|
|
|
|
(if url
|
2015-10-21 12:12:59 +02:00
|
|
|
|
;; Use a relatively small TTL for the archive itself.
|
|
|
|
|
(parameterize ((%http-cache-ttl (* 6 3600)))
|
|
|
|
|
(call-with-downloaded-file url read))
|
2015-06-16 10:50:06 +02:00
|
|
|
|
(leave (_ "~A: currently not supported~%") repo))))
|
|
|
|
|
|
2015-07-22 15:26:12 +02:00
|
|
|
|
(define* (call-with-downloaded-file url proc #:optional (error-thunk #f))
|
2015-06-16 10:50:06 +02:00
|
|
|
|
"Fetch URL, store the content in a temporary file and call PROC with that
|
2015-07-22 15:26:12 +02:00
|
|
|
|
file. Returns the value returned by PROC. On error call ERROR-THUNK and
|
|
|
|
|
return its value or leave if it's false."
|
2016-12-14 02:31:12 +01:00
|
|
|
|
(catch #t
|
|
|
|
|
(lambda ()
|
|
|
|
|
(proc (http-fetch/cached (string->uri url))))
|
|
|
|
|
(lambda (key . args)
|
|
|
|
|
(if error-thunk
|
|
|
|
|
(error-thunk)
|
|
|
|
|
(leave (_ "~A: download failed~%") url)))))
|
2015-06-16 10:50:06 +02:00
|
|
|
|
|
|
|
|
|
(define (is-elpa-package? name elpa-pkg-spec)
|
|
|
|
|
"Return true if the string NAME corresponds to the name of the package
|
|
|
|
|
defined by ELPA-PKG-SPEC, a package specification as in an archive
|
|
|
|
|
'archive-contents' file."
|
|
|
|
|
(eq? (first elpa-pkg-spec) (string->symbol name)))
|
|
|
|
|
|
|
|
|
|
(define* (elpa-package-info name #:optional (repo 'gnu))
|
|
|
|
|
"Extract the information about the package NAME from the package archieve of
|
|
|
|
|
REPO."
|
|
|
|
|
(let* ((archive (elpa-fetch-archive repo))
|
|
|
|
|
(pkgs (match archive ((version pkg-spec ...) pkg-spec)))
|
|
|
|
|
(info (filter (cut is-elpa-package? name <>) pkgs)))
|
|
|
|
|
(if (pair? info) (first info) #f)))
|
|
|
|
|
|
|
|
|
|
;; Object to store information about an ELPA package.
|
|
|
|
|
(define-record-type <elpa-package>
|
|
|
|
|
(make-elpa-package name version inputs synopsis kind home-page description
|
|
|
|
|
source-url)
|
|
|
|
|
elpa-package?
|
|
|
|
|
(name elpa-package-name)
|
|
|
|
|
(version elpa-package-version)
|
|
|
|
|
(inputs elpa-package-inputs)
|
|
|
|
|
(synopsis elpa-package-synopsis)
|
|
|
|
|
(kind elpa-package-kind)
|
|
|
|
|
(home-page elpa-package-home-page)
|
|
|
|
|
(description elpa-package-description)
|
|
|
|
|
(source-url elpa-package-source-url))
|
|
|
|
|
|
|
|
|
|
(set-record-type-printer! <elpa-package>
|
|
|
|
|
(lambda (package port)
|
2016-01-24 17:42:39 +01:00
|
|
|
|
(format port "#<elpa-package ~a@~a>"
|
2015-06-16 10:50:06 +02:00
|
|
|
|
(elpa-package-name package)
|
|
|
|
|
(elpa-package-version package))))
|
|
|
|
|
|
|
|
|
|
(define (elpa-version->string elpa-version)
|
|
|
|
|
"Convert the package version as used in Emacs package files into a string."
|
|
|
|
|
(if (pair? elpa-version)
|
|
|
|
|
(let-values (((ms rest) (match elpa-version
|
|
|
|
|
((ms . rest)
|
|
|
|
|
(values ms rest)))))
|
|
|
|
|
(fold (lambda (n s) (string-append s "." (number->string n)))
|
|
|
|
|
(number->string ms) rest))
|
|
|
|
|
#f))
|
|
|
|
|
|
|
|
|
|
(define (package-home-page alist)
|
|
|
|
|
"Extract the package home-page from ALIST."
|
|
|
|
|
(or (assq-ref alist ':url) "unspecified"))
|
|
|
|
|
|
|
|
|
|
(define (ensure-list alist)
|
|
|
|
|
"If ALIST is the symbol 'nil return the empty list. Otherwise, return ALIST."
|
|
|
|
|
(if (eq? alist 'nil)
|
|
|
|
|
'()
|
|
|
|
|
alist))
|
|
|
|
|
|
|
|
|
|
(define (package-source-url kind name version repo)
|
|
|
|
|
"Return the source URL of the package described the the strings NAME and
|
|
|
|
|
VERSION at REPO. KIND is either the symbol 'single or 'tar."
|
|
|
|
|
(case kind
|
|
|
|
|
((single) (full-url repo name ".el" version))
|
|
|
|
|
((tar) (full-url repo name ".tar" version))
|
|
|
|
|
(else
|
|
|
|
|
#f)))
|
|
|
|
|
|
|
|
|
|
(define* (full-url repo name suffix #:optional (version #f))
|
|
|
|
|
"Return the full URL of the package NAME at REPO and the SUFFIX. Maybe
|
|
|
|
|
include VERSION."
|
|
|
|
|
(if version
|
|
|
|
|
(string-append (elpa-url repo) "/" name "-" version suffix)
|
|
|
|
|
(string-append (elpa-url repo) "/" name suffix)))
|
|
|
|
|
|
|
|
|
|
(define (fetch-package-description kind name repo)
|
|
|
|
|
"Fetch the description of package NAME of type KIND from REPO."
|
2015-07-22 15:26:12 +02:00
|
|
|
|
(let ((url (full-url repo name "-readme.txt"))
|
|
|
|
|
(error-thunk (lambda () "No description available.")))
|
|
|
|
|
(call-with-downloaded-file url read-string error-thunk)))
|
2015-06-16 10:50:06 +02:00
|
|
|
|
|
|
|
|
|
(define* (fetch-elpa-package name #:optional (repo 'gnu))
|
|
|
|
|
"Fetch package NAME from REPO."
|
|
|
|
|
(let ((pkg (elpa-package-info name repo)))
|
|
|
|
|
(match pkg
|
|
|
|
|
((name version reqs synopsis kind . rest)
|
|
|
|
|
(let* ((name (symbol->string name))
|
|
|
|
|
(ver (elpa-version->string version))
|
|
|
|
|
(url (package-source-url kind name ver repo)))
|
|
|
|
|
(make-elpa-package name ver
|
|
|
|
|
(ensure-list reqs) synopsis kind
|
|
|
|
|
(package-home-page (first rest))
|
|
|
|
|
(fetch-package-description kind name repo)
|
|
|
|
|
url)))
|
|
|
|
|
(_ #f))))
|
|
|
|
|
|
|
|
|
|
(define* (elpa-package->sexp pkg)
|
|
|
|
|
"Return the `package' S-expression for the Emacs package PKG, a record of
|
|
|
|
|
type '<elpa-package>'."
|
|
|
|
|
|
|
|
|
|
(define name (elpa-package-name pkg))
|
|
|
|
|
|
|
|
|
|
(define version (elpa-package-version pkg))
|
|
|
|
|
|
|
|
|
|
(define source-url (elpa-package-source-url pkg))
|
|
|
|
|
|
|
|
|
|
(define dependencies
|
|
|
|
|
(let* ((deps (elpa-package-inputs pkg))
|
|
|
|
|
(names (filter-dependencies (elpa-dependencies->names deps))))
|
|
|
|
|
(map (lambda (n)
|
|
|
|
|
(let ((new-n (elpa-name->package-name n)))
|
|
|
|
|
(list new-n (list 'unquote (string->symbol new-n)))))
|
|
|
|
|
names)))
|
|
|
|
|
|
|
|
|
|
(define (maybe-inputs input-type inputs)
|
|
|
|
|
(match inputs
|
|
|
|
|
(()
|
|
|
|
|
'())
|
|
|
|
|
((inputs ...)
|
|
|
|
|
(list (list input-type
|
|
|
|
|
(list 'quasiquote inputs))))))
|
|
|
|
|
|
|
|
|
|
(let ((tarball (with-store store
|
|
|
|
|
(download-to-store store source-url))))
|
|
|
|
|
`(package
|
|
|
|
|
(name ,(elpa-name->package-name name))
|
|
|
|
|
(version ,version)
|
|
|
|
|
(source (origin
|
|
|
|
|
(method url-fetch)
|
|
|
|
|
(uri (string-append ,@(factorize-uri source-url version)))
|
|
|
|
|
(sha256
|
|
|
|
|
(base32
|
|
|
|
|
,(if tarball
|
|
|
|
|
(bytevector->nix-base32-string (file-sha256 tarball))
|
|
|
|
|
"failed to download package")))))
|
|
|
|
|
(build-system emacs-build-system)
|
2016-12-14 02:34:15 +01:00
|
|
|
|
,@(maybe-inputs 'propagated-inputs dependencies)
|
2015-06-16 10:50:06 +02:00
|
|
|
|
(home-page ,(elpa-package-home-page pkg))
|
|
|
|
|
(synopsis ,(elpa-package-synopsis pkg))
|
|
|
|
|
(description ,(elpa-package-description pkg))
|
|
|
|
|
(license license:gpl3+))))
|
|
|
|
|
|
|
|
|
|
(define* (elpa->guix-package name #:optional (repo 'gnu))
|
|
|
|
|
"Fetch the package NAME from REPO and produce a Guix package S-expression."
|
|
|
|
|
(let ((pkg (fetch-elpa-package name repo)))
|
|
|
|
|
(and=> pkg elpa-package->sexp)))
|
|
|
|
|
|
2015-10-21 12:25:06 +02:00
|
|
|
|
|
|
|
|
|
;;;
|
|
|
|
|
;;; Updates.
|
|
|
|
|
;;;
|
|
|
|
|
|
|
|
|
|
(define (latest-release package)
|
2016-04-14 21:40:20 +02:00
|
|
|
|
"Return an <upstream-release> for the latest release of PACKAGE."
|
2015-10-21 12:25:06 +02:00
|
|
|
|
(define name
|
2016-04-14 21:40:20 +02:00
|
|
|
|
(if (string-prefix? "emacs-" (package-name package))
|
|
|
|
|
(string-drop (package-name package) 6)
|
|
|
|
|
(package-name package)))
|
2015-10-21 12:25:06 +02:00
|
|
|
|
|
|
|
|
|
(let* ((repo 'gnu)
|
|
|
|
|
(info (elpa-package-info name repo))
|
|
|
|
|
(version (match info
|
|
|
|
|
((name raw-version . _)
|
|
|
|
|
(elpa-version->string raw-version))))
|
|
|
|
|
(url (match info
|
|
|
|
|
((_ raw-version reqs synopsis kind . rest)
|
|
|
|
|
(package-source-url kind name version repo)))))
|
|
|
|
|
(upstream-source
|
2016-04-14 21:40:20 +02:00
|
|
|
|
(package (package-name package))
|
2015-10-21 12:25:06 +02:00
|
|
|
|
(version version)
|
|
|
|
|
(urls (list url))
|
|
|
|
|
(signature-urls (list (string-append url ".sig"))))))
|
|
|
|
|
|
|
|
|
|
(define (package-from-gnu.org? package)
|
|
|
|
|
"Return true if PACKAGE is from elpa.gnu.org."
|
|
|
|
|
(match (and=> (package-source package) origin-uri)
|
|
|
|
|
((? string? uri)
|
|
|
|
|
(let ((uri (string->uri uri)))
|
|
|
|
|
(and uri (string=? (uri-host uri) "elpa.gnu.org"))))
|
|
|
|
|
(_ #f)))
|
|
|
|
|
|
|
|
|
|
(define %elpa-updater
|
|
|
|
|
;; The ELPA updater. We restrict it to packages hosted on elpa.gnu.org
|
|
|
|
|
;; because for other repositories, we typically grab the source elsewhere.
|
2015-10-26 19:24:53 +01:00
|
|
|
|
(upstream-updater
|
|
|
|
|
(name 'elpa)
|
|
|
|
|
(description "Updater for ELPA packages")
|
|
|
|
|
(pred package-from-gnu.org?)
|
|
|
|
|
(latest latest-release)))
|
2015-10-21 12:25:06 +02:00
|
|
|
|
|
2015-06-16 10:50:06 +02:00
|
|
|
|
;;; elpa.scm ends here
|