Binding container port(s) to a specific IP address

By default, when one publishes a port for a container, all interfaces (IP addresses) listen for traffic on that port. There may be instances where we only want to publish the port on a specific IP address.

In the many Docker Compose snippets I usually see, the ports section of the configuration looks something like the following:

ports:
  - 8080:3000
  - 2432:2432
  - 2222:22

With the above configuration, all IP addresses (localhost, LAN IP addresses, etc) would be listening for traffic on ports 8080, 2432, or 2222.

While this is desirable for most use cases, there are times we just want to bind, for example, port 2432 to a specific IP address, such as 127.0.0.1 (localhost). We can update the above configuration bit to the following where we add an IP address to the source half for that port:

ports:
  - 8080:3000
  - "127.0.0.1:2432:2432"
  - 2222:22

A Use Case: Self-hosted Git Forge

An example use case of this is a self-hosted Git forge such as Forgejo or Gitea. One of the ways one can push/fetch a repository is using SSH

Out of the box, the ports section looks like the following

ports:
  - '3000:3000'
  - '222:22'

Web interface (port 3000) aside, notice that port 222 published to port 22 inside the container. This is because most Linux servers have an SSH server running on them and SSH’s port is 22.

If we keep the configuration as is, our git clone URIs would look like the following

git clone git@forgejo.localdomain:222:username/repo.git

What if we wanted to drop the port specification from the URI? With some modifications to our SSH server and the Docker Compose ports section, we can bind port 22 to an IP address to the container.

In order to accomplish this, the server needs to have access to an additional IP address. In the following example, 192.168.10.56 is the IP address of the VM while 192.168.10.59 is the IP address we have allocated for Forgejo.

Reconfigure OpenSSH

Out of the box, OpenSSH is configured to listen on all interfaces. One approach is to change the listening port, but that is undesirable (and misses the point). Instead, we will reconfigure the SSH server to listen on port 22 on a specific IP address (in my case 192.168.10.56).

We add/update the OpenSSH configuration with the following such that it is only listening on 192.168.10.56:

ListenAddress 192.168.10.56

Additionally, we need to also override the Systemd ssh.socket unit such that it only listens on 192.168.10.56. To override the ssh.socket unit, run systemctl edit ssh.socket to bring up an editor such that we can override the unit with the following:

[Socket]
ListenStream=
ListenStream=192.168.10.56:22

Once that is done, we reload Systemd with systemctl daemon-reload and then restart SSH with systemctl restart ssh.

Reconfigure (Docker) container

Last but not least, we update our Docker Compose config such that we publish port 22 on our second IP address (192.168.10.59)

ports:
  - '3000:3000'
  - '192.168.10.59:22:22'

Now, we should be able to recreate the container such that 192.168.10.59:22 is published to port 22 inside the container. Now our Git URIs get to look like the following (assuming that the DNS record for forgejo.localdomain exists and is pointing at the [second] IP address)

git clone git@forgejo.localdomain:username/repo.git

Note: We can do the same thing to the web interface assuming that we want to simply use what is there and not expose it through a reverse proxy such as Traefik or Nginx.


Posted

in

by