mirror of
https://github.com/openfaas/faas.git
synced 2025-06-09 00:36:46 +00:00
Enables publishing to various topics according to annotations on the functions. The function cache is moved up one level so that it can be shared between the scale from zero code and the queue proxy. Unit tests added for new internal methods. Tested e2e with arkade and the newest queue-worker and RC gateway image with two queues and an annotation on one of the functions of com.openfaas.queue. It worked as expected including with multiple namespace support. Signed-off-by: Alex Ellis (OpenFaaS Ltd) <alexellis2@gmail.com>
157 lines
3.5 KiB
Go
157 lines
3.5 KiB
Go
package scaling
|
|
|
|
import (
|
|
"fmt"
|
|
"log"
|
|
"time"
|
|
)
|
|
|
|
// NewFunctionScaler create a new scaler with the specified
|
|
// ScalingConfig
|
|
func NewFunctionScaler(config ScalingConfig, functionCacher FunctionCacher) FunctionScaler {
|
|
return FunctionScaler{
|
|
Cache: functionCacher,
|
|
Config: config,
|
|
}
|
|
}
|
|
|
|
// FunctionScaler scales from zero
|
|
type FunctionScaler struct {
|
|
Cache FunctionCacher
|
|
Config ScalingConfig
|
|
}
|
|
|
|
// FunctionScaleResult holds the result of scaling from zero
|
|
type FunctionScaleResult struct {
|
|
Available bool
|
|
Error error
|
|
Found bool
|
|
Duration time.Duration
|
|
}
|
|
|
|
// Scale scales a function from zero replicas to 1 or the value set in
|
|
// the minimum replicas metadata
|
|
func (f *FunctionScaler) Scale(functionName, namespace string) FunctionScaleResult {
|
|
start := time.Now()
|
|
|
|
if cachedResponse, hit := f.Cache.Get(functionName, namespace); hit &&
|
|
cachedResponse.AvailableReplicas > 0 {
|
|
return FunctionScaleResult{
|
|
Error: nil,
|
|
Available: true,
|
|
Found: true,
|
|
Duration: time.Since(start),
|
|
}
|
|
}
|
|
|
|
queryResponse, err := f.Config.ServiceQuery.GetReplicas(functionName, namespace)
|
|
|
|
if err != nil {
|
|
return FunctionScaleResult{
|
|
Error: err,
|
|
Available: false,
|
|
Found: false,
|
|
Duration: time.Since(start),
|
|
}
|
|
}
|
|
|
|
f.Cache.Set(functionName, namespace, queryResponse)
|
|
|
|
if queryResponse.AvailableReplicas == 0 {
|
|
minReplicas := uint64(1)
|
|
if queryResponse.MinReplicas > 0 {
|
|
minReplicas = queryResponse.MinReplicas
|
|
}
|
|
|
|
scaleResult := backoff(func(attempt int) error {
|
|
queryResponse, err := f.Config.ServiceQuery.GetReplicas(functionName, namespace)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
f.Cache.Set(functionName, namespace, queryResponse)
|
|
|
|
if queryResponse.Replicas > 0 {
|
|
return nil
|
|
}
|
|
|
|
log.Printf("[Scale %d] function=%s 0 => %d requested", attempt, functionName, minReplicas)
|
|
setScaleErr := f.Config.ServiceQuery.SetReplicas(functionName, namespace, minReplicas)
|
|
if setScaleErr != nil {
|
|
return fmt.Errorf("unable to scale function [%s], err: %s", functionName, setScaleErr)
|
|
}
|
|
|
|
return nil
|
|
|
|
}, int(f.Config.SetScaleRetries), f.Config.FunctionPollInterval)
|
|
|
|
if scaleResult != nil {
|
|
return FunctionScaleResult{
|
|
Error: scaleResult,
|
|
Available: false,
|
|
Found: true,
|
|
Duration: time.Since(start),
|
|
}
|
|
}
|
|
|
|
for i := 0; i < int(f.Config.MaxPollCount); i++ {
|
|
queryResponse, err := f.Config.ServiceQuery.GetReplicas(functionName, namespace)
|
|
if err == nil {
|
|
f.Cache.Set(functionName, namespace, queryResponse)
|
|
}
|
|
totalTime := time.Since(start)
|
|
|
|
if err != nil {
|
|
return FunctionScaleResult{
|
|
Error: err,
|
|
Available: false,
|
|
Found: true,
|
|
Duration: totalTime,
|
|
}
|
|
}
|
|
|
|
if queryResponse.AvailableReplicas > 0 {
|
|
|
|
log.Printf("[Scale] function=%s 0 => %d successful - %fs",
|
|
functionName, queryResponse.AvailableReplicas, totalTime.Seconds())
|
|
|
|
return FunctionScaleResult{
|
|
Error: nil,
|
|
Available: true,
|
|
Found: true,
|
|
Duration: totalTime,
|
|
}
|
|
}
|
|
|
|
time.Sleep(f.Config.FunctionPollInterval)
|
|
}
|
|
}
|
|
|
|
return FunctionScaleResult{
|
|
Error: nil,
|
|
Available: true,
|
|
Found: true,
|
|
Duration: time.Since(start),
|
|
}
|
|
}
|
|
|
|
type routine func(attempt int) error
|
|
|
|
func backoff(r routine, attempts int, interval time.Duration) error {
|
|
var err error
|
|
|
|
for i := 0; i < attempts; i++ {
|
|
res := r(i)
|
|
if res != nil {
|
|
err = res
|
|
|
|
log.Printf("Attempt: %d, had error: %s\n", i, res)
|
|
} else {
|
|
err = nil
|
|
break
|
|
}
|
|
time.Sleep(interval)
|
|
}
|
|
return err
|
|
}
|