services: 'fold-service-types' honors its seed.
[jackhill/guix/guix.git] / gnu / services.scm
CommitLineData
db4fdc04 1;;; GNU Guix --- Functional package management for GNU
caa78166 2;;; Copyright © 2015, 2016, 2017 Ludovic Courtès <ludo@gnu.org>
4d343a14 3;;; Copyright © 2016 Chris Marusich <cmmarusich@gmail.com>
db4fdc04
LC
4;;;
5;;; This file is part of GNU Guix.
6;;;
7;;; GNU Guix is free software; you can redistribute it and/or modify it
8;;; under the terms of the GNU General Public License as published by
9;;; the Free Software Foundation; either version 3 of the License, or (at
10;;; your option) any later version.
11;;;
12;;; GNU Guix is distributed in the hope that it will be useful, but
13;;; WITHOUT ANY WARRANTY; without even the implied warranty of
14;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15;;; GNU General Public License for more details.
16;;;
17;;; You should have received a copy of the GNU General Public License
18;;; along with GNU Guix. If not, see <http://www.gnu.org/licenses/>.
19
20(define-module (gnu services)
bebc8681 21 #:use-module (guix gexp)
0adfe95a
LC
22 #:use-module (guix monads)
23 #:use-module (guix store)
db4fdc04 24 #:use-module (guix records)
af4c3fd5 25 #:use-module (guix profiles)
0c0c1b21 26 #:use-module (guix discovery)
0adfe95a
LC
27 #:use-module (guix sets)
28 #:use-module (guix ui)
1bb895ea 29 #:use-module ((guix utils) #:select (source-properties->location))
232ccbef 30 #:use-module (guix modules)
0adfe95a
LC
31 #:use-module (gnu packages base)
32 #:use-module (gnu packages bash)
33 #:use-module (srfi srfi-1)
34 #:use-module (srfi srfi-9)
35 #:use-module (srfi srfi-9 gnu)
36 #:use-module (srfi srfi-26)
37 #:use-module (srfi srfi-34)
38 #:use-module (srfi srfi-35)
39 #:use-module (ice-9 vlist)
40 #:use-module (ice-9 match)
41 #:export (service-extension
42 service-extension?
7d8b5913
CB
43 service-extension-target
44 service-extension-compute
0adfe95a
LC
45
46 service-type
47 service-type?
5152d13b
LC
48 service-type-name
49 service-type-extensions
50 service-type-compose
51 service-type-extend
1bb895ea 52 service-type-default-value
b714395a
LC
53 service-type-description
54 service-type-location
55
0c0c1b21
LC
56 %service-type-path
57 fold-service-types
0adfe95a 58
db4fdc04 59 service
0adfe95a
LC
60 service?
61 service-kind
efe7d19a
LC
62 service-value
63 service-parameters ;deprecated
0adfe95a 64
71654dfd 65 simple-service
cd6f6c22 66 modify-services
5152d13b 67 service-back-edges
0adfe95a
LC
68 fold-services
69
70 service-error?
1bb895ea
LC
71 missing-value-service-error?
72 missing-value-service-error-type
73 missing-value-service-error-location
0adfe95a
LC
74 missing-target-service-error?
75 missing-target-service-error-service
76 missing-target-service-error-target-type
77 ambiguous-target-service-error?
78 ambiguous-target-service-error-service
79 ambiguous-target-service-error-target-type
80
d62e201c 81 system-service-type
0adfe95a 82 boot-service-type
be7be9e8 83 cleanup-service-type
0adfe95a
LC
84 activation-service-type
85 activation-service->script
a241a7ac 86 %linux-bare-metal-service
387e1754
LC
87 special-files-service-type
88 extra-special-file
0adfe95a
LC
89 etc-service-type
90 etc-directory
91 setuid-program-service-type
af4c3fd5 92 profile-service-type
0adfe95a 93 firmware-service-type
e0b47290 94 gc-root-service-type
0adfe95a
LC
95
96 %boot-service
97 %activation-service
d298c815 98 etc-service))
0adfe95a
LC
99
100;;; Comment:
101;;;
102;;; This module defines a broad notion of "service types" and "services."
db4fdc04 103;;;
0adfe95a
LC
104;;; A service type describe how its instances extend instances of other
105;;; service types. For instance, some services extend the instance of
106;;; ACCOUNT-SERVICE-TYPE by providing it with accounts and groups to create;
d4053c71
AK
107;;; others extend SHEPHERD-ROOT-SERVICE-TYPE by passing it instances of
108;;; <shepherd-service>.
0adfe95a
LC
109;;;
110;;; When applicable, the service type defines how it can itself be extended,
111;;; by providing one procedure to compose extensions, and one procedure to
112;;; extend itself.
113;;;
d62e201c
LC
114;;; A notable service type is SYSTEM-SERVICE-TYPE, which has a single
115;;; instance, which is the root of the service DAG. Its value is the
116;;; derivation that produces the 'system' directory as returned by
117;;; 'operating-system-derivation'.
0adfe95a
LC
118;;;
119;;; The 'fold-services' procedure can be passed a list of procedures, which it
120;;; "folds" by propagating extensions down the graph; it returns the root
121;;; service after the applying all its extensions.
db4fdc04
LC
122;;;
123;;; Code:
124
0adfe95a
LC
125(define-record-type <service-extension>
126 (service-extension target compute)
127 service-extension?
128 (target service-extension-target) ;<service-type>
129 (compute service-extension-compute)) ;params -> params
130
1bb895ea
LC
131(define &no-default-value
132 ;; Value used to denote service types that have no associated default value.
133 '(no default value))
134
0adfe95a
LC
135(define-record-type* <service-type> service-type make-service-type
136 service-type?
137 (name service-type-name) ;symbol (for debugging)
138
139 ;; Things extended by services of this type.
140 (extensions service-type-extensions) ;list of <service-extensions>
141
142 ;; Given a list of extensions, "compose" them.
143 (compose service-type-compose ;list of Any -> Any
144 (default #f))
145
146 ;; Extend the services' own parameters with the extension composition.
147 (extend service-type-extend ;list of Any -> parameters
1bb895ea
LC
148 (default #f))
149
150 ;; Optional default value for instances of this type.
151 (default-value service-type-default-value ;Any
b714395a
LC
152 (default &no-default-value))
153
154 ;; Meta-data.
155 (description service-type-description ;string
156 (default #f))
157 (location service-type-location ;<location>
158 (default (and=> (current-source-location)
159 source-properties->location))
160 (innate)))
0adfe95a
LC
161
162(define (write-service-type type port)
163 (format port "#<service-type ~a ~a>"
164 (service-type-name type)
165 (number->string (object-address type) 16)))
166
167(set-record-type-printer! <service-type> write-service-type)
168
0c0c1b21
LC
169(define %distro-root-directory
170 ;; Absolute file name of the module hierarchy.
171 (dirname (search-path %load-path "guix.scm")))
172
173(define %service-type-path
174 ;; Search path for service types.
175 (make-parameter `((,%distro-root-directory . "gnu/services")
176 (,%distro-root-directory . "gnu/system"))))
177
178(define* (fold-service-types proc seed
179 #:optional
180 (modules (all-modules (%service-type-path))))
181 "For each service type exported by one of MODULES, call (PROC RESULT). SEED
182is used as the initial value of RESULT."
183 (fold-module-public-variables (lambda (object result)
184 (if (service-type? object)
185 (proc object result)
186 result))
a3d37f3a 187 seed
0c0c1b21
LC
188 modules))
189
0adfe95a
LC
190;; Services of a given type.
191(define-record-type <service>
1bb895ea 192 (make-service type value)
db4fdc04 193 service?
0adfe95a 194 (type service-kind)
efe7d19a
LC
195 (value service-value))
196
1bb895ea
LC
197(define-syntax service
198 (syntax-rules ()
199 "Return a service instance of TYPE. The service value is VALUE or, if
200omitted, TYPE's default value."
201 ((_ type value)
202 (make-service type value))
203 ((_ type)
204 (%service-with-default-value (current-source-location)
205 type))))
206
207(define (%service-with-default-value location type)
208 "Return a instance of service type TYPE with its default value, if any. If
209TYPE does not have a default value, an error is raised."
210 ;; TODO: Currently this is a run-time error but with a little bit macrology
211 ;; we could turn it into an expansion-time error.
212 (let ((default (service-type-default-value type)))
213 (if (eq? default &no-default-value)
214 (let ((location (source-properties->location location)))
215 (raise
216 (condition
217 (&missing-value-service-error (type type) (location location))
218 (&message
69daee23 219 (message (format #f (G_ "~a: no value specified \
1bb895ea
LC
220for service of type '~a'")
221 (location->string location)
222 (service-type-name type)))))))
223 (service type default))))
224
225(define-condition-type &service-error &error
226 service-error?)
227
228(define-condition-type &missing-value-service-error &service-error
229 missing-value-service-error?
230 (type missing-value-service-error-type)
231 (location missing-value-service-error-location))
232
233
234\f
235;;;
236;;; Helpers.
237;;;
238
efe7d19a
LC
239(define service-parameters
240 ;; Deprecated alias.
241 service-value)
0adfe95a 242
71654dfd
LC
243(define (simple-service name target value)
244 "Return a service that extends TARGET with VALUE. This works by creating a
245singleton service type NAME, of which the returned service is an instance."
246 (let* ((extension (service-extension target identity))
247 (type (service-type (name name)
248 (extensions (list extension)))))
249 (service type value)))
0adfe95a 250
cd6f6c22
LC
251(define-syntax %modify-service
252 (syntax-rules (=>)
253 ((_ service)
254 service)
255 ((_ svc (kind param => exp ...) clauses ...)
256 (if (eq? (service-kind svc) kind)
efe7d19a 257 (let ((param (service-value svc)))
cd6f6c22
LC
258 (service (service-kind svc)
259 (begin exp ...)))
260 (%modify-service svc clauses ...)))))
261
262(define-syntax modify-services
263 (syntax-rules ()
4d343a14
CM
264 "Modify the services listed in SERVICES according to CLAUSES and return
265the resulting list of services. Each clause must have the form:
cd6f6c22
LC
266
267 (TYPE VARIABLE => BODY)
268
269where TYPE is a service type, such as 'guix-service-type', and VARIABLE is an
270identifier that is bound within BODY to the value of the service of that
271TYPE. Consider this example:
272
273 (modify-services %base-services
274 (guix-service-type config =>
275 (guix-configuration
276 (inherit config)
277 (use-substitutes? #f)
278 (extra-options '(\"--gc-keep-derivations\"))))
279 (mingetty-service-type config =>
280 (mingetty-configuration
281 (inherit config)
282 (motd (plain-file \"motd\" \"Hi there!\")))))
283
284It changes the configuration of the GUIX-SERVICE-TYPE instance, and that of
285all the MINGETTY-SERVICE-TYPE instances.
286
287This is a shorthand for (map (lambda (svc) ...) %base-services)."
288 ((_ services clauses ...)
289 (map (lambda (service)
290 (%modify-service service clauses ...))
291 services))))
0adfe95a
LC
292
293\f
294;;;
295;;; Core services.
296;;;
297
d62e201c
LC
298(define (system-derivation mentries mextensions)
299 "Return as a monadic value the derivation of the 'system' directory
300containing the given entries."
301 (mlet %store-monad ((entries mentries)
302 (extensions (sequence %store-monad mextensions)))
303 (lower-object
304 (file-union "system"
305 (append entries (concatenate extensions))))))
306
307(define system-service-type
308 ;; This is the ultimate service type, the root of the service DAG. The
309 ;; service of this type is extended by monadic name/item pairs. These items
310 ;; end up in the "system directory" as returned by
311 ;; 'operating-system-derivation'.
312 (service-type (name 'system)
313 (extensions '())
314 (compose identity)
315 (extend system-derivation)))
316
3a391e68 317(define (compute-boot-script _ mexps)
be7be9e8 318 (mlet %store-monad ((gexps (sequence %store-monad mexps)))
0adfe95a 319 (gexp->file "boot"
d4053c71 320 ;; Clean up and activate the system, then spawn shepherd.
be7be9e8 321 #~(begin #$@gexps))))
0adfe95a 322
d62e201c
LC
323(define (boot-script-entry mboot)
324 "Return, as a monadic value, an entry for the boot script in the system
325directory."
326 (mlet %store-monad ((boot mboot))
327 (return `(("boot" ,boot)))))
328
0adfe95a
LC
329(define boot-service-type
330 ;; The service of this type is extended by being passed gexps as monadic
331 ;; values. It aggregates them in a single script, as a monadic value, which
332 ;; becomes its 'parameters'. It is the only service that extends nothing.
333 (service-type (name 'boot)
d62e201c
LC
334 (extensions
335 (list (service-extension system-service-type
336 boot-script-entry)))
3a391e68
LC
337 (compose append)
338 (extend compute-boot-script)))
0adfe95a
LC
339
340(define %boot-service
d62e201c 341 ;; The service that produces the boot script.
0adfe95a 342 (service boot-service-type #t))
be7be9e8
LC
343
344(define (cleanup-gexp _)
345 "Return as a monadic value a gexp to clean up /tmp and similar places upon
346boot."
fd129893
LC
347 (with-monad %store-monad
348 (with-imported-modules '((guix build utils))
349 (return #~(begin
350 (use-modules (guix build utils))
351
352 ;; Clean out /tmp and /var/run.
353 ;;
354 ;; XXX This needs to happen before service activations, so it
355 ;; has to be here, but this also implicitly assumes that /tmp
356 ;; and /var/run are on the root partition.
357 (letrec-syntax ((fail-safe (syntax-rules ()
358 ((_ exp rest ...)
359 (begin
360 (catch 'system-error
361 (lambda () exp)
362 (const #f))
363 (fail-safe rest ...)))
364 ((_)
365 #t))))
366 ;; Ignore I/O errors so the system can boot.
367 (fail-safe
aad8a143
LC
368 ;; Remove stale Shadow lock files as they would lead to
369 ;; failures of 'useradd' & co.
370 (delete-file "/etc/group.lock")
371 (delete-file "/etc/passwd.lock")
372 (delete-file "/etc/.pwd.lock") ;from 'lckpwdf'
373
fd129893
LC
374 (delete-file-recursively "/tmp")
375 (delete-file-recursively "/var/run")
376 (mkdir "/tmp")
377 (chmod "/tmp" #o1777)
378 (mkdir "/var/run")
379 (chmod "/var/run" #o755))))))))
be7be9e8
LC
380
381(define cleanup-service-type
382 ;; Service that cleans things up in /tmp and similar.
383 (service-type (name 'cleanup)
384 (extensions
385 (list (service-extension boot-service-type
386 cleanup-gexp)))))
0adfe95a 387
0adfe95a
LC
388(define* (activation-service->script service)
389 "Return as a monadic value the activation script for SERVICE, a service of
390ACTIVATION-SCRIPT-TYPE."
efe7d19a 391 (activation-script (service-value service)))
0adfe95a
LC
392
393(define (activation-script gexps)
394 "Return the system's activation script, which evaluates GEXPS."
0adfe95a
LC
395 (define (service-activations)
396 ;; Return the activation scripts for SERVICES.
397 (mapm %store-monad
398 (cut gexp->file "activate-service" <>)
399 gexps))
400
fd129893 401 (mlet* %store-monad ((actions (service-activations)))
0adfe95a 402 (gexp->file "activate"
232ccbef 403 (with-imported-modules (source-module-closure
fe2b6434
CB
404 '((gnu build activation)
405 (guix build utils)))
fd129893 406 #~(begin
fe2b6434
CB
407 (use-modules (gnu build activation)
408 (guix build utils))
0adfe95a 409
caa78166
LC
410 ;; Make sure the user accounting database exists. If it
411 ;; does not exist, 'setutxent' does not create it and
412 ;; thus there is no accounting at all.
413 (close-port (open-file "/var/run/utmpx" "a0"))
414
2986995b
LC
415 ;; Same for 'wtmp', which is populated by mingetty et
416 ;; al.
fe2b6434 417 (mkdir-p "/var/log")
2986995b
LC
418 (close-port (open-file "/var/log/wtmp" "a0"))
419
97bb1ab6
CB
420 ;; Set up /run/current-system. Among other things this
421 ;; sets up locales, which the activation snippets
422 ;; executed below may expect.
423 (activate-current-system)
424
fd129893
LC
425 ;; Run the services' activation snippets.
426 ;; TODO: Use 'load-compiled'.
97bb1ab6 427 (for-each primitive-load '#$actions))))))
0adfe95a
LC
428
429(define (gexps->activation-gexp gexps)
430 "Return a gexp that runs the activation script containing GEXPS."
431 (mlet %store-monad ((script (activation-script gexps)))
432 (return #~(primitive-load #$script))))
433
3a391e68
LC
434(define (second-argument a b) b)
435
0adfe95a
LC
436(define activation-service-type
437 (service-type (name 'activate)
438 (extensions
439 (list (service-extension boot-service-type
440 gexps->activation-gexp)))
441 (compose append)
442 (extend second-argument)))
443
444(define %activation-service
445 ;; The activation service produces the activation script from the gexps it
446 ;; receives.
447 (service activation-service-type #t))
448
a241a7ac
LC
449(define %modprobe-wrapper
450 ;; Wrapper for the 'modprobe' command that knows where modules live.
451 ;;
452 ;; This wrapper is typically invoked by the Linux kernel ('call_modprobe',
453 ;; in kernel/kmod.c), a situation where the 'LINUX_MODULE_DIRECTORY'
454 ;; environment variable is not set---hence the need for this wrapper.
455 (let ((modprobe "/run/current-system/profile/bin/modprobe"))
456 (program-file "modprobe"
457 #~(begin
458 (setenv "LINUX_MODULE_DIRECTORY"
459 "/run/booted-system/kernel/lib/modules")
460 (apply execl #$modprobe
461 (cons #$modprobe (cdr (command-line))))))))
462
463(define %linux-kernel-activation
464 ;; Activation of the Linux kernel running on the bare metal (as opposed to
465 ;; running in a container.)
466 #~(begin
467 ;; Tell the kernel to use our 'modprobe' command.
468 (activate-modprobe #$%modprobe-wrapper)
469
470 ;; Let users debug their own processes!
471 (activate-ptrace-attach)))
472
a241a7ac
LC
473(define %linux-bare-metal-service
474 ;; The service that does things that are needed on the "bare metal", but not
475 ;; necessary or impossible in a container.
6ddf4fcf
LC
476 (simple-service 'linux-bare-metal
477 activation-service-type
478 %linux-kernel-activation))
479
a241a7ac 480
387e1754
LC
481(define special-files-service-type
482 ;; Service to install "special files" such as /bin/sh and /usr/bin/env.
483 (service-type
484 (name 'special-files)
485 (extensions
486 (list (service-extension activation-service-type
487 (lambda (files)
488 #~(activate-special-files '#$files)))))
489 (compose concatenate)
490 (extend append)))
491
492(define (extra-special-file file target)
493 "Use TARGET as the \"special file\" FILE. For example, TARGET might be
494 (file-append coreutils \"/bin/env\")
495and FILE could be \"/usr/bin/env\"."
496 (simple-service (string->symbol (string-append "special-file-" file))
497 special-files-service-type
498 `((,file ,target))))
499
0adfe95a
LC
500(define (etc-directory service)
501 "Return the directory for SERVICE, a service of type ETC-SERVICE-TYPE."
efe7d19a 502 (files->etc-directory (service-value service)))
0adfe95a
LC
503
504(define (files->etc-directory files)
505 (file-union "etc" files))
506
d62e201c
LC
507(define (etc-entry files)
508 "Return an entry for the /etc directory consisting of FILES in the system
509directory."
510 (with-monad %store-monad
511 (return `(("etc" ,(files->etc-directory files))))))
512
0adfe95a
LC
513(define etc-service-type
514 (service-type (name 'etc)
515 (extensions
516 (list
517 (service-extension activation-service-type
518 (lambda (files)
519 (let ((etc
520 (files->etc-directory files)))
d62e201c
LC
521 #~(activate-etc #$etc))))
522 (service-extension system-service-type etc-entry)))
0adfe95a
LC
523 (compose concatenate)
524 (extend append)))
525
526(define (etc-service files)
527 "Return a new service of ETC-SERVICE-TYPE that populates /etc with FILES.
528FILES must be a list of name/file-like object pairs."
529 (service etc-service-type files))
530
531(define setuid-program-service-type
532 (service-type (name 'setuid-program)
533 (extensions
534 (list (service-extension activation-service-type
535 (lambda (programs)
536 #~(activate-setuid-programs
537 (list #$@programs))))))
538 (compose concatenate)
539 (extend append)))
540
af4c3fd5
LC
541(define (packages->profile-entry packages)
542 "Return a system entry for the profile containing PACKAGES."
543 (mlet %store-monad ((profile (profile-derivation
60a0886d
SB
544 (packages->manifest
545 (delete-duplicates packages eq?)))))
af4c3fd5
LC
546 (return `(("profile" ,profile)))))
547
548(define profile-service-type
549 ;; The service that populates the system's profile---i.e.,
550 ;; /run/current-system/profile. It is extended by package lists.
551 (service-type (name 'profile)
552 (extensions
553 (list (service-extension system-service-type
554 packages->profile-entry)))
555 (compose concatenate)
556 (extend append)))
557
0adfe95a
LC
558(define (firmware->activation-gexp firmware)
559 "Return a gexp to make the packages listed in FIRMWARE loadable by the
560kernel."
561 (let ((directory (directory-union "firmware" firmware)))
562 ;; Tell the kernel where firmware is.
563 #~(activate-firmware (string-append #$directory "/lib/firmware"))))
564
565(define firmware-service-type
566 ;; The service that collects firmware.
567 (service-type (name 'firmware)
568 (extensions
569 (list (service-extension activation-service-type
570 firmware->activation-gexp)))
571 (compose concatenate)
572 (extend append)))
573
e0b47290
LC
574(define (gc-roots->system-entry roots)
575 "Return an entry in the system's output containing symlinks to ROOTS."
576 (mlet %store-monad ((entry (gexp->derivation
577 "gc-roots"
578 #~(let ((roots '#$roots))
579 (mkdir #$output)
580 (chdir #$output)
581 (for-each symlink
582 roots
583 (map number->string
584 (iota (length roots))))))))
585 (return (if (null? roots)
586 '()
587 `(("gc-roots" ,entry))))))
588
589(define gc-root-service-type
590 ;; A service to associate extra garbage-collector roots to the system. This
591 ;; is a simple hack that guarantees that the system retains references to
592 ;; the given list of roots. Roots must be "lowerable" objects like
593 ;; packages, or derivations.
594 (service-type (name 'gc-roots)
595 (extensions
596 (list (service-extension system-service-type
597 gc-roots->system-entry)))
598 (compose concatenate)
599 (extend append)))
600
0adfe95a
LC
601\f
602;;;
603;;; Service folding.
604;;;
605
0adfe95a
LC
606(define-condition-type &missing-target-service-error &service-error
607 missing-target-service-error?
608 (service missing-target-service-error-service)
609 (target-type missing-target-service-error-target-type))
610
611(define-condition-type &ambiguous-target-service-error &service-error
612 ambiguous-target-service-error?
613 (service ambiguous-target-service-error-service)
614 (target-type ambiguous-target-service-error-target-type))
615
616(define (service-back-edges services)
617 "Return a procedure that, when passed a <service>, returns the list of
618<service> objects that depend on it."
619 (define (add-edges service edges)
620 (define (add-edge extension edges)
621 (let ((target-type (service-extension-target extension)))
622 (match (filter (lambda (service)
623 (eq? (service-kind service) target-type))
624 services)
625 ((target)
626 (vhash-consq target service edges))
627 (()
628 (raise
629 (condition (&missing-target-service-error
630 (service service)
631 (target-type target-type))
632 (&message
633 (message
638e9dea 634 (format #f (G_ "no target of type '~a' for service '~a'")
0adfe95a 635 (service-type-name target-type)
638e9dea
LC
636 (service-type-name
637 (service-kind service))))))))
0adfe95a
LC
638 (x
639 (raise
640 (condition (&ambiguous-target-service-error
641 (service service)
642 (target-type target-type))
643 (&message
644 (message
645 (format #f
69daee23 646 (G_ "more than one target service of type '~a'")
0adfe95a
LC
647 (service-type-name target-type))))))))))
648
649 (fold add-edge edges (service-type-extensions (service-kind service))))
650
651 (let ((edges (fold add-edges vlist-null services)))
652 (lambda (node)
653 (reverse (vhash-foldq* cons '() node edges)))))
654
d62e201c
LC
655(define* (fold-services services
656 #:key (target-type system-service-type))
0adfe95a
LC
657 "Fold SERVICES by propagating their extensions down to the root of type
658TARGET-TYPE; return the root service adjusted accordingly."
659 (define dependents
660 (service-back-edges services))
661
662 (define (matching-extension target)
663 (let ((target (service-kind target)))
664 (match-lambda
665 (($ <service-extension> type)
666 (eq? type target)))))
667
668 (define (apply-extension target)
669 (lambda (service)
670 (match (find (matching-extension target)
671 (service-type-extensions (service-kind service)))
672 (($ <service-extension> _ compute)
efe7d19a 673 (compute (service-value service))))))
0adfe95a
LC
674
675 (match (filter (lambda (service)
676 (eq? (service-kind service) target-type))
677 services)
678 ((sink)
679 (let loop ((sink sink))
680 (let* ((dependents (map loop (dependents sink)))
681 (extensions (map (apply-extension sink) dependents))
682 (extend (service-type-extend (service-kind sink)))
683 (compose (service-type-compose (service-kind sink)))
efe7d19a 684 (params (service-value sink)))
0adfe95a
LC
685 ;; We distinguish COMPOSE and EXTEND because PARAMS typically has a
686 ;; different type than the elements of EXTENSIONS.
687 (if extend
688 (service (service-kind sink)
689 (extend params (compose extensions)))
690 sink))))
691 (()
692 (raise
693 (condition (&missing-target-service-error
694 (service #f)
695 (target-type target-type))
696 (&message
69daee23 697 (message (format #f (G_ "service of type '~a' not found")
0adfe95a
LC
698 (service-type-name target-type)))))))
699 (x
700 (raise
701 (condition (&ambiguous-target-service-error
702 (service #f)
703 (target-type target-type))
704 (&message
705 (message
706 (format #f
69daee23 707 (G_ "more than one target service of type '~a'")
0adfe95a 708 (service-type-name target-type)))))))))
db4fdc04
LC
709
710;;; services.scm ends here.