@lenne.tech/nest-server 9.3.0 → 9.4.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (102) hide show
  1. package/dist/config.env.js +0 -3
  2. package/dist/config.env.js.map +1 -1
  3. package/dist/core/common/args/filter.args.js.map +1 -1
  4. package/dist/core/common/args/pagination.args.js.map +1 -1
  5. package/dist/core/common/decorators/restricted.decorator.js +1 -0
  6. package/dist/core/common/decorators/restricted.decorator.js.map +1 -1
  7. package/dist/core/common/enums/role.enum.d.ts +2 -1
  8. package/dist/core/common/enums/role.enum.js +1 -0
  9. package/dist/core/common/enums/role.enum.js.map +1 -1
  10. package/dist/core/common/filters/http-exception-log.filter.js.map +1 -1
  11. package/dist/core/common/helpers/input.helper.d.ts +2 -0
  12. package/dist/core/common/helpers/input.helper.js +28 -1
  13. package/dist/core/common/helpers/input.helper.js.map +1 -1
  14. package/dist/core/common/inputs/combined-filter.input.js.map +1 -1
  15. package/dist/core/common/inputs/filter.input.js.map +1 -1
  16. package/dist/core/common/inputs/single-filter.input.js.map +1 -1
  17. package/dist/core/common/inputs/sort.input.js.map +1 -1
  18. package/dist/core/common/interceptors/check-response.interceptor.js.map +1 -1
  19. package/dist/core/common/interceptors/check-security.interceptor.js.map +1 -1
  20. package/dist/core/common/middlewares/to-lower-case.middleware.d.ts +2 -0
  21. package/dist/core/common/middlewares/to-lower-case.middleware.js +9 -0
  22. package/dist/core/common/middlewares/to-lower-case.middleware.js.map +1 -0
  23. package/dist/core/common/models/core-persistence.model.js.map +1 -1
  24. package/dist/core/common/pipes/check-input.pipe.js.map +1 -1
  25. package/dist/core/common/pipes/map-and-validate.pipe.js.map +1 -1
  26. package/dist/core/common/plugins/complexity.plugin.js.map +1 -1
  27. package/dist/core/common/scalars/any.scalar.js.map +1 -1
  28. package/dist/core/common/scalars/date-timestamp.scalar.d.ts +7 -0
  29. package/dist/core/common/scalars/date-timestamp.scalar.js +30 -0
  30. package/dist/core/common/scalars/date-timestamp.scalar.js.map +1 -0
  31. package/dist/core/common/scalars/date.scalar.js +3 -20
  32. package/dist/core/common/scalars/date.scalar.js.map +1 -1
  33. package/dist/core/common/scalars/json.scalar.js.map +1 -1
  34. package/dist/core/common/services/config.service.js +1 -1
  35. package/dist/core/common/services/config.service.js.map +1 -1
  36. package/dist/core/common/services/email.service.js.map +1 -1
  37. package/dist/core/common/services/mailjet.service.js.map +1 -1
  38. package/dist/core/common/services/template.service.js.map +1 -1
  39. package/dist/core/modules/auth/core-auth.controller.js.map +1 -1
  40. package/dist/core/modules/auth/core-auth.model.js.map +1 -1
  41. package/dist/core/modules/auth/core-auth.module.js.map +1 -1
  42. package/dist/core/modules/auth/core-auth.resolver.js.map +1 -1
  43. package/dist/core/modules/auth/guards/refresh-token.guard.js.map +1 -1
  44. package/dist/core/modules/auth/guards/roles.guard.js.map +1 -1
  45. package/dist/core/modules/auth/inputs/core-auth-sign-in.input.js.map +1 -1
  46. package/dist/core/modules/auth/inputs/core-auth-sign-up.input.js.map +1 -1
  47. package/dist/core/modules/auth/services/core-auth.service.js.map +1 -1
  48. package/dist/core/modules/auth/strategies/jwt-refresh.strategy.js.map +1 -1
  49. package/dist/core/modules/auth/strategies/jwt.strategy.js.map +1 -1
  50. package/dist/core/modules/file/core-file-info.model.js.map +1 -1
  51. package/dist/core/modules/file/core-file.controller.js.map +1 -1
  52. package/dist/core/modules/file/core-file.resolver.js.map +1 -1
  53. package/dist/core/modules/user/core-user.model.js +1 -1
  54. package/dist/core/modules/user/core-user.model.js.map +1 -1
  55. package/dist/core/modules/user/inputs/core-user-create.input.js.map +1 -1
  56. package/dist/core/modules/user/inputs/core-user.input.js.map +1 -1
  57. package/dist/core.module.js.map +1 -1
  58. package/dist/index.d.ts +2 -0
  59. package/dist/index.js +2 -0
  60. package/dist/index.js.map +1 -1
  61. package/dist/server/common/models/persistence.model.js.map +1 -1
  62. package/dist/server/common/services/cron-jobs.service.js.map +1 -1
  63. package/dist/server/modules/auth/auth.controller.js.map +1 -1
  64. package/dist/server/modules/auth/auth.model.js.map +1 -1
  65. package/dist/server/modules/auth/auth.module.js.map +1 -1
  66. package/dist/server/modules/auth/auth.resolver.js.map +1 -1
  67. package/dist/server/modules/auth/auth.service.js.map +1 -1
  68. package/dist/server/modules/auth/inputs/auth-sign-in.input.js.map +1 -1
  69. package/dist/server/modules/auth/inputs/auth-sign-up.input.js.map +1 -1
  70. package/dist/server/modules/file/file-info.model.js.map +1 -1
  71. package/dist/server/modules/file/file.controller.js.map +1 -1
  72. package/dist/server/modules/file/file.module.js.map +1 -1
  73. package/dist/server/modules/file/file.resolver.js.map +1 -1
  74. package/dist/server/modules/file/file.service.js.map +1 -1
  75. package/dist/server/modules/file/multer-config.service.js.map +1 -1
  76. package/dist/server/modules/user/avatar.controller.js.map +1 -1
  77. package/dist/server/modules/user/inputs/user-create.input.js.map +1 -1
  78. package/dist/server/modules/user/inputs/user.input.js.map +1 -1
  79. package/dist/server/modules/user/outputs/find-and-count-users-result.output.js.map +1 -1
  80. package/dist/server/modules/user/user.model.js.map +1 -1
  81. package/dist/server/modules/user/user.module.js.map +1 -1
  82. package/dist/server/modules/user/user.resolver.js +3 -3
  83. package/dist/server/modules/user/user.resolver.js.map +1 -1
  84. package/dist/server/modules/user/user.service.js.map +1 -1
  85. package/dist/server/server.controller.js.map +1 -1
  86. package/dist/server/server.module.js.map +1 -1
  87. package/dist/test/test.helper.js +1 -0
  88. package/dist/test/test.helper.js.map +1 -1
  89. package/dist/tsconfig.build.tsbuildinfo +1 -1
  90. package/package.json +28 -28
  91. package/src/config.env.ts +0 -3
  92. package/src/core/common/decorators/restricted.decorator.ts +1 -0
  93. package/src/core/common/enums/role.enum.ts +3 -0
  94. package/src/core/common/helpers/input.helper.ts +46 -0
  95. package/src/core/common/middlewares/to-lower-case.middleware.ts +9 -0
  96. package/src/core/common/scalars/date-timestamp.scalar.ts +32 -0
  97. package/src/core/common/scalars/date.scalar.ts +3 -34
  98. package/src/core/modules/user/core-user.model.ts +1 -1
  99. package/src/core/modules/user/inputs/core-user.input.ts +1 -0
  100. package/src/index.ts +2 -0
  101. package/src/server/modules/user/user.resolver.ts +3 -3
  102. package/src/test/test.helper.ts +2 -0
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@lenne.tech/nest-server",
3
- "version": "9.3.0",
3
+ "version": "9.4.1",
4
4
  "description": "Modern, fast, powerful Node.js web framework in TypeScript based on Nest with a GraphQL API and a connection to MongoDB (or other databases).",
