How to Expose Ports in Docker

Table of Contents

The Docker open-source platform has revolutionized the way we create, deploy, and manage containerized applications. To containerize an application, you’ll need to write a Dockerfile—which has instructions Docker uses for building and running images.

At times, you may need to set out some networking rules to enable smooth interaction between containers in multi-container applications or make your Docker ports accessible by services in the outside world. 

You can do this in the following ways:

  • Add an EXPOSE instruction in the Dockerfile
  • Use the –expose flag at runtime to expose a port 
  • Use the -p flag or -P flag in the Docker run string to publish a port

Whereas each of the above rules may realize mostly similar results, they work differently. 

So, which rule should you go for?

This article will demonstrate how to apply different networking rules when implementing Docker expose ports instructions in your code. 

Exposing Docker ports via EXPOSE or –expose

There are two ways of exposing ports in Docker:

  • Including an EXPOSE instruction in the Dockerfile
  • Using the –expose flag at runtime

While the two commands are equivalent, they differ in how they work. 

Let’s talk about each of them. 

Using EXPOSE

With the EXPOSE rule, you can tell Docker that the container listens on the stated network ports during runtime.

Here is an example of how to expose a port in Dockerfile:

EXPOSE 8080

The above line will instruct Docker that the container’s service can be connected to via port 8080. 

You can also expose multiple ports:

EXPOSE 80
EXPOSE 8080
EXPOSE 3306
EXPOSE 27018

By default, the EXPOSE keyword specifies that the port listens on TCP protocol. 

Here is how to expose on UDP:

EXPOSE 8080/udp

If you want to specify both UDP and TCP, add two lines:

EXPOSE 8080/udp
EXPOSE 8080/tcp

Using –expose

On the other hand, –expose is a runtime flag that lets you expose a specific port or a range of ports inside the container.

Using the flag is additive, which means that it will expose additional ports together with those stated by the EXPOSE keyword.

Here is an example of how to use the flag in a Docker run string:

docker run --expose=8080 test

You can also provide a range of ports as an argument:

--expose=2000-3000

How EXPOSE and –expose work

Basically, EXPOSE is a documentation mechanism that gives configuration information another command can use, provides a hint about which initial incoming ports will provide services, or informs the decisions that the container operator makes. EXPOSE does not provide much networking control to an image developer. 

As earlier explained, you can use the –expose flag in a Docker run string to add to the exposed ports.

By default, the EXPOSE instruction does not expose the container’s ports to be accessible from the host. In other words, it only makes the stated ports available for inter-container interaction. 

For example, let’s say you have a Node.js application and a Redis server deployed on the same Docker network. To ensure the Node.js application communicates with the Redis server, the Redis container should expose a port. 

If you check the Dockerfile of the official Redis image, a line is included that says EXPOSE 6379. This is what allows the two containers to talk with one another. 

Therefore, when your Node.js application connects to the 6379 port of the Redis container, the EXPOSE directive is what ensures the inter-container communication takes place.

Publishing Docker ports via -P or -p

There are two ways of publishing ports in Docker:

  • Using the -P flag
  • Using the -p flag

Let’s talk about each of them. 

Using the -P flag

Using the -P (upper case) flag at runtime lets you publish all exposed ports to random ports on the host interfaces. It’s short for –publish-all.

As earlier mentioned, EXPOSE is usually used as a documentation mechanism; that is, hinting to the container operator about the port(s) providing services.

Docker allows you to add -P at runtime and convert the EXPOSE instructions in the Dockerfile to specific port mapping rules.

Docker identifies all ports exposed using the EXPOSE directive and those exposed using the –expose parameter. Then, each exposed port is mapped automatically to a random port on the host interface. This automatic mapping also prevents potential port mapping conflicts.

Using the -p flag

Using the -p (lower case) flag at runtime lets you publish a container’s specific port(s) to the Docker host. It’s short for –publish.

It allows you to map a container’s port or a range of ports to the host explicitly—instead of exposing all Docker ports.

Note that irrespective of the EXPOSE instructions in the Dockerfile, using the -p flag at runtime allows you to override them.

There are different formats for declaring the -p flag:

ip:hostPort:containerPort| ip::containerPort \
 | hostPort:containerPort | containerPort

Here is how you can run it:

docker run -p <hostPort>:<containerPort> imageName

Basically, you may leave out either ip or hostPort references. However, you should always state the containerPort to expose. If ip or hostPort is left out, Docker will automatically provide them.

Furthermore, each of the publishing rules defaults to the TCP protocol. If you want UDP, you’ll need to specify it.

Here are some Docker expose port examples using the -p flag:

How publishing ports works

By default, if you use the docker run or docker create command to run or create a container, they do not make any Docker’s ports accessible by services in the outside world.

So, while it’s possible for your Docker containers to connect to the outside world without making any changes to your code, it’s not possible for the outside world to connect to your Docker containers.

If you want to override this default behavior, you can use either the -P or the -p flag in your Docker run string. 

Publishing ports produce a firewall rule that binds a container port to a port on the Docker host, ensuring the ports are accessible to any client that can communicate with the host. 

It’s what makes a port accessible to Docker containers that are not connected to the container’s network, or services that are outside of your Docker environment.

Differences between EXPOSE and publish 

Whereas publishing a port using either -P or -p exposes it, exposing a port using EXPOSE or –expose does not publish it. 

So, while exposed ports can only be accessed internally, published ports can be accessible by external containers and services.  

That’s the main difference between exposing and publishing ports in Docker.

Conclusion

As we’ve demonstrated in this article, implementing various Docker expose ports scenarios during development is not difficult.

We hope you now understand how exposing ports, port binding, and other related concepts work in Docker.

Recent resources

What is LDAP Injection? Types, Examples and How to Prevent It

Learn what LDAP Injection is, its types, examples, and how to prevent it. Secure your applications against LDAP attacks.

Read more

How to Use Dependency Injection in Java: Tutorial with Examples

Learn how to use Dependency Injection in Java with this comprehensive tutorial. Discover its benefits, types, and practical examples.

Read more

Idempotency: The Microservices Architect’s Shield Against Chaos

Discover the power of idempotency in microservices architecture. Learn how to maintain data consistency and predictability.

Read more