gnu: Add webssh service.
* gnu/services/ssh.scm: (<webssh-configuration>): New record type. (%webssh-configuration-nginx, webssh-service-type): New variables. (webssh-account, webssh-activation, webssh-shepherd-service): New procedures. * doc/guix.texi: Document this.master
parent
124562df27
commit
da6aec32cf
|
@ -15547,6 +15547,81 @@ may cause undefined behaviour.
|
||||||
@end table
|
@end table
|
||||||
@end deftp
|
@end deftp
|
||||||
|
|
||||||
|
@cindex WebSSH
|
||||||
|
@deffn {Scheme Variable} webssh-service-type
|
||||||
|
This is the type for the @uref{https://webssh.huashengdun.org/, WebSSH}
|
||||||
|
program that runs a web SSH client. WebSSH can be run manually from the
|
||||||
|
command-line by passing arguments to the binary @command{wssh} from the
|
||||||
|
package @code{webssh}, but it can also be run as a Guix service. This
|
||||||
|
latter use case is documented here.
|
||||||
|
|
||||||
|
For example, to specify a service running WebSSH on loopback interface
|
||||||
|
on port @code{8888} with reject policy with a list of allowed to
|
||||||
|
connection hosts, and NGINX as a reverse-proxy to this service listening
|
||||||
|
for HTTPS connection, add this call to the operating system's
|
||||||
|
@code{services} field:
|
||||||
|
|
||||||
|
@lisp
|
||||||
|
(service webssh-service-type
|
||||||
|
(webssh-configuration (address "127.0.0.1")
|
||||||
|
(port 8888)
|
||||||
|
(policy 'reject)
|
||||||
|
(known-hosts '("localhost ecdsa-sha2-nistp256 AAAA…"
|
||||||
|
"127.0.0.1 ecdsa-sha2-nistp256 AAAA…"))))
|
||||||
|
|
||||||
|
(service nginx-service-type
|
||||||
|
(nginx-configuration
|
||||||
|
(server-blocks
|
||||||
|
(list
|
||||||
|
(nginx-server-configuration
|
||||||
|
(inherit %webssh-configuration-nginx)
|
||||||
|
(server-name '("webssh.example.com"))
|
||||||
|
(listen '("443 ssl"))
|
||||||
|
(ssl-certificate (letsencrypt-certificate "webssh.example.com"))
|
||||||
|
(ssl-certificate-key (letsencrypt-key "webssh.example.com"))
|
||||||
|
(locations
|
||||||
|
(cons (nginx-location-configuration
|
||||||
|
(uri "/.well-known")
|
||||||
|
(body '("root /var/www;")))
|
||||||
|
(nginx-server-configuration-locations %webssh-configuration-nginx))))))))
|
||||||
|
@end lisp
|
||||||
|
@end deffn
|
||||||
|
|
||||||
|
@deftp {Data Type} webssh-configuration
|
||||||
|
Data type representing the configuration for @code{webssh-service}.
|
||||||
|
|
||||||
|
@table @asis
|
||||||
|
@item @code{package} (default: @var{webssh})
|
||||||
|
@code{webssh} package to use.
|
||||||
|
|
||||||
|
@item @code{user-name} (default: @var{"webssh"})
|
||||||
|
User name or user ID that file transfers to and from that module should take
|
||||||
|
place.
|
||||||
|
|
||||||
|
@item @code{group-name} (default: @var{"webssh"})
|
||||||
|
Group name or group ID that will be used when accessing the module.
|
||||||
|
|
||||||
|
@item @code{address} (default: @var{#f})
|
||||||
|
IP address on which @command{webssh} listens for incoming connections.
|
||||||
|
|
||||||
|
@item @code{port} (default: @var{8888})
|
||||||
|
TCP port on which @command{webssh} listens for incoming connections.
|
||||||
|
|
||||||
|
@item @code{policy} (default: @var{#f})
|
||||||
|
Connection policy. @var{reject} policy requires to specify @var{known-hosts}.
|
||||||
|
|
||||||
|
@item @code{known-hosts} (default: @var{'()})
|
||||||
|
List of hosts which allowed for SSH connection from @command{webssh}.
|
||||||
|
|
||||||
|
@item @code{log-file} (default: @file{"/var/log/webssh.log"})
|
||||||
|
Name of the file where @command{rsync} writes its log file.
|
||||||
|
|
||||||
|
@item @code{log-level} (default: @var{#f})
|
||||||
|
Logging level.
|
||||||
|
|
||||||
|
@end table
|
||||||
|
@end deftp
|
||||||
|
|
||||||
@defvr {Scheme Variable} %facebook-host-aliases
|
@defvr {Scheme Variable} %facebook-host-aliases
|
||||||
This variable contains a string for use in @file{/etc/hosts}
|
This variable contains a string for use in @file{/etc/hosts}
|
||||||
(@pxref{Host Names,,, libc, The GNU C Library Reference Manual}). Each
|
(@pxref{Host Names,,, libc, The GNU C Library Reference Manual}). Each
|
||||||
|
|
|
@ -5,6 +5,7 @@
|
||||||
;;; Copyright © 2017 Clément Lassieur <clement@lassieur.org>
|
;;; Copyright © 2017 Clément Lassieur <clement@lassieur.org>
|
||||||
;;; Copyright © 2019 Ricardo Wurmus <rekado@elephly.net>
|
;;; Copyright © 2019 Ricardo Wurmus <rekado@elephly.net>
|
||||||
;;; Copyright © 2020 pinoaffe <pinoaffe@airmail.cc>
|
;;; Copyright © 2020 pinoaffe <pinoaffe@airmail.cc>
|
||||||
|
;;; Copyright © 2020 Oleg Pykhalov <go.wigust@gmail.com>
|
||||||
;;;
|
;;;
|
||||||
;;; This file is part of GNU Guix.
|
;;; This file is part of GNU Guix.
|
||||||
;;;
|
;;;
|
||||||
|
@ -26,6 +27,7 @@
|
||||||
#:use-module (gnu packages admin)
|
#:use-module (gnu packages admin)
|
||||||
#:use-module (gnu services)
|
#:use-module (gnu services)
|
||||||
#:use-module (gnu services shepherd)
|
#:use-module (gnu services shepherd)
|
||||||
|
#:use-module (gnu services web)
|
||||||
#:use-module (gnu system pam)
|
#:use-module (gnu system pam)
|
||||||
#:use-module (gnu system shadow)
|
#:use-module (gnu system shadow)
|
||||||
#:use-module (guix gexp)
|
#:use-module (guix gexp)
|
||||||
|
@ -50,7 +52,12 @@
|
||||||
|
|
||||||
autossh-configuration
|
autossh-configuration
|
||||||
autossh-configuration?
|
autossh-configuration?
|
||||||
autossh-service-type))
|
autossh-service-type
|
||||||
|
|
||||||
|
webssh-configuration
|
||||||
|
webssh-configuration?
|
||||||
|
webssh-service-type
|
||||||
|
%webssh-configuration-nginx))
|
||||||
|
|
||||||
;;; Commentary:
|
;;; Commentary:
|
||||||
;;;
|
;;;
|
||||||
|
@ -732,4 +739,126 @@ object."
|
||||||
autossh-service-activation)))
|
autossh-service-activation)))
|
||||||
(default-value (autossh-configuration))))
|
(default-value (autossh-configuration))))
|
||||||
|
|
||||||
|
|
||||||
|
;;;
|
||||||
|
;;; WebSSH
|
||||||
|
;;;
|
||||||
|
|
||||||
|
(define-record-type* <webssh-configuration>
|
||||||
|
webssh-configuration make-webssh-configuration
|
||||||
|
webssh-configuration?
|
||||||
|
(package webssh-configuration-package ;package
|
||||||
|
(default webssh))
|
||||||
|
(user-name webssh-configuration-user-name ;string
|
||||||
|
(default "webssh"))
|
||||||
|
(group-name webssh-configuration-group-name ;string
|
||||||
|
(default "webssh"))
|
||||||
|
(policy webssh-configuration-policy ;symbol
|
||||||
|
(default #f))
|
||||||
|
(known-hosts webssh-configuration-known-hosts ;list of strings
|
||||||
|
(default #f))
|
||||||
|
(port webssh-configuration-port ;number
|
||||||
|
(default #f))
|
||||||
|
(address webssh-configuration-address ;string
|
||||||
|
(default #f))
|
||||||
|
(log-file webssh-configuration-log-file ;string
|
||||||
|
(default "/var/log/webssh.log"))
|
||||||
|
(log-level webssh-configuration-log-level ;symbol
|
||||||
|
(default #f)))
|
||||||
|
|
||||||
|
(define %webssh-configuration-nginx
|
||||||
|
(nginx-server-configuration
|
||||||
|
(listen '("80"))
|
||||||
|
(locations
|
||||||
|
(list (nginx-location-configuration
|
||||||
|
(uri "/")
|
||||||
|
(body '("proxy_pass http://127.0.0.1:8888;"
|
||||||
|
"proxy_http_version 1.1;"
|
||||||
|
"proxy_read_timeout 300;"
|
||||||
|
"proxy_set_header Upgrade $http_upgrade;"
|
||||||
|
"proxy_set_header Connection \"upgrade\";"
|
||||||
|
"proxy_set_header Host $http_host;"
|
||||||
|
"proxy_set_header X-Real-IP $remote_addr;"
|
||||||
|
"proxy_set_header X-Real-PORT $remote_port;")))))))
|
||||||
|
|
||||||
|
(define webssh-account
|
||||||
|
;; Return the user accounts and user groups for CONFIG.
|
||||||
|
(match-lambda
|
||||||
|
(($ <webssh-configuration> _ user-name group-name _ _ _ _ _ _)
|
||||||
|
(list (user-group
|
||||||
|
(name group-name))
|
||||||
|
(user-account
|
||||||
|
(name user-name)
|
||||||
|
(group group-name)
|
||||||
|
(comment "webssh privilege separation user")
|
||||||
|
(home-directory (string-append "/var/run/" user-name))
|
||||||
|
(shell #~(string-append #$shadow "/sbin/nologin")))))))
|
||||||
|
|
||||||
|
(define webssh-activation
|
||||||
|
;; Return the activation GEXP for CONFIG.
|
||||||
|
(match-lambda
|
||||||
|
(($ <webssh-configuration> _ user-name group-name policy known-hosts _ _
|
||||||
|
log-file _)
|
||||||
|
(with-imported-modules '((guix build utils))
|
||||||
|
#~(begin
|
||||||
|
(let* ((home-dir (string-append "/var/run/" #$user-name))
|
||||||
|
(ssh-dir (string-append home-dir "/.ssh"))
|
||||||
|
(known-hosts-file (string-append ssh-dir "/known_hosts")))
|
||||||
|
(call-with-output-file #$log-file (const #t))
|
||||||
|
(mkdir-p ssh-dir)
|
||||||
|
(case '#$policy
|
||||||
|
((reject)
|
||||||
|
(if '#$known-hosts
|
||||||
|
(call-with-output-file known-hosts-file
|
||||||
|
(lambda (port)
|
||||||
|
(for-each (lambda (host) (display host port) (newline port))
|
||||||
|
'#$known-hosts)))
|
||||||
|
(display-hint (G_ "webssh: reject policy requires `known-hosts'.")))))
|
||||||
|
(for-each (lambda (file)
|
||||||
|
(chown file
|
||||||
|
(passwd:uid (getpw #$user-name))
|
||||||
|
(group:gid (getpw #$group-name))))
|
||||||
|
(list #$log-file ssh-dir known-hosts-file))
|
||||||
|
(chmod ssh-dir #o700)))))))
|
||||||
|
|
||||||
|
(define webssh-shepherd-service
|
||||||
|
(match-lambda
|
||||||
|
(($ <webssh-configuration> package user-name group-name policy _ port
|
||||||
|
address log-file log-level)
|
||||||
|
(list (shepherd-service
|
||||||
|
(provision '(webssh))
|
||||||
|
(documentation "Run webssh daemon.")
|
||||||
|
(start #~(make-forkexec-constructor
|
||||||
|
`(,(string-append #$webssh "/bin/wssh")
|
||||||
|
,(string-append "--log-file-prefix=" #$log-file)
|
||||||
|
,@(case '#$log-level
|
||||||
|
((debug) '("--logging=debug"))
|
||||||
|
(else '()))
|
||||||
|
,@(case '#$policy
|
||||||
|
((reject) '("--policy=reject"))
|
||||||
|
(else '()))
|
||||||
|
,@(if #$port
|
||||||
|
(list (string-append "--port=" (number->string #$port)))
|
||||||
|
'())
|
||||||
|
,@(if #$address
|
||||||
|
(list (string-append "--address=" #$address))
|
||||||
|
'()))
|
||||||
|
#:user #$user-name
|
||||||
|
#:group #$group-name))
|
||||||
|
(stop #~(make-kill-destructor)))))))
|
||||||
|
|
||||||
|
(define webssh-service-type
|
||||||
|
(service-type
|
||||||
|
(name 'webssh)
|
||||||
|
(extensions
|
||||||
|
(list (service-extension shepherd-root-service-type
|
||||||
|
webssh-shepherd-service)
|
||||||
|
(service-extension account-service-type
|
||||||
|
webssh-account)
|
||||||
|
(service-extension activation-service-type
|
||||||
|
webssh-activation)))
|
||||||
|
(default-value (webssh-configuration))
|
||||||
|
(description
|
||||||
|
"Run the webssh.")))
|
||||||
|
|
||||||
;;; ssh.scm ends here
|
;;; ssh.scm ends here
|
||||||
|
|
Reference in New Issue