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.
Files changed (191) hide show
  1. package/README.md +45 -17
  2. package/lib/AccountLockout.js +11 -26
  3. package/lib/Adapters/AdapterLoader.js +8 -14
  4. package/lib/Adapters/Analytics/AnalyticsAdapter.js +2 -8
  5. package/lib/Adapters/Auth/AuthAdapter.js +7 -16
  6. package/lib/Adapters/Auth/OAuth1Client.js +32 -57
  7. package/lib/Adapters/Auth/apple.js +6 -22
  8. package/lib/Adapters/Auth/facebook.js +7 -37
  9. package/lib/Adapters/Auth/gcenter.js +8 -37
  10. package/lib/Adapters/Auth/github.js +7 -10
  11. package/lib/Adapters/Auth/google.js +11 -34
  12. package/lib/Adapters/Auth/gpgames.js +5 -8
  13. package/lib/Adapters/Auth/httpsRequest.js +1 -7
  14. package/lib/Adapters/Auth/index.js +20 -65
  15. package/lib/Adapters/Auth/instagram.js +5 -9
  16. package/lib/Adapters/Auth/janraincapture.js +8 -12
  17. package/lib/Adapters/Auth/janrainengage.js +7 -11
  18. package/lib/Adapters/Auth/keycloak.js +5 -19
  19. package/lib/Adapters/Auth/ldap.js +1 -15
  20. package/lib/Adapters/Auth/line.js +7 -10
  21. package/lib/Adapters/Auth/linkedin.js +7 -12
  22. package/lib/Adapters/Auth/meetup.js +7 -10
  23. package/lib/Adapters/Auth/microsoft.js +7 -10
  24. package/lib/Adapters/Auth/oauth2.js +6 -18
  25. package/lib/Adapters/Auth/phantauth.js +8 -10
  26. package/lib/Adapters/Auth/qq.js +7 -13
  27. package/lib/Adapters/Auth/spotify.js +7 -14
  28. package/lib/Adapters/Auth/twitter.js +5 -15
  29. package/lib/Adapters/Auth/vkontakte.js +9 -15
  30. package/lib/Adapters/Auth/wechat.js +7 -10
  31. package/lib/Adapters/Auth/weibo.js +7 -11
  32. package/lib/Adapters/Cache/CacheAdapter.js +4 -12
  33. package/lib/Adapters/Cache/InMemoryCache.js +5 -19
  34. package/lib/Adapters/Cache/InMemoryCacheAdapter.js +1 -11
  35. package/lib/Adapters/Cache/LRUCache.js +1 -11
  36. package/lib/Adapters/Cache/NullCacheAdapter.js +1 -8
  37. package/lib/Adapters/Cache/RedisCacheAdapter.js +46 -87
  38. package/lib/Adapters/Cache/SchemaCache.js +1 -6
  39. package/lib/Adapters/Email/MailAdapter.js +2 -7
  40. package/lib/Adapters/Files/FilesAdapter.js +7 -21
  41. package/lib/Adapters/Files/GridFSBucketAdapter.js +6 -44
  42. package/lib/Adapters/Files/GridStoreAdapter.js +1 -1
  43. package/lib/Adapters/Logger/LoggerAdapter.js +2 -11
  44. package/lib/Adapters/Logger/WinstonLogger.js +3 -30
  45. package/lib/Adapters/Logger/WinstonLoggerAdapter.js +5 -16
  46. package/lib/Adapters/MessageQueue/EventEmitterMQ.js +3 -20
  47. package/lib/Adapters/PubSub/EventEmitterPubSub.js +1 -16
  48. package/lib/Adapters/PubSub/PubSubAdapter.js +2 -9
  49. package/lib/Adapters/PubSub/RedisPubSub.js +13 -10
  50. package/lib/Adapters/Push/PushAdapter.js +2 -8
  51. package/lib/Adapters/Storage/Mongo/MongoCollection.js +12 -37
  52. package/lib/Adapters/Storage/Mongo/MongoSchemaCollection.js +26 -79
  53. package/lib/Adapters/Storage/Mongo/MongoStorageAdapter.js +78 -209
  54. package/lib/Adapters/Storage/Mongo/MongoTransform.js +82 -371
  55. package/lib/Adapters/Storage/Postgres/PostgresClient.js +1 -13
  56. package/lib/Adapters/Storage/Postgres/PostgresConfigParser.js +1 -20
  57. package/lib/Adapters/Storage/Postgres/PostgresStorageAdapter.js +119 -446
  58. package/lib/Adapters/Storage/Postgres/sql/index.js +4 -7
  59. package/lib/Adapters/Storage/StorageAdapter.js +1 -1
  60. package/lib/Adapters/WebSocketServer/WSAdapter.js +3 -12
  61. package/lib/Adapters/WebSocketServer/WSSAdapter.js +7 -12
  62. package/lib/Auth.js +54 -121
  63. package/lib/ClientSDK.js +3 -11
  64. package/lib/Config.js +69 -113
  65. package/lib/Controllers/AdaptableController.js +6 -18
  66. package/lib/Controllers/AnalyticsController.js +1 -9
  67. package/lib/Controllers/CacheController.js +3 -23
  68. package/lib/Controllers/DatabaseController.js +147 -345
  69. package/lib/Controllers/FilesController.js +5 -34
  70. package/lib/Controllers/HooksController.js +1 -51
  71. package/lib/Controllers/LiveQueryController.js +4 -23
  72. package/lib/Controllers/LoggerController.js +15 -54
  73. package/lib/Controllers/ParseGraphQLController.js +49 -104
  74. package/lib/Controllers/PushController.js +20 -59
  75. package/lib/Controllers/SchemaController.js +154 -344
  76. package/lib/Controllers/UserController.js +11 -72
  77. package/lib/Controllers/index.js +19 -68
  78. package/lib/Controllers/types.js +1 -1
  79. package/lib/Deprecator/Deprecations.js +1 -8
  80. package/lib/Deprecator/Deprecator.js +9 -18
  81. package/lib/GraphQL/ParseGraphQLSchema.js +16 -100
  82. package/lib/GraphQL/ParseGraphQLServer.js +2 -29
  83. package/lib/GraphQL/helpers/objectsMutations.js +2 -12
  84. package/lib/GraphQL/helpers/objectsQueries.js +18 -76
  85. package/lib/GraphQL/loaders/defaultGraphQLMutations.js +1 -9
  86. package/lib/GraphQL/loaders/defaultGraphQLQueries.js +1 -8
  87. package/lib/GraphQL/loaders/defaultGraphQLTypes.js +9 -115
  88. package/lib/GraphQL/loaders/defaultRelaySchema.js +6 -18
  89. package/lib/GraphQL/loaders/filesMutations.js +2 -19
  90. package/lib/GraphQL/loaders/functionsMutations.js +6 -17
  91. package/lib/GraphQL/loaders/parseClassMutations.js +6 -44
  92. package/lib/GraphQL/loaders/parseClassQueries.js +1 -26
  93. package/lib/GraphQL/loaders/parseClassTypes.js +10 -64
  94. package/lib/GraphQL/loaders/schemaDirectives.js +1 -17
  95. package/lib/GraphQL/loaders/schemaMutations.js +1 -20
  96. package/lib/GraphQL/loaders/schemaQueries.js +1 -14
  97. package/lib/GraphQL/loaders/schemaTypes.js +2 -6
  98. package/lib/GraphQL/loaders/usersMutations.js +6 -28
  99. package/lib/GraphQL/loaders/usersQueries.js +4 -26
  100. package/lib/GraphQL/parseGraphQLUtils.js +6 -19
  101. package/lib/GraphQL/transformers/className.js +1 -4
  102. package/lib/GraphQL/transformers/constraintType.js +1 -20
  103. package/lib/GraphQL/transformers/inputType.js +1 -20
  104. package/lib/GraphQL/transformers/mutation.js +6 -51
  105. package/lib/GraphQL/transformers/outputType.js +1 -20
  106. package/lib/GraphQL/transformers/query.js +6 -42
  107. package/lib/GraphQL/transformers/schemaFields.js +7 -34
  108. package/lib/KeyPromiseQueue.js +1 -12
  109. package/lib/LiveQuery/Client.js +1 -25
  110. package/lib/LiveQuery/Id.js +1 -7
  111. package/lib/LiveQuery/ParseCloudCodePublisher.js +13 -19
  112. package/lib/LiveQuery/ParseLiveQueryServer.js +92 -306
  113. package/lib/LiveQuery/ParsePubSub.js +1 -12
  114. package/lib/LiveQuery/ParseWebSocketServer.js +4 -26
  115. package/lib/LiveQuery/QueryTools.js +14 -116
  116. package/lib/LiveQuery/RequestSchema.js +1 -1
  117. package/lib/LiveQuery/SessionTokenCache.js +1 -17
  118. package/lib/LiveQuery/Subscription.js +4 -18
  119. package/lib/LiveQuery/equalObjects.js +2 -14
  120. package/lib/Options/Definitions.js +79 -10
  121. package/lib/Options/docs.js +23 -3
  122. package/lib/Options/index.js +4 -12
  123. package/lib/Options/parsers.js +1 -18
  124. package/lib/Page.js +1 -9
  125. package/lib/ParseMessageQueue.js +1 -10
  126. package/lib/ParseServer.js +144 -182
  127. package/lib/ParseServerRESTController.js +6 -33
  128. package/lib/PromiseRouter.js +16 -50
  129. package/lib/Push/PushQueue.js +3 -15
  130. package/lib/Push/PushWorker.js +7 -32
  131. package/lib/Push/utils.js +9 -38
  132. package/lib/RestQuery.js +105 -242
  133. package/lib/RestWrite.js +212 -377
  134. package/lib/Routers/AggregateRouter.js +14 -51
  135. package/lib/Routers/AnalyticsRouter.js +2 -8
  136. package/lib/Routers/AudiencesRouter.js +1 -15
  137. package/lib/Routers/ClassesRouter.js +3 -53
  138. package/lib/Routers/CloudCodeRouter.js +1 -19
  139. package/lib/Routers/FeaturesRouter.js +1 -10
  140. package/lib/Routers/FilesRouter.js +29 -76
  141. package/lib/Routers/FunctionsRouter.js +5 -28
  142. package/lib/Routers/GlobalConfigRouter.js +4 -18
  143. package/lib/Routers/GraphQLRouter.js +1 -14
  144. package/lib/Routers/HooksRouter.js +1 -29
  145. package/lib/Routers/IAPValidationRouter.js +6 -29
  146. package/lib/Routers/InstallationsRouter.js +2 -12
  147. package/lib/Routers/LogsRouter.js +4 -16
  148. package/lib/Routers/PagesRouter.js +69 -129
  149. package/lib/Routers/PublicAPIRouter.js +3 -62
  150. package/lib/Routers/PurgeRouter.js +1 -15
  151. package/lib/Routers/PushRouter.js +2 -18
  152. package/lib/Routers/RolesRouter.js +1 -7
  153. package/lib/Routers/SchemasRouter.js +4 -34
  154. package/lib/Routers/SecurityRouter.js +1 -12
  155. package/lib/Routers/SessionsRouter.js +3 -19
  156. package/lib/Routers/UsersRouter.js +58 -155
  157. package/lib/SchemaMigrations/DefinedSchemas.js +56 -115
  158. package/lib/SchemaMigrations/Migrations.js +2 -8
  159. package/lib/Security/Check.js +8 -16
  160. package/lib/Security/CheckGroup.js +4 -11
  161. package/lib/Security/CheckGroups/CheckGroupDatabase.js +8 -18
  162. package/lib/Security/CheckGroups/CheckGroupServerConfig.js +5 -15
  163. package/lib/Security/CheckGroups/CheckGroups.js +1 -4
  164. package/lib/Security/CheckRunner.js +22 -41
  165. package/lib/StatusHandler.js +12 -69
  166. package/lib/TestUtils.js +1 -6
  167. package/lib/Utils.js +27 -66
  168. package/lib/batch.js +17 -28
  169. package/lib/cache.js +1 -3
  170. package/lib/cli/definitions/parse-live-query-server.js +1 -3
  171. package/lib/cli/definitions/parse-server.js +1 -3
  172. package/lib/cli/parse-live-query-server.js +1 -6
  173. package/lib/cli/parse-server.js +11 -21
  174. package/lib/cli/utils/commander.js +13 -51
  175. package/lib/cli/utils/runner.js +1 -14
  176. package/lib/cloud-code/Parse.Cloud.js +71 -92
  177. package/lib/cryptoUtils.js +11 -19
  178. package/lib/defaults.js +2 -14
  179. package/lib/deprecated.js +1 -2
  180. package/lib/index.js +16 -34
  181. package/lib/logger.js +6 -13
  182. package/lib/middlewares.js +147 -151
  183. package/lib/password.js +6 -10
  184. package/lib/request.js +173 -2
  185. package/lib/requiredParameter.js +1 -3
  186. package/lib/rest.js +19 -41
  187. package/lib/triggers.js +54 -252
  188. package/lib/vendor/mongodbUrl.js +125 -305
  189. package/package.json +22 -19
  190. package/lib/cloud-code/HTTPResponse.js +0 -73
  191. package/lib/cloud-code/httpRequest.js +0 -192
package/lib/RestWrite.js CHANGED
@@ -4,46 +4,30 @@ Object.defineProperty(exports, "__esModule", {
4
4
  value: true
5
5
  });
6
6
  exports.default = void 0;
7
-
8
7
  var _RestQuery = _interopRequireDefault(require("./RestQuery"));
9
-
10
8
  var _lodash = _interopRequireDefault(require("lodash"));
11
-
12
9
  var _logger = _interopRequireDefault(require("./logger"));
13
-
14
10
  var _SchemaController = require("./Controllers/SchemaController");
15
-
16
11
  function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
17
-
18
- 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; }
19
-
20
- 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; }
21
-
22
- 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; }
23
-
12
+ 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; }
13
+ 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; }
14
+ 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; }
15
+ function _toPropertyKey(arg) { var key = _toPrimitive(arg, "string"); return typeof key === "symbol" ? key : String(key); }
16
+ 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); }
24
17
  // A RestWrite encapsulates everything we need to run an operation
25
18
  // that writes to the database.
26
19
  // This could be either a "create" or an "update".
27
- var SchemaController = require('./Controllers/SchemaController');
28
20
 
21
+ var SchemaController = require('./Controllers/SchemaController');
29
22
  var deepcopy = require('deepcopy');
30
-
31
23
  const Auth = require('./Auth');
32
-
33
24
  const Utils = require('./Utils');
34
-
35
25
  var cryptoUtils = require('./cryptoUtils');
36
-
37
26
  var passwordCrypto = require('./password');
38
-
39
27
  var Parse = require('parse/node');
40
-
41
28
  var triggers = require('./triggers');
42
-
43
29
  var ClientSDK = require('./ClientSDK');
44
-
45
30
  const util = require('util');
46
-
47
31
  // query and data are both provided in REST API format. So data
48
32
  // types are encoded by plain old objects.
49
33
  // If query is null, this is a "create" and the data in data should be
@@ -57,7 +41,6 @@ function RestWrite(config, auth, className, query, data, originalData, clientSDK
57
41
  if (auth.isReadOnly) {
58
42
  throw new Parse.Error(Parse.Error.OPERATION_FORBIDDEN, 'Cannot perform a write operation when using readOnlyMasterKey');
59
43
  }
60
-
61
44
  this.config = config;
62
45
  this.auth = auth;
63
46
  this.className = className;
@@ -65,11 +48,9 @@ function RestWrite(config, auth, className, query, data, originalData, clientSDK
65
48
  this.storage = {};
66
49
  this.runOptions = {};
67
50
  this.context = context || {};
68
-
69
51
  if (action) {
70
52
  this.runOptions.action = action;
71
53
  }
72
-
73
54
  if (!query) {
74
55
  if (this.config.allowCustomObjectId) {
75
56
  if (Object.prototype.hasOwnProperty.call(data, 'objectId') && !data.objectId) {
@@ -79,38 +60,43 @@ function RestWrite(config, auth, className, query, data, originalData, clientSDK
79
60
  if (data.objectId) {
80
61
  throw new Parse.Error(Parse.Error.INVALID_KEY_NAME, 'objectId is an invalid field name.');
81
62
  }
82
-
83
63
  if (data.id) {
84
64
  throw new Parse.Error(Parse.Error.INVALID_KEY_NAME, 'id is an invalid field name.');
85
65
  }
86
66
  }
87
67
  }
68
+ this.checkProhibitedKeywords(data);
88
69
 
89
- this.checkProhibitedKeywords(data); // When the operation is complete, this.response may have several
70
+ // When the operation is complete, this.response may have several
90
71
  // fields.
91
72
  // response: the actual data to be returned
92
73
  // status: the http status code. if not present, treated like a 200
93
74
  // location: the location header. if not present, no location header
75
+ this.response = null;
94
76
 
95
- this.response = null; // Processing this operation may mutate our data, so we operate on a
77
+ // Processing this operation may mutate our data, so we operate on a
96
78
  // copy
97
-
98
79
  this.query = deepcopy(query);
99
- this.data = deepcopy(data); // We never change originalData, so we do not need a deep copy
80
+ this.data = deepcopy(data);
81
+ // We never change originalData, so we do not need a deep copy
82
+ this.originalData = originalData;
100
83
 
101
- this.originalData = originalData; // The timestamp we'll use for this whole operation
84
+ // The timestamp we'll use for this whole operation
85
+ this.updatedAt = Parse._encode(new Date()).iso;
102
86
 
103
- this.updatedAt = Parse._encode(new Date()).iso; // Shared SchemaController to be reused to reduce the number of loadSchema() calls per request
87
+ // Shared SchemaController to be reused to reduce the number of loadSchema() calls per request
104
88
  // Once set the schemaData should be immutable
105
-
106
89
  this.validSchemaController = null;
107
- this.pendingOps = {};
108
- } // A convenient method to perform all the steps of processing the
90
+ this.pendingOps = {
91
+ operations: null,
92
+ identifier: null
93
+ };
94
+ }
95
+
96
+ // A convenient method to perform all the steps of processing the
109
97
  // write, in order.
110
98
  // Returns a promise for a {response, status, location} object.
111
99
  // status and location are optional.
112
-
113
-
114
100
  RestWrite.prototype.execute = function () {
115
101
  return Promise.resolve().then(() => {
116
102
  return this.getUserAndRoleACL();
@@ -156,19 +142,16 @@ RestWrite.prototype.execute = function () {
156
142
  this.response.response.authDataResponse = this.authDataResponse;
157
143
  }
158
144
  }
159
-
160
145
  return this.response;
161
146
  });
162
- }; // Uses the Auth object to get the list of roles, adds the user id
163
-
147
+ };
164
148
 
149
+ // Uses the Auth object to get the list of roles, adds the user id
165
150
  RestWrite.prototype.getUserAndRoleACL = function () {
166
151
  if (this.auth.isMaster) {
167
152
  return Promise.resolve();
168
153
  }
169
-
170
154
  this.runOptions.acl = ['*'];
171
-
172
155
  if (this.auth.user) {
173
156
  return this.auth.getUserRoles().then(roles => {
174
157
  this.runOptions.acl = this.runOptions.acl.concat(roles, [this.auth.user.id]);
@@ -177,9 +160,9 @@ RestWrite.prototype.getUserAndRoleACL = function () {
177
160
  } else {
178
161
  return Promise.resolve();
179
162
  }
180
- }; // Validates this operation against the allowClientClassCreation config.
181
-
163
+ };
182
164
 
165
+ // Validates this operation against the allowClientClassCreation config.
183
166
  RestWrite.prototype.validateClientClassCreation = function () {
184
167
  if (this.config.allowClientClassCreation === false && !this.auth.isMaster && SchemaController.systemClasses.indexOf(this.className) === -1) {
185
168
  return this.config.database.loadSchema().then(schemaController => schemaController.hasClass(this.className)).then(hasClass => {
@@ -190,45 +173,46 @@ RestWrite.prototype.validateClientClassCreation = function () {
190
173
  } else {
191
174
  return Promise.resolve();
192
175
  }
193
- }; // Validates this operation against the schema.
194
-
176
+ };
195
177
 
178
+ // Validates this operation against the schema.
196
179
  RestWrite.prototype.validateSchema = function () {
197
180
  return this.config.database.validateObject(this.className, this.data, this.query, this.runOptions);
198
- }; // Runs any beforeSave triggers against this operation.
199
- // Any change leads to our data being mutated.
200
-
181
+ };
201
182
 
183
+ // Runs any beforeSave triggers against this operation.
184
+ // Any change leads to our data being mutated.
202
185
  RestWrite.prototype.runBeforeSaveTrigger = function () {
203
186
  if (this.response || this.runOptions.many) {
204
187
  return;
205
- } // Avoid doing any setup for triggers if there is no 'beforeSave' trigger for this class.
206
-
188
+ }
207
189
 
190
+ // Avoid doing any setup for triggers if there is no 'beforeSave' trigger for this class.
208
191
  if (!triggers.triggerExists(this.className, triggers.Types.beforeSave, this.config.applicationId)) {
209
192
  return Promise.resolve();
210
193
  }
211
-
212
194
  const {
213
195
  originalObject,
214
196
  updatedObject
215
197
  } = this.buildParseObjects();
198
+ const identifier = updatedObject._getStateIdentifier();
216
199
  const stateController = Parse.CoreManager.getObjectStateController();
217
- const [pending] = stateController.getPendingOps(updatedObject._getStateIdentifier());
218
- this.pendingOps = _objectSpread({}, pending);
200
+ const [pending] = stateController.getPendingOps(identifier);
201
+ this.pendingOps = {
202
+ operations: _objectSpread({}, pending),
203
+ identifier
204
+ };
219
205
  return Promise.resolve().then(() => {
220
206
  // Before calling the trigger, validate the permissions for the save operation
221
207
  let databasePromise = null;
222
-
223
208
  if (this.query) {
224
209
  // Validate for updating
225
210
  databasePromise = this.config.database.update(this.className, this.query, this.data, this.runOptions, true, true);
226
211
  } else {
227
212
  // Validate for creating
228
213
  databasePromise = this.config.database.create(this.className, this.data, this.runOptions, true);
229
- } // In the case that there is no permission for the operation, it throws an error
230
-
231
-
214
+ }
215
+ // In the case that there is no permission for the operation, it throws an error
232
216
  return databasePromise.then(result => {
233
217
  if (!result || result.length <= 0) {
234
218
  throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Object not found.');
@@ -242,48 +226,44 @@ RestWrite.prototype.runBeforeSaveTrigger = function () {
242
226
  if (!_lodash.default.isEqual(this.data[key], value)) {
243
227
  result.push(key);
244
228
  }
245
-
246
229
  return result;
247
230
  }, []);
248
- this.data = response.object; // We should delete the objectId for an update write
249
-
231
+ this.data = response.object;
232
+ // We should delete the objectId for an update write
250
233
  if (this.query && this.query.objectId) {
251
234
  delete this.data.objectId;
252
235
  }
253
236
  }
254
-
255
237
  this.checkProhibitedKeywords(this.data);
256
238
  });
257
239
  };
258
-
259
240
  RestWrite.prototype.runBeforeLoginTrigger = async function (userData) {
260
241
  // Avoid doing any setup for triggers if there is no 'beforeLogin' trigger
261
242
  if (!triggers.triggerExists(this.className, triggers.Types.beforeLogin, this.config.applicationId)) {
262
243
  return;
263
- } // Cloud code gets a bit of extra data for its objects
264
-
244
+ }
265
245
 
246
+ // Cloud code gets a bit of extra data for its objects
266
247
  const extraData = {
267
248
  className: this.className
268
- }; // Expand file objects
249
+ };
269
250
 
251
+ // Expand file objects
270
252
  this.config.filesController.expandFilesInObject(this.config, userData);
271
- const user = triggers.inflate(extraData, userData); // no need to return a response
253
+ const user = triggers.inflate(extraData, userData);
272
254
 
255
+ // no need to return a response
273
256
  await triggers.maybeRunTrigger(triggers.Types.beforeLogin, this.auth, user, null, this.config, this.context);
274
257
  };
