Tag Archives: Kubernetes - Page 2

Setup a TLS enabled Ingress in Kubernetes

TLS is a very common security mechanism on web servers through out the Internet. It creates a transport tunnel of encrypted data between the server and client. This tunnel is created with the help of the server certificate and is dependent of that the client trusts the server. We are now going to see an example of how to set this up in the Ingress component of Kubernetes

All certificates are going to be self-signed in this example, regular certificates from trusted sources like Thwate, GlobalSign and many others will also work.

For Kubernetes I will use Minikube with the Ingress addon:

1minikube addons enable ingress

1. We start with creating the server certificate

1openssl req -x509 -nodes -days 365 -newkey rsa:2048 -keyout server.key -out server.crt -subj "/CN=test.localdev.me/O=test.localdev.me"

This should give you two files, a server.key and a server.crt file with the private key and the certificate to present to clients trying to connect.
NOTE: the “domain” test.localdev.me will normally return 127.0.0.1 automatically which makes it perfect to use in cases like this

2. Lets add the certificate to the cluster via a Secret and the special type tls

1kubectl create secret tls server-certificate --key server.key --cert server.crt

3. Now we need an application to call. We create one with the Deployment below:

01apiVersion: apps/v1
02kind: Deployment
03metadata:
04  labels:
05    app: mywebserver
06  name: mywebserver
07spec:
08  replicas: 1
09  selector:
10    matchLabels:
11      app: mywebserver
12  template:
13    metadata:
14      labels:
15        app: mywebserver
16    spec:
17      containers:
18      - image: httpd
19        name: httpd
20        ports:
21        - containerPort: 80

4. A Service to expose the application to the cluster

01apiVersion: v1
02kind: Service
03metadata:
04  labels:
05    app: my-service
06  name: my-service
07spec:
08  ports:
09  - port: 80
10    protocol: TCP
11    targetPort: 80
12  selector:
13    app: mywebserver

5. A Ingress to handle the authentication

01apiVersion: networking.k8s.io/v1
02kind: Ingress
03metadata:
04  name: mtls-ingress
05  namespace: default
06spec:
07  ingressClassName: nginx
08  rules:
09  - host: test.localdev.me
10    http:
11      paths:
12      - backend:
13          service:
14            name: my-service
15            port:
16              number: 80
17        path: /
18        pathType: Prefix
19  tls:
20  - hosts:
21    - test.localdev.me
22    secretName: server-certificate

6. Time to test our TLS setup
First we need to setup a port binding to port 443 on our local machine

1sudo kubectl port-forward -n ingress-nginx service/ingress-nginx-controller 443:443

and now we can test with curl and the test.localdev.me domain

1curl -k -v https://test.localdev.me/

If everything is working we should get “It works!” from the Web Server

Tested in Minikube 1.26.0 and with OpenSSL 1.1.1f on Ubuntu 20.08

Setup a mTLS enabled Ingress in Kubernetes

mTLS is a widely used for securing sensitive data the world over. It is dependent on that both sides of a communication trust each other before a transaction can happen. The trust is based on certificates and can be devided into two parts, one part where the clients needs to trust the servers presented certificate and another part where the server needs to trust the clients presented certificate. We are now going to see an example of how to set this up in Kubernetes

All certificates are going to be self-signed in this example, regular certificates from trusted sources like Thwate, GlobalSign and many others will also work.

For Kubernetes I will use Minikube with the Ingress addon:

1minikube addons enable ingress

1. First we need a server certificate to present to any client wanting to connect

1openssl req -x509 -nodes -days 365 -newkey rsa:2048 -keyout server.key -out server.crt -subj "/CN=test.localdev.me/O=test.localdev.me"

This should give you two files, a server.key and a server.crt file with the private key and the certificate to present to clients trying to connect.
NOTE: the “domain” test.localdev.me will normally return 127.0.0.1 automatically which makes it perfect to use in cases like this

2. Lets add the certificate to the cluster via a Secret and the special type tls

1kubectl create secret tls server-certificate --key server.key --cert server.crt

