Comparing version
{ | ||
"name": "grats", | ||
"version": "0.0.2", | ||
"version": "0.0.10", | ||
"main": "dist/src/index.js", | ||
@@ -12,14 +12,15 @@ "bin": "dist/src/cli.js", | ||
"scripts": { | ||
"test": "ts-node --esm src/tests/test.ts", | ||
"test": "ts-node src/tests/test.ts", | ||
"integration-tests": "node src/tests/integration.mjs", | ||
"build": "tsc --build", | ||
"lint": "eslint src/**/*.ts" | ||
"format": "prettier . --write", | ||
"lint": "eslint src/**/*.ts && prettier . --check" | ||
}, | ||
"dependencies": { | ||
"@graphql-tools/utils": "^9.2.1", | ||
"commander": "^10.0.0", | ||
"graphql": "^16.6.0", | ||
"typescript": "^4.9.5" | ||
"typescript": "^5.0.2" | ||
}, | ||
"devDependencies": { | ||
"@graphql-tools/utils": "^9.2.1", | ||
"@types/node": "^18.14.6", | ||
@@ -32,2 +33,3 @@ "@typescript-eslint/eslint-plugin": "^5.55.0", | ||
"path-browserify": "^1.0.1", | ||
"prettier": "^2.8.7", | ||
"process": "^0.11.10", | ||
@@ -39,3 +41,7 @@ "ts-node": "^10.9.1" | ||
}, | ||
"packageManager": "pnpm@8.1.1" | ||
"packageManager": "pnpm@8.12.0", | ||
"engines": { | ||
"node": ">=16 <=21", | ||
"pnpm": "^8" | ||
} | ||
} |
#!/usr/bin/env node | ||
export {}; | ||
import { Location } from "graphql"; | ||
export declare function formatLoc(loc: Location): string; |
@@ -39,7 +39,7 @@ #!/usr/bin/env node | ||
}; | ||
exports.__esModule = true; | ||
Object.defineProperty(exports, "__esModule", { value: true }); | ||
exports.formatLoc = void 0; | ||
var graphql_1 = require("graphql"); | ||
var _1 = require("./"); | ||
var lib_1 = require("./lib"); | ||
var utils_1 = require("@graphql-tools/utils"); | ||
var commander_1 = require("commander"); | ||
@@ -49,2 +49,5 @@ var fs_1 = require("fs"); | ||
var package_json_1 = require("../package.json"); | ||
var Locate_1 = require("./Locate"); | ||
var printSchema_1 = require("./printSchema"); | ||
var ts = require("typescript"); | ||
var program = new commander_1.Command(); | ||
@@ -55,9 +58,8 @@ program | ||
.version(package_json_1.version) | ||
.option("-o, --output <SCHEMA_FILE>", "Where to write the schema file. Defaults to stdout") | ||
.option("--tsconfig <TSCONFIG>", "Path to tsconfig.json. Defaults to auto-detecting based on the current working directory") | ||
.action(function (_a) { | ||
var output = _a.output, tsconfig = _a.tsconfig; | ||
var tsconfig = _a.tsconfig; | ||
return __awaiter(void 0, void 0, void 0, function () { | ||
return __generator(this, function (_b) { | ||
build(output, tsconfig); | ||
build(tsconfig); | ||
return [2 /*return*/]; | ||
@@ -67,12 +69,47 @@ }); | ||
}); | ||
program | ||
.command("locate") | ||
.argument("<ENTITY>", "GraphQL entity to locate. E.g. `User` or `User.id`") | ||
.option("--tsconfig <TSCONFIG>", "Path to tsconfig.json. Defaults to auto-detecting based on the current working directory") | ||
.action(function (entity, _a) { | ||
var tsconfig = _a.tsconfig; | ||
var config = getTsConfig(tsconfig).config; | ||
var schema = buildSchema(config); | ||
var loc = (0, Locate_1.locate)(schema, entity); | ||
if (loc.kind === "ERROR") { | ||
console.error(loc.err); | ||
process.exit(1); | ||
} | ||
console.log(formatLoc(loc.value)); | ||
}); | ||
program.parse(); | ||
function build(output, tsconfig) { | ||
if (tsconfig && !(0, fs_1.existsSync)(tsconfig)) { | ||
console.error("Grats: Could not find tsconfig.json at `".concat(tsconfig, "`.")); | ||
function build(tsconfig) { | ||
var _a = getTsConfig(tsconfig), config = _a.config, configPath = _a.configPath; | ||
var schema = buildSchema(config); | ||
var sortedSchema = (0, graphql_1.lexicographicSortSchema)(schema); | ||
var gratsOptions = config.raw.grats; | ||
var dest = (0, path_1.resolve)((0, path_1.dirname)(configPath), gratsOptions.tsSchema); | ||
var code = (0, printSchema_1.printExecutableSchema)(sortedSchema, gratsOptions, dest); | ||
(0, fs_1.writeFileSync)(dest, code); | ||
console.error("Grats: Wrote TypeScript schema to `".concat(dest, "`.")); | ||
var schemaStr = (0, printSchema_1.printGratsSDL)(sortedSchema, gratsOptions); | ||
var absOutput = (0, path_1.resolve)((0, path_1.dirname)(configPath), gratsOptions.graphqlSchema); | ||
(0, fs_1.writeFileSync)(absOutput, schemaStr); | ||
console.error("Grats: Wrote schema to `".concat(absOutput, "`.")); | ||
} | ||
// Locate and read the tsconfig.json file | ||
function getTsConfig(tsconfig) { | ||
var configPath = tsconfig || ts.findConfigFile(process.cwd(), ts.sys.fileExists); | ||
if (configPath == null) { | ||
throw new Error("Grats: Could not find tsconfig.json"); | ||
} | ||
var optionsResult = (0, _1.getParsedTsConfig)(configPath); | ||
if (optionsResult.kind === "ERROR") { | ||
console.error(optionsResult.err.formatDiagnosticsWithColorAndContext()); | ||
process.exit(1); | ||
} | ||
var parsed = (0, _1.getParsedTsConfig)(tsconfig); | ||
// FIXME: Validate config! | ||
// https://github.com/tsconfig/bases | ||
var schemaResult = (0, lib_1.buildSchemaResult)(parsed); | ||
return { configPath: configPath, config: optionsResult.value }; | ||
} | ||
function buildSchema(options) { | ||
var schemaResult = (0, lib_1.buildSchemaResult)(options); | ||
if (schemaResult.kind === "ERROR") { | ||
@@ -82,12 +119,9 @@ console.error(schemaResult.err.formatDiagnosticsWithColorAndContext()); | ||
} | ||
var schema = (0, graphql_1.lexicographicSortSchema)(schemaResult.value); | ||
var schemaStr = (0, utils_1.printSchemaWithDirectives)(schema, { assumeValid: true }); | ||
if (output) { | ||
var absOutput = (0, path_1.resolve)(process.cwd(), output); | ||
(0, fs_1.writeFileSync)(absOutput, schemaStr); | ||
console.error("Grats: Wrote schema to `".concat(absOutput, "`.")); | ||
} | ||
else { | ||
console.log(schemaStr); | ||
} | ||
return schemaResult.value; | ||
} | ||
// Format a location for printing to the console. Tools like VS Code and iTerm | ||
// will automatically turn this into a clickable link. | ||
function formatLoc(loc) { | ||
return "".concat(loc.source.name, ":").concat(loc.startToken.line + 1, ":").concat(loc.startToken.column + 1); | ||
} | ||
exports.formatLoc = formatLoc; |
@@ -34,4 +34,4 @@ /** | ||
export declare function inputFieldUntyped(): string; | ||
export declare function typeTagOnUnamedClass(): string; | ||
export declare function typeTagOnAliasOfNonObject(): string; | ||
export declare function typeTagOnUnnamedClass(): string; | ||
export declare function typeTagOnAliasOfNonObjectOrUnknown(): string; | ||
export declare function typeNameNotDeclaration(): string; | ||
@@ -66,6 +66,29 @@ export declare function typeNameMissingInitializer(): string; | ||
export declare function pluralTypeMissingParameter(): string; | ||
export declare function expectedIdentifer(): string; | ||
export declare function expectedIdentifier(): string; | ||
export declare function killsParentOnExceptionWithWrongConfig(): string; | ||
export declare function killsParentOnExceptionOnNullable(): string; | ||
export declare function nonNullTypeCannotBeOptional(): string; | ||
export declare function mergedInterfaces(interfaceName: string): string; | ||
export declare function mergedInterfaces(): string; | ||
export declare function implementsTagMissingValue(): string; | ||
export declare function implementsTagOnClass(): string; | ||
export declare function implementsTagOnInterface(): string; | ||
export declare function implementsTagOnTypeAlias(): string; | ||
export declare function duplicateTag(tagName: string): string; | ||
export declare function duplicateInterfaceTag(): string; | ||
export declare function parameterWithoutModifiers(): string; | ||
export declare function parameterPropertyNotPublic(): string; | ||
export declare function parameterPropertyMissingType(): string; | ||
export declare function invalidTypePassedToFieldFunction(): string; | ||
export declare function unresolvedTypeReference(): string; | ||
export declare function expectedTypeAnnotationOnContext(): string; | ||
export declare function expectedTypeAnnotationOfReferenceOnContext(): string; | ||
export declare function expectedTypeAnnotationOnContextToBeResolvable(): string; | ||
export declare function expectedTypeAnnotationOnContextToHaveDeclaration(): string; | ||
export declare function unexpectedParamSpreadForContextParam(): string; | ||
export declare function multipleContextTypes(): string; | ||
export declare function graphQLNameHasLeadingNewlines(name: string, tagName: string): string; | ||
export declare function graphQLTagNameHasWhitespace(tagName: string): string; | ||
export declare function subscriptionFieldNotAsyncIterable(): string; | ||
export declare function nonSubscriptionFieldAsyncIterable(): string; | ||
export declare function operationTypeNotUnknown(): string; | ||
export declare function expectedNullableArgumentToBeOptional(): string; |
"use strict"; | ||
exports.__esModule = true; | ||
exports.defaultArgPropertyMissingName = exports.defaultArgElementIsNotAssignment = exports.defaultValueIsNotLiteral = exports.ambiguousNumberType = exports.expectedOneNonNullishType = exports.propertyFieldMissingType = exports.cannotResolveSymbolForDescription = exports.promiseMissingTypeArg = exports.methodMissingType = exports.gqlEntityMissingName = exports.enumVariantMissingInitializer = exports.enumVariantNotStringLiteral = exports.enumTagOnInvalidNode = exports.argNotTyped = exports.argNameNotLiteral = exports.argIsNotProperty = exports.argumentParamIsNotObject = exports.argumentParamIsMissingType = exports.typeNameDoesNotMatchExpected = exports.typeNameTypeNotStringLiteral = exports.typeNameMissingTypeAnnotation = exports.typeNameInitializerWrong = exports.typeNameInitializeNotString = exports.typeNameMissingInitializer = exports.typeNameNotDeclaration = exports.typeTagOnAliasOfNonObject = exports.typeTagOnUnamedClass = exports.inputFieldUntyped = exports.inputTypeFieldNotProperty = exports.inputTypeNotLiteral = exports.functionFieldNotNamedExport = exports.functionFieldDefaultExport = exports.functionFieldNotNamed = exports.functionFieldParentTypeNotValid = exports.functionFieldParentTypeMissing = exports.functionFieldNotTopLevel = exports.invalidReturnTypeForFunctionField = exports.invalidParentArgForFunctionField = exports.expectedUnionTypeReference = exports.expectedUnionTypeNode = exports.invalidUnionTagUsage = exports.invalidInputTagUsage = exports.invalidEnumTagUsage = exports.invalidInterfaceTagUsage = exports.invalidScalarTagUsage = exports.invalidTypeTagUsage = exports.invalidGratsTag = exports.wrongCasingForGratsTag = exports.killsParentOnExceptionOnWrongNode = exports.fieldTagOnWrongNode = void 0; | ||
exports.mergedInterfaces = exports.nonNullTypeCannotBeOptional = exports.killsParentOnExceptionOnNullable = exports.killsParentOnExceptionWithWrongConfig = exports.expectedIdentifer = exports.pluralTypeMissingParameter = exports.unknownGraphQLType = exports.unsupportedTypeLiteral = exports.defaultArgPropertyMissingInitializer = void 0; | ||
Object.defineProperty(exports, "__esModule", { value: true }); | ||
exports.defaultArgPropertyMissingName = exports.defaultArgElementIsNotAssignment = exports.defaultValueIsNotLiteral = exports.ambiguousNumberType = exports.expectedOneNonNullishType = exports.propertyFieldMissingType = exports.cannotResolveSymbolForDescription = exports.promiseMissingTypeArg = exports.methodMissingType = exports.gqlEntityMissingName = exports.enumVariantMissingInitializer = exports.enumVariantNotStringLiteral = exports.enumTagOnInvalidNode = exports.argNotTyped = exports.argNameNotLiteral = exports.argIsNotProperty = exports.argumentParamIsNotObject = exports.argumentParamIsMissingType = exports.typeNameDoesNotMatchExpected = exports.typeNameTypeNotStringLiteral = exports.typeNameMissingTypeAnnotation = exports.typeNameInitializerWrong = exports.typeNameInitializeNotString = exports.typeNameMissingInitializer = exports.typeNameNotDeclaration = exports.typeTagOnAliasOfNonObjectOrUnknown = exports.typeTagOnUnnamedClass = exports.inputFieldUntyped = exports.inputTypeFieldNotProperty = exports.inputTypeNotLiteral = exports.functionFieldNotNamedExport = exports.functionFieldDefaultExport = exports.functionFieldNotNamed = exports.functionFieldParentTypeNotValid = exports.functionFieldParentTypeMissing = exports.functionFieldNotTopLevel = exports.invalidReturnTypeForFunctionField = exports.invalidParentArgForFunctionField = exports.expectedUnionTypeReference = exports.expectedUnionTypeNode = exports.invalidUnionTagUsage = exports.invalidInputTagUsage = exports.invalidEnumTagUsage = exports.invalidInterfaceTagUsage = exports.invalidScalarTagUsage = exports.invalidTypeTagUsage = exports.invalidGratsTag = exports.wrongCasingForGratsTag = exports.killsParentOnExceptionOnWrongNode = exports.fieldTagOnWrongNode = void 0; | ||
exports.expectedNullableArgumentToBeOptional = exports.operationTypeNotUnknown = exports.nonSubscriptionFieldAsyncIterable = exports.subscriptionFieldNotAsyncIterable = exports.graphQLTagNameHasWhitespace = exports.graphQLNameHasLeadingNewlines = exports.multipleContextTypes = exports.unexpectedParamSpreadForContextParam = exports.expectedTypeAnnotationOnContextToHaveDeclaration = exports.expectedTypeAnnotationOnContextToBeResolvable = exports.expectedTypeAnnotationOfReferenceOnContext = exports.expectedTypeAnnotationOnContext = exports.unresolvedTypeReference = exports.invalidTypePassedToFieldFunction = exports.parameterPropertyMissingType = exports.parameterPropertyNotPublic = exports.parameterWithoutModifiers = exports.duplicateInterfaceTag = exports.duplicateTag = exports.implementsTagOnTypeAlias = exports.implementsTagOnInterface = exports.implementsTagOnClass = exports.implementsTagMissingValue = exports.mergedInterfaces = exports.nonNullTypeCannotBeOptional = exports.killsParentOnExceptionOnNullable = exports.killsParentOnExceptionWithWrongConfig = exports.expectedIdentifier = exports.pluralTypeMissingParameter = exports.unknownGraphQLType = exports.unsupportedTypeLiteral = exports.defaultArgPropertyMissingInitializer = void 0; | ||
var Extractor_1 = require("./Extractor"); | ||
// TODO: Move these to short URLS that are easier to keep from breaking. | ||
var DOC_URLS = { | ||
mergedInterfaces: "https://grats.capt.dev/docs/dockblock-tags/interfaces/#merged-interfaces" | ||
mergedInterfaces: "https://grats.capt.dev/docs/docblock-tags/interfaces/#merged-interfaces", | ||
parameterProperties: "https://grats.capt.dev/docs/docblock-tags/fields#class-based-fields", | ||
}; | ||
@@ -70,3 +71,3 @@ /** | ||
function invalidParentArgForFunctionField() { | ||
return "Expected `@".concat(Extractor_1.FIELD_TAG, "` function to have a first argument representing the type to extend."); | ||
return "Expected `@".concat(Extractor_1.FIELD_TAG, "` function to have a first argument representing the type to extend. If you don't need access to the parent object in the function, you can name the variable `_` to indicate that it is unused. e.g. `function myField(_: ParentType) {}`"); | ||
} | ||
@@ -114,10 +115,10 @@ exports.invalidParentArgForFunctionField = invalidParentArgForFunctionField; | ||
exports.inputFieldUntyped = inputFieldUntyped; | ||
function typeTagOnUnamedClass() { | ||
function typeTagOnUnnamedClass() { | ||
return "Unexpected `@".concat(Extractor_1.TYPE_TAG, "` annotation on unnamed class declaration."); | ||
} | ||
exports.typeTagOnUnamedClass = typeTagOnUnamedClass; | ||
function typeTagOnAliasOfNonObject() { | ||
return "Expected `@".concat(Extractor_1.TYPE_TAG, "` type to be a type literal. For example: `type Foo = { bar: string }`"); | ||
exports.typeTagOnUnnamedClass = typeTagOnUnnamedClass; | ||
function typeTagOnAliasOfNonObjectOrUnknown() { | ||
return "Expected `@".concat(Extractor_1.TYPE_TAG, "` type to be an object type literal (`{ }`) or `unknown`. For example: `type Foo = { bar: string }` or `type Query = unknown`."); | ||
} | ||
exports.typeTagOnAliasOfNonObject = typeTagOnAliasOfNonObject; | ||
exports.typeTagOnAliasOfNonObjectOrUnknown = typeTagOnAliasOfNonObjectOrUnknown; | ||
function typeNameNotDeclaration() { | ||
@@ -152,7 +153,7 @@ return "Expected `__typename` to be a property declaration."; | ||
function argumentParamIsMissingType() { | ||
return "Expected GraphQL field arguments to have a TypeScript type. If there are no arguments, you can use `args: never`."; | ||
return "Expected GraphQL field arguments to have a TypeScript type. If there are no arguments, you can use `args: unknown`."; | ||
} | ||
exports.argumentParamIsMissingType = argumentParamIsMissingType; | ||
function argumentParamIsNotObject() { | ||
return "Expected GraphQL field arguments to be typed using a literal object: `{someField: string}`."; | ||
return "Expected GraphQL field arguments to be typed using a literal object: `{someField: string}`. If there are no arguments, you can use `args: unknown`."; | ||
} | ||
@@ -240,6 +241,6 @@ exports.argumentParamIsNotObject = argumentParamIsNotObject; | ||
exports.pluralTypeMissingParameter = pluralTypeMissingParameter; | ||
function expectedIdentifer() { | ||
function expectedIdentifier() { | ||
return "Expected an identifier."; | ||
} | ||
exports.expectedIdentifer = expectedIdentifer; | ||
exports.expectedIdentifier = expectedIdentifier; | ||
function killsParentOnExceptionWithWrongConfig() { | ||
@@ -250,12 +251,12 @@ return "Unexpected `@".concat(Extractor_1.KILLS_PARENT_ON_EXCEPTION_TAG, "` tag. `@").concat(Extractor_1.KILLS_PARENT_ON_EXCEPTION_TAG, "` is only supported when the Grats config `nullableByDefault` is enabled."); | ||
function killsParentOnExceptionOnNullable() { | ||
return "Unexpected `@".concat(Extractor_1.KILLS_PARENT_ON_EXCEPTION_TAG, "` tag. `@").concat(Extractor_1.KILLS_PARENT_ON_EXCEPTION_TAG, "` is unnessesary on fields that are already nullable."); | ||
return "Unexpected `@".concat(Extractor_1.KILLS_PARENT_ON_EXCEPTION_TAG, "` tag. `@").concat(Extractor_1.KILLS_PARENT_ON_EXCEPTION_TAG, "` is unnecessary on fields that are already nullable."); | ||
} | ||
exports.killsParentOnExceptionOnNullable = killsParentOnExceptionOnNullable; | ||
function nonNullTypeCannotBeOptional() { | ||
return "Unexpected optional argument that does not also accept `null`. Optional arguments in GraphQL may get passed an explict `null` value. This means optional arguments must be typed to also accept `null`."; | ||
return "Unexpected optional argument that does not also accept `null`. Optional arguments in GraphQL may get passed an explicit `null` value. This means optional arguments must be typed to also accept `null`."; | ||
} | ||
exports.nonNullTypeCannotBeOptional = nonNullTypeCannotBeOptional; | ||
function mergedInterfaces(interfaceName) { | ||
function mergedInterfaces() { | ||
return [ | ||
"Unexpected merged interface `".concat(interfaceName, "`."), | ||
"Unexpected merged interface.", | ||
"If an interface is declared multiple times in a scope, TypeScript merges them.", | ||
@@ -268,1 +269,101 @@ "To avoid ambiguity Grats does not support using merged interfaces as GraphQL interfaces.", | ||
exports.mergedInterfaces = mergedInterfaces; | ||
function implementsTagMissingValue() { | ||
return "Expected `@".concat(Extractor_1.IMPLEMENTS_TAG_DEPRECATED, "` to be followed by one or more interface names."); | ||
} | ||
exports.implementsTagMissingValue = implementsTagMissingValue; | ||
function implementsTagOnClass() { | ||
return "`@".concat(Extractor_1.IMPLEMENTS_TAG_DEPRECATED, "` has been deprecated. Instead use `class MyType implements MyInterface`."); | ||
} | ||
exports.implementsTagOnClass = implementsTagOnClass; | ||
function implementsTagOnInterface() { | ||
return "`@".concat(Extractor_1.IMPLEMENTS_TAG_DEPRECATED, "` has been deprecated. Instead use `interface MyType extends MyInterface`."); | ||
} | ||
exports.implementsTagOnInterface = implementsTagOnInterface; | ||
function implementsTagOnTypeAlias() { | ||
return "`@".concat(Extractor_1.IMPLEMENTS_TAG_DEPRECATED, "` has been deprecated. Types which implement GraphQL interfaces should be defined using TypeScript class or interface declarations."); | ||
} | ||
exports.implementsTagOnTypeAlias = implementsTagOnTypeAlias; | ||
function duplicateTag(tagName) { | ||
return "Unexpected duplicate `@".concat(tagName, "` tag. Grats does not accept multiple instances of the same tag."); | ||
} | ||
exports.duplicateTag = duplicateTag; | ||
function duplicateInterfaceTag() { | ||
return "Unexpected duplicate `@".concat(Extractor_1.IMPLEMENTS_TAG_DEPRECATED, "` tag. To declare that a type or interface implements multiple interfaces list them as comma separated values: `@").concat(Extractor_1.IMPLEMENTS_TAG_DEPRECATED, " interfaceA, interfaceB`."); | ||
} | ||
exports.duplicateInterfaceTag = duplicateInterfaceTag; | ||
function parameterWithoutModifiers() { | ||
return [ | ||
"Expected `@".concat(Extractor_1.FIELD_TAG, "` constructor parameter to be a parameter property. This requires a modifier such as `public` or `readonly` before the parameter name.\n\n"), | ||
"Learn more: ".concat(DOC_URLS.parameterProperties), | ||
].join(""); | ||
} | ||
exports.parameterWithoutModifiers = parameterWithoutModifiers; | ||
function parameterPropertyNotPublic() { | ||
return [ | ||
"Expected `@".concat(Extractor_1.FIELD_TAG, "` parameter property to be public. Valid modifiers for `@").concat(Extractor_1.FIELD_TAG, "` parameter properties are `public` and `readonly`.\n\n"), | ||
"Learn more: ".concat(DOC_URLS.parameterProperties), | ||
].join(""); | ||
} | ||
exports.parameterPropertyNotPublic = parameterPropertyNotPublic; | ||
function parameterPropertyMissingType() { | ||
return "Expected `@".concat(Extractor_1.FIELD_TAG, "` parameter property to have a type annotation."); | ||
} | ||
exports.parameterPropertyMissingType = parameterPropertyMissingType; | ||
function invalidTypePassedToFieldFunction() { | ||
return "Unexpected type passed to `@".concat(Extractor_1.FIELD_TAG, "` function. `@").concat(Extractor_1.FIELD_TAG, "` functions can only be used to extend `@").concat(Extractor_1.TYPE_TAG, "` and `@").concat(Extractor_1.INTERFACE_TAG, "` types."); | ||
} | ||
exports.invalidTypePassedToFieldFunction = invalidTypePassedToFieldFunction; | ||
function unresolvedTypeReference() { | ||
return "This type is not a valid GraphQL type. Did you mean to annotate it's definition with a `/** @gql */` tag such as `/** @gqlType */` or `/** @gqlInput **/`?"; | ||
} | ||
exports.unresolvedTypeReference = unresolvedTypeReference; | ||
function expectedTypeAnnotationOnContext() { | ||
return "Expected context parameter to have a type annotation. Grats validates that your context parameter is type-safe by checking all context values reference the same type declaration."; | ||
} | ||
exports.expectedTypeAnnotationOnContext = expectedTypeAnnotationOnContext; | ||
function expectedTypeAnnotationOfReferenceOnContext() { | ||
return "Expected context parameter's type to be a type reference Grats validates that your context parameter is type-safe by checking all context values reference the same type declaration."; | ||
} | ||
exports.expectedTypeAnnotationOfReferenceOnContext = expectedTypeAnnotationOfReferenceOnContext; | ||
function expectedTypeAnnotationOnContextToBeResolvable() { | ||
// TODO: Provide guidance? | ||
// TODO: I don't think we have a test case that triggers this error. | ||
return "Unable to resolve context parameter type. Grats validates that your context parameter is type-safe by checking all context values reference the same type declaration."; | ||
} | ||
exports.expectedTypeAnnotationOnContextToBeResolvable = expectedTypeAnnotationOnContextToBeResolvable; | ||
function expectedTypeAnnotationOnContextToHaveDeclaration() { | ||
return "Unable to locate the declaration of the context parameter's type. Grats validates that your context parameter is type-safe by checking all context values reference the same type declaration. Did you forget to import or define this type?"; | ||
} | ||
exports.expectedTypeAnnotationOnContextToHaveDeclaration = expectedTypeAnnotationOnContextToHaveDeclaration; | ||
function unexpectedParamSpreadForContextParam() { | ||
return "Unexpected spread parameter in context parameter position. Grats expects the context parameter to be a single, explicitly typed, argument."; | ||
} | ||
exports.unexpectedParamSpreadForContextParam = unexpectedParamSpreadForContextParam; | ||
function multipleContextTypes() { | ||
return "Context argument's type does not match. Grats expects all resolvers that read the context argument to use the same type for that argument. Did you use the incorrect type in one of your resolvers?"; | ||
} | ||
exports.multipleContextTypes = multipleContextTypes; | ||
function graphQLNameHasLeadingNewlines(name, tagName) { | ||
return "Expected the GraphQL name `".concat(name, "` to be on the same line as it's `@").concat(tagName, "` tag."); | ||
} | ||
exports.graphQLNameHasLeadingNewlines = graphQLNameHasLeadingNewlines; | ||
function graphQLTagNameHasWhitespace(tagName) { | ||
return "Expected text following a `@".concat(tagName, "` tag to be a GraphQL name. If you intended this text to be a description, place it at the top of the docblock before any `@tags`."); | ||
} | ||
exports.graphQLTagNameHasWhitespace = graphQLTagNameHasWhitespace; | ||
function subscriptionFieldNotAsyncIterable() { | ||
return "Expected fields on `Subscription` to return an AsyncIterable."; | ||
} | ||
exports.subscriptionFieldNotAsyncIterable = subscriptionFieldNotAsyncIterable; | ||
function nonSubscriptionFieldAsyncIterable() { | ||
return "Unexpected AsyncIterable. Only fields on `Subscription` should return an AsyncIterable."; | ||
} | ||
exports.nonSubscriptionFieldAsyncIterable = nonSubscriptionFieldAsyncIterable; | ||
function operationTypeNotUnknown() { | ||
return "Operation types `Query`, `Mutation`, and `Subscription` must be defined as type aliases of `unknown`. E.g. `type Query = unknown`."; | ||
} | ||
exports.operationTypeNotUnknown = operationTypeNotUnknown; | ||
function expectedNullableArgumentToBeOptional() { | ||
return "Expected nullable argument to be optional. graphql-js may not define properties where an undefined argument is passed. To guard against this add a `?` to the end of the argument name to make it optional."; | ||
} | ||
exports.expectedNullableArgumentToBeOptional = expectedNullableArgumentToBeOptional; |
@@ -1,6 +0,7 @@ | ||
import { DefinitionNode, FieldDefinitionNode, InputValueDefinitionNode, ListTypeNode, NamedTypeNode, Location as GraphQLLocation, NameNode, Token, TypeNode, NonNullTypeNode, StringValueNode, ConstValueNode, ConstDirectiveNode, ConstArgumentNode, EnumValueDefinitionNode, ConstObjectFieldNode, ConstObjectValueNode, ConstListValueNode } from "graphql"; | ||
import { NameNode } from "graphql"; | ||
import { DiagnosticsResult } from "./utils/DiagnosticError"; | ||
import * as ts from "typescript"; | ||
import { TypeContext } from "./TypeContext"; | ||
import { NameDefinition } from "./TypeContext"; | ||
import { ConfigOptions } from "./lib"; | ||
import { GratsDefinitionNode } from "./GraphQLConstructor"; | ||
export declare const LIBRARY_IMPORT_NAME = "grats"; | ||
@@ -16,5 +17,13 @@ export declare const LIBRARY_NAME = "Grats"; | ||
export declare const INPUT_TAG = "gqlInput"; | ||
export declare const IMPLEMENTS_TAG_DEPRECATED = "gqlImplements"; | ||
export declare const KILLS_PARENT_ON_EXCEPTION_TAG = "killsParentOnException"; | ||
export declare const ALL_TAGS: string[]; | ||
type ArgDefaults = Map<string, ts.Expression>; | ||
export type ExtractionSnapshot = { | ||
readonly definitions: GratsDefinitionNode[]; | ||
readonly unresolvedNames: Map<ts.Node, NameNode>; | ||
readonly nameDefinitions: Map<ts.Node, NameDefinition>; | ||
readonly contextReferences: Array<ts.Node>; | ||
readonly typesWithTypenameField: Set<string>; | ||
readonly interfaceDeclarations: Array<ts.InterfaceDeclaration>; | ||
}; | ||
/** | ||
@@ -30,84 +39,2 @@ * Extracts GraphQL definitions from TypeScript source code. | ||
*/ | ||
export declare class Extractor { | ||
definitions: DefinitionNode[]; | ||
sourceFile: ts.SourceFile; | ||
ctx: TypeContext; | ||
configOptions: ConfigOptions; | ||
errors: ts.Diagnostic[]; | ||
constructor(sourceFile: ts.SourceFile, ctx: TypeContext, buildOptions: ConfigOptions); | ||
extract(): DiagnosticsResult<DefinitionNode[]>; | ||
extractType(node: ts.Node, tag: ts.JSDocTag): void; | ||
extractScalar(node: ts.Node, tag: ts.JSDocTag): void; | ||
extractInterface(node: ts.Node, tag: ts.JSDocTag): void; | ||
extractEnum(node: ts.Node, tag: ts.JSDocTag): void; | ||
extractInput(node: ts.Node, tag: ts.JSDocTag): void; | ||
extractUnion(node: ts.Node, tag: ts.JSDocTag): void; | ||
/** Error handling and location juggling */ | ||
report(node: ts.Node, message: string, relatedInformation?: ts.DiagnosticRelatedInformation[]): null; | ||
reportUnhandled(node: ts.Node, message: string, relatedInformation?: ts.DiagnosticRelatedInformation[]): null; | ||
related(node: ts.Node, message: string): ts.DiagnosticRelatedInformation; | ||
diagnosticAnnotatedLocation(node: ts.Node): { | ||
start: number; | ||
length: number; | ||
filepath: ts.SourceFile; | ||
}; | ||
loc(node: ts.Node): GraphQLLocation; | ||
gqlDummyToken(pos: number): Token; | ||
/** TypeScript traversals */ | ||
unionTypeAliasDeclaration(node: ts.TypeAliasDeclaration, tag: ts.JSDocTag): null | undefined; | ||
functionDeclarationExtendType(node: ts.FunctionDeclaration, tag: ts.JSDocTag): null | undefined; | ||
typeReferenceFromParam(typeParam: ts.ParameterDeclaration): NameNode | null; | ||
namedFunctionExportName(node: ts.FunctionDeclaration): ts.Identifier | null; | ||
scalarTypeAliasDeclaration(node: ts.TypeAliasDeclaration, tag: ts.JSDocTag): null | undefined; | ||
inputTypeAliasDeclaration(node: ts.TypeAliasDeclaration, tag: ts.JSDocTag): null | undefined; | ||
collectInputFields(node: ts.TypeAliasDeclaration): Array<InputValueDefinitionNode> | null; | ||
collectInputField(node: ts.PropertySignature): InputValueDefinitionNode | null; | ||
typeClassDeclaration(node: ts.ClassDeclaration, tag: ts.JSDocTag): null | undefined; | ||
typeInterfaceDeclaration(node: ts.InterfaceDeclaration, tag: ts.JSDocTag): null | undefined; | ||
typeTypeAliasDeclaration(node: ts.TypeAliasDeclaration, tag: ts.JSDocTag): null | undefined; | ||
checkForTypenameProperty(node: ts.ClassDeclaration | ts.InterfaceDeclaration | ts.TypeLiteralNode, expectedName: string): void; | ||
isValidTypeNameProperty(member: ts.ClassElement | ts.TypeElement, expectedName: string): boolean; | ||
isValidTypenamePropertyDeclaration(node: ts.PropertyDeclaration, expectedName: string): boolean; | ||
isValidTypenamePropertySignature(node: ts.PropertySignature, expectedName: string): boolean; | ||
isValidTypenamePropertyType(node: ts.TypeNode, expectedName: string): boolean; | ||
collectInterfaces(node: ts.ClassDeclaration | ts.InterfaceDeclaration): Array<NamedTypeNode> | null; | ||
interfaceInterfaceDeclaration(node: ts.InterfaceDeclaration, tag: ts.JSDocTag): null | undefined; | ||
collectFields(node: ts.ClassDeclaration | ts.InterfaceDeclaration | ts.TypeLiteralNode): Array<FieldDefinitionNode>; | ||
collectArgs(argsParam: ts.ParameterDeclaration): ReadonlyArray<InputValueDefinitionNode> | null; | ||
collectArgDefaults(node: ts.ObjectBindingPattern): ArgDefaults; | ||
collectConstValue(node: ts.Expression): ConstValueNode | null; | ||
collectArrayLiteral(node: ts.ArrayLiteralExpression): ConstListValueNode | null; | ||
cellectObjectLiteral(node: ts.ObjectLiteralExpression): ConstObjectValueNode | null; | ||
collectObjectField(node: ts.ObjectLiteralElementLike): ConstObjectFieldNode | null; | ||
collectArg(node: ts.TypeElement, defaults?: Map<string, ts.Expression> | null): InputValueDefinitionNode | null; | ||
enumEnumDeclaration(node: ts.EnumDeclaration, tag: ts.JSDocTag): void; | ||
enumTypeAliasDeclaration(node: ts.TypeAliasDeclaration, tag: ts.JSDocTag): void; | ||
enumTypeAliasVariants(node: ts.TypeAliasDeclaration): EnumValueDefinitionNode[] | null; | ||
collectEnumValues(node: ts.EnumDeclaration): ReadonlyArray<EnumValueDefinitionNode>; | ||
entityName(node: ts.ClassDeclaration | ts.MethodDeclaration | ts.MethodSignature | ts.PropertyDeclaration | ts.InterfaceDeclaration | ts.PropertySignature | ts.EnumDeclaration | ts.TypeAliasDeclaration | ts.FunctionDeclaration, tag: ts.JSDocTag): NameNode | null; | ||
methodDeclaration(node: ts.MethodDeclaration | ts.MethodSignature): FieldDefinitionNode | null; | ||
collectMethodType(node: ts.TypeNode): TypeNode | null; | ||
collectPropertyType(node: ts.TypeNode): TypeNode | null; | ||
maybeUnwrapePromise(node: ts.TypeNode): ts.TypeNode | null; | ||
collectDescription(node: ts.Node): StringValueNode | null; | ||
collectDeprecated(node: ts.Node): ConstDirectiveNode | null; | ||
property(node: ts.PropertyDeclaration | ts.PropertySignature): FieldDefinitionNode | null; | ||
collectType(node: ts.TypeNode): TypeNode | null; | ||
typeReference(node: ts.TypeReferenceNode): TypeNode | null; | ||
isNullish(node: ts.Node): boolean; | ||
expectIdentifier(node: ts.Node): ts.Identifier | null; | ||
findTag(node: ts.Node, tagName: string): ts.JSDocTag | null; | ||
handleErrorBubbling(parentNode: ts.Node, type: TypeNode): TypeNode; | ||
methodNameDirective(nameNode: ts.Node, name: string): ConstDirectiveNode; | ||
exportDirective(nameNode: ts.Node, filename: string, functionName: string): ConstDirectiveNode; | ||
/** GraphQL AST node helper methods */ | ||
gqlName(node: ts.Node, value: string): NameNode; | ||
gqlNamedType(node: ts.Node, value: string): NamedTypeNode; | ||
gqlNonNullType(node: ts.Node, type: TypeNode): NonNullTypeNode; | ||
gqlNullableType(type: TypeNode): NamedTypeNode | ListTypeNode; | ||
gqlListType(node: ts.Node, type: TypeNode): ListTypeNode; | ||
gqlConstArgument(node: ts.Node, name: NameNode, value: ConstValueNode): ConstArgumentNode; | ||
gqlConstDirective(node: ts.Node, name: NameNode, args: ReadonlyArray<ConstArgumentNode>): ConstDirectiveNode; | ||
gqlString(node: ts.Node, value: string): StringValueNode; | ||
} | ||
export {}; | ||
export declare function extract(options: ConfigOptions, sourceFile: ts.SourceFile): DiagnosticsResult<ExtractionSnapshot>; |
@@ -29,4 +29,4 @@ "use strict"; | ||
}; | ||
exports.__esModule = true; | ||
exports.Extractor = exports.ALL_TAGS = exports.KILLS_PARENT_ON_EXCEPTION_TAG = exports.INPUT_TAG = exports.UNION_TAG = exports.ENUM_TAG = exports.INTERFACE_TAG = exports.SCALAR_TAG = exports.FIELD_TAG = exports.TYPE_TAG = exports.ISSUE_URL = exports.LIBRARY_NAME = exports.LIBRARY_IMPORT_NAME = void 0; | ||
Object.defineProperty(exports, "__esModule", { value: true }); | ||
exports.extract = exports.ALL_TAGS = exports.KILLS_PARENT_ON_EXCEPTION_TAG = exports.IMPLEMENTS_TAG_DEPRECATED = exports.INPUT_TAG = exports.UNION_TAG = exports.ENUM_TAG = exports.INTERFACE_TAG = exports.SCALAR_TAG = exports.FIELD_TAG = exports.TYPE_TAG = exports.ISSUE_URL = exports.LIBRARY_NAME = exports.LIBRARY_IMPORT_NAME = void 0; | ||
var graphql_1 = require("graphql"); | ||
@@ -36,5 +36,6 @@ var DiagnosticError_1 = require("./utils/DiagnosticError"); | ||
var TypeContext_1 = require("./TypeContext"); | ||
var serverDirectives_1 = require("./serverDirectives"); | ||
var E = require("./Errors"); | ||
var JSDoc_1 = require("./utils/JSDoc"); | ||
var GraphQLConstructor_1 = require("./GraphQLConstructor"); | ||
var gratsRoot_1 = require("./gratsRoot"); | ||
exports.LIBRARY_IMPORT_NAME = "grats"; | ||
@@ -50,3 +51,5 @@ exports.LIBRARY_NAME = "Grats"; | ||
exports.INPUT_TAG = "gqlInput"; | ||
exports.IMPLEMENTS_TAG_DEPRECATED = "gqlImplements"; | ||
exports.KILLS_PARENT_ON_EXCEPTION_TAG = "killsParentOnException"; | ||
// All the tags that start with gql | ||
exports.ALL_TAGS = [ | ||
@@ -62,2 +65,3 @@ exports.TYPE_TAG, | ||
var DEPRECATED_TAG = "deprecated"; | ||
var OPERATION_TYPES = new Set(["Query", "Mutation", "Subscription"]); | ||
/** | ||
@@ -73,10 +77,26 @@ * Extracts GraphQL definitions from TypeScript source code. | ||
*/ | ||
function extract(options, sourceFile) { | ||
var extractor = new Extractor(options); | ||
return extractor.extract(sourceFile); | ||
} | ||
exports.extract = extract; | ||
var Extractor = /** @class */ (function () { | ||
function Extractor(sourceFile, ctx, buildOptions) { | ||
function Extractor(buildOptions) { | ||
this.definitions = []; | ||
// Snapshot data | ||
this.unresolvedNames = new Map(); | ||
this.nameDefinitions = new Map(); | ||
this.contextReferences = []; | ||
this.typesWithTypenameField = new Set(); | ||
this.interfaceDeclarations = []; | ||
this.errors = []; | ||
this.sourceFile = sourceFile; | ||
this.ctx = ctx; | ||
this.configOptions = buildOptions; | ||
this.gql = new GraphQLConstructor_1.GraphQLConstructor(); | ||
} | ||
Extractor.prototype.markUnresolvedType = function (node, name) { | ||
this.unresolvedNames.set(node, name); | ||
}; | ||
Extractor.prototype.recordTypeName = function (node, name, kind) { | ||
this.nameDefinitions.set(node, { name: name, kind: kind }); | ||
}; | ||
// Traverse all nodes, checking each one for its JSDoc tags. | ||
@@ -86,5 +106,5 @@ // If we find a tag we recognize, we extract the relevant information, | ||
// supported. | ||
Extractor.prototype.extract = function () { | ||
Extractor.prototype.extract = function (sourceFile) { | ||
var _this = this; | ||
(0, JSDoc_1.traverseJSDocTags)(this.sourceFile, function (node, tag) { | ||
(0, JSDoc_1.traverseJSDocTags)(sourceFile, function (node, tag) { | ||
var e_1, _a; | ||
@@ -114,3 +134,4 @@ switch (tag.tagName.text) { | ||
} | ||
else if (!(ts.isMethodDeclaration(node) || | ||
else if (!(ts.isParameter(node) || | ||
ts.isMethodDeclaration(node) || | ||
ts.isPropertyDeclaration(node) || | ||
@@ -121,3 +142,3 @@ ts.isMethodSignature(node) || | ||
// Note: Keep this in sync with `collectFields` | ||
_this.reportUnhandled(node, E.fieldTagOnWrongNode()); | ||
_this.reportUnhandled(node, "field", E.fieldTagOnWrongNode()); | ||
} | ||
@@ -150,3 +171,3 @@ break; | ||
try { | ||
if (ALL_TAGS_1_1 && !ALL_TAGS_1_1.done && (_a = ALL_TAGS_1["return"])) _a.call(ALL_TAGS_1); | ||
if (ALL_TAGS_1_1 && !ALL_TAGS_1_1.done && (_a = ALL_TAGS_1.return)) _a.call(ALL_TAGS_1); | ||
} | ||
@@ -164,3 +185,10 @@ finally { if (e_1) throw e_1.error; } | ||
} | ||
return (0, DiagnosticError_1.ok)(this.definitions); | ||
return (0, DiagnosticError_1.ok)({ | ||
definitions: this.definitions, | ||
unresolvedNames: this.unresolvedNames, | ||
nameDefinitions: this.nameDefinitions, | ||
contextReferences: this.contextReferences, | ||
typesWithTypenameField: this.typesWithTypenameField, | ||
interfaceDeclarations: this.interfaceDeclarations, | ||
}); | ||
}; | ||
@@ -226,13 +254,3 @@ Extractor.prototype.extractType = function (node, tag) { | ||
Extractor.prototype.report = function (node, message, relatedInformation) { | ||
var start = node.getStart(); | ||
var length = node.getEnd() - start; | ||
this.errors.push({ | ||
messageText: message, | ||
file: this.sourceFile, | ||
code: DiagnosticError_1.FAKE_ERROR_CODE, | ||
category: ts.DiagnosticCategory.Error, | ||
start: start, | ||
length: length, | ||
relatedInformation: relatedInformation | ||
}); | ||
this.errors.push((0, DiagnosticError_1.tsErr)(node, message, relatedInformation)); | ||
return null; | ||
@@ -242,35 +260,7 @@ }; | ||
// Gives the user a path forward if they think we should be able to infer this type. | ||
Extractor.prototype.reportUnhandled = function (node, message, relatedInformation) { | ||
var suggestion = "If you think ".concat(exports.LIBRARY_NAME, " should be able to infer this type, please report an issue at ").concat(exports.ISSUE_URL, "."); | ||
Extractor.prototype.reportUnhandled = function (node, positionKind, message, relatedInformation) { | ||
var suggestion = "If you think ".concat(exports.LIBRARY_NAME, " should be able to infer this ").concat(positionKind, ", please report an issue at ").concat(exports.ISSUE_URL, "."); | ||
var completedMessage = "".concat(message, "\n\n").concat(suggestion); | ||
return this.report(node, completedMessage, relatedInformation); | ||
}; | ||
Extractor.prototype.related = function (node, message) { | ||
return { | ||
category: ts.DiagnosticCategory.Message, | ||
code: 0, | ||
file: node.getSourceFile(), | ||
start: node.getStart(), | ||
length: node.getWidth(), | ||
messageText: message | ||
}; | ||
}; | ||
Extractor.prototype.diagnosticAnnotatedLocation = function (node) { | ||
var start = node.getStart(); | ||
var end = node.getEnd(); | ||
return { start: start, length: end - start, filepath: this.sourceFile }; | ||
}; | ||
// TODO: This is potentially quite expensive, and we only need it if we report | ||
// an error at one of these locations. We could consider some trick to return a | ||
// proxy object that would lazily compute the line/column info. | ||
Extractor.prototype.loc = function (node) { | ||
var source = new graphql_1.Source(this.sourceFile.text, this.sourceFile.fileName); | ||
var startToken = this.gqlDummyToken(node.getStart()); | ||
var endToken = this.gqlDummyToken(node.getEnd()); | ||
return new graphql_1.Location(startToken, endToken, source); | ||
}; | ||
Extractor.prototype.gqlDummyToken = function (pos) { | ||
var _a = this.sourceFile.getLineAndCharacterOfPosition(pos), line = _a.line, character = _a.character; | ||
return new graphql_1.Token(graphql_1.TokenKind.SOF, pos, pos, line, character, undefined); | ||
}; | ||
/** TypeScript traversals */ | ||
@@ -285,3 +275,3 @@ Extractor.prototype.unionTypeAliasDeclaration = function (node, tag) { | ||
} | ||
var description = this.collectDescription(node.name); | ||
var description = this.collectDescription(node); | ||
var types = []; | ||
@@ -292,6 +282,6 @@ try { | ||
if (!ts.isTypeReferenceNode(member)) { | ||
return this.reportUnhandled(member, E.expectedUnionTypeReference()); | ||
return this.reportUnhandled(member, "union member", E.expectedUnionTypeReference()); | ||
} | ||
var namedType = this.gqlNamedType(member.typeName, TypeContext_1.UNRESOLVED_REFERENCE_NAME); | ||
this.ctx.markUnresolvedType(member.typeName, namedType.name); | ||
var namedType = this.gql.namedType(member.typeName, TypeContext_1.UNRESOLVED_REFERENCE_NAME); | ||
this.markUnresolvedType(member.typeName, namedType.name); | ||
types.push(namedType); | ||
@@ -303,14 +293,8 @@ } | ||
try { | ||
if (_c && !_c.done && (_a = _b["return"])) _a.call(_b); | ||
if (_c && !_c.done && (_a = _b.return)) _a.call(_b); | ||
} | ||
finally { if (e_2) throw e_2.error; } | ||
} | ||
this.ctx.recordTypeName(node.name, name.value); | ||
this.definitions.push({ | ||
kind: graphql_1.Kind.UNION_TYPE_DEFINITION, | ||
loc: this.loc(node), | ||
description: description !== null && description !== void 0 ? description : undefined, | ||
name: name, | ||
types: types | ||
}); | ||
this.recordTypeName(node.name, name, "UNION"); | ||
this.definitions.push(this.gql.unionTypeDefinition(node, name, types, description)); | ||
}; | ||
@@ -334,5 +318,6 @@ Extractor.prototype.functionDeclarationExtendType = function (node, tag) { | ||
} | ||
var type = this.collectMethodType(node.type); | ||
if (type == null) | ||
var returnType = this.collectReturnType(node.type); | ||
if (returnType == null) | ||
return null; | ||
var type = returnType.type, isStream = returnType.isStream; | ||
var args = null; | ||
@@ -343,3 +328,7 @@ var argsParam = node.parameters[1]; | ||
} | ||
var description = this.collectDescription(funcName); | ||
var context = node.parameters[2]; | ||
if (context != null) { | ||
this.validateContextParameter(context); | ||
} | ||
var description = this.collectDescription(node); | ||
if (!ts.isSourceFile(node.parent)) { | ||
@@ -349,8 +338,12 @@ return this.report(node, E.functionFieldNotTopLevel()); | ||
// TODO: Does this work in the browser? | ||
var filename = this.ctx.getDestFilePath(node.parent); | ||
var tsModulePath = (0, gratsRoot_1.relativePath)(node.getSourceFile().fileName); | ||
var directives = [ | ||
this.exportDirective(funcName, filename, funcName.text), | ||
this.gql.exportedDirective(funcName, { | ||
tsModulePath: tsModulePath, | ||
exportedFunctionName: funcName.text, | ||
argCount: node.parameters.length, | ||
}), | ||
]; | ||
if (funcName.text !== name.value) { | ||
directives.push(this.methodNameDirective(funcName, funcName.text)); | ||
if (isStream) { | ||
directives.push(this.gql.asyncIterableDirective(node.type)); | ||
} | ||
@@ -361,18 +354,4 @@ var deprecated = this.collectDeprecated(node); | ||
} | ||
this.definitions.push({ | ||
kind: graphql_1.Kind.OBJECT_TYPE_EXTENSION, | ||
loc: this.loc(node), | ||
name: typeName, | ||
fields: [ | ||
{ | ||
kind: graphql_1.Kind.FIELD_DEFINITION, | ||
loc: this.loc(node), | ||
description: description || undefined, | ||
name: name, | ||
arguments: args || undefined, | ||
type: this.handleErrorBubbling(node, type), | ||
directives: directives.length === 0 ? undefined : directives | ||
}, | ||
] | ||
}); | ||
var field = this.gql.fieldDefinition(node, name, this.handleErrorBubbling(node, type), args, directives, description); | ||
this.definitions.push(this.gql.abstractFieldDefinition(node, typeName, field)); | ||
}; | ||
@@ -387,4 +366,4 @@ Extractor.prototype.typeReferenceFromParam = function (typeParam) { | ||
var nameNode = typeParam.type.typeName; | ||
var typeName = this.gqlName(nameNode, TypeContext_1.UNRESOLVED_REFERENCE_NAME); | ||
this.ctx.markUnresolvedType(nameNode, typeName); | ||
var typeName = this.gql.name(nameNode, TypeContext_1.UNRESOLVED_REFERENCE_NAME); | ||
this.markUnresolvedType(nameNode, typeName); | ||
return typeName; | ||
@@ -416,10 +395,5 @@ }; | ||
return null; | ||
var description = this.collectDescription(node.name); | ||
this.ctx.recordTypeName(node.name, name.value); | ||
this.definitions.push({ | ||
kind: graphql_1.Kind.SCALAR_TYPE_DEFINITION, | ||
loc: this.loc(node), | ||
description: description !== null && description !== void 0 ? description : undefined, | ||
name: name | ||
}); | ||
var description = this.collectDescription(node); | ||
this.recordTypeName(node.name, name, "SCALAR"); | ||
this.definitions.push(this.gql.scalarTypeDefinition(node, name, description)); | ||
}; | ||
@@ -430,14 +404,7 @@ Extractor.prototype.inputTypeAliasDeclaration = function (node, tag) { | ||
return null; | ||
var description = this.collectDescription(node.name); | ||
this.ctx.recordTypeName(node.name, name.value); | ||
var description = this.collectDescription(node); | ||
this.recordTypeName(node.name, name, "INPUT_OBJECT"); | ||
var fields = this.collectInputFields(node); | ||
var deprecatedDirective = this.collectDeprecated(node); | ||
this.definitions.push({ | ||
kind: graphql_1.Kind.INPUT_OBJECT_TYPE_DEFINITION, | ||
loc: this.loc(node), | ||
description: description !== null && description !== void 0 ? description : undefined, | ||
name: name, | ||
fields: fields !== null && fields !== void 0 ? fields : undefined, | ||
directives: deprecatedDirective == null ? undefined : [deprecatedDirective] | ||
}); | ||
this.definitions.push(this.gql.inputObjectTypeDefinition(node, name, fields, deprecatedDirective == null ? null : [deprecatedDirective], description)); | ||
}; | ||
@@ -448,3 +415,3 @@ Extractor.prototype.collectInputFields = function (node) { | ||
if (!ts.isTypeLiteralNode(node.type)) { | ||
return this.reportUnhandled(node, E.inputTypeNotLiteral()); | ||
return this.reportUnhandled(node, "input", E.inputTypeNotLiteral()); | ||
} | ||
@@ -455,3 +422,3 @@ try { | ||
if (!ts.isPropertySignature(member)) { | ||
this.reportUnhandled(member, E.inputTypeFieldNotProperty()); | ||
this.reportUnhandled(member, "input field", E.inputTypeFieldNotProperty()); | ||
continue; | ||
@@ -467,3 +434,3 @@ } | ||
try { | ||
if (_c && !_c.done && (_a = _b["return"])) _a.call(_b); | ||
if (_c && !_c.done && (_a = _b.return)) _a.call(_b); | ||
} | ||
@@ -484,18 +451,10 @@ finally { if (e_3) throw e_3.error; } | ||
return null; | ||
var type = node.questionToken == null ? inner : this.gqlNullableType(inner); | ||
var description = this.collectDescription(node.name); | ||
var type = node.questionToken == null ? inner : this.gql.nullableType(inner); | ||
var description = this.collectDescription(node); | ||
var deprecatedDirective = this.collectDeprecated(node); | ||
return { | ||
kind: graphql_1.Kind.INPUT_VALUE_DEFINITION, | ||
loc: this.loc(node), | ||
description: description !== null && description !== void 0 ? description : undefined, | ||
name: this.gqlName(id, id.text), | ||
type: type, | ||
defaultValue: undefined, | ||
directives: deprecatedDirective == null ? undefined : [deprecatedDirective] | ||
}; | ||
return this.gql.inputValueDefinition(node, this.gql.name(id, id.text), type, deprecatedDirective == null ? null : [deprecatedDirective], null, description); | ||
}; | ||
Extractor.prototype.typeClassDeclaration = function (node, tag) { | ||
if (node.name == null) { | ||
return this.report(node, E.typeTagOnUnamedClass()); | ||
return this.report(node, E.typeTagOnUnnamedClass()); | ||
} | ||
@@ -505,17 +464,17 @@ var name = this.entityName(node, tag); | ||
return null; | ||
var description = this.collectDescription(node.name); | ||
this.validateOperationTypes(node.name, name.value); | ||
var description = this.collectDescription(node); | ||
var fields = this.collectFields(node); | ||
var interfaces = this.collectInterfaces(node); | ||
this.ctx.recordTypeName(node.name, name.value); | ||
this.recordTypeName(node.name, name, "TYPE"); | ||
this.checkForTypenameProperty(node, name.value); | ||
this.definitions.push({ | ||
kind: graphql_1.Kind.OBJECT_TYPE_DEFINITION, | ||
loc: this.loc(node), | ||
description: description !== null && description !== void 0 ? description : undefined, | ||
directives: undefined, | ||
name: name, | ||
fields: fields, | ||
interfaces: interfaces !== null && interfaces !== void 0 ? interfaces : undefined | ||
}); | ||
this.definitions.push(this.gql.objectTypeDefinition(node, name, fields, interfaces, description)); | ||
}; | ||
Extractor.prototype.validateOperationTypes = function (node, name) { | ||
// TODO: If we start supporting defining operation types using | ||
// non-standard names, we will need to update this logic. | ||
if (OPERATION_TYPES.has(name)) { | ||
this.report(node, E.operationTypeNotUnknown()); | ||
} | ||
}; | ||
Extractor.prototype.typeInterfaceDeclaration = function (node, tag) { | ||
@@ -525,16 +484,9 @@ var name = this.entityName(node, tag); | ||
return null; | ||
var description = this.collectDescription(node.name); | ||
this.validateOperationTypes(node.name, name.value); | ||
var description = this.collectDescription(node); | ||
var fields = this.collectFields(node); | ||
var interfaces = this.collectInterfaces(node); | ||
this.ctx.recordTypeName(node.name, name.value); | ||
this.recordTypeName(node.name, name, "INTERFACE"); | ||
this.checkForTypenameProperty(node, name.value); | ||
this.definitions.push({ | ||
kind: graphql_1.Kind.OBJECT_TYPE_DEFINITION, | ||
loc: this.loc(node), | ||
description: description !== null && description !== void 0 ? description : undefined, | ||
directives: undefined, | ||
name: name, | ||
fields: fields, | ||
interfaces: interfaces !== null && interfaces !== void 0 ? interfaces : undefined | ||
}); | ||
this.definitions.push(this.gql.objectTypeDefinition(node, name, fields, interfaces, description)); | ||
}; | ||
@@ -545,21 +497,21 @@ Extractor.prototype.typeTypeAliasDeclaration = function (node, tag) { | ||
return null; | ||
if (!ts.isTypeLiteralNode(node.type)) { | ||
this.reportUnhandled(node.type, E.typeTagOnAliasOfNonObject()); | ||
return; | ||
var fields = []; | ||
var interfaces = null; | ||
if (ts.isTypeLiteralNode(node.type)) { | ||
this.validateOperationTypes(node.type, name.value); | ||
fields = this.collectFields(node.type); | ||
interfaces = this.collectInterfaces(node); | ||
this.checkForTypenameProperty(node.type, name.value); | ||
} | ||
var description = this.collectDescription(node.name); | ||
var fields = this.collectFields(node.type); | ||
this.ctx.recordTypeName(node.name, name.value); | ||
this.checkForTypenameProperty(node.type, name.value); | ||
this.definitions.push({ | ||
kind: graphql_1.Kind.OBJECT_TYPE_DEFINITION, | ||
loc: this.loc(node), | ||
description: description !== null && description !== void 0 ? description : undefined, | ||
directives: undefined, | ||
name: name, | ||
fields: fields, | ||
// I don't believe there is a reasonable way to specify that a type | ||
// implements an interface. | ||
interfaces: undefined | ||
}); | ||
else if (node.type.kind === ts.SyntaxKind.UnknownKeyword) { | ||
// This is fine, we just don't know what it is. This should be the expected | ||
// case for operation types such as `Query`, `Mutation`, and `Subscription` | ||
// where there is not strong convention around. | ||
} | ||
else { | ||
return this.report(node.type, E.typeTagOnAliasOfNonObjectOrUnknown()); | ||
} | ||
var description = this.collectDescription(node); | ||
this.recordTypeName(node.name, name, "TYPE"); | ||
this.definitions.push(this.gql.objectTypeDefinition(node, name, fields, interfaces, description)); | ||
}; | ||
@@ -572,3 +524,3 @@ Extractor.prototype.checkForTypenameProperty = function (node, expectedName) { | ||
if (hasTypename) { | ||
this.ctx.recordHasTypenameField(expectedName); | ||
this.typesWithTypenameField.add(expectedName); | ||
} | ||
@@ -632,15 +584,40 @@ }; | ||
Extractor.prototype.collectInterfaces = function (node) { | ||
this.reportTagInterfaces(node); | ||
return ts.isClassDeclaration(node) || ts.isInterfaceDeclaration(node) | ||
? this.collectHeritageInterfaces(node) | ||
: null; | ||
}; | ||
Extractor.prototype.reportTagInterfaces = function (node) { | ||
var tag = this.findTag(node, exports.IMPLEMENTS_TAG_DEPRECATED); | ||
if (tag == null) | ||
return null; | ||
if (node.kind === ts.SyntaxKind.ClassDeclaration) { | ||
this.report(tag, E.implementsTagOnClass()); | ||
} | ||
if (node.kind === ts.SyntaxKind.InterfaceDeclaration) { | ||
this.report(tag, E.implementsTagOnInterface()); | ||
} | ||
if (node.kind === ts.SyntaxKind.TypeAliasDeclaration) { | ||
this.report(tag, E.implementsTagOnTypeAlias()); | ||
} | ||
}; | ||
Extractor.prototype.collectHeritageInterfaces = function (node) { | ||
var _this = this; | ||
if (node.heritageClauses == null) | ||
return null; | ||
var maybeInterfaces = node.heritageClauses.flatMap(function (clause) { | ||
if (clause.token !== ts.SyntaxKind.ImplementsKeyword) | ||
return []; | ||
return clause.types.map(function (type) { | ||
if (!ts.isIdentifier(type.expression)) { | ||
// TODO: Are there valid cases we want to cover here? | ||
return null; | ||
} | ||
var namedType = _this.gqlNamedType(type.expression, TypeContext_1.UNRESOLVED_REFERENCE_NAME); | ||
_this.ctx.markUnresolvedType(type.expression, namedType.name); | ||
var maybeInterfaces = node.heritageClauses | ||
.filter(function (clause) { | ||
if (node.kind === ts.SyntaxKind.ClassDeclaration) { | ||
return clause.token === ts.SyntaxKind.ImplementsKeyword; | ||
} | ||
// Interfaces can only have extends clauses, and those are allowed. | ||
return true; | ||
}) | ||
.flatMap(function (clause) { | ||
return clause.types | ||
.map(function (type) { return type.expression; }) | ||
.filter(function (expression) { return ts.isIdentifier(expression); }) | ||
.map(function (expression) { | ||
var namedType = _this.gql.namedType(expression, TypeContext_1.UNRESOLVED_REFERENCE_NAME); | ||
_this.markUnresolvedType(expression, namedType.name); | ||
return namedType; | ||
@@ -655,4 +632,8 @@ }); | ||
}; | ||
Extractor.prototype.hasGqlTag = function (node) { | ||
return ts.getJSDocTags(node).some(function (tag) { | ||
return exports.ALL_TAGS.includes(tag.tagName.text); | ||
}); | ||
}; | ||
Extractor.prototype.interfaceInterfaceDeclaration = function (node, tag) { | ||
var _this = this; | ||
var name = this.entityName(node, tag); | ||
@@ -662,31 +643,8 @@ if (name == null || name.value == null) { | ||
} | ||
// Prevent using merged interfaces as GraphQL interfaces. | ||
// https://www.typescriptlang.org/docs/handbook/declaration-merging.html#merging-interfaces | ||
var symbol = this.ctx.checker.getSymbolAtLocation(node.name); | ||
if (symbol != null && | ||
symbol.declarations != null && | ||
symbol.declarations.length > 1) { | ||
var otherLocations = symbol.declarations | ||
.filter(function (d) { return d !== node; }) | ||
.map(function (d) { | ||
var _a; | ||
var locNode = (_a = ts.getNameOfDeclaration(d)) !== null && _a !== void 0 ? _a : d; | ||
return _this.related(locNode, "Other declaration"); | ||
}); | ||
return this.report(node.name, E.mergedInterfaces(name.value), otherLocations); | ||
} | ||
var description = this.collectDescription(node.name); | ||
this.interfaceDeclarations.push(node); | ||
var description = this.collectDescription(node); | ||
var interfaces = this.collectInterfaces(node); | ||
var fields = this.collectFields(node); | ||
this.ctx.recordTypeName(node.name, name.value); | ||
// While GraphQL supports interfaces that extend other interfaces, | ||
// TypeScript does not. So we can't support that here either. | ||
// In the future we could support classes that act as interfaces through | ||
// inheritance. | ||
this.definitions.push({ | ||
kind: graphql_1.Kind.INTERFACE_TYPE_DEFINITION, | ||
loc: this.loc(node), | ||
description: description || undefined, | ||
name: name, | ||
fields: fields | ||
}); | ||
this.recordTypeName(node.name, name, "INTERFACE"); | ||
this.definitions.push(this.gql.interfaceTypeDefinition(node, name, fields, interfaces, description)); | ||
}; | ||
@@ -697,5 +655,26 @@ Extractor.prototype.collectFields = function (node) { | ||
ts.forEachChild(node, function (node) { | ||
var e_4, _a; | ||
if (ts.isConstructorDeclaration(node)) { | ||
try { | ||
// Handle parameter properties | ||
// https://www.typescriptlang.org/docs/handbook/2/classes.html#parameter-properties | ||
for (var _b = __values(node.parameters), _c = _b.next(); !_c.done; _c = _b.next()) { | ||
var param = _c.value; | ||
var field = _this.constructorParam(param); | ||
if (field != null) { | ||
fields.push(field); | ||
} | ||
} | ||
} | ||
catch (e_4_1) { e_4 = { error: e_4_1 }; } | ||
finally { | ||
try { | ||
if (_c && !_c.done && (_a = _b.return)) _a.call(_b); | ||
} | ||
finally { if (e_4) throw e_4.error; } | ||
} | ||
} | ||
if (ts.isMethodDeclaration(node) || ts.isMethodSignature(node)) { | ||
var field = _this.methodDeclaration(node); | ||
if (field) { | ||
if (field != null) { | ||
fields.push(field); | ||
@@ -714,4 +693,56 @@ } | ||
}; | ||
Extractor.prototype.constructorParam = function (node) { | ||
var tag = this.findTag(node, exports.FIELD_TAG); | ||
if (tag == null) | ||
return null; | ||
if (node.modifiers == null) { | ||
return this.report(node, E.parameterWithoutModifiers()); | ||
} | ||
var isParameterProperty = node.modifiers.some(function (modifier) { | ||
return modifier.kind === ts.SyntaxKind.PublicKeyword || | ||
modifier.kind === ts.SyntaxKind.PrivateKeyword || | ||
modifier.kind === ts.SyntaxKind.ProtectedKeyword || | ||
modifier.kind === ts.SyntaxKind.ReadonlyKeyword; | ||
}); | ||
if (!isParameterProperty) { | ||
return this.report(node, E.parameterWithoutModifiers()); | ||
} | ||
var notPublic = node.modifiers.find(function (modifier) { | ||
return modifier.kind === ts.SyntaxKind.PrivateKeyword || | ||
modifier.kind === ts.SyntaxKind.ProtectedKeyword; | ||
}); | ||
if (notPublic != null) { | ||
return this.report(notPublic, E.parameterPropertyNotPublic()); | ||
} | ||
var name = this.entityName(node, tag); | ||
if (name == null) | ||
return null; | ||
if (node.type == null) { | ||
return this.report(node, E.parameterPropertyMissingType()); | ||
} | ||
var id = node.name; | ||
if (ts.isArrayBindingPattern(id) || ts.isObjectBindingPattern(id)) { | ||
// TypeScript triggers an error if a binding pattern is used for a | ||
// parameter property, so we don't need to report them. | ||
// https://www.typescriptlang.org/play?#code/MYGwhgzhAEBiD29oG8BQ1rHgOwgFwCcBXYPeAgCgAciAjEAS2BQDNEBfAShXdXaA | ||
return null; | ||
} | ||
var directives = []; | ||
if (id.text !== name.value) { | ||
directives = [ | ||
this.gql.propertyNameDirective(node.name, { name: id.text }), | ||
]; | ||
} | ||
var type = this.collectType(node.type); | ||
if (type == null) | ||
return null; | ||
var deprecated = this.collectDeprecated(node); | ||
if (deprecated != null) { | ||
directives.push(deprecated); | ||
} | ||
var description = this.collectDescription(node); | ||
return this.gql.fieldDefinition(node, name, this.handleErrorBubbling(node, type), null, directives, description); | ||
}; | ||
Extractor.prototype.collectArgs = function (argsParam) { | ||
var e_4, _a; | ||
var e_5, _a; | ||
var args = []; | ||
@@ -722,3 +753,3 @@ var argsType = argsParam.type; | ||
} | ||
if (argsType.kind === ts.SyntaxKind.NeverKeyword) { | ||
if (argsType.kind === ts.SyntaxKind.UnknownKeyword) { | ||
return []; | ||
@@ -742,8 +773,8 @@ } | ||
} | ||
catch (e_4_1) { e_4 = { error: e_4_1 }; } | ||
catch (e_5_1) { e_5 = { error: e_5_1 }; } | ||
finally { | ||
try { | ||
if (_c && !_c.done && (_a = _b["return"])) _a.call(_b); | ||
if (_c && !_c.done && (_a = _b.return)) _a.call(_b); | ||
} | ||
finally { if (e_4) throw e_4.error; } | ||
finally { if (e_5) throw e_5.error; } | ||
} | ||
@@ -753,3 +784,3 @@ return args; | ||
Extractor.prototype.collectArgDefaults = function (node) { | ||
var e_5, _a; | ||
var e_6, _a; | ||
var defaults = new Map(); | ||
@@ -766,8 +797,8 @@ try { | ||
} | ||
catch (e_5_1) { e_5 = { error: e_5_1 }; } | ||
catch (e_6_1) { e_6 = { error: e_6_1 }; } | ||
finally { | ||
try { | ||
if (_c && !_c.done && (_a = _b["return"])) _a.call(_b); | ||
if (_c && !_c.done && (_a = _b.return)) _a.call(_b); | ||
} | ||
finally { if (e_5) throw e_5.error; } | ||
finally { if (e_6) throw e_6.error; } | ||
} | ||
@@ -778,19 +809,20 @@ return defaults; | ||
if (ts.isStringLiteral(node)) { | ||
return { kind: graphql_1.Kind.STRING, loc: this.loc(node), value: node.text }; | ||
return this.gql.string(node, node.text); | ||
} | ||
else if (ts.isNumericLiteral(node)) { | ||
var kind = node.text.includes(".") ? graphql_1.Kind.FLOAT : graphql_1.Kind.INT; | ||
return { kind: kind, loc: this.loc(node), value: node.text }; | ||
return node.text.includes(".") | ||
? this.gql.float(node, node.text) | ||
: this.gql.int(node, node.text); | ||
} | ||
else if (this.isNullish(node)) { | ||
return { kind: graphql_1.Kind.NULL, loc: this.loc(node) }; | ||
return this.gql.null(node); | ||
} | ||
else if (node.kind === ts.SyntaxKind.TrueKeyword) { | ||
return { kind: graphql_1.Kind.BOOLEAN, loc: this.loc(node), value: true }; | ||
return this.gql.boolean(node, true); | ||
} | ||
else if (node.kind === ts.SyntaxKind.FalseKeyword) { | ||
return { kind: graphql_1.Kind.BOOLEAN, loc: this.loc(node), value: false }; | ||
return this.gql.boolean(node, false); | ||
} | ||
else if (ts.isObjectLiteralExpression(node)) { | ||
return this.cellectObjectLiteral(node); | ||
return this.collectObjectLiteral(node); | ||
} | ||
@@ -800,6 +832,6 @@ else if (ts.isArrayLiteralExpression(node)) { | ||
} | ||
return this.reportUnhandled(node, E.defaultValueIsNotLiteral()); | ||
return this.reportUnhandled(node, "constant value", E.defaultValueIsNotLiteral()); | ||
}; | ||
Extractor.prototype.collectArrayLiteral = function (node) { | ||
var e_6, _a; | ||
var e_7, _a; | ||
var values = []; | ||
@@ -819,8 +851,8 @@ var errors = false; | ||
} | ||
catch (e_6_1) { e_6 = { error: e_6_1 }; } | ||
catch (e_7_1) { e_7 = { error: e_7_1 }; } | ||
finally { | ||
try { | ||
if (_c && !_c.done && (_a = _b["return"])) _a.call(_b); | ||
if (_c && !_c.done && (_a = _b.return)) _a.call(_b); | ||
} | ||
finally { if (e_6) throw e_6.error; } | ||
finally { if (e_7) throw e_7.error; } | ||
} | ||
@@ -830,10 +862,6 @@ if (errors) { | ||
} | ||
return { | ||
kind: graphql_1.Kind.LIST, | ||
loc: this.loc(node), | ||
values: values | ||
}; | ||
return this.gql.list(node, values); | ||
}; | ||
Extractor.prototype.cellectObjectLiteral = function (node) { | ||
var e_7, _a; | ||
Extractor.prototype.collectObjectLiteral = function (node) { | ||
var e_8, _a; | ||
var fields = []; | ||
@@ -853,8 +881,8 @@ var errors = false; | ||
} | ||
catch (e_7_1) { e_7 = { error: e_7_1 }; } | ||
catch (e_8_1) { e_8 = { error: e_8_1 }; } | ||
finally { | ||
try { | ||
if (_c && !_c.done && (_a = _b["return"])) _a.call(_b); | ||
if (_c && !_c.done && (_a = _b.return)) _a.call(_b); | ||
} | ||
finally { if (e_7) throw e_7.error; } | ||
finally { if (e_8) throw e_8.error; } | ||
} | ||
@@ -864,14 +892,10 @@ if (errors) { | ||
} | ||
return { | ||
kind: graphql_1.Kind.OBJECT, | ||
loc: this.loc(node), | ||
fields: fields | ||
}; | ||
return this.gql.object(node, fields); | ||
}; | ||
Extractor.prototype.collectObjectField = function (node) { | ||
if (!ts.isPropertyAssignment(node)) { | ||
return this.reportUnhandled(node, E.defaultArgElementIsNotAssignment()); | ||
return this.reportUnhandled(node, "constant value", E.defaultArgElementIsNotAssignment()); | ||
} | ||
if (node.name == null) { | ||
return this.reportUnhandled(node, E.defaultArgPropertyMissingName()); | ||
return this.reportUnhandled(node, "field", E.defaultArgPropertyMissingName()); | ||
} | ||
@@ -888,8 +912,3 @@ var name = this.expectIdentifier(node.name); | ||
return null; | ||
return { | ||
kind: graphql_1.Kind.OBJECT_FIELD, | ||
loc: this.loc(node), | ||
name: this.gqlName(node.name, name.text), | ||
value: value | ||
}; | ||
return this.gql.constObjectField(node, this.gql.name(node.name, name.text), value); | ||
}; | ||
@@ -911,12 +930,20 @@ Extractor.prototype.collectArg = function (node, defaults) { | ||
return null; | ||
if (type.kind !== graphql_1.Kind.NON_NULL_TYPE && !node.questionToken) { | ||
// If a field is passed an argument value, and that argument is not defined in the request, | ||
// `graphql-js` will not define the argument property. Therefore we must ensure the argument | ||
// is not just nullable, but optional. | ||
return this.report(node.name, E.expectedNullableArgumentToBeOptional()); | ||
} | ||
if (node.questionToken) { | ||
/* | ||
// TODO: Don't allow args that are optional but don't accept null | ||
if (type.kind === Kind.NON_NULL_TYPE) { | ||
return this.report(node.questionToken, E.nonNullTypeCannotBeOptional()); | ||
// Question mark means we can handle the argument being undefined in the | ||
// object literal, but if we are going to type the GraphQL arg as | ||
// optional, the code must also be able to handle an explicit null. | ||
// | ||
// TODO: This will catch { a?: string } but not { a?: string | undefined }. | ||
if (type.kind === graphql_1.Kind.NON_NULL_TYPE) { | ||
return this.report(node.questionToken, E.nonNullTypeCannotBeOptional()); | ||
} | ||
*/ | ||
type = this.gqlNullableType(type); | ||
type = this.gql.nullableType(type); | ||
} | ||
var description = this.collectDescription(node.name); | ||
var description = this.collectDescription(node); | ||
var defaultValue = null; | ||
@@ -930,11 +957,3 @@ if (defaults != null) { | ||
var deprecatedDirective = this.collectDeprecated(node); | ||
return { | ||
kind: graphql_1.Kind.INPUT_VALUE_DEFINITION, | ||
loc: this.loc(node), | ||
description: description || undefined, | ||
name: this.gqlName(node.name, node.name.text), | ||
type: type, | ||
defaultValue: defaultValue || undefined, | ||
directives: deprecatedDirective != null ? [deprecatedDirective] : undefined | ||
}; | ||
return this.gql.inputValueDefinition(node, this.gql.name(node.name, node.name.text), type, deprecatedDirective == null ? null : [deprecatedDirective], defaultValue, description); | ||
}; | ||
@@ -946,12 +965,6 @@ Extractor.prototype.enumEnumDeclaration = function (node, tag) { | ||
} | ||
var description = this.collectDescription(node.name); | ||
var description = this.collectDescription(node); | ||
var values = this.collectEnumValues(node); | ||
this.ctx.recordTypeName(node.name, name.value); | ||
this.definitions.push({ | ||
kind: graphql_1.Kind.ENUM_TYPE_DEFINITION, | ||
loc: this.loc(node), | ||
description: description || undefined, | ||
name: name, | ||
values: values | ||
}); | ||
this.recordTypeName(node.name, name, "ENUM"); | ||
this.definitions.push(this.gql.enumTypeDefinition(node, name, values, description)); | ||
}; | ||
@@ -966,15 +979,8 @@ Extractor.prototype.enumTypeAliasDeclaration = function (node, tag) { | ||
return; | ||
var description = this.collectDescription(node.name); | ||
this.ctx.recordTypeName(node.name, name.value); | ||
this.definitions.push({ | ||
kind: graphql_1.Kind.ENUM_TYPE_DEFINITION, | ||
loc: this.loc(node), | ||
description: description || undefined, | ||
name: name, | ||
values: values | ||
}); | ||
var description = this.collectDescription(node); | ||
this.recordTypeName(node.name, name, "ENUM"); | ||
this.definitions.push(this.gql.enumTypeDefinition(node, name, values, description)); | ||
}; | ||
Extractor.prototype.enumTypeAliasVariants = function (node) { | ||
var e_8, _a; | ||
var _b; | ||
var e_9, _a; | ||
// Semantically we only support deriving enums from type aliases that | ||
@@ -988,12 +994,7 @@ // are unions of string literals. However, in the edge case of a union | ||
return [ | ||
{ | ||
kind: graphql_1.Kind.ENUM_VALUE_DEFINITION, | ||
name: this.gqlName(node.type.literal, node.type.literal.text), | ||
description: undefined, | ||
loc: this.loc(node) | ||
}, | ||
this.gql.enumValueDefinition(node, this.gql.name(node.type.literal, node.type.literal.text), undefined, null), | ||
]; | ||
} | ||
if (!ts.isUnionTypeNode(node.type)) { | ||
this.reportUnhandled(node.type, E.enumTagOnInvalidNode()); | ||
this.reportUnhandled(node.type, "union", E.enumTagOnInvalidNode()); | ||
return null; | ||
@@ -1003,32 +1004,7 @@ } | ||
try { | ||
for (var _c = __values(node.type.types), _d = _c.next(); !_d.done; _d = _c.next()) { | ||
var member = _d.value; | ||
// TODO: Complete this feature | ||
if (ts.isTypeReferenceNode(member)) { | ||
if (member.typeName.kind === ts.SyntaxKind.Identifier) { | ||
var symbol = this.ctx.checker.getSymbolAtLocation(member.typeName); | ||
if (((_b = symbol === null || symbol === void 0 ? void 0 : symbol.declarations) === null || _b === void 0 ? void 0 : _b.length) === 1) { | ||
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion | ||
var declaration = symbol.declarations[0]; | ||
if (ts.isTypeAliasDeclaration(declaration)) { | ||
if (ts.isLiteralTypeNode(declaration.type) && | ||
ts.isStringLiteral(declaration.type.literal)) { | ||
var deprecatedDirective = this.collectDeprecated(declaration); | ||
var memberDescription = this.collectDescription(declaration.name); | ||
values.push({ | ||
kind: graphql_1.Kind.ENUM_VALUE_DEFINITION, | ||
name: this.gqlName(declaration.type.literal, declaration.type.literal.text), | ||
directives: deprecatedDirective ? [deprecatedDirective] : [], | ||
description: memberDescription || undefined, | ||
loc: this.loc(node) | ||
}); | ||
continue; | ||
} | ||
} | ||
} | ||
} | ||
} | ||
for (var _b = __values(node.type.types), _c = _b.next(); !_c.done; _c = _b.next()) { | ||
var member = _c.value; | ||
if (!ts.isLiteralTypeNode(member) || | ||
!ts.isStringLiteral(member.literal)) { | ||
this.reportUnhandled(member, E.enumVariantNotStringLiteral()); | ||
this.reportUnhandled(member, "union member", E.enumVariantNotStringLiteral()); | ||
continue; | ||
@@ -1038,16 +1014,11 @@ } | ||
// does not allow comments attached to string literal types. | ||
values.push({ | ||
kind: graphql_1.Kind.ENUM_VALUE_DEFINITION, | ||
name: this.gqlName(member.literal, member.literal.text), | ||
description: undefined, | ||
loc: this.loc(member) | ||
}); | ||
values.push(this.gql.enumValueDefinition(node, this.gql.name(member.literal, member.literal.text), undefined, null)); | ||
} | ||
} | ||
catch (e_8_1) { e_8 = { error: e_8_1 }; } | ||
catch (e_9_1) { e_9 = { error: e_9_1 }; } | ||
finally { | ||
try { | ||
if (_d && !_d.done && (_a = _c["return"])) _a.call(_c); | ||
if (_c && !_c.done && (_a = _b.return)) _a.call(_b); | ||
} | ||
finally { if (e_8) throw e_8.error; } | ||
finally { if (e_9) throw e_9.error; } | ||
} | ||
@@ -1057,3 +1028,3 @@ return values; | ||
Extractor.prototype.collectEnumValues = function (node) { | ||
var e_9, _a; | ||
var e_10, _a; | ||
var values = []; | ||
@@ -1065,22 +1036,16 @@ try { | ||
!ts.isStringLiteral(member.initializer)) { | ||
this.reportUnhandled(member, E.enumVariantMissingInitializer()); | ||
this.reportUnhandled(member, "enum value", E.enumVariantMissingInitializer()); | ||
continue; | ||
} | ||
var description = this.collectDescription(member.name); | ||
var description = this.collectDescription(member); | ||
var deprecated = this.collectDeprecated(member); | ||
values.push({ | ||
kind: graphql_1.Kind.ENUM_VALUE_DEFINITION, | ||
loc: this.loc(member), | ||
description: description || undefined, | ||
name: this.gqlName(member.initializer, member.initializer.text), | ||
directives: deprecated ? [deprecated] : undefined | ||
}); | ||
values.push(this.gql.enumValueDefinition(member, this.gql.name(member.initializer, member.initializer.text), deprecated ? [deprecated] : undefined, description)); | ||
} | ||
} | ||
catch (e_9_1) { e_9 = { error: e_9_1 }; } | ||
catch (e_10_1) { e_10 = { error: e_10_1 }; } | ||
finally { | ||
try { | ||
if (_c && !_c.done && (_a = _b["return"])) _a.call(_b); | ||
if (_c && !_c.done && (_a = _b.return)) _a.call(_b); | ||
} | ||
finally { if (e_9) throw e_9.error; } | ||
finally { if (e_10) throw e_10.error; } | ||
} | ||
@@ -1094,3 +1059,24 @@ return values; | ||
// FIXME: Use the _value_'s location not the tag's | ||
return this.gqlName(tag, commentName); | ||
var locNode = tag; | ||
// Test for leading newlines using the raw text | ||
var hasLeadingNewlines = /\n/.test(tag.getText().trimEnd()); | ||
var hasInternalWhitespace = /\s/.test(commentName); | ||
var validationMessage = graphQLNameValidationMessage(commentName); | ||
if (hasLeadingNewlines && validationMessage == null) { | ||
// TODO: Offer quick fix. | ||
return this.report(locNode, E.graphQLNameHasLeadingNewlines(commentName, tag.tagName.text)); | ||
} | ||
if (hasLeadingNewlines || hasInternalWhitespace) { | ||
return this.report(locNode, E.graphQLTagNameHasWhitespace(tag.tagName.text)); | ||
} | ||
// No whitespace, but still invalid. We will assume they meant this to | ||
// be a GraphQL name but didn't provide a valid identifier. | ||
// | ||
// NOTE: We can't let GraphQL validation handle this, because it throws rather | ||
// than returning a validation message. Presumably because it expects token | ||
// validation to be done during lexing/parsing. | ||
if (validationMessage !== null) { | ||
return this.report(locNode, validationMessage); | ||
} | ||
return this.gql.name(locNode, commentName); | ||
} | ||
@@ -1104,4 +1090,24 @@ } | ||
return null; | ||
return this.gqlName(id, id.text); | ||
return this.gql.name(id, id.text); | ||
}; | ||
// Ensure the type of the ctx param resolves to the declaration | ||
// annotated with `@gqlContext`. | ||
Extractor.prototype.validateContextParameter = function (node) { | ||
if (node.type == null) { | ||
return this.report(node, E.expectedTypeAnnotationOnContext()); | ||
} | ||
if (node.type.kind === ts.SyntaxKind.UnknownKeyword) { | ||
// If the user just needs to define the argument to get to a later parameter, | ||
// they can use `ctx: unknown` to safely avoid triggering a Grats error. | ||
return; | ||
} | ||
if (!ts.isTypeReferenceNode(node.type)) { | ||
return this.report(node.type, E.expectedTypeAnnotationOfReferenceOnContext()); | ||
} | ||
// Check for ... | ||
if (node.dotDotDotToken != null) { | ||
return this.report(node.dotDotDotToken, E.unexpectedParamSpreadForContextParam()); | ||
} | ||
this.contextReferences.push(node.type.typeName); | ||
}; | ||
Extractor.prototype.methodDeclaration = function (node) { | ||
@@ -1117,3 +1123,6 @@ var tag = this.findTag(node, exports.FIELD_TAG); | ||
} | ||
var type = this.collectMethodType(node.type); | ||
var returnType = this.collectReturnType(node.type); | ||
if (returnType == null) | ||
return null; | ||
var type = returnType.type, isStream = returnType.isStream; | ||
// We already reported an error | ||
@@ -1127,3 +1136,7 @@ if (type == null) | ||
} | ||
var description = this.collectDescription(node.name); | ||
var context = node.parameters[1]; | ||
if (context != null) { | ||
this.validateContextParameter(context); | ||
} | ||
var description = this.collectDescription(node); | ||
var id = this.expectIdentifier(node.name); | ||
@@ -1134,4 +1147,9 @@ if (id == null) | ||
if (id.text !== name.value) { | ||
directives = [this.methodNameDirective(node.name, id.text)]; | ||
directives = [ | ||
this.gql.propertyNameDirective(node.name, { name: id.text }), | ||
]; | ||
} | ||
if (isStream) { | ||
directives.push(this.gql.asyncIterableDirective(node.type)); | ||
} | ||
var deprecated = this.collectDeprecated(node); | ||
@@ -1141,21 +1159,31 @@ if (deprecated != null) { | ||
} | ||
return { | ||
kind: graphql_1.Kind.FIELD_DEFINITION, | ||
loc: this.loc(node), | ||
description: description || undefined, | ||
name: name, | ||
arguments: args || undefined, | ||
type: this.handleErrorBubbling(node, type), | ||
directives: directives.length === 0 ? undefined : directives | ||
}; | ||
return this.gql.fieldDefinition(node, name, this.handleErrorBubbling(node, type), args, directives, description); | ||
}; | ||
Extractor.prototype.collectMethodType = function (node) { | ||
var inner = this.maybeUnwrapePromise(node); | ||
Extractor.prototype.collectReturnType = function (node) { | ||
if (ts.isTypeReferenceNode(node)) { | ||
var identifier = this.expectIdentifier(node.typeName); | ||
if (identifier == null) | ||
return null; | ||
if (identifier.text == "AsyncIterable") { | ||
if (node.typeArguments == null || node.typeArguments.length === 0) { | ||
// TODO: Better error? | ||
return this.report(node, E.promiseMissingTypeArg()); | ||
} | ||
var t_1 = this.collectType(node.typeArguments[0]); | ||
if (t_1 == null) | ||
return null; | ||
return { type: t_1, isStream: true }; | ||
} | ||
} | ||
var inner = this.maybeUnwrapPromise(node); | ||
if (inner == null) | ||
return null; | ||
return this.collectType(inner); | ||
var t = this.collectType(inner); | ||
if (t == null) | ||
return null; | ||
return { type: t, isStream: false }; | ||
}; | ||
Extractor.prototype.collectPropertyType = function (node) { | ||
// TODO: Handle function types here. | ||
var inner = this.maybeUnwrapePromise(node); | ||
var inner = this.maybeUnwrapPromise(node); | ||
if (inner == null) | ||
@@ -1165,3 +1193,3 @@ return null; | ||
}; | ||
Extractor.prototype.maybeUnwrapePromise = function (node) { | ||
Extractor.prototype.maybeUnwrapPromise = function (node) { | ||
if (ts.isTypeReferenceNode(node)) { | ||
@@ -1172,3 +1200,3 @@ var identifier = this.expectIdentifier(node.typeName); | ||
if (identifier.text === "Promise") { | ||
if (node.typeArguments == null) { | ||
if (node.typeArguments == null || node.typeArguments.length === 0) { | ||
return this.report(node, E.promiseMissingTypeArg()); | ||
@@ -1182,16 +1210,12 @@ } | ||
Extractor.prototype.collectDescription = function (node) { | ||
var symbol = this.ctx.checker.getSymbolAtLocation(node); | ||
if (symbol == null) { | ||
return this.report(node, E.cannotResolveSymbolForDescription()); | ||
var docs = | ||
// @ts-ignore Exposed as stable in https://github.com/microsoft/TypeScript/pull/53627 | ||
ts.getJSDocCommentsAndTags(node); | ||
var comment = docs | ||
.filter(function (doc) { return doc.kind === ts.SyntaxKind.JSDoc; }) | ||
.map(function (doc) { return doc.comment; }) | ||
.join(""); | ||
if (comment) { | ||
return this.gql.string(node, comment.trim(), true); | ||
} | ||
var doc = symbol.getDocumentationComment(this.ctx.checker); | ||
var description = ts.displayPartsToString(doc); | ||
if (description) { | ||
return { | ||
kind: graphql_1.Kind.STRING, | ||
loc: this.loc(node), | ||
value: description, | ||
block: true | ||
}; | ||
} | ||
return null; | ||
@@ -1208,12 +1232,6 @@ }; | ||
// FIXME: Use the _value_'s location not the tag's | ||
reason = this.gqlConstArgument(tag, this.gqlName(tag, "reason"), this.gqlString(tag, reasonComment)); | ||
reason = this.gql.constArgument(tag, this.gql.name(tag, "reason"), this.gql.string(tag, reasonComment)); | ||
} | ||
} | ||
var args = reason == null ? undefined : [reason]; | ||
return { | ||
kind: graphql_1.Kind.DIRECTIVE, | ||
loc: this.loc(tag.tagName), | ||
name: this.gqlName(tag.tagName, DEPRECATED_TAG), | ||
arguments: args | ||
}; | ||
return this.gql.constDirective(tag.tagName, this.gql.name(node, DEPRECATED_TAG), reason == null ? null : [reason]); | ||
}; | ||
@@ -1235,4 +1253,4 @@ Extractor.prototype.property = function (node) { | ||
return null; | ||
var type = node.questionToken == null ? inner : this.gqlNullableType(inner); | ||
var description = this.collectDescription(node.name); | ||
var type = node.questionToken == null ? inner : this.gql.nullableType(inner); | ||
var description = this.collectDescription(node); | ||
var directives = []; | ||
@@ -1247,13 +1265,7 @@ var id = this.expectIdentifier(node.name); | ||
if (id.text !== name.value) { | ||
directives = [this.methodNameDirective(node.name, id.text)]; | ||
directives = [ | ||
this.gql.propertyNameDirective(node.name, { name: id.text }), | ||
]; | ||
} | ||
return { | ||
kind: graphql_1.Kind.FIELD_DEFINITION, | ||
loc: this.loc(node), | ||
description: description || undefined, | ||
name: name, | ||
arguments: undefined, | ||
type: this.handleErrorBubbling(node, type), | ||
directives: directives.length === 0 ? undefined : directives | ||
}; | ||
return this.gql.fieldDefinition(node, name, this.handleErrorBubbling(node, type), null, directives, description); | ||
}; | ||
@@ -1274,3 +1286,3 @@ // TODO: Support separate modes for input and output types | ||
return null; | ||
return this.gqlNonNullType(node, this.gqlListType(node, element)); | ||
return this.gql.nonNullType(node, this.gql.listType(node, element)); | ||
} | ||
@@ -1289,3 +1301,3 @@ else if (ts.isUnionTypeNode(node)) { | ||
var incompatibleVariants = rest.map(function (tsType) { | ||
return _this.related(tsType, "Other non-nullish type"); | ||
return (0, DiagnosticError_1.tsRelated)(tsType, "Other non-nullish type"); | ||
}); | ||
@@ -1296,5 +1308,5 @@ this.report(first, E.expectedOneNonNullishType(), incompatibleVariants); | ||
if (node.types.length > 1) { | ||
return this.gqlNullableType(type); | ||
return this.gql.nullableType(type); | ||
} | ||
return this.gqlNonNullType(node, type); | ||
return this.gql.nonNullType(node, type); | ||
} | ||
@@ -1305,6 +1317,6 @@ else if (ts.isParenthesizedTypeNode(node)) { | ||
else if (node.kind === ts.SyntaxKind.StringKeyword) { | ||
return this.gqlNonNullType(node, this.gqlNamedType(node, "String")); | ||
return this.gql.nonNullType(node, this.gql.namedType(node, "String")); | ||
} | ||
else if (node.kind === ts.SyntaxKind.BooleanKeyword) { | ||
return this.gqlNonNullType(node, this.gqlNamedType(node, "Boolean")); | ||
return this.gql.nonNullType(node, this.gql.namedType(node, "Boolean")); | ||
} | ||
@@ -1318,3 +1330,3 @@ else if (node.kind === ts.SyntaxKind.NumberKeyword) { | ||
// TODO: Better error message. This is okay if it's a type reference, but everything else is not. | ||
this.reportUnhandled(node, E.unknownGraphQLType()); | ||
this.reportUnhandled(node, "type", E.unknownGraphQLType()); | ||
return null; | ||
@@ -1337,3 +1349,3 @@ }; | ||
return null; | ||
return this.gqlNonNullType(node, this.gqlListType(node, element)); | ||
return this.gql.nonNullType(node, this.gql.listType(node, element)); | ||
} | ||
@@ -1345,5 +1357,5 @@ default: { | ||
// A later pass will resolve the type. | ||
var namedType = this.gqlNamedType(node, TypeContext_1.UNRESOLVED_REFERENCE_NAME); | ||
this.ctx.markUnresolvedType(node.typeName, namedType.name); | ||
return this.gqlNonNullType(node, namedType); | ||
var namedType = this.gql.namedType(node, TypeContext_1.UNRESOLVED_REFERENCE_NAME); | ||
this.markUnresolvedType(node.typeName, namedType.name); | ||
return this.gql.nonNullType(node, namedType); | ||
} | ||
@@ -1368,12 +1380,24 @@ } | ||
} | ||
return this.report(node, E.expectedIdentifer()); | ||
return this.report(node, E.expectedIdentifier()); | ||
}; | ||
Extractor.prototype.findTag = function (node, tagName) { | ||
var _a; | ||
return ((_a = ts | ||
var tags = ts | ||
.getJSDocTags(node) | ||
.find(function (tag) { return tag.tagName.escapedText === tagName; })) !== null && _a !== void 0 ? _a : null); | ||
.filter(function (tag) { return tag.tagName.escapedText === tagName; }); | ||
if (tags.length === 0) { | ||
return null; | ||
} | ||
if (tags.length > 1) { | ||
var additionalTags = tags.slice(1).map(function (tag) { | ||
return (0, DiagnosticError_1.tsRelated)(tag, "Additional tag"); | ||
}); | ||
var message = tagName === exports.IMPLEMENTS_TAG_DEPRECATED | ||
? E.duplicateInterfaceTag() | ||
: E.duplicateTag(tagName); | ||
return this.report(tags[0], message, additionalTags); | ||
} | ||
return tags[0]; | ||
}; | ||
// It is a GraphQL best practice to model all fields as nullable. This allows | ||
// the server to handle field level exections by simply returning null for | ||
// the server to handle field level executions by simply returning null for | ||
// that field. | ||
@@ -1394,55 +1418,16 @@ // https://graphql.org/learn/best-practices/#nullability | ||
if (this.configOptions.nullableByDefault) { | ||
return this.gqlNullableType(type); | ||
return this.gql.nullableType(type); | ||
} | ||
return type; | ||
}; | ||
Extractor.prototype.methodNameDirective = function (nameNode, name) { | ||
return this.gqlConstDirective(nameNode, this.gqlName(nameNode, serverDirectives_1.METHOD_NAME_DIRECTIVE), [ | ||
this.gqlConstArgument(nameNode, this.gqlName(nameNode, serverDirectives_1.METHOD_NAME_ARG), this.gqlString(nameNode, name)), | ||
]); | ||
}; | ||
Extractor.prototype.exportDirective = function (nameNode, filename, functionName) { | ||
return this.gqlConstDirective(nameNode, this.gqlName(nameNode, serverDirectives_1.EXPORTED_DIRECTIVE), [ | ||
this.gqlConstArgument(nameNode, this.gqlName(nameNode, serverDirectives_1.EXPORTED_FILENAME_ARG), this.gqlString(nameNode, filename)), | ||
this.gqlConstArgument(nameNode, this.gqlName(nameNode, serverDirectives_1.EXPORTED_FUNCTION_NAME_ARG), this.gqlString(nameNode, functionName)), | ||
]); | ||
}; | ||
/** GraphQL AST node helper methods */ | ||
Extractor.prototype.gqlName = function (node, value) { | ||
return { kind: graphql_1.Kind.NAME, loc: this.loc(node), value: value }; | ||
}; | ||
Extractor.prototype.gqlNamedType = function (node, value) { | ||
return { | ||
kind: graphql_1.Kind.NAMED_TYPE, | ||
loc: this.loc(node), | ||
name: this.gqlName(node, value) | ||
}; | ||
}; | ||
Extractor.prototype.gqlNonNullType = function (node, type) { | ||
if (type.kind === graphql_1.Kind.NON_NULL_TYPE) { | ||
return type; | ||
} | ||
return { kind: graphql_1.Kind.NON_NULL_TYPE, loc: this.loc(node), type: type }; | ||
}; | ||
Extractor.prototype.gqlNullableType = function (type) { | ||
var inner = type; | ||
while (inner.kind === graphql_1.Kind.NON_NULL_TYPE) { | ||
inner = inner.type; | ||
} | ||
return inner; | ||
}; | ||
Extractor.prototype.gqlListType = function (node, type) { | ||
return { kind: graphql_1.Kind.LIST_TYPE, loc: this.loc(node), type: type }; | ||
}; | ||
Extractor.prototype.gqlConstArgument = function (node, name, value) { | ||
return { kind: graphql_1.Kind.ARGUMENT, loc: this.loc(node), name: name, value: value }; | ||
}; | ||
Extractor.prototype.gqlConstDirective = function (node, name, args) { | ||
return { kind: graphql_1.Kind.DIRECTIVE, loc: this.loc(node), name: name, arguments: args }; | ||
}; | ||
Extractor.prototype.gqlString = function (node, value) { | ||
return { kind: graphql_1.Kind.STRING, loc: this.loc(node), value: value }; | ||
}; | ||
return Extractor; | ||
}()); | ||
exports.Extractor = Extractor; | ||
function graphQLNameValidationMessage(name) { | ||
try { | ||
(0, graphql_1.assertName)(name); | ||
return null; | ||
} | ||
catch (e) { | ||
return e.message; | ||
} | ||
} |
@@ -1,3 +0,2 @@ | ||
import * as ts from "typescript"; | ||
export declare function getRelativeOutputPath(options: ts.ParsedCommandLine, sourceFile: ts.SourceFile): string; | ||
export declare function relativePath(absolute: string): string; | ||
export declare function resolveRelativePath(relativePath: string): string; |
"use strict"; | ||
exports.__esModule = true; | ||
exports.resolveRelativePath = exports.getRelativeOutputPath = void 0; | ||
Object.defineProperty(exports, "__esModule", { value: true }); | ||
exports.resolveRelativePath = exports.relativePath = void 0; | ||
var path_1 = require("path"); | ||
var ts = require("typescript"); | ||
// Grats parses TypeScript files and finds resolvers. If the field resolver is a | ||
@@ -13,16 +12,7 @@ // named export, Grats needs to be able to import that file during execution. | ||
// step and the runtime can agree on. This path is that thing. | ||
var gratsRoot = __dirname; | ||
function getRelativeOutputPath(options, sourceFile) { | ||
var fileNames = ts.getOutputFileNames(options, sourceFile.fileName, true); | ||
// ts.getOutputFileNames returns a list of files that includes both the .d.ts | ||
// and .js files. | ||
var jsFileNames = fileNames.filter(function (fileName) { return fileName.endsWith(".js"); }); | ||
if (jsFileNames.length !== 1) { | ||
throw new Error("Grats: Expected ts.getOutputFileNames to return exactly one `.js` file. " + | ||
"Found ".concat(jsFileNames.length, "}. This is a bug in Grats. I'd appreciate it if ") + | ||
"you could open an issue."); | ||
} | ||
return (0, path_1.relative)(gratsRoot, fileNames[0]); | ||
var gratsRoot = (0, path_1.join)(__dirname, "../.."); | ||
function relativePath(absolute) { | ||
return (0, path_1.relative)(gratsRoot, absolute); | ||
} | ||
exports.getRelativeOutputPath = getRelativeOutputPath; | ||
exports.relativePath = relativePath; | ||
function resolveRelativePath(relativePath) { | ||
@@ -29,0 +19,0 @@ return (0, path_1.resolve)(gratsRoot, relativePath); |
@@ -1,10 +0,7 @@ | ||
import { GraphQLSchema } from "graphql"; | ||
import * as ts from "typescript"; | ||
import { ParsedCommandLineGrats } from "./lib"; | ||
import { ReportableDiagnostics, Result } from "./utils/DiagnosticError"; | ||
export { printSDLWithoutDirectives } from "./printSchema"; | ||
export * from "./Types"; | ||
export * from "./lib"; | ||
type RuntimeOptions = { | ||
emitSchemaFile?: string; | ||
}; | ||
export declare function extractGratsSchemaAtRuntime(runtimeOptions: RuntimeOptions): GraphQLSchema; | ||
export declare function buildSchemaFromSDL(sdl: string): GraphQLSchema; | ||
export declare function getParsedTsConfig(configPath?: string): ts.ParsedCommandLine; | ||
export { codegen } from "./codegen"; | ||
export declare function getParsedTsConfig(configFile: string): Result<ParsedCommandLineGrats, ReportableDiagnostics>; |
@@ -16,39 +16,15 @@ "use strict"; | ||
}; | ||
exports.__esModule = true; | ||
exports.getParsedTsConfig = exports.buildSchemaFromSDL = exports.extractGratsSchemaAtRuntime = void 0; | ||
var graphql_1 = require("graphql"); | ||
var utils_1 = require("@graphql-tools/utils"); | ||
var fs = require("fs"); | ||
Object.defineProperty(exports, "__esModule", { value: true }); | ||
exports.getParsedTsConfig = exports.codegen = exports.printSDLWithoutDirectives = void 0; | ||
var ts = require("typescript"); | ||
var lib_1 = require("./lib"); | ||
var DiagnosticError_1 = require("./utils/DiagnosticError"); | ||
var printSchema_1 = require("./printSchema"); | ||
Object.defineProperty(exports, "printSDLWithoutDirectives", { enumerable: true, get: function () { return printSchema_1.printSDLWithoutDirectives; } }); | ||
__exportStar(require("./Types"), exports); | ||
__exportStar(require("./lib"), exports); | ||
// Build an executable schema from a set of files. Note that if extraction | ||
// fails, this function will exit the process and print a helpful error | ||
// message. | ||
function extractGratsSchemaAtRuntime(runtimeOptions) { | ||
var parsedTsConfig = getParsedTsConfig(); | ||
var schemaResult = (0, lib_1.buildSchemaResult)(parsedTsConfig); | ||
if (schemaResult.kind === "ERROR") { | ||
console.error(schemaResult.err.formatDiagnosticsWithColorAndContext()); | ||
process.exit(1); | ||
} | ||
var runtimeSchema = schemaResult.value; | ||
if (runtimeOptions.emitSchemaFile) { | ||
runtimeSchema = (0, graphql_1.lexicographicSortSchema)(runtimeSchema); | ||
var sdl = (0, utils_1.printSchemaWithDirectives)(runtimeSchema, { assumeValid: true }); | ||
var filePath = runtimeOptions.emitSchemaFile; | ||
fs.writeFileSync(filePath, sdl); | ||
} | ||
return runtimeSchema; | ||
} | ||
exports.extractGratsSchemaAtRuntime = extractGratsSchemaAtRuntime; | ||
function buildSchemaFromSDL(sdl) { | ||
var schema = (0, graphql_1.buildSchema)(sdl); | ||
return (0, lib_1.applyServerDirectives)(schema); | ||
} | ||
exports.buildSchemaFromSDL = buildSchemaFromSDL; | ||
var codegen_1 = require("./codegen"); | ||
Object.defineProperty(exports, "codegen", { enumerable: true, get: function () { return codegen_1.codegen; } }); | ||
// #FIXME: Report diagnostics instead of throwing! | ||
function getParsedTsConfig(configPath) { | ||
var configFile = configPath || ts.findConfigFile(process.cwd(), ts.sys.fileExists); | ||
function getParsedTsConfig(configFile) { | ||
if (!configFile) { | ||
@@ -60,7 +36,10 @@ throw new Error("Grats: Could not find tsconfig.json"); | ||
var parsed = ts.getParsedCommandLineOfConfigFile(configFile, undefined, configFileHost); | ||
if (!parsed || parsed.errors.length > 0) { | ||
throw new Error("Grats: Could not parse tsconfig.json"); | ||
if (!parsed) { | ||
throw new Error("Grats: Could not locate tsconfig.json"); | ||
} | ||
return parsed; | ||
if (parsed.errors.length > 0) { | ||
return (0, DiagnosticError_1.err)(DiagnosticError_1.ReportableDiagnostics.fromDiagnostics(parsed.errors)); | ||
} | ||
return (0, DiagnosticError_1.ok)((0, lib_1.validateGratsOptions)(parsed)); | ||
} | ||
exports.getParsedTsConfig = getParsedTsConfig; |
@@ -1,10 +0,13 @@ | ||
import { GraphQLSchema } from "graphql"; | ||
import { Result, ReportableDiagnostics } from "./utils/DiagnosticError"; | ||
import { DocumentNode, GraphQLSchema } from "graphql"; | ||
import { DiagnosticsResult, Result, ReportableDiagnostics } from "./utils/DiagnosticError"; | ||
import * as ts from "typescript"; | ||
export { applyServerDirectives } from "./serverDirectives"; | ||
export type ConfigOptions = { | ||
nullableByDefault?: boolean; | ||
reportTypeScriptTypeErrors?: boolean; | ||
}; | ||
export declare function buildSchemaResult(options: ts.ParsedCommandLine): Result<GraphQLSchema, ReportableDiagnostics>; | ||
export declare function buildSchemaResultWithHost(options: ts.ParsedCommandLine, compilerHost: ts.CompilerHost): Result<GraphQLSchema, ReportableDiagnostics>; | ||
import { ExtractionSnapshot } from "./Extractor"; | ||
import { ParsedCommandLineGrats } from "./gratsConfig"; | ||
export * from "./gratsConfig"; | ||
export declare function buildSchemaResult(options: ParsedCommandLineGrats): Result<GraphQLSchema, ReportableDiagnostics>; | ||
export declare function buildSchemaResultWithHost(options: ParsedCommandLineGrats, compilerHost: ts.CompilerHost): Result<GraphQLSchema, ReportableDiagnostics>; | ||
/** | ||
* Given a merged snapshot representing the whole program, construct a GraphQL | ||
* schema document with metadata directives attached. | ||
*/ | ||
export declare function docFromSnapshot(program: ts.Program, host: ts.CompilerHost, snapshot: ExtractionSnapshot): DiagnosticsResult<DocumentNode>; |
"use strict"; | ||
var __assign = (this && this.__assign) || function () { | ||
__assign = Object.assign || function(t) { | ||
for (var s, i = 1, n = arguments.length; i < n; i++) { | ||
s = arguments[i]; | ||
for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p)) | ||
t[p] = s[p]; | ||
} | ||
return t; | ||
}; | ||
return __assign.apply(this, arguments); | ||
}; | ||
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) { | ||
@@ -24,2 +13,5 @@ if (k2 === undefined) k2 = k; | ||
})); | ||
var __exportStar = (this && this.__exportStar) || function(m, exports) { | ||
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p); | ||
}; | ||
var __values = (this && this.__values) || function(o) { | ||
@@ -36,13 +28,36 @@ var s = typeof Symbol === "function" && Symbol.iterator, m = s && o[s], i = 0; | ||
}; | ||
exports.__esModule = true; | ||
exports.buildSchemaResultWithHost = exports.buildSchemaResult = exports.applyServerDirectives = void 0; | ||
var __read = (this && this.__read) || function (o, n) { | ||
var m = typeof Symbol === "function" && o[Symbol.iterator]; | ||
if (!m) return o; | ||
var i = m.call(o), r, ar = [], e; | ||
try { | ||
while ((n === void 0 || n-- > 0) && !(r = i.next()).done) ar.push(r.value); | ||
} | ||
catch (error) { e = { error: error }; } | ||
finally { | ||
try { | ||
if (r && !r.done && (m = i["return"])) m.call(i); | ||
} | ||
finally { if (e) throw e.error; } | ||
} | ||
return ar; | ||
}; | ||
Object.defineProperty(exports, "__esModule", { value: true }); | ||
exports.docFromSnapshot = exports.buildSchemaResultWithHost = exports.buildSchemaResult = void 0; | ||
var graphql_1 = require("graphql"); | ||
var DiagnosticError_1 = require("./utils/DiagnosticError"); | ||
var ts = require("typescript"); | ||
var Extractor_1 = require("./Extractor"); | ||
var TypeContext_1 = require("./TypeContext"); | ||
var validate_1 = require("graphql/validation/validate"); | ||
var serverDirectives_1 = require("./serverDirectives"); | ||
var serverDirectives_2 = require("./serverDirectives"); | ||
__createBinding(exports, serverDirectives_2, "applyServerDirectives"); | ||
var validateTypenames_1 = require("./validations/validateTypenames"); | ||
var snapshotsFromProgram_1 = require("./transforms/snapshotsFromProgram"); | ||
var validateMergedInterfaces_1 = require("./validations/validateMergedInterfaces"); | ||
var validateContextReferences_1 = require("./validations/validateContextReferences"); | ||
var metadataDirectives_1 = require("./metadataDirectives"); | ||
var helpers_1 = require("./utils/helpers"); | ||
var addInterfaceFields_1 = require("./transforms/addInterfaceFields"); | ||
var filterNonGqlInterfaces_1 = require("./transforms/filterNonGqlInterfaces"); | ||
var resolveTypes_1 = require("./transforms/resolveTypes"); | ||
var validateAsyncIterable_1 = require("./validations/validateAsyncIterable"); | ||
__exportStar(require("./gratsConfig"), exports); | ||
// Construct a schema, using GraphQL schema language | ||
@@ -59,89 +74,24 @@ // Exported for tests that want to intercept diagnostic errors. | ||
function buildSchemaResultWithHost(options, compilerHost) { | ||
var gratsOptions = parseGratsOptions(options); | ||
var schemaResult = extractSchema(options, gratsOptions, compilerHost); | ||
var schemaResult = extractSchema(options, compilerHost); | ||
if (schemaResult.kind === "ERROR") { | ||
return (0, DiagnosticError_1.err)(new DiagnosticError_1.ReportableDiagnostics(compilerHost, schemaResult.err)); | ||
} | ||
return (0, DiagnosticError_1.ok)((0, serverDirectives_1.applyServerDirectives)(schemaResult.value)); | ||
return (0, DiagnosticError_1.ok)(schemaResult.value); | ||
} | ||
exports.buildSchemaResultWithHost = buildSchemaResultWithHost; | ||
// TODO: Make this return diagnostics | ||
function parseGratsOptions(options) { | ||
var _a, _b; | ||
var gratsOptions = __assign({}, ((_b = (_a = options.raw) === null || _a === void 0 ? void 0 : _a.grats) !== null && _b !== void 0 ? _b : {})); | ||
if (gratsOptions.nullableByDefault === undefined) { | ||
gratsOptions.nullableByDefault = true; | ||
function extractSchema(options, host) { | ||
var program = ts.createProgram(options.fileNames, options.options, host); | ||
var snapshotsResult = (0, snapshotsFromProgram_1.snapshotsFromProgram)(program, options); | ||
if (snapshotsResult.kind === "ERROR") { | ||
return snapshotsResult; | ||
} | ||
else if (typeof gratsOptions.nullableByDefault !== "boolean") { | ||
throw new Error("Grats: The Grats config option `nullableByDefault` must be a boolean if provided."); | ||
var snapshot = reduceSnapshots(snapshotsResult.value); | ||
var docResult = docFromSnapshot(program, host, snapshot); | ||
if (docResult.kind === "ERROR") { | ||
return docResult; | ||
} | ||
if (gratsOptions.reportTypeScriptTypeErrors === undefined) { | ||
gratsOptions.reportTypeScriptTypeErrors = false; | ||
} | ||
else if (typeof gratsOptions.reportTypeScriptTypeErrors !== "boolean") { | ||
throw new Error("Grats: The Grats config option `reportTypeScriptTypeErrors` must be a boolean if provided"); | ||
} | ||
// FIXME: Check for unknown options | ||
return gratsOptions; | ||
return buildSchemaFromDocumentNode(docResult.value, snapshot.typesWithTypenameField); | ||
} | ||
function extractSchema(options, gratsOptions, host) { | ||
var e_1, _a, e_2, _b; | ||
var program = ts.createProgram(options.fileNames, options.options, host); | ||
var checker = program.getTypeChecker(); | ||
var ctx = new TypeContext_1.TypeContext(options, checker, host); | ||
var definitions = Array.from(serverDirectives_1.DIRECTIVES_AST.definitions); | ||
try { | ||
for (var _c = __values(program.getSourceFiles()), _d = _c.next(); !_d.done; _d = _c.next()) { | ||
var sourceFile = _d.value; | ||
// If the file doesn't contain any GraphQL definitions, skip it. | ||
if (!/@gql/i.test(sourceFile.text)) { | ||
continue; | ||
} | ||
if (gratsOptions.reportTypeScriptTypeErrors) { | ||
// If the user asked for us to report TypeScript errors, then we'll report them. | ||
var typeErrors = ts.getPreEmitDiagnostics(program, sourceFile); | ||
if (typeErrors.length > 0) { | ||
return (0, DiagnosticError_1.err)(typeErrors); | ||
} | ||
} | ||
else { | ||
// Otherwise, we will only report syntax errors, since they will prevent us from | ||
// extracting any GraphQL definitions. | ||
var syntaxErrors = program.getSyntacticDiagnostics(sourceFile); | ||
if (syntaxErrors.length > 0) { | ||
// It's not very helpful to report multiple syntax errors, so just report | ||
// the first one. | ||
return (0, DiagnosticError_1.err)([syntaxErrors[0]]); | ||
} | ||
} | ||
var extractor = new Extractor_1.Extractor(sourceFile, ctx, gratsOptions); | ||
var extractedResult = extractor.extract(); | ||
if (extractedResult.kind === "ERROR") | ||
return extractedResult; | ||
try { | ||
for (var _e = (e_2 = void 0, __values(extractedResult.value)), _f = _e.next(); !_f.done; _f = _e.next()) { | ||
var definition = _f.value; | ||
definitions.push(definition); | ||
} | ||
} | ||
catch (e_2_1) { e_2 = { error: e_2_1 }; } | ||
finally { | ||
try { | ||
if (_f && !_f.done && (_b = _e["return"])) _b.call(_e); | ||
} | ||
finally { if (e_2) throw e_2.error; } | ||
} | ||
} | ||
} | ||
catch (e_1_1) { e_1 = { error: e_1_1 }; } | ||
finally { | ||
try { | ||
if (_d && !_d.done && (_a = _c["return"])) _a.call(_c); | ||
} | ||
finally { if (e_1) throw e_1.error; } | ||
} | ||
var docResult = ctx.resolveTypes({ kind: graphql_1.Kind.DOCUMENT, definitions: definitions }); | ||
if (docResult.kind === "ERROR") | ||
return docResult; | ||
var doc = docResult.value; | ||
// Given a SDL AST, build and validate a GraphQLSchema. | ||
function buildSchemaFromDocumentNode(doc, typesWithTypenameField) { | ||
// TODO: Currently this does not detect definitions that shadow builtins | ||
@@ -165,3 +115,3 @@ // (`String`, `Int`, etc). However, if we pass a second param (extending an | ||
} | ||
var typenameDiagnostics = validateTypename(schema, ctx); | ||
var typenameDiagnostics = (0, validateTypenames_1.validateTypenames)(schema, typesWithTypenameField); | ||
if (typenameDiagnostics.length > 0) | ||
@@ -171,22 +121,85 @@ return (0, DiagnosticError_1.err)(typenameDiagnostics); | ||
} | ||
function validateTypename(schema, ctx) { | ||
var e_3, _a, e_4, _b; | ||
var _c, _d; | ||
var typenameDiagnostics = []; | ||
var abstractTypes = Object.values(schema.getTypeMap()).filter(graphql_1.isAbstractType); | ||
/** | ||
* Given a merged snapshot representing the whole program, construct a GraphQL | ||
* schema document with metadata directives attached. | ||
*/ | ||
function docFromSnapshot(program, host, snapshot) { | ||
var e_1, _a, e_2, _b; | ||
var checker = program.getTypeChecker(); | ||
var ctx = new TypeContext_1.TypeContext(checker, host); | ||
// Validate the snapshot | ||
var mergedResult = (0, DiagnosticError_1.combineResults)((0, validateMergedInterfaces_1.validateMergedInterfaces)(checker, snapshot.interfaceDeclarations), (0, validateContextReferences_1.validateContextReferences)(ctx, snapshot.contextReferences)); | ||
if (mergedResult.kind === "ERROR") { | ||
return mergedResult; | ||
} | ||
try { | ||
for (var abstractTypes_1 = __values(abstractTypes), abstractTypes_1_1 = abstractTypes_1.next(); !abstractTypes_1_1.done; abstractTypes_1_1 = abstractTypes_1.next()) { | ||
var type = abstractTypes_1_1.value; | ||
// TODO: If we already implement resolveType, we don't need to check implementors | ||
var typeImplementors = schema.getPossibleTypes(type).filter(graphql_1.isType); | ||
// Propagate snapshot data to type context | ||
for (var _c = __values(snapshot.unresolvedNames), _d = _c.next(); !_d.done; _d = _c.next()) { | ||
var _e = __read(_d.value, 2), node = _e[0], typeName = _e[1]; | ||
ctx.markUnresolvedType(node, typeName); | ||
} | ||
} | ||
catch (e_1_1) { e_1 = { error: e_1_1 }; } | ||
finally { | ||
try { | ||
if (_d && !_d.done && (_a = _c.return)) _a.call(_c); | ||
} | ||
finally { if (e_1) throw e_1.error; } | ||
} | ||
try { | ||
for (var _f = __values(snapshot.nameDefinitions), _g = _f.next(); !_g.done; _g = _f.next()) { | ||
var _h = __read(_g.value, 2), node = _h[0], definition = _h[1]; | ||
ctx.recordTypeName(node, definition.name, definition.kind); | ||
} | ||
} | ||
catch (e_2_1) { e_2 = { error: e_2_1 }; } | ||
finally { | ||
try { | ||
if (_g && !_g.done && (_b = _f.return)) _b.call(_f); | ||
} | ||
finally { if (e_2) throw e_2.error; } | ||
} | ||
// Fixup the schema SDL | ||
var definitions = Array.from(metadataDirectives_1.DIRECTIVES_AST.definitions); | ||
(0, helpers_1.extend)(definitions, snapshot.definitions); | ||
// If you define a field on an interface using the functional style, we need to add | ||
// that field to each concrete type as well. This must be done after all types are created, | ||
// but before we validate the schema. | ||
var definitionsResult = (0, addInterfaceFields_1.addInterfaceFields)(ctx, definitions); | ||
if (definitionsResult.kind === "ERROR") { | ||
return definitionsResult; | ||
} | ||
var filteredDoc = (0, filterNonGqlInterfaces_1.filterNonGqlInterfaces)(ctx, { | ||
kind: graphql_1.Kind.DOCUMENT, | ||
definitions: definitionsResult.value, | ||
}); | ||
var docResult = (0, resolveTypes_1.resolveTypes)(ctx, filteredDoc); | ||
if (docResult.kind === "ERROR") | ||
return docResult; | ||
var doc = docResult.value; | ||
var subscriptionsValidationResult = (0, validateAsyncIterable_1.validateAsyncIterable)(doc); | ||
if (subscriptionsValidationResult.kind === "ERROR") { | ||
return subscriptionsValidationResult; | ||
} | ||
return (0, DiagnosticError_1.ok)(doc); | ||
} | ||
exports.docFromSnapshot = docFromSnapshot; | ||
// Given a list of snapshots, merge them into a single snapshot. | ||
function reduceSnapshots(snapshots) { | ||
var e_3, _a, e_4, _b, e_5, _c, e_6, _d, e_7, _e, e_8, _f, e_9, _g; | ||
var result = { | ||
definitions: [], | ||
nameDefinitions: new Map(), | ||
unresolvedNames: new Map(), | ||
contextReferences: [], | ||
typesWithTypenameField: new Set(), | ||
interfaceDeclarations: [], | ||
}; | ||
try { | ||
for (var snapshots_1 = __values(snapshots), snapshots_1_1 = snapshots_1.next(); !snapshots_1_1.done; snapshots_1_1 = snapshots_1.next()) { | ||
var snapshot = snapshots_1_1.value; | ||
try { | ||
for (var typeImplementors_1 = (e_4 = void 0, __values(typeImplementors)), typeImplementors_1_1 = typeImplementors_1.next(); !typeImplementors_1_1.done; typeImplementors_1_1 = typeImplementors_1.next()) { | ||
var implementor = typeImplementors_1_1.value; | ||
if (!ctx.hasTypename.has(implementor.name)) { | ||
var loc = (_d = (_c = implementor.astNode) === null || _c === void 0 ? void 0 : _c.name) === null || _d === void 0 ? void 0 : _d.loc; | ||
if (loc == null) { | ||
throw new Error("Grats expected the parsed type `".concat(implementor.name, "` to have location information. This is a bug in Grats. Please report it.")); | ||
} | ||
typenameDiagnostics.push((0, DiagnosticError_1.diagnosticAtGraphQLLocation)("Missing __typename on `".concat(implementor.name, "`. The type `").concat(type.name, "` is used in a union or interface, so it must have a `__typename` field."), loc)); | ||
} | ||
for (var _h = (e_4 = void 0, __values(snapshot.definitions)), _j = _h.next(); !_j.done; _j = _h.next()) { | ||
var definition = _j.value; | ||
result.definitions.push(definition); | ||
} | ||
@@ -197,6 +210,71 @@ } | ||
try { | ||
if (typeImplementors_1_1 && !typeImplementors_1_1.done && (_b = typeImplementors_1["return"])) _b.call(typeImplementors_1); | ||
if (_j && !_j.done && (_b = _h.return)) _b.call(_h); | ||
} | ||
finally { if (e_4) throw e_4.error; } | ||
} | ||
try { | ||
for (var _k = (e_5 = void 0, __values(snapshot.nameDefinitions)), _l = _k.next(); !_l.done; _l = _k.next()) { | ||
var _m = __read(_l.value, 2), node = _m[0], definition = _m[1]; | ||
result.nameDefinitions.set(node, definition); | ||
} | ||
} | ||
catch (e_5_1) { e_5 = { error: e_5_1 }; } | ||
finally { | ||
try { | ||
if (_l && !_l.done && (_c = _k.return)) _c.call(_k); | ||
} | ||
finally { if (e_5) throw e_5.error; } | ||
} | ||
try { | ||
for (var _o = (e_6 = void 0, __values(snapshot.unresolvedNames)), _p = _o.next(); !_p.done; _p = _o.next()) { | ||
var _q = __read(_p.value, 2), node = _q[0], typeName = _q[1]; | ||
result.unresolvedNames.set(node, typeName); | ||
} | ||
} | ||
catch (e_6_1) { e_6 = { error: e_6_1 }; } | ||
finally { | ||
try { | ||
if (_p && !_p.done && (_d = _o.return)) _d.call(_o); | ||
} | ||
finally { if (e_6) throw e_6.error; } | ||
} | ||
try { | ||
for (var _r = (e_7 = void 0, __values(snapshot.contextReferences)), _s = _r.next(); !_s.done; _s = _r.next()) { | ||
var contextReference = _s.value; | ||
result.contextReferences.push(contextReference); | ||
} | ||
} | ||
catch (e_7_1) { e_7 = { error: e_7_1 }; } | ||
finally { | ||
try { | ||
if (_s && !_s.done && (_e = _r.return)) _e.call(_r); | ||
} | ||
finally { if (e_7) throw e_7.error; } | ||
} | ||
try { | ||
for (var _t = (e_8 = void 0, __values(snapshot.typesWithTypenameField)), _u = _t.next(); !_u.done; _u = _t.next()) { | ||
var typeName = _u.value; | ||
result.typesWithTypenameField.add(typeName); | ||
} | ||
} | ||
catch (e_8_1) { e_8 = { error: e_8_1 }; } | ||
finally { | ||
try { | ||
if (_u && !_u.done && (_f = _t.return)) _f.call(_t); | ||
} | ||
finally { if (e_8) throw e_8.error; } | ||
} | ||
try { | ||
for (var _v = (e_9 = void 0, __values(snapshot.interfaceDeclarations)), _w = _v.next(); !_w.done; _w = _v.next()) { | ||
var interfaceDeclaration = _w.value; | ||
result.interfaceDeclarations.push(interfaceDeclaration); | ||
} | ||
} | ||
catch (e_9_1) { e_9 = { error: e_9_1 }; } | ||
finally { | ||
try { | ||
if (_w && !_w.done && (_g = _v.return)) _g.call(_v); | ||
} | ||
finally { if (e_9) throw e_9.error; } | ||
} | ||
} | ||
@@ -207,7 +285,7 @@ } | ||
try { | ||
if (abstractTypes_1_1 && !abstractTypes_1_1.done && (_a = abstractTypes_1["return"])) _a.call(abstractTypes_1); | ||
if (snapshots_1_1 && !snapshots_1_1.done && (_a = snapshots_1.return)) _a.call(snapshots_1); | ||
} | ||
finally { if (e_3) throw e_3.error; } | ||
} | ||
return typenameDiagnostics; | ||
return result; | ||
} |
@@ -60,9 +60,16 @@ "use strict"; | ||
}; | ||
exports.__esModule = true; | ||
Object.defineProperty(exports, "__esModule", { value: true }); | ||
var path = require("path"); | ||
var TestRunner_1 = require("./TestRunner"); | ||
var lib_1 = require("../lib"); | ||
var utils_1 = require("@graphql-tools/utils"); | ||
var ts = require("typescript"); | ||
var graphql_1 = require("graphql"); | ||
var commander_1 = require("commander"); | ||
var Locate_1 = require("../Locate"); | ||
var DiagnosticError_1 = require("../utils/DiagnosticError"); | ||
var fs_1 = require("fs"); | ||
var codegen_1 = require("../codegen"); | ||
var utils_1 = require("@graphql-tools/utils"); | ||
var jest_diff_1 = require("jest-diff"); | ||
var printSchema_1 = require("../printSchema"); | ||
var program = new commander_1.Command(); | ||
@@ -72,3 +79,3 @@ program | ||
.description("Run Grats' internal tests") | ||
.option("-w, --write", "Write the actual ouput of the test to the expected output files. Useful for updating tests.") | ||
.option("-w, --write", "Write the actual output of the test to the expected output files. Useful for updating tests.") | ||
.option("-f, --filter <FILTER_REGEX>", "A regex to filter the tests to run. Only tests with a file path matching the regex will be run.") | ||
@@ -78,3 +85,3 @@ .action(function (_a) { | ||
return __awaiter(void 0, void 0, void 0, function () { | ||
var filterRegex, failures, testDirs_1, testDirs_1_1, _b, fixturesDir_1, transformer, runner, e_1_1; | ||
var filterRegex, failures, testDirs_1, testDirs_1_1, _b, fixturesDir_1, transformer, testFilePattern, ignoreFilePattern, runner, e_1_1; | ||
var e_1, _c; | ||
@@ -93,4 +100,4 @@ return __generator(this, function (_d) { | ||
if (!!testDirs_1_1.done) return [3 /*break*/, 5]; | ||
_b = testDirs_1_1.value, fixturesDir_1 = _b.fixturesDir, transformer = _b.transformer; | ||
runner = new TestRunner_1["default"](fixturesDir_1, !!write, filterRegex, transformer); | ||
_b = testDirs_1_1.value, fixturesDir_1 = _b.fixturesDir, transformer = _b.transformer, testFilePattern = _b.testFilePattern, ignoreFilePattern = _b.ignoreFilePattern; | ||
runner = new TestRunner_1.default(fixturesDir_1, !!write, filterRegex, testFilePattern, ignoreFilePattern, transformer); | ||
return [4 /*yield*/, runner.run()]; | ||
@@ -110,3 +117,3 @@ case 3: | ||
try { | ||
if (testDirs_1_1 && !testDirs_1_1.done && (_c = testDirs_1["return"])) _c.call(testDirs_1); | ||
if (testDirs_1_1 && !testDirs_1_1.done && (_c = testDirs_1.return)) _c.call(testDirs_1); | ||
} | ||
@@ -124,2 +131,3 @@ finally { if (e_1) throw e_1.error; } | ||
}); | ||
var gratsDir = path.join(__dirname, "../.."); | ||
var fixturesDir = path.join(__dirname, "fixtures"); | ||
@@ -130,6 +138,9 @@ var integrationFixturesDir = path.join(__dirname, "integrationFixtures"); | ||
fixturesDir: fixturesDir, | ||
testFilePattern: /\.ts$/, | ||
ignoreFilePattern: null, | ||
transformer: function (code, fileName) { | ||
var firstLine = code.split("\n")[0]; | ||
var options = { | ||
nullableByDefault: true | ||
nullableByDefault: true, | ||
schemaHeader: null, | ||
}; | ||
@@ -141,50 +152,71 @@ if (firstLine.startsWith("// {")) { | ||
} | ||
var files = ["".concat(fixturesDir, "/").concat(fileName), "src/Types.ts"]; | ||
var parsedOptions = { | ||
var files = [ | ||
"".concat(fixturesDir, "/").concat(fileName), | ||
path.join(__dirname, "../Types.ts"), | ||
]; | ||
var parsedOptions = (0, lib_1.validateGratsOptions)({ | ||
options: {}, | ||
raw: { | ||
grats: options | ||
grats: options, | ||
}, | ||
errors: [], | ||
fileNames: files | ||
}; | ||
var schemaResult = (0, lib_1.buildSchemaResult)(parsedOptions); | ||
fileNames: files, | ||
}); | ||
// https://stackoverflow.com/a/66604532/1263117 | ||
var compilerHost = ts.createCompilerHost(parsedOptions.options, | ||
/* setParentNodes this is needed for finding jsDocs */ | ||
true); | ||
var schemaResult = (0, lib_1.buildSchemaResultWithHost)(parsedOptions, compilerHost); | ||
if (schemaResult.kind === "ERROR") { | ||
return schemaResult.err.formatDiagnosticsWithContext(); | ||
} | ||
return (0, utils_1.printSchemaWithDirectives)(schemaResult.value, { | ||
assumeValid: true | ||
}); | ||
} | ||
// We run codegen here just ensure that it doesn't throw. | ||
var executableSchema = (0, codegen_1.codegen)(schemaResult.value, "".concat(fixturesDir, "/").concat(fileName)); | ||
var LOCATION_REGEX = /^\/\/ Locate: (.*)/; | ||
var locationMatch = code.match(LOCATION_REGEX); | ||
if (locationMatch != null) { | ||
var locResult = (0, Locate_1.locate)(schemaResult.value, locationMatch[1].trim()); | ||
if (locResult.kind === "ERROR") { | ||
return locResult.err; | ||
} | ||
return new DiagnosticError_1.ReportableDiagnostics(compilerHost, [ | ||
(0, DiagnosticError_1.gqlErr)(locResult.value, "Located here"), | ||
]).formatDiagnosticsWithContext(); | ||
} | ||
else { | ||
var sdl = (0, utils_1.printSchemaWithDirectives)(schemaResult.value, { | ||
assumeValid: true, | ||
}); | ||
return "-- SDL --\n".concat(sdl, "\n-- TypeScript --\n").concat(executableSchema); | ||
} | ||
}, | ||
}, | ||
{ | ||
fixturesDir: integrationFixturesDir, | ||
testFilePattern: /index.ts$/, | ||
ignoreFilePattern: /schema.ts$/, | ||
transformer: function (code, fileName) { return __awaiter(void 0, void 0, void 0, function () { | ||
var filePath, server, options, files, parsedOptions, schemaResult, schema, data; | ||
var filePath, schemaPath, options, files, parsedOptions, schemaResult, tsSchema, server, schemaModule, schemaDiff, data; | ||
return __generator(this, function (_a) { | ||
var _b; | ||
switch (_a.label) { | ||
case 0: | ||
filePath = "".concat(integrationFixturesDir, "/").concat(fileName); | ||
return [4 /*yield*/, (_b = filePath, Promise.resolve().then(function () { return require(_b); }))]; | ||
case 1: | ||
server = _a.sent(); | ||
if (server.query == null || typeof server.query !== "string") { | ||
throw new Error("Expected `".concat(filePath, "` to export a query text as `query`")); | ||
} | ||
if (server.Query == null || typeof server.Query !== "function") { | ||
throw new Error("Expected `".concat(filePath, "` to export a Query type as `Query`")); | ||
} | ||
schemaPath = path.join(path.dirname(filePath), "schema.ts"); | ||
options = { | ||
nullableByDefault: true | ||
nullableByDefault: true, | ||
}; | ||
files = [filePath, "src/Types.ts"]; | ||
parsedOptions = { | ||
options: {}, | ||
files = [filePath, path.join(__dirname, "../Types.ts")]; | ||
parsedOptions = (0, lib_1.validateGratsOptions)({ | ||
options: { | ||
// Required to enable ts-node to locate function exports | ||
rootDir: gratsDir, | ||
outDir: "dist", | ||
configFilePath: "tsconfig.json", | ||
}, | ||
raw: { | ||
grats: options | ||
grats: options, | ||
}, | ||
errors: [], | ||
fileNames: files | ||
}; | ||
fileNames: files, | ||
}); | ||
schemaResult = (0, lib_1.buildSchemaResult)(parsedOptions); | ||
@@ -194,9 +226,25 @@ if (schemaResult.kind === "ERROR") { | ||
} | ||
schema = schemaResult.value; | ||
tsSchema = (0, codegen_1.codegen)(schemaResult.value, schemaPath); | ||
(0, fs_1.writeFileSync)(schemaPath, tsSchema); | ||
return [4 /*yield*/, Promise.resolve("".concat(filePath)).then(function (s) { return require(s); })]; | ||
case 1: | ||
server = _a.sent(); | ||
if (server.query == null || typeof server.query !== "string") { | ||
throw new Error("Expected `".concat(filePath, "` to export a query text as `query`")); | ||
} | ||
return [4 /*yield*/, Promise.resolve("".concat(schemaPath)).then(function (s) { return require(s); })]; | ||
case 2: | ||
schemaModule = _a.sent(); | ||
schemaDiff = compareSchemas(schemaModule.schema, schemaResult.value); | ||
if (schemaDiff) { | ||
console.log(schemaDiff); | ||
// TODO: Make this an actual test failure, not an error | ||
throw new Error("The codegen schema does not match the SDL schema."); | ||
} | ||
return [4 /*yield*/, (0, graphql_1.graphql)({ | ||
schema: schema, | ||
schema: schemaModule.schema, | ||
source: server.query, | ||
rootValue: new server.Query() | ||
variableValues: server.variables, | ||
})]; | ||
case 2: | ||
case 3: | ||
data = _a.sent(); | ||
@@ -206,5 +254,14 @@ return [2 /*return*/, JSON.stringify(data, null, 2)]; | ||
}); | ||
}); } | ||
}); }, | ||
}, | ||
]; | ||
// Returns null if the schemas are equal, otherwise returns a string diff. | ||
function compareSchemas(actual, expected) { | ||
var actualSDL = (0, printSchema_1.printSDLWithoutDirectives)(actual); | ||
var expectedSDL = (0, printSchema_1.printSDLWithoutDirectives)(expected); | ||
if (actualSDL === expectedSDL) { | ||
return null; | ||
} | ||
return (0, jest_diff_1.diff)(expectedSDL, actualSDL); | ||
} | ||
program.parse(); |
@@ -15,3 +15,3 @@ type Transformer = (code: string, filename: string) => Promise<string> | string; | ||
_transformer: Transformer; | ||
constructor(fixturesDir: string, write: boolean, filter: string | null, transformer: Transformer); | ||
constructor(fixturesDir: string, write: boolean, filter: string | null, testFilePattern: RegExp, ignoreFilePattern: RegExp | null, transformer: Transformer); | ||
run(): Promise<boolean>; | ||
@@ -18,0 +18,0 @@ _testFixture(fixture: string): Promise<void>; |
@@ -49,3 +49,3 @@ "use strict"; | ||
}; | ||
exports.__esModule = true; | ||
Object.defineProperty(exports, "__esModule", { value: true }); | ||
var fs = require("fs"); | ||
@@ -60,3 +60,3 @@ var path = require("path"); | ||
var TestRunner = /** @class */ (function () { | ||
function TestRunner(fixturesDir, write, filter, transformer) { | ||
function TestRunner(fixturesDir, write, filter, testFilePattern, ignoreFilePattern, transformer) { | ||
var e_1, _a; | ||
@@ -74,3 +74,3 @@ this._testFixtures = []; | ||
var fileName = _c.value; | ||
if (fileName.endsWith(".ts")) { | ||
if (testFilePattern.test(fileName)) { | ||
this._testFixtures.push(fileName); | ||
@@ -82,3 +82,3 @@ var filePath = path.join(fixturesDir, fileName); | ||
} | ||
else { | ||
else if (!ignoreFilePattern || !ignoreFilePattern.test(fileName)) { | ||
this._otherFiles.add(fileName); | ||
@@ -91,3 +91,3 @@ } | ||
try { | ||
if (_c && !_c.done && (_a = _b["return"])) _a.call(_b); | ||
if (_c && !_c.done && (_a = _b.return)) _a.call(_b); | ||
} | ||
@@ -125,3 +125,3 @@ finally { if (e_1) throw e_1.error; } | ||
try { | ||
if (_b && !_b.done && (_g = _a["return"])) _g.call(_a); | ||
if (_b && !_b.done && (_g = _a.return)) _g.call(_a); | ||
} | ||
@@ -151,3 +151,3 @@ finally { if (e_2) throw e_2.error; } | ||
try { | ||
if (_d && !_d.done && (_h = _c["return"])) _h.call(_c); | ||
if (_d && !_d.done && (_h = _c.return)) _h.call(_c); | ||
} | ||
@@ -168,3 +168,3 @@ finally { if (e_3) throw e_3.error; } | ||
try { | ||
if (_f && !_f.done && (_j = _e["return"])) _j.call(_e); | ||
if (_f && !_f.done && (_j = _e.return)) _j.call(_e); | ||
} | ||
@@ -191,3 +191,3 @@ finally { if (e_4) throw e_4.error; } | ||
if (this._otherFiles.has(expectedFileName)) { | ||
this._otherFiles["delete"](expectedFileName); | ||
this._otherFiles.delete(expectedFileName); | ||
} | ||
@@ -247,3 +247,3 @@ else { | ||
}()); | ||
exports["default"] = TestRunner; | ||
exports.default = TestRunner; | ||
function readdirSyncRecursive(dir) { | ||
@@ -266,3 +266,3 @@ var e_6, _a, e_7, _b; | ||
try { | ||
if (_f && !_f.done && (_b = _e["return"])) _b.call(_e); | ||
if (_f && !_f.done && (_b = _e.return)) _b.call(_e); | ||
} | ||
@@ -280,3 +280,3 @@ finally { if (e_7) throw e_7.error; } | ||
try { | ||
if (_d && !_d.done && (_a = _c["return"])) _a.call(_c); | ||
if (_d && !_d.done && (_a = _c.return)) _a.call(_c); | ||
} | ||
@@ -283,0 +283,0 @@ finally { if (e_6) throw e_6.error; } |
@@ -1,5 +0,9 @@ | ||
import { DocumentNode, NameNode } from "graphql"; | ||
import { NameNode } from "graphql"; | ||
import * as ts from "typescript"; | ||
import { DiagnosticResult, DiagnosticsResult } from "./utils/DiagnosticError"; | ||
export declare const UNRESOLVED_REFERENCE_NAME = "__UNRESOLVED_REFERENCE__"; | ||
export type NameDefinition = { | ||
name: NameNode; | ||
kind: "TYPE" | "INTERFACE" | "UNION" | "SCALAR" | "INPUT_OBJECT" | "ENUM"; | ||
}; | ||
/** | ||
@@ -20,14 +24,12 @@ * Used to track TypeScript references. | ||
host: ts.CompilerHost; | ||
_options: ts.ParsedCommandLine; | ||
_symbolToName: Map<ts.Symbol, string>; | ||
_symbolToName: Map<ts.Symbol, NameDefinition>; | ||
_unresolvedTypes: Map<NameNode, ts.Symbol>; | ||
hasTypename: Set<string>; | ||
constructor(options: ts.ParsedCommandLine, checker: ts.TypeChecker, host: ts.CompilerHost); | ||
recordTypeName(node: ts.Node, name: string): void; | ||
recordHasTypenameField(name: string): void; | ||
constructor(checker: ts.TypeChecker, host: ts.CompilerHost); | ||
recordTypeName(node: ts.Node, name: NameNode, kind: NameDefinition["kind"]): void; | ||
markUnresolvedType(node: ts.Node, name: NameNode): void; | ||
resolveTypes(doc: DocumentNode): DiagnosticsResult<DocumentNode>; | ||
findSymbolDeclaration(startSymbol: ts.Symbol): ts.Declaration | null; | ||
resolveSymbol(startSymbol: ts.Symbol): ts.Symbol; | ||
resolveNamedType(unresolved: NameNode): DiagnosticResult<NameNode>; | ||
validateInterfaceImplementorsHaveTypenameField(): DiagnosticResult<null>; | ||
getDestFilePath(sourceFile: ts.SourceFile): string; | ||
unresolvedNameIsGraphQL(unresolved: NameNode): boolean; | ||
getNameDefinition(nameNode: NameNode): DiagnosticsResult<NameDefinition>; | ||
} |
@@ -13,8 +13,7 @@ "use strict"; | ||
}; | ||
exports.__esModule = true; | ||
Object.defineProperty(exports, "__esModule", { value: true }); | ||
exports.TypeContext = exports.UNRESOLVED_REFERENCE_NAME = void 0; | ||
var graphql_1 = require("graphql"); | ||
var ts = require("typescript"); | ||
var DiagnosticError_1 = require("./utils/DiagnosticError"); | ||
var gratsRoot_1 = require("./gratsRoot"); | ||
var E = require("./Errors"); | ||
exports.UNRESOLVED_REFERENCE_NAME = "__UNRESOLVED_REFERENCE__"; | ||
@@ -34,11 +33,11 @@ /** | ||
var TypeContext = /** @class */ (function () { | ||
function TypeContext(options, checker, host) { | ||
function TypeContext(checker, host) { | ||
this._symbolToName = new Map(); | ||
this._unresolvedTypes = new Map(); | ||
this.hasTypename = new Set(); | ||
this._options = options; | ||
this.checker = checker; | ||
this.host = host; | ||
} | ||
TypeContext.prototype.recordTypeName = function (node, name) { | ||
// Record that a GraphQL construct of type `kind` with the name `name` is | ||
// declared at `node`. | ||
TypeContext.prototype.recordTypeName = function (node, name, kind) { | ||
var symbol = this.checker.getSymbolAtLocation(node); | ||
@@ -53,7 +52,5 @@ if (symbol == null) { | ||
} | ||
this._symbolToName.set(symbol, name); | ||
this._symbolToName.set(symbol, { name: name, kind: kind }); | ||
}; | ||
TypeContext.prototype.recordHasTypenameField = function (name) { | ||
this.hasTypename.add(name); | ||
}; | ||
// Record that a type reference `node` | ||
TypeContext.prototype.markUnresolvedType = function (node, name) { | ||
@@ -65,25 +62,23 @@ var symbol = this.checker.getSymbolAtLocation(node); | ||
} | ||
if (symbol.flags & ts.SymbolFlags.Alias) { | ||
// Follow any aliases to get the real type declaration. | ||
symbol = this.checker.getAliasedSymbol(symbol); | ||
} | ||
this._unresolvedTypes.set(name, symbol); | ||
this._unresolvedTypes.set(name, this.resolveSymbol(symbol)); | ||
}; | ||
TypeContext.prototype.resolveTypes = function (doc) { | ||
var _this = this; | ||
var errors = []; | ||
var newDoc = (0, graphql_1.visit)(doc, { | ||
Name: function (t) { | ||
var namedTypeResult = _this.resolveNamedType(t); | ||
if (namedTypeResult.kind === "ERROR") { | ||
errors.push(namedTypeResult.err); | ||
return t; | ||
} | ||
return namedTypeResult.value; | ||
TypeContext.prototype.findSymbolDeclaration = function (startSymbol) { | ||
var _a; | ||
var symbol = this.resolveSymbol(startSymbol); | ||
var declaration = (_a = symbol.declarations) === null || _a === void 0 ? void 0 : _a[0]; | ||
return declaration !== null && declaration !== void 0 ? declaration : null; | ||
}; | ||
// Follow symbol aliases until we find the original symbol. Accounts for | ||
// cyclical aliases. | ||
TypeContext.prototype.resolveSymbol = function (startSymbol) { | ||
var symbol = startSymbol; | ||
var visitedSymbols = new Set(); | ||
while (ts.SymbolFlags.Alias & symbol.flags) { | ||
if (visitedSymbols.has(symbol)) { | ||
throw new Error("Cyclical alias detected. Breaking resolution."); | ||
} | ||
}); | ||
if (errors.length > 0) { | ||
return (0, DiagnosticError_1.err)(errors); | ||
visitedSymbols.add(symbol); | ||
symbol = this.checker.getAliasedSymbol(symbol); | ||
} | ||
return (0, DiagnosticError_1.ok)(newDoc); | ||
return symbol; | ||
}; | ||
@@ -99,23 +94,32 @@ TypeContext.prototype.resolveNamedType = function (unresolved) { | ||
} | ||
var name = this._symbolToName.get(symbol); | ||
if (name == null) { | ||
var nameDefinition = this._symbolToName.get(symbol); | ||
if (nameDefinition == null) { | ||
if (unresolved.loc == null) { | ||
throw new Error("Expected namedType to have a location."); | ||
} | ||
return (0, DiagnosticError_1.err)({ | ||
messageText: "This type is not a valid GraphQL type. Did you mean to annotate it's definition with a `/** @gql */` tag such as `/** @gqlType */` or `/** @gqlInput **/`?", | ||
start: unresolved.loc.start, | ||
length: unresolved.loc.end - unresolved.loc.start, | ||
category: ts.DiagnosticCategory.Error, | ||
code: DiagnosticError_1.FAKE_ERROR_CODE, | ||
file: ts.createSourceFile(unresolved.loc.source.name, unresolved.loc.source.body, ts.ScriptTarget.Latest) | ||
}); | ||
return (0, DiagnosticError_1.err)((0, DiagnosticError_1.gqlErr)(unresolved.loc, E.unresolvedTypeReference())); | ||
} | ||
return (0, DiagnosticError_1.ok)(__assign(__assign({}, unresolved), { value: name })); | ||
return (0, DiagnosticError_1.ok)(__assign(__assign({}, unresolved), { value: nameDefinition.name.value })); | ||
}; | ||
TypeContext.prototype.validateInterfaceImplementorsHaveTypenameField = function () { | ||
return (0, DiagnosticError_1.ok)(null); | ||
TypeContext.prototype.unresolvedNameIsGraphQL = function (unresolved) { | ||
var symbol = this._unresolvedTypes.get(unresolved); | ||
return symbol != null && this._symbolToName.has(symbol); | ||
}; | ||
TypeContext.prototype.getDestFilePath = function (sourceFile) { | ||
return (0, gratsRoot_1.getRelativeOutputPath)(this._options, sourceFile); | ||
// TODO: Merge this with resolveNamedType | ||
TypeContext.prototype.getNameDefinition = function (nameNode) { | ||
var typeNameResult = this.resolveNamedType(nameNode); | ||
if (typeNameResult.kind === "ERROR") { | ||
return (0, DiagnosticError_1.err)([typeNameResult.err]); | ||
} | ||
var symbol = this._unresolvedTypes.get(nameNode); | ||
if (symbol == null) { | ||
// This should have already been handled by resolveNamedType | ||
throw new Error("Expected to find unresolved type."); | ||
} | ||
var nameDefinition = this._symbolToName.get(symbol); | ||
if (nameDefinition == null) { | ||
// This should have already been handled by resolveNamedType | ||
throw new Error("Expected to find name definition."); | ||
} | ||
return (0, DiagnosticError_1.ok)(nameDefinition); | ||
}; | ||
@@ -122,0 +126,0 @@ return TypeContext; |
"use strict"; | ||
exports.__esModule = true; | ||
Object.defineProperty(exports, "__esModule", { value: true }); |
@@ -16,6 +16,9 @@ import { GraphQLError, Location, Source } from "graphql"; | ||
export declare function err<E>(err: E): Err<E>; | ||
export declare function collectResults<T>(results: DiagnosticsResult<T>[]): DiagnosticsResult<T[]>; | ||
export declare function combineResults<T, U>(result1: DiagnosticsResult<T>, result2: DiagnosticsResult<U>): DiagnosticsResult<[T, U]>; | ||
export declare class ReportableDiagnostics { | ||
_host: ts.CompilerHost; | ||
_host: ts.FormatDiagnosticsHost; | ||
_diagnostics: ts.Diagnostic[]; | ||
constructor(host: ts.CompilerHost, diagnostics: ts.Diagnostic[]); | ||
constructor(host: ts.FormatDiagnosticsHost, diagnostics: ts.Diagnostic[]); | ||
static fromDiagnostics(diagnostics: ts.Diagnostic[]): ReportableDiagnostics; | ||
formatDiagnosticsWithColorAndContext(): string; | ||
@@ -26,4 +29,7 @@ formatDiagnosticsWithContext(): string; | ||
export declare function graphQlErrorToDiagnostic(error: GraphQLError): ts.Diagnostic; | ||
export declare function diagnosticAtGraphQLLocation(message: string, loc: Location): ts.Diagnostic; | ||
export declare function gqlErr(loc: Location, message: string, relatedInformation?: ts.DiagnosticRelatedInformation[]): ts.Diagnostic; | ||
export declare function gqlRelated(loc: Location, message: string): ts.DiagnosticRelatedInformation; | ||
export declare function tsErr(node: ts.Node, message: string, relatedInformation?: ts.DiagnosticRelatedInformation[]): ts.Diagnostic; | ||
export declare function tsRelated(node: ts.Node, message: string): ts.DiagnosticRelatedInformation; | ||
export declare function graphqlSourceToSourceFile(source: Source): ts.SourceFile; | ||
export {}; |
"use strict"; | ||
var __values = (this && this.__values) || function(o) { | ||
var s = typeof Symbol === "function" && Symbol.iterator, m = s && o[s], i = 0; | ||
if (m) return m.call(o); | ||
if (o && typeof o.length === "number") return { | ||
next: function () { | ||
if (o && i >= o.length) o = void 0; | ||
return { value: o && o[i++], done: !o }; | ||
} | ||
}; | ||
throw new TypeError(s ? "Object is not iterable." : "Symbol.iterator is not defined."); | ||
}; | ||
var __read = (this && this.__read) || function (o, n) { | ||
@@ -18,15 +29,13 @@ var m = typeof Symbol === "function" && o[Symbol.iterator]; | ||
}; | ||
var __values = (this && this.__values) || function(o) { | ||
var s = typeof Symbol === "function" && Symbol.iterator, m = s && o[s], i = 0; | ||
if (m) return m.call(o); | ||
if (o && typeof o.length === "number") return { | ||
next: function () { | ||
if (o && i >= o.length) o = void 0; | ||
return { value: o && o[i++], done: !o }; | ||
var __spreadArray = (this && this.__spreadArray) || function (to, from, pack) { | ||
if (pack || arguments.length === 2) for (var i = 0, l = from.length, ar; i < l; i++) { | ||
if (ar || !(i in from)) { | ||
if (!ar) ar = Array.prototype.slice.call(from, 0, i); | ||
ar[i] = from[i]; | ||
} | ||
}; | ||
throw new TypeError(s ? "Object is not iterable." : "Symbol.iterator is not defined."); | ||
} | ||
return to.concat(ar || Array.prototype.slice.call(from)); | ||
}; | ||
exports.__esModule = true; | ||
exports.graphqlSourceToSourceFile = exports.diagnosticAtGraphQLLocation = exports.graphQlErrorToDiagnostic = exports.FAKE_ERROR_CODE = exports.ReportableDiagnostics = exports.err = exports.ok = void 0; | ||
Object.defineProperty(exports, "__esModule", { value: true }); | ||
exports.graphqlSourceToSourceFile = exports.tsRelated = exports.tsErr = exports.gqlRelated = exports.gqlErr = exports.graphQlErrorToDiagnostic = exports.FAKE_ERROR_CODE = exports.ReportableDiagnostics = exports.combineResults = exports.collectResults = exports.err = exports.ok = void 0; | ||
var ts = require("typescript"); | ||
@@ -41,2 +50,43 @@ function ok(value) { | ||
exports.err = err; | ||
function collectResults(results) { | ||
var e_1, _a; | ||
var errors = []; | ||
var values = []; | ||
try { | ||
for (var results_1 = __values(results), results_1_1 = results_1.next(); !results_1_1.done; results_1_1 = results_1.next()) { | ||
var result = results_1_1.value; | ||
if (result.kind === "ERROR") { | ||
errors.push.apply(errors, __spreadArray([], __read(result.err), false)); | ||
} | ||
else { | ||
values.push(result.value); | ||
} | ||
} | ||
} | ||
catch (e_1_1) { e_1 = { error: e_1_1 }; } | ||
finally { | ||
try { | ||
if (results_1_1 && !results_1_1.done && (_a = results_1.return)) _a.call(results_1); | ||
} | ||
finally { if (e_1) throw e_1.error; } | ||
} | ||
if (errors.length > 0) { | ||
return err(errors); | ||
} | ||
return ok(values); | ||
} | ||
exports.collectResults = collectResults; | ||
function combineResults(result1, result2) { | ||
if (result1.kind === "ERROR" && result2.kind === "ERROR") { | ||
return err(__spreadArray(__spreadArray([], __read(result1.err), false), __read(result2.err), false)); | ||
} | ||
if (result1.kind === "ERROR") { | ||
return result1; | ||
} | ||
if (result2.kind === "ERROR") { | ||
return result2; | ||
} | ||
return ok([result1.value, result2.value]); | ||
} | ||
exports.combineResults = combineResults; | ||
var ReportableDiagnostics = /** @class */ (function () { | ||
@@ -47,2 +97,12 @@ function ReportableDiagnostics(host, diagnostics) { | ||
} | ||
// If you don't have a host, for example if you error while parsing the | ||
// tsconfig, you can use this method and one will be created for you. | ||
ReportableDiagnostics.fromDiagnostics = function (diagnostics) { | ||
var formatHost = { | ||
getCanonicalFileName: function (path) { return path; }, | ||
getCurrentDirectory: ts.sys.getCurrentDirectory, | ||
getNewLine: function () { return ts.sys.newLine; }, | ||
}; | ||
return new ReportableDiagnostics(formatHost, diagnostics); | ||
}; | ||
ReportableDiagnostics.prototype.formatDiagnosticsWithColorAndContext = function () { | ||
@@ -61,3 +121,3 @@ var formatted = ts.formatDiagnosticsWithColorAndContext(this._diagnostics, this._host); | ||
exports.ReportableDiagnostics = ReportableDiagnostics; | ||
// A madeup error code that we use to fake a TypeScript error code. | ||
// A made-up error code that we use to fake a TypeScript error code. | ||
// We pick a very random number to avoid collisions with real error messages. | ||
@@ -73,3 +133,3 @@ exports.FAKE_ERROR_CODE = 349389149282; | ||
function graphQlErrorToDiagnostic(error) { | ||
var e_1, _a; | ||
var e_2, _a; | ||
var position = error.positions[0]; | ||
@@ -79,3 +139,3 @@ if (position == null) { | ||
} | ||
// Start with baseline location infromation | ||
// Start with baseline location information | ||
var start = position; | ||
@@ -99,18 +159,11 @@ var length = 1; | ||
} | ||
relatedInformation.push({ | ||
category: ts.DiagnosticCategory.Message, | ||
code: exports.FAKE_ERROR_CODE, | ||
messageText: "Related location", | ||
file: graphqlSourceToSourceFile(relatedNode.loc.source), | ||
start: relatedNode.loc.start, | ||
length: relatedNode.loc.end - relatedNode.loc.start | ||
}); | ||
relatedInformation.push(gqlRelated(relatedNode.loc, "Related location")); | ||
} | ||
} | ||
catch (e_1_1) { e_1 = { error: e_1_1 }; } | ||
catch (e_2_1) { e_2 = { error: e_2_1 }; } | ||
finally { | ||
try { | ||
if (rest_1_1 && !rest_1_1.done && (_a = rest_1["return"])) _a.call(rest_1); | ||
if (rest_1_1 && !rest_1_1.done && (_a = rest_1.return)) _a.call(rest_1); | ||
} | ||
finally { if (e_1) throw e_1.error; } | ||
finally { if (e_2) throw e_2.error; } | ||
} | ||
@@ -131,7 +184,7 @@ } | ||
length: length, | ||
relatedInformation: relatedInformation | ||
relatedInformation: relatedInformation, | ||
}; | ||
} | ||
exports.graphQlErrorToDiagnostic = graphQlErrorToDiagnostic; | ||
function diagnosticAtGraphQLLocation(message, loc) { | ||
function gqlErr(loc, message, relatedInformation) { | ||
return { | ||
@@ -143,6 +196,44 @@ messageText: message, | ||
start: loc.start, | ||
length: loc.end - loc.start | ||
length: loc.end - loc.start, | ||
relatedInformation: relatedInformation, | ||
}; | ||
} | ||
exports.diagnosticAtGraphQLLocation = diagnosticAtGraphQLLocation; | ||
exports.gqlErr = gqlErr; | ||
function gqlRelated(loc, message) { | ||
return { | ||
category: ts.DiagnosticCategory.Message, | ||
code: exports.FAKE_ERROR_CODE, | ||
messageText: message, | ||
file: graphqlSourceToSourceFile(loc.source), | ||
start: loc.start, | ||
length: loc.end - loc.start, | ||
}; | ||
} | ||
exports.gqlRelated = gqlRelated; | ||
function tsErr(node, message, relatedInformation) { | ||
var start = node.getStart(); | ||
var length = node.getEnd() - start; | ||
var sourceFile = node.getSourceFile(); | ||
return { | ||
messageText: message, | ||
file: sourceFile, | ||
code: exports.FAKE_ERROR_CODE, | ||
category: ts.DiagnosticCategory.Error, | ||
start: start, | ||
length: length, | ||
relatedInformation: relatedInformation, | ||
}; | ||
} | ||
exports.tsErr = tsErr; | ||
function tsRelated(node, message) { | ||
return { | ||
category: ts.DiagnosticCategory.Message, | ||
code: 0, | ||
file: node.getSourceFile(), | ||
start: node.getStart(), | ||
length: node.getWidth(), | ||
messageText: message, | ||
}; | ||
} | ||
exports.tsRelated = tsRelated; | ||
function graphqlSourceToSourceFile(source) { | ||
@@ -149,0 +240,0 @@ return ts.createSourceFile(source.name, source.body, ts.ScriptTarget.Latest); |
@@ -13,3 +13,3 @@ "use strict"; | ||
}; | ||
exports.__esModule = true; | ||
Object.defineProperty(exports, "__esModule", { value: true }); | ||
exports.traverseJSDocTags = void 0; | ||
@@ -42,3 +42,3 @@ var ts = require("typescript"); | ||
try { | ||
if (_c && !_c.done && (_a = _b["return"])) _a.call(_b); | ||
if (_c && !_c.done && (_a = _b.return)) _a.call(_b); | ||
} | ||
@@ -45,0 +45,0 @@ finally { if (e_1) throw e_1.error; } |
{ | ||
"name": "grats", | ||
"version": "0.0.0-main-8de1c8ac", | ||
"version": "0.0.0-main-8e63ea66", | ||
"main": "dist/src/index.js", | ||
@@ -12,8 +12,8 @@ "bin": "dist/src/cli.js", | ||
"dependencies": { | ||
"@graphql-tools/utils": "^9.2.1", | ||
"commander": "^10.0.0", | ||
"graphql": "^16.6.0", | ||
"typescript": "^4.9.5" | ||
"typescript": "^5.0.2" | ||
}, | ||
"devDependencies": { | ||
"@graphql-tools/utils": "^9.2.1", | ||
"@types/node": "^18.14.6", | ||
@@ -26,2 +26,3 @@ "@typescript-eslint/eslint-plugin": "^5.55.0", | ||
"path-browserify": "^1.0.1", | ||
"prettier": "^2.8.7", | ||
"process": "^0.11.10", | ||
@@ -33,9 +34,14 @@ "ts-node": "^10.9.1" | ||
}, | ||
"packageManager": "pnpm@8.1.1", | ||
"packageManager": "pnpm@8.12.0", | ||
"engines": { | ||
"node": ">=16 <=21", | ||
"pnpm": "^8" | ||
}, | ||
"scripts": { | ||
"test": "ts-node --esm src/tests/test.ts", | ||
"test": "ts-node src/tests/test.ts", | ||
"integration-tests": "node src/tests/integration.mjs", | ||
"build": "tsc --build", | ||
"lint": "eslint src/**/*.ts" | ||
"format": "prettier . --write", | ||
"lint": "eslint src/**/*.ts && prettier . --check" | ||
} | ||
} |
@@ -1,5 +0,1 @@ | ||
# -=[ ALPHA SOFTWARE ]=- | ||
**Grats is still experimental. Feel free to try it out and give feedback, but they api is still in flux** | ||
# Grats: Implementation-First GraphQL for TypeScript | ||
@@ -9,5 +5,7 @@ | ||
_Beta Software: Grats is largely stable and being used in production in multiple places. If you encounter any issues, dont hesitate to let us know._ | ||
**What if building a GraphQL server were as simple as just writing functions?** | ||
When you write your GraphQL server in TypeScript, your fields and resovlers | ||
When you write your GraphQL server in TypeScript, your fields and resolvers | ||
are _already_ annotated with type information. _Grats leverages your existing | ||
@@ -18,7 +16,33 @@ type annotations to automatically extract an executable GraphQL schema from your | ||
By making your TypeScript implementation the source of truth, you never have to | ||
worry about valiating that your implementiaton matches your schema. Your | ||
worry about validating that your implementation matches your schema. Your | ||
implementation _is_ your schema! | ||
## Read the docs: https://grats.capt.dev/ | ||
## Example | ||
Here's what it looks like to define a User type with a greeting field using Grats: | ||
```ts | ||
/** @gqlType */ | ||
class User { | ||
/** @gqlField */ | ||
name: string; | ||
/** @gqlField */ | ||
greet(args: { greeting: string }): string { | ||
return `${args.greeting}, ${this.name}`; | ||
} | ||
} | ||
``` | ||
After running `npx grats`, you'll find a `schema.ts` module that exports an executable schema, and a `schema.graphql` file contins your GraphQL schema definition: | ||
```graphql | ||
type User { | ||
name: String | ||
greet(greeting: String!): String | ||
} | ||
``` | ||
That's just the begining! To learn more, **Read the docs: https://grats.capt.dev/** | ||
## Contributing | ||
@@ -30,9 +54,9 @@ | ||
* [@mofeiZ](https://github.com/mofeiZ) and [@alunyov](https://github/alunyov) for their Relay hack-week project exploring a similar idea. | ||
* [@josephsavona](https://github.com/josephsavona) for input on the design of [Relay Resolvers](https://relay.dev/docs/guides/relay-resolvers/) which inspired this project. | ||
* [@bradzacher](https://github.com/bradzacher) for tips on how to handle TypeScript ASTs. | ||
* Everyone who worked on Meta's Hack GraphQL server, the developer experince of which inspired this project. | ||
* A number of other projects which seem to have explored similar ideas in the past: | ||
* [ts2gql](https://github.com/convoyinc/ts2gql) | ||
* [ts2graphql](https://github.com/cevek/ts2graphql) | ||
* [typegraphql-reflection-poc](https://github.com/MichalLytek/typegraphql-reflection-poc) | ||
- [@mofeiZ](https://github.com/mofeiZ) and [@alunyov](https://github/alunyov) for their Relay hack-week project exploring a similar idea. | ||
- [@josephsavona](https://github.com/josephsavona) for input on the design of [Relay Resolvers](https://relay.dev/docs/guides/relay-resolvers/) which inspired this project. | ||
- [@bradzacher](https://github.com/bradzacher) for tips on how to handle TypeScript ASTs. | ||
- Everyone who worked on Meta's Hack GraphQL server, the developer experince of which inspired this project. | ||
- A number of other projects which seem to have explored similar ideas in the past: | ||
- [ts2gql](https://github.com/convoyinc/ts2gql) | ||
- [ts2graphql](https://github.com/cevek/ts2graphql) | ||
- [typegraphql-reflection-poc](https://github.com/MichalLytek/typegraphql-reflection-poc) |
Major refactor
Supply chain riskPackage has recently undergone a major refactor. It may be unstable or indicate significant internal changes. Use caution when updating to versions that include significant changes.
Found 1 instance in 1 package
Dynamic require
Supply chain riskDynamic require can indicate the package is performing dangerous or unsafe dynamic code execution.
Found 1 instance in 1 package
Dynamic require
Supply chain riskDynamic require can indicate the package is performing dangerous or unsafe dynamic code execution.
Found 1 instance in 1 package
277347
74%3
-25%107
205.71%6100
81.98%60
66.67%11
22.22%1
Infinity%+ Added
- Removed
- Removed
- Removed
- Removed
- Removed
Updated