275
-
276
258
  RestWrite.prototype.setRequiredFieldsIfNeeded = function () {
277
259
  if (this.data) {
278
260
  return this.validSchemaController.getAllClasses().then(allClasses => {
279
261
  const schema = allClasses.find(oneClass => oneClass.className === this.className);
280
-
281
262
  const setRequiredFieldIfNeeded = (fieldName, setDefault) => {
282
263
  if (this.data[fieldName] === undefined || this.data[fieldName] === null || this.data[fieldName] === '' || typeof this.data[fieldName] === 'object' && this.data[fieldName].__op === 'Delete') {
283
264
  if (setDefault && schema.fields[fieldName] && schema.fields[fieldName].defaultValue !== null && schema.fields[fieldName].defaultValue !== undefined && (this.data[fieldName] === undefined || typeof this.data[fieldName] === 'object' && this.data[fieldName].__op === 'Delete')) {
284
265
  this.data[fieldName] = schema.fields[fieldName].defaultValue;
285
266
  this.storage.fieldsChangedByTrigger = this.storage.fieldsChangedByTrigger || [];
286
-
287
267
  if (this.storage.fieldsChangedByTrigger.indexOf(fieldName) < 0) {
288
268
  this.storage.fieldsChangedByTrigger.push(fieldName);
289
269
  }
@@ -291,18 +271,17 @@ RestWrite.prototype.setRequiredFieldsIfNeeded = function () {
291
271
  throw new Parse.Error(Parse.Error.VALIDATION_ERROR, `${fieldName} is required`);
292
272
  }
293
273
  }
294
- }; // Add default fields
295
-
274
+ };
296
275
 
276
+ // Add default fields
297
277
  this.data.updatedAt = this.updatedAt;
298
-
299
278
  if (!this.query) {
300
- this.data.createdAt = this.updatedAt; // Only assign new objectId if we are creating new object
279
+ this.data.createdAt = this.updatedAt;
301
280
 
281
+ // Only assign new objectId if we are creating new object
302
282
  if (!this.data.objectId) {
303
283
  this.data.objectId = cryptoUtils.newObjectId(this.config.objectIdSize);
304
284
  }
305
-
306
285
  if (schema) {
307
286
  Object.keys(schema.fields).forEach(fieldName => {
308
287
  setRequiredFieldIfNeeded(fieldName, true);
@@ -315,31 +294,26 @@ RestWrite.prototype.setRequiredFieldsIfNeeded = function () {
315
294
  }
316
295
  });
317
296
  }
318
-
319
297
  return Promise.resolve();
320
- }; // Transforms auth data for a user object.
298
+ };
299
+
300
+ // Transforms auth data for a user object.
321
301
  // Does nothing if this isn't a user object.
322
302
  // Returns a promise for when we're done if it can't finish this tick.
323
-
324
-
325
303
  RestWrite.prototype.validateAuthData = function () {
326
304
  if (this.className !== '_User') {
327
305
  return;
328
306
  }
329
-
330
307
  const authData = this.data.authData;
331
308
  const hasUsernameAndPassword = typeof this.data.username === 'string' && typeof this.data.password === 'string';
332
-
333
309
  if (!this.query && !authData) {
334
310
  if (typeof this.data.username !== 'string' || _lodash.default.isEmpty(this.data.username)) {
335
311
  throw new Parse.Error(Parse.Error.USERNAME_MISSING, 'bad or missing username');
336
312
  }
337
-
338
313
  if (typeof this.data.password !== 'string' || _lodash.default.isEmpty(this.data.password)) {
339
314
  throw new Parse.Error(Parse.Error.PASSWORD_MISSING, 'password is required');
340
315
  }
341
316
  }
342
-
343
317
  if (authData && !Object.keys(authData).length || !Object.prototype.hasOwnProperty.call(this.data, 'authData')) {
344
318
  // Nothing to validate here
345
319
  return;
@@ -347,104 +321,89 @@ RestWrite.prototype.validateAuthData = function () {
347
321
  // Handle saving authData to null
348
322
  throw new Parse.Error(Parse.Error.UNSUPPORTED_SERVICE, 'This authentication method is unsupported.');
349
323
  }
350
-
351
324
  var providers = Object.keys(authData);
352
-
353
325
  if (providers.length > 0) {
354
326
  const canHandleAuthData = providers.some(provider => {
355
327
  var providerAuthData = authData[provider];
356
328
  var hasToken = providerAuthData && providerAuthData.id;
357
329
  return hasToken || providerAuthData === null;
358
330
  });
359
-
360
331
  if (canHandleAuthData || hasUsernameAndPassword || this.auth.isMaster || this.getUserId()) {
361
332
  return this.handleAuthData(authData);
362
333
  }
363
334
  }
364
-
365
335
  throw new Parse.Error(Parse.Error.UNSUPPORTED_SERVICE, 'This authentication method is unsupported.');
366
336
  };
367
-
368
337
  RestWrite.prototype.filteredObjectsByACL = function (objects) {
369
338
  if (this.auth.isMaster) {
370
339
  return objects;
371
340
  }
372
-
373
341
  return objects.filter(object => {
374
342
  if (!object.ACL) {
375
343
  return true; // legacy users that have no ACL field on them
376
- } // Regular users that have been locked out.
377
-
378
-
344
+ }
345
+ // Regular users that have been locked out.
379
346
  return object.ACL && Object.keys(object.ACL).length > 0;
380
347
  });
381
348
  };
382
-
383
349
  RestWrite.prototype.getUserId = function () {
384
350
  if (this.query && this.query.objectId && this.className === '_User') {
385
351
  return this.query.objectId;
386
352
  } else if (this.auth && this.auth.user && this.auth.user.id) {
387
353
  return this.auth.user.id;
388
354
  }
389
- }; // Developers are allowed to change authData via before save trigger
355
+ };
356
+
357
+ // Developers are allowed to change authData via before save trigger
390
358
  // we need after before save to ensure that the developer
391
359
  // is not currently duplicating auth data ID
392
-
393
-
394
360
  RestWrite.prototype.ensureUniqueAuthDataId = async function () {
395
361
  if (this.className !== '_User' || !this.data.authData) {
396
362
  return;
397
363
  }
398
-
399
364
  const hasAuthDataId = Object.keys(this.data.authData).some(key => this.data.authData[key] && this.data.authData[key].id);
400
365
  if (!hasAuthDataId) return;
401
366
  const r = await Auth.findUsersWithAuthData(this.config, this.data.authData);
402
367
  const results = this.filteredObjectsByACL(r);
403
-
404
368
  if (results.length > 1) {
405
369
  throw new Parse.Error(Parse.Error.ACCOUNT_ALREADY_LINKED, 'this auth is already used');
406
- } // use data.objectId in case of login time and found user during handle validateAuthData
407
-
408
-
370
+ }
371
+ // use data.objectId in case of login time and found user during handle validateAuthData
409
372
  const userId = this.getUserId() || this.data.objectId;
410
-
411
373
  if (results.length === 1 && userId !== results[0].objectId) {
412
374
  throw new Parse.Error(Parse.Error.ACCOUNT_ALREADY_LINKED, 'this auth is already used');
413
375
  }
414
376
  };
415
-
416
377
  RestWrite.prototype.handleAuthData = async function (authData) {
417
378
  const r = await Auth.findUsersWithAuthData(this.config, authData);
418
379
  const results = this.filteredObjectsByACL(r);
419
-
420
380
  if (results.length > 1) {
421
381
  // To avoid https://github.com/parse-community/parse-server/security/advisories/GHSA-8w3j-g983-8jh5
422
382
  // Let's run some validation before throwing
423
383
  await Auth.handleAuthDataValidation(authData, this, results[0]);
424
384
  throw new Parse.Error(Parse.Error.ACCOUNT_ALREADY_LINKED, 'this auth is already used');
425
- } // No user found with provided authData we need to validate
426
-
385
+ }
427
386
 
387
+ // No user found with provided authData we need to validate
428
388
  if (!results.length) {
429
389
  const {
430
390
  authData: validatedAuthData,
431
391
  authDataResponse
432
392
  } = await Auth.handleAuthDataValidation(authData, this);
433
- this.authDataResponse = authDataResponse; // Replace current authData by the new validated one
434
-
393
+ this.authDataResponse = authDataResponse;
394
+ // Replace current authData by the new validated one
435
395
  this.data.authData = validatedAuthData;
436
396
  return;
437
- } // User found with provided authData
438
-
397
+ }
439
398
 
399
+ // User found with provided authData
440
400
  if (results.length === 1) {
441
401
  const userId = this.getUserId();
442
- const userResult = results[0]; // Prevent duplicate authData id
443
-
402
+ const userResult = results[0];
403
+ // Prevent duplicate authData id
444
404
  if (userId && userId !== userResult.objectId) {
445
405
  throw new Parse.Error(Parse.Error.ACCOUNT_ALREADY_LINKED, 'this auth is already used');
446
406
  }
447
-
448
407
  this.storage.authProvider = Object.keys(authData).join(',');
449
408
  const {
450
409
  hasMutatedAuthData,
@@ -452,56 +411,57 @@ RestWrite.prototype.handleAuthData = async function (authData) {
452
411
  } = Auth.hasMutatedAuthData(authData, userResult.authData);
453
412
  const isCurrentUserLoggedOrMaster = this.auth && this.auth.user && this.auth.user.id === userResult.objectId || this.auth.isMaster;
454
413
  const isLogin = !userId;
455
-
456
414
  if (isLogin || isCurrentUserLoggedOrMaster) {
457
415
  // no user making the call
458
416
  // OR the user making the call is the right one
459
417
  // Login with auth data
460
- delete results[0].password; // need to set the objectId first otherwise location has trailing undefined
418
+ delete results[0].password;
461
419
 
420
+ // need to set the objectId first otherwise location has trailing undefined
462
421
  this.data.objectId = userResult.objectId;
463
-
464
422
  if (!this.query || !this.query.objectId) {
465
423
  this.response = {
466
424
  response: userResult,
467
425
  location: this.location()
468
- }; // Run beforeLogin hook before storing any updates
426
+ };
427
+ // Run beforeLogin hook before storing any updates
469
428
  // to authData on the db; changes to userResult
470
429
  // will be ignored.
430
+ await this.runBeforeLoginTrigger(deepcopy(userResult));
471
431
 
472
- await this.runBeforeLoginTrigger(deepcopy(userResult)); // If we are in login operation via authData
432
+ // If we are in login operation via authData
473
433
  // we need to be sure that the user has provided
474
434
  // required authData
475
-
476
435
  Auth.checkIfUserHasProvidedConfiguredProvidersForLogin(authData, userResult.authData, this.config);
477
- } // Prevent validating if no mutated data detected on update
478
-
436
+ }
479
437
 
438
+ // Prevent validating if no mutated data detected on update
480
439
  if (!hasMutatedAuthData && isCurrentUserLoggedOrMaster) {
481
440
  return;
482
- } // Force to validate all provided authData on login
483
- // on update only validate mutated ones
484
-
441
+ }
485
442
 
443
+ // Force to validate all provided authData on login
444
+ // on update only validate mutated ones
486
445
  if (hasMutatedAuthData || !this.config.allowExpiredAuthDataToken) {
487
446
  const res = await Auth.handleAuthDataValidation(isLogin ? authData : mutatedAuthData, this, userResult);
488
447
  this.data.authData = res.authData;
489
448
  this.authDataResponse = res.authDataResponse;
490
- } // IF we are in login we'll skip the database operation / beforeSave / afterSave etc...
449
+ }
450
+
451
+ // IF we are in login we'll skip the database operation / beforeSave / afterSave etc...
491
452
  // we need to set it up there.
492
453
  // We are supposed to have a response only on LOGIN with authData, so we skip those
493
454
  // If we're not logging in, but just updating the current user, we can safely skip that part
494
-
495
-
496
455
  if (this.response) {
497
456
  // Assign the new authData in the response
498
457
  Object.keys(mutatedAuthData).forEach(provider => {
499
458
  this.response.response.authData[provider] = mutatedAuthData[provider];
500
- }); // Run the DB update directly, as 'master' only if authData contains some keys
459
+ });
460
+
461
+ // Run the DB update directly, as 'master' only if authData contains some keys
501
462
  // authData could not contains keys after validation if the authAdapter
502
463
  // uses the `doNotSave` option. Just update the authData part
503
464
  // Then we're good for the user, early exit of sorts
504
-
505
465
  if (Object.keys(this.data.authData).length) {
506
466
  await this.config.database.update(this.className, {
507
467
  objectId: this.data.objectId
@@ -512,22 +472,20 @@ RestWrite.prototype.handleAuthData = async function (authData) {
512
472
  }
513
473
  }
514
474
  }
515
- }; // The non-third-party parts of User transformation
516
-
475
+ };
517
476
 
477
+ // The non-third-party parts of User transformation
518
478
  RestWrite.prototype.transformUser = function () {
519
479
  var promise = Promise.resolve();
520
-
521
480
  if (this.className !== '_User') {
522
481
  return promise;
523
482
  }
524
-
525
483
  if (!this.auth.isMaster && 'emailVerified' in this.data) {
526
484
  const error = `Clients aren't allowed to manually update email verification.`;
527
485
  throw new Parse.Error(Parse.Error.OPERATION_FORBIDDEN, error);
528
- } // Do not cleanup session if objectId is not set
529
-
486
+ }
530
487
 
488
+ // Do not cleanup session if objectId is not set
531
489
  if (this.query && this.objectId()) {
532
490
  // If we're updating a _User object, we need to clear out the cache for that user. Find all their
533
491
  // session tokens, and remove them from the cache.
@@ -541,22 +499,19 @@ RestWrite.prototype.transformUser = function () {
541
499
  results.results.forEach(session => this.config.cacheController.user.del(session.sessionToken));
542
500
  });
543
501
  }
544
-
545
502
  return promise.then(() => {
546
503
  // Transform the password
547
504
  if (this.data.password === undefined) {
548
505
  // ignore only if undefined. should proceed if empty ('')
549
506
  return Promise.resolve();
550
507
  }
551
-
552
508
  if (this.query) {
553
- this.storage['clearSessions'] = true; // Generate a new session only if the user requested
554
-
509
+ this.storage['clearSessions'] = true;
510
+ // Generate a new session only if the user requested
555
511
  if (!this.auth.isMaster) {
556
512
  this.storage['generateNewSession'] = true;
557
513
  }
558
514
  }
559
-
560
515
  return this._validatePasswordPolicy().then(() => {
561
516
  return passwordCrypto.hash(this.data.password).then(hashedPassword => {
562
517
  this.data._hashed_password = hashedPassword;
@@ -569,7 +524,6 @@ RestWrite.prototype.transformUser = function () {
569
524
  return this._validateEmail();
570
525
  });
571
526
  };
572
-
573
527
  RestWrite.prototype._validateUserName = function () {
574
528
  // Check for username uniqueness
575
529
  if (!this.data.username) {
@@ -577,7 +531,6 @@ RestWrite.prototype._validateUserName = function () {
577
531
  this.data.username = cryptoUtils.randomString(25);
578
532
  this.responseShouldHaveUsername = true;
579
533
  }
580
-
581
534
  return Promise.resolve();
582
535
  }
583
536
  /*
@@ -586,8 +539,6 @@ RestWrite.prototype._validateUserName = function () {
586
539
  login using the case they entered. I.e. 'Snoopy' should preclude
587
540
  'snoopy' as a valid username.
588
541
  */
589
-
590
-
591
542
  return this.config.database.find(this.className, {
592
543
  username: this.data.username,
593
544
  objectId: {
@@ -600,10 +551,10 @@ RestWrite.prototype._validateUserName = function () {
600
551
  if (results.length > 0) {
601
552
  throw new Parse.Error(Parse.Error.USERNAME_TAKEN, 'Account already exists for this username.');
602
553
  }
603
-
604
554
  return;
605
555
  });
606
556
  };
557
+
607
558
  /*
608
559
  As with usernames, Parse should not allow case insensitive collisions of email.
609
560
  unlike with usernames (which can have case insensitive collisions in the case of
@@ -616,19 +567,15 @@ RestWrite.prototype._validateUserName = function () {
616
567
  Given that this lookup should be a relatively low use case and that the case sensitive
617
568
  unique index will be used by the db for the query, this is an adequate solution.
618
569
  */
619
-
620
-
621
570
  RestWrite.prototype._validateEmail = function () {
622
571
  if (!this.data.email || this.data.email.__op === 'Delete') {
623
572
  return Promise.resolve();
624
- } // Validate basic email address format
625
-
626
-
573
+ }
574
+ // Validate basic email address format
627
575
  if (!this.data.email.match(/^.+@.+$/)) {
628
576
  return Promise.reject(new Parse.Error(Parse.Error.INVALID_EMAIL_ADDRESS, 'Email address format is invalid.'));
629
- } // Case insensitive match, see note above function.
630
-
631
-
577
+ }
578
+ // Case insensitive match, see note above function.
632
579
  return this.config.database.find(this.className, {
633
580
  email: this.data.email,
634
581
  objectId: {
@@ -641,7 +588,6 @@ RestWrite.prototype._validateEmail = function () {
641
588
  if (results.length > 0) {
642
589
  throw new Parse.Error(Parse.Error.EMAIL_TAKEN, 'Account already exists for this email address.');
643
590
  }
644
-
645
591
  if (!this.data.authData || !Object.keys(this.data.authData).length || Object.keys(this.data.authData).length === 1 && Object.keys(this.data.authData)[0] === 'anonymous') {
646
592
  // We updated the email, send a new validation
647
593
  this.storage['sendVerificationEmail'] = true;
@@ -649,14 +595,12 @@ RestWrite.prototype._validateEmail = function () {
649
595
  }
650
596
  });
651
597
  };
652
-
653
598
  RestWrite.prototype._validatePasswordPolicy = function () {
654
599
  if (!this.config.passwordPolicy) return Promise.resolve();
655
600
  return this._validatePasswordRequirements().then(() => {
656
601
  return this._validatePasswordHistory();
657
602
  });
658
603
  };
659
-
660
604
  RestWrite.prototype._validatePasswordRequirements = function () {
661
605
  // check if the password conforms to the defined password policy if configured
662
606
  // If we specified a custom error in our configuration use it.
@@ -667,13 +611,14 @@ RestWrite.prototype._validatePasswordRequirements = function () {
667
611
  // a. making the user guess whats wrong
668
612
  // b. making a custom password reset page that shows the requirements
669
613
  const policyError = this.config.passwordPolicy.validationError ? this.config.passwordPolicy.validationError : 'Password does not meet the Password Policy requirements.';
670
- const containsUsernameError = 'Password cannot contain your username.'; // check whether the password meets the password strength requirements
614
+ const containsUsernameError = 'Password cannot contain your username.';
671
615
 
616
+ // check whether the password meets the password strength requirements
672
617
  if (this.config.passwordPolicy.patternValidator && !this.config.passwordPolicy.patternValidator(this.data.password) || this.config.passwordPolicy.validatorCallback && !this.config.passwordPolicy.validatorCallback(this.data.password)) {
673
618
  return Promise.reject(new Parse.Error(Parse.Error.VALIDATION_ERROR, policyError));
674
- } // check whether password contain username
675
-
619
+ }
676
620
 
621
+ // check whether password contain username
677
622
  if (this.config.passwordPolicy.doNotAllowUsername === true) {
678
623
  if (this.data.username) {
679
624
  // username is not passed during password reset
@@ -686,16 +631,13 @@ RestWrite.prototype._validatePasswordRequirements = function () {
686
631
  if (results.length != 1) {
687
632
  throw undefined;
688
633
  }
689
-
690
634
  if (this.data.password.indexOf(results[0].username) >= 0) return Promise.reject(new Parse.Error(Parse.Error.VALIDATION_ERROR, containsUsernameError));
691
635
  return Promise.resolve();
692
636
  });
693
637
  }
694
638
  }
695
-
696
639
  return Promise.resolve();
697
640
  };
698
-
699
641
  RestWrite.prototype._validatePasswordHistory = function () {
700
642
  // check whether password is repeating from specified history
701
643
  if (this.query && this.config.passwordPolicy.maxPasswordHistory) {
@@ -707,51 +649,49 @@ RestWrite.prototype._validatePasswordHistory = function () {
707
649
  if (results.length != 1) {
708
650
  throw undefined;
709
651
  }
710
-
711
652
  const user = results[0];
712
653
  let oldPasswords = [];
713
654
  if (user._password_history) oldPasswords = _lodash.default.take(user._password_history, this.config.passwordPolicy.maxPasswordHistory - 1);
714
655
  oldPasswords.push(user.password);
715
- const newPassword = this.data.password; // compare the new password hash with all old password hashes
716
-
656
+ const newPassword = this.data.password;
657
+ // compare the new password hash with all old password hashes
717
658
  const promises = oldPasswords.map(function (hash) {
718
659
  return passwordCrypto.compare(newPassword, hash).then(result => {
719
- if (result) // reject if there is a match
660
+ if (result)
661
+ // reject if there is a match
720
662
  return Promise.reject('REPEAT_PASSWORD');
721
663
  return Promise.resolve();
722
664
  });
723
- }); // wait for all comparisons to complete
724
-
665
+ });
666
+ // wait for all comparisons to complete
725
667
  return Promise.all(promises).then(() => {
726
668
  return Promise.resolve();
727
669
  }).catch(err => {
728
- if (err === 'REPEAT_PASSWORD') // a match was found
670
+ if (err === 'REPEAT_PASSWORD')
671
+ // a match was found
729
672
  return Promise.reject(new Parse.Error(Parse.Error.VALIDATION_ERROR, `New password should not be the same as last ${this.config.passwordPolicy.maxPasswordHistory} passwords.`));
730
673
  throw err;
731
674
  });
732
675
  });
733
676
  }
734
-
735
677
  return Promise.resolve();
736
678
  };
737
-
738
679
  RestWrite.prototype.createSessionTokenIfNeeded = function () {
739
680
  if (this.className !== '_User') {
740
681
  return;
741
- } // Don't generate session for updating user (this.query is set) unless authData exists
742
-
743
-
682
+ }
683
+ // Don't generate session for updating user (this.query is set) unless authData exists
744
684
  if (this.query && !this.data.authData) {
745
685
  return;
746
- } // Don't generate new sessionToken if linking via sessionToken
747
-
748
-
686
+ }
687
+ // Don't generate new sessionToken if linking via sessionToken
749
688
  if (this.auth.user && this.data.authData) {
750
689
  return;
751
690
  }
752
-
753
- if (!this.storage.authProvider && // signup call, with
754
- this.config.preventLoginWithUnverifiedEmail && // no login without verification
691
+ if (!this.storage.authProvider &&
692
+ // signup call, with
693
+ this.config.preventLoginWithUnverifiedEmail &&
694
+ // no login without verification
755
695
  this.config.verifyUserEmails) {
756
696
  // verification is on
757
697
  return; // do not create the session token in that case!
@@ -759,18 +699,15 @@ RestWrite.prototype.createSessionTokenIfNeeded = function () {
759
699
 
760
700
  return this.createSessionToken();
761
701
  };
762
-
763
702
  RestWrite.prototype.createSessionToken = async function () {
764
703
  // cloud installationId from Cloud Code,
765
704
  // never create session tokens from there.
766
705
  if (this.auth.installationId && this.auth.installationId === 'cloud') {
767
706
  return;
768
707
  }
769
-
770
708
  if (this.storage.authProvider == null && this.data.authData) {
771
709
  this.storage.authProvider = Object.keys(this.data.authData).join(',');
772
710
  }
773
-
774
711
  const {
775
712
  sessionData,
776
713
  createSession
@@ -782,14 +719,11 @@ RestWrite.prototype.createSessionToken = async function () {
782
719
  },
783
720
  installationId: this.auth.installationId
784
721
  });
785
-
786
722
  if (this.response && this.response.response) {
787
723
  this.response.response.sessionToken = sessionData.sessionToken;
788
724
  }
789
-
790
725
  return createSession();
791
726
  };
792
-
793
727
  RestWrite.createSession = function (config, {
794
728
  userId,
795
729
  createdWith,
@@ -808,25 +742,22 @@ RestWrite.createSession = function (config, {
808
742
  createdWith,
809
743
  expiresAt: Parse._encode(expiresAt)
810
744
  };
811
-
812
745
  if (installationId) {
813
746
  sessionData.installationId = installationId;
814
747
  }
815
-
816
748
  Object.assign(sessionData, additionalSessionData);
817
749
  return {
818
750
  sessionData,
819
751
  createSession: () => new RestWrite(config, Auth.master(config), '_Session', null, sessionData).execute()
820
752
  };
821
- }; // Delete email reset tokens if user is changing password or email.
822
-
753
+ };
823
754
 
755
+ // Delete email reset tokens if user is changing password or email.
824
756
  RestWrite.prototype.deleteEmailResetTokenIfNeeded = function () {
825
757
  if (this.className !== '_User' || this.query === null) {
826
758
  // null query means create
827
759
  return;
828
760
  }
829
-
830
761
  if ('password' in this.data || 'email' in this.data) {
831
762
  const addOps = {
832
763
  _perishable_token: {
@@ -839,28 +770,23 @@ RestWrite.prototype.deleteEmailResetTokenIfNeeded = function () {
839
770
  this.data = Object.assign(this.data, addOps);
840
771
  }
841
772
  };
842
-
843
773
  RestWrite.prototype.destroyDuplicatedSessions = function () {
844
774
  // Only for _Session, and at creation time
845
775
  if (this.className != '_Session' || this.query) {
846
776
  return;
847
- } // Destroy the sessions in 'Background'
848
-
849
-
777
+ }
778
+ // Destroy the sessions in 'Background'
850
779
  const {
851
780
  user,
852
781
  installationId,
853
782
  sessionToken
854
783
  } = this.data;
855
-
856
784
  if (!user || !installationId) {
857
785
  return;
858
786
  }
859
-
860
787
  if (!user.objectId) {
861
788
  return;
862
789
  }
863
-
864
790
  this.config.database.destroy('_Session', {
865
791
  user,
866
792
  installationId,
@@ -868,9 +794,9 @@ RestWrite.prototype.destroyDuplicatedSessions = function () {
868
794
  $ne: sessionToken
869
795
  }
870
796
  }, {}, this.validSchemaController);
871
- }; // Handles any followup logic
872
-
797
+ };
873
798
 
799
+ // Handles any followup logic
874
800
  RestWrite.prototype.handleFollowup = function () {
875
801
  if (this.storage && this.storage['clearSessions'] && this.config.revokeSessionOnPasswordReset) {
876
802
  var sessionQuery = {
@@ -883,36 +809,32 @@ RestWrite.prototype.handleFollowup = function () {
883
809
  delete this.storage['clearSessions'];
884
810
  return this.config.database.destroy('_Session', sessionQuery).then(this.handleFollowup.bind(this));
885
811
  }
886
-
887
812
  if (this.storage && this.storage['generateNewSession']) {
888
813
  delete this.storage['generateNewSession'];
889
814
  return this.createSessionToken().then(this.handleFollowup.bind(this));
890
815
  }
891
-
892
816
  if (this.storage && this.storage['sendVerificationEmail']) {
893
- delete this.storage['sendVerificationEmail']; // Fire and forget!
894
-
817
+ delete this.storage['sendVerificationEmail'];
818
+ // Fire and forget!
895
819
  this.config.userController.sendVerificationEmail(this.data);
896
820
  return this.handleFollowup.bind(this);
897
821
  }
898
- }; // Handles the _Session class specialness.
899
- // Does nothing if this isn't an _Session object.
900
-
822
+ };
901
823
 
824
+ // Handles the _Session class specialness.
825
+ // Does nothing if this isn't an _Session object.
902
826
  RestWrite.prototype.handleSession = function () {
903
827
  if (this.response || this.className !== '_Session') {
904
828
  return;
905
829
  }
906
-
907
830
  if (!this.auth.user && !this.auth.isMaster) {
908
831
  throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'Session token required.');
909
- } // TODO: Verify proper error to throw
910
-
832
+ }
911
833
 
834
+ // TODO: Verify proper error to throw
912
835
  if (this.data.ACL) {
913
836
  throw new Parse.Error(Parse.Error.INVALID_KEY_NAME, 'Cannot set ' + 'ACL on a Session.');
914
837
  }
915
-
916
838
  if (this.query) {
917
839
  if (this.data.user && !this.auth.isMaster && this.data.user.objectId != this.auth.user.id) {
918
840
  throw new Parse.Error(Parse.Error.INVALID_KEY_NAME);
@@ -921,7 +843,6 @@ RestWrite.prototype.handleSession = function () {
921
843
  } else if (this.data.sessionToken) {
922
844
  throw new Parse.Error(Parse.Error.INVALID_KEY_NAME);
923
845
  }
924
-
925
846
  if (!this.auth.isMaster) {
926
847
  this.query = {
927
848
  $and: [this.query, {
@@ -934,18 +855,14 @@ RestWrite.prototype.handleSession = function () {
934
855
  };
935
856
  }
936
857
  }
937
-
938
858
  if (!this.query && !this.auth.isMaster) {
939
859
  const additionalSessionData = {};
940
-
941
860
  for (var key in this.data) {
942
861
  if (key === 'objectId' || key === 'user') {
943
862
  continue;
944
863
  }
945
-
946
864
  additionalSessionData[key] = this.data[key];
947
865
  }
948
-
949
866
  const {
950
867
  sessionData,
951
868
  createSession
@@ -960,7 +877,6 @@ RestWrite.prototype.handleSession = function () {
960
877
  if (!results.response) {
961
878
  throw new Parse.Error(Parse.Error.INTERNAL_SERVER_ERROR, 'Error creating session.');
962
879
  }
963
-
964
880
  sessionData['objectId'] = results.response['objectId'];
965
881
  this.response = {
966
882
  status: 201,
@@ -969,79 +885,71 @@ RestWrite.prototype.handleSession = function () {
969
885
  };
970
886
  });
971
887
  }
972
- }; // Handles the _Installation class specialness.
888
+ };
889
+
890
+ // Handles the _Installation class specialness.
973
891
  // Does nothing if this isn't an installation object.
974
892
  // If an installation is found, this can mutate this.query and turn a create
975
893
  // into an update.
976
894
  // Returns a promise for when we're done if it can't finish this tick.
977
-
978
-
979
895
  RestWrite.prototype.handleInstallation = function () {
980
896
  if (this.response || this.className !== '_Installation') {
981
897
  return;
982
898
  }
983
-
984
899
  if (!this.query && !this.data.deviceToken && !this.data.installationId && !this.auth.installationId) {
985
900
  throw new Parse.Error(135, 'at least one ID field (deviceToken, installationId) ' + 'must be specified in this operation');
986
- } // If the device token is 64 characters long, we assume it is for iOS
987
- // and lowercase it.
988
-
901
+ }
989
902
 
903
+ // If the device token is 64 characters long, we assume it is for iOS
904
+ // and lowercase it.
990
905
  if (this.data.deviceToken && this.data.deviceToken.length == 64) {
991
906
  this.data.deviceToken = this.data.deviceToken.toLowerCase();
992
- } // We lowercase the installationId if present
993
-
907
+ }
994
908
 
909
+ // We lowercase the installationId if present
995
910
  if (this.data.installationId) {
996
911
  this.data.installationId = this.data.installationId.toLowerCase();
997
912
  }
913
+ let installationId = this.data.installationId;
998
914
 
999
- let installationId = this.data.installationId; // If data.installationId is not set and we're not master, we can lookup in auth
1000
-
915
+ // If data.installationId is not set and we're not master, we can lookup in auth
1001
916
  if (!installationId && !this.auth.isMaster) {
1002
917
  installationId = this.auth.installationId;
1003
918
  }
1004
-
1005
919
  if (installationId) {
1006
920
  installationId = installationId.toLowerCase();
1007
- } // Updating _Installation but not updating anything critical
1008
-
921
+ }
1009
922
 
923
+ // Updating _Installation but not updating anything critical
1010
924
  if (this.query && !this.data.deviceToken && !installationId && !this.data.deviceType) {
1011
925
  return;
1012
926
  }
1013
-
1014
927
  var promise = Promise.resolve();
1015
928
  var idMatch; // Will be a match on either objectId or installationId
1016
-
1017
929
  var objectIdMatch;
1018
930
  var installationIdMatch;
1019
- var deviceTokenMatches = []; // Instead of issuing 3 reads, let's do it with one OR.
931
+ var deviceTokenMatches = [];
1020
932
 
933
+ // Instead of issuing 3 reads, let's do it with one OR.
1021
934
  const orQueries = [];
1022
-
1023
935
  if (this.query && this.query.objectId) {
1024
936
  orQueries.push({
1025
937
  objectId: this.query.objectId
1026
938
  });
1027
939
  }
1028
-
1029
940
  if (installationId) {
1030
941
  orQueries.push({
1031
942
  installationId: installationId
1032
943
  });
1033
944
  }
1034
-
1035
945
  if (this.data.deviceToken) {
1036
946
  orQueries.push({
1037
947
  deviceToken: this.data.deviceToken
1038
948
  });
1039
949
  }
1040
-
1041
950
  if (orQueries.length == 0) {
1042
951
  return;
1043
952
  }
1044
-
1045
953
  promise = promise.then(() => {
1046
954
  return this.config.database.find('_Installation', {
1047
955
  $or: orQueries
@@ -1051,43 +959,36 @@ RestWrite.prototype.handleInstallation = function () {
1051
959
  if (this.query && this.query.objectId && result.objectId == this.query.objectId) {
1052
960
  objectIdMatch = result;
1053
961
  }
1054
-
1055
962
  if (result.installationId == installationId) {
1056
963
  installationIdMatch = result;
1057
964
  }
1058
-
1059
965
  if (result.deviceToken == this.data.deviceToken) {
1060
966
  deviceTokenMatches.push(result);
1061
967
  }
1062
- }); // Sanity checks when running a query
968
+ });
1063
969
 
970
+ // Sanity checks when running a query
1064
971
  if (this.query && this.query.objectId) {
1065
972
  if (!objectIdMatch) {
1066
973
  throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Object not found for update.');
1067
974
  }
1068
-
1069
975
  if (this.data.installationId && objectIdMatch.installationId && this.data.installationId !== objectIdMatch.installationId) {
1070
976
  throw new Parse.Error(136, 'installationId may not be changed in this ' + 'operation');
1071
977
  }
1072
-
1073
978
  if (this.data.deviceToken && objectIdMatch.deviceToken && this.data.deviceToken !== objectIdMatch.deviceToken && !this.data.installationId && !objectIdMatch.installationId) {
1074
979
  throw new Parse.Error(136, 'deviceToken may not be changed in this ' + 'operation');
1075
980
  }
1076
-
1077
981
  if (this.data.deviceType && this.data.deviceType && this.data.deviceType !== objectIdMatch.deviceType) {
1078
982
  throw new Parse.Error(136, 'deviceType may not be changed in this ' + 'operation');
1079
983
  }
1080
984
  }
1081
-
1082
985
  if (this.query && this.query.objectId && objectIdMatch) {
1083
986
  idMatch = objectIdMatch;
1084
987
  }
1085
-
1086
988
  if (installationId && installationIdMatch) {
1087
989
  idMatch = installationIdMatch;
1088
- } // need to specify deviceType only if it's new
1089
-
1090
-
990
+ }
991
+ // need to specify deviceType only if it's new
1091
992
  if (!this.query && !this.data.deviceType && !idMatch) {
1092
993
  throw new Parse.Error(135, 'deviceType must be specified in this operation');
1093
994
  }
@@ -1114,18 +1015,15 @@ RestWrite.prototype.handleInstallation = function () {
1114
1015
  $ne: installationId
1115
1016
  }
1116
1017
  };
1117
-
1118
1018
  if (this.data.appIdentifier) {
1119
1019
  delQuery['appIdentifier'] = this.data.appIdentifier;
1120
1020
  }
1121
-
1122
1021
  this.config.database.destroy('_Installation', delQuery).catch(err => {
1123
1022
  if (err.code == Parse.Error.OBJECT_NOT_FOUND) {
1124
1023
  // no deletions were made. Can be ignored.
1125
1024
  return;
1126
- } // rethrow the error
1127
-
1128
-
1025
+ }
1026
+ // rethrow the error
1129
1027
  throw err;
1130
1028
  });
1131
1029
  return;
@@ -1144,9 +1042,8 @@ RestWrite.prototype.handleInstallation = function () {
1144
1042
  if (err.code == Parse.Error.OBJECT_NOT_FOUND) {
1145
1043
  // no deletions were made. Can be ignored
1146
1044
  return;
1147
- } // rethrow the error
1148
-
1149
-
1045
+ }
1046
+ // rethrow the error
1150
1047
  throw err;
1151
1048
  });
1152
1049
  } else {
@@ -1156,9 +1053,9 @@ RestWrite.prototype.handleInstallation = function () {
1156
1053
  // device token.
1157
1054
  const delQuery = {
1158
1055
  deviceToken: this.data.deviceToken
1159
- }; // We have a unique install Id, use that to preserve
1056
+ };
1057
+ // We have a unique install Id, use that to preserve
1160
1058
  // the interesting installation
1161
-
1162
1059
  if (this.data.installationId) {
1163
1060
  delQuery['installationId'] = {
1164
1061
  $ne: this.data.installationId
@@ -1172,23 +1069,19 @@ RestWrite.prototype.handleInstallation = function () {
1172
1069
  // What to do here? can't really clean up everything...
1173
1070
  return idMatch.objectId;
1174
1071
  }
1175
-
1176
1072
  if (this.data.appIdentifier) {
1177
1073
  delQuery['appIdentifier'] = this.data.appIdentifier;
1178
1074
  }
1179
-
1180
1075
  this.config.database.destroy('_Installation', delQuery).catch(err => {
1181
1076
  if (err.code == Parse.Error.OBJECT_NOT_FOUND) {
1182
1077
  // no deletions were made. Can be ignored.
1183
1078
  return;
1184
- } // rethrow the error
1185
-
1186
-
1079
+ }
1080
+ // rethrow the error
1187
1081
  throw err;
1188
1082
  });
1189
- } // In non-merge scenarios, just return the installation match id
1190
-
1191
-
1083
+ }
1084
+ // In non-merge scenarios, just return the installation match id
1192
1085
  return idMatch.objectId;
1193
1086
  }
