parse-server 2.8.4 → 8.6.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +167 -25
- package/NOTICE +10 -0
- package/README.md +929 -278
- package/lib/AccountLockout.js +47 -30
- package/lib/Adapters/AdapterLoader.js +21 -6
- package/lib/Adapters/Analytics/AnalyticsAdapter.js +15 -12
- package/lib/Adapters/Auth/AuthAdapter.js +116 -13
- package/lib/Adapters/Auth/BaseCodeAuthAdapter.js +99 -0
- package/lib/Adapters/Auth/OAuth1Client.js +27 -46
- package/lib/Adapters/Auth/apple.js +123 -0
- package/lib/Adapters/Auth/facebook.js +162 -35
- package/lib/Adapters/Auth/gcenter.js +217 -0
- package/lib/Adapters/Auth/github.js +118 -48
- package/lib/Adapters/Auth/google.js +160 -51
- package/lib/Adapters/Auth/gpgames.js +125 -0
- package/lib/Adapters/Auth/httpsRequest.js +6 -7
- package/lib/Adapters/Auth/index.js +170 -62
- package/lib/Adapters/Auth/instagram.js +114 -40
- package/lib/Adapters/Auth/janraincapture.js +52 -23
- package/lib/Adapters/Auth/janrainengage.js +19 -36
- package/lib/Adapters/Auth/keycloak.js +148 -0
- package/lib/Adapters/Auth/ldap.js +167 -0
- package/lib/Adapters/Auth/line.js +125 -0
- package/lib/Adapters/Auth/linkedin.js +111 -55
- package/lib/Adapters/Auth/meetup.js +24 -34
- package/lib/Adapters/Auth/mfa.js +324 -0
- package/lib/Adapters/Auth/microsoft.js +111 -0
- package/lib/Adapters/Auth/oauth2.js +97 -162
- package/lib/Adapters/Auth/phantauth.js +53 -0
- package/lib/Adapters/Auth/qq.js +108 -49
- package/lib/Adapters/Auth/spotify.js +107 -55
- package/lib/Adapters/Auth/twitter.js +188 -48
- package/lib/Adapters/Auth/utils.js +28 -0
- package/lib/Adapters/Auth/vkontakte.js +26 -39
- package/lib/Adapters/Auth/wechat.js +106 -44
- package/lib/Adapters/Auth/weibo.js +132 -58
- package/lib/Adapters/Cache/CacheAdapter.js +13 -8
- package/lib/Adapters/Cache/InMemoryCache.js +3 -13
- package/lib/Adapters/Cache/InMemoryCacheAdapter.js +5 -13
- package/lib/Adapters/Cache/LRUCache.js +13 -27
- package/lib/Adapters/Cache/NullCacheAdapter.js +3 -8
- package/lib/Adapters/Cache/RedisCacheAdapter.js +85 -76
- package/lib/Adapters/Cache/SchemaCache.js +25 -0
- package/lib/Adapters/Email/MailAdapter.js +10 -8
- package/lib/Adapters/Files/FilesAdapter.js +83 -25
- package/lib/Adapters/Files/GridFSBucketAdapter.js +231 -0
- package/lib/Adapters/Files/GridStoreAdapter.js +4 -91
- package/lib/Adapters/Logger/LoggerAdapter.js +18 -14
- package/lib/Adapters/Logger/WinstonLogger.js +69 -88
- package/lib/Adapters/Logger/WinstonLoggerAdapter.js +7 -16
- package/lib/Adapters/MessageQueue/EventEmitterMQ.js +8 -26
- package/lib/Adapters/PubSub/EventEmitterPubSub.js +12 -25
- package/lib/Adapters/PubSub/PubSubAdapter.js +34 -0
- package/lib/Adapters/PubSub/RedisPubSub.js +42 -19
- package/lib/Adapters/Push/PushAdapter.js +14 -7
- package/lib/Adapters/Storage/Mongo/MongoCollection.js +137 -45
- package/lib/Adapters/Storage/Mongo/MongoSchemaCollection.js +158 -63
- package/lib/Adapters/Storage/Mongo/MongoStorageAdapter.js +320 -168
- package/lib/Adapters/Storage/Mongo/MongoTransform.js +279 -306
- package/lib/Adapters/Storage/Postgres/PostgresClient.js +14 -10
- package/lib/Adapters/Storage/Postgres/PostgresConfigParser.js +47 -21
- package/lib/Adapters/Storage/Postgres/PostgresStorageAdapter.js +854 -468
- package/lib/Adapters/Storage/Postgres/sql/index.js +4 -6
- package/lib/Adapters/Storage/StorageAdapter.js +1 -1
- package/lib/Adapters/WebSocketServer/WSAdapter.js +35 -0
- package/lib/Adapters/WebSocketServer/WSSAdapter.js +66 -0
- package/lib/Auth.js +488 -125
- package/lib/ClientSDK.js +2 -6
- package/lib/Config.js +525 -94
- package/lib/Controllers/AdaptableController.js +5 -25
- package/lib/Controllers/AnalyticsController.js +22 -23
- package/lib/Controllers/CacheController.js +10 -31
- package/lib/Controllers/DatabaseController.js +767 -313
- package/lib/Controllers/FilesController.js +49 -54
- package/lib/Controllers/HooksController.js +80 -84
- package/lib/Controllers/LiveQueryController.js +35 -22
- package/lib/Controllers/LoggerController.js +22 -58
- package/lib/Controllers/ParseGraphQLController.js +293 -0
- package/lib/Controllers/PushController.js +58 -49
- package/lib/Controllers/SchemaController.js +916 -422
- package/lib/Controllers/UserController.js +265 -180
- package/lib/Controllers/index.js +90 -125
- package/lib/Controllers/types.js +1 -1
- package/lib/Deprecator/Deprecations.js +30 -0
- package/lib/Deprecator/Deprecator.js +127 -0
- package/lib/Error.js +48 -0
- package/lib/GraphQL/ParseGraphQLSchema.js +375 -0
- package/lib/GraphQL/ParseGraphQLServer.js +214 -0
- package/lib/GraphQL/helpers/objectsMutations.js +30 -0
- package/lib/GraphQL/helpers/objectsQueries.js +246 -0
- package/lib/GraphQL/loaders/configMutations.js +87 -0
- package/lib/GraphQL/loaders/configQueries.js +79 -0
- package/lib/GraphQL/loaders/defaultGraphQLMutations.js +21 -0
- package/lib/GraphQL/loaders/defaultGraphQLQueries.js +23 -0
- package/lib/GraphQL/loaders/defaultGraphQLTypes.js +1098 -0
- package/lib/GraphQL/loaders/defaultRelaySchema.js +53 -0
- package/lib/GraphQL/loaders/filesMutations.js +107 -0
- package/lib/GraphQL/loaders/functionsMutations.js +78 -0
- package/lib/GraphQL/loaders/parseClassMutations.js +268 -0
- package/lib/GraphQL/loaders/parseClassQueries.js +127 -0
- package/lib/GraphQL/loaders/parseClassTypes.js +493 -0
- package/lib/GraphQL/loaders/schemaDirectives.js +62 -0
- package/lib/GraphQL/loaders/schemaMutations.js +162 -0
- package/lib/GraphQL/loaders/schemaQueries.js +81 -0
- package/lib/GraphQL/loaders/schemaTypes.js +341 -0
- package/lib/GraphQL/loaders/usersMutations.js +433 -0
- package/lib/GraphQL/loaders/usersQueries.js +90 -0
- package/lib/GraphQL/parseGraphQLUtils.js +63 -0
- package/lib/GraphQL/transformers/className.js +14 -0
- package/lib/GraphQL/transformers/constraintType.js +53 -0
- package/lib/GraphQL/transformers/inputType.js +51 -0
- package/lib/GraphQL/transformers/mutation.js +274 -0
- package/lib/GraphQL/transformers/outputType.js +51 -0
- package/lib/GraphQL/transformers/query.js +237 -0
- package/lib/GraphQL/transformers/schemaFields.js +99 -0
- package/lib/KeyPromiseQueue.js +48 -0
- package/lib/LiveQuery/Client.js +25 -33
- package/lib/LiveQuery/Id.js +2 -5
- package/lib/LiveQuery/ParseCloudCodePublisher.js +26 -23
- package/lib/LiveQuery/ParseLiveQueryServer.js +560 -285
- package/lib/LiveQuery/ParsePubSub.js +7 -16
- package/lib/LiveQuery/ParseWebSocketServer.js +42 -39
- package/lib/LiveQuery/QueryTools.js +76 -15
- package/lib/LiveQuery/RequestSchema.js +111 -97
- package/lib/LiveQuery/SessionTokenCache.js +23 -36
- package/lib/LiveQuery/Subscription.js +8 -17
- package/lib/LiveQuery/equalObjects.js +2 -3
- package/lib/Options/Definitions.js +1355 -382
- package/lib/Options/docs.js +301 -62
- package/lib/Options/index.js +11 -1
- package/lib/Options/parsers.js +14 -10
- package/lib/Page.js +44 -0
- package/lib/ParseMessageQueue.js +6 -13
- package/lib/ParseServer.js +474 -235
- package/lib/ParseServerRESTController.js +102 -40
- package/lib/PromiseRouter.js +39 -50
- package/lib/Push/PushQueue.js +24 -30
- package/lib/Push/PushWorker.js +32 -56
- package/lib/Push/utils.js +22 -35
- package/lib/RestQuery.js +361 -139
- package/lib/RestWrite.js +713 -344
- package/lib/Routers/AggregateRouter.js +97 -71
- package/lib/Routers/AnalyticsRouter.js +8 -14
- package/lib/Routers/AudiencesRouter.js +16 -35
- package/lib/Routers/ClassesRouter.js +86 -72
- package/lib/Routers/CloudCodeRouter.js +28 -37
- package/lib/Routers/FeaturesRouter.js +22 -25
- package/lib/Routers/FilesRouter.js +266 -171
- package/lib/Routers/FunctionsRouter.js +87 -103
- package/lib/Routers/GlobalConfigRouter.js +94 -33
- package/lib/Routers/GraphQLRouter.js +41 -0
- package/lib/Routers/HooksRouter.js +43 -47
- package/lib/Routers/IAPValidationRouter.js +57 -70
- package/lib/Routers/InstallationsRouter.js +17 -25
- package/lib/Routers/LogsRouter.js +10 -25
- package/lib/Routers/PagesRouter.js +647 -0
- package/lib/Routers/PublicAPIRouter.js +104 -112
- package/lib/Routers/PurgeRouter.js +19 -29
- package/lib/Routers/PushRouter.js +14 -28
- package/lib/Routers/RolesRouter.js +7 -14
- package/lib/Routers/SchemasRouter.js +63 -42
- package/lib/Routers/SecurityRouter.js +34 -0
- package/lib/Routers/SessionsRouter.js +25 -38
- package/lib/Routers/UsersRouter.js +463 -190
- package/lib/SchemaMigrations/DefinedSchemas.js +379 -0
- package/lib/SchemaMigrations/Migrations.js +30 -0
- package/lib/Security/Check.js +109 -0
- package/lib/Security/CheckGroup.js +44 -0
- package/lib/Security/CheckGroups/CheckGroupDatabase.js +44 -0
- package/lib/Security/CheckGroups/CheckGroupServerConfig.js +96 -0
- package/lib/Security/CheckGroups/CheckGroups.js +21 -0
- package/lib/Security/CheckRunner.js +213 -0
- package/lib/SharedRest.js +29 -0
- package/lib/StatusHandler.js +96 -93
- package/lib/TestUtils.js +70 -14
- package/lib/Utils.js +468 -0
- package/lib/batch.js +74 -40
- package/lib/cache.js +8 -8
- package/lib/cli/definitions/parse-live-query-server.js +4 -3
- package/lib/cli/definitions/parse-server.js +4 -3
- package/lib/cli/parse-live-query-server.js +9 -17
- package/lib/cli/parse-server.js +49 -47
- package/lib/cli/utils/commander.js +20 -29
- package/lib/cli/utils/runner.js +31 -32
- package/lib/cloud-code/Parse.Cloud.js +711 -36
- package/lib/cloud-code/Parse.Server.js +21 -0
- package/lib/cryptoUtils.js +6 -11
- package/lib/defaults.js +21 -15
- package/lib/deprecated.js +1 -1
- package/lib/index.js +78 -67
- package/lib/logger.js +12 -20
- package/lib/middlewares.js +484 -160
- package/lib/password.js +10 -6
- package/lib/request.js +175 -0
- package/lib/requiredParameter.js +4 -3
- package/lib/rest.js +157 -82
- package/lib/triggers.js +627 -185
- package/lib/vendor/README.md +3 -3
- package/lib/vendor/mongodbUrl.js +224 -137
- package/package.json +135 -57
- package/postinstall.js +38 -50
- package/public_html/invalid_verification_link.html +3 -3
- package/types/@types/@parse/fs-files-adapter/index.d.ts +5 -0
- package/types/@types/deepcopy/index.d.ts +5 -0
- package/types/LiveQuery/ParseLiveQueryServer.d.ts +40 -0
- package/types/Options/index.d.ts +301 -0
- package/types/ParseServer.d.ts +65 -0
- package/types/eslint.config.mjs +30 -0
- package/types/index.d.ts +21 -0
- package/types/logger.d.ts +2 -0
- package/types/tests.ts +44 -0
- package/types/tsconfig.json +24 -0
- package/CHANGELOG.md +0 -1246
- package/PATENTS +0 -37
- package/bin/dev +0 -37
- package/lib/.DS_Store +0 -0
- package/lib/Adapters/Auth/common.js +0 -2
- package/lib/Adapters/Auth/facebookaccountkit.js +0 -69
- package/lib/Controllers/SchemaCache.js +0 -97
- package/lib/LiveQuery/.DS_Store +0 -0
- package/lib/cli/utils/parsers.js +0 -77
- package/lib/cloud-code/.DS_Store +0 -0
- package/lib/cloud-code/HTTPResponse.js +0 -57
- package/lib/cloud-code/Untitled-1 +0 -123
- package/lib/cloud-code/httpRequest.js +0 -102
- package/lib/cloud-code/team.html +0 -123
- package/lib/graphql/ParseClass.js +0 -234
- package/lib/graphql/Schema.js +0 -197
- package/lib/graphql/index.js +0 -1
- package/lib/graphql/types/ACL.js +0 -35
- package/lib/graphql/types/Date.js +0 -25
- package/lib/graphql/types/File.js +0 -24
- package/lib/graphql/types/GeoPoint.js +0 -35
- package/lib/graphql/types/JSONObject.js +0 -30
- package/lib/graphql/types/NumberInput.js +0 -43
- package/lib/graphql/types/NumberQuery.js +0 -42
- package/lib/graphql/types/Pointer.js +0 -35
- package/lib/graphql/types/QueryConstraint.js +0 -61
- package/lib/graphql/types/StringQuery.js +0 -39
- package/lib/graphql/types/index.js +0 -110
package/lib/Auth.js
CHANGED
|
@@ -1,16 +1,30 @@
|
|
|
1
|
-
|
|
1
|
+
"use strict";
|
|
2
2
|
|
|
3
|
-
|
|
4
|
-
|
|
3
|
+
var _util = require("util");
|
|
4
|
+
var _triggers = require("./triggers");
|
|
5
|
+
var _logger = require("./logger");
|
|
6
|
+
var _lruCache = require("lru-cache");
|
|
7
|
+
var _RestQuery = _interopRequireDefault(require("./RestQuery"));
|
|
8
|
+
var _RestWrite = _interopRequireDefault(require("./RestWrite"));
|
|
9
|
+
function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
|
|
5
10
|
const Parse = require('parse/node');
|
|
6
|
-
|
|
7
11
|
// An Auth object tells you who is requesting something and whether
|
|
8
12
|
// the master key was used.
|
|
9
13
|
// userObject is a Parse.User and can be null if there's no user.
|
|
10
|
-
function Auth({
|
|
14
|
+
function Auth({
|
|
15
|
+
config,
|
|
16
|
+
cacheController = undefined,
|
|
17
|
+
isMaster = false,
|
|
18
|
+
isMaintenance = false,
|
|
19
|
+
isReadOnly = false,
|
|
20
|
+
user,
|
|
21
|
+
installationId
|
|
22
|
+
}) {
|
|
11
23
|
this.config = config;
|
|
24
|
+
this.cacheController = cacheController || config && config.cacheController;
|
|
12
25
|
this.installationId = installationId;
|
|
13
26
|
this.isMaster = isMaster;
|
|
27
|
+
this.isMaintenance = isMaintenance;
|
|
14
28
|
this.user = user;
|
|
15
29
|
this.isReadOnly = isReadOnly;
|
|
16
30
|
|
|
@@ -27,6 +41,9 @@ Auth.prototype.isUnauthenticated = function () {
|
|
|
27
41
|
if (this.isMaster) {
|
|
28
42
|
return false;
|
|
29
43
|
}
|
|
44
|
+
if (this.isMaintenance) {
|
|
45
|
+
return false;
|
|
46
|
+
}
|
|
30
47
|
if (this.user) {
|
|
31
48
|
return false;
|
|
32
49
|
}
|
|
@@ -35,60 +52,200 @@ Auth.prototype.isUnauthenticated = function () {
|
|
|
35
52
|
|
|
36
53
|
// A helper to get a master-level Auth object
|
|
37
54
|
function master(config) {
|
|
38
|
-
return new Auth({
|
|
55
|
+
return new Auth({
|
|
56
|
+
config,
|
|
57
|
+
isMaster: true
|
|
58
|
+
});
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
// A helper to get a maintenance-level Auth object
|
|
62
|
+
function maintenance(config) {
|
|
63
|
+
return new Auth({
|
|
64
|
+
config,
|
|
65
|
+
isMaintenance: true
|
|
66
|
+
});
|
|
39
67
|
}
|
|
40
68
|
|
|
41
69
|
// A helper to get a master-level Auth object
|
|
42
70
|
function readOnly(config) {
|
|
43
|
-
return new Auth({
|
|
71
|
+
return new Auth({
|
|
72
|
+
config,
|
|
73
|
+
isMaster: true,
|
|
74
|
+
isReadOnly: true
|
|
75
|
+
});
|
|
44
76
|
}
|
|
45
77
|
|
|
46
78
|
// A helper to get a nobody-level Auth object
|
|
47
79
|
function nobody(config) {
|
|
48
|
-
return new Auth({
|
|
80
|
+
return new Auth({
|
|
81
|
+
config,
|
|
82
|
+
isMaster: false
|
|
83
|
+
});
|
|
84
|
+
}
|
|
85
|
+
const throttle = new _lruCache.LRUCache({
|
|
86
|
+
max: 10000,
|
|
87
|
+
ttl: 500
|
|
88
|
+
});
|
|
89
|
+
/**
|
|
90
|
+
* Checks whether session should be updated based on last update time & session length.
|
|
91
|
+
*/
|
|
92
|
+
function shouldUpdateSessionExpiry(config, session) {
|
|
93
|
+
const resetAfter = config.sessionLength / 2;
|
|
94
|
+
const lastUpdated = new Date(session?.updatedAt);
|
|
95
|
+
const skipRange = new Date();
|
|
96
|
+
skipRange.setTime(skipRange.getTime() - resetAfter * 1000);
|
|
97
|
+
return lastUpdated <= skipRange;
|
|
49
98
|
}
|
|
99
|
+
const renewSessionIfNeeded = async ({
|
|
100
|
+
config,
|
|
101
|
+
session,
|
|
102
|
+
sessionToken
|
|
103
|
+
}) => {
|
|
104
|
+
if (!config?.extendSessionOnUse) {
|
|
105
|
+
return;
|
|
106
|
+
}
|
|
107
|
+
if (throttle.get(sessionToken)) {
|
|
108
|
+
return;
|
|
109
|
+
}
|
|
110
|
+
throttle.set(sessionToken, true);
|
|
111
|
+
try {
|
|
112
|
+
if (!session) {
|
|
113
|
+
const query = await (0, _RestQuery.default)({
|
|
114
|
+
method: _RestQuery.default.Method.get,
|
|
115
|
+
config,
|
|
116
|
+
auth: master(config),
|
|
117
|
+
runBeforeFind: false,
|
|
118
|
+
className: '_Session',
|
|
119
|
+
restWhere: {
|
|
120
|
+
sessionToken
|
|
121
|
+
},
|
|
122
|
+
restOptions: {
|
|
123
|
+
limit: 1
|
|
124
|
+
}
|
|
125
|
+
});
|
|
126
|
+
const {
|
|
127
|
+
results
|
|
128
|
+
} = await query.execute();
|
|
129
|
+
session = results[0];
|
|
130
|
+
}
|
|
131
|
+
if (!shouldUpdateSessionExpiry(config, session) || !session) {
|
|
132
|
+
return;
|
|
133
|
+
}
|
|
134
|
+
const expiresAt = config.generateSessionExpiresAt();
|
|
135
|
+
await new _RestWrite.default(config, master(config), '_Session', {
|
|
136
|
+
objectId: session.objectId
|
|
137
|
+
}, {
|
|
138
|
+
expiresAt: Parse._encode(expiresAt)
|
|
139
|
+
}).execute();
|
|
140
|
+
} catch (e) {
|
|
141
|
+
if (e?.code !== Parse.Error.OBJECT_NOT_FOUND) {
|
|
142
|
+
_logger.logger.error('Could not update session expiry: ', e);
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
};
|
|
50
146
|
|
|
51
147
|
// Returns a promise that resolves to an Auth object
|
|
52
|
-
|
|
53
|
-
|
|
148
|
+
const getAuthForSessionToken = async function ({
|
|
149
|
+
config,
|
|
150
|
+
cacheController,
|
|
151
|
+
sessionToken,
|
|
152
|
+
installationId
|
|
153
|
+
}) {
|
|
154
|
+
cacheController = cacheController || config && config.cacheController;
|
|
155
|
+
if (cacheController) {
|
|
156
|
+
const userJSON = await cacheController.user.get(sessionToken);
|
|
54
157
|
if (userJSON) {
|
|
55
158
|
const cachedUser = Parse.Object.fromJSON(userJSON);
|
|
56
|
-
|
|
159
|
+
renewSessionIfNeeded({
|
|
160
|
+
config,
|
|
161
|
+
sessionToken
|
|
162
|
+
});
|
|
163
|
+
return Promise.resolve(new Auth({
|
|
164
|
+
config,
|
|
165
|
+
cacheController,
|
|
166
|
+
isMaster: false,
|
|
167
|
+
installationId,
|
|
168
|
+
user: cachedUser
|
|
169
|
+
}));
|
|
57
170
|
}
|
|
58
|
-
|
|
59
|
-
|
|
171
|
+
}
|
|
172
|
+
let results;
|
|
173
|
+
if (config) {
|
|
174
|
+
const restOptions = {
|
|
60
175
|
limit: 1,
|
|
61
176
|
include: 'user'
|
|
62
177
|
};
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'Session token is expired.');
|
|
75
|
-
}
|
|
76
|
-
var obj = results[0]['user'];
|
|
77
|
-
delete obj.password;
|
|
78
|
-
obj['className'] = '_User';
|
|
79
|
-
obj['sessionToken'] = sessionToken;
|
|
80
|
-
config.cacheController.user.put(sessionToken, obj);
|
|
81
|
-
const userObject = Parse.Object.fromJSON(obj);
|
|
82
|
-
return new Auth({ config, isMaster: false, installationId, user: userObject });
|
|
178
|
+
const RestQuery = require('./RestQuery');
|
|
179
|
+
const query = await RestQuery({
|
|
180
|
+
method: RestQuery.Method.get,
|
|
181
|
+
config,
|
|
182
|
+
runBeforeFind: false,
|
|
183
|
+
auth: master(config),
|
|
184
|
+
className: '_Session',
|
|
185
|
+
restWhere: {
|
|
186
|
+
sessionToken
|
|
187
|
+
},
|
|
188
|
+
restOptions
|
|
83
189
|
});
|
|
190
|
+
results = (await query.execute()).results;
|
|
191
|
+
} else {
|
|
192
|
+
results = (await new Parse.Query(Parse.Session).limit(1).include('user').equalTo('sessionToken', sessionToken).find({
|
|
193
|
+
useMasterKey: true
|
|
194
|
+
})).map(obj => obj.toJSON());
|
|
195
|
+
}
|
|
196
|
+
if (results.length !== 1 || !results[0]['user']) {
|
|
197
|
+
throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'Invalid session token');
|
|
198
|
+
}
|
|
199
|
+
const session = results[0];
|
|
200
|
+
const now = new Date(),
|
|
201
|
+
expiresAt = session.expiresAt ? new Date(session.expiresAt.iso) : undefined;
|
|
202
|
+
if (expiresAt < now) {
|
|
203
|
+
throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'Session token is expired.');
|
|
204
|
+
}
|
|
205
|
+
const obj = session.user;
|
|
206
|
+
if (typeof obj['objectId'] === 'string' && obj['objectId'].startsWith('role:')) {
|
|
207
|
+
throw new Parse.Error(Parse.Error.INTERNAL_SERVER_ERROR, 'Invalid object ID.');
|
|
208
|
+
}
|
|
209
|
+
delete obj.password;
|
|
210
|
+
obj['className'] = '_User';
|
|
211
|
+
obj['sessionToken'] = sessionToken;
|
|
212
|
+
if (cacheController) {
|
|
213
|
+
cacheController.user.put(sessionToken, obj);
|
|
214
|
+
}
|
|
215
|
+
renewSessionIfNeeded({
|
|
216
|
+
config,
|
|
217
|
+
session,
|
|
218
|
+
sessionToken
|
|
219
|
+
});
|
|
220
|
+
const userObject = Parse.Object.fromJSON(obj);
|
|
221
|
+
return new Auth({
|
|
222
|
+
config,
|
|
223
|
+
cacheController,
|
|
224
|
+
isMaster: false,
|
|
225
|
+
installationId,
|
|
226
|
+
user: userObject
|
|
84
227
|
});
|
|
85
228
|
};
|
|
86
|
-
|
|
87
|
-
|
|
229
|
+
var getAuthForLegacySessionToken = async function ({
|
|
230
|
+
config,
|
|
231
|
+
sessionToken,
|
|
232
|
+
installationId
|
|
233
|
+
}) {
|
|
88
234
|
var restOptions = {
|
|
89
235
|
limit: 1
|
|
90
236
|
};
|
|
91
|
-
|
|
237
|
+
const RestQuery = require('./RestQuery');
|
|
238
|
+
var query = await RestQuery({
|
|
239
|
+
method: RestQuery.Method.get,
|
|
240
|
+
config,
|
|
241
|
+
runBeforeFind: false,
|
|
242
|
+
auth: master(config),
|
|
243
|
+
className: '_User',
|
|
244
|
+
restWhere: {
|
|
245
|
+
_session_token: sessionToken
|
|
246
|
+
},
|
|
247
|
+
restOptions
|
|
248
|
+
});
|
|
92
249
|
return query.execute().then(response => {
|
|
93
250
|
var results = response.results;
|
|
94
251
|
if (results.length !== 1) {
|
|
@@ -97,13 +254,18 @@ var getAuthForLegacySessionToken = function ({ config, sessionToken, installatio
|
|
|
97
254
|
const obj = results[0];
|
|
98
255
|
obj.className = '_User';
|
|
99
256
|
const userObject = Parse.Object.fromJSON(obj);
|
|
100
|
-
return new Auth({
|
|
257
|
+
return new Auth({
|
|
258
|
+
config,
|
|
259
|
+
isMaster: false,
|
|
260
|
+
installationId,
|
|
261
|
+
user: userObject
|
|
262
|
+
});
|
|
101
263
|
});
|
|
102
264
|
};
|
|
103
265
|
|
|
104
266
|
// Returns a promise that resolves to an array of role names
|
|
105
267
|
Auth.prototype.getUserRoles = function () {
|
|
106
|
-
if (this.isMaster || !this.user) {
|
|
268
|
+
if (this.isMaster || this.isMaintenance || !this.user) {
|
|
107
269
|
return Promise.resolve([]);
|
|
108
270
|
}
|
|
109
271
|
if (this.fetchedRoles) {
|
|
@@ -115,84 +277,140 @@ Auth.prototype.getUserRoles = function () {
|
|
|
115
277
|
this.rolePromise = this._loadRoles();
|
|
116
278
|
return this.rolePromise;
|
|
117
279
|
};
|
|
280
|
+
Auth.prototype.getRolesForUser = async function () {
|
|
281
|
+
//Stack all Parse.Role
|
|
282
|
+
const results = [];
|
|
283
|
+
if (this.config) {
|
|
284
|
+
const restWhere = {
|
|
285
|
+
users: {
|
|
286
|
+
__type: 'Pointer',
|
|
287
|
+
className: '_User',
|
|
288
|
+
objectId: this.user.id
|
|
289
|
+
}
|
|
290
|
+
};
|
|
291
|
+
const RestQuery = require('./RestQuery');
|
|
292
|
+
const query = await RestQuery({
|
|
293
|
+
method: RestQuery.Method.find,
|
|
294
|
+
runBeforeFind: false,
|
|
295
|
+
config: this.config,
|
|
296
|
+
auth: master(this.config),
|
|
297
|
+
className: '_Role',
|
|
298
|
+
restWhere
|
|
299
|
+
});
|
|
300
|
+
await query.each(result => results.push(result));
|
|
301
|
+
} else {
|
|
302
|
+
await new Parse.Query(Parse.Role).equalTo('users', this.user).each(result => results.push(result.toJSON()), {
|
|
303
|
+
useMasterKey: true
|
|
304
|
+
});
|
|
305
|
+
}
|
|
306
|
+
return results;
|
|
307
|
+
};
|
|
118
308
|
|
|
119
|
-
// Iterates through the role tree and compiles a
|
|
120
|
-
Auth.prototype._loadRoles = function () {
|
|
121
|
-
|
|
122
|
-
|
|
309
|
+
// Iterates through the role tree and compiles a user's roles
|
|
310
|
+
Auth.prototype._loadRoles = async function () {
|
|
311
|
+
if (this.cacheController) {
|
|
312
|
+
const cachedRoles = await this.cacheController.role.get(this.user.id);
|
|
123
313
|
if (cachedRoles != null) {
|
|
124
314
|
this.fetchedRoles = true;
|
|
125
315
|
this.userRoles = cachedRoles;
|
|
126
|
-
return
|
|
316
|
+
return cachedRoles;
|
|
127
317
|
}
|
|
318
|
+
}
|
|
319
|
+
|
|
320
|
+
// First get the role ids this user is directly a member of
|
|
321
|
+
const results = await this.getRolesForUser();
|
|
322
|
+
if (!results.length) {
|
|
323
|
+
this.userRoles = [];
|
|
324
|
+
this.fetchedRoles = true;
|
|
325
|
+
this.rolePromise = null;
|
|
326
|
+
this.cacheRoles();
|
|
327
|
+
return this.userRoles;
|
|
328
|
+
}
|
|
329
|
+
const rolesMap = results.reduce((m, r) => {
|
|
330
|
+
m.names.push(r.name);
|
|
331
|
+
m.ids.push(r.objectId);
|
|
332
|
+
return m;
|
|
333
|
+
}, {
|
|
334
|
+
ids: [],
|
|
335
|
+
names: []
|
|
336
|
+
});
|
|
128
337
|
|
|
129
|
-
|
|
130
|
-
|
|
338
|
+
// run the recursive finding
|
|
339
|
+
const roleNames = await this._getAllRolesNamesForRoleIds(rolesMap.ids, rolesMap.names);
|
|
340
|
+
this.userRoles = roleNames.map(r => {
|
|
341
|
+
return 'role:' + r;
|
|
342
|
+
});
|
|
343
|
+
this.fetchedRoles = true;
|
|
344
|
+
this.rolePromise = null;
|
|
345
|
+
this.cacheRoles();
|
|
346
|
+
return this.userRoles;
|
|
347
|
+
};
|
|
348
|
+
Auth.prototype.cacheRoles = function () {
|
|
349
|
+
if (!this.cacheController) {
|
|
350
|
+
return false;
|
|
351
|
+
}
|
|
352
|
+
this.cacheController.role.put(this.user.id, Array(...this.userRoles));
|
|
353
|
+
return true;
|
|
354
|
+
};
|
|
355
|
+
Auth.prototype.clearRoleCache = function (sessionToken) {
|
|
356
|
+
if (!this.cacheController) {
|
|
357
|
+
return false;
|
|
358
|
+
}
|
|
359
|
+
this.cacheController.role.del(this.user.id);
|
|
360
|
+
this.cacheController.user.del(sessionToken);
|
|
361
|
+
return true;
|
|
362
|
+
};
|
|
363
|
+
Auth.prototype.getRolesByIds = async function (ins) {
|
|
364
|
+
const results = [];
|
|
365
|
+
// Build an OR query across all parentRoles
|
|
366
|
+
if (!this.config) {
|
|
367
|
+
await new Parse.Query(Parse.Role).containedIn('roles', ins.map(id => {
|
|
368
|
+
const role = new Parse.Object(Parse.Role);
|
|
369
|
+
role.id = id;
|
|
370
|
+
return role;
|
|
371
|
+
})).each(result => results.push(result.toJSON()), {
|
|
372
|
+
useMasterKey: true
|
|
373
|
+
});
|
|
374
|
+
} else {
|
|
375
|
+
const roles = ins.map(id => {
|
|
376
|
+
return {
|
|
131
377
|
__type: 'Pointer',
|
|
132
|
-
className: '
|
|
133
|
-
objectId:
|
|
378
|
+
className: '_Role',
|
|
379
|
+
objectId: id
|
|
380
|
+
};
|
|
381
|
+
});
|
|
382
|
+
const restWhere = {
|
|
383
|
+
roles: {
|
|
384
|
+
$in: roles
|
|
134
385
|
}
|
|
135
386
|
};
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
cacheAdapter.role.put(this.user.id, Array(...this.userRoles));
|
|
146
|
-
return Promise.resolve(this.userRoles);
|
|
147
|
-
}
|
|
148
|
-
var rolesMap = results.reduce((m, r) => {
|
|
149
|
-
m.names.push(r.name);
|
|
150
|
-
m.ids.push(r.objectId);
|
|
151
|
-
return m;
|
|
152
|
-
}, { ids: [], names: [] });
|
|
153
|
-
|
|
154
|
-
// run the recursive finding
|
|
155
|
-
return this._getAllRolesNamesForRoleIds(rolesMap.ids, rolesMap.names).then(roleNames => {
|
|
156
|
-
this.userRoles = roleNames.map(r => {
|
|
157
|
-
return 'role:' + r;
|
|
158
|
-
});
|
|
159
|
-
this.fetchedRoles = true;
|
|
160
|
-
this.rolePromise = null;
|
|
161
|
-
cacheAdapter.role.put(this.user.id, Array(...this.userRoles));
|
|
162
|
-
return Promise.resolve(this.userRoles);
|
|
163
|
-
});
|
|
387
|
+
const RestQuery = require('./RestQuery');
|
|
388
|
+
const query = await RestQuery({
|
|
389
|
+
method: RestQuery.Method.find,
|
|
390
|
+
config: this.config,
|
|
391
|
+
runBeforeFind: false,
|
|
392
|
+
auth: master(this.config),
|
|
393
|
+
className: '_Role',
|
|
394
|
+
restWhere
|
|
164
395
|
});
|
|
165
|
-
|
|
396
|
+
await query.each(result => results.push(result));
|
|
397
|
+
}
|
|
398
|
+
return results;
|
|
166
399
|
};
|
|
167
400
|
|
|
168
401
|
// Given a list of roleIds, find all the parent roles, returns a promise with all names
|
|
169
402
|
Auth.prototype._getAllRolesNamesForRoleIds = function (roleIDs, names = [], queriedRoles = {}) {
|
|
170
403
|
const ins = roleIDs.filter(roleID => {
|
|
171
|
-
|
|
172
|
-
}).map(roleID => {
|
|
173
|
-
// mark as queried
|
|
404
|
+
const wasQueried = queriedRoles[roleID] !== true;
|
|
174
405
|
queriedRoles[roleID] = true;
|
|
175
|
-
return
|
|
176
|
-
__type: 'Pointer',
|
|
177
|
-
className: '_Role',
|
|
178
|
-
objectId: roleID
|
|
179
|
-
};
|
|
406
|
+
return wasQueried;
|
|
180
407
|
});
|
|
181
408
|
|
|
182
409
|
// all roles are accounted for, return the names
|
|
183
410
|
if (ins.length == 0) {
|
|
184
411
|
return Promise.resolve([...new Set(names)]);
|
|
185
412
|
}
|
|
186
|
-
|
|
187
|
-
let restWhere;
|
|
188
|
-
if (ins.length == 1) {
|
|
189
|
-
restWhere = { 'roles': ins[0] };
|
|
190
|
-
} else {
|
|
191
|
-
restWhere = { 'roles': { '$in': ins } };
|
|
192
|
-
}
|
|
193
|
-
const query = new RestQuery(this.config, master(this.config), '_Role', restWhere, {});
|
|
194
|
-
return query.execute().then(response => {
|
|
195
|
-
var results = response.results;
|
|
413
|
+
return this.getRolesByIds(ins).then(results => {
|
|
196
414
|
// Nothing found
|
|
197
415
|
if (!results.length) {
|
|
198
416
|
return Promise.resolve(names);
|
|
@@ -202,7 +420,10 @@ Auth.prototype._getAllRolesNamesForRoleIds = function (roleIDs, names = [], quer
|
|
|
202
420
|
memo.names.push(role.name);
|
|
203
421
|
memo.ids.push(role.objectId);
|
|
204
422
|
return memo;
|
|
205
|
-
}, {
|
|
423
|
+
}, {
|
|
424
|
+
ids: [],
|
|
425
|
+
names: []
|
|
426
|
+
});
|
|
206
427
|
// store the new found names
|
|
207
428
|
names = names.concat(resultMap.names);
|
|
208
429
|
// find the next ones, circular roles will be cut
|
|
@@ -211,48 +432,190 @@ Auth.prototype._getAllRolesNamesForRoleIds = function (roleIDs, names = [], quer
|
|
|
211
432
|
return Promise.resolve([...new Set(names)]);
|
|
212
433
|
});
|
|
213
434
|
};
|
|
435
|
+
const findUsersWithAuthData = async (config, authData, beforeFind) => {
|
|
436
|
+
const providers = Object.keys(authData);
|
|
437
|
+
const queries = await Promise.all(providers.map(async provider => {
|
|
438
|
+
const providerAuthData = authData[provider];
|
|
439
|
+
const adapter = config.authDataManager.getValidatorForProvider(provider)?.adapter;
|
|
440
|
+
if (beforeFind && typeof adapter?.beforeFind === 'function') {
|
|
441
|
+
await adapter.beforeFind(providerAuthData);
|
|
442
|
+
}
|
|
443
|
+
if (!providerAuthData?.id) {
|
|
444
|
+
return null;
|
|
445
|
+
}
|
|
446
|
+
return {
|
|
447
|
+
[`authData.${provider}.id`]: providerAuthData.id
|
|
448
|
+
};
|
|
449
|
+
}));
|
|
214
450
|
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
additionalSessionData
|
|
220
|
-
}) {
|
|
221
|
-
const token = 'r:' + cryptoUtils.newToken();
|
|
222
|
-
const expiresAt = config.generateSessionExpiresAt();
|
|
223
|
-
const sessionData = {
|
|
224
|
-
sessionToken: token,
|
|
225
|
-
user: {
|
|
226
|
-
__type: 'Pointer',
|
|
227
|
-
className: '_User',
|
|
228
|
-
objectId: userId
|
|
229
|
-
},
|
|
230
|
-
createdWith,
|
|
231
|
-
restricted: false,
|
|
232
|
-
expiresAt: Parse._encode(expiresAt)
|
|
233
|
-
};
|
|
234
|
-
|
|
235
|
-
if (installationId) {
|
|
236
|
-
sessionData.installationId = installationId;
|
|
451
|
+
// Filter out null queries
|
|
452
|
+
const validQueries = queries.filter(query => query !== null);
|
|
453
|
+
if (!validQueries.length) {
|
|
454
|
+
return [];
|
|
237
455
|
}
|
|
238
456
|
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
|
|
242
|
-
|
|
457
|
+
// Perform database query
|
|
458
|
+
return config.database.find('_User', {
|
|
459
|
+
$or: validQueries
|
|
460
|
+
}, {
|
|
461
|
+
limit: 2
|
|
462
|
+
});
|
|
463
|
+
};
|
|
464
|
+
const hasMutatedAuthData = (authData, userAuthData) => {
|
|
465
|
+
if (!userAuthData) {
|
|
466
|
+
return {
|
|
467
|
+
hasMutatedAuthData: true,
|
|
468
|
+
mutatedAuthData: authData
|
|
469
|
+
};
|
|
470
|
+
}
|
|
471
|
+
const mutatedAuthData = {};
|
|
472
|
+
Object.keys(authData).forEach(provider => {
|
|
473
|
+
// Anonymous provider is not handled this way
|
|
474
|
+
if (provider === 'anonymous') {
|
|
475
|
+
return;
|
|
476
|
+
}
|
|
477
|
+
const providerData = authData[provider];
|
|
478
|
+
const userProviderAuthData = userAuthData[provider];
|
|
479
|
+
if (!(0, _util.isDeepStrictEqual)(providerData, userProviderAuthData)) {
|
|
480
|
+
mutatedAuthData[provider] = providerData;
|
|
481
|
+
}
|
|
482
|
+
});
|
|
483
|
+
const hasMutatedAuthData = Object.keys(mutatedAuthData).length !== 0;
|
|
243
484
|
return {
|
|
244
|
-
|
|
245
|
-
|
|
485
|
+
hasMutatedAuthData,
|
|
486
|
+
mutatedAuthData
|
|
246
487
|
};
|
|
247
488
|
};
|
|
489
|
+
const checkIfUserHasProvidedConfiguredProvidersForLogin = (req = {}, authData = {}, userAuthData = {}, config) => {
|
|
490
|
+
const savedUserProviders = Object.keys(userAuthData).map(provider => ({
|
|
491
|
+
name: provider,
|
|
492
|
+
adapter: config.authDataManager.getValidatorForProvider(provider).adapter
|
|
493
|
+
}));
|
|
494
|
+
const hasProvidedASoloProvider = savedUserProviders.some(provider => provider && provider.adapter && provider.adapter.policy === 'solo' && authData[provider.name]);
|
|
495
|
+
|
|
496
|
+
// Solo providers can be considered as safe, so we do not have to check if the user needs
|
|
497
|
+
// to provide an additional provider to login. An auth adapter with "solo" (like webauthn) means
|
|
498
|
+
// no "additional" auth needs to be provided to login (like OTP, MFA)
|
|
499
|
+
if (hasProvidedASoloProvider) {
|
|
500
|
+
return;
|
|
501
|
+
}
|
|
502
|
+
const additionProvidersNotFound = [];
|
|
503
|
+
const hasProvidedAtLeastOneAdditionalProvider = savedUserProviders.some(provider => {
|
|
504
|
+
let policy = provider.adapter.policy;
|
|
505
|
+
if (typeof policy === 'function') {
|
|
506
|
+
const requestObject = {
|
|
507
|
+
ip: req.config.ip,
|
|
508
|
+
user: req.auth.user,
|
|
509
|
+
master: req.auth.isMaster
|
|
510
|
+
};
|
|
511
|
+
policy = policy.call(provider.adapter, requestObject, userAuthData[provider.name]);
|
|
512
|
+
}
|
|
513
|
+
if (policy === 'additional') {
|
|
514
|
+
if (authData[provider.name]) {
|
|
515
|
+
return true;
|
|
516
|
+
} else {
|
|
517
|
+
// Push missing provider for error message
|
|
518
|
+
additionProvidersNotFound.push(provider.name);
|
|
519
|
+
}
|
|
520
|
+
}
|
|
521
|
+
});
|
|
522
|
+
if (hasProvidedAtLeastOneAdditionalProvider || !additionProvidersNotFound.length) {
|
|
523
|
+
return;
|
|
524
|
+
}
|
|
525
|
+
throw new Parse.Error(Parse.Error.OTHER_CAUSE, `Missing additional authData ${additionProvidersNotFound.join(',')}`);
|
|
526
|
+
};
|
|
248
527
|
|
|
528
|
+
// Validate each authData step-by-step and return the provider responses
|
|
529
|
+
const handleAuthDataValidation = async (authData, req, foundUser) => {
|
|
530
|
+
let user;
|
|
531
|
+
if (foundUser) {
|
|
532
|
+
user = Parse.User.fromJSON({
|
|
533
|
+
className: '_User',
|
|
534
|
+
...foundUser
|
|
535
|
+
});
|
|
536
|
+
// Find user by session and current objectId; only pass user if it's the current user or master key is provided
|
|
537
|
+
} else if (req.auth && req.auth.user && typeof req.getUserId === 'function' && req.getUserId() === req.auth.user.id || req.auth && req.auth.isMaster && typeof req.getUserId === 'function' && req.getUserId()) {
|
|
538
|
+
user = new Parse.User();
|
|
539
|
+
user.id = req.auth.isMaster ? req.getUserId() : req.auth.user.id;
|
|
540
|
+
await user.fetch({
|
|
541
|
+
useMasterKey: true
|
|
542
|
+
});
|
|
543
|
+
}
|
|
544
|
+
const {
|
|
545
|
+
updatedObject
|
|
546
|
+
} = req.buildParseObjects();
|
|
547
|
+
const requestObject = (0, _triggers.getRequestObject)(undefined, req.auth, updatedObject, user, req.config);
|
|
548
|
+
// Perform validation as step-by-step pipeline for better error consistency
|
|
549
|
+
// and also to avoid to trigger a provider (like OTP SMS) if another one fails
|
|
550
|
+
const acc = {
|
|
551
|
+
authData: {},
|
|
552
|
+
authDataResponse: {}
|
|
553
|
+
};
|
|
554
|
+
const authKeys = Object.keys(authData).sort();
|
|
555
|
+
for (const provider of authKeys) {
|
|
556
|
+
let method = '';
|
|
557
|
+
try {
|
|
558
|
+
if (authData[provider] === null) {
|
|
559
|
+
acc.authData[provider] = null;
|
|
560
|
+
continue;
|
|
561
|
+
}
|
|
562
|
+
const {
|
|
563
|
+
validator
|
|
564
|
+
} = req.config.authDataManager.getValidatorForProvider(provider) || {};
|
|
565
|
+
const authProvider = (req.config.auth || {})[provider] || {};
|
|
566
|
+
if (!validator || authProvider.enabled === false) {
|
|
567
|
+
throw new Parse.Error(Parse.Error.UNSUPPORTED_SERVICE, 'This authentication method is unsupported.');
|
|
568
|
+
}
|
|
569
|
+
let validationResult = await validator(authData[provider], req, user, requestObject);
|
|
570
|
+
method = validationResult && validationResult.method;
|
|
571
|
+
requestObject.triggerName = method;
|
|
572
|
+
if (validationResult && validationResult.validator) {
|
|
573
|
+
validationResult = await validationResult.validator();
|
|
574
|
+
}
|
|
575
|
+
if (!validationResult) {
|
|
576
|
+
acc.authData[provider] = authData[provider];
|
|
577
|
+
continue;
|
|
578
|
+
}
|
|
579
|
+
if (!Object.keys(validationResult).length) {
|
|
580
|
+
acc.authData[provider] = authData[provider];
|
|
581
|
+
continue;
|
|
582
|
+
}
|
|
583
|
+
if (validationResult.response) {
|
|
584
|
+
acc.authDataResponse[provider] = validationResult.response;
|
|
585
|
+
}
|
|
586
|
+
// Some auth providers after initialization will avoid to replace authData already stored
|
|
587
|
+
if (!validationResult.doNotSave) {
|
|
588
|
+
acc.authData[provider] = validationResult.save || authData[provider];
|
|
589
|
+
}
|
|
590
|
+
} catch (err) {
|
|
591
|
+
const e = (0, _triggers.resolveError)(err, {
|
|
592
|
+
code: Parse.Error.SCRIPT_FAILED,
|
|
593
|
+
message: 'Auth failed. Unknown error.'
|
|
594
|
+
});
|
|
595
|
+
const userString = req.auth && req.auth.user ? req.auth.user.id : req.data.objectId || undefined;
|
|
596
|
+
_logger.logger.error(`Failed running auth step ${method} for ${provider} for user ${userString} with Error: ` + JSON.stringify(e), {
|
|
597
|
+
authenticationStep: method,
|
|
598
|
+
error: e,
|
|
599
|
+
user: userString,
|
|
600
|
+
provider
|
|
601
|
+
});
|
|
602
|
+
throw e;
|
|
603
|
+
}
|
|
604
|
+
}
|
|
605
|
+
return acc;
|
|
606
|
+
};
|
|
249
607
|
module.exports = {
|
|
250
608
|
Auth,
|
|
251
609
|
master,
|
|
610
|
+
maintenance,
|
|
252
611
|
nobody,
|
|
253
612
|
readOnly,
|
|
613
|
+
shouldUpdateSessionExpiry,
|
|
254
614
|
getAuthForSessionToken,
|
|
255
615
|
getAuthForLegacySessionToken,
|
|
256
|
-
|
|
616
|
+
findUsersWithAuthData,
|
|
617
|
+
hasMutatedAuthData,
|
|
618
|
+
checkIfUserHasProvidedConfiguredProvidersForLogin,
|
|
619
|
+
handleAuthDataValidation
|
|
257
620
|
};
|
|
258
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/Auth.js"],"names":["cryptoUtils","require","RestQuery","Parse","Auth","config","isMaster","isReadOnly","user","installationId","userRoles","fetchedRoles","rolePromise","prototype","isUnauthenticated","master","readOnly","nobody","getAuthForSessionToken","sessionToken","cacheController","get","then","userJSON","cachedUser","Object","fromJSON","Promise","resolve","restOptions","limit","include","query","execute","response","results","length","Error","INVALID_SESSION_TOKEN","now","Date","expiresAt","iso","undefined","obj","password","put","userObject","getAuthForLegacySessionToken","className","getUserRoles","_loadRoles","cacheAdapter","role","id","cachedRoles","restWhere","__type","objectId","Array","rolesMap","reduce","m","r","names","push","name","ids","_getAllRolesNamesForRoleIds","roleNames","map","roleIDs","queriedRoles","ins","filter","roleID","Set","resultMap","memo","concat","createSession","userId","createdWith","additionalSessionData","token","newToken","generateSessionExpiresAt","sessionData","restricted","_encode","assign","RestWrite","module","exports"],"mappings":"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","file":"Auth.js","sourcesContent":["const cryptoUtils = require('./cryptoUtils');\nconst RestQuery = require('./RestQuery');\nconst Parse = require('parse/node');\n\n// An Auth object tells you who is requesting something and whether\n// the master key was used.\n// userObject is a Parse.User and can be null if there's no user.\nfunction Auth({ config, isMaster = false, isReadOnly = false, user, installationId } = {}) {\n  this.config = config;\n  this.installationId = installationId;\n  this.isMaster = isMaster;\n  this.user = user;\n  this.isReadOnly = isReadOnly;\n\n  // Assuming a users roles won't change during a single request, we'll\n  // only load them once.\n  this.userRoles = [];\n  this.fetchedRoles = false;\n  this.rolePromise = null;\n}\n\n// Whether this auth could possibly modify the given user id.\n// It still could be forbidden via ACLs even if this returns true.\nAuth.prototype.isUnauthenticated = function() {\n  if (this.isMaster) {\n    return false;\n  }\n  if (this.user) {\n    return false;\n  }\n  return true;\n};\n\n// A helper to get a master-level Auth object\nfunction master(config) {\n  return new Auth({ config, isMaster: true });\n}\n\n// A helper to get a master-level Auth object\nfunction readOnly(config) {\n  return new Auth({ config, isMaster: true, isReadOnly: true });\n}\n\n// A helper to get a nobody-level Auth object\nfunction nobody(config) {\n  return new Auth({ config, isMaster: false });\n}\n\n\n// Returns a promise that resolves to an Auth object\nvar getAuthForSessionToken = function({ config, sessionToken, installationId } = {}) {\n  return config.cacheController.user.get(sessionToken).then((userJSON) => {\n    if (userJSON) {\n      const cachedUser = Parse.Object.fromJSON(userJSON);\n      return Promise.resolve(new Auth({config, isMaster: false, installationId, user: cachedUser}));\n    }\n\n    var restOptions = {\n      limit: 1,\n      include: 'user'\n    };\n\n    var query = new RestQuery(config, master(config), '_Session', {sessionToken}, restOptions);\n    return query.execute().then((response) => {\n      var results = response.results;\n      if (results.length !== 1 || !results[0]['user']) {\n        throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'Invalid session token');\n      }\n\n      var now = new Date(),\n        expiresAt = results[0].expiresAt ? new Date(results[0].expiresAt.iso) : undefined;\n      if (expiresAt < now) {\n        throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN,\n          'Session token is expired.');\n      }\n      var obj = results[0]['user'];\n      delete obj.password;\n      obj['className'] = '_User';\n      obj['sessionToken'] = sessionToken;\n      config.cacheController.user.put(sessionToken, obj);\n      const userObject = Parse.Object.fromJSON(obj);\n      return new Auth({config, isMaster: false, installationId, user: userObject});\n    });\n  });\n};\n\nvar getAuthForLegacySessionToken = function({config, sessionToken, installationId } = {}) {\n  var restOptions = {\n    limit: 1\n  };\n  var query = new RestQuery(config, master(config), '_User', { sessionToken: sessionToken}, restOptions);\n  return query.execute().then((response) => {\n    var results = response.results;\n    if (results.length !== 1) {\n      throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'invalid legacy session token');\n    }\n    const obj = results[0];\n    obj.className = '_User';\n    const userObject = Parse.Object.fromJSON(obj);\n    return new Auth({config, isMaster: false, installationId, user: userObject});\n  });\n}\n\n// Returns a promise that resolves to an array of role names\nAuth.prototype.getUserRoles = function() {\n  if (this.isMaster || !this.user) {\n    return Promise.resolve([]);\n  }\n  if (this.fetchedRoles) {\n    return Promise.resolve(this.userRoles);\n  }\n  if (this.rolePromise) {\n    return this.rolePromise;\n  }\n  this.rolePromise = this._loadRoles();\n  return this.rolePromise;\n};\n\n// Iterates through the role tree and compiles a users roles\nAuth.prototype._loadRoles = function() {\n  var cacheAdapter = this.config.cacheController;\n  return cacheAdapter.role.get(this.user.id).then((cachedRoles) => {\n    if (cachedRoles != null) {\n      this.fetchedRoles = true;\n      this.userRoles = cachedRoles;\n      return Promise.resolve(cachedRoles);\n    }\n\n    var restWhere = {\n      'users': {\n        __type: 'Pointer',\n        className: '_User',\n        objectId: this.user.id\n      }\n    };\n    // First get the role ids this user is directly a member of\n    var query = new RestQuery(this.config, master(this.config), '_Role', restWhere, {});\n    return query.execute().then((response) => {\n      var results = response.results;\n      if (!results.length) {\n        this.userRoles = [];\n        this.fetchedRoles = true;\n        this.rolePromise = null;\n\n        cacheAdapter.role.put(this.user.id, Array(...this.userRoles));\n        return Promise.resolve(this.userRoles);\n      }\n      var rolesMap = results.reduce((m, r) => {\n        m.names.push(r.name);\n        m.ids.push(r.objectId);\n        return m;\n      }, {ids: [], names: []});\n\n      // run the recursive finding\n      return this._getAllRolesNamesForRoleIds(rolesMap.ids, rolesMap.names)\n        .then((roleNames) => {\n          this.userRoles = roleNames.map((r) => {\n            return 'role:' + r;\n          });\n          this.fetchedRoles = true;\n          this.rolePromise = null;\n          cacheAdapter.role.put(this.user.id, Array(...this.userRoles));\n          return Promise.resolve(this.userRoles);\n        });\n    });\n  });\n};\n\n// Given a list of roleIds, find all the parent roles, returns a promise with all names\nAuth.prototype._getAllRolesNamesForRoleIds = function(roleIDs, names = [], queriedRoles = {}) {\n  const ins = roleIDs.filter((roleID) => {\n    return queriedRoles[roleID] !== true;\n  }).map((roleID) => {\n    // mark as queried\n    queriedRoles[roleID] = true;\n    return {\n      __type: 'Pointer',\n      className: '_Role',\n      objectId: roleID\n    }\n  });\n\n  // all roles are accounted for, return the names\n  if (ins.length == 0) {\n    return Promise.resolve([...new Set(names)]);\n  }\n  // Build an OR query across all parentRoles\n  let restWhere;\n  if (ins.length == 1) {\n    restWhere = { 'roles': ins[0] };\n  } else {\n    restWhere = { 'roles': { '$in': ins }}\n  }\n  const query = new RestQuery(this.config, master(this.config), '_Role', restWhere, {});\n  return query.execute().then((response) => {\n    var results = response.results;\n    // Nothing found\n    if (!results.length) {\n      return Promise.resolve(names);\n    }\n    // Map the results with all Ids and names\n    const resultMap = results.reduce((memo, role) => {\n      memo.names.push(role.name);\n      memo.ids.push(role.objectId);\n      return memo;\n    }, {ids: [], names: []});\n    // store the new found names\n    names = names.concat(resultMap.names);\n    // find the next ones, circular roles will be cut\n    return this._getAllRolesNamesForRoleIds(resultMap.ids, names, queriedRoles)\n  }).then((names) => {\n    return Promise.resolve([...new Set(names)])\n  })\n}\n\nconst createSession = function(config, {\n  userId,\n  createdWith,\n  installationId,\n  additionalSessionData,\n}) {\n  const token = 'r:' + cryptoUtils.newToken();\n  const expiresAt = config.generateSessionExpiresAt();\n  const sessionData = {\n    sessionToken: token,\n    user: {\n      __type: 'Pointer',\n      className: '_User',\n      objectId: userId\n    },\n    createdWith,\n    restricted: false,\n    expiresAt: Parse._encode(expiresAt)\n  };\n\n  if (installationId) {\n    sessionData.installationId = installationId\n  }\n\n  Object.assign(sessionData, additionalSessionData);\n  // We need to import RestWrite at this point for the cyclic dependency it has to it\n  const RestWrite = require('./RestWrite');\n\n  return {\n    sessionData,\n    createSession: () => new RestWrite(config, master(config), '_Session', null, sessionData).execute()\n  }\n}\n\nmodule.exports = {\n  Auth,\n  master,\n  nobody,\n  readOnly,\n  getAuthForSessionToken,\n  getAuthForLegacySessionToken,\n  createSession,\n};\n"]}
|
|
621
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_util","require","_triggers","_logger","_lruCache","_RestQuery","_interopRequireDefault","_RestWrite","e","__esModule","default","Parse","Auth","config","cacheController","undefined","isMaster","isMaintenance","isReadOnly","user","installationId","userRoles","fetchedRoles","rolePromise","prototype","isUnauthenticated","master","maintenance","readOnly","nobody","throttle","LRU","max","ttl","shouldUpdateSessionExpiry","session","resetAfter","sessionLength","lastUpdated","Date","updatedAt","skipRange","setTime","getTime","renewSessionIfNeeded","sessionToken","extendSessionOnUse","get","set","query","RestQuery","method","Method","auth","runBeforeFind","className","restWhere","restOptions","limit","results","execute","expiresAt","generateSessionExpiresAt","RestWrite","objectId","_encode","code","Error","OBJECT_NOT_FOUND","logger","error","getAuthForSessionToken","userJSON","cachedUser","Object","fromJSON","Promise","resolve","include","Query","Session","equalTo","find","useMasterKey","map","obj","toJSON","length","INVALID_SESSION_TOKEN","now","iso","startsWith","INTERNAL_SERVER_ERROR","password","put","userObject","getAuthForLegacySessionToken","_session_token","then","response","getUserRoles","_loadRoles","getRolesForUser","users","__type","id","each","result","push","Role","cachedRoles","role","cacheRoles","rolesMap","reduce","m","r","names","name","ids","roleNames","_getAllRolesNamesForRoleIds","Array","clearRoleCache","del","getRolesByIds","ins","containedIn","roles","$in","roleIDs","queriedRoles","filter","roleID","wasQueried","Set","resultMap","memo","concat","findUsersWithAuthData","authData","beforeFind","providers","keys","queries","all","provider","providerAuthData","adapter","authDataManager","getValidatorForProvider","validQueries","database","$or","hasMutatedAuthData","userAuthData","mutatedAuthData","forEach","providerData","userProviderAuthData","isDeepStrictEqual","checkIfUserHasProvidedConfiguredProvidersForLogin","req","savedUserProviders","hasProvidedASoloProvider","some","policy","additionProvidersNotFound","hasProvidedAtLeastOneAdditionalProvider","requestObject","ip","call","OTHER_CAUSE","join","handleAuthDataValidation","foundUser","User","getUserId","fetch","updatedObject","buildParseObjects","getRequestObject","acc","authDataResponse","authKeys","sort","validator","authProvider","enabled","UNSUPPORTED_SERVICE","validationResult","triggerName","doNotSave","save","err","resolveError","SCRIPT_FAILED","message","userString","data","JSON","stringify","authenticationStep","module","exports"],"sources":["../src/Auth.js"],"sourcesContent":["const Parse = require('parse/node');\nimport { isDeepStrictEqual } from 'util';\nimport { getRequestObject, resolveError } from './triggers';\nimport { logger } from './logger';\nimport { LRUCache as LRU } from 'lru-cache';\nimport RestQuery from './RestQuery';\nimport RestWrite from './RestWrite';\n\n// An Auth object tells you who is requesting something and whether\n// the master key was used.\n// userObject is a Parse.User and can be null if there's no user.\nfunction Auth({\n  config,\n  cacheController = undefined,\n  isMaster = false,\n  isMaintenance = false,\n  isReadOnly = false,\n  user,\n  installationId,\n}) {\n  this.config = config;\n  this.cacheController = cacheController || (config && config.cacheController);\n  this.installationId = installationId;\n  this.isMaster = isMaster;\n  this.isMaintenance = isMaintenance;\n  this.user = user;\n  this.isReadOnly = isReadOnly;\n\n  // Assuming a users roles won't change during a single request, we'll\n  // only load them once.\n  this.userRoles = [];\n  this.fetchedRoles = false;\n  this.rolePromise = null;\n}\n\n// Whether this auth could possibly modify the given user id.\n// It still could be forbidden via ACLs even if this returns true.\nAuth.prototype.isUnauthenticated = function () {\n  if (this.isMaster) {\n    return false;\n  }\n  if (this.isMaintenance) {\n    return false;\n  }\n  if (this.user) {\n    return false;\n  }\n  return true;\n};\n\n// A helper to get a master-level Auth object\nfunction master(config) {\n  return new Auth({ config, isMaster: true });\n}\n\n// A helper to get a maintenance-level Auth object\nfunction maintenance(config) {\n  return new Auth({ config, isMaintenance: true });\n}\n\n// A helper to get a master-level Auth object\nfunction readOnly(config) {\n  return new Auth({ config, isMaster: true, isReadOnly: true });\n}\n\n// A helper to get a nobody-level Auth object\nfunction nobody(config) {\n  return new Auth({ config, isMaster: false });\n}\n\nconst throttle = new LRU({\n  max: 10000,\n  ttl: 500,\n});\n/**\n * Checks whether session should be updated based on last update time & session length.\n */\nfunction shouldUpdateSessionExpiry(config, session) {\n  const resetAfter = config.sessionLength / 2;\n  const lastUpdated = new Date(session?.updatedAt);\n  const skipRange = new Date();\n  skipRange.setTime(skipRange.getTime() - resetAfter * 1000);\n  return lastUpdated <= skipRange;\n}\n\nconst renewSessionIfNeeded = async ({ config, session, sessionToken }) => {\n  if (!config?.extendSessionOnUse) {\n    return;\n  }\n  if (throttle.get(sessionToken)) {\n    return;\n  }\n  throttle.set(sessionToken, true);\n  try {\n    if (!session) {\n      const query = await RestQuery({\n        method: RestQuery.Method.get,\n        config,\n        auth: master(config),\n        runBeforeFind: false,\n        className: '_Session',\n        restWhere: { sessionToken },\n        restOptions: { limit: 1 },\n      });\n      const { results } = await query.execute();\n      session = results[0];\n    }\n\n    if (!shouldUpdateSessionExpiry(config, session) || !session) {\n      return;\n    }\n    const expiresAt = config.generateSessionExpiresAt();\n    await new RestWrite(\n      config,\n      master(config),\n      '_Session',\n      { objectId: session.objectId },\n      { expiresAt: Parse._encode(expiresAt) }\n    ).execute();\n  } catch (e) {\n    if (e?.code !== Parse.Error.OBJECT_NOT_FOUND) {\n      logger.error('Could not update session expiry: ', e);\n    }\n  }\n};\n\n// Returns a promise that resolves to an Auth object\nconst getAuthForSessionToken = async function ({\n  config,\n  cacheController,\n  sessionToken,\n  installationId,\n}) {\n  cacheController = cacheController || (config && config.cacheController);\n  if (cacheController) {\n    const userJSON = await cacheController.user.get(sessionToken);\n    if (userJSON) {\n      const cachedUser = Parse.Object.fromJSON(userJSON);\n      renewSessionIfNeeded({ config, sessionToken });\n      return Promise.resolve(\n        new Auth({\n          config,\n          cacheController,\n          isMaster: false,\n          installationId,\n          user: cachedUser,\n        })\n      );\n    }\n  }\n\n  let results;\n  if (config) {\n    const restOptions = {\n      limit: 1,\n      include: 'user',\n    };\n    const RestQuery = require('./RestQuery');\n    const query = await RestQuery({\n      method: RestQuery.Method.get,\n      config,\n      runBeforeFind: false,\n      auth: master(config),\n      className: '_Session',\n      restWhere: { sessionToken },\n      restOptions,\n    });\n    results = (await query.execute()).results;\n  } else {\n    results = (\n      await new Parse.Query(Parse.Session)\n        .limit(1)\n        .include('user')\n        .equalTo('sessionToken', sessionToken)\n        .find({ useMasterKey: true })\n    ).map(obj => obj.toJSON());\n  }\n\n  if (results.length !== 1 || !results[0]['user']) {\n    throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'Invalid session token');\n  }\n  const session = results[0];\n  const now = new Date(),\n    expiresAt = session.expiresAt ? new Date(session.expiresAt.iso) : undefined;\n  if (expiresAt < now) {\n    throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'Session token is expired.');\n  }\n  const obj = session.user;\n\n  if (typeof obj['objectId'] === 'string' && obj['objectId'].startsWith('role:')) {\n    throw new Parse.Error(Parse.Error.INTERNAL_SERVER_ERROR, 'Invalid object ID.');\n  }\n\n  delete obj.password;\n  obj['className'] = '_User';\n  obj['sessionToken'] = sessionToken;\n  if (cacheController) {\n    cacheController.user.put(sessionToken, obj);\n  }\n  renewSessionIfNeeded({ config, session, sessionToken });\n  const userObject = Parse.Object.fromJSON(obj);\n  return new Auth({\n    config,\n    cacheController,\n    isMaster: false,\n    installationId,\n    user: userObject,\n  });\n};\n\nvar getAuthForLegacySessionToken = async function ({ config, sessionToken, installationId }) {\n  var restOptions = {\n    limit: 1,\n  };\n  const RestQuery = require('./RestQuery');\n  var query = await RestQuery({\n    method: RestQuery.Method.get,\n    config,\n    runBeforeFind: false,\n    auth: master(config),\n    className: '_User',\n    restWhere: { _session_token: sessionToken },\n    restOptions,\n  });\n  return query.execute().then(response => {\n    var results = response.results;\n    if (results.length !== 1) {\n      throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'invalid legacy session token');\n    }\n    const obj = results[0];\n    obj.className = '_User';\n    const userObject = Parse.Object.fromJSON(obj);\n    return new Auth({\n      config,\n      isMaster: false,\n      installationId,\n      user: userObject,\n    });\n  });\n};\n\n// Returns a promise that resolves to an array of role names\nAuth.prototype.getUserRoles = function () {\n  if (this.isMaster || this.isMaintenance || !this.user) {\n    return Promise.resolve([]);\n  }\n  if (this.fetchedRoles) {\n    return Promise.resolve(this.userRoles);\n  }\n  if (this.rolePromise) {\n    return this.rolePromise;\n  }\n  this.rolePromise = this._loadRoles();\n  return this.rolePromise;\n};\n\nAuth.prototype.getRolesForUser = async function () {\n  //Stack all Parse.Role\n  const results = [];\n  if (this.config) {\n    const restWhere = {\n      users: {\n        __type: 'Pointer',\n        className: '_User',\n        objectId: this.user.id,\n      },\n    };\n    const RestQuery = require('./RestQuery');\n    const query = await RestQuery({\n      method: RestQuery.Method.find,\n      runBeforeFind: false,\n      config: this.config,\n      auth: master(this.config),\n      className: '_Role',\n      restWhere,\n    });\n    await query.each(result => results.push(result));\n  } else {\n    await new Parse.Query(Parse.Role)\n      .equalTo('users', this.user)\n      .each(result => results.push(result.toJSON()), { useMasterKey: true });\n  }\n  return results;\n};\n\n// Iterates through the role tree and compiles a user's roles\nAuth.prototype._loadRoles = async function () {\n  if (this.cacheController) {\n    const cachedRoles = await this.cacheController.role.get(this.user.id);\n    if (cachedRoles != null) {\n      this.fetchedRoles = true;\n      this.userRoles = cachedRoles;\n      return cachedRoles;\n    }\n  }\n\n  // First get the role ids this user is directly a member of\n  const results = await this.getRolesForUser();\n  if (!results.length) {\n    this.userRoles = [];\n    this.fetchedRoles = true;\n    this.rolePromise = null;\n\n    this.cacheRoles();\n    return this.userRoles;\n  }\n\n  const rolesMap = results.reduce(\n    (m, r) => {\n      m.names.push(r.name);\n      m.ids.push(r.objectId);\n      return m;\n    },\n    { ids: [], names: [] }\n  );\n\n  // run the recursive finding\n  const roleNames = await this._getAllRolesNamesForRoleIds(rolesMap.ids, rolesMap.names);\n  this.userRoles = roleNames.map(r => {\n    return 'role:' + r;\n  });\n  this.fetchedRoles = true;\n  this.rolePromise = null;\n  this.cacheRoles();\n  return this.userRoles;\n};\n\nAuth.prototype.cacheRoles = function () {\n  if (!this.cacheController) {\n    return false;\n  }\n  this.cacheController.role.put(this.user.id, Array(...this.userRoles));\n  return true;\n};\n\nAuth.prototype.clearRoleCache = function (sessionToken) {\n  if (!this.cacheController) {\n    return false;\n  }\n  this.cacheController.role.del(this.user.id);\n  this.cacheController.user.del(sessionToken);\n  return true;\n};\n\nAuth.prototype.getRolesByIds = async function (ins) {\n  const results = [];\n  // Build an OR query across all parentRoles\n  if (!this.config) {\n    await new Parse.Query(Parse.Role)\n      .containedIn(\n        'roles',\n        ins.map(id => {\n          const role = new Parse.Object(Parse.Role);\n          role.id = id;\n          return role;\n        })\n      )\n      .each(result => results.push(result.toJSON()), { useMasterKey: true });\n  } else {\n    const roles = ins.map(id => {\n      return {\n        __type: 'Pointer',\n        className: '_Role',\n        objectId: id,\n      };\n    });\n    const restWhere = { roles: { $in: roles } };\n    const RestQuery = require('./RestQuery');\n    const query = await RestQuery({\n      method: RestQuery.Method.find,\n      config: this.config,\n      runBeforeFind: false,\n      auth: master(this.config),\n      className: '_Role',\n      restWhere,\n    });\n    await query.each(result => results.push(result));\n  }\n  return results;\n};\n\n// Given a list of roleIds, find all the parent roles, returns a promise with all names\nAuth.prototype._getAllRolesNamesForRoleIds = function (roleIDs, names = [], queriedRoles = {}) {\n  const ins = roleIDs.filter(roleID => {\n    const wasQueried = queriedRoles[roleID] !== true;\n    queriedRoles[roleID] = true;\n    return wasQueried;\n  });\n\n  // all roles are accounted for, return the names\n  if (ins.length == 0) {\n    return Promise.resolve([...new Set(names)]);\n  }\n\n  return this.getRolesByIds(ins)\n    .then(results => {\n      // Nothing found\n      if (!results.length) {\n        return Promise.resolve(names);\n      }\n      // Map the results with all Ids and names\n      const resultMap = results.reduce(\n        (memo, role) => {\n          memo.names.push(role.name);\n          memo.ids.push(role.objectId);\n          return memo;\n        },\n        { ids: [], names: [] }\n      );\n      // store the new found names\n      names = names.concat(resultMap.names);\n      // find the next ones, circular roles will be cut\n      return this._getAllRolesNamesForRoleIds(resultMap.ids, names, queriedRoles);\n    })\n    .then(names => {\n      return Promise.resolve([...new Set(names)]);\n    });\n};\n\nconst findUsersWithAuthData = async (config, authData, beforeFind) => {\n  const providers = Object.keys(authData);\n\n  const queries = await Promise.all(\n    providers.map(async provider => {\n      const providerAuthData = authData[provider];\n\n      const adapter = config.authDataManager.getValidatorForProvider(provider)?.adapter;\n      if (beforeFind && typeof adapter?.beforeFind === 'function') {\n        await adapter.beforeFind(providerAuthData);\n      }\n\n      if (!providerAuthData?.id) {\n        return null;\n      }\n\n      return { [`authData.${provider}.id`]: providerAuthData.id };\n    })\n  );\n\n  // Filter out null queries\n  const validQueries = queries.filter(query => query !== null);\n\n  if (!validQueries.length) {\n    return [];\n  }\n\n  // Perform database query\n  return config.database.find('_User', { $or: validQueries }, { limit: 2 });\n};\n\nconst hasMutatedAuthData = (authData, userAuthData) => {\n  if (!userAuthData) { return { hasMutatedAuthData: true, mutatedAuthData: authData }; }\n  const mutatedAuthData = {};\n  Object.keys(authData).forEach(provider => {\n    // Anonymous provider is not handled this way\n    if (provider === 'anonymous') { return; }\n    const providerData = authData[provider];\n    const userProviderAuthData = userAuthData[provider];\n    if (!isDeepStrictEqual(providerData, userProviderAuthData)) {\n      mutatedAuthData[provider] = providerData;\n    }\n  });\n  const hasMutatedAuthData = Object.keys(mutatedAuthData).length !== 0;\n  return { hasMutatedAuthData, mutatedAuthData };\n};\n\nconst checkIfUserHasProvidedConfiguredProvidersForLogin = (\n  req = {},\n  authData = {},\n  userAuthData = {},\n  config\n) => {\n  const savedUserProviders = Object.keys(userAuthData).map(provider => ({\n    name: provider,\n    adapter: config.authDataManager.getValidatorForProvider(provider).adapter,\n  }));\n\n  const hasProvidedASoloProvider = savedUserProviders.some(\n    provider =>\n      provider && provider.adapter && provider.adapter.policy === 'solo' && authData[provider.name]\n  );\n\n  // Solo providers can be considered as safe, so we do not have to check if the user needs\n  // to provide an additional provider to login. An auth adapter with \"solo\" (like webauthn) means\n  // no \"additional\" auth needs to be provided to login (like OTP, MFA)\n  if (hasProvidedASoloProvider) {\n    return;\n  }\n\n  const additionProvidersNotFound = [];\n  const hasProvidedAtLeastOneAdditionalProvider = savedUserProviders.some(provider => {\n    let policy = provider.adapter.policy;\n    if (typeof policy === 'function') {\n      const requestObject = {\n        ip: req.config.ip,\n        user: req.auth.user,\n        master: req.auth.isMaster,\n      };\n      policy = policy.call(provider.adapter, requestObject, userAuthData[provider.name]);\n    }\n    if (policy === 'additional') {\n      if (authData[provider.name]) {\n        return true;\n      } else {\n        // Push missing provider for error message\n        additionProvidersNotFound.push(provider.name);\n      }\n    }\n  });\n  if (hasProvidedAtLeastOneAdditionalProvider || !additionProvidersNotFound.length) {\n    return;\n  }\n\n  throw new Parse.Error(\n    Parse.Error.OTHER_CAUSE,\n    `Missing additional authData ${additionProvidersNotFound.join(',')}`\n  );\n};\n\n// Validate each authData step-by-step and return the provider responses\nconst handleAuthDataValidation = async (authData, req, foundUser) => {\n  let user;\n  if (foundUser) {\n    user = Parse.User.fromJSON({ className: '_User', ...foundUser });\n    // Find user by session and current objectId; only pass user if it's the current user or master key is provided\n  } else if (\n    (req.auth &&\n      req.auth.user &&\n      typeof req.getUserId === 'function' &&\n      req.getUserId() === req.auth.user.id) ||\n    (req.auth && req.auth.isMaster && typeof req.getUserId === 'function' && req.getUserId())\n  ) {\n    user = new Parse.User();\n    user.id = req.auth.isMaster ? req.getUserId() : req.auth.user.id;\n    await user.fetch({ useMasterKey: true });\n  }\n\n  const { updatedObject } = req.buildParseObjects();\n  const requestObject = getRequestObject(undefined, req.auth, updatedObject, user, req.config);\n  // Perform validation as step-by-step pipeline for better error consistency\n  // and also to avoid to trigger a provider (like OTP SMS) if another one fails\n  const acc = { authData: {}, authDataResponse: {} };\n  const authKeys = Object.keys(authData).sort();\n  for (const provider of authKeys) {\n    let method = '';\n    try {\n      if (authData[provider] === null) {\n        acc.authData[provider] = null;\n        continue;\n      }\n      const { validator } = req.config.authDataManager.getValidatorForProvider(provider) || {};\n      const authProvider = (req.config.auth || {})[provider] || {};\n      if (!validator || authProvider.enabled === false) {\n        throw new Parse.Error(\n          Parse.Error.UNSUPPORTED_SERVICE,\n          'This authentication method is unsupported.'\n        );\n      }\n      let validationResult = await validator(authData[provider], req, user, requestObject);\n      method = validationResult && validationResult.method;\n      requestObject.triggerName = method;\n      if (validationResult && validationResult.validator) {\n        validationResult = await validationResult.validator();\n      }\n      if (!validationResult) {\n        acc.authData[provider] = authData[provider];\n        continue;\n      }\n      if (!Object.keys(validationResult).length) {\n        acc.authData[provider] = authData[provider];\n        continue;\n      }\n\n      if (validationResult.response) {\n        acc.authDataResponse[provider] = validationResult.response;\n      }\n      // Some auth providers after initialization will avoid to replace authData already stored\n      if (!validationResult.doNotSave) {\n        acc.authData[provider] = validationResult.save || authData[provider];\n      }\n    } catch (err) {\n      const e = resolveError(err, {\n        code: Parse.Error.SCRIPT_FAILED,\n        message: 'Auth failed. Unknown error.',\n      });\n      const userString =\n        req.auth && req.auth.user ? req.auth.user.id : req.data.objectId || undefined;\n      logger.error(\n        `Failed running auth step ${method} for ${provider} for user ${userString} with Error: ` +\n          JSON.stringify(e),\n        {\n          authenticationStep: method,\n          error: e,\n          user: userString,\n          provider,\n        }\n      );\n      throw e;\n    }\n  }\n  return acc;\n};\n\nmodule.exports = {\n  Auth,\n  master,\n  maintenance,\n  nobody,\n  readOnly,\n  shouldUpdateSessionExpiry,\n  getAuthForSessionToken,\n  getAuthForLegacySessionToken,\n  findUsersWithAuthData,\n  hasMutatedAuthData,\n  checkIfUserHasProvidedConfiguredProvidersForLogin,\n  handleAuthDataValidation,\n};\n"],"mappings":"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","ignoreList":[]}
|