Scalars
A GraphQL object type has a name and fields, but at some point those fields have to resolve to some concrete data. That’s where the scalar types come in: they represent the leaves of the query (read more here). GraphQL includes the following default types: Int
, Float
, String
, Boolean
and ID
. In addition to these built-in types, you may need to support custom atomic data types (e.g., Date
).
Code first
The code-first approach ships with five scalars in which three of them are simple aliases for the existing GraphQL types.
ID
(alias forGraphQLID
) - represents a unique identifier, often used to refetch an object or as the key for a cacheInt
(alias forGraphQLInt
) - a signed 32‐bit integerFloat
(alias forGraphQLFloat
) - a signed double-precision floating-point valueGraphQLISODateTime
- a date-time string at UTC (used by default to representDate
type)GraphQLTimestamp
- a numeric string which represents time and date as number of milliseconds from start of UNIX epoch
The GraphQLISODateTime
(e.g. 2019-12-03T09:54:33Z
) is used by default to represent the Date
type. To use the GraphQLTimestamp
instead, set the dateScalarMode
of the buildSchemaOptions
object to 'timestamp'
as follows:
GraphQLModule.forRoot({
buildSchemaOptions: {
dateScalarMode: 'timestamp',
}
}),
In addition, you can create custom scalars. For example, to create a Date
scalar, simply create a new class.
import { Scalar, CustomScalar } from '@nestjs/graphql';
import { Kind, ValueNode } from 'graphql';
@Scalar('Date', type => Date)
export class DateScalar implements CustomScalar<number, Date> {
description = 'Date custom scalar type';
parseValue(value: number): Date {
return new Date(value); // value from the client
}
serialize(value: Date): number {
return value.getTime(); // value sent to the client
}
parseLiteral(ast: ValueNode): Date {
if (ast.kind === Kind.INT) {
return new Date(ast.value);
}
return null;
}
}
With this in place, register DateScalar
as a provider.
@Module({
providers: [DateScalar],
})
export class CommonModule {}
Now we can use the Date
type in our classes.
@Field()
creationDate: Date;
Schema first
To define a custom scalar (read more about scalars here), create a type definition and a dedicated resolver. Here (as in the official documentation), we’ll use the graphql-type-json
package for demonstration purposes. This npm package defines a JSON
GraphQL scalar type.
Start by installing the package:
$ npm i --save graphql-type-json
Once the package is installed, we pass a custom resolver to the forRoot()
method:
import * as GraphQLJSON from 'graphql-type-json';
@Module({
imports: [
GraphQLModule.forRoot({
typePaths: ['./**/*.graphql'],
resolvers: { JSON: GraphQLJSON },
}),
],
})
export class ApplicationModule {}
Now we can use the JSON
scalar in our type definitions:
scalar JSON
type Foo {
field: JSON
}
Another method to define a scalar type is to create a simple class. Assume we want to enhance our schema with the Date
type.
import { Scalar, CustomScalar } from '@nestjs/graphql';
import { Kind, ValueNode } from 'graphql';
@Scalar('Date')
export class DateScalar implements CustomScalar<number, Date> {
description = 'Date custom scalar type';
parseValue(value: number): Date {
return new Date(value); // value from the client
}
serialize(value: Date): number {
return value.getTime(); // value sent to the client
}
parseLiteral(ast: ValueNode): Date {
if (ast.kind === Kind.INT) {
return new Date(ast.value);
}
return null;
}
}
With this in place, register DateScalar
as a provider.
@Module({
providers: [DateScalar],
})
export class CommonModule {}
Now we can use the Date
scalar in type definitions.
scalar Date