Skip to content
Snippets Groups Projects
Commit d9f0a237 authored by Ludovic Courtès's avatar Ludovic Courtès
Browse files

gnu: vm: Rewrite helper functions as monadic functions.

* gnu/system/dmd.scm (host-name-service, nscd-service, mingetty-service,
  syslog-service, guix-service, static-networking-service): Rewrite as
  monadic functions.
  (dmd-configuration-file): Use 'text-file' instead of
  'add-text-to-store'.
* gnu/system/grub.scm (grub-configuration-file): Rewrite as a monadic
  function.
* gnu/system/linux.scm (pam-services->directory): Likewise.
* gnu/system/shadow.scm (group-file, passwd-file, guix-build-accounts):
  Likewise.
* gnu/system/vm.scm (expression->derivation-in-linux-vm, qemu-image,
  union, system-qemu-image): Likewise.
parent b860f382
No related branches found
No related tags found
No related merge requests found
...@@ -31,6 +31,7 @@ (define-module (gnu system dmd) ...@@ -31,6 +31,7 @@ (define-module (gnu system dmd)
#:select (net-tools)) #:select (net-tools))
#:use-module (ice-9 match) #:use-module (ice-9 match)
#:use-module (srfi srfi-1) #:use-module (srfi srfi-1)
#:use-module (guix monads)
#:export (service? #:export (service?
service service
service-provision service-provision
...@@ -69,53 +70,51 @@ (define-record-type* <service> ...@@ -69,53 +70,51 @@ (define-record-type* <service>
(inputs service-inputs ; list of inputs (inputs service-inputs ; list of inputs
(default '()))) (default '())))
(define (host-name-service store name) (define (host-name-service name)
"Return a service that sets the host name to NAME." "Return a service that sets the host name to NAME."
(service (with-monad %store-monad
(provision '(host-name)) (return (service
(start `(lambda _ (provision '(host-name))
(sethostname ,name))) (start `(lambda _
(respawn? #f))) (sethostname ,name)))
(respawn? #f)))))
(define (mingetty-service store tty)
(define (mingetty-service tty)
"Return a service to run mingetty on TTY." "Return a service to run mingetty on TTY."
(let* ((mingetty-drv (package-derivation store mingetty)) (mlet %store-monad ((mingetty-bin (package-file mingetty "sbin/mingetty")))
(mingetty-bin (string-append (derivation->output-path mingetty-drv) (return
"/sbin/mingetty"))) (service
(service (provision (list (symbol-append 'term- (string->symbol tty))))
(provision (list (symbol-append 'term- (string->symbol tty))))
;; Since the login prompt shows the host name, wait for the 'host-name' ;; Since the login prompt shows the host name, wait for the 'host-name'
;; service to be done. ;; service to be done.
(requirement '(host-name)) (requirement '(host-name))
(start `(make-forkexec-constructor ,mingetty-bin "--noclear" ,tty)) (start `(make-forkexec-constructor ,mingetty-bin "--noclear" ,tty))
(inputs `(("mingetty" ,mingetty)))))) (inputs `(("mingetty" ,mingetty)))))))
(define* (nscd-service store (define* (nscd-service #:key (glibc glibc-final))
#:key (glibc glibc-final))
"Return a service that runs libc's name service cache daemon (nscd)." "Return a service that runs libc's name service cache daemon (nscd)."
(let ((nscd (string-append (package-output store glibc) "/sbin/nscd"))) (mlet %store-monad ((nscd (package-file glibc "sbin/nscd")))
(service (return (service
(provision '(nscd)) (provision '(nscd))
(start `(make-forkexec-constructor ,nscd "-f" "/dev/null")) (start `(make-forkexec-constructor ,nscd "-f" "/dev/null"))
;; XXX: Local copy of 'make-kill-destructor' because the one upstream ;; XXX: Local copy of 'make-kill-destructor' because the one upstream
;; uses the broken 'opt-lambda' macro. ;; uses the broken 'opt-lambda' macro.
(stop `(lambda* (#:optional (signal SIGTERM)) (stop `(lambda* (#:optional (signal SIGTERM))
(lambda (pid . args) (lambda (pid . args)
(kill pid signal) (kill pid signal)
#f))) #f)))
(respawn? #f) (respawn? #f)
(inputs `(("glibc" ,glibc)))))) (inputs `(("glibc" ,glibc)))))))
(define (syslog-service store) (define (syslog-service)
"Return a service that runs 'syslogd' with reasonable default settings." "Return a service that runs 'syslogd' with reasonable default settings."
(define syslog.conf ;; Snippet adapted from the GNU inetutils manual.
;; Snippet adapted from the GNU inetutils manual. (define contents "
(add-text-to-store store "syslog.conf" "
# Log all kernel messages, authentication messages of # Log all kernel messages, authentication messages of
# level notice or higher and anything of level err or # level notice or higher and anything of level err or
# higher to the console. # higher to the console.
...@@ -134,31 +133,30 @@ (define syslog.conf ...@@ -134,31 +133,30 @@ (define syslog.conf
# Log all the mail messages in one place. # Log all the mail messages in one place.
mail.* /var/log/maillog mail.* /var/log/maillog
")) ")
(let* ((inetutils-drv (package-derivation store inetutils)) (mlet %store-monad
(syslogd (string-append (derivation->output-path inetutils-drv) ((syslog.conf (text-file "syslog.conf" contents))
"/libexec/syslogd"))) (syslogd (package-file inetutils "libexec/syslogd")))
(service (return
(provision '(syslogd)) (service
(start `(make-forkexec-constructor ,syslogd (provision '(syslogd))
"--rcfile" ,syslog.conf)) (start `(make-forkexec-constructor ,syslogd
(inputs `(("inetutils" ,inetutils) "--rcfile" ,syslog.conf))
("syslog.conf" ,syslog.conf)))))) (inputs `(("inetutils" ,inetutils)
("syslog.conf" ,syslog.conf)))))))
(define* (guix-service store #:key (guix guix) (builder-group "guixbuild"))
(define* (guix-service #:key (guix guix) (builder-group "guixbuild"))
"Return a service that runs the build daemon from GUIX." "Return a service that runs the build daemon from GUIX."
(let* ((drv (package-derivation store guix)) (mlet %store-monad ((daemon (package-file guix "bin/guix-daemon")))
(daemon (string-append (derivation->output-path drv) (return (service
"/bin/guix-daemon"))) (provision '(guix-daemon))
(service (start `(make-forkexec-constructor ,daemon
(provision '(guix-daemon)) "--build-users-group"
(start `(make-forkexec-constructor ,daemon ,builder-group))
"--build-users-group" (inputs `(("guix" ,guix)))))))
,builder-group))
(inputs `(("guix" ,guix)))))) (define* (static-networking-service interface ip
(define* (static-networking-service store interface ip
#:key #:key
gateway gateway
(inetutils inetutils) (inetutils inetutils)
...@@ -169,31 +167,30 @@ (define* (static-networking-service store interface ip ...@@ -169,31 +167,30 @@ (define* (static-networking-service store interface ip
;; TODO: Eventually we should do this using Guile's networking procedures, ;; TODO: Eventually we should do this using Guile's networking procedures,
;; like 'configure-qemu-networking' does, but the patch that does this is ;; like 'configure-qemu-networking' does, but the patch that does this is
;; not yet in stock Guile. ;; not yet in stock Guile.
(let ((ifconfig (string-append (package-output store inetutils) (mlet %store-monad ((ifconfig (package-file inetutils "bin/ifconfig"))
"/bin/ifconfig")) (route (package-file net-tools "sbin/route")))
(route (string-append (package-output store net-tools) (return
"/sbin/route"))) (service
(service (provision '(networking))
(provision '(networking)) (start `(lambda _
(start `(lambda _ (and (zero? (system* ,ifconfig ,interface ,ip "up"))
(and (zero? (system* ,ifconfig ,interface ,ip "up")) ,(if gateway
,(if gateway `(begin
`(begin (sleep 3) ; XXX
(sleep 3) ; XXX (zero? (system* ,route "add" "-net" "default"
(zero? (system* ,route "add" "-net" "default" "gw" ,gateway)))
"gw" ,gateway))) #t))))
#t)))) (stop `(lambda _
(stop `(lambda _ (system* ,ifconfig ,interface "down")
(system* ,ifconfig ,interface "down") (system* ,route "del" "-net" "default")))
(system* ,route "del" "-net" "default"))) (respawn? #f)
(respawn? #f) (inputs `(("inetutils" ,inetutils)
(inputs `(("inetutils" ,inetutils) ,@(if gateway
,@(if gateway `(("net-tools" ,net-tools))
`(("net-tools" ,net-tools)) '())))))))
'()))))))
(define (dmd-configuration-file store services) (define (dmd-configuration-file services)
"Return the dmd configuration file for SERVICES." "Return the dmd configuration file for SERVICES."
(define config (define config
`(begin `(begin
...@@ -209,7 +206,6 @@ (define config ...@@ -209,7 +206,6 @@ (define config
services)) services))
(for-each start ',(append-map service-provision services)))) (for-each start ',(append-map service-provision services))))
(add-text-to-store store "dmd.conf" (text-file "dmd.conf" (object->string config)))
(object->string config)))
;;; dmd.scm ends here ;;; dmd.scm ends here
...@@ -21,6 +21,7 @@ (define-module (gnu system grub) ...@@ -21,6 +21,7 @@ (define-module (gnu system grub)
#:use-module (guix packages) #:use-module (guix packages)
#:use-module (guix derivations) #:use-module (guix derivations)
#:use-module (guix records) #:use-module (guix records)
#:use-module (guix monads)
#:use-module (ice-9 match) #:use-module (ice-9 match)
#:use-module (srfi srfi-1) #:use-module (srfi srfi-1)
#:export (menu-entry #:export (menu-entry
...@@ -42,43 +43,45 @@ (define-record-type* <menu-entry> ...@@ -42,43 +43,45 @@ (define-record-type* <menu-entry>
(default '())) (default '()))
(initrd menu-entry-initrd)) (initrd menu-entry-initrd))
(define* (grub-configuration-file store entries (define* (grub-configuration-file entries
#:key (default-entry 1) (timeout 5) #:key (default-entry 1) (timeout 5)
(system (%current-system))) (system (%current-system)))
"Return the GRUB configuration file in STORE for ENTRIES, a list of "Return the GRUB configuration file for ENTRIES, a list of
<menu-entry> objects, defaulting to DEFAULT-ENTRY and with the given TIMEOUT." <menu-entry> objects, defaulting to DEFAULT-ENTRY and with the given TIMEOUT."
(define prologue (define (prologue kernel)
(format #f " (format #f "
set default=~a set default=~a
set timeout=~a set timeout=~a
search.file ~a~%" search.file ~a~%"
default-entry timeout default-entry timeout kernel))
(any (match-lambda
(($ <menu-entry> _ linux) (define (bzImage)
(let* ((drv (package-derivation store linux system)) (anym %store-monad
(out (derivation->output-path drv))) (match-lambda
(string-append out "/bzImage")))) (($ <menu-entry> _ linux)
entries))) (package-file linux "bzImage"
#:system system)))
entries))
(define entry->text (define entry->text
(match-lambda (match-lambda
(($ <menu-entry> label linux arguments initrd) (($ <menu-entry> label linux arguments initrd)
(let ((linux-drv (package-derivation store linux system)) (mlet %store-monad ((linux (package-file linux "bzImage"
(initrd-drv (package-derivation store initrd system))) #:system system))
(initrd (package-file initrd "initrd"
#:system system)))
;; XXX: Assume that INITRD is a directory containing an 'initrd' file. ;; XXX: Assume that INITRD is a directory containing an 'initrd' file.
(format #f "menuentry ~s { (return (format #f "menuentry ~s {
linux ~a/bzImage ~a linux ~a ~a
initrd ~a/initrd initrd ~a
}~%" }~%"
label label
(derivation->output-path linux-drv) linux (string-join arguments) initrd))))))
(string-join arguments)
(derivation->output-path initrd-drv))))))
(add-text-to-store store "grub.cfg" (mlet %store-monad ((kernel (bzImage))
(string-append prologue (body (mapm %store-monad entry->text entries)))
(string-concatenate (text-file "grub.cfg"
(map entry->text entries))) (string-append (prologue kernel)
'())) (string-concatenate body)))))
;;; grub.scm ends here ;;; grub.scm ends here
...@@ -20,6 +20,7 @@ (define-module (gnu system linux) ...@@ -20,6 +20,7 @@ (define-module (gnu system linux)
#:use-module (guix store) #:use-module (guix store)
#:use-module (guix records) #:use-module (guix records)
#:use-module (guix derivations) #:use-module (guix derivations)
#:use-module (guix monads)
#:use-module (ice-9 match) #:use-module (ice-9 match)
#:use-module (srfi srfi-1) #:use-module (srfi srfi-1)
#:use-module (srfi srfi-26) #:use-module (srfi srfi-26)
...@@ -81,17 +82,20 @@ (define (entry->string type entry) ...@@ -81,17 +82,20 @@ (define (entry->string type entry)
(map (cut entry->string "password" <>) password) (map (cut entry->string "password" <>) password)
(map (cut entry->string "session" <>) session)))))) (map (cut entry->string "session" <>) session))))))
(define (pam-services->directory store services) (define (pam-services->directory services)
"Return the derivation to build the configuration directory to be used as "Return the derivation to build the configuration directory to be used as
/etc/pam.d for SERVICES." /etc/pam.d for SERVICES."
(let ((names (map pam-service-name services)) (mlet %store-monad
(files (map (match-lambda ((names -> (map pam-service-name services))
(files (mapm %store-monad
(match-lambda
((and service ($ <pam-service> name)) ((and service ($ <pam-service> name))
(let ((config (pam-service->configuration service))) (let ((config (pam-service->configuration service)))
(add-text-to-store store (text-file (string-append name ".pam") config))))
(string-append name ".pam")
config '())))) ;; XXX: Eventually, SERVICES may be a list of monadic
services))) ;; values instead of plain values.
(map return services))))
(define builder (define builder
'(begin '(begin
(use-modules (ice-9 match)) (use-modules (ice-9 match))
...@@ -104,9 +108,7 @@ (define builder ...@@ -104,9 +108,7 @@ (define builder
%build-inputs) %build-inputs)
#t))) #t)))
(build-expression->derivation store "pam.d" (%current-system) (derivation-expression "pam.d" (%current-system) builder (zip names files))))
builder
(zip names files))))
(define %pam-other-services (define %pam-other-services
;; The "other" PAM configuration, which denies everything (see ;; The "other" PAM configuration, which denies everything (see
......
...@@ -20,6 +20,7 @@ (define-module (gnu system shadow) ...@@ -20,6 +20,7 @@ (define-module (gnu system shadow)
#:use-module (guix store) #:use-module (guix store)
#:use-module (guix records) #:use-module (guix records)
#:use-module (guix packages) #:use-module (guix packages)
#:use-module (guix monads)
#:use-module ((gnu packages system) #:use-module ((gnu packages system)
#:select (shadow)) #:select (shadow))
#:use-module (srfi srfi-1) #:use-module (srfi srfi-1)
...@@ -72,7 +73,7 @@ (define-record-type* <user-group> ...@@ -72,7 +73,7 @@ (define-record-type* <user-group>
(id user-group-id) (id user-group-id)
(members user-group-members (default '()))) (members user-group-members (default '())))
(define (group-file store groups) (define (group-file groups)
"Return a /etc/group file for GROUPS, a list of <user-group> objects." "Return a /etc/group file for GROUPS, a list of <user-group> objects."
(define contents (define contents
(let loop ((groups groups) (let loop ((groups groups)
...@@ -87,9 +88,9 @@ (define contents ...@@ -87,9 +88,9 @@ (define contents
(() (()
(string-join (reverse result) "\n" 'suffix))))) (string-join (reverse result) "\n" 'suffix)))))
(add-text-to-store store "group" contents)) (text-file "group" contents))
(define* (passwd-file store accounts #:key shadow?) (define* (passwd-file accounts #:key shadow?)
"Return a password file for ACCOUNTS, a list of <user-account> objects. If "Return a password file for ACCOUNTS, a list of <user-account> objects. If
SHADOW? is true, then it is a /etc/shadow file, otherwise it is a /etc/passwd SHADOW? is true, then it is a /etc/shadow file, otherwise it is a /etc/passwd
file." file."
...@@ -114,28 +115,27 @@ (define contents ...@@ -114,28 +115,27 @@ (define contents
(() (()
(string-join (reverse result) "\n" 'suffix))))) (string-join (reverse result) "\n" 'suffix)))))
(add-text-to-store store (if shadow? "shadow" "passwd") (text-file (if shadow? "shadow" "passwd") contents))
contents '()))
(define* (guix-build-accounts store count #:key (define* (guix-build-accounts count #:key
(first-uid 30001) (first-uid 30001)
(gid 30000) (gid 30000)
(shadow shadow)) (shadow shadow))
"Return a list of COUNT user accounts for Guix build users, with UIDs "Return a list of COUNT user accounts for Guix build users, with UIDs
starting at FIRST-UID, and under GID." starting at FIRST-UID, and under GID."
(let* ((gid* gid) (mlet* %store-monad ((gid* -> gid)
(no-login (string-append (package-output store shadow) "/sbin/nologin"))) (no-login (package-file shadow "sbin/nologin")))
(unfold (cut > <> count) (return (unfold (cut > <> count)
(lambda (n) (lambda (n)
(user-account (user-account
(name (format #f "guixbuilder~2,'0d" n)) (name (format #f "guixbuilder~2,'0d" n))
(password "!") (password "!")
(uid (+ first-uid n -1)) (uid (+ first-uid n -1))
(gid gid*) (gid gid*)
(comment (format #f "Guix Build User ~2d" n)) (comment (format #f "Guix Build User ~2d" n))
(home-directory "/var/empty") (home-directory "/var/empty")
(shell no-login))) (shell no-login)))
1+ 1+
1))) 1))))
;;; shadow.scm ends here ;;; shadow.scm ends here
This diff is collapsed.
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment