parse-server 9.9.0-alpha.2 → 9.9.0
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 +44 -0
- package/lib/Config.js +35 -2
- package/lib/Deprecator/Deprecations.js +5 -1
- package/lib/InstallationDedup.js +178 -0
- package/lib/Options/Definitions.js +26 -1
- package/lib/Options/docs.js +9 -1
- package/lib/Options/index.js +1 -1
- package/lib/RestWrite.js +36 -35
- package/package.json +1 -1
|
@@ -0,0 +1,178 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
Object.defineProperty(exports, "__esModule", {
|
|
4
|
+
value: true
|
|
5
|
+
});
|
|
6
|
+
exports.applyDuplicateDeviceTokenMerge = applyDuplicateDeviceTokenMerge;
|
|
7
|
+
exports.default = void 0;
|
|
8
|
+
exports.removeConflictingDeviceToken = removeConflictingDeviceToken;
|
|
9
|
+
var _node = _interopRequireDefault(require("parse/node"));
|
|
10
|
+
var _logger = _interopRequireDefault(require("./logger"));
|
|
11
|
+
function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
|
|
12
|
+
const CLASS_NAME = '_Installation';
|
|
13
|
+
function logResult(action, count, err) {
|
|
14
|
+
if (err && err.code === _node.default.Error.OBJECT_NOT_FOUND) {
|
|
15
|
+
_logger.default.verbose(`Installation dedup ${action} matched no rows; nothing to do.`);
|
|
16
|
+
return;
|
|
17
|
+
}
|
|
18
|
+
if (err && err.code === _node.default.Error.OPERATION_FORBIDDEN) {
|
|
19
|
+
_logger.default.warn(`Installation dedup ${action} skipped: caller has no permission to ${action} the conflicting row(s). The conflicting row remains.`);
|
|
20
|
+
return;
|
|
21
|
+
}
|
|
22
|
+
if (err) {
|
|
23
|
+
_logger.default.error(`Installation dedup ${action} failed: ${err.message || err}`);
|
|
24
|
+
return;
|
|
25
|
+
}
|
|
26
|
+
_logger.default.verbose(`Installation dedup ${action} applied to ${count == null ? 'matching' : count} conflicting row(s).`);
|
|
27
|
+
}
|
|
28
|
+
async function performAction({
|
|
29
|
+
database,
|
|
30
|
+
query,
|
|
31
|
+
action,
|
|
32
|
+
fieldToClear,
|
|
33
|
+
runOptions,
|
|
34
|
+
many,
|
|
35
|
+
validSchemaController
|
|
36
|
+
}) {
|
|
37
|
+
if (action === 'delete') {
|
|
38
|
+
return database.destroy(CLASS_NAME, query, runOptions, validSchemaController);
|
|
39
|
+
}
|
|
40
|
+
if (action === 'update') {
|
|
41
|
+
return database.update(CLASS_NAME, query, {
|
|
42
|
+
[fieldToClear]: {
|
|
43
|
+
__op: 'Delete'
|
|
44
|
+
}
|
|
45
|
+
}, {
|
|
46
|
+
...runOptions,
|
|
47
|
+
many
|
|
48
|
+
}, false, false, validSchemaController);
|
|
49
|
+
}
|
|
50
|
+
throw new Error(`Unknown installation dedup action: ${action}`);
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
/**
|
|
54
|
+
* Removes or updates `_Installation` rows that hold a `deviceToken` matching the query,
|
|
55
|
+
* allowing the caller to claim that `deviceToken` exclusively. Used when a new or updated
|
|
56
|
+
* install collides with one or more existing rows on `deviceToken`.
|
|
57
|
+
*
|
|
58
|
+
* @param {Object} options
|
|
59
|
+
* @param {DatabaseController} options.database
|
|
60
|
+
* @param {Object} options.query e.g. { deviceToken: 'X', installationId: { $ne: 'I' } }
|
|
61
|
+
* @param {'delete'|'update'} options.action
|
|
62
|
+
* @param {boolean} options.enforceAuth
|
|
63
|
+
* @param {Object} options.runOptions RestWrite.runOptions
|
|
64
|
+
* @param {SchemaController} options.validSchemaController
|
|
65
|
+
*/
|
|
66
|
+
async function removeConflictingDeviceToken({
|
|
67
|
+
database,
|
|
68
|
+
query,
|
|
69
|
+
action,
|
|
70
|
+
enforceAuth,
|
|
71
|
+
runOptions,
|
|
72
|
+
validSchemaController
|
|
73
|
+
}) {
|
|
74
|
+
const opts = enforceAuth ? runOptions : {};
|
|
75
|
+
try {
|
|
76
|
+
await performAction({
|
|
77
|
+
database,
|
|
78
|
+
query,
|
|
79
|
+
action,
|
|
80
|
+
fieldToClear: 'deviceToken',
|
|
81
|
+
runOptions: opts,
|
|
82
|
+
many: true,
|
|
83
|
+
validSchemaController
|
|
84
|
+
});
|
|
85
|
+
logResult(action, null, null);
|
|
86
|
+
} catch (err) {
|
|
87
|
+
if (err && err.code === _node.default.Error.OBJECT_NOT_FOUND) {
|
|
88
|
+
logResult(action, 0, err);
|
|
89
|
+
return;
|
|
90
|
+
}
|
|
91
|
+
if (err && err.code === _node.default.Error.OPERATION_FORBIDDEN) {
|
|
92
|
+
logResult(action, null, err);
|
|
93
|
+
return;
|
|
94
|
+
}
|
|
95
|
+
logResult(action, null, err);
|
|
96
|
+
throw err;
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
/**
|
|
101
|
+
* Resolves a merge conflict between two `_Installation` rows that together represent the
|
|
102
|
+
* same install: one matched by `installationId`/`objectId` (`idMatch`), and another holding
|
|
103
|
+
* the same `deviceToken` but no `installationId` (`deviceTokenMatch`). The `mergePriority`
|
|
104
|
+
* determines which row survives; the loser receives the configured `action`. Returns the
|
|
105
|
+
* survivor's `objectId` so the save flow can target it.
|
|
106
|
+
*
|
|
107
|
+
* @param {Object} options
|
|
108
|
+
* @param {DatabaseController} options.database
|
|
109
|
+
* @param {{ objectId: string, installationId?: string, deviceToken?: string }} options.idMatch
|
|
110
|
+
* @param {{ objectId: string, deviceToken?: string }} options.deviceTokenMatch
|
|
111
|
+
* @param {'delete'|'update'} options.action
|
|
112
|
+
* @param {'deviceToken'|'installationId'} options.mergePriority
|
|
113
|
+
* @param {boolean} options.enforceAuth
|
|
114
|
+
* @param {Object} options.runOptions
|
|
115
|
+
* @param {SchemaController} options.validSchemaController
|
|
116
|
+
* @returns {Promise<string>} survivor's objectId
|
|
117
|
+
*/
|
|
118
|
+
async function applyDuplicateDeviceTokenMerge({
|
|
119
|
+
database,
|
|
120
|
+
idMatch,
|
|
121
|
+
deviceTokenMatch,
|
|
122
|
+
action,
|
|
123
|
+
mergePriority,
|
|
124
|
+
enforceAuth,
|
|
125
|
+
runOptions,
|
|
126
|
+
validSchemaController
|
|
127
|
+
}) {
|
|
128
|
+
// Self-merge guard: when both matches resolve to the same row, there's
|
|
129
|
+
// nothing to clean up. Skip the action so we don't destroy/update the row
|
|
130
|
+
// we're about to return as the survivor.
|
|
131
|
+
if (idMatch.objectId === deviceTokenMatch.objectId) {
|
|
132
|
+
return idMatch.objectId;
|
|
133
|
+
}
|
|
134
|
+
const opts = enforceAuth ? runOptions : {};
|
|
135
|
+
let loser;
|
|
136
|
+
let survivorId;
|
|
137
|
+
let fieldToClear;
|
|
138
|
+
if (mergePriority === 'deviceToken') {
|
|
139
|
+
loser = idMatch;
|
|
140
|
+
survivorId = deviceTokenMatch.objectId;
|
|
141
|
+
fieldToClear = 'installationId';
|
|
142
|
+
} else if (mergePriority === 'installationId') {
|
|
143
|
+
loser = deviceTokenMatch;
|
|
144
|
+
survivorId = idMatch.objectId;
|
|
145
|
+
fieldToClear = 'deviceToken';
|
|
146
|
+
} else {
|
|
147
|
+
throw new Error(`Unknown installation dedup mergePriority: ${mergePriority}`);
|
|
148
|
+
}
|
|
149
|
+
try {
|
|
150
|
+
await performAction({
|
|
151
|
+
database,
|
|
152
|
+
query: {
|
|
153
|
+
objectId: loser.objectId
|
|
154
|
+
},
|
|
155
|
+
action,
|
|
156
|
+
fieldToClear,
|
|
157
|
+
runOptions: opts,
|
|
158
|
+
many: false,
|
|
159
|
+
validSchemaController
|
|
160
|
+
});
|
|
161
|
+
logResult(action, 1, null);
|
|
162
|
+
} catch (err) {
|
|
163
|
+
if (err && err.code === _node.default.Error.OBJECT_NOT_FOUND) {
|
|
164
|
+
logResult(action, 0, err);
|
|
165
|
+
} else if (err && err.code === _node.default.Error.OPERATION_FORBIDDEN) {
|
|
166
|
+
logResult(action, null, err);
|
|
167
|
+
} else {
|
|
168
|
+
logResult(action, null, err);
|
|
169
|
+
throw err;
|
|
170
|
+
}
|
|
171
|
+
}
|
|
172
|
+
return survivorId;
|
|
173
|
+
}
|
|
174
|
+
var _default = exports.default = {
|
|
175
|
+
removeConflictingDeviceToken,
|
|
176
|
+
applyDuplicateDeviceTokenMerge
|
|
177
|
+
};
|
|
178
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_node","_interopRequireDefault","require","_logger","e","__esModule","default","CLASS_NAME","logResult","action","count","err","code","Parse","Error","OBJECT_NOT_FOUND","logger","verbose","OPERATION_FORBIDDEN","warn","error","message","performAction","database","query","fieldToClear","runOptions","many","validSchemaController","destroy","update","__op","removeConflictingDeviceToken","enforceAuth","opts","applyDuplicateDeviceTokenMerge","idMatch","deviceTokenMatch","mergePriority","objectId","loser","survivorId","_default","exports"],"sources":["../src/InstallationDedup.js"],"sourcesContent":["import Parse from 'parse/node';\nimport logger from './logger';\n\nconst CLASS_NAME = '_Installation';\n\nfunction logResult(action, count, err) {\n  if (err && err.code === Parse.Error.OBJECT_NOT_FOUND) {\n    logger.verbose(`Installation dedup ${action} matched no rows; nothing to do.`);\n    return;\n  }\n  if (err && err.code === Parse.Error.OPERATION_FORBIDDEN) {\n    logger.warn(\n      `Installation dedup ${action} skipped: caller has no permission to ${action} the conflicting row(s). The conflicting row remains.`\n    );\n    return;\n  }\n  if (err) {\n    logger.error(`Installation dedup ${action} failed: ${err.message || err}`);\n    return;\n  }\n  logger.verbose(\n    `Installation dedup ${action} applied to ${count == null ? 'matching' : count} conflicting row(s).`\n  );\n}\n\nasync function performAction({\n  database,\n  query,\n  action,\n  fieldToClear,\n  runOptions,\n  many,\n  validSchemaController,\n}) {\n  if (action === 'delete') {\n    return database.destroy(CLASS_NAME, query, runOptions, validSchemaController);\n  }\n  if (action === 'update') {\n    return database.update(\n      CLASS_NAME,\n      query,\n      { [fieldToClear]: { __op: 'Delete' } },\n      { ...runOptions, many },\n      false,\n      false,\n      validSchemaController\n    );\n  }\n  throw new Error(`Unknown installation dedup action: ${action}`);\n}\n\n/**\n * Removes or updates `_Installation` rows that hold a `deviceToken` matching the query,\n * allowing the caller to claim that `deviceToken` exclusively. Used when a new or updated\n * install collides with one or more existing rows on `deviceToken`.\n *\n * @param {Object} options\n * @param {DatabaseController} options.database\n * @param {Object} options.query e.g. { deviceToken: 'X', installationId: { $ne: 'I' } }\n * @param {'delete'|'update'} options.action\n * @param {boolean} options.enforceAuth\n * @param {Object} options.runOptions RestWrite.runOptions\n * @param {SchemaController} options.validSchemaController\n */\nexport async function removeConflictingDeviceToken({\n  database,\n  query,\n  action,\n  enforceAuth,\n  runOptions,\n  validSchemaController,\n}) {\n  const opts = enforceAuth ? runOptions : {};\n  try {\n    await performAction({\n      database,\n      query,\n      action,\n      fieldToClear: 'deviceToken',\n      runOptions: opts,\n      many: true,\n      validSchemaController,\n    });\n    logResult(action, null, null);\n  } catch (err) {\n    if (err && err.code === Parse.Error.OBJECT_NOT_FOUND) {\n      logResult(action, 0, err);\n      return;\n    }\n    if (err && err.code === Parse.Error.OPERATION_FORBIDDEN) {\n      logResult(action, null, err);\n      return;\n    }\n    logResult(action, null, err);\n    throw err;\n  }\n}\n\n/**\n * Resolves a merge conflict between two `_Installation` rows that together represent the\n * same install: one matched by `installationId`/`objectId` (`idMatch`), and another holding\n * the same `deviceToken` but no `installationId` (`deviceTokenMatch`). The `mergePriority`\n * determines which row survives; the loser receives the configured `action`. Returns the\n * survivor's `objectId` so the save flow can target it.\n *\n * @param {Object} options\n * @param {DatabaseController} options.database\n * @param {{ objectId: string, installationId?: string, deviceToken?: string }} options.idMatch\n * @param {{ objectId: string, deviceToken?: string }} options.deviceTokenMatch\n * @param {'delete'|'update'} options.action\n * @param {'deviceToken'|'installationId'} options.mergePriority\n * @param {boolean} options.enforceAuth\n * @param {Object} options.runOptions\n * @param {SchemaController} options.validSchemaController\n * @returns {Promise<string>} survivor's objectId\n */\nexport async function applyDuplicateDeviceTokenMerge({\n  database,\n  idMatch,\n  deviceTokenMatch,\n  action,\n  mergePriority,\n  enforceAuth,\n  runOptions,\n  validSchemaController,\n}) {\n  // Self-merge guard: when both matches resolve to the same row, there's\n  // nothing to clean up. Skip the action so we don't destroy/update the row\n  // we're about to return as the survivor.\n  if (idMatch.objectId === deviceTokenMatch.objectId) {\n    return idMatch.objectId;\n  }\n  const opts = enforceAuth ? runOptions : {};\n  let loser;\n  let survivorId;\n  let fieldToClear;\n  if (mergePriority === 'deviceToken') {\n    loser = idMatch;\n    survivorId = deviceTokenMatch.objectId;\n    fieldToClear = 'installationId';\n  } else if (mergePriority === 'installationId') {\n    loser = deviceTokenMatch;\n    survivorId = idMatch.objectId;\n    fieldToClear = 'deviceToken';\n  } else {\n    throw new Error(`Unknown installation dedup mergePriority: ${mergePriority}`);\n  }\n\n  try {\n    await performAction({\n      database,\n      query: { objectId: loser.objectId },\n      action,\n      fieldToClear,\n      runOptions: opts,\n      many: false,\n      validSchemaController,\n    });\n    logResult(action, 1, null);\n  } catch (err) {\n    if (err && err.code === Parse.Error.OBJECT_NOT_FOUND) {\n      logResult(action, 0, err);\n    } else if (err && err.code === Parse.Error.OPERATION_FORBIDDEN) {\n      logResult(action, null, err);\n    } else {\n      logResult(action, null, err);\n      throw err;\n    }\n  }\n  return survivorId;\n}\n\nexport default {\n  removeConflictingDeviceToken,\n  applyDuplicateDeviceTokenMerge,\n};\n"],"mappings":"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","ignoreList":[]}
|