parse-server 2.8.4 → 8.6.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (240) hide show
  1. package/LICENSE +167 -25
  2. package/NOTICE +10 -0
  3. package/README.md +929 -278
  4. package/lib/AccountLockout.js +47 -30
  5. package/lib/Adapters/AdapterLoader.js +21 -6
  6. package/lib/Adapters/Analytics/AnalyticsAdapter.js +15 -12
  7. package/lib/Adapters/Auth/AuthAdapter.js +116 -13
  8. package/lib/Adapters/Auth/BaseCodeAuthAdapter.js +99 -0
  9. package/lib/Adapters/Auth/OAuth1Client.js +27 -46
  10. package/lib/Adapters/Auth/apple.js +123 -0
  11. package/lib/Adapters/Auth/facebook.js +162 -35
  12. package/lib/Adapters/Auth/gcenter.js +217 -0
  13. package/lib/Adapters/Auth/github.js +118 -48
  14. package/lib/Adapters/Auth/google.js +160 -51
  15. package/lib/Adapters/Auth/gpgames.js +125 -0
  16. package/lib/Adapters/Auth/httpsRequest.js +6 -7
  17. package/lib/Adapters/Auth/index.js +170 -62
  18. package/lib/Adapters/Auth/instagram.js +114 -40
  19. package/lib/Adapters/Auth/janraincapture.js +52 -23
  20. package/lib/Adapters/Auth/janrainengage.js +19 -36
  21. package/lib/Adapters/Auth/keycloak.js +148 -0
  22. package/lib/Adapters/Auth/ldap.js +167 -0
  23. package/lib/Adapters/Auth/line.js +125 -0
  24. package/lib/Adapters/Auth/linkedin.js +111 -55
  25. package/lib/Adapters/Auth/meetup.js +24 -34
  26. package/lib/Adapters/Auth/mfa.js +324 -0
  27. package/lib/Adapters/Auth/microsoft.js +111 -0
  28. package/lib/Adapters/Auth/oauth2.js +97 -162
  29. package/lib/Adapters/Auth/phantauth.js +53 -0
  30. package/lib/Adapters/Auth/qq.js +108 -49
  31. package/lib/Adapters/Auth/spotify.js +107 -55
  32. package/lib/Adapters/Auth/twitter.js +188 -48
  33. package/lib/Adapters/Auth/utils.js +28 -0
  34. package/lib/Adapters/Auth/vkontakte.js +26 -39
  35. package/lib/Adapters/Auth/wechat.js +106 -44
  36. package/lib/Adapters/Auth/weibo.js +132 -58
  37. package/lib/Adapters/Cache/CacheAdapter.js +13 -8
  38. package/lib/Adapters/Cache/InMemoryCache.js +3 -13
  39. package/lib/Adapters/Cache/InMemoryCacheAdapter.js +5 -13
  40. package/lib/Adapters/Cache/LRUCache.js +13 -27
  41. package/lib/Adapters/Cache/NullCacheAdapter.js +3 -8
  42. package/lib/Adapters/Cache/RedisCacheAdapter.js +85 -76
  43. package/lib/Adapters/Cache/SchemaCache.js +25 -0
  44. package/lib/Adapters/Email/MailAdapter.js +10 -8
  45. package/lib/Adapters/Files/FilesAdapter.js +83 -25
  46. package/lib/Adapters/Files/GridFSBucketAdapter.js +231 -0
  47. package/lib/Adapters/Files/GridStoreAdapter.js +4 -91
  48. package/lib/Adapters/Logger/LoggerAdapter.js +18 -14
  49. package/lib/Adapters/Logger/WinstonLogger.js +69 -88
  50. package/lib/Adapters/Logger/WinstonLoggerAdapter.js +7 -16
  51. package/lib/Adapters/MessageQueue/EventEmitterMQ.js +8 -26
  52. package/lib/Adapters/PubSub/EventEmitterPubSub.js +12 -25
  53. package/lib/Adapters/PubSub/PubSubAdapter.js +34 -0
  54. package/lib/Adapters/PubSub/RedisPubSub.js +42 -19
  55. package/lib/Adapters/Push/PushAdapter.js +14 -7
  56. package/lib/Adapters/Storage/Mongo/MongoCollection.js +137 -45
  57. package/lib/Adapters/Storage/Mongo/MongoSchemaCollection.js +158 -63
  58. package/lib/Adapters/Storage/Mongo/MongoStorageAdapter.js +320 -168
  59. package/lib/Adapters/Storage/Mongo/MongoTransform.js +279 -306
  60. package/lib/Adapters/Storage/Postgres/PostgresClient.js +14 -10
  61. package/lib/Adapters/Storage/Postgres/PostgresConfigParser.js +47 -21
  62. package/lib/Adapters/Storage/Postgres/PostgresStorageAdapter.js +854 -468
  63. package/lib/Adapters/Storage/Postgres/sql/index.js +4 -6
  64. package/lib/Adapters/Storage/StorageAdapter.js +1 -1
  65. package/lib/Adapters/WebSocketServer/WSAdapter.js +35 -0
  66. package/lib/Adapters/WebSocketServer/WSSAdapter.js +66 -0
  67. package/lib/Auth.js +488 -125
  68. package/lib/ClientSDK.js +2 -6
  69. package/lib/Config.js +525 -94
  70. package/lib/Controllers/AdaptableController.js +5 -25
  71. package/lib/Controllers/AnalyticsController.js +22 -23
  72. package/lib/Controllers/CacheController.js +10 -31
  73. package/lib/Controllers/DatabaseController.js +767 -313
  74. package/lib/Controllers/FilesController.js +49 -54
  75. package/lib/Controllers/HooksController.js +80 -84
  76. package/lib/Controllers/LiveQueryController.js +35 -22
  77. package/lib/Controllers/LoggerController.js +22 -58
  78. package/lib/Controllers/ParseGraphQLController.js +293 -0
  79. package/lib/Controllers/PushController.js +58 -49
  80. package/lib/Controllers/SchemaController.js +916 -422
  81. package/lib/Controllers/UserController.js +265 -180
  82. package/lib/Controllers/index.js +90 -125
  83. package/lib/Controllers/types.js +1 -1
  84. package/lib/Deprecator/Deprecations.js +30 -0
  85. package/lib/Deprecator/Deprecator.js +127 -0
  86. package/lib/Error.js +48 -0
  87. package/lib/GraphQL/ParseGraphQLSchema.js +375 -0
  88. package/lib/GraphQL/ParseGraphQLServer.js +214 -0
  89. package/lib/GraphQL/helpers/objectsMutations.js +30 -0
  90. package/lib/GraphQL/helpers/objectsQueries.js +246 -0
  91. package/lib/GraphQL/loaders/configMutations.js +87 -0
  92. package/lib/GraphQL/loaders/configQueries.js +79 -0
  93. package/lib/GraphQL/loaders/defaultGraphQLMutations.js +21 -0
  94. package/lib/GraphQL/loaders/defaultGraphQLQueries.js +23 -0
  95. package/lib/GraphQL/loaders/defaultGraphQLTypes.js +1098 -0
  96. package/lib/GraphQL/loaders/defaultRelaySchema.js +53 -0
  97. package/lib/GraphQL/loaders/filesMutations.js +107 -0
  98. package/lib/GraphQL/loaders/functionsMutations.js +78 -0
  99. package/lib/GraphQL/loaders/parseClassMutations.js +268 -0
  100. package/lib/GraphQL/loaders/parseClassQueries.js +127 -0
  101. package/lib/GraphQL/loaders/parseClassTypes.js +493 -0
  102. package/lib/GraphQL/loaders/schemaDirectives.js +62 -0
  103. package/lib/GraphQL/loaders/schemaMutations.js +162 -0
  104. package/lib/GraphQL/loaders/schemaQueries.js +81 -0
  105. package/lib/GraphQL/loaders/schemaTypes.js +341 -0
  106. package/lib/GraphQL/loaders/usersMutations.js +433 -0
  107. package/lib/GraphQL/loaders/usersQueries.js +90 -0
  108. package/lib/GraphQL/parseGraphQLUtils.js +63 -0
  109. package/lib/GraphQL/transformers/className.js +14 -0
  110. package/lib/GraphQL/transformers/constraintType.js +53 -0
  111. package/lib/GraphQL/transformers/inputType.js +51 -0
  112. package/lib/GraphQL/transformers/mutation.js +274 -0
  113. package/lib/GraphQL/transformers/outputType.js +51 -0
  114. package/lib/GraphQL/transformers/query.js +237 -0
  115. package/lib/GraphQL/transformers/schemaFields.js +99 -0
  116. package/lib/KeyPromiseQueue.js +48 -0
  117. package/lib/LiveQuery/Client.js +25 -33
  118. package/lib/LiveQuery/Id.js +2 -5
  119. package/lib/LiveQuery/ParseCloudCodePublisher.js +26 -23
  120. package/lib/LiveQuery/ParseLiveQueryServer.js +560 -285
  121. package/lib/LiveQuery/ParsePubSub.js +7 -16
  122. package/lib/LiveQuery/ParseWebSocketServer.js +42 -39
  123. package/lib/LiveQuery/QueryTools.js +76 -15
  124. package/lib/LiveQuery/RequestSchema.js +111 -97
  125. package/lib/LiveQuery/SessionTokenCache.js +23 -36
  126. package/lib/LiveQuery/Subscription.js +8 -17
  127. package/lib/LiveQuery/equalObjects.js +2 -3
  128. package/lib/Options/Definitions.js +1355 -382
  129. package/lib/Options/docs.js +301 -62
  130. package/lib/Options/index.js +11 -1
  131. package/lib/Options/parsers.js +14 -10
  132. package/lib/Page.js +44 -0
  133. package/lib/ParseMessageQueue.js +6 -13
  134. package/lib/ParseServer.js +474 -235
  135. package/lib/ParseServerRESTController.js +102 -40
  136. package/lib/PromiseRouter.js +39 -50
  137. package/lib/Push/PushQueue.js +24 -30
  138. package/lib/Push/PushWorker.js +32 -56
  139. package/lib/Push/utils.js +22 -35
  140. package/lib/RestQuery.js +361 -139
  141. package/lib/RestWrite.js +713 -344
  142. package/lib/Routers/AggregateRouter.js +97 -71
  143. package/lib/Routers/AnalyticsRouter.js +8 -14
  144. package/lib/Routers/AudiencesRouter.js +16 -35
  145. package/lib/Routers/ClassesRouter.js +86 -72
  146. package/lib/Routers/CloudCodeRouter.js +28 -37
  147. package/lib/Routers/FeaturesRouter.js +22 -25
  148. package/lib/Routers/FilesRouter.js +266 -171
  149. package/lib/Routers/FunctionsRouter.js +87 -103
  150. package/lib/Routers/GlobalConfigRouter.js +94 -33
  151. package/lib/Routers/GraphQLRouter.js +41 -0
  152. package/lib/Routers/HooksRouter.js +43 -47
  153. package/lib/Routers/IAPValidationRouter.js +57 -70
  154. package/lib/Routers/InstallationsRouter.js +17 -25
  155. package/lib/Routers/LogsRouter.js +10 -25
  156. package/lib/Routers/PagesRouter.js +647 -0
  157. package/lib/Routers/PublicAPIRouter.js +104 -112
  158. package/lib/Routers/PurgeRouter.js +19 -29
  159. package/lib/Routers/PushRouter.js +14 -28
  160. package/lib/Routers/RolesRouter.js +7 -14
  161. package/lib/Routers/SchemasRouter.js +63 -42
  162. package/lib/Routers/SecurityRouter.js +34 -0
  163. package/lib/Routers/SessionsRouter.js +25 -38
  164. package/lib/Routers/UsersRouter.js +463 -190
  165. package/lib/SchemaMigrations/DefinedSchemas.js +379 -0
  166. package/lib/SchemaMigrations/Migrations.js +30 -0
  167. package/lib/Security/Check.js +109 -0
  168. package/lib/Security/CheckGroup.js +44 -0
  169. package/lib/Security/CheckGroups/CheckGroupDatabase.js +44 -0
  170. package/lib/Security/CheckGroups/CheckGroupServerConfig.js +96 -0
  171. package/lib/Security/CheckGroups/CheckGroups.js +21 -0
  172. package/lib/Security/CheckRunner.js +213 -0
  173. package/lib/SharedRest.js +29 -0
  174. package/lib/StatusHandler.js +96 -93
  175. package/lib/TestUtils.js +70 -14
  176. package/lib/Utils.js +468 -0
  177. package/lib/batch.js +74 -40
  178. package/lib/cache.js +8 -8
  179. package/lib/cli/definitions/parse-live-query-server.js +4 -3
  180. package/lib/cli/definitions/parse-server.js +4 -3
  181. package/lib/cli/parse-live-query-server.js +9 -17
  182. package/lib/cli/parse-server.js +49 -47
  183. package/lib/cli/utils/commander.js +20 -29
  184. package/lib/cli/utils/runner.js +31 -32
  185. package/lib/cloud-code/Parse.Cloud.js +711 -36
  186. package/lib/cloud-code/Parse.Server.js +21 -0
  187. package/lib/cryptoUtils.js +6 -11
  188. package/lib/defaults.js +21 -15
  189. package/lib/deprecated.js +1 -1
  190. package/lib/index.js +78 -67
  191. package/lib/logger.js +12 -20
  192. package/lib/middlewares.js +484 -160
  193. package/lib/password.js +10 -6
  194. package/lib/request.js +175 -0
  195. package/lib/requiredParameter.js +4 -3
  196. package/lib/rest.js +157 -82
  197. package/lib/triggers.js +627 -185
  198. package/lib/vendor/README.md +3 -3
  199. package/lib/vendor/mongodbUrl.js +224 -137
  200. package/package.json +135 -57
  201. package/postinstall.js +38 -50
  202. package/public_html/invalid_verification_link.html +3 -3
  203. package/types/@types/@parse/fs-files-adapter/index.d.ts +5 -0
  204. package/types/@types/deepcopy/index.d.ts +5 -0
  205. package/types/LiveQuery/ParseLiveQueryServer.d.ts +40 -0
  206. package/types/Options/index.d.ts +301 -0
  207. package/types/ParseServer.d.ts +65 -0
  208. package/types/eslint.config.mjs +30 -0
  209. package/types/index.d.ts +21 -0
  210. package/types/logger.d.ts +2 -0
  211. package/types/tests.ts +44 -0
  212. package/types/tsconfig.json +24 -0
  213. package/CHANGELOG.md +0 -1246
  214. package/PATENTS +0 -37
  215. package/bin/dev +0 -37
  216. package/lib/.DS_Store +0 -0
  217. package/lib/Adapters/Auth/common.js +0 -2
  218. package/lib/Adapters/Auth/facebookaccountkit.js +0 -69
  219. package/lib/Controllers/SchemaCache.js +0 -97
  220. package/lib/LiveQuery/.DS_Store +0 -0
  221. package/lib/cli/utils/parsers.js +0 -77
  222. package/lib/cloud-code/.DS_Store +0 -0
  223. package/lib/cloud-code/HTTPResponse.js +0 -57
  224. package/lib/cloud-code/Untitled-1 +0 -123
  225. package/lib/cloud-code/httpRequest.js +0 -102
  226. package/lib/cloud-code/team.html +0 -123
  227. package/lib/graphql/ParseClass.js +0 -234
  228. package/lib/graphql/Schema.js +0 -197
  229. package/lib/graphql/index.js +0 -1
  230. package/lib/graphql/types/ACL.js +0 -35
  231. package/lib/graphql/types/Date.js +0 -25
  232. package/lib/graphql/types/File.js +0 -24
  233. package/lib/graphql/types/GeoPoint.js +0 -35
  234. package/lib/graphql/types/JSONObject.js +0 -30
  235. package/lib/graphql/types/NumberInput.js +0 -43
  236. package/lib/graphql/types/NumberQuery.js +0 -42
  237. package/lib/graphql/types/Pointer.js +0 -35
  238. package/lib/graphql/types/QueryConstraint.js +0 -61
  239. package/lib/graphql/types/StringQuery.js +0 -39
  240. package/lib/graphql/types/index.js +0 -110
