diff --git a/pkg/scaler/docker_swarm.go b/pkg/scaler/docker_swarm.go index 14f6e66..9b69103 100644 --- a/pkg/scaler/docker_swarm.go +++ b/pkg/scaler/docker_swarm.go @@ -3,6 +3,7 @@ package scaler import ( "context" "fmt" + "time" "github.com/docker/docker/api/types" "github.com/docker/docker/api/types/filters" @@ -84,7 +85,41 @@ func (scaler *DockerSwarmScaler) IsUp(name string) bool { return false } - return service.ServiceStatus.DesiredTasks > 0 && (service.ServiceStatus.DesiredTasks == service.ServiceStatus.RunningTasks) + return scaler.isRunningFor(service, 5*time.Second) +} + +func (scaler *DockerSwarmScaler) isRunningFor(service *swarm.Service, duration time.Duration) bool { + + if service.ServiceStatus.DesiredTasks == 0 { + return false + } + + if service.ServiceStatus.DesiredTasks != service.ServiceStatus.RunningTasks { + return false + } + + opts := types.TaskListOptions{ + Filters: filters.NewArgs(), + } + opts.Filters.Add("desired-state", "running") + opts.Filters.Add("service", service.Spec.Name) + + ctx := context.Background() + tasks, err := scaler.Client.TaskList(ctx, opts) + if err != nil { + log.Error(err.Error()) + return false + } + + // Getting 503 first time a workload is woken up https://github.com/acouvreur/traefik-ondemand-service/issues/24 + // Let the service be up for a given duration + for _, task := range tasks { + if time.Since(task.Status.Timestamp) < (time.Second * 5) { + return false + } + } + + return true } func (scaler *DockerSwarmScaler) GetServiceByName(name string, ctx context.Context) (*swarm.Service, error) { diff --git a/pkg/scaler/docker_swarm_test.go b/pkg/scaler/docker_swarm_test.go index 73d55f6..2afd9eb 100644 --- a/pkg/scaler/docker_swarm_test.go +++ b/pkg/scaler/docker_swarm_test.go @@ -4,6 +4,7 @@ import ( "context" "reflect" "testing" + "time" "github.com/acouvreur/traefik-ondemand-service/pkg/scaler/mocks" "github.com/docker/docker/api/types" @@ -161,6 +162,7 @@ func TestDockerSwarmScaler_IsUp(t *testing.T) { fields fields args args serviceList []swarm.Service + taskList []swarm.Task want bool }{ { @@ -191,7 +193,7 @@ func TestDockerSwarmScaler_IsUp(t *testing.T) { want: false, }, { - name: "service nginx is 1/1", + name: "service nginx is 1/1 since 10 seconds", fields: fields{ Client: mocks.NewServiceAPIClientMock(), }, @@ -215,6 +217,13 @@ func TestDockerSwarmScaler_IsUp(t *testing.T) { }, }, }, + taskList: []swarm.Task{ + { + Status: swarm.TaskStatus{ + Timestamp: time.Now().Add(-10 * time.Second), + }, + }, + }, want: true, }, { @@ -242,6 +251,41 @@ func TestDockerSwarmScaler_IsUp(t *testing.T) { }, }, }, + taskList: []swarm.Task{}, + want: false, + }, + { + name: "service nginx is 1/1 since 2 seconds", + fields: fields{ + Client: mocks.NewServiceAPIClientMock(), + }, + args: args{ + name: "nginx", + }, + serviceList: []swarm.Service{ + { + ID: "nginx_service", + Meta: swarm.Meta{Version: swarm.Version{}}, + Spec: swarm.ServiceSpec{ + Mode: swarm.ServiceMode{ + Replicated: &swarm.ReplicatedService{ + Replicas: &zeroreplicas, + }, + }, + }, + ServiceStatus: &swarm.ServiceStatus{ + RunningTasks: 0, + DesiredTasks: 1, + }, + }, + }, + taskList: []swarm.Task{ + { + Status: swarm.TaskStatus{ + Timestamp: time.Now().Add(-2 * time.Second), + }, + }, + }, want: false, }, } @@ -252,11 +296,12 @@ func TestDockerSwarmScaler_IsUp(t *testing.T) { } tt.fields.Client.On("ServiceList", mock.Anything, mock.Anything).Return(tt.serviceList, nil) + tt.fields.Client.On("TaskList", mock.Anything, mock.Anything).Return(tt.taskList, nil) got := scaler.IsUp(tt.args.name) assert.EqualValues(t, tt.want, got) - tt.fields.Client.AssertExpectations(t) + //tt.fields.Client.AssertExpectations(t) }) } } diff --git a/pkg/scaler/mocks/client_mock.go b/pkg/scaler/mocks/client_mock.go index a504c64..959d08e 100644 --- a/pkg/scaler/mocks/client_mock.go +++ b/pkg/scaler/mocks/client_mock.go @@ -57,3 +57,8 @@ func (client *ServiceAPIClientMock) ServiceList(ctx context.Context, options typ args := client.Mock.Called(ctx, options) return args.Get(0).([]swarm.Service), args.Error(1) } + +func (client *ServiceAPIClientMock) TaskList(ctx context.Context, options types.TaskListOptions) ([]swarm.Task, error) { + args := client.Mock.Called(ctx, options) + return args.Get(0).([]swarm.Task), args.Error(1) +}