Skip to content

Commit f6909d1

Browse files
authoredMar 21, 2025··
[resolvers] CODEGEN-500: Support semanticNonNull custom directive (#10315)
* Refactor FieldDefinition to bring related logic together * Implement customDirectives.semanticNonNull * Add changeset * Use graphql-sock to transform schema when generating * Fix doc and errors
1 parent ba18e84 commit f6909d1

File tree

6 files changed

+168
-18
lines changed

6 files changed

+168
-18
lines changed
 

‎.changeset/old-gorillas-taste.md

+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
---
2+
'@graphql-codegen/visitor-plugin-common': minor
3+
'@graphql-codegen/typescript-resolvers': minor
4+
---
5+
6+
Implement semanticNonNull custom directive

‎packages/plugins/other/visitor-plugin-common/src/base-resolvers-visitor.ts

+67-14
Original file line numberDiff line numberDiff line change
@@ -549,6 +549,40 @@ export interface RawResolversConfig extends RawConfig {
549549
* ```
550550
*/
551551
enumSuffix?: boolean;
552+
/**
553+
* @description Configures behavior for custom directives from various GraphQL libraries.
554+
* @exampleMarkdown
555+
* ## `@semanticNonNull`
556+
* First, install `graphql-sock` peer dependency:
557+
*
558+
* ```sh npm2yarn
559+
* npm install -D graphql-sock
560+
* ```
561+
*
562+
* Now, you can enable support for `@semanticNonNull` directive:
563+
*
564+
* ```ts filename="codegen.ts"
565+
* import type { CodegenConfig } from '@graphql-codegen/cli';
566+
*
567+
* const config: CodegenConfig = {
568+
* // ...
569+
* generates: {
570+
* 'path/to/file.ts': {
571+
* plugins: ['typescript-resolvers'],
572+
* config: {
573+
* customDirectives: {
574+
* semanticNonNull: true
575+
* }
576+
* },
577+
* },
578+
* },
579+
* };
580+
* export default config;
581+
* ```
582+
*/
583+
customDirectives?: {
584+
semanticNonNull?: boolean;
585+
};
552586
/**
553587
* @default false
554588
* @description Sets the `__resolveType` field as optional field.
@@ -1449,8 +1483,6 @@ export class BaseResolversVisitor<
14491483

14501484
return (parentName, avoidResolverOptionals) => {
14511485
const original: FieldDefinitionNode = parent[key];
1452-
const baseType = getBaseTypeNode(original.type);
1453-
const realType = baseType.name.value;
14541486
const parentType = this.schema.getType(parentName);
14551487

14561488
if (this._federation.skipField({ fieldNode: original, parentType })) {
@@ -1459,11 +1491,6 @@ export class BaseResolversVisitor<
14591491

14601492
const contextType = this.getContextType(parentName, node);
14611493

1462-
const typeToUse = this.getTypeToUse(realType);
1463-
const mappedType = this._variablesTransformer.wrapAstTypeWithModifiers(typeToUse, original.type);
1464-
const subscriptionType = this._schema.getSubscriptionType();
1465-
const isSubscriptionType = subscriptionType && subscriptionType.name === parentName;
1466-
14671494
let argsType = hasArguments
14681495
? this.convertName(
14691496
parentName +
@@ -1499,15 +1526,41 @@ export class BaseResolversVisitor<
14991526
parentType,
15001527
parentTypeSignature: this.getParentTypeForSignature(node),
15011528
});
1502-
const mappedTypeKey = isSubscriptionType ? `${mappedType}, "${node.name}"` : mappedType;
15031529

1504-
const directiveMappings =
1505-
node.directives
1506-
?.map(directive => this._directiveResolverMappings[directive.name as any])
1507-
.filter(Boolean)
1508-
.reverse() ?? [];
1530+
const { mappedTypeKey, resolverType } = ((): { mappedTypeKey: string; resolverType: string } => {
1531+
const baseType = getBaseTypeNode(original.type);
1532+
const realType = baseType.name.value;
1533+
const typeToUse = this.getTypeToUse(realType);
1534+
/**
1535+
* Turns GraphQL type to TypeScript types (`mappedType`) e.g.
1536+
* - String! -> ResolversTypes['String']>
1537+
* - String -> Maybe<ResolversTypes['String']>
1538+
* - [String] -> Maybe<Array<Maybe<ResolversTypes['String']>>>
1539+
* - [String!]! -> Array<ResolversTypes['String']>
1540+
*/
1541+
const mappedType = this._variablesTransformer.wrapAstTypeWithModifiers(typeToUse, original.type);
1542+
1543+
const subscriptionType = this._schema.getSubscriptionType();
1544+
const isSubscriptionType = subscriptionType && subscriptionType.name === parentName;
1545+
1546+
if (isSubscriptionType) {
1547+
return {
1548+
mappedTypeKey: `${mappedType}, "${node.name}"`,
1549+
resolverType: 'SubscriptionResolver',
1550+
};
1551+
}
1552+
1553+
const directiveMappings =
1554+
node.directives
1555+
?.map(directive => this._directiveResolverMappings[directive.name as any])
1556+
.filter(Boolean)
1557+
.reverse() ?? [];
15091558

1510-
const resolverType = isSubscriptionType ? 'SubscriptionResolver' : directiveMappings[0] ?? 'Resolver';
1559+
return {
1560+
mappedTypeKey: mappedType,
1561+
resolverType: directiveMappings[0] ?? 'Resolver',
1562+
};
1563+
})();
15111564

15121565
const signature: {
15131566
name: string;

‎packages/plugins/typescript/resolvers/package.json

+4-2
Original file line numberDiff line numberDiff line change
@@ -21,10 +21,12 @@
2121
"tslib": "~2.6.0"
2222
},
2323
"peerDependencies": {
24-
"graphql": "^0.8.0 || ^0.9.0 || ^0.10.0 || ^0.11.0 || ^0.12.0 || ^0.13.0 || ^14.0.0 || ^15.0.0 || ^16.0.0"
24+
"graphql": "^0.8.0 || ^0.9.0 || ^0.10.0 || ^0.11.0 || ^0.12.0 || ^0.13.0 || ^14.0.0 || ^15.0.0 || ^16.0.0",
25+
"graphql-sock": "^1.0.0"
2526
},
2627
"devDependencies": {
27-
"graphql-subscriptions": "3.0.0"
28+
"graphql-subscriptions": "3.0.0",
29+
"graphql-sock": "1.0.0"
2830
},
2931
"main": "dist/cjs/index.js",
3032
"module": "dist/esm/index.js",

‎packages/plugins/typescript/resolvers/src/index.ts

+17-2
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,7 @@ export const plugin: PluginFunction<
1717
Types.ComplexPluginOutput<{
1818
generatedResolverTypes: RootResolver['generatedResolverTypes'];
1919
}>
20-
> = (schema: GraphQLSchema, documents: Types.DocumentFile[], config: TypeScriptResolversPluginConfig) => {
20+
> = async (schema: GraphQLSchema, documents: Types.DocumentFile[], config: TypeScriptResolversPluginConfig) => {
2121
const imports = [];
2222
if (!config.customResolveInfo) {
2323
imports.push('GraphQLResolveInfo');
@@ -75,7 +75,11 @@ export type Resolver${capitalizedDirectiveName}WithResolve<TResult, TParent, TCo
7575
}
7676
}
7777

78-
const transformedSchema = config.federation ? addFederationReferencesToSchema(schema) : schema;
78+
let transformedSchema = config.federation ? addFederationReferencesToSchema(schema) : schema;
79+
transformedSchema = config.customDirectives?.semanticNonNull
80+
? await semanticToStrict(transformedSchema)
81+
: transformedSchema;
82+
7983
const visitor = new TypeScriptResolversVisitor({ ...config, directiveResolverMappings }, transformedSchema);
8084
const namespacedImportPrefix = visitor.config.namespacedImportName ? `${visitor.config.namespacedImportName}.` : '';
8185

@@ -302,3 +306,14 @@ export type DirectiveResolverFn<TResult = {}, TParent = {}, TContext = {}, TArgs
302306
};
303307

304308
export { TypeScriptResolversPluginConfig, TypeScriptResolversVisitor };
309+
310+
const semanticToStrict = async (schema: GraphQLSchema): Promise<GraphQLSchema> => {
311+
try {
312+
const sock = await import('graphql-sock');
313+
return sock.semanticToStrict(schema);
314+
} catch {
315+
throw new Error(
316+
"To use the `customDirective.semanticNonNull` option, you must install the 'graphql-sock' package."
317+
);
318+
}
319+
};
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,69 @@
1+
import { buildSchema } from 'graphql';
2+
import '@graphql-codegen/testing';
3+
import { plugin } from '../src/index.js';
4+
5+
describe('customDirectives.sematicNonNull', () => {
6+
it('allowSemanticNonNull - should build strict type if annotated by @semanticNonNull directive', async () => {
7+
const testingSchema = buildSchema(/* GraphQL */ `
8+
directive @semanticNonNull(levels: [Int] = [0]) on FIELD_DEFINITION
9+
10+
type TestingType {
11+
field: String @semanticNonNull
12+
fieldLevel0: String @semanticNonNull(levels: [0])
13+
fieldLevel1: String @semanticNonNull(levels: [1])
14+
fieldBothLevels: String @semanticNonNull(levels: [0, 1])
15+
list: [String] @semanticNonNull
16+
listLevel0: [String] @semanticNonNull(levels: [0])
17+
listLevel1: [String] @semanticNonNull(levels: [1])
18+
listBothLevels: [String] @semanticNonNull(levels: [0, 1])
19+
nonNullableList: [String]! @semanticNonNull
20+
nonNullableListLevel0: [String]! @semanticNonNull(levels: [0])
21+
nonNullableListLevel1: [String]! @semanticNonNull(levels: [1])
22+
nonNullableListBothLevels: [String]! @semanticNonNull(levels: [0, 1])
23+
listWithNonNullableItem: [String!] @semanticNonNull
24+
listWithNonNullableItemLevel0: [String!] @semanticNonNull(levels: [0])
25+
listWithNonNullableItemLevel1: [String!] @semanticNonNull(levels: [1])
26+
listWithNonNullableItemBothLevels: [String!] @semanticNonNull(levels: [0, 1])
27+
nonNullableListWithNonNullableItem: [String!]! @semanticNonNull
28+
nonNullableListWithNonNullableItemLevel0: [String!]! @semanticNonNull(levels: [0])
29+
nonNullableListWithNonNullableItemLevel1: [String!]! @semanticNonNull(levels: [1])
30+
nonNullableListWithNonNullableItemBothLevels: [String!]! @semanticNonNull(levels: [0, 1])
31+
}
32+
`);
33+
34+
const result = await plugin(
35+
testingSchema,
36+
[],
37+
{
38+
customDirectives: { semanticNonNull: true },
39+
},
40+
{ outputFile: '' }
41+
);
42+
43+
expect(result.content).toBeSimilarStringTo(`
44+
export type TestingTypeResolvers<ContextType = any, ParentType extends ResolversParentTypes['TestingType'] = ResolversParentTypes['TestingType']> = {
45+
field?: Resolver<ResolversTypes['String'], ParentType, ContextType>;
46+
fieldLevel0?: Resolver<ResolversTypes['String'], ParentType, ContextType>;
47+
fieldLevel1?: Resolver<Maybe<ResolversTypes['String']>, ParentType, ContextType>;
48+
fieldBothLevels?: Resolver<ResolversTypes['String'], ParentType, ContextType>;
49+
list?: Resolver<Array<Maybe<ResolversTypes['String']>>, ParentType, ContextType>;
50+
listLevel0?: Resolver<Array<Maybe<ResolversTypes['String']>>, ParentType, ContextType>;
51+
listLevel1?: Resolver<Maybe<Array<ResolversTypes['String']>>, ParentType, ContextType>;
52+
listBothLevels?: Resolver<Array<ResolversTypes['String']>, ParentType, ContextType>;
53+
nonNullableList?: Resolver<Array<Maybe<ResolversTypes['String']>>, ParentType, ContextType>;
54+
nonNullableListLevel0?: Resolver<Array<Maybe<ResolversTypes['String']>>, ParentType, ContextType>;
55+
nonNullableListLevel1?: Resolver<Array<ResolversTypes['String']>, ParentType, ContextType>;
56+
nonNullableListBothLevels?: Resolver<Array<ResolversTypes['String']>, ParentType, ContextType>;
57+
listWithNonNullableItem?: Resolver<Array<ResolversTypes['String']>, ParentType, ContextType>;
58+
listWithNonNullableItemLevel0?: Resolver<Array<ResolversTypes['String']>, ParentType, ContextType>;
59+
listWithNonNullableItemLevel1?: Resolver<Maybe<Array<ResolversTypes['String']>>, ParentType, ContextType>;
60+
listWithNonNullableItemBothLevels?: Resolver<Array<ResolversTypes['String']>, ParentType, ContextType>;
61+
nonNullableListWithNonNullableItem?: Resolver<Array<ResolversTypes['String']>, ParentType, ContextType>;
62+
nonNullableListWithNonNullableItemLevel0?: Resolver<Array<ResolversTypes['String']>, ParentType, ContextType>;
63+
nonNullableListWithNonNullableItemLevel1?: Resolver<Array<ResolversTypes['String']>, ParentType, ContextType>;
64+
nonNullableListWithNonNullableItemBothLevels?: Resolver<Array<ResolversTypes['String']>, ParentType, ContextType>;
65+
__isTypeOf?: IsTypeOfResolverFn<ParentType, ContextType>;
66+
};
67+
`);
68+
});
69+
});

‎yarn.lock

+5
Original file line numberDiff line numberDiff line change
@@ -9094,6 +9094,11 @@ graphql-request@^6.0.0:
90949094
"@graphql-typed-document-node/core" "^3.2.0"
90959095
cross-fetch "^3.1.5"
90969096

9097+
graphql-sock@1.0.0:
9098+
version "1.0.0"
9099+
resolved "https://registry.yarnpkg.com/graphql-sock/-/graphql-sock-1.0.0.tgz#efdfb991cbd8a37da91d7e2e81c2955945c6df37"
9100+
integrity sha512-pvODB7YaQ/K80pBOG7AnQC65R0UgE4hZv67VtwVjyQ0u2N+KMKVy8HEcAnYTK27BfVyTQCunSfX114u0xHOBMQ==
9101+
90979102
graphql-subscriptions@3.0.0:
90989103
version "3.0.0"
90999104
resolved "https://registry.yarnpkg.com/graphql-subscriptions/-/graphql-subscriptions-3.0.0.tgz#820c846ef271414c08f64827b5c9a192801e1b6f"

0 commit comments

Comments
 (0)
Please sign in to comment.