| // Copyright 2019 CUE 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 cue |
| |
| import ( |
| "encoding" |
| "encoding/json" |
| "fmt" |
| "math/big" |
| "reflect" |
| "sort" |
| "strings" |
| "unicode/utf8" |
| |
| "github.com/cockroachdb/apd/v2" |
| |
| "cuelang.org/go/cue/ast" |
| "cuelang.org/go/cue/parser" |
| "cuelang.org/go/cue/token" |
| "cuelang.org/go/internal" |
| ) |
| |
| // This file contains functionality for converting Go to CUE. |
| // |
| // The code in this file is a prototype implementation and is far from |
| // optimized. |
| |
| func init() { |
| internal.FromGoValue = func(runtime, x interface{}, nilIsTop bool) interface{} { |
| return convertValue(runtime.(*Runtime), x, nilIsTop) |
| } |
| |
| internal.FromGoType = func(runtime, x interface{}) interface{} { |
| return convertType(runtime.(*Runtime), x) |
| } |
| } |
| |
| func convertValue(r *Runtime, x interface{}, nilIsTop bool) Value { |
| ctx := r.index().newContext() |
| v := convert(ctx, baseValue{}, nilIsTop, x) |
| return newValueRoot(ctx, v) |
| } |
| |
| func convertType(r *Runtime, x interface{}) Value { |
| ctx := r.index().newContext() |
| v := convertGoType(r, reflect.TypeOf(x)) |
| return newValueRoot(ctx, v) |
| |
| } |
| |
| // parseTag parses a CUE expression from a cue tag. |
| func parseTag(ctx *context, obj *structLit, field label, tag string) value { |
| if p := strings.Index(tag, ","); p >= 0 { |
| tag = tag[:p] |
| } |
| if tag == "" { |
| return &top{} |
| } |
| expr, err := parser.ParseExpr("<field:>", tag) |
| if err != nil { |
| field := ctx.labelStr(field) |
| return ctx.mkErr(baseValue{}, "invalid tag %q for field %q: %v", tag, field, err) |
| } |
| v := newVisitor(ctx.index, nil, nil, obj, true) |
| return v.walk(expr) |
| } |
| |
| // TODO: should we allow mapping names in cue tags? This only seems like a good |
| // idea if we ever want to allow mapping CUE to a different name than JSON. |
| var tagsWithNames = []string{"json", "yaml", "protobuf"} |
| |
| func getName(f *reflect.StructField) string { |
| name := f.Name |
| for _, s := range tagsWithNames { |
| if tag, ok := f.Tag.Lookup(s); ok { |
| if p := strings.Index(tag, ","); p >= 0 { |
| tag = tag[:p] |
| } |
| if tag != "" { |
| name = tag |
| break |
| } |
| } |
| } |
| return name |
| } |
| |
| // isOptional indicates whether a field should be marked as optional. |
| func isOptional(f *reflect.StructField) bool { |
| isOptional := false |
| switch f.Type.Kind() { |
| case reflect.Ptr, reflect.Map, reflect.Chan, reflect.Interface, reflect.Slice: |
| // Note: it may be confusing to distinguish between an empty slice and |
| // a nil slice. However, it is also surprizing to not be able to specify |
| // a default value for a slice. So for now we will allow it. |
| isOptional = true |
| } |
| if tag, ok := f.Tag.Lookup("cue"); ok { |
| // TODO: only if first field is not empty. |
| isOptional = false |
| for _, f := range strings.Split(tag, ",")[1:] { |
| switch f { |
| case "opt": |
| isOptional = true |
| case "req": |
| return false |
| } |
| } |
| } else if tag, ok = f.Tag.Lookup("json"); ok { |
| isOptional = false |
| for _, f := range strings.Split(tag, ",")[1:] { |
| if f == "omitempty" { |
| return true |
| } |
| } |
| } |
| return isOptional |
| } |
| |
| // isOmitEmpty means that the zero value is interpreted as undefined. |
| func isOmitEmpty(f *reflect.StructField) bool { |
| isOmitEmpty := false |
| switch f.Type.Kind() { |
| case reflect.Ptr, reflect.Map, reflect.Chan, reflect.Interface, reflect.Slice: |
| // Note: it may be confusing to distinguish between an empty slice and |
| // a nil slice. However, it is also surprizing to not be able to specify |
| // a default value for a slice. So for now we will allow it. |
| isOmitEmpty = true |
| |
| default: |
| // TODO: we can also infer omit empty if a type cannot be nil if there |
| // is a constraint that unconditionally disallows the zero value. |
| } |
| tag, ok := f.Tag.Lookup("json") |
| if ok { |
| isOmitEmpty = false |
| for _, f := range strings.Split(tag, ",")[1:] { |
| if f == "omitempty" { |
| return true |
| } |
| } |
| } |
| return isOmitEmpty |
| } |
| |
| // parseJSON parses JSON into a CUE value. b must be valid JSON. |
| func parseJSON(ctx *context, b []byte) evaluated { |
| expr, err := parser.ParseExpr("json", b) |
| if err != nil { |
| panic(err) // cannot happen |
| } |
| v := newVisitor(ctx.index, nil, nil, nil, false) |
| return v.walk(expr).evalPartial(ctx) |
| } |
| |
| func isZero(v reflect.Value) bool { |
| x := v.Interface() |
| if x == nil { |
| return true |
| } |
| switch k := v.Kind(); k { |
| case reflect.Struct, reflect.Array: |
| // we never allow optional values for these types. |
| return false |
| |
| case reflect.Chan, reflect.Func, reflect.Interface, reflect.Map, |
| reflect.Slice: |
| // Note that for maps we preserve the distinction between a nil map and |
| // an empty map. |
| return v.IsNil() |
| |
| case reflect.String: |
| return v.Len() == 0 |
| |
| default: |
| return x == reflect.Zero(v.Type()).Interface() |
| } |
| } |
| |
| func convert(ctx *context, src source, nilIsTop bool, x interface{}) evaluated { |
| v := convertRec(ctx, src, nilIsTop, x) |
| if v == nil { |
| return ctx.mkErr(baseValue{}, "unsupported Go type (%v)", v) |
| } |
| return v |
| } |
| |
| func isNil(x reflect.Value) bool { |
| switch x.Kind() { |
| // Only check for supported types; ignore func and chan. |
| case reflect.Ptr, reflect.Map, reflect.Slice, reflect.Interface: |
| return x.IsNil() |
| } |
| return false |
| } |
| |
| func convertRec(ctx *context, src source, nilIsTop bool, x interface{}) evaluated { |
| switch v := x.(type) { |
| case nil: |
| if nilIsTop { |
| return &top{src.base()} |
| } |
| return &nullLit{src.base()} |
| |
| case *ast.File: |
| x := newVisitorCtx(ctx, nil, nil, nil, false) |
| return ctx.manifest(x.walk(v)) |
| |
| case ast.Expr: |
| x := newVisitorCtx(ctx, nil, nil, nil, false) |
| return ctx.manifest(x.walk(v)) |
| |
| case *big.Int: |
| n := newInt(src.base(), 0) |
| n.v.Coeff.Set(v) |
| if v.Sign() < 0 { |
| n.v.Coeff.Neg(&n.v.Coeff) |
| n.v.Negative = true |
| } |
| return n |
| |
| case *big.Rat: |
| // should we represent this as a binary operation? |
| n := newNum(src, numKind, 0) |
| _, err := ctx.Quo(&n.v, apd.NewWithBigInt(v.Num(), 0), apd.NewWithBigInt(v.Denom(), 0)) |
| if err != nil { |
| return ctx.mkErr(src, err) |
| } |
| if !v.IsInt() { |
| n.k = floatKind |
| } |
| return n |
| |
| case *big.Float: |
| return newFloat(src, 0).setString(v.String()) |
| |
| case *apd.Decimal: |
| n := newNum(src, numKind, 0).set(v) |
| if !n.isInt(ctx) { |
| n.k = floatKind |
| } |
| return n |
| |
| case json.Marshaler: |
| b, err := v.MarshalJSON() |
| if err != nil { |
| return ctx.mkErr(src, err) |
| } |
| |
| return parseJSON(ctx, b) |
| |
| case encoding.TextMarshaler: |
| b, err := v.MarshalText() |
| if err != nil { |
| return ctx.mkErr(src, err) |
| } |
| b, err = json.Marshal(string(b)) |
| if err != nil { |
| return ctx.mkErr(src, err) |
| } |
| return parseJSON(ctx, b) |
| |
| case error: |
| return ctx.mkErr(src, v.Error()) |
| case bool: |
| return &boolLit{src.base(), v} |
| case string: |
| if !utf8.ValidString(v) { |
| return ctx.mkErr(src, |
| "cannot convert result to string: invalid UTF-8") |
| } |
| return &stringLit{src.base(), v, nil} |
| case []byte: |
| return &bytesLit{src.base(), v, nil} |
| case int: |
| return toInt(ctx, src, int64(v)) |
| case int8: |
| return toInt(ctx, src, int64(v)) |
| case int16: |
| return toInt(ctx, src, int64(v)) |
| case int32: |
| return toInt(ctx, src, int64(v)) |
| case int64: |
| return toInt(ctx, src, int64(v)) |
| case uint: |
| return toUint(ctx, src, uint64(v)) |
| case uint8: |
| return toUint(ctx, src, uint64(v)) |
| case uint16: |
| return toUint(ctx, src, uint64(v)) |
| case uint32: |
| return toUint(ctx, src, uint64(v)) |
| case uint64: |
| return toUint(ctx, src, uint64(v)) |
| case uintptr: |
| return toUint(ctx, src, uint64(v)) |
| case float64: |
| return newFloat(src, 0).setString(fmt.Sprintf("%g", v)) |
| case float32: |
| return newFloat(src, 0).setString(fmt.Sprintf("%g", v)) |
| |
| case reflect.Value: |
| if v.CanInterface() { |
| return convertRec(ctx, src, nilIsTop, v.Interface()) |
| } |
| |
| default: |
| value := reflect.ValueOf(v) |
| switch value.Kind() { |
| case reflect.Bool: |
| return &boolLit{src.base(), value.Bool()} |
| |
| case reflect.String: |
| str := value.String() |
| if !utf8.ValidString(str) { |
| return ctx.mkErr(src, |
| "cannot convert result to string: invalid UTF-8") |
| } |
| return &stringLit{src.base(), str, nil} |
| |
| case reflect.Int, reflect.Int8, reflect.Int16, |
| reflect.Int32, reflect.Int64: |
| return toInt(ctx, src, value.Int()) |
| |
| case reflect.Uint, reflect.Uint8, reflect.Uint16, |
| reflect.Uint32, reflect.Uint64, reflect.Uintptr: |
| return toUint(ctx, src, value.Uint()) |
| |
| case reflect.Float32, reflect.Float64: |
| return convertRec(ctx, src, nilIsTop, value.Float()) |
| |
| case reflect.Ptr: |
| if value.IsNil() { |
| if nilIsTop { |
| return &top{src.base()} |
| } |
| return &nullLit{src.base()} |
| } |
| return convertRec(ctx, src, nilIsTop, value.Elem().Interface()) |
| |
| case reflect.Struct: |
| obj := newStruct(src) |
| t := value.Type() |
| for i := 0; i < value.NumField(); i++ { |
| t := t.Field(i) |
| if t.PkgPath != "" { |
| continue |
| } |
| val := value.Field(i) |
| if !nilIsTop && isNil(val) { |
| continue |
| } |
| if tag, _ := t.Tag.Lookup("json"); tag == "-" { |
| continue |
| } |
| if isOmitEmpty(&t) && isZero(val) { |
| continue |
| } |
| sub := convertRec(ctx, src, nilIsTop, val.Interface()) |
| if sub == nil { |
| // mimic behavior of encoding/json: skip fields of unsupported types |
| continue |
| } |
| if isBottom(sub) { |
| return sub |
| } |
| |
| // leave errors like we do during normal evaluation or do we |
| // want to return the error? |
| name := getName(&t) |
| if name == "-" { |
| continue |
| } |
| f := ctx.strLabel(name) |
| obj.arcs = append(obj.arcs, arc{feature: f, v: sub}) |
| } |
| sort.Sort(obj) |
| return obj |
| |
| case reflect.Map: |
| obj := newStruct(src) |
| |
| sorted := []string{} |
| keys := []string{} |
| t := value.Type() |
| switch key := t.Key(); key.Kind() { |
| case reflect.String, |
| reflect.Int, reflect.Int8, reflect.Int16, |
| reflect.Int32, reflect.Int64, |
| reflect.Uint, reflect.Uint8, reflect.Uint16, |
| reflect.Uint32, reflect.Uint64, reflect.Uintptr: |
| for _, k := range value.MapKeys() { |
| val := value.MapIndex(k) |
| // if isNil(val) { |
| // continue |
| // } |
| |
| sub := convertRec(ctx, src, nilIsTop, val.Interface()) |
| // mimic behavior of encoding/json: report error of |
| // unsupported type. |
| if sub == nil { |
| return ctx.mkErr(baseValue{}, "unsupported Go type (%v)", val) |
| } |
| if isBottom(sub) { |
| return sub |
| } |
| |
| s := fmt.Sprint(k) |
| keys = append(keys, s) |
| sorted = append(sorted, s) |
| |
| // Set feature later. |
| obj.arcs = append(obj.arcs, arc{feature: 0, v: sub}) |
| } |
| |
| default: |
| return ctx.mkErr(baseValue{}, "unsupported Go type for map key (%v)", key) |
| } |
| |
| // Assign label in normalized order. |
| sort.Strings(sorted) |
| for _, k := range sorted { |
| ctx.strLabel(k) |
| } |
| |
| // Now assign the labels to the arcs. |
| for i, k := range keys { |
| obj.arcs[i].feature = ctx.strLabel(k) |
| } |
| sort.Sort(obj) |
| return obj |
| |
| case reflect.Slice, reflect.Array: |
| list := &list{baseValue: src.base()} |
| arcs := []arc{} |
| for i := 0; i < value.Len(); i++ { |
| val := value.Index(i) |
| x := convertRec(ctx, src, nilIsTop, val.Interface()) |
| if x == nil { |
| return ctx.mkErr(baseValue{}, "unsupported Go type (%v)", val) |
| } |
| if isBottom(x) { |
| return x |
| } |
| arcs = append(arcs, arc{feature: label(len(arcs)), v: x}) |
| } |
| list.elem = &structLit{baseValue: list.baseValue, arcs: arcs} |
| list.initLit() |
| // There is no need to set the type of the list, as the list will |
| // be of fixed size and all elements will already have a defined |
| // value. |
| return list |
| } |
| } |
| return nil |
| } |
| |
| func toInt(ctx *context, src source, x int64) evaluated { |
| return newInt(src, 0).setInt64(x) |
| } |
| |
| func toUint(ctx *context, src source, x uint64) evaluated { |
| return newInt(src, 0).setUInt64(x) |
| } |
| |
| func convertGoType(r *Runtime, t reflect.Type) value { |
| ctx := r.index().newContext() |
| // TODO: this can be much more efficient. |
| ctx.mutex.Lock() |
| defer ctx.mutex.Unlock() |
| return goTypeToValue(ctx, true, t) |
| } |
| |
| var ( |
| jsonMarshaler = reflect.TypeOf(new(json.Marshaler)).Elem() |
| textMarshaler = reflect.TypeOf(new(encoding.TextMarshaler)).Elem() |
| topSentinel = &top{} |
| ) |
| |
| // goTypeToValue converts a Go Type to a value. |
| // |
| // TODO: if this value will always be unified with a concrete type in Go, then |
| // many of the fields may be omitted. |
| func goTypeToValue(ctx *context, allowNullDefault bool, t reflect.Type) value { |
| v := goTypeToValueRec(ctx, allowNullDefault, t) |
| if v == nil { |
| return ctx.mkErr(baseValue{}, "unsupported Go type (%v)", t) |
| } |
| return v |
| } |
| |
| func goTypeToValueRec(ctx *context, allowNullDefault bool, t reflect.Type) (e value) { |
| if e, ok := ctx.typeCache.Load(t); ok { |
| return e.(value) |
| } |
| |
| switch reflect.Zero(t).Interface().(type) { |
| case *big.Int, big.Int: |
| e = &basicType{k: intKind} |
| goto store |
| |
| case *big.Float, big.Float, *big.Rat, big.Rat: |
| e = &basicType{k: numKind} |
| goto store |
| |
| case *apd.Decimal, apd.Decimal: |
| e = &basicType{k: numKind} |
| goto store |
| } |
| |
| // Even if this is for types that we know cast to a certain type, it can't |
| // hurt to return top, as in these cases the concrete values will be |
| // strict instances and there cannot be any tags that further constrain |
| // the values. |
| if t.Implements(jsonMarshaler) || t.Implements(textMarshaler) { |
| return topSentinel |
| } |
| |
| switch k := t.Kind(); k { |
| case reflect.Ptr: |
| elem := t.Elem() |
| for elem.Kind() == reflect.Ptr { |
| elem = elem.Elem() |
| } |
| e = goTypeToValueRec(ctx, false, elem) |
| if allowNullDefault { |
| e = wrapOrNull(e) |
| } |
| |
| case reflect.Interface: |
| switch t.Name() { |
| case "error": |
| // This is really null | _|_. There is no error if the error is null. |
| e = &nullLit{} // null |
| default: |
| e = topSentinel // `_` |
| } |
| |
| case reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64, |
| reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64: |
| e = predefinedRanges[t.Kind().String()] |
| |
| case reflect.Uint, reflect.Uintptr: |
| e = predefinedRanges["uint64"] |
| |
| case reflect.Int: |
| e = predefinedRanges["int64"] |
| |
| case reflect.String: |
| e = &basicType{k: stringKind} |
| |
| case reflect.Bool: |
| e = &basicType{k: boolKind} |
| |
| case reflect.Float32, reflect.Float64: |
| e = &basicType{k: floatKind} |
| |
| case reflect.Struct: |
| // First iterate to create struct, then iterate another time to |
| // resolve field tags to allow field tags to refer to the struct fields. |
| tags := map[label]string{} |
| obj := newStruct(baseValue{}) |
| ctx.typeCache.Store(t, obj) |
| |
| for i := 0; i < t.NumField(); i++ { |
| f := t.Field(i) |
| if f.PkgPath != "" { |
| continue |
| } |
| _, ok := f.Tag.Lookup("cue") |
| elem := goTypeToValueRec(ctx, !ok, f.Type) |
| if elem == nil || isBottom(elem) { |
| continue // Ignore fields for unsupported types |
| } |
| |
| // leave errors like we do during normal evaluation or do we |
| // want to return the error? |
| name := getName(&f) |
| if name == "-" { |
| continue |
| } |
| l := ctx.strLabel(name) |
| obj.arcs = append(obj.arcs, arc{ |
| feature: l, |
| // The GO JSON decoder always allows a value to be undefined. |
| optional: isOptional(&f), |
| v: elem, |
| }) |
| |
| if tag, ok := f.Tag.Lookup("cue"); ok { |
| tags[l] = tag |
| } |
| } |
| sort.Sort(obj) |
| |
| for label, tag := range tags { |
| v := parseTag(ctx, obj, label, tag) |
| if isBottom(v) { |
| return v |
| } |
| for i, a := range obj.arcs { |
| if a.feature == label { |
| // Instead of unifying with the existing type, we substitute |
| // with the constraints from the tags. The type constraints |
| // will be implied when unified with a concrete value. |
| obj.arcs[i].v = mkBin(ctx, token.NoPos, opUnify, a.v, v) |
| } |
| } |
| } |
| |
| return obj |
| |
| case reflect.Array, reflect.Slice: |
| if t.Elem().Kind() == reflect.Uint8 { |
| e = &basicType{k: bytesKind} |
| } else { |
| elem := goTypeToValueRec(ctx, allowNullDefault, t.Elem()) |
| if elem == nil { |
| return ctx.mkErr(baseValue{}, "unsupported Go type (%v)", t.Elem()) |
| } |
| |
| var ln value = &top{} |
| if t.Kind() == reflect.Array { |
| ln = toInt(ctx, baseValue{}, int64(t.Len())) |
| } |
| e = &list{elem: &structLit{}, typ: elem, len: ln} |
| } |
| if k == reflect.Slice { |
| e = wrapOrNull(e) |
| } |
| |
| case reflect.Map: |
| switch key := t.Key(); key.Kind() { |
| case reflect.String, reflect.Int, reflect.Int8, reflect.Int16, |
| reflect.Int32, reflect.Int64, reflect.Uint, reflect.Uint8, |
| reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr: |
| default: |
| return ctx.mkErr(baseValue{}, "unsupported Go type for map key (%v)", key) |
| } |
| |
| obj := newStruct(baseValue{}) |
| sig := ¶ms{} |
| sig.add(ctx.label("_", true), &basicType{k: stringKind}) |
| v := goTypeToValueRec(ctx, allowNullDefault, t.Elem()) |
| if v == nil { |
| return ctx.mkErr(baseValue{}, "unsupported Go type (%v)", t.Elem()) |
| } |
| if isBottom(v) { |
| return v |
| } |
| obj.optionals = newOptional(nil, &lambdaExpr{params: sig, value: v}) |
| |
| e = wrapOrNull(obj) |
| } |
| |
| store: |
| // TODO: store error if not nil? |
| if e != nil { |
| ctx.typeCache.Store(t, e) |
| } |
| return e |
| } |
| |
| func wrapOrNull(e value) value { |
| if e == nil || isBottom(e) || e.kind().isAnyOf(nullKind) { |
| return e |
| } |
| return makeNullable(e, true) |
| } |
| |
| func makeNullable(e value, nullIsDefault bool) value { |
| return &disjunction{ |
| baseValue: baseValue{e}, |
| values: []dValue{ |
| {val: &nullLit{}, marked: nullIsDefault}, |
| {val: e}}, |
| errors: nil, |
| hasDefaults: nullIsDefault, |
| } |
| } |