mirror of https://github.com/grafana/loki
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
181 lines
7.4 KiB
181 lines
7.4 KiB
|
3 years ago
|
---
|
||
|
|
title: "0003: Query fairness across users within tenants"
|
||
|
|
description: "Improve the query scheduler to ensure fairness between individual users of individual tenants."
|
||
|
|
---
|
||
|
|
|
||
|
|
# 0003: Query fairness across users within tenants
|
||
|
|
|
||
|
|
**Author:** Christian Haudum (<christian.haudum@grafana.com>)
|
||
|
|
|
||
|
|
**Date:** 02/2023
|
||
|
|
|
||
|
|
**Sponsor(s):** @chaudum @owen-d
|
||
|
|
|
||
|
|
**Type:** Feature
|
||
|
|
|
||
|
|
**Status:** Accepted
|
||
|
|
|
||
|
|
**Related issues/PRs:**
|
||
|
|
|
||
|
|
**Thread from [mailing list](https://groups.google.com/forum/#!forum/lokiproject):**
|
||
|
|
|
||
|
|
---
|
||
|
|
|
||
|
|
## Background
|
||
|
|
|
||
|
|
The query scheduler (or short scheduler) is a component of Loki that distributes requests (sub-queries) from the query frontend (or short frontend) to the querier workers so that execution fairness between tenants can be guaranteed.
|
||
|
|
|
||
|
|
By maintaining separate FIFO queues for each tenant and assigning the correct amount of querier workers to these queues, the scheduler takes care that a single tenant cannot compromise all other tenants' query capabilities.
|
||
|
|
|
||
|
|
**Component diagram:**
|
||
|
|
|
||
|
|

|
||
|
|
|
||
|
|
**Sequence diagram:**
|
||
|
|
|
||
|
|

|
||
|
|
|
||
|
|
## Problem Statement
|
||
|
|
|
||
|
|
Even though Loki is built as multi-tenant system by default, there are use-cases where a Loki installation only has a very large, single tenant, e.g. dedicated Loki cells for customers in Grafana Cloud.
|
||
|
|
|
||
|
|
However, there are potentially a lot of different users using the same tenant to query logs, such as users accessing Loki from Grafana or via CLI or HTTP API. This can lead to contention between queries of different users, because they all share the same tenant.
|
||
|
|
|
||
|
|
While the current implementation of the scheduler queues allows for QoS guarantees between tenants, it does not account for QoS guarantees across individual users within a single tenant.
|
||
|
|
|
||
|
|
That said, Loki does not have the notation of individual users.
|
||
|
|
|
||
|
|
## Goals
|
||
|
|
|
||
|
|
The main goal of the following proposals is to lay out ideas how to improve the scheduler component to not only assure QoS across tenants, but also across actors (users) within a tenant, without requiring any changes to the deployment model of frontend, scheduler and queriers.
|
||
|
|
This should also include changes to the queue structure to be easily extensible for future scheduling improvements.
|
||
|
|
|
||
|
|
## Non-Goals (optional)
|
||
|
|
|
||
|
|
While changing and extending the scheduler requires also user-facing API changes, the public API is not part of the discussion of this document.
|
||
|
|
|
||
|
|
## Proposals
|
||
|
|
|
||
|
|
### Proposal 0: Do nothing
|
||
|
|
|
||
|
|
An alternative to changing the scheduling mechanism is to handle QoS control via multiple tenants and multi-tenant querying.
|
||
|
|
|
||
|
|
**Pros:**
|
||
|
|
* Keeps the scheduler as simple as it is now
|
||
|
|
* No development time
|
||
|
|
|
||
|
|
**Cons:**
|
||
|
|
* While that separation into tenants may work for some prospects, it might not be feasible to implement for others.
|
||
|
|
|
||
|
|
### Proposal 1: Add fixed second level to scheduler
|
||
|
|
|
||
|
|
The current scheduler is implemented in a way that it maintains a separate FIFO queue for each tenant. When a request (sub-query) is enqueued, the scheduler puts it into the existing queue for that tenant. If the queue does not exist yet, it creates it first and re-assignes the connected querier workers to the available tenant queues. Each querier worker pulls round-robin from the assigned queues in a loop.
|
||
|
|
|
||
|
|
Now, instead of enqueuing and pulling directly from the per-tenant queue, requests get enqueued in per-user queues and the per-tenant queue pulls round-robin from the user queues that are assigned to the tenant queues.
|
||
|
|
|
||
|
|
**Component diagram:**
|
||
|
|
|
||
|
|

|
||
|
|
|
||
|
|
Like the current implementation, the scheduler enqueues requests based on the `X-Scope-OrgID` header (or equivalent key in the request context), but also takes a second key (such as `X-Scope-UserID`) into account. This ensembles a fixed hierarchy with two levels where the tenant-to-user relation is a one-to-many relation.
|
||
|
|
However, this has the disadvantage that the concept of users (that does not exist yet in Loki) leaks into the scheduler domain.
|
||
|
|
|
||
|
|
**Pros:**
|
||
|
|
* Relatively simple to to implement
|
||
|
|
|
||
|
|
**Cons:**
|
||
|
|
* Not extensible
|
||
|
|
* Leaks domain knowledge
|
||
|
|
|
||
|
|
### Proposal 2: Fully hierarchical scheduler
|
||
|
|
|
||
|
|
This proposal is similar to _Proposal 1_, but with the difference that there are no fixed levels and levels can be nested arbitrarily.
|
||
|
|
|
||
|
|
**Component diagram:**
|
||
|
|
|
||
|
|

|
||
|
|
|
||
|
|
The implementation of the `RequestQueue`, which controls what querier workers are connected to which root queues (aka tenant queues), can be kept as is. However, the concept of tenants and users is dropped and replaced by by a concept of hierarchical actors, which can be represented as a slice of identifiers. Note, this does **not** drop the concept of tenants throughout Loki (represented in the `X-Scope-OrgID` header and/or request context).
|
||
|
|
|
||
|
|
**Example of identifiers:**
|
||
|
|
|
||
|
|
```go
|
||
|
|
actorA := []string{"tenant_a", "user_1"}
|
||
|
|
actorB := []string{"tenant_b", "user_2"}
|
||
|
|
actorC := []string{"tenant_b", "user_3", "service_foo"}
|
||
|
|
actorD := []string{"tenant_b", "user_3", "service_bar"}
|
||
|
|
```
|
||
|
|
|
||
|
|
More generally:
|
||
|
|
|
||
|
|
```go
|
||
|
|
actorN := []string{"L0 Queue", "L1 Queue", "L2 Queue", ... "Ln Queue"}
|
||
|
|
```
|
||
|
|
|
||
|
|
The L0 queue (root queue) needs to be able to handle worker connections and therefore needs additional functionality compared to its leaf queues.
|
||
|
|
|
||
|
|
The following code snippet is meant to show the simplified recursive structure of the queues.
|
||
|
|
|
||
|
|
```go
|
||
|
|
type Request interface{}
|
||
|
|
|
||
|
|
type Queue interface {
|
||
|
|
Deqeue(actor []string) Request
|
||
|
|
Enqueue(r Request, actor []string) error
|
||
|
|
}
|
||
|
|
|
||
|
|
// RequestQueue implements Queue
|
||
|
|
type RequestQueue struct {
|
||
|
|
queriers map[string]*querier
|
||
|
|
rootQueues map[string]*RootQueue
|
||
|
|
}
|
||
|
|
|
||
|
|
// RootQueue implements Queue
|
||
|
|
type RootQueue struct {
|
||
|
|
queriers map[string]*querier
|
||
|
|
leafs map[string]*LeafQueue
|
||
|
|
ch chan Request
|
||
|
|
}
|
||
|
|
|
||
|
|
// LeafQueue implements Queue
|
||
|
|
type LeafQueue struct {
|
||
|
|
leafs map[string]*LeafQueue
|
||
|
|
ch chan Request
|
||
|
|
}
|
||
|
|
```
|
||
|
|
|
||
|
|
**Pros:**
|
||
|
|
* Backwards compatible, because tenant can be identified as `[]string{"tenantID"}`
|
||
|
|
* Queue hierarchy can be extended without changing the scheduler implementation
|
||
|
|
* Implementation does not require knowledge outside of its domain
|
||
|
|
|
||
|
|
**Cons:**
|
||
|
|
* More complex to implement than fixed amount of levels
|
||
|
|
* Each queue comes with memory overhead
|
||
|
|
|
||
|
|
### Proposal 3: Multiple per-tenant sub-queues
|
||
|
|
|
||
|
|
Another option to keep the concept of users out of Loki and still provide some query fairness guarantees would be to simply shard request across multiple sub-queues within a tenant's queue. The shard size could be a per-tenant setting to account for different tenant sizes.
|
||
|
|
|
||
|
|
This is similar to Proposal 1, in the sense of adding another fixed level of sub-queues.
|
||
|
|
However, with the difference, that in this case, a single query request is assigned a random identifier that is hashed. When the query is split, the sub-requests maintain the same hashed identifier. The modulor of the hash defines to which sub-queue of a tenant requests will be enqueued.
|
||
|
|
|
||
|
|
**Pros:**
|
||
|
|
* User agnostic per-request QoS control
|
||
|
|
|
||
|
|
**Cons:**
|
||
|
|
* Requests of individual users can still effect other users
|
||
|
|
* Not extensible
|
||
|
|
|
||
|
|
**Alternative:**
|
||
|
|
|
||
|
|
Sharding on a per-request basis can still be achieved with Proposal 2, by adding the request hash as an additional level in the hierarchy.
|
||
|
|
|
||
|
|
```go
|
||
|
|
actor := []string{"tenant", "user", "request_hash"}
|
||
|
|
```
|
||
|
|
|
||
|
|
## Consensus
|
||
|
|
|
||
|
|
Proposal 2 is going to be implemented.
|