Skip to content
New issue

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

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

Already on GitHub? Sign in to your account

feat: refactor to maintain policy state in-memory #49

Open
wants to merge 25 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 10 commits
Commits
Show all changes
25 commits
Select commit Hold shift + click to select a range
75b3f7d
maintain state of a placement policy in-memory rather than use pod an…
cmaclaughlin Jan 19, 2022
67eb90e
attempt to resolve issues
cmaclaughlin Jan 21, 2022
f457f74
initial PR changes
cmaclaughlin Jan 24, 2022
6d0aea4
fix compilation error
cmaclaughlin Jan 24, 2022
cba7a0c
Update pkg/plugins/placementpolicy/core/core.go
cmaclaughlin Feb 8, 2022
8059236
Update pkg/plugins/placementpolicy/core/policy.go
cmaclaughlin Feb 8, 2022
138af22
Merge branch 'Azure:main' into inmemory-policy-state
cmaclaughlin Feb 8, 2022
1f17889
changes per initial PR comments
cmaclaughlin Feb 8, 2022
744c441
Merge branch 'Azure:main' into inmemory-policy-state
cmaclaughlin Feb 22, 2022
f104ce5
adding comments for clarity
cmaclaughlin Feb 22, 2022
da2c4e6
Update pkg/plugins/placementpolicy/placementpolicy.go
cmaclaughlin Feb 25, 2022
8f418f9
Update pkg/plugins/placementpolicy/placementpolicy.go
cmaclaughlin Feb 25, 2022
db96728
Update pkg/plugins/placementpolicy/core/policy.go
cmaclaughlin Feb 25, 2022
cedd819
Update pkg/plugins/placementpolicy/core/policy.go
cmaclaughlin Feb 25, 2022
3adaed7
Update pkg/plugins/placementpolicy/core/policy.go
cmaclaughlin Feb 25, 2022
cd95269
Update pkg/plugins/placementpolicy/core/policy.go
cmaclaughlin Feb 25, 2022
0d482b2
Update pkg/plugins/placementpolicy/core/policy.go
cmaclaughlin Feb 25, 2022
34b0d77
Update pkg/plugins/placementpolicy/core/policy.go
cmaclaughlin Feb 25, 2022
c2fb51e
Update pkg/plugins/placementpolicy/core/policy.go
cmaclaughlin Feb 25, 2022
5f93723
Update pkg/plugins/placementpolicy/placementpolicy.go
cmaclaughlin Feb 25, 2022
e346776
initial changes for next round of feedback
cmaclaughlin Feb 25, 2022
ef85acc
Additional changes per PR feedback
cmaclaughlin Feb 25, 2022
934623e
remove stage dependency
cmaclaughlin Feb 25, 2022
d2fc5ef
final cleanup/fixes
cmaclaughlin Feb 25, 2022
0d7a9f3
Merge branch 'Azure:main' into inmemory-policy-state
cmaclaughlin May 5, 2022
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion .github/workflows/workflow.yml
Original file line number Diff line number Diff line change
Expand Up @@ -54,7 +54,7 @@ jobs:
e2e-test:
name: "E2E Test"
runs-on: ubuntu-latest
timeout-minutes: 10
timeout-minutes: 15
steps:
- name: Set up Go 1.17
uses: actions/setup-go@v2
Expand Down
146 changes: 98 additions & 48 deletions pkg/plugins/placementpolicy/core/core.go
Original file line number Diff line number Diff line change
Expand Up @@ -5,52 +5,36 @@ import (
"sort"

"github.com/Azure/placement-policy-scheduler-plugins/apis/v1alpha1"
ppclientset "github.com/Azure/placement-policy-scheduler-plugins/pkg/client/clientset/versioned"
ppinformers "github.com/Azure/placement-policy-scheduler-plugins/pkg/client/informers/externalversions/apis/v1alpha1"
pplisters "github.com/Azure/placement-policy-scheduler-plugins/pkg/client/listers/apis/v1alpha1"
"github.com/Azure/placement-policy-scheduler-plugins/pkg/utils"

corev1 "k8s.io/api/core/v1"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/labels"
"k8s.io/client-go/kubernetes"
corelisters "k8s.io/client-go/listers/core/v1"
"k8s.io/kubernetes/pkg/scheduler/framework"
)

// Manager defines the interfaces for PlacementPolicy management.
type Manager interface {
GetPlacementPolicyForPod(context.Context, *corev1.Pod) (*v1alpha1.PlacementPolicy, error)
GetPodsWithLabels(context.Context, map[string]string) ([]*corev1.Pod, error)
AnnotatePod(context.Context, *corev1.Pod, *v1alpha1.PlacementPolicy, bool) (*corev1.Pod, error)
GetPlacementPolicy(context.Context, string, string) (*v1alpha1.PlacementPolicy, error)
GetPolicyInfo(*v1alpha1.PlacementPolicy) *PolicyInfo
MatchPod(context.Context, *corev1.Pod, *PolicyInfo) (*PolicyInfo, error)
AddPod(context.Context, *corev1.Pod, *PolicyInfo) (*PolicyInfo, error)
RemovePod(*corev1.Pod) error
}

type PlacementPolicyManager struct {
// client is a clientset for the kube API server.
client kubernetes.Interface
// client is a placementPolicy client
ppClient ppclientset.Interface
// podLister is pod lister
podLister corelisters.PodLister
// snapshotSharedLister is pod shared list
snapshotSharedLister framework.SharedLister
// ppLister is placementPolicy lister
ppLister pplisters.PlacementPolicyLister
// available policies by namespace
policies PolicyInfos
}

