(define-module (ruther home services dconf)
#:use-module (guix gexp)
#:use-module (guix modules)
#:use-module (guix monads)
#:use-module (guix store)
#:use-module (gnu packages gnome)
#:use-module (gnu services)
#:use-module (gnu services configuration)
#:use-module (gnu home services)
#:use-module (srfi srfi-1)
#:use-module (rde serializers ini)
#:export (home-dconf-configuration
home-dconf-service-type))
;; The state will be a scm file.
;; Create a file under the home folder that will
(define-configuration home-dconf-configuration
(dconf (file-like dconf) "The dconf package to use for loading environment")
(config
(ini-config '())
"The configuration of dconf to load. Stuff that's not
declared here is not touched. The stuff that was declared in previous
generation, and got removed, will also be removed from dconf.
The config is fed into dconf load. The ini sections should point to the
location in dconf settings, ie. org/gnome/desktop/interface, and
the values in there refer to values to set, ie. cursor-theme='my-pretty-cursors'."))
(define home-dconf-settings-file "state/dconf-settings.scm")
(define (update-dconf-settings-script dconf dconf-config)
(program-file
"update-dconf-settings"
#~(begin
(define (get-dconf-keys config)
(apply
append
(apply
append
(map
(lambda (section)
(let ((section-name (symbol->string (car section))))
(map
(lambda (section-fields)
(map
(lambda (section-field)
(string-append "/" section-name "/" (symbol->string (car section-field))))
section-fields))
(cdr section))))
config))))
(define (get-deleted-dconf-keys old-config new-config)
(let ((old-keys (get-dconf-keys old-config))
(new-keys (get-dconf-keys new-config)))
(filter
(lambda (key)
(not (member key new-keys)))
old-keys)))
(use-modules (ice-9 popen))
(let* ((dconf #$(file-append dconf "/bin/dconf"))
(new-home (getenv "GUIX_NEW_HOME"))
(old-home (getenv "GUIX_OLD_HOME"))
(new-home-dconf-file (string-append new-home "/" #$home-dconf-settings-file))
(old-home-dconf-file (when old-home
(string-append old-home "/" #$home-dconf-settings-file)))
(new-dconf-settings (with-input-from-file new-home-dconf-file read))
(old-dconf-settings (if old-home-dconf-file
(with-input-from-file old-home-dconf-file read)
'()))
(deleted-dconf-keys (get-deleted-dconf-keys old-dconf-settings new-dconf-settings))
(dconf-ini #$(apply string-append
(serialize-ini-config dconf-config
#:equal-string "="))))
;; Remove settings that are not managed anymore
(display "Removing old dconf keys...")
(newline)
(for-each
(lambda (deleted-key)
(system* dconf "reset" deleted-key))
deleted-dconf-keys)
;; Load the dconf settings
(display "Loading dconf...")
(newline)
(let ((dconf-load-pipe (open-output-pipe (string-append dconf
" load /"))))
(display dconf-ini dconf-load-pipe)
(close-pipe dconf-load-pipe))
(display "Configured dconf.")
(newline)))))
(define (dconf-activation config)
#~(primitive-load #$(update-dconf-settings-script (home-dconf-configuration-dconf config)
(home-dconf-configuration-config config))))
(define (dconf-entry config)
(with-monad %store-monad
(return `((,home-dconf-settings-file
,(computed-file
"dconf-settings.scm"
#~(call-with-output-file #$output
(lambda (port)
(write '#$(home-dconf-configuration-config config) port)))))))))
(define (home-dconf-extensions original-config sections)
(home-dconf-configuration
(inherit original-config)
(config
(append
(home-dconf-configuration-config original-config)
sections))))
(define home-dconf-service-type
(service-type
(name 'home-dconf)
(description "A service to populate dconf settings by given configuration.")
(extend home-dconf-extensions)
(compose concatenate)
(extensions
(list (service-extension home-service-type
dconf-entry)
(service-extension home-activation-service-type
dconf-activation)))
(default-value (home-dconf-configuration))))