Skip to content
New issue

Have a question about this project? # for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “#”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? # to your account

broken ScaledObjects scaling #64

Open
jonathan-mayer opened this issue Oct 16, 2024 · 2 comments · May be fixed by #78
Open

broken ScaledObjects scaling #64

jonathan-mayer opened this issue Oct 16, 2024 · 2 comments · May be fixed by #78
Labels
bug Something isn't working

Comments

@jonathan-mayer
Copy link
Member

Issue

Workloads managed by ScaledObjects are currently also scaled by the Downscaler. This results in the ScaledObject and the downscaler fighting to downscale those workloads to (possibly) different Replicas.

Further details

Bug was found here: caas-team/py-kube-downscaler#96
Bug was fixed in py-kube-downscaler here: caas-team/py-kube-downscaler#100

Proposal

Implement some way to avoid this.
A possible way would be to automatically exclude the workloads managed by ScaledObjects from being scaled by the Downscaler.

Who can address the issue

Other links/references

@jonathan-mayer jonathan-mayer added the bug Something isn't working label Oct 16, 2024
@samuel-esp
Copy link
Contributor

samuel-esp commented Oct 16, 2024

A possible way would be to automatically exclude the workloads managed by ScaledObjects from being scaled by the Downscaler.

My suggestion to implement the approach you said would be something like:

// GetWorkloads gets all workloads of the specified resources for the specified namespaces
func (c client) GetWorkloads(namespaces []string, resourceTypes []string, ctx context.Context) ([]scalable.Workload, error) {
	var results []scalable.Workload
	if namespaces == nil {
		namespaces = append(namespaces, "")
	}
	for _, namespace := range namespaces {
		for _, resourceType := range resourceTypes {
			slog.Debug("getting workloads from resource type", "resourceType", resourceType)
			getWorkloads, ok := scalable.GetWorkloads[strings.ToLower(resourceType)]
			if !ok {
				return nil, errResourceNotSupported
			}
			workloads, err := getWorkloads(namespace, c.clientsets, ctx)
			if err != nil {
				return nil, fmt.Errorf("failed to get workloads: %w", err)
			}
			results = append(results, workloads...)
		}
	}

	return results, nil
}
  • Insert a conditional for this case:
    • When ScaledObjects are retrieved, iterate over each object and insert the Deployment/StatefulSet reference (name) inside an HashMap
  • Insert a conditional for this case:
    • When Deployments are retrieved, iterate over each object. They are not appended to the results list if the name is inside the HashMap
    • When StatefulSets are retrieved, iterate over each object. They are not appended to the results list if the name is inside the HashMap

This approach requires the ScaledObjects to be processed before Deployments/StatefulSets

Things to consider: clarity of code (right now is perfect), performance (should still be 0(N), but we will iterate many times inside lists that could be pretty big, especially the Deployment list)

@jonathan-mayer
Copy link
Member Author

jonathan-mayer commented Oct 16, 2024

I'm not sure if putting all this in the GetWorkloads function is right. I will have a look on how to implement it next week as I am still out of office for the rest of this week.

# for free to join this conversation on GitHub. Already have an account? # to comment
Labels
bug Something isn't working
Projects
None yet
Development

Successfully merging a pull request may close this issue.

2 participants