diff --git a/lib/graphql.go b/lib/graphql.go index e328c46..4b09fbe 100644 --- a/lib/graphql.go +++ b/lib/graphql.go @@ -60,6 +60,7 @@ type Type struct { Fields []*Field Directives []*Directive Descriptions *[]string + Extend bool } type Arg struct { diff --git a/lib/lex.go b/lib/lex.go index 415e083..054290e 100644 --- a/lib/lex.go +++ b/lib/lex.go @@ -51,6 +51,7 @@ const ( tokEnum // enum tokScalar // scalar tokUnion // union + tokExtend // extend tokSchema // schema ) @@ -134,6 +135,8 @@ func (typ tokenType) String() string { return "scalar" case tokUnion: return "union" + case tokExtend: + return "extend" case tokSchema: return "schema" default: @@ -328,6 +331,8 @@ func (l *lexer) next() *token { return mkToken(tokScalar, "scalar") case "union": return mkToken(tokUnion, "union") + case "extend": + return mkToken(tokExtend, "extend") case "schema": return mkToken(tokSchema, "schema") default: diff --git a/lib/schema.go b/lib/schema.go index f2a7479..a0ed7f8 100644 --- a/lib/schema.go +++ b/lib/schema.go @@ -60,6 +60,7 @@ func (sc *Schema) ReadSchema(path string) { } func (s *Schema) Parse(p *Parser) { + isExtended := false for { tok := p.lex.next() if tok.typ == tokEOF { @@ -133,6 +134,9 @@ func (s *Schema) Parse(p *Parser) { } s.DirectiveDefinitions = append(s.DirectiveDefinitions, &d) + case tokExtend: + isExtended = true + case tokScalar: c := Scalar{} c.Filename = p.lex.filename @@ -346,6 +350,8 @@ func (s *Schema) Parse(p *Parser) { case tokType: t := Type{} + t.Extend = isExtended + isExtended = false t.Filename = p.lex.filename t.Line = p.lex.line t.Column = p.lex.col @@ -357,6 +363,9 @@ func (s *Schema) Parse(p *Parser) { next := p.lex.next() switch next.typ { case tokImplements: + if len(t.Directives) > 0 { + errorf(`%s:%d:%d: directives cann't be placed in front of implements`, p.lex.filename, p.lex.line, p.lex.col) + } t.Impl = true name, _ := p.lex.consumeIdent() t.ImplTypes = append(t.ImplTypes, name.String()) @@ -533,22 +542,28 @@ func (s *Schema) MergeTypeName(wg *sync.WaitGroup) { if _, ok := seen[v.Name]; ok { for i := 0; i < j; i++ { if s.Types[i].Name == v.Name { - if reflect.DeepEqual(s.Types[i].ImplTypes, v.ImplTypes) && IsEqualWithoutDescriptions(s.Types[i].Directives, v.Directives) { + if v.Extend { s.Types[i].Fields = mergeFields(s.Types[i].Fields, v.Fields) - mergeDescriptionsAndComments(s.Types[i].Directives, v.Directives) + s.Types[i].Directives = mergeDirectives(s.Types[i].Directives, v.Directives) break } else { + if reflect.DeepEqual(s.Types[i].ImplTypes, v.ImplTypes) && IsEqualWithoutDescriptions(s.Types[i].Directives, v.Directives) { + s.Types[i].Fields = mergeFields(s.Types[i].Fields, v.Fields) + mergeDescriptionsAndComments(s.Types[i].Directives, v.Directives) + break + } else { - rel1, err := GetRelPath(s.Types[i].Filename) - if err != nil { - panic(err) - } - rel2, err := GetRelPath(v.Filename) - if err != nil { - panic(err) - } + rel1, err := GetRelPath(s.Types[i].Filename) + if err != nil { + panic(err) + } + rel2, err := GetRelPath(v.Filename) + if err != nil { + panic(err) + } - errorf("Duplicated Types: %s(%s:%v:%v) and (%s:%v:%v)", s.Types[i].Name, *rel1, s.Types[i].Line, s.Types[i].Column, *rel2, v.Line, v.Column) + errorf("Duplicated Types: %s(%s:%v:%v) and (%s:%v:%v)", s.Types[i].Name, *rel1, s.Types[i].Line, s.Types[i].Column, *rel2, v.Line, v.Column) + } } } } diff --git a/lib/schema_test.go b/lib/schema_test.go index d9b54f0..97c71c8 100644 --- a/lib/schema_test.go +++ b/lib/schema_test.go @@ -38,3 +38,39 @@ func TestGetSchema(t *testing.T) { } } + +func TestMergeDirectives(t *testing.T) { + str := make([]string, 0) + a := []*Directive{ + { + Name: "talkable", + DirectiveArgs: []*DirectiveArg{}, + Descriptions: &str, + }, + } + b := []*Directive{ + { + Name: "talkable", + DirectiveArgs: []*DirectiveArg{}, + Descriptions: &str, + }, + } + c := []*Directive{ + { + Name: "walkable", + DirectiveArgs: []*DirectiveArg{}, + Descriptions: &str, + }, + } + ds := mergeDirectives(a, b) + + if len(ds) == 0 { + t.Fatal("should be more than 0") + } + + ds = mergeDirectives(a, c) + + if len(ds) == 0 || len(ds) == 1 { + t.Fatal("should be more than 1") + } +} diff --git a/lib/util.go b/lib/util.go index 0f21f84..6692090 100644 --- a/lib/util.go +++ b/lib/util.go @@ -137,3 +137,78 @@ func mergeDescriptionsAndComments(a, b interface{}) { } } } + +func mergeDirectiveArgs(a, b []*DirectiveArg) []*DirectiveArg { + merged := make([]*DirectiveArg, len(a)) + copy(merged, a) + + for _, bArg := range b { + found := false + for i, mArg := range merged { + if mArg.Name == bArg.Name && compareValuesAndIsList(mArg, bArg) { + merged[i].Descriptions = mergeDescriptions(mArg.Descriptions, bArg.Descriptions) + found = true + break + } + } + if !found { + merged = append(merged, bArg) + } + } + return merged +} + +func compareValuesAndIsList(a, b *DirectiveArg) bool { + if a.IsList != b.IsList { + return false + } + if len(a.Value) != len(b.Value) { + return false + } + for i := range a.Value { + if a.Value[i] != b.Value[i] { + return false + } + } + return true +} + +func mergeDescriptions(a, b *[]string) *[]string { + if a == nil && b == nil { + return nil + } + if a == nil { + return b + } + if b == nil { + return a + } + merged := make([]string, len(*a)+len(*b)) + copy(merged, *a) + merged = append(merged, *b...) + return &merged +} + +func mergeDirectives(a, b []*Directive) []*Directive { + directiveMap := make(map[string]*Directive) + + for _, dir := range a { + directiveMap[dir.Name] = dir + } + + for _, dirB := range b { + if dirA, exists := directiveMap[dirB.Name]; exists { + dirA.DirectiveArgs = mergeDirectiveArgs(dirA.DirectiveArgs, dirB.DirectiveArgs) + dirA.Descriptions = mergeDescriptions(dirA.Descriptions, dirB.Descriptions) + } else { + directiveMap[dirB.Name] = dirB + } + } + + merged := make([]*Directive, 0, len(directiveMap)) + for _, dir := range directiveMap { + merged = append(merged, dir) + } + + return merged +} diff --git a/test/object_extension/generated.graphql b/test/object_extension/generated.graphql new file mode 100644 index 0000000..df1238f --- /dev/null +++ b/test/object_extension/generated.graphql @@ -0,0 +1,11 @@ +type Person implements Node @talkable @walkable { + id: ID! + createTime: Time! + updateTime: Time! + name: String! + hasCar: Boolean! +} + + + + diff --git a/test/object_extension/schema/a.graphql b/test/object_extension/schema/a.graphql new file mode 100644 index 0000000..77200ec --- /dev/null +++ b/test/object_extension/schema/a.graphql @@ -0,0 +1,6 @@ +type Person implements Node @talkable{ + id: ID! + createTime: Time! + updateTime: Time! + name: String! +} diff --git a/test/object_extension/schema/b.graphql b/test/object_extension/schema/b.graphql new file mode 100644 index 0000000..6b9a6b4 --- /dev/null +++ b/test/object_extension/schema/b.graphql @@ -0,0 +1,3 @@ +extend type Person @walkable { + hasCar: Boolean! +}