DefinedSchemas.js 60 KB

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