Skip to content

Commit

Permalink
Add a user document for the scheduling framework
Browse files Browse the repository at this point in the history
  • Loading branch information
bsalamat committed May 17, 2019
1 parent a1f811d commit 0e07174
Show file tree
Hide file tree
Showing 2 changed files with 230 additions and 0 deletions.
230 changes: 230 additions & 0 deletions content/en/docs/concepts/configuration/scheduling-framework.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,230 @@
---
reviewers:
- ahg-g
title: Scheduling Framework
content_template: templates/concept
weight: 70
---

{{% capture overview %}}

{{< feature-state for_k8s_version="1.15" state="alpha" >}}

The scheduling framework is a new plugable architecture for Kubernetes Scheduler
that makes scheduler customizations easy. It adds a new set of "plugin" APIs to
the existing scheduler. Plugins are compiled into the scheduler, and these APIs
allow many scheduling features to be implemented as plugins, while keeping the
scheduling "core" simple and maintainable. Refer to the [design proposal of the
scheduling framework][kep] for more technical information on the design of the
framework.

[kep]: (https://github.com/kubernetes/enhancements/blob/master/keps/sig-scheduling/20180409-scheduling-framework.md)

{{% /capture %}}

{{% capture body %}}

# Framework workflow

The Scheduling Framework defines a few extension points. Scheduler plugins
register to be invoked at one or more extension points. Some of these plugins
can change the scheduling decisions and some are informational only.

Each attempt to schedule one pod is split into two phases, the **scheduling
cycle** and the **binding cycle**.

## Scheduling Cycle & Binding Cycle

The scheduling cycle selects a node for the pod, and the binding cycle applies
that decision to the cluster. Together, a scheduling cycle and binding cycle are
referred to as a "scheduling context".

Scheduling cycles are run serially, while binding cycles may run concurrently.

A scheduling cycle or binding cycle can be aborted if the pod is determined to
be unschedulable or if there is an internal error. The pod will be returned to
the queue and retried.

## Extension points

The following picture shows the scheduling context of a pod and the extension
points that the scheduling framework exposes. In this picture "Filter" is
equivalent to "Predicate" and "Scoring" is equivalent to "Priority function".

One plugin may register at multiple extension points to perform more complex or
stateful tasks.

{{< figure src="/images/docs/scheduling-framework.png" title="scheduling framework extension points" >}}

### Queue sort

These plugins are used to sort pods in the scheduling queue. A queue sort plugin
essentially will provide a "less(pod1, pod2)" function. Only one queue sort
plugin may be enabled at a time.

### Pre-filter

These plugins are used to pre-process info about the pod, or to check certain
conditions that the cluster or the pod must meet. If a pre-filter plugin returns
an error, the scheduling cycle is aborted.

### Filter

These plugins are used to filter out nodes that cannot run the Pod. For each
node, the scheduler will call filter plugins in their configured order. If any
filter plugin marks the node as infeasible, the remaining plugins will not be
called for that node. Nodes may be evaluated concurrently.

### Post-filter

This is an informational extension point. Plugins will be called with a list of
nodes that passed the filtering phase. A plugin may use this data to update
internal state or to generate logs/metrics.

**Note:** Plugins wishing to perform "pre-scoring" work should use the
post-filter extension point.

### Scoring

These plugins are used to rank nodes that have passed the filtering phase. The
scheduler will call each scoring plugin for each node. There will be a well
defined range of integers representing the minimum and maximum scores. After the
[normalize scoring](#normalize-scoring) phase, the scheduler will combine node
scores from all plugins according to the configured plugin weights.

### Normalize scoring

These plugins are used to modify scores before the scheduler computes a final
ranking of Nodes. A plugin that registers for this extension point will be
called with the [scoring](#scoring) results from the same plugin. This is called
once per plugin per scheduling cycle.

For example, suppose a plugin `BlinkingLightScorer` ranks Nodes based on how
many blinking lights they have.

```go
func ScoreNode(_ *v1.Pod, n *v1.Node) (int, error) {
return getBlinkingLightCount(n)
}
```

However, the maximum count of blinking lights may be small compared to
`NodeScoreMax`. To fix this, `BlinkingLightScorer` should also register for this
extension point.

```go
func NormalizeScores(scores map[string]int) {
highest := 0
for _, score := range scores {
highest = max(highest, score)
}
for node, score := range scores {
scores[node] = score*NodeScoreMax/highest
}
}
```

If any normalize-scoring plugin returns an error, the scheduling cycle is
aborted.

**Note:** Plugins wishing to perform "pre-reserve" work should use the
normalize-scoring extension point.

### Reserve

This is an informational extension point. Plugins which maintain runtime state
(aka "stateful plugins") should use this extension point to be notified by the
scheduler when resources on a node are being reserved for a given Pod. This
happens before the scheduler actually binds the pod to the Node, and it exists
to prevent race conditions while the scheduler waits for the bind to succeed.

This is the last step in a scheduling cycle. Once a pod is in the reserved
state, it will either trigger [Un-reserve](#un-reserve) plugins (on failure) or
[Post-bind](#post-bind) plugins (on success) at the end of the binding cycle.

*Note: This concept used to be referred to as "assume".*

### Permit

These plugins are used to prevent or delay the binding of a Pod. A permit plugin
can do one of three things.

1. **approve** \
Once all permit plugins approve a pod, it is sent for binding.

1. **deny** \
If any permit plugin denies a pod, it is returned to the scheduling queue.
This will trigger [Un-reserve](#un-reserve) plugins.

1. **wait** (with a timeout) \
If a permit plugin returns "wait", then the pod is kept in the permit phase
until a [plugin approves it](#frameworkhandle). If a timeout occurs, **wait**
becomes **deny** and the pod is returned to the scheduling queue, triggering
[un-reserve](#un-reserve) plugins.

**Approving a pod binding**

While any plugin can receive the list of reserved pods from the cache and
approve them (see [`FrameworkHandle`](#frameworkhandle)) we expect only the permit
plugins to approve binding of reserved Pods that are in "waiting" state. Once a
pod is approved, it is sent to the pre-bind phase.

### Pre-bind

These plugins are used to perform any work required before a pod is bound. For
example, a pre-bind plugin may provision a network volume and mount it on the
target node before allowing the pod to run there.

If any pre-bind plugin returns an error, the pod is [rejected](#un-reserve) and
returned to the scheduling queue.

### Bind

These plugins are used to bind a pod to a Node. Bind plugins will not be called
until all pre-bind plugins have completed. Each bind plugin is called in the
configured order. A bind plugin may choose whether or not to handle the given
Pod. If a bind plugin chooses to handle a Pod, **the remaining bind plugins are
skipped**.

### Post-bind

This is an informational extension point. Post-bind plugins are called after a
pod is successfully bound. This is the end of a binding cycle, and can be used
to clean up associated resources.

### Unreserve

This is an informational extension point. If a pod was reserved and then
rejected in a later phase, then unreserve plugins will be notified. Unreserve
plugins should clean up state associated with the reserved Pod.

Plugins that use this extension point usually should also use
[Reserve](#reserve).

## Plugin API

There are two steps to the plugin API. First, plugins must register and get
configured, then they use the extension point interfaces. Extension point
interfaces have the following form.

```go
type Plugin interface {
Name() string
}

type QueueSortPlugin interface {
Plugin
Less(*v1.Pod, *v1.Pod) bool
}

type PreFilterPlugin interface {
Plugin
PreFilter(PluginContext, *v1.Pod) error
}

// ...
```

# Plugin Configuration

{{% /capture %}}
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.

0 comments on commit 0e07174

Please sign in to comment.