1194
1087
  }
@@ -1199,49 +1092,44 @@ RestWrite.prototype.handleInstallation = function () {
1199
1092
  };
1200
1093
  delete this.data.objectId;
1201
1094
  delete this.data.createdAt;
1202
- } // TODO: Validate ops (add/remove on channels, $inc on badge, etc.)
1203
-
1095
+ }
1096
+ // TODO: Validate ops (add/remove on channels, $inc on badge, etc.)
1204
1097
  });
1098
+
1205
1099
  return promise;
1206
- }; // If we short-circuited the object response - then we need to make sure we expand all the files,
1100
+ };
1101
+
1102
+ // If we short-circuited the object response - then we need to make sure we expand all the files,
1207
1103
  // since this might not have a query, meaning it won't return the full result back.
1208
1104
  // TODO: (nlutsenko) This should die when we move to per-class based controllers on _Session/_User
1209
-
1210
-
1211
1105
  RestWrite.prototype.expandFilesForExistingObjects = function () {
1212
1106
  // Check whether we have a short-circuited response - only then run expansion.
1213
1107
  if (this.response && this.response.response) {
1214
1108
  this.config.filesController.expandFilesInObject(this.config, this.response.response);
1215
1109
  }
1216
1110
  };
1217
-
1218
1111
  RestWrite.prototype.runDatabaseOperation = function () {
1219
1112
  if (this.response) {
1220
1113
  return;
1221
1114
  }
1222
-
1223
1115
  if (this.className === '_Role') {
1224
1116
  this.config.cacheController.role.clear();
1225
-
1226
1117
  if (this.config.liveQueryController) {
1227
1118
  this.config.liveQueryController.clearCachedRoles(this.auth.user);
1228
1119
  }
1229
1120
  }
1230
-
1231
1121
  if (this.className === '_User' && this.query && this.auth.isUnauthenticated()) {
1232
1122
  throw new Parse.Error(Parse.Error.SESSION_MISSING, `Cannot modify user ${this.query.objectId}.`);
1233
1123
  }
1234
-
1235
1124
  if (this.className === '_Product' && this.data.download) {
1236
1125
  this.data.downloadName = this.data.download.name;
1237
- } // TODO: Add better detection for ACL, ensuring a user can't be locked from
1238
- // their own user record.
1239
-
1126
+ }
1240
1127
 
1128
+ // TODO: Add better detection for ACL, ensuring a user can't be locked from
1129
+ // their own user record.
1241
1130
  if (this.data.ACL && this.data.ACL['*unresolved']) {
1242
1131
  throw new Parse.Error(Parse.Error.INVALID_ACL, 'Invalid ACL.');
1243
1132
  }
1244
-
1245
1133
  if (this.query) {
1246
1134
  // Force the user to not lockout
1247
1135
  // Matched with parse.com
@@ -1250,17 +1138,15 @@ RestWrite.prototype.runDatabaseOperation = function () {
1250
1138
  read: true,
1251
1139
  write: true
1252
1140
  };
1253
- } // update password timestamp if user password is being changed
1254
-
1255
-
1141
+ }
1142
+ // update password timestamp if user password is being changed
1256
1143
  if (this.className === '_User' && this.data._hashed_password && this.config.passwordPolicy && this.config.passwordPolicy.maxPasswordAge) {
1257
1144
  this.data._password_changed_at = Parse._encode(new Date());
1258
- } // Ignore createdAt when update
1259
-
1260
-
1145
+ }
1146
+ // Ignore createdAt when update
1261
1147
  delete this.data.createdAt;
1262
- let defer = Promise.resolve(); // if password history is enabled then save the current password to history
1263
-
1148
+ let defer = Promise.resolve();
1149
+ // if password history is enabled then save the current password to history
1264
1150
  if (this.className === '_User' && this.data._hashed_password && this.config.passwordPolicy && this.config.passwordPolicy.maxPasswordHistory) {
1265
1151
  defer = this.config.database.find('_User', {
1266
1152
  objectId: this.objectId()
@@ -1270,31 +1156,24 @@ RestWrite.prototype.runDatabaseOperation = function () {
1270
1156
  if (results.length != 1) {
1271
1157
  throw undefined;
1272
1158
  }
1273
-
1274
1159
  const user = results[0];
1275
1160
  let oldPasswords = [];
1276
-
1277
1161
  if (user._password_history) {
1278
1162
  oldPasswords = _lodash.default.take(user._password_history, this.config.passwordPolicy.maxPasswordHistory);
1279
- } //n-1 passwords go into history including last password
1280
-
1281
-
1163
+ }
1164
+ //n-1 passwords go into history including last password
1282
1165
  while (oldPasswords.length > Math.max(0, this.config.passwordPolicy.maxPasswordHistory - 2)) {
1283
1166
  oldPasswords.shift();
1284
1167
  }
1285
-
1286
1168
  oldPasswords.push(user.password);
1287
1169
  this.data._password_history = oldPasswords;
1288
1170
  });
1289
1171
  }
1290
-
1291
1172
  return defer.then(() => {
1292
1173
  // Run an update
1293
1174
  return this.config.database.update(this.className, this.query, this.data, this.runOptions, false, false, this.validSchemaController).then(response => {
1294
1175
  response.updatedAt = this.updatedAt;
1295
-
1296
1176
  this._updateResponseWithData(response, this.data);
1297
-
1298
1177
  this.response = {
1299
1178
  response
1300
1179
  };
@@ -1303,50 +1182,47 @@ RestWrite.prototype.runDatabaseOperation = function () {
1303
1182
  } else {
1304
1183
  // Set the default ACL and password timestamp for the new _User
1305
1184
  if (this.className === '_User') {
1306
- var ACL = this.data.ACL; // default public r/w ACL
1307
-
1185
+ var ACL = this.data.ACL;
1186
+ // default public r/w ACL
1308
1187
  if (!ACL) {
1309
1188
  ACL = {};
1310
-
1311
1189
  if (!this.config.enforcePrivateUsers) {
1312
1190
  ACL['*'] = {
1313
1191
  read: true,
1314
1192
  write: false
1315
1193
  };
1316
1194
  }
1317
- } // make sure the user is not locked down
1318
-
1319
-
1195
+ }
1196
+ // make sure the user is not locked down
1320
1197
  ACL[this.data.objectId] = {
1321
1198
  read: true,
1322
1199
  write: true
1323
1200
  };
1324
- this.data.ACL = ACL; // password timestamp to be used when password expiry policy is enforced
1325
-
1201
+ this.data.ACL = ACL;
1202
+ // password timestamp to be used when password expiry policy is enforced
1326
1203
  if (this.config.passwordPolicy && this.config.passwordPolicy.maxPasswordAge) {
1327
1204
  this.data._password_changed_at = Parse._encode(new Date());
1328
1205
  }
1329
- } // Run a create
1330
-
1206
+ }
1331
1207
 
1208
+ // Run a create
1332
1209
  return this.config.database.create(this.className, this.data, this.runOptions, false, this.validSchemaController).catch(error => {
1333
1210
  if (this.className !== '_User' || error.code !== Parse.Error.DUPLICATE_VALUE) {
1334
1211
  throw error;
1335
- } // Quick check, if we were able to infer the duplicated field name
1336
-
1212
+ }
1337
1213
 
1214
+ // Quick check, if we were able to infer the duplicated field name
1338
1215
  if (error && error.userInfo && error.userInfo.duplicated_field === 'username') {
1339
1216
  throw new Parse.Error(Parse.Error.USERNAME_TAKEN, 'Account already exists for this username.');
1340
1217
  }
1341
-
1342
1218
  if (error && error.userInfo && error.userInfo.duplicated_field === 'email') {
1343
1219
  throw new Parse.Error(Parse.Error.EMAIL_TAKEN, 'Account already exists for this email address.');
1344
- } // If this was a failed user creation due to username or email already taken, we need to
1220
+ }
1221
+
1222
+ // If this was a failed user creation due to username or email already taken, we need to
1345
1223
  // check whether it was username or email and return the appropriate error.
1346
1224
  // Fallback to the original method
1347
1225
  // TODO: See if we can later do this without additional queries by using named indexes.
1348
-
1349
-
1350
1226
  return this.config.database.find(this.className, {
1351
1227
  username: this.data.username,
1352
1228
  objectId: {
@@ -1358,7 +1234,6 @@ RestWrite.prototype.runDatabaseOperation = function () {
1358
1234
  if (results.length > 0) {
1359
1235
  throw new Parse.Error(Parse.Error.USERNAME_TAKEN, 'Account already exists for this username.');
1360
1236
  }
1361
-
1362
1237
  return this.config.database.find(this.className, {
1363
1238
  email: this.data.email,
1364
1239
  objectId: {
@@ -1371,19 +1246,15 @@ RestWrite.prototype.runDatabaseOperation = function () {
1371
1246
  if (results.length > 0) {
1372
1247
  throw new Parse.Error(Parse.Error.EMAIL_TAKEN, 'Account already exists for this email address.');
1373
1248
  }
1374
-
1375
1249
  throw new Parse.Error(Parse.Error.DUPLICATE_VALUE, 'A duplicate value for a field with unique values was provided');
1376
1250
  });
1377
1251
  }).then(response => {
1378
1252
  response.objectId = this.data.objectId;
1379
1253
  response.createdAt = this.data.createdAt;
1380
-
1381
1254
  if (this.responseShouldHaveUsername) {
1382
1255
  response.username = this.data.username;
1383
1256
  }
1384
-
1385
1257
  this._updateResponseWithData(response, this.data);
1386
-
1387
1258
  this.response = {
1388
1259
  status: 201,
1389
1260
  response,
@@ -1391,40 +1262,36 @@ RestWrite.prototype.runDatabaseOperation = function () {
1391
1262
  };
1392
1263
  });
1393
1264
  }
1394
- }; // Returns nothing - doesn't wait for the trigger.
1395
-
1265
+ };
1396
1266
 
1267
+ // Returns nothing - doesn't wait for the trigger.
1397
1268
  RestWrite.prototype.runAfterSaveTrigger = function () {
1398
1269
  if (!this.response || !this.response.response || this.runOptions.many) {
1399
1270
  return;
1400
- } // Avoid doing any setup for triggers if there is no 'afterSave' trigger for this class.
1401
-
1271
+ }
1402
1272
 
1273
+ // Avoid doing any setup for triggers if there is no 'afterSave' trigger for this class.
1403
1274
  const hasAfterSaveHook = triggers.triggerExists(this.className, triggers.Types.afterSave, this.config.applicationId);
1404
1275
  const hasLiveQuery = this.config.liveQueryController.hasLiveQuery(this.className);
1405
-
1406
1276
  if (!hasAfterSaveHook && !hasLiveQuery) {
1407
1277
  return Promise.resolve();
1408
1278
  }
1409
-
1410
1279
  const {
1411
1280
  originalObject,
1412
1281
  updatedObject
1413
1282
  } = this.buildParseObjects();
1414
-
1415
1283
  updatedObject._handleSaveResponse(this.response.response, this.response.status || 200);
1416
-
1417
1284
  this.config.database.loadSchema().then(schemaController => {
1418
1285
  // Notifiy LiveQueryServer if possible
1419
1286
  const perms = schemaController.getClassLevelPermissions(updatedObject.className);
1420
1287
  this.config.liveQueryController.onAfterSave(updatedObject.className, updatedObject, originalObject, perms);
1421
- }); // Run afterSave trigger
1288
+ });
1422
1289
 
1290
+ // Run afterSave trigger
1423
1291
  return triggers.maybeRunTrigger(triggers.Types.afterSave, this.auth, updatedObject, originalObject, this.config, this.context).then(result => {
1424
1292
  const jsonReturned = result && !result._toFullJSON;
1425
-
1426
1293
  if (jsonReturned) {
1427
- this.pendingOps = {};
1294
+ this.pendingOps.operations = {};
1428
1295
  this.response.response = result;
1429
1296
  } else {
1430
1297
  this.response.response = this._updateResponseWithData((result || updatedObject).toJSON(), this.data);
@@ -1432,57 +1299,51 @@ RestWrite.prototype.runAfterSaveTrigger = function () {
1432
1299
  }).catch(function (err) {
1433
1300
  _logger.default.warn('afterSave caught an error', err);
1434
1301
  });
1435
- }; // A helper to figure out what location this operation happens at.
1436
-
1302
+ };
1437
1303
 
1304
+ // A helper to figure out what location this operation happens at.
1438
1305
  RestWrite.prototype.location = function () {
1439
1306
  var middle = this.className === '_User' ? '/users/' : '/classes/' + this.className + '/';
1440
1307
  const mount = this.config.mount || this.config.serverURL;
1441
1308
  return mount + middle + this.data.objectId;
1442
- }; // A helper to get the object id for this operation.
1443
- // Because it could be either on the query or on the data
1444
-
1309
+ };
1445
1310
 
1311
+ // A helper to get the object id for this operation.
1312
+ // Because it could be either on the query or on the data
1446
1313
  RestWrite.prototype.objectId = function () {
1447
1314
  return this.data.objectId || this.query.objectId;
1448
- }; // Returns a copy of the data and delete bad keys (_auth_data, _hashed_password...)
1449
-
1315
+ };
1450
1316
 
1317
+ // Returns a copy of the data and delete bad keys (_auth_data, _hashed_password...)
1451
1318
  RestWrite.prototype.sanitizedData = function () {
1452
1319
  const data = Object.keys(this.data).reduce((data, key) => {
1453
1320
  // Regexp comes from Parse.Object.prototype.validate
1454
1321
  if (!/^[A-Za-z][0-9A-Za-z_]*$/.test(key)) {
1455
1322
  delete data[key];
1456
1323
  }
1457
-
1458
1324
  return data;
1459
1325
  }, deepcopy(this.data));
1460
1326
  return Parse._decode(undefined, data);
1461
- }; // Returns an updated copy of the object
1462
-
1327
+ };
1463
1328
 
