parse-server 9.5.2-alpha.1 → 9.5.2-alpha.2
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.
- package/lib/Config.js +28 -2
- package/lib/GraphQL/ParseGraphQLServer.js +3 -2
- package/lib/GraphQL/helpers/queryComplexity.js +105 -0
- package/lib/Options/Definitions.js +40 -1
- package/lib/Options/docs.js +11 -1
- package/lib/Options/index.js +1 -1
- package/lib/RestQuery.js +63 -5
- package/lib/Security/CheckGroups/CheckGroupServerConfig.js +15 -1
- package/package.json +1 -1
|
@@ -0,0 +1,105 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
Object.defineProperty(exports, "__esModule", {
|
|
4
|
+
value: true
|
|
5
|
+
});
|
|
6
|
+
exports.calculateQueryComplexity = calculateQueryComplexity;
|
|
7
|
+
exports.createComplexityValidationPlugin = createComplexityValidationPlugin;
|
|
8
|
+
var _graphql = require("graphql");
|
|
9
|
+
var _logger = _interopRequireDefault(require("../../logger"));
|
|
10
|
+
function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
|
|
11
|
+
function calculateQueryComplexity(operation, fragments) {
|
|
12
|
+
let maxDepth = 0;
|
|
13
|
+
let totalFields = 0;
|
|
14
|
+
function visitSelectionSet(selectionSet, depth, visitedFragments) {
|
|
15
|
+
if (!selectionSet) {
|
|
16
|
+
return;
|
|
17
|
+
}
|
|
18
|
+
for (const selection of selectionSet.selections) {
|
|
19
|
+
if (selection.kind === 'Field') {
|
|
20
|
+
totalFields++;
|
|
21
|
+
const newDepth = depth + 1;
|
|
22
|
+
if (newDepth > maxDepth) {
|
|
23
|
+
maxDepth = newDepth;
|
|
24
|
+
}
|
|
25
|
+
if (selection.selectionSet) {
|
|
26
|
+
visitSelectionSet(selection.selectionSet, newDepth, visitedFragments);
|
|
27
|
+
}
|
|
28
|
+
} else if (selection.kind === 'InlineFragment') {
|
|
29
|
+
visitSelectionSet(selection.selectionSet, depth, visitedFragments);
|
|
30
|
+
} else if (selection.kind === 'FragmentSpread') {
|
|
31
|
+
const name = selection.name.value;
|
|
32
|
+
if (visitedFragments.has(name)) {
|
|
33
|
+
continue;
|
|
34
|
+
}
|
|
35
|
+
const fragment = fragments[name];
|
|
36
|
+
if (fragment) {
|
|
37
|
+
const branchVisited = new Set(visitedFragments);
|
|
38
|
+
branchVisited.add(name);
|
|
39
|
+
visitSelectionSet(fragment.selectionSet, depth, branchVisited);
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
visitSelectionSet(operation.selectionSet, 0, new Set());
|
|
45
|
+
return {
|
|
46
|
+
depth: maxDepth,
|
|
47
|
+
fields: totalFields
|
|
48
|
+
};
|
|
49
|
+
}
|
|
50
|
+
function createComplexityValidationPlugin(getConfig) {
|
|
51
|
+
return {
|
|
52
|
+
requestDidStart: requestContext => ({
|
|
53
|
+
didResolveOperation: async () => {
|
|
54
|
+
const auth = requestContext.contextValue?.auth;
|
|
55
|
+
if (auth?.isMaster || auth?.isMaintenance) {
|
|
56
|
+
return;
|
|
57
|
+
}
|
|
58
|
+
const config = getConfig();
|
|
59
|
+
if (!config) {
|
|
60
|
+
return;
|
|
61
|
+
}
|
|
62
|
+
const {
|
|
63
|
+
graphQLDepth,
|
|
64
|
+
graphQLFields
|
|
65
|
+
} = config;
|
|
66
|
+
if (graphQLDepth === -1 && graphQLFields === -1) {
|
|
67
|
+
return;
|
|
68
|
+
}
|
|
69
|
+
const fragments = {};
|
|
70
|
+
for (const definition of requestContext.document.definitions) {
|
|
71
|
+
if (definition.kind === 'FragmentDefinition') {
|
|
72
|
+
fragments[definition.name.value] = definition;
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
const {
|
|
76
|
+
depth,
|
|
77
|
+
fields
|
|
78
|
+
} = calculateQueryComplexity(requestContext.operation, fragments);
|
|
79
|
+
if (graphQLDepth !== -1 && depth > graphQLDepth) {
|
|
80
|
+
const message = `GraphQL query depth of ${depth} exceeds maximum allowed depth of ${graphQLDepth}`;
|
|
81
|
+
_logger.default.warn(message);
|
|
82
|
+
throw new _graphql.GraphQLError(message, {
|
|
83
|
+
extensions: {
|
|
84
|
+
http: {
|
|
85
|
+
status: 400
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
});
|
|
89
|
+
}
|
|
90
|
+
if (graphQLFields !== -1 && fields > graphQLFields) {
|
|
91
|
+
const message = `Number of GraphQL fields (${fields}) exceeds maximum allowed (${graphQLFields})`;
|
|
92
|
+
_logger.default.warn(message);
|
|
93
|
+
throw new _graphql.GraphQLError(message, {
|
|
94
|
+
extensions: {
|
|
95
|
+
http: {
|
|
96
|
+
status: 400
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
});
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
})
|
|
103
|
+
};
|
|
104
|
+
}
|
|
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":"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","ignoreList":[]}
|