mirror of
https://github.com/ent/ent.git
synced 2026-05-22 09:31:45 +03:00
* schema/edge: an storage-key option for edges * schema/edge: use Column(s) with functional options
378 lines
9.7 KiB
Go
378 lines
9.7 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"
|
|
"math"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/facebookincubator/ent"
|
|
"github.com/facebookincubator/ent/schema/edge"
|
|
"github.com/facebookincubator/ent/schema/field"
|
|
"github.com/facebookincubator/ent/schema/index"
|
|
"github.com/facebookincubator/ent/schema/mixin"
|
|
|
|
"github.com/google/uuid"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
type User struct {
|
|
ent.Schema
|
|
}
|
|
|
|
func (User) Fields() []ent.Field {
|
|
return []ent.Field{
|
|
field.Int("age"),
|
|
field.String("name").
|
|
Default("unknown"),
|
|
field.String("nillable").
|
|
Nillable(),
|
|
field.String("optional").
|
|
Optional(),
|
|
field.Enum("state").
|
|
Values("on", "off").
|
|
Optional(),
|
|
field.String("sensitive").
|
|
Sensitive(),
|
|
field.Time("creation_time").
|
|
Default(time.Now),
|
|
field.UUID("uuid", uuid.UUID{}).
|
|
Default(uuid.New),
|
|
}
|
|
}
|
|
|
|
func (User) Edges() []ent.Edge {
|
|
return []ent.Edge{
|
|
edge.To("groups", Group.Type),
|
|
edge.To("parent", User.Type).
|
|
Unique().
|
|
StorageKey(edge.Column("user_parent_id")).
|
|
From("children"),
|
|
}
|
|
}
|
|
|
|
func (User) Indexes() []ent.Index {
|
|
return []ent.Index{
|
|
index.Fields("name", "address").
|
|
Unique(),
|
|
index.Fields("name").
|
|
Edges("parent").
|
|
StorageKey("user_parent_name").
|
|
Unique(),
|
|
}
|
|
}
|
|
|
|
type Group struct{ ent.Schema }
|
|
|
|
func (Group) Fields() []ent.Field { return nil }
|
|
|
|
func (Group) Edges() []ent.Edge {
|
|
return []ent.Edge{
|
|
edge.From("users", User.Type),
|
|
}
|
|
}
|
|
|
|
func TestMarshalSchema(t *testing.T) {
|
|
for _, u := range []ent.Interface{User{}, &User{}} {
|
|
buf, err := MarshalSchema(u)
|
|
require.NoError(t, err)
|
|
|
|
schema := &Schema{}
|
|
require.NoError(t, json.Unmarshal(buf, schema))
|
|
require.Equal(t, "User", schema.Name)
|
|
require.Len(t, schema.Fields, 8)
|
|
require.Equal(t, "age", schema.Fields[0].Name)
|
|
require.Equal(t, field.TypeInt, schema.Fields[0].Info.Type)
|
|
|
|
require.Equal(t, "name", schema.Fields[1].Name)
|
|
require.Equal(t, field.TypeString, schema.Fields[1].Info.Type)
|
|
require.Equal(t, "unknown", schema.Fields[1].DefaultValue)
|
|
|
|
require.Equal(t, "nillable", schema.Fields[2].Name)
|
|
require.Equal(t, field.TypeString, schema.Fields[2].Info.Type)
|
|
require.True(t, schema.Fields[2].Nillable)
|
|
require.False(t, schema.Fields[2].Optional)
|
|
require.False(t, schema.Fields[2].Sensitive)
|
|
|
|
require.Equal(t, "optional", schema.Fields[3].Name)
|
|
require.Equal(t, field.TypeString, schema.Fields[3].Info.Type)
|
|
require.False(t, schema.Fields[3].Nillable)
|
|
require.True(t, schema.Fields[3].Optional)
|
|
|
|
require.Equal(t, "state", schema.Fields[4].Name)
|
|
require.Equal(t, field.TypeEnum, schema.Fields[4].Info.Type)
|
|
require.Equal(t, []string{"on", "off"}, schema.Fields[4].Enums)
|
|
|
|
require.Equal(t, "sensitive", schema.Fields[5].Name)
|
|
require.Equal(t, field.TypeString, schema.Fields[5].Info.Type)
|
|
require.True(t, schema.Fields[5].Sensitive)
|
|
|
|
require.Equal(t, "creation_time", schema.Fields[6].Name)
|
|
require.Equal(t, field.TypeTime, schema.Fields[6].Info.Type)
|
|
require.Nil(t, schema.Fields[6].DefaultValue)
|
|
|
|
require.Equal(t, "uuid", schema.Fields[7].Name)
|
|
require.Equal(t, field.TypeUUID, schema.Fields[7].Info.Type)
|
|
require.True(t, schema.Fields[7].Default)
|
|
|
|
require.Len(t, schema.Edges, 2)
|
|
require.Equal(t, "groups", schema.Edges[0].Name)
|
|
require.Equal(t, "Group", schema.Edges[0].Type)
|
|
require.False(t, schema.Edges[0].Inverse)
|
|
require.Equal(t, "children", schema.Edges[1].Name)
|
|
require.Equal(t, "user_parent_id", schema.Edges[1].StorageKey.Columns[0])
|
|
require.Equal(t, "User", schema.Edges[1].Type)
|
|
require.True(t, schema.Edges[1].Inverse)
|
|
require.Equal(t, "parent", schema.Edges[1].Ref.Name)
|
|
require.True(t, schema.Edges[1].Ref.Unique)
|
|
require.Equal(t, "user_parent_id", schema.Edges[1].Ref.StorageKey.Columns[0])
|
|
|
|
require.Equal(t, []string{"name", "address"}, schema.Indexes[0].Fields)
|
|
require.True(t, schema.Indexes[0].Unique)
|
|
require.Equal(t, []string{"name"}, schema.Indexes[1].Fields)
|
|
require.Equal(t, []string{"parent"}, schema.Indexes[1].Edges)
|
|
require.Equal(t, "user_parent_name", schema.Indexes[1].StorageKey)
|
|
require.True(t, schema.Indexes[1].Unique)
|
|
}
|
|
}
|
|
|
|
type InvalidEdge struct {
|
|
ent.Schema
|
|
}
|
|
|
|
// Edge panics because the edge declaration is invalid.
|
|
func (InvalidEdge) Edges() []ent.Edge {
|
|
return []ent.Edge{
|
|
edge.From("invalid", InvalidEdge{}.Type),
|
|
}
|
|
}
|
|
|
|
type InvalidUUID struct {
|
|
ent.Schema
|
|
}
|
|
|
|
func (InvalidUUID) Fields() []ent.Field {
|
|
return []ent.Field{
|
|
field.UUID("invalid", uuid.New()).
|
|
Default(time.Now),
|
|
}
|
|
}
|
|
|
|
func TestMarshalFails(t *testing.T) {
|
|
i1 := InvalidEdge{}
|
|
buf, err := MarshalSchema(i1)
|
|
require.Error(t, err)
|
|
require.Nil(t, buf)
|
|
|
|
i2 := InvalidUUID{}
|
|
buf, err = MarshalSchema(i2)
|
|
require.Nil(t, buf)
|
|
require.EqualError(t, err, `schema "InvalidUUID": field "invalid": expect type (func() uuid.UUID) for uuid default value`)
|
|
}
|
|
|
|
type WithDefaults struct {
|
|
ent.Schema
|
|
}
|
|
|
|
func (WithDefaults) Fields() []ent.Field {
|
|
return []ent.Field{
|
|
field.Int("int").
|
|
Default(1),
|
|
field.Float("float").
|
|
Default(math.Pi),
|
|
field.String("string").
|
|
Default("foo"),
|
|
field.Bool("string").
|
|
Default(true),
|
|
field.Time("updated_at").
|
|
UpdateDefault(time.Now),
|
|
}
|
|
}
|
|
|
|
func (WithDefaults) Edges() []ent.Edge {
|
|
return nil
|
|
}
|
|
|
|
func (WithDefaults) Indexes() []ent.Index {
|
|
return nil
|
|
}
|
|
|
|
func TestMarshalDefaults(t *testing.T) {
|
|
d := WithDefaults{}
|
|
buf, err := MarshalSchema(d)
|
|
require.NoError(t, err)
|
|
|
|
schema := &Schema{}
|
|
err = json.Unmarshal(buf, schema)
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, "WithDefaults", schema.Name)
|
|
require.True(t, schema.Fields[0].Default)
|
|
require.True(t, schema.Fields[1].Default)
|
|
require.True(t, schema.Fields[2].Default)
|
|
require.True(t, schema.Fields[3].Default)
|
|
require.False(t, schema.Fields[4].Default)
|
|
require.True(t, schema.Fields[4].UpdateDefault)
|
|
}
|
|
|
|
type TimeMixin struct {
|
|
mixin.Schema
|
|
}
|
|
|
|
func (TimeMixin) Fields() []ent.Field {
|
|
return []ent.Field{
|
|
field.Time("created_at").
|
|
Immutable().
|
|
Default(time.Now),
|
|
field.Time("updated_at").
|
|
Default(time.Now).
|
|
UpdateDefault(time.Now),
|
|
}
|
|
}
|
|
|
|
type HooksMixin struct {
|
|
mixin.Schema
|
|
}
|
|
|
|
func (HooksMixin) Fields() []ent.Field {
|
|
return []ent.Field{
|
|
field.String("boring"),
|
|
}
|
|
}
|
|
|
|
func (HooksMixin) Edges() []ent.Edge {
|
|
return []ent.Edge{
|
|
edge.To("user", User.Type).
|
|
Unique(),
|
|
}
|
|
}
|
|
|
|
func (HooksMixin) Indexes() []ent.Index {
|
|
return []ent.Index{
|
|
index.Fields("boring").
|
|
Edges("user"),
|
|
}
|
|
}
|
|
|
|
func (HooksMixin) Hooks() []ent.Hook {
|
|
return []ent.Hook{
|
|
func(ent.Mutator) ent.Mutator { return nil },
|
|
func(ent.Mutator) ent.Mutator { return nil },
|
|
}
|
|
}
|
|
|
|
type WithMixin struct {
|
|
ent.Schema
|
|
}
|
|
|
|
func (WithMixin) Mixin() []ent.Mixin {
|
|
return []ent.Mixin{
|
|
TimeMixin{},
|
|
HooksMixin{},
|
|
}
|
|
}
|
|
|
|
func (WithMixin) Fields() []ent.Field {
|
|
return []ent.Field{
|
|
field.Int("field"),
|
|
}
|
|
}
|
|
|
|
func (WithMixin) Edges() []ent.Edge {
|
|
return []ent.Edge{
|
|
edge.To("owner", User.Type),
|
|
}
|
|
}
|
|
|
|
func (WithMixin) Indexes() []ent.Index {
|
|
return []ent.Index{
|
|
index.Fields("field").
|
|
Edges("owner").
|
|
Unique(),
|
|
}
|
|
}
|
|
|
|
func (WithMixin) Hooks() []ent.Hook {
|
|
return []ent.Hook{
|
|
func(ent.Mutator) ent.Mutator { return nil },
|
|
}
|
|
}
|
|
|
|
func TestMarshalMixin(t *testing.T) {
|
|
d := WithMixin{}
|
|
buf, err := MarshalSchema(d)
|
|
require.NoError(t, err)
|
|
|
|
schema := &Schema{}
|
|
err = json.Unmarshal(buf, schema)
|
|
require.NoError(t, err)
|
|
|
|
t.Run("Fields", func(t *testing.T) {
|
|
require.Equal(t, "WithMixin", schema.Name)
|
|
require.Equal(t, "created_at", schema.Fields[0].Name)
|
|
require.True(t, schema.Fields[0].Default)
|
|
require.True(t, schema.Fields[0].Position.MixedIn)
|
|
require.Equal(t, 0, schema.Fields[0].Position.MixinIndex)
|
|
require.Equal(t, 0, schema.Fields[0].Position.Index)
|
|
|
|
require.Equal(t, "updated_at", schema.Fields[1].Name)
|
|
require.True(t, schema.Fields[1].Default)
|
|
require.True(t, schema.Fields[1].UpdateDefault)
|
|
require.True(t, schema.Fields[1].Position.MixedIn)
|
|
require.Equal(t, 0, schema.Fields[1].Position.MixinIndex)
|
|
require.Equal(t, 1, schema.Fields[1].Position.Index)
|
|
|
|
require.Equal(t, "boring", schema.Fields[2].Name)
|
|
require.False(t, schema.Fields[2].Default)
|
|
require.False(t, schema.Fields[2].UpdateDefault)
|
|
require.True(t, schema.Fields[2].Position.MixedIn)
|
|
require.Equal(t, 1, schema.Fields[2].Position.MixinIndex)
|
|
require.Equal(t, 0, schema.Fields[2].Position.Index)
|
|
|
|
require.Equal(t, "field", schema.Fields[3].Name)
|
|
require.False(t, schema.Fields[3].Default)
|
|
require.False(t, schema.Fields[3].Position.MixedIn)
|
|
require.Equal(t, 0, schema.Fields[3].Position.Index)
|
|
})
|
|
|
|
t.Run("Hooks", func(t *testing.T) {
|
|
require.True(t, schema.Hooks[0].MixedIn)
|
|
require.True(t, schema.Hooks[1].MixedIn)
|
|
|
|
require.Equal(t, 1, schema.Hooks[0].MixinIndex)
|
|
require.Equal(t, 1, schema.Hooks[1].MixinIndex)
|
|
require.Equal(t, 0, schema.Hooks[0].Index)
|
|
require.Equal(t, 1, schema.Hooks[1].Index)
|
|
|
|
require.False(t, schema.Hooks[2].MixedIn)
|
|
require.Equal(t, 0, schema.Hooks[2].Index)
|
|
require.Equal(t, 0, schema.Hooks[2].MixinIndex)
|
|
})
|
|
|
|
t.Run("Edges", func(t *testing.T) {
|
|
require.Len(t, schema.Edges, 2)
|
|
require.Equal(t, "user", schema.Edges[0].Name)
|
|
require.Equal(t, "User", schema.Edges[0].Type)
|
|
require.True(t, schema.Edges[0].Unique)
|
|
|
|
require.Equal(t, "owner", schema.Edges[1].Name)
|
|
require.Equal(t, "User", schema.Edges[1].Type)
|
|
require.False(t, schema.Edges[1].Unique)
|
|
})
|
|
|
|
t.Run("Indexes", func(t *testing.T) {
|
|
require.Len(t, schema.Indexes, 2)
|
|
require.Equal(t, []string{"boring"}, schema.Indexes[0].Fields)
|
|
require.Equal(t, []string{"user"}, schema.Indexes[0].Edges)
|
|
require.False(t, schema.Indexes[0].Unique)
|
|
|
|
require.Equal(t, []string{"field"}, schema.Indexes[1].Fields)
|
|
require.Equal(t, []string{"owner"}, schema.Indexes[1].Edges)
|
|
require.True(t, schema.Indexes[1].Unique)
|
|
})
|
|
}
|