parse-server 6.0.0-alpha.3 → 6.0.0-alpha.31

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (192) hide show
  1. package/LICENSE +167 -25
  2. package/NOTICE +10 -0
  3. package/README.md +58 -34
  4. package/lib/AccountLockout.js +11 -26
  5. package/lib/Adapters/AdapterLoader.js +8 -14
  6. package/lib/Adapters/Analytics/AnalyticsAdapter.js +2 -8
  7. package/lib/Adapters/Auth/AuthAdapter.js +7 -16
  8. package/lib/Adapters/Auth/OAuth1Client.js +32 -57
  9. package/lib/Adapters/Auth/apple.js +6 -22
  10. package/lib/Adapters/Auth/facebook.js +7 -37
  11. package/lib/Adapters/Auth/gcenter.js +8 -37
  12. package/lib/Adapters/Auth/github.js +7 -10
  13. package/lib/Adapters/Auth/google.js +11 -34
  14. package/lib/Adapters/Auth/gpgames.js +5 -8
  15. package/lib/Adapters/Auth/httpsRequest.js +1 -7
  16. package/lib/Adapters/Auth/index.js +20 -65
  17. package/lib/Adapters/Auth/instagram.js +5 -9
  18. package/lib/Adapters/Auth/janraincapture.js +8 -12
  19. package/lib/Adapters/Auth/janrainengage.js +7 -11
  20. package/lib/Adapters/Auth/keycloak.js +5 -19
  21. package/lib/Adapters/Auth/ldap.js +1 -15
  22. package/lib/Adapters/Auth/line.js +7 -10
  23. package/lib/Adapters/Auth/linkedin.js +7 -12
  24. package/lib/Adapters/Auth/meetup.js +7 -10
  25. package/lib/Adapters/Auth/microsoft.js +7 -10
  26. package/lib/Adapters/Auth/oauth2.js +6 -18
  27. package/lib/Adapters/Auth/phantauth.js +8 -10
  28. package/lib/Adapters/Auth/qq.js +7 -13
  29. package/lib/Adapters/Auth/spotify.js +7 -14
  30. package/lib/Adapters/Auth/twitter.js +5 -15
  31. package/lib/Adapters/Auth/vkontakte.js +9 -15
  32. package/lib/Adapters/Auth/wechat.js +7 -10
  33. package/lib/Adapters/Auth/weibo.js +7 -11
  34. package/lib/Adapters/Cache/CacheAdapter.js +4 -12
  35. package/lib/Adapters/Cache/InMemoryCache.js +5 -19
  36. package/lib/Adapters/Cache/InMemoryCacheAdapter.js +1 -11
  37. package/lib/Adapters/Cache/LRUCache.js +1 -11
  38. package/lib/Adapters/Cache/NullCacheAdapter.js +1 -8
  39. package/lib/Adapters/Cache/RedisCacheAdapter.js +46 -87
  40. package/lib/Adapters/Cache/SchemaCache.js +1 -6
  41. package/lib/Adapters/Email/MailAdapter.js +2 -7
  42. package/lib/Adapters/Files/FilesAdapter.js +7 -21
  43. package/lib/Adapters/Files/GridFSBucketAdapter.js +6 -44
  44. package/lib/Adapters/Files/GridStoreAdapter.js +1 -1
  45. package/lib/Adapters/Logger/LoggerAdapter.js +2 -11
  46. package/lib/Adapters/Logger/WinstonLogger.js +3 -30
  47. package/lib/Adapters/Logger/WinstonLoggerAdapter.js +5 -16
  48. package/lib/Adapters/MessageQueue/EventEmitterMQ.js +3 -20
  49. package/lib/Adapters/PubSub/EventEmitterPubSub.js +1 -16
  50. package/lib/Adapters/PubSub/PubSubAdapter.js +2 -9
  51. package/lib/Adapters/PubSub/RedisPubSub.js +13 -10
  52. package/lib/Adapters/Push/PushAdapter.js +2 -8
  53. package/lib/Adapters/Storage/Mongo/MongoCollection.js +12 -37
  54. package/lib/Adapters/Storage/Mongo/MongoSchemaCollection.js +26 -79
  55. package/lib/Adapters/Storage/Mongo/MongoStorageAdapter.js +78 -209
  56. package/lib/Adapters/Storage/Mongo/MongoTransform.js +82 -371
  57. package/lib/Adapters/Storage/Postgres/PostgresClient.js +1 -13
  58. package/lib/Adapters/Storage/Postgres/PostgresConfigParser.js +1 -20
  59. package/lib/Adapters/Storage/Postgres/PostgresStorageAdapter.js +119 -446
  60. package/lib/Adapters/Storage/Postgres/sql/index.js +4 -7
  61. package/lib/Adapters/Storage/StorageAdapter.js +1 -1
  62. package/lib/Adapters/WebSocketServer/WSAdapter.js +3 -12
  63. package/lib/Adapters/WebSocketServer/WSSAdapter.js +7 -12
  64. package/lib/Auth.js +68 -121
  65. package/lib/ClientSDK.js +3 -11
  66. package/lib/Config.js +77 -115
  67. package/lib/Controllers/AdaptableController.js +6 -18
  68. package/lib/Controllers/AnalyticsController.js +1 -9
  69. package/lib/Controllers/CacheController.js +3 -23
  70. package/lib/Controllers/DatabaseController.js +171 -364
  71. package/lib/Controllers/FilesController.js +5 -34
  72. package/lib/Controllers/HooksController.js +1 -51
  73. package/lib/Controllers/LiveQueryController.js +4 -23
  74. package/lib/Controllers/LoggerController.js +15 -54
  75. package/lib/Controllers/ParseGraphQLController.js +49 -104
  76. package/lib/Controllers/PushController.js +20 -59
  77. package/lib/Controllers/SchemaController.js +162 -348
  78. package/lib/Controllers/UserController.js +17 -78
  79. package/lib/Controllers/index.js +19 -68
  80. package/lib/Controllers/types.js +1 -1
  81. package/lib/Deprecator/Deprecations.js +1 -8
  82. package/lib/Deprecator/Deprecator.js +9 -18
  83. package/lib/GraphQL/ParseGraphQLSchema.js +16 -100
  84. package/lib/GraphQL/ParseGraphQLServer.js +2 -29
  85. package/lib/GraphQL/helpers/objectsMutations.js +2 -12
  86. package/lib/GraphQL/helpers/objectsQueries.js +18 -76
  87. package/lib/GraphQL/loaders/defaultGraphQLMutations.js +1 -9
  88. package/lib/GraphQL/loaders/defaultGraphQLQueries.js +1 -8
  89. package/lib/GraphQL/loaders/defaultGraphQLTypes.js +9 -115
  90. package/lib/GraphQL/loaders/defaultRelaySchema.js +6 -18
  91. package/lib/GraphQL/loaders/filesMutations.js +2 -19
  92. package/lib/GraphQL/loaders/functionsMutations.js +6 -17
  93. package/lib/GraphQL/loaders/parseClassMutations.js +6 -44
  94. package/lib/GraphQL/loaders/parseClassQueries.js +1 -26
  95. package/lib/GraphQL/loaders/parseClassTypes.js +10 -64
  96. package/lib/GraphQL/loaders/schemaDirectives.js +1 -17
  97. package/lib/GraphQL/loaders/schemaMutations.js +1 -20
  98. package/lib/GraphQL/loaders/schemaQueries.js +1 -14
  99. package/lib/GraphQL/loaders/schemaTypes.js +2 -6
  100. package/lib/GraphQL/loaders/usersMutations.js +6 -28
  101. package/lib/GraphQL/loaders/usersQueries.js +4 -26
  102. package/lib/GraphQL/parseGraphQLUtils.js +6 -19
  103. package/lib/GraphQL/transformers/className.js +1 -4
  104. package/lib/GraphQL/transformers/constraintType.js +1 -20
  105. package/lib/GraphQL/transformers/inputType.js +1 -20
  106. package/lib/GraphQL/transformers/mutation.js +6 -51
  107. package/lib/GraphQL/transformers/outputType.js +1 -20
  108. package/lib/GraphQL/transformers/query.js +6 -42
  109. package/lib/GraphQL/transformers/schemaFields.js +7 -34
  110. package/lib/KeyPromiseQueue.js +1 -12
  111. package/lib/LiveQuery/Client.js +1 -25
  112. package/lib/LiveQuery/Id.js +1 -7
  113. package/lib/LiveQuery/ParseCloudCodePublisher.js +13 -19
  114. package/lib/LiveQuery/ParseLiveQueryServer.js +111 -307
  115. package/lib/LiveQuery/ParsePubSub.js +1 -12
  116. package/lib/LiveQuery/ParseWebSocketServer.js +4 -26
  117. package/lib/LiveQuery/QueryTools.js +14 -116
  118. package/lib/LiveQuery/RequestSchema.js +1 -1
  119. package/lib/LiveQuery/SessionTokenCache.js +1 -17
  120. package/lib/LiveQuery/Subscription.js +4 -18
  121. package/lib/LiveQuery/equalObjects.js +2 -14
  122. package/lib/Options/Definitions.js +90 -10
  123. package/lib/Options/docs.js +25 -3
  124. package/lib/Options/index.js +4 -12
  125. package/lib/Options/parsers.js +1 -18
  126. package/lib/Page.js +1 -9
  127. package/lib/ParseMessageQueue.js +1 -10
  128. package/lib/ParseServer.js +178 -202
  129. package/lib/ParseServerRESTController.js +6 -33
  130. package/lib/PromiseRouter.js +16 -50
  131. package/lib/Push/PushQueue.js +3 -15
  132. package/lib/Push/PushWorker.js +7 -32
  133. package/lib/Push/utils.js +9 -38
  134. package/lib/RestQuery.js +105 -242
  135. package/lib/RestWrite.js +224 -389
  136. package/lib/Routers/AggregateRouter.js +14 -51
  137. package/lib/Routers/AnalyticsRouter.js +2 -8
  138. package/lib/Routers/AudiencesRouter.js +1 -15
  139. package/lib/Routers/ClassesRouter.js +3 -53
  140. package/lib/Routers/CloudCodeRouter.js +1 -19
  141. package/lib/Routers/FeaturesRouter.js +1 -10
  142. package/lib/Routers/FilesRouter.js +29 -76
  143. package/lib/Routers/FunctionsRouter.js +5 -28
  144. package/lib/Routers/GlobalConfigRouter.js +4 -18
  145. package/lib/Routers/GraphQLRouter.js +1 -14
  146. package/lib/Routers/HooksRouter.js +1 -29
  147. package/lib/Routers/IAPValidationRouter.js +6 -29
  148. package/lib/Routers/InstallationsRouter.js +2 -12
  149. package/lib/Routers/LogsRouter.js +4 -16
  150. package/lib/Routers/PagesRouter.js +69 -129
  151. package/lib/Routers/PublicAPIRouter.js +3 -62
  152. package/lib/Routers/PurgeRouter.js +1 -15
  153. package/lib/Routers/PushRouter.js +2 -18
  154. package/lib/Routers/RolesRouter.js +1 -7
  155. package/lib/Routers/SchemasRouter.js +4 -34
  156. package/lib/Routers/SecurityRouter.js +1 -12
  157. package/lib/Routers/SessionsRouter.js +3 -19
  158. package/lib/Routers/UsersRouter.js +59 -156
  159. package/lib/SchemaMigrations/DefinedSchemas.js +56 -115
  160. package/lib/SchemaMigrations/Migrations.js +2 -8
  161. package/lib/Security/Check.js +8 -16
  162. package/lib/Security/CheckGroup.js +4 -11
  163. package/lib/Security/CheckGroups/CheckGroupDatabase.js +8 -18
  164. package/lib/Security/CheckGroups/CheckGroupServerConfig.js +5 -15
  165. package/lib/Security/CheckGroups/CheckGroups.js +1 -4
  166. package/lib/Security/CheckRunner.js +22 -41
  167. package/lib/StatusHandler.js +12 -69
  168. package/lib/TestUtils.js +1 -6
  169. package/lib/Utils.js +27 -66
  170. package/lib/batch.js +17 -28
  171. package/lib/cache.js +1 -3
  172. package/lib/cli/definitions/parse-live-query-server.js +1 -3
  173. package/lib/cli/definitions/parse-server.js +1 -3
  174. package/lib/cli/parse-live-query-server.js +1 -6
  175. package/lib/cli/parse-server.js +11 -21
  176. package/lib/cli/utils/commander.js +13 -51
  177. package/lib/cli/utils/runner.js +1 -14
  178. package/lib/cloud-code/Parse.Cloud.js +71 -81
  179. package/lib/cryptoUtils.js +11 -19
  180. package/lib/defaults.js +2 -14
  181. package/lib/deprecated.js +1 -2
  182. package/lib/index.js +16 -34
  183. package/lib/logger.js +6 -13
  184. package/lib/middlewares.js +170 -151
  185. package/lib/password.js +6 -10
  186. package/lib/request.js +8 -42
  187. package/lib/requiredParameter.js +1 -3
  188. package/lib/rest.js +25 -47
  189. package/lib/triggers.js +54 -252
  190. package/lib/vendor/mongodbUrl.js +129 -310
  191. package/package.json +25 -22
  192. package/PATENTS +0 -37
@@ -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 ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) { symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); } keys.push.apply(keys, symbols); } return keys; }
33
-
34
- function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i] != null ? arguments[i] : {}; if (i % 2) { ownKeys(Object(source), true).forEach(function (key) { _defineProperty(target, key, source[key]); }); } else if (Object.getOwnPropertyDescriptors) { Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)); } else { ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } } return target; }
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; // Sometimes the authData still has null on that keys
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; // TODO: use the right error codes / descriptions.
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
- return req.config.database.find('_User', query).then(results => {
116
+ return req.config.database.find('_User', query, {}, _Auth.default.maintenance(req.config)).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
- } // Ensure the user isn't locked out
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; // Send token back on the login, because SDKs expect that.
198
-
199
- user.sessionToken = sessionToken; // Remove hidden properties.
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; // Check if user has provided their required auth providers
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
- } // handle password expiry policy
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
- } // Calculate the expiry time.
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()) // fail of current time is past password expiry time
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
- } // Remove hidden properties.
251
-
216
+ }
252
217
 
218
+ // Remove hidden properties.
253
219
  UsersRouter.removeHiddenProperties(user);
254
- req.config.filesController.expandFilesInObject(req.config, user); // Before login trigger; throws if failure
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); // If we have some new validated authData update directly
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
- return _rest.default.find(req.config, _Auth.default.master(req.config), '_Session', {
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).then(records => {
377
- if (records.results && records.results.length) {
378
- return _rest.default.del(req.config, _Auth.default.master(req.config), '_Session', records.results[0].objectId, req.info.context).then(() => {
379
- this._runAfterLogoutTrigger(req, records.results[0]);
380
-
381
- return Promise.resolve(success);
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
- const user = results[0]; // remove password field, messes with saving on postgres
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; // if username or email provided with password try to authenticate the user by username
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; // Try to find user by authData
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
- } // Find the provider used to find the user
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; // Validate authData used to identify the user to avoid brute-force attack on `id`
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
- const acc = {}; // Execute challenge step-by-step with consistent order for better error feedback
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","Auth","maintenance","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","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, {}, Auth.maintenance(req.config))\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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"}