mirror of
https://github.com/kubevela/kubevela.git
synced 2026-05-17 06:46:47 +00:00
* Add JSON format appfile support Co-authored-by: 摩羯 <wangxun@cai-inc.com> Co-authored-by: wangxun <wangxun@zcydeMBP.lan>
262 lines
7.4 KiB
Go
262 lines
7.4 KiB
Go
package commands
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"fmt"
|
|
"io/ioutil"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
|
|
"github.com/pkg/errors"
|
|
"github.com/spf13/cobra"
|
|
apierrors "k8s.io/apimachinery/pkg/api/errors"
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
k8sjson "k8s.io/apimachinery/pkg/runtime/serializer/json"
|
|
apitypes "k8s.io/apimachinery/pkg/types"
|
|
"sigs.k8s.io/controller-runtime/pkg/client"
|
|
|
|
"github.com/oam-dev/kubevela/apis/core.oam.dev/v1alpha2"
|
|
"github.com/oam-dev/kubevela/apis/types"
|
|
"github.com/oam-dev/kubevela/pkg/appfile"
|
|
"github.com/oam-dev/kubevela/pkg/appfile/template"
|
|
"github.com/oam-dev/kubevela/pkg/application"
|
|
cmdutil "github.com/oam-dev/kubevela/pkg/commands/util"
|
|
"github.com/oam-dev/kubevela/pkg/oam"
|
|
"github.com/oam-dev/kubevela/pkg/utils/common"
|
|
)
|
|
|
|
var (
|
|
appFilePath string
|
|
)
|
|
|
|
// NewUpCommand will create command for applying an AppFile
|
|
func NewUpCommand(c types.Args, ioStream cmdutil.IOStreams) *cobra.Command {
|
|
cmd := &cobra.Command{
|
|
Use: "up",
|
|
DisableFlagsInUseLine: true,
|
|
Short: "Apply an appfile",
|
|
Long: "Apply an appfile",
|
|
Annotations: map[string]string{
|
|
types.TagCommandType: types.TypeStart,
|
|
},
|
|
PersistentPreRunE: func(cmd *cobra.Command, args []string) error {
|
|
return c.SetConfig()
|
|
},
|
|
RunE: func(cmd *cobra.Command, args []string) error {
|
|
velaEnv, err := GetEnv(cmd)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
kubecli, err := client.New(c.Config, client.Options{Scheme: c.Schema})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
o := &AppfileOptions{
|
|
Kubecli: kubecli,
|
|
IO: ioStream,
|
|
Env: velaEnv,
|
|
}
|
|
filePath, err := cmd.Flags().GetString(appFilePath)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return o.Run(filePath)
|
|
},
|
|
}
|
|
cmd.SetOut(ioStream.Out)
|
|
|
|
cmd.Flags().StringP(appFilePath, "f", "", "specify file path for appfile")
|
|
return cmd
|
|
}
|
|
|
|
// AppfileOptions is some configuration that modify options for an Appfile
|
|
type AppfileOptions struct {
|
|
Kubecli client.Client
|
|
IO cmdutil.IOStreams
|
|
Env *types.EnvMeta
|
|
}
|
|
|
|
func saveRemoteAppfile(url string) (string, error) {
|
|
body, err := common.HTTPGet(context.Background(), url)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
ext := filepath.Ext(url)
|
|
dest := "Appfile"
|
|
if ext == ".json" {
|
|
dest = "vela.json"
|
|
} else if ext == ".yaml" || ext == ".yml" {
|
|
dest = "vela.yaml"
|
|
}
|
|
//nolint:gosec
|
|
return dest, ioutil.WriteFile(dest, body, 0644)
|
|
}
|
|
|
|
type buildResult struct {
|
|
app *appfile.AppFile
|
|
appConfig *v1alpha2.ApplicationConfiguration
|
|
comps []*v1alpha2.Component
|
|
scopes []oam.Object
|
|
}
|
|
|
|
func (o *AppfileOptions) export(filePath string, quiet bool) (*buildResult, []byte, error) {
|
|
var app *appfile.AppFile
|
|
var err error
|
|
if !quiet {
|
|
o.IO.Info("Parsing vela appfile ...")
|
|
}
|
|
if filePath != "" {
|
|
if strings.HasPrefix(filePath, "https://") || strings.HasPrefix(filePath, "http://") {
|
|
filePath, err = saveRemoteAppfile(filePath)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
}
|
|
app, err = appfile.LoadFromFile(filePath)
|
|
} else {
|
|
app, err = appfile.Load()
|
|
}
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
|
|
if !quiet {
|
|
o.IO.Info("Loading templates ...")
|
|
}
|
|
tm, err := template.Load()
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
|
|
comps, appConfig, scopes, err := app.BuildOAM(o.Env.Namespace, o.IO, tm, quiet)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
|
|
var w bytes.Buffer
|
|
|
|
enc := k8sjson.NewYAMLSerializer(k8sjson.DefaultMetaFactory, nil, nil)
|
|
appConfig.TypeMeta = metav1.TypeMeta{
|
|
APIVersion: v1alpha2.ApplicationConfigurationGroupVersionKind.GroupVersion().String(),
|
|
Kind: v1alpha2.ApplicationConfigurationKind,
|
|
}
|
|
err = enc.Encode(appConfig, &w)
|
|
if err != nil {
|
|
return nil, nil, fmt.Errorf("yaml encode AppConfig failed: %w", err)
|
|
}
|
|
w.WriteByte('\n')
|
|
|
|
for _, comp := range comps {
|
|
w.WriteString("---\n")
|
|
comp.TypeMeta = metav1.TypeMeta{
|
|
APIVersion: v1alpha2.ComponentGroupVersionKind.GroupVersion().String(),
|
|
Kind: v1alpha2.ComponentKind,
|
|
}
|
|
err = enc.Encode(comp, &w)
|
|
if err != nil {
|
|
return nil, nil, fmt.Errorf("yaml encode service (%s) failed: %w", comp.Name, err)
|
|
}
|
|
w.WriteByte('\n')
|
|
}
|
|
for _, scope := range scopes {
|
|
w.WriteString("---\n")
|
|
err = enc.Encode(scope, &w)
|
|
if err != nil {
|
|
return nil, nil, fmt.Errorf("yaml encode scope (%s) failed: %w", scope.GetName(), err)
|
|
}
|
|
w.WriteByte('\n')
|
|
}
|
|
br := &buildResult{
|
|
app: app,
|
|
appConfig: appConfig,
|
|
comps: comps,
|
|
scopes: scopes,
|
|
}
|
|
return br, w.Bytes(), nil
|
|
}
|
|
|
|
// Run starts an application according to Appfile
|
|
func (o *AppfileOptions) Run(filePath string) error {
|
|
res, data, err := o.export(filePath, false)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
deployFilePath := ".vela/deploy.yaml"
|
|
o.IO.Infof("Writing deploy config to (%s)\n", deployFilePath)
|
|
if err := os.MkdirAll(filepath.Dir(deployFilePath), 0700); err != nil {
|
|
return err
|
|
}
|
|
if err := ioutil.WriteFile(deployFilePath, data, 0600); err != nil {
|
|
return errors.Wrap(err, "write deploy config manifests failed")
|
|
}
|
|
|
|
if err := o.saveToAppDir(res.app); err != nil {
|
|
return errors.Wrap(err, "save to app dir failed")
|
|
}
|
|
|
|
o.IO.Infof("\nApplying deploy configs ...\n")
|
|
return o.ApplyAppConfig(res.appConfig, res.comps, res.scopes)
|
|
}
|
|
|
|
func (o *AppfileOptions) saveToAppDir(f *appfile.AppFile) error {
|
|
app := &application.Application{AppFile: f}
|
|
return app.Save(o.Env.Name)
|
|
}
|
|
|
|
// ApplyAppConfig applys config resources for the app.
|
|
// It differs by create and update:
|
|
// - for create, it displays app status along with information of url, metrics, ssh, logging.
|
|
// - for update, it rolls out a canary deployment and prints its information. User can verify the canary deployment.
|
|
// This will wait for user approval. If approved, it continues upgrading the whole; otherwise, it would rollback.
|
|
func (o *AppfileOptions) ApplyAppConfig(ac *v1alpha2.ApplicationConfiguration, comps []*v1alpha2.Component, scopes []oam.Object) error {
|
|
key := apitypes.NamespacedName{
|
|
Namespace: ac.Namespace,
|
|
Name: ac.Name,
|
|
}
|
|
o.IO.Infof("Checking if app has been deployed...\n")
|
|
var tmpAC v1alpha2.ApplicationConfiguration
|
|
err := o.Kubecli.Get(context.TODO(), key, &tmpAC)
|
|
switch {
|
|
case apierrors.IsNotFound(err):
|
|
o.IO.Infof("App has not been deployed, creating a new deployment...\n")
|
|
case err == nil:
|
|
o.IO.Infof("App exists, updating existing deployment...\n")
|
|
default:
|
|
return err
|
|
}
|
|
if err := o.apply(ac, comps, scopes); err != nil {
|
|
return err
|
|
}
|
|
o.IO.Infof(o.Info(ac.Name, comps))
|
|
return nil
|
|
}
|
|
|
|
func (o *AppfileOptions) apply(ac *v1alpha2.ApplicationConfiguration, comps []*v1alpha2.Component, scopes []oam.Object) error {
|
|
for _, comp := range comps {
|
|
if err := application.CreateOrUpdateComponent(context.TODO(), o.Kubecli, comp); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
if err := application.CreateScopes(context.TODO(), o.Kubecli, scopes); err != nil {
|
|
return err
|
|
}
|
|
return application.CreateOrUpdateAppConfig(context.TODO(), o.Kubecli, ac)
|
|
}
|
|
|
|
// Info shows the status of each service in the Appfile
|
|
func (o *AppfileOptions) Info(appName string, comps []*v1alpha2.Component) string {
|
|
var appUpMessage = "✅ App has been deployed 🚀🚀🚀\n" +
|
|
fmt.Sprintf(" Port forward: vela port-forward %s\n", appName) +
|
|
fmt.Sprintf(" SSH: vela exec %s\n", appName) +
|
|
fmt.Sprintf(" Logging: vela logs %s\n", appName) +
|
|
fmt.Sprintf(" App status: vela status %s\n", appName)
|
|
for _, comp := range comps {
|
|
appUpMessage += fmt.Sprintf(" Service status: vela status %s --svc %s\n", appName, comp.Name)
|
|
}
|
|
return appUpMessage
|
|
}
|