2020-03-26 21:07:15 +00:00
|
|
|
package schemas
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"reflect"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/rancher/wrangler/pkg/data/convert"
|
|
|
|
"github.com/rancher/wrangler/pkg/slice"
|
|
|
|
"github.com/sirupsen/logrus"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
blacklistNames = map[string]bool{
|
|
|
|
"links": true,
|
|
|
|
"actions": true,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
func (s *Schemas) TypeName(name string, obj interface{}) *Schemas {
|
|
|
|
s.typeNames[reflect.TypeOf(obj)] = name
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Schemas) getTypeName(t reflect.Type) string {
|
|
|
|
if name, ok := s.typeNames[t]; ok {
|
|
|
|
return name
|
|
|
|
}
|
|
|
|
return convert.LowerTitle(t.Name())
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Schemas) SchemaFor(t reflect.Type) *Schema {
|
|
|
|
name := s.getTypeName(t)
|
|
|
|
return s.Schema(name)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Schemas) AddMapperForType(obj interface{}, mapper ...Mapper) *Schemas {
|
|
|
|
if len(mapper) == 0 {
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
|
|
|
t := reflect.TypeOf(obj)
|
|
|
|
typeName := s.getTypeName(t)
|
|
|
|
if len(mapper) == 1 {
|
|
|
|
return s.AddMapper(typeName, mapper[0])
|
|
|
|
}
|
|
|
|
return s.AddMapper(typeName, Mappers(mapper))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Schemas) MustImport(obj interface{}, externalOverrides ...interface{}) *Schemas {
|
|
|
|
if reflect.ValueOf(obj).Kind() == reflect.Ptr {
|
|
|
|
panic(fmt.Errorf("obj cannot be a pointer"))
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, err := s.Import(obj, externalOverrides...); err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Schemas) MustImportAndCustomize(obj interface{}, f func(*Schema), externalOverrides ...interface{}) *Schemas {
|
|
|
|
return s.MustImport(obj, externalOverrides...).
|
|
|
|
MustCustomizeType(obj, f)
|
|
|
|
}
|
|
|
|
|
2020-04-22 22:34:19 +00:00
|
|
|
func getType(obj interface{}) reflect.Type {
|
|
|
|
if t, ok := obj.(reflect.Type); ok {
|
|
|
|
return t
|
2020-03-26 21:07:15 +00:00
|
|
|
}
|
|
|
|
|
2020-04-22 22:34:19 +00:00
|
|
|
t := reflect.TypeOf(obj)
|
|
|
|
if t.Kind() == reflect.Ptr {
|
|
|
|
t = t.Elem()
|
|
|
|
}
|
|
|
|
return t
|
|
|
|
}
|
2020-03-26 21:07:15 +00:00
|
|
|
|
2020-04-22 22:34:19 +00:00
|
|
|
func (s *Schemas) Import(obj interface{}, externalOverrides ...interface{}) (*Schema, error) {
|
|
|
|
var types []reflect.Type
|
|
|
|
for _, override := range externalOverrides {
|
|
|
|
types = append(types, getType(override))
|
2020-03-26 21:07:15 +00:00
|
|
|
}
|
|
|
|
|
2020-04-22 22:34:19 +00:00
|
|
|
t := getType(obj)
|
2020-03-26 21:07:15 +00:00
|
|
|
return s.importType(t, types...)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Schemas) newSchemaFromType(t reflect.Type, typeName string) (*Schema, error) {
|
|
|
|
schema := &Schema{
|
|
|
|
ID: typeName,
|
|
|
|
CodeName: t.Name(),
|
|
|
|
PkgName: t.PkgPath(),
|
|
|
|
ResourceFields: map[string]Field{},
|
|
|
|
ResourceActions: map[string]Action{},
|
|
|
|
CollectionActions: map[string]Action{},
|
|
|
|
Attributes: map[string]interface{}{},
|
|
|
|
}
|
|
|
|
|
|
|
|
s.processingTypes[t] = schema
|
|
|
|
defer delete(s.processingTypes, t)
|
|
|
|
|
|
|
|
if err := s.readFields(schema, t); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return schema, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Schemas) MustCustomizeType(obj interface{}, f func(*Schema)) *Schemas {
|
|
|
|
name := s.getTypeName(reflect.TypeOf(obj))
|
|
|
|
schema := s.Schema(name)
|
|
|
|
if schema == nil {
|
|
|
|
panic("Failed to find schema " + name)
|
|
|
|
}
|
|
|
|
|
|
|
|
f(schema)
|
|
|
|
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Schemas) assignMappers(schema *Schema) error {
|
|
|
|
if schema.Mapper != nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
mappers := s.mapper(schema.ID)
|
|
|
|
if canList(schema) {
|
|
|
|
if s.DefaultMapper != nil {
|
|
|
|
mappers = append([]Mapper{s.DefaultMapper()}, mappers...)
|
|
|
|
}
|
|
|
|
if s.DefaultPostMapper != nil {
|
|
|
|
mappers = append(mappers, s.DefaultPostMapper())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(mappers) > 0 {
|
|
|
|
schema.InternalSchema = schema.DeepCopy()
|
|
|
|
}
|
|
|
|
|
|
|
|
mapper := &typeMapper{
|
|
|
|
Mappers: mappers,
|
|
|
|
root: canList(schema),
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := mapper.ModifySchema(schema, s); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
schema.Mapper = mapper
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func canList(schema *Schema) bool {
|
|
|
|
return slice.ContainsString(schema.CollectionMethods, "GET")
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Schemas) importType(t reflect.Type, overrides ...reflect.Type) (*Schema, error) {
|
|
|
|
typeName := s.getTypeName(t)
|
|
|
|
|
|
|
|
existing := s.Schema(typeName)
|
|
|
|
if existing != nil {
|
|
|
|
return existing, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if s, ok := s.processingTypes[t]; ok {
|
|
|
|
logrus.Debugf("Returning half built schema %s for %v", typeName, t)
|
|
|
|
return s, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
logrus.Debugf("Inspecting schema %s for %v", typeName, t)
|
|
|
|
|
|
|
|
schema, err := s.newSchemaFromType(t, typeName)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, override := range overrides {
|
|
|
|
if err := s.readFields(schema, override); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := s.assignMappers(schema); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = s.AddSchema(*schema)
|
|
|
|
return s.Schema(schema.ID), err
|
|
|
|
}
|
|
|
|
|
|
|
|
func jsonName(f reflect.StructField) string {
|
|
|
|
return strings.SplitN(f.Tag.Get("json"), ",", 2)[0]
|
|
|
|
}
|
|
|
|
|
|
|
|
func k8sType(field reflect.StructField) bool {
|
|
|
|
return field.Type.Name() == "TypeMeta" &&
|
|
|
|
strings.HasSuffix(field.Type.PkgPath(), "k8s.io/apimachinery/pkg/apis/meta/v1")
|
|
|
|
}
|
|
|
|
|
|
|
|
func k8sObject(field reflect.StructField) bool {
|
|
|
|
return field.Type.Name() == "ObjectMeta" &&
|
|
|
|
strings.HasSuffix(field.Type.PkgPath(), "k8s.io/apimachinery/pkg/apis/meta/v1")
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Schemas) readFields(schema *Schema, t reflect.Type) error {
|
|
|
|
hasType := false
|
|
|
|
hasMeta := false
|
|
|
|
|
|
|
|
for i := 0; i < t.NumField(); i++ {
|
|
|
|
field := t.Field(i)
|
|
|
|
|
|
|
|
if field.PkgPath != "" {
|
|
|
|
// unexported field
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
jsonName := jsonName(field)
|
|
|
|
if jsonName == "-" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if field.Anonymous && jsonName == "" && k8sType(field) {
|
|
|
|
hasType = true
|
|
|
|
}
|
|
|
|
|
|
|
|
if field.Anonymous && jsonName == "metadata" && k8sObject(field) {
|
|
|
|
hasMeta = true
|
|
|
|
}
|
|
|
|
|
|
|
|
if field.Anonymous && jsonName == "" {
|
|
|
|
t := field.Type
|
|
|
|
if t.Kind() == reflect.Ptr {
|
|
|
|
t = t.Elem()
|
|
|
|
}
|
|
|
|
if t.Kind() == reflect.Struct {
|
|
|
|
if err := s.readFields(schema, t); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
fieldName := jsonName
|
|
|
|
if fieldName == "" {
|
|
|
|
fieldName = convert.LowerTitle(field.Name)
|
|
|
|
if strings.HasSuffix(fieldName, "ID") {
|
|
|
|
fieldName = strings.TrimSuffix(fieldName, "ID") + "Id"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if blacklistNames[fieldName] {
|
|
|
|
logrus.Debugf("Ignoring blacklisted field %s.%s for %v", schema.ID, fieldName, field)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
logrus.Debugf("Inspecting field %s.%s for %v", schema.ID, fieldName, field)
|
|
|
|
|
|
|
|
schemaField := Field{
|
|
|
|
CodeName: field.Name,
|
|
|
|
Create: true,
|
|
|
|
Update: true,
|
|
|
|
}
|
|
|
|
|
|
|
|
fieldType := field.Type
|
|
|
|
if fieldType.Kind() == reflect.Ptr {
|
|
|
|
schemaField.Nullable = true
|
|
|
|
fieldType = fieldType.Elem()
|
|
|
|
} else if fieldType.Kind() == reflect.Bool {
|
|
|
|
schemaField.Nullable = false
|
|
|
|
schemaField.Default = false
|
|
|
|
} else if fieldType.Kind() == reflect.Int ||
|
|
|
|
fieldType.Kind() == reflect.Uint32 ||
|
|
|
|
fieldType.Kind() == reflect.Int32 ||
|
|
|
|
fieldType.Kind() == reflect.Uint64 ||
|
|
|
|
fieldType.Kind() == reflect.Int64 ||
|
|
|
|
fieldType.Kind() == reflect.Float32 ||
|
|
|
|
fieldType.Kind() == reflect.Float64 {
|
|
|
|
schemaField.Nullable = false
|
|
|
|
schemaField.Default = 0
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := applyTag(&field, &schemaField); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if schemaField.Type == "" {
|
|
|
|
inferredType, err := s.determineSchemaType(fieldType)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("failed inspecting type %s, field %s: %v", t, fieldName, err)
|
|
|
|
}
|
|
|
|
schemaField.Type = inferredType
|
|
|
|
}
|
|
|
|
|
|
|
|
if schemaField.Default != nil {
|
|
|
|
switch schemaField.Type {
|
|
|
|
case "int":
|
|
|
|
n, err := convert.ToNumber(schemaField.Default)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
schemaField.Default = n
|
|
|
|
case "float":
|
|
|
|
n, err := convert.ToFloat(schemaField.Default)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
schemaField.Default = n
|
|
|
|
case "boolean":
|
|
|
|
schemaField.Default = convert.ToBool(schemaField.Default)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if s.fieldMappers != nil {
|
|
|
|
if err := s.processFieldsMappers(t, fieldName, schema, field); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
logrus.Debugf("Setting field %s.%s: %#v", schema.ID, fieldName, schemaField)
|
|
|
|
schema.ResourceFields[fieldName] = schemaField
|
|
|
|
}
|
|
|
|
|
|
|
|
if hasType && hasMeta {
|
2020-04-22 22:34:19 +00:00
|
|
|
delete(schema.ResourceFields, "kind")
|
|
|
|
delete(schema.ResourceFields, "apiVersion")
|
|
|
|
delete(schema.ResourceFields, "metadata")
|
2020-03-26 21:07:15 +00:00
|
|
|
schema.CollectionMethods = []string{"GET", "POST"}
|
|
|
|
schema.ResourceMethods = []string{"GET", "PUT", "DELETE"}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Schemas) processFieldsMappers(t reflect.Type, fieldName string, schema *Schema, field reflect.StructField) error {
|
|
|
|
for _, fieldMapper := range strings.Split(field.Tag.Get("mapper"), ",") {
|
|
|
|
if fieldMapper == "" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
var (
|
|
|
|
name string
|
|
|
|
opts []string
|
|
|
|
)
|
|
|
|
parts := strings.SplitN(fieldMapper, "=", 2)
|
|
|
|
name = parts[0]
|
|
|
|
if len(parts) == 2 {
|
|
|
|
for _, opt := range strings.Split(parts[1], "|") {
|
|
|
|
opts = append(opts, opt)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
factory, ok := s.fieldMappers[name]
|
|
|
|
if !ok {
|
|
|
|
return fmt.Errorf("failed to find field mapper [%s] for type [%v]", name, t)
|
|
|
|
}
|
|
|
|
|
|
|
|
s.AddMapper(schema.ID, factory(fieldName, opts...))
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func applyTag(structField *reflect.StructField, field *Field) error {
|
2020-04-22 22:34:19 +00:00
|
|
|
t, ok := structField.Tag.Lookup("wrangler")
|
|
|
|
if !ok {
|
|
|
|
t = structField.Tag.Get("norman")
|
|
|
|
}
|
|
|
|
for _, part := range strings.Split(t, ",") {
|
2020-03-26 21:07:15 +00:00
|
|
|
if part == "" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
var err error
|
|
|
|
key, value := getKeyValue(part)
|
|
|
|
|
|
|
|
switch key {
|
|
|
|
case "type":
|
|
|
|
field.Type = value
|
|
|
|
case "codeName":
|
|
|
|
field.CodeName = value
|
|
|
|
case "default":
|
|
|
|
field.Default = value
|
|
|
|
case "nullable":
|
|
|
|
field.Nullable = true
|
|
|
|
case "notnullable":
|
|
|
|
field.Nullable = false
|
|
|
|
case "create":
|
|
|
|
field.Create = true
|
|
|
|
case "nocreate":
|
|
|
|
field.Create = false
|
|
|
|
case "writeOnly":
|
|
|
|
field.WriteOnly = true
|
|
|
|
case "required":
|
|
|
|
field.Required = true
|
|
|
|
case "update":
|
|
|
|
field.Update = true
|
|
|
|
case "noupdate":
|
|
|
|
field.Update = false
|
|
|
|
case "minLength":
|
|
|
|
field.MinLength, err = toInt(value, structField)
|
|
|
|
case "maxLength":
|
|
|
|
field.MaxLength, err = toInt(value, structField)
|
|
|
|
case "min":
|
|
|
|
field.Min, err = toInt(value, structField)
|
|
|
|
case "max":
|
|
|
|
field.Max, err = toInt(value, structField)
|
|
|
|
case "options":
|
|
|
|
field.Options = split(value)
|
|
|
|
if field.Type == "" {
|
|
|
|
field.Type = "enum"
|
|
|
|
}
|
|
|
|
case "validChars":
|
|
|
|
field.ValidChars = value
|
|
|
|
case "invalidChars":
|
|
|
|
field.InvalidChars = value
|
|
|
|
default:
|
|
|
|
return fmt.Errorf("invalid tag %s on field %s", key, structField.Name)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func toInt(value string, structField *reflect.StructField) (*int64, error) {
|
|
|
|
i, err := strconv.ParseInt(value, 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("invalid number on field %s: %v", structField.Name, err)
|
|
|
|
}
|
|
|
|
return &i, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func split(input string) []string {
|
|
|
|
var result []string
|
|
|
|
for _, i := range strings.Split(input, "|") {
|
|
|
|
for _, part := range strings.Split(i, " ") {
|
|
|
|
part = strings.TrimSpace(part)
|
|
|
|
if len(part) > 0 {
|
|
|
|
result = append(result, part)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
func getKeyValue(input string) (string, string) {
|
|
|
|
var (
|
|
|
|
key, value string
|
|
|
|
)
|
|
|
|
parts := strings.SplitN(input, "=", 2)
|
|
|
|
key = parts[0]
|
|
|
|
if len(parts) > 1 {
|
|
|
|
value = parts[1]
|
|
|
|
}
|
|
|
|
|
|
|
|
return key, value
|
|
|
|
}
|
|
|
|
|
|
|
|
func deRef(p reflect.Type) reflect.Type {
|
|
|
|
if p.Kind() == reflect.Ptr {
|
|
|
|
return p.Elem()
|
|
|
|
}
|
|
|
|
return p
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Schemas) determineSchemaType(t reflect.Type) (string, error) {
|
|
|
|
switch t.Kind() {
|
|
|
|
case reflect.Uint8:
|
|
|
|
return "byte", nil
|
|
|
|
case reflect.Bool:
|
|
|
|
return "boolean", nil
|
|
|
|
case reflect.Int:
|
|
|
|
fallthrough
|
|
|
|
case reflect.Int32:
|
|
|
|
fallthrough
|
|
|
|
case reflect.Uint32:
|
|
|
|
fallthrough
|
|
|
|
case reflect.Uint64:
|
|
|
|
fallthrough
|
|
|
|
case reflect.Int64:
|
|
|
|
return "int", nil
|
|
|
|
case reflect.Float32:
|
|
|
|
fallthrough
|
|
|
|
case reflect.Float64:
|
|
|
|
return "float", nil
|
|
|
|
case reflect.Interface:
|
|
|
|
return "json", nil
|
|
|
|
case reflect.Map:
|
|
|
|
subType, err := s.determineSchemaType(deRef(t.Elem()))
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
return fmt.Sprintf("map[%s]", subType), nil
|
|
|
|
case reflect.Slice:
|
|
|
|
subType, err := s.determineSchemaType(deRef(t.Elem()))
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
if subType == "byte" {
|
|
|
|
return "base64", nil
|
|
|
|
}
|
|
|
|
return fmt.Sprintf("array[%s]", subType), nil
|
|
|
|
case reflect.String:
|
|
|
|
return "string", nil
|
|
|
|
case reflect.Struct:
|
|
|
|
if t.Name() == "Time" {
|
|
|
|
return "date", nil
|
|
|
|
}
|
|
|
|
if t.Name() == "IntOrString" {
|
|
|
|
return "intOrString", nil
|
|
|
|
}
|
|
|
|
if t.Name() == "Quantity" {
|
|
|
|
return "string", nil
|
|
|
|
}
|
|
|
|
schema, err := s.importType(t)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
if t.Name() == "Duration" && strings.Contains(schema.PkgName, "k8s.io/apimachinery/pkg/apis/meta/v1") {
|
|
|
|
return "string", nil
|
|
|
|
}
|
|
|
|
return schema.ID, nil
|
|
|
|
default:
|
|
|
|
return "", fmt.Errorf("unknown type kind %s", t.Kind())
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|