;;; GNU Guix --- Functional package management for GNU ;;; Copyright © 2020 Katherine Cox-Buday ;;; Copyright © 2020 Helio Machado <0x2b3bfa0+guix@googlemail.com> ;;; Copyright © 2021 François Joulaud ;;; Copyright © 2021 Maxim Cournoyer ;;; Copyright © 2021-2022 Ludovic Courtès ;;; Copyright © 2021 Xinglu Chen ;;; Copyright © 2021 Sarah Morgensen ;;; Copyright © 2021 Simon Tournier ;;; Copyright © 2023 Efraim Flashner ;;; Copyright © 2024 Christina O'Donnell ;;; ;;; 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 . (define-module (guix import go) #:use-module (guix build-system go) #:use-module (guix git) #:use-module (guix hash) #:use-module (guix i18n) #:use-module ((guix utils) #:select (version>?)) #:use-module (guix diagnostics) #:use-module (guix import utils) #:use-module (guix import json) #:use-module (guix packages) #:use-module (guix http-client) #:use-module (guix memoization) #:autoload (htmlprag) (html->sxml) ;from Guile-Lib #:autoload (guix base32) (bytevector->nix-base32-string) #:autoload (guix build utils) (mkdir-p) #:autoload (guix ui) (warning) #:autoload (gcrypt hash) (hash-algorithm sha256) #:autoload (git structs) (git-error-message) #:use-module (ice-9 format) #:use-module (ice-9 match) #:use-module (ice-9 peg) #:use-module (ice-9 receive) #:use-module (ice-9 regex) #:use-module (ice-9 textual-ports) #:use-module ((rnrs io ports) #:select (call-with-port)) #:use-module (srfi srfi-1) #:use-module (srfi srfi-9) #:use-module (srfi srfi-11) #:use-module (srfi srfi-26) #:use-module (srfi srfi-34) #:use-module (srfi srfi-35) #:use-module (sxml match) #:use-module ((sxml xpath) #:renamer (lambda (s) (if (eq? 'filter s) 'xfilter s))) #:use-module (web uri) #:export (go-module->guix-package go-module->guix-package* go-module-recursive-import)) ;;; Commentary: ;;; ;;; (guix import go) attempts to make it easier to create Guix package ;;; declarations for Go modules. ;;; ;;; Modules in Go are a "collection of related Go packages" which are "the ;;; unit of source code interchange and versioning". Modules are generally ;;; hosted in a repository. ;;; ;;; Monorepo is a collection of modules within the same VCS source. Each ;;; module of monorepo may be released individually by assigning ;;; "/v" tag (see: https://go.dev/ref/mod#modules-overview). ;;; ;;; At this point it should handle correctly modules which have only Go ;;; dependencies and are accessible from proxy.golang.org (or configured via ;;; GOPROXY). ;;; ;;; We want it to work more or less this way: ;;; - get latest version for the module from GOPROXY ;;; - infer VCS root repo from which we will check-out source by ;;; + recognising known patterns (like github.com) ;;; + or recognizing .vcs suffix ;;; + or parsing meta tag in HTML served at the URL ;;; + or (TODO) if nothing else works by using zip file served by GOPROXY ;;; - get go.mod from GOPROXY (which is able to synthetize one if needed) ;;; - extract list of dependencies from this go.mod ;;; ;;; The Go module paths are translated to a Guix package name under the ;;; assumption that there will be no collision. ;;; TODO list ;;; - get correct hash in vcs->origin for Mercurial and Subversion ;;; Code: (define (go-package) "Return the 'go' package. This is a lazy reference so that we don't depend on (gnu packages golang)." (module-ref (resolve-interface '(gnu packages golang)) 'go)) (define http-fetch* ;; Like http-fetch, but memoized and returning the body as a string. (memoize (lambda args (call-with-port (apply http-fetch args) get-string-all)))) (define json-fetch* (memoize json-fetch)) (define (go-path-escape path) "Escape a module path by replacing every uppercase letter with an exclamation mark followed with its lowercase equivalent, as per the module Escaped Paths specification (see: https://godoc.org/golang.org/x/mod/module#hdr-Escaped_Paths)." (define (escape occurrence) (string-append "!" (string-downcase (match:substring occurrence)))) (regexp-substitute/global #f "[A-Z]" path 'pre escape 'post)) ;; Prevent inlining of this procedure, which is accessed by unit tests. (set! go-path-escape go-path-escape) (define (go.pkg.dev-info name) (http-fetch* (string-append "https://pkg.go.dev/" name))) (define* (go-module-version-info goproxy name #:key version) "Fetch a JSON object encoding about the lastest version for NAME from the given GOPROXY server, or for VERSION when specified." (let ((file (if version (string-append "@v/" version ".info") "@latest"))) (json-fetch* (format #f "~a/~a/~a" goproxy (go-path-escape name) file)))) (define* (go-module-available-versions goproxy name) "Retrieve the available versions for a given module from the module proxy. Versions are being returned **unordered** and may contain different versioning styles for the same package." (let* ((url (string-append goproxy "/" (go-path-escape name) "/@v/list")) (body (http-fetch* url)) (versions (remove string-null? (string-split body #\newline)))) (if (null? versions) (begin (warning (G_ "Empty list of versions on proxy ~a for package '~a'. Using latest.~%") goproxy name) ;; If we haven't recieved any versions, look in the version-info json ;; object and return a one-element list if found. (or (and=> (assoc-ref (go-module-version-info goproxy name) "Version") list) (raise (make-compound-condition (formatted-message (G_ "No versions available for '~a' on proxy ~a.") name goproxy)))))) versions)) (define (go-package-licenses name) "Retrieve the list of licenses that apply to NAME, a Go package or module name (e.g. \"github.com/golang/protobuf/proto\")." (let* ((body (go.pkg.dev-info (string-append name "?tab=licenses"))) ;; Extract the text contained in a h2 child node of any ;; element marked with a "License" class attribute. (select (sxpath `(// (* (@ (equal? (class "License")))) h2 // div // *text*)))) (select (html->sxml body #:strict? #t)))) (define (sxml->texi sxml-node) "A very basic SXML to Texinfo converter which attempts to preserve HTML formatting and links as text." (sxml-match sxml-node ((strong ,text) (format #f "@strong{~a}" text)) ((a (@ (href ,url)) ,text) (format #f "@url{~a,~a}" url text)) ((code ,text) (format #f "@code{~a}" text)) (,something-else something-else))) (define (go-package-description name) "Retrieve a short description for NAME, a Go package name, e.g. \"google.golang.org/protobuf/proto\"." (let* ((body (go.pkg.dev-info name)) (sxml (html->sxml body #:strict? #t)) (overview ((sxpath `(// (* (@ (equal? (class "Documentation-overview")))) (p 1))) sxml)) ;; Sometimes, the first paragraph just contains images/links that ;; has only "\n" for text. The following filter is designed to ;; omit it. (contains-text? (lambda (node) (remove string-null? (map string-trim-both (filter (node-typeof? '*text*) (cdr node)))))) (select-content (sxpath `(// (* (@ (equal? (class "UnitReadme-content")))) div // p ,(xfilter contains-text?)))) ;; Fall-back to use content; this is less desirable as it is more ;; verbose, but not every page has an overview. (description (if (not (null? overview)) overview (select-content sxml))) (description* (if (not (null? description)) (first description) description))) (match description* (() #f) ;nothing selected ((p elements ...) (apply string-append (filter string? (map sxml->texi elements))))))) (define (go-package-synopsis module-name) "Retrieve a short synopsis for a Go module named MODULE-NAME, e.g. \"google.golang.org/protobuf\". The data is scraped from the https://pkg.go.dev/ web site." ;; Note: Only the *module* (rather than package) page has the README title ;; used as a synopsis on the https://pkg.go.dev web site. (let* ((url (string-append "https://pkg.go.dev/" module-name)) (body (http-fetch* url)) ;; Extract the text contained in a h2 child node of any ;; element marked with a "License" class attribute. (select-title (sxpath `(// (div (@ (equal? (class "UnitReadme-content")))) // h3 *text*)))) (match (select-title (html->sxml body #:strict? #t)) (() #f) ;nothing selected ((title more ...) ;title is the first string of the list (string-trim-both title))))) (define (list->licenses licenses) "Given a list of LICENSES mostly following the SPDX conventions, return the corresponding Guix license or 'unknown-license!" (filter-map (lambda (license) (and (not (string-null? license)) (not (any (cut string=? <> license) '("AND" "OR" "WITH"))) ;; Adjust the license names scraped from ;; https://pkg.go.dev to an equivalent SPDX identifier, ;; if they differ (see: https://github.com/golang/pkgsite ;; /internal/licenses/licenses.go#L174). (or (spdx-string->license (match license ("BlueOak-1.0" "BlueOak-1.0.0") ("BSD-0-Clause" "0BSD") ("BSD-2-Clause" "BSD-2-Clause-FreeBSD") ("GPL2" "GPL-2.0") ("GPL3" "GPL-3.0") ("NIST" "NIST-PD") (_ license))) 'unknown-license!))) licenses)) (define (fetch-go.mod goproxy module-path version) "Fetch go.mod from the given GOPROXY server for the given MODULE-PATH and VERSION and return an input port." (let ((url (format #f "~a/~a/@v/~a.mod" goproxy (go-path-escape module-path) (go-path-escape version)))) (http-fetch* url))) (define (parse-go.mod content) "Parse the go.mod file CONTENT, returning a list of directives, comments, and unknown lines. Each sublist begins with a symbol (go, module, require, replace, exclude, retract, comment, or unknown) and is followed by one or more sublists. Each sublist begins with a symbol (module-path, version, file-path, comment, or unknown) and is followed by the indicated data." ;; https://golang.org/ref/mod#go-mod-file-grammar (define-peg-pattern NL none "\n") (define-peg-pattern WS none (or " " "\t" "\r")) (define-peg-pattern => none (and (* WS) "=>")) (define-peg-pattern punctuation none (or "," "=>" "[" "]" "(" ")")) (define-peg-pattern comment all (and (ignore "//") (* WS) (* (and (not-followed-by NL) peg-any)))) (define-peg-pattern EOL body (and (* WS) (? comment) NL)) (define-peg-pattern block-start none (and (* WS) "(" EOL)) (define-peg-pattern block-end none (and (* WS) ")" EOL)) (define-peg-pattern any-line body (and (* WS) (* (and (not-followed-by NL) peg-any)) EOL)) ;; Strings and identifiers (define-peg-pattern identifier body (+ (and (not-followed-by (or NL WS punctuation)) peg-any))) (define-peg-pattern string-raw body (and (ignore "`") (+ (and (not-followed-by "`") peg-any)) (ignore "`"))) (define-peg-pattern string-quoted body (and (ignore "\"") (+ (or (and (ignore "\\") peg-any) (and (not-followed-by "\"") peg-any))) (ignore "\""))) (define-peg-pattern string-or-ident body (and (* WS) (or string-raw string-quoted identifier))) (define-peg-pattern version all string-or-ident) (define-peg-pattern module-path all string-or-ident) (define-peg-pattern file-path all string-or-ident) ;; Non-directive lines (define-peg-pattern unknown all any-line) (define-peg-pattern block-line body (or EOL (and (not-followed-by block-end) unknown))) ;; GoDirective = "go" GoVersion newline . (define-peg-pattern go all (and (ignore "go") version EOL)) ;; ModuleDirective = "module" ( ModulePath | "(" newline ModulePath newline ")" ) newline . (define-peg-pattern module all (and (ignore "module") (or (and block-start module-path EOL block-end) (and module-path EOL)))) ;; The following directives may all be used solo or in a block ;; RequireSpec = ModulePath Version newline . (define-peg-pattern require all (and module-path version ;; We don't want the transitive dependencies. (not-followed-by (and (* WS) "//" (* WS) "indirect")) EOL)) (define-peg-pattern require-top body (and (ignore "require") (or (and block-start (* (or require block-line)) block-end) require))) ;; ExcludeSpec = ModulePath Version newline . (define-peg-pattern exclude all (and module-path version EOL)) (define-peg-pattern exclude-top body (and (ignore "exclude") (or (and block-start (* (or exclude block-line)) block-end) exclude))) ;; ReplaceSpec = ModulePath [ Version ] "=>" FilePath newline ;; | ModulePath [ Version ] "=>" ModulePath Version newline . (define-peg-pattern original all (or (and module-path version) module-path)) (define-peg-pattern with all (or (and module-path version) file-path)) (define-peg-pattern replace all (and original => with EOL)) (define-peg-pattern replace-top body (and (ignore "replace") (or (and block-start (* (or replace block-line)) block-end) replace))) ;; RetractSpec = ( Version | "[" Version "," Version "]" ) newline . (define-peg-pattern range all (and (* WS) (ignore "[") version (* WS) (ignore ",") version (* WS) (ignore "]"))) (define-peg-pattern retract all (and (or range version) EOL)) (define-peg-pattern retract-top body (and (ignore "retract") (or (and block-start (* (or retract block-line)) block-end) retract))) (define-peg-pattern go-mod body (* (and (* WS) (or go module require-top exclude-top replace-top retract-top EOL unknown)))) (let ((tree (peg:tree (match-pattern go-mod content))) (keywords '(go module require replace exclude retract comment unknown))) (keyword-flatten keywords tree))) ;; Prevent inlining of this procedure, which is accessed by unit tests. (set! parse-go.mod parse-go.mod) (define (go.mod-directives go.mod directive) "Return the list of top-level directive bodies in GO.MOD matching the symbol DIRECTIVE." (filter-map (match-lambda (((? (cut eq? <> directive) head) . rest) rest) (_ #f)) go.mod)) (define (go.mod-requirements go.mod) "Compute and return the list of requirements specified by GO.MOD." (define (replace directive requirements) (define (maybe-replace module-path new-requirement) ;; Do not allow version updates for indirect dependencies (see: ;; https://golang.org/ref/mod#go-mod-file-replace). (if (and (equal? module-path (first new-requirement)) (not (assoc-ref requirements module-path))) requirements (cons new-requirement (alist-delete module-path requirements)))) (match directive ((('original ('module-path module-path) . _) with . _) (match with (('with ('file-path _) . _) (alist-delete module-path requirements)) (('with ('module-path new-module-path) ('version new-version) . _) (maybe-replace module-path (list new-module-path new-version))))))) (define (require directive requirements) (match directive ((('module-path module-path) ('version version) . _) (cons (list module-path version) requirements)))) (let* ((requires (go.mod-directives go.mod 'require)) (replaces (go.mod-directives go.mod 'replace)) (requirements (fold require '() requires))) (fold replace requirements replaces))) ;; Prevent inlining of this procedure, which is accessed by unit tests. (set! go.mod-requirements go.mod-requirements) (define (go.mod-go-version go.mod) "Return the minimum version of go required to specified by GO.MOD." (let ((go-version (go.mod-directives go.mod 'go))) (if (null? go-version) ;; If the go directive is missing, go 1.16 is assumed. '(version "1.16") (flatten go-version)))) ;; Prevent inlining of this procedure, which is accessed by unit tests. (set! go.mod-go-version go.mod-go-version) (define-record-type (%make-vcs url-prefix root-regex type) vcs? (url-prefix vcs-url-prefix) (root-regex vcs-root-regex) (type vcs-type)) (define (make-vcs prefix regexp type) (%make-vcs prefix (make-regexp regexp) type)) (define known-vcs ;; See the following URL for the official Go equivalent: ;; https://github.com/golang/go/blob/846dce9d05f19a1f53465e62a304dea21b99f910/src/cmd/go/internal/vcs/vcs.go#L1026-L1087 (list (make-vcs "github.com" "^(github\\.com/[A-Za-z0-9_.\\-]+/[A-Za-z0-9_.\\-]+)(/[A-Za-z0-9_.\\-]+)*$" 'git) (make-vcs "bitbucket.org" "^(bitbucket\\.org/([A-Za-z0-9_.\\-]+/[A-Za-z0-9_.\\-]+))(/[A-Za-z0-9_.\\-]+)*$" 'unknown) (make-vcs "hub.jazz.net/git/" "^(hub\\.jazz\\.net/git/[a-z0-9]+/[A-Za-z0-9_.\\-]+)(/[A-Za-z0-9_.\\-]+)*$" 'git) (make-vcs "git.apache.org" "^(git\\.apache\\.org/[a-z0-9_.\\-]+\\.git)(/[A-Za-z0-9_.\\-]+)*$" 'git) (make-vcs "git.openstack.org" "^(git\\.openstack\\.org/[A-Za-z0-9_.\\-]+/[A-Za-z0-9_.\\-]+)(\\.git)?\ (/[A-Za-z0-9_.\\-]+)*$" 'git))) (define (module-path->repository-root module-path version-info) "Infer the repository root from a module path. Go modules can be defined at any level of a repository tree, but querying for the meta tag usually can only be done from the web page at the root of the repository, hence the need to derive this information." ;; For reference, see: https://golang.org/ref/mod#vcs-find. (define vcs-qualifiers '(".bzr" ".fossil" ".git" ".hg" ".svn")) (define (vcs-qualified-module-path->root-repo-url module-path) (let* ((vcs-qualifiers-group (string-join vcs-qualifiers "|")) (pattern (format #f "^(.*(~a))(/|$)" vcs-qualifiers-group)) (m (string-match pattern module-path))) (and=> m (cut match:substring <> 1)))) (or (and=> (find (lambda (vcs) (string-prefix? (vcs-url-prefix vcs) module-path)) known-vcs) (lambda (vcs) (match:substring (regexp-exec (vcs-root-regex vcs) module-path) 1))) (and=> (assoc-ref version-info "Origin") (lambda (origin) (and=> (assoc-ref origin "Subdir") (lambda (subdir) ;; If version-info contains a 'subdir' and that is a suffix, ;; then the repo-root can be found by stripping off the ;; suffix. (if (string-suffix? (string-append "/" subdir) module-path) (string-drop-right module-path (+ 1 (string-length subdir))) #f))))) (vcs-qualified-module-path->root-repo-url module-path) (begin (warning (G_ "Unable to determine repository root of '~a'. Guessing '~a'.~%") module-path module-path) module-path))) (define* (go-module->guix-package-name module-path #:optional version) "Converts a module's path to the canonical Guix format for Go packages. Optionally include a VERSION string to append to the name." ;; Map dot, slash, underscore and tilde characters to hyphens. (let ((module-path* (string-map (lambda (c) (if (member c '(#\. #\/ #\_ #\~)) #\- c)) module-path))) (string-downcase (string-append "go-" module-path* (if version (string-append "-" version) ""))))) (define (strip-.git-suffix/maybe repo-url) "Strip a repository URL '.git' suffix from REPO-URL if hosted at GitHub." (match repo-url ((and (? (cut string-prefix? "https://github.com" <>)) (? (cut string-suffix? ".git" <>))) (string-drop-right repo-url 4)) (_ repo-url))) (define-record-type (make-module-meta import-prefix vcs repo-root) module-meta? (import-prefix module-meta-import-prefix) (vcs module-meta-vcs) ;a symbol (repo-root module-meta-repo-root)) (define (fetch-module-meta-data module-path) "Retrieve the module meta-data from its landing page. This is necessary because goproxy servers don't currently provide all the information needed to build a package." (define (go-import->module-meta content-text) (match (string-tokenize content-text char-set:graphic) ((root-path vcs repo-url) (make-module-meta root-path (string->symbol vcs) (strip-.git-suffix/maybe repo-url))))) ;; (let* ((meta-data (http-fetch* (format #f "https://~a?go-get=1" module-path))) (select (sxpath `(// (meta (@ (equal? (name "go-import")))) // content)))) (match (select (html->sxml meta-data #:strict? #t)) (() (raise (make-compound-condition (formatted-message (G_ "no element in result when accessing module path '~a' using go-get") module-path)))) ((('content content-text) ..1) (or (find (lambda (meta) (string-prefix? (module-meta-import-prefix meta) module-path)) (map go-import->module-meta content-text)) ;; Fallback to the first meta if no import prefixes match. (go-import->module-meta (first content-text)) (raise (make-compound-condition (formatted-message (G_ "unable to parse when accessing module path '~a' using go-get") module-path)))))))) (define (module-meta-data-repo-url meta-data goproxy) "Return the URL where the fetcher which will be used can download the source." (if (member (module-meta-vcs meta-data) '(fossil mod)) goproxy (module-meta-repo-root meta-data))) (define* (git-checkout-hash url reference algorithm) "Return the ALGORITHM hash of the checkout of URL at REFERENCE, a commit or tag." (define cache (string-append (or (getenv "TMPDIR") "/tmp") "/guix-import-go-" (passwd:name (getpwuid (getuid))))) ;; Use a custom cache to avoid cluttering the default one under ;; ~/.cache/guix, but choose one under /tmp so that it's persistent across ;; subsequent "guix import" invocations. (mkdir-p cache) (chmod cache #o700) (let-values (((checkout commit _) (parameterize ((%repository-cache-directory cache)) (update-cached-checkout url #:ref `(tag-or-commit . ,reference))))) (file-hash* checkout #:algorithm algorithm #:recursive? #true))) (define (vcs->origin vcs-type vcs-repo-url version subdir) "Generate the `origin' block of a package depending on what type of source control system is being used." (case vcs-type ((git) (let* ((plain-version? (string=? version (go-version->git-ref version #:subdir subdir))) (v-prefixed? (string-prefix? "v" version)) ;; This is done because the version field of the package, ;; which the generated quoted expression refers to, has been ;; stripped of any 'v' prefixed. (version-expr (if (and plain-version? v-prefixed?) '(string-append "v" version) `(go-version->git-ref version ,@(if subdir `(#:subdir ,subdir) '()))))) `(origin (method git-fetch) (uri (git-reference (url ,vcs-repo-url) ;; This is done because the version field of the package, ;; which the generated quoted expression refers to, has been ;; stripped of any 'v' prefixed. (commit ,version-expr))) (file-name (git-file-name name version)) (sha256 (base32 ,(bytevector->nix-base32-string (git-checkout-hash vcs-repo-url (go-version->git-ref version #:subdir subdir) (hash-algorithm sha256)))))))) ((hg) `(origin (method hg-fetch) (uri (hg-reference (url ,vcs-repo-url) (changeset ,version))) (file-name (string-append name "-" version "-checkout")) (sha256 (base32 ;; FIXME: populate hash for hg repo checkout "0000000000000000000000000000000000000000000000000000")))) ((svn) `(origin (method svn-fetch) (uri (svn-reference (url ,vcs-repo-url) (revision (string->number version)))) (file-name (string-append name "-" version "-checkout")) (sha256 (base32 ;; FIXME: populate hash for svn repo checkout "0000000000000000000000000000000000000000000000000000")))) (else (raise (formatted-message (G_ "unsupported vcs type '~a' for package '~a'") vcs-type vcs-repo-url))))) (define (strip-v-prefix version) "Strip from VERSION the \"v\" prefix that Go uses." (string-trim version #\v)) (define (ensure-v-prefix version) "Add a \"v\" prefix to VERSION if it does not already have one." (if (string-prefix? "v" version) version (string-append "v" version))) (define (validate-version version available-versions module-path) "Raise an error if VERSION is not among AVAILABLE-VERSIONS, unless VERSION is a pseudo-version. Return VERSION." ;; Pseudo-versions do not appear in the versions list; skip the ;; following check. (if (or (go-pseudo-version? version) (member version available-versions)) version (raise (make-compound-condition (formatted-message (G_ "version ~a of ~a is not available~%") version module-path available-versions) (condition (&fix-hint (hint (format #f (G_ "Pick one of the following \ available versions:~{ ~a~}.") (map strip-v-prefix available-versions))))))))) (define (path-diff parent child) (if (and (string-prefix? parent child) (not (string=? parent child))) (let ((parent-len (string-length parent))) (string-trim (substring child parent-len) (char-set #\/))) #f)) (define* (go-module->guix-package module-path #:key (goproxy "https://proxy.golang.org") version pin-versions? #:allow-other-keys) "Return the package S-expression corresponding to MODULE-PATH at VERSION, a Go package. The meta-data is fetched from the GOPROXY server and https://pkg.go.dev/. When VERSION is unspecified, the latest version available is used." (let* ((available-versions (go-module-available-versions goproxy module-path)) (version* (validate-version (or (and version (ensure-v-prefix version)) (assoc-ref (go-module-version-info goproxy module-path) "Version")) ;latest available-versions module-path)) (version-info (go-module-version-info goproxy module-path #:version version*)) (content (fetch-go.mod goproxy module-path version*)) (min-go-version (second (go.mod-go-version (parse-go.mod content)))) (dependencies+versions (go.mod-requirements (parse-go.mod content))) (dependencies (if pin-versions? dependencies+versions (map car dependencies+versions))) (module-path-sans-suffix (match:prefix (string-match "([\\./]v[0-9]+)?$" module-path))) (guix-name (go-module->guix-package-name module-path-sans-suffix )) (root-module-path (module-path->repository-root module-path-sans-suffix version-info)) ;; The VCS type and URL are not included in goproxy information. For ;; this we need to fetch it from the official module page. (meta-data (fetch-module-meta-data root-module-path)) (subdir (path-diff root-module-path module-path-sans-suffix)) (vcs-type (module-meta-vcs meta-data)) (vcs-repo-url (module-meta-data-repo-url meta-data goproxy)) (synopsis (go-package-synopsis module-path)) (description (go-package-description module-path)) (licenses (go-package-licenses module-path))) (values `(package (name ,guix-name) (version ,(strip-v-prefix version*)) (source ,(vcs->origin vcs-type vcs-repo-url version* subdir)) (build-system go-build-system) (arguments (list ,@(if (version>? min-go-version (package-version (go-package))) `(#:go ,(string->number min-go-version)) '()) #:import-path ,module-path ,@(if (string=? module-path root-module-path) '() `(#:unpack-path ,root-module-path)))) ,@(maybe-propagated-inputs (map (match-lambda ((name version) (go-module->guix-package-name name (strip-v-prefix version))) (name (go-module->guix-package-name name))) dependencies)) (home-page ,(format #f "https://~a" root-module-path)) (synopsis ,synopsis) (description ,(and=> description beautify-description)) (license ,(match (list->licenses licenses) (() #f) ;unknown license ((license) ;a single license license) ((license ...) ;a list of licenses `(list ,@license))))) (if pin-versions? dependencies+versions dependencies)))) (define go-module->guix-package* (lambda args ;; Disable output buffering so that the following warning gets printed ;; consistently. (setvbuf (current-error-port) 'none) (let ((package-name (match args ((name _ ...) name)))) (begin (info (G_ "Importing package ~s...~%") package-name) (guard (c ((http-get-error? c) (warning (G_ "Failed to import package ~s. reason: ~s could not be fetched: HTTP error ~a (~s). This package and its dependencies won't be imported.~%") package-name (uri->string (http-get-error-uri c)) (http-get-error-code c) (http-get-error-reason c)) (values #f '())) ((formatted-message? c) (warning (G_ "Failed to import package ~s. reason: ~a This package and its dependencies won't be imported.~%") package-name (apply format #f (formatted-message-string c) (formatted-message-arguments c))) (values #f '())) ((eq? (exception-kind c) 'git-error) (warning (G_ "Failed to import package ~s. reason: ~a This package and its dependencies won't be imported.~%") package-name (git-error-message c)) (values #f '()))) (apply go-module->guix-package args)))))) (define* (go-module-recursive-import package-name #:key (goproxy "https://proxy.golang.org") version pin-versions?) (recursive-import package-name #:repo->guix-package (memoize (lambda* (name #:key version repo #:allow-other-keys) (receive (package-sexp dependencies) (go-module->guix-package* name #:goproxy goproxy #:version version #:pin-versions? pin-versions?) (values package-sexp dependencies)))) #:guix-name go-module->guix-package-name #:version version))