Post

Using Docker Multi stage build

A walkthrough of using Docker multi stage build Docker files to build an image containing an example Go application

Using Docker Multi stage build

I have been using using Docker and Kubernetes off and on for work and personal for a few years now, but I was recently shown a feature in the Docker file that I wasn’t aware of.

A bit of background - I was building a Go application that would sit alongside Squid to perform some updates. Although I was building the appliction with GOOS configured, the created Docker image would only work on my Macbook. Setting GOARCH also fixed it, but this is more interesting and certainly more portable solution.

Multi Stage

Since version 17.06.1-ee-1, Docker has added support for multi stage build in the Docker file. This means you can build your code in on container and make it available to other image builds. The obvious benefit of this for my case is that the target base image can be used as the build env too.

An Example

I wanted to provide a very basic example, so I thought I would create a dummy application that I could build the Dockerfile for.

The Application

The application is going to provide a simple web server that when passed a status code in the path with return a response with that code. This could be potentially useful if you want to test getting various status codes in the wild.

I am writing the application in Go because that is the language I have started using at work - so its all good practice.

HTTP Server

The application creates an HTTP server to listen on port 8080 and return the response with an appropriate status code.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
package example

import (
	"fmt"
	"net/http"
	"os"
	"strconv"
)

func StartServer(stop chan bool) {
	start := make(chan bool, 1)

	http.HandleFunc("/", HttpCodeServer)3

	go func() {
		for {
			select {
			case <-start:
				println(fmt.Sprintf("Starting the service on %s", port))
				err := http.ListenAndServe("0.0.0.0:8080", nil)
				if err != nil {
					println("Restarting the server...")
					start <- true
				}
			case <-stop:
				break
			}

		}
	}()
	println("Starting the server...")
	start <- true
}


func HttpCodeServer(w http.ResponseWriter, r *http.Request) {
	statusCode, err := strconv.Atoi(r.URL.Path[1:])
	if err != nil {
		fmt.Fprintf(w, "Couldn't handle status code %s!", r.URL.Path[1:])
	}
	w.WriteHeader(statusCode)
}

Main command

The application needs an entry point, this is found in main.go under cmd/httpcodes path.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
package main

import (
	"httpcodes/internal/app/example"
	"os"
	"os/signal"

)

func main() {
	stopServer := make(chan bool, 1)
	stop := make(chan os.Signal, 1)
	signal.Notify(stop, os.Interrupt)

	go func() {
		stopServer <- <-stop == os.Interrupt
	}()
	httpcodes.StartServer(stopServer)
	<-stop
}

This code starts the httpcodes server which will do the listening. A channel is used to stop the server when the application recieves a SIGINT.

Folder Structure

I have followed the folder structure laid down in the Go project layout guide.

This puts the Dockerfile into the build/packages structure.

1
2
3
4
5
6
7
8
9
10
11
12
.
├── build
│   └── package
│       └── httpcodes
│           └── Dockerfile
├── cmd
│   └── httpcodes
│       └── main.go
└── internal
    └── app
        └── httpcodes
            └── server.go

Dockerfile

This was the original point of the post, the Dockerfile.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
FROM golang:alpine AS builder

ENV SRCPATH $GOPATH/src/httpcodes
COPY ./ $SRCPATH
RUN go install httpcodes/cmd/httpcodes

FROM alpine:3.11.3

EXPOSE 8080

RUN mkdir -p /app
COPY --from=builder /go/bin/httpcodes /app/
RUN chmod +x /app/httpcodes

CMD ["./app/httpcodes"]

I have kept it deliberately straight forward to highlight the main point.

At the top I am using golang:alpine which has all the required packages already installed to build my application. Notice that I mark this AS builder.

Now, when I create the actually build of the main image, in the COPY command you can see that the --from=builder attribute is used. This attribute is telling the second image to grab the /go/bin/httpcodes binary from the output of the first stage and copy them to the /app folder.

Building and Running

All that is left now is to build and run. From the root of the project run;

1
docker build -t httpcodes -f build/package/httpcodes/Dockerfile .

This will build the project and you can now run it with;

1
docker run -d -p 8080:8080 httpcodes

This command will start the image with port 8080 mapped to the host machine so we can access it. The -d just runs it in detached mode.

Testing

To test, execute a curl command against localhost:8080/ passing a status code.

for example;

1
curl -v http://localhost:8080/502

This should return a verbose message saying we got a 502

1
2
3
4
5
6
7
8
9
10
11
12
13
14
*   Trying 127.0.0.1...
* TCP_NODELAY set
* Connected to localhost (127.0.0.1) port 8080 (#0)
> GET /502 HTTP/1.1
> Host: localhost:8080
> User-Agent: curl/7.64.1
> Accept: */*
>
< HTTP/1.1 502 Bad Gateway
< Date: Sat, 25 Jan 2020 13:02:31 GMT
< Content-Length: 0
<
* Connection #0 to host localhost left intact
* Closing connection 0
This post is licensed under CC BY 4.0 by the author.