Skip to content

Commit 2399194

Browse files
committed
clean up sorts
1 parent faa564f commit 2399194

14 files changed

+63
-105
lines changed

go.mod

+1-1
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@ go 1.18
55
require (
66
github.com/go-resty/resty/v2 v2.5.0
77
github.com/jstemmer/go-junit-report v0.9.1
8-
github.com/mattfenwick/collections v0.1.0
8+
github.com/mattfenwick/collections v0.1.5
99
github.com/olekukonko/tablewriter v0.0.4
1010
github.com/onsi/ginkgo/v2 v2.1.4
1111
github.com/onsi/gomega v1.19.0

go.sum

+2-2
Original file line numberDiff line numberDiff line change
@@ -241,8 +241,8 @@ github.com/mailru/easyjson v0.0.0-20190614124828-94de47d64c63/go.mod h1:C1wdFJiN
241241
github.com/mailru/easyjson v0.0.0-20190626092158-b2ccc519800e/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc=
242242
github.com/mailru/easyjson v0.7.6 h1:8yTIVnZgCoiM1TgqoeTl+LfU5Jg6/xL3QhGQnimLYnA=
243243
github.com/mailru/easyjson v0.7.6/go.mod h1:xzfreul335JAWq5oZzymOObrkdz5UnU4kGfJJLY9Nlc=
244-
github.com/mattfenwick/collections v0.1.0 h1:4wN8xVs5gzU8amlRGeQNbDLTQvPsBRi79ZxiSZJBbVY=
245-
github.com/mattfenwick/collections v0.1.0/go.mod h1:6JmZGWNsbA4g6OzJQGAOTalNWMPghEOl1q4abMZAJSQ=
244+
github.com/mattfenwick/collections v0.1.5 h1:uffG2TZh2hdiiB9i6v/lJ2xck9IIIBA6FokctWiK5QM=
245+
github.com/mattfenwick/collections v0.1.5/go.mod h1:6JmZGWNsbA4g6OzJQGAOTalNWMPghEOl1q4abMZAJSQ=
246246
github.com/mattn/go-runewidth v0.0.7 h1:Ei8KR0497xHyKJPAv59M1dkC+rOZCMBJ+t3fZ+twI54=
247247
github.com/mattn/go-runewidth v0.0.7/go.mod h1:H031xJmbD/WCDINGzjvQ9THkh0rPKHF+m2gUSrubnMI=
248248
github.com/matttproud/golang_protobuf_extensions v1.0.1/go.mod h1:D8He9yQNgCq6Z5Ld7szi9bcBfOoFv/3dc6xSMkL2PC0=

pkg/connectivity/printer.go

+6-10
Original file line numberDiff line numberDiff line change
@@ -2,12 +2,10 @@ package connectivity
22

33
import (
44
"fmt"
5-
"github.com/mattfenwick/collections/pkg/builtins"
65
"github.com/mattfenwick/collections/pkg/slices"
76
log "github.com/sirupsen/logrus"
87
"golang.org/x/exp/maps"
98
"math"
10-
"sort"
119
"strings"
1210

1311
"github.com/mattfenwick/cyclonus/pkg/generator"
@@ -75,11 +73,11 @@ func (m *markdownRow) GetResult() string {
7573
}
7674

7775
func (t *Printer) printMarkdownFeatureTable(primaryCounts map[string]map[bool]int, tagCounts map[string]map[string]map[bool]int) string {
78-
primaries := slices.SortBy(builtins.CompareOrdered[string], maps.Keys(tagCounts))
76+
primaries := slices.Sort(maps.Keys(tagCounts))
7977

8078
var rows []*markdownRow
8179
for _, primary := range primaries {
82-
subs := slices.SortBy(builtins.CompareOrdered[string], maps.Keys(tagCounts[primary]))
80+
subs := slices.Sort(maps.Keys(tagCounts[primary]))
8381
rows = append(rows, &markdownRow{Name: primary, IsPrimary: true, Pass: primaryCounts[primary][true], Fail: primaryCounts[primary][false]})
8482
for _, sub := range subs {
8583
counts := tagCounts[primary][sub]
@@ -120,7 +118,7 @@ type passFailRow struct {
120118
Failed int
121119
}
122120

123-
func (p *passFailRow) PassedPercentage() float64 {
121+
func PassedPercentage(p *passFailRow) float64 {
124122
return percentage(p.Passed, p.Passed+p.Failed)
125123
}
126124

@@ -140,16 +138,14 @@ func passFailTable(caption string, passFailCounts map[string]map[bool]int, passe
140138
Failed: passFailCounts[feature][false],
141139
})
142140
}
141+
rows = slices.SortOn(PassedPercentage, rows)
143142

144-
sort.Slice(rows, func(i, j int) bool {
145-
return rows[i].PassedPercentage() < rows[j].PassedPercentage()
146-
})
147143
if passedTotal != nil || failedTotal != nil {
148144
rows = append(rows, &passFailRow{Feature: "Total", Passed: *passedTotal, Failed: *failedTotal})
149145
}
150146

151147
for _, row := range rows {
152-
table.Append([]string{row.Feature, intToString(row.Passed), intToString(row.Failed), fmt.Sprintf("%.0f", row.PassedPercentage())})
148+
table.Append([]string{row.Feature, intToString(row.Passed), intToString(row.Failed), fmt.Sprintf("%.0f", PassedPercentage(row))})
153149
}
154150

155151
table.Render()
@@ -174,7 +170,7 @@ func protocolPassFailTable(protocolCounts map[v1.Protocol]map[Comparison]int) st
174170
}
175171

176172
for _, row := range rows {
177-
table.Append([]string{row.Feature, intToString(row.Passed), intToString(row.Failed), fmt.Sprintf("%.0f", row.PassedPercentage())})
173+
table.Append([]string{row.Feature, intToString(row.Passed), intToString(row.Failed), fmt.Sprintf("%.0f", PassedPercentage(row))})
178174
}
179175

180176
table.Render()

pkg/connectivity/probe/resource-printer.go

+3-9
Original file line numberDiff line numberDiff line change
@@ -2,12 +2,10 @@ package probe
22

33
import (
44
"fmt"
5-
"github.com/mattfenwick/collections/pkg/builtins"
65
"github.com/mattfenwick/collections/pkg/slices"
76
"github.com/olekukonko/tablewriter"
87
"github.com/pkg/errors"
98
"golang.org/x/exp/maps"
10-
"sort"
119
"strings"
1210
)
1311

@@ -19,24 +17,20 @@ func (r *Resources) RenderTable() string {
1917
table.SetAutoMergeCells(true)
2018
table.SetRowLine(true)
2119

22-
var nsSlice []string
2320
nsToPod := map[string][]*Pod{}
2421
for _, pod := range r.Pods {
2522
ns := pod.Namespace
2623
if _, ok := nsToPod[ns]; !ok {
2724
nsToPod[ns] = []*Pod{}
28-
nsSlice = append(nsSlice, ns)
2925
}
3026
if _, ok := r.Namespaces[ns]; !ok {
3127
panic(errors.Errorf("cannot handle pod %s/%s: namespace not found", ns, pod.Name))
3228
}
3329
nsToPod[ns] = append(nsToPod[ns], pod)
3430
}
3531

36-
sort.Slice(nsSlice, func(i, j int) bool {
37-
return nsSlice[i] < nsSlice[j]
38-
})
39-
for _, ns := range nsSlice {
32+
namespaces := slices.Sort(maps.Keys(nsToPod))
33+
for _, ns := range namespaces {
4034
labels := r.Namespaces[ns]
4135
nsLabelLines := labelsToLines(labels)
4236
for _, pod := range nsToPod[ns] {
@@ -59,7 +53,7 @@ func (r *Resources) RenderTable() string {
5953
}
6054

6155
func labelsToLines(labels map[string]string) string {
62-
keys := slices.SortBy(builtins.CompareOrdered[string], maps.Keys(labels))
56+
keys := slices.Sort(maps.Keys(labels))
6357
var lines []string
6458
for _, key := range keys {
6559
lines = append(lines, fmt.Sprintf("%s: %s", key, labels[key]))

pkg/connectivity/probe/resources.go

+3-4
Original file line numberDiff line numberDiff line change
@@ -1,15 +1,13 @@
11
package probe
22

33
import (
4-
"github.com/mattfenwick/collections/pkg/builtins"
54
"github.com/mattfenwick/collections/pkg/slices"
65
"github.com/mattfenwick/cyclonus/pkg/kube"
76
"github.com/pkg/errors"
87
"github.com/sirupsen/logrus"
98
"golang.org/x/exp/maps"
109
v1 "k8s.io/api/core/v1"
1110
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
12-
"sort"
1311
"time"
1412
)
1513

@@ -20,7 +18,8 @@ type Resources struct {
2018
}
2119

2220
func NewDefaultResources(kubernetes kube.IKubernetes, namespaces []string, podNames []string, ports []int, protocols []v1.Protocol, externalIPs []string, podCreationTimeoutSeconds int, batchJobs bool) (*Resources, error) {
23-
sort.Strings(externalIPs)
21+
//sort.Strings(externalIPs) // TODO why is this here?
22+
2423
r := &Resources{
2524
Namespaces: map[string]map[string]string{},
2625
//ExternalIPs: externalIPs,
@@ -242,7 +241,7 @@ func (r *Resources) DeletePod(ns string, podName string) (*Resources, error) {
242241
}
243242

244243
func (r *Resources) SortedPodNames() []string {
245-
return slices.SortBy(builtins.CompareOrdered[string], slices.Map(
244+
return slices.Sort(slices.Map(
246245
func(p *Pod) string { return p.PodString().String() },
247246
r.Pods))
248247
}

pkg/connectivity/probe/table.go

+3-4
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,6 @@
11
package probe
22

33
import (
4-
"github.com/mattfenwick/collections/pkg/builtins"
54
"github.com/mattfenwick/collections/pkg/slices"
65
"github.com/mattfenwick/cyclonus/pkg/utils"
76
"github.com/pkg/errors"
@@ -79,7 +78,7 @@ func (t *Table) renderTableHelper(render func(*JobResult) string) string {
7978
isSingleElement = false
8079
break
8180
}
82-
keys := slices.SortBy(builtins.CompareOrdered[string], maps.Keys(dict))
81+
keys := slices.Sort(maps.Keys(dict))
8382
schema[strings.Join(keys, "_")] = true
8483
if len(schema) > 1 {
8584
isSchemaUniform = false
@@ -121,7 +120,7 @@ func (t *Table) renderSimpleTable(render func(*JobResult) string) string {
121120
func (t *Table) renderUniformMultiTable(render func(*JobResult) string) string {
122121
key := t.Wrapped.Keys()[0]
123122
first := t.Get(key.From, key.To)
124-
keys := slices.SortBy(builtins.CompareOrdered[string], maps.Keys(first.JobResults))
123+
keys := slices.Sort(maps.Keys(first.JobResults))
125124
schema := strings.Join(keys, "\n")
126125
return t.Wrapped.Table(schema, true, func(fr, to string, i interface{}) string {
127126
dict := t.Get(fr, to).JobResults
@@ -136,7 +135,7 @@ func (t *Table) renderUniformMultiTable(render func(*JobResult) string) string {
136135
func (t *Table) renderNonuniformTable(render func(*JobResult) string) string {
137136
return t.Wrapped.Table("", true, func(fr, to string, i interface{}) string {
138137
dict := t.Get(fr, to).JobResults
139-
keys := slices.SortBy(builtins.CompareOrdered[string], maps.Keys(dict))
138+
keys := slices.Sort(maps.Keys(dict))
140139
var lines []string
141140
for _, k := range keys {
142141
lines = append(lines, k+": "+render(dict[k]))

pkg/kube/labelselector.go

+4-7
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,6 @@ package kube
22

33
import (
44
"fmt"
5-
"github.com/mattfenwick/collections/pkg/builtins"
65
"github.com/mattfenwick/collections/pkg/slices"
76
"github.com/mattfenwick/cyclonus/pkg/utils"
87
"golang.org/x/exp/maps"
@@ -93,10 +92,9 @@ func IsLabelSelectorEmpty(l metav1.LabelSelector) bool {
9392
// SerializeLabelSelector deterministically converts a metav1.LabelSelector
9493
// into a string
9594
func SerializeLabelSelector(ls metav1.LabelSelector) string {
96-
labelKeys := slices.SortBy(builtins.CompareOrdered[string], maps.Keys(ls.MatchLabels))
9795
keyVals := slices.Map(func(key string) string {
9896
return fmt.Sprintf("%s: %s", key, ls.MatchLabels[key])
99-
}, labelKeys)
97+
}, slices.Sort(maps.Keys(ls.MatchLabels)))
10098
// this looks weird -- we're using an array to make the order deterministic
10199
return utils.JsonStringNoIndent([]interface{}{"MatchLabels", keyVals, "MatchExpression", ls.MatchExpressions})
102100
}
@@ -108,19 +106,18 @@ func LabelSelectorTableLines(selector metav1.LabelSelector) string {
108106
var lines []string
109107
if len(selector.MatchLabels) > 0 {
110108
lines = append(lines, "Match labels:")
111-
for _, key := range slices.SortBy(builtins.CompareOrdered[string], maps.Keys(selector.MatchLabels)) {
109+
for _, key := range slices.Sort(maps.Keys(selector.MatchLabels)) {
112110
val := selector.MatchLabels[key]
113111
lines = append(lines, fmt.Sprintf(" %s: %s", key, val))
114112
}
115113
}
116114
if len(selector.MatchExpressions) > 0 {
117115
lines = append(lines, "Match expressions:")
118-
sortedMatchExpressions := slices.SortOnBy(
116+
sortedMatchExpressions := slices.SortOn(
119117
func(l metav1.LabelSelectorRequirement) string { return l.Key },
120-
builtins.CompareOrdered[string],
121118
selector.MatchExpressions)
122119
for _, exp := range sortedMatchExpressions {
123-
lines = append(lines, fmt.Sprintf(" %s %s %+v", exp.Key, exp.Operator, slices.SortBy(builtins.CompareOrdered[string], exp.Values)))
120+
lines = append(lines, fmt.Sprintf(" %s %s %+v", exp.Key, exp.Operator, slices.Sort(exp.Values)))
124121
}
125122
}
126123
return strings.Join(lines, "\n")

pkg/kube/netpol/policies.go

+3-8
Original file line numberDiff line numberDiff line change
@@ -2,11 +2,12 @@ package netpol
22

33
import (
44
"fmt"
5+
"github.com/mattfenwick/collections/pkg/slices"
6+
"golang.org/x/exp/maps"
57
v1 "k8s.io/api/core/v1"
68
networkingv1 "k8s.io/api/networking/v1"
79
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
810
"k8s.io/apimachinery/pkg/util/intstr"
9-
"sort"
1011
"strings"
1112
)
1213

@@ -16,13 +17,7 @@ func label(key, val string) map[string]string {
1617

1718
func LabelString(labels map[string]string) string {
1819
// 1. first, sort the keys so we get a deterministic answer
19-
var keys []string
20-
for key := range labels {
21-
keys = append(keys, key)
22-
}
23-
sort.Slice(keys, func(i, j int) bool {
24-
return keys[i] < keys[j]
25-
})
20+
keys := slices.Sort(maps.Keys(labels))
2621
// 2. now use the sorted keys to generate chunks
2722
var chunks []string
2823
for _, key := range keys {

pkg/linter/checks.go

+11-16
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,6 @@ package linter
33
import (
44
"fmt"
55
collections "github.com/mattfenwick/collections/pkg"
6-
"github.com/mattfenwick/collections/pkg/builtins"
76
"github.com/mattfenwick/collections/pkg/slices"
87
"github.com/mattfenwick/cyclonus/pkg/matcher"
98
"github.com/mattfenwick/cyclonus/pkg/utils"
@@ -24,16 +23,17 @@ warnings:
2423
type Check string
2524

2625
const (
27-
// omitting the namespace will create the policy in the default namespace
26+
// CheckSourceMissingNamespace omitting the namespace will create the policy in the default namespace
2827
CheckSourceMissingNamespace Check = "CheckSourceMissingNamespace"
29-
// omitting the protocol from a NetworkPolicyPort will default to TCP
28+
// CheckSourcePortMissingProtocol omitting the protocol from a NetworkPolicyPort will default to TCP
3029
CheckSourcePortMissingProtocol Check = "CheckSourcePortMissingProtocol"
31-
// omitting the types can sometimes be automatically handled; but it's better to explicitly list them
30+
// CheckSourceMissingPolicyTypes omitting the types can sometimes be automatically handled; but it's better to explicitly list them
3231
CheckSourceMissingPolicyTypes Check = "CheckSourceMissingPolicyTypes"
33-
// if the policy has ingress/egress rules, then the corresponding type should be present
32+
// CheckSourceMissingPolicyTypeIngress if the policy has ingress rules, then that type should be present
3433
CheckSourceMissingPolicyTypeIngress Check = "CheckSourceMissingPolicyTypeIngress"
35-
CheckSourceMissingPolicyTypeEgress Check = "CheckSourceMissingPolicyTypeEgress"
36-
// duplicate names
34+
// CheckSourceMissingPolicyTypeEgress if the policy has egress rules, then that type should be present
35+
CheckSourceMissingPolicyTypeEgress Check = "CheckSourceMissingPolicyTypeEgress"
36+
// CheckSourceDuplicatePolicyName duplicate names of source network policies
3737
CheckSourceDuplicatePolicyName Check = "CheckSourceDuplicatePolicyName"
3838

3939
CheckDNSBlockedOnTCP Check = "CheckDNSBlockedOnTCP"
@@ -46,11 +46,6 @@ const (
4646
// TODO add check that rule is unnecessary b/c another rule exactly supersedes it
4747
)
4848

49-
func (a Check) Equal(b Check) bool {
50-
// TODO why is this necessary? why can't we use existing String implementation?
51-
return a == b
52-
}
53-
5449
type Warning interface {
5550
OriginIsSource() bool
5651
GetCheck() Check
@@ -83,7 +78,7 @@ func NetpolKey(netpol *networkingv1.NetworkPolicy) string {
8378
return fmt.Sprintf("%s/%s", netpol.Namespace, netpol.Name)
8479
}
8580

86-
func sortOn(w Warning) []string {
81+
func sortKey(w Warning) []string {
8782
origin := "1"
8883
if w.OriginIsSource() {
8984
origin = "0"
@@ -110,7 +105,7 @@ func (r *resolvedWarning) GetTarget() string {
110105
}
111106

112107
func (r *resolvedWarning) GetSourcePolicies() string {
113-
target := slices.SortBy(builtins.CompareOrdered[string], slices.Map(NetpolKey, r.Target.SourceRules))
108+
target := slices.Sort(slices.Map(NetpolKey, r.Target.SourceRules))
114109
return strings.Join(target, "\n")
115110
}
116111

@@ -122,7 +117,7 @@ func WarningsTable(warnings []Warning) string {
122117
table.SetReflowDuringAutoWrap(false)
123118
table.SetAutoWrapText(false)
124119

125-
sortedWarnings := slices.SortOnBy(sortOn, slices.CompareSlice[string](builtins.CompareOrdered[string]), warnings)
120+
sortedWarnings := slices.SortOnBy(sortKey, slices.CompareSlicePairwise[string](), warnings)
126121
for _, w := range sortedWarnings {
127122
origin := "Source"
128123
if !w.OriginIsSource() {
@@ -135,7 +130,7 @@ func WarningsTable(warnings []Warning) string {
135130
return str.String()
136131
}
137132

138-
func Lint(kubePolicies []*networkingv1.NetworkPolicy, skip *collections.Set[Check]) []Warning {
133+
func Lint(kubePolicies []*networkingv1.NetworkPolicy, skip *collections.Set[Check, Check]) []Warning {
139134
policies := matcher.BuildNetworkPolicies(false, kubePolicies)
140135
warnings := append(LintSourcePolicies(kubePolicies), LintResolvedPolicies(policies)...)
141136

pkg/matcher/explain.go

+2-4
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,6 @@ package matcher
22

33
import (
44
"fmt"
5-
"github.com/mattfenwick/collections/pkg/builtins"
65
"github.com/mattfenwick/collections/pkg/slices"
76
"github.com/mattfenwick/cyclonus/pkg/kube"
87
"github.com/mattfenwick/cyclonus/pkg/utils"
@@ -49,8 +48,7 @@ func (s *SliceBuilder) TargetsTableLines(targets []*Target, isIngress bool) {
4948
ruleType = "Egress"
5049
}
5150
for _, target := range targets {
52-
sourceRules := slices.SortBy(
53-
builtins.CompareOrdered[string],
51+
sourceRules := slices.Sort(
5452
slices.Map(func(sr *networkingv1.NetworkPolicy) string {
5553
return fmt.Sprintf("%s/%s", sr.Namespace, sr.Name)
5654
}, target.SourceRules))
@@ -61,7 +59,7 @@ func (s *SliceBuilder) TargetsTableLines(targets []*Target, isIngress bool) {
6159
if len(target.Peers) == 0 {
6260
s.Append("no pods, no ips", "no ports, no protocols")
6361
} else {
64-
for _, peer := range slices.SortOnBy(func(p PeerMatcher) string { return utils.DumpJSON(p) }, builtins.CompareOrdered[string], target.Peers) {
62+
for _, peer := range slices.SortOn(func(p PeerMatcher) string { return utils.DumpJSON(p) }, target.Peers) {
6563
switch a := peer.(type) {
6664
case *AllPeersMatcher:
6765
s.Append("all pods, all ips", "all ports, all protocols")

0 commit comments

Comments
 (0)