summaryrefslogtreecommitdiff
path: root/tw/services/restic.scm
diff options
context:
space:
mode:
authorTimo Wilken2023-04-20 23:12:01 +0200
committerTimo Wilken2023-04-21 14:32:02 +0200
commit79ea4090f9b5591e1e93c0bc5a5250ce4b390981 (patch)
treea0a8bca56baa09fab5e420fd385ce75c5f731528 /tw/services/restic.scm
parent5c1c824fb6fc7881e2cf9bedcce7ff24550f8210 (diff)
Add initial restic-server code
Diffstat (limited to 'tw/services/restic.scm')
-rw-r--r--tw/services/restic.scm84
1 files changed, 84 insertions, 0 deletions
diff --git a/tw/services/restic.scm b/tw/services/restic.scm
new file mode 100644
index 00000000..637c7104
--- /dev/null
+++ b/tw/services/restic.scm
@@ -0,0 +1,84 @@
+(define-module (tw services restic)
+ #:use-module (gnu)
+ #:use-module ((gnu packages admin)
+ #:select (shadow))
+ #:use-module (gnu services)
+ #:use-module (gnu services configuration)
+ #:use-module (gnu services shepherd)
+ #:use-module (guix gexp)
+ #:use-module (guix packages)
+ #:use-module (tw packages restic)
+ #:export (restic-server-service-type
+ restic-server-configuration))
+
+(define-maybe/no-serialization integer)
+(define-maybe/no-serialization string)
+
+;; TODO: implement --tls, --tls-cert and --tls-key, maybe using certbot-service-type?
+(define-configuration/no-serialization restic-server-configuration
+ (repository-path (string "/var/lib/restic") "The directory containing
+restic's repositories and @code{.htpasswd} file, unless otherwise configured
+using @code{htpasswd-file}.")
+ (restic-server (package restic-rest-server) "The restic REST server package to use.")
+ (bind-address (string ":8000") "The listen address (including port) to bind to.")
+ (htpasswd-file (maybe-string #f) "Location of @code{.htpasswd} file
+(default: @code{REPOSITORY-PATH/.htpasswd}).")
+ (auth? (boolean #t) "Whether to authenticate users at all (using .htpasswd).")
+ (verify-upload? (boolean #t) "Whether to verify the integrity of uploaded
+data. @emph{Do not disable} unless the restic server is to be run on a very
+low-power device.")
+ (append-only? (boolean #f) "Whether to run the restic server in append-only mode.")
+ (max-repository-size (maybe-integer %unset-value) "Maximum repository size
+in bytes, if any.")
+ (private-repos-only? (boolean #f) "Whether to let users only access their
+private restic repos.")
+ (prometheus? (boolean #f) "Whether to serve Prometheus metrics.")
+ (prometheus-auth? (boolean #t) "Whether to require authentication as the
+@code{metrics} user to access the Prometheus /metrics endpoint."))
+
+(define (restic-server-arguments config)
+ "Turn CONFIG into a list of arguments to the restic-rest-server executable."
+ `("--path" ,(restic-server-configuration-repository-path config)
+ "--log" "/var/log/restic-server.log"
+ "--listen" ,(restic-server-configuration-bind-address config)
+ ,@(if (restic-server-configuration-append-only? config) '("--append-only") '())
+ ,@(let ((max-size (restic-server-configuration-max-repository-size config)))
+ (if (integer? max-size) `("--max-size" ,max-size) '()))
+ ,@(if (restic-server-configuration-private-repos-only? config) '("--private-repos") '())
+ ,@(if (restic-server-configuration-prometheus? config) '("--prometheus") '())
+ ,@(if (restic-server-configuration-prometheus-auth? config) '() '("--prometheus-no-auth"))))
+
+(define (restic-server-service config)
+ "Create a `shepherd-service' for the restic REST server from CONFIG."
+ (list (shepherd-service
+ (provision '(restic-server))
+ (requirement '(networking))
+ (documentation "Run the Restic REST server to serve backup repositories via HTTP.")
+ (start #~(make-forkexec-constructor
+ (list #$(file-append (restic-server-configuration-restic-server config)
+ "/bin/restic-rest-server")
+ #$@(restic-server-arguments config))
+ #:user "restic" #:group "restic"))
+ (stop #~(make-kill-destructor)))))
+
+(define (restic-server-accounts config)
+ "Create user accounts and groups for the restic REST server defined in CONFIG."
+ (list (user-account
+ (name "restic")
+ (group "restic")
+ (comment "Restic server user")
+ (system? #t)
+ (home-directory (restic-server-configuration-repository-path config))
+ (shell (file-append shadow "/sbin/nologin")))
+ (user-group
+ (name "restic")
+ (system? #t))))
+
+(define restic-server-service-type
+ (service-type
+ (name 'restic-server)
+ (extensions
+ (list (service-extension shepherd-root-service-type restic-server-service)
+ (service-extension account-service-type restic-server-accounts)))
+ (description
+ "Restic REST server, running as a service user instead of root.")))