From 18eab5d66cd95ea2e14e7101f63154d5f9182daa Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Micha=C5=82=20Lytek?= Date: Mon, 18 Feb 2019 20:07:38 +0100 Subject: [PATCH] docs(interfaces-inheritance): split docs to separate pages --- ...aces-and-inheritance.md => inheritance.md} | 68 ++++++------------- docs/interfaces.md | 50 ++++++++++++++ website/i18n/en.json | 7 +- website/sidebars.json | 15 ++-- 4 files changed, 84 insertions(+), 56 deletions(-) rename docs/{interfaces-and-inheritance.md => inheritance.md} (59%) create mode 100644 docs/interfaces.md diff --git a/docs/interfaces-and-inheritance.md b/docs/inheritance.md similarity index 59% rename from docs/interfaces-and-inheritance.md rename to docs/inheritance.md index 53dd0353e..ea4a35cb9 100644 --- a/docs/interfaces-and-inheritance.md +++ b/docs/inheritance.md @@ -1,53 +1,17 @@ --- -title: Interfaces and inheritance +title: Inheritance --- The main idea of TypeGraphQL is to create GraphQL types based on TypeScript classes. -In object-oriented programming it is common to create interfaces which describes the contract that classes implementing them has to fulfill. We also compose classes using inheritance. Hence TypeGraphQL supports both GraphQL interfaces as well as composing type definitions by extending the classes. - -## Interfaces -TypeScript has first class support for interfaces. Unfortunately, they only exist at compile-time, so we can't use them to build GraphQL schema at runtime by using decorators. - -Luckily, we can use an abstract class for this purpose. It behaves almost like an interface (can't be "newed", can be implemented by class), it just won't stop developers from implementing a method or initializing a field. But as long as we treat it like an interface, we can safely use it. - -So, how do you create GraphQL interface definition? We create an abstract class and decorate it with `@InterfaceType()`. The rest is exactly the same as with object types: we use `@Field` to declare the shape of the type: - -```typescript -@InterfaceType() -abstract class IPerson { - @Field(type => ID) - id: string; - - @Field() - name: string; - - @Field(type => Int) - age: number; -} -``` - -Then we can use this "interface" in object type class definition: - -```typescript -@ObjectType({ implements: IPerson }) -class Person implements IPerson { - id: string; - name: string; - age: number; -} -``` - -The only difference is that we have to let TypeGraphQL know that this `ObjectType` is implementing the `InterfaceType`. We do it by passing the param `({ implements: IPerson })` to the decorator. If we implemented more interfaces, we would pass the array of interfaces, like `({ implements: [IPerson, IAnimal, IMachine] })`. - -We can also omit the decorators as the GraphQL types will be copied from the interface definition - this way we don't have to maintain two definitions and just rely on TypeScript type checking of correct interface implementation. - -Be aware that when your object type is implementing GraphQL interface type, __you have to return an instance of the type class__ in your resolvers. Otherwise, `graphql-js` will not be able to detect the underlying GraphQL type correctly. +In object-oriented programming it is common to compose classes using inheritance. Hence TypeGraphQL supports composing type definitions by extending the classes. ## Types inheritance + One of the most known principles of software development is DRY - don't repeat yourself - which tells about avoiding redundancy in our code. -While creating GraphQL API, it's a common pattern to have pagination args in resolvers, like `skip` and `take`. So instead of repeating yourself, you can declare it once: +While creating GraphQL API, it's a common pattern to have pagination args in resolvers, like `skip` and `take`. So instead of repeating yourself, you can declare it once: + ```typescript @ArgsType() class PaginationArgs { @@ -60,6 +24,7 @@ class PaginationArgs { ``` and then reuse it everywhere: + ```typescript @ArgsType() class GetTodosArgs extends PaginationArgs { @@ -69,6 +34,7 @@ class GetTodosArgs extends PaginationArgs { ``` This technique also works with input type classes, as well as with object type classes: + ```typescript @ObjectType() class Person { @@ -86,9 +52,11 @@ class Student extends Person { Note that both the subclass and the parent class must be decorated with the same type of decorator, like `@ObjectType()` in the example `Person -> Student` above. Mixing decorator types across parent and child classes is prohibited and might result in schema building error - you can't e.g decorate the subclass with `@ObjectType()` and the parent with `@InputType()`. ## Resolvers inheritance + The special kind of inheritance in TypeGraphQL is a resolver classes inheritance. This pattern allows you to e.g. create a base CRUD resolver class for your resource/entity, so you don't have to repeat the common boilerplate code all the time. As we need to generate unique query/mutation names, we have to create a factory function for our base class: + ```typescript function createBaseResolver() { abstract class BaseResolver {} @@ -96,9 +64,11 @@ function createBaseResolver() { return BaseResolver; } ``` -Be aware that with some `tsconfig.json` settings you might receive `[ts] Return type of exported function has or is using private name 'BaseResolver'` error - in that case you might need to use `any` as a return type or create a separate class/interface describing the class methods and properties. + +Be aware that with some `tsconfig.json` settings (like `declarations: true`) you might receive `[ts] Return type of exported function has or is using private name 'BaseResolver'` error - in that case you might need to use `any` as a return type or create a separate class/interface describing the class methods and properties. This factory should take a parameter that we can use to generate queries/mutations names, as well as the type that we would return from the resolvers: + ```typescript function createBaseResolver(suffix: string, objectTypeCls: T) { abstract class BaseResolver {} @@ -108,6 +78,7 @@ function createBaseResolver(suffix: string, objectTypeCls: ``` It's very important to mark the `BaseResolver` class using `@Resolver` decorator with `{ isAbstract: true }` option that will prevent throwing error due to registering multiple queries/mutations with the same name. + ```typescript function createBaseResolver(suffix: string, objectTypeCls: T) { @Resolver({ isAbstract: true }) @@ -118,6 +89,7 @@ function createBaseResolver(suffix: string, objectTypeCls: ``` Then we can implement the resolvers methods in the same way as always. The only difference is that we can use `name` decorator option for `@Query`, `@Mutation` and `@Subscription` decorators to overwrite the name that will be emitted in schema: + ```typescript function createBaseResolver(suffix: string, objectTypeCls: T) { @Resolver({ isAbstract: true }) @@ -125,9 +97,7 @@ function createBaseResolver(suffix: string, objectTypeCls: protected items: T[] = []; @Query(type => [objectTypeCls], { name: `getAll${suffix}` }) - async getAll( - @Arg("first", type => Int) first: number, - ): Promise { + async getAll(@Arg("first", type => Int) first: number): Promise { return this.items.slice(0, first); } } @@ -137,6 +107,7 @@ function createBaseResolver(suffix: string, objectTypeCls: ``` After that we can create a specific resolver class that will extend the base resolver class: + ```typescript const PersonBaseResolver = createBaseResolver("person", Person); @@ -147,6 +118,7 @@ export class PersonResolver extends PersonBaseResolver { ``` We can also add specific queries and mutation in our resolver class, just like always: + ```typescript const PersonBaseResolver = createBaseResolver("person", Person); @@ -159,11 +131,13 @@ export class PersonResolver extends PersonBaseResolver { } } ``` + And that's it! We just need to normally register `PersonResolver` in `buildSchema` and the extended resolver will be working correctly. Be aware that if you want to overwrite the query/mutation/subscription from parent resolver class, you need to generate the same schema name (using `name` decorator option or the class method name). It will overwrite the implementation along with GraphQL args and return types. If you only provide different implementation of the inherited method like `getOne`, it won't work. ## Examples -More advanced usage example of interfaces and types inheritance (e.g. with query returning interface type) you can see in [this examples folder](https://github.com/19majkel94/type-graphql/tree/master/examples/interfaces-inheritance). -For more advanced resolvers inheritance example, please go to [the example folder](https://github.com/19majkel94/type-graphql/tree/master/examples/resolvers-inheritance). +More advanced usage example of types inheritance (and interfaces) you can see in [the example folder](https://github.com/19majkel94/type-graphql/tree/master/examples/interfaces-inheritance). + +For more advanced resolvers inheritance example, please go to [this example folder](https://github.com/19majkel94/type-graphql/tree/master/examples/resolvers-inheritance). diff --git a/docs/interfaces.md b/docs/interfaces.md new file mode 100644 index 000000000..4d6d3b528 --- /dev/null +++ b/docs/interfaces.md @@ -0,0 +1,50 @@ +--- +title: Interfaces +--- + +The main idea of TypeGraphQL is to create GraphQL types based on TypeScript classes. + +In object-oriented programming it is common to create interfaces which describes the contract that classes implementing them has to fulfill - hence TypeGraphQL supports defining GraphQL interfaces. + +## How to? + +TypeScript has first class support for interfaces. Unfortunately, they only exist at compile-time, so we can't use them to build GraphQL schema at runtime by using decorators. + +Luckily, we can use an abstract class for this purpose. It behaves almost like an interface (can't be "newed", can be implemented by class), it just won't stop developers from implementing a method or initializing a field. But as long as we treat it like an interface, we can safely use it. + +So, how do you create GraphQL interface definition? We create an abstract class and decorate it with `@InterfaceType()`. The rest is exactly the same as with object types: we use `@Field` to declare the shape of the type: + +```typescript +@InterfaceType() +abstract class IPerson { + @Field(type => ID) + id: string; + + @Field() + name: string; + + @Field(type => Int) + age: number; +} +``` + +Then we can use this "interface" in object type class definition: + +```typescript +@ObjectType({ implements: IPerson }) +class Person implements IPerson { + id: string; + name: string; + age: number; +} +``` + +The only difference is that we have to let TypeGraphQL know that this `ObjectType` is implementing the `InterfaceType`. We do it by passing the param `({ implements: IPerson })` to the decorator. If we implemented more interfaces, we would pass the array of interfaces, like `({ implements: [IPerson, IAnimal, IMachine] })`. + +We can also omit the decorators as the GraphQL types will be copied from the interface definition - this way we don't have to maintain two definitions and just rely on TypeScript type checking of correct interface implementation. + +Be aware that when your object type is implementing GraphQL interface type, **you have to return an instance of the type class** in your resolvers. Otherwise, `graphql-js` will not be able to detect the underlying GraphQL type correctly. + +## Examples + +More advanced usage example of interfaces (and types inheritance), e.g. with query returning interface type, you can see in [this examples folder](https://github.com/19majkel94/type-graphql/tree/master/examples/interfaces-inheritance). diff --git a/website/i18n/en.json b/website/i18n/en.json index 51878a8d4..bde870468 100644 --- a/website/i18n/en.json +++ b/website/i18n/en.json @@ -36,11 +36,14 @@ "getting-started": { "title": "Getting started" }, + "inheritance": { + "title": "Inheritance" + }, "installation": { "title": "Installation" }, - "interfaces-and-inheritance": { - "title": "Interfaces and inheritance" + "interfaces": { + "title": "Interfaces" }, "introduction": { "title": "Introduction", diff --git a/website/sidebars.json b/website/sidebars.json index 1c88ca9b7..e6bf70ba0 100644 --- a/website/sidebars.json +++ b/website/sidebars.json @@ -1,18 +1,19 @@ { "docs": { "Introduction": ["introduction"], - "Beginner guides": ["installation", "getting-started", "types-and-fields", "resolvers", "bootstrap"], - "Advanced guides": [ - "scalars", - "enums", - "unions", - "subscriptions", - "interfaces-and-inheritance" + "Beginner guides": [ + "installation", + "getting-started", + "types-and-fields", + "resolvers", + "bootstrap" ], + "Advanced guides": ["scalars", "enums", "unions", "interfaces", "subscriptions"], "Features": [ "dependency-injection", "authorization", "validation", + "inheritance", "middlewares", "complexity" ],