1329
+ // Returns an updated copy of the object
1464
1330
  RestWrite.prototype.buildParseObjects = function () {
1465
1331
  var _this$query;
1466
-
1467
1332
  const extraData = {
1468
1333
  className: this.className,
1469
1334
  objectId: (_this$query = this.query) === null || _this$query === void 0 ? void 0 : _this$query.objectId
1470
1335
  };
1471
1336
  let originalObject;
1472
-
1473
1337
  if (this.query && this.query.objectId) {
1474
1338
  originalObject = triggers.inflate(extraData, this.originalData);
1475
1339
  }
1476
-
1477
1340
  const className = Parse.Object.fromJSON(extraData);
1478
1341
  const readOnlyAttributes = className.constructor.readOnlyAttributes ? className.constructor.readOnlyAttributes() : [];
1479
-
1480
1342
  if (!this.originalData) {
1481
1343
  for (const attribute of readOnlyAttributes) {
1482
1344
  extraData[attribute] = this.data[attribute];
1483
1345
  }
1484
1346
  }
1485
-
1486
1347
  const updatedObject = triggers.inflate(extraData, this.originalData);
1487
1348
  Object.keys(this.data).reduce(function (data, key) {
1488
1349
  if (key.indexOf('.') > 0) {
@@ -1495,59 +1356,45 @@ RestWrite.prototype.buildParseObjects = function () {
1495
1356
  const splittedKey = key.split('.');
1496
1357
  const parentProp = splittedKey[0];
1497
1358
  let parentVal = updatedObject.get(parentProp);
1498
-
1499
1359
  if (typeof parentVal !== 'object') {
1500
1360
  parentVal = {};
1501
1361
  }
1502
-
1503
1362
  parentVal[splittedKey[1]] = data[key];
1504
1363
  updatedObject.set(parentProp, parentVal);
1505
1364
  }
1506
-
1507
1365
  delete data[key];
1508
1366
  }
1509
-
1510
1367
  return data;
1511
1368
  }, deepcopy(this.data));
1512
1369
  const sanitized = this.sanitizedData();
1513
-
1514
1370
  for (const attribute of readOnlyAttributes) {
1515
1371
  delete sanitized[attribute];
1516
1372
  }
1517
-
1518
1373
  updatedObject.set(sanitized);
1519
1374
  return {
1520
1375
  updatedObject,
1521
1376
  originalObject
1522
1377
  };
1523
1378
  };
1524
-
1525
1379
  RestWrite.prototype.cleanUserAuthData = function () {
1526
1380
  if (this.response && this.response.response && this.className === '_User') {
1527
1381
  const user = this.response.response;
1528
-
1529
1382
  if (user.authData) {
1530
1383
  Object.keys(user.authData).forEach(provider => {
1531
1384
  if (user.authData[provider] === null) {
1532
1385
  delete user.authData[provider];
1533
1386
  }
1534
1387
  });
1535
-
1536
1388
  if (Object.keys(user.authData).length == 0) {
1537
1389
  delete user.authData;
1538
1390
  }
1539
1391
  }
1540
1392
  }
1541
1393
  };
