1
0
mirror of https://github.com/rancher/norman.git synced 2025-05-30 18:55:06 +00:00
norman/types/schemas.go

408 lines
8.3 KiB
Go
Raw Normal View History

2017-11-11 04:44:02 +00:00
package types
import (
"bytes"
"fmt"
2017-12-23 06:16:28 +00:00
"reflect"
2017-11-11 04:44:02 +00:00
"strings"
2017-12-16 08:25:28 +00:00
"sync"
2017-11-11 04:44:02 +00:00
"github.com/rancher/norman/types/convert"
2017-12-13 15:53:28 +00:00
"github.com/rancher/norman/types/definition"
2024-06-04 10:02:09 +00:00
"github.com/rancher/wrangler/v3/pkg/name"
2017-11-11 04:44:02 +00:00
)
type SchemaCollection struct {
Data []Schema
}
2017-12-16 08:25:28 +00:00
type SchemasInitFunc func(*Schemas) *Schemas
type SchemaHook func(*Schema)
2017-11-15 04:33:57 +00:00
2017-11-29 21:27:02 +00:00
type MappersFactory func() []Mapper
2017-12-13 15:53:28 +00:00
type BackReference struct {
FieldName string
Schema *Schema
}
2017-11-11 04:44:02 +00:00
type Schemas struct {
2017-12-16 08:25:28 +00:00
sync.Mutex
2018-12-17 22:40:55 +00:00
processingTypes map[reflect.Type]*Schema
2018-02-09 20:31:12 +00:00
typeNames map[reflect.Type]string
schemasByPath map[string]map[string]*Schema
mappers map[string]map[string][]Mapper
references map[string][]BackReference
embedded map[string]*Schema
DefaultMappers MappersFactory
DefaultPostMappers MappersFactory
versions []APIVersion
schemas []*Schema
AddHook SchemaHook
errors []error
2017-11-11 04:44:02 +00:00
}
func NewSchemas() *Schemas {
return &Schemas{
2018-12-17 22:40:55 +00:00
processingTypes: map[reflect.Type]*Schema{},
typeNames: map[reflect.Type]string{},
schemasByPath: map[string]map[string]*Schema{},
mappers: map[string]map[string][]Mapper{},
references: map[string][]BackReference{},
embedded: map[string]*Schema{},
2017-11-11 04:44:02 +00:00
}
}
2017-12-16 08:25:28 +00:00
func (s *Schemas) Init(initFunc SchemasInitFunc) *Schemas {
2017-11-15 04:33:57 +00:00
return initFunc(s)
}
2017-11-11 04:44:02 +00:00
func (s *Schemas) Err() error {
2018-03-04 04:45:18 +00:00
return NewErrors(s.errors...)
2017-11-11 04:44:02 +00:00
}
2017-11-12 00:07:09 +00:00
func (s *Schemas) AddSchemas(schema *Schemas) *Schemas {
for _, schema := range schema.Schemas() {
2017-12-12 03:58:43 +00:00
s.AddSchema(*schema)
2017-11-12 00:07:09 +00:00
}
return s
}
2017-12-16 08:25:28 +00:00
func (s *Schemas) RemoveSchema(schema Schema) *Schemas {
s.Lock()
defer s.Unlock()
return s.doRemoveSchema(schema)
}
func (s *Schemas) doRemoveSchema(schema Schema) *Schemas {
delete(s.schemasByPath[schema.Version.Path], schema.ID)
s.removeReferences(&schema)
if schema.Embed {
s.removeEmbed(&schema)
2017-11-11 04:44:02 +00:00
}
2017-12-16 08:25:28 +00:00
return s
}
func (s *Schemas) removeReferences(schema *Schema) {
for name, values := range s.references {
changed := false
var modified []BackReference
for _, value := range values {
if value.Schema.ID == schema.ID && value.Schema.Version.Path == schema.Version.Path {
changed = true
continue
}
modified = append(modified, value)
}
if changed {
s.references[name] = modified
}
2017-11-13 19:50:25 +00:00
}
2017-12-16 08:25:28 +00:00
}
func (s *Schemas) AddSchema(schema Schema) *Schemas {
s.Lock()
defer s.Unlock()
return s.doAddSchema(schema, false)
2017-12-16 08:25:28 +00:00
}
func (s *Schemas) ForceAddSchema(schema Schema) *Schemas {
s.Lock()
defer s.Unlock()
return s.doAddSchema(schema, true)
}
func (s *Schemas) doAddSchema(schema Schema, replace bool) *Schemas {
2017-12-16 08:25:28 +00:00
s.setupDefaults(&schema)
if s.AddHook != nil {
s.AddHook(&schema)
2017-11-21 20:46:30 +00:00
}
2017-11-11 04:44:02 +00:00
schemas, ok := s.schemasByPath[schema.Version.Path]
if !ok {
schemas = map[string]*Schema{}
s.schemasByPath[schema.Version.Path] = schemas
s.versions = append(s.versions, schema.Version)
}
if _, ok := schemas[schema.ID]; !ok ||
(replace && schema.DynamicSchemaVersion != schemas[schema.ID].DynamicSchemaVersion) {
2017-12-12 03:58:43 +00:00
schemas[schema.ID] = &schema
if replace {
for i, candidate := range s.schemas {
if candidate.ID == schema.ID {
s.schemas[i] = &schema
break
}
}
} else {
s.schemas = append(s.schemas, &schema)
}
2017-12-13 15:53:28 +00:00
2017-12-16 08:25:28 +00:00
if !schema.Embed {
s.addReferences(&schema)
2017-12-13 15:53:28 +00:00
}
2017-11-11 04:44:02 +00:00
}
2017-12-16 08:25:28 +00:00
if schema.Embed {
s.embed(&schema)
}
2017-11-11 04:44:02 +00:00
return s
}
2017-12-16 08:25:28 +00:00
func (s *Schemas) removeEmbed(schema *Schema) {
target := s.doSchema(&schema.Version, schema.EmbedType, false)
if target == nil {
return
}
newSchema := *target
newSchema.ResourceFields = map[string]Field{}
for k, v := range target.ResourceFields {
newSchema.ResourceFields[k] = v
}
for k := range schema.ResourceFields {
delete(newSchema.ResourceFields, k)
}
s.doRemoveSchema(*target)
s.doAddSchema(newSchema, false)
2017-12-16 08:25:28 +00:00
}
func (s *Schemas) embed(schema *Schema) {
target := s.doSchema(&schema.Version, schema.EmbedType, false)
if target == nil {
return
}
newSchema := *target
newSchema.ResourceFields = map[string]Field{}
for k, v := range target.ResourceFields {
// We remove the dynamic fields off the existing schema in case
// they've been removed from the dynamic schema so they won't
// be accidentally left over
if !v.DynamicField {
newSchema.ResourceFields[k] = v
}
2017-12-16 08:25:28 +00:00
}
for k, v := range schema.ResourceFields {
newSchema.ResourceFields[k] = v
}
s.doRemoveSchema(*target)
s.doAddSchema(newSchema, false)
2017-12-16 08:25:28 +00:00
}
func (s *Schemas) addReferences(schema *Schema) {
for name, field := range schema.ResourceFields {
if !definition.IsReferenceType(field.Type) {
continue
}
refType := definition.SubType(field.Type)
if !strings.HasPrefix(refType, "/") {
refType = convert.ToFullReference(schema.Version.Path, refType)
}
s.references[refType] = append(s.references[refType], BackReference{
FieldName: name,
Schema: schema,
})
}
}
func (s *Schemas) setupDefaults(schema *Schema) {
schema.Type = "/meta/schemas/schema"
if schema.ID == "" {
s.errors = append(s.errors, fmt.Errorf("ID is not set on schema: %v", schema))
return
}
if schema.Version.Path == "" || schema.Version.Version == "" {
s.errors = append(s.errors, fmt.Errorf("version is not set on schema: %s", schema.ID))
return
}
if schema.PluralName == "" {
schema.PluralName = name.GuessPluralName(schema.ID)
}
if schema.CodeName == "" {
schema.CodeName = convert.Capitalize(schema.ID)
}
if schema.CodeNamePlural == "" {
schema.CodeNamePlural = name.GuessPluralName(schema.CodeName)
}
if schema.BaseType == "" {
schema.BaseType = schema.ID
}
}
2017-12-13 15:53:28 +00:00
func (s *Schemas) References(schema *Schema) []BackReference {
refType := convert.ToFullReference(schema.Version.Path, schema.ID)
2017-12-16 08:25:28 +00:00
s.Lock()
defer s.Unlock()
2017-12-13 15:53:28 +00:00
return s.references[refType]
}
2017-11-11 04:44:02 +00:00
func (s *Schemas) AddMapper(version *APIVersion, schemaID string, mapper Mapper) *Schemas {
mappers, ok := s.mappers[version.Path]
if !ok {
2017-11-21 20:46:30 +00:00
mappers = map[string][]Mapper{}
2017-11-11 04:44:02 +00:00
s.mappers[version.Path] = mappers
}
2017-11-21 20:46:30 +00:00
mappers[schemaID] = append(mappers[schemaID], mapper)
2017-11-11 04:44:02 +00:00
return s
}
func (s *Schemas) SchemasForVersion(version APIVersion) map[string]*Schema {
2017-12-16 08:25:28 +00:00
s.Lock()
defer s.Unlock()
2017-11-11 04:44:02 +00:00
return s.schemasByPath[version.Path]
}
func (s *Schemas) Versions() []APIVersion {
return s.versions
}
func (s *Schemas) Schemas() []*Schema {
return s.schemas
}
2017-11-21 20:46:30 +00:00
func (s *Schemas) mapper(version *APIVersion, name string) []Mapper {
2017-11-11 04:44:02 +00:00
var (
path string
)
if strings.Contains(name, "/") {
idx := strings.LastIndex(name, "/")
path = name[0:idx]
name = name[idx+1:]
} else if version != nil {
path = version.Path
} else {
path = "core"
}
mappers, ok := s.mappers[path]
if !ok {
return nil
}
mapper := mappers[name]
if mapper != nil {
return mapper
}
return nil
}
func (s *Schemas) Schema(version *APIVersion, name string) *Schema {
2017-12-16 08:25:28 +00:00
return s.doSchema(version, name, true)
}
func (s *Schemas) doSchema(version *APIVersion, name string, lock bool) *Schema {
2017-11-11 04:44:02 +00:00
var (
path string
)
2017-11-21 20:46:30 +00:00
if strings.Contains(name, "/schemas/") {
parts := strings.SplitN(name, "/schemas/", 2)
path = parts[0]
name = parts[1]
2017-11-11 04:44:02 +00:00
} else if version != nil {
path = version.Path
} else {
path = "core"
}
2017-12-16 08:25:28 +00:00
if lock {
s.Lock()
2021-09-22 05:14:50 +00:00
defer s.Unlock()
2017-12-16 08:25:28 +00:00
}
2017-11-11 04:44:02 +00:00
schemas, ok := s.schemasByPath[path]
if !ok {
return nil
}
schema := schemas[name]
if schema != nil {
return schema
}
for _, check := range schemas {
if strings.EqualFold(check.ID, name) || strings.EqualFold(check.PluralName, name) {
return check
}
}
return nil
}
2018-02-09 20:31:12 +00:00
func (s *Schemas) SubContextVersionForSchema(schema *Schema) *APIVersion {
fullName := fmt.Sprintf("%s/schemas/%s", schema.Version.Path, schema.ID)
for _, version := range s.Versions() {
if version.SubContextSchema == fullName {
return &version
}
}
return nil
}
2018-03-22 22:53:36 +00:00
type MultiErrors struct {
Errors []error
2017-11-11 04:44:02 +00:00
}
type Errors struct {
errors []error
}
func (e *Errors) Add(err error) {
if err != nil {
e.errors = append(e.errors, err)
}
}
func (e *Errors) Err() error {
return NewErrors(e.errors...)
}
func NewErrors(inErrors ...error) error {
var errors []error
for _, err := range inErrors {
if err != nil {
errors = append(errors, err)
}
}
2017-11-11 04:44:02 +00:00
if len(errors) == 0 {
return nil
} else if len(errors) == 1 {
return errors[0]
}
2018-03-22 22:53:36 +00:00
return &MultiErrors{
Errors: errors,
2017-11-11 04:44:02 +00:00
}
}
2018-03-22 22:53:36 +00:00
func (m *MultiErrors) Error() string {
2017-11-11 04:44:02 +00:00
buf := bytes.NewBuffer(nil)
2018-03-22 22:53:36 +00:00
for _, err := range m.Errors {
2017-11-11 04:44:02 +00:00
if buf.Len() > 0 {
buf.WriteString(", ")
}
buf.WriteString(err.Error())
}
return buf.String()
}