5
5
  "keywords": [
6
6
  "node",
@@ -60,15 +60,15 @@
60
60
  "node": ">= 16.13.0"
61
61
  },
62
62
  "dependencies": {
63
- "@apollo/gateway": "2.3.2",
64
- "@nestjs/apollo": "10.2.0",
65
- "@nestjs/common": "9.3.9",
66
- "@nestjs/core": "9.3.9",
67
- "@nestjs/graphql": "10.2.0",
63
+ "@apollo/gateway": "2.4.0",
64
+ "@nestjs/apollo": "11.0.4",
65
+ "@nestjs/common": "9.3.12",
66
+ "@nestjs/core": "9.3.12",
67
+ "@nestjs/graphql": "11.0.4",
68
68
  "@nestjs/jwt": "10.0.2",
69
- "@nestjs/mongoose": "9.2.1",
69
+ "@nestjs/mongoose": "9.2.2",
70
70
  "@nestjs/passport": "9.0.3",
71
- "@nestjs/platform-express": "9.3.9",
71
+ "@nestjs/platform-express": "9.3.12",
72
72
  "@nestjs/schedule": "2.2.0",
73
73
  "apollo-server-core": "3.11.1",
74
74
  "apollo-server-express": "3.11.1",
@@ -77,48 +77,48 @@
77
77
  "class-validator": "0.14.0",
78
78
  "compression": "1.7.4",
79
79
  "cookie-parser": "1.4.6",
80
- "ejs": "3.1.8",
80
+ "ejs": "3.1.9",
81
81
  "graphql": "16.6.0",
82
82
  "graphql-query-complexity": "0.12.0",
83
83
  "graphql-subscriptions": "2.0.0",
84
84
  "graphql-upload": "15.0.2",
85
85
  "js-sha256": "0.9.0",
86
86
  "json-to-graphql-query": "2.2.5",
87
- "light-my-request": "5.8.0",
87
+ "light-my-request": "5.9.1",
88
88
  "lodash": "4.17.21",
89
89
  "mongodb": "4.14.0",
90
- "mongoose": "6.9.1",
90
+ "mongoose": "6.10.4",
91
91
  "mongoose-gridfs": "1.3.0",
92
92
  "multer": "1.4.5-lts.1",
93
93
  "multer-gridfs-storage": "5.0.2",
94
94
  "node-mailjet": "6.0.2",
95
95
  "nodemailer": "6.9.1",
96
- "nodemon": "2.0.20",
96
+ "nodemon": "2.0.22",
97
97
  "passport": "0.6.0",
98
98
  "passport-jwt": "4.0.1",
99
99
  "reflect-metadata": "0.1.13",
100
100
  "rfdc": "1.3.0",
101
- "rimraf": "4.1.2",
101
+ "rimraf": "4.4.1",
102
102
  "rxjs": "7.8.0"
103
103
  },
