Skip to content

Commit

Permalink
Add HPAv2 tutorial
Browse files Browse the repository at this point in the history
Fixes: openfaas/faas-netes#483

Signed-off-by: Alex Ellis (OpenFaaS Ltd) <[email protected]>
  • Loading branch information
alexellis committed Aug 11, 2019
1 parent 4f19add commit e994467
Show file tree
Hide file tree
Showing 3 changed files with 317 additions and 4 deletions.
7 changes: 3 additions & 4 deletions docs/architecture/autoscaling.md
Original file line number Diff line number Diff line change
Expand Up @@ -37,11 +37,11 @@ For each alert fired the auto-scaler will add a number of replicas, which is a d

When using Kubernetes the built-in [Horizontal Pod Autoscaler (HPA)](https://kubernetes.io/docs/tasks/run-application/horizontal-pod-autoscale/) can be used instead of AlertManager.

Find out more in Stefan Prodan's blog post below:
* Try the 2019 tutorial: [Kubernetes HPAv2 with OpenFaaS](/tutorials/kubernetes-hpa/).

https://stefanprodan.com/2018/kubernetes-scaleway-baremetal-arm-terraform-installer/#horizontal-pod-autoscaling
* Stefan Prodan also wrote a blog post about [HPA with OpenFaaS in 2018](https://stefanprodan.com/2018/kubernetes-scaleway-baremetal-arm-terraform-installer/#horizontal-pod-autoscaling)

In addition to the above, both of the OpenFaaS watchdogs automatically provide custom metrics that can be used for HPAv2 scaling rules.
> Note: In addition to the above, both of the OpenFaaS watchdogs automatically provide custom metrics that can be used for HPAv2 scaling rules.
## Zero-scale

Expand Down Expand Up @@ -82,4 +82,3 @@ The [faas-idler](https://github.com/openfaas-incubator/faas-idler) is deployed b
#### 2) OpenFaaS REST API

If you want to use your own set of criteria for idling functions then you can make use of the OpenFaaS REST API to decide when to scale functions to zero. You can build and deploy your own custom controller for this task.

313 changes: 313 additions & 0 deletions docs/tutorials/kubernetes-hpa.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,313 @@
# Kubernetes HPAv2 with OpenFaaS

Kubernetes has a Pod auto-scaling mechanism built-in called Horizontal Pod Autoscaler (HPA) which can be used as an alternative to the built-in OpenFaaS auto-scaling.

## 1. Pre-reqs

### `hey` load-testing tool

We'll use the [hey](https://github.com/rakyll/hey) tool which also features in the OpenFaaS workshop for auto-scaling.

Download the latest release for your OS from this page and rename it to `hey` or `hey.exe`: [https://github.com/rakyll/hey/releases](https://github.com/rakyll/hey/releases)

### Helm

Some of the dependencies in this tutorial may rely on `helm`. The client-side CLI called `helm` is not insecure, but some people have concerns about using the server-side component named `tiller`. If you think that using helm's tiller component is insecure, then you should use the `helm template` command instead.

### Install OpenFaaS

Use the Deployment guide or an existing installation, you should also install the CLI.

### Install the metrics server

HPAv2 relies on the [Kubernetes metrics-server](https://github.com/kubernetes-incubator/metrics-server) which can be installed using a helm chart.

Most cloud providers are compatible with the metrics-server, but some are not, or require an additional "insecure" flag to be configured.

```sh
helm install --name metrics-server --namespace kube-system \
stable/metrics-server
```

You can see the [Helm chart](https://github.com/helm/charts/tree/master/stable/metrics-server) for more options.

## 2. Check your metrics-server

When you see metrics appear from this command, continue with the tutorial.

```sh
kubectl get --raw "/apis/metrics.k8s.io/v1beta1/nodes"

kubectl get --raw /metrics
```

Check the logs of the metrics-server for any errors:

```
kubectl logs deploy/metrics-server -n kube-system
```

If you don't see any metrics, then you may be using a cloud which needs the "insecure" work-around.

```sh
helm del --purge metrics-server

helm install --name metrics-server --namespace kube-system \
stable/metrics-server \
--set args="{--kubelet-insecure-tls,--kubelet-preferred-address-types=InternalIP,ExternalIP,Hostname}"
```

You will now get Pod metrics along with Node metrics, these take a while to propagate.

Find out the CPU and memory usage across nodes:

```sh
kubectl top node
NAME CPU(cores) CPU% MEMORY(bytes) MEMORY%
nuc7 377m 4% 7955Mi 24%
```

Find out the usage across Pods:

```sh
# View the functions
kubectl top pod -n openfaas-fn
NAME CPU(cores) MEMORY(bytes)
nodeinfo-6f48f9b548-gbtr4 2m 3Mi

# View the core services
kubectl top pod -n openfaas
NAME CPU(cores) MEMORY(bytes)
alertmanager-666c65c694-k8q6h 2m 10Mi
basic-auth-plugin-6d97c6dc5b-rbw29 1m 4Mi
faas-idler-67f9dcd4fc-85rbt 1m 4Mi
gateway-7c687d498f-nvjh8 4m 21Mi
nats-d4c9d8d95-fxrc2 1m 6Mi
prometheus-549c7d687d-zznmq 9m 41Mi
queue-worker-544bcb7c67-72l72 1m 3Mi
```

> Note: you can view function usage using `-n openfaas-fn`, or core-service usage with `-n openfaas`
### Disable auto-scaling with OpenFaaS

Disable auto-scaling by scaling alertmanager down to zero replicas, this will stop it from firing alerts.

```sh
kubectl scale -n openfaas deploy/alertmanager --replicas=0
```

### Configure Prometheus to scrape the Kubernetes API server

The metrics available via kubectl top / pod can now be scraped with Prometheus. OpenFaaS ships with Prometheus, and you could reconfigure that version to do this task, but for the tutorial we will install it again.

```
kubectl create namespace prometheus
helm install stable/prometheus \
--name prometheus \
--namespace prometheus \
--set alertmanager.persistentVolume.enabled=false,server.persistentVolume.enabled=false,pushgateway.enabled=false
```

You will see that a number of pods have been created including "node-exporter" which fetches statistics about each node such as free disk space and CPU utilization.

```
kubectl get pods -n prometheus
```

Now port-forward the Prometheus dashboard:

```
kubectl --namespace=prometheus port-forward deploy/prometheus-server 9090
```

From here you can open a browser at `http://127.0.0.1:9090`

Prometheus has been configured to use the Kubernetes service discovery mechanism. It will find any Pods with metrics then scrape them.

* See your targets at: http://127.0.0.1:9090/targets
* Enter query such as `container_memory_usage_bytes` on the Graph page.

### Deploy your first function

For HPAv2 to work, we need to deploy a function with a minimum request value for CPU.

Let's create a `stack.yml` file:

```yaml
version: 1.0
provider:
name: openfaas
gateway: http://127.0.0.1:31112
functions:
nodeinfo:
image: functions/nodeinfo:latest
skip_build: true
requests:
cpu: 10m
```
```sh
faas-cli deploy
```

Check that the CPU request was created on the Deployment:

```sh
kubectl describe deploy/nodeinfo -n openfaas-fn | grep cpu

cpu: 10m
```

Note: you can find the Docker image for other store functions use the following:

```sh
faas-cli store list

faas-cli store inspect FUNCTION_NAME
```

### Create a HPAv2 rule for CPU/memory

You can create a rule via YAML, or create one via the CLI:

```sh
kubectl autoscale deployment -n openfaas-fn \
nodeinfo \
--cpu-percent=50 \
--min=1 \
--max=10

horizontalpodautoscaler.autoscaling/nodeinfo autoscaled
```

* `-n openfaas-fn` refers to where the function is deployed
* `nodeinfo` is the name of the function
* `--cpu-percentage` is the level of CPU the Pod should reach before additional replicas are added
* `--min` minimum number of Pods
* `--max` maximum number of Pods

View the HPA record with:

```
kubectl get hpa/nodeinfo -n openfaas-fn
NAME REFERENCE TARGETS MINPODS MAXPODS REPLICAS AGE
nodeinfo Deployment/nodeinfo <unknown>/50% 1 10 0 15s
# view the YAML
kubectl get hpa/nodeinfo -n openfaas-fn -o yaml
```

You can use `kubectl describe hpa/nodeinfo -n openfaas-fn` to get detailed information including any events such as scaling up and down.

```
kubectl describe hpa/nodeinfo -n openfaas-fn
Name: nodeinfo
Namespace: openfaas-fn
Labels: <none>
Annotations: <none>
CreationTimestamp: Sat, 10 Aug 2019 11:48:42 +0000
Reference: Deployment/nodeinfo
Metrics: ( current / target )
resource cpu on pods (as a percentage of request): 20% (2m) / 50%
Min replicas: 1
Max replicas: 10
Deployment pods: 4 current / 4 desired
Conditions:
Type Status Reason Message
---- ------ ------ -------
AbleToScale True ScaleDownStabilized recent recommendations were higher than current one, applying the highest recent recommendation
ScalingActive True ValidMetricFound the HPA was able to successfully calculate a replica count from cpu resource utilization (percentage of request)
ScalingLimited False DesiredWithinRange the desired count is within the acceptable range
Events: <none>
```

### Generate some traffic

Use the following command with hey

```
export OPENFAAS_URL=http://127.0.0.1:31112
hey -c 5 \
-z 5m \
$OPENFAAS_URL/function/nodeinfo
```

* `-c` will simulate `5` concurrent users
* `-z` will run for `5m` and then complete

You should note that HPA is designed to react slowly to changes in traffic, both for scaling up and for scaling down. In some instances you may wait 20 minutes for all your Pods to scale back down to a normal level after the load has stopped.

Now in a new window monitor the progress:

```
kubectl describe hpa/nodeinfo -n openfaas
```

Or in an automated fashion:

```
# watch -n 5 "kubectl describe hpa/nodeinfo -n openfaas-fn"
```

You can also monitor the replicas of your function in the OpenFaaS UI or via the CLI:

```
watch -n 5 "faas-cli list"
```

Here is ane example of the replicas scaling up in response to the traffic created by `hey`:

```sh
Name: nodeinfo
Namespace: openfaas-fn
Labels: <none>
Annotations: <none>
CreationTimestamp: Sat, 10 Aug 2019 11:48:42 +0000
Reference: Deployment/nodeinfo
Metrics: ( current / target )
resource cpu on pods (as a percentage of request): 12325% (1232m) / 50%
Min replicas: 1
Max replicas: 10
Deployment pods: 8 current / 10 desired
Conditions:
Type Status Reason Message
---- ------ ------ -------
AbleToScale True SucceededRescale the HPA controller was able to update the target scale to 10
ScalingActive True ValidMetricFound the HPA was able to successfully calculate a replica count from cpu resource utilization (percentage of request)
ScalingLimited True TooManyReplicas the desired replica count is more than the maximum replica count
Events:
Type Reason Age From Message
---- ------ ---- ---- -------
Normal SuccessfulRescale 18s horizontal-pod-autoscaler New size: 8; reason: cpu resource utilization (percentage of request) above target
Normal SuccessfulRescale 3s horizontal-pod-autoscaler New size: 10; reason: cpu resource utilization (percentage of request) above target
```

Note that whilst the scaling up was relatively quick, the scale-down may take significantly longer.

## 3. Wrapping up

In this tutorial we disabled the auto-scaling built into OpenFaaS which uses Prometheus and Alertmanager, and added in Kubernetes' own HPAv2 mechanism, its metrics-server and another deployment of Prometheus.

### Notes and caveats

* Additional resource usage by Kubernetes

The additional services and workloads mentioned above do not come for free, and you may notice an increase in CPU and memory consumption across your cluster. Generally the auto-scaler in OpenFaaS is more efficient and light-weight.

* Scale to zero

You may not be able to scale your functions to zero if they are being managed by a HPAv2 rule with a minimum value of `1`, this is because after they are scaled down, the HPA rule will override the setting.

Nothing should block scaling up from zero, but HPAv2 is unlikely to allow any function to scale down and stay at `0/0` replicas

### Take it further

* Scale based upon RAM usage, instead of CPU
* Mix OpenFaaS auto-scaling with HPA for different use-cases
* Look into scaling with HPAv2 using "custom metrics" - [k8s-prom-hpa tutorial by Stefan Prodan](https://github.com/stefanprodan/k8s-prom-hpa)
1 change: 1 addition & 0 deletions mkdocs.yml
Original file line number Diff line number Diff line change
Expand Up @@ -155,6 +155,7 @@ nav:
- CLI with Node.js: ./tutorials/CLI-with-node.md
- First Python Function: ./tutorials/first-python-function.md
- Test Drive: ./tutorials/test-drive.md
- Kubernetes HPAv2: ./tutorials/kubernetes-hpa.md
- Featured: ./tutorials/featured.md
- Contributing:
- Get Started: ./contributing/get-started.md
Expand Down

0 comments on commit e994467

Please sign in to comment.