parse-server 9.7.0-alpha.9 → 9.7.0

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.
@@ -4,7 +4,6 @@ Object.defineProperty(exports, "__esModule", {
4
4
  value: true
5
5
  });
6
6
  exports.ParseGraphQLServer = void 0;
7
- var _cors = _interopRequireDefault(require("cors"));
8
7
  var _graphqlUploadExpress = _interopRequireDefault(require("graphql-upload/graphqlUploadExpress.js"));
9
8
  var _server = require("@apollo/server");
10
9
  var _express = require("@as-integrations/express5");
@@ -108,10 +107,8 @@ class ParseGraphQLServer {
108
107
  return {
109
108
  schema: await this.parseGraphQLSchema.load(),
110
109
  context: async ({
111
- req,
112
- res
110
+ req
113
111
  }) => {
114
- res.set('access-control-allow-origin', req.get('origin') || '*');
115
112
  return {
116
113
  info: req.info,
117
114
  config: req.config,
@@ -194,7 +191,7 @@ class ParseGraphQLServer {
194
191
  if (!app || !app.use) {
195
192
  (0, _requiredParameter.default)('You must provide an Express.js app instance!');
196
193
  }
197
- app.use(this.config.graphQLPath, (0, _cors.default)());
194
+ app.use(this.config.graphQLPath, (0, _middlewares.allowCrossDomain)(this.parseServer.config.appId));
198
195
  app.use(this.config.graphQLPath, _middlewares.handleParseHeaders);
199
196
  app.use(this.config.graphQLPath, _middlewares.handleParseSession);
200
197
  this.applyRequestContextMiddleware(app, this.parseServer.config);
@@ -241,4 +238,4 @@ class ParseGraphQLServer {
241
238
  }
242
239
  }
243
240
  exports.ParseGraphQLServer = ParseGraphQLServer;
244
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_cors","_interopRequireDefault","require","_graphqlUploadExpress","_server","_express","_disabled","_express2","_graphql","_middlewares","_requiredParameter","_logger","_ParseGraphQLSchema","_ParseGraphQLController","_interopRequireWildcard","_queryComplexity","e","t","WeakMap","r","n","__esModule","o","i","f","__proto__","default","has","get","set","hasOwnProperty","call","Object","defineProperty","getOwnPropertyDescriptor","hasTypeIntrospection","query","ast","parse","checkSelections","selections","selection","kind","name","value","arguments","length","selectionSet","definition","definitions","throwIntrospectionError","GraphQLError","extensions","http","status","IntrospectionControlPlugin","publicIntrospection","requestDidStart","requestContext","didResolveOperation","isMasterOrMaintenance","contextValue","auth","isMaster","isMaintenance","request","includes","ParseGraphQLServer","constructor","parseServer","config","requiredParameter","graphQLPath","parseGraphQLController","log","loggerController","defaultLogger","parseGraphQLSchema","ParseGraphQLSchema","databaseController","graphQLCustomTypeDefs","appId","_getGraphQLOptions","schema","load","context","req","res","info","error","stack","toString","_getServer","schemaRef","graphQLSchema","newSchemaRef","_schemaRefMutex","createServer","apollo","ApolloServer","csrfPrevention","requestHeaders","introspection","plugins","ApolloServerPluginCacheControlDisabled","graphQLPublicIntrospection","createComplexityValidationPlugin","requestComplexity","start","expressMiddleware","_transformMaxUploadSizeToBytes","maxUploadSize","unitMap","kb","mb","gb","Number","slice","Math","pow","toLowerCase","applyRequestContextMiddleware","api","options","requestContextMiddleware","Error","use","applyGraphQL","app","corsMiddleware","handleParseHeaders","handleParseSession","handleParseErrors","graphqlUploadExpress","maxFileSize","express","json","next","server","applyPlayground","playgroundPath","_req","setHeader","write","JSON","stringify","masterKey","end","setGraphQLConfig","graphQLConfig","updateGraphQLConfig","exports"],"sources":["../../src/GraphQL/ParseGraphQLServer.js"],"sourcesContent":["import corsMiddleware from 'cors';\nimport graphqlUploadExpress from 'graphql-upload/graphqlUploadExpress.js';\nimport { ApolloServer } from '@apollo/server';\nimport { expressMiddleware } from '@as-integrations/express5';\nimport { ApolloServerPluginCacheControlDisabled } from '@apollo/server/plugin/disabled';\nimport express from 'express';\nimport { GraphQLError, parse } from 'graphql';\nimport { handleParseErrors, handleParseHeaders, handleParseSession } from '../middlewares';\nimport requiredParameter from '../requiredParameter';\nimport defaultLogger from '../logger';\nimport { ParseGraphQLSchema } from './ParseGraphQLSchema';\nimport ParseGraphQLController, { ParseGraphQLConfig } from '../Controllers/ParseGraphQLController';\nimport { createComplexityValidationPlugin } from './helpers/queryComplexity';\n\n\nconst hasTypeIntrospection = (query) => {\n  try {\n    const ast = parse(query);\n    const checkSelections = (selections) => {\n      for (const selection of selections) {\n        if (selection.kind === 'Field' && selection.name.value === '__type') {\n          if (selection.arguments && selection.arguments.length > 0) {\n            return true;\n          }\n        }\n        if (selection.selectionSet) {\n          if (checkSelections(selection.selectionSet.selections)) {\n            return true;\n          }\n        }\n      }\n      return false;\n    };\n    for (const definition of ast.definitions) {\n      if (definition.selectionSet) {\n        if (checkSelections(definition.selectionSet.selections)) {\n          return true;\n        }\n      }\n    }\n    return false;\n  } catch {\n    return false;\n  }\n};\n\nconst throwIntrospectionError = () => {\n  throw new GraphQLError('Introspection is not allowed', {\n    extensions: {\n      http: {\n        status: 403,\n      },\n    }\n  });\n};\n\nconst IntrospectionControlPlugin = (publicIntrospection) => ({\n\n\n  requestDidStart: (requestContext) => ({\n\n    didResolveOperation: async () => {\n      // If public introspection is enabled, we allow all introspection queries\n      if (publicIntrospection) {\n        return;\n      }\n\n      const isMasterOrMaintenance = requestContext.contextValue.auth?.isMaster || requestContext.contextValue.auth?.isMaintenance\n      if (isMasterOrMaintenance) {\n        return;\n      }\n\n      const query = requestContext.request.query;\n\n\n      // Fast path: simple string check for __schema\n      // This avoids parsing the query in most cases\n      if (query?.includes('__schema')) {\n        return throwIntrospectionError();\n      }\n\n      // Smart check for __type: only parse if the string is present\n      // This avoids false positives (e.g., \"__type\" in strings or comments)\n      // while still being efficient for the common case\n      if (query?.includes('__type') && hasTypeIntrospection(query)) {\n        return throwIntrospectionError();\n      }\n    },\n\n  })\n\n});\n\nclass ParseGraphQLServer {\n  parseGraphQLController: ParseGraphQLController;\n\n  constructor(parseServer, config) {\n    this.parseServer = parseServer || requiredParameter('You must provide a parseServer instance!');\n    if (!config || !config.graphQLPath) {\n      requiredParameter('You must provide a config.graphQLPath!');\n    }\n    this.config = config;\n    this.parseGraphQLController = this.parseServer.config.parseGraphQLController;\n    this.log =\n      (this.parseServer.config && this.parseServer.config.loggerController) || defaultLogger;\n    this.parseGraphQLSchema = new ParseGraphQLSchema({\n      parseGraphQLController: this.parseGraphQLController,\n      databaseController: this.parseServer.config.databaseController,\n      log: this.log,\n      graphQLCustomTypeDefs: this.config.graphQLCustomTypeDefs,\n      appId: this.parseServer.config.appId,\n    });\n  }\n\n  async _getGraphQLOptions() {\n    try {\n      return {\n        schema: await this.parseGraphQLSchema.load(),\n        context: async ({ req, res }) => {\n          res.set('access-control-allow-origin', req.get('origin') || '*');\n          return {\n            info: req.info,\n            config: req.config,\n            auth: req.auth,\n          };\n        },\n      };\n    } catch (e) {\n      this.log.error(e.stack || (typeof e.toString === 'function' && e.toString()) || e);\n      throw e;\n    }\n  }\n\n  async _getServer() {\n    const schemaRef = this.parseGraphQLSchema.graphQLSchema;\n    const newSchemaRef = await this.parseGraphQLSchema.load();\n    if (schemaRef === newSchemaRef && this._server) {\n      return this._server;\n    }\n    // It means a parallel _getServer call is already in progress\n    if (this._schemaRefMutex === newSchemaRef) {\n      return this._server;\n    }\n    // Update the schema ref mutex to avoid parallel _getServer calls\n    this._schemaRefMutex = newSchemaRef;\n    const createServer = async () => {\n      try {\n        const { schema, context } = await this._getGraphQLOptions();\n        const apollo = new ApolloServer({\n          csrfPrevention: {\n            // See https://www.apollographql.com/docs/router/configuration/csrf/\n            // needed since we use graphql upload\n            requestHeaders: ['X-Parse-Application-Id'],\n          },\n          // We need always true introspection because apollo server have changing behavior based on the NODE_ENV variable\n          // we delegate the introspection control to the IntrospectionControlPlugin\n          introspection: true,\n          plugins: [ApolloServerPluginCacheControlDisabled(), IntrospectionControlPlugin(this.config.graphQLPublicIntrospection), createComplexityValidationPlugin(() => this.parseServer.config.requestComplexity)],\n          schema,\n        });\n        await apollo.start();\n        return expressMiddleware(apollo, {\n          context,\n        });\n      } catch (e) {\n        // Reset all mutexes and forward the error\n        this._server = null;\n        this._schemaRefMutex = null;\n        throw e;\n      }\n    }\n    // Do not await so parallel request will wait the same promise ref\n    this._server = createServer();\n    return this._server;\n  }\n\n  _transformMaxUploadSizeToBytes(maxUploadSize) {\n    const unitMap = {\n      kb: 1,\n      mb: 2,\n      gb: 3,\n    };\n\n    return (\n      Number(maxUploadSize.slice(0, -2)) *\n      Math.pow(1024, unitMap[maxUploadSize.slice(-2).toLowerCase()])\n    );\n  }\n\n  /**\n   * @static\n   * Allow developers to customize each request with inversion of control/dependency injection\n   */\n  applyRequestContextMiddleware(api, options) {\n    if (options.requestContextMiddleware) {\n      if (typeof options.requestContextMiddleware !== 'function') {\n        throw new Error('requestContextMiddleware must be a function');\n      }\n      api.use(this.config.graphQLPath, options.requestContextMiddleware);\n    }\n  }\n\n  applyGraphQL(app) {\n    if (!app || !app.use) {\n      requiredParameter('You must provide an Express.js app instance!');\n    }\n    app.use(this.config.graphQLPath, corsMiddleware());\n    app.use(this.config.graphQLPath, handleParseHeaders);\n    app.use(this.config.graphQLPath, handleParseSession);\n    this.applyRequestContextMiddleware(app, this.parseServer.config);\n    app.use(this.config.graphQLPath, handleParseErrors);\n    app.use(\n      this.config.graphQLPath,\n      graphqlUploadExpress({\n        maxFileSize: this._transformMaxUploadSizeToBytes(\n          this.parseServer.config.maxUploadSize || '20mb'\n        ),\n      })\n    );\n    app.use(this.config.graphQLPath, express.json(), async (req, res, next) => {\n      const server = await this._getServer();\n      return server(req, res, next);\n    });\n  }\n\n  applyPlayground(app) {\n    if (!app || !app.get) {\n      requiredParameter('You must provide an Express.js app instance!');\n    }\n\n    app.get(\n      this.config.playgroundPath ||\n      requiredParameter('You must provide a config.playgroundPath to applyPlayground!'),\n      (_req, res) => {\n        res.setHeader('Content-Type', 'text/html');\n        res.write(\n          `<div id=\"sandbox\" style=\"position:absolute;top:0;right:0;bottom:0;left:0\"></div>\n          <script src=\"https://embeddable-sandbox.cdn.apollographql.com/_latest/embeddable-sandbox.umd.production.min.js\"></script>\n          <script>\n           new window.EmbeddedSandbox({\n             target: \"#sandbox\",\n             endpointIsEditable: false,\n             initialEndpoint: ${JSON.stringify(this.config.graphQLPath)},\n             handleRequest: (endpointUrl, options) => {\n              return fetch(endpointUrl, {\n                ...options,\n                headers: {\n                    ...options.headers,\n                    'X-Parse-Application-Id': ${JSON.stringify(this.parseServer.config.appId)},\n                    'X-Parse-Master-Key': ${JSON.stringify(this.parseServer.config.masterKey)},\n                },\n              })\n            },\n           });\n           // advanced options: https://www.apollographql.com/docs/studio/explorer/sandbox#embedding-sandbox\n          </script>`\n        );\n        res.end();\n      }\n    );\n  }\n\n  setGraphQLConfig(graphQLConfig: ParseGraphQLConfig): Promise {\n    return this.parseGraphQLController.updateGraphQLConfig(graphQLConfig);\n  }\n}\n\nexport { ParseGraphQLServer };\n"],"mappings":"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","ignoreList":[]}
241
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_graphqlUploadExpress","_interopRequireDefault","require","_server","_express","_disabled","_express2","_graphql","_middlewares","_requiredParameter","_logger","_ParseGraphQLSchema","_ParseGraphQLController","_interopRequireWildcard","_queryComplexity","e","t","WeakMap","r","n","__esModule","o","i","f","__proto__","default","has","get","set","hasOwnProperty","call","Object","defineProperty","getOwnPropertyDescriptor","hasTypeIntrospection","query","ast","parse","checkSelections","selections","selection","kind","name","value","arguments","length","selectionSet","definition","definitions","throwIntrospectionError","GraphQLError","extensions","http","status","IntrospectionControlPlugin","publicIntrospection","requestDidStart","requestContext","didResolveOperation","isMasterOrMaintenance","contextValue","auth","isMaster","isMaintenance","request","includes","ParseGraphQLServer","constructor","parseServer","config","requiredParameter","graphQLPath","parseGraphQLController","log","loggerController","defaultLogger","parseGraphQLSchema","ParseGraphQLSchema","databaseController","graphQLCustomTypeDefs","appId","_getGraphQLOptions","schema","load","context","req","info","error","stack","toString","_getServer","schemaRef","graphQLSchema","newSchemaRef","_schemaRefMutex","createServer","apollo","ApolloServer","csrfPrevention","requestHeaders","introspection","plugins","ApolloServerPluginCacheControlDisabled","graphQLPublicIntrospection","createComplexityValidationPlugin","requestComplexity","start","expressMiddleware","_transformMaxUploadSizeToBytes","maxUploadSize","unitMap","kb","mb","gb","Number","slice","Math","pow","toLowerCase","applyRequestContextMiddleware","api","options","requestContextMiddleware","Error","use","applyGraphQL","app","allowCrossDomain","handleParseHeaders","handleParseSession","handleParseErrors","graphqlUploadExpress","maxFileSize","express","json","res","next","server","applyPlayground","playgroundPath","_req","setHeader","write","JSON","stringify","masterKey","end","setGraphQLConfig","graphQLConfig","updateGraphQLConfig","exports"],"sources":["../../src/GraphQL/ParseGraphQLServer.js"],"sourcesContent":["import graphqlUploadExpress from 'graphql-upload/graphqlUploadExpress.js';\nimport { ApolloServer } from '@apollo/server';\nimport { expressMiddleware } from '@as-integrations/express5';\nimport { ApolloServerPluginCacheControlDisabled } from '@apollo/server/plugin/disabled';\nimport express from 'express';\nimport { GraphQLError, parse } from 'graphql';\nimport { allowCrossDomain, handleParseErrors, handleParseHeaders, handleParseSession } from '../middlewares';\nimport requiredParameter from '../requiredParameter';\nimport defaultLogger from '../logger';\nimport { ParseGraphQLSchema } from './ParseGraphQLSchema';\nimport ParseGraphQLController, { ParseGraphQLConfig } from '../Controllers/ParseGraphQLController';\nimport { createComplexityValidationPlugin } from './helpers/queryComplexity';\n\n\nconst hasTypeIntrospection = (query) => {\n  try {\n    const ast = parse(query);\n    const checkSelections = (selections) => {\n      for (const selection of selections) {\n        if (selection.kind === 'Field' && selection.name.value === '__type') {\n          if (selection.arguments && selection.arguments.length > 0) {\n            return true;\n          }\n        }\n        if (selection.selectionSet) {\n          if (checkSelections(selection.selectionSet.selections)) {\n            return true;\n          }\n        }\n      }\n      return false;\n    };\n    for (const definition of ast.definitions) {\n      if (definition.selectionSet) {\n        if (checkSelections(definition.selectionSet.selections)) {\n          return true;\n        }\n      }\n    }\n    return false;\n  } catch {\n    return false;\n  }\n};\n\nconst throwIntrospectionError = () => {\n  throw new GraphQLError('Introspection is not allowed', {\n    extensions: {\n      http: {\n        status: 403,\n      },\n    }\n  });\n};\n\nconst IntrospectionControlPlugin = (publicIntrospection) => ({\n\n\n  requestDidStart: (requestContext) => ({\n\n    didResolveOperation: async () => {\n      // If public introspection is enabled, we allow all introspection queries\n      if (publicIntrospection) {\n        return;\n      }\n\n      const isMasterOrMaintenance = requestContext.contextValue.auth?.isMaster || requestContext.contextValue.auth?.isMaintenance\n      if (isMasterOrMaintenance) {\n        return;\n      }\n\n      const query = requestContext.request.query;\n\n\n      // Fast path: simple string check for __schema\n      // This avoids parsing the query in most cases\n      if (query?.includes('__schema')) {\n        return throwIntrospectionError();\n      }\n\n      // Smart check for __type: only parse if the string is present\n      // This avoids false positives (e.g., \"__type\" in strings or comments)\n      // while still being efficient for the common case\n      if (query?.includes('__type') && hasTypeIntrospection(query)) {\n        return throwIntrospectionError();\n      }\n    },\n\n  })\n\n});\n\nclass ParseGraphQLServer {\n  parseGraphQLController: ParseGraphQLController;\n\n  constructor(parseServer, config) {\n    this.parseServer = parseServer || requiredParameter('You must provide a parseServer instance!');\n    if (!config || !config.graphQLPath) {\n      requiredParameter('You must provide a config.graphQLPath!');\n    }\n    this.config = config;\n    this.parseGraphQLController = this.parseServer.config.parseGraphQLController;\n    this.log =\n      (this.parseServer.config && this.parseServer.config.loggerController) || defaultLogger;\n    this.parseGraphQLSchema = new ParseGraphQLSchema({\n      parseGraphQLController: this.parseGraphQLController,\n      databaseController: this.parseServer.config.databaseController,\n      log: this.log,\n      graphQLCustomTypeDefs: this.config.graphQLCustomTypeDefs,\n      appId: this.parseServer.config.appId,\n    });\n  }\n\n  async _getGraphQLOptions() {\n    try {\n      return {\n        schema: await this.parseGraphQLSchema.load(),\n        context: async ({ req }) => {\n          return {\n            info: req.info,\n            config: req.config,\n            auth: req.auth,\n          };\n        },\n      };\n    } catch (e) {\n      this.log.error(e.stack || (typeof e.toString === 'function' && e.toString()) || e);\n      throw e;\n    }\n  }\n\n  async _getServer() {\n    const schemaRef = this.parseGraphQLSchema.graphQLSchema;\n    const newSchemaRef = await this.parseGraphQLSchema.load();\n    if (schemaRef === newSchemaRef && this._server) {\n      return this._server;\n    }\n    // It means a parallel _getServer call is already in progress\n    if (this._schemaRefMutex === newSchemaRef) {\n      return this._server;\n    }\n    // Update the schema ref mutex to avoid parallel _getServer calls\n    this._schemaRefMutex = newSchemaRef;\n    const createServer = async () => {\n      try {\n        const { schema, context } = await this._getGraphQLOptions();\n        const apollo = new ApolloServer({\n          csrfPrevention: {\n            // See https://www.apollographql.com/docs/router/configuration/csrf/\n            // needed since we use graphql upload\n            requestHeaders: ['X-Parse-Application-Id'],\n          },\n          // We need always true introspection because apollo server have changing behavior based on the NODE_ENV variable\n          // we delegate the introspection control to the IntrospectionControlPlugin\n          introspection: true,\n          plugins: [ApolloServerPluginCacheControlDisabled(), IntrospectionControlPlugin(this.config.graphQLPublicIntrospection), createComplexityValidationPlugin(() => this.parseServer.config.requestComplexity)],\n          schema,\n        });\n        await apollo.start();\n        return expressMiddleware(apollo, {\n          context,\n        });\n      } catch (e) {\n        // Reset all mutexes and forward the error\n        this._server = null;\n        this._schemaRefMutex = null;\n        throw e;\n      }\n    }\n    // Do not await so parallel request will wait the same promise ref\n    this._server = createServer();\n    return this._server;\n  }\n\n  _transformMaxUploadSizeToBytes(maxUploadSize) {\n    const unitMap = {\n      kb: 1,\n      mb: 2,\n      gb: 3,\n    };\n\n    return (\n      Number(maxUploadSize.slice(0, -2)) *\n      Math.pow(1024, unitMap[maxUploadSize.slice(-2).toLowerCase()])\n    );\n  }\n\n  /**\n   * @static\n   * Allow developers to customize each request with inversion of control/dependency injection\n   */\n  applyRequestContextMiddleware(api, options) {\n    if (options.requestContextMiddleware) {\n      if (typeof options.requestContextMiddleware !== 'function') {\n        throw new Error('requestContextMiddleware must be a function');\n      }\n      api.use(this.config.graphQLPath, options.requestContextMiddleware);\n    }\n  }\n\n  applyGraphQL(app) {\n    if (!app || !app.use) {\n      requiredParameter('You must provide an Express.js app instance!');\n    }\n    app.use(this.config.graphQLPath, allowCrossDomain(this.parseServer.config.appId));\n    app.use(this.config.graphQLPath, handleParseHeaders);\n    app.use(this.config.graphQLPath, handleParseSession);\n    this.applyRequestContextMiddleware(app, this.parseServer.config);\n    app.use(this.config.graphQLPath, handleParseErrors);\n    app.use(\n      this.config.graphQLPath,\n      graphqlUploadExpress({\n        maxFileSize: this._transformMaxUploadSizeToBytes(\n          this.parseServer.config.maxUploadSize || '20mb'\n        ),\n      })\n    );\n    app.use(this.config.graphQLPath, express.json(), async (req, res, next) => {\n      const server = await this._getServer();\n      return server(req, res, next);\n    });\n  }\n\n  applyPlayground(app) {\n    if (!app || !app.get) {\n      requiredParameter('You must provide an Express.js app instance!');\n    }\n\n    app.get(\n      this.config.playgroundPath ||\n      requiredParameter('You must provide a config.playgroundPath to applyPlayground!'),\n      (_req, res) => {\n        res.setHeader('Content-Type', 'text/html');\n        res.write(\n          `<div id=\"sandbox\" style=\"position:absolute;top:0;right:0;bottom:0;left:0\"></div>\n          <script src=\"https://embeddable-sandbox.cdn.apollographql.com/_latest/embeddable-sandbox.umd.production.min.js\"></script>\n          <script>\n           new window.EmbeddedSandbox({\n             target: \"#sandbox\",\n             endpointIsEditable: false,\n             initialEndpoint: ${JSON.stringify(this.config.graphQLPath)},\n             handleRequest: (endpointUrl, options) => {\n              return fetch(endpointUrl, {\n                ...options,\n                headers: {\n                    ...options.headers,\n                    'X-Parse-Application-Id': ${JSON.stringify(this.parseServer.config.appId)},\n                    'X-Parse-Master-Key': ${JSON.stringify(this.parseServer.config.masterKey)},\n                },\n              })\n            },\n           });\n           // advanced options: https://www.apollographql.com/docs/studio/explorer/sandbox#embedding-sandbox\n          </script>`\n        );\n        res.end();\n      }\n    );\n  }\n\n  setGraphQLConfig(graphQLConfig: ParseGraphQLConfig): Promise {\n    return this.parseGraphQLController.updateGraphQLConfig(graphQLConfig);\n  }\n}\n\nexport { ParseGraphQLServer };\n"],"mappings":";;;;;;AAAA,IAAAA,qBAAA,GAAAC,sBAAA,CAAAC,OAAA;AACA,IAAAC,OAAA,GAAAD,OAAA;AACA,IAAAE,QAAA,GAAAF,OAAA;AACA,IAAAG,SAAA,GAAAH,OAAA;AACA,IAAAI,SAAA,GAAAL,sBAAA,CAAAC,OAAA;AACA,IAAAK,QAAA,GAAAL,OAAA;AACA,IAAAM,YAAA,GAAAN,OAAA;AACA,IAAAO,kBAAA,GAAAR,sBAAA,CAAAC,OAAA;AACA,IAAAQ,OAAA,GAAAT,sBAAA,CAAAC,OAAA;AACA,IAAAS,mBAAA,GAAAT,OAAA;AACA,IAAAU,uBAAA,GAAAC,uBAAA,CAAAX,OAAA;AACA,IAAAY,gBAAA,GAAAZ,OAAA;AAA6E,SAAAW,wBAAAE,CAAA,EAAAC,CAAA,6BAAAC,OAAA,MAAAC,CAAA,OAAAD,OAAA,IAAAE,CAAA,OAAAF,OAAA,YAAAJ,uBAAA,YAAAA,CAAAE,CAAA,EAAAC,CAAA,SAAAA,CAAA,IAAAD,CAAA,IAAAA,CAAA,CAAAK,UAAA,SAAAL,CAAA,MAAAM,CAAA,EAAAC,CAAA,EAAAC,CAAA,KAAAC,SAAA,QAAAC,OAAA,EAAAV,CAAA,iBAAAA,CAAA,uBAAAA,CAAA,yBAAAA,CAAA,SAAAQ,CAAA,MAAAF,CAAA,GAAAL,CAAA,GAAAG,CAAA,GAAAD,CAAA,QAAAG,CAAA,CAAAK,GAAA,CAAAX,CAAA,UAAAM,CAAA,CAAAM,GAAA,CAAAZ,CAAA,GAAAM,CAAA,CAAAO,GAAA,CAAAb,CAAA,EAAAQ,CAAA,gBAAAP,CAAA,IAAAD,CAAA,gBAAAC,CAAA,OAAAa,cAAA,CAAAC,IAAA,CAAAf,CAAA,EAAAC,CAAA,OAAAM,CAAA,IAAAD,CAAA,GAAAU,MAAA,CAAAC,cAAA,KAAAD,MAAA,CAAAE,wBAAA,CAAAlB,CAAA,EAAAC,CAAA,OAAAM,CAAA,CAAAK,GAAA,IAAAL,CAAA,CAAAM,GAAA,IAAAP,CAAA,CAAAE,CAAA,EAAAP,CAAA,EAAAM,CAAA,IAAAC,CAAA,CAAAP,CAAA,IAAAD,CAAA,CAAAC,CAAA,WAAAO,CAAA,KAAAR,CAAA,EAAAC,CAAA;AAAA,SAAAf,uBAAAc,CAAA,WAAAA,CAAA,IAAAA,CAAA,CAAAK,UAAA,GAAAL,CAAA,KAAAU,OAAA,EAAAV,CAAA;AAG7E,MAAMmB,oBAAoB,GAAIC,KAAK,IAAK;EACtC,IAAI;IACF,MAAMC,GAAG,GAAG,IAAAC,cAAK,EAACF,KAAK,CAAC;IACxB,MAAMG,eAAe,GAAIC,UAAU,IAAK;MACtC,KAAK,MAAMC,SAAS,IAAID,UAAU,EAAE;QAClC,IAAIC,SAAS,CAACC,IAAI,KAAK,OAAO,IAAID,SAAS,CAACE,IAAI,CAACC,KAAK,KAAK,QAAQ,EAAE;UACnE,IAAIH,SAAS,CAACI,SAAS,IAAIJ,SAAS,CAACI,SAAS,CAACC,MAAM,GAAG,CAAC,EAAE;YACzD,OAAO,IAAI;UACb;QACF;QACA,IAAIL,SAAS,CAACM,YAAY,EAAE;UAC1B,IAAIR,eAAe,CAACE,SAAS,CAACM,YAAY,CAACP,UAAU,CAAC,EAAE;YACtD,OAAO,IAAI;UACb;QACF;MACF;MACA,OAAO,KAAK;IACd,CAAC;IACD,KAAK,MAAMQ,UAAU,IAAIX,GAAG,CAACY,WAAW,EAAE;MACxC,IAAID,UAAU,CAACD,YAAY,EAAE;QAC3B,IAAIR,eAAe,CAACS,UAAU,CAACD,YAAY,CAACP,UAAU,CAAC,EAAE;UACvD,OAAO,IAAI;QACb;MACF;IACF;IACA,OAAO,KAAK;EACd,CAAC,CAAC,MAAM;IACN,OAAO,KAAK;EACd;AACF,CAAC;AAED,MAAMU,uBAAuB,GAAGA,CAAA,KAAM;EACpC,MAAM,IAAIC,qBAAY,CAAC,8BAA8B,EAAE;IACrDC,UAAU,EAAE;MACVC,IAAI,EAAE;QACJC,MAAM,EAAE;MACV;IACF;EACF,CAAC,CAAC;AACJ,CAAC;AAED,MAAMC,0BAA0B,GAAIC,mBAAmB,KAAM;EAG3DC,eAAe,EAAGC,cAAc,KAAM;IAEpCC,mBAAmB,EAAE,MAAAA,CAAA,KAAY;MAC/B;MACA,IAAIH,mBAAmB,EAAE;QACvB;MACF;MAEA,MAAMI,qBAAqB,GAAGF,cAAc,CAACG,YAAY,CAACC,IAAI,EAAEC,QAAQ,IAAIL,cAAc,CAACG,YAAY,CAACC,IAAI,EAAEE,aAAa;MAC3H,IAAIJ,qBAAqB,EAAE;QACzB;MACF;MAEA,MAAMxB,KAAK,GAAGsB,cAAc,CAACO,OAAO,CAAC7B,KAAK;;MAG1C;MACA;MACA,IAAIA,KAAK,EAAE8B,QAAQ,CAAC,UAAU,CAAC,EAAE;QAC/B,OAAOhB,uBAAuB,CAAC,CAAC;MAClC;;MAEA;MACA;MACA;MACA,IAAId,KAAK,EAAE8B,QAAQ,CAAC,QAAQ,CAAC,IAAI/B,oBAAoB,CAACC,KAAK,CAAC,EAAE;QAC5D,OAAOc,uBAAuB,CAAC,CAAC;MAClC;IACF;EAEF,CAAC;AAEH,CAAC,CAAC;AAEF,MAAMiB,kBAAkB,CAAC;EAGvBC,WAAWA,CAACC,WAAW,EAAEC,MAAM,EAAE;IAC/B,IAAI,CAACD,WAAW,GAAGA,WAAW,IAAI,IAAAE,0BAAiB,EAAC,0CAA0C,CAAC;IAC/F,IAAI,CAACD,MAAM,IAAI,CAACA,MAAM,CAACE,WAAW,EAAE;MAClC,IAAAD,0BAAiB,EAAC,wCAAwC,CAAC;IAC7D;IACA,IAAI,CAACD,MAAM,GAAGA,MAAM;IACpB,IAAI,CAACG,sBAAsB,GAAG,IAAI,CAACJ,WAAW,CAACC,MAAM,CAACG,sBAAsB;IAC5E,IAAI,CAACC,GAAG,GACL,IAAI,CAACL,WAAW,CAACC,MAAM,IAAI,IAAI,CAACD,WAAW,CAACC,MAAM,CAACK,gBAAgB,IAAKC,eAAa;IACxF,IAAI,CAACC,kBAAkB,GAAG,IAAIC,sCAAkB,CAAC;MAC/CL,sBAAsB,EAAE,IAAI,CAACA,sBAAsB;MACnDM,kBAAkB,EAAE,IAAI,CAACV,WAAW,CAACC,MAAM,CAACS,kBAAkB;MAC9DL,GAAG,EAAE,IAAI,CAACA,GAAG;MACbM,qBAAqB,EAAE,IAAI,CAACV,MAAM,CAACU,qBAAqB;MACxDC,KAAK,EAAE,IAAI,CAACZ,WAAW,CAACC,MAAM,CAACW;IACjC,CAAC,CAAC;EACJ;EAEA,MAAMC,kBAAkBA,CAAA,EAAG;IACzB,IAAI;MACF,OAAO;QACLC,MAAM,EAAE,MAAM,IAAI,CAACN,kBAAkB,CAACO,IAAI,CAAC,CAAC;QAC5CC,OAAO,EAAE,MAAAA,CAAO;UAAEC;QAAI,CAAC,KAAK;UAC1B,OAAO;YACLC,IAAI,EAAED,GAAG,CAACC,IAAI;YACdjB,MAAM,EAAEgB,GAAG,CAAChB,MAAM;YAClBR,IAAI,EAAEwB,GAAG,CAACxB;UACZ,CAAC;QACH;MACF,CAAC;IACH,CAAC,CAAC,OAAO9C,CAAC,EAAE;MACV,IAAI,CAAC0D,GAAG,CAACc,KAAK,CAACxE,CAAC,CAACyE,KAAK,IAAK,OAAOzE,CAAC,CAAC0E,QAAQ,KAAK,UAAU,IAAI1E,CAAC,CAAC0E,QAAQ,CAAC,CAAE,IAAI1E,CAAC,CAAC;MAClF,MAAMA,CAAC;IACT;EACF;EAEA,MAAM2E,UAAUA,CAAA,EAAG;IACjB,MAAMC,SAAS,GAAG,IAAI,CAACf,kBAAkB,CAACgB,aAAa;IACvD,MAAMC,YAAY,GAAG,MAAM,IAAI,CAACjB,kBAAkB,CAACO,IAAI,CAAC,CAAC;IACzD,IAAIQ,SAAS,KAAKE,YAAY,IAAI,IAAI,CAAC1F,OAAO,EAAE;MAC9C,OAAO,IAAI,CAACA,OAAO;IACrB;IACA;IACA,IAAI,IAAI,CAAC2F,eAAe,KAAKD,YAAY,EAAE;MACzC,OAAO,IAAI,CAAC1F,OAAO;IACrB;IACA;IACA,IAAI,CAAC2F,eAAe,GAAGD,YAAY;IACnC,MAAME,YAAY,GAAG,MAAAA,CAAA,KAAY;MAC/B,IAAI;QACF,MAAM;UAAEb,MAAM;UAAEE;QAAQ,CAAC,GAAG,MAAM,IAAI,CAACH,kBAAkB,CAAC,CAAC;QAC3D,MAAMe,MAAM,GAAG,IAAIC,oBAAY,CAAC;UAC9BC,cAAc,EAAE;YACd;YACA;YACAC,cAAc,EAAE,CAAC,wBAAwB;UAC3C,CAAC;UACD;UACA;UACAC,aAAa,EAAE,IAAI;UACnBC,OAAO,EAAE,CAAC,IAAAC,gDAAsC,EAAC,CAAC,EAAEhD,0BAA0B,CAAC,IAAI,CAACe,MAAM,CAACkC,0BAA0B,CAAC,EAAE,IAAAC,iDAAgC,EAAC,MAAM,IAAI,CAACpC,WAAW,CAACC,MAAM,CAACoC,iBAAiB,CAAC,CAAC;UAC1MvB;QACF,CAAC,CAAC;QACF,MAAMc,MAAM,CAACU,KAAK,CAAC,CAAC;QACpB,OAAO,IAAAC,0BAAiB,EAACX,MAAM,EAAE;UAC/BZ;QACF,CAAC,CAAC;MACJ,CAAC,CAAC,OAAOrE,CAAC,EAAE;QACV;QACA,IAAI,CAACZ,OAAO,GAAG,IAAI;QACnB,IAAI,CAAC2F,eAAe,GAAG,IAAI;QAC3B,MAAM/E,CAAC;MACT;IACF,CAAC;IACD;IACA,IAAI,CAACZ,OAAO,GAAG4F,YAAY,CAAC,CAAC;IAC7B,OAAO,IAAI,CAAC5F,OAAO;EACrB;EAEAyG,8BAA8BA,CAACC,aAAa,EAAE;IAC5C,MAAMC,OAAO,GAAG;MACdC,EAAE,EAAE,CAAC;MACLC,EAAE,EAAE,CAAC;MACLC,EAAE,EAAE;IACN,CAAC;IAED,OACEC,MAAM,CAACL,aAAa,CAACM,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,GAClCC,IAAI,CAACC,GAAG,CAAC,IAAI,EAAEP,OAAO,CAACD,aAAa,CAACM,KAAK,CAAC,CAAC,CAAC,CAAC,CAACG,WAAW,CAAC,CAAC,CAAC,CAAC;EAElE;;EAEA;AACF;AACA;AACA;EACEC,6BAA6BA,CAACC,GAAG,EAAEC,OAAO,EAAE;IAC1C,IAAIA,OAAO,CAACC,wBAAwB,EAAE;MACpC,IAAI,OAAOD,OAAO,CAACC,wBAAwB,KAAK,UAAU,EAAE;QAC1D,MAAM,IAAIC,KAAK,CAAC,6CAA6C,CAAC;MAChE;MACAH,GAAG,CAACI,GAAG,CAAC,IAAI,CAACvD,MAAM,CAACE,WAAW,EAAEkD,OAAO,CAACC,wBAAwB,CAAC;IACpE;EACF;EAEAG,YAAYA,CAACC,GAAG,EAAE;IAChB,IAAI,CAACA,GAAG,IAAI,CAACA,GAAG,CAACF,GAAG,EAAE;MACpB,IAAAtD,0BAAiB,EAAC,8CAA8C,CAAC;IACnE;IACAwD,GAAG,CAACF,GAAG,CAAC,IAAI,CAACvD,MAAM,CAACE,WAAW,EAAE,IAAAwD,6BAAgB,EAAC,IAAI,CAAC3D,WAAW,CAACC,MAAM,CAACW,KAAK,CAAC,CAAC;IACjF8C,GAAG,CAACF,GAAG,CAAC,IAAI,CAACvD,MAAM,CAACE,WAAW,EAAEyD,+BAAkB,CAAC;IACpDF,GAAG,CAACF,GAAG,CAAC,IAAI,CAACvD,MAAM,CAACE,WAAW,EAAE0D,+BAAkB,CAAC;IACpD,IAAI,CAACV,6BAA6B,CAACO,GAAG,EAAE,IAAI,CAAC1D,WAAW,CAACC,MAAM,CAAC;IAChEyD,GAAG,CAACF,GAAG,CAAC,IAAI,CAACvD,MAAM,CAACE,WAAW,EAAE2D,8BAAiB,CAAC;IACnDJ,GAAG,CAACF,GAAG,CACL,IAAI,CAACvD,MAAM,CAACE,WAAW,EACvB,IAAA4D,6BAAoB,EAAC;MACnBC,WAAW,EAAE,IAAI,CAACxB,8BAA8B,CAC9C,IAAI,CAACxC,WAAW,CAACC,MAAM,CAACwC,aAAa,IAAI,MAC3C;IACF,CAAC,CACH,CAAC;IACDiB,GAAG,CAACF,GAAG,CAAC,IAAI,CAACvD,MAAM,CAACE,WAAW,EAAE8D,iBAAO,CAACC,IAAI,CAAC,CAAC,EAAE,OAAOjD,GAAG,EAAEkD,GAAG,EAAEC,IAAI,KAAK;MACzE,MAAMC,MAAM,GAAG,MAAM,IAAI,CAAC/C,UAAU,CAAC,CAAC;MACtC,OAAO+C,MAAM,CAACpD,GAAG,EAAEkD,GAAG,EAAEC,IAAI,CAAC;IAC/B,CAAC,CAAC;EACJ;EAEAE,eAAeA,CAACZ,GAAG,EAAE;IACnB,IAAI,CAACA,GAAG,IAAI,CAACA,GAAG,CAACnG,GAAG,EAAE;MACpB,IAAA2C,0BAAiB,EAAC,8CAA8C,CAAC;IACnE;IAEAwD,GAAG,CAACnG,GAAG,CACL,IAAI,CAAC0C,MAAM,CAACsE,cAAc,IAC1B,IAAArE,0BAAiB,EAAC,8DAA8D,CAAC,EACjF,CAACsE,IAAI,EAAEL,GAAG,KAAK;MACbA,GAAG,CAACM,SAAS,CAAC,cAAc,EAAE,WAAW,CAAC;MAC1CN,GAAG,CAACO,KAAK,CACP;AACV;AACA;AACA;AACA;AACA;AACA,gCAAgCC,IAAI,CAACC,SAAS,CAAC,IAAI,CAAC3E,MAAM,CAACE,WAAW,CAAC;AACvE;AACA;AACA;AACA;AACA;AACA,gDAAgDwE,IAAI,CAACC,SAAS,CAAC,IAAI,CAAC5E,WAAW,CAACC,MAAM,CAACW,KAAK,CAAC;AAC7F,4CAA4C+D,IAAI,CAACC,SAAS,CAAC,IAAI,CAAC5E,WAAW,CAACC,MAAM,CAAC4E,SAAS,CAAC;AAC7F;AACA;AACA;AACA;AACA;AACA,oBACQ,CAAC;MACDV,GAAG,CAACW,GAAG,CAAC,CAAC;IACX,CACF,CAAC;EACH;EAEAC,gBAAgBA,CAACC,aAAiC,EAAW;IAC3D,OAAO,IAAI,CAAC5E,sBAAsB,CAAC6E,mBAAmB,CAACD,aAAa,CAAC;EACvE;AACF;AAACE,OAAA,CAAApF,kBAAA,GAAAA,kBAAA","ignoreList":[]}
@@ -8,13 +8,21 @@ exports.createComplexityValidationPlugin = createComplexityValidationPlugin;
8
8
  var _graphql = require("graphql");
9
9
  var _logger = _interopRequireDefault(require("../../logger"));
10
10
  function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
11
- function calculateQueryComplexity(operation, fragments) {
11
+ function calculateQueryComplexity(operation, fragments, limits = {}) {
12
12
  let maxDepth = 0;
13
13
  let totalFields = 0;
14
+ const fragmentCache = new Map();
15
+ const {
16
+ maxDepth: allowedMaxDepth,
17
+ maxFields: allowedMaxFields
18
+ } = limits;
14
19
  function visitSelectionSet(selectionSet, depth, visitedFragments) {
15
20
  if (!selectionSet) {
16
21
  return;
17
22
  }
23
+ if (allowedMaxFields !== undefined && allowedMaxFields !== -1 && totalFields > allowedMaxFields || allowedMaxDepth !== undefined && allowedMaxDepth !== -1 && maxDepth > allowedMaxDepth) {
24
+ return;
25
+ }
18
26
  for (const selection of selectionSet.selections) {
19
27
  if (selection.kind === 'Field') {
20
28
  totalFields++;
@@ -29,14 +37,36 @@ function calculateQueryComplexity(operation, fragments) {
29
37
  visitSelectionSet(selection.selectionSet, depth, visitedFragments);
30
38
  } else if (selection.kind === 'FragmentSpread') {
31
39
  const name = selection.name.value;
40
+ if (fragmentCache.has(name)) {
41
+ const cached = fragmentCache.get(name);
42
+ totalFields += cached.fields;
43
+ const adjustedDepth = depth + cached.maxDepthDelta;
44
+ if (adjustedDepth > maxDepth) {
45
+ maxDepth = adjustedDepth;
46
+ }
47
+ continue;
48
+ }
32
49
  if (visitedFragments.has(name)) {
33
50
  continue;
34
51
  }
35
52
  const fragment = fragments[name];
36
53
  if (fragment) {
37
- const branchVisited = new Set(visitedFragments);
38
- branchVisited.add(name);
39
- visitSelectionSet(fragment.selectionSet, depth, branchVisited);
54
+ if (allowedMaxFields !== undefined && allowedMaxFields !== -1 && totalFields > allowedMaxFields || allowedMaxDepth !== undefined && allowedMaxDepth !== -1 && maxDepth > allowedMaxDepth) {
55
+ continue;
56
+ }
57
+ visitedFragments.add(name);
58
+ const savedFields = totalFields;
59
+ const savedMaxDepth = maxDepth;
60
+ maxDepth = depth;
61
+ visitSelectionSet(fragment.selectionSet, depth, visitedFragments);
62
+ const fieldsContribution = totalFields - savedFields;
63
+ const maxDepthDelta = maxDepth - depth;
64
+ fragmentCache.set(name, {
65
+ fields: fieldsContribution,
66
+ maxDepthDelta
67
+ });
68
+ maxDepth = Math.max(savedMaxDepth, maxDepth);
69
+ visitedFragments.delete(name);
40
70
  }
41
71
  }
42
72
  }
@@ -75,7 +105,10 @@ function createComplexityValidationPlugin(getConfig) {
75
105
  const {
76
106
  depth,
77
107
  fields
78
- } = calculateQueryComplexity(requestContext.operation, fragments);
108
+ } = calculateQueryComplexity(requestContext.operation, fragments, {
109
+ maxDepth: graphQLDepth,
110
+ maxFields: graphQLFields
111
+ });
79
112
  if (graphQLDepth !== -1 && depth > graphQLDepth) {
80
113
  const message = `GraphQL query depth of ${depth} exceeds maximum allowed depth of ${graphQLDepth}`;
81
114
  _logger.default.warn(message);
@@ -102,4 +135,4 @@ function createComplexityValidationPlugin(getConfig) {
102
135
  })
103
136
  };
104
137
  }
105
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_graphql","require","_logger","_interopRequireDefault","e","__esModule","default","calculateQueryComplexity","operation","fragments","maxDepth","totalFields","visitSelectionSet","selectionSet","depth","visitedFragments","selection","selections","kind","newDepth","name","value","has","fragment","branchVisited","Set","add","fields","createComplexityValidationPlugin","getConfig","requestDidStart","requestContext","didResolveOperation","auth","contextValue","isMaster","isMaintenance","config","graphQLDepth","graphQLFields","definition","document","definitions","message","logger","warn","GraphQLError","extensions","http","status"],"sources":["../../../src/GraphQL/helpers/queryComplexity.js"],"sourcesContent":["import { GraphQLError } from 'graphql';\nimport logger from '../../logger';\n\nfunction calculateQueryComplexity(operation, fragments) {\n  let maxDepth = 0;\n  let totalFields = 0;\n\n  function visitSelectionSet(selectionSet, depth, visitedFragments) {\n    if (!selectionSet) {\n      return;\n    }\n    for (const selection of selectionSet.selections) {\n      if (selection.kind === 'Field') {\n        totalFields++;\n        const newDepth = depth + 1;\n        if (newDepth > maxDepth) {\n          maxDepth = newDepth;\n        }\n        if (selection.selectionSet) {\n          visitSelectionSet(selection.selectionSet, newDepth, visitedFragments);\n        }\n      } else if (selection.kind === 'InlineFragment') {\n        visitSelectionSet(selection.selectionSet, depth, visitedFragments);\n      } else if (selection.kind === 'FragmentSpread') {\n        const name = selection.name.value;\n        if (visitedFragments.has(name)) {\n          continue;\n        }\n        const fragment = fragments[name];\n        if (fragment) {\n          const branchVisited = new Set(visitedFragments);\n          branchVisited.add(name);\n          visitSelectionSet(fragment.selectionSet, depth, branchVisited);\n        }\n      }\n    }\n  }\n\n  visitSelectionSet(operation.selectionSet, 0, new Set());\n\n  return { depth: maxDepth, fields: totalFields };\n}\n\nfunction createComplexityValidationPlugin(getConfig) {\n  return {\n    requestDidStart: (requestContext) => ({\n      didResolveOperation: async () => {\n        const auth = requestContext.contextValue?.auth;\n        if (auth?.isMaster || auth?.isMaintenance) {\n          return;\n        }\n\n        const config = getConfig();\n        if (!config) {\n          return;\n        }\n\n        const { graphQLDepth, graphQLFields } = config;\n        if (graphQLDepth === -1 && graphQLFields === -1) {\n          return;\n        }\n\n        const fragments = {};\n        for (const definition of requestContext.document.definitions) {\n          if (definition.kind === 'FragmentDefinition') {\n            fragments[definition.name.value] = definition;\n          }\n        }\n\n        const { depth, fields } = calculateQueryComplexity(\n          requestContext.operation,\n          fragments\n        );\n\n        if (graphQLDepth !== -1 && depth > graphQLDepth) {\n          const message = `GraphQL query depth of ${depth} exceeds maximum allowed depth of ${graphQLDepth}`;\n          logger.warn(message);\n          throw new GraphQLError(message, {\n            extensions: {\n              http: { status: 400 },\n            },\n          });\n        }\n\n        if (graphQLFields !== -1 && fields > graphQLFields) {\n          const message = `Number of GraphQL fields (${fields}) exceeds maximum allowed (${graphQLFields})`;\n          logger.warn(message);\n          throw new GraphQLError(message, {\n            extensions: {\n              http: { status: 400 },\n            },\n          });\n        }\n      },\n    }),\n  };\n}\n\nexport { calculateQueryComplexity, createComplexityValidationPlugin };\n"],"mappings":";;;;;;;AAAA,IAAAA,QAAA,GAAAC,OAAA;AACA,IAAAC,OAAA,GAAAC,sBAAA,CAAAF,OAAA;AAAkC,SAAAE,uBAAAC,CAAA,WAAAA,CAAA,IAAAA,CAAA,CAAAC,UAAA,GAAAD,CAAA,KAAAE,OAAA,EAAAF,CAAA;AAElC,SAASG,wBAAwBA,CAACC,SAAS,EAAEC,SAAS,EAAE;EACtD,IAAIC,QAAQ,GAAG,CAAC;EAChB,IAAIC,WAAW,GAAG,CAAC;EAEnB,SAASC,iBAAiBA,CAACC,YAAY,EAAEC,KAAK,EAAEC,gBAAgB,EAAE;IAChE,IAAI,CAACF,YAAY,EAAE;MACjB;IACF;IACA,KAAK,MAAMG,SAAS,IAAIH,YAAY,CAACI,UAAU,EAAE;MAC/C,IAAID,SAAS,CAACE,IAAI,KAAK,OAAO,EAAE;QAC9BP,WAAW,EAAE;QACb,MAAMQ,QAAQ,GAAGL,KAAK,GAAG,CAAC;QAC1B,IAAIK,QAAQ,GAAGT,QAAQ,EAAE;UACvBA,QAAQ,GAAGS,QAAQ;QACrB;QACA,IAAIH,SAAS,CAACH,YAAY,EAAE;UAC1BD,iBAAiB,CAACI,SAAS,CAACH,YAAY,EAAEM,QAAQ,EAAEJ,gBAAgB,CAAC;QACvE;MACF,CAAC,MAAM,IAAIC,SAAS,CAACE,IAAI,KAAK,gBAAgB,EAAE;QAC9CN,iBAAiB,CAACI,SAAS,CAACH,YAAY,EAAEC,KAAK,EAAEC,gBAAgB,CAAC;MACpE,CAAC,MAAM,IAAIC,SAAS,CAACE,IAAI,KAAK,gBAAgB,EAAE;QAC9C,MAAME,IAAI,GAAGJ,SAAS,CAACI,IAAI,CAACC,KAAK;QACjC,IAAIN,gBAAgB,CAACO,GAAG,CAACF,IAAI,CAAC,EAAE;UAC9B;QACF;QACA,MAAMG,QAAQ,GAAGd,SAAS,CAACW,IAAI,CAAC;QAChC,IAAIG,QAAQ,EAAE;UACZ,MAAMC,aAAa,GAAG,IAAIC,GAAG,CAACV,gBAAgB,CAAC;UAC/CS,aAAa,CAACE,GAAG,CAACN,IAAI,CAAC;UACvBR,iBAAiB,CAACW,QAAQ,CAACV,YAAY,EAAEC,KAAK,EAAEU,aAAa,CAAC;QAChE;MACF;IACF;EACF;EAEAZ,iBAAiB,CAACJ,SAAS,CAACK,YAAY,EAAE,CAAC,EAAE,IAAIY,GAAG,CAAC,CAAC,CAAC;EAEvD,OAAO;IAAEX,KAAK,EAAEJ,QAAQ;IAAEiB,MAAM,EAAEhB;EAAY,CAAC;AACjD;AAEA,SAASiB,gCAAgCA,CAACC,SAAS,EAAE;EACnD,OAAO;IACLC,eAAe,EAAGC,cAAc,KAAM;MACpCC,mBAAmB,EAAE,MAAAA,CAAA,KAAY;QAC/B,MAAMC,IAAI,GAAGF,cAAc,CAACG,YAAY,EAAED,IAAI;QAC9C,IAAIA,IAAI,EAAEE,QAAQ,IAAIF,IAAI,EAAEG,aAAa,EAAE;UACzC;QACF;QAEA,MAAMC,MAAM,GAAGR,SAAS,CAAC,CAAC;QAC1B,IAAI,CAACQ,MAAM,EAAE;UACX;QACF;QAEA,MAAM;UAAEC,YAAY;UAAEC;QAAc,CAAC,GAAGF,MAAM;QAC9C,IAAIC,YAAY,KAAK,CAAC,CAAC,IAAIC,aAAa,KAAK,CAAC,CAAC,EAAE;UAC/C;QACF;QAEA,MAAM9B,SAAS,GAAG,CAAC,CAAC;QACpB,KAAK,MAAM+B,UAAU,IAAIT,cAAc,CAACU,QAAQ,CAACC,WAAW,EAAE;UAC5D,IAAIF,UAAU,CAACtB,IAAI,KAAK,oBAAoB,EAAE;YAC5CT,SAAS,CAAC+B,UAAU,CAACpB,IAAI,CAACC,KAAK,CAAC,GAAGmB,UAAU;UAC/C;QACF;QAEA,MAAM;UAAE1B,KAAK;UAAEa;QAAO,CAAC,GAAGpB,wBAAwB,CAChDwB,cAAc,CAACvB,SAAS,EACxBC,SACF,CAAC;QAED,IAAI6B,YAAY,KAAK,CAAC,CAAC,IAAIxB,KAAK,GAAGwB,YAAY,EAAE;UAC/C,MAAMK,OAAO,GAAG,0BAA0B7B,KAAK,qCAAqCwB,YAAY,EAAE;UAClGM,eAAM,CAACC,IAAI,CAACF,OAAO,CAAC;UACpB,MAAM,IAAIG,qBAAY,CAACH,OAAO,EAAE;YAC9BI,UAAU,EAAE;cACVC,IAAI,EAAE;gBAAEC,MAAM,EAAE;cAAI;YACtB;UACF,CAAC,CAAC;QACJ;QAEA,IAAIV,aAAa,KAAK,CAAC,CAAC,IAAIZ,MAAM,GAAGY,aAAa,EAAE;UAClD,MAAMI,OAAO,GAAG,6BAA6BhB,MAAM,8BAA8BY,aAAa,GAAG;UACjGK,eAAM,CAACC,IAAI,CAACF,OAAO,CAAC;UACpB,MAAM,IAAIG,qBAAY,CAACH,OAAO,EAAE;YAC9BI,UAAU,EAAE;cACVC,IAAI,EAAE;gBAAEC,MAAM,EAAE;cAAI;YACtB;UACF,CAAC,CAAC;QACJ;MACF;IACF,CAAC;EACH,CAAC;AACH","ignoreList":[]}
138
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_graphql","require","_logger","_interopRequireDefault","e","__esModule","default","calculateQueryComplexity","operation","fragments","limits","maxDepth","totalFields","fragmentCache","Map","allowedMaxDepth","maxFields","allowedMaxFields","visitSelectionSet","selectionSet","depth","visitedFragments","undefined","selection","selections","kind","newDepth","name","value","has","cached","get","fields","adjustedDepth","maxDepthDelta","fragment","add","savedFields","savedMaxDepth","fieldsContribution","set","Math","max","delete","Set","createComplexityValidationPlugin","getConfig","requestDidStart","requestContext","didResolveOperation","auth","contextValue","isMaster","isMaintenance","config","graphQLDepth","graphQLFields","definition","document","definitions","message","logger","warn","GraphQLError","extensions","http","status"],"sources":["../../../src/GraphQL/helpers/queryComplexity.js"],"sourcesContent":["import { GraphQLError } from 'graphql';\nimport logger from '../../logger';\n\nfunction calculateQueryComplexity(operation, fragments, limits = {}) {\n  let maxDepth = 0;\n  let totalFields = 0;\n  const fragmentCache = new Map();\n  const { maxDepth: allowedMaxDepth, maxFields: allowedMaxFields } = limits;\n\n  function visitSelectionSet(selectionSet, depth, visitedFragments) {\n    if (!selectionSet) {\n      return;\n    }\n    if (\n      (allowedMaxFields !== undefined && allowedMaxFields !== -1 && totalFields > allowedMaxFields) ||\n      (allowedMaxDepth !== undefined && allowedMaxDepth !== -1 && maxDepth > allowedMaxDepth)\n    ) {\n      return;\n    }\n    for (const selection of selectionSet.selections) {\n      if (selection.kind === 'Field') {\n        totalFields++;\n        const newDepth = depth + 1;\n        if (newDepth > maxDepth) {\n          maxDepth = newDepth;\n        }\n        if (selection.selectionSet) {\n          visitSelectionSet(selection.selectionSet, newDepth, visitedFragments);\n        }\n      } else if (selection.kind === 'InlineFragment') {\n        visitSelectionSet(selection.selectionSet, depth, visitedFragments);\n      } else if (selection.kind === 'FragmentSpread') {\n        const name = selection.name.value;\n        if (fragmentCache.has(name)) {\n          const cached = fragmentCache.get(name);\n          totalFields += cached.fields;\n          const adjustedDepth = depth + cached.maxDepthDelta;\n          if (adjustedDepth > maxDepth) {\n            maxDepth = adjustedDepth;\n          }\n          continue;\n        }\n        if (visitedFragments.has(name)) {\n          continue;\n        }\n        const fragment = fragments[name];\n        if (fragment) {\n          if (\n            (allowedMaxFields !== undefined && allowedMaxFields !== -1 && totalFields > allowedMaxFields) ||\n            (allowedMaxDepth !== undefined && allowedMaxDepth !== -1 && maxDepth > allowedMaxDepth)\n          ) {\n            continue;\n          }\n          visitedFragments.add(name);\n          const savedFields = totalFields;\n          const savedMaxDepth = maxDepth;\n          maxDepth = depth;\n          visitSelectionSet(fragment.selectionSet, depth, visitedFragments);\n          const fieldsContribution = totalFields - savedFields;\n          const maxDepthDelta = maxDepth - depth;\n          fragmentCache.set(name, { fields: fieldsContribution, maxDepthDelta });\n          maxDepth = Math.max(savedMaxDepth, maxDepth);\n          visitedFragments.delete(name);\n        }\n      }\n    }\n  }\n\n  visitSelectionSet(operation.selectionSet, 0, new Set());\n\n  return { depth: maxDepth, fields: totalFields };\n}\n\nfunction createComplexityValidationPlugin(getConfig) {\n  return {\n    requestDidStart: (requestContext) => ({\n      didResolveOperation: async () => {\n        const auth = requestContext.contextValue?.auth;\n        if (auth?.isMaster || auth?.isMaintenance) {\n          return;\n        }\n\n        const config = getConfig();\n        if (!config) {\n          return;\n        }\n\n        const { graphQLDepth, graphQLFields } = config;\n        if (graphQLDepth === -1 && graphQLFields === -1) {\n          return;\n        }\n\n        const fragments = {};\n        for (const definition of requestContext.document.definitions) {\n          if (definition.kind === 'FragmentDefinition') {\n            fragments[definition.name.value] = definition;\n          }\n        }\n\n        const { depth, fields } = calculateQueryComplexity(\n          requestContext.operation,\n          fragments,\n          { maxDepth: graphQLDepth, maxFields: graphQLFields }\n        );\n\n        if (graphQLDepth !== -1 && depth > graphQLDepth) {\n          const message = `GraphQL query depth of ${depth} exceeds maximum allowed depth of ${graphQLDepth}`;\n          logger.warn(message);\n          throw new GraphQLError(message, {\n            extensions: {\n              http: { status: 400 },\n            },\n          });\n        }\n\n        if (graphQLFields !== -1 && fields > graphQLFields) {\n          const message = `Number of GraphQL fields (${fields}) exceeds maximum allowed (${graphQLFields})`;\n          logger.warn(message);\n          throw new GraphQLError(message, {\n            extensions: {\n              http: { status: 400 },\n            },\n          });\n        }\n      },\n    }),\n  };\n}\n\nexport { calculateQueryComplexity, createComplexityValidationPlugin };\n"],"mappings":"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","ignoreList":[]}