@@ -1,4 +1,4 @@
1
- 'use strict';
1
+ "use strict";
2
2
 
3
3
  Object.defineProperty(exports, "__esModule", {
4
4
  value: true
@@ -6,33 +6,27 @@ Object.defineProperty(exports, "__esModule", {
6
6
  exports.flatten = flatten;
7
7
  exports.jobStatusHandler = jobStatusHandler;
8
8
  exports.pushStatusHandler = pushStatusHandler;
9
-
10
- var _cryptoUtils = require('./cryptoUtils');
11
-
12
- var _logger = require('./logger');
13
-
14
- var _rest = require('./rest');
15
-
16
- var _rest2 = _interopRequireDefault(_rest);
17
-
18
- var _Auth = require('./Auth');
19
-
20
- var _Auth2 = _interopRequireDefault(_Auth);
21
-
22
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
23
-
9
+ var _cryptoUtils = require("./cryptoUtils");
10
+ var _KeyPromiseQueue = require("./KeyPromiseQueue");
11
+ var _logger = require("./logger");
12
+ var _rest = _interopRequireDefault(require("./rest"));
13
+ var _Auth = _interopRequireDefault(require("./Auth"));
14
+ function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
24
15
  const PUSH_STATUS_COLLECTION = '_PushStatus';
25
16
  const JOB_STATUS_COLLECTION = '_JobStatus';
26
-
17
+ const pushPromiseQueue = new _KeyPromiseQueue.KeyPromiseQueue();
18
+ const jobPromiseQueue = new _KeyPromiseQueue.KeyPromiseQueue();
27
19
  const incrementOp = function (object = {}, key, amount = 1) {
28
20
  if (!object[key]) {
29
- object[key] = { __op: 'Increment', amount: amount };
21
+ object[key] = {
22
+ __op: 'Increment',
23
+ amount: amount
24
+ };
30
25
  } else {
31
26
  object[key].amount += amount;
32
27
  }
33
28
  return object[key];
34
29
  };
35
-
36
30
  function flatten(array) {
37
31
  var flattened = [];
38
32
  for (var i = 0; i < array.length; i++) {
@@ -44,107 +38,99 @@ function flatten(array) {
44
38
  }
45
39
  return flattened;
46
40
  }
47
-
48
41
  function statusHandler(className, database) {
49
- let lastPromise = Promise.resolve();
50
-
51
42
  function create(object) {
52
- lastPromise = lastPromise.then(() => {
53
- return database.create(className, object).then(() => {
54
- return Promise.resolve(object);
55
- });
43
+ return database.create(className, object).then(() => {
44
+ return Promise.resolve(object);
56
45
  });
57
- return lastPromise;
58
46
  }
59
-
60
47
  function update(where, object) {
61
- lastPromise = lastPromise.then(() => {
62
- return database.update(className, where, object);
63
- });
64
- return lastPromise;
48
+ return jobPromiseQueue.enqueue(where.objectId, () => database.update(className, where, object));
65
49
  }
66
-
67
50
  return Object.freeze({
68
51
  create,
69
52
  update
70
53
  });
71
54
  }
72
-
73
55
  function restStatusHandler(className, config) {
74
- let lastPromise = Promise.resolve();
75
- const auth = _Auth2.default.master(config);
56
+ const auth = _Auth.default.master(config);
76
57
  function create(object) {
77
- lastPromise = lastPromise.then(() => {
78
- return _rest2.default.create(config, auth, className, object).then(({ response }) => {
79
- // merge the objects
80
- return Promise.resolve(Object.assign({}, object, response));
81
- });
58
+ return _rest.default.create(config, auth, className, object).then(({
59
+ response
60
+ }) => {
61
+ return {
62
+ ...object,
63
+ ...response
64
+ };
82
65
  });
83
- return lastPromise;
84
66
  }
85
-
86
67
  function update(where, object) {
87
- // TODO: when we have updateWhere, use that for proper interfacing
88
- lastPromise = lastPromise.then(() => {
89
- return _rest2.default.update(config, auth, className, { objectId: where.objectId }, object).then(({ response }) => {
90
- // merge the objects
91
- return Promise.resolve(Object.assign({}, object, response));
92
- });
93
- });
94
- return lastPromise;
68
+ return pushPromiseQueue.enqueue(where.objectId, () => _rest.default.update(config, auth, className, {
69
+ objectId: where.objectId
70
+ }, object).then(({
71
+ response
72
+ }) => {
73
+ return {
74
+ ...object,
75
+ ...response
76
+ };
77
+ }));
95
78
  }
96
-
97
79
  return Object.freeze({
98
80
  create,
99
81
  update
100
82
  });
101
83
  }
102
-
103
84
  function jobStatusHandler(config) {
104
85
  let jobStatus;
105
86
  const objectId = (0, _cryptoUtils.newObjectId)(config.objectIdSize);
106
87
  const database = config.database;
107
88
  const handler = statusHandler(JOB_STATUS_COLLECTION, database);
108
- const setRunning = function (jobName, params) {
89
+ const setRunning = function (jobName) {
109
90
  const now = new Date();
110
91
  jobStatus = {
111
92
  objectId,
112
93
  jobName,
113
- params,
114
94
  status: 'running',
115
95
  source: 'api',
116
96
  createdAt: now,
117
97
  // lockdown!
118
98
  ACL: {}
119
99
  };
120
-
121
100
  return handler.create(jobStatus);
122
101
  };
123
-
124
102
  const setMessage = function (message) {
125
103
  if (!message || typeof message !== 'string') {
126
104
  return Promise.resolve();
127
105
  }
128
- return handler.update({ objectId }, { message });
106
+ return handler.update({
107
+ objectId
108
+ }, {
109
+ message
110
+ });
129
111
  };
130
-
131
112
  const setSucceeded = function (message) {
132
113
  return setFinalStatus('succeeded', message);
133
114
  };
134
-
135
115
  const setFailed = function (message) {
136
116
  return setFinalStatus('failed', message);
137
117
  };
138
-
139
118
  const setFinalStatus = function (status, message = undefined) {
140
119
  const finishedAt = new Date();
141
- const update = { status, finishedAt };
120
+ const update = {
121
+ status,
122
+ finishedAt
123
+ };
142
124
  if (message && typeof message === 'string') {
143
125
  update.message = message;
144
126
  }
145
- return handler.update({ objectId }, update);
127
+ if (message instanceof Error && typeof message.message === 'string') {
128
+ update.message = message.message;
129
+ }
130
+ return handler.update({
131
+ objectId
132
+ }, update);
146
133
  };
147
-
148
134
  return Object.freeze({
149
135
  setRunning,
150
136
  setSucceeded,
@@ -152,18 +138,18 @@ function jobStatusHandler(config) {
152
138
  setFailed
153
139
  });
154
140
  }
155
-
156
141
  function pushStatusHandler(config, existingObjectId) {
157
-
158
142
  let pushStatus;
159
143
  const database = config.database;
160
144
  const handler = restStatusHandler(PUSH_STATUS_COLLECTION, config);
161
145
  let objectId = existingObjectId;
162
- const setInitial = function (body = {}, where, options = { source: 'rest' }) {
146
+ const setInitial = function (body = {}, where, options = {
147
+ source: 'rest'
148
+ }) {
163
149
  const now = new Date();
164
150
  let pushTime = now.toISOString();
165
151
  let status = 'pending';
166
- if (body.hasOwnProperty('push_time')) {
152
+ if (Object.prototype.hasOwnProperty.call(body, 'push_time')) {
167
153
  if (config.hasPushScheduledSupport) {
168
154
  pushTime = body.push_time;
169
155
  status = 'scheduled';
@@ -172,7 +158,6 @@ function pushStatusHandler(config, existingObjectId) {
172
158
  _logger.logger.warn('Push will be sent immediately');
173
159
  }
174
160
  }
175
-
176
161
  const data = body.data || {};
177
162
  const payloadString = JSON.stringify(data);
178
163
  let pushHash;
@@ -205,18 +190,16 @@ function pushStatusHandler(config, existingObjectId) {
205
190
  return Promise.resolve(pushStatus);
206
191
  });
207
192
  };
208
-
209
193
  const setRunning = function (batches) {
210
194
  _logger.logger.verbose(`_PushStatus ${objectId}: sending push to installations with %d batches`, batches);
211
195
  return handler.update({
212
- status: "pending",
196
+ status: 'pending',
213
197
  objectId: objectId
214
198
  }, {
215
- status: "running",
199
+ status: 'running',
216
200
  count: batches
217
201
  });
218
202
  };
219
-
220
203
  const trackSent = function (results, UTCOffset, cleanupInstallations = process.env.PARSE_SERVER_CLEANUP_INVALID_INSTALLATIONS) {
221
204
  const update = {
222
205
  numSent: 0,
@@ -243,11 +226,19 @@ function pushStatusHandler(config, existingObjectId) {
243
226
  if (result && result.response && result.response.error && result.device && result.device.deviceToken) {
244
227
  const token = result.device.deviceToken;
245
228
  const error = result.response.error;
246
- // GCM errors
229
+ // GCM / FCM HTTP v1 API errors; see:
230
+ // https://firebase.google.com/docs/reference/fcm/rest/v1/ErrorCode
247
231
  if (error === 'NotRegistered' || error === 'InvalidRegistration') {
248
232
  devicesToRemove.push(token);
249
233
  }
250
- // APNS errors
234
+ // FCM API v2 errors; see:
235
+ // https://firebase.google.com/docs/cloud-messaging/manage-tokens
236
+ // https://github.com/firebase/functions-samples/blob/703c0359eacf07a551751d1319d34f912a2cd828/Node/fcm-notifications/functions/index.js#L89-L93C16
237
+ if (error?.code === 'messaging/registration-token-not-registered' || error?.code === 'messaging/invalid-registration-token' || error?.code === 'messaging/invalid-argument' && error?.message === 'The registration token is not a valid FCM registration token') {
238
+ devicesToRemove.push(token);
239
+ }
240
+ // APNS errors; see:
241
+ // https://developer.apple.com/documentation/usernotifications/handling-notification-responses-from-apns
251
242
  if (error === 'Unregistered' || error === 'BadDeviceToken') {
252
243
  devicesToRemove.push(token);
253
244
  }
@@ -257,9 +248,10 @@ function pushStatusHandler(config, existingObjectId) {
257
248
  return memo;
258
249
  }, update);
259
250
  }
260
-
261
251
  _logger.logger.verbose(`_PushStatus ${objectId}: sent push! %d success, %d failures`, update.numSent, update.numFailed);
262
- _logger.logger.verbose(`_PushStatus ${objectId}: needs cleanup`, { devicesToRemove });
252
+ _logger.logger.verbose(`_PushStatus ${objectId}: needs cleanup`, {
253
+ devicesToRemove
254
+ });
263
255
  ['numSent', 'numFailed'].forEach(key => {
264
256
  if (update[key] > 0) {
265
257
  update[key] = {
@@ -270,43 +262,55 @@ function pushStatusHandler(config, existingObjectId) {
270
262
  delete update[key];
271
263
  }
272
264
  });
273
-
274
265
  if (devicesToRemove.length > 0 && cleanupInstallations) {
275
266
  _logger.logger.info(`Removing device tokens on ${devicesToRemove.length} _Installations`);
276
- database.update('_Installation', { deviceToken: { '$in': devicesToRemove } }, { deviceToken: { "__op": "Delete" } }, {
267
+ database.update('_Installation', {
268
+ deviceToken: {
269
+ $in: devicesToRemove
270
+ }
271
+ }, {
272
+ deviceToken: {
273
+ __op: 'Delete'
274
+ }
275
+ }, {
277
276
  acl: undefined,
278
277
  many: true
279
278
  });
280
279
  }
281
-
282
- // indicate this batch is complete
283
280
  incrementOp(update, 'count', -1);
284
-
285
- return handler.update({ objectId }, update).then(res => {
281
+ update.status = 'running';
282
+ return handler.update({
283
+ objectId
284
+ }, update).then(res => {
286
285
  if (res && res.count === 0) {
287
286
  return this.complete();
288
287
  }
289
288
  });
290
289
  };
291
-
292
290
  const complete = function () {
293
- return handler.update({ objectId }, {
291
+ return handler.update({
292
+ objectId
293
+ }, {
294
294
  status: 'succeeded',
295
- count: { __op: 'Delete' }
295
+ count: {
296
+ __op: 'Delete'
297
+ }
296
298
  });
297
299
  };
298
-
299
300
  const fail = function (err) {
300
301
  if (typeof err === 'string') {
301
- err = { message: err };
302
+ err = {
303
+ message: err
304
+ };
302
305
  }
303
306
  const update = {
304
307
  errorMessage: err,
305
308
  status: 'failed'
306
309
  };
307
- return handler.update({ objectId }, update);
310
+ return handler.update({
311
+ objectId
312
+ }, update);
308
313
  };
309
-
310
314
  const rval = {
311
315
  setInitial,
312
316
  setRunning,
@@ -316,10 +320,9 @@ function pushStatusHandler(config, existingObjectId) {
316
320
  };
317
321
 
318
322
  // define objectId to be dynamic
319
- Object.defineProperty(rval, "objectId", {
323
+ Object.defineProperty(rval, 'objectId', {
320
324
  get: () => objectId
321
325
  });
322
-
323
326
  return Object.freeze(rval);
324
327
  }
325
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/StatusHandler.js"],"names":["flatten","jobStatusHandler","pushStatusHandler","PUSH_STATUS_COLLECTION","JOB_STATUS_COLLECTION","incrementOp","object","key","amount","__op","array","flattened","i","length","Array","isArray","concat","push","statusHandler","className","database","lastPromise","Promise","resolve","create","then","update","where","Object","freeze","restStatusHandler","config","auth","Auth","master","rest","response","assign","objectId","jobStatus","objectIdSize","handler","setRunning","jobName","params","now","Date","status","source","createdAt","ACL","setMessage","message","setSucceeded","setFinalStatus","setFailed","undefined","finishedAt","existingObjectId","pushStatus","setInitial","body","options","pushTime","toISOString","hasOwnProperty","hasPushScheduledSupport","push_time","logger","warn","data","payloadString","JSON","stringify","pushHash","alert","query","payload","title","expiry","expiration_time","expiration_interval","numSent","result","batches","verbose","count","trackSent","results","UTCOffset","cleanupInstallations","process","env","PARSE_SERVER_CLEANUP_INVALID_INSTALLATIONS","numFailed","devicesToRemove","reduce","memo","device","deviceType","transmitted","offsetKey","error","deviceToken","token","forEach","info","acl","many","res","complete","fail","err","errorMessage","rval","defineProperty","get"],"mappings":"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","file":"StatusHandler.js","sourcesContent":["import { md5Hash, newObjectId } from './cryptoUtils';\nimport { logger }               from './logger';\nimport rest                     from './rest';\nimport Auth                     from './Auth';\n\nconst PUSH_STATUS_COLLECTION = '_PushStatus';\nconst JOB_STATUS_COLLECTION = '_JobStatus';\n\nconst incrementOp = function(object = {}, key, amount = 1) {\n  if (!object[key]) {\n    object[key] = {__op: 'Increment', amount: amount}\n  } else {\n    object[key].amount += amount;\n  }\n  return object[key];\n}\n\nexport function flatten(array) {\n  var flattened = [];\n  for(var i = 0; i < array.length; i++) {\n    if(Array.isArray(array[i])) {\n      flattened = flattened.concat(flatten(array[i]));\n    } else {\n      flattened.push(array[i]);\n    }\n  }\n  return flattened;\n}\n\nfunction statusHandler(className, database) {\n  let lastPromise = Promise.resolve();\n\n  function create(object) {\n    lastPromise = lastPromise.then(() => {\n      return database.create(className, object).then(() => {\n        return Promise.resolve(object);\n      });\n    });\n    return lastPromise;\n  }\n\n  function update(where, object) {\n    lastPromise = lastPromise.then(() => {\n      return database.update(className, where, object);\n    });\n    return lastPromise;\n  }\n\n  return Object.freeze({\n    create,\n    update\n  })\n}\n\nfunction restStatusHandler(className, config) {\n  let lastPromise = Promise.resolve();\n  const auth = Auth.master(config);\n  function create(object) {\n    lastPromise = lastPromise.then(() => {\n      return rest.create(config, auth, className, object)\n        .then(({ response }) => {\n          // merge the objects\n          return Promise.resolve(Object.assign({}, object, response));\n        });\n    });\n    return lastPromise;\n  }\n\n  function update(where, object) {\n    // TODO: when we have updateWhere, use that for proper interfacing\n    lastPromise = lastPromise.then(() => {\n      return rest.update(config, auth, className, { objectId: where.objectId }, object)\n        .then(({ response }) => {\n          // merge the objects\n          return Promise.resolve(Object.assign({}, object, response));\n        });\n    });\n    return lastPromise;\n  }\n\n  return Object.freeze({\n    create,\n    update\n  })\n}\n\nexport function jobStatusHandler(config) {\n  let jobStatus;\n  const objectId = newObjectId(config.objectIdSize);\n  const database = config.database;\n  const handler = statusHandler(JOB_STATUS_COLLECTION, database);\n  const setRunning = function(jobName, params) {\n    const now = new Date();\n    jobStatus = {\n      objectId,\n      jobName,\n      params,\n      status: 'running',\n      source: 'api',\n      createdAt: now,\n      // lockdown!\n      ACL: {}\n    }\n\n    return handler.create(jobStatus);\n  }\n\n  const setMessage = function(message) {\n    if (!message || typeof message !== 'string') {\n      return Promise.resolve();\n    }\n    return handler.update({ objectId }, { message });\n  }\n\n  const setSucceeded = function(message) {\n    return setFinalStatus('succeeded', message);\n  }\n\n  const setFailed = function(message) {\n    return setFinalStatus('failed', message);\n  }\n\n  const setFinalStatus = function(status, message = undefined) {\n    const finishedAt = new Date();\n    const update = { status, finishedAt };\n    if (message && typeof message === 'string') {\n      update.message = message;\n    }\n    return handler.update({ objectId }, update);\n  }\n\n  return Object.freeze({\n    setRunning,\n    setSucceeded,\n    setMessage,\n    setFailed\n  });\n}\n\nexport function pushStatusHandler(config, existingObjectId) {\n\n  let pushStatus;\n  const database = config.database;\n  const handler = restStatusHandler(PUSH_STATUS_COLLECTION, config);\n  let objectId = existingObjectId;\n  const setInitial = function(body = {}, where, options = {source: 'rest'}) {\n    const now = new Date();\n    let pushTime = now.toISOString();\n    let status = 'pending';\n    if (body.hasOwnProperty('push_time')) {\n      if (config.hasPushScheduledSupport) {\n        pushTime = body.push_time;\n        status = 'scheduled';\n      } else {\n        logger.warn('Trying to schedule a push while server is not configured.');\n        logger.warn('Push will be sent immediately');\n      }\n    }\n\n    const data =  body.data || {};\n    const payloadString = JSON.stringify(data);\n    let pushHash;\n    if (typeof data.alert === 'string') {\n      pushHash = md5Hash(data.alert);\n    } else if (typeof data.alert === 'object') {\n      pushHash = md5Hash(JSON.stringify(data.alert));\n    } else {\n      pushHash = 'd41d8cd98f00b204e9800998ecf8427e';\n    }\n    const object = {\n      pushTime,\n      query: JSON.stringify(where),\n      payload: payloadString,\n      source: options.source,\n      title: options.title,\n      expiry: body.expiration_time,\n      expiration_interval: body.expiration_interval,\n      status: status,\n      numSent: 0,\n      pushHash,\n      // lockdown!\n      ACL: {}\n    }\n    return handler.create(object).then((result) => {\n      objectId = result.objectId;\n      pushStatus = {\n        objectId\n      };\n      return Promise.resolve(pushStatus);\n    });\n  }\n\n  const setRunning = function(batches) {\n    logger.verbose(`_PushStatus ${objectId}: sending push to installations with %d batches`, batches);\n    return handler.update(\n      {\n        status:\"pending\",\n        objectId: objectId\n      },\n      {\n        status: \"running\",\n        count: batches\n      }\n    );\n  }\n\n  const trackSent = function(results, UTCOffset, cleanupInstallations = process.env.PARSE_SERVER_CLEANUP_INVALID_INSTALLATIONS) {\n    const update = {\n      numSent: 0,\n      numFailed: 0\n    };\n    const devicesToRemove = [];\n    if (Array.isArray(results)) {\n      results = flatten(results);\n      results.reduce((memo, result) => {\n        // Cannot handle that\n        if (!result || !result.device || !result.device.deviceType) {\n          return memo;\n        }\n        const deviceType = result.device.deviceType;\n        const key = result.transmitted ? `sentPerType.${deviceType}` : `failedPerType.${deviceType}`;\n        memo[key] = incrementOp(memo, key);\n        if (typeof UTCOffset !== 'undefined') {\n          const offsetKey = result.transmitted ? `sentPerUTCOffset.${UTCOffset}` : `failedPerUTCOffset.${UTCOffset}`;\n          memo[offsetKey] = incrementOp(memo, offsetKey);\n        }\n        if (result.transmitted) {\n          memo.numSent++;\n        } else {\n          if (result && result.response && result.response.error && result.device && result.device.deviceToken) {\n            const token = result.device.deviceToken;\n            const error = result.response.error;\n            // GCM errors\n            if (error === 'NotRegistered' || error === 'InvalidRegistration') {\n              devicesToRemove.push(token);\n            }\n            // APNS errors\n            if (error === 'Unregistered' || error === 'BadDeviceToken') {\n              devicesToRemove.push(token);\n            }\n          }\n          memo.numFailed++;\n        }\n        return memo;\n      }, update);\n    }\n\n    logger.verbose(`_PushStatus ${objectId}: sent push! %d success, %d failures`, update.numSent, update.numFailed);\n    logger.verbose(`_PushStatus ${objectId}: needs cleanup`, { devicesToRemove });\n    ['numSent', 'numFailed'].forEach((key) => {\n      if (update[key] > 0) {\n        update[key] = {\n          __op: 'Increment',\n          amount: update[key]\n        };\n      } else {\n        delete update[key];\n      }\n    });\n\n    if (devicesToRemove.length > 0 && cleanupInstallations) {\n      logger.info(`Removing device tokens on ${devicesToRemove.length} _Installations`);\n      database.update('_Installation', { deviceToken: { '$in': devicesToRemove }}, { deviceToken: {\"__op\": \"Delete\"} }, {\n        acl: undefined,\n        many: true\n      });\n    }\n\n    // indicate this batch is complete\n    incrementOp(update, 'count', -1);\n\n    return handler.update({ objectId }, update).then((res) => {\n      if (res && res.count === 0) {\n        return this.complete();\n      }\n    })\n  }\n\n  const complete = function() {\n    return handler.update({ objectId }, {\n      status: 'succeeded',\n      count: {__op: 'Delete'}\n    });\n  }\n\n  const fail = function(err) {\n    if (typeof err === 'string') {\n      err = { message: err };\n    }\n    const update = {\n      errorMessage: err,\n      status: 'failed'\n    }\n    return handler.update({ objectId }, update);\n  }\n\n  const rval = {\n    setInitial,\n    setRunning,\n    trackSent,\n    complete,\n    fail\n  };\n\n  // define objectId to be dynamic\n  Object.defineProperty(rval, \"objectId\", {\n    get: () => objectId\n  });\n\n  return Object.freeze(rval);\n}\n"]}
328
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_cryptoUtils","require","_KeyPromiseQueue","_logger","_rest","_interopRequireDefault","_Auth","e","__esModule","default","PUSH_STATUS_COLLECTION","JOB_STATUS_COLLECTION","pushPromiseQueue","KeyPromiseQueue","jobPromiseQueue","incrementOp","object","key","amount","__op","flatten","array","flattened","i","length","Array","isArray","concat","push","statusHandler","className","database","create","then","Promise","resolve","update","where","enqueue","objectId","Object","freeze","restStatusHandler","config","auth","Auth","master","rest","response","jobStatusHandler","jobStatus","newObjectId","objectIdSize","handler","setRunning","jobName","now","Date","status","source","createdAt","ACL","setMessage","message","setSucceeded","setFinalStatus","setFailed","undefined","finishedAt","Error","pushStatusHandler","existingObjectId","pushStatus","setInitial","body","options","pushTime","toISOString","prototype","hasOwnProperty","call","hasPushScheduledSupport","push_time","logger","warn","data","payloadString","JSON","stringify","pushHash","alert","md5Hash","query","payload","title","expiry","expiration_time","expiration_interval","numSent","result","batches","verbose","count","trackSent","results","UTCOffset","cleanupInstallations","process","env","PARSE_SERVER_CLEANUP_INVALID_INSTALLATIONS","numFailed","devicesToRemove","reduce","memo","device","deviceType","transmitted","offsetKey","error","deviceToken","token","code","forEach","info","$in","acl","many","res","complete","fail","err","errorMessage","rval","defineProperty","get"],"sources":["../src/StatusHandler.js"],"sourcesContent":["import { md5Hash, newObjectId } from './cryptoUtils';\nimport { KeyPromiseQueue } from './KeyPromiseQueue';\nimport { logger } from './logger';\nimport rest from './rest';\nimport Auth from './Auth';\n\nconst PUSH_STATUS_COLLECTION = '_PushStatus';\nconst JOB_STATUS_COLLECTION = '_JobStatus';\n\nconst pushPromiseQueue = new KeyPromiseQueue();\nconst jobPromiseQueue = new KeyPromiseQueue();\n\nconst incrementOp = function (object = {}, key, amount = 1) {\n  if (!object[key]) {\n    object[key] = { __op: 'Increment', amount: amount };\n  } else {\n    object[key].amount += amount;\n  }\n  return object[key];\n};\n\nexport function flatten(array) {\n  var flattened = [];\n  for (var i = 0; i < array.length; i++) {\n    if (Array.isArray(array[i])) {\n      flattened = flattened.concat(flatten(array[i]));\n    } else {\n      flattened.push(array[i]);\n    }\n  }\n  return flattened;\n}\n\nfunction statusHandler(className, database) {\n  function create(object) {\n    return database.create(className, object).then(() => {\n      return Promise.resolve(object);\n    });\n  }\n\n  function update(where, object) {\n    return jobPromiseQueue.enqueue(where.objectId, () => database.update(className, where, object));\n  }\n\n  return Object.freeze({\n    create,\n    update,\n  });\n}\n\nfunction restStatusHandler(className, config) {\n  const auth = Auth.master(config);\n  function create(object) {\n    return rest.create(config, auth, className, object).then(({ response }) => {\n      return { ...object, ...response };\n    });\n  }\n\n  function update(where, object) {\n    return pushPromiseQueue.enqueue(where.objectId, () =>\n      rest\n        .update(config, auth, className, { objectId: where.objectId }, object)\n        .then(({ response }) => {\n          return { ...object, ...response };\n        })\n    );\n  }\n\n  return Object.freeze({\n    create,\n    update,\n  });\n}\n\nexport function jobStatusHandler(config) {\n  let jobStatus;\n  const objectId = newObjectId(config.objectIdSize);\n  const database = config.database;\n  const handler = statusHandler(JOB_STATUS_COLLECTION, database);\n  const setRunning = function (jobName) {\n    const now = new Date();\n    jobStatus = {\n      objectId,\n      jobName,\n      status: 'running',\n      source: 'api',\n      createdAt: now,\n      // lockdown!\n      ACL: {},\n    };\n\n    return handler.create(jobStatus);\n  };\n\n  const setMessage = function (message) {\n    if (!message || typeof message !== 'string') {\n      return Promise.resolve();\n    }\n    return handler.update({ objectId }, { message });\n  };\n\n  const setSucceeded = function (message) {\n    return setFinalStatus('succeeded', message);\n  };\n\n  const setFailed = function (message) {\n    return setFinalStatus('failed', message);\n  };\n\n  const setFinalStatus = function (status, message = undefined) {\n    const finishedAt = new Date();\n    const update = { status, finishedAt };\n    if (message && typeof message === 'string') {\n      update.message = message;\n    }\n    if (message instanceof Error && typeof message.message === 'string') {\n      update.message = message.message;\n    }\n    return handler.update({ objectId }, update);\n  };\n\n  return Object.freeze({\n    setRunning,\n    setSucceeded,\n    setMessage,\n    setFailed,\n  });\n}\n\nexport function pushStatusHandler(config, existingObjectId) {\n  let pushStatus;\n  const database = config.database;\n  const handler = restStatusHandler(PUSH_STATUS_COLLECTION, config);\n  let objectId = existingObjectId;\n  const setInitial = function (body = {}, where, options = { source: 'rest' }) {\n    const now = new Date();\n    let pushTime = now.toISOString();\n    let status = 'pending';\n    if (Object.prototype.hasOwnProperty.call(body, 'push_time')) {\n      if (config.hasPushScheduledSupport) {\n        pushTime = body.push_time;\n        status = 'scheduled';\n      } else {\n        logger.warn('Trying to schedule a push while server is not configured.');\n        logger.warn('Push will be sent immediately');\n      }\n    }\n\n    const data = body.data || {};\n    const payloadString = JSON.stringify(data);\n    let pushHash;\n    if (typeof data.alert === 'string') {\n      pushHash = md5Hash(data.alert);\n    } else if (typeof data.alert === 'object') {\n      pushHash = md5Hash(JSON.stringify(data.alert));\n    } else {\n      pushHash = 'd41d8cd98f00b204e9800998ecf8427e';\n    }\n    const object = {\n      pushTime,\n      query: JSON.stringify(where),\n      payload: payloadString,\n      source: options.source,\n      title: options.title,\n      expiry: body.expiration_time,\n      expiration_interval: body.expiration_interval,\n      status: status,\n      numSent: 0,\n      pushHash,\n      // lockdown!\n      ACL: {},\n    };\n    return handler.create(object).then(result => {\n      objectId = result.objectId;\n      pushStatus = {\n        objectId,\n      };\n      return Promise.resolve(pushStatus);\n    });\n  };\n\n  const setRunning = function (batches) {\n    logger.verbose(\n      `_PushStatus ${objectId}: sending push to installations with %d batches`,\n      batches\n    );\n    return handler.update(\n      {\n        status: 'pending',\n        objectId: objectId,\n      },\n      {\n        status: 'running',\n        count: batches,\n      }\n    );\n  };\n\n  const trackSent = function (\n    results,\n    UTCOffset,\n    cleanupInstallations = process.env.PARSE_SERVER_CLEANUP_INVALID_INSTALLATIONS\n  ) {\n    const update = {\n      numSent: 0,\n      numFailed: 0,\n    };\n    const devicesToRemove = [];\n    if (Array.isArray(results)) {\n      results = flatten(results);\n      results.reduce((memo, result) => {\n        // Cannot handle that\n        if (!result || !result.device || !result.device.deviceType) {\n          return memo;\n        }\n        const deviceType = result.device.deviceType;\n        const key = result.transmitted\n          ? `sentPerType.${deviceType}`\n          : `failedPerType.${deviceType}`;\n        memo[key] = incrementOp(memo, key);\n        if (typeof UTCOffset !== 'undefined') {\n          const offsetKey = result.transmitted\n            ? `sentPerUTCOffset.${UTCOffset}`\n            : `failedPerUTCOffset.${UTCOffset}`;\n          memo[offsetKey] = incrementOp(memo, offsetKey);\n        }\n        if (result.transmitted) {\n          memo.numSent++;\n        } else {\n          if (\n            result &&\n            result.response &&\n            result.response.error &&\n            result.device &&\n            result.device.deviceToken\n          ) {\n            const token = result.device.deviceToken;\n            const error = result.response.error;\n            // GCM / FCM HTTP v1 API errors; see:\n            // https://firebase.google.com/docs/reference/fcm/rest/v1/ErrorCode\n            if (error === 'NotRegistered' || error === 'InvalidRegistration') {\n              devicesToRemove.push(token);\n            }\n            // FCM API v2 errors; see:\n            // https://firebase.google.com/docs/cloud-messaging/manage-tokens\n            // https://github.com/firebase/functions-samples/blob/703c0359eacf07a551751d1319d34f912a2cd828/Node/fcm-notifications/functions/index.js#L89-L93C16\n            if (\n              error?.code === 'messaging/registration-token-not-registered' ||\n              error?.code === 'messaging/invalid-registration-token' ||\n              (error?.code === 'messaging/invalid-argument' && error?.message === 'The registration token is not a valid FCM registration token')\n            ) {\n              devicesToRemove.push(token);\n            }\n            // APNS errors; see:\n            // https://developer.apple.com/documentation/usernotifications/handling-notification-responses-from-apns\n            if (error === 'Unregistered' || error === 'BadDeviceToken') {\n              devicesToRemove.push(token);\n            }\n          }\n          memo.numFailed++;\n        }\n        return memo;\n      }, update);\n    }\n\n    logger.verbose(\n      `_PushStatus ${objectId}: sent push! %d success, %d failures`,\n      update.numSent,\n      update.numFailed\n    );\n    logger.verbose(`_PushStatus ${objectId}: needs cleanup`, {\n      devicesToRemove,\n    });\n    ['numSent', 'numFailed'].forEach(key => {\n      if (update[key] > 0) {\n        update[key] = {\n          __op: 'Increment',\n          amount: update[key],\n        };\n      } else {\n        delete update[key];\n      }\n    });\n\n    if (devicesToRemove.length > 0 && cleanupInstallations) {\n      logger.info(`Removing device tokens on ${devicesToRemove.length} _Installations`);\n      database.update(\n        '_Installation',\n        { deviceToken: { $in: devicesToRemove } },\n        { deviceToken: { __op: 'Delete' } },\n        {\n          acl: undefined,\n          many: true,\n        }\n      );\n    }\n    incrementOp(update, 'count', -1);\n    update.status = 'running';\n\n    return handler.update({ objectId }, update).then(res => {\n      if (res && res.count === 0) {\n        return this.complete();\n      }\n    });\n  };\n\n  const complete = function () {\n    return handler.update(\n      { objectId },\n      {\n        status: 'succeeded',\n        count: { __op: 'Delete' },\n      }\n    );\n  };\n\n  const fail = function (err) {\n    if (typeof err === 'string') {\n      err = { message: err };\n    }\n    const update = {\n      errorMessage: err,\n      status: 'failed',\n    };\n    return handler.update({ objectId }, update);\n  };\n\n  const rval = {\n    setInitial,\n    setRunning,\n    trackSent,\n    complete,\n    fail,\n  };\n\n  // define objectId to be dynamic\n  Object.defineProperty(rval, 'objectId', {\n    get: () => objectId,\n  });\n\n  return Object.freeze(rval);\n}\n"],"mappings":"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","ignoreList":[]}
package/lib/TestUtils.js CHANGED
@@ -1,16 +1,16 @@
1
- 'use strict';
1
+ "use strict";
2
2
 
3
3
  Object.defineProperty(exports, "__esModule", {
4
4
  value: true
5
5
  });
6
+ exports.Connections = void 0;
6
7
  exports.destroyAllDataPermanently = destroyAllDataPermanently;
7
-
8
- var _cache = require('./cache');
9
-
10
- var _cache2 = _interopRequireDefault(_cache);
11
-
12
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
13
-
8
+ exports.getConnectionsCount = getConnectionsCount;
9
+ exports.resolvingPromise = resolvingPromise;
10
+ exports.sleep = sleep;
11
+ var _cache = _interopRequireDefault(require("./cache"));
12
+ var _SchemaCache = _interopRequireDefault(require("./Adapters/Cache/SchemaCache"));
13
+ function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
14
14
  /**
15
15
  * Destroys all data in the database
16
16
  * @param {boolean} fast set to true if it's ok to just drop objects and not indexes.
@@ -19,13 +19,69 @@ function destroyAllDataPermanently(fast) {
19
19
  if (!process.env.TESTING) {
20
20
  throw 'Only supported in test environment';
21
21
  }
22
- return Promise.all(Object.keys(_cache2.default.cache).map(appId => {
23
- const app = _cache2.default.get(appId);
22
+ return Promise.all(Object.keys(_cache.default.cache).map(appId => {
23
+ const app = _cache.default.get(appId);
24
+ const deletePromises = [];
25
+ if (app.cacheAdapter && app.cacheAdapter.clear) {
26
+ deletePromises.push(app.cacheAdapter.clear());
27
+ }
24
28
  if (app.databaseController) {
25
- return app.databaseController.deleteEverything(fast);
26
- } else {
27
- return Promise.resolve();
29
+ deletePromises.push(app.databaseController.deleteEverything(fast));
30
+ } else if (app.databaseAdapter) {
31
+ _SchemaCache.default.clear();
32
+ deletePromises.push(app.databaseAdapter.deleteAllClasses(fast));
28
33
  }
34
+ return Promise.all(deletePromises);
29
35
  }));
30
36
  }
31
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uL3NyYy9UZXN0VXRpbHMuanMiXSwibmFtZXMiOlsiZGVzdHJveUFsbERhdGFQZXJtYW5lbnRseSIsImZhc3QiLCJwcm9jZXNzIiwiZW52IiwiVEVTVElORyIsIlByb21pc2UiLCJhbGwiLCJPYmplY3QiLCJrZXlzIiwiQXBwQ2FjaGUiLCJjYWNoZSIsIm1hcCIsImFwcElkIiwiYXBwIiwiZ2V0IiwiZGF0YWJhc2VDb250cm9sbGVyIiwiZGVsZXRlRXZlcnl0aGluZyIsInJlc29sdmUiXSwibWFwcGluZ3MiOiI7Ozs7O1FBTWdCQSx5QixHQUFBQSx5Qjs7QUFOaEI7Ozs7OztBQUVBOzs7O0FBSU8sU0FBU0EseUJBQVQsQ0FBbUNDLElBQW5DLEVBQXlDO0FBQzlDLE1BQUksQ0FBQ0MsUUFBUUMsR0FBUixDQUFZQyxPQUFqQixFQUEwQjtBQUN4QixVQUFNLG9DQUFOO0FBQ0Q7QUFDRCxTQUFPQyxRQUFRQyxHQUFSLENBQVlDLE9BQU9DLElBQVAsQ0FBWUMsZ0JBQVNDLEtBQXJCLEVBQTRCQyxHQUE1QixDQUFnQ0MsU0FBUztBQUMxRCxVQUFNQyxNQUFNSixnQkFBU0ssR0FBVCxDQUFhRixLQUFiLENBQVo7QUFDQSxRQUFJQyxJQUFJRSxrQkFBUixFQUE0QjtBQUMxQixhQUFPRixJQUFJRSxrQkFBSixDQUF1QkMsZ0JBQXZCLENBQXdDZixJQUF4QyxDQUFQO0FBQ0QsS0FGRCxNQUVPO0FBQ0wsYUFBT0ksUUFBUVksT0FBUixFQUFQO0FBQ0Q7QUFDRixHQVBrQixDQUFaLENBQVA7QUFRRCIsImZpbGUiOiJUZXN0VXRpbHMuanMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgQXBwQ2FjaGUgZnJvbSAnLi9jYWNoZSc7XG5cbi8qKlxuICogRGVzdHJveXMgYWxsIGRhdGEgaW4gdGhlIGRhdGFiYXNlXG4gKiBAcGFyYW0ge2Jvb2xlYW59IGZhc3Qgc2V0IHRvIHRydWUgaWYgaXQncyBvayB0byBqdXN0IGRyb3Agb2JqZWN0cyBhbmQgbm90IGluZGV4ZXMuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBkZXN0cm95QWxsRGF0YVBlcm1hbmVudGx5KGZhc3QpIHtcbiAgaWYgKCFwcm9jZXNzLmVudi5URVNUSU5HKSB7XG4gICAgdGhyb3cgJ09ubHkgc3VwcG9ydGVkIGluIHRlc3QgZW52aXJvbm1lbnQnO1xuICB9XG4gIHJldHVybiBQcm9taXNlLmFsbChPYmplY3Qua2V5cyhBcHBDYWNoZS5jYWNoZSkubWFwKGFwcElkID0+IHtcbiAgICBjb25zdCBhcHAgPSBBcHBDYWNoZS5nZXQoYXBwSWQpO1xuICAgIGlmIChhcHAuZGF0YWJhc2VDb250cm9sbGVyKSB7XG4gICAgICByZXR1cm4gYXBwLmRhdGFiYXNlQ29udHJvbGxlci5kZWxldGVFdmVyeXRoaW5nKGZhc3QpO1xuICAgIH0gZWxzZSB7XG4gICAgICByZXR1cm4gUHJvbWlzZS5yZXNvbHZlKCk7XG4gICAgfVxuICB9KSk7XG59XG4iXX0=
37
+ function resolvingPromise() {
38
+ let res;
39
+ let rej;
40
+ const promise = new Promise((resolve, reject) => {
41
+ res = resolve;
42
+ rej = reject;
43
+ });
44
+ promise.resolve = res;
45
+ promise.reject = rej;
46
+ return promise;
47
+ }
48
+ function sleep(ms) {
49
+ return new Promise(resolve => setTimeout(resolve, ms));
50
+ }
51
+ function getConnectionsCount(server) {
52
+ return new Promise((resolve, reject) => {
53
+ server.getConnections((err, count) => {
54
+ /* istanbul ignore next */
55
+ if (err) {
56
+ reject(err);
57
+ } else {
58
+ resolve(count);
59
+ }
60
+ });
61
+ });
62
+ }
63
+ ;
64
+ class Connections {
65
+ constructor() {
66
+ this.sockets = new Set();
67
+ }
68
+ track(server) {
69
+ server.on('connection', socket => {
70
+ this.sockets.add(socket);
71
+ socket.on('close', () => {
72
+ this.sockets.delete(socket);
73
+ });
74
+ });
75
+ }
76
+ destroyAll() {
77
+ for (const socket of this.sockets.values()) {
78
+ socket.destroy();
79
+ }
80
+ this.sockets.clear();
81
+ }
82
+ count() {
83
+ return this.sockets.size;
84
+ }
85
+ }
86
+ exports.Connections = Connections;
87
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJfY2FjaGUiLCJfaW50ZXJvcFJlcXVpcmVEZWZhdWx0IiwicmVxdWlyZSIsIl9TY2hlbWFDYWNoZSIsImUiLCJfX2VzTW9kdWxlIiwiZGVmYXVsdCIsImRlc3Ryb3lBbGxEYXRhUGVybWFuZW50bHkiLCJmYXN0IiwicHJvY2VzcyIsImVudiIsIlRFU1RJTkciLCJQcm9taXNlIiwiYWxsIiwiT2JqZWN0Iiwia2V5cyIsIkFwcENhY2hlIiwiY2FjaGUiLCJtYXAiLCJhcHBJZCIsImFwcCIsImdldCIsImRlbGV0ZVByb21pc2VzIiwiY2FjaGVBZGFwdGVyIiwiY2xlYXIiLCJwdXNoIiwiZGF0YWJhc2VDb250cm9sbGVyIiwiZGVsZXRlRXZlcnl0aGluZyIsImRhdGFiYXNlQWRhcHRlciIsIlNjaGVtYUNhY2hlIiwiZGVsZXRlQWxsQ2xhc3NlcyIsInJlc29sdmluZ1Byb21pc2UiLCJyZXMiLCJyZWoiLCJwcm9taXNlIiwicmVzb2x2ZSIsInJlamVjdCIsInNsZWVwIiwibXMiLCJzZXRUaW1lb3V0IiwiZ2V0Q29ubmVjdGlvbnNDb3VudCIsInNlcnZlciIsImdldENvbm5lY3Rpb25zIiwiZXJyIiwiY291bnQiLCJDb25uZWN0aW9ucyIsImNvbnN0cnVjdG9yIiwic29ja2V0cyIsIlNldCIsInRyYWNrIiwib24iLCJzb2NrZXQiLCJhZGQiLCJkZWxldGUiLCJkZXN0cm95QWxsIiwidmFsdWVzIiwiZGVzdHJveSIsInNpemUiLCJleHBvcnRzIl0sInNvdXJjZXMiOlsiLi4vc3JjL1Rlc3RVdGlscy5qcyJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgQXBwQ2FjaGUgZnJvbSAnLi9jYWNoZSc7XG5pbXBvcnQgU2NoZW1hQ2FjaGUgZnJvbSAnLi9BZGFwdGVycy9DYWNoZS9TY2hlbWFDYWNoZSc7XG5cbi8qKlxuICogRGVzdHJveXMgYWxsIGRhdGEgaW4gdGhlIGRhdGFiYXNlXG4gKiBAcGFyYW0ge2Jvb2xlYW59IGZhc3Qgc2V0IHRvIHRydWUgaWYgaXQncyBvayB0byBqdXN0IGRyb3Agb2JqZWN0cyBhbmQgbm90IGluZGV4ZXMuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBkZXN0cm95QWxsRGF0YVBlcm1hbmVudGx5KGZhc3QpIHtcbiAgaWYgKCFwcm9jZXNzLmVudi5URVNUSU5HKSB7XG4gICAgdGhyb3cgJ09ubHkgc3VwcG9ydGVkIGluIHRlc3QgZW52aXJvbm1lbnQnO1xuICB9XG4gIHJldHVybiBQcm9taXNlLmFsbChcbiAgICBPYmplY3Qua2V5cyhBcHBDYWNoZS5jYWNoZSkubWFwKGFwcElkID0+IHtcbiAgICAgIGNvbnN0IGFwcCA9IEFwcENhY2hlLmdldChhcHBJZCk7XG4gICAgICBjb25zdCBkZWxldGVQcm9taXNlcyA9IFtdO1xuICAgICAgaWYgKGFwcC5jYWNoZUFkYXB0ZXIgJiYgYXBwLmNhY2hlQWRhcHRlci5jbGVhcikge1xuICAgICAgICBkZWxldGVQcm9taXNlcy5wdXNoKGFwcC5jYWNoZUFkYXB0ZXIuY2xlYXIoKSk7XG4gICAgICB9XG4gICAgICBpZiAoYXBwLmRhdGFiYXNlQ29udHJvbGxlcikge1xuICAgICAgICBkZWxldGVQcm9taXNlcy5wdXNoKGFwcC5kYXRhYmFzZUNvbnRyb2xsZXIuZGVsZXRlRXZlcnl0aGluZyhmYXN0KSk7XG4gICAgICB9IGVsc2UgaWYgKGFwcC5kYXRhYmFzZUFkYXB0ZXIpIHtcbiAgICAgICAgU2NoZW1hQ2FjaGUuY2xlYXIoKTtcbiAgICAgICAgZGVsZXRlUHJvbWlzZXMucHVzaChhcHAuZGF0YWJhc2VBZGFwdGVyLmRlbGV0ZUFsbENsYXNzZXMoZmFzdCkpO1xuICAgICAgfVxuICAgICAgcmV0dXJuIFByb21pc2UuYWxsKGRlbGV0ZVByb21pc2VzKTtcbiAgICB9KVxuICApO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gcmVzb2x2aW5nUHJvbWlzZSgpIHtcbiAgbGV0IHJlcztcbiAgbGV0IHJlajtcbiAgY29uc3QgcHJvbWlzZSA9IG5ldyBQcm9taXNlKChyZXNvbHZlLCByZWplY3QpID0+IHtcbiAgICByZXMgPSByZXNvbHZlO1xuICAgIHJlaiA9IHJlamVjdDtcbiAgfSk7XG4gIHByb21pc2UucmVzb2x2ZSA9IHJlcztcbiAgcHJvbWlzZS5yZWplY3QgPSByZWo7XG4gIHJldHVybiBwcm9taXNlO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gc2xlZXAobXMpIHtcbiAgcmV0dXJuIG5ldyBQcm9taXNlKChyZXNvbHZlKSA9PiBzZXRUaW1lb3V0KHJlc29sdmUsIG1zKSk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBnZXRDb25uZWN0aW9uc0NvdW50KHNlcnZlcikge1xuICByZXR1cm4gbmV3IFByb21pc2UoKHJlc29sdmUsIHJlamVjdCkgPT4ge1xuICAgIHNlcnZlci5nZXRDb25uZWN0aW9ucygoZXJyLCBjb3VudCkgPT4ge1xuICAgICAgLyogaXN0YW5idWwgaWdub3JlIG5leHQgKi9cbiAgICAgIGlmIChlcnIpIHtcbiAgICAgICAgcmVqZWN0KGVycik7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICByZXNvbHZlKGNvdW50KTtcbiAgICAgIH1cbiAgICB9KTtcbiAgfSk7XG59O1xuXG5leHBvcnQgY2xhc3MgQ29ubmVjdGlvbnMge1xuICBjb25zdHJ1Y3RvcigpIHtcbiAgICB0aGlzLnNvY2tldHMgPSBuZXcgU2V0KCk7XG4gIH1cblxuICB0cmFjayhzZXJ2ZXIpIHtcbiAgICBzZXJ2ZXIub24oJ2Nvbm5lY3Rpb24nLCBzb2NrZXQgPT4ge1xuICAgICAgdGhpcy5zb2NrZXRzLmFkZChzb2NrZXQpO1xuICAgICAgc29ja2V0Lm9uKCdjbG9zZScsICgpID0+IHtcbiAgICAgICAgdGhpcy5zb2NrZXRzLmRlbGV0ZShzb2NrZXQpO1xuICAgICAgfSk7XG4gICAgfSk7XG4gIH1cblxuICBkZXN0cm95QWxsKCkge1xuICAgIGZvciAoY29uc3Qgc29ja2V0IG9mIHRoaXMuc29ja2V0cy52YWx1ZXMoKSkge1xuICAgICAgc29ja2V0LmRlc3Ryb3koKTtcbiAgICB9XG4gICAgdGhpcy5zb2NrZXRzLmNsZWFyKCk7XG4gIH1cblxuICBjb3VudCgpIHtcbiAgICByZXR1cm4gdGhpcy5zb2NrZXRzLnNpemU7XG4gIH1cbn1cblxuIl0sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7O0FBQUEsSUFBQUEsTUFBQSxHQUFBQyxzQkFBQSxDQUFBQyxPQUFBO0FBQ0EsSUFBQUMsWUFBQSxHQUFBRixzQkFBQSxDQUFBQyxPQUFBO0FBQXVELFNBQUFELHVCQUFBRyxDQUFBLFdBQUFBLENBQUEsSUFBQUEsQ0FBQSxDQUFBQyxVQUFBLEdBQUFELENBQUEsS0FBQUUsT0FBQSxFQUFBRixDQUFBO0FBRXZEO0FBQ0E7QUFDQTtBQUNBO0FBQ08sU0FBU0cseUJBQXlCQSxDQUFDQyxJQUFJLEVBQUU7RUFDOUMsSUFBSSxDQUFDQyxPQUFPLENBQUNDLEdBQUcsQ0FBQ0MsT0FBTyxFQUFFO0lBQ3hCLE1BQU0sb0NBQW9DO0VBQzVDO0VBQ0EsT0FBT0MsT0FBTyxDQUFDQyxHQUFHLENBQ2hCQyxNQUFNLENBQUNDLElBQUksQ0FBQ0MsY0FBUSxDQUFDQyxLQUFLLENBQUMsQ0FBQ0MsR0FBRyxDQUFDQyxLQUFLLElBQUk7SUFDdkMsTUFBTUMsR0FBRyxHQUFHSixjQUFRLENBQUNLLEdBQUcsQ0FBQ0YsS0FBSyxDQUFDO0lBQy9CLE1BQU1HLGNBQWMsR0FBRyxFQUFFO0lBQ3pCLElBQUlGLEdBQUcsQ0FBQ0csWUFBWSxJQUFJSCxHQUFHLENBQUNHLFlBQVksQ0FBQ0MsS0FBSyxFQUFFO01BQzlDRixjQUFjLENBQUNHLElBQUksQ0FBQ0wsR0FBRyxDQUFDRyxZQUFZLENBQUNDLEtBQUssQ0FBQyxDQUFDLENBQUM7SUFDL0M7SUFDQSxJQUFJSixHQUFHLENBQUNNLGtCQUFrQixFQUFFO01BQzFCSixjQUFjLENBQUNHLElBQUksQ0FBQ0wsR0FBRyxDQUFDTSxrQkFBa0IsQ0FBQ0MsZ0JBQWdCLENBQUNuQixJQUFJLENBQUMsQ0FBQztJQUNwRSxDQUFDLE1BQU0sSUFBSVksR0FBRyxDQUFDUSxlQUFlLEVBQUU7TUFDOUJDLG9CQUFXLENBQUNMLEtBQUssQ0FBQyxDQUFDO01BQ25CRixjQUFjLENBQUNHLElBQUksQ0FBQ0wsR0FBRyxDQUFDUSxlQUFlLENBQUNFLGdCQUFnQixDQUFDdEIsSUFBSSxDQUFDLENBQUM7SUFDakU7SUFDQSxPQUFPSSxPQUFPLENBQUNDLEdBQUcsQ0FBQ1MsY0FBYyxDQUFDO0VBQ3BDLENBQUMsQ0FDSCxDQUFDO0FBQ0g7QUFFTyxTQUFTUyxnQkFBZ0JBLENBQUEsRUFBRztFQUNqQyxJQUFJQyxHQUFHO0VBQ1AsSUFBSUMsR0FBRztFQUNQLE1BQU1DLE9BQU8sR0FBRyxJQUFJdEIsT0FBTyxDQUFDLENBQUN1QixPQUFPLEVBQUVDLE1BQU0sS0FBSztJQUMvQ0osR0FBRyxHQUFHRyxPQUFPO0lBQ2JGLEdBQUcsR0FBR0csTUFBTTtFQUNkLENBQUMsQ0FBQztFQUNGRixPQUFPLENBQUNDLE9BQU8sR0FBR0gsR0FBRztFQUNyQkUsT0FBTyxDQUFDRSxNQUFNLEdBQUdILEdBQUc7RUFDcEIsT0FBT0MsT0FBTztBQUNoQjtBQUVPLFNBQVNHLEtBQUtBLENBQUNDLEVBQUUsRUFBRTtFQUN4QixPQUFPLElBQUkxQixPQUFPLENBQUV1QixPQUFPLElBQUtJLFVBQVUsQ0FBQ0osT0FBTyxFQUFFRyxFQUFFLENBQUMsQ0FBQztBQUMxRDtBQUVPLFNBQVNFLG1CQUFtQkEsQ0FBQ0MsTUFBTSxFQUFFO0VBQzFDLE9BQU8sSUFBSTdCLE9BQU8sQ0FBQyxDQUFDdUIsT0FBTyxFQUFFQyxNQUFNLEtBQUs7SUFDdENLLE1BQU0sQ0FBQ0MsY0FBYyxDQUFDLENBQUNDLEdBQUcsRUFBRUMsS0FBSyxLQUFLO01BQ3BDO01BQ0EsSUFBSUQsR0FBRyxFQUFFO1FBQ1BQLE1BQU0sQ0FBQ08sR0FBRyxDQUFDO01BQ2IsQ0FBQyxNQUFNO1FBQ0xSLE9BQU8sQ0FBQ1MsS0FBSyxDQUFDO01BQ2hCO0lBQ0YsQ0FBQyxDQUFDO0VBQ0osQ0FBQyxDQUFDO0FBQ0o7QUFBQztBQUVNLE1BQU1DLFdBQVcsQ0FBQztFQUN2QkMsV0FBV0EsQ0FBQSxFQUFHO0lBQ1osSUFBSSxDQUFDQyxPQUFPLEdBQUcsSUFBSUMsR0FBRyxDQUFDLENBQUM7RUFDMUI7RUFFQUMsS0FBS0EsQ0FBQ1IsTUFBTSxFQUFFO0lBQ1pBLE1BQU0sQ0FBQ1MsRUFBRSxDQUFDLFlBQVksRUFBRUMsTUFBTSxJQUFJO01BQ2hDLElBQUksQ0FBQ0osT0FBTyxDQUFDSyxHQUFHLENBQUNELE1BQU0sQ0FBQztNQUN4QkEsTUFBTSxDQUFDRCxFQUFFLENBQUMsT0FBTyxFQUFFLE1BQU07UUFDdkIsSUFBSSxDQUFDSCxPQUFPLENBQUNNLE1BQU0sQ0FBQ0YsTUFBTSxDQUFDO01BQzdCLENBQUMsQ0FBQztJQUNKLENBQUMsQ0FBQztFQUNKO0VBRUFHLFVBQVVBLENBQUEsRUFBRztJQUNYLEtBQUssTUFBTUgsTUFBTSxJQUFJLElBQUksQ0FBQ0osT0FBTyxDQUFDUSxNQUFNLENBQUMsQ0FBQyxFQUFFO01BQzFDSixNQUFNLENBQUNLLE9BQU8sQ0FBQyxDQUFDO0lBQ2xCO0lBQ0EsSUFBSSxDQUFDVCxPQUFPLENBQUN2QixLQUFLLENBQUMsQ0FBQztFQUN0QjtFQUVBb0IsS0FBS0EsQ0FBQSxFQUFHO0lBQ04sT0FBTyxJQUFJLENBQUNHLE9BQU8sQ0FBQ1UsSUFBSTtFQUMxQjtBQUNGO0FBQUNDLE9BQUEsQ0FBQWIsV0FBQSxHQUFBQSxXQUFBIiwiaWdub3JlTGlzdCI6W119