diff --git a/src/InterfaceGraph.ts b/src/InterfaceGraph.ts deleted file mode 100644 index 3f35bb4f..00000000 --- a/src/InterfaceGraph.ts +++ /dev/null @@ -1,56 +0,0 @@ -import { GratsDefinitionNode } from "./GraphQLConstructor"; -import { TypeContext } from "./TypeContext"; -import { DefaultMap } from "./utils/helpers"; -import { Kind } from "graphql"; - -export type InterfaceImplementor = { kind: "TYPE" | "INTERFACE"; name: string }; -export type InterfaceMap = DefaultMap>; - -/** - * Compute a map of interfaces to the types and interfaces that implement them. - */ -export function computeInterfaceMap( - typeContext: TypeContext, - docs: GratsDefinitionNode[], -): InterfaceMap { - // For each interface definition, we need to know which types and interfaces implement it. - const graph = new DefaultMap>( - () => new Set(), - ); - - const add = (interfaceName: string, implementor: InterfaceImplementor) => { - graph.get(interfaceName).add(implementor); - }; - - for (const doc of docs) { - switch (doc.kind) { - case Kind.INTERFACE_TYPE_DEFINITION: - case Kind.INTERFACE_TYPE_EXTENSION: - for (const implementor of doc.interfaces ?? []) { - const resolved = typeContext.resolveNamedType(implementor.name); - if (resolved.kind === "ERROR") { - // We trust that these errors will be reported elsewhere. - continue; - } - add(resolved.value.value, { - kind: "INTERFACE", - name: doc.name.value, - }); - } - break; - case Kind.OBJECT_TYPE_DEFINITION: - case Kind.OBJECT_TYPE_EXTENSION: - for (const implementor of doc.interfaces ?? []) { - const resolved = typeContext.resolveNamedType(implementor.name); - if (resolved.kind === "ERROR") { - // We trust that these errors will be reported elsewhere. - continue; - } - add(resolved.value.value, { kind: "TYPE", name: doc.name.value }); - } - break; - } - } - - return graph; -} diff --git a/src/tests/fixtures/extend_interface/addStringFieldToInterface.ts.expected b/src/tests/fixtures/extend_interface/addStringFieldToInterface.ts.expected index b9998b2c..1bb89f01 100644 --- a/src/tests/fixtures/extend_interface/addStringFieldToInterface.ts.expected +++ b/src/tests/fixtures/extend_interface/addStringFieldToInterface.ts.expected @@ -34,7 +34,7 @@ OUTPUT ----------------- -- SDL -- interface IPerson { - greeting: String + greeting: String @metadata(argCount: 1, name: "greeting", tsModulePath: "grats/src/tests/fixtures/extend_interface/addStringFieldToInterface.ts") hello: String @metadata } diff --git a/src/tests/fixtures/extend_interface/addStringFieldToInterfaceImplementedByInterface.ts.expected b/src/tests/fixtures/extend_interface/addStringFieldToInterfaceImplementedByInterface.ts.expected index 4ae26e0e..e868ba80 100644 --- a/src/tests/fixtures/extend_interface/addStringFieldToInterfaceImplementedByInterface.ts.expected +++ b/src/tests/fixtures/extend_interface/addStringFieldToInterfaceImplementedByInterface.ts.expected @@ -39,11 +39,11 @@ OUTPUT ----------------- -- SDL -- interface IPerson implements IThing { - greeting: String + greeting: String @metadata(argCount: 1, name: "greeting", tsModulePath: "grats/src/tests/fixtures/extend_interface/addStringFieldToInterfaceImplementedByInterface.ts") } interface IThing { - greeting: String + greeting: String @metadata(argCount: 1, name: "greeting", tsModulePath: "grats/src/tests/fixtures/extend_interface/addStringFieldToInterfaceImplementedByInterface.ts") } type Admin implements IPerson & IThing { diff --git a/src/tests/fixtures/extend_interface/addStringFieldToInterfaceTwice.invalid.ts.expected b/src/tests/fixtures/extend_interface/addStringFieldToInterfaceTwice.invalid.ts.expected index 3b898e1a..1c2e6843 100644 --- a/src/tests/fixtures/extend_interface/addStringFieldToInterfaceTwice.invalid.ts.expected +++ b/src/tests/fixtures/extend_interface/addStringFieldToInterfaceTwice.invalid.ts.expected @@ -39,24 +39,6 @@ OUTPUT ----------------- src/tests/fixtures/extend_interface/addStringFieldToInterfaceTwice.invalid.ts:9:17 - error: Field "IPerson.greeting" can only be defined once. -9 export function greeting(person: IPerson): string { - ~~~~~~~~ - - src/tests/fixtures/extend_interface/addStringFieldToInterfaceTwice.invalid.ts:13:5 - 13 /** @gqlField greeting */ - ~~~~~~~~~~~~~~~~~~~ - Related location -src/tests/fixtures/extend_interface/addStringFieldToInterfaceTwice.invalid.ts:9:17 - error: Field "Admin.greeting" can only be defined once. - -9 export function greeting(person: IPerson): string { - ~~~~~~~~ - - src/tests/fixtures/extend_interface/addStringFieldToInterfaceTwice.invalid.ts:13:5 - 13 /** @gqlField greeting */ - ~~~~~~~~~~~~~~~~~~~ - Related location -src/tests/fixtures/extend_interface/addStringFieldToInterfaceTwice.invalid.ts:9:17 - error: Field "User.greeting" can only be defined once. - 9 export function greeting(person: IPerson): string { ~~~~~~~~ diff --git a/src/tests/fixtures/extend_interface/redefineFiledThatExistsOnConcreteType.invalid.ts.expected b/src/tests/fixtures/extend_interface/redefineFiledThatExistsOnConcreteType.invalid.ts.expected deleted file mode 100644 index ec598487..00000000 --- a/src/tests/fixtures/extend_interface/redefineFiledThatExistsOnConcreteType.invalid.ts.expected +++ /dev/null @@ -1,40 +0,0 @@ ------------------ -INPUT ------------------ -/** @gqlInterface */ -interface IPerson { - name: string; - /** @gqlField */ - hello: string; -} - -/** @gqlField */ -export function greeting(person: IPerson): string { - return `Hello ${person.name}!`; -} - -/** @gqlType */ -class User implements IPerson { - __typename: "User"; - name: string; - /** @gqlField */ - hello: string; - - /** @gqlField */ - greeting(): string { - return `Hello ${this.name}!`; - } -} - ------------------ -OUTPUT ------------------ -src/tests/fixtures/extend_interface/redefineFiledThatExistsOnConcreteType.invalid.ts:21:3 - error: Field "User.greeting" can only be defined once. - -21 greeting(): string { - ~~~~~~~~ - - src/tests/fixtures/extend_interface/redefineFiledThatExistsOnConcreteType.invalid.ts:9:17 - 9 export function greeting(person: IPerson): string { - ~~~~~~~~ - Related location diff --git a/src/tests/fixtures/extend_interface/redefineFiledThatExistsOnConcreteType.invalid.ts b/src/tests/fixtures/extend_interface/redefineFiledThatExistsOnConcreteType.ts similarity index 100% rename from src/tests/fixtures/extend_interface/redefineFiledThatExistsOnConcreteType.invalid.ts rename to src/tests/fixtures/extend_interface/redefineFiledThatExistsOnConcreteType.ts diff --git a/src/tests/fixtures/extend_interface/redefineFiledThatExistsOnConcreteType.ts.expected b/src/tests/fixtures/extend_interface/redefineFiledThatExistsOnConcreteType.ts.expected new file mode 100644 index 00000000..b706bb00 --- /dev/null +++ b/src/tests/fixtures/extend_interface/redefineFiledThatExistsOnConcreteType.ts.expected @@ -0,0 +1,81 @@ +----------------- +INPUT +----------------- +/** @gqlInterface */ +interface IPerson { + name: string; + /** @gqlField */ + hello: string; +} + +/** @gqlField */ +export function greeting(person: IPerson): string { + return `Hello ${person.name}!`; +} + +/** @gqlType */ +class User implements IPerson { + __typename: "User"; + name: string; + /** @gqlField */ + hello: string; + + /** @gqlField */ + greeting(): string { + return `Hello ${this.name}!`; + } +} + +----------------- +OUTPUT +----------------- +-- SDL -- +interface IPerson { + greeting: String @metadata(argCount: 1, name: "greeting", tsModulePath: "grats/src/tests/fixtures/extend_interface/redefineFiledThatExistsOnConcreteType.ts") + hello: String @metadata +} + +type User implements IPerson { + greeting: String @metadata(argCount: 0) + hello: String @metadata +} +-- TypeScript -- +import { GraphQLSchema, GraphQLInterfaceType, GraphQLString, GraphQLObjectType } from "graphql"; +export function getSchema(): GraphQLSchema { + const IPersonType: GraphQLInterfaceType = new GraphQLInterfaceType({ + name: "IPerson", + fields() { + return { + greeting: { + name: "greeting", + type: GraphQLString + }, + hello: { + name: "hello", + type: GraphQLString + } + }; + } + }); + const UserType: GraphQLObjectType = new GraphQLObjectType({ + name: "User", + fields() { + return { + greeting: { + name: "greeting", + type: GraphQLString + }, + hello: { + name: "hello", + type: GraphQLString + } + }; + }, + interfaces() { + return [IPersonType]; + } + }); + return new GraphQLSchema({ + types: [IPersonType, UserType] + }); +} diff --git a/src/tests/fixtures/extend_interface/redefineFiledThatExistsOnImplementingInterface.invalid.ts.expected b/src/tests/fixtures/extend_interface/redefineFiledThatExistsOnImplementingInterface.invalid.ts.expected deleted file mode 100644 index 21db8a89..00000000 --- a/src/tests/fixtures/extend_interface/redefineFiledThatExistsOnImplementingInterface.invalid.ts.expected +++ /dev/null @@ -1,38 +0,0 @@ ------------------ -INPUT ------------------ -/** @gqlInterface */ -interface IPerson { - name: string; - /** @gqlField */ - hello: string; -} - -/** @gqlField */ -export function greeting(person: IPerson): string { - return `Hello ${person.name}!`; -} - -/** @gqlInterface */ -interface User extends IPerson { - __typename: "User"; - name: string; - /** @gqlField */ - hello: string; - - /** @gqlField */ - greeting(): string; -} - ------------------ -OUTPUT ------------------ -src/tests/fixtures/extend_interface/redefineFiledThatExistsOnImplementingInterface.invalid.ts:21:3 - error: Field "User.greeting" can only be defined once. - -21 greeting(): string; - ~~~~~~~~ - - src/tests/fixtures/extend_interface/redefineFiledThatExistsOnImplementingInterface.invalid.ts:9:17 - 9 export function greeting(person: IPerson): string { - ~~~~~~~~ - Related location diff --git a/src/tests/fixtures/extend_interface/redefineFiledThatExistsOnImplementingInterface.invalid.ts b/src/tests/fixtures/extend_interface/redefineFiledThatExistsOnImplementingInterface.ts similarity index 100% rename from src/tests/fixtures/extend_interface/redefineFiledThatExistsOnImplementingInterface.invalid.ts rename to src/tests/fixtures/extend_interface/redefineFiledThatExistsOnImplementingInterface.ts diff --git a/src/tests/fixtures/extend_interface/redefineFiledThatExistsOnImplementingInterface.ts.expected b/src/tests/fixtures/extend_interface/redefineFiledThatExistsOnImplementingInterface.ts.expected new file mode 100644 index 00000000..902bc01c --- /dev/null +++ b/src/tests/fixtures/extend_interface/redefineFiledThatExistsOnImplementingInterface.ts.expected @@ -0,0 +1,79 @@ +----------------- +INPUT +----------------- +/** @gqlInterface */ +interface IPerson { + name: string; + /** @gqlField */ + hello: string; +} + +/** @gqlField */ +export function greeting(person: IPerson): string { + return `Hello ${person.name}!`; +} + +/** @gqlInterface */ +interface User extends IPerson { + __typename: "User"; + name: string; + /** @gqlField */ + hello: string; + + /** @gqlField */ + greeting(): string; +} + +----------------- +OUTPUT +----------------- +-- SDL -- +interface IPerson { + greeting: String @metadata(argCount: 1, name: "greeting", tsModulePath: "grats/src/tests/fixtures/extend_interface/redefineFiledThatExistsOnImplementingInterface.ts") + hello: String @metadata +} + +interface User implements IPerson { + greeting: String @metadata(argCount: 0) + hello: String @metadata +} +-- TypeScript -- +import { GraphQLSchema, GraphQLInterfaceType, GraphQLString } from "graphql"; +export function getSchema(): GraphQLSchema { + const IPersonType: GraphQLInterfaceType = new GraphQLInterfaceType({ + name: "IPerson", + fields() { + return { + greeting: { + name: "greeting", + type: GraphQLString + }, + hello: { + name: "hello", + type: GraphQLString + } + }; + } + }); + const UserType: GraphQLInterfaceType = new GraphQLInterfaceType({ + name: "User", + fields() { + return { + greeting: { + name: "greeting", + type: GraphQLString + }, + hello: { + name: "hello", + type: GraphQLString + } + }; + }, + interfaces() { + return [IPersonType]; + } + }); + return new GraphQLSchema({ + types: [IPersonType, UserType] + }); +} diff --git a/src/tests/fixtures/extend_type/interfaceFirstArgumentType.ts.expected b/src/tests/fixtures/extend_type/interfaceFirstArgumentType.ts.expected index 4a357890..a9970d0c 100644 --- a/src/tests/fixtures/extend_type/interfaceFirstArgumentType.ts.expected +++ b/src/tests/fixtures/extend_type/interfaceFirstArgumentType.ts.expected @@ -24,7 +24,7 @@ OUTPUT -- SDL -- interface IFoo { bar: String @metadata - greeting: String + greeting: String @metadata(argCount: 1, name: "greeting", tsModulePath: "grats/src/tests/fixtures/extend_type/interfaceFirstArgumentType.ts") } type SomeType { diff --git a/src/tests/fixtures/interfaces/concreteTypeInheritsFieldFromTransitiveInterface.ts b/src/tests/fixtures/interfaces/concreteTypeInheritsFieldFromTransitiveInterface.ts new file mode 100644 index 00000000..ae5b1dce --- /dev/null +++ b/src/tests/fixtures/interfaces/concreteTypeInheritsFieldFromTransitiveInterface.ts @@ -0,0 +1,16 @@ +/** @gqlInterface */ +interface Entity { + /** @gqlField */ + name: string | null; +} + +/** @gqlInterface */ +interface IThing extends Entity { + name: string | null; +} + +/** @gqlType */ +export class Doohickey implements IThing, Entity { + __typename: "Doohickey"; + name: string; +} diff --git a/src/tests/fixtures/interfaces/concreteTypeInheritsFieldFromTransitiveInterface.ts.expected b/src/tests/fixtures/interfaces/concreteTypeInheritsFieldFromTransitiveInterface.ts.expected new file mode 100644 index 00000000..94d22287 --- /dev/null +++ b/src/tests/fixtures/interfaces/concreteTypeInheritsFieldFromTransitiveInterface.ts.expected @@ -0,0 +1,81 @@ +----------------- +INPUT +----------------- +/** @gqlInterface */ +interface Entity { + /** @gqlField */ + name: string | null; +} + +/** @gqlInterface */ +interface IThing extends Entity { + name: string | null; +} + +/** @gqlType */ +export class Doohickey implements IThing, Entity { + __typename: "Doohickey"; + name: string; +} + +----------------- +OUTPUT +----------------- +-- SDL -- +interface Entity { + name: String @metadata +} + +interface IThing implements Entity { + name: String @metadata +} + +type Doohickey implements Entity & IThing { + name: String @metadata +} +-- TypeScript -- +import { GraphQLSchema, GraphQLInterfaceType, GraphQLString, GraphQLObjectType } from "graphql"; +export function getSchema(): GraphQLSchema { + const EntityType: GraphQLInterfaceType = new GraphQLInterfaceType({ + name: "Entity", + fields() { + return { + name: { + name: "name", + type: GraphQLString + } + }; + } + }); + const IThingType: GraphQLInterfaceType = new GraphQLInterfaceType({ + name: "IThing", + fields() { + return { + name: { + name: "name", + type: GraphQLString + } + }; + }, + interfaces() { + return [EntityType]; + } + }); + const DoohickeyType: GraphQLObjectType = new GraphQLObjectType({ + name: "Doohickey", + fields() { + return { + name: { + name: "name", + type: GraphQLString + } + }; + }, + interfaces() { + return [EntityType, IThingType]; + } + }); + return new GraphQLSchema({ + types: [EntityType, IThingType, DoohickeyType] + }); +} diff --git a/src/tests/fixtures/interfaces/concreteTypeInheritsFunctionFieldFromClosestTransitiveInterfacets.ts b/src/tests/fixtures/interfaces/concreteTypeInheritsFunctionFieldFromClosestTransitiveInterfacets.ts new file mode 100644 index 00000000..d776a6a1 --- /dev/null +++ b/src/tests/fixtures/interfaces/concreteTypeInheritsFunctionFieldFromClosestTransitiveInterfacets.ts @@ -0,0 +1,32 @@ +/** @gqlInterface */ +interface Entity { + name: string | null; +} + +/** @gqlField */ +export function greeting(entity: Entity): string { + return `Hello, ${entity.name ?? "World"}!`; +} + +/** @gqlInterface */ +interface IThing extends Entity { + name: string | null; +} + +/** @gqlField greeting */ +export function iThingGreeting(iThing: IThing): string { + return `Hello, ${iThing.name ?? "IThing"}!`; +} + +/** @gqlType */ +export class Doohickey implements IThing, Entity { + __typename: "Doohickey"; + name: string; +} + +// Reverse the order of the interfaces to test that the order doesn't matter +/** @gqlType */ +export class Widget implements Entity, IThing { + __typename: "Widget"; + name: string; +} diff --git a/src/tests/fixtures/interfaces/concreteTypeInheritsFunctionFieldFromClosestTransitiveInterfacets.ts.expected b/src/tests/fixtures/interfaces/concreteTypeInheritsFunctionFieldFromClosestTransitiveInterfacets.ts.expected new file mode 100644 index 00000000..fad6ebb0 --- /dev/null +++ b/src/tests/fixtures/interfaces/concreteTypeInheritsFunctionFieldFromClosestTransitiveInterfacets.ts.expected @@ -0,0 +1,123 @@ +----------------- +INPUT +----------------- +/** @gqlInterface */ +interface Entity { + name: string | null; +} + +/** @gqlField */ +export function greeting(entity: Entity): string { + return `Hello, ${entity.name ?? "World"}!`; +} + +/** @gqlInterface */ +interface IThing extends Entity { + name: string | null; +} + +/** @gqlField greeting */ +export function iThingGreeting(iThing: IThing): string { + return `Hello, ${iThing.name ?? "IThing"}!`; +} + +/** @gqlType */ +export class Doohickey implements IThing, Entity { + __typename: "Doohickey"; + name: string; +} + +// Reverse the order of the interfaces to test that the order doesn't matter +/** @gqlType */ +export class Widget implements Entity, IThing { + __typename: "Widget"; + name: string; +} + +----------------- +OUTPUT +----------------- +-- SDL -- +interface Entity { + greeting: String @metadata(argCount: 1, name: "greeting", tsModulePath: "grats/src/tests/fixtures/interfaces/concreteTypeInheritsFunctionFieldFromClosestTransitiveInterfacets.ts") +} + +interface IThing implements Entity { + greeting: String @metadata(argCount: 1, name: "iThingGreeting", tsModulePath: "grats/src/tests/fixtures/interfaces/concreteTypeInheritsFunctionFieldFromClosestTransitiveInterfacets.ts") +} + +type Doohickey implements Entity & IThing { + greeting: String @metadata(argCount: 1, name: "iThingGreeting", tsModulePath: "grats/src/tests/fixtures/interfaces/concreteTypeInheritsFunctionFieldFromClosestTransitiveInterfacets.ts") +} + +type Widget implements Entity & IThing { + greeting: String @metadata(argCount: 1, name: "greeting", tsModulePath: "grats/src/tests/fixtures/interfaces/concreteTypeInheritsFunctionFieldFromClosestTransitiveInterfacets.ts") +} +-- TypeScript -- +import { iThingGreeting as doohickeyIThingGreetingResolver } from "./concreteTypeInheritsFunctionFieldFromClosestTransitiveInterfacets"; +import { greeting as widgetGreetingResolver } from "./concreteTypeInheritsFunctionFieldFromClosestTransitiveInterfacets"; +import { GraphQLSchema, GraphQLInterfaceType, GraphQLString, GraphQLObjectType } from "graphql"; +export function getSchema(): GraphQLSchema { + const EntityType: GraphQLInterfaceType = new GraphQLInterfaceType({ + name: "Entity", + fields() { + return { + greeting: { + name: "greeting", + type: GraphQLString + } + }; + } + }); + const IThingType: GraphQLInterfaceType = new GraphQLInterfaceType({ + name: "IThing", + fields() { + return { + greeting: { + name: "greeting", + type: GraphQLString + } + }; + }, + interfaces() { + return [EntityType]; + } + }); + const DoohickeyType: GraphQLObjectType = new GraphQLObjectType({ + name: "Doohickey", + fields() { + return { + greeting: { + name: "greeting", + type: GraphQLString, + resolve(source) { + return doohickeyIThingGreetingResolver(source); + } + } + }; + }, + interfaces() { + return [EntityType, IThingType]; + } + }); + const WidgetType: GraphQLObjectType = new GraphQLObjectType({ + name: "Widget", + fields() { + return { + greeting: { + name: "greeting", + type: GraphQLString, + resolve(source) { + return widgetGreetingResolver(source); + } + } + }; + }, + interfaces() { + return [EntityType, IThingType]; + } + }); + return new GraphQLSchema({ + types: [EntityType, IThingType, DoohickeyType, WidgetType] + }); +} diff --git a/src/tests/fixtures/interfaces/concreteTypeInheritsFunctionFieldFromMultipleInterfacets.ts b/src/tests/fixtures/interfaces/concreteTypeInheritsFunctionFieldFromMultipleInterfacets.ts new file mode 100644 index 00000000..5d617470 --- /dev/null +++ b/src/tests/fixtures/interfaces/concreteTypeInheritsFunctionFieldFromMultipleInterfacets.ts @@ -0,0 +1,19 @@ +/** @gqlInterface */ +interface Entity { + /** @gqlField */ + name: string | null; +} + +/** @gqlInterface */ +interface NotEntity {} + +/** @gqlField */ +export function name(_: NotEntity): string | null { + return "Hello"; +} + +/** @gqlType */ +export class Doohickey implements Entity, NotEntity { + __typename: "Doohickey"; + name: string; +} diff --git a/src/tests/fixtures/interfaces/concreteTypeInheritsFunctionFieldFromMultipleInterfacets.ts.expected b/src/tests/fixtures/interfaces/concreteTypeInheritsFunctionFieldFromMultipleInterfacets.ts.expected new file mode 100644 index 00000000..a496c069 --- /dev/null +++ b/src/tests/fixtures/interfaces/concreteTypeInheritsFunctionFieldFromMultipleInterfacets.ts.expected @@ -0,0 +1,81 @@ +----------------- +INPUT +----------------- +/** @gqlInterface */ +interface Entity { + /** @gqlField */ + name: string | null; +} + +/** @gqlInterface */ +interface NotEntity {} + +/** @gqlField */ +export function name(_: NotEntity): string | null { + return "Hello"; +} + +/** @gqlType */ +export class Doohickey implements Entity, NotEntity { + __typename: "Doohickey"; + name: string; +} + +----------------- +OUTPUT +----------------- +-- SDL -- +interface Entity { + name: String @metadata +} + +interface NotEntity { + name: String @metadata(argCount: 1, name: "name", tsModulePath: "grats/src/tests/fixtures/interfaces/concreteTypeInheritsFunctionFieldFromMultipleInterfacets.ts") +} + +type Doohickey implements Entity & NotEntity { + name: String @metadata +} +-- TypeScript -- +import { GraphQLSchema, GraphQLInterfaceType, GraphQLString, GraphQLObjectType } from "graphql"; +export function getSchema(): GraphQLSchema { + const EntityType: GraphQLInterfaceType = new GraphQLInterfaceType({ + name: "Entity", + fields() { + return { + name: { + name: "name", + type: GraphQLString + } + }; + } + }); + const NotEntityType: GraphQLInterfaceType = new GraphQLInterfaceType({ + name: "NotEntity", + fields() { + return { + name: { + name: "name", + type: GraphQLString + } + }; + } + }); + const DoohickeyType: GraphQLObjectType = new GraphQLObjectType({ + name: "Doohickey", + fields() { + return { + name: { + name: "name", + type: GraphQLString + } + }; + }, + interfaces() { + return [EntityType, NotEntityType]; + } + }); + return new GraphQLSchema({ + types: [EntityType, NotEntityType, DoohickeyType] + }); +} diff --git a/src/tests/fixtures/interfaces/concreteTypeInheritsFunctionFieldFromTransitiveInterfacets.ts b/src/tests/fixtures/interfaces/concreteTypeInheritsFunctionFieldFromTransitiveInterfacets.ts new file mode 100644 index 00000000..227418f8 --- /dev/null +++ b/src/tests/fixtures/interfaces/concreteTypeInheritsFunctionFieldFromTransitiveInterfacets.ts @@ -0,0 +1,20 @@ +/** @gqlInterface */ +interface Entity { + name: string | null; +} + +/** @gqlField */ +export function greeting(entity: Entity): string { + return `Hello, ${entity.name ?? "World"}!`; +} + +/** @gqlInterface */ +interface IThing extends Entity { + name: string | null; +} + +/** @gqlType */ +export class Doohickey implements IThing, Entity { + __typename: "Doohickey"; + name: string; +} diff --git a/src/tests/fixtures/interfaces/concreteTypeInheritsFunctionFieldFromTransitiveInterfacets.ts.expected b/src/tests/fixtures/interfaces/concreteTypeInheritsFunctionFieldFromTransitiveInterfacets.ts.expected new file mode 100644 index 00000000..4dfc9ffd --- /dev/null +++ b/src/tests/fixtures/interfaces/concreteTypeInheritsFunctionFieldFromTransitiveInterfacets.ts.expected @@ -0,0 +1,89 @@ +----------------- +INPUT +----------------- +/** @gqlInterface */ +interface Entity { + name: string | null; +} + +/** @gqlField */ +export function greeting(entity: Entity): string { + return `Hello, ${entity.name ?? "World"}!`; +} + +/** @gqlInterface */ +interface IThing extends Entity { + name: string | null; +} + +/** @gqlType */ +export class Doohickey implements IThing, Entity { + __typename: "Doohickey"; + name: string; +} + +----------------- +OUTPUT +----------------- +-- SDL -- +interface Entity { + greeting: String @metadata(argCount: 1, name: "greeting", tsModulePath: "grats/src/tests/fixtures/interfaces/concreteTypeInheritsFunctionFieldFromTransitiveInterfacets.ts") +} + +interface IThing implements Entity { + greeting: String @metadata(argCount: 1, name: "greeting", tsModulePath: "grats/src/tests/fixtures/interfaces/concreteTypeInheritsFunctionFieldFromTransitiveInterfacets.ts") +} + +type Doohickey implements Entity & IThing { + greeting: String @metadata(argCount: 1, name: "greeting", tsModulePath: "grats/src/tests/fixtures/interfaces/concreteTypeInheritsFunctionFieldFromTransitiveInterfacets.ts") +} +-- TypeScript -- +import { greeting as doohickeyGreetingResolver } from "./concreteTypeInheritsFunctionFieldFromTransitiveInterfacets"; +import { GraphQLSchema, GraphQLInterfaceType, GraphQLString, GraphQLObjectType } from "graphql"; +export function getSchema(): GraphQLSchema { + const EntityType: GraphQLInterfaceType = new GraphQLInterfaceType({ + name: "Entity", + fields() { + return { + greeting: { + name: "greeting", + type: GraphQLString + } + }; + } + }); + const IThingType: GraphQLInterfaceType = new GraphQLInterfaceType({ + name: "IThing", + fields() { + return { + greeting: { + name: "greeting", + type: GraphQLString + } + }; + }, + interfaces() { + return [EntityType]; + } + }); + const DoohickeyType: GraphQLObjectType = new GraphQLObjectType({ + name: "Doohickey", + fields() { + return { + greeting: { + name: "greeting", + type: GraphQLString, + resolve(source) { + return doohickeyGreetingResolver(source); + } + } + }; + }, + interfaces() { + return [EntityType, IThingType]; + } + }); + return new GraphQLSchema({ + types: [EntityType, IThingType, DoohickeyType] + }); +} diff --git a/src/tests/fixtures/interfaces/concreteTypeRefinesInterfaceFieldWithAlternateMethod.ts b/src/tests/fixtures/interfaces/concreteTypeRefinesInterfaceFieldWithAlternateMethod.ts new file mode 100644 index 00000000..ca5dd94e --- /dev/null +++ b/src/tests/fixtures/interfaces/concreteTypeRefinesInterfaceFieldWithAlternateMethod.ts @@ -0,0 +1,18 @@ +// { "nullableByDefault": false } +/** @gqlInterface */ +interface IThing { + /** @gqlField */ + name: string | null; +} + +/** @gqlType */ +export class Doohickey implements IThing { + __typename: "Doohickey"; + name: string; + /** + * @gqlField name + */ + someOtherMethod(): string { + return this.name + "!"; + } +} diff --git a/src/tests/fixtures/interfaces/concreteTypeRefinesInterfaceFieldWithAlternateMethod.ts.expected b/src/tests/fixtures/interfaces/concreteTypeRefinesInterfaceFieldWithAlternateMethod.ts.expected new file mode 100644 index 00000000..77516f56 --- /dev/null +++ b/src/tests/fixtures/interfaces/concreteTypeRefinesInterfaceFieldWithAlternateMethod.ts.expected @@ -0,0 +1,68 @@ +----------------- +INPUT +----------------- +// { "nullableByDefault": false } +/** @gqlInterface */ +interface IThing { + /** @gqlField */ + name: string | null; +} + +/** @gqlType */ +export class Doohickey implements IThing { + __typename: "Doohickey"; + name: string; + /** + * @gqlField name + */ + someOtherMethod(): string { + return this.name + "!"; + } +} + +----------------- +OUTPUT +----------------- +-- SDL -- +interface IThing { + name: String @metadata +} + +type Doohickey implements IThing { + name: String! @metadata(argCount: 0, name: "someOtherMethod") +} +-- TypeScript -- +import { GraphQLSchema, GraphQLInterfaceType, GraphQLString, GraphQLObjectType, GraphQLNonNull } from "graphql"; +export function getSchema(): GraphQLSchema { + const IThingType: GraphQLInterfaceType = new GraphQLInterfaceType({ + name: "IThing", + fields() { + return { + name: { + name: "name", + type: GraphQLString + } + }; + } + }); + const DoohickeyType: GraphQLObjectType = new GraphQLObjectType({ + name: "Doohickey", + fields() { + return { + name: { + name: "name", + type: new GraphQLNonNull(GraphQLString), + resolve(source, args, context, info) { + return source.someOtherMethod(source, args, context, info); + } + } + }; + }, + interfaces() { + return [IThingType]; + } + }); + return new GraphQLSchema({ + types: [IThingType, DoohickeyType] + }); +} diff --git a/src/tests/fixtures/interfaces/concreteTypeRefinesInterfaceFieldWithNewDescription.ts b/src/tests/fixtures/interfaces/concreteTypeRefinesInterfaceFieldWithNewDescription.ts new file mode 100644 index 00000000..09381e41 --- /dev/null +++ b/src/tests/fixtures/interfaces/concreteTypeRefinesInterfaceFieldWithNewDescription.ts @@ -0,0 +1,18 @@ +/** @gqlInterface */ +interface IThing { + /** + * This description is on the interface type. + * @gqlField + */ + name: string; +} + +/** @gqlType */ +export class Doohickey implements IThing { + __typename: "Doohickey"; + /** + * This description is on the concrete type. + * @gqlField + */ + name: string; +} diff --git a/src/tests/fixtures/interfaces/concreteTypeRefinesInterfaceFieldWithNewDescription.ts.expected b/src/tests/fixtures/interfaces/concreteTypeRefinesInterfaceFieldWithNewDescription.ts.expected new file mode 100644 index 00000000..e49550b8 --- /dev/null +++ b/src/tests/fixtures/interfaces/concreteTypeRefinesInterfaceFieldWithNewDescription.ts.expected @@ -0,0 +1,69 @@ +----------------- +INPUT +----------------- +/** @gqlInterface */ +interface IThing { + /** + * This description is on the interface type. + * @gqlField + */ + name: string; +} + +/** @gqlType */ +export class Doohickey implements IThing { + __typename: "Doohickey"; + /** + * This description is on the concrete type. + * @gqlField + */ + name: string; +} + +----------------- +OUTPUT +----------------- +-- SDL -- +interface IThing { + """This description is on the interface type.""" + name: String @metadata +} + +type Doohickey implements IThing { + """This description is on the concrete type.""" + name: String @metadata +} +-- TypeScript -- +import { GraphQLSchema, GraphQLInterfaceType, GraphQLString, GraphQLObjectType } from "graphql"; +export function getSchema(): GraphQLSchema { + const IThingType: GraphQLInterfaceType = new GraphQLInterfaceType({ + name: "IThing", + fields() { + return { + name: { + description: "This description is on the interface type.", + name: "name", + type: GraphQLString + } + }; + } + }); + const DoohickeyType: GraphQLObjectType = new GraphQLObjectType({ + name: "Doohickey", + fields() { + return { + name: { + description: "This description is on the concrete type.", + name: "name", + type: GraphQLString + } + }; + }, + interfaces() { + return [IThingType]; + } + }); + return new GraphQLSchema({ + types: [IThingType, DoohickeyType] + }); +} diff --git a/src/tests/fixtures/interfaces/concreteTypeRefinesInterfaceFieldWithSubtype.ts b/src/tests/fixtures/interfaces/concreteTypeRefinesInterfaceFieldWithSubtype.ts new file mode 100644 index 00000000..b0aa3ea3 --- /dev/null +++ b/src/tests/fixtures/interfaces/concreteTypeRefinesInterfaceFieldWithSubtype.ts @@ -0,0 +1,13 @@ +// { "nullableByDefault": false } +/** @gqlInterface */ +interface IThing { + /** @gqlField */ + name: string | null; +} + +/** @gqlType */ +export class Doohickey implements IThing { + __typename: "Doohickey"; + /** @gqlField */ + name: string; +} diff --git a/src/tests/fixtures/interfaces/concreteTypeRefinesInterfaceFieldWithSubtype.ts.expected b/src/tests/fixtures/interfaces/concreteTypeRefinesInterfaceFieldWithSubtype.ts.expected new file mode 100644 index 00000000..6d3cb191 --- /dev/null +++ b/src/tests/fixtures/interfaces/concreteTypeRefinesInterfaceFieldWithSubtype.ts.expected @@ -0,0 +1,60 @@ +----------------- +INPUT +----------------- +// { "nullableByDefault": false } +/** @gqlInterface */ +interface IThing { + /** @gqlField */ + name: string | null; +} + +/** @gqlType */ +export class Doohickey implements IThing { + __typename: "Doohickey"; + /** @gqlField */ + name: string; +} + +----------------- +OUTPUT +----------------- +-- SDL -- +interface IThing { + name: String @metadata +} + +type Doohickey implements IThing { + name: String! @metadata +} +-- TypeScript -- +import { GraphQLSchema, GraphQLInterfaceType, GraphQLString, GraphQLObjectType, GraphQLNonNull } from "graphql"; +export function getSchema(): GraphQLSchema { + const IThingType: GraphQLInterfaceType = new GraphQLInterfaceType({ + name: "IThing", + fields() { + return { + name: { + name: "name", + type: GraphQLString + } + }; + } + }); + const DoohickeyType: GraphQLObjectType = new GraphQLObjectType({ + name: "Doohickey", + fields() { + return { + name: { + name: "name", + type: new GraphQLNonNull(GraphQLString) + } + }; + }, + interfaces() { + return [IThingType]; + } + }); + return new GraphQLSchema({ + types: [IThingType, DoohickeyType] + }); +} diff --git a/src/tests/fixtures/interfaces/interfaceFieldsInheritedByConcreteType.ts b/src/tests/fixtures/interfaces/interfaceFieldsInheritedByConcreteType.ts new file mode 100644 index 00000000..42637b77 --- /dev/null +++ b/src/tests/fixtures/interfaces/interfaceFieldsInheritedByConcreteType.ts @@ -0,0 +1,11 @@ +/** @gqlInterface */ +interface IThing { + /** @gqlField */ + name: string; +} + +/** @gqlType */ +class Doohickey implements IThing { + __typename: "Doohickey"; + name: string; +} diff --git a/src/tests/fixtures/interfaces/interfaceFieldsInheritedByConcreteType.ts.expected b/src/tests/fixtures/interfaces/interfaceFieldsInheritedByConcreteType.ts.expected new file mode 100644 index 00000000..3752abaa --- /dev/null +++ b/src/tests/fixtures/interfaces/interfaceFieldsInheritedByConcreteType.ts.expected @@ -0,0 +1,58 @@ +----------------- +INPUT +----------------- +/** @gqlInterface */ +interface IThing { + /** @gqlField */ + name: string; +} + +/** @gqlType */ +class Doohickey implements IThing { + __typename: "Doohickey"; + name: string; +} + +----------------- +OUTPUT +----------------- +-- SDL -- +interface IThing { + name: String @metadata +} + +type Doohickey implements IThing { + name: String @metadata +} +-- TypeScript -- +import { GraphQLSchema, GraphQLInterfaceType, GraphQLString, GraphQLObjectType } from "graphql"; +export function getSchema(): GraphQLSchema { + const IThingType: GraphQLInterfaceType = new GraphQLInterfaceType({ + name: "IThing", + fields() { + return { + name: { + name: "name", + type: GraphQLString + } + }; + } + }); + const DoohickeyType: GraphQLObjectType = new GraphQLObjectType({ + name: "Doohickey", + fields() { + return { + name: { + name: "name", + type: GraphQLString + } + }; + }, + interfaces() { + return [IThingType]; + } + }); + return new GraphQLSchema({ + types: [IThingType, DoohickeyType] + }); +} diff --git a/src/transforms/addInterfaceFields.ts b/src/transforms/addInterfaceFields.ts index 6f27b4c6..823852f0 100644 --- a/src/transforms/addInterfaceFields.ts +++ b/src/transforms/addInterfaceFields.ts @@ -8,14 +8,12 @@ import { gqlRelated, } from "../utils/DiagnosticError"; import { err, ok } from "../utils/Result"; -import { InterfaceMap, computeInterfaceMap } from "../InterfaceGraph"; import { extend } from "../utils/helpers"; import { FIELD_TAG } from "../Extractor"; import { AbstractFieldDefinitionNode, GratsDefinitionNode, } from "../GraphQLConstructor"; -import { FIELD_METADATA_DIRECTIVE } from "../metadataDirectives"; /** * Grats allows you to define GraphQL fields on TypeScript interfaces using @@ -32,15 +30,9 @@ export function addInterfaceFields( const newDocs: DefinitionNode[] = []; const errors: ts.DiagnosticWithLocation[] = []; - const interfaceGraph = computeInterfaceMap(ctx, docs); - for (const doc of docs) { if (doc.kind === "AbstractFieldDefinition") { - const abstractDocResults = addAbstractFieldDefinition( - ctx, - doc, - interfaceGraph, - ); + const abstractDocResults = addAbstractFieldDefinition(ctx, doc); if (abstractDocResults.kind === "ERROR") { extend(errors, abstractDocResults.err); } else { @@ -61,7 +53,6 @@ export function addInterfaceFields( function addAbstractFieldDefinition( ctx: TypeContext, doc: AbstractFieldDefinitionNode, - interfaceGraph: InterfaceMap, ): DiagnosticsResult { const newDocs: DefinitionNode[] = []; const definitionResult = ctx.getNameDefinition(doc.onType); @@ -74,7 +65,6 @@ function addAbstractFieldDefinition( switch (nameDefinition.kind) { case "TYPE": - // Extending a type, is just adding a field to it. newDocs.push({ kind: Kind.OBJECT_TYPE_EXTENSION, name: doc.onType, @@ -83,45 +73,11 @@ function addAbstractFieldDefinition( }); break; case "INTERFACE": { - // Extending an interface is a bit more complicated. We need to add the field - // to the interface, and to each type that implements the interface. - - // The interface field definition is not executable, so we don't - // need to annotate it with the details of the implementation. - const directives = doc.field.directives?.filter((directive) => { - return directive.name.value !== FIELD_METADATA_DIRECTIVE; - }); newDocs.push({ kind: Kind.INTERFACE_TYPE_EXTENSION, name: doc.onType, - fields: [{ ...doc.field, directives }], + fields: [doc.field], }); - - for (const implementor of interfaceGraph.get(nameDefinition.name.value)) { - const name = { - kind: Kind.NAME, - value: implementor.name, - loc: doc.loc, // Bit of a lie, but I don't see a better option. - } as const; - switch (implementor.kind) { - case "TYPE": - newDocs.push({ - kind: Kind.OBJECT_TYPE_EXTENSION, - name, - fields: [doc.field], - loc: doc.loc, - }); - break; - case "INTERFACE": - newDocs.push({ - kind: Kind.INTERFACE_TYPE_EXTENSION, - name, - fields: [{ ...doc.field, directives }], - loc: doc.loc, - }); - break; - } - } break; } default: { diff --git a/src/transforms/mergeExtensions.ts b/src/transforms/mergeExtensions.ts index 48a4320f..7498fb68 100644 --- a/src/transforms/mergeExtensions.ts +++ b/src/transforms/mergeExtensions.ts @@ -1,86 +1,152 @@ -import { DocumentNode, FieldDefinitionNode, visit } from "graphql"; -import { extend } from "../utils/helpers"; +import { + DocumentNode, + FieldDefinitionNode, + NamedTypeNode, + visit, +} from "graphql"; +import { DefaultMap, extend } from "../utils/helpers"; /** * Takes every example of `extend type Foo` and `extend interface Foo` and * merges them into the original type/interface definition. + * + * Also, propagates fields from interfaces to their implementing types. */ export function mergeExtensions(doc: DocumentNode): DocumentNode { - const fields = new MultiMap(); + const merger = new ExtensionMerger(); + return merger.mergeExtensions(doc); +} - // Collect all the fields from the extensions and trim them from the AST. - const sansExtensions = visit(doc, { - ObjectTypeExtension(t) { - if (t.directives != null || t.interfaces != null) { - throw new Error("Unexpected directives or interfaces on Extension"); - } - fields.extend(t.name.value, t.fields); - return null; - }, - InterfaceTypeExtension(t) { - if (t.directives != null || t.interfaces != null) { - throw new Error("Unexpected directives or interfaces on Extension"); - } - fields.extend(t.name.value, t.fields); - return null; - }, - // Grats does not create these extension types - ScalarTypeExtension(_) { - throw new Error("Unexpected ScalarTypeExtension"); - }, - EnumTypeExtension(_) { - throw new Error("Unexpected EnumTypeExtension"); - }, - SchemaExtension(_) { - throw new Error("Unexpected SchemaExtension"); - }, - }); +class ExtensionMerger { + _fields: DefaultMap; + _interfaceMap: DefaultMap>; + constructor() { + this._fields = new DefaultMap(() => []); + this._interfaceMap = new DefaultMap(() => new Set()); + } - // Merge collected extension fields into the original type/interface definition. - return visit(sansExtensions, { - ObjectTypeDefinition(t) { - const extensions = fields.get(t.name.value); - if (t.fields == null) { - return { ...t, fields: extensions }; - } - return { ...t, fields: [...t.fields, ...extensions] }; - }, - InterfaceTypeDefinition(t) { - const extensions = fields.get(t.name.value); - if (t.fields == null) { - return { ...t, fields: extensions }; + mergeExtensions(doc: DocumentNode): DocumentNode { + const sansExtensions = this.collectFieldsAndExtensions(doc); + + this.propagateMissingFields(sansExtensions); + + // Merge collected extension fields into the original type/interface definition. + return visit(sansExtensions, { + ObjectTypeDefinition: (t) => { + return { ...t, fields: this._fields.get(t.name.value) }; + }, + InterfaceTypeDefinition: (t) => { + return { ...t, fields: this._fields.get(t.name.value) }; + }, + }); + } + + // Propagate fields from interfaces to their implementing types. + // GraphQL expects all types and interfaces to explicitly define all fields, including + // those required by interfaces they implement. They way Grats works, if a field is defined + // on an interface, that field is provably present on all implementing types. + propagateMissingFields(sansExtensions: DocumentNode) { + const processedInterfaces = new Set(); + + const extendType = (name: string): void => { + const ownFields = this._fields.get(name); + for (const iface of this._interfaceMap.get(name)) { + // First make sure the interface has been fully populated + extendInterface(iface); + const ifaceFields = this._fields.get(iface); + for (const field of ifaceFields) { + // TODO: This is not very efficient. Every interface's field requires an O(n) search. + // Modeling fields a a name => field map would be more efficient but + // ends up implicitly removing duplicates, which should be reported as + // an error by later phases of the pipeline. + if (!ownFields.some((f) => f.name.value === field.name.value)) { + ownFields.push(field); + } + } } - return { ...t, fields: [...t.fields, ...extensions] }; - }, - }); -} + }; -// Map a key to an array of values. -class MultiMap { - private readonly map = new Map(); + const extendInterface = (name: string): void => { + // Avoid duplicate processing and infinite recursion + if (processedInterfaces.has(name)) { + return; + } + // This does not _correctly_ handle recursive interfaces, but that's okay + // since some other validation pass should detect that. The main issues is + // to avoid infinite recursion leading to a stack overflow. + processedInterfaces.add(name); + extendType(name); + }; - push(key: K, value: V): void { - let existing = this.map.get(key); - if (existing == null) { - existing = []; - this.map.set(key, existing); + for (const doc of sansExtensions.definitions) { + switch (doc.kind) { + case "ObjectTypeDefinition": + extendType(doc.name.value); + break; + case "InterfaceTypeDefinition": + extendInterface(doc.name.value); + break; + } } - existing.push(value); } - extend(key: K, values?: readonly V[]): void { - if (values == null) { - return; - } - let existing = this.map.get(key); - if (existing == null) { - existing = []; - this.map.set(key, existing); + // Collect all the fields and interfaces from the extensions and trim them from the AST. + collectFieldsAndExtensions(doc: DocumentNode): DocumentNode { + return visit(doc, { + ObjectTypeExtension: (t) => { + if (t.directives != null || t.interfaces != null) { + throw new Error("Unexpected directives or interfaces on Extension"); + } + this.addFields(t.name.value, t.fields); + return null; + }, + InterfaceTypeExtension: (t) => { + if (t.directives != null || t.interfaces != null) { + throw new Error("Unexpected directives or interfaces on Extension"); + } + this.addFields(t.name.value, t.fields); + return null; + }, + ObjectTypeDefinition: (t) => { + this.addInterfaces(t.name.value, t.interfaces); + this.addFields(t.name.value, t.fields); + return t; + }, + InterfaceTypeDefinition: (t) => { + this.addInterfaces(t.name.value, t.interfaces); + this.addFields(t.name.value, t.fields); + return t; + }, + // Grats does not create these extension types + ScalarTypeExtension(_) { + throw new Error("Unexpected ScalarTypeExtension"); + }, + EnumTypeExtension(_) { + throw new Error("Unexpected EnumTypeExtension"); + }, + SchemaExtension(_) { + throw new Error("Unexpected SchemaExtension"); + }, + }); + } + + addFields( + name: string, + fields: readonly FieldDefinitionNode[] | undefined, + ): void { + if (fields != null) { + extend(this._fields.get(name), fields); } - extend(existing, values); } - get(key: K): V[] { - return this.map.get(key) ?? []; + addInterfaces( + name: string, + interfaces: readonly NamedTypeNode[] | undefined, + ): void { + if (interfaces != null) { + for (const i of interfaces) { + this._interfaceMap.get(name).add(i.name.value); + } + } } } diff --git a/website/docs/04-docblock-tags/05-interfaces.mdx b/website/docs/04-docblock-tags/05-interfaces.mdx index e1f8efe1..2be32394 100644 --- a/website/docs/04-docblock-tags/05-interfaces.mdx +++ b/website/docs/04-docblock-tags/05-interfaces.mdx @@ -3,6 +3,7 @@ import InterfaceDeclaration from "!!raw-loader!./snippets/04-interface-declarati import NodeInterface from "!!raw-loader!./snippets/04-merged-interface-renaming.out"; import InterfaceImplementingInterface from "!!raw-loader!./snippets/04-interface-implement-interface.out"; import InterfaceFieldCommonImpl from "!!raw-loader!./snippets/04-interface-field-common-impl.out"; +import InterfaceOverrideFieldImpl from "!!raw-loader!./snippets/04-interface-override-field-impl.out"; # Interfaces @@ -32,11 +33,13 @@ Which will generate the following GraphQL schema: ---- +## Overriding Interface Fields -:::note -Each implementor of an interface must declare define all the fields required by the interface with `/** @gqlField */`. This means that if you have an interface that implements another interface, you must define all the fields required by both interfaces. -::: +Unlike GraphQL, when you add a field to an interface with `/** @gqlField */`, the field will be automatically be added to the GraphQL definitions of all the implementors of the interface. + +If you wish to override the field in a specific implementor, you can add a `/** @gqlField */` with the same name to the implementor. + + ## Merged Interfaces diff --git a/website/docs/04-docblock-tags/snippets/04-interface-override-field-impl.grats.ts b/website/docs/04-docblock-tags/snippets/04-interface-override-field-impl.grats.ts new file mode 100644 index 00000000..6756d804 --- /dev/null +++ b/website/docs/04-docblock-tags/snippets/04-interface-override-field-impl.grats.ts @@ -0,0 +1,22 @@ +/** @gqlInterface */ +interface Person { + /** @gqlField */ + name: string; +} + +/** @gqlType */ +class User implements Person { + __typename = "User"; + name: string; + + // highlight-start + /** + * For `User` this method will be used instead of the `name` property. + * + * @gqlField name + */ + userSpecificName(): string { + return `User: this.name`; + } + // highlight-end +} diff --git a/website/docs/04-docblock-tags/snippets/04-interface-override-field-impl.out b/website/docs/04-docblock-tags/snippets/04-interface-override-field-impl.out new file mode 100644 index 00000000..f010a419 --- /dev/null +++ b/website/docs/04-docblock-tags/snippets/04-interface-override-field-impl.out @@ -0,0 +1,68 @@ +/** @gqlInterface */ +interface Person { + /** @gqlField */ + name: string; +} + +/** @gqlType */ +class User implements Person { + __typename = "User"; + name: string; + + // highlight-start + /** + * For `User` this method will be used instead of the `name` property. + * + * @gqlField name + */ + userSpecificName(): string { + return `User: this.name`; + } + // highlight-end +} + +=== SNIP === +interface Person { + name: String +} + +type User implements Person { + """For `User` this method will be used instead of the `name` property.""" + name: String +} +=== SNIP === +import { GraphQLSchema, GraphQLInterfaceType, GraphQLString, GraphQLObjectType } from "graphql"; +export function getSchema(): GraphQLSchema { + const PersonType: GraphQLInterfaceType = new GraphQLInterfaceType({ + name: "Person", + fields() { + return { + name: { + name: "name", + type: GraphQLString + } + }; + } + }); + const UserType: GraphQLObjectType = new GraphQLObjectType({ + name: "User", + fields() { + return { + name: { + description: "For `User` this method will be used instead of the `name` property.", + name: "name", + type: GraphQLString, + resolve(source, args, context, info) { + return source.userSpecificName(source, args, context, info); + } + } + }; + }, + interfaces() { + return [PersonType]; + } + }); + return new GraphQLSchema({ + types: [PersonType, UserType] + }); +}