Security News
The Risks of Misguided Research in Supply Chain Security
Snyk's use of malicious npm packages for research raises ethical concerns, highlighting risks in public deployment, data exfiltration, and unauthorized testing.
**This is currently a proof of concept. It won't yet work on any real projects.**
This is currently a proof of concept. It won't yet work on any real projects.
Grats is a tool for statically infering GraphQL schema from your vanilla TypeScript code.
Just write your types and resolvers as regular TypeScript and annotate your types and fields with simple JSDoc tags. From there, Grats can extract your GraphQL schema automatically by statically analyzing your code and its types. No convoluted directive APIs to remember. No need to define your Schema at runtime with verbose builder APIs.
By making your TypeScript implementation the source of truth, you entirely remove the question of mismatches between your implementation and your GraphQL schema definition. Your implementation is the schema definition!
/** @gqlType */
export default class Query {
/** @gqlField */
me(): UserResolver {
return new UserResolver();
}
/**
* @gqlField
* @deprecated Please use `me` instead.
*/
viewer(): UserResolver {
return new UserResolver();
}
}
/**
* A user in our kick-ass system!
* @gqlType User
*/
class UserResolver {
/** @gqlField */
name: string = 'Alice';
/** @gqlField */
greeting(args: { salutation: string }): string {
return `${args.salutation}, ${this.name}`;
}
}
Extracts the following GraphQL schema:
type Query {
me: User
viewer: User @deprecated(reason: "Please use `me` instead.")
}
"""A user in our kick-ass system!"""
type User {
name: String
greeting(salutation: String!): String
}
Give it a try in the online playground!
For dev mode or small projects, Grats offers a runtime extraction mode. This is the easiest way to get started with Grats, although you may find that it causes a slow startup time. For larger projects, you probably want to use the build mode (documentation to come).
npm install express express-graphql grats
Ensure your project has a tsconfig.json
file.
import * as express from "express";
import { graphqlHTTP } from "express-graphql";
import { extractGratsSchemaAtRuntime } from "grats";
/** @gqlType */
class Query {
/** @gqlField */
hello(): string {
return "Hello world!";
}
}
const app = express();
// At runtime Grats will parse your TypeScript project (including this file!) and
// extract the GraphQL schema.
const schema = extractGratsSchemaAtRuntime({
emitSchemaFile: "./schema.graphql",
});
app.use(
"/graphql",
graphqlHTTP({ schema, rootValue: new Query(), graphiql: true }),
);
app.listen(4000);
console.log("Running a GraphQL API server at http://localhost:4000/graphql");
Try it out on CodeSandbox!
Grats has a few configuration options. They can be set in your project's
tsconfig.json
file:
{
"grats": {
// Should all fields be typed as nullable in accordance with GraphQL best
// practices?
// https://graphql.org/learn/best-practices/#nullability
"nullableByDefault": true, // Default: true
},
"compilerOptions": {
// ... TypeScript config...
}
}
In order for Grats to extract GraphQL schema from your code, simply mark which
TypeScript structures should be included in the schema by marking them with
special JSDoc tags such as /** @gqlType */
or /** @gqlField */
.
Any comment text preceding the JSDoc @
tag will be used as that element's description.
Note that JSDocs must being with
/**
(two asterix). However, they may be consolidated into a single line.
The following JSDoc tags are supported:
GraphQL types can be defined by placing a @gqlType
docblock directly before a:
/**
* Here I can write a description of my type that will be included in the schema.
* @gqlType <optional name of the type, if different from class name>
*/
class MyClass {
/** @gqlField */
someField: string;
}
/**
* Here I can write a description of my type that will be included in the schema.
* @gqlType <optional name of the type, if different from interface name>
*/
interface MyInterface {
/** @gqlField */
someField: string;
}
Note: If your type implements a GraphQL interface or is a member of a GraphQL
union, Grats will remind you to add a __typename: "MyType"
property to your
class or interface.
GraphQL interfaces can be defined by placing a @gqlInterface
docblock directly before an:
/**
* A description of my interface.
* @gqlInterface <optional name of the type, if different from class name>
*/
interface MyClass {
/** @gqlField */
someField: string;
}
All @gqlType
types which implement the interface in TypeScript will
automatically implement it in GraphQL as well.
You can define GraphQL fields by placing a @gqlField
directly before a:
/**
* A description of some field.
* @gqlField <optional name of the field, if different from property name>
*/
someField: string;
/**
* A description of my field.
* @gqlField <optional name of the field, if different from method name>
*/
myField(): string {
return "Hello World";
}
Note: By default, Grats makes all fields nullable in keeping with GraphQL
*best practices. This
*behavior can be changed by setting the config option nullableByDefault
to
false
.
If you wish to define arguments for a field, define your argument types inline:
/** @gqlField */
myField(args: { greeting: string }): string {
return `${args.greeting} World`;
}
Default values for arguments can be defined by using the =
operator with destructuring:
/** @gqlField */
myField({ greeting = "Hello" }: { greeting: string }): string {
return `${greeting} World`;
}
Arguments can be given descriptions by using the /**
syntax:
/** @gqlField */
myField(args: {
/** A description of the greeting argument */
greeting: string
}): string {
return `${args.greeting} World`;
}
To mark a field as deprecated, use the @deprecated
JSDoc tag:
/**
* @gqlField
* @deprecated Please use myNewField instead.
*/
myOldField(): string {
return "Hello World";
}
Sometimes you want to add a computed field to a non-class type, or extend base
types like Query
or Mutation
from another file. Both of these usecases are
enabled by placing a @gqlField
before an exported function declaration.
In this case, the function should expect an instance of the base type as the first argument, and an object representing the GraphQL field arguments as the second argument. The function should return the value of the field.
Extending Query:
/**
* Description of my field
* @gqlField <optional name of the field, if different from function name>
*/
export function userById(_: Query, args: {id: string}): User {
return DB.getUserById(args.id);
}
Extending Mutation:
/**
* Delete a user. GOODBYE!
* @gqlField <optional name of the mutation, if different from function name>
*/
export function deleteUser(_: Mutation, args: {id: string}): boolean {
return DB.deleteUser(args.id);
}
Note that Grats will use the type of the first argument to determine which type is being extended. So, as seen in the previous examples, even if you don't need access to the instance you should still define a typed first argument.
GraphQL unions can be defined by placing a @gqlUnion
docblock directly before a:
/**
* A description of my union.
* @gqlUnion <optional name of the union, if different from type name>
*/
type MyUnion = User | Post;
GraphQL custom sclars can be defined by placing a @gqlScalar
docblock directly before a:
/**
* A description of my custom scalar.
* @gqlScalar <optional name of the scalar, if different from type name>
*/
type MyCustomString = string;
Note: For the built-in GraphQL scalars that don't have a corresponding TypeScript type, Grats ships with type aliases you can import. You may be promted to use one of these by Grat if you try to use number
in a positon from which Grat needs to infer a GraphQL type.
import { Float, Int } from "grats";
/** @gqlType */
class Query {
/** @gqlField */
round(args: {float: Float}): Int {
return Math.round(args.float);
}
}
GraphQL enums can be defined by placing a @gqlEnum
docblock directly before a:
/**
* A description of my enum.
* @gqlEnum <optional name of the enum, if different from type name>
*/
enum MyEnum {
/** A description of my variant */
OK = "OK",
/** A description of my other variant */
ERROR = "ERROR"
}
Note that the values of the enum are used as the GraphQL enum values, and must be string literals.
To mark a variants as deprecated, use the @deprecated
JSDoc tag directly before it:
/** @gqlEnum */
enum MyEnum {
OK = "OK"
/** @deprecated Please use OK instead. */
OKAY = "OKAY"
ERROR = "ERROR"
}
We also support defining enums using a union of string literals, however there are some limitations to this approach:
This is due to the fact that TypeScript does not see JSDoc comments as "attaching" to string literal types.
/**
* A description of my enum.
* @gqlEnum <optional name of the enum, if different from type name>
*/
type MyEnum = "OK" | "ERROR";
GraphQL input types can be defined by placing a @gqlInput
docblock directly before a:
/**
* Description of my input type
* @gqlInput <optional name of the input, if different from type name>
*/
type MyInput = {
name: string;
age: number;
};
See example-server/
in the repo root for a working example. Here we run the static
analysis at startup time. Nice for development, but not ideal for production
where you would want to cache the schema and write it to disk for other tools to
see.
See CONTRIBUTING.md
in the repo root for details on how to make changes to this project.
Because Grats relies on static analysis to infer types, it requires that your GraphQL fields use types that can be statically analyzed. This means that you can't use complex derived types in positions where Grats needs to be able to infer the type. For example, field arguments and return values.
Currently, Grats does not have a great way to handle the case where you want to expose structures that are not owned by your codebase. For example, if you want to expose a field that returns a type from a third-party library, or a type that is generated by some other codegen tool. Today, your best option is to define a wrapper resolver class.
Using decorators to signal that a class/method/etc should be included in the schema would have some advantages:
However, it also has some disadvantages:
Given these tradeoffs, we've decided to use comments instead of decorators.
FAQs
[![Join our Discord!](https://img.shields.io/discord/1089650710796320868?logo=discord)](https://capt.dev/grats-chat)
The npm package grats receives a total of 104 weekly downloads. As such, grats popularity was classified as not popular.
We found that grats demonstrated a healthy version release cadence and project activity because the last version was released less than a year ago. It has 0 open source maintainers collaborating on the project.
Did you know?
Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.
Security News
Snyk's use of malicious npm packages for research raises ethical concerns, highlighting risks in public deployment, data exfiltration, and unauthorized testing.
Research
Security News
Socket researchers found several malicious npm packages typosquatting Chalk and Chokidar, targeting Node.js developers with kill switches and data theft.
Security News
pnpm 10 blocks lifecycle scripts by default to improve security, addressing supply chain attack risks but sparking debate over compatibility and workflow changes.