mirror of
https://github.com/kubevela/kubevela.git
synced 2026-05-11 03:47:59 +00:00
196 lines
5.7 KiB
Go
196 lines
5.7 KiB
Go
package appfile
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"reflect"
|
|
"sort"
|
|
|
|
"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/api"
|
|
"github.com/oam-dev/kubevela/pkg/appfile/template"
|
|
)
|
|
|
|
// NewEmptyApplication new empty application, only set tm
|
|
func NewEmptyApplication() (*api.Application, error) {
|
|
tm, err := template.Load()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return NewApplication(nil, tm), nil
|
|
}
|
|
|
|
// NewApplication will create application object
|
|
func NewApplication(f *api.AppFile, tm template.Manager) *api.Application {
|
|
if f == nil {
|
|
f = api.NewAppFile()
|
|
}
|
|
return &api.Application{AppFile: f, Tm: tm}
|
|
}
|
|
|
|
// Validate will validate whether an Appfile is valid.
|
|
func Validate(app *api.Application) error {
|
|
if app.Name == "" {
|
|
return errors.New("name is required")
|
|
}
|
|
if len(app.Services) == 0 {
|
|
return errors.New("at least one service is required")
|
|
}
|
|
for name, svc := range app.Services {
|
|
for traitName, traitData := range svc.GetApplicationConfig() {
|
|
if app.Tm.IsTrait(traitName) {
|
|
if _, ok := traitData.(map[string]interface{}); !ok {
|
|
return fmt.Errorf("trait %s in '%s' must be map", traitName, name)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// IsNotFound is application not found error
|
|
func IsNotFound(appName string, err error) bool {
|
|
return err != nil && err.Error() == fmt.Sprintf(`application "%s" not found`, appName)
|
|
}
|
|
|
|
// LoadApplication will load application with env and name from default vela home dir.
|
|
func LoadApplication(envName, appName string) (*api.Application, error) {
|
|
app, err := GetStorage().Get(envName, appName)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
err = Validate(app)
|
|
return app, err
|
|
}
|
|
|
|
// Delete will delete an app along with it's appfile.
|
|
func Delete(envName, appName string) error {
|
|
return GetStorage().Delete(envName, appName)
|
|
}
|
|
|
|
// List will list all apps
|
|
func List(envName string) ([]*api.Application, error) {
|
|
respApps, err := GetStorage().List(envName)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
var apps []*api.Application
|
|
for _, resp := range respApps {
|
|
app := NewApplication(resp.AppFile, resp.Tm)
|
|
err := Validate(app)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
apps = append(apps, app)
|
|
}
|
|
return apps, nil
|
|
}
|
|
|
|
// MatchAppByComp will get application with componentName without AppName.
|
|
func MatchAppByComp(envName, compName string) (*api.Application, error) {
|
|
apps, err := List(envName)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
for _, subapp := range apps {
|
|
for _, v := range GetComponents(subapp) {
|
|
if v == compName {
|
|
return subapp, nil
|
|
}
|
|
}
|
|
}
|
|
return nil, fmt.Errorf("no app found contains %s in env %s", compName, envName)
|
|
}
|
|
|
|
// Save will save appfile into default dir.
|
|
func Save(app *api.Application, envName string) error {
|
|
return GetStorage().Save(app, envName)
|
|
}
|
|
|
|
// GetComponents will get oam components from Appfile.
|
|
func GetComponents(app *api.Application) []string {
|
|
var components []string
|
|
for name := range app.Services {
|
|
components = append(components, name)
|
|
}
|
|
sort.Strings(components)
|
|
return components
|
|
}
|
|
|
|
// GetServiceConfig will get service type and it's configuration
|
|
func GetServiceConfig(app *api.Application, componentName string) (string, map[string]interface{}) {
|
|
svc, ok := app.Services[componentName]
|
|
if !ok {
|
|
return "", make(map[string]interface{})
|
|
}
|
|
return svc.GetType(), svc.GetApplicationConfig()
|
|
}
|
|
|
|
// GetWorkload will get workload type and it's configuration
|
|
func GetWorkload(app *api.Application, componentName string) (string, map[string]interface{}) {
|
|
svcType, config := GetServiceConfig(app, componentName)
|
|
if svcType == "" {
|
|
return "", make(map[string]interface{})
|
|
}
|
|
workloadData := make(map[string]interface{})
|
|
for k, v := range config {
|
|
if app.Tm.IsTrait(k) {
|
|
continue
|
|
}
|
|
workloadData[k] = v
|
|
}
|
|
return svcType, workloadData
|
|
}
|
|
|
|
// GetTraits will list all traits and it's configurations attached to the specified component.
|
|
func GetTraits(app *api.Application, componentName string) (map[string]map[string]interface{}, error) {
|
|
_, config := GetServiceConfig(app, componentName)
|
|
traitsData := make(map[string]map[string]interface{})
|
|
for k, v := range config {
|
|
if !app.Tm.IsTrait(k) {
|
|
continue
|
|
}
|
|
newV, ok := v.(map[string]interface{})
|
|
if !ok {
|
|
return nil, fmt.Errorf("%s is trait, but with invalid format %s, should be map[string]interface{}", k, reflect.TypeOf(v))
|
|
}
|
|
traitsData[k] = newV
|
|
}
|
|
return traitsData, nil
|
|
}
|
|
|
|
// GetTraitsByType will get trait configuration with specified component and trait type, we assume one type of trait can only attach to a component once.
|
|
func GetTraitsByType(app *api.Application, componentName, traitType string) (map[string]interface{}, error) {
|
|
service, ok := app.Services[componentName]
|
|
if !ok {
|
|
return nil, fmt.Errorf("service name (%s) doesn't exist", componentName)
|
|
}
|
|
t, ok := service[traitType]
|
|
if !ok {
|
|
return make(map[string]interface{}), nil
|
|
}
|
|
return t.(map[string]interface{}), nil
|
|
}
|
|
|
|
// GetAppConfig will get AppConfig from K8s cluster.
|
|
func GetAppConfig(ctx context.Context, c client.Client, app *api.Application, env *types.EnvMeta) (*v1alpha2.ApplicationConfiguration, error) {
|
|
appConfig := &v1alpha2.ApplicationConfiguration{}
|
|
if err := c.Get(ctx, client.ObjectKey{Namespace: env.Namespace, Name: app.Name}, appConfig); err != nil {
|
|
return nil, err
|
|
}
|
|
return appConfig, nil
|
|
}
|
|
|
|
// GetApplication will get Application from K8s cluster.
|
|
func GetApplication(ctx context.Context, c client.Client, app *api.Application, env *types.EnvMeta) (*v1alpha2.Application, error) {
|
|
appl := &v1alpha2.Application{}
|
|
if err := c.Get(ctx, client.ObjectKey{Namespace: env.Namespace, Name: app.Name}, appl); err != nil {
|
|
return nil, err
|
|
}
|
|
return appl, nil
|
|
}
|