Merge pull request #44899 from smarterclayton/burst
Automatic merge from submit-queue (batch tested with PRs 38990, 45781, 46225, 44899, 43663) Support parallel scaling on StatefulSets Fixes #41255 ```release-note StatefulSets now include an alpha scaling feature accessible by setting the `spec.podManagementPolicy` field to `Parallel`. The controller will not wait for pods to be ready before adding the other pods, and will replace deleted pods as needed. Since parallel scaling creates pods out of order, you cannot depend on predictable membership changes within your set. ```
This commit is contained in:
@@ -47458,25 +47458,29 @@
|
|||||||
"serviceName"
|
"serviceName"
|
||||||
],
|
],
|
||||||
"properties": {
|
"properties": {
|
||||||
|
"podManagementPolicy": {
|
||||||
|
"description": "podManagementPolicy controls how pods are created during initial scale up, when replacing pods on nodes, or when scaling down. The default policy is `OrderedReady`, where pods are created in increasing order (pod-0, then pod-1, etc) and the controller will wait until each pod is ready before continuing. When scaling down, the pods are removed in the opposite order. The alternative policy is `Parallel` which will create pods in parallel to match the desired scale without waiting, and on scale down will delete all pods at once.",
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
"replicas": {
|
"replicas": {
|
||||||
"description": "Replicas is the desired number of replicas of the given Template. These are replicas in the sense that they are instantiations of the same Template, but individual replicas also have a consistent identity. If unspecified, defaults to 1.",
|
"description": "replicas is the desired number of replicas of the given Template. These are replicas in the sense that they are instantiations of the same Template, but individual replicas also have a consistent identity. If unspecified, defaults to 1.",
|
||||||
"type": "integer",
|
"type": "integer",
|
||||||
"format": "int32"
|
"format": "int32"
|
||||||
},
|
},
|
||||||
"selector": {
|
"selector": {
|
||||||
"description": "Selector is a label query over pods that should match the replica count. If empty, defaulted to labels on the pod template. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors",
|
"description": "selector is a label query over pods that should match the replica count. If empty, defaulted to labels on the pod template. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors",
|
||||||
"$ref": "#/definitions/io.k8s.apimachinery.pkg.apis.meta.v1.LabelSelector"
|
"$ref": "#/definitions/io.k8s.apimachinery.pkg.apis.meta.v1.LabelSelector"
|
||||||
},
|
},
|
||||||
"serviceName": {
|
"serviceName": {
|
||||||
"description": "ServiceName is the name of the service that governs this StatefulSet. This service must exist before the StatefulSet, and is responsible for the network identity of the set. Pods get DNS/hostnames that follow the pattern: pod-specific-string.serviceName.default.svc.cluster.local where \"pod-specific-string\" is managed by the StatefulSet controller.",
|
"description": "serviceName is the name of the service that governs this StatefulSet. This service must exist before the StatefulSet, and is responsible for the network identity of the set. Pods get DNS/hostnames that follow the pattern: pod-specific-string.serviceName.default.svc.cluster.local where \"pod-specific-string\" is managed by the StatefulSet controller.",
|
||||||
"type": "string"
|
"type": "string"
|
||||||
},
|
},
|
||||||
"template": {
|
"template": {
|
||||||
"description": "Template is the object that describes the pod that will be created if insufficient replicas are detected. Each pod stamped out by the StatefulSet will fulfill this Template, but have a unique identity from the rest of the StatefulSet.",
|
"description": "template is the object that describes the pod that will be created if insufficient replicas are detected. Each pod stamped out by the StatefulSet will fulfill this Template, but have a unique identity from the rest of the StatefulSet.",
|
||||||
"$ref": "#/definitions/io.k8s.kubernetes.pkg.api.v1.PodTemplateSpec"
|
"$ref": "#/definitions/io.k8s.kubernetes.pkg.api.v1.PodTemplateSpec"
|
||||||
},
|
},
|
||||||
"volumeClaimTemplates": {
|
"volumeClaimTemplates": {
|
||||||
"description": "VolumeClaimTemplates is a list of claims that pods are allowed to reference. The StatefulSet controller is responsible for mapping network identities to claims in a way that maintains the identity of a pod. Every claim in this list must have at least one matching (by name) volumeMount in one container in the template. A claim in this list takes precedence over any volumes in the template, with the same name.",
|
"description": "volumeClaimTemplates is a list of claims that pods are allowed to reference. The StatefulSet controller is responsible for mapping network identities to claims in a way that maintains the identity of a pod. Every claim in this list must have at least one matching (by name) volumeMount in one container in the template. A claim in this list takes precedence over any volumes in the template, with the same name.",
|
||||||
"type": "array",
|
"type": "array",
|
||||||
"items": {
|
"items": {
|
||||||
"$ref": "#/definitions/io.k8s.kubernetes.pkg.api.v1.PersistentVolumeClaim"
|
"$ref": "#/definitions/io.k8s.kubernetes.pkg.api.v1.PersistentVolumeClaim"
|
||||||
@@ -47491,12 +47495,12 @@
|
|||||||
],
|
],
|
||||||
"properties": {
|
"properties": {
|
||||||
"observedGeneration": {
|
"observedGeneration": {
|
||||||
"description": "most recent generation observed by this StatefulSet.",
|
"description": "observedGeneration is the most recent generation observed by this StatefulSet.",
|
||||||
"type": "integer",
|
"type": "integer",
|
||||||
"format": "int64"
|
"format": "int64"
|
||||||
},
|
},
|
||||||
"replicas": {
|
"replicas": {
|
||||||
"description": "Replicas is the number of actual replicas.",
|
"description": "replicas is the number of actual replicas.",
|
||||||
"type": "integer",
|
"type": "integer",
|
||||||
"format": "int32"
|
"format": "int32"
|
||||||
}
|
}
|
||||||
|
@@ -4955,26 +4955,30 @@
|
|||||||
"replicas": {
|
"replicas": {
|
||||||
"type": "integer",
|
"type": "integer",
|
||||||
"format": "int32",
|
"format": "int32",
|
||||||
"description": "Replicas is the desired number of replicas of the given Template. These are replicas in the sense that they are instantiations of the same Template, but individual replicas also have a consistent identity. If unspecified, defaults to 1."
|
"description": "replicas is the desired number of replicas of the given Template. These are replicas in the sense that they are instantiations of the same Template, but individual replicas also have a consistent identity. If unspecified, defaults to 1."
|
||||||
},
|
},
|
||||||
"selector": {
|
"selector": {
|
||||||
"$ref": "v1.LabelSelector",
|
"$ref": "v1.LabelSelector",
|
||||||
"description": "Selector is a label query over pods that should match the replica count. If empty, defaulted to labels on the pod template. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors"
|
"description": "selector is a label query over pods that should match the replica count. If empty, defaulted to labels on the pod template. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors"
|
||||||
},
|
},
|
||||||
"template": {
|
"template": {
|
||||||
"$ref": "v1.PodTemplateSpec",
|
"$ref": "v1.PodTemplateSpec",
|
||||||
"description": "Template is the object that describes the pod that will be created if insufficient replicas are detected. Each pod stamped out by the StatefulSet will fulfill this Template, but have a unique identity from the rest of the StatefulSet."
|
"description": "template is the object that describes the pod that will be created if insufficient replicas are detected. Each pod stamped out by the StatefulSet will fulfill this Template, but have a unique identity from the rest of the StatefulSet."
|
||||||
},
|
},
|
||||||
"volumeClaimTemplates": {
|
"volumeClaimTemplates": {
|
||||||
"type": "array",
|
"type": "array",
|
||||||
"items": {
|
"items": {
|
||||||
"$ref": "v1.PersistentVolumeClaim"
|
"$ref": "v1.PersistentVolumeClaim"
|
||||||
},
|
},
|
||||||
"description": "VolumeClaimTemplates is a list of claims that pods are allowed to reference. The StatefulSet controller is responsible for mapping network identities to claims in a way that maintains the identity of a pod. Every claim in this list must have at least one matching (by name) volumeMount in one container in the template. A claim in this list takes precedence over any volumes in the template, with the same name."
|
"description": "volumeClaimTemplates is a list of claims that pods are allowed to reference. The StatefulSet controller is responsible for mapping network identities to claims in a way that maintains the identity of a pod. Every claim in this list must have at least one matching (by name) volumeMount in one container in the template. A claim in this list takes precedence over any volumes in the template, with the same name."
|
||||||
},
|
},
|
||||||
"serviceName": {
|
"serviceName": {
|
||||||
"type": "string",
|
"type": "string",
|
||||||
"description": "ServiceName is the name of the service that governs this StatefulSet. This service must exist before the StatefulSet, and is responsible for the network identity of the set. Pods get DNS/hostnames that follow the pattern: pod-specific-string.serviceName.default.svc.cluster.local where \"pod-specific-string\" is managed by the StatefulSet controller."
|
"description": "serviceName is the name of the service that governs this StatefulSet. This service must exist before the StatefulSet, and is responsible for the network identity of the set. Pods get DNS/hostnames that follow the pattern: pod-specific-string.serviceName.default.svc.cluster.local where \"pod-specific-string\" is managed by the StatefulSet controller."
|
||||||
|
},
|
||||||
|
"podManagementPolicy": {
|
||||||
|
"type": "string",
|
||||||
|
"description": "podManagementPolicy controls how pods are created during initial scale up, when replacing pods on nodes, or when scaling down. The default policy is `OrderedReady`, where pods are created in increasing order (pod-0, then pod-1, etc) and the controller will wait until each pod is ready before continuing. When scaling down, the pods are removed in the opposite order. The alternative policy is `Parallel` which will create pods in parallel to match the desired scale without waiting, and on scale down will delete all pods at once."
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@@ -5068,12 +5072,12 @@
|
|||||||
"observedGeneration": {
|
"observedGeneration": {
|
||||||
"type": "integer",
|
"type": "integer",
|
||||||
"format": "int64",
|
"format": "int64",
|
||||||
"description": "most recent generation observed by this StatefulSet."
|
"description": "observedGeneration is the most recent generation observed by this StatefulSet."
|
||||||
},
|
},
|
||||||
"replicas": {
|
"replicas": {
|
||||||
"type": "integer",
|
"type": "integer",
|
||||||
"format": "int32",
|
"format": "int32",
|
||||||
"description": "Replicas is the number of actual replicas."
|
"description": "replicas is the number of actual replicas."
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
@@ -1511,39 +1511,46 @@ span.icon > [class^="icon-"], span.icon > [class*=" icon-"] { cursor: default; }
|
|||||||
<tbody>
|
<tbody>
|
||||||
<tr>
|
<tr>
|
||||||
<td class="tableblock halign-left valign-top"><p class="tableblock">replicas</p></td>
|
<td class="tableblock halign-left valign-top"><p class="tableblock">replicas</p></td>
|
||||||
<td class="tableblock halign-left valign-top"><p class="tableblock">Replicas is the desired number of replicas of the given Template. These are replicas in the sense that they are instantiations of the same Template, but individual replicas also have a consistent identity. If unspecified, defaults to 1.</p></td>
|
<td class="tableblock halign-left valign-top"><p class="tableblock">replicas is the desired number of replicas of the given Template. These are replicas in the sense that they are instantiations of the same Template, but individual replicas also have a consistent identity. If unspecified, defaults to 1.</p></td>
|
||||||
<td class="tableblock halign-left valign-top"><p class="tableblock">false</p></td>
|
<td class="tableblock halign-left valign-top"><p class="tableblock">false</p></td>
|
||||||
<td class="tableblock halign-left valign-top"><p class="tableblock">integer (int32)</p></td>
|
<td class="tableblock halign-left valign-top"><p class="tableblock">integer (int32)</p></td>
|
||||||
<td class="tableblock halign-left valign-top"></td>
|
<td class="tableblock halign-left valign-top"></td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
<td class="tableblock halign-left valign-top"><p class="tableblock">selector</p></td>
|
<td class="tableblock halign-left valign-top"><p class="tableblock">selector</p></td>
|
||||||
<td class="tableblock halign-left valign-top"><p class="tableblock">Selector is a label query over pods that should match the replica count. If empty, defaulted to labels on the pod template. More info: <a href="https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors">https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors</a></p></td>
|
<td class="tableblock halign-left valign-top"><p class="tableblock">selector is a label query over pods that should match the replica count. If empty, defaulted to labels on the pod template. More info: <a href="https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors">https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors</a></p></td>
|
||||||
<td class="tableblock halign-left valign-top"><p class="tableblock">false</p></td>
|
<td class="tableblock halign-left valign-top"><p class="tableblock">false</p></td>
|
||||||
<td class="tableblock halign-left valign-top"><p class="tableblock"><a href="#_v1_labelselector">v1.LabelSelector</a></p></td>
|
<td class="tableblock halign-left valign-top"><p class="tableblock"><a href="#_v1_labelselector">v1.LabelSelector</a></p></td>
|
||||||
<td class="tableblock halign-left valign-top"></td>
|
<td class="tableblock halign-left valign-top"></td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
<td class="tableblock halign-left valign-top"><p class="tableblock">template</p></td>
|
<td class="tableblock halign-left valign-top"><p class="tableblock">template</p></td>
|
||||||
<td class="tableblock halign-left valign-top"><p class="tableblock">Template is the object that describes the pod that will be created if insufficient replicas are detected. Each pod stamped out by the StatefulSet will fulfill this Template, but have a unique identity from the rest of the StatefulSet.</p></td>
|
<td class="tableblock halign-left valign-top"><p class="tableblock">template is the object that describes the pod that will be created if insufficient replicas are detected. Each pod stamped out by the StatefulSet will fulfill this Template, but have a unique identity from the rest of the StatefulSet.</p></td>
|
||||||
<td class="tableblock halign-left valign-top"><p class="tableblock">true</p></td>
|
<td class="tableblock halign-left valign-top"><p class="tableblock">true</p></td>
|
||||||
<td class="tableblock halign-left valign-top"><p class="tableblock"><a href="#_v1_podtemplatespec">v1.PodTemplateSpec</a></p></td>
|
<td class="tableblock halign-left valign-top"><p class="tableblock"><a href="#_v1_podtemplatespec">v1.PodTemplateSpec</a></p></td>
|
||||||
<td class="tableblock halign-left valign-top"></td>
|
<td class="tableblock halign-left valign-top"></td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
<td class="tableblock halign-left valign-top"><p class="tableblock">volumeClaimTemplates</p></td>
|
<td class="tableblock halign-left valign-top"><p class="tableblock">volumeClaimTemplates</p></td>
|
||||||
<td class="tableblock halign-left valign-top"><p class="tableblock">VolumeClaimTemplates is a list of claims that pods are allowed to reference. The StatefulSet controller is responsible for mapping network identities to claims in a way that maintains the identity of a pod. Every claim in this list must have at least one matching (by name) volumeMount in one container in the template. A claim in this list takes precedence over any volumes in the template, with the same name.</p></td>
|
<td class="tableblock halign-left valign-top"><p class="tableblock">volumeClaimTemplates is a list of claims that pods are allowed to reference. The StatefulSet controller is responsible for mapping network identities to claims in a way that maintains the identity of a pod. Every claim in this list must have at least one matching (by name) volumeMount in one container in the template. A claim in this list takes precedence over any volumes in the template, with the same name.</p></td>
|
||||||
<td class="tableblock halign-left valign-top"><p class="tableblock">false</p></td>
|
<td class="tableblock halign-left valign-top"><p class="tableblock">false</p></td>
|
||||||
<td class="tableblock halign-left valign-top"><p class="tableblock"><a href="#_v1_persistentvolumeclaim">v1.PersistentVolumeClaim</a> array</p></td>
|
<td class="tableblock halign-left valign-top"><p class="tableblock"><a href="#_v1_persistentvolumeclaim">v1.PersistentVolumeClaim</a> array</p></td>
|
||||||
<td class="tableblock halign-left valign-top"></td>
|
<td class="tableblock halign-left valign-top"></td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
<td class="tableblock halign-left valign-top"><p class="tableblock">serviceName</p></td>
|
<td class="tableblock halign-left valign-top"><p class="tableblock">serviceName</p></td>
|
||||||
<td class="tableblock halign-left valign-top"><p class="tableblock">ServiceName is the name of the service that governs this StatefulSet. This service must exist before the StatefulSet, and is responsible for the network identity of the set. Pods get DNS/hostnames that follow the pattern: pod-specific-string.serviceName.default.svc.cluster.local where "pod-specific-string" is managed by the StatefulSet controller.</p></td>
|
<td class="tableblock halign-left valign-top"><p class="tableblock">serviceName is the name of the service that governs this StatefulSet. This service must exist before the StatefulSet, and is responsible for the network identity of the set. Pods get DNS/hostnames that follow the pattern: pod-specific-string.serviceName.default.svc.cluster.local where "pod-specific-string" is managed by the StatefulSet controller.</p></td>
|
||||||
<td class="tableblock halign-left valign-top"><p class="tableblock">true</p></td>
|
<td class="tableblock halign-left valign-top"><p class="tableblock">true</p></td>
|
||||||
<td class="tableblock halign-left valign-top"><p class="tableblock">string</p></td>
|
<td class="tableblock halign-left valign-top"><p class="tableblock">string</p></td>
|
||||||
<td class="tableblock halign-left valign-top"></td>
|
<td class="tableblock halign-left valign-top"></td>
|
||||||
</tr>
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td class="tableblock halign-left valign-top"><p class="tableblock">podManagementPolicy</p></td>
|
||||||
|
<td class="tableblock halign-left valign-top"><p class="tableblock">podManagementPolicy controls how pods are created during initial scale up, when replacing pods on nodes, or when scaling down. The default policy is <code>OrderedReady</code>, where pods are created in increasing order (pod-0, then pod-1, etc) and the controller will wait until each pod is ready before continuing. When scaling down, the pods are removed in the opposite order. The alternative policy is <code>Parallel</code> which will create pods in parallel to match the desired scale without waiting, and on scale down will delete all pods at once.</p></td>
|
||||||
|
<td class="tableblock halign-left valign-top"><p class="tableblock">false</p></td>
|
||||||
|
<td class="tableblock halign-left valign-top"><p class="tableblock">string</p></td>
|
||||||
|
<td class="tableblock halign-left valign-top"></td>
|
||||||
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
|
|
||||||
@@ -4914,14 +4921,14 @@ Examples:<br>
|
|||||||
<tbody>
|
<tbody>
|
||||||
<tr>
|
<tr>
|
||||||
<td class="tableblock halign-left valign-top"><p class="tableblock">observedGeneration</p></td>
|
<td class="tableblock halign-left valign-top"><p class="tableblock">observedGeneration</p></td>
|
||||||
<td class="tableblock halign-left valign-top"><p class="tableblock">most recent generation observed by this StatefulSet.</p></td>
|
<td class="tableblock halign-left valign-top"><p class="tableblock">observedGeneration is the most recent generation observed by this StatefulSet.</p></td>
|
||||||
<td class="tableblock halign-left valign-top"><p class="tableblock">false</p></td>
|
<td class="tableblock halign-left valign-top"><p class="tableblock">false</p></td>
|
||||||
<td class="tableblock halign-left valign-top"><p class="tableblock">integer (int64)</p></td>
|
<td class="tableblock halign-left valign-top"><p class="tableblock">integer (int64)</p></td>
|
||||||
<td class="tableblock halign-left valign-top"></td>
|
<td class="tableblock halign-left valign-top"></td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
<td class="tableblock halign-left valign-top"><p class="tableblock">replicas</p></td>
|
<td class="tableblock halign-left valign-top"><p class="tableblock">replicas</p></td>
|
||||||
<td class="tableblock halign-left valign-top"><p class="tableblock">Replicas is the number of actual replicas.</p></td>
|
<td class="tableblock halign-left valign-top"><p class="tableblock">replicas is the number of actual replicas.</p></td>
|
||||||
<td class="tableblock halign-left valign-top"><p class="tableblock">true</p></td>
|
<td class="tableblock halign-left valign-top"><p class="tableblock">true</p></td>
|
||||||
<td class="tableblock halign-left valign-top"><p class="tableblock">integer (int32)</p></td>
|
<td class="tableblock halign-left valign-top"><p class="tableblock">integer (int32)</p></td>
|
||||||
<td class="tableblock halign-left valign-top"></td>
|
<td class="tableblock halign-left valign-top"></td>
|
||||||
@@ -6449,7 +6456,7 @@ Examples:<br>
|
|||||||
</div>
|
</div>
|
||||||
<div id="footer">
|
<div id="footer">
|
||||||
<div id="footer-text">
|
<div id="footer-text">
|
||||||
Last updated 2017-05-18 19:28:33 UTC
|
Last updated 2017-05-19 16:32:56 UTC
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</body>
|
</body>
|
||||||
|
@@ -20,6 +20,7 @@ go_library(
|
|||||||
"//pkg/api:go_default_library",
|
"//pkg/api:go_default_library",
|
||||||
"//pkg/api/testapi:go_default_library",
|
"//pkg/api/testapi:go_default_library",
|
||||||
"//pkg/api/v1:go_default_library",
|
"//pkg/api/v1:go_default_library",
|
||||||
|
"//pkg/apis/apps:go_default_library",
|
||||||
"//pkg/apis/autoscaling:go_default_library",
|
"//pkg/apis/autoscaling:go_default_library",
|
||||||
"//pkg/apis/batch:go_default_library",
|
"//pkg/apis/batch:go_default_library",
|
||||||
"//pkg/apis/certificates:go_default_library",
|
"//pkg/apis/certificates:go_default_library",
|
||||||
|
@@ -36,6 +36,7 @@ import (
|
|||||||
"k8s.io/kubernetes/pkg/api"
|
"k8s.io/kubernetes/pkg/api"
|
||||||
"k8s.io/kubernetes/pkg/api/testapi"
|
"k8s.io/kubernetes/pkg/api/testapi"
|
||||||
"k8s.io/kubernetes/pkg/api/v1"
|
"k8s.io/kubernetes/pkg/api/v1"
|
||||||
|
"k8s.io/kubernetes/pkg/apis/apps"
|
||||||
"k8s.io/kubernetes/pkg/apis/autoscaling"
|
"k8s.io/kubernetes/pkg/apis/autoscaling"
|
||||||
"k8s.io/kubernetes/pkg/apis/batch"
|
"k8s.io/kubernetes/pkg/apis/batch"
|
||||||
"k8s.io/kubernetes/pkg/apis/certificates"
|
"k8s.io/kubernetes/pkg/apis/certificates"
|
||||||
@@ -707,6 +708,18 @@ func rbacFuncs(t apitesting.TestingCommon) []interface{} {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func appsFuncs(t apitesting.TestingCommon) []interface{} {
|
||||||
|
return []interface{}{
|
||||||
|
func(s *apps.StatefulSet, c fuzz.Continue) {
|
||||||
|
c.FuzzNoCustom(s) // fuzz self without calling this function again
|
||||||
|
|
||||||
|
// match defaulter
|
||||||
|
if len(s.Spec.PodManagementPolicy) == 0 {
|
||||||
|
s.Spec.PodManagementPolicy = apps.OrderedReadyPodManagement
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
func policyFuncs(t apitesting.TestingCommon) []interface{} {
|
func policyFuncs(t apitesting.TestingCommon) []interface{} {
|
||||||
return []interface{}{
|
return []interface{}{
|
||||||
func(s *policy.PodDisruptionBudgetStatus, c fuzz.Continue) {
|
func(s *policy.PodDisruptionBudgetStatus, c fuzz.Continue) {
|
||||||
@@ -731,6 +744,7 @@ func FuzzerFuncs(t apitesting.TestingCommon, codecs runtimeserializer.CodecFacto
|
|||||||
overrideGenericFuncs(t, codecs),
|
overrideGenericFuncs(t, codecs),
|
||||||
coreFuncs(t),
|
coreFuncs(t),
|
||||||
extensionFuncs(t),
|
extensionFuncs(t),
|
||||||
|
appsFuncs(t),
|
||||||
batchFuncs(t),
|
batchFuncs(t),
|
||||||
autoscalingFuncs(t),
|
autoscalingFuncs(t),
|
||||||
rbacFuncs(t),
|
rbacFuncs(t),
|
||||||
|
@@ -44,6 +44,21 @@ type StatefulSet struct {
|
|||||||
Status StatefulSetStatus
|
Status StatefulSetStatus
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// PodManagementPolicyType defines the policy for creating pods under a stateful set.
|
||||||
|
type PodManagementPolicyType string
|
||||||
|
|
||||||
|
const (
|
||||||
|
// OrderedReadyPodManagement will create pods in strictly increasing order on
|
||||||
|
// scale up and strictly decreasing order on scale down, progressing only when
|
||||||
|
// the previous pod is ready or terminated. At most one pod will be changed
|
||||||
|
// at any time.
|
||||||
|
OrderedReadyPodManagement PodManagementPolicyType = "OrderedReady"
|
||||||
|
// ParallelPodManagement will create and delete pods as soon as the stateful set
|
||||||
|
// replica count is changed, and will not wait for pods to be ready or complete
|
||||||
|
// termination.
|
||||||
|
ParallelPodManagement = "Parallel"
|
||||||
|
)
|
||||||
|
|
||||||
// A StatefulSetSpec is the specification of a StatefulSet.
|
// A StatefulSetSpec is the specification of a StatefulSet.
|
||||||
type StatefulSetSpec struct {
|
type StatefulSetSpec struct {
|
||||||
// Replicas is the desired number of replicas of the given Template.
|
// Replicas is the desired number of replicas of the given Template.
|
||||||
@@ -82,6 +97,17 @@ type StatefulSetSpec struct {
|
|||||||
// pattern: pod-specific-string.serviceName.default.svc.cluster.local
|
// pattern: pod-specific-string.serviceName.default.svc.cluster.local
|
||||||
// where "pod-specific-string" is managed by the StatefulSet controller.
|
// where "pod-specific-string" is managed by the StatefulSet controller.
|
||||||
ServiceName string
|
ServiceName string
|
||||||
|
|
||||||
|
// PodManagementPolicy controls how pods are created during initial scale up,
|
||||||
|
// when replacing pods on nodes, or when scaling down. The default policy is
|
||||||
|
// `OrderedReady`, where pods are created in increasing order (pod-0, then
|
||||||
|
// pod-1, etc) and the controller will wait until each pod is ready before
|
||||||
|
// continuing. When scaling down, the pods are removed in the opposite order.
|
||||||
|
// The alternative policy is `Parallel` which will create pods in parallel
|
||||||
|
// to match the desired scale without waiting, and on scale down will delete
|
||||||
|
// all pods at once.
|
||||||
|
// +optional
|
||||||
|
PodManagementPolicy PodManagementPolicyType
|
||||||
}
|
}
|
||||||
|
|
||||||
// StatefulSetStatus represents the current state of a StatefulSet.
|
// StatefulSetStatus represents the current state of a StatefulSet.
|
||||||
|
@@ -110,6 +110,7 @@ func Convert_v1beta1_StatefulSetSpec_To_apps_StatefulSetSpec(in *StatefulSetSpec
|
|||||||
out.VolumeClaimTemplates = nil
|
out.VolumeClaimTemplates = nil
|
||||||
}
|
}
|
||||||
out.ServiceName = in.ServiceName
|
out.ServiceName = in.ServiceName
|
||||||
|
out.PodManagementPolicy = apps.PodManagementPolicyType(in.PodManagementPolicy)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -140,6 +141,7 @@ func Convert_apps_StatefulSetSpec_To_v1beta1_StatefulSetSpec(in *apps.StatefulSe
|
|||||||
out.VolumeClaimTemplates = nil
|
out.VolumeClaimTemplates = nil
|
||||||
}
|
}
|
||||||
out.ServiceName = in.ServiceName
|
out.ServiceName = in.ServiceName
|
||||||
|
out.PodManagementPolicy = PodManagementPolicyType(in.PodManagementPolicy)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -27,6 +27,9 @@ func addDefaultingFuncs(scheme *runtime.Scheme) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func SetDefaults_StatefulSet(obj *StatefulSet) {
|
func SetDefaults_StatefulSet(obj *StatefulSet) {
|
||||||
|
if len(obj.Spec.PodManagementPolicy) == 0 {
|
||||||
|
obj.Spec.PodManagementPolicy = OrderedReadyPodManagement
|
||||||
|
}
|
||||||
labels := obj.Spec.Template.Labels
|
labels := obj.Spec.Template.Labels
|
||||||
if labels != nil {
|
if labels != nil {
|
||||||
if obj.Spec.Selector == nil {
|
if obj.Spec.Selector == nil {
|
||||||
|
@@ -788,6 +788,10 @@ func (m *StatefulSetSpec) MarshalTo(dAtA []byte) (int, error) {
|
|||||||
i++
|
i++
|
||||||
i = encodeVarintGenerated(dAtA, i, uint64(len(m.ServiceName)))
|
i = encodeVarintGenerated(dAtA, i, uint64(len(m.ServiceName)))
|
||||||
i += copy(dAtA[i:], m.ServiceName)
|
i += copy(dAtA[i:], m.ServiceName)
|
||||||
|
dAtA[i] = 0x32
|
||||||
|
i++
|
||||||
|
i = encodeVarintGenerated(dAtA, i, uint64(len(m.PodManagementPolicy)))
|
||||||
|
i += copy(dAtA[i:], m.PodManagementPolicy)
|
||||||
return i, nil
|
return i, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1068,6 +1072,8 @@ func (m *StatefulSetSpec) Size() (n int) {
|
|||||||
}
|
}
|
||||||
l = len(m.ServiceName)
|
l = len(m.ServiceName)
|
||||||
n += 1 + l + sovGenerated(uint64(l))
|
n += 1 + l + sovGenerated(uint64(l))
|
||||||
|
l = len(m.PodManagementPolicy)
|
||||||
|
n += 1 + l + sovGenerated(uint64(l))
|
||||||
return n
|
return n
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1297,6 +1303,7 @@ func (this *StatefulSetSpec) String() string {
|
|||||||
`Template:` + strings.Replace(strings.Replace(this.Template.String(), "PodTemplateSpec", "k8s_io_kubernetes_pkg_api_v1.PodTemplateSpec", 1), `&`, ``, 1) + `,`,
|
`Template:` + strings.Replace(strings.Replace(this.Template.String(), "PodTemplateSpec", "k8s_io_kubernetes_pkg_api_v1.PodTemplateSpec", 1), `&`, ``, 1) + `,`,
|
||||||
`VolumeClaimTemplates:` + strings.Replace(strings.Replace(fmt.Sprintf("%v", this.VolumeClaimTemplates), "PersistentVolumeClaim", "k8s_io_kubernetes_pkg_api_v1.PersistentVolumeClaim", 1), `&`, ``, 1) + `,`,
|
`VolumeClaimTemplates:` + strings.Replace(strings.Replace(fmt.Sprintf("%v", this.VolumeClaimTemplates), "PersistentVolumeClaim", "k8s_io_kubernetes_pkg_api_v1.PersistentVolumeClaim", 1), `&`, ``, 1) + `,`,
|
||||||
`ServiceName:` + fmt.Sprintf("%v", this.ServiceName) + `,`,
|
`ServiceName:` + fmt.Sprintf("%v", this.ServiceName) + `,`,
|
||||||
|
`PodManagementPolicy:` + fmt.Sprintf("%v", this.PodManagementPolicy) + `,`,
|
||||||
`}`,
|
`}`,
|
||||||
}, "")
|
}, "")
|
||||||
return s
|
return s
|
||||||
@@ -3635,6 +3642,35 @@ func (m *StatefulSetSpec) Unmarshal(dAtA []byte) error {
|
|||||||
}
|
}
|
||||||
m.ServiceName = string(dAtA[iNdEx:postIndex])
|
m.ServiceName = string(dAtA[iNdEx:postIndex])
|
||||||
iNdEx = postIndex
|
iNdEx = postIndex
|
||||||
|
case 6:
|
||||||
|
if wireType != 2 {
|
||||||
|
return fmt.Errorf("proto: wrong wireType = %d for field PodManagementPolicy", wireType)
|
||||||
|
}
|
||||||
|
var stringLen uint64
|
||||||
|
for shift := uint(0); ; shift += 7 {
|
||||||
|
if shift >= 64 {
|
||||||
|
return ErrIntOverflowGenerated
|
||||||
|
}
|
||||||
|
if iNdEx >= l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
b := dAtA[iNdEx]
|
||||||
|
iNdEx++
|
||||||
|
stringLen |= (uint64(b) & 0x7F) << shift
|
||||||
|
if b < 0x80 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
intStringLen := int(stringLen)
|
||||||
|
if intStringLen < 0 {
|
||||||
|
return ErrInvalidLengthGenerated
|
||||||
|
}
|
||||||
|
postIndex := iNdEx + intStringLen
|
||||||
|
if postIndex > l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
m.PodManagementPolicy = PodManagementPolicyType(dAtA[iNdEx:postIndex])
|
||||||
|
iNdEx = postIndex
|
||||||
default:
|
default:
|
||||||
iNdEx = preIndex
|
iNdEx = preIndex
|
||||||
skippy, err := skipGenerated(dAtA[iNdEx:])
|
skippy, err := skipGenerated(dAtA[iNdEx:])
|
||||||
@@ -3855,101 +3891,103 @@ func init() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
var fileDescriptorGenerated = []byte{
|
var fileDescriptorGenerated = []byte{
|
||||||
// 1525 bytes of a gzipped FileDescriptorProto
|
// 1563 bytes of a gzipped FileDescriptorProto
|
||||||
0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0xe4, 0x58, 0xcb, 0x6f, 0x5b, 0xc5,
|
0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0xe4, 0x58, 0xcb, 0x6f, 0x1b, 0x37,
|
||||||
0x17, 0xce, 0x4d, 0xec, 0xc4, 0x99, 0x34, 0x4e, 0x33, 0xc9, 0xaf, 0xf1, 0x2f, 0x45, 0x4e, 0xe5,
|
0x13, 0xf7, 0xda, 0x92, 0x2d, 0xd3, 0xb1, 0x1c, 0xd3, 0xfe, 0x62, 0x7d, 0xce, 0x07, 0x39, 0xd0,
|
||||||
0x45, 0x1f, 0xa8, 0xbd, 0xa6, 0x69, 0xa1, 0x8f, 0x40, 0x45, 0xdc, 0x96, 0x52, 0x94, 0xd0, 0x6a,
|
0x21, 0x8f, 0x0f, 0xc9, 0xaa, 0x71, 0xd2, 0xbc, 0xdc, 0x06, 0xb5, 0x92, 0x34, 0x4d, 0x61, 0x37,
|
||||||
0xec, 0x54, 0xb4, 0x14, 0x89, 0xb1, 0x3d, 0xbd, 0x9d, 0xfa, 0xbe, 0x74, 0x67, 0x6c, 0xc5, 0x3b,
|
0x06, 0x65, 0x07, 0x4d, 0x9a, 0x02, 0xa5, 0x24, 0x66, 0xcd, 0x78, 0x5f, 0x58, 0x52, 0x82, 0x75,
|
||||||
0x36, 0x2c, 0x90, 0x58, 0xb0, 0x62, 0x87, 0xd8, 0x23, 0x24, 0x76, 0xfc, 0x0d, 0x11, 0x6c, 0xba,
|
0xeb, 0xa5, 0x87, 0x02, 0x3d, 0xf4, 0xd4, 0x5b, 0xd1, 0x9e, 0x8b, 0x02, 0xfd, 0x37, 0x8c, 0xf6,
|
||||||
0x44, 0x2c, 0x22, 0xe2, 0xfe, 0x17, 0x5d, 0xa1, 0x99, 0x3b, 0xf7, 0xe5, 0x6b, 0x27, 0x8e, 0x11,
|
0x92, 0x63, 0xd1, 0x83, 0x51, 0x3b, 0xff, 0x45, 0x4e, 0x05, 0xb9, 0xdc, 0x97, 0x76, 0x65, 0xcb,
|
||||||
0xdd, 0xb0, 0xf3, 0x9d, 0x39, 0xdf, 0x77, 0xce, 0xcc, 0x7c, 0xe7, 0xcc, 0x19, 0x83, 0x6b, 0xad,
|
0x2a, 0x9a, 0x4b, 0x6f, 0x5a, 0xce, 0xfc, 0x7e, 0x33, 0x24, 0x67, 0x86, 0x33, 0x02, 0x37, 0x77,
|
||||||
0xeb, 0x4c, 0xa7, 0x4e, 0xb9, 0xd5, 0xae, 0x13, 0xcf, 0x26, 0x9c, 0xb0, 0xb2, 0xdb, 0x32, 0xca,
|
0x6e, 0x31, 0x9d, 0x3a, 0xd5, 0x9d, 0x76, 0x83, 0x78, 0x36, 0xe1, 0x84, 0x55, 0xdd, 0x1d, 0xa3,
|
||||||
0xd8, 0xa5, 0xac, 0x8c, 0x5d, 0x97, 0x95, 0x3b, 0x97, 0xeb, 0x84, 0xe3, 0xcb, 0x65, 0x83, 0xd8,
|
0x8a, 0x5d, 0xca, 0xaa, 0xd8, 0x75, 0x59, 0xb5, 0x73, 0xb5, 0x41, 0x38, 0xbe, 0x5a, 0x35, 0x88,
|
||||||
0xc4, 0xc3, 0x9c, 0x34, 0x75, 0xd7, 0x73, 0xb8, 0x03, 0xcf, 0xf9, 0x40, 0x3d, 0x02, 0xea, 0x6e,
|
0x4d, 0x3c, 0xcc, 0x49, 0x4b, 0x77, 0x3d, 0x87, 0x3b, 0xf0, 0x82, 0x0f, 0xd4, 0x23, 0xa0, 0xee,
|
||||||
0xcb, 0xd0, 0x05, 0x50, 0x17, 0x40, 0x5d, 0x01, 0x57, 0x2f, 0x19, 0x94, 0x3f, 0x6f, 0xd7, 0xf5,
|
0xee, 0x18, 0xba, 0x00, 0xea, 0x02, 0xa8, 0x2b, 0xe0, 0xe2, 0x15, 0x83, 0xf2, 0xed, 0x76, 0x43,
|
||||||
0x86, 0x63, 0x95, 0x0d, 0xc7, 0x70, 0xca, 0x12, 0x5f, 0x6f, 0x3f, 0x93, 0x5f, 0xf2, 0x43, 0xfe,
|
0x6f, 0x3a, 0x56, 0xd5, 0x70, 0x0c, 0xa7, 0x2a, 0xf1, 0x8d, 0xf6, 0x0b, 0xf9, 0x25, 0x3f, 0xe4,
|
||||||
0xf2, 0x79, 0x57, 0xaf, 0xaa, 0x80, 0xb0, 0x4b, 0x2d, 0xdc, 0x78, 0x4e, 0x6d, 0xe2, 0x75, 0xa3,
|
0x2f, 0x9f, 0x77, 0xf1, 0xba, 0x72, 0x08, 0xbb, 0xd4, 0xc2, 0xcd, 0x6d, 0x6a, 0x13, 0xaf, 0x1b,
|
||||||
0x90, 0x2c, 0xc2, 0x71, 0xb9, 0x93, 0x8a, 0x66, 0xb5, 0x3c, 0x0c, 0xe5, 0xb5, 0x6d, 0x4e, 0x2d,
|
0xb9, 0x64, 0x11, 0x8e, 0xab, 0x9d, 0x94, 0x37, 0x8b, 0xd5, 0x7e, 0x28, 0xaf, 0x6d, 0x73, 0x6a,
|
||||||
0x92, 0x02, 0xbc, 0x77, 0x14, 0x80, 0x35, 0x9e, 0x13, 0x0b, 0xa7, 0x70, 0x57, 0x86, 0xe1, 0xda,
|
0x91, 0x14, 0xe0, 0xc6, 0x71, 0x00, 0xd6, 0xdc, 0x26, 0x16, 0x4e, 0xe1, 0xae, 0xf5, 0xc3, 0xb5,
|
||||||
0x9c, 0x9a, 0x65, 0x6a, 0x73, 0xc6, 0xbd, 0x14, 0x28, 0xb6, 0x26, 0x46, 0xbc, 0x0e, 0xf1, 0xa2,
|
0x39, 0x35, 0xab, 0xd4, 0xe6, 0x8c, 0x7b, 0x29, 0x50, 0x6c, 0x4f, 0x8c, 0x78, 0x1d, 0xe2, 0x45,
|
||||||
0x05, 0x91, 0x5d, 0x6c, 0xb9, 0x26, 0x19, 0xb4, 0xa6, 0x8b, 0x43, 0x8f, 0x66, 0x90, 0xf5, 0x07,
|
0x1b, 0x22, 0xbb, 0xd8, 0x72, 0x4d, 0x92, 0xb5, 0xa7, 0xcb, 0x7d, 0xaf, 0x26, 0x4b, 0xfb, 0xfd,
|
||||||
0x87, 0x1c, 0x24, 0xd9, 0xe5, 0xc4, 0x66, 0xd4, 0xb1, 0x87, 0x1e, 0x67, 0xe9, 0xe7, 0x49, 0x00,
|
0x23, 0x2e, 0x92, 0xec, 0x72, 0x62, 0x33, 0xea, 0xd8, 0x7d, 0xaf, 0xb3, 0xf2, 0xf3, 0x28, 0x00,
|
||||||
0xee, 0x10, 0xd7, 0x74, 0xba, 0x16, 0xb1, 0x39, 0xfc, 0x12, 0xe4, 0xc4, 0x56, 0x37, 0x31, 0xc7,
|
0xf7, 0x89, 0x6b, 0x3a, 0x5d, 0x8b, 0xd8, 0x1c, 0x7e, 0x01, 0x0a, 0xe2, 0xa8, 0x5b, 0x98, 0xe3,
|
||||||
0x05, 0xed, 0x8c, 0x76, 0x7e, 0x6e, 0xfd, 0x1d, 0x5d, 0x1d, 0x78, 0x7c, 0xe5, 0xd1, 0x91, 0x0b,
|
0x92, 0x76, 0x4e, 0xbb, 0x38, 0xb5, 0xfc, 0x8e, 0xae, 0x2e, 0x3c, 0xbe, 0xf3, 0xe8, 0xca, 0x85,
|
||||||
0x6b, 0xbd, 0x73, 0x59, 0x7f, 0x50, 0x7f, 0x41, 0x1a, 0x7c, 0x9b, 0x70, 0x5c, 0x81, 0x7b, 0xfb,
|
0xb6, 0xde, 0xb9, 0xaa, 0x3f, 0x6e, 0xbc, 0x24, 0x4d, 0xbe, 0x4e, 0x38, 0xae, 0xc1, 0xbd, 0xfd,
|
||||||
0x6b, 0x13, 0xbd, 0xfd, 0x35, 0x10, 0x8d, 0xa1, 0x90, 0x15, 0x3e, 0x06, 0x19, 0xe6, 0x92, 0x46,
|
0xa5, 0x91, 0xc3, 0xfd, 0x25, 0x10, 0xad, 0xa1, 0x90, 0x15, 0x3e, 0x05, 0x39, 0xe6, 0x92, 0x66,
|
||||||
0x61, 0x52, 0xb2, 0x5f, 0xd3, 0x47, 0x94, 0x93, 0x1e, 0x05, 0x59, 0x75, 0x49, 0xa3, 0x72, 0x42,
|
0x69, 0x54, 0xb2, 0xdf, 0xd4, 0x07, 0x0c, 0x27, 0x3d, 0x72, 0xb2, 0xee, 0x92, 0x66, 0xed, 0x94,
|
||||||
0x39, 0xc9, 0x88, 0x2f, 0x24, 0x29, 0x21, 0x06, 0xd3, 0x8c, 0x63, 0xde, 0x66, 0x85, 0x29, 0x49,
|
0x32, 0x92, 0x13, 0x5f, 0x48, 0x52, 0x42, 0x0c, 0xc6, 0x19, 0xc7, 0xbc, 0xcd, 0x4a, 0x63, 0x92,
|
||||||
0x7e, 0x63, 0x1c, 0x72, 0x49, 0x50, 0xc9, 0x2b, 0xfa, 0x69, 0xff, 0x1b, 0x29, 0xe2, 0xd2, 0xc1,
|
0xfc, 0xf6, 0x30, 0xe4, 0x92, 0xa0, 0x56, 0x54, 0xf4, 0xe3, 0xfe, 0x37, 0x52, 0xc4, 0x95, 0x83,
|
||||||
0x14, 0x58, 0x8a, 0x8c, 0x6f, 0x3b, 0x76, 0x93, 0x72, 0xea, 0xd8, 0x70, 0x03, 0x64, 0x78, 0xd7,
|
0x31, 0x30, 0x17, 0x29, 0xdf, 0x73, 0xec, 0x16, 0xe5, 0xd4, 0xb1, 0xe1, 0x0a, 0xc8, 0xf1, 0xae,
|
||||||
0x25, 0x72, 0xcf, 0x66, 0x2b, 0xe7, 0x82, 0xe0, 0x6a, 0x5d, 0x97, 0xbc, 0xde, 0x5f, 0x5b, 0x19,
|
0x4b, 0xe4, 0x99, 0x4d, 0xd6, 0x2e, 0x04, 0xce, 0x6d, 0x76, 0x5d, 0xf2, 0x66, 0x7f, 0x69, 0x21,
|
||||||
0x00, 0x11, 0x53, 0x48, 0x82, 0xe0, 0xa3, 0x30, 0xee, 0x49, 0x09, 0xbf, 0x95, 0x74, 0xfe, 0x7a,
|
0x03, 0x22, 0x44, 0x48, 0x82, 0xe0, 0x93, 0xd0, 0xef, 0x51, 0x09, 0xbf, 0x9b, 0x34, 0xfe, 0x66,
|
||||||
0x7f, 0xed, 0x50, 0x49, 0xe8, 0x21, 0x67, 0x32, 0x58, 0x78, 0x16, 0x4c, 0x7b, 0x04, 0x33, 0xc7,
|
0x7f, 0xe9, 0xc8, 0x90, 0xd0, 0x43, 0xce, 0xa4, 0xb3, 0xf0, 0x3c, 0x18, 0xf7, 0x08, 0x66, 0x8e,
|
||||||
0x2e, 0x64, 0x24, 0x6f, 0xb8, 0x28, 0x24, 0x47, 0x91, 0x9a, 0x85, 0x17, 0xc0, 0x8c, 0x45, 0x18,
|
0x5d, 0xca, 0x49, 0xde, 0x70, 0x53, 0x48, 0xae, 0x22, 0x25, 0x85, 0x97, 0xc0, 0x84, 0x45, 0x18,
|
||||||
0xc3, 0x06, 0x29, 0x64, 0xa5, 0xe1, 0x82, 0x32, 0x9c, 0xd9, 0xf6, 0x87, 0x51, 0x30, 0x0f, 0x5f,
|
0xc3, 0x06, 0x29, 0xe5, 0xa5, 0xe2, 0x8c, 0x52, 0x9c, 0x58, 0xf7, 0x97, 0x51, 0x20, 0x87, 0x2f,
|
||||||
0x80, 0xbc, 0x89, 0x19, 0xdf, 0x71, 0x9b, 0x98, 0x93, 0x1a, 0xb5, 0x48, 0x61, 0x5a, 0x6e, 0xf5,
|
0x41, 0xd1, 0xc4, 0x8c, 0x6f, 0xb9, 0x2d, 0xcc, 0xc9, 0x26, 0xb5, 0x48, 0x69, 0x5c, 0x1e, 0xf5,
|
||||||
0xdb, 0xa3, 0xa9, 0x44, 0x20, 0x2a, 0xa7, 0x14, 0x7b, 0x7e, 0x2b, 0xc1, 0x84, 0xfa, 0x98, 0x61,
|
0xff, 0x07, 0x8b, 0x12, 0x81, 0xa8, 0x9d, 0x51, 0xec, 0xc5, 0xb5, 0x04, 0x13, 0xea, 0x61, 0x86,
|
||||||
0x07, 0x40, 0x31, 0x52, 0xf3, 0xb0, 0xcd, 0xfc, 0x2d, 0x13, 0xfe, 0x66, 0x8e, 0xed, 0x6f, 0x55,
|
0x1d, 0x00, 0xc5, 0xca, 0xa6, 0x87, 0x6d, 0xe6, 0x1f, 0x99, 0xb0, 0x37, 0x71, 0x62, 0x7b, 0x8b,
|
||||||
0xf9, 0x83, 0x5b, 0x29, 0x36, 0x34, 0xc0, 0x43, 0x69, 0x4f, 0x03, 0xf9, 0xe8, 0xc0, 0xb6, 0x28,
|
0xca, 0x1e, 0x5c, 0x4b, 0xb1, 0xa1, 0x0c, 0x0b, 0x95, 0x3d, 0x0d, 0x14, 0xa3, 0x0b, 0x5b, 0xa3,
|
||||||
0xe3, 0xf0, 0x69, 0x2a, 0x2d, 0xf4, 0xd1, 0x02, 0x10, 0x68, 0x99, 0x14, 0x27, 0x55, 0x10, 0xb9,
|
0x8c, 0xc3, 0xe7, 0xa9, 0xb4, 0xd0, 0x07, 0x73, 0x40, 0xa0, 0x65, 0x52, 0x9c, 0x56, 0x4e, 0x14,
|
||||||
0x60, 0x24, 0x96, 0x12, 0x9f, 0x81, 0x2c, 0xe5, 0xc4, 0x12, 0xc7, 0x3f, 0x75, 0x7e, 0x6e, 0xfd,
|
0x82, 0x95, 0x58, 0x4a, 0x7c, 0x0a, 0xf2, 0x94, 0x13, 0x4b, 0x5c, 0xff, 0xd8, 0xc5, 0xa9, 0xe5,
|
||||||
0xca, 0x18, 0xb2, 0xad, 0xcc, 0x2b, 0xfe, 0xec, 0x7d, 0xc1, 0x84, 0x7c, 0xc2, 0xd2, 0xb7, 0x53,
|
0x6b, 0x43, 0x84, 0x6d, 0x6d, 0x5a, 0xf1, 0xe7, 0x1f, 0x09, 0x26, 0xe4, 0x13, 0x56, 0xbe, 0x19,
|
||||||
0x00, 0x46, 0x46, 0xc8, 0x31, 0xcd, 0x3a, 0x6e, 0xb4, 0xe0, 0x19, 0x90, 0xb1, 0xb1, 0x15, 0xa8,
|
0x03, 0x30, 0x52, 0x42, 0x8e, 0x69, 0x36, 0x70, 0x73, 0x07, 0x9e, 0x03, 0x39, 0x1b, 0x5b, 0x41,
|
||||||
0x35, 0x4c, 0xa5, 0x4f, 0xb1, 0x45, 0x90, 0x9c, 0x81, 0x3f, 0x6a, 0x00, 0xb6, 0xe5, 0x51, 0x34,
|
0xb4, 0x86, 0xa9, 0xf4, 0x09, 0xb6, 0x08, 0x92, 0x12, 0xf8, 0x83, 0x06, 0x60, 0x5b, 0x5e, 0x45,
|
||||||
0x37, 0x6d, 0xdb, 0xe1, 0x58, 0xec, 0x4e, 0x10, 0x60, 0x75, 0x8c, 0x00, 0x03, 0xdf, 0xfa, 0x4e,
|
0x6b, 0xd5, 0xb6, 0x1d, 0x8e, 0xc5, 0xe9, 0x04, 0x0e, 0xd6, 0x87, 0x70, 0x30, 0xb0, 0xad, 0x6f,
|
||||||
0x8a, 0xf5, 0xae, 0xcd, 0xbd, 0x6e, 0x74, 0x4a, 0x69, 0x03, 0x34, 0x20, 0x14, 0xd8, 0x02, 0xc0,
|
0xa5, 0x58, 0x1f, 0xd8, 0xdc, 0xeb, 0x46, 0xb7, 0x94, 0x56, 0x40, 0x19, 0xae, 0xc0, 0x1d, 0x00,
|
||||||
0x53, 0x9c, 0x35, 0x47, 0x25, 0xfc, 0xe8, 0xd5, 0x24, 0x08, 0xe7, 0xb6, 0x63, 0x3f, 0xa3, 0x46,
|
0x3c, 0xc5, 0xb9, 0xe9, 0xa8, 0x84, 0x1f, 0xbc, 0x9a, 0x04, 0xee, 0xdc, 0x73, 0xec, 0x17, 0xd4,
|
||||||
0x54, 0xb2, 0x50, 0x48, 0x89, 0x62, 0xf4, 0xab, 0x77, 0xc1, 0xca, 0x90, 0xb8, 0xe1, 0x49, 0x30,
|
0x88, 0x4a, 0x16, 0x0a, 0x29, 0x51, 0x8c, 0x7e, 0xf1, 0x01, 0x58, 0xe8, 0xe3, 0x37, 0x3c, 0x0d,
|
||||||
0xd5, 0x22, 0x5d, 0x7f, 0x2b, 0x91, 0xf8, 0x09, 0x97, 0x41, 0xb6, 0x83, 0xcd, 0x36, 0xf1, 0xb3,
|
0xc6, 0x76, 0x48, 0xd7, 0x3f, 0x4a, 0x24, 0x7e, 0xc2, 0x79, 0x90, 0xef, 0x60, 0xb3, 0x4d, 0xfc,
|
||||||
0x19, 0xf9, 0x1f, 0x37, 0x27, 0xaf, 0x6b, 0xa5, 0x3f, 0xb3, 0x71, 0x65, 0x89, 0xca, 0x05, 0xcf,
|
0x6c, 0x46, 0xfe, 0xc7, 0x9d, 0xd1, 0x5b, 0x5a, 0xe5, 0x8f, 0x7c, 0x3c, 0xb2, 0x44, 0xe5, 0x82,
|
||||||
0x83, 0x9c, 0x47, 0x5c, 0x93, 0x36, 0x30, 0x93, 0x1c, 0xd9, 0xca, 0x09, 0xa1, 0x12, 0xa4, 0xc6,
|
0x17, 0x41, 0xc1, 0x23, 0xae, 0x49, 0x9b, 0x98, 0x49, 0x8e, 0x7c, 0xed, 0x94, 0x88, 0x12, 0xa4,
|
||||||
0x50, 0x38, 0x0b, 0xbf, 0x00, 0x39, 0x46, 0x4c, 0xd2, 0xe0, 0x8e, 0xa7, 0x8a, 0xe7, 0x95, 0x11,
|
0xd6, 0x50, 0x28, 0x85, 0x9f, 0x83, 0x02, 0x23, 0x26, 0x69, 0x72, 0xc7, 0x53, 0xc5, 0xf3, 0xda,
|
||||||
0x35, 0x88, 0xeb, 0xc4, 0xac, 0x2a, 0xa8, 0x4f, 0x1f, 0x7c, 0xa1, 0x90, 0x12, 0x7e, 0x0e, 0x72,
|
0x80, 0x31, 0x88, 0x1b, 0xc4, 0xac, 0x2b, 0xa8, 0x4f, 0x1f, 0x7c, 0xa1, 0x90, 0x12, 0x7e, 0x06,
|
||||||
0x9c, 0x58, 0xae, 0x89, 0x39, 0x51, 0xbb, 0x79, 0x69, 0xf8, 0x6e, 0x0a, 0xda, 0x87, 0x4e, 0xb3,
|
0x0a, 0x9c, 0x58, 0xae, 0x89, 0x39, 0x51, 0xa7, 0x79, 0xa5, 0xff, 0x69, 0x0a, 0xda, 0x0d, 0xa7,
|
||||||
0xa6, 0x00, 0xb2, 0x22, 0x87, 0x0a, 0x0f, 0x46, 0x51, 0x48, 0x08, 0x29, 0xc8, 0x31, 0x2e, 0xae,
|
0xb5, 0xa9, 0x00, 0xb2, 0x22, 0x87, 0x11, 0x1e, 0xac, 0xa2, 0x90, 0x10, 0x52, 0x50, 0x60, 0x5c,
|
||||||
0x1d, 0xa3, 0x2b, 0x6b, 0xd1, 0xdc, 0xfa, 0xc6, 0x58, 0xb5, 0xd9, 0xa7, 0x88, 0x5c, 0x05, 0x23,
|
0x3c, 0x3b, 0x46, 0x57, 0xd6, 0xa2, 0xa9, 0xe5, 0x95, 0xa1, 0x6a, 0xb3, 0x4f, 0x11, 0x99, 0x0a,
|
||||||
0x28, 0xa4, 0x87, 0x9b, 0x60, 0xc1, 0xa2, 0x36, 0x22, 0xb8, 0xd9, 0xad, 0x92, 0x86, 0x63, 0x37,
|
0x56, 0x50, 0x48, 0x0f, 0x57, 0xc1, 0x8c, 0x45, 0x6d, 0x44, 0x70, 0xab, 0x5b, 0x27, 0x4d, 0xc7,
|
||||||
0x99, 0x2c, 0x6a, 0xd9, 0xca, 0x8a, 0x02, 0x2d, 0x6c, 0x27, 0xa7, 0x51, 0xbf, 0x3d, 0xdc, 0x02,
|
0x6e, 0x31, 0x59, 0xd4, 0xf2, 0xb5, 0x05, 0x05, 0x9a, 0x59, 0x4f, 0x8a, 0x51, 0xaf, 0x3e, 0x5c,
|
||||||
0xcb, 0x1e, 0xe9, 0x50, 0x71, 0x71, 0x7e, 0x4c, 0x19, 0x77, 0xbc, 0xee, 0x16, 0xb5, 0x28, 0x97,
|
0x03, 0xf3, 0x1e, 0xe9, 0x50, 0xf1, 0x70, 0x7e, 0x44, 0x19, 0x77, 0xbc, 0xee, 0x1a, 0xb5, 0x28,
|
||||||
0xa5, 0x2e, 0x5b, 0x29, 0xf4, 0xf6, 0xd7, 0x96, 0xd1, 0x80, 0x79, 0x34, 0x10, 0x25, 0xaa, 0xb0,
|
0x97, 0xa5, 0x2e, 0x5f, 0x2b, 0x1d, 0xee, 0x2f, 0xcd, 0xa3, 0x0c, 0x39, 0xca, 0x44, 0x89, 0x2a,
|
||||||
0x8b, 0xdb, 0x8c, 0x34, 0x65, 0xe9, 0xca, 0x45, 0x55, 0xf8, 0xa1, 0x1c, 0x45, 0x6a, 0x16, 0x1a,
|
0xec, 0xe2, 0x36, 0x23, 0x2d, 0x59, 0xba, 0x0a, 0x51, 0x15, 0xde, 0x90, 0xab, 0x48, 0x49, 0xa1,
|
||||||
0x09, 0x41, 0xe7, 0xfe, 0x99, 0xa0, 0xf3, 0xc3, 0xc5, 0x0c, 0x77, 0xc0, 0x8a, 0xeb, 0x39, 0x86,
|
0x91, 0x08, 0xe8, 0xc2, 0xdf, 0x0b, 0xe8, 0x62, 0xff, 0x60, 0x86, 0x5b, 0x60, 0xc1, 0xf5, 0x1c,
|
||||||
0x47, 0x18, 0xbb, 0x43, 0x70, 0xd3, 0xa4, 0x36, 0x09, 0x76, 0x6a, 0x56, 0xae, 0xf0, 0x74, 0x6f,
|
0xc3, 0x23, 0x8c, 0xdd, 0x27, 0xb8, 0x65, 0x52, 0x9b, 0x04, 0x27, 0x35, 0x29, 0x77, 0x78, 0xf6,
|
||||||
0x7f, 0x6d, 0xe5, 0xe1, 0x60, 0x13, 0x34, 0x0c, 0x5b, 0xfa, 0x3e, 0x03, 0x4e, 0xf6, 0xdf, 0xa3,
|
0x70, 0x7f, 0x69, 0x61, 0x23, 0x5b, 0x05, 0xf5, 0xc3, 0x56, 0xbe, 0xcb, 0x81, 0xd3, 0xbd, 0xef,
|
||||||
0xf0, 0x13, 0x00, 0x9d, 0xba, 0xec, 0x7d, 0x9a, 0xf7, 0xfc, 0xce, 0x83, 0x3a, 0xb6, 0x14, 0xfa,
|
0x28, 0xfc, 0x18, 0x40, 0xa7, 0x21, 0x7b, 0x9f, 0xd6, 0x43, 0xbf, 0xf3, 0xa0, 0x8e, 0x2d, 0x03,
|
||||||
0x54, 0x94, 0xf1, 0x0f, 0x52, 0x16, 0x68, 0x00, 0x0a, 0x5e, 0x8c, 0xa5, 0xca, 0xa4, 0x0c, 0x34,
|
0x7d, 0x2c, 0xca, 0xf8, 0xc7, 0x29, 0x0d, 0x94, 0x81, 0x82, 0x97, 0x63, 0xa9, 0x32, 0x2a, 0x1d,
|
||||||
0xd4, 0xc1, 0x80, 0x74, 0xd9, 0x04, 0x0b, 0xaa, 0x6a, 0x04, 0x93, 0x52, 0xd6, 0x31, 0x1d, 0xec,
|
0x0d, 0xe3, 0x20, 0x23, 0x5d, 0x56, 0xc1, 0x8c, 0xaa, 0x1a, 0x81, 0x50, 0x86, 0x75, 0x2c, 0x0e,
|
||||||
0x24, 0xa7, 0x51, 0xbf, 0x3d, 0xbc, 0x07, 0x16, 0x71, 0x07, 0x53, 0x13, 0xd7, 0x4d, 0x12, 0x92,
|
0xb6, 0x92, 0x62, 0xd4, 0xab, 0x0f, 0x1f, 0x82, 0x59, 0xdc, 0xc1, 0xd4, 0xc4, 0x0d, 0x93, 0x84,
|
||||||
0x64, 0x24, 0xc9, 0xff, 0x15, 0xc9, 0xe2, 0x66, 0xbf, 0x01, 0x4a, 0x63, 0xe0, 0x36, 0x58, 0x6a,
|
0x24, 0x39, 0x49, 0xf2, 0x5f, 0x45, 0x32, 0xbb, 0xda, 0xab, 0x80, 0xd2, 0x18, 0xb8, 0x0e, 0xe6,
|
||||||
0xdb, 0x69, 0x2a, 0x5f, 0x97, 0xa7, 0x15, 0xd5, 0xd2, 0x4e, 0xda, 0x04, 0x0d, 0xc2, 0x41, 0x17,
|
0xda, 0x76, 0x9a, 0xca, 0x8f, 0xcb, 0xb3, 0x8a, 0x6a, 0x6e, 0x2b, 0xad, 0x82, 0xb2, 0x70, 0xd0,
|
||||||
0x80, 0x46, 0x70, 0xe5, 0xb3, 0xc2, 0xb4, 0xac, 0xc9, 0xef, 0x8f, 0x91, 0x4f, 0x61, 0xdf, 0x10,
|
0x05, 0xa0, 0x19, 0x3c, 0xf9, 0xac, 0x34, 0x2e, 0x6b, 0xf2, 0x7b, 0x43, 0xe4, 0x53, 0xd8, 0x37,
|
||||||
0xd5, 0xbf, 0x70, 0x88, 0xa1, 0x98, 0x0f, 0xb8, 0x01, 0xe6, 0x3d, 0x91, 0x21, 0x61, 0xe8, 0x33,
|
0x44, 0xf5, 0x2f, 0x5c, 0x62, 0x28, 0x66, 0x03, 0xae, 0x80, 0x69, 0x4f, 0x64, 0x48, 0xe8, 0xfa,
|
||||||
0x32, 0xf4, 0xff, 0x29, 0xd8, 0x3c, 0x8a, 0x4f, 0xa2, 0xa4, 0x6d, 0xe9, 0x77, 0x2d, 0x7e, 0x09,
|
0x84, 0x74, 0xfd, 0x3f, 0x0a, 0x36, 0x8d, 0xe2, 0x42, 0x94, 0xd4, 0xad, 0xfc, 0xa6, 0xc5, 0x1f,
|
||||||
0x05, 0x29, 0x0b, 0x6f, 0x26, 0x5a, 0xa6, 0xb3, 0x7d, 0x2d, 0xd3, 0xa9, 0x34, 0x22, 0xd6, 0x31,
|
0xa1, 0x20, 0x65, 0xe1, 0x9d, 0x44, 0xcb, 0x74, 0xbe, 0xa7, 0x65, 0x3a, 0x93, 0x46, 0xc4, 0x3a,
|
||||||
0x75, 0xc1, 0xbc, 0x10, 0x34, 0xb5, 0x0d, 0xff, 0x10, 0x55, 0x41, 0xfc, 0xf0, 0x58, 0xe9, 0x12,
|
0xa6, 0x2e, 0x98, 0x16, 0x01, 0x4d, 0x6d, 0xc3, 0xbf, 0x44, 0x55, 0x10, 0x3f, 0x38, 0x51, 0xba,
|
||||||
0xa2, 0x63, 0xd7, 0xe8, 0xa2, 0x5c, 0x4d, 0x7c, 0x12, 0x25, 0x3d, 0x95, 0x6e, 0x81, 0x7c, 0x32,
|
0x84, 0xe8, 0xd8, 0x33, 0x3a, 0x2b, 0x77, 0x13, 0x17, 0xa2, 0xa4, 0xa5, 0xca, 0x5d, 0x50, 0x4c,
|
||||||
0xd7, 0x7c, 0x5d, 0xfa, 0x89, 0xaf, 0x94, 0x1d, 0xd3, 0xa5, 0x3f, 0x8e, 0x42, 0x8b, 0xd2, 0x2b,
|
0xe6, 0x9a, 0x1f, 0x97, 0x7e, 0xe2, 0xab, 0xc8, 0x8e, 0xc5, 0xa5, 0xbf, 0x8e, 0x42, 0x8d, 0xca,
|
||||||
0x0d, 0xac, 0x0c, 0xf1, 0x0e, 0x4d, 0x90, 0xb7, 0xf0, 0x6e, 0x4c, 0x07, 0x47, 0xf6, 0xe0, 0xe2,
|
0x6b, 0x0d, 0x2c, 0xf4, 0xb1, 0x0e, 0x4d, 0x50, 0xb4, 0xf0, 0x6e, 0x2c, 0x0e, 0x8e, 0xed, 0xc1,
|
||||||
0xf5, 0xa1, 0xfb, 0xaf, 0x0f, 0xfd, 0xbe, 0xcd, 0x1f, 0x78, 0x55, 0xee, 0x51, 0xdb, 0xa8, 0x40,
|
0xc5, 0xf4, 0xa1, 0xfb, 0xd3, 0x87, 0xfe, 0xc8, 0xe6, 0x8f, 0xbd, 0x3a, 0xf7, 0xa8, 0x6d, 0xd4,
|
||||||
0xd1, 0x5f, 0x6d, 0x27, 0xb8, 0x50, 0x1f, 0x37, 0x7c, 0x02, 0x72, 0x16, 0xde, 0xad, 0xb6, 0x3d,
|
0xa0, 0xe8, 0xaf, 0xd6, 0x13, 0x5c, 0xa8, 0x87, 0x1b, 0x3e, 0x03, 0x05, 0x0b, 0xef, 0xd6, 0xdb,
|
||||||
0x23, 0xd8, 0xbf, 0xe3, 0xfb, 0x91, 0xb7, 0xc9, 0xb6, 0x62, 0x41, 0x21, 0x5f, 0xe9, 0x87, 0x49,
|
0x9e, 0x11, 0x9c, 0xdf, 0xc9, 0xed, 0xc8, 0xd7, 0x64, 0x5d, 0xb1, 0xa0, 0x90, 0xaf, 0xf2, 0xfd,
|
||||||
0x90, 0xad, 0x36, 0xb0, 0x49, 0xde, 0xc0, 0x8b, 0xa2, 0x96, 0x78, 0x51, 0xac, 0x8f, 0xac, 0x01,
|
0x28, 0xc8, 0xd7, 0x9b, 0xd8, 0x24, 0x6f, 0x61, 0xa2, 0xd8, 0x4c, 0x4c, 0x14, 0xcb, 0x03, 0xc7,
|
||||||
0x19, 0xdf, 0xd0, 0xc7, 0xc4, 0xd3, 0xbe, 0xc7, 0xc4, 0xd5, 0x63, 0xf2, 0x1e, 0xfe, 0x8e, 0xb8,
|
0x80, 0xf4, 0xaf, 0xef, 0x30, 0xf1, 0xbc, 0x67, 0x98, 0xb8, 0x7e, 0x42, 0xde, 0xa3, 0xe7, 0x88,
|
||||||
0x01, 0x66, 0x43, 0xf7, 0x89, 0xc2, 0xa6, 0x1d, 0x55, 0xd8, 0x4a, 0x3f, 0x4d, 0x82, 0xb9, 0x98,
|
0xdb, 0x60, 0x32, 0x34, 0x9f, 0x28, 0x6c, 0xda, 0x71, 0x85, 0xad, 0xf2, 0xd3, 0x28, 0x98, 0x8a,
|
||||||
0x8b, 0xe3, 0xa1, 0xa1, 0x9b, 0xe8, 0x22, 0x44, 0xe5, 0xa8, 0x8c, 0xb3, 0x30, 0x3d, 0xe8, 0x20,
|
0x99, 0x38, 0x19, 0x1a, 0xba, 0x89, 0x2e, 0x42, 0x54, 0x8e, 0xda, 0x30, 0x1b, 0xd3, 0x83, 0x0e,
|
||||||
0xfc, 0xe6, 0x2d, 0xba, 0x90, 0xd3, 0x8d, 0xc5, 0x2d, 0x90, 0xe7, 0xd8, 0x33, 0x08, 0x0f, 0xe6,
|
0xc2, 0x6f, 0xde, 0xa2, 0x07, 0x39, 0xdd, 0x58, 0xdc, 0x05, 0x45, 0x8e, 0x3d, 0x83, 0xf0, 0x40,
|
||||||
0xe4, 0x86, 0xce, 0x46, 0xcf, 0x80, 0x5a, 0x62, 0x16, 0xf5, 0x59, 0xaf, 0x6e, 0x80, 0xf9, 0x84,
|
0x26, 0x0f, 0x74, 0x32, 0x1a, 0x03, 0x36, 0x13, 0x52, 0xd4, 0xa3, 0xbd, 0xb8, 0x02, 0xa6, 0x13,
|
||||||
0xb3, 0x63, 0x75, 0x5c, 0xbf, 0x88, 0xcd, 0xe2, 0x98, 0x93, 0x67, 0x6d, 0xb3, 0x4a, 0xde, 0xc4,
|
0xc6, 0x4e, 0xd4, 0x71, 0xfd, 0x22, 0x0e, 0x8b, 0x63, 0x4e, 0x5e, 0xb4, 0xcd, 0x3a, 0x79, 0x1b,
|
||||||
0xfb, 0xf6, 0x49, 0x42, 0x8d, 0xd7, 0x47, 0xdf, 0xdc, 0x28, 0xca, 0xa1, 0x9a, 0xac, 0xf7, 0x69,
|
0xf3, 0xed, 0xb3, 0x44, 0x34, 0xde, 0x1a, 0xfc, 0x70, 0x23, 0x2f, 0xfb, 0xc6, 0x64, 0xa3, 0x27,
|
||||||
0xf2, 0xe6, 0x58, 0xec, 0x87, 0x2b, 0xf3, 0x37, 0x0d, 0x2c, 0xc4, 0xac, 0xdf, 0xc0, 0xf3, 0xe7,
|
0x26, 0xef, 0x0c, 0xc5, 0x7e, 0x74, 0x64, 0xfe, 0xaa, 0x81, 0x99, 0x98, 0xf6, 0x5b, 0x18, 0x7f,
|
||||||
0x71, 0xf2, 0xf9, 0x73, 0x75, 0x9c, 0x45, 0x0d, 0x79, 0xff, 0xfc, 0x3a, 0x95, 0x58, 0xcc, 0x7f,
|
0x9e, 0x26, 0xc7, 0x9f, 0xeb, 0xc3, 0x6c, 0xaa, 0xcf, 0xfc, 0xf3, 0x63, 0x2e, 0xb1, 0x99, 0x7f,
|
||||||
0xa8, 0xe3, 0xfe, 0x5a, 0x03, 0xcb, 0x1d, 0xc7, 0x6c, 0x5b, 0xe4, 0xb6, 0x89, 0xa9, 0x15, 0x58,
|
0x51, 0xc7, 0xfd, 0x95, 0x06, 0xe6, 0x3b, 0x8e, 0xd9, 0xb6, 0xc8, 0x3d, 0x13, 0x53, 0x2b, 0xd0,
|
||||||
0x88, 0xfe, 0xe5, 0x88, 0x37, 0xa6, 0xf4, 0x44, 0x3c, 0x46, 0x19, 0x27, 0x36, 0x7f, 0x14, 0x71,
|
0x10, 0xfd, 0xcb, 0x31, 0x33, 0xa6, 0xb4, 0x44, 0x3c, 0x46, 0x19, 0x27, 0x36, 0x7f, 0x12, 0x71,
|
||||||
0x54, 0xde, 0x52, 0xfe, 0x96, 0x1f, 0x0d, 0x20, 0x46, 0x03, 0xdd, 0xc1, 0x77, 0xc1, 0x9c, 0x68,
|
0xd4, 0xfe, 0xa7, 0xec, 0xcd, 0x3f, 0xc9, 0x20, 0x46, 0x99, 0xe6, 0xe0, 0xbb, 0x60, 0x4a, 0x34,
|
||||||
0xe4, 0x68, 0x83, 0x88, 0xd7, 0xa5, 0xfa, 0x7f, 0x61, 0x49, 0x11, 0xcd, 0x55, 0xa3, 0x29, 0x14,
|
0x72, 0xb4, 0x49, 0xc4, 0x74, 0xa9, 0xfe, 0x5f, 0x98, 0x53, 0x44, 0x53, 0xf5, 0x48, 0x84, 0xe2,
|
||||||
0xb7, 0x2b, 0x7d, 0xa3, 0x81, 0xc5, 0x94, 0x66, 0xe1, 0x47, 0x87, 0x74, 0x93, 0xa7, 0xfe, 0xad,
|
0x7a, 0x70, 0x1b, 0xcc, 0xb9, 0x4e, 0x6b, 0x1d, 0xdb, 0xd8, 0x20, 0xe2, 0x69, 0xdc, 0x70, 0x4c,
|
||||||
0x4e, 0xb2, 0x72, 0x61, 0xef, 0xa0, 0x38, 0xf1, 0xf2, 0xa0, 0x38, 0xf1, 0xc7, 0x41, 0x71, 0xe2,
|
0xda, 0xec, 0xca, 0x0e, 0x7c, 0xb2, 0x76, 0x23, 0xe8, 0x98, 0x36, 0xd2, 0x2a, 0x6f, 0x44, 0xeb,
|
||||||
0xab, 0x5e, 0x51, 0xdb, 0xeb, 0x15, 0xb5, 0x97, 0xbd, 0xa2, 0xf6, 0x57, 0xaf, 0xa8, 0x7d, 0xf7,
|
0x9a, 0x5e, 0x96, 0xbd, 0x43, 0x16, 0x65, 0xe5, 0x6b, 0x0d, 0xcc, 0xa6, 0xb2, 0x03, 0x7e, 0x78,
|
||||||
0xaa, 0x38, 0xf1, 0x64, 0x46, 0x29, 0xf2, 0xef, 0x00, 0x00, 0x00, 0xff, 0xff, 0xd6, 0xb9, 0xde,
|
0x44, 0xdf, 0x7a, 0xe6, 0x9f, 0xea, 0x59, 0x6b, 0x97, 0xf6, 0x0e, 0xca, 0x23, 0xaf, 0x0e, 0xca,
|
||||||
0x1a, 0x56, 0x15, 0x00, 0x00,
|
0x23, 0xbf, 0x1f, 0x94, 0x47, 0xbe, 0x3c, 0x2c, 0x6b, 0x7b, 0x87, 0x65, 0xed, 0xd5, 0x61, 0x59,
|
||||||
|
0xfb, 0xf3, 0xb0, 0xac, 0x7d, 0xfb, 0xba, 0x3c, 0xf2, 0x6c, 0x42, 0xc5, 0xfe, 0x5f, 0x01, 0x00,
|
||||||
|
0x00, 0xff, 0xff, 0xdf, 0x80, 0xf3, 0x34, 0xc0, 0x15, 0x00, 0x00,
|
||||||
}
|
}
|
||||||
|
@@ -294,7 +294,7 @@ message StatefulSetList {
|
|||||||
|
|
||||||
// A StatefulSetSpec is the specification of a StatefulSet.
|
// A StatefulSetSpec is the specification of a StatefulSet.
|
||||||
message StatefulSetSpec {
|
message StatefulSetSpec {
|
||||||
// Replicas is the desired number of replicas of the given Template.
|
// replicas is the desired number of replicas of the given Template.
|
||||||
// These are replicas in the sense that they are instantiations of the
|
// These are replicas in the sense that they are instantiations of the
|
||||||
// same Template, but individual replicas also have a consistent identity.
|
// same Template, but individual replicas also have a consistent identity.
|
||||||
// If unspecified, defaults to 1.
|
// If unspecified, defaults to 1.
|
||||||
@@ -302,19 +302,19 @@ message StatefulSetSpec {
|
|||||||
// +optional
|
// +optional
|
||||||
optional int32 replicas = 1;
|
optional int32 replicas = 1;
|
||||||
|
|
||||||
// Selector is a label query over pods that should match the replica count.
|
// selector is a label query over pods that should match the replica count.
|
||||||
// If empty, defaulted to labels on the pod template.
|
// If empty, defaulted to labels on the pod template.
|
||||||
// More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors
|
// More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors
|
||||||
// +optional
|
// +optional
|
||||||
optional k8s.io.apimachinery.pkg.apis.meta.v1.LabelSelector selector = 2;
|
optional k8s.io.apimachinery.pkg.apis.meta.v1.LabelSelector selector = 2;
|
||||||
|
|
||||||
// Template is the object that describes the pod that will be created if
|
// template is the object that describes the pod that will be created if
|
||||||
// insufficient replicas are detected. Each pod stamped out by the StatefulSet
|
// insufficient replicas are detected. Each pod stamped out by the StatefulSet
|
||||||
// will fulfill this Template, but have a unique identity from the rest
|
// will fulfill this Template, but have a unique identity from the rest
|
||||||
// of the StatefulSet.
|
// of the StatefulSet.
|
||||||
optional k8s.io.kubernetes.pkg.api.v1.PodTemplateSpec template = 3;
|
optional k8s.io.kubernetes.pkg.api.v1.PodTemplateSpec template = 3;
|
||||||
|
|
||||||
// VolumeClaimTemplates is a list of claims that pods are allowed to reference.
|
// volumeClaimTemplates is a list of claims that pods are allowed to reference.
|
||||||
// The StatefulSet controller is responsible for mapping network identities to
|
// The StatefulSet controller is responsible for mapping network identities to
|
||||||
// claims in a way that maintains the identity of a pod. Every claim in
|
// claims in a way that maintains the identity of a pod. Every claim in
|
||||||
// this list must have at least one matching (by name) volumeMount in one
|
// this list must have at least one matching (by name) volumeMount in one
|
||||||
@@ -324,21 +324,32 @@ message StatefulSetSpec {
|
|||||||
// +optional
|
// +optional
|
||||||
repeated k8s.io.kubernetes.pkg.api.v1.PersistentVolumeClaim volumeClaimTemplates = 4;
|
repeated k8s.io.kubernetes.pkg.api.v1.PersistentVolumeClaim volumeClaimTemplates = 4;
|
||||||
|
|
||||||
// ServiceName is the name of the service that governs this StatefulSet.
|
// serviceName is the name of the service that governs this StatefulSet.
|
||||||
// This service must exist before the StatefulSet, and is responsible for
|
// This service must exist before the StatefulSet, and is responsible for
|
||||||
// the network identity of the set. Pods get DNS/hostnames that follow the
|
// the network identity of the set. Pods get DNS/hostnames that follow the
|
||||||
// pattern: pod-specific-string.serviceName.default.svc.cluster.local
|
// pattern: pod-specific-string.serviceName.default.svc.cluster.local
|
||||||
// where "pod-specific-string" is managed by the StatefulSet controller.
|
// where "pod-specific-string" is managed by the StatefulSet controller.
|
||||||
optional string serviceName = 5;
|
optional string serviceName = 5;
|
||||||
|
|
||||||
|
// podManagementPolicy controls how pods are created during initial scale up,
|
||||||
|
// when replacing pods on nodes, or when scaling down. The default policy is
|
||||||
|
// `OrderedReady`, where pods are created in increasing order (pod-0, then
|
||||||
|
// pod-1, etc) and the controller will wait until each pod is ready before
|
||||||
|
// continuing. When scaling down, the pods are removed in the opposite order.
|
||||||
|
// The alternative policy is `Parallel` which will create pods in parallel
|
||||||
|
// to match the desired scale without waiting, and on scale down will delete
|
||||||
|
// all pods at once.
|
||||||
|
// +optional
|
||||||
|
optional string podManagementPolicy = 6;
|
||||||
}
|
}
|
||||||
|
|
||||||
// StatefulSetStatus represents the current state of a StatefulSet.
|
// StatefulSetStatus represents the current state of a StatefulSet.
|
||||||
message StatefulSetStatus {
|
message StatefulSetStatus {
|
||||||
// most recent generation observed by this StatefulSet.
|
// observedGeneration is the most recent generation observed by this StatefulSet.
|
||||||
// +optional
|
// +optional
|
||||||
optional int64 observedGeneration = 1;
|
optional int64 observedGeneration = 1;
|
||||||
|
|
||||||
// Replicas is the number of actual replicas.
|
// replicas is the number of actual replicas.
|
||||||
optional int32 replicas = 2;
|
optional int32 replicas = 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -1340,6 +1340,32 @@ func (x *StatefulSet) codecDecodeSelfFromArray(l int, d *codec1978.Decoder) {
|
|||||||
z.DecSendContainerState(codecSelfer_containerArrayEnd1234)
|
z.DecSendContainerState(codecSelfer_containerArrayEnd1234)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (x PodManagementPolicyType) CodecEncodeSelf(e *codec1978.Encoder) {
|
||||||
|
var h codecSelfer1234
|
||||||
|
z, r := codec1978.GenHelperEncoder(e)
|
||||||
|
_, _, _ = h, z, r
|
||||||
|
yym1 := z.EncBinary()
|
||||||
|
_ = yym1
|
||||||
|
if false {
|
||||||
|
} else if z.HasExtensions() && z.EncExt(x) {
|
||||||
|
} else {
|
||||||
|
r.EncodeString(codecSelferC_UTF81234, string(x))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *PodManagementPolicyType) CodecDecodeSelf(d *codec1978.Decoder) {
|
||||||
|
var h codecSelfer1234
|
||||||
|
z, r := codec1978.GenHelperDecoder(d)
|
||||||
|
_, _, _ = h, z, r
|
||||||
|
yym1 := z.DecBinary()
|
||||||
|
_ = yym1
|
||||||
|
if false {
|
||||||
|
} else if z.HasExtensions() && z.DecExt(x) {
|
||||||
|
} else {
|
||||||
|
*((*string)(x)) = r.DecodeString()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func (x *StatefulSetSpec) CodecEncodeSelf(e *codec1978.Encoder) {
|
func (x *StatefulSetSpec) CodecEncodeSelf(e *codec1978.Encoder) {
|
||||||
var h codecSelfer1234
|
var h codecSelfer1234
|
||||||
z, r := codec1978.GenHelperEncoder(e)
|
z, r := codec1978.GenHelperEncoder(e)
|
||||||
@@ -1354,15 +1380,16 @@ func (x *StatefulSetSpec) CodecEncodeSelf(e *codec1978.Encoder) {
|
|||||||
} else {
|
} else {
|
||||||
yysep2 := !z.EncBinary()
|
yysep2 := !z.EncBinary()
|
||||||
yy2arr2 := z.EncBasicHandle().StructToArray
|
yy2arr2 := z.EncBasicHandle().StructToArray
|
||||||
var yyq2 [5]bool
|
var yyq2 [6]bool
|
||||||
_, _, _ = yysep2, yyq2, yy2arr2
|
_, _, _ = yysep2, yyq2, yy2arr2
|
||||||
const yyr2 bool = false
|
const yyr2 bool = false
|
||||||
yyq2[0] = x.Replicas != nil
|
yyq2[0] = x.Replicas != nil
|
||||||
yyq2[1] = x.Selector != nil
|
yyq2[1] = x.Selector != nil
|
||||||
yyq2[3] = len(x.VolumeClaimTemplates) != 0
|
yyq2[3] = len(x.VolumeClaimTemplates) != 0
|
||||||
|
yyq2[5] = x.PodManagementPolicy != ""
|
||||||
var yynn2 int
|
var yynn2 int
|
||||||
if yyr2 || yy2arr2 {
|
if yyr2 || yy2arr2 {
|
||||||
r.EncodeArrayStart(5)
|
r.EncodeArrayStart(6)
|
||||||
} else {
|
} else {
|
||||||
yynn2 = 2
|
yynn2 = 2
|
||||||
for _, b := range yyq2 {
|
for _, b := range yyq2 {
|
||||||
@@ -1506,6 +1533,21 @@ func (x *StatefulSetSpec) CodecEncodeSelf(e *codec1978.Encoder) {
|
|||||||
r.EncodeString(codecSelferC_UTF81234, string(x.ServiceName))
|
r.EncodeString(codecSelferC_UTF81234, string(x.ServiceName))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if yyr2 || yy2arr2 {
|
||||||
|
z.EncSendContainerState(codecSelfer_containerArrayElem1234)
|
||||||
|
if yyq2[5] {
|
||||||
|
x.PodManagementPolicy.CodecEncodeSelf(e)
|
||||||
|
} else {
|
||||||
|
r.EncodeString(codecSelferC_UTF81234, "")
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if yyq2[5] {
|
||||||
|
z.EncSendContainerState(codecSelfer_containerMapKey1234)
|
||||||
|
r.EncodeString(codecSelferC_UTF81234, string("podManagementPolicy"))
|
||||||
|
z.EncSendContainerState(codecSelfer_containerMapValue1234)
|
||||||
|
x.PodManagementPolicy.CodecEncodeSelf(e)
|
||||||
|
}
|
||||||
|
}
|
||||||
if yyr2 || yy2arr2 {
|
if yyr2 || yy2arr2 {
|
||||||
z.EncSendContainerState(codecSelfer_containerArrayEnd1234)
|
z.EncSendContainerState(codecSelfer_containerArrayEnd1234)
|
||||||
} else {
|
} else {
|
||||||
@@ -1631,6 +1673,13 @@ func (x *StatefulSetSpec) codecDecodeSelfFromMap(l int, d *codec1978.Decoder) {
|
|||||||
*((*string)(yyv11)) = r.DecodeString()
|
*((*string)(yyv11)) = r.DecodeString()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
case "podManagementPolicy":
|
||||||
|
if r.TryDecodeAsNil() {
|
||||||
|
x.PodManagementPolicy = ""
|
||||||
|
} else {
|
||||||
|
yyv13 := &x.PodManagementPolicy
|
||||||
|
yyv13.CodecDecodeSelf(d)
|
||||||
|
}
|
||||||
default:
|
default:
|
||||||
z.DecStructFieldNotFound(-1, yys3)
|
z.DecStructFieldNotFound(-1, yys3)
|
||||||
} // end switch yys3
|
} // end switch yys3
|
||||||
@@ -1642,16 +1691,16 @@ func (x *StatefulSetSpec) codecDecodeSelfFromArray(l int, d *codec1978.Decoder)
|
|||||||
var h codecSelfer1234
|
var h codecSelfer1234
|
||||||
z, r := codec1978.GenHelperDecoder(d)
|
z, r := codec1978.GenHelperDecoder(d)
|
||||||
_, _, _ = h, z, r
|
_, _, _ = h, z, r
|
||||||
var yyj13 int
|
var yyj14 int
|
||||||
var yyb13 bool
|
var yyb14 bool
|
||||||
var yyhl13 bool = l >= 0
|
var yyhl14 bool = l >= 0
|
||||||
yyj13++
|
yyj14++
|
||||||
if yyhl13 {
|
if yyhl14 {
|
||||||
yyb13 = yyj13 > l
|
yyb14 = yyj14 > l
|
||||||
} else {
|
} else {
|
||||||
yyb13 = r.CheckBreak()
|
yyb14 = r.CheckBreak()
|
||||||
}
|
}
|
||||||
if yyb13 {
|
if yyb14 {
|
||||||
z.DecSendContainerState(codecSelfer_containerArrayEnd1234)
|
z.DecSendContainerState(codecSelfer_containerArrayEnd1234)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@@ -1664,20 +1713,20 @@ func (x *StatefulSetSpec) codecDecodeSelfFromArray(l int, d *codec1978.Decoder)
|
|||||||
if x.Replicas == nil {
|
if x.Replicas == nil {
|
||||||
x.Replicas = new(int32)
|
x.Replicas = new(int32)
|
||||||
}
|
}
|
||||||
yym15 := z.DecBinary()
|
yym16 := z.DecBinary()
|
||||||
_ = yym15
|
_ = yym16
|
||||||
if false {
|
if false {
|
||||||
} else {
|
} else {
|
||||||
*((*int32)(x.Replicas)) = int32(r.DecodeInt(32))
|
*((*int32)(x.Replicas)) = int32(r.DecodeInt(32))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
yyj13++
|
yyj14++
|
||||||
if yyhl13 {
|
if yyhl14 {
|
||||||
yyb13 = yyj13 > l
|
yyb14 = yyj14 > l
|
||||||
} else {
|
} else {
|
||||||
yyb13 = r.CheckBreak()
|
yyb14 = r.CheckBreak()
|
||||||
}
|
}
|
||||||
if yyb13 {
|
if yyb14 {
|
||||||
z.DecSendContainerState(codecSelfer_containerArrayEnd1234)
|
z.DecSendContainerState(codecSelfer_containerArrayEnd1234)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@@ -1690,21 +1739,21 @@ func (x *StatefulSetSpec) codecDecodeSelfFromArray(l int, d *codec1978.Decoder)
|
|||||||
if x.Selector == nil {
|
if x.Selector == nil {
|
||||||
x.Selector = new(pkg1_v1.LabelSelector)
|
x.Selector = new(pkg1_v1.LabelSelector)
|
||||||
}
|
}
|
||||||
yym17 := z.DecBinary()
|
yym18 := z.DecBinary()
|
||||||
_ = yym17
|
_ = yym18
|
||||||
if false {
|
if false {
|
||||||
} else if z.HasExtensions() && z.DecExt(x.Selector) {
|
} else if z.HasExtensions() && z.DecExt(x.Selector) {
|
||||||
} else {
|
} else {
|
||||||
z.DecFallback(x.Selector, false)
|
z.DecFallback(x.Selector, false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
yyj13++
|
yyj14++
|
||||||
if yyhl13 {
|
if yyhl14 {
|
||||||
yyb13 = yyj13 > l
|
yyb14 = yyj14 > l
|
||||||
} else {
|
} else {
|
||||||
yyb13 = r.CheckBreak()
|
yyb14 = r.CheckBreak()
|
||||||
}
|
}
|
||||||
if yyb13 {
|
if yyb14 {
|
||||||
z.DecSendContainerState(codecSelfer_containerArrayEnd1234)
|
z.DecSendContainerState(codecSelfer_containerArrayEnd1234)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@@ -1712,16 +1761,16 @@ func (x *StatefulSetSpec) codecDecodeSelfFromArray(l int, d *codec1978.Decoder)
|
|||||||
if r.TryDecodeAsNil() {
|
if r.TryDecodeAsNil() {
|
||||||
x.Template = pkg3_v1.PodTemplateSpec{}
|
x.Template = pkg3_v1.PodTemplateSpec{}
|
||||||
} else {
|
} else {
|
||||||
yyv18 := &x.Template
|
yyv19 := &x.Template
|
||||||
yyv18.CodecDecodeSelf(d)
|
yyv19.CodecDecodeSelf(d)
|
||||||
}
|
}
|
||||||
yyj13++
|
yyj14++
|
||||||
if yyhl13 {
|
if yyhl14 {
|
||||||
yyb13 = yyj13 > l
|
yyb14 = yyj14 > l
|
||||||
} else {
|
} else {
|
||||||
yyb13 = r.CheckBreak()
|
yyb14 = r.CheckBreak()
|
||||||
}
|
}
|
||||||
if yyb13 {
|
if yyb14 {
|
||||||
z.DecSendContainerState(codecSelfer_containerArrayEnd1234)
|
z.DecSendContainerState(codecSelfer_containerArrayEnd1234)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@@ -1729,21 +1778,21 @@ func (x *StatefulSetSpec) codecDecodeSelfFromArray(l int, d *codec1978.Decoder)
|
|||||||
if r.TryDecodeAsNil() {
|
if r.TryDecodeAsNil() {
|
||||||
x.VolumeClaimTemplates = nil
|
x.VolumeClaimTemplates = nil
|
||||||
} else {
|
} else {
|
||||||
yyv19 := &x.VolumeClaimTemplates
|
yyv20 := &x.VolumeClaimTemplates
|
||||||
yym20 := z.DecBinary()
|
yym21 := z.DecBinary()
|
||||||
_ = yym20
|
_ = yym21
|
||||||
if false {
|
if false {
|
||||||
} else {
|
} else {
|
||||||
h.decSlicev1_PersistentVolumeClaim((*[]pkg3_v1.PersistentVolumeClaim)(yyv19), d)
|
h.decSlicev1_PersistentVolumeClaim((*[]pkg3_v1.PersistentVolumeClaim)(yyv20), d)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
yyj13++
|
yyj14++
|
||||||
if yyhl13 {
|
if yyhl14 {
|
||||||
yyb13 = yyj13 > l
|
yyb14 = yyj14 > l
|
||||||
} else {
|
} else {
|
||||||
yyb13 = r.CheckBreak()
|
yyb14 = r.CheckBreak()
|
||||||
}
|
}
|
||||||
if yyb13 {
|
if yyb14 {
|
||||||
z.DecSendContainerState(codecSelfer_containerArrayEnd1234)
|
z.DecSendContainerState(codecSelfer_containerArrayEnd1234)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@@ -1751,26 +1800,43 @@ func (x *StatefulSetSpec) codecDecodeSelfFromArray(l int, d *codec1978.Decoder)
|
|||||||
if r.TryDecodeAsNil() {
|
if r.TryDecodeAsNil() {
|
||||||
x.ServiceName = ""
|
x.ServiceName = ""
|
||||||
} else {
|
} else {
|
||||||
yyv21 := &x.ServiceName
|
yyv22 := &x.ServiceName
|
||||||
yym22 := z.DecBinary()
|
yym23 := z.DecBinary()
|
||||||
_ = yym22
|
_ = yym23
|
||||||
if false {
|
if false {
|
||||||
} else {
|
} else {
|
||||||
*((*string)(yyv21)) = r.DecodeString()
|
*((*string)(yyv22)) = r.DecodeString()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
yyj14++
|
||||||
|
if yyhl14 {
|
||||||
|
yyb14 = yyj14 > l
|
||||||
|
} else {
|
||||||
|
yyb14 = r.CheckBreak()
|
||||||
|
}
|
||||||
|
if yyb14 {
|
||||||
|
z.DecSendContainerState(codecSelfer_containerArrayEnd1234)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
z.DecSendContainerState(codecSelfer_containerArrayElem1234)
|
||||||
|
if r.TryDecodeAsNil() {
|
||||||
|
x.PodManagementPolicy = ""
|
||||||
|
} else {
|
||||||
|
yyv24 := &x.PodManagementPolicy
|
||||||
|
yyv24.CodecDecodeSelf(d)
|
||||||
|
}
|
||||||
for {
|
for {
|
||||||
yyj13++
|
yyj14++
|
||||||
if yyhl13 {
|
if yyhl14 {
|
||||||
yyb13 = yyj13 > l
|
yyb14 = yyj14 > l
|
||||||
} else {
|
} else {
|
||||||
yyb13 = r.CheckBreak()
|
yyb14 = r.CheckBreak()
|
||||||
}
|
}
|
||||||
if yyb13 {
|
if yyb14 {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
z.DecSendContainerState(codecSelfer_containerArrayElem1234)
|
z.DecSendContainerState(codecSelfer_containerArrayElem1234)
|
||||||
z.DecStructFieldNotFound(yyj13-1, "")
|
z.DecStructFieldNotFound(yyj14-1, "")
|
||||||
}
|
}
|
||||||
z.DecSendContainerState(codecSelfer_containerArrayEnd1234)
|
z.DecSendContainerState(codecSelfer_containerArrayEnd1234)
|
||||||
}
|
}
|
||||||
@@ -6166,7 +6232,7 @@ func (x codecSelfer1234) decSliceStatefulSet(v *[]StatefulSet, d *codec1978.Deco
|
|||||||
|
|
||||||
yyrg1 := len(yyv1) > 0
|
yyrg1 := len(yyv1) > 0
|
||||||
yyv21 := yyv1
|
yyv21 := yyv1
|
||||||
yyrl1, yyrt1 = z.DecInferLen(yyl1, z.DecBasicHandle().MaxInitLen, 896)
|
yyrl1, yyrt1 = z.DecInferLen(yyl1, z.DecBasicHandle().MaxInitLen, 912)
|
||||||
if yyrt1 {
|
if yyrt1 {
|
||||||
if yyrl1 <= cap(yyv1) {
|
if yyrl1 <= cap(yyv1) {
|
||||||
yyv1 = yyv1[:yyrl1]
|
yyv1 = yyv1[:yyrl1]
|
||||||
|
@@ -95,9 +95,24 @@ type StatefulSet struct {
|
|||||||
Status StatefulSetStatus `json:"status,omitempty" protobuf:"bytes,3,opt,name=status"`
|
Status StatefulSetStatus `json:"status,omitempty" protobuf:"bytes,3,opt,name=status"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// PodManagementPolicyType defines the policy for creating pods under a stateful set.
|
||||||
|
type PodManagementPolicyType string
|
||||||
|
|
||||||
|
const (
|
||||||
|
// OrderedReadyPodManagement will create pods in strictly increasing order on
|
||||||
|
// scale up and strictly decreasing order on scale down, progressing only when
|
||||||
|
// the previous pod is ready or terminated. At most one pod will be changed
|
||||||
|
// at any time.
|
||||||
|
OrderedReadyPodManagement PodManagementPolicyType = "OrderedReady"
|
||||||
|
// ParallelPodManagement will create and delete pods as soon as the stateful set
|
||||||
|
// replica count is changed, and will not wait for pods to be ready or complete
|
||||||
|
// termination.
|
||||||
|
ParallelPodManagement = "Parallel"
|
||||||
|
)
|
||||||
|
|
||||||
// A StatefulSetSpec is the specification of a StatefulSet.
|
// A StatefulSetSpec is the specification of a StatefulSet.
|
||||||
type StatefulSetSpec struct {
|
type StatefulSetSpec struct {
|
||||||
// Replicas is the desired number of replicas of the given Template.
|
// replicas is the desired number of replicas of the given Template.
|
||||||
// These are replicas in the sense that they are instantiations of the
|
// These are replicas in the sense that they are instantiations of the
|
||||||
// same Template, but individual replicas also have a consistent identity.
|
// same Template, but individual replicas also have a consistent identity.
|
||||||
// If unspecified, defaults to 1.
|
// If unspecified, defaults to 1.
|
||||||
@@ -105,19 +120,19 @@ type StatefulSetSpec struct {
|
|||||||
// +optional
|
// +optional
|
||||||
Replicas *int32 `json:"replicas,omitempty" protobuf:"varint,1,opt,name=replicas"`
|
Replicas *int32 `json:"replicas,omitempty" protobuf:"varint,1,opt,name=replicas"`
|
||||||
|
|
||||||
// Selector is a label query over pods that should match the replica count.
|
// selector is a label query over pods that should match the replica count.
|
||||||
// If empty, defaulted to labels on the pod template.
|
// If empty, defaulted to labels on the pod template.
|
||||||
// More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors
|
// More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors
|
||||||
// +optional
|
// +optional
|
||||||
Selector *metav1.LabelSelector `json:"selector,omitempty" protobuf:"bytes,2,opt,name=selector"`
|
Selector *metav1.LabelSelector `json:"selector,omitempty" protobuf:"bytes,2,opt,name=selector"`
|
||||||
|
|
||||||
// Template is the object that describes the pod that will be created if
|
// template is the object that describes the pod that will be created if
|
||||||
// insufficient replicas are detected. Each pod stamped out by the StatefulSet
|
// insufficient replicas are detected. Each pod stamped out by the StatefulSet
|
||||||
// will fulfill this Template, but have a unique identity from the rest
|
// will fulfill this Template, but have a unique identity from the rest
|
||||||
// of the StatefulSet.
|
// of the StatefulSet.
|
||||||
Template v1.PodTemplateSpec `json:"template" protobuf:"bytes,3,opt,name=template"`
|
Template v1.PodTemplateSpec `json:"template" protobuf:"bytes,3,opt,name=template"`
|
||||||
|
|
||||||
// VolumeClaimTemplates is a list of claims that pods are allowed to reference.
|
// volumeClaimTemplates is a list of claims that pods are allowed to reference.
|
||||||
// The StatefulSet controller is responsible for mapping network identities to
|
// The StatefulSet controller is responsible for mapping network identities to
|
||||||
// claims in a way that maintains the identity of a pod. Every claim in
|
// claims in a way that maintains the identity of a pod. Every claim in
|
||||||
// this list must have at least one matching (by name) volumeMount in one
|
// this list must have at least one matching (by name) volumeMount in one
|
||||||
@@ -127,21 +142,32 @@ type StatefulSetSpec struct {
|
|||||||
// +optional
|
// +optional
|
||||||
VolumeClaimTemplates []v1.PersistentVolumeClaim `json:"volumeClaimTemplates,omitempty" protobuf:"bytes,4,rep,name=volumeClaimTemplates"`
|
VolumeClaimTemplates []v1.PersistentVolumeClaim `json:"volumeClaimTemplates,omitempty" protobuf:"bytes,4,rep,name=volumeClaimTemplates"`
|
||||||
|
|
||||||
// ServiceName is the name of the service that governs this StatefulSet.
|
// serviceName is the name of the service that governs this StatefulSet.
|
||||||
// This service must exist before the StatefulSet, and is responsible for
|
// This service must exist before the StatefulSet, and is responsible for
|
||||||
// the network identity of the set. Pods get DNS/hostnames that follow the
|
// the network identity of the set. Pods get DNS/hostnames that follow the
|
||||||
// pattern: pod-specific-string.serviceName.default.svc.cluster.local
|
// pattern: pod-specific-string.serviceName.default.svc.cluster.local
|
||||||
// where "pod-specific-string" is managed by the StatefulSet controller.
|
// where "pod-specific-string" is managed by the StatefulSet controller.
|
||||||
ServiceName string `json:"serviceName" protobuf:"bytes,5,opt,name=serviceName"`
|
ServiceName string `json:"serviceName" protobuf:"bytes,5,opt,name=serviceName"`
|
||||||
|
|
||||||
|
// podManagementPolicy controls how pods are created during initial scale up,
|
||||||
|
// when replacing pods on nodes, or when scaling down. The default policy is
|
||||||
|
// `OrderedReady`, where pods are created in increasing order (pod-0, then
|
||||||
|
// pod-1, etc) and the controller will wait until each pod is ready before
|
||||||
|
// continuing. When scaling down, the pods are removed in the opposite order.
|
||||||
|
// The alternative policy is `Parallel` which will create pods in parallel
|
||||||
|
// to match the desired scale without waiting, and on scale down will delete
|
||||||
|
// all pods at once.
|
||||||
|
// +optional
|
||||||
|
PodManagementPolicy PodManagementPolicyType `json:"podManagementPolicy,omitempty" protobuf:"bytes,6,opt,name=podManagementPolicy,casttype=PodManagementPolicyType"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// StatefulSetStatus represents the current state of a StatefulSet.
|
// StatefulSetStatus represents the current state of a StatefulSet.
|
||||||
type StatefulSetStatus struct {
|
type StatefulSetStatus struct {
|
||||||
// most recent generation observed by this StatefulSet.
|
// observedGeneration is the most recent generation observed by this StatefulSet.
|
||||||
// +optional
|
// +optional
|
||||||
ObservedGeneration *int64 `json:"observedGeneration,omitempty" protobuf:"varint,1,opt,name=observedGeneration"`
|
ObservedGeneration *int64 `json:"observedGeneration,omitempty" protobuf:"varint,1,opt,name=observedGeneration"`
|
||||||
|
|
||||||
// Replicas is the number of actual replicas.
|
// replicas is the number of actual replicas.
|
||||||
Replicas int32 `json:"replicas" protobuf:"varint,2,opt,name=replicas"`
|
Replicas int32 `json:"replicas" protobuf:"varint,2,opt,name=replicas"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -184,11 +184,12 @@ func (StatefulSetList) SwaggerDoc() map[string]string {
|
|||||||
|
|
||||||
var map_StatefulSetSpec = map[string]string{
|
var map_StatefulSetSpec = map[string]string{
|
||||||
"": "A StatefulSetSpec is the specification of a StatefulSet.",
|
"": "A StatefulSetSpec is the specification of a StatefulSet.",
|
||||||
"replicas": "Replicas is the desired number of replicas of the given Template. These are replicas in the sense that they are instantiations of the same Template, but individual replicas also have a consistent identity. If unspecified, defaults to 1.",
|
"replicas": "replicas is the desired number of replicas of the given Template. These are replicas in the sense that they are instantiations of the same Template, but individual replicas also have a consistent identity. If unspecified, defaults to 1.",
|
||||||
"selector": "Selector is a label query over pods that should match the replica count. If empty, defaulted to labels on the pod template. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors",
|
"selector": "selector is a label query over pods that should match the replica count. If empty, defaulted to labels on the pod template. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors",
|
||||||
"template": "Template is the object that describes the pod that will be created if insufficient replicas are detected. Each pod stamped out by the StatefulSet will fulfill this Template, but have a unique identity from the rest of the StatefulSet.",
|
"template": "template is the object that describes the pod that will be created if insufficient replicas are detected. Each pod stamped out by the StatefulSet will fulfill this Template, but have a unique identity from the rest of the StatefulSet.",
|
||||||
"volumeClaimTemplates": "VolumeClaimTemplates is a list of claims that pods are allowed to reference. The StatefulSet controller is responsible for mapping network identities to claims in a way that maintains the identity of a pod. Every claim in this list must have at least one matching (by name) volumeMount in one container in the template. A claim in this list takes precedence over any volumes in the template, with the same name.",
|
"volumeClaimTemplates": "volumeClaimTemplates is a list of claims that pods are allowed to reference. The StatefulSet controller is responsible for mapping network identities to claims in a way that maintains the identity of a pod. Every claim in this list must have at least one matching (by name) volumeMount in one container in the template. A claim in this list takes precedence over any volumes in the template, with the same name.",
|
||||||
"serviceName": "ServiceName is the name of the service that governs this StatefulSet. This service must exist before the StatefulSet, and is responsible for the network identity of the set. Pods get DNS/hostnames that follow the pattern: pod-specific-string.serviceName.default.svc.cluster.local where \"pod-specific-string\" is managed by the StatefulSet controller.",
|
"serviceName": "serviceName is the name of the service that governs this StatefulSet. This service must exist before the StatefulSet, and is responsible for the network identity of the set. Pods get DNS/hostnames that follow the pattern: pod-specific-string.serviceName.default.svc.cluster.local where \"pod-specific-string\" is managed by the StatefulSet controller.",
|
||||||
|
"podManagementPolicy": "podManagementPolicy controls how pods are created during initial scale up, when replacing pods on nodes, or when scaling down. The default policy is `OrderedReady`, where pods are created in increasing order (pod-0, then pod-1, etc) and the controller will wait until each pod is ready before continuing. When scaling down, the pods are removed in the opposite order. The alternative policy is `Parallel` which will create pods in parallel to match the desired scale without waiting, and on scale down will delete all pods at once.",
|
||||||
}
|
}
|
||||||
|
|
||||||
func (StatefulSetSpec) SwaggerDoc() map[string]string {
|
func (StatefulSetSpec) SwaggerDoc() map[string]string {
|
||||||
@@ -197,8 +198,8 @@ func (StatefulSetSpec) SwaggerDoc() map[string]string {
|
|||||||
|
|
||||||
var map_StatefulSetStatus = map[string]string{
|
var map_StatefulSetStatus = map[string]string{
|
||||||
"": "StatefulSetStatus represents the current state of a StatefulSet.",
|
"": "StatefulSetStatus represents the current state of a StatefulSet.",
|
||||||
"observedGeneration": "most recent generation observed by this StatefulSet.",
|
"observedGeneration": "observedGeneration is the most recent generation observed by this StatefulSet.",
|
||||||
"replicas": "Replicas is the number of actual replicas.",
|
"replicas": "replicas is the number of actual replicas.",
|
||||||
}
|
}
|
||||||
|
|
||||||
func (StatefulSetStatus) SwaggerDoc() map[string]string {
|
func (StatefulSetStatus) SwaggerDoc() map[string]string {
|
||||||
|
@@ -133,6 +133,7 @@ func autoConvert_v1beta1_StatefulSetSpec_To_apps_StatefulSetSpec(in *StatefulSet
|
|||||||
}
|
}
|
||||||
out.VolumeClaimTemplates = *(*[]api.PersistentVolumeClaim)(unsafe.Pointer(&in.VolumeClaimTemplates))
|
out.VolumeClaimTemplates = *(*[]api.PersistentVolumeClaim)(unsafe.Pointer(&in.VolumeClaimTemplates))
|
||||||
out.ServiceName = in.ServiceName
|
out.ServiceName = in.ServiceName
|
||||||
|
out.PodManagementPolicy = apps.PodManagementPolicyType(in.PodManagementPolicy)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -146,6 +147,7 @@ func autoConvert_apps_StatefulSetSpec_To_v1beta1_StatefulSetSpec(in *apps.Statef
|
|||||||
}
|
}
|
||||||
out.VolumeClaimTemplates = *(*[]api_v1.PersistentVolumeClaim)(unsafe.Pointer(&in.VolumeClaimTemplates))
|
out.VolumeClaimTemplates = *(*[]api_v1.PersistentVolumeClaim)(unsafe.Pointer(&in.VolumeClaimTemplates))
|
||||||
out.ServiceName = in.ServiceName
|
out.ServiceName = in.ServiceName
|
||||||
|
out.PodManagementPolicy = PodManagementPolicyType(in.PodManagementPolicy)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -17,6 +17,7 @@ limitations under the License.
|
|||||||
package validation
|
package validation
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"fmt"
|
||||||
"reflect"
|
"reflect"
|
||||||
|
|
||||||
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
||||||
@@ -66,6 +67,14 @@ func ValidatePodTemplateSpecForStatefulSet(template *api.PodTemplateSpec, select
|
|||||||
func ValidateStatefulSetSpec(spec *apps.StatefulSetSpec, fldPath *field.Path) field.ErrorList {
|
func ValidateStatefulSetSpec(spec *apps.StatefulSetSpec, fldPath *field.Path) field.ErrorList {
|
||||||
allErrs := field.ErrorList{}
|
allErrs := field.ErrorList{}
|
||||||
|
|
||||||
|
switch spec.PodManagementPolicy {
|
||||||
|
case "":
|
||||||
|
allErrs = append(allErrs, field.Required(fldPath.Child("podManagementPolicy"), ""))
|
||||||
|
case apps.OrderedReadyPodManagement, apps.ParallelPodManagement:
|
||||||
|
default:
|
||||||
|
allErrs = append(allErrs, field.Invalid(fldPath.Child("podManagementPolicy"), spec.PodManagementPolicy, fmt.Sprintf("must be '%s' or '%s'", apps.OrderedReadyPodManagement, apps.ParallelPodManagement)))
|
||||||
|
}
|
||||||
|
|
||||||
allErrs = append(allErrs, apivalidation.ValidateNonnegativeField(int64(spec.Replicas), fldPath.Child("replicas"))...)
|
allErrs = append(allErrs, apivalidation.ValidateNonnegativeField(int64(spec.Replicas), fldPath.Child("replicas"))...)
|
||||||
if spec.Selector == nil {
|
if spec.Selector == nil {
|
||||||
allErrs = append(allErrs, field.Required(fldPath.Child("selector"), ""))
|
allErrs = append(allErrs, field.Required(fldPath.Child("selector"), ""))
|
||||||
|
@@ -55,6 +55,7 @@ func TestValidateStatefulSet(t *testing.T) {
|
|||||||
{
|
{
|
||||||
ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: metav1.NamespaceDefault},
|
ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: metav1.NamespaceDefault},
|
||||||
Spec: apps.StatefulSetSpec{
|
Spec: apps.StatefulSetSpec{
|
||||||
|
PodManagementPolicy: apps.OrderedReadyPodManagement,
|
||||||
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
||||||
Template: validPodTemplate.Template,
|
Template: validPodTemplate.Template,
|
||||||
},
|
},
|
||||||
@@ -62,6 +63,7 @@ func TestValidateStatefulSet(t *testing.T) {
|
|||||||
{
|
{
|
||||||
ObjectMeta: metav1.ObjectMeta{Name: "abc-123", Namespace: metav1.NamespaceDefault},
|
ObjectMeta: metav1.ObjectMeta{Name: "abc-123", Namespace: metav1.NamespaceDefault},
|
||||||
Spec: apps.StatefulSetSpec{
|
Spec: apps.StatefulSetSpec{
|
||||||
|
PodManagementPolicy: apps.OrderedReadyPodManagement,
|
||||||
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
||||||
Template: validPodTemplate.Template,
|
Template: validPodTemplate.Template,
|
||||||
},
|
},
|
||||||
@@ -77,6 +79,7 @@ func TestValidateStatefulSet(t *testing.T) {
|
|||||||
"zero-length ID": {
|
"zero-length ID": {
|
||||||
ObjectMeta: metav1.ObjectMeta{Name: "", Namespace: metav1.NamespaceDefault},
|
ObjectMeta: metav1.ObjectMeta{Name: "", Namespace: metav1.NamespaceDefault},
|
||||||
Spec: apps.StatefulSetSpec{
|
Spec: apps.StatefulSetSpec{
|
||||||
|
PodManagementPolicy: apps.OrderedReadyPodManagement,
|
||||||
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
||||||
Template: validPodTemplate.Template,
|
Template: validPodTemplate.Template,
|
||||||
},
|
},
|
||||||
@@ -84,6 +87,7 @@ func TestValidateStatefulSet(t *testing.T) {
|
|||||||
"missing-namespace": {
|
"missing-namespace": {
|
||||||
ObjectMeta: metav1.ObjectMeta{Name: "abc-123"},
|
ObjectMeta: metav1.ObjectMeta{Name: "abc-123"},
|
||||||
Spec: apps.StatefulSetSpec{
|
Spec: apps.StatefulSetSpec{
|
||||||
|
PodManagementPolicy: apps.OrderedReadyPodManagement,
|
||||||
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
||||||
Template: validPodTemplate.Template,
|
Template: validPodTemplate.Template,
|
||||||
},
|
},
|
||||||
@@ -91,12 +95,14 @@ func TestValidateStatefulSet(t *testing.T) {
|
|||||||
"empty selector": {
|
"empty selector": {
|
||||||
ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: metav1.NamespaceDefault},
|
ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: metav1.NamespaceDefault},
|
||||||
Spec: apps.StatefulSetSpec{
|
Spec: apps.StatefulSetSpec{
|
||||||
|
PodManagementPolicy: apps.OrderedReadyPodManagement,
|
||||||
Template: validPodTemplate.Template,
|
Template: validPodTemplate.Template,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
"selector_doesnt_match": {
|
"selector_doesnt_match": {
|
||||||
ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: metav1.NamespaceDefault},
|
ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: metav1.NamespaceDefault},
|
||||||
Spec: apps.StatefulSetSpec{
|
Spec: apps.StatefulSetSpec{
|
||||||
|
PodManagementPolicy: apps.OrderedReadyPodManagement,
|
||||||
Selector: &metav1.LabelSelector{MatchLabels: map[string]string{"foo": "bar"}},
|
Selector: &metav1.LabelSelector{MatchLabels: map[string]string{"foo": "bar"}},
|
||||||
Template: validPodTemplate.Template,
|
Template: validPodTemplate.Template,
|
||||||
},
|
},
|
||||||
@@ -104,12 +110,14 @@ func TestValidateStatefulSet(t *testing.T) {
|
|||||||
"invalid manifest": {
|
"invalid manifest": {
|
||||||
ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: metav1.NamespaceDefault},
|
ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: metav1.NamespaceDefault},
|
||||||
Spec: apps.StatefulSetSpec{
|
Spec: apps.StatefulSetSpec{
|
||||||
|
PodManagementPolicy: apps.OrderedReadyPodManagement,
|
||||||
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
"negative_replicas": {
|
"negative_replicas": {
|
||||||
ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: metav1.NamespaceDefault},
|
ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: metav1.NamespaceDefault},
|
||||||
Spec: apps.StatefulSetSpec{
|
Spec: apps.StatefulSetSpec{
|
||||||
|
PodManagementPolicy: apps.OrderedReadyPodManagement,
|
||||||
Replicas: -1,
|
Replicas: -1,
|
||||||
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
||||||
},
|
},
|
||||||
@@ -123,6 +131,7 @@ func TestValidateStatefulSet(t *testing.T) {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
Spec: apps.StatefulSetSpec{
|
Spec: apps.StatefulSetSpec{
|
||||||
|
PodManagementPolicy: apps.OrderedReadyPodManagement,
|
||||||
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
||||||
Template: validPodTemplate.Template,
|
Template: validPodTemplate.Template,
|
||||||
},
|
},
|
||||||
@@ -136,6 +145,7 @@ func TestValidateStatefulSet(t *testing.T) {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
Spec: apps.StatefulSetSpec{
|
Spec: apps.StatefulSetSpec{
|
||||||
|
PodManagementPolicy: apps.OrderedReadyPodManagement,
|
||||||
Template: invalidPodTemplate.Template,
|
Template: invalidPodTemplate.Template,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
@@ -148,6 +158,7 @@ func TestValidateStatefulSet(t *testing.T) {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
Spec: apps.StatefulSetSpec{
|
Spec: apps.StatefulSetSpec{
|
||||||
|
PodManagementPolicy: apps.OrderedReadyPodManagement,
|
||||||
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
||||||
Template: validPodTemplate.Template,
|
Template: validPodTemplate.Template,
|
||||||
},
|
},
|
||||||
@@ -158,6 +169,7 @@ func TestValidateStatefulSet(t *testing.T) {
|
|||||||
Namespace: metav1.NamespaceDefault,
|
Namespace: metav1.NamespaceDefault,
|
||||||
},
|
},
|
||||||
Spec: apps.StatefulSetSpec{
|
Spec: apps.StatefulSetSpec{
|
||||||
|
PodManagementPolicy: apps.OrderedReadyPodManagement,
|
||||||
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
||||||
Template: api.PodTemplateSpec{
|
Template: api.PodTemplateSpec{
|
||||||
Spec: api.PodSpec{
|
Spec: api.PodSpec{
|
||||||
@@ -177,6 +189,7 @@ func TestValidateStatefulSet(t *testing.T) {
|
|||||||
Namespace: metav1.NamespaceDefault,
|
Namespace: metav1.NamespaceDefault,
|
||||||
},
|
},
|
||||||
Spec: apps.StatefulSetSpec{
|
Spec: apps.StatefulSetSpec{
|
||||||
|
PodManagementPolicy: apps.OrderedReadyPodManagement,
|
||||||
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
||||||
Template: api.PodTemplateSpec{
|
Template: api.PodTemplateSpec{
|
||||||
Spec: api.PodSpec{
|
Spec: api.PodSpec{
|
||||||
@@ -263,6 +276,7 @@ func TestValidateStatefulSetUpdate(t *testing.T) {
|
|||||||
old: apps.StatefulSet{
|
old: apps.StatefulSet{
|
||||||
ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: metav1.NamespaceDefault},
|
ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: metav1.NamespaceDefault},
|
||||||
Spec: apps.StatefulSetSpec{
|
Spec: apps.StatefulSetSpec{
|
||||||
|
PodManagementPolicy: apps.OrderedReadyPodManagement,
|
||||||
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
||||||
Template: validPodTemplate.Template,
|
Template: validPodTemplate.Template,
|
||||||
},
|
},
|
||||||
@@ -270,6 +284,7 @@ func TestValidateStatefulSetUpdate(t *testing.T) {
|
|||||||
update: apps.StatefulSet{
|
update: apps.StatefulSet{
|
||||||
ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: metav1.NamespaceDefault},
|
ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: metav1.NamespaceDefault},
|
||||||
Spec: apps.StatefulSetSpec{
|
Spec: apps.StatefulSetSpec{
|
||||||
|
PodManagementPolicy: apps.OrderedReadyPodManagement,
|
||||||
Replicas: 3,
|
Replicas: 3,
|
||||||
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
||||||
Template: validPodTemplate.Template,
|
Template: validPodTemplate.Template,
|
||||||
@@ -296,12 +311,48 @@ func TestValidateStatefulSetUpdate(t *testing.T) {
|
|||||||
update: apps.StatefulSet{
|
update: apps.StatefulSet{
|
||||||
ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: metav1.NamespaceDefault},
|
ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: metav1.NamespaceDefault},
|
||||||
Spec: apps.StatefulSetSpec{
|
Spec: apps.StatefulSetSpec{
|
||||||
|
PodManagementPolicy: apps.OrderedReadyPodManagement,
|
||||||
Replicas: 2,
|
Replicas: 2,
|
||||||
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
||||||
Template: readWriteVolumePodTemplate.Template,
|
Template: readWriteVolumePodTemplate.Template,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
"empty pod creation policy": {
|
||||||
|
old: apps.StatefulSet{
|
||||||
|
ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: metav1.NamespaceDefault},
|
||||||
|
Spec: apps.StatefulSetSpec{
|
||||||
|
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
||||||
|
Template: validPodTemplate.Template,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
update: apps.StatefulSet{
|
||||||
|
ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: metav1.NamespaceDefault},
|
||||||
|
Spec: apps.StatefulSetSpec{
|
||||||
|
Replicas: 3,
|
||||||
|
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
||||||
|
Template: validPodTemplate.Template,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"invalid pod creation policy": {
|
||||||
|
old: apps.StatefulSet{
|
||||||
|
ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: metav1.NamespaceDefault},
|
||||||
|
Spec: apps.StatefulSetSpec{
|
||||||
|
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
||||||
|
Template: validPodTemplate.Template,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
update: apps.StatefulSet{
|
||||||
|
ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: metav1.NamespaceDefault},
|
||||||
|
Spec: apps.StatefulSetSpec{
|
||||||
|
PodManagementPolicy: apps.PodManagementPolicyType("Other"),
|
||||||
|
Replicas: 3,
|
||||||
|
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
||||||
|
Template: validPodTemplate.Template,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
"updates to a field other than spec.Replicas": {
|
"updates to a field other than spec.Replicas": {
|
||||||
old: apps.StatefulSet{
|
old: apps.StatefulSet{
|
||||||
ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: metav1.NamespaceDefault},
|
ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: metav1.NamespaceDefault},
|
||||||
@@ -313,6 +364,7 @@ func TestValidateStatefulSetUpdate(t *testing.T) {
|
|||||||
update: apps.StatefulSet{
|
update: apps.StatefulSet{
|
||||||
ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: metav1.NamespaceDefault},
|
ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: metav1.NamespaceDefault},
|
||||||
Spec: apps.StatefulSetSpec{
|
Spec: apps.StatefulSetSpec{
|
||||||
|
PodManagementPolicy: apps.OrderedReadyPodManagement,
|
||||||
Replicas: 1,
|
Replicas: 1,
|
||||||
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
||||||
Template: readWriteVolumePodTemplate.Template,
|
Template: readWriteVolumePodTemplate.Template,
|
||||||
@@ -330,6 +382,7 @@ func TestValidateStatefulSetUpdate(t *testing.T) {
|
|||||||
update: apps.StatefulSet{
|
update: apps.StatefulSet{
|
||||||
ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: metav1.NamespaceDefault},
|
ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: metav1.NamespaceDefault},
|
||||||
Spec: apps.StatefulSetSpec{
|
Spec: apps.StatefulSetSpec{
|
||||||
|
PodManagementPolicy: apps.OrderedReadyPodManagement,
|
||||||
Replicas: 2,
|
Replicas: 2,
|
||||||
Selector: &metav1.LabelSelector{MatchLabels: invalidLabels},
|
Selector: &metav1.LabelSelector{MatchLabels: invalidLabels},
|
||||||
Template: validPodTemplate.Template,
|
Template: validPodTemplate.Template,
|
||||||
@@ -340,6 +393,7 @@ func TestValidateStatefulSetUpdate(t *testing.T) {
|
|||||||
old: apps.StatefulSet{
|
old: apps.StatefulSet{
|
||||||
ObjectMeta: metav1.ObjectMeta{Name: "", Namespace: metav1.NamespaceDefault},
|
ObjectMeta: metav1.ObjectMeta{Name: "", Namespace: metav1.NamespaceDefault},
|
||||||
Spec: apps.StatefulSetSpec{
|
Spec: apps.StatefulSetSpec{
|
||||||
|
PodManagementPolicy: apps.OrderedReadyPodManagement,
|
||||||
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
||||||
Template: validPodTemplate.Template,
|
Template: validPodTemplate.Template,
|
||||||
},
|
},
|
||||||
@@ -364,6 +418,7 @@ func TestValidateStatefulSetUpdate(t *testing.T) {
|
|||||||
update: apps.StatefulSet{
|
update: apps.StatefulSet{
|
||||||
ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: metav1.NamespaceDefault},
|
ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: metav1.NamespaceDefault},
|
||||||
Spec: apps.StatefulSetSpec{
|
Spec: apps.StatefulSetSpec{
|
||||||
|
PodManagementPolicy: apps.OrderedReadyPodManagement,
|
||||||
Replicas: -1,
|
Replicas: -1,
|
||||||
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
Selector: &metav1.LabelSelector{MatchLabels: validLabels},
|
||||||
Template: validPodTemplate.Template,
|
Template: validPodTemplate.Template,
|
||||||
|
@@ -56,9 +56,11 @@ go_test(
|
|||||||
library = ":go_default_library",
|
library = ":go_default_library",
|
||||||
tags = ["automanaged"],
|
tags = ["automanaged"],
|
||||||
deps = [
|
deps = [
|
||||||
|
"//pkg/api:go_default_library",
|
||||||
"//pkg/api/v1:go_default_library",
|
"//pkg/api/v1:go_default_library",
|
||||||
"//pkg/api/v1/pod:go_default_library",
|
"//pkg/api/v1/pod:go_default_library",
|
||||||
"//pkg/apis/apps/v1beta1:go_default_library",
|
"//pkg/apis/apps/v1beta1:go_default_library",
|
||||||
|
"//pkg/client/clientset_generated/clientset:go_default_library",
|
||||||
"//pkg/client/clientset_generated/clientset/fake:go_default_library",
|
"//pkg/client/clientset_generated/clientset/fake:go_default_library",
|
||||||
"//pkg/client/informers/informers_generated/externalversions:go_default_library",
|
"//pkg/client/informers/informers_generated/externalversions:go_default_library",
|
||||||
"//pkg/client/informers/informers_generated/externalversions/apps/v1beta1:go_default_library",
|
"//pkg/client/informers/informers_generated/externalversions/apps/v1beta1:go_default_library",
|
||||||
|
@@ -50,6 +50,12 @@ type defaultStatefulSetControl struct {
|
|||||||
podControl StatefulPodControlInterface
|
podControl StatefulPodControlInterface
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// UpdateStatefulSet executes the core logic loop for a stateful set, applying the predictable and
|
||||||
|
// consistent monotonic update strategy by default - scale up proceeds in ordinal order, no new pod
|
||||||
|
// is created while any pod is unhealthy, and pods are terminated in descending order. The burst
|
||||||
|
// strategy allows these constraints to be relaxed - pods will be created and deleted eagerly and
|
||||||
|
// in no particular order. Clients using the burst strategy should be careful to ensure they
|
||||||
|
// understand the consistency implications of having unpredictable numbers of pods available.
|
||||||
func (ssc *defaultStatefulSetControl) UpdateStatefulSet(set *apps.StatefulSet, pods []*v1.Pod) error {
|
func (ssc *defaultStatefulSetControl) UpdateStatefulSet(set *apps.StatefulSet, pods []*v1.Pod) error {
|
||||||
replicaCount := int(*set.Spec.Replicas)
|
replicaCount := int(*set.Spec.Replicas)
|
||||||
// slice that will contain all Pods such that 0 <= getOrdinal(pod) < set.Spec.Replicas
|
// slice that will contain all Pods such that 0 <= getOrdinal(pod) < set.Spec.Replicas
|
||||||
@@ -118,6 +124,8 @@ func (ssc *defaultStatefulSetControl) UpdateStatefulSet(set *apps.StatefulSet, p
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
monotonic := !allowsBurst(set)
|
||||||
|
|
||||||
// Examine each replica with respect to its ordinal
|
// Examine each replica with respect to its ordinal
|
||||||
for i := range replicas {
|
for i := range replicas {
|
||||||
// delete and recreate failed pods
|
// delete and recreate failed pods
|
||||||
@@ -128,23 +136,29 @@ func (ssc *defaultStatefulSetControl) UpdateStatefulSet(set *apps.StatefulSet, p
|
|||||||
}
|
}
|
||||||
replicas[i] = newStatefulSetPod(set, i)
|
replicas[i] = newStatefulSetPod(set, i)
|
||||||
}
|
}
|
||||||
// If we find a Pod that has not been created we create the Pod immediately and return
|
// If we find a Pod that has not been created we create the Pod
|
||||||
if !isCreated(replicas[i]) {
|
if !isCreated(replicas[i]) {
|
||||||
return ssc.podControl.CreateStatefulPod(set, replicas[i])
|
if err := ssc.podControl.CreateStatefulPod(set, replicas[i]); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
// if the set does not allow bursting, return immediately
|
||||||
|
if monotonic {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
// pod created, no more work possible for this round
|
||||||
|
continue
|
||||||
}
|
}
|
||||||
// If we find a Pod that is currently terminating, we must wait until graceful deletion
|
// If we find a Pod that is currently terminating, we must wait until graceful deletion
|
||||||
// completes before we continue to make progress.
|
// completes before we continue to make progress.
|
||||||
if isTerminating(replicas[i]) {
|
if isTerminating(replicas[i]) && monotonic {
|
||||||
glog.V(2).Infof("StatefulSet %s is waiting for Pod %s to Terminate",
|
glog.V(2).Infof("StatefulSet %s is waiting for Pod %s to Terminate", set.Name, replicas[i].Name)
|
||||||
set.Name, replicas[i].Name)
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
// If we have a Pod that has been created but is not running and ready we can not make progress.
|
// If we have a Pod that has been created but is not running and ready we can not make progress.
|
||||||
// We must ensure that all for each Pod, when we create it, all of its predecessors, with respect to its
|
// We must ensure that all for each Pod, when we create it, all of its predecessors, with respect to its
|
||||||
// ordinal, are Running and Ready.
|
// ordinal, are Running and Ready.
|
||||||
if !isRunningAndReady(replicas[i]) {
|
if !isRunningAndReady(replicas[i]) && monotonic {
|
||||||
glog.V(2).Infof("StatefulSet %s is waiting for Pod %s to be Running and Ready",
|
glog.V(2).Infof("StatefulSet %s is waiting for Pod %s to be Running and Ready", set.Name, replicas[i].Name)
|
||||||
set.Name, replicas[i].Name)
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
// Enforce the StatefulSet invariants
|
// Enforce the StatefulSet invariants
|
||||||
@@ -166,13 +180,18 @@ func (ssc *defaultStatefulSetControl) UpdateStatefulSet(set *apps.StatefulSet, p
|
|||||||
// We will wait for all predecessors to be Running and Ready prior to attempting a deletion.
|
// We will wait for all predecessors to be Running and Ready prior to attempting a deletion.
|
||||||
// We will terminate Pods in a monotonically decreasing order over [len(pods),set.Spec.Replicas).
|
// We will terminate Pods in a monotonically decreasing order over [len(pods),set.Spec.Replicas).
|
||||||
// Note that we do not resurrect Pods in this interval.
|
// Note that we do not resurrect Pods in this interval.
|
||||||
if unhealthy > 0 {
|
if unhealthy > 0 && monotonic {
|
||||||
glog.V(2).Infof("StatefulSet %s is waiting on %d Pods", set.Name, unhealthy)
|
glog.V(2).Infof("StatefulSet %s is waiting on %d Pods", set.Name, unhealthy)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
if target := len(condemned) - 1; target >= 0 {
|
for target := len(condemned) - 1; target >= 0; target-- {
|
||||||
glog.V(2).Infof("StatefulSet %s terminating Pod %s", set.Name, condemned[target])
|
glog.V(2).Infof("StatefulSet %s terminating Pod %s", set.Name, condemned[target])
|
||||||
return ssc.podControl.DeleteStatefulPod(set, condemned[target])
|
if err := ssc.podControl.DeleteStatefulPod(set, condemned[target]); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if monotonic {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
@@ -19,8 +19,12 @@ package statefulset
|
|||||||
import (
|
import (
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"math/rand"
|
||||||
|
"reflect"
|
||||||
|
"runtime"
|
||||||
"sort"
|
"sort"
|
||||||
"strconv"
|
"strconv"
|
||||||
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
@@ -28,9 +32,11 @@ import (
|
|||||||
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
||||||
"k8s.io/apimachinery/pkg/labels"
|
"k8s.io/apimachinery/pkg/labels"
|
||||||
"k8s.io/client-go/tools/cache"
|
"k8s.io/client-go/tools/cache"
|
||||||
|
"k8s.io/kubernetes/pkg/api"
|
||||||
"k8s.io/kubernetes/pkg/api/v1"
|
"k8s.io/kubernetes/pkg/api/v1"
|
||||||
podutil "k8s.io/kubernetes/pkg/api/v1/pod"
|
podutil "k8s.io/kubernetes/pkg/api/v1/pod"
|
||||||
apps "k8s.io/kubernetes/pkg/apis/apps/v1beta1"
|
apps "k8s.io/kubernetes/pkg/apis/apps/v1beta1"
|
||||||
|
"k8s.io/kubernetes/pkg/client/clientset_generated/clientset"
|
||||||
"k8s.io/kubernetes/pkg/client/clientset_generated/clientset/fake"
|
"k8s.io/kubernetes/pkg/client/clientset_generated/clientset/fake"
|
||||||
informers "k8s.io/kubernetes/pkg/client/informers/informers_generated/externalversions"
|
informers "k8s.io/kubernetes/pkg/client/informers/informers_generated/externalversions"
|
||||||
appsinformers "k8s.io/kubernetes/pkg/client/informers/informers_generated/externalversions/apps/v1beta1"
|
appsinformers "k8s.io/kubernetes/pkg/client/informers/informers_generated/externalversions/apps/v1beta1"
|
||||||
@@ -40,24 +46,75 @@ import (
|
|||||||
"k8s.io/kubernetes/pkg/controller"
|
"k8s.io/kubernetes/pkg/controller"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestDefaultStatefulSetControlCreatesPods(t *testing.T) {
|
type invariantFunc func(set *apps.StatefulSet, spc *fakeStatefulPodControl) error
|
||||||
set := newStatefulSet(3)
|
|
||||||
client := fake.NewSimpleClientset(set)
|
|
||||||
|
|
||||||
|
func setupController(client clientset.Interface) (*fakeStatefulPodControl, StatefulSetControlInterface, chan struct{}) {
|
||||||
informerFactory := informers.NewSharedInformerFactory(client, controller.NoResyncPeriodFunc())
|
informerFactory := informers.NewSharedInformerFactory(client, controller.NoResyncPeriodFunc())
|
||||||
spc := newFakeStatefulPodControl(informerFactory.Core().V1().Pods(), informerFactory.Apps().V1beta1().StatefulSets())
|
spc := newFakeStatefulPodControl(informerFactory.Core().V1().Pods(), informerFactory.Apps().V1beta1().StatefulSets())
|
||||||
ssc := NewDefaultStatefulSetControl(spc)
|
ssc := NewDefaultStatefulSetControl(spc)
|
||||||
|
|
||||||
stop := make(chan struct{})
|
stop := make(chan struct{})
|
||||||
defer close(stop)
|
|
||||||
informerFactory.Start(stop)
|
informerFactory.Start(stop)
|
||||||
cache.WaitForCacheSync(
|
cache.WaitForCacheSync(
|
||||||
stop,
|
stop,
|
||||||
informerFactory.Apps().V1beta1().StatefulSets().Informer().HasSynced,
|
informerFactory.Apps().V1beta1().StatefulSets().Informer().HasSynced,
|
||||||
informerFactory.Core().V1().Pods().Informer().HasSynced,
|
informerFactory.Core().V1().Pods().Informer().HasSynced,
|
||||||
)
|
)
|
||||||
|
return spc, ssc, stop
|
||||||
|
}
|
||||||
|
|
||||||
if err := scaleUpStatefulSetControl(set, ssc, spc); err != nil {
|
func burst(set *apps.StatefulSet) *apps.StatefulSet {
|
||||||
|
set.Spec.PodManagementPolicy = apps.ParallelPodManagement
|
||||||
|
return set
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestStatefulSetControl(t *testing.T) {
|
||||||
|
simpleSetFn := func() *apps.StatefulSet { return newStatefulSet(3) }
|
||||||
|
largeSetFn := func() *apps.StatefulSet { return newStatefulSet(5) }
|
||||||
|
|
||||||
|
testCases := []struct {
|
||||||
|
fn func(*testing.T, *apps.StatefulSet, invariantFunc)
|
||||||
|
obj func() *apps.StatefulSet
|
||||||
|
}{
|
||||||
|
{CreatesPods, simpleSetFn},
|
||||||
|
{ScalesUp, simpleSetFn},
|
||||||
|
{ScalesDown, simpleSetFn},
|
||||||
|
{ReplacesPods, largeSetFn},
|
||||||
|
{RecreatesFailedPod, simpleSetFn},
|
||||||
|
{SetsInitAnnotation, simpleSetFn},
|
||||||
|
{CreatePodFailure, simpleSetFn},
|
||||||
|
{UpdatePodFailure, simpleSetFn},
|
||||||
|
{UpdateSetStatusFailure, simpleSetFn},
|
||||||
|
{PodRecreateDeleteFailure, simpleSetFn},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, testCase := range testCases {
|
||||||
|
fnName := runtime.FuncForPC(reflect.ValueOf(testCase.fn).Pointer()).Name()
|
||||||
|
if i := strings.LastIndex(fnName, "."); i != -1 {
|
||||||
|
fnName = fnName[i+1:]
|
||||||
|
}
|
||||||
|
t.Run(
|
||||||
|
fmt.Sprintf("%s/Monotonic", fnName),
|
||||||
|
func(t *testing.T) {
|
||||||
|
testCase.fn(t, testCase.obj(), assertMonotonicInvariants)
|
||||||
|
},
|
||||||
|
)
|
||||||
|
t.Run(
|
||||||
|
fmt.Sprintf("%s/Burst", fnName),
|
||||||
|
func(t *testing.T) {
|
||||||
|
set := burst(testCase.obj())
|
||||||
|
testCase.fn(t, set, assertBurstInvariants)
|
||||||
|
},
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func CreatesPods(t *testing.T, set *apps.StatefulSet, invariants invariantFunc) {
|
||||||
|
client := fake.NewSimpleClientset(set)
|
||||||
|
spc, ssc, stop := setupController(client)
|
||||||
|
defer close(stop)
|
||||||
|
|
||||||
|
if err := scaleUpStatefulSetControl(t, set, ssc, spc, invariants); err != nil {
|
||||||
t.Errorf("Failed to turn up StatefulSet : %s", err)
|
t.Errorf("Failed to turn up StatefulSet : %s", err)
|
||||||
}
|
}
|
||||||
var err error
|
var err error
|
||||||
@@ -70,28 +127,16 @@ func TestDefaultStatefulSetControlCreatesPods(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestStatefulSetControlScaleUp(t *testing.T) {
|
func ScalesUp(t *testing.T, set *apps.StatefulSet, invariants invariantFunc) {
|
||||||
set := newStatefulSet(3)
|
|
||||||
client := fake.NewSimpleClientset(set)
|
client := fake.NewSimpleClientset(set)
|
||||||
|
spc, ssc, stop := setupController(client)
|
||||||
informerFactory := informers.NewSharedInformerFactory(client, controller.NoResyncPeriodFunc())
|
|
||||||
spc := newFakeStatefulPodControl(informerFactory.Core().V1().Pods(), informerFactory.Apps().V1beta1().StatefulSets())
|
|
||||||
ssc := NewDefaultStatefulSetControl(spc)
|
|
||||||
|
|
||||||
stop := make(chan struct{})
|
|
||||||
defer close(stop)
|
defer close(stop)
|
||||||
informerFactory.Start(stop)
|
|
||||||
cache.WaitForCacheSync(
|
|
||||||
stop,
|
|
||||||
informerFactory.Apps().V1beta1().StatefulSets().Informer().HasSynced,
|
|
||||||
informerFactory.Core().V1().Pods().Informer().HasSynced,
|
|
||||||
)
|
|
||||||
|
|
||||||
if err := scaleUpStatefulSetControl(set, ssc, spc); err != nil {
|
if err := scaleUpStatefulSetControl(t, set, ssc, spc, invariants); err != nil {
|
||||||
t.Errorf("Failed to turn up StatefulSet : %s", err)
|
t.Errorf("Failed to turn up StatefulSet : %s", err)
|
||||||
}
|
}
|
||||||
*set.Spec.Replicas = 4
|
*set.Spec.Replicas = 4
|
||||||
if err := scaleUpStatefulSetControl(set, ssc, spc); err != nil {
|
if err := scaleUpStatefulSetControl(t, set, ssc, spc, invariants); err != nil {
|
||||||
t.Errorf("Failed to scale StatefulSet : %s", err)
|
t.Errorf("Failed to scale StatefulSet : %s", err)
|
||||||
}
|
}
|
||||||
var err error
|
var err error
|
||||||
@@ -104,28 +149,16 @@ func TestStatefulSetControlScaleUp(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestStatefulSetControlScaleDown(t *testing.T) {
|
func ScalesDown(t *testing.T, set *apps.StatefulSet, invariants invariantFunc) {
|
||||||
set := newStatefulSet(3)
|
|
||||||
client := fake.NewSimpleClientset(set)
|
client := fake.NewSimpleClientset(set)
|
||||||
|
spc, ssc, stop := setupController(client)
|
||||||
informerFactory := informers.NewSharedInformerFactory(client, controller.NoResyncPeriodFunc())
|
|
||||||
spc := newFakeStatefulPodControl(informerFactory.Core().V1().Pods(), informerFactory.Apps().V1beta1().StatefulSets())
|
|
||||||
ssc := NewDefaultStatefulSetControl(spc)
|
|
||||||
|
|
||||||
stop := make(chan struct{})
|
|
||||||
defer close(stop)
|
defer close(stop)
|
||||||
informerFactory.Start(stop)
|
|
||||||
cache.WaitForCacheSync(
|
|
||||||
stop,
|
|
||||||
informerFactory.Apps().V1beta1().StatefulSets().Informer().HasSynced,
|
|
||||||
informerFactory.Core().V1().Pods().Informer().HasSynced,
|
|
||||||
)
|
|
||||||
|
|
||||||
if err := scaleUpStatefulSetControl(set, ssc, spc); err != nil {
|
if err := scaleUpStatefulSetControl(t, set, ssc, spc, invariants); err != nil {
|
||||||
t.Errorf("Failed to turn up StatefulSet : %s", err)
|
t.Errorf("Failed to turn up StatefulSet : %s", err)
|
||||||
}
|
}
|
||||||
*set.Spec.Replicas = 0
|
*set.Spec.Replicas = 0
|
||||||
if err := scaleDownStatefulSetControl(set, ssc, spc); err != nil {
|
if err := scaleDownStatefulSetControl(t, set, ssc, spc, invariants); err != nil {
|
||||||
t.Errorf("Failed to scale StatefulSet : %s", err)
|
t.Errorf("Failed to scale StatefulSet : %s", err)
|
||||||
}
|
}
|
||||||
if set.Status.Replicas != 0 {
|
if set.Status.Replicas != 0 {
|
||||||
@@ -133,24 +166,12 @@ func TestStatefulSetControlScaleDown(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestStatefulSetControlReplacesPods(t *testing.T) {
|
func ReplacesPods(t *testing.T, set *apps.StatefulSet, invariants invariantFunc) {
|
||||||
set := newStatefulSet(5)
|
|
||||||
client := fake.NewSimpleClientset(set)
|
client := fake.NewSimpleClientset(set)
|
||||||
|
spc, ssc, stop := setupController(client)
|
||||||
informerFactory := informers.NewSharedInformerFactory(client, controller.NoResyncPeriodFunc())
|
|
||||||
spc := newFakeStatefulPodControl(informerFactory.Core().V1().Pods(), informerFactory.Apps().V1beta1().StatefulSets())
|
|
||||||
ssc := NewDefaultStatefulSetControl(spc)
|
|
||||||
|
|
||||||
stop := make(chan struct{})
|
|
||||||
defer close(stop)
|
defer close(stop)
|
||||||
informerFactory.Start(stop)
|
|
||||||
cache.WaitForCacheSync(
|
|
||||||
stop,
|
|
||||||
informerFactory.Apps().V1beta1().StatefulSets().Informer().HasSynced,
|
|
||||||
informerFactory.Core().V1().Pods().Informer().HasSynced,
|
|
||||||
)
|
|
||||||
|
|
||||||
if err := scaleUpStatefulSetControl(set, ssc, spc); err != nil {
|
if err := scaleUpStatefulSetControl(t, set, ssc, spc, invariants); err != nil {
|
||||||
t.Errorf("Failed to turn up StatefulSet : %s", err)
|
t.Errorf("Failed to turn up StatefulSet : %s", err)
|
||||||
}
|
}
|
||||||
var err error
|
var err error
|
||||||
@@ -215,12 +236,11 @@ func TestStatefulSetControlReplacesPods(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestDefaultStatefulSetControlRecreatesFailedPod(t *testing.T) {
|
func RecreatesFailedPod(t *testing.T, set *apps.StatefulSet, invariants invariantFunc) {
|
||||||
client := fake.NewSimpleClientset()
|
client := fake.NewSimpleClientset()
|
||||||
informerFactory := informers.NewSharedInformerFactory(client, controller.NoResyncPeriodFunc())
|
informerFactory := informers.NewSharedInformerFactory(client, controller.NoResyncPeriodFunc())
|
||||||
spc := newFakeStatefulPodControl(informerFactory.Core().V1().Pods(), informerFactory.Apps().V1beta1().StatefulSets())
|
spc := newFakeStatefulPodControl(informerFactory.Core().V1().Pods(), informerFactory.Apps().V1beta1().StatefulSets())
|
||||||
ssc := NewDefaultStatefulSetControl(spc)
|
ssc := NewDefaultStatefulSetControl(spc)
|
||||||
set := newStatefulSet(3)
|
|
||||||
selector, err := metav1.LabelSelectorAsSelector(set.Spec.Selector)
|
selector, err := metav1.LabelSelectorAsSelector(set.Spec.Selector)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Error(err)
|
t.Error(err)
|
||||||
@@ -232,7 +252,7 @@ func TestDefaultStatefulSetControlRecreatesFailedPod(t *testing.T) {
|
|||||||
if err := ssc.UpdateStatefulSet(set, pods); err != nil {
|
if err := ssc.UpdateStatefulSet(set, pods); err != nil {
|
||||||
t.Errorf("Error updating StatefulSet %s", err)
|
t.Errorf("Error updating StatefulSet %s", err)
|
||||||
}
|
}
|
||||||
if err := assertInvariants(set, spc); err != nil {
|
if err := invariants(set, spc); err != nil {
|
||||||
t.Error(err)
|
t.Error(err)
|
||||||
}
|
}
|
||||||
pods, err = spc.podsLister.Pods(set.Namespace).List(selector)
|
pods, err = spc.podsLister.Pods(set.Namespace).List(selector)
|
||||||
@@ -244,7 +264,7 @@ func TestDefaultStatefulSetControlRecreatesFailedPod(t *testing.T) {
|
|||||||
if err := ssc.UpdateStatefulSet(set, pods); err != nil {
|
if err := ssc.UpdateStatefulSet(set, pods); err != nil {
|
||||||
t.Errorf("Error updating StatefulSet %s", err)
|
t.Errorf("Error updating StatefulSet %s", err)
|
||||||
}
|
}
|
||||||
if err := assertInvariants(set, spc); err != nil {
|
if err := invariants(set, spc); err != nil {
|
||||||
t.Error(err)
|
t.Error(err)
|
||||||
}
|
}
|
||||||
pods, err = spc.podsLister.Pods(set.Namespace).List(selector)
|
pods, err = spc.podsLister.Pods(set.Namespace).List(selector)
|
||||||
@@ -256,22 +276,10 @@ func TestDefaultStatefulSetControlRecreatesFailedPod(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestDefaultStatefulSetControlInitAnnotation(t *testing.T) {
|
func SetsInitAnnotation(t *testing.T, set *apps.StatefulSet, invariants invariantFunc) {
|
||||||
set := newStatefulSet(3)
|
|
||||||
client := fake.NewSimpleClientset(set)
|
client := fake.NewSimpleClientset(set)
|
||||||
|
spc, ssc, stop := setupController(client)
|
||||||
informerFactory := informers.NewSharedInformerFactory(client, controller.NoResyncPeriodFunc())
|
|
||||||
spc := newFakeStatefulPodControl(informerFactory.Core().V1().Pods(), informerFactory.Apps().V1beta1().StatefulSets())
|
|
||||||
ssc := NewDefaultStatefulSetControl(spc)
|
|
||||||
|
|
||||||
stop := make(chan struct{})
|
|
||||||
defer close(stop)
|
defer close(stop)
|
||||||
informerFactory.Start(stop)
|
|
||||||
cache.WaitForCacheSync(
|
|
||||||
stop,
|
|
||||||
informerFactory.Apps().V1beta1().StatefulSets().Informer().HasSynced,
|
|
||||||
informerFactory.Core().V1().Pods().Informer().HasSynced,
|
|
||||||
)
|
|
||||||
|
|
||||||
selector, err := metav1.LabelSelectorAsSelector(set.Spec.Selector)
|
selector, err := metav1.LabelSelectorAsSelector(set.Spec.Selector)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -284,7 +292,7 @@ func TestDefaultStatefulSetControlInitAnnotation(t *testing.T) {
|
|||||||
if err = ssc.UpdateStatefulSet(set, pods); err != nil {
|
if err = ssc.UpdateStatefulSet(set, pods); err != nil {
|
||||||
t.Errorf("Error updating StatefulSet %s", err)
|
t.Errorf("Error updating StatefulSet %s", err)
|
||||||
}
|
}
|
||||||
if err = assertInvariants(set, spc); err != nil {
|
if err = invariants(set, spc); err != nil {
|
||||||
t.Error(err)
|
t.Error(err)
|
||||||
}
|
}
|
||||||
if pods, err = spc.setPodRunning(set, 0); err != nil {
|
if pods, err = spc.setPodRunning(set, 0); err != nil {
|
||||||
@@ -300,7 +308,7 @@ func TestDefaultStatefulSetControlInitAnnotation(t *testing.T) {
|
|||||||
if err := ssc.UpdateStatefulSet(set, pods); err != nil {
|
if err := ssc.UpdateStatefulSet(set, pods); err != nil {
|
||||||
t.Errorf("Error updating StatefulSet %s", err)
|
t.Errorf("Error updating StatefulSet %s", err)
|
||||||
}
|
}
|
||||||
if err := assertInvariants(set, spc); err != nil {
|
if err := invariants(set, spc); err != nil {
|
||||||
t.Error(err)
|
t.Error(err)
|
||||||
}
|
}
|
||||||
if replicas != int(set.Status.Replicas) {
|
if replicas != int(set.Status.Replicas) {
|
||||||
@@ -309,7 +317,7 @@ func TestDefaultStatefulSetControlInitAnnotation(t *testing.T) {
|
|||||||
if pods, err = spc.setPodInitStatus(set, 0, true); err != nil {
|
if pods, err = spc.setPodInitStatus(set, 0, true); err != nil {
|
||||||
t.Error(err)
|
t.Error(err)
|
||||||
}
|
}
|
||||||
if err := scaleUpStatefulSetControl(set, ssc, spc); err != nil {
|
if err := scaleUpStatefulSetControl(t, set, ssc, spc, invariants); err != nil {
|
||||||
t.Errorf("Failed to turn up StatefulSet : %s", err)
|
t.Errorf("Failed to turn up StatefulSet : %s", err)
|
||||||
}
|
}
|
||||||
set, err = spc.setsLister.StatefulSets(set.Namespace).Get(set.Name)
|
set, err = spc.setsLister.StatefulSets(set.Namespace).Get(set.Name)
|
||||||
@@ -321,28 +329,16 @@ func TestDefaultStatefulSetControlInitAnnotation(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestDefaultStatefulSetControlCreatePodFailure(t *testing.T) {
|
func CreatePodFailure(t *testing.T, set *apps.StatefulSet, invariants invariantFunc) {
|
||||||
set := newStatefulSet(3)
|
|
||||||
client := fake.NewSimpleClientset(set)
|
client := fake.NewSimpleClientset(set)
|
||||||
|
spc, ssc, stop := setupController(client)
|
||||||
informerFactory := informers.NewSharedInformerFactory(client, controller.NoResyncPeriodFunc())
|
defer close(stop)
|
||||||
spc := newFakeStatefulPodControl(informerFactory.Core().V1().Pods(), informerFactory.Apps().V1beta1().StatefulSets())
|
|
||||||
ssc := NewDefaultStatefulSetControl(spc)
|
|
||||||
spc.SetCreateStatefulPodError(apierrors.NewInternalError(errors.New("API server failed")), 2)
|
spc.SetCreateStatefulPodError(apierrors.NewInternalError(errors.New("API server failed")), 2)
|
||||||
|
|
||||||
stop := make(chan struct{})
|
if err := scaleUpStatefulSetControl(t, set, ssc, spc, invariants); !apierrors.IsInternalError(err) {
|
||||||
defer close(stop)
|
|
||||||
informerFactory.Start(stop)
|
|
||||||
cache.WaitForCacheSync(
|
|
||||||
stop,
|
|
||||||
informerFactory.Apps().V1beta1().StatefulSets().Informer().HasSynced,
|
|
||||||
informerFactory.Core().V1().Pods().Informer().HasSynced,
|
|
||||||
)
|
|
||||||
|
|
||||||
if err := scaleUpStatefulSetControl(set, ssc, spc); !apierrors.IsInternalError(err) {
|
|
||||||
t.Errorf("StatefulSetControl did not return InternalError found %s", err)
|
t.Errorf("StatefulSetControl did not return InternalError found %s", err)
|
||||||
}
|
}
|
||||||
if err := scaleUpStatefulSetControl(set, ssc, spc); err != nil {
|
if err := scaleUpStatefulSetControl(t, set, ssc, spc, invariants); err != nil {
|
||||||
t.Errorf("Failed to turn up StatefulSet : %s", err)
|
t.Errorf("Failed to turn up StatefulSet : %s", err)
|
||||||
}
|
}
|
||||||
var err error
|
var err error
|
||||||
@@ -355,26 +351,14 @@ func TestDefaultStatefulSetControlCreatePodFailure(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestDefaultStatefulSetControlUpdatePodFailure(t *testing.T) {
|
func UpdatePodFailure(t *testing.T, set *apps.StatefulSet, invariants invariantFunc) {
|
||||||
set := newStatefulSet(3)
|
|
||||||
client := fake.NewSimpleClientset(set)
|
client := fake.NewSimpleClientset(set)
|
||||||
|
spc, ssc, stop := setupController(client)
|
||||||
informerFactory := informers.NewSharedInformerFactory(client, controller.NoResyncPeriodFunc())
|
defer close(stop)
|
||||||
spc := newFakeStatefulPodControl(informerFactory.Core().V1().Pods(), informerFactory.Apps().V1beta1().StatefulSets())
|
|
||||||
ssc := NewDefaultStatefulSetControl(spc)
|
|
||||||
spc.SetUpdateStatefulPodError(apierrors.NewInternalError(errors.New("API server failed")), 0)
|
spc.SetUpdateStatefulPodError(apierrors.NewInternalError(errors.New("API server failed")), 0)
|
||||||
|
|
||||||
stop := make(chan struct{})
|
|
||||||
defer close(stop)
|
|
||||||
informerFactory.Start(stop)
|
|
||||||
cache.WaitForCacheSync(
|
|
||||||
stop,
|
|
||||||
informerFactory.Apps().V1beta1().StatefulSets().Informer().HasSynced,
|
|
||||||
informerFactory.Core().V1().Pods().Informer().HasSynced,
|
|
||||||
)
|
|
||||||
|
|
||||||
// have to have 1 successful loop first
|
// have to have 1 successful loop first
|
||||||
if err := scaleUpStatefulSetControl(set, ssc, spc); err != nil {
|
if err := scaleUpStatefulSetControl(t, set, ssc, spc, invariants); err != nil {
|
||||||
t.Fatalf("Unexpected error: %v", err)
|
t.Fatalf("Unexpected error: %v", err)
|
||||||
}
|
}
|
||||||
var err error
|
var err error
|
||||||
@@ -404,25 +388,13 @@ func TestDefaultStatefulSetControlUpdatePodFailure(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestDefaultStatefulSetControlBlocksOnTerminating(t *testing.T) {
|
func testDefaultStatefulSetControlBlocksOnTerminating(t *testing.T, set *apps.StatefulSet, invariants invariantFunc) {
|
||||||
set := newStatefulSet(3)
|
|
||||||
client := fake.NewSimpleClientset(set)
|
client := fake.NewSimpleClientset(set)
|
||||||
|
spc, ssc, stop := setupController(client)
|
||||||
informerFactory := informers.NewSharedInformerFactory(client, controller.NoResyncPeriodFunc())
|
defer close(stop)
|
||||||
spc := newFakeStatefulPodControl(informerFactory.Core().V1().Pods(), informerFactory.Apps().V1beta1().StatefulSets())
|
|
||||||
ssc := NewDefaultStatefulSetControl(spc)
|
|
||||||
spc.SetUpdateStatefulPodError(apierrors.NewInternalError(errors.New("API server failed")), 0)
|
spc.SetUpdateStatefulPodError(apierrors.NewInternalError(errors.New("API server failed")), 0)
|
||||||
|
|
||||||
stop := make(chan struct{})
|
if err := scaleUpStatefulSetControl(t, set, ssc, spc, invariants); err != nil {
|
||||||
defer close(stop)
|
|
||||||
informerFactory.Start(stop)
|
|
||||||
cache.WaitForCacheSync(
|
|
||||||
stop,
|
|
||||||
informerFactory.Apps().V1beta1().StatefulSets().Informer().HasSynced,
|
|
||||||
informerFactory.Core().V1().Pods().Informer().HasSynced,
|
|
||||||
)
|
|
||||||
|
|
||||||
if err := scaleUpStatefulSetControl(set, ssc, spc); err != nil {
|
|
||||||
t.Fatalf("Unexpected error: %v", err)
|
t.Fatalf("Unexpected error: %v", err)
|
||||||
}
|
}
|
||||||
var err error
|
var err error
|
||||||
@@ -458,7 +430,7 @@ func TestDefaultStatefulSetControlBlocksOnTerminating(t *testing.T) {
|
|||||||
if len(pods) != 2 {
|
if len(pods) != 2 {
|
||||||
t.Fatalf("Expected 3 pods, got %d", len(pods))
|
t.Fatalf("Expected 3 pods, got %d", len(pods))
|
||||||
}
|
}
|
||||||
if err := scaleUpStatefulSetControl(set, ssc, spc); err != nil {
|
if err := scaleUpStatefulSetControl(t, set, ssc, spc, invariants); err != nil {
|
||||||
t.Fatalf("Unexpected error: %v", err)
|
t.Fatalf("Unexpected error: %v", err)
|
||||||
}
|
}
|
||||||
set, err = spc.setsLister.StatefulSets(set.Namespace).Get(set.Name)
|
set, err = spc.setsLister.StatefulSets(set.Namespace).Get(set.Name)
|
||||||
@@ -470,28 +442,16 @@ func TestDefaultStatefulSetControlBlocksOnTerminating(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestDefaultStatefulSetControlUpdateSetStatusFailure(t *testing.T) {
|
func UpdateSetStatusFailure(t *testing.T, set *apps.StatefulSet, invariants invariantFunc) {
|
||||||
set := newStatefulSet(3)
|
|
||||||
client := fake.NewSimpleClientset(set)
|
client := fake.NewSimpleClientset(set)
|
||||||
|
spc, ssc, stop := setupController(client)
|
||||||
informerFactory := informers.NewSharedInformerFactory(client, controller.NoResyncPeriodFunc())
|
defer close(stop)
|
||||||
spc := newFakeStatefulPodControl(informerFactory.Core().V1().Pods(), informerFactory.Apps().V1beta1().StatefulSets())
|
|
||||||
ssc := NewDefaultStatefulSetControl(spc)
|
|
||||||
spc.SetUpdateStatefulSetStatusError(apierrors.NewInternalError(errors.New("API server failed")), 2)
|
spc.SetUpdateStatefulSetStatusError(apierrors.NewInternalError(errors.New("API server failed")), 2)
|
||||||
|
|
||||||
stop := make(chan struct{})
|
if err := scaleUpStatefulSetControl(t, set, ssc, spc, invariants); !apierrors.IsInternalError(err) {
|
||||||
defer close(stop)
|
|
||||||
informerFactory.Start(stop)
|
|
||||||
cache.WaitForCacheSync(
|
|
||||||
stop,
|
|
||||||
informerFactory.Apps().V1beta1().StatefulSets().Informer().HasSynced,
|
|
||||||
informerFactory.Core().V1().Pods().Informer().HasSynced,
|
|
||||||
)
|
|
||||||
|
|
||||||
if err := scaleUpStatefulSetControl(set, ssc, spc); !apierrors.IsInternalError(err) {
|
|
||||||
t.Errorf("StatefulSetControl did not return InternalError found %s", err)
|
t.Errorf("StatefulSetControl did not return InternalError found %s", err)
|
||||||
}
|
}
|
||||||
if err := scaleUpStatefulSetControl(set, ssc, spc); err != nil {
|
if err := scaleUpStatefulSetControl(t, set, ssc, spc, invariants); err != nil {
|
||||||
t.Errorf("Failed to turn up StatefulSet : %s", err)
|
t.Errorf("Failed to turn up StatefulSet : %s", err)
|
||||||
}
|
}
|
||||||
var err error
|
var err error
|
||||||
@@ -504,22 +464,10 @@ func TestDefaultStatefulSetControlUpdateSetStatusFailure(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestDefaultStatefulSetControlPodRecreateDeleteError(t *testing.T) {
|
func PodRecreateDeleteFailure(t *testing.T, set *apps.StatefulSet, invariants invariantFunc) {
|
||||||
set := newStatefulSet(3)
|
|
||||||
client := fake.NewSimpleClientset(set)
|
client := fake.NewSimpleClientset(set)
|
||||||
|
spc, ssc, stop := setupController(client)
|
||||||
informerFactory := informers.NewSharedInformerFactory(client, controller.NoResyncPeriodFunc())
|
|
||||||
spc := newFakeStatefulPodControl(informerFactory.Core().V1().Pods(), informerFactory.Apps().V1beta1().StatefulSets())
|
|
||||||
ssc := NewDefaultStatefulSetControl(spc)
|
|
||||||
|
|
||||||
stop := make(chan struct{})
|
|
||||||
defer close(stop)
|
defer close(stop)
|
||||||
informerFactory.Start(stop)
|
|
||||||
cache.WaitForCacheSync(
|
|
||||||
stop,
|
|
||||||
informerFactory.Apps().V1beta1().StatefulSets().Informer().HasSynced,
|
|
||||||
informerFactory.Core().V1().Pods().Informer().HasSynced,
|
|
||||||
)
|
|
||||||
|
|
||||||
selector, err := metav1.LabelSelectorAsSelector(set.Spec.Selector)
|
selector, err := metav1.LabelSelectorAsSelector(set.Spec.Selector)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -532,7 +480,7 @@ func TestDefaultStatefulSetControlPodRecreateDeleteError(t *testing.T) {
|
|||||||
if err := ssc.UpdateStatefulSet(set, pods); err != nil {
|
if err := ssc.UpdateStatefulSet(set, pods); err != nil {
|
||||||
t.Errorf("Error updating StatefulSet %s", err)
|
t.Errorf("Error updating StatefulSet %s", err)
|
||||||
}
|
}
|
||||||
if err := assertInvariants(set, spc); err != nil {
|
if err := invariants(set, spc); err != nil {
|
||||||
t.Error(err)
|
t.Error(err)
|
||||||
}
|
}
|
||||||
pods, err = spc.podsLister.Pods(set.Namespace).List(selector)
|
pods, err = spc.podsLister.Pods(set.Namespace).List(selector)
|
||||||
@@ -545,13 +493,13 @@ func TestDefaultStatefulSetControlPodRecreateDeleteError(t *testing.T) {
|
|||||||
if err := ssc.UpdateStatefulSet(set, pods); !apierrors.IsInternalError(err) {
|
if err := ssc.UpdateStatefulSet(set, pods); !apierrors.IsInternalError(err) {
|
||||||
t.Errorf("StatefulSet failed to %s", err)
|
t.Errorf("StatefulSet failed to %s", err)
|
||||||
}
|
}
|
||||||
if err := assertInvariants(set, spc); err != nil {
|
if err := invariants(set, spc); err != nil {
|
||||||
t.Error(err)
|
t.Error(err)
|
||||||
}
|
}
|
||||||
if err := ssc.UpdateStatefulSet(set, pods); err != nil {
|
if err := ssc.UpdateStatefulSet(set, pods); err != nil {
|
||||||
t.Errorf("Error updating StatefulSet %s", err)
|
t.Errorf("Error updating StatefulSet %s", err)
|
||||||
}
|
}
|
||||||
if err := assertInvariants(set, spc); err != nil {
|
if err := invariants(set, spc); err != nil {
|
||||||
t.Error(err)
|
t.Error(err)
|
||||||
}
|
}
|
||||||
pods, err = spc.podsLister.Pods(set.Namespace).List(selector)
|
pods, err = spc.podsLister.Pods(set.Namespace).List(selector)
|
||||||
@@ -564,23 +512,13 @@ func TestDefaultStatefulSetControlPodRecreateDeleteError(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestStatefulSetControlScaleDownDeleteError(t *testing.T) {
|
func TestStatefulSetControlScaleDownDeleteError(t *testing.T) {
|
||||||
|
invariants := assertMonotonicInvariants
|
||||||
set := newStatefulSet(3)
|
set := newStatefulSet(3)
|
||||||
client := fake.NewSimpleClientset(set)
|
client := fake.NewSimpleClientset(set)
|
||||||
|
spc, ssc, stop := setupController(client)
|
||||||
informerFactory := informers.NewSharedInformerFactory(client, controller.NoResyncPeriodFunc())
|
|
||||||
spc := newFakeStatefulPodControl(informerFactory.Core().V1().Pods(), informerFactory.Apps().V1beta1().StatefulSets())
|
|
||||||
ssc := NewDefaultStatefulSetControl(spc)
|
|
||||||
|
|
||||||
stop := make(chan struct{})
|
|
||||||
defer close(stop)
|
defer close(stop)
|
||||||
informerFactory.Start(stop)
|
|
||||||
cache.WaitForCacheSync(
|
|
||||||
stop,
|
|
||||||
informerFactory.Apps().V1beta1().StatefulSets().Informer().HasSynced,
|
|
||||||
informerFactory.Core().V1().Pods().Informer().HasSynced,
|
|
||||||
)
|
|
||||||
|
|
||||||
if err := scaleUpStatefulSetControl(set, ssc, spc); err != nil {
|
if err := scaleUpStatefulSetControl(t, set, ssc, spc, invariants); err != nil {
|
||||||
t.Errorf("Failed to turn up StatefulSet : %s", err)
|
t.Errorf("Failed to turn up StatefulSet : %s", err)
|
||||||
}
|
}
|
||||||
var err error
|
var err error
|
||||||
@@ -590,10 +528,10 @@ func TestStatefulSetControlScaleDownDeleteError(t *testing.T) {
|
|||||||
}
|
}
|
||||||
*set.Spec.Replicas = 0
|
*set.Spec.Replicas = 0
|
||||||
spc.SetDeleteStatefulPodError(apierrors.NewInternalError(errors.New("API server failed")), 2)
|
spc.SetDeleteStatefulPodError(apierrors.NewInternalError(errors.New("API server failed")), 2)
|
||||||
if err := scaleDownStatefulSetControl(set, ssc, spc); !apierrors.IsInternalError(err) {
|
if err := scaleDownStatefulSetControl(t, set, ssc, spc, invariants); !apierrors.IsInternalError(err) {
|
||||||
t.Errorf("StatefulSetControl failed to throw error on delete %s", err)
|
t.Errorf("StatefulSetControl failed to throw error on delete %s", err)
|
||||||
}
|
}
|
||||||
if err := scaleDownStatefulSetControl(set, ssc, spc); err != nil {
|
if err := scaleDownStatefulSetControl(t, set, ssc, spc, invariants); err != nil {
|
||||||
t.Errorf("Failed to turn down StatefulSet %s", err)
|
t.Errorf("Failed to turn down StatefulSet %s", err)
|
||||||
}
|
}
|
||||||
set, err = spc.setsLister.StatefulSets(set.Namespace).Get(set.Name)
|
set, err = spc.setsLister.StatefulSets(set.Namespace).Get(set.Name)
|
||||||
@@ -672,6 +610,14 @@ func (spc *fakeStatefulPodControl) SetUpdateStatefulSetStatusError(err error, af
|
|||||||
spc.updateStatusTracker.after = after
|
spc.updateStatusTracker.after = after
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func copyPod(pod *v1.Pod) *v1.Pod {
|
||||||
|
obj, err := api.Scheme.Copy(pod)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
return obj.(*v1.Pod)
|
||||||
|
}
|
||||||
|
|
||||||
func (spc *fakeStatefulPodControl) setPodPending(set *apps.StatefulSet, ordinal int) ([]*v1.Pod, error) {
|
func (spc *fakeStatefulPodControl) setPodPending(set *apps.StatefulSet, ordinal int) ([]*v1.Pod, error) {
|
||||||
selector, err := metav1.LabelSelectorAsSelector(set.Spec.Selector)
|
selector, err := metav1.LabelSelectorAsSelector(set.Spec.Selector)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -685,7 +631,7 @@ func (spc *fakeStatefulPodControl) setPodPending(set *apps.StatefulSet, ordinal
|
|||||||
return nil, fmt.Errorf("ordinal %d out of range [0,%d)", ordinal, len(pods))
|
return nil, fmt.Errorf("ordinal %d out of range [0,%d)", ordinal, len(pods))
|
||||||
}
|
}
|
||||||
sort.Sort(ascendingOrdinal(pods))
|
sort.Sort(ascendingOrdinal(pods))
|
||||||
pod := pods[ordinal]
|
pod := copyPod(pods[ordinal])
|
||||||
pod.Status.Phase = v1.PodPending
|
pod.Status.Phase = v1.PodPending
|
||||||
fakeResourceVersion(pod)
|
fakeResourceVersion(pod)
|
||||||
spc.podsIndexer.Update(pod)
|
spc.podsIndexer.Update(pod)
|
||||||
@@ -705,7 +651,7 @@ func (spc *fakeStatefulPodControl) setPodRunning(set *apps.StatefulSet, ordinal
|
|||||||
return nil, fmt.Errorf("ordinal %d out of range [0,%d)", ordinal, len(pods))
|
return nil, fmt.Errorf("ordinal %d out of range [0,%d)", ordinal, len(pods))
|
||||||
}
|
}
|
||||||
sort.Sort(ascendingOrdinal(pods))
|
sort.Sort(ascendingOrdinal(pods))
|
||||||
pod := pods[ordinal]
|
pod := copyPod(pods[ordinal])
|
||||||
pod.Status.Phase = v1.PodRunning
|
pod.Status.Phase = v1.PodRunning
|
||||||
fakeResourceVersion(pod)
|
fakeResourceVersion(pod)
|
||||||
spc.podsIndexer.Update(pod)
|
spc.podsIndexer.Update(pod)
|
||||||
@@ -725,7 +671,7 @@ func (spc *fakeStatefulPodControl) setPodReady(set *apps.StatefulSet, ordinal in
|
|||||||
return nil, fmt.Errorf("ordinal %d out of range [0,%d)", ordinal, len(pods))
|
return nil, fmt.Errorf("ordinal %d out of range [0,%d)", ordinal, len(pods))
|
||||||
}
|
}
|
||||||
sort.Sort(ascendingOrdinal(pods))
|
sort.Sort(ascendingOrdinal(pods))
|
||||||
pod := pods[ordinal]
|
pod := copyPod(pods[ordinal])
|
||||||
condition := v1.PodCondition{Type: v1.PodReady, Status: v1.ConditionTrue}
|
condition := v1.PodCondition{Type: v1.PodReady, Status: v1.ConditionTrue}
|
||||||
podutil.UpdatePodCondition(&pod.Status, &condition)
|
podutil.UpdatePodCondition(&pod.Status, &condition)
|
||||||
fakeResourceVersion(pod)
|
fakeResourceVersion(pod)
|
||||||
@@ -746,7 +692,7 @@ func (spc *fakeStatefulPodControl) setPodInitStatus(set *apps.StatefulSet, ordin
|
|||||||
return nil, fmt.Errorf("ordinal %d out of range [0,%d)", ordinal, len(pods))
|
return nil, fmt.Errorf("ordinal %d out of range [0,%d)", ordinal, len(pods))
|
||||||
}
|
}
|
||||||
sort.Sort(ascendingOrdinal(pods))
|
sort.Sort(ascendingOrdinal(pods))
|
||||||
pod := pods[ordinal]
|
pod := copyPod(pods[ordinal])
|
||||||
if init {
|
if init {
|
||||||
pod.Annotations[apps.StatefulSetInitAnnotation] = "true"
|
pod.Annotations[apps.StatefulSetInitAnnotation] = "true"
|
||||||
} else {
|
} else {
|
||||||
@@ -850,7 +796,7 @@ func (spc *fakeStatefulPodControl) UpdateStatefulSetStatus(set *apps.StatefulSet
|
|||||||
|
|
||||||
var _ StatefulPodControlInterface = &fakeStatefulPodControl{}
|
var _ StatefulPodControlInterface = &fakeStatefulPodControl{}
|
||||||
|
|
||||||
func assertInvariants(set *apps.StatefulSet, spc *fakeStatefulPodControl) error {
|
func assertMonotonicInvariants(set *apps.StatefulSet, spc *fakeStatefulPodControl) error {
|
||||||
selector, err := metav1.LabelSelectorAsSelector(set.Spec.Selector)
|
selector, err := metav1.LabelSelectorAsSelector(set.Spec.Selector)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
@@ -862,31 +808,61 @@ func assertInvariants(set *apps.StatefulSet, spc *fakeStatefulPodControl) error
|
|||||||
sort.Sort(ascendingOrdinal(pods))
|
sort.Sort(ascendingOrdinal(pods))
|
||||||
for ord := 0; ord < len(pods); ord++ {
|
for ord := 0; ord < len(pods); ord++ {
|
||||||
if ord > 0 && isRunningAndReady(pods[ord]) && !isRunningAndReady(pods[ord-1]) {
|
if ord > 0 && isRunningAndReady(pods[ord]) && !isRunningAndReady(pods[ord-1]) {
|
||||||
return fmt.Errorf("Predecessor %s is Running and Ready while %s is not",
|
return fmt.Errorf("Successor %s is Running and Ready while %s is not", pods[ord].Name, pods[ord-1].Name)
|
||||||
pods[ord-1].Name,
|
|
||||||
pods[ord].Name)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if getOrdinal(pods[ord]) != ord {
|
if getOrdinal(pods[ord]) != ord {
|
||||||
return fmt.Errorf("Pods %s deployed in the wrong order",
|
return fmt.Errorf("pods %s deployed in the wrong order", pods[ord].Name)
|
||||||
pods[ord].Name)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if !storageMatches(set, pods[ord]) {
|
if !storageMatches(set, pods[ord]) {
|
||||||
return fmt.Errorf("Pods %s does not match the storage specification of StatefulSet %s ",
|
return fmt.Errorf("pods %s does not match the storage specification of StatefulSet %s ", pods[ord].Name, set.Name)
|
||||||
pods[ord].
|
}
|
||||||
Name, set.Name)
|
|
||||||
} else {
|
|
||||||
for _, claim := range getPersistentVolumeClaims(set, pods[ord]) {
|
for _, claim := range getPersistentVolumeClaims(set, pods[ord]) {
|
||||||
if claim, err := spc.claimsLister.PersistentVolumeClaims(set.Namespace).Get(claim.Name); err != nil {
|
claim, err := spc.claimsLister.PersistentVolumeClaims(set.Namespace).Get(claim.Name)
|
||||||
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
} else if claim == nil {
|
}
|
||||||
return fmt.Errorf("claim %s for Pod %s was not created",
|
if claim == nil {
|
||||||
claim.Name,
|
return fmt.Errorf("claim %s for Pod %s was not created", claim.Name, pods[ord].Name)
|
||||||
pods[ord].Name)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if !identityMatches(set, pods[ord]) {
|
if !identityMatches(set, pods[ord]) {
|
||||||
return fmt.Errorf("Pods %s does not match the identity specification of StatefulSet %s ",
|
return fmt.Errorf("pods %s does not match the identity specification of StatefulSet %s ", pods[ord].Name, set.Name)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func assertBurstInvariants(set *apps.StatefulSet, spc *fakeStatefulPodControl) error {
|
||||||
|
selector, err := metav1.LabelSelectorAsSelector(set.Spec.Selector)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
pods, err := spc.podsLister.Pods(set.Namespace).List(selector)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
sort.Sort(ascendingOrdinal(pods))
|
||||||
|
for ord := 0; ord < len(pods); ord++ {
|
||||||
|
if !storageMatches(set, pods[ord]) {
|
||||||
|
return fmt.Errorf("pods %s does not match the storage specification of StatefulSet %s ", pods[ord].Name, set.Name)
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, claim := range getPersistentVolumeClaims(set, pods[ord]) {
|
||||||
|
claim, err := spc.claimsLister.PersistentVolumeClaims(set.Namespace).Get(claim.Name)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if claim == nil {
|
||||||
|
return fmt.Errorf("claim %s for Pod %s was not created", claim.Name, pods[ord].Name)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if !identityMatches(set, pods[ord]) {
|
||||||
|
return fmt.Errorf("pods %s does not match the identity specification of StatefulSet %s ",
|
||||||
pods[ord].Name,
|
pods[ord].Name,
|
||||||
set.Name)
|
set.Name)
|
||||||
}
|
}
|
||||||
@@ -898,14 +874,15 @@ func fakeResourceVersion(object interface{}) {
|
|||||||
obj, isObj := object.(metav1.Object)
|
obj, isObj := object.(metav1.Object)
|
||||||
if !isObj {
|
if !isObj {
|
||||||
return
|
return
|
||||||
} else if version := obj.GetResourceVersion(); version == "" {
|
}
|
||||||
|
if version := obj.GetResourceVersion(); version == "" {
|
||||||
obj.SetResourceVersion("1")
|
obj.SetResourceVersion("1")
|
||||||
} else if intValue, err := strconv.ParseInt(version, 10, 32); err == nil {
|
} else if intValue, err := strconv.ParseInt(version, 10, 32); err == nil {
|
||||||
obj.SetResourceVersion(strconv.FormatInt(intValue+1, 10))
|
obj.SetResourceVersion(strconv.FormatInt(intValue+1, 10))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func scaleUpStatefulSetControl(set *apps.StatefulSet, ssc StatefulSetControlInterface, spc *fakeStatefulPodControl) error {
|
func scaleUpStatefulSetControl(t *testing.T, set *apps.StatefulSet, ssc StatefulSetControlInterface, spc *fakeStatefulPodControl, invariants invariantFunc) error {
|
||||||
selector, err := metav1.LabelSelectorAsSelector(set.Spec.Selector)
|
selector, err := metav1.LabelSelectorAsSelector(set.Spec.Selector)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
@@ -915,21 +892,44 @@ func scaleUpStatefulSetControl(set *apps.StatefulSet, ssc StatefulSetControlInte
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
if ord := len(pods) - 1; ord >= 0 {
|
sort.Sort(ascendingOrdinal(pods))
|
||||||
ord := len(pods) - 1
|
|
||||||
|
// ensure all pods are valid (have a phase)
|
||||||
|
initialized := false
|
||||||
|
for ord, pod := range pods {
|
||||||
|
if pod.Status.Phase == "" {
|
||||||
|
t.Logf("found pod %s pending", pod.Name)
|
||||||
if pods, err = spc.setPodPending(set, ord); err != nil {
|
if pods, err = spc.setPodPending(set, ord); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
if err = ssc.UpdateStatefulSet(set, pods); err != nil {
|
break
|
||||||
return err
|
|
||||||
}
|
}
|
||||||
set, err = spc.setsLister.StatefulSets(set.Namespace).Get(set.Name)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
}
|
||||||
|
if initialized {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
// select one of the pods and move it forward in status
|
||||||
|
if len(pods) > 0 {
|
||||||
|
ord := int(rand.Int63n(int64(len(pods))))
|
||||||
|
pod := pods[ord]
|
||||||
|
switch pod.Status.Phase {
|
||||||
|
case v1.PodPending:
|
||||||
|
t.Logf("set pod %s running", pod.Name)
|
||||||
if pods, err = spc.setPodRunning(set, ord); err != nil {
|
if pods, err = spc.setPodRunning(set, ord); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
case v1.PodRunning:
|
||||||
|
t.Logf("set pod %s ready", pod.Name)
|
||||||
|
if pods, err = spc.setPodReady(set, ord); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// run the controller once and check invariants
|
||||||
if err = ssc.UpdateStatefulSet(set, pods); err != nil {
|
if err = ssc.UpdateStatefulSet(set, pods); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@@ -937,25 +937,14 @@ func scaleUpStatefulSetControl(set *apps.StatefulSet, ssc StatefulSetControlInte
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
if pods, err = spc.setPodReady(set, ord); err != nil {
|
if err := invariants(set, spc); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if err := ssc.UpdateStatefulSet(set, pods); err != nil {
|
return invariants(set, spc)
|
||||||
return err
|
|
||||||
}
|
|
||||||
set, err = spc.setsLister.StatefulSets(set.Namespace).Get(set.Name)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err := assertInvariants(set, spc); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return assertInvariants(set, spc)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func scaleDownStatefulSetControl(set *apps.StatefulSet, ssc StatefulSetControlInterface, spc *fakeStatefulPodControl) error {
|
func scaleDownStatefulSetControl(t *testing.T, set *apps.StatefulSet, ssc StatefulSetControlInterface, spc *fakeStatefulPodControl, invariants invariantFunc) error {
|
||||||
selector, err := metav1.LabelSelectorAsSelector(set.Spec.Selector)
|
selector, err := metav1.LabelSelectorAsSelector(set.Spec.Selector)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
@@ -965,6 +954,7 @@ func scaleDownStatefulSetControl(set *apps.StatefulSet, ssc StatefulSetControlIn
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
sort.Sort(ascendingOrdinal(pods))
|
||||||
if ordinal := len(pods) - 1; ordinal >= 0 {
|
if ordinal := len(pods) - 1; ordinal >= 0 {
|
||||||
if err := ssc.UpdateStatefulSet(set, pods); err != nil {
|
if err := ssc.UpdateStatefulSet(set, pods); err != nil {
|
||||||
return err
|
return err
|
||||||
@@ -997,9 +987,9 @@ func scaleDownStatefulSetControl(set *apps.StatefulSet, ssc StatefulSetControlIn
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
if err := assertInvariants(set, spc); err != nil {
|
if err := invariants(set, spc); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return assertInvariants(set, spc)
|
return invariants(set, spc)
|
||||||
}
|
}
|
||||||
|
@@ -640,7 +640,7 @@ func scaleUpStatefulSetController(set *apps.StatefulSet, ssc *StatefulSetControl
|
|||||||
pod = getPodAtOrdinal(pods, ord)
|
pod = getPodAtOrdinal(pods, ord)
|
||||||
ssc.updatePod(&prev, pod)
|
ssc.updatePod(&prev, pod)
|
||||||
fakeWorker(ssc)
|
fakeWorker(ssc)
|
||||||
if err := assertInvariants(set, spc); err != nil {
|
if err := assertMonotonicInvariants(set, spc); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
if obj, _, err := spc.setsIndexer.Get(set); err != nil {
|
if obj, _, err := spc.setsIndexer.Get(set); err != nil {
|
||||||
@@ -650,7 +650,7 @@ func scaleUpStatefulSetController(set *apps.StatefulSet, ssc *StatefulSetControl
|
|||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
return assertInvariants(set, spc)
|
return assertMonotonicInvariants(set, spc)
|
||||||
}
|
}
|
||||||
|
|
||||||
func scaleDownStatefulSetController(set *apps.StatefulSet, ssc *StatefulSetController, spc *fakeStatefulPodControl) error {
|
func scaleDownStatefulSetController(set *apps.StatefulSet, ssc *StatefulSetController, spc *fakeStatefulPodControl) error {
|
||||||
@@ -692,5 +692,5 @@ func scaleDownStatefulSetController(set *apps.StatefulSet, ssc *StatefulSetContr
|
|||||||
set = obj.(*apps.StatefulSet)
|
set = obj.(*apps.StatefulSet)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return assertInvariants(set, spc)
|
return assertMonotonicInvariants(set, spc)
|
||||||
}
|
}
|
||||||
|
@@ -227,6 +227,11 @@ func isHealthy(pod *v1.Pod) bool {
|
|||||||
return isRunningAndReady(pod) && !isTerminating(pod)
|
return isRunningAndReady(pod) && !isTerminating(pod)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// allowsBurst is true if the alpha burst annotation is set.
|
||||||
|
func allowsBurst(set *apps.StatefulSet) bool {
|
||||||
|
return set.Spec.PodManagementPolicy == apps.ParallelPodManagement
|
||||||
|
}
|
||||||
|
|
||||||
// newControllerRef returns an ControllerRef pointing to a given StatefulSet.
|
// newControllerRef returns an ControllerRef pointing to a given StatefulSet.
|
||||||
func newControllerRef(set *apps.StatefulSet) *metav1.OwnerReference {
|
func newControllerRef(set *apps.StatefulSet) *metav1.OwnerReference {
|
||||||
blockOwnerDeletion := true
|
blockOwnerDeletion := true
|
||||||
|
@@ -58,6 +58,7 @@ func validNewStatefulSet() *apps.StatefulSet {
|
|||||||
Labels: map[string]string{"a": "b"},
|
Labels: map[string]string{"a": "b"},
|
||||||
},
|
},
|
||||||
Spec: apps.StatefulSetSpec{
|
Spec: apps.StatefulSetSpec{
|
||||||
|
PodManagementPolicy: apps.OrderedReadyPodManagement,
|
||||||
Selector: &metav1.LabelSelector{MatchLabels: map[string]string{"a": "b"}},
|
Selector: &metav1.LabelSelector{MatchLabels: map[string]string{"a": "b"}},
|
||||||
Template: api.PodTemplateSpec{
|
Template: api.PodTemplateSpec{
|
||||||
ObjectMeta: metav1.ObjectMeta{
|
ObjectMeta: metav1.ObjectMeta{
|
||||||
|
@@ -51,6 +51,7 @@ func TestStatefulSetStrategy(t *testing.T) {
|
|||||||
ps := &apps.StatefulSet{
|
ps := &apps.StatefulSet{
|
||||||
ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: metav1.NamespaceDefault},
|
ObjectMeta: metav1.ObjectMeta{Name: "abc", Namespace: metav1.NamespaceDefault},
|
||||||
Spec: apps.StatefulSetSpec{
|
Spec: apps.StatefulSetSpec{
|
||||||
|
PodManagementPolicy: apps.OrderedReadyPodManagement,
|
||||||
Selector: &metav1.LabelSelector{MatchLabels: validSelector},
|
Selector: &metav1.LabelSelector{MatchLabels: validSelector},
|
||||||
Template: validPodTemplate.Template,
|
Template: validPodTemplate.Template,
|
||||||
},
|
},
|
||||||
@@ -70,6 +71,7 @@ func TestStatefulSetStrategy(t *testing.T) {
|
|||||||
validPs := &apps.StatefulSet{
|
validPs := &apps.StatefulSet{
|
||||||
ObjectMeta: metav1.ObjectMeta{Name: ps.Name, Namespace: ps.Namespace, ResourceVersion: "1", Generation: 1},
|
ObjectMeta: metav1.ObjectMeta{Name: ps.Name, Namespace: ps.Namespace, ResourceVersion: "1", Generation: 1},
|
||||||
Spec: apps.StatefulSetSpec{
|
Spec: apps.StatefulSetSpec{
|
||||||
|
PodManagementPolicy: apps.OrderedReadyPodManagement,
|
||||||
Selector: ps.Spec.Selector,
|
Selector: ps.Spec.Selector,
|
||||||
Template: validPodTemplate.Template,
|
Template: validPodTemplate.Template,
|
||||||
},
|
},
|
||||||
|
@@ -44,6 +44,21 @@ type StatefulSet struct {
|
|||||||
Status StatefulSetStatus
|
Status StatefulSetStatus
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// PodManagementPolicyType defines the policy for creating pods under a stateful set.
|
||||||
|
type PodManagementPolicyType string
|
||||||
|
|
||||||
|
const (
|
||||||
|
// OrderedReadyPodManagement will create pods in strictly increasing order on
|
||||||
|
// scale up and strictly decreasing order on scale down, progressing only when
|
||||||
|
// the previous pod is ready or terminated. At most one pod will be changed
|
||||||
|
// at any time.
|
||||||
|
OrderedReadyPodManagement PodManagementPolicyType = "OrderedReady"
|
||||||
|
// ParallelPodManagement will create and delete pods as soon as the stateful set
|
||||||
|
// replica count is changed, and will not wait for pods to be ready or complete
|
||||||
|
// termination.
|
||||||
|
ParallelPodManagement = "Parallel"
|
||||||
|
)
|
||||||
|
|
||||||
// A StatefulSetSpec is the specification of a StatefulSet.
|
// A StatefulSetSpec is the specification of a StatefulSet.
|
||||||
type StatefulSetSpec struct {
|
type StatefulSetSpec struct {
|
||||||
// Replicas is the desired number of replicas of the given Template.
|
// Replicas is the desired number of replicas of the given Template.
|
||||||
@@ -82,6 +97,17 @@ type StatefulSetSpec struct {
|
|||||||
// pattern: pod-specific-string.serviceName.default.svc.cluster.local
|
// pattern: pod-specific-string.serviceName.default.svc.cluster.local
|
||||||
// where "pod-specific-string" is managed by the StatefulSet controller.
|
// where "pod-specific-string" is managed by the StatefulSet controller.
|
||||||
ServiceName string
|
ServiceName string
|
||||||
|
|
||||||
|
// PodManagementPolicy controls how pods are created during initial scale up,
|
||||||
|
// when replacing pods on nodes, or when scaling down. The default policy is
|
||||||
|
// `OrderedReady`, where pods are created in increasing order (pod-0, then
|
||||||
|
// pod-1, etc) and the controller will wait until each pod is ready before
|
||||||
|
// continuing. When scaling down, the pods are removed in the opposite order.
|
||||||
|
// The alternative policy is `Parallel` which will create pods in parallel
|
||||||
|
// to match the desired scale without waiting, and on scale down will delete
|
||||||
|
// all pods at once.
|
||||||
|
// +optional
|
||||||
|
PodManagementPolicy PodManagementPolicyType
|
||||||
}
|
}
|
||||||
|
|
||||||
// StatefulSetStatus represents the current state of a StatefulSet.
|
// StatefulSetStatus represents the current state of a StatefulSet.
|
||||||
|
@@ -110,6 +110,7 @@ func Convert_v1beta1_StatefulSetSpec_To_apps_StatefulSetSpec(in *StatefulSetSpec
|
|||||||
out.VolumeClaimTemplates = nil
|
out.VolumeClaimTemplates = nil
|
||||||
}
|
}
|
||||||
out.ServiceName = in.ServiceName
|
out.ServiceName = in.ServiceName
|
||||||
|
out.PodManagementPolicy = apps.PodManagementPolicyType(in.PodManagementPolicy)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -140,6 +141,7 @@ func Convert_apps_StatefulSetSpec_To_v1beta1_StatefulSetSpec(in *apps.StatefulSe
|
|||||||
out.VolumeClaimTemplates = nil
|
out.VolumeClaimTemplates = nil
|
||||||
}
|
}
|
||||||
out.ServiceName = in.ServiceName
|
out.ServiceName = in.ServiceName
|
||||||
|
out.PodManagementPolicy = PodManagementPolicyType(in.PodManagementPolicy)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -27,6 +27,9 @@ func addDefaultingFuncs(scheme *runtime.Scheme) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func SetDefaults_StatefulSet(obj *StatefulSet) {
|
func SetDefaults_StatefulSet(obj *StatefulSet) {
|
||||||
|
if len(obj.Spec.PodManagementPolicy) == 0 {
|
||||||
|
obj.Spec.PodManagementPolicy = OrderedReadyPodManagement
|
||||||
|
}
|
||||||
labels := obj.Spec.Template.Labels
|
labels := obj.Spec.Template.Labels
|
||||||
if labels != nil {
|
if labels != nil {
|
||||||
if obj.Spec.Selector == nil {
|
if obj.Spec.Selector == nil {
|
||||||
|
@@ -788,6 +788,10 @@ func (m *StatefulSetSpec) MarshalTo(dAtA []byte) (int, error) {
|
|||||||
i++
|
i++
|
||||||
i = encodeVarintGenerated(dAtA, i, uint64(len(m.ServiceName)))
|
i = encodeVarintGenerated(dAtA, i, uint64(len(m.ServiceName)))
|
||||||
i += copy(dAtA[i:], m.ServiceName)
|
i += copy(dAtA[i:], m.ServiceName)
|
||||||
|
dAtA[i] = 0x32
|
||||||
|
i++
|
||||||
|
i = encodeVarintGenerated(dAtA, i, uint64(len(m.PodManagementPolicy)))
|
||||||
|
i += copy(dAtA[i:], m.PodManagementPolicy)
|
||||||
return i, nil
|
return i, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1068,6 +1072,8 @@ func (m *StatefulSetSpec) Size() (n int) {
|
|||||||
}
|
}
|
||||||
l = len(m.ServiceName)
|
l = len(m.ServiceName)
|
||||||
n += 1 + l + sovGenerated(uint64(l))
|
n += 1 + l + sovGenerated(uint64(l))
|
||||||
|
l = len(m.PodManagementPolicy)
|
||||||
|
n += 1 + l + sovGenerated(uint64(l))
|
||||||
return n
|
return n
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1297,6 +1303,7 @@ func (this *StatefulSetSpec) String() string {
|
|||||||
`Template:` + strings.Replace(strings.Replace(this.Template.String(), "PodTemplateSpec", "k8s_io_kubernetes_pkg_api_v1.PodTemplateSpec", 1), `&`, ``, 1) + `,`,
|
`Template:` + strings.Replace(strings.Replace(this.Template.String(), "PodTemplateSpec", "k8s_io_kubernetes_pkg_api_v1.PodTemplateSpec", 1), `&`, ``, 1) + `,`,
|
||||||
`VolumeClaimTemplates:` + strings.Replace(strings.Replace(fmt.Sprintf("%v", this.VolumeClaimTemplates), "PersistentVolumeClaim", "k8s_io_kubernetes_pkg_api_v1.PersistentVolumeClaim", 1), `&`, ``, 1) + `,`,
|
`VolumeClaimTemplates:` + strings.Replace(strings.Replace(fmt.Sprintf("%v", this.VolumeClaimTemplates), "PersistentVolumeClaim", "k8s_io_kubernetes_pkg_api_v1.PersistentVolumeClaim", 1), `&`, ``, 1) + `,`,
|
||||||
`ServiceName:` + fmt.Sprintf("%v", this.ServiceName) + `,`,
|
`ServiceName:` + fmt.Sprintf("%v", this.ServiceName) + `,`,
|
||||||
|
`PodManagementPolicy:` + fmt.Sprintf("%v", this.PodManagementPolicy) + `,`,
|
||||||
`}`,
|
`}`,
|
||||||
}, "")
|
}, "")
|
||||||
return s
|
return s
|
||||||
@@ -3635,6 +3642,35 @@ func (m *StatefulSetSpec) Unmarshal(dAtA []byte) error {
|
|||||||
}
|
}
|
||||||
m.ServiceName = string(dAtA[iNdEx:postIndex])
|
m.ServiceName = string(dAtA[iNdEx:postIndex])
|
||||||
iNdEx = postIndex
|
iNdEx = postIndex
|
||||||
|
case 6:
|
||||||
|
if wireType != 2 {
|
||||||
|
return fmt.Errorf("proto: wrong wireType = %d for field PodManagementPolicy", wireType)
|
||||||
|
}
|
||||||
|
var stringLen uint64
|
||||||
|
for shift := uint(0); ; shift += 7 {
|
||||||
|
if shift >= 64 {
|
||||||
|
return ErrIntOverflowGenerated
|
||||||
|
}
|
||||||
|
if iNdEx >= l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
b := dAtA[iNdEx]
|
||||||
|
iNdEx++
|
||||||
|
stringLen |= (uint64(b) & 0x7F) << shift
|
||||||
|
if b < 0x80 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
intStringLen := int(stringLen)
|
||||||
|
if intStringLen < 0 {
|
||||||
|
return ErrInvalidLengthGenerated
|
||||||
|
}
|
||||||
|
postIndex := iNdEx + intStringLen
|
||||||
|
if postIndex > l {
|
||||||
|
return io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
m.PodManagementPolicy = PodManagementPolicyType(dAtA[iNdEx:postIndex])
|
||||||
|
iNdEx = postIndex
|
||||||
default:
|
default:
|
||||||
iNdEx = preIndex
|
iNdEx = preIndex
|
||||||
skippy, err := skipGenerated(dAtA[iNdEx:])
|
skippy, err := skipGenerated(dAtA[iNdEx:])
|
||||||
@@ -3855,101 +3891,103 @@ func init() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
var fileDescriptorGenerated = []byte{
|
var fileDescriptorGenerated = []byte{
|
||||||
// 1525 bytes of a gzipped FileDescriptorProto
|
// 1563 bytes of a gzipped FileDescriptorProto
|
||||||
0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0xe4, 0x58, 0xcb, 0x6f, 0x5b, 0xc5,
|
0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0xe4, 0x58, 0xcb, 0x6f, 0x1b, 0x37,
|
||||||
0x17, 0xce, 0x4d, 0xec, 0xc4, 0x99, 0x34, 0x4e, 0x33, 0xc9, 0xaf, 0xf1, 0x2f, 0x45, 0x4e, 0xe5,
|
0x13, 0xf7, 0xda, 0x92, 0x2d, 0xd3, 0xb1, 0x1c, 0xd3, 0xfe, 0x62, 0x7d, 0xce, 0x07, 0x39, 0xd0,
|
||||||
0x45, 0x1f, 0xa8, 0xbd, 0xa6, 0x69, 0xa1, 0x8f, 0x40, 0x45, 0xdc, 0x96, 0x52, 0x94, 0xd0, 0x6a,
|
0x21, 0x8f, 0x0f, 0xc9, 0xaa, 0x71, 0xd2, 0xbc, 0xdc, 0x06, 0xb5, 0x92, 0x34, 0x4d, 0x61, 0x37,
|
||||||
0xec, 0x54, 0xb4, 0x14, 0x89, 0xb1, 0x3d, 0xbd, 0x9d, 0xfa, 0xbe, 0x74, 0x67, 0x6c, 0xc5, 0x3b,
|
0x06, 0x65, 0x07, 0x4d, 0x9a, 0x02, 0xa5, 0x24, 0x66, 0xcd, 0x78, 0x5f, 0x58, 0x52, 0x82, 0x75,
|
||||||
0x36, 0x2c, 0x90, 0x58, 0xb0, 0x62, 0x87, 0xd8, 0x23, 0x24, 0x76, 0xfc, 0x0d, 0x11, 0x6c, 0xba,
|
0xeb, 0xa5, 0x87, 0x02, 0x3d, 0xf4, 0xd4, 0x5b, 0xd1, 0x9e, 0x8b, 0x02, 0xfd, 0x37, 0x8c, 0xf6,
|
||||||
0x44, 0x2c, 0x22, 0xe2, 0xfe, 0x17, 0x5d, 0xa1, 0x99, 0x3b, 0xf7, 0xe5, 0x6b, 0x27, 0x8e, 0x11,
|
0x92, 0x63, 0xd1, 0x83, 0x51, 0x3b, 0xff, 0x45, 0x4e, 0x05, 0xb9, 0xdc, 0x97, 0x76, 0x65, 0xcb,
|
||||||
0xdd, 0xb0, 0xf3, 0x9d, 0x39, 0xdf, 0x77, 0xce, 0xcc, 0x7c, 0xe7, 0xcc, 0x19, 0x83, 0x6b, 0xad,
|
0x2a, 0x9a, 0x4b, 0x6f, 0x5a, 0xce, 0xfc, 0x7e, 0x33, 0x24, 0x67, 0x86, 0x33, 0x02, 0x37, 0x77,
|
||||||
0xeb, 0x4c, 0xa7, 0x4e, 0xb9, 0xd5, 0xae, 0x13, 0xcf, 0x26, 0x9c, 0xb0, 0xb2, 0xdb, 0x32, 0xca,
|
0x6e, 0x31, 0x9d, 0x3a, 0xd5, 0x9d, 0x76, 0x83, 0x78, 0x36, 0xe1, 0x84, 0x55, 0xdd, 0x1d, 0xa3,
|
||||||
0xd8, 0xa5, 0xac, 0x8c, 0x5d, 0x97, 0x95, 0x3b, 0x97, 0xeb, 0x84, 0xe3, 0xcb, 0x65, 0x83, 0xd8,
|
0x8a, 0x5d, 0xca, 0xaa, 0xd8, 0x75, 0x59, 0xb5, 0x73, 0xb5, 0x41, 0x38, 0xbe, 0x5a, 0x35, 0x88,
|
||||||
0xc4, 0xc3, 0x9c, 0x34, 0x75, 0xd7, 0x73, 0xb8, 0x03, 0xcf, 0xf9, 0x40, 0x3d, 0x02, 0xea, 0x6e,
|
0x4d, 0x3c, 0xcc, 0x49, 0x4b, 0x77, 0x3d, 0x87, 0x3b, 0xf0, 0x82, 0x0f, 0xd4, 0x23, 0xa0, 0xee,
|
||||||
0xcb, 0xd0, 0x05, 0x50, 0x17, 0x40, 0x5d, 0x01, 0x57, 0x2f, 0x19, 0x94, 0x3f, 0x6f, 0xd7, 0xf5,
|
0xee, 0x18, 0xba, 0x00, 0xea, 0x02, 0xa8, 0x2b, 0xe0, 0xe2, 0x15, 0x83, 0xf2, 0xed, 0x76, 0x43,
|
||||||
0x86, 0x63, 0x95, 0x0d, 0xc7, 0x70, 0xca, 0x12, 0x5f, 0x6f, 0x3f, 0x93, 0x5f, 0xf2, 0x43, 0xfe,
|
0x6f, 0x3a, 0x56, 0xd5, 0x70, 0x0c, 0xa7, 0x2a, 0xf1, 0x8d, 0xf6, 0x0b, 0xf9, 0x25, 0x3f, 0xe4,
|
||||||
0xf2, 0x79, 0x57, 0xaf, 0xaa, 0x80, 0xb0, 0x4b, 0x2d, 0xdc, 0x78, 0x4e, 0x6d, 0xe2, 0x75, 0xa3,
|
0x2f, 0x9f, 0x77, 0xf1, 0xba, 0x72, 0x08, 0xbb, 0xd4, 0xc2, 0xcd, 0x6d, 0x6a, 0x13, 0xaf, 0x1b,
|
||||||
0x90, 0x2c, 0xc2, 0x71, 0xb9, 0x93, 0x8a, 0x66, 0xb5, 0x3c, 0x0c, 0xe5, 0xb5, 0x6d, 0x4e, 0x2d,
|
0xb9, 0x64, 0x11, 0x8e, 0xab, 0x9d, 0x94, 0x37, 0x8b, 0xd5, 0x7e, 0x28, 0xaf, 0x6d, 0x73, 0x6a,
|
||||||
0x92, 0x02, 0xbc, 0x77, 0x14, 0x80, 0x35, 0x9e, 0x13, 0x0b, 0xa7, 0x70, 0x57, 0x86, 0xe1, 0xda,
|
0x91, 0x14, 0xe0, 0xc6, 0x71, 0x00, 0xd6, 0xdc, 0x26, 0x16, 0x4e, 0xe1, 0xae, 0xf5, 0xc3, 0xb5,
|
||||||
0x9c, 0x9a, 0x65, 0x6a, 0x73, 0xc6, 0xbd, 0x14, 0x28, 0xb6, 0x26, 0x46, 0xbc, 0x0e, 0xf1, 0xa2,
|
0x39, 0x35, 0xab, 0xd4, 0xe6, 0x8c, 0x7b, 0x29, 0x50, 0x6c, 0x4f, 0x8c, 0x78, 0x1d, 0xe2, 0x45,
|
||||||
0x05, 0x91, 0x5d, 0x6c, 0xb9, 0x26, 0x19, 0xb4, 0xa6, 0x8b, 0x43, 0x8f, 0x66, 0x90, 0xf5, 0x07,
|
0x1b, 0x22, 0xbb, 0xd8, 0x72, 0x4d, 0x92, 0xb5, 0xa7, 0xcb, 0x7d, 0xaf, 0x26, 0x4b, 0xfb, 0xfd,
|
||||||
0x87, 0x1c, 0x24, 0xd9, 0xe5, 0xc4, 0x66, 0xd4, 0xb1, 0x87, 0x1e, 0x67, 0xe9, 0xe7, 0x49, 0x00,
|
0x23, 0x2e, 0x92, 0xec, 0x72, 0x62, 0x33, 0xea, 0xd8, 0x7d, 0xaf, 0xb3, 0xf2, 0xf3, 0x28, 0x00,
|
||||||
0xee, 0x10, 0xd7, 0x74, 0xba, 0x16, 0xb1, 0x39, 0xfc, 0x12, 0xe4, 0xc4, 0x56, 0x37, 0x31, 0xc7,
|
0xf7, 0x89, 0x6b, 0x3a, 0x5d, 0x8b, 0xd8, 0x1c, 0x7e, 0x01, 0x0a, 0xe2, 0xa8, 0x5b, 0x98, 0xe3,
|
||||||
0x05, 0xed, 0x8c, 0x76, 0x7e, 0x6e, 0xfd, 0x1d, 0x5d, 0x1d, 0x78, 0x7c, 0xe5, 0xd1, 0x91, 0x0b,
|
0x92, 0x76, 0x4e, 0xbb, 0x38, 0xb5, 0xfc, 0x8e, 0xae, 0x2e, 0x3c, 0xbe, 0xf3, 0xe8, 0xca, 0x85,
|
||||||
0x6b, 0xbd, 0x73, 0x59, 0x7f, 0x50, 0x7f, 0x41, 0x1a, 0x7c, 0x9b, 0x70, 0x5c, 0x81, 0x7b, 0xfb,
|
0xb6, 0xde, 0xb9, 0xaa, 0x3f, 0x6e, 0xbc, 0x24, 0x4d, 0xbe, 0x4e, 0x38, 0xae, 0xc1, 0xbd, 0xfd,
|
||||||
0x6b, 0x13, 0xbd, 0xfd, 0x35, 0x10, 0x8d, 0xa1, 0x90, 0x15, 0x3e, 0x06, 0x19, 0xe6, 0x92, 0x46,
|
0xa5, 0x91, 0xc3, 0xfd, 0x25, 0x10, 0xad, 0xa1, 0x90, 0x15, 0x3e, 0x05, 0x39, 0xe6, 0x92, 0x66,
|
||||||
0x61, 0x52, 0xb2, 0x5f, 0xd3, 0x47, 0x94, 0x93, 0x1e, 0x05, 0x59, 0x75, 0x49, 0xa3, 0x72, 0x42,
|
0x69, 0x54, 0xb2, 0xdf, 0xd4, 0x07, 0x0c, 0x27, 0x3d, 0x72, 0xb2, 0xee, 0x92, 0x66, 0xed, 0x94,
|
||||||
0x39, 0xc9, 0x88, 0x2f, 0x24, 0x29, 0x21, 0x06, 0xd3, 0x8c, 0x63, 0xde, 0x66, 0x85, 0x29, 0x49,
|
0x32, 0x92, 0x13, 0x5f, 0x48, 0x52, 0x42, 0x0c, 0xc6, 0x19, 0xc7, 0xbc, 0xcd, 0x4a, 0x63, 0x92,
|
||||||
0x7e, 0x63, 0x1c, 0x72, 0x49, 0x50, 0xc9, 0x2b, 0xfa, 0x69, 0xff, 0x1b, 0x29, 0xe2, 0xd2, 0xc1,
|
0xfc, 0xf6, 0x30, 0xe4, 0x92, 0xa0, 0x56, 0x54, 0xf4, 0xe3, 0xfe, 0x37, 0x52, 0xc4, 0x95, 0x83,
|
||||||
0x14, 0x58, 0x8a, 0x8c, 0x6f, 0x3b, 0x76, 0x93, 0x72, 0xea, 0xd8, 0x70, 0x03, 0x64, 0x78, 0xd7,
|
0x31, 0x30, 0x17, 0x29, 0xdf, 0x73, 0xec, 0x16, 0xe5, 0xd4, 0xb1, 0xe1, 0x0a, 0xc8, 0xf1, 0xae,
|
||||||
0x25, 0x72, 0xcf, 0x66, 0x2b, 0xe7, 0x82, 0xe0, 0x6a, 0x5d, 0x97, 0xbc, 0xde, 0x5f, 0x5b, 0x19,
|
0x4b, 0xe4, 0x99, 0x4d, 0xd6, 0x2e, 0x04, 0xce, 0x6d, 0x76, 0x5d, 0xf2, 0x66, 0x7f, 0x69, 0x21,
|
||||||
0x00, 0x11, 0x53, 0x48, 0x82, 0xe0, 0xa3, 0x30, 0xee, 0x49, 0x09, 0xbf, 0x95, 0x74, 0xfe, 0x7a,
|
0x03, 0x22, 0x44, 0x48, 0x82, 0xe0, 0x93, 0xd0, 0xef, 0x51, 0x09, 0xbf, 0x9b, 0x34, 0xfe, 0x66,
|
||||||
0x7f, 0xed, 0x50, 0x49, 0xe8, 0x21, 0x67, 0x32, 0x58, 0x78, 0x16, 0x4c, 0x7b, 0x04, 0x33, 0xc7,
|
0x7f, 0xe9, 0xc8, 0x90, 0xd0, 0x43, 0xce, 0xa4, 0xb3, 0xf0, 0x3c, 0x18, 0xf7, 0x08, 0x66, 0x8e,
|
||||||
0x2e, 0x64, 0x24, 0x6f, 0xb8, 0x28, 0x24, 0x47, 0x91, 0x9a, 0x85, 0x17, 0xc0, 0x8c, 0x45, 0x18,
|
0x5d, 0xca, 0x49, 0xde, 0x70, 0x53, 0x48, 0xae, 0x22, 0x25, 0x85, 0x97, 0xc0, 0x84, 0x45, 0x18,
|
||||||
0xc3, 0x06, 0x29, 0x64, 0xa5, 0xe1, 0x82, 0x32, 0x9c, 0xd9, 0xf6, 0x87, 0x51, 0x30, 0x0f, 0x5f,
|
0xc3, 0x06, 0x29, 0xe5, 0xa5, 0xe2, 0x8c, 0x52, 0x9c, 0x58, 0xf7, 0x97, 0x51, 0x20, 0x87, 0x2f,
|
||||||
0x80, 0xbc, 0x89, 0x19, 0xdf, 0x71, 0x9b, 0x98, 0x93, 0x1a, 0xb5, 0x48, 0x61, 0x5a, 0x6e, 0xf5,
|
0x41, 0xd1, 0xc4, 0x8c, 0x6f, 0xb9, 0x2d, 0xcc, 0xc9, 0x26, 0xb5, 0x48, 0x69, 0x5c, 0x1e, 0xf5,
|
||||||
0xdb, 0xa3, 0xa9, 0x44, 0x20, 0x2a, 0xa7, 0x14, 0x7b, 0x7e, 0x2b, 0xc1, 0x84, 0xfa, 0x98, 0x61,
|
0xff, 0x07, 0x8b, 0x12, 0x81, 0xa8, 0x9d, 0x51, 0xec, 0xc5, 0xb5, 0x04, 0x13, 0xea, 0x61, 0x86,
|
||||||
0x07, 0x40, 0x31, 0x52, 0xf3, 0xb0, 0xcd, 0xfc, 0x2d, 0x13, 0xfe, 0x66, 0x8e, 0xed, 0x6f, 0x55,
|
0x1d, 0x00, 0xc5, 0xca, 0xa6, 0x87, 0x6d, 0xe6, 0x1f, 0x99, 0xb0, 0x37, 0x71, 0x62, 0x7b, 0x8b,
|
||||||
0xf9, 0x83, 0x5b, 0x29, 0x36, 0x34, 0xc0, 0x43, 0x69, 0x4f, 0x03, 0xf9, 0xe8, 0xc0, 0xb6, 0x28,
|
0xca, 0x1e, 0x5c, 0x4b, 0xb1, 0xa1, 0x0c, 0x0b, 0x95, 0x3d, 0x0d, 0x14, 0xa3, 0x0b, 0x5b, 0xa3,
|
||||||
0xe3, 0xf0, 0x69, 0x2a, 0x2d, 0xf4, 0xd1, 0x02, 0x10, 0x68, 0x99, 0x14, 0x27, 0x55, 0x10, 0xb9,
|
0x8c, 0xc3, 0xe7, 0xa9, 0xb4, 0xd0, 0x07, 0x73, 0x40, 0xa0, 0x65, 0x52, 0x9c, 0x56, 0x4e, 0x14,
|
||||||
0x60, 0x24, 0x96, 0x12, 0x9f, 0x81, 0x2c, 0xe5, 0xc4, 0x12, 0xc7, 0x3f, 0x75, 0x7e, 0x6e, 0xfd,
|
0x82, 0x95, 0x58, 0x4a, 0x7c, 0x0a, 0xf2, 0x94, 0x13, 0x4b, 0x5c, 0xff, 0xd8, 0xc5, 0xa9, 0xe5,
|
||||||
0xca, 0x18, 0xb2, 0xad, 0xcc, 0x2b, 0xfe, 0xec, 0x7d, 0xc1, 0x84, 0x7c, 0xc2, 0xd2, 0xb7, 0x53,
|
0x6b, 0x43, 0x84, 0x6d, 0x6d, 0x5a, 0xf1, 0xe7, 0x1f, 0x09, 0x26, 0xe4, 0x13, 0x56, 0xbe, 0x19,
|
||||||
0x00, 0x46, 0x46, 0xc8, 0x31, 0xcd, 0x3a, 0x6e, 0xb4, 0xe0, 0x19, 0x90, 0xb1, 0xb1, 0x15, 0xa8,
|
0x03, 0x30, 0x52, 0x42, 0x8e, 0x69, 0x36, 0x70, 0x73, 0x07, 0x9e, 0x03, 0x39, 0x1b, 0x5b, 0x41,
|
||||||
0x35, 0x4c, 0xa5, 0x4f, 0xb1, 0x45, 0x90, 0x9c, 0x81, 0x3f, 0x6a, 0x00, 0xb6, 0xe5, 0x51, 0x34,
|
0xb4, 0x86, 0xa9, 0xf4, 0x09, 0xb6, 0x08, 0x92, 0x12, 0xf8, 0x83, 0x06, 0x60, 0x5b, 0x5e, 0x45,
|
||||||
0x37, 0x6d, 0xdb, 0xe1, 0x58, 0xec, 0x4e, 0x10, 0x60, 0x75, 0x8c, 0x00, 0x03, 0xdf, 0xfa, 0x4e,
|
0x6b, 0xd5, 0xb6, 0x1d, 0x8e, 0xc5, 0xe9, 0x04, 0x0e, 0xd6, 0x87, 0x70, 0x30, 0xb0, 0xad, 0x6f,
|
||||||
0x8a, 0xf5, 0xae, 0xcd, 0xbd, 0x6e, 0x74, 0x4a, 0x69, 0x03, 0x34, 0x20, 0x14, 0xd8, 0x02, 0xc0,
|
0xa5, 0x58, 0x1f, 0xd8, 0xdc, 0xeb, 0x46, 0xb7, 0x94, 0x56, 0x40, 0x19, 0xae, 0xc0, 0x1d, 0x00,
|
||||||
0x53, 0x9c, 0x35, 0x47, 0x25, 0xfc, 0xe8, 0xd5, 0x24, 0x08, 0xe7, 0xb6, 0x63, 0x3f, 0xa3, 0x46,
|
0x3c, 0xc5, 0xb9, 0xe9, 0xa8, 0x84, 0x1f, 0xbc, 0x9a, 0x04, 0xee, 0xdc, 0x73, 0xec, 0x17, 0xd4,
|
||||||
0x54, 0xb2, 0x50, 0x48, 0x89, 0x62, 0xf4, 0xab, 0x77, 0xc1, 0xca, 0x90, 0xb8, 0xe1, 0x49, 0x30,
|
0x88, 0x4a, 0x16, 0x0a, 0x29, 0x51, 0x8c, 0x7e, 0xf1, 0x01, 0x58, 0xe8, 0xe3, 0x37, 0x3c, 0x0d,
|
||||||
0xd5, 0x22, 0x5d, 0x7f, 0x2b, 0x91, 0xf8, 0x09, 0x97, 0x41, 0xb6, 0x83, 0xcd, 0x36, 0xf1, 0xb3,
|
0xc6, 0x76, 0x48, 0xd7, 0x3f, 0x4a, 0x24, 0x7e, 0xc2, 0x79, 0x90, 0xef, 0x60, 0xb3, 0x4d, 0xfc,
|
||||||
0x19, 0xf9, 0x1f, 0x37, 0x27, 0xaf, 0x6b, 0xa5, 0x3f, 0xb3, 0x71, 0x65, 0x89, 0xca, 0x05, 0xcf,
|
0x6c, 0x46, 0xfe, 0xc7, 0x9d, 0xd1, 0x5b, 0x5a, 0xe5, 0x8f, 0x7c, 0x3c, 0xb2, 0x44, 0xe5, 0x82,
|
||||||
0x83, 0x9c, 0x47, 0x5c, 0x93, 0x36, 0x30, 0x93, 0x1c, 0xd9, 0xca, 0x09, 0xa1, 0x12, 0xa4, 0xc6,
|
0x17, 0x41, 0xc1, 0x23, 0xae, 0x49, 0x9b, 0x98, 0x49, 0x8e, 0x7c, 0xed, 0x94, 0x88, 0x12, 0xa4,
|
||||||
0x50, 0x38, 0x0b, 0xbf, 0x00, 0x39, 0x46, 0x4c, 0xd2, 0xe0, 0x8e, 0xa7, 0x8a, 0xe7, 0x95, 0x11,
|
0xd6, 0x50, 0x28, 0x85, 0x9f, 0x83, 0x02, 0x23, 0x26, 0x69, 0x72, 0xc7, 0x53, 0xc5, 0xf3, 0xda,
|
||||||
0x35, 0x88, 0xeb, 0xc4, 0xac, 0x2a, 0xa8, 0x4f, 0x1f, 0x7c, 0xa1, 0x90, 0x12, 0x7e, 0x0e, 0x72,
|
0x80, 0x31, 0x88, 0x1b, 0xc4, 0xac, 0x2b, 0xa8, 0x4f, 0x1f, 0x7c, 0xa1, 0x90, 0x12, 0x7e, 0x06,
|
||||||
0x9c, 0x58, 0xae, 0x89, 0x39, 0x51, 0xbb, 0x79, 0x69, 0xf8, 0x6e, 0x0a, 0xda, 0x87, 0x4e, 0xb3,
|
0x0a, 0x9c, 0x58, 0xae, 0x89, 0x39, 0x51, 0xa7, 0x79, 0xa5, 0xff, 0x69, 0x0a, 0xda, 0x0d, 0xa7,
|
||||||
0xa6, 0x00, 0xb2, 0x22, 0x87, 0x0a, 0x0f, 0x46, 0x51, 0x48, 0x08, 0x29, 0xc8, 0x31, 0x2e, 0xae,
|
0xb5, 0xa9, 0x00, 0xb2, 0x22, 0x87, 0x11, 0x1e, 0xac, 0xa2, 0x90, 0x10, 0x52, 0x50, 0x60, 0x5c,
|
||||||
0x1d, 0xa3, 0x2b, 0x6b, 0xd1, 0xdc, 0xfa, 0xc6, 0x58, 0xb5, 0xd9, 0xa7, 0x88, 0x5c, 0x05, 0x23,
|
0x3c, 0x3b, 0x46, 0x57, 0xd6, 0xa2, 0xa9, 0xe5, 0x95, 0xa1, 0x6a, 0xb3, 0x4f, 0x11, 0x99, 0x0a,
|
||||||
0x28, 0xa4, 0x87, 0x9b, 0x60, 0xc1, 0xa2, 0x36, 0x22, 0xb8, 0xd9, 0xad, 0x92, 0x86, 0x63, 0x37,
|
0x56, 0x50, 0x48, 0x0f, 0x57, 0xc1, 0x8c, 0x45, 0x6d, 0x44, 0x70, 0xab, 0x5b, 0x27, 0x4d, 0xc7,
|
||||||
0x99, 0x2c, 0x6a, 0xd9, 0xca, 0x8a, 0x02, 0x2d, 0x6c, 0x27, 0xa7, 0x51, 0xbf, 0x3d, 0xdc, 0x02,
|
0x6e, 0x31, 0x59, 0xd4, 0xf2, 0xb5, 0x05, 0x05, 0x9a, 0x59, 0x4f, 0x8a, 0x51, 0xaf, 0x3e, 0x5c,
|
||||||
0xcb, 0x1e, 0xe9, 0x50, 0x71, 0x71, 0x7e, 0x4c, 0x19, 0x77, 0xbc, 0xee, 0x16, 0xb5, 0x28, 0x97,
|
0x03, 0xf3, 0x1e, 0xe9, 0x50, 0xf1, 0x70, 0x7e, 0x44, 0x19, 0x77, 0xbc, 0xee, 0x1a, 0xb5, 0x28,
|
||||||
0xa5, 0x2e, 0x5b, 0x29, 0xf4, 0xf6, 0xd7, 0x96, 0xd1, 0x80, 0x79, 0x34, 0x10, 0x25, 0xaa, 0xb0,
|
0x97, 0xa5, 0x2e, 0x5f, 0x2b, 0x1d, 0xee, 0x2f, 0xcd, 0xa3, 0x0c, 0x39, 0xca, 0x44, 0x89, 0x2a,
|
||||||
0x8b, 0xdb, 0x8c, 0x34, 0x65, 0xe9, 0xca, 0x45, 0x55, 0xf8, 0xa1, 0x1c, 0x45, 0x6a, 0x16, 0x1a,
|
0xec, 0xe2, 0x36, 0x23, 0x2d, 0x59, 0xba, 0x0a, 0x51, 0x15, 0xde, 0x90, 0xab, 0x48, 0x49, 0xa1,
|
||||||
0x09, 0x41, 0xe7, 0xfe, 0x99, 0xa0, 0xf3, 0xc3, 0xc5, 0x0c, 0x77, 0xc0, 0x8a, 0xeb, 0x39, 0x86,
|
0x91, 0x08, 0xe8, 0xc2, 0xdf, 0x0b, 0xe8, 0x62, 0xff, 0x60, 0x86, 0x5b, 0x60, 0xc1, 0xf5, 0x1c,
|
||||||
0x47, 0x18, 0xbb, 0x43, 0x70, 0xd3, 0xa4, 0x36, 0x09, 0x76, 0x6a, 0x56, 0xae, 0xf0, 0x74, 0x6f,
|
0xc3, 0x23, 0x8c, 0xdd, 0x27, 0xb8, 0x65, 0x52, 0x9b, 0x04, 0x27, 0x35, 0x29, 0x77, 0x78, 0xf6,
|
||||||
0x7f, 0x6d, 0xe5, 0xe1, 0x60, 0x13, 0x34, 0x0c, 0x5b, 0xfa, 0x3e, 0x03, 0x4e, 0xf6, 0xdf, 0xa3,
|
0x70, 0x7f, 0x69, 0x61, 0x23, 0x5b, 0x05, 0xf5, 0xc3, 0x56, 0xbe, 0xcb, 0x81, 0xd3, 0xbd, 0xef,
|
||||||
0xf0, 0x13, 0x00, 0x9d, 0xba, 0xec, 0x7d, 0x9a, 0xf7, 0xfc, 0xce, 0x83, 0x3a, 0xb6, 0x14, 0xfa,
|
0x28, 0xfc, 0x18, 0x40, 0xa7, 0x21, 0x7b, 0x9f, 0xd6, 0x43, 0xbf, 0xf3, 0xa0, 0x8e, 0x2d, 0x03,
|
||||||
0x54, 0x94, 0xf1, 0x0f, 0x52, 0x16, 0x68, 0x00, 0x0a, 0x5e, 0x8c, 0xa5, 0xca, 0xa4, 0x0c, 0x34,
|
0x7d, 0x2c, 0xca, 0xf8, 0xc7, 0x29, 0x0d, 0x94, 0x81, 0x82, 0x97, 0x63, 0xa9, 0x32, 0x2a, 0x1d,
|
||||||
0xd4, 0xc1, 0x80, 0x74, 0xd9, 0x04, 0x0b, 0xaa, 0x6a, 0x04, 0x93, 0x52, 0xd6, 0x31, 0x1d, 0xec,
|
0x0d, 0xe3, 0x20, 0x23, 0x5d, 0x56, 0xc1, 0x8c, 0xaa, 0x1a, 0x81, 0x50, 0x86, 0x75, 0x2c, 0x0e,
|
||||||
0x24, 0xa7, 0x51, 0xbf, 0x3d, 0xbc, 0x07, 0x16, 0x71, 0x07, 0x53, 0x13, 0xd7, 0x4d, 0x12, 0x92,
|
0xb6, 0x92, 0x62, 0xd4, 0xab, 0x0f, 0x1f, 0x82, 0x59, 0xdc, 0xc1, 0xd4, 0xc4, 0x0d, 0x93, 0x84,
|
||||||
0x64, 0x24, 0xc9, 0xff, 0x15, 0xc9, 0xe2, 0x66, 0xbf, 0x01, 0x4a, 0x63, 0xe0, 0x36, 0x58, 0x6a,
|
0x24, 0x39, 0x49, 0xf2, 0x5f, 0x45, 0x32, 0xbb, 0xda, 0xab, 0x80, 0xd2, 0x18, 0xb8, 0x0e, 0xe6,
|
||||||
0xdb, 0x69, 0x2a, 0x5f, 0x97, 0xa7, 0x15, 0xd5, 0xd2, 0x4e, 0xda, 0x04, 0x0d, 0xc2, 0x41, 0x17,
|
0xda, 0x76, 0x9a, 0xca, 0x8f, 0xcb, 0xb3, 0x8a, 0x6a, 0x6e, 0x2b, 0xad, 0x82, 0xb2, 0x70, 0xd0,
|
||||||
0x80, 0x46, 0x70, 0xe5, 0xb3, 0xc2, 0xb4, 0xac, 0xc9, 0xef, 0x8f, 0x91, 0x4f, 0x61, 0xdf, 0x10,
|
0x05, 0xa0, 0x19, 0x3c, 0xf9, 0xac, 0x34, 0x2e, 0x6b, 0xf2, 0x7b, 0x43, 0xe4, 0x53, 0xd8, 0x37,
|
||||||
0xd5, 0xbf, 0x70, 0x88, 0xa1, 0x98, 0x0f, 0xb8, 0x01, 0xe6, 0x3d, 0x91, 0x21, 0x61, 0xe8, 0x33,
|
0x44, 0xf5, 0x2f, 0x5c, 0x62, 0x28, 0x66, 0x03, 0xae, 0x80, 0x69, 0x4f, 0x64, 0x48, 0xe8, 0xfa,
|
||||||
0x32, 0xf4, 0xff, 0x29, 0xd8, 0x3c, 0x8a, 0x4f, 0xa2, 0xa4, 0x6d, 0xe9, 0x77, 0x2d, 0x7e, 0x09,
|
0x84, 0x74, 0xfd, 0x3f, 0x0a, 0x36, 0x8d, 0xe2, 0x42, 0x94, 0xd4, 0xad, 0xfc, 0xa6, 0xc5, 0x1f,
|
||||||
0x05, 0x29, 0x0b, 0x6f, 0x26, 0x5a, 0xa6, 0xb3, 0x7d, 0x2d, 0xd3, 0xa9, 0x34, 0x22, 0xd6, 0x31,
|
0xa1, 0x20, 0x65, 0xe1, 0x9d, 0x44, 0xcb, 0x74, 0xbe, 0xa7, 0x65, 0x3a, 0x93, 0x46, 0xc4, 0x3a,
|
||||||
0x75, 0xc1, 0xbc, 0x10, 0x34, 0xb5, 0x0d, 0xff, 0x10, 0x55, 0x41, 0xfc, 0xf0, 0x58, 0xe9, 0x12,
|
0xa6, 0x2e, 0x98, 0x16, 0x01, 0x4d, 0x6d, 0xc3, 0xbf, 0x44, 0x55, 0x10, 0x3f, 0x38, 0x51, 0xba,
|
||||||
0xa2, 0x63, 0xd7, 0xe8, 0xa2, 0x5c, 0x4d, 0x7c, 0x12, 0x25, 0x3d, 0x95, 0x6e, 0x81, 0x7c, 0x32,
|
0x84, 0xe8, 0xd8, 0x33, 0x3a, 0x2b, 0x77, 0x13, 0x17, 0xa2, 0xa4, 0xa5, 0xca, 0x5d, 0x50, 0x4c,
|
||||||
0xd7, 0x7c, 0x5d, 0xfa, 0x89, 0xaf, 0x94, 0x1d, 0xd3, 0xa5, 0x3f, 0x8e, 0x42, 0x8b, 0xd2, 0x2b,
|
0xe6, 0x9a, 0x1f, 0x97, 0x7e, 0xe2, 0xab, 0xc8, 0x8e, 0xc5, 0xa5, 0xbf, 0x8e, 0x42, 0x8d, 0xca,
|
||||||
0x0d, 0xac, 0x0c, 0xf1, 0x0e, 0x4d, 0x90, 0xb7, 0xf0, 0x6e, 0x4c, 0x07, 0x47, 0xf6, 0xe0, 0xe2,
|
0x6b, 0x0d, 0x2c, 0xf4, 0xb1, 0x0e, 0x4d, 0x50, 0xb4, 0xf0, 0x6e, 0x2c, 0x0e, 0x8e, 0xed, 0xc1,
|
||||||
0xf5, 0xa1, 0xfb, 0xaf, 0x0f, 0xfd, 0xbe, 0xcd, 0x1f, 0x78, 0x55, 0xee, 0x51, 0xdb, 0xa8, 0x40,
|
0xc5, 0xf4, 0xa1, 0xfb, 0xd3, 0x87, 0xfe, 0xc8, 0xe6, 0x8f, 0xbd, 0x3a, 0xf7, 0xa8, 0x6d, 0xd4,
|
||||||
0xd1, 0x5f, 0x6d, 0x27, 0xb8, 0x50, 0x1f, 0x37, 0x7c, 0x02, 0x72, 0x16, 0xde, 0xad, 0xb6, 0x3d,
|
0xa0, 0xe8, 0xaf, 0xd6, 0x13, 0x5c, 0xa8, 0x87, 0x1b, 0x3e, 0x03, 0x05, 0x0b, 0xef, 0xd6, 0xdb,
|
||||||
0x23, 0xd8, 0xbf, 0xe3, 0xfb, 0x91, 0xb7, 0xc9, 0xb6, 0x62, 0x41, 0x21, 0x5f, 0xe9, 0x87, 0x49,
|
0x9e, 0x11, 0x9c, 0xdf, 0xc9, 0xed, 0xc8, 0xd7, 0x64, 0x5d, 0xb1, 0xa0, 0x90, 0xaf, 0xf2, 0xfd,
|
||||||
0x90, 0xad, 0x36, 0xb0, 0x49, 0xde, 0xc0, 0x8b, 0xa2, 0x96, 0x78, 0x51, 0xac, 0x8f, 0xac, 0x01,
|
0x28, 0xc8, 0xd7, 0x9b, 0xd8, 0x24, 0x6f, 0x61, 0xa2, 0xd8, 0x4c, 0x4c, 0x14, 0xcb, 0x03, 0xc7,
|
||||||
0x19, 0xdf, 0xd0, 0xc7, 0xc4, 0xd3, 0xbe, 0xc7, 0xc4, 0xd5, 0x63, 0xf2, 0x1e, 0xfe, 0x8e, 0xb8,
|
0x80, 0xf4, 0xaf, 0xef, 0x30, 0xf1, 0xbc, 0x67, 0x98, 0xb8, 0x7e, 0x42, 0xde, 0xa3, 0xe7, 0x88,
|
||||||
0x01, 0x66, 0x43, 0xf7, 0x89, 0xc2, 0xa6, 0x1d, 0x55, 0xd8, 0x4a, 0x3f, 0x4d, 0x82, 0xb9, 0x98,
|
0xdb, 0x60, 0x32, 0x34, 0x9f, 0x28, 0x6c, 0xda, 0x71, 0x85, 0xad, 0xf2, 0xd3, 0x28, 0x98, 0x8a,
|
||||||
0x8b, 0xe3, 0xa1, 0xa1, 0x9b, 0xe8, 0x22, 0x44, 0xe5, 0xa8, 0x8c, 0xb3, 0x30, 0x3d, 0xe8, 0x20,
|
0x99, 0x38, 0x19, 0x1a, 0xba, 0x89, 0x2e, 0x42, 0x54, 0x8e, 0xda, 0x30, 0x1b, 0xd3, 0x83, 0x0e,
|
||||||
0xfc, 0xe6, 0x2d, 0xba, 0x90, 0xd3, 0x8d, 0xc5, 0x2d, 0x90, 0xe7, 0xd8, 0x33, 0x08, 0x0f, 0xe6,
|
0xc2, 0x6f, 0xde, 0xa2, 0x07, 0x39, 0xdd, 0x58, 0xdc, 0x05, 0x45, 0x8e, 0x3d, 0x83, 0xf0, 0x40,
|
||||||
0xe4, 0x86, 0xce, 0x46, 0xcf, 0x80, 0x5a, 0x62, 0x16, 0xf5, 0x59, 0xaf, 0x6e, 0x80, 0xf9, 0x84,
|
0x26, 0x0f, 0x74, 0x32, 0x1a, 0x03, 0x36, 0x13, 0x52, 0xd4, 0xa3, 0xbd, 0xb8, 0x02, 0xa6, 0x13,
|
||||||
0xb3, 0x63, 0x75, 0x5c, 0xbf, 0x88, 0xcd, 0xe2, 0x98, 0x93, 0x67, 0x6d, 0xb3, 0x4a, 0xde, 0xc4,
|
0xc6, 0x4e, 0xd4, 0x71, 0xfd, 0x22, 0x0e, 0x8b, 0x63, 0x4e, 0x5e, 0xb4, 0xcd, 0x3a, 0x79, 0x1b,
|
||||||
0xfb, 0xf6, 0x49, 0x42, 0x8d, 0xd7, 0x47, 0xdf, 0xdc, 0x28, 0xca, 0xa1, 0x9a, 0xac, 0xf7, 0x69,
|
0xf3, 0xed, 0xb3, 0x44, 0x34, 0xde, 0x1a, 0xfc, 0x70, 0x23, 0x2f, 0xfb, 0xc6, 0x64, 0xa3, 0x27,
|
||||||
0xf2, 0xe6, 0x58, 0xec, 0x87, 0x2b, 0xf3, 0x37, 0x0d, 0x2c, 0xc4, 0xac, 0xdf, 0xc0, 0xf3, 0xe7,
|
0x26, 0xef, 0x0c, 0xc5, 0x7e, 0x74, 0x64, 0xfe, 0xaa, 0x81, 0x99, 0x98, 0xf6, 0x5b, 0x18, 0x7f,
|
||||||
0x71, 0xf2, 0xf9, 0x73, 0x75, 0x9c, 0x45, 0x0d, 0x79, 0xff, 0xfc, 0x3a, 0x95, 0x58, 0xcc, 0x7f,
|
0x9e, 0x26, 0xc7, 0x9f, 0xeb, 0xc3, 0x6c, 0xaa, 0xcf, 0xfc, 0xf3, 0x63, 0x2e, 0xb1, 0x99, 0x7f,
|
||||||
0xa8, 0xe3, 0xfe, 0x5a, 0x03, 0xcb, 0x1d, 0xc7, 0x6c, 0x5b, 0xe4, 0xb6, 0x89, 0xa9, 0x15, 0x58,
|
0x51, 0xc7, 0xfd, 0x95, 0x06, 0xe6, 0x3b, 0x8e, 0xd9, 0xb6, 0xc8, 0x3d, 0x13, 0x53, 0x2b, 0xd0,
|
||||||
0x88, 0xfe, 0xe5, 0x88, 0x37, 0xa6, 0xf4, 0x44, 0x3c, 0x46, 0x19, 0x27, 0x36, 0x7f, 0x14, 0x71,
|
0x10, 0xfd, 0xcb, 0x31, 0x33, 0xa6, 0xb4, 0x44, 0x3c, 0x46, 0x19, 0x27, 0x36, 0x7f, 0x12, 0x71,
|
||||||
0x54, 0xde, 0x52, 0xfe, 0x96, 0x1f, 0x0d, 0x20, 0x46, 0x03, 0xdd, 0xc1, 0x77, 0xc1, 0x9c, 0x68,
|
0xd4, 0xfe, 0xa7, 0xec, 0xcd, 0x3f, 0xc9, 0x20, 0x46, 0x99, 0xe6, 0xe0, 0xbb, 0x60, 0x4a, 0x34,
|
||||||
0xe4, 0x68, 0x83, 0x88, 0xd7, 0xa5, 0xfa, 0x7f, 0x61, 0x49, 0x11, 0xcd, 0x55, 0xa3, 0x29, 0x14,
|
0x72, 0xb4, 0x49, 0xc4, 0x74, 0xa9, 0xfe, 0x5f, 0x98, 0x53, 0x44, 0x53, 0xf5, 0x48, 0x84, 0xe2,
|
||||||
0xb7, 0x2b, 0x7d, 0xa3, 0x81, 0xc5, 0x94, 0x66, 0xe1, 0x47, 0x87, 0x74, 0x93, 0xa7, 0xfe, 0xad,
|
0x7a, 0x70, 0x1b, 0xcc, 0xb9, 0x4e, 0x6b, 0x1d, 0xdb, 0xd8, 0x20, 0xe2, 0x69, 0xdc, 0x70, 0x4c,
|
||||||
0x4e, 0xb2, 0x72, 0x61, 0xef, 0xa0, 0x38, 0xf1, 0xf2, 0xa0, 0x38, 0xf1, 0xc7, 0x41, 0x71, 0xe2,
|
0xda, 0xec, 0xca, 0x0e, 0x7c, 0xb2, 0x76, 0x23, 0xe8, 0x98, 0x36, 0xd2, 0x2a, 0x6f, 0x44, 0xeb,
|
||||||
0xab, 0x5e, 0x51, 0xdb, 0xeb, 0x15, 0xb5, 0x97, 0xbd, 0xa2, 0xf6, 0x57, 0xaf, 0xa8, 0x7d, 0xf7,
|
0x9a, 0x5e, 0x96, 0xbd, 0x43, 0x16, 0x65, 0xe5, 0x6b, 0x0d, 0xcc, 0xa6, 0xb2, 0x03, 0x7e, 0x78,
|
||||||
0xaa, 0x38, 0xf1, 0x64, 0x46, 0x29, 0xf2, 0xef, 0x00, 0x00, 0x00, 0xff, 0xff, 0xd6, 0xb9, 0xde,
|
0x44, 0xdf, 0x7a, 0xe6, 0x9f, 0xea, 0x59, 0x6b, 0x97, 0xf6, 0x0e, 0xca, 0x23, 0xaf, 0x0e, 0xca,
|
||||||
0x1a, 0x56, 0x15, 0x00, 0x00,
|
0x23, 0xbf, 0x1f, 0x94, 0x47, 0xbe, 0x3c, 0x2c, 0x6b, 0x7b, 0x87, 0x65, 0xed, 0xd5, 0x61, 0x59,
|
||||||
|
0xfb, 0xf3, 0xb0, 0xac, 0x7d, 0xfb, 0xba, 0x3c, 0xf2, 0x6c, 0x42, 0xc5, 0xfe, 0x5f, 0x01, 0x00,
|
||||||
|
0x00, 0xff, 0xff, 0xdf, 0x80, 0xf3, 0x34, 0xc0, 0x15, 0x00, 0x00,
|
||||||
}
|
}
|
||||||
|
@@ -294,7 +294,7 @@ message StatefulSetList {
|
|||||||
|
|
||||||
// A StatefulSetSpec is the specification of a StatefulSet.
|
// A StatefulSetSpec is the specification of a StatefulSet.
|
||||||
message StatefulSetSpec {
|
message StatefulSetSpec {
|
||||||
// Replicas is the desired number of replicas of the given Template.
|
// replicas is the desired number of replicas of the given Template.
|
||||||
// These are replicas in the sense that they are instantiations of the
|
// These are replicas in the sense that they are instantiations of the
|
||||||
// same Template, but individual replicas also have a consistent identity.
|
// same Template, but individual replicas also have a consistent identity.
|
||||||
// If unspecified, defaults to 1.
|
// If unspecified, defaults to 1.
|
||||||
@@ -302,19 +302,19 @@ message StatefulSetSpec {
|
|||||||
// +optional
|
// +optional
|
||||||
optional int32 replicas = 1;
|
optional int32 replicas = 1;
|
||||||
|
|
||||||
// Selector is a label query over pods that should match the replica count.
|
// selector is a label query over pods that should match the replica count.
|
||||||
// If empty, defaulted to labels on the pod template.
|
// If empty, defaulted to labels on the pod template.
|
||||||
// More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors
|
// More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors
|
||||||
// +optional
|
// +optional
|
||||||
optional k8s.io.apimachinery.pkg.apis.meta.v1.LabelSelector selector = 2;
|
optional k8s.io.apimachinery.pkg.apis.meta.v1.LabelSelector selector = 2;
|
||||||
|
|
||||||
// Template is the object that describes the pod that will be created if
|
// template is the object that describes the pod that will be created if
|
||||||
// insufficient replicas are detected. Each pod stamped out by the StatefulSet
|
// insufficient replicas are detected. Each pod stamped out by the StatefulSet
|
||||||
// will fulfill this Template, but have a unique identity from the rest
|
// will fulfill this Template, but have a unique identity from the rest
|
||||||
// of the StatefulSet.
|
// of the StatefulSet.
|
||||||
optional k8s.io.kubernetes.pkg.api.v1.PodTemplateSpec template = 3;
|
optional k8s.io.kubernetes.pkg.api.v1.PodTemplateSpec template = 3;
|
||||||
|
|
||||||
// VolumeClaimTemplates is a list of claims that pods are allowed to reference.
|
// volumeClaimTemplates is a list of claims that pods are allowed to reference.
|
||||||
// The StatefulSet controller is responsible for mapping network identities to
|
// The StatefulSet controller is responsible for mapping network identities to
|
||||||
// claims in a way that maintains the identity of a pod. Every claim in
|
// claims in a way that maintains the identity of a pod. Every claim in
|
||||||
// this list must have at least one matching (by name) volumeMount in one
|
// this list must have at least one matching (by name) volumeMount in one
|
||||||
@@ -324,21 +324,32 @@ message StatefulSetSpec {
|
|||||||
// +optional
|
// +optional
|
||||||
repeated k8s.io.kubernetes.pkg.api.v1.PersistentVolumeClaim volumeClaimTemplates = 4;
|
repeated k8s.io.kubernetes.pkg.api.v1.PersistentVolumeClaim volumeClaimTemplates = 4;
|
||||||
|
|
||||||
// ServiceName is the name of the service that governs this StatefulSet.
|
// serviceName is the name of the service that governs this StatefulSet.
|
||||||
// This service must exist before the StatefulSet, and is responsible for
|
// This service must exist before the StatefulSet, and is responsible for
|
||||||
// the network identity of the set. Pods get DNS/hostnames that follow the
|
// the network identity of the set. Pods get DNS/hostnames that follow the
|
||||||
// pattern: pod-specific-string.serviceName.default.svc.cluster.local
|
// pattern: pod-specific-string.serviceName.default.svc.cluster.local
|
||||||
// where "pod-specific-string" is managed by the StatefulSet controller.
|
// where "pod-specific-string" is managed by the StatefulSet controller.
|
||||||
optional string serviceName = 5;
|
optional string serviceName = 5;
|
||||||
|
|
||||||
|
// podManagementPolicy controls how pods are created during initial scale up,
|
||||||
|
// when replacing pods on nodes, or when scaling down. The default policy is
|
||||||
|
// `OrderedReady`, where pods are created in increasing order (pod-0, then
|
||||||
|
// pod-1, etc) and the controller will wait until each pod is ready before
|
||||||
|
// continuing. When scaling down, the pods are removed in the opposite order.
|
||||||
|
// The alternative policy is `Parallel` which will create pods in parallel
|
||||||
|
// to match the desired scale without waiting, and on scale down will delete
|
||||||
|
// all pods at once.
|
||||||
|
// +optional
|
||||||
|
optional string podManagementPolicy = 6;
|
||||||
}
|
}
|
||||||
|
|
||||||
// StatefulSetStatus represents the current state of a StatefulSet.
|
// StatefulSetStatus represents the current state of a StatefulSet.
|
||||||
message StatefulSetStatus {
|
message StatefulSetStatus {
|
||||||
// most recent generation observed by this StatefulSet.
|
// observedGeneration is the most recent generation observed by this StatefulSet.
|
||||||
// +optional
|
// +optional
|
||||||
optional int64 observedGeneration = 1;
|
optional int64 observedGeneration = 1;
|
||||||
|
|
||||||
// Replicas is the number of actual replicas.
|
// replicas is the number of actual replicas.
|
||||||
optional int32 replicas = 2;
|
optional int32 replicas = 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -1340,6 +1340,32 @@ func (x *StatefulSet) codecDecodeSelfFromArray(l int, d *codec1978.Decoder) {
|
|||||||
z.DecSendContainerState(codecSelfer_containerArrayEnd1234)
|
z.DecSendContainerState(codecSelfer_containerArrayEnd1234)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (x PodManagementPolicyType) CodecEncodeSelf(e *codec1978.Encoder) {
|
||||||
|
var h codecSelfer1234
|
||||||
|
z, r := codec1978.GenHelperEncoder(e)
|
||||||
|
_, _, _ = h, z, r
|
||||||
|
yym1 := z.EncBinary()
|
||||||
|
_ = yym1
|
||||||
|
if false {
|
||||||
|
} else if z.HasExtensions() && z.EncExt(x) {
|
||||||
|
} else {
|
||||||
|
r.EncodeString(codecSelferC_UTF81234, string(x))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *PodManagementPolicyType) CodecDecodeSelf(d *codec1978.Decoder) {
|
||||||
|
var h codecSelfer1234
|
||||||
|
z, r := codec1978.GenHelperDecoder(d)
|
||||||
|
_, _, _ = h, z, r
|
||||||
|
yym1 := z.DecBinary()
|
||||||
|
_ = yym1
|
||||||
|
if false {
|
||||||
|
} else if z.HasExtensions() && z.DecExt(x) {
|
||||||
|
} else {
|
||||||
|
*((*string)(x)) = r.DecodeString()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func (x *StatefulSetSpec) CodecEncodeSelf(e *codec1978.Encoder) {
|
func (x *StatefulSetSpec) CodecEncodeSelf(e *codec1978.Encoder) {
|
||||||
var h codecSelfer1234
|
var h codecSelfer1234
|
||||||
z, r := codec1978.GenHelperEncoder(e)
|
z, r := codec1978.GenHelperEncoder(e)
|
||||||
@@ -1354,15 +1380,16 @@ func (x *StatefulSetSpec) CodecEncodeSelf(e *codec1978.Encoder) {
|
|||||||
} else {
|
} else {
|
||||||
yysep2 := !z.EncBinary()
|
yysep2 := !z.EncBinary()
|
||||||
yy2arr2 := z.EncBasicHandle().StructToArray
|
yy2arr2 := z.EncBasicHandle().StructToArray
|
||||||
var yyq2 [5]bool
|
var yyq2 [6]bool
|
||||||
_, _, _ = yysep2, yyq2, yy2arr2
|
_, _, _ = yysep2, yyq2, yy2arr2
|
||||||
const yyr2 bool = false
|
const yyr2 bool = false
|
||||||
yyq2[0] = x.Replicas != nil
|
yyq2[0] = x.Replicas != nil
|
||||||
yyq2[1] = x.Selector != nil
|
yyq2[1] = x.Selector != nil
|
||||||
yyq2[3] = len(x.VolumeClaimTemplates) != 0
|
yyq2[3] = len(x.VolumeClaimTemplates) != 0
|
||||||
|
yyq2[5] = x.PodManagementPolicy != ""
|
||||||
var yynn2 int
|
var yynn2 int
|
||||||
if yyr2 || yy2arr2 {
|
if yyr2 || yy2arr2 {
|
||||||
r.EncodeArrayStart(5)
|
r.EncodeArrayStart(6)
|
||||||
} else {
|
} else {
|
||||||
yynn2 = 2
|
yynn2 = 2
|
||||||
for _, b := range yyq2 {
|
for _, b := range yyq2 {
|
||||||
@@ -1506,6 +1533,21 @@ func (x *StatefulSetSpec) CodecEncodeSelf(e *codec1978.Encoder) {
|
|||||||
r.EncodeString(codecSelferC_UTF81234, string(x.ServiceName))
|
r.EncodeString(codecSelferC_UTF81234, string(x.ServiceName))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if yyr2 || yy2arr2 {
|
||||||
|
z.EncSendContainerState(codecSelfer_containerArrayElem1234)
|
||||||
|
if yyq2[5] {
|
||||||
|
x.PodManagementPolicy.CodecEncodeSelf(e)
|
||||||
|
} else {
|
||||||
|
r.EncodeString(codecSelferC_UTF81234, "")
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if yyq2[5] {
|
||||||
|
z.EncSendContainerState(codecSelfer_containerMapKey1234)
|
||||||
|
r.EncodeString(codecSelferC_UTF81234, string("podManagementPolicy"))
|
||||||
|
z.EncSendContainerState(codecSelfer_containerMapValue1234)
|
||||||
|
x.PodManagementPolicy.CodecEncodeSelf(e)
|
||||||
|
}
|
||||||
|
}
|
||||||
if yyr2 || yy2arr2 {
|
if yyr2 || yy2arr2 {
|
||||||
z.EncSendContainerState(codecSelfer_containerArrayEnd1234)
|
z.EncSendContainerState(codecSelfer_containerArrayEnd1234)
|
||||||
} else {
|
} else {
|
||||||
@@ -1631,6 +1673,13 @@ func (x *StatefulSetSpec) codecDecodeSelfFromMap(l int, d *codec1978.Decoder) {
|
|||||||
*((*string)(yyv11)) = r.DecodeString()
|
*((*string)(yyv11)) = r.DecodeString()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
case "podManagementPolicy":
|
||||||
|
if r.TryDecodeAsNil() {
|
||||||
|
x.PodManagementPolicy = ""
|
||||||
|
} else {
|
||||||
|
yyv13 := &x.PodManagementPolicy
|
||||||
|
yyv13.CodecDecodeSelf(d)
|
||||||
|
}
|
||||||
default:
|
default:
|
||||||
z.DecStructFieldNotFound(-1, yys3)
|
z.DecStructFieldNotFound(-1, yys3)
|
||||||
} // end switch yys3
|
} // end switch yys3
|
||||||
@@ -1642,16 +1691,16 @@ func (x *StatefulSetSpec) codecDecodeSelfFromArray(l int, d *codec1978.Decoder)
|
|||||||
var h codecSelfer1234
|
var h codecSelfer1234
|
||||||
z, r := codec1978.GenHelperDecoder(d)
|
z, r := codec1978.GenHelperDecoder(d)
|
||||||
_, _, _ = h, z, r
|
_, _, _ = h, z, r
|
||||||
var yyj13 int
|
var yyj14 int
|
||||||
var yyb13 bool
|
var yyb14 bool
|
||||||
var yyhl13 bool = l >= 0
|
var yyhl14 bool = l >= 0
|
||||||
yyj13++
|
yyj14++
|
||||||
if yyhl13 {
|
if yyhl14 {
|
||||||
yyb13 = yyj13 > l
|
yyb14 = yyj14 > l
|
||||||
} else {
|
} else {
|
||||||
yyb13 = r.CheckBreak()
|
yyb14 = r.CheckBreak()
|
||||||
}
|
}
|
||||||
if yyb13 {
|
if yyb14 {
|
||||||
z.DecSendContainerState(codecSelfer_containerArrayEnd1234)
|
z.DecSendContainerState(codecSelfer_containerArrayEnd1234)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@@ -1664,20 +1713,20 @@ func (x *StatefulSetSpec) codecDecodeSelfFromArray(l int, d *codec1978.Decoder)
|
|||||||
if x.Replicas == nil {
|
if x.Replicas == nil {
|
||||||
x.Replicas = new(int32)
|
x.Replicas = new(int32)
|
||||||
}
|
}
|
||||||
yym15 := z.DecBinary()
|
yym16 := z.DecBinary()
|
||||||
_ = yym15
|
_ = yym16
|
||||||
if false {
|
if false {
|
||||||
} else {
|
} else {
|
||||||
*((*int32)(x.Replicas)) = int32(r.DecodeInt(32))
|
*((*int32)(x.Replicas)) = int32(r.DecodeInt(32))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
yyj13++
|
yyj14++
|
||||||
if yyhl13 {
|
if yyhl14 {
|
||||||
yyb13 = yyj13 > l
|
yyb14 = yyj14 > l
|
||||||
} else {
|
} else {
|
||||||
yyb13 = r.CheckBreak()
|
yyb14 = r.CheckBreak()
|
||||||
}
|
}
|
||||||
if yyb13 {
|
if yyb14 {
|
||||||
z.DecSendContainerState(codecSelfer_containerArrayEnd1234)
|
z.DecSendContainerState(codecSelfer_containerArrayEnd1234)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@@ -1690,21 +1739,21 @@ func (x *StatefulSetSpec) codecDecodeSelfFromArray(l int, d *codec1978.Decoder)
|
|||||||
if x.Selector == nil {
|
if x.Selector == nil {
|
||||||
x.Selector = new(pkg1_v1.LabelSelector)
|
x.Selector = new(pkg1_v1.LabelSelector)
|
||||||
}
|
}
|
||||||
yym17 := z.DecBinary()
|
yym18 := z.DecBinary()
|
||||||
_ = yym17
|
_ = yym18
|
||||||
if false {
|
if false {
|
||||||
} else if z.HasExtensions() && z.DecExt(x.Selector) {
|
} else if z.HasExtensions() && z.DecExt(x.Selector) {
|
||||||
} else {
|
} else {
|
||||||
z.DecFallback(x.Selector, false)
|
z.DecFallback(x.Selector, false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
yyj13++
|
yyj14++
|
||||||
if yyhl13 {
|
if yyhl14 {
|
||||||
yyb13 = yyj13 > l
|
yyb14 = yyj14 > l
|
||||||
} else {
|
} else {
|
||||||
yyb13 = r.CheckBreak()
|
yyb14 = r.CheckBreak()
|
||||||
}
|
}
|
||||||
if yyb13 {
|
if yyb14 {
|
||||||
z.DecSendContainerState(codecSelfer_containerArrayEnd1234)
|
z.DecSendContainerState(codecSelfer_containerArrayEnd1234)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@@ -1712,16 +1761,16 @@ func (x *StatefulSetSpec) codecDecodeSelfFromArray(l int, d *codec1978.Decoder)
|
|||||||
if r.TryDecodeAsNil() {
|
if r.TryDecodeAsNil() {
|
||||||
x.Template = pkg3_v1.PodTemplateSpec{}
|
x.Template = pkg3_v1.PodTemplateSpec{}
|
||||||
} else {
|
} else {
|
||||||
yyv18 := &x.Template
|
yyv19 := &x.Template
|
||||||
yyv18.CodecDecodeSelf(d)
|
yyv19.CodecDecodeSelf(d)
|
||||||
}
|
}
|
||||||
yyj13++
|
yyj14++
|
||||||
if yyhl13 {
|
if yyhl14 {
|
||||||
yyb13 = yyj13 > l
|
yyb14 = yyj14 > l
|
||||||
} else {
|
} else {
|
||||||
yyb13 = r.CheckBreak()
|
yyb14 = r.CheckBreak()
|
||||||
}
|
}
|
||||||
if yyb13 {
|
if yyb14 {
|
||||||
z.DecSendContainerState(codecSelfer_containerArrayEnd1234)
|
z.DecSendContainerState(codecSelfer_containerArrayEnd1234)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@@ -1729,21 +1778,21 @@ func (x *StatefulSetSpec) codecDecodeSelfFromArray(l int, d *codec1978.Decoder)
|
|||||||
if r.TryDecodeAsNil() {
|
if r.TryDecodeAsNil() {
|
||||||
x.VolumeClaimTemplates = nil
|
x.VolumeClaimTemplates = nil
|
||||||
} else {
|
} else {
|
||||||
yyv19 := &x.VolumeClaimTemplates
|
yyv20 := &x.VolumeClaimTemplates
|
||||||
yym20 := z.DecBinary()
|
yym21 := z.DecBinary()
|
||||||
_ = yym20
|
_ = yym21
|
||||||
if false {
|
if false {
|
||||||
} else {
|
} else {
|
||||||
h.decSlicev1_PersistentVolumeClaim((*[]pkg3_v1.PersistentVolumeClaim)(yyv19), d)
|
h.decSlicev1_PersistentVolumeClaim((*[]pkg3_v1.PersistentVolumeClaim)(yyv20), d)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
yyj13++
|
yyj14++
|
||||||
if yyhl13 {
|
if yyhl14 {
|
||||||
yyb13 = yyj13 > l
|
yyb14 = yyj14 > l
|
||||||
} else {
|
} else {
|
||||||
yyb13 = r.CheckBreak()
|
yyb14 = r.CheckBreak()
|
||||||
}
|
}
|
||||||
if yyb13 {
|
if yyb14 {
|
||||||
z.DecSendContainerState(codecSelfer_containerArrayEnd1234)
|
z.DecSendContainerState(codecSelfer_containerArrayEnd1234)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@@ -1751,26 +1800,43 @@ func (x *StatefulSetSpec) codecDecodeSelfFromArray(l int, d *codec1978.Decoder)
|
|||||||
if r.TryDecodeAsNil() {
|
if r.TryDecodeAsNil() {
|
||||||
x.ServiceName = ""
|
x.ServiceName = ""
|
||||||
} else {
|
} else {
|
||||||
yyv21 := &x.ServiceName
|
yyv22 := &x.ServiceName
|
||||||
yym22 := z.DecBinary()
|
yym23 := z.DecBinary()
|
||||||
_ = yym22
|
_ = yym23
|
||||||
if false {
|
if false {
|
||||||
} else {
|
} else {
|
||||||
*((*string)(yyv21)) = r.DecodeString()
|
*((*string)(yyv22)) = r.DecodeString()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
yyj14++
|
||||||
|
if yyhl14 {
|
||||||
|
yyb14 = yyj14 > l
|
||||||
|
} else {
|
||||||
|
yyb14 = r.CheckBreak()
|
||||||
|
}
|
||||||
|
if yyb14 {
|
||||||
|
z.DecSendContainerState(codecSelfer_containerArrayEnd1234)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
z.DecSendContainerState(codecSelfer_containerArrayElem1234)
|
||||||
|
if r.TryDecodeAsNil() {
|
||||||
|
x.PodManagementPolicy = ""
|
||||||
|
} else {
|
||||||
|
yyv24 := &x.PodManagementPolicy
|
||||||
|
yyv24.CodecDecodeSelf(d)
|
||||||
|
}
|
||||||
for {
|
for {
|
||||||
yyj13++
|
yyj14++
|
||||||
if yyhl13 {
|
if yyhl14 {
|
||||||
yyb13 = yyj13 > l
|
yyb14 = yyj14 > l
|
||||||
} else {
|
} else {
|
||||||
yyb13 = r.CheckBreak()
|
yyb14 = r.CheckBreak()
|
||||||
}
|
}
|
||||||
if yyb13 {
|
if yyb14 {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
z.DecSendContainerState(codecSelfer_containerArrayElem1234)
|
z.DecSendContainerState(codecSelfer_containerArrayElem1234)
|
||||||
z.DecStructFieldNotFound(yyj13-1, "")
|
z.DecStructFieldNotFound(yyj14-1, "")
|
||||||
}
|
}
|
||||||
z.DecSendContainerState(codecSelfer_containerArrayEnd1234)
|
z.DecSendContainerState(codecSelfer_containerArrayEnd1234)
|
||||||
}
|
}
|
||||||
@@ -6166,7 +6232,7 @@ func (x codecSelfer1234) decSliceStatefulSet(v *[]StatefulSet, d *codec1978.Deco
|
|||||||
|
|
||||||
yyrg1 := len(yyv1) > 0
|
yyrg1 := len(yyv1) > 0
|
||||||
yyv21 := yyv1
|
yyv21 := yyv1
|
||||||
yyrl1, yyrt1 = z.DecInferLen(yyl1, z.DecBasicHandle().MaxInitLen, 896)
|
yyrl1, yyrt1 = z.DecInferLen(yyl1, z.DecBasicHandle().MaxInitLen, 912)
|
||||||
if yyrt1 {
|
if yyrt1 {
|
||||||
if yyrl1 <= cap(yyv1) {
|
if yyrl1 <= cap(yyv1) {
|
||||||
yyv1 = yyv1[:yyrl1]
|
yyv1 = yyv1[:yyrl1]
|
||||||
|
@@ -95,9 +95,24 @@ type StatefulSet struct {
|
|||||||
Status StatefulSetStatus `json:"status,omitempty" protobuf:"bytes,3,opt,name=status"`
|
Status StatefulSetStatus `json:"status,omitempty" protobuf:"bytes,3,opt,name=status"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// PodManagementPolicyType defines the policy for creating pods under a stateful set.
|
||||||
|
type PodManagementPolicyType string
|
||||||
|
|
||||||
|
const (
|
||||||
|
// OrderedReadyPodManagement will create pods in strictly increasing order on
|
||||||
|
// scale up and strictly decreasing order on scale down, progressing only when
|
||||||
|
// the previous pod is ready or terminated. At most one pod will be changed
|
||||||
|
// at any time.
|
||||||
|
OrderedReadyPodManagement PodManagementPolicyType = "OrderedReady"
|
||||||
|
// ParallelPodManagement will create and delete pods as soon as the stateful set
|
||||||
|
// replica count is changed, and will not wait for pods to be ready or complete
|
||||||
|
// termination.
|
||||||
|
ParallelPodManagement = "Parallel"
|
||||||
|
)
|
||||||
|
|
||||||
// A StatefulSetSpec is the specification of a StatefulSet.
|
// A StatefulSetSpec is the specification of a StatefulSet.
|
||||||
type StatefulSetSpec struct {
|
type StatefulSetSpec struct {
|
||||||
// Replicas is the desired number of replicas of the given Template.
|
// replicas is the desired number of replicas of the given Template.
|
||||||
// These are replicas in the sense that they are instantiations of the
|
// These are replicas in the sense that they are instantiations of the
|
||||||
// same Template, but individual replicas also have a consistent identity.
|
// same Template, but individual replicas also have a consistent identity.
|
||||||
// If unspecified, defaults to 1.
|
// If unspecified, defaults to 1.
|
||||||
@@ -105,19 +120,19 @@ type StatefulSetSpec struct {
|
|||||||
// +optional
|
// +optional
|
||||||
Replicas *int32 `json:"replicas,omitempty" protobuf:"varint,1,opt,name=replicas"`
|
Replicas *int32 `json:"replicas,omitempty" protobuf:"varint,1,opt,name=replicas"`
|
||||||
|
|
||||||
// Selector is a label query over pods that should match the replica count.
|
// selector is a label query over pods that should match the replica count.
|
||||||
// If empty, defaulted to labels on the pod template.
|
// If empty, defaulted to labels on the pod template.
|
||||||
// More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors
|
// More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors
|
||||||
// +optional
|
// +optional
|
||||||
Selector *metav1.LabelSelector `json:"selector,omitempty" protobuf:"bytes,2,opt,name=selector"`
|
Selector *metav1.LabelSelector `json:"selector,omitempty" protobuf:"bytes,2,opt,name=selector"`
|
||||||
|
|
||||||
// Template is the object that describes the pod that will be created if
|
// template is the object that describes the pod that will be created if
|
||||||
// insufficient replicas are detected. Each pod stamped out by the StatefulSet
|
// insufficient replicas are detected. Each pod stamped out by the StatefulSet
|
||||||
// will fulfill this Template, but have a unique identity from the rest
|
// will fulfill this Template, but have a unique identity from the rest
|
||||||
// of the StatefulSet.
|
// of the StatefulSet.
|
||||||
Template v1.PodTemplateSpec `json:"template" protobuf:"bytes,3,opt,name=template"`
|
Template v1.PodTemplateSpec `json:"template" protobuf:"bytes,3,opt,name=template"`
|
||||||
|
|
||||||
// VolumeClaimTemplates is a list of claims that pods are allowed to reference.
|
// volumeClaimTemplates is a list of claims that pods are allowed to reference.
|
||||||
// The StatefulSet controller is responsible for mapping network identities to
|
// The StatefulSet controller is responsible for mapping network identities to
|
||||||
// claims in a way that maintains the identity of a pod. Every claim in
|
// claims in a way that maintains the identity of a pod. Every claim in
|
||||||
// this list must have at least one matching (by name) volumeMount in one
|
// this list must have at least one matching (by name) volumeMount in one
|
||||||
@@ -127,21 +142,32 @@ type StatefulSetSpec struct {
|
|||||||
// +optional
|
// +optional
|
||||||
VolumeClaimTemplates []v1.PersistentVolumeClaim `json:"volumeClaimTemplates,omitempty" protobuf:"bytes,4,rep,name=volumeClaimTemplates"`
|
VolumeClaimTemplates []v1.PersistentVolumeClaim `json:"volumeClaimTemplates,omitempty" protobuf:"bytes,4,rep,name=volumeClaimTemplates"`
|
||||||
|
|
||||||
// ServiceName is the name of the service that governs this StatefulSet.
|
// serviceName is the name of the service that governs this StatefulSet.
|
||||||
// This service must exist before the StatefulSet, and is responsible for
|
// This service must exist before the StatefulSet, and is responsible for
|
||||||
// the network identity of the set. Pods get DNS/hostnames that follow the
|
// the network identity of the set. Pods get DNS/hostnames that follow the
|
||||||
// pattern: pod-specific-string.serviceName.default.svc.cluster.local
|
// pattern: pod-specific-string.serviceName.default.svc.cluster.local
|
||||||
// where "pod-specific-string" is managed by the StatefulSet controller.
|
// where "pod-specific-string" is managed by the StatefulSet controller.
|
||||||
ServiceName string `json:"serviceName" protobuf:"bytes,5,opt,name=serviceName"`
|
ServiceName string `json:"serviceName" protobuf:"bytes,5,opt,name=serviceName"`
|
||||||
|
|
||||||
|
// podManagementPolicy controls how pods are created during initial scale up,
|
||||||
|
// when replacing pods on nodes, or when scaling down. The default policy is
|
||||||
|
// `OrderedReady`, where pods are created in increasing order (pod-0, then
|
||||||
|
// pod-1, etc) and the controller will wait until each pod is ready before
|
||||||
|
// continuing. When scaling down, the pods are removed in the opposite order.
|
||||||
|
// The alternative policy is `Parallel` which will create pods in parallel
|
||||||
|
// to match the desired scale without waiting, and on scale down will delete
|
||||||
|
// all pods at once.
|
||||||
|
// +optional
|
||||||
|
PodManagementPolicy PodManagementPolicyType `json:"podManagementPolicy,omitempty" protobuf:"bytes,6,opt,name=podManagementPolicy,casttype=PodManagementPolicyType"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// StatefulSetStatus represents the current state of a StatefulSet.
|
// StatefulSetStatus represents the current state of a StatefulSet.
|
||||||
type StatefulSetStatus struct {
|
type StatefulSetStatus struct {
|
||||||
// most recent generation observed by this StatefulSet.
|
// observedGeneration is the most recent generation observed by this StatefulSet.
|
||||||
// +optional
|
// +optional
|
||||||
ObservedGeneration *int64 `json:"observedGeneration,omitempty" protobuf:"varint,1,opt,name=observedGeneration"`
|
ObservedGeneration *int64 `json:"observedGeneration,omitempty" protobuf:"varint,1,opt,name=observedGeneration"`
|
||||||
|
|
||||||
// Replicas is the number of actual replicas.
|
// replicas is the number of actual replicas.
|
||||||
Replicas int32 `json:"replicas" protobuf:"varint,2,opt,name=replicas"`
|
Replicas int32 `json:"replicas" protobuf:"varint,2,opt,name=replicas"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -184,11 +184,12 @@ func (StatefulSetList) SwaggerDoc() map[string]string {
|
|||||||
|
|
||||||
var map_StatefulSetSpec = map[string]string{
|
var map_StatefulSetSpec = map[string]string{
|
||||||
"": "A StatefulSetSpec is the specification of a StatefulSet.",
|
"": "A StatefulSetSpec is the specification of a StatefulSet.",
|
||||||
"replicas": "Replicas is the desired number of replicas of the given Template. These are replicas in the sense that they are instantiations of the same Template, but individual replicas also have a consistent identity. If unspecified, defaults to 1.",
|
"replicas": "replicas is the desired number of replicas of the given Template. These are replicas in the sense that they are instantiations of the same Template, but individual replicas also have a consistent identity. If unspecified, defaults to 1.",
|
||||||
"selector": "Selector is a label query over pods that should match the replica count. If empty, defaulted to labels on the pod template. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors",
|
"selector": "selector is a label query over pods that should match the replica count. If empty, defaulted to labels on the pod template. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#label-selectors",
|
||||||
"template": "Template is the object that describes the pod that will be created if insufficient replicas are detected. Each pod stamped out by the StatefulSet will fulfill this Template, but have a unique identity from the rest of the StatefulSet.",
|
"template": "template is the object that describes the pod that will be created if insufficient replicas are detected. Each pod stamped out by the StatefulSet will fulfill this Template, but have a unique identity from the rest of the StatefulSet.",
|
||||||
"volumeClaimTemplates": "VolumeClaimTemplates is a list of claims that pods are allowed to reference. The StatefulSet controller is responsible for mapping network identities to claims in a way that maintains the identity of a pod. Every claim in this list must have at least one matching (by name) volumeMount in one container in the template. A claim in this list takes precedence over any volumes in the template, with the same name.",
|
"volumeClaimTemplates": "volumeClaimTemplates is a list of claims that pods are allowed to reference. The StatefulSet controller is responsible for mapping network identities to claims in a way that maintains the identity of a pod. Every claim in this list must have at least one matching (by name) volumeMount in one container in the template. A claim in this list takes precedence over any volumes in the template, with the same name.",
|
||||||
"serviceName": "ServiceName is the name of the service that governs this StatefulSet. This service must exist before the StatefulSet, and is responsible for the network identity of the set. Pods get DNS/hostnames that follow the pattern: pod-specific-string.serviceName.default.svc.cluster.local where \"pod-specific-string\" is managed by the StatefulSet controller.",
|
"serviceName": "serviceName is the name of the service that governs this StatefulSet. This service must exist before the StatefulSet, and is responsible for the network identity of the set. Pods get DNS/hostnames that follow the pattern: pod-specific-string.serviceName.default.svc.cluster.local where \"pod-specific-string\" is managed by the StatefulSet controller.",
|
||||||
|
"podManagementPolicy": "podManagementPolicy controls how pods are created during initial scale up, when replacing pods on nodes, or when scaling down. The default policy is `OrderedReady`, where pods are created in increasing order (pod-0, then pod-1, etc) and the controller will wait until each pod is ready before continuing. When scaling down, the pods are removed in the opposite order. The alternative policy is `Parallel` which will create pods in parallel to match the desired scale without waiting, and on scale down will delete all pods at once.",
|
||||||
}
|
}
|
||||||
|
|
||||||
func (StatefulSetSpec) SwaggerDoc() map[string]string {
|
func (StatefulSetSpec) SwaggerDoc() map[string]string {
|
||||||
@@ -197,8 +198,8 @@ func (StatefulSetSpec) SwaggerDoc() map[string]string {
|
|||||||
|
|
||||||
var map_StatefulSetStatus = map[string]string{
|
var map_StatefulSetStatus = map[string]string{
|
||||||
"": "StatefulSetStatus represents the current state of a StatefulSet.",
|
"": "StatefulSetStatus represents the current state of a StatefulSet.",
|
||||||
"observedGeneration": "most recent generation observed by this StatefulSet.",
|
"observedGeneration": "observedGeneration is the most recent generation observed by this StatefulSet.",
|
||||||
"replicas": "Replicas is the number of actual replicas.",
|
"replicas": "replicas is the number of actual replicas.",
|
||||||
}
|
}
|
||||||
|
|
||||||
func (StatefulSetStatus) SwaggerDoc() map[string]string {
|
func (StatefulSetStatus) SwaggerDoc() map[string]string {
|
||||||
|
@@ -133,6 +133,7 @@ func autoConvert_v1beta1_StatefulSetSpec_To_apps_StatefulSetSpec(in *StatefulSet
|
|||||||
}
|
}
|
||||||
out.VolumeClaimTemplates = *(*[]api.PersistentVolumeClaim)(unsafe.Pointer(&in.VolumeClaimTemplates))
|
out.VolumeClaimTemplates = *(*[]api.PersistentVolumeClaim)(unsafe.Pointer(&in.VolumeClaimTemplates))
|
||||||
out.ServiceName = in.ServiceName
|
out.ServiceName = in.ServiceName
|
||||||
|
out.PodManagementPolicy = apps.PodManagementPolicyType(in.PodManagementPolicy)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -146,6 +147,7 @@ func autoConvert_apps_StatefulSetSpec_To_v1beta1_StatefulSetSpec(in *apps.Statef
|
|||||||
}
|
}
|
||||||
out.VolumeClaimTemplates = *(*[]api_v1.PersistentVolumeClaim)(unsafe.Pointer(&in.VolumeClaimTemplates))
|
out.VolumeClaimTemplates = *(*[]api_v1.PersistentVolumeClaim)(unsafe.Pointer(&in.VolumeClaimTemplates))
|
||||||
out.ServiceName = in.ServiceName
|
out.ServiceName = in.ServiceName
|
||||||
|
out.PodManagementPolicy = PodManagementPolicyType(in.PodManagementPolicy)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -265,14 +265,21 @@ func (s *StatefulSetTester) GetPodList(ss *apps.StatefulSet) *v1.PodList {
|
|||||||
|
|
||||||
// ConfirmStatefulPodCount asserts that the current number of Pods in ss is count waiting up to timeout for ss to
|
// ConfirmStatefulPodCount asserts that the current number of Pods in ss is count waiting up to timeout for ss to
|
||||||
// to scale to count.
|
// to scale to count.
|
||||||
func (s *StatefulSetTester) ConfirmStatefulPodCount(count int, ss *apps.StatefulSet, timeout time.Duration) {
|
func (s *StatefulSetTester) ConfirmStatefulPodCount(count int, ss *apps.StatefulSet, timeout time.Duration, hard bool) {
|
||||||
start := time.Now()
|
start := time.Now()
|
||||||
deadline := start.Add(timeout)
|
deadline := start.Add(timeout)
|
||||||
for t := time.Now(); t.Before(deadline); t = time.Now() {
|
for t := time.Now(); t.Before(deadline); t = time.Now() {
|
||||||
podList := s.GetPodList(ss)
|
podList := s.GetPodList(ss)
|
||||||
statefulPodCount := len(podList.Items)
|
statefulPodCount := len(podList.Items)
|
||||||
if statefulPodCount != count {
|
if statefulPodCount != count {
|
||||||
Failf("StatefulSet %v scaled unexpectedly scaled to %d -> %d replicas: %+v", ss.Name, count, len(podList.Items), podList)
|
logPodStates(podList.Items)
|
||||||
|
if hard {
|
||||||
|
Failf("StatefulSet %v scaled unexpectedly scaled to %d -> %d replicas", ss.Name, count, len(podList.Items))
|
||||||
|
} else {
|
||||||
|
Logf("StatefulSet %v has not reached scale %d, at %d", ss.Name, count, statefulPodCount)
|
||||||
|
}
|
||||||
|
time.Sleep(1 * time.Second)
|
||||||
|
continue
|
||||||
}
|
}
|
||||||
Logf("Verifying statefulset %v doesn't scale past %d for another %+v", ss.Name, count, deadline.Sub(t))
|
Logf("Verifying statefulset %v doesn't scale past %d for another %+v", ss.Name, count, deadline.Sub(t))
|
||||||
time.Sleep(1 * time.Second)
|
time.Sleep(1 * time.Second)
|
||||||
|
@@ -311,7 +311,7 @@ var _ = framework.KubeDescribe("StatefulSet", func() {
|
|||||||
sst.UpdateReplicas(ss, 1)
|
sst.UpdateReplicas(ss, 1)
|
||||||
|
|
||||||
By("Verifying that the 2nd pod wont be removed if it is not running and ready")
|
By("Verifying that the 2nd pod wont be removed if it is not running and ready")
|
||||||
sst.ConfirmStatefulPodCount(2, ss, 10*time.Second)
|
sst.ConfirmStatefulPodCount(2, ss, 10*time.Second, true)
|
||||||
expectedPodName := ss.Name + "-1"
|
expectedPodName := ss.Name + "-1"
|
||||||
expectedPod, err := f.ClientSet.Core().Pods(ns).Get(expectedPodName, metav1.GetOptions{})
|
expectedPod, err := f.ClientSet.Core().Pods(ns).Get(expectedPodName, metav1.GetOptions{})
|
||||||
Expect(err).NotTo(HaveOccurred())
|
Expect(err).NotTo(HaveOccurred())
|
||||||
@@ -369,7 +369,7 @@ var _ = framework.KubeDescribe("StatefulSet", func() {
|
|||||||
sst.WaitForRunningAndNotReady(*ss.Spec.Replicas, ss)
|
sst.WaitForRunningAndNotReady(*ss.Spec.Replicas, ss)
|
||||||
sst.WaitForStatus(ss, 0)
|
sst.WaitForStatus(ss, 0)
|
||||||
sst.UpdateReplicas(ss, 3)
|
sst.UpdateReplicas(ss, 3)
|
||||||
sst.ConfirmStatefulPodCount(1, ss, 10*time.Second)
|
sst.ConfirmStatefulPodCount(1, ss, 10*time.Second, true)
|
||||||
|
|
||||||
By("Scaling up stateful set " + ssName + " to 3 replicas and waiting until all of them will be running in namespace " + ns)
|
By("Scaling up stateful set " + ssName + " to 3 replicas and waiting until all of them will be running in namespace " + ns)
|
||||||
sst.RestoreProbe(ss, testProbe)
|
sst.RestoreProbe(ss, testProbe)
|
||||||
@@ -400,7 +400,7 @@ var _ = framework.KubeDescribe("StatefulSet", func() {
|
|||||||
sst.WaitForStatus(ss, 0)
|
sst.WaitForStatus(ss, 0)
|
||||||
sst.WaitForRunningAndNotReady(3, ss)
|
sst.WaitForRunningAndNotReady(3, ss)
|
||||||
sst.UpdateReplicas(ss, 0)
|
sst.UpdateReplicas(ss, 0)
|
||||||
sst.ConfirmStatefulPodCount(3, ss, 10*time.Second)
|
sst.ConfirmStatefulPodCount(3, ss, 10*time.Second, true)
|
||||||
|
|
||||||
By("Scaling down stateful set " + ssName + " to 0 replicas and waiting until none of pods will run in namespace" + ns)
|
By("Scaling down stateful set " + ssName + " to 0 replicas and waiting until none of pods will run in namespace" + ns)
|
||||||
sst.RestoreProbe(ss, testProbe)
|
sst.RestoreProbe(ss, testProbe)
|
||||||
@@ -422,6 +422,47 @@ var _ = framework.KubeDescribe("StatefulSet", func() {
|
|||||||
Expect(err).NotTo(HaveOccurred())
|
Expect(err).NotTo(HaveOccurred())
|
||||||
})
|
})
|
||||||
|
|
||||||
|
It("Burst scaling should run to completion even with unhealthy pods", func() {
|
||||||
|
psLabels := klabels.Set(labels)
|
||||||
|
|
||||||
|
By("Creating stateful set " + ssName + " in namespace " + ns)
|
||||||
|
testProbe := &v1.Probe{Handler: v1.Handler{HTTPGet: &v1.HTTPGetAction{
|
||||||
|
Path: "/index.html",
|
||||||
|
Port: intstr.IntOrString{IntVal: 80}}}}
|
||||||
|
ss := framework.NewStatefulSet(ssName, ns, headlessSvcName, 1, nil, nil, psLabels)
|
||||||
|
ss.Spec.PodManagementPolicy = apps.ParallelPodManagement
|
||||||
|
ss.Spec.Template.Spec.Containers[0].ReadinessProbe = testProbe
|
||||||
|
ss, err := c.Apps().StatefulSets(ns).Create(ss)
|
||||||
|
Expect(err).NotTo(HaveOccurred())
|
||||||
|
|
||||||
|
By("Waiting until all stateful set " + ssName + " replicas will be running in namespace " + ns)
|
||||||
|
sst := framework.NewStatefulSetTester(c)
|
||||||
|
sst.WaitForRunningAndReady(*ss.Spec.Replicas, ss)
|
||||||
|
|
||||||
|
By("Confirming that stateful set scale up will not halt with unhealthy stateful pod")
|
||||||
|
sst.BreakProbe(ss, testProbe)
|
||||||
|
sst.WaitForRunningAndNotReady(*ss.Spec.Replicas, ss)
|
||||||
|
sst.WaitForStatus(ss, 0)
|
||||||
|
sst.UpdateReplicas(ss, 3)
|
||||||
|
sst.ConfirmStatefulPodCount(3, ss, 10*time.Second, false)
|
||||||
|
|
||||||
|
By("Scaling up stateful set " + ssName + " to 3 replicas and waiting until all of them will be running in namespace " + ns)
|
||||||
|
sst.RestoreProbe(ss, testProbe)
|
||||||
|
sst.WaitForRunningAndReady(3, ss)
|
||||||
|
|
||||||
|
By("Scale down will not halt with unhealthy stateful pod")
|
||||||
|
sst.BreakProbe(ss, testProbe)
|
||||||
|
sst.WaitForStatus(ss, 0)
|
||||||
|
sst.WaitForRunningAndNotReady(3, ss)
|
||||||
|
sst.UpdateReplicas(ss, 0)
|
||||||
|
sst.ConfirmStatefulPodCount(0, ss, 10*time.Second, false)
|
||||||
|
|
||||||
|
By("Scaling down stateful set " + ssName + " to 0 replicas and waiting until none of pods will run in namespace" + ns)
|
||||||
|
sst.RestoreProbe(ss, testProbe)
|
||||||
|
sst.Scale(ss, 0)
|
||||||
|
sst.WaitForStatus(ss, 0)
|
||||||
|
})
|
||||||
|
|
||||||
It("Should recreate evicted statefulset", func() {
|
It("Should recreate evicted statefulset", func() {
|
||||||
podName := "test-pod"
|
podName := "test-pod"
|
||||||
statefulPodName := ssName + "-0"
|
statefulPodName := ssName + "-0"
|
||||||
|
Reference in New Issue
Block a user