parse-server 4.10.4 → 5.0.0-alpha.12

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 (134) hide show
  1. package/README.md +461 -157
  2. package/lib/AccountLockout.js +23 -2
  3. package/lib/Adapters/AdapterLoader.js +1 -1
  4. package/lib/Adapters/Analytics/AnalyticsAdapter.js +1 -1
  5. package/lib/Adapters/Auth/AuthAdapter.js +1 -1
  6. package/lib/Adapters/Auth/OAuth1Client.js +1 -1
  7. package/lib/Adapters/Auth/facebook.js +110 -10
  8. package/lib/Adapters/Auth/gcenter.js +1 -1
  9. package/lib/Adapters/Auth/gpgames.js +1 -1
  10. package/lib/Adapters/Auth/instagram.js +4 -2
  11. package/lib/Adapters/Auth/keycloak.js +1 -1
  12. package/lib/Adapters/Auth/ldap.js +3 -1
  13. package/lib/Adapters/Auth/oauth2.js +1 -1
  14. package/lib/Adapters/Auth/phantauth.js +1 -1
  15. package/lib/Adapters/Cache/CacheAdapter.js +1 -1
  16. package/lib/Adapters/Cache/RedisCacheAdapter.js +143 -0
  17. package/lib/Adapters/Cache/SchemaCache.js +31 -0
  18. package/lib/Adapters/Email/MailAdapter.js +1 -1
  19. package/lib/Adapters/Files/FilesAdapter.js +1 -1
  20. package/lib/Adapters/Files/GridFSBucketAdapter.js +1 -1
  21. package/lib/Adapters/Files/GridStoreAdapter.js +1 -1
  22. package/lib/Adapters/Logger/LoggerAdapter.js +1 -1
  23. package/lib/Adapters/Logger/WinstonLogger.js +4 -4
  24. package/lib/Adapters/PubSub/EventEmitterPubSub.js +5 -1
  25. package/lib/Adapters/PubSub/PubSubAdapter.js +1 -1
  26. package/lib/Adapters/Push/PushAdapter.js +1 -1
  27. package/lib/Adapters/Storage/Mongo/MongoCollection.js +1 -1
  28. package/lib/Adapters/Storage/Mongo/MongoSchemaCollection.js +21 -3
  29. package/lib/Adapters/Storage/Mongo/MongoStorageAdapter.js +43 -11
  30. package/lib/Adapters/Storage/Mongo/MongoTransform.js +9 -6
  31. package/lib/Adapters/Storage/Postgres/PostgresClient.js +11 -1
  32. package/lib/Adapters/Storage/Postgres/PostgresStorageAdapter.js +107 -62
  33. package/lib/Adapters/WebSocketServer/WSAdapter.js +1 -1
  34. package/lib/Adapters/WebSocketServer/WSSAdapter.js +1 -1
  35. package/lib/Auth.js +2 -39
  36. package/lib/Config.js +201 -8
  37. package/lib/Controllers/AdaptableController.js +1 -9
  38. package/lib/Controllers/CacheController.js +1 -1
  39. package/lib/Controllers/DatabaseController.js +166 -45
  40. package/lib/Controllers/FilesController.js +1 -1
  41. package/lib/Controllers/HooksController.js +2 -2
  42. package/lib/Controllers/LiveQueryController.js +16 -3
  43. package/lib/Controllers/LoggerController.js +1 -1
  44. package/lib/Controllers/ParseGraphQLController.js +2 -2
  45. package/lib/Controllers/PushController.js +1 -1
  46. package/lib/Controllers/SchemaController.js +114 -93
  47. package/lib/Controllers/UserController.js +16 -5
  48. package/lib/Controllers/index.js +10 -11
  49. package/lib/Deprecator/Deprecations.js +28 -0
  50. package/lib/Deprecator/Deprecator.js +135 -0
  51. package/lib/GraphQL/ParseGraphQLSchema.js +71 -39
  52. package/lib/GraphQL/ParseGraphQLServer.js +3 -3
  53. package/lib/GraphQL/loaders/defaultGraphQLMutations.js +2 -2
  54. package/lib/GraphQL/loaders/defaultGraphQLQueries.js +2 -2
  55. package/lib/GraphQL/loaders/defaultGraphQLTypes.js +4 -7
  56. package/lib/GraphQL/loaders/defaultRelaySchema.js +3 -3
  57. package/lib/GraphQL/loaders/filesMutations.js +2 -2
  58. package/lib/GraphQL/loaders/functionsMutations.js +9 -5
  59. package/lib/GraphQL/loaders/parseClassMutations.js +21 -9
  60. package/lib/GraphQL/loaders/parseClassQueries.js +9 -6
  61. package/lib/GraphQL/loaders/parseClassTypes.js +5 -5
  62. package/lib/GraphQL/loaders/schemaDirectives.js +1 -1
  63. package/lib/GraphQL/loaders/schemaMutations.js +8 -6
  64. package/lib/GraphQL/loaders/schemaQueries.js +6 -4
  65. package/lib/GraphQL/loaders/usersMutations.js +65 -7
  66. package/lib/GraphQL/transformers/constraintType.js +2 -2
  67. package/lib/GraphQL/transformers/inputType.js +2 -2
  68. package/lib/GraphQL/transformers/mutation.js +45 -10
  69. package/lib/GraphQL/transformers/outputType.js +2 -2
  70. package/lib/GraphQL/transformers/query.js +2 -2
  71. package/lib/GraphQL/transformers/schemaFields.js +1 -1
  72. package/lib/KeyPromiseQueue.js +59 -0
  73. package/lib/LiveQuery/Client.js +1 -1
  74. package/lib/LiveQuery/Id.js +1 -1
  75. package/lib/LiveQuery/ParseLiveQueryServer.js +144 -38
  76. package/lib/LiveQuery/ParseWebSocketServer.js +2 -2
  77. package/lib/LiveQuery/QueryTools.js +50 -1
  78. package/lib/LiveQuery/equalObjects.js +1 -1
  79. package/lib/Options/Definitions.js +256 -33
  80. package/lib/Options/docs.js +79 -21
  81. package/lib/Options/index.js +3 -1
  82. package/lib/Page.js +53 -0
  83. package/lib/ParseServer.js +40 -16
  84. package/lib/ParseServerRESTController.js +55 -45
  85. package/lib/PromiseRouter.js +7 -20
  86. package/lib/Push/PushQueue.js +1 -1
  87. package/lib/Push/PushWorker.js +2 -2
  88. package/lib/Push/utils.js +1 -1
  89. package/lib/RestQuery.js +44 -8
  90. package/lib/RestWrite.js +60 -10
  91. package/lib/Routers/AggregateRouter.js +23 -18
  92. package/lib/Routers/AudiencesRouter.js +2 -2
  93. package/lib/Routers/ClassesRouter.js +11 -11
  94. package/lib/Routers/CloudCodeRouter.js +1 -1
  95. package/lib/Routers/FeaturesRouter.js +2 -2
  96. package/lib/Routers/FilesRouter.js +34 -7
  97. package/lib/Routers/FunctionsRouter.js +2 -2
  98. package/lib/Routers/GlobalConfigRouter.js +2 -2
  99. package/lib/Routers/GraphQLRouter.js +3 -3
  100. package/lib/Routers/HooksRouter.js +2 -2
  101. package/lib/Routers/LogsRouter.js +2 -2
  102. package/lib/Routers/PagesRouter.js +722 -0
  103. package/lib/Routers/PurgeRouter.js +2 -2
  104. package/lib/Routers/PushRouter.js +3 -3
  105. package/lib/Routers/SchemasRouter.js +43 -16
  106. package/lib/Routers/SecurityRouter.js +47 -0
  107. package/lib/Routers/SessionsRouter.js +4 -2
  108. package/lib/Routers/UsersRouter.js +88 -17
  109. package/lib/SchemaMigrations/DefinedSchemas.js +421 -0
  110. package/lib/SchemaMigrations/Migrations.js +36 -0
  111. package/lib/Security/Check.js +118 -0
  112. package/lib/Security/CheckGroup.js +54 -0
  113. package/lib/Security/CheckGroups/CheckGroupDatabase.js +57 -0
  114. package/lib/Security/CheckGroups/CheckGroupServerConfig.js +82 -0
  115. package/lib/Security/CheckGroups/CheckGroups.js +24 -0
  116. package/lib/Security/CheckRunner.js +236 -0
  117. package/lib/StatusHandler.js +27 -36
  118. package/lib/TestUtils.js +1 -1
  119. package/lib/Utils.js +226 -0
  120. package/lib/batch.js +55 -44
  121. package/lib/cli/utils/commander.js +8 -3
  122. package/lib/cloud-code/HTTPResponse.js +1 -1
  123. package/lib/cloud-code/Parse.Cloud.js +155 -19
  124. package/lib/cloud-code/httpRequest.js +1 -1
  125. package/lib/index.js +10 -12
  126. package/lib/middlewares.js +39 -4
  127. package/lib/rest.js +4 -4
  128. package/lib/triggers.js +134 -121
  129. package/lib/vendor/mongodbUrl.js +8 -10
  130. package/package.json +60 -39
  131. package/CHANGELOG.md +0 -1780
  132. package/lib/Adapters/Cache/RedisCacheAdapter/KeyPromiseQueue.js +0 -59
  133. package/lib/Adapters/Cache/RedisCacheAdapter/index.js +0 -130
  134. package/lib/Controllers/SchemaCache.js +0 -75
@@ -0,0 +1,421 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.DefinedSchemas = void 0;
7
+
8
+ var _logger = require("../logger");
9
+
10
+ var _Config = _interopRequireDefault(require("../Config"));
11
+
12
+ var _SchemasRouter = require("../Routers/SchemasRouter");
13
+
14
+ var _SchemaController = require("../Controllers/SchemaController");
15
+
16
+ var _Options = require("../Options");
17
+
18
+ var Migrations = _interopRequireWildcard(require("./Migrations"));
19
+
20
+ function _getRequireWildcardCache(nodeInterop) { if (typeof WeakMap !== "function") return null; var cacheBabelInterop = new WeakMap(); var cacheNodeInterop = new WeakMap(); return (_getRequireWildcardCache = function (nodeInterop) { return nodeInterop ? cacheNodeInterop : cacheBabelInterop; })(nodeInterop); }
21
+
22
+ function _interopRequireWildcard(obj, nodeInterop) { if (!nodeInterop && obj && obj.__esModule) { return obj; } if (obj === null || typeof obj !== "object" && typeof obj !== "function") { return { default: obj }; } var cache = _getRequireWildcardCache(nodeInterop); if (cache && cache.has(obj)) { return cache.get(obj); } var newObj = {}; var hasPropertyDescriptor = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var key in obj) { if (key !== "default" && Object.prototype.hasOwnProperty.call(obj, key)) { var desc = hasPropertyDescriptor ? Object.getOwnPropertyDescriptor(obj, key) : null; if (desc && (desc.get || desc.set)) { Object.defineProperty(newObj, key, desc); } else { newObj[key] = obj[key]; } } } newObj.default = obj; if (cache) { cache.set(obj, newObj); } return newObj; }
23
+
24
+ function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
25
+
26
+ 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; }
27
+
28
+ 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; }
29
+
30
+ 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; }
31
+
32
+ // -disable-next Cannot resolve module `parse/node`.
33
+ const Parse = require('parse/node');
34
+
35
+ class DefinedSchemas {
36
+ constructor(schemaOptions, config) {
37
+ this.localSchemas = [];
38
+ this.config = _Config.default.get(config.appId);
39
+ this.schemaOptions = schemaOptions;
40
+
41
+ if (schemaOptions && schemaOptions.definitions) {
42
+ if (!Array.isArray(schemaOptions.definitions)) {
43
+ throw `"schema.definitions" must be an array of schemas`;
44
+ }
45
+
46
+ this.localSchemas = schemaOptions.definitions;
47
+ }
48
+
49
+ this.retries = 0;
50
+ this.maxRetries = 3;
51
+ }
52
+
53
+ async saveSchemaToDB(schema) {
54
+ const payload = {
55
+ className: schema.className,
56
+ fields: schema._fields,
57
+ indexes: schema._indexes,
58
+ classLevelPermissions: schema._clp
59
+ };
60
+ await (0, _SchemasRouter.internalCreateSchema)(schema.className, payload, this.config);
61
+ this.resetSchemaOps(schema);
62
+ }
63
+
64
+ resetSchemaOps(schema) {
65
+ // Reset ops like SDK
66
+ schema._fields = {};
67
+ schema._indexes = {};
68
+ } // Simulate update like the SDK
69
+ // We cannot use SDK since routes are disabled
70
+
71
+
72
+ async updateSchemaToDB(schema) {
73
+ const payload = {
74
+ className: schema.className,
75
+ fields: schema._fields,
76
+ indexes: schema._indexes,
77
+ classLevelPermissions: schema._clp
78
+ };
79
+ await (0, _SchemasRouter.internalUpdateSchema)(schema.className, payload, this.config);
80
+ this.resetSchemaOps(schema);
81
+ }
82
+
83
+ async execute() {
84
+ try {
85
+ _logger.logger.info('Running Migrations');
86
+
87
+ if (this.schemaOptions && this.schemaOptions.beforeMigration) {
88
+ await Promise.resolve(this.schemaOptions.beforeMigration());
89
+ }
90
+
91
+ await this.executeMigrations();
92
+
93
+ if (this.schemaOptions && this.schemaOptions.afterMigration) {
94
+ await Promise.resolve(this.schemaOptions.afterMigration());
95
+ }
96
+
97
+ _logger.logger.info('Running Migrations Completed');
98
+ } catch (e) {
99
+ _logger.logger.error(`Failed to run migrations: ${e}`);
100
+
101
+ if (process.env.NODE_ENV === 'production') process.exit(1);
102
+ }
103
+ }
104
+
105
+ async executeMigrations() {
106
+ let timeout = null;
107
+
108
+ try {
109
+ // Set up a time out in production
110
+ // if we fail to get schema
111
+ // pm2 or K8s and many other process managers will try to restart the process
112
+ // after the exit
113
+ if (process.env.NODE_ENV === 'production') {
114
+ timeout = setTimeout(() => {
115
+ _logger.logger.error('Timeout occurred during execution of migrations. Exiting...');
116
+
117
+ process.exit(1);
118
+ }, 20000);
119
+ } // Hack to force session schema to be created
120
+
121
+
122
+ await this.createDeleteSession();
123
+ this.allCloudSchemas = await Parse.Schema.all();
124
+ clearTimeout(timeout);
125
+ await Promise.all(this.localSchemas.map(async localSchema => this.saveOrUpdate(localSchema)));
126
+ this.checkForMissingSchemas();
127
+ await this.enforceCLPForNonProvidedClass();
128
+ } catch (e) {
129
+ if (timeout) clearTimeout(timeout);
130
+
131
+ if (this.retries < this.maxRetries) {
132
+ this.retries++; // first retry 1sec, 2sec, 3sec total 6sec retry sequence
133
+ // retry will only happen in case of deploying multi parse server instance
134
+ // at the same time. Modern systems like k8 avoid this by doing rolling updates
135
+
136
+ await this.wait(1000 * this.retries);
137
+ await this.executeMigrations();
138
+ } else {
139
+ _logger.logger.error(`Failed to run migrations: ${e}`);
140
+
141
+ if (process.env.NODE_ENV === 'production') process.exit(1);
142
+ }
143
+ }
144
+ }
145
+
146
+ checkForMissingSchemas() {
147
+ if (this.schemaOptions.strict !== true) {
148
+ return;
149
+ }
150
+
151
+ const cloudSchemas = this.allCloudSchemas.map(s => s.className);
152
+ const localSchemas = this.localSchemas.map(s => s.className);
153
+ const missingSchemas = cloudSchemas.filter(c => !localSchemas.includes(c) && !_SchemaController.systemClasses.includes(c));
154
+
155
+ if (new Set(localSchemas).size !== localSchemas.length) {
156
+ _logger.logger.error(`The list of schemas provided contains duplicated "className" "${localSchemas.join('","')}"`);
157
+
158
+ process.exit(1);
159
+ }
160
+
161
+ if (this.schemaOptions.strict && missingSchemas.length) {
162
+ _logger.logger.warn(`The following schemas are currently present in the database, but not explicitly defined in a schema: "${missingSchemas.join('", "')}"`);
163
+ }
164
+ } // Required for testing purpose
165
+
166
+
167
+ wait(time) {
168
+ return new Promise(resolve => setTimeout(resolve, time));
169
+ }
170
+
171
+ async enforceCLPForNonProvidedClass() {
172
+ const nonProvidedClasses = this.allCloudSchemas.filter(cloudSchema => !this.localSchemas.some(localSchema => localSchema.className === cloudSchema.className));
173
+ await Promise.all(nonProvidedClasses.map(async schema => {
174
+ const parseSchema = new Parse.Schema(schema.className);
175
+ this.handleCLP(schema, parseSchema);
176
+ await this.updateSchemaToDB(parseSchema);
177
+ }));
178
+ } // Create a fake session since Parse do not create the _Session until
179
+ // a session is created
180
+
181
+
182
+ async createDeleteSession() {
183
+ const session = new Parse.Session();
184
+ await session.save(null, {
185
+ useMasterKey: true
186
+ });
187
+ await session.destroy({
188
+ useMasterKey: true
189
+ });
190
+ }
191
+
192
+ async saveOrUpdate(localSchema) {
193
+ const cloudSchema = this.allCloudSchemas.find(sc => sc.className === localSchema.className);
194
+
195
+ if (cloudSchema) {
196
+ try {
197
+ await this.updateSchema(localSchema, cloudSchema);
198
+ } catch (e) {
199
+ throw `Error during update of schema for type ${cloudSchema.className}: ${e}`;
200
+ }
201
+ } else {
202
+ try {
203
+ await this.saveSchema(localSchema);
204
+ } catch (e) {
205
+ throw `Error while saving Schema for type ${localSchema.className}: ${e}`;
206
+ }
207
+ }
208
+ }
209
+
210
+ async saveSchema(localSchema) {
211
+ const newLocalSchema = new Parse.Schema(localSchema.className);
212
+
213
+ if (localSchema.fields) {
214
+ // Handle fields
215
+ Object.keys(localSchema.fields).filter(fieldName => !this.isProtectedFields(localSchema.className, fieldName)).forEach(fieldName => {
216
+ if (localSchema.fields) {
217
+ const field = localSchema.fields[fieldName];
218
+ this.handleFields(newLocalSchema, fieldName, field);
219
+ }
220
+ });
221
+ } // Handle indexes
222
+
223
+
224
+ if (localSchema.indexes) {
225
+ Object.keys(localSchema.indexes).forEach(indexName => {
226
+ if (localSchema.indexes && !this.isProtectedIndex(localSchema.className, indexName)) {
227
+ newLocalSchema.addIndex(indexName, localSchema.indexes[indexName]);
228
+ }
229
+ });
230
+ }
231
+
232
+ this.handleCLP(localSchema, newLocalSchema);
233
+ return await this.saveSchemaToDB(newLocalSchema);
234
+ }
235
+
236
+ async updateSchema(localSchema, cloudSchema) {
237
+ const newLocalSchema = new Parse.Schema(localSchema.className); // Handle fields
238
+ // Check addition
239
+
240
+ if (localSchema.fields) {
241
+ Object.keys(localSchema.fields).filter(fieldName => !this.isProtectedFields(localSchema.className, fieldName)).forEach(fieldName => {
242
+ // -disable-next
243
+ const field = localSchema.fields[fieldName];
244
+
245
+ if (!cloudSchema.fields[fieldName]) {
246
+ this.handleFields(newLocalSchema, fieldName, field);
247
+ }
248
+ });
249
+ }
250
+
251
+ const fieldsToDelete = [];
252
+ const fieldsToRecreate = [];
253
+ const fieldsWithChangedParams = []; // Check deletion
254
+
255
+ Object.keys(cloudSchema.fields).filter(fieldName => !this.isProtectedFields(localSchema.className, fieldName)).forEach(fieldName => {
256
+ const field = cloudSchema.fields[fieldName];
257
+
258
+ if (!localSchema.fields || !localSchema.fields[fieldName]) {
259
+ fieldsToDelete.push(fieldName);
260
+ return;
261
+ }
262
+
263
+ const localField = localSchema.fields[fieldName]; // Check if field has a changed type
264
+
265
+ if (!this.paramsAreEquals({
266
+ type: field.type,
267
+ targetClass: field.targetClass
268
+ }, {
269
+ type: localField.type,
270
+ targetClass: localField.targetClass
271
+ })) {
272
+ fieldsToRecreate.push({
273
+ fieldName,
274
+ from: {
275
+ type: field.type,
276
+ targetClass: field.targetClass
277
+ },
278
+ to: {
279
+ type: localField.type,
280
+ targetClass: localField.targetClass
281
+ }
282
+ });
283
+ return;
284
+ } // Check if something changed other than the type (like required, defaultValue)
285
+
286
+
287
+ if (!this.paramsAreEquals(field, localField)) {
288
+ fieldsWithChangedParams.push(fieldName);
289
+ }
290
+ });
291
+
292
+ if (this.schemaOptions.deleteExtraFields === true) {
293
+ fieldsToDelete.forEach(fieldName => {
294
+ newLocalSchema.deleteField(fieldName);
295
+ }); // Delete fields from the schema then apply changes
296
+
297
+ await this.updateSchemaToDB(newLocalSchema);
298
+ } else if (this.schemaOptions.strict === true && fieldsToDelete.length) {
299
+ _logger.logger.warn(`The following fields exist in the database for "${localSchema.className}", but are missing in the schema : "${fieldsToDelete.join('" ,"')}"`);
300
+ }
301
+
302
+ if (this.schemaOptions.recreateModifiedFields === true) {
303
+ fieldsToRecreate.forEach(field => {
304
+ newLocalSchema.deleteField(field.fieldName);
305
+ }); // Delete fields from the schema then apply changes
306
+
307
+ await this.updateSchemaToDB(newLocalSchema);
308
+ fieldsToRecreate.forEach(fieldInfo => {
309
+ if (localSchema.fields) {
310
+ const field = localSchema.fields[fieldInfo.fieldName];
311
+ this.handleFields(newLocalSchema, fieldInfo.fieldName, field);
312
+ }
313
+ });
314
+ } else if (this.schemaOptions.strict === true && fieldsToRecreate.length) {
315
+ fieldsToRecreate.forEach(field => {
316
+ const from = field.from.type + (field.from.targetClass ? ` (${field.from.targetClass})` : '');
317
+ const to = field.to.type + (field.to.targetClass ? ` (${field.to.targetClass})` : '');
318
+
319
+ _logger.logger.warn(`The field "${field.fieldName}" type differ between the schema and the database for "${localSchema.className}"; Schema is defined as "${to}" and current database type is "${from}"`);
320
+ });
321
+ }
322
+
323
+ fieldsWithChangedParams.forEach(fieldName => {
324
+ if (localSchema.fields) {
325
+ const field = localSchema.fields[fieldName];
326
+ this.handleFields(newLocalSchema, fieldName, field);
327
+ }
328
+ }); // Handle Indexes
329
+ // Check addition
330
+
331
+ if (localSchema.indexes) {
332
+ Object.keys(localSchema.indexes).forEach(indexName => {
333
+ if ((!cloudSchema.indexes || !cloudSchema.indexes[indexName]) && !this.isProtectedIndex(localSchema.className, indexName)) {
334
+ if (localSchema.indexes) {
335
+ newLocalSchema.addIndex(indexName, localSchema.indexes[indexName]);
336
+ }
337
+ }
338
+ });
339
+ }
340
+
341
+ const indexesToAdd = []; // Check deletion
342
+
343
+ if (cloudSchema.indexes) {
344
+ Object.keys(cloudSchema.indexes).forEach(indexName => {
345
+ if (!this.isProtectedIndex(localSchema.className, indexName)) {
346
+ if (!localSchema.indexes || !localSchema.indexes[indexName]) {
347
+ newLocalSchema.deleteIndex(indexName);
348
+ } else if (!this.paramsAreEquals(localSchema.indexes[indexName], cloudSchema.indexes[indexName])) {
349
+ newLocalSchema.deleteIndex(indexName);
350
+
351
+ if (localSchema.indexes) {
352
+ indexesToAdd.push({
353
+ indexName,
354
+ index: localSchema.indexes[indexName]
355
+ });
356
+ }
357
+ }
358
+ }
359
+ });
360
+ }
361
+
362
+ this.handleCLP(localSchema, newLocalSchema, cloudSchema); // Apply changes
363
+
364
+ await this.updateSchemaToDB(newLocalSchema); // Apply new/changed indexes
365
+
366
+ if (indexesToAdd.length) {
367
+ _logger.logger.debug(`Updating indexes for "${newLocalSchema.className}" : ${indexesToAdd.join(' ,')}`);
368
+
369
+ indexesToAdd.forEach(o => newLocalSchema.addIndex(o.indexName, o.index));
370
+ await this.updateSchemaToDB(newLocalSchema);
371
+ }
372
+ }
373
+
374
+ handleCLP(localSchema, newLocalSchema, cloudSchema) {
375
+ if (!localSchema.classLevelPermissions && !cloudSchema) {
376
+ _logger.logger.warn(`classLevelPermissions not provided for ${localSchema.className}.`);
377
+ } // Use spread to avoid read only issue (encountered by Moumouls using directAccess)
378
+
379
+
380
+ const clp = _objectSpread({}, localSchema.classLevelPermissions) || {}; // To avoid inconsistency we need to remove all rights on addField
381
+
382
+ clp.addField = {};
383
+ newLocalSchema.setCLP(clp);
384
+ }
385
+
386
+ isProtectedFields(className, fieldName) {
387
+ return !!_SchemaController.defaultColumns._Default[fieldName] || !!(_SchemaController.defaultColumns[className] && _SchemaController.defaultColumns[className][fieldName]);
388
+ }
389
+
390
+ isProtectedIndex(className, indexName) {
391
+ let indexes = ['_id_'];
392
+
393
+ if (className === '_User') {
394
+ indexes = [...indexes, 'case_insensitive_username', 'case_insensitive_email', 'username_1', 'email_1'];
395
+ }
396
+
397
+ return indexes.indexOf(indexName) !== -1;
398
+ }
399
+
400
+ paramsAreEquals(objA, objB) {
401
+ const keysA = Object.keys(objA);
402
+ const keysB = Object.keys(objB); // Check key name
403
+
404
+ if (keysA.length !== keysB.length) return false;
405
+ return keysA.every(k => objA[k] === objB[k]);
406
+ }
407
+
408
+ handleFields(newLocalSchema, fieldName, field) {
409
+ if (field.type === 'Relation') {
410
+ newLocalSchema.addRelation(fieldName, field.targetClass);
411
+ } else if (field.type === 'Pointer') {
412
+ newLocalSchema.addPointer(fieldName, field.targetClass, field);
413
+ } else {
414
+ newLocalSchema.addField(fieldName, field.type, field);
415
+ }
416
+ }
417
+
418
+ }
419
+
420
+ exports.DefinedSchemas = DefinedSchemas;
421
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/SchemaMigrations/DefinedSchemas.js"],"names":["Parse","require","DefinedSchemas","constructor","schemaOptions","config","localSchemas","Config","get","appId","definitions","Array","isArray","retries","maxRetries","saveSchemaToDB","schema","payload","className","fields","_fields","indexes","_indexes","classLevelPermissions","_clp","resetSchemaOps","updateSchemaToDB","execute","logger","info","beforeMigration","Promise","resolve","executeMigrations","afterMigration","e","error","process","env","NODE_ENV","exit","timeout","setTimeout","createDeleteSession","allCloudSchemas","Schema","all","clearTimeout","map","localSchema","saveOrUpdate","checkForMissingSchemas","enforceCLPForNonProvidedClass","wait","strict","cloudSchemas","s","missingSchemas","filter","c","includes","systemClasses","Set","size","length","join","warn","time","nonProvidedClasses","cloudSchema","some","parseSchema","handleCLP","session","Session","save","useMasterKey","destroy","find","sc","updateSchema","saveSchema","newLocalSchema","Object","keys","fieldName","isProtectedFields","forEach","field","handleFields","indexName","isProtectedIndex","addIndex","fieldsToDelete","fieldsToRecreate","fieldsWithChangedParams","push","localField","paramsAreEquals","type","targetClass","from","to","deleteExtraFields","deleteField","recreateModifiedFields","fieldInfo","indexesToAdd","deleteIndex","index","debug","o","clp","addField","setCLP","defaultColumns","_Default","indexOf","objA","objB","keysA","keysB","every","k","addRelation","addPointer"],"mappings":"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","sourcesContent":["// @flow\n// @flow-disable-next Cannot resolve module `parse/node`.\nconst Parse = require('parse/node');\nimport { logger } from '../logger';\nimport Config from '../Config';\nimport { internalCreateSchema, internalUpdateSchema } from '../Routers/SchemasRouter';\nimport { defaultColumns, systemClasses } from '../Controllers/SchemaController';\nimport { ParseServerOptions } from '../Options';\nimport * as Migrations from './Migrations';\n\nexport class DefinedSchemas {\n  config: ParseServerOptions;\n  schemaOptions: Migrations.SchemaOptions;\n  localSchemas: Migrations.JSONSchema[];\n  retries: number;\n  maxRetries: number;\n  allCloudSchemas: Parse.Schema[];\n\n  constructor(schemaOptions: Migrations.SchemaOptions, config: ParseServerOptions) {\n    this.localSchemas = [];\n    this.config = Config.get(config.appId);\n    this.schemaOptions = schemaOptions;\n    if (schemaOptions && schemaOptions.definitions) {\n      if (!Array.isArray(schemaOptions.definitions)) {\n        throw `\"schema.definitions\" must be an array of schemas`;\n      }\n\n      this.localSchemas = schemaOptions.definitions;\n    }\n\n    this.retries = 0;\n    this.maxRetries = 3;\n  }\n\n  async saveSchemaToDB(schema: Parse.Schema): Promise<void> {\n    const payload = {\n      className: schema.className,\n      fields: schema._fields,\n      indexes: schema._indexes,\n      classLevelPermissions: schema._clp,\n    };\n    await internalCreateSchema(schema.className, payload, this.config);\n    this.resetSchemaOps(schema);\n  }\n\n  resetSchemaOps(schema: Parse.Schema) {\n    // Reset ops like SDK\n    schema._fields = {};\n    schema._indexes = {};\n  }\n\n  // Simulate update like the SDK\n  // We cannot use SDK since routes are disabled\n  async updateSchemaToDB(schema: Parse.Schema) {\n    const payload = {\n      className: schema.className,\n      fields: schema._fields,\n      indexes: schema._indexes,\n      classLevelPermissions: schema._clp,\n    };\n    await internalUpdateSchema(schema.className, payload, this.config);\n    this.resetSchemaOps(schema);\n  }\n\n  async execute() {\n    try {\n      logger.info('Running Migrations');\n      if (this.schemaOptions && this.schemaOptions.beforeMigration) {\n        await Promise.resolve(this.schemaOptions.beforeMigration());\n      }\n\n      await this.executeMigrations();\n\n      if (this.schemaOptions && this.schemaOptions.afterMigration) {\n        await Promise.resolve(this.schemaOptions.afterMigration());\n      }\n\n      logger.info('Running Migrations Completed');\n    } catch (e) {\n      logger.error(`Failed to run migrations: ${e}`);\n      if (process.env.NODE_ENV === 'production') process.exit(1);\n    }\n  }\n\n  async executeMigrations() {\n    let timeout = null;\n    try {\n      // Set up a time out in production\n      // if we fail to get schema\n      // pm2 or K8s and many other process managers will try to restart the process\n      // after the exit\n      if (process.env.NODE_ENV === 'production') {\n        timeout = setTimeout(() => {\n          logger.error('Timeout occurred during execution of migrations. Exiting...');\n          process.exit(1);\n        }, 20000);\n      }\n\n      // Hack to force session schema to be created\n      await this.createDeleteSession();\n      this.allCloudSchemas = await Parse.Schema.all();\n      clearTimeout(timeout);\n      await Promise.all(this.localSchemas.map(async localSchema => this.saveOrUpdate(localSchema)));\n\n      this.checkForMissingSchemas();\n      await this.enforceCLPForNonProvidedClass();\n    } catch (e) {\n      if (timeout) clearTimeout(timeout);\n      if (this.retries < this.maxRetries) {\n        this.retries++;\n        // first retry 1sec, 2sec, 3sec total 6sec retry sequence\n        // retry will only happen in case of deploying multi parse server instance\n        // at the same time. Modern systems like k8 avoid this by doing rolling updates\n        await this.wait(1000 * this.retries);\n        await this.executeMigrations();\n      } else {\n        logger.error(`Failed to run migrations: ${e}`);\n        if (process.env.NODE_ENV === 'production') process.exit(1);\n      }\n    }\n  }\n\n  checkForMissingSchemas() {\n    if (this.schemaOptions.strict !== true) {\n      return;\n    }\n\n    const cloudSchemas = this.allCloudSchemas.map(s => s.className);\n    const localSchemas = this.localSchemas.map(s => s.className);\n    const missingSchemas = cloudSchemas.filter(\n      c => !localSchemas.includes(c) && !systemClasses.includes(c)\n    );\n\n    if (new Set(localSchemas).size !== localSchemas.length) {\n      logger.error(\n        `The list of schemas provided contains duplicated \"className\"  \"${localSchemas.join(\n          '\",\"'\n        )}\"`\n      );\n      process.exit(1);\n    }\n\n    if (this.schemaOptions.strict && missingSchemas.length) {\n      logger.warn(\n        `The following schemas are currently present in the database, but not explicitly defined in a schema: \"${missingSchemas.join(\n          '\", \"'\n        )}\"`\n      );\n    }\n  }\n\n  // Required for testing purpose\n  wait(time: number) {\n    return new Promise<void>(resolve => setTimeout(resolve, time));\n  }\n\n  async enforceCLPForNonProvidedClass(): Promise<void> {\n    const nonProvidedClasses = this.allCloudSchemas.filter(\n      cloudSchema =>\n        !this.localSchemas.some(localSchema => localSchema.className === cloudSchema.className)\n    );\n    await Promise.all(\n      nonProvidedClasses.map(async schema => {\n        const parseSchema = new Parse.Schema(schema.className);\n        this.handleCLP(schema, parseSchema);\n        await this.updateSchemaToDB(parseSchema);\n      })\n    );\n  }\n\n  // Create a fake session since Parse do not create the _Session until\n  // a session is created\n  async createDeleteSession() {\n    const session = new Parse.Session();\n    await session.save(null, { useMasterKey: true });\n    await session.destroy({ useMasterKey: true });\n  }\n\n  async saveOrUpdate(localSchema: Migrations.JSONSchema) {\n    const cloudSchema = this.allCloudSchemas.find(sc => sc.className === localSchema.className);\n    if (cloudSchema) {\n      try {\n        await this.updateSchema(localSchema, cloudSchema);\n      } catch (e) {\n        throw `Error during update of schema for type ${cloudSchema.className}: ${e}`;\n      }\n    } else {\n      try {\n        await this.saveSchema(localSchema);\n      } catch (e) {\n        throw `Error while saving Schema for type ${localSchema.className}: ${e}`;\n      }\n    }\n  }\n\n  async saveSchema(localSchema: Migrations.JSONSchema) {\n    const newLocalSchema = new Parse.Schema(localSchema.className);\n    if (localSchema.fields) {\n      // Handle fields\n      Object.keys(localSchema.fields)\n        .filter(fieldName => !this.isProtectedFields(localSchema.className, fieldName))\n        .forEach(fieldName => {\n          if (localSchema.fields) {\n            const field = localSchema.fields[fieldName];\n            this.handleFields(newLocalSchema, fieldName, field);\n          }\n        });\n    }\n    // Handle indexes\n    if (localSchema.indexes) {\n      Object.keys(localSchema.indexes).forEach(indexName => {\n        if (localSchema.indexes && !this.isProtectedIndex(localSchema.className, indexName)) {\n          newLocalSchema.addIndex(indexName, localSchema.indexes[indexName]);\n        }\n      });\n    }\n\n    this.handleCLP(localSchema, newLocalSchema);\n\n    return await this.saveSchemaToDB(newLocalSchema);\n  }\n\n  async updateSchema(localSchema: Migrations.JSONSchema, cloudSchema: Parse.Schema) {\n    const newLocalSchema = new Parse.Schema(localSchema.className);\n\n    // Handle fields\n    // Check addition\n    if (localSchema.fields) {\n      Object.keys(localSchema.fields)\n        .filter(fieldName => !this.isProtectedFields(localSchema.className, fieldName))\n        .forEach(fieldName => {\n          // @flow-disable-next\n          const field = localSchema.fields[fieldName];\n          if (!cloudSchema.fields[fieldName]) {\n            this.handleFields(newLocalSchema, fieldName, field);\n          }\n        });\n    }\n\n    const fieldsToDelete: string[] = [];\n    const fieldsToRecreate: {\n      fieldName: string,\n      from: { type: string, targetClass?: string },\n      to: { type: string, targetClass?: string },\n    }[] = [];\n    const fieldsWithChangedParams: string[] = [];\n\n    // Check deletion\n    Object.keys(cloudSchema.fields)\n      .filter(fieldName => !this.isProtectedFields(localSchema.className, fieldName))\n      .forEach(fieldName => {\n        const field = cloudSchema.fields[fieldName];\n        if (!localSchema.fields || !localSchema.fields[fieldName]) {\n          fieldsToDelete.push(fieldName);\n          return;\n        }\n\n        const localField = localSchema.fields[fieldName];\n        // Check if field has a changed type\n        if (\n          !this.paramsAreEquals(\n            { type: field.type, targetClass: field.targetClass },\n            { type: localField.type, targetClass: localField.targetClass }\n          )\n        ) {\n          fieldsToRecreate.push({\n            fieldName,\n            from: { type: field.type, targetClass: field.targetClass },\n            to: { type: localField.type, targetClass: localField.targetClass },\n          });\n          return;\n        }\n\n        // Check if something changed other than the type (like required, defaultValue)\n        if (!this.paramsAreEquals(field, localField)) {\n          fieldsWithChangedParams.push(fieldName);\n        }\n      });\n\n    if (this.schemaOptions.deleteExtraFields === true) {\n      fieldsToDelete.forEach(fieldName => {\n        newLocalSchema.deleteField(fieldName);\n      });\n\n      // Delete fields from the schema then apply changes\n      await this.updateSchemaToDB(newLocalSchema);\n    } else if (this.schemaOptions.strict === true && fieldsToDelete.length) {\n      logger.warn(\n        `The following fields exist in the database for \"${\n          localSchema.className\n        }\", but are missing in the schema : \"${fieldsToDelete.join('\" ,\"')}\"`\n      );\n    }\n\n    if (this.schemaOptions.recreateModifiedFields === true) {\n      fieldsToRecreate.forEach(field => {\n        newLocalSchema.deleteField(field.fieldName);\n      });\n\n      // Delete fields from the schema then apply changes\n      await this.updateSchemaToDB(newLocalSchema);\n\n      fieldsToRecreate.forEach(fieldInfo => {\n        if (localSchema.fields) {\n          const field = localSchema.fields[fieldInfo.fieldName];\n          this.handleFields(newLocalSchema, fieldInfo.fieldName, field);\n        }\n      });\n    } else if (this.schemaOptions.strict === true && fieldsToRecreate.length) {\n      fieldsToRecreate.forEach(field => {\n        const from =\n          field.from.type + (field.from.targetClass ? ` (${field.from.targetClass})` : '');\n        const to = field.to.type + (field.to.targetClass ? ` (${field.to.targetClass})` : '');\n\n        logger.warn(\n          `The field \"${field.fieldName}\" type differ between the schema and the database for \"${localSchema.className}\"; Schema is defined as \"${to}\" and current database type is \"${from}\"`\n        );\n      });\n    }\n\n    fieldsWithChangedParams.forEach(fieldName => {\n      if (localSchema.fields) {\n        const field = localSchema.fields[fieldName];\n        this.handleFields(newLocalSchema, fieldName, field);\n      }\n    });\n\n    // Handle Indexes\n    // Check addition\n    if (localSchema.indexes) {\n      Object.keys(localSchema.indexes).forEach(indexName => {\n        if (\n          (!cloudSchema.indexes || !cloudSchema.indexes[indexName]) &&\n          !this.isProtectedIndex(localSchema.className, indexName)\n        ) {\n          if (localSchema.indexes) {\n            newLocalSchema.addIndex(indexName, localSchema.indexes[indexName]);\n          }\n        }\n      });\n    }\n\n    const indexesToAdd = [];\n\n    // Check deletion\n    if (cloudSchema.indexes) {\n      Object.keys(cloudSchema.indexes).forEach(indexName => {\n        if (!this.isProtectedIndex(localSchema.className, indexName)) {\n          if (!localSchema.indexes || !localSchema.indexes[indexName]) {\n            newLocalSchema.deleteIndex(indexName);\n          } else if (\n            !this.paramsAreEquals(localSchema.indexes[indexName], cloudSchema.indexes[indexName])\n          ) {\n            newLocalSchema.deleteIndex(indexName);\n            if (localSchema.indexes) {\n              indexesToAdd.push({\n                indexName,\n                index: localSchema.indexes[indexName],\n              });\n            }\n          }\n        }\n      });\n    }\n\n    this.handleCLP(localSchema, newLocalSchema, cloudSchema);\n    // Apply changes\n    await this.updateSchemaToDB(newLocalSchema);\n    // Apply new/changed indexes\n    if (indexesToAdd.length) {\n      logger.debug(\n        `Updating indexes for \"${newLocalSchema.className}\" :  ${indexesToAdd.join(' ,')}`\n      );\n      indexesToAdd.forEach(o => newLocalSchema.addIndex(o.indexName, o.index));\n      await this.updateSchemaToDB(newLocalSchema);\n    }\n  }\n\n  handleCLP(\n    localSchema: Migrations.JSONSchema,\n    newLocalSchema: Parse.Schema,\n    cloudSchema: Parse.Schema\n  ) {\n    if (!localSchema.classLevelPermissions && !cloudSchema) {\n      logger.warn(`classLevelPermissions not provided for ${localSchema.className}.`);\n    }\n    // Use spread to avoid read only issue (encountered by Moumouls using directAccess)\n    const clp = ({ ...localSchema.classLevelPermissions } || {}: Parse.CLP.PermissionsMap);\n    // To avoid inconsistency we need to remove all rights on addField\n    clp.addField = {};\n    newLocalSchema.setCLP(clp);\n  }\n\n  isProtectedFields(className: string, fieldName: string) {\n    return (\n      !!defaultColumns._Default[fieldName] ||\n      !!(defaultColumns[className] && defaultColumns[className][fieldName])\n    );\n  }\n\n  isProtectedIndex(className: string, indexName: string) {\n    let indexes = ['_id_'];\n    if (className === '_User') {\n      indexes = [\n        ...indexes,\n        'case_insensitive_username',\n        'case_insensitive_email',\n        'username_1',\n        'email_1',\n      ];\n    }\n\n    return indexes.indexOf(indexName) !== -1;\n  }\n\n  paramsAreEquals<T: { [key: string]: any }>(objA: T, objB: T) {\n    const keysA: string[] = Object.keys(objA);\n    const keysB: string[] = Object.keys(objB);\n\n    // Check key name\n    if (keysA.length !== keysB.length) return false;\n    return keysA.every(k => objA[k] === objB[k]);\n  }\n\n  handleFields(newLocalSchema: Parse.Schema, fieldName: string, field: Migrations.FieldType) {\n    if (field.type === 'Relation') {\n      newLocalSchema.addRelation(fieldName, field.targetClass);\n    } else if (field.type === 'Pointer') {\n      newLocalSchema.addPointer(fieldName, field.targetClass, field);\n    } else {\n      newLocalSchema.addField(fieldName, field.type, field);\n    }\n  }\n}\n"]}
@@ -0,0 +1,36 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.makeSchema = makeSchema;
7
+ exports.CLP = void 0;
8
+
9
+ // @Typescript 4.1+ // type CLPPermission = 'requiresAuthentication' | '*' | `user:${string}` | `role:${string}`
10
+ class CLP {
11
+ static allow(perms) {
12
+ const out = {};
13
+
14
+ for (const [perm, ops] of Object.entries(perms)) {
15
+ // -disable-next Property `@@iterator` is missing in mixed [1] but exists in `$Iterable` [2].
16
+ for (const op of ops) {
17
+ out[op] = out[op] || {};
18
+ out[op][perm] = true;
19
+ }
20
+ }
21
+
22
+ return out;
23
+ }
24
+
25
+ }
26
+
27
+ exports.CLP = CLP;
28
+
29
+ function makeSchema(className, schema) {
30
+ // This function solve two things:
31
+ // 1. It provides auto-completion to the users who are implementing schemas
32
+ // 2. It allows forward-compatible point in order to allow future changes to the internal structure of JSONSchema without affecting all the users
33
+ schema.className = className;
34
+ return schema;
35
+ }
36
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9TY2hlbWFNaWdyYXRpb25zL01pZ3JhdGlvbnMuanMiXSwibmFtZXMiOlsiQ0xQIiwiYWxsb3ciLCJwZXJtcyIsIm91dCIsInBlcm0iLCJvcHMiLCJPYmplY3QiLCJlbnRyaWVzIiwib3AiLCJtYWtlU2NoZW1hIiwiY2xhc3NOYW1lIiwic2NoZW1hIl0sIm1hcHBpbmdzIjoiOzs7Ozs7OztBQWlEQTtBQXNCTyxNQUFNQSxHQUFOLENBQVU7QUFDSCxTQUFMQyxLQUFLLENBQUNDLEtBQUQsRUFBa0Q7QUFDNUQsVUFBTUMsR0FBRyxHQUFHLEVBQVo7O0FBRUEsU0FBSyxNQUFNLENBQUNDLElBQUQsRUFBT0MsR0FBUCxDQUFYLElBQTBCQyxNQUFNLENBQUNDLE9BQVAsQ0FBZUwsS0FBZixDQUExQixFQUFpRDtBQUMvQztBQUNBLFdBQUssTUFBTU0sRUFBWCxJQUFpQkgsR0FBakIsRUFBc0I7QUFDcEJGLFFBQUFBLEdBQUcsQ0FBQ0ssRUFBRCxDQUFILEdBQVVMLEdBQUcsQ0FBQ0ssRUFBRCxDQUFILElBQVcsRUFBckI7QUFDQUwsUUFBQUEsR0FBRyxDQUFDSyxFQUFELENBQUgsQ0FBUUosSUFBUixJQUFnQixJQUFoQjtBQUNEO0FBQ0Y7O0FBRUQsV0FBT0QsR0FBUDtBQUNEOztBQWJjOzs7O0FBZ0JWLFNBQVNNLFVBQVQsQ0FBb0JDLFNBQXBCLEVBQThDQyxNQUE5QyxFQUE4RTtBQUNuRjtBQUNBO0FBQ0E7QUFDQUEsRUFBQUEsTUFBTSxDQUFDRCxTQUFQLEdBQW1CQSxTQUFuQjtBQUVBLFNBQU9DLE1BQVA7QUFDRCIsInNvdXJjZXNDb250ZW50IjpbIi8vIEBmbG93XG5cbmV4cG9ydCB0eXBlIEZpZWxkVmFsdWVUeXBlID1cbiAgfCAnU3RyaW5nJ1xuICB8ICdCb29sZWFuJ1xuICB8ICdGaWxlJ1xuICB8ICdOdW1iZXInXG4gIHwgJ1JlbGF0aW9uJ1xuICB8ICdQb2ludGVyJ1xuICB8ICdEYXRlJ1xuICB8ICdHZW9Qb2ludCdcbiAgfCAnUG9seWdvbidcbiAgfCAnQXJyYXknXG4gIHwgJ09iamVjdCdcbiAgfCAnQUNMJztcblxuZXhwb3J0IGludGVyZmFjZSBGaWVsZFR5cGUge1xuICB0eXBlOiBGaWVsZFZhbHVlVHlwZTtcbiAgcmVxdWlyZWQ/OiBib29sZWFuO1xuICBkZWZhdWx0VmFsdWU/OiBtaXhlZDtcbiAgdGFyZ2V0Q2xhc3M/OiBzdHJpbmc7XG59XG5cbnR5cGUgQ2xhc3NOYW1lVHlwZSA9ICdfVXNlcicgfCAnX1JvbGUnIHwgc3RyaW5nO1xuXG5leHBvcnQgaW50ZXJmYWNlIFByb3RlY3RlZEZpZWxkc0ludGVyZmFjZSB7XG4gIFtrZXk6IHN0cmluZ106IHN0cmluZ1tdO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIEluZGV4SW50ZXJmYWNlIHtcbiAgW2tleTogc3RyaW5nXTogbnVtYmVyO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIEluZGV4ZXNJbnRlcmZhY2Uge1xuICBba2V5OiBzdHJpbmddOiBJbmRleEludGVyZmFjZTtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBTY2hlbWFPcHRpb25zIHtcbiAgZGVmaW5pdGlvbnM6IEpTT05TY2hlbWFbXTtcbiAgc3RyaWN0OiA/Ym9vbGVhbjtcbiAgZGVsZXRlRXh0cmFGaWVsZHM6ID9ib29sZWFuO1xuICByZWNyZWF0ZU1vZGlmaWVkRmllbGRzOiA/Ym9vbGVhbjtcbiAgbG9ja1NjaGVtYXM6ID9ib29sZWFuO1xuICAvKiBDYWxsYmFjayB3aGVuIHNlcnZlciBoYXMgc3RhcnRlZCBhbmQgYmVmb3JlIHJ1bm5pbmcgc2NoZW1hcyBtaWdyYXRpb24gb3BlcmF0aW9ucyBpZiBzY2hlbWFzIGtleSBwcm92aWRlZCAqL1xuICBiZWZvcmVNaWdyYXRpb246ID8oKSA9PiB2b2lkIHwgUHJvbWlzZTx2b2lkPjtcbiAgYWZ0ZXJNaWdyYXRpb246ID8oKSA9PiB2b2lkIHwgUHJvbWlzZTx2b2lkPjtcbn1cblxuZXhwb3J0IHR5cGUgQ0xQT3BlcmF0aW9uID0gJ2ZpbmQnIHwgJ2NvdW50JyB8ICdnZXQnIHwgJ3VwZGF0ZScgfCAnY3JlYXRlJyB8ICdkZWxldGUnO1xuLy8gQFR5cGVzY3JpcHQgNC4xKyAvLyB0eXBlIENMUFBlcm1pc3Npb24gPSAncmVxdWlyZXNBdXRoZW50aWNhdGlvbicgfCAnKicgfCAgYHVzZXI6JHtzdHJpbmd9YCB8IGByb2xlOiR7c3RyaW5nfWBcblxudHlwZSBDTFBWYWx1ZSA9IHsgW2tleTogc3RyaW5nXTogYm9vbGVhbiB9O1xudHlwZSBDTFBEYXRhID0geyBba2V5OiBzdHJpbmddOiBDTFBPcGVyYXRpb25bXSB9O1xudHlwZSBDTFBJbnRlcmZhY2UgPSB7IFtrZXk6IHN0cmluZ106IENMUFZhbHVlIH07XG5cbmV4cG9ydCBpbnRlcmZhY2UgSlNPTlNjaGVtYSB7XG4gIGNsYXNzTmFtZTogQ2xhc3NOYW1lVHlwZTtcbiAgZmllbGRzPzogeyBba2V5OiBzdHJpbmddOiBGaWVsZFR5cGUgfTtcbiAgaW5kZXhlcz86IEluZGV4ZXNJbnRlcmZhY2U7XG4gIGNsYXNzTGV2ZWxQZXJtaXNzaW9ucz86IHtcbiAgICBmaW5kPzogQ0xQVmFsdWUsXG4gICAgY291bnQ/OiBDTFBWYWx1ZSxcbiAgICBnZXQ/OiBDTFBWYWx1ZSxcbiAgICB1cGRhdGU/OiBDTFBWYWx1ZSxcbiAgICBjcmVhdGU/OiBDTFBWYWx1ZSxcbiAgICBkZWxldGU/OiBDTFBWYWx1ZSxcbiAgICBhZGRGaWVsZD86IENMUFZhbHVlLFxuICAgIHByb3RlY3RlZEZpZWxkcz86IFByb3RlY3RlZEZpZWxkc0ludGVyZmFjZSxcbiAgfTtcbn1cblxuZXhwb3J0IGNsYXNzIENMUCB7XG4gIHN0YXRpYyBhbGxvdyhwZXJtczogeyBba2V5OiBzdHJpbmddOiBDTFBEYXRhIH0pOiBDTFBJbnRlcmZhY2Uge1xuICAgIGNvbnN0IG91dCA9IHt9O1xuXG4gICAgZm9yIChjb25zdCBbcGVybSwgb3BzXSBvZiBPYmplY3QuZW50cmllcyhwZXJtcykpIHtcbiAgICAgIC8vIEBmbG93LWRpc2FibGUtbmV4dCBQcm9wZXJ0eSBgQEBpdGVyYXRvcmAgaXMgbWlzc2luZyBpbiBtaXhlZCBbMV0gYnV0IGV4aXN0cyBpbiBgJEl0ZXJhYmxlYCBbMl0uXG4gICAgICBmb3IgKGNvbnN0IG9wIG9mIG9wcykge1xuICAgICAgICBvdXRbb3BdID0gb3V0W29wXSB8fCB7fTtcbiAgICAgICAgb3V0W29wXVtwZXJtXSA9IHRydWU7XG4gICAgICB9XG4gICAgfVxuXG4gICAgcmV0dXJuIG91dDtcbiAgfVxufVxuXG5leHBvcnQgZnVuY3Rpb24gbWFrZVNjaGVtYShjbGFzc05hbWU6IENsYXNzTmFtZVR5cGUsIHNjaGVtYTogSlNPTlNjaGVtYSk6IEpTT05TY2hlbWEge1xuICAvLyBUaGlzIGZ1bmN0aW9uIHNvbHZlIHR3byB0aGluZ3M6XG4gIC8vIDEuIEl0IHByb3ZpZGVzIGF1dG8tY29tcGxldGlvbiB0byB0aGUgdXNlcnMgd2hvIGFyZSBpbXBsZW1lbnRpbmcgc2NoZW1hc1xuICAvLyAyLiBJdCBhbGxvd3MgZm9yd2FyZC1jb21wYXRpYmxlIHBvaW50IGluIG9yZGVyIHRvIGFsbG93IGZ1dHVyZSBjaGFuZ2VzIHRvIHRoZSBpbnRlcm5hbCBzdHJ1Y3R1cmUgb2YgSlNPTlNjaGVtYSB3aXRob3V0IGFmZmVjdGluZyBhbGwgdGhlIHVzZXJzXG4gIHNjaGVtYS5jbGFzc05hbWUgPSBjbGFzc05hbWU7XG5cbiAgcmV0dXJuIHNjaGVtYTtcbn1cbiJdfQ==
@@ -0,0 +1,118 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.default = void 0;
7
+
8
+ var _Utils = _interopRequireDefault(require("../Utils"));
9
+
10
+ var _lodash = require("lodash");
11
+
12
+ function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
13
+
14
+ /**
15
+ * @module SecurityCheck
16
+ */
17
+
18
+ /**
19
+ * A security check.
20
+ * @class Check
21
+ */
22
+ class Check {
23
+ /**
24
+ * Constructs a new security check.
25
+ * @param {Object} params The parameters.
26
+ * @param {String} params.title The title.
27
+ * @param {String} params.warning The warning message if the check fails.
28
+ * @param {String} params.solution The solution to fix the check.
29
+ * @param {Promise} params.check The check as synchronous or asynchronous function.
30
+ */
31
+ constructor(params) {
32
+ this._validateParams(params);
33
+
34
+ const {
35
+ title,
36
+ warning,
37
+ solution,
38
+ check
39
+ } = params;
40
+ this.title = title;
41
+ this.warning = warning;
42
+ this.solution = solution;
43
+ this.check = check; // Set default properties
44
+
45
+ this._checkState = CheckState.none;
46
+ this.error;
47
+ }
48
+ /**
49
+ * Returns the current check state.
50
+ * @return {CheckState} The check state.
51
+ */
52
+
53
+
54
+ checkState() {
55
+ return this._checkState;
56
+ }
57
+
58
+ async run() {
59
+ // Get check as synchronous or asynchronous function
60
+ const check = this.check instanceof Promise ? await this.check : this.check; // Run check
61
+
62
+ try {
63
+ check();
64
+ this._checkState = CheckState.success;
65
+ } catch (e) {
66
+ this.stateFailError = e;
67
+ this._checkState = CheckState.fail;
68
+ }
69
+ }
70
+ /**
71
+ * Validates the constructor parameters.
72
+ * @param {Object} params The parameters to validate.
73
+ */
74
+
75
+
76
+ _validateParams(params) {
77
+ _Utils.default.validateParams(params, {
78
+ group: {
79
+ t: 'string',
80
+ v: _lodash.isString
81
+ },
82
+ title: {
83
+ t: 'string',
84
+ v: _lodash.isString
85
+ },
86
+ warning: {
87
+ t: 'string',
88
+ v: _lodash.isString
89
+ },
90
+ solution: {
91
+ t: 'string',
92
+ v: _lodash.isString
93
+ },
94
+ check: {
95
+ t: 'function',
96
+ v: _lodash.isFunction
97
+ }
98
+ });
99
+ }
100
+
101
+ }
102
+ /**
103
+ * The check state.
104
+ */
105
+
106
+
107
+ const CheckState = Object.freeze({
108
+ none: 'none',
109
+ fail: 'fail',
110
+ success: 'success'
111
+ });
112
+ var _default = Check;
113
+ exports.default = _default;
114
+ module.exports = {
115
+ Check,
116
+ CheckState
117
+ };
118
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9TZWN1cml0eS9DaGVjay5qcyJdLCJuYW1lcyI6WyJDaGVjayIsImNvbnN0cnVjdG9yIiwicGFyYW1zIiwiX3ZhbGlkYXRlUGFyYW1zIiwidGl0bGUiLCJ3YXJuaW5nIiwic29sdXRpb24iLCJjaGVjayIsIl9jaGVja1N0YXRlIiwiQ2hlY2tTdGF0ZSIsIm5vbmUiLCJlcnJvciIsImNoZWNrU3RhdGUiLCJydW4iLCJQcm9taXNlIiwic3VjY2VzcyIsImUiLCJzdGF0ZUZhaWxFcnJvciIsImZhaWwiLCJVdGlscyIsInZhbGlkYXRlUGFyYW1zIiwiZ3JvdXAiLCJ0IiwidiIsImlzU3RyaW5nIiwiaXNGdW5jdGlvbiIsIk9iamVjdCIsImZyZWV6ZSIsIm1vZHVsZSIsImV4cG9ydHMiXSwibWFwcGluZ3MiOiI7Ozs7Ozs7QUFJQTs7QUFDQTs7OztBQUxBO0FBQ0E7QUFDQTs7QUFLQTtBQUNBO0FBQ0E7QUFDQTtBQUNBLE1BQU1BLEtBQU4sQ0FBWTtBQUNWO0FBQ0Y7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDRUMsRUFBQUEsV0FBVyxDQUFDQyxNQUFELEVBQVM7QUFDbEIsU0FBS0MsZUFBTCxDQUFxQkQsTUFBckI7O0FBQ0EsVUFBTTtBQUFFRSxNQUFBQSxLQUFGO0FBQVNDLE1BQUFBLE9BQVQ7QUFBa0JDLE1BQUFBLFFBQWxCO0FBQTRCQyxNQUFBQTtBQUE1QixRQUFzQ0wsTUFBNUM7QUFFQSxTQUFLRSxLQUFMLEdBQWFBLEtBQWI7QUFDQSxTQUFLQyxPQUFMLEdBQWVBLE9BQWY7QUFDQSxTQUFLQyxRQUFMLEdBQWdCQSxRQUFoQjtBQUNBLFNBQUtDLEtBQUwsR0FBYUEsS0FBYixDQVBrQixDQVNsQjs7QUFDQSxTQUFLQyxXQUFMLEdBQW1CQyxVQUFVLENBQUNDLElBQTlCO0FBQ0EsU0FBS0MsS0FBTDtBQUNEO0FBRUQ7QUFDRjtBQUNBO0FBQ0E7OztBQUNFQyxFQUFBQSxVQUFVLEdBQUc7QUFDWCxXQUFPLEtBQUtKLFdBQVo7QUFDRDs7QUFFUSxRQUFISyxHQUFHLEdBQUc7QUFDVjtBQUNBLFVBQU1OLEtBQUssR0FBRyxLQUFLQSxLQUFMLFlBQXNCTyxPQUF0QixHQUFnQyxNQUFNLEtBQUtQLEtBQTNDLEdBQW1ELEtBQUtBLEtBQXRFLENBRlUsQ0FJVjs7QUFDQSxRQUFJO0FBQ0ZBLE1BQUFBLEtBQUs7QUFDTCxXQUFLQyxXQUFMLEdBQW1CQyxVQUFVLENBQUNNLE9BQTlCO0FBQ0QsS0FIRCxDQUdFLE9BQU9DLENBQVAsRUFBVTtBQUNWLFdBQUtDLGNBQUwsR0FBc0JELENBQXRCO0FBQ0EsV0FBS1IsV0FBTCxHQUFtQkMsVUFBVSxDQUFDUyxJQUE5QjtBQUNEO0FBQ0Y7QUFFRDtBQUNGO0FBQ0E7QUFDQTs7O0FBQ0VmLEVBQUFBLGVBQWUsQ0FBQ0QsTUFBRCxFQUFTO0FBQ3RCaUIsbUJBQU1DLGNBQU4sQ0FBcUJsQixNQUFyQixFQUE2QjtBQUMzQm1CLE1BQUFBLEtBQUssRUFBRTtBQUFFQyxRQUFBQSxDQUFDLEVBQUUsUUFBTDtBQUFlQyxRQUFBQSxDQUFDLEVBQUVDO0FBQWxCLE9BRG9CO0FBRTNCcEIsTUFBQUEsS0FBSyxFQUFFO0FBQUVrQixRQUFBQSxDQUFDLEVBQUUsUUFBTDtBQUFlQyxRQUFBQSxDQUFDLEVBQUVDO0FBQWxCLE9BRm9CO0FBRzNCbkIsTUFBQUEsT0FBTyxFQUFFO0FBQUVpQixRQUFBQSxDQUFDLEVBQUUsUUFBTDtBQUFlQyxRQUFBQSxDQUFDLEVBQUVDO0FBQWxCLE9BSGtCO0FBSTNCbEIsTUFBQUEsUUFBUSxFQUFFO0FBQUVnQixRQUFBQSxDQUFDLEVBQUUsUUFBTDtBQUFlQyxRQUFBQSxDQUFDLEVBQUVDO0FBQWxCLE9BSmlCO0FBSzNCakIsTUFBQUEsS0FBSyxFQUFFO0FBQUVlLFFBQUFBLENBQUMsRUFBRSxVQUFMO0FBQWlCQyxRQUFBQSxDQUFDLEVBQUVFO0FBQXBCO0FBTG9CLEtBQTdCO0FBT0Q7O0FBekRTO0FBNERaO0FBQ0E7QUFDQTs7O0FBQ0EsTUFBTWhCLFVBQVUsR0FBR2lCLE1BQU0sQ0FBQ0MsTUFBUCxDQUFjO0FBQy9CakIsRUFBQUEsSUFBSSxFQUFFLE1BRHlCO0FBRS9CUSxFQUFBQSxJQUFJLEVBQUUsTUFGeUI7QUFHL0JILEVBQUFBLE9BQU8sRUFBRTtBQUhzQixDQUFkLENBQW5CO2VBTWVmLEs7O0FBQ2Y0QixNQUFNLENBQUNDLE9BQVAsR0FBaUI7QUFDZjdCLEVBQUFBLEtBRGU7QUFFZlMsRUFBQUE7QUFGZSxDQUFqQiIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQG1vZHVsZSBTZWN1cml0eUNoZWNrXG4gKi9cblxuaW1wb3J0IFV0aWxzIGZyb20gJy4uL1V0aWxzJztcbmltcG9ydCB7IGlzRnVuY3Rpb24sIGlzU3RyaW5nIH0gZnJvbSAnbG9kYXNoJztcblxuLyoqXG4gKiBBIHNlY3VyaXR5IGNoZWNrLlxuICogQGNsYXNzIENoZWNrXG4gKi9cbmNsYXNzIENoZWNrIHtcbiAgLyoqXG4gICAqIENvbnN0cnVjdHMgYSBuZXcgc2VjdXJpdHkgY2hlY2suXG4gICAqIEBwYXJhbSB7T2JqZWN0fSBwYXJhbXMgVGhlIHBhcmFtZXRlcnMuXG4gICAqIEBwYXJhbSB7U3RyaW5nfSBwYXJhbXMudGl0bGUgVGhlIHRpdGxlLlxuICAgKiBAcGFyYW0ge1N0cmluZ30gcGFyYW1zLndhcm5pbmcgVGhlIHdhcm5pbmcgbWVzc2FnZSBpZiB0aGUgY2hlY2sgZmFpbHMuXG4gICAqIEBwYXJhbSB7U3RyaW5nfSBwYXJhbXMuc29sdXRpb24gVGhlIHNvbHV0aW9uIHRvIGZpeCB0aGUgY2hlY2suXG4gICAqIEBwYXJhbSB7UHJvbWlzZX0gcGFyYW1zLmNoZWNrIFRoZSBjaGVjayBhcyBzeW5jaHJvbm91cyBvciBhc3luY2hyb25vdXMgZnVuY3Rpb24uXG4gICAqL1xuICBjb25zdHJ1Y3RvcihwYXJhbXMpIHtcbiAgICB0aGlzLl92YWxpZGF0ZVBhcmFtcyhwYXJhbXMpO1xuICAgIGNvbnN0IHsgdGl0bGUsIHdhcm5pbmcsIHNvbHV0aW9uLCBjaGVjayB9ID0gcGFyYW1zO1xuXG4gICAgdGhpcy50aXRsZSA9IHRpdGxlO1xuICAgIHRoaXMud2FybmluZyA9IHdhcm5pbmc7XG4gICAgdGhpcy5zb2x1dGlvbiA9IHNvbHV0aW9uO1xuICAgIHRoaXMuY2hlY2sgPSBjaGVjaztcblxuICAgIC8vIFNldCBkZWZhdWx0IHByb3BlcnRpZXNcbiAgICB0aGlzLl9jaGVja1N0YXRlID0gQ2hlY2tTdGF0ZS5ub25lO1xuICAgIHRoaXMuZXJyb3I7XG4gIH1cblxuICAvKipcbiAgICogUmV0dXJucyB0aGUgY3VycmVudCBjaGVjayBzdGF0ZS5cbiAgICogQHJldHVybiB7Q2hlY2tTdGF0ZX0gVGhlIGNoZWNrIHN0YXRlLlxuICAgKi9cbiAgY2hlY2tTdGF0ZSgpIHtcbiAgICByZXR1cm4gdGhpcy5fY2hlY2tTdGF0ZTtcbiAgfVxuXG4gIGFzeW5jIHJ1bigpIHtcbiAgICAvLyBHZXQgY2hlY2sgYXMgc3luY2hyb25vdXMgb3IgYXN5bmNocm9ub3VzIGZ1bmN0aW9uXG4gICAgY29uc3QgY2hlY2sgPSB0aGlzLmNoZWNrIGluc3RhbmNlb2YgUHJvbWlzZSA/IGF3YWl0IHRoaXMuY2hlY2sgOiB0aGlzLmNoZWNrO1xuXG4gICAgLy8gUnVuIGNoZWNrXG4gICAgdHJ5IHtcbiAgICAgIGNoZWNrKCk7XG4gICAgICB0aGlzLl9jaGVja1N0YXRlID0gQ2hlY2tTdGF0ZS5zdWNjZXNzO1xuICAgIH0gY2F0Y2ggKGUpIHtcbiAgICAgIHRoaXMuc3RhdGVGYWlsRXJyb3IgPSBlO1xuICAgICAgdGhpcy5fY2hlY2tTdGF0ZSA9IENoZWNrU3RhdGUuZmFpbDtcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogVmFsaWRhdGVzIHRoZSBjb25zdHJ1Y3RvciBwYXJhbWV0ZXJzLlxuICAgKiBAcGFyYW0ge09iamVjdH0gcGFyYW1zIFRoZSBwYXJhbWV0ZXJzIHRvIHZhbGlkYXRlLlxuICAgKi9cbiAgX3ZhbGlkYXRlUGFyYW1zKHBhcmFtcykge1xuICAgIFV0aWxzLnZhbGlkYXRlUGFyYW1zKHBhcmFtcywge1xuICAgICAgZ3JvdXA6IHsgdDogJ3N0cmluZycsIHY6IGlzU3RyaW5nIH0sXG4gICAgICB0aXRsZTogeyB0OiAnc3RyaW5nJywgdjogaXNTdHJpbmcgfSxcbiAgICAgIHdhcm5pbmc6IHsgdDogJ3N0cmluZycsIHY6IGlzU3RyaW5nIH0sXG4gICAgICBzb2x1dGlvbjogeyB0OiAnc3RyaW5nJywgdjogaXNTdHJpbmcgfSxcbiAgICAgIGNoZWNrOiB7IHQ6ICdmdW5jdGlvbicsIHY6IGlzRnVuY3Rpb24gfSxcbiAgICB9KTtcbiAgfVxufVxuXG4vKipcbiAqIFRoZSBjaGVjayBzdGF0ZS5cbiAqL1xuY29uc3QgQ2hlY2tTdGF0ZSA9IE9iamVjdC5mcmVlemUoe1xuICBub25lOiAnbm9uZScsXG4gIGZhaWw6ICdmYWlsJyxcbiAgc3VjY2VzczogJ3N1Y2Nlc3MnLFxufSk7XG5cbmV4cG9ydCBkZWZhdWx0IENoZWNrO1xubW9kdWxlLmV4cG9ydHMgPSB7XG4gIENoZWNrLFxuICBDaGVja1N0YXRlLFxufTtcbiJdfQ==