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

Reconcile deleted namespaces out of cluster quota status #12123

Merged
merged 1 commit into from
Dec 3, 2016
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
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
18 changes: 14 additions & 4 deletions pkg/quota/controller/clusterquotamapping/mapper.go
Original file line number Diff line number Diff line change
Expand Up @@ -133,8 +133,13 @@ func (m *clusterQuotaMapper) removeQuota(quotaName string) {
delete(m.requiredQuotaToSelector, quotaName)
delete(m.completedQuotaToSelector, quotaName)
delete(m.quotaToNamespaces, quotaName)
for _, quotas := range m.namespaceToQuota {
quotas.Delete(quotaName)
for namespaceName, quotas := range m.namespaceToQuota {
if quotas.Has(quotaName) {
quotas.Delete(quotaName)
for _, listener := range m.listeners {
listener.RemoveMapping(quotaName, namespaceName)
}
}
}
}

Expand Down Expand Up @@ -176,8 +181,13 @@ func (m *clusterQuotaMapper) removeNamespace(namespaceName string) {
delete(m.requiredNamespaceToLabels, namespaceName)
delete(m.completedNamespaceToLabels, namespaceName)
delete(m.namespaceToQuota, namespaceName)
for _, namespaces := range m.quotaToNamespaces {
namespaces.Delete(namespaceName)
for quotaName, namespaces := range m.quotaToNamespaces {
if namespaces.Has(namespaceName) {
namespaces.Delete(namespaceName)
for _, listener := range m.listeners {
listener.RemoveMapping(quotaName, namespaceName)
}
}
}
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -179,8 +179,19 @@ func (c *ClusterQuotaReconcilationController) calculateAll() {
}

for _, quota := range quotas {
// If we have namespaces we map to, force calculating those namespaces
namespaces, _ := c.clusterQuotaMapper.GetNamespacesFor(quota.Name)
c.forceCalculation(quota.Name, namespaces...)
if len(namespaces) > 0 {
c.forceCalculation(quota.Name, namespaces...)
continue
}

// If the quota status has namespaces when our mapper doesn't think it should,
// add it directly to the queue without any work items
if quota.Status.Namespaces.OrderedKeys().Front() != nil {
c.queue.AddWithData(quota.Name)
continue
}
}
}

Expand Down Expand Up @@ -285,6 +296,17 @@ func (c *ClusterQuotaReconcilationController) syncQuotaForNamespaces(originalQuo
quota.Status.Namespaces.Insert(namespaceName, recalculatedStatus)
}

// Remove any namespaces from quota.status that no longer match.
// Needed because we will never get workitems for namespaces that no longer exist if we missed the delete event (e.g. on startup)
for e := quota.Status.Namespaces.OrderedKeys().Front(); e != nil; e = e.Next() {
namespaceName := e.Value.(string)
namespaceTotals, _ := quota.Status.Namespaces.Get(namespaceName)
if !matchingNamespaceNames.Has(namespaceName) {
quota.Status.Total.Used = utilquota.Subtract(quota.Status.Total.Used, namespaceTotals.Used)
quota.Status.Namespaces.Remove(namespaceName)
}
}

quota.Status.Total.Hard = quota.Spec.Quota.Hard

// if there's no change, no update, return early. NewAggregate returns nil on empty input
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -161,7 +161,7 @@ func TestSyncFunc(t *testing.T) {
expectedRetries: []workItem{},
},
{
name: "update one, remove two, ignore three, fail four",
name: "update one, remove two, ignore three, fail four, remove deleted",
startingQuota: func() *quotaapi.ClusterResourceQuota {
ret := defaultQuota()
ret.Status.Total.Hard = ret.Spec.Quota.Hard
Expand All @@ -178,6 +178,10 @@ func TestSyncFunc(t *testing.T) {
Hard: ret.Spec.Quota.Hard,
Used: kapi.ResourceList{kapi.ResourcePods: resource.MustParse("15")},
})
ret.Status.Namespaces.Insert("deleted", kapi.ResourceQuotaStatus{
Hard: ret.Spec.Quota.Hard,
Used: kapi.ResourceList{kapi.ResourcePods: resource.MustParse("0")},
})
return ret
},
workItems: []workItem{
Expand Down
4 changes: 2 additions & 2 deletions test/cmd/quota.sh
Original file line number Diff line number Diff line change
Expand Up @@ -29,9 +29,9 @@ os::cmd::expect_success 'oc new-project asmail [email protected]'
os::cmd::try_until_text 'oc get appliedclusterresourcequota -n bar --as deads -o name' "for-deads-by-annotation"
os::cmd::try_until_text 'oc get appliedclusterresourcequota -n foo --as deads -o name' "for-deads-by-annotation"
os::cmd::try_until_text 'oc get appliedclusterresourcequota -n asmail --as [email protected] -o name' "for-deads-email-by-annotation"
os::cmd::try_until_text 'oc describe appliedclusterresourcequota/for-deads-by-annotation -n bar --as deads' "secrets.*[1-4][0-9]"

os::cmd::try_until_text 'oc describe appliedclusterresourcequota/for-deads-by-annotation -n bar --as deads' "secrets.*1[0-9]"
os::cmd::expect_success 'oc delete project foo'
os::cmd::try_until_text 'oc get clusterresourcequota/for-deads-by-annotation -o jsonpath="{.status.namespaces[*].Namespace}"' '^bar$'
os::cmd::expect_success 'oc delete project bar'
os::cmd::expect_success 'oc delete project asmail'

Expand Down