mirror of
https://github.com/k3s-io/kubernetes.git
synced 2025-08-11 21:12:07 +00:00
Merge pull request #54436 from msau42/sc-api
Automatic merge from submit-queue (batch tested with PRs 54436, 53148, 55153, 55614, 55484). If you want to cherry-pick this change to another branch, please follow the instructions <a href="https://github.com/kubernetes/community/blob/master/contributors/devel/cherry-picks.md">here</a>. Add VolumeBindingMode to StorageClass API **What this PR does / why we need it**: Adds a new field `VolumeBindingMode` to `StorageClass`, as specified in kubernetes/community#1168 **Which issue this PR fixes** *(optional, in `fixes #<issue number>(, fixes #<issue_number>, ...)` format, will close that issue when PR gets merged)*: fixes #54434 **Special notes for your reviewer**: API changes only. The scheduler and PV controller work will be submitted as a separate PR. **Release note**: NONE @kubernetes/sig-storage-pr-reviews
This commit is contained in:
commit
5c59d66a41
8
api/openapi-spec/swagger.json
generated
8
api/openapi-spec/swagger.json
generated
@ -80871,6 +80871,10 @@
|
|||||||
"reclaimPolicy": {
|
"reclaimPolicy": {
|
||||||
"description": "Dynamically provisioned PersistentVolumes of this storage class are created with this reclaimPolicy. Defaults to Delete.",
|
"description": "Dynamically provisioned PersistentVolumes of this storage class are created with this reclaimPolicy. Defaults to Delete.",
|
||||||
"type": "string"
|
"type": "string"
|
||||||
|
},
|
||||||
|
"volumeBindingMode": {
|
||||||
|
"description": "VolumeBindingMode indicates how PersistentVolumeClaims should be provisioned and bound. When unset, VolumeBindingImmediate is used. This field is alpha-level and is only honored by servers that enable the VolumeScheduling feature.",
|
||||||
|
"type": "string"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"x-kubernetes-group-version-kind": [
|
"x-kubernetes-group-version-kind": [
|
||||||
@ -81098,6 +81102,10 @@
|
|||||||
"reclaimPolicy": {
|
"reclaimPolicy": {
|
||||||
"description": "Dynamically provisioned PersistentVolumes of this storage class are created with this reclaimPolicy. Defaults to Delete.",
|
"description": "Dynamically provisioned PersistentVolumes of this storage class are created with this reclaimPolicy. Defaults to Delete.",
|
||||||
"type": "string"
|
"type": "string"
|
||||||
|
},
|
||||||
|
"volumeBindingMode": {
|
||||||
|
"description": "VolumeBindingMode indicates how PersistentVolumeClaims should be provisioned and bound. When unset, VolumeBindingImmediate is used. This field is alpha-level and is only honored by servers that enable the VolumeScheduling feature.",
|
||||||
|
"type": "string"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"x-kubernetes-group-version-kind": [
|
"x-kubernetes-group-version-kind": [
|
||||||
|
8
api/swagger-spec/storage.k8s.io_v1.json
generated
8
api/swagger-spec/storage.k8s.io_v1.json
generated
@ -814,6 +814,10 @@
|
|||||||
"allowVolumeExpansion": {
|
"allowVolumeExpansion": {
|
||||||
"type": "boolean",
|
"type": "boolean",
|
||||||
"description": "AllowVolumeExpansion shows whether the storage class allow volume expand"
|
"description": "AllowVolumeExpansion shows whether the storage class allow volume expand"
|
||||||
|
},
|
||||||
|
"volumeBindingMode": {
|
||||||
|
"$ref": "v1.VolumeBindingMode",
|
||||||
|
"description": "VolumeBindingMode indicates how PersistentVolumeClaims should be provisioned and bound. When unset, VolumeBindingImmediate is used. This field is alpha-level and is only honored by servers that enable the VolumeScheduling feature."
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@ -1059,6 +1063,10 @@
|
|||||||
"id": "v1.PersistentVolumeReclaimPolicy",
|
"id": "v1.PersistentVolumeReclaimPolicy",
|
||||||
"properties": {}
|
"properties": {}
|
||||||
},
|
},
|
||||||
|
"v1.VolumeBindingMode": {
|
||||||
|
"id": "v1.VolumeBindingMode",
|
||||||
|
"properties": {}
|
||||||
|
},
|
||||||
"v1.WatchEvent": {
|
"v1.WatchEvent": {
|
||||||
"id": "v1.WatchEvent",
|
"id": "v1.WatchEvent",
|
||||||
"required": [
|
"required": [
|
||||||
|
8
api/swagger-spec/storage.k8s.io_v1beta1.json
generated
8
api/swagger-spec/storage.k8s.io_v1beta1.json
generated
@ -814,6 +814,10 @@
|
|||||||
"allowVolumeExpansion": {
|
"allowVolumeExpansion": {
|
||||||
"type": "boolean",
|
"type": "boolean",
|
||||||
"description": "AllowVolumeExpansion shows whether the storage class allow volume expand"
|
"description": "AllowVolumeExpansion shows whether the storage class allow volume expand"
|
||||||
|
},
|
||||||
|
"volumeBindingMode": {
|
||||||
|
"$ref": "v1beta1.VolumeBindingMode",
|
||||||
|
"description": "VolumeBindingMode indicates how PersistentVolumeClaims should be provisioned and bound. When unset, VolumeBindingImmediate is used. This field is alpha-level and is only honored by servers that enable the VolumeScheduling feature."
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@ -1059,6 +1063,10 @@
|
|||||||
"id": "v1.PersistentVolumeReclaimPolicy",
|
"id": "v1.PersistentVolumeReclaimPolicy",
|
||||||
"properties": {}
|
"properties": {}
|
||||||
},
|
},
|
||||||
|
"v1beta1.VolumeBindingMode": {
|
||||||
|
"id": "v1beta1.VolumeBindingMode",
|
||||||
|
"properties": {}
|
||||||
|
},
|
||||||
"v1.WatchEvent": {
|
"v1.WatchEvent": {
|
||||||
"id": "v1.WatchEvent",
|
"id": "v1.WatchEvent",
|
||||||
"required": [
|
"required": [
|
||||||
|
@ -996,6 +996,13 @@ span.icon > [class^="icon-"], span.icon > [class*=" icon-"] { cursor: default; }
|
|||||||
<td class="tableblock halign-left valign-top"><p class="tableblock">boolean</p></td>
|
<td class="tableblock halign-left valign-top"><p class="tableblock">boolean</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>
|
||||||
</tr>
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td class="tableblock halign-left valign-top"><p class="tableblock">volumeBindingMode</p></td>
|
||||||
|
<td class="tableblock halign-left valign-top"><p class="tableblock">VolumeBindingMode indicates how PersistentVolumeClaims should be provisioned and bound. When unset, VolumeBindingImmediate is used. This field is alpha-level and is only honored by servers that enable the VolumeScheduling feature.</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_volumebindingmode">v1.VolumeBindingMode</a></p></td>
|
||||||
|
<td class="tableblock halign-left valign-top"></td>
|
||||||
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
|
|
||||||
@ -1377,6 +1384,10 @@ Examples:<br>
|
|||||||
<div class="sect2">
|
<div class="sect2">
|
||||||
<h3 id="_v1_deletionpropagation">v1.DeletionPropagation</h3>
|
<h3 id="_v1_deletionpropagation">v1.DeletionPropagation</h3>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
<div class="sect2">
|
||||||
|
<h3 id="_v1_volumebindingmode">v1.VolumeBindingMode</h3>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
<div class="sect2">
|
<div class="sect2">
|
||||||
<h3 id="_any">any</h3>
|
<h3 id="_any">any</h3>
|
||||||
|
@ -958,6 +958,13 @@ span.icon > [class^="icon-"], span.icon > [class*=" icon-"] { cursor: default; }
|
|||||||
<td class="tableblock halign-left valign-top"><p class="tableblock">boolean</p></td>
|
<td class="tableblock halign-left valign-top"><p class="tableblock">boolean</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>
|
||||||
</tr>
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td class="tableblock halign-left valign-top"><p class="tableblock">volumeBindingMode</p></td>
|
||||||
|
<td class="tableblock halign-left valign-top"><p class="tableblock">VolumeBindingMode indicates how PersistentVolumeClaims should be provisioned and bound. When unset, VolumeBindingImmediate is used. This field is alpha-level and is only honored by servers that enable the VolumeScheduling feature.</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="#_v1beta1_volumebindingmode">v1beta1.VolumeBindingMode</a></p></td>
|
||||||
|
<td class="tableblock halign-left valign-top"></td>
|
||||||
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
|
|
||||||
@ -1227,6 +1234,10 @@ When an object is created, the system will populate this list with the current s
|
|||||||
<div class="sect2">
|
<div class="sect2">
|
||||||
<h3 id="_v1_persistentvolumereclaimpolicy">v1.PersistentVolumeReclaimPolicy</h3>
|
<h3 id="_v1_persistentvolumereclaimpolicy">v1.PersistentVolumeReclaimPolicy</h3>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
<div class="sect2">
|
||||||
|
<h3 id="_v1beta1_volumebindingmode">v1beta1.VolumeBindingMode</h3>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
<div class="sect2">
|
<div class="sect2">
|
||||||
<h3 id="_v1_apiresource">v1.APIResource</h3>
|
<h3 id="_v1_apiresource">v1.APIResource</h3>
|
||||||
|
@ -71,10 +71,8 @@ pkg/apis/settings
|
|||||||
pkg/apis/settings/v1alpha1
|
pkg/apis/settings/v1alpha1
|
||||||
pkg/apis/storage
|
pkg/apis/storage
|
||||||
pkg/apis/storage/util
|
pkg/apis/storage/util
|
||||||
pkg/apis/storage/v1
|
|
||||||
pkg/apis/storage/v1/util
|
pkg/apis/storage/v1/util
|
||||||
pkg/apis/storage/v1alpha1
|
pkg/apis/storage/v1alpha1
|
||||||
pkg/apis/storage/v1beta1
|
|
||||||
pkg/apis/storage/v1beta1/util
|
pkg/apis/storage/v1beta1/util
|
||||||
pkg/auth/authorizer/abac
|
pkg/auth/authorizer/abac
|
||||||
pkg/capabilities
|
pkg/capabilities
|
||||||
|
@ -65,6 +65,13 @@ type StorageClass struct {
|
|||||||
// for all PVs created from this storageclass.
|
// for all PVs created from this storageclass.
|
||||||
// +optional
|
// +optional
|
||||||
AllowVolumeExpansion *bool
|
AllowVolumeExpansion *bool
|
||||||
|
|
||||||
|
// VolumeBindingMode indicates how PersistentVolumeClaims should be
|
||||||
|
// provisioned and bound. When unset, VolumeBindingImmediate is used.
|
||||||
|
// This field is alpha-level and is only honored by servers that enable
|
||||||
|
// the VolumeScheduling feature.
|
||||||
|
// +optional
|
||||||
|
VolumeBindingMode *VolumeBindingMode
|
||||||
}
|
}
|
||||||
|
|
||||||
// +k8s:deepcopy-gen:interfaces=k8s.io/apimachinery/pkg/runtime.Object
|
// +k8s:deepcopy-gen:interfaces=k8s.io/apimachinery/pkg/runtime.Object
|
||||||
@ -187,3 +194,18 @@ type VolumeError struct {
|
|||||||
// +optional
|
// +optional
|
||||||
Message string
|
Message string
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// VolumeBindingMode indicates how PersistentVolumeClaims should be bound.
|
||||||
|
type VolumeBindingMode string
|
||||||
|
|
||||||
|
const (
|
||||||
|
// VolumeBindingImmediate indicates that PersistentVolumeClaims should be
|
||||||
|
// immediately provisioned and bound.
|
||||||
|
VolumeBindingImmediate VolumeBindingMode = "Immediate"
|
||||||
|
|
||||||
|
// VolumeBindingWaitForFirstConsumer indicates that PersistentVolumeClaims
|
||||||
|
// should not be provisioned and bound until the first Pod is created that
|
||||||
|
// references the PeristentVolumeClaim. The volume provisioning and
|
||||||
|
// binding will occur during Pod scheduing.
|
||||||
|
VolumeBindingWaitForFirstConsumer VolumeBindingMode = "WaitForFirstConsumer"
|
||||||
|
)
|
||||||
|
@ -3,13 +3,22 @@ package(default_visibility = ["//visibility:public"])
|
|||||||
load(
|
load(
|
||||||
"@io_bazel_rules_go//go:def.bzl",
|
"@io_bazel_rules_go//go:def.bzl",
|
||||||
"go_library",
|
"go_library",
|
||||||
|
"go_test",
|
||||||
)
|
)
|
||||||
|
|
||||||
go_library(
|
go_library(
|
||||||
name = "go_default_library",
|
name = "go_default_library",
|
||||||
srcs = ["helpers.go"],
|
srcs = [
|
||||||
|
"helpers.go",
|
||||||
|
"util.go",
|
||||||
|
],
|
||||||
importpath = "k8s.io/kubernetes/pkg/apis/storage/util",
|
importpath = "k8s.io/kubernetes/pkg/apis/storage/util",
|
||||||
deps = ["//vendor/k8s.io/apimachinery/pkg/apis/meta/v1:go_default_library"],
|
deps = [
|
||||||
|
"//pkg/apis/storage:go_default_library",
|
||||||
|
"//pkg/features:go_default_library",
|
||||||
|
"//vendor/k8s.io/apimachinery/pkg/apis/meta/v1:go_default_library",
|
||||||
|
"//vendor/k8s.io/apiserver/pkg/util/feature:go_default_library",
|
||||||
|
],
|
||||||
)
|
)
|
||||||
|
|
||||||
filegroup(
|
filegroup(
|
||||||
@ -24,3 +33,14 @@ filegroup(
|
|||||||
srcs = [":package-srcs"],
|
srcs = [":package-srcs"],
|
||||||
tags = ["automanaged"],
|
tags = ["automanaged"],
|
||||||
)
|
)
|
||||||
|
|
||||||
|
go_test(
|
||||||
|
name = "go_default_test",
|
||||||
|
srcs = ["util_test.go"],
|
||||||
|
importpath = "k8s.io/kubernetes/pkg/apis/storage/util",
|
||||||
|
library = ":go_default_library",
|
||||||
|
deps = [
|
||||||
|
"//pkg/apis/storage:go_default_library",
|
||||||
|
"//vendor/k8s.io/apiserver/pkg/util/feature:go_default_library",
|
||||||
|
],
|
||||||
|
)
|
||||||
|
30
pkg/apis/storage/util/util.go
Normal file
30
pkg/apis/storage/util/util.go
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
/*
|
||||||
|
Copyright 2017 The Kubernetes Authors.
|
||||||
|
|
||||||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
you may not use this file except in compliance with the License.
|
||||||
|
You may obtain a copy of the License at
|
||||||
|
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package util
|
||||||
|
|
||||||
|
import (
|
||||||
|
utilfeature "k8s.io/apiserver/pkg/util/feature"
|
||||||
|
"k8s.io/kubernetes/pkg/apis/storage"
|
||||||
|
"k8s.io/kubernetes/pkg/features"
|
||||||
|
)
|
||||||
|
|
||||||
|
// DropDisabledAlphaFields removes disabled fields from the StorageClass object.
|
||||||
|
func DropDisabledAlphaFields(class *storage.StorageClass) {
|
||||||
|
if !utilfeature.DefaultFeatureGate.Enabled(features.VolumeScheduling) {
|
||||||
|
class.VolumeBindingMode = nil
|
||||||
|
}
|
||||||
|
}
|
52
pkg/apis/storage/util/util_test.go
Normal file
52
pkg/apis/storage/util/util_test.go
Normal file
@ -0,0 +1,52 @@
|
|||||||
|
/*
|
||||||
|
Copyright 2017 The Kubernetes Authors.
|
||||||
|
|
||||||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
you may not use this file except in compliance with the License.
|
||||||
|
You may obtain a copy of the License at
|
||||||
|
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package util
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
utilfeature "k8s.io/apiserver/pkg/util/feature"
|
||||||
|
"k8s.io/kubernetes/pkg/apis/storage"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestDropAlphaFields(t *testing.T) {
|
||||||
|
bindingMode := storage.VolumeBindingWaitForFirstConsumer
|
||||||
|
|
||||||
|
// Test that field gets dropped when feature gate is not set
|
||||||
|
class := &storage.StorageClass{
|
||||||
|
VolumeBindingMode: &bindingMode,
|
||||||
|
}
|
||||||
|
DropDisabledAlphaFields(class)
|
||||||
|
if class.VolumeBindingMode != nil {
|
||||||
|
t.Errorf("VolumeBindingMode field didn't get dropped: %+v", class.VolumeBindingMode)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Test that field does not get dropped when feature gate is set
|
||||||
|
class = &storage.StorageClass{
|
||||||
|
VolumeBindingMode: &bindingMode,
|
||||||
|
}
|
||||||
|
if err := utilfeature.DefaultFeatureGate.Set("VolumeScheduling=true"); err != nil {
|
||||||
|
t.Fatalf("Failed to set feature gate for VolumeScheduling: %v", err)
|
||||||
|
}
|
||||||
|
DropDisabledAlphaFields(class)
|
||||||
|
if class.VolumeBindingMode != &bindingMode {
|
||||||
|
t.Errorf("VolumeBindingMode field got unexpectantly modified: %+v", class.VolumeBindingMode)
|
||||||
|
}
|
||||||
|
if err := utilfeature.DefaultFeatureGate.Set("VolumeScheduling=false"); err != nil {
|
||||||
|
t.Fatalf("Failed to disable feature gate for VolumeScheduling: %v", err)
|
||||||
|
}
|
||||||
|
}
|
@ -3,6 +3,7 @@ package(default_visibility = ["//visibility:public"])
|
|||||||
load(
|
load(
|
||||||
"@io_bazel_rules_go//go:def.bzl",
|
"@io_bazel_rules_go//go:def.bzl",
|
||||||
"go_library",
|
"go_library",
|
||||||
|
"go_test",
|
||||||
)
|
)
|
||||||
|
|
||||||
go_library(
|
go_library(
|
||||||
@ -18,11 +19,13 @@ go_library(
|
|||||||
deps = [
|
deps = [
|
||||||
"//pkg/apis/core:go_default_library",
|
"//pkg/apis/core:go_default_library",
|
||||||
"//pkg/apis/storage:go_default_library",
|
"//pkg/apis/storage:go_default_library",
|
||||||
|
"//pkg/features:go_default_library",
|
||||||
"//vendor/k8s.io/api/core/v1:go_default_library",
|
"//vendor/k8s.io/api/core/v1:go_default_library",
|
||||||
"//vendor/k8s.io/api/storage/v1:go_default_library",
|
"//vendor/k8s.io/api/storage/v1:go_default_library",
|
||||||
"//vendor/k8s.io/apimachinery/pkg/conversion:go_default_library",
|
"//vendor/k8s.io/apimachinery/pkg/conversion:go_default_library",
|
||||||
"//vendor/k8s.io/apimachinery/pkg/runtime:go_default_library",
|
"//vendor/k8s.io/apimachinery/pkg/runtime:go_default_library",
|
||||||
"//vendor/k8s.io/apimachinery/pkg/runtime/schema:go_default_library",
|
"//vendor/k8s.io/apimachinery/pkg/runtime/schema:go_default_library",
|
||||||
|
"//vendor/k8s.io/apiserver/pkg/util/feature:go_default_library",
|
||||||
],
|
],
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -41,3 +44,16 @@ filegroup(
|
|||||||
],
|
],
|
||||||
tags = ["automanaged"],
|
tags = ["automanaged"],
|
||||||
)
|
)
|
||||||
|
|
||||||
|
go_test(
|
||||||
|
name = "go_default_xtest",
|
||||||
|
srcs = ["defaults_test.go"],
|
||||||
|
importpath = "k8s.io/kubernetes/pkg/apis/storage/v1_test",
|
||||||
|
deps = [
|
||||||
|
"//pkg/api/legacyscheme:go_default_library",
|
||||||
|
"//pkg/apis/storage/install:go_default_library",
|
||||||
|
"//vendor/k8s.io/api/storage/v1:go_default_library",
|
||||||
|
"//vendor/k8s.io/apimachinery/pkg/runtime:go_default_library",
|
||||||
|
"//vendor/k8s.io/apiserver/pkg/util/feature:go_default_library",
|
||||||
|
],
|
||||||
|
)
|
||||||
|
@ -20,6 +20,8 @@ import (
|
|||||||
"k8s.io/api/core/v1"
|
"k8s.io/api/core/v1"
|
||||||
storagev1 "k8s.io/api/storage/v1"
|
storagev1 "k8s.io/api/storage/v1"
|
||||||
"k8s.io/apimachinery/pkg/runtime"
|
"k8s.io/apimachinery/pkg/runtime"
|
||||||
|
utilfeature "k8s.io/apiserver/pkg/util/feature"
|
||||||
|
"k8s.io/kubernetes/pkg/features"
|
||||||
)
|
)
|
||||||
|
|
||||||
func addDefaultingFuncs(scheme *runtime.Scheme) error {
|
func addDefaultingFuncs(scheme *runtime.Scheme) error {
|
||||||
@ -31,4 +33,9 @@ func SetDefaults_StorageClass(obj *storagev1.StorageClass) {
|
|||||||
obj.ReclaimPolicy = new(v1.PersistentVolumeReclaimPolicy)
|
obj.ReclaimPolicy = new(v1.PersistentVolumeReclaimPolicy)
|
||||||
*obj.ReclaimPolicy = v1.PersistentVolumeReclaimDelete
|
*obj.ReclaimPolicy = v1.PersistentVolumeReclaimDelete
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if obj.VolumeBindingMode == nil && utilfeature.DefaultFeatureGate.Enabled(features.VolumeScheduling) {
|
||||||
|
obj.VolumeBindingMode = new(storagev1.VolumeBindingMode)
|
||||||
|
*obj.VolumeBindingMode = storagev1.VolumeBindingImmediate
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
81
pkg/apis/storage/v1/defaults_test.go
Normal file
81
pkg/apis/storage/v1/defaults_test.go
Normal file
@ -0,0 +1,81 @@
|
|||||||
|
/*
|
||||||
|
Copyright 2017 The Kubernetes Authors.
|
||||||
|
|
||||||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
you may not use this file except in compliance with the License.
|
||||||
|
You may obtain a copy of the License at
|
||||||
|
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package v1_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"reflect"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
storagev1 "k8s.io/api/storage/v1"
|
||||||
|
"k8s.io/apimachinery/pkg/runtime"
|
||||||
|
utilfeature "k8s.io/apiserver/pkg/util/feature"
|
||||||
|
"k8s.io/kubernetes/pkg/api/legacyscheme"
|
||||||
|
_ "k8s.io/kubernetes/pkg/apis/storage/install"
|
||||||
|
)
|
||||||
|
|
||||||
|
func roundTrip(t *testing.T, obj runtime.Object) runtime.Object {
|
||||||
|
codec := legacyscheme.Codecs.LegacyCodec(storagev1.SchemeGroupVersion)
|
||||||
|
data, err := runtime.Encode(codec, obj)
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("%v\n %#v", err, obj)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
obj2, err := runtime.Decode(codec, data)
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("%v\nData: %s\nSource: %#v", err, string(data), obj)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
obj3 := reflect.New(reflect.TypeOf(obj).Elem()).Interface().(runtime.Object)
|
||||||
|
err = legacyscheme.Scheme.Convert(obj2, obj3, nil)
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("%v\nSource: %#v", err, obj2)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return obj3
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestSetDefaultVolumeBindingMode(t *testing.T) {
|
||||||
|
class := &storagev1.StorageClass{}
|
||||||
|
|
||||||
|
// When feature gate is disabled, field should not be defaulted
|
||||||
|
output := roundTrip(t, runtime.Object(class)).(*storagev1.StorageClass)
|
||||||
|
if output.VolumeBindingMode != nil {
|
||||||
|
t.Errorf("Expected VolumeBindingMode to not be defaulted, got: %+v", output.VolumeBindingMode)
|
||||||
|
}
|
||||||
|
|
||||||
|
class = &storagev1.StorageClass{}
|
||||||
|
|
||||||
|
err := utilfeature.DefaultFeatureGate.Set("VolumeScheduling=true")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to enable feature gate for VolumeScheduling: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// When feature gate is enabled, field should be defaulted
|
||||||
|
defaultMode := storagev1.VolumeBindingImmediate
|
||||||
|
output = roundTrip(t, runtime.Object(class)).(*storagev1.StorageClass)
|
||||||
|
outMode := output.VolumeBindingMode
|
||||||
|
if outMode == nil {
|
||||||
|
t.Errorf("Expected VolumeBindingMode to be defaulted to: %+v, got: nil", defaultMode)
|
||||||
|
} else if *outMode != defaultMode {
|
||||||
|
t.Errorf("Expected VolumeBindingMode to be defaulted to: %+v, got: %+v", defaultMode, outMode)
|
||||||
|
}
|
||||||
|
|
||||||
|
err = utilfeature.DefaultFeatureGate.Set("VolumeScheduling=false")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to disable feature gate for VolumeScheduling: %v", err)
|
||||||
|
}
|
||||||
|
}
|
@ -16,7 +16,9 @@ limitations under the License.
|
|||||||
|
|
||||||
package util
|
package util
|
||||||
|
|
||||||
import metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
import (
|
||||||
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
||||||
|
)
|
||||||
|
|
||||||
// IsDefaultStorageClassAnnotation represents a StorageClass annotation that
|
// IsDefaultStorageClassAnnotation represents a StorageClass annotation that
|
||||||
// marks a class as the default StorageClass
|
// marks a class as the default StorageClass
|
||||||
|
2
pkg/apis/storage/v1/zz_generated.conversion.go
generated
2
pkg/apis/storage/v1/zz_generated.conversion.go
generated
@ -52,6 +52,7 @@ func autoConvert_v1_StorageClass_To_storage_StorageClass(in *v1.StorageClass, ou
|
|||||||
out.ReclaimPolicy = (*core.PersistentVolumeReclaimPolicy)(unsafe.Pointer(in.ReclaimPolicy))
|
out.ReclaimPolicy = (*core.PersistentVolumeReclaimPolicy)(unsafe.Pointer(in.ReclaimPolicy))
|
||||||
out.MountOptions = *(*[]string)(unsafe.Pointer(&in.MountOptions))
|
out.MountOptions = *(*[]string)(unsafe.Pointer(&in.MountOptions))
|
||||||
out.AllowVolumeExpansion = (*bool)(unsafe.Pointer(in.AllowVolumeExpansion))
|
out.AllowVolumeExpansion = (*bool)(unsafe.Pointer(in.AllowVolumeExpansion))
|
||||||
|
out.VolumeBindingMode = (*storage.VolumeBindingMode)(unsafe.Pointer(in.VolumeBindingMode))
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -67,6 +68,7 @@ func autoConvert_storage_StorageClass_To_v1_StorageClass(in *storage.StorageClas
|
|||||||
out.ReclaimPolicy = (*core_v1.PersistentVolumeReclaimPolicy)(unsafe.Pointer(in.ReclaimPolicy))
|
out.ReclaimPolicy = (*core_v1.PersistentVolumeReclaimPolicy)(unsafe.Pointer(in.ReclaimPolicy))
|
||||||
out.MountOptions = *(*[]string)(unsafe.Pointer(&in.MountOptions))
|
out.MountOptions = *(*[]string)(unsafe.Pointer(&in.MountOptions))
|
||||||
out.AllowVolumeExpansion = (*bool)(unsafe.Pointer(in.AllowVolumeExpansion))
|
out.AllowVolumeExpansion = (*bool)(unsafe.Pointer(in.AllowVolumeExpansion))
|
||||||
|
out.VolumeBindingMode = (*v1.VolumeBindingMode)(unsafe.Pointer(in.VolumeBindingMode))
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3,6 +3,7 @@ package(default_visibility = ["//visibility:public"])
|
|||||||
load(
|
load(
|
||||||
"@io_bazel_rules_go//go:def.bzl",
|
"@io_bazel_rules_go//go:def.bzl",
|
||||||
"go_library",
|
"go_library",
|
||||||
|
"go_test",
|
||||||
)
|
)
|
||||||
|
|
||||||
go_library(
|
go_library(
|
||||||
@ -18,11 +19,13 @@ go_library(
|
|||||||
deps = [
|
deps = [
|
||||||
"//pkg/apis/core:go_default_library",
|
"//pkg/apis/core:go_default_library",
|
||||||
"//pkg/apis/storage:go_default_library",
|
"//pkg/apis/storage:go_default_library",
|
||||||
|
"//pkg/features:go_default_library",
|
||||||
"//vendor/k8s.io/api/core/v1:go_default_library",
|
"//vendor/k8s.io/api/core/v1:go_default_library",
|
||||||
"//vendor/k8s.io/api/storage/v1beta1:go_default_library",
|
"//vendor/k8s.io/api/storage/v1beta1:go_default_library",
|
||||||
"//vendor/k8s.io/apimachinery/pkg/conversion:go_default_library",
|
"//vendor/k8s.io/apimachinery/pkg/conversion:go_default_library",
|
||||||
"//vendor/k8s.io/apimachinery/pkg/runtime:go_default_library",
|
"//vendor/k8s.io/apimachinery/pkg/runtime:go_default_library",
|
||||||
"//vendor/k8s.io/apimachinery/pkg/runtime/schema:go_default_library",
|
"//vendor/k8s.io/apimachinery/pkg/runtime/schema:go_default_library",
|
||||||
|
"//vendor/k8s.io/apiserver/pkg/util/feature:go_default_library",
|
||||||
],
|
],
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -41,3 +44,16 @@ filegroup(
|
|||||||
],
|
],
|
||||||
tags = ["automanaged"],
|
tags = ["automanaged"],
|
||||||
)
|
)
|
||||||
|
|
||||||
|
go_test(
|
||||||
|
name = "go_default_xtest",
|
||||||
|
srcs = ["defaults_test.go"],
|
||||||
|
importpath = "k8s.io/kubernetes/pkg/apis/storage/v1beta1_test",
|
||||||
|
deps = [
|
||||||
|
"//pkg/api/legacyscheme:go_default_library",
|
||||||
|
"//pkg/apis/storage/install:go_default_library",
|
||||||
|
"//vendor/k8s.io/api/storage/v1beta1:go_default_library",
|
||||||
|
"//vendor/k8s.io/apimachinery/pkg/runtime:go_default_library",
|
||||||
|
"//vendor/k8s.io/apiserver/pkg/util/feature:go_default_library",
|
||||||
|
],
|
||||||
|
)
|
||||||
|
@ -20,6 +20,8 @@ import (
|
|||||||
"k8s.io/api/core/v1"
|
"k8s.io/api/core/v1"
|
||||||
storagev1beta1 "k8s.io/api/storage/v1beta1"
|
storagev1beta1 "k8s.io/api/storage/v1beta1"
|
||||||
"k8s.io/apimachinery/pkg/runtime"
|
"k8s.io/apimachinery/pkg/runtime"
|
||||||
|
utilfeature "k8s.io/apiserver/pkg/util/feature"
|
||||||
|
"k8s.io/kubernetes/pkg/features"
|
||||||
)
|
)
|
||||||
|
|
||||||
func addDefaultingFuncs(scheme *runtime.Scheme) error {
|
func addDefaultingFuncs(scheme *runtime.Scheme) error {
|
||||||
@ -31,4 +33,9 @@ func SetDefaults_StorageClass(obj *storagev1beta1.StorageClass) {
|
|||||||
obj.ReclaimPolicy = new(v1.PersistentVolumeReclaimPolicy)
|
obj.ReclaimPolicy = new(v1.PersistentVolumeReclaimPolicy)
|
||||||
*obj.ReclaimPolicy = v1.PersistentVolumeReclaimDelete
|
*obj.ReclaimPolicy = v1.PersistentVolumeReclaimDelete
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if obj.VolumeBindingMode == nil && utilfeature.DefaultFeatureGate.Enabled(features.VolumeScheduling) {
|
||||||
|
obj.VolumeBindingMode = new(storagev1beta1.VolumeBindingMode)
|
||||||
|
*obj.VolumeBindingMode = storagev1beta1.VolumeBindingImmediate
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
81
pkg/apis/storage/v1beta1/defaults_test.go
Normal file
81
pkg/apis/storage/v1beta1/defaults_test.go
Normal file
@ -0,0 +1,81 @@
|
|||||||
|
/*
|
||||||
|
Copyright 2017 The Kubernetes Authors.
|
||||||
|
|
||||||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
you may not use this file except in compliance with the License.
|
||||||
|
You may obtain a copy of the License at
|
||||||
|
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package v1beta1_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"reflect"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
storagev1beta1 "k8s.io/api/storage/v1beta1"
|
||||||
|
"k8s.io/apimachinery/pkg/runtime"
|
||||||
|
utilfeature "k8s.io/apiserver/pkg/util/feature"
|
||||||
|
"k8s.io/kubernetes/pkg/api/legacyscheme"
|
||||||
|
_ "k8s.io/kubernetes/pkg/apis/storage/install"
|
||||||
|
)
|
||||||
|
|
||||||
|
func roundTrip(t *testing.T, obj runtime.Object) runtime.Object {
|
||||||
|
codec := legacyscheme.Codecs.LegacyCodec(storagev1beta1.SchemeGroupVersion)
|
||||||
|
data, err := runtime.Encode(codec, obj)
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("%v\n %#v", err, obj)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
obj2, err := runtime.Decode(codec, data)
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("%v\nData: %s\nSource: %#v", err, string(data), obj)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
obj3 := reflect.New(reflect.TypeOf(obj).Elem()).Interface().(runtime.Object)
|
||||||
|
err = legacyscheme.Scheme.Convert(obj2, obj3, nil)
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("%v\nSource: %#v", err, obj2)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return obj3
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestSetDefaultVolumeBindingMode(t *testing.T) {
|
||||||
|
class := &storagev1beta1.StorageClass{}
|
||||||
|
|
||||||
|
// When feature gate is disabled, field should not be defaulted
|
||||||
|
output := roundTrip(t, runtime.Object(class)).(*storagev1beta1.StorageClass)
|
||||||
|
if output.VolumeBindingMode != nil {
|
||||||
|
t.Errorf("Expected VolumeBindingMode to not be defaulted, got: %+v", output.VolumeBindingMode)
|
||||||
|
}
|
||||||
|
|
||||||
|
class = &storagev1beta1.StorageClass{}
|
||||||
|
|
||||||
|
err := utilfeature.DefaultFeatureGate.Set("VolumeScheduling=true")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to enable feature gate for VolumeScheduling: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// When feature gate is enabled, field should be defaulted
|
||||||
|
defaultMode := storagev1beta1.VolumeBindingImmediate
|
||||||
|
output = roundTrip(t, runtime.Object(class)).(*storagev1beta1.StorageClass)
|
||||||
|
outMode := output.VolumeBindingMode
|
||||||
|
if outMode == nil {
|
||||||
|
t.Errorf("Expected VolumeBindingMode to be defaulted to: %+v, got: nil", defaultMode)
|
||||||
|
} else if *outMode != defaultMode {
|
||||||
|
t.Errorf("Expected VolumeBindingMode to be defaulted to: %+v, got: %+v", defaultMode, outMode)
|
||||||
|
}
|
||||||
|
|
||||||
|
err = utilfeature.DefaultFeatureGate.Set("VolumeScheduling=false")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to disable feature gate for VolumeScheduling: %v", err)
|
||||||
|
}
|
||||||
|
}
|
@ -52,6 +52,7 @@ func autoConvert_v1beta1_StorageClass_To_storage_StorageClass(in *v1beta1.Storag
|
|||||||
out.ReclaimPolicy = (*core.PersistentVolumeReclaimPolicy)(unsafe.Pointer(in.ReclaimPolicy))
|
out.ReclaimPolicy = (*core.PersistentVolumeReclaimPolicy)(unsafe.Pointer(in.ReclaimPolicy))
|
||||||
out.MountOptions = *(*[]string)(unsafe.Pointer(&in.MountOptions))
|
out.MountOptions = *(*[]string)(unsafe.Pointer(&in.MountOptions))
|
||||||
out.AllowVolumeExpansion = (*bool)(unsafe.Pointer(in.AllowVolumeExpansion))
|
out.AllowVolumeExpansion = (*bool)(unsafe.Pointer(in.AllowVolumeExpansion))
|
||||||
|
out.VolumeBindingMode = (*storage.VolumeBindingMode)(unsafe.Pointer(in.VolumeBindingMode))
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -67,6 +68,7 @@ func autoConvert_storage_StorageClass_To_v1beta1_StorageClass(in *storage.Storag
|
|||||||
out.ReclaimPolicy = (*v1.PersistentVolumeReclaimPolicy)(unsafe.Pointer(in.ReclaimPolicy))
|
out.ReclaimPolicy = (*v1.PersistentVolumeReclaimPolicy)(unsafe.Pointer(in.ReclaimPolicy))
|
||||||
out.MountOptions = *(*[]string)(unsafe.Pointer(&in.MountOptions))
|
out.MountOptions = *(*[]string)(unsafe.Pointer(&in.MountOptions))
|
||||||
out.AllowVolumeExpansion = (*bool)(unsafe.Pointer(in.AllowVolumeExpansion))
|
out.AllowVolumeExpansion = (*bool)(unsafe.Pointer(in.AllowVolumeExpansion))
|
||||||
|
out.VolumeBindingMode = (*v1beta1.VolumeBindingMode)(unsafe.Pointer(in.VolumeBindingMode))
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -46,6 +46,7 @@ func ValidateStorageClass(storageClass *storage.StorageClass) field.ErrorList {
|
|||||||
allErrs = append(allErrs, validateParameters(storageClass.Parameters, field.NewPath("parameters"))...)
|
allErrs = append(allErrs, validateParameters(storageClass.Parameters, field.NewPath("parameters"))...)
|
||||||
allErrs = append(allErrs, validateReclaimPolicy(storageClass.ReclaimPolicy, field.NewPath("reclaimPolicy"))...)
|
allErrs = append(allErrs, validateReclaimPolicy(storageClass.ReclaimPolicy, field.NewPath("reclaimPolicy"))...)
|
||||||
allErrs = append(allErrs, validateAllowVolumeExpansion(storageClass.AllowVolumeExpansion, field.NewPath("allowVolumeExpansion"))...)
|
allErrs = append(allErrs, validateAllowVolumeExpansion(storageClass.AllowVolumeExpansion, field.NewPath("allowVolumeExpansion"))...)
|
||||||
|
allErrs = append(allErrs, validateVolumeBindingMode(storageClass.VolumeBindingMode, field.NewPath("volumeBindingMode"))...)
|
||||||
|
|
||||||
return allErrs
|
return allErrs
|
||||||
}
|
}
|
||||||
@ -64,6 +65,8 @@ func ValidateStorageClassUpdate(storageClass, oldStorageClass *storage.StorageCl
|
|||||||
if *storageClass.ReclaimPolicy != *oldStorageClass.ReclaimPolicy {
|
if *storageClass.ReclaimPolicy != *oldStorageClass.ReclaimPolicy {
|
||||||
allErrs = append(allErrs, field.Forbidden(field.NewPath("reclaimPolicy"), "updates to reclaimPolicy are forbidden."))
|
allErrs = append(allErrs, field.Forbidden(field.NewPath("reclaimPolicy"), "updates to reclaimPolicy are forbidden."))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
allErrs = append(allErrs, apivalidation.ValidateImmutableField(storageClass.VolumeBindingMode, oldStorageClass.VolumeBindingMode, field.NewPath("volumeBindingMode"))...)
|
||||||
return allErrs
|
return allErrs
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -218,3 +221,21 @@ func ValidateVolumeAttachmentUpdate(new, old *storage.VolumeAttachment) field.Er
|
|||||||
}
|
}
|
||||||
return allErrs
|
return allErrs
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var supportedVolumeBindingModes = sets.NewString(string(storage.VolumeBindingImmediate), string(storage.VolumeBindingWaitForFirstConsumer))
|
||||||
|
|
||||||
|
// validateVolumeBindingMode tests that VolumeBindingMode specifies valid values.
|
||||||
|
func validateVolumeBindingMode(mode *storage.VolumeBindingMode, fldPath *field.Path) field.ErrorList {
|
||||||
|
allErrs := field.ErrorList{}
|
||||||
|
if utilfeature.DefaultFeatureGate.Enabled(features.VolumeScheduling) {
|
||||||
|
if mode == nil {
|
||||||
|
allErrs = append(allErrs, field.Required(fldPath, ""))
|
||||||
|
} else if !supportedVolumeBindingModes.Has(string(*mode)) {
|
||||||
|
allErrs = append(allErrs, field.NotSupported(fldPath, mode, supportedVolumeBindingModes.List()))
|
||||||
|
}
|
||||||
|
} else if mode != nil {
|
||||||
|
allErrs = append(allErrs, field.Forbidden(fldPath, "field is disabled by feature-gate VolumeScheduling"))
|
||||||
|
}
|
||||||
|
|
||||||
|
return allErrs
|
||||||
|
}
|
||||||
|
@ -27,6 +27,14 @@ import (
|
|||||||
"k8s.io/kubernetes/pkg/apis/storage"
|
"k8s.io/kubernetes/pkg/apis/storage"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
deleteReclaimPolicy = api.PersistentVolumeReclaimDelete
|
||||||
|
immediateMode1 = storage.VolumeBindingImmediate
|
||||||
|
immediateMode2 = storage.VolumeBindingImmediate
|
||||||
|
waitingMode = storage.VolumeBindingWaitForFirstConsumer
|
||||||
|
invalidMode = storage.VolumeBindingMode("foo")
|
||||||
|
)
|
||||||
|
|
||||||
func TestValidateStorageClass(t *testing.T) {
|
func TestValidateStorageClass(t *testing.T) {
|
||||||
deleteReclaimPolicy := api.PersistentVolumeReclaimPolicy("Delete")
|
deleteReclaimPolicy := api.PersistentVolumeReclaimPolicy("Delete")
|
||||||
retainReclaimPolicy := api.PersistentVolumeReclaimPolicy("Retain")
|
retainReclaimPolicy := api.PersistentVolumeReclaimPolicy("Retain")
|
||||||
@ -436,3 +444,141 @@ func TestVolumeAttachmentUpdateValidation(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func makeClassWithBinding(mode *storage.VolumeBindingMode) *storage.StorageClass {
|
||||||
|
return &storage.StorageClass{
|
||||||
|
ObjectMeta: metav1.ObjectMeta{Name: "foo", ResourceVersion: "foo"},
|
||||||
|
Provisioner: "kubernetes.io/foo-provisioner",
|
||||||
|
ReclaimPolicy: &deleteReclaimPolicy,
|
||||||
|
VolumeBindingMode: mode,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: Remove these tests once feature gate is not required
|
||||||
|
func TestValidateVolumeBindingModeAlphaDisabled(t *testing.T) {
|
||||||
|
errorCases := map[string]*storage.StorageClass{
|
||||||
|
"immediate mode": makeClassWithBinding(&immediateMode1),
|
||||||
|
"waiting mode": makeClassWithBinding(&waitingMode),
|
||||||
|
"invalid mode": makeClassWithBinding(&invalidMode),
|
||||||
|
}
|
||||||
|
|
||||||
|
for testName, storageClass := range errorCases {
|
||||||
|
if errs := ValidateStorageClass(storageClass); len(errs) == 0 {
|
||||||
|
t.Errorf("Expected failure for test: %v", testName)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type bindingTest struct {
|
||||||
|
class *storage.StorageClass
|
||||||
|
shouldSucceed bool
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestValidateVolumeBindingMode(t *testing.T) {
|
||||||
|
cases := map[string]bindingTest{
|
||||||
|
"no mode": {
|
||||||
|
class: makeClassWithBinding(nil),
|
||||||
|
shouldSucceed: false,
|
||||||
|
},
|
||||||
|
"immediate mode": {
|
||||||
|
class: makeClassWithBinding(&immediateMode1),
|
||||||
|
shouldSucceed: true,
|
||||||
|
},
|
||||||
|
"waiting mode": {
|
||||||
|
class: makeClassWithBinding(&waitingMode),
|
||||||
|
shouldSucceed: true,
|
||||||
|
},
|
||||||
|
"invalid mode": {
|
||||||
|
class: makeClassWithBinding(&invalidMode),
|
||||||
|
shouldSucceed: false,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: remove when feature gate not required
|
||||||
|
err := utilfeature.DefaultFeatureGate.Set("VolumeScheduling=true")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to enable feature gate for VolumeScheduling: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
for testName, testCase := range cases {
|
||||||
|
errs := ValidateStorageClass(testCase.class)
|
||||||
|
if testCase.shouldSucceed && len(errs) != 0 {
|
||||||
|
t.Errorf("Expected success for test %q, got %v", testName, errs)
|
||||||
|
}
|
||||||
|
if !testCase.shouldSucceed && len(errs) == 0 {
|
||||||
|
t.Errorf("Expected failure for test %q, got success", testName)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
err = utilfeature.DefaultFeatureGate.Set("VolumeScheduling=false")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to disable feature gate for VolumeScheduling: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type updateTest struct {
|
||||||
|
oldClass *storage.StorageClass
|
||||||
|
newClass *storage.StorageClass
|
||||||
|
shouldSucceed bool
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestValidateUpdateVolumeBindingMode(t *testing.T) {
|
||||||
|
noBinding := makeClassWithBinding(nil)
|
||||||
|
immediateBinding1 := makeClassWithBinding(&immediateMode1)
|
||||||
|
immediateBinding2 := makeClassWithBinding(&immediateMode2)
|
||||||
|
waitBinding := makeClassWithBinding(&waitingMode)
|
||||||
|
|
||||||
|
cases := map[string]updateTest{
|
||||||
|
"old and new no mode": {
|
||||||
|
oldClass: noBinding,
|
||||||
|
newClass: noBinding,
|
||||||
|
shouldSucceed: true,
|
||||||
|
},
|
||||||
|
"old and new same mode ptr": {
|
||||||
|
oldClass: immediateBinding1,
|
||||||
|
newClass: immediateBinding1,
|
||||||
|
shouldSucceed: true,
|
||||||
|
},
|
||||||
|
"old and new same mode value": {
|
||||||
|
oldClass: immediateBinding1,
|
||||||
|
newClass: immediateBinding2,
|
||||||
|
shouldSucceed: true,
|
||||||
|
},
|
||||||
|
"old no mode, new mode": {
|
||||||
|
oldClass: noBinding,
|
||||||
|
newClass: waitBinding,
|
||||||
|
shouldSucceed: false,
|
||||||
|
},
|
||||||
|
"old mode, new no mode": {
|
||||||
|
oldClass: waitBinding,
|
||||||
|
newClass: noBinding,
|
||||||
|
shouldSucceed: false,
|
||||||
|
},
|
||||||
|
"old and new different modes": {
|
||||||
|
oldClass: waitBinding,
|
||||||
|
newClass: immediateBinding1,
|
||||||
|
shouldSucceed: false,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: remove when feature gate not required
|
||||||
|
err := utilfeature.DefaultFeatureGate.Set("VolumeScheduling=true")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to enable feature gate for VolumeScheduling: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
for testName, testCase := range cases {
|
||||||
|
errs := ValidateStorageClassUpdate(testCase.newClass, testCase.oldClass)
|
||||||
|
if testCase.shouldSucceed && len(errs) != 0 {
|
||||||
|
t.Errorf("Expected success for %v, got %v", testName, errs)
|
||||||
|
}
|
||||||
|
if !testCase.shouldSucceed && len(errs) == 0 {
|
||||||
|
t.Errorf("Expected failure for %v, got success", testName)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
err = utilfeature.DefaultFeatureGate.Set("VolumeScheduling=false")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to disable feature gate for VolumeScheduling: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
9
pkg/apis/storage/zz_generated.deepcopy.go
generated
9
pkg/apis/storage/zz_generated.deepcopy.go
generated
@ -60,6 +60,15 @@ func (in *StorageClass) DeepCopyInto(out *StorageClass) {
|
|||||||
**out = **in
|
**out = **in
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if in.VolumeBindingMode != nil {
|
||||||
|
in, out := &in.VolumeBindingMode, &out.VolumeBindingMode
|
||||||
|
if *in == nil {
|
||||||
|
*out = nil
|
||||||
|
} else {
|
||||||
|
*out = new(VolumeBindingMode)
|
||||||
|
**out = **in
|
||||||
|
}
|
||||||
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -98,7 +98,7 @@ const (
|
|||||||
// the API server as the certificate approaches expiration.
|
// the API server as the certificate approaches expiration.
|
||||||
RotateKubeletClientCertificate utilfeature.Feature = "RotateKubeletClientCertificate"
|
RotateKubeletClientCertificate utilfeature.Feature = "RotateKubeletClientCertificate"
|
||||||
|
|
||||||
// owner: @msau
|
// owner: @msau42
|
||||||
// alpha: v1.7
|
// alpha: v1.7
|
||||||
//
|
//
|
||||||
// A new volume type that supports local disks on a node.
|
// A new volume type that supports local disks on a node.
|
||||||
@ -175,6 +175,12 @@ const (
|
|||||||
//
|
//
|
||||||
// Enable running mount utilities in containers.
|
// Enable running mount utilities in containers.
|
||||||
MountContainers utilfeature.Feature = "MountContainers"
|
MountContainers utilfeature.Feature = "MountContainers"
|
||||||
|
|
||||||
|
// owner: @msau42
|
||||||
|
// alpha: v1.9
|
||||||
|
//
|
||||||
|
// Extend the default scheduler to be aware of PV topology and handle PV binding
|
||||||
|
VolumeScheduling utilfeature.Feature = "VolumeScheduling"
|
||||||
)
|
)
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
@ -208,6 +214,7 @@ var defaultKubernetesFeatureGates = map[utilfeature.Feature]utilfeature.FeatureS
|
|||||||
CPUManager: {Default: false, PreRelease: utilfeature.Alpha},
|
CPUManager: {Default: false, PreRelease: utilfeature.Alpha},
|
||||||
ServiceNodeExclusion: {Default: false, PreRelease: utilfeature.Alpha},
|
ServiceNodeExclusion: {Default: false, PreRelease: utilfeature.Alpha},
|
||||||
MountContainers: {Default: false, PreRelease: utilfeature.Alpha},
|
MountContainers: {Default: false, PreRelease: utilfeature.Alpha},
|
||||||
|
VolumeScheduling: {Default: false, PreRelease: utilfeature.Alpha},
|
||||||
|
|
||||||
// inherited features from generic apiserver, relisted here to get a conflict if it is changed
|
// inherited features from generic apiserver, relisted here to get a conflict if it is changed
|
||||||
// unintentionally on either side:
|
// unintentionally on either side:
|
||||||
|
@ -3214,6 +3214,9 @@ func describeStorageClass(sc *storage.StorageClass, events *api.EventList) (stri
|
|||||||
if sc.ReclaimPolicy != nil {
|
if sc.ReclaimPolicy != nil {
|
||||||
w.Write(LEVEL_0, "ReclaimPolicy:\t%s\n", *sc.ReclaimPolicy)
|
w.Write(LEVEL_0, "ReclaimPolicy:\t%s\n", *sc.ReclaimPolicy)
|
||||||
}
|
}
|
||||||
|
if sc.VolumeBindingMode != nil {
|
||||||
|
w.Write(LEVEL_0, "VolumeBindingMode:\t%s\n", *sc.VolumeBindingMode)
|
||||||
|
}
|
||||||
if events != nil {
|
if events != nil {
|
||||||
DescribeEvents(events, w)
|
DescribeEvents(events, w)
|
||||||
}
|
}
|
||||||
|
@ -943,6 +943,7 @@ func TestDescribeDeployment(t *testing.T) {
|
|||||||
|
|
||||||
func TestDescribeStorageClass(t *testing.T) {
|
func TestDescribeStorageClass(t *testing.T) {
|
||||||
reclaimPolicy := api.PersistentVolumeReclaimRetain
|
reclaimPolicy := api.PersistentVolumeReclaimRetain
|
||||||
|
bindingMode := storage.VolumeBindingMode("bindingmode")
|
||||||
f := fake.NewSimpleClientset(&storage.StorageClass{
|
f := fake.NewSimpleClientset(&storage.StorageClass{
|
||||||
ObjectMeta: metav1.ObjectMeta{
|
ObjectMeta: metav1.ObjectMeta{
|
||||||
Name: "foo",
|
Name: "foo",
|
||||||
@ -956,14 +957,22 @@ func TestDescribeStorageClass(t *testing.T) {
|
|||||||
"param1": "value1",
|
"param1": "value1",
|
||||||
"param2": "value2",
|
"param2": "value2",
|
||||||
},
|
},
|
||||||
ReclaimPolicy: &reclaimPolicy,
|
ReclaimPolicy: &reclaimPolicy,
|
||||||
|
VolumeBindingMode: &bindingMode,
|
||||||
})
|
})
|
||||||
s := StorageClassDescriber{f}
|
s := StorageClassDescriber{f}
|
||||||
out, err := s.Describe("", "foo", printers.DescriberSettings{ShowEvents: true})
|
out, err := s.Describe("", "foo", printers.DescriberSettings{ShowEvents: true})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Errorf("unexpected error: %v", err)
|
t.Errorf("unexpected error: %v", err)
|
||||||
}
|
}
|
||||||
if !strings.Contains(out, "foo") {
|
if !strings.Contains(out, "foo") ||
|
||||||
|
!strings.Contains(out, "my-provisioner") ||
|
||||||
|
!strings.Contains(out, "param1") ||
|
||||||
|
!strings.Contains(out, "param2") ||
|
||||||
|
!strings.Contains(out, "value1") ||
|
||||||
|
!strings.Contains(out, "value2") ||
|
||||||
|
!strings.Contains(out, "Retain") ||
|
||||||
|
!strings.Contains(out, "bindingmode") {
|
||||||
t.Errorf("unexpected out: %s", out)
|
t.Errorf("unexpected out: %s", out)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -16,6 +16,7 @@ go_library(
|
|||||||
deps = [
|
deps = [
|
||||||
"//pkg/api/legacyscheme:go_default_library",
|
"//pkg/api/legacyscheme:go_default_library",
|
||||||
"//pkg/apis/storage:go_default_library",
|
"//pkg/apis/storage:go_default_library",
|
||||||
|
"//pkg/apis/storage/util:go_default_library",
|
||||||
"//pkg/apis/storage/validation:go_default_library",
|
"//pkg/apis/storage/validation:go_default_library",
|
||||||
"//pkg/features:go_default_library",
|
"//pkg/features:go_default_library",
|
||||||
"//vendor/k8s.io/apimachinery/pkg/runtime:go_default_library",
|
"//vendor/k8s.io/apimachinery/pkg/runtime:go_default_library",
|
||||||
|
@ -24,6 +24,7 @@ import (
|
|||||||
utilfeature "k8s.io/apiserver/pkg/util/feature"
|
utilfeature "k8s.io/apiserver/pkg/util/feature"
|
||||||
"k8s.io/kubernetes/pkg/api/legacyscheme"
|
"k8s.io/kubernetes/pkg/api/legacyscheme"
|
||||||
"k8s.io/kubernetes/pkg/apis/storage"
|
"k8s.io/kubernetes/pkg/apis/storage"
|
||||||
|
storageutil "k8s.io/kubernetes/pkg/apis/storage/util"
|
||||||
"k8s.io/kubernetes/pkg/apis/storage/validation"
|
"k8s.io/kubernetes/pkg/apis/storage/validation"
|
||||||
"k8s.io/kubernetes/pkg/features"
|
"k8s.io/kubernetes/pkg/features"
|
||||||
)
|
)
|
||||||
@ -49,6 +50,8 @@ func (storageClassStrategy) PrepareForCreate(ctx genericapirequest.Context, obj
|
|||||||
if !utilfeature.DefaultFeatureGate.Enabled(features.ExpandPersistentVolumes) {
|
if !utilfeature.DefaultFeatureGate.Enabled(features.ExpandPersistentVolumes) {
|
||||||
class.AllowVolumeExpansion = nil
|
class.AllowVolumeExpansion = nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
storageutil.DropDisabledAlphaFields(class)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (storageClassStrategy) Validate(ctx genericapirequest.Context, obj runtime.Object) field.ErrorList {
|
func (storageClassStrategy) Validate(ctx genericapirequest.Context, obj runtime.Object) field.ErrorList {
|
||||||
@ -73,6 +76,8 @@ func (storageClassStrategy) PrepareForUpdate(ctx genericapirequest.Context, obj,
|
|||||||
newClass.AllowVolumeExpansion = nil
|
newClass.AllowVolumeExpansion = nil
|
||||||
oldClass.AllowVolumeExpansion = nil
|
oldClass.AllowVolumeExpansion = nil
|
||||||
}
|
}
|
||||||
|
storageutil.DropDisabledAlphaFields(oldClass)
|
||||||
|
storageutil.DropDisabledAlphaFields(newClass)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (storageClassStrategy) ValidateUpdate(ctx genericapirequest.Context, obj, old runtime.Object) field.ErrorList {
|
func (storageClassStrategy) ValidateUpdate(ctx genericapirequest.Context, obj, old runtime.Object) field.ErrorList {
|
||||||
|
@ -35,6 +35,7 @@ func TestStorageClassStrategy(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
deleteReclaimPolicy := api.PersistentVolumeReclaimDelete
|
deleteReclaimPolicy := api.PersistentVolumeReclaimDelete
|
||||||
|
bindingMode := storage.VolumeBindingWaitForFirstConsumer
|
||||||
storageClass := &storage.StorageClass{
|
storageClass := &storage.StorageClass{
|
||||||
ObjectMeta: metav1.ObjectMeta{
|
ObjectMeta: metav1.ObjectMeta{
|
||||||
Name: "valid-class",
|
Name: "valid-class",
|
||||||
@ -43,7 +44,8 @@ func TestStorageClassStrategy(t *testing.T) {
|
|||||||
Parameters: map[string]string{
|
Parameters: map[string]string{
|
||||||
"foo": "bar",
|
"foo": "bar",
|
||||||
},
|
},
|
||||||
ReclaimPolicy: &deleteReclaimPolicy,
|
ReclaimPolicy: &deleteReclaimPolicy,
|
||||||
|
VolumeBindingMode: &bindingMode,
|
||||||
}
|
}
|
||||||
|
|
||||||
Strategy.PrepareForCreate(ctx, storageClass)
|
Strategy.PrepareForCreate(ctx, storageClass)
|
||||||
@ -62,7 +64,8 @@ func TestStorageClassStrategy(t *testing.T) {
|
|||||||
Parameters: map[string]string{
|
Parameters: map[string]string{
|
||||||
"foo": "bar",
|
"foo": "bar",
|
||||||
},
|
},
|
||||||
ReclaimPolicy: &deleteReclaimPolicy,
|
ReclaimPolicy: &deleteReclaimPolicy,
|
||||||
|
VolumeBindingMode: &bindingMode,
|
||||||
}
|
}
|
||||||
|
|
||||||
Strategy.PrepareForUpdate(ctx, newStorageClass, storageClass)
|
Strategy.PrepareForUpdate(ctx, newStorageClass, storageClass)
|
||||||
|
120
staging/src/k8s.io/api/storage/v1/generated.pb.go
generated
120
staging/src/k8s.io/api/storage/v1/generated.pb.go
generated
@ -146,6 +146,12 @@ func (m *StorageClass) MarshalTo(dAtA []byte) (int, error) {
|
|||||||
}
|
}
|
||||||
i++
|
i++
|
||||||
}
|
}
|
||||||
|
if m.VolumeBindingMode != nil {
|
||||||
|
dAtA[i] = 0x3a
|
||||||
|
i++
|
||||||
|
i = encodeVarintGenerated(dAtA, i, uint64(len(*m.VolumeBindingMode)))
|
||||||
|
i += copy(dAtA[i:], *m.VolumeBindingMode)
|
||||||
|
}
|
||||||
return i, nil
|
return i, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -242,6 +248,10 @@ func (m *StorageClass) Size() (n int) {
|
|||||||
if m.AllowVolumeExpansion != nil {
|
if m.AllowVolumeExpansion != nil {
|
||||||
n += 2
|
n += 2
|
||||||
}
|
}
|
||||||
|
if m.VolumeBindingMode != nil {
|
||||||
|
l = len(*m.VolumeBindingMode)
|
||||||
|
n += 1 + l + sovGenerated(uint64(l))
|
||||||
|
}
|
||||||
return n
|
return n
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -293,6 +303,7 @@ func (this *StorageClass) String() string {
|
|||||||
`ReclaimPolicy:` + valueToStringGenerated(this.ReclaimPolicy) + `,`,
|
`ReclaimPolicy:` + valueToStringGenerated(this.ReclaimPolicy) + `,`,
|
||||||
`MountOptions:` + fmt.Sprintf("%v", this.MountOptions) + `,`,
|
`MountOptions:` + fmt.Sprintf("%v", this.MountOptions) + `,`,
|
||||||
`AllowVolumeExpansion:` + valueToStringGenerated(this.AllowVolumeExpansion) + `,`,
|
`AllowVolumeExpansion:` + valueToStringGenerated(this.AllowVolumeExpansion) + `,`,
|
||||||
|
`VolumeBindingMode:` + valueToStringGenerated(this.VolumeBindingMode) + `,`,
|
||||||
`}`,
|
`}`,
|
||||||
}, "")
|
}, "")
|
||||||
return s
|
return s
|
||||||
@ -600,6 +611,36 @@ func (m *StorageClass) Unmarshal(dAtA []byte) error {
|
|||||||
}
|
}
|
||||||
b := bool(v != 0)
|
b := bool(v != 0)
|
||||||
m.AllowVolumeExpansion = &b
|
m.AllowVolumeExpansion = &b
|
||||||
|
case 7:
|
||||||
|
if wireType != 2 {
|
||||||
|
return fmt.Errorf("proto: wrong wireType = %d for field VolumeBindingMode", 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
|
||||||
|
}
|
||||||
|
s := VolumeBindingMode(dAtA[iNdEx:postIndex])
|
||||||
|
m.VolumeBindingMode = &s
|
||||||
|
iNdEx = postIndex
|
||||||
default:
|
default:
|
||||||
iNdEx = preIndex
|
iNdEx = preIndex
|
||||||
skippy, err := skipGenerated(dAtA[iNdEx:])
|
skippy, err := skipGenerated(dAtA[iNdEx:])
|
||||||
@ -842,43 +883,44 @@ func init() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
var fileDescriptorGenerated = []byte{
|
var fileDescriptorGenerated = []byte{
|
||||||
// 593 bytes of a gzipped FileDescriptorProto
|
// 623 bytes of a gzipped FileDescriptorProto
|
||||||
0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0x8c, 0x92, 0x41, 0x6f, 0xd3, 0x3e,
|
0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0x8c, 0x52, 0x4f, 0x6f, 0xd3, 0x3c,
|
||||||
0x18, 0xc6, 0x9b, 0x76, 0xfd, 0x6b, 0x73, 0x37, 0xfd, 0xab, 0x30, 0xa4, 0xa8, 0x87, 0xb4, 0x1a,
|
0x18, 0x6f, 0xda, 0xb7, 0x2f, 0x9b, 0xbb, 0x89, 0x2e, 0x0c, 0x29, 0xea, 0x21, 0xa9, 0xc6, 0xa5,
|
||||||
0x97, 0x0a, 0x09, 0x7b, 0xdd, 0x06, 0x9a, 0x90, 0x40, 0xa2, 0x68, 0x12, 0x48, 0x9b, 0x56, 0x05,
|
0x9a, 0x84, 0xb3, 0x6e, 0x03, 0x4d, 0x48, 0x20, 0x11, 0x34, 0x09, 0xa4, 0x4d, 0xab, 0x82, 0x34,
|
||||||
0x89, 0x03, 0xe2, 0x80, 0x9b, 0xbd, 0x64, 0x26, 0x89, 0x1d, 0xd9, 0x4e, 0xa0, 0x37, 0x3e, 0x02,
|
0x21, 0xc4, 0x01, 0x37, 0x7d, 0xc8, 0x4c, 0x13, 0x3b, 0xb2, 0x9d, 0x40, 0x6f, 0x7c, 0x04, 0xce,
|
||||||
0x9f, 0x87, 0x13, 0xc7, 0x1d, 0x77, 0xdc, 0x29, 0x62, 0xe1, 0x5b, 0x70, 0x42, 0x49, 0xca, 0x92,
|
0x7c, 0x14, 0x3e, 0xc1, 0x8e, 0x3b, 0xee, 0x14, 0xb1, 0xf0, 0x2d, 0x76, 0x42, 0x49, 0xca, 0x9a,
|
||||||
0xad, 0x9d, 0xd8, 0xcd, 0x7e, 0xdf, 0xe7, 0xf7, 0xd8, 0x7e, 0xfd, 0xa0, 0xe7, 0xfe, 0xbe, 0xc2,
|
0xad, 0x9d, 0xd8, 0x2d, 0xfe, 0xfd, 0xb3, 0x9f, 0x27, 0x3f, 0xf4, 0x62, 0xbc, 0x27, 0x31, 0xe5,
|
||||||
0x4c, 0x10, 0x3f, 0x9e, 0x82, 0xe4, 0xa0, 0x41, 0x91, 0x04, 0xf8, 0x89, 0x90, 0x64, 0xde, 0xa0,
|
0xf6, 0x38, 0x1e, 0x82, 0x60, 0xa0, 0x40, 0xda, 0x09, 0xb0, 0x11, 0x17, 0xf6, 0x94, 0x20, 0x11,
|
||||||
0x11, 0x23, 0x4a, 0x0b, 0x49, 0x3d, 0x20, 0xc9, 0x88, 0x78, 0xc0, 0x41, 0x52, 0x0d, 0x27, 0x38,
|
0xb5, 0xa5, 0xe2, 0x82, 0xf8, 0x60, 0x27, 0x7d, 0xdb, 0x07, 0x06, 0x82, 0x28, 0x18, 0xe1, 0x48,
|
||||||
0x92, 0x42, 0x0b, 0xf3, 0x7e, 0x29, 0xc3, 0x34, 0x62, 0x78, 0x2e, 0xc3, 0xc9, 0xa8, 0xf7, 0xc8,
|
0x70, 0xc5, 0xf5, 0x87, 0xa5, 0x0c, 0x93, 0x88, 0xe2, 0xa9, 0x0c, 0x27, 0xfd, 0xce, 0x63, 0x9f,
|
||||||
0x63, 0xfa, 0x34, 0x9e, 0x62, 0x57, 0x84, 0xc4, 0x13, 0x9e, 0x20, 0x85, 0x7a, 0x1a, 0x7f, 0x2c,
|
0xaa, 0x93, 0x78, 0x88, 0x3d, 0x1e, 0xda, 0x3e, 0xf7, 0xb9, 0x5d, 0xa8, 0x87, 0xf1, 0xa7, 0xe2,
|
||||||
0x76, 0xc5, 0xa6, 0x58, 0x95, 0x2e, 0xbd, 0x87, 0x4b, 0x0f, 0x9b, 0x82, 0xa6, 0x0b, 0x27, 0xf6,
|
0x54, 0x1c, 0x8a, 0xaf, 0x32, 0xa5, 0xb3, 0xb9, 0xf0, 0xb2, 0x21, 0x28, 0x32, 0x77, 0x63, 0x67,
|
||||||
0xf6, 0x2a, 0x6d, 0x48, 0xdd, 0x53, 0xc6, 0x41, 0xce, 0x48, 0xe4, 0x7b, 0x79, 0x41, 0x91, 0x10,
|
0x77, 0xa6, 0x0d, 0x89, 0x77, 0x42, 0x19, 0x88, 0x89, 0x1d, 0x8d, 0xfd, 0x1c, 0x90, 0x76, 0x08,
|
||||||
0x34, 0x5d, 0x72, 0xcf, 0x1e, 0xb9, 0x8d, 0x92, 0x31, 0xd7, 0x2c, 0x84, 0x05, 0xe0, 0xc9, 0xbf,
|
0x8a, 0x2c, 0x78, 0x67, 0xc7, 0xbe, 0xcd, 0x25, 0x62, 0xa6, 0x68, 0x08, 0x73, 0x86, 0xa7, 0xff,
|
||||||
0x00, 0xe5, 0x9e, 0x42, 0x48, 0x17, 0xb8, 0xdd, 0xdb, 0xb8, 0x58, 0xb3, 0x80, 0x30, 0xae, 0x95,
|
0x32, 0x48, 0xef, 0x04, 0x42, 0x32, 0xe7, 0xdb, 0xb9, 0xcd, 0x17, 0x2b, 0x1a, 0xd8, 0x94, 0x29,
|
||||||
0x96, 0x37, 0xa1, 0xad, 0x1f, 0x2b, 0x68, 0xfd, 0x4d, 0xf9, 0xee, 0x97, 0x01, 0x55, 0xca, 0xfc,
|
0xa9, 0xc4, 0x4d, 0xd3, 0xc6, 0x8f, 0x26, 0x5a, 0x79, 0x5b, 0xce, 0xfd, 0x2a, 0x20, 0x52, 0xea,
|
||||||
0x80, 0x56, 0xf3, 0x97, 0x9c, 0x50, 0x4d, 0x2d, 0x63, 0x60, 0x0c, 0x3b, 0x3b, 0xdb, 0xb8, 0x9a,
|
0x1f, 0xd1, 0x52, 0x3e, 0xc9, 0x88, 0x28, 0x62, 0x68, 0x5d, 0xad, 0xd7, 0xda, 0xde, 0xc2, 0xb3,
|
||||||
0xf4, 0x95, 0x31, 0x8e, 0x7c, 0x2f, 0x2f, 0x28, 0x9c, 0xab, 0x71, 0x32, 0xc2, 0xc7, 0xd3, 0x4f,
|
0x4d, 0x5f, 0x05, 0xe3, 0x68, 0xec, 0xe7, 0x80, 0xc4, 0xb9, 0x1a, 0x27, 0x7d, 0x7c, 0x34, 0xfc,
|
||||||
0xe0, 0xea, 0x23, 0xd0, 0x74, 0x6c, 0x9e, 0xa5, 0xfd, 0x46, 0x96, 0xf6, 0x51, 0x55, 0x73, 0xae,
|
0x0c, 0x9e, 0x3a, 0x04, 0x45, 0x1c, 0xfd, 0x34, 0xb5, 0x6a, 0x59, 0x6a, 0xa1, 0x19, 0xe6, 0x5e,
|
||||||
0x5c, 0xcd, 0xc7, 0xa8, 0x13, 0x49, 0x91, 0x30, 0xc5, 0x04, 0x07, 0x69, 0x35, 0x07, 0xc6, 0x70,
|
0xa5, 0xea, 0x4f, 0x50, 0x2b, 0x12, 0x3c, 0xa1, 0x92, 0x72, 0x06, 0xc2, 0xa8, 0x77, 0xb5, 0xde,
|
||||||
0x6d, 0x7c, 0x6f, 0x8e, 0x74, 0x26, 0x55, 0xcb, 0xa9, 0xeb, 0x4c, 0x0f, 0xa1, 0x88, 0x4a, 0x1a,
|
0xb2, 0xf3, 0x60, 0x6a, 0x69, 0x0d, 0x66, 0x94, 0x5b, 0xd5, 0xe9, 0x3e, 0x42, 0x11, 0x11, 0x24,
|
||||||
0x82, 0x06, 0xa9, 0xac, 0xd6, 0xa0, 0x35, 0xec, 0xec, 0xec, 0xe2, 0xa5, 0x21, 0xc0, 0xf5, 0x17,
|
0x04, 0x05, 0x42, 0x1a, 0x8d, 0x6e, 0xa3, 0xd7, 0xda, 0xde, 0xc1, 0x0b, 0x4b, 0x80, 0xab, 0x13,
|
||||||
0xe1, 0xc9, 0x15, 0x75, 0xc0, 0xb5, 0x9c, 0x55, 0xb7, 0xab, 0x1a, 0x4e, 0xcd, 0xda, 0xf4, 0xd1,
|
0xe1, 0xc1, 0x95, 0x6b, 0x9f, 0x29, 0x31, 0x99, 0xbd, 0x6e, 0x46, 0xb8, 0x95, 0x68, 0x7d, 0x8c,
|
||||||
0x86, 0x04, 0x37, 0xa0, 0x2c, 0x9c, 0x88, 0x80, 0xb9, 0x33, 0x6b, 0xa5, 0xb8, 0xe1, 0x41, 0x96,
|
0x56, 0x05, 0x78, 0x01, 0xa1, 0xe1, 0x80, 0x07, 0xd4, 0x9b, 0x18, 0xff, 0x15, 0x2f, 0xdc, 0xcf,
|
||||||
0xf6, 0x37, 0x9c, 0x7a, 0xe3, 0x77, 0xda, 0xdf, 0xae, 0xc5, 0xc7, 0x15, 0x32, 0xcf, 0x0e, 0x9e,
|
0x52, 0x6b, 0xd5, 0xad, 0x12, 0x97, 0xa9, 0xb5, 0x55, 0xa9, 0x8f, 0xc7, 0x45, 0xde, 0x1d, 0x3c,
|
||||||
0x80, 0x54, 0x4c, 0x69, 0xe0, 0xfa, 0xad, 0x08, 0xe2, 0x10, 0xae, 0x31, 0xce, 0x75, 0x6f, 0x73,
|
0x00, 0x21, 0xa9, 0x54, 0xc0, 0xd4, 0x31, 0x0f, 0xe2, 0x10, 0xae, 0x79, 0xdc, 0xeb, 0xd9, 0xfa,
|
||||||
0x0f, 0xad, 0x87, 0x22, 0xe6, 0xfa, 0x38, 0xd2, 0x4c, 0x70, 0x65, 0xb5, 0x07, 0xad, 0xe1, 0xda,
|
0x2e, 0x5a, 0x09, 0x79, 0xcc, 0xd4, 0x51, 0xa4, 0x28, 0x67, 0xd2, 0x68, 0x76, 0x1b, 0xbd, 0x65,
|
||||||
0xb8, 0x9b, 0xa5, 0xfd, 0xf5, 0xa3, 0x5a, 0xdd, 0xb9, 0xa6, 0x32, 0x0f, 0xd1, 0x26, 0x0d, 0x02,
|
0xa7, 0x9d, 0xa5, 0xd6, 0xca, 0x61, 0x05, 0x77, 0xaf, 0xa9, 0xf4, 0x03, 0xb4, 0x4e, 0x82, 0x80,
|
||||||
0xf1, 0xb9, 0x3c, 0xe0, 0xe0, 0x4b, 0x44, 0x79, 0x3e, 0x25, 0xeb, 0xbf, 0x81, 0x31, 0x5c, 0x1d,
|
0x7f, 0x29, 0x2f, 0xd8, 0xff, 0x1a, 0x11, 0x96, 0x6f, 0xc9, 0xf8, 0xbf, 0xab, 0xf5, 0x96, 0x1c,
|
||||||
0x5b, 0x59, 0xda, 0xdf, 0x7c, 0xb1, 0xa4, 0xef, 0x2c, 0xa5, 0x7a, 0xcf, 0xd0, 0xff, 0x37, 0x66,
|
0x23, 0x4b, 0xad, 0xf5, 0x97, 0x0b, 0x78, 0x77, 0xa1, 0x4b, 0x7f, 0x87, 0xd6, 0x92, 0x02, 0x72,
|
||||||
0x64, 0x76, 0x51, 0xcb, 0x87, 0x59, 0x11, 0x80, 0x35, 0x27, 0x5f, 0x9a, 0x9b, 0xa8, 0x9d, 0xd0,
|
0x28, 0x1b, 0x51, 0xe6, 0x1f, 0xf2, 0x11, 0x18, 0xf7, 0x8a, 0xa1, 0x37, 0xb3, 0xd4, 0x5a, 0x3b,
|
||||||
0x20, 0x86, 0xf2, 0xbf, 0x9c, 0x72, 0xf3, 0xb4, 0xb9, 0x6f, 0x6c, 0x7d, 0x37, 0x50, 0xb7, 0x3e,
|
0xbe, 0x49, 0x5e, 0x2e, 0x02, 0xdd, 0xf9, 0x90, 0xce, 0x73, 0x74, 0xff, 0xc6, 0xf6, 0xf5, 0x36,
|
||||||
0xf0, 0x43, 0xa6, 0xb4, 0xf9, 0x7e, 0x21, 0x46, 0xf8, 0x6e, 0x31, 0xca, 0xe9, 0x22, 0x44, 0xdd,
|
0x6a, 0x8c, 0x61, 0x52, 0x54, 0x6b, 0xd9, 0xcd, 0x3f, 0xf5, 0x75, 0xd4, 0x4c, 0x48, 0x10, 0x43,
|
||||||
0xf9, 0x37, 0xad, 0xfe, 0xad, 0xd4, 0x22, 0xf4, 0x0a, 0xb5, 0x99, 0x86, 0x50, 0x59, 0xcd, 0x22,
|
0xd9, 0x04, 0xb7, 0x3c, 0x3c, 0xab, 0xef, 0x69, 0x1b, 0x3f, 0x35, 0xd4, 0xae, 0xfe, 0xca, 0x03,
|
||||||
0x06, 0x0f, 0xee, 0x10, 0x83, 0xf1, 0xc6, 0xdc, 0xaf, 0xfd, 0x3a, 0x27, 0x9d, 0xd2, 0x60, 0x3c,
|
0x2a, 0x95, 0xfe, 0x61, 0xae, 0xa0, 0xf8, 0x6e, 0x05, 0xcd, 0xdd, 0x45, 0x3d, 0xdb, 0xd3, 0x02,
|
||||||
0x3c, 0xbb, 0xb4, 0x1b, 0xe7, 0x97, 0x76, 0xe3, 0xe2, 0xd2, 0x6e, 0x7c, 0xcd, 0x6c, 0xe3, 0x2c,
|
0x2c, 0xfd, 0x45, 0x2a, 0xe5, 0x7c, 0x8d, 0x9a, 0x54, 0x41, 0x28, 0x8d, 0x7a, 0x51, 0xb0, 0x47,
|
||||||
0xb3, 0x8d, 0xf3, 0xcc, 0x36, 0x2e, 0x32, 0xdb, 0xf8, 0x99, 0xd9, 0xc6, 0xb7, 0x5f, 0x76, 0xe3,
|
0x77, 0x28, 0x98, 0xb3, 0x3a, 0xcd, 0x6b, 0xbe, 0xc9, 0x9d, 0x6e, 0x19, 0xe0, 0xf4, 0x4e, 0x2f,
|
||||||
0x5d, 0x33, 0x19, 0xfd, 0x09, 0x00, 0x00, 0xff, 0xff, 0xbb, 0x57, 0xe7, 0x15, 0xb0, 0x04, 0x00,
|
0xcc, 0xda, 0xd9, 0x85, 0x59, 0x3b, 0xbf, 0x30, 0x6b, 0xdf, 0x32, 0x53, 0x3b, 0xcd, 0x4c, 0xed,
|
||||||
0x00,
|
0x2c, 0x33, 0xb5, 0xf3, 0xcc, 0xd4, 0x7e, 0x65, 0xa6, 0xf6, 0xfd, 0xb7, 0x59, 0x7b, 0x5f, 0x4f,
|
||||||
|
0xfa, 0x7f, 0x02, 0x00, 0x00, 0xff, 0xff, 0xee, 0x56, 0xcc, 0xfd, 0x0a, 0x05, 0x00, 0x00,
|
||||||
}
|
}
|
||||||
|
@ -63,6 +63,13 @@ message StorageClass {
|
|||||||
// AllowVolumeExpansion shows whether the storage class allow volume expand
|
// AllowVolumeExpansion shows whether the storage class allow volume expand
|
||||||
// +optional
|
// +optional
|
||||||
optional bool allowVolumeExpansion = 6;
|
optional bool allowVolumeExpansion = 6;
|
||||||
|
|
||||||
|
// VolumeBindingMode indicates how PersistentVolumeClaims should be
|
||||||
|
// provisioned and bound. When unset, VolumeBindingImmediate is used.
|
||||||
|
// This field is alpha-level and is only honored by servers that enable
|
||||||
|
// the VolumeScheduling feature.
|
||||||
|
// +optional
|
||||||
|
optional string volumeBindingMode = 7;
|
||||||
}
|
}
|
||||||
|
|
||||||
// StorageClassList is a collection of storage classes.
|
// StorageClassList is a collection of storage classes.
|
||||||
|
@ -59,6 +59,13 @@ type StorageClass struct {
|
|||||||
// AllowVolumeExpansion shows whether the storage class allow volume expand
|
// AllowVolumeExpansion shows whether the storage class allow volume expand
|
||||||
// +optional
|
// +optional
|
||||||
AllowVolumeExpansion *bool `json:"allowVolumeExpansion,omitempty" protobuf:"varint,6,opt,name=allowVolumeExpansion"`
|
AllowVolumeExpansion *bool `json:"allowVolumeExpansion,omitempty" protobuf:"varint,6,opt,name=allowVolumeExpansion"`
|
||||||
|
|
||||||
|
// VolumeBindingMode indicates how PersistentVolumeClaims should be
|
||||||
|
// provisioned and bound. When unset, VolumeBindingImmediate is used.
|
||||||
|
// This field is alpha-level and is only honored by servers that enable
|
||||||
|
// the VolumeScheduling feature.
|
||||||
|
// +optional
|
||||||
|
VolumeBindingMode *VolumeBindingMode `json:"volumeBindingMode,omitempty" protobuf:"bytes,7,opt,name=volumeBindingMode"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// +k8s:deepcopy-gen:interfaces=k8s.io/apimachinery/pkg/runtime.Object
|
// +k8s:deepcopy-gen:interfaces=k8s.io/apimachinery/pkg/runtime.Object
|
||||||
@ -74,3 +81,18 @@ type StorageClassList struct {
|
|||||||
// Items is the list of StorageClasses
|
// Items is the list of StorageClasses
|
||||||
Items []StorageClass `json:"items" protobuf:"bytes,2,rep,name=items"`
|
Items []StorageClass `json:"items" protobuf:"bytes,2,rep,name=items"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// VolumeBindingMode indicates how PersistentVolumeClaims should be bound.
|
||||||
|
type VolumeBindingMode string
|
||||||
|
|
||||||
|
const (
|
||||||
|
// VolumeBindingImmediate indicates that PersistentVolumeClaims should be
|
||||||
|
// immediately provisioned and bound. This is the default mode.
|
||||||
|
VolumeBindingImmediate VolumeBindingMode = "Immediate"
|
||||||
|
|
||||||
|
// VolumeBindingWaitForFirstConsumer indicates that PersistentVolumeClaims
|
||||||
|
// should not be provisioned and bound until the first Pod is created that
|
||||||
|
// references the PeristentVolumeClaim. The volume provisioning and
|
||||||
|
// binding will occur during Pod scheduing.
|
||||||
|
VolumeBindingWaitForFirstConsumer VolumeBindingMode = "WaitForFirstConsumer"
|
||||||
|
)
|
||||||
|
@ -35,6 +35,7 @@ var map_StorageClass = map[string]string{
|
|||||||
"reclaimPolicy": "Dynamically provisioned PersistentVolumes of this storage class are created with this reclaimPolicy. Defaults to Delete.",
|
"reclaimPolicy": "Dynamically provisioned PersistentVolumes of this storage class are created with this reclaimPolicy. Defaults to Delete.",
|
||||||
"mountOptions": "Dynamically provisioned PersistentVolumes of this storage class are created with these mountOptions, e.g. [\"ro\", \"soft\"]. Not validated - mount of the PVs will simply fail if one is invalid.",
|
"mountOptions": "Dynamically provisioned PersistentVolumes of this storage class are created with these mountOptions, e.g. [\"ro\", \"soft\"]. Not validated - mount of the PVs will simply fail if one is invalid.",
|
||||||
"allowVolumeExpansion": "AllowVolumeExpansion shows whether the storage class allow volume expand",
|
"allowVolumeExpansion": "AllowVolumeExpansion shows whether the storage class allow volume expand",
|
||||||
|
"volumeBindingMode": "VolumeBindingMode indicates how PersistentVolumeClaims should be provisioned and bound. When unset, VolumeBindingImmediate is used. This field is alpha-level and is only honored by servers that enable the VolumeScheduling feature.",
|
||||||
}
|
}
|
||||||
|
|
||||||
func (StorageClass) SwaggerDoc() map[string]string {
|
func (StorageClass) SwaggerDoc() map[string]string {
|
||||||
|
@ -60,6 +60,15 @@ func (in *StorageClass) DeepCopyInto(out *StorageClass) {
|
|||||||
**out = **in
|
**out = **in
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if in.VolumeBindingMode != nil {
|
||||||
|
in, out := &in.VolumeBindingMode, &out.VolumeBindingMode
|
||||||
|
if *in == nil {
|
||||||
|
*out = nil
|
||||||
|
} else {
|
||||||
|
*out = new(VolumeBindingMode)
|
||||||
|
**out = **in
|
||||||
|
}
|
||||||
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
119
staging/src/k8s.io/api/storage/v1beta1/generated.pb.go
generated
119
staging/src/k8s.io/api/storage/v1beta1/generated.pb.go
generated
@ -146,6 +146,12 @@ func (m *StorageClass) MarshalTo(dAtA []byte) (int, error) {
|
|||||||
}
|
}
|
||||||
i++
|
i++
|
||||||
}
|
}
|
||||||
|
if m.VolumeBindingMode != nil {
|
||||||
|
dAtA[i] = 0x3a
|
||||||
|
i++
|
||||||
|
i = encodeVarintGenerated(dAtA, i, uint64(len(*m.VolumeBindingMode)))
|
||||||
|
i += copy(dAtA[i:], *m.VolumeBindingMode)
|
||||||
|
}
|
||||||
return i, nil
|
return i, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -242,6 +248,10 @@ func (m *StorageClass) Size() (n int) {
|
|||||||
if m.AllowVolumeExpansion != nil {
|
if m.AllowVolumeExpansion != nil {
|
||||||
n += 2
|
n += 2
|
||||||
}
|
}
|
||||||
|
if m.VolumeBindingMode != nil {
|
||||||
|
l = len(*m.VolumeBindingMode)
|
||||||
|
n += 1 + l + sovGenerated(uint64(l))
|
||||||
|
}
|
||||||
return n
|
return n
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -293,6 +303,7 @@ func (this *StorageClass) String() string {
|
|||||||
`ReclaimPolicy:` + valueToStringGenerated(this.ReclaimPolicy) + `,`,
|
`ReclaimPolicy:` + valueToStringGenerated(this.ReclaimPolicy) + `,`,
|
||||||
`MountOptions:` + fmt.Sprintf("%v", this.MountOptions) + `,`,
|
`MountOptions:` + fmt.Sprintf("%v", this.MountOptions) + `,`,
|
||||||
`AllowVolumeExpansion:` + valueToStringGenerated(this.AllowVolumeExpansion) + `,`,
|
`AllowVolumeExpansion:` + valueToStringGenerated(this.AllowVolumeExpansion) + `,`,
|
||||||
|
`VolumeBindingMode:` + valueToStringGenerated(this.VolumeBindingMode) + `,`,
|
||||||
`}`,
|
`}`,
|
||||||
}, "")
|
}, "")
|
||||||
return s
|
return s
|
||||||
@ -600,6 +611,36 @@ func (m *StorageClass) Unmarshal(dAtA []byte) error {
|
|||||||
}
|
}
|
||||||
b := bool(v != 0)
|
b := bool(v != 0)
|
||||||
m.AllowVolumeExpansion = &b
|
m.AllowVolumeExpansion = &b
|
||||||
|
case 7:
|
||||||
|
if wireType != 2 {
|
||||||
|
return fmt.Errorf("proto: wrong wireType = %d for field VolumeBindingMode", 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
|
||||||
|
}
|
||||||
|
s := VolumeBindingMode(dAtA[iNdEx:postIndex])
|
||||||
|
m.VolumeBindingMode = &s
|
||||||
|
iNdEx = postIndex
|
||||||
default:
|
default:
|
||||||
iNdEx = preIndex
|
iNdEx = preIndex
|
||||||
skippy, err := skipGenerated(dAtA[iNdEx:])
|
skippy, err := skipGenerated(dAtA[iNdEx:])
|
||||||
@ -842,42 +883,44 @@ func init() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
var fileDescriptorGenerated = []byte{
|
var fileDescriptorGenerated = []byte{
|
||||||
// 589 bytes of a gzipped FileDescriptorProto
|
// 622 bytes of a gzipped FileDescriptorProto
|
||||||
0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0x8c, 0x92, 0x4f, 0x4f, 0xd4, 0x40,
|
0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0x8c, 0x92, 0xcd, 0x6e, 0xd3, 0x40,
|
||||||
0x18, 0xc6, 0xb7, 0x2c, 0xab, 0x30, 0x0b, 0x71, 0x53, 0x39, 0x34, 0x7b, 0xe8, 0x6e, 0x38, 0xf5,
|
0x14, 0x85, 0xe3, 0x86, 0xd0, 0x76, 0xd2, 0x8a, 0xd4, 0x74, 0x61, 0x65, 0x61, 0x47, 0x5d, 0x45,
|
||||||
0xc2, 0x0c, 0x20, 0x1a, 0x62, 0xe2, 0xc1, 0x12, 0x0e, 0x26, 0x10, 0x36, 0x35, 0xf1, 0x60, 0x3c,
|
0x48, 0x1d, 0xb7, 0xa5, 0xa0, 0x0a, 0x89, 0x05, 0xae, 0xba, 0x40, 0x6a, 0xd4, 0xc8, 0x48, 0x15,
|
||||||
0x38, 0x5b, 0x5e, 0xcb, 0xd8, 0x76, 0xa6, 0x99, 0x99, 0xae, 0xee, 0xcd, 0x8f, 0xe0, 0x37, 0xf2,
|
0x42, 0x2c, 0x98, 0x38, 0x17, 0x77, 0x88, 0x3d, 0x63, 0xcd, 0x8c, 0x03, 0xd9, 0xf1, 0x08, 0xbc,
|
||||||
0x64, 0xc2, 0x91, 0x23, 0xa7, 0x46, 0xea, 0xb7, 0xf0, 0x64, 0xfa, 0x47, 0x5a, 0x58, 0x88, 0xdc,
|
0x01, 0x8f, 0xc2, 0xb6, 0xcb, 0x2e, 0xbb, 0xb2, 0xa8, 0x79, 0x8b, 0xae, 0x90, 0x7f, 0x68, 0xdc,
|
||||||
0x3a, 0xef, 0xfb, 0xfc, 0x9e, 0xb7, 0xf3, 0xce, 0x83, 0x0e, 0xc2, 0x7d, 0x85, 0x99, 0x20, 0x61,
|
0xfc, 0x88, 0xee, 0x3c, 0xf7, 0x9e, 0xef, 0xdc, 0x99, 0xeb, 0x83, 0x8e, 0x47, 0x47, 0x12, 0x53,
|
||||||
0x3a, 0x05, 0xc9, 0x41, 0x83, 0x22, 0x33, 0xe0, 0xa7, 0x42, 0x92, 0xba, 0x41, 0x13, 0x46, 0x94,
|
0x6e, 0x8f, 0xe2, 0x01, 0x08, 0x06, 0x0a, 0xa4, 0x3d, 0x06, 0x36, 0xe4, 0xc2, 0x2e, 0x1b, 0x24,
|
||||||
0x16, 0x92, 0x06, 0x40, 0x66, 0x3b, 0x53, 0xd0, 0x74, 0x87, 0x04, 0xc0, 0x41, 0x52, 0x0d, 0xa7,
|
0xa2, 0xb6, 0x54, 0x5c, 0x10, 0x1f, 0xec, 0xf1, 0xfe, 0x00, 0x14, 0xd9, 0xb7, 0x7d, 0x60, 0x20,
|
||||||
0x38, 0x91, 0x42, 0x0b, 0x73, 0x58, 0x69, 0x31, 0x4d, 0x18, 0xae, 0xb5, 0xb8, 0xd6, 0x0e, 0xb7,
|
0x88, 0x82, 0x21, 0x8e, 0x04, 0x57, 0x5c, 0x6f, 0x17, 0x5a, 0x4c, 0x22, 0x8a, 0x4b, 0x2d, 0x2e,
|
||||||
0x02, 0xa6, 0xcf, 0xd2, 0x29, 0xf6, 0x45, 0x4c, 0x02, 0x11, 0x08, 0x52, 0x22, 0xd3, 0xf4, 0x53,
|
0xb5, 0xed, 0x5d, 0x9f, 0xaa, 0x8b, 0x78, 0x80, 0x3d, 0x1e, 0xda, 0x3e, 0xf7, 0xb9, 0x9d, 0x23,
|
||||||
0x79, 0x2a, 0x0f, 0xe5, 0x57, 0x65, 0x35, 0xdc, 0x6c, 0x8d, 0xf5, 0x85, 0x2c, 0x66, 0xde, 0x1e,
|
0x83, 0xf8, 0x73, 0x7e, 0xca, 0x0f, 0xf9, 0x57, 0x61, 0xd5, 0xde, 0xa9, 0x8c, 0xf5, 0xb8, 0xc8,
|
||||||
0x37, 0xdc, 0x6b, 0x34, 0x31, 0xf5, 0xcf, 0x18, 0x07, 0x39, 0x27, 0x49, 0x18, 0x14, 0x05, 0x45,
|
0x66, 0xce, 0x8e, 0x6b, 0x1f, 0x4e, 0x35, 0x21, 0xf1, 0x2e, 0x28, 0x03, 0x31, 0xb1, 0xa3, 0x91,
|
||||||
0x62, 0xd0, 0xf4, 0x2e, 0x8a, 0xdc, 0x47, 0xc9, 0x94, 0x6b, 0x16, 0xc3, 0x02, 0xf0, 0xe2, 0x7f,
|
0x9f, 0x15, 0xa4, 0x1d, 0x82, 0x22, 0x8b, 0x28, 0x7b, 0x19, 0x25, 0x62, 0xa6, 0x68, 0x08, 0x73,
|
||||||
0x80, 0xf2, 0xcf, 0x20, 0xa6, 0x0b, 0xdc, 0xb3, 0xfb, 0xb8, 0x54, 0xb3, 0x88, 0x30, 0xae, 0x95,
|
0xc0, 0xcb, 0xff, 0x01, 0xd2, 0xbb, 0x80, 0x90, 0xcc, 0x71, 0xcf, 0x97, 0x71, 0xb1, 0xa2, 0x81,
|
||||||
0x96, 0xb7, 0xa1, 0xcd, 0x9f, 0xcb, 0x68, 0xed, 0x6d, 0xb5, 0xba, 0x83, 0x88, 0x2a, 0x65, 0x7e,
|
0x4d, 0x99, 0x92, 0x4a, 0xcc, 0x42, 0x3b, 0x3f, 0x1b, 0x68, 0xe3, 0x5d, 0xb1, 0xba, 0xe3, 0x80,
|
||||||
0x44, 0x2b, 0xc5, 0x4d, 0x4e, 0xa9, 0xa6, 0x96, 0x31, 0x36, 0x9c, 0xfe, 0xee, 0x36, 0x6e, 0xd6,
|
0x48, 0xa9, 0x7f, 0x42, 0x6b, 0xd9, 0x4b, 0x86, 0x44, 0x11, 0x43, 0xeb, 0x68, 0xdd, 0xe6, 0xc1,
|
||||||
0x7c, 0x6d, 0x8c, 0x93, 0x30, 0x28, 0x0a, 0x0a, 0x17, 0x6a, 0x3c, 0xdb, 0xc1, 0x27, 0xd3, 0xcf,
|
0x1e, 0x9e, 0xae, 0xf9, 0xce, 0x18, 0x47, 0x23, 0x3f, 0x2b, 0x48, 0x9c, 0xa9, 0xf1, 0x78, 0x1f,
|
||||||
0xe0, 0xeb, 0x63, 0xd0, 0xd4, 0x35, 0xcf, 0xb3, 0x51, 0x27, 0xcf, 0x46, 0xa8, 0xa9, 0x79, 0xd7,
|
0x9f, 0x0d, 0xbe, 0x80, 0xa7, 0x7a, 0xa0, 0x88, 0xa3, 0x5f, 0x26, 0x56, 0x2d, 0x4d, 0x2c, 0x34,
|
||||||
0xae, 0xe6, 0x73, 0xd4, 0x4f, 0xa4, 0x98, 0x31, 0xc5, 0x04, 0x07, 0x69, 0x2d, 0x8d, 0x0d, 0x67,
|
0xad, 0xb9, 0x77, 0xae, 0xfa, 0x0b, 0xd4, 0x8c, 0x04, 0x1f, 0x53, 0x49, 0x39, 0x03, 0x61, 0xac,
|
||||||
0xd5, 0x7d, 0x5a, 0x23, 0xfd, 0x49, 0xd3, 0xf2, 0xda, 0x3a, 0x33, 0x42, 0x28, 0xa1, 0x92, 0xc6,
|
0x74, 0xb4, 0xee, 0xba, 0xf3, 0xb4, 0x44, 0x9a, 0xfd, 0x69, 0xcb, 0xad, 0xea, 0xf4, 0x00, 0xa1,
|
||||||
0xa0, 0x41, 0x2a, 0xab, 0x3b, 0xee, 0x3a, 0xfd, 0xdd, 0x7d, 0x7c, 0x7f, 0x02, 0x70, 0xfb, 0x5a,
|
0x88, 0x08, 0x12, 0x82, 0x02, 0x21, 0x8d, 0x7a, 0xa7, 0xde, 0x6d, 0x1e, 0x1c, 0xe1, 0xe5, 0x09,
|
||||||
0x78, 0x72, 0x8d, 0x1e, 0x72, 0x2d, 0xe7, 0xcd, 0x2f, 0x36, 0x0d, 0xaf, 0xe5, 0x6f, 0x86, 0x68,
|
0xc0, 0xd5, 0x67, 0xe1, 0xfe, 0x1d, 0x7a, 0xc2, 0x94, 0x98, 0x4c, 0xaf, 0x38, 0x6d, 0xb8, 0x15,
|
||||||
0x5d, 0x82, 0x1f, 0x51, 0x16, 0x4f, 0x44, 0xc4, 0xfc, 0xb9, 0xb5, 0x5c, 0xfe, 0xe6, 0x61, 0x9e,
|
0x7f, 0x7d, 0x84, 0x36, 0x05, 0x78, 0x01, 0xa1, 0x61, 0x9f, 0x07, 0xd4, 0x9b, 0x18, 0x8f, 0xf2,
|
||||||
0x8d, 0xd6, 0xbd, 0x76, 0xe3, 0x4f, 0x36, 0xda, 0x5e, 0xcc, 0x0e, 0x9e, 0x80, 0x54, 0x4c, 0x69,
|
0x6b, 0x9e, 0xa4, 0x89, 0xb5, 0xe9, 0x56, 0x1b, 0xb7, 0x89, 0xb5, 0x37, 0x9f, 0x1d, 0xdc, 0x07,
|
||||||
0xe0, 0xfa, 0x9d, 0x88, 0xd2, 0x18, 0x6e, 0x30, 0xde, 0x4d, 0x6f, 0x73, 0x0f, 0xad, 0xc5, 0x22,
|
0x21, 0xa9, 0x54, 0xc0, 0xd4, 0x39, 0x0f, 0xe2, 0x10, 0xee, 0x31, 0xee, 0x7d, 0x6f, 0xfd, 0x10,
|
||||||
0xe5, 0xfa, 0x24, 0xd1, 0x4c, 0x70, 0x65, 0xf5, 0xc6, 0x5d, 0x67, 0xd5, 0x1d, 0xe4, 0xd9, 0x68,
|
0x6d, 0x84, 0x3c, 0x66, 0xea, 0x2c, 0x52, 0x94, 0x33, 0x69, 0x34, 0x3a, 0xf5, 0xee, 0xba, 0xd3,
|
||||||
0xed, 0xb8, 0x55, 0xf7, 0x6e, 0xa8, 0xcc, 0x23, 0xb4, 0x41, 0xa3, 0x48, 0x7c, 0xa9, 0x06, 0x1c,
|
0x4a, 0x13, 0x6b, 0xa3, 0x57, 0xa9, 0xbb, 0xf7, 0x54, 0xfa, 0x29, 0xda, 0x26, 0x41, 0xc0, 0xbf,
|
||||||
0x7e, 0x4d, 0x28, 0x2f, 0x56, 0x65, 0x3d, 0x1a, 0x1b, 0xce, 0x8a, 0x6b, 0xe5, 0xd9, 0x68, 0xe3,
|
0x16, 0x03, 0x4e, 0xbe, 0x45, 0x84, 0x65, 0xab, 0x32, 0x1e, 0x77, 0xb4, 0xee, 0x9a, 0x63, 0xa4,
|
||||||
0xf5, 0x1d, 0x7d, 0xef, 0x4e, 0x6a, 0xf8, 0x0a, 0x3d, 0xb9, 0xb5, 0x23, 0x73, 0x80, 0xba, 0x21,
|
0x89, 0xb5, 0xfd, 0x66, 0x41, 0xdf, 0x5d, 0x48, 0xe9, 0xef, 0xd1, 0xd6, 0x38, 0x2f, 0x39, 0x94,
|
||||||
0xcc, 0xcb, 0x14, 0xac, 0x7a, 0xc5, 0xa7, 0xb9, 0x81, 0x7a, 0x33, 0x1a, 0xa5, 0x50, 0x3d, 0x9a,
|
0x0d, 0x29, 0xf3, 0x7b, 0x7c, 0x08, 0xc6, 0x6a, 0xfe, 0xe8, 0x67, 0x69, 0x62, 0x6d, 0x9d, 0xcf,
|
||||||
0x57, 0x1d, 0x5e, 0x2e, 0xed, 0x1b, 0x9b, 0x3f, 0x0c, 0x34, 0x68, 0x2f, 0xfc, 0x88, 0x29, 0x6d,
|
0x36, 0x6f, 0x17, 0x15, 0xdd, 0x79, 0x93, 0xf6, 0x6b, 0xf4, 0x64, 0x66, 0xfb, 0x7a, 0x0b, 0xd5,
|
||||||
0x7e, 0x58, 0xc8, 0x12, 0x7e, 0x58, 0x96, 0x0a, 0xba, 0x4c, 0xd2, 0xa0, 0x7e, 0xa6, 0x95, 0x7f,
|
0x47, 0x30, 0xc9, 0xf3, 0xb5, 0xee, 0x66, 0x9f, 0xfa, 0x36, 0x6a, 0x8c, 0x49, 0x10, 0x43, 0x11,
|
||||||
0x95, 0x56, 0x8e, 0x8e, 0x51, 0x8f, 0x69, 0x88, 0x95, 0xb5, 0x54, 0x66, 0xc1, 0x79, 0x68, 0x16,
|
0x07, 0xb7, 0x38, 0xbc, 0x5a, 0x39, 0xd2, 0x76, 0x7e, 0x69, 0xa8, 0x55, 0xfd, 0x95, 0xa7, 0x54,
|
||||||
0xdc, 0xf5, 0xda, 0xb4, 0xf7, 0xa6, 0xc0, 0xbd, 0xca, 0xc5, 0xdd, 0x3a, 0xbf, 0xb2, 0x3b, 0x17,
|
0x2a, 0xfd, 0xe3, 0x5c, 0x4a, 0xf1, 0xc3, 0x52, 0x9a, 0xd1, 0x79, 0x46, 0x5b, 0x65, 0x00, 0xd6,
|
||||||
0x57, 0x76, 0xe7, 0xf2, 0xca, 0xee, 0x7c, 0xcb, 0x6d, 0xe3, 0x3c, 0xb7, 0x8d, 0x8b, 0xdc, 0x36,
|
0xfe, 0x55, 0x2a, 0x09, 0xed, 0xa1, 0x06, 0x55, 0x10, 0x4a, 0x63, 0x25, 0x4f, 0x59, 0xf7, 0xa1,
|
||||||
0x2e, 0x73, 0xdb, 0xf8, 0x95, 0xdb, 0xc6, 0xf7, 0xdf, 0x76, 0xe7, 0xfd, 0xe3, 0xda, 0xf1, 0x6f,
|
0x29, 0x73, 0x36, 0x4b, 0xd3, 0xc6, 0xdb, 0x0c, 0x77, 0x0b, 0x17, 0x67, 0xf7, 0xf2, 0xc6, 0xac,
|
||||||
0x00, 0x00, 0x00, 0xff, 0xff, 0x1b, 0xae, 0x44, 0x72, 0xc1, 0x04, 0x00, 0x00,
|
0x5d, 0xdd, 0x98, 0xb5, 0xeb, 0x1b, 0xb3, 0xf6, 0x3d, 0x35, 0xb5, 0xcb, 0xd4, 0xd4, 0xae, 0x52,
|
||||||
|
0x53, 0xbb, 0x4e, 0x4d, 0xed, 0x77, 0x6a, 0x6a, 0x3f, 0xfe, 0x98, 0xb5, 0x0f, 0xab, 0xa5, 0xe3,
|
||||||
|
0xdf, 0x00, 0x00, 0x00, 0xff, 0xff, 0x66, 0xe2, 0x8e, 0x84, 0x1b, 0x05, 0x00, 0x00,
|
||||||
}
|
}
|
||||||
|
@ -63,6 +63,13 @@ message StorageClass {
|
|||||||
// AllowVolumeExpansion shows whether the storage class allow volume expand
|
// AllowVolumeExpansion shows whether the storage class allow volume expand
|
||||||
// +optional
|
// +optional
|
||||||
optional bool allowVolumeExpansion = 6;
|
optional bool allowVolumeExpansion = 6;
|
||||||
|
|
||||||
|
// VolumeBindingMode indicates how PersistentVolumeClaims should be
|
||||||
|
// provisioned and bound. When unset, VolumeBindingImmediate is used.
|
||||||
|
// This field is alpha-level and is only honored by servers that enable
|
||||||
|
// the VolumeScheduling feature.
|
||||||
|
// +optional
|
||||||
|
optional string volumeBindingMode = 7;
|
||||||
}
|
}
|
||||||
|
|
||||||
// StorageClassList is a collection of storage classes.
|
// StorageClassList is a collection of storage classes.
|
||||||
|
@ -59,6 +59,13 @@ type StorageClass struct {
|
|||||||
// AllowVolumeExpansion shows whether the storage class allow volume expand
|
// AllowVolumeExpansion shows whether the storage class allow volume expand
|
||||||
// +optional
|
// +optional
|
||||||
AllowVolumeExpansion *bool `json:"allowVolumeExpansion,omitempty" protobuf:"varint,6,opt,name=allowVolumeExpansion"`
|
AllowVolumeExpansion *bool `json:"allowVolumeExpansion,omitempty" protobuf:"varint,6,opt,name=allowVolumeExpansion"`
|
||||||
|
|
||||||
|
// VolumeBindingMode indicates how PersistentVolumeClaims should be
|
||||||
|
// provisioned and bound. When unset, VolumeBindingImmediate is used.
|
||||||
|
// This field is alpha-level and is only honored by servers that enable
|
||||||
|
// the VolumeScheduling feature.
|
||||||
|
// +optional
|
||||||
|
VolumeBindingMode *VolumeBindingMode `json:"volumeBindingMode,omitempty" protobuf:"bytes,7,opt,name=volumeBindingMode"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// +k8s:deepcopy-gen:interfaces=k8s.io/apimachinery/pkg/runtime.Object
|
// +k8s:deepcopy-gen:interfaces=k8s.io/apimachinery/pkg/runtime.Object
|
||||||
@ -74,3 +81,18 @@ type StorageClassList struct {
|
|||||||
// Items is the list of StorageClasses
|
// Items is the list of StorageClasses
|
||||||
Items []StorageClass `json:"items" protobuf:"bytes,2,rep,name=items"`
|
Items []StorageClass `json:"items" protobuf:"bytes,2,rep,name=items"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// VolumeBindingMode indicates how PersistentVolumeClaims should be bound.
|
||||||
|
type VolumeBindingMode string
|
||||||
|
|
||||||
|
const (
|
||||||
|
// VolumeBindingImmediate indicates that PersistentVolumeClaims should be
|
||||||
|
// immediately provisioned and bound. This is the default mode.
|
||||||
|
VolumeBindingImmediate VolumeBindingMode = "Immediate"
|
||||||
|
|
||||||
|
// VolumeBindingWaitForFirstConsumer indicates that PersistentVolumeClaims
|
||||||
|
// should not be provisioned and bound until the first Pod is created that
|
||||||
|
// references the PeristentVolumeClaim. The volume provisioning and
|
||||||
|
// binding will occur during Pod scheduing.
|
||||||
|
VolumeBindingWaitForFirstConsumer VolumeBindingMode = "WaitForFirstConsumer"
|
||||||
|
)
|
||||||
|
@ -35,6 +35,7 @@ var map_StorageClass = map[string]string{
|
|||||||
"reclaimPolicy": "Dynamically provisioned PersistentVolumes of this storage class are created with this reclaimPolicy. Defaults to Delete.",
|
"reclaimPolicy": "Dynamically provisioned PersistentVolumes of this storage class are created with this reclaimPolicy. Defaults to Delete.",
|
||||||
"mountOptions": "Dynamically provisioned PersistentVolumes of this storage class are created with these mountOptions, e.g. [\"ro\", \"soft\"]. Not validated - mount of the PVs will simply fail if one is invalid.",
|
"mountOptions": "Dynamically provisioned PersistentVolumes of this storage class are created with these mountOptions, e.g. [\"ro\", \"soft\"]. Not validated - mount of the PVs will simply fail if one is invalid.",
|
||||||
"allowVolumeExpansion": "AllowVolumeExpansion shows whether the storage class allow volume expand",
|
"allowVolumeExpansion": "AllowVolumeExpansion shows whether the storage class allow volume expand",
|
||||||
|
"volumeBindingMode": "VolumeBindingMode indicates how PersistentVolumeClaims should be provisioned and bound. When unset, VolumeBindingImmediate is used. This field is alpha-level and is only honored by servers that enable the VolumeScheduling feature.",
|
||||||
}
|
}
|
||||||
|
|
||||||
func (StorageClass) SwaggerDoc() map[string]string {
|
func (StorageClass) SwaggerDoc() map[string]string {
|
||||||
|
@ -60,6 +60,15 @@ func (in *StorageClass) DeepCopyInto(out *StorageClass) {
|
|||||||
**out = **in
|
**out = **in
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if in.VolumeBindingMode != nil {
|
||||||
|
in, out := &in.VolumeBindingMode, &out.VolumeBindingMode
|
||||||
|
if *in == nil {
|
||||||
|
*out = nil
|
||||||
|
} else {
|
||||||
|
*out = new(VolumeBindingMode)
|
||||||
|
**out = **in
|
||||||
|
}
|
||||||
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user