3. Now we need the client key and certificate. We start by creating our own “CA Authority”

1openssl req -x509 -sha256 -newkey rsa:4096 -keyout ca.key -out ca.crt -days 356 -nodes -subj "/CN=My CA"

4. Add the CA to the cluster as a Secret with the type ca-secret

1kubectl create secret generic ca-secret --from-file=ca.crt=ca.crt

5. A CSR for our client cert

1openssl req -new -newkey rsa:4096 -keyout client.key -out client.csr -nodes -subj "/CN=My Client"

6. Sign the CSR with our CA (same we put into the cluster)

1openssl x509 -req -sha256 -days 365 -in client.csr -CA ca.crt -CAkey ca.key -set_serial 02 -out client.crt

We should now have a client.key and a client.crt ready to use

7. Now we need an application to call. We create one with the Deployment below:

01apiVersion: apps/v1
02kind: Deployment
03metadata:
04  labels:
05    app: mywebserver
06  name: mywebserver
07spec:
08  replicas: 1
09  selector:
10    matchLabels:
11      app: mywebserver
12  template:
13    metadata:
14      labels:
15        app: mywebserver
16    spec:
17      containers:
18      - image: httpd
19        name: httpd
20        ports:
21        - containerPort: 80

8. A Service to expose the application to the cluster

01apiVersion: v1
02kind: Service
03metadata:
04  labels:
05    app: my-service
06  name: my-service
07spec:
08  ports:
09  - port: 80
10    protocol: TCP
11    targetPort: 80
12  selector:
13    app: mywebserver

9. A Ingress to handle the authentication

01apiVersion: networking.k8s.io/v1
02kind: Ingress
03metadata:
04  annotations:
05    nginx.ingress.kubernetes.io/auth-tls-pass-certificate-to-upstream: "true"
06    nginx.ingress.kubernetes.io/auth-tls-secret: default/ca-secret
07    nginx.ingress.kubernetes.io/auth-tls-verify-client: "on"
08    nginx.ingress.kubernetes.io/auth-tls-verify-depth: "1"
09  name: mtls-ingress
10  namespace: default
11spec:
12  ingressClassName: nginx
13  rules:
14  - host: test.localdev.me
15    http:
16      paths:
17      - backend:
18          service:
19            name: my-service
20            port:
21              number: 80
22        path: /
23        pathType: Prefix
24  tls:
25  - hosts:
26    - test.localdev.me
27    secretName: server-certificate

10. Time to test our mTLS setup
First we need to setup a port binding to port 443 on our local machine

1sudo kubectl port-forward -n ingress-nginx service/ingress-nginx-controller 443:443

and now we can test with a call with our client certificate and key

1curl -k -v https://test.localdev.me/ --key client.key --cert client.crt

If everything is working we should get “It works!” from the Web Server

Tested in Minikube 1.26.0 and with OpenSSL 1.1.1f on Ubuntu 20.08

‘kubectl’ on WSL2 gives you “The connection to the server localhost:8080 was refused – did you specify the right host or port?” when using ‘sudo’

I ran into this problem trying to do a port-forward on port 443 in WSL2. Normally you don’t use ‘sudo’ for ‘kubectl’ commands but when it comes to port-forwarding ports under port 1024, 443 in this case, you need ‘sudo’ to get your OS to accept the binding of the port

The solution for me was to make sure the root user had access to the .kube/config file of my normal user. I did this via the environment variable KUBECONFIG:

1. Open a root shell

2. Export the KUBECONFIG with a path to my regular users .kube/config

1export KUBECONFIG=/home/niklas/.kube/config

3. Do the port-forward to port 443 (or anything else that needs root access)

1kubectl port-forward -n ingress-nginx service/ingress-nginx-controller 443:443

I wasn’t looking for a permanent solution so this worked for the few tests I needed to do while setting up my Ingress for mTLS authentication

Tested on ‘kubectl’ 1.26, WSL2 with Ubuntu 20.08 and Kubernetes v1.26 on Minikube