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
|
@@ -4,47 +4,32 @@ Object.defineProperty(exports, "__esModule", {
|
|
|
4
4
|
value: true
|
|
5
5
|
});
|
|
6
6
|
exports.default = exports.UsersRouter = void 0;
|
|
7
|
-
|
|
8
7
|
var _node = _interopRequireDefault(require("parse/node"));
|
|
9
|
-
|
|
10
8
|
var _Config = _interopRequireDefault(require("../Config"));
|
|
11
|
-
|
|
12
9
|
var _AccountLockout = _interopRequireDefault(require("../AccountLockout"));
|
|
13
|
-
|
|
14
10
|
var _ClassesRouter = _interopRequireDefault(require("./ClassesRouter"));
|
|
15
|
-
|
|
16
11
|
var _rest = _interopRequireDefault(require("../rest"));
|
|
17
|
-
|
|
18
12
|
var _Auth = _interopRequireDefault(require("../Auth"));
|
|
19
|
-
|
|
20
13
|
var _password = _interopRequireDefault(require("../password"));
|
|
21
|
-
|
|
22
14
|
var _triggers = require("../triggers");
|
|
23
|
-
|
|
24
15
|
var _middlewares = require("../middlewares");
|
|
25
|
-
|
|
26
16
|
var _RestWrite = _interopRequireDefault(require("../RestWrite"));
|
|
27
|
-
|
|
28
17
|
var _logger = require("../logger");
|
|
29
|
-
|
|
30
18
|
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
31
|
-
|
|
32
|
-
function
|
|
33
|
-
|
|
34
|
-
function
|
|
35
|
-
|
|
36
|
-
function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
|
|
37
|
-
|
|
19
|
+
function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); enumerableOnly && (symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; })), keys.push.apply(keys, symbols); } return keys; }
|
|
20
|
+
function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = null != arguments[i] ? arguments[i] : {}; i % 2 ? ownKeys(Object(source), !0).forEach(function (key) { _defineProperty(target, key, source[key]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)) : ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } return target; }
|
|
21
|
+
function _defineProperty(obj, key, value) { key = _toPropertyKey(key); if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
|
|
22
|
+
function _toPropertyKey(arg) { var key = _toPrimitive(arg, "string"); return typeof key === "symbol" ? key : String(key); }
|
|
23
|
+
function _toPrimitive(input, hint) { if (typeof input !== "object" || input === null) return input; var prim = input[Symbol.toPrimitive]; if (prim !== undefined) { var res = prim.call(input, hint || "default"); if (typeof res !== "object") return res; throw new TypeError("@@toPrimitive must return a primitive value."); } return (hint === "string" ? String : Number)(input); }
|
|
38
24
|
class UsersRouter extends _ClassesRouter.default {
|
|
39
25
|
className() {
|
|
40
26
|
return '_User';
|
|
41
27
|
}
|
|
28
|
+
|
|
42
29
|
/**
|
|
43
30
|
* Removes all "_" prefixed properties from an object, except "__type"
|
|
44
31
|
* @param {Object} obj An object.
|
|
45
32
|
*/
|
|
46
|
-
|
|
47
|
-
|
|
48
33
|
static removeHiddenProperties(obj) {
|
|
49
34
|
for (var key in obj) {
|
|
50
35
|
if (Object.prototype.hasOwnProperty.call(obj, key)) {
|
|
@@ -55,68 +40,61 @@ class UsersRouter extends _ClassesRouter.default {
|
|
|
55
40
|
}
|
|
56
41
|
}
|
|
57
42
|
}
|
|
43
|
+
|
|
58
44
|
/**
|
|
59
45
|
* After retrieving a user directly from the database, we need to remove the
|
|
60
46
|
* password from the object (for security), and fix an issue some SDKs have
|
|
61
47
|
* with null values
|
|
62
48
|
*/
|
|
63
|
-
|
|
64
|
-
|
|
65
49
|
_sanitizeAuthData(user) {
|
|
66
|
-
delete user.password;
|
|
67
|
-
// https://github.com/parse-community/parse-server/issues/935
|
|
50
|
+
delete user.password;
|
|
68
51
|
|
|
52
|
+
// Sometimes the authData still has null on that keys
|
|
53
|
+
// https://github.com/parse-community/parse-server/issues/935
|
|
69
54
|
if (user.authData) {
|
|
70
55
|
Object.keys(user.authData).forEach(provider => {
|
|
71
56
|
if (user.authData[provider] === null) {
|
|
72
57
|
delete user.authData[provider];
|
|
73
58
|
}
|
|
74
59
|
});
|
|
75
|
-
|
|
76
60
|
if (Object.keys(user.authData).length == 0) {
|
|
77
61
|
delete user.authData;
|
|
78
62
|
}
|
|
79
63
|
}
|
|
80
64
|
}
|
|
65
|
+
|
|
81
66
|
/**
|
|
82
67
|
* Validates a password request in login and verifyPassword
|
|
83
68
|
* @param {Object} req The request
|
|
84
69
|
* @returns {Object} User object
|
|
85
70
|
* @private
|
|
86
71
|
*/
|
|
87
|
-
|
|
88
|
-
|
|
89
72
|
_authenticateUserFromRequest(req) {
|
|
90
73
|
return new Promise((resolve, reject) => {
|
|
91
74
|
// Use query parameters instead if provided in url
|
|
92
75
|
let payload = req.body;
|
|
93
|
-
|
|
94
76
|
if (!payload.username && req.query && req.query.username || !payload.email && req.query && req.query.email) {
|
|
95
77
|
payload = req.query;
|
|
96
78
|
}
|
|
97
|
-
|
|
98
79
|
const {
|
|
99
80
|
username,
|
|
100
81
|
email,
|
|
101
82
|
password
|
|
102
|
-
} = payload;
|
|
83
|
+
} = payload;
|
|
103
84
|
|
|
85
|
+
// TODO: use the right error codes / descriptions.
|
|
104
86
|
if (!username && !email) {
|
|
105
87
|
throw new _node.default.Error(_node.default.Error.USERNAME_MISSING, 'username/email is required.');
|
|
106
88
|
}
|
|
107
|
-
|
|
108
89
|
if (!password) {
|
|
109
90
|
throw new _node.default.Error(_node.default.Error.PASSWORD_MISSING, 'password is required.');
|
|
110
91
|
}
|
|
111
|
-
|
|
112
92
|
if (typeof password !== 'string' || email && typeof email !== 'string' || username && typeof username !== 'string') {
|
|
113
93
|
throw new _node.default.Error(_node.default.Error.OBJECT_NOT_FOUND, 'Invalid username/password.');
|
|
114
94
|
}
|
|
115
|
-
|
|
116
95
|
let user;
|
|
117
96
|
let isValidPassword = false;
|
|
118
97
|
let query;
|
|
119
|
-
|
|
120
98
|
if (email && username) {
|
|
121
99
|
query = {
|
|
122
100
|
email,
|
|
@@ -135,12 +113,10 @@ class UsersRouter extends _ClassesRouter.default {
|
|
|
135
113
|
}]
|
|
136
114
|
};
|
|
137
115
|
}
|
|
138
|
-
|
|
139
116
|
return req.config.database.find('_User', query).then(results => {
|
|
140
117
|
if (!results.length) {
|
|
141
118
|
throw new _node.default.Error(_node.default.Error.OBJECT_NOT_FOUND, 'Invalid username/password.');
|
|
142
119
|
}
|
|
143
|
-
|
|
144
120
|
if (results.length > 1) {
|
|
145
121
|
// corner case where user1 has username == user2 email
|
|
146
122
|
req.config.loggerController.warn("There is a user which email is the same as another user's username, logging in based on username");
|
|
@@ -148,7 +124,6 @@ class UsersRouter extends _ClassesRouter.default {
|
|
|
148
124
|
} else {
|
|
149
125
|
user = results[0];
|
|
150
126
|
}
|
|
151
|
-
|
|
152
127
|
return _password.default.compare(password, user.password);
|
|
153
128
|
}).then(correct => {
|
|
154
129
|
isValidPassword = correct;
|
|
@@ -157,34 +132,28 @@ class UsersRouter extends _ClassesRouter.default {
|
|
|
157
132
|
}).then(() => {
|
|
158
133
|
if (!isValidPassword) {
|
|
159
134
|
throw new _node.default.Error(_node.default.Error.OBJECT_NOT_FOUND, 'Invalid username/password.');
|
|
160
|
-
}
|
|
135
|
+
}
|
|
136
|
+
// Ensure the user isn't locked out
|
|
161
137
|
// A locked out user won't be able to login
|
|
162
138
|
// To lock a user out, just set the ACL to `masterKey` only ({}).
|
|
163
139
|
// Empty ACL is OK
|
|
164
|
-
|
|
165
|
-
|
|
166
140
|
if (!req.auth.isMaster && user.ACL && Object.keys(user.ACL).length == 0) {
|
|
167
141
|
throw new _node.default.Error(_node.default.Error.OBJECT_NOT_FOUND, 'Invalid username/password.');
|
|
168
142
|
}
|
|
169
|
-
|
|
170
143
|
if (req.config.verifyUserEmails && req.config.preventLoginWithUnverifiedEmail && !user.emailVerified) {
|
|
171
144
|
throw new _node.default.Error(_node.default.Error.EMAIL_NOT_FOUND, 'User email is not verified.');
|
|
172
145
|
}
|
|
173
|
-
|
|
174
146
|
this._sanitizeAuthData(user);
|
|
175
|
-
|
|
176
147
|
return resolve(user);
|
|
177
148
|
}).catch(error => {
|
|
178
149
|
return reject(error);
|
|
179
150
|
});
|
|
180
151
|
});
|
|
181
152
|
}
|
|
182
|
-
|
|
183
153
|
handleMe(req) {
|
|
184
154
|
if (!req.info || !req.info.sessionToken) {
|
|
185
155
|
throw new _node.default.Error(_node.default.Error.INVALID_SESSION_TOKEN, 'Invalid session token');
|
|
186
156
|
}
|
|
187
|
-
|
|
188
157
|
const sessionToken = req.info.sessionToken;
|
|
189
158
|
return _rest.default.find(req.config, _Auth.default.master(req.config), '_Session', {
|
|
190
159
|
sessionToken
|
|
@@ -194,10 +163,11 @@ class UsersRouter extends _ClassesRouter.default {
|
|
|
194
163
|
if (!response.results || response.results.length == 0 || !response.results[0].user) {
|
|
195
164
|
throw new _node.default.Error(_node.default.Error.INVALID_SESSION_TOKEN, 'Invalid session token');
|
|
196
165
|
} else {
|
|
197
|
-
const user = response.results[0].user;
|
|
198
|
-
|
|
199
|
-
user.sessionToken = sessionToken;
|
|
166
|
+
const user = response.results[0].user;
|
|
167
|
+
// Send token back on the login, because SDKs expect that.
|
|
168
|
+
user.sessionToken = sessionToken;
|
|
200
169
|
|
|
170
|
+
// Remove hidden properties.
|
|
201
171
|
UsersRouter.removeHiddenProperties(user);
|
|
202
172
|
return {
|
|
203
173
|
response: user
|
|
@@ -205,28 +175,24 @@ class UsersRouter extends _ClassesRouter.default {
|
|
|
205
175
|
}
|
|
206
176
|
});
|
|
207
177
|
}
|
|
208
|
-
|
|
209
178
|
async handleLogIn(req) {
|
|
210
179
|
const user = await this._authenticateUserFromRequest(req);
|
|
211
|
-
const authData = req.body && req.body.authData;
|
|
212
|
-
|
|
180
|
+
const authData = req.body && req.body.authData;
|
|
181
|
+
// Check if user has provided their required auth providers
|
|
213
182
|
_Auth.default.checkIfUserHasProvidedConfiguredProvidersForLogin(authData, user.authData, req.config);
|
|
214
|
-
|
|
215
183
|
let authDataResponse;
|
|
216
184
|
let validatedAuthData;
|
|
217
|
-
|
|
218
185
|
if (authData) {
|
|
219
186
|
const res = await _Auth.default.handleAuthDataValidation(authData, new _RestWrite.default(req.config, req.auth, '_User', {
|
|
220
187
|
objectId: user.objectId
|
|
221
188
|
}, req.body, user, req.info.clientSDK, req.info.context), user);
|
|
222
189
|
authDataResponse = res.authDataResponse;
|
|
223
190
|
validatedAuthData = res.authData;
|
|
224
|
-
}
|
|
225
|
-
|
|
191
|
+
}
|
|
226
192
|
|
|
193
|
+
// handle password expiry policy
|
|
227
194
|
if (req.config.passwordPolicy && req.config.passwordPolicy.maxPasswordAge) {
|
|
228
195
|
let changedAt = user._password_changed_at;
|
|
229
|
-
|
|
230
196
|
if (!changedAt) {
|
|
231
197
|
// password was created before expiry policy was enabled.
|
|
232
198
|
// simply update _User object so that it will start enforcing from now
|
|
@@ -240,23 +206,25 @@ class UsersRouter extends _ClassesRouter.default {
|
|
|
240
206
|
// check whether the password has expired
|
|
241
207
|
if (changedAt.__type == 'Date') {
|
|
242
208
|
changedAt = new Date(changedAt.iso);
|
|
243
|
-
}
|
|
244
|
-
|
|
245
|
-
|
|
209
|
+
}
|
|
210
|
+
// Calculate the expiry time.
|
|
246
211
|
const expiresAt = new Date(changedAt.getTime() + 86400000 * req.config.passwordPolicy.maxPasswordAge);
|
|
247
|
-
if (expiresAt < new Date())
|
|
212
|
+
if (expiresAt < new Date())
|
|
213
|
+
// fail of current time is past password expiry time
|
|
248
214
|
throw new _node.default.Error(_node.default.Error.OBJECT_NOT_FOUND, 'Your password has expired. Please reset your password.');
|
|
249
215
|
}
|
|
250
|
-
}
|
|
251
|
-
|
|
216
|
+
}
|
|
252
217
|
|
|
218
|
+
// Remove hidden properties.
|
|
253
219
|
UsersRouter.removeHiddenProperties(user);
|
|
254
|
-
req.config.filesController.expandFilesInObject(req.config, user);
|
|
220
|
+
req.config.filesController.expandFilesInObject(req.config, user);
|
|
255
221
|
|
|
222
|
+
// Before login trigger; throws if failure
|
|
256
223
|
await (0, _triggers.maybeRunTrigger)(_triggers.Types.beforeLogin, req.auth, _node.default.User.fromJSON(Object.assign({
|
|
257
224
|
className: '_User'
|
|
258
|
-
}, user)), null, req.config);
|
|
225
|
+
}, user)), null, req.config);
|
|
259
226
|
|
|
227
|
+
// If we have some new validated authData update directly
|
|
260
228
|
if (validatedAuthData && Object.keys(validatedAuthData).length) {
|
|
261
229
|
await req.config.database.update('_User', {
|
|
262
230
|
objectId: user.objectId
|
|
@@ -264,7 +232,6 @@ class UsersRouter extends _ClassesRouter.default {
|
|
|
264
232
|
authData: validatedAuthData
|
|
265
233
|
}, {});
|
|
266
234
|
}
|
|
267
|
-
|
|
268
235
|
const {
|
|
269
236
|
sessionData,
|
|
270
237
|
createSession
|
|
@@ -276,26 +243,22 @@ class UsersRouter extends _ClassesRouter.default {
|
|
|
276
243
|
},
|
|
277
244
|
installationId: req.info.installationId
|
|
278
245
|
});
|
|
279
|
-
|
|
280
246
|
user.sessionToken = sessionData.sessionToken;
|
|
281
247
|
await createSession();
|
|
282
|
-
|
|
283
248
|
const afterLoginUser = _node.default.User.fromJSON(Object.assign({
|
|
284
249
|
className: '_User'
|
|
285
250
|
}, user));
|
|
286
|
-
|
|
287
|
-
(0, _triggers.maybeRunTrigger)(_triggers.Types.afterLogin, _objectSpread(_objectSpread({}, req.auth), {}, {
|
|
251
|
+
await (0, _triggers.maybeRunTrigger)(_triggers.Types.afterLogin, _objectSpread(_objectSpread({}, req.auth), {}, {
|
|
288
252
|
user: afterLoginUser
|
|
289
253
|
}), afterLoginUser, null, req.config);
|
|
290
|
-
|
|
291
254
|
if (authDataResponse) {
|
|
292
255
|
user.authDataResponse = authDataResponse;
|
|
293
256
|
}
|
|
294
|
-
|
|
295
257
|
return {
|
|
296
258
|
response: user
|
|
297
259
|
};
|
|
298
260
|
}
|
|
261
|
+
|
|
299
262
|
/**
|
|
300
263
|
* This allows master-key clients to create user sessions without access to
|
|
301
264
|
* user credentials. This enables systems that can authenticate access another
|
|
@@ -310,30 +273,22 @@ class UsersRouter extends _ClassesRouter.default {
|
|
|
310
273
|
* immediate use cases suggest /loginAs could be used for semantically
|
|
311
274
|
* different reasons from /login
|
|
312
275
|
*/
|
|
313
|
-
|
|
314
|
-
|
|
315
276
|
async handleLogInAs(req) {
|
|
316
277
|
if (!req.auth.isMaster) {
|
|
317
278
|
throw new _node.default.Error(_node.default.Error.OPERATION_FORBIDDEN, 'master key is required');
|
|
318
279
|
}
|
|
319
|
-
|
|
320
280
|
const userId = req.body.userId || req.query.userId;
|
|
321
|
-
|
|
322
281
|
if (!userId) {
|
|
323
282
|
throw new _node.default.Error(_node.default.Error.INVALID_VALUE, 'userId must not be empty, null, or undefined');
|
|
324
283
|
}
|
|
325
|
-
|
|
326
284
|
const queryResults = await req.config.database.find('_User', {
|
|
327
285
|
objectId: userId
|
|
328
286
|
});
|
|
329
287
|
const user = queryResults[0];
|
|
330
|
-
|
|
331
288
|
if (!user) {
|
|
332
289
|
throw new _node.default.Error(_node.default.Error.OBJECT_NOT_FOUND, 'user not found');
|
|
333
290
|
}
|
|
334
|
-
|
|
335
291
|
this._sanitizeAuthData(user);
|
|
336
|
-
|
|
337
292
|
const {
|
|
338
293
|
sessionData,
|
|
339
294
|
createSession
|
|
@@ -345,14 +300,12 @@ class UsersRouter extends _ClassesRouter.default {
|
|
|
345
300
|
},
|
|
346
301
|
installationId: req.info.installationId
|
|
347
302
|
});
|
|
348
|
-
|
|
349
303
|
user.sessionToken = sessionData.sessionToken;
|
|
350
304
|
await createSession();
|
|
351
305
|
return {
|
|
352
306
|
response: user
|
|
353
307
|
};
|
|
354
308
|
}
|
|
355
|
-
|
|
356
309
|
handleVerifyPassword(req) {
|
|
357
310
|
return this._authenticateUserFromRequest(req).then(user => {
|
|
358
311
|
// Remove hidden properties.
|
|
@@ -364,38 +317,23 @@ class UsersRouter extends _ClassesRouter.default {
|
|
|
364
317
|
throw error;
|
|
365
318
|
});
|
|
366
319
|
}
|
|
367
|
-
|
|
368
|
-
handleLogOut(req) {
|
|
320
|
+
async handleLogOut(req) {
|
|
369
321
|
const success = {
|
|
370
322
|
response: {}
|
|
371
323
|
};
|
|
372
|
-
|
|
373
324
|
if (req.info && req.info.sessionToken) {
|
|
374
|
-
|
|
325
|
+
const records = await _rest.default.find(req.config, _Auth.default.master(req.config), '_Session', {
|
|
375
326
|
sessionToken: req.info.sessionToken
|
|
376
|
-
}, undefined, req.info.clientSDK, req.info.context)
|
|
377
|
-
|
|
378
|
-
|
|
379
|
-
|
|
380
|
-
|
|
381
|
-
|
|
382
|
-
|
|
383
|
-
}
|
|
384
|
-
|
|
385
|
-
return Promise.resolve(success);
|
|
386
|
-
});
|
|
327
|
+
}, undefined, req.info.clientSDK, req.info.context);
|
|
328
|
+
if (records.results && records.results.length) {
|
|
329
|
+
await _rest.default.del(req.config, _Auth.default.master(req.config), '_Session', records.results[0].objectId, req.info.context);
|
|
330
|
+
await (0, _triggers.maybeRunTrigger)(_triggers.Types.afterLogout, req.auth, _node.default.Session.fromJSON(Object.assign({
|
|
331
|
+
className: '_Session'
|
|
332
|
+
}, records.results[0])), null, req.config);
|
|
333
|
+
}
|
|
387
334
|
}
|
|
388
|
-
|
|
389
|
-
return Promise.resolve(success);
|
|
390
|
-
}
|
|
391
|
-
|
|
392
|
-
_runAfterLogoutTrigger(req, session) {
|
|
393
|
-
// After logout trigger
|
|
394
|
-
(0, _triggers.maybeRunTrigger)(_triggers.Types.afterLogout, req.auth, _node.default.Session.fromJSON(Object.assign({
|
|
395
|
-
className: '_Session'
|
|
396
|
-
}, session)), null, req.config);
|
|
335
|
+
return success;
|
|
397
336
|
}
|
|
398
|
-
|
|
399
337
|
_throwOnBadEmailConfig(req) {
|
|
400
338
|
try {
|
|
401
339
|
_Config.default.validateEmailConfiguration({
|
|
@@ -414,22 +352,17 @@ class UsersRouter extends _ClassesRouter.default {
|
|
|
414
352
|
}
|
|
415
353
|
}
|
|
416
354
|
}
|
|
417
|
-
|
|
418
355
|
handleResetRequest(req) {
|
|
419
356
|
this._throwOnBadEmailConfig(req);
|
|
420
|
-
|
|
421
357
|
const {
|
|
422
358
|
email
|
|
423
359
|
} = req.body;
|
|
424
|
-
|
|
425
360
|
if (!email) {
|
|
426
361
|
throw new _node.default.Error(_node.default.Error.EMAIL_MISSING, 'you must provide an email');
|
|
427
362
|
}
|
|
428
|
-
|
|
429
363
|
if (typeof email !== 'string') {
|
|
430
364
|
throw new _node.default.Error(_node.default.Error.INVALID_EMAIL_ADDRESS, 'you must provide a valid email string');
|
|
431
365
|
}
|
|
432
|
-
|
|
433
366
|
const userController = req.config.userController;
|
|
434
367
|
return userController.sendPasswordResetEmail(email).then(() => {
|
|
435
368
|
return Promise.resolve({
|
|
@@ -447,37 +380,30 @@ class UsersRouter extends _ClassesRouter.default {
|
|
|
447
380
|
}
|
|
448
381
|
});
|
|
449
382
|
}
|
|
450
|
-
|
|
451
383
|
handleVerificationEmailRequest(req) {
|
|
452
384
|
this._throwOnBadEmailConfig(req);
|
|
453
|
-
|
|
454
385
|
const {
|
|
455
386
|
email
|
|
456
387
|
} = req.body;
|
|
457
|
-
|
|
458
388
|
if (!email) {
|
|
459
389
|
throw new _node.default.Error(_node.default.Error.EMAIL_MISSING, 'you must provide an email');
|
|
460
390
|
}
|
|
461
|
-
|
|
462
391
|
if (typeof email !== 'string') {
|
|
463
392
|
throw new _node.default.Error(_node.default.Error.INVALID_EMAIL_ADDRESS, 'you must provide a valid email string');
|
|
464
393
|
}
|
|
465
|
-
|
|
466
394
|
return req.config.database.find('_User', {
|
|
467
395
|
email: email
|
|
468
396
|
}).then(results => {
|
|
469
397
|
if (!results.length || results.length < 1) {
|
|
470
398
|
throw new _node.default.Error(_node.default.Error.EMAIL_NOT_FOUND, `No user found with email ${email}`);
|
|
471
399
|
}
|
|
400
|
+
const user = results[0];
|
|
472
401
|
|
|
473
|
-
|
|
474
|
-
|
|
402
|
+
// remove password field, messes with saving on postgres
|
|
475
403
|
delete user.password;
|
|
476
|
-
|
|
477
404
|
if (user.emailVerified) {
|
|
478
405
|
throw new _node.default.Error(_node.default.Error.OTHER_CAUSE, `Email ${email} is already verified.`);
|
|
479
406
|
}
|
|
480
|
-
|
|
481
407
|
const userController = req.config.userController;
|
|
482
408
|
return userController.regenerateEmailVerifyToken(user).then(() => {
|
|
483
409
|
userController.sendVerificationEmail(user);
|
|
@@ -487,7 +413,6 @@ class UsersRouter extends _ClassesRouter.default {
|
|
|
487
413
|
});
|
|
488
414
|
});
|
|
489
415
|
}
|
|
490
|
-
|
|
491
416
|
async handleChallenge(req) {
|
|
492
417
|
const {
|
|
493
418
|
username,
|
|
@@ -495,101 +420,85 @@ class UsersRouter extends _ClassesRouter.default {
|
|
|
495
420
|
password,
|
|
496
421
|
authData,
|
|
497
422
|
challengeData
|
|
498
|
-
} = req.body;
|
|
423
|
+
} = req.body;
|
|
499
424
|
|
|
425
|
+
// if username or email provided with password try to authenticate the user by username
|
|
500
426
|
let user;
|
|
501
|
-
|
|
502
427
|
if (username || email) {
|
|
503
428
|
if (!password) {
|
|
504
429
|
throw new _node.default.Error(_node.default.Error.OTHER_CAUSE, 'You provided username or email, you need to also provide password.');
|
|
505
430
|
}
|
|
506
|
-
|
|
507
431
|
user = await this._authenticateUserFromRequest(req);
|
|
508
432
|
}
|
|
509
|
-
|
|
510
433
|
if (!challengeData) {
|
|
511
434
|
throw new _node.default.Error(_node.default.Error.OTHER_CAUSE, 'Nothing to challenge.');
|
|
512
435
|
}
|
|
513
|
-
|
|
514
436
|
if (typeof challengeData !== 'object') {
|
|
515
437
|
throw new _node.default.Error(_node.default.Error.OTHER_CAUSE, 'challengeData should be an object.');
|
|
516
438
|
}
|
|
517
|
-
|
|
518
439
|
let request;
|
|
519
|
-
let parseUser;
|
|
440
|
+
let parseUser;
|
|
520
441
|
|
|
442
|
+
// Try to find user by authData
|
|
521
443
|
if (authData) {
|
|
522
444
|
if (typeof authData !== 'object') {
|
|
523
445
|
throw new _node.default.Error(_node.default.Error.OTHER_CAUSE, 'authData should be an object.');
|
|
524
446
|
}
|
|
525
|
-
|
|
526
447
|
if (user) {
|
|
527
448
|
throw new _node.default.Error(_node.default.Error.OTHER_CAUSE, 'You cannot provide username/email and authData, only use one identification method.');
|
|
528
449
|
}
|
|
529
|
-
|
|
530
450
|
if (Object.keys(authData).filter(key => authData[key].id).length > 1) {
|
|
531
451
|
throw new _node.default.Error(_node.default.Error.OTHER_CAUSE, 'You cannot provide more than one authData provider with an id.');
|
|
532
452
|
}
|
|
533
|
-
|
|
534
453
|
const results = await _Auth.default.findUsersWithAuthData(req.config, authData);
|
|
535
|
-
|
|
536
454
|
try {
|
|
537
455
|
if (!results[0] || results.length > 1) {
|
|
538
456
|
throw new _node.default.Error(_node.default.Error.OBJECT_NOT_FOUND, 'User not found.');
|
|
539
|
-
}
|
|
540
|
-
|
|
541
|
-
|
|
457
|
+
}
|
|
458
|
+
// Find the provider used to find the user
|
|
542
459
|
const provider = Object.keys(authData).find(key => authData[key].id);
|
|
543
460
|
parseUser = _node.default.User.fromJSON(_objectSpread({
|
|
544
461
|
className: '_User'
|
|
545
462
|
}, results[0]));
|
|
546
463
|
request = (0, _triggers.getRequestObject)(undefined, req.auth, parseUser, parseUser, req.config);
|
|
547
|
-
request.isChallenge = true;
|
|
548
|
-
|
|
464
|
+
request.isChallenge = true;
|
|
465
|
+
// Validate authData used to identify the user to avoid brute-force attack on `id`
|
|
549
466
|
const {
|
|
550
467
|
validator
|
|
551
468
|
} = req.config.authDataManager.getValidatorForProvider(provider);
|
|
552
469
|
const validatorResponse = await validator(authData[provider], req, parseUser, request);
|
|
553
|
-
|
|
554
470
|
if (validatorResponse && validatorResponse.validator) {
|
|
555
471
|
await validatorResponse.validator();
|
|
556
472
|
}
|
|
557
473
|
} catch (e) {
|
|
558
474
|
// Rewrite the error to avoid guess id attack
|
|
559
475
|
_logger.logger.error(e);
|
|
560
|
-
|
|
561
476
|
throw new _node.default.Error(_node.default.Error.OBJECT_NOT_FOUND, 'User not found.');
|
|
562
477
|
}
|
|
563
478
|
}
|
|
564
|
-
|
|
565
479
|
if (!parseUser) {
|
|
566
480
|
parseUser = user ? _node.default.User.fromJSON(_objectSpread({
|
|
567
481
|
className: '_User'
|
|
568
482
|
}, user)) : undefined;
|
|
569
483
|
}
|
|
570
|
-
|
|
571
484
|
if (!request) {
|
|
572
485
|
request = (0, _triggers.getRequestObject)(undefined, req.auth, parseUser, parseUser, req.config);
|
|
573
486
|
request.isChallenge = true;
|
|
574
487
|
}
|
|
575
|
-
|
|
576
|
-
|
|
488
|
+
const acc = {};
|
|
489
|
+
// Execute challenge step-by-step with consistent order for better error feedback
|
|
577
490
|
// and to avoid to trigger others challenges if one of them fails
|
|
578
|
-
|
|
579
491
|
for (const provider of Object.keys(challengeData).sort()) {
|
|
580
492
|
try {
|
|
581
493
|
const authAdapter = req.config.authDataManager.getValidatorForProvider(provider);
|
|
582
|
-
|
|
583
494
|
if (!authAdapter) {
|
|
584
495
|
continue;
|
|
585
496
|
}
|
|
586
|
-
|
|
587
497
|
const {
|
|
588
498
|
adapter: {
|
|
589
499
|
challenge
|
|
590
500
|
}
|
|
591
501
|
} = authAdapter;
|
|
592
|
-
|
|
593
502
|
if (typeof challenge === 'function') {
|
|
594
503
|
const providerChallengeResponse = await challenge(challengeData[provider], authData && authData[provider], req.config.auth[provider], request);
|
|
595
504
|
acc[provider] = providerChallengeResponse || true;
|
|
@@ -600,25 +509,21 @@ class UsersRouter extends _ClassesRouter.default {
|
|
|
600
509
|
message: 'Challenge failed. Unknown error.'
|
|
601
510
|
});
|
|
602
511
|
const userString = req.auth && req.auth.user ? req.auth.user.id : undefined;
|
|
603
|
-
|
|
604
512
|
_logger.logger.error(`Failed running auth step challenge for ${provider} for user ${userString} with Error: ` + JSON.stringify(e), {
|
|
605
513
|
authenticationStep: 'challenge',
|
|
606
514
|
error: e,
|
|
607
515
|
user: userString,
|
|
608
516
|
provider
|
|
609
517
|
});
|
|
610
|
-
|
|
611
518
|
throw e;
|
|
612
519
|
}
|
|
613
520
|
}
|
|
614
|
-
|
|
615
521
|
return {
|
|
616
522
|
response: {
|
|
617
523
|
challengeData: acc
|
|
618
524
|
}
|
|
619
525
|
};
|
|
620
526
|
}
|
|
621
|
-
|
|
622
527
|
mountRoutes() {
|
|
623
528
|
this.route('GET', '/users', req => {
|
|
624
529
|
return this.handleFind(req);
|
|
@@ -663,10 +568,8 @@ class UsersRouter extends _ClassesRouter.default {
|
|
|
663
568
|
return this.handleChallenge(req);
|
|
664
569
|
});
|
|
665
570
|
}
|
|
666
|
-
|
|
667
571
|
}
|
|
668
|
-
|
|
669
572
|
exports.UsersRouter = UsersRouter;
|
|
670
573
|
var _default = UsersRouter;
|
|
671
574
|
exports.default = _default;
|
|
672
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/Routers/UsersRouter.js"],"names":["UsersRouter","ClassesRouter","className","removeHiddenProperties","obj","key","Object","prototype","hasOwnProperty","call","test","_sanitizeAuthData","user","password","authData","keys","forEach","provider","length","_authenticateUserFromRequest","req","Promise","resolve","reject","payload","body","username","query","email","Parse","Error","USERNAME_MISSING","PASSWORD_MISSING","OBJECT_NOT_FOUND","isValidPassword","$or","config","database","find","then","results","loggerController","warn","filter","passwordCrypto","compare","correct","accountLockoutPolicy","AccountLockout","handleLoginAttempt","auth","isMaster","ACL","verifyUserEmails","preventLoginWithUnverifiedEmail","emailVerified","EMAIL_NOT_FOUND","catch","error","handleMe","info","sessionToken","INVALID_SESSION_TOKEN","rest","Auth","master","include","clientSDK","context","response","handleLogIn","checkIfUserHasProvidedConfiguredProvidersForLogin","authDataResponse","validatedAuthData","res","handleAuthDataValidation","RestWrite","objectId","passwordPolicy","maxPasswordAge","changedAt","_password_changed_at","Date","update","_encode","__type","iso","expiresAt","getTime","filesController","expandFilesInObject","TriggerTypes","beforeLogin","User","fromJSON","assign","sessionData","createSession","userId","createdWith","action","authProvider","installationId","afterLoginUser","afterLogin","handleLogInAs","OPERATION_FORBIDDEN","INVALID_VALUE","queryResults","handleVerifyPassword","handleLogOut","success","undefined","records","del","_runAfterLogoutTrigger","session","afterLogout","Session","_throwOnBadEmailConfig","Config","validateEmailConfiguration","emailAdapter","userController","adapter","appName","publicServerURL","emailVerifyTokenValidityDuration","emailVerifyTokenReuseIfValid","e","INTERNAL_SERVER_ERROR","handleResetRequest","EMAIL_MISSING","INVALID_EMAIL_ADDRESS","sendPasswordResetEmail","err","code","handleVerificationEmailRequest","OTHER_CAUSE","regenerateEmailVerifyToken","sendVerificationEmail","handleChallenge","challengeData","request","parseUser","id","findUsersWithAuthData","isChallenge","validator","authDataManager","getValidatorForProvider","validatorResponse","logger","acc","sort","authAdapter","challenge","providerChallengeResponse","SCRIPT_FAILED","message","userString","JSON","stringify","authenticationStep","mountRoutes","route","handleFind","promiseEnsureIdempotency","handleCreate","handleGet","handleUpdate","handleDelete"],"mappings":"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","sourcesContent":["// These methods handle the User-related routes.\n\nimport Parse from 'parse/node';\nimport Config from '../Config';\nimport AccountLockout from '../AccountLockout';\nimport ClassesRouter from './ClassesRouter';\nimport rest from '../rest';\nimport Auth from '../Auth';\nimport passwordCrypto from '../password';\nimport {\n  maybeRunTrigger,\n  Types as TriggerTypes,\n  getRequestObject,\n  resolveError,\n} from '../triggers';\nimport { promiseEnsureIdempotency } from '../middlewares';\nimport RestWrite from '../RestWrite';\nimport { logger } from '../logger';\n\nexport class UsersRouter extends ClassesRouter {\n  className() {\n    return '_User';\n  }\n\n  /**\n   * Removes all \"_\" prefixed properties from an object, except \"__type\"\n   * @param {Object} obj An object.\n   */\n  static removeHiddenProperties(obj) {\n    for (var key in obj) {\n      if (Object.prototype.hasOwnProperty.call(obj, key)) {\n        // Regexp comes from Parse.Object.prototype.validate\n        if (key !== '__type' && !/^[A-Za-z][0-9A-Za-z_]*$/.test(key)) {\n          delete obj[key];\n        }\n      }\n    }\n  }\n\n  /**\n   * After retrieving a user directly from the database, we need to remove the\n   * password from the object (for security), and fix an issue some SDKs have\n   * with null values\n   */\n  _sanitizeAuthData(user) {\n    delete user.password;\n\n    // Sometimes the authData still has null on that keys\n    // https://github.com/parse-community/parse-server/issues/935\n    if (user.authData) {\n      Object.keys(user.authData).forEach(provider => {\n        if (user.authData[provider] === null) {\n          delete user.authData[provider];\n        }\n      });\n      if (Object.keys(user.authData).length == 0) {\n        delete user.authData;\n      }\n    }\n  }\n\n  /**\n   * Validates a password request in login and verifyPassword\n   * @param {Object} req The request\n   * @returns {Object} User object\n   * @private\n   */\n  _authenticateUserFromRequest(req) {\n    return new Promise((resolve, reject) => {\n      // Use query parameters instead if provided in url\n      let payload = req.body;\n      if (\n        (!payload.username && req.query && req.query.username) ||\n        (!payload.email && req.query && req.query.email)\n      ) {\n        payload = req.query;\n      }\n      const { username, email, password } = payload;\n\n      // TODO: use the right error codes / descriptions.\n      if (!username && !email) {\n        throw new Parse.Error(Parse.Error.USERNAME_MISSING, 'username/email is required.');\n      }\n      if (!password) {\n        throw new Parse.Error(Parse.Error.PASSWORD_MISSING, 'password is required.');\n      }\n      if (\n        typeof password !== 'string' ||\n        (email && typeof email !== 'string') ||\n        (username && typeof username !== 'string')\n      ) {\n        throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Invalid username/password.');\n      }\n\n      let user;\n      let isValidPassword = false;\n      let query;\n      if (email && username) {\n        query = { email, username };\n      } else if (email) {\n        query = { email };\n      } else {\n        query = { $or: [{ username }, { email: username }] };\n      }\n      return req.config.database\n        .find('_User', query)\n        .then(results => {\n          if (!results.length) {\n            throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Invalid username/password.');\n          }\n\n          if (results.length > 1) {\n            // corner case where user1 has username == user2 email\n            req.config.loggerController.warn(\n              \"There is a user which email is the same as another user's username, logging in based on username\"\n            );\n            user = results.filter(user => user.username === username)[0];\n          } else {\n            user = results[0];\n          }\n\n          return passwordCrypto.compare(password, user.password);\n        })\n        .then(correct => {\n          isValidPassword = correct;\n          const accountLockoutPolicy = new AccountLockout(user, req.config);\n          return accountLockoutPolicy.handleLoginAttempt(isValidPassword);\n        })\n        .then(() => {\n          if (!isValidPassword) {\n            throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Invalid username/password.');\n          }\n          // Ensure the user isn't locked out\n          // A locked out user won't be able to login\n          // To lock a user out, just set the ACL to `masterKey` only  ({}).\n          // Empty ACL is OK\n          if (!req.auth.isMaster && user.ACL && Object.keys(user.ACL).length == 0) {\n            throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Invalid username/password.');\n          }\n          if (\n            req.config.verifyUserEmails &&\n            req.config.preventLoginWithUnverifiedEmail &&\n            !user.emailVerified\n          ) {\n            throw new Parse.Error(Parse.Error.EMAIL_NOT_FOUND, 'User email is not verified.');\n          }\n\n          this._sanitizeAuthData(user);\n\n          return resolve(user);\n        })\n        .catch(error => {\n          return reject(error);\n        });\n    });\n  }\n\n  handleMe(req) {\n    if (!req.info || !req.info.sessionToken) {\n      throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'Invalid session token');\n    }\n    const sessionToken = req.info.sessionToken;\n    return rest\n      .find(\n        req.config,\n        Auth.master(req.config),\n        '_Session',\n        { sessionToken },\n        { include: 'user' },\n        req.info.clientSDK,\n        req.info.context\n      )\n      .then(response => {\n        if (!response.results || response.results.length == 0 || !response.results[0].user) {\n          throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'Invalid session token');\n        } else {\n          const user = response.results[0].user;\n          // Send token back on the login, because SDKs expect that.\n          user.sessionToken = sessionToken;\n\n          // Remove hidden properties.\n          UsersRouter.removeHiddenProperties(user);\n          return { response: user };\n        }\n      });\n  }\n\n  async handleLogIn(req) {\n    const user = await this._authenticateUserFromRequest(req);\n    const authData = req.body && req.body.authData;\n    // Check if user has provided their required auth providers\n    Auth.checkIfUserHasProvidedConfiguredProvidersForLogin(authData, user.authData, req.config);\n\n    let authDataResponse;\n    let validatedAuthData;\n    if (authData) {\n      const res = await Auth.handleAuthDataValidation(\n        authData,\n        new RestWrite(\n          req.config,\n          req.auth,\n          '_User',\n          { objectId: user.objectId },\n          req.body,\n          user,\n          req.info.clientSDK,\n          req.info.context\n        ),\n        user\n      );\n      authDataResponse = res.authDataResponse;\n      validatedAuthData = res.authData;\n    }\n\n    // handle password expiry policy\n    if (req.config.passwordPolicy && req.config.passwordPolicy.maxPasswordAge) {\n      let changedAt = user._password_changed_at;\n\n      if (!changedAt) {\n        // password was created before expiry policy was enabled.\n        // simply update _User object so that it will start enforcing from now\n        changedAt = new Date();\n        req.config.database.update(\n          '_User',\n          { username: user.username },\n          { _password_changed_at: Parse._encode(changedAt) }\n        );\n      } else {\n        // check whether the password has expired\n        if (changedAt.__type == 'Date') {\n          changedAt = new Date(changedAt.iso);\n        }\n        // Calculate the expiry time.\n        const expiresAt = new Date(\n          changedAt.getTime() + 86400000 * req.config.passwordPolicy.maxPasswordAge\n        );\n        if (expiresAt < new Date())\n          // fail of current time is past password expiry time\n          throw new Parse.Error(\n            Parse.Error.OBJECT_NOT_FOUND,\n            'Your password has expired. Please reset your password.'\n          );\n      }\n    }\n\n    // Remove hidden properties.\n    UsersRouter.removeHiddenProperties(user);\n\n    req.config.filesController.expandFilesInObject(req.config, user);\n\n    // Before login trigger; throws if failure\n    await maybeRunTrigger(\n      TriggerTypes.beforeLogin,\n      req.auth,\n      Parse.User.fromJSON(Object.assign({ className: '_User' }, user)),\n      null,\n      req.config\n    );\n\n    // If we have some new validated authData update directly\n    if (validatedAuthData && Object.keys(validatedAuthData).length) {\n      await req.config.database.update(\n        '_User',\n        { objectId: user.objectId },\n        { authData: validatedAuthData },\n        {}\n      );\n    }\n\n    const { sessionData, createSession } = RestWrite.createSession(req.config, {\n      userId: user.objectId,\n      createdWith: {\n        action: 'login',\n        authProvider: 'password',\n      },\n      installationId: req.info.installationId,\n    });\n\n    user.sessionToken = sessionData.sessionToken;\n\n    await createSession();\n\n    const afterLoginUser = Parse.User.fromJSON(Object.assign({ className: '_User' }, user));\n    maybeRunTrigger(\n      TriggerTypes.afterLogin,\n      { ...req.auth, user: afterLoginUser },\n      afterLoginUser,\n      null,\n      req.config\n    );\n\n    if (authDataResponse) {\n      user.authDataResponse = authDataResponse;\n    }\n\n    return { response: user };\n  }\n\n  /**\n   * This allows master-key clients to create user sessions without access to\n   * user credentials. This enables systems that can authenticate access another\n   * way (API key, app administrators) to act on a user's behalf.\n   *\n   * We create a new session rather than looking for an existing session; we\n   * want this to work in situations where the user is logged out on all\n   * devices, since this can be used by automated systems acting on the user's\n   * behalf.\n   *\n   * For the moment, we're omitting event hooks and lockout checks, since\n   * immediate use cases suggest /loginAs could be used for semantically\n   * different reasons from /login\n   */\n  async handleLogInAs(req) {\n    if (!req.auth.isMaster) {\n      throw new Parse.Error(Parse.Error.OPERATION_FORBIDDEN, 'master key is required');\n    }\n\n    const userId = req.body.userId || req.query.userId;\n    if (!userId) {\n      throw new Parse.Error(\n        Parse.Error.INVALID_VALUE,\n        'userId must not be empty, null, or undefined'\n      );\n    }\n\n    const queryResults = await req.config.database.find('_User', { objectId: userId });\n    const user = queryResults[0];\n    if (!user) {\n      throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'user not found');\n    }\n\n    this._sanitizeAuthData(user);\n\n    const { sessionData, createSession } = RestWrite.createSession(req.config, {\n      userId,\n      createdWith: {\n        action: 'login',\n        authProvider: 'masterkey',\n      },\n      installationId: req.info.installationId,\n    });\n\n    user.sessionToken = sessionData.sessionToken;\n\n    await createSession();\n\n    return { response: user };\n  }\n\n  handleVerifyPassword(req) {\n    return this._authenticateUserFromRequest(req)\n      .then(user => {\n        // Remove hidden properties.\n        UsersRouter.removeHiddenProperties(user);\n\n        return { response: user };\n      })\n      .catch(error => {\n        throw error;\n      });\n  }\n\n  handleLogOut(req) {\n    const success = { response: {} };\n    if (req.info && req.info.sessionToken) {\n      return rest\n        .find(\n          req.config,\n          Auth.master(req.config),\n          '_Session',\n          { sessionToken: req.info.sessionToken },\n          undefined,\n          req.info.clientSDK,\n          req.info.context\n        )\n        .then(records => {\n          if (records.results && records.results.length) {\n            return rest\n              .del(\n                req.config,\n                Auth.master(req.config),\n                '_Session',\n                records.results[0].objectId,\n                req.info.context\n              )\n              .then(() => {\n                this._runAfterLogoutTrigger(req, records.results[0]);\n                return Promise.resolve(success);\n              });\n          }\n          return Promise.resolve(success);\n        });\n    }\n    return Promise.resolve(success);\n  }\n\n  _runAfterLogoutTrigger(req, session) {\n    // After logout trigger\n    maybeRunTrigger(\n      TriggerTypes.afterLogout,\n      req.auth,\n      Parse.Session.fromJSON(Object.assign({ className: '_Session' }, session)),\n      null,\n      req.config\n    );\n  }\n\n  _throwOnBadEmailConfig(req) {\n    try {\n      Config.validateEmailConfiguration({\n        emailAdapter: req.config.userController.adapter,\n        appName: req.config.appName,\n        publicServerURL: req.config.publicServerURL,\n        emailVerifyTokenValidityDuration: req.config.emailVerifyTokenValidityDuration,\n        emailVerifyTokenReuseIfValid: req.config.emailVerifyTokenReuseIfValid,\n      });\n    } catch (e) {\n      if (typeof e === 'string') {\n        // Maybe we need a Bad Configuration error, but the SDKs won't understand it. For now, Internal Server Error.\n        throw new Parse.Error(\n          Parse.Error.INTERNAL_SERVER_ERROR,\n          'An appName, publicServerURL, and emailAdapter are required for password reset and email verification functionality.'\n        );\n      } else {\n        throw e;\n      }\n    }\n  }\n\n  handleResetRequest(req) {\n    this._throwOnBadEmailConfig(req);\n\n    const { email } = req.body;\n    if (!email) {\n      throw new Parse.Error(Parse.Error.EMAIL_MISSING, 'you must provide an email');\n    }\n    if (typeof email !== 'string') {\n      throw new Parse.Error(\n        Parse.Error.INVALID_EMAIL_ADDRESS,\n        'you must provide a valid email string'\n      );\n    }\n    const userController = req.config.userController;\n    return userController.sendPasswordResetEmail(email).then(\n      () => {\n        return Promise.resolve({\n          response: {},\n        });\n      },\n      err => {\n        if (err.code === Parse.Error.OBJECT_NOT_FOUND) {\n          // Return success so that this endpoint can't\n          // be used to enumerate valid emails\n          return Promise.resolve({\n            response: {},\n          });\n        } else {\n          throw err;\n        }\n      }\n    );\n  }\n\n  handleVerificationEmailRequest(req) {\n    this._throwOnBadEmailConfig(req);\n\n    const { email } = req.body;\n    if (!email) {\n      throw new Parse.Error(Parse.Error.EMAIL_MISSING, 'you must provide an email');\n    }\n    if (typeof email !== 'string') {\n      throw new Parse.Error(\n        Parse.Error.INVALID_EMAIL_ADDRESS,\n        'you must provide a valid email string'\n      );\n    }\n\n    return req.config.database.find('_User', { email: email }).then(results => {\n      if (!results.length || results.length < 1) {\n        throw new Parse.Error(Parse.Error.EMAIL_NOT_FOUND, `No user found with email ${email}`);\n      }\n      const user = results[0];\n\n      // remove password field, messes with saving on postgres\n      delete user.password;\n\n      if (user.emailVerified) {\n        throw new Parse.Error(Parse.Error.OTHER_CAUSE, `Email ${email} is already verified.`);\n      }\n\n      const userController = req.config.userController;\n      return userController.regenerateEmailVerifyToken(user).then(() => {\n        userController.sendVerificationEmail(user);\n        return { response: {} };\n      });\n    });\n  }\n\n  async handleChallenge(req) {\n    const { username, email, password, authData, challengeData } = req.body;\n\n    // if username or email provided with password try to authenticate the user by username\n    let user;\n    if (username || email) {\n      if (!password) {\n        throw new Parse.Error(\n          Parse.Error.OTHER_CAUSE,\n          'You provided username or email, you need to also provide password.'\n        );\n      }\n      user = await this._authenticateUserFromRequest(req);\n    }\n\n    if (!challengeData) {\n      throw new Parse.Error(Parse.Error.OTHER_CAUSE, 'Nothing to challenge.');\n    }\n\n    if (typeof challengeData !== 'object') {\n      throw new Parse.Error(Parse.Error.OTHER_CAUSE, 'challengeData should be an object.');\n    }\n\n    let request;\n    let parseUser;\n\n    // Try to find user by authData\n    if (authData) {\n      if (typeof authData !== 'object') {\n        throw new Parse.Error(Parse.Error.OTHER_CAUSE, 'authData should be an object.');\n      }\n      if (user) {\n        throw new Parse.Error(\n          Parse.Error.OTHER_CAUSE,\n          'You cannot provide username/email and authData, only use one identification method.'\n        );\n      }\n\n      if (Object.keys(authData).filter(key => authData[key].id).length > 1) {\n        throw new Parse.Error(\n          Parse.Error.OTHER_CAUSE,\n          'You cannot provide more than one authData provider with an id.'\n        );\n      }\n\n      const results = await Auth.findUsersWithAuthData(req.config, authData);\n\n      try {\n        if (!results[0] || results.length > 1) {\n          throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'User not found.');\n        }\n        // Find the provider used to find the user\n        const provider = Object.keys(authData).find(key => authData[key].id);\n\n        parseUser = Parse.User.fromJSON({ className: '_User', ...results[0] });\n        request = getRequestObject(undefined, req.auth, parseUser, parseUser, req.config);\n        request.isChallenge = true;\n        // Validate authData used to identify the user to avoid brute-force attack on `id`\n        const { validator } = req.config.authDataManager.getValidatorForProvider(provider);\n        const validatorResponse = await validator(authData[provider], req, parseUser, request);\n        if (validatorResponse && validatorResponse.validator) {\n          await validatorResponse.validator();\n        }\n      } catch (e) {\n        // Rewrite the error to avoid guess id attack\n        logger.error(e);\n        throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'User not found.');\n      }\n    }\n\n    if (!parseUser) {\n      parseUser = user ? Parse.User.fromJSON({ className: '_User', ...user }) : undefined;\n    }\n\n    if (!request) {\n      request = getRequestObject(undefined, req.auth, parseUser, parseUser, req.config);\n      request.isChallenge = true;\n    }\n    const acc = {};\n    // Execute challenge step-by-step with consistent order for better error feedback\n    // and to avoid to trigger others challenges if one of them fails\n    for (const provider of Object.keys(challengeData).sort()) {\n      try {\n        const authAdapter = req.config.authDataManager.getValidatorForProvider(provider);\n        if (!authAdapter) {\n          continue;\n        }\n        const {\n          adapter: { challenge },\n        } = authAdapter;\n        if (typeof challenge === 'function') {\n          const providerChallengeResponse = await challenge(\n            challengeData[provider],\n            authData && authData[provider],\n            req.config.auth[provider],\n            request\n          );\n          acc[provider] = providerChallengeResponse || true;\n        }\n      } catch (err) {\n        const e = resolveError(err, {\n          code: Parse.Error.SCRIPT_FAILED,\n          message: 'Challenge failed. Unknown error.',\n        });\n        const userString = req.auth && req.auth.user ? req.auth.user.id : undefined;\n        logger.error(\n          `Failed running auth step challenge for ${provider} for user ${userString} with Error: ` +\n            JSON.stringify(e),\n          {\n            authenticationStep: 'challenge',\n            error: e,\n            user: userString,\n            provider,\n          }\n        );\n        throw e;\n      }\n    }\n    return { response: { challengeData: acc } };\n  }\n\n  mountRoutes() {\n    this.route('GET', '/users', req => {\n      return this.handleFind(req);\n    });\n    this.route('POST', '/users', promiseEnsureIdempotency, req => {\n      return this.handleCreate(req);\n    });\n    this.route('GET', '/users/me', req => {\n      return this.handleMe(req);\n    });\n    this.route('GET', '/users/:objectId', req => {\n      return this.handleGet(req);\n    });\n    this.route('PUT', '/users/:objectId', promiseEnsureIdempotency, req => {\n      return this.handleUpdate(req);\n    });\n    this.route('DELETE', '/users/:objectId', req => {\n      return this.handleDelete(req);\n    });\n    this.route('GET', '/login', req => {\n      return this.handleLogIn(req);\n    });\n    this.route('POST', '/login', req => {\n      return this.handleLogIn(req);\n    });\n    this.route('POST', '/loginAs', req => {\n      return this.handleLogInAs(req);\n    });\n    this.route('POST', '/logout', req => {\n      return this.handleLogOut(req);\n    });\n    this.route('POST', '/requestPasswordReset', req => {\n      return this.handleResetRequest(req);\n    });\n    this.route('POST', '/verificationEmailRequest', req => {\n      return this.handleVerificationEmailRequest(req);\n    });\n    this.route('GET', '/verifyPassword', req => {\n      return this.handleVerifyPassword(req);\n    });\n    this.route('POST', '/challenge', req => {\n      return this.handleChallenge(req);\n    });\n  }\n}\n\nexport default UsersRouter;\n"]}
|
|
575
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["UsersRouter","ClassesRouter","className","removeHiddenProperties","obj","key","Object","prototype","hasOwnProperty","call","test","_sanitizeAuthData","user","password","authData","keys","forEach","provider","length","_authenticateUserFromRequest","req","Promise","resolve","reject","payload","body","username","query","email","Parse","Error","USERNAME_MISSING","PASSWORD_MISSING","OBJECT_NOT_FOUND","isValidPassword","$or","config","database","find","then","results","loggerController","warn","filter","passwordCrypto","compare","correct","accountLockoutPolicy","AccountLockout","handleLoginAttempt","auth","isMaster","ACL","verifyUserEmails","preventLoginWithUnverifiedEmail","emailVerified","EMAIL_NOT_FOUND","catch","error","handleMe","info","sessionToken","INVALID_SESSION_TOKEN","rest","Auth","master","include","clientSDK","context","response","handleLogIn","checkIfUserHasProvidedConfiguredProvidersForLogin","authDataResponse","validatedAuthData","res","handleAuthDataValidation","RestWrite","objectId","passwordPolicy","maxPasswordAge","changedAt","_password_changed_at","Date","update","_encode","__type","iso","expiresAt","getTime","filesController","expandFilesInObject","maybeRunTrigger","TriggerTypes","beforeLogin","User","fromJSON","assign","sessionData","createSession","userId","createdWith","action","authProvider","installationId","afterLoginUser","afterLogin","handleLogInAs","OPERATION_FORBIDDEN","INVALID_VALUE","queryResults","handleVerifyPassword","handleLogOut","success","records","undefined","del","afterLogout","Session","_throwOnBadEmailConfig","Config","validateEmailConfiguration","emailAdapter","userController","adapter","appName","publicServerURL","emailVerifyTokenValidityDuration","emailVerifyTokenReuseIfValid","e","INTERNAL_SERVER_ERROR","handleResetRequest","EMAIL_MISSING","INVALID_EMAIL_ADDRESS","sendPasswordResetEmail","err","code","handleVerificationEmailRequest","OTHER_CAUSE","regenerateEmailVerifyToken","sendVerificationEmail","handleChallenge","challengeData","request","parseUser","id","findUsersWithAuthData","getRequestObject","isChallenge","validator","authDataManager","getValidatorForProvider","validatorResponse","logger","acc","sort","authAdapter","challenge","providerChallengeResponse","resolveError","SCRIPT_FAILED","message","userString","JSON","stringify","authenticationStep","mountRoutes","route","handleFind","promiseEnsureIdempotency","handleCreate","handleGet","handleUpdate","handleDelete"],"sources":["../../src/Routers/UsersRouter.js"],"sourcesContent":["// These methods handle the User-related routes.\n\nimport Parse from 'parse/node';\nimport Config from '../Config';\nimport AccountLockout from '../AccountLockout';\nimport ClassesRouter from './ClassesRouter';\nimport rest from '../rest';\nimport Auth from '../Auth';\nimport passwordCrypto from '../password';\nimport {\n  maybeRunTrigger,\n  Types as TriggerTypes,\n  getRequestObject,\n  resolveError,\n} from '../triggers';\nimport { promiseEnsureIdempotency } from '../middlewares';\nimport RestWrite from '../RestWrite';\nimport { logger } from '../logger';\n\nexport class UsersRouter extends ClassesRouter {\n  className() {\n    return '_User';\n  }\n\n  /**\n   * Removes all \"_\" prefixed properties from an object, except \"__type\"\n   * @param {Object} obj An object.\n   */\n  static removeHiddenProperties(obj) {\n    for (var key in obj) {\n      if (Object.prototype.hasOwnProperty.call(obj, key)) {\n        // Regexp comes from Parse.Object.prototype.validate\n        if (key !== '__type' && !/^[A-Za-z][0-9A-Za-z_]*$/.test(key)) {\n          delete obj[key];\n        }\n      }\n    }\n  }\n\n  /**\n   * After retrieving a user directly from the database, we need to remove the\n   * password from the object (for security), and fix an issue some SDKs have\n   * with null values\n   */\n  _sanitizeAuthData(user) {\n    delete user.password;\n\n    // Sometimes the authData still has null on that keys\n    // https://github.com/parse-community/parse-server/issues/935\n    if (user.authData) {\n      Object.keys(user.authData).forEach(provider => {\n        if (user.authData[provider] === null) {\n          delete user.authData[provider];\n        }\n      });\n      if (Object.keys(user.authData).length == 0) {\n        delete user.authData;\n      }\n    }\n  }\n\n  /**\n   * Validates a password request in login and verifyPassword\n   * @param {Object} req The request\n   * @returns {Object} User object\n   * @private\n   */\n  _authenticateUserFromRequest(req) {\n    return new Promise((resolve, reject) => {\n      // Use query parameters instead if provided in url\n      let payload = req.body;\n      if (\n        (!payload.username && req.query && req.query.username) ||\n        (!payload.email && req.query && req.query.email)\n      ) {\n        payload = req.query;\n      }\n      const { username, email, password } = payload;\n\n      // TODO: use the right error codes / descriptions.\n      if (!username && !email) {\n        throw new Parse.Error(Parse.Error.USERNAME_MISSING, 'username/email is required.');\n      }\n      if (!password) {\n        throw new Parse.Error(Parse.Error.PASSWORD_MISSING, 'password is required.');\n      }\n      if (\n        typeof password !== 'string' ||\n        (email && typeof email !== 'string') ||\n        (username && typeof username !== 'string')\n      ) {\n        throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Invalid username/password.');\n      }\n\n      let user;\n      let isValidPassword = false;\n      let query;\n      if (email && username) {\n        query = { email, username };\n      } else if (email) {\n        query = { email };\n      } else {\n        query = { $or: [{ username }, { email: username }] };\n      }\n      return req.config.database\n        .find('_User', query)\n        .then(results => {\n          if (!results.length) {\n            throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Invalid username/password.');\n          }\n\n          if (results.length > 1) {\n            // corner case where user1 has username == user2 email\n            req.config.loggerController.warn(\n              \"There is a user which email is the same as another user's username, logging in based on username\"\n            );\n            user = results.filter(user => user.username === username)[0];\n          } else {\n            user = results[0];\n          }\n\n          return passwordCrypto.compare(password, user.password);\n        })\n        .then(correct => {\n          isValidPassword = correct;\n          const accountLockoutPolicy = new AccountLockout(user, req.config);\n          return accountLockoutPolicy.handleLoginAttempt(isValidPassword);\n        })\n        .then(() => {\n          if (!isValidPassword) {\n            throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Invalid username/password.');\n          }\n          // Ensure the user isn't locked out\n          // A locked out user won't be able to login\n          // To lock a user out, just set the ACL to `masterKey` only  ({}).\n          // Empty ACL is OK\n          if (!req.auth.isMaster && user.ACL && Object.keys(user.ACL).length == 0) {\n            throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Invalid username/password.');\n          }\n          if (\n            req.config.verifyUserEmails &&\n            req.config.preventLoginWithUnverifiedEmail &&\n            !user.emailVerified\n          ) {\n            throw new Parse.Error(Parse.Error.EMAIL_NOT_FOUND, 'User email is not verified.');\n          }\n\n          this._sanitizeAuthData(user);\n\n          return resolve(user);\n        })\n        .catch(error => {\n          return reject(error);\n        });\n    });\n  }\n\n  handleMe(req) {\n    if (!req.info || !req.info.sessionToken) {\n      throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'Invalid session token');\n    }\n    const sessionToken = req.info.sessionToken;\n    return rest\n      .find(\n        req.config,\n        Auth.master(req.config),\n        '_Session',\n        { sessionToken },\n        { include: 'user' },\n        req.info.clientSDK,\n        req.info.context\n      )\n      .then(response => {\n        if (!response.results || response.results.length == 0 || !response.results[0].user) {\n          throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'Invalid session token');\n        } else {\n          const user = response.results[0].user;\n          // Send token back on the login, because SDKs expect that.\n          user.sessionToken = sessionToken;\n\n          // Remove hidden properties.\n          UsersRouter.removeHiddenProperties(user);\n          return { response: user };\n        }\n      });\n  }\n\n  async handleLogIn(req) {\n    const user = await this._authenticateUserFromRequest(req);\n    const authData = req.body && req.body.authData;\n    // Check if user has provided their required auth providers\n    Auth.checkIfUserHasProvidedConfiguredProvidersForLogin(authData, user.authData, req.config);\n\n    let authDataResponse;\n    let validatedAuthData;\n    if (authData) {\n      const res = await Auth.handleAuthDataValidation(\n        authData,\n        new RestWrite(\n          req.config,\n          req.auth,\n          '_User',\n          { objectId: user.objectId },\n          req.body,\n          user,\n          req.info.clientSDK,\n          req.info.context\n        ),\n        user\n      );\n      authDataResponse = res.authDataResponse;\n      validatedAuthData = res.authData;\n    }\n\n    // handle password expiry policy\n    if (req.config.passwordPolicy && req.config.passwordPolicy.maxPasswordAge) {\n      let changedAt = user._password_changed_at;\n\n      if (!changedAt) {\n        // password was created before expiry policy was enabled.\n        // simply update _User object so that it will start enforcing from now\n        changedAt = new Date();\n        req.config.database.update(\n          '_User',\n          { username: user.username },\n          { _password_changed_at: Parse._encode(changedAt) }\n        );\n      } else {\n        // check whether the password has expired\n        if (changedAt.__type == 'Date') {\n          changedAt = new Date(changedAt.iso);\n        }\n        // Calculate the expiry time.\n        const expiresAt = new Date(\n          changedAt.getTime() + 86400000 * req.config.passwordPolicy.maxPasswordAge\n        );\n        if (expiresAt < new Date())\n          // fail of current time is past password expiry time\n          throw new Parse.Error(\n            Parse.Error.OBJECT_NOT_FOUND,\n            'Your password has expired. Please reset your password.'\n          );\n      }\n    }\n\n    // Remove hidden properties.\n    UsersRouter.removeHiddenProperties(user);\n\n    req.config.filesController.expandFilesInObject(req.config, user);\n\n    // Before login trigger; throws if failure\n    await maybeRunTrigger(\n      TriggerTypes.beforeLogin,\n      req.auth,\n      Parse.User.fromJSON(Object.assign({ className: '_User' }, user)),\n      null,\n      req.config\n    );\n\n    // If we have some new validated authData update directly\n    if (validatedAuthData && Object.keys(validatedAuthData).length) {\n      await req.config.database.update(\n        '_User',\n        { objectId: user.objectId },\n        { authData: validatedAuthData },\n        {}\n      );\n    }\n\n    const { sessionData, createSession } = RestWrite.createSession(req.config, {\n      userId: user.objectId,\n      createdWith: {\n        action: 'login',\n        authProvider: 'password',\n      },\n      installationId: req.info.installationId,\n    });\n\n    user.sessionToken = sessionData.sessionToken;\n\n    await createSession();\n\n    const afterLoginUser = Parse.User.fromJSON(Object.assign({ className: '_User' }, user));\n    await maybeRunTrigger(\n      TriggerTypes.afterLogin,\n      { ...req.auth, user: afterLoginUser },\n      afterLoginUser,\n      null,\n      req.config\n    );\n\n    if (authDataResponse) {\n      user.authDataResponse = authDataResponse;\n    }\n\n    return { response: user };\n  }\n\n  /**\n   * This allows master-key clients to create user sessions without access to\n   * user credentials. This enables systems that can authenticate access another\n   * way (API key, app administrators) to act on a user's behalf.\n   *\n   * We create a new session rather than looking for an existing session; we\n   * want this to work in situations where the user is logged out on all\n   * devices, since this can be used by automated systems acting on the user's\n   * behalf.\n   *\n   * For the moment, we're omitting event hooks and lockout checks, since\n   * immediate use cases suggest /loginAs could be used for semantically\n   * different reasons from /login\n   */\n  async handleLogInAs(req) {\n    if (!req.auth.isMaster) {\n      throw new Parse.Error(Parse.Error.OPERATION_FORBIDDEN, 'master key is required');\n    }\n\n    const userId = req.body.userId || req.query.userId;\n    if (!userId) {\n      throw new Parse.Error(\n        Parse.Error.INVALID_VALUE,\n        'userId must not be empty, null, or undefined'\n      );\n    }\n\n    const queryResults = await req.config.database.find('_User', { objectId: userId });\n    const user = queryResults[0];\n    if (!user) {\n      throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'user not found');\n    }\n\n    this._sanitizeAuthData(user);\n\n    const { sessionData, createSession } = RestWrite.createSession(req.config, {\n      userId,\n      createdWith: {\n        action: 'login',\n        authProvider: 'masterkey',\n      },\n      installationId: req.info.installationId,\n    });\n\n    user.sessionToken = sessionData.sessionToken;\n\n    await createSession();\n\n    return { response: user };\n  }\n\n  handleVerifyPassword(req) {\n    return this._authenticateUserFromRequest(req)\n      .then(user => {\n        // Remove hidden properties.\n        UsersRouter.removeHiddenProperties(user);\n\n        return { response: user };\n      })\n      .catch(error => {\n        throw error;\n      });\n  }\n\n  async handleLogOut(req) {\n    const success = { response: {} };\n    if (req.info && req.info.sessionToken) {\n      const records = await rest.find(\n        req.config,\n        Auth.master(req.config),\n        '_Session',\n        { sessionToken: req.info.sessionToken },\n        undefined,\n        req.info.clientSDK,\n        req.info.context\n      );\n      if (records.results && records.results.length) {\n        await rest.del(\n          req.config,\n          Auth.master(req.config),\n          '_Session',\n          records.results[0].objectId,\n          req.info.context\n        );\n        await maybeRunTrigger(\n          TriggerTypes.afterLogout,\n          req.auth,\n          Parse.Session.fromJSON(Object.assign({ className: '_Session' }, records.results[0])),\n          null,\n          req.config\n        );\n      }\n    }\n    return success;\n  }\n\n  _throwOnBadEmailConfig(req) {\n    try {\n      Config.validateEmailConfiguration({\n        emailAdapter: req.config.userController.adapter,\n        appName: req.config.appName,\n        publicServerURL: req.config.publicServerURL,\n        emailVerifyTokenValidityDuration: req.config.emailVerifyTokenValidityDuration,\n        emailVerifyTokenReuseIfValid: req.config.emailVerifyTokenReuseIfValid,\n      });\n    } catch (e) {\n      if (typeof e === 'string') {\n        // Maybe we need a Bad Configuration error, but the SDKs won't understand it. For now, Internal Server Error.\n        throw new Parse.Error(\n          Parse.Error.INTERNAL_SERVER_ERROR,\n          'An appName, publicServerURL, and emailAdapter are required for password reset and email verification functionality.'\n        );\n      } else {\n        throw e;\n      }\n    }\n  }\n\n  handleResetRequest(req) {\n    this._throwOnBadEmailConfig(req);\n\n    const { email } = req.body;\n    if (!email) {\n      throw new Parse.Error(Parse.Error.EMAIL_MISSING, 'you must provide an email');\n    }\n    if (typeof email !== 'string') {\n      throw new Parse.Error(\n        Parse.Error.INVALID_EMAIL_ADDRESS,\n        'you must provide a valid email string'\n      );\n    }\n    const userController = req.config.userController;\n    return userController.sendPasswordResetEmail(email).then(\n      () => {\n        return Promise.resolve({\n          response: {},\n        });\n      },\n      err => {\n        if (err.code === Parse.Error.OBJECT_NOT_FOUND) {\n          // Return success so that this endpoint can't\n          // be used to enumerate valid emails\n          return Promise.resolve({\n            response: {},\n          });\n        } else {\n          throw err;\n        }\n      }\n    );\n  }\n\n  handleVerificationEmailRequest(req) {\n    this._throwOnBadEmailConfig(req);\n\n    const { email } = req.body;\n    if (!email) {\n      throw new Parse.Error(Parse.Error.EMAIL_MISSING, 'you must provide an email');\n    }\n    if (typeof email !== 'string') {\n      throw new Parse.Error(\n        Parse.Error.INVALID_EMAIL_ADDRESS,\n        'you must provide a valid email string'\n      );\n    }\n\n    return req.config.database.find('_User', { email: email }).then(results => {\n      if (!results.length || results.length < 1) {\n        throw new Parse.Error(Parse.Error.EMAIL_NOT_FOUND, `No user found with email ${email}`);\n      }\n      const user = results[0];\n\n      // remove password field, messes with saving on postgres\n      delete user.password;\n\n      if (user.emailVerified) {\n        throw new Parse.Error(Parse.Error.OTHER_CAUSE, `Email ${email} is already verified.`);\n      }\n\n      const userController = req.config.userController;\n      return userController.regenerateEmailVerifyToken(user).then(() => {\n        userController.sendVerificationEmail(user);\n        return { response: {} };\n      });\n    });\n  }\n\n  async handleChallenge(req) {\n    const { username, email, password, authData, challengeData } = req.body;\n\n    // if username or email provided with password try to authenticate the user by username\n    let user;\n    if (username || email) {\n      if (!password) {\n        throw new Parse.Error(\n          Parse.Error.OTHER_CAUSE,\n          'You provided username or email, you need to also provide password.'\n        );\n      }\n      user = await this._authenticateUserFromRequest(req);\n    }\n\n    if (!challengeData) {\n      throw new Parse.Error(Parse.Error.OTHER_CAUSE, 'Nothing to challenge.');\n    }\n\n    if (typeof challengeData !== 'object') {\n      throw new Parse.Error(Parse.Error.OTHER_CAUSE, 'challengeData should be an object.');\n    }\n\n    let request;\n    let parseUser;\n\n    // Try to find user by authData\n    if (authData) {\n      if (typeof authData !== 'object') {\n        throw new Parse.Error(Parse.Error.OTHER_CAUSE, 'authData should be an object.');\n      }\n      if (user) {\n        throw new Parse.Error(\n          Parse.Error.OTHER_CAUSE,\n          'You cannot provide username/email and authData, only use one identification method.'\n        );\n      }\n\n      if (Object.keys(authData).filter(key => authData[key].id).length > 1) {\n        throw new Parse.Error(\n          Parse.Error.OTHER_CAUSE,\n          'You cannot provide more than one authData provider with an id.'\n        );\n      }\n\n      const results = await Auth.findUsersWithAuthData(req.config, authData);\n\n      try {\n        if (!results[0] || results.length > 1) {\n          throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'User not found.');\n        }\n        // Find the provider used to find the user\n        const provider = Object.keys(authData).find(key => authData[key].id);\n\n        parseUser = Parse.User.fromJSON({ className: '_User', ...results[0] });\n        request = getRequestObject(undefined, req.auth, parseUser, parseUser, req.config);\n        request.isChallenge = true;\n        // Validate authData used to identify the user to avoid brute-force attack on `id`\n        const { validator } = req.config.authDataManager.getValidatorForProvider(provider);\n        const validatorResponse = await validator(authData[provider], req, parseUser, request);\n        if (validatorResponse && validatorResponse.validator) {\n          await validatorResponse.validator();\n        }\n      } catch (e) {\n        // Rewrite the error to avoid guess id attack\n        logger.error(e);\n        throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'User not found.');\n      }\n    }\n\n    if (!parseUser) {\n      parseUser = user ? Parse.User.fromJSON({ className: '_User', ...user }) : undefined;\n    }\n\n    if (!request) {\n      request = getRequestObject(undefined, req.auth, parseUser, parseUser, req.config);\n      request.isChallenge = true;\n    }\n    const acc = {};\n    // Execute challenge step-by-step with consistent order for better error feedback\n    // and to avoid to trigger others challenges if one of them fails\n    for (const provider of Object.keys(challengeData).sort()) {\n      try {\n        const authAdapter = req.config.authDataManager.getValidatorForProvider(provider);\n        if (!authAdapter) {\n          continue;\n        }\n        const {\n          adapter: { challenge },\n        } = authAdapter;\n        if (typeof challenge === 'function') {\n          const providerChallengeResponse = await challenge(\n            challengeData[provider],\n            authData && authData[provider],\n            req.config.auth[provider],\n            request\n          );\n          acc[provider] = providerChallengeResponse || true;\n        }\n      } catch (err) {\n        const e = resolveError(err, {\n          code: Parse.Error.SCRIPT_FAILED,\n          message: 'Challenge failed. Unknown error.',\n        });\n        const userString = req.auth && req.auth.user ? req.auth.user.id : undefined;\n        logger.error(\n          `Failed running auth step challenge for ${provider} for user ${userString} with Error: ` +\n            JSON.stringify(e),\n          {\n            authenticationStep: 'challenge',\n            error: e,\n            user: userString,\n            provider,\n          }\n        );\n        throw e;\n      }\n    }\n    return { response: { challengeData: acc } };\n  }\n\n  mountRoutes() {\n    this.route('GET', '/users', req => {\n      return this.handleFind(req);\n    });\n    this.route('POST', '/users', promiseEnsureIdempotency, req => {\n      return this.handleCreate(req);\n    });\n    this.route('GET', '/users/me', req => {\n      return this.handleMe(req);\n    });\n    this.route('GET', '/users/:objectId', req => {\n      return this.handleGet(req);\n    });\n    this.route('PUT', '/users/:objectId', promiseEnsureIdempotency, req => {\n      return this.handleUpdate(req);\n    });\n    this.route('DELETE', '/users/:objectId', req => {\n      return this.handleDelete(req);\n    });\n    this.route('GET', '/login', req => {\n      return this.handleLogIn(req);\n    });\n    this.route('POST', '/login', req => {\n      return this.handleLogIn(req);\n    });\n    this.route('POST', '/loginAs', req => {\n      return this.handleLogInAs(req);\n    });\n    this.route('POST', '/logout', req => {\n      return this.handleLogOut(req);\n    });\n    this.route('POST', '/requestPasswordReset', req => {\n      return this.handleResetRequest(req);\n    });\n    this.route('POST', '/verificationEmailRequest', req => {\n      return this.handleVerificationEmailRequest(req);\n    });\n    this.route('GET', '/verifyPassword', req => {\n      return this.handleVerifyPassword(req);\n    });\n    this.route('POST', '/challenge', req => {\n      return this.handleChallenge(req);\n    });\n  }\n}\n\nexport default UsersRouter;\n"],"mappings":"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"}
|