Skip to content

Commit b3fbf98

Browse files
authored
Scale test (#1115)
Problem: We need to understand how NGF performs when resources are scaled. Solution: Adds results from manual scale tests and steps on how to run the scale tests.
1 parent 331467a commit b3fbf98

23 files changed

+3645
-0
lines changed

tests/scale/generate_manifests.go

+277
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,277 @@
1+
//go:build scale
2+
// +build scale
3+
4+
package scale
5+
6+
import (
7+
"bytes"
8+
"fmt"
9+
"os"
10+
"path/filepath"
11+
"text/template"
12+
)
13+
14+
var gwTmplTxt = `apiVersion: gateway.networking.k8s.io/v1beta1
15+
kind: Gateway
16+
metadata:
17+
name: gateway
18+
spec:
19+
gatewayClassName: nginx
20+
listeners:
21+
{{- range $l := . }}
22+
- name: {{ $l.Name }}
23+
hostname: "{{ $l.HostnamePrefix }}.example.com"{{ if ne $l.SecretName "" }}
24+
port: 443
25+
protocol: HTTPS
26+
tls:
27+
mode: Terminate
28+
certificateRefs:
29+
- kind: Secret
30+
name: {{ $l.SecretName }}{{ else }}
31+
port: 80
32+
protocol: HTTP
33+
{{- end -}}
34+
{{- end -}}`
35+
36+
var hrTmplTxt = `apiVersion: gateway.networking.k8s.io/v1beta1
37+
kind: HTTPRoute
38+
metadata:
39+
name: {{ .Name }}
40+
spec:
41+
parentRefs:
42+
- name: gateway
43+
sectionName: {{ .ListenerName }}
44+
hostnames:
45+
- "{{ .HostnamePrefix }}.example.com"
46+
rules:
47+
- matches:
48+
- path:
49+
type: PathPrefix
50+
value: /
51+
backendRefs:
52+
- name: {{ .BackendName }}
53+
port: 80`
54+
55+
// nolint:all
56+
var secretTmplTxt = `apiVersion: v1
57+
kind: Secret
58+
metadata:
59+
name: {{ . }}
60+
type: kubernetes.io/tls
61+
data:
62+
tls.crt: LS0tLS1CRUdJTiBDRVJUSUZJQ0FURS0tLS0tCk1JSUNzakNDQVpvQ0NRQzdCdVdXdWRtRkNEQU5CZ2txaGtpRzl3MEJBUXNGQURBYk1Sa3dGd1lEVlFRRERCQmoKWVdabExtVjRZVzF3YkdVdVkyOXRNQjRYRFRJeU1EY3hOREl4TlRJek9Wb1hEVEl6TURjeE5ESXhOVEl6T1ZvdwpHekVaTUJjR0ExVUVBd3dRWTJGbVpTNWxlR0Z0Y0d4bExtTnZiVENDQVNJd0RRWUpLb1pJaHZjTkFRRUJCUUFECmdnRVBBRENDQVFvQ2dnRUJBTHFZMnRHNFc5aStFYzJhdnV4Q2prb2tnUUx1ek10U1Rnc1RNaEhuK3ZRUmxIam8KVzFLRnMvQVdlS25UUStyTWVKVWNseis4M3QwRGtyRThwUisxR2NKSE50WlNMb0NEYUlRN0Nhck5nY1daS0o4Qgo1WDNnVS9YeVJHZjI2c1REd2xzU3NkSEQ1U2U3K2Vab3NPcTdHTVF3K25HR2NVZ0VtL1Q1UEMvY05PWE0zZWxGClRPL051MStoMzROVG9BbDNQdTF2QlpMcDNQVERtQ0thaEROV0NWbUJQUWpNNFI4VERsbFhhMHQ5Z1o1MTRSRzUKWHlZWTNtdzZpUzIrR1dYVXllMjFuWVV4UEhZbDV4RHY0c0FXaGRXbElweHlZQlNCRURjczN6QlI2bFF1OWkxZAp0R1k4dGJ3blVmcUVUR3NZdWxzc05qcU95V1VEcFdJelhibHhJZVVDQXdFQUFUQU5CZ2txaGtpRzl3MEJBUXNGCkFBT0NBUUVBcjkrZWJ0U1dzSnhLTGtLZlRkek1ISFhOd2Y5ZXFVbHNtTXZmMGdBdWVKTUpUR215dG1iWjlpbXQKL2RnWlpYVE9hTElHUG9oZ3BpS0l5eVVRZVdGQ2F0NHRxWkNPVWRhbUloOGk0Q1h6QVJYVHNvcUNOenNNLzZMRQphM25XbFZyS2lmZHYrWkxyRi8vblc0VVNvOEoxaCtQeDljY0tpRDZZU0RVUERDRGh1RUtFWXcvbHpoUDJVOXNmCnl6cEJKVGQ4enFyM3paTjNGWWlITmgzYlRhQS82di9jU2lyamNTK1EwQXg4RWpzQzYxRjRVMTc4QzdWNWRCKzQKcmtPTy9QNlA0UFlWNTRZZHMvRjE2WkZJTHFBNENCYnExRExuYWRxamxyN3NPbzl2ZzNnWFNMYXBVVkdtZ2todAp6VlZPWG1mU0Z4OS90MDBHUi95bUdPbERJbWlXMGc9PQotLS0tLUVORCBDRVJUSUZJQ0FURS0tLS0tCg==
63+
tls.key: LS0tLS1CRUdJTiBQUklWQVRFIEtFWS0tLS0tCk1JSUV2UUlCQURBTkJna3Foa2lHOXcwQkFRRUZBQVNDQktjd2dnU2pBZ0VBQW9JQkFRQzZtTnJSdUZ2WXZoSE4KbXI3c1FvNUtKSUVDN3N6TFVrNExFeklSNS9yMEVaUjQ2RnRTaGJQd0ZuaXAwMFBxekhpVkhKYy92TjdkQTVLeApQS1VmdFJuQ1J6YldVaTZBZzJpRU93bXF6WUhGbVNpZkFlVjk0RlAxOGtSbjl1ckV3OEpiRXJIUncrVW51L25tCmFMRHF1eGpFTVBweGhuRklCSnYwK1R3djNEVGx6TjNwUlV6dnpidGZvZCtEVTZBSmR6N3Rid1dTNmR6MHc1Z2kKbW9RelZnbFpnVDBJek9FZkV3NVpWMnRMZllHZWRlRVJ1VjhtR041c09va3R2aGxsMU1udHRaMkZNVHgySmVjUQo3K0xBRm9YVnBTS2NjbUFVZ1JBM0xOOHdVZXBVTHZZdFhiUm1QTFc4SjFINmhFeHJHTHBiTERZNmpzbGxBNlZpCk0xMjVjU0hsQWdNQkFBRUNnZ0VBQnpaRE50bmVTdWxGdk9HZlFYaHRFWGFKdWZoSzJBenRVVVpEcUNlRUxvekQKWlV6dHdxbkNRNlJLczUyandWNTN4cU9kUU94bTNMbjNvSHdNa2NZcEliWW82MjJ2dUczYnkwaVEzaFlsVHVMVgpqQmZCcS9UUXFlL2NMdngvSkczQWhFNmJxdFRjZFlXeGFmTmY2eUtpR1dzZk11WVVXTWs4MGVJVUxuRmZaZ1pOCklYNTlSOHlqdE9CVm9Sa3hjYTVoMW1ZTDFsSlJNM3ZqVHNHTHFybmpOTjNBdWZ3ZGRpK1VDbGZVL2l0K1EvZkUKV216aFFoTlRpNVFkRWJLVStOTnYvNnYvb2JvandNb25HVVBCdEFTUE05cmxFemIralQ1WHdWQjgvLzRGY3VoSwoyVzNpcjhtNHVlQ1JHSVlrbGxlLzhuQmZ0eVhiVkNocVRyZFBlaGlPM1FLQmdRRGlrR3JTOTc3cjg3Y1JPOCtQClpoeXltNXo4NVIzTHVVbFNTazJiOTI1QlhvakpZL2RRZDVTdFVsSWE4OUZKZnNWc1JRcEhHaTFCYzBMaTY1YjIKazR0cE5xcVFoUmZ1UVh0UG9GYXRuQzlPRnJVTXJXbDVJN0ZFejZnNkNQMVBXMEg5d2hPemFKZUdpZVpNYjlYTQoybDdSSFZOcC9jTDlYbmhNMnN0Q1lua2Iwd0tCZ1FEUzF4K0crakEyUVNtRVFWNXA1RnRONGcyamsyZEFjMEhNClRIQ2tTazFDRjhkR0Z2UWtsWm5ZbUt0dXFYeXNtekJGcnZKdmt2eUhqbUNYYTducXlpajBEdDZtODViN3BGcVAKQWxtajdtbXI3Z1pUeG1ZMXBhRWFLMXY4SDNINGtRNVl3MWdrTWRybVJHcVAvaTBGaDVpaGtSZS9DOUtGTFVkSQpDcnJjTzhkUVp3S0JnSHA1MzRXVWNCMVZibzFlYStIMUxXWlFRUmxsTWlwRFM2TzBqeWZWSmtFb1BZSEJESnp2ClIrdzZLREJ4eFoyWmJsZ05LblV0YlhHSVFZd3lGelhNcFB5SGxNVHpiZkJhYmJLcDFyR2JVT2RCMXpXM09PRkgKcmppb21TUm1YNmxhaDk0SjRHU0lFZ0drNGw1SHhxZ3JGRDZ2UDd4NGRjUktJWFpLZ0w2dVJSSUpBb0dCQU1CVApaL2p5WStRNTBLdEtEZHUrYU9ORW4zaGxUN3hrNXRKN3NBek5rbWdGMU10RXlQUk9Xd1pQVGFJbWpRbk9qbHdpCldCZ2JGcXg0M2ZlQ1Z4ZXJ6V3ZEM0txaWJVbWpCTkNMTGtYeGh3ZEVteFQwVit2NzZGYzgwaTNNYVdSNnZZR08KditwVVovL0F6UXdJcWZ6dlVmV2ZxdStrMHlhVXhQOGNlcFBIRyt0bEFvR0FmQUtVVWhqeFU0Ym5vVzVwVUhKegpwWWZXZXZ5TW54NWZyT2VsSmRmNzlvNGMvMHhVSjh1eFBFWDFkRmNrZW96dHNpaVFTNkN6MENRY09XVWxtSkRwCnVrdERvVzM3VmNSQU1BVjY3NlgxQVZlM0UwNm5aL2g2Tkd4Z28rT042Q3pwL0lkMkJPUm9IMFAxa2RjY1NLT3kKMUtFZlNnb1B0c1N1eEpBZXdUZmxDMXc9Ci0tLS0tRU5EIFBSSVZBVEUgS0VZLS0tLS0K
64+
`
65+
66+
var appTmplTxt = `apiVersion: v1
67+
apiVersion: apps/v1
68+
kind: Deployment
69+
metadata:
70+
name: {{ . }}
71+
spec:
72+
replicas: 1
73+
selector:
74+
matchLabels:
75+
app: {{ . }}
76+
template:
77+
metadata:
78+
labels:
79+
app: {{ . }}
80+
spec:
81+
containers:
82+
- name: nginx
83+
image: nginxdemos/nginx-hello:plain-text
84+
ports:
85+
- containerPort: 8080
86+
---
87+
apiVersion: v1
88+
kind: Service
89+
metadata:
90+
name: {{ . }}
91+
spec:
92+
ports:
93+
- port: 80
94+
targetPort: 8080
95+
protocol: TCP
96+
name: http
97+
selector:
98+
app: {{ . }}
99+
`
100+
101+
var (
102+
gwTmpl = template.Must(template.New("gw").Parse(gwTmplTxt))
103+
hrTmpl = template.Must(template.New("hr").Parse(hrTmplTxt))
104+
secretTmpl = template.Must(template.New("secret").Parse(secretTmplTxt))
105+
appTmpl = template.Must(template.New("app").Parse(appTmplTxt))
106+
)
107+
108+
type Listener struct {
109+
Name string
110+
HostnamePrefix string
111+
SecretName string
112+
}
113+
114+
type Route struct {
115+
Name string
116+
ListenerName string
117+
HostnamePrefix string
118+
BackendName string
119+
}
120+
121+
func getPrereqDirName(manifestDir string) string {
122+
return filepath.Join(manifestDir, "prereqs")
123+
}
124+
125+
func generateScaleListenerManifests(numListeners int, manifestDir string, tls bool) error {
126+
listeners := make([]Listener, 0)
127+
backends := make([]string, 0)
128+
secrets := make([]string, 0)
129+
130+
for i := 0; i < numListeners; i++ {
131+
listenerName := fmt.Sprintf("listener-%d", i)
132+
hostnamePrefix := fmt.Sprintf("%d", i)
133+
backendName := fmt.Sprintf("backend-%d", i)
134+
135+
var secretName string
136+
if tls {
137+
secretName = fmt.Sprintf("secret-%d", i)
138+
secrets = append(secrets, secretName)
139+
}
140+
141+
listeners = append(listeners, Listener{
142+
Name: listenerName,
143+
HostnamePrefix: hostnamePrefix,
144+
SecretName: secretName,
145+
})
146+
147+
route := Route{
148+
Name: fmt.Sprintf("route-%d", i),
149+
ListenerName: listenerName,
150+
HostnamePrefix: hostnamePrefix,
151+
BackendName: backendName,
152+
}
153+
154+
backends = append(backends, backendName)
155+
156+
if err := generateManifests(manifestDir, i, listeners, []Route{route}); err != nil {
157+
return err
158+
}
159+
}
160+
161+
if err := generateSecrets(getPrereqDirName(manifestDir), secrets); err != nil {
162+
return err
163+
}
164+
165+
return generateBackendAppManifests(getPrereqDirName(manifestDir), backends)
166+
}
167+
168+
func generateSecrets(secretsDir string, secrets []string) error {
169+
err := os.Mkdir(secretsDir, 0o750)
170+
if err != nil && !os.IsExist(err) {
171+
return err
172+
}
173+
174+
for _, secret := range secrets {
175+
var buf bytes.Buffer
176+
177+
if err = secretTmpl.Execute(&buf, secret); err != nil {
178+
return err
179+
}
180+
181+
path := filepath.Join(secretsDir, fmt.Sprintf("%s.yaml", secret))
182+
183+
fmt.Println("Writing", path)
184+
if err := os.WriteFile(path, buf.Bytes(), 0o600); err != nil {
185+
return err
186+
}
187+
}
188+
189+
return nil
190+
}
191+
192+
func generateScaleHTTPRouteManifests(numRoutes int, manifestDir string) error {
193+
l := Listener{
194+
Name: "listener",
195+
HostnamePrefix: "*",
196+
}
197+
198+
backendName := "backend"
199+
200+
for i := 0; i < numRoutes; i++ {
201+
202+
route := Route{
203+
Name: fmt.Sprintf("route-%d", i),
204+
HostnamePrefix: fmt.Sprintf("%d", i),
205+
ListenerName: "listener",
206+
BackendName: backendName,
207+
}
208+
209+
var listeners []Listener
210+
if i == 0 {
211+
// only generate a Gateway on the first iteration
212+
listeners = []Listener{l}
213+
}
214+
215+
if err := generateManifests(manifestDir, i, listeners, []Route{route}); err != nil {
216+
return err
217+
}
218+
219+
}
220+
221+
return generateBackendAppManifests(getPrereqDirName(manifestDir), []string{backendName})
222+
}
223+
224+
func generateManifests(outDir string, version int, listeners []Listener, routes []Route) error {
225+
var buf bytes.Buffer
226+
227+
if len(listeners) > 0 {
228+
if err := gwTmpl.Execute(&buf, listeners); err != nil {
229+
return err
230+
}
231+
}
232+
233+
for _, r := range routes {
234+
if buf.Len() > 0 {
235+
buf.Write([]byte("\n---\n"))
236+
}
237+
238+
if err := hrTmpl.Execute(&buf, r); err != nil {
239+
return err
240+
}
241+
}
242+
243+
err := os.Mkdir(outDir, 0o750)
244+
if err != nil && !os.IsExist(err) {
245+
return err
246+
}
247+
248+
filename := fmt.Sprintf("manifest-%d.yaml", version)
249+
path := filepath.Join(outDir, filename)
250+
251+
fmt.Println("Writing", path)
252+
return os.WriteFile(path, buf.Bytes(), 0o600)
253+
}
254+
255+
func generateBackendAppManifests(outDir string, backends []string) error {
256+
err := os.Mkdir(outDir, 0o750)
257+
if err != nil && !os.IsExist(err) {
258+
return err
259+
}
260+
261+
for _, backend := range backends {
262+
var buf bytes.Buffer
263+
264+
if err = appTmpl.Execute(&buf, backend); err != nil {
265+
return err
266+
}
267+
268+
path := filepath.Join(outDir, fmt.Sprintf("%s.yaml", backend))
269+
270+
fmt.Println("Writing", path)
271+
if err := os.WriteFile(path, buf.Bytes(), 0o600); err != nil {
272+
return err
273+
}
274+
}
275+
276+
return nil
277+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,44 @@
1+
apiVersion: v1
2+
kind: Namespace
3+
metadata:
4+
name: prom
5+
---
6+
apiVersion: v1
7+
kind: ServiceAccount
8+
metadata:
9+
name: prometheus
10+
namespace: prom
11+
---
12+
apiVersion: rbac.authorization.k8s.io/v1
13+
kind: ClusterRole
14+
metadata:
15+
name: prometheus
16+
namespace: prom
17+
rules:
18+
- apiGroups: [""]
19+
resources:
20+
- nodes
21+
- services
22+
- endpoints
23+
- pods
24+
verbs: ["get", "list", "watch"]
25+
- apiGroups: [""]
26+
resources:
27+
- configmaps
28+
verbs: ["get"]
29+
- nonResourceURLs: ["/metrics"]
30+
verbs: ["get"]
31+
---
32+
apiVersion: rbac.authorization.k8s.io/v1
33+
kind: ClusterRoleBinding
34+
metadata:
35+
name: prometheus
36+
namespace: prom
37+
roleRef:
38+
apiGroup: rbac.authorization.k8s.io
39+
kind: ClusterRole
40+
name: prometheus
41+
subjects:
42+
- kind: ServiceAccount
43+
name: prometheus
44+
namespace: prom

0 commit comments

Comments
 (0)