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
|
@@ -4,52 +4,38 @@ Object.defineProperty(exports, "__esModule", {
|
|
|
4
4
|
value: true
|
|
5
5
|
});
|
|
6
6
|
exports.DefinedSchemas = void 0;
|
|
7
|
-
|
|
8
7
|
var _logger = require("../logger");
|
|
9
|
-
|
|
10
8
|
var _Config = _interopRequireDefault(require("../Config"));
|
|
11
|
-
|
|
12
9
|
var _SchemasRouter = require("../Routers/SchemasRouter");
|
|
13
|
-
|
|
14
10
|
var _SchemaController = require("../Controllers/SchemaController");
|
|
15
|
-
|
|
16
11
|
var _Options = require("../Options");
|
|
17
|
-
|
|
18
12
|
var Migrations = _interopRequireWildcard(require("./Migrations"));
|
|
19
|
-
|
|
13
|
+
var _Auth = _interopRequireDefault(require("../Auth"));
|
|
14
|
+
var _rest = _interopRequireDefault(require("../rest"));
|
|
20
15
|
function _getRequireWildcardCache(nodeInterop) { if (typeof WeakMap !== "function") return null; var cacheBabelInterop = new WeakMap(); var cacheNodeInterop = new WeakMap(); return (_getRequireWildcardCache = function (nodeInterop) { return nodeInterop ? cacheNodeInterop : cacheBabelInterop; })(nodeInterop); }
|
|
21
|
-
|
|
22
16
|
function _interopRequireWildcard(obj, nodeInterop) { if (!nodeInterop && obj && obj.__esModule) { return obj; } if (obj === null || typeof obj !== "object" && typeof obj !== "function") { return { default: obj }; } var cache = _getRequireWildcardCache(nodeInterop); if (cache && cache.has(obj)) { return cache.get(obj); } var newObj = {}; var hasPropertyDescriptor = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var key in obj) { if (key !== "default" && Object.prototype.hasOwnProperty.call(obj, key)) { var desc = hasPropertyDescriptor ? Object.getOwnPropertyDescriptor(obj, key) : null; if (desc && (desc.get || desc.set)) { Object.defineProperty(newObj, key, desc); } else { newObj[key] = obj[key]; } } } newObj.default = obj; if (cache) { cache.set(obj, newObj); } return newObj; }
|
|
23
|
-
|
|
24
17
|
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
25
|
-
|
|
26
|
-
function
|
|
27
|
-
|
|
28
|
-
function
|
|
29
|
-
|
|
30
|
-
function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
|
|
31
|
-
|
|
18
|
+
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; }
|
|
19
|
+
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; }
|
|
20
|
+
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; }
|
|
21
|
+
function _toPropertyKey(arg) { var key = _toPrimitive(arg, "string"); return typeof key === "symbol" ? key : String(key); }
|
|
22
|
+
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); }
|
|
32
23
|
// -disable-next Cannot resolve module `parse/node`.
|
|
33
24
|
const Parse = require('parse/node');
|
|
34
|
-
|
|
35
25
|
class DefinedSchemas {
|
|
36
26
|
constructor(schemaOptions, config) {
|
|
37
27
|
this.localSchemas = [];
|
|
38
28
|
this.config = _Config.default.get(config.appId);
|
|
39
29
|
this.schemaOptions = schemaOptions;
|
|
40
|
-
|
|
41
30
|
if (schemaOptions && schemaOptions.definitions) {
|
|
42
31
|
if (!Array.isArray(schemaOptions.definitions)) {
|
|
43
32
|
throw `"schema.definitions" must be an array of schemas`;
|
|
44
33
|
}
|
|
45
|
-
|
|
46
34
|
this.localSchemas = schemaOptions.definitions;
|
|
47
35
|
}
|
|
48
|
-
|
|
49
36
|
this.retries = 0;
|
|
50
37
|
this.maxRetries = 3;
|
|
51
38
|
}
|
|
52
|
-
|
|
53
39
|
async saveSchemaToDB(schema) {
|
|
54
40
|
const payload = {
|
|
55
41
|
className: schema.className,
|
|
@@ -60,15 +46,14 @@ class DefinedSchemas {
|
|
|
60
46
|
await (0, _SchemasRouter.internalCreateSchema)(schema.className, payload, this.config);
|
|
61
47
|
this.resetSchemaOps(schema);
|
|
62
48
|
}
|
|
63
|
-
|
|
64
49
|
resetSchemaOps(schema) {
|
|
65
50
|
// Reset ops like SDK
|
|
66
51
|
schema._fields = {};
|
|
67
52
|
schema._indexes = {};
|
|
68
|
-
}
|
|
69
|
-
// We cannot use SDK since routes are disabled
|
|
70
|
-
|
|
53
|
+
}
|
|
71
54
|
|
|
55
|
+
// Simulate update like the SDK
|
|
56
|
+
// We cannot use SDK since routes are disabled
|
|
72
57
|
async updateSchemaToDB(schema) {
|
|
73
58
|
const payload = {
|
|
74
59
|
className: schema.className,
|
|
@@ -79,32 +64,24 @@ class DefinedSchemas {
|
|
|
79
64
|
await (0, _SchemasRouter.internalUpdateSchema)(schema.className, payload, this.config);
|
|
80
65
|
this.resetSchemaOps(schema);
|
|
81
66
|
}
|
|
82
|
-
|
|
83
67
|
async execute() {
|
|
84
68
|
try {
|
|
85
69
|
_logger.logger.info('Running Migrations');
|
|
86
|
-
|
|
87
70
|
if (this.schemaOptions && this.schemaOptions.beforeMigration) {
|
|
88
71
|
await Promise.resolve(this.schemaOptions.beforeMigration());
|
|
89
72
|
}
|
|
90
|
-
|
|
91
73
|
await this.executeMigrations();
|
|
92
|
-
|
|
93
74
|
if (this.schemaOptions && this.schemaOptions.afterMigration) {
|
|
94
75
|
await Promise.resolve(this.schemaOptions.afterMigration());
|
|
95
76
|
}
|
|
96
|
-
|
|
97
77
|
_logger.logger.info('Running Migrations Completed');
|
|
98
78
|
} catch (e) {
|
|
99
79
|
_logger.logger.error(`Failed to run migrations: ${e}`);
|
|
100
|
-
|
|
101
80
|
if (process.env.NODE_ENV === 'production') process.exit(1);
|
|
102
81
|
}
|
|
103
82
|
}
|
|
104
|
-
|
|
105
83
|
async executeMigrations() {
|
|
106
84
|
let timeout = null;
|
|
107
|
-
|
|
108
85
|
try {
|
|
109
86
|
// Set up a time out in production
|
|
110
87
|
// if we fail to get schema
|
|
@@ -113,61 +90,52 @@ class DefinedSchemas {
|
|
|
113
90
|
if (process.env.NODE_ENV === 'production') {
|
|
114
91
|
timeout = setTimeout(() => {
|
|
115
92
|
_logger.logger.error('Timeout occurred during execution of migrations. Exiting...');
|
|
116
|
-
|
|
117
93
|
process.exit(1);
|
|
118
94
|
}, 20000);
|
|
119
|
-
}
|
|
120
|
-
|
|
121
|
-
|
|
95
|
+
}
|
|
122
96
|
await this.createDeleteSession();
|
|
123
|
-
|
|
97
|
+
// -disable-next-line
|
|
98
|
+
const schemaController = await this.config.database.loadSchema();
|
|
99
|
+
this.allCloudSchemas = await schemaController.getAllClasses();
|
|
124
100
|
clearTimeout(timeout);
|
|
125
101
|
await Promise.all(this.localSchemas.map(async localSchema => this.saveOrUpdate(localSchema)));
|
|
126
102
|
this.checkForMissingSchemas();
|
|
127
103
|
await this.enforceCLPForNonProvidedClass();
|
|
128
104
|
} catch (e) {
|
|
129
105
|
if (timeout) clearTimeout(timeout);
|
|
130
|
-
|
|
131
106
|
if (this.retries < this.maxRetries) {
|
|
132
|
-
this.retries++;
|
|
107
|
+
this.retries++;
|
|
108
|
+
// first retry 1sec, 2sec, 3sec total 6sec retry sequence
|
|
133
109
|
// retry will only happen in case of deploying multi parse server instance
|
|
134
110
|
// at the same time. Modern systems like k8 avoid this by doing rolling updates
|
|
135
|
-
|
|
136
111
|
await this.wait(1000 * this.retries);
|
|
137
112
|
await this.executeMigrations();
|
|
138
113
|
} else {
|
|
139
114
|
_logger.logger.error(`Failed to run migrations: ${e}`);
|
|
140
|
-
|
|
141
115
|
if (process.env.NODE_ENV === 'production') process.exit(1);
|
|
142
116
|
}
|
|
143
117
|
}
|
|
144
118
|
}
|
|
145
|
-
|
|
146
119
|
checkForMissingSchemas() {
|
|
147
120
|
if (this.schemaOptions.strict !== true) {
|
|
148
121
|
return;
|
|
149
122
|
}
|
|
150
|
-
|
|
151
123
|
const cloudSchemas = this.allCloudSchemas.map(s => s.className);
|
|
152
124
|
const localSchemas = this.localSchemas.map(s => s.className);
|
|
153
125
|
const missingSchemas = cloudSchemas.filter(c => !localSchemas.includes(c) && !_SchemaController.systemClasses.includes(c));
|
|
154
|
-
|
|
155
126
|
if (new Set(localSchemas).size !== localSchemas.length) {
|
|
156
127
|
_logger.logger.error(`The list of schemas provided contains duplicated "className" "${localSchemas.join('","')}"`);
|
|
157
|
-
|
|
158
128
|
process.exit(1);
|
|
159
129
|
}
|
|
160
|
-
|
|
161
130
|
if (this.schemaOptions.strict && missingSchemas.length) {
|
|
162
131
|
_logger.logger.warn(`The following schemas are currently present in the database, but not explicitly defined in a schema: "${missingSchemas.join('", "')}"`);
|
|
163
132
|
}
|
|
164
|
-
}
|
|
165
|
-
|
|
133
|
+
}
|
|
166
134
|
|
|
135
|
+
// Required for testing purpose
|
|
167
136
|
wait(time) {
|
|
168
137
|
return new Promise(resolve => setTimeout(resolve, time));
|
|
169
138
|
}
|
|
170
|
-
|
|
171
139
|
async enforceCLPForNonProvidedClass() {
|
|
172
140
|
const nonProvidedClasses = this.allCloudSchemas.filter(cloudSchema => !this.localSchemas.some(localSchema => localSchema.className === cloudSchema.className));
|
|
173
141
|
await Promise.all(nonProvidedClasses.map(async schema => {
|
|
@@ -175,23 +143,18 @@ class DefinedSchemas {
|
|
|
175
143
|
this.handleCLP(schema, parseSchema);
|
|
176
144
|
await this.updateSchemaToDB(parseSchema);
|
|
177
145
|
}));
|
|
178
|
-
}
|
|
179
|
-
// a session is created
|
|
180
|
-
|
|
146
|
+
}
|
|
181
147
|
|
|
148
|
+
// Create a fake session since Parse do not create the _Session until
|
|
149
|
+
// a session is created
|
|
182
150
|
async createDeleteSession() {
|
|
183
|
-
const
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
await session.destroy({
|
|
188
|
-
useMasterKey: true
|
|
189
|
-
});
|
|
151
|
+
const {
|
|
152
|
+
response
|
|
153
|
+
} = await _rest.default.create(this.config, _Auth.default.master(this.config), '_Session', {});
|
|
154
|
+
await _rest.default.del(this.config, _Auth.default.master(this.config), '_Session', response.objectId);
|
|
190
155
|
}
|
|
191
|
-
|
|
192
156
|
async saveOrUpdate(localSchema) {
|
|
193
157
|
const cloudSchema = this.allCloudSchemas.find(sc => sc.className === localSchema.className);
|
|
194
|
-
|
|
195
158
|
if (cloudSchema) {
|
|
196
159
|
try {
|
|
197
160
|
await this.updateSchema(localSchema, cloudSchema);
|
|
@@ -206,10 +169,8 @@ class DefinedSchemas {
|
|
|
206
169
|
}
|
|
207
170
|
}
|
|
208
171
|
}
|
|
209
|
-
|
|
210
172
|
async saveSchema(localSchema) {
|
|
211
173
|
const newLocalSchema = new Parse.Schema(localSchema.className);
|
|
212
|
-
|
|
213
174
|
if (localSchema.fields) {
|
|
214
175
|
// Handle fields
|
|
215
176
|
Object.keys(localSchema.fields).filter(fieldName => !this.isProtectedFields(localSchema.className, fieldName)).forEach(fieldName => {
|
|
@@ -218,9 +179,8 @@ class DefinedSchemas {
|
|
|
218
179
|
this.handleFields(newLocalSchema, fieldName, field);
|
|
219
180
|
}
|
|
220
181
|
});
|
|
221
|
-
}
|
|
222
|
-
|
|
223
|
-
|
|
182
|
+
}
|
|
183
|
+
// Handle indexes
|
|
224
184
|
if (localSchema.indexes) {
|
|
225
185
|
Object.keys(localSchema.indexes).forEach(indexName => {
|
|
226
186
|
if (localSchema.indexes && !this.isProtectedIndex(localSchema.className, indexName)) {
|
|
@@ -228,40 +188,36 @@ class DefinedSchemas {
|
|
|
228
188
|
}
|
|
229
189
|
});
|
|
230
190
|
}
|
|
231
|
-
|
|
232
191
|
this.handleCLP(localSchema, newLocalSchema);
|
|
233
192
|
return await this.saveSchemaToDB(newLocalSchema);
|
|
234
193
|
}
|
|
235
|
-
|
|
236
194
|
async updateSchema(localSchema, cloudSchema) {
|
|
237
|
-
const newLocalSchema = new Parse.Schema(localSchema.className);
|
|
238
|
-
// Check addition
|
|
195
|
+
const newLocalSchema = new Parse.Schema(localSchema.className);
|
|
239
196
|
|
|
197
|
+
// Handle fields
|
|
198
|
+
// Check addition
|
|
240
199
|
if (localSchema.fields) {
|
|
241
200
|
Object.keys(localSchema.fields).filter(fieldName => !this.isProtectedFields(localSchema.className, fieldName)).forEach(fieldName => {
|
|
242
201
|
// -disable-next
|
|
243
202
|
const field = localSchema.fields[fieldName];
|
|
244
|
-
|
|
245
203
|
if (!cloudSchema.fields[fieldName]) {
|
|
246
204
|
this.handleFields(newLocalSchema, fieldName, field);
|
|
247
205
|
}
|
|
248
206
|
});
|
|
249
207
|
}
|
|
250
|
-
|
|
251
208
|
const fieldsToDelete = [];
|
|
252
209
|
const fieldsToRecreate = [];
|
|
253
|
-
const fieldsWithChangedParams = [];
|
|
210
|
+
const fieldsWithChangedParams = [];
|
|
254
211
|
|
|
212
|
+
// Check deletion
|
|
255
213
|
Object.keys(cloudSchema.fields).filter(fieldName => !this.isProtectedFields(localSchema.className, fieldName)).forEach(fieldName => {
|
|
256
214
|
const field = cloudSchema.fields[fieldName];
|
|
257
|
-
|
|
258
215
|
if (!localSchema.fields || !localSchema.fields[fieldName]) {
|
|
259
216
|
fieldsToDelete.push(fieldName);
|
|
260
217
|
return;
|
|
261
218
|
}
|
|
262
|
-
|
|
263
|
-
|
|
264
|
-
|
|
219
|
+
const localField = localSchema.fields[fieldName];
|
|
220
|
+
// Check if field has a changed type
|
|
265
221
|
if (!this.paramsAreEquals({
|
|
266
222
|
type: field.type,
|
|
267
223
|
targetClass: field.targetClass
|
|
@@ -281,29 +237,29 @@ class DefinedSchemas {
|
|
|
281
237
|
}
|
|
282
238
|
});
|
|
283
239
|
return;
|
|
284
|
-
}
|
|
285
|
-
|
|
240
|
+
}
|
|
286
241
|
|
|
242
|
+
// Check if something changed other than the type (like required, defaultValue)
|
|
287
243
|
if (!this.paramsAreEquals(field, localField)) {
|
|
288
244
|
fieldsWithChangedParams.push(fieldName);
|
|
289
245
|
}
|
|
290
246
|
});
|
|
291
|
-
|
|
292
247
|
if (this.schemaOptions.deleteExtraFields === true) {
|
|
293
248
|
fieldsToDelete.forEach(fieldName => {
|
|
294
249
|
newLocalSchema.deleteField(fieldName);
|
|
295
|
-
});
|
|
250
|
+
});
|
|
296
251
|
|
|
252
|
+
// Delete fields from the schema then apply changes
|
|
297
253
|
await this.updateSchemaToDB(newLocalSchema);
|
|
298
254
|
} else if (this.schemaOptions.strict === true && fieldsToDelete.length) {
|
|
299
255
|
_logger.logger.warn(`The following fields exist in the database for "${localSchema.className}", but are missing in the schema : "${fieldsToDelete.join('" ,"')}"`);
|
|
300
256
|
}
|
|
301
|
-
|
|
302
257
|
if (this.schemaOptions.recreateModifiedFields === true) {
|
|
303
258
|
fieldsToRecreate.forEach(field => {
|
|
304
259
|
newLocalSchema.deleteField(field.fieldName);
|
|
305
|
-
});
|
|
260
|
+
});
|
|
306
261
|
|
|
262
|
+
// Delete fields from the schema then apply changes
|
|
307
263
|
await this.updateSchemaToDB(newLocalSchema);
|
|
308
264
|
fieldsToRecreate.forEach(fieldInfo => {
|
|
309
265
|
if (localSchema.fields) {
|
|
@@ -315,19 +271,18 @@ class DefinedSchemas {
|
|
|
315
271
|
fieldsToRecreate.forEach(field => {
|
|
316
272
|
const from = field.from.type + (field.from.targetClass ? ` (${field.from.targetClass})` : '');
|
|
317
273
|
const to = field.to.type + (field.to.targetClass ? ` (${field.to.targetClass})` : '');
|
|
318
|
-
|
|
319
274
|
_logger.logger.warn(`The field "${field.fieldName}" type differ between the schema and the database for "${localSchema.className}"; Schema is defined as "${to}" and current database type is "${from}"`);
|
|
320
275
|
});
|
|
321
276
|
}
|
|
322
|
-
|
|
323
277
|
fieldsWithChangedParams.forEach(fieldName => {
|
|
324
278
|
if (localSchema.fields) {
|
|
325
279
|
const field = localSchema.fields[fieldName];
|
|
326
280
|
this.handleFields(newLocalSchema, fieldName, field);
|
|
327
281
|
}
|
|
328
|
-
});
|
|
329
|
-
// Check addition
|
|
282
|
+
});
|
|
330
283
|
|
|
284
|
+
// Handle Indexes
|
|
285
|
+
// Check addition
|
|
331
286
|
if (localSchema.indexes) {
|
|
332
287
|
Object.keys(localSchema.indexes).forEach(indexName => {
|
|
333
288
|
if ((!cloudSchema.indexes || !cloudSchema.indexes[indexName]) && !this.isProtectedIndex(localSchema.className, indexName)) {
|
|
@@ -337,9 +292,9 @@ class DefinedSchemas {
|
|
|
337
292
|
}
|
|
338
293
|
});
|
|
339
294
|
}
|
|
295
|
+
const indexesToAdd = [];
|
|
340
296
|
|
|
341
|
-
|
|
342
|
-
|
|
297
|
+
// Check deletion
|
|
343
298
|
if (cloudSchema.indexes) {
|
|
344
299
|
Object.keys(cloudSchema.indexes).forEach(indexName => {
|
|
345
300
|
if (!this.isProtectedIndex(localSchema.className, indexName)) {
|
|
@@ -347,7 +302,6 @@ class DefinedSchemas {
|
|
|
347
302
|
newLocalSchema.deleteIndex(indexName);
|
|
348
303
|
} else if (!this.paramsAreEquals(localSchema.indexes[indexName], cloudSchema.indexes[indexName])) {
|
|
349
304
|
newLocalSchema.deleteIndex(indexName);
|
|
350
|
-
|
|
351
305
|
if (localSchema.indexes) {
|
|
352
306
|
indexesToAdd.push({
|
|
353
307
|
indexName,
|
|
@@ -358,63 +312,52 @@ class DefinedSchemas {
|
|
|
358
312
|
}
|
|
359
313
|
});
|
|
360
314
|
}
|
|
361
|
-
|
|
362
|
-
|
|
363
|
-
|
|
364
|
-
|
|
365
|
-
|
|
315
|
+
this.handleCLP(localSchema, newLocalSchema, cloudSchema);
|
|
316
|
+
// Apply changes
|
|
317
|
+
await this.updateSchemaToDB(newLocalSchema);
|
|
318
|
+
// Apply new/changed indexes
|
|
366
319
|
if (indexesToAdd.length) {
|
|
367
320
|
_logger.logger.debug(`Updating indexes for "${newLocalSchema.className}" : ${indexesToAdd.join(' ,')}`);
|
|
368
|
-
|
|
369
321
|
indexesToAdd.forEach(o => newLocalSchema.addIndex(o.indexName, o.index));
|
|
370
322
|
await this.updateSchemaToDB(newLocalSchema);
|
|
371
323
|
}
|
|
372
324
|
}
|
|
373
|
-
|
|
374
325
|
handleCLP(localSchema, newLocalSchema, cloudSchema) {
|
|
375
326
|
if (!localSchema.classLevelPermissions && !cloudSchema) {
|
|
376
327
|
_logger.logger.warn(`classLevelPermissions not provided for ${localSchema.className}.`);
|
|
377
|
-
}
|
|
378
|
-
|
|
379
|
-
|
|
380
|
-
|
|
381
|
-
|
|
328
|
+
}
|
|
329
|
+
// Use spread to avoid read only issue (encountered by Moumouls using directAccess)
|
|
330
|
+
const clp = _objectSpread({}, localSchema.classLevelPermissions) || {};
|
|
331
|
+
// To avoid inconsistency we need to remove all rights on addField
|
|
382
332
|
clp.addField = {};
|
|
383
333
|
newLocalSchema.setCLP(clp);
|
|
384
334
|
}
|
|
385
|
-
|
|
386
335
|
isProtectedFields(className, fieldName) {
|
|
387
336
|
return !!_SchemaController.defaultColumns._Default[fieldName] || !!(_SchemaController.defaultColumns[className] && _SchemaController.defaultColumns[className][fieldName]);
|
|
388
337
|
}
|
|
389
|
-
|
|
390
338
|
isProtectedIndex(className, indexName) {
|
|
391
339
|
const indexes = ['_id_'];
|
|
392
|
-
|
|
393
340
|
switch (className) {
|
|
394
341
|
case '_User':
|
|
395
342
|
indexes.push('case_insensitive_username', 'case_insensitive_email', 'username_1', 'email_1');
|
|
396
343
|
break;
|
|
397
|
-
|
|
398
344
|
case '_Role':
|
|
399
345
|
indexes.push('name_1');
|
|
400
346
|
break;
|
|
401
|
-
|
|
402
347
|
case '_Idempotency':
|
|
403
348
|
indexes.push('reqId_1');
|
|
404
349
|
break;
|
|
405
350
|
}
|
|
406
|
-
|
|
407
351
|
return indexes.indexOf(indexName) !== -1;
|
|
408
352
|
}
|
|
409
|
-
|
|
410
353
|
paramsAreEquals(objA, objB) {
|
|
411
354
|
const keysA = Object.keys(objA);
|
|
412
|
-
const keysB = Object.keys(objB);
|
|
355
|
+
const keysB = Object.keys(objB);
|
|
413
356
|
|
|
357
|
+
// Check key name
|
|
414
358
|
if (keysA.length !== keysB.length) return false;
|
|
415
359
|
return keysA.every(k => objA[k] === objB[k]);
|
|
416
360
|
}
|
|
417
|
-
|
|
418
361
|
handleFields(newLocalSchema, fieldName, field) {
|
|
419
362
|
if (field.type === 'Relation') {
|
|
420
363
|
newLocalSchema.addRelation(fieldName, field.targetClass);
|
|
@@ -424,8 +367,6 @@ class DefinedSchemas {
|
|
|
424
367
|
newLocalSchema.addField(fieldName, field.type, field);
|
|
425
368
|
}
|
|
426
369
|
}
|
|
427
|
-
|
|
428
370
|
}
|
|
429
|
-
|
|
430
371
|
exports.DefinedSchemas = DefinedSchemas;
|
|
431
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/SchemaMigrations/DefinedSchemas.js"],"names":["Parse","require","DefinedSchemas","constructor","schemaOptions","config","localSchemas","Config","get","appId","definitions","Array","isArray","retries","maxRetries","saveSchemaToDB","schema","payload","className","fields","_fields","indexes","_indexes","classLevelPermissions","_clp","resetSchemaOps","updateSchemaToDB","execute","logger","info","beforeMigration","Promise","resolve","executeMigrations","afterMigration","e","error","process","env","NODE_ENV","exit","timeout","setTimeout","createDeleteSession","allCloudSchemas","Schema","all","clearTimeout","map","localSchema","saveOrUpdate","checkForMissingSchemas","enforceCLPForNonProvidedClass","wait","strict","cloudSchemas","s","missingSchemas","filter","c","includes","systemClasses","Set","size","length","join","warn","time","nonProvidedClasses","cloudSchema","some","parseSchema","handleCLP","session","Session","save","useMasterKey","destroy","find","sc","updateSchema","saveSchema","newLocalSchema","Object","keys","fieldName","isProtectedFields","forEach","field","handleFields","indexName","isProtectedIndex","addIndex","fieldsToDelete","fieldsToRecreate","fieldsWithChangedParams","push","localField","paramsAreEquals","type","targetClass","from","to","deleteExtraFields","deleteField","recreateModifiedFields","fieldInfo","indexesToAdd","deleteIndex","index","debug","o","clp","addField","setCLP","defaultColumns","_Default","indexOf","objA","objB","keysA","keysB","every","k","addRelation","addPointer"],"mappings":"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","sourcesContent":["// @flow\n// @flow-disable-next Cannot resolve module `parse/node`.\nconst Parse = require('parse/node');\nimport { logger } from '../logger';\nimport Config from '../Config';\nimport { internalCreateSchema, internalUpdateSchema } from '../Routers/SchemasRouter';\nimport { defaultColumns, systemClasses } from '../Controllers/SchemaController';\nimport { ParseServerOptions } from '../Options';\nimport * as Migrations from './Migrations';\n\nexport class DefinedSchemas {\n  config: ParseServerOptions;\n  schemaOptions: Migrations.SchemaOptions;\n  localSchemas: Migrations.JSONSchema[];\n  retries: number;\n  maxRetries: number;\n  allCloudSchemas: Parse.Schema[];\n\n  constructor(schemaOptions: Migrations.SchemaOptions, config: ParseServerOptions) {\n    this.localSchemas = [];\n    this.config = Config.get(config.appId);\n    this.schemaOptions = schemaOptions;\n    if (schemaOptions && schemaOptions.definitions) {\n      if (!Array.isArray(schemaOptions.definitions)) {\n        throw `\"schema.definitions\" must be an array of schemas`;\n      }\n\n      this.localSchemas = schemaOptions.definitions;\n    }\n\n    this.retries = 0;\n    this.maxRetries = 3;\n  }\n\n  async saveSchemaToDB(schema: Parse.Schema): Promise<void> {\n    const payload = {\n      className: schema.className,\n      fields: schema._fields,\n      indexes: schema._indexes,\n      classLevelPermissions: schema._clp,\n    };\n    await internalCreateSchema(schema.className, payload, this.config);\n    this.resetSchemaOps(schema);\n  }\n\n  resetSchemaOps(schema: Parse.Schema) {\n    // Reset ops like SDK\n    schema._fields = {};\n    schema._indexes = {};\n  }\n\n  // Simulate update like the SDK\n  // We cannot use SDK since routes are disabled\n  async updateSchemaToDB(schema: Parse.Schema) {\n    const payload = {\n      className: schema.className,\n      fields: schema._fields,\n      indexes: schema._indexes,\n      classLevelPermissions: schema._clp,\n    };\n    await internalUpdateSchema(schema.className, payload, this.config);\n    this.resetSchemaOps(schema);\n  }\n\n  async execute() {\n    try {\n      logger.info('Running Migrations');\n      if (this.schemaOptions && this.schemaOptions.beforeMigration) {\n        await Promise.resolve(this.schemaOptions.beforeMigration());\n      }\n\n      await this.executeMigrations();\n\n      if (this.schemaOptions && this.schemaOptions.afterMigration) {\n        await Promise.resolve(this.schemaOptions.afterMigration());\n      }\n\n      logger.info('Running Migrations Completed');\n    } catch (e) {\n      logger.error(`Failed to run migrations: ${e}`);\n      if (process.env.NODE_ENV === 'production') process.exit(1);\n    }\n  }\n\n  async executeMigrations() {\n    let timeout = null;\n    try {\n      // Set up a time out in production\n      // if we fail to get schema\n      // pm2 or K8s and many other process managers will try to restart the process\n      // after the exit\n      if (process.env.NODE_ENV === 'production') {\n        timeout = setTimeout(() => {\n          logger.error('Timeout occurred during execution of migrations. Exiting...');\n          process.exit(1);\n        }, 20000);\n      }\n\n      // Hack to force session schema to be created\n      await this.createDeleteSession();\n      this.allCloudSchemas = await Parse.Schema.all();\n      clearTimeout(timeout);\n      await Promise.all(this.localSchemas.map(async localSchema => this.saveOrUpdate(localSchema)));\n\n      this.checkForMissingSchemas();\n      await this.enforceCLPForNonProvidedClass();\n    } catch (e) {\n      if (timeout) clearTimeout(timeout);\n      if (this.retries < this.maxRetries) {\n        this.retries++;\n        // first retry 1sec, 2sec, 3sec total 6sec retry sequence\n        // retry will only happen in case of deploying multi parse server instance\n        // at the same time. Modern systems like k8 avoid this by doing rolling updates\n        await this.wait(1000 * this.retries);\n        await this.executeMigrations();\n      } else {\n        logger.error(`Failed to run migrations: ${e}`);\n        if (process.env.NODE_ENV === 'production') process.exit(1);\n      }\n    }\n  }\n\n  checkForMissingSchemas() {\n    if (this.schemaOptions.strict !== true) {\n      return;\n    }\n\n    const cloudSchemas = this.allCloudSchemas.map(s => s.className);\n    const localSchemas = this.localSchemas.map(s => s.className);\n    const missingSchemas = cloudSchemas.filter(\n      c => !localSchemas.includes(c) && !systemClasses.includes(c)\n    );\n\n    if (new Set(localSchemas).size !== localSchemas.length) {\n      logger.error(\n        `The list of schemas provided contains duplicated \"className\"  \"${localSchemas.join(\n          '\",\"'\n        )}\"`\n      );\n      process.exit(1);\n    }\n\n    if (this.schemaOptions.strict && missingSchemas.length) {\n      logger.warn(\n        `The following schemas are currently present in the database, but not explicitly defined in a schema: \"${missingSchemas.join(\n          '\", \"'\n        )}\"`\n      );\n    }\n  }\n\n  // Required for testing purpose\n  wait(time: number) {\n    return new Promise<void>(resolve => setTimeout(resolve, time));\n  }\n\n  async enforceCLPForNonProvidedClass(): Promise<void> {\n    const nonProvidedClasses = this.allCloudSchemas.filter(\n      cloudSchema =>\n        !this.localSchemas.some(localSchema => localSchema.className === cloudSchema.className)\n    );\n    await Promise.all(\n      nonProvidedClasses.map(async schema => {\n        const parseSchema = new Parse.Schema(schema.className);\n        this.handleCLP(schema, parseSchema);\n        await this.updateSchemaToDB(parseSchema);\n      })\n    );\n  }\n\n  // Create a fake session since Parse do not create the _Session until\n  // a session is created\n  async createDeleteSession() {\n    const session = new Parse.Session();\n    await session.save(null, { useMasterKey: true });\n    await session.destroy({ useMasterKey: true });\n  }\n\n  async saveOrUpdate(localSchema: Migrations.JSONSchema) {\n    const cloudSchema = this.allCloudSchemas.find(sc => sc.className === localSchema.className);\n    if (cloudSchema) {\n      try {\n        await this.updateSchema(localSchema, cloudSchema);\n      } catch (e) {\n        throw `Error during update of schema for type ${cloudSchema.className}: ${e}`;\n      }\n    } else {\n      try {\n        await this.saveSchema(localSchema);\n      } catch (e) {\n        throw `Error while saving Schema for type ${localSchema.className}: ${e}`;\n      }\n    }\n  }\n\n  async saveSchema(localSchema: Migrations.JSONSchema) {\n    const newLocalSchema = new Parse.Schema(localSchema.className);\n    if (localSchema.fields) {\n      // Handle fields\n      Object.keys(localSchema.fields)\n        .filter(fieldName => !this.isProtectedFields(localSchema.className, fieldName))\n        .forEach(fieldName => {\n          if (localSchema.fields) {\n            const field = localSchema.fields[fieldName];\n            this.handleFields(newLocalSchema, fieldName, field);\n          }\n        });\n    }\n    // Handle indexes\n    if (localSchema.indexes) {\n      Object.keys(localSchema.indexes).forEach(indexName => {\n        if (localSchema.indexes && !this.isProtectedIndex(localSchema.className, indexName)) {\n          newLocalSchema.addIndex(indexName, localSchema.indexes[indexName]);\n        }\n      });\n    }\n\n    this.handleCLP(localSchema, newLocalSchema);\n\n    return await this.saveSchemaToDB(newLocalSchema);\n  }\n\n  async updateSchema(localSchema: Migrations.JSONSchema, cloudSchema: Parse.Schema) {\n    const newLocalSchema = new Parse.Schema(localSchema.className);\n\n    // Handle fields\n    // Check addition\n    if (localSchema.fields) {\n      Object.keys(localSchema.fields)\n        .filter(fieldName => !this.isProtectedFields(localSchema.className, fieldName))\n        .forEach(fieldName => {\n          // @flow-disable-next\n          const field = localSchema.fields[fieldName];\n          if (!cloudSchema.fields[fieldName]) {\n            this.handleFields(newLocalSchema, fieldName, field);\n          }\n        });\n    }\n\n    const fieldsToDelete: string[] = [];\n    const fieldsToRecreate: {\n      fieldName: string,\n      from: { type: string, targetClass?: string },\n      to: { type: string, targetClass?: string },\n    }[] = [];\n    const fieldsWithChangedParams: string[] = [];\n\n    // Check deletion\n    Object.keys(cloudSchema.fields)\n      .filter(fieldName => !this.isProtectedFields(localSchema.className, fieldName))\n      .forEach(fieldName => {\n        const field = cloudSchema.fields[fieldName];\n        if (!localSchema.fields || !localSchema.fields[fieldName]) {\n          fieldsToDelete.push(fieldName);\n          return;\n        }\n\n        const localField = localSchema.fields[fieldName];\n        // Check if field has a changed type\n        if (\n          !this.paramsAreEquals(\n            { type: field.type, targetClass: field.targetClass },\n            { type: localField.type, targetClass: localField.targetClass }\n          )\n        ) {\n          fieldsToRecreate.push({\n            fieldName,\n            from: { type: field.type, targetClass: field.targetClass },\n            to: { type: localField.type, targetClass: localField.targetClass },\n          });\n          return;\n        }\n\n        // Check if something changed other than the type (like required, defaultValue)\n        if (!this.paramsAreEquals(field, localField)) {\n          fieldsWithChangedParams.push(fieldName);\n        }\n      });\n\n    if (this.schemaOptions.deleteExtraFields === true) {\n      fieldsToDelete.forEach(fieldName => {\n        newLocalSchema.deleteField(fieldName);\n      });\n\n      // Delete fields from the schema then apply changes\n      await this.updateSchemaToDB(newLocalSchema);\n    } else if (this.schemaOptions.strict === true && fieldsToDelete.length) {\n      logger.warn(\n        `The following fields exist in the database for \"${\n          localSchema.className\n        }\", but are missing in the schema : \"${fieldsToDelete.join('\" ,\"')}\"`\n      );\n    }\n\n    if (this.schemaOptions.recreateModifiedFields === true) {\n      fieldsToRecreate.forEach(field => {\n        newLocalSchema.deleteField(field.fieldName);\n      });\n\n      // Delete fields from the schema then apply changes\n      await this.updateSchemaToDB(newLocalSchema);\n\n      fieldsToRecreate.forEach(fieldInfo => {\n        if (localSchema.fields) {\n          const field = localSchema.fields[fieldInfo.fieldName];\n          this.handleFields(newLocalSchema, fieldInfo.fieldName, field);\n        }\n      });\n    } else if (this.schemaOptions.strict === true && fieldsToRecreate.length) {\n      fieldsToRecreate.forEach(field => {\n        const from =\n          field.from.type + (field.from.targetClass ? ` (${field.from.targetClass})` : '');\n        const to = field.to.type + (field.to.targetClass ? ` (${field.to.targetClass})` : '');\n\n        logger.warn(\n          `The field \"${field.fieldName}\" type differ between the schema and the database for \"${localSchema.className}\"; Schema is defined as \"${to}\" and current database type is \"${from}\"`\n        );\n      });\n    }\n\n    fieldsWithChangedParams.forEach(fieldName => {\n      if (localSchema.fields) {\n        const field = localSchema.fields[fieldName];\n        this.handleFields(newLocalSchema, fieldName, field);\n      }\n    });\n\n    // Handle Indexes\n    // Check addition\n    if (localSchema.indexes) {\n      Object.keys(localSchema.indexes).forEach(indexName => {\n        if (\n          (!cloudSchema.indexes || !cloudSchema.indexes[indexName]) &&\n          !this.isProtectedIndex(localSchema.className, indexName)\n        ) {\n          if (localSchema.indexes) {\n            newLocalSchema.addIndex(indexName, localSchema.indexes[indexName]);\n          }\n        }\n      });\n    }\n\n    const indexesToAdd = [];\n\n    // Check deletion\n    if (cloudSchema.indexes) {\n      Object.keys(cloudSchema.indexes).forEach(indexName => {\n        if (!this.isProtectedIndex(localSchema.className, indexName)) {\n          if (!localSchema.indexes || !localSchema.indexes[indexName]) {\n            newLocalSchema.deleteIndex(indexName);\n          } else if (\n            !this.paramsAreEquals(localSchema.indexes[indexName], cloudSchema.indexes[indexName])\n          ) {\n            newLocalSchema.deleteIndex(indexName);\n            if (localSchema.indexes) {\n              indexesToAdd.push({\n                indexName,\n                index: localSchema.indexes[indexName],\n              });\n            }\n          }\n        }\n      });\n    }\n\n    this.handleCLP(localSchema, newLocalSchema, cloudSchema);\n    // Apply changes\n    await this.updateSchemaToDB(newLocalSchema);\n    // Apply new/changed indexes\n    if (indexesToAdd.length) {\n      logger.debug(\n        `Updating indexes for \"${newLocalSchema.className}\" :  ${indexesToAdd.join(' ,')}`\n      );\n      indexesToAdd.forEach(o => newLocalSchema.addIndex(o.indexName, o.index));\n      await this.updateSchemaToDB(newLocalSchema);\n    }\n  }\n\n  handleCLP(\n    localSchema: Migrations.JSONSchema,\n    newLocalSchema: Parse.Schema,\n    cloudSchema: Parse.Schema\n  ) {\n    if (!localSchema.classLevelPermissions && !cloudSchema) {\n      logger.warn(`classLevelPermissions not provided for ${localSchema.className}.`);\n    }\n    // Use spread to avoid read only issue (encountered by Moumouls using directAccess)\n    const clp = ({ ...localSchema.classLevelPermissions } || {}: Parse.CLP.PermissionsMap);\n    // To avoid inconsistency we need to remove all rights on addField\n    clp.addField = {};\n    newLocalSchema.setCLP(clp);\n  }\n\n  isProtectedFields(className: string, fieldName: string) {\n    return (\n      !!defaultColumns._Default[fieldName] ||\n      !!(defaultColumns[className] && defaultColumns[className][fieldName])\n    );\n  }\n\n  isProtectedIndex(className: string, indexName: string) {\n    const indexes = ['_id_'];\n    switch (className) {\n      case '_User':\n        indexes.push(\n          'case_insensitive_username',\n          'case_insensitive_email',\n          'username_1',\n          'email_1'\n        );\n        break;\n      case '_Role':\n        indexes.push('name_1');\n        break;\n\n      case '_Idempotency':\n        indexes.push('reqId_1');\n        break;\n    }\n\n    return indexes.indexOf(indexName) !== -1;\n  }\n\n  paramsAreEquals<T: { [key: string]: any }>(objA: T, objB: T) {\n    const keysA: string[] = Object.keys(objA);\n    const keysB: string[] = Object.keys(objB);\n\n    // Check key name\n    if (keysA.length !== keysB.length) return false;\n    return keysA.every(k => objA[k] === objB[k]);\n  }\n\n  handleFields(newLocalSchema: Parse.Schema, fieldName: string, field: Migrations.FieldType) {\n    if (field.type === 'Relation') {\n      newLocalSchema.addRelation(fieldName, field.targetClass);\n    } else if (field.type === 'Pointer') {\n      newLocalSchema.addPointer(fieldName, field.targetClass, field);\n    } else {\n      newLocalSchema.addField(fieldName, field.type, field);\n    }\n  }\n}\n"]}
|
|
372
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Parse","require","DefinedSchemas","constructor","schemaOptions","config","localSchemas","Config","get","appId","definitions","Array","isArray","retries","maxRetries","saveSchemaToDB","schema","payload","className","fields","_fields","indexes","_indexes","classLevelPermissions","_clp","internalCreateSchema","resetSchemaOps","updateSchemaToDB","internalUpdateSchema","execute","logger","info","beforeMigration","Promise","resolve","executeMigrations","afterMigration","e","error","process","env","NODE_ENV","exit","timeout","setTimeout","createDeleteSession","schemaController","database","loadSchema","allCloudSchemas","getAllClasses","clearTimeout","all","map","localSchema","saveOrUpdate","checkForMissingSchemas","enforceCLPForNonProvidedClass","wait","strict","cloudSchemas","s","missingSchemas","filter","c","includes","systemClasses","Set","size","length","join","warn","time","nonProvidedClasses","cloudSchema","some","parseSchema","Schema","handleCLP","response","rest","create","Auth","master","del","objectId","find","sc","updateSchema","saveSchema","newLocalSchema","Object","keys","fieldName","isProtectedFields","forEach","field","handleFields","indexName","isProtectedIndex","addIndex","fieldsToDelete","fieldsToRecreate","fieldsWithChangedParams","push","localField","paramsAreEquals","type","targetClass","from","to","deleteExtraFields","deleteField","recreateModifiedFields","fieldInfo","indexesToAdd","deleteIndex","index","debug","o","clp","addField","setCLP","defaultColumns","_Default","indexOf","objA","objB","keysA","keysB","every","k","addRelation","addPointer"],"sources":["../../src/SchemaMigrations/DefinedSchemas.js"],"sourcesContent":["// @flow\n// @flow-disable-next Cannot resolve module `parse/node`.\nconst Parse = require('parse/node');\nimport { logger } from '../logger';\nimport Config from '../Config';\nimport { internalCreateSchema, internalUpdateSchema } from '../Routers/SchemasRouter';\nimport { defaultColumns, systemClasses } from '../Controllers/SchemaController';\nimport { ParseServerOptions } from '../Options';\nimport * as Migrations from './Migrations';\nimport Auth from '../Auth';\nimport rest from '../rest';\n\nexport class DefinedSchemas {\n  config: ParseServerOptions;\n  schemaOptions: Migrations.SchemaOptions;\n  localSchemas: Migrations.JSONSchema[];\n  retries: number;\n  maxRetries: number;\n  allCloudSchemas: Parse.Schema[];\n\n  constructor(schemaOptions: Migrations.SchemaOptions, config: ParseServerOptions) {\n    this.localSchemas = [];\n    this.config = Config.get(config.appId);\n    this.schemaOptions = schemaOptions;\n    if (schemaOptions && schemaOptions.definitions) {\n      if (!Array.isArray(schemaOptions.definitions)) {\n        throw `\"schema.definitions\" must be an array of schemas`;\n      }\n\n      this.localSchemas = schemaOptions.definitions;\n    }\n\n    this.retries = 0;\n    this.maxRetries = 3;\n  }\n\n  async saveSchemaToDB(schema: Parse.Schema): Promise<void> {\n    const payload = {\n      className: schema.className,\n      fields: schema._fields,\n      indexes: schema._indexes,\n      classLevelPermissions: schema._clp,\n    };\n    await internalCreateSchema(schema.className, payload, this.config);\n    this.resetSchemaOps(schema);\n  }\n\n  resetSchemaOps(schema: Parse.Schema) {\n    // Reset ops like SDK\n    schema._fields = {};\n    schema._indexes = {};\n  }\n\n  // Simulate update like the SDK\n  // We cannot use SDK since routes are disabled\n  async updateSchemaToDB(schema: Parse.Schema) {\n    const payload = {\n      className: schema.className,\n      fields: schema._fields,\n      indexes: schema._indexes,\n      classLevelPermissions: schema._clp,\n    };\n    await internalUpdateSchema(schema.className, payload, this.config);\n    this.resetSchemaOps(schema);\n  }\n\n  async execute() {\n    try {\n      logger.info('Running Migrations');\n      if (this.schemaOptions && this.schemaOptions.beforeMigration) {\n        await Promise.resolve(this.schemaOptions.beforeMigration());\n      }\n\n      await this.executeMigrations();\n\n      if (this.schemaOptions && this.schemaOptions.afterMigration) {\n        await Promise.resolve(this.schemaOptions.afterMigration());\n      }\n\n      logger.info('Running Migrations Completed');\n    } catch (e) {\n      logger.error(`Failed to run migrations: ${e}`);\n      if (process.env.NODE_ENV === 'production') process.exit(1);\n    }\n  }\n\n  async executeMigrations() {\n    let timeout = null;\n    try {\n      // Set up a time out in production\n      // if we fail to get schema\n      // pm2 or K8s and many other process managers will try to restart the process\n      // after the exit\n      if (process.env.NODE_ENV === 'production') {\n        timeout = setTimeout(() => {\n          logger.error('Timeout occurred during execution of migrations. Exiting...');\n          process.exit(1);\n        }, 20000);\n      }\n\n      await this.createDeleteSession();\n      // @flow-disable-next-line\n      const schemaController = await this.config.database.loadSchema();\n      this.allCloudSchemas = await schemaController.getAllClasses();\n      clearTimeout(timeout);\n      await Promise.all(this.localSchemas.map(async localSchema => this.saveOrUpdate(localSchema)));\n\n      this.checkForMissingSchemas();\n      await this.enforceCLPForNonProvidedClass();\n    } catch (e) {\n      if (timeout) clearTimeout(timeout);\n      if (this.retries < this.maxRetries) {\n        this.retries++;\n        // first retry 1sec, 2sec, 3sec total 6sec retry sequence\n        // retry will only happen in case of deploying multi parse server instance\n        // at the same time. Modern systems like k8 avoid this by doing rolling updates\n        await this.wait(1000 * this.retries);\n        await this.executeMigrations();\n      } else {\n        logger.error(`Failed to run migrations: ${e}`);\n        if (process.env.NODE_ENV === 'production') process.exit(1);\n      }\n    }\n  }\n\n  checkForMissingSchemas() {\n    if (this.schemaOptions.strict !== true) {\n      return;\n    }\n\n    const cloudSchemas = this.allCloudSchemas.map(s => s.className);\n    const localSchemas = this.localSchemas.map(s => s.className);\n    const missingSchemas = cloudSchemas.filter(\n      c => !localSchemas.includes(c) && !systemClasses.includes(c)\n    );\n\n    if (new Set(localSchemas).size !== localSchemas.length) {\n      logger.error(\n        `The list of schemas provided contains duplicated \"className\"  \"${localSchemas.join(\n          '\",\"'\n        )}\"`\n      );\n      process.exit(1);\n    }\n\n    if (this.schemaOptions.strict && missingSchemas.length) {\n      logger.warn(\n        `The following schemas are currently present in the database, but not explicitly defined in a schema: \"${missingSchemas.join(\n          '\", \"'\n        )}\"`\n      );\n    }\n  }\n\n  // Required for testing purpose\n  wait(time: number) {\n    return new Promise<void>(resolve => setTimeout(resolve, time));\n  }\n\n  async enforceCLPForNonProvidedClass(): Promise<void> {\n    const nonProvidedClasses = this.allCloudSchemas.filter(\n      cloudSchema =>\n        !this.localSchemas.some(localSchema => localSchema.className === cloudSchema.className)\n    );\n    await Promise.all(\n      nonProvidedClasses.map(async schema => {\n        const parseSchema = new Parse.Schema(schema.className);\n        this.handleCLP(schema, parseSchema);\n        await this.updateSchemaToDB(parseSchema);\n      })\n    );\n  }\n\n  // Create a fake session since Parse do not create the _Session until\n  // a session is created\n  async createDeleteSession() {\n    const { response } = await rest.create(this.config, Auth.master(this.config), '_Session', {});\n    await rest.del(this.config, Auth.master(this.config), '_Session', response.objectId);\n  }\n\n  async saveOrUpdate(localSchema: Migrations.JSONSchema) {\n    const cloudSchema = this.allCloudSchemas.find(sc => sc.className === localSchema.className);\n    if (cloudSchema) {\n      try {\n        await this.updateSchema(localSchema, cloudSchema);\n      } catch (e) {\n        throw `Error during update of schema for type ${cloudSchema.className}: ${e}`;\n      }\n    } else {\n      try {\n        await this.saveSchema(localSchema);\n      } catch (e) {\n        throw `Error while saving Schema for type ${localSchema.className}: ${e}`;\n      }\n    }\n  }\n\n  async saveSchema(localSchema: Migrations.JSONSchema) {\n    const newLocalSchema = new Parse.Schema(localSchema.className);\n    if (localSchema.fields) {\n      // Handle fields\n      Object.keys(localSchema.fields)\n        .filter(fieldName => !this.isProtectedFields(localSchema.className, fieldName))\n        .forEach(fieldName => {\n          if (localSchema.fields) {\n            const field = localSchema.fields[fieldName];\n            this.handleFields(newLocalSchema, fieldName, field);\n          }\n        });\n    }\n    // Handle indexes\n    if (localSchema.indexes) {\n      Object.keys(localSchema.indexes).forEach(indexName => {\n        if (localSchema.indexes && !this.isProtectedIndex(localSchema.className, indexName)) {\n          newLocalSchema.addIndex(indexName, localSchema.indexes[indexName]);\n        }\n      });\n    }\n\n    this.handleCLP(localSchema, newLocalSchema);\n\n    return await this.saveSchemaToDB(newLocalSchema);\n  }\n\n  async updateSchema(localSchema: Migrations.JSONSchema, cloudSchema: Parse.Schema) {\n    const newLocalSchema = new Parse.Schema(localSchema.className);\n\n    // Handle fields\n    // Check addition\n    if (localSchema.fields) {\n      Object.keys(localSchema.fields)\n        .filter(fieldName => !this.isProtectedFields(localSchema.className, fieldName))\n        .forEach(fieldName => {\n          // @flow-disable-next\n          const field = localSchema.fields[fieldName];\n          if (!cloudSchema.fields[fieldName]) {\n            this.handleFields(newLocalSchema, fieldName, field);\n          }\n        });\n    }\n\n    const fieldsToDelete: string[] = [];\n    const fieldsToRecreate: {\n      fieldName: string,\n      from: { type: string, targetClass?: string },\n      to: { type: string, targetClass?: string },\n    }[] = [];\n    const fieldsWithChangedParams: string[] = [];\n\n    // Check deletion\n    Object.keys(cloudSchema.fields)\n      .filter(fieldName => !this.isProtectedFields(localSchema.className, fieldName))\n      .forEach(fieldName => {\n        const field = cloudSchema.fields[fieldName];\n        if (!localSchema.fields || !localSchema.fields[fieldName]) {\n          fieldsToDelete.push(fieldName);\n          return;\n        }\n\n        const localField = localSchema.fields[fieldName];\n        // Check if field has a changed type\n        if (\n          !this.paramsAreEquals(\n            { type: field.type, targetClass: field.targetClass },\n            { type: localField.type, targetClass: localField.targetClass }\n          )\n        ) {\n          fieldsToRecreate.push({\n            fieldName,\n            from: { type: field.type, targetClass: field.targetClass },\n            to: { type: localField.type, targetClass: localField.targetClass },\n          });\n          return;\n        }\n\n        // Check if something changed other than the type (like required, defaultValue)\n        if (!this.paramsAreEquals(field, localField)) {\n          fieldsWithChangedParams.push(fieldName);\n        }\n      });\n\n    if (this.schemaOptions.deleteExtraFields === true) {\n      fieldsToDelete.forEach(fieldName => {\n        newLocalSchema.deleteField(fieldName);\n      });\n\n      // Delete fields from the schema then apply changes\n      await this.updateSchemaToDB(newLocalSchema);\n    } else if (this.schemaOptions.strict === true && fieldsToDelete.length) {\n      logger.warn(\n        `The following fields exist in the database for \"${\n          localSchema.className\n        }\", but are missing in the schema : \"${fieldsToDelete.join('\" ,\"')}\"`\n      );\n    }\n\n    if (this.schemaOptions.recreateModifiedFields === true) {\n      fieldsToRecreate.forEach(field => {\n        newLocalSchema.deleteField(field.fieldName);\n      });\n\n      // Delete fields from the schema then apply changes\n      await this.updateSchemaToDB(newLocalSchema);\n\n      fieldsToRecreate.forEach(fieldInfo => {\n        if (localSchema.fields) {\n          const field = localSchema.fields[fieldInfo.fieldName];\n          this.handleFields(newLocalSchema, fieldInfo.fieldName, field);\n        }\n      });\n    } else if (this.schemaOptions.strict === true && fieldsToRecreate.length) {\n      fieldsToRecreate.forEach(field => {\n        const from =\n          field.from.type + (field.from.targetClass ? ` (${field.from.targetClass})` : '');\n        const to = field.to.type + (field.to.targetClass ? ` (${field.to.targetClass})` : '');\n\n        logger.warn(\n          `The field \"${field.fieldName}\" type differ between the schema and the database for \"${localSchema.className}\"; Schema is defined as \"${to}\" and current database type is \"${from}\"`\n        );\n      });\n    }\n\n    fieldsWithChangedParams.forEach(fieldName => {\n      if (localSchema.fields) {\n        const field = localSchema.fields[fieldName];\n        this.handleFields(newLocalSchema, fieldName, field);\n      }\n    });\n\n    // Handle Indexes\n    // Check addition\n    if (localSchema.indexes) {\n      Object.keys(localSchema.indexes).forEach(indexName => {\n        if (\n          (!cloudSchema.indexes || !cloudSchema.indexes[indexName]) &&\n          !this.isProtectedIndex(localSchema.className, indexName)\n        ) {\n          if (localSchema.indexes) {\n            newLocalSchema.addIndex(indexName, localSchema.indexes[indexName]);\n          }\n        }\n      });\n    }\n\n    const indexesToAdd = [];\n\n    // Check deletion\n    if (cloudSchema.indexes) {\n      Object.keys(cloudSchema.indexes).forEach(indexName => {\n        if (!this.isProtectedIndex(localSchema.className, indexName)) {\n          if (!localSchema.indexes || !localSchema.indexes[indexName]) {\n            newLocalSchema.deleteIndex(indexName);\n          } else if (\n            !this.paramsAreEquals(localSchema.indexes[indexName], cloudSchema.indexes[indexName])\n          ) {\n            newLocalSchema.deleteIndex(indexName);\n            if (localSchema.indexes) {\n              indexesToAdd.push({\n                indexName,\n                index: localSchema.indexes[indexName],\n              });\n            }\n          }\n        }\n      });\n    }\n\n    this.handleCLP(localSchema, newLocalSchema, cloudSchema);\n    // Apply changes\n    await this.updateSchemaToDB(newLocalSchema);\n    // Apply new/changed indexes\n    if (indexesToAdd.length) {\n      logger.debug(\n        `Updating indexes for \"${newLocalSchema.className}\" :  ${indexesToAdd.join(' ,')}`\n      );\n      indexesToAdd.forEach(o => newLocalSchema.addIndex(o.indexName, o.index));\n      await this.updateSchemaToDB(newLocalSchema);\n    }\n  }\n\n  handleCLP(\n    localSchema: Migrations.JSONSchema,\n    newLocalSchema: Parse.Schema,\n    cloudSchema: Parse.Schema\n  ) {\n    if (!localSchema.classLevelPermissions && !cloudSchema) {\n      logger.warn(`classLevelPermissions not provided for ${localSchema.className}.`);\n    }\n    // Use spread to avoid read only issue (encountered by Moumouls using directAccess)\n    const clp = ({ ...localSchema.classLevelPermissions } || {}: Parse.CLP.PermissionsMap);\n    // To avoid inconsistency we need to remove all rights on addField\n    clp.addField = {};\n    newLocalSchema.setCLP(clp);\n  }\n\n  isProtectedFields(className: string, fieldName: string) {\n    return (\n      !!defaultColumns._Default[fieldName] ||\n      !!(defaultColumns[className] && defaultColumns[className][fieldName])\n    );\n  }\n\n  isProtectedIndex(className: string, indexName: string) {\n    const indexes = ['_id_'];\n    switch (className) {\n      case '_User':\n        indexes.push(\n          'case_insensitive_username',\n          'case_insensitive_email',\n          'username_1',\n          'email_1'\n        );\n        break;\n      case '_Role':\n        indexes.push('name_1');\n        break;\n\n      case '_Idempotency':\n        indexes.push('reqId_1');\n        break;\n    }\n\n    return indexes.indexOf(indexName) !== -1;\n  }\n\n  paramsAreEquals<T: { [key: string]: any }>(objA: T, objB: T) {\n    const keysA: string[] = Object.keys(objA);\n    const keysB: string[] = Object.keys(objB);\n\n    // Check key name\n    if (keysA.length !== keysB.length) return false;\n    return keysA.every(k => objA[k] === objB[k]);\n  }\n\n  handleFields(newLocalSchema: Parse.Schema, fieldName: string, field: Migrations.FieldType) {\n    if (field.type === 'Relation') {\n      newLocalSchema.addRelation(fieldName, field.targetClass);\n    } else if (field.type === 'Pointer') {\n      newLocalSchema.addPointer(fieldName, field.targetClass, field);\n    } else {\n      newLocalSchema.addField(fieldName, field.type, field);\n    }\n  }\n}\n"],"mappings":"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"}
|