Skip to content

Commit feb73d0

Browse files
committed
Support specifying StorageClass while creating volumes
Add support for specifying StorageClass when user is creating volumes via oc set volume command
1 parent 512825b commit feb73d0

File tree

2 files changed

+90
-2
lines changed

2 files changed

+90
-2
lines changed

pkg/cmd/cli/cmd/set/volume.go

+20-2
Original file line numberDiff line numberDiff line change
@@ -85,7 +85,8 @@ For descriptions on other volume types, see https://docs.openshift.com`
8585
# Ceph, Gluster, NFS, ISCSI, ...)
8686
%[1]s volume dc/registry --add -m /repo --source=<json-string>`
8787

88-
volumePrefix = "volume-"
88+
volumePrefix = "volume-"
89+
storageAnnClass = "volume.beta.kubernetes.io/storage-class"
8990
)
9091

9192
type VolumeOptions struct {
@@ -134,6 +135,7 @@ type AddVolumeOptions struct {
134135
ClaimName string
135136
ClaimSize string
136137
ClaimMode string
138+
ClaimClass string
137139

138140
TypeChanged bool
139141
}
@@ -184,6 +186,7 @@ func NewCmdVolume(fullName string, f *clientcmd.Factory, out, errOut io.Writer)
184186
cmd.Flags().StringVar(&addOpts.ConfigMapName, "configmap-name", "", "Name of the persisted config map. Must be provided for configmap volume type")
185187
cmd.Flags().StringVar(&addOpts.SecretName, "secret-name", "", "Name of the persisted secret. Must be provided for secret volume type")
186188
cmd.Flags().StringVar(&addOpts.ClaimName, "claim-name", "", "Persistent volume claim name. Must be provided for persistentVolumeClaim volume type")
189+
cmd.Flags().StringVar(&addOpts.ClaimClass, "claim-class", "", "StorageClass to use for provisioning the persistent volume.")
187190
cmd.Flags().StringVar(&addOpts.ClaimSize, "claim-size", "", "If specified along with a persistent volume type, create a new claim with the given size in bytes. Accepts SI notation: 10, 10G, 10Gi")
188191
cmd.Flags().StringVar(&addOpts.ClaimMode, "claim-mode", "ReadWriteOnce", "Set the access mode of the claim to be created. Valid values are ReadWriteOnce (rwo), ReadWriteMany (rwm), or ReadOnlyMany (rom)")
189192
cmd.Flags().StringVar(&addOpts.Source, "source", "", "Details of volume source as json string. This can be used if the required volume type is not supported by --type option. (e.g.: '{\"gitRepo\": {\"repository\": <git-url>, \"revision\": <commit-hash>}}')")
@@ -312,6 +315,15 @@ func (a *AddVolumeOptions) Validate(isAddOp bool) error {
312315
return err
313316
}
314317
}
318+
if len(a.ClaimClass) > 0 {
319+
selectedLowerType := strings.ToLower(a.Type)
320+
if selectedLowerType != "persistentvolumeclaim" && selectedLowerType != "pvc" {
321+
return errors.New("must provide --type as persistentVolumeClaim")
322+
}
323+
if len(a.ClaimSize) == 0 {
324+
return errors.New("must provide --claim-size to create new pvc with claim-class")
325+
}
326+
}
315327
} else if len(a.Source) > 0 || len(a.Path) > 0 || len(a.SecretName) > 0 || len(a.ConfigMapName) > 0 || len(a.ClaimName) > 0 || a.Overwrite {
316328
return errors.New("--type|--path|--configmap-name|--secret-name|--claim-name|--source|--overwrite are only valid for --add operation")
317329
}
@@ -561,7 +573,7 @@ func (v *VolumeOptions) printVolumes(infos []*resource.Info) []error {
561573
}
562574

563575
func (v *AddVolumeOptions) createClaim() *kapi.PersistentVolumeClaim {
564-
return &kapi.PersistentVolumeClaim{
576+
pvc := &kapi.PersistentVolumeClaim{
565577
ObjectMeta: kapi.ObjectMeta{
566578
Name: v.ClaimName,
567579
},
@@ -574,6 +586,12 @@ func (v *AddVolumeOptions) createClaim() *kapi.PersistentVolumeClaim {
574586
},
575587
},
576588
}
589+
if len(v.ClaimClass) > 0 {
590+
pvc.Annotations = map[string]string{
591+
storageAnnClass: v.ClaimClass,
592+
}
593+
}
594+
return pvc
577595
}
578596

579597
func (v *VolumeOptions) setVolumeSource(kv *kapi.Volume) error {

pkg/cmd/cli/cmd/set/volume_test.go

+70
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
package set
22

33
import (
4+
"errors"
45
"net/http"
56
"testing"
67

@@ -161,3 +162,72 @@ func TestAddVolume(t *testing.T) {
161162
t.Error(patchError)
162163
}
163164
}
165+
166+
func TestCreateClaim(t *testing.T) {
167+
addOpts := &AddVolumeOptions{
168+
Type: "persistentVolumeClaim",
169+
ClaimClass: "foobar",
170+
ClaimName: "foo-vol",
171+
ClaimSize: "5G",
172+
MountPath: "/sandbox",
173+
}
174+
175+
pvc := addOpts.createClaim()
176+
if len(pvc.Annotations) == 0 {
177+
t.Errorf("Expected storage class annotation")
178+
}
179+
180+
if pvc.Annotations[storageAnnClass] != "foobar" {
181+
t.Errorf("Expected storage annotated class to be %s", addOpts.ClaimClass)
182+
}
183+
}
184+
185+
func TestValidateAddOptions(t *testing.T) {
186+
tests := []struct {
187+
name string
188+
addOpts *AddVolumeOptions
189+
expectedError error
190+
}{
191+
{
192+
"using existing pvc",
193+
&AddVolumeOptions{Type: "persistentVolumeClaim"},
194+
errors.New("must provide --claim-name or --claim-size (to create a new claim) for --type=pvc"),
195+
},
196+
{
197+
"creating new pvc",
198+
&AddVolumeOptions{Type: "persistentVolumeClaim", ClaimName: "sandbox-pvc", ClaimSize: "5G"},
199+
nil,
200+
},
201+
{
202+
"error creating pvc with storage class",
203+
&AddVolumeOptions{Type: "persistentVolumeClaim", ClaimName: "sandbox-pvc", ClaimClass: "slow"},
204+
errors.New("must provide --claim-size to create new pvc with claim-class"),
205+
},
206+
{
207+
"creating pvc with storage class",
208+
&AddVolumeOptions{Type: "persistentVolumeClaim", ClaimName: "sandbox-pvc", ClaimClass: "slow", ClaimSize: "5G"},
209+
nil,
210+
},
211+
}
212+
213+
for _, testCase := range tests {
214+
addOpts := testCase.addOpts
215+
err := addOpts.Validate(true)
216+
if testCase.expectedError == nil && err != nil {
217+
t.Errorf("Expected nil error for %s got %s", testCase.name, err)
218+
continue
219+
}
220+
221+
if testCase.expectedError != nil {
222+
if err == nil {
223+
t.Errorf("Expected %s, got nil", testCase.expectedError)
224+
continue
225+
}
226+
227+
if testCase.expectedError.Error() != err.Error() {
228+
t.Errorf("Expected %s, got %s", testCase.expectedError, err)
229+
}
230+
}
231+
232+
}
233+
}

0 commit comments

Comments
 (0)