ubuntu:systemd:about_systemd
Differences
This shows you the differences between two versions of the page.
Both sides previous revisionPrevious revisionNext revision | Previous revision | ||
ubuntu:systemd:about_systemd [2023/06/03 09:05] – removed - external edit (Unknown date) 127.0.0.1 | ubuntu:systemd:about_systemd [2023/06/03 09:05] (current) – peter | ||
---|---|---|---|
Line 1: | Line 1: | ||
+ | ====== Ubuntu - SystemD - About SystemD ====== | ||
+ | |||
+ | ===== Introduction systemd ===== | ||
+ | |||
+ | systemd is an init system that provides many powerful features for starting, stopping and managing processes. systemd is also used to manage the lifecycle of Docker containers. | ||
+ | |||
+ | ---- | ||
+ | |||
+ | ===== Terminology ===== | ||
+ | |||
+ | systemd consists of two main concepts: a unit and a target. A unit is a configuration file that describes the properties of the process that you'd like to run. This is normally a **docker run** command or something similar. | ||
+ | |||
+ | systemd is the first process started and it reads different targets and starts the processes specified which allows the operating system to start. | ||
+ | |||
+ | Each target is actually a collection of symlinks to our unit files. | ||
+ | |||
+ | ---- | ||
+ | |||
+ | ===== Unit File ===== | ||
+ | |||
+ | Unit files are generally located within the R/W filesystem at **/ | ||
+ | |||
+ | <file bash / | ||
+ | [Unit] | ||
+ | Description=MyApp | ||
+ | After=docker.service | ||
+ | Requires=docker.service | ||
+ | |||
+ | [Service] | ||
+ | TimeoutStartSec=0 | ||
+ | ExecStartPre=-/ | ||
+ | ExecStartPre=-/ | ||
+ | ExecStartPre=/ | ||
+ | ExecStart=/ | ||
+ | |||
+ | [Install] | ||
+ | WantedBy=multi-user.target | ||
+ | </ | ||
+ | |||
+ | <WRAP info> | ||
+ | **NOTE: | ||
+ | |||
+ | * **Description**: | ||
+ | * **After=docker.service** and **Requires=docker.service**: | ||
+ | * **ExecStart=**: | ||
+ | * The pid assigned to this process is what systemd will monitor to determine whether the process has crashed or not. | ||
+ | * Do not run docker containers with **-d** as this will prevent the container from starting as a child of this pid. systemd will think the process has exited and the unit will be stopped. | ||
+ | * **WantedBy=**: | ||
+ | |||
+ | </ | ||
+ | |||
+ | To start a new unit, we need to tell systemd to create the symlink and then start the file: | ||
+ | |||
+ | <code bash> | ||
+ | sudo systemctl enable / | ||
+ | sudo systemctl start hello.service | ||
+ | </ | ||
+ | |||
+ | To verify the unit started, you can see the list of containers running with **docker ps** and read the unit's output with **journalctl**: | ||
+ | |||
+ | <code bash> | ||
+ | journalctl -f -u hello.service | ||
+ | </ | ||
+ | |||
+ | returns: | ||
+ | |||
+ | <code bash> | ||
+ | -- Logs begin at Fri 2014-02-07 00:05:55 UTC. -- | ||
+ | Feb 11 17:46:26 localhost docker[23470]: | ||
+ | Feb 11 17:46:27 localhost docker[23470]: | ||
+ | Feb 11 17:46:28 localhost docker[23470]: | ||
+ | ... | ||
+ | </ | ||
+ | |||
+ | * [[https:// | ||
+ | |||
+ | * [[https:// | ||
+ | |||
+ | ---- | ||
+ | |||
+ | ===== Advanced Unit Files ===== | ||
+ | |||
+ | systemd provides a high degree of functionality in your unit files. | ||
+ | |||
+ | |||
+ | ^NAME^DESCRIPTION^ | ||
+ | |ExecStartPre|Commands that will run before **ExecStart**.| | ||
+ | |ExecStart|Main commands to run for this unit.| | ||
+ | |ExecStartPost|Commands that will run after all **ExecStart** commands have completed.| | ||
+ | |ExecReload|Commands that will run when this unit is reloaded via **systemctl reload foo.service**.| | ||
+ | |ExecStop|Commands that will run when this unit is considered failed or if it is stopped via **systemctl stop foo.service**.| | ||
+ | |ExecStopPost|Commands that will run after **ExecStop** has completed.| | ||
+ | |RestartSec|The amount of time to sleep before restarting a service. Useful to prevent your failed service from attempting to restart itself every 100ms.| | ||
+ | |||
+ | |||
+ | The full list is located on the [[http:// | ||
+ | |||
+ | Let's put a few of these concepts together to register new units within etcd. Imagine we had another container running that would read these values from etcd and act upon them. | ||
+ | |||
+ | We can use **ExecStartPre** to scrub existing container state. | ||
+ | |||
+ | **docker rm** will remove the container and **docker pull** will pull down the latest version. | ||
+ | |||
+ | **ExecStart** is where the container is started from the container image that we pulled above. | ||
+ | |||
+ | Since our container will be started in **ExecStart**, | ||
+ | |||
+ | When the service is told to stop, we need to stop the docker container using its **< | ||
+ | |||
+ | < | ||
+ | [Unit] | ||
+ | Description=My Advanced Service | ||
+ | After=etcd.service | ||
+ | After=docker.service | ||
+ | |||
+ | [Service] | ||
+ | TimeoutStartSec=0 | ||
+ | ExecStartPre=-/ | ||
+ | ExecStartPre=-/ | ||
+ | ExecStartPre=/ | ||
+ | ExecStart=/ | ||
+ | ExecStartPost=/ | ||
+ | ExecStop=/ | ||
+ | ExecStopPost=/ | ||
+ | |||
+ | [Install] | ||
+ | WantedBy=multi-user.target | ||
+ | </ | ||
+ | |||
+ | While it's possible to manage the starting, stopping, and removal of the container in a single **ExecStart** command by using docker **< | ||
+ | |||
+ | ---- | ||
+ | |||
+ | ===== Unit Specifiers ===== | ||
+ | |||
+ | In our last example we had to hardcode our IP address when we announced our container in etcd. That's not scalable and systemd has a few variables built in to help us out. Here's a few of the most useful: | ||
+ | |||
+ | |||
+ | ^VARIABLE^MEANING^DESCRIPTION^ | ||
+ | |%n|Full unit name|Useful if the name of your unit is unique enough to be used as an argument on a command.| | ||
+ | |%m|Machine ID|Useful for namespacing etcd keys by machine. Example: **/ | ||
+ | |%b|BootID|Similar to the machine ID, but this value is random and changes on each boot.| | ||
+ | |%H|Hostname|Allows you to run the same unit file across many machines. Useful for service discovery. Example: **/ | ||
+ | |||
+ | |||
+ | A full list of specifiers can be found on the [[http:// | ||
+ | |||
+ | ---- | ||
+ | |||
+ | ===== Instantiated Units ===== | ||
+ | |||
+ | Since systemd is based on symlinks, there are a few interesting tricks you can leverage that are very powerful when used with containers. | ||
+ | |||
+ | |||
+ | ^VARIABLE^MEANING^DESCRIPTION^ | ||
+ | |%p|Prefix name|Refers to any string before **@** in your unit name.| | ||
+ | |%i|Instance name|Refers to the string between the **@** and the suffix.| | ||
+ | |||
+ | In our earlier example we had to hardcode our IP address when registering within etcd: | ||
+ | |||
+ | < | ||
+ | ExecStartPost=/ | ||
+ | </ | ||
+ | |||
+ | We can enhance this by using **%H** and **%i** to dynamically announce the hostname and port. Specify the port after the **@** by using two unit files named **foo@123.service** and **foo@456.service**: | ||
+ | |||
+ | < | ||
+ | ExecStartPost=/ | ||
+ | </ | ||
+ | |||
+ | This gives us the flexibility to use a single unit file to announce multiple copies of the same container on a single machine (no port overlap) and on multiple machines (no hostname overlap). | ||
+ | |||
+ | ---- | ||
+ | |||
+ | ===== More Information ===== | ||
+ | |||
+ | * [[http:// | ||
+ | |||
+ | * [[http:// | ||
+ | |||
+ | * [[http:// | ||
+ | |||