2021-03-26 15:24:19 +08:00
|
|
|
/*
|
|
|
|
|
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.
|
|
|
|
|
*/
|
|
|
|
|
|
2020-11-16 18:55:01 +08:00
|
|
|
package common
|
2020-09-24 16:30:12 +08:00
|
|
|
|
|
|
|
|
import (
|
2020-12-18 10:18:16 +08:00
|
|
|
"bytes"
|
2020-11-18 16:08:16 +08:00
|
|
|
"context"
|
2020-12-18 10:18:16 +08:00
|
|
|
"encoding/json"
|
|
|
|
|
"errors"
|
2020-09-24 16:30:12 +08:00
|
|
|
"fmt"
|
2021-01-28 18:36:52 +08:00
|
|
|
"io"
|
2020-11-18 16:08:16 +08:00
|
|
|
"io/ioutil"
|
|
|
|
|
"net/http"
|
2020-09-24 16:30:12 +08:00
|
|
|
"os"
|
2021-01-28 18:36:52 +08:00
|
|
|
"os/exec"
|
2020-12-18 10:18:16 +08:00
|
|
|
"path/filepath"
|
2020-09-24 16:30:12 +08:00
|
|
|
|
2020-12-18 10:18:16 +08:00
|
|
|
"cuelang.org/go/cue"
|
2021-08-11 13:59:46 +08:00
|
|
|
"cuelang.org/go/cue/ast"
|
2021-06-15 13:36:23 +08:00
|
|
|
"cuelang.org/go/cue/format"
|
2020-12-18 10:18:16 +08:00
|
|
|
"cuelang.org/go/encoding/openapi"
|
2021-02-20 04:11:26 +08:00
|
|
|
"github.com/AlecAivazis/survey/v2"
|
2021-03-20 12:08:15 +08:00
|
|
|
"github.com/ghodss/yaml"
|
2021-06-03 11:23:20 +08:00
|
|
|
"github.com/hashicorp/hcl/v2/hclparse"
|
|
|
|
|
"github.com/oam-dev/terraform-config-inspect/tfconfig"
|
2021-05-13 21:46:45 +08:00
|
|
|
terraformv1beta1 "github.com/oam-dev/terraform-controller/api/v1beta1"
|
2021-03-24 08:01:24 +08:00
|
|
|
kruise "github.com/openkruise/kruise-api/apps/v1alpha1"
|
2020-10-15 11:51:41 +08:00
|
|
|
certmanager "github.com/wonderflow/cert-manager-api/pkg/apis/certmanager/v1"
|
2021-03-24 08:01:24 +08:00
|
|
|
istioclientv1beta1 "istio.io/client-go/pkg/apis/networking/v1beta1"
|
|
|
|
|
crdv1 "k8s.io/apiextensions-apiserver/pkg/apis/apiextensions/v1"
|
2021-07-30 10:02:51 +08:00
|
|
|
"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
|
2020-11-17 14:25:58 +08:00
|
|
|
k8sruntime "k8s.io/apimachinery/pkg/runtime"
|
2021-07-30 10:02:51 +08:00
|
|
|
"k8s.io/apimachinery/pkg/runtime/schema"
|
2020-09-24 16:30:12 +08:00
|
|
|
clientgoscheme "k8s.io/client-go/kubernetes/scheme"
|
2021-08-06 10:46:31 +08:00
|
|
|
ocmclusterv1alpha1 "open-cluster-management.io/api/cluster/v1alpha1"
|
|
|
|
|
ocmworkv1 "open-cluster-management.io/api/work/v1"
|
2021-08-11 23:05:10 +08:00
|
|
|
"sigs.k8s.io/controller-runtime/pkg/client"
|
2020-11-17 14:25:58 +08:00
|
|
|
"sigs.k8s.io/controller-runtime/pkg/client/config"
|
2020-09-24 16:30:12 +08:00
|
|
|
|
2021-03-24 08:01:24 +08:00
|
|
|
oamcore "github.com/oam-dev/kubevela/apis/core.oam.dev"
|
|
|
|
|
oamstandard "github.com/oam-dev/kubevela/apis/standard.oam.dev/v1alpha1"
|
2021-06-02 15:37:06 +08:00
|
|
|
velacue "github.com/oam-dev/kubevela/pkg/cue"
|
2020-09-24 16:30:12 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
var (
|
2020-11-17 17:46:56 +08:00
|
|
|
// Scheme defines the default KubeVela schema
|
2020-09-24 16:30:12 +08:00
|
|
|
Scheme = k8sruntime.NewScheme()
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
|
_ = clientgoscheme.AddToScheme(Scheme)
|
2021-03-24 08:01:24 +08:00
|
|
|
_ = crdv1.AddToScheme(Scheme)
|
|
|
|
|
_ = oamcore.AddToScheme(Scheme)
|
|
|
|
|
_ = oamstandard.AddToScheme(Scheme)
|
|
|
|
|
_ = istioclientv1beta1.AddToScheme(Scheme)
|
2020-09-24 16:30:12 +08:00
|
|
|
_ = certmanager.AddToScheme(Scheme)
|
2021-03-24 08:01:24 +08:00
|
|
|
_ = kruise.AddToScheme(Scheme)
|
2021-05-13 21:46:45 +08:00
|
|
|
_ = terraformv1beta1.AddToScheme(Scheme)
|
2021-08-06 10:46:31 +08:00
|
|
|
_ = ocmclusterv1alpha1.Install(Scheme)
|
|
|
|
|
_ = ocmworkv1.Install(Scheme)
|
2020-09-24 16:30:12 +08:00
|
|
|
// +kubebuilder:scaffold:scheme
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-17 17:46:56 +08:00
|
|
|
// InitBaseRestConfig will return reset config for create controller runtime client
|
2021-03-23 16:30:49 +08:00
|
|
|
func InitBaseRestConfig() (Args, error) {
|
2020-09-24 16:30:12 +08:00
|
|
|
restConf, err := config.GetConfig()
|
2021-08-06 17:10:52 +08:00
|
|
|
if err != nil && os.Getenv("IGNORE_KUBE_CONFIG") != "true" {
|
2020-09-24 16:30:12 +08:00
|
|
|
fmt.Println("get kubeConfig err", err)
|
|
|
|
|
os.Exit(1)
|
|
|
|
|
}
|
|
|
|
|
|
2021-03-23 16:30:49 +08:00
|
|
|
return Args{
|
2020-09-24 16:30:12 +08:00
|
|
|
Config: restConf,
|
|
|
|
|
Schema: Scheme,
|
2020-11-12 13:25:06 +08:00
|
|
|
}, nil
|
2020-09-24 16:30:12 +08:00
|
|
|
}
|
2020-11-18 16:08:16 +08:00
|
|
|
|
|
|
|
|
// HTTPGet will send GET http request with context
|
|
|
|
|
func HTTPGet(ctx context.Context, url string) ([]byte, error) {
|
|
|
|
|
// Change NewRequest to NewRequestWithContext and pass context it
|
|
|
|
|
req, err := http.NewRequestWithContext(ctx, http.MethodGet, url, nil)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
resp, err := http.DefaultClient.Do(req)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
//nolint:errcheck
|
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
return ioutil.ReadAll(resp.Body)
|
|
|
|
|
}
|
2020-12-18 10:18:16 +08:00
|
|
|
|
|
|
|
|
// GetCUEParameterValue converts definitions to cue format
|
|
|
|
|
func GetCUEParameterValue(cueStr string) (cue.Value, error) {
|
|
|
|
|
r := cue.Runtime{}
|
2021-06-02 15:37:06 +08:00
|
|
|
template, err := r.Compile("", cueStr+velacue.BaseTemplate)
|
2020-12-18 10:18:16 +08:00
|
|
|
if err != nil {
|
|
|
|
|
return cue.Value{}, err
|
|
|
|
|
}
|
|
|
|
|
tempStruct, err := template.Value().Struct()
|
|
|
|
|
if err != nil {
|
|
|
|
|
return cue.Value{}, err
|
|
|
|
|
}
|
|
|
|
|
// find the parameter definition
|
|
|
|
|
var paraDef cue.FieldInfo
|
|
|
|
|
var found bool
|
|
|
|
|
for i := 0; i < tempStruct.Len(); i++ {
|
|
|
|
|
paraDef = tempStruct.Field(i)
|
2021-06-02 15:37:06 +08:00
|
|
|
if paraDef.Name == velacue.ParameterTag {
|
2020-12-18 10:18:16 +08:00
|
|
|
found = true
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if !found {
|
|
|
|
|
return cue.Value{}, errors.New("parameter not exist")
|
|
|
|
|
}
|
|
|
|
|
arguments := paraDef.Value
|
|
|
|
|
|
|
|
|
|
return arguments, nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// GenOpenAPI generates OpenAPI json schema from cue.Instance
|
|
|
|
|
func GenOpenAPI(inst *cue.Instance) ([]byte, error) {
|
|
|
|
|
if inst.Err != nil {
|
|
|
|
|
return nil, inst.Err
|
|
|
|
|
}
|
2021-06-15 13:36:23 +08:00
|
|
|
paramOnlyIns, err := RefineParameterInstance(inst)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
2020-12-18 10:18:16 +08:00
|
|
|
defaultConfig := &openapi.Config{}
|
2021-06-15 13:36:23 +08:00
|
|
|
b, err := openapi.Gen(paramOnlyIns, defaultConfig)
|
2020-12-18 10:18:16 +08:00
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
var out = &bytes.Buffer{}
|
|
|
|
|
_ = json.Indent(out, b, "", " ")
|
|
|
|
|
return out.Bytes(), nil
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-11 13:59:46 +08:00
|
|
|
// extractParameterDefinitionNodeFromInstance extracts the `#parameter` ast.Node from root instance, if failed fall back to `parameter` by LookUpDef
|
|
|
|
|
func extractParameterDefinitionNodeFromInstance(inst *cue.Instance) ast.Node {
|
|
|
|
|
opts := []cue.Option{cue.All(), cue.DisallowCycles(true), cue.ResolveReferences(true), cue.Docs(true)}
|
|
|
|
|
node := inst.Value().Syntax(opts...)
|
|
|
|
|
if fileNode, ok := node.(*ast.File); ok {
|
|
|
|
|
for _, decl := range fileNode.Decls {
|
|
|
|
|
if field, ok := decl.(*ast.Field); ok {
|
|
|
|
|
if label, ok := field.Label.(*ast.Ident); ok && label.Name == "#"+velacue.ParameterTag {
|
|
|
|
|
return decl.(*ast.Field).Value
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
paramVal := inst.LookupDef(velacue.ParameterTag)
|
|
|
|
|
return paramVal.Syntax(opts...)
|
|
|
|
|
}
|
|
|
|
|
|
2021-06-15 13:36:23 +08:00
|
|
|
// RefineParameterInstance refines cue instance to merely include `parameter` identifier
|
|
|
|
|
func RefineParameterInstance(inst *cue.Instance) (*cue.Instance, error) {
|
|
|
|
|
r := cue.Runtime{}
|
|
|
|
|
paramVal := inst.LookupDef(velacue.ParameterTag)
|
|
|
|
|
var paramOnlyStr string
|
|
|
|
|
switch k := paramVal.IncompleteKind(); k {
|
|
|
|
|
case cue.StructKind, cue.ListKind:
|
2021-08-11 13:59:46 +08:00
|
|
|
paramSyntax, _ := format.Node(extractParameterDefinitionNodeFromInstance(inst))
|
2021-06-15 13:36:23 +08:00
|
|
|
paramOnlyStr = fmt.Sprintf("#%s: %s\n", velacue.ParameterTag, string(paramSyntax))
|
|
|
|
|
case cue.IntKind, cue.StringKind, cue.FloatKind, cue.BoolKind:
|
|
|
|
|
paramOnlyStr = fmt.Sprintf("#%s: %v", velacue.ParameterTag, paramVal)
|
|
|
|
|
case cue.BottomKind:
|
|
|
|
|
paramOnlyStr = fmt.Sprintf("#%s: {}", velacue.ParameterTag)
|
|
|
|
|
default:
|
|
|
|
|
return nil, fmt.Errorf("unsupport parameter kind: %s", k.String())
|
|
|
|
|
}
|
|
|
|
|
paramOnlyIns, err := r.Compile("-", paramOnlyStr)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
return paramOnlyIns, nil
|
|
|
|
|
}
|
|
|
|
|
|
2021-01-28 18:36:52 +08:00
|
|
|
// RealtimePrintCommandOutput prints command output in real time
|
|
|
|
|
// If logFile is "", it will prints the stdout, or it will write to local file
|
|
|
|
|
func RealtimePrintCommandOutput(cmd *exec.Cmd, logFile string) error {
|
|
|
|
|
var writer io.Writer
|
|
|
|
|
if logFile == "" {
|
|
|
|
|
writer = io.MultiWriter(os.Stdout)
|
|
|
|
|
} else {
|
|
|
|
|
if _, err := os.Stat(filepath.Dir(logFile)); err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
f, err := os.Create(filepath.Clean(logFile))
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
writer = io.MultiWriter(f)
|
|
|
|
|
}
|
|
|
|
|
cmd.Stdout = writer
|
|
|
|
|
cmd.Stderr = writer
|
|
|
|
|
if err := cmd.Run(); err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
return nil
|
|
|
|
|
}
|
2021-02-20 04:11:26 +08:00
|
|
|
|
|
|
|
|
// AskToChooseOneService will ask users to select one service of the application if more than one exidi
|
|
|
|
|
func AskToChooseOneService(svcNames []string) (string, error) {
|
|
|
|
|
if len(svcNames) == 0 {
|
|
|
|
|
return "", fmt.Errorf("no service exist in the application")
|
|
|
|
|
}
|
|
|
|
|
if len(svcNames) == 1 {
|
|
|
|
|
return svcNames[0], nil
|
|
|
|
|
}
|
|
|
|
|
prompt := &survey.Select{
|
|
|
|
|
Message: "You have multiple services in your app. Please choose one service: ",
|
|
|
|
|
Options: svcNames,
|
|
|
|
|
}
|
|
|
|
|
var svcName string
|
|
|
|
|
err := survey.AskOne(prompt, &svcName)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return "", fmt.Errorf("choosing service err %w", err)
|
|
|
|
|
}
|
|
|
|
|
return svcName, nil
|
|
|
|
|
}
|
2021-03-20 12:08:15 +08:00
|
|
|
|
|
|
|
|
// ReadYamlToObject will read a yaml K8s object to runtime.Object
|
|
|
|
|
func ReadYamlToObject(path string, object k8sruntime.Object) error {
|
|
|
|
|
data, err := ioutil.ReadFile(filepath.Clean(path))
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
return yaml.Unmarshal(data, object)
|
|
|
|
|
}
|
2021-06-03 11:23:20 +08:00
|
|
|
|
|
|
|
|
// ParseTerraformVariables get variables from Terraform Configuration
|
|
|
|
|
func ParseTerraformVariables(configuration string) (map[string]*tfconfig.Variable, error) {
|
|
|
|
|
p := hclparse.NewParser()
|
|
|
|
|
hclFile, diagnostic := p.ParseHCL([]byte(configuration), "")
|
|
|
|
|
if diagnostic != nil {
|
|
|
|
|
return nil, errors.New(diagnostic.Error())
|
|
|
|
|
}
|
|
|
|
|
mod := tfconfig.Module{Variables: map[string]*tfconfig.Variable{}}
|
|
|
|
|
diagnostic = tfconfig.LoadModuleFromFile(hclFile, &mod)
|
|
|
|
|
if diagnostic != nil {
|
|
|
|
|
return nil, errors.New(diagnostic.Error())
|
|
|
|
|
}
|
|
|
|
|
return mod.Variables, nil
|
|
|
|
|
}
|
2021-07-30 10:02:51 +08:00
|
|
|
|
|
|
|
|
// GenerateUnstructuredObj generate UnstructuredObj
|
|
|
|
|
func GenerateUnstructuredObj(name, ns string, gvk schema.GroupVersionKind) *unstructured.Unstructured {
|
|
|
|
|
u := &unstructured.Unstructured{}
|
|
|
|
|
u.SetGroupVersionKind(gvk)
|
|
|
|
|
u.SetName(name)
|
|
|
|
|
u.SetNamespace(ns)
|
|
|
|
|
return u
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// SetSpecObjIntoUnstructuredObj set UnstructuredObj spec field
|
|
|
|
|
func SetSpecObjIntoUnstructuredObj(spec interface{}, u *unstructured.Unstructured) error {
|
|
|
|
|
bts, err := json.Marshal(spec)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
data := make(map[string]interface{})
|
|
|
|
|
if err := json.Unmarshal(bts, &data); err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
_ = unstructured.SetNestedMap(u.Object, data, "spec")
|
|
|
|
|
return nil
|
|
|
|
|
}
|
2021-08-11 23:05:10 +08:00
|
|
|
|
|
|
|
|
// NewK8sClient init a local k8s client which add oamcore scheme
|
|
|
|
|
func NewK8sClient() (client.Client, error) {
|
|
|
|
|
conf, err := config.GetConfig()
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
scheme := k8sruntime.NewScheme()
|
|
|
|
|
if err := clientgoscheme.AddToScheme(scheme); err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
if err := oamcore.AddToScheme(scheme); err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
k8sClient, err := client.New(conf, client.Options{Scheme: scheme})
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
return k8sClient, nil
|
|
|
|
|
}
|