mirror of
https://codeberg.org/guix/guix.git
synced 2025-10-02 02:15:12 +00:00
* guix/import/utils.scm (find-version): New procedure. * guix/scripts/refresh.scm (<update-spec>) [partial?]: New field. (update-spec-partial?): New accessor. (update-spec): Add a PARTIAL? optional argument. (update-specification->update-spec) <update-spec>: Call with its new PARTIAL? optional argument when FALLBACK-VERSION is provided, i.e. when '--target-version' was used. (update-package): Remove the PACKAGE and VERSION positional arguments, and replace them with UPDATE-SPEC. Update doc. Call `package-update' with its new #:partial-version? argument. (check-for-package-update) <package-latest-release>: Pass the new #:partial-version? argument to it. (guix-refresh) <update-package>: Adjust call accordingly. (show-help): Udate doc. * guix/upstream.scm (package-latest-release): Add #:partial-version? argument, and apply it to the importer call. (package-update): Add #:partial-version?> argument. Update doc. Pass it to the `package-latest-release' call. * guix/gnu-maintenance.scm (rewrite-url): Add #:partial-version? argument. Update doc. Crawl URL for newer compatible versions when provided. (import-html-release): Add #:partial-version? argument, and pass it to the `rewrite-url' call. Use `find-version' to find the best version. (import-release, import-ftp-release, import-gnu-release) (import-release*): Add #:partial-version? argument and honor it. (import-html-updatable-release): Add #:partial-version? argument, and pass it to the `import-html-release' call. * guix/import/gnome.scm (import-gnome-release) <#:partial-version?>: Add new argument and honor it. * guix/import/texlive.scm (latest-texlive-tag): Rename to... (texlive-tags): ... this, and have it return all tags. (texlive->guix-package): Adjust accordingly. (latest-release): Add a #:partial-version? argument. Update doc. * guix/import/stackage.scm (latest-lts-release): New #:partial-version? argument. * guix/import/pypi.scm (import-release): New #:partial-version? argument; pass it to `pypi-package->upstream-source'. * guix/import/opam.scm (latest-release): New #:partial-version? argument. * guix/import/minetest.scm (latest-minetest-release): New #:partial-version? argument. (pypi-package->upstream-source): New #:partial-version? argument. Update doc. * guix/import/launchpad.scm (latest-released-version): Rename to... (release-versions): ... this, making it return all versions. (import-release) <#:partial-version?>: New argument. * guix/import/kde.scm (import-kde-release) <#:partial-version?>: New argument. Update doc. Refactor to honor argument. * guix/import/hexpm.scm (lookup-hexpm): Update doc. (hexpm-latest-release): Rename to... (hexpm-releases): ... this; return all release strings. (hexpm->guix-package): Adjust accordingly. (import-release): Add and honor a #:partial-version? argument. Update doc. * guix/import/hackage.scm (import-release): New #:partial-version? argument. * guix/import/cpan.scm (latest-release): New #:partial-version? argument. * guix/import/crate.scm (max-crate-version-of-semver): Improve doc. (import-release): Add a #:partial-version? argument and honor it. * guix/import/egg.scm (find-latest-version): Rename to... (get-versions): ... this, returning all versions. (egg-metadata): Adjust accordingly. (egg->guix-package): Likewise. (import-release): Add a new #:partial-version? argument and honor it. * guix/import/elpa.scm (latest-release): New #:partial-version? argument. * guix/import/gem.scm (get-versions): New procedure. (import-release): Add a new #:partial-version? argument and honor it. * guix/import/git.scm (version-mapping): Update doc; streamline a bit. (latest-tag): Rename to... (get-tags): ... this, dropping the #:version keyword and returning the complete tags alist. Update doc. (latest-git-tag-version): Rename to... (get-package-tags): ... this, returning the complete tags alist of the package. Update doc. (import-git-release): Add a new #:partial-version? argument and honor it. Update doc. * guix/import/github.scm (latest-released-version): Rename to... (get-package-tags): ... this, returning all tags. Update doc. (import-release): Add a new #:partial-version? argument and honor it. * guix/import/cran.scm (latest-cran-release) (latest-bioconductor-release): Add #:partial-version? argument. * guix/import/composer.scm (latest-version): Delete procedure. (composer-fetch): Add #:partial-version? keyword and honor it. Update doc. (import-release): Likewise. * guix/import/test.scm (import-release): Add #:partial-version? argument. * tests/guix-refresh.sh: Add test. * tests/gem.scm (test-foo-versions-json): New variable. (package-latest-release): Mock new URL. * tests/import-git.scm (latest-git-tag-version): New procedure. * tests/gnu-maintenance.scm (libuv-dist-html) (libuv-dist-1.46.0-html, libuv-dist-1.44.2-html) (libuv-html-data): New variables. (mock-http-fetch/cached): New procedure. ("rewrite-url, without to-version"): Rewrite using the above. ("rewrite-url, partial to-version"): New test. * doc/guix.texi <"Invoking guix refresh">: Update doc. Series-to: 75871@debbugs.gnu.org Change-Id: I092a58b57ac42e54a2fa55e7761e8c6993af8ad4
338 lines
13 KiB
Scheme
338 lines
13 KiB
Scheme
;;; GNU Guix --- Functional package management for GNU
|
||
;;; Copyright © 2015 Cyril Roelandt <tipecaml@gmail.com>
|
||
;;; Copyright © 2016 David Craven <david@craven.ch>
|
||
;;; Copyright © 2017, 2019-2021, 2024 Ludovic Courtès <ludo@gnu.org>
|
||
;;; Copyright © 2019 Martin Becze <mjbecze@riseup.net>
|
||
;;; Copyright © 2019, 2024 Maxim Cournoyer <maxim.cournoyer@gmail.com>
|
||
;;; Copyright © 2020-2022 Hartmut Goebel <h.goebel@crazy-compilers.com>
|
||
;;;
|
||
;;; 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 hexpm)
|
||
#:use-module (json)
|
||
#:use-module (guix import utils)
|
||
#:use-module ((guix import json) #:select (json-fetch))
|
||
#:use-module ((guix build utils)
|
||
#:select ((package-name->name+version
|
||
. hyphen-package-name->name+version)
|
||
dump-port
|
||
call-with-temporary-output-file))
|
||
#:use-module (guix packages)
|
||
#:use-module (guix upstream)
|
||
#:autoload (guix utils) (file-sans-extension version>? version-prefix?)
|
||
#:use-module (ice-9 match)
|
||
#:use-module (srfi srfi-1)
|
||
#:use-module (srfi srfi-26)
|
||
#:use-module (guix build-system rebar)
|
||
#:export (hexpm->guix-package
|
||
guix-package->hexpm-name
|
||
strings->licenses ;; why used here?
|
||
hexpm-recursive-import
|
||
%hexpm-updater))
|
||
|
||
;;;
|
||
;;; Interface to https://hex.pm/api, version 2.
|
||
;;; REST-API end-points:
|
||
;;; https://github.com/hexpm/specifications/blob/master/apiary.apib
|
||
;;; Repository end-points:
|
||
;;; https://github.com/hexpm/specifications/blob/master/endpoints.md
|
||
;;;
|
||
|
||
(define %hexpm-api-url
|
||
(make-parameter "https://hex.pm/api"))
|
||
|
||
(define (package-url name)
|
||
(string-append (%hexpm-api-url) "/packages/" name))
|
||
|
||
;;
|
||
;; Hexpm Package. /packages/${name}
|
||
;; https://github.com/hexpm/specifications/blob/master/apiary.apib#Package
|
||
;;
|
||
;; Each package can have several "releases", each of which has its own set of
|
||
;; requirements, build-tool, etc. - see <hexpm-release> below.
|
||
(define-json-mapping <hexpm-pkgdef> make-hexpm-pkgdef hexpm-pkgdef?
|
||
json->hexpm
|
||
(name hexpm-name) ; string
|
||
(html-url hexpm-html-url "html_url") ; string
|
||
(docs-html-url hexpm-docs-html-url "docs_html_url") ; string | 'null
|
||
(meta hexpm-meta "meta" json->hexpm-meta)
|
||
(versions hexpm-versions "releases" ; list of <hexpm-version>
|
||
(lambda (vector)
|
||
(map json->hexpm-version
|
||
(vector->list vector))))
|
||
;; "latest_version" and "latest_stable_version" are not named in the
|
||
;; specification, butt seen in practice.
|
||
(latest-version hexpm-latest-version "latest_version") ; string
|
||
(latest-stable hexpm-latest-stable "latest_stable_version")) ; string
|
||
|
||
;; Hexpm package metadata.
|
||
(define-json-mapping <hexpm-meta> make-hexpm-meta hexpm-meta?
|
||
json->hexpm-meta
|
||
(description hexpm-meta-description) ;string
|
||
(licenses hexpm-meta-licenses "licenses" ;list of strings
|
||
(lambda (vector)
|
||
(or (and vector (vector->list vector))
|
||
#f))))
|
||
|
||
;; Hexpm package versions.
|
||
(define-json-mapping <hexpm-version> make-hexpm-version hexpm-version?
|
||
json->hexpm-version
|
||
(number hexpm-version-number "version") ;string
|
||
(url hexpm-version-url)) ;string
|
||
|
||
|
||
(define (lookup-hexpm name)
|
||
"Look up NAME on hex.pm and return the corresponding <hexpm-pkgdef> record
|
||
or #f if it was not found."
|
||
(and=> (json-fetch (package-url name))
|
||
json->hexpm))
|
||
|
||
;;
|
||
;; Hexpm release. /packages/${name}/releases/${version}
|
||
;; https://github.com/hexpm/specifications/blob/master/apiary.apib#Release
|
||
;;
|
||
(define-json-mapping <hexpm-release> make-hexpm-release hexpm-release?
|
||
json->hexpm-release
|
||
(version hexpm-release-version) ; string
|
||
(url hexpm-release-url) ; string
|
||
(meta hexpm-release-meta "meta" json->hexpm-release-meta)
|
||
;; Specification names the next fields "dependencies", but in practice it is
|
||
;; "requirements".
|
||
(dependencies hexpm-requirements "requirements")) ; list of <hexpm-dependency>
|
||
|
||
;; Hexpm release meta.
|
||
;; https://github.com/hexpm/specifications/blob/main/package_metadata.md
|
||
(define-json-mapping <hexpm-release-meta>
|
||
make-hexpm-release-meta hexpm-release-meta?
|
||
json->hexpm-release-meta
|
||
(app hexpm-release-meta-app) ; string
|
||
(elixir hexpm-release-meta-elixir) ; string
|
||
(build-tools hexpm-release-meta-build-tools "build_tools" ; list of strings
|
||
(lambda (vector)
|
||
(or (and vector (vector->list vector))
|
||
(list)))))
|
||
|
||
;; Hexpm dependency. Each requirement has information about the required
|
||
;; version, such as "~> 2.1.2" or ">= 2.1.2 and < 2.2.0", see
|
||
;; <https://hexdocs.pm/elixir/Version.html#module-requirements>, and whether
|
||
;; the dependency is optional.
|
||
(define-json-mapping <hexpm-dependency> make-hexpm-dependency
|
||
hexpm-dependency?
|
||
json->hexpm-dependency
|
||
(name hexpm-dependency-name "app") ; string
|
||
(requirement hexpm-dependency-requirement) ; string
|
||
(optional hexpm-dependency-optional)) ; bool
|
||
|
||
(define (hexpm-release-dependencies release)
|
||
"Return the list of dependency names of RELEASE, a <hexpm-release>."
|
||
(let ((reqs (or (hexpm-requirements release) '#())))
|
||
(map first reqs))) ;; TODO: also return required version
|
||
|
||
|
||
(define (lookup-hexpm-release version*)
|
||
"Look up RELEASE on hexpm-version-url and return the corresponding
|
||
<hexpm-release> record or #f if it was not found."
|
||
(and=> (json-fetch (hexpm-version-url version*))
|
||
json->hexpm-release))
|
||
|
||
|
||
;;;
|
||
;;; Converting hex.pm packages to Guix packages.
|
||
;;;
|
||
|
||
(define (maybe-inputs package-inputs input-type)
|
||
"Given a list of PACKAGE-INPUTS, tries to generate the 'inputs' field of a
|
||
package definition. INPUT-TYPE, a symbol, is used to populate the name of
|
||
the input field."
|
||
(match package-inputs
|
||
(()
|
||
'())
|
||
((package-inputs ...)
|
||
`((,input-type (list ,@package-inputs))))))
|
||
|
||
(define (dependencies->package-names names)
|
||
"Given a list of hexpm package NAMES, returns a list of guix package names
|
||
as symbols."
|
||
;; TODO: Base name on language of dependency.
|
||
;; The language used for implementing the dependency is not know without
|
||
;; recursing the dependencies. So for now assume more packages are based on
|
||
;; Erlang and prefix all dependencies with "erlang-" (the default).
|
||
(map string->symbol
|
||
(map hexpm-name->package-name
|
||
(sort names string-ci<?))))
|
||
|
||
(define* (make-hexpm-sexp #:key name version tarball-url
|
||
home-page synopsis description license
|
||
language build-system dependencies
|
||
#:allow-other-keys)
|
||
"Return the `package' s-expression for a hexpm package with the given NAME,
|
||
VERSION, TARBALL-URL, HOME-PAGE, SYNOPSIS, DESCRIPTION, and LICENSE. The
|
||
created package's name will stem from LANGUAGE. BUILD-SYSTEM defined the
|
||
build-system, and DEPENDENCIES the inputs for the package."
|
||
(call-with-temporary-output-file
|
||
(lambda (temp port)
|
||
(and (url-fetch tarball-url temp)
|
||
(values
|
||
`(package
|
||
(name ,(hexpm-name->package-name name language))
|
||
(version ,version)
|
||
(source (origin
|
||
(method url-fetch)
|
||
(uri (hexpm-uri ,name version))
|
||
(sha256 (base32 ,(guix-hash-url temp)))))
|
||
(build-system ,build-system)
|
||
,@(maybe-inputs (dependencies->package-names dependencies) 'inputs)
|
||
(synopsis ,synopsis)
|
||
(description ,(beautify-description description))
|
||
(home-page ,(match home-page
|
||
(() "")
|
||
(_ home-page)))
|
||
(license ,(match license
|
||
(() #f)
|
||
((license) license)
|
||
(_ `(list ,@license))))))))))
|
||
|
||
(define (strings->licenses strings)
|
||
"Convert the list of STRINGS into a list of license objects."
|
||
(filter-map (lambda (license)
|
||
(and (not (string-null? license))
|
||
(not (any (lambda (elem) (string=? elem license))
|
||
'("AND" "OR" "WITH")))
|
||
(or (spdx-string->license license)
|
||
license)))
|
||
strings))
|
||
|
||
(define (hexpm-releases package)
|
||
"Return the version strings for releases of PACKAGE, a <hexpm-pkgdef>
|
||
object, ordered from newest to oldest."
|
||
(sort (map hexpm-version-number (hexpm-versions package))
|
||
version>?))
|
||
|
||
(define* (hexpm->guix-package package-name #:key version #:allow-other-keys)
|
||
"Fetch the metadata for PACKAGE-NAME from hexpms.io, and return the
|
||
`package' s-expression corresponding to that package, or #f on failure.
|
||
When VERSION is specified, attempt to fetch that version; otherwise fetch the
|
||
latest version of PACKAGE-NAME."
|
||
|
||
(define package
|
||
(lookup-hexpm package-name))
|
||
|
||
(define version-number
|
||
(and package
|
||
(or version
|
||
(first (hexpm-releases package)))))
|
||
|
||
(define version*
|
||
(and package
|
||
(find (lambda (version)
|
||
(string=? (hexpm-version-number version)
|
||
version-number))
|
||
(hexpm-versions package))))
|
||
|
||
(define release
|
||
(and package version*
|
||
(lookup-hexpm-release version*)))
|
||
|
||
(define release-meta
|
||
(and package version*
|
||
(hexpm-release-meta release)))
|
||
|
||
(define build-system
|
||
(and package version*
|
||
(let ((build-tools (hexpm-release-meta-build-tools release-meta)))
|
||
(cond
|
||
((member "rebar3" build-tools) 'rebar-build-system)
|
||
((member "mix" build-tools) 'mix-build-system)
|
||
((member "make" build-tools) 'gnu-build-system)
|
||
(else #f)))))
|
||
|
||
(define language
|
||
(and package version*
|
||
(let ((elixir (hexpm-release-meta-elixir release-meta)))
|
||
(cond
|
||
((and (string? elixir) (not (string-null? elixir))) "elixir")
|
||
(else "erlang")))))
|
||
|
||
(and package version*
|
||
(let ((dependencies (hexpm-release-dependencies release))
|
||
(pkg-meta (hexpm-meta package))
|
||
(docs-html-url (hexpm-docs-html-url package)))
|
||
(values
|
||
(make-hexpm-sexp
|
||
#:language language
|
||
#:build-system build-system
|
||
#:name package-name
|
||
#:version version-number
|
||
#:dependencies dependencies
|
||
#:home-page (or (and (not (eq? docs-html-url 'null))
|
||
docs-html-url)
|
||
;; TODO: Homepage?
|
||
(hexpm-html-url package))
|
||
#:synopsis (hexpm-meta-description pkg-meta)
|
||
#:description (hexpm-meta-description pkg-meta)
|
||
#:license (or (and=> (hexpm-meta-licenses pkg-meta)
|
||
strings->licenses))
|
||
#:tarball-url (hexpm-uri package-name version-number))
|
||
dependencies))))
|
||
|
||
(define* (hexpm-recursive-import pkg-name #:optional version)
|
||
(recursive-import pkg-name
|
||
#:version version
|
||
#:repo->guix-package hexpm->guix-package
|
||
#:guix-name hexpm-name->package-name))
|
||
|
||
(define (guix-package->hexpm-name package)
|
||
"Return the hex.pm name of PACKAGE."
|
||
(define (url->hexpm-name url)
|
||
(hyphen-package-name->name+version
|
||
(basename (file-sans-extension url))))
|
||
|
||
(match (and=> (package-source package) origin-uri)
|
||
((? string? url)
|
||
(url->hexpm-name url))
|
||
((lst ...)
|
||
(any url->hexpm-name lst))
|
||
(#f #f)))
|
||
|
||
(define* (hexpm-name->package-name name #:optional (language "erlang"))
|
||
(string-append language "-" (string-join (string-split name #\_) "-")))
|
||
|
||
|
||
;;;
|
||
;;; Updater
|
||
;;;
|
||
|
||
(define* (import-release package #:key version partial-version?)
|
||
"Return an <upstream-source> for the latest release of PACKAGE. Optionally
|
||
include a VERSION string to fetch a specific version, which may be a version
|
||
prefix when PARTIAL-VERSION? is #t."
|
||
(let* ((hexpm-name (guix-package->hexpm-name package))
|
||
(hexpm (lookup-hexpm hexpm-name))
|
||
(latest-stable (hexpm-latest-stable hexpm))
|
||
(releases (hexpm-releases hexpm))
|
||
(version (find-version releases version partial-version?)))
|
||
(and version
|
||
(upstream-source
|
||
(package (package-name package))
|
||
(version version)
|
||
(urls (list (hexpm-uri hexpm-name version)))))))
|
||
|
||
(define %hexpm-updater
|
||
(upstream-updater
|
||
(name 'hexpm)
|
||
(description "Updater for hex.pm packages")
|
||
(pred (url-prefix-predicate hexpm-package-url))
|
||
(import import-release)))
|