Speed up ingester rollout using zone awareness
The Loki zone aware ingesters are used by Grafana Labs in order to allow for easier rollouts of large Loki deployments. You can think of them as three logical zones, however with some extra Kubernetes configuration you could deploy them in separate zones.
By default, an incoming log stream’s logs are replicated to 3 random ingesters. Except in the case of some replica scaling up or down, a given stream will always be replicated to the same 3 ingesters. This means that if one of those ingesters is restarted no data is lost. However two or more ingesters restarting can result in data loss and also impacts the systems ability to ingest logs because of an unhealthy ring status.
With zone awareness enabled, an incoming log line will be replicated to one ingester in each zone. This means that we’re not only concerned about ingesters in multiple zones restarting at the same time, we can now rollout or lose an entire zone at once without impacting writes. This allows deployments with a large number of ingesters to be deployed much more quickly.
At Grafana Labs, we also make use of rollout-operator to manage rollouts to the 3 StatefulSets gracefully. The rollout-operator looks for labels on StatefulSets to know which ones are part of a certain rollout group, and coordinates rollouts of pods only from a single StatefulSet in the group at a time. See the README in the rollout-operator repo for a more in depth explanation.
Migration
Migrating from a single ingester StatefulSet to 3 zone aware ingester StatefulSets. The migration follows a few general steps, regardless of deployment method.
- Configure your existing ingesters to be part of a zone, for example
zone-default
, this will allow us to later exclude them from the write path while still allowing for graceful shutdowns. - Prep for the increase in active streams (due to the way streams are split between ingesters) by increasing the number of active streams allowed for your tenants.
- Add and scale up your new zone-aware ingester StatefulSets such that each has 1/3rd of the total number of replicas you want to run.
- Enable zone awareness on the write path by setting
distributor.zone-awareness-enabled
totrue
for distributors and rulers. - Wait some time to ensure that the new zone-aware ingesters have data for the time period they are queried for (
query_ingesters_within
). - Enable zone awareness on the read path by setting
distributor.zone-awareness-enabled
to true for queriers. - Configure distributors and rulers to exclude ingesters in the
zone-default
so those ingesters no longer receive write traffic usingdistributor.excluded-zones
. - Use the shutdown endpoint to flush data from the default ingesters, then scale down and remove the associated StatefulSet.
- Clean up any config remaining from the migration.
Detailed Migration Steps
The following are steps to live migrate (no downtime) an existing Loki deployment from a single ingester StatefulSet to 3 zone aware ingester StatefulSets.
These instructions assume you are using the zone aware ingester jsonnet deployment code from this repo, see here. If you are not using jsonnet see the relevant annotations in some steps that describe how to perform that step manually.
Configure the zone for the existing ingester StatefulSet as
zone-default
by settingmulti_zone_default_ingester_zone: true
, this allows us to later filter out that zone from the write path.Configure ingester-pdb with
maxUnavailable
as 0 and deploy 3x zone-aware StatefulSets with 0 replicas by setting_config+:: { multi_zone_ingester_enabled: true, multi_zone_ingester_migration_enabled: true, multi_zone_ingester_replicas: 0, // These last two lines are necessary now that we enable zone aware ingester by default // so that newly created cells will not be migrated later on. If you miss them you will // break writes in the cell. multi_zone_ingester_replication_write_path_enabled: false, multi_zone_ingester_replication_read_path_enabled: false, },
If you’re not using jsonnet, the new ingester StatefulSets should have a label with
rollout-group: ingester
, annotationrollout-max-unavailable: x
(put a placeholder value in, later you should set the value of this to be some portion of the StatefulSets total replicas, for example in jsonnet we template this so that each StatefulSet runs 1/3 of the total replicas and the max unavailable is 1/3 of each StatefulSets replicas), and set the update strategy toOnDelete
.Diff ingester and ingester-zone-a StatefulSets and make sure all config matches
kubectl get statefulset -n loki-dev-008 ingester -o yaml > ingester.yaml kubectl get statefulset -n loki-dev-008 ingester-zone-a -o yaml > ingester-zone-a.yaml diff ingester.yaml ingester-zone-a.yaml
Expected in diffs are values like: creation time and revision number, the zone, fields used by rollout operator, number of replicas, anything related to kustomize/flux, and PVC for the WAL since the containers don’t exist yet.
Temporarily double max series limits for users that are using more than 50% of their current limit, the queries are as follows (add label selectors as appropriate):
sum by (tenant)(sum (loki_ingester_memory_streams) by (cluster, namespace, tenant) / on (namespace) group_left max by(namespace) (loki_distributor_replication_factor)) > on (tenant) ( max by (tenant) (label_replace(loki_overrides{limit_name="max_global_streams_per_user"} / 2.5, "tenant", "$1", "user", "(.+)")) )
(sum (loki_ingester_memory_streams) by (cluster, namespace, tenant) / on (namespace) group_left max by(namespace) (loki_distributor_replication_factor) ) / ignoring(tenant) group_left max by (cluster, namespace)(loki_overrides_defaults{limit_name="max_global_streams_per_user"}) > 0.4) unless on (tenant) ( (label_replace(loki_overrides{limit_name="max_global_streams_per_user"},"tenant", "$1", "user", "(.+)")))
Scale up zone-aware StatefulSets until they have 1/3rd of replicas each. In smaller cells you can do this all at once, in larger cells it is safer to do it in chunks. The config value you need to change is
multi_zone_ingester_replicas: 6
, the value will be split across the three StatefulSets. In this case, each StatefulSet would run 2 replicas.If you’re not using jsonnet, this is the step where you would also set the annotation
rollout-max-unavailable
to some value that is less than or equal to the number of replicas each StatefulSet is running.Enable zone awareness on the write path by setting
multi_zone_ingester_replication_write_path_enabled: true
, this causes distributors and rulers to reshuffle series to distributors in each zone. Be sure to check that all the distributors and rulers have restarted properly.If you’re not using jsonnet, enable zone awareness on the write path by setting
distributor.zone-awareness-enabled
to true for distributors and rulers.Wait for
query_ingesters_within
configured hours. The default is3h
. This ensures that no data will be missing if we query a new ingester. However, because we cut chunks at least every 30m due tochunk_idle_period
we can likely reduce this amount of time.Check that rule evaluations are still correct on the migration, look for increases in the rate for metrics with names with the following suffixes:
rule_evaluations_total rule_evaluation_failures_total rule_group_iterations_missed_total
Enable zone-aware replication on the read path
multi_zone_ingester_replication_read_path_enabled: true
. If you’re not using jsonnet, setdistributor.zone-awareness-enabled
to true for queriers.Check that queries are still executing correctly, for example look at
loki_logql_querystats_latency_seconds_count
to see that you don’t have a big increase in latency or error count for a specific query type.Configure distributor / ruler to exclude ingesters in the
zone-default
so those ingesters no longer receive write traffic by settingmulti_zone_ingester_exclude_default: true
. If you’re not using jsonnet setdistributor.excluded-zones
on distributors and rulers.It is a good idea to check rules evaluations again at this point, and also that the zone aware ingester StatefulSet is now receiving all the write traffic, you can compare
sum(loki_ingester_memory_streams{cluster="<cluster>",job=~"(<namespace>)/ingester"})
tosum(loki_ingester_memory_streams{cluster="<cluster>",job=~"(<namespace>)/ingester-zone.*"})
If you’re using an automated reconciliation or deployment system like flux, disable it now (for example using flux ignore) if possible for just the default ingester StatefulSet.
Shutdown flush the default ingesters, unregistering them from the ring, you can do this by port-forwarding each ingester Pod and using the endpoint:
"http://url:PORT/ingester/shutdown?flush=true&delete_ring_tokens=true&terminate=false"
manually scale down the default ingester StatefulSet to 0 replicas, we do this via
tk apply
but you could do it via modifying the yaml.merge a PR to your central config repo to keep the StatefulSet 0’d, and then remove the flux ignore.
clean up any remaining temporary config from the migration, for example
multi_zone_ingester_migration_enabled: true
is no longer needed.ensure that all the old default ingester PVC/PV are removed.