;;; GNU Guix --- Functional package management for GNU ;;; Copyright © 2017 nee ;;; Copyright © 2021, 2022 Maxim Cournoyer ;;; ;;; This file is part of GNU Guix. ;;; ;;; GNU Guix is free software; you can redistribute it and/or modify it ;;; under the terms of the GNU General Public License as published by ;;; the Free Software Foundation; either version 3 of the License, or (at ;;; your option) any later version. ;;; ;;; GNU Guix is distributed in the hope that it will be useful, but ;;; WITHOUT ANY WARRANTY; without even the implied warranty of ;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the ;;; GNU General Public License for more details. ;;; ;;; You should have received a copy of the GNU General Public License ;;; along with GNU Guix. If not, see . (define-module (gnu services telephony) #:use-module ((gnu build jami-service) #:select (account-fingerprint?)) #:use-module ((gnu services) #:hide (delete)) #:use-modul
aboutsummaryrefslogtreecommitdiff
blob: f5c8407b8949e50607bf4ba2b81290c53753d291 (about) (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
;;; GNU Guix --- Functional package management for GNU
;;; Copyright © 2013, 2014, 2015, 2016, 2017, 2018, 2019, 2020 Ludovic Courtès <ludo@gnu.org>
;;; Copyright © 2016 Chris Marusich <cmmarusich@gmail.com>
;;;
;;; This file is part of GNU Guix.
;;;
;;; GNU Guix is free software; you can redistribute it and/or modify it
;;; under the terms of the GNU General Public License as published by
;;; the Free Software Foundation; either version 3 of the License, or (at
;;; your option) any later version.
;;;
;;; GNU Guix is distributed in the hope that it will be useful, but
;;; WITHOUT ANY WARRANTY; without even the implied warranty of
;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
;;; GNU General Public License for more details.
;;;
;;; You should have received a copy of the GNU General Public License
;;; along with GNU Guix.  If not, see <http://www.gnu.org/licenses/>.

(define-module (gnu build install)
  #:use-module (guix build syscalls)
  #:use-module (guix build utils)
  #:use-module (guix build store-copy)
  #:use-module (srfi srfi-26)
  #:use-module (ice-9 match)
  #:export (install-boot-config
            evaluate-populate-directive
            populate-root-file-system
            install-database-and-gc-roots
            populate-single-profile-directory
            mount-cow-store
            unmount-cow-store))

;;; Commentary:
;;;
;;; This module supports the installation of the GNU system on a hard disk.
;;; It is meant to be used both in a build environment (in derivations that
;;; build VM images), and on the bare metal (when really installing the
;;; system.)
;;;
;;; Code:

(define (install-boot-config bootcfg bootcfg-location mount-point)
  "Atomically copy BOOTCFG into BOOTCFG-LOCATION on the MOUNT-POINT.  Note
that the caller must make sure that BOOTCFG is registered as a GC root so
that the fonts, background images, etc. referred to by BOOTCFG are not GC'd."
  (let* ((target (string-append mount-point bootcfg-location))
         (pivot  (string-append target ".new")))
    (mkdir-p (dirname target))

    ;; Copy BOOTCFG instead of just symlinking it, because symlinks won't
    ;; work when /boot is on a separate partition.  Do that atomically.
    (copy-file bootcfg pivot)
    (rename-file pivot target)))

(define* (evaluate-populate-directive directive target
                                      #:key
                                      (default-gid 0)
                                      (default-uid 0))
  "Evaluate DIRECTIVE, an sexp describing a file or directory to create under
directory TARGET.  DEFAULT-UID and DEFAULT-GID are the default UID and GID in
the context of the caller.  If the directive matches those defaults then,
'chown' won't be run."
  (let loop ((directive directive))
    (catch 'system-error
      (lambda ()
        (match directive
          (('directory name)
           (mkdir-p (string-append target name)))
          (('directory name uid gid)
           (let ((dir (string-append target name)))
             (mkdir-p dir)
             ;; If called from a context without "root" permissions, "chown"
             ;; to root will fail.  In that case, do not try to run "chown"
             ;; and assume that the file will be chowned elsewhere (when
             ;; interned in the store for instance).
             (or (and (= uid default-uid) (= gid default-gid))
                 (chown dir uid gid))))
          (('directory name uid gid mode)
           (loop `(directory ,name ,uid ,gid))
           (chmod (string-append target name) mode))
          (('file name)
           (call-with-output-file (string-append target name)
             (const #t)))
          (('file name (? string? content))
           (call-with-output-file (string-append target name)
             (lambda (port)
               (display content port))))
          ((new '-> old)
           (let try ()
             (catch 'system-error
               (lambda ()
                 (symlink old (string-append target new)))
               (lambda args
                 ;; When doing 'guix system init' on the current '/', some
                 ;; symlinks may already exists.  Override them.
                 (if (= EEXIST (system-error-errno args))
                     (begin
                       (delete-file (string-append target new))
                       (try))
                     (apply throw args))))))))
      (lambda args
        ;; Usually we can only get here when installing to an existing root,
        ;; as with 'guix system init foo.scm /'.
        (format (current-error-port)
                "error: failed to evaluate directive: ~s~%"
                directive)
        (apply throw args)))))

(define (directives store)
  "Return a list of directives to populate the root file system that will host
STORE."
  `((directory ,store 0 0 #o1775)

    (directory "/etc")
    (directory "/var/log")                          ; for shepherd
    (directory "/var/guix/gcroots")
    (directory "/var/empty")                        ; for no-login accounts
    (directory "/var/db")                           ; for dhclient, etc.
    (directory "/var/run")
    (directory "/run")
    (directory "/mnt")
    (directory "/var/guix/profiles/per-user/root" 0 0)

    ;; Link to the initial system generation.
    ("/var/guix/profiles/system" -> "system-1-link")

    ("/var/guix/gcroots/booted-system" -> "/run/booted-system")
    ("/var/guix/gcroots/current-system" -> "/run/current-system")
    ("/var/guix/gcroots/profiles" -> "/var/guix/profiles")

    (directory "/bin")
    (directory "/tmp" 0 0 #o1777)                 ; sticky bit
    (directory "/var/tmp" 0 0 #o1777)
    (directory "/var/lock" 0 0 #o1777)

    (directory "/home" 0 0)))

(define* (populate-root-file-system system target
                                    #:key (extras '()))
  "Make the essential non-store files and directories on TARGET.  This
includes /etc, /var, /run, /bin/sh, etc., and all the symlinks to SYSTEM.
EXTRAS is a list of directives appended to the built-in directives to populate
TARGET."
  (for-each (cut evaluate-populate-directive <> target)
            (append (directives (%store-directory)) extras))

  ;; Add system generation 1.
  (let ((generation-1 (string-append target
                                     "/var/guix/profiles/system-1-link")))
    (let try ()
      (catch 'system-error
        (lambda ()
          (symlink system generation-1))
        (lambda args
          ;; If GENERATION-1 already exists, overwrite it.
          (if (= EEXIST (system-error-errno args))
              (begin
                (delete-file generation-1)
                (try))
              (apply throw args)))))))

(define %root-profile
  "/var/guix/profiles/per-user/root")

(define* (install-database-and-gc-roots root database profile
                                        #:key (profile-name "guix-profile"))
  "Install DATABASE, the store database, under directory ROOT.  Create
PROFILE-NAME and have it link to PROFILE, a store item."
  (define (scope file)
    (string-append root "/" file))

  (define (mkdir-p* dir)
    (mkdir-p (scope dir)))

  (define (symlink* old new)
    (symlink old (scope new)))

  (install-file database (scope "/var/guix/db/"))
  (chmod (scope "/var/guix/db/db.sqlite") #o644)
  (mkdir-p* "/var/guix/profiles")
  (mkdir-p* "/var/guix/gcroots")
  (symlink* "/var/guix/profiles" "/var/guix/gcroots/profiles")

  ;; Make root's profile, which makes it a GC root.
  (mkdir-p* %root-profile)
  (symlink* profile
            (string-append %root-profile "/" profile-name "-1-link"))
  (symlink* (string-append profile-name "-1-link")
            (string-append %root-profile "/" profile-name)))

(define* (populate-single-profile-directory directory
                                            #:key profile closure
                                            (profile-name "guix-profile")
                                            database)
  "Populate DIRECTORY with a store containing PROFILE, whose closure is given
in the file called CLOSURE (as generated by #:references-graphs.)  DIRECTORY
is initialized to contain a single profile under /root pointing to PROFILE.

When DATABASE is true, copy it to DIRECTORY/var/guix/db and create
DIRECTORY/var/guix/gcroots and friends.

PROFILE-NAME is the name of the profile being created under
/var/guix/profiles, typically either \"guix-profile\" or \"current-guix\".

This is used to create the self-contained tarballs with 'guix pack'."
  (define (scope file)
    (string-append directory "/" file))

  (define (mkdir-p* dir)
    (mkdir-p (scope dir)))

  (define (symlink* old new)
    (symlink old (scope new)))

  ;; Populate the store.
  (populate-store (list closure) directory
                  #:deduplicate? #f)

  (when database
    (install-database-and-gc-roots directory database profile
                                   #:profile-name profile-name))

  (match profile-name
    ("guix-profile"
     (mkdir-p* "/root")
     (symlink* (string-append %root-profile "/guix-profile")
               "/root/.guix-profile"))
    ("current-guix"
     (mkdir-p* "/root/.config/guix")
     (symlink* (string-append %root-profile "/current-guix")
               "/root/.config/guix/current"))
    (_
     #t)))

(define (mount-cow-store target backing-directory)
  "Make the store copy-on-write, using TARGET as the backing store.  This is
useful when TARGET is on a hard disk, whereas the current store is on a RAM
disk."
  (define (set-store-permissions directory)
    "Set the right perms on DIRECTORY to use it as the store."
    (chown directory 0 30000)      ;use the fixed 'guixbuild' GID
    (chmod directory #o1775))

  (let ((tmpdir (string-append target "/tmp")))
    (mkdir-p tmpdir)
    (mount tmpdir "/tmp" "none" MS_BIND))

  (let* ((rw-dir (string-append target backing-directory))
         (work-dir (string-append rw-dir "/../.overlayfs-workdir")))
    (mkdir-p rw-dir)
    (mkdir-p work-dir)
    (mkdir-p "/.rw-store")
    (set-store-permissions rw-dir)
    (set-store-permissions "/.rw-store")

    ;; Mount the overlay, then atomically make it the store.
    (mount "none" "/.rw-store" "overlay" 0
           (string-append "lowerdir=" (%store-directory) ","
                          "upperdir=" rw-dir ","
                          "workdir=" work-dir))
    (mount "/.rw-store" (%store-directory) "" MS_MOVE)
    (rmdir "/.rw-store")))

(define (unmount-cow-store target backing-directory)
  "Unmount copy-on-write store."
  (let ((tmp-dir "/remove"))
    (mkdir-p tmp-dir)
    (mount (%store-directory) tmp-dir "" MS_MOVE)
    (umount tmp-dir)
    (rmdir tmp-dir)
    (delete-file-recursively
     (string-append target backing-directory))))

;;; install.scm ends here
list-moderators-action add-moderator-action ban-contact-action list-banned-contacts-action enable-account-action disable-account-action)) (requirement '(jami-dbus-session)) (modules `((ice-9 format) (ice-9 ftw) (ice-9 match) (ice-9 receive) (srfi srfi-1) (srfi srfi-26) (gnu build dbus-service) (gnu build jami-service) (gnu build shepherd) (gnu system file-systems) ,@%default-modules)) (start #~(lambda args (define (delete-file-recursively/safe file) ;; Ensure we're not deleting things outside of ;; /var/lib/jami. This prevents a possible attack in case ;; the daemon is compromised and an attacker gains write ;; access to /var/lib/jami. (let ((parent-directory (dirname file))) (if (eq? 'symlink (stat:type (stat parent-directory))) (error "abnormality detected; unexpected symlink found at" parent-directory) (delete-file-recursively file)))) (when #$declarative-mode? ;; Clear the Jami configuration and accounts, to enforce the ;; declared state. (catch #t (lambda () (for-each (cut delete-file-recursively/safe <>) '("/var/lib/jami/.cache/jami" "/var/lib/jami/.config/jami" "/var/lib/jami/.local/share/jami" "/var/lib/jami/accounts"))) (lambda args #t)) ;; Copy the Jami account archives from somewhere readable ;; by root to a place only the jami user can read. (let* ((accounts-dir "/var/lib/jami/accounts/") (pwd (getpwnam "jami")) (user (passwd:uid pwd)) (group (passwd:gid pwd))) (mkdir-p accounts-dir) (chown accounts-dir user group) (for-each (lambda (f) (let ((dest (string-append accounts-dir (basename f)))) (copy-file f dest) (chown dest user group))) '#$(and declarative-mode? (map jami-account-archive accounts))))) ;; Start the daemon. (define daemon-pid ((make-forkexec-constructor/container '#$(jami-configuration->command-line-arguments config) #:mappings (list (file-system-mapping (source "/dev/log") ;for syslog (target source)) (file-system-mapping (source "/var/lib/jami") (target source) (writable? #t)) (file-system-mapping (source "/var/run/jami") (target source) (writable? #t)) ;; Expose TLS certificates for GnuTLS. (file-system-mapping (source #$(file-append nss-certs "/etc/ssl/certs")) (target "/etc/ssl/certs"))) #:user "jami" #:group "jami" #:environment-variables (list (string-append "DBUS_SESSION_BUS_ADDRESS=" "unix:path=/var/run/jami/bus") ;; Expose TLS certificates for OpenSSL. "SSL_CERT_DIR=/etc/ssl/certs")))) (setenv "DBUS_SESSION_BUS_ADDRESS" "unix:path=/var/run/jami/bus") ;; Wait until the service name has been acquired by D-Bus. (with-retries 20 1 (jami-service-available?)) (when #$declarative-mode? ;; Provision the accounts via the D-Bus API of the daemon. (let* ((jami-account-archives (map (cut string-append "/var/lib/jami/accounts/" <>) (scandir "/var/lib/jami/accounts/" (lambda (f) (not (member f '("." ".."))))))) (usernames (map-in-order (cut add-account <>) jami-account-archives))) (define (archive-name->username archive) (list-ref usernames (list-index (lambda (f) (string-suffix? (basename archive) f)) jami-account-archives))) (for-each (lambda (archive allowed-contacts moderators account-details) (let ((username (archive-name->username archive))) (when (not (unspecified? allowed-contacts)) ;; Reject calls from unknown contacts. (set-account-details '(("DHT.PublicInCalls" . "false")) username) ;; Remove all contacts. (for-each (cut remove-contact <> username) (username->contacts username)) ;; Add allowed ones. (for-each (cut add-contact <> username) allowed-contacts)) (when (not (unspecified? moderators)) ;; Disable the 'AllModerators' property. (set-all-moderators #f username) ;; Remove all moderators. (for-each (cut set-moderator <> #f username) (username->moderators username)) ;; Add declared moderators. (for-each (cut set-moderator <> #t username) moderators)) ;; Set the various account parameters. (set-account-details account-details username))) '#$(and declarative-mode? (map-in-order (cut jami-account-archive <>) accounts)) '#$(and declarative-mode? (map-in-order (cut jami-account-allowed-contacts <>) accounts)) '#$(and declarative-mode? (map-in-order (cut jami-account-moderators <>) accounts)) '#$(and declarative-mode? (map-in-order jami-account->alist accounts))))) ;; Finally, return the PID of the daemon process. daemon-pid)) (stop #~(lambda (pid . args) (kill pid SIGKILL) ;; Wait for the process to exit; this prevents overlapping ;; processes when issuing 'herd restart'. (waitpid pid) #f)))))))) (define jami-service-type (service-type (name 'jami) (default-value (jami-configuration)) (extensions (list (service-extension shepherd-root-service-type jami-shepherd-services) (service-extension account-service-type (const %jami-accounts)) (service-extension activation-service-type jami-dbus-session-activation))) (description "Run the Jami daemon (@command{jamid}). This service is geared toward the use case of hosting Jami rendezvous points over a headless server. If you use Jami on your local machine, you may prefer to setup a user Shepherd service for it instead; this way, the daemon will be shared via your normal user D-Bus session bus."))) ;;; ;;; Mumble server. ;;; ;; https://github.com/mumble-voip/mumble/blob/master/scripts/murmur.ini (define-record-type* mumble-server-configuration make-mumble-server-configuration mumble-server-configuration? (package mumble-server-configuration-package ;file-like (default mumble)) (user mumble-server-configuration-user (default "mumble-server")) (group mumble-server-configuration-group (default "mumble-server")) (port mumble-server-configuration-port (default 64738)) (welcome-text mumble-server-configuration-welcome-text (default "")) (server-password mumble-server-configuration-server-password (default "")) (max-users mumble-server-configuration-max-users (default 100)) (max-user-bandwidth mumble-server-configuration-max-user-bandwidth (default #f)) (database-file mumble-server-configuration-database-file (default "/var/lib/mumble-server/db.sqlite")) (log-file mumble-server-configuration-log-file (default "/var/log/mumble-server/mumble-server.log")) (pid-file mumble-server-configuration-pid-file (default "/var/run/mumble-server/mumble-server.pid")) (autoban-attempts mumble-server-configuration-autoban-attempts (default 10)) (autoban-timeframe mumble-server-configuration-autoban-timeframe (default 120)) (autoban-time mumble-server-configuration-autoban-time (default 300)) (opus-threshold mumble-server-configuration-opus-threshold (default 100)) ; integer percent (channel-nesting-limit mumble-server-configuration-channel-nesting-limit (default 10)) (channelname-regex mumble-server-configuration-channelname-regex (default #f)) (username-regex mumble-server-configuration-username-regex (default #f)) (text-message-length mumble-server-configuration-text-message-length (default 5000)) (image-message-length mumble-server-configuration-image-message-length (default (* 128 1024))) ; 128 Kilobytes (cert-required? mumble-server-configuration-cert-required? (default #f)) (remember-channel? mumble-server-configuration-remember-channel? (default #f)) (allow-html? mumble-server-configuration-allow-html? (default #f)) (allow-ping? mumble-server-configuration-allow-ping? (default #f)) (bonjour? mumble-server-configuration-bonjour? (default #f)) (send-version? mumble-server-configuration-send-version? (default #f)) (log-days mumble-server-configuration-log-days (default 31)) (obfuscate-ips? mumble-server-obfuscate-ips? (default #t)) (ssl-cert mumble-server-configuration-ssl-cert (default #f)) (ssl-key mumble-server-configuration-ssl-key (default #f)) (ssl-dh-params mumble-server-configuration-ssl-dh-params (default #f)) (ssl-ciphers mumble-server-configuration-ssl-ciphers (default #f)) (public-registration mumble-server-configuration-public-registration (default #f)) ; (file mumble-server-configuration-file (default #f))) (define-record-type* mumble-server-public-registration-configuration make-mumble-server-public-registration-configuration mumble-server-public-registration-configuration? (name mumble-server-public-registration-configuration-name) (password mumble-server-public-registration-configuration-password) (url mumble-server-public-registration-configuration-url) (hostname mumble-server-public-registration-configuration-hostname (default #f))) (define (flatten . lst) "Return a list that recursively concatenates all sub-lists of LST." (define (flatten1 head out) (if (list? head) (fold-right flatten1 out head) (cons head out))) (fold-right flatten1 '() lst)) (define (default-mumble-server-config config) (match-record config (user port welcome-text server-password max-users max-user-bandwidth database-file log-file pid-file autoban-attempts autoban-timeframe autoban-time opus-threshold channel-nesting-limit channelname-regex username-regex text-message-length image-message-length cert-required? remember-channel? allow-html? allow-ping? bonjour? send-version? log-days obfuscate-ips? ssl-cert ssl-key ssl-dh-params ssl-ciphers public-registration) (apply mixed-text-file "mumble-server.ini" (flatten "welcometext=" welcome-text "\n" "port=" (number->string port) "\n" (if server-password (list "serverpassword=" server-password "\n") '()) (if max-user-bandwidth (list "bandwidth=" (number->string max-user-bandwidth) "\n") '()) "users=" (number->string max-users) "\n" "uname=" user "\n" "database=" database-file "\n" "logfile=" log-file "\n" "pidfile=" pid-file "\n" (if autoban-attempts (list "autobanAttempts=" (number->string autoban-attempts) "\n") '()) (if autoban-timeframe (list "autobanTimeframe=" (number->string autoban-timeframe) "\n") '()) (if autoban-time (list "autobanTime=" (number->string autoban-time) "\n") '()) (if opus-threshold (list "opusthreshold=" (number->string opus-threshold) "\n") '()) (if channel-nesting-limit (list "channelnestinglimit=" (number->string channel-nesting-limit) "\n") '()) (if channelname-regex (list "channelname=" channelname-regex "\n") '()) (if username-regex (list "username=" username-regex "\n") '()) (if text-message-length (list "textmessagelength=" (number->string text-message-length) "\n") '()) (if image-message-length (list "imagemessagelength=" (number->string image-message-length) "\n") '()) (if log-days (list "logdays=" (number->string log-days) "\n") '()) "obfuscate=" (if obfuscate-ips? "true" "false") "\n" "certrequired=" (if cert-required? "true" "false") "\n" "rememberchannel=" (if remember-channel? "true" "false") "\n" "allowhtml=" (if allow-html? "true" "false") "\n" "allowping=" (if allow-ping? "true" "false") "\n" "bonjour=" (if bonjour? "true" "false") "\n" "sendversion=" (if send-version? "true" "false") "\n" (cond ((and ssl-cert ssl-key) (list "sslCert=" ssl-cert "\n" "sslKey=" ssl-key "\n")) ((or ssl-cert ssl-key) (error "ssl-cert and ssl-key must both be set" ssl-cert ssl-key)) (else '())) (if ssl-dh-params (list "sslDHParams=" ssl-dh-params) '()) (if ssl-ciphers (list "sslCiphers=" ssl-ciphers) '()) (match public-registration (#f '()) (($ name password url hostname) (if (and (or (not server-password) (string-null? server-password)) allow-ping?) (list "registerName=" name "\n" "registerPassword=" password "\n" "registerUrl=" url "\n" (if hostname (string-append "registerHostname=" hostname "\n") "")) (error "To publicly register your mumble-server server your server must be publicy visible and users must be able to join without a password. To fix this set: (allow-ping? #t) (server-password \"\") Or set public-registration to #f")))))))) (define (mumble-server-activation config) #~(begin (use-modules (guix build utils)) (let* ((log-dir (dirname #$(mumble-server-configuration-log-file config))) (pid-dir (dirname #$(mumble-server-configuration-pid-file config))) (db-dir (dirname #$(mumble-server-configuration-database-file config))) (user (getpwnam #$(mumble-server-configuration-user config))) (init-dir (lambda (name dir) (format #t "creating mumble-server ~a directory '~a'\n" name dir) (mkdir-p dir) (chown dir (passwd:uid user) (passwd:gid user)) (chmod dir #o700))) (ini #$(or (mumble-server-configuration-file config) (default-mumble-server-config config)))) (init-dir "log" log-dir) (init-dir "pid" pid-dir) (init-dir "database" db-dir) (format #t "mumble-server: use config file: ~a~%\n" ini) (format #t "mumble-server: to set the SuperUser password run: `~a -ini ~a -readsupw`\n" #$(file-append (mumble-server-configuration-package config) "/bin/mumble-server") ini) #t))) (define mumble-server-accounts (match-lambda (($ _ user group) (list (user-group (name group) (system? #t)) (user-account (name user) (group group) (system? #t) (comment "Mumble server daemon") (home-directory "/var/empty") (shell (file-append shadow "/sbin/nologin"))))))) (define (mumble-server-shepherd-service config) (list (shepherd-service (provision '(mumble-server)) (documentation "Run the Mumble server.") (requirement '(networking)) (start #~(make-forkexec-constructor '(#$(file-append (mumble-server-configuration-package config) "/bin/mumble-server") "-ini" #$(or (mumble-server-configuration-file config) (default-mumble-server-config config))) #:pid-file #$(mumble-server-configuration-pid-file config))) (stop #~(make-kill-destructor))))) (define mumble-server-service-type (service-type (name 'mumble-server) (description "Run the Mumble voice-over-IP (VoIP) server.") (extensions (list (service-extension shepherd-root-service-type mumble-server-shepherd-service) (service-extension activation-service-type mumble-server-activation) (service-extension account-service-type mumble-server-accounts))) (default-value (mumble-server-configuration)))) (define-deprecated/public-alias murmur-configuration mumble-server-configuration) (define-deprecated/public-alias make-murmur-configuration make-mumble-server-configuration) (define-deprecated/public-alias murmur-configuration? mumble-server-configuration?) (define-deprecated/public-alias murmur-configuration-package mumble-server-configuration-package) (define-deprecated/public-alias murmur-configuration-user mumble-server-configuration-user) (define-deprecated/public-alias murmur-configuration-group mumble-server-configuration-group) (define-deprecated/public-alias murmur-configuration-port mumble-server-configuration-port) (define-deprecated/public-alias murmur-configuration-welcome-text mumble-server-configuration-welcome-text) (define-deprecated/public-alias murmur-configuration-server-password mumble-server-configuration-server-password) (define-deprecated/public-alias murmur-configuration-max-users mumble-server-configuration-max-users) (define-deprecated/public-alias murmur-configuration-max-user-bandwidth mumble-server-configuration-max-user-bandwidth) (define-deprecated/public-alias murmur-configuration-database-file mumble-server-configuration-database-file) (define-deprecated/public-alias murmur-configuration-log-file mumble-server-configuration-log-file) (define-deprecated/public-alias murmur-configuration-pid-file mumble-server-configuration-pid-file) (define-deprecated/public-alias murmur-configuration-autoban-attempts mumble-server-configuration-autoban-attempts) (define-deprecated/public-alias murmur-configuration-autoban-timeframe mumble-server-configuration-autoban-timeframe) (define-deprecated/public-alias murmur-configuration-autoban-time mumble-server-configuration-autoban-time) (define-deprecated/public-alias murmur-configuration-opus-threshold mumble-server-configuration-opus-threshold) (define-deprecated/public-alias murmur-configuration-channel-nesting-limit mumble-server-configuration-channel-nesting-limit) (define-deprecated/public-alias murmur-configuration-channelname-regex mumble-server-configuration-channelname-regex) (define-deprecated/public-alias murmur-configuration-username-regex mumble-server-configuration-username-regex) (define-deprecated/public-alias murmur-configuration-text-message-length mumble-server-configuration-text-message-length) (define-deprecated/public-alias murmur-configuration-image-message-length mumble-server-configuration-image-message-length) (define-deprecated/public-alias murmur-configuration-cert-required? mumble-server-configuration-cert-required?) (define-deprecated/public-alias murmur-configuration-remember-channel? mumble-server-configuration-remember-channel?) (define-deprecated/public-alias murmur-configuration-allow-html? mumble-server-configuration-allow-html?) (define-deprecated/public-alias murmur-configuration-allow-ping? mumble-server-configuration-allow-ping?) (define-deprecated/public-alias murmur-configuration-bonjour? mumble-server-configuration-bonjour?) (define-deprecated/public-alias murmur-configuration-send-version? mumble-server-configuration-send-version?) (define-deprecated/public-alias murmur-configuration-log-days mumble-server-configuration-log-days) (define-deprecated/public-alias murmur-configuration-obfuscate-ips? mumble-server-configuration-obfuscate-ips?) (define-deprecated/public-alias murmur-configuration-ssl-cert mumble-server-configuration-ssl-cert) (define-deprecated/public-alias murmur-configuration-ssl-key mumble-server-configuration-ssl-key) (define-deprecated/public-alias murmur-configuration-ssl-dh-params mumble-server-configuration-ssl-dh-params) (define-deprecated/public-alias murmur-configuration-ssl-ciphers mumble-server-configuration-ssl-ciphers) (define-deprecated/public-alias murmur-configuration-public-registration mumble-server-configuration-public-registration) (define-deprecated/public-alias murmur-configuration-file mumble-server-configuration-file) (define-deprecated/public-alias murmur-public-registration-configuration mumble-server-public-registration-configuration) (define-deprecated/public-alias make-murmur-public-registration-configuration make-mumble-server-public-registration-configuration) (define-deprecated/public-alias murmur-public-registration-configuration? mumble-server-public-registration-configuration?) (define-deprecated/public-alias murmur-public-registration-configuration-name mumble-server-public-registration-configuration-name) (define-deprecated/public-alias murmur-public-registration-configuration-url mumble-server-public-registration-configuration-url) (define-deprecated/public-alias murmur-public-registration-configuration-password mumble-server-public-registration-configuration-password) (define-deprecated/public-alias murmur-public-registration-configuration-hostname mumble-server-public-registration-configuration-hostname) (define-deprecated/public-alias murmur-service-type mumble-server-service-type) ;; Local Variables: ;; eval: (put 'with-retries 'scheme-indent-function 2) ;; End: