parse-server 6.0.0-alpha.3 → 6.0.0-alpha.31
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/LICENSE +167 -25
- package/NOTICE +10 -0
- package/README.md +58 -34
- package/lib/AccountLockout.js +11 -26
- package/lib/Adapters/AdapterLoader.js +8 -14
- package/lib/Adapters/Analytics/AnalyticsAdapter.js +2 -8
- package/lib/Adapters/Auth/AuthAdapter.js +7 -16
- package/lib/Adapters/Auth/OAuth1Client.js +32 -57
- package/lib/Adapters/Auth/apple.js +6 -22
- package/lib/Adapters/Auth/facebook.js +7 -37
- package/lib/Adapters/Auth/gcenter.js +8 -37
- package/lib/Adapters/Auth/github.js +7 -10
- package/lib/Adapters/Auth/google.js +11 -34
- package/lib/Adapters/Auth/gpgames.js +5 -8
- package/lib/Adapters/Auth/httpsRequest.js +1 -7
- package/lib/Adapters/Auth/index.js +20 -65
- package/lib/Adapters/Auth/instagram.js +5 -9
- package/lib/Adapters/Auth/janraincapture.js +8 -12
- package/lib/Adapters/Auth/janrainengage.js +7 -11
- package/lib/Adapters/Auth/keycloak.js +5 -19
- package/lib/Adapters/Auth/ldap.js +1 -15
- package/lib/Adapters/Auth/line.js +7 -10
- package/lib/Adapters/Auth/linkedin.js +7 -12
- package/lib/Adapters/Auth/meetup.js +7 -10
- package/lib/Adapters/Auth/microsoft.js +7 -10
- package/lib/Adapters/Auth/oauth2.js +6 -18
- package/lib/Adapters/Auth/phantauth.js +8 -10
- package/lib/Adapters/Auth/qq.js +7 -13
- package/lib/Adapters/Auth/spotify.js +7 -14
- package/lib/Adapters/Auth/twitter.js +5 -15
- package/lib/Adapters/Auth/vkontakte.js +9 -15
- package/lib/Adapters/Auth/wechat.js +7 -10
- package/lib/Adapters/Auth/weibo.js +7 -11
- package/lib/Adapters/Cache/CacheAdapter.js +4 -12
- package/lib/Adapters/Cache/InMemoryCache.js +5 -19
- package/lib/Adapters/Cache/InMemoryCacheAdapter.js +1 -11
- package/lib/Adapters/Cache/LRUCache.js +1 -11
- package/lib/Adapters/Cache/NullCacheAdapter.js +1 -8
- package/lib/Adapters/Cache/RedisCacheAdapter.js +46 -87
- package/lib/Adapters/Cache/SchemaCache.js +1 -6
- package/lib/Adapters/Email/MailAdapter.js +2 -7
- package/lib/Adapters/Files/FilesAdapter.js +7 -21
- package/lib/Adapters/Files/GridFSBucketAdapter.js +6 -44
- package/lib/Adapters/Files/GridStoreAdapter.js +1 -1
- package/lib/Adapters/Logger/LoggerAdapter.js +2 -11
- package/lib/Adapters/Logger/WinstonLogger.js +3 -30
- package/lib/Adapters/Logger/WinstonLoggerAdapter.js +5 -16
- package/lib/Adapters/MessageQueue/EventEmitterMQ.js +3 -20
- package/lib/Adapters/PubSub/EventEmitterPubSub.js +1 -16
- package/lib/Adapters/PubSub/PubSubAdapter.js +2 -9
- package/lib/Adapters/PubSub/RedisPubSub.js +13 -10
- package/lib/Adapters/Push/PushAdapter.js +2 -8
- package/lib/Adapters/Storage/Mongo/MongoCollection.js +12 -37
- package/lib/Adapters/Storage/Mongo/MongoSchemaCollection.js +26 -79
- package/lib/Adapters/Storage/Mongo/MongoStorageAdapter.js +78 -209
- package/lib/Adapters/Storage/Mongo/MongoTransform.js +82 -371
- package/lib/Adapters/Storage/Postgres/PostgresClient.js +1 -13
- package/lib/Adapters/Storage/Postgres/PostgresConfigParser.js +1 -20
- package/lib/Adapters/Storage/Postgres/PostgresStorageAdapter.js +119 -446
- package/lib/Adapters/Storage/Postgres/sql/index.js +4 -7
- package/lib/Adapters/Storage/StorageAdapter.js +1 -1
- package/lib/Adapters/WebSocketServer/WSAdapter.js +3 -12
- package/lib/Adapters/WebSocketServer/WSSAdapter.js +7 -12
- package/lib/Auth.js +68 -121
- package/lib/ClientSDK.js +3 -11
- package/lib/Config.js +77 -115
- package/lib/Controllers/AdaptableController.js +6 -18
- package/lib/Controllers/AnalyticsController.js +1 -9
- package/lib/Controllers/CacheController.js +3 -23
- package/lib/Controllers/DatabaseController.js +171 -364
- package/lib/Controllers/FilesController.js +5 -34
- package/lib/Controllers/HooksController.js +1 -51
- package/lib/Controllers/LiveQueryController.js +4 -23
- package/lib/Controllers/LoggerController.js +15 -54
- package/lib/Controllers/ParseGraphQLController.js +49 -104
- package/lib/Controllers/PushController.js +20 -59
- package/lib/Controllers/SchemaController.js +162 -348
- package/lib/Controllers/UserController.js +17 -78
- package/lib/Controllers/index.js +19 -68
- package/lib/Controllers/types.js +1 -1
- package/lib/Deprecator/Deprecations.js +1 -8
- package/lib/Deprecator/Deprecator.js +9 -18
- package/lib/GraphQL/ParseGraphQLSchema.js +16 -100
- package/lib/GraphQL/ParseGraphQLServer.js +2 -29
- package/lib/GraphQL/helpers/objectsMutations.js +2 -12
- package/lib/GraphQL/helpers/objectsQueries.js +18 -76
- package/lib/GraphQL/loaders/defaultGraphQLMutations.js +1 -9
- package/lib/GraphQL/loaders/defaultGraphQLQueries.js +1 -8
- package/lib/GraphQL/loaders/defaultGraphQLTypes.js +9 -115
- package/lib/GraphQL/loaders/defaultRelaySchema.js +6 -18
- package/lib/GraphQL/loaders/filesMutations.js +2 -19
- package/lib/GraphQL/loaders/functionsMutations.js +6 -17
- package/lib/GraphQL/loaders/parseClassMutations.js +6 -44
- package/lib/GraphQL/loaders/parseClassQueries.js +1 -26
- package/lib/GraphQL/loaders/parseClassTypes.js +10 -64
- package/lib/GraphQL/loaders/schemaDirectives.js +1 -17
- package/lib/GraphQL/loaders/schemaMutations.js +1 -20
- package/lib/GraphQL/loaders/schemaQueries.js +1 -14
- package/lib/GraphQL/loaders/schemaTypes.js +2 -6
- package/lib/GraphQL/loaders/usersMutations.js +6 -28
- package/lib/GraphQL/loaders/usersQueries.js +4 -26
- package/lib/GraphQL/parseGraphQLUtils.js +6 -19
- package/lib/GraphQL/transformers/className.js +1 -4
- package/lib/GraphQL/transformers/constraintType.js +1 -20
- package/lib/GraphQL/transformers/inputType.js +1 -20
- package/lib/GraphQL/transformers/mutation.js +6 -51
- package/lib/GraphQL/transformers/outputType.js +1 -20
- package/lib/GraphQL/transformers/query.js +6 -42
- package/lib/GraphQL/transformers/schemaFields.js +7 -34
- package/lib/KeyPromiseQueue.js +1 -12
- package/lib/LiveQuery/Client.js +1 -25
- package/lib/LiveQuery/Id.js +1 -7
- package/lib/LiveQuery/ParseCloudCodePublisher.js +13 -19
- package/lib/LiveQuery/ParseLiveQueryServer.js +111 -307
- package/lib/LiveQuery/ParsePubSub.js +1 -12
- package/lib/LiveQuery/ParseWebSocketServer.js +4 -26
- package/lib/LiveQuery/QueryTools.js +14 -116
- package/lib/LiveQuery/RequestSchema.js +1 -1
- package/lib/LiveQuery/SessionTokenCache.js +1 -17
- package/lib/LiveQuery/Subscription.js +4 -18
- package/lib/LiveQuery/equalObjects.js +2 -14
- package/lib/Options/Definitions.js +90 -10
- package/lib/Options/docs.js +25 -3
- package/lib/Options/index.js +4 -12
- package/lib/Options/parsers.js +1 -18
- package/lib/Page.js +1 -9
- package/lib/ParseMessageQueue.js +1 -10
- package/lib/ParseServer.js +178 -202
- package/lib/ParseServerRESTController.js +6 -33
- package/lib/PromiseRouter.js +16 -50
- package/lib/Push/PushQueue.js +3 -15
- package/lib/Push/PushWorker.js +7 -32
- package/lib/Push/utils.js +9 -38
- package/lib/RestQuery.js +105 -242
- package/lib/RestWrite.js +224 -389
- package/lib/Routers/AggregateRouter.js +14 -51
- package/lib/Routers/AnalyticsRouter.js +2 -8
- package/lib/Routers/AudiencesRouter.js +1 -15
- package/lib/Routers/ClassesRouter.js +3 -53
- package/lib/Routers/CloudCodeRouter.js +1 -19
- package/lib/Routers/FeaturesRouter.js +1 -10
- package/lib/Routers/FilesRouter.js +29 -76
- package/lib/Routers/FunctionsRouter.js +5 -28
- package/lib/Routers/GlobalConfigRouter.js +4 -18
- package/lib/Routers/GraphQLRouter.js +1 -14
- package/lib/Routers/HooksRouter.js +1 -29
- package/lib/Routers/IAPValidationRouter.js +6 -29
- package/lib/Routers/InstallationsRouter.js +2 -12
- package/lib/Routers/LogsRouter.js +4 -16
- package/lib/Routers/PagesRouter.js +69 -129
- package/lib/Routers/PublicAPIRouter.js +3 -62
- package/lib/Routers/PurgeRouter.js +1 -15
- package/lib/Routers/PushRouter.js +2 -18
- package/lib/Routers/RolesRouter.js +1 -7
- package/lib/Routers/SchemasRouter.js +4 -34
- package/lib/Routers/SecurityRouter.js +1 -12
- package/lib/Routers/SessionsRouter.js +3 -19
- package/lib/Routers/UsersRouter.js +59 -156
- package/lib/SchemaMigrations/DefinedSchemas.js +56 -115
- package/lib/SchemaMigrations/Migrations.js +2 -8
- package/lib/Security/Check.js +8 -16
- package/lib/Security/CheckGroup.js +4 -11
- package/lib/Security/CheckGroups/CheckGroupDatabase.js +8 -18
- package/lib/Security/CheckGroups/CheckGroupServerConfig.js +5 -15
- package/lib/Security/CheckGroups/CheckGroups.js +1 -4
- package/lib/Security/CheckRunner.js +22 -41
- package/lib/StatusHandler.js +12 -69
- package/lib/TestUtils.js +1 -6
- package/lib/Utils.js +27 -66
- package/lib/batch.js +17 -28
- package/lib/cache.js +1 -3
- package/lib/cli/definitions/parse-live-query-server.js +1 -3
- package/lib/cli/definitions/parse-server.js +1 -3
- package/lib/cli/parse-live-query-server.js +1 -6
- package/lib/cli/parse-server.js +11 -21
- package/lib/cli/utils/commander.js +13 -51
- package/lib/cli/utils/runner.js +1 -14
- package/lib/cloud-code/Parse.Cloud.js +71 -81
- package/lib/cryptoUtils.js +11 -19
- package/lib/defaults.js +2 -14
- package/lib/deprecated.js +1 -2
- package/lib/index.js +16 -34
- package/lib/logger.js +6 -13
- package/lib/middlewares.js +170 -151
- package/lib/password.js +6 -10
- package/lib/request.js +8 -42
- package/lib/requiredParameter.js +1 -3
- package/lib/rest.js +25 -47
- package/lib/triggers.js +54 -252
- package/lib/vendor/mongodbUrl.js +129 -310
- package/package.json +25 -22
- package/PATENTS +0 -37
package/lib/middlewares.js
CHANGED
|
@@ -3,58 +3,49 @@
|
|
|
3
3
|
Object.defineProperty(exports, "__esModule", {
|
|
4
4
|
value: true
|
|
5
5
|
});
|
|
6
|
-
exports.
|
|
6
|
+
exports.addRateLimit = exports.DEFAULT_ALLOWED_HEADERS = void 0;
|
|
7
7
|
exports.allowCrossDomain = allowCrossDomain;
|
|
8
8
|
exports.allowMethodOverride = allowMethodOverride;
|
|
9
|
-
exports.handleParseErrors = handleParseErrors;
|
|
10
9
|
exports.enforceMasterKeyAccess = enforceMasterKeyAccess;
|
|
10
|
+
exports.handleParseErrors = handleParseErrors;
|
|
11
|
+
exports.handleParseHeaders = handleParseHeaders;
|
|
12
|
+
exports.handleParseSession = void 0;
|
|
11
13
|
exports.promiseEnforceMasterKeyAccess = promiseEnforceMasterKeyAccess;
|
|
12
14
|
exports.promiseEnsureIdempotency = promiseEnsureIdempotency;
|
|
13
|
-
exports.DEFAULT_ALLOWED_HEADERS = void 0;
|
|
14
|
-
|
|
15
15
|
var _cache = _interopRequireDefault(require("./cache"));
|
|
16
|
-
|
|
17
16
|
var _node = _interopRequireDefault(require("parse/node"));
|
|
18
|
-
|
|
19
17
|
var _Auth = _interopRequireDefault(require("./Auth"));
|
|
20
|
-
|
|
21
18
|
var _Config = _interopRequireDefault(require("./Config"));
|
|
22
|
-
|
|
23
19
|
var _ClientSDK = _interopRequireDefault(require("./ClientSDK"));
|
|
24
|
-
|
|
25
20
|
var _logger = _interopRequireDefault(require("./logger"));
|
|
26
|
-
|
|
27
21
|
var _rest = _interopRequireDefault(require("./rest"));
|
|
28
|
-
|
|
29
22
|
var _MongoStorageAdapter = _interopRequireDefault(require("./Adapters/Storage/Mongo/MongoStorageAdapter"));
|
|
30
|
-
|
|
31
23
|
var _PostgresStorageAdapter = _interopRequireDefault(require("./Adapters/Storage/Postgres/PostgresStorageAdapter"));
|
|
32
|
-
|
|
24
|
+
var _expressRateLimit = _interopRequireDefault(require("express-rate-limit"));
|
|
25
|
+
var _Definitions = require("./Options/Definitions");
|
|
26
|
+
var _pathToRegexp = _interopRequireDefault(require("path-to-regexp"));
|
|
27
|
+
var _ipRangeCheck = _interopRequireDefault(require("ip-range-check"));
|
|
33
28
|
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
34
|
-
|
|
35
29
|
const DEFAULT_ALLOWED_HEADERS = 'X-Parse-Master-Key, X-Parse-REST-API-Key, X-Parse-Javascript-Key, X-Parse-Application-Id, X-Parse-Client-Version, X-Parse-Session-Token, X-Requested-With, X-Parse-Revocable-Session, X-Parse-Request-Id, Content-Type, Pragma, Cache-Control';
|
|
36
30
|
exports.DEFAULT_ALLOWED_HEADERS = DEFAULT_ALLOWED_HEADERS;
|
|
37
|
-
|
|
38
31
|
const getMountForRequest = function (req) {
|
|
39
32
|
const mountPathLength = req.originalUrl.length - req.url.length;
|
|
40
33
|
const mountPath = req.originalUrl.slice(0, mountPathLength);
|
|
41
34
|
return req.protocol + '://' + req.get('host') + mountPath;
|
|
42
|
-
};
|
|
35
|
+
};
|
|
36
|
+
|
|
37
|
+
// Checks that the request is authorized for this app and checks user
|
|
43
38
|
// auth too.
|
|
44
39
|
// The bodyparser should run before this middleware.
|
|
45
40
|
// Adds info to the request:
|
|
46
41
|
// req.config - the Config for this app
|
|
47
42
|
// req.auth - the Auth for this request
|
|
48
|
-
|
|
49
|
-
|
|
50
43
|
function handleParseHeaders(req, res, next) {
|
|
51
44
|
var mount = getMountForRequest(req);
|
|
52
45
|
let context = {};
|
|
53
|
-
|
|
54
46
|
if (req.get('X-Parse-Cloud-Context') != null) {
|
|
55
47
|
try {
|
|
56
48
|
context = JSON.parse(req.get('X-Parse-Cloud-Context'));
|
|
57
|
-
|
|
58
49
|
if (Object.prototype.toString.call(context) !== '[object Object]') {
|
|
59
50
|
throw 'Context is not an object';
|
|
60
51
|
}
|
|
@@ -62,11 +53,11 @@ function handleParseHeaders(req, res, next) {
|
|
|
62
53
|
return malformedContext(req, res);
|
|
63
54
|
}
|
|
64
55
|
}
|
|
65
|
-
|
|
66
56
|
var info = {
|
|
67
57
|
appId: req.get('X-Parse-Application-Id'),
|
|
68
58
|
sessionToken: req.get('X-Parse-Session-Token'),
|
|
69
59
|
masterKey: req.get('X-Parse-Master-Key'),
|
|
60
|
+
maintenanceKey: req.get('X-Parse-Maintenance-Key'),
|
|
70
61
|
installationId: req.get('X-Parse-Installation-Id'),
|
|
71
62
|
clientKey: req.get('X-Parse-Client-Key'),
|
|
72
63
|
javascriptKey: req.get('X-Parse-Javascript-Key'),
|
|
@@ -76,25 +67,20 @@ function handleParseHeaders(req, res, next) {
|
|
|
76
67
|
context: context
|
|
77
68
|
};
|
|
78
69
|
var basicAuth = httpAuth(req);
|
|
79
|
-
|
|
80
70
|
if (basicAuth) {
|
|
81
71
|
var basicAuthAppId = basicAuth.appId;
|
|
82
|
-
|
|
83
72
|
if (_cache.default.get(basicAuthAppId)) {
|
|
84
73
|
info.appId = basicAuthAppId;
|
|
85
74
|
info.masterKey = basicAuth.masterKey || info.masterKey;
|
|
86
75
|
info.javascriptKey = basicAuth.javascriptKey || info.javascriptKey;
|
|
87
76
|
}
|
|
88
77
|
}
|
|
89
|
-
|
|
90
78
|
if (req.body) {
|
|
91
79
|
// Unity SDK sends a _noBody key which needs to be removed.
|
|
92
80
|
// Unclear at this point if action needs to be taken.
|
|
93
81
|
delete req.body._noBody;
|
|
94
82
|
}
|
|
95
|
-
|
|
96
83
|
var fileViaJSON = false;
|
|
97
|
-
|
|
98
84
|
if (!info.appId || !_cache.default.get(info.appId)) {
|
|
99
85
|
// See if we can find the app id on the body.
|
|
100
86
|
if (req.body instanceof Buffer) {
|
|
@@ -108,48 +94,40 @@ function handleParseHeaders(req, res, next) {
|
|
|
108
94
|
} catch (e) {
|
|
109
95
|
return invalidRequest(req, res);
|
|
110
96
|
}
|
|
111
|
-
|
|
112
97
|
fileViaJSON = true;
|
|
113
98
|
}
|
|
114
|
-
|
|
115
99
|
if (req.body) {
|
|
116
100
|
delete req.body._RevocableSession;
|
|
117
101
|
}
|
|
118
|
-
|
|
119
102
|
if (req.body && req.body._ApplicationId && _cache.default.get(req.body._ApplicationId) && (!info.masterKey || _cache.default.get(req.body._ApplicationId).masterKey === info.masterKey)) {
|
|
120
103
|
info.appId = req.body._ApplicationId;
|
|
121
104
|
info.javascriptKey = req.body._JavaScriptKey || '';
|
|
122
105
|
delete req.body._ApplicationId;
|
|
123
|
-
delete req.body._JavaScriptKey;
|
|
106
|
+
delete req.body._JavaScriptKey;
|
|
107
|
+
// TODO: test that the REST API formats generated by the other
|
|
124
108
|
// SDKs are handled ok
|
|
125
|
-
|
|
126
109
|
if (req.body._ClientVersion) {
|
|
127
110
|
info.clientVersion = req.body._ClientVersion;
|
|
128
111
|
delete req.body._ClientVersion;
|
|
129
112
|
}
|
|
130
|
-
|
|
131
113
|
if (req.body._InstallationId) {
|
|
132
114
|
info.installationId = req.body._InstallationId;
|
|
133
115
|
delete req.body._InstallationId;
|
|
134
116
|
}
|
|
135
|
-
|
|
136
117
|
if (req.body._SessionToken) {
|
|
137
118
|
info.sessionToken = req.body._SessionToken;
|
|
138
119
|
delete req.body._SessionToken;
|
|
139
120
|
}
|
|
140
|
-
|
|
141
121
|
if (req.body._MasterKey) {
|
|
142
122
|
info.masterKey = req.body._MasterKey;
|
|
143
123
|
delete req.body._MasterKey;
|
|
144
124
|
}
|
|
145
|
-
|
|
146
125
|
if (req.body._context) {
|
|
147
126
|
if (req.body._context instanceof Object) {
|
|
148
127
|
info.context = req.body._context;
|
|
149
128
|
} else {
|
|
150
129
|
try {
|
|
151
130
|
info.context = JSON.parse(req.body._context);
|
|
152
|
-
|
|
153
131
|
if (Object.prototype.toString.call(info.context) !== '[object Object]') {
|
|
154
132
|
throw 'Context is not an object';
|
|
155
133
|
}
|
|
@@ -157,10 +135,8 @@ function handleParseHeaders(req, res, next) {
|
|
|
157
135
|
return malformedContext(req, res);
|
|
158
136
|
}
|
|
159
137
|
}
|
|
160
|
-
|
|
161
138
|
delete req.body._context;
|
|
162
139
|
}
|
|
163
|
-
|
|
164
140
|
if (req.body._ContentType) {
|
|
165
141
|
req.headers['content-type'] = req.body._ContentType;
|
|
166
142
|
delete req.body._ContentType;
|
|
@@ -169,47 +145,64 @@ function handleParseHeaders(req, res, next) {
|
|
|
169
145
|
return invalidRequest(req, res);
|
|
170
146
|
}
|
|
171
147
|
}
|
|
172
|
-
|
|
173
148
|
if (info.sessionToken && typeof info.sessionToken !== 'string') {
|
|
174
149
|
info.sessionToken = info.sessionToken.toString();
|
|
175
150
|
}
|
|
176
|
-
|
|
177
151
|
if (info.clientVersion) {
|
|
178
152
|
info.clientSDK = _ClientSDK.default.fromString(info.clientVersion);
|
|
179
153
|
}
|
|
180
|
-
|
|
181
154
|
if (fileViaJSON) {
|
|
182
|
-
req.fileData = req.body.fileData;
|
|
183
|
-
|
|
155
|
+
req.fileData = req.body.fileData;
|
|
156
|
+
// We need to repopulate req.body with a buffer
|
|
184
157
|
var base64 = req.body.base64;
|
|
185
158
|
req.body = Buffer.from(base64, 'base64');
|
|
186
159
|
}
|
|
187
|
-
|
|
188
160
|
const clientIp = getClientIp(req);
|
|
161
|
+
const config = _Config.default.get(info.appId, mount);
|
|
162
|
+
if (config.state && config.state !== 'ok') {
|
|
163
|
+
res.status(500);
|
|
164
|
+
res.json({
|
|
165
|
+
code: _node.default.Error.INTERNAL_SERVER_ERROR,
|
|
166
|
+
error: `Invalid server state: ${config.state}`
|
|
167
|
+
});
|
|
168
|
+
return;
|
|
169
|
+
}
|
|
189
170
|
info.app = _cache.default.get(info.appId);
|
|
190
|
-
req.config =
|
|
171
|
+
req.config = config;
|
|
191
172
|
req.config.headers = req.headers || {};
|
|
192
173
|
req.config.ip = clientIp;
|
|
193
174
|
req.info = info;
|
|
194
|
-
|
|
195
|
-
if (
|
|
196
|
-
|
|
175
|
+
const isMaintenance = req.config.maintenanceKey && info.maintenanceKey === req.config.maintenanceKey;
|
|
176
|
+
if (isMaintenance) {
|
|
177
|
+
var _req$config;
|
|
178
|
+
if ((0, _ipRangeCheck.default)(clientIp, req.config.maintenanceKeyIps || [])) {
|
|
179
|
+
req.auth = new _Auth.default.Auth({
|
|
180
|
+
config: req.config,
|
|
181
|
+
installationId: info.installationId,
|
|
182
|
+
isMaintenance: true
|
|
183
|
+
});
|
|
184
|
+
next();
|
|
185
|
+
return;
|
|
186
|
+
}
|
|
187
|
+
const log = ((_req$config = req.config) === null || _req$config === void 0 ? void 0 : _req$config.loggerController) || _logger.default;
|
|
188
|
+
log.error(`Request using maintenance key rejected as the request IP address '${clientIp}' is not set in Parse Server option 'maintenanceKeyIps'.`);
|
|
189
|
+
}
|
|
190
|
+
let isMaster = info.masterKey === req.config.masterKey;
|
|
191
|
+
if (isMaster && !(0, _ipRangeCheck.default)(clientIp, req.config.masterKeyIps || [])) {
|
|
192
|
+
var _req$config2;
|
|
193
|
+
const log = ((_req$config2 = req.config) === null || _req$config2 === void 0 ? void 0 : _req$config2.loggerController) || _logger.default;
|
|
194
|
+
log.error(`Request using master key rejected as the request IP address '${clientIp}' is not set in Parse Server option 'masterKeyIps'.`);
|
|
195
|
+
isMaster = false;
|
|
197
196
|
}
|
|
198
|
-
|
|
199
|
-
var isMaster = info.masterKey === req.config.masterKey;
|
|
200
|
-
|
|
201
197
|
if (isMaster) {
|
|
202
198
|
req.auth = new _Auth.default.Auth({
|
|
203
199
|
config: req.config,
|
|
204
200
|
installationId: info.installationId,
|
|
205
201
|
isMaster: true
|
|
206
202
|
});
|
|
207
|
-
next
|
|
208
|
-
return;
|
|
203
|
+
return handleRateLimit(req, res, next);
|
|
209
204
|
}
|
|
210
|
-
|
|
211
205
|
var isReadOnlyMaster = info.masterKey === req.config.readOnlyMasterKey;
|
|
212
|
-
|
|
213
206
|
if (typeof req.config.readOnlyMasterKey != 'undefined' && req.config.readOnlyMasterKey && isReadOnlyMaster) {
|
|
214
207
|
req.auth = new _Auth.default.Auth({
|
|
215
208
|
config: req.config,
|
|
@@ -217,12 +210,11 @@ function handleParseHeaders(req, res, next) {
|
|
|
217
210
|
isMaster: true,
|
|
218
211
|
isReadOnly: true
|
|
219
212
|
});
|
|
220
|
-
next
|
|
221
|
-
|
|
222
|
-
} // Client keys are not required in parse-server, but if any have been configured in the server, validate them
|
|
223
|
-
// to preserve original behavior.
|
|
224
|
-
|
|
213
|
+
return handleRateLimit(req, res, next);
|
|
214
|
+
}
|
|
225
215
|
|
|
216
|
+
// Client keys are not required in parse-server, but if any have been configured in the server, validate them
|
|
217
|
+
// to preserve original behavior.
|
|
226
218
|
const keys = ['clientKey', 'javascriptKey', 'dotNetKey', 'restAPIKey'];
|
|
227
219
|
const oneKeyConfigured = keys.some(function (key) {
|
|
228
220
|
return req.config[key] !== undefined;
|
|
@@ -230,15 +222,12 @@ function handleParseHeaders(req, res, next) {
|
|
|
230
222
|
const oneKeyMatches = keys.some(function (key) {
|
|
231
223
|
return req.config[key] !== undefined && info[key] === req.config[key];
|
|
232
224
|
});
|
|
233
|
-
|
|
234
225
|
if (oneKeyConfigured && !oneKeyMatches) {
|
|
235
226
|
return invalidRequest(req, res);
|
|
236
227
|
}
|
|
237
|
-
|
|
238
228
|
if (req.url == '/login') {
|
|
239
229
|
delete info.sessionToken;
|
|
240
230
|
}
|
|
241
|
-
|
|
242
231
|
if (req.userFromJWT) {
|
|
243
232
|
req.auth = new _Auth.default.Auth({
|
|
244
233
|
config: req.config,
|
|
@@ -246,89 +235,96 @@ function handleParseHeaders(req, res, next) {
|
|
|
246
235
|
isMaster: false,
|
|
247
236
|
user: req.userFromJWT
|
|
248
237
|
});
|
|
249
|
-
next
|
|
250
|
-
return;
|
|
238
|
+
return handleRateLimit(req, res, next);
|
|
251
239
|
}
|
|
252
|
-
|
|
253
240
|
if (!info.sessionToken) {
|
|
254
241
|
req.auth = new _Auth.default.Auth({
|
|
255
242
|
config: req.config,
|
|
256
243
|
installationId: info.installationId,
|
|
257
244
|
isMaster: false
|
|
258
245
|
});
|
|
259
|
-
|
|
246
|
+
}
|
|
247
|
+
handleRateLimit(req, res, next);
|
|
248
|
+
}
|
|
249
|
+
const handleRateLimit = async (req, res, next) => {
|
|
250
|
+
const rateLimits = req.config.rateLimits || [];
|
|
251
|
+
try {
|
|
252
|
+
await Promise.all(rateLimits.map(async limit => {
|
|
253
|
+
const pathExp = new RegExp(limit.path);
|
|
254
|
+
if (pathExp.test(req.url)) {
|
|
255
|
+
await limit.handler(req, res, err => {
|
|
256
|
+
if (err) {
|
|
257
|
+
if (err.code === _node.default.Error.CONNECTION_FAILED) {
|
|
258
|
+
throw err;
|
|
259
|
+
}
|
|
260
|
+
req.config.loggerController.error('An unknown error occured when attempting to apply the rate limiter: ', err);
|
|
261
|
+
}
|
|
262
|
+
});
|
|
263
|
+
}
|
|
264
|
+
}));
|
|
265
|
+
} catch (error) {
|
|
266
|
+
res.status(429);
|
|
267
|
+
res.json({
|
|
268
|
+
code: _node.default.Error.CONNECTION_FAILED,
|
|
269
|
+
error: error.message
|
|
270
|
+
});
|
|
260
271
|
return;
|
|
261
272
|
}
|
|
262
|
-
|
|
263
|
-
|
|
264
|
-
|
|
273
|
+
next();
|
|
274
|
+
};
|
|
275
|
+
const handleParseSession = async (req, res, next) => {
|
|
276
|
+
try {
|
|
277
|
+
const info = req.info;
|
|
278
|
+
if (req.auth) {
|
|
279
|
+
next();
|
|
280
|
+
return;
|
|
281
|
+
}
|
|
282
|
+
let requestAuth = null;
|
|
265
283
|
if (info.sessionToken && req.url === '/upgradeToRevocableSession' && info.sessionToken.indexOf('r:') != 0) {
|
|
266
|
-
|
|
284
|
+
requestAuth = await _Auth.default.getAuthForLegacySessionToken({
|
|
267
285
|
config: req.config,
|
|
268
286
|
installationId: info.installationId,
|
|
269
287
|
sessionToken: info.sessionToken
|
|
270
288
|
});
|
|
271
289
|
} else {
|
|
272
|
-
|
|
290
|
+
requestAuth = await _Auth.default.getAuthForSessionToken({
|
|
273
291
|
config: req.config,
|
|
274
292
|
installationId: info.installationId,
|
|
275
293
|
sessionToken: info.sessionToken
|
|
276
294
|
});
|
|
277
295
|
}
|
|
278
|
-
|
|
279
|
-
|
|
280
|
-
|
|
281
|
-
next();
|
|
282
|
-
}
|
|
283
|
-
}).catch(error => {
|
|
296
|
+
req.auth = requestAuth;
|
|
297
|
+
next();
|
|
298
|
+
} catch (error) {
|
|
284
299
|
if (error instanceof _node.default.Error) {
|
|
285
300
|
next(error);
|
|
286
301
|
return;
|
|
287
|
-
} else {
|
|
288
|
-
// TODO: Determine the correct error scenario.
|
|
289
|
-
req.config.loggerController.error('error getting auth for sessionToken', error);
|
|
290
|
-
throw new _node.default.Error(_node.default.Error.UNKNOWN_ERROR, error);
|
|
291
302
|
}
|
|
292
|
-
|
|
293
|
-
|
|
294
|
-
|
|
295
|
-
function getClientIp(req) {
|
|
296
|
-
if (req.headers['x-forwarded-for']) {
|
|
297
|
-
// try to get from x-forwared-for if it set (behind reverse proxy)
|
|
298
|
-
return req.headers['x-forwarded-for'].split(',')[0];
|
|
299
|
-
} else if (req.connection && req.connection.remoteAddress) {
|
|
300
|
-
// no proxy, try getting from connection.remoteAddress
|
|
301
|
-
return req.connection.remoteAddress;
|
|
302
|
-
} else if (req.socket) {
|
|
303
|
-
// try to get it from req.socket
|
|
304
|
-
return req.socket.remoteAddress;
|
|
305
|
-
} else if (req.connection && req.connection.socket) {
|
|
306
|
-
// try to get it form the connection.socket
|
|
307
|
-
return req.connection.socket.remoteAddress;
|
|
308
|
-
} else {
|
|
309
|
-
// if non above, fallback.
|
|
310
|
-
return req.ip;
|
|
303
|
+
// TODO: Determine the correct error scenario.
|
|
304
|
+
req.config.loggerController.error('error getting auth for sessionToken', error);
|
|
305
|
+
throw new _node.default.Error(_node.default.Error.UNKNOWN_ERROR, error);
|
|
311
306
|
}
|
|
307
|
+
};
|
|
308
|
+
exports.handleParseSession = handleParseSession;
|
|
309
|
+
function getClientIp(req) {
|
|
310
|
+
return req.ip;
|
|
312
311
|
}
|
|
313
|
-
|
|
314
312
|
function httpAuth(req) {
|
|
315
313
|
if (!(req.req || req).headers.authorization) return;
|
|
316
314
|
var header = (req.req || req).headers.authorization;
|
|
317
|
-
var appId, masterKey, javascriptKey;
|
|
315
|
+
var appId, masterKey, javascriptKey;
|
|
318
316
|
|
|
317
|
+
// parse header
|
|
319
318
|
var authPrefix = 'basic ';
|
|
320
319
|
var match = header.toLowerCase().indexOf(authPrefix);
|
|
321
|
-
|
|
322
320
|
if (match == 0) {
|
|
323
321
|
var encodedAuth = header.substring(authPrefix.length, header.length);
|
|
324
322
|
var credentials = decodeBase64(encodedAuth).split(':');
|
|
325
|
-
|
|
326
323
|
if (credentials.length == 2) {
|
|
327
324
|
appId = credentials[0];
|
|
328
325
|
var key = credentials[1];
|
|
329
326
|
var jsKeyPrefix = 'javascript-key=';
|
|
330
327
|
var matchKey = key.indexOf(jsKeyPrefix);
|
|
331
|
-
|
|
332
328
|
if (matchKey == 0) {
|
|
333
329
|
javascriptKey = key.substring(jsKeyPrefix.length, key.length);
|
|
334
330
|
} else {
|
|
@@ -336,34 +332,28 @@ function httpAuth(req) {
|
|
|
336
332
|
}
|
|
337
333
|
}
|
|
338
334
|
}
|
|
339
|
-
|
|
340
335
|
return {
|
|
341
336
|
appId: appId,
|
|
342
337
|
masterKey: masterKey,
|
|
343
338
|
javascriptKey: javascriptKey
|
|
344
339
|
};
|
|
345
340
|
}
|
|
346
|
-
|
|
347
341
|
function decodeBase64(str) {
|
|
348
342
|
return Buffer.from(str, 'base64').toString();
|
|
349
343
|
}
|
|
350
|
-
|
|
351
344
|
function allowCrossDomain(appId) {
|
|
352
345
|
return (req, res, next) => {
|
|
353
346
|
const config = _Config.default.get(appId, getMountForRequest(req));
|
|
354
|
-
|
|
355
347
|
let allowHeaders = DEFAULT_ALLOWED_HEADERS;
|
|
356
|
-
|
|
357
348
|
if (config && config.allowHeaders) {
|
|
358
349
|
allowHeaders += `, ${config.allowHeaders.join(', ')}`;
|
|
359
350
|
}
|
|
360
|
-
|
|
361
351
|
const allowOrigin = config && config.allowOrigin || '*';
|
|
362
352
|
res.header('Access-Control-Allow-Origin', allowOrigin);
|
|
363
353
|
res.header('Access-Control-Allow-Methods', 'GET,PUT,POST,DELETE,OPTIONS');
|
|
364
354
|
res.header('Access-Control-Allow-Headers', allowHeaders);
|
|
365
|
-
res.header('Access-Control-Expose-Headers', 'X-Parse-Job-Status-Id, X-Parse-Push-Status-Id');
|
|
366
|
-
|
|
355
|
+
res.header('Access-Control-Expose-Headers', 'X-Parse-Job-Status-Id, X-Parse-Push-Status-Id');
|
|
356
|
+
// intercept OPTIONS method
|
|
367
357
|
if ('OPTIONS' == req.method) {
|
|
368
358
|
res.sendStatus(200);
|
|
369
359
|
} else {
|
|
@@ -371,40 +361,32 @@ function allowCrossDomain(appId) {
|
|
|
371
361
|
}
|
|
372
362
|
};
|
|
373
363
|
}
|
|
374
|
-
|
|
375
364
|
function allowMethodOverride(req, res, next) {
|
|
376
365
|
if (req.method === 'POST' && req.body._method) {
|
|
377
366
|
req.originalMethod = req.method;
|
|
378
367
|
req.method = req.body._method;
|
|
379
368
|
delete req.body._method;
|
|
380
369
|
}
|
|
381
|
-
|
|
382
370
|
next();
|
|
383
371
|
}
|
|
384
|
-
|
|
385
372
|
function handleParseErrors(err, req, res, next) {
|
|
386
373
|
const log = req.config && req.config.loggerController || _logger.default;
|
|
387
|
-
|
|
388
374
|
if (err instanceof _node.default.Error) {
|
|
389
375
|
if (req.config && req.config.enableExpressErrorHandler) {
|
|
390
376
|
return next(err);
|
|
391
377
|
}
|
|
392
|
-
|
|
393
|
-
|
|
394
|
-
|
|
378
|
+
let httpStatus;
|
|
379
|
+
// TODO: fill out this mapping
|
|
395
380
|
switch (err.code) {
|
|
396
381
|
case _node.default.Error.INTERNAL_SERVER_ERROR:
|
|
397
382
|
httpStatus = 500;
|
|
398
383
|
break;
|
|
399
|
-
|
|
400
384
|
case _node.default.Error.OBJECT_NOT_FOUND:
|
|
401
385
|
httpStatus = 404;
|
|
402
386
|
break;
|
|
403
|
-
|
|
404
387
|
default:
|
|
405
388
|
httpStatus = 400;
|
|
406
389
|
}
|
|
407
|
-
|
|
408
390
|
res.status(httpStatus);
|
|
409
391
|
res.json({
|
|
410
392
|
code: err.code,
|
|
@@ -416,7 +398,6 @@ function handleParseErrors(err, req, res, next) {
|
|
|
416
398
|
res.json({
|
|
417
399
|
error: err.message
|
|
418
400
|
});
|
|
419
|
-
|
|
420
401
|
if (!(process && process.env.TESTING)) {
|
|
421
402
|
next(err);
|
|
422
403
|
}
|
|
@@ -427,23 +408,19 @@ function handleParseErrors(err, req, res, next) {
|
|
|
427
408
|
code: _node.default.Error.INTERNAL_SERVER_ERROR,
|
|
428
409
|
message: 'Internal server error.'
|
|
429
410
|
});
|
|
430
|
-
|
|
431
411
|
if (!(process && process.env.TESTING)) {
|
|
432
412
|
next(err);
|
|
433
413
|
}
|
|
434
414
|
}
|
|
435
415
|
}
|
|
436
|
-
|
|
437
416
|
function enforceMasterKeyAccess(req, res, next) {
|
|
438
417
|
if (!req.auth.isMaster) {
|
|
439
418
|
res.status(403);
|
|
440
419
|
res.end('{"error":"unauthorized: master key is required"}');
|
|
441
420
|
return;
|
|
442
421
|
}
|
|
443
|
-
|
|
444
422
|
next();
|
|
445
423
|
}
|
|
446
|
-
|
|
447
424
|
function promiseEnforceMasterKeyAccess(request) {
|
|
448
425
|
if (!request.auth.isMaster) {
|
|
449
426
|
const error = new Error();
|
|
@@ -451,56 +428,101 @@ function promiseEnforceMasterKeyAccess(request) {
|
|
|
451
428
|
error.message = 'unauthorized: master key is required';
|
|
452
429
|
throw error;
|
|
453
430
|
}
|
|
454
|
-
|
|
455
431
|
return Promise.resolve();
|
|
456
432
|
}
|
|
433
|
+
const addRateLimit = (route, config) => {
|
|
434
|
+
if (typeof config === 'string') {
|
|
435
|
+
config = _Config.default.get(config);
|
|
436
|
+
}
|
|
437
|
+
for (const key in route) {
|
|
438
|
+
if (!_Definitions.RateLimitOptions[key]) {
|
|
439
|
+
throw `Invalid rate limit option "${key}"`;
|
|
440
|
+
}
|
|
441
|
+
}
|
|
442
|
+
if (!config.rateLimits) {
|
|
443
|
+
config.rateLimits = [];
|
|
444
|
+
}
|
|
445
|
+
config.rateLimits.push({
|
|
446
|
+
path: (0, _pathToRegexp.default)(route.requestPath),
|
|
447
|
+
handler: (0, _expressRateLimit.default)({
|
|
448
|
+
windowMs: route.requestTimeWindow,
|
|
449
|
+
max: route.requestCount,
|
|
450
|
+
message: route.errorResponseMessage || _Definitions.RateLimitOptions.errorResponseMessage.default,
|
|
451
|
+
handler: (request, response, next, options) => {
|
|
452
|
+
throw {
|
|
453
|
+
code: _node.default.Error.CONNECTION_FAILED,
|
|
454
|
+
message: options.message
|
|
455
|
+
};
|
|
456
|
+
},
|
|
457
|
+
skip: request => {
|
|
458
|
+
var _request$auth;
|
|
459
|
+
if (request.ip === '127.0.0.1' && !route.includeInternalRequests) {
|
|
460
|
+
return true;
|
|
461
|
+
}
|
|
462
|
+
if (route.includeMasterKey) {
|
|
463
|
+
return false;
|
|
464
|
+
}
|
|
465
|
+
if (route.requestMethods) {
|
|
466
|
+
if (Array.isArray(route.requestMethods)) {
|
|
467
|
+
if (!route.requestMethods.includes(request.method)) {
|
|
468
|
+
return true;
|
|
469
|
+
}
|
|
470
|
+
} else {
|
|
471
|
+
const regExp = new RegExp(route.requestMethods);
|
|
472
|
+
if (!regExp.test(request.method)) {
|
|
473
|
+
return true;
|
|
474
|
+
}
|
|
475
|
+
}
|
|
476
|
+
}
|
|
477
|
+
return (_request$auth = request.auth) === null || _request$auth === void 0 ? void 0 : _request$auth.isMaster;
|
|
478
|
+
},
|
|
479
|
+
keyGenerator: request => {
|
|
480
|
+
return request.config.ip;
|
|
481
|
+
}
|
|
482
|
+
})
|
|
483
|
+
});
|
|
484
|
+
_Config.default.put(config);
|
|
485
|
+
};
|
|
486
|
+
|
|
457
487
|
/**
|
|
458
488
|
* Deduplicates a request to ensure idempotency. Duplicates are determined by the request ID
|
|
459
489
|
* in the request header. If a request has no request ID, it is executed anyway.
|
|
460
490
|
* @param {*} req The request to evaluate.
|
|
461
491
|
* @returns Promise<{}>
|
|
462
492
|
*/
|
|
463
|
-
|
|
464
|
-
|
|
493
|
+
exports.addRateLimit = addRateLimit;
|
|
465
494
|
function promiseEnsureIdempotency(req) {
|
|
466
495
|
// Enable feature only for MongoDB
|
|
467
496
|
if (!(req.config.database.adapter instanceof _MongoStorageAdapter.default || req.config.database.adapter instanceof _PostgresStorageAdapter.default)) {
|
|
468
497
|
return Promise.resolve();
|
|
469
|
-
}
|
|
470
|
-
|
|
471
|
-
|
|
498
|
+
}
|
|
499
|
+
// Get parameters
|
|
472
500
|
const config = req.config;
|
|
473
501
|
const requestId = ((req || {}).headers || {})['x-parse-request-id'];
|
|
474
502
|
const {
|
|
475
503
|
paths,
|
|
476
504
|
ttl
|
|
477
505
|
} = config.idempotencyOptions;
|
|
478
|
-
|
|
479
506
|
if (!requestId || !config.idempotencyOptions) {
|
|
480
507
|
return Promise.resolve();
|
|
481
|
-
}
|
|
508
|
+
}
|
|
509
|
+
// Request path may contain trailing slashes, depending on the original request, so remove
|
|
482
510
|
// leading and trailing slashes to make it easier to specify paths in the configuration
|
|
483
|
-
|
|
484
|
-
|
|
485
|
-
const reqPath = req.path.replace(/^\/|\/$/, ''); // Determine whether idempotency is enabled for current request path
|
|
486
|
-
|
|
511
|
+
const reqPath = req.path.replace(/^\/|\/$/, '');
|
|
512
|
+
// Determine whether idempotency is enabled for current request path
|
|
487
513
|
let match = false;
|
|
488
|
-
|
|
489
514
|
for (const path of paths) {
|
|
490
515
|
// Assume one wants a path to always match from the beginning to prevent any mistakes
|
|
491
516
|
const regex = new RegExp(path.charAt(0) === '^' ? path : '^' + path);
|
|
492
|
-
|
|
493
517
|
if (reqPath.match(regex)) {
|
|
494
518
|
match = true;
|
|
495
519
|
break;
|
|
496
520
|
}
|
|
497
521
|
}
|
|
498
|
-
|
|
499
522
|
if (!match) {
|
|
500
523
|
return Promise.resolve();
|
|
501
|
-
}
|
|
502
|
-
|
|
503
|
-
|
|
524
|
+
}
|
|
525
|
+
// Try to store request
|
|
504
526
|
const expiryDate = new Date(new Date().setSeconds(new Date().getSeconds() + ttl));
|
|
505
527
|
return _rest.default.create(config, _Auth.default.master(config), '_Idempotency', {
|
|
506
528
|
reqId: requestId,
|
|
@@ -509,16 +531,13 @@ function promiseEnsureIdempotency(req) {
|
|
|
509
531
|
if (e.code == _node.default.Error.DUPLICATE_VALUE) {
|
|
510
532
|
throw new _node.default.Error(_node.default.Error.DUPLICATE_REQUEST, 'Duplicate request');
|
|
511
533
|
}
|
|
512
|
-
|
|
513
534
|
throw e;
|
|
514
535
|
});
|
|
515
536
|
}
|
|
516
|
-
|
|
517
537
|
function invalidRequest(req, res) {
|
|
518
538
|
res.status(403);
|
|
519
539
|
res.end('{"error":"unauthorized"}');
|
|
520
540
|
}
|
|
521
|
-
|
|
522
541
|
function malformedContext(req, res) {
|
|
523
542
|
res.status(400);
|
|
524
543
|
res.json({
|
|
@@ -526,4 +545,4 @@ function malformedContext(req, res) {
|
|
|
526
545
|
error: 'Invalid object for context.'
|
|
527
546
|
});
|
|
528
547
|
}
|
|
529
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/middlewares.js"],"names":["DEFAULT_ALLOWED_HEADERS","getMountForRequest","req","mountPathLength","originalUrl","length","url","mountPath","slice","protocol","get","handleParseHeaders","res","next","mount","context","JSON","parse","Object","prototype","toString","call","e","malformedContext","info","appId","sessionToken","masterKey","installationId","clientKey","javascriptKey","dotNetKey","restAPIKey","clientVersion","basicAuth","httpAuth","basicAuthAppId","AppCache","body","_noBody","fileViaJSON","Buffer","invalidRequest","_RevocableSession","_ApplicationId","_JavaScriptKey","_ClientVersion","_InstallationId","_SessionToken","_MasterKey","_context","_ContentType","headers","clientSDK","ClientSDK","fromString","fileData","base64","from","clientIp","getClientIp","app","config","Config","ip","masterKeyIps","indexOf","isMaster","auth","Auth","isReadOnlyMaster","readOnlyMasterKey","isReadOnly","keys","oneKeyConfigured","some","key","undefined","oneKeyMatches","userFromJWT","user","Promise","resolve","then","getAuthForLegacySessionToken","getAuthForSessionToken","catch","error","Parse","Error","loggerController","UNKNOWN_ERROR","split","connection","remoteAddress","socket","authorization","header","authPrefix","match","toLowerCase","encodedAuth","substring","credentials","decodeBase64","jsKeyPrefix","matchKey","str","allowCrossDomain","allowHeaders","join","allowOrigin","method","sendStatus","allowMethodOverride","_method","originalMethod","handleParseErrors","err","log","defaultLogger","enableExpressErrorHandler","httpStatus","code","INTERNAL_SERVER_ERROR","OBJECT_NOT_FOUND","status","json","message","process","env","TESTING","stack","enforceMasterKeyAccess","end","promiseEnforceMasterKeyAccess","request","promiseEnsureIdempotency","database","adapter","MongoStorageAdapter","PostgresStorageAdapter","requestId","paths","ttl","idempotencyOptions","reqPath","path","replace","regex","RegExp","charAt","expiryDate","Date","setSeconds","getSeconds","rest","create","master","reqId","expire","_encode","DUPLICATE_VALUE","DUPLICATE_REQUEST","INVALID_JSON"],"mappings":"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","sourcesContent":["import AppCache from './cache';\nimport Parse from 'parse/node';\nimport auth from './Auth';\nimport Config from './Config';\nimport ClientSDK from './ClientSDK';\nimport defaultLogger from './logger';\nimport rest from './rest';\nimport MongoStorageAdapter from './Adapters/Storage/Mongo/MongoStorageAdapter';\nimport PostgresStorageAdapter from './Adapters/Storage/Postgres/PostgresStorageAdapter';\n\nexport const DEFAULT_ALLOWED_HEADERS =\n  'X-Parse-Master-Key, X-Parse-REST-API-Key, X-Parse-Javascript-Key, X-Parse-Application-Id, X-Parse-Client-Version, X-Parse-Session-Token, X-Requested-With, X-Parse-Revocable-Session, X-Parse-Request-Id, Content-Type, Pragma, Cache-Control';\n\nconst getMountForRequest = function (req) {\n  const mountPathLength = req.originalUrl.length - req.url.length;\n  const mountPath = req.originalUrl.slice(0, mountPathLength);\n  return req.protocol + '://' + req.get('host') + mountPath;\n};\n\n// Checks that the request is authorized for this app and checks user\n// auth too.\n// The bodyparser should run before this middleware.\n// Adds info to the request:\n// req.config - the Config for this app\n// req.auth - the Auth for this request\nexport function handleParseHeaders(req, res, next) {\n  var mount = getMountForRequest(req);\n\n  let context = {};\n  if (req.get('X-Parse-Cloud-Context') != null) {\n    try {\n      context = JSON.parse(req.get('X-Parse-Cloud-Context'));\n      if (Object.prototype.toString.call(context) !== '[object Object]') {\n        throw 'Context is not an object';\n      }\n    } catch (e) {\n      return malformedContext(req, res);\n    }\n  }\n  var info = {\n    appId: req.get('X-Parse-Application-Id'),\n    sessionToken: req.get('X-Parse-Session-Token'),\n    masterKey: req.get('X-Parse-Master-Key'),\n    installationId: req.get('X-Parse-Installation-Id'),\n    clientKey: req.get('X-Parse-Client-Key'),\n    javascriptKey: req.get('X-Parse-Javascript-Key'),\n    dotNetKey: req.get('X-Parse-Windows-Key'),\n    restAPIKey: req.get('X-Parse-REST-API-Key'),\n    clientVersion: req.get('X-Parse-Client-Version'),\n    context: context,\n  };\n\n  var basicAuth = httpAuth(req);\n\n  if (basicAuth) {\n    var basicAuthAppId = basicAuth.appId;\n    if (AppCache.get(basicAuthAppId)) {\n      info.appId = basicAuthAppId;\n      info.masterKey = basicAuth.masterKey || info.masterKey;\n      info.javascriptKey = basicAuth.javascriptKey || info.javascriptKey;\n    }\n  }\n\n  if (req.body) {\n    // Unity SDK sends a _noBody key which needs to be removed.\n    // Unclear at this point if action needs to be taken.\n    delete req.body._noBody;\n  }\n\n  var fileViaJSON = false;\n\n  if (!info.appId || !AppCache.get(info.appId)) {\n    // See if we can find the app id on the body.\n    if (req.body instanceof Buffer) {\n      // The only chance to find the app id is if this is a file\n      // upload that actually is a JSON body. So try to parse it.\n      // https://github.com/parse-community/parse-server/issues/6589\n      // It is also possible that the client is trying to upload a file but forgot\n      // to provide x-parse-app-id in header and parse a binary file will fail\n      try {\n        req.body = JSON.parse(req.body);\n      } catch (e) {\n        return invalidRequest(req, res);\n      }\n      fileViaJSON = true;\n    }\n\n    if (req.body) {\n      delete req.body._RevocableSession;\n    }\n\n    if (\n      req.body &&\n      req.body._ApplicationId &&\n      AppCache.get(req.body._ApplicationId) &&\n      (!info.masterKey || AppCache.get(req.body._ApplicationId).masterKey === info.masterKey)\n    ) {\n      info.appId = req.body._ApplicationId;\n      info.javascriptKey = req.body._JavaScriptKey || '';\n      delete req.body._ApplicationId;\n      delete req.body._JavaScriptKey;\n      // TODO: test that the REST API formats generated by the other\n      // SDKs are handled ok\n      if (req.body._ClientVersion) {\n        info.clientVersion = req.body._ClientVersion;\n        delete req.body._ClientVersion;\n      }\n      if (req.body._InstallationId) {\n        info.installationId = req.body._InstallationId;\n        delete req.body._InstallationId;\n      }\n      if (req.body._SessionToken) {\n        info.sessionToken = req.body._SessionToken;\n        delete req.body._SessionToken;\n      }\n      if (req.body._MasterKey) {\n        info.masterKey = req.body._MasterKey;\n        delete req.body._MasterKey;\n      }\n      if (req.body._context) {\n        if (req.body._context instanceof Object) {\n          info.context = req.body._context;\n        } else {\n          try {\n            info.context = JSON.parse(req.body._context);\n            if (Object.prototype.toString.call(info.context) !== '[object Object]') {\n              throw 'Context is not an object';\n            }\n          } catch (e) {\n            return malformedContext(req, res);\n          }\n        }\n        delete req.body._context;\n      }\n      if (req.body._ContentType) {\n        req.headers['content-type'] = req.body._ContentType;\n        delete req.body._ContentType;\n      }\n    } else {\n      return invalidRequest(req, res);\n    }\n  }\n\n  if (info.sessionToken && typeof info.sessionToken !== 'string') {\n    info.sessionToken = info.sessionToken.toString();\n  }\n\n  if (info.clientVersion) {\n    info.clientSDK = ClientSDK.fromString(info.clientVersion);\n  }\n\n  if (fileViaJSON) {\n    req.fileData = req.body.fileData;\n    // We need to repopulate req.body with a buffer\n    var base64 = req.body.base64;\n    req.body = Buffer.from(base64, 'base64');\n  }\n\n  const clientIp = getClientIp(req);\n\n  info.app = AppCache.get(info.appId);\n  req.config = Config.get(info.appId, mount);\n  req.config.headers = req.headers || {};\n  req.config.ip = clientIp;\n  req.info = info;\n\n  if (\n    info.masterKey &&\n    req.config.masterKeyIps &&\n    req.config.masterKeyIps.length !== 0 &&\n    req.config.masterKeyIps.indexOf(clientIp) === -1\n  ) {\n    return invalidRequest(req, res);\n  }\n\n  var isMaster = info.masterKey === req.config.masterKey;\n\n  if (isMaster) {\n    req.auth = new auth.Auth({\n      config: req.config,\n      installationId: info.installationId,\n      isMaster: true,\n    });\n    next();\n    return;\n  }\n\n  var isReadOnlyMaster = info.masterKey === req.config.readOnlyMasterKey;\n  if (\n    typeof req.config.readOnlyMasterKey != 'undefined' &&\n    req.config.readOnlyMasterKey &&\n    isReadOnlyMaster\n  ) {\n    req.auth = new auth.Auth({\n      config: req.config,\n      installationId: info.installationId,\n      isMaster: true,\n      isReadOnly: true,\n    });\n    next();\n    return;\n  }\n\n  // Client keys are not required in parse-server, but if any have been configured in the server, validate them\n  //  to preserve original behavior.\n  const keys = ['clientKey', 'javascriptKey', 'dotNetKey', 'restAPIKey'];\n  const oneKeyConfigured = keys.some(function (key) {\n    return req.config[key] !== undefined;\n  });\n  const oneKeyMatches = keys.some(function (key) {\n    return req.config[key] !== undefined && info[key] === req.config[key];\n  });\n\n  if (oneKeyConfigured && !oneKeyMatches) {\n    return invalidRequest(req, res);\n  }\n\n  if (req.url == '/login') {\n    delete info.sessionToken;\n  }\n\n  if (req.userFromJWT) {\n    req.auth = new auth.Auth({\n      config: req.config,\n      installationId: info.installationId,\n      isMaster: false,\n      user: req.userFromJWT,\n    });\n    next();\n    return;\n  }\n\n  if (!info.sessionToken) {\n    req.auth = new auth.Auth({\n      config: req.config,\n      installationId: info.installationId,\n      isMaster: false,\n    });\n    next();\n    return;\n  }\n\n  return Promise.resolve()\n    .then(() => {\n      // handle the upgradeToRevocableSession path on it's own\n      if (\n        info.sessionToken &&\n        req.url === '/upgradeToRevocableSession' &&\n        info.sessionToken.indexOf('r:') != 0\n      ) {\n        return auth.getAuthForLegacySessionToken({\n          config: req.config,\n          installationId: info.installationId,\n          sessionToken: info.sessionToken,\n        });\n      } else {\n        return auth.getAuthForSessionToken({\n          config: req.config,\n          installationId: info.installationId,\n          sessionToken: info.sessionToken,\n        });\n      }\n    })\n    .then(auth => {\n      if (auth) {\n        req.auth = auth;\n        next();\n      }\n    })\n    .catch(error => {\n      if (error instanceof Parse.Error) {\n        next(error);\n        return;\n      } else {\n        // TODO: Determine the correct error scenario.\n        req.config.loggerController.error('error getting auth for sessionToken', error);\n        throw new Parse.Error(Parse.Error.UNKNOWN_ERROR, error);\n      }\n    });\n}\n\nfunction getClientIp(req) {\n  if (req.headers['x-forwarded-for']) {\n    // try to get from x-forwared-for if it set (behind reverse proxy)\n    return req.headers['x-forwarded-for'].split(',')[0];\n  } else if (req.connection && req.connection.remoteAddress) {\n    // no proxy, try getting from connection.remoteAddress\n    return req.connection.remoteAddress;\n  } else if (req.socket) {\n    // try to get it from req.socket\n    return req.socket.remoteAddress;\n  } else if (req.connection && req.connection.socket) {\n    // try to get it form the connection.socket\n    return req.connection.socket.remoteAddress;\n  } else {\n    // if non above, fallback.\n    return req.ip;\n  }\n}\n\nfunction httpAuth(req) {\n  if (!(req.req || req).headers.authorization) return;\n\n  var header = (req.req || req).headers.authorization;\n  var appId, masterKey, javascriptKey;\n\n  // parse header\n  var authPrefix = 'basic ';\n\n  var match = header.toLowerCase().indexOf(authPrefix);\n\n  if (match == 0) {\n    var encodedAuth = header.substring(authPrefix.length, header.length);\n    var credentials = decodeBase64(encodedAuth).split(':');\n\n    if (credentials.length == 2) {\n      appId = credentials[0];\n      var key = credentials[1];\n\n      var jsKeyPrefix = 'javascript-key=';\n\n      var matchKey = key.indexOf(jsKeyPrefix);\n      if (matchKey == 0) {\n        javascriptKey = key.substring(jsKeyPrefix.length, key.length);\n      } else {\n        masterKey = key;\n      }\n    }\n  }\n\n  return { appId: appId, masterKey: masterKey, javascriptKey: javascriptKey };\n}\n\nfunction decodeBase64(str) {\n  return Buffer.from(str, 'base64').toString();\n}\n\nexport function allowCrossDomain(appId) {\n  return (req, res, next) => {\n    const config = Config.get(appId, getMountForRequest(req));\n    let allowHeaders = DEFAULT_ALLOWED_HEADERS;\n    if (config && config.allowHeaders) {\n      allowHeaders += `, ${config.allowHeaders.join(', ')}`;\n    }\n    const allowOrigin = (config && config.allowOrigin) || '*';\n    res.header('Access-Control-Allow-Origin', allowOrigin);\n    res.header('Access-Control-Allow-Methods', 'GET,PUT,POST,DELETE,OPTIONS');\n    res.header('Access-Control-Allow-Headers', allowHeaders);\n    res.header('Access-Control-Expose-Headers', 'X-Parse-Job-Status-Id, X-Parse-Push-Status-Id');\n    // intercept OPTIONS method\n    if ('OPTIONS' == req.method) {\n      res.sendStatus(200);\n    } else {\n      next();\n    }\n  };\n}\n\nexport function allowMethodOverride(req, res, next) {\n  if (req.method === 'POST' && req.body._method) {\n    req.originalMethod = req.method;\n    req.method = req.body._method;\n    delete req.body._method;\n  }\n  next();\n}\n\nexport function handleParseErrors(err, req, res, next) {\n  const log = (req.config && req.config.loggerController) || defaultLogger;\n  if (err instanceof Parse.Error) {\n    if (req.config && req.config.enableExpressErrorHandler) {\n      return next(err);\n    }\n    let httpStatus;\n    // TODO: fill out this mapping\n    switch (err.code) {\n      case Parse.Error.INTERNAL_SERVER_ERROR:\n        httpStatus = 500;\n        break;\n      case Parse.Error.OBJECT_NOT_FOUND:\n        httpStatus = 404;\n        break;\n      default:\n        httpStatus = 400;\n    }\n    res.status(httpStatus);\n    res.json({ code: err.code, error: err.message });\n    log.error('Parse error: ', err);\n  } else if (err.status && err.message) {\n    res.status(err.status);\n    res.json({ error: err.message });\n    if (!(process && process.env.TESTING)) {\n      next(err);\n    }\n  } else {\n    log.error('Uncaught internal server error.', err, err.stack);\n    res.status(500);\n    res.json({\n      code: Parse.Error.INTERNAL_SERVER_ERROR,\n      message: 'Internal server error.',\n    });\n    if (!(process && process.env.TESTING)) {\n      next(err);\n    }\n  }\n}\n\nexport function enforceMasterKeyAccess(req, res, next) {\n  if (!req.auth.isMaster) {\n    res.status(403);\n    res.end('{\"error\":\"unauthorized: master key is required\"}');\n    return;\n  }\n  next();\n}\n\nexport function promiseEnforceMasterKeyAccess(request) {\n  if (!request.auth.isMaster) {\n    const error = new Error();\n    error.status = 403;\n    error.message = 'unauthorized: master key is required';\n    throw error;\n  }\n  return Promise.resolve();\n}\n\n/**\n * Deduplicates a request to ensure idempotency. Duplicates are determined by the request ID\n * in the request header. If a request has no request ID, it is executed anyway.\n * @param {*} req The request to evaluate.\n * @returns Promise<{}>\n */\nexport function promiseEnsureIdempotency(req) {\n  // Enable feature only for MongoDB\n  if (\n    !(\n      req.config.database.adapter instanceof MongoStorageAdapter ||\n      req.config.database.adapter instanceof PostgresStorageAdapter\n    )\n  ) {\n    return Promise.resolve();\n  }\n  // Get parameters\n  const config = req.config;\n  const requestId = ((req || {}).headers || {})['x-parse-request-id'];\n  const { paths, ttl } = config.idempotencyOptions;\n  if (!requestId || !config.idempotencyOptions) {\n    return Promise.resolve();\n  }\n  // Request path may contain trailing slashes, depending on the original request, so remove\n  // leading and trailing slashes to make it easier to specify paths in the configuration\n  const reqPath = req.path.replace(/^\\/|\\/$/, '');\n  // Determine whether idempotency is enabled for current request path\n  let match = false;\n  for (const path of paths) {\n    // Assume one wants a path to always match from the beginning to prevent any mistakes\n    const regex = new RegExp(path.charAt(0) === '^' ? path : '^' + path);\n    if (reqPath.match(regex)) {\n      match = true;\n      break;\n    }\n  }\n  if (!match) {\n    return Promise.resolve();\n  }\n  // Try to store request\n  const expiryDate = new Date(new Date().setSeconds(new Date().getSeconds() + ttl));\n  return rest\n    .create(config, auth.master(config), '_Idempotency', {\n      reqId: requestId,\n      expire: Parse._encode(expiryDate),\n    })\n    .catch(e => {\n      if (e.code == Parse.Error.DUPLICATE_VALUE) {\n        throw new Parse.Error(Parse.Error.DUPLICATE_REQUEST, 'Duplicate request');\n      }\n      throw e;\n    });\n}\n\nfunction invalidRequest(req, res) {\n  res.status(403);\n  res.end('{\"error\":\"unauthorized\"}');\n}\n\nfunction malformedContext(req, res) {\n  res.status(400);\n  res.json({ code: Parse.Error.INVALID_JSON, error: 'Invalid object for context.' });\n}\n"]}
|
|
548
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["DEFAULT_ALLOWED_HEADERS","getMountForRequest","req","mountPathLength","originalUrl","length","url","mountPath","slice","protocol","get","handleParseHeaders","res","next","mount","context","JSON","parse","Object","prototype","toString","call","e","malformedContext","info","appId","sessionToken","masterKey","maintenanceKey","installationId","clientKey","javascriptKey","dotNetKey","restAPIKey","clientVersion","basicAuth","httpAuth","basicAuthAppId","AppCache","body","_noBody","fileViaJSON","Buffer","invalidRequest","_RevocableSession","_ApplicationId","_JavaScriptKey","_ClientVersion","_InstallationId","_SessionToken","_MasterKey","_context","_ContentType","headers","clientSDK","ClientSDK","fromString","fileData","base64","from","clientIp","getClientIp","config","Config","state","status","json","code","Parse","Error","INTERNAL_SERVER_ERROR","error","app","ip","isMaintenance","ipRangeCheck","maintenanceKeyIps","auth","Auth","log","loggerController","defaultLogger","isMaster","masterKeyIps","handleRateLimit","isReadOnlyMaster","readOnlyMasterKey","isReadOnly","keys","oneKeyConfigured","some","key","undefined","oneKeyMatches","userFromJWT","user","rateLimits","Promise","all","map","limit","pathExp","RegExp","path","test","handler","err","CONNECTION_FAILED","message","handleParseSession","requestAuth","indexOf","getAuthForLegacySessionToken","getAuthForSessionToken","UNKNOWN_ERROR","authorization","header","authPrefix","match","toLowerCase","encodedAuth","substring","credentials","decodeBase64","split","jsKeyPrefix","matchKey","str","allowCrossDomain","allowHeaders","join","allowOrigin","method","sendStatus","allowMethodOverride","_method","originalMethod","handleParseErrors","enableExpressErrorHandler","httpStatus","OBJECT_NOT_FOUND","process","env","TESTING","stack","enforceMasterKeyAccess","end","promiseEnforceMasterKeyAccess","request","resolve","addRateLimit","route","RateLimitOptions","push","pathToRegexp","requestPath","rateLimit","windowMs","requestTimeWindow","max","requestCount","errorResponseMessage","default","response","options","skip","includeInternalRequests","includeMasterKey","requestMethods","Array","isArray","includes","regExp","keyGenerator","put","promiseEnsureIdempotency","database","adapter","MongoStorageAdapter","PostgresStorageAdapter","requestId","paths","ttl","idempotencyOptions","reqPath","replace","regex","charAt","expiryDate","Date","setSeconds","getSeconds","rest","create","master","reqId","expire","_encode","catch","DUPLICATE_VALUE","DUPLICATE_REQUEST","INVALID_JSON"],"sources":["../src/middlewares.js"],"sourcesContent":["import AppCache from './cache';\nimport Parse from 'parse/node';\nimport auth from './Auth';\nimport Config from './Config';\nimport ClientSDK from './ClientSDK';\nimport defaultLogger from './logger';\nimport rest from './rest';\nimport MongoStorageAdapter from './Adapters/Storage/Mongo/MongoStorageAdapter';\nimport PostgresStorageAdapter from './Adapters/Storage/Postgres/PostgresStorageAdapter';\nimport rateLimit from 'express-rate-limit';\nimport { RateLimitOptions } from './Options/Definitions';\nimport pathToRegexp from 'path-to-regexp';\nimport ipRangeCheck from 'ip-range-check';\n\nexport const DEFAULT_ALLOWED_HEADERS =\n  'X-Parse-Master-Key, X-Parse-REST-API-Key, X-Parse-Javascript-Key, X-Parse-Application-Id, X-Parse-Client-Version, X-Parse-Session-Token, X-Requested-With, X-Parse-Revocable-Session, X-Parse-Request-Id, Content-Type, Pragma, Cache-Control';\n\nconst getMountForRequest = function (req) {\n  const mountPathLength = req.originalUrl.length - req.url.length;\n  const mountPath = req.originalUrl.slice(0, mountPathLength);\n  return req.protocol + '://' + req.get('host') + mountPath;\n};\n\n// Checks that the request is authorized for this app and checks user\n// auth too.\n// The bodyparser should run before this middleware.\n// Adds info to the request:\n// req.config - the Config for this app\n// req.auth - the Auth for this request\nexport function handleParseHeaders(req, res, next) {\n  var mount = getMountForRequest(req);\n\n  let context = {};\n  if (req.get('X-Parse-Cloud-Context') != null) {\n    try {\n      context = JSON.parse(req.get('X-Parse-Cloud-Context'));\n      if (Object.prototype.toString.call(context) !== '[object Object]') {\n        throw 'Context is not an object';\n      }\n    } catch (e) {\n      return malformedContext(req, res);\n    }\n  }\n  var info = {\n    appId: req.get('X-Parse-Application-Id'),\n    sessionToken: req.get('X-Parse-Session-Token'),\n    masterKey: req.get('X-Parse-Master-Key'),\n    maintenanceKey: req.get('X-Parse-Maintenance-Key'),\n    installationId: req.get('X-Parse-Installation-Id'),\n    clientKey: req.get('X-Parse-Client-Key'),\n    javascriptKey: req.get('X-Parse-Javascript-Key'),\n    dotNetKey: req.get('X-Parse-Windows-Key'),\n    restAPIKey: req.get('X-Parse-REST-API-Key'),\n    clientVersion: req.get('X-Parse-Client-Version'),\n    context: context,\n  };\n\n  var basicAuth = httpAuth(req);\n\n  if (basicAuth) {\n    var basicAuthAppId = basicAuth.appId;\n    if (AppCache.get(basicAuthAppId)) {\n      info.appId = basicAuthAppId;\n      info.masterKey = basicAuth.masterKey || info.masterKey;\n      info.javascriptKey = basicAuth.javascriptKey || info.javascriptKey;\n    }\n  }\n\n  if (req.body) {\n    // Unity SDK sends a _noBody key which needs to be removed.\n    // Unclear at this point if action needs to be taken.\n    delete req.body._noBody;\n  }\n\n  var fileViaJSON = false;\n\n  if (!info.appId || !AppCache.get(info.appId)) {\n    // See if we can find the app id on the body.\n    if (req.body instanceof Buffer) {\n      // The only chance to find the app id is if this is a file\n      // upload that actually is a JSON body. So try to parse it.\n      // https://github.com/parse-community/parse-server/issues/6589\n      // It is also possible that the client is trying to upload a file but forgot\n      // to provide x-parse-app-id in header and parse a binary file will fail\n      try {\n        req.body = JSON.parse(req.body);\n      } catch (e) {\n        return invalidRequest(req, res);\n      }\n      fileViaJSON = true;\n    }\n\n    if (req.body) {\n      delete req.body._RevocableSession;\n    }\n\n    if (\n      req.body &&\n      req.body._ApplicationId &&\n      AppCache.get(req.body._ApplicationId) &&\n      (!info.masterKey || AppCache.get(req.body._ApplicationId).masterKey === info.masterKey)\n    ) {\n      info.appId = req.body._ApplicationId;\n      info.javascriptKey = req.body._JavaScriptKey || '';\n      delete req.body._ApplicationId;\n      delete req.body._JavaScriptKey;\n      // TODO: test that the REST API formats generated by the other\n      // SDKs are handled ok\n      if (req.body._ClientVersion) {\n        info.clientVersion = req.body._ClientVersion;\n        delete req.body._ClientVersion;\n      }\n      if (req.body._InstallationId) {\n        info.installationId = req.body._InstallationId;\n        delete req.body._InstallationId;\n      }\n      if (req.body._SessionToken) {\n        info.sessionToken = req.body._SessionToken;\n        delete req.body._SessionToken;\n      }\n      if (req.body._MasterKey) {\n        info.masterKey = req.body._MasterKey;\n        delete req.body._MasterKey;\n      }\n      if (req.body._context) {\n        if (req.body._context instanceof Object) {\n          info.context = req.body._context;\n        } else {\n          try {\n            info.context = JSON.parse(req.body._context);\n            if (Object.prototype.toString.call(info.context) !== '[object Object]') {\n              throw 'Context is not an object';\n            }\n          } catch (e) {\n            return malformedContext(req, res);\n          }\n        }\n        delete req.body._context;\n      }\n      if (req.body._ContentType) {\n        req.headers['content-type'] = req.body._ContentType;\n        delete req.body._ContentType;\n      }\n    } else {\n      return invalidRequest(req, res);\n    }\n  }\n\n  if (info.sessionToken && typeof info.sessionToken !== 'string') {\n    info.sessionToken = info.sessionToken.toString();\n  }\n\n  if (info.clientVersion) {\n    info.clientSDK = ClientSDK.fromString(info.clientVersion);\n  }\n\n  if (fileViaJSON) {\n    req.fileData = req.body.fileData;\n    // We need to repopulate req.body with a buffer\n    var base64 = req.body.base64;\n    req.body = Buffer.from(base64, 'base64');\n  }\n\n  const clientIp = getClientIp(req);\n  const config = Config.get(info.appId, mount);\n  if (config.state && config.state !== 'ok') {\n    res.status(500);\n    res.json({\n      code: Parse.Error.INTERNAL_SERVER_ERROR,\n      error: `Invalid server state: ${config.state}`,\n    });\n    return;\n  }\n\n  info.app = AppCache.get(info.appId);\n  req.config = config;\n  req.config.headers = req.headers || {};\n  req.config.ip = clientIp;\n  req.info = info;\n\n  const isMaintenance =\n    req.config.maintenanceKey && info.maintenanceKey === req.config.maintenanceKey;\n  if (isMaintenance) {\n    if (ipRangeCheck(clientIp, req.config.maintenanceKeyIps || [])) {\n      req.auth = new auth.Auth({\n        config: req.config,\n        installationId: info.installationId,\n        isMaintenance: true,\n      });\n      next();\n      return;\n    }\n    const log = req.config?.loggerController || defaultLogger;\n    log.error(\n      `Request using maintenance key rejected as the request IP address '${clientIp}' is not set in Parse Server option 'maintenanceKeyIps'.`\n    );\n  }\n\n  let isMaster = info.masterKey === req.config.masterKey;\n  if (isMaster && !ipRangeCheck(clientIp, req.config.masterKeyIps || [])) {\n    const log = req.config?.loggerController || defaultLogger;\n    log.error(\n      `Request using master key rejected as the request IP address '${clientIp}' is not set in Parse Server option 'masterKeyIps'.`\n    );\n    isMaster = false;\n  }\n\n  if (isMaster) {\n    req.auth = new auth.Auth({\n      config: req.config,\n      installationId: info.installationId,\n      isMaster: true,\n    });\n    return handleRateLimit(req, res, next);\n  }\n\n  var isReadOnlyMaster = info.masterKey === req.config.readOnlyMasterKey;\n  if (\n    typeof req.config.readOnlyMasterKey != 'undefined' &&\n    req.config.readOnlyMasterKey &&\n    isReadOnlyMaster\n  ) {\n    req.auth = new auth.Auth({\n      config: req.config,\n      installationId: info.installationId,\n      isMaster: true,\n      isReadOnly: true,\n    });\n    return handleRateLimit(req, res, next);\n  }\n\n  // Client keys are not required in parse-server, but if any have been configured in the server, validate them\n  //  to preserve original behavior.\n  const keys = ['clientKey', 'javascriptKey', 'dotNetKey', 'restAPIKey'];\n  const oneKeyConfigured = keys.some(function (key) {\n    return req.config[key] !== undefined;\n  });\n  const oneKeyMatches = keys.some(function (key) {\n    return req.config[key] !== undefined && info[key] === req.config[key];\n  });\n\n  if (oneKeyConfigured && !oneKeyMatches) {\n    return invalidRequest(req, res);\n  }\n\n  if (req.url == '/login') {\n    delete info.sessionToken;\n  }\n\n  if (req.userFromJWT) {\n    req.auth = new auth.Auth({\n      config: req.config,\n      installationId: info.installationId,\n      isMaster: false,\n      user: req.userFromJWT,\n    });\n    return handleRateLimit(req, res, next);\n  }\n\n  if (!info.sessionToken) {\n    req.auth = new auth.Auth({\n      config: req.config,\n      installationId: info.installationId,\n      isMaster: false,\n    });\n  }\n  handleRateLimit(req, res, next);\n}\n\nconst handleRateLimit = async (req, res, next) => {\n  const rateLimits = req.config.rateLimits || [];\n  try {\n    await Promise.all(\n      rateLimits.map(async limit => {\n        const pathExp = new RegExp(limit.path);\n        if (pathExp.test(req.url)) {\n          await limit.handler(req, res, err => {\n            if (err) {\n              if (err.code === Parse.Error.CONNECTION_FAILED) {\n                throw err;\n              }\n              req.config.loggerController.error(\n                'An unknown error occured when attempting to apply the rate limiter: ',\n                err\n              );\n            }\n          });\n        }\n      })\n    );\n  } catch (error) {\n    res.status(429);\n    res.json({ code: Parse.Error.CONNECTION_FAILED, error: error.message });\n    return;\n  }\n  next();\n};\n\nexport const handleParseSession = async (req, res, next) => {\n  try {\n    const info = req.info;\n    if (req.auth) {\n      next();\n      return;\n    }\n    let requestAuth = null;\n    if (\n      info.sessionToken &&\n      req.url === '/upgradeToRevocableSession' &&\n      info.sessionToken.indexOf('r:') != 0\n    ) {\n      requestAuth = await auth.getAuthForLegacySessionToken({\n        config: req.config,\n        installationId: info.installationId,\n        sessionToken: info.sessionToken,\n      });\n    } else {\n      requestAuth = await auth.getAuthForSessionToken({\n        config: req.config,\n        installationId: info.installationId,\n        sessionToken: info.sessionToken,\n      });\n    }\n    req.auth = requestAuth;\n    next();\n  } catch (error) {\n    if (error instanceof Parse.Error) {\n      next(error);\n      return;\n    }\n    // TODO: Determine the correct error scenario.\n    req.config.loggerController.error('error getting auth for sessionToken', error);\n    throw new Parse.Error(Parse.Error.UNKNOWN_ERROR, error);\n  }\n};\n\nfunction getClientIp(req) {\n  return req.ip;\n}\n\nfunction httpAuth(req) {\n  if (!(req.req || req).headers.authorization) return;\n\n  var header = (req.req || req).headers.authorization;\n  var appId, masterKey, javascriptKey;\n\n  // parse header\n  var authPrefix = 'basic ';\n\n  var match = header.toLowerCase().indexOf(authPrefix);\n\n  if (match == 0) {\n    var encodedAuth = header.substring(authPrefix.length, header.length);\n    var credentials = decodeBase64(encodedAuth).split(':');\n\n    if (credentials.length == 2) {\n      appId = credentials[0];\n      var key = credentials[1];\n\n      var jsKeyPrefix = 'javascript-key=';\n\n      var matchKey = key.indexOf(jsKeyPrefix);\n      if (matchKey == 0) {\n        javascriptKey = key.substring(jsKeyPrefix.length, key.length);\n      } else {\n        masterKey = key;\n      }\n    }\n  }\n\n  return { appId: appId, masterKey: masterKey, javascriptKey: javascriptKey };\n}\n\nfunction decodeBase64(str) {\n  return Buffer.from(str, 'base64').toString();\n}\n\nexport function allowCrossDomain(appId) {\n  return (req, res, next) => {\n    const config = Config.get(appId, getMountForRequest(req));\n    let allowHeaders = DEFAULT_ALLOWED_HEADERS;\n    if (config && config.allowHeaders) {\n      allowHeaders += `, ${config.allowHeaders.join(', ')}`;\n    }\n    const allowOrigin = (config && config.allowOrigin) || '*';\n    res.header('Access-Control-Allow-Origin', allowOrigin);\n    res.header('Access-Control-Allow-Methods', 'GET,PUT,POST,DELETE,OPTIONS');\n    res.header('Access-Control-Allow-Headers', allowHeaders);\n    res.header('Access-Control-Expose-Headers', 'X-Parse-Job-Status-Id, X-Parse-Push-Status-Id');\n    // intercept OPTIONS method\n    if ('OPTIONS' == req.method) {\n      res.sendStatus(200);\n    } else {\n      next();\n    }\n  };\n}\n\nexport function allowMethodOverride(req, res, next) {\n  if (req.method === 'POST' && req.body._method) {\n    req.originalMethod = req.method;\n    req.method = req.body._method;\n    delete req.body._method;\n  }\n  next();\n}\n\nexport function handleParseErrors(err, req, res, next) {\n  const log = (req.config && req.config.loggerController) || defaultLogger;\n  if (err instanceof Parse.Error) {\n    if (req.config && req.config.enableExpressErrorHandler) {\n      return next(err);\n    }\n    let httpStatus;\n    // TODO: fill out this mapping\n    switch (err.code) {\n      case Parse.Error.INTERNAL_SERVER_ERROR:\n        httpStatus = 500;\n        break;\n      case Parse.Error.OBJECT_NOT_FOUND:\n        httpStatus = 404;\n        break;\n      default:\n        httpStatus = 400;\n    }\n    res.status(httpStatus);\n    res.json({ code: err.code, error: err.message });\n    log.error('Parse error: ', err);\n  } else if (err.status && err.message) {\n    res.status(err.status);\n    res.json({ error: err.message });\n    if (!(process && process.env.TESTING)) {\n      next(err);\n    }\n  } else {\n    log.error('Uncaught internal server error.', err, err.stack);\n    res.status(500);\n    res.json({\n      code: Parse.Error.INTERNAL_SERVER_ERROR,\n      message: 'Internal server error.',\n    });\n    if (!(process && process.env.TESTING)) {\n      next(err);\n    }\n  }\n}\n\nexport function enforceMasterKeyAccess(req, res, next) {\n  if (!req.auth.isMaster) {\n    res.status(403);\n    res.end('{\"error\":\"unauthorized: master key is required\"}');\n    return;\n  }\n  next();\n}\n\nexport function promiseEnforceMasterKeyAccess(request) {\n  if (!request.auth.isMaster) {\n    const error = new Error();\n    error.status = 403;\n    error.message = 'unauthorized: master key is required';\n    throw error;\n  }\n  return Promise.resolve();\n}\n\nexport const addRateLimit = (route, config) => {\n  if (typeof config === 'string') {\n    config = Config.get(config);\n  }\n  for (const key in route) {\n    if (!RateLimitOptions[key]) {\n      throw `Invalid rate limit option \"${key}\"`;\n    }\n  }\n  if (!config.rateLimits) {\n    config.rateLimits = [];\n  }\n  config.rateLimits.push({\n    path: pathToRegexp(route.requestPath),\n    handler: rateLimit({\n      windowMs: route.requestTimeWindow,\n      max: route.requestCount,\n      message: route.errorResponseMessage || RateLimitOptions.errorResponseMessage.default,\n      handler: (request, response, next, options) => {\n        throw {\n          code: Parse.Error.CONNECTION_FAILED,\n          message: options.message,\n        };\n      },\n      skip: request => {\n        if (request.ip === '127.0.0.1' && !route.includeInternalRequests) {\n          return true;\n        }\n        if (route.includeMasterKey) {\n          return false;\n        }\n        if (route.requestMethods) {\n          if (Array.isArray(route.requestMethods)) {\n            if (!route.requestMethods.includes(request.method)) {\n              return true;\n            }\n          } else {\n            const regExp = new RegExp(route.requestMethods);\n            if (!regExp.test(request.method)) {\n              return true;\n            }\n          }\n        }\n        return request.auth?.isMaster;\n      },\n      keyGenerator: request => {\n        return request.config.ip;\n      },\n    }),\n  });\n  Config.put(config);\n};\n\n/**\n * Deduplicates a request to ensure idempotency. Duplicates are determined by the request ID\n * in the request header. If a request has no request ID, it is executed anyway.\n * @param {*} req The request to evaluate.\n * @returns Promise<{}>\n */\nexport function promiseEnsureIdempotency(req) {\n  // Enable feature only for MongoDB\n  if (\n    !(\n      req.config.database.adapter instanceof MongoStorageAdapter ||\n      req.config.database.adapter instanceof PostgresStorageAdapter\n    )\n  ) {\n    return Promise.resolve();\n  }\n  // Get parameters\n  const config = req.config;\n  const requestId = ((req || {}).headers || {})['x-parse-request-id'];\n  const { paths, ttl } = config.idempotencyOptions;\n  if (!requestId || !config.idempotencyOptions) {\n    return Promise.resolve();\n  }\n  // Request path may contain trailing slashes, depending on the original request, so remove\n  // leading and trailing slashes to make it easier to specify paths in the configuration\n  const reqPath = req.path.replace(/^\\/|\\/$/, '');\n  // Determine whether idempotency is enabled for current request path\n  let match = false;\n  for (const path of paths) {\n    // Assume one wants a path to always match from the beginning to prevent any mistakes\n    const regex = new RegExp(path.charAt(0) === '^' ? path : '^' + path);\n    if (reqPath.match(regex)) {\n      match = true;\n      break;\n    }\n  }\n  if (!match) {\n    return Promise.resolve();\n  }\n  // Try to store request\n  const expiryDate = new Date(new Date().setSeconds(new Date().getSeconds() + ttl));\n  return rest\n    .create(config, auth.master(config), '_Idempotency', {\n      reqId: requestId,\n      expire: Parse._encode(expiryDate),\n    })\n    .catch(e => {\n      if (e.code == Parse.Error.DUPLICATE_VALUE) {\n        throw new Parse.Error(Parse.Error.DUPLICATE_REQUEST, 'Duplicate request');\n      }\n      throw e;\n    });\n}\n\nfunction invalidRequest(req, res) {\n  res.status(403);\n  res.end('{\"error\":\"unauthorized\"}');\n}\n\nfunction malformedContext(req, res) {\n  res.status(400);\n  res.json({ code: Parse.Error.INVALID_JSON, error: 'Invalid object for context.' });\n}\n"],"mappings":"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"}
|