1542
-
1543
1394
  RestWrite.prototype._updateResponseWithData = function (response, data) {
1544
- const {
1545
- updatedObject
1546
- } = this.buildParseObjects();
1547
1395
  const stateController = Parse.CoreManager.getObjectStateController();
1548
- const [pending] = stateController.getPendingOps(updatedObject._getStateIdentifier());
1549
-
1550
- for (const key in this.pendingOps) {
1396
+ const [pending] = stateController.getPendingOps(this.pendingOps.identifier);
1397
+ for (const key in this.pendingOps.operations) {
1551
1398
  if (!pending[key]) {
1552
1399
  data[key] = this.originalData ? this.originalData[key] : {
1553
1400
  __op: 'Delete'
@@ -1555,44 +1402,35 @@ RestWrite.prototype._updateResponseWithData = function (response, data) {
1555
1402
  this.storage.fieldsChangedByTrigger.push(key);
1556
1403
  }
1557
1404
  }
1558
-
1559
1405
  const skipKeys = [...(_SchemaController.requiredColumns.read[this.className] || [])];
1560
-
1561
1406
  if (!this.query) {
1562
1407
  skipKeys.push('objectId', 'createdAt');
1563
1408
  } else {
1564
1409
  skipKeys.push('updatedAt');
1565
1410
  delete response.objectId;
1566
1411
  }
1567
-
1568
1412
  for (const key in response) {
1569
1413
  if (skipKeys.includes(key)) {
1570
1414
  continue;
1571
1415
  }
1572
-
1573
1416
  const value = response[key];
1574
-
1575
1417
  if (value == null || value.__type && value.__type === 'Pointer' || util.isDeepStrictEqual(data[key], value) || util.isDeepStrictEqual((this.originalData || {})[key], value)) {
1576
1418
  delete response[key];
1577
1419
  }
1578
1420
  }
1579
-
1580
1421
  if (_lodash.default.isEmpty(this.storage.fieldsChangedByTrigger)) {
1581
1422
  return response;
1582
1423
  }
1583
-
1584
1424
  const clientSupportsDelete = ClientSDK.supportsForwardDelete(this.clientSDK);
1585
1425
  this.storage.fieldsChangedByTrigger.forEach(fieldName => {
1586
1426
  const dataValue = data[fieldName];
1587
-
1588
1427
  if (!Object.prototype.hasOwnProperty.call(response, fieldName)) {
1589
1428
  response[fieldName] = dataValue;
1590
- } // Strips operations from responses
1591
-
1429
+ }
1592
1430
 
1431
+ // Strips operations from responses
1593
1432
  if (response[fieldName] && response[fieldName].__op) {
1594
1433
  delete response[fieldName];
1595
-
1596
1434
  if (clientSupportsDelete && dataValue.__op == 'Delete') {
1597
1435
  response[fieldName] = dataValue;
1598
1436
  }
@@ -1600,21 +1438,18 @@ RestWrite.prototype._updateResponseWithData = function (response, data) {
1600
1438
  });
1601
1439
  return response;
1602
1440
  };
1603
-
1604
1441
  RestWrite.prototype.checkProhibitedKeywords = function (data) {
1605
1442
  if (this.config.requestKeywordDenylist) {
1606
1443
  // Scan request data for denied keywords
1607
1444
  for (const keyword of this.config.requestKeywordDenylist) {
1608
1445
  const match = Utils.objectContainsKeyValue(data, keyword.key, keyword.value);
1609
-
1610
1446
  if (match) {
1611
1447
  throw new Parse.Error(Parse.Error.INVALID_KEY_NAME, `Prohibited keyword in request data: ${JSON.stringify(keyword)}.`);
1612
1448
  }
1613
1449
  }
1614
1450
  }
1615
1451
  };
1616
-
1617
1452
  var _default = RestWrite;
1618
1453
  exports.default = _default;
1619
1454
  module.exports = RestWrite;
1620
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/RestWrite.js"],"names":["SchemaController","require","deepcopy","Auth","Utils","cryptoUtils","passwordCrypto","Parse","triggers","ClientSDK","util","RestWrite","config","auth","className","query","data","originalData","clientSDK","context","action","isReadOnly","Error","OPERATION_FORBIDDEN","storage","runOptions","allowCustomObjectId","Object","prototype","hasOwnProperty","call","objectId","MISSING_OBJECT_ID","INVALID_KEY_NAME","id","checkProhibitedKeywords","response","updatedAt","_encode","Date","iso","validSchemaController","pendingOps","execute","Promise","resolve","then","getUserAndRoleACL","validateClientClassCreation","handleInstallation","handleSession","validateAuthData","runBeforeSaveTrigger","ensureUniqueAuthDataId","deleteEmailResetTokenIfNeeded","validateSchema","schemaController","setRequiredFieldsIfNeeded","transformUser","expandFilesForExistingObjects","destroyDuplicatedSessions","runDatabaseOperation","createSessionTokenIfNeeded","handleFollowup","runAfterSaveTrigger","cleanUserAuthData","authDataResponse","isMaster","acl","user","getUserRoles","roles","concat","allowClientClassCreation","systemClasses","indexOf","database","loadSchema","hasClass","validateObject","many","triggerExists","Types","beforeSave","applicationId","originalObject","updatedObject","buildParseObjects","stateController","CoreManager","getObjectStateController","pending","getPendingOps","_getStateIdentifier","databasePromise","update","create","result","length","OBJECT_NOT_FOUND","maybeRunTrigger","object","fieldsChangedByTrigger","_","reduce","value","key","isEqual","push","runBeforeLoginTrigger","userData","beforeLogin","extraData","filesController","expandFilesInObject","inflate","getAllClasses","allClasses","schema","find","oneClass","setRequiredFieldIfNeeded","fieldName","setDefault","undefined","__op","fields","defaultValue","required","VALIDATION_ERROR","createdAt","newObjectId","objectIdSize","keys","forEach","authData","hasUsernameAndPassword","username","password","isEmpty","USERNAME_MISSING","PASSWORD_MISSING","UNSUPPORTED_SERVICE","providers","canHandleAuthData","some","provider","providerAuthData","hasToken","getUserId","handleAuthData","filteredObjectsByACL","objects","filter","ACL","hasAuthDataId","r","findUsersWithAuthData","results","ACCOUNT_ALREADY_LINKED","userId","handleAuthDataValidation","validatedAuthData","userResult","authProvider","join","hasMutatedAuthData","mutatedAuthData","isCurrentUserLoggedOrMaster","isLogin","location","checkIfUserHasProvidedConfiguredProvidersForLogin","allowExpiredAuthDataToken","res","promise","error","RestQuery","master","__type","session","cacheController","del","sessionToken","_validatePasswordPolicy","hash","hashedPassword","_hashed_password","_validateUserName","_validateEmail","randomString","responseShouldHaveUsername","$ne","limit","caseInsensitive","USERNAME_TAKEN","email","match","reject","INVALID_EMAIL_ADDRESS","EMAIL_TAKEN","userController","setEmailVerifyToken","passwordPolicy","_validatePasswordRequirements","_validatePasswordHistory","policyError","validationError","containsUsernameError","patternValidator","validatorCallback","doNotAllowUsername","maxPasswordHistory","oldPasswords","_password_history","take","newPassword","promises","map","compare","all","catch","err","preventLoginWithUnverifiedEmail","verifyUserEmails","createSessionToken","installationId","sessionData","createSession","createdWith","additionalSessionData","token","newToken","expiresAt","generateSessionExpiresAt","assign","addOps","_perishable_token","_perishable_token_expires_at","destroy","revokeSessionOnPasswordReset","sessionQuery","bind","sendVerificationEmail","INVALID_SESSION_TOKEN","$and","INTERNAL_SERVER_ERROR","status","deviceToken","toLowerCase","deviceType","idMatch","objectIdMatch","installationIdMatch","deviceTokenMatches","orQueries","$or","delQuery","appIdentifier","code","objId","role","clear","liveQueryController","clearCachedRoles","isUnauthenticated","SESSION_MISSING","download","downloadName","name","INVALID_ACL","read","write","maxPasswordAge","_password_changed_at","defer","Math","max","shift","_updateResponseWithData","enforcePrivateUsers","DUPLICATE_VALUE","userInfo","duplicated_field","hasAfterSaveHook","afterSave","hasLiveQuery","_handleSaveResponse","perms","getClassLevelPermissions","onAfterSave","jsonReturned","_toFullJSON","toJSON","logger","warn","middle","mount","serverURL","sanitizedData","test","_decode","fromJSON","readOnlyAttributes","constructor","attribute","includes","set","splittedKey","split","parentProp","parentVal","get","sanitized","skipKeys","requiredColumns","isDeepStrictEqual","clientSupportsDelete","supportsForwardDelete","dataValue","requestKeywordDenylist","keyword","objectContainsKeyValue","JSON","stringify","module","exports"],"mappings":"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","sourcesContent":["// A RestWrite encapsulates everything we need to run an operation\n// that writes to the database.\n// This could be either a \"create\" or an \"update\".\n\nvar SchemaController = require('./Controllers/SchemaController');\nvar deepcopy = require('deepcopy');\n\nconst Auth = require('./Auth');\nconst Utils = require('./Utils');\nvar cryptoUtils = require('./cryptoUtils');\nvar passwordCrypto = require('./password');\nvar Parse = require('parse/node');\nvar triggers = require('./triggers');\nvar ClientSDK = require('./ClientSDK');\nconst util = require('util');\nimport RestQuery from './RestQuery';\nimport _ from 'lodash';\nimport logger from './logger';\nimport { requiredColumns } from './Controllers/SchemaController';\n\n// query and data are both provided in REST API format. So data\n// types are encoded by plain old objects.\n// If query is null, this is a \"create\" and the data in data should be\n// created.\n// Otherwise this is an \"update\" - the object matching the query\n// should get updated with data.\n// RestWrite will handle objectId, createdAt, and updatedAt for\n// everything. It also knows to use triggers and special modifications\n// for the _User class.\nfunction RestWrite(config, auth, className, query, data, originalData, clientSDK, context, action) {\n  if (auth.isReadOnly) {\n    throw new Parse.Error(\n      Parse.Error.OPERATION_FORBIDDEN,\n      'Cannot perform a write operation when using readOnlyMasterKey'\n    );\n  }\n  this.config = config;\n  this.auth = auth;\n  this.className = className;\n  this.clientSDK = clientSDK;\n  this.storage = {};\n  this.runOptions = {};\n  this.context = context || {};\n\n  if (action) {\n    this.runOptions.action = action;\n  }\n\n  if (!query) {\n    if (this.config.allowCustomObjectId) {\n      if (Object.prototype.hasOwnProperty.call(data, 'objectId') && !data.objectId) {\n        throw new Parse.Error(\n          Parse.Error.MISSING_OBJECT_ID,\n          'objectId must not be empty, null or undefined'\n        );\n      }\n    } else {\n      if (data.objectId) {\n        throw new Parse.Error(Parse.Error.INVALID_KEY_NAME, 'objectId is an invalid field name.');\n      }\n      if (data.id) {\n        throw new Parse.Error(Parse.Error.INVALID_KEY_NAME, 'id is an invalid field name.');\n      }\n    }\n  }\n\n  this.checkProhibitedKeywords(data);\n\n  // When the operation is complete, this.response may have several\n  // fields.\n  // response: the actual data to be returned\n  // status: the http status code. if not present, treated like a 200\n  // location: the location header. if not present, no location header\n  this.response = null;\n\n  // Processing this operation may mutate our data, so we operate on a\n  // copy\n  this.query = deepcopy(query);\n  this.data = deepcopy(data);\n  // We never change originalData, so we do not need a deep copy\n  this.originalData = originalData;\n\n  // The timestamp we'll use for this whole operation\n  this.updatedAt = Parse._encode(new Date()).iso;\n\n  // Shared SchemaController to be reused to reduce the number of loadSchema() calls per request\n  // Once set the schemaData should be immutable\n  this.validSchemaController = null;\n  this.pendingOps = {};\n}\n\n// A convenient method to perform all the steps of processing the\n// write, in order.\n// Returns a promise for a {response, status, location} object.\n// status and location are optional.\nRestWrite.prototype.execute = function () {\n  return Promise.resolve()\n    .then(() => {\n      return this.getUserAndRoleACL();\n    })\n    .then(() => {\n      return this.validateClientClassCreation();\n    })\n    .then(() => {\n      return this.handleInstallation();\n    })\n    .then(() => {\n      return this.handleSession();\n    })\n    .then(() => {\n      return this.validateAuthData();\n    })\n    .then(() => {\n      return this.runBeforeSaveTrigger();\n    })\n    .then(() => {\n      return this.ensureUniqueAuthDataId();\n    })\n    .then(() => {\n      return this.deleteEmailResetTokenIfNeeded();\n    })\n    .then(() => {\n      return this.validateSchema();\n    })\n    .then(schemaController => {\n      this.validSchemaController = schemaController;\n      return this.setRequiredFieldsIfNeeded();\n    })\n    .then(() => {\n      return this.transformUser();\n    })\n    .then(() => {\n      return this.expandFilesForExistingObjects();\n    })\n    .then(() => {\n      return this.destroyDuplicatedSessions();\n    })\n    .then(() => {\n      return this.runDatabaseOperation();\n    })\n    .then(() => {\n      return this.createSessionTokenIfNeeded();\n    })\n    .then(() => {\n      return this.handleFollowup();\n    })\n    .then(() => {\n      return this.runAfterSaveTrigger();\n    })\n    .then(() => {\n      return this.cleanUserAuthData();\n    })\n    .then(() => {\n      // Append the authDataResponse if exists\n      if (this.authDataResponse) {\n        if (this.response && this.response.response) {\n          this.response.response.authDataResponse = this.authDataResponse;\n        }\n      }\n      return this.response;\n    });\n};\n\n// Uses the Auth object to get the list of roles, adds the user id\nRestWrite.prototype.getUserAndRoleACL = function () {\n  if (this.auth.isMaster) {\n    return Promise.resolve();\n  }\n\n  this.runOptions.acl = ['*'];\n\n  if (this.auth.user) {\n    return this.auth.getUserRoles().then(roles => {\n      this.runOptions.acl = this.runOptions.acl.concat(roles, [this.auth.user.id]);\n      return;\n    });\n  } else {\n    return Promise.resolve();\n  }\n};\n\n// Validates this operation against the allowClientClassCreation config.\nRestWrite.prototype.validateClientClassCreation = function () {\n  if (\n    this.config.allowClientClassCreation === false &&\n    !this.auth.isMaster &&\n    SchemaController.systemClasses.indexOf(this.className) === -1\n  ) {\n    return this.config.database\n      .loadSchema()\n      .then(schemaController => schemaController.hasClass(this.className))\n      .then(hasClass => {\n        if (hasClass !== true) {\n          throw new Parse.Error(\n            Parse.Error.OPERATION_FORBIDDEN,\n            'This user is not allowed to access ' + 'non-existent class: ' + this.className\n          );\n        }\n      });\n  } else {\n    return Promise.resolve();\n  }\n};\n\n// Validates this operation against the schema.\nRestWrite.prototype.validateSchema = function () {\n  return this.config.database.validateObject(\n    this.className,\n    this.data,\n    this.query,\n    this.runOptions\n  );\n};\n\n// Runs any beforeSave triggers against this operation.\n// Any change leads to our data being mutated.\nRestWrite.prototype.runBeforeSaveTrigger = function () {\n  if (this.response || this.runOptions.many) {\n    return;\n  }\n\n  // Avoid doing any setup for triggers if there is no 'beforeSave' trigger for this class.\n  if (\n    !triggers.triggerExists(this.className, triggers.Types.beforeSave, this.config.applicationId)\n  ) {\n    return Promise.resolve();\n  }\n\n  const { originalObject, updatedObject } = this.buildParseObjects();\n\n  const stateController = Parse.CoreManager.getObjectStateController();\n  const [pending] = stateController.getPendingOps(updatedObject._getStateIdentifier());\n  this.pendingOps = { ...pending };\n\n  return Promise.resolve()\n    .then(() => {\n      // Before calling the trigger, validate the permissions for the save operation\n      let databasePromise = null;\n      if (this.query) {\n        // Validate for updating\n        databasePromise = this.config.database.update(\n          this.className,\n          this.query,\n          this.data,\n          this.runOptions,\n          true,\n          true\n        );\n      } else {\n        // Validate for creating\n        databasePromise = this.config.database.create(\n          this.className,\n          this.data,\n          this.runOptions,\n          true\n        );\n      }\n      // In the case that there is no permission for the operation, it throws an error\n      return databasePromise.then(result => {\n        if (!result || result.length <= 0) {\n          throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Object not found.');\n        }\n      });\n    })\n    .then(() => {\n      return triggers.maybeRunTrigger(\n        triggers.Types.beforeSave,\n        this.auth,\n        updatedObject,\n        originalObject,\n        this.config,\n        this.context\n      );\n    })\n    .then(response => {\n      if (response && response.object) {\n        this.storage.fieldsChangedByTrigger = _.reduce(\n          response.object,\n          (result, value, key) => {\n            if (!_.isEqual(this.data[key], value)) {\n              result.push(key);\n            }\n            return result;\n          },\n          []\n        );\n        this.data = response.object;\n        // We should delete the objectId for an update write\n        if (this.query && this.query.objectId) {\n          delete this.data.objectId;\n        }\n      }\n      this.checkProhibitedKeywords(this.data);\n    });\n};\n\nRestWrite.prototype.runBeforeLoginTrigger = async function (userData) {\n  // Avoid doing any setup for triggers if there is no 'beforeLogin' trigger\n  if (\n    !triggers.triggerExists(this.className, triggers.Types.beforeLogin, this.config.applicationId)\n  ) {\n    return;\n  }\n\n  // Cloud code gets a bit of extra data for its objects\n  const extraData = { className: this.className };\n\n  // Expand file objects\n  this.config.filesController.expandFilesInObject(this.config, userData);\n\n  const user = triggers.inflate(extraData, userData);\n\n  // no need to return a response\n  await triggers.maybeRunTrigger(\n    triggers.Types.beforeLogin,\n    this.auth,\n    user,\n    null,\n    this.config,\n    this.context\n  );\n};\n\nRestWrite.prototype.setRequiredFieldsIfNeeded = function () {\n  if (this.data) {\n    return this.validSchemaController.getAllClasses().then(allClasses => {\n      const schema = allClasses.find(oneClass => oneClass.className === this.className);\n      const setRequiredFieldIfNeeded = (fieldName, setDefault) => {\n        if (\n          this.data[fieldName] === undefined ||\n          this.data[fieldName] === null ||\n          this.data[fieldName] === '' ||\n          (typeof this.data[fieldName] === 'object' && this.data[fieldName].__op === 'Delete')\n        ) {\n          if (\n            setDefault &&\n            schema.fields[fieldName] &&\n            schema.fields[fieldName].defaultValue !== null &&\n            schema.fields[fieldName].defaultValue !== undefined &&\n            (this.data[fieldName] === undefined ||\n              (typeof this.data[fieldName] === 'object' && this.data[fieldName].__op === 'Delete'))\n          ) {\n            this.data[fieldName] = schema.fields[fieldName].defaultValue;\n            this.storage.fieldsChangedByTrigger = this.storage.fieldsChangedByTrigger || [];\n            if (this.storage.fieldsChangedByTrigger.indexOf(fieldName) < 0) {\n              this.storage.fieldsChangedByTrigger.push(fieldName);\n            }\n          } else if (schema.fields[fieldName] && schema.fields[fieldName].required === true) {\n            throw new Parse.Error(Parse.Error.VALIDATION_ERROR, `${fieldName} is required`);\n          }\n        }\n      };\n\n      // Add default fields\n      this.data.updatedAt = this.updatedAt;\n      if (!this.query) {\n        this.data.createdAt = this.updatedAt;\n\n        // Only assign new objectId if we are creating new object\n        if (!this.data.objectId) {\n          this.data.objectId = cryptoUtils.newObjectId(this.config.objectIdSize);\n        }\n        if (schema) {\n          Object.keys(schema.fields).forEach(fieldName => {\n            setRequiredFieldIfNeeded(fieldName, true);\n          });\n        }\n      } else if (schema) {\n        Object.keys(this.data).forEach(fieldName => {\n          setRequiredFieldIfNeeded(fieldName, false);\n        });\n      }\n    });\n  }\n  return Promise.resolve();\n};\n\n// Transforms auth data for a user object.\n// Does nothing if this isn't a user object.\n// Returns a promise for when we're done if it can't finish this tick.\nRestWrite.prototype.validateAuthData = function () {\n  if (this.className !== '_User') {\n    return;\n  }\n\n  const authData = this.data.authData;\n  const hasUsernameAndPassword =\n    typeof this.data.username === 'string' && typeof this.data.password === 'string';\n\n  if (!this.query && !authData) {\n    if (typeof this.data.username !== 'string' || _.isEmpty(this.data.username)) {\n      throw new Parse.Error(Parse.Error.USERNAME_MISSING, 'bad or missing username');\n    }\n    if (typeof this.data.password !== 'string' || _.isEmpty(this.data.password)) {\n      throw new Parse.Error(Parse.Error.PASSWORD_MISSING, 'password is required');\n    }\n  }\n\n  if (\n    (authData && !Object.keys(authData).length) ||\n    !Object.prototype.hasOwnProperty.call(this.data, 'authData')\n  ) {\n    // Nothing to validate here\n    return;\n  } else if (Object.prototype.hasOwnProperty.call(this.data, 'authData') && !this.data.authData) {\n    // Handle saving authData to null\n    throw new Parse.Error(\n      Parse.Error.UNSUPPORTED_SERVICE,\n      'This authentication method is unsupported.'\n    );\n  }\n\n  var providers = Object.keys(authData);\n  if (providers.length > 0) {\n    const canHandleAuthData = providers.some(provider => {\n      var providerAuthData = authData[provider];\n      var hasToken = providerAuthData && providerAuthData.id;\n      return hasToken || providerAuthData === null;\n    });\n    if (canHandleAuthData || hasUsernameAndPassword || this.auth.isMaster || this.getUserId()) {\n      return this.handleAuthData(authData);\n    }\n  }\n  throw new Parse.Error(\n    Parse.Error.UNSUPPORTED_SERVICE,\n    'This authentication method is unsupported.'\n  );\n};\n\nRestWrite.prototype.filteredObjectsByACL = function (objects) {\n  if (this.auth.isMaster) {\n    return objects;\n  }\n  return objects.filter(object => {\n    if (!object.ACL) {\n      return true; // legacy users that have no ACL field on them\n    }\n    // Regular users that have been locked out.\n    return object.ACL && Object.keys(object.ACL).length > 0;\n  });\n};\n\nRestWrite.prototype.getUserId = function () {\n  if (this.query && this.query.objectId && this.className === '_User') {\n    return this.query.objectId;\n  } else if (this.auth && this.auth.user && this.auth.user.id) {\n    return this.auth.user.id;\n  }\n};\n\n// Developers are allowed to change authData via before save trigger\n// we need after before save to ensure that the developer\n// is not currently duplicating auth data ID\nRestWrite.prototype.ensureUniqueAuthDataId = async function () {\n  if (this.className !== '_User' || !this.data.authData) {\n    return;\n  }\n\n  const hasAuthDataId = Object.keys(this.data.authData).some(\n    key => this.data.authData[key] && this.data.authData[key].id\n  );\n\n  if (!hasAuthDataId) return;\n\n  const r = await Auth.findUsersWithAuthData(this.config, this.data.authData);\n  const results = this.filteredObjectsByACL(r);\n  if (results.length > 1) {\n    throw new Parse.Error(Parse.Error.ACCOUNT_ALREADY_LINKED, 'this auth is already used');\n  }\n  // use data.objectId in case of login time and found user during handle validateAuthData\n  const userId = this.getUserId() || this.data.objectId;\n  if (results.length === 1 && userId !== results[0].objectId) {\n    throw new Parse.Error(Parse.Error.ACCOUNT_ALREADY_LINKED, 'this auth is already used');\n  }\n};\n\nRestWrite.prototype.handleAuthData = async function (authData) {\n  const r = await Auth.findUsersWithAuthData(this.config, authData);\n  const results = this.filteredObjectsByACL(r);\n\n  if (results.length > 1) {\n    // To avoid https://github.com/parse-community/parse-server/security/advisories/GHSA-8w3j-g983-8jh5\n    // Let's run some validation before throwing\n    await Auth.handleAuthDataValidation(authData, this, results[0]);\n    throw new Parse.Error(Parse.Error.ACCOUNT_ALREADY_LINKED, 'this auth is already used');\n  }\n\n  // No user found with provided authData we need to validate\n  if (!results.length) {\n    const { authData: validatedAuthData, authDataResponse } = await Auth.handleAuthDataValidation(\n      authData,\n      this\n    );\n    this.authDataResponse = authDataResponse;\n    // Replace current authData by the new validated one\n    this.data.authData = validatedAuthData;\n    return;\n  }\n\n  // User found with provided authData\n  if (results.length === 1) {\n    const userId = this.getUserId();\n    const userResult = results[0];\n    // Prevent duplicate authData id\n    if (userId && userId !== userResult.objectId) {\n      throw new Parse.Error(Parse.Error.ACCOUNT_ALREADY_LINKED, 'this auth is already used');\n    }\n\n    this.storage.authProvider = Object.keys(authData).join(',');\n\n    const { hasMutatedAuthData, mutatedAuthData } = Auth.hasMutatedAuthData(\n      authData,\n      userResult.authData\n    );\n\n    const isCurrentUserLoggedOrMaster =\n      (this.auth && this.auth.user && this.auth.user.id === userResult.objectId) ||\n      this.auth.isMaster;\n\n    const isLogin = !userId;\n\n    if (isLogin || isCurrentUserLoggedOrMaster) {\n      // no user making the call\n      // OR the user making the call is the right one\n      // Login with auth data\n      delete results[0].password;\n\n      // need to set the objectId first otherwise location has trailing undefined\n      this.data.objectId = userResult.objectId;\n\n      if (!this.query || !this.query.objectId) {\n        this.response = {\n          response: userResult,\n          location: this.location(),\n        };\n        // Run beforeLogin hook before storing any updates\n        // to authData on the db; changes to userResult\n        // will be ignored.\n        await this.runBeforeLoginTrigger(deepcopy(userResult));\n\n        // If we are in login operation via authData\n        // we need to be sure that the user has provided\n        // required authData\n        Auth.checkIfUserHasProvidedConfiguredProvidersForLogin(\n          authData,\n          userResult.authData,\n          this.config\n        );\n      }\n\n      // Prevent validating if no mutated data detected on update\n      if (!hasMutatedAuthData && isCurrentUserLoggedOrMaster) {\n        return;\n      }\n\n      // Force to validate all provided authData on login\n      // on update only validate mutated ones\n      if (hasMutatedAuthData || !this.config.allowExpiredAuthDataToken) {\n        const res = await Auth.handleAuthDataValidation(\n          isLogin ? authData : mutatedAuthData,\n          this,\n          userResult\n        );\n        this.data.authData = res.authData;\n        this.authDataResponse = res.authDataResponse;\n      }\n\n      // IF we are in login we'll skip the database operation / beforeSave / afterSave etc...\n      // we need to set it up there.\n      // We are supposed to have a response only on LOGIN with authData, so we skip those\n      // If we're not logging in, but just updating the current user, we can safely skip that part\n      if (this.response) {\n        // Assign the new authData in the response\n        Object.keys(mutatedAuthData).forEach(provider => {\n          this.response.response.authData[provider] = mutatedAuthData[provider];\n        });\n\n        // Run the DB update directly, as 'master' only if authData contains some keys\n        // authData could not contains keys after validation if the authAdapter\n        // uses the `doNotSave` option. Just update the authData part\n        // Then we're good for the user, early exit of sorts\n        if (Object.keys(this.data.authData).length) {\n          await this.config.database.update(\n            this.className,\n            { objectId: this.data.objectId },\n            { authData: this.data.authData },\n            {}\n          );\n        }\n      }\n    }\n  }\n};\n\n// The non-third-party parts of User transformation\nRestWrite.prototype.transformUser = function () {\n  var promise = Promise.resolve();\n  if (this.className !== '_User') {\n    return promise;\n  }\n\n  if (!this.auth.isMaster && 'emailVerified' in this.data) {\n    const error = `Clients aren't allowed to manually update email verification.`;\n    throw new Parse.Error(Parse.Error.OPERATION_FORBIDDEN, error);\n  }\n\n  // Do not cleanup session if objectId is not set\n  if (this.query && this.objectId()) {\n    // If we're updating a _User object, we need to clear out the cache for that user. Find all their\n    // session tokens, and remove them from the cache.\n    promise = new RestQuery(this.config, Auth.master(this.config), '_Session', {\n      user: {\n        __type: 'Pointer',\n        className: '_User',\n        objectId: this.objectId(),\n      },\n    })\n      .execute()\n      .then(results => {\n        results.results.forEach(session =>\n          this.config.cacheController.user.del(session.sessionToken)\n        );\n      });\n  }\n\n  return promise\n    .then(() => {\n      // Transform the password\n      if (this.data.password === undefined) {\n        // ignore only if undefined. should proceed if empty ('')\n        return Promise.resolve();\n      }\n\n      if (this.query) {\n        this.storage['clearSessions'] = true;\n        // Generate a new session only if the user requested\n        if (!this.auth.isMaster) {\n          this.storage['generateNewSession'] = true;\n        }\n      }\n\n      return this._validatePasswordPolicy().then(() => {\n        return passwordCrypto.hash(this.data.password).then(hashedPassword => {\n          this.data._hashed_password = hashedPassword;\n          delete this.data.password;\n        });\n      });\n    })\n    .then(() => {\n      return this._validateUserName();\n    })\n    .then(() => {\n      return this._validateEmail();\n    });\n};\n\nRestWrite.prototype._validateUserName = function () {\n  // Check for username uniqueness\n  if (!this.data.username) {\n    if (!this.query) {\n      this.data.username = cryptoUtils.randomString(25);\n      this.responseShouldHaveUsername = true;\n    }\n    return Promise.resolve();\n  }\n  /*\n    Usernames should be unique when compared case insensitively\n\n    Users should be able to make case sensitive usernames and\n    login using the case they entered.  I.e. 'Snoopy' should preclude\n    'snoopy' as a valid username.\n  */\n  return this.config.database\n    .find(\n      this.className,\n      {\n        username: this.data.username,\n        objectId: { $ne: this.objectId() },\n      },\n      { limit: 1, caseInsensitive: true },\n      {},\n      this.validSchemaController\n    )\n    .then(results => {\n      if (results.length > 0) {\n        throw new Parse.Error(\n          Parse.Error.USERNAME_TAKEN,\n          'Account already exists for this username.'\n        );\n      }\n      return;\n    });\n};\n\n/*\n  As with usernames, Parse should not allow case insensitive collisions of email.\n  unlike with usernames (which can have case insensitive collisions in the case of\n  auth adapters), emails should never have a case insensitive collision.\n\n  This behavior can be enforced through a properly configured index see:\n  https://docs.mongodb.com/manual/core/index-case-insensitive/#create-a-case-insensitive-index\n  which could be implemented instead of this code based validation.\n\n  Given that this lookup should be a relatively low use case and that the case sensitive\n  unique index will be used by the db for the query, this is an adequate solution.\n*/\nRestWrite.prototype._validateEmail = function () {\n  if (!this.data.email || this.data.email.__op === 'Delete') {\n    return Promise.resolve();\n  }\n  // Validate basic email address format\n  if (!this.data.email.match(/^.+@.+$/)) {\n    return Promise.reject(\n      new Parse.Error(Parse.Error.INVALID_EMAIL_ADDRESS, 'Email address format is invalid.')\n    );\n  }\n  // Case insensitive match, see note above function.\n  return this.config.database\n    .find(\n      this.className,\n      {\n        email: this.data.email,\n        objectId: { $ne: this.objectId() },\n      },\n      { limit: 1, caseInsensitive: true },\n      {},\n      this.validSchemaController\n    )\n    .then(results => {\n      if (results.length > 0) {\n        throw new Parse.Error(\n          Parse.Error.EMAIL_TAKEN,\n          'Account already exists for this email address.'\n        );\n      }\n      if (\n        !this.data.authData ||\n        !Object.keys(this.data.authData).length ||\n        (Object.keys(this.data.authData).length === 1 &&\n          Object.keys(this.data.authData)[0] === 'anonymous')\n      ) {\n        // We updated the email, send a new validation\n        this.storage['sendVerificationEmail'] = true;\n        this.config.userController.setEmailVerifyToken(this.data);\n      }\n    });\n};\n\nRestWrite.prototype._validatePasswordPolicy = function () {\n  if (!this.config.passwordPolicy) return Promise.resolve();\n  return this._validatePasswordRequirements().then(() => {\n    return this._validatePasswordHistory();\n  });\n};\n\nRestWrite.prototype._validatePasswordRequirements = function () {\n  // check if the password conforms to the defined password policy if configured\n  // If we specified a custom error in our configuration use it.\n  // Example: \"Passwords must include a Capital Letter, Lowercase Letter, and a number.\"\n  //\n  // This is especially useful on the generic \"password reset\" page,\n  // as it allows the programmer to communicate specific requirements instead of:\n  // a. making the user guess whats wrong\n  // b. making a custom password reset page that shows the requirements\n  const policyError = this.config.passwordPolicy.validationError\n    ? this.config.passwordPolicy.validationError\n    : 'Password does not meet the Password Policy requirements.';\n  const containsUsernameError = 'Password cannot contain your username.';\n\n  // check whether the password meets the password strength requirements\n  if (\n    (this.config.passwordPolicy.patternValidator &&\n      !this.config.passwordPolicy.patternValidator(this.data.password)) ||\n    (this.config.passwordPolicy.validatorCallback &&\n      !this.config.passwordPolicy.validatorCallback(this.data.password))\n  ) {\n    return Promise.reject(new Parse.Error(Parse.Error.VALIDATION_ERROR, policyError));\n  }\n\n  // check whether password contain username\n  if (this.config.passwordPolicy.doNotAllowUsername === true) {\n    if (this.data.username) {\n      // username is not passed during password reset\n      if (this.data.password.indexOf(this.data.username) >= 0)\n        return Promise.reject(new Parse.Error(Parse.Error.VALIDATION_ERROR, containsUsernameError));\n    } else {\n      // retrieve the User object using objectId during password reset\n      return this.config.database.find('_User', { objectId: this.objectId() }).then(results => {\n        if (results.length != 1) {\n          throw undefined;\n        }\n        if (this.data.password.indexOf(results[0].username) >= 0)\n          return Promise.reject(\n            new Parse.Error(Parse.Error.VALIDATION_ERROR, containsUsernameError)\n          );\n        return Promise.resolve();\n      });\n    }\n  }\n  return Promise.resolve();\n};\n\nRestWrite.prototype._validatePasswordHistory = function () {\n  // check whether password is repeating from specified history\n  if (this.query && this.config.passwordPolicy.maxPasswordHistory) {\n    return this.config.database\n      .find(\n        '_User',\n        { objectId: this.objectId() },\n        { keys: ['_password_history', '_hashed_password'] }\n      )\n      .then(results => {\n        if (results.length != 1) {\n          throw undefined;\n        }\n        const user = results[0];\n        let oldPasswords = [];\n        if (user._password_history)\n          oldPasswords = _.take(\n            user._password_history,\n            this.config.passwordPolicy.maxPasswordHistory - 1\n          );\n        oldPasswords.push(user.password);\n        const newPassword = this.data.password;\n        // compare the new password hash with all old password hashes\n        const promises = oldPasswords.map(function (hash) {\n          return passwordCrypto.compare(newPassword, hash).then(result => {\n            if (result)\n              // reject if there is a match\n              return Promise.reject('REPEAT_PASSWORD');\n            return Promise.resolve();\n          });\n        });\n        // wait for all comparisons to complete\n        return Promise.all(promises)\n          .then(() => {\n            return Promise.resolve();\n          })\n          .catch(err => {\n            if (err === 'REPEAT_PASSWORD')\n              // a match was found\n              return Promise.reject(\n                new Parse.Error(\n                  Parse.Error.VALIDATION_ERROR,\n                  `New password should not be the same as last ${this.config.passwordPolicy.maxPasswordHistory} passwords.`\n                )\n              );\n            throw err;\n          });\n      });\n  }\n  return Promise.resolve();\n};\n\nRestWrite.prototype.createSessionTokenIfNeeded = function () {\n  if (this.className !== '_User') {\n    return;\n  }\n  // Don't generate session for updating user (this.query is set) unless authData exists\n  if (this.query && !this.data.authData) {\n    return;\n  }\n  // Don't generate new sessionToken if linking via sessionToken\n  if (this.auth.user && this.data.authData) {\n    return;\n  }\n  if (\n    !this.storage.authProvider && // signup call, with\n    this.config.preventLoginWithUnverifiedEmail && // no login without verification\n    this.config.verifyUserEmails\n  ) {\n    // verification is on\n    return; // do not create the session token in that case!\n  }\n  return this.createSessionToken();\n};\n\nRestWrite.prototype.createSessionToken = async function () {\n  // cloud installationId from Cloud Code,\n  // never create session tokens from there.\n  if (this.auth.installationId && this.auth.installationId === 'cloud') {\n    return;\n  }\n\n  if (this.storage.authProvider == null && this.data.authData) {\n    this.storage.authProvider = Object.keys(this.data.authData).join(',');\n  }\n\n  const { sessionData, createSession } = RestWrite.createSession(this.config, {\n    userId: this.objectId(),\n    createdWith: {\n      action: this.storage.authProvider ? 'login' : 'signup',\n      authProvider: this.storage.authProvider || 'password',\n    },\n    installationId: this.auth.installationId,\n  });\n\n  if (this.response && this.response.response) {\n    this.response.response.sessionToken = sessionData.sessionToken;\n  }\n\n  return createSession();\n};\n\nRestWrite.createSession = function (\n  config,\n  { userId, createdWith, installationId, additionalSessionData }\n) {\n  const token = 'r:' + cryptoUtils.newToken();\n  const expiresAt = config.generateSessionExpiresAt();\n  const sessionData = {\n    sessionToken: token,\n    user: {\n      __type: 'Pointer',\n      className: '_User',\n      objectId: userId,\n    },\n    createdWith,\n    expiresAt: Parse._encode(expiresAt),\n  };\n\n  if (installationId) {\n    sessionData.installationId = installationId;\n  }\n\n  Object.assign(sessionData, additionalSessionData);\n\n  return {\n    sessionData,\n    createSession: () =>\n      new RestWrite(config, Auth.master(config), '_Session', null, sessionData).execute(),\n  };\n};\n\n// Delete email reset tokens if user is changing password or email.\nRestWrite.prototype.deleteEmailResetTokenIfNeeded = function () {\n  if (this.className !== '_User' || this.query === null) {\n    // null query means create\n    return;\n  }\n\n  if ('password' in this.data || 'email' in this.data) {\n    const addOps = {\n      _perishable_token: { __op: 'Delete' },\n      _perishable_token_expires_at: { __op: 'Delete' },\n    };\n    this.data = Object.assign(this.data, addOps);\n  }\n};\n\nRestWrite.prototype.destroyDuplicatedSessions = function () {\n  // Only for _Session, and at creation time\n  if (this.className != '_Session' || this.query) {\n    return;\n  }\n  // Destroy the sessions in 'Background'\n  const { user, installationId, sessionToken } = this.data;\n  if (!user || !installationId) {\n    return;\n  }\n  if (!user.objectId) {\n    return;\n  }\n  this.config.database.destroy(\n    '_Session',\n    {\n      user,\n      installationId,\n      sessionToken: { $ne: sessionToken },\n    },\n    {},\n    this.validSchemaController\n  );\n};\n\n// Handles any followup logic\nRestWrite.prototype.handleFollowup = function () {\n  if (this.storage && this.storage['clearSessions'] && this.config.revokeSessionOnPasswordReset) {\n    var sessionQuery = {\n      user: {\n        __type: 'Pointer',\n        className: '_User',\n        objectId: this.objectId(),\n      },\n    };\n    delete this.storage['clearSessions'];\n    return this.config.database\n      .destroy('_Session', sessionQuery)\n      .then(this.handleFollowup.bind(this));\n  }\n\n  if (this.storage && this.storage['generateNewSession']) {\n    delete this.storage['generateNewSession'];\n    return this.createSessionToken().then(this.handleFollowup.bind(this));\n  }\n\n  if (this.storage && this.storage['sendVerificationEmail']) {\n    delete this.storage['sendVerificationEmail'];\n    // Fire and forget!\n    this.config.userController.sendVerificationEmail(this.data);\n    return this.handleFollowup.bind(this);\n  }\n};\n\n// Handles the _Session class specialness.\n// Does nothing if this isn't an _Session object.\nRestWrite.prototype.handleSession = function () {\n  if (this.response || this.className !== '_Session') {\n    return;\n  }\n\n  if (!this.auth.user && !this.auth.isMaster) {\n    throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'Session token required.');\n  }\n\n  // TODO: Verify proper error to throw\n  if (this.data.ACL) {\n    throw new Parse.Error(Parse.Error.INVALID_KEY_NAME, 'Cannot set ' + 'ACL on a Session.');\n  }\n\n  if (this.query) {\n    if (this.data.user && !this.auth.isMaster && this.data.user.objectId != this.auth.user.id) {\n      throw new Parse.Error(Parse.Error.INVALID_KEY_NAME);\n    } else if (this.data.installationId) {\n      throw new Parse.Error(Parse.Error.INVALID_KEY_NAME);\n    } else if (this.data.sessionToken) {\n      throw new Parse.Error(Parse.Error.INVALID_KEY_NAME);\n    }\n    if (!this.auth.isMaster) {\n      this.query = {\n        $and: [\n          this.query,\n          {\n            user: {\n              __type: 'Pointer',\n              className: '_User',\n              objectId: this.auth.user.id,\n            },\n          },\n        ],\n      };\n    }\n  }\n\n  if (!this.query && !this.auth.isMaster) {\n    const additionalSessionData = {};\n    for (var key in this.data) {\n      if (key === 'objectId' || key === 'user') {\n        continue;\n      }\n      additionalSessionData[key] = this.data[key];\n    }\n\n    const { sessionData, createSession } = RestWrite.createSession(this.config, {\n      userId: this.auth.user.id,\n      createdWith: {\n        action: 'create',\n      },\n      additionalSessionData,\n    });\n\n    return createSession().then(results => {\n      if (!results.response) {\n        throw new Parse.Error(Parse.Error.INTERNAL_SERVER_ERROR, 'Error creating session.');\n      }\n      sessionData['objectId'] = results.response['objectId'];\n      this.response = {\n        status: 201,\n        location: results.location,\n        response: sessionData,\n      };\n    });\n  }\n};\n\n// Handles the _Installation class specialness.\n// Does nothing if this isn't an installation object.\n// If an installation is found, this can mutate this.query and turn a create\n// into an update.\n// Returns a promise for when we're done if it can't finish this tick.\nRestWrite.prototype.handleInstallation = function () {\n  if (this.response || this.className !== '_Installation') {\n    return;\n  }\n\n  if (\n    !this.query &&\n    !this.data.deviceToken &&\n    !this.data.installationId &&\n    !this.auth.installationId\n  ) {\n    throw new Parse.Error(\n      135,\n      'at least one ID field (deviceToken, installationId) ' + 'must be specified in this operation'\n    );\n  }\n\n  // If the device token is 64 characters long, we assume it is for iOS\n  // and lowercase it.\n  if (this.data.deviceToken && this.data.deviceToken.length == 64) {\n    this.data.deviceToken = this.data.deviceToken.toLowerCase();\n  }\n\n  // We lowercase the installationId if present\n  if (this.data.installationId) {\n    this.data.installationId = this.data.installationId.toLowerCase();\n  }\n\n  let installationId = this.data.installationId;\n\n  // If data.installationId is not set and we're not master, we can lookup in auth\n  if (!installationId && !this.auth.isMaster) {\n    installationId = this.auth.installationId;\n  }\n\n  if (installationId) {\n    installationId = installationId.toLowerCase();\n  }\n\n  // Updating _Installation but not updating anything critical\n  if (this.query && !this.data.deviceToken && !installationId && !this.data.deviceType) {\n    return;\n  }\n\n  var promise = Promise.resolve();\n\n  var idMatch; // Will be a match on either objectId or installationId\n  var objectIdMatch;\n  var installationIdMatch;\n  var deviceTokenMatches = [];\n\n  // Instead of issuing 3 reads, let's do it with one OR.\n  const orQueries = [];\n  if (this.query && this.query.objectId) {\n    orQueries.push({\n      objectId: this.query.objectId,\n    });\n  }\n  if (installationId) {\n    orQueries.push({\n      installationId: installationId,\n    });\n  }\n  if (this.data.deviceToken) {\n    orQueries.push({ deviceToken: this.data.deviceToken });\n  }\n\n  if (orQueries.length == 0) {\n    return;\n  }\n\n  promise = promise\n    .then(() => {\n      return this.config.database.find(\n        '_Installation',\n        {\n          $or: orQueries,\n        },\n        {}\n      );\n    })\n    .then(results => {\n      results.forEach(result => {\n        if (this.query && this.query.objectId && result.objectId == this.query.objectId) {\n          objectIdMatch = result;\n        }\n        if (result.installationId == installationId) {\n          installationIdMatch = result;\n        }\n        if (result.deviceToken == this.data.deviceToken) {\n          deviceTokenMatches.push(result);\n        }\n      });\n\n      // Sanity checks when running a query\n      if (this.query && this.query.objectId) {\n        if (!objectIdMatch) {\n          throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Object not found for update.');\n        }\n        if (\n          this.data.installationId &&\n          objectIdMatch.installationId &&\n          this.data.installationId !== objectIdMatch.installationId\n        ) {\n          throw new Parse.Error(136, 'installationId may not be changed in this ' + 'operation');\n        }\n        if (\n          this.data.deviceToken &&\n          objectIdMatch.deviceToken &&\n          this.data.deviceToken !== objectIdMatch.deviceToken &&\n          !this.data.installationId &&\n          !objectIdMatch.installationId\n        ) {\n          throw new Parse.Error(136, 'deviceToken may not be changed in this ' + 'operation');\n        }\n        if (\n          this.data.deviceType &&\n          this.data.deviceType &&\n          this.data.deviceType !== objectIdMatch.deviceType\n        ) {\n          throw new Parse.Error(136, 'deviceType may not be changed in this ' + 'operation');\n        }\n      }\n\n      if (this.query && this.query.objectId && objectIdMatch) {\n        idMatch = objectIdMatch;\n      }\n\n      if (installationId && installationIdMatch) {\n        idMatch = installationIdMatch;\n      }\n      // need to specify deviceType only if it's new\n      if (!this.query && !this.data.deviceType && !idMatch) {\n        throw new Parse.Error(135, 'deviceType must be specified in this operation');\n      }\n    })\n    .then(() => {\n      if (!idMatch) {\n        if (!deviceTokenMatches.length) {\n          return;\n        } else if (\n          deviceTokenMatches.length == 1 &&\n          (!deviceTokenMatches[0]['installationId'] || !installationId)\n        ) {\n          // Single match on device token but none on installationId, and either\n          // the passed object or the match is missing an installationId, so we\n          // can just return the match.\n          return deviceTokenMatches[0]['objectId'];\n        } else if (!this.data.installationId) {\n          throw new Parse.Error(\n            132,\n            'Must specify installationId when deviceToken ' +\n              'matches multiple Installation objects'\n          );\n        } else {\n          // Multiple device token matches and we specified an installation ID,\n          // or a single match where both the passed and matching objects have\n          // an installation ID. Try cleaning out old installations that match\n          // the deviceToken, and return nil to signal that a new object should\n          // be created.\n          var delQuery = {\n            deviceToken: this.data.deviceToken,\n            installationId: {\n              $ne: installationId,\n            },\n          };\n          if (this.data.appIdentifier) {\n            delQuery['appIdentifier'] = this.data.appIdentifier;\n          }\n          this.config.database.destroy('_Installation', delQuery).catch(err => {\n            if (err.code == Parse.Error.OBJECT_NOT_FOUND) {\n              // no deletions were made. Can be ignored.\n              return;\n            }\n            // rethrow the error\n            throw err;\n          });\n          return;\n        }\n      } else {\n        if (deviceTokenMatches.length == 1 && !deviceTokenMatches[0]['installationId']) {\n          // Exactly one device token match and it doesn't have an installation\n          // ID. This is the one case where we want to merge with the existing\n          // object.\n          const delQuery = { objectId: idMatch.objectId };\n          return this.config.database\n            .destroy('_Installation', delQuery)\n            .then(() => {\n              return deviceTokenMatches[0]['objectId'];\n            })\n            .catch(err => {\n              if (err.code == Parse.Error.OBJECT_NOT_FOUND) {\n                // no deletions were made. Can be ignored\n                return;\n              }\n              // rethrow the error\n              throw err;\n            });\n        } else {\n          if (this.data.deviceToken && idMatch.deviceToken != this.data.deviceToken) {\n            // We're setting the device token on an existing installation, so\n            // we should try cleaning out old installations that match this\n            // device token.\n            const delQuery = {\n              deviceToken: this.data.deviceToken,\n            };\n            // We have a unique install Id, use that to preserve\n            // the interesting installation\n            if (this.data.installationId) {\n              delQuery['installationId'] = {\n                $ne: this.data.installationId,\n              };\n            } else if (\n              idMatch.objectId &&\n              this.data.objectId &&\n              idMatch.objectId == this.data.objectId\n            ) {\n              // we passed an objectId, preserve that instalation\n              delQuery['objectId'] = {\n                $ne: idMatch.objectId,\n              };\n            } else {\n              // What to do here? can't really clean up everything...\n              return idMatch.objectId;\n            }\n            if (this.data.appIdentifier) {\n              delQuery['appIdentifier'] = this.data.appIdentifier;\n            }\n            this.config.database.destroy('_Installation', delQuery).catch(err => {\n              if (err.code == Parse.Error.OBJECT_NOT_FOUND) {\n                // no deletions were made. Can be ignored.\n                return;\n              }\n              // rethrow the error\n              throw err;\n            });\n          }\n          // In non-merge scenarios, just return the installation match id\n          return idMatch.objectId;\n        }\n      }\n    })\n    .then(objId => {\n      if (objId) {\n        this.query = { objectId: objId };\n        delete this.data.objectId;\n        delete this.data.createdAt;\n      }\n      // TODO: Validate ops (add/remove on channels, $inc on badge, etc.)\n    });\n  return promise;\n};\n\n// If we short-circuited the object response - then we need to make sure we expand all the files,\n// since this might not have a query, meaning it won't return the full result back.\n// TODO: (nlutsenko) This should die when we move to per-class based controllers on _Session/_User\nRestWrite.prototype.expandFilesForExistingObjects = function () {\n  // Check whether we have a short-circuited response - only then run expansion.\n  if (this.response && this.response.response) {\n    this.config.filesController.expandFilesInObject(this.config, this.response.response);\n  }\n};\n\nRestWrite.prototype.runDatabaseOperation = function () {\n  if (this.response) {\n    return;\n  }\n\n  if (this.className === '_Role') {\n    this.config.cacheController.role.clear();\n    if (this.config.liveQueryController) {\n      this.config.liveQueryController.clearCachedRoles(this.auth.user);\n    }\n  }\n\n  if (this.className === '_User' && this.query && this.auth.isUnauthenticated()) {\n    throw new Parse.Error(\n      Parse.Error.SESSION_MISSING,\n      `Cannot modify user ${this.query.objectId}.`\n    );\n  }\n\n  if (this.className === '_Product' && this.data.download) {\n    this.data.downloadName = this.data.download.name;\n  }\n\n  // TODO: Add better detection for ACL, ensuring a user can't be locked from\n  //       their own user record.\n  if (this.data.ACL && this.data.ACL['*unresolved']) {\n    throw new Parse.Error(Parse.Error.INVALID_ACL, 'Invalid ACL.');\n  }\n\n  if (this.query) {\n    // Force the user to not lockout\n    // Matched with parse.com\n    if (this.className === '_User' && this.data.ACL && this.auth.isMaster !== true) {\n      this.data.ACL[this.query.objectId] = { read: true, write: true };\n    }\n    // update password timestamp if user password is being changed\n    if (\n      this.className === '_User' &&\n      this.data._hashed_password &&\n      this.config.passwordPolicy &&\n      this.config.passwordPolicy.maxPasswordAge\n    ) {\n      this.data._password_changed_at = Parse._encode(new Date());\n    }\n    // Ignore createdAt when update\n    delete this.data.createdAt;\n\n    let defer = Promise.resolve();\n    // if password history is enabled then save the current password to history\n    if (\n      this.className === '_User' &&\n      this.data._hashed_password &&\n      this.config.passwordPolicy &&\n      this.config.passwordPolicy.maxPasswordHistory\n    ) {\n      defer = this.config.database\n        .find(\n          '_User',\n          { objectId: this.objectId() },\n          { keys: ['_password_history', '_hashed_password'] }\n        )\n        .then(results => {\n          if (results.length != 1) {\n            throw undefined;\n          }\n          const user = results[0];\n          let oldPasswords = [];\n          if (user._password_history) {\n            oldPasswords = _.take(\n              user._password_history,\n              this.config.passwordPolicy.maxPasswordHistory\n            );\n          }\n          //n-1 passwords go into history including last password\n          while (\n            oldPasswords.length > Math.max(0, this.config.passwordPolicy.maxPasswordHistory - 2)\n          ) {\n            oldPasswords.shift();\n          }\n          oldPasswords.push(user.password);\n          this.data._password_history = oldPasswords;\n        });\n    }\n\n    return defer.then(() => {\n      // Run an update\n      return this.config.database\n        .update(\n          this.className,\n          this.query,\n          this.data,\n          this.runOptions,\n          false,\n          false,\n          this.validSchemaController\n        )\n        .then(response => {\n          response.updatedAt = this.updatedAt;\n          this._updateResponseWithData(response, this.data);\n          this.response = { response };\n        });\n    });\n  } else {\n    // Set the default ACL and password timestamp for the new _User\n    if (this.className === '_User') {\n      var ACL = this.data.ACL;\n      // default public r/w ACL\n      if (!ACL) {\n        ACL = {};\n        if (!this.config.enforcePrivateUsers) {\n          ACL['*'] = { read: true, write: false };\n        }\n      }\n      // make sure the user is not locked down\n      ACL[this.data.objectId] = { read: true, write: true };\n      this.data.ACL = ACL;\n      // password timestamp to be used when password expiry policy is enforced\n      if (this.config.passwordPolicy && this.config.passwordPolicy.maxPasswordAge) {\n        this.data._password_changed_at = Parse._encode(new Date());\n      }\n    }\n\n    // Run a create\n    return this.config.database\n      .create(this.className, this.data, this.runOptions, false, this.validSchemaController)\n      .catch(error => {\n        if (this.className !== '_User' || error.code !== Parse.Error.DUPLICATE_VALUE) {\n          throw error;\n        }\n\n        // Quick check, if we were able to infer the duplicated field name\n        if (error && error.userInfo && error.userInfo.duplicated_field === 'username') {\n          throw new Parse.Error(\n            Parse.Error.USERNAME_TAKEN,\n            'Account already exists for this username.'\n          );\n        }\n\n        if (error && error.userInfo && error.userInfo.duplicated_field === 'email') {\n          throw new Parse.Error(\n            Parse.Error.EMAIL_TAKEN,\n            'Account already exists for this email address.'\n          );\n        }\n\n        // If this was a failed user creation due to username or email already taken, we need to\n        // check whether it was username or email and return the appropriate error.\n        // Fallback to the original method\n        // TODO: See if we can later do this without additional queries by using named indexes.\n        return this.config.database\n          .find(\n            this.className,\n            {\n              username: this.data.username,\n              objectId: { $ne: this.objectId() },\n            },\n            { limit: 1 }\n          )\n          .then(results => {\n            if (results.length > 0) {\n              throw new Parse.Error(\n                Parse.Error.USERNAME_TAKEN,\n                'Account already exists for this username.'\n              );\n            }\n            return this.config.database.find(\n              this.className,\n              { email: this.data.email, objectId: { $ne: this.objectId() } },\n              { limit: 1 }\n            );\n          })\n          .then(results => {\n            if (results.length > 0) {\n              throw new Parse.Error(\n                Parse.Error.EMAIL_TAKEN,\n                'Account already exists for this email address.'\n              );\n            }\n            throw new Parse.Error(\n              Parse.Error.DUPLICATE_VALUE,\n              'A duplicate value for a field with unique values was provided'\n            );\n          });\n      })\n      .then(response => {\n        response.objectId = this.data.objectId;\n        response.createdAt = this.data.createdAt;\n\n        if (this.responseShouldHaveUsername) {\n          response.username = this.data.username;\n        }\n        this._updateResponseWithData(response, this.data);\n        this.response = {\n          status: 201,\n          response,\n          location: this.location(),\n        };\n      });\n  }\n};\n\n// Returns nothing - doesn't wait for the trigger.\nRestWrite.prototype.runAfterSaveTrigger = function () {\n  if (!this.response || !this.response.response || this.runOptions.many) {\n    return;\n  }\n\n  // Avoid doing any setup for triggers if there is no 'afterSave' trigger for this class.\n  const hasAfterSaveHook = triggers.triggerExists(\n    this.className,\n    triggers.Types.afterSave,\n    this.config.applicationId\n  );\n  const hasLiveQuery = this.config.liveQueryController.hasLiveQuery(this.className);\n  if (!hasAfterSaveHook && !hasLiveQuery) {\n    return Promise.resolve();\n  }\n\n  const { originalObject, updatedObject } = this.buildParseObjects();\n  updatedObject._handleSaveResponse(this.response.response, this.response.status || 200);\n\n  this.config.database.loadSchema().then(schemaController => {\n    // Notifiy LiveQueryServer if possible\n    const perms = schemaController.getClassLevelPermissions(updatedObject.className);\n    this.config.liveQueryController.onAfterSave(\n      updatedObject.className,\n      updatedObject,\n      originalObject,\n      perms\n    );\n  });\n\n  // Run afterSave trigger\n  return triggers\n    .maybeRunTrigger(\n      triggers.Types.afterSave,\n      this.auth,\n      updatedObject,\n      originalObject,\n      this.config,\n      this.context\n    )\n    .then(result => {\n      const jsonReturned = result && !result._toFullJSON;\n      if (jsonReturned) {\n        this.pendingOps = {};\n        this.response.response = result;\n      } else {\n        this.response.response = this._updateResponseWithData(\n          (result || updatedObject).toJSON(),\n          this.data\n        );\n      }\n    })\n    .catch(function (err) {\n      logger.warn('afterSave caught an error', err);\n    });\n};\n\n// A helper to figure out what location this operation happens at.\nRestWrite.prototype.location = function () {\n  var middle = this.className === '_User' ? '/users/' : '/classes/' + this.className + '/';\n  const mount = this.config.mount || this.config.serverURL;\n  return mount + middle + this.data.objectId;\n};\n\n// A helper to get the object id for this operation.\n// Because it could be either on the query or on the data\nRestWrite.prototype.objectId = function () {\n  return this.data.objectId || this.query.objectId;\n};\n\n// Returns a copy of the data and delete bad keys (_auth_data, _hashed_password...)\nRestWrite.prototype.sanitizedData = function () {\n  const data = Object.keys(this.data).reduce((data, key) => {\n    // Regexp comes from Parse.Object.prototype.validate\n    if (!/^[A-Za-z][0-9A-Za-z_]*$/.test(key)) {\n      delete data[key];\n    }\n    return data;\n  }, deepcopy(this.data));\n  return Parse._decode(undefined, data);\n};\n\n// Returns an updated copy of the object\nRestWrite.prototype.buildParseObjects = function () {\n  const extraData = { className: this.className, objectId: this.query?.objectId };\n  let originalObject;\n  if (this.query && this.query.objectId) {\n    originalObject = triggers.inflate(extraData, this.originalData);\n  }\n\n  const className = Parse.Object.fromJSON(extraData);\n  const readOnlyAttributes = className.constructor.readOnlyAttributes\n    ? className.constructor.readOnlyAttributes()\n    : [];\n  if (!this.originalData) {\n    for (const attribute of readOnlyAttributes) {\n      extraData[attribute] = this.data[attribute];\n    }\n  }\n  const updatedObject = triggers.inflate(extraData, this.originalData);\n  Object.keys(this.data).reduce(function (data, key) {\n    if (key.indexOf('.') > 0) {\n      if (typeof data[key].__op === 'string') {\n        if (!readOnlyAttributes.includes(key)) {\n          updatedObject.set(key, data[key]);\n        }\n      } else {\n        // subdocument key with dot notation { 'x.y': v } => { 'x': { 'y' : v } })\n        const splittedKey = key.split('.');\n        const parentProp = splittedKey[0];\n        let parentVal = updatedObject.get(parentProp);\n        if (typeof parentVal !== 'object') {\n          parentVal = {};\n        }\n        parentVal[splittedKey[1]] = data[key];\n        updatedObject.set(parentProp, parentVal);\n      }\n      delete data[key];\n    }\n    return data;\n  }, deepcopy(this.data));\n\n  const sanitized = this.sanitizedData();\n  for (const attribute of readOnlyAttributes) {\n    delete sanitized[attribute];\n  }\n  updatedObject.set(sanitized);\n  return { updatedObject, originalObject };\n};\n\nRestWrite.prototype.cleanUserAuthData = function () {\n  if (this.response && this.response.response && this.className === '_User') {\n    const user = this.response.response;\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\nRestWrite.prototype._updateResponseWithData = function (response, data) {\n  const { updatedObject } = this.buildParseObjects();\n  const stateController = Parse.CoreManager.getObjectStateController();\n  const [pending] = stateController.getPendingOps(updatedObject._getStateIdentifier());\n  for (const key in this.pendingOps) {\n    if (!pending[key]) {\n      data[key] = this.originalData ? this.originalData[key] : { __op: 'Delete' };\n      this.storage.fieldsChangedByTrigger.push(key);\n    }\n  }\n  const skipKeys = [...(requiredColumns.read[this.className] || [])];\n  if (!this.query) {\n    skipKeys.push('objectId', 'createdAt');\n  } else {\n    skipKeys.push('updatedAt');\n    delete response.objectId;\n  }\n  for (const key in response) {\n    if (skipKeys.includes(key)) {\n      continue;\n    }\n    const value = response[key];\n    if (\n      value == null ||\n      (value.__type && value.__type === 'Pointer') ||\n      util.isDeepStrictEqual(data[key], value) ||\n      util.isDeepStrictEqual((this.originalData || {})[key], value)\n    ) {\n      delete response[key];\n    }\n  }\n  if (_.isEmpty(this.storage.fieldsChangedByTrigger)) {\n    return response;\n  }\n  const clientSupportsDelete = ClientSDK.supportsForwardDelete(this.clientSDK);\n  this.storage.fieldsChangedByTrigger.forEach(fieldName => {\n    const dataValue = data[fieldName];\n\n    if (!Object.prototype.hasOwnProperty.call(response, fieldName)) {\n      response[fieldName] = dataValue;\n    }\n\n    // Strips operations from responses\n    if (response[fieldName] && response[fieldName].__op) {\n      delete response[fieldName];\n      if (clientSupportsDelete && dataValue.__op == 'Delete') {\n        response[fieldName] = dataValue;\n      }\n    }\n  });\n  return response;\n};\n\nRestWrite.prototype.checkProhibitedKeywords = function (data) {\n  if (this.config.requestKeywordDenylist) {\n    // Scan request data for denied keywords\n    for (const keyword of this.config.requestKeywordDenylist) {\n      const match = Utils.objectContainsKeyValue(data, keyword.key, keyword.value);\n      if (match) {\n        throw new Parse.Error(\n          Parse.Error.INVALID_KEY_NAME,\n          `Prohibited keyword in request data: ${JSON.stringify(keyword)}.`\n        );\n      }\n    }\n  }\n};\n\nexport default RestWrite;\nmodule.exports = RestWrite;\n"]}
1455
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["SchemaController","require","deepcopy","Auth","Utils","cryptoUtils","passwordCrypto","Parse","triggers","ClientSDK","util","RestWrite","config","auth","className","query","data","originalData","clientSDK","context","action","isReadOnly","Error","OPERATION_FORBIDDEN","storage","runOptions","allowCustomObjectId","Object","prototype","hasOwnProperty","call","objectId","MISSING_OBJECT_ID","INVALID_KEY_NAME","id","checkProhibitedKeywords","response","updatedAt","_encode","Date","iso","validSchemaController","pendingOps","operations","identifier","execute","Promise","resolve","then","getUserAndRoleACL","validateClientClassCreation","handleInstallation","handleSession","validateAuthData","runBeforeSaveTrigger","ensureUniqueAuthDataId","deleteEmailResetTokenIfNeeded","validateSchema","schemaController","setRequiredFieldsIfNeeded","transformUser","expandFilesForExistingObjects","destroyDuplicatedSessions","runDatabaseOperation","createSessionTokenIfNeeded","handleFollowup","runAfterSaveTrigger","cleanUserAuthData","authDataResponse","isMaster","acl","user","getUserRoles","roles","concat","allowClientClassCreation","systemClasses","indexOf","database","loadSchema","hasClass","validateObject","many","triggerExists","Types","beforeSave","applicationId","originalObject","updatedObject","buildParseObjects","_getStateIdentifier","stateController","CoreManager","getObjectStateController","pending","getPendingOps","databasePromise","update","create","result","length","OBJECT_NOT_FOUND","maybeRunTrigger","object","fieldsChangedByTrigger","_","reduce","value","key","isEqual","push","runBeforeLoginTrigger","userData","beforeLogin","extraData","filesController","expandFilesInObject","inflate","getAllClasses","allClasses","schema","find","oneClass","setRequiredFieldIfNeeded","fieldName","setDefault","undefined","__op","fields","defaultValue","required","VALIDATION_ERROR","createdAt","newObjectId","objectIdSize","keys","forEach","authData","hasUsernameAndPassword","username","password","isEmpty","USERNAME_MISSING","PASSWORD_MISSING","UNSUPPORTED_SERVICE","providers","canHandleAuthData","some","provider","providerAuthData","hasToken","getUserId","handleAuthData","filteredObjectsByACL","objects","filter","ACL","hasAuthDataId","r","findUsersWithAuthData","results","ACCOUNT_ALREADY_LINKED","userId","handleAuthDataValidation","validatedAuthData","userResult","authProvider","join","hasMutatedAuthData","mutatedAuthData","isCurrentUserLoggedOrMaster","isLogin","location","checkIfUserHasProvidedConfiguredProvidersForLogin","allowExpiredAuthDataToken","res","promise","error","RestQuery","master","__type","session","cacheController","del","sessionToken","_validatePasswordPolicy","hash","hashedPassword","_hashed_password","_validateUserName","_validateEmail","randomString","responseShouldHaveUsername","$ne","limit","caseInsensitive","USERNAME_TAKEN","email","match","reject","INVALID_EMAIL_ADDRESS","EMAIL_TAKEN","userController","setEmailVerifyToken","passwordPolicy","_validatePasswordRequirements","_validatePasswordHistory","policyError","validationError","containsUsernameError","patternValidator","validatorCallback","doNotAllowUsername","maxPasswordHistory","oldPasswords","_password_history","take","newPassword","promises","map","compare","all","catch","err","preventLoginWithUnverifiedEmail","verifyUserEmails","createSessionToken","installationId","sessionData","createSession","createdWith","additionalSessionData","token","newToken","expiresAt","generateSessionExpiresAt","assign","addOps","_perishable_token","_perishable_token_expires_at","destroy","revokeSessionOnPasswordReset","sessionQuery","bind","sendVerificationEmail","INVALID_SESSION_TOKEN","$and","INTERNAL_SERVER_ERROR","status","deviceToken","toLowerCase","deviceType","idMatch","objectIdMatch","installationIdMatch","deviceTokenMatches","orQueries","$or","delQuery","appIdentifier","code","objId","role","clear","liveQueryController","clearCachedRoles","isUnauthenticated","SESSION_MISSING","download","downloadName","name","INVALID_ACL","read","write","maxPasswordAge","_password_changed_at","defer","Math","max","shift","_updateResponseWithData","enforcePrivateUsers","DUPLICATE_VALUE","userInfo","duplicated_field","hasAfterSaveHook","afterSave","hasLiveQuery","_handleSaveResponse","perms","getClassLevelPermissions","onAfterSave","jsonReturned","_toFullJSON","toJSON","logger","warn","middle","mount","serverURL","sanitizedData","test","_decode","fromJSON","readOnlyAttributes","constructor","attribute","includes","set","splittedKey","split","parentProp","parentVal","get","sanitized","skipKeys","requiredColumns","isDeepStrictEqual","clientSupportsDelete","supportsForwardDelete","dataValue","requestKeywordDenylist","keyword","objectContainsKeyValue","JSON","stringify","module","exports"],"sources":["../src/RestWrite.js"],"sourcesContent":["// A RestWrite encapsulates everything we need to run an operation\n// that writes to the database.\n// This could be either a \"create\" or an \"update\".\n\nvar SchemaController = require('./Controllers/SchemaController');\nvar deepcopy = require('deepcopy');\n\nconst Auth = require('./Auth');\nconst Utils = require('./Utils');\nvar cryptoUtils = require('./cryptoUtils');\nvar passwordCrypto = require('./password');\nvar Parse = require('parse/node');\nvar triggers = require('./triggers');\nvar ClientSDK = require('./ClientSDK');\nconst util = require('util');\nimport RestQuery from './RestQuery';\nimport _ from 'lodash';\nimport logger from './logger';\nimport { requiredColumns } from './Controllers/SchemaController';\n\n// query and data are both provided in REST API format. So data\n// types are encoded by plain old objects.\n// If query is null, this is a \"create\" and the data in data should be\n// created.\n// Otherwise this is an \"update\" - the object matching the query\n// should get updated with data.\n// RestWrite will handle objectId, createdAt, and updatedAt for\n// everything. It also knows to use triggers and special modifications\n// for the _User class.\nfunction RestWrite(config, auth, className, query, data, originalData, clientSDK, context, action) {\n  if (auth.isReadOnly) {\n    throw new Parse.Error(\n      Parse.Error.OPERATION_FORBIDDEN,\n      'Cannot perform a write operation when using readOnlyMasterKey'\n    );\n  }\n  this.config = config;\n  this.auth = auth;\n  this.className = className;\n  this.clientSDK = clientSDK;\n  this.storage = {};\n  this.runOptions = {};\n  this.context = context || {};\n\n  if (action) {\n    this.runOptions.action = action;\n  }\n\n  if (!query) {\n    if (this.config.allowCustomObjectId) {\n      if (Object.prototype.hasOwnProperty.call(data, 'objectId') && !data.objectId) {\n        throw new Parse.Error(\n          Parse.Error.MISSING_OBJECT_ID,\n          'objectId must not be empty, null or undefined'\n        );\n      }\n    } else {\n      if (data.objectId) {\n        throw new Parse.Error(Parse.Error.INVALID_KEY_NAME, 'objectId is an invalid field name.');\n      }\n      if (data.id) {\n        throw new Parse.Error(Parse.Error.INVALID_KEY_NAME, 'id is an invalid field name.');\n      }\n    }\n  }\n\n  this.checkProhibitedKeywords(data);\n\n  // When the operation is complete, this.response may have several\n  // fields.\n  // response: the actual data to be returned\n  // status: the http status code. if not present, treated like a 200\n  // location: the location header. if not present, no location header\n  this.response = null;\n\n  // Processing this operation may mutate our data, so we operate on a\n  // copy\n  this.query = deepcopy(query);\n  this.data = deepcopy(data);\n  // We never change originalData, so we do not need a deep copy\n  this.originalData = originalData;\n\n  // The timestamp we'll use for this whole operation\n  this.updatedAt = Parse._encode(new Date()).iso;\n\n  // Shared SchemaController to be reused to reduce the number of loadSchema() calls per request\n  // Once set the schemaData should be immutable\n  this.validSchemaController = null;\n  this.pendingOps = {\n    operations: null,\n    identifier: null,\n  };\n}\n\n// A convenient method to perform all the steps of processing the\n// write, in order.\n// Returns a promise for a {response, status, location} object.\n// status and location are optional.\nRestWrite.prototype.execute = function () {\n  return Promise.resolve()\n    .then(() => {\n      return this.getUserAndRoleACL();\n    })\n    .then(() => {\n      return this.validateClientClassCreation();\n    })\n    .then(() => {\n      return this.handleInstallation();\n    })\n    .then(() => {\n      return this.handleSession();\n    })\n    .then(() => {\n      return this.validateAuthData();\n    })\n    .then(() => {\n      return this.runBeforeSaveTrigger();\n    })\n    .then(() => {\n      return this.ensureUniqueAuthDataId();\n    })\n    .then(() => {\n      return this.deleteEmailResetTokenIfNeeded();\n    })\n    .then(() => {\n      return this.validateSchema();\n    })\n    .then(schemaController => {\n      this.validSchemaController = schemaController;\n      return this.setRequiredFieldsIfNeeded();\n    })\n    .then(() => {\n      return this.transformUser();\n    })\n    .then(() => {\n      return this.expandFilesForExistingObjects();\n    })\n    .then(() => {\n      return this.destroyDuplicatedSessions();\n    })\n    .then(() => {\n      return this.runDatabaseOperation();\n    })\n    .then(() => {\n      return this.createSessionTokenIfNeeded();\n    })\n    .then(() => {\n      return this.handleFollowup();\n    })\n    .then(() => {\n      return this.runAfterSaveTrigger();\n    })\n    .then(() => {\n      return this.cleanUserAuthData();\n    })\n    .then(() => {\n      // Append the authDataResponse if exists\n      if (this.authDataResponse) {\n        if (this.response && this.response.response) {\n          this.response.response.authDataResponse = this.authDataResponse;\n        }\n      }\n      return this.response;\n    });\n};\n\n// Uses the Auth object to get the list of roles, adds the user id\nRestWrite.prototype.getUserAndRoleACL = function () {\n  if (this.auth.isMaster) {\n    return Promise.resolve();\n  }\n\n  this.runOptions.acl = ['*'];\n\n  if (this.auth.user) {\n    return this.auth.getUserRoles().then(roles => {\n      this.runOptions.acl = this.runOptions.acl.concat(roles, [this.auth.user.id]);\n      return;\n    });\n  } else {\n    return Promise.resolve();\n  }\n};\n\n// Validates this operation against the allowClientClassCreation config.\nRestWrite.prototype.validateClientClassCreation = function () {\n  if (\n    this.config.allowClientClassCreation === false &&\n    !this.auth.isMaster &&\n    SchemaController.systemClasses.indexOf(this.className) === -1\n  ) {\n    return this.config.database\n      .loadSchema()\n      .then(schemaController => schemaController.hasClass(this.className))\n      .then(hasClass => {\n        if (hasClass !== true) {\n          throw new Parse.Error(\n            Parse.Error.OPERATION_FORBIDDEN,\n            'This user is not allowed to access ' + 'non-existent class: ' + this.className\n          );\n        }\n      });\n  } else {\n    return Promise.resolve();\n  }\n};\n\n// Validates this operation against the schema.\nRestWrite.prototype.validateSchema = function () {\n  return this.config.database.validateObject(\n    this.className,\n    this.data,\n    this.query,\n    this.runOptions\n  );\n};\n\n// Runs any beforeSave triggers against this operation.\n// Any change leads to our data being mutated.\nRestWrite.prototype.runBeforeSaveTrigger = function () {\n  if (this.response || this.runOptions.many) {\n    return;\n  }\n\n  // Avoid doing any setup for triggers if there is no 'beforeSave' trigger for this class.\n  if (\n    !triggers.triggerExists(this.className, triggers.Types.beforeSave, this.config.applicationId)\n  ) {\n    return Promise.resolve();\n  }\n\n  const { originalObject, updatedObject } = this.buildParseObjects();\n  const identifier = updatedObject._getStateIdentifier();\n  const stateController = Parse.CoreManager.getObjectStateController();\n  const [pending] = stateController.getPendingOps(identifier);\n  this.pendingOps = {\n    operations: { ...pending },\n    identifier,\n  };\n\n  return Promise.resolve()\n    .then(() => {\n      // Before calling the trigger, validate the permissions for the save operation\n      let databasePromise = null;\n      if (this.query) {\n        // Validate for updating\n        databasePromise = this.config.database.update(\n          this.className,\n          this.query,\n          this.data,\n          this.runOptions,\n          true,\n          true\n        );\n      } else {\n        // Validate for creating\n        databasePromise = this.config.database.create(\n          this.className,\n          this.data,\n          this.runOptions,\n          true\n        );\n      }\n      // In the case that there is no permission for the operation, it throws an error\n      return databasePromise.then(result => {\n        if (!result || result.length <= 0) {\n          throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Object not found.');\n        }\n      });\n    })\n    .then(() => {\n      return triggers.maybeRunTrigger(\n        triggers.Types.beforeSave,\n        this.auth,\n        updatedObject,\n        originalObject,\n        this.config,\n        this.context\n      );\n    })\n    .then(response => {\n      if (response && response.object) {\n        this.storage.fieldsChangedByTrigger = _.reduce(\n          response.object,\n          (result, value, key) => {\n            if (!_.isEqual(this.data[key], value)) {\n              result.push(key);\n            }\n            return result;\n          },\n          []\n        );\n        this.data = response.object;\n        // We should delete the objectId for an update write\n        if (this.query && this.query.objectId) {\n          delete this.data.objectId;\n        }\n      }\n      this.checkProhibitedKeywords(this.data);\n    });\n};\n\nRestWrite.prototype.runBeforeLoginTrigger = async function (userData) {\n  // Avoid doing any setup for triggers if there is no 'beforeLogin' trigger\n  if (\n    !triggers.triggerExists(this.className, triggers.Types.beforeLogin, this.config.applicationId)\n  ) {\n    return;\n  }\n\n  // Cloud code gets a bit of extra data for its objects\n  const extraData = { className: this.className };\n\n  // Expand file objects\n  this.config.filesController.expandFilesInObject(this.config, userData);\n\n  const user = triggers.inflate(extraData, userData);\n\n  // no need to return a response\n  await triggers.maybeRunTrigger(\n    triggers.Types.beforeLogin,\n    this.auth,\n    user,\n    null,\n    this.config,\n    this.context\n  );\n};\n\nRestWrite.prototype.setRequiredFieldsIfNeeded = function () {\n  if (this.data) {\n    return this.validSchemaController.getAllClasses().then(allClasses => {\n      const schema = allClasses.find(oneClass => oneClass.className === this.className);\n      const setRequiredFieldIfNeeded = (fieldName, setDefault) => {\n        if (\n          this.data[fieldName] === undefined ||\n          this.data[fieldName] === null ||\n          this.data[fieldName] === '' ||\n          (typeof this.data[fieldName] === 'object' && this.data[fieldName].__op === 'Delete')\n        ) {\n          if (\n            setDefault &&\n            schema.fields[fieldName] &&\n            schema.fields[fieldName].defaultValue !== null &&\n            schema.fields[fieldName].defaultValue !== undefined &&\n            (this.data[fieldName] === undefined ||\n              (typeof this.data[fieldName] === 'object' && this.data[fieldName].__op === 'Delete'))\n          ) {\n            this.data[fieldName] = schema.fields[fieldName].defaultValue;\n            this.storage.fieldsChangedByTrigger = this.storage.fieldsChangedByTrigger || [];\n            if (this.storage.fieldsChangedByTrigger.indexOf(fieldName) < 0) {\n              this.storage.fieldsChangedByTrigger.push(fieldName);\n            }\n          } else if (schema.fields[fieldName] && schema.fields[fieldName].required === true) {\n            throw new Parse.Error(Parse.Error.VALIDATION_ERROR, `${fieldName} is required`);\n          }\n        }\n      };\n\n      // Add default fields\n      this.data.updatedAt = this.updatedAt;\n      if (!this.query) {\n        this.data.createdAt = this.updatedAt;\n\n        // Only assign new objectId if we are creating new object\n        if (!this.data.objectId) {\n          this.data.objectId = cryptoUtils.newObjectId(this.config.objectIdSize);\n        }\n        if (schema) {\n          Object.keys(schema.fields).forEach(fieldName => {\n            setRequiredFieldIfNeeded(fieldName, true);\n          });\n        }\n      } else if (schema) {\n        Object.keys(this.data).forEach(fieldName => {\n          setRequiredFieldIfNeeded(fieldName, false);\n        });\n      }\n    });\n  }\n  return Promise.resolve();\n};\n\n// Transforms auth data for a user object.\n// Does nothing if this isn't a user object.\n// Returns a promise for when we're done if it can't finish this tick.\nRestWrite.prototype.validateAuthData = function () {\n  if (this.className !== '_User') {\n    return;\n  }\n\n  const authData = this.data.authData;\n  const hasUsernameAndPassword =\n    typeof this.data.username === 'string' && typeof this.data.password === 'string';\n\n  if (!this.query && !authData) {\n    if (typeof this.data.username !== 'string' || _.isEmpty(this.data.username)) {\n      throw new Parse.Error(Parse.Error.USERNAME_MISSING, 'bad or missing username');\n    }\n    if (typeof this.data.password !== 'string' || _.isEmpty(this.data.password)) {\n      throw new Parse.Error(Parse.Error.PASSWORD_MISSING, 'password is required');\n    }\n  }\n\n  if (\n    (authData && !Object.keys(authData).length) ||\n    !Object.prototype.hasOwnProperty.call(this.data, 'authData')\n  ) {\n    // Nothing to validate here\n    return;\n  } else if (Object.prototype.hasOwnProperty.call(this.data, 'authData') && !this.data.authData) {\n    // Handle saving authData to null\n    throw new Parse.Error(\n      Parse.Error.UNSUPPORTED_SERVICE,\n      'This authentication method is unsupported.'\n    );\n  }\n\n  var providers = Object.keys(authData);\n  if (providers.length > 0) {\n    const canHandleAuthData = providers.some(provider => {\n      var providerAuthData = authData[provider];\n      var hasToken = providerAuthData && providerAuthData.id;\n      return hasToken || providerAuthData === null;\n    });\n    if (canHandleAuthData || hasUsernameAndPassword || this.auth.isMaster || this.getUserId()) {\n      return this.handleAuthData(authData);\n    }\n  }\n  throw new Parse.Error(\n    Parse.Error.UNSUPPORTED_SERVICE,\n    'This authentication method is unsupported.'\n  );\n};\n\nRestWrite.prototype.filteredObjectsByACL = function (objects) {\n  if (this.auth.isMaster) {\n    return objects;\n  }\n  return objects.filter(object => {\n    if (!object.ACL) {\n      return true; // legacy users that have no ACL field on them\n    }\n    // Regular users that have been locked out.\n    return object.ACL && Object.keys(object.ACL).length > 0;\n  });\n};\n\nRestWrite.prototype.getUserId = function () {\n  if (this.query && this.query.objectId && this.className === '_User') {\n    return this.query.objectId;\n  } else if (this.auth && this.auth.user && this.auth.user.id) {\n    return this.auth.user.id;\n  }\n};\n\n// Developers are allowed to change authData via before save trigger\n// we need after before save to ensure that the developer\n// is not currently duplicating auth data ID\nRestWrite.prototype.ensureUniqueAuthDataId = async function () {\n  if (this.className !== '_User' || !this.data.authData) {\n    return;\n  }\n\n  const hasAuthDataId = Object.keys(this.data.authData).some(\n    key => this.data.authData[key] && this.data.authData[key].id\n  );\n\n  if (!hasAuthDataId) return;\n\n  const r = await Auth.findUsersWithAuthData(this.config, this.data.authData);\n  const results = this.filteredObjectsByACL(r);\n  if (results.length > 1) {\n    throw new Parse.Error(Parse.Error.ACCOUNT_ALREADY_LINKED, 'this auth is already used');\n  }\n  // use data.objectId in case of login time and found user during handle validateAuthData\n  const userId = this.getUserId() || this.data.objectId;\n  if (results.length === 1 && userId !== results[0].objectId) {\n    throw new Parse.Error(Parse.Error.ACCOUNT_ALREADY_LINKED, 'this auth is already used');\n  }\n};\n\nRestWrite.prototype.handleAuthData = async function (authData) {\n  const r = await Auth.findUsersWithAuthData(this.config, authData);\n  const results = this.filteredObjectsByACL(r);\n\n  if (results.length > 1) {\n    // To avoid https://github.com/parse-community/parse-server/security/advisories/GHSA-8w3j-g983-8jh5\n    // Let's run some validation before throwing\n    await Auth.handleAuthDataValidation(authData, this, results[0]);\n    throw new Parse.Error(Parse.Error.ACCOUNT_ALREADY_LINKED, 'this auth is already used');\n  }\n\n  // No user found with provided authData we need to validate\n  if (!results.length) {\n    const { authData: validatedAuthData, authDataResponse } = await Auth.handleAuthDataValidation(\n      authData,\n      this\n    );\n    this.authDataResponse = authDataResponse;\n    // Replace current authData by the new validated one\n    this.data.authData = validatedAuthData;\n    return;\n  }\n\n  // User found with provided authData\n  if (results.length === 1) {\n    const userId = this.getUserId();\n    const userResult = results[0];\n    // Prevent duplicate authData id\n    if (userId && userId !== userResult.objectId) {\n      throw new Parse.Error(Parse.Error.ACCOUNT_ALREADY_LINKED, 'this auth is already used');\n    }\n\n    this.storage.authProvider = Object.keys(authData).join(',');\n\n    const { hasMutatedAuthData, mutatedAuthData } = Auth.hasMutatedAuthData(\n      authData,\n      userResult.authData\n    );\n\n    const isCurrentUserLoggedOrMaster =\n      (this.auth && this.auth.user && this.auth.user.id === userResult.objectId) ||\n      this.auth.isMaster;\n\n    const isLogin = !userId;\n\n    if (isLogin || isCurrentUserLoggedOrMaster) {\n      // no user making the call\n      // OR the user making the call is the right one\n      // Login with auth data\n      delete results[0].password;\n\n      // need to set the objectId first otherwise location has trailing undefined\n      this.data.objectId = userResult.objectId;\n\n      if (!this.query || !this.query.objectId) {\n        this.response = {\n          response: userResult,\n          location: this.location(),\n        };\n        // Run beforeLogin hook before storing any updates\n        // to authData on the db; changes to userResult\n        // will be ignored.\n        await this.runBeforeLoginTrigger(deepcopy(userResult));\n\n        // If we are in login operation via authData\n        // we need to be sure that the user has provided\n        // required authData\n        Auth.checkIfUserHasProvidedConfiguredProvidersForLogin(\n          authData,\n          userResult.authData,\n          this.config\n        );\n      }\n\n      // Prevent validating if no mutated data detected on update\n      if (!hasMutatedAuthData && isCurrentUserLoggedOrMaster) {\n        return;\n      }\n\n      // Force to validate all provided authData on login\n      // on update only validate mutated ones\n      if (hasMutatedAuthData || !this.config.allowExpiredAuthDataToken) {\n        const res = await Auth.handleAuthDataValidation(\n          isLogin ? authData : mutatedAuthData,\n          this,\n          userResult\n        );\n        this.data.authData = res.authData;\n        this.authDataResponse = res.authDataResponse;\n      }\n\n      // IF we are in login we'll skip the database operation / beforeSave / afterSave etc...\n      // we need to set it up there.\n      // We are supposed to have a response only on LOGIN with authData, so we skip those\n      // If we're not logging in, but just updating the current user, we can safely skip that part\n      if (this.response) {\n        // Assign the new authData in the response\n        Object.keys(mutatedAuthData).forEach(provider => {\n          this.response.response.authData[provider] = mutatedAuthData[provider];\n        });\n\n        // Run the DB update directly, as 'master' only if authData contains some keys\n        // authData could not contains keys after validation if the authAdapter\n        // uses the `doNotSave` option. Just update the authData part\n        // Then we're good for the user, early exit of sorts\n        if (Object.keys(this.data.authData).length) {\n          await this.config.database.update(\n            this.className,\n            { objectId: this.data.objectId },\n            { authData: this.data.authData },\n            {}\n          );\n        }\n      }\n    }\n  }\n};\n\n// The non-third-party parts of User transformation\nRestWrite.prototype.transformUser = function () {\n  var promise = Promise.resolve();\n  if (this.className !== '_User') {\n    return promise;\n  }\n\n  if (!this.auth.isMaster && 'emailVerified' in this.data) {\n    const error = `Clients aren't allowed to manually update email verification.`;\n    throw new Parse.Error(Parse.Error.OPERATION_FORBIDDEN, error);\n  }\n\n  // Do not cleanup session if objectId is not set\n  if (this.query && this.objectId()) {\n    // If we're updating a _User object, we need to clear out the cache for that user. Find all their\n    // session tokens, and remove them from the cache.\n    promise = new RestQuery(this.config, Auth.master(this.config), '_Session', {\n      user: {\n        __type: 'Pointer',\n        className: '_User',\n        objectId: this.objectId(),\n      },\n    })\n      .execute()\n      .then(results => {\n        results.results.forEach(session =>\n          this.config.cacheController.user.del(session.sessionToken)\n        );\n      });\n  }\n\n  return promise\n    .then(() => {\n      // Transform the password\n      if (this.data.password === undefined) {\n        // ignore only if undefined. should proceed if empty ('')\n        return Promise.resolve();\n      }\n\n      if (this.query) {\n        this.storage['clearSessions'] = true;\n        // Generate a new session only if the user requested\n        if (!this.auth.isMaster) {\n          this.storage['generateNewSession'] = true;\n        }\n      }\n\n      return this._validatePasswordPolicy().then(() => {\n        return passwordCrypto.hash(this.data.password).then(hashedPassword => {\n          this.data._hashed_password = hashedPassword;\n          delete this.data.password;\n        });\n      });\n    })\n    .then(() => {\n      return this._validateUserName();\n    })\n    .then(() => {\n      return this._validateEmail();\n    });\n};\n\nRestWrite.prototype._validateUserName = function () {\n  // Check for username uniqueness\n  if (!this.data.username) {\n    if (!this.query) {\n      this.data.username = cryptoUtils.randomString(25);\n      this.responseShouldHaveUsername = true;\n    }\n    return Promise.resolve();\n  }\n  /*\n    Usernames should be unique when compared case insensitively\n\n    Users should be able to make case sensitive usernames and\n    login using the case they entered.  I.e. 'Snoopy' should preclude\n    'snoopy' as a valid username.\n  */\n  return this.config.database\n    .find(\n      this.className,\n      {\n        username: this.data.username,\n        objectId: { $ne: this.objectId() },\n      },\n      { limit: 1, caseInsensitive: true },\n      {},\n      this.validSchemaController\n    )\n    .then(results => {\n      if (results.length > 0) {\n        throw new Parse.Error(\n          Parse.Error.USERNAME_TAKEN,\n          'Account already exists for this username.'\n        );\n      }\n      return;\n    });\n};\n\n/*\n  As with usernames, Parse should not allow case insensitive collisions of email.\n  unlike with usernames (which can have case insensitive collisions in the case of\n  auth adapters), emails should never have a case insensitive collision.\n\n  This behavior can be enforced through a properly configured index see:\n  https://docs.mongodb.com/manual/core/index-case-insensitive/#create-a-case-insensitive-index\n  which could be implemented instead of this code based validation.\n\n  Given that this lookup should be a relatively low use case and that the case sensitive\n  unique index will be used by the db for the query, this is an adequate solution.\n*/\nRestWrite.prototype._validateEmail = function () {\n  if (!this.data.email || this.data.email.__op === 'Delete') {\n    return Promise.resolve();\n  }\n  // Validate basic email address format\n  if (!this.data.email.match(/^.+@.+$/)) {\n    return Promise.reject(\n      new Parse.Error(Parse.Error.INVALID_EMAIL_ADDRESS, 'Email address format is invalid.')\n    );\n  }\n  // Case insensitive match, see note above function.\n  return this.config.database\n    .find(\n      this.className,\n      {\n        email: this.data.email,\n        objectId: { $ne: this.objectId() },\n      },\n      { limit: 1, caseInsensitive: true },\n      {},\n      this.validSchemaController\n    )\n    .then(results => {\n      if (results.length > 0) {\n        throw new Parse.Error(\n          Parse.Error.EMAIL_TAKEN,\n          'Account already exists for this email address.'\n        );\n      }\n      if (\n        !this.data.authData ||\n        !Object.keys(this.data.authData).length ||\n        (Object.keys(this.data.authData).length === 1 &&\n          Object.keys(this.data.authData)[0] === 'anonymous')\n      ) {\n        // We updated the email, send a new validation\n        this.storage['sendVerificationEmail'] = true;\n        this.config.userController.setEmailVerifyToken(this.data);\n      }\n    });\n};\n\nRestWrite.prototype._validatePasswordPolicy = function () {\n  if (!this.config.passwordPolicy) return Promise.resolve();\n  return this._validatePasswordRequirements().then(() => {\n    return this._validatePasswordHistory();\n  });\n};\n\nRestWrite.prototype._validatePasswordRequirements = function () {\n  // check if the password conforms to the defined password policy if configured\n  // If we specified a custom error in our configuration use it.\n  // Example: \"Passwords must include a Capital Letter, Lowercase Letter, and a number.\"\n  //\n  // This is especially useful on the generic \"password reset\" page,\n  // as it allows the programmer to communicate specific requirements instead of:\n  // a. making the user guess whats wrong\n  // b. making a custom password reset page that shows the requirements\n  const policyError = this.config.passwordPolicy.validationError\n    ? this.config.passwordPolicy.validationError\n    : 'Password does not meet the Password Policy requirements.';\n  const containsUsernameError = 'Password cannot contain your username.';\n\n  // check whether the password meets the password strength requirements\n  if (\n    (this.config.passwordPolicy.patternValidator &&\n      !this.config.passwordPolicy.patternValidator(this.data.password)) ||\n    (this.config.passwordPolicy.validatorCallback &&\n      !this.config.passwordPolicy.validatorCallback(this.data.password))\n  ) {\n    return Promise.reject(new Parse.Error(Parse.Error.VALIDATION_ERROR, policyError));\n  }\n\n  // check whether password contain username\n  if (this.config.passwordPolicy.doNotAllowUsername === true) {\n    if (this.data.username) {\n      // username is not passed during password reset\n      if (this.data.password.indexOf(this.data.username) >= 0)\n        return Promise.reject(new Parse.Error(Parse.Error.VALIDATION_ERROR, containsUsernameError));\n    } else {\n      // retrieve the User object using objectId during password reset\n      return this.config.database.find('_User', { objectId: this.objectId() }).then(results => {\n        if (results.length != 1) {\n          throw undefined;\n        }\n        if (this.data.password.indexOf(results[0].username) >= 0)\n          return Promise.reject(\n            new Parse.Error(Parse.Error.VALIDATION_ERROR, containsUsernameError)\n          );\n        return Promise.resolve();\n      });\n    }\n  }\n  return Promise.resolve();\n};\n\nRestWrite.prototype._validatePasswordHistory = function () {\n  // check whether password is repeating from specified history\n  if (this.query && this.config.passwordPolicy.maxPasswordHistory) {\n    return this.config.database\n      .find(\n        '_User',\n        { objectId: this.objectId() },\n        { keys: ['_password_history', '_hashed_password'] }\n      )\n      .then(results => {\n        if (results.length != 1) {\n          throw undefined;\n        }\n        const user = results[0];\n        let oldPasswords = [];\n        if (user._password_history)\n          oldPasswords = _.take(\n            user._password_history,\n            this.config.passwordPolicy.maxPasswordHistory - 1\n          );\n        oldPasswords.push(user.password);\n        const newPassword = this.data.password;\n        // compare the new password hash with all old password hashes\n        const promises = oldPasswords.map(function (hash) {\n          return passwordCrypto.compare(newPassword, hash).then(result => {\n            if (result)\n              // reject if there is a match\n              return Promise.reject('REPEAT_PASSWORD');\n            return Promise.resolve();\n          });\n        });\n        // wait for all comparisons to complete\n        return Promise.all(promises)\n          .then(() => {\n            return Promise.resolve();\n          })\n          .catch(err => {\n            if (err === 'REPEAT_PASSWORD')\n              // a match was found\n              return Promise.reject(\n                new Parse.Error(\n                  Parse.Error.VALIDATION_ERROR,\n                  `New password should not be the same as last ${this.config.passwordPolicy.maxPasswordHistory} passwords.`\n                )\n              );\n            throw err;\n          });\n      });\n  }\n  return Promise.resolve();\n};\n\nRestWrite.prototype.createSessionTokenIfNeeded = function () {\n  if (this.className !== '_User') {\n    return;\n  }\n  // Don't generate session for updating user (this.query is set) unless authData exists\n  if (this.query && !this.data.authData) {\n    return;\n  }\n  // Don't generate new sessionToken if linking via sessionToken\n  if (this.auth.user && this.data.authData) {\n    return;\n  }\n  if (\n    !this.storage.authProvider && // signup call, with\n    this.config.preventLoginWithUnverifiedEmail && // no login without verification\n    this.config.verifyUserEmails\n  ) {\n    // verification is on\n    return; // do not create the session token in that case!\n  }\n  return this.createSessionToken();\n};\n\nRestWrite.prototype.createSessionToken = async function () {\n  // cloud installationId from Cloud Code,\n  // never create session tokens from there.\n  if (this.auth.installationId && this.auth.installationId === 'cloud') {\n    return;\n  }\n\n  if (this.storage.authProvider == null && this.data.authData) {\n    this.storage.authProvider = Object.keys(this.data.authData).join(',');\n  }\n\n  const { sessionData, createSession } = RestWrite.createSession(this.config, {\n    userId: this.objectId(),\n    createdWith: {\n      action: this.storage.authProvider ? 'login' : 'signup',\n      authProvider: this.storage.authProvider || 'password',\n    },\n    installationId: this.auth.installationId,\n  });\n\n  if (this.response && this.response.response) {\n    this.response.response.sessionToken = sessionData.sessionToken;\n  }\n\n  return createSession();\n};\n\nRestWrite.createSession = function (\n  config,\n  { userId, createdWith, installationId, additionalSessionData }\n) {\n  const token = 'r:' + cryptoUtils.newToken();\n  const expiresAt = config.generateSessionExpiresAt();\n  const sessionData = {\n    sessionToken: token,\n    user: {\n      __type: 'Pointer',\n      className: '_User',\n      objectId: userId,\n    },\n    createdWith,\n    expiresAt: Parse._encode(expiresAt),\n  };\n\n  if (installationId) {\n    sessionData.installationId = installationId;\n  }\n\n  Object.assign(sessionData, additionalSessionData);\n\n  return {\n    sessionData,\n    createSession: () =>\n      new RestWrite(config, Auth.master(config), '_Session', null, sessionData).execute(),\n  };\n};\n\n// Delete email reset tokens if user is changing password or email.\nRestWrite.prototype.deleteEmailResetTokenIfNeeded = function () {\n  if (this.className !== '_User' || this.query === null) {\n    // null query means create\n    return;\n  }\n\n  if ('password' in this.data || 'email' in this.data) {\n    const addOps = {\n      _perishable_token: { __op: 'Delete' },\n      _perishable_token_expires_at: { __op: 'Delete' },\n    };\n    this.data = Object.assign(this.data, addOps);\n  }\n};\n\nRestWrite.prototype.destroyDuplicatedSessions = function () {\n  // Only for _Session, and at creation time\n  if (this.className != '_Session' || this.query) {\n    return;\n  }\n  // Destroy the sessions in 'Background'\n  const { user, installationId, sessionToken } = this.data;\n  if (!user || !installationId) {\n    return;\n  }\n  if (!user.objectId) {\n    return;\n  }\n  this.config.database.destroy(\n    '_Session',\n    {\n      user,\n      installationId,\n      sessionToken: { $ne: sessionToken },\n    },\n    {},\n    this.validSchemaController\n  );\n};\n\n// Handles any followup logic\nRestWrite.prototype.handleFollowup = function () {\n  if (this.storage && this.storage['clearSessions'] && this.config.revokeSessionOnPasswordReset) {\n    var sessionQuery = {\n      user: {\n        __type: 'Pointer',\n        className: '_User',\n        objectId: this.objectId(),\n      },\n    };\n    delete this.storage['clearSessions'];\n    return this.config.database\n      .destroy('_Session', sessionQuery)\n      .then(this.handleFollowup.bind(this));\n  }\n\n  if (this.storage && this.storage['generateNewSession']) {\n    delete this.storage['generateNewSession'];\n    return this.createSessionToken().then(this.handleFollowup.bind(this));\n  }\n\n  if (this.storage && this.storage['sendVerificationEmail']) {\n    delete this.storage['sendVerificationEmail'];\n    // Fire and forget!\n    this.config.userController.sendVerificationEmail(this.data);\n    return this.handleFollowup.bind(this);\n  }\n};\n\n// Handles the _Session class specialness.\n// Does nothing if this isn't an _Session object.\nRestWrite.prototype.handleSession = function () {\n  if (this.response || this.className !== '_Session') {\n    return;\n  }\n\n  if (!this.auth.user && !this.auth.isMaster) {\n    throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'Session token required.');\n  }\n\n  // TODO: Verify proper error to throw\n  if (this.data.ACL) {\n    throw new Parse.Error(Parse.Error.INVALID_KEY_NAME, 'Cannot set ' + 'ACL on a Session.');\n  }\n\n  if (this.query) {\n    if (this.data.user && !this.auth.isMaster && this.data.user.objectId != this.auth.user.id) {\n      throw new Parse.Error(Parse.Error.INVALID_KEY_NAME);\n    } else if (this.data.installationId) {\n      throw new Parse.Error(Parse.Error.INVALID_KEY_NAME);\n    } else if (this.data.sessionToken) {\n      throw new Parse.Error(Parse.Error.INVALID_KEY_NAME);\n    }\n    if (!this.auth.isMaster) {\n      this.query = {\n        $and: [\n          this.query,\n          {\n            user: {\n              __type: 'Pointer',\n              className: '_User',\n              objectId: this.auth.user.id,\n            },\n          },\n        ],\n      };\n    }\n  }\n\n  if (!this.query && !this.auth.isMaster) {\n    const additionalSessionData = {};\n    for (var key in this.data) {\n      if (key === 'objectId' || key === 'user') {\n        continue;\n      }\n      additionalSessionData[key] = this.data[key];\n    }\n\n    const { sessionData, createSession } = RestWrite.createSession(this.config, {\n      userId: this.auth.user.id,\n      createdWith: {\n        action: 'create',\n      },\n      additionalSessionData,\n    });\n\n    return createSession().then(results => {\n      if (!results.response) {\n        throw new Parse.Error(Parse.Error.INTERNAL_SERVER_ERROR, 'Error creating session.');\n      }\n      sessionData['objectId'] = results.response['objectId'];\n      this.response = {\n        status: 201,\n        location: results.location,\n        response: sessionData,\n      };\n    });\n  }\n};\n\n// Handles the _Installation class specialness.\n// Does nothing if this isn't an installation object.\n// If an installation is found, this can mutate this.query and turn a create\n// into an update.\n// Returns a promise for when we're done if it can't finish this tick.\nRestWrite.prototype.handleInstallation = function () {\n  if (this.response || this.className !== '_Installation') {\n    return;\n  }\n\n  if (\n    !this.query &&\n    !this.data.deviceToken &&\n    !this.data.installationId &&\n    !this.auth.installationId\n  ) {\n    throw new Parse.Error(\n      135,\n      'at least one ID field (deviceToken, installationId) ' + 'must be specified in this operation'\n    );\n  }\n\n  // If the device token is 64 characters long, we assume it is for iOS\n  // and lowercase it.\n  if (this.data.deviceToken && this.data.deviceToken.length == 64) {\n    this.data.deviceToken = this.data.deviceToken.toLowerCase();\n  }\n\n  // We lowercase the installationId if present\n  if (this.data.installationId) {\n    this.data.installationId = this.data.installationId.toLowerCase();\n  }\n\n  let installationId = this.data.installationId;\n\n  // If data.installationId is not set and we're not master, we can lookup in auth\n  if (!installationId && !this.auth.isMaster) {\n    installationId = this.auth.installationId;\n  }\n\n  if (installationId) {\n    installationId = installationId.toLowerCase();\n  }\n\n  // Updating _Installation but not updating anything critical\n  if (this.query && !this.data.deviceToken && !installationId && !this.data.deviceType) {\n    return;\n  }\n\n  var promise = Promise.resolve();\n\n  var idMatch; // Will be a match on either objectId or installationId\n  var objectIdMatch;\n  var installationIdMatch;\n  var deviceTokenMatches = [];\n\n  // Instead of issuing 3 reads, let's do it with one OR.\n  const orQueries = [];\n  if (this.query && this.query.objectId) {\n    orQueries.push({\n      objectId: this.query.objectId,\n    });\n  }\n  if (installationId) {\n    orQueries.push({\n      installationId: installationId,\n    });\n  }\n  if (this.data.deviceToken) {\n    orQueries.push({ deviceToken: this.data.deviceToken });\n  }\n\n  if (orQueries.length == 0) {\n    return;\n  }\n\n  promise = promise\n    .then(() => {\n      return this.config.database.find(\n        '_Installation',\n        {\n          $or: orQueries,\n        },\n        {}\n      );\n    })\n    .then(results => {\n      results.forEach(result => {\n        if (this.query && this.query.objectId && result.objectId == this.query.objectId) {\n          objectIdMatch = result;\n        }\n        if (result.installationId == installationId) {\n          installationIdMatch = result;\n        }\n        if (result.deviceToken == this.data.deviceToken) {\n          deviceTokenMatches.push(result);\n        }\n      });\n\n      // Sanity checks when running a query\n      if (this.query && this.query.objectId) {\n        if (!objectIdMatch) {\n          throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Object not found for update.');\n        }\n        if (\n          this.data.installationId &&\n          objectIdMatch.installationId &&\n          this.data.installationId !== objectIdMatch.installationId\n        ) {\n          throw new Parse.Error(136, 'installationId may not be changed in this ' + 'operation');\n        }\n        if (\n          this.data.deviceToken &&\n          objectIdMatch.deviceToken &&\n          this.data.deviceToken !== objectIdMatch.deviceToken &&\n          !this.data.installationId &&\n          !objectIdMatch.installationId\n        ) {\n          throw new Parse.Error(136, 'deviceToken may not be changed in this ' + 'operation');\n        }\n        if (\n          this.data.deviceType &&\n          this.data.deviceType &&\n          this.data.deviceType !== objectIdMatch.deviceType\n        ) {\n          throw new Parse.Error(136, 'deviceType may not be changed in this ' + 'operation');\n        }\n      }\n\n      if (this.query && this.query.objectId && objectIdMatch) {\n        idMatch = objectIdMatch;\n      }\n\n      if (installationId && installationIdMatch) {\n        idMatch = installationIdMatch;\n      }\n      // need to specify deviceType only if it's new\n      if (!this.query && !this.data.deviceType && !idMatch) {\n        throw new Parse.Error(135, 'deviceType must be specified in this operation');\n      }\n    })\n    .then(() => {\n      if (!idMatch) {\n        if (!deviceTokenMatches.length) {\n          return;\n        } else if (\n          deviceTokenMatches.length == 1 &&\n          (!deviceTokenMatches[0]['installationId'] || !installationId)\n        ) {\n          // Single match on device token but none on installationId, and either\n          // the passed object or the match is missing an installationId, so we\n          // can just return the match.\n          return deviceTokenMatches[0]['objectId'];\n        } else if (!this.data.installationId) {\n          throw new Parse.Error(\n            132,\n            'Must specify installationId when deviceToken ' +\n              'matches multiple Installation objects'\n          );\n        } else {\n          // Multiple device token matches and we specified an installation ID,\n          // or a single match where both the passed and matching objects have\n          // an installation ID. Try cleaning out old installations that match\n          // the deviceToken, and return nil to signal that a new object should\n          // be created.\n          var delQuery = {\n            deviceToken: this.data.deviceToken,\n            installationId: {\n              $ne: installationId,\n            },\n          };\n          if (this.data.appIdentifier) {\n            delQuery['appIdentifier'] = this.data.appIdentifier;\n          }\n          this.config.database.destroy('_Installation', delQuery).catch(err => {\n            if (err.code == Parse.Error.OBJECT_NOT_FOUND) {\n              // no deletions were made. Can be ignored.\n              return;\n            }\n            // rethrow the error\n            throw err;\n          });\n          return;\n        }\n      } else {\n        if (deviceTokenMatches.length == 1 && !deviceTokenMatches[0]['installationId']) {\n          // Exactly one device token match and it doesn't have an installation\n          // ID. This is the one case where we want to merge with the existing\n          // object.\n          const delQuery = { objectId: idMatch.objectId };\n          return this.config.database\n            .destroy('_Installation', delQuery)\n            .then(() => {\n              return deviceTokenMatches[0]['objectId'];\n            })\n            .catch(err => {\n              if (err.code == Parse.Error.OBJECT_NOT_FOUND) {\n                // no deletions were made. Can be ignored\n                return;\n              }\n              // rethrow the error\n              throw err;\n            });\n        } else {\n          if (this.data.deviceToken && idMatch.deviceToken != this.data.deviceToken) {\n            // We're setting the device token on an existing installation, so\n            // we should try cleaning out old installations that match this\n            // device token.\n            const delQuery = {\n              deviceToken: this.data.deviceToken,\n            };\n            // We have a unique install Id, use that to preserve\n            // the interesting installation\n            if (this.data.installationId) {\n              delQuery['installationId'] = {\n                $ne: this.data.installationId,\n              };\n            } else if (\n              idMatch.objectId &&\n              this.data.objectId &&\n              idMatch.objectId == this.data.objectId\n            ) {\n              // we passed an objectId, preserve that instalation\n              delQuery['objectId'] = {\n                $ne: idMatch.objectId,\n              };\n            } else {\n              // What to do here? can't really clean up everything...\n              return idMatch.objectId;\n            }\n            if (this.data.appIdentifier) {\n              delQuery['appIdentifier'] = this.data.appIdentifier;\n            }\n            this.config.database.destroy('_Installation', delQuery).catch(err => {\n              if (err.code == Parse.Error.OBJECT_NOT_FOUND) {\n                // no deletions were made. Can be ignored.\n                return;\n              }\n              // rethrow the error\n              throw err;\n            });\n          }\n          // In non-merge scenarios, just return the installation match id\n          return idMatch.objectId;\n        }\n      }\n    })\n    .then(objId => {\n      if (objId) {\n        this.query = { objectId: objId };\n        delete this.data.objectId;\n        delete this.data.createdAt;\n      }\n      // TODO: Validate ops (add/remove on channels, $inc on badge, etc.)\n    });\n  return promise;\n};\n\n// If we short-circuited the object response - then we need to make sure we expand all the files,\n// since this might not have a query, meaning it won't return the full result back.\n// TODO: (nlutsenko) This should die when we move to per-class based controllers on _Session/_User\nRestWrite.prototype.expandFilesForExistingObjects = function () {\n  // Check whether we have a short-circuited response - only then run expansion.\n  if (this.response && this.response.response) {\n    this.config.filesController.expandFilesInObject(this.config, this.response.response);\n  }\n};\n\nRestWrite.prototype.runDatabaseOperation = function () {\n  if (this.response) {\n    return;\n  }\n\n  if (this.className === '_Role') {\n    this.config.cacheController.role.clear();\n    if (this.config.liveQueryController) {\n      this.config.liveQueryController.clearCachedRoles(this.auth.user);\n    }\n  }\n\n  if (this.className === '_User' && this.query && this.auth.isUnauthenticated()) {\n    throw new Parse.Error(\n      Parse.Error.SESSION_MISSING,\n      `Cannot modify user ${this.query.objectId}.`\n    );\n  }\n\n  if (this.className === '_Product' && this.data.download) {\n    this.data.downloadName = this.data.download.name;\n  }\n\n  // TODO: Add better detection for ACL, ensuring a user can't be locked from\n  //       their own user record.\n  if (this.data.ACL && this.data.ACL['*unresolved']) {\n    throw new Parse.Error(Parse.Error.INVALID_ACL, 'Invalid ACL.');\n  }\n\n  if (this.query) {\n    // Force the user to not lockout\n    // Matched with parse.com\n    if (this.className === '_User' && this.data.ACL && this.auth.isMaster !== true) {\n      this.data.ACL[this.query.objectId] = { read: true, write: true };\n    }\n    // update password timestamp if user password is being changed\n    if (\n      this.className === '_User' &&\n      this.data._hashed_password &&\n      this.config.passwordPolicy &&\n      this.config.passwordPolicy.maxPasswordAge\n    ) {\n      this.data._password_changed_at = Parse._encode(new Date());\n    }\n    // Ignore createdAt when update\n    delete this.data.createdAt;\n\n    let defer = Promise.resolve();\n    // if password history is enabled then save the current password to history\n    if (\n      this.className === '_User' &&\n      this.data._hashed_password &&\n      this.config.passwordPolicy &&\n      this.config.passwordPolicy.maxPasswordHistory\n    ) {\n      defer = this.config.database\n        .find(\n          '_User',\n          { objectId: this.objectId() },\n          { keys: ['_password_history', '_hashed_password'] }\n        )\n        .then(results => {\n          if (results.length != 1) {\n            throw undefined;\n          }\n          const user = results[0];\n          let oldPasswords = [];\n          if (user._password_history) {\n            oldPasswords = _.take(\n              user._password_history,\n              this.config.passwordPolicy.maxPasswordHistory\n            );\n          }\n          //n-1 passwords go into history including last password\n          while (\n            oldPasswords.length > Math.max(0, this.config.passwordPolicy.maxPasswordHistory - 2)\n          ) {\n            oldPasswords.shift();\n          }\n          oldPasswords.push(user.password);\n          this.data._password_history = oldPasswords;\n        });\n    }\n\n    return defer.then(() => {\n      // Run an update\n      return this.config.database\n        .update(\n          this.className,\n          this.query,\n          this.data,\n          this.runOptions,\n          false,\n          false,\n          this.validSchemaController\n        )\n        .then(response => {\n          response.updatedAt = this.updatedAt;\n          this._updateResponseWithData(response, this.data);\n          this.response = { response };\n        });\n    });\n  } else {\n    // Set the default ACL and password timestamp for the new _User\n    if (this.className === '_User') {\n      var ACL = this.data.ACL;\n      // default public r/w ACL\n      if (!ACL) {\n        ACL = {};\n        if (!this.config.enforcePrivateUsers) {\n          ACL['*'] = { read: true, write: false };\n        }\n      }\n      // make sure the user is not locked down\n      ACL[this.data.objectId] = { read: true, write: true };\n      this.data.ACL = ACL;\n      // password timestamp to be used when password expiry policy is enforced\n      if (this.config.passwordPolicy && this.config.passwordPolicy.maxPasswordAge) {\n        this.data._password_changed_at = Parse._encode(new Date());\n      }\n    }\n\n    // Run a create\n    return this.config.database\n      .create(this.className, this.data, this.runOptions, false, this.validSchemaController)\n      .catch(error => {\n        if (this.className !== '_User' || error.code !== Parse.Error.DUPLICATE_VALUE) {\n          throw error;\n        }\n\n        // Quick check, if we were able to infer the duplicated field name\n        if (error && error.userInfo && error.userInfo.duplicated_field === 'username') {\n          throw new Parse.Error(\n            Parse.Error.USERNAME_TAKEN,\n            'Account already exists for this username.'\n          );\n        }\n\n        if (error && error.userInfo && error.userInfo.duplicated_field === 'email') {\n          throw new Parse.Error(\n            Parse.Error.EMAIL_TAKEN,\n            'Account already exists for this email address.'\n          );\n        }\n\n        // If this was a failed user creation due to username or email already taken, we need to\n        // check whether it was username or email and return the appropriate error.\n        // Fallback to the original method\n        // TODO: See if we can later do this without additional queries by using named indexes.\n        return this.config.database\n          .find(\n            this.className,\n            {\n              username: this.data.username,\n              objectId: { $ne: this.objectId() },\n            },\n            { limit: 1 }\n          )\n          .then(results => {\n            if (results.length > 0) {\n              throw new Parse.Error(\n                Parse.Error.USERNAME_TAKEN,\n                'Account already exists for this username.'\n              );\n            }\n            return this.config.database.find(\n              this.className,\n              { email: this.data.email, objectId: { $ne: this.objectId() } },\n              { limit: 1 }\n            );\n          })\n          .then(results => {\n            if (results.length > 0) {\n              throw new Parse.Error(\n                Parse.Error.EMAIL_TAKEN,\n                'Account already exists for this email address.'\n              );\n            }\n            throw new Parse.Error(\n              Parse.Error.DUPLICATE_VALUE,\n              'A duplicate value for a field with unique values was provided'\n            );\n          });\n      })\n      .then(response => {\n        response.objectId = this.data.objectId;\n        response.createdAt = this.data.createdAt;\n\n        if (this.responseShouldHaveUsername) {\n          response.username = this.data.username;\n        }\n        this._updateResponseWithData(response, this.data);\n        this.response = {\n          status: 201,\n          response,\n          location: this.location(),\n        };\n      });\n  }\n};\n\n// Returns nothing - doesn't wait for the trigger.\nRestWrite.prototype.runAfterSaveTrigger = function () {\n  if (!this.response || !this.response.response || this.runOptions.many) {\n    return;\n  }\n\n  // Avoid doing any setup for triggers if there is no 'afterSave' trigger for this class.\n  const hasAfterSaveHook = triggers.triggerExists(\n    this.className,\n    triggers.Types.afterSave,\n    this.config.applicationId\n  );\n  const hasLiveQuery = this.config.liveQueryController.hasLiveQuery(this.className);\n  if (!hasAfterSaveHook && !hasLiveQuery) {\n    return Promise.resolve();\n  }\n\n  const { originalObject, updatedObject } = this.buildParseObjects();\n  updatedObject._handleSaveResponse(this.response.response, this.response.status || 200);\n\n  this.config.database.loadSchema().then(schemaController => {\n    // Notifiy LiveQueryServer if possible\n    const perms = schemaController.getClassLevelPermissions(updatedObject.className);\n    this.config.liveQueryController.onAfterSave(\n      updatedObject.className,\n      updatedObject,\n      originalObject,\n      perms\n    );\n  });\n\n  // Run afterSave trigger\n  return triggers\n    .maybeRunTrigger(\n      triggers.Types.afterSave,\n      this.auth,\n      updatedObject,\n      originalObject,\n      this.config,\n      this.context\n    )\n    .then(result => {\n      const jsonReturned = result && !result._toFullJSON;\n      if (jsonReturned) {\n        this.pendingOps.operations = {};\n        this.response.response = result;\n      } else {\n        this.response.response = this._updateResponseWithData(\n          (result || updatedObject).toJSON(),\n          this.data\n        );\n      }\n    })\n    .catch(function (err) {\n      logger.warn('afterSave caught an error', err);\n    });\n};\n\n// A helper to figure out what location this operation happens at.\nRestWrite.prototype.location = function () {\n  var middle = this.className === '_User' ? '/users/' : '/classes/' + this.className + '/';\n  const mount = this.config.mount || this.config.serverURL;\n  return mount + middle + this.data.objectId;\n};\n\n// A helper to get the object id for this operation.\n// Because it could be either on the query or on the data\nRestWrite.prototype.objectId = function () {\n  return this.data.objectId || this.query.objectId;\n};\n\n// Returns a copy of the data and delete bad keys (_auth_data, _hashed_password...)\nRestWrite.prototype.sanitizedData = function () {\n  const data = Object.keys(this.data).reduce((data, key) => {\n    // Regexp comes from Parse.Object.prototype.validate\n    if (!/^[A-Za-z][0-9A-Za-z_]*$/.test(key)) {\n      delete data[key];\n    }\n    return data;\n  }, deepcopy(this.data));\n  return Parse._decode(undefined, data);\n};\n\n// Returns an updated copy of the object\nRestWrite.prototype.buildParseObjects = function () {\n  const extraData = { className: this.className, objectId: this.query?.objectId };\n  let originalObject;\n  if (this.query && this.query.objectId) {\n    originalObject = triggers.inflate(extraData, this.originalData);\n  }\n\n  const className = Parse.Object.fromJSON(extraData);\n  const readOnlyAttributes = className.constructor.readOnlyAttributes\n    ? className.constructor.readOnlyAttributes()\n    : [];\n  if (!this.originalData) {\n    for (const attribute of readOnlyAttributes) {\n      extraData[attribute] = this.data[attribute];\n    }\n  }\n  const updatedObject = triggers.inflate(extraData, this.originalData);\n  Object.keys(this.data).reduce(function (data, key) {\n    if (key.indexOf('.') > 0) {\n      if (typeof data[key].__op === 'string') {\n        if (!readOnlyAttributes.includes(key)) {\n          updatedObject.set(key, data[key]);\n        }\n      } else {\n        // subdocument key with dot notation { 'x.y': v } => { 'x': { 'y' : v } })\n        const splittedKey = key.split('.');\n        const parentProp = splittedKey[0];\n        let parentVal = updatedObject.get(parentProp);\n        if (typeof parentVal !== 'object') {\n          parentVal = {};\n        }\n        parentVal[splittedKey[1]] = data[key];\n        updatedObject.set(parentProp, parentVal);\n      }\n      delete data[key];\n    }\n    return data;\n  }, deepcopy(this.data));\n\n  const sanitized = this.sanitizedData();\n  for (const attribute of readOnlyAttributes) {\n    delete sanitized[attribute];\n  }\n  updatedObject.set(sanitized);\n  return { updatedObject, originalObject };\n};\n\nRestWrite.prototype.cleanUserAuthData = function () {\n  if (this.response && this.response.response && this.className === '_User') {\n    const user = this.response.response;\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\nRestWrite.prototype._updateResponseWithData = function (response, data) {\n  const stateController = Parse.CoreManager.getObjectStateController();\n  const [pending] = stateController.getPendingOps(this.pendingOps.identifier);\n  for (const key in this.pendingOps.operations) {\n    if (!pending[key]) {\n      data[key] = this.originalData ? this.originalData[key] : { __op: 'Delete' };\n      this.storage.fieldsChangedByTrigger.push(key);\n    }\n  }\n  const skipKeys = [...(requiredColumns.read[this.className] || [])];\n  if (!this.query) {\n    skipKeys.push('objectId', 'createdAt');\n  } else {\n    skipKeys.push('updatedAt');\n    delete response.objectId;\n  }\n  for (const key in response) {\n    if (skipKeys.includes(key)) {\n      continue;\n    }\n    const value = response[key];\n    if (\n      value == null ||\n      (value.__type && value.__type === 'Pointer') ||\n      util.isDeepStrictEqual(data[key], value) ||\n      util.isDeepStrictEqual((this.originalData || {})[key], value)\n    ) {\n      delete response[key];\n    }\n  }\n  if (_.isEmpty(this.storage.fieldsChangedByTrigger)) {\n    return response;\n  }\n  const clientSupportsDelete = ClientSDK.supportsForwardDelete(this.clientSDK);\n  this.storage.fieldsChangedByTrigger.forEach(fieldName => {\n    const dataValue = data[fieldName];\n\n    if (!Object.prototype.hasOwnProperty.call(response, fieldName)) {\n      response[fieldName] = dataValue;\n    }\n\n    // Strips operations from responses\n    if (response[fieldName] && response[fieldName].__op) {\n      delete response[fieldName];\n      if (clientSupportsDelete && dataValue.__op == 'Delete') {\n        response[fieldName] = dataValue;\n      }\n    }\n  });\n  return response;\n};\n\nRestWrite.prototype.checkProhibitedKeywords = function (data) {\n  if (this.config.requestKeywordDenylist) {\n    // Scan request data for denied keywords\n    for (const keyword of this.config.requestKeywordDenylist) {\n      const match = Utils.objectContainsKeyValue(data, keyword.key, keyword.value);\n      if (match) {\n        throw new Parse.Error(\n          Parse.Error.INVALID_KEY_NAME,\n          `Prohibited keyword in request data: ${JSON.stringify(keyword)}.`\n        );\n      }\n    }\n  }\n};\n\nexport default RestWrite;\nmodule.exports = RestWrite;\n"],"mappings":"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"}