TLS for OpenFaaS
TLS for OpenFaaS¶
Transport Layer Security (TLS) is a cryptographic protocol that provides secure encryption on top of HTTP. It is required for any OpenFaaS gateway which is exposed to the Internet.
This guide explains how to obtain TLS certificates for the OpenFaaS Gateway running on Kubernetes. For faasd, see the instructions in the eBook.
- Setup an Ingress Controller
- Configure cert-manager to obtain a certificate from Let's Encrypt
- Configure the an Ingress record for the OpenFaaS Gateway
Pre-requisites¶
- A domain name under your control, and access to create A or CNAME records
- A public IP address with NodePorts, a Load Balancer or a tunnel such as inlets
- A Kubernetes cluster
Where you see example.com
given in an example, replace that with your own domain name.
Make sure you can obtain public IP addresses¶
Managed Kubernetes services have a built-in LoadBalancer provisioner, which will provide a public IP address or CNAME for you, once you create a Service of type LoadBalancer.
If you're running self-managed Kubernetes, where each node has its own Public IP address, then you can configure your Ingress Controller to use a NodePort mapped to port 80 and 443 on the host.
If you are running on a local or private network, you can use inlets-operator instead, which provisions a VM and uses its public IP address over a websocket tunnel.
Set up an Ingress Controller¶
We recommend ingress-nginx for OpenFaaS, however any Ingress controller will work, or you can use Istio with separate instructions.
To install ingress-nginx, use either the Helm chart, or arkade:
$ arkade install ingress-nginx
See also: ingress-nginx installation
Timeouts for synchronous invocations¶
Despite configuring OpenFaaS and your functions for extended timeouts, you may find that your Ingress Controller, Istio Gateway, or Cloud Load Balancer implements its own timeouts on connections. If you think you have everything configured correctly for OpenFaaS, but see a timeout at a very specific number such as 30s or 60s, then check the timeouts on your Ingress Controller or Load Balancer.
For Ingress Nginx, to extend a synchronous invocation beyond one minute, add the nginx.ingress.kubernetes.io/proxy-read-timeout
annotation to your Ingress resource. This annotation is specified in seconds - for example, to extend the timeout to 30 minutes, use nginx.ingress.kubernetes.io/proxy-read-timeout: "1800"
.
Install cert-manager¶
cert-manager is a Kubernetes operator maintained by the Cloud Native Computing Foundation (CNCF) which automates TLS certificate management.
To install cert-manager, use either the Helm chart, or arkade:
$ arkade install cert-manager
See also: cert-manager installation
Configure cert-manager¶
You'll need to create an Issuer or ClusterIssuer for your cert-manager installation. This will tell cert-manager which domain it is operating on, and how to register an account for you.
The below will create an Issuer that only operates in the openfaas namespace, with a HTTP01 challenge. Note the ingress class specified in the HTTP01 challenge, this should match the class of your Ingress controller. You can view ingress classes with kubectl get ingressclass
.
export EMAIL="you@example.com"
cat > issuer.yaml <<EOF
apiVersion: cert-manager.io/v1
kind: Issuer
metadata:
name: letsencrypt-prod
namespace: openfaas
spec:
acme:
server: https://acme-v02.api.letsencrypt.org/directory
email: $EMAIL
privateKeySecretRef:
name: letsencrypt
solvers:
- selector: {}
http01:
ingress:
class: nginx
---
apiVersion: cert-manager.io/v1
kind: Issuer
metadata:
name: letsencrypt-staging
namespace: openfaas
spec:
acme:
server: https://acme-staging-v02.api.letsencrypt.org/directory
email: $EMAIL
privateKeySecretRef:
name: letsencrypt-staging
solvers:
- selector: {}
http01:
ingress:
class: nginx
---
EOF
Apply the staging and production Issuers:
$ kubectl apply -f issuer.yaml
Create the required DNS records¶
You will need to create an A or CNAME record for your domain, pointing to the public IP address of your Ingress controller.
If you created the Ingress Controller with arkade, you'll see a new service in the default namespace called ingress-nginx-controller
. You can find the public IP address with:
$ kubectl get svc -n default ingress-nginx-controller
NAME TYPE CLUSTER-IP EXTERNAL-IP PORT(S) AGE
ingress-nginx-controller LoadBalancer 10.43.87.4 18.136.136.18 80:31876/TCP,443:30108/TCP 28d
Take the IP address from the EXTERNAL-IP
column and create an A record for your domain in your domain management software, or a CNAME record if you're using AWS EKS, and see a domain name in this field.
All users should create an entry for: gateway.example.com
and then OpenFaaS dashboard users should create an additional record pointing at the same address for: dashboard.example.com
.
Configure TLS for the OpenFaaS gateway¶
You can now configure the OpenFaaS gateway to use TLS by setting the following Helm values, you can save them in a file called tls.yaml
:
export DOMAIN="gw.example.com"
export NGINX_TIMEOUT_SECS="1800" # 30 minutes
cat > tls.yaml <<EOF
ingress:
enabled: true
ingressClassName: nginx
annotations:
cert-manager.io/issuer: letsencrypt-prod
nginx.ingress.kubernetes.io/proxy-read-timeout: "$NGINX_TIMEOUT_SECS"
tls:
- hosts:
- $DOMAIN
secretName: openfaas-gateway-cert
hosts:
- host: $DOMAIN
http:
paths:
- path: /
pathType: Prefix
backend:
service:
name: gateway
port:
number: 8080
EOF
If you're using something other than ingress-nginx, then change the ingressClassName
field accordingly. Note that the kubernetes.io/ingress.class
annotation is deprecated and should not be used.
The cert-manager.io/issuer
annotation is used to pick between the staging and production Issuers for Let's Encrypt. If this is your first time working with cert-manager, you may want to use the staging issuer first to avoid running into rate limits if you have something misconfigured.
Now upgrade OpenFaaS via helm, use any custom values.yaml files that you have saved from a previous installation:
helm repo update && \
helm upgrade --install openfaas openfaas/openfaas \
--namespace openfaas \
--values tls.yaml \
--values values-custom.yaml
Configure TLS for the OpenFaaS dashboard¶
If you're using OpenFaaS Standard or OpenFaaS for Enterprises, you will probably want to create an additional Ingress record for the OpenFaaS dashboard.
Edit the previous example:
export DOMAIN="gw.example.com"
export DOMAIN_DASHBOARD="dashboard.example.com"
export NGINX_TIMEOUT_SECS="1800" # 30 minutes
cat > tls.yaml <<EOF
ingress:
enabled: true
ingressClassName: nginx
annotations:
cert-manager.io/issuer: letsencrypt-prod
nginx.ingress.kubernetes.io/proxy-read-timeout: "$NGINX_TIMEOUT_SECS"
tls:
- hosts:
- $DOMAIN
secretName: openfaas-gateway-cert
- hosts:
- $DOMAIN_DASHBOARD
secretName: openfaas-dashboard-cert
hosts:
- host: $DOMAIN
http:
paths:
- path: /
pathType: Prefix
backend:
service:
name: gateway
port:
number: 8080
- host: $DOMAIN_DASHBOARD
http:
paths:
- path: /
pathType: Prefix
backend:
service:
name: dashboard
port:
number: 8080
EOF
As above, run the helm upgrade
command to apply the changes.
Verifying the installation¶
First, check that the DNS records you created have taken effect. You can use nslookup
or dig
to check that the domain names resolve to the public address of your Ingress Controller's service.
$ nslookup gw.example.com
Next, verify that the Ingress records have the desired domains in the "HOSTS" field:
$ kubectl get ingress -n openfaas
Next, check that the certificates have been issued and that they're ready:
$ kubectl get certificates -n openfaas -o wide
If you're still encountering issues, you can check the logs of the cert-manager controller:
$ kubectl logs -n cert-manager deploy/cert-manager
Log into the OpenFaaS gateway using its new URL:
$ export OPENFAAS_URL=https://gw.example.com
$ PASSWORD=$(kubectl get secret -n openfaas basic-auth -o jsonpath="{.data.basic-auth-password}" | base64 --decode; echo)
$ echo -n $PASSWORD | faas-cli login --username admin --password-stdin
# List some functions:
$ faas-cli list
If you're using Identity and Access Management (IAM) for OpenFaaS, see the SSO instructions instead.