wiki/HowtoSystemd.md

353 lines
8.7 KiB
Markdown
Raw Normal View History

2016-12-14 16:14:44 +01:00
---
2016-12-14 16:15:13 +01:00
categories: system sysadmin
2016-12-14 16:14:44 +01:00
title: Howto systemd
...
2016-12-14 15:09:19 +01:00
systemd est un gestionnaire de services (démon avec le PID 1) en alternative au System V. Il est utilisé par défaut depuis Debian 8.
2016-12-14 16:07:01 +01:00
> **Note**: systemd ne prend aucune majuscule. Ce n'est pas SystemD ou systemD.
2016-12-14 15:09:19 +01:00
2016-12-14 16:07:01 +01:00
Site officiel du projet : <https://www.freedesktop.org/wiki/Software/systemd/>
2016-12-14 15:09:19 +01:00
2016-12-14 16:07:01 +01:00
Toutes les man pages : <https://www.freedesktop.org/software/systemd/man/index.html>
Man pages utiles :
2016-12-14 15:09:19 +01:00
2016-12-14 16:07:01 +01:00
- <https://www.freedesktop.org/software/systemd/man/systemd.unit.html>
- <https://www.freedesktop.org/software/systemd/man/systemd.service.html>
- <https://www.freedesktop.org/software/systemd/man/systemd.exec.html>
# Utilisation de base
## Status de systemd
Afficher le status :
~~~
2016-12-14 16:13:33 +01:00
# systemctl status
2016-12-14 16:07:01 +01:00
~~~
Lister les unités qui tournent :
~~~
2016-12-14 16:13:33 +01:00
# systemctl
# systemctl list-units
2016-12-14 16:07:01 +01:00
~~~
Lister les unités en échecs :
~~~
2016-12-14 16:13:33 +01:00
# systemctl --failed
2016-12-14 16:07:01 +01:00
~~~
2016-12-14 16:18:01 +01:00
Lister les unités installées :
2016-12-14 16:07:01 +01:00
~~~
2016-12-14 16:13:33 +01:00
# systemctl list-unit-files
2016-12-14 16:07:01 +01:00
~~~
> **Note** : Les unités sont dans `/usr/lib/systemd/system/` (packagé par debian) et `/etc/systemd/system/` (géré par l'administrateur).
## Utilisation des unités
Démarrer une unité :
~~~
# systemctl start unit
~~~
Arrêter une unité :
~~~
# systemctl stop unit
~~~
Redémarrer une unité :
~~~
# systemctl restart unit
~~~
2016-12-14 16:18:01 +01:00
Recharger la configuration d'une unité :
2016-12-14 16:07:01 +01:00
~~~
# systemctl reload unit
~~~
Afficher le status d'une unité :
~~~
2016-12-14 16:13:33 +01:00
# systemctl status unit
2016-12-14 16:07:01 +01:00
~~~
2016-12-14 16:18:01 +01:00
Savoir si une unité est activée ou non :
2016-12-14 16:07:01 +01:00
~~~
2016-12-14 16:13:33 +01:00
# systemctl is-enabled unit
2016-12-14 16:07:01 +01:00
~~~
> **Note** : Attention, si l'unité est en fait un script d'init dans `/etc/init.d/`, `is-enabled` ne fonctionne pas.
Activer une unité au boot :
~~~
# systemctl enable unit
~~~
Désactiver une unité au boot :
~~~
# systemctl disable unit
~~~
Masquer une unité, rendant impossible tout stop/start/… :
~~~
# systemctl mask unit
~~~
Démasquer une unité :
~~~
# systemctl unmask unit
~~~
Afficher la man page correspondante au service de l'unité :
~~~
2016-12-14 16:13:33 +01:00
# systemctl help unit
2016-12-14 16:07:01 +01:00
~~~
2016-12-14 16:18:01 +01:00
Recharger systemd pour prendre en compte les unités modifiées :
2016-12-14 16:07:01 +01:00
~~~
# systemctl daemon-reload
~~~
2016-12-14 15:09:19 +01:00
2016-12-14 16:07:01 +01:00
## Gestion de l'alimentation
Éteindre et redémarrer la machine :
~~~
# systemctl reboot
~~~
Éteindre et couper l'alimentation de la machine :
~~~
# systemctl poweroff
~~~
##### TODO
journald ? Autre page… ?
2016-12-14 15:09:19 +01:00
2016-12-14 16:07:01 +01:00
#### TODO
# Modifier une unité système
~~~
# cp -a /lib/systemd/system/<service>.service /etc/systemd/system/
# vim /etc/systemd/system/<service>.service
# systemctl daemon-reload
~~~
2016-12-14 15:09:19 +01:00
Exemple avec ssh.service sous Debian 8 :
2016-12-14 16:07:01 +01:00
~~~
2016-12-14 16:07:31 +01:00
[Unit]
Description=OpenBSD Secure Shell server
After=network.target auditd.service
ConditionPathExists=!/etc/ssh/sshd_not_to_be_run
[Service]
EnvironmentFile=-/etc/default/ssh
ExecStart=/usr/sbin/sshd -D $SSHD_OPTS
ExecReload=/bin/kill -HUP $MAINPID
KillMode=process
Restart=on-failure
[Install]
WantedBy=multi-user.target
Alias=sshd.service
2016-12-14 16:07:01 +01:00
~~~
2016-12-14 15:09:19 +01:00
2016-12-14 16:31:11 +01:00
> **Note** : Chaque fois qu'une unité est modifiée, il est nécessaire de recharger systemd avec `systemctl daemon-reload`.
2016-12-14 16:07:01 +01:00
# Modifier partiellement une unité
On peut « override » une partie de l'unité en créant un fichier dans `/etc/systemd/system/unit.d/` qui précisera les modifications à faire.
2016-12-14 16:28:38 +01:00
Exemple avec l'unité de varnish :
2016-12-14 16:07:01 +01:00
2016-12-14 16:28:38 +01:00
Dossier `/etc/systemd/system/varnish.service.d`, fichier `/etc/systemd/system/varnish.service.d/varnish.service` :
~~~
[Service]
2016-12-14 16:28:58 +01:00
ExecStart=/usr/sbin/varnishd -a 0.0.0.0:80 -T localhost:6082 -f /etc/varnish/default.vcl -S /etc/varnish/secret -s malloc,16G -p thread_pools=8 -p thread_pool_add_delay=2 -p thread_pool_min=500 -p thread_pool_max=5000
ExecReload=/etc/varnish/reload-vcl.sh
2016-12-14 16:28:38 +01:00
~~~
2016-12-14 16:07:01 +01:00
# Utiliser les templates d'unités
On peut créer/utiliser des templates d'unités. Cela permet de gérer à la volée des instances de services. Par exemple pour redis via `/etc/systemd/system/redis@.service`.
~~~
[Unit]
Description=Advanced key-value store instance %i
After=network.target
[Service]
Type=forking
ExecStart=/usr/bin/redis-server /etc/redis/%i.conf
ExecStartPost=/bin/chgrp %i /var/run/redis/%i.sock
ExecStop=/usr/bin/redis-cli shutdown
Restart=always
User=redis
Group=%i
[Install]
WantedBy=multi-user.target
~~~
`%i` correspond au nom d'instance après le `@`.
On pourra ensuite démarrer l'instance foo et bar avec :
~~~
# systemctl start redis@foo redis@bar
~~~
# systemd par utilisateur
2016-12-14 16:18:01 +01:00
systemd fonctionne avec le PID 1. Mais il est aussi possible de lancer des instances par utilisateur avec le paramètre `--user`. C'est très pratique pour donner les droits à un utilisateur de gérer lui-même ses services. (Instances Tomcat par exemple).
2016-12-14 16:07:01 +01:00
Dans un premier temps il faut s'assurer d'avoir le paquet `libpam-systemd` installé. Ensuite si l'on veut que l'utilisateur `foo` ait toujours une session ouverte (pour faire tourner un service) il faut utiliser loginctl :
~~~
# loginctl enable-linger foo
~~~
De cette façon systemd tournera toujours même si l'utilisateur n'est pas connecté.
Exemple d'unité utilisateur pour tomcat :
`/etc/systemd/user/tomcat.service`
2016-12-14 16:11:37 +01:00
2016-12-14 16:07:01 +01:00
~~~
[Unit]
Description=Tomcat %u.
After=network.target
[Service]
WorkingDirectory=/home/%u
Environment="CATALINA_BASE=/home/%u/tomcat"
EnvironmentFile=/home/%u/tomcat/conf/env
UMask=0002
ExecStart=/usr/share/tomcat7/bin/startup.sh
ExecStop=/usr/share/tomcat7/bin/shutdown.sh
ExecStopPost=/bin/sh -c date | /usr/bin/mail -s "%H/%u : Shutdown instance" foo@bar.com
Type=forking
[Install]
WantedBy=default.target
~~~
2016-12-14 16:18:01 +01:00
L'utilisateur pourra gérer lui-même le service.
2016-12-14 16:12:36 +01:00
2016-12-14 16:07:01 +01:00
~~~
$ systemctl --user enable tomcat
$ systemctl --user start tomcat
$ systemctl --user status -l tomcat
$ systemctl --user restart tomcat
$ systemctl --user stop tomcat
~~~
# Utilisation avancée
2016-12-14 15:09:19 +01:00
2016-12-14 16:07:01 +01:00
## Débug
2016-12-14 15:09:19 +01:00
2016-12-14 16:18:01 +01:00
Afficher plein d'informations sur l'unité :
2016-12-14 16:11:37 +01:00
~~~
# systemctl show <service>
~~~
2016-12-14 16:07:01 +01:00
Débug au démarrage accessible sur tty9 :
2016-12-14 15:09:19 +01:00
2016-12-14 16:07:01 +01:00
~~~
# systemctl enable debug-shell
~~~
2016-12-14 15:09:19 +01:00
2016-12-14 16:07:01 +01:00
Débug d'une unité :
2016-12-14 15:09:19 +01:00
2016-12-14 16:07:01 +01:00
~~~
# systemctl stop unit
# SYSTEMD_LOG_LEVEL=debug /lib/systemd/unit
~~~
2016-12-14 15:09:19 +01:00
2016-12-14 16:07:01 +01:00
Analyser ce qui prend du temps au démarrage :
2016-12-14 15:09:19 +01:00
2016-12-14 16:07:01 +01:00
~~~
# systemd-analyze
Startup finished in 5.019s (firmware) + 6.128s (loader) + 5min 35.864s (kernel) + 20min 3.148s (userspace) = 25min 50.160s
# systemd-analyze blame | head -3
2.029s systemd-udev-settle.service
1.400s docker.service
1.215s uml-utilities.service
2016-12-14 16:12:23 +01:00
~~~
2016-12-14 15:09:19 +01:00
2016-12-14 16:07:01 +01:00
# Migrer de sysvinit à systemd
Cette action est souvent requise lorsque l'on migre de Debian Wheezy à Jessie.
~~~
# apt install systemd-sysv
The following packages will be REMOVED:
sysvinit-core
# reboot
~~~
Vérifier que GRUB ne se lance pas avec l'option `init=/lib/sysvinit/init`.
2016-12-14 15:09:19 +01:00
2016-12-14 16:07:01 +01:00
# FAQ
2016-12-14 15:09:19 +01:00
2016-12-14 16:07:01 +01:00
## Bind9
2016-12-14 15:09:19 +01:00
2016-12-14 16:34:42 +01:00
Sous Debian 8, l'unité par défaut ne gère par les options dans `/etc/default/bind`.
2016-12-14 15:09:19 +01:00
2016-12-14 16:11:37 +01:00
Voir <http://trac.evolix.net/infogerance/wiki/HowtoDebian/MigrationWheezyJessie#Bind>
2016-12-14 15:09:19 +01:00
2016-12-14 16:07:01 +01:00
## strace Apache (TODO)
2016-12-14 15:09:19 +01:00
2016-12-14 16:34:42 +01:00
Une astuce efficace pour stracer un serveur de type Apache était : `strace -s65535 -ff /etc/init.d/apache2 restart`. Ensuite on joue la requête et on des infos (impossible à faire en prod bien sûr!).
Avec systemd, le strace ne suit pas les forks malgré `-ff`. À voir pourquoi systemd empêche ça…
2016-12-14 15:09:19 +01:00
2016-12-14 16:07:01 +01:00
## networking
2016-12-14 15:09:19 +01:00
Sous Debian 8, le démarrage du système se bloque complètement (dead lock) si l'initialisation de process dépendant du réseau (comme NFS, NTP, firewalling…) se fait avant que le réseau soit démarré.
2016-12-14 16:34:42 +01:00
Notamment quand ils sont démarrés via des hooks dans `/etc/network/if-*/`… Cela n'était pas bloquant avec SysV, cf https://bugs.debian.org/cgi-bin/bugreport.cgi?bug=754218#30
2016-12-14 15:09:19 +01:00
2016-12-14 16:34:42 +01:00
Astuces « bourrin » : ajouter `exit 0` dans `/etc/init.d/networking` et voir le souci via `systemctl status networking -l` ; modifier `JobTimeoutUSer` => `systemctl show networking -p JobTimeoutUSer`
2016-12-14 15:09:19 +01:00
2016-12-14 16:07:01 +01:00
## Failed to get D-Bus connection: Unknown error -1
2016-12-14 15:09:19 +01:00
2016-12-14 16:34:42 +01:00
Si en utilisant `systemctl` vous obtenez une erreur "Failed to get D-Bus connection: Unknown error -1" vérifiez si vous êtes bien passé à systemd et n'êtes resté à sysvinit !
2016-12-14 15:09:19 +01:00
2016-12-14 16:07:01 +01:00
## systemd-fsck : Dependency failed for File System Check on /dev/disk/xxx
2016-12-14 15:09:19 +01:00
2016-12-14 16:34:42 +01:00
Par défaut, `systemd-fsck` attend 1m30 l'accès à chaque device disque. Pour diverses raisons (accès à un SAN externe, etc.) cela peut échouer, vous aurez alors :
2016-12-14 15:09:19 +01:00
2016-12-14 16:11:37 +01:00
~~~
systemd[1]: Job dev-disk-by\x2dlabel-foo.device/start timed out.
systemd[1]: Timed out waiting for device dev-disk-by\x2dlabel-foo.device.
systemd[1]: Dependency failed for /backup.
systemd[1]: Dependency failed for Local File Systems.
systemd[1]: Triggering OnFailure= dependencies of local-fs.target.
systemd[1]: Dependency failed for File System Check on /dev/disk/by-label/foo.
~~~
2016-12-14 15:09:19 +01:00
Vous pouvez alors ajouter différentes options via le fstab, du type :
2016-12-14 16:11:37 +01:00
~~~
LABEL=foo /foo ext4 defaults,nofail,x-systemd.device-timeout=120 0 0
~~~
2016-12-14 15:09:19 +01:00
2016-12-14 16:11:37 +01:00
Notamment `nofail` pour éviter que cela fasse échouer votre séquence de boot !