parse-server 6.0.0-alpha.2 → 6.0.0-alpha.21
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +45 -17
- package/lib/AccountLockout.js +11 -26
- package/lib/Adapters/AdapterLoader.js +8 -14
- package/lib/Adapters/Analytics/AnalyticsAdapter.js +2 -8
- package/lib/Adapters/Auth/AuthAdapter.js +7 -16
- package/lib/Adapters/Auth/OAuth1Client.js +32 -57
- package/lib/Adapters/Auth/apple.js +6 -22
- package/lib/Adapters/Auth/facebook.js +7 -37
- package/lib/Adapters/Auth/gcenter.js +8 -37
- package/lib/Adapters/Auth/github.js +7 -10
- package/lib/Adapters/Auth/google.js +11 -34
- package/lib/Adapters/Auth/gpgames.js +5 -8
- package/lib/Adapters/Auth/httpsRequest.js +1 -7
- package/lib/Adapters/Auth/index.js +20 -65
- package/lib/Adapters/Auth/instagram.js +5 -9
- package/lib/Adapters/Auth/janraincapture.js +8 -12
- package/lib/Adapters/Auth/janrainengage.js +7 -11
- package/lib/Adapters/Auth/keycloak.js +5 -19
- package/lib/Adapters/Auth/ldap.js +1 -15
- package/lib/Adapters/Auth/line.js +7 -10
- package/lib/Adapters/Auth/linkedin.js +7 -12
- package/lib/Adapters/Auth/meetup.js +7 -10
- package/lib/Adapters/Auth/microsoft.js +7 -10
- package/lib/Adapters/Auth/oauth2.js +6 -18
- package/lib/Adapters/Auth/phantauth.js +8 -10
- package/lib/Adapters/Auth/qq.js +7 -13
- package/lib/Adapters/Auth/spotify.js +7 -14
- package/lib/Adapters/Auth/twitter.js +5 -15
- package/lib/Adapters/Auth/vkontakte.js +9 -15
- package/lib/Adapters/Auth/wechat.js +7 -10
- package/lib/Adapters/Auth/weibo.js +7 -11
- package/lib/Adapters/Cache/CacheAdapter.js +4 -12
- package/lib/Adapters/Cache/InMemoryCache.js +5 -19
- package/lib/Adapters/Cache/InMemoryCacheAdapter.js +1 -11
- package/lib/Adapters/Cache/LRUCache.js +1 -11
- package/lib/Adapters/Cache/NullCacheAdapter.js +1 -8
- package/lib/Adapters/Cache/RedisCacheAdapter.js +46 -87
- package/lib/Adapters/Cache/SchemaCache.js +1 -6
- package/lib/Adapters/Email/MailAdapter.js +2 -7
- package/lib/Adapters/Files/FilesAdapter.js +7 -21
- package/lib/Adapters/Files/GridFSBucketAdapter.js +6 -44
- package/lib/Adapters/Files/GridStoreAdapter.js +1 -1
- package/lib/Adapters/Logger/LoggerAdapter.js +2 -11
- package/lib/Adapters/Logger/WinstonLogger.js +3 -30
- package/lib/Adapters/Logger/WinstonLoggerAdapter.js +5 -16
- package/lib/Adapters/MessageQueue/EventEmitterMQ.js +3 -20
- package/lib/Adapters/PubSub/EventEmitterPubSub.js +1 -16
- package/lib/Adapters/PubSub/PubSubAdapter.js +2 -9
- package/lib/Adapters/PubSub/RedisPubSub.js +13 -10
- package/lib/Adapters/Push/PushAdapter.js +2 -8
- package/lib/Adapters/Storage/Mongo/MongoCollection.js +12 -37
- package/lib/Adapters/Storage/Mongo/MongoSchemaCollection.js +26 -79
- package/lib/Adapters/Storage/Mongo/MongoStorageAdapter.js +78 -209
- package/lib/Adapters/Storage/Mongo/MongoTransform.js +82 -371
- package/lib/Adapters/Storage/Postgres/PostgresClient.js +1 -13
- package/lib/Adapters/Storage/Postgres/PostgresConfigParser.js +1 -20
- package/lib/Adapters/Storage/Postgres/PostgresStorageAdapter.js +119 -446
- package/lib/Adapters/Storage/Postgres/sql/index.js +4 -7
- package/lib/Adapters/Storage/StorageAdapter.js +1 -1
- package/lib/Adapters/WebSocketServer/WSAdapter.js +3 -12
- package/lib/Adapters/WebSocketServer/WSSAdapter.js +7 -12
- package/lib/Auth.js +54 -121
- package/lib/ClientSDK.js +3 -11
- package/lib/Config.js +69 -113
- package/lib/Controllers/AdaptableController.js +6 -18
- package/lib/Controllers/AnalyticsController.js +1 -9
- package/lib/Controllers/CacheController.js +3 -23
- package/lib/Controllers/DatabaseController.js +147 -345
- package/lib/Controllers/FilesController.js +5 -34
- package/lib/Controllers/HooksController.js +1 -51
- package/lib/Controllers/LiveQueryController.js +4 -23
- package/lib/Controllers/LoggerController.js +15 -54
- package/lib/Controllers/ParseGraphQLController.js +49 -104
- package/lib/Controllers/PushController.js +20 -59
- package/lib/Controllers/SchemaController.js +154 -344
- package/lib/Controllers/UserController.js +11 -72
- package/lib/Controllers/index.js +19 -68
- package/lib/Controllers/types.js +1 -1
- package/lib/Deprecator/Deprecations.js +1 -8
- package/lib/Deprecator/Deprecator.js +9 -18
- package/lib/GraphQL/ParseGraphQLSchema.js +16 -100
- package/lib/GraphQL/ParseGraphQLServer.js +2 -29
- package/lib/GraphQL/helpers/objectsMutations.js +2 -12
- package/lib/GraphQL/helpers/objectsQueries.js +18 -76
- package/lib/GraphQL/loaders/defaultGraphQLMutations.js +1 -9
- package/lib/GraphQL/loaders/defaultGraphQLQueries.js +1 -8
- package/lib/GraphQL/loaders/defaultGraphQLTypes.js +9 -115
- package/lib/GraphQL/loaders/defaultRelaySchema.js +6 -18
- package/lib/GraphQL/loaders/filesMutations.js +2 -19
- package/lib/GraphQL/loaders/functionsMutations.js +6 -17
- package/lib/GraphQL/loaders/parseClassMutations.js +6 -44
- package/lib/GraphQL/loaders/parseClassQueries.js +1 -26
- package/lib/GraphQL/loaders/parseClassTypes.js +10 -64
- package/lib/GraphQL/loaders/schemaDirectives.js +1 -17
- package/lib/GraphQL/loaders/schemaMutations.js +1 -20
- package/lib/GraphQL/loaders/schemaQueries.js +1 -14
- package/lib/GraphQL/loaders/schemaTypes.js +2 -6
- package/lib/GraphQL/loaders/usersMutations.js +6 -28
- package/lib/GraphQL/loaders/usersQueries.js +4 -26
- package/lib/GraphQL/parseGraphQLUtils.js +6 -19
- package/lib/GraphQL/transformers/className.js +1 -4
- package/lib/GraphQL/transformers/constraintType.js +1 -20
- package/lib/GraphQL/transformers/inputType.js +1 -20
- package/lib/GraphQL/transformers/mutation.js +6 -51
- package/lib/GraphQL/transformers/outputType.js +1 -20
- package/lib/GraphQL/transformers/query.js +6 -42
- package/lib/GraphQL/transformers/schemaFields.js +7 -34
- package/lib/KeyPromiseQueue.js +1 -12
- package/lib/LiveQuery/Client.js +1 -25
- package/lib/LiveQuery/Id.js +1 -7
- package/lib/LiveQuery/ParseCloudCodePublisher.js +13 -19
- package/lib/LiveQuery/ParseLiveQueryServer.js +92 -306
- package/lib/LiveQuery/ParsePubSub.js +1 -12
- package/lib/LiveQuery/ParseWebSocketServer.js +4 -26
- package/lib/LiveQuery/QueryTools.js +14 -116
- package/lib/LiveQuery/RequestSchema.js +1 -1
- package/lib/LiveQuery/SessionTokenCache.js +1 -17
- package/lib/LiveQuery/Subscription.js +4 -18
- package/lib/LiveQuery/equalObjects.js +2 -14
- package/lib/Options/Definitions.js +79 -10
- package/lib/Options/docs.js +23 -3
- package/lib/Options/index.js +4 -12
- package/lib/Options/parsers.js +1 -18
- package/lib/Page.js +1 -9
- package/lib/ParseMessageQueue.js +1 -10
- package/lib/ParseServer.js +144 -182
- package/lib/ParseServerRESTController.js +6 -33
- package/lib/PromiseRouter.js +16 -50
- package/lib/Push/PushQueue.js +3 -15
- package/lib/Push/PushWorker.js +7 -32
- package/lib/Push/utils.js +9 -38
- package/lib/RestQuery.js +105 -242
- package/lib/RestWrite.js +212 -377
- package/lib/Routers/AggregateRouter.js +14 -51
- package/lib/Routers/AnalyticsRouter.js +2 -8
- package/lib/Routers/AudiencesRouter.js +1 -15
- package/lib/Routers/ClassesRouter.js +3 -53
- package/lib/Routers/CloudCodeRouter.js +1 -19
- package/lib/Routers/FeaturesRouter.js +1 -10
- package/lib/Routers/FilesRouter.js +29 -76
- package/lib/Routers/FunctionsRouter.js +5 -28
- package/lib/Routers/GlobalConfigRouter.js +4 -18
- package/lib/Routers/GraphQLRouter.js +1 -14
- package/lib/Routers/HooksRouter.js +1 -29
- package/lib/Routers/IAPValidationRouter.js +6 -29
- package/lib/Routers/InstallationsRouter.js +2 -12
- package/lib/Routers/LogsRouter.js +4 -16
- package/lib/Routers/PagesRouter.js +69 -129
- package/lib/Routers/PublicAPIRouter.js +3 -62
- package/lib/Routers/PurgeRouter.js +1 -15
- package/lib/Routers/PushRouter.js +2 -18
- package/lib/Routers/RolesRouter.js +1 -7
- package/lib/Routers/SchemasRouter.js +4 -34
- package/lib/Routers/SecurityRouter.js +1 -12
- package/lib/Routers/SessionsRouter.js +3 -19
- package/lib/Routers/UsersRouter.js +58 -155
- package/lib/SchemaMigrations/DefinedSchemas.js +56 -115
- package/lib/SchemaMigrations/Migrations.js +2 -8
- package/lib/Security/Check.js +8 -16
- package/lib/Security/CheckGroup.js +4 -11
- package/lib/Security/CheckGroups/CheckGroupDatabase.js +8 -18
- package/lib/Security/CheckGroups/CheckGroupServerConfig.js +5 -15
- package/lib/Security/CheckGroups/CheckGroups.js +1 -4
- package/lib/Security/CheckRunner.js +22 -41
- package/lib/StatusHandler.js +12 -69
- package/lib/TestUtils.js +1 -6
- package/lib/Utils.js +27 -66
- package/lib/batch.js +17 -28
- package/lib/cache.js +1 -3
- package/lib/cli/definitions/parse-live-query-server.js +1 -3
- package/lib/cli/definitions/parse-server.js +1 -3
- package/lib/cli/parse-live-query-server.js +1 -6
- package/lib/cli/parse-server.js +11 -21
- package/lib/cli/utils/commander.js +13 -51
- package/lib/cli/utils/runner.js +1 -14
- package/lib/cloud-code/Parse.Cloud.js +71 -92
- package/lib/cryptoUtils.js +11 -19
- package/lib/defaults.js +2 -14
- package/lib/deprecated.js +1 -2
- package/lib/index.js +16 -34
- package/lib/logger.js +6 -13
- package/lib/middlewares.js +147 -151
- package/lib/password.js +6 -10
- package/lib/request.js +173 -2
- package/lib/requiredParameter.js +1 -3
- package/lib/rest.js +19 -41
- package/lib/triggers.js +54 -252
- package/lib/vendor/mongodbUrl.js +125 -305
- package/package.json +22 -19
- package/lib/cloud-code/HTTPResponse.js +0 -73
- package/lib/cloud-code/httpRequest.js +0 -192
|
@@ -3,32 +3,25 @@
|
|
|
3
3
|
Object.defineProperty(exports, "__esModule", {
|
|
4
4
|
value: true
|
|
5
5
|
});
|
|
6
|
+
exports.VolatileClassesSchemas = exports.SchemaController = void 0;
|
|
7
|
+
exports.buildMergedSchemaObject = buildMergedSchemaObject;
|
|
6
8
|
exports.classNameIsValid = classNameIsValid;
|
|
9
|
+
exports.defaultColumns = exports.default = exports.convertSchemaToAdapterSchema = void 0;
|
|
7
10
|
exports.fieldNameIsValid = fieldNameIsValid;
|
|
8
11
|
exports.invalidClassNameMessage = invalidClassNameMessage;
|
|
9
|
-
exports.
|
|
10
|
-
exports.requiredColumns = exports.VolatileClassesSchemas = exports.convertSchemaToAdapterSchema = exports.defaultColumns = exports.systemClasses = exports.load = exports.SchemaController = exports.default = void 0;
|
|
11
|
-
|
|
12
|
+
exports.systemClasses = exports.requiredColumns = exports.load = void 0;
|
|
12
13
|
var _StorageAdapter = require("../Adapters/Storage/StorageAdapter");
|
|
13
|
-
|
|
14
14
|
var _SchemaCache = _interopRequireDefault(require("../Adapters/Cache/SchemaCache"));
|
|
15
|
-
|
|
16
15
|
var _DatabaseController = _interopRequireDefault(require("./DatabaseController"));
|
|
17
|
-
|
|
18
16
|
var _Config = _interopRequireDefault(require("../Config"));
|
|
19
|
-
|
|
20
17
|
var _deepcopy = _interopRequireDefault(require("deepcopy"));
|
|
21
|
-
|
|
22
18
|
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
23
|
-
|
|
24
|
-
function
|
|
25
|
-
|
|
26
|
-
function
|
|
27
|
-
|
|
28
|
-
function
|
|
29
|
-
|
|
30
|
-
function _extends() { _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; }; return _extends.apply(this, arguments); }
|
|
31
|
-
|
|
19
|
+
function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); enumerableOnly && (symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; })), keys.push.apply(keys, symbols); } return keys; }
|
|
20
|
+
function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = null != arguments[i] ? arguments[i] : {}; i % 2 ? ownKeys(Object(source), !0).forEach(function (key) { _defineProperty(target, key, source[key]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)) : ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } return target; }
|
|
21
|
+
function _defineProperty(obj, key, value) { key = _toPropertyKey(key); if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
|
|
22
|
+
function _toPropertyKey(arg) { var key = _toPrimitive(arg, "string"); return typeof key === "symbol" ? key : String(key); }
|
|
23
|
+
function _toPrimitive(input, hint) { if (typeof input !== "object" || input === null) return input; var prim = input[Symbol.toPrimitive]; if (prim !== undefined) { var res = prim.call(input, hint || "default"); if (typeof res !== "object") return res; throw new TypeError("@@toPrimitive must return a primitive value."); } return (hint === "string" ? String : Number)(input); }
|
|
24
|
+
function _extends() { _extends = Object.assign ? Object.assign.bind() : function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; }; return _extends.apply(this, arguments); }
|
|
32
25
|
// This class handles schema validation, persistence, and modification.
|
|
33
26
|
//
|
|
34
27
|
// Each individual Schema object should be immutable. The helpers to
|
|
@@ -46,7 +39,6 @@ function _extends() { _extends = Object.assign || function (target) { for (var i
|
|
|
46
39
|
// TODO: hide all schema logic inside the database adapter.
|
|
47
40
|
// -disable-next
|
|
48
41
|
const Parse = require('parse/node').Parse;
|
|
49
|
-
|
|
50
42
|
const defaultColumns = Object.freeze({
|
|
51
43
|
// Contain the default columns for every parse object type (except _Join collection)
|
|
52
44
|
_Default: {
|
|
@@ -234,8 +226,8 @@ const defaultColumns = Object.freeze({
|
|
|
234
226
|
count: {
|
|
235
227
|
type: 'Number'
|
|
236
228
|
} // tracks # of batches queued and pending
|
|
237
|
-
|
|
238
229
|
},
|
|
230
|
+
|
|
239
231
|
_JobStatus: {
|
|
240
232
|
jobName: {
|
|
241
233
|
type: 'String'
|
|
@@ -342,8 +334,9 @@ const defaultColumns = Object.freeze({
|
|
|
342
334
|
type: 'Date'
|
|
343
335
|
}
|
|
344
336
|
}
|
|
345
|
-
});
|
|
337
|
+
});
|
|
346
338
|
|
|
339
|
+
// fields required for read or write operations on their respective classes.
|
|
347
340
|
exports.defaultColumns = defaultColumns;
|
|
348
341
|
const requiredColumns = Object.freeze({
|
|
349
342
|
read: {
|
|
@@ -358,152 +351,140 @@ exports.requiredColumns = requiredColumns;
|
|
|
358
351
|
const invalidColumns = ['length'];
|
|
359
352
|
const systemClasses = Object.freeze(['_User', '_Installation', '_Role', '_Session', '_Product', '_PushStatus', '_JobStatus', '_JobSchedule', '_Audience', '_Idempotency']);
|
|
360
353
|
exports.systemClasses = systemClasses;
|
|
361
|
-
const volatileClasses = Object.freeze(['_JobStatus', '_PushStatus', '_Hooks', '_GlobalConfig', '_GraphQLConfig', '_JobSchedule', '_Audience', '_Idempotency']);
|
|
362
|
-
|
|
363
|
-
const roleRegex = /^role:.*/; // Anything that starts with userField (allowed for protected fields only)
|
|
364
|
-
|
|
365
|
-
const protectedFieldsPointerRegex = /^userField:.*/; // * permission
|
|
354
|
+
const volatileClasses = Object.freeze(['_JobStatus', '_PushStatus', '_Hooks', '_GlobalConfig', '_GraphQLConfig', '_JobSchedule', '_Audience', '_Idempotency']);
|
|
366
355
|
|
|
356
|
+
// Anything that start with role
|
|
357
|
+
const roleRegex = /^role:.*/;
|
|
358
|
+
// Anything that starts with userField (allowed for protected fields only)
|
|
359
|
+
const protectedFieldsPointerRegex = /^userField:.*/;
|
|
360
|
+
// * permission
|
|
367
361
|
const publicRegex = /^\*$/;
|
|
368
362
|
const authenticatedRegex = /^authenticated$/;
|
|
369
363
|
const requiresAuthenticationRegex = /^requiresAuthentication$/;
|
|
370
|
-
const clpPointerRegex = /^pointerFields$/;
|
|
364
|
+
const clpPointerRegex = /^pointerFields$/;
|
|
371
365
|
|
|
372
|
-
|
|
366
|
+
// regex for validating entities in protectedFields object
|
|
367
|
+
const protectedFieldsRegex = Object.freeze([protectedFieldsPointerRegex, publicRegex, authenticatedRegex, roleRegex]);
|
|
373
368
|
|
|
369
|
+
// clp regex
|
|
374
370
|
const clpFieldsRegex = Object.freeze([clpPointerRegex, publicRegex, requiresAuthenticationRegex, roleRegex]);
|
|
375
|
-
|
|
376
371
|
function validatePermissionKey(key, userIdRegExp) {
|
|
377
372
|
let matchesSome = false;
|
|
378
|
-
|
|
379
373
|
for (const regEx of clpFieldsRegex) {
|
|
380
374
|
if (key.match(regEx) !== null) {
|
|
381
375
|
matchesSome = true;
|
|
382
376
|
break;
|
|
383
377
|
}
|
|
384
|
-
}
|
|
385
|
-
|
|
378
|
+
}
|
|
386
379
|
|
|
380
|
+
// userId depends on startup options so it's dynamic
|
|
387
381
|
const valid = matchesSome || key.match(userIdRegExp) !== null;
|
|
388
|
-
|
|
389
382
|
if (!valid) {
|
|
390
383
|
throw new Parse.Error(Parse.Error.INVALID_JSON, `'${key}' is not a valid key for class level permissions`);
|
|
391
384
|
}
|
|
392
385
|
}
|
|
393
|
-
|
|
394
386
|
function validateProtectedFieldsKey(key, userIdRegExp) {
|
|
395
387
|
let matchesSome = false;
|
|
396
|
-
|
|
397
388
|
for (const regEx of protectedFieldsRegex) {
|
|
398
389
|
if (key.match(regEx) !== null) {
|
|
399
390
|
matchesSome = true;
|
|
400
391
|
break;
|
|
401
392
|
}
|
|
402
|
-
}
|
|
403
|
-
|
|
393
|
+
}
|
|
404
394
|
|
|
395
|
+
// userId regex depends on launch options so it's dynamic
|
|
405
396
|
const valid = matchesSome || key.match(userIdRegExp) !== null;
|
|
406
|
-
|
|
407
397
|
if (!valid) {
|
|
408
398
|
throw new Parse.Error(Parse.Error.INVALID_JSON, `'${key}' is not a valid key for class level permissions`);
|
|
409
399
|
}
|
|
410
400
|
}
|
|
401
|
+
const CLPValidKeys = Object.freeze(['find', 'count', 'get', 'create', 'update', 'delete', 'addField', 'readUserFields', 'writeUserFields', 'protectedFields']);
|
|
411
402
|
|
|
412
|
-
|
|
413
|
-
|
|
403
|
+
// validation before setting class-level permissions on collection
|
|
414
404
|
function validateCLP(perms, fields, userIdRegExp) {
|
|
415
405
|
if (!perms) {
|
|
416
406
|
return;
|
|
417
407
|
}
|
|
418
|
-
|
|
419
408
|
for (const operationKey in perms) {
|
|
420
409
|
if (CLPValidKeys.indexOf(operationKey) == -1) {
|
|
421
410
|
throw new Parse.Error(Parse.Error.INVALID_JSON, `${operationKey} is not a valid operation for class level permissions`);
|
|
422
411
|
}
|
|
412
|
+
const operation = perms[operationKey];
|
|
413
|
+
// proceed with next operationKey
|
|
423
414
|
|
|
424
|
-
const operation = perms[operationKey]; // proceed with next operationKey
|
|
425
415
|
// throws when root fields are of wrong type
|
|
426
|
-
|
|
427
416
|
validateCLPjson(operation, operationKey);
|
|
428
|
-
|
|
429
417
|
if (operationKey === 'readUserFields' || operationKey === 'writeUserFields') {
|
|
430
418
|
// validate grouped pointer permissions
|
|
431
419
|
// must be an array with field names
|
|
432
420
|
for (const fieldName of operation) {
|
|
433
421
|
validatePointerPermission(fieldName, fields, operationKey);
|
|
434
|
-
}
|
|
422
|
+
}
|
|
423
|
+
// readUserFields and writerUserFields do not have nesdted fields
|
|
435
424
|
// proceed with next operationKey
|
|
436
|
-
|
|
437
|
-
|
|
438
425
|
continue;
|
|
439
|
-
}
|
|
440
|
-
|
|
426
|
+
}
|
|
441
427
|
|
|
428
|
+
// validate protected fields
|
|
442
429
|
if (operationKey === 'protectedFields') {
|
|
443
430
|
for (const entity in operation) {
|
|
444
431
|
// throws on unexpected key
|
|
445
432
|
validateProtectedFieldsKey(entity, userIdRegExp);
|
|
446
433
|
const protectedFields = operation[entity];
|
|
447
|
-
|
|
448
434
|
if (!Array.isArray(protectedFields)) {
|
|
449
435
|
throw new Parse.Error(Parse.Error.INVALID_JSON, `'${protectedFields}' is not a valid value for protectedFields[${entity}] - expected an array.`);
|
|
450
|
-
}
|
|
451
|
-
|
|
436
|
+
}
|
|
452
437
|
|
|
438
|
+
// if the field is in form of array
|
|
453
439
|
for (const field of protectedFields) {
|
|
454
440
|
// do not alloow to protect default fields
|
|
455
441
|
if (defaultColumns._Default[field]) {
|
|
456
442
|
throw new Parse.Error(Parse.Error.INVALID_JSON, `Default field '${field}' can not be protected`);
|
|
457
|
-
}
|
|
458
|
-
|
|
459
|
-
|
|
443
|
+
}
|
|
444
|
+
// field should exist on collection
|
|
460
445
|
if (!Object.prototype.hasOwnProperty.call(fields, field)) {
|
|
461
446
|
throw new Parse.Error(Parse.Error.INVALID_JSON, `Field '${field}' in protectedFields:${entity} does not exist`);
|
|
462
447
|
}
|
|
463
448
|
}
|
|
464
|
-
}
|
|
465
|
-
|
|
466
|
-
|
|
449
|
+
}
|
|
450
|
+
// proceed with next operationKey
|
|
467
451
|
continue;
|
|
468
|
-
}
|
|
452
|
+
}
|
|
453
|
+
|
|
454
|
+
// validate other fields
|
|
469
455
|
// Entity can be:
|
|
470
456
|
// "*" - Public,
|
|
471
457
|
// "requiresAuthentication" - authenticated users,
|
|
472
458
|
// "objectId" - _User id,
|
|
473
459
|
// "role:rolename",
|
|
474
460
|
// "pointerFields" - array of field names containing pointers to users
|
|
475
|
-
|
|
476
|
-
|
|
477
461
|
for (const entity in operation) {
|
|
478
462
|
// throws on unexpected key
|
|
479
|
-
validatePermissionKey(entity, userIdRegExp);
|
|
480
|
-
// "pointerFields": string[]
|
|
463
|
+
validatePermissionKey(entity, userIdRegExp);
|
|
481
464
|
|
|
465
|
+
// entity can be either:
|
|
466
|
+
// "pointerFields": string[]
|
|
482
467
|
if (entity === 'pointerFields') {
|
|
483
468
|
const pointerFields = operation[entity];
|
|
484
|
-
|
|
485
469
|
if (Array.isArray(pointerFields)) {
|
|
486
470
|
for (const pointerField of pointerFields) {
|
|
487
471
|
validatePointerPermission(pointerField, fields, operation);
|
|
488
472
|
}
|
|
489
473
|
} else {
|
|
490
474
|
throw new Parse.Error(Parse.Error.INVALID_JSON, `'${pointerFields}' is not a valid value for ${operationKey}[${entity}] - expected an array.`);
|
|
491
|
-
}
|
|
492
|
-
|
|
493
|
-
|
|
475
|
+
}
|
|
476
|
+
// proceed with next entity key
|
|
494
477
|
continue;
|
|
495
|
-
}
|
|
496
|
-
|
|
478
|
+
}
|
|
497
479
|
|
|
480
|
+
// or [entity]: boolean
|
|
498
481
|
const permit = operation[entity];
|
|
499
|
-
|
|
500
482
|
if (permit !== true) {
|
|
501
483
|
throw new Parse.Error(Parse.Error.INVALID_JSON, `'${permit}' is not a valid value for class level permissions ${operationKey}:${entity}:${permit}`);
|
|
502
484
|
}
|
|
503
485
|
}
|
|
504
486
|
}
|
|
505
487
|
}
|
|
506
|
-
|
|
507
488
|
function validateCLPjson(operation, operationKey) {
|
|
508
489
|
if (operationKey === 'readUserFields' || operationKey === 'writeUserFields') {
|
|
509
490
|
if (!Array.isArray(operation)) {
|
|
@@ -518,7 +499,6 @@ function validateCLPjson(operation, operationKey) {
|
|
|
518
499
|
}
|
|
519
500
|
}
|
|
520
501
|
}
|
|
521
|
-
|
|
522
502
|
function validatePointerPermission(fieldName, fields, operation) {
|
|
523
503
|
// Uses collection schema to ensure the field is of type:
|
|
524
504
|
// - Pointer<_User> (pointers)
|
|
@@ -531,55 +511,50 @@ function validatePointerPermission(fieldName, fields, operation) {
|
|
|
531
511
|
throw new Parse.Error(Parse.Error.INVALID_JSON, `'${fieldName}' is not a valid column for class level pointer permissions ${operation}`);
|
|
532
512
|
}
|
|
533
513
|
}
|
|
534
|
-
|
|
535
514
|
const joinClassRegex = /^_Join:[A-Za-z0-9_]+:[A-Za-z0-9_]+/;
|
|
536
515
|
const classAndFieldRegex = /^[A-Za-z][A-Za-z0-9_]*$/;
|
|
537
|
-
|
|
538
516
|
function classNameIsValid(className) {
|
|
539
517
|
// Valid classes must:
|
|
540
|
-
return (
|
|
541
|
-
|
|
542
|
-
|
|
518
|
+
return (
|
|
519
|
+
// Be one of _User, _Installation, _Role, _Session OR
|
|
520
|
+
systemClasses.indexOf(className) > -1 ||
|
|
521
|
+
// Be a join table OR
|
|
522
|
+
joinClassRegex.test(className) ||
|
|
523
|
+
// Include only alpha-numeric and underscores, and not start with an underscore or number
|
|
543
524
|
fieldNameIsValid(className, className)
|
|
544
525
|
);
|
|
545
|
-
}
|
|
546
|
-
// must not be a reserved key
|
|
547
|
-
|
|
526
|
+
}
|
|
548
527
|
|
|
528
|
+
// Valid fields must be alpha-numeric, and not start with an underscore or number
|
|
529
|
+
// must not be a reserved key
|
|
549
530
|
function fieldNameIsValid(fieldName, className) {
|
|
550
531
|
if (className && className !== '_Hooks') {
|
|
551
532
|
if (fieldName === 'className') {
|
|
552
533
|
return false;
|
|
553
534
|
}
|
|
554
535
|
}
|
|
555
|
-
|
|
556
536
|
return classAndFieldRegex.test(fieldName) && !invalidColumns.includes(fieldName);
|
|
557
|
-
}
|
|
558
|
-
|
|
537
|
+
}
|
|
559
538
|
|
|
539
|
+
// Checks that it's not trying to clobber one of the default fields of the class.
|
|
560
540
|
function fieldNameIsValidForClass(fieldName, className) {
|
|
561
541
|
if (!fieldNameIsValid(fieldName, className)) {
|
|
562
542
|
return false;
|
|
563
543
|
}
|
|
564
|
-
|
|
565
544
|
if (defaultColumns._Default[fieldName]) {
|
|
566
545
|
return false;
|
|
567
546
|
}
|
|
568
|
-
|
|
569
547
|
if (defaultColumns[className] && defaultColumns[className][fieldName]) {
|
|
570
548
|
return false;
|
|
571
549
|
}
|
|
572
|
-
|
|
573
550
|
return true;
|
|
574
551
|
}
|
|
575
|
-
|
|
576
552
|
function invalidClassNameMessage(className) {
|
|
577
553
|
return 'Invalid classname: ' + className + ', classnames can only have alphanumeric characters and _, and must start with an alpha character ';
|
|
578
554
|
}
|
|
579
|
-
|
|
580
555
|
const invalidJsonError = new Parse.Error(Parse.Error.INVALID_JSON, 'invalid JSON');
|
|
581
|
-
const validNonRelationOrPointerTypes = ['Number', 'String', 'Boolean', 'Date', 'Object', 'Array', 'GeoPoint', 'File', 'Bytes', 'Polygon'];
|
|
582
|
-
|
|
556
|
+
const validNonRelationOrPointerTypes = ['Number', 'String', 'Boolean', 'Date', 'Object', 'Array', 'GeoPoint', 'File', 'Bytes', 'Polygon'];
|
|
557
|
+
// Returns an error suitable for throwing if the type is invalid
|
|
583
558
|
const fieldTypeIsInvalid = ({
|
|
584
559
|
type,
|
|
585
560
|
targetClass
|
|
@@ -595,18 +570,14 @@ const fieldTypeIsInvalid = ({
|
|
|
595
570
|
return undefined;
|
|
596
571
|
}
|
|
597
572
|
}
|
|
598
|
-
|
|
599
573
|
if (typeof type !== 'string') {
|
|
600
574
|
return invalidJsonError;
|
|
601
575
|
}
|
|
602
|
-
|
|
603
576
|
if (validNonRelationOrPointerTypes.indexOf(type) < 0) {
|
|
604
577
|
return new Parse.Error(Parse.Error.INCORRECT_TYPE, `invalid field type: ${type}`);
|
|
605
578
|
}
|
|
606
|
-
|
|
607
579
|
return undefined;
|
|
608
580
|
};
|
|
609
|
-
|
|
610
581
|
const convertSchemaToAdapterSchema = schema => {
|
|
611
582
|
schema = injectDefaultSchema(schema);
|
|
612
583
|
delete schema.fields.ACL;
|
|
@@ -616,44 +587,34 @@ const convertSchemaToAdapterSchema = schema => {
|
|
|
616
587
|
schema.fields._wperm = {
|
|
617
588
|
type: 'Array'
|
|
618
589
|
};
|
|
619
|
-
|
|
620
590
|
if (schema.className === '_User') {
|
|
621
591
|
delete schema.fields.password;
|
|
622
592
|
schema.fields._hashed_password = {
|
|
623
593
|
type: 'String'
|
|
624
594
|
};
|
|
625
595
|
}
|
|
626
|
-
|
|
627
596
|
return schema;
|
|
628
597
|
};
|
|
629
|
-
|
|
630
598
|
exports.convertSchemaToAdapterSchema = convertSchemaToAdapterSchema;
|
|
631
|
-
|
|
632
599
|
const convertAdapterSchemaToParseSchema = _ref => {
|
|
633
600
|
let schema = _extends({}, _ref);
|
|
634
|
-
|
|
635
601
|
delete schema.fields._rperm;
|
|
636
602
|
delete schema.fields._wperm;
|
|
637
603
|
schema.fields.ACL = {
|
|
638
604
|
type: 'ACL'
|
|
639
605
|
};
|
|
640
|
-
|
|
641
606
|
if (schema.className === '_User') {
|
|
642
607
|
delete schema.fields.authData; //Auth data is implicit
|
|
643
|
-
|
|
644
608
|
delete schema.fields._hashed_password;
|
|
645
609
|
schema.fields.password = {
|
|
646
610
|
type: 'String'
|
|
647
611
|
};
|
|
648
612
|
}
|
|
649
|
-
|
|
650
613
|
if (schema.indexes && Object.keys(schema.indexes).length === 0) {
|
|
651
614
|
delete schema.indexes;
|
|
652
615
|
}
|
|
653
|
-
|
|
654
616
|
return schema;
|
|
655
617
|
};
|
|
656
|
-
|
|
657
618
|
class SchemaData {
|
|
658
619
|
constructor(allSchemas = [], protectedFields = {}) {
|
|
659
620
|
this.__data = {};
|
|
@@ -662,7 +623,6 @@ class SchemaData {
|
|
|
662
623
|
if (volatileClasses.includes(schema.className)) {
|
|
663
624
|
return;
|
|
664
625
|
}
|
|
665
|
-
|
|
666
626
|
Object.defineProperty(this, schema.className, {
|
|
667
627
|
get: () => {
|
|
668
628
|
if (!this.__data[schema.className]) {
|
|
@@ -671,22 +631,20 @@ class SchemaData {
|
|
|
671
631
|
data.classLevelPermissions = (0, _deepcopy.default)(schema.classLevelPermissions);
|
|
672
632
|
data.indexes = schema.indexes;
|
|
673
633
|
const classProtectedFields = this.__protectedFields[schema.className];
|
|
674
|
-
|
|
675
634
|
if (classProtectedFields) {
|
|
676
635
|
for (const key in classProtectedFields) {
|
|
677
636
|
const unq = new Set([...(data.classLevelPermissions.protectedFields[key] || []), ...classProtectedFields[key]]);
|
|
678
637
|
data.classLevelPermissions.protectedFields[key] = Array.from(unq);
|
|
679
638
|
}
|
|
680
639
|
}
|
|
681
|
-
|
|
682
640
|
this.__data[schema.className] = data;
|
|
683
641
|
}
|
|
684
|
-
|
|
685
642
|
return this.__data[schema.className];
|
|
686
643
|
}
|
|
687
644
|
});
|
|
688
|
-
});
|
|
645
|
+
});
|
|
689
646
|
|
|
647
|
+
// Inject the in-memory classes
|
|
690
648
|
volatileClasses.forEach(className => {
|
|
691
649
|
Object.defineProperty(this, className, {
|
|
692
650
|
get: () => {
|
|
@@ -702,15 +660,12 @@ class SchemaData {
|
|
|
702
660
|
data.indexes = schema.indexes;
|
|
703
661
|
this.__data[className] = data;
|
|
704
662
|
}
|
|
705
|
-
|
|
706
663
|
return this.__data[className];
|
|
707
664
|
}
|
|
708
665
|
});
|
|
709
666
|
});
|
|
710
667
|
}
|
|
711
|
-
|
|
712
668
|
}
|
|
713
|
-
|
|
714
669
|
const injectDefaultSchema = ({
|
|
715
670
|
className,
|
|
716
671
|
fields,
|
|
@@ -722,14 +677,11 @@ const injectDefaultSchema = ({
|
|
|
722
677
|
fields: _objectSpread(_objectSpread(_objectSpread({}, defaultColumns._Default), defaultColumns[className] || {}), fields),
|
|
723
678
|
classLevelPermissions
|
|
724
679
|
};
|
|
725
|
-
|
|
726
680
|
if (indexes && Object.keys(indexes).length !== 0) {
|
|
727
681
|
defaultSchema.indexes = indexes;
|
|
728
682
|
}
|
|
729
|
-
|
|
730
683
|
return defaultSchema;
|
|
731
684
|
};
|
|
732
|
-
|
|
733
685
|
const _HooksSchema = {
|
|
734
686
|
className: '_Hooks',
|
|
735
687
|
fields: defaultColumns._Hooks
|
|
@@ -742,40 +694,33 @@ const _GraphQLConfigSchema = {
|
|
|
742
694
|
className: '_GraphQLConfig',
|
|
743
695
|
fields: defaultColumns._GraphQLConfig
|
|
744
696
|
};
|
|
745
|
-
|
|
746
697
|
const _PushStatusSchema = convertSchemaToAdapterSchema(injectDefaultSchema({
|
|
747
698
|
className: '_PushStatus',
|
|
748
699
|
fields: {},
|
|
749
700
|
classLevelPermissions: {}
|
|
750
701
|
}));
|
|
751
|
-
|
|
752
702
|
const _JobStatusSchema = convertSchemaToAdapterSchema(injectDefaultSchema({
|
|
753
703
|
className: '_JobStatus',
|
|
754
704
|
fields: {},
|
|
755
705
|
classLevelPermissions: {}
|
|
756
706
|
}));
|
|
757
|
-
|
|
758
707
|
const _JobScheduleSchema = convertSchemaToAdapterSchema(injectDefaultSchema({
|
|
759
708
|
className: '_JobSchedule',
|
|
760
709
|
fields: {},
|
|
761
710
|
classLevelPermissions: {}
|
|
762
711
|
}));
|
|
763
|
-
|
|
764
712
|
const _AudienceSchema = convertSchemaToAdapterSchema(injectDefaultSchema({
|
|
765
713
|
className: '_Audience',
|
|
766
714
|
fields: defaultColumns._Audience,
|
|
767
715
|
classLevelPermissions: {}
|
|
768
716
|
}));
|
|
769
|
-
|
|
770
717
|
const _IdempotencySchema = convertSchemaToAdapterSchema(injectDefaultSchema({
|
|
771
718
|
className: '_Idempotency',
|
|
772
719
|
fields: defaultColumns._Idempotency,
|
|
773
720
|
classLevelPermissions: {}
|
|
774
721
|
}));
|
|
775
|
-
|
|
776
722
|
const VolatileClassesSchemas = [_HooksSchema, _JobStatusSchema, _JobScheduleSchema, _PushStatusSchema, _GlobalConfigSchema, _GraphQLConfigSchema, _AudienceSchema, _IdempotencySchema];
|
|
777
723
|
exports.VolatileClassesSchemas = VolatileClassesSchemas;
|
|
778
|
-
|
|
779
724
|
const dbTypeMatchesObjectType = (dbType, objectType) => {
|
|
780
725
|
if (dbType.type !== objectType.type) return false;
|
|
781
726
|
if (dbType.targetClass !== objectType.targetClass) return false;
|
|
@@ -783,48 +728,39 @@ const dbTypeMatchesObjectType = (dbType, objectType) => {
|
|
|
783
728
|
if (dbType.type === objectType.type) return true;
|
|
784
729
|
return false;
|
|
785
730
|
};
|
|
786
|
-
|
|
787
731
|
const typeToString = type => {
|
|
788
732
|
if (typeof type === 'string') {
|
|
789
733
|
return type;
|
|
790
734
|
}
|
|
791
|
-
|
|
792
735
|
if (type.targetClass) {
|
|
793
736
|
return `${type.type}<${type.targetClass}>`;
|
|
794
737
|
}
|
|
795
|
-
|
|
796
738
|
return `${type.type}`;
|
|
797
|
-
};
|
|
798
|
-
// the mongo format and the Parse format. Soon, this will all be Parse format.
|
|
799
|
-
|
|
739
|
+
};
|
|
800
740
|
|
|
741
|
+
// Stores the entire schema of the app in a weird hybrid format somewhere between
|
|
742
|
+
// the mongo format and the Parse format. Soon, this will all be Parse format.
|
|
801
743
|
class SchemaController {
|
|
802
744
|
constructor(databaseAdapter) {
|
|
803
745
|
this._dbAdapter = databaseAdapter;
|
|
804
746
|
this.schemaData = new SchemaData(_SchemaCache.default.all(), this.protectedFields);
|
|
805
747
|
this.protectedFields = _Config.default.get(Parse.applicationId).protectedFields;
|
|
806
|
-
|
|
807
748
|
const customIds = _Config.default.get(Parse.applicationId).allowCustomObjectId;
|
|
808
|
-
|
|
809
749
|
const customIdRegEx = /^.{1,}$/u; // 1+ chars
|
|
810
|
-
|
|
811
750
|
const autoIdRegEx = /^[a-zA-Z0-9]{1,}$/;
|
|
812
751
|
this.userIdRegEx = customIds ? customIdRegEx : autoIdRegEx;
|
|
813
|
-
|
|
814
752
|
this._dbAdapter.watch(() => {
|
|
815
753
|
this.reloadData({
|
|
816
754
|
clearCache: true
|
|
817
755
|
});
|
|
818
756
|
});
|
|
819
757
|
}
|
|
820
|
-
|
|
821
758
|
reloadData(options = {
|
|
822
759
|
clearCache: false
|
|
823
760
|
}) {
|
|
824
761
|
if (this.reloadDataPromise && !options.clearCache) {
|
|
825
762
|
return this.reloadDataPromise;
|
|
826
763
|
}
|
|
827
|
-
|
|
828
764
|
this.reloadDataPromise = this.getAllClasses(options).then(allSchemas => {
|
|
829
765
|
this.schemaData = new SchemaData(allSchemas, this.protectedFields);
|
|
830
766
|
delete this.reloadDataPromise;
|
|
@@ -835,38 +771,30 @@ class SchemaController {
|
|
|
835
771
|
}).then(() => {});
|
|
836
772
|
return this.reloadDataPromise;
|
|
837
773
|
}
|
|
838
|
-
|
|
839
774
|
getAllClasses(options = {
|
|
840
775
|
clearCache: false
|
|
841
776
|
}) {
|
|
842
777
|
if (options.clearCache) {
|
|
843
778
|
return this.setAllClasses();
|
|
844
779
|
}
|
|
845
|
-
|
|
846
780
|
const cached = _SchemaCache.default.all();
|
|
847
|
-
|
|
848
781
|
if (cached && cached.length) {
|
|
849
782
|
return Promise.resolve(cached);
|
|
850
783
|
}
|
|
851
|
-
|
|
852
784
|
return this.setAllClasses();
|
|
853
785
|
}
|
|
854
|
-
|
|
855
786
|
setAllClasses() {
|
|
856
787
|
return this._dbAdapter.getAllClasses().then(allSchemas => allSchemas.map(injectDefaultSchema)).then(allSchemas => {
|
|
857
788
|
_SchemaCache.default.put(allSchemas);
|
|
858
|
-
|
|
859
789
|
return allSchemas;
|
|
860
790
|
});
|
|
861
791
|
}
|
|
862
|
-
|
|
863
792
|
getOneSchema(className, allowVolatileClasses = false, options = {
|
|
864
793
|
clearCache: false
|
|
865
794
|
}) {
|
|
866
795
|
if (options.clearCache) {
|
|
867
796
|
_SchemaCache.default.clear();
|
|
868
797
|
}
|
|
869
|
-
|
|
870
798
|
if (allowVolatileClasses && volatileClasses.indexOf(className) > -1) {
|
|
871
799
|
const data = this.schemaData[className];
|
|
872
800
|
return Promise.resolve({
|
|
@@ -876,52 +804,44 @@ class SchemaController {
|
|
|
876
804
|
indexes: data.indexes
|
|
877
805
|
});
|
|
878
806
|
}
|
|
879
|
-
|
|
880
807
|
const cached = _SchemaCache.default.get(className);
|
|
881
|
-
|
|
882
808
|
if (cached && !options.clearCache) {
|
|
883
809
|
return Promise.resolve(cached);
|
|
884
810
|
}
|
|
885
|
-
|
|
886
811
|
return this.setAllClasses().then(allSchemas => {
|
|
887
812
|
const oneSchema = allSchemas.find(schema => schema.className === className);
|
|
888
|
-
|
|
889
813
|
if (!oneSchema) {
|
|
890
814
|
return Promise.reject(undefined);
|
|
891
815
|
}
|
|
892
|
-
|
|
893
816
|
return oneSchema;
|
|
894
817
|
});
|
|
895
|
-
}
|
|
818
|
+
}
|
|
819
|
+
|
|
820
|
+
// Create a new class that includes the three default fields.
|
|
896
821
|
// ACL is an implicit column that does not get an entry in the
|
|
897
822
|
// _SCHEMAS database. Returns a promise that resolves with the
|
|
898
823
|
// created schema, in mongo format.
|
|
899
824
|
// on success, and rejects with an error on fail. Ensure you
|
|
900
825
|
// have authorization (master key, or client class creation
|
|
901
826
|
// enabled) before calling this function.
|
|
902
|
-
|
|
903
|
-
|
|
904
827
|
async addClassIfNotExists(className, fields = {}, classLevelPermissions, indexes = {}) {
|
|
905
828
|
var validationError = this.validateNewClass(className, fields, classLevelPermissions);
|
|
906
|
-
|
|
907
829
|
if (validationError) {
|
|
908
830
|
if (validationError instanceof Parse.Error) {
|
|
909
831
|
return Promise.reject(validationError);
|
|
910
832
|
} else if (validationError.code && validationError.error) {
|
|
911
833
|
return Promise.reject(new Parse.Error(validationError.code, validationError.error));
|
|
912
834
|
}
|
|
913
|
-
|
|
914
835
|
return Promise.reject(validationError);
|
|
915
836
|
}
|
|
916
|
-
|
|
917
837
|
try {
|
|
918
838
|
const adapterSchema = await this._dbAdapter.createClass(className, convertSchemaToAdapterSchema({
|
|
919
839
|
fields,
|
|
920
840
|
classLevelPermissions,
|
|
921
841
|
indexes,
|
|
922
842
|
className
|
|
923
|
-
}));
|
|
924
|
-
|
|
843
|
+
}));
|
|
844
|
+
// TODO: Remove by updating schema cache directly
|
|
925
845
|
await this.reloadData({
|
|
926
846
|
clearCache: true
|
|
927
847
|
});
|
|
@@ -935,17 +855,14 @@ class SchemaController {
|
|
|
935
855
|
}
|
|
936
856
|
}
|
|
937
857
|
}
|
|
938
|
-
|
|
939
858
|
updateClass(className, submittedFields, classLevelPermissions, indexes, database) {
|
|
940
859
|
return this.getOneSchema(className).then(schema => {
|
|
941
860
|
const existingFields = schema.fields;
|
|
942
861
|
Object.keys(submittedFields).forEach(name => {
|
|
943
862
|
const field = submittedFields[name];
|
|
944
|
-
|
|
945
863
|
if (existingFields[name] && existingFields[name].type !== field.type && field.__op !== 'Delete') {
|
|
946
864
|
throw new Parse.Error(255, `Field ${name} exists, cannot update.`);
|
|
947
865
|
}
|
|
948
|
-
|
|
949
866
|
if (!existingFields[name] && field.__op === 'Delete') {
|
|
950
867
|
throw new Parse.Error(255, `Field ${name} does not exist, cannot delete.`);
|
|
951
868
|
}
|
|
@@ -956,13 +873,12 @@ class SchemaController {
|
|
|
956
873
|
const defaultFields = defaultColumns[className] || defaultColumns._Default;
|
|
957
874
|
const fullNewSchema = Object.assign({}, newSchema, defaultFields);
|
|
958
875
|
const validationError = this.validateSchemaData(className, newSchema, classLevelPermissions, Object.keys(existingFields));
|
|
959
|
-
|
|
960
876
|
if (validationError) {
|
|
961
877
|
throw new Parse.Error(validationError.code, validationError.error);
|
|
962
|
-
}
|
|
963
|
-
// Do all deletions first, then a single save to _SCHEMA collection to handle all additions.
|
|
964
|
-
|
|
878
|
+
}
|
|
965
879
|
|
|
880
|
+
// Finally we have checked to make sure the request is valid and we can start deleting fields.
|
|
881
|
+
// Do all deletions first, then a single save to _SCHEMA collection to handle all additions.
|
|
966
882
|
const deletedFields = [];
|
|
967
883
|
const insertedFields = [];
|
|
968
884
|
Object.keys(submittedFields).forEach(fieldName => {
|
|
@@ -973,11 +889,9 @@ class SchemaController {
|
|
|
973
889
|
}
|
|
974
890
|
});
|
|
975
891
|
let deletePromise = Promise.resolve();
|
|
976
|
-
|
|
977
892
|
if (deletedFields.length > 0) {
|
|
978
893
|
deletePromise = this.deleteFields(deletedFields, className, database);
|
|
979
894
|
}
|
|
980
|
-
|
|
981
895
|
let enforceFields = [];
|
|
982
896
|
return deletePromise // Delete Everything
|
|
983
897
|
.then(() => this.reloadData({
|
|
@@ -994,7 +908,8 @@ class SchemaController {
|
|
|
994
908
|
return this.setPermissions(className, classLevelPermissions, newSchema);
|
|
995
909
|
}).then(() => this._dbAdapter.setIndexesWithSchemaFormat(className, indexes, schema.indexes, fullNewSchema)).then(() => this.reloadData({
|
|
996
910
|
clearCache: true
|
|
997
|
-
}))
|
|
911
|
+
}))
|
|
912
|
+
//TODO: Move this logic into the database adapter
|
|
998
913
|
.then(() => {
|
|
999
914
|
this.ensureFields(enforceFields);
|
|
1000
915
|
const schema = this.schemaData[className];
|
|
@@ -1003,11 +918,9 @@ class SchemaController {
|
|
|
1003
918
|
fields: schema.fields,
|
|
1004
919
|
classLevelPermissions: schema.classLevelPermissions
|
|
1005
920
|
};
|
|
1006
|
-
|
|
1007
921
|
if (schema.indexes && Object.keys(schema.indexes).length !== 0) {
|
|
1008
922
|
reloadedSchema.indexes = schema.indexes;
|
|
1009
923
|
}
|
|
1010
|
-
|
|
1011
924
|
return reloadedSchema;
|
|
1012
925
|
});
|
|
1013
926
|
}).catch(error => {
|
|
@@ -1017,17 +930,17 @@ class SchemaController {
|
|
|
1017
930
|
throw error;
|
|
1018
931
|
}
|
|
1019
932
|
});
|
|
1020
|
-
}
|
|
1021
|
-
// object or fails with a reason.
|
|
1022
|
-
|
|
933
|
+
}
|
|
1023
934
|
|
|
935
|
+
// Returns a promise that resolves successfully to the new schema
|
|
936
|
+
// object or fails with a reason.
|
|
1024
937
|
enforceClassExists(className) {
|
|
1025
938
|
if (this.schemaData[className]) {
|
|
1026
939
|
return Promise.resolve(this);
|
|
1027
|
-
}
|
|
1028
|
-
|
|
1029
|
-
|
|
1030
|
-
|
|
940
|
+
}
|
|
941
|
+
// We don't have this class. Update the schema
|
|
942
|
+
return (
|
|
943
|
+
// The schema update succeeded. Reload the schema
|
|
1031
944
|
this.addClassIfNotExists(className).catch(() => {
|
|
1032
945
|
// The schema update failed. This can be okay - it might
|
|
1033
946
|
// have failed because there's a race condition and a different
|
|
@@ -1049,22 +962,18 @@ class SchemaController {
|
|
|
1049
962
|
})
|
|
1050
963
|
);
|
|
1051
964
|
}
|
|
1052
|
-
|
|
1053
965
|
validateNewClass(className, fields = {}, classLevelPermissions) {
|
|
1054
966
|
if (this.schemaData[className]) {
|
|
1055
967
|
throw new Parse.Error(Parse.Error.INVALID_CLASS_NAME, `Class ${className} already exists.`);
|
|
1056
968
|
}
|
|
1057
|
-
|
|
1058
969
|
if (!classNameIsValid(className)) {
|
|
1059
970
|
return {
|
|
1060
971
|
code: Parse.Error.INVALID_CLASS_NAME,
|
|
1061
972
|
error: invalidClassNameMessage(className)
|
|
1062
973
|
};
|
|
1063
974
|
}
|
|
1064
|
-
|
|
1065
975
|
return this.validateSchemaData(className, fields, classLevelPermissions, []);
|
|
1066
976
|
}
|
|
1067
|
-
|
|
1068
977
|
validateSchemaData(className, fields, classLevelPermissions, existingFieldNames) {
|
|
1069
978
|
for (const fieldName in fields) {
|
|
1070
979
|
if (existingFieldNames.indexOf(fieldName) < 0) {
|
|
@@ -1074,24 +983,20 @@ class SchemaController {
|
|
|
1074
983
|
error: 'invalid field name: ' + fieldName
|
|
1075
984
|
};
|
|
1076
985
|
}
|
|
1077
|
-
|
|
1078
986
|
if (!fieldNameIsValidForClass(fieldName, className)) {
|
|
1079
987
|
return {
|
|
1080
988
|
code: 136,
|
|
1081
989
|
error: 'field ' + fieldName + ' cannot be added'
|
|
1082
990
|
};
|
|
1083
991
|
}
|
|
1084
|
-
|
|
1085
992
|
const fieldType = fields[fieldName];
|
|
1086
993
|
const error = fieldTypeIsInvalid(fieldType);
|
|
1087
994
|
if (error) return {
|
|
1088
995
|
code: error.code,
|
|
1089
996
|
error: error.message
|
|
1090
997
|
};
|
|
1091
|
-
|
|
1092
998
|
if (fieldType.defaultValue !== undefined) {
|
|
1093
999
|
let defaultValueType = getType(fieldType.defaultValue);
|
|
1094
|
-
|
|
1095
1000
|
if (typeof defaultValueType === 'string') {
|
|
1096
1001
|
defaultValueType = {
|
|
1097
1002
|
type: defaultValueType
|
|
@@ -1102,7 +1007,6 @@ class SchemaController {
|
|
|
1102
1007
|
error: `The 'default value' option is not applicable for ${typeToString(fieldType)}`
|
|
1103
1008
|
};
|
|
1104
1009
|
}
|
|
1105
|
-
|
|
1106
1010
|
if (!dbTypeMatchesObjectType(fieldType, defaultValueType)) {
|
|
1107
1011
|
return {
|
|
1108
1012
|
code: Parse.Error.INCORRECT_TYPE,
|
|
@@ -1119,106 +1023,88 @@ class SchemaController {
|
|
|
1119
1023
|
}
|
|
1120
1024
|
}
|
|
1121
1025
|
}
|
|
1122
|
-
|
|
1123
1026
|
for (const fieldName in defaultColumns[className]) {
|
|
1124
1027
|
fields[fieldName] = defaultColumns[className][fieldName];
|
|
1125
1028
|
}
|
|
1126
|
-
|
|
1127
1029
|
const geoPoints = Object.keys(fields).filter(key => fields[key] && fields[key].type === 'GeoPoint');
|
|
1128
|
-
|
|
1129
1030
|
if (geoPoints.length > 1) {
|
|
1130
1031
|
return {
|
|
1131
1032
|
code: Parse.Error.INCORRECT_TYPE,
|
|
1132
1033
|
error: 'currently, only one GeoPoint field may exist in an object. Adding ' + geoPoints[1] + ' when ' + geoPoints[0] + ' already exists.'
|
|
1133
1034
|
};
|
|
1134
1035
|
}
|
|
1135
|
-
|
|
1136
1036
|
validateCLP(classLevelPermissions, fields, this.userIdRegEx);
|
|
1137
|
-
}
|
|
1138
|
-
|
|
1037
|
+
}
|
|
1139
1038
|
|
|
1039
|
+
// Sets the Class-level permissions for a given className, which must exist.
|
|
1140
1040
|
async setPermissions(className, perms, newSchema) {
|
|
1141
1041
|
if (typeof perms === 'undefined') {
|
|
1142
1042
|
return Promise.resolve();
|
|
1143
1043
|
}
|
|
1144
|
-
|
|
1145
1044
|
validateCLP(perms, newSchema, this.userIdRegEx);
|
|
1146
1045
|
await this._dbAdapter.setClassLevelPermissions(className, perms);
|
|
1147
|
-
|
|
1148
1046
|
const cached = _SchemaCache.default.get(className);
|
|
1149
|
-
|
|
1150
1047
|
if (cached) {
|
|
1151
1048
|
cached.classLevelPermissions = perms;
|
|
1152
1049
|
}
|
|
1153
|
-
}
|
|
1050
|
+
}
|
|
1051
|
+
|
|
1052
|
+
// Returns a promise that resolves successfully to the new schema
|
|
1154
1053
|
// object if the provided className-fieldName-type tuple is valid.
|
|
1155
1054
|
// The className must already be validated.
|
|
1156
1055
|
// If 'freeze' is true, refuse to update the schema for this field.
|
|
1157
|
-
|
|
1158
|
-
|
|
1159
1056
|
enforceFieldExists(className, fieldName, type, isValidation) {
|
|
1160
1057
|
if (fieldName.indexOf('.') > 0) {
|
|
1161
1058
|
// subdocument key (x.y) => ok if x is of type 'object'
|
|
1162
1059
|
fieldName = fieldName.split('.')[0];
|
|
1163
1060
|
type = 'Object';
|
|
1164
1061
|
}
|
|
1165
|
-
|
|
1166
1062
|
if (!fieldNameIsValid(fieldName, className)) {
|
|
1167
1063
|
throw new Parse.Error(Parse.Error.INVALID_KEY_NAME, `Invalid field name: ${fieldName}.`);
|
|
1168
|
-
}
|
|
1169
|
-
|
|
1064
|
+
}
|
|
1170
1065
|
|
|
1066
|
+
// If someone tries to create a new field with null/undefined as the value, return;
|
|
1171
1067
|
if (!type) {
|
|
1172
1068
|
return undefined;
|
|
1173
1069
|
}
|
|
1174
|
-
|
|
1175
1070
|
const expectedType = this.getExpectedType(className, fieldName);
|
|
1176
|
-
|
|
1177
1071
|
if (typeof type === 'string') {
|
|
1178
1072
|
type = {
|
|
1179
1073
|
type
|
|
1180
1074
|
};
|
|
1181
1075
|
}
|
|
1182
|
-
|
|
1183
1076
|
if (type.defaultValue !== undefined) {
|
|
1184
1077
|
let defaultValueType = getType(type.defaultValue);
|
|
1185
|
-
|
|
1186
1078
|
if (typeof defaultValueType === 'string') {
|
|
1187
1079
|
defaultValueType = {
|
|
1188
1080
|
type: defaultValueType
|
|
1189
1081
|
};
|
|
1190
1082
|
}
|
|
1191
|
-
|
|
1192
1083
|
if (!dbTypeMatchesObjectType(type, defaultValueType)) {
|
|
1193
1084
|
throw new Parse.Error(Parse.Error.INCORRECT_TYPE, `schema mismatch for ${className}.${fieldName} default value; expected ${typeToString(type)} but got ${typeToString(defaultValueType)}`);
|
|
1194
1085
|
}
|
|
1195
1086
|
}
|
|
1196
|
-
|
|
1197
1087
|
if (expectedType) {
|
|
1198
1088
|
if (!dbTypeMatchesObjectType(expectedType, type)) {
|
|
1199
1089
|
throw new Parse.Error(Parse.Error.INCORRECT_TYPE, `schema mismatch for ${className}.${fieldName}; expected ${typeToString(expectedType)} but got ${typeToString(type)}`);
|
|
1200
|
-
}
|
|
1090
|
+
}
|
|
1091
|
+
// If type options do not change
|
|
1201
1092
|
// we can safely return
|
|
1202
|
-
|
|
1203
|
-
|
|
1204
1093
|
if (isValidation || JSON.stringify(expectedType) === JSON.stringify(type)) {
|
|
1205
1094
|
return undefined;
|
|
1206
|
-
}
|
|
1095
|
+
}
|
|
1096
|
+
// Field options are may be changed
|
|
1207
1097
|
// ensure to have an update to date schema field
|
|
1208
|
-
|
|
1209
|
-
|
|
1210
1098
|
return this._dbAdapter.updateFieldOptions(className, fieldName, type);
|
|
1211
1099
|
}
|
|
1212
|
-
|
|
1213
1100
|
return this._dbAdapter.addFieldIfNotExists(className, fieldName, type).catch(error => {
|
|
1214
1101
|
if (error.code == Parse.Error.INCORRECT_TYPE) {
|
|
1215
1102
|
// Make sure that we throw errors when it is appropriate to do so.
|
|
1216
1103
|
throw error;
|
|
1217
|
-
}
|
|
1104
|
+
}
|
|
1105
|
+
// The update failed. This can be okay - it might have been a race
|
|
1218
1106
|
// condition where another client updated the schema in the same
|
|
1219
1107
|
// way that we wanted to. So, just reload the schema
|
|
1220
|
-
|
|
1221
|
-
|
|
1222
1108
|
return Promise.resolve();
|
|
1223
1109
|
}).then(() => {
|
|
1224
1110
|
return {
|
|
@@ -1228,7 +1114,6 @@ class SchemaController {
|
|
|
1228
1114
|
};
|
|
1229
1115
|
});
|
|
1230
1116
|
}
|
|
1231
|
-
|
|
1232
1117
|
ensureFields(fields) {
|
|
1233
1118
|
for (let i = 0; i < fields.length; i += 1) {
|
|
1234
1119
|
const {
|
|
@@ -1239,42 +1124,38 @@ class SchemaController {
|
|
|
1239
1124
|
type
|
|
1240
1125
|
} = fields[i];
|
|
1241
1126
|
const expectedType = this.getExpectedType(className, fieldName);
|
|
1242
|
-
|
|
1243
1127
|
if (typeof type === 'string') {
|
|
1244
1128
|
type = {
|
|
1245
1129
|
type: type
|
|
1246
1130
|
};
|
|
1247
1131
|
}
|
|
1248
|
-
|
|
1249
1132
|
if (!expectedType || !dbTypeMatchesObjectType(expectedType, type)) {
|
|
1250
1133
|
throw new Parse.Error(Parse.Error.INVALID_JSON, `Could not add field ${fieldName}`);
|
|
1251
1134
|
}
|
|
1252
1135
|
}
|
|
1253
|
-
}
|
|
1254
|
-
|
|
1136
|
+
}
|
|
1255
1137
|
|
|
1138
|
+
// maintain compatibility
|
|
1256
1139
|
deleteField(fieldName, className, database) {
|
|
1257
1140
|
return this.deleteFields([fieldName], className, database);
|
|
1258
|
-
}
|
|
1141
|
+
}
|
|
1142
|
+
|
|
1143
|
+
// Delete fields, and remove that data from all objects. This is intended
|
|
1259
1144
|
// to remove unused fields, if other writers are writing objects that include
|
|
1260
1145
|
// this field, the field may reappear. Returns a Promise that resolves with
|
|
1261
1146
|
// no object on success, or rejects with { code, error } on failure.
|
|
1262
1147
|
// Passing the database and prefix is necessary in order to drop relation collections
|
|
1263
1148
|
// and remove fields from objects. Ideally the database would belong to
|
|
1264
1149
|
// a database adapter and this function would close over it or access it via member.
|
|
1265
|
-
|
|
1266
|
-
|
|
1267
1150
|
deleteFields(fieldNames, className, database) {
|
|
1268
1151
|
if (!classNameIsValid(className)) {
|
|
1269
1152
|
throw new Parse.Error(Parse.Error.INVALID_CLASS_NAME, invalidClassNameMessage(className));
|
|
1270
1153
|
}
|
|
1271
|
-
|
|
1272
1154
|
fieldNames.forEach(fieldName => {
|
|
1273
1155
|
if (!fieldNameIsValid(fieldName, className)) {
|
|
1274
1156
|
throw new Parse.Error(Parse.Error.INVALID_KEY_NAME, `invalid field name: ${fieldName}`);
|
|
1275
|
-
}
|
|
1276
|
-
|
|
1277
|
-
|
|
1157
|
+
}
|
|
1158
|
+
//Don't allow deleting the default fields.
|
|
1278
1159
|
if (!fieldNameIsValidForClass(fieldName, className)) {
|
|
1279
1160
|
throw new Parse.Error(136, `field ${fieldName} cannot be changed`);
|
|
1280
1161
|
}
|
|
@@ -1293,174 +1174,145 @@ class SchemaController {
|
|
|
1293
1174
|
throw new Parse.Error(255, `Field ${fieldName} does not exist, cannot delete.`);
|
|
1294
1175
|
}
|
|
1295
1176
|
});
|
|
1296
|
-
|
|
1297
1177
|
const schemaFields = _objectSpread({}, schema.fields);
|
|
1298
|
-
|
|
1299
1178
|
return database.adapter.deleteFields(className, schema, fieldNames).then(() => {
|
|
1300
1179
|
return Promise.all(fieldNames.map(fieldName => {
|
|
1301
1180
|
const field = schemaFields[fieldName];
|
|
1302
|
-
|
|
1303
1181
|
if (field && field.type === 'Relation') {
|
|
1304
1182
|
//For relations, drop the _Join table
|
|
1305
1183
|
return database.adapter.deleteClass(`_Join:${fieldName}:${className}`);
|
|
1306
1184
|
}
|
|
1307
|
-
|
|
1308
1185
|
return Promise.resolve();
|
|
1309
1186
|
}));
|
|
1310
1187
|
});
|
|
1311
1188
|
}).then(() => {
|
|
1312
1189
|
_SchemaCache.default.clear();
|
|
1313
1190
|
});
|
|
1314
|
-
}
|
|
1191
|
+
}
|
|
1192
|
+
|
|
1193
|
+
// Validates an object provided in REST format.
|
|
1315
1194
|
// Returns a promise that resolves to the new schema if this object is
|
|
1316
1195
|
// valid.
|
|
1317
|
-
|
|
1318
|
-
|
|
1319
1196
|
async validateObject(className, object, query) {
|
|
1320
1197
|
let geocount = 0;
|
|
1321
1198
|
const schema = await this.enforceClassExists(className);
|
|
1322
1199
|
const promises = [];
|
|
1323
|
-
|
|
1324
1200
|
for (const fieldName in object) {
|
|
1325
1201
|
if (object[fieldName] && getType(object[fieldName]) === 'GeoPoint') {
|
|
1326
1202
|
geocount++;
|
|
1327
1203
|
}
|
|
1328
|
-
|
|
1329
1204
|
if (geocount > 1) {
|
|
1330
1205
|
return Promise.reject(new Parse.Error(Parse.Error.INCORRECT_TYPE, 'there can only be one geopoint field in a class'));
|
|
1331
1206
|
}
|
|
1332
1207
|
}
|
|
1333
|
-
|
|
1334
1208
|
for (const fieldName in object) {
|
|
1335
1209
|
if (object[fieldName] === undefined) {
|
|
1336
1210
|
continue;
|
|
1337
1211
|
}
|
|
1338
|
-
|
|
1339
1212
|
const expected = getType(object[fieldName]);
|
|
1340
|
-
|
|
1341
1213
|
if (!expected) {
|
|
1342
1214
|
continue;
|
|
1343
1215
|
}
|
|
1344
|
-
|
|
1345
1216
|
if (fieldName === 'ACL') {
|
|
1346
1217
|
// Every object has ACL implicitly.
|
|
1347
1218
|
continue;
|
|
1348
1219
|
}
|
|
1349
|
-
|
|
1350
1220
|
promises.push(schema.enforceFieldExists(className, fieldName, expected, true));
|
|
1351
1221
|
}
|
|
1352
|
-
|
|
1353
1222
|
const results = await Promise.all(promises);
|
|
1354
1223
|
const enforceFields = results.filter(result => !!result);
|
|
1355
|
-
|
|
1356
1224
|
if (enforceFields.length !== 0) {
|
|
1357
1225
|
// TODO: Remove by updating schema cache directly
|
|
1358
1226
|
await this.reloadData({
|
|
1359
1227
|
clearCache: true
|
|
1360
1228
|
});
|
|
1361
1229
|
}
|
|
1362
|
-
|
|
1363
1230
|
this.ensureFields(enforceFields);
|
|
1364
1231
|
const promise = Promise.resolve(schema);
|
|
1365
1232
|
return thenValidateRequiredColumns(promise, className, object, query);
|
|
1366
|
-
}
|
|
1367
|
-
|
|
1233
|
+
}
|
|
1368
1234
|
|
|
1235
|
+
// Validates that all the properties are set for the object
|
|
1369
1236
|
validateRequiredColumns(className, object, query) {
|
|
1370
1237
|
const columns = requiredColumns.write[className];
|
|
1371
|
-
|
|
1372
1238
|
if (!columns || columns.length == 0) {
|
|
1373
1239
|
return Promise.resolve(this);
|
|
1374
1240
|
}
|
|
1375
|
-
|
|
1376
1241
|
const missingColumns = columns.filter(function (column) {
|
|
1377
1242
|
if (query && query.objectId) {
|
|
1378
1243
|
if (object[column] && typeof object[column] === 'object') {
|
|
1379
1244
|
// Trying to delete a required column
|
|
1380
1245
|
return object[column].__op == 'Delete';
|
|
1381
|
-
}
|
|
1382
|
-
|
|
1383
|
-
|
|
1246
|
+
}
|
|
1247
|
+
// Not trying to do anything there
|
|
1384
1248
|
return false;
|
|
1385
1249
|
}
|
|
1386
|
-
|
|
1387
1250
|
return !object[column];
|
|
1388
1251
|
});
|
|
1389
|
-
|
|
1390
1252
|
if (missingColumns.length > 0) {
|
|
1391
1253
|
throw new Parse.Error(Parse.Error.INCORRECT_TYPE, missingColumns[0] + ' is required.');
|
|
1392
1254
|
}
|
|
1393
|
-
|
|
1394
1255
|
return Promise.resolve(this);
|
|
1395
1256
|
}
|
|
1396
|
-
|
|
1397
1257
|
testPermissionsForClassName(className, aclGroup, operation) {
|
|
1398
1258
|
return SchemaController.testPermissions(this.getClassLevelPermissions(className), aclGroup, operation);
|
|
1399
|
-
}
|
|
1400
|
-
|
|
1259
|
+
}
|
|
1401
1260
|
|
|
1261
|
+
// Tests that the class level permission let pass the operation for a given aclGroup
|
|
1402
1262
|
static testPermissions(classPermissions, aclGroup, operation) {
|
|
1403
1263
|
if (!classPermissions || !classPermissions[operation]) {
|
|
1404
1264
|
return true;
|
|
1405
1265
|
}
|
|
1406
|
-
|
|
1407
1266
|
const perms = classPermissions[operation];
|
|
1408
|
-
|
|
1409
1267
|
if (perms['*']) {
|
|
1410
1268
|
return true;
|
|
1411
|
-
}
|
|
1412
|
-
|
|
1413
|
-
|
|
1269
|
+
}
|
|
1270
|
+
// Check permissions against the aclGroup provided (array of userId/roles)
|
|
1414
1271
|
if (aclGroup.some(acl => {
|
|
1415
1272
|
return perms[acl] === true;
|
|
1416
1273
|
})) {
|
|
1417
1274
|
return true;
|
|
1418
1275
|
}
|
|
1419
|
-
|
|
1420
1276
|
return false;
|
|
1421
|
-
}
|
|
1422
|
-
|
|
1277
|
+
}
|
|
1423
1278
|
|
|
1279
|
+
// Validates an operation passes class-level-permissions set in the schema
|
|
1424
1280
|
static validatePermission(classPermissions, className, aclGroup, operation, action) {
|
|
1425
1281
|
if (SchemaController.testPermissions(classPermissions, aclGroup, operation)) {
|
|
1426
1282
|
return Promise.resolve();
|
|
1427
1283
|
}
|
|
1428
|
-
|
|
1429
1284
|
if (!classPermissions || !classPermissions[operation]) {
|
|
1430
1285
|
return true;
|
|
1431
1286
|
}
|
|
1432
|
-
|
|
1433
|
-
|
|
1287
|
+
const perms = classPermissions[operation];
|
|
1288
|
+
// If only for authenticated users
|
|
1434
1289
|
// make sure we have an aclGroup
|
|
1435
|
-
|
|
1436
1290
|
if (perms['requiresAuthentication']) {
|
|
1437
1291
|
// If aclGroup has * (public)
|
|
1438
1292
|
if (!aclGroup || aclGroup.length == 0) {
|
|
1439
1293
|
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Permission denied, user needs to be authenticated.');
|
|
1440
1294
|
} else if (aclGroup.indexOf('*') > -1 && aclGroup.length == 1) {
|
|
1441
1295
|
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Permission denied, user needs to be authenticated.');
|
|
1442
|
-
}
|
|
1296
|
+
}
|
|
1297
|
+
// requiresAuthentication passed, just move forward
|
|
1443
1298
|
// probably would be wise at some point to rename to 'authenticatedUser'
|
|
1444
|
-
|
|
1445
|
-
|
|
1446
1299
|
return Promise.resolve();
|
|
1447
|
-
}
|
|
1448
|
-
// And handle those later
|
|
1449
|
-
|
|
1300
|
+
}
|
|
1450
1301
|
|
|
1451
|
-
|
|
1302
|
+
// No matching CLP, let's check the Pointer permissions
|
|
1303
|
+
// And handle those later
|
|
1304
|
+
const permissionField = ['get', 'find', 'count'].indexOf(operation) > -1 ? 'readUserFields' : 'writeUserFields';
|
|
1452
1305
|
|
|
1306
|
+
// Reject create when write lockdown
|
|
1453
1307
|
if (permissionField == 'writeUserFields' && operation == 'create') {
|
|
1454
1308
|
throw new Parse.Error(Parse.Error.OPERATION_FORBIDDEN, `Permission denied for action ${operation} on class ${className}.`);
|
|
1455
|
-
}
|
|
1456
|
-
|
|
1309
|
+
}
|
|
1457
1310
|
|
|
1311
|
+
// Process the readUserFields later
|
|
1458
1312
|
if (Array.isArray(classPermissions[permissionField]) && classPermissions[permissionField].length > 0) {
|
|
1459
1313
|
return Promise.resolve();
|
|
1460
1314
|
}
|
|
1461
|
-
|
|
1462
1315
|
const pointerFields = classPermissions[operation].pointerFields;
|
|
1463
|
-
|
|
1464
1316
|
if (Array.isArray(pointerFields) && pointerFields.length > 0) {
|
|
1465
1317
|
// any op except 'addField as part of create' is ok.
|
|
1466
1318
|
if (operation !== 'addField' || action === 'update') {
|
|
@@ -1468,138 +1320,118 @@ class SchemaController {
|
|
|
1468
1320
|
return Promise.resolve();
|
|
1469
1321
|
}
|
|
1470
1322
|
}
|
|
1471
|
-
|
|
1472
1323
|
throw new Parse.Error(Parse.Error.OPERATION_FORBIDDEN, `Permission denied for action ${operation} on class ${className}.`);
|
|
1473
|
-
}
|
|
1474
|
-
|
|
1324
|
+
}
|
|
1475
1325
|
|
|
1326
|
+
// Validates an operation passes class-level-permissions set in the schema
|
|
1476
1327
|
validatePermission(className, aclGroup, operation, action) {
|
|
1477
1328
|
return SchemaController.validatePermission(this.getClassLevelPermissions(className), className, aclGroup, operation, action);
|
|
1478
1329
|
}
|
|
1479
|
-
|
|
1480
1330
|
getClassLevelPermissions(className) {
|
|
1481
1331
|
return this.schemaData[className] && this.schemaData[className].classLevelPermissions;
|
|
1482
|
-
}
|
|
1483
|
-
// or undefined if the schema is not set
|
|
1484
|
-
|
|
1332
|
+
}
|
|
1485
1333
|
|
|
1334
|
+
// Returns the expected type for a className+key combination
|
|
1335
|
+
// or undefined if the schema is not set
|
|
1486
1336
|
getExpectedType(className, fieldName) {
|
|
1487
1337
|
if (this.schemaData[className]) {
|
|
1488
1338
|
const expectedType = this.schemaData[className].fields[fieldName];
|
|
1489
1339
|
return expectedType === 'map' ? 'Object' : expectedType;
|
|
1490
1340
|
}
|
|
1491
|
-
|
|
1492
1341
|
return undefined;
|
|
1493
|
-
}
|
|
1494
|
-
|
|
1342
|
+
}
|
|
1495
1343
|
|
|
1344
|
+
// Checks if a given class is in the schema.
|
|
1496
1345
|
hasClass(className) {
|
|
1497
1346
|
if (this.schemaData[className]) {
|
|
1498
1347
|
return Promise.resolve(true);
|
|
1499
1348
|
}
|
|
1500
|
-
|
|
1501
1349
|
return this.reloadData().then(() => !!this.schemaData[className]);
|
|
1502
1350
|
}
|
|
1351
|
+
}
|
|
1503
1352
|
|
|
1504
|
-
|
|
1505
|
-
|
|
1506
|
-
|
|
1353
|
+
// Returns a promise for a new Schema.
|
|
1507
1354
|
exports.SchemaController = exports.default = SchemaController;
|
|
1508
|
-
|
|
1509
1355
|
const load = (dbAdapter, options) => {
|
|
1510
1356
|
const schema = new SchemaController(dbAdapter);
|
|
1511
1357
|
return schema.reloadData(options).then(() => schema);
|
|
1512
|
-
};
|
|
1358
|
+
};
|
|
1359
|
+
|
|
1360
|
+
// Builds a new schema (in schema API response format) out of an
|
|
1513
1361
|
// existing mongo schema + a schemas API put request. This response
|
|
1514
1362
|
// does not include the default fields, as it is intended to be passed
|
|
1515
1363
|
// to mongoSchemaFromFieldsAndClassName. No validation is done here, it
|
|
1516
1364
|
// is done in mongoSchemaFromFieldsAndClassName.
|
|
1517
|
-
|
|
1518
|
-
|
|
1519
1365
|
exports.load = load;
|
|
1520
|
-
|
|
1521
1366
|
function buildMergedSchemaObject(existingFields, putRequest) {
|
|
1522
|
-
const newSchema = {};
|
|
1523
|
-
|
|
1367
|
+
const newSchema = {};
|
|
1368
|
+
// -disable-next
|
|
1524
1369
|
const sysSchemaField = Object.keys(defaultColumns).indexOf(existingFields._id) === -1 ? [] : Object.keys(defaultColumns[existingFields._id]);
|
|
1525
|
-
|
|
1526
1370
|
for (const oldField in existingFields) {
|
|
1527
1371
|
if (oldField !== '_id' && oldField !== 'ACL' && oldField !== 'updatedAt' && oldField !== 'createdAt' && oldField !== 'objectId') {
|
|
1528
1372
|
if (sysSchemaField.length > 0 && sysSchemaField.indexOf(oldField) !== -1) {
|
|
1529
1373
|
continue;
|
|
1530
1374
|
}
|
|
1531
|
-
|
|
1532
1375
|
const fieldIsDeleted = putRequest[oldField] && putRequest[oldField].__op === 'Delete';
|
|
1533
|
-
|
|
1534
1376
|
if (!fieldIsDeleted) {
|
|
1535
1377
|
newSchema[oldField] = existingFields[oldField];
|
|
1536
1378
|
}
|
|
1537
1379
|
}
|
|
1538
1380
|
}
|
|
1539
|
-
|
|
1540
1381
|
for (const newField in putRequest) {
|
|
1541
1382
|
if (newField !== 'objectId' && putRequest[newField].__op !== 'Delete') {
|
|
1542
1383
|
if (sysSchemaField.length > 0 && sysSchemaField.indexOf(newField) !== -1) {
|
|
1543
1384
|
continue;
|
|
1544
1385
|
}
|
|
1545
|
-
|
|
1546
1386
|
newSchema[newField] = putRequest[newField];
|
|
1547
1387
|
}
|
|
1548
1388
|
}
|
|
1549
|
-
|
|
1550
1389
|
return newSchema;
|
|
1551
|
-
}
|
|
1552
|
-
// validates this field once the schema loads.
|
|
1553
|
-
|
|
1390
|
+
}
|
|
1554
1391
|
|
|
1392
|
+
// Given a schema promise, construct another schema promise that
|
|
1393
|
+
// validates this field once the schema loads.
|
|
1555
1394
|
function thenValidateRequiredColumns(schemaPromise, className, object, query) {
|
|
1556
1395
|
return schemaPromise.then(schema => {
|
|
1557
1396
|
return schema.validateRequiredColumns(className, object, query);
|
|
1558
1397
|
});
|
|
1559
|
-
}
|
|
1398
|
+
}
|
|
1399
|
+
|
|
1400
|
+
// Gets the type from a REST API formatted object, where 'type' is
|
|
1560
1401
|
// extended past javascript types to include the rest of the Parse
|
|
1561
1402
|
// type system.
|
|
1562
1403
|
// The output should be a valid schema value.
|
|
1563
1404
|
// TODO: ensure that this is compatible with the format used in Open DB
|
|
1564
|
-
|
|
1565
|
-
|
|
1566
1405
|
function getType(obj) {
|
|
1567
1406
|
const type = typeof obj;
|
|
1568
|
-
|
|
1569
1407
|
switch (type) {
|
|
1570
1408
|
case 'boolean':
|
|
1571
1409
|
return 'Boolean';
|
|
1572
|
-
|
|
1573
1410
|
case 'string':
|
|
1574
1411
|
return 'String';
|
|
1575
|
-
|
|
1576
1412
|
case 'number':
|
|
1577
1413
|
return 'Number';
|
|
1578
|
-
|
|
1579
1414
|
case 'map':
|
|
1580
1415
|
case 'object':
|
|
1581
1416
|
if (!obj) {
|
|
1582
1417
|
return undefined;
|
|
1583
1418
|
}
|
|
1584
|
-
|
|
1585
1419
|
return getObjectType(obj);
|
|
1586
|
-
|
|
1587
1420
|
case 'function':
|
|
1588
1421
|
case 'symbol':
|
|
1589
1422
|
case 'undefined':
|
|
1590
1423
|
default:
|
|
1591
1424
|
throw 'bad obj: ' + obj;
|
|
1592
1425
|
}
|
|
1593
|
-
}
|
|
1426
|
+
}
|
|
1427
|
+
|
|
1428
|
+
// This gets the type for non-JSON types like pointers and files, but
|
|
1594
1429
|
// also gets the appropriate type for $ operators.
|
|
1595
1430
|
// Returns null if the type is unknown.
|
|
1596
|
-
|
|
1597
|
-
|
|
1598
1431
|
function getObjectType(obj) {
|
|
1599
1432
|
if (obj instanceof Array) {
|
|
1600
1433
|
return 'Array';
|
|
1601
1434
|
}
|
|
1602
|
-
|
|
1603
1435
|
if (obj.__type) {
|
|
1604
1436
|
switch (obj.__type) {
|
|
1605
1437
|
case 'Pointer':
|
|
@@ -1609,9 +1441,7 @@ function getObjectType(obj) {
|
|
|
1609
1441
|
targetClass: obj.className
|
|
1610
1442
|
};
|
|
1611
1443
|
}
|
|
1612
|
-
|
|
1613
1444
|
break;
|
|
1614
|
-
|
|
1615
1445
|
case 'Relation':
|
|
1616
1446
|
if (obj.className) {
|
|
1617
1447
|
return {
|
|
@@ -1619,80 +1449,60 @@ function getObjectType(obj) {
|
|
|
1619
1449
|
targetClass: obj.className
|
|
1620
1450
|
};
|
|
1621
1451
|
}
|
|
1622
|
-
|
|
1623
1452
|
break;
|
|
1624
|
-
|
|
1625
1453
|
case 'File':
|
|
1626
1454
|
if (obj.name) {
|
|
1627
1455
|
return 'File';
|
|
1628
1456
|
}
|
|
1629
|
-
|
|
1630
1457
|
break;
|
|
1631
|
-
|
|
1632
1458
|
case 'Date':
|
|
1633
1459
|
if (obj.iso) {
|
|
1634
1460
|
return 'Date';
|
|
1635
1461
|
}
|
|
1636
|
-
|
|
1637
1462
|
break;
|
|
1638
|
-
|
|
1639
1463
|
case 'GeoPoint':
|
|
1640
1464
|
if (obj.latitude != null && obj.longitude != null) {
|
|
1641
1465
|
return 'GeoPoint';
|
|
1642
1466
|
}
|
|
1643
|
-
|
|
1644
1467
|
break;
|
|
1645
|
-
|
|
1646
1468
|
case 'Bytes':
|
|
1647
1469
|
if (obj.base64) {
|
|
1648
1470
|
return 'Bytes';
|
|
1649
1471
|
}
|
|
1650
|
-
|
|
1651
1472
|
break;
|
|
1652
|
-
|
|
1653
1473
|
case 'Polygon':
|
|
1654
1474
|
if (obj.coordinates) {
|
|
1655
1475
|
return 'Polygon';
|
|
1656
1476
|
}
|
|
1657
|
-
|
|
1658
1477
|
break;
|
|
1659
1478
|
}
|
|
1660
|
-
|
|
1661
1479
|
throw new Parse.Error(Parse.Error.INCORRECT_TYPE, 'This is not a valid ' + obj.__type);
|
|
1662
1480
|
}
|
|
1663
|
-
|
|
1664
1481
|
if (obj['$ne']) {
|
|
1665
1482
|
return getObjectType(obj['$ne']);
|
|
1666
1483
|
}
|
|
1667
|
-
|
|
1668
1484
|
if (obj.__op) {
|
|
1669
1485
|
switch (obj.__op) {
|
|
1670
1486
|
case 'Increment':
|
|
1671
1487
|
return 'Number';
|
|
1672
|
-
|
|
1673
1488
|
case 'Delete':
|
|
1674
1489
|
return null;
|
|
1675
|
-
|
|
1676
1490
|
case 'Add':
|
|
1677
1491
|
case 'AddUnique':
|
|
1678
1492
|
case 'Remove':
|
|
1679
1493
|
return 'Array';
|
|
1680
|
-
|
|
1681
1494
|
case 'AddRelation':
|
|
1682
1495
|
case 'RemoveRelation':
|
|
1683
1496
|
return {
|
|
1684
1497
|
type: 'Relation',
|
|
1685
1498
|
targetClass: obj.objects[0].className
|
|
1686
1499
|
};
|
|
1687
|
-
|
|
1688
1500
|
case 'Batch':
|
|
1689
1501
|
return getObjectType(obj.ops[0]);
|
|
1690
|
-
|
|
1691
1502
|
default:
|
|
1692
1503
|
throw 'unexpected op: ' + obj.__op;
|
|
1693
1504
|
}
|
|
1694
1505
|
}
|
|
1695
|
-
|
|
1696
1506
|
return 'Object';
|
|
1697
1507
|
}
|
|
1698
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/Controllers/SchemaController.js"],"names":["Parse","require","defaultColumns","Object","freeze","_Default","objectId","type","createdAt","updatedAt","ACL","_User","username","password","email","emailVerified","authData","_Installation","installationId","deviceToken","channels","deviceType","pushType","GCMSenderId","timeZone","localeIdentifier","badge","appVersion","appName","appIdentifier","parseVersion","_Role","name","users","targetClass","roles","_Session","user","sessionToken","expiresAt","createdWith","_Product","productIdentifier","download","downloadName","icon","order","title","subtitle","_PushStatus","pushTime","source","query","payload","expiry","expiration_interval","status","numSent","numFailed","pushHash","errorMessage","sentPerType","failedPerType","sentPerUTCOffset","failedPerUTCOffset","count","_JobStatus","jobName","message","params","finishedAt","_JobSchedule","description","startAfter","daysOfWeek","timeOfDay","lastRun","repeatMinutes","_Hooks","functionName","className","triggerName","url","_GlobalConfig","masterKeyOnly","_GraphQLConfig","config","_Audience","lastUsed","timesUsed","_Idempotency","reqId","expire","requiredColumns","read","write","invalidColumns","systemClasses","volatileClasses","roleRegex","protectedFieldsPointerRegex","publicRegex","authenticatedRegex","requiresAuthenticationRegex","clpPointerRegex","protectedFieldsRegex","clpFieldsRegex","validatePermissionKey","key","userIdRegExp","matchesSome","regEx","match","valid","Error","INVALID_JSON","validateProtectedFieldsKey","CLPValidKeys","validateCLP","perms","fields","operationKey","indexOf","operation","validateCLPjson","fieldName","validatePointerPermission","entity","protectedFields","Array","isArray","field","prototype","hasOwnProperty","call","pointerFields","pointerField","permit","joinClassRegex","classAndFieldRegex","classNameIsValid","test","fieldNameIsValid","includes","fieldNameIsValidForClass","invalidClassNameMessage","invalidJsonError","validNonRelationOrPointerTypes","fieldTypeIsInvalid","INVALID_CLASS_NAME","undefined","INCORRECT_TYPE","convertSchemaToAdapterSchema","schema","injectDefaultSchema","_rperm","_wperm","_hashed_password","convertAdapterSchemaToParseSchema","indexes","keys","length","SchemaData","constructor","allSchemas","__data","__protectedFields","forEach","defineProperty","get","data","classLevelPermissions","classProtectedFields","unq","Set","from","defaultSchema","_HooksSchema","_GlobalConfigSchema","_GraphQLConfigSchema","_PushStatusSchema","_JobStatusSchema","_JobScheduleSchema","_AudienceSchema","_IdempotencySchema","VolatileClassesSchemas","dbTypeMatchesObjectType","dbType","objectType","typeToString","SchemaController","databaseAdapter","_dbAdapter","schemaData","SchemaCache","all","Config","applicationId","customIds","allowCustomObjectId","customIdRegEx","autoIdRegEx","userIdRegEx","watch","reloadData","clearCache","options","reloadDataPromise","getAllClasses","then","err","setAllClasses","cached","Promise","resolve","map","put","getOneSchema","allowVolatileClasses","clear","oneSchema","find","reject","addClassIfNotExists","validationError","validateNewClass","code","error","adapterSchema","createClass","parseSchema","DUPLICATE_VALUE","updateClass","submittedFields","database","existingFields","__op","newSchema","buildMergedSchemaObject","defaultFields","fullNewSchema","assign","validateSchemaData","deletedFields","insertedFields","push","deletePromise","deleteFields","enforceFields","promises","enforceFieldExists","results","filter","result","setPermissions","setIndexesWithSchemaFormat","ensureFields","reloadedSchema","catch","enforceClassExists","existingFieldNames","INVALID_KEY_NAME","fieldType","defaultValue","defaultValueType","getType","required","geoPoints","setClassLevelPermissions","isValidation","split","expectedType","getExpectedType","JSON","stringify","updateFieldOptions","addFieldIfNotExists","i","deleteField","fieldNames","schemaFields","adapter","deleteClass","validateObject","object","geocount","expected","promise","thenValidateRequiredColumns","validateRequiredColumns","columns","missingColumns","column","testPermissionsForClassName","aclGroup","testPermissions","getClassLevelPermissions","classPermissions","some","acl","validatePermission","action","OBJECT_NOT_FOUND","permissionField","OPERATION_FORBIDDEN","hasClass","load","dbAdapter","putRequest","sysSchemaField","_id","oldField","fieldIsDeleted","newField","schemaPromise","obj","getObjectType","__type","iso","latitude","longitude","base64","coordinates","objects","ops"],"mappings":"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","sourcesContent":["// @flow\n// This class handles schema validation, persistence, and modification.\n//\n// Each individual Schema object should be immutable. The helpers to\n// do things with the Schema just return a new schema when the schema\n// is changed.\n//\n// The canonical place to store this Schema is in the database itself,\n// in a _SCHEMA collection. This is not the right way to do it for an\n// open source framework, but it's backward compatible, so we're\n// keeping it this way for now.\n//\n// In API-handling code, you should only use the Schema class via the\n// DatabaseController. This will let us replace the schema logic for\n// different databases.\n// TODO: hide all schema logic inside the database adapter.\n// @flow-disable-next\nconst Parse = require('parse/node').Parse;\nimport { StorageAdapter } from '../Adapters/Storage/StorageAdapter';\nimport SchemaCache from '../Adapters/Cache/SchemaCache';\nimport DatabaseController from './DatabaseController';\nimport Config from '../Config';\n// @flow-disable-next\nimport deepcopy from 'deepcopy';\nimport type {\n  Schema,\n  SchemaFields,\n  ClassLevelPermissions,\n  SchemaField,\n  LoadSchemaOptions,\n} from './types';\n\nconst defaultColumns: { [string]: SchemaFields } = Object.freeze({\n  // Contain the default columns for every parse object type (except _Join collection)\n  _Default: {\n    objectId: { type: 'String' },\n    createdAt: { type: 'Date' },\n    updatedAt: { type: 'Date' },\n    ACL: { type: 'ACL' },\n  },\n  // The additional default columns for the _User collection (in addition to DefaultCols)\n  _User: {\n    username: { type: 'String' },\n    password: { type: 'String' },\n    email: { type: 'String' },\n    emailVerified: { type: 'Boolean' },\n    authData: { type: 'Object' },\n  },\n  // The additional default columns for the _Installation collection (in addition to DefaultCols)\n  _Installation: {\n    installationId: { type: 'String' },\n    deviceToken: { type: 'String' },\n    channels: { type: 'Array' },\n    deviceType: { type: 'String' },\n    pushType: { type: 'String' },\n    GCMSenderId: { type: 'String' },\n    timeZone: { type: 'String' },\n    localeIdentifier: { type: 'String' },\n    badge: { type: 'Number' },\n    appVersion: { type: 'String' },\n    appName: { type: 'String' },\n    appIdentifier: { type: 'String' },\n    parseVersion: { type: 'String' },\n  },\n  // The additional default columns for the _Role collection (in addition to DefaultCols)\n  _Role: {\n    name: { type: 'String' },\n    users: { type: 'Relation', targetClass: '_User' },\n    roles: { type: 'Relation', targetClass: '_Role' },\n  },\n  // The additional default columns for the _Session collection (in addition to DefaultCols)\n  _Session: {\n    user: { type: 'Pointer', targetClass: '_User' },\n    installationId: { type: 'String' },\n    sessionToken: { type: 'String' },\n    expiresAt: { type: 'Date' },\n    createdWith: { type: 'Object' },\n  },\n  _Product: {\n    productIdentifier: { type: 'String' },\n    download: { type: 'File' },\n    downloadName: { type: 'String' },\n    icon: { type: 'File' },\n    order: { type: 'Number' },\n    title: { type: 'String' },\n    subtitle: { type: 'String' },\n  },\n  _PushStatus: {\n    pushTime: { type: 'String' },\n    source: { type: 'String' }, // rest or webui\n    query: { type: 'String' }, // the stringified JSON query\n    payload: { type: 'String' }, // the stringified JSON payload,\n    title: { type: 'String' },\n    expiry: { type: 'Number' },\n    expiration_interval: { type: 'Number' },\n    status: { type: 'String' },\n    numSent: { type: 'Number' },\n    numFailed: { type: 'Number' },\n    pushHash: { type: 'String' },\n    errorMessage: { type: 'Object' },\n    sentPerType: { type: 'Object' },\n    failedPerType: { type: 'Object' },\n    sentPerUTCOffset: { type: 'Object' },\n    failedPerUTCOffset: { type: 'Object' },\n    count: { type: 'Number' }, // tracks # of batches queued and pending\n  },\n  _JobStatus: {\n    jobName: { type: 'String' },\n    source: { type: 'String' },\n    status: { type: 'String' },\n    message: { type: 'String' },\n    params: { type: 'Object' }, // params received when calling the job\n    finishedAt: { type: 'Date' },\n  },\n  _JobSchedule: {\n    jobName: { type: 'String' },\n    description: { type: 'String' },\n    params: { type: 'String' },\n    startAfter: { type: 'String' },\n    daysOfWeek: { type: 'Array' },\n    timeOfDay: { type: 'String' },\n    lastRun: { type: 'Number' },\n    repeatMinutes: { type: 'Number' },\n  },\n  _Hooks: {\n    functionName: { type: 'String' },\n    className: { type: 'String' },\n    triggerName: { type: 'String' },\n    url: { type: 'String' },\n  },\n  _GlobalConfig: {\n    objectId: { type: 'String' },\n    params: { type: 'Object' },\n    masterKeyOnly: { type: 'Object' },\n  },\n  _GraphQLConfig: {\n    objectId: { type: 'String' },\n    config: { type: 'Object' },\n  },\n  _Audience: {\n    objectId: { type: 'String' },\n    name: { type: 'String' },\n    query: { type: 'String' }, //storing query as JSON string to prevent \"Nested keys should not contain the '$' or '.' characters\" error\n    lastUsed: { type: 'Date' },\n    timesUsed: { type: 'Number' },\n  },\n  _Idempotency: {\n    reqId: { type: 'String' },\n    expire: { type: 'Date' },\n  },\n});\n\n// fields required for read or write operations on their respective classes.\nconst requiredColumns = Object.freeze({\n  read: {\n    _User: ['username'],\n  },\n  write: {\n    _Product: ['productIdentifier', 'icon', 'order', 'title', 'subtitle'],\n    _Role: ['name', 'ACL'],\n  },\n});\n\nconst invalidColumns = ['length'];\n\nconst systemClasses = Object.freeze([\n  '_User',\n  '_Installation',\n  '_Role',\n  '_Session',\n  '_Product',\n  '_PushStatus',\n  '_JobStatus',\n  '_JobSchedule',\n  '_Audience',\n  '_Idempotency',\n]);\n\nconst volatileClasses = Object.freeze([\n  '_JobStatus',\n  '_PushStatus',\n  '_Hooks',\n  '_GlobalConfig',\n  '_GraphQLConfig',\n  '_JobSchedule',\n  '_Audience',\n  '_Idempotency',\n]);\n\n// Anything that start with role\nconst roleRegex = /^role:.*/;\n// Anything that starts with userField (allowed for protected fields only)\nconst protectedFieldsPointerRegex = /^userField:.*/;\n// * permission\nconst publicRegex = /^\\*$/;\n\nconst authenticatedRegex = /^authenticated$/;\n\nconst requiresAuthenticationRegex = /^requiresAuthentication$/;\n\nconst clpPointerRegex = /^pointerFields$/;\n\n// regex for validating entities in protectedFields object\nconst protectedFieldsRegex = Object.freeze([\n  protectedFieldsPointerRegex,\n  publicRegex,\n  authenticatedRegex,\n  roleRegex,\n]);\n\n// clp regex\nconst clpFieldsRegex = Object.freeze([\n  clpPointerRegex,\n  publicRegex,\n  requiresAuthenticationRegex,\n  roleRegex,\n]);\n\nfunction validatePermissionKey(key, userIdRegExp) {\n  let matchesSome = false;\n  for (const regEx of clpFieldsRegex) {\n    if (key.match(regEx) !== null) {\n      matchesSome = true;\n      break;\n    }\n  }\n\n  // userId depends on startup options so it's dynamic\n  const valid = matchesSome || key.match(userIdRegExp) !== null;\n  if (!valid) {\n    throw new Parse.Error(\n      Parse.Error.INVALID_JSON,\n      `'${key}' is not a valid key for class level permissions`\n    );\n  }\n}\n\nfunction validateProtectedFieldsKey(key, userIdRegExp) {\n  let matchesSome = false;\n  for (const regEx of protectedFieldsRegex) {\n    if (key.match(regEx) !== null) {\n      matchesSome = true;\n      break;\n    }\n  }\n\n  // userId regex depends on launch options so it's dynamic\n  const valid = matchesSome || key.match(userIdRegExp) !== null;\n  if (!valid) {\n    throw new Parse.Error(\n      Parse.Error.INVALID_JSON,\n      `'${key}' is not a valid key for class level permissions`\n    );\n  }\n}\n\nconst CLPValidKeys = Object.freeze([\n  'find',\n  'count',\n  'get',\n  'create',\n  'update',\n  'delete',\n  'addField',\n  'readUserFields',\n  'writeUserFields',\n  'protectedFields',\n]);\n\n// validation before setting class-level permissions on collection\nfunction validateCLP(perms: ClassLevelPermissions, fields: SchemaFields, userIdRegExp: RegExp) {\n  if (!perms) {\n    return;\n  }\n  for (const operationKey in perms) {\n    if (CLPValidKeys.indexOf(operationKey) == -1) {\n      throw new Parse.Error(\n        Parse.Error.INVALID_JSON,\n        `${operationKey} is not a valid operation for class level permissions`\n      );\n    }\n\n    const operation = perms[operationKey];\n    // proceed with next operationKey\n\n    // throws when root fields are of wrong type\n    validateCLPjson(operation, operationKey);\n\n    if (operationKey === 'readUserFields' || operationKey === 'writeUserFields') {\n      // validate grouped pointer permissions\n      // must be an array with field names\n      for (const fieldName of operation) {\n        validatePointerPermission(fieldName, fields, operationKey);\n      }\n      // readUserFields and writerUserFields do not have nesdted fields\n      // proceed with next operationKey\n      continue;\n    }\n\n    // validate protected fields\n    if (operationKey === 'protectedFields') {\n      for (const entity in operation) {\n        // throws on unexpected key\n        validateProtectedFieldsKey(entity, userIdRegExp);\n\n        const protectedFields = operation[entity];\n\n        if (!Array.isArray(protectedFields)) {\n          throw new Parse.Error(\n            Parse.Error.INVALID_JSON,\n            `'${protectedFields}' is not a valid value for protectedFields[${entity}] - expected an array.`\n          );\n        }\n\n        // if the field is in form of array\n        for (const field of protectedFields) {\n          // do not alloow to protect default fields\n          if (defaultColumns._Default[field]) {\n            throw new Parse.Error(\n              Parse.Error.INVALID_JSON,\n              `Default field '${field}' can not be protected`\n            );\n          }\n          // field should exist on collection\n          if (!Object.prototype.hasOwnProperty.call(fields, field)) {\n            throw new Parse.Error(\n              Parse.Error.INVALID_JSON,\n              `Field '${field}' in protectedFields:${entity} does not exist`\n            );\n          }\n        }\n      }\n      // proceed with next operationKey\n      continue;\n    }\n\n    // validate other fields\n    // Entity can be:\n    // \"*\" - Public,\n    // \"requiresAuthentication\" - authenticated users,\n    // \"objectId\" - _User id,\n    // \"role:rolename\",\n    // \"pointerFields\" - array of field names containing pointers to users\n    for (const entity in operation) {\n      // throws on unexpected key\n      validatePermissionKey(entity, userIdRegExp);\n\n      // entity can be either:\n      // \"pointerFields\": string[]\n      if (entity === 'pointerFields') {\n        const pointerFields = operation[entity];\n\n        if (Array.isArray(pointerFields)) {\n          for (const pointerField of pointerFields) {\n            validatePointerPermission(pointerField, fields, operation);\n          }\n        } else {\n          throw new Parse.Error(\n            Parse.Error.INVALID_JSON,\n            `'${pointerFields}' is not a valid value for ${operationKey}[${entity}] - expected an array.`\n          );\n        }\n        // proceed with next entity key\n        continue;\n      }\n\n      // or [entity]: boolean\n      const permit = operation[entity];\n\n      if (permit !== true) {\n        throw new Parse.Error(\n          Parse.Error.INVALID_JSON,\n          `'${permit}' is not a valid value for class level permissions ${operationKey}:${entity}:${permit}`\n        );\n      }\n    }\n  }\n}\n\nfunction validateCLPjson(operation: any, operationKey: string) {\n  if (operationKey === 'readUserFields' || operationKey === 'writeUserFields') {\n    if (!Array.isArray(operation)) {\n      throw new Parse.Error(\n        Parse.Error.INVALID_JSON,\n        `'${operation}' is not a valid value for class level permissions ${operationKey} - must be an array`\n      );\n    }\n  } else {\n    if (typeof operation === 'object' && operation !== null) {\n      // ok to proceed\n      return;\n    } else {\n      throw new Parse.Error(\n        Parse.Error.INVALID_JSON,\n        `'${operation}' is not a valid value for class level permissions ${operationKey} - must be an object`\n      );\n    }\n  }\n}\n\nfunction validatePointerPermission(fieldName: string, fields: Object, operation: string) {\n  // Uses collection schema to ensure the field is of type:\n  // - Pointer<_User> (pointers)\n  // - Array\n  //\n  //    It's not possible to enforce type on Array's items in schema\n  //  so we accept any Array field, and later when applying permissions\n  //  only items that are pointers to _User are considered.\n  if (\n    !(\n      fields[fieldName] &&\n      ((fields[fieldName].type == 'Pointer' && fields[fieldName].targetClass == '_User') ||\n        fields[fieldName].type == 'Array')\n    )\n  ) {\n    throw new Parse.Error(\n      Parse.Error.INVALID_JSON,\n      `'${fieldName}' is not a valid column for class level pointer permissions ${operation}`\n    );\n  }\n}\n\nconst joinClassRegex = /^_Join:[A-Za-z0-9_]+:[A-Za-z0-9_]+/;\nconst classAndFieldRegex = /^[A-Za-z][A-Za-z0-9_]*$/;\nfunction classNameIsValid(className: string): boolean {\n  // Valid classes must:\n  return (\n    // Be one of _User, _Installation, _Role, _Session OR\n    systemClasses.indexOf(className) > -1 ||\n    // Be a join table OR\n    joinClassRegex.test(className) ||\n    // Include only alpha-numeric and underscores, and not start with an underscore or number\n    fieldNameIsValid(className, className)\n  );\n}\n\n// Valid fields must be alpha-numeric, and not start with an underscore or number\n// must not be a reserved key\nfunction fieldNameIsValid(fieldName: string, className: string): boolean {\n  if (className && className !== '_Hooks') {\n    if (fieldName === 'className') {\n      return false;\n    }\n  }\n  return classAndFieldRegex.test(fieldName) && !invalidColumns.includes(fieldName);\n}\n\n// Checks that it's not trying to clobber one of the default fields of the class.\nfunction fieldNameIsValidForClass(fieldName: string, className: string): boolean {\n  if (!fieldNameIsValid(fieldName, className)) {\n    return false;\n  }\n  if (defaultColumns._Default[fieldName]) {\n    return false;\n  }\n  if (defaultColumns[className] && defaultColumns[className][fieldName]) {\n    return false;\n  }\n  return true;\n}\n\nfunction invalidClassNameMessage(className: string): string {\n  return (\n    'Invalid classname: ' +\n    className +\n    ', classnames can only have alphanumeric characters and _, and must start with an alpha character '\n  );\n}\n\nconst invalidJsonError = new Parse.Error(Parse.Error.INVALID_JSON, 'invalid JSON');\nconst validNonRelationOrPointerTypes = [\n  'Number',\n  'String',\n  'Boolean',\n  'Date',\n  'Object',\n  'Array',\n  'GeoPoint',\n  'File',\n  'Bytes',\n  'Polygon',\n];\n// Returns an error suitable for throwing if the type is invalid\nconst fieldTypeIsInvalid = ({ type, targetClass }) => {\n  if (['Pointer', 'Relation'].indexOf(type) >= 0) {\n    if (!targetClass) {\n      return new Parse.Error(135, `type ${type} needs a class name`);\n    } else if (typeof targetClass !== 'string') {\n      return invalidJsonError;\n    } else if (!classNameIsValid(targetClass)) {\n      return new Parse.Error(Parse.Error.INVALID_CLASS_NAME, invalidClassNameMessage(targetClass));\n    } else {\n      return undefined;\n    }\n  }\n  if (typeof type !== 'string') {\n    return invalidJsonError;\n  }\n  if (validNonRelationOrPointerTypes.indexOf(type) < 0) {\n    return new Parse.Error(Parse.Error.INCORRECT_TYPE, `invalid field type: ${type}`);\n  }\n  return undefined;\n};\n\nconst convertSchemaToAdapterSchema = (schema: any) => {\n  schema = injectDefaultSchema(schema);\n  delete schema.fields.ACL;\n  schema.fields._rperm = { type: 'Array' };\n  schema.fields._wperm = { type: 'Array' };\n\n  if (schema.className === '_User') {\n    delete schema.fields.password;\n    schema.fields._hashed_password = { type: 'String' };\n  }\n\n  return schema;\n};\n\nconst convertAdapterSchemaToParseSchema = ({ ...schema }) => {\n  delete schema.fields._rperm;\n  delete schema.fields._wperm;\n\n  schema.fields.ACL = { type: 'ACL' };\n\n  if (schema.className === '_User') {\n    delete schema.fields.authData; //Auth data is implicit\n    delete schema.fields._hashed_password;\n    schema.fields.password = { type: 'String' };\n  }\n\n  if (schema.indexes && Object.keys(schema.indexes).length === 0) {\n    delete schema.indexes;\n  }\n\n  return schema;\n};\n\nclass SchemaData {\n  __data: any;\n  __protectedFields: any;\n  constructor(allSchemas = [], protectedFields = {}) {\n    this.__data = {};\n    this.__protectedFields = protectedFields;\n    allSchemas.forEach(schema => {\n      if (volatileClasses.includes(schema.className)) {\n        return;\n      }\n      Object.defineProperty(this, schema.className, {\n        get: () => {\n          if (!this.__data[schema.className]) {\n            const data = {};\n            data.fields = injectDefaultSchema(schema).fields;\n            data.classLevelPermissions = deepcopy(schema.classLevelPermissions);\n            data.indexes = schema.indexes;\n\n            const classProtectedFields = this.__protectedFields[schema.className];\n            if (classProtectedFields) {\n              for (const key in classProtectedFields) {\n                const unq = new Set([\n                  ...(data.classLevelPermissions.protectedFields[key] || []),\n                  ...classProtectedFields[key],\n                ]);\n                data.classLevelPermissions.protectedFields[key] = Array.from(unq);\n              }\n            }\n\n            this.__data[schema.className] = data;\n          }\n          return this.__data[schema.className];\n        },\n      });\n    });\n\n    // Inject the in-memory classes\n    volatileClasses.forEach(className => {\n      Object.defineProperty(this, className, {\n        get: () => {\n          if (!this.__data[className]) {\n            const schema = injectDefaultSchema({\n              className,\n              fields: {},\n              classLevelPermissions: {},\n            });\n            const data = {};\n            data.fields = schema.fields;\n            data.classLevelPermissions = schema.classLevelPermissions;\n            data.indexes = schema.indexes;\n            this.__data[className] = data;\n          }\n          return this.__data[className];\n        },\n      });\n    });\n  }\n}\n\nconst injectDefaultSchema = ({ className, fields, classLevelPermissions, indexes }: Schema) => {\n  const defaultSchema: Schema = {\n    className,\n    fields: {\n      ...defaultColumns._Default,\n      ...(defaultColumns[className] || {}),\n      ...fields,\n    },\n    classLevelPermissions,\n  };\n  if (indexes && Object.keys(indexes).length !== 0) {\n    defaultSchema.indexes = indexes;\n  }\n  return defaultSchema;\n};\n\nconst _HooksSchema = { className: '_Hooks', fields: defaultColumns._Hooks };\nconst _GlobalConfigSchema = {\n  className: '_GlobalConfig',\n  fields: defaultColumns._GlobalConfig,\n};\nconst _GraphQLConfigSchema = {\n  className: '_GraphQLConfig',\n  fields: defaultColumns._GraphQLConfig,\n};\nconst _PushStatusSchema = convertSchemaToAdapterSchema(\n  injectDefaultSchema({\n    className: '_PushStatus',\n    fields: {},\n    classLevelPermissions: {},\n  })\n);\nconst _JobStatusSchema = convertSchemaToAdapterSchema(\n  injectDefaultSchema({\n    className: '_JobStatus',\n    fields: {},\n    classLevelPermissions: {},\n  })\n);\nconst _JobScheduleSchema = convertSchemaToAdapterSchema(\n  injectDefaultSchema({\n    className: '_JobSchedule',\n    fields: {},\n    classLevelPermissions: {},\n  })\n);\nconst _AudienceSchema = convertSchemaToAdapterSchema(\n  injectDefaultSchema({\n    className: '_Audience',\n    fields: defaultColumns._Audience,\n    classLevelPermissions: {},\n  })\n);\nconst _IdempotencySchema = convertSchemaToAdapterSchema(\n  injectDefaultSchema({\n    className: '_Idempotency',\n    fields: defaultColumns._Idempotency,\n    classLevelPermissions: {},\n  })\n);\nconst VolatileClassesSchemas = [\n  _HooksSchema,\n  _JobStatusSchema,\n  _JobScheduleSchema,\n  _PushStatusSchema,\n  _GlobalConfigSchema,\n  _GraphQLConfigSchema,\n  _AudienceSchema,\n  _IdempotencySchema,\n];\n\nconst dbTypeMatchesObjectType = (dbType: SchemaField | string, objectType: SchemaField) => {\n  if (dbType.type !== objectType.type) return false;\n  if (dbType.targetClass !== objectType.targetClass) return false;\n  if (dbType === objectType.type) return true;\n  if (dbType.type === objectType.type) return true;\n  return false;\n};\n\nconst typeToString = (type: SchemaField | string): string => {\n  if (typeof type === 'string') {\n    return type;\n  }\n  if (type.targetClass) {\n    return `${type.type}<${type.targetClass}>`;\n  }\n  return `${type.type}`;\n};\n\n// Stores the entire schema of the app in a weird hybrid format somewhere between\n// the mongo format and the Parse format. Soon, this will all be Parse format.\nexport default class SchemaController {\n  _dbAdapter: StorageAdapter;\n  schemaData: { [string]: Schema };\n  reloadDataPromise: ?Promise<any>;\n  protectedFields: any;\n  userIdRegEx: RegExp;\n\n  constructor(databaseAdapter: StorageAdapter) {\n    this._dbAdapter = databaseAdapter;\n    this.schemaData = new SchemaData(SchemaCache.all(), this.protectedFields);\n    this.protectedFields = Config.get(Parse.applicationId).protectedFields;\n\n    const customIds = Config.get(Parse.applicationId).allowCustomObjectId;\n\n    const customIdRegEx = /^.{1,}$/u; // 1+ chars\n    const autoIdRegEx = /^[a-zA-Z0-9]{1,}$/;\n\n    this.userIdRegEx = customIds ? customIdRegEx : autoIdRegEx;\n\n    this._dbAdapter.watch(() => {\n      this.reloadData({ clearCache: true });\n    });\n  }\n\n  reloadData(options: LoadSchemaOptions = { clearCache: false }): Promise<any> {\n    if (this.reloadDataPromise && !options.clearCache) {\n      return this.reloadDataPromise;\n    }\n    this.reloadDataPromise = this.getAllClasses(options)\n      .then(\n        allSchemas => {\n          this.schemaData = new SchemaData(allSchemas, this.protectedFields);\n          delete this.reloadDataPromise;\n        },\n        err => {\n          this.schemaData = new SchemaData();\n          delete this.reloadDataPromise;\n          throw err;\n        }\n      )\n      .then(() => {});\n    return this.reloadDataPromise;\n  }\n\n  getAllClasses(options: LoadSchemaOptions = { clearCache: false }): Promise<Array<Schema>> {\n    if (options.clearCache) {\n      return this.setAllClasses();\n    }\n    const cached = SchemaCache.all();\n    if (cached && cached.length) {\n      return Promise.resolve(cached);\n    }\n    return this.setAllClasses();\n  }\n\n  setAllClasses(): Promise<Array<Schema>> {\n    return this._dbAdapter\n      .getAllClasses()\n      .then(allSchemas => allSchemas.map(injectDefaultSchema))\n      .then(allSchemas => {\n        SchemaCache.put(allSchemas);\n        return allSchemas;\n      });\n  }\n\n  getOneSchema(\n    className: string,\n    allowVolatileClasses: boolean = false,\n    options: LoadSchemaOptions = { clearCache: false }\n  ): Promise<Schema> {\n    if (options.clearCache) {\n      SchemaCache.clear();\n    }\n    if (allowVolatileClasses && volatileClasses.indexOf(className) > -1) {\n      const data = this.schemaData[className];\n      return Promise.resolve({\n        className,\n        fields: data.fields,\n        classLevelPermissions: data.classLevelPermissions,\n        indexes: data.indexes,\n      });\n    }\n    const cached = SchemaCache.get(className);\n    if (cached && !options.clearCache) {\n      return Promise.resolve(cached);\n    }\n    return this.setAllClasses().then(allSchemas => {\n      const oneSchema = allSchemas.find(schema => schema.className === className);\n      if (!oneSchema) {\n        return Promise.reject(undefined);\n      }\n      return oneSchema;\n    });\n  }\n\n  // Create a new class that includes the three default fields.\n  // ACL is an implicit column that does not get an entry in the\n  // _SCHEMAS database. Returns a promise that resolves with the\n  // created schema, in mongo format.\n  // on success, and rejects with an error on fail. Ensure you\n  // have authorization (master key, or client class creation\n  // enabled) before calling this function.\n  async addClassIfNotExists(\n    className: string,\n    fields: SchemaFields = {},\n    classLevelPermissions: any,\n    indexes: any = {}\n  ): Promise<void | Schema> {\n    var validationError = this.validateNewClass(className, fields, classLevelPermissions);\n    if (validationError) {\n      if (validationError instanceof Parse.Error) {\n        return Promise.reject(validationError);\n      } else if (validationError.code && validationError.error) {\n        return Promise.reject(new Parse.Error(validationError.code, validationError.error));\n      }\n      return Promise.reject(validationError);\n    }\n    try {\n      const adapterSchema = await this._dbAdapter.createClass(\n        className,\n        convertSchemaToAdapterSchema({\n          fields,\n          classLevelPermissions,\n          indexes,\n          className,\n        })\n      );\n      // TODO: Remove by updating schema cache directly\n      await this.reloadData({ clearCache: true });\n      const parseSchema = convertAdapterSchemaToParseSchema(adapterSchema);\n      return parseSchema;\n    } catch (error) {\n      if (error && error.code === Parse.Error.DUPLICATE_VALUE) {\n        throw new Parse.Error(Parse.Error.INVALID_CLASS_NAME, `Class ${className} already exists.`);\n      } else {\n        throw error;\n      }\n    }\n  }\n\n  updateClass(\n    className: string,\n    submittedFields: SchemaFields,\n    classLevelPermissions: any,\n    indexes: any,\n    database: DatabaseController\n  ) {\n    return this.getOneSchema(className)\n      .then(schema => {\n        const existingFields = schema.fields;\n        Object.keys(submittedFields).forEach(name => {\n          const field = submittedFields[name];\n          if (\n            existingFields[name] &&\n            existingFields[name].type !== field.type &&\n            field.__op !== 'Delete'\n          ) {\n            throw new Parse.Error(255, `Field ${name} exists, cannot update.`);\n          }\n          if (!existingFields[name] && field.__op === 'Delete') {\n            throw new Parse.Error(255, `Field ${name} does not exist, cannot delete.`);\n          }\n        });\n\n        delete existingFields._rperm;\n        delete existingFields._wperm;\n        const newSchema = buildMergedSchemaObject(existingFields, submittedFields);\n        const defaultFields = defaultColumns[className] || defaultColumns._Default;\n        const fullNewSchema = Object.assign({}, newSchema, defaultFields);\n        const validationError = this.validateSchemaData(\n          className,\n          newSchema,\n          classLevelPermissions,\n          Object.keys(existingFields)\n        );\n        if (validationError) {\n          throw new Parse.Error(validationError.code, validationError.error);\n        }\n\n        // Finally we have checked to make sure the request is valid and we can start deleting fields.\n        // Do all deletions first, then a single save to _SCHEMA collection to handle all additions.\n        const deletedFields: string[] = [];\n        const insertedFields = [];\n        Object.keys(submittedFields).forEach(fieldName => {\n          if (submittedFields[fieldName].__op === 'Delete') {\n            deletedFields.push(fieldName);\n          } else {\n            insertedFields.push(fieldName);\n          }\n        });\n\n        let deletePromise = Promise.resolve();\n        if (deletedFields.length > 0) {\n          deletePromise = this.deleteFields(deletedFields, className, database);\n        }\n        let enforceFields = [];\n        return (\n          deletePromise // Delete Everything\n            .then(() => this.reloadData({ clearCache: true })) // Reload our Schema, so we have all the new values\n            .then(() => {\n              const promises = insertedFields.map(fieldName => {\n                const type = submittedFields[fieldName];\n                return this.enforceFieldExists(className, fieldName, type);\n              });\n              return Promise.all(promises);\n            })\n            .then(results => {\n              enforceFields = results.filter(result => !!result);\n              return this.setPermissions(className, classLevelPermissions, newSchema);\n            })\n            .then(() =>\n              this._dbAdapter.setIndexesWithSchemaFormat(\n                className,\n                indexes,\n                schema.indexes,\n                fullNewSchema\n              )\n            )\n            .then(() => this.reloadData({ clearCache: true }))\n            //TODO: Move this logic into the database adapter\n            .then(() => {\n              this.ensureFields(enforceFields);\n              const schema = this.schemaData[className];\n              const reloadedSchema: Schema = {\n                className: className,\n                fields: schema.fields,\n                classLevelPermissions: schema.classLevelPermissions,\n              };\n              if (schema.indexes && Object.keys(schema.indexes).length !== 0) {\n                reloadedSchema.indexes = schema.indexes;\n              }\n              return reloadedSchema;\n            })\n        );\n      })\n      .catch(error => {\n        if (error === undefined) {\n          throw new Parse.Error(\n            Parse.Error.INVALID_CLASS_NAME,\n            `Class ${className} does not exist.`\n          );\n        } else {\n          throw error;\n        }\n      });\n  }\n\n  // Returns a promise that resolves successfully to the new schema\n  // object or fails with a reason.\n  enforceClassExists(className: string): Promise<SchemaController> {\n    if (this.schemaData[className]) {\n      return Promise.resolve(this);\n    }\n    // We don't have this class. Update the schema\n    return (\n      // The schema update succeeded. Reload the schema\n      this.addClassIfNotExists(className)\n        .catch(() => {\n          // The schema update failed. This can be okay - it might\n          // have failed because there's a race condition and a different\n          // client is making the exact same schema update that we want.\n          // So just reload the schema.\n          return this.reloadData({ clearCache: true });\n        })\n        .then(() => {\n          // Ensure that the schema now validates\n          if (this.schemaData[className]) {\n            return this;\n          } else {\n            throw new Parse.Error(Parse.Error.INVALID_JSON, `Failed to add ${className}`);\n          }\n        })\n        .catch(() => {\n          // The schema still doesn't validate. Give up\n          throw new Parse.Error(Parse.Error.INVALID_JSON, 'schema class name does not revalidate');\n        })\n    );\n  }\n\n  validateNewClass(className: string, fields: SchemaFields = {}, classLevelPermissions: any): any {\n    if (this.schemaData[className]) {\n      throw new Parse.Error(Parse.Error.INVALID_CLASS_NAME, `Class ${className} already exists.`);\n    }\n    if (!classNameIsValid(className)) {\n      return {\n        code: Parse.Error.INVALID_CLASS_NAME,\n        error: invalidClassNameMessage(className),\n      };\n    }\n    return this.validateSchemaData(className, fields, classLevelPermissions, []);\n  }\n\n  validateSchemaData(\n    className: string,\n    fields: SchemaFields,\n    classLevelPermissions: ClassLevelPermissions,\n    existingFieldNames: Array<string>\n  ) {\n    for (const fieldName in fields) {\n      if (existingFieldNames.indexOf(fieldName) < 0) {\n        if (!fieldNameIsValid(fieldName, className)) {\n          return {\n            code: Parse.Error.INVALID_KEY_NAME,\n            error: 'invalid field name: ' + fieldName,\n          };\n        }\n        if (!fieldNameIsValidForClass(fieldName, className)) {\n          return {\n            code: 136,\n            error: 'field ' + fieldName + ' cannot be added',\n          };\n        }\n        const fieldType = fields[fieldName];\n        const error = fieldTypeIsInvalid(fieldType);\n        if (error) return { code: error.code, error: error.message };\n        if (fieldType.defaultValue !== undefined) {\n          let defaultValueType = getType(fieldType.defaultValue);\n          if (typeof defaultValueType === 'string') {\n            defaultValueType = { type: defaultValueType };\n          } else if (typeof defaultValueType === 'object' && fieldType.type === 'Relation') {\n            return {\n              code: Parse.Error.INCORRECT_TYPE,\n              error: `The 'default value' option is not applicable for ${typeToString(fieldType)}`,\n            };\n          }\n          if (!dbTypeMatchesObjectType(fieldType, defaultValueType)) {\n            return {\n              code: Parse.Error.INCORRECT_TYPE,\n              error: `schema mismatch for ${className}.${fieldName} default value; expected ${typeToString(\n                fieldType\n              )} but got ${typeToString(defaultValueType)}`,\n            };\n          }\n        } else if (fieldType.required) {\n          if (typeof fieldType === 'object' && fieldType.type === 'Relation') {\n            return {\n              code: Parse.Error.INCORRECT_TYPE,\n              error: `The 'required' option is not applicable for ${typeToString(fieldType)}`,\n            };\n          }\n        }\n      }\n    }\n\n    for (const fieldName in defaultColumns[className]) {\n      fields[fieldName] = defaultColumns[className][fieldName];\n    }\n\n    const geoPoints = Object.keys(fields).filter(\n      key => fields[key] && fields[key].type === 'GeoPoint'\n    );\n    if (geoPoints.length > 1) {\n      return {\n        code: Parse.Error.INCORRECT_TYPE,\n        error:\n          'currently, only one GeoPoint field may exist in an object. Adding ' +\n          geoPoints[1] +\n          ' when ' +\n          geoPoints[0] +\n          ' already exists.',\n      };\n    }\n    validateCLP(classLevelPermissions, fields, this.userIdRegEx);\n  }\n\n  // Sets the Class-level permissions for a given className, which must exist.\n  async setPermissions(className: string, perms: any, newSchema: SchemaFields) {\n    if (typeof perms === 'undefined') {\n      return Promise.resolve();\n    }\n    validateCLP(perms, newSchema, this.userIdRegEx);\n    await this._dbAdapter.setClassLevelPermissions(className, perms);\n    const cached = SchemaCache.get(className);\n    if (cached) {\n      cached.classLevelPermissions = perms;\n    }\n  }\n\n  // Returns a promise that resolves successfully to the new schema\n  // object if the provided className-fieldName-type tuple is valid.\n  // The className must already be validated.\n  // If 'freeze' is true, refuse to update the schema for this field.\n  enforceFieldExists(\n    className: string,\n    fieldName: string,\n    type: string | SchemaField,\n    isValidation?: boolean\n  ) {\n    if (fieldName.indexOf('.') > 0) {\n      // subdocument key (x.y) => ok if x is of type 'object'\n      fieldName = fieldName.split('.')[0];\n      type = 'Object';\n    }\n    if (!fieldNameIsValid(fieldName, className)) {\n      throw new Parse.Error(Parse.Error.INVALID_KEY_NAME, `Invalid field name: ${fieldName}.`);\n    }\n\n    // If someone tries to create a new field with null/undefined as the value, return;\n    if (!type) {\n      return undefined;\n    }\n\n    const expectedType = this.getExpectedType(className, fieldName);\n    if (typeof type === 'string') {\n      type = ({ type }: SchemaField);\n    }\n\n    if (type.defaultValue !== undefined) {\n      let defaultValueType = getType(type.defaultValue);\n      if (typeof defaultValueType === 'string') {\n        defaultValueType = { type: defaultValueType };\n      }\n      if (!dbTypeMatchesObjectType(type, defaultValueType)) {\n        throw new Parse.Error(\n          Parse.Error.INCORRECT_TYPE,\n          `schema mismatch for ${className}.${fieldName} default value; expected ${typeToString(\n            type\n          )} but got ${typeToString(defaultValueType)}`\n        );\n      }\n    }\n\n    if (expectedType) {\n      if (!dbTypeMatchesObjectType(expectedType, type)) {\n        throw new Parse.Error(\n          Parse.Error.INCORRECT_TYPE,\n          `schema mismatch for ${className}.${fieldName}; expected ${typeToString(\n            expectedType\n          )} but got ${typeToString(type)}`\n        );\n      }\n      // If type options do not change\n      // we can safely return\n      if (isValidation || JSON.stringify(expectedType) === JSON.stringify(type)) {\n        return undefined;\n      }\n      // Field options are may be changed\n      // ensure to have an update to date schema field\n      return this._dbAdapter.updateFieldOptions(className, fieldName, type);\n    }\n\n    return this._dbAdapter\n      .addFieldIfNotExists(className, fieldName, type)\n      .catch(error => {\n        if (error.code == Parse.Error.INCORRECT_TYPE) {\n          // Make sure that we throw errors when it is appropriate to do so.\n          throw error;\n        }\n        // The update failed. This can be okay - it might have been a race\n        // condition where another client updated the schema in the same\n        // way that we wanted to. So, just reload the schema\n        return Promise.resolve();\n      })\n      .then(() => {\n        return {\n          className,\n          fieldName,\n          type,\n        };\n      });\n  }\n\n  ensureFields(fields: any) {\n    for (let i = 0; i < fields.length; i += 1) {\n      const { className, fieldName } = fields[i];\n      let { type } = fields[i];\n      const expectedType = this.getExpectedType(className, fieldName);\n      if (typeof type === 'string') {\n        type = { type: type };\n      }\n      if (!expectedType || !dbTypeMatchesObjectType(expectedType, type)) {\n        throw new Parse.Error(Parse.Error.INVALID_JSON, `Could not add field ${fieldName}`);\n      }\n    }\n  }\n\n  // maintain compatibility\n  deleteField(fieldName: string, className: string, database: DatabaseController) {\n    return this.deleteFields([fieldName], className, database);\n  }\n\n  // Delete fields, and remove that data from all objects. This is intended\n  // to remove unused fields, if other writers are writing objects that include\n  // this field, the field may reappear. Returns a Promise that resolves with\n  // no object on success, or rejects with { code, error } on failure.\n  // Passing the database and prefix is necessary in order to drop relation collections\n  // and remove fields from objects. Ideally the database would belong to\n  // a database adapter and this function would close over it or access it via member.\n  deleteFields(fieldNames: Array<string>, className: string, database: DatabaseController) {\n    if (!classNameIsValid(className)) {\n      throw new Parse.Error(Parse.Error.INVALID_CLASS_NAME, invalidClassNameMessage(className));\n    }\n\n    fieldNames.forEach(fieldName => {\n      if (!fieldNameIsValid(fieldName, className)) {\n        throw new Parse.Error(Parse.Error.INVALID_KEY_NAME, `invalid field name: ${fieldName}`);\n      }\n      //Don't allow deleting the default fields.\n      if (!fieldNameIsValidForClass(fieldName, className)) {\n        throw new Parse.Error(136, `field ${fieldName} cannot be changed`);\n      }\n    });\n\n    return this.getOneSchema(className, false, { clearCache: true })\n      .catch(error => {\n        if (error === undefined) {\n          throw new Parse.Error(\n            Parse.Error.INVALID_CLASS_NAME,\n            `Class ${className} does not exist.`\n          );\n        } else {\n          throw error;\n        }\n      })\n      .then(schema => {\n        fieldNames.forEach(fieldName => {\n          if (!schema.fields[fieldName]) {\n            throw new Parse.Error(255, `Field ${fieldName} does not exist, cannot delete.`);\n          }\n        });\n\n        const schemaFields = { ...schema.fields };\n        return database.adapter.deleteFields(className, schema, fieldNames).then(() => {\n          return Promise.all(\n            fieldNames.map(fieldName => {\n              const field = schemaFields[fieldName];\n              if (field && field.type === 'Relation') {\n                //For relations, drop the _Join table\n                return database.adapter.deleteClass(`_Join:${fieldName}:${className}`);\n              }\n              return Promise.resolve();\n            })\n          );\n        });\n      })\n      .then(() => {\n        SchemaCache.clear();\n      });\n  }\n\n  // Validates an object provided in REST format.\n  // Returns a promise that resolves to the new schema if this object is\n  // valid.\n  async validateObject(className: string, object: any, query: any) {\n    let geocount = 0;\n    const schema = await this.enforceClassExists(className);\n    const promises = [];\n\n    for (const fieldName in object) {\n      if (object[fieldName] && getType(object[fieldName]) === 'GeoPoint') {\n        geocount++;\n      }\n      if (geocount > 1) {\n        return Promise.reject(\n          new Parse.Error(\n            Parse.Error.INCORRECT_TYPE,\n            'there can only be one geopoint field in a class'\n          )\n        );\n      }\n    }\n    for (const fieldName in object) {\n      if (object[fieldName] === undefined) {\n        continue;\n      }\n      const expected = getType(object[fieldName]);\n      if (!expected) {\n        continue;\n      }\n      if (fieldName === 'ACL') {\n        // Every object has ACL implicitly.\n        continue;\n      }\n      promises.push(schema.enforceFieldExists(className, fieldName, expected, true));\n    }\n    const results = await Promise.all(promises);\n    const enforceFields = results.filter(result => !!result);\n\n    if (enforceFields.length !== 0) {\n      // TODO: Remove by updating schema cache directly\n      await this.reloadData({ clearCache: true });\n    }\n    this.ensureFields(enforceFields);\n\n    const promise = Promise.resolve(schema);\n    return thenValidateRequiredColumns(promise, className, object, query);\n  }\n\n  // Validates that all the properties are set for the object\n  validateRequiredColumns(className: string, object: any, query: any) {\n    const columns = requiredColumns.write[className];\n    if (!columns || columns.length == 0) {\n      return Promise.resolve(this);\n    }\n\n    const missingColumns = columns.filter(function (column) {\n      if (query && query.objectId) {\n        if (object[column] && typeof object[column] === 'object') {\n          // Trying to delete a required column\n          return object[column].__op == 'Delete';\n        }\n        // Not trying to do anything there\n        return false;\n      }\n      return !object[column];\n    });\n\n    if (missingColumns.length > 0) {\n      throw new Parse.Error(Parse.Error.INCORRECT_TYPE, missingColumns[0] + ' is required.');\n    }\n    return Promise.resolve(this);\n  }\n\n  testPermissionsForClassName(className: string, aclGroup: string[], operation: string) {\n    return SchemaController.testPermissions(\n      this.getClassLevelPermissions(className),\n      aclGroup,\n      operation\n    );\n  }\n\n  // Tests that the class level permission let pass the operation for a given aclGroup\n  static testPermissions(classPermissions: ?any, aclGroup: string[], operation: string): boolean {\n    if (!classPermissions || !classPermissions[operation]) {\n      return true;\n    }\n    const perms = classPermissions[operation];\n    if (perms['*']) {\n      return true;\n    }\n    // Check permissions against the aclGroup provided (array of userId/roles)\n    if (\n      aclGroup.some(acl => {\n        return perms[acl] === true;\n      })\n    ) {\n      return true;\n    }\n    return false;\n  }\n\n  // Validates an operation passes class-level-permissions set in the schema\n  static validatePermission(\n    classPermissions: ?any,\n    className: string,\n    aclGroup: string[],\n    operation: string,\n    action?: string\n  ) {\n    if (SchemaController.testPermissions(classPermissions, aclGroup, operation)) {\n      return Promise.resolve();\n    }\n\n    if (!classPermissions || !classPermissions[operation]) {\n      return true;\n    }\n    const perms = classPermissions[operation];\n    // If only for authenticated users\n    // make sure we have an aclGroup\n    if (perms['requiresAuthentication']) {\n      // If aclGroup has * (public)\n      if (!aclGroup || aclGroup.length == 0) {\n        throw new Parse.Error(\n          Parse.Error.OBJECT_NOT_FOUND,\n          'Permission denied, user needs to be authenticated.'\n        );\n      } else if (aclGroup.indexOf('*') > -1 && aclGroup.length == 1) {\n        throw new Parse.Error(\n          Parse.Error.OBJECT_NOT_FOUND,\n          'Permission denied, user needs to be authenticated.'\n        );\n      }\n      // requiresAuthentication passed, just move forward\n      // probably would be wise at some point to rename to 'authenticatedUser'\n      return Promise.resolve();\n    }\n\n    // No matching CLP, let's check the Pointer permissions\n    // And handle those later\n    const permissionField =\n      ['get', 'find', 'count'].indexOf(operation) > -1 ? 'readUserFields' : 'writeUserFields';\n\n    // Reject create when write lockdown\n    if (permissionField == 'writeUserFields' && operation == 'create') {\n      throw new Parse.Error(\n        Parse.Error.OPERATION_FORBIDDEN,\n        `Permission denied for action ${operation} on class ${className}.`\n      );\n    }\n\n    // Process the readUserFields later\n    if (\n      Array.isArray(classPermissions[permissionField]) &&\n      classPermissions[permissionField].length > 0\n    ) {\n      return Promise.resolve();\n    }\n\n    const pointerFields = classPermissions[operation].pointerFields;\n    if (Array.isArray(pointerFields) && pointerFields.length > 0) {\n      // any op except 'addField as part of create' is ok.\n      if (operation !== 'addField' || action === 'update') {\n        // We can allow adding field on update flow only.\n        return Promise.resolve();\n      }\n    }\n\n    throw new Parse.Error(\n      Parse.Error.OPERATION_FORBIDDEN,\n      `Permission denied for action ${operation} on class ${className}.`\n    );\n  }\n\n  // Validates an operation passes class-level-permissions set in the schema\n  validatePermission(className: string, aclGroup: string[], operation: string, action?: string) {\n    return SchemaController.validatePermission(\n      this.getClassLevelPermissions(className),\n      className,\n      aclGroup,\n      operation,\n      action\n    );\n  }\n\n  getClassLevelPermissions(className: string): any {\n    return this.schemaData[className] && this.schemaData[className].classLevelPermissions;\n  }\n\n  // Returns the expected type for a className+key combination\n  // or undefined if the schema is not set\n  getExpectedType(className: string, fieldName: string): ?(SchemaField | string) {\n    if (this.schemaData[className]) {\n      const expectedType = this.schemaData[className].fields[fieldName];\n      return expectedType === 'map' ? 'Object' : expectedType;\n    }\n    return undefined;\n  }\n\n  // Checks if a given class is in the schema.\n  hasClass(className: string) {\n    if (this.schemaData[className]) {\n      return Promise.resolve(true);\n    }\n    return this.reloadData().then(() => !!this.schemaData[className]);\n  }\n}\n\n// Returns a promise for a new Schema.\nconst load = (dbAdapter: StorageAdapter, options: any): Promise<SchemaController> => {\n  const schema = new SchemaController(dbAdapter);\n  return schema.reloadData(options).then(() => schema);\n};\n\n// Builds a new schema (in schema API response format) out of an\n// existing mongo schema + a schemas API put request. This response\n// does not include the default fields, as it is intended to be passed\n// to mongoSchemaFromFieldsAndClassName. No validation is done here, it\n// is done in mongoSchemaFromFieldsAndClassName.\nfunction buildMergedSchemaObject(existingFields: SchemaFields, putRequest: any): SchemaFields {\n  const newSchema = {};\n  // @flow-disable-next\n  const sysSchemaField =\n    Object.keys(defaultColumns).indexOf(existingFields._id) === -1\n      ? []\n      : Object.keys(defaultColumns[existingFields._id]);\n  for (const oldField in existingFields) {\n    if (\n      oldField !== '_id' &&\n      oldField !== 'ACL' &&\n      oldField !== 'updatedAt' &&\n      oldField !== 'createdAt' &&\n      oldField !== 'objectId'\n    ) {\n      if (sysSchemaField.length > 0 && sysSchemaField.indexOf(oldField) !== -1) {\n        continue;\n      }\n      const fieldIsDeleted = putRequest[oldField] && putRequest[oldField].__op === 'Delete';\n      if (!fieldIsDeleted) {\n        newSchema[oldField] = existingFields[oldField];\n      }\n    }\n  }\n  for (const newField in putRequest) {\n    if (newField !== 'objectId' && putRequest[newField].__op !== 'Delete') {\n      if (sysSchemaField.length > 0 && sysSchemaField.indexOf(newField) !== -1) {\n        continue;\n      }\n      newSchema[newField] = putRequest[newField];\n    }\n  }\n  return newSchema;\n}\n\n// Given a schema promise, construct another schema promise that\n// validates this field once the schema loads.\nfunction thenValidateRequiredColumns(schemaPromise, className, object, query) {\n  return schemaPromise.then(schema => {\n    return schema.validateRequiredColumns(className, object, query);\n  });\n}\n\n// Gets the type from a REST API formatted object, where 'type' is\n// extended past javascript types to include the rest of the Parse\n// type system.\n// The output should be a valid schema value.\n// TODO: ensure that this is compatible with the format used in Open DB\nfunction getType(obj: any): ?(SchemaField | string) {\n  const type = typeof obj;\n  switch (type) {\n    case 'boolean':\n      return 'Boolean';\n    case 'string':\n      return 'String';\n    case 'number':\n      return 'Number';\n    case 'map':\n    case 'object':\n      if (!obj) {\n        return undefined;\n      }\n      return getObjectType(obj);\n    case 'function':\n    case 'symbol':\n    case 'undefined':\n    default:\n      throw 'bad obj: ' + obj;\n  }\n}\n\n// This gets the type for non-JSON types like pointers and files, but\n// also gets the appropriate type for $ operators.\n// Returns null if the type is unknown.\nfunction getObjectType(obj): ?(SchemaField | string) {\n  if (obj instanceof Array) {\n    return 'Array';\n  }\n  if (obj.__type) {\n    switch (obj.__type) {\n      case 'Pointer':\n        if (obj.className) {\n          return {\n            type: 'Pointer',\n            targetClass: obj.className,\n          };\n        }\n        break;\n      case 'Relation':\n        if (obj.className) {\n          return {\n            type: 'Relation',\n            targetClass: obj.className,\n          };\n        }\n        break;\n      case 'File':\n        if (obj.name) {\n          return 'File';\n        }\n        break;\n      case 'Date':\n        if (obj.iso) {\n          return 'Date';\n        }\n        break;\n      case 'GeoPoint':\n        if (obj.latitude != null && obj.longitude != null) {\n          return 'GeoPoint';\n        }\n        break;\n      case 'Bytes':\n        if (obj.base64) {\n          return 'Bytes';\n        }\n        break;\n      case 'Polygon':\n        if (obj.coordinates) {\n          return 'Polygon';\n        }\n        break;\n    }\n    throw new Parse.Error(Parse.Error.INCORRECT_TYPE, 'This is not a valid ' + obj.__type);\n  }\n  if (obj['$ne']) {\n    return getObjectType(obj['$ne']);\n  }\n  if (obj.__op) {\n    switch (obj.__op) {\n      case 'Increment':\n        return 'Number';\n      case 'Delete':\n        return null;\n      case 'Add':\n      case 'AddUnique':\n      case 'Remove':\n        return 'Array';\n      case 'AddRelation':\n      case 'RemoveRelation':\n        return {\n          type: 'Relation',\n          targetClass: obj.objects[0].className,\n        };\n      case 'Batch':\n        return getObjectType(obj.ops[0]);\n      default:\n        throw 'unexpected op: ' + obj.__op;\n    }\n  }\n  return 'Object';\n}\n\nexport {\n  load,\n  classNameIsValid,\n  fieldNameIsValid,\n  invalidClassNameMessage,\n  buildMergedSchemaObject,\n  systemClasses,\n  defaultColumns,\n  convertSchemaToAdapterSchema,\n  VolatileClassesSchemas,\n  SchemaController,\n  requiredColumns,\n};\n"]}
|
|
1508
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Parse","require","defaultColumns","Object","freeze","_Default","objectId","type","createdAt","updatedAt","ACL","_User","username","password","email","emailVerified","authData","_Installation","installationId","deviceToken","channels","deviceType","pushType","GCMSenderId","timeZone","localeIdentifier","badge","appVersion","appName","appIdentifier","parseVersion","_Role","name","users","targetClass","roles","_Session","user","sessionToken","expiresAt","createdWith","_Product","productIdentifier","download","downloadName","icon","order","title","subtitle","_PushStatus","pushTime","source","query","payload","expiry","expiration_interval","status","numSent","numFailed","pushHash","errorMessage","sentPerType","failedPerType","sentPerUTCOffset","failedPerUTCOffset","count","_JobStatus","jobName","message","params","finishedAt","_JobSchedule","description","startAfter","daysOfWeek","timeOfDay","lastRun","repeatMinutes","_Hooks","functionName","className","triggerName","url","_GlobalConfig","masterKeyOnly","_GraphQLConfig","config","_Audience","lastUsed","timesUsed","_Idempotency","reqId","expire","requiredColumns","read","write","invalidColumns","systemClasses","volatileClasses","roleRegex","protectedFieldsPointerRegex","publicRegex","authenticatedRegex","requiresAuthenticationRegex","clpPointerRegex","protectedFieldsRegex","clpFieldsRegex","validatePermissionKey","key","userIdRegExp","matchesSome","regEx","match","valid","Error","INVALID_JSON","validateProtectedFieldsKey","CLPValidKeys","validateCLP","perms","fields","operationKey","indexOf","operation","validateCLPjson","fieldName","validatePointerPermission","entity","protectedFields","Array","isArray","field","prototype","hasOwnProperty","call","pointerFields","pointerField","permit","joinClassRegex","classAndFieldRegex","classNameIsValid","test","fieldNameIsValid","includes","fieldNameIsValidForClass","invalidClassNameMessage","invalidJsonError","validNonRelationOrPointerTypes","fieldTypeIsInvalid","INVALID_CLASS_NAME","undefined","INCORRECT_TYPE","convertSchemaToAdapterSchema","schema","injectDefaultSchema","_rperm","_wperm","_hashed_password","convertAdapterSchemaToParseSchema","indexes","keys","length","SchemaData","constructor","allSchemas","__data","__protectedFields","forEach","defineProperty","get","data","classLevelPermissions","deepcopy","classProtectedFields","unq","Set","from","defaultSchema","_HooksSchema","_GlobalConfigSchema","_GraphQLConfigSchema","_PushStatusSchema","_JobStatusSchema","_JobScheduleSchema","_AudienceSchema","_IdempotencySchema","VolatileClassesSchemas","dbTypeMatchesObjectType","dbType","objectType","typeToString","SchemaController","databaseAdapter","_dbAdapter","schemaData","SchemaCache","all","Config","applicationId","customIds","allowCustomObjectId","customIdRegEx","autoIdRegEx","userIdRegEx","watch","reloadData","clearCache","options","reloadDataPromise","getAllClasses","then","err","setAllClasses","cached","Promise","resolve","map","put","getOneSchema","allowVolatileClasses","clear","oneSchema","find","reject","addClassIfNotExists","validationError","validateNewClass","code","error","adapterSchema","createClass","parseSchema","DUPLICATE_VALUE","updateClass","submittedFields","database","existingFields","__op","newSchema","buildMergedSchemaObject","defaultFields","fullNewSchema","assign","validateSchemaData","deletedFields","insertedFields","push","deletePromise","deleteFields","enforceFields","promises","enforceFieldExists","results","filter","result","setPermissions","setIndexesWithSchemaFormat","ensureFields","reloadedSchema","catch","enforceClassExists","existingFieldNames","INVALID_KEY_NAME","fieldType","defaultValue","defaultValueType","getType","required","geoPoints","setClassLevelPermissions","isValidation","split","expectedType","getExpectedType","JSON","stringify","updateFieldOptions","addFieldIfNotExists","i","deleteField","fieldNames","schemaFields","adapter","deleteClass","validateObject","object","geocount","expected","promise","thenValidateRequiredColumns","validateRequiredColumns","columns","missingColumns","column","testPermissionsForClassName","aclGroup","testPermissions","getClassLevelPermissions","classPermissions","some","acl","validatePermission","action","OBJECT_NOT_FOUND","permissionField","OPERATION_FORBIDDEN","hasClass","load","dbAdapter","putRequest","sysSchemaField","_id","oldField","fieldIsDeleted","newField","schemaPromise","obj","getObjectType","__type","iso","latitude","longitude","base64","coordinates","objects","ops"],"sources":["../../src/Controllers/SchemaController.js"],"sourcesContent":["// @flow\n// This class handles schema validation, persistence, and modification.\n//\n// Each individual Schema object should be immutable. The helpers to\n// do things with the Schema just return a new schema when the schema\n// is changed.\n//\n// The canonical place to store this Schema is in the database itself,\n// in a _SCHEMA collection. This is not the right way to do it for an\n// open source framework, but it's backward compatible, so we're\n// keeping it this way for now.\n//\n// In API-handling code, you should only use the Schema class via the\n// DatabaseController. This will let us replace the schema logic for\n// different databases.\n// TODO: hide all schema logic inside the database adapter.\n// @flow-disable-next\nconst Parse = require('parse/node').Parse;\nimport { StorageAdapter } from '../Adapters/Storage/StorageAdapter';\nimport SchemaCache from '../Adapters/Cache/SchemaCache';\nimport DatabaseController from './DatabaseController';\nimport Config from '../Config';\n// @flow-disable-next\nimport deepcopy from 'deepcopy';\nimport type {\n  Schema,\n  SchemaFields,\n  ClassLevelPermissions,\n  SchemaField,\n  LoadSchemaOptions,\n} from './types';\n\nconst defaultColumns: { [string]: SchemaFields } = Object.freeze({\n  // Contain the default columns for every parse object type (except _Join collection)\n  _Default: {\n    objectId: { type: 'String' },\n    createdAt: { type: 'Date' },\n    updatedAt: { type: 'Date' },\n    ACL: { type: 'ACL' },\n  },\n  // The additional default columns for the _User collection (in addition to DefaultCols)\n  _User: {\n    username: { type: 'String' },\n    password: { type: 'String' },\n    email: { type: 'String' },\n    emailVerified: { type: 'Boolean' },\n    authData: { type: 'Object' },\n  },\n  // The additional default columns for the _Installation collection (in addition to DefaultCols)\n  _Installation: {\n    installationId: { type: 'String' },\n    deviceToken: { type: 'String' },\n    channels: { type: 'Array' },\n    deviceType: { type: 'String' },\n    pushType: { type: 'String' },\n    GCMSenderId: { type: 'String' },\n    timeZone: { type: 'String' },\n    localeIdentifier: { type: 'String' },\n    badge: { type: 'Number' },\n    appVersion: { type: 'String' },\n    appName: { type: 'String' },\n    appIdentifier: { type: 'String' },\n    parseVersion: { type: 'String' },\n  },\n  // The additional default columns for the _Role collection (in addition to DefaultCols)\n  _Role: {\n    name: { type: 'String' },\n    users: { type: 'Relation', targetClass: '_User' },\n    roles: { type: 'Relation', targetClass: '_Role' },\n  },\n  // The additional default columns for the _Session collection (in addition to DefaultCols)\n  _Session: {\n    user: { type: 'Pointer', targetClass: '_User' },\n    installationId: { type: 'String' },\n    sessionToken: { type: 'String' },\n    expiresAt: { type: 'Date' },\n    createdWith: { type: 'Object' },\n  },\n  _Product: {\n    productIdentifier: { type: 'String' },\n    download: { type: 'File' },\n    downloadName: { type: 'String' },\n    icon: { type: 'File' },\n    order: { type: 'Number' },\n    title: { type: 'String' },\n    subtitle: { type: 'String' },\n  },\n  _PushStatus: {\n    pushTime: { type: 'String' },\n    source: { type: 'String' }, // rest or webui\n    query: { type: 'String' }, // the stringified JSON query\n    payload: { type: 'String' }, // the stringified JSON payload,\n    title: { type: 'String' },\n    expiry: { type: 'Number' },\n    expiration_interval: { type: 'Number' },\n    status: { type: 'String' },\n    numSent: { type: 'Number' },\n    numFailed: { type: 'Number' },\n    pushHash: { type: 'String' },\n    errorMessage: { type: 'Object' },\n    sentPerType: { type: 'Object' },\n    failedPerType: { type: 'Object' },\n    sentPerUTCOffset: { type: 'Object' },\n    failedPerUTCOffset: { type: 'Object' },\n    count: { type: 'Number' }, // tracks # of batches queued and pending\n  },\n  _JobStatus: {\n    jobName: { type: 'String' },\n    source: { type: 'String' },\n    status: { type: 'String' },\n    message: { type: 'String' },\n    params: { type: 'Object' }, // params received when calling the job\n    finishedAt: { type: 'Date' },\n  },\n  _JobSchedule: {\n    jobName: { type: 'String' },\n    description: { type: 'String' },\n    params: { type: 'String' },\n    startAfter: { type: 'String' },\n    daysOfWeek: { type: 'Array' },\n    timeOfDay: { type: 'String' },\n    lastRun: { type: 'Number' },\n    repeatMinutes: { type: 'Number' },\n  },\n  _Hooks: {\n    functionName: { type: 'String' },\n    className: { type: 'String' },\n    triggerName: { type: 'String' },\n    url: { type: 'String' },\n  },\n  _GlobalConfig: {\n    objectId: { type: 'String' },\n    params: { type: 'Object' },\n    masterKeyOnly: { type: 'Object' },\n  },\n  _GraphQLConfig: {\n    objectId: { type: 'String' },\n    config: { type: 'Object' },\n  },\n  _Audience: {\n    objectId: { type: 'String' },\n    name: { type: 'String' },\n    query: { type: 'String' }, //storing query as JSON string to prevent \"Nested keys should not contain the '$' or '.' characters\" error\n    lastUsed: { type: 'Date' },\n    timesUsed: { type: 'Number' },\n  },\n  _Idempotency: {\n    reqId: { type: 'String' },\n    expire: { type: 'Date' },\n  },\n});\n\n// fields required for read or write operations on their respective classes.\nconst requiredColumns = Object.freeze({\n  read: {\n    _User: ['username'],\n  },\n  write: {\n    _Product: ['productIdentifier', 'icon', 'order', 'title', 'subtitle'],\n    _Role: ['name', 'ACL'],\n  },\n});\n\nconst invalidColumns = ['length'];\n\nconst systemClasses = Object.freeze([\n  '_User',\n  '_Installation',\n  '_Role',\n  '_Session',\n  '_Product',\n  '_PushStatus',\n  '_JobStatus',\n  '_JobSchedule',\n  '_Audience',\n  '_Idempotency',\n]);\n\nconst volatileClasses = Object.freeze([\n  '_JobStatus',\n  '_PushStatus',\n  '_Hooks',\n  '_GlobalConfig',\n  '_GraphQLConfig',\n  '_JobSchedule',\n  '_Audience',\n  '_Idempotency',\n]);\n\n// Anything that start with role\nconst roleRegex = /^role:.*/;\n// Anything that starts with userField (allowed for protected fields only)\nconst protectedFieldsPointerRegex = /^userField:.*/;\n// * permission\nconst publicRegex = /^\\*$/;\n\nconst authenticatedRegex = /^authenticated$/;\n\nconst requiresAuthenticationRegex = /^requiresAuthentication$/;\n\nconst clpPointerRegex = /^pointerFields$/;\n\n// regex for validating entities in protectedFields object\nconst protectedFieldsRegex = Object.freeze([\n  protectedFieldsPointerRegex,\n  publicRegex,\n  authenticatedRegex,\n  roleRegex,\n]);\n\n// clp regex\nconst clpFieldsRegex = Object.freeze([\n  clpPointerRegex,\n  publicRegex,\n  requiresAuthenticationRegex,\n  roleRegex,\n]);\n\nfunction validatePermissionKey(key, userIdRegExp) {\n  let matchesSome = false;\n  for (const regEx of clpFieldsRegex) {\n    if (key.match(regEx) !== null) {\n      matchesSome = true;\n      break;\n    }\n  }\n\n  // userId depends on startup options so it's dynamic\n  const valid = matchesSome || key.match(userIdRegExp) !== null;\n  if (!valid) {\n    throw new Parse.Error(\n      Parse.Error.INVALID_JSON,\n      `'${key}' is not a valid key for class level permissions`\n    );\n  }\n}\n\nfunction validateProtectedFieldsKey(key, userIdRegExp) {\n  let matchesSome = false;\n  for (const regEx of protectedFieldsRegex) {\n    if (key.match(regEx) !== null) {\n      matchesSome = true;\n      break;\n    }\n  }\n\n  // userId regex depends on launch options so it's dynamic\n  const valid = matchesSome || key.match(userIdRegExp) !== null;\n  if (!valid) {\n    throw new Parse.Error(\n      Parse.Error.INVALID_JSON,\n      `'${key}' is not a valid key for class level permissions`\n    );\n  }\n}\n\nconst CLPValidKeys = Object.freeze([\n  'find',\n  'count',\n  'get',\n  'create',\n  'update',\n  'delete',\n  'addField',\n  'readUserFields',\n  'writeUserFields',\n  'protectedFields',\n]);\n\n// validation before setting class-level permissions on collection\nfunction validateCLP(perms: ClassLevelPermissions, fields: SchemaFields, userIdRegExp: RegExp) {\n  if (!perms) {\n    return;\n  }\n  for (const operationKey in perms) {\n    if (CLPValidKeys.indexOf(operationKey) == -1) {\n      throw new Parse.Error(\n        Parse.Error.INVALID_JSON,\n        `${operationKey} is not a valid operation for class level permissions`\n      );\n    }\n\n    const operation = perms[operationKey];\n    // proceed with next operationKey\n\n    // throws when root fields are of wrong type\n    validateCLPjson(operation, operationKey);\n\n    if (operationKey === 'readUserFields' || operationKey === 'writeUserFields') {\n      // validate grouped pointer permissions\n      // must be an array with field names\n      for (const fieldName of operation) {\n        validatePointerPermission(fieldName, fields, operationKey);\n      }\n      // readUserFields and writerUserFields do not have nesdted fields\n      // proceed with next operationKey\n      continue;\n    }\n\n    // validate protected fields\n    if (operationKey === 'protectedFields') {\n      for (const entity in operation) {\n        // throws on unexpected key\n        validateProtectedFieldsKey(entity, userIdRegExp);\n\n        const protectedFields = operation[entity];\n\n        if (!Array.isArray(protectedFields)) {\n          throw new Parse.Error(\n            Parse.Error.INVALID_JSON,\n            `'${protectedFields}' is not a valid value for protectedFields[${entity}] - expected an array.`\n          );\n        }\n\n        // if the field is in form of array\n        for (const field of protectedFields) {\n          // do not alloow to protect default fields\n          if (defaultColumns._Default[field]) {\n            throw new Parse.Error(\n              Parse.Error.INVALID_JSON,\n              `Default field '${field}' can not be protected`\n            );\n          }\n          // field should exist on collection\n          if (!Object.prototype.hasOwnProperty.call(fields, field)) {\n            throw new Parse.Error(\n              Parse.Error.INVALID_JSON,\n              `Field '${field}' in protectedFields:${entity} does not exist`\n            );\n          }\n        }\n      }\n      // proceed with next operationKey\n      continue;\n    }\n\n    // validate other fields\n    // Entity can be:\n    // \"*\" - Public,\n    // \"requiresAuthentication\" - authenticated users,\n    // \"objectId\" - _User id,\n    // \"role:rolename\",\n    // \"pointerFields\" - array of field names containing pointers to users\n    for (const entity in operation) {\n      // throws on unexpected key\n      validatePermissionKey(entity, userIdRegExp);\n\n      // entity can be either:\n      // \"pointerFields\": string[]\n      if (entity === 'pointerFields') {\n        const pointerFields = operation[entity];\n\n        if (Array.isArray(pointerFields)) {\n          for (const pointerField of pointerFields) {\n            validatePointerPermission(pointerField, fields, operation);\n          }\n        } else {\n          throw new Parse.Error(\n            Parse.Error.INVALID_JSON,\n            `'${pointerFields}' is not a valid value for ${operationKey}[${entity}] - expected an array.`\n          );\n        }\n        // proceed with next entity key\n        continue;\n      }\n\n      // or [entity]: boolean\n      const permit = operation[entity];\n\n      if (permit !== true) {\n        throw new Parse.Error(\n          Parse.Error.INVALID_JSON,\n          `'${permit}' is not a valid value for class level permissions ${operationKey}:${entity}:${permit}`\n        );\n      }\n    }\n  }\n}\n\nfunction validateCLPjson(operation: any, operationKey: string) {\n  if (operationKey === 'readUserFields' || operationKey === 'writeUserFields') {\n    if (!Array.isArray(operation)) {\n      throw new Parse.Error(\n        Parse.Error.INVALID_JSON,\n        `'${operation}' is not a valid value for class level permissions ${operationKey} - must be an array`\n      );\n    }\n  } else {\n    if (typeof operation === 'object' && operation !== null) {\n      // ok to proceed\n      return;\n    } else {\n      throw new Parse.Error(\n        Parse.Error.INVALID_JSON,\n        `'${operation}' is not a valid value for class level permissions ${operationKey} - must be an object`\n      );\n    }\n  }\n}\n\nfunction validatePointerPermission(fieldName: string, fields: Object, operation: string) {\n  // Uses collection schema to ensure the field is of type:\n  // - Pointer<_User> (pointers)\n  // - Array\n  //\n  //    It's not possible to enforce type on Array's items in schema\n  //  so we accept any Array field, and later when applying permissions\n  //  only items that are pointers to _User are considered.\n  if (\n    !(\n      fields[fieldName] &&\n      ((fields[fieldName].type == 'Pointer' && fields[fieldName].targetClass == '_User') ||\n        fields[fieldName].type == 'Array')\n    )\n  ) {\n    throw new Parse.Error(\n      Parse.Error.INVALID_JSON,\n      `'${fieldName}' is not a valid column for class level pointer permissions ${operation}`\n    );\n  }\n}\n\nconst joinClassRegex = /^_Join:[A-Za-z0-9_]+:[A-Za-z0-9_]+/;\nconst classAndFieldRegex = /^[A-Za-z][A-Za-z0-9_]*$/;\nfunction classNameIsValid(className: string): boolean {\n  // Valid classes must:\n  return (\n    // Be one of _User, _Installation, _Role, _Session OR\n    systemClasses.indexOf(className) > -1 ||\n    // Be a join table OR\n    joinClassRegex.test(className) ||\n    // Include only alpha-numeric and underscores, and not start with an underscore or number\n    fieldNameIsValid(className, className)\n  );\n}\n\n// Valid fields must be alpha-numeric, and not start with an underscore or number\n// must not be a reserved key\nfunction fieldNameIsValid(fieldName: string, className: string): boolean {\n  if (className && className !== '_Hooks') {\n    if (fieldName === 'className') {\n      return false;\n    }\n  }\n  return classAndFieldRegex.test(fieldName) && !invalidColumns.includes(fieldName);\n}\n\n// Checks that it's not trying to clobber one of the default fields of the class.\nfunction fieldNameIsValidForClass(fieldName: string, className: string): boolean {\n  if (!fieldNameIsValid(fieldName, className)) {\n    return false;\n  }\n  if (defaultColumns._Default[fieldName]) {\n    return false;\n  }\n  if (defaultColumns[className] && defaultColumns[className][fieldName]) {\n    return false;\n  }\n  return true;\n}\n\nfunction invalidClassNameMessage(className: string): string {\n  return (\n    'Invalid classname: ' +\n    className +\n    ', classnames can only have alphanumeric characters and _, and must start with an alpha character '\n  );\n}\n\nconst invalidJsonError = new Parse.Error(Parse.Error.INVALID_JSON, 'invalid JSON');\nconst validNonRelationOrPointerTypes = [\n  'Number',\n  'String',\n  'Boolean',\n  'Date',\n  'Object',\n  'Array',\n  'GeoPoint',\n  'File',\n  'Bytes',\n  'Polygon',\n];\n// Returns an error suitable for throwing if the type is invalid\nconst fieldTypeIsInvalid = ({ type, targetClass }) => {\n  if (['Pointer', 'Relation'].indexOf(type) >= 0) {\n    if (!targetClass) {\n      return new Parse.Error(135, `type ${type} needs a class name`);\n    } else if (typeof targetClass !== 'string') {\n      return invalidJsonError;\n    } else if (!classNameIsValid(targetClass)) {\n      return new Parse.Error(Parse.Error.INVALID_CLASS_NAME, invalidClassNameMessage(targetClass));\n    } else {\n      return undefined;\n    }\n  }\n  if (typeof type !== 'string') {\n    return invalidJsonError;\n  }\n  if (validNonRelationOrPointerTypes.indexOf(type) < 0) {\n    return new Parse.Error(Parse.Error.INCORRECT_TYPE, `invalid field type: ${type}`);\n  }\n  return undefined;\n};\n\nconst convertSchemaToAdapterSchema = (schema: any) => {\n  schema = injectDefaultSchema(schema);\n  delete schema.fields.ACL;\n  schema.fields._rperm = { type: 'Array' };\n  schema.fields._wperm = { type: 'Array' };\n\n  if (schema.className === '_User') {\n    delete schema.fields.password;\n    schema.fields._hashed_password = { type: 'String' };\n  }\n\n  return schema;\n};\n\nconst convertAdapterSchemaToParseSchema = ({ ...schema }) => {\n  delete schema.fields._rperm;\n  delete schema.fields._wperm;\n\n  schema.fields.ACL = { type: 'ACL' };\n\n  if (schema.className === '_User') {\n    delete schema.fields.authData; //Auth data is implicit\n    delete schema.fields._hashed_password;\n    schema.fields.password = { type: 'String' };\n  }\n\n  if (schema.indexes && Object.keys(schema.indexes).length === 0) {\n    delete schema.indexes;\n  }\n\n  return schema;\n};\n\nclass SchemaData {\n  __data: any;\n  __protectedFields: any;\n  constructor(allSchemas = [], protectedFields = {}) {\n    this.__data = {};\n    this.__protectedFields = protectedFields;\n    allSchemas.forEach(schema => {\n      if (volatileClasses.includes(schema.className)) {\n        return;\n      }\n      Object.defineProperty(this, schema.className, {\n        get: () => {\n          if (!this.__data[schema.className]) {\n            const data = {};\n            data.fields = injectDefaultSchema(schema).fields;\n            data.classLevelPermissions = deepcopy(schema.classLevelPermissions);\n            data.indexes = schema.indexes;\n\n            const classProtectedFields = this.__protectedFields[schema.className];\n            if (classProtectedFields) {\n              for (const key in classProtectedFields) {\n                const unq = new Set([\n                  ...(data.classLevelPermissions.protectedFields[key] || []),\n                  ...classProtectedFields[key],\n                ]);\n                data.classLevelPermissions.protectedFields[key] = Array.from(unq);\n              }\n            }\n\n            this.__data[schema.className] = data;\n          }\n          return this.__data[schema.className];\n        },\n      });\n    });\n\n    // Inject the in-memory classes\n    volatileClasses.forEach(className => {\n      Object.defineProperty(this, className, {\n        get: () => {\n          if (!this.__data[className]) {\n            const schema = injectDefaultSchema({\n              className,\n              fields: {},\n              classLevelPermissions: {},\n            });\n            const data = {};\n            data.fields = schema.fields;\n            data.classLevelPermissions = schema.classLevelPermissions;\n            data.indexes = schema.indexes;\n            this.__data[className] = data;\n          }\n          return this.__data[className];\n        },\n      });\n    });\n  }\n}\n\nconst injectDefaultSchema = ({ className, fields, classLevelPermissions, indexes }: Schema) => {\n  const defaultSchema: Schema = {\n    className,\n    fields: {\n      ...defaultColumns._Default,\n      ...(defaultColumns[className] || {}),\n      ...fields,\n    },\n    classLevelPermissions,\n  };\n  if (indexes && Object.keys(indexes).length !== 0) {\n    defaultSchema.indexes = indexes;\n  }\n  return defaultSchema;\n};\n\nconst _HooksSchema = { className: '_Hooks', fields: defaultColumns._Hooks };\nconst _GlobalConfigSchema = {\n  className: '_GlobalConfig',\n  fields: defaultColumns._GlobalConfig,\n};\nconst _GraphQLConfigSchema = {\n  className: '_GraphQLConfig',\n  fields: defaultColumns._GraphQLConfig,\n};\nconst _PushStatusSchema = convertSchemaToAdapterSchema(\n  injectDefaultSchema({\n    className: '_PushStatus',\n    fields: {},\n    classLevelPermissions: {},\n  })\n);\nconst _JobStatusSchema = convertSchemaToAdapterSchema(\n  injectDefaultSchema({\n    className: '_JobStatus',\n    fields: {},\n    classLevelPermissions: {},\n  })\n);\nconst _JobScheduleSchema = convertSchemaToAdapterSchema(\n  injectDefaultSchema({\n    className: '_JobSchedule',\n    fields: {},\n    classLevelPermissions: {},\n  })\n);\nconst _AudienceSchema = convertSchemaToAdapterSchema(\n  injectDefaultSchema({\n    className: '_Audience',\n    fields: defaultColumns._Audience,\n    classLevelPermissions: {},\n  })\n);\nconst _IdempotencySchema = convertSchemaToAdapterSchema(\n  injectDefaultSchema({\n    className: '_Idempotency',\n    fields: defaultColumns._Idempotency,\n    classLevelPermissions: {},\n  })\n);\nconst VolatileClassesSchemas = [\n  _HooksSchema,\n  _JobStatusSchema,\n  _JobScheduleSchema,\n  _PushStatusSchema,\n  _GlobalConfigSchema,\n  _GraphQLConfigSchema,\n  _AudienceSchema,\n  _IdempotencySchema,\n];\n\nconst dbTypeMatchesObjectType = (dbType: SchemaField | string, objectType: SchemaField) => {\n  if (dbType.type !== objectType.type) return false;\n  if (dbType.targetClass !== objectType.targetClass) return false;\n  if (dbType === objectType.type) return true;\n  if (dbType.type === objectType.type) return true;\n  return false;\n};\n\nconst typeToString = (type: SchemaField | string): string => {\n  if (typeof type === 'string') {\n    return type;\n  }\n  if (type.targetClass) {\n    return `${type.type}<${type.targetClass}>`;\n  }\n  return `${type.type}`;\n};\n\n// Stores the entire schema of the app in a weird hybrid format somewhere between\n// the mongo format and the Parse format. Soon, this will all be Parse format.\nexport default class SchemaController {\n  _dbAdapter: StorageAdapter;\n  schemaData: { [string]: Schema };\n  reloadDataPromise: ?Promise<any>;\n  protectedFields: any;\n  userIdRegEx: RegExp;\n\n  constructor(databaseAdapter: StorageAdapter) {\n    this._dbAdapter = databaseAdapter;\n    this.schemaData = new SchemaData(SchemaCache.all(), this.protectedFields);\n    this.protectedFields = Config.get(Parse.applicationId).protectedFields;\n\n    const customIds = Config.get(Parse.applicationId).allowCustomObjectId;\n\n    const customIdRegEx = /^.{1,}$/u; // 1+ chars\n    const autoIdRegEx = /^[a-zA-Z0-9]{1,}$/;\n\n    this.userIdRegEx = customIds ? customIdRegEx : autoIdRegEx;\n\n    this._dbAdapter.watch(() => {\n      this.reloadData({ clearCache: true });\n    });\n  }\n\n  reloadData(options: LoadSchemaOptions = { clearCache: false }): Promise<any> {\n    if (this.reloadDataPromise && !options.clearCache) {\n      return this.reloadDataPromise;\n    }\n    this.reloadDataPromise = this.getAllClasses(options)\n      .then(\n        allSchemas => {\n          this.schemaData = new SchemaData(allSchemas, this.protectedFields);\n          delete this.reloadDataPromise;\n        },\n        err => {\n          this.schemaData = new SchemaData();\n          delete this.reloadDataPromise;\n          throw err;\n        }\n      )\n      .then(() => {});\n    return this.reloadDataPromise;\n  }\n\n  getAllClasses(options: LoadSchemaOptions = { clearCache: false }): Promise<Array<Schema>> {\n    if (options.clearCache) {\n      return this.setAllClasses();\n    }\n    const cached = SchemaCache.all();\n    if (cached && cached.length) {\n      return Promise.resolve(cached);\n    }\n    return this.setAllClasses();\n  }\n\n  setAllClasses(): Promise<Array<Schema>> {\n    return this._dbAdapter\n      .getAllClasses()\n      .then(allSchemas => allSchemas.map(injectDefaultSchema))\n      .then(allSchemas => {\n        SchemaCache.put(allSchemas);\n        return allSchemas;\n      });\n  }\n\n  getOneSchema(\n    className: string,\n    allowVolatileClasses: boolean = false,\n    options: LoadSchemaOptions = { clearCache: false }\n  ): Promise<Schema> {\n    if (options.clearCache) {\n      SchemaCache.clear();\n    }\n    if (allowVolatileClasses && volatileClasses.indexOf(className) > -1) {\n      const data = this.schemaData[className];\n      return Promise.resolve({\n        className,\n        fields: data.fields,\n        classLevelPermissions: data.classLevelPermissions,\n        indexes: data.indexes,\n      });\n    }\n    const cached = SchemaCache.get(className);\n    if (cached && !options.clearCache) {\n      return Promise.resolve(cached);\n    }\n    return this.setAllClasses().then(allSchemas => {\n      const oneSchema = allSchemas.find(schema => schema.className === className);\n      if (!oneSchema) {\n        return Promise.reject(undefined);\n      }\n      return oneSchema;\n    });\n  }\n\n  // Create a new class that includes the three default fields.\n  // ACL is an implicit column that does not get an entry in the\n  // _SCHEMAS database. Returns a promise that resolves with the\n  // created schema, in mongo format.\n  // on success, and rejects with an error on fail. Ensure you\n  // have authorization (master key, or client class creation\n  // enabled) before calling this function.\n  async addClassIfNotExists(\n    className: string,\n    fields: SchemaFields = {},\n    classLevelPermissions: any,\n    indexes: any = {}\n  ): Promise<void | Schema> {\n    var validationError = this.validateNewClass(className, fields, classLevelPermissions);\n    if (validationError) {\n      if (validationError instanceof Parse.Error) {\n        return Promise.reject(validationError);\n      } else if (validationError.code && validationError.error) {\n        return Promise.reject(new Parse.Error(validationError.code, validationError.error));\n      }\n      return Promise.reject(validationError);\n    }\n    try {\n      const adapterSchema = await this._dbAdapter.createClass(\n        className,\n        convertSchemaToAdapterSchema({\n          fields,\n          classLevelPermissions,\n          indexes,\n          className,\n        })\n      );\n      // TODO: Remove by updating schema cache directly\n      await this.reloadData({ clearCache: true });\n      const parseSchema = convertAdapterSchemaToParseSchema(adapterSchema);\n      return parseSchema;\n    } catch (error) {\n      if (error && error.code === Parse.Error.DUPLICATE_VALUE) {\n        throw new Parse.Error(Parse.Error.INVALID_CLASS_NAME, `Class ${className} already exists.`);\n      } else {\n        throw error;\n      }\n    }\n  }\n\n  updateClass(\n    className: string,\n    submittedFields: SchemaFields,\n    classLevelPermissions: any,\n    indexes: any,\n    database: DatabaseController\n  ) {\n    return this.getOneSchema(className)\n      .then(schema => {\n        const existingFields = schema.fields;\n        Object.keys(submittedFields).forEach(name => {\n          const field = submittedFields[name];\n          if (\n            existingFields[name] &&\n            existingFields[name].type !== field.type &&\n            field.__op !== 'Delete'\n          ) {\n            throw new Parse.Error(255, `Field ${name} exists, cannot update.`);\n          }\n          if (!existingFields[name] && field.__op === 'Delete') {\n            throw new Parse.Error(255, `Field ${name} does not exist, cannot delete.`);\n          }\n        });\n\n        delete existingFields._rperm;\n        delete existingFields._wperm;\n        const newSchema = buildMergedSchemaObject(existingFields, submittedFields);\n        const defaultFields = defaultColumns[className] || defaultColumns._Default;\n        const fullNewSchema = Object.assign({}, newSchema, defaultFields);\n        const validationError = this.validateSchemaData(\n          className,\n          newSchema,\n          classLevelPermissions,\n          Object.keys(existingFields)\n        );\n        if (validationError) {\n          throw new Parse.Error(validationError.code, validationError.error);\n        }\n\n        // Finally we have checked to make sure the request is valid and we can start deleting fields.\n        // Do all deletions first, then a single save to _SCHEMA collection to handle all additions.\n        const deletedFields: string[] = [];\n        const insertedFields = [];\n        Object.keys(submittedFields).forEach(fieldName => {\n          if (submittedFields[fieldName].__op === 'Delete') {\n            deletedFields.push(fieldName);\n          } else {\n            insertedFields.push(fieldName);\n          }\n        });\n\n        let deletePromise = Promise.resolve();\n        if (deletedFields.length > 0) {\n          deletePromise = this.deleteFields(deletedFields, className, database);\n        }\n        let enforceFields = [];\n        return (\n          deletePromise // Delete Everything\n            .then(() => this.reloadData({ clearCache: true })) // Reload our Schema, so we have all the new values\n            .then(() => {\n              const promises = insertedFields.map(fieldName => {\n                const type = submittedFields[fieldName];\n                return this.enforceFieldExists(className, fieldName, type);\n              });\n              return Promise.all(promises);\n            })\n            .then(results => {\n              enforceFields = results.filter(result => !!result);\n              return this.setPermissions(className, classLevelPermissions, newSchema);\n            })\n            .then(() =>\n              this._dbAdapter.setIndexesWithSchemaFormat(\n                className,\n                indexes,\n                schema.indexes,\n                fullNewSchema\n              )\n            )\n            .then(() => this.reloadData({ clearCache: true }))\n            //TODO: Move this logic into the database adapter\n            .then(() => {\n              this.ensureFields(enforceFields);\n              const schema = this.schemaData[className];\n              const reloadedSchema: Schema = {\n                className: className,\n                fields: schema.fields,\n                classLevelPermissions: schema.classLevelPermissions,\n              };\n              if (schema.indexes && Object.keys(schema.indexes).length !== 0) {\n                reloadedSchema.indexes = schema.indexes;\n              }\n              return reloadedSchema;\n            })\n        );\n      })\n      .catch(error => {\n        if (error === undefined) {\n          throw new Parse.Error(\n            Parse.Error.INVALID_CLASS_NAME,\n            `Class ${className} does not exist.`\n          );\n        } else {\n          throw error;\n        }\n      });\n  }\n\n  // Returns a promise that resolves successfully to the new schema\n  // object or fails with a reason.\n  enforceClassExists(className: string): Promise<SchemaController> {\n    if (this.schemaData[className]) {\n      return Promise.resolve(this);\n    }\n    // We don't have this class. Update the schema\n    return (\n      // The schema update succeeded. Reload the schema\n      this.addClassIfNotExists(className)\n        .catch(() => {\n          // The schema update failed. This can be okay - it might\n          // have failed because there's a race condition and a different\n          // client is making the exact same schema update that we want.\n          // So just reload the schema.\n          return this.reloadData({ clearCache: true });\n        })\n        .then(() => {\n          // Ensure that the schema now validates\n          if (this.schemaData[className]) {\n            return this;\n          } else {\n            throw new Parse.Error(Parse.Error.INVALID_JSON, `Failed to add ${className}`);\n          }\n        })\n        .catch(() => {\n          // The schema still doesn't validate. Give up\n          throw new Parse.Error(Parse.Error.INVALID_JSON, 'schema class name does not revalidate');\n        })\n    );\n  }\n\n  validateNewClass(className: string, fields: SchemaFields = {}, classLevelPermissions: any): any {\n    if (this.schemaData[className]) {\n      throw new Parse.Error(Parse.Error.INVALID_CLASS_NAME, `Class ${className} already exists.`);\n    }\n    if (!classNameIsValid(className)) {\n      return {\n        code: Parse.Error.INVALID_CLASS_NAME,\n        error: invalidClassNameMessage(className),\n      };\n    }\n    return this.validateSchemaData(className, fields, classLevelPermissions, []);\n  }\n\n  validateSchemaData(\n    className: string,\n    fields: SchemaFields,\n    classLevelPermissions: ClassLevelPermissions,\n    existingFieldNames: Array<string>\n  ) {\n    for (const fieldName in fields) {\n      if (existingFieldNames.indexOf(fieldName) < 0) {\n        if (!fieldNameIsValid(fieldName, className)) {\n          return {\n            code: Parse.Error.INVALID_KEY_NAME,\n            error: 'invalid field name: ' + fieldName,\n          };\n        }\n        if (!fieldNameIsValidForClass(fieldName, className)) {\n          return {\n            code: 136,\n            error: 'field ' + fieldName + ' cannot be added',\n          };\n        }\n        const fieldType = fields[fieldName];\n        const error = fieldTypeIsInvalid(fieldType);\n        if (error) return { code: error.code, error: error.message };\n        if (fieldType.defaultValue !== undefined) {\n          let defaultValueType = getType(fieldType.defaultValue);\n          if (typeof defaultValueType === 'string') {\n            defaultValueType = { type: defaultValueType };\n          } else if (typeof defaultValueType === 'object' && fieldType.type === 'Relation') {\n            return {\n              code: Parse.Error.INCORRECT_TYPE,\n              error: `The 'default value' option is not applicable for ${typeToString(fieldType)}`,\n            };\n          }\n          if (!dbTypeMatchesObjectType(fieldType, defaultValueType)) {\n            return {\n              code: Parse.Error.INCORRECT_TYPE,\n              error: `schema mismatch for ${className}.${fieldName} default value; expected ${typeToString(\n                fieldType\n              )} but got ${typeToString(defaultValueType)}`,\n            };\n          }\n        } else if (fieldType.required) {\n          if (typeof fieldType === 'object' && fieldType.type === 'Relation') {\n            return {\n              code: Parse.Error.INCORRECT_TYPE,\n              error: `The 'required' option is not applicable for ${typeToString(fieldType)}`,\n            };\n          }\n        }\n      }\n    }\n\n    for (const fieldName in defaultColumns[className]) {\n      fields[fieldName] = defaultColumns[className][fieldName];\n    }\n\n    const geoPoints = Object.keys(fields).filter(\n      key => fields[key] && fields[key].type === 'GeoPoint'\n    );\n    if (geoPoints.length > 1) {\n      return {\n        code: Parse.Error.INCORRECT_TYPE,\n        error:\n          'currently, only one GeoPoint field may exist in an object. Adding ' +\n          geoPoints[1] +\n          ' when ' +\n          geoPoints[0] +\n          ' already exists.',\n      };\n    }\n    validateCLP(classLevelPermissions, fields, this.userIdRegEx);\n  }\n\n  // Sets the Class-level permissions for a given className, which must exist.\n  async setPermissions(className: string, perms: any, newSchema: SchemaFields) {\n    if (typeof perms === 'undefined') {\n      return Promise.resolve();\n    }\n    validateCLP(perms, newSchema, this.userIdRegEx);\n    await this._dbAdapter.setClassLevelPermissions(className, perms);\n    const cached = SchemaCache.get(className);\n    if (cached) {\n      cached.classLevelPermissions = perms;\n    }\n  }\n\n  // Returns a promise that resolves successfully to the new schema\n  // object if the provided className-fieldName-type tuple is valid.\n  // The className must already be validated.\n  // If 'freeze' is true, refuse to update the schema for this field.\n  enforceFieldExists(\n    className: string,\n    fieldName: string,\n    type: string | SchemaField,\n    isValidation?: boolean\n  ) {\n    if (fieldName.indexOf('.') > 0) {\n      // subdocument key (x.y) => ok if x is of type 'object'\n      fieldName = fieldName.split('.')[0];\n      type = 'Object';\n    }\n    if (!fieldNameIsValid(fieldName, className)) {\n      throw new Parse.Error(Parse.Error.INVALID_KEY_NAME, `Invalid field name: ${fieldName}.`);\n    }\n\n    // If someone tries to create a new field with null/undefined as the value, return;\n    if (!type) {\n      return undefined;\n    }\n\n    const expectedType = this.getExpectedType(className, fieldName);\n    if (typeof type === 'string') {\n      type = ({ type }: SchemaField);\n    }\n\n    if (type.defaultValue !== undefined) {\n      let defaultValueType = getType(type.defaultValue);\n      if (typeof defaultValueType === 'string') {\n        defaultValueType = { type: defaultValueType };\n      }\n      if (!dbTypeMatchesObjectType(type, defaultValueType)) {\n        throw new Parse.Error(\n          Parse.Error.INCORRECT_TYPE,\n          `schema mismatch for ${className}.${fieldName} default value; expected ${typeToString(\n            type\n          )} but got ${typeToString(defaultValueType)}`\n        );\n      }\n    }\n\n    if (expectedType) {\n      if (!dbTypeMatchesObjectType(expectedType, type)) {\n        throw new Parse.Error(\n          Parse.Error.INCORRECT_TYPE,\n          `schema mismatch for ${className}.${fieldName}; expected ${typeToString(\n            expectedType\n          )} but got ${typeToString(type)}`\n        );\n      }\n      // If type options do not change\n      // we can safely return\n      if (isValidation || JSON.stringify(expectedType) === JSON.stringify(type)) {\n        return undefined;\n      }\n      // Field options are may be changed\n      // ensure to have an update to date schema field\n      return this._dbAdapter.updateFieldOptions(className, fieldName, type);\n    }\n\n    return this._dbAdapter\n      .addFieldIfNotExists(className, fieldName, type)\n      .catch(error => {\n        if (error.code == Parse.Error.INCORRECT_TYPE) {\n          // Make sure that we throw errors when it is appropriate to do so.\n          throw error;\n        }\n        // The update failed. This can be okay - it might have been a race\n        // condition where another client updated the schema in the same\n        // way that we wanted to. So, just reload the schema\n        return Promise.resolve();\n      })\n      .then(() => {\n        return {\n          className,\n          fieldName,\n          type,\n        };\n      });\n  }\n\n  ensureFields(fields: any) {\n    for (let i = 0; i < fields.length; i += 1) {\n      const { className, fieldName } = fields[i];\n      let { type } = fields[i];\n      const expectedType = this.getExpectedType(className, fieldName);\n      if (typeof type === 'string') {\n        type = { type: type };\n      }\n      if (!expectedType || !dbTypeMatchesObjectType(expectedType, type)) {\n        throw new Parse.Error(Parse.Error.INVALID_JSON, `Could not add field ${fieldName}`);\n      }\n    }\n  }\n\n  // maintain compatibility\n  deleteField(fieldName: string, className: string, database: DatabaseController) {\n    return this.deleteFields([fieldName], className, database);\n  }\n\n  // Delete fields, and remove that data from all objects. This is intended\n  // to remove unused fields, if other writers are writing objects that include\n  // this field, the field may reappear. Returns a Promise that resolves with\n  // no object on success, or rejects with { code, error } on failure.\n  // Passing the database and prefix is necessary in order to drop relation collections\n  // and remove fields from objects. Ideally the database would belong to\n  // a database adapter and this function would close over it or access it via member.\n  deleteFields(fieldNames: Array<string>, className: string, database: DatabaseController) {\n    if (!classNameIsValid(className)) {\n      throw new Parse.Error(Parse.Error.INVALID_CLASS_NAME, invalidClassNameMessage(className));\n    }\n\n    fieldNames.forEach(fieldName => {\n      if (!fieldNameIsValid(fieldName, className)) {\n        throw new Parse.Error(Parse.Error.INVALID_KEY_NAME, `invalid field name: ${fieldName}`);\n      }\n      //Don't allow deleting the default fields.\n      if (!fieldNameIsValidForClass(fieldName, className)) {\n        throw new Parse.Error(136, `field ${fieldName} cannot be changed`);\n      }\n    });\n\n    return this.getOneSchema(className, false, { clearCache: true })\n      .catch(error => {\n        if (error === undefined) {\n          throw new Parse.Error(\n            Parse.Error.INVALID_CLASS_NAME,\n            `Class ${className} does not exist.`\n          );\n        } else {\n          throw error;\n        }\n      })\n      .then(schema => {\n        fieldNames.forEach(fieldName => {\n          if (!schema.fields[fieldName]) {\n            throw new Parse.Error(255, `Field ${fieldName} does not exist, cannot delete.`);\n          }\n        });\n\n        const schemaFields = { ...schema.fields };\n        return database.adapter.deleteFields(className, schema, fieldNames).then(() => {\n          return Promise.all(\n            fieldNames.map(fieldName => {\n              const field = schemaFields[fieldName];\n              if (field && field.type === 'Relation') {\n                //For relations, drop the _Join table\n                return database.adapter.deleteClass(`_Join:${fieldName}:${className}`);\n              }\n              return Promise.resolve();\n            })\n          );\n        });\n      })\n      .then(() => {\n        SchemaCache.clear();\n      });\n  }\n\n  // Validates an object provided in REST format.\n  // Returns a promise that resolves to the new schema if this object is\n  // valid.\n  async validateObject(className: string, object: any, query: any) {\n    let geocount = 0;\n    const schema = await this.enforceClassExists(className);\n    const promises = [];\n\n    for (const fieldName in object) {\n      if (object[fieldName] && getType(object[fieldName]) === 'GeoPoint') {\n        geocount++;\n      }\n      if (geocount > 1) {\n        return Promise.reject(\n          new Parse.Error(\n            Parse.Error.INCORRECT_TYPE,\n            'there can only be one geopoint field in a class'\n          )\n        );\n      }\n    }\n    for (const fieldName in object) {\n      if (object[fieldName] === undefined) {\n        continue;\n      }\n      const expected = getType(object[fieldName]);\n      if (!expected) {\n        continue;\n      }\n      if (fieldName === 'ACL') {\n        // Every object has ACL implicitly.\n        continue;\n      }\n      promises.push(schema.enforceFieldExists(className, fieldName, expected, true));\n    }\n    const results = await Promise.all(promises);\n    const enforceFields = results.filter(result => !!result);\n\n    if (enforceFields.length !== 0) {\n      // TODO: Remove by updating schema cache directly\n      await this.reloadData({ clearCache: true });\n    }\n    this.ensureFields(enforceFields);\n\n    const promise = Promise.resolve(schema);\n    return thenValidateRequiredColumns(promise, className, object, query);\n  }\n\n  // Validates that all the properties are set for the object\n  validateRequiredColumns(className: string, object: any, query: any) {\n    const columns = requiredColumns.write[className];\n    if (!columns || columns.length == 0) {\n      return Promise.resolve(this);\n    }\n\n    const missingColumns = columns.filter(function (column) {\n      if (query && query.objectId) {\n        if (object[column] && typeof object[column] === 'object') {\n          // Trying to delete a required column\n          return object[column].__op == 'Delete';\n        }\n        // Not trying to do anything there\n        return false;\n      }\n      return !object[column];\n    });\n\n    if (missingColumns.length > 0) {\n      throw new Parse.Error(Parse.Error.INCORRECT_TYPE, missingColumns[0] + ' is required.');\n    }\n    return Promise.resolve(this);\n  }\n\n  testPermissionsForClassName(className: string, aclGroup: string[], operation: string) {\n    return SchemaController.testPermissions(\n      this.getClassLevelPermissions(className),\n      aclGroup,\n      operation\n    );\n  }\n\n  // Tests that the class level permission let pass the operation for a given aclGroup\n  static testPermissions(classPermissions: ?any, aclGroup: string[], operation: string): boolean {\n    if (!classPermissions || !classPermissions[operation]) {\n      return true;\n    }\n    const perms = classPermissions[operation];\n    if (perms['*']) {\n      return true;\n    }\n    // Check permissions against the aclGroup provided (array of userId/roles)\n    if (\n      aclGroup.some(acl => {\n        return perms[acl] === true;\n      })\n    ) {\n      return true;\n    }\n    return false;\n  }\n\n  // Validates an operation passes class-level-permissions set in the schema\n  static validatePermission(\n    classPermissions: ?any,\n    className: string,\n    aclGroup: string[],\n    operation: string,\n    action?: string\n  ) {\n    if (SchemaController.testPermissions(classPermissions, aclGroup, operation)) {\n      return Promise.resolve();\n    }\n\n    if (!classPermissions || !classPermissions[operation]) {\n      return true;\n    }\n    const perms = classPermissions[operation];\n    // If only for authenticated users\n    // make sure we have an aclGroup\n    if (perms['requiresAuthentication']) {\n      // If aclGroup has * (public)\n      if (!aclGroup || aclGroup.length == 0) {\n        throw new Parse.Error(\n          Parse.Error.OBJECT_NOT_FOUND,\n          'Permission denied, user needs to be authenticated.'\n        );\n      } else if (aclGroup.indexOf('*') > -1 && aclGroup.length == 1) {\n        throw new Parse.Error(\n          Parse.Error.OBJECT_NOT_FOUND,\n          'Permission denied, user needs to be authenticated.'\n        );\n      }\n      // requiresAuthentication passed, just move forward\n      // probably would be wise at some point to rename to 'authenticatedUser'\n      return Promise.resolve();\n    }\n\n    // No matching CLP, let's check the Pointer permissions\n    // And handle those later\n    const permissionField =\n      ['get', 'find', 'count'].indexOf(operation) > -1 ? 'readUserFields' : 'writeUserFields';\n\n    // Reject create when write lockdown\n    if (permissionField == 'writeUserFields' && operation == 'create') {\n      throw new Parse.Error(\n        Parse.Error.OPERATION_FORBIDDEN,\n        `Permission denied for action ${operation} on class ${className}.`\n      );\n    }\n\n    // Process the readUserFields later\n    if (\n      Array.isArray(classPermissions[permissionField]) &&\n      classPermissions[permissionField].length > 0\n    ) {\n      return Promise.resolve();\n    }\n\n    const pointerFields = classPermissions[operation].pointerFields;\n    if (Array.isArray(pointerFields) && pointerFields.length > 0) {\n      // any op except 'addField as part of create' is ok.\n      if (operation !== 'addField' || action === 'update') {\n        // We can allow adding field on update flow only.\n        return Promise.resolve();\n      }\n    }\n\n    throw new Parse.Error(\n      Parse.Error.OPERATION_FORBIDDEN,\n      `Permission denied for action ${operation} on class ${className}.`\n    );\n  }\n\n  // Validates an operation passes class-level-permissions set in the schema\n  validatePermission(className: string, aclGroup: string[], operation: string, action?: string) {\n    return SchemaController.validatePermission(\n      this.getClassLevelPermissions(className),\n      className,\n      aclGroup,\n      operation,\n      action\n    );\n  }\n\n  getClassLevelPermissions(className: string): any {\n    return this.schemaData[className] && this.schemaData[className].classLevelPermissions;\n  }\n\n  // Returns the expected type for a className+key combination\n  // or undefined if the schema is not set\n  getExpectedType(className: string, fieldName: string): ?(SchemaField | string) {\n    if (this.schemaData[className]) {\n      const expectedType = this.schemaData[className].fields[fieldName];\n      return expectedType === 'map' ? 'Object' : expectedType;\n    }\n    return undefined;\n  }\n\n  // Checks if a given class is in the schema.\n  hasClass(className: string) {\n    if (this.schemaData[className]) {\n      return Promise.resolve(true);\n    }\n    return this.reloadData().then(() => !!this.schemaData[className]);\n  }\n}\n\n// Returns a promise for a new Schema.\nconst load = (dbAdapter: StorageAdapter, options: any): Promise<SchemaController> => {\n  const schema = new SchemaController(dbAdapter);\n  return schema.reloadData(options).then(() => schema);\n};\n\n// Builds a new schema (in schema API response format) out of an\n// existing mongo schema + a schemas API put request. This response\n// does not include the default fields, as it is intended to be passed\n// to mongoSchemaFromFieldsAndClassName. No validation is done here, it\n// is done in mongoSchemaFromFieldsAndClassName.\nfunction buildMergedSchemaObject(existingFields: SchemaFields, putRequest: any): SchemaFields {\n  const newSchema = {};\n  // @flow-disable-next\n  const sysSchemaField =\n    Object.keys(defaultColumns).indexOf(existingFields._id) === -1\n      ? []\n      : Object.keys(defaultColumns[existingFields._id]);\n  for (const oldField in existingFields) {\n    if (\n      oldField !== '_id' &&\n      oldField !== 'ACL' &&\n      oldField !== 'updatedAt' &&\n      oldField !== 'createdAt' &&\n      oldField !== 'objectId'\n    ) {\n      if (sysSchemaField.length > 0 && sysSchemaField.indexOf(oldField) !== -1) {\n        continue;\n      }\n      const fieldIsDeleted = putRequest[oldField] && putRequest[oldField].__op === 'Delete';\n      if (!fieldIsDeleted) {\n        newSchema[oldField] = existingFields[oldField];\n      }\n    }\n  }\n  for (const newField in putRequest) {\n    if (newField !== 'objectId' && putRequest[newField].__op !== 'Delete') {\n      if (sysSchemaField.length > 0 && sysSchemaField.indexOf(newField) !== -1) {\n        continue;\n      }\n      newSchema[newField] = putRequest[newField];\n    }\n  }\n  return newSchema;\n}\n\n// Given a schema promise, construct another schema promise that\n// validates this field once the schema loads.\nfunction thenValidateRequiredColumns(schemaPromise, className, object, query) {\n  return schemaPromise.then(schema => {\n    return schema.validateRequiredColumns(className, object, query);\n  });\n}\n\n// Gets the type from a REST API formatted object, where 'type' is\n// extended past javascript types to include the rest of the Parse\n// type system.\n// The output should be a valid schema value.\n// TODO: ensure that this is compatible with the format used in Open DB\nfunction getType(obj: any): ?(SchemaField | string) {\n  const type = typeof obj;\n  switch (type) {\n    case 'boolean':\n      return 'Boolean';\n    case 'string':\n      return 'String';\n    case 'number':\n      return 'Number';\n    case 'map':\n    case 'object':\n      if (!obj) {\n        return undefined;\n      }\n      return getObjectType(obj);\n    case 'function':\n    case 'symbol':\n    case 'undefined':\n    default:\n      throw 'bad obj: ' + obj;\n  }\n}\n\n// This gets the type for non-JSON types like pointers and files, but\n// also gets the appropriate type for $ operators.\n// Returns null if the type is unknown.\nfunction getObjectType(obj): ?(SchemaField | string) {\n  if (obj instanceof Array) {\n    return 'Array';\n  }\n  if (obj.__type) {\n    switch (obj.__type) {\n      case 'Pointer':\n        if (obj.className) {\n          return {\n            type: 'Pointer',\n            targetClass: obj.className,\n          };\n        }\n        break;\n      case 'Relation':\n        if (obj.className) {\n          return {\n            type: 'Relation',\n            targetClass: obj.className,\n          };\n        }\n        break;\n      case 'File':\n        if (obj.name) {\n          return 'File';\n        }\n        break;\n      case 'Date':\n        if (obj.iso) {\n          return 'Date';\n        }\n        break;\n      case 'GeoPoint':\n        if (obj.latitude != null && obj.longitude != null) {\n          return 'GeoPoint';\n        }\n        break;\n      case 'Bytes':\n        if (obj.base64) {\n          return 'Bytes';\n        }\n        break;\n      case 'Polygon':\n        if (obj.coordinates) {\n          return 'Polygon';\n        }\n        break;\n    }\n    throw new Parse.Error(Parse.Error.INCORRECT_TYPE, 'This is not a valid ' + obj.__type);\n  }\n  if (obj['$ne']) {\n    return getObjectType(obj['$ne']);\n  }\n  if (obj.__op) {\n    switch (obj.__op) {\n      case 'Increment':\n        return 'Number';\n      case 'Delete':\n        return null;\n      case 'Add':\n      case 'AddUnique':\n      case 'Remove':\n        return 'Array';\n      case 'AddRelation':\n      case 'RemoveRelation':\n        return {\n          type: 'Relation',\n          targetClass: obj.objects[0].className,\n        };\n      case 'Batch':\n        return getObjectType(obj.ops[0]);\n      default:\n        throw 'unexpected op: ' + obj.__op;\n    }\n  }\n  return 'Object';\n}\n\nexport {\n  load,\n  classNameIsValid,\n  fieldNameIsValid,\n  invalidClassNameMessage,\n  buildMergedSchemaObject,\n  systemClasses,\n  defaultColumns,\n  convertSchemaToAdapterSchema,\n  VolatileClassesSchemas,\n  SchemaController,\n  requiredColumns,\n};\n"],"mappings":"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"}
|