104
104
  "devDependencies": {
105
- "@nestjs/testing": "9.3.9",
105
+ "@nestjs/testing": "9.3.12",
106
106
  "@types/compression": "1.7.2",
107
107
  "@types/cookie-parser": "1.4.3",
108
- "@types/cron": "2.0.0",
109
- "@types/ejs": "3.1.1",
110
- "@types/jest": "29.4.0",
111
- "@types/lodash": "4.14.191",
108
+ "@types/cron": "2.0.1",
109
+ "@types/ejs": "3.1.2",
110
+ "@types/jest": "29.5.0",
111
+ "@types/lodash": "4.14.192",
112
112
  "@types/multer": "1.4.7",
113
- "@types/node": "18.13.0",
113
+ "@types/node": "18.15.10",
114
114
  "@types/nodemailer": "6.4.7",
115
- "@types/passport": "1.0.11",
115
+ "@types/passport": "1.0.12",
116
116
  "@types/supertest": "2.0.12",
117
- "@typescript-eslint/eslint-plugin": "5.52.0",
118
- "@typescript-eslint/parser": "5.52.0",
117
+ "@typescript-eslint/eslint-plugin": "5.57.0",
118
+ "@typescript-eslint/parser": "5.57.0",
119
119
  "coffeescript": "2.7.0",
120
- "eslint": "8.34.0",
121
- "eslint-config-prettier": "8.6.0",
120
+ "eslint": "8.36.0",
121
+ "eslint-config-prettier": "8.8.0",
122
122
  "find-file-up": "2.0.1",
123
123
  "grunt": "1.6.1",
124
124
  "grunt-bg-shell": "2.3.3",
@@ -126,17 +126,17 @@
126
126
  "grunt-contrib-watch": "1.1.0",
127
127
  "grunt-sync": "0.8.2",
128
128
  "husky": "8.0.3",
129
- "jest": "29.4.2",
129
+ "jest": "29.5.0",
130
130
  "npm-watch": "0.11.0",
131
- "pm2": "5.2.2",
132
- "prettier": "2.8.4",
131
+ "pm2": "5.3.0",
132
+ "prettier": "2.8.7",
133
133
  "pretty-quick": "3.1.3",
134
134
  "supertest": "6.3.3",
135
135
  "ts-jest": "29.0.5",
136
136
  "ts-morph": "17.0.1",
137
137
  "ts-node": "10.9.1",
138
138
  "tsconfig-paths": "4.1.2",
139
- "typescript": "4.9.5",
139
+ "typescript": "5.0.2",
140
140
  "yalc": "1.0.0-pre.53"
141
141
  },
