mirror of
https://github.com/kubevela/kubevela.git
synced 2026-03-05 19:22:03 +00:00
* Feat: add support for json-patch and json-merge-patch Signed-off-by: Somefive <yd219913@alibaba-inc.com> * Fix: add e2e test Signed-off-by: Somefive <yd219913@alibaba-inc.com> * Fix: refactor json-patch field Signed-off-by: Somefive <yd219913@alibaba-inc.com>
158 lines
3.8 KiB
Go
158 lines
3.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 model
|
|
|
|
import (
|
|
"regexp"
|
|
"strings"
|
|
|
|
"cuelang.org/go/cue"
|
|
"cuelang.org/go/cue/build"
|
|
"cuelang.org/go/cue/format"
|
|
"github.com/pkg/errors"
|
|
"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
|
|
"k8s.io/klog/v2"
|
|
|
|
"github.com/oam-dev/kubevela/pkg/cue/model/sets"
|
|
)
|
|
|
|
// Instance defines Model Interface
|
|
type Instance interface {
|
|
String() string
|
|
Unstructured() (*unstructured.Unstructured, error)
|
|
IsBase() bool
|
|
Unify(other Instance, options ...sets.UnifyOption) error
|
|
Compile() ([]byte, error)
|
|
}
|
|
|
|
type instance struct {
|
|
v string
|
|
base bool
|
|
}
|
|
|
|
// String return instance's cue format string
|
|
func (inst *instance) String() string {
|
|
return inst.v
|
|
}
|
|
|
|
// IsBase indicate whether the instance is base model
|
|
func (inst *instance) IsBase() bool {
|
|
return inst.base
|
|
}
|
|
|
|
func (inst *instance) Compile() ([]byte, error) {
|
|
bi := build.NewContext().NewInstance("", nil)
|
|
err := bi.AddFile("-", inst.v)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
var r cue.Runtime
|
|
it, err := r.Build(bi)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
// compiled object should be final and concrete value
|
|
if err := it.Value().Validate(cue.Concrete(true), cue.Final()); err != nil {
|
|
return nil, err
|
|
}
|
|
return it.Value().MarshalJSON()
|
|
}
|
|
|
|
// Unstructured convert cue values to unstructured.Unstructured
|
|
// TODO(wonderflow): will it be better if we try to decode it to concrete object(such as K8s Deployment) by using runtime.Schema?
|
|
func (inst *instance) Unstructured() (*unstructured.Unstructured, error) {
|
|
jsonv, err := inst.Compile()
|
|
if err != nil {
|
|
klog.ErrorS(err, "failed to have the workload/trait unstructured", "Definition", inst.String())
|
|
return nil, errors.Wrap(err, "failed to have the workload/trait unstructured")
|
|
}
|
|
o := &unstructured.Unstructured{}
|
|
if err := o.UnmarshalJSON(jsonv); err != nil {
|
|
return nil, err
|
|
}
|
|
return o, nil
|
|
}
|
|
|
|
// Unify implement unity operations between instances
|
|
func (inst *instance) Unify(other Instance, options ...sets.UnifyOption) error {
|
|
pv, err := sets.StrategyUnify(inst.v, other.String(), options...)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
inst.v = pv
|
|
return nil
|
|
}
|
|
|
|
// NewBase create a base instance
|
|
func NewBase(v cue.Value) (Instance, error) {
|
|
vs, err := openPrint(v)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &instance{
|
|
v: vs,
|
|
base: true,
|
|
}, nil
|
|
}
|
|
|
|
// NewOther create a non-base instance
|
|
func NewOther(v cue.Value) (Instance, error) {
|
|
vs, err := openPrint(v)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &instance{
|
|
v: vs,
|
|
}, nil
|
|
}
|
|
|
|
func openPrint(v cue.Value) (string, error) {
|
|
sysopts := []cue.Option{cue.All(), cue.DisallowCycles(true), cue.ResolveReferences(true), cue.Docs(true)}
|
|
f, err := sets.ToFile(v.Syntax(sysopts...))
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
for _, decl := range f.Decls {
|
|
sets.ListOpen(decl)
|
|
}
|
|
|
|
ret, err := format.Node(f)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
errInfo, contain := IndexMatchLine(string(ret), "_|_")
|
|
if contain {
|
|
return "", errors.New(errInfo)
|
|
}
|
|
return string(ret), nil
|
|
}
|
|
|
|
// IndexMatchLine will index and extract the line contains the pattern.
|
|
func IndexMatchLine(ret, target string) (string, bool) {
|
|
if strings.Contains(ret, target) {
|
|
if target == "_|_" {
|
|
r := regexp.MustCompile(`_\|_[\s]//.*`)
|
|
match := r.FindAllString(ret, -1)
|
|
if len(match) > 0 {
|
|
return strings.Join(match, ","), true
|
|
}
|
|
}
|
|
}
|
|
return "", false
|
|
}
|