cdk8s-plus-34 0.0.1 → 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 +101649 -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 +21 -33
- package/SECURITY.md +5 -0
- package/cdk8s.yaml +11 -0
- package/docs/java.md +150009 -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 +84 -0
- package/docs/python.md +161196 -0
- package/docs/typescript.md +139649 -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 +26 -0
- package/lib/_action.js +39 -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 +235 -0
- package/lib/deployment.js +216 -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 +25809 -0
- package/lib/imports/k8s.js +19238 -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 +1147 -0
- package/lib/probe.d.ts +181 -0
- package/lib/probe.js +88 -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 +206 -0
- package/lib/stateful-set.js +244 -0
- package/lib/utils.d.ts +4 -0
- package/lib/utils.js +33 -0
- package/lib/volume.d.ts +577 -0
- package/lib/volume.js +377 -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 +202 -0
- package/node_modules/brace-expansion/package.json +49 -0
- package/node_modules/minimatch/LICENSE +15 -0
- package/node_modules/minimatch/README.md +491 -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 +747 -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 +98 -0
- package/node_modules/minimatch/dist/commonjs/index.d.ts.map +1 -0
- package/node_modules/minimatch/dist/commonjs/index.js +1023 -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 +743 -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 +98 -0
- package/node_modules/minimatch/dist/esm/index.d.ts.map +1 -0
- package/node_modules/minimatch/dist/esm/index.js +1007 -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 +176 -6
- package/rotate.md +85 -0
|
@@ -0,0 +1,206 @@
|
|
|
1
|
+
import { ApiObject, Duration } from 'cdk8s';
|
|
2
|
+
import { Construct } from 'constructs';
|
|
3
|
+
import { IScalable, ScalingTarget } from './horizontal-pod-autoscaler';
|
|
4
|
+
import * as k8s from './imports/k8s';
|
|
5
|
+
import { PersistentVolumeClaimProps } from './pvc';
|
|
6
|
+
import * as service from './service';
|
|
7
|
+
import * as workload from './workload';
|
|
8
|
+
/**
|
|
9
|
+
* Controls how pods are created during initial scale up, when replacing pods on nodes,
|
|
10
|
+
* or when scaling down.
|
|
11
|
+
*
|
|
12
|
+
* The default policy is `OrderedReady`, where pods are created in increasing order
|
|
13
|
+
* (pod-0, then pod-1, etc) and the controller will wait until each pod is ready before
|
|
14
|
+
* continuing. When scaling down, the pods are removed in the opposite order.
|
|
15
|
+
*
|
|
16
|
+
* The alternative policy is `Parallel` which will create pods in parallel to match the
|
|
17
|
+
* desired scale without waiting, and on scale down will delete all pods at once.
|
|
18
|
+
*/
|
|
19
|
+
export declare enum PodManagementPolicy {
|
|
20
|
+
ORDERED_READY = "OrderedReady",
|
|
21
|
+
PARALLEL = "Parallel"
|
|
22
|
+
}
|
|
23
|
+
/**
|
|
24
|
+
* Properties for initialization of `StatefulSet`.
|
|
25
|
+
*/
|
|
26
|
+
export interface StatefulSetProps extends workload.WorkloadProps {
|
|
27
|
+
/**
|
|
28
|
+
* Service to associate with the statefulset.
|
|
29
|
+
*
|
|
30
|
+
* @default - A new headless service will be created.
|
|
31
|
+
*/
|
|
32
|
+
readonly service?: service.Service;
|
|
33
|
+
/**
|
|
34
|
+
* Number of desired pods.
|
|
35
|
+
*
|
|
36
|
+
* @default 1
|
|
37
|
+
*/
|
|
38
|
+
readonly replicas?: number;
|
|
39
|
+
/**
|
|
40
|
+
* Pod management policy to use for this statefulset.
|
|
41
|
+
*
|
|
42
|
+
* @default PodManagementPolicy.ORDERED_READY
|
|
43
|
+
*/
|
|
44
|
+
readonly podManagementPolicy?: PodManagementPolicy;
|
|
45
|
+
/**
|
|
46
|
+
* Indicates the StatefulSetUpdateStrategy that will be employed to update Pods in the StatefulSet when a revision is made to Template.
|
|
47
|
+
*
|
|
48
|
+
* @default - RollingUpdate with partition set to 0
|
|
49
|
+
*/
|
|
50
|
+
readonly strategy?: StatefulSetUpdateStrategy;
|
|
51
|
+
/**
|
|
52
|
+
* Minimum duration for which a newly created pod should be ready without any of its container crashing,
|
|
53
|
+
* for it to be considered available. Zero means the pod will be considered available as soon as it is ready.
|
|
54
|
+
*
|
|
55
|
+
* This is an alpha field and requires enabling StatefulSetMinReadySeconds feature gate.
|
|
56
|
+
*
|
|
57
|
+
* @see https://kubernetes.io/docs/concepts/workloads/controllers/deployment/#min-ready-seconds
|
|
58
|
+
* @default Duration.seconds(0)
|
|
59
|
+
*/
|
|
60
|
+
readonly minReady?: Duration;
|
|
61
|
+
/**
|
|
62
|
+
* A list of PersistentVolumeClaim templates that will be created for each pod in the StatefulSet.
|
|
63
|
+
* The StatefulSet controller creates a PVC and a PV for each template based on the pod's ordinal index,
|
|
64
|
+
* ensuring stable storage across pod restarts and rescheduling.
|
|
65
|
+
*
|
|
66
|
+
* Each claim in this list must have at least one matching (by name) volumeMount in one of the containers.
|
|
67
|
+
*
|
|
68
|
+
* @default - No volume claim templates will be created.
|
|
69
|
+
*/
|
|
70
|
+
readonly volumeClaimTemplates?: PersistentVolumeClaimTemplateProps[];
|
|
71
|
+
}
|
|
72
|
+
/**
|
|
73
|
+
* StatefulSet is the workload API object used to manage stateful applications.
|
|
74
|
+
*
|
|
75
|
+
* Manages the deployment and scaling of a set of Pods, and provides guarantees
|
|
76
|
+
* about the ordering and uniqueness of these Pods.
|
|
77
|
+
*
|
|
78
|
+
* Like a Deployment, a StatefulSet manages Pods that are based on an identical
|
|
79
|
+
* container spec. Unlike a Deployment, a StatefulSet maintains a sticky identity
|
|
80
|
+
* for each of their Pods. These pods are created from the same spec, but are not
|
|
81
|
+
* interchangeable: each has a persistent identifier that it maintains across any
|
|
82
|
+
* rescheduling.
|
|
83
|
+
*
|
|
84
|
+
* If you want to use storage volumes to provide persistence for your workload, you
|
|
85
|
+
* can use a StatefulSet as part of the solution. Although individual Pods in a StatefulSet
|
|
86
|
+
* are susceptible to failure, the persistent Pod identifiers make it easier to match existing
|
|
87
|
+
* volumes to the new Pods that replace any that have failed.
|
|
88
|
+
*
|
|
89
|
+
* Using StatefulSets
|
|
90
|
+
* ------------------
|
|
91
|
+
* StatefulSets are valuable for applications that require one or more of the following.
|
|
92
|
+
*
|
|
93
|
+
* - Stable, unique network identifiers.
|
|
94
|
+
* - Stable, persistent storage.
|
|
95
|
+
* - Ordered, graceful deployment and scaling.
|
|
96
|
+
* - Ordered, automated rolling updates.
|
|
97
|
+
*/
|
|
98
|
+
export declare class StatefulSet extends workload.Workload implements IScalable {
|
|
99
|
+
/**
|
|
100
|
+
* Number of desired pods.
|
|
101
|
+
*/
|
|
102
|
+
readonly replicas?: number;
|
|
103
|
+
/**
|
|
104
|
+
* Management policy to use for the set.
|
|
105
|
+
*/
|
|
106
|
+
readonly podManagementPolicy: PodManagementPolicy;
|
|
107
|
+
/**
|
|
108
|
+
* The update startegy of this stateful set.
|
|
109
|
+
*/
|
|
110
|
+
readonly strategy: StatefulSetUpdateStrategy;
|
|
111
|
+
/**
|
|
112
|
+
* Minimum duration for which a newly created pod should be ready without
|
|
113
|
+
* any of its container crashing, for it to be considered available.
|
|
114
|
+
*/
|
|
115
|
+
readonly minReady: Duration;
|
|
116
|
+
/**
|
|
117
|
+
* @see base.Resource.apiObject
|
|
118
|
+
*/
|
|
119
|
+
protected readonly apiObject: ApiObject;
|
|
120
|
+
readonly resourceType = "statefulsets";
|
|
121
|
+
hasAutoscaler: boolean;
|
|
122
|
+
readonly service: service.Service;
|
|
123
|
+
volumeClaimTemplates?: PersistentVolumeClaimTemplateProps[];
|
|
124
|
+
constructor(scope: Construct, id: string, props: StatefulSetProps);
|
|
125
|
+
addVolumeClaimTemplate(template: PersistentVolumeClaimTemplateProps): void;
|
|
126
|
+
private _createHeadlessService;
|
|
127
|
+
/**
|
|
128
|
+
* @internal
|
|
129
|
+
*/
|
|
130
|
+
_toKube(): k8s.StatefulSetSpec;
|
|
131
|
+
/**
|
|
132
|
+
* @internal
|
|
133
|
+
*/
|
|
134
|
+
_toPodSpec(): k8s.PodSpec;
|
|
135
|
+
/**
|
|
136
|
+
* @internal
|
|
137
|
+
*/
|
|
138
|
+
private _filterPodSpecVolumes;
|
|
139
|
+
/**
|
|
140
|
+
* @see IScalable.markHasAutoscaler()
|
|
141
|
+
*/
|
|
142
|
+
markHasAutoscaler(): void;
|
|
143
|
+
/**
|
|
144
|
+
* @see IScalable.toScalingTarget()
|
|
145
|
+
*/
|
|
146
|
+
toScalingTarget(): ScalingTarget;
|
|
147
|
+
/**
|
|
148
|
+
* @internal
|
|
149
|
+
*/
|
|
150
|
+
private _toPersistentVolumeClaims;
|
|
151
|
+
}
|
|
152
|
+
/**
|
|
153
|
+
* Options for `StatefulSetUpdateStrategy.rollingUpdate`.
|
|
154
|
+
*/
|
|
155
|
+
export interface StatefulSetUpdateStrategyRollingUpdateOptions {
|
|
156
|
+
/**
|
|
157
|
+
* If specified, all Pods with an ordinal that is greater than or equal to the partition will
|
|
158
|
+
* be updated when the StatefulSet's .spec.template is updated. All Pods with an ordinal that
|
|
159
|
+
* is less than the partition will not be updated, and, even if they are deleted, they will be
|
|
160
|
+
* recreated at the previous version.
|
|
161
|
+
*
|
|
162
|
+
* If the partition is greater than replicas, updates to the pod template will not be propagated to Pods.
|
|
163
|
+
* In most cases you will not need to use a partition, but they are useful if you want to stage an
|
|
164
|
+
* update, roll out a canary, or perform a phased roll out.
|
|
165
|
+
*
|
|
166
|
+
* @see https://kubernetes.io/docs/concepts/workloads/controllers/statefulset/#partitions
|
|
167
|
+
* @default 0
|
|
168
|
+
*/
|
|
169
|
+
readonly partition?: number;
|
|
170
|
+
}
|
|
171
|
+
/**
|
|
172
|
+
* StatefulSet update strategies.
|
|
173
|
+
*/
|
|
174
|
+
export declare class StatefulSetUpdateStrategy {
|
|
175
|
+
private readonly strategy;
|
|
176
|
+
/**
|
|
177
|
+
* The controller will not automatically update the Pods in a StatefulSet.
|
|
178
|
+
* Users must manually delete Pods to cause the controller to create new Pods
|
|
179
|
+
* that reflect modifications.
|
|
180
|
+
*/
|
|
181
|
+
static onDelete(): StatefulSetUpdateStrategy;
|
|
182
|
+
/**
|
|
183
|
+
* The controller will delete and recreate each Pod in the StatefulSet.
|
|
184
|
+
* It will proceed in the same order as Pod termination (from the largest ordinal to the smallest),
|
|
185
|
+
* updating each Pod one at a time. The Kubernetes control plane waits until an updated
|
|
186
|
+
* Pod is Running and Ready prior to updating its predecessor.
|
|
187
|
+
*/
|
|
188
|
+
static rollingUpdate(options?: StatefulSetUpdateStrategyRollingUpdateOptions): StatefulSetUpdateStrategy;
|
|
189
|
+
private constructor();
|
|
190
|
+
/**
|
|
191
|
+
* @internal
|
|
192
|
+
*/
|
|
193
|
+
_toKube(): k8s.StatefulSetUpdateStrategy;
|
|
194
|
+
}
|
|
195
|
+
/**
|
|
196
|
+
* A PersistentVolumeClaim template for StatefulSets
|
|
197
|
+
*/
|
|
198
|
+
export interface PersistentVolumeClaimTemplateProps extends PersistentVolumeClaimProps {
|
|
199
|
+
/**
|
|
200
|
+
* The name of the claim that the StatefulSet controller will create for each pod.
|
|
201
|
+
* This will be used to name the created PVC in the format <claim-name>-<pod-name>
|
|
202
|
+
*
|
|
203
|
+
* This name should match the name of a volume mount in one of the containers.
|
|
204
|
+
*/
|
|
205
|
+
readonly name: string;
|
|
206
|
+
}
|
|
@@ -0,0 +1,244 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var _a, _b;
|
|
3
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
+
exports.StatefulSetUpdateStrategy = exports.StatefulSet = exports.PodManagementPolicy = void 0;
|
|
5
|
+
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
|
6
|
+
const cdk8s_1 = require("cdk8s");
|
|
7
|
+
const container = require("./container");
|
|
8
|
+
const k8s = require("./imports/k8s");
|
|
9
|
+
const k8s_1 = require("./imports/k8s");
|
|
10
|
+
const service = require("./service");
|
|
11
|
+
const workload = require("./workload");
|
|
12
|
+
/**
|
|
13
|
+
* Controls how pods are created during initial scale up, when replacing pods on nodes,
|
|
14
|
+
* or when scaling down.
|
|
15
|
+
*
|
|
16
|
+
* The default policy is `OrderedReady`, where pods are created in increasing order
|
|
17
|
+
* (pod-0, then pod-1, etc) and the controller will wait until each pod is ready before
|
|
18
|
+
* continuing. When scaling down, the pods are removed in the opposite order.
|
|
19
|
+
*
|
|
20
|
+
* The alternative policy is `Parallel` which will create pods in parallel to match the
|
|
21
|
+
* desired scale without waiting, and on scale down will delete all pods at once.
|
|
22
|
+
*/
|
|
23
|
+
var PodManagementPolicy;
|
|
24
|
+
(function (PodManagementPolicy) {
|
|
25
|
+
PodManagementPolicy["ORDERED_READY"] = "OrderedReady";
|
|
26
|
+
PodManagementPolicy["PARALLEL"] = "Parallel";
|
|
27
|
+
})(PodManagementPolicy || (exports.PodManagementPolicy = PodManagementPolicy = {}));
|
|
28
|
+
/**
|
|
29
|
+
* StatefulSet is the workload API object used to manage stateful applications.
|
|
30
|
+
*
|
|
31
|
+
* Manages the deployment and scaling of a set of Pods, and provides guarantees
|
|
32
|
+
* about the ordering and uniqueness of these Pods.
|
|
33
|
+
*
|
|
34
|
+
* Like a Deployment, a StatefulSet manages Pods that are based on an identical
|
|
35
|
+
* container spec. Unlike a Deployment, a StatefulSet maintains a sticky identity
|
|
36
|
+
* for each of their Pods. These pods are created from the same spec, but are not
|
|
37
|
+
* interchangeable: each has a persistent identifier that it maintains across any
|
|
38
|
+
* rescheduling.
|
|
39
|
+
*
|
|
40
|
+
* If you want to use storage volumes to provide persistence for your workload, you
|
|
41
|
+
* can use a StatefulSet as part of the solution. Although individual Pods in a StatefulSet
|
|
42
|
+
* are susceptible to failure, the persistent Pod identifiers make it easier to match existing
|
|
43
|
+
* volumes to the new Pods that replace any that have failed.
|
|
44
|
+
*
|
|
45
|
+
* Using StatefulSets
|
|
46
|
+
* ------------------
|
|
47
|
+
* StatefulSets are valuable for applications that require one or more of the following.
|
|
48
|
+
*
|
|
49
|
+
* - Stable, unique network identifiers.
|
|
50
|
+
* - Stable, persistent storage.
|
|
51
|
+
* - Ordered, graceful deployment and scaling.
|
|
52
|
+
* - Ordered, automated rolling updates.
|
|
53
|
+
*/
|
|
54
|
+
class StatefulSet extends workload.Workload {
|
|
55
|
+
constructor(scope, id, props) {
|
|
56
|
+
super(scope, id, props);
|
|
57
|
+
this.resourceType = 'statefulsets';
|
|
58
|
+
this.hasAutoscaler = false;
|
|
59
|
+
this.apiObject = new k8s.KubeStatefulSet(this, 'Resource', {
|
|
60
|
+
metadata: props.metadata,
|
|
61
|
+
spec: cdk8s_1.Lazy.any({ produce: () => this._toKube() }),
|
|
62
|
+
});
|
|
63
|
+
this.service = props.service ?? this._createHeadlessService();
|
|
64
|
+
this.apiObject.addDependency(this.service);
|
|
65
|
+
this.replicas = props.replicas;
|
|
66
|
+
this.strategy = props.strategy ?? StatefulSetUpdateStrategy.rollingUpdate();
|
|
67
|
+
this.podManagementPolicy = props.podManagementPolicy ?? PodManagementPolicy.ORDERED_READY;
|
|
68
|
+
this.minReady = props.minReady ?? cdk8s_1.Duration.seconds(0);
|
|
69
|
+
props?.volumeClaimTemplates?.forEach(template => this.addVolumeClaimTemplate(template));
|
|
70
|
+
this.service.select(this);
|
|
71
|
+
if (this.isolate) {
|
|
72
|
+
this.connections.isolate();
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
addVolumeClaimTemplate(template) {
|
|
76
|
+
if (this.volumeClaimTemplates?.some(t => t.name === template.name)) {
|
|
77
|
+
throw new Error(`A volume claim template with name "${template.name}" already exists`);
|
|
78
|
+
}
|
|
79
|
+
this.volumeClaimTemplates = this.volumeClaimTemplates
|
|
80
|
+
? [...this.volumeClaimTemplates, template]
|
|
81
|
+
: [template];
|
|
82
|
+
}
|
|
83
|
+
_createHeadlessService() {
|
|
84
|
+
const myPorts = container.extractContainerPorts(this);
|
|
85
|
+
const myPortNumbers = myPorts.map(p => p.number);
|
|
86
|
+
const ports = myPorts.map(p => ({
|
|
87
|
+
port: p.number,
|
|
88
|
+
targetPort: p.number,
|
|
89
|
+
protocol: p.protocol,
|
|
90
|
+
name: p.name,
|
|
91
|
+
}));
|
|
92
|
+
if (ports.length === 0) {
|
|
93
|
+
throw new Error(`Unable to create a service for the stateful set ${this.name}: StatefulSet ports cannot be determined.`);
|
|
94
|
+
}
|
|
95
|
+
// validate the ports are owned by our containers
|
|
96
|
+
for (const port of ports) {
|
|
97
|
+
const targetPort = port.targetPort ?? port.port;
|
|
98
|
+
if (!myPortNumbers.includes(targetPort)) {
|
|
99
|
+
throw new Error(`Unable to expose stateful set ${this.name} via a service: Port ${targetPort} is not exposed by any container`);
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
const metadata = { namespace: this.metadata.namespace };
|
|
103
|
+
return new service.Service(this, 'Service', {
|
|
104
|
+
selector: this,
|
|
105
|
+
ports,
|
|
106
|
+
metadata,
|
|
107
|
+
clusterIP: 'None',
|
|
108
|
+
type: service.ServiceType.CLUSTER_IP,
|
|
109
|
+
});
|
|
110
|
+
}
|
|
111
|
+
/**
|
|
112
|
+
* @internal
|
|
113
|
+
*/
|
|
114
|
+
_toKube() {
|
|
115
|
+
return {
|
|
116
|
+
replicas: this.hasAutoscaler ? undefined : (this.replicas ?? 1),
|
|
117
|
+
serviceName: this.service.name,
|
|
118
|
+
minReadySeconds: this.minReady.toSeconds(),
|
|
119
|
+
template: {
|
|
120
|
+
metadata: this.podMetadata.toJson(),
|
|
121
|
+
spec: this._toPodSpec(),
|
|
122
|
+
},
|
|
123
|
+
selector: this._toLabelSelector(),
|
|
124
|
+
podManagementPolicy: this.podManagementPolicy,
|
|
125
|
+
volumeClaimTemplates: this._toPersistentVolumeClaims(),
|
|
126
|
+
updateStrategy: this.strategy._toKube(),
|
|
127
|
+
};
|
|
128
|
+
}
|
|
129
|
+
/**
|
|
130
|
+
* @internal
|
|
131
|
+
*/
|
|
132
|
+
_toPodSpec() {
|
|
133
|
+
const podSpec = super._toPodSpec();
|
|
134
|
+
return {
|
|
135
|
+
...podSpec,
|
|
136
|
+
volumes: this._filterPodSpecVolumes(podSpec),
|
|
137
|
+
};
|
|
138
|
+
}
|
|
139
|
+
/**
|
|
140
|
+
* @internal
|
|
141
|
+
*/
|
|
142
|
+
_filterPodSpecVolumes(podSpec) {
|
|
143
|
+
// When using volumeClaimTemplates, the volumes with matching names should not be included in the pod spec
|
|
144
|
+
if (this.volumeClaimTemplates && podSpec.volumes) {
|
|
145
|
+
const volumeClaimNames = this.volumeClaimTemplates.map(vct => vct.name);
|
|
146
|
+
const volumesWithNoTemplates = podSpec.volumes.filter(vol => !volumeClaimNames.includes(vol.name));
|
|
147
|
+
// If there are no volumes after filtering, don't include volumes property
|
|
148
|
+
if (volumesWithNoTemplates.length === 0) {
|
|
149
|
+
return undefined;
|
|
150
|
+
}
|
|
151
|
+
else {
|
|
152
|
+
return volumesWithNoTemplates;
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
return undefined;
|
|
156
|
+
}
|
|
157
|
+
/**
|
|
158
|
+
* @see IScalable.markHasAutoscaler()
|
|
159
|
+
*/
|
|
160
|
+
markHasAutoscaler() {
|
|
161
|
+
this.hasAutoscaler = true;
|
|
162
|
+
}
|
|
163
|
+
/**
|
|
164
|
+
* @see IScalable.toScalingTarget()
|
|
165
|
+
*/
|
|
166
|
+
toScalingTarget() {
|
|
167
|
+
return {
|
|
168
|
+
kind: this.apiObject.kind,
|
|
169
|
+
apiVersion: this.apiObject.apiVersion,
|
|
170
|
+
name: this.name,
|
|
171
|
+
containers: this.containers,
|
|
172
|
+
replicas: this.replicas,
|
|
173
|
+
};
|
|
174
|
+
}
|
|
175
|
+
/**
|
|
176
|
+
* @internal
|
|
177
|
+
*/
|
|
178
|
+
_toPersistentVolumeClaims() {
|
|
179
|
+
const volumeNames = this.containers.flatMap(it => it.mounts).map(mount => mount.volume.name);
|
|
180
|
+
this.volumeClaimTemplates?.forEach(t => {
|
|
181
|
+
if (!volumeNames.includes(t.name)) {
|
|
182
|
+
throw new Error(`Volume claim template with name "${t.name}" is not used by any container mount`);
|
|
183
|
+
}
|
|
184
|
+
});
|
|
185
|
+
return this.volumeClaimTemplates?.map(template => {
|
|
186
|
+
const resources = template.storage
|
|
187
|
+
? { requests: { storage: k8s_1.Quantity.fromString(template.storage.asString()) } }
|
|
188
|
+
: {};
|
|
189
|
+
return {
|
|
190
|
+
metadata: {
|
|
191
|
+
name: template.name,
|
|
192
|
+
},
|
|
193
|
+
spec: {
|
|
194
|
+
accessModes: template.accessModes,
|
|
195
|
+
storageClassName: template.storageClassName,
|
|
196
|
+
resources,
|
|
197
|
+
},
|
|
198
|
+
};
|
|
199
|
+
});
|
|
200
|
+
}
|
|
201
|
+
}
|
|
202
|
+
exports.StatefulSet = StatefulSet;
|
|
203
|
+
_a = JSII_RTTI_SYMBOL_1;
|
|
204
|
+
StatefulSet[_a] = { fqn: "cdk8s-plus-34.StatefulSet", version: "2.0.0" };
|
|
205
|
+
/**
|
|
206
|
+
* StatefulSet update strategies.
|
|
207
|
+
*/
|
|
208
|
+
class StatefulSetUpdateStrategy {
|
|
209
|
+
/**
|
|
210
|
+
* The controller will not automatically update the Pods in a StatefulSet.
|
|
211
|
+
* Users must manually delete Pods to cause the controller to create new Pods
|
|
212
|
+
* that reflect modifications.
|
|
213
|
+
*/
|
|
214
|
+
static onDelete() {
|
|
215
|
+
return new StatefulSetUpdateStrategy({
|
|
216
|
+
type: 'OnDelete',
|
|
217
|
+
});
|
|
218
|
+
}
|
|
219
|
+
/**
|
|
220
|
+
* The controller will delete and recreate each Pod in the StatefulSet.
|
|
221
|
+
* It will proceed in the same order as Pod termination (from the largest ordinal to the smallest),
|
|
222
|
+
* updating each Pod one at a time. The Kubernetes control plane waits until an updated
|
|
223
|
+
* Pod is Running and Ready prior to updating its predecessor.
|
|
224
|
+
*/
|
|
225
|
+
static rollingUpdate(options = {}) {
|
|
226
|
+
return new StatefulSetUpdateStrategy({
|
|
227
|
+
type: 'RollingUpdate',
|
|
228
|
+
rollingUpdate: { partition: options.partition ?? 0 },
|
|
229
|
+
});
|
|
230
|
+
}
|
|
231
|
+
constructor(strategy) {
|
|
232
|
+
this.strategy = strategy;
|
|
233
|
+
}
|
|
234
|
+
/**
|
|
235
|
+
* @internal
|
|
236
|
+
*/
|
|
237
|
+
_toKube() {
|
|
238
|
+
return this.strategy;
|
|
239
|
+
}
|
|
240
|
+
}
|
|
241
|
+
exports.StatefulSetUpdateStrategy = StatefulSetUpdateStrategy;
|
|
242
|
+
_b = JSII_RTTI_SYMBOL_1;
|
|
243
|
+
StatefulSetUpdateStrategy[_b] = { fqn: "cdk8s-plus-34.StatefulSetUpdateStrategy", version: "2.0.0" };
|
|
244
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RhdGVmdWwtc2V0LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vc3JjL3N0YXRlZnVsLXNldC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7OztBQUFBLGlDQUFrRDtBQUVsRCx5Q0FBeUM7QUFFekMscUNBQXFDO0FBQ3JDLHVDQUE4RztBQUU5RyxxQ0FBcUM7QUFDckMsdUNBQXVDO0FBRXZDOzs7Ozs7Ozs7O0dBVUc7QUFDSCxJQUFZLG1CQUdYO0FBSEQsV0FBWSxtQkFBbUI7SUFDN0IscURBQThCLENBQUE7SUFDOUIsNENBQXFCLENBQUE7QUFDdkIsQ0FBQyxFQUhXLG1CQUFtQixtQ0FBbkIsbUJBQW1CLFFBRzlCO0FBeUREOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBeUJHO0FBQ0gsTUFBYSxXQUFZLFNBQVEsUUFBUSxDQUFDLFFBQVE7SUFtQ2hELFlBQVksS0FBZ0IsRUFBRSxFQUFVLEVBQUUsS0FBdUI7UUFDL0QsS0FBSyxDQUFDLEtBQUssRUFBRSxFQUFFLEVBQUUsS0FBSyxDQUFDLENBQUM7UUFUVixpQkFBWSxHQUFHLGNBQWMsQ0FBQztRQUV2QyxrQkFBYSxHQUFHLEtBQUssQ0FBQztRQVMzQixJQUFJLENBQUMsU0FBUyxHQUFHLElBQUksR0FBRyxDQUFDLGVBQWUsQ0FBQyxJQUFJLEVBQUUsVUFBVSxFQUFFO1lBQ3pELFFBQVEsRUFBRSxLQUFLLENBQUMsUUFBUTtZQUN4QixJQUFJLEVBQUUsWUFBSSxDQUFDLEdBQUcsQ0FBQyxFQUFFLE9BQU8sRUFBRSxHQUFHLEVBQUUsQ0FBQyxJQUFJLENBQUMsT0FBTyxFQUFFLEVBQUUsQ0FBQztTQUNsRCxDQUFDLENBQUM7UUFDSCxJQUFJLENBQUMsT0FBTyxHQUFHLEtBQUssQ0FBQyxPQUFPLElBQUksSUFBSSxDQUFDLHNCQUFzQixFQUFFLENBQUM7UUFFOUQsSUFBSSxDQUFDLFNBQVMsQ0FBQyxhQUFhLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxDQUFDO1FBRTNDLElBQUks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QyxFQUFFO1NBQ3JELENBQUMsQ0FBQztJQUNMLENBQUM7SUFFRCxZQUFxQyxRQUF1QztRQUF2QyxhQUFRLEdBQVIsUUFBUSxDQUErQjtJQUM1RSxDQUFDO0lBRUQ7O09BRUc7SUFDSSxPQUFPO1FBQ1osT0FBTyxJQUFJLENBQUMsUUFBUSxDQUFDO0lBQ3ZCLENBQUM7O0FBbkNILDhEQXFDQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEFwaU9iamVjdCwgTGF6eSwgRHVyYXRpb24gfSBmcm9tICdjZGs4cyc7XG5pbXBvcnQgeyBDb25zdHJ1Y3QgfSBmcm9tICdjb25zdHJ1Y3RzJztcbmltcG9ydCAqIGFzIGNvbnRhaW5lciBmcm9tICcuL2NvbnRhaW5lcic7XG5pbXBvcnQgeyBJU2NhbGFibGUsIFNjYWxpbmdUYXJnZXQgfSBmcm9tICcuL2hvcml6b250YWwtcG9kLWF1dG9zY2FsZXInO1xuaW1wb3J0ICogYXMgazhzIGZyb20gJy4vaW1wb3J0cy9rOHMnO1xuaW1wb3J0IHsgS3ViZVBlcnNpc3RlbnRWb2x1bWVDbGFpbVByb3BzLCBQb2RTcGVjLCBRdWFudGl0eSwgVm9sdW1lUmVzb3VyY2VSZXF1aXJlbWVudHMgfSBmcm9tICcuL2ltcG9ydHMvazhzJztcbmltcG9ydCB7IFBlcnNpc3RlbnRWb2x1bWVDbGFpbVByb3BzIH0gZnJvbSAnLi9wdmMnO1xuaW1wb3J0ICogYXMgc2VydmljZSBmcm9tICcuL3NlcnZpY2UnO1xuaW1wb3J0ICogYXMgd29ya2xvYWQgZnJvbSAnLi93b3JrbG9hZCc7XG5cbi8qKlxuICogQ29udHJvbHMgaG93IHBvZHMgYXJlIGNyZWF0ZWQgZHVyaW5nIGluaXRpYWwgc2NhbGUgdXAsIHdoZW4gcmVwbGFjaW5nIHBvZHMgb24gbm9kZXMsXG4gKiBvciB3aGVuIHNjYWxpbmcgZG93bi5cbiAqXG4gKiBUaGUgZGVmYXVsdCBwb2xpY3kgaXMgYE9yZGVyZWRSZWFkeWAsIHdoZXJlIHBvZHMgYXJlIGNyZWF0ZWQgaW4gaW5jcmVhc2luZyBvcmRlclxuICogKHBvZC0wLCB0aGVuIHBvZC0xLCBldGMpIGFuZCB0aGUgY29udHJvbGxlciB3aWxsIHdhaXQgdW50aWwgZWFjaCBwb2QgaXMgcmVhZHkgYmVmb3JlXG4gKiBjb250aW51aW5nLiBXaGVuIHNjYWxpbmcgZG93biwgdGhlIHBvZHMgYXJlIHJlbW92ZWQgaW4gdGhlIG9wcG9zaXRlIG9yZGVyLlxuICpcbiAqIFRoZSBhbHRlcm5hdGl2ZSBwb2xpY3kgaXMgYFBhcmFsbGVsYCB3aGljaCB3aWxsIGNyZWF0ZSBwb2RzIGluIHBhcmFsbGVsIHRvIG1hdGNoIHRoZVxuICogZGVzaXJlZCBzY2FsZSB3aXRob3V0IHdhaXRpbmcsIGFuZCBvbiBzY2FsZSBkb3duIHdpbGwgZGVsZXRlIGFsbCBwb2RzIGF0IG9uY2UuXG4gKi9cbmV4cG9ydCBlbnVtIFBvZE1hbmFnZW1lbnRQb2xpY3kge1xuICBPUkRFUkVEX1JFQURZID0gJ09yZGVyZWRSZWFkeScsXG4gIFBBUkFMTEVMID0gJ1BhcmFsbGVsJyxcbn1cblxuLyoqXG4gKiBQcm9wZXJ0aWVzIGZvciBpbml0aWFsaXphdGlvbiBvZiBgU3RhdGVmdWxTZXRgLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIFN0YXRlZnVsU2V0UHJvcHMgZXh0ZW5kcyB3b3JrbG9hZC5Xb3JrbG9hZFByb3BzIHtcbiAgLyoqXG4gICAqIFNlcnZpY2UgdG8gYXNzb2NpYXRlIHdpdGggdGhlIHN0YXRlZnVsc2V0LlxuICAgKlxuICAgKiBAZGVmYXVsdCAtIEEgbmV3IGhlYWRsZXNzIHNlcnZpY2Ugd2lsbCBiZSBjcmVhdGVkLlxuICAgKi9cbiAgcmVhZG9ubHkgc2VydmljZT86IHNlcnZpY2UuU2VydmljZTtcblxuICAvKipcbiAgICogTnVtYmVyIG9mIGRlc2lyZWQgcG9kcy5cbiAgICpcbiAgICogQGRlZmF1bHQgMVxuICAgKi9cbiAgcmVhZG9ubHkgcmVwbGljYXM/OiBudW1iZXI7XG5cbiAgLyoqXG4gICAqIFBvZCBtYW5hZ2VtZW50IHBvbGljeSB0byB1c2UgZm9yIHRoaXMgc3RhdGVmdWxzZXQuXG4gICAqXG4gICAqIEBkZWZhdWx0IFBvZE1hbmFnZW1lbnRQb2xpY3kuT1JERVJFRF9SRUFEWVxuICAgKi9cbiAgcmVhZG9ubHkgcG9kTWFuYWdlbWVudFBvbGljeT86IFBvZE1hbmFnZW1lbnRQb2xpY3k7XG5cbiAgLyoqXG4gICAqIEluZGljYXRlcyB0aGUgU3RhdGVmdWxTZXRVcGRhdGVTdHJhdGVneSB0aGF0IHdpbGwgYmUgZW1wbG95ZWQgdG8gdXBkYXRlIFBvZHMgaW4gdGhlIFN0YXRlZnVsU2V0IHdoZW4gYSByZXZpc2lvbiBpcyBtYWRlIHRvIFRlbXBsYXRlLlxuICAgKlxuICAgKiBAZGVmYXVsdCAtIFJvbGxpbmdVcGRhdGUgd2l0aCBwYXJ0aXRpb24gc2V0IHRvIDBcbiAgICovXG4gIHJlYWRvbmx5IHN0cmF0ZWd5PzogU3RhdGVmdWxTZXRVcGRhdGVTdHJhdGVneTtcblxuICAvKipcbiAgICogTWluaW11bSBkdXJhdGlvbiBmb3Igd2hpY2ggYSBuZXdseSBjcmVhdGVkIHBvZCBzaG91bGQgYmUgcmVhZHkgd2l0aG91dCBhbnkgb2YgaXRzIGNvbnRhaW5lciBjcmFzaGluZyxcbiAgICogZm9yIGl0IHRvIGJlIGNvbnNpZGVyZWQgYXZhaWxhYmxlLiBaZXJvIG1lYW5zIHRoZSBwb2Qgd2lsbCBiZSBjb25zaWRlcmVkIGF2YWlsYWJsZSBhcyBzb29uIGFzIGl0IGlzIHJlYWR5LlxuICAgKlxuICAgKiBUaGlzIGlzIGFuIGFscGhhIGZpZWxkIGFuZCByZXF1aXJlcyBlbmFibGluZyBTdGF0ZWZ1bFNldE1pblJlYWR5U2Vjb25kcyBmZWF0dXJlIGdhdGUuXG4gICAqXG4gICAqIEBzZWUgaHR0cHM6Ly9rdWJlcm5ldGVzLmlvL2RvY3MvY29uY2VwdHMvd29ya2xvYWRzL2NvbnRyb2xsZXJzL2RlcGxveW1lbnQvI21pbi1yZWFkeS1zZWNvbmRzXG4gICAqIEBkZWZhdWx0IER1cmF0aW9uLnNlY29uZHMoMClcbiAgICovXG4gIHJlYWRvbmx5IG1pblJlYWR5PzogRHVyYXRpb247XG5cbiAgLyoqXG4gICAqIEEgbGlzdCBvZiBQZXJzaXN0ZW50Vm9sdW1lQ2xhaW0gdGVtcGxhdGVzIHRoYXQgd2lsbCBiZSBjcmVhdGVkIGZvciBlYWNoIHBvZCBpbiB0aGUgU3RhdGVmdWxTZXQuXG4gICAqIFRoZSBTdGF0ZWZ1bFNldCBjb250cm9sbGVyIGNyZWF0ZXMgYSBQVkMgYW5kIGEgUFYgZm9yIGVhY2ggdGVtcGxhdGUgYmFzZWQgb24gdGhlIHBvZCdzIG9yZGluYWwgaW5kZXgsXG4gICAqIGVuc3VyaW5nIHN0YWJsZSBzdG9yYWdlIGFjcm9zcyBwb2QgcmVzdGFydHMgYW5kIHJlc2NoZWR1bGluZy5cbiAgICpcbiAgICogRWFjaCBjbGFpbSBpbiB0aGlzIGxpc3QgbXVzdCBoYXZlIGF0IGxlYXN0IG9uZSBtYXRjaGluZyAoYnkgbmFtZSkgdm9sdW1lTW91bnQgaW4gb25lIG9mIHRoZSBjb250YWluZXJzLlxuICAgKlxuICAgKiBAZGVmYXVsdCAtIE5vIHZvbHVtZSBjbGFpbSB0ZW1wbGF0ZXMgd2lsbCBiZSBjcmVhdGVkLlxuICAgKi9cbiAgcmVhZG9ubHkgdm9sdW1lQ2xhaW1UZW1wbGF0ZXM/OiBQZXJzaXN0ZW50Vm9sdW1lQ2xhaW1UZW1wbGF0ZVByb3BzW107XG59XG5cbi8qKlxuICogU3RhdGVmdWxTZXQgaXMgdGhlIHdvcmtsb2FkIEFQSSBvYmplY3QgdXNlZCB0byBtYW5hZ2Ugc3RhdGVmdWwgYXBwbGljYXRpb25zLlxuICpcbiAqIE1hbmFnZXMgdGhlIGRlcGxveW1lbnQgYW5kIHNjYWxpbmcgb2YgYSBzZXQgb2YgUG9kcywgYW5kIHByb3ZpZGVzIGd1YXJhbnRlZXNcbiAqIGFib3V0IHRoZSBvcmRlcmluZyBhbmQgdW5pcXVlbmVzcyBvZiB0aGVzZSBQb2RzLlxuICpcbiAqIExpa2UgYSBEZXBsb3ltZW50LCBhIFN0YXRlZnVsU2V0IG1hbmFnZXMgUG9kcyB0aGF0IGFyZSBiYXNlZCBvbiBhbiBpZGVudGljYWxcbiAqIGNvbnRhaW5lciBzcGVjLiBVbmxpa2UgYSBEZXBsb3ltZW50LCBhIFN0YXRlZnVsU2V0IG1haW50YWlucyBhIHN0aWNreSBpZGVudGl0eVxuICogZm9yIGVhY2ggb2YgdGhlaXIgUG9kcy4gVGhlc2UgcG9kcyBhcmUgY3JlYXRlZCBmcm9tIHRoZSBzYW1lIHNwZWMsIGJ1dCBhcmUgbm90XG4gKiBpbnRlcmNoYW5nZWFibGU6IGVhY2ggaGFzIGEgcGVyc2lzdGVudCBpZGVudGlmaWVyIHRoYXQgaXQgbWFpbnRhaW5zIGFjcm9zcyBhbnlcbiAqIHJlc2NoZWR1bGluZy5cbiAqXG4gKiBJZiB5b3Ugd2FudCB0byB1c2Ugc3RvcmFnZSB2b2x1bWVzIHRvIHByb3ZpZGUgcGVyc2lzdGVuY2UgZm9yIHlvdXIgd29ya2xvYWQsIHlvdVxuICogY2FuIHVzZSBhIFN0YXRlZnVsU2V0IGFzIHBhcnQgb2YgdGhlIHNvbHV0aW9uLiBBbHRob3VnaCBpbmRpdmlkdWFsIFBvZHMgaW4gYSBTdGF0ZWZ1bFNldFxuICogYXJlIHN1c2NlcHRpYmxlIHRvIGZhaWx1cmUsIHRoZSBwZXJzaXN0ZW50IFBvZCBpZGVudGlmaWVycyBtYWtlIGl0IGVhc2llciB0byBtYXRjaCBleGlzdGluZ1xuICogdm9sdW1lcyB0byB0aGUgbmV3IFBvZHMgdGhhdCByZXBsYWNlIGFueSB0aGF0IGhhdmUgZmFpbGVkLlxuICpcbiAqIFVzaW5nIFN0YXRlZnVsU2V0c1xuICogLS0tLS0tLS0tLS0tLS0tLS0tXG4gKiBTdGF0ZWZ1bFNldHMgYXJlIHZhbHVhYmxlIGZvciBhcHBsaWNhdGlvbnMgdGhhdCByZXF1aXJlIG9uZSBvciBtb3JlIG9mIHRoZSBmb2xsb3dpbmcuXG4gKlxuICogLSBTdGFibGUsIHVuaXF1ZSBuZXR3b3JrIGlkZW50aWZpZXJzLlxuICogLSBTdGFibGUsIHBlcnNpc3RlbnQgc3RvcmFnZS5cbiAqIC0gT3JkZXJlZCwgZ3JhY2VmdWwgZGVwbG95bWVudCBhbmQgc2NhbGluZy5cbiAqIC0gT3JkZXJlZCwgYXV0b21hdGVkIHJvbGxpbmcgdXBkYXRlcy5cbiAqL1xuZXhwb3J0IGNsYXNzIFN0YXRlZnVsU2V0IGV4dGVuZHMgd29ya2xvYWQuV29ya2xvYWQgaW1wbGVtZW50cyBJU2NhbGFibGUge1xuICAvKipcbiAgICogTnVtYmVyIG9mIGRlc2lyZWQgcG9kcy5cbiAgICovXG4gIHB1YmxpYyByZWFkb25seSByZXBsaWNhcz86IG51bWJlcjtcblxuICAvKipcbiAgICogTWFuYWdlbWVudCBwb2xpY3kgdG8gdXNlIGZvciB0aGUgc2V0LlxuICAgKi9cbiAgcHVibGljIHJlYWRvbmx5IHBvZE1hbmFnZW1lbnRQb2xpY3k6IFBvZE1hbmFnZW1lbnRQb2xpY3k7XG5cbiAgLyoqXG4gICAqIFRoZSB1cGRhdGUgc3RhcnRlZ3kgb2YgdGhpcyBzdGF0ZWZ1bCBzZXQuXG4gICAqL1xuICBwdWJsaWMgcmVhZG9ubHkgc3RyYXRlZ3k6IFN0YXRlZnVsU2V0VXBkYXRlU3RyYXRlZ3k7XG5cbiAgLyoqXG4gICAqIE1pbmltdW0gZHVyYXRpb24gZm9yIHdoaWNoIGEgbmV3bHkgY3JlYXRlZCBwb2Qgc2hvdWxkIGJlIHJlYWR5IHdpdGhvdXRcbiAgICogYW55IG9mIGl0cyBjb250YWluZXIgY3Jhc2hpbmcsIGZvciBpdCB0byBiZSBjb25zaWRlcmVkIGF2YWlsYWJsZS5cbiAgICovXG4gIHB1YmxpYyByZWFkb25seSBtaW5SZWFkeTogRHVyYXRpb247XG5cbiAgLyoqXG4gICAqIEBzZWUgYmFzZS5SZXNvdXJjZS5hcGlPYmplY3RcbiAgICovXG4gIHByb3RlY3RlZCByZWFkb25seSBhcGlPYmplY3Q6IEFwaU9iamVjdDtcblxuICBwdWJsaWMgcmVhZG9ubHkgcmVzb3VyY2VUeXBlID0gJ3N0YXRlZnVsc2V0cyc7XG5cbiAgcHVibGljIGhhc0F1dG9zY2FsZXIgPSBmYWxzZTtcblxuICBwdWJsaWMgcmVhZG9ubHkgc2VydmljZTogc2VydmljZS5TZXJ2aWNlO1xuXG4gIHB1YmxpYyB2b2x1bWVDbGFpbVRlbXBsYXRlcz86IFBlcnNpc3RlbnRWb2x1bWVDbGFpbVRlbXBsYXRlUHJvcHNbXTtcblxuICBjb25zdHJ1Y3RvcihzY29wZTogQ29uc3RydWN0LCBpZDogc3RyaW5nLCBwcm9wczogU3RhdGVmdWxTZXRQcm9wcykge1xuICAgIHN1cGVyKHNjb3BlLCBpZCwgcHJvcHMpO1xuXG4gICAgdGhpcy5hcGlPYmplY3QgPSBuZXcgazhzLkt1YmVTdGF0ZWZ1bFNldCh0aGlzLCAnUmVzb3VyY2UnLCB7XG4gICAgICBtZXRhZGF0YTogcHJvcHMubWV0YWRhdGEsXG4gICAgICBzcGVjOiBMYXp5LmFueSh7IHByb2R1Y2U6ICgpID0+IHRoaXMuX3RvS3ViZSgpIH0pLFxuICAgIH0pO1xuICAgIHRoaXMuc2VydmljZSA9IHByb3BzLnNlcnZpY2UgPz8gdGhpcy5fY3JlYXRlSGVhZGxlc3NTZXJ2aWNlKCk7XG5cbiAgICB0aGlzLmFwaU9iamVjdC5hZGREZXBlbmRlbmN5KHRoaXMuc2VydmljZSk7XG5cbiAgICB0aGlzLnJlcGxpY2FzID0gcHJvcHMucmVwbGljYXM7XG4gICAgdGhpcy5zdHJhdGVneSA9IHByb3BzLnN0cmF0ZWd5ID8/IFN0YXRlZnVsU2V0VXBkYXRlU3RyYXRlZ3kucm9sbGluZ1VwZGF0ZSgpO1xuICAgIHRoaXMucG9kTWFuYWdlbWVudFBvbGljeSA9IHByb3BzLnBvZE1hbmFnZW1lbnRQb2xpY3kgPz8gUG9kTWFuYWdlbWVudFBvbGljeS5PUkRFUkVEX1JFQURZO1xuICAgIHRoaXMubWluUmVhZHkgPSBwcm9wcy5taW5SZWFkeSA/PyBEdXJhdGlvbi5zZWNvbmRzKDApO1xuICAgIHByb3BzPy52b2x1bWVDbGFpbVRlbXBsYXRlcz8uZm9yRWFjaCh0ZW1wbGF0ZSA9PiB0aGlzLmFkZFZvbHVtZUNsYWltVGVtcGxhdGUodGVtcGxhdGUpKTtcbiAgICB0aGlzLnNlcnZpY2Uuc2VsZWN0KHRoaXMpO1xuXG4gICAgaWYgKHRoaXMuaXNvbGF0ZSkge1xuICAgICAgdGhpcy5jb25uZWN0aW9ucy5pc29sYXRlKCk7XG4gICAgfVxuICB9XG5cbiAgcHVibGljIGFkZFZvbHVtZUNsYWltVGVtcGxhdGUodGVtcGxhdGU6IFBlcnNpc3RlbnRWb2x1bWVDbGFpbVRlbXBsYXRlUHJvcHMpIHtcbiAgICBpZiAodGhpcy52b2x1bWVDbGFpbVRlbXBsYXRlcz8uc29tZSh0ID0+IHQubmFtZSA9PT0gdGVtcGxhdGUubmFtZSkpIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcihgQSB2b2x1bWUgY2xhaW0gdGVtcGxhdGUgd2l0aCBuYW1lIFwiJHt0ZW1wbGF0ZS5uYW1lfVwiIGFscmVhZHkgZXhpc3RzYCk7XG4gICAgfVxuICAgIHRoaXMudm9sdW1lQ2xhaW1UZW1wbGF0ZXMgPSB0aGlzLnZvbHVtZUNsYWltVGVtcGxhdGVzXG4gICAgICA/IFsuLi50aGlzLnZvbHVtZUNsYWltVGVtcGxhdGVzLCB0ZW1wbGF0ZV1cbiAgICAgIDogW3RlbXBsYXRlXTtcbiAgfVxuXG4gIHByaXZhdGUgX2NyZWF0ZUhlYWRsZXNzU2VydmljZSgpIHtcbiAgICBjb25zdCBteVBvcnRzID0gY29udGFpbmVyLmV4dHJhY3RDb250YWluZXJQb3J0cyh0aGlzKTtcbiAgICBjb25zdCBteVBvcnROdW1iZXJzID0gbXlQb3J0cy5tYXAocCA9PiBwLm51bWJlcik7XG4gICAgY29uc3QgcG9ydHM6IHNlcnZpY2UuU2VydmljZVBvcnRbXSA9IG15UG9ydHMubWFwKHAgPT4gKHtcbiAgICAgIHBvcnQ6IHAubnVtYmVyLFxuICAgICAgdGFyZ2V0UG9ydDogcC5udW1iZXIsXG4gICAgICBwcm90b2NvbDogcC5wcm90b2NvbCxcbiAgICAgIG5hbWU6IHAubmFtZSxcbiAgICB9KSk7XG4gICAgaWYgKHBvcnRzLmxlbmd0aCA9PT0gMCkge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKGBVbmFibGUgdG8gY3JlYXRlIGEgc2VydmljZSBmb3IgdGhlIHN0YXRlZnVsIHNldCAke3RoaXMubmFtZX06IFN0YXRlZnVsU2V0IHBvcnRzIGNhbm5vdCBiZSBkZXRlcm1pbmVkLmApO1xuICAgIH1cblxuICAgIC8vIHZhbGlkYXRlIHRoZSBwb3J0cyBhcmUgb3duZWQgYnkgb3VyIGNvbnRhaW5lcnNcbiAgICBmb3IgKGNvbnN0IHBvcnQgb2YgcG9ydHMpIHtcbiAgICAgIGNvbnN0IHRhcmdldFBvcnQgPSBwb3J0LnRhcmdldFBvcnQgPz8gcG9ydC5wb3J0O1xuICAgICAgaWYgKCFteVBvcnROdW1iZXJzLmluY2x1ZGVzKHRhcmdldFBvcnQpKSB7XG4gICAgICAgIHRocm93IG5ldyBFcnJvcihgVW5hYmxlIHRvIGV4cG9zZSBzdGF0ZWZ1bCBzZXQgJHt0aGlzLm5hbWV9IHZpYSBhIHNlcnZpY2U6IFBvcnQgJHt0YXJnZXRQb3J0fSBpcyBub3QgZXhwb3NlZCBieSBhbnkgY29udGFpbmVyYCk7XG4gICAgICB9XG4gICAgfVxuXG4gICAgY29uc3QgbWV0YWRhdGE6IGFueSA9IHsgbmFtZXNwYWNlOiB0aGlzLm1ldGFkYXRhLm5hbWVzcGFjZSB9O1xuICAgIHJldHVybiBuZXcgc2VydmljZS5TZXJ2aWNlKHRoaXMsICdTZXJ2aWNlJywge1xuICAgICAgc2VsZWN0b3I6IHRoaXMsXG4gICAgICBwb3J0cyxcbiAgICAgIG1ldGFkYXRhLFxuICAgICAgY2x1c3RlcklQOiAnTm9uZScsXG4gICAgICB0eXBlOiBzZXJ2aWNlLlNlcnZpY2VUeXBlLkNMVVNURVJfSVAsXG4gICAgfSk7XG5cbiAgfVxuXG4gIC8qKlxuICAgKiBAaW50ZXJuYWxcbiAgICovXG4gIHB1YmxpYyBfdG9LdWJlKCk6IGs4cy5TdGF0ZWZ1bFNldFNwZWMge1xuICAgIHJldHVybiB7XG4gICAgICByZXBsaWNhczogdGhpcy5oYXNBdXRvc2NhbGVyID8gdW5kZWZpbmVkIDogKHRoaXMucmVwbGljYXMgPz8gMSksXG4gICAgICBzZXJ2aWNlTmFtZTogdGhpcy5zZXJ2aWNlLm5hbWUsXG4gICAgICBtaW5SZWFkeVNlY29uZHM6IHRoaXMubWluUmVhZHkudG9TZWNvbmRzKCksXG4gICAgICB0ZW1wbGF0ZToge1xuICAgICAgICBtZXRhZGF0YTogdGhpcy5wb2RNZXRhZGF0YS50b0pzb24oKSxcbiAgICAgICAgc3BlYzogdGhpcy5fdG9Qb2RTcGVjKCksXG4gICAgICB9LFxuICAgICAgc2VsZWN0b3I6IHRoaXMuX3RvTGFiZWxTZWxlY3RvcigpLFxuICAgICAgcG9kTWFuYWdlbWVudFBvbGljeTogdGhpcy5wb2RNYW5hZ2VtZW50UG9saWN5LFxuICAgICAgdm9sdW1lQ2xhaW1UZW1wbGF0ZXM6IHRoaXMuX3RvUGVyc2lzdGVudFZvbHVtZUNsYWltcygpLFxuICAgICAgdXBkYXRlU3RyYXRlZ3k6IHRoaXMuc3RyYXRlZ3kuX3RvS3ViZSgpLFxuICAgIH07XG4gIH1cblxuICAvKipcbiAgICogQGludGVybmFsXG4gICAqL1xuICBwdWJsaWMgX3RvUG9kU3BlYygpOiBrOHMuUG9kU3BlYyB7XG4gICAgY29uc3QgcG9kU3BlYyA9IHN1cGVyLl90b1BvZFNwZWMoKTtcbiAgICByZXR1cm4ge1xuICAgICAgLi4ucG9kU3BlYyxcbiAgICAgIHZvbHVtZXM6IHRoaXMuX2ZpbHRlclBvZFNwZWNWb2x1bWVzKHBvZFNwZWMpLFxuICAgIH07XG4gIH1cblxuICAvKipcbiAgICogQGludGVybmFsXG4gICAqL1xuICBwcml2YXRlIF9maWx0ZXJQb2RTcGVjVm9sdW1lcyhwb2RTcGVjOiBQb2RTcGVjKSB7XG4gICAgLy8gV2hlbiB1c2luZyB2b2x1bWVDbGFpbVRlbXBsYXRlcywgdGhlIHZvbHVtZXMgd2l0aCBtYXRjaGluZyBuYW1lcyBzaG91bGQgbm90IGJlIGluY2x1ZGVkIGluIHRoZSBwb2Qgc3BlY1xuICAgIGlmICh0aGlzLnZvbHVtZUNsYWltVGVtcGxhdGVzICYmIHBvZFNwZWMudm9sdW1lcykge1xuICAgICAgY29uc3Qgdm9sdW1lQ2xhaW1OYW1lcyA9IHRoaXMudm9sdW1lQ2xhaW1UZW1wbGF0ZXMubWFwKHZjdCA9PiB2Y3QubmFtZSk7XG4gICAgICBjb25zdCB2b2x1bWVzV2l0aE5vVGVtcGxhdGVzID0gcG9kU3BlYy52b2x1bWVzLmZpbHRlcih2b2wgPT4gIXZvbHVtZUNsYWltTmFtZXMuaW5jbHVkZXModm9sLm5hbWUpKTtcblxuICAgICAgLy8gSWYgdGhlcmUgYXJlIG5vIHZvbHVtZXMgYWZ0ZXIgZmlsdGVyaW5nLCBkb24ndCBpbmNsdWRlIHZvbHVtZXMgcHJvcGVydHlcbiAgICAgIGlmICh2b2x1bWVzV2l0aE5vVGVtcGxhdGVzLmxlbmd0aCA9PT0gMCkge1xuICAgICAgICByZXR1cm4gdW5kZWZpbmVkO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgcmV0dXJuIHZvbHVtZXNXaXRoTm9UZW1wbGF0ZXM7XG4gICAgICB9XG4gICAgfVxuICAgIHJldHVybiB1bmRlZmluZWQ7XG4gIH1cblxuICAvKipcbiAgICogQHNlZSBJU2NhbGFibGUubWFya0hhc0F1dG9zY2FsZXIoKVxuICAgKi9cbiAgcHVibGljIG1hcmtIYXNBdXRvc2NhbGVyKCkge1xuICAgIHRoaXMuaGFzQXV0b3NjYWxlciA9IHRydWU7XG4gIH1cblxuICAvKipcbiAgICogQHNlZSBJU2NhbGFibGUudG9TY2FsaW5nVGFyZ2V0KClcbiAgICovXG4gIHB1YmxpYyB0b1NjYWxpbmdUYXJnZXQoKTogU2NhbGluZ1RhcmdldCB7XG4gICAgcmV0dXJuIHtcbiAgICAgIGtpbmQ6IHRoaXMuYXBpT2JqZWN0LmtpbmQsXG4gICAgICBhcGlWZXJzaW9uOiB0aGlzLmFwaU9iamVjdC5hcGlWZXJzaW9uLFxuICAgICAgbmFtZTogdGhpcy5uYW1lLFxuICAgICAgY29udGFpbmVyczogdGhpcy5jb250YWluZXJzLFxuICAgICAgcmVwbGljYXM6IHRoaXMucmVwbGljYXMsXG4gICAgfTtcbiAgfVxuXG4gIC8qKlxuICAgKiBAaW50ZXJuYWxcbiAgICovXG4gIHByaXZhdGUgX3RvUGVyc2lzdGVudFZvbHVtZUNsYWltcygpOiBLdWJlUGVyc2lzdGVudFZvbHVtZUNsYWltUHJvcHNbXSB8IHVuZGVmaW5lZCB7XG4gICAgY29uc3Qgdm9sdW1lTmFtZXMgPSB0aGlzLmNvbnRhaW5lcnMuZmxhdE1hcChpdCA9PiBpdC5tb3VudHMpLm1hcChtb3VudCA9PiBtb3VudC52b2x1bWUubmFtZSk7XG4gICAgdGhpcy52b2x1bWVDbGFpbVRlbXBsYXRlcz8uZm9yRWFjaCh0ID0+IHtcbiAgICAgIGlmICghdm9sdW1lTmFtZXMuaW5jbHVkZXModC5uYW1lKSkge1xuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYFZvbHVtZSBjbGFpbSB0ZW1wbGF0ZSB3aXRoIG5hbWUgXCIke3QubmFtZX1cIiBpcyBub3QgdXNlZCBieSBhbnkgY29udGFpbmVyIG1vdW50YCk7XG4gICAgICB9XG4gICAgfSk7XG4gICAgcmV0dXJuIHRoaXMudm9sdW1lQ2xhaW1UZW1wbGF0ZXM/Lm1hcCh0ZW1wbGF0ZSA9PiB7XG4gICAgICBjb25zdCByZXNvdXJjZXM6IFZvbHVtZVJlc291cmNlUmVxdWlyZW1lbnRzID0gdGVtcGxhdGUuc3RvcmFnZVxuICAgICAgICA/IHsgcmVxdWVzdHM6IHsgc3RvcmFnZTogUXVhbnRpdHkuZnJvbVN0cmluZyh0ZW1wbGF0ZS5zdG9yYWdlLmFzU3RyaW5nKCkpIH0gfVxuICAgICAgICA6IHt9O1xuICAgICAgcmV0dXJuIHtcbiAgICAgICAgbWV0YWRhdGE6IHtcbiAgICAgICAgICBuYW1lOiB0ZW1wbGF0ZS5uYW1lLFxuICAgICAgICB9LFxuICAgICAgICBzcGVjOiB7XG4gICAgICAgICAgYWNjZXNzTW9kZXM6IHRlbXBsYXRlLmFjY2Vzc01vZGVzLFxuICAgICAgICAgIHN0b3JhZ2VDbGFzc05hbWU6IHRlbXBsYXRlLnN0b3JhZ2VDbGFzc05hbWUsXG4gICAgICAgICAgcmVzb3VyY2VzLFxuICAgICAgICB9LFxuICAgICAgfTtcbiAgICB9KTtcbiAgfVxufVxuXG4vKipcbiAqIE9wdGlvbnMgZm9yIGBTdGF0ZWZ1bFNldFVwZGF0ZVN0cmF0ZWd5LnJvbGxpbmdVcGRhdGVgLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIFN0YXRlZnVsU2V0VXBkYXRlU3RyYXRlZ3lSb2xsaW5nVXBkYXRlT3B0aW9ucyB7XG5cbiAgLyoqXG4gICAqIElmIHNwZWNpZmllZCwgYWxsIFBvZHMgd2l0aCBhbiBvcmRpbmFsIHRoYXQgaXMgZ3JlYXRlciB0aGFuIG9yIGVxdWFsIHRvIHRoZSBwYXJ0aXRpb24gd2lsbFxuICAgKiBiZSB1cGRhdGVkIHdoZW4gdGhlIFN0YXRlZnVsU2V0J3MgLnNwZWMudGVtcGxhdGUgaXMgdXBkYXRlZC4gQWxsIFBvZHMgd2l0aCBhbiBvcmRpbmFsIHRoYXRcbiAgICogaXMgbGVzcyB0aGFuIHRoZSBwYXJ0aXRpb24gd2lsbCBub3QgYmUgdXBkYXRlZCwgYW5kLCBldmVuIGlmIHRoZXkgYXJlIGRlbGV0ZWQsIHRoZXkgd2lsbCBiZVxuICAgKiByZWNyZWF0ZWQgYXQgdGhlIHByZXZpb3VzIHZlcnNpb24uXG4gICAqXG4gICAqIElmIHRoZSBwYXJ0aXRpb24gaXMgZ3JlYXRlciB0aGFuIHJlcGxpY2FzLCB1cGRhdGVzIHRvIHRoZSBwb2QgdGVtcGxhdGUgd2lsbCBub3QgYmUgcHJvcGFnYXRlZCB0byBQb2RzLlxuICAgKiBJbiBtb3N0IGNhc2VzIHlvdSB3aWxsIG5vdCBuZWVkIHRvIHVzZSBhIHBhcnRpdGlvbiwgYnV0IHRoZXkgYXJlIHVzZWZ1bCBpZiB5b3Ugd2FudCB0byBzdGFnZSBhblxuICAgKiB1cGRhdGUsIHJvbGwgb3V0IGEgY2FuYXJ5LCBvciBwZXJmb3JtIGEgcGhhc2VkIHJvbGwgb3V0LlxuICAgKlxuICAgKiBAc2VlIGh0dHBzOi8va3ViZXJuZXRlcy5pby9kb2NzL2NvbmNlcHRzL3dvcmtsb2Fkcy9jb250cm9sbGVycy9zdGF0ZWZ1bHNldC8jcGFydGl0aW9uc1xuICAgKiBAZGVmYXVsdCAwXG4gICAqL1xuICByZWFkb25seSBwYXJ0aXRpb24/OiBudW1iZXI7XG5cbn1cblxuLyoqXG4gKiBTdGF0ZWZ1bFNldCB1cGRhdGUgc3RyYXRlZ2llcy5cbiAqL1xuZXhwb3J0IGNsYXNzIFN0YXRlZnVsU2V0VXBkYXRlU3RyYXRlZ3kge1xuXG4gIC8qKlxuICAgKiBUaGUgY29udHJvbGxlciB3aWxsIG5vdCBhdXRvbWF0aWNhbGx5IHVwZGF0ZSB0aGUgUG9kcyBpbiBhIFN0YXRlZnVsU2V0LlxuICAgKiBVc2VycyBtdXN0IG1hbnVhbGx5IGRlbGV0ZSBQb2RzIHRvIGNhdXNlIHRoZSBjb250cm9sbGVyIHRvIGNyZWF0ZSBuZXcgUG9kc1xuICAgKiB0aGF0IHJlZmxlY3QgbW9kaWZpY2F0aW9ucy5cbiAgICovXG4gIHB1YmxpYyBzdGF0aWMgb25EZWxldGUoKTogU3RhdGVmdWxTZXRVcGRhdGVTdHJhdGVneSB7XG4gICAgcmV0dXJuIG5ldyBTdGF0ZWZ1bFNldFVwZGF0ZVN0cmF0ZWd5KHtcbiAgICAgIHR5cGU6ICdPbkRlbGV0ZScsXG4gICAgfSk7XG4gIH1cblxuICAvKipcbiAgICogVGhlIGNvbnRyb2xsZXIgd2lsbCBkZWxldGUgYW5kIHJlY3JlYXRlIGVhY2ggUG9kIGluIHRoZSBTdGF0ZWZ1bFNldC5cbiAgICogSXQgd2lsbCBwcm9jZWVkIGluIHRoZSBzYW1lIG9yZGVyIGFzIFBvZCB0ZXJtaW5hdGlvbiAoZnJvbSB0aGUgbGFyZ2VzdCBvcmRpbmFsIHRvIHRoZSBzbWFsbGVzdCksXG4gICAqIHVwZGF0aW5nIGVhY2ggUG9kIG9uZSBhdCBhIHRpbWUuIFRoZSBLdWJlcm5ldGVzIGNvbnRyb2wgcGxhbmUgd2FpdHMgdW50aWwgYW4gdXBkYXRlZFxuICAgKiBQb2QgaXMgUnVubmluZyBhbmQgUmVhZHkgcHJpb3IgdG8gdXBkYXRpbmcgaXRzIHByZWRlY2Vzc29yLlxuICAgKi9cbiAgcHVibGljIHN0YXRpYyByb2xsaW5nVXBkYXRlKG9wdGlvbnM6IFN0YXRlZnVsU2V0VXBkYXRlU3RyYXRlZ3lSb2xsaW5nVXBkYXRlT3B0aW9ucyA9IHt9KTogU3RhdGVmdWxTZXRVcGRhdGVTdHJhdGVneSB7XG5cbiAgICByZXR1cm4gbmV3IFN0YXRlZnVsU2V0VXBkYXRlU3RyYXRlZ3koe1xuICAgICAgdHlwZTogJ1JvbGxpbmdVcGRhdGUnLFxuICAgICAgcm9sbGluZ1VwZGF0ZTogeyBwYXJ0aXRpb246IG9wdGlvbnMucGFydGl0aW9uID8/IDAgfSxcbiAgICB9KTtcbiAgfVxuXG4gIHByaXZhdGUgY29uc3RydWN0b3IocHJpdmF0ZSByZWFkb25seSBzdHJhdGVneTogazhzLlN0YXRlZnVsU2V0VXBkYXRlU3RyYXRlZ3kpIHtcbiAgfVxuXG4gIC8qKlxuICAgKiBAaW50ZXJuYWxcbiAgICovXG4gIHB1YmxpYyBfdG9LdWJlKCk6IGs4cy5TdGF0ZWZ1bFNldFVwZGF0ZVN0cmF0ZWd5IHtcbiAgICByZXR1cm4gdGhpcy5zdHJhdGVneTtcbiAgfVxuXG59XG5cbi8qKlxuICogQSBQZXJzaXN0ZW50Vm9sdW1lQ2xhaW0gdGVtcGxhdGUgZm9yIFN0YXRlZnVsU2V0c1xuICovXG5leHBvcnQgaW50ZXJmYWNlIFBlcnNpc3RlbnRWb2x1bWVDbGFpbVRlbXBsYXRlUHJvcHMgZXh0ZW5kcyBQZXJzaXN0ZW50Vm9sdW1lQ2xhaW1Qcm9wcyB7XG4gIC8qKlxuICAgKiBUaGUgbmFtZSBvZiB0aGUgY2xhaW0gdGhhdCB0aGUgU3RhdGVmdWxTZXQgY29udHJvbGxlciB3aWxsIGNyZWF0ZSBmb3IgZWFjaCBwb2QuXG4gICAqIFRoaXMgd2lsbCBiZSB1c2VkIHRvIG5hbWUgdGhlIGNyZWF0ZWQgUFZDIGluIHRoZSBmb3JtYXQgPGNsYWltLW5hbWU+LTxwb2QtbmFtZT5cbiAgICpcbiAgICogVGhpcyBuYW1lIHNob3VsZCBtYXRjaCB0aGUgbmFtZSBvZiBhIHZvbHVtZSBtb3VudCBpbiBvbmUgb2YgdGhlIGNvbnRhaW5lcnMuXG4gICAqL1xuICByZWFkb25seSBuYW1lOiBzdHJpbmc7XG59XG4iXX0=
|
package/lib/utils.d.ts
ADDED
package/lib/utils.js
ADDED
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.undefinedIfEmpty = undefinedIfEmpty;
|
|
4
|
+
exports.filterUndefined = filterUndefined;
|
|
5
|
+
exports.address = address;
|
|
6
|
+
function undefinedIfEmpty(obj) {
|
|
7
|
+
if (typeof (obj) === 'string' && obj === '') {
|
|
8
|
+
return undefined;
|
|
9
|
+
}
|
|
10
|
+
if (Array.isArray(obj) && obj.length === 0) {
|
|
11
|
+
return undefined;
|
|
12
|
+
}
|
|
13
|
+
if (typeof (obj) === 'object' && (Object.keys(obj).length === 0 || Object.values(obj).filter(x => x).length === 0)) {
|
|
14
|
+
return undefined;
|
|
15
|
+
}
|
|
16
|
+
return obj;
|
|
17
|
+
}
|
|
18
|
+
function filterUndefined(obj) {
|
|
19
|
+
const ret = {};
|
|
20
|
+
for (const [k, v] of Object.entries(obj)) {
|
|
21
|
+
if (v !== undefined) {
|
|
22
|
+
ret[k] = v;
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
return ret;
|
|
26
|
+
}
|
|
27
|
+
function address(...constructs) {
|
|
28
|
+
const addresses = constructs
|
|
29
|
+
.map(c => c.node.addr)
|
|
30
|
+
.sort((a, b) => a.localeCompare(b));
|
|
31
|
+
return addresses.join('');
|
|
32
|
+
}
|
|
33
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvdXRpbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFFQSw0Q0FLQztBQUVELDBDQVFDO0FBRUQsMEJBS0M7QUF0QkQsU0FBZ0IsZ0JBQWdCLENBQUksR0FBTTtJQUN4QyxJQUFJLE9BQU0sQ0FBQyxHQUFHLENBQUMsS0FBSyxRQUFRLElBQUksR0FBRyxLQUFLLEVBQUUsRUFBRSxDQUFDO1FBQUMsT0FBTyxTQUFTLENBQUM7SUFBQyxDQUFDO0lBQ2pFLElBQUksS0FBSyxDQUFDLE9BQU8sQ0FBQyxHQUFHLENBQUMsSUFBSSxHQUFHLENBQUMsTUFBTSxLQUFLLENBQUMsRUFBRSxDQUFDO1FBQUMsT0FBTyxTQUFTLENBQUM7SUFBQyxDQUFDO0lBQ2pFLElBQUksT0FBTSxDQUFDLEdBQUcsQ0FBQyxLQUFLLFFBQVEsSUFBSSxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsR0FBd0IsQ0FBQyxDQUFDLE1BQU0sS0FBSyxDQUFDLElBQUksTUFBTSxDQUFDLE1BQU0sQ0FBQyxHQUF3QixDQUFDLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDLENBQUMsTUFBTSxLQUFLLENBQUMsQ0FBQyxFQUFFLENBQUM7UUFBQyxPQUFPLFNBQVMsQ0FBQztJQUFDLENBQUM7SUFDbEwsT0FBTyxHQUFHLENBQUM7QUFDYixDQUFDO0FBRUQsU0FBZ0IsZUFBZSxDQUFDLEdBQVE7SUFDdEMsTUFBTSxHQUFHLEdBQVEsRUFBRSxDQUFDO0lBQ3BCLEtBQUssTUFBTSxDQUFDLENBQUMsRUFBRSxDQUFDLENBQUMsSUFBSSxNQUFNLENBQUMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUM7UUFDekMsSUFBSSxDQUFDLEtBQUssU0FBUyxFQUFFLENBQUM7WUFDcEIsR0FBRyxDQUFDLENBQUMsQ0FBQyxHQUFHLENBQUMsQ0FBQztRQUNiLENBQUM7SUFDSCxDQUFDO0lBQ0QsT0FBTyxHQUFHLENBQUM7QUFDYixDQUFDO0FBRUQsU0FBZ0IsT0FBTyxDQUFDLEdBQUcsVUFBd0I7SUFDakQsTUFBTSxTQUFTLEdBQUcsVUFBVTtTQUN6QixHQUFHLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQztTQUNyQixJQUFJLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxDQUFDLENBQUMsYUFBYSxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUM7SUFDdEMsT0FBTyxTQUFTLENBQUMsSUFBSSxDQUFDLEVBQUUsQ0FBQyxDQUFDO0FBQzVCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJQ29uc3RydWN0IH0gZnJvbSAnY29uc3RydWN0cyc7XG5cbmV4cG9ydCBmdW5jdGlvbiB1bmRlZmluZWRJZkVtcHR5PFQ+KG9iajogVCk6IFQgfCB1bmRlZmluZWQge1xuICBpZiAodHlwZW9mKG9iaikgPT09ICdzdHJpbmcnICYmIG9iaiA9PT0gJycpIHsgcmV0dXJuIHVuZGVmaW5lZDsgfVxuICBpZiAoQXJyYXkuaXNBcnJheShvYmopICYmIG9iai5sZW5ndGggPT09IDApIHsgcmV0dXJuIHVuZGVmaW5lZDsgfVxuICBpZiAodHlwZW9mKG9iaikgPT09ICdvYmplY3QnICYmIChPYmplY3Qua2V5cyhvYmogYXMgdW5rbm93biBhcyBvYmplY3QpLmxlbmd0aCA9PT0gMCB8fCBPYmplY3QudmFsdWVzKG9iaiBhcyB1bmtub3duIGFzIG9iamVjdCkuZmlsdGVyKHggPT4geCkubGVuZ3RoID09PSAwKSkgeyByZXR1cm4gdW5kZWZpbmVkOyB9XG4gIHJldHVybiBvYmo7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBmaWx0ZXJVbmRlZmluZWQob2JqOiBhbnkpOiBhbnkge1xuICBjb25zdCByZXQ6IGFueSA9IHt9O1xuICBmb3IgKGNvbnN0IFtrLCB2XSBvZiBPYmplY3QuZW50cmllcyhvYmopKSB7XG4gICAgaWYgKHYgIT09IHVuZGVmaW5lZCkge1xuICAgICAgcmV0W2tdID0gdjtcbiAgICB9XG4gIH1cbiAgcmV0dXJuIHJldDtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGFkZHJlc3MoLi4uY29uc3RydWN0czogSUNvbnN0cnVjdFtdKSB7XG4gIGNvbnN0IGFkZHJlc3NlcyA9IGNvbnN0cnVjdHNcbiAgICAubWFwKGMgPT4gYy5ub2RlLmFkZHIpXG4gICAgLnNvcnQoKGEsIGIpID0+IGEubG9jYWxlQ29tcGFyZShiKSk7XG4gIHJldHVybiBhZGRyZXNzZXMuam9pbignJyk7XG59XG4iXX0=
|