Skip to content

Commit

Permalink
[controller] Add LocalStorageClass validation webhook (#20)
Browse files Browse the repository at this point in the history
Signed-off-by: v.oleynikov <[email protected]>
  • Loading branch information
duckhawk authored Mar 27, 2024
1 parent 1e1ad1e commit 1d05d65
Show file tree
Hide file tree
Showing 7 changed files with 438 additions and 6 deletions.
36 changes: 30 additions & 6 deletions images/webhooks/src/main.go
Original file line number Diff line number Diff line change
Expand Up @@ -23,9 +23,11 @@ import (
kwhhttp "github.com/slok/kubewebhook/v2/pkg/http"
kwhlogrus "github.com/slok/kubewebhook/v2/pkg/log/logrus"
kwhmutating "github.com/slok/kubewebhook/v2/pkg/webhook/mutating"
kwhvalidating "github.com/slok/kubewebhook/v2/pkg/webhook/validating"
corev1 "k8s.io/api/core/v1"
"net/http"
"os"
"webhooks/v1alpha1"
"webhooks/validators"
)

Expand Down Expand Up @@ -61,29 +63,51 @@ func main() {

cfg := initFlags()

mt := kwhmutating.MutatorFunc(validators.PodSchedulerMutation)
pscmMF := kwhmutating.MutatorFunc(validators.PodSchedulerMutation)

mcfg := kwhmutating.WebhookConfig{
pscmMCfg := kwhmutating.WebhookConfig{
ID: "PodSchedulerMutation",
Obj: &corev1.Pod{},
Mutator: mt,
Mutator: pscmMF,
Logger: logger,
}
wh, err := kwhmutating.NewWebhook(mcfg)
pscmWh, err := kwhmutating.NewWebhook(pscmMCfg)
if err != nil {
fmt.Fprintf(os.Stderr, "error creating webhook: %s", err)
os.Exit(1)
}

// Get the handler for our webhook.
whHandler, err := kwhhttp.HandlerFor(kwhhttp.HandlerConfig{Webhook: wh, Logger: logger})
pscmWhHandler, err := kwhhttp.HandlerFor(kwhhttp.HandlerConfig{Webhook: pscmWh, Logger: logger})
if err != nil {
fmt.Fprintf(os.Stderr, "error creating webhook handler: %s", err)
os.Exit(1)
}

scuMF := kwhvalidating.ValidatorFunc(validators.StorageClassUpdate)

scuMCfg := kwhvalidating.WebhookConfig{
ID: "StorageClassUpdate",
Obj: &v1alpha1.LocalStorageClass{},
Validator: scuMF,
Logger: logger,
}
scuWh, err := kwhvalidating.NewWebhook(scuMCfg)
if err != nil {
fmt.Fprintf(os.Stderr, "error creating webhook: %s", err)
os.Exit(1)
}

// Get the handler for our webhook.
scuWhHandler, err := kwhhttp.HandlerFor(kwhhttp.HandlerConfig{Webhook: scuWh, Logger: logger})
if err != nil {
fmt.Fprintf(os.Stderr, "error creating webhook handler: %s", err)
os.Exit(1)
}

mux := http.NewServeMux()
mux.Handle("/pod-scheduler-mutation", whHandler)
mux.Handle("/pod-scheduler-mutation", pscmWhHandler)
mux.Handle("/storage-class-update", scuWhHandler)
mux.HandleFunc("/healthz", httpHandlerHealthz)

logger.Infof("Listening on %s", port)
Expand Down
59 changes: 59 additions & 0 deletions images/webhooks/src/v1alpha1/local_storage_class.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,59 @@
/*
Copyright 2024 Flant JSC
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 v1alpha1

import metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"

type LocalStorageClass struct {
metav1.TypeMeta `json:",inline"`
metav1.ObjectMeta `json:"metadata,omitempty"`
Spec LocalStorageClassSpec `json:"spec"`
Status *LocalStorageClassStatus `json:"status,omitempty"`
}

// LocalStorageClassList contains a list of empty block device
type LocalStorageClassList struct {
metav1.TypeMeta `json:",inline"`
metav1.ListMeta `json:"metadata"`
Items []LocalStorageClass `json:"items"`
}

type LocalStorageClassSpec struct {
IsDefault bool `json:"isDefault"`
ReclaimPolicy string `json:"reclaimPolicy"`
VolumeBindingMode string `json:"volumeBindingMode"`
LVM *LocalStorageClassLVM `json:"lvm,omitempty"`
}

type LocalStorageClassLVM struct {
Type string `json:"type"`
LVMVolumeGroups []LocalStorageClassLVG `json:"lvmVolumeGroups"`
}

type LocalStorageClassStatus struct {
Phase string `json:"phase,omitempty"`
Reason string `json:"reason,omitempty"`
}

type LocalStorageClassLVG struct {
Name string `json:"name"`
Thin *LocalStorageClassThinPool `json:"thin,omitempty"`
}

type LocalStorageClassThinPool struct {
PoolName string `json:"poolName"`
}
75 changes: 75 additions & 0 deletions images/webhooks/src/v1alpha1/lvm_volume_group.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,75 @@
/*
Copyright 2024 Flant JSC
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 v1alpha1

import (
"k8s.io/apimachinery/pkg/api/resource"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
)

type LvmVolumeGroupList struct {
metav1.TypeMeta `json:",inline"`
metav1.ListMeta `json:"metadata"`

Items []LvmVolumeGroup `json:"items"`
}

type LvmVolumeGroup struct {
metav1.TypeMeta `json:",inline"`
metav1.ObjectMeta `json:"metadata,omitempty"`

Spec LvmVolumeGroupSpec `json:"spec"`
Status LvmVolumeGroupStatus `json:"status,omitempty"`
}

type SpecThinPool struct {
Name string `json:"name"`
Size resource.Quantity `json:"size"`
}

type LvmVolumeGroupSpec struct {
ActualVGNameOnTheNode string `json:"actualVGNameOnTheNode"`
BlockDeviceNames []string `json:"blockDeviceNames"`
ThinPools []SpecThinPool `json:"thinPools"`
Type string `json:"type"`
}

type LvmVolumeGroupDevice struct {
BlockDevice string `json:"blockDevice"`
DevSize resource.Quantity `json:"devSize"`
PVSize string `json:"pvSize"`
PVUuid string `json:"pvUUID"`
Path string `json:"path"`
}

type LvmVolumeGroupNode struct {
Devices []LvmVolumeGroupDevice `json:"devices"`
Name string `json:"name"`
}

type StatusThinPool struct {
Name string `json:"name"`
ActualSize resource.Quantity `json:"actualSize"`
UsedSize string `json:"usedSize"`
}

type LvmVolumeGroupStatus struct {
AllocatedSize string `json:"allocatedSize"`
Health string `json:"health"`
Message string `json:"message"`
Nodes []LvmVolumeGroupNode `json:"nodes"`
ThinPools []StatusThinPool `json:"thinPools"`
VGSize string `json:"vgSize"`
VGUuid string `json:"vgUUID"`
}
51 changes: 51 additions & 0 deletions images/webhooks/src/v1alpha1/register.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,51 @@
/*
Copyright 2024 Flant JSC
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 v1alpha1

import (
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/runtime"
"k8s.io/apimachinery/pkg/runtime/schema"
)

const (
LocalStorageClassKind = "LocalStorageClass"
APIGroup = "storage.deckhouse.io"
APIVersion = "v1alpha1"
)

// SchemeGroupVersion is group version used to register these objects
var (
SchemeGroupVersion = schema.GroupVersion{
Group: APIGroup,
Version: APIVersion,
}
SchemeBuilder = runtime.NewSchemeBuilder(addKnownTypes)
AddToScheme = SchemeBuilder.AddToScheme
)

// Adds the list of known types to Scheme.
func addKnownTypes(scheme *runtime.Scheme) error {
scheme.AddKnownTypes(SchemeGroupVersion,
&LocalStorageClass{},
&LocalStorageClassList{},
&LvmVolumeGroup{},
&LvmVolumeGroupList{},
)
metav1.AddToGroupVersion(scheme, SchemeGroupVersion)
return nil
}
135 changes: 135 additions & 0 deletions images/webhooks/src/v1alpha1/zz_generated.deepcopy.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,135 @@
/*
Copyright 2024 Flant JSC
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 v1alpha1

import "k8s.io/apimachinery/pkg/runtime"

// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
func (in *LocalStorageClass) DeepCopyInto(out *LocalStorageClass) {
*out = *in
out.TypeMeta = in.TypeMeta
in.ObjectMeta.DeepCopyInto(&out.ObjectMeta)

}

// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new EmptyBlockDevice.
func (in *LocalStorageClass) DeepCopy() *LocalStorageClass {
if in == nil {
return nil
}
out := new(LocalStorageClass)
in.DeepCopyInto(out)
return out
}

// DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object.
func (in *LocalStorageClass) DeepCopyObject() runtime.Object {
if c := in.DeepCopy(); c != nil {
return c
}
return nil
}

// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
func (in *LocalStorageClassList) DeepCopyInto(out *LocalStorageClassList) {
*out = *in
out.TypeMeta = in.TypeMeta
in.ListMeta.DeepCopyInto(&out.ListMeta)
if in.Items != nil {
in, out := &in.Items, &out.Items
*out = make([]LocalStorageClass, len(*in))
for i := range *in {
(*in)[i].DeepCopyInto(&(*out)[i])
}
}
}

// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new GuestbookList.
func (in *LocalStorageClassList) DeepCopy() *LocalStorageClassList {
if in == nil {
return nil
}
out := new(LocalStorageClassList)
in.DeepCopyInto(out)
return out
}

// DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object.
func (in *LocalStorageClassList) DeepCopyObject() runtime.Object {
if c := in.DeepCopy(); c != nil {
return c
}
return nil
}

// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
func (in *LvmVolumeGroup) DeepCopyInto(out *LvmVolumeGroup) {
*out = *in
out.TypeMeta = in.TypeMeta
in.ObjectMeta.DeepCopyInto(&out.ObjectMeta)

}

// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new EmptyBlockDevice.
func (in *LvmVolumeGroup) DeepCopy() *LvmVolumeGroup {
if in == nil {
return nil
}
out := new(LvmVolumeGroup)
in.DeepCopyInto(out)
return out
}

// DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object.
func (in *LvmVolumeGroup) DeepCopyObject() runtime.Object {
if c := in.DeepCopy(); c != nil {
return c
}
return nil
}

// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
func (in *LvmVolumeGroupList) DeepCopyInto(out *LvmVolumeGroupList) {
*out = *in
out.TypeMeta = in.TypeMeta
in.ListMeta.DeepCopyInto(&out.ListMeta)
if in.Items != nil {
in, out := &in.Items, &out.Items
*out = make([]LvmVolumeGroup, len(*in))
for i := range *in {
(*in)[i].DeepCopyInto(&(*out)[i])
}
}
}

// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new GuestbookList.
func (in *LvmVolumeGroupList) DeepCopy() *LvmVolumeGroupList {
if in == nil {
return nil
}
out := new(LvmVolumeGroupList)
in.DeepCopyInto(out)
return out
}

// DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object.
func (in *LvmVolumeGroupList) DeepCopyObject() runtime.Object {
if c := in.DeepCopy(); c != nil {
return c
}
return nil
}
Loading

0 comments on commit 1d05d65

Please sign in to comment.