validate-custom-fields-interceptor.ts 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171
  1. import { CallHandler, ExecutionContext, Injectable, NestInterceptor } from '@nestjs/common';
  2. import { ModuleRef } from '@nestjs/core';
  3. import { GqlExecutionContext } from '@nestjs/graphql';
  4. import { LanguageCode } from '@vendure/common/lib/generated-types';
  5. import { getGraphQlInputName } from '@vendure/common/lib/shared-utils';
  6. import {
  7. GraphQLInputType,
  8. GraphQLList,
  9. GraphQLNonNull,
  10. GraphQLSchema,
  11. OperationDefinitionNode,
  12. TypeNode,
  13. } from 'graphql';
  14. import { REQUEST_CONTEXT_KEY } from '../../common/constants';
  15. import { Injector } from '../../common/injector';
  16. import { ConfigService } from '../../config/config.service';
  17. import { CustomFieldConfig, CustomFields } from '../../config/custom-field/custom-field-types';
  18. import { parseContext } from '../common/parse-context';
  19. import { RequestContext } from '../common/request-context';
  20. import { validateCustomFieldValue } from '../common/validate-custom-field-value';
  21. /**
  22. * This interceptor is responsible for enforcing the validation constraints defined for any CustomFields.
  23. * For example, if a custom 'int' field has a "min" value of 0, and a mutation attempts to set its value
  24. * to a negative integer, then that mutation will fail with an error.
  25. */
  26. @Injectable()
  27. export class ValidateCustomFieldsInterceptor implements NestInterceptor {
  28. private readonly inputsWithCustomFields: Set<string>;
  29. constructor(private configService: ConfigService, private moduleRef: ModuleRef) {
  30. this.inputsWithCustomFields = Object.keys(configService.customFields).reduce((inputs, entityName) => {
  31. inputs.add(`Create${entityName}Input`);
  32. inputs.add(`Update${entityName}Input`);
  33. return inputs;
  34. }, new Set<string>());
  35. this.inputsWithCustomFields.add('OrderLineCustomFieldsInput');
  36. }
  37. async intercept(context: ExecutionContext, next: CallHandler<any>) {
  38. const parsedContext = parseContext(context);
  39. const injector = new Injector(this.moduleRef);
  40. if (parsedContext.isGraphQL) {
  41. const gqlExecutionContext = GqlExecutionContext.create(context);
  42. const { operation, schema } = parsedContext.info;
  43. const variables = gqlExecutionContext.getArgs();
  44. const ctx: RequestContext = (parsedContext.req as any)[REQUEST_CONTEXT_KEY];
  45. if (operation.operation === 'mutation') {
  46. const inputTypeNames = this.getArgumentMap(operation, schema);
  47. for (const [inputName, typeName] of Object.entries(inputTypeNames)) {
  48. if (this.inputsWithCustomFields.has(typeName)) {
  49. if (variables[inputName]) {
  50. const inputVariables: Array<Record<string, any>> = Array.isArray(
  51. variables[inputName],
  52. )
  53. ? variables[inputName]
  54. : [variables[inputName]];
  55. for (const inputVariable of inputVariables) {
  56. await this.validateInput(typeName, ctx.languageCode, injector, inputVariable);
  57. }
  58. }
  59. }
  60. }
  61. }
  62. }
  63. return next.handle();
  64. }
  65. private async validateInput(
  66. typeName: string,
  67. languageCode: LanguageCode,
  68. injector: Injector,
  69. variableValues?: { [key: string]: any },
  70. ) {
  71. if (variableValues) {
  72. const entityName = typeName.replace(/(Create|Update)(.+)Input/, '$2');
  73. const customFieldConfig = this.configService.customFields[entityName as keyof CustomFields];
  74. if (typeName === 'OrderLineCustomFieldsInput') {
  75. // special case needed to handle custom fields passed via addItemToOrder or adjustOrderLine
  76. // mutations.
  77. await this.validateCustomFieldsObject(
  78. this.configService.customFields.OrderLine,
  79. languageCode,
  80. variableValues,
  81. injector,
  82. );
  83. }
  84. if (variableValues.customFields) {
  85. await this.validateCustomFieldsObject(
  86. customFieldConfig,
  87. languageCode,
  88. variableValues.customFields,
  89. injector,
  90. );
  91. }
  92. const translations = variableValues.translations;
  93. if (Array.isArray(translations)) {
  94. for (const translation of translations) {
  95. if (translation.customFields) {
  96. await this.validateCustomFieldsObject(
  97. customFieldConfig,
  98. languageCode,
  99. translation.customFields,
  100. injector,
  101. );
  102. }
  103. }
  104. }
  105. }
  106. }
  107. private async validateCustomFieldsObject(
  108. customFieldConfig: CustomFieldConfig[],
  109. languageCode: LanguageCode,
  110. customFieldsObject: { [key: string]: any },
  111. injector: Injector,
  112. ) {
  113. for (const [key, value] of Object.entries(customFieldsObject)) {
  114. const config = customFieldConfig.find(c => getGraphQlInputName(c) === key);
  115. if (config) {
  116. await validateCustomFieldValue(config, value, injector, languageCode);
  117. }
  118. }
  119. }
  120. private getArgumentMap(
  121. operation: OperationDefinitionNode,
  122. schema: GraphQLSchema,
  123. ): { [inputName: string]: string } {
  124. const mutationType = schema.getMutationType();
  125. if (!mutationType) {
  126. return {};
  127. }
  128. const map: { [inputName: string]: string } = {};
  129. for (const selection of operation.selectionSet.selections) {
  130. if (selection.kind === 'Field') {
  131. const name = selection.name.value;
  132. const inputType = mutationType.getFields()[name];
  133. if (!inputType) continue;
  134. for (const arg of inputType.args) {
  135. map[arg.name] = this.getInputTypeName(arg.type);
  136. }
  137. }
  138. }
  139. return map;
  140. }
  141. private getNamedTypeName(type: TypeNode): string {
  142. if (type.kind === 'NonNullType' || type.kind === 'ListType') {
  143. return this.getNamedTypeName(type.type);
  144. } else {
  145. return type.name.value;
  146. }
  147. }
  148. private getInputTypeName(type: GraphQLInputType): string {
  149. if (type instanceof GraphQLNonNull) {
  150. return this.getInputTypeName(type.ofType);
  151. }
  152. if (type instanceof GraphQLList) {
  153. return this.getInputTypeName(type.ofType);
  154. }
  155. return type.name;
  156. }
  157. }