2014-01-23 17:48:34 -05:00
|
|
|
|
;;; GNU Guix --- Functional package management for GNU
|
2022-01-03 13:31:05 -05:00
|
|
|
|
;;; Copyright © 2014-2022 Ludovic Courtès <ludo@gnu.org>
|
2017-12-13 17:42:40 -05:00
|
|
|
|
;;; Copyright © 2017 Ricardo Wurmus <rekado@elephly.net>
|
2020-08-24 16:26:14 -04:00
|
|
|
|
;;; Copyright © 2020 Maxim Cournoyer <maxim.cournoyer@gmail.com>
|
2020-12-03 14:22:50 -05:00
|
|
|
|
;;; Copyright © 2020 Julien Lepiller <julien@lepiller.eu>
|
2014-01-23 17:48:34 -05: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 scripts offload)
|
2021-12-15 06:45:11 -05:00
|
|
|
|
#:autoload (ssh key) (private-key-from-file
|
|
|
|
|
public-key-from-file)
|
|
|
|
|
#:autoload (ssh auth) (userauth-public-key!)
|
|
|
|
|
#:autoload (ssh session) (make-session
|
|
|
|
|
connect! get-error
|
2022-11-22 01:17:17 -05:00
|
|
|
|
disconnect! session-set! session-get)
|
2021-12-15 06:45:11 -05:00
|
|
|
|
#:autoload (ssh version) (zlib-support?)
|
2014-01-23 17:48:34 -05:00
|
|
|
|
#:use-module (guix config)
|
|
|
|
|
#:use-module (guix records)
|
2021-12-15 06:45:11 -05:00
|
|
|
|
#:autoload (guix ssh) (authenticate-server*
|
|
|
|
|
connect-to-remote-daemon
|
|
|
|
|
send-files retrieve-files retrieve-files*
|
|
|
|
|
remote-inferior report-guile-error)
|
2014-01-23 17:48:34 -05:00
|
|
|
|
#:use-module (guix store)
|
2022-11-22 01:17:17 -05:00
|
|
|
|
#:autoload (guix inferior) (inferior-eval close-inferior
|
|
|
|
|
inferior? inferior-protocol-error?)
|
2021-12-15 06:45:11 -05:00
|
|
|
|
#:autoload (guix derivations) (read-derivation-from-file
|
|
|
|
|
derivation-file-name
|
|
|
|
|
build-derivations)
|
|
|
|
|
#:autoload (guix nar) (restore-file-set)
|
2020-07-25 12:26:18 -04:00
|
|
|
|
#:use-module ((guix utils) #:select (%current-system))
|
2017-05-28 10:09:32 -04:00
|
|
|
|
#:use-module ((guix build syscalls)
|
|
|
|
|
#:select (fcntl-flock set-thread-name))
|
2014-03-06 15:38:45 -05:00
|
|
|
|
#:use-module ((guix build utils) #:select (which mkdir-p))
|
2014-01-23 17:48:34 -05:00
|
|
|
|
#:use-module (guix ui)
|
2020-09-01 16:13:11 -04:00
|
|
|
|
#:use-module (guix scripts)
|
2020-07-25 12:26:18 -04:00
|
|
|
|
#:use-module (guix diagnostics)
|
2014-01-23 17:48:34 -05:00
|
|
|
|
#:use-module (srfi srfi-1)
|
2014-03-26 11:22:41 -04:00
|
|
|
|
#:use-module (srfi srfi-11)
|
2014-01-23 17:48:34 -05:00
|
|
|
|
#:use-module (srfi srfi-26)
|
|
|
|
|
#:use-module (srfi srfi-34)
|
|
|
|
|
#:use-module (srfi srfi-35)
|
|
|
|
|
#:use-module (ice-9 rdelim)
|
|
|
|
|
#:use-module (ice-9 match)
|
|
|
|
|
#:use-module (ice-9 regex)
|
|
|
|
|
#:use-module (ice-9 format)
|
|
|
|
|
#:export (build-machine
|
2020-10-24 18:14:00 -04:00
|
|
|
|
build-machine?
|
|
|
|
|
build-machine-name
|
|
|
|
|
build-machine-port
|
|
|
|
|
build-machine-systems
|
|
|
|
|
build-machine-user
|
|
|
|
|
build-machine-private-key
|
|
|
|
|
build-machine-host-key
|
|
|
|
|
build-machine-compression
|
|
|
|
|
build-machine-daemon-socket
|
|
|
|
|
build-machine-overload-threshold
|
|
|
|
|
build-machine-systems
|
|
|
|
|
build-machine-features
|
2020-12-03 14:22:50 -05:00
|
|
|
|
build-machine-location
|
2020-10-24 18:14:00 -04:00
|
|
|
|
|
2014-01-23 17:48:34 -05:00
|
|
|
|
build-requirements
|
2020-10-24 18:14:00 -04:00
|
|
|
|
build-requirements?
|
|
|
|
|
|
2014-01-23 17:48:34 -05:00
|
|
|
|
guix-offload))
|
|
|
|
|
|
|
|
|
|
;;; Commentary:
|
|
|
|
|
;;;
|
|
|
|
|
;;; Attempt to offload builds to the machines listed in
|
|
|
|
|
;;; /etc/guix/machines.scm, transferring missing dependencies over SSH, and
|
|
|
|
|
;;; retrieving the build output(s) over SSH upon success.
|
|
|
|
|
;;;
|
|
|
|
|
;;; This command should not be used directly; instead, it is called on-demand
|
2019-11-21 09:14:33 -05:00
|
|
|
|
;;; by the daemon, unless it was started with '--no-offload' or a client
|
2014-01-23 17:48:34 -05:00
|
|
|
|
;;; inhibited build hooks.
|
|
|
|
|
;;;
|
|
|
|
|
;;; Code:
|
|
|
|
|
|
|
|
|
|
(define-record-type* <build-machine>
|
|
|
|
|
build-machine make-build-machine
|
|
|
|
|
build-machine?
|
|
|
|
|
(name build-machine-name) ; string
|
2014-03-13 16:58:04 -04:00
|
|
|
|
(port build-machine-port ; number
|
|
|
|
|
(default 22))
|
2020-08-24 16:26:14 -04:00
|
|
|
|
(systems %build-machine-systems ; list of strings
|
|
|
|
|
(default #f)) ; drop default after system is removed
|
|
|
|
|
(system %build-machine-system ; deprecated
|
|
|
|
|
(default #f))
|
2014-01-23 17:48:34 -05:00
|
|
|
|
(user build-machine-user) ; string
|
|
|
|
|
(private-key build-machine-private-key ; file name
|
2014-03-02 16:39:48 -05:00
|
|
|
|
(default (user-openssh-private-key)))
|
|
|
|
|
(host-key build-machine-host-key) ; string
|
2016-12-05 17:15:17 -05:00
|
|
|
|
(compression build-machine-compression ; string
|
|
|
|
|
(default "zlib@openssh.com,zlib"))
|
|
|
|
|
(compression-level build-machine-compression-level ;integer
|
|
|
|
|
(default 3))
|
2016-11-04 19:47:34 -04:00
|
|
|
|
(daemon-socket build-machine-daemon-socket ; string
|
|
|
|
|
(default "/var/guix/daemon-socket/socket"))
|
2020-10-03 01:17:54 -04:00
|
|
|
|
;; A #f value tells the offload scheduler to disregard the load of the build
|
|
|
|
|
;; machine when selecting the best offload machine.
|
|
|
|
|
(overload-threshold build-machine-overload-threshold ; inexact real between
|
2022-11-23 13:32:04 -05:00
|
|
|
|
(default 0.8)) ; 0.0 and 1.0 | #f
|
2014-01-23 17:48:34 -05:00
|
|
|
|
(parallel-builds build-machine-parallel-builds ; number
|
|
|
|
|
(default 1))
|
|
|
|
|
(speed build-machine-speed ; inexact real
|
|
|
|
|
(default 1.0))
|
|
|
|
|
(features build-machine-features ; list of strings
|
2020-12-03 14:22:50 -05:00
|
|
|
|
(default '()))
|
|
|
|
|
(location build-machine-location
|
|
|
|
|
(default (and=> (current-source-location)
|
|
|
|
|
source-properties->location))
|
|
|
|
|
(innate)))
|
2014-01-23 17:48:34 -05:00
|
|
|
|
|
2020-08-24 16:26:14 -04:00
|
|
|
|
;;; Deprecated.
|
|
|
|
|
(define (build-machine-system machine)
|
2020-12-03 14:22:50 -05:00
|
|
|
|
(warning
|
|
|
|
|
(build-machine-location machine)
|
|
|
|
|
(G_ "The 'system' field is deprecated, \
|
2020-08-24 16:26:14 -04:00
|
|
|
|
please use 'systems' instead.~%"))
|
|
|
|
|
(%build-machine-system machine))
|
|
|
|
|
|
|
|
|
|
;;; TODO: Remove after the deprecated 'system' field is removed.
|
|
|
|
|
(define (build-machine-systems machine)
|
|
|
|
|
(or (%build-machine-systems machine)
|
|
|
|
|
(list (build-machine-system machine))
|
|
|
|
|
(leave (G_ "The build-machine object lacks a value for its 'systems'
|
|
|
|
|
field."))))
|
|
|
|
|
|
2014-01-23 17:48:34 -05:00
|
|
|
|
(define-record-type* <build-requirements>
|
|
|
|
|
build-requirements make-build-requirements
|
|
|
|
|
build-requirements?
|
|
|
|
|
(system build-requirements-system) ; string
|
|
|
|
|
(features build-requirements-features ; list of strings
|
|
|
|
|
(default '())))
|
|
|
|
|
|
|
|
|
|
(define %machine-file
|
|
|
|
|
;; File that lists machines available as build slaves.
|
|
|
|
|
(string-append %config-directory "/machines.scm"))
|
|
|
|
|
|
2014-03-02 16:39:48 -05:00
|
|
|
|
(define (user-openssh-private-key)
|
|
|
|
|
"Return the user's default SSH private key, or #f if it could not be
|
2014-01-23 17:48:34 -05:00
|
|
|
|
determined."
|
|
|
|
|
(and=> (getenv "HOME")
|
2014-03-02 16:39:48 -05:00
|
|
|
|
(cut string-append <> "/.ssh/id_rsa")))
|
2014-01-23 17:48:34 -05:00
|
|
|
|
|
|
|
|
|
(define %user-module
|
|
|
|
|
;; Module in which the machine description file is loaded.
|
|
|
|
|
(let ((module (make-fresh-user-module)))
|
|
|
|
|
(module-use! module (resolve-interface '(guix scripts offload)))
|
|
|
|
|
module))
|
|
|
|
|
|
|
|
|
|
(define* (build-machines #:optional (file %machine-file))
|
|
|
|
|
"Read the list of build machines from FILE and return it."
|
|
|
|
|
(catch #t
|
|
|
|
|
(lambda ()
|
|
|
|
|
;; Avoid ABI incompatibility with the <build-machine> record.
|
2017-12-01 12:31:16 -05:00
|
|
|
|
;; (set! %fresh-auto-compile #t)
|
2014-01-23 17:48:34 -05:00
|
|
|
|
|
|
|
|
|
(save-module-excursion
|
|
|
|
|
(lambda ()
|
|
|
|
|
(set-current-module %user-module)
|
2018-06-14 17:08:08 -04:00
|
|
|
|
(match (primitive-load file)
|
|
|
|
|
(((? build-machine? machines) ...)
|
|
|
|
|
machines)
|
|
|
|
|
(_
|
|
|
|
|
;; Instead of crashing, assume the empty list.
|
|
|
|
|
(warning (G_ "'~a' did not return a list of build machines; \
|
|
|
|
|
ignoring it~%")
|
|
|
|
|
file)
|
|
|
|
|
'())))))
|
2014-01-23 17:48:34 -05:00
|
|
|
|
(lambda args
|
|
|
|
|
(match args
|
2016-09-20 04:51:39 -04:00
|
|
|
|
(('system-error . rest)
|
2014-01-23 17:48:34 -05:00
|
|
|
|
(let ((err (system-error-errno args)))
|
|
|
|
|
;; Silently ignore missing file since this is a common case.
|
|
|
|
|
(if (= ENOENT err)
|
|
|
|
|
'()
|
ui: Rename '_' to 'G_'.
This avoids collisions with '_' when the latter is used as a 'match'
pattern for instance. See
<https://lists.gnu.org/archive/html/guix-devel/2017-04/msg00464.html>.
* guix/ui.scm: Rename '_' to 'G_'.
* po/guix/Makevars (XGETTEXT_OPTIONS): Adjust accordingly.
* build-aux/compile-all.scm (warnings): Remove 'format'.
* gnu/packages.scm,
gnu/services.scm,
gnu/services/shepherd.scm,
gnu/system.scm,
gnu/system/shadow.scm,
guix/gnupg.scm,
guix/http-client.scm,
guix/import/cpan.scm,
guix/import/elpa.scm,
guix/import/pypi.scm,
guix/nar.scm,
guix/scripts.scm,
guix/scripts/archive.scm,
guix/scripts/authenticate.scm,
guix/scripts/build.scm,
guix/scripts/challenge.scm,
guix/scripts/container.scm,
guix/scripts/container/exec.scm,
guix/scripts/copy.scm,
guix/scripts/download.scm,
guix/scripts/edit.scm,
guix/scripts/environment.scm,
guix/scripts/gc.scm,
guix/scripts/graph.scm,
guix/scripts/hash.scm,
guix/scripts/import.scm,
guix/scripts/import/cpan.scm,
guix/scripts/import/cran.scm,
guix/scripts/import/crate.scm,
guix/scripts/import/elpa.scm,
guix/scripts/import/gem.scm,
guix/scripts/import/gnu.scm,
guix/scripts/import/hackage.scm,
guix/scripts/import/nix.scm,
guix/scripts/import/pypi.scm,
guix/scripts/import/stackage.scm,
guix/scripts/lint.scm,
guix/scripts/offload.scm,
guix/scripts/pack.scm,
guix/scripts/package.scm,
guix/scripts/perform-download.scm,
guix/scripts/publish.scm,
guix/scripts/pull.scm,
guix/scripts/refresh.scm,
guix/scripts/size.scm,
guix/scripts/substitute.scm,
guix/scripts/system.scm,
guix/ssh.scm,
guix/upstream.scm: Use 'G_' instead of '_'. Most of this change was
obtained by running: "sed -i -e's/(_ "/(G_ "/g' `find -name \*.scm`".
2017-05-03 09:57:02 -04:00
|
|
|
|
(leave (G_ "failed to open machine file '~a': ~a~%")
|
2014-02-01 19:32:50 -05:00
|
|
|
|
file (strerror err)))))
|
2014-06-26 16:23:36 -04:00
|
|
|
|
(('syntax-error proc message properties form . rest)
|
|
|
|
|
(let ((loc (source-properties->location properties)))
|
ui: Rename '_' to 'G_'.
This avoids collisions with '_' when the latter is used as a 'match'
pattern for instance. See
<https://lists.gnu.org/archive/html/guix-devel/2017-04/msg00464.html>.
* guix/ui.scm: Rename '_' to 'G_'.
* po/guix/Makevars (XGETTEXT_OPTIONS): Adjust accordingly.
* build-aux/compile-all.scm (warnings): Remove 'format'.
* gnu/packages.scm,
gnu/services.scm,
gnu/services/shepherd.scm,
gnu/system.scm,
gnu/system/shadow.scm,
guix/gnupg.scm,
guix/http-client.scm,
guix/import/cpan.scm,
guix/import/elpa.scm,
guix/import/pypi.scm,
guix/nar.scm,
guix/scripts.scm,
guix/scripts/archive.scm,
guix/scripts/authenticate.scm,
guix/scripts/build.scm,
guix/scripts/challenge.scm,
guix/scripts/container.scm,
guix/scripts/container/exec.scm,
guix/scripts/copy.scm,
guix/scripts/download.scm,
guix/scripts/edit.scm,
guix/scripts/environment.scm,
guix/scripts/gc.scm,
guix/scripts/graph.scm,
guix/scripts/hash.scm,
guix/scripts/import.scm,
guix/scripts/import/cpan.scm,
guix/scripts/import/cran.scm,
guix/scripts/import/crate.scm,
guix/scripts/import/elpa.scm,
guix/scripts/import/gem.scm,
guix/scripts/import/gnu.scm,
guix/scripts/import/hackage.scm,
guix/scripts/import/nix.scm,
guix/scripts/import/pypi.scm,
guix/scripts/import/stackage.scm,
guix/scripts/lint.scm,
guix/scripts/offload.scm,
guix/scripts/pack.scm,
guix/scripts/package.scm,
guix/scripts/perform-download.scm,
guix/scripts/publish.scm,
guix/scripts/pull.scm,
guix/scripts/refresh.scm,
guix/scripts/size.scm,
guix/scripts/substitute.scm,
guix/scripts/system.scm,
guix/ssh.scm,
guix/upstream.scm: Use 'G_' instead of '_'. Most of this change was
obtained by running: "sed -i -e's/(_ "/(G_ "/g' `find -name \*.scm`".
2017-05-03 09:57:02 -04:00
|
|
|
|
(leave (G_ "~a: ~a~%")
|
2014-06-26 16:23:36 -04:00
|
|
|
|
(location->string loc) message)))
|
2016-09-20 04:51:39 -04:00
|
|
|
|
(x
|
ui: Rename '_' to 'G_'.
This avoids collisions with '_' when the latter is used as a 'match'
pattern for instance. See
<https://lists.gnu.org/archive/html/guix-devel/2017-04/msg00464.html>.
* guix/ui.scm: Rename '_' to 'G_'.
* po/guix/Makevars (XGETTEXT_OPTIONS): Adjust accordingly.
* build-aux/compile-all.scm (warnings): Remove 'format'.
* gnu/packages.scm,
gnu/services.scm,
gnu/services/shepherd.scm,
gnu/system.scm,
gnu/system/shadow.scm,
guix/gnupg.scm,
guix/http-client.scm,
guix/import/cpan.scm,
guix/import/elpa.scm,
guix/import/pypi.scm,
guix/nar.scm,
guix/scripts.scm,
guix/scripts/archive.scm,
guix/scripts/authenticate.scm,
guix/scripts/build.scm,
guix/scripts/challenge.scm,
guix/scripts/container.scm,
guix/scripts/container/exec.scm,
guix/scripts/copy.scm,
guix/scripts/download.scm,
guix/scripts/edit.scm,
guix/scripts/environment.scm,
guix/scripts/gc.scm,
guix/scripts/graph.scm,
guix/scripts/hash.scm,
guix/scripts/import.scm,
guix/scripts/import/cpan.scm,
guix/scripts/import/cran.scm,
guix/scripts/import/crate.scm,
guix/scripts/import/elpa.scm,
guix/scripts/import/gem.scm,
guix/scripts/import/gnu.scm,
guix/scripts/import/hackage.scm,
guix/scripts/import/nix.scm,
guix/scripts/import/pypi.scm,
guix/scripts/import/stackage.scm,
guix/scripts/lint.scm,
guix/scripts/offload.scm,
guix/scripts/pack.scm,
guix/scripts/package.scm,
guix/scripts/perform-download.scm,
guix/scripts/publish.scm,
guix/scripts/pull.scm,
guix/scripts/refresh.scm,
guix/scripts/size.scm,
guix/scripts/substitute.scm,
guix/scripts/system.scm,
guix/ssh.scm,
guix/upstream.scm: Use 'G_' instead of '_'. Most of this change was
obtained by running: "sed -i -e's/(_ "/(G_ "/g' `find -name \*.scm`".
2017-05-03 09:57:02 -04:00
|
|
|
|
(leave (G_ "failed to load machine file '~a': ~s~%")
|
2014-02-01 19:32:50 -05:00
|
|
|
|
file args))))))
|
2014-01-23 17:48:34 -05:00
|
|
|
|
|
2014-03-02 16:39:48 -05:00
|
|
|
|
(define (private-key-from-file* file)
|
|
|
|
|
"Like 'private-key-from-file', but raise an error that 'with-error-handling'
|
|
|
|
|
can interpret meaningfully."
|
|
|
|
|
(catch 'guile-ssh-error
|
|
|
|
|
(lambda ()
|
|
|
|
|
(private-key-from-file file))
|
|
|
|
|
(lambda (key proc str . rest)
|
2020-07-25 12:26:18 -04:00
|
|
|
|
(raise (formatted-message (G_ "failed to load SSH \
|
2014-03-02 16:39:48 -05:00
|
|
|
|
private key from '~a': ~a")
|
2020-07-25 12:26:18 -04:00
|
|
|
|
file str)))))
|
2014-03-02 16:39:48 -05:00
|
|
|
|
|
2020-10-24 18:20:12 -04:00
|
|
|
|
(define* (open-ssh-session machine #:optional max-silent-time)
|
|
|
|
|
"Open an SSH session for MACHINE and return it. Throw an error on failure.
|
|
|
|
|
When MAX-SILENT-TIME is true, it must be a positive integer denoting the
|
|
|
|
|
number of seconds after which the connection times out."
|
2014-03-02 16:39:48 -05:00
|
|
|
|
(let ((private (private-key-from-file* (build-machine-private-key machine)))
|
|
|
|
|
(public (public-key-from-file
|
|
|
|
|
(string-append (build-machine-private-key machine)
|
|
|
|
|
".pub")))
|
|
|
|
|
(session (make-session #:user (build-machine-user machine)
|
|
|
|
|
#:host (build-machine-name machine)
|
|
|
|
|
#:port (build-machine-port machine)
|
2023-02-24 21:24:52 -05:00
|
|
|
|
;; Multiple derivations may be offloaded in
|
|
|
|
|
;; parallel, and when there is a large amount
|
|
|
|
|
;; of data to be sent, it can choke lower
|
|
|
|
|
;; bandwidth connections and cause timeouts, so
|
|
|
|
|
;; set it to a large enough value.
|
|
|
|
|
#:timeout 30 ;initial timeout (seconds)
|
2014-03-02 16:39:48 -05:00
|
|
|
|
;; #:log-verbosity 'protocol
|
|
|
|
|
#:identity (build-machine-private-key machine)
|
|
|
|
|
|
2016-12-09 08:54:42 -05:00
|
|
|
|
;; By default libssh reads ~/.ssh/known_hosts
|
|
|
|
|
;; and uses that to adjust its choice of cipher
|
|
|
|
|
;; suites, which changes the type of host key
|
|
|
|
|
;; that the server sends (RSA vs. Ed25519,
|
|
|
|
|
;; etc.). Opt for something reproducible and
|
|
|
|
|
;; stateless instead.
|
|
|
|
|
#:knownhosts "/dev/null"
|
|
|
|
|
|
2022-01-03 13:31:05 -05:00
|
|
|
|
;; Likewise for ~/.ssh/config.
|
|
|
|
|
#:config "/dev/null"
|
|
|
|
|
|
2014-03-02 16:39:48 -05:00
|
|
|
|
;; We need lightweight compression when
|
|
|
|
|
;; exchanging full archives.
|
2016-12-05 17:15:17 -05:00
|
|
|
|
#:compression
|
|
|
|
|
(build-machine-compression machine)
|
|
|
|
|
#:compression-level
|
|
|
|
|
(build-machine-compression-level machine))))
|
2016-12-01 17:20:18 -05:00
|
|
|
|
(match (connect! session)
|
|
|
|
|
('ok
|
2019-12-03 15:28:23 -05:00
|
|
|
|
;; Make sure the server's key is what we expect.
|
|
|
|
|
(authenticate-server* session (build-machine-host-key machine))
|
2016-12-01 17:20:18 -05:00
|
|
|
|
|
|
|
|
|
(let ((auth (userauth-public-key! session private)))
|
|
|
|
|
(unless (eq? 'success auth)
|
|
|
|
|
(disconnect! session)
|
ui: Rename '_' to 'G_'.
This avoids collisions with '_' when the latter is used as a 'match'
pattern for instance. See
<https://lists.gnu.org/archive/html/guix-devel/2017-04/msg00464.html>.
* guix/ui.scm: Rename '_' to 'G_'.
* po/guix/Makevars (XGETTEXT_OPTIONS): Adjust accordingly.
* build-aux/compile-all.scm (warnings): Remove 'format'.
* gnu/packages.scm,
gnu/services.scm,
gnu/services/shepherd.scm,
gnu/system.scm,
gnu/system/shadow.scm,
guix/gnupg.scm,
guix/http-client.scm,
guix/import/cpan.scm,
guix/import/elpa.scm,
guix/import/pypi.scm,
guix/nar.scm,
guix/scripts.scm,
guix/scripts/archive.scm,
guix/scripts/authenticate.scm,
guix/scripts/build.scm,
guix/scripts/challenge.scm,
guix/scripts/container.scm,
guix/scripts/container/exec.scm,
guix/scripts/copy.scm,
guix/scripts/download.scm,
guix/scripts/edit.scm,
guix/scripts/environment.scm,
guix/scripts/gc.scm,
guix/scripts/graph.scm,
guix/scripts/hash.scm,
guix/scripts/import.scm,
guix/scripts/import/cpan.scm,
guix/scripts/import/cran.scm,
guix/scripts/import/crate.scm,
guix/scripts/import/elpa.scm,
guix/scripts/import/gem.scm,
guix/scripts/import/gnu.scm,
guix/scripts/import/hackage.scm,
guix/scripts/import/nix.scm,
guix/scripts/import/pypi.scm,
guix/scripts/import/stackage.scm,
guix/scripts/lint.scm,
guix/scripts/offload.scm,
guix/scripts/pack.scm,
guix/scripts/package.scm,
guix/scripts/perform-download.scm,
guix/scripts/publish.scm,
guix/scripts/pull.scm,
guix/scripts/refresh.scm,
guix/scripts/size.scm,
guix/scripts/substitute.scm,
guix/scripts/system.scm,
guix/ssh.scm,
guix/upstream.scm: Use 'G_' instead of '_'. Most of this change was
obtained by running: "sed -i -e's/(_ "/(G_ "/g' `find -name \*.scm`".
2017-05-03 09:57:02 -04:00
|
|
|
|
(leave (G_ "SSH public key authentication failed for '~a': ~a~%")
|
2016-12-01 17:20:18 -05:00
|
|
|
|
(build-machine-name machine) (get-error session))))
|
|
|
|
|
|
2020-10-24 18:20:12 -04:00
|
|
|
|
(when max-silent-time
|
|
|
|
|
;; From then on use MAX-SILENT-TIME as the absolute timeout when
|
|
|
|
|
;; reading from or write to a channel for this session.
|
|
|
|
|
(session-set! session 'timeout max-silent-time))
|
2019-10-15 06:24:09 -04:00
|
|
|
|
|
2016-12-01 17:20:18 -05:00
|
|
|
|
session)
|
|
|
|
|
(x
|
|
|
|
|
;; Connection failed or timeout expired.
|
ui: Rename '_' to 'G_'.
This avoids collisions with '_' when the latter is used as a 'match'
pattern for instance. See
<https://lists.gnu.org/archive/html/guix-devel/2017-04/msg00464.html>.
* guix/ui.scm: Rename '_' to 'G_'.
* po/guix/Makevars (XGETTEXT_OPTIONS): Adjust accordingly.
* build-aux/compile-all.scm (warnings): Remove 'format'.
* gnu/packages.scm,
gnu/services.scm,
gnu/services/shepherd.scm,
gnu/system.scm,
gnu/system/shadow.scm,
guix/gnupg.scm,
guix/http-client.scm,
guix/import/cpan.scm,
guix/import/elpa.scm,
guix/import/pypi.scm,
guix/nar.scm,
guix/scripts.scm,
guix/scripts/archive.scm,
guix/scripts/authenticate.scm,
guix/scripts/build.scm,
guix/scripts/challenge.scm,
guix/scripts/container.scm,
guix/scripts/container/exec.scm,
guix/scripts/copy.scm,
guix/scripts/download.scm,
guix/scripts/edit.scm,
guix/scripts/environment.scm,
guix/scripts/gc.scm,
guix/scripts/graph.scm,
guix/scripts/hash.scm,
guix/scripts/import.scm,
guix/scripts/import/cpan.scm,
guix/scripts/import/cran.scm,
guix/scripts/import/crate.scm,
guix/scripts/import/elpa.scm,
guix/scripts/import/gem.scm,
guix/scripts/import/gnu.scm,
guix/scripts/import/hackage.scm,
guix/scripts/import/nix.scm,
guix/scripts/import/pypi.scm,
guix/scripts/import/stackage.scm,
guix/scripts/lint.scm,
guix/scripts/offload.scm,
guix/scripts/pack.scm,
guix/scripts/package.scm,
guix/scripts/perform-download.scm,
guix/scripts/publish.scm,
guix/scripts/pull.scm,
guix/scripts/refresh.scm,
guix/scripts/size.scm,
guix/scripts/substitute.scm,
guix/scripts/system.scm,
guix/ssh.scm,
guix/upstream.scm: Use 'G_' instead of '_'. Most of this change was
obtained by running: "sed -i -e's/(_ "/(G_ "/g' `find -name \*.scm`".
2017-05-03 09:57:02 -04:00
|
|
|
|
(leave (G_ "failed to connect to '~a': ~a~%")
|
2016-12-01 17:20:18 -05:00
|
|
|
|
(build-machine-name machine) (get-error session))))))
|
2014-03-02 16:39:48 -05:00
|
|
|
|
|
2014-03-09 13:08:21 -04:00
|
|
|
|
|
|
|
|
|
;;;
|
|
|
|
|
;;; Synchronization.
|
|
|
|
|
;;;
|
|
|
|
|
|
|
|
|
|
(define (machine-slot-file machine slot)
|
|
|
|
|
"Return the file name of MACHINE's file for SLOT."
|
|
|
|
|
;; For each machine we have a bunch of files representing each build slot.
|
|
|
|
|
;; When choosing a build machine, we attempt to get an exclusive lock on one
|
|
|
|
|
;; of these; if we fail, that means all the build slots are already taken.
|
|
|
|
|
;; Inspired by Nix's build-remote.pl.
|
|
|
|
|
(string-append (string-append %state-directory "/offload/"
|
2019-09-28 11:22:39 -04:00
|
|
|
|
(build-machine-name machine) ":"
|
|
|
|
|
(number->string (build-machine-port machine))
|
2014-03-09 13:08:21 -04:00
|
|
|
|
"/" (number->string slot))))
|
|
|
|
|
|
|
|
|
|
(define (acquire-build-slot machine)
|
|
|
|
|
"Attempt to acquire a build slot on MACHINE. Return the port representing
|
|
|
|
|
the slot, or #f if none is available.
|
|
|
|
|
|
|
|
|
|
This mechanism allows us to set a hard limit on the number of simultaneous
|
|
|
|
|
connections allowed to MACHINE."
|
|
|
|
|
(mkdir-p (dirname (machine-slot-file machine 0)))
|
2018-12-26 11:42:02 -05:00
|
|
|
|
|
|
|
|
|
;; When several 'guix offload' processes run in parallel, there's a race
|
|
|
|
|
;; among them, but since they try the slots in the same order, we're fine.
|
|
|
|
|
(any (lambda (slot)
|
|
|
|
|
(let ((port (open-file (machine-slot-file machine slot)
|
|
|
|
|
"w0")))
|
|
|
|
|
(catch 'flock-error
|
|
|
|
|
(lambda ()
|
|
|
|
|
(fcntl-flock port 'write-lock #:wait? #f)
|
|
|
|
|
;; Got it!
|
|
|
|
|
(format (current-error-port)
|
|
|
|
|
"process ~a acquired build slot '~a'~%"
|
|
|
|
|
(getpid) (port-filename port))
|
|
|
|
|
port)
|
|
|
|
|
(lambda args
|
|
|
|
|
;; PORT is already locked by another process.
|
|
|
|
|
(close-port port)
|
|
|
|
|
#f))))
|
|
|
|
|
(iota (build-machine-parallel-builds machine))))
|
2014-03-09 13:08:21 -04:00
|
|
|
|
|
|
|
|
|
(define (release-build-slot slot)
|
|
|
|
|
"Release SLOT, a build slot as returned as by 'acquire-build-slot'."
|
|
|
|
|
(close-port slot))
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
;;;
|
|
|
|
|
;;; Offloading.
|
|
|
|
|
;;;
|
|
|
|
|
|
2014-03-19 18:12:06 -04:00
|
|
|
|
(define (build-log-port)
|
|
|
|
|
"Return the default port where build logs should be sent. The default is
|
|
|
|
|
file descriptor 4, which is open by the daemon before running the offload
|
|
|
|
|
hook."
|
|
|
|
|
(let ((port (fdopen 4 "w0")))
|
|
|
|
|
;; Make sure file descriptor 4 isn't closed when PORT is GC'd.
|
|
|
|
|
(set-port-revealed! port 1)
|
|
|
|
|
port))
|
|
|
|
|
|
offload: Use (guix inferior) instead of (ssh dist node).
Using inferiors and thus 'guix repl' simplifies setup on build
machines (no need to worry about GUILE_LOAD_PATH etc.)
Furthermore, the 'guix repl -t machine' protocol running in a remote
pipe addresses several issues with the current implementation of nodes
and RREPLs in Guile-SSH: fewer round trips, doesn't leave a 'guile
--listen' process behind it, stateless (since a new process is started
each time), more efficient (the SSH channel can be reused), more
reliable (no 'pgrep', 'pkill', and shellology; see
<https://github.com/artyom-poptsov/guile-ssh/issues/11> as an example.)
* guix/ssh.scm (inferior-remote-eval): New procedure.
(send-files): Use it instead of 'make-node' and 'node-eval'.
* guix/scripts/offload.scm (node-guile-version): New procedure.
(node-free-disk-space, transfer-and-offload, node-load)
(choose-build-machine, assert-node-has-guix): Use 'remote-inferior'
instead of 'make-node' and 'inferior-eval' instead of 'node-eval'.
(assert-node-can-import, assert-node-can-export): Likewise, and add
'session' parameter.
(check-machine-availability): Likewise, and add calls to
'close-inferior' and 'disconnect!'.
(check-machine-status): Likewise.
* doc/guix.texi (Daemon Offload Setup): Remove bit related to 'guile' in
$PATH and $GUILE_LOAD_PATH; mention 'guix' alone.
2018-12-24 09:40:04 -05:00
|
|
|
|
(define (node-guile-version node)
|
|
|
|
|
(inferior-eval '(version) node))
|
|
|
|
|
|
2018-12-21 17:12:52 -05:00
|
|
|
|
(define (node-free-disk-space node)
|
|
|
|
|
"Return the free disk space, in bytes, in NODE's store."
|
offload: Use (guix inferior) instead of (ssh dist node).
Using inferiors and thus 'guix repl' simplifies setup on build
machines (no need to worry about GUILE_LOAD_PATH etc.)
Furthermore, the 'guix repl -t machine' protocol running in a remote
pipe addresses several issues with the current implementation of nodes
and RREPLs in Guile-SSH: fewer round trips, doesn't leave a 'guile
--listen' process behind it, stateless (since a new process is started
each time), more efficient (the SSH channel can be reused), more
reliable (no 'pgrep', 'pkill', and shellology; see
<https://github.com/artyom-poptsov/guile-ssh/issues/11> as an example.)
* guix/ssh.scm (inferior-remote-eval): New procedure.
(send-files): Use it instead of 'make-node' and 'node-eval'.
* guix/scripts/offload.scm (node-guile-version): New procedure.
(node-free-disk-space, transfer-and-offload, node-load)
(choose-build-machine, assert-node-has-guix): Use 'remote-inferior'
instead of 'make-node' and 'inferior-eval' instead of 'node-eval'.
(assert-node-can-import, assert-node-can-export): Likewise, and add
'session' parameter.
(check-machine-availability): Likewise, and add calls to
'close-inferior' and 'disconnect!'.
(check-machine-status): Likewise.
* doc/guix.texi (Daemon Offload Setup): Remove bit related to 'guile' in
$PATH and $GUILE_LOAD_PATH; mention 'guix' alone.
2018-12-24 09:40:04 -05:00
|
|
|
|
(inferior-eval `(begin
|
|
|
|
|
(use-modules (guix build syscalls))
|
|
|
|
|
(free-disk-space ,(%store-prefix)))
|
|
|
|
|
node))
|
2018-12-21 17:12:52 -05:00
|
|
|
|
|
2014-03-09 09:05:30 -04:00
|
|
|
|
(define* (transfer-and-offload drv machine
|
|
|
|
|
#:key
|
|
|
|
|
(inputs '())
|
|
|
|
|
(outputs '())
|
|
|
|
|
(max-silent-time 3600)
|
2014-03-09 18:13:53 -04:00
|
|
|
|
build-timeout
|
2014-03-09 09:05:30 -04:00
|
|
|
|
print-build-trace?)
|
|
|
|
|
"Offload DRV to MACHINE. Prior to the actual offloading, transfer all of
|
|
|
|
|
INPUTS to MACHINE; if building DRV succeeds, retrieve all of OUTPUTS from
|
|
|
|
|
MACHINE."
|
2016-11-02 07:00:47 -04:00
|
|
|
|
(define session
|
2019-10-15 06:24:09 -04:00
|
|
|
|
(open-ssh-session machine max-silent-time))
|
2016-11-02 07:00:47 -04:00
|
|
|
|
|
2016-11-04 19:47:34 -04:00
|
|
|
|
(define store
|
|
|
|
|
(connect-to-remote-daemon session
|
|
|
|
|
(build-machine-daemon-socket machine)))
|
|
|
|
|
|
|
|
|
|
(set-build-options store
|
|
|
|
|
#:print-build-trace print-build-trace?
|
|
|
|
|
#:max-silent-time max-silent-time
|
|
|
|
|
#:timeout build-timeout)
|
|
|
|
|
|
|
|
|
|
;; Protect DRV from garbage collection.
|
|
|
|
|
(add-temp-root store (derivation-file-name drv))
|
|
|
|
|
|
2016-12-30 17:22:27 -05:00
|
|
|
|
(with-store local
|
|
|
|
|
(send-files local (cons (derivation-file-name drv) inputs) store
|
|
|
|
|
#:log-port (current-output-port)))
|
2016-11-04 19:47:34 -04:00
|
|
|
|
(format (current-error-port) "offloading '~a' to '~a'...~%"
|
|
|
|
|
(derivation-file-name drv) (build-machine-name machine))
|
|
|
|
|
(format (current-error-port) "@ build-remote ~a ~a~%"
|
|
|
|
|
(derivation-file-name drv) (build-machine-name machine))
|
|
|
|
|
|
store: Rename '&nix-error' to '&store-error'.
* guix/store.scm (&nix-error): Rename to...
(&store-error): ... this, and adjust users.
(&nix-connection-error): Rename to...
(&store-connection-error): ... this, and adjust users.
(&nix-protocol-error): Rename to...
(&store-protocol-error): ... this, adjust users.
(&nix-error, &nix-connection-error, &nix-protocol-error): Define these
condition types and their getters as deprecrated aliases.
* build-aux/run-system-tests.scm, guix/derivations.scm,
guix/grafts.scm, guix/scripts/challenge.scm,
guix/scripts/graph.scm, guix/scripts/lint.scm,
guix/scripts/offload.scm, guix/serialization.scm,
guix/ssh.scm, guix/tests.scm, guix/ui.scm,
tests/derivations.scm, tests/gexp.scm, tests/guix-daemon.sh,
tests/packages.scm, tests/store.scm, doc/guix.texi: Adjust to use the
new names.
2019-01-21 11:41:11 -05:00
|
|
|
|
(guard (c ((store-protocol-error? c)
|
2016-11-04 19:47:34 -04:00
|
|
|
|
(format (current-error-port)
|
ui: Rename '_' to 'G_'.
This avoids collisions with '_' when the latter is used as a 'match'
pattern for instance. See
<https://lists.gnu.org/archive/html/guix-devel/2017-04/msg00464.html>.
* guix/ui.scm: Rename '_' to 'G_'.
* po/guix/Makevars (XGETTEXT_OPTIONS): Adjust accordingly.
* build-aux/compile-all.scm (warnings): Remove 'format'.
* gnu/packages.scm,
gnu/services.scm,
gnu/services/shepherd.scm,
gnu/system.scm,
gnu/system/shadow.scm,
guix/gnupg.scm,
guix/http-client.scm,
guix/import/cpan.scm,
guix/import/elpa.scm,
guix/import/pypi.scm,
guix/nar.scm,
guix/scripts.scm,
guix/scripts/archive.scm,
guix/scripts/authenticate.scm,
guix/scripts/build.scm,
guix/scripts/challenge.scm,
guix/scripts/container.scm,
guix/scripts/container/exec.scm,
guix/scripts/copy.scm,
guix/scripts/download.scm,
guix/scripts/edit.scm,
guix/scripts/environment.scm,
guix/scripts/gc.scm,
guix/scripts/graph.scm,
guix/scripts/hash.scm,
guix/scripts/import.scm,
guix/scripts/import/cpan.scm,
guix/scripts/import/cran.scm,
guix/scripts/import/crate.scm,
guix/scripts/import/elpa.scm,
guix/scripts/import/gem.scm,
guix/scripts/import/gnu.scm,
guix/scripts/import/hackage.scm,
guix/scripts/import/nix.scm,
guix/scripts/import/pypi.scm,
guix/scripts/import/stackage.scm,
guix/scripts/lint.scm,
guix/scripts/offload.scm,
guix/scripts/pack.scm,
guix/scripts/package.scm,
guix/scripts/perform-download.scm,
guix/scripts/publish.scm,
guix/scripts/pull.scm,
guix/scripts/refresh.scm,
guix/scripts/size.scm,
guix/scripts/substitute.scm,
guix/scripts/system.scm,
guix/ssh.scm,
guix/upstream.scm: Use 'G_' instead of '_'. Most of this change was
obtained by running: "sed -i -e's/(_ "/(G_ "/g' `find -name \*.scm`".
2017-05-03 09:57:02 -04:00
|
|
|
|
(G_ "derivation '~a' offloaded to '~a' failed: ~a~%")
|
2016-11-04 19:47:34 -04:00
|
|
|
|
(derivation-file-name drv)
|
|
|
|
|
(build-machine-name machine)
|
store: Rename '&nix-error' to '&store-error'.
* guix/store.scm (&nix-error): Rename to...
(&store-error): ... this, and adjust users.
(&nix-connection-error): Rename to...
(&store-connection-error): ... this, and adjust users.
(&nix-protocol-error): Rename to...
(&store-protocol-error): ... this, adjust users.
(&nix-error, &nix-connection-error, &nix-protocol-error): Define these
condition types and their getters as deprecrated aliases.
* build-aux/run-system-tests.scm, guix/derivations.scm,
guix/grafts.scm, guix/scripts/challenge.scm,
guix/scripts/graph.scm, guix/scripts/lint.scm,
guix/scripts/offload.scm, guix/serialization.scm,
guix/ssh.scm, guix/tests.scm, guix/ui.scm,
tests/derivations.scm, tests/gexp.scm, tests/guix-daemon.sh,
tests/packages.scm, tests/store.scm, doc/guix.texi: Adjust to use the
new names.
2019-01-21 11:41:11 -05:00
|
|
|
|
(store-protocol-error-message c))
|
offload: Use (guix inferior) instead of (ssh dist node).
Using inferiors and thus 'guix repl' simplifies setup on build
machines (no need to worry about GUILE_LOAD_PATH etc.)
Furthermore, the 'guix repl -t machine' protocol running in a remote
pipe addresses several issues with the current implementation of nodes
and RREPLs in Guile-SSH: fewer round trips, doesn't leave a 'guile
--listen' process behind it, stateless (since a new process is started
each time), more efficient (the SSH channel can be reused), more
reliable (no 'pgrep', 'pkill', and shellology; see
<https://github.com/artyom-poptsov/guile-ssh/issues/11> as an example.)
* guix/ssh.scm (inferior-remote-eval): New procedure.
(send-files): Use it instead of 'make-node' and 'node-eval'.
* guix/scripts/offload.scm (node-guile-version): New procedure.
(node-free-disk-space, transfer-and-offload, node-load)
(choose-build-machine, assert-node-has-guix): Use 'remote-inferior'
instead of 'make-node' and 'inferior-eval' instead of 'node-eval'.
(assert-node-can-import, assert-node-can-export): Likewise, and add
'session' parameter.
(check-machine-availability): Likewise, and add calls to
'close-inferior' and 'disconnect!'.
(check-machine-status): Likewise.
* doc/guix.texi (Daemon Offload Setup): Remove bit related to 'guile' in
$PATH and $GUILE_LOAD_PATH; mention 'guix' alone.
2018-12-24 09:40:04 -05:00
|
|
|
|
(let* ((inferior (false-if-exception (remote-inferior session)))
|
|
|
|
|
(space (false-if-exception
|
|
|
|
|
(node-free-disk-space inferior))))
|
|
|
|
|
|
|
|
|
|
(when inferior
|
|
|
|
|
(close-inferior inferior))
|
2018-12-21 17:31:19 -05:00
|
|
|
|
|
|
|
|
|
;; Use exit code 100 for a permanent build failure. The daemon
|
|
|
|
|
;; interprets other non-zero codes as transient build failures.
|
|
|
|
|
(if (and space (< space (* 10 (expt 2 20))))
|
|
|
|
|
(begin
|
|
|
|
|
(format (current-error-port)
|
|
|
|
|
(G_ "build failure may have been caused by lack \
|
|
|
|
|
of free disk space on '~a'~%")
|
|
|
|
|
(build-machine-name machine))
|
|
|
|
|
(primitive-exit 1))
|
|
|
|
|
(primitive-exit 100)))))
|
2016-12-05 18:50:08 -05:00
|
|
|
|
(parameterize ((current-build-output-port (build-log-port)))
|
|
|
|
|
(build-derivations store (list drv))))
|
2016-11-04 19:47:34 -04:00
|
|
|
|
|
2018-01-12 16:20:30 -05:00
|
|
|
|
(retrieve-files* outputs store
|
|
|
|
|
|
|
|
|
|
;; We cannot use the 'import-paths' RPC here because we
|
|
|
|
|
;; already hold the locks for FILES.
|
|
|
|
|
#:import
|
|
|
|
|
(lambda (port)
|
|
|
|
|
(restore-file-set port
|
|
|
|
|
#:log-port (current-error-port)
|
|
|
|
|
#:lock? #f)))
|
|
|
|
|
|
2020-08-29 10:05:05 -04:00
|
|
|
|
(close-connection store)
|
|
|
|
|
(disconnect! session)
|
2016-11-04 19:47:34 -04:00
|
|
|
|
(format (current-error-port) "done with offloaded '~a'~%"
|
|
|
|
|
(derivation-file-name drv)))
|
|
|
|
|
|
2014-03-09 13:08:21 -04:00
|
|
|
|
|
|
|
|
|
;;;
|
|
|
|
|
;;; Scheduling.
|
|
|
|
|
;;;
|
|
|
|
|
|
2014-01-23 17:48:34 -05:00
|
|
|
|
(define (machine-matches? machine requirements)
|
|
|
|
|
"Return #t if MACHINE matches REQUIREMENTS."
|
2020-08-24 16:26:14 -04:00
|
|
|
|
(and (member (build-requirements-system requirements)
|
|
|
|
|
(build-machine-systems machine))
|
2014-01-23 17:48:34 -05:00
|
|
|
|
(lset<= string=?
|
|
|
|
|
(build-requirements-features requirements)
|
|
|
|
|
(build-machine-features machine))))
|
|
|
|
|
|
2018-12-21 17:12:52 -05:00
|
|
|
|
(define %minimum-disk-space
|
|
|
|
|
;; Minimum disk space required on the build machine for a build to be
|
|
|
|
|
;; offloaded. This keeps us from offloading to machines that are bound to
|
|
|
|
|
;; run out of disk space.
|
|
|
|
|
(* 100 (expt 2 20))) ;100 MiB
|
|
|
|
|
|
2018-12-21 16:54:02 -05:00
|
|
|
|
(define (node-load node)
|
2020-10-03 01:17:54 -04:00
|
|
|
|
"Return the load on NODE, a normalized value between 0.0 and 1.0. The value
|
|
|
|
|
is derived from /proc/loadavg and normalized according to the number of
|
|
|
|
|
logical cores available, to give a rough estimation of CPU usage. Return
|
|
|
|
|
1.0 (fully loaded) if NODE is misbehaving."
|
offload: Use (guix inferior) instead of (ssh dist node).
Using inferiors and thus 'guix repl' simplifies setup on build
machines (no need to worry about GUILE_LOAD_PATH etc.)
Furthermore, the 'guix repl -t machine' protocol running in a remote
pipe addresses several issues with the current implementation of nodes
and RREPLs in Guile-SSH: fewer round trips, doesn't leave a 'guile
--listen' process behind it, stateless (since a new process is started
each time), more efficient (the SSH channel can be reused), more
reliable (no 'pgrep', 'pkill', and shellology; see
<https://github.com/artyom-poptsov/guile-ssh/issues/11> as an example.)
* guix/ssh.scm (inferior-remote-eval): New procedure.
(send-files): Use it instead of 'make-node' and 'node-eval'.
* guix/scripts/offload.scm (node-guile-version): New procedure.
(node-free-disk-space, transfer-and-offload, node-load)
(choose-build-machine, assert-node-has-guix): Use 'remote-inferior'
instead of 'make-node' and 'inferior-eval' instead of 'node-eval'.
(assert-node-can-import, assert-node-can-export): Likewise, and add
'session' parameter.
(check-machine-availability): Likewise, and add calls to
'close-inferior' and 'disconnect!'.
(check-machine-status): Likewise.
* doc/guix.texi (Daemon Offload Setup): Remove bit related to 'guile' in
$PATH and $GUILE_LOAD_PATH; mention 'guix' alone.
2018-12-24 09:40:04 -05:00
|
|
|
|
(let ((line (inferior-eval '(begin
|
|
|
|
|
(use-modules (ice-9 rdelim))
|
|
|
|
|
(call-with-input-file "/proc/loadavg"
|
|
|
|
|
read-string))
|
2020-10-03 01:17:54 -04:00
|
|
|
|
node))
|
|
|
|
|
(ncores (inferior-eval '(begin
|
|
|
|
|
(use-modules (ice-9 threads))
|
|
|
|
|
(current-processor-count))
|
|
|
|
|
node)))
|
|
|
|
|
(if (or (eof-object? line) (eof-object? ncores))
|
|
|
|
|
1.0 ;MACHINE does not respond, so assume it is fully loaded
|
2018-12-21 16:54:02 -05:00
|
|
|
|
(match (string-tokenize line)
|
|
|
|
|
((one five fifteen . x)
|
2020-10-03 01:17:54 -04:00
|
|
|
|
(let ((load (/ (string->number one) ncores)))
|
|
|
|
|
(if (> load 1.0)
|
|
|
|
|
1.0
|
|
|
|
|
load)))
|
2018-12-21 16:54:02 -05:00
|
|
|
|
(x
|
2020-10-03 01:17:54 -04:00
|
|
|
|
1.0)))))
|
|
|
|
|
|
|
|
|
|
(define (report-load machine load)
|
|
|
|
|
(format (current-error-port)
|
|
|
|
|
"normalized load on machine '~a' is ~,2f~%"
|
|
|
|
|
(build-machine-name machine) load))
|
2014-02-28 19:31:18 -05:00
|
|
|
|
|
2017-10-12 08:21:54 -04:00
|
|
|
|
(define (random-seed)
|
|
|
|
|
(logxor (getpid) (car (gettimeofday))))
|
|
|
|
|
|
|
|
|
|
(define shuffle
|
|
|
|
|
(let ((state (seed->random-state (random-seed))))
|
|
|
|
|
(lambda (lst)
|
|
|
|
|
"Return LST shuffled (using the Fisher-Yates algorithm.)"
|
|
|
|
|
(define vec (list->vector lst))
|
|
|
|
|
(let loop ((result '())
|
|
|
|
|
(i (vector-length vec)))
|
|
|
|
|
(if (zero? i)
|
|
|
|
|
result
|
|
|
|
|
(let* ((j (random i state))
|
|
|
|
|
(val (vector-ref vec j)))
|
|
|
|
|
(vector-set! vec j (vector-ref vec (- i 1)))
|
|
|
|
|
(loop (cons val result) (- i 1))))))))
|
|
|
|
|
|
2022-11-22 01:17:17 -05:00
|
|
|
|
(define (remote-inferior* session)
|
|
|
|
|
"Like 'remote-inferior', but upon error return #f."
|
|
|
|
|
(or (guard (c ((inferior-protocol-error? c) #f))
|
|
|
|
|
(remote-inferior session))
|
|
|
|
|
(begin
|
|
|
|
|
(warning (G_ "failed to run 'guix repl' on machine '~a'~%")
|
|
|
|
|
(session-get session 'host))
|
|
|
|
|
#f)))
|
|
|
|
|
|
2014-03-09 09:05:30 -04:00
|
|
|
|
(define (choose-build-machine machines)
|
2017-07-25 15:55:20 -04:00
|
|
|
|
"Return two values: the best machine among MACHINES and its build
|
|
|
|
|
slot (which must later be released with 'release-build-slot'), or #f and #f."
|
2014-03-08 06:15:38 -05:00
|
|
|
|
|
|
|
|
|
;; Proceed like this:
|
2018-12-26 11:30:56 -05:00
|
|
|
|
;; 1. For all MACHINES, attempt to acquire a build slot, and filter out
|
2014-03-08 06:15:38 -05:00
|
|
|
|
;; those machines for which we failed.
|
2018-12-26 11:30:56 -05:00
|
|
|
|
;; 2. Choose the best machine among those that are left.
|
|
|
|
|
;; 3. Release the previously-acquired build slots of the other machines.
|
|
|
|
|
|
|
|
|
|
(define machines+slots
|
|
|
|
|
(filter-map (lambda (machine)
|
|
|
|
|
(let ((slot (acquire-build-slot machine)))
|
|
|
|
|
(and slot (list machine slot))))
|
|
|
|
|
(shuffle machines)))
|
|
|
|
|
|
|
|
|
|
(define (undecorate pred)
|
|
|
|
|
(lambda (a b)
|
|
|
|
|
(match a
|
|
|
|
|
((machine1 slot1)
|
|
|
|
|
(match b
|
|
|
|
|
((machine2 slot2)
|
|
|
|
|
(pred machine1 machine2)))))))
|
|
|
|
|
|
|
|
|
|
(define (machine-faster? m1 m2)
|
|
|
|
|
;; Return #t if M1 is faster than M2.
|
|
|
|
|
(> (build-machine-speed m1)
|
|
|
|
|
(build-machine-speed m2)))
|
|
|
|
|
|
|
|
|
|
(let loop ((machines+slots
|
|
|
|
|
(sort machines+slots (undecorate machine-faster?))))
|
|
|
|
|
(match machines+slots
|
|
|
|
|
(((best slot) others ...)
|
|
|
|
|
;; Return the best machine unless it's already overloaded.
|
|
|
|
|
;; Note: We call 'node-load' only as a last resort because it is
|
|
|
|
|
;; too costly to call it once for every machine.
|
2019-10-15 06:24:09 -04:00
|
|
|
|
(let* ((session (false-if-exception (open-ssh-session best
|
|
|
|
|
%short-timeout)))
|
2022-11-22 01:17:17 -05:00
|
|
|
|
(node (and session (remote-inferior* session)))
|
2020-10-03 01:17:54 -04:00
|
|
|
|
(load (and node (node-load node)))
|
|
|
|
|
(threshold (build-machine-overload-threshold best))
|
2018-12-26 11:30:56 -05:00
|
|
|
|
(space (and node (node-free-disk-space node))))
|
2020-10-03 01:17:54 -04:00
|
|
|
|
(when load (report-load best load))
|
2018-12-26 11:30:56 -05:00
|
|
|
|
(when node (close-inferior node))
|
|
|
|
|
(when session (disconnect! session))
|
2020-10-03 01:17:54 -04:00
|
|
|
|
(if (and node
|
|
|
|
|
(or (not threshold) (< load threshold))
|
|
|
|
|
(>= space %minimum-disk-space))
|
2018-12-26 11:30:56 -05:00
|
|
|
|
(match others
|
|
|
|
|
(((machines slots) ...)
|
|
|
|
|
;; Release slots from the uninteresting machines.
|
|
|
|
|
(for-each release-build-slot slots)
|
|
|
|
|
|
|
|
|
|
;; The caller must keep SLOT to protect it from GC and to
|
|
|
|
|
;; eventually release it.
|
|
|
|
|
(values best slot)))
|
|
|
|
|
(begin
|
|
|
|
|
;; BEST is unsuitable, so try the next one.
|
|
|
|
|
(when (and space (< space %minimum-disk-space))
|
|
|
|
|
(format (current-error-port)
|
|
|
|
|
"skipping machine '~a' because it is low \
|
2018-12-21 17:12:52 -05:00
|
|
|
|
on disk space (~,2f MiB free)~%"
|
2018-12-26 11:30:56 -05:00
|
|
|
|
(build-machine-name best)
|
|
|
|
|
(/ space (expt 2 20) 1.)))
|
|
|
|
|
(release-build-slot slot)
|
|
|
|
|
(loop others)))))
|
|
|
|
|
(()
|
|
|
|
|
(values #f #f)))))
|
2014-01-23 17:48:34 -05:00
|
|
|
|
|
2018-06-11 05:42:59 -04:00
|
|
|
|
(define (call-with-timeout timeout drv thunk)
|
|
|
|
|
"Call THUNK and leave after TIMEOUT seconds. If TIMEOUT is #f, simply call
|
|
|
|
|
THUNK. Use DRV as an indication of what we were building when the timeout
|
|
|
|
|
expired."
|
|
|
|
|
(if (number? timeout)
|
|
|
|
|
(dynamic-wind
|
|
|
|
|
(lambda ()
|
|
|
|
|
(sigaction SIGALRM
|
|
|
|
|
(lambda _
|
|
|
|
|
;; The exit code here will be 1, which guix-daemon will
|
|
|
|
|
;; interpret as a transient failure.
|
|
|
|
|
(leave (G_ "timeout expired while offloading '~a'~%")
|
|
|
|
|
(derivation-file-name drv))))
|
|
|
|
|
(alarm timeout))
|
|
|
|
|
thunk
|
|
|
|
|
(lambda ()
|
|
|
|
|
(alarm 0)))
|
|
|
|
|
(thunk)))
|
|
|
|
|
|
|
|
|
|
(define-syntax-rule (with-timeout timeout drv exp ...)
|
|
|
|
|
"Evaluate EXP... and leave after TIMEOUT seconds if EXP hasn't completed.
|
|
|
|
|
If TIMEOUT is #f, simply evaluate EXP..."
|
|
|
|
|
(call-with-timeout timeout drv (lambda () exp ...)))
|
|
|
|
|
|
2021-12-15 06:45:11 -05:00
|
|
|
|
(define (check-ssh-zlib-support)
|
|
|
|
|
"Warn once if libssh lacks zlib support."
|
|
|
|
|
;; We rely on protocol-level compression from libssh to optimize large data
|
|
|
|
|
;; transfers. Warn if it's missing.
|
|
|
|
|
(unless (zlib-support?)
|
|
|
|
|
(warning (G_ "Guile-SSH lacks zlib support"))
|
|
|
|
|
(warning (G_ "data transfers will *not* be compressed!")))
|
|
|
|
|
(set! check-ssh-zlib-support (const #t)))
|
|
|
|
|
|
2014-01-23 17:48:34 -05:00
|
|
|
|
(define* (process-request wants-local? system drv features
|
|
|
|
|
#:key
|
|
|
|
|
print-build-trace? (max-silent-time 3600)
|
2014-03-09 18:13:53 -04:00
|
|
|
|
build-timeout)
|
2014-01-23 17:48:34 -05:00
|
|
|
|
"Process a request to build DRV."
|
2014-03-09 09:05:30 -04:00
|
|
|
|
(let* ((local? (and wants-local? (string=? system (%current-system))))
|
|
|
|
|
(reqs (build-requirements
|
|
|
|
|
(system system)
|
|
|
|
|
(features features)))
|
|
|
|
|
(candidates (filter (cut machine-matches? <> reqs)
|
|
|
|
|
(build-machines))))
|
|
|
|
|
(match candidates
|
|
|
|
|
(()
|
|
|
|
|
;; We'll never be able to match REQS.
|
|
|
|
|
(display "# decline\n"))
|
2017-04-21 11:18:54 -04:00
|
|
|
|
((x ...)
|
2017-07-25 15:55:20 -04:00
|
|
|
|
(let-values (((machine slot)
|
|
|
|
|
(choose-build-machine candidates)))
|
2014-03-09 09:05:30 -04:00
|
|
|
|
(if machine
|
2017-07-25 15:55:20 -04:00
|
|
|
|
(dynamic-wind
|
|
|
|
|
(const #f)
|
|
|
|
|
(lambda ()
|
|
|
|
|
;; Offload DRV to MACHINE.
|
|
|
|
|
(display "# accept\n")
|
2021-12-15 06:45:11 -05:00
|
|
|
|
(check-ssh-zlib-support)
|
2021-12-15 06:33:28 -05:00
|
|
|
|
(let ((drv (read-derivation-from-file drv))
|
|
|
|
|
(inputs (string-tokenize (read-line)))
|
2017-07-25 15:55:20 -04:00
|
|
|
|
(outputs (string-tokenize (read-line))))
|
2018-06-11 05:42:59 -04:00
|
|
|
|
;; Even if BUILD-TIMEOUT is honored by MACHINE, there can
|
|
|
|
|
;; be issues with the connection or deadlocks that could
|
|
|
|
|
;; lead the 'guix offload' process to remain stuck forever.
|
|
|
|
|
;; To avoid that, install a timeout here as well.
|
|
|
|
|
(with-timeout build-timeout drv
|
|
|
|
|
(transfer-and-offload drv machine
|
|
|
|
|
#:inputs inputs
|
|
|
|
|
#:outputs outputs
|
|
|
|
|
#:max-silent-time max-silent-time
|
|
|
|
|
#:build-timeout build-timeout
|
|
|
|
|
#:print-build-trace?
|
|
|
|
|
print-build-trace?))))
|
2017-07-25 15:55:20 -04:00
|
|
|
|
(lambda ()
|
|
|
|
|
(release-build-slot slot)))
|
2014-03-09 09:05:30 -04:00
|
|
|
|
|
|
|
|
|
;; Not now, all the machines are busy.
|
|
|
|
|
(display "# postpone\n")))))))
|
2014-01-23 17:48:34 -05:00
|
|
|
|
|
2016-12-05 12:16:04 -05:00
|
|
|
|
|
|
|
|
|
;;;
|
|
|
|
|
;;; Installation tests.
|
|
|
|
|
;;;
|
|
|
|
|
|
2019-10-15 06:24:09 -04:00
|
|
|
|
(define %short-timeout
|
|
|
|
|
;; Timeout in seconds used on SSH connections where reads and writes
|
|
|
|
|
;; shouldn't take long.
|
|
|
|
|
15)
|
|
|
|
|
|
2016-12-05 12:16:04 -05:00
|
|
|
|
(define (assert-node-repl node name)
|
|
|
|
|
"Bail out if NODE is not running Guile."
|
|
|
|
|
(match (node-guile-version node)
|
|
|
|
|
(#f
|
2018-01-12 17:16:53 -05:00
|
|
|
|
(report-guile-error name))
|
2016-12-05 12:16:04 -05:00
|
|
|
|
((? string? version)
|
offload: Use (guix inferior) instead of (ssh dist node).
Using inferiors and thus 'guix repl' simplifies setup on build
machines (no need to worry about GUILE_LOAD_PATH etc.)
Furthermore, the 'guix repl -t machine' protocol running in a remote
pipe addresses several issues with the current implementation of nodes
and RREPLs in Guile-SSH: fewer round trips, doesn't leave a 'guile
--listen' process behind it, stateless (since a new process is started
each time), more efficient (the SSH channel can be reused), more
reliable (no 'pgrep', 'pkill', and shellology; see
<https://github.com/artyom-poptsov/guile-ssh/issues/11> as an example.)
* guix/ssh.scm (inferior-remote-eval): New procedure.
(send-files): Use it instead of 'make-node' and 'node-eval'.
* guix/scripts/offload.scm (node-guile-version): New procedure.
(node-free-disk-space, transfer-and-offload, node-load)
(choose-build-machine, assert-node-has-guix): Use 'remote-inferior'
instead of 'make-node' and 'inferior-eval' instead of 'node-eval'.
(assert-node-can-import, assert-node-can-export): Likewise, and add
'session' parameter.
(check-machine-availability): Likewise, and add calls to
'close-inferior' and 'disconnect!'.
(check-machine-status): Likewise.
* doc/guix.texi (Daemon Offload Setup): Remove bit related to 'guile' in
$PATH and $GUILE_LOAD_PATH; mention 'guix' alone.
2018-12-24 09:40:04 -05:00
|
|
|
|
(info (G_ "'~a' is running GNU Guile ~a~%")
|
2016-12-05 12:16:04 -05:00
|
|
|
|
name (node-guile-version node)))))
|
|
|
|
|
|
|
|
|
|
(define (assert-node-has-guix node name)
|
2018-12-25 11:03:37 -05:00
|
|
|
|
"Bail out if NODE if #f or if we fail to use the (guix) module, or if its
|
|
|
|
|
daemon is not running."
|
|
|
|
|
(unless (inferior? node)
|
|
|
|
|
(leave (G_ "failed to run 'guix repl' on '~a'~%") name))
|
|
|
|
|
|
|
|
|
|
(match (inferior-eval '(begin
|
|
|
|
|
(use-modules (guix))
|
|
|
|
|
(and add-text-to-store 'alright))
|
|
|
|
|
node)
|
|
|
|
|
('alright #t)
|
2020-12-18 05:42:57 -05:00
|
|
|
|
(_ (leave (G_ "(guix) module not usable on remote host '~a'")
|
|
|
|
|
name)))
|
2018-12-25 11:03:37 -05:00
|
|
|
|
|
|
|
|
|
(match (inferior-eval '(begin
|
|
|
|
|
(use-modules (guix))
|
|
|
|
|
(with-store store
|
|
|
|
|
(add-text-to-store store "test"
|
|
|
|
|
"Hello, build machine!")))
|
|
|
|
|
node)
|
|
|
|
|
((? string? str)
|
|
|
|
|
(info (G_ "Guix is usable on '~a' (test returned ~s)~%")
|
|
|
|
|
name str))
|
|
|
|
|
(x
|
|
|
|
|
(leave (G_ "failed to talk to guix-daemon on '~a' (test returned ~s)~%")
|
|
|
|
|
name x))))
|
2016-12-05 12:16:04 -05:00
|
|
|
|
|
|
|
|
|
(define %random-state
|
|
|
|
|
(delay
|
|
|
|
|
(seed->random-state (logxor (getpid) (car (gettimeofday))))))
|
|
|
|
|
|
2016-12-30 17:22:27 -05:00
|
|
|
|
(define* (nonce #:optional (name (gethostname)))
|
|
|
|
|
(string-append name "-"
|
2016-12-05 12:16:04 -05:00
|
|
|
|
(number->string (random 1000000 (force %random-state)))))
|
|
|
|
|
|
offload: Use (guix inferior) instead of (ssh dist node).
Using inferiors and thus 'guix repl' simplifies setup on build
machines (no need to worry about GUILE_LOAD_PATH etc.)
Furthermore, the 'guix repl -t machine' protocol running in a remote
pipe addresses several issues with the current implementation of nodes
and RREPLs in Guile-SSH: fewer round trips, doesn't leave a 'guile
--listen' process behind it, stateless (since a new process is started
each time), more efficient (the SSH channel can be reused), more
reliable (no 'pgrep', 'pkill', and shellology; see
<https://github.com/artyom-poptsov/guile-ssh/issues/11> as an example.)
* guix/ssh.scm (inferior-remote-eval): New procedure.
(send-files): Use it instead of 'make-node' and 'node-eval'.
* guix/scripts/offload.scm (node-guile-version): New procedure.
(node-free-disk-space, transfer-and-offload, node-load)
(choose-build-machine, assert-node-has-guix): Use 'remote-inferior'
instead of 'make-node' and 'inferior-eval' instead of 'node-eval'.
(assert-node-can-import, assert-node-can-export): Likewise, and add
'session' parameter.
(check-machine-availability): Likewise, and add calls to
'close-inferior' and 'disconnect!'.
(check-machine-status): Likewise.
* doc/guix.texi (Daemon Offload Setup): Remove bit related to 'guile' in
$PATH and $GUILE_LOAD_PATH; mention 'guix' alone.
2018-12-24 09:40:04 -05:00
|
|
|
|
(define (assert-node-can-import session node name daemon-socket)
|
2016-12-05 12:16:04 -05:00
|
|
|
|
"Bail out if NODE refuses to import our archives."
|
offload: Use (guix inferior) instead of (ssh dist node).
Using inferiors and thus 'guix repl' simplifies setup on build
machines (no need to worry about GUILE_LOAD_PATH etc.)
Furthermore, the 'guix repl -t machine' protocol running in a remote
pipe addresses several issues with the current implementation of nodes
and RREPLs in Guile-SSH: fewer round trips, doesn't leave a 'guile
--listen' process behind it, stateless (since a new process is started
each time), more efficient (the SSH channel can be reused), more
reliable (no 'pgrep', 'pkill', and shellology; see
<https://github.com/artyom-poptsov/guile-ssh/issues/11> as an example.)
* guix/ssh.scm (inferior-remote-eval): New procedure.
(send-files): Use it instead of 'make-node' and 'node-eval'.
* guix/scripts/offload.scm (node-guile-version): New procedure.
(node-free-disk-space, transfer-and-offload, node-load)
(choose-build-machine, assert-node-has-guix): Use 'remote-inferior'
instead of 'make-node' and 'inferior-eval' instead of 'node-eval'.
(assert-node-can-import, assert-node-can-export): Likewise, and add
'session' parameter.
(check-machine-availability): Likewise, and add calls to
'close-inferior' and 'disconnect!'.
(check-machine-status): Likewise.
* doc/guix.texi (Daemon Offload Setup): Remove bit related to 'guile' in
$PATH and $GUILE_LOAD_PATH; mention 'guix' alone.
2018-12-24 09:40:04 -05:00
|
|
|
|
(with-store store
|
|
|
|
|
(let* ((item (add-text-to-store store "export-test" (nonce)))
|
|
|
|
|
(remote (connect-to-remote-daemon session daemon-socket)))
|
|
|
|
|
(with-store local
|
|
|
|
|
(send-files local (list item) remote))
|
|
|
|
|
|
|
|
|
|
(if (valid-path? remote item)
|
|
|
|
|
(info (G_ "'~a' successfully imported '~a'~%")
|
|
|
|
|
name item)
|
|
|
|
|
(leave (G_ "'~a' was not properly imported on '~a'~%")
|
|
|
|
|
item name)))))
|
|
|
|
|
|
|
|
|
|
(define (assert-node-can-export session node name daemon-socket)
|
2016-12-05 12:16:04 -05:00
|
|
|
|
"Bail out if we cannot import signed archives from NODE."
|
offload: Use (guix inferior) instead of (ssh dist node).
Using inferiors and thus 'guix repl' simplifies setup on build
machines (no need to worry about GUILE_LOAD_PATH etc.)
Furthermore, the 'guix repl -t machine' protocol running in a remote
pipe addresses several issues with the current implementation of nodes
and RREPLs in Guile-SSH: fewer round trips, doesn't leave a 'guile
--listen' process behind it, stateless (since a new process is started
each time), more efficient (the SSH channel can be reused), more
reliable (no 'pgrep', 'pkill', and shellology; see
<https://github.com/artyom-poptsov/guile-ssh/issues/11> as an example.)
* guix/ssh.scm (inferior-remote-eval): New procedure.
(send-files): Use it instead of 'make-node' and 'node-eval'.
* guix/scripts/offload.scm (node-guile-version): New procedure.
(node-free-disk-space, transfer-and-offload, node-load)
(choose-build-machine, assert-node-has-guix): Use 'remote-inferior'
instead of 'make-node' and 'inferior-eval' instead of 'node-eval'.
(assert-node-can-import, assert-node-can-export): Likewise, and add
'session' parameter.
(check-machine-availability): Likewise, and add calls to
'close-inferior' and 'disconnect!'.
(check-machine-status): Likewise.
* doc/guix.texi (Daemon Offload Setup): Remove bit related to 'guile' in
$PATH and $GUILE_LOAD_PATH; mention 'guix' alone.
2018-12-24 09:40:04 -05:00
|
|
|
|
(let* ((remote (connect-to-remote-daemon session daemon-socket))
|
2016-12-30 17:22:27 -05:00
|
|
|
|
(item (add-text-to-store remote "import-test" (nonce name))))
|
2016-12-05 12:16:04 -05:00
|
|
|
|
(with-store store
|
2016-12-30 17:22:27 -05:00
|
|
|
|
(if (and (retrieve-files store (list item) remote)
|
2016-12-05 12:16:04 -05:00
|
|
|
|
(valid-path? store item))
|
ui: Rename '_' to 'G_'.
This avoids collisions with '_' when the latter is used as a 'match'
pattern for instance. See
<https://lists.gnu.org/archive/html/guix-devel/2017-04/msg00464.html>.
* guix/ui.scm: Rename '_' to 'G_'.
* po/guix/Makevars (XGETTEXT_OPTIONS): Adjust accordingly.
* build-aux/compile-all.scm (warnings): Remove 'format'.
* gnu/packages.scm,
gnu/services.scm,
gnu/services/shepherd.scm,
gnu/system.scm,
gnu/system/shadow.scm,
guix/gnupg.scm,
guix/http-client.scm,
guix/import/cpan.scm,
guix/import/elpa.scm,
guix/import/pypi.scm,
guix/nar.scm,
guix/scripts.scm,
guix/scripts/archive.scm,
guix/scripts/authenticate.scm,
guix/scripts/build.scm,
guix/scripts/challenge.scm,
guix/scripts/container.scm,
guix/scripts/container/exec.scm,
guix/scripts/copy.scm,
guix/scripts/download.scm,
guix/scripts/edit.scm,
guix/scripts/environment.scm,
guix/scripts/gc.scm,
guix/scripts/graph.scm,
guix/scripts/hash.scm,
guix/scripts/import.scm,
guix/scripts/import/cpan.scm,
guix/scripts/import/cran.scm,
guix/scripts/import/crate.scm,
guix/scripts/import/elpa.scm,
guix/scripts/import/gem.scm,
guix/scripts/import/gnu.scm,
guix/scripts/import/hackage.scm,
guix/scripts/import/nix.scm,
guix/scripts/import/pypi.scm,
guix/scripts/import/stackage.scm,
guix/scripts/lint.scm,
guix/scripts/offload.scm,
guix/scripts/pack.scm,
guix/scripts/package.scm,
guix/scripts/perform-download.scm,
guix/scripts/publish.scm,
guix/scripts/pull.scm,
guix/scripts/refresh.scm,
guix/scripts/size.scm,
guix/scripts/substitute.scm,
guix/scripts/system.scm,
guix/ssh.scm,
guix/upstream.scm: Use 'G_' instead of '_'. Most of this change was
obtained by running: "sed -i -e's/(_ "/(G_ "/g' `find -name \*.scm`".
2017-05-03 09:57:02 -04:00
|
|
|
|
(info (G_ "successfully imported '~a' from '~a'~%")
|
2016-12-05 12:16:04 -05:00
|
|
|
|
item name)
|
ui: Rename '_' to 'G_'.
This avoids collisions with '_' when the latter is used as a 'match'
pattern for instance. See
<https://lists.gnu.org/archive/html/guix-devel/2017-04/msg00464.html>.
* guix/ui.scm: Rename '_' to 'G_'.
* po/guix/Makevars (XGETTEXT_OPTIONS): Adjust accordingly.
* build-aux/compile-all.scm (warnings): Remove 'format'.
* gnu/packages.scm,
gnu/services.scm,
gnu/services/shepherd.scm,
gnu/system.scm,
gnu/system/shadow.scm,
guix/gnupg.scm,
guix/http-client.scm,
guix/import/cpan.scm,
guix/import/elpa.scm,
guix/import/pypi.scm,
guix/nar.scm,
guix/scripts.scm,
guix/scripts/archive.scm,
guix/scripts/authenticate.scm,
guix/scripts/build.scm,
guix/scripts/challenge.scm,
guix/scripts/container.scm,
guix/scripts/container/exec.scm,
guix/scripts/copy.scm,
guix/scripts/download.scm,
guix/scripts/edit.scm,
guix/scripts/environment.scm,
guix/scripts/gc.scm,
guix/scripts/graph.scm,
guix/scripts/hash.scm,
guix/scripts/import.scm,
guix/scripts/import/cpan.scm,
guix/scripts/import/cran.scm,
guix/scripts/import/crate.scm,
guix/scripts/import/elpa.scm,
guix/scripts/import/gem.scm,
guix/scripts/import/gnu.scm,
guix/scripts/import/hackage.scm,
guix/scripts/import/nix.scm,
guix/scripts/import/pypi.scm,
guix/scripts/import/stackage.scm,
guix/scripts/lint.scm,
guix/scripts/offload.scm,
guix/scripts/pack.scm,
guix/scripts/package.scm,
guix/scripts/perform-download.scm,
guix/scripts/publish.scm,
guix/scripts/pull.scm,
guix/scripts/refresh.scm,
guix/scripts/size.scm,
guix/scripts/substitute.scm,
guix/scripts/system.scm,
guix/ssh.scm,
guix/upstream.scm: Use 'G_' instead of '_'. Most of this change was
obtained by running: "sed -i -e's/(_ "/(G_ "/g' `find -name \*.scm`".
2017-05-03 09:57:02 -04:00
|
|
|
|
(leave (G_ "failed to import '~a' from '~a'~%")
|
2016-12-05 12:16:04 -05:00
|
|
|
|
item name)))))
|
|
|
|
|
|
2016-12-09 17:12:06 -05:00
|
|
|
|
(define (check-machine-availability machine-file pred)
|
|
|
|
|
"Check that each machine matching PRED in MACHINE-FILE is usable as a build
|
|
|
|
|
machine."
|
2016-12-09 17:00:08 -05:00
|
|
|
|
(define (build-machine=? m1 m2)
|
|
|
|
|
(and (string=? (build-machine-name m1) (build-machine-name m2))
|
|
|
|
|
(= (build-machine-port m1) (build-machine-port m2))))
|
|
|
|
|
|
2022-11-22 01:17:17 -05:00
|
|
|
|
(define (if-true proc)
|
|
|
|
|
(lambda args
|
|
|
|
|
(when (every ->bool args)
|
|
|
|
|
(apply proc args))))
|
|
|
|
|
|
2016-12-09 17:00:08 -05:00
|
|
|
|
;; A given build machine may appear several times (e.g., once for
|
|
|
|
|
;; "x86_64-linux" and a second time for "i686-linux"); test them only once.
|
2016-12-09 17:12:06 -05:00
|
|
|
|
(let ((machines (filter pred
|
|
|
|
|
(delete-duplicates (build-machines machine-file)
|
|
|
|
|
build-machine=?))))
|
ui: Rename '_' to 'G_'.
This avoids collisions with '_' when the latter is used as a 'match'
pattern for instance. See
<https://lists.gnu.org/archive/html/guix-devel/2017-04/msg00464.html>.
* guix/ui.scm: Rename '_' to 'G_'.
* po/guix/Makevars (XGETTEXT_OPTIONS): Adjust accordingly.
* build-aux/compile-all.scm (warnings): Remove 'format'.
* gnu/packages.scm,
gnu/services.scm,
gnu/services/shepherd.scm,
gnu/system.scm,
gnu/system/shadow.scm,
guix/gnupg.scm,
guix/http-client.scm,
guix/import/cpan.scm,
guix/import/elpa.scm,
guix/import/pypi.scm,
guix/nar.scm,
guix/scripts.scm,
guix/scripts/archive.scm,
guix/scripts/authenticate.scm,
guix/scripts/build.scm,
guix/scripts/challenge.scm,
guix/scripts/container.scm,
guix/scripts/container/exec.scm,
guix/scripts/copy.scm,
guix/scripts/download.scm,
guix/scripts/edit.scm,
guix/scripts/environment.scm,
guix/scripts/gc.scm,
guix/scripts/graph.scm,
guix/scripts/hash.scm,
guix/scripts/import.scm,
guix/scripts/import/cpan.scm,
guix/scripts/import/cran.scm,
guix/scripts/import/crate.scm,
guix/scripts/import/elpa.scm,
guix/scripts/import/gem.scm,
guix/scripts/import/gnu.scm,
guix/scripts/import/hackage.scm,
guix/scripts/import/nix.scm,
guix/scripts/import/pypi.scm,
guix/scripts/import/stackage.scm,
guix/scripts/lint.scm,
guix/scripts/offload.scm,
guix/scripts/pack.scm,
guix/scripts/package.scm,
guix/scripts/perform-download.scm,
guix/scripts/publish.scm,
guix/scripts/pull.scm,
guix/scripts/refresh.scm,
guix/scripts/size.scm,
guix/scripts/substitute.scm,
guix/scripts/system.scm,
guix/ssh.scm,
guix/upstream.scm: Use 'G_' instead of '_'. Most of this change was
obtained by running: "sed -i -e's/(_ "/(G_ "/g' `find -name \*.scm`".
2017-05-03 09:57:02 -04:00
|
|
|
|
(info (G_ "testing ~a build machines defined in '~a'...~%")
|
2016-12-05 12:16:04 -05:00
|
|
|
|
(length machines) machine-file)
|
|
|
|
|
(let* ((names (map build-machine-name machines))
|
|
|
|
|
(sockets (map build-machine-daemon-socket machines))
|
2019-10-15 06:24:09 -04:00
|
|
|
|
(sessions (map (cut open-ssh-session <> %short-timeout) machines))
|
2022-11-22 01:17:17 -05:00
|
|
|
|
(nodes (map remote-inferior* sessions)))
|
|
|
|
|
(for-each (if-true assert-node-has-guix) nodes names)
|
|
|
|
|
(for-each (if-true assert-node-repl) nodes names)
|
|
|
|
|
(for-each (if-true assert-node-can-import) sessions nodes names sockets)
|
|
|
|
|
(for-each (if-true assert-node-can-export) sessions nodes names sockets)
|
|
|
|
|
(for-each (if-true close-inferior) nodes)
|
offload: Use (guix inferior) instead of (ssh dist node).
Using inferiors and thus 'guix repl' simplifies setup on build
machines (no need to worry about GUILE_LOAD_PATH etc.)
Furthermore, the 'guix repl -t machine' protocol running in a remote
pipe addresses several issues with the current implementation of nodes
and RREPLs in Guile-SSH: fewer round trips, doesn't leave a 'guile
--listen' process behind it, stateless (since a new process is started
each time), more efficient (the SSH channel can be reused), more
reliable (no 'pgrep', 'pkill', and shellology; see
<https://github.com/artyom-poptsov/guile-ssh/issues/11> as an example.)
* guix/ssh.scm (inferior-remote-eval): New procedure.
(send-files): Use it instead of 'make-node' and 'node-eval'.
* guix/scripts/offload.scm (node-guile-version): New procedure.
(node-free-disk-space, transfer-and-offload, node-load)
(choose-build-machine, assert-node-has-guix): Use 'remote-inferior'
instead of 'make-node' and 'inferior-eval' instead of 'node-eval'.
(assert-node-can-import, assert-node-can-export): Likewise, and add
'session' parameter.
(check-machine-availability): Likewise, and add calls to
'close-inferior' and 'disconnect!'.
(check-machine-status): Likewise.
* doc/guix.texi (Daemon Offload Setup): Remove bit related to 'guile' in
$PATH and $GUILE_LOAD_PATH; mention 'guix' alone.
2018-12-24 09:40:04 -05:00
|
|
|
|
(for-each disconnect! sessions))))
|
2016-12-05 12:16:04 -05:00
|
|
|
|
|
2017-12-13 17:42:40 -05:00
|
|
|
|
(define (check-machine-status machine-file pred)
|
|
|
|
|
"Print the load of each machine matching PRED in MACHINE-FILE."
|
|
|
|
|
(define (build-machine=? m1 m2)
|
|
|
|
|
(and (string=? (build-machine-name m1) (build-machine-name m2))
|
|
|
|
|
(= (build-machine-port m1) (build-machine-port m2))))
|
|
|
|
|
|
|
|
|
|
;; A given build machine may appear several times (e.g., once for
|
|
|
|
|
;; "x86_64-linux" and a second time for "i686-linux"); test them only once.
|
|
|
|
|
(let ((machines (filter pred
|
|
|
|
|
(delete-duplicates (build-machines machine-file)
|
|
|
|
|
build-machine=?))))
|
|
|
|
|
(info (G_ "getting status of ~a build machines defined in '~a'...~%")
|
|
|
|
|
(length machines) machine-file)
|
|
|
|
|
(for-each (lambda (machine)
|
2018-12-25 11:03:37 -05:00
|
|
|
|
(define session
|
2019-10-15 06:24:09 -04:00
|
|
|
|
(open-ssh-session machine %short-timeout))
|
2018-12-25 11:03:37 -05:00
|
|
|
|
|
2022-11-22 01:17:17 -05:00
|
|
|
|
(match (remote-inferior* session)
|
2018-12-25 11:03:37 -05:00
|
|
|
|
(#f
|
2022-11-22 01:17:17 -05:00
|
|
|
|
#f)
|
2018-12-25 11:03:37 -05:00
|
|
|
|
((? inferior? inferior)
|
2019-01-22 11:37:59 -05:00
|
|
|
|
(let ((now (car (gettimeofday))))
|
|
|
|
|
(match (inferior-eval '(list (uname)
|
|
|
|
|
(car (gettimeofday)))
|
|
|
|
|
inferior)
|
|
|
|
|
((uts time)
|
|
|
|
|
(when (< time now)
|
|
|
|
|
;; Build machine clocks must not be behind as this
|
|
|
|
|
;; could cause timestamp issues.
|
|
|
|
|
(warning (G_ "machine '~a' is ~a seconds behind~%")
|
|
|
|
|
(build-machine-name machine)
|
|
|
|
|
(- now time)))
|
|
|
|
|
|
|
|
|
|
(let ((load (node-load inferior))
|
|
|
|
|
(free (node-free-disk-space inferior)))
|
|
|
|
|
(close-inferior inferior)
|
|
|
|
|
(format #t "~a~% kernel: ~a ~a~% architecture: ~a~%\
|
2020-10-03 01:17:54 -04:00
|
|
|
|
host name: ~a~% normalized load: ~,2f~% free disk space: ~,2f MiB~%\
|
2019-01-22 11:37:59 -05:00
|
|
|
|
time difference: ~a s~%"
|
|
|
|
|
(build-machine-name machine)
|
|
|
|
|
(utsname:sysname uts) (utsname:release uts)
|
|
|
|
|
(utsname:machine uts)
|
|
|
|
|
(utsname:nodename uts)
|
2020-10-03 01:17:54 -04:00
|
|
|
|
load
|
2019-01-22 11:37:59 -05:00
|
|
|
|
(/ free (expt 2 20) 1.)
|
|
|
|
|
(- time now))))))))
|
2018-12-25 11:03:37 -05:00
|
|
|
|
|
|
|
|
|
(disconnect! session))
|
2017-12-13 17:42:40 -05:00
|
|
|
|
machines)))
|
|
|
|
|
|
2014-01-23 17:48:34 -05:00
|
|
|
|
|
|
|
|
|
;;;
|
|
|
|
|
;;; Entry point.
|
|
|
|
|
;;;
|
|
|
|
|
|
2020-09-01 16:13:11 -04:00
|
|
|
|
(define-command (guix-offload . args)
|
|
|
|
|
(category plumbing)
|
|
|
|
|
(synopsis "set up and operate build offloading")
|
|
|
|
|
|
2014-01-23 17:48:34 -05:00
|
|
|
|
(define request-line-rx
|
|
|
|
|
;; The request format. See 'tryBuildHook' method in build.cc.
|
|
|
|
|
(make-regexp "([01]) ([a-z0-9_-]+) (/[[:graph:]]+.drv) ([[:graph:]]*)"))
|
|
|
|
|
|
|
|
|
|
(define not-coma
|
|
|
|
|
(char-set-complement (char-set #\,)))
|
|
|
|
|
|
|
|
|
|
;; Make sure $HOME really corresponds to the current user. This is
|
|
|
|
|
;; necessary since lsh uses that to determine the location of the yarrow
|
|
|
|
|
;; seed file, and fails if it's owned by someone else.
|
|
|
|
|
(and=> (passwd:dir (getpw (getuid)))
|
|
|
|
|
(cut setenv "HOME" <>))
|
|
|
|
|
|
|
|
|
|
(match args
|
|
|
|
|
((system max-silent-time print-build-trace? build-timeout)
|
|
|
|
|
(let ((max-silent-time (string->number max-silent-time))
|
|
|
|
|
(build-timeout (string->number build-timeout))
|
|
|
|
|
(print-build-trace? (string=? print-build-trace? "1")))
|
2017-05-28 10:09:32 -04:00
|
|
|
|
(set-thread-name "guix offload")
|
2014-01-23 17:48:34 -05:00
|
|
|
|
(parameterize ((%current-system system))
|
|
|
|
|
(let loop ((line (read-line)))
|
|
|
|
|
(unless (eof-object? line)
|
|
|
|
|
(cond ((regexp-exec request-line-rx line)
|
|
|
|
|
=>
|
|
|
|
|
(lambda (match)
|
2016-11-04 19:46:04 -04:00
|
|
|
|
(with-error-handling
|
2014-01-23 17:48:34 -05:00
|
|
|
|
(process-request (equal? (match:substring match 1) "1")
|
|
|
|
|
(match:substring match 2) ; system
|
2021-12-15 06:33:28 -05:00
|
|
|
|
(match:substring match 3)
|
2014-01-23 17:48:34 -05:00
|
|
|
|
(string-tokenize
|
|
|
|
|
(match:substring match 4) not-coma)
|
|
|
|
|
#:print-build-trace? print-build-trace?
|
|
|
|
|
#:max-silent-time max-silent-time
|
|
|
|
|
#:build-timeout build-timeout))))
|
|
|
|
|
(else
|
ui: Rename '_' to 'G_'.
This avoids collisions with '_' when the latter is used as a 'match'
pattern for instance. See
<https://lists.gnu.org/archive/html/guix-devel/2017-04/msg00464.html>.
* guix/ui.scm: Rename '_' to 'G_'.
* po/guix/Makevars (XGETTEXT_OPTIONS): Adjust accordingly.
* build-aux/compile-all.scm (warnings): Remove 'format'.
* gnu/packages.scm,
gnu/services.scm,
gnu/services/shepherd.scm,
gnu/system.scm,
gnu/system/shadow.scm,
guix/gnupg.scm,
guix/http-client.scm,
guix/import/cpan.scm,
guix/import/elpa.scm,
guix/import/pypi.scm,
guix/nar.scm,
guix/scripts.scm,
guix/scripts/archive.scm,
guix/scripts/authenticate.scm,
guix/scripts/build.scm,
guix/scripts/challenge.scm,
guix/scripts/container.scm,
guix/scripts/container/exec.scm,
guix/scripts/copy.scm,
guix/scripts/download.scm,
guix/scripts/edit.scm,
guix/scripts/environment.scm,
guix/scripts/gc.scm,
guix/scripts/graph.scm,
guix/scripts/hash.scm,
guix/scripts/import.scm,
guix/scripts/import/cpan.scm,
guix/scripts/import/cran.scm,
guix/scripts/import/crate.scm,
guix/scripts/import/elpa.scm,
guix/scripts/import/gem.scm,
guix/scripts/import/gnu.scm,
guix/scripts/import/hackage.scm,
guix/scripts/import/nix.scm,
guix/scripts/import/pypi.scm,
guix/scripts/import/stackage.scm,
guix/scripts/lint.scm,
guix/scripts/offload.scm,
guix/scripts/pack.scm,
guix/scripts/package.scm,
guix/scripts/perform-download.scm,
guix/scripts/publish.scm,
guix/scripts/pull.scm,
guix/scripts/refresh.scm,
guix/scripts/size.scm,
guix/scripts/substitute.scm,
guix/scripts/system.scm,
guix/ssh.scm,
guix/upstream.scm: Use 'G_' instead of '_'. Most of this change was
obtained by running: "sed -i -e's/(_ "/(G_ "/g' `find -name \*.scm`".
2017-05-03 09:57:02 -04:00
|
|
|
|
(leave (G_ "invalid request line: ~s~%") line)))
|
2014-01-23 17:48:34 -05:00
|
|
|
|
(loop (read-line)))))))
|
2016-12-05 12:16:04 -05:00
|
|
|
|
(("test" rest ...)
|
|
|
|
|
(with-error-handling
|
2016-12-09 17:12:06 -05:00
|
|
|
|
(let-values (((file pred)
|
|
|
|
|
(match rest
|
|
|
|
|
((file regexp)
|
|
|
|
|
(values file
|
|
|
|
|
(compose (cut string-match regexp <>)
|
|
|
|
|
build-machine-name)))
|
|
|
|
|
((file) (values file (const #t)))
|
|
|
|
|
(() (values %machine-file (const #t)))
|
ui: Rename '_' to 'G_'.
This avoids collisions with '_' when the latter is used as a 'match'
pattern for instance. See
<https://lists.gnu.org/archive/html/guix-devel/2017-04/msg00464.html>.
* guix/ui.scm: Rename '_' to 'G_'.
* po/guix/Makevars (XGETTEXT_OPTIONS): Adjust accordingly.
* build-aux/compile-all.scm (warnings): Remove 'format'.
* gnu/packages.scm,
gnu/services.scm,
gnu/services/shepherd.scm,
gnu/system.scm,
gnu/system/shadow.scm,
guix/gnupg.scm,
guix/http-client.scm,
guix/import/cpan.scm,
guix/import/elpa.scm,
guix/import/pypi.scm,
guix/nar.scm,
guix/scripts.scm,
guix/scripts/archive.scm,
guix/scripts/authenticate.scm,
guix/scripts/build.scm,
guix/scripts/challenge.scm,
guix/scripts/container.scm,
guix/scripts/container/exec.scm,
guix/scripts/copy.scm,
guix/scripts/download.scm,
guix/scripts/edit.scm,
guix/scripts/environment.scm,
guix/scripts/gc.scm,
guix/scripts/graph.scm,
guix/scripts/hash.scm,
guix/scripts/import.scm,
guix/scripts/import/cpan.scm,
guix/scripts/import/cran.scm,
guix/scripts/import/crate.scm,
guix/scripts/import/elpa.scm,
guix/scripts/import/gem.scm,
guix/scripts/import/gnu.scm,
guix/scripts/import/hackage.scm,
guix/scripts/import/nix.scm,
guix/scripts/import/pypi.scm,
guix/scripts/import/stackage.scm,
guix/scripts/lint.scm,
guix/scripts/offload.scm,
guix/scripts/pack.scm,
guix/scripts/package.scm,
guix/scripts/perform-download.scm,
guix/scripts/publish.scm,
guix/scripts/pull.scm,
guix/scripts/refresh.scm,
guix/scripts/size.scm,
guix/scripts/substitute.scm,
guix/scripts/system.scm,
guix/ssh.scm,
guix/upstream.scm: Use 'G_' instead of '_'. Most of this change was
obtained by running: "sed -i -e's/(_ "/(G_ "/g' `find -name \*.scm`".
2017-05-03 09:57:02 -04:00
|
|
|
|
(x (leave (G_ "wrong number of arguments~%"))))))
|
2016-12-09 17:12:06 -05:00
|
|
|
|
(check-machine-availability (or file %machine-file) pred))))
|
2017-12-13 17:42:40 -05:00
|
|
|
|
(("status" rest ...)
|
|
|
|
|
(with-error-handling
|
|
|
|
|
(let-values (((file pred)
|
|
|
|
|
(match rest
|
|
|
|
|
((file regexp)
|
|
|
|
|
(values file
|
|
|
|
|
(compose (cut string-match regexp <>)
|
|
|
|
|
build-machine-name)))
|
|
|
|
|
((file) (values file (const #t)))
|
|
|
|
|
(() (values %machine-file (const #t)))
|
|
|
|
|
(x (leave (G_ "wrong number of arguments~%"))))))
|
|
|
|
|
(check-machine-status (or file %machine-file) pred))))
|
2014-01-23 17:48:34 -05:00
|
|
|
|
(("--version")
|
|
|
|
|
(show-version-and-exit "guix offload"))
|
|
|
|
|
(("--help")
|
2023-10-16 12:42:43 -04:00
|
|
|
|
(leave-on-EPIPE
|
|
|
|
|
(format #t (G_ "Usage: guix offload SYSTEM MAX-SILENT-TIME \
|
2020-08-24 16:23:16 -04:00
|
|
|
|
PRINT-BUILD-TRACE? BUILD-TIMEOUT
|
2014-01-23 17:48:34 -05:00
|
|
|
|
Process build offload requests written on the standard input, possibly
|
|
|
|
|
offloading builds to the machines listed in '~a'.~%")
|
2023-10-16 12:42:43 -04:00
|
|
|
|
%machine-file))
|
ui: Rename '_' to 'G_'.
This avoids collisions with '_' when the latter is used as a 'match'
pattern for instance. See
<https://lists.gnu.org/archive/html/guix-devel/2017-04/msg00464.html>.
* guix/ui.scm: Rename '_' to 'G_'.
* po/guix/Makevars (XGETTEXT_OPTIONS): Adjust accordingly.
* build-aux/compile-all.scm (warnings): Remove 'format'.
* gnu/packages.scm,
gnu/services.scm,
gnu/services/shepherd.scm,
gnu/system.scm,
gnu/system/shadow.scm,
guix/gnupg.scm,
guix/http-client.scm,
guix/import/cpan.scm,
guix/import/elpa.scm,
guix/import/pypi.scm,
guix/nar.scm,
guix/scripts.scm,
guix/scripts/archive.scm,
guix/scripts/authenticate.scm,
guix/scripts/build.scm,
guix/scripts/challenge.scm,
guix/scripts/container.scm,
guix/scripts/container/exec.scm,
guix/scripts/copy.scm,
guix/scripts/download.scm,
guix/scripts/edit.scm,
guix/scripts/environment.scm,
guix/scripts/gc.scm,
guix/scripts/graph.scm,
guix/scripts/hash.scm,
guix/scripts/import.scm,
guix/scripts/import/cpan.scm,
guix/scripts/import/cran.scm,
guix/scripts/import/crate.scm,
guix/scripts/import/elpa.scm,
guix/scripts/import/gem.scm,
guix/scripts/import/gnu.scm,
guix/scripts/import/hackage.scm,
guix/scripts/import/nix.scm,
guix/scripts/import/pypi.scm,
guix/scripts/import/stackage.scm,
guix/scripts/lint.scm,
guix/scripts/offload.scm,
guix/scripts/pack.scm,
guix/scripts/package.scm,
guix/scripts/perform-download.scm,
guix/scripts/publish.scm,
guix/scripts/pull.scm,
guix/scripts/refresh.scm,
guix/scripts/size.scm,
guix/scripts/substitute.scm,
guix/scripts/system.scm,
guix/ssh.scm,
guix/upstream.scm: Use 'G_' instead of '_'. Most of this change was
obtained by running: "sed -i -e's/(_ "/(G_ "/g' `find -name \*.scm`".
2017-05-03 09:57:02 -04:00
|
|
|
|
(display (G_ "
|
2014-01-23 17:48:34 -05:00
|
|
|
|
This tool is meant to be used internally by 'guix-daemon'.\n"))
|
|
|
|
|
(show-bug-report-information))
|
|
|
|
|
(x
|
ui: Rename '_' to 'G_'.
This avoids collisions with '_' when the latter is used as a 'match'
pattern for instance. See
<https://lists.gnu.org/archive/html/guix-devel/2017-04/msg00464.html>.
* guix/ui.scm: Rename '_' to 'G_'.
* po/guix/Makevars (XGETTEXT_OPTIONS): Adjust accordingly.
* build-aux/compile-all.scm (warnings): Remove 'format'.
* gnu/packages.scm,
gnu/services.scm,
gnu/services/shepherd.scm,
gnu/system.scm,
gnu/system/shadow.scm,
guix/gnupg.scm,
guix/http-client.scm,
guix/import/cpan.scm,
guix/import/elpa.scm,
guix/import/pypi.scm,
guix/nar.scm,
guix/scripts.scm,
guix/scripts/archive.scm,
guix/scripts/authenticate.scm,
guix/scripts/build.scm,
guix/scripts/challenge.scm,
guix/scripts/container.scm,
guix/scripts/container/exec.scm,
guix/scripts/copy.scm,
guix/scripts/download.scm,
guix/scripts/edit.scm,
guix/scripts/environment.scm,
guix/scripts/gc.scm,
guix/scripts/graph.scm,
guix/scripts/hash.scm,
guix/scripts/import.scm,
guix/scripts/import/cpan.scm,
guix/scripts/import/cran.scm,
guix/scripts/import/crate.scm,
guix/scripts/import/elpa.scm,
guix/scripts/import/gem.scm,
guix/scripts/import/gnu.scm,
guix/scripts/import/hackage.scm,
guix/scripts/import/nix.scm,
guix/scripts/import/pypi.scm,
guix/scripts/import/stackage.scm,
guix/scripts/lint.scm,
guix/scripts/offload.scm,
guix/scripts/pack.scm,
guix/scripts/package.scm,
guix/scripts/perform-download.scm,
guix/scripts/publish.scm,
guix/scripts/pull.scm,
guix/scripts/refresh.scm,
guix/scripts/size.scm,
guix/scripts/substitute.scm,
guix/scripts/system.scm,
guix/ssh.scm,
guix/upstream.scm: Use 'G_' instead of '_'. Most of this change was
obtained by running: "sed -i -e's/(_ "/(G_ "/g' `find -name \*.scm`".
2017-05-03 09:57:02 -04:00
|
|
|
|
(leave (G_ "invalid arguments: ~{~s ~}~%") x))))
|
2014-01-23 17:48:34 -05:00
|
|
|
|
|
2014-03-06 15:38:45 -05:00
|
|
|
|
;;; Local Variables:
|
2014-03-19 18:12:06 -04:00
|
|
|
|
;;; eval: (put 'with-error-to-port 'scheme-indent-function 1)
|
2018-06-11 05:42:59 -04:00
|
|
|
|
;;; eval: (put 'with-timeout 'scheme-indent-function 2)
|
2014-03-06 15:38:45 -05:00
|
|
|
|
;;; End:
|
|
|
|
|
|
2014-01-23 17:48:34 -05:00
|
|
|
|
;;; offload.scm ends here
|