142
142
  "overrides": {
package/src/config.env.ts CHANGED
@@ -51,7 +51,6 @@ const config: { [env: string]: IServerOptions } = {
51
51
  },
52
52
  graphQl: {
53
53
  driver: {
54
- debug: true,
55
54
  introspection: true,
56
55
  },
57
56
  maxComplexity: 20,
@@ -125,7 +124,6 @@ const config: { [env: string]: IServerOptions } = {
125
124
  },
126
125
  graphQl: {
127
126
  driver: {
128
- debug: true,
129
127
  introspection: true,
130
128
  },
131
129
  maxComplexity: 20,
@@ -199,7 +197,6 @@ const config: { [env: string]: IServerOptions } = {
199
197
  },
200
198
  graphQl: {
201
199
  driver: {
202
- debug: false,
203
200
  introspection: true,
204
201
  },
205
202
  maxComplexity: 20,
@@ -137,6 +137,7 @@ export const checkRestricted = (
137
137
  roles.includes(RoleEnum.S_EVERYONE) ||
138
138
  user?.hasRole?.(roles) ||
139
139
  (user?.id && roles.includes(RoleEnum.S_USER)) ||
140
+ (roles.includes(RoleEnum.S_SELF) && getIncludedIds(config.dbObject, user)) ||
140
141
  (roles.includes(RoleEnum.S_CREATOR) && getIncludedIds(config.dbObject?.createdBy, user))
141
142
  ) {
142
143
  valid = true;
@@ -53,4 +53,7 @@ export enum RoleEnum {
53
53
 
54
54
  // User must be the creator of the processed object(s) (see createdBy property of object(s))
55
55
  S_CREATOR = 's_creator',
56
+
57
+ // User must be herself/himself
58
+ S_SELF = 's_self',
56
59
  }
@@ -2,6 +2,7 @@ import { BadRequestException, HttpException, UnauthorizedException } from '@nest
2
2
  import { plainToInstance } from 'class-transformer';
3
3
  import { validate } from 'class-validator';
4
4
  import { ValidatorOptions } from 'class-validator/types/validation/ValidatorOptions';
5
+ import { Kind } from 'graphql/index';
5
6
  import * as _ from 'lodash';
6
7
  import * as rfdc from 'rfdc';
7
8
  import { checkRestricted } from '../decorators/restricted.decorator';
@@ -249,6 +250,8 @@ export async function check(
249
250
  (roles.includes(RoleEnum.S_USER) && user?.id) ||
250
251
  // check if the user has at least one of the required roles
251
252
  user?.hasRole?.(roles) ||
253
+ // check if the user is herself / himself
254
+ (roles.includes(RoleEnum.S_SELF) && equalIds(config.dbObject, user)) ||
252
255
  // check if the user is the creator
253
256
  (roles.includes(RoleEnum.S_CREATOR) && equalIds(config.dbObject?.createdBy, user))
254
257
  ) {
@@ -300,6 +303,25 @@ export async function check(
300
303
  return value;
301
304
  }
302
305
 
306
+ /**
307
+ * Check if input is a valid Date format and return a new Date
308
+ */
309
+ export function checkAndGetDate(input: any): Date {
310
+ // Create date from value
311
+ const date = new Date(input);
312
+
313
+ // Check value
314
+ if (date.toString() === 'Invalid Date') {
315
+ throw new Error('Invalid value for date');
316
+ }
317
+
318
+ // Check if range is valid
319
+ date.toISOString();
320
+
321
+ // Return date if everything is fine
322
+ return date;
323
+ }
324
+
303
325
  /**
304
326
  * Clone object
305
327
  * @param object Any object
@@ -379,6 +401,30 @@ export function filterProperties<T = Record<string, any>>(
379
401
  .reduce((res, key) => Object.assign(res, { [key]: obj[key] }), {});
380
402
  }
381
403
 
404
+ export function getDateFromGraphQL(input: any): Date {
405
+ // Check value
406
+ if (input.value === undefined || input.value === null) {
407
+ return input.value;
408
+ }
409
+
410
+ // Check nullable
411
+ if (!input.value) {
412
+ throw new Error('Invalid value for date');
413
+ }
414
+
415
+ // Check value type
416
+ if (input.kind !== Kind.INT && input.kind !== Kind.STRING) {
417
+ throw new Error('Invalid value type for date');
418
+ }
419
+
420
+ // Check format if value is a string
421
+ if (input.kind === Kind.STRING && isNaN(Date.parse(input.value))) {
422
+ throw new Error('Invalid ISO 8601 format for date');
423
+ }
424
+
425
+ return checkAndGetDate(input.value);
426
+ }
427
+
382
428
  /**
383
429
  * Get plain copy of object
384
430
  */
@@ -0,0 +1,9 @@
1
+ import { FieldMiddleware, MiddlewareContext, NextFn } from '@nestjs/graphql';
2
+
3
+ /**
4
+ * Field middleware to convert string to lowercase letters
5
+ */
6
+ export const toLowerCase: FieldMiddleware = async (ctx: MiddlewareContext, next: NextFn) => {
7
+ const value = await next();
8
+ return value?.toLowerCase();
9
+ };
@@ -0,0 +1,32 @@
1
+ import { CustomScalar, Scalar } from '@nestjs/graphql';
2
+ import { Kind } from 'graphql';
3
+ import { checkAndGetDate, getDateFromGraphQL } from '../helpers/input.helper';
4
+
5
+ /**
6
+ * Date-Timestamp-Scalar to convert timestamp to date and vice versa
7
+ */
8
+ @Scalar('Date', (type) => Date)
9
+ export class DateTimestampScalar implements CustomScalar<number, Date> {
10
+ description = 'Date (by Timestamp) custom scalar type';
11
+
12
+ /**
13
+ * Parse value from the client input variables
14
+ */
15
+ parseValue(value: number): Date {
16
+ return checkAndGetDate(value); // value from the client
17
+ }
18
+
19
+ /**
20
+ * Serialize value to send to the client
21
+ */
22
+ serialize(value: Date): number {
23
+ return value.getTime(); // value sent to the client
24
+ }
25
+
26
+ /**
27
+ * Parse value from the client query
28
+ */
29
+ parseLiteral(ast: any): Date {
30
+ return getDateFromGraphQL(ast);
31
+ }
32
+ }
@@ -1,5 +1,6 @@
1
1
  import { CustomScalar, Scalar } from '@nestjs/graphql';
2
2
  import { Kind } from 'graphql';
3
+ import { checkAndGetDate, getDateFromGraphQL } from '../helpers/input.helper';
3
4
 
4
5
  /**
5
6
  * Date scalar to convert string into date
@@ -12,7 +13,7 @@ export class DateScalar implements CustomScalar<string, Date> {
12
13
  * Parse value from the client input variables
13
14
  */
14
15
  parseValue(value: number): Date {
15
- return new Date(value); // value from the client
16
+ return checkAndGetDate(value); // value from the client
16
17
  }
17
18
 
18
19
  /**
@@ -26,38 +27,6 @@ export class DateScalar implements CustomScalar<string, Date> {
26
27
  * Parse value from the client query
27
28
  */
28
29
  parseLiteral(ast: any): Date {
29
- // Check value
30
- if (ast.value === undefined || ast.value === null) {
31
- return ast.value;
32
- }
33
-
34
- // Check nullable
35
- if (!ast.value) {
36
- throw new Error('Invalid value for date');
37
- }
38
-
39
- // Check value type
40
- if (ast.kind !== Kind.INT && ast.kind !== Kind.STRING) {
41
- throw new Error('Invalid value type for date');
42
- }
43
-
44
- // Check format if value is a string
45
- if (ast.kind === Kind.STRING && isNaN(Date.parse(ast.value))) {
46
- throw new Error('Invalid ISO 8601 format for date');
47
- }
48
-
49
- // Create date from value
50
- const date = new Date(ast.value);
51
-
52
- // Check value
53
- if (date.toString() === 'Invalid Date') {
54
- throw new Error('Invalid value for date');
55
- }
56
-
57
- // Check if range is valid
58
- date.toISOString();
59
-
60
- // Return date if everything is fine
61
- return date;
30
+ return getDateFromGraphQL(ast);
62
31
  }
63
32
  }
@@ -22,7 +22,7 @@ export abstract class CoreUserModel extends CorePersistenceModel {
22
22
  */
23
23
  @Field({ description: 'Email of the user', nullable: true })
24
24
  @IsEmail()
25
- @Prop({ unique: true })
25
+ @Prop({ unique: true, lowercase: true, trim: true })
26
26
  email: string = undefined;
27
27
 
28
28
  /**
@@ -4,6 +4,7 @@ import { Restricted } from '../../../common/decorators/restricted.decorator';
4
4
  import { ProcessType } from '../../../common/enums/process-type.enum';
5
5
  import { RoleEnum } from '../../../common/enums/role.enum';
6
6
  import { CoreInput } from '../../../common/inputs/core-input.input';
7
+ import { toLowerCase } from '../../../common/middlewares/to-lower-case.middleware';
7
8
 
8
9
  /**
9
10
  * User input to update a user
package/src/index.ts CHANGED
@@ -48,6 +48,7 @@ export * from './core/common/interfaces/prepare-output-options.interface';
48
48
  export * from './core/common/interfaces/resolve-selector.interface';
49
49
  export * from './core/common/interfaces/server-options.interface';
50
50
  export * from './core/common/interfaces/service-options.interface';
51
+ export * from './core/common/middlewares/to-lower-case.middleware';
51
52
  export * from './core/common/models/core-model.model';
52
53
  export * from './core/common/models/core-persistence.model';
53
54
  export * from './core/common/pipes/check-input.pipe';
@@ -56,6 +57,7 @@ export * from './core/common/plugins/complexity.plugin';
56
57
  export * from './core/common/plugins/mongoose-id.plugin';
57
58
  export * from './core/common/scalars/any.scalar';
58
59
  export * from './core/common/scalars/date.scalar';
60
+ export * from './core/common/scalars/date-timestamp.scalar';
59
61
  export * from './core/common/scalars/json.scalar';
60
62
  export * from './core/common/services/config.service';
61
63
  export * from './core/common/services/core-cron-jobs.service';
@@ -62,7 +62,7 @@ export class UserResolver {
62
62
  async getUser(@GraphQLServiceOptions() serviceOptions: ServiceOptions, @Args('id') id: string): Promise<User> {
63
63
  return await this.userService.get(id, {
64
64
  ...serviceOptions,
65
- roles: [RoleEnum.ADMIN, RoleEnum.S_CREATOR],
65
+ roles: [RoleEnum.ADMIN, RoleEnum.S_CREATOR, RoleEnum.S_SELF],
66
66
  });
67
67
  }
68
68
 
@@ -111,7 +111,7 @@ export class UserResolver {
111
111
  async deleteUser(@GraphQLServiceOptions() serviceOptions: ServiceOptions, @Args('id') id: string): Promise<User> {
112
112
  return await this.userService.delete(id, {
113
113
  ...serviceOptions,
114
- roles: [RoleEnum.ADMIN, RoleEnum.S_CREATOR],
114
+ roles: [RoleEnum.ADMIN, RoleEnum.S_CREATOR, RoleEnum.S_SELF],
115
115
  });
116
116
  }
117
117
 
@@ -138,7 +138,7 @@ export class UserResolver {
138
138
  return await this.userService.update(id, input, {
139
139
  ...serviceOptions,
140
140
  inputType: UserInput,
141
- roles: [RoleEnum.ADMIN, RoleEnum.S_CREATOR],
141
+ roles: [RoleEnum.ADMIN, RoleEnum.S_CREATOR, RoleEnum.S_SELF],
142
142
  });
143
143
  }
144
144
 
@@ -520,6 +520,8 @@ export class TestHelper {
520
520
  // Add variables as attachment or field
521
521
  mapArray.forEach((variable, i) => {
522
522
  if (variable.type === 'attachment') {
523
+ // See https://stackoverflow.com/questions/74581070/apollo-client-this-operation-has-been-blocked-as-a-potential-cross-site-request
524
+ request.set('Apollo-Require-Preflight', 'true');
523
525
  if (typeof variable.value === 'object' && variable.value.file) {
524
526
  request.attach(`${i}`, variable.value.file, variable.value.options);
525
527
  } else {