func NewPlacementPolicyManager(
client kubernetes.Interface,
ppClient ppclientset.Interface,
snapshotSharedLister framework.SharedLister,
ppInformer ppinformers.PlacementPolicyInformer,
podLister corelisters.PodLister) *PlacementPolicyManager {
ppInformer ppinformers.PlacementPolicyInformer) *PlacementPolicyManager {
return &PlacementPolicyManager{
client: client,
ppClient: ppClient,
snapshotSharedLister: snapshotSharedLister,
ppLister: ppInformer.Lister(),
podLister: podLister,
ppLister: ppInformer.Lister(),
policies: NewPolicyInfos(),
}
}

Expand All @@ -73,38 +57,104 @@ func (m *PlacementPolicyManager) GetPlacementPolicyForPod(ctx context.Context, p
return ppList[0], nil
}

func (m *PlacementPolicyManager) GetPodsWithLabels(ctx context.Context, podLabels map[string]string) ([]*corev1.Pod, error) {
return m.podLister.List(labels.Set(podLabels).AsSelector())
func (m *PlacementPolicyManager) filterPlacementPolicyList(ppList []*v1alpha1.PlacementPolicy, pod *corev1.Pod) []*v1alpha1.PlacementPolicy {
var filteredPPList []*v1alpha1.PlacementPolicy
for _, pp := range ppList {
labels := pp.Spec.PodSelector.MatchLabels
if utils.HasMatchingLabels(pod.Labels, labels) {
filteredPPList = append(filteredPPList, pp)
}
}
return filteredPPList
}

// AnnotatePod annotates the pod with the placement policy.
func (m *PlacementPolicyManager) AnnotatePod(ctx context.Context, pod *corev1.Pod, pp *v1alpha1.PlacementPolicy, preferredNodeWithMatchingLabels bool) (*corev1.Pod, error) {
annotations := map[string]string{}
if pod.Annotations != nil {
annotations = pod.Annotations
}
// GetPolicyInfo returns the PolicyInfo held in-memory for the given placement policy
func (m *PlacementPolicyManager) GetPolicyInfo(pp *v1alpha1.PlacementPolicy) *PolicyInfo {
policyNamespace := pp.Namespace
policyName := pp.Name
corePolicy := pp.Spec.Policy
action := corePolicy.Action
targetSize := corePolicy.TargetSize

return m.policies.GetPolicy(policyNamespace, policyName, action, targetSize)
}

preference := "false"
if preferredNodeWithMatchingLabels {
preference = "true"
type PodAction int16

const (
Match PodAction = iota
Add
Remove
)

// MatchPod attaches the provided pod to the provided PolicyInfo as a match
func (m *PlacementPolicyManager) MatchPod(ctx context.Context, pod *corev1.Pod, policy *PolicyInfo) (*PolicyInfo, error) {
cmaclaughlin marked this conversation as resolved.
Show resolved Hide resolved
err := policy.addMatch(pod)
if err != nil {
return nil, err
}
annotations[v1alpha1.PlacementPolicyAnnotationKey] = pp.Name
annotations[v1alpha1.PlacementPolicyPreferenceAnnotationKey] = preference
pod.Annotations = annotations
return m.client.CoreV1().Pods(pod.Namespace).Update(ctx, pod, metav1.UpdateOptions{})

m.updatePolicies(policy, Match)
return policy, nil
}

func (m *PlacementPolicyManager) GetPlacementPolicy(ctx context.Context, namespace, name string) (*v1alpha1.PlacementPolicy, error) {
return m.ppLister.PlacementPolicies(namespace).Get(name)
// AddPod adds the provided pod to the provided PolicyInfo and calculates whether target was met
func (m *PlacementPolicyManager) AddPod(ctx context.Context, pod *corev1.Pod, policy *PolicyInfo) (*PolicyInfo, error) {
err := policy.addPodIfNotPresent(pod)
if err != nil {
return nil, err
}

m.updatePolicies(policy, Add)
return policy, nil
}

func (m *PlacementPolicyManager) filterPlacementPolicyList(ppList []*v1alpha1.PlacementPolicy, pod *corev1.Pod) []*v1alpha1.PlacementPolicy {
var filteredPPList []*v1alpha1.PlacementPolicy
// RemovePod removes the provided pod from PolicyInfo held in-memory and re-calculates policy's status (if applicable)
func (m *PlacementPolicyManager) RemovePod(pod *corev1.Pod) error {
key, err := framework.GetPodKey(pod)
if err != nil {
return err
}

podNamespace := pod.Namespace
ppList := m.policies.GetPoliciesByNamespace(podNamespace)

cmaclaughlin marked this conversation as resolved.
Show resolved Hide resolved
if ppList == nil {
return nil
}

var matchingPolicy *PolicyInfo

for _, pp := range ppList {
labels := pp.Spec.PodSelector.MatchLabels
if utils.HasMatchingLabels(pod.Labels, labels) {
filteredPPList = append(filteredPPList, pp)
if pp.PodQualifiesForPolicy(key) {
matchingPolicy = pp
break
}
}
return filteredPPList

if matchingPolicy != nil {
err := matchingPolicy.removePodIfPresent(pod)
if err != nil {
return err
}

m.updatePolicies(matchingPolicy, Remove)
}
return nil
}

func (m *PlacementPolicyManager) updatePolicies(policy *PolicyInfo, action PodAction) {
namespace := policy.Namespace

if action == Remove {
qualifyingCount := len(policy.qualifiedPods)

if qualifyingCount == 0 {
name := policy.Name
m.policies.RemovePolicy(namespace, name)
return
}
}

m.policies.UpdatePolicy(namespace, policy)
}
Loading