mirror of
https://github.com/ent/ent.git
synced 2026-05-22 09:31:45 +03:00
373 lines
10 KiB
Go
373 lines
10 KiB
Go
// Copyright 2019-present Facebook Inc. All rights reserved.
|
|
// This source code is licensed under the Apache 2.0 license found
|
|
// in the LICENSE file in the root directory of this source tree.
|
|
|
|
package load
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"reflect"
|
|
|
|
"github.com/facebookincubator/ent"
|
|
"github.com/facebookincubator/ent/schema/edge"
|
|
"github.com/facebookincubator/ent/schema/field"
|
|
"github.com/facebookincubator/ent/schema/index"
|
|
)
|
|
|
|
// Schema represents an ent.Schema that was loaded from a complied user package.
|
|
type Schema struct {
|
|
Name string `json:"name,omitempty"`
|
|
Config ent.Config `json:"config,omitempty"`
|
|
Edges []*Edge `json:"edges,omitempty"`
|
|
Fields []*Field `json:"fields,omitempty"`
|
|
Indexes []*Index `json:"indexes,omitempty"`
|
|
Hooks []*Position `json:"hooks,omitempty"`
|
|
Policy bool `json:"policy,omitempty"`
|
|
}
|
|
|
|
// Position describes a field position in the schema.
|
|
type Position struct {
|
|
Index int // Field index in the field list.
|
|
MixedIn bool // Indicates if the field was mixed-in.
|
|
MixinIndex int // Mixin index in the mixin list.
|
|
}
|
|
|
|
// Field represents an ent.Field that was loaded from a complied user package.
|
|
type Field struct {
|
|
Name string `json:"name,omitempty"`
|
|
Info *field.TypeInfo `json:"type,omitempty"`
|
|
Tag string `json:"tag,omitempty"`
|
|
Size *int64 `json:"size,omitempty"`
|
|
Enums []string `json:"enums,omitempty"`
|
|
Unique bool `json:"unique,omitempty"`
|
|
Nillable bool `json:"nillable,omitempty"`
|
|
Optional bool `json:"optional,omitempty"`
|
|
Default bool `json:"default,omitempty"`
|
|
DefaultValue interface{} `json:"default_value,omitempty"`
|
|
UpdateDefault bool `json:"update_default,omitempty"`
|
|
Immutable bool `json:"immutable,omitempty"`
|
|
Validators int `json:"validators,omitempty"`
|
|
StorageKey string `json:"storage_key,omitempty"`
|
|
Position *Position `json:"position,omitempty"`
|
|
Sensitive bool `json:"sensitive,omitempty"`
|
|
SchemaType map[string]string `json:"schema_type,omitempty"`
|
|
}
|
|
|
|
// Edge represents an ent.Edge that was loaded from a complied user package.
|
|
type Edge struct {
|
|
Name string `json:"name,omitempty"`
|
|
Type string `json:"type,omitempty"`
|
|
Tag string `json:"tag,omitempty"`
|
|
RefName string `json:"ref_name,omitempty"`
|
|
Ref *Edge `json:"ref,omitempty"`
|
|
Unique bool `json:"unique,omitempty"`
|
|
Inverse bool `json:"inverse,omitempty"`
|
|
Required bool `json:"required,omitempty"`
|
|
}
|
|
|
|
// Index represents an ent.Index that was loaded from a complied user package.
|
|
type Index struct {
|
|
Unique bool `json:"unique,omitempty"`
|
|
Edges []string `json:"edges,omitempty"`
|
|
Fields []string `json:"fields,omitempty"`
|
|
StorageKey string `json:"storage_key,omitempty"`
|
|
}
|
|
|
|
// NewEdge creates an loaded edge from edge descriptor.
|
|
func NewEdge(ed *edge.Descriptor) *Edge {
|
|
ne := &Edge{
|
|
Tag: ed.Tag,
|
|
Type: ed.Type,
|
|
Name: ed.Name,
|
|
Unique: ed.Unique,
|
|
Inverse: ed.Inverse,
|
|
Required: ed.Required,
|
|
RefName: ed.RefName,
|
|
}
|
|
if ref := ed.Ref; ref != nil {
|
|
ne.Ref = NewEdge(ref)
|
|
}
|
|
return ne
|
|
}
|
|
|
|
// NewField creates an loaded field from field descriptor.
|
|
func NewField(fd *field.Descriptor) (*Field, error) {
|
|
sf := &Field{
|
|
Name: fd.Name,
|
|
Info: fd.Info,
|
|
Tag: fd.Tag,
|
|
Enums: fd.Enums,
|
|
Unique: fd.Unique,
|
|
Nillable: fd.Nillable,
|
|
Optional: fd.Optional,
|
|
Default: fd.Default != nil,
|
|
UpdateDefault: fd.UpdateDefault != nil,
|
|
Immutable: fd.Immutable,
|
|
StorageKey: fd.StorageKey,
|
|
Validators: len(fd.Validators),
|
|
Sensitive: fd.Sensitive,
|
|
SchemaType: fd.SchemaType,
|
|
}
|
|
if sf.Info == nil {
|
|
return nil, fmt.Errorf("missing type info for field %q", sf.Name)
|
|
}
|
|
if size := int64(fd.Size); size != 0 {
|
|
sf.Size = &size
|
|
}
|
|
// If the default value can be encoded to the generator.
|
|
// For example, not a function like time.Now.
|
|
if _, err := json.Marshal(fd.Default); err == nil {
|
|
sf.DefaultValue = fd.Default
|
|
}
|
|
if fd.Info.Type == field.TypeUUID && fd.Default != nil {
|
|
typ := reflect.TypeOf(fd.Default)
|
|
if typ.Kind() != reflect.Func || typ.NumIn() != 0 || typ.NumOut() != 1 || typ.Out(0).String() != fd.Info.String() {
|
|
return nil, fmt.Errorf("expect type (func() %s) for uuid default value", fd.Info.String())
|
|
}
|
|
}
|
|
return sf, nil
|
|
}
|
|
|
|
// NewIndex creates an loaded index from index descriptor.
|
|
func NewIndex(idx *index.Descriptor) *Index {
|
|
return &Index{
|
|
Edges: idx.Edges,
|
|
Fields: idx.Fields,
|
|
Unique: idx.Unique,
|
|
StorageKey: idx.StorageKey,
|
|
}
|
|
}
|
|
|
|
// MarshalSchema encode the ent.Schema interface into a JSON
|
|
// that can be decoded into the Schema object object.
|
|
func MarshalSchema(schema ent.Interface) (b []byte, err error) {
|
|
s := &Schema{
|
|
Config: schema.Config(),
|
|
Name: indirect(reflect.TypeOf(schema)).Name(),
|
|
}
|
|
if err := s.loadMixin(schema); err != nil {
|
|
return nil, fmt.Errorf("schema %q: %v", s.Name, err)
|
|
}
|
|
if err := s.loadFields(schema); err != nil {
|
|
return nil, fmt.Errorf("schema %q: %v", s.Name, err)
|
|
}
|
|
edges, err := safeEdges(schema)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("schema %q: %v", s.Name, err)
|
|
}
|
|
for _, e := range edges {
|
|
s.Edges = append(s.Edges, NewEdge(e.Descriptor()))
|
|
}
|
|
indexes, err := safeIndexes(schema)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("schema %q: %v", s.Name, err)
|
|
}
|
|
for _, idx := range indexes {
|
|
s.Indexes = append(s.Indexes, NewIndex(idx.Descriptor()))
|
|
}
|
|
if err := s.loadHooks(schema); err != nil {
|
|
return nil, fmt.Errorf("schema %q: %v", s.Name, err)
|
|
}
|
|
if err := s.loadPolicy(schema); err != nil {
|
|
return nil, fmt.Errorf("schema %q: %v", s.Name, err)
|
|
}
|
|
return json.Marshal(s)
|
|
}
|
|
|
|
// UnmarshalSchema decodes the given buffer to a loaded schema.
|
|
func UnmarshalSchema(buf []byte) (*Schema, error) {
|
|
s := &Schema{}
|
|
if err := json.Unmarshal(buf, s); err != nil {
|
|
return nil, err
|
|
}
|
|
for _, f := range s.Fields {
|
|
if err := f.defaults(); err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
return s, nil
|
|
}
|
|
|
|
// loadMixin loads mixin to schema from ent.Interface.
|
|
func (s *Schema) loadMixin(schema ent.Interface) error {
|
|
mixin, err := safeMixin(schema)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
for i, mx := range mixin {
|
|
fields, err := safeFields(mx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
for j, f := range fields {
|
|
sf, err := NewField(f.Descriptor())
|
|
if err != nil {
|
|
return err
|
|
}
|
|
sf.Position = &Position{
|
|
Index: j,
|
|
MixedIn: true,
|
|
MixinIndex: i,
|
|
}
|
|
s.Fields = append(s.Fields, sf)
|
|
}
|
|
edges, err := safeEdges(mx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
for _, e := range edges {
|
|
s.Edges = append(s.Edges, NewEdge(e.Descriptor()))
|
|
}
|
|
indexes, err := safeIndexes(mx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
for _, idx := range indexes {
|
|
s.Indexes = append(s.Indexes, NewIndex(idx.Descriptor()))
|
|
}
|
|
hooks, err := safeHooks(mx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
for j := range hooks {
|
|
s.Hooks = append(s.Hooks, &Position{
|
|
Index: j,
|
|
MixedIn: true,
|
|
MixinIndex: i,
|
|
})
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// loadFields loads field to schema from ent.Interface.
|
|
func (s *Schema) loadFields(schema ent.Interface) error {
|
|
fields, err := safeFields(schema)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
for i, f := range fields {
|
|
sf, err := NewField(f.Descriptor())
|
|
if err != nil {
|
|
return err
|
|
}
|
|
sf.Position = &Position{Index: i}
|
|
s.Fields = append(s.Fields, sf)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (s *Schema) loadHooks(schema ent.Interface) error {
|
|
hooks, err := safeHooks(schema)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
for i := range hooks {
|
|
s.Hooks = append(s.Hooks, &Position{
|
|
Index: i,
|
|
MixedIn: false,
|
|
})
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (s *Schema) loadPolicy(schema ent.Interface) error {
|
|
policy, err := safePolicy(schema)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
s.Policy = policy != nil
|
|
return nil
|
|
}
|
|
|
|
func (f *Field) defaults() error {
|
|
if !f.Default || !f.Info.Numeric() {
|
|
return nil
|
|
}
|
|
n, ok := f.DefaultValue.(float64)
|
|
if !ok {
|
|
return fmt.Errorf("unexpected default value type for field: %q", f.Name)
|
|
}
|
|
switch t := f.Info.Type; {
|
|
case t >= field.TypeInt8 && t <= field.TypeInt64:
|
|
f.DefaultValue = int64(n)
|
|
case t >= field.TypeUint8 && t <= field.TypeUint64:
|
|
f.DefaultValue = uint64(n)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// safeFields wraps the schema.Fields and mixin.Fields method with recover to ensure no panics in marshaling.
|
|
func safeFields(fd interface{ Fields() []ent.Field }) (fields []ent.Field, err error) {
|
|
defer func() {
|
|
if v := recover(); v != nil {
|
|
err = fmt.Errorf("%T.Fields panics: %v", fd, v)
|
|
fields = nil
|
|
}
|
|
}()
|
|
return fd.Fields(), nil
|
|
}
|
|
|
|
// safeEdges wraps the schema.Edges method with recover to ensure no panics in marshaling.
|
|
func safeEdges(schema interface{ Edges() []ent.Edge }) (edges []ent.Edge, err error) {
|
|
defer func() {
|
|
if v := recover(); v != nil {
|
|
err = fmt.Errorf("schema.Edges panics: %v", v)
|
|
edges = nil
|
|
}
|
|
}()
|
|
return schema.Edges(), nil
|
|
}
|
|
|
|
// safeIndexes wraps the schema.Indexes method with recover to ensure no panics in marshaling.
|
|
func safeIndexes(schema interface{ Indexes() []ent.Index }) (indexes []ent.Index, err error) {
|
|
defer func() {
|
|
if v := recover(); v != nil {
|
|
err = fmt.Errorf("schema.Indexes panics: %v", v)
|
|
indexes = nil
|
|
}
|
|
}()
|
|
return schema.Indexes(), nil
|
|
}
|
|
|
|
// safeMixin wraps the schema.Mixin method with recover to ensure no panics in marshaling.
|
|
func safeMixin(schema ent.Interface) (mixin []ent.Mixin, err error) {
|
|
defer func() {
|
|
if v := recover(); v != nil {
|
|
err = fmt.Errorf("schema.Mixin panics: %v", v)
|
|
mixin = nil
|
|
}
|
|
}()
|
|
return schema.Mixin(), nil
|
|
}
|
|
|
|
// safeHooks wraps the schema.Hooks method with recover to ensure no panics in marshaling.
|
|
func safeHooks(schema interface{ Hooks() []ent.Hook }) (hooks []ent.Hook, err error) {
|
|
defer func() {
|
|
if v := recover(); v != nil {
|
|
err = fmt.Errorf("schema.Hooks panics: %v", v)
|
|
hooks = nil
|
|
}
|
|
}()
|
|
return schema.Hooks(), nil
|
|
}
|
|
|
|
// safePolicy wraps the schema.Policy method with recover to ensure no panics in marshaling.
|
|
func safePolicy(schema ent.Interface) (policy ent.Policy, err error) {
|
|
defer func() {
|
|
if v := recover(); v != nil {
|
|
err = fmt.Errorf("schema.Policy panics: %v", v)
|
|
policy = nil
|
|
}
|
|
}()
|
|
return schema.Policy(), nil
|
|
}
|
|
|
|
func indirect(t reflect.Type) reflect.Type {
|
|
for t.Kind() == reflect.Ptr {
|
|
t = t.Elem()
|
|
}
|
|
return t
|
|
}
|