cdk8s-plus-33 2.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.jsii +98351 -0
- package/CODE_OF_CONDUCT.md +3 -0
- package/CONTRIBUTING.md +185 -0
- package/DCO +34 -0
- package/LICENSE +202 -0
- package/NOTICE +1 -0
- package/OWNERS.md +5 -0
- package/README.md +32 -0
- package/SECURITY.md +5 -0
- package/cdk8s.yaml +10 -0
- package/docs/java.md +24323 -0
- package/docs/plus/config-map.md +98 -0
- package/docs/plus/container.md +133 -0
- package/docs/plus/cronjob.md +67 -0
- package/docs/plus/deployment.md +232 -0
- package/docs/plus/horizontal-pod-autoscaler.md +226 -0
- package/docs/plus/ingress.md +68 -0
- package/docs/plus/job.md +48 -0
- package/docs/plus/namespace.md +58 -0
- package/docs/plus/network-policy.md +341 -0
- package/docs/plus/pod.md +455 -0
- package/docs/plus/pv.md +82 -0
- package/docs/plus/pvc.md +77 -0
- package/docs/plus/rbac.md +104 -0
- package/docs/plus/secret.md +32 -0
- package/docs/plus/service-account.md +35 -0
- package/docs/plus/service.md +41 -0
- package/docs/plus/volume.md +38 -0
- package/docs/python.md +26547 -0
- package/docs/typescript.md +19825 -0
- package/git-hooks/README.md +9 -0
- package/git-hooks/prepare-commit-msg +18 -0
- package/git-hooks/setup.sh +10 -0
- package/lib/_action.d.ts +21 -0
- package/lib/_action.js +32 -0
- package/lib/api-resource.d.ts +298 -0
- package/lib/api-resource.js +430 -0
- package/lib/base.d.ts +79 -0
- package/lib/base.js +92 -0
- package/lib/config-map.d.ts +126 -0
- package/lib/config-map.js +159 -0
- package/lib/container.d.ts +1057 -0
- package/lib/container.js +845 -0
- package/lib/cron-job.d.ts +138 -0
- package/lib/cron-job.js +103 -0
- package/lib/daemon-set.d.ts +45 -0
- package/lib/daemon-set.js +55 -0
- package/lib/deployment.d.ts +223 -0
- package/lib/deployment.js +214 -0
- package/lib/handler.d.ts +62 -0
- package/lib/handler.js +54 -0
- package/lib/horizontal-pod-autoscaler.d.ts +500 -0
- package/lib/horizontal-pod-autoscaler.js +569 -0
- package/lib/imports/k8s.d.ts +24537 -0
- package/lib/imports/k8s.js +18615 -0
- package/lib/index.d.ts +26 -0
- package/lib/index.js +44 -0
- package/lib/ingress.d.ts +230 -0
- package/lib/ingress.js +246 -0
- package/lib/job.d.ts +64 -0
- package/lib/job.js +54 -0
- package/lib/namespace.d.ts +128 -0
- package/lib/namespace.js +109 -0
- package/lib/network-policy.d.ts +311 -0
- package/lib/network-policy.js +344 -0
- package/lib/pod.d.ts +1097 -0
- package/lib/pod.js +1143 -0
- package/lib/probe.d.ts +141 -0
- package/lib/probe.js +77 -0
- package/lib/pv.d.ts +375 -0
- package/lib/pv.js +273 -0
- package/lib/pvc.d.ts +163 -0
- package/lib/pvc.js +154 -0
- package/lib/role-binding.d.ts +138 -0
- package/lib/role-binding.js +165 -0
- package/lib/role.d.ts +268 -0
- package/lib/role.js +401 -0
- package/lib/secret.d.ts +195 -0
- package/lib/secret.js +185 -0
- package/lib/service-account.d.ts +83 -0
- package/lib/service-account.js +105 -0
- package/lib/service.d.ts +289 -0
- package/lib/service.js +182 -0
- package/lib/stateful-set.d.ts +169 -0
- package/lib/stateful-set.js +174 -0
- package/lib/utils.d.ts +4 -0
- package/lib/utils.js +33 -0
- package/lib/volume.d.ts +573 -0
- package/lib/volume.js +371 -0
- package/lib/workload.d.ts +121 -0
- package/lib/workload.js +122 -0
- package/node_modules/balanced-match/.github/FUNDING.yml +2 -0
- package/node_modules/balanced-match/LICENSE.md +21 -0
- package/node_modules/balanced-match/README.md +97 -0
- package/node_modules/balanced-match/index.js +62 -0
- package/node_modules/balanced-match/package.json +48 -0
- package/node_modules/brace-expansion/.github/FUNDING.yml +2 -0
- package/node_modules/brace-expansion/LICENSE +21 -0
- package/node_modules/brace-expansion/README.md +135 -0
- package/node_modules/brace-expansion/index.js +203 -0
- package/node_modules/brace-expansion/package.json +49 -0
- package/node_modules/minimatch/LICENSE +15 -0
- package/node_modules/minimatch/README.md +454 -0
- package/node_modules/minimatch/dist/commonjs/assert-valid-pattern.d.ts +2 -0
- package/node_modules/minimatch/dist/commonjs/assert-valid-pattern.d.ts.map +1 -0
- package/node_modules/minimatch/dist/commonjs/assert-valid-pattern.js +14 -0
- package/node_modules/minimatch/dist/commonjs/assert-valid-pattern.js.map +1 -0
- package/node_modules/minimatch/dist/commonjs/ast.d.ts +20 -0
- package/node_modules/minimatch/dist/commonjs/ast.d.ts.map +1 -0
- package/node_modules/minimatch/dist/commonjs/ast.js +592 -0
- package/node_modules/minimatch/dist/commonjs/ast.js.map +1 -0
- package/node_modules/minimatch/dist/commonjs/brace-expressions.d.ts +8 -0
- package/node_modules/minimatch/dist/commonjs/brace-expressions.d.ts.map +1 -0
- package/node_modules/minimatch/dist/commonjs/brace-expressions.js +152 -0
- package/node_modules/minimatch/dist/commonjs/brace-expressions.js.map +1 -0
- package/node_modules/minimatch/dist/commonjs/escape.d.ts +12 -0
- package/node_modules/minimatch/dist/commonjs/escape.d.ts.map +1 -0
- package/node_modules/minimatch/dist/commonjs/escape.js +22 -0
- package/node_modules/minimatch/dist/commonjs/escape.js.map +1 -0
- package/node_modules/minimatch/dist/commonjs/index.d.ts +94 -0
- package/node_modules/minimatch/dist/commonjs/index.d.ts.map +1 -0
- package/node_modules/minimatch/dist/commonjs/index.js +1017 -0
- package/node_modules/minimatch/dist/commonjs/index.js.map +1 -0
- package/node_modules/minimatch/dist/commonjs/package.json +3 -0
- package/node_modules/minimatch/dist/commonjs/unescape.d.ts +17 -0
- package/node_modules/minimatch/dist/commonjs/unescape.d.ts.map +1 -0
- package/node_modules/minimatch/dist/commonjs/unescape.js +24 -0
- package/node_modules/minimatch/dist/commonjs/unescape.js.map +1 -0
- package/node_modules/minimatch/dist/esm/assert-valid-pattern.d.ts +2 -0
- package/node_modules/minimatch/dist/esm/assert-valid-pattern.d.ts.map +1 -0
- package/node_modules/minimatch/dist/esm/assert-valid-pattern.js +10 -0
- package/node_modules/minimatch/dist/esm/assert-valid-pattern.js.map +1 -0
- package/node_modules/minimatch/dist/esm/ast.d.ts +20 -0
- package/node_modules/minimatch/dist/esm/ast.d.ts.map +1 -0
- package/node_modules/minimatch/dist/esm/ast.js +588 -0
- package/node_modules/minimatch/dist/esm/ast.js.map +1 -0
- package/node_modules/minimatch/dist/esm/brace-expressions.d.ts +8 -0
- package/node_modules/minimatch/dist/esm/brace-expressions.d.ts.map +1 -0
- package/node_modules/minimatch/dist/esm/brace-expressions.js +148 -0
- package/node_modules/minimatch/dist/esm/brace-expressions.js.map +1 -0
- package/node_modules/minimatch/dist/esm/escape.d.ts +12 -0
- package/node_modules/minimatch/dist/esm/escape.d.ts.map +1 -0
- package/node_modules/minimatch/dist/esm/escape.js +18 -0
- package/node_modules/minimatch/dist/esm/escape.js.map +1 -0
- package/node_modules/minimatch/dist/esm/index.d.ts +94 -0
- package/node_modules/minimatch/dist/esm/index.d.ts.map +1 -0
- package/node_modules/minimatch/dist/esm/index.js +1001 -0
- package/node_modules/minimatch/dist/esm/index.js.map +1 -0
- package/node_modules/minimatch/dist/esm/package.json +3 -0
- package/node_modules/minimatch/dist/esm/unescape.d.ts +17 -0
- package/node_modules/minimatch/dist/esm/unescape.d.ts.map +1 -0
- package/node_modules/minimatch/dist/esm/unescape.js +20 -0
- package/node_modules/minimatch/dist/esm/unescape.js.map +1 -0
- package/node_modules/minimatch/package.json +82 -0
- package/package.json +181 -0
- package/rotate.md +84 -0
package/lib/pod.js
ADDED
|
@@ -0,0 +1,1143 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var _a, _b, _c, _d, _e, _f, _g, _h, _j, _k, _l, _m, _o, _p, _q, _r;
|
|
3
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
+
exports.PodConnections = exports.PodConnectionsIsolation = exports.PodScheduling = exports.Topology = exports.Node = exports.NamedNode = exports.TaintedNode = exports.LabeledNode = exports.Pods = exports.NodeTaintQuery = exports.TaintEffect = exports.LabelExpression = exports.NodeLabelQuery = exports.DnsPolicy = exports.FsGroupChangePolicy = exports.RestartPolicy = exports.PodSecurityContext = exports.PodDns = exports.Pod = exports.LabelSelector = exports.AbstractPod = void 0;
|
|
5
|
+
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
|
6
|
+
const cdk8s_1 = require("cdk8s");
|
|
7
|
+
const constructs_1 = require("constructs");
|
|
8
|
+
const base = require("./base");
|
|
9
|
+
const container = require("./container");
|
|
10
|
+
const k8s = require("./imports/k8s");
|
|
11
|
+
const networkpolicy = require("./network-policy");
|
|
12
|
+
const utils_1 = require("./utils");
|
|
13
|
+
class AbstractPod extends base.Resource {
|
|
14
|
+
constructor(scope, id, props = {}) {
|
|
15
|
+
super(scope, id);
|
|
16
|
+
this._containers = [];
|
|
17
|
+
this._initContainers = [];
|
|
18
|
+
this._hostAliases = [];
|
|
19
|
+
this._volumes = new Map();
|
|
20
|
+
this.restartPolicy = props.restartPolicy ?? RestartPolicy.ALWAYS;
|
|
21
|
+
this.serviceAccount = props.serviceAccount;
|
|
22
|
+
this.securityContext = new PodSecurityContext(props.securityContext);
|
|
23
|
+
this.dns = new PodDns(props.dns);
|
|
24
|
+
this.dockerRegistryAuth = props.dockerRegistryAuth;
|
|
25
|
+
this.automountServiceAccountToken = props.automountServiceAccountToken ?? false;
|
|
26
|
+
this.shareProcessNamespace = props.shareProcessNamespace ?? false;
|
|
27
|
+
this.isolate = props.isolate ?? false;
|
|
28
|
+
this.hostNetwork = props.hostNetwork ?? false;
|
|
29
|
+
this.terminationGracePeriod = props.terminationGracePeriod ?? cdk8s_1.Duration.seconds(30);
|
|
30
|
+
this.enableServiceLinks = props.enableServiceLinks;
|
|
31
|
+
if (props.containers) {
|
|
32
|
+
props.containers.forEach(c => this.addContainer(c));
|
|
33
|
+
}
|
|
34
|
+
if (props.volumes) {
|
|
35
|
+
props.volumes.forEach(v => this.addVolume(v));
|
|
36
|
+
}
|
|
37
|
+
if (props.initContainers) {
|
|
38
|
+
props.initContainers.forEach(c => this.addInitContainer(c));
|
|
39
|
+
}
|
|
40
|
+
if (props.hostAliases) {
|
|
41
|
+
props.hostAliases.forEach(c => this.addHostAlias(c));
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
get containers() {
|
|
45
|
+
return [...this._containers];
|
|
46
|
+
}
|
|
47
|
+
get initContainers() {
|
|
48
|
+
return [...this._initContainers];
|
|
49
|
+
}
|
|
50
|
+
get volumes() {
|
|
51
|
+
return Array.from(this._volumes.values());
|
|
52
|
+
}
|
|
53
|
+
get hostAliases() {
|
|
54
|
+
return [...this._hostAliases];
|
|
55
|
+
}
|
|
56
|
+
/**
|
|
57
|
+
* @see IPodSelector.toPodSelectorConfig()
|
|
58
|
+
*/
|
|
59
|
+
toPodSelectorConfig() {
|
|
60
|
+
const podAddress = this.podMetadata.getLabel(Pod.ADDRESS_LABEL);
|
|
61
|
+
if (!podAddress) {
|
|
62
|
+
// shouldn't happen because we add this label automatically in both pods and workloads.
|
|
63
|
+
throw new Error(`Unable to create a label selector since ${Pod.ADDRESS_LABEL} label is missing`);
|
|
64
|
+
}
|
|
65
|
+
return {
|
|
66
|
+
labelSelector: LabelSelector.of({ labels: { [Pod.ADDRESS_LABEL]: podAddress } }),
|
|
67
|
+
namespaces: this.metadata.namespace ? {
|
|
68
|
+
names: [this.metadata.namespace],
|
|
69
|
+
} : undefined,
|
|
70
|
+
};
|
|
71
|
+
}
|
|
72
|
+
/**
|
|
73
|
+
* @see INetworkPolicyPeer.toNetworkPolicyPeerConfig()
|
|
74
|
+
*/
|
|
75
|
+
toNetworkPolicyPeerConfig() {
|
|
76
|
+
return { podSelector: this.toPodSelectorConfig() };
|
|
77
|
+
}
|
|
78
|
+
/**
|
|
79
|
+
* @see INetworkPolicyPeer.toPodSelector()
|
|
80
|
+
*/
|
|
81
|
+
toPodSelector() {
|
|
82
|
+
return this;
|
|
83
|
+
}
|
|
84
|
+
addContainer(cont) {
|
|
85
|
+
const impl = new container.Container(cont);
|
|
86
|
+
this.attachContainer(impl);
|
|
87
|
+
return impl;
|
|
88
|
+
}
|
|
89
|
+
attachContainer(cont) {
|
|
90
|
+
this._containers.push(cont);
|
|
91
|
+
}
|
|
92
|
+
addInitContainer(cont) {
|
|
93
|
+
// https://kubernetes.io/docs/concepts/workloads/pods/init-containers/#differences-from-regular-containers
|
|
94
|
+
if (!this.isSidecarContainer(cont) && cont.readiness) {
|
|
95
|
+
throw new Error('Init containers must not have a readiness probe');
|
|
96
|
+
}
|
|
97
|
+
if (!this.isSidecarContainer(cont) && cont.liveness) {
|
|
98
|
+
throw new Error('Init containers must not have a liveness probe');
|
|
99
|
+
}
|
|
100
|
+
if (!this.isSidecarContainer(cont) && cont.startup) {
|
|
101
|
+
throw new Error('Init containers must not have a startup probe');
|
|
102
|
+
}
|
|
103
|
+
const impl = new container.Container({
|
|
104
|
+
...cont,
|
|
105
|
+
name: cont.name ?? `init-${this._initContainers.length}`,
|
|
106
|
+
});
|
|
107
|
+
this._initContainers.push(impl);
|
|
108
|
+
return impl;
|
|
109
|
+
}
|
|
110
|
+
// Any initContainer that has `restartPolicy=Always` is a sidecar container. Please refer to
|
|
111
|
+
// documentation for more details:
|
|
112
|
+
// https://kubernetes.io/docs/concepts/workloads/pods/sidecar-containers/#differences-from-init-containers
|
|
113
|
+
isSidecarContainer(cont) {
|
|
114
|
+
return cont.restartPolicy === container.ContainerRestartPolicy.ALWAYS;
|
|
115
|
+
}
|
|
116
|
+
addHostAlias(hostAlias) {
|
|
117
|
+
this._hostAliases.push(hostAlias);
|
|
118
|
+
}
|
|
119
|
+
addVolume(vol) {
|
|
120
|
+
const existingVolume = this._volumes.get(vol.name);
|
|
121
|
+
if (existingVolume) {
|
|
122
|
+
throw new Error(`Volume with name ${vol.name} already exists`);
|
|
123
|
+
}
|
|
124
|
+
this._volumes.set(vol.name, vol);
|
|
125
|
+
}
|
|
126
|
+
/**
|
|
127
|
+
* @see ISubect.toSubjectConfiguration()
|
|
128
|
+
*/
|
|
129
|
+
toSubjectConfiguration() {
|
|
130
|
+
if (!this.serviceAccount && !this.automountServiceAccountToken) {
|
|
131
|
+
throw new Error(`${this.name} cannot be converted to a role binding subject:`
|
|
132
|
+
+ ' You must either assign a service account to it, or use \'automountServiceAccountToken: true\'');
|
|
133
|
+
}
|
|
134
|
+
// 'default' is assumed to be the name of the default service account
|
|
135
|
+
// in the cluster.
|
|
136
|
+
const serviceAccountName = this.serviceAccount?.name ?? 'default';
|
|
137
|
+
return {
|
|
138
|
+
kind: 'ServiceAccount',
|
|
139
|
+
name: serviceAccountName,
|
|
140
|
+
apiGroup: '',
|
|
141
|
+
};
|
|
142
|
+
}
|
|
143
|
+
/**
|
|
144
|
+
* @internal
|
|
145
|
+
*/
|
|
146
|
+
_toPodSpec() {
|
|
147
|
+
if (this.containers.length === 0) {
|
|
148
|
+
throw new Error('PodSpec must have at least 1 container');
|
|
149
|
+
}
|
|
150
|
+
const volumes = new Map();
|
|
151
|
+
const containers = [];
|
|
152
|
+
const initContainers = [];
|
|
153
|
+
for (const cont of this.containers) {
|
|
154
|
+
// check if restartPolicy is defined for containers
|
|
155
|
+
if (cont.restartPolicy) {
|
|
156
|
+
throw new Error(`Invalid container spec: ${cont.name} has non-empty restartPolicy field. The field can only be specified for initContainers`);
|
|
157
|
+
}
|
|
158
|
+
// automatically add volume from the container mount
|
|
159
|
+
// to this pod so thats its available to the container.
|
|
160
|
+
for (const mount of cont.mounts) {
|
|
161
|
+
addVolume(mount.volume);
|
|
162
|
+
}
|
|
163
|
+
containers.push(cont._toKube());
|
|
164
|
+
}
|
|
165
|
+
for (const cont of this.initContainers) {
|
|
166
|
+
// automatically add volume from the container mount
|
|
167
|
+
// to this pod so thats its available to the container.
|
|
168
|
+
for (const mount of cont.mounts) {
|
|
169
|
+
addVolume(mount.volume);
|
|
170
|
+
}
|
|
171
|
+
initContainers.push(cont._toKube());
|
|
172
|
+
}
|
|
173
|
+
for (const vol of this.volumes) {
|
|
174
|
+
addVolume(vol);
|
|
175
|
+
}
|
|
176
|
+
function addVolume(vol) {
|
|
177
|
+
const existingVolume = volumes.get(vol.name);
|
|
178
|
+
// its ok to call this function twice on the same volume, but its not ok to
|
|
179
|
+
// call it twice on a different volume with the same name.
|
|
180
|
+
if (existingVolume && existingVolume !== vol) {
|
|
181
|
+
throw new Error(`Invalid mount configuration. At least two different volumes have the same name: ${vol.name}`);
|
|
182
|
+
}
|
|
183
|
+
volumes.set(vol.name, vol);
|
|
184
|
+
}
|
|
185
|
+
const dns = this.dns._toKube();
|
|
186
|
+
return {
|
|
187
|
+
restartPolicy: this.restartPolicy,
|
|
188
|
+
serviceAccountName: this.serviceAccount?.name,
|
|
189
|
+
containers: containers,
|
|
190
|
+
securityContext: (0, utils_1.undefinedIfEmpty)(this.securityContext._toKube()),
|
|
191
|
+
initContainers: (0, utils_1.undefinedIfEmpty)(initContainers),
|
|
192
|
+
hostAliases: (0, utils_1.undefinedIfEmpty)(this.hostAliases),
|
|
193
|
+
volumes: (0, utils_1.undefinedIfEmpty)(Array.from(volumes.values()).map(v => v._toKube())),
|
|
194
|
+
dnsPolicy: dns.policy,
|
|
195
|
+
dnsConfig: (0, utils_1.undefinedIfEmpty)(dns.config),
|
|
196
|
+
hostname: dns.hostname,
|
|
197
|
+
subdomain: dns.subdomain,
|
|
198
|
+
setHostnameAsFqdn: dns.hostnameAsFQDN,
|
|
199
|
+
imagePullSecrets: this.dockerRegistryAuth ? [{ name: this.dockerRegistryAuth.name }] : undefined,
|
|
200
|
+
automountServiceAccountToken: this.automountServiceAccountToken,
|
|
201
|
+
shareProcessNamespace: this.shareProcessNamespace,
|
|
202
|
+
hostNetwork: this.hostNetwork,
|
|
203
|
+
terminationGracePeriodSeconds: this.terminationGracePeriod?.toSeconds(),
|
|
204
|
+
enableServiceLinks: this.enableServiceLinks,
|
|
205
|
+
};
|
|
206
|
+
}
|
|
207
|
+
}
|
|
208
|
+
exports.AbstractPod = AbstractPod;
|
|
209
|
+
_a = JSII_RTTI_SYMBOL_1;
|
|
210
|
+
AbstractPod[_a] = { fqn: "cdk8s-plus-33.AbstractPod", version: "2.0.0" };
|
|
211
|
+
/**
|
|
212
|
+
* Match a resource by labels.
|
|
213
|
+
*/
|
|
214
|
+
class LabelSelector {
|
|
215
|
+
static of(options = {}) {
|
|
216
|
+
return new LabelSelector(options.expressions ?? [], options.labels ?? {});
|
|
217
|
+
}
|
|
218
|
+
constructor(expressions, labels) {
|
|
219
|
+
this.expressions = expressions;
|
|
220
|
+
this.labels = labels;
|
|
221
|
+
}
|
|
222
|
+
isEmpty() {
|
|
223
|
+
return this.expressions.length === 0 && Object.keys(this.labels).length === 0;
|
|
224
|
+
}
|
|
225
|
+
/**
|
|
226
|
+
* @internal
|
|
227
|
+
*/
|
|
228
|
+
_toKube() {
|
|
229
|
+
if (this.isEmpty()) {
|
|
230
|
+
return {};
|
|
231
|
+
}
|
|
232
|
+
return {
|
|
233
|
+
matchExpressions: (0, utils_1.undefinedIfEmpty)(this.expressions.map(q => ({ key: q.key, operator: q.operator, values: q.values }))),
|
|
234
|
+
matchLabels: (0, utils_1.undefinedIfEmpty)(this.labels),
|
|
235
|
+
};
|
|
236
|
+
}
|
|
237
|
+
}
|
|
238
|
+
exports.LabelSelector = LabelSelector;
|
|
239
|
+
_b = JSII_RTTI_SYMBOL_1;
|
|
240
|
+
LabelSelector[_b] = { fqn: "cdk8s-plus-33.LabelSelector", version: "2.0.0" };
|
|
241
|
+
/**
|
|
242
|
+
* Pod is a collection of containers that can run on a host. This resource is
|
|
243
|
+
* created by clients and scheduled onto hosts.
|
|
244
|
+
*/
|
|
245
|
+
class Pod extends AbstractPod {
|
|
246
|
+
constructor(scope, id, props = {}) {
|
|
247
|
+
super(scope, id, props);
|
|
248
|
+
this.resourceType = 'pods';
|
|
249
|
+
this.apiObject = new k8s.KubePod(this, 'Resource', {
|
|
250
|
+
metadata: props.metadata,
|
|
251
|
+
spec: cdk8s_1.Lazy.any({ produce: () => this._toKube() }),
|
|
252
|
+
});
|
|
253
|
+
this.metadata.addLabel(Pod.ADDRESS_LABEL, cdk8s_1.Names.toLabelValue(this));
|
|
254
|
+
this.scheduling = new PodScheduling(this);
|
|
255
|
+
this.connections = new PodConnections(this);
|
|
256
|
+
if (this.isolate) {
|
|
257
|
+
this.connections.isolate();
|
|
258
|
+
}
|
|
259
|
+
}
|
|
260
|
+
get podMetadata() {
|
|
261
|
+
return this.metadata;
|
|
262
|
+
}
|
|
263
|
+
/**
|
|
264
|
+
* @internal
|
|
265
|
+
*/
|
|
266
|
+
_toKube() {
|
|
267
|
+
const scheduling = this.scheduling._toKube();
|
|
268
|
+
return {
|
|
269
|
+
...this._toPodSpec(),
|
|
270
|
+
affinity: scheduling.affinity,
|
|
271
|
+
nodeName: scheduling.nodeName,
|
|
272
|
+
tolerations: scheduling.tolerations,
|
|
273
|
+
};
|
|
274
|
+
}
|
|
275
|
+
}
|
|
276
|
+
exports.Pod = Pod;
|
|
277
|
+
_c = JSII_RTTI_SYMBOL_1;
|
|
278
|
+
Pod[_c] = { fqn: "cdk8s-plus-33.Pod", version: "2.0.0" };
|
|
279
|
+
/**
|
|
280
|
+
* This label is autoamtically added by cdk8s to any pod. It provides
|
|
281
|
+
* a unique and stable identifier for the pod.
|
|
282
|
+
*/
|
|
283
|
+
Pod.ADDRESS_LABEL = 'cdk8s.io/metadata.addr';
|
|
284
|
+
/**
|
|
285
|
+
* Holds dns settings of the pod.
|
|
286
|
+
*/
|
|
287
|
+
class PodDns {
|
|
288
|
+
constructor(props = {}) {
|
|
289
|
+
this.hostname = props.hostname;
|
|
290
|
+
this.subdomain = props.subdomain;
|
|
291
|
+
this.policy = props.policy ?? DnsPolicy.CLUSTER_FIRST;
|
|
292
|
+
this.hostnameAsFQDN = props.hostnameAsFQDN ?? false;
|
|
293
|
+
this._nameservers = props.nameservers ?? [];
|
|
294
|
+
this._searches = props.searches ?? [];
|
|
295
|
+
this._options = props.options ?? [];
|
|
296
|
+
}
|
|
297
|
+
/**
|
|
298
|
+
* Nameservers defined for this pod.
|
|
299
|
+
*/
|
|
300
|
+
get nameservers() {
|
|
301
|
+
return [...this._nameservers];
|
|
302
|
+
}
|
|
303
|
+
/**
|
|
304
|
+
* Search domains defined for this pod.
|
|
305
|
+
*/
|
|
306
|
+
get searches() {
|
|
307
|
+
return [...this._searches];
|
|
308
|
+
}
|
|
309
|
+
/**
|
|
310
|
+
* Custom dns options defined for this pod.
|
|
311
|
+
*/
|
|
312
|
+
get options() {
|
|
313
|
+
return [...this._options];
|
|
314
|
+
}
|
|
315
|
+
/**
|
|
316
|
+
* Add a nameserver.
|
|
317
|
+
*/
|
|
318
|
+
addNameserver(...nameservers) {
|
|
319
|
+
this._nameservers.push(...nameservers);
|
|
320
|
+
}
|
|
321
|
+
/**
|
|
322
|
+
* Add a search domain.
|
|
323
|
+
*/
|
|
324
|
+
addSearch(...searches) {
|
|
325
|
+
this._searches.push(...searches);
|
|
326
|
+
}
|
|
327
|
+
/**
|
|
328
|
+
* Add a custom option.
|
|
329
|
+
*/
|
|
330
|
+
addOption(...options) {
|
|
331
|
+
this._options.push(...options);
|
|
332
|
+
}
|
|
333
|
+
/**
|
|
334
|
+
* @internal
|
|
335
|
+
*/
|
|
336
|
+
_toKube() {
|
|
337
|
+
if (this.policy === DnsPolicy.NONE && this.nameservers.length === 0) {
|
|
338
|
+
throw new Error('When dns policy is set to NONE, at least one nameserver is required');
|
|
339
|
+
}
|
|
340
|
+
if (this.nameservers.length > 3) {
|
|
341
|
+
throw new Error('There can be at most 3 nameservers specified');
|
|
342
|
+
}
|
|
343
|
+
if (this.searches.length > 6) {
|
|
344
|
+
throw new Error('There can be at most 6 search domains specified');
|
|
345
|
+
}
|
|
346
|
+
return {
|
|
347
|
+
hostname: this.hostname,
|
|
348
|
+
subdomain: this.subdomain,
|
|
349
|
+
hostnameAsFQDN: this.hostnameAsFQDN,
|
|
350
|
+
policy: this.policy,
|
|
351
|
+
config: {
|
|
352
|
+
nameservers: (0, utils_1.undefinedIfEmpty)(this.nameservers),
|
|
353
|
+
searches: (0, utils_1.undefinedIfEmpty)(this.searches),
|
|
354
|
+
options: (0, utils_1.undefinedIfEmpty)(this.options),
|
|
355
|
+
},
|
|
356
|
+
};
|
|
357
|
+
}
|
|
358
|
+
}
|
|
359
|
+
exports.PodDns = PodDns;
|
|
360
|
+
_d = JSII_RTTI_SYMBOL_1;
|
|
361
|
+
PodDns[_d] = { fqn: "cdk8s-plus-33.PodDns", version: "2.0.0" };
|
|
362
|
+
/**
|
|
363
|
+
* Holds pod-level security attributes and common container settings.
|
|
364
|
+
*/
|
|
365
|
+
class PodSecurityContext {
|
|
366
|
+
constructor(props = {}) {
|
|
367
|
+
this._sysctls = [];
|
|
368
|
+
this.ensureNonRoot = props.ensureNonRoot ?? true;
|
|
369
|
+
this.fsGroupChangePolicy = props.fsGroupChangePolicy ?? FsGroupChangePolicy.ALWAYS;
|
|
370
|
+
this.user = props.user;
|
|
371
|
+
this.group = props.group;
|
|
372
|
+
this.fsGroup = props.fsGroup;
|
|
373
|
+
for (const sysctl of props.sysctls ?? []) {
|
|
374
|
+
this._sysctls.push(sysctl);
|
|
375
|
+
}
|
|
376
|
+
}
|
|
377
|
+
get sysctls() {
|
|
378
|
+
return [...this._sysctls];
|
|
379
|
+
}
|
|
380
|
+
/**
|
|
381
|
+
* @internal
|
|
382
|
+
*/
|
|
383
|
+
_toKube() {
|
|
384
|
+
return {
|
|
385
|
+
runAsGroup: this.group,
|
|
386
|
+
runAsUser: this.user,
|
|
387
|
+
fsGroup: this.fsGroup,
|
|
388
|
+
runAsNonRoot: this.ensureNonRoot,
|
|
389
|
+
fsGroupChangePolicy: this.fsGroupChangePolicy,
|
|
390
|
+
sysctls: (0, utils_1.undefinedIfEmpty)(this._sysctls),
|
|
391
|
+
};
|
|
392
|
+
}
|
|
393
|
+
}
|
|
394
|
+
exports.PodSecurityContext = PodSecurityContext;
|
|
395
|
+
_e = JSII_RTTI_SYMBOL_1;
|
|
396
|
+
PodSecurityContext[_e] = { fqn: "cdk8s-plus-33.PodSecurityContext", version: "2.0.0" };
|
|
397
|
+
/**
|
|
398
|
+
* Restart policy for all containers within the pod.
|
|
399
|
+
*/
|
|
400
|
+
var RestartPolicy;
|
|
401
|
+
(function (RestartPolicy) {
|
|
402
|
+
/**
|
|
403
|
+
* Always restart the pod after it exits.
|
|
404
|
+
*/
|
|
405
|
+
RestartPolicy["ALWAYS"] = "Always";
|
|
406
|
+
/**
|
|
407
|
+
* Only restart if the pod exits with a non-zero exit code.
|
|
408
|
+
*/
|
|
409
|
+
RestartPolicy["ON_FAILURE"] = "OnFailure";
|
|
410
|
+
/**
|
|
411
|
+
* Never restart the pod.
|
|
412
|
+
*/
|
|
413
|
+
RestartPolicy["NEVER"] = "Never";
|
|
414
|
+
})(RestartPolicy || (exports.RestartPolicy = RestartPolicy = {}));
|
|
415
|
+
var FsGroupChangePolicy;
|
|
416
|
+
(function (FsGroupChangePolicy) {
|
|
417
|
+
/**
|
|
418
|
+
* Only change permissions and ownership if permission and ownership of root directory does
|
|
419
|
+
* not match with expected permissions of the volume.
|
|
420
|
+
* This could help shorten the time it takes to change ownership and permission of a volume
|
|
421
|
+
*/
|
|
422
|
+
FsGroupChangePolicy["ON_ROOT_MISMATCH"] = "OnRootMismatch";
|
|
423
|
+
/**
|
|
424
|
+
* Always change permission and ownership of the volume when volume is mounted.
|
|
425
|
+
*/
|
|
426
|
+
FsGroupChangePolicy["ALWAYS"] = "Always";
|
|
427
|
+
})(FsGroupChangePolicy || (exports.FsGroupChangePolicy = FsGroupChangePolicy = {}));
|
|
428
|
+
/**
|
|
429
|
+
* Pod DNS policies.
|
|
430
|
+
*/
|
|
431
|
+
var DnsPolicy;
|
|
432
|
+
(function (DnsPolicy) {
|
|
433
|
+
/**
|
|
434
|
+
* Any DNS query that does not match the configured cluster domain suffix,
|
|
435
|
+
* such as "www.kubernetes.io", is forwarded to the
|
|
436
|
+
* upstream nameserver inherited from the node.
|
|
437
|
+
* Cluster administrators may have extra stub-domain and upstream DNS servers configured.
|
|
438
|
+
*/
|
|
439
|
+
DnsPolicy["CLUSTER_FIRST"] = "ClusterFirst";
|
|
440
|
+
/**
|
|
441
|
+
* For Pods running with hostNetwork, you should
|
|
442
|
+
* explicitly set its DNS policy "ClusterFirstWithHostNet".
|
|
443
|
+
*/
|
|
444
|
+
DnsPolicy["CLUSTER_FIRST_WITH_HOST_NET"] = "ClusterFirstWithHostNet";
|
|
445
|
+
/**
|
|
446
|
+
* The Pod inherits the name resolution configuration
|
|
447
|
+
* from the node that the pods run on.
|
|
448
|
+
*/
|
|
449
|
+
DnsPolicy["DEFAULT"] = "Default";
|
|
450
|
+
/**
|
|
451
|
+
* It allows a Pod to ignore DNS settings from the Kubernetes environment.
|
|
452
|
+
* All DNS settings are supposed to be provided using the dnsConfig
|
|
453
|
+
* field in the Pod Spec.
|
|
454
|
+
*/
|
|
455
|
+
DnsPolicy["NONE"] = "None";
|
|
456
|
+
})(DnsPolicy || (exports.DnsPolicy = DnsPolicy = {}));
|
|
457
|
+
/**
|
|
458
|
+
* Represents a query that can be performed against nodes with labels.
|
|
459
|
+
*/
|
|
460
|
+
class NodeLabelQuery {
|
|
461
|
+
/**
|
|
462
|
+
* Requires value of label `key` to equal `value`.
|
|
463
|
+
*/
|
|
464
|
+
static is(key, value) {
|
|
465
|
+
return NodeLabelQuery.in(key, [value]);
|
|
466
|
+
}
|
|
467
|
+
/**
|
|
468
|
+
* Requires value of label `key` to be one of `values`.
|
|
469
|
+
*/
|
|
470
|
+
static in(key, values) {
|
|
471
|
+
return new NodeLabelQuery(key, 'In', values);
|
|
472
|
+
}
|
|
473
|
+
/**
|
|
474
|
+
* Requires value of label `key` to be none of `values`.
|
|
475
|
+
*/
|
|
476
|
+
static notIn(key, values) {
|
|
477
|
+
return new NodeLabelQuery(key, 'NotIn', values);
|
|
478
|
+
}
|
|
479
|
+
/**
|
|
480
|
+
* Requires label `key` to exist.
|
|
481
|
+
*/
|
|
482
|
+
static exists(key) {
|
|
483
|
+
return new NodeLabelQuery(key, 'Exists', undefined);
|
|
484
|
+
}
|
|
485
|
+
/**
|
|
486
|
+
* Requires label `key` to not exist.
|
|
487
|
+
*/
|
|
488
|
+
static doesNotExist(key) {
|
|
489
|
+
return new NodeLabelQuery(key, 'DoesNotExist', undefined);
|
|
490
|
+
}
|
|
491
|
+
/**
|
|
492
|
+
* Requires value of label `key` to greater than all elements in `values`.
|
|
493
|
+
*/
|
|
494
|
+
static gt(key, values) {
|
|
495
|
+
return new NodeLabelQuery(key, 'Gt', values);
|
|
496
|
+
}
|
|
497
|
+
/**
|
|
498
|
+
* Requires value of label `key` to less than all elements in `values`.
|
|
499
|
+
*/
|
|
500
|
+
static lt(key, values) {
|
|
501
|
+
return new NodeLabelQuery(key, 'Lt', values);
|
|
502
|
+
}
|
|
503
|
+
constructor(key, operator, values) {
|
|
504
|
+
this.key = key;
|
|
505
|
+
this.operator = operator;
|
|
506
|
+
this.values = values;
|
|
507
|
+
}
|
|
508
|
+
/**
|
|
509
|
+
* @internal
|
|
510
|
+
*/
|
|
511
|
+
_toKube() {
|
|
512
|
+
return {
|
|
513
|
+
key: this.key,
|
|
514
|
+
operator: this.operator,
|
|
515
|
+
values: this.values,
|
|
516
|
+
};
|
|
517
|
+
}
|
|
518
|
+
}
|
|
519
|
+
exports.NodeLabelQuery = NodeLabelQuery;
|
|
520
|
+
_f = JSII_RTTI_SYMBOL_1;
|
|
521
|
+
NodeLabelQuery[_f] = { fqn: "cdk8s-plus-33.NodeLabelQuery", version: "2.0.0" };
|
|
522
|
+
/**
|
|
523
|
+
* Represents a query that can be performed against resources with labels.
|
|
524
|
+
*/
|
|
525
|
+
class LabelExpression {
|
|
526
|
+
/**
|
|
527
|
+
* Requires value of label `key` to be one of `values`.
|
|
528
|
+
*/
|
|
529
|
+
static in(key, values) {
|
|
530
|
+
return new LabelExpression(key, 'In', values);
|
|
531
|
+
}
|
|
532
|
+
/**
|
|
533
|
+
* Requires value of label `key` to be none of `values`.
|
|
534
|
+
*/
|
|
535
|
+
static notIn(key, values) {
|
|
536
|
+
return new LabelExpression(key, 'NotIn', values);
|
|
537
|
+
}
|
|
538
|
+
/**
|
|
539
|
+
* Requires label `key` to exist.
|
|
540
|
+
*/
|
|
541
|
+
static exists(key) {
|
|
542
|
+
return new LabelExpression(key, 'Exists', undefined);
|
|
543
|
+
}
|
|
544
|
+
/**
|
|
545
|
+
* Requires label `key` to not exist.
|
|
546
|
+
*/
|
|
547
|
+
static doesNotExist(key) {
|
|
548
|
+
return new LabelExpression(key, 'DoesNotExist', undefined);
|
|
549
|
+
}
|
|
550
|
+
constructor(key, operator, values) {
|
|
551
|
+
this.key = key;
|
|
552
|
+
this.operator = operator;
|
|
553
|
+
this.values = values;
|
|
554
|
+
}
|
|
555
|
+
}
|
|
556
|
+
exports.LabelExpression = LabelExpression;
|
|
557
|
+
_g = JSII_RTTI_SYMBOL_1;
|
|
558
|
+
LabelExpression[_g] = { fqn: "cdk8s-plus-33.LabelExpression", version: "2.0.0" };
|
|
559
|
+
/**
|
|
560
|
+
* Taint effects.
|
|
561
|
+
*/
|
|
562
|
+
var TaintEffect;
|
|
563
|
+
(function (TaintEffect) {
|
|
564
|
+
/**
|
|
565
|
+
* This means that no pod will be able to schedule
|
|
566
|
+
* onto the node unless it has a matching toleration.
|
|
567
|
+
*/
|
|
568
|
+
TaintEffect["NO_SCHEDULE"] = "NoSchedule";
|
|
569
|
+
/**
|
|
570
|
+
* This is a "preference" or "soft" version of `NO_SCHEDULE` -- the system
|
|
571
|
+
* will try to avoid placing a pod that does not tolerate the taint on the node,
|
|
572
|
+
* but it is not required
|
|
573
|
+
*/
|
|
574
|
+
TaintEffect["PREFER_NO_SCHEDULE"] = "PreferNoSchedule";
|
|
575
|
+
/**
|
|
576
|
+
* This affects pods that are already running on the node as follows:
|
|
577
|
+
*
|
|
578
|
+
* - Pods that do not tolerate the taint are evicted immediately.
|
|
579
|
+
* - Pods that tolerate the taint without specifying `duration` remain bound forever.
|
|
580
|
+
* - Pods that tolerate the taint with a specified `duration` remain bound for
|
|
581
|
+
* the specified amount of time.
|
|
582
|
+
*/
|
|
583
|
+
TaintEffect["NO_EXECUTE"] = "NoExecute";
|
|
584
|
+
})(TaintEffect || (exports.TaintEffect = TaintEffect = {}));
|
|
585
|
+
/**
|
|
586
|
+
* Taint queries that can be perfomed against nodes.
|
|
587
|
+
*/
|
|
588
|
+
class NodeTaintQuery {
|
|
589
|
+
/**
|
|
590
|
+
* Matches a taint with a specific key and value.
|
|
591
|
+
*/
|
|
592
|
+
static is(key, value, options = {}) {
|
|
593
|
+
return new NodeTaintQuery('Equal', key, value, options.effect, options.evictAfter);
|
|
594
|
+
}
|
|
595
|
+
/**
|
|
596
|
+
* Matches a tain with any value of a specific key.
|
|
597
|
+
*/
|
|
598
|
+
static exists(key, options = {}) {
|
|
599
|
+
return new NodeTaintQuery('Exists', key, undefined, options.effect, options.evictAfter);
|
|
600
|
+
}
|
|
601
|
+
/**
|
|
602
|
+
* Matches any taint.
|
|
603
|
+
*/
|
|
604
|
+
static any() {
|
|
605
|
+
return new NodeTaintQuery('Exists');
|
|
606
|
+
}
|
|
607
|
+
constructor(operator, key, value, effect, evictAfter) {
|
|
608
|
+
this.operator = operator;
|
|
609
|
+
this.key = key;
|
|
610
|
+
this.value = value;
|
|
611
|
+
this.effect = effect;
|
|
612
|
+
this.evictAfter = evictAfter;
|
|
613
|
+
if (evictAfter && effect !== TaintEffect.NO_EXECUTE) {
|
|
614
|
+
throw new Error('Only \'NO_EXECUTE\' effects can specify \'evictAfter\'');
|
|
615
|
+
}
|
|
616
|
+
}
|
|
617
|
+
/**
|
|
618
|
+
* @internal
|
|
619
|
+
*/
|
|
620
|
+
_toKube() {
|
|
621
|
+
return {
|
|
622
|
+
effect: this.effect,
|
|
623
|
+
key: this.key,
|
|
624
|
+
operator: this.operator,
|
|
625
|
+
tolerationSeconds: this.evictAfter?.toSeconds(),
|
|
626
|
+
value: this.value,
|
|
627
|
+
};
|
|
628
|
+
}
|
|
629
|
+
}
|
|
630
|
+
exports.NodeTaintQuery = NodeTaintQuery;
|
|
631
|
+
_h = JSII_RTTI_SYMBOL_1;
|
|
632
|
+
NodeTaintQuery[_h] = { fqn: "cdk8s-plus-33.NodeTaintQuery", version: "2.0.0" };
|
|
633
|
+
/**
|
|
634
|
+
* Represents a group of pods.
|
|
635
|
+
*/
|
|
636
|
+
class Pods extends constructs_1.Construct {
|
|
637
|
+
/**
|
|
638
|
+
* Select pods in the cluster with various selectors.
|
|
639
|
+
*/
|
|
640
|
+
static select(scope, id, options) {
|
|
641
|
+
return new Pods(scope, id, options.expressions, options.labels, options.namespaces);
|
|
642
|
+
}
|
|
643
|
+
/**
|
|
644
|
+
* Select all pods.
|
|
645
|
+
*/
|
|
646
|
+
static all(scope, id, options = {}) {
|
|
647
|
+
return Pods.select(scope, id, { namespaces: options.namespaces });
|
|
648
|
+
}
|
|
649
|
+
constructor(scope, id, expressions, labels, namespaces) {
|
|
650
|
+
super(scope, id);
|
|
651
|
+
this.expressions = expressions;
|
|
652
|
+
this.labels = labels;
|
|
653
|
+
this.namespaces = namespaces;
|
|
654
|
+
}
|
|
655
|
+
/**
|
|
656
|
+
* @see IPodSelector.toPodSelectorConfig()
|
|
657
|
+
*/
|
|
658
|
+
toPodSelectorConfig() {
|
|
659
|
+
return {
|
|
660
|
+
labelSelector: LabelSelector.of({ expressions: this.expressions, labels: this.labels }),
|
|
661
|
+
namespaces: this.namespaces?.toNamespaceSelectorConfig(),
|
|
662
|
+
};
|
|
663
|
+
}
|
|
664
|
+
/**
|
|
665
|
+
* @see INetworkPolicyPeer.toNetworkPolicyPeerConfig()
|
|
666
|
+
*/
|
|
667
|
+
toNetworkPolicyPeerConfig() {
|
|
668
|
+
return { podSelector: this.toPodSelectorConfig() };
|
|
669
|
+
}
|
|
670
|
+
/**
|
|
671
|
+
* @see INetworkPolicyPeer.toPodSelector()
|
|
672
|
+
*/
|
|
673
|
+
toPodSelector() {
|
|
674
|
+
return this;
|
|
675
|
+
}
|
|
676
|
+
}
|
|
677
|
+
exports.Pods = Pods;
|
|
678
|
+
_j = JSII_RTTI_SYMBOL_1;
|
|
679
|
+
Pods[_j] = { fqn: "cdk8s-plus-33.Pods", version: "2.0.0" };
|
|
680
|
+
/**
|
|
681
|
+
* A node that is matched by label selectors.
|
|
682
|
+
*/
|
|
683
|
+
class LabeledNode {
|
|
684
|
+
constructor(labelSelector) {
|
|
685
|
+
this.labelSelector = labelSelector;
|
|
686
|
+
}
|
|
687
|
+
;
|
|
688
|
+
}
|
|
689
|
+
exports.LabeledNode = LabeledNode;
|
|
690
|
+
_k = JSII_RTTI_SYMBOL_1;
|
|
691
|
+
LabeledNode[_k] = { fqn: "cdk8s-plus-33.LabeledNode", version: "2.0.0" };
|
|
692
|
+
/**
|
|
693
|
+
* A node that is matched by taint selectors.
|
|
694
|
+
*/
|
|
695
|
+
class TaintedNode {
|
|
696
|
+
constructor(taintSelector) {
|
|
697
|
+
this.taintSelector = taintSelector;
|
|
698
|
+
}
|
|
699
|
+
;
|
|
700
|
+
}
|
|
701
|
+
exports.TaintedNode = TaintedNode;
|
|
702
|
+
_l = JSII_RTTI_SYMBOL_1;
|
|
703
|
+
TaintedNode[_l] = { fqn: "cdk8s-plus-33.TaintedNode", version: "2.0.0" };
|
|
704
|
+
/**
|
|
705
|
+
* A node that is matched by its name.
|
|
706
|
+
*/
|
|
707
|
+
class NamedNode {
|
|
708
|
+
constructor(name) {
|
|
709
|
+
this.name = name;
|
|
710
|
+
}
|
|
711
|
+
;
|
|
712
|
+
}
|
|
713
|
+
exports.NamedNode = NamedNode;
|
|
714
|
+
_m = JSII_RTTI_SYMBOL_1;
|
|
715
|
+
NamedNode[_m] = { fqn: "cdk8s-plus-33.NamedNode", version: "2.0.0" };
|
|
716
|
+
/**
|
|
717
|
+
* Represents a node in the cluster.
|
|
718
|
+
*/
|
|
719
|
+
class Node {
|
|
720
|
+
/**
|
|
721
|
+
* Match a node by its labels.
|
|
722
|
+
*/
|
|
723
|
+
static labeled(...labelSelector) {
|
|
724
|
+
return new LabeledNode(labelSelector);
|
|
725
|
+
}
|
|
726
|
+
/**
|
|
727
|
+
* Match a node by its name.
|
|
728
|
+
*/
|
|
729
|
+
static named(nodeName) {
|
|
730
|
+
return new NamedNode(nodeName);
|
|
731
|
+
}
|
|
732
|
+
/**
|
|
733
|
+
* Match a node by its taints.
|
|
734
|
+
*/
|
|
735
|
+
static tainted(...taintSelector) {
|
|
736
|
+
return new TaintedNode(taintSelector);
|
|
737
|
+
}
|
|
738
|
+
}
|
|
739
|
+
exports.Node = Node;
|
|
740
|
+
_o = JSII_RTTI_SYMBOL_1;
|
|
741
|
+
Node[_o] = { fqn: "cdk8s-plus-33.Node", version: "2.0.0" };
|
|
742
|
+
/**
|
|
743
|
+
* Available topology domains.
|
|
744
|
+
*/
|
|
745
|
+
class Topology {
|
|
746
|
+
/**
|
|
747
|
+
* Custom key for the node label that the system uses to denote the topology domain.
|
|
748
|
+
*/
|
|
749
|
+
static custom(key) {
|
|
750
|
+
return new Topology(key);
|
|
751
|
+
}
|
|
752
|
+
constructor(key) {
|
|
753
|
+
this.key = key;
|
|
754
|
+
}
|
|
755
|
+
;
|
|
756
|
+
}
|
|
757
|
+
exports.Topology = Topology;
|
|
758
|
+
_p = JSII_RTTI_SYMBOL_1;
|
|
759
|
+
Topology[_p] = { fqn: "cdk8s-plus-33.Topology", version: "2.0.0" };
|
|
760
|
+
/**
|
|
761
|
+
* A hostname represents a single node in the cluster.
|
|
762
|
+
*
|
|
763
|
+
* @see https://kubernetes.io/docs/reference/labels-annotations-taints/#kubernetesiohostname
|
|
764
|
+
*/
|
|
765
|
+
Topology.HOSTNAME = new Topology('kubernetes.io/hostname');
|
|
766
|
+
/**
|
|
767
|
+
* A zone represents a logical failure domain. It is common for Kubernetes clusters to
|
|
768
|
+
* span multiple zones for increased availability. While the exact definition of a zone is
|
|
769
|
+
* left to infrastructure implementations, common properties of a zone include very low
|
|
770
|
+
* network latency within a zone, no-cost network traffic within a zone, and failure
|
|
771
|
+
* independence from other zones. For example, nodes within a zone might share a network
|
|
772
|
+
* switch, but nodes in different zones should not.
|
|
773
|
+
*
|
|
774
|
+
* @see https://kubernetes.io/docs/reference/labels-annotations-taints/#topologykubernetesiozone
|
|
775
|
+
*/
|
|
776
|
+
Topology.ZONE = new Topology('topology.kubernetes.io/zone');
|
|
777
|
+
/**
|
|
778
|
+
* A region represents a larger domain, made up of one or more zones. It is uncommon
|
|
779
|
+
* for Kubernetes clusters to span multiple regions. While the exact definition of a
|
|
780
|
+
* zone or region is left to infrastructure implementations, common properties of a region
|
|
781
|
+
* include higher network latency between them than within them, non-zero cost for network
|
|
782
|
+
* traffic between them, and failure independence from other zones or regions.
|
|
783
|
+
*
|
|
784
|
+
* For example, nodes within a region might share power infrastructure (e.g. a UPS or generator), but
|
|
785
|
+
* nodes in different regions typically would not.
|
|
786
|
+
*
|
|
787
|
+
* @see https://kubernetes.io/docs/reference/labels-annotations-taints/#topologykubernetesioregion
|
|
788
|
+
*/
|
|
789
|
+
Topology.REGION = new Topology('topology.kubernetes.io/region');
|
|
790
|
+
/**
|
|
791
|
+
* Controls the pod scheduling strategy.
|
|
792
|
+
*/
|
|
793
|
+
class PodScheduling {
|
|
794
|
+
constructor(instance) {
|
|
795
|
+
this.instance = instance;
|
|
796
|
+
this._nodeAffinityPreferred = [];
|
|
797
|
+
this._nodeAffinityRequired = [];
|
|
798
|
+
this._podAffinityPreferred = [];
|
|
799
|
+
this._podAffinityRequired = [];
|
|
800
|
+
this._podAntiAffinityPreferred = [];
|
|
801
|
+
this._podAntiAffinityRequired = [];
|
|
802
|
+
this._tolerations = [];
|
|
803
|
+
}
|
|
804
|
+
/**
|
|
805
|
+
* Assign this pod a specific node by name.
|
|
806
|
+
*
|
|
807
|
+
* The scheduler ignores the Pod, and the kubelet on the named node
|
|
808
|
+
* tries to place the Pod on that node. Overrules any affinity rules of the pod.
|
|
809
|
+
*
|
|
810
|
+
* Some limitations of static assignment are:
|
|
811
|
+
*
|
|
812
|
+
* - If the named node does not exist, the Pod will not run, and in some
|
|
813
|
+
* cases may be automatically deleted.
|
|
814
|
+
* - If the named node does not have the resources to accommodate the Pod,
|
|
815
|
+
* the Pod will fail and its reason will indicate why, for example OutOfmemory or OutOfcpu.
|
|
816
|
+
* - Node names in cloud environments are not always predictable or stable.
|
|
817
|
+
*
|
|
818
|
+
* Will throw is the pod is already assigned to named node.
|
|
819
|
+
*
|
|
820
|
+
* Under the hood, this method utilizes the `nodeName` property.
|
|
821
|
+
*/
|
|
822
|
+
assign(node) {
|
|
823
|
+
if (this._nodeName) {
|
|
824
|
+
// disallow overriding an static node assignment
|
|
825
|
+
throw new Error(`Cannot assign ${this.instance.podMetadata.name} to node ${node.name}. It is already assigned to node ${this._nodeName}`);
|
|
826
|
+
}
|
|
827
|
+
else {
|
|
828
|
+
this._nodeName = node.name;
|
|
829
|
+
}
|
|
830
|
+
}
|
|
831
|
+
/**
|
|
832
|
+
* Allow this pod to tolerate taints matching these tolerations.
|
|
833
|
+
*
|
|
834
|
+
* You can put multiple taints on the same node and multiple tolerations on the same pod.
|
|
835
|
+
* The way Kubernetes processes multiple taints and tolerations is like a filter: start with
|
|
836
|
+
* all of a node's taints, then ignore the ones for which the pod has a matching toleration;
|
|
837
|
+
* the remaining un-ignored taints have the indicated effects on the pod. In particular:
|
|
838
|
+
*
|
|
839
|
+
* - if there is at least one un-ignored taint with effect NoSchedule then Kubernetes will
|
|
840
|
+
* not schedule the pod onto that node
|
|
841
|
+
* - if there is no un-ignored taint with effect NoSchedule but there is at least one un-ignored
|
|
842
|
+
* taint with effect PreferNoSchedule then Kubernetes will try to not schedule the pod onto the node
|
|
843
|
+
* - if there is at least one un-ignored taint with effect NoExecute then the pod will be evicted from
|
|
844
|
+
* the node (if it is already running on the node), and will not be scheduled onto the node (if it is
|
|
845
|
+
* not yet running on the node).
|
|
846
|
+
*
|
|
847
|
+
* Under the hood, this method utilizes the `tolerations` property.
|
|
848
|
+
*
|
|
849
|
+
* @see https://kubernetes.io/docs/concepts/scheduling-eviction/taint-and-toleration/
|
|
850
|
+
*/
|
|
851
|
+
tolerate(node) {
|
|
852
|
+
for (const query of node.taintSelector) {
|
|
853
|
+
this._tolerations.push(query._toKube());
|
|
854
|
+
}
|
|
855
|
+
}
|
|
856
|
+
/**
|
|
857
|
+
* Attract this pod to a node matched by selectors.
|
|
858
|
+
* You can select a node by using `Node.labeled()`.
|
|
859
|
+
*
|
|
860
|
+
* Attracting to multiple nodes (i.e invoking this method multiple times) acts as
|
|
861
|
+
* an OR condition, meaning the pod will be assigned to either one of the nodes.
|
|
862
|
+
*
|
|
863
|
+
* Under the hood, this method utilizes the `nodeAffinity` property.
|
|
864
|
+
*
|
|
865
|
+
* @see https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#node-affinity
|
|
866
|
+
*/
|
|
867
|
+
attract(node, options = {}) {
|
|
868
|
+
const term = this.createNodeAffinityTerm(node);
|
|
869
|
+
if (options.weight) {
|
|
870
|
+
this.validateWeight(options.weight);
|
|
871
|
+
this._nodeAffinityPreferred.push({ weight: options.weight, preference: term });
|
|
872
|
+
}
|
|
873
|
+
else {
|
|
874
|
+
this._nodeAffinityRequired.push(term);
|
|
875
|
+
}
|
|
876
|
+
}
|
|
877
|
+
/**
|
|
878
|
+
* Co-locate this pod with a scheduling selection.
|
|
879
|
+
*
|
|
880
|
+
* A selection can be one of:
|
|
881
|
+
*
|
|
882
|
+
* - An instance of a `Pod`.
|
|
883
|
+
* - An instance of a `Workload` (e.g `Deployment`, `StatefulSet`).
|
|
884
|
+
* - An un-managed pod that can be selected via `Pods.select()`.
|
|
885
|
+
*
|
|
886
|
+
* Co-locating with multiple selections ((i.e invoking this method multiple times)) acts as
|
|
887
|
+
* an AND condition. meaning the pod will be assigned to a node that satisfies all
|
|
888
|
+
* selections (i.e runs at least one pod that satisifies each selection).
|
|
889
|
+
*
|
|
890
|
+
* Under the hood, this method utilizes the `podAffinity` property.
|
|
891
|
+
*
|
|
892
|
+
* @see https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#inter-pod-affinity-and-anti-affinity
|
|
893
|
+
*/
|
|
894
|
+
colocate(selector, options = {}) {
|
|
895
|
+
const topology = options.topology ?? Topology.HOSTNAME;
|
|
896
|
+
const term = this.createPodAffinityTerm(topology, selector);
|
|
897
|
+
if (options.weight) {
|
|
898
|
+
this.validateWeight(options.weight);
|
|
899
|
+
this._podAffinityPreferred.push({ weight: options.weight, podAffinityTerm: term });
|
|
900
|
+
}
|
|
901
|
+
else {
|
|
902
|
+
this._podAffinityRequired.push(term);
|
|
903
|
+
}
|
|
904
|
+
}
|
|
905
|
+
/**
|
|
906
|
+
* Seperate this pod from a scheduling selection.
|
|
907
|
+
*
|
|
908
|
+
* A selection can be one of:
|
|
909
|
+
*
|
|
910
|
+
* - An instance of a `Pod`.
|
|
911
|
+
* - An instance of a `Workload` (e.g `Deployment`, `StatefulSet`).
|
|
912
|
+
* - An un-managed pod that can be selected via `Pods.select()`.
|
|
913
|
+
*
|
|
914
|
+
* Seperating from multiple selections acts as an AND condition. meaning the pod
|
|
915
|
+
* will not be assigned to a node that satisfies all selections (i.e runs at least one pod that satisifies each selection).
|
|
916
|
+
*
|
|
917
|
+
* Under the hood, this method utilizes the `podAntiAffinity` property.
|
|
918
|
+
*
|
|
919
|
+
* @see https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#inter-pod-affinity-and-anti-affinity
|
|
920
|
+
*/
|
|
921
|
+
separate(selector, options = {}) {
|
|
922
|
+
const topology = options.topology ?? Topology.HOSTNAME;
|
|
923
|
+
const term = this.createPodAffinityTerm(topology, selector);
|
|
924
|
+
if (options.weight) {
|
|
925
|
+
this.validateWeight(options.weight);
|
|
926
|
+
this._podAntiAffinityPreferred.push({ weight: options.weight, podAffinityTerm: term });
|
|
927
|
+
}
|
|
928
|
+
else {
|
|
929
|
+
this._podAntiAffinityRequired.push(term);
|
|
930
|
+
}
|
|
931
|
+
}
|
|
932
|
+
createPodAffinityTerm(topology, selector) {
|
|
933
|
+
const config = selector.toPodSelectorConfig();
|
|
934
|
+
return {
|
|
935
|
+
topologyKey: topology.key,
|
|
936
|
+
labelSelector: config.labelSelector._toKube(),
|
|
937
|
+
namespaceSelector: config.namespaces?.labelSelector?._toKube(),
|
|
938
|
+
namespaces: config.namespaces?.names,
|
|
939
|
+
};
|
|
940
|
+
}
|
|
941
|
+
createNodeAffinityTerm(node) {
|
|
942
|
+
return { matchExpressions: node.labelSelector.map(s => s._toKube()) };
|
|
943
|
+
}
|
|
944
|
+
validateWeight(weight) {
|
|
945
|
+
if (weight < 1 || weight > 100) {
|
|
946
|
+
// https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#node-affinity-weight
|
|
947
|
+
throw new Error(`Invalid affinity weight: ${weight}. Must be in range 1-100`);
|
|
948
|
+
}
|
|
949
|
+
}
|
|
950
|
+
/**
|
|
951
|
+
* @internal
|
|
952
|
+
*/
|
|
953
|
+
_toKube() {
|
|
954
|
+
const atLeastOne = (...arrays) => {
|
|
955
|
+
return arrays.flat().length > 0;
|
|
956
|
+
};
|
|
957
|
+
const hasNodeAffinity = atLeastOne(this._nodeAffinityPreferred, this._nodeAffinityRequired);
|
|
958
|
+
const hasPodAffinity = atLeastOne(this._podAffinityPreferred, this._podAffinityRequired);
|
|
959
|
+
const hasPodAntiAffinty = atLeastOne(this._podAntiAffinityPreferred, this._podAntiAffinityRequired);
|
|
960
|
+
const hasAffinity = hasNodeAffinity || hasPodAffinity || hasPodAntiAffinty;
|
|
961
|
+
return {
|
|
962
|
+
affinity: hasAffinity ? {
|
|
963
|
+
nodeAffinity: hasNodeAffinity ? {
|
|
964
|
+
preferredDuringSchedulingIgnoredDuringExecution: (0, utils_1.undefinedIfEmpty)(this._nodeAffinityPreferred),
|
|
965
|
+
requiredDuringSchedulingIgnoredDuringExecution: this._nodeAffinityRequired.length > 0 ? {
|
|
966
|
+
nodeSelectorTerms: this._nodeAffinityRequired,
|
|
967
|
+
} : undefined,
|
|
968
|
+
} : undefined,
|
|
969
|
+
podAffinity: hasPodAffinity ? {
|
|
970
|
+
preferredDuringSchedulingIgnoredDuringExecution: (0, utils_1.undefinedIfEmpty)(this._podAffinityPreferred),
|
|
971
|
+
requiredDuringSchedulingIgnoredDuringExecution: (0, utils_1.undefinedIfEmpty)(this._podAffinityRequired),
|
|
972
|
+
} : undefined,
|
|
973
|
+
podAntiAffinity: hasPodAntiAffinty ? {
|
|
974
|
+
preferredDuringSchedulingIgnoredDuringExecution: (0, utils_1.undefinedIfEmpty)(this._podAntiAffinityPreferred),
|
|
975
|
+
requiredDuringSchedulingIgnoredDuringExecution: (0, utils_1.undefinedIfEmpty)(this._podAntiAffinityRequired),
|
|
976
|
+
} : undefined,
|
|
977
|
+
} : undefined,
|
|
978
|
+
nodeName: this._nodeName,
|
|
979
|
+
tolerations: (0, utils_1.undefinedIfEmpty)(this._tolerations),
|
|
980
|
+
};
|
|
981
|
+
}
|
|
982
|
+
}
|
|
983
|
+
exports.PodScheduling = PodScheduling;
|
|
984
|
+
_q = JSII_RTTI_SYMBOL_1;
|
|
985
|
+
PodScheduling[_q] = { fqn: "cdk8s-plus-33.PodScheduling", version: "2.0.0" };
|
|
986
|
+
/**
|
|
987
|
+
* Isolation determines which policies are created
|
|
988
|
+
* when allowing connections from a a pod / workload to peers.
|
|
989
|
+
*/
|
|
990
|
+
var PodConnectionsIsolation;
|
|
991
|
+
(function (PodConnectionsIsolation) {
|
|
992
|
+
/**
|
|
993
|
+
* Only creates network policies that select the pod.
|
|
994
|
+
*/
|
|
995
|
+
PodConnectionsIsolation["POD"] = "POD";
|
|
996
|
+
/**
|
|
997
|
+
* Only creates network policies that select the peer.
|
|
998
|
+
*/
|
|
999
|
+
PodConnectionsIsolation["PEER"] = "PEER";
|
|
1000
|
+
})(PodConnectionsIsolation || (exports.PodConnectionsIsolation = PodConnectionsIsolation = {}));
|
|
1001
|
+
/**
|
|
1002
|
+
* Controls network isolation rules for inter-pod communication.
|
|
1003
|
+
*/
|
|
1004
|
+
class PodConnections {
|
|
1005
|
+
constructor(instance) {
|
|
1006
|
+
this.instance = instance;
|
|
1007
|
+
}
|
|
1008
|
+
/**
|
|
1009
|
+
* Allow network traffic from this pod to the peer.
|
|
1010
|
+
*
|
|
1011
|
+
* By default, this will create an egress network policy for this pod, and an ingress
|
|
1012
|
+
* network policy for the peer. This is required if both sides are already isolated.
|
|
1013
|
+
* Use `options.isolation` to control this behavior.
|
|
1014
|
+
*
|
|
1015
|
+
* @example
|
|
1016
|
+
*
|
|
1017
|
+
* // create only an egress policy that selects the 'web' pod to allow outgoing traffic
|
|
1018
|
+
* // to the 'redis' pod. this requires the 'redis' pod to not be isolated for ingress.
|
|
1019
|
+
* web.connections.allowTo(redis, { isolation: Isolation.POD })
|
|
1020
|
+
*
|
|
1021
|
+
* // create only an ingress policy that selects the 'redis' peer to allow incoming traffic
|
|
1022
|
+
* // from the 'web' pod. this requires the 'web' pod to not be isolated for egress.
|
|
1023
|
+
* web.connections.allowTo(redis, { isolation: Isolation.PEER })
|
|
1024
|
+
*
|
|
1025
|
+
*/
|
|
1026
|
+
allowTo(peer, options = {}) {
|
|
1027
|
+
return this.allow('Egress', peer, { ports: this.extractPorts(peer), ...options });
|
|
1028
|
+
}
|
|
1029
|
+
/**
|
|
1030
|
+
* Allow network traffic from the peer to this pod.
|
|
1031
|
+
*
|
|
1032
|
+
* By default, this will create an ingress network policy for this pod, and an egress
|
|
1033
|
+
* network policy for the peer. This is required if both sides are already isolated.
|
|
1034
|
+
* Use `options.isolation` to control this behavior.
|
|
1035
|
+
*
|
|
1036
|
+
* @example
|
|
1037
|
+
*
|
|
1038
|
+
* // create only an egress policy that selects the 'web' pod to allow outgoing traffic
|
|
1039
|
+
* // to the 'redis' pod. this requires the 'redis' pod to not be isolated for ingress.
|
|
1040
|
+
* redis.connections.allowFrom(web, { isolation: Isolation.PEER })
|
|
1041
|
+
*
|
|
1042
|
+
* // create only an ingress policy that selects the 'redis' peer to allow incoming traffic
|
|
1043
|
+
* // from the 'web' pod. this requires the 'web' pod to not be isolated for egress.
|
|
1044
|
+
* redis.connections.allowFrom(web, { isolation: Isolation.POD })
|
|
1045
|
+
*
|
|
1046
|
+
*/
|
|
1047
|
+
allowFrom(peer, options = {}) {
|
|
1048
|
+
return this.allow('Ingress', peer, { ports: this.extractPorts(this.instance), ...options });
|
|
1049
|
+
}
|
|
1050
|
+
allow(direction, peer, options = {}) {
|
|
1051
|
+
const config = peer.toNetworkPolicyPeerConfig();
|
|
1052
|
+
networkpolicy.validatePeerConfig(config);
|
|
1053
|
+
const peerAddress = (0, utils_1.address)(peer);
|
|
1054
|
+
if (!options.isolation || options.isolation === PodConnectionsIsolation.POD) {
|
|
1055
|
+
const src = new networkpolicy.NetworkPolicy(this.instance, `Allow${direction}${peerAddress}`, {
|
|
1056
|
+
selector: this.instance,
|
|
1057
|
+
// the policy must be defined in the namespace of the pod
|
|
1058
|
+
// so it can select it.
|
|
1059
|
+
metadata: { namespace: this.instance.metadata.namespace },
|
|
1060
|
+
});
|
|
1061
|
+
switch (direction) {
|
|
1062
|
+
case 'Egress':
|
|
1063
|
+
src.addEgressRule(peer, options.ports);
|
|
1064
|
+
break;
|
|
1065
|
+
case 'Ingress':
|
|
1066
|
+
src.addIngressRule(peer, options.ports);
|
|
1067
|
+
}
|
|
1068
|
+
}
|
|
1069
|
+
if (!options.isolation || options.isolation === PodConnectionsIsolation.PEER) {
|
|
1070
|
+
if (config.ipBlock) {
|
|
1071
|
+
// for an ip block we don't need to create the opposite policies
|
|
1072
|
+
return;
|
|
1073
|
+
}
|
|
1074
|
+
const podSelector = peer.toPodSelector();
|
|
1075
|
+
if (!podSelector) {
|
|
1076
|
+
throw new Error(`Unable to create policies for peer '${peer.node.addr}' since its not a pod selector`);
|
|
1077
|
+
}
|
|
1078
|
+
const oppositeDirection = direction === 'Egress' ? 'Ingress' : 'Egress';
|
|
1079
|
+
const podSelectorConfig = podSelector.toPodSelectorConfig();
|
|
1080
|
+
let namespaces;
|
|
1081
|
+
if (!podSelectorConfig.namespaces) {
|
|
1082
|
+
// if the peer doesn't specify namespaces, we assume the same namespace.
|
|
1083
|
+
namespaces = [this.instance.metadata.namespace];
|
|
1084
|
+
}
|
|
1085
|
+
else {
|
|
1086
|
+
// a peer cannot specify namespaces by labels because
|
|
1087
|
+
// we won't be able to extract the names of those namespaces.
|
|
1088
|
+
if (podSelectorConfig.namespaces.labelSelector && !podSelectorConfig.namespaces.labelSelector.isEmpty()) {
|
|
1089
|
+
throw new Error(`Unable to create an ${oppositeDirection} policy for peer '${peer.node.path}' (pod=${this.instance.name}). Peer must specify namespaces only by name`);
|
|
1090
|
+
}
|
|
1091
|
+
// a peer must specify namespaces by name.
|
|
1092
|
+
if (!podSelectorConfig.namespaces.names) {
|
|
1093
|
+
throw new Error(`Unable to create an ${oppositeDirection} policy for peer '${peer.node.path}' (pod=${this.instance.name}). Peer must specify namespace names`);
|
|
1094
|
+
}
|
|
1095
|
+
namespaces = podSelectorConfig.namespaces.names;
|
|
1096
|
+
}
|
|
1097
|
+
for (const name of namespaces) {
|
|
1098
|
+
switch (direction) {
|
|
1099
|
+
case 'Egress':
|
|
1100
|
+
new networkpolicy.NetworkPolicy(this.instance, `AllowIngress${name}${peerAddress}`, {
|
|
1101
|
+
selector: podSelector,
|
|
1102
|
+
metadata: { namespace: name },
|
|
1103
|
+
ingress: { rules: [{ peer: this.instance, ports: options.ports }] },
|
|
1104
|
+
});
|
|
1105
|
+
break;
|
|
1106
|
+
case 'Ingress':
|
|
1107
|
+
new networkpolicy.NetworkPolicy(this.instance, `AllowEgress${name}${peerAddress}`, {
|
|
1108
|
+
selector: podSelector,
|
|
1109
|
+
metadata: { namespace: name },
|
|
1110
|
+
egress: { rules: [{ peer: this.instance, ports: options.ports }] },
|
|
1111
|
+
});
|
|
1112
|
+
break;
|
|
1113
|
+
default:
|
|
1114
|
+
throw new Error(`Unsupported direction: ${direction}`);
|
|
1115
|
+
}
|
|
1116
|
+
}
|
|
1117
|
+
}
|
|
1118
|
+
}
|
|
1119
|
+
extractPorts(selector) {
|
|
1120
|
+
return container.extractContainerPorts(selector).map(n => networkpolicy.NetworkPolicyPort.tcp(n.number));
|
|
1121
|
+
}
|
|
1122
|
+
/**
|
|
1123
|
+
* Sets the default network policy for Pod/Workload to have all egress and ingress connections as disabled
|
|
1124
|
+
*/
|
|
1125
|
+
isolate() {
|
|
1126
|
+
new networkpolicy.NetworkPolicy(this.instance, 'DefaultDenyAll', {
|
|
1127
|
+
selector: this.instance,
|
|
1128
|
+
// the policy must be defined in the namespace of the pod
|
|
1129
|
+
// so it can select it.
|
|
1130
|
+
metadata: { namespace: this.instance.metadata.namespace },
|
|
1131
|
+
egress: {
|
|
1132
|
+
default: networkpolicy.NetworkPolicyTrafficDefault.DENY,
|
|
1133
|
+
},
|
|
1134
|
+
ingress: {
|
|
1135
|
+
default: networkpolicy.NetworkPolicyTrafficDefault.DENY,
|
|
1136
|
+
},
|
|
1137
|
+
});
|
|
1138
|
+
}
|
|
1139
|
+
}
|
|
1140
|
+
exports.PodConnections = PodConnections;
|
|
1141
|
+
_r = JSII_RTTI_SYMBOL_1;
|
|
1142
|
+
PodConnections[_r] = { fqn: "cdk8s-plus-33.PodConnections", version: "2.0.0" };
|
|
1143
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pod.js","sourceRoot":"","sources":["../src/pod.ts"],"names":[],"mappings":"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","sourcesContent":["import { ApiObject, ApiObjectMetadataDefinition, Duration, Lazy, Names } from 'cdk8s';\nimport { Construct, IConstruct } from 'constructs';\nimport * as base from './base';\nimport * as container from './container';\nimport * as k8s from './imports/k8s';\nimport * as namespace from './namespace';\nimport * as networkpolicy from './network-policy';\nimport * as rb from './role-binding';\nimport * as secret from './secret';\nimport * as serviceaccount from './service-account';\nimport { undefinedIfEmpty, address } from './utils';\nimport * as volume from './volume';\n\nexport abstract class AbstractPod extends base.Resource implements IPodSelector, networkpolicy.INetworkPolicyPeer, rb.ISubject {\n\n  public readonly restartPolicy?: RestartPolicy;\n  public readonly serviceAccount?: serviceaccount.IServiceAccount;\n  public readonly securityContext: PodSecurityContext;\n  public readonly dns: PodDns;\n  public readonly dockerRegistryAuth?: secret.ISecret;\n  public readonly automountServiceAccountToken: boolean;\n  public readonly shareProcessNamespace: boolean;\n  public readonly hostNetwork?: boolean;\n  public readonly terminationGracePeriod?: Duration;\n  public readonly enableServiceLinks?: boolean;\n\n  protected readonly isolate: boolean;\n\n  private readonly _containers: container.Container[] = [];\n  private readonly _initContainers: container.Container[] = [];\n  private readonly _hostAliases: HostAlias[] = [];\n  private readonly _volumes: Map<string, volume.Volume> = new Map();\n\n  public abstract readonly podMetadata: ApiObjectMetadataDefinition;\n\n  constructor(scope: Construct, id: string, props: AbstractPodProps = {}) {\n    super(scope, id);\n\n    this.restartPolicy = props.restartPolicy ?? RestartPolicy.ALWAYS;\n    this.serviceAccount = props.serviceAccount;\n    this.securityContext = new PodSecurityContext(props.securityContext);\n    this.dns = new PodDns(props.dns);\n    this.dockerRegistryAuth = props.dockerRegistryAuth;\n    this.automountServiceAccountToken = props.automountServiceAccountToken ?? false;\n    this.shareProcessNamespace = props.shareProcessNamespace ?? false;\n    this.isolate = props.isolate ?? false;\n    this.hostNetwork = props.hostNetwork ?? false;\n    this.terminationGracePeriod = props.terminationGracePeriod ?? Duration.seconds(30);\n    this.enableServiceLinks = props.enableServiceLinks;\n\n    if (props.containers) {\n      props.containers.forEach(c => this.addContainer(c));\n    }\n\n    if (props.volumes) {\n      props.volumes.forEach(v => this.addVolume(v));\n    }\n\n    if (props.initContainers) {\n      props.initContainers.forEach(c => this.addInitContainer(c));\n    }\n\n    if (props.hostAliases) {\n      props.hostAliases.forEach(c => this.addHostAlias(c));\n    }\n\n  }\n\n  public get containers(): container.Container[] {\n    return [...this._containers];\n  }\n\n  public get initContainers(): container.Container[] {\n    return [...this._initContainers];\n  }\n\n  public get volumes(): volume.Volume[] {\n    return Array.from(this._volumes.values());\n  }\n\n  public get hostAliases(): HostAlias[] {\n    return [...this._hostAliases];\n  }\n\n  /**\n   * @see IPodSelector.toPodSelectorConfig()\n   */\n  public toPodSelectorConfig(): PodSelectorConfig {\n    const podAddress = this.podMetadata.getLabel(Pod.ADDRESS_LABEL);\n    if (!podAddress) {\n      // shouldn't happen because we add this label automatically in both pods and workloads.\n      throw new Error(`Unable to create a label selector since ${Pod.ADDRESS_LABEL} label is missing`);\n    }\n    return {\n      labelSelector: LabelSelector.of({ labels: { [Pod.ADDRESS_LABEL]: podAddress } }),\n      namespaces: this.metadata.namespace ? {\n        names: [this.metadata.namespace],\n      } : undefined,\n    };\n  }\n\n  /**\n   * @see INetworkPolicyPeer.toNetworkPolicyPeerConfig()\n   */\n  public toNetworkPolicyPeerConfig(): networkpolicy.NetworkPolicyPeerConfig {\n    return { podSelector: this.toPodSelectorConfig() };\n  }\n\n  /**\n   * @see INetworkPolicyPeer.toPodSelector()\n   */\n  public toPodSelector(): IPodSelector | undefined {\n    return this;\n  }\n\n  public addContainer(cont: container.ContainerProps): container.Container {\n    const impl = new container.Container(cont);\n    this.attachContainer(impl);\n    return impl;\n  }\n\n  public attachContainer(cont: container.Container) {\n    this._containers.push(cont);\n  }\n\n  public addInitContainer(cont: container.ContainerProps): container.Container {\n\n    // https://kubernetes.io/docs/concepts/workloads/pods/init-containers/#differences-from-regular-containers\n    if (!this.isSidecarContainer(cont) && cont.readiness) {\n      throw new Error('Init containers must not have a readiness probe');\n    }\n\n    if (!this.isSidecarContainer(cont) && cont.liveness) {\n      throw new Error('Init containers must not have a liveness probe');\n    }\n\n    if (!this.isSidecarContainer(cont) && cont.startup) {\n      throw new Error('Init containers must not have a startup probe');\n    }\n\n    const impl = new container.Container({\n      ...cont,\n      name: cont.name ?? `init-${this._initContainers.length}`,\n    });\n\n    this._initContainers.push(impl);\n    return impl;\n  }\n\n  // Any initContainer that has `restartPolicy=Always` is a sidecar container. Please refer to\n  // documentation for more details:\n  // https://kubernetes.io/docs/concepts/workloads/pods/sidecar-containers/#differences-from-init-containers\n  private isSidecarContainer(cont: container.ContainerProps) {\n    return cont.restartPolicy === container.ContainerRestartPolicy.ALWAYS;\n  }\n\n  public addHostAlias(hostAlias: HostAlias): void {\n    this._hostAliases.push(hostAlias);\n  }\n\n  public addVolume(vol: volume.Volume): void {\n    const existingVolume = this._volumes.get(vol.name);\n    if (existingVolume) {\n      throw new Error(`Volume with name ${vol.name} already exists`);\n    }\n    this._volumes.set(vol.name, vol);\n  }\n\n  /**\n   * @see ISubect.toSubjectConfiguration()\n   */\n  public toSubjectConfiguration(): rb.SubjectConfiguration {\n\n    if (!this.serviceAccount && !this.automountServiceAccountToken) {\n      throw new Error(`${this.name} cannot be converted to a role binding subject:`\n        + ' You must either assign a service account to it, or use \\'automountServiceAccountToken: true\\'');\n    }\n\n    // 'default' is assumed to be the name of the default service account\n    // in the cluster.\n    const serviceAccountName = this.serviceAccount?.name ?? 'default';\n\n    return {\n      kind: 'ServiceAccount',\n      name: serviceAccountName,\n      apiGroup: '',\n    };\n  }\n\n  /**\n   * @internal\n   */\n  public _toPodSpec(): k8s.PodSpec {\n\n    if (this.containers.length === 0) {\n      throw new Error('PodSpec must have at least 1 container');\n    }\n\n    const volumes: Map<string, volume.Volume> = new Map();\n    const containers: k8s.Container[] = [];\n    const initContainers: k8s.Container[] = [];\n\n    for (const cont of this.containers) {\n      // check if restartPolicy is defined for containers\n      if (cont.restartPolicy) {\n        throw new Error(`Invalid container spec: ${cont.name} has non-empty restartPolicy field. The field can only be specified for initContainers`);\n      }\n      // automatically add volume from the container mount\n      // to this pod so thats its available to the container.\n      for (const mount of cont.mounts) {\n        addVolume(mount.volume);\n      }\n      containers.push(cont._toKube());\n    }\n\n    for (const cont of this.initContainers) {\n      // automatically add volume from the container mount\n      // to this pod so thats its available to the container.\n      for (const mount of cont.mounts) {\n        addVolume(mount.volume);\n      }\n      initContainers.push(cont._toKube());\n    }\n\n    for (const vol of this.volumes) {\n      addVolume(vol);\n    }\n\n    function addVolume(vol: volume.Volume) {\n      const existingVolume = volumes.get(vol.name);\n      // its ok to call this function twice on the same volume, but its not ok to\n      // call it twice on a different volume with the same name.\n      if (existingVolume && existingVolume !== vol) {\n        throw new Error(`Invalid mount configuration. At least two different volumes have the same name: ${vol.name}`);\n      }\n      volumes.set(vol.name, vol);\n    }\n\n    const dns = this.dns._toKube();\n\n    return {\n      restartPolicy: this.restartPolicy,\n      serviceAccountName: this.serviceAccount?.name,\n      containers: containers,\n      securityContext: undefinedIfEmpty(this.securityContext._toKube()),\n      initContainers: undefinedIfEmpty(initContainers),\n      hostAliases: undefinedIfEmpty(this.hostAliases),\n      volumes: undefinedIfEmpty(Array.from(volumes.values()).map(v => v._toKube())),\n      dnsPolicy: dns.policy,\n      dnsConfig: undefinedIfEmpty(dns.config),\n      hostname: dns.hostname,\n      subdomain: dns.subdomain,\n      setHostnameAsFqdn: dns.hostnameAsFQDN,\n      imagePullSecrets: this.dockerRegistryAuth ? [{ name: this.dockerRegistryAuth.name }] : undefined,\n      automountServiceAccountToken: this.automountServiceAccountToken,\n      shareProcessNamespace: this.shareProcessNamespace,\n      hostNetwork: this.hostNetwork,\n      terminationGracePeriodSeconds: this.terminationGracePeriod?.toSeconds(),\n      enableServiceLinks: this.enableServiceLinks,\n    };\n\n  }\n\n}\n\n/**\n * Sysctl defines a kernel parameter to be set\n */\nexport interface Sysctl {\n  /**\n   * Name of a property to set\n   */\n  readonly name: string;\n\n  /**\n   * Value of a property to set\n   */\n  readonly value: string;\n}\n\n/**\n * Properties for `PodSecurityContext`\n */\nexport interface PodSecurityContextProps {\n\n  /**\n   * Modify the ownership and permissions of pod volumes to this GID.\n   *\n   * @default - Volume ownership is not changed.\n   */\n  readonly fsGroup?: number;\n\n  /**\n   * Defines behavior of changing ownership and permission of the volume before being exposed inside Pod.\n   * This field will only apply to volume types which support fsGroup based ownership(and permissions).\n   * It will have no effect on ephemeral volume types such as: secret, configmaps and emptydir.\n   *\n   * @default FsGroupChangePolicy.ALWAYS\n   */\n  readonly fsGroupChangePolicy?: FsGroupChangePolicy;\n\n  /**\n   * The UID to run the entrypoint of the container process.\n   *\n   * @default - User specified in image metadata\n   */\n  readonly user?: number;\n\n  /**\n   * The GID to run the entrypoint of the container process.\n   *\n   * @default - Group configured by container runtime\n   */\n  readonly group?: number;\n\n  /**\n   * Indicates that the container must run as a non-root user.\n   * If true, the Kubelet will validate the image at runtime to ensure that it does\n   * not run as UID 0 (root) and fail to start the container if it does.\n   *\n   * @default true\n   */\n  readonly ensureNonRoot?: boolean;\n\n  /**\n   * Sysctls hold a list of namespaced sysctls used for the pod.\n   * Pods with unsupported sysctls (by the container runtime) might fail to launch.\n   *\n   * @default - No sysctls\n   */\n  readonly sysctls?: Sysctl[];\n}\n\n/**\n * Properties for `AbstractPod`.\n */\nexport interface AbstractPodProps extends base.ResourceProps {\n\n  /**\n   * List of containers belonging to the pod. Containers cannot currently be\n   * added or removed. There must be at least one container in a Pod.\n   *\n   * You can add additionnal containers using `podSpec.addContainer()`\n   *\n   * @default - No containers. Note that a pod spec must include at least one container.\n   */\n  readonly containers?: container.ContainerProps[];\n\n  /**\n   * List of initialization containers belonging to the pod.\n   * Init containers are executed in order prior to containers being started.\n   * If any init container fails, the pod is considered to have failed and is handled according to its restartPolicy.\n   * The name for an init container or normal container must be unique among all containers.\n   * Init containers may not have Lifecycle actions, Readiness probes, Liveness probes, or Startup probes.\n   * The resourceRequirements of an init container are taken into account during scheduling by finding the highest request/limit\n   * for each resource type, and then using the max of of that value or the sum of the normal containers.\n   * Limits are applied to init containers in a similar fashion.\n   *\n   * Init containers cannot currently be added ,removed or updated.\n   *\n   * @see https://kubernetes.io/docs/concepts/workloads/pods/init-containers/\n   * @default - No init containers.\n   */\n  readonly initContainers?: container.ContainerProps[];\n\n  /**\n   * List of volumes that can be mounted by containers belonging to the pod.\n   *\n   * You can also add volumes later using `podSpec.addVolume()`\n   *\n   * @see https://kubernetes.io/docs/concepts/storage/volumes\n   *\n   * @default - No volumes.\n   */\n  readonly volumes?: volume.Volume[];\n\n  /**\n   * Restart policy for all containers within the pod.\n   *\n   * @see https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle/#restart-policy\n   *\n   * @default RestartPolicy.ALWAYS\n   */\n  readonly restartPolicy?: RestartPolicy;\n\n  /**\n   * A service account provides an identity for processes that run in a Pod.\n   *\n   * When you (a human) access the cluster (for example, using kubectl), you are\n   * authenticated by the apiserver as a particular User Account (currently this\n   * is usually admin, unless your cluster administrator has customized your\n   * cluster). Processes in containers inside pods can also contact the\n   * apiserver. When they do, they are authenticated as a particular Service\n   * Account (for example, default).\n   *\n   * @see https://kubernetes.io/docs/tasks/configure-pod-container/configure-service-account/\n   *\n   * @default - No service account.\n   */\n  readonly serviceAccount?: serviceaccount.IServiceAccount;\n\n  /**\n   * SecurityContext holds pod-level security attributes and common container settings.\n   *\n   * @default\n   *\n   *   fsGroupChangePolicy: FsGroupChangePolicy.FsGroupChangePolicy.ALWAYS\n   *   ensureNonRoot: true\n   */\n  readonly securityContext?: PodSecurityContextProps;\n\n  /**\n   * HostAlias holds the mapping between IP and hostnames that will be injected as an entry in the pod's hosts file.\n   *\n   * @schema io.k8s.api.core.v1.HostAlias\n   */\n  readonly hostAliases?: HostAlias[];\n\n  /**\n   * DNS settings for the pod.\n   *\n   * @see https://kubernetes.io/docs/concepts/services-networking/dns-pod-service/\n   *\n   * @default\n   *\n   *  policy: DnsPolicy.CLUSTER_FIRST\n   *  hostnameAsFQDN: false\n   */\n  readonly dns?: PodDnsProps;\n\n  /**\n   * A secret containing docker credentials for authenticating to a registry.\n   *\n   * @default - No auth. Images are assumed to be publicly available.\n   */\n  readonly dockerRegistryAuth?: secret.ISecret;\n\n  /**\n   * Indicates whether a service account token should be automatically mounted.\n   *\n   * @default false\n   * @see https://kubernetes.io/docs/tasks/configure-pod-container/configure-service-account/#use-the-default-service-account-to-access-the-api-server\n   */\n  readonly automountServiceAccountToken?: boolean;\n\n  /**\n   * When process namespace sharing is enabled, processes in a container are visible to all other containers in the same pod.\n   *\n   * @default false\n   * @see https://kubernetes.io/docs/tasks/configure-pod-container/share-process-namespace/\n   */\n  readonly shareProcessNamespace?: boolean;\n\n  /**\n   * Isolates the pod. This will prevent any ingress or egress connections to / from this pod.\n   * You can however allow explicit connections post instantiation by using the `.connections` property.\n   *\n   * @default false\n   */\n  readonly isolate?: boolean;\n\n  /**\n   * Host network for the pod.\n   *\n   * @default false\n   */\n  readonly hostNetwork?: boolean;\n\n  /**\n   * Grace period until the pod is terminated\n   *\n   * @default Duration.seconds(30)\n   */\n  readonly terminationGracePeriod?: Duration;\n\n  /**\n   * Indicates whether information about services should be injected into pod's\n   * environment variables, matching the syntax of Docker links.\n   *\n   * @default true\n   * @see https://kubernetes.io/docs/concepts/services-networking/connect-applications-service/#accessing-the-service\n   */\n  readonly enableServiceLinks?: boolean;\n}\n\n/**\n * Properties for `Pod`.\n */\nexport interface PodProps extends AbstractPodProps {}\n\n/**\n * Options for `LabelSelector.of`.\n */\nexport interface LabelSelectorOptions {\n\n  /**\n   * Strict label matchers.\n   */\n  readonly labels?: { [key: string]: string };\n\n  /**\n   * Expression based label matchers.\n   */\n  readonly expressions?: LabelExpression[];\n}\n\n/**\n * Match a resource by labels.\n */\nexport class LabelSelector {\n\n  public static of(options: LabelSelectorOptions = {}) {\n    return new LabelSelector(options.expressions ?? [], options.labels ?? {});\n  }\n\n  private constructor(\n    private readonly expressions: LabelExpression[],\n    private readonly labels: { [key: string]: string }) {}\n\n  public isEmpty() {\n    return this.expressions.length === 0 && Object.keys(this.labels).length === 0;\n  }\n\n  /**\n   * @internal\n   */\n  public _toKube(): k8s.LabelSelector {\n    if (this.isEmpty()) {\n      return {};\n    }\n    return {\n      matchExpressions: undefinedIfEmpty(this.expressions.map(q => ({ key: q.key, operator: q.operator, values: q.values }))),\n      matchLabels: undefinedIfEmpty(this.labels),\n    };\n  }\n}\n\n/**\n * Configuration for selecting pods, optionally in particular namespaces.\n */\nexport interface PodSelectorConfig {\n\n  /**\n   * A selector to select pods by labels.\n   */\n  readonly labelSelector: LabelSelector;\n\n  /**\n   * Configuration for selecting which namepsaces are the pods allowed to be in.\n   */\n  readonly namespaces?: namespace.NamespaceSelectorConfig;\n\n}\n\n/**\n * Represents an object that can select pods.\n */\nexport interface IPodSelector extends IConstruct {\n  /**\n   * Return the configuration of this selector.\n   */\n  toPodSelectorConfig(): PodSelectorConfig;\n}\n\n/**\n * Pod is a collection of containers that can run on a host. This resource is\n * created by clients and scheduled onto hosts.\n */\nexport class Pod extends AbstractPod {\n\n  /**\n   * This label is autoamtically added by cdk8s to any pod. It provides\n   * a unique and stable identifier for the pod.\n   */\n  public static readonly ADDRESS_LABEL = 'cdk8s.io/metadata.addr';\n\n  /**\n   * @see base.Resource.apiObject\n   */\n  protected readonly apiObject: ApiObject;\n\n  public readonly resourceType = 'pods';\n\n  public readonly scheduling: PodScheduling;\n  public readonly connections: PodConnections;\n\n  constructor(scope: Construct, id: string, props: PodProps = {}) {\n    super(scope, id, props);\n\n    this.apiObject = new k8s.KubePod(this, 'Resource', {\n      metadata: props.metadata,\n      spec: Lazy.any({ produce: () => this._toKube() }),\n    });\n\n    this.metadata.addLabel(Pod.ADDRESS_LABEL, Names.toLabelValue(this));\n\n    this.scheduling = new PodScheduling(this);\n    this.connections = new PodConnections(this);\n\n    if (this.isolate) {\n      this.connections.isolate();\n    }\n  }\n\n  public get podMetadata(): ApiObjectMetadataDefinition {\n    return this.metadata;\n  }\n\n  /**\n   * @internal\n   */\n  public _toKube(): k8s.PodSpec {\n    const scheduling = this.scheduling._toKube();\n\n    return {\n      ...this._toPodSpec(),\n      affinity: scheduling.affinity,\n      nodeName: scheduling.nodeName,\n      tolerations: scheduling.tolerations,\n    };\n  }\n\n}\n\n/**\n * Properties for `PodDns`.\n */\nexport interface PodDnsProps {\n\n  /**\n   * Specifies the hostname of the Pod.\n   *\n   * @default - Set to a system-defined value.\n   */\n  readonly hostname?: string;\n\n  /**\n   * If specified, the fully qualified Pod hostname will be \"<hostname>.<subdomain>.<pod namespace>.svc.<cluster domain>\".\n   *\n   * @default - No subdomain.\n   */\n  readonly subdomain?: string;\n\n  /**\n   * If true the pod's hostname will be configured as the pod's FQDN, rather than the leaf name (the default).\n   * In Linux containers, this means setting the FQDN in the hostname field of the kernel (the nodename field of struct utsname).\n   * In Windows containers, this means setting the registry value of hostname for the registry\n   * key HKEY_LOCAL_MACHINE\\SYSTEM\\CurrentControlSet\\Services\\Tcpip\\Parameters to FQDN.\n   * If a pod does not have FQDN, this has no effect.\n   *\n   * @default false\n   */\n  readonly hostnameAsFQDN?: boolean;\n\n  /**\n   * Set DNS policy for the pod.\n   *\n   * If policy is set to `None`, other configuration must be supplied.\n   *\n   * @default DnsPolicy.CLUSTER_FIRST\n   */\n  readonly policy?: DnsPolicy;\n\n  /**\n   * A list of IP addresses that will be used as DNS servers for the Pod. There can be at most 3 IP addresses specified.\n   * When the policy is set to \"NONE\", the list must contain at least one IP address,\n   * otherwise this property is optional.\n   * The servers listed will be combined to the base nameservers generated from\n   * the specified DNS policy with duplicate addresses removed.\n   */\n  readonly nameservers?: string[];\n\n  /**\n   * A list of DNS search domains for hostname lookup in the Pod.\n   * When specified, the provided list will be merged into the base\n   * search domain names generated from the chosen DNS policy.\n   * Duplicate domain names are removed.\n   *\n   * Kubernetes allows for at most 6 search domains.\n   */\n  readonly searches?: string[];\n\n  /**\n   * List of objects where each object may have a name property (required)\n   * and a value property (optional). The contents in this property\n   * will be merged to the options generated from the specified DNS policy.\n   * Duplicate entries are removed.\n   */\n  readonly options?: DnsOption[];\n}\n\n/**\n * Holds dns settings of the pod.\n */\nexport class PodDns {\n\n  /**\n   * The DNS policy of this pod.\n   */\n  public readonly policy: DnsPolicy;\n\n  /**\n   * The configured hostname of the pod. Undefined means its set to a system-defined value.\n   */\n  public readonly hostname?: string;\n\n  /**\n   * The configured subdomain of the pod.\n   */\n  public readonly subdomain?: string;\n\n  /**\n   * Whether or not the pods hostname is set to its FQDN.\n   */\n  public readonly hostnameAsFQDN: boolean;\n\n  private readonly _nameservers: string[];\n  private readonly _searches: string[];\n  private readonly _options: DnsOption[];\n\n  constructor(props: PodDnsProps = {}) {\n    this.hostname = props.hostname;\n    this.subdomain = props.subdomain;\n    this.policy = props.policy ?? DnsPolicy.CLUSTER_FIRST;\n    this.hostnameAsFQDN = props.hostnameAsFQDN ?? false;\n    this._nameservers = props.nameservers ?? [];\n    this._searches = props.searches ?? [];\n    this._options = props.options ?? [];\n  }\n\n  /**\n   * Nameservers defined for this pod.\n   */\n  public get nameservers(): string[] {\n    return [...this._nameservers];\n  }\n\n  /**\n   * Search domains defined for this pod.\n   */\n  public get searches(): string[] {\n    return [...this._searches];\n  }\n\n  /**\n   * Custom dns options defined for this pod.\n   */\n  public get options(): DnsOption[] {\n    return [...this._options];\n  }\n\n  /**\n   * Add a nameserver.\n   */\n  public addNameserver(...nameservers: string[]) {\n    this._nameservers.push(...nameservers);\n  }\n\n  /**\n   * Add a search domain.\n   */\n  public addSearch(...searches: string[]) {\n    this._searches.push(...searches);\n  }\n\n  /**\n   * Add a custom option.\n   */\n  public addOption(...options: DnsOption[]) {\n    this._options.push(...options);\n  }\n\n  /**\n   * @internal\n   */\n  public _toKube(): {\n    hostname?: string;\n    subdomain?: string;\n    hostnameAsFQDN: boolean;\n    policy: string;\n    config: k8s.PodDnsConfig;\n  } {\n\n    if (this.policy === DnsPolicy.NONE && this.nameservers.length === 0) {\n      throw new Error('When dns policy is set to NONE, at least one nameserver is required');\n    }\n\n    if (this.nameservers.length > 3) {\n      throw new Error('There can be at most 3 nameservers specified');\n    }\n\n    if (this.searches.length > 6) {\n      throw new Error('There can be at most 6 search domains specified');\n    }\n\n    return {\n      hostname: this.hostname,\n      subdomain: this.subdomain,\n      hostnameAsFQDN: this.hostnameAsFQDN,\n      policy: this.policy,\n      config: {\n        nameservers: undefinedIfEmpty(this.nameservers),\n        searches: undefinedIfEmpty(this.searches),\n        options: undefinedIfEmpty(this.options),\n      },\n    };\n  }\n\n}\n\n/**\n * Holds pod-level security attributes and common container settings.\n */\nexport class PodSecurityContext {\n\n  public readonly ensureNonRoot: boolean;\n  public readonly user?: number;\n  public readonly group?: number;\n  public readonly fsGroup?: number;\n  public readonly fsGroupChangePolicy: FsGroupChangePolicy;\n\n  private readonly _sysctls: Sysctl[] = [];\n\n  constructor(props: PodSecurityContextProps = {}) {\n    this.ensureNonRoot = props.ensureNonRoot ?? true;\n    this.fsGroupChangePolicy = props.fsGroupChangePolicy ?? FsGroupChangePolicy.ALWAYS;\n    this.user = props.user;\n    this.group = props.group;\n    this.fsGroup = props.fsGroup;\n\n    for (const sysctl of props.sysctls ?? []) {\n      this._sysctls.push(sysctl);\n    }\n\n  }\n\n  public get sysctls(): Sysctl[] {\n    return [...this._sysctls];\n  }\n\n  /**\n   * @internal\n   */\n  public _toKube(): k8s.PodSecurityContext {\n    return {\n      runAsGroup: this.group,\n      runAsUser: this.user,\n      fsGroup: this.fsGroup,\n      runAsNonRoot: this.ensureNonRoot,\n      fsGroupChangePolicy: this.fsGroupChangePolicy,\n      sysctls: undefinedIfEmpty(this._sysctls),\n    };\n  }\n\n}\n\n/**\n * Restart policy for all containers within the pod.\n */\nexport enum RestartPolicy {\n  /**\n   * Always restart the pod after it exits.\n   */\n  ALWAYS = 'Always',\n\n  /**\n   * Only restart if the pod exits with a non-zero exit code.\n   */\n  ON_FAILURE = 'OnFailure',\n\n  /**\n   * Never restart the pod.\n   */\n  NEVER = 'Never',\n}\n\nexport enum FsGroupChangePolicy {\n\n  /**\n   * Only change permissions and ownership if permission and ownership of root directory does\n   * not match with expected permissions of the volume.\n   * This could help shorten the time it takes to change ownership and permission of a volume\n   */\n  ON_ROOT_MISMATCH = 'OnRootMismatch',\n\n  /**\n   * Always change permission and ownership of the volume when volume is mounted.\n   */\n  ALWAYS = 'Always',\n}\n\n/**\n * Custom DNS option.\n */\nexport interface DnsOption {\n\n  /**\n   * Option name.\n   */\n  readonly name: string;\n\n  /**\n   * Option value.\n   *\n   * @default - No value.\n   */\n  readonly value?: string;\n}\n\n/**\n * Pod DNS policies.\n */\nexport enum DnsPolicy {\n\n  /**\n   * Any DNS query that does not match the configured cluster domain suffix,\n   * such as \"www.kubernetes.io\", is forwarded to the\n   * upstream nameserver inherited from the node.\n   * Cluster administrators may have extra stub-domain and upstream DNS servers configured.\n   */\n  CLUSTER_FIRST = 'ClusterFirst',\n\n  /**\n   * For Pods running with hostNetwork, you should\n   * explicitly set its DNS policy \"ClusterFirstWithHostNet\".\n   */\n  CLUSTER_FIRST_WITH_HOST_NET = 'ClusterFirstWithHostNet',\n\n  /**\n   * The Pod inherits the name resolution configuration\n   * from the node that the pods run on.\n   */\n  DEFAULT = 'Default',\n\n  /**\n   * It allows a Pod to ignore DNS settings from the Kubernetes environment.\n   * All DNS settings are supposed to be provided using the dnsConfig\n   * field in the Pod Spec.\n   */\n  NONE = 'None',\n\n}\n\n/**\n * HostAlias holds the mapping between IP and hostnames that will be injected as\n * an entry in the pod's /etc/hosts file.\n */\nexport interface HostAlias {\n  /**\n   * Hostnames for the chosen IP address.\n   */\n  readonly hostnames: string[];\n\n  /**\n   * IP address of the host file entry.\n   */\n  readonly ip: string;\n}\n\n/**\n * Represents a query that can be performed against nodes with labels.\n */\nexport class NodeLabelQuery {\n\n  /**\n   * Requires value of label `key` to equal `value`.\n   */\n  public static is(key: string, value: string) {\n    return NodeLabelQuery.in(key, [value]);\n  }\n\n  /**\n   * Requires value of label `key` to be one of `values`.\n   */\n  public static in(key: string, values: string[]) {\n    return new NodeLabelQuery(key, 'In', values);\n  }\n\n  /**\n   * Requires value of label `key` to be none of `values`.\n   */\n  public static notIn(key: string, values: string[]) {\n    return new NodeLabelQuery(key, 'NotIn', values);\n  }\n\n  /**\n   * Requires label `key` to exist.\n   */\n  public static exists(key: string) {\n    return new NodeLabelQuery(key, 'Exists', undefined);\n  }\n\n  /**\n   * Requires label `key` to not exist.\n   */\n  public static doesNotExist(key: string) {\n    return new NodeLabelQuery(key, 'DoesNotExist', undefined);\n  }\n\n  /**\n   * Requires value of label `key` to greater than all elements in `values`.\n   */\n  public static gt(key: string, values: string[]) {\n    return new NodeLabelQuery(key, 'Gt', values);\n  }\n\n  /**\n   * Requires value of label `key` to less than all elements in `values`.\n   */\n  public static lt(key: string, values: string[]) {\n    return new NodeLabelQuery(key, 'Lt', values);\n  }\n\n  private constructor(\n    private readonly key: string,\n    private readonly operator: string,\n    private readonly values?: string[]) {\n  }\n\n  /**\n   * @internal\n   */\n  public _toKube(): k8s.NodeSelectorRequirement {\n    return {\n      key: this.key,\n      operator: this.operator,\n      values: this.values,\n    };\n  }\n}\n\n/**\n * Represents a query that can be performed against resources with labels.\n */\nexport class LabelExpression {\n\n  /**\n   * Requires value of label `key` to be one of `values`.\n   */\n  public static in(key: string, values: string[]) {\n    return new LabelExpression(key, 'In', values);\n  }\n\n  /**\n   * Requires value of label `key` to be none of `values`.\n   */\n  public static notIn(key: string, values: string[]) {\n    return new LabelExpression(key, 'NotIn', values);\n  }\n\n  /**\n   * Requires label `key` to exist.\n   */\n  public static exists(key: string) {\n    return new LabelExpression(key, 'Exists', undefined);\n  }\n\n  /**\n   * Requires label `key` to not exist.\n   */\n  public static doesNotExist(key: string) {\n    return new LabelExpression(key, 'DoesNotExist', undefined);\n  }\n\n  private constructor(\n    public readonly key: string,\n    public readonly operator: string,\n    public readonly values?: string[]) {\n  }\n\n}\n\n/**\n * Taint effects.\n */\nexport enum TaintEffect {\n  /**\n   * This means that no pod will be able to schedule\n   * onto the node unless it has a matching toleration.\n   */\n  NO_SCHEDULE = 'NoSchedule',\n\n  /**\n   * This is a \"preference\" or \"soft\" version of `NO_SCHEDULE` -- the system\n   * will try to avoid placing a pod that does not tolerate the taint on the node,\n   * but it is not required\n   */\n  PREFER_NO_SCHEDULE = 'PreferNoSchedule',\n\n  /**\n   * This affects pods that are already running on the node as follows:\n   *\n   * - Pods that do not tolerate the taint are evicted immediately.\n   * - Pods that tolerate the taint without specifying `duration` remain bound forever.\n   * - Pods that tolerate the taint with a specified `duration` remain bound for\n   *   the specified amount of time.\n   */\n  NO_EXECUTE = 'NoExecute',\n}\n\n/**\n * Options for `NodeTaintQuery`.\n */\nexport interface NodeTaintQueryOptions {\n  /**\n   * The taint effect to match.\n   *\n   * @default - all effects are matched.\n   */\n  readonly effect?: TaintEffect;\n\n  /**\n   * How much time should a pod that tolerates the `NO_EXECUTE` effect\n   * be bound to the node. Only applies for the `NO_EXECUTE` effect.\n   *\n   * @default - bound forever.\n   */\n  readonly evictAfter?: Duration;\n}\n\n/**\n * Taint queries that can be perfomed against nodes.\n */\nexport class NodeTaintQuery {\n\n  /**\n   * Matches a taint with a specific key and value.\n   */\n  public static is(key: string, value: string, options: NodeTaintQueryOptions = {}): NodeTaintQuery {\n    return new NodeTaintQuery('Equal', key, value, options.effect, options.evictAfter);\n  }\n\n  /**\n   * Matches a tain with any value of a specific key.\n   */\n  public static exists(key: string, options: NodeTaintQueryOptions = {}): NodeTaintQuery {\n    return new NodeTaintQuery('Exists', key, undefined, options.effect, options.evictAfter);\n  }\n\n  /**\n   * Matches any taint.\n   */\n  public static any(): NodeTaintQuery {\n    return new NodeTaintQuery('Exists');\n  }\n\n  private constructor(\n    private readonly operator: string,\n    private readonly key?: string,\n    private readonly value?: string,\n    private readonly effect?: TaintEffect,\n    private readonly evictAfter?: Duration,\n  ) {\n    if (evictAfter && effect !== TaintEffect.NO_EXECUTE) {\n      throw new Error('Only \\'NO_EXECUTE\\' effects can specify \\'evictAfter\\'');\n    }\n  }\n\n  /**\n   * @internal\n   */\n  public _toKube(): k8s.Toleration {\n\n    return {\n      effect: this.effect,\n      key: this.key,\n      operator: this.operator,\n      tolerationSeconds: this.evictAfter?.toSeconds(),\n      value: this.value,\n    };\n  }\n\n}\n\n/**\n * Options for `Pods.all`.\n */\nexport interface PodsAllOptions {\n\n  /**\n   * Namespaces the pods are allowed to be in.\n   * Use `Namespaces.all()` to allow all namespaces.\n   *\n   * @default - unset, implies the namespace of the resource this selection is used in.\n   */\n  readonly namespaces?: namespace.Namespaces;\n}\n\n/**\n * Options for `Pods.select`.\n */\nexport interface PodsSelectOptions {\n\n  /**\n   * Labels the pods must have.\n   *\n   * @default - no strict labels requirements.\n   */\n  readonly labels?: { [key: string]: string };\n\n  /**\n    * Expressions the pods must satisify.\n    *\n    * @default - no expressions requirements.\n    */\n  readonly expressions?: LabelExpression[];\n\n  /**\n   * Namespaces the pods are allowed to be in.\n   * Use `Namespaces.all()` to allow all namespaces.\n   *\n   * @default - unset, implies the namespace of the resource this selection is used in.\n   */\n  readonly namespaces?: namespace.Namespaces;\n\n}\n\n/**\n * Represents a group of pods.\n */\nexport class Pods extends Construct implements IPodSelector {\n\n  /**\n   * Select pods in the cluster with various selectors.\n   */\n  public static select(scope: Construct, id: string, options: PodsSelectOptions): Pods {\n    return new Pods(scope, id, options.expressions, options.labels, options.namespaces);\n  }\n\n  /**\n   * Select all pods.\n   */\n  public static all(scope: Construct, id: string, options: PodsAllOptions = {}) {\n    return Pods.select(scope, id, { namespaces: options.namespaces });\n  }\n\n  constructor(scope: Construct, id: string,\n    private readonly expressions?: LabelExpression[],\n    private readonly labels?: { [key: string]: string },\n    private readonly namespaces?: namespace.INamespaceSelector) {\n    super(scope, id);\n  }\n\n  /**\n   * @see IPodSelector.toPodSelectorConfig()\n   */\n  public toPodSelectorConfig(): PodSelectorConfig {\n    return {\n      labelSelector: LabelSelector.of({ expressions: this.expressions, labels: this.labels }),\n      namespaces: this.namespaces?.toNamespaceSelectorConfig(),\n    };\n  }\n\n  /**\n   * @see INetworkPolicyPeer.toNetworkPolicyPeerConfig()\n   */\n  public toNetworkPolicyPeerConfig(): networkpolicy.NetworkPolicyPeerConfig {\n    return { podSelector: this.toPodSelectorConfig() };\n  }\n\n  /**\n   * @see INetworkPolicyPeer.toPodSelector()\n   */\n  public toPodSelector(): IPodSelector | undefined {\n    return this;\n  }\n\n}\n\n/**\n * A node that is matched by label selectors.\n */\nexport class LabeledNode {\n  public constructor(public readonly labelSelector: NodeLabelQuery[]) {};\n}\n\n/**\n * A node that is matched by taint selectors.\n */\nexport class TaintedNode {\n  public constructor(public readonly taintSelector: NodeTaintQuery[]) {};\n}\n\n/**\n * A node that is matched by its name.\n */\nexport class NamedNode {\n  public constructor(public readonly name: string) {};\n}\n\n/**\n * Represents a node in the cluster.\n */\nexport class Node {\n\n  /**\n   * Match a node by its labels.\n   */\n  public static labeled(...labelSelector: NodeLabelQuery[]): LabeledNode {\n    return new LabeledNode(labelSelector);\n  }\n\n  /**\n   * Match a node by its name.\n   */\n  public static named(nodeName: string): NamedNode {\n    return new NamedNode(nodeName);\n  }\n\n  /**\n   * Match a node by its taints.\n   */\n  public static tainted(...taintSelector: NodeTaintQuery[]): TaintedNode {\n    return new TaintedNode(taintSelector);\n  }\n\n}\n\n/**\n * Available topology domains.\n */\nexport class Topology {\n\n  /**\n   * A hostname represents a single node in the cluster.\n   *\n   * @see https://kubernetes.io/docs/reference/labels-annotations-taints/#kubernetesiohostname\n   */\n  public static readonly HOSTNAME = new Topology('kubernetes.io/hostname');\n\n  /**\n   * A zone represents a logical failure domain. It is common for Kubernetes clusters to\n   * span multiple zones for increased availability. While the exact definition of a zone is\n   * left to infrastructure implementations, common properties of a zone include very low\n   * network latency within a zone, no-cost network traffic within a zone, and failure\n   * independence from other zones. For example, nodes within a zone might share a network\n   * switch, but nodes in different zones should not.\n   *\n   * @see https://kubernetes.io/docs/reference/labels-annotations-taints/#topologykubernetesiozone\n   */\n  public static readonly ZONE = new Topology('topology.kubernetes.io/zone');\n\n  /**\n   * A region represents a larger domain, made up of one or more zones. It is uncommon\n   * for Kubernetes clusters to span multiple regions. While the exact definition of a\n   * zone or region is left to infrastructure implementations, common properties of a region\n   * include higher network latency between them than within them, non-zero cost for network\n   * traffic between them, and failure independence from other zones or regions.\n   *\n   * For example, nodes within a region might share power infrastructure (e.g. a UPS or generator), but\n   * nodes in different regions typically would not.\n   *\n   * @see https://kubernetes.io/docs/reference/labels-annotations-taints/#topologykubernetesioregion\n   */\n  public static readonly REGION = new Topology('topology.kubernetes.io/region');\n\n  /**\n   * Custom key for the node label that the system uses to denote the topology domain.\n   */\n  public static custom(key: string): Topology {\n    return new Topology(key);\n  }\n\n  private constructor(public readonly key: string) {};\n}\n\n/**\n * Options for `PodScheduling.colocate`.\n */\nexport interface PodSchedulingColocateOptions {\n  /**\n   * Which topology to coloate on.\n   *\n   * @default - Topology.HOSTNAME\n   */\n  readonly topology?: Topology;\n\n  /**\n   * Indicates the co-location is optional (soft), with this weight score.\n   *\n   * @default - no weight. co-location is assumed to be required (hard).\n   */\n  readonly weight?: number;\n}\n\n/**\n * Options for `PodScheduling.separate`.\n */\nexport interface PodSchedulingSeparateOptions {\n  /**\n   * Which topology to separate on.\n   *\n   * @default - Topology.HOSTNAME\n   */\n  readonly topology?: Topology;\n\n  /**\n   * Indicates the separation is optional (soft), with this weight score.\n   *\n   * @default - no weight. separation is assumed to be required (hard).\n   */\n  readonly weight?: number;\n}\n\n/**\n * Options for `PodScheduling.attract`.\n */\nexport interface PodSchedulingAttractOptions {\n  /**\n   * Indicates the attraction is optional (soft), with this weight score.\n   *\n   * @default - no weight. assignment is assumed to be required (hard).\n   */\n  readonly weight?: number;\n}\n\n/**\n * Controls the pod scheduling strategy.\n */\nexport class PodScheduling {\n\n  private _nodeAffinityPreferred: k8s.PreferredSchedulingTerm[] = [];\n  private _nodeAffinityRequired: k8s.NodeSelectorTerm[] = [];\n  private _podAffinityPreferred: k8s.WeightedPodAffinityTerm[] = [];\n  private _podAffinityRequired: k8s.PodAffinityTerm[] = [];\n  private _podAntiAffinityPreferred: k8s.WeightedPodAffinityTerm[] = [];\n  private _podAntiAffinityRequired: k8s.PodAffinityTerm[] = [];\n  private _tolerations: k8s.Toleration[] = [];\n  private _nodeName?: string;\n\n  constructor(protected readonly instance: AbstractPod) {}\n\n  /**\n   * Assign this pod a specific node by name.\n   *\n   * The scheduler ignores the Pod, and the kubelet on the named node\n   * tries to place the Pod on that node. Overrules any affinity rules of the pod.\n   *\n   * Some limitations of static assignment are:\n   *\n   * - If the named node does not exist, the Pod will not run, and in some\n   *   cases may be automatically deleted.\n   * - If the named node does not have the resources to accommodate the Pod,\n   *   the Pod will fail and its reason will indicate why, for example OutOfmemory or OutOfcpu.\n   * - Node names in cloud environments are not always predictable or stable.\n   *\n   * Will throw is the pod is already assigned to named node.\n   *\n   * Under the hood, this method utilizes the `nodeName` property.\n   */\n  public assign(node: NamedNode) {\n\n    if (this._nodeName) {\n      // disallow overriding an static node assignment\n      throw new Error(`Cannot assign ${this.instance.podMetadata.name} to node ${node.name}. It is already assigned to node ${this._nodeName}`);\n    } else {\n      this._nodeName = node.name;\n    }\n  }\n\n  /**\n   * Allow this pod to tolerate taints matching these tolerations.\n   *\n   * You can put multiple taints on the same node and multiple tolerations on the same pod.\n   * The way Kubernetes processes multiple taints and tolerations is like a filter: start with\n   * all of a node's taints, then ignore the ones for which the pod has a matching toleration;\n   * the remaining un-ignored taints have the indicated effects on the pod. In particular:\n   *\n   * - if there is at least one un-ignored taint with effect NoSchedule then Kubernetes will\n   *   not schedule the pod onto that node\n   * - if there is no un-ignored taint with effect NoSchedule but there is at least one un-ignored\n   *   taint with effect PreferNoSchedule then Kubernetes will try to not schedule the pod onto the node\n   * - if there is at least one un-ignored taint with effect NoExecute then the pod will be evicted from\n   *   the node (if it is already running on the node), and will not be scheduled onto the node (if it is\n   *   not yet running on the node).\n   *\n   * Under the hood, this method utilizes the `tolerations` property.\n   *\n   * @see https://kubernetes.io/docs/concepts/scheduling-eviction/taint-and-toleration/\n   */\n  public tolerate(node: TaintedNode) {\n    for (const query of node.taintSelector) {\n      this._tolerations.push(query._toKube());\n    }\n  }\n\n  /**\n   * Attract this pod to a node matched by selectors.\n   * You can select a node by using `Node.labeled()`.\n   *\n   * Attracting to multiple nodes (i.e invoking this method multiple times) acts as\n   * an OR condition, meaning the pod will be assigned to either one of the nodes.\n   *\n   * Under the hood, this method utilizes the `nodeAffinity` property.\n   *\n   * @see https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#node-affinity\n   */\n  public attract(node: LabeledNode, options: PodSchedulingAttractOptions = {}) {\n\n    const term = this.createNodeAffinityTerm(node);\n\n    if (options.weight) {\n      this.validateWeight(options.weight);\n      this._nodeAffinityPreferred.push({ weight: options.weight, preference: term });\n    } else {\n      this._nodeAffinityRequired.push(term);\n    }\n  }\n\n  /**\n   * Co-locate this pod with a scheduling selection.\n   *\n   * A selection can be one of:\n   *\n   * - An instance of a `Pod`.\n   * - An instance of a `Workload` (e.g `Deployment`, `StatefulSet`).\n   * - An un-managed pod that can be selected via `Pods.select()`.\n   *\n   * Co-locating with multiple selections ((i.e invoking this method multiple times)) acts as\n   * an AND condition. meaning the pod will be assigned to a node that satisfies all\n   * selections (i.e runs at least one pod that satisifies each selection).\n   *\n   * Under the hood, this method utilizes the `podAffinity` property.\n   *\n   * @see https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#inter-pod-affinity-and-anti-affinity\n   */\n  public colocate(selector: IPodSelector, options: PodSchedulingColocateOptions = {}) {\n\n    const topology = options.topology ?? Topology.HOSTNAME;\n    const term = this.createPodAffinityTerm(topology, selector);\n\n    if (options.weight) {\n      this.validateWeight(options.weight);\n      this._podAffinityPreferred.push({ weight: options.weight, podAffinityTerm: term });\n    } else {\n      this._podAffinityRequired.push(term);\n    }\n  }\n\n  /**\n   * Seperate this pod from a scheduling selection.\n   *\n   * A selection can be one of:\n   *\n   * - An instance of a `Pod`.\n   * - An instance of a `Workload` (e.g `Deployment`, `StatefulSet`).\n   * - An un-managed pod that can be selected via `Pods.select()`.\n   *\n   * Seperating from multiple selections acts as an AND condition. meaning the pod\n   * will not be assigned to a node that satisfies all selections (i.e runs at least one pod that satisifies each selection).\n   *\n   * Under the hood, this method utilizes the `podAntiAffinity` property.\n   *\n   * @see https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#inter-pod-affinity-and-anti-affinity\n   */\n  public separate(selector: IPodSelector, options: PodSchedulingSeparateOptions = {}) {\n\n    const topology = options.topology ?? Topology.HOSTNAME;\n    const term = this.createPodAffinityTerm(topology, selector);\n\n    if (options.weight) {\n      this.validateWeight(options.weight);\n      this._podAntiAffinityPreferred.push({ weight: options.weight, podAffinityTerm: term });\n    } else {\n      this._podAntiAffinityRequired.push(term);\n    }\n\n  }\n\n  private createPodAffinityTerm(topology: Topology, selector: IPodSelector): k8s.PodAffinityTerm {\n    const config = selector.toPodSelectorConfig();\n    return {\n      topologyKey: topology.key,\n      labelSelector: config.labelSelector._toKube(),\n      namespaceSelector: config.namespaces?.labelSelector?._toKube(),\n      namespaces: config.namespaces?.names,\n    };\n  }\n\n  private createNodeAffinityTerm(node: LabeledNode): k8s.NodeSelectorTerm {\n    return { matchExpressions: node.labelSelector.map(s => s._toKube()) };\n  }\n\n  private validateWeight(weight: number) {\n    if (weight < 1 || weight > 100) {\n      // https://kubernetes.io/docs/concepts/scheduling-eviction/assign-pod-node/#node-affinity-weight\n      throw new Error(`Invalid affinity weight: ${weight}. Must be in range 1-100`);\n    }\n  }\n\n  /**\n   * @internal\n   */\n  public _toKube(): { affinity?: k8s.Affinity; nodeName?: string; tolerations?: k8s.Toleration[] } {\n\n    const atLeastOne = (...arrays: Array<any>[]) => {\n      return arrays.flat().length > 0;\n    };\n\n    const hasNodeAffinity = atLeastOne(this._nodeAffinityPreferred, this._nodeAffinityRequired);\n    const hasPodAffinity = atLeastOne(this._podAffinityPreferred, this._podAffinityRequired);\n    const hasPodAntiAffinty = atLeastOne(this._podAntiAffinityPreferred, this._podAntiAffinityRequired);\n    const hasAffinity = hasNodeAffinity || hasPodAffinity || hasPodAntiAffinty;\n\n    return {\n      affinity: hasAffinity ? {\n        nodeAffinity: hasNodeAffinity ? {\n          preferredDuringSchedulingIgnoredDuringExecution: undefinedIfEmpty(this._nodeAffinityPreferred),\n          requiredDuringSchedulingIgnoredDuringExecution: this._nodeAffinityRequired.length > 0 ? {\n            nodeSelectorTerms: this._nodeAffinityRequired,\n          } : undefined,\n        } : undefined,\n        podAffinity: hasPodAffinity ? {\n          preferredDuringSchedulingIgnoredDuringExecution: undefinedIfEmpty(this._podAffinityPreferred),\n          requiredDuringSchedulingIgnoredDuringExecution: undefinedIfEmpty(this._podAffinityRequired),\n        } : undefined,\n        podAntiAffinity: hasPodAntiAffinty ? {\n          preferredDuringSchedulingIgnoredDuringExecution: undefinedIfEmpty(this._podAntiAffinityPreferred),\n          requiredDuringSchedulingIgnoredDuringExecution: undefinedIfEmpty(this._podAntiAffinityRequired),\n        } : undefined,\n      } : undefined,\n      nodeName: this._nodeName,\n      tolerations: undefinedIfEmpty(this._tolerations),\n    };\n  }\n}\n\n/**\n * Isolation determines which policies are created\n * when allowing connections from a a pod / workload to peers.\n */\nexport enum PodConnectionsIsolation {\n\n  /**\n   * Only creates network policies that select the pod.\n   */\n  POD = 'POD',\n\n  /**\n   * Only creates network policies that select the peer.\n   */\n  PEER = 'PEER',\n\n}\n\n/**\n * Options for `PodConnections.allowTo`.\n */\nexport interface PodConnectionsAllowToOptions {\n\n  /**\n   * Which isolation should be applied to establish the connection.\n   *\n   * @default - unset, isolates both the pod and the peer.\n   */\n  readonly isolation?: PodConnectionsIsolation;\n\n  /**\n   * Ports to allow outgoing traffic to.\n   *\n   * @default - If the peer is a managed pod, take its ports. Otherwise, all ports are allowed.\n   */\n  readonly ports?: networkpolicy.NetworkPolicyPort[];\n\n}\n\n/**\n * Options for `PodConnections.allowFrom`.\n */\nexport interface PodConnectionsAllowFromOptions {\n\n  /**\n   * Which isolation should be applied to establish the connection.\n   *\n   * @default - unset, isolates both the pod and the peer.\n   */\n  readonly isolation?: PodConnectionsIsolation;\n\n  /**\n   * Ports to allow incoming traffic to.\n   *\n   * @default - The pod ports.\n   */\n  readonly ports?: networkpolicy.NetworkPolicyPort[];\n\n}\n\n/**\n * Controls network isolation rules for inter-pod communication.\n */\nexport class PodConnections {\n\n  constructor(protected readonly instance: AbstractPod) {}\n\n  /**\n   * Allow network traffic from this pod to the peer.\n   *\n   * By default, this will create an egress network policy for this pod, and an ingress\n   * network policy for the peer. This is required if both sides are already isolated.\n   * Use `options.isolation` to control this behavior.\n   *\n   * @example\n   *\n   * // create only an egress policy that selects the 'web' pod to allow outgoing traffic\n   * // to the 'redis' pod. this requires the 'redis' pod to not be isolated for ingress.\n   * web.connections.allowTo(redis, { isolation: Isolation.POD })\n   *\n   * // create only an ingress policy that selects the 'redis' peer to allow incoming traffic\n   * // from the 'web' pod. this requires the 'web' pod to not be isolated for egress.\n   * web.connections.allowTo(redis, { isolation: Isolation.PEER })\n   *\n   */\n  public allowTo(peer: networkpolicy.INetworkPolicyPeer, options: PodConnectionsAllowToOptions = {}) {\n    return this.allow('Egress', peer, { ports: this.extractPorts(peer), ...options });\n  }\n\n  /**\n   * Allow network traffic from the peer to this pod.\n   *\n   * By default, this will create an ingress network policy for this pod, and an egress\n   * network policy for the peer. This is required if both sides are already isolated.\n   * Use `options.isolation` to control this behavior.\n   *\n   * @example\n   *\n   * // create only an egress policy that selects the 'web' pod to allow outgoing traffic\n   * // to the 'redis' pod. this requires the 'redis' pod to not be isolated for ingress.\n   * redis.connections.allowFrom(web, { isolation: Isolation.PEER })\n   *\n   * // create only an ingress policy that selects the 'redis' peer to allow incoming traffic\n   * // from the 'web' pod. this requires the 'web' pod to not be isolated for egress.\n   * redis.connections.allowFrom(web, { isolation: Isolation.POD })\n   *\n   */\n  public allowFrom(peer: networkpolicy.INetworkPolicyPeer, options: PodConnectionsAllowFromOptions = {}) {\n    return this.allow('Ingress', peer, { ports: this.extractPorts(this.instance), ...options });\n  }\n\n  private allow(direction: 'Ingress' | 'Egress', peer: networkpolicy.INetworkPolicyPeer, options: PodConnectionsAllowToOptions | PodConnectionsAllowFromOptions = {}) {\n\n    const config = peer.toNetworkPolicyPeerConfig();\n    networkpolicy.validatePeerConfig(config);\n\n    const peerAddress = address(peer);\n\n    if (!options.isolation || options.isolation === PodConnectionsIsolation.POD) {\n\n      const src = new networkpolicy.NetworkPolicy(this.instance, `Allow${direction}${peerAddress}`, {\n        selector: this.instance,\n        // the policy must be defined in the namespace of the pod\n        // so it can select it.\n        metadata: { namespace: this.instance.metadata.namespace },\n      });\n\n      switch (direction) {\n        case 'Egress':\n          src.addEgressRule(peer, options.ports);\n          break;\n        case 'Ingress':\n          src.addIngressRule(peer, options.ports);\n      }\n\n    }\n\n    if (!options.isolation || options.isolation === PodConnectionsIsolation.PEER) {\n\n      if (config.ipBlock) {\n        // for an ip block we don't need to create the opposite policies\n        return;\n      }\n\n      const podSelector = peer.toPodSelector();\n      if (!podSelector) {\n        throw new Error(`Unable to create policies for peer '${peer.node.addr}' since its not a pod selector`);\n      }\n\n      const oppositeDirection = direction === 'Egress' ? 'Ingress' : 'Egress';\n\n      const podSelectorConfig = podSelector.toPodSelectorConfig();\n      let namespaces: (string | undefined)[];\n\n      if (!podSelectorConfig.namespaces) {\n\n        // if the peer doesn't specify namespaces, we assume the same namespace.\n        namespaces = [this.instance.metadata.namespace];\n\n      } else {\n\n        // a peer cannot specify namespaces by labels because\n        // we won't be able to extract the names of those namespaces.\n        if (podSelectorConfig.namespaces.labelSelector && !podSelectorConfig.namespaces.labelSelector.isEmpty()) {\n          throw new Error(`Unable to create an ${oppositeDirection} policy for peer '${peer.node.path}' (pod=${this.instance.name}). Peer must specify namespaces only by name`);\n        }\n\n        // a peer must specify namespaces by name.\n        if (!podSelectorConfig.namespaces.names) {\n          throw new Error(`Unable to create an ${oppositeDirection} policy for peer '${peer.node.path}' (pod=${this.instance.name}). Peer must specify namespace names`);\n        }\n\n        namespaces = podSelectorConfig.namespaces.names;\n      }\n\n      for (const name of namespaces) {\n        switch (direction) {\n          case 'Egress':\n            new networkpolicy.NetworkPolicy(this.instance, `AllowIngress${name}${peerAddress}`, {\n              selector: podSelector,\n              metadata: { namespace: name },\n              ingress: { rules: [{ peer: this.instance, ports: options.ports }] },\n            });\n            break;\n          case 'Ingress':\n            new networkpolicy.NetworkPolicy(this.instance, `AllowEgress${name}${peerAddress}`, {\n              selector: podSelector,\n              metadata: { namespace: name },\n              egress: { rules: [{ peer: this.instance, ports: options.ports }] },\n            });\n            break;\n          default:\n            throw new Error(`Unsupported direction: ${direction}`);\n        }\n      }\n\n    }\n  }\n\n  private extractPorts(selector?: networkpolicy.INetworkPolicyPeer): networkpolicy.NetworkPolicyPort[] {\n    return container.extractContainerPorts(selector).map(n => networkpolicy.NetworkPolicyPort.tcp(n.number));\n  }\n\n  /**\n   * Sets the default network policy for Pod/Workload to have all egress and ingress connections as disabled\n   */\n  public isolate() {\n    new networkpolicy.NetworkPolicy(this.instance, 'DefaultDenyAll', {\n      selector: this.instance,\n      // the policy must be defined in the namespace of the pod\n      // so it can select it.\n      metadata: { namespace: this.instance.metadata.namespace },\n      egress: {\n        default: networkpolicy.NetworkPolicyTrafficDefault.DENY,\n      },\n      ingress: {\n        default: networkpolicy.NetworkPolicyTrafficDefault.DENY,\n      },\n    });\n  }\n}\n"]}
|