/* 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 }