mirror of
https://git.in.rschanz.org/ryan77627/guix.git
synced 2025-01-11 13:49:23 -05:00
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.
This commit is contained in:
parent
af15fe13b6
commit
ed7b44370f
3 changed files with 80 additions and 63 deletions
|
@ -1051,13 +1051,11 @@ name, and they will be scheduled on matching build machines.
|
||||||
@end table
|
@end table
|
||||||
@end deftp
|
@end deftp
|
||||||
|
|
||||||
The @code{guile} command must be in the search path on the build
|
The @command{guix} command must be in the search path on the build
|
||||||
machines. In addition, the Guix modules must be in
|
machines. You can check whether this is the case by running:
|
||||||
@code{$GUILE_LOAD_PATH} on the build machine---you can check whether
|
|
||||||
this is the case by running:
|
|
||||||
|
|
||||||
@example
|
@example
|
||||||
ssh build-machine guile -c "'(use-modules (guix config))'"
|
ssh build-machine guix repl --version
|
||||||
@end example
|
@end example
|
||||||
|
|
||||||
There is one last thing to do once @file{machines.scm} is in place. As
|
There is one last thing to do once @file{machines.scm} is in place. As
|
||||||
|
|
|
@ -23,13 +23,12 @@ (define-module (guix scripts offload)
|
||||||
#:use-module (ssh session)
|
#:use-module (ssh session)
|
||||||
#:use-module (ssh channel)
|
#:use-module (ssh channel)
|
||||||
#:use-module (ssh popen)
|
#:use-module (ssh popen)
|
||||||
#:use-module (ssh dist)
|
|
||||||
#:use-module (ssh dist node)
|
|
||||||
#:use-module (ssh version)
|
#:use-module (ssh version)
|
||||||
#:use-module (guix config)
|
#:use-module (guix config)
|
||||||
#:use-module (guix records)
|
#:use-module (guix records)
|
||||||
#:use-module (guix ssh)
|
#:use-module (guix ssh)
|
||||||
#:use-module (guix store)
|
#:use-module (guix store)
|
||||||
|
#:use-module (guix inferior)
|
||||||
#:use-module (guix derivations)
|
#:use-module (guix derivations)
|
||||||
#:use-module ((guix serialization)
|
#:use-module ((guix serialization)
|
||||||
#:select (nar-error? nar-error-file))
|
#:select (nar-error? nar-error-file))
|
||||||
|
@ -321,12 +320,15 @@ (define (build-log-port)
|
||||||
(set-port-revealed! port 1)
|
(set-port-revealed! port 1)
|
||||||
port))
|
port))
|
||||||
|
|
||||||
|
(define (node-guile-version node)
|
||||||
|
(inferior-eval '(version) node))
|
||||||
|
|
||||||
(define (node-free-disk-space node)
|
(define (node-free-disk-space node)
|
||||||
"Return the free disk space, in bytes, in NODE's store."
|
"Return the free disk space, in bytes, in NODE's store."
|
||||||
(node-eval node
|
(inferior-eval `(begin
|
||||||
`(begin
|
(use-modules (guix build syscalls))
|
||||||
(use-modules (guix build syscalls))
|
(free-disk-space ,(%store-prefix)))
|
||||||
(free-disk-space ,(%store-prefix)))))
|
node))
|
||||||
|
|
||||||
(define* (transfer-and-offload drv machine
|
(define* (transfer-and-offload drv machine
|
||||||
#:key
|
#:key
|
||||||
|
@ -367,8 +369,12 @@ (define store
|
||||||
(derivation-file-name drv)
|
(derivation-file-name drv)
|
||||||
(build-machine-name machine)
|
(build-machine-name machine)
|
||||||
(nix-protocol-error-message c))
|
(nix-protocol-error-message c))
|
||||||
(let* ((space (false-if-exception
|
(let* ((inferior (false-if-exception (remote-inferior session)))
|
||||||
(node-free-disk-space (make-node session)))))
|
(space (false-if-exception
|
||||||
|
(node-free-disk-space inferior))))
|
||||||
|
|
||||||
|
(when inferior
|
||||||
|
(close-inferior inferior))
|
||||||
|
|
||||||
;; Use exit code 100 for a permanent build failure. The daemon
|
;; Use exit code 100 for a permanent build failure. The daemon
|
||||||
;; interprets other non-zero codes as transient build failures.
|
;; interprets other non-zero codes as transient build failures.
|
||||||
|
@ -417,11 +423,11 @@ (define %minimum-disk-space
|
||||||
|
|
||||||
(define (node-load node)
|
(define (node-load node)
|
||||||
"Return the load on NODE. Return +∞ if NODE is misbehaving."
|
"Return the load on NODE. Return +∞ if NODE is misbehaving."
|
||||||
(let ((line (node-eval node
|
(let ((line (inferior-eval '(begin
|
||||||
'(begin
|
(use-modules (ice-9 rdelim))
|
||||||
(use-modules (ice-9 rdelim))
|
(call-with-input-file "/proc/loadavg"
|
||||||
(call-with-input-file "/proc/loadavg"
|
read-string))
|
||||||
read-string)))))
|
node)))
|
||||||
(if (eof-object? line)
|
(if (eof-object? line)
|
||||||
+inf.0 ;MACHINE does not respond, so assume it is infinitely loaded
|
+inf.0 ;MACHINE does not respond, so assume it is infinitely loaded
|
||||||
(match (string-tokenize line)
|
(match (string-tokenize line)
|
||||||
|
@ -508,9 +514,10 @@ (define (machine-faster? m1 m2)
|
||||||
;; Note: We call 'node-load' only as a last resort because it is
|
;; Note: We call 'node-load' only as a last resort because it is
|
||||||
;; too costly to call it once for every machine.
|
;; too costly to call it once for every machine.
|
||||||
(let* ((session (false-if-exception (open-ssh-session best)))
|
(let* ((session (false-if-exception (open-ssh-session best)))
|
||||||
(node (and session (make-node session)))
|
(node (and session (remote-inferior session)))
|
||||||
(load (and node (normalized-load best (node-load node))))
|
(load (and node (normalized-load best (node-load node))))
|
||||||
(space (and node (node-free-disk-space node))))
|
(space (and node (node-free-disk-space node))))
|
||||||
|
(when node (close-inferior node))
|
||||||
(when session (disconnect! session))
|
(when session (disconnect! session))
|
||||||
(if (and node (< load 2.) (>= space %minimum-disk-space))
|
(if (and node (< load 2.) (>= space %minimum-disk-space))
|
||||||
(match others
|
(match others
|
||||||
|
@ -613,18 +620,17 @@ (define (assert-node-repl node name)
|
||||||
(#f
|
(#f
|
||||||
(report-guile-error name))
|
(report-guile-error name))
|
||||||
((? string? version)
|
((? string? version)
|
||||||
;; Note: The version string already contains the word "Guile".
|
(info (G_ "'~a' is running GNU Guile ~a~%")
|
||||||
(info (G_ "'~a' is running ~a~%")
|
|
||||||
name (node-guile-version node)))))
|
name (node-guile-version node)))))
|
||||||
|
|
||||||
(define (assert-node-has-guix node name)
|
(define (assert-node-has-guix node name)
|
||||||
"Bail out if NODE lacks the (guix) module, or if its daemon is not running."
|
"Bail out if NODE lacks the (guix) module, or if its daemon is not running."
|
||||||
(catch 'node-repl-error
|
(catch 'node-repl-error
|
||||||
(lambda ()
|
(lambda ()
|
||||||
(match (node-eval node
|
(match (inferior-eval '(begin
|
||||||
'(begin
|
(use-modules (guix))
|
||||||
(use-modules (guix))
|
(and add-text-to-store 'alright))
|
||||||
(and add-text-to-store 'alright)))
|
node)
|
||||||
('alright #t)
|
('alright #t)
|
||||||
(_ (report-module-error name))))
|
(_ (report-module-error name))))
|
||||||
(lambda (key . args)
|
(lambda (key . args)
|
||||||
|
@ -632,12 +638,12 @@ (define (assert-node-has-guix node name)
|
||||||
|
|
||||||
(catch 'node-repl-error
|
(catch 'node-repl-error
|
||||||
(lambda ()
|
(lambda ()
|
||||||
(match (node-eval node
|
(match (inferior-eval '(begin
|
||||||
'(begin
|
(use-modules (guix))
|
||||||
(use-modules (guix))
|
(with-store store
|
||||||
(with-store store
|
(add-text-to-store store "test"
|
||||||
(add-text-to-store store "test"
|
"Hello, build machine!")))
|
||||||
"Hello, build machine!"))))
|
node)
|
||||||
((? string? str)
|
((? string? str)
|
||||||
(info (G_ "Guix is usable on '~a' (test returned ~s)~%")
|
(info (G_ "Guix is usable on '~a' (test returned ~s)~%")
|
||||||
name str))
|
name str))
|
||||||
|
@ -656,25 +662,23 @@ (define* (nonce #:optional (name (gethostname)))
|
||||||
(string-append name "-"
|
(string-append name "-"
|
||||||
(number->string (random 1000000 (force %random-state)))))
|
(number->string (random 1000000 (force %random-state)))))
|
||||||
|
|
||||||
(define (assert-node-can-import node name daemon-socket)
|
(define (assert-node-can-import session node name daemon-socket)
|
||||||
"Bail out if NODE refuses to import our archives."
|
"Bail out if NODE refuses to import our archives."
|
||||||
(let ((session (node-session node)))
|
(with-store store
|
||||||
(with-store store
|
(let* ((item (add-text-to-store store "export-test" (nonce)))
|
||||||
(let* ((item (add-text-to-store store "export-test" (nonce)))
|
(remote (connect-to-remote-daemon session daemon-socket)))
|
||||||
(remote (connect-to-remote-daemon session daemon-socket)))
|
(with-store local
|
||||||
(with-store local
|
(send-files local (list item) remote))
|
||||||
(send-files local (list item) remote))
|
|
||||||
|
|
||||||
(if (valid-path? remote item)
|
(if (valid-path? remote item)
|
||||||
(info (G_ "'~a' successfully imported '~a'~%")
|
(info (G_ "'~a' successfully imported '~a'~%")
|
||||||
name item)
|
name item)
|
||||||
(leave (G_ "'~a' was not properly imported on '~a'~%")
|
(leave (G_ "'~a' was not properly imported on '~a'~%")
|
||||||
item name))))))
|
item name)))))
|
||||||
|
|
||||||
(define (assert-node-can-export node name daemon-socket)
|
(define (assert-node-can-export session node name daemon-socket)
|
||||||
"Bail out if we cannot import signed archives from NODE."
|
"Bail out if we cannot import signed archives from NODE."
|
||||||
(let* ((session (node-session node))
|
(let* ((remote (connect-to-remote-daemon session daemon-socket))
|
||||||
(remote (connect-to-remote-daemon session daemon-socket))
|
|
||||||
(item (add-text-to-store remote "import-test" (nonce name))))
|
(item (add-text-to-store remote "import-test" (nonce name))))
|
||||||
(with-store store
|
(with-store store
|
||||||
(if (and (retrieve-files store (list item) remote)
|
(if (and (retrieve-files store (list item) remote)
|
||||||
|
@ -701,11 +705,13 @@ (define (build-machine=? m1 m2)
|
||||||
(let* ((names (map build-machine-name machines))
|
(let* ((names (map build-machine-name machines))
|
||||||
(sockets (map build-machine-daemon-socket machines))
|
(sockets (map build-machine-daemon-socket machines))
|
||||||
(sessions (map open-ssh-session machines))
|
(sessions (map open-ssh-session machines))
|
||||||
(nodes (map make-node sessions)))
|
(nodes (map remote-inferior sessions)))
|
||||||
(for-each assert-node-repl nodes names)
|
(for-each assert-node-repl nodes names)
|
||||||
(for-each assert-node-has-guix nodes names)
|
(for-each assert-node-has-guix nodes names)
|
||||||
(for-each assert-node-can-import nodes names sockets)
|
(for-each assert-node-can-import sessions nodes names sockets)
|
||||||
(for-each assert-node-can-export nodes names sockets))))
|
(for-each assert-node-can-export sessions nodes names sockets)
|
||||||
|
(for-each close-inferior nodes)
|
||||||
|
(for-each disconnect! sessions))))
|
||||||
|
|
||||||
(define (check-machine-status machine-file pred)
|
(define (check-machine-status machine-file pred)
|
||||||
"Print the load of each machine matching PRED in MACHINE-FILE."
|
"Print the load of each machine matching PRED in MACHINE-FILE."
|
||||||
|
@ -722,10 +728,11 @@ (define (build-machine=? m1 m2)
|
||||||
(length machines) machine-file)
|
(length machines) machine-file)
|
||||||
(for-each (lambda (machine)
|
(for-each (lambda (machine)
|
||||||
(let* ((session (open-ssh-session machine))
|
(let* ((session (open-ssh-session machine))
|
||||||
(node (make-node session))
|
(inferior (remote-inferior session))
|
||||||
(uts (node-eval node '(uname)))
|
(uts (inferior-eval '(uname) inferior))
|
||||||
(load (node-load node))
|
(load (node-load inferior))
|
||||||
(free (node-free-disk-space node)))
|
(free (node-free-disk-space inferior)))
|
||||||
|
(close-inferior inferior)
|
||||||
(disconnect! session)
|
(disconnect! session)
|
||||||
(format #t "~a~% kernel: ~a ~a~% architecture: ~a~%\
|
(format #t "~a~% kernel: ~a ~a~% architecture: ~a~%\
|
||||||
host name: ~a~% normalized load: ~a~% free disk space: ~,2f MiB~%"
|
host name: ~a~% normalized load: ~a~% free disk space: ~,2f MiB~%"
|
||||||
|
|
32
guix/ssh.scm
32
guix/ssh.scm
|
@ -27,8 +27,6 @@ (define-module (guix ssh)
|
||||||
#:use-module (ssh channel)
|
#:use-module (ssh channel)
|
||||||
#:use-module (ssh popen)
|
#:use-module (ssh popen)
|
||||||
#:use-module (ssh session)
|
#:use-module (ssh session)
|
||||||
#:use-module (ssh dist)
|
|
||||||
#:use-module (ssh dist node)
|
|
||||||
#:use-module (srfi srfi-1)
|
#:use-module (srfi srfi-1)
|
||||||
#:use-module (srfi srfi-11)
|
#:use-module (srfi srfi-11)
|
||||||
#:use-module (srfi srfi-26)
|
#:use-module (srfi srfi-26)
|
||||||
|
@ -102,6 +100,20 @@ (define (remote-inferior session)
|
||||||
"guix" "repl" "-t" "machine")))
|
"guix" "repl" "-t" "machine")))
|
||||||
(port->inferior pipe)))
|
(port->inferior pipe)))
|
||||||
|
|
||||||
|
(define (inferior-remote-eval exp session)
|
||||||
|
"Evaluate EXP in a new inferior running in SESSION, and close the inferior
|
||||||
|
right away."
|
||||||
|
(let ((inferior (remote-inferior session)))
|
||||||
|
(dynamic-wind
|
||||||
|
(const #t)
|
||||||
|
(lambda ()
|
||||||
|
(inferior-eval exp inferior))
|
||||||
|
(lambda ()
|
||||||
|
;; Close INFERIOR right away to prevent finalization from happening in
|
||||||
|
;; another thread at the wrong time (see
|
||||||
|
;; <https://bugs.gnu.org/26976>.)
|
||||||
|
(close-inferior inferior)))))
|
||||||
|
|
||||||
(define* (remote-daemon-channel session
|
(define* (remote-daemon-channel session
|
||||||
#:optional
|
#:optional
|
||||||
(socket-name
|
(socket-name
|
||||||
|
@ -277,15 +289,15 @@ (define* (send-files local files remote
|
||||||
;; Compute the subset of FILES missing on SESSION and send them.
|
;; Compute the subset of FILES missing on SESSION and send them.
|
||||||
(let* ((files (if recursive? (requisites local files) files))
|
(let* ((files (if recursive? (requisites local files) files))
|
||||||
(session (channel-get-session (nix-server-socket remote)))
|
(session (channel-get-session (nix-server-socket remote)))
|
||||||
(node (make-node session))
|
(missing (inferior-remote-eval
|
||||||
(missing (node-eval node
|
`(begin
|
||||||
`(begin
|
(use-modules (guix)
|
||||||
(use-modules (guix)
|
(srfi srfi-1) (srfi srfi-26))
|
||||||
(srfi srfi-1) (srfi srfi-26))
|
|
||||||
|
|
||||||
(with-store store
|
(with-store store
|
||||||
(remove (cut valid-path? store <>)
|
(remove (cut valid-path? store <>)
|
||||||
',files)))))
|
',files)))
|
||||||
|
session))
|
||||||
(count (length missing))
|
(count (length missing))
|
||||||
(sizes (map (lambda (item)
|
(sizes (map (lambda (item)
|
||||||
(path-info-nar-size (query-path-info local item)))
|
(path-info-nar-size (query-path-info local item)))
|
||||||
|
|
Loading…
Reference in a new issue