parse-server 6.0.0-alpha.2 → 6.0.0-alpha.21
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/README.md +45 -17
- 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 +54 -121
- package/lib/ClientSDK.js +3 -11
- package/lib/Config.js +69 -113
- 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 +147 -345
- 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 +154 -344
- package/lib/Controllers/UserController.js +11 -72
- 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 +92 -306
- 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 +79 -10
- package/lib/Options/docs.js +23 -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 +144 -182
- 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 +212 -377
- 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 +58 -155
- 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 -92
- 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 +147 -151
- package/lib/password.js +6 -10
- package/lib/request.js +173 -2
- package/lib/requiredParameter.js +1 -3
- package/lib/rest.js +19 -41
- package/lib/triggers.js +54 -252
- package/lib/vendor/mongodbUrl.js +125 -305
- package/package.json +22 -19
- package/lib/cloud-code/HTTPResponse.js +0 -73
- package/lib/cloud-code/httpRequest.js +0 -192
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,7 +53,6 @@ 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'),
|
|
@@ -76,25 +66,20 @@ function handleParseHeaders(req, res, next) {
|
|
|
76
66
|
context: context
|
|
77
67
|
};
|
|
78
68
|
var basicAuth = httpAuth(req);
|
|
79
|
-
|
|
80
69
|
if (basicAuth) {
|
|
81
70
|
var basicAuthAppId = basicAuth.appId;
|
|
82
|
-
|
|
83
71
|
if (_cache.default.get(basicAuthAppId)) {
|
|
84
72
|
info.appId = basicAuthAppId;
|
|
85
73
|
info.masterKey = basicAuth.masterKey || info.masterKey;
|
|
86
74
|
info.javascriptKey = basicAuth.javascriptKey || info.javascriptKey;
|
|
87
75
|
}
|
|
88
76
|
}
|
|
89
|
-
|
|
90
77
|
if (req.body) {
|
|
91
78
|
// Unity SDK sends a _noBody key which needs to be removed.
|
|
92
79
|
// Unclear at this point if action needs to be taken.
|
|
93
80
|
delete req.body._noBody;
|
|
94
81
|
}
|
|
95
|
-
|
|
96
82
|
var fileViaJSON = false;
|
|
97
|
-
|
|
98
83
|
if (!info.appId || !_cache.default.get(info.appId)) {
|
|
99
84
|
// See if we can find the app id on the body.
|
|
100
85
|
if (req.body instanceof Buffer) {
|
|
@@ -108,48 +93,40 @@ function handleParseHeaders(req, res, next) {
|
|
|
108
93
|
} catch (e) {
|
|
109
94
|
return invalidRequest(req, res);
|
|
110
95
|
}
|
|
111
|
-
|
|
112
96
|
fileViaJSON = true;
|
|
113
97
|
}
|
|
114
|
-
|
|
115
98
|
if (req.body) {
|
|
116
99
|
delete req.body._RevocableSession;
|
|
117
100
|
}
|
|
118
|
-
|
|
119
101
|
if (req.body && req.body._ApplicationId && _cache.default.get(req.body._ApplicationId) && (!info.masterKey || _cache.default.get(req.body._ApplicationId).masterKey === info.masterKey)) {
|
|
120
102
|
info.appId = req.body._ApplicationId;
|
|
121
103
|
info.javascriptKey = req.body._JavaScriptKey || '';
|
|
122
104
|
delete req.body._ApplicationId;
|
|
123
|
-
delete req.body._JavaScriptKey;
|
|
105
|
+
delete req.body._JavaScriptKey;
|
|
106
|
+
// TODO: test that the REST API formats generated by the other
|
|
124
107
|
// SDKs are handled ok
|
|
125
|
-
|
|
126
108
|
if (req.body._ClientVersion) {
|
|
127
109
|
info.clientVersion = req.body._ClientVersion;
|
|
128
110
|
delete req.body._ClientVersion;
|
|
129
111
|
}
|
|
130
|
-
|
|
131
112
|
if (req.body._InstallationId) {
|
|
132
113
|
info.installationId = req.body._InstallationId;
|
|
133
114
|
delete req.body._InstallationId;
|
|
134
115
|
}
|
|
135
|
-
|
|
136
116
|
if (req.body._SessionToken) {
|
|
137
117
|
info.sessionToken = req.body._SessionToken;
|
|
138
118
|
delete req.body._SessionToken;
|
|
139
119
|
}
|
|
140
|
-
|
|
141
120
|
if (req.body._MasterKey) {
|
|
142
121
|
info.masterKey = req.body._MasterKey;
|
|
143
122
|
delete req.body._MasterKey;
|
|
144
123
|
}
|
|
145
|
-
|
|
146
124
|
if (req.body._context) {
|
|
147
125
|
if (req.body._context instanceof Object) {
|
|
148
126
|
info.context = req.body._context;
|
|
149
127
|
} else {
|
|
150
128
|
try {
|
|
151
129
|
info.context = JSON.parse(req.body._context);
|
|
152
|
-
|
|
153
130
|
if (Object.prototype.toString.call(info.context) !== '[object Object]') {
|
|
154
131
|
throw 'Context is not an object';
|
|
155
132
|
}
|
|
@@ -157,10 +134,8 @@ function handleParseHeaders(req, res, next) {
|
|
|
157
134
|
return malformedContext(req, res);
|
|
158
135
|
}
|
|
159
136
|
}
|
|
160
|
-
|
|
161
137
|
delete req.body._context;
|
|
162
138
|
}
|
|
163
|
-
|
|
164
139
|
if (req.body._ContentType) {
|
|
165
140
|
req.headers['content-type'] = req.body._ContentType;
|
|
166
141
|
delete req.body._ContentType;
|
|
@@ -169,47 +144,49 @@ function handleParseHeaders(req, res, next) {
|
|
|
169
144
|
return invalidRequest(req, res);
|
|
170
145
|
}
|
|
171
146
|
}
|
|
172
|
-
|
|
173
147
|
if (info.sessionToken && typeof info.sessionToken !== 'string') {
|
|
174
148
|
info.sessionToken = info.sessionToken.toString();
|
|
175
149
|
}
|
|
176
|
-
|
|
177
150
|
if (info.clientVersion) {
|
|
178
151
|
info.clientSDK = _ClientSDK.default.fromString(info.clientVersion);
|
|
179
152
|
}
|
|
180
|
-
|
|
181
153
|
if (fileViaJSON) {
|
|
182
|
-
req.fileData = req.body.fileData;
|
|
183
|
-
|
|
154
|
+
req.fileData = req.body.fileData;
|
|
155
|
+
// We need to repopulate req.body with a buffer
|
|
184
156
|
var base64 = req.body.base64;
|
|
185
157
|
req.body = Buffer.from(base64, 'base64');
|
|
186
158
|
}
|
|
187
|
-
|
|
188
159
|
const clientIp = getClientIp(req);
|
|
160
|
+
const config = _Config.default.get(info.appId, mount);
|
|
161
|
+
if (config.state && config.state !== 'ok') {
|
|
162
|
+
res.status(500);
|
|
163
|
+
res.json({
|
|
164
|
+
code: _node.default.Error.INTERNAL_SERVER_ERROR,
|
|
165
|
+
error: `Invalid server state: ${config.state}`
|
|
166
|
+
});
|
|
167
|
+
return;
|
|
168
|
+
}
|
|
189
169
|
info.app = _cache.default.get(info.appId);
|
|
190
|
-
req.config =
|
|
170
|
+
req.config = config;
|
|
191
171
|
req.config.headers = req.headers || {};
|
|
192
172
|
req.config.ip = clientIp;
|
|
193
173
|
req.info = info;
|
|
194
|
-
|
|
195
|
-
if (
|
|
196
|
-
|
|
174
|
+
let isMaster = info.masterKey === req.config.masterKey;
|
|
175
|
+
if (isMaster && !(0, _ipRangeCheck.default)(clientIp, req.config.masterKeyIps || [])) {
|
|
176
|
+
var _req$config;
|
|
177
|
+
const log = ((_req$config = req.config) === null || _req$config === void 0 ? void 0 : _req$config.loggerController) || _logger.default;
|
|
178
|
+
log.error(`Request using master key rejected as the request IP address '${clientIp}' is not set in Parse Server option 'masterKeyIps'.`);
|
|
179
|
+
isMaster = false;
|
|
197
180
|
}
|
|
198
|
-
|
|
199
|
-
var isMaster = info.masterKey === req.config.masterKey;
|
|
200
|
-
|
|
201
181
|
if (isMaster) {
|
|
202
182
|
req.auth = new _Auth.default.Auth({
|
|
203
183
|
config: req.config,
|
|
204
184
|
installationId: info.installationId,
|
|
205
185
|
isMaster: true
|
|
206
186
|
});
|
|
207
|
-
next
|
|
208
|
-
return;
|
|
187
|
+
return handleRateLimit(req, res, next);
|
|
209
188
|
}
|
|
210
|
-
|
|
211
189
|
var isReadOnlyMaster = info.masterKey === req.config.readOnlyMasterKey;
|
|
212
|
-
|
|
213
190
|
if (typeof req.config.readOnlyMasterKey != 'undefined' && req.config.readOnlyMasterKey && isReadOnlyMaster) {
|
|
214
191
|
req.auth = new _Auth.default.Auth({
|
|
215
192
|
config: req.config,
|
|
@@ -217,12 +194,11 @@ function handleParseHeaders(req, res, next) {
|
|
|
217
194
|
isMaster: true,
|
|
218
195
|
isReadOnly: true
|
|
219
196
|
});
|
|
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
|
-
|
|
197
|
+
return handleRateLimit(req, res, next);
|
|
198
|
+
}
|
|
225
199
|
|
|
200
|
+
// Client keys are not required in parse-server, but if any have been configured in the server, validate them
|
|
201
|
+
// to preserve original behavior.
|
|
226
202
|
const keys = ['clientKey', 'javascriptKey', 'dotNetKey', 'restAPIKey'];
|
|
227
203
|
const oneKeyConfigured = keys.some(function (key) {
|
|
228
204
|
return req.config[key] !== undefined;
|
|
@@ -230,15 +206,12 @@ function handleParseHeaders(req, res, next) {
|
|
|
230
206
|
const oneKeyMatches = keys.some(function (key) {
|
|
231
207
|
return req.config[key] !== undefined && info[key] === req.config[key];
|
|
232
208
|
});
|
|
233
|
-
|
|
234
209
|
if (oneKeyConfigured && !oneKeyMatches) {
|
|
235
210
|
return invalidRequest(req, res);
|
|
236
211
|
}
|
|
237
|
-
|
|
238
212
|
if (req.url == '/login') {
|
|
239
213
|
delete info.sessionToken;
|
|
240
214
|
}
|
|
241
|
-
|
|
242
215
|
if (req.userFromJWT) {
|
|
243
216
|
req.auth = new _Auth.default.Auth({
|
|
244
217
|
config: req.config,
|
|
@@ -246,89 +219,93 @@ function handleParseHeaders(req, res, next) {
|
|
|
246
219
|
isMaster: false,
|
|
247
220
|
user: req.userFromJWT
|
|
248
221
|
});
|
|
249
|
-
next
|
|
250
|
-
return;
|
|
222
|
+
return handleRateLimit(req, res, next);
|
|
251
223
|
}
|
|
252
|
-
|
|
253
224
|
if (!info.sessionToken) {
|
|
254
225
|
req.auth = new _Auth.default.Auth({
|
|
255
226
|
config: req.config,
|
|
256
227
|
installationId: info.installationId,
|
|
257
228
|
isMaster: false
|
|
258
229
|
});
|
|
259
|
-
|
|
230
|
+
}
|
|
231
|
+
handleRateLimit(req, res, next);
|
|
232
|
+
}
|
|
233
|
+
const handleRateLimit = async (req, res, next) => {
|
|
234
|
+
const rateLimits = req.config.rateLimits || [];
|
|
235
|
+
try {
|
|
236
|
+
await Promise.all(rateLimits.map(async limit => {
|
|
237
|
+
const pathExp = new RegExp(limit.path);
|
|
238
|
+
if (pathExp.test(req.url)) {
|
|
239
|
+
await limit.handler(req, res, err => {
|
|
240
|
+
if (err) {
|
|
241
|
+
throw err;
|
|
242
|
+
}
|
|
243
|
+
});
|
|
244
|
+
}
|
|
245
|
+
}));
|
|
246
|
+
} catch (error) {
|
|
247
|
+
res.status(429);
|
|
248
|
+
res.json({
|
|
249
|
+
code: _node.default.Error.CONNECTION_FAILED,
|
|
250
|
+
error
|
|
251
|
+
});
|
|
260
252
|
return;
|
|
261
253
|
}
|
|
262
|
-
|
|
263
|
-
|
|
264
|
-
|
|
254
|
+
next();
|
|
255
|
+
};
|
|
256
|
+
const handleParseSession = async (req, res, next) => {
|
|
257
|
+
try {
|
|
258
|
+
const info = req.info;
|
|
259
|
+
if (req.auth) {
|
|
260
|
+
next();
|
|
261
|
+
return;
|
|
262
|
+
}
|
|
263
|
+
let requestAuth = null;
|
|
265
264
|
if (info.sessionToken && req.url === '/upgradeToRevocableSession' && info.sessionToken.indexOf('r:') != 0) {
|
|
266
|
-
|
|
265
|
+
requestAuth = await _Auth.default.getAuthForLegacySessionToken({
|
|
267
266
|
config: req.config,
|
|
268
267
|
installationId: info.installationId,
|
|
269
268
|
sessionToken: info.sessionToken
|
|
270
269
|
});
|
|
271
270
|
} else {
|
|
272
|
-
|
|
271
|
+
requestAuth = await _Auth.default.getAuthForSessionToken({
|
|
273
272
|
config: req.config,
|
|
274
273
|
installationId: info.installationId,
|
|
275
274
|
sessionToken: info.sessionToken
|
|
276
275
|
});
|
|
277
276
|
}
|
|
278
|
-
|
|
279
|
-
|
|
280
|
-
|
|
281
|
-
next();
|
|
282
|
-
}
|
|
283
|
-
}).catch(error => {
|
|
277
|
+
req.auth = requestAuth;
|
|
278
|
+
next();
|
|
279
|
+
} catch (error) {
|
|
284
280
|
if (error instanceof _node.default.Error) {
|
|
285
281
|
next(error);
|
|
286
282
|
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
283
|
}
|
|
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;
|
|
284
|
+
// TODO: Determine the correct error scenario.
|
|
285
|
+
req.config.loggerController.error('error getting auth for sessionToken', error);
|
|
286
|
+
throw new _node.default.Error(_node.default.Error.UNKNOWN_ERROR, error);
|
|
311
287
|
}
|
|
288
|
+
};
|
|
289
|
+
exports.handleParseSession = handleParseSession;
|
|
290
|
+
function getClientIp(req) {
|
|
291
|
+
return req.ip;
|
|
312
292
|
}
|
|
313
|
-
|
|
314
293
|
function httpAuth(req) {
|
|
315
294
|
if (!(req.req || req).headers.authorization) return;
|
|
316
295
|
var header = (req.req || req).headers.authorization;
|
|
317
|
-
var appId, masterKey, javascriptKey;
|
|
296
|
+
var appId, masterKey, javascriptKey;
|
|
318
297
|
|
|
298
|
+
// parse header
|
|
319
299
|
var authPrefix = 'basic ';
|
|
320
300
|
var match = header.toLowerCase().indexOf(authPrefix);
|
|
321
|
-
|
|
322
301
|
if (match == 0) {
|
|
323
302
|
var encodedAuth = header.substring(authPrefix.length, header.length);
|
|
324
303
|
var credentials = decodeBase64(encodedAuth).split(':');
|
|
325
|
-
|
|
326
304
|
if (credentials.length == 2) {
|
|
327
305
|
appId = credentials[0];
|
|
328
306
|
var key = credentials[1];
|
|
329
307
|
var jsKeyPrefix = 'javascript-key=';
|
|
330
308
|
var matchKey = key.indexOf(jsKeyPrefix);
|
|
331
|
-
|
|
332
309
|
if (matchKey == 0) {
|
|
333
310
|
javascriptKey = key.substring(jsKeyPrefix.length, key.length);
|
|
334
311
|
} else {
|
|
@@ -336,34 +313,28 @@ function httpAuth(req) {
|
|
|
336
313
|
}
|
|
337
314
|
}
|
|
338
315
|
}
|
|
339
|
-
|
|
340
316
|
return {
|
|
341
317
|
appId: appId,
|
|
342
318
|
masterKey: masterKey,
|
|
343
319
|
javascriptKey: javascriptKey
|
|
344
320
|
};
|
|
345
321
|
}
|
|
346
|
-
|
|
347
322
|
function decodeBase64(str) {
|
|
348
323
|
return Buffer.from(str, 'base64').toString();
|
|
349
324
|
}
|
|
350
|
-
|
|
351
325
|
function allowCrossDomain(appId) {
|
|
352
326
|
return (req, res, next) => {
|
|
353
327
|
const config = _Config.default.get(appId, getMountForRequest(req));
|
|
354
|
-
|
|
355
328
|
let allowHeaders = DEFAULT_ALLOWED_HEADERS;
|
|
356
|
-
|
|
357
329
|
if (config && config.allowHeaders) {
|
|
358
330
|
allowHeaders += `, ${config.allowHeaders.join(', ')}`;
|
|
359
331
|
}
|
|
360
|
-
|
|
361
332
|
const allowOrigin = config && config.allowOrigin || '*';
|
|
362
333
|
res.header('Access-Control-Allow-Origin', allowOrigin);
|
|
363
334
|
res.header('Access-Control-Allow-Methods', 'GET,PUT,POST,DELETE,OPTIONS');
|
|
364
335
|
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
|
-
|
|
336
|
+
res.header('Access-Control-Expose-Headers', 'X-Parse-Job-Status-Id, X-Parse-Push-Status-Id');
|
|
337
|
+
// intercept OPTIONS method
|
|
367
338
|
if ('OPTIONS' == req.method) {
|
|
368
339
|
res.sendStatus(200);
|
|
369
340
|
} else {
|
|
@@ -371,40 +342,32 @@ function allowCrossDomain(appId) {
|
|
|
371
342
|
}
|
|
372
343
|
};
|
|
373
344
|
}
|
|
374
|
-
|
|
375
345
|
function allowMethodOverride(req, res, next) {
|
|
376
346
|
if (req.method === 'POST' && req.body._method) {
|
|
377
347
|
req.originalMethod = req.method;
|
|
378
348
|
req.method = req.body._method;
|
|
379
349
|
delete req.body._method;
|
|
380
350
|
}
|
|
381
|
-
|
|
382
351
|
next();
|
|
383
352
|
}
|
|
384
|
-
|
|
385
353
|
function handleParseErrors(err, req, res, next) {
|
|
386
354
|
const log = req.config && req.config.loggerController || _logger.default;
|
|
387
|
-
|
|
388
355
|
if (err instanceof _node.default.Error) {
|
|
389
356
|
if (req.config && req.config.enableExpressErrorHandler) {
|
|
390
357
|
return next(err);
|
|
391
358
|
}
|
|
392
|
-
|
|
393
|
-
|
|
394
|
-
|
|
359
|
+
let httpStatus;
|
|
360
|
+
// TODO: fill out this mapping
|
|
395
361
|
switch (err.code) {
|
|
396
362
|
case _node.default.Error.INTERNAL_SERVER_ERROR:
|
|
397
363
|
httpStatus = 500;
|
|
398
364
|
break;
|
|
399
|
-
|
|
400
365
|
case _node.default.Error.OBJECT_NOT_FOUND:
|
|
401
366
|
httpStatus = 404;
|
|
402
367
|
break;
|
|
403
|
-
|
|
404
368
|
default:
|
|
405
369
|
httpStatus = 400;
|
|
406
370
|
}
|
|
407
|
-
|
|
408
371
|
res.status(httpStatus);
|
|
409
372
|
res.json({
|
|
410
373
|
code: err.code,
|
|
@@ -416,7 +379,6 @@ function handleParseErrors(err, req, res, next) {
|
|
|
416
379
|
res.json({
|
|
417
380
|
error: err.message
|
|
418
381
|
});
|
|
419
|
-
|
|
420
382
|
if (!(process && process.env.TESTING)) {
|
|
421
383
|
next(err);
|
|
422
384
|
}
|
|
@@ -427,23 +389,19 @@ function handleParseErrors(err, req, res, next) {
|
|
|
427
389
|
code: _node.default.Error.INTERNAL_SERVER_ERROR,
|
|
428
390
|
message: 'Internal server error.'
|
|
429
391
|
});
|
|
430
|
-
|
|
431
392
|
if (!(process && process.env.TESTING)) {
|
|
432
393
|
next(err);
|
|
433
394
|
}
|
|
434
395
|
}
|
|
435
396
|
}
|
|
436
|
-
|
|
437
397
|
function enforceMasterKeyAccess(req, res, next) {
|
|
438
398
|
if (!req.auth.isMaster) {
|
|
439
399
|
res.status(403);
|
|
440
400
|
res.end('{"error":"unauthorized: master key is required"}');
|
|
441
401
|
return;
|
|
442
402
|
}
|
|
443
|
-
|
|
444
403
|
next();
|
|
445
404
|
}
|
|
446
|
-
|
|
447
405
|
function promiseEnforceMasterKeyAccess(request) {
|
|
448
406
|
if (!request.auth.isMaster) {
|
|
449
407
|
const error = new Error();
|
|
@@ -451,56 +409,97 @@ function promiseEnforceMasterKeyAccess(request) {
|
|
|
451
409
|
error.message = 'unauthorized: master key is required';
|
|
452
410
|
throw error;
|
|
453
411
|
}
|
|
454
|
-
|
|
455
412
|
return Promise.resolve();
|
|
456
413
|
}
|
|
414
|
+
const addRateLimit = (route, config) => {
|
|
415
|
+
if (typeof config === 'string') {
|
|
416
|
+
config = _Config.default.get(config);
|
|
417
|
+
}
|
|
418
|
+
for (const key in route) {
|
|
419
|
+
if (!_Definitions.RateLimitOptions[key]) {
|
|
420
|
+
throw `Invalid rate limit option "${key}"`;
|
|
421
|
+
}
|
|
422
|
+
}
|
|
423
|
+
if (!config.rateLimits) {
|
|
424
|
+
config.rateLimits = [];
|
|
425
|
+
}
|
|
426
|
+
config.rateLimits.push({
|
|
427
|
+
path: (0, _pathToRegexp.default)(route.requestPath),
|
|
428
|
+
handler: (0, _expressRateLimit.default)({
|
|
429
|
+
windowMs: route.requestTimeWindow,
|
|
430
|
+
max: route.requestCount,
|
|
431
|
+
message: route.errorResponseMessage || _Definitions.RateLimitOptions.errorResponseMessage.default,
|
|
432
|
+
handler: (request, response, next, options) => {
|
|
433
|
+
throw options.message;
|
|
434
|
+
},
|
|
435
|
+
skip: request => {
|
|
436
|
+
if (request.ip === '127.0.0.1' && !route.includeInternalRequests) {
|
|
437
|
+
return true;
|
|
438
|
+
}
|
|
439
|
+
if (route.includeMasterKey) {
|
|
440
|
+
return false;
|
|
441
|
+
}
|
|
442
|
+
if (route.requestMethods) {
|
|
443
|
+
if (Array.isArray(route.requestMethods)) {
|
|
444
|
+
if (!route.requestMethods.includes(request.method)) {
|
|
445
|
+
return true;
|
|
446
|
+
}
|
|
447
|
+
} else {
|
|
448
|
+
const regExp = new RegExp(route.requestMethods);
|
|
449
|
+
if (!regExp.test(request.method)) {
|
|
450
|
+
return true;
|
|
451
|
+
}
|
|
452
|
+
}
|
|
453
|
+
}
|
|
454
|
+
return request.auth.isMaster;
|
|
455
|
+
},
|
|
456
|
+
keyGenerator: request => {
|
|
457
|
+
return request.config.ip;
|
|
458
|
+
}
|
|
459
|
+
})
|
|
460
|
+
});
|
|
461
|
+
_Config.default.put(config);
|
|
462
|
+
};
|
|
463
|
+
|
|
457
464
|
/**
|
|
458
465
|
* Deduplicates a request to ensure idempotency. Duplicates are determined by the request ID
|
|
459
466
|
* in the request header. If a request has no request ID, it is executed anyway.
|
|
460
467
|
* @param {*} req The request to evaluate.
|
|
461
468
|
* @returns Promise<{}>
|
|
462
469
|
*/
|
|
463
|
-
|
|
464
|
-
|
|
470
|
+
exports.addRateLimit = addRateLimit;
|
|
465
471
|
function promiseEnsureIdempotency(req) {
|
|
466
472
|
// Enable feature only for MongoDB
|
|
467
473
|
if (!(req.config.database.adapter instanceof _MongoStorageAdapter.default || req.config.database.adapter instanceof _PostgresStorageAdapter.default)) {
|
|
468
474
|
return Promise.resolve();
|
|
469
|
-
}
|
|
470
|
-
|
|
471
|
-
|
|
475
|
+
}
|
|
476
|
+
// Get parameters
|
|
472
477
|
const config = req.config;
|
|
473
478
|
const requestId = ((req || {}).headers || {})['x-parse-request-id'];
|
|
474
479
|
const {
|
|
475
480
|
paths,
|
|
476
481
|
ttl
|
|
477
482
|
} = config.idempotencyOptions;
|
|
478
|
-
|
|
479
483
|
if (!requestId || !config.idempotencyOptions) {
|
|
480
484
|
return Promise.resolve();
|
|
481
|
-
}
|
|
485
|
+
}
|
|
486
|
+
// Request path may contain trailing slashes, depending on the original request, so remove
|
|
482
487
|
// 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
|
-
|
|
488
|
+
const reqPath = req.path.replace(/^\/|\/$/, '');
|
|
489
|
+
// Determine whether idempotency is enabled for current request path
|
|
487
490
|
let match = false;
|
|
488
|
-
|
|
489
491
|
for (const path of paths) {
|
|
490
492
|
// Assume one wants a path to always match from the beginning to prevent any mistakes
|
|
491
493
|
const regex = new RegExp(path.charAt(0) === '^' ? path : '^' + path);
|
|
492
|
-
|
|
493
494
|
if (reqPath.match(regex)) {
|
|
494
495
|
match = true;
|
|
495
496
|
break;
|
|
496
497
|
}
|
|
497
498
|
}
|
|
498
|
-
|
|
499
499
|
if (!match) {
|
|
500
500
|
return Promise.resolve();
|
|
501
|
-
}
|
|
502
|
-
|
|
503
|
-
|
|
501
|
+
}
|
|
502
|
+
// Try to store request
|
|
504
503
|
const expiryDate = new Date(new Date().setSeconds(new Date().getSeconds() + ttl));
|
|
505
504
|
return _rest.default.create(config, _Auth.default.master(config), '_Idempotency', {
|
|
506
505
|
reqId: requestId,
|
|
@@ -509,16 +508,13 @@ function promiseEnsureIdempotency(req) {
|
|
|
509
508
|
if (e.code == _node.default.Error.DUPLICATE_VALUE) {
|
|
510
509
|
throw new _node.default.Error(_node.default.Error.DUPLICATE_REQUEST, 'Duplicate request');
|
|
511
510
|
}
|
|
512
|
-
|
|
513
511
|
throw e;
|
|
514
512
|
});
|
|
515
513
|
}
|
|
516
|
-
|
|
517
514
|
function invalidRequest(req, res) {
|
|
518
515
|
res.status(403);
|
|
519
516
|
res.end('{"error":"unauthorized"}');
|
|
520
517
|
}
|
|
521
|
-
|
|
522
518
|
function malformedContext(req, res) {
|
|
523
519
|
res.status(400);
|
|
524
520
|
res.json({
|
|
@@ -526,4 +522,4 @@ function malformedContext(req, res) {
|
|
|
526
522
|
error: 'Invalid object for context.'
|
|
527
523
|
});
|
|
528
524
|
}
|
|
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"]}
|
|
525
|
+
//# 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","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","isMaster","ipRangeCheck","masterKeyIps","log","loggerController","defaultLogger","auth","Auth","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","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","message","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    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  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              throw err;\n            }\n          });\n        }\n      })\n    );\n  } catch (error) {\n    res.status(429);\n    res.json({ code: Parse.Error.CONNECTION_FAILED, error });\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 options.message;\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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"}
|