123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366 |
- ;;; GNU Guix --- Functional package management for GNU
- ;;; Copyright © 2014 Eric Bavier <bavier@member.fsf.org>
- ;;; Copyright © 2015 Mark H Weaver <mhw@netris.org>
- ;;; Copyright © 2016 Alex Sassmannshausen <alex@pompo.co>
- ;;; Copyright © 2017, 2018 Tobias Geerinckx-Rice <me@tobias.gr>
- ;;; Copyright © 2020, 2021, 2023 Ludovic Courtès <ludo@gnu.org>
- ;;; Copyright © 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 cpan)
- #:use-module (ice-9 match)
- #:use-module (ice-9 regex)
- #:use-module ((ice-9 popen) #:select (open-pipe* close-pipe))
- #:use-module ((ice-9 rdelim) #:select (read-line))
- #:use-module (srfi srfi-1)
- #:use-module (srfi srfi-26)
- #:use-module (srfi srfi-34)
- #:use-module (json)
- #:use-module (gcrypt hash)
- #:use-module (guix diagnostics)
- #:use-module (guix i18n)
- #:use-module (guix store)
- #:use-module (guix utils)
- #:use-module (guix base32)
- #:use-module ((guix download) #:select (download-to-store url-fetch))
- #:use-module ((guix import utils) #:select (factorize-uri))
- #:use-module (guix import json)
- #:use-module (guix packages)
- #:use-module (guix upstream)
- #:use-module (guix derivations)
- #:export (cpan->guix-package
- metacpan-url->mirror-url
- %cpan-updater
- %metacpan-base-url))
- ;;; Commentary:
- ;;;
- ;;; Generate a package declaration template for the latest version of a CPAN
- ;;; module, using meta-data from metacpan.org.
- ;;;
- ;;; Code:
- (define %metacpan-base-url
- ;; Base URL of the MetaCPAN API.
- (make-parameter "https://fastapi.metacpan.org/v1/"))
- ;; Dependency of a "release".
- (define-json-mapping <cpan-dependency> make-cpan-dependency cpan-dependency?
- json->cpan-dependency
- (relationship cpan-dependency-relationship "relationship"
- string->symbol) ;requires | suggests
- (phase cpan-dependency-phase "phase"
- string->symbol) ;develop | configure | test | runtime
- (module cpan-dependency-module) ;string
- (version cpan-dependency-version)) ;string
- ;; Release as returned by <https://fastapi.metacpan.org/v1/release/PKG>.
- (define-json-mapping <cpan-release> make-cpan-release cpan-release?
- json->cpan-release
- (license cpan-release-license)
- (author cpan-release-author)
- (version cpan-release-version "version"
- (match-lambda
- ((? number? version)
- ;; Version is sometimes not quoted in the module json, so
- ;; it gets imported into Guile as a number, so convert it
- ;; to a string (example: "X11-Protocol-Other").
- (number->string version))
- ((? string? version)
- ;; Sometimes we get a "v" prefix. Strip it.
- (if (string-prefix? "v" version)
- (string-drop version 1)
- version))))
- (module cpan-release-module "main_module") ;e.g., "Test::Script"
- (distribution cpan-release-distribution) ;e.g., "Test-Script"
- (download-url cpan-release-download-url "download_url")
- (abstract cpan-release-abstract "abstract")
- (home-page cpan-release-home-page "resources"
- (match-lambda
- (#f #f)
- ((? unspecified?) #f)
- ((lst ...) (assoc-ref lst "homepage"))))
- (dependencies cpan-release-dependencies "dependency"
- (lambda (vector)
- (map json->cpan-dependency (vector->list vector)))))
- (define string->license
- (match-lambda
- ;; List of valid values from https://metacpan.org/pod/CPAN::Meta::Spec.
- ;; Some licenses are excluded based on their absense from (guix licenses).
- ("agpl_3" 'agpl3)
- ;; apache_1_1
- ("apache_2_0" 'asl2.0)
- ;; artistic_1
- ("artistic_2" 'artistic2.0)
- ("bsd" 'bsd-3)
- ("freebsd" 'bsd-2)
- ;; gfdl_1_2
- ("gfdl_1_3" 'fdl1.3+)
- ("gpl_1" 'gpl1)
- ("gpl_2" 'gpl2)
- ("gpl_3" 'gpl3)
- ("lgpl_2_1" 'lgpl2.1)
- ("lgpl_3_0" 'lgpl3)
- ("mit" 'x11)
- ;; mozilla_1_0
- ("mozilla_1_1" 'mpl1.1)
- ("openssl" 'openssl)
- ("perl_5" 'perl-license) ;GPL1+ and Artistic 1
- ("qpl_1_0" 'qpl)
- ;; ssleay
- ;; sun
- ("zlib" 'zlib)
- (#(x) (string->license x))
- (#(lst ...) `(list ,@(map string->license lst)))
- (_ #f)))
- (define (module->name module)
- "Transform a 'module' name into a 'release' name"
- (regexp-substitute/global #f "::" module 'pre "-" 'post))
- (define (module->dist-name module)
- "Return the base distribution module for a given module. E.g. the 'ok'
- module is distributed with 'Test::Simple', so (module->dist-name \"ok\") would
- return \"Test-Simple\""
- (assoc-ref (json-fetch (string-append
- (%metacpan-base-url) "/module/"
- module
- "?fields=distribution"))
- "distribution"))
- (define (package->upstream-name package)
- "Return the CPAN name of PACKAGE."
- (let* ((properties (package-properties package))
- (upstream-name (and=> properties
- (cut assoc-ref <> 'upstream-name))))
- (or upstream-name
- (match (package-source package)
- ((? origin? origin)
- (match (origin-uri origin)
- ((or (? string? url) (url _ ...))
- (match (string-match "([^/]*)-v?[0-9\\.]+" url)
- (#f #f)
- (m (match:substring m 1))))
- (_ #f)))
- (_ #f)))))
- (define (cpan-fetch name)
- "Return a <cpan-release> record for Perl module MODULE,
- or #f on failure. MODULE should be the distribution name, such as
- \"Test-Script\" for the \"Test::Script\" module."
- ;; This API always returns the latest release of the module.
- (and=> (json-fetch (string-append (%metacpan-base-url) "/release/"
- name))
- json->cpan-release))
- (define (cpan-home name)
- (string-append "https://metacpan.org/release/" name))
- (define (metacpan-url->mirror-url url)
- "Replace 'https://cpan.metacpan.org' in URL with 'mirror://cpan'."
- (regexp-substitute/global #f "http[s]?://cpan.metacpan.org"
- url
- 'pre "mirror://cpan" 'post))
- (define cpan-source-url
- (compose metacpan-url->mirror-url cpan-release-download-url))
- (define (perl-package)
- "Return the 'perl' package. This is a lazy reference so that we don't
- depend on (gnu packages perl)."
- (module-ref (resolve-interface '(gnu packages perl)) 'perl))
- (define %corelist
- (delay
- (let* ((perl (with-store store
- (derivation->output-path
- (package-derivation store (perl-package)))))
- (core (string-append perl "/bin/corelist")))
- (and (access? core X_OK)
- core))))
- (define core-module?
- (let ((rx (make-regexp
- (string-append "released with perl v?([0-9\\.]*)"
- "(.*and removed from v?([0-9\\.]*))?"))))
- (lambda (name)
- (define perl-version
- (package-version (perl-package)))
- (define (version-between? lower version upper)
- (and (version>=? version lower)
- (or (not upper)
- (version>? upper version))))
- (and (force %corelist)
- (parameterize ((current-error-port (%make-void-port "w")))
- (let* ((corelist (open-pipe* OPEN_READ (force %corelist) name)))
- (let loop ()
- (let ((line (read-line corelist)))
- (if (eof-object? line)
- (begin (close-pipe corelist) #f)
- (or (and=> (regexp-exec rx line)
- (lambda (m)
- (let ((first (match:substring m 1))
- (last (match:substring m 3)))
- (version-between?
- first perl-version last))))
- (loop)))))))))))
- (define (cpan-name->downstream-name name)
- "Return the Guix package name corresponding to NAME."
- (if (string-prefix? "perl-" name)
- (string-downcase name)
- (string-append "perl-" (string-downcase name))))
- (define (cran-dependency->upstream-input dependency)
- "Return the <upstream-input> corresponding to DEPENDENCY, or #f if
- DEPENDENCY denotes an implicit or otherwise unnecessary dependency."
- (match (cpan-dependency-module dependency)
- ("perl" #f) ;implicit dependency
- (module
- (let ((type (match (cpan-dependency-phase dependency)
- ((or 'configure 'build 'test)
- ;; "runtime" may also be needed here. See
- ;; https://metacpan.org/pod/CPAN::Meta::Spec#Phases,
- ;; which says they are required during
- ;; building. We have not yet had a need for
- ;; cross-compiled Perl modules, however, so
- ;; we leave it out.
- 'native)
- ('runtime
- 'propagated)
- (_
- #f))))
- (and type
- (not (core-module? module)) ;expensive call!
- (upstream-input
- (name (module->dist-name module))
- (downstream-name (cpan-name->downstream-name name))
- (type type)))))))
- (define (cpan-module-inputs release)
- "Return the list of <upstream-input> for dependencies of RELEASE, a
- <cpan-release>."
- (define (upstream-input<? a b)
- (string<? (upstream-input-downstream-name a)
- (upstream-input-downstream-name b)))
- (sort (delete-duplicates
- (filter-map cran-dependency->upstream-input
- (cpan-release-dependencies release)))
- upstream-input<?))
- (define (cpan-module->sexp release)
- "Return the 'package' s-expression for a CPAN module from the release data
- in RELEASE, a <cpan-release> record."
- (define name
- (cpan-release-distribution release))
- (define version (cpan-release-version release))
- (define source-url (cpan-source-url release))
- (define (maybe-inputs input-type inputs)
- (match inputs
- (()
- '())
- ((inputs ...)
- `((,input-type (list ,@(map (compose string->symbol
- upstream-input-downstream-name)
- inputs)))))))
- (let ((tarball (with-store store
- (download-to-store store source-url)))
- (inputs (cpan-module-inputs release)))
- `(package
- (name ,(cpan-name->downstream-name name))
- (version ,version)
- (source (origin
- (method url-fetch)
- (uri (string-append ,@(factorize-uri source-url version)))
- (sha256
- (base32
- ,(bytevector->nix-base32-string (file-sha256 tarball))))))
- (build-system perl-build-system)
- ,@(maybe-inputs 'native-inputs
- (filter (upstream-input-type-predicate 'native)
- inputs))
- ,@(maybe-inputs 'propagated-inputs
- (filter (upstream-input-type-predicate 'propagated)
- inputs))
- (home-page ,(cpan-home name))
- (synopsis ,(cpan-release-abstract release))
- (description fill-in-yourself!)
- (license ,(string->license (cpan-release-license release))))))
- (define (cpan->guix-package module-name)
- "Fetch the metadata for PACKAGE-NAME from metacpan.org, and return the
- `package' s-expression corresponding to that package, or #f on failure."
- (let ((release (cpan-fetch (module->name module-name))))
- (and=> release cpan-module->sexp)))
- (define cpan-package?
- (let ((cpan-rx (make-regexp (string-append "("
- "mirror://cpan" "|"
- "https?://www.cpan.org" "|"
- "https?://cpan.metacpan.org"
- ")"))))
- (url-predicate (cut regexp-exec cpan-rx <>))))
- (define* (latest-release package #:key (version #f))
- "Return an <upstream-source> for the latest release of PACKAGE."
- (when version
- (raise
- (formatted-message
- (G_ "~a updater doesn't support updating to a specific version, sorry.")
- "cpan")))
- (match (cpan-fetch (package->upstream-name package))
- (#f #f)
- (release
- (let ((core-inputs
- (match (package-direct-inputs package)
- (((_ inputs _ ...) ...)
- (filter-map (match-lambda
- ((and (? package?)
- (? cpan-package?)
- (= package->upstream-name
- (? core-module? name)))
- name)
- (else #f))
- inputs)))))
- ;; Warn about inputs that are part of perl's core
- (unless (null? core-inputs)
- (for-each (lambda (module)
- (warning (G_ "input '~a' of ~a is in Perl core~%")
- module (package-name package)))
- core-inputs)))
- (let ((version (cpan-release-version release))
- (url (cpan-source-url release)))
- (upstream-source
- (package (package-name package))
- (version version)
- (urls (list url))
- (inputs (cpan-module-inputs release)))))))
- (define %cpan-updater
- (upstream-updater
- (name 'cpan)
- (description "Updater for CPAN packages")
- (pred cpan-package?)
- (import latest-release)))
|