Files
kubevela/references/cli/kube.go
Somefive 897f07fbec Feat: vela kube delete (#3952)
Signed-off-by: Somefive <yd219913@alibaba-inc.com>
2022-05-24 09:38:22 +08:00

364 lines
12 KiB
Go

/*
Copyright 2022 The KubeVela 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 cli
import (
"bytes"
"fmt"
"io"
"strings"
"github.com/pkg/errors"
"github.com/spf13/cobra"
"gopkg.in/yaml.v3"
apierrors "k8s.io/apimachinery/pkg/api/errors"
"k8s.io/apimachinery/pkg/api/meta"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
"k8s.io/apimachinery/pkg/runtime"
"k8s.io/apimachinery/pkg/runtime/schema"
apitypes "k8s.io/apimachinery/pkg/types"
"k8s.io/kubectl/pkg/util/i18n"
"k8s.io/kubectl/pkg/util/templates"
"k8s.io/utils/strings/slices"
"sigs.k8s.io/controller-runtime/pkg/client"
"github.com/oam-dev/kubevela/apis/types"
velacmd "github.com/oam-dev/kubevela/pkg/cmd"
cmdutil "github.com/oam-dev/kubevela/pkg/cmd/util"
"github.com/oam-dev/kubevela/pkg/multicluster"
"github.com/oam-dev/kubevela/pkg/utils"
"github.com/oam-dev/kubevela/pkg/utils/util"
)
// KubeCommandGroup command group for native resource management
func KubeCommandGroup(f velacmd.Factory, streams util.IOStreams) *cobra.Command {
cmd := &cobra.Command{
Use: "kube",
Short: i18n.T("Managing native Kubernetes resources across clusters."),
Annotations: map[string]string{
types.TagCommandType: types.TypeCD,
},
Run: func(cmd *cobra.Command, args []string) {
},
}
cmd.AddCommand(NewKubeApplyCommand(f, streams))
cmd.AddCommand(NewKubeDeleteCommand(f, streams))
return cmd
}
// KubeApplyOptions options for kube apply
type KubeApplyOptions struct {
files []string
clusters []string
namespace string
dryRun bool
filesData []utils.FileData
objects []*unstructured.Unstructured
util.IOStreams
}
// Complete .
func (opt *KubeApplyOptions) Complete(f velacmd.Factory, cmd *cobra.Command) error {
opt.namespace = velacmd.GetNamespace(f, cmd)
opt.clusters = velacmd.GetClusters(cmd)
var paths []string
for _, file := range opt.files {
path := strings.TrimSpace(file)
if !slices.Contains(paths, path) {
paths = append(paths, path)
}
}
for _, path := range paths {
data, err := utils.LoadDataFromPath(cmd.Context(), path, utils.IsJSONOrYAMLFile)
if err != nil {
return err
}
opt.filesData = append(opt.filesData, data...)
}
return nil
}
// Validate .
func (opt *KubeApplyOptions) Validate() error {
if len(opt.files) == 0 {
return fmt.Errorf("at least one file should be specified with the --file flag")
}
if len(opt.filesData) == 0 {
return fmt.Errorf("not file found")
}
for _, fileData := range opt.filesData {
decoder := yaml.NewDecoder(bytes.NewReader(fileData.Data))
for {
obj := &unstructured.Unstructured{Object: map[string]interface{}{}}
err := decoder.Decode(obj.Object)
if err != nil {
if errors.Is(err, io.EOF) {
break
}
return fmt.Errorf("failed to decode object in %s: %w", fileData.Path, err)
}
if opt.namespace != "" {
obj.SetNamespace(opt.namespace)
} else if obj.GetNamespace() == "" {
obj.SetNamespace(metav1.NamespaceDefault)
}
opt.objects = append(opt.objects, obj)
}
}
return nil
}
// Run .
func (opt *KubeApplyOptions) Run(f velacmd.Factory, cmd *cobra.Command) error {
if opt.dryRun {
for i, obj := range opt.objects {
if i > 0 {
_, _ = fmt.Fprintf(opt.Out, "---\n")
}
bs, err := yaml.Marshal(obj.Object)
if err != nil {
return err
}
_, _ = opt.Out.Write(bs)
}
return nil
}
for i, cluster := range opt.clusters {
if i > 0 {
_, _ = fmt.Fprintf(opt.Out, "\n")
}
_, _ = fmt.Fprintf(opt.Out, "Apply objects in cluster %s.\n", cluster)
ctx := multicluster.ContextWithClusterName(cmd.Context(), cluster)
for _, obj := range opt.objects {
copiedObj := &unstructured.Unstructured{}
bs, err := obj.MarshalJSON()
if err != nil {
return err
}
if err = copiedObj.UnmarshalJSON(bs); err != nil {
return err
}
res, err := utils.CreateOrUpdate(ctx, f.Client(), copiedObj)
if err != nil {
return err
}
key := strings.TrimPrefix(obj.GetNamespace()+"/"+obj.GetName(), "/")
_, _ = fmt.Fprintf(opt.Out, " %s %s %s.\n", obj.GetKind(), key, res)
}
}
return nil
}
var (
kubeApplyLong = templates.LongDesc(i18n.T(`
Apply Kubernetes objects in clusters
Apply Kubernetes objects in multiple clusters. Use --clusters to specify which clusters to
apply. If -n/--namespace is used, the original object namespace will be overrode.
You can use -f/--file to specify the object file to apply. Multiple file inputs are allowed.
Directory input and web url input is supported as well.`))
kubeApplyExample = templates.Examples(i18n.T(`
# Apply single object file in managed cluster
vela kube apply -f my.yaml --cluster cluster-1
# Apply multiple object files in multiple managed clusters
vela kube apply -f my-1.yaml -f my-2.yaml --cluster cluster-1 --cluster cluster-2
# Apply object file with web url in control plane
vela kube apply -f https://raw.githubusercontent.com/kubevela/kubevela/master/docs/examples/app-with-probe/app-with-probe.yaml
# Apply object files in directory to specified namespace in managed clusters
vela kube apply -f ./resources -n demo --cluster cluster-1 --cluster cluster-2`))
)
// NewKubeApplyCommand kube apply command
func NewKubeApplyCommand(f velacmd.Factory, streams util.IOStreams) *cobra.Command {
o := &KubeApplyOptions{IOStreams: streams}
cmd := &cobra.Command{
Use: "apply",
Short: i18n.T("Apply resources in Kubernetes YAML file to clusters."),
Long: kubeApplyLong,
Example: kubeApplyExample,
Annotations: map[string]string{
types.TagCommandType: types.TypeCD,
},
Args: cobra.ExactValidArgs(0),
Run: func(cmd *cobra.Command, args []string) {
cmdutil.CheckErr(o.Complete(f, cmd))
cmdutil.CheckErr(o.Validate())
cmdutil.CheckErr(o.Run(f, cmd))
},
}
cmd.Flags().StringSliceVarP(&o.files, "file", "f", o.files, "Files that include native Kubernetes objects to apply.")
cmd.Flags().BoolVarP(&o.dryRun, "dryrun", "", o.dryRun, "Setting this flag will not apply resources in clusters. It will print out the resource to be applied.")
return velacmd.NewCommandBuilder(f, cmd).
WithNamespaceFlag(
velacmd.NamespaceFlagDisableEnvOption{},
velacmd.UsageOption("The namespace to apply objects. If empty, the namespace declared in the YAML will be used."),
).
WithClusterFlag(velacmd.UsageOption("The cluster to apply objects. Setting multiple clusters will apply objects in order.")).
WithStreams(streams).
WithResponsiveWriter().
Build()
}
// KubeDeleteOptions options for kube delete
type KubeDeleteOptions struct {
clusters []string
namespace string
deleteAll bool
resource string
resourceName string
util.IOStreams
}
var (
kubeDeleteLong = templates.LongDesc(i18n.T(`
Delete Kubernetes objects in clusters
Delete Kubernetes objects in multiple clusters. Use --clusters to specify which clusters to
delete. Use -n/--namespace flags to specify which cluster the target resource locates.
Use --all flag to delete all this kind of objects in the target namespace and clusters.`))
kubeDeleteExample = templates.Examples(i18n.T(`
# Delete the deployment nginx in default namespace in cluster-1
vela kube delete deployment nginx --cluster cluster-1
# Delete the deployment nginx in demo namespace in cluster-1 and cluster-2
vela kube delete deployment nginx -n demo --cluster cluster-1 --cluster cluster-2
# Delete all deployments in demo namespace in cluster-1
vela kube delete deployment --all -n demo --cluster cluster-1`))
)
// Complete .
func (opt *KubeDeleteOptions) Complete(f velacmd.Factory, cmd *cobra.Command, args []string) {
opt.namespace = velacmd.GetNamespace(f, cmd)
if opt.namespace == "" {
opt.namespace = metav1.NamespaceDefault
}
opt.clusters = velacmd.GetClusters(cmd)
opt.resource = args[0]
if len(args) == 2 {
opt.resourceName = args[1]
}
}
// Validate .
func (opt *KubeDeleteOptions) Validate() error {
if opt.resourceName == "" && !opt.deleteAll {
return fmt.Errorf("either resource name or flag --all should be set")
}
if opt.resourceName != "" && opt.deleteAll {
return fmt.Errorf("cannot set resource name and flag --all at the same time")
}
return nil
}
// Run .
func (opt *KubeDeleteOptions) Run(f velacmd.Factory, cmd *cobra.Command) error {
gvks, err := f.Client().RESTMapper().KindsFor(schema.GroupVersionResource{Resource: opt.resource})
if err != nil {
return fmt.Errorf("failed to find kinds for resource %s: %w", opt.resource, err)
}
if len(gvks) == 0 {
return fmt.Errorf("no kinds found for resource %s", opt.resource)
}
gvk := gvks[0]
mappings, err := f.Client().RESTMapper().RESTMappings(gvk.GroupKind(), gvk.Version)
if err != nil {
return fmt.Errorf("failed to get mappings for resource %s: %w", opt.resource, err)
}
if len(mappings) == 0 {
return fmt.Errorf("no mappings found for resource %s", opt.resource)
}
mapping := mappings[0]
namespaced := mapping.Scope.Name() == meta.RESTScopeNameNamespace
for _, cluster := range opt.clusters {
ctx := multicluster.ContextWithClusterName(cmd.Context(), cluster)
objs, obj := &unstructured.UnstructuredList{}, &unstructured.Unstructured{}
objs.SetGroupVersionKind(gvk)
obj.SetGroupVersionKind(gvk)
switch {
case opt.deleteAll && namespaced:
err = f.Client().List(ctx, objs, client.InNamespace(opt.namespace))
case opt.deleteAll && !namespaced:
err = f.Client().List(ctx, objs)
case !opt.deleteAll && namespaced:
err = f.Client().Get(ctx, apitypes.NamespacedName{Namespace: opt.namespace, Name: opt.resourceName}, obj)
case !opt.deleteAll && !namespaced:
err = f.Client().Get(ctx, apitypes.NamespacedName{Name: opt.resourceName}, obj)
}
if err != nil && !apierrors.IsNotFound(err) && !runtime.IsNotRegisteredError(err) && !meta.IsNoMatchError(err) {
return fmt.Errorf("failed to retrieve %s in cluster %s: %w", opt.resource, cluster, err)
}
for _, toDel := range append(objs.Items, *obj) {
key := toDel.GetName()
if key == "" {
continue
}
if namespaced {
key = toDel.GetNamespace() + "/" + key
}
if err = f.Client().Delete(ctx, toDel.DeepCopy()); err != nil {
return fmt.Errorf("failed to delete %s %s in cluster %s: %w", opt.resource, key, cluster, err)
}
_, _ = fmt.Fprintf(opt.IOStreams.Out, "%s %s in cluster %s deleted.\n", opt.resource, key, cluster)
}
}
return nil
}
// NewKubeDeleteCommand kube delete command
func NewKubeDeleteCommand(f velacmd.Factory, streams util.IOStreams) *cobra.Command {
o := &KubeDeleteOptions{IOStreams: streams}
cmd := &cobra.Command{
Use: "delete",
Short: i18n.T("Delete resources in clusters."),
Long: kubeDeleteLong,
Example: kubeDeleteExample,
Annotations: map[string]string{
types.TagCommandType: types.TypeCD,
},
Args: cobra.RangeArgs(1, 2),
Run: func(cmd *cobra.Command, args []string) {
o.Complete(f, cmd, args)
cmdutil.CheckErr(o.Validate())
cmdutil.CheckErr(o.Run(f, cmd))
},
}
cmd.Flags().BoolVarP(&o.deleteAll, "all", "", o.deleteAll, "Setting this flag will delete all this kind of resources.")
return velacmd.NewCommandBuilder(f, cmd).
WithNamespaceFlag(
velacmd.NamespaceFlagDisableEnvOption{},
velacmd.UsageOption("The namespace to delete objects. If empty, the default namespace will be used."),
).
WithClusterFlag(velacmd.UsageOption("The cluster to delete objects. Setting multiple clusters will delete objects in order.")).
WithStreams(streams).
WithResponsiveWriter().
Build()
}