Files
kubevela/references/cli/utils.go
barnettZQG 976d683185 Feat: refactor CLI commands related to resources (#4500)
* Feat: refactor CLI commands related to resources

Signed-off-by: barnettZQG <barnett.zqg@gmail.com>

* Fix: remove the old test case.

Signed-off-by: barnettZQG <barnett.zqg@gmail.com>

* Fix: e2e test

Signed-off-by: barnettZQG <barnett.zqg@gmail.com>

* Fix: optimize test cases

Signed-off-by: barnettZQG <barnett.zqg@gmail.com>

* Feat: rename 'vela pods' to 'vela status --pod'

Signed-off-by: barnettZQG <barnett.zqg@gmail.com>

* Feat: optimize the e2e test case

Signed-off-by: barnettZQG <barnett.zqg@gmail.com>

* Fix: sort the objects

Signed-off-by: barnettZQG <barnett.zqg@gmail.com>

* Fix: optimize the e2e test case

Signed-off-by: barnettZQG <barnett.zqg@gmail.com>

* Fix: list the pod by the labels

Signed-off-by: barnettZQG <barnett.zqg@gmail.com>

* Fix: order the tree resource

Signed-off-by: barnettZQG <barnett.zqg@gmail.com>

* Fix: set multicluster config

Signed-off-by: barnettZQG <barnett.zqg@gmail.com>
2022-08-01 19:44:27 +08:00

232 lines
5.8 KiB
Go

/*
Copyright 2021 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 (
"bufio"
"bytes"
"encoding/json"
"fmt"
"io"
"log"
"os"
"strings"
"github.com/AlecAivazis/survey/v2"
"github.com/pkg/errors"
corev1 "k8s.io/api/core/v1"
"k8s.io/apimachinery/pkg/runtime"
"k8s.io/client-go/util/jsonpath"
"k8s.io/kubectl/pkg/cmd/get"
"sigs.k8s.io/yaml"
"github.com/oam-dev/kubevela/apis/core.oam.dev/v1beta1"
"github.com/oam-dev/kubevela/pkg/velaql/providers/query/types"
)
// UserInput user input in command
type UserInput struct {
Writer io.Writer
Reader *bufio.Reader
}
// UserInputOptions user input options
type UserInputOptions struct {
AssumeYes bool
}
// NewUserInput new user input util
func NewUserInput() *UserInput {
return &UserInput{
Writer: os.Stdout,
Reader: bufio.NewReader(os.Stdin),
}
}
// AskBool format the answer to bool type
func (ui *UserInput) AskBool(question string, opts *UserInputOptions) bool {
fmt.Fprintf(ui.Writer, "%s (y/n)", question)
if opts.AssumeYes {
return true
}
line, err := ui.read()
if err != nil {
log.Fatal(err.Error())
}
if input := strings.TrimSpace(strings.ToLower(line)); input == "y" || input == "yes" {
return true
}
return false
}
func (ui *UserInput) read() (string, error) {
line, err := ui.Reader.ReadString('\n')
if err != nil && !errors.Is(err, io.EOF) {
return "", err
}
resultStr := strings.TrimSuffix(line, "\n")
return resultStr, err
}
// formatApplicationString formats an Application to string in yaml/json/jsonpath for printing (without managedFields).
//
// format = "yaml" / "json" / "jsonpath={.field}"
func formatApplicationString(format string, app *v1beta1.Application) (string, error) {
var ret string
if format == "" {
return "", fmt.Errorf("no format provided")
}
// No, we don't want managedFields, get rid of it.
app.ManagedFields = nil
switch format {
case "yaml":
b, err := yaml.Marshal(app)
if err != nil {
return "", err
}
ret = string(b)
case "json":
b, err := json.MarshalIndent(app, "", " ")
if err != nil {
return "", err
}
ret = string(b)
default:
// format is not any of json/yaml/jsonpath, not supported
if !strings.HasPrefix(format, "jsonpath") {
return "", fmt.Errorf("output %s is not supported", format)
}
// format = jsonpath
s := strings.Split(format, "=")
if len(s) < 2 {
return "", fmt.Errorf("jsonpath template format specified but no template given")
}
path, err := get.RelaxedJSONPathExpression(s[1])
if err != nil {
return "", err
}
jp := jsonpath.New("").AllowMissingKeys(true)
err = jp.Parse(path)
if err != nil {
return "", err
}
buf := &bytes.Buffer{}
err = jp.Execute(buf, app)
if err != nil {
return "", err
}
ret = buf.String()
}
return ret, nil
}
// AskToChooseOnePod will ask user to select one pod
func AskToChooseOnePod(pods []types.PodBase) (*types.PodBase, error) {
if len(pods) == 0 {
return nil, errors.New("no pod found in your application")
}
if len(pods) == 1 {
return &pods[0], nil
}
var ops []string
for i := 0; i < len(pods); i++ {
pod := pods[i]
ops = append(ops, fmt.Sprintf("%s | %s | %s", pod.Cluster, pod.Component, pod.Metadata.Name))
}
prompt := &survey.Select{
Message: fmt.Sprintf("There are %d pods match your filter conditions. Please choose one:\nCluster | Component | Pod", len(ops)),
Options: ops,
}
var selectedRsc string
err := survey.AskOne(prompt, &selectedRsc)
if err != nil {
return nil, fmt.Errorf("choosing pod err %w", err)
}
for k, resource := range ops {
if selectedRsc == resource {
return &pods[k], nil
}
}
// it should never happen.
return nil, errors.New("no pod match for your choice")
}
// AskToChooseOneService will ask user to select one service and/or port
func AskToChooseOneService(services []types.ResourceItem, selectPort bool) (*types.ResourceItem, int, error) {
if len(services) == 0 {
return nil, 0, errors.New("no service found in your application")
}
var ops []string
var res []struct {
item types.ResourceItem
port int
}
for i := 0; i < len(services); i++ {
obj := services[i]
service := &corev1.Service{}
if err := runtime.DefaultUnstructuredConverter.FromUnstructured(obj.Object.Object, service); err == nil {
if selectPort {
for _, port := range service.Spec.Ports {
ops = append(ops, fmt.Sprintf("%s | %s | %s:%d", obj.Cluster, obj.Component, obj.Object.GetName(), port.Port))
res = append(res, struct {
item types.ResourceItem
port int
}{
item: obj,
port: int(port.Port),
})
}
} else {
ops = append(ops, fmt.Sprintf("%s | %s | %s", obj.Cluster, obj.Component, obj.Object.GetName()))
res = append(res, struct {
item types.ResourceItem
port int
}{
item: obj,
})
}
}
}
if len(ops) == 1 {
return &res[0].item, res[0].port, nil
}
prompt := &survey.Select{
Message: fmt.Sprintf("There are %d services match your filter conditions. Please choose one:\nCluster | Component | Service", len(ops)),
Options: ops,
}
var selectedRsc string
err := survey.AskOne(prompt, &selectedRsc)
if err != nil {
return nil, 0, fmt.Errorf("choosing service err %w", err)
}
for k, resource := range ops {
if selectedRsc == resource {
return &res[k].item, res[k].port, nil
}
}
// it should never happen.
return nil, 0, errors.New("no service match for your choice")
}