Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[V2] Storage Analysis Tool for Azure Disk CSI Driver V2 #1277

Merged
merged 28 commits into from
Apr 25, 2022
Merged
Show file tree
Hide file tree
Changes from 17 commits
Commits
Show all changes
28 commits
Select commit Hold shift + click to select a range
7e9e368
initial commit
hccheng72 Apr 7, 2022
45a433e
added display functions
hccheng72 Apr 8, 2022
39de770
tested access to cluster
hccheng72 Apr 8, 2022
540b4d5
updated go.mod
hccheng72 Apr 8, 2022
ef8b17e
deleted extra go.mod and go.sum
hccheng72 Apr 11, 2022
1713f7d
added func GetAzVolumesByPod
hccheng72 Apr 12, 2022
70e6bd4
Merge branch 'main_v2' into storageCli
hccheng72 Apr 12, 2022
52471b7
moved directory, deleted LICENSE, edited description of commands
hccheng72 Apr 12, 2022
71a97a5
added packages to vendor
hccheng72 Apr 12, 2022
67e1fe0
fixed func GetAzVolumesByPod and removed testing lines
hccheng72 Apr 12, 2022
9f82cd6
moved common functions for commands to common.go
hccheng72 Apr 13, 2022
69c8afa
feat: Add new functions to get AzVolumeAttachments by pod/node/zone
hccheng72 Apr 14, 2022
ae380a9
Merge branch 'storageCli' of https://github.com/hccheng72/azuredisk-c…
hccheng72 Apr 14, 2022
730ee3a
feat: Update go.mod, go.sum, and verdor folder
hccheng72 Apr 14, 2022
4c8398d
fix: Fix formatting
hccheng72 Apr 15, 2022
8ad2ab5
fix: fix boilerplate format
hccheng72 Apr 15, 2022
261f49c
fix: Fix boilerplate in main.go
hccheng72 Apr 15, 2022
8fe1c50
perf: Get only one pod when a podName is provided
hccheng72 Apr 18, 2022
fa03d70
perf: use constants for string literals
hccheng72 Apr 18, 2022
5d32e79
feat: if no flag values, list all of the pods/nodes/zone
hccheng72 Apr 18, 2022
fb664dc
perf: change the way to access config and age format
hccheng72 Apr 20, 2022
b46a1a3
fix: fix format
hccheng72 Apr 20, 2022
3058ae7
perf: change .az-analyz extention to yaml, access driverNamaspace by …
hccheng72 Apr 20, 2022
2efaf78
fix: using default path if kubeconfig isn't provided
hccheng72 Apr 20, 2022
4ff15a2
perf: change format of get azva and add default value to driverNamespace
hccheng72 Apr 21, 2022
f5bbd06
fix: fix format
hccheng72 Apr 21, 2022
ea746d7
Merge branch 'main_v2' into storageCli
hccheng72 Apr 21, 2022
f5c75f7
fix: update go.sum
hccheng72 Apr 21, 2022
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
136 changes: 136 additions & 0 deletions cmd/az-analyze/cmd/azv.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,136 @@
/*
Copyright 2022 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 cmd

import (
"context"
"fmt"
"os"

//"reflect"

"github.com/olekukonko/tablewriter"
"github.com/spf13/cobra"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
v1beta1 "sigs.k8s.io/azuredisk-csi-driver/pkg/apis/azuredisk/v1beta1"
consts "sigs.k8s.io/azuredisk-csi-driver/pkg/azureconstants"
)

// azvCmd represents the azv command
var azvCmd = &cobra.Command{
Use: "azv",
Short: "Azure Volume",
Long: `Azure Volume is a Kubernetes Custom Resource.`,
Run: func(cmd *cobra.Command, args []string) {
pod, _ := cmd.Flags().GetString("pod")
namespace, _ := cmd.Flags().GetString("namespace")

var result []AzvResource
result = GetAzVolumesByPod(pod, namespace)

// display
if len(result) != 0 {
displayAzv(result)
} else {
// not found, display an error
fmt.Printf("azVolumes not found in the %s\n", namespace)
}

},
}

func init() {
getCmd.AddCommand(azvCmd)
azvCmd.PersistentFlags().StringP("pod", "p", "", "insert-pod-name")
azvCmd.PersistentFlags().StringP("namespace", "n", "default", "insert-namespace (optional).")

// Here you will define your flags and configuration settings.

// Cobra supports Persistent Flags which will work for this command
// and all subcommands, e.g.:
// azvCmd.PersistentFlags().String("foo", "", "A help for foo")

// Cobra supports local flags which will only run when this command
// is called directly, e.g.:
// azvCmd.Flags().BoolP("toggle", "t", false, "Help message for toggle")
}

type AzvResource struct {
ResourceType string
Namespace string
Name string
State v1beta1.AzVolumeState
}

func GetAzVolumesByPod(podName string, namespace string) []AzvResource {
result := make([]AzvResource, 0)

// access to Config and Clientsets
config := getConfig()
clientsetK8s := getKubernetesClientset(config)
clientsetAzDisk := getAzDiskClientset(config)

// get pvc claim name set of pod
pvcClaimNameSet := make(map[string]string)

pods, err := clientsetK8s.CoreV1().Pods(namespace).List(context.Background(), metav1.ListOptions{})
hccheng72 marked this conversation as resolved.
Show resolved Hide resolved
if err != nil {
panic(err.Error())
}

for _, pod := range pods.Items {
// if pod flag isn't provided, print all pods
if podName == "" || pod.Name == podName {
for _, v := range pod.Spec.Volumes {
if v.PersistentVolumeClaim != nil {
pvcClaimNameSet[v.PersistentVolumeClaim.ClaimName] = pod.Name
}
}
}
}

// get azVolumes with the same claim name in pvcSet
azVolumes, err := clientsetAzDisk.DiskV1beta1().AzVolumes(consts.DefaultAzureDiskCrdNamespace).List(context.Background(), metav1.ListOptions{})
hccheng72 marked this conversation as resolved.
Show resolved Hide resolved
if err != nil {
panic(err.Error())
}
for _, azVolume := range azVolumes.Items {
pvcClaimName := azVolume.Spec.Parameters["csi.storage.k8s.io/pvc/name"]

// if pvcClaimName is contained in pvcClaimNameSet, add the azVolume to result
if pName, ok := pvcClaimNameSet[pvcClaimName]; ok {
result = append(result, AzvResource{
ResourceType: pName,
Namespace: azVolume.Namespace,
Name: azVolume.Spec.VolumeName,
State: azVolume.Status.State})
}
}

return result
}

func displayAzv(result []AzvResource) {
table := tablewriter.NewWriter(os.Stdout)
table.SetHeader([]string{"PODNAME", "NAMESPACE", "NAME", "STATE"})

for _, azv := range result {
table.Append([]string{azv.ResourceType, azv.Namespace, azv.Name, string(azv.State)})
}

table.Render()
}
244 changes: 244 additions & 0 deletions cmd/az-analyze/cmd/azva.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,244 @@
/*
Copyright 2022 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 cmd

import (
"context"
"fmt"
"os"
"strings"
"time"

"github.com/olekukonko/tablewriter"
"github.com/spf13/cobra"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
v1beta1 "sigs.k8s.io/azuredisk-csi-driver/pkg/apis/azuredisk/v1beta1"
consts "sigs.k8s.io/azuredisk-csi-driver/pkg/azureconstants"
)

// azvaCmd represents the azva command
var azvaCmd = &cobra.Command{
Use: "azva",
Short: "Azure Volume Attachment",
Long: `Azure Volume Attachment is a Kubernetes Custom Resource.`,
Run: func(cmd *cobra.Command, args []string) {
// typesFlag := []string{"pod", "node", "zone", "namespace"}
// valuesFlag := []string{pod, node, zone, namespace}

// for _, value := range valuesFlag {

// }

pod, _ := cmd.Flags().GetString("pod")
node, _ := cmd.Flags().GetString("node")
zone, _ := cmd.Flags().GetString("zone")
namespace, _ := cmd.Flags().GetString("namespace")

numFlag := cmd.Flags().NFlag()
if hasNamespace := namespace != ""; hasNamespace {
numFlag--
}

var azva []AzvaResource

if numFlag > 1 {
fmt.Printf("only one of the flags is allowed.\n" + "Run 'az-analyze --help' for usage.\n")
} else {
if numFlag == 0 {
// TODO: the same as kubectl get AzVolumeAttachment
fmt.Println("no flags")
} else if pod != "" {
azva = GetAzVolumeAttachementsByPod(pod, namespace)
displayAzva(azva, "POD")
} else if node != "" {
azva = GetAzVolumeAttachementsByNode(node)
displayAzva(azva, "NODE")
} else if zone != "" {
azva = GetAzVolumeAttachementsByZone(zone)
displayAzva(azva, "ZONE")
} else {
fmt.Printf("invalid flag name\n" + "Run 'az-analyze --help' for usage.\n")
}
}
},
}

func init() {
getCmd.AddCommand(azvaCmd)
azvaCmd.PersistentFlags().StringP("pod", "p", "", "insert-pod-name (only one of the flags is allowed).")
azvaCmd.PersistentFlags().StringP("node", "d", "", "insert-node-name (only one of the flags is allowed).")
azvaCmd.PersistentFlags().StringP("zone", "z", "", "insert-zone-name (only one of the flags is allowed).")
azvaCmd.PersistentFlags().StringP("namespace", "n", "", "insert-namespace (optional).")

// Here you will define your flags and configuration settings.

// Cobra supports Persistent Flags which will work for this command
// and all subcommands, e.g.:
// azvaCmd.PersistentFlags().String("foo", "", "A help for foo")

// Cobra supports local flags which will only run when this command
// is called directly, e.g.:
// azvaCmd.Flags().BoolP("toggle", "t", false, "Help message for toggle")
}

type AzvaResource struct {
ResourceType string
Namespace string
Name string
Age time.Duration
RequestRole v1beta1.Role
Role v1beta1.Role
State v1beta1.AzVolumeAttachmentAttachmentState
}

func GetAzVolumeAttachementsByPod(podName string, namespace string) []AzvaResource {
result := make([]AzvaResource, 0)

if namespace == "" {
namespace = "default"
}

// access to config and Clientsets
config := getConfig()
clientsetK8s := getKubernetesClientset(config)
clientsetAzDisk := getAzDiskClientset(config)

// get pvc claim names of pod
pvcClaimNameSet := make(map[string]bool)

pods, err := clientsetK8s.CoreV1().Pods(namespace).List(context.Background(), metav1.ListOptions{})
hccheng72 marked this conversation as resolved.
Show resolved Hide resolved
if err != nil {
panic(err.Error())
}

for _, pod := range pods.Items {
if pod.Name == podName {
hccheng72 marked this conversation as resolved.
Show resolved Hide resolved
for _, v := range pod.Spec.Volumes {
if v.PersistentVolumeClaim != nil {
pvcClaimNameSet[v.PersistentVolumeClaim.ClaimName] = true
}
}
}
}

// get azVolumes with the same claim name in pvcClaimNameSet
hccheng72 marked this conversation as resolved.
Show resolved Hide resolved
azVolumeAttachments, err := clientsetAzDisk.DiskV1beta1().AzVolumeAttachments(consts.DefaultAzureDiskCrdNamespace).List(context.Background(), metav1.ListOptions{})
hccheng72 marked this conversation as resolved.
Show resolved Hide resolved
if err != nil {
panic(err.Error())
}

for _, azVolumeAttachment := range azVolumeAttachments.Items {
pvcClaimName := azVolumeAttachment.Spec.VolumeContext["csi.storage.k8s.io/pvc/name"]
hccheng72 marked this conversation as resolved.
Show resolved Hide resolved

// if pvcClaimName is contained in pvcClaimNameSet, add the azVolumeattachment to result
if pvcClaimNameSet[pvcClaimName] {
result = append(result, AzvaResource{
ResourceType: podName,
Namespace: azVolumeAttachment.Namespace,
Name: azVolumeAttachment.Name,
Age: time.Duration(metav1.Now().Sub(azVolumeAttachment.CreationTimestamp.Time).Hours()), //TODO: change format of age
RequestRole: azVolumeAttachment.Spec.RequestedRole,
Role: azVolumeAttachment.Status.Detail.Role,
State: azVolumeAttachment.Status.State})
}
}

return result
}

func GetAzVolumeAttachementsByNode(nodeName string) []AzvaResource {
result := make([]AzvaResource, 0)

// access to config and Clientset
config := getConfig()
clientsetAzDisk := getAzDiskClientset(config)

azVolumeAttachments, err := clientsetAzDisk.DiskV1beta1().AzVolumeAttachments(consts.DefaultAzureDiskCrdNamespace).List(context.Background(), metav1.ListOptions{})
if err != nil {
panic(err.Error())
}

for _, azVolumeAttachment := range azVolumeAttachments.Items {
if azVolumeAttachment.Spec.NodeName == nodeName {
result = append(result, AzvaResource{
ResourceType: azVolumeAttachment.Spec.NodeName,
Namespace: azVolumeAttachment.Namespace,
Name: azVolumeAttachment.Name,
Age: metav1.Now().Sub(azVolumeAttachment.CreationTimestamp.Time),
RequestRole: azVolumeAttachment.Spec.RequestedRole,
Role: azVolumeAttachment.Status.Detail.Role,
State: azVolumeAttachment.Status.State})
}
}

return result
}

func GetAzVolumeAttachementsByZone(zoneName string) []AzvaResource {
result := make([]AzvaResource, 0)

// access to config and Clientsets
config := getConfig()
clientsetK8s := getKubernetesClientset(config)
clientsetAzDisk := getAzDiskClientset(config)

// get nodes in the zone
nodeSet := make(map[string]bool)

nodes, err := clientsetK8s.CoreV1().Nodes().List(context.Background(), metav1.ListOptions{})
if err != nil {
panic(err.Error())
}

for _, node := range nodes.Items {
if node.Labels["topology.kubernetes.io/zone"] == zoneName {
nodeSet[node.Name] = true
}
}

// get azVolumeAttachments of the nodes in the zone
azVolumeAttachments, err := clientsetAzDisk.DiskV1beta1().AzVolumeAttachments(consts.DefaultAzureDiskCrdNamespace).List(context.Background(), metav1.ListOptions{})
if err != nil {
panic(err.Error())
}

for _, azVolumeAttachment := range azVolumeAttachments.Items {
if nodeSet[azVolumeAttachment.Spec.NodeName] {
result = append(result, AzvaResource{
ResourceType: zoneName,
Namespace: azVolumeAttachment.Namespace,
Name: azVolumeAttachment.Name,
Age: metav1.Now().Sub(azVolumeAttachment.CreationTimestamp.Time),
RequestRole: azVolumeAttachment.Spec.RequestedRole,
Role: azVolumeAttachment.Status.Detail.Role,
State: azVolumeAttachment.Status.State})
}
}

return result
}

func displayAzva(result []AzvaResource, typeName string) {
table := tablewriter.NewWriter(os.Stdout)
table.SetHeader([]string{strings.ToUpper(typeName) + "NAME", "NAMESPACE", "NAME", "AGE", "REQUESTEDROLE", "ROLE", "STATE"})

for _, azva := range result {
table.Append([]string{azva.ResourceType, azva.Namespace, azva.Name, azva.Age.String()[:2] + "h", string(azva.RequestRole), string(azva.Role), string(azva.State)})
}

table.Render()
}
Loading