Skip to content

Latest commit

 

History

History
162 lines (123 loc) · 5.91 KB

blog.md

File metadata and controls

162 lines (123 loc) · 5.91 KB

Title: "Use socket activation with Podman to get improved security"

Subtitle: "Learn how to restrict network access for a containerized network server"


Network services that are facing the public internet are exposed to the risk of being attacked. There are different ways of how we could better protect them. For instance, using strong passwords and keeping the systems up to date with the latest security updates, will reduce the risk of an intrusion.

Another advice is trying to minimize the potential damages caused by a compromised network daemon. Assuming the compromise is not caused by a kernel bug, the intruder will not be able to gain any more privileges than those of the running network daemon. A good strategy is therefore to run the network daemon with as few privileges as possible. There is a new feature in Podman that lets us run network daemons with restricted network access. Since version 3.4.0 Podman supports socket activation of containers, i.e., passing a socket-activated socket to the container. Interestingly, it's possible for a container to use such a socket-activated socket even when the network is disabled (i.e., when the option --network=none is given to podman run).

Not all software daemons support socket activation but it's getting more popular. For instance, Apache HTTP server, MariaDB, DBUS, PipeWire, Gunicorn and CUPS all have socket activation support.

An echo server example

Let's try out socket-activate-echo, a simple echo server container that supports socket activation.

Create the container

$ podman create --rm --name echo --network=none ghcr.io/eriksjolund/socket-activate-echo

Generate the systemd service unit

$ mkdir -p ~/.config/systemd/user
$ podman generate systemd --name --new echo > ~/.config/systemd/user/echo.service

A socket-activated service also requires a systemd socket unit. Create the file ~/.config/systemd/user/echo.socket where we define the sockets that the container should use

[Unit]
Description=echo server

[Socket]
ListenStream=127.0.0.1:5040
ListenDatagram=127.0.0.1:5040
ListenStream=[::1]:5040
ListenDatagram=[::1]:5040
ListenStream=%h/echo_stream_sock

[Install]
WantedBy=default.target

%h is a systemd specifier that expands to the user's home directory.

Start the socket unit

$ systemctl --user start echo.socket

Test the echo server with the program socat

$ echo hello | socat - tcp4:127.0.0.1:5040
hello
$ echo hello | socat - tcp6:[::1]:5040
hello
$ echo hello | socat - udp4:127.0.0.1:5040
hello
$ echo hello | socat - udp6:[::1]:5040
hello
$ echo hello | socat - unix:$HOME/echo_stream_sock
hello

The echo server works as expected. It replies "hello" after receiving the text "hello".

Improved security by using --network=none

In case the echo server would get compromised due to a security vulnerability, the container might be used to launch attacks against other PCs or devices on the network. An echo server does not need the ability to establish outgoing connections. It just needs to accept incoming connections on the socket-activated socket it inherited. Fortunately, the command-line option --network=none, given to podman run in the service unit file, provides those restrictions.

$ grep -B8 -- --network=none ~/.config/systemd/user/echo.service
ExecStart=/usr/bin/podman run \
	--cidfile=%t/%n.ctr-id \
	--cgroups=no-conmon \
	--rm \
	--sdnotify=conmon \
	-d \
	--replace \
	--name echo \
	--network=none ghcr.io/eriksjolund/socket-activate-echo

Assume an intruder has shell access in the container. The situation can be simulated by executing commands with podman exec.

$ podman exec -ti echo /usr/sbin/ip -brief addr
lo               UNKNOWN        127.0.0.1/8 ::1/128

Only the loopback interface is available.

$ podman exec -ti echo /usr/bin/curl https://podman.io
curl: (6) Could not resolve host: podman.io

curl is not able to download any web page. The network interface tap0 that rootless Podman normally uses to access the internet is not available.

If we instead remove the option --network=none from the file ~/.config/systemd/user/echo.service and get the service up and running with the new configuration

$ sed -i "s/--network=none//" ~/.config/systemd/user/echo.service
$ systemctl --user daemon-reload
$ systemctl --user stop echo.service
Warning: Stopping echo.service, but it can still be activated by:
  echo.socket
$ echo hello | socat - tcp4:127.0.0.1:5040
hello

and then run the same podman exec commands, we see that the network interface tap0 is available

$ podman exec -ti echo /usr/sbin/ip -brief addr
lo               UNKNOWN        127.0.0.1/8 ::1/128
tap0             UNKNOWN        10.0.2.100/24 fd00::9847:3aff:fe5d:97ea/64 fe80::9847:3aff:fe5d:97ea/64

and that curl is able to download the web page.

$ podman exec -ti echo /usr/bin/curl https://podman.io | head -2
<!doctype html>
<html lang="en-US">

Wrap up

Using socket activation together with the option --network=none for containerized network daemons is a new way to improve security in Podman. It limits the possibilities for an intruder to use a compromised container as a starting point for attacks on other PCs. The socket activation tutorial provides more information about socket activation support in Podman.

Note about SELinux

Note If your computer is running SELinux, you need to have container-selinux 2.183.0 or newer installed to run the examples as specified above. If you are using an older version of container-selinux, add --security-opt label=disable to podman run as a work around.