parse-server 2.8.4 → 8.6.2
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/LICENSE +167 -25
- package/NOTICE +10 -0
- package/README.md +929 -278
- package/lib/AccountLockout.js +47 -30
- package/lib/Adapters/AdapterLoader.js +21 -6
- package/lib/Adapters/Analytics/AnalyticsAdapter.js +15 -12
- package/lib/Adapters/Auth/AuthAdapter.js +116 -13
- package/lib/Adapters/Auth/BaseCodeAuthAdapter.js +99 -0
- package/lib/Adapters/Auth/OAuth1Client.js +27 -46
- package/lib/Adapters/Auth/apple.js +123 -0
- package/lib/Adapters/Auth/facebook.js +162 -35
- package/lib/Adapters/Auth/gcenter.js +217 -0
- package/lib/Adapters/Auth/github.js +118 -48
- package/lib/Adapters/Auth/google.js +160 -51
- package/lib/Adapters/Auth/gpgames.js +125 -0
- package/lib/Adapters/Auth/httpsRequest.js +6 -7
- package/lib/Adapters/Auth/index.js +170 -62
- package/lib/Adapters/Auth/instagram.js +114 -40
- package/lib/Adapters/Auth/janraincapture.js +52 -23
- package/lib/Adapters/Auth/janrainengage.js +19 -36
- package/lib/Adapters/Auth/keycloak.js +148 -0
- package/lib/Adapters/Auth/ldap.js +167 -0
- package/lib/Adapters/Auth/line.js +125 -0
- package/lib/Adapters/Auth/linkedin.js +111 -55
- package/lib/Adapters/Auth/meetup.js +24 -34
- package/lib/Adapters/Auth/mfa.js +324 -0
- package/lib/Adapters/Auth/microsoft.js +111 -0
- package/lib/Adapters/Auth/oauth2.js +97 -162
- package/lib/Adapters/Auth/phantauth.js +53 -0
- package/lib/Adapters/Auth/qq.js +108 -49
- package/lib/Adapters/Auth/spotify.js +107 -55
- package/lib/Adapters/Auth/twitter.js +188 -48
- package/lib/Adapters/Auth/utils.js +28 -0
- package/lib/Adapters/Auth/vkontakte.js +26 -39
- package/lib/Adapters/Auth/wechat.js +106 -44
- package/lib/Adapters/Auth/weibo.js +132 -58
- package/lib/Adapters/Cache/CacheAdapter.js +13 -8
- package/lib/Adapters/Cache/InMemoryCache.js +3 -13
- package/lib/Adapters/Cache/InMemoryCacheAdapter.js +5 -13
- package/lib/Adapters/Cache/LRUCache.js +13 -27
- package/lib/Adapters/Cache/NullCacheAdapter.js +3 -8
- package/lib/Adapters/Cache/RedisCacheAdapter.js +85 -76
- package/lib/Adapters/Cache/SchemaCache.js +25 -0
- package/lib/Adapters/Email/MailAdapter.js +10 -8
- package/lib/Adapters/Files/FilesAdapter.js +83 -25
- package/lib/Adapters/Files/GridFSBucketAdapter.js +231 -0
- package/lib/Adapters/Files/GridStoreAdapter.js +4 -91
- package/lib/Adapters/Logger/LoggerAdapter.js +18 -14
- package/lib/Adapters/Logger/WinstonLogger.js +69 -88
- package/lib/Adapters/Logger/WinstonLoggerAdapter.js +7 -16
- package/lib/Adapters/MessageQueue/EventEmitterMQ.js +8 -26
- package/lib/Adapters/PubSub/EventEmitterPubSub.js +12 -25
- package/lib/Adapters/PubSub/PubSubAdapter.js +34 -0
- package/lib/Adapters/PubSub/RedisPubSub.js +42 -19
- package/lib/Adapters/Push/PushAdapter.js +14 -7
- package/lib/Adapters/Storage/Mongo/MongoCollection.js +137 -45
- package/lib/Adapters/Storage/Mongo/MongoSchemaCollection.js +158 -63
- package/lib/Adapters/Storage/Mongo/MongoStorageAdapter.js +320 -168
- package/lib/Adapters/Storage/Mongo/MongoTransform.js +279 -306
- package/lib/Adapters/Storage/Postgres/PostgresClient.js +14 -10
- package/lib/Adapters/Storage/Postgres/PostgresConfigParser.js +47 -21
- package/lib/Adapters/Storage/Postgres/PostgresStorageAdapter.js +854 -468
- package/lib/Adapters/Storage/Postgres/sql/index.js +4 -6
- package/lib/Adapters/Storage/StorageAdapter.js +1 -1
- package/lib/Adapters/WebSocketServer/WSAdapter.js +35 -0
- package/lib/Adapters/WebSocketServer/WSSAdapter.js +66 -0
- package/lib/Auth.js +488 -125
- package/lib/ClientSDK.js +2 -6
- package/lib/Config.js +525 -94
- package/lib/Controllers/AdaptableController.js +5 -25
- package/lib/Controllers/AnalyticsController.js +22 -23
- package/lib/Controllers/CacheController.js +10 -31
- package/lib/Controllers/DatabaseController.js +767 -313
- package/lib/Controllers/FilesController.js +49 -54
- package/lib/Controllers/HooksController.js +80 -84
- package/lib/Controllers/LiveQueryController.js +35 -22
- package/lib/Controllers/LoggerController.js +22 -58
- package/lib/Controllers/ParseGraphQLController.js +293 -0
- package/lib/Controllers/PushController.js +58 -49
- package/lib/Controllers/SchemaController.js +916 -422
- package/lib/Controllers/UserController.js +265 -180
- package/lib/Controllers/index.js +90 -125
- package/lib/Controllers/types.js +1 -1
- package/lib/Deprecator/Deprecations.js +30 -0
- package/lib/Deprecator/Deprecator.js +127 -0
- package/lib/Error.js +48 -0
- package/lib/GraphQL/ParseGraphQLSchema.js +375 -0
- package/lib/GraphQL/ParseGraphQLServer.js +214 -0
- package/lib/GraphQL/helpers/objectsMutations.js +30 -0
- package/lib/GraphQL/helpers/objectsQueries.js +246 -0
- package/lib/GraphQL/loaders/configMutations.js +87 -0
- package/lib/GraphQL/loaders/configQueries.js +79 -0
- package/lib/GraphQL/loaders/defaultGraphQLMutations.js +21 -0
- package/lib/GraphQL/loaders/defaultGraphQLQueries.js +23 -0
- package/lib/GraphQL/loaders/defaultGraphQLTypes.js +1098 -0
- package/lib/GraphQL/loaders/defaultRelaySchema.js +53 -0
- package/lib/GraphQL/loaders/filesMutations.js +107 -0
- package/lib/GraphQL/loaders/functionsMutations.js +78 -0
- package/lib/GraphQL/loaders/parseClassMutations.js +268 -0
- package/lib/GraphQL/loaders/parseClassQueries.js +127 -0
- package/lib/GraphQL/loaders/parseClassTypes.js +493 -0
- package/lib/GraphQL/loaders/schemaDirectives.js +62 -0
- package/lib/GraphQL/loaders/schemaMutations.js +162 -0
- package/lib/GraphQL/loaders/schemaQueries.js +81 -0
- package/lib/GraphQL/loaders/schemaTypes.js +341 -0
- package/lib/GraphQL/loaders/usersMutations.js +433 -0
- package/lib/GraphQL/loaders/usersQueries.js +90 -0
- package/lib/GraphQL/parseGraphQLUtils.js +63 -0
- package/lib/GraphQL/transformers/className.js +14 -0
- package/lib/GraphQL/transformers/constraintType.js +53 -0
- package/lib/GraphQL/transformers/inputType.js +51 -0
- package/lib/GraphQL/transformers/mutation.js +274 -0
- package/lib/GraphQL/transformers/outputType.js +51 -0
- package/lib/GraphQL/transformers/query.js +237 -0
- package/lib/GraphQL/transformers/schemaFields.js +99 -0
- package/lib/KeyPromiseQueue.js +48 -0
- package/lib/LiveQuery/Client.js +25 -33
- package/lib/LiveQuery/Id.js +2 -5
- package/lib/LiveQuery/ParseCloudCodePublisher.js +26 -23
- package/lib/LiveQuery/ParseLiveQueryServer.js +560 -285
- package/lib/LiveQuery/ParsePubSub.js +7 -16
- package/lib/LiveQuery/ParseWebSocketServer.js +42 -39
- package/lib/LiveQuery/QueryTools.js +76 -15
- package/lib/LiveQuery/RequestSchema.js +111 -97
- package/lib/LiveQuery/SessionTokenCache.js +23 -36
- package/lib/LiveQuery/Subscription.js +8 -17
- package/lib/LiveQuery/equalObjects.js +2 -3
- package/lib/Options/Definitions.js +1355 -382
- package/lib/Options/docs.js +301 -62
- package/lib/Options/index.js +11 -1
- package/lib/Options/parsers.js +14 -10
- package/lib/Page.js +44 -0
- package/lib/ParseMessageQueue.js +6 -13
- package/lib/ParseServer.js +474 -235
- package/lib/ParseServerRESTController.js +102 -40
- package/lib/PromiseRouter.js +39 -50
- package/lib/Push/PushQueue.js +24 -30
- package/lib/Push/PushWorker.js +32 -56
- package/lib/Push/utils.js +22 -35
- package/lib/RestQuery.js +361 -139
- package/lib/RestWrite.js +713 -344
- package/lib/Routers/AggregateRouter.js +97 -71
- package/lib/Routers/AnalyticsRouter.js +8 -14
- package/lib/Routers/AudiencesRouter.js +16 -35
- package/lib/Routers/ClassesRouter.js +86 -72
- package/lib/Routers/CloudCodeRouter.js +28 -37
- package/lib/Routers/FeaturesRouter.js +22 -25
- package/lib/Routers/FilesRouter.js +266 -171
- package/lib/Routers/FunctionsRouter.js +87 -103
- package/lib/Routers/GlobalConfigRouter.js +94 -33
- package/lib/Routers/GraphQLRouter.js +41 -0
- package/lib/Routers/HooksRouter.js +43 -47
- package/lib/Routers/IAPValidationRouter.js +57 -70
- package/lib/Routers/InstallationsRouter.js +17 -25
- package/lib/Routers/LogsRouter.js +10 -25
- package/lib/Routers/PagesRouter.js +647 -0
- package/lib/Routers/PublicAPIRouter.js +104 -112
- package/lib/Routers/PurgeRouter.js +19 -29
- package/lib/Routers/PushRouter.js +14 -28
- package/lib/Routers/RolesRouter.js +7 -14
- package/lib/Routers/SchemasRouter.js +63 -42
- package/lib/Routers/SecurityRouter.js +34 -0
- package/lib/Routers/SessionsRouter.js +25 -38
- package/lib/Routers/UsersRouter.js +463 -190
- package/lib/SchemaMigrations/DefinedSchemas.js +379 -0
- package/lib/SchemaMigrations/Migrations.js +30 -0
- package/lib/Security/Check.js +109 -0
- package/lib/Security/CheckGroup.js +44 -0
- package/lib/Security/CheckGroups/CheckGroupDatabase.js +44 -0
- package/lib/Security/CheckGroups/CheckGroupServerConfig.js +96 -0
- package/lib/Security/CheckGroups/CheckGroups.js +21 -0
- package/lib/Security/CheckRunner.js +213 -0
- package/lib/SharedRest.js +29 -0
- package/lib/StatusHandler.js +96 -93
- package/lib/TestUtils.js +70 -14
- package/lib/Utils.js +468 -0
- package/lib/batch.js +74 -40
- package/lib/cache.js +8 -8
- package/lib/cli/definitions/parse-live-query-server.js +4 -3
- package/lib/cli/definitions/parse-server.js +4 -3
- package/lib/cli/parse-live-query-server.js +9 -17
- package/lib/cli/parse-server.js +49 -47
- package/lib/cli/utils/commander.js +20 -29
- package/lib/cli/utils/runner.js +31 -32
- package/lib/cloud-code/Parse.Cloud.js +711 -36
- package/lib/cloud-code/Parse.Server.js +21 -0
- package/lib/cryptoUtils.js +6 -11
- package/lib/defaults.js +21 -15
- package/lib/deprecated.js +1 -1
- package/lib/index.js +78 -67
- package/lib/logger.js +12 -20
- package/lib/middlewares.js +484 -160
- package/lib/password.js +10 -6
- package/lib/request.js +175 -0
- package/lib/requiredParameter.js +4 -3
- package/lib/rest.js +157 -82
- package/lib/triggers.js +627 -185
- package/lib/vendor/README.md +3 -3
- package/lib/vendor/mongodbUrl.js +224 -137
- package/package.json +135 -57
- package/postinstall.js +38 -50
- package/public_html/invalid_verification_link.html +3 -3
- package/types/@types/@parse/fs-files-adapter/index.d.ts +5 -0
- package/types/@types/deepcopy/index.d.ts +5 -0
- package/types/LiveQuery/ParseLiveQueryServer.d.ts +40 -0
- package/types/Options/index.d.ts +301 -0
- package/types/ParseServer.d.ts +65 -0
- package/types/eslint.config.mjs +30 -0
- package/types/index.d.ts +21 -0
- package/types/logger.d.ts +2 -0
- package/types/tests.ts +44 -0
- package/types/tsconfig.json +24 -0
- package/CHANGELOG.md +0 -1246
- package/PATENTS +0 -37
- package/bin/dev +0 -37
- package/lib/.DS_Store +0 -0
- package/lib/Adapters/Auth/common.js +0 -2
- package/lib/Adapters/Auth/facebookaccountkit.js +0 -69
- package/lib/Controllers/SchemaCache.js +0 -97
- package/lib/LiveQuery/.DS_Store +0 -0
- package/lib/cli/utils/parsers.js +0 -77
- package/lib/cloud-code/.DS_Store +0 -0
- package/lib/cloud-code/HTTPResponse.js +0 -57
- package/lib/cloud-code/Untitled-1 +0 -123
- package/lib/cloud-code/httpRequest.js +0 -102
- package/lib/cloud-code/team.html +0 -123
- package/lib/graphql/ParseClass.js +0 -234
- package/lib/graphql/Schema.js +0 -197
- package/lib/graphql/index.js +0 -1
- package/lib/graphql/types/ACL.js +0 -35
- package/lib/graphql/types/Date.js +0 -25
- package/lib/graphql/types/File.js +0 -24
- package/lib/graphql/types/GeoPoint.js +0 -35
- package/lib/graphql/types/JSONObject.js +0 -30
- package/lib/graphql/types/NumberInput.js +0 -43
- package/lib/graphql/types/NumberQuery.js +0 -42
- package/lib/graphql/types/Pointer.js +0 -35
- package/lib/graphql/types/QueryConstraint.js +0 -61
- package/lib/graphql/types/StringQuery.js +0 -39
- package/lib/graphql/types/index.js +0 -110
package/lib/StatusHandler.js
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
|
|
1
|
+
"use strict";
|
|
2
2
|
|
|
3
3
|
Object.defineProperty(exports, "__esModule", {
|
|
4
4
|
value: true
|
|
@@ -6,33 +6,27 @@ Object.defineProperty(exports, "__esModule", {
|
|
|
6
6
|
exports.flatten = flatten;
|
|
7
7
|
exports.jobStatusHandler = jobStatusHandler;
|
|
8
8
|
exports.pushStatusHandler = pushStatusHandler;
|
|
9
|
-
|
|
10
|
-
var
|
|
11
|
-
|
|
12
|
-
var
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
var _rest2 = _interopRequireDefault(_rest);
|
|
17
|
-
|
|
18
|
-
var _Auth = require('./Auth');
|
|
19
|
-
|
|
20
|
-
var _Auth2 = _interopRequireDefault(_Auth);
|
|
21
|
-
|
|
22
|
-
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
23
|
-
|
|
9
|
+
var _cryptoUtils = require("./cryptoUtils");
|
|
10
|
+
var _KeyPromiseQueue = require("./KeyPromiseQueue");
|
|
11
|
+
var _logger = require("./logger");
|
|
12
|
+
var _rest = _interopRequireDefault(require("./rest"));
|
|
13
|
+
var _Auth = _interopRequireDefault(require("./Auth"));
|
|
14
|
+
function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
|
|
24
15
|
const PUSH_STATUS_COLLECTION = '_PushStatus';
|
|
25
16
|
const JOB_STATUS_COLLECTION = '_JobStatus';
|
|
26
|
-
|
|
17
|
+
const pushPromiseQueue = new _KeyPromiseQueue.KeyPromiseQueue();
|
|
18
|
+
const jobPromiseQueue = new _KeyPromiseQueue.KeyPromiseQueue();
|
|
27
19
|
const incrementOp = function (object = {}, key, amount = 1) {
|
|
28
20
|
if (!object[key]) {
|
|
29
|
-
object[key] = {
|
|
21
|
+
object[key] = {
|
|
22
|
+
__op: 'Increment',
|
|
23
|
+
amount: amount
|
|
24
|
+
};
|
|
30
25
|
} else {
|
|
31
26
|
object[key].amount += amount;
|
|
32
27
|
}
|
|
33
28
|
return object[key];
|
|
34
29
|
};
|
|
35
|
-
|
|
36
30
|
function flatten(array) {
|
|
37
31
|
var flattened = [];
|
|
38
32
|
for (var i = 0; i < array.length; i++) {
|
|
@@ -44,107 +38,99 @@ function flatten(array) {
|
|
|
44
38
|
}
|
|
45
39
|
return flattened;
|
|
46
40
|
}
|
|
47
|
-
|
|
48
41
|
function statusHandler(className, database) {
|
|
49
|
-
let lastPromise = Promise.resolve();
|
|
50
|
-
|
|
51
42
|
function create(object) {
|
|
52
|
-
|
|
53
|
-
return
|
|
54
|
-
return Promise.resolve(object);
|
|
55
|
-
});
|
|
43
|
+
return database.create(className, object).then(() => {
|
|
44
|
+
return Promise.resolve(object);
|
|
56
45
|
});
|
|
57
|
-
return lastPromise;
|
|
58
46
|
}
|
|
59
|
-
|
|
60
47
|
function update(where, object) {
|
|
61
|
-
|
|
62
|
-
return database.update(className, where, object);
|
|
63
|
-
});
|
|
64
|
-
return lastPromise;
|
|
48
|
+
return jobPromiseQueue.enqueue(where.objectId, () => database.update(className, where, object));
|
|
65
49
|
}
|
|
66
|
-
|
|
67
50
|
return Object.freeze({
|
|
68
51
|
create,
|
|
69
52
|
update
|
|
70
53
|
});
|
|
71
54
|
}
|
|
72
|
-
|
|
73
55
|
function restStatusHandler(className, config) {
|
|
74
|
-
|
|
75
|
-
const auth = _Auth2.default.master(config);
|
|
56
|
+
const auth = _Auth.default.master(config);
|
|
76
57
|
function create(object) {
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
58
|
+
return _rest.default.create(config, auth, className, object).then(({
|
|
59
|
+
response
|
|
60
|
+
}) => {
|
|
61
|
+
return {
|
|
62
|
+
...object,
|
|
63
|
+
...response
|
|
64
|
+
};
|
|
82
65
|
});
|
|
83
|
-
return lastPromise;
|
|
84
66
|
}
|
|
85
|
-
|
|
86
67
|
function update(where, object) {
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
68
|
+
return pushPromiseQueue.enqueue(where.objectId, () => _rest.default.update(config, auth, className, {
|
|
69
|
+
objectId: where.objectId
|
|
70
|
+
}, object).then(({
|
|
71
|
+
response
|
|
72
|
+
}) => {
|
|
73
|
+
return {
|
|
74
|
+
...object,
|
|
75
|
+
...response
|
|
76
|
+
};
|
|
77
|
+
}));
|
|
95
78
|
}
|
|
96
|
-
|
|
97
79
|
return Object.freeze({
|
|
98
80
|
create,
|
|
99
81
|
update
|
|
100
82
|
});
|
|
101
83
|
}
|
|
102
|
-
|
|
103
84
|
function jobStatusHandler(config) {
|
|
104
85
|
let jobStatus;
|
|
105
86
|
const objectId = (0, _cryptoUtils.newObjectId)(config.objectIdSize);
|
|
106
87
|
const database = config.database;
|
|
107
88
|
const handler = statusHandler(JOB_STATUS_COLLECTION, database);
|
|
108
|
-
const setRunning = function (jobName
|
|
89
|
+
const setRunning = function (jobName) {
|
|
109
90
|
const now = new Date();
|
|
110
91
|
jobStatus = {
|
|
111
92
|
objectId,
|
|
112
93
|
jobName,
|
|
113
|
-
params,
|
|
114
94
|
status: 'running',
|
|
115
95
|
source: 'api',
|
|
116
96
|
createdAt: now,
|
|
117
97
|
// lockdown!
|
|
118
98
|
ACL: {}
|
|
119
99
|
};
|
|
120
|
-
|
|
121
100
|
return handler.create(jobStatus);
|
|
122
101
|
};
|
|
123
|
-
|
|
124
102
|
const setMessage = function (message) {
|
|
125
103
|
if (!message || typeof message !== 'string') {
|
|
126
104
|
return Promise.resolve();
|
|
127
105
|
}
|
|
128
|
-
return handler.update({
|
|
106
|
+
return handler.update({
|
|
107
|
+
objectId
|
|
108
|
+
}, {
|
|
109
|
+
message
|
|
110
|
+
});
|
|
129
111
|
};
|
|
130
|
-
|
|
131
112
|
const setSucceeded = function (message) {
|
|
132
113
|
return setFinalStatus('succeeded', message);
|
|
133
114
|
};
|
|
134
|
-
|
|
135
115
|
const setFailed = function (message) {
|
|
136
116
|
return setFinalStatus('failed', message);
|
|
137
117
|
};
|
|
138
|
-
|
|
139
118
|
const setFinalStatus = function (status, message = undefined) {
|
|
140
119
|
const finishedAt = new Date();
|
|
141
|
-
const update = {
|
|
120
|
+
const update = {
|
|
121
|
+
status,
|
|
122
|
+
finishedAt
|
|
123
|
+
};
|
|
142
124
|
if (message && typeof message === 'string') {
|
|
143
125
|
update.message = message;
|
|
144
126
|
}
|
|
145
|
-
|
|
127
|
+
if (message instanceof Error && typeof message.message === 'string') {
|
|
128
|
+
update.message = message.message;
|
|
129
|
+
}
|
|
130
|
+
return handler.update({
|
|
131
|
+
objectId
|
|
132
|
+
}, update);
|
|
146
133
|
};
|
|
147
|
-
|
|
148
134
|
return Object.freeze({
|
|
149
135
|
setRunning,
|
|
150
136
|
setSucceeded,
|
|
@@ -152,18 +138,18 @@ function jobStatusHandler(config) {
|
|
|
152
138
|
setFailed
|
|
153
139
|
});
|
|
154
140
|
}
|
|
155
|
-
|
|
156
141
|
function pushStatusHandler(config, existingObjectId) {
|
|
157
|
-
|
|
158
142
|
let pushStatus;
|
|
159
143
|
const database = config.database;
|
|
160
144
|
const handler = restStatusHandler(PUSH_STATUS_COLLECTION, config);
|
|
161
145
|
let objectId = existingObjectId;
|
|
162
|
-
const setInitial = function (body = {}, where, options = {
|
|
146
|
+
const setInitial = function (body = {}, where, options = {
|
|
147
|
+
source: 'rest'
|
|
148
|
+
}) {
|
|
163
149
|
const now = new Date();
|
|
164
150
|
let pushTime = now.toISOString();
|
|
165
151
|
let status = 'pending';
|
|
166
|
-
if (
|
|
152
|
+
if (Object.prototype.hasOwnProperty.call(body, 'push_time')) {
|
|
167
153
|
if (config.hasPushScheduledSupport) {
|
|
168
154
|
pushTime = body.push_time;
|
|
169
155
|
status = 'scheduled';
|
|
@@ -172,7 +158,6 @@ function pushStatusHandler(config, existingObjectId) {
|
|
|
172
158
|
_logger.logger.warn('Push will be sent immediately');
|
|
173
159
|
}
|
|
174
160
|
}
|
|
175
|
-
|
|
176
161
|
const data = body.data || {};
|
|
177
162
|
const payloadString = JSON.stringify(data);
|
|
178
163
|
let pushHash;
|
|
@@ -205,18 +190,16 @@ function pushStatusHandler(config, existingObjectId) {
|
|
|
205
190
|
return Promise.resolve(pushStatus);
|
|
206
191
|
});
|
|
207
192
|
};
|
|
208
|
-
|
|
209
193
|
const setRunning = function (batches) {
|
|
210
194
|
_logger.logger.verbose(`_PushStatus ${objectId}: sending push to installations with %d batches`, batches);
|
|
211
195
|
return handler.update({
|
|
212
|
-
status:
|
|
196
|
+
status: 'pending',
|
|
213
197
|
objectId: objectId
|
|
214
198
|
}, {
|
|
215
|
-
status:
|
|
199
|
+
status: 'running',
|
|
216
200
|
count: batches
|
|
217
201
|
});
|
|
218
202
|
};
|
|
219
|
-
|
|
220
203
|
const trackSent = function (results, UTCOffset, cleanupInstallations = process.env.PARSE_SERVER_CLEANUP_INVALID_INSTALLATIONS) {
|
|
221
204
|
const update = {
|
|
222
205
|
numSent: 0,
|
|
@@ -243,11 +226,19 @@ function pushStatusHandler(config, existingObjectId) {
|
|
|
243
226
|
if (result && result.response && result.response.error && result.device && result.device.deviceToken) {
|
|
244
227
|
const token = result.device.deviceToken;
|
|
245
228
|
const error = result.response.error;
|
|
246
|
-
// GCM errors
|
|
229
|
+
// GCM / FCM HTTP v1 API errors; see:
|
|
230
|
+
// https://firebase.google.com/docs/reference/fcm/rest/v1/ErrorCode
|
|
247
231
|
if (error === 'NotRegistered' || error === 'InvalidRegistration') {
|
|
248
232
|
devicesToRemove.push(token);
|
|
249
233
|
}
|
|
250
|
-
//
|
|
234
|
+
// FCM API v2 errors; see:
|
|
235
|
+
// https://firebase.google.com/docs/cloud-messaging/manage-tokens
|
|
236
|
+
// https://github.com/firebase/functions-samples/blob/703c0359eacf07a551751d1319d34f912a2cd828/Node/fcm-notifications/functions/index.js#L89-L93C16
|
|
237
|
+
if (error?.code === 'messaging/registration-token-not-registered' || error?.code === 'messaging/invalid-registration-token' || error?.code === 'messaging/invalid-argument' && error?.message === 'The registration token is not a valid FCM registration token') {
|
|
238
|
+
devicesToRemove.push(token);
|
|
239
|
+
}
|
|
240
|
+
// APNS errors; see:
|
|
241
|
+
// https://developer.apple.com/documentation/usernotifications/handling-notification-responses-from-apns
|
|
251
242
|
if (error === 'Unregistered' || error === 'BadDeviceToken') {
|
|
252
243
|
devicesToRemove.push(token);
|
|
253
244
|
}
|
|
@@ -257,9 +248,10 @@ function pushStatusHandler(config, existingObjectId) {
|
|
|
257
248
|
return memo;
|
|
258
249
|
}, update);
|
|
259
250
|
}
|
|
260
|
-
|
|
261
251
|
_logger.logger.verbose(`_PushStatus ${objectId}: sent push! %d success, %d failures`, update.numSent, update.numFailed);
|
|
262
|
-
_logger.logger.verbose(`_PushStatus ${objectId}: needs cleanup`, {
|
|
252
|
+
_logger.logger.verbose(`_PushStatus ${objectId}: needs cleanup`, {
|
|
253
|
+
devicesToRemove
|
|
254
|
+
});
|
|
263
255
|
['numSent', 'numFailed'].forEach(key => {
|
|
264
256
|
if (update[key] > 0) {
|
|
265
257
|
update[key] = {
|
|
@@ -270,43 +262,55 @@ function pushStatusHandler(config, existingObjectId) {
|
|
|
270
262
|
delete update[key];
|
|
271
263
|
}
|
|
272
264
|
});
|
|
273
|
-
|
|
274
265
|
if (devicesToRemove.length > 0 && cleanupInstallations) {
|
|
275
266
|
_logger.logger.info(`Removing device tokens on ${devicesToRemove.length} _Installations`);
|
|
276
|
-
database.update('_Installation', {
|
|
267
|
+
database.update('_Installation', {
|
|
268
|
+
deviceToken: {
|
|
269
|
+
$in: devicesToRemove
|
|
270
|
+
}
|
|
271
|
+
}, {
|
|
272
|
+
deviceToken: {
|
|
273
|
+
__op: 'Delete'
|
|
274
|
+
}
|
|
275
|
+
}, {
|
|
277
276
|
acl: undefined,
|
|
278
277
|
many: true
|
|
279
278
|
});
|
|
280
279
|
}
|
|
281
|
-
|
|
282
|
-
// indicate this batch is complete
|
|
283
280
|
incrementOp(update, 'count', -1);
|
|
284
|
-
|
|
285
|
-
return handler.update({
|
|
281
|
+
update.status = 'running';
|
|
282
|
+
return handler.update({
|
|
283
|
+
objectId
|
|
284
|
+
}, update).then(res => {
|
|
286
285
|
if (res && res.count === 0) {
|
|
287
286
|
return this.complete();
|
|
288
287
|
}
|
|
289
288
|
});
|
|
290
289
|
};
|
|
291
|
-
|
|
292
290
|
const complete = function () {
|
|
293
|
-
return handler.update({
|
|
291
|
+
return handler.update({
|
|
292
|
+
objectId
|
|
293
|
+
}, {
|
|
294
294
|
status: 'succeeded',
|
|
295
|
-
count: {
|
|
295
|
+
count: {
|
|
296
|
+
__op: 'Delete'
|
|
297
|
+
}
|
|
296
298
|
});
|
|
297
299
|
};
|
|
298
|
-
|
|
299
300
|
const fail = function (err) {
|
|
300
301
|
if (typeof err === 'string') {
|
|
301
|
-
err = {
|
|
302
|
+
err = {
|
|
303
|
+
message: err
|
|
304
|
+
};
|
|
302
305
|
}
|
|
303
306
|
const update = {
|
|
304
307
|
errorMessage: err,
|
|
305
308
|
status: 'failed'
|
|
306
309
|
};
|
|
307
|
-
return handler.update({
|
|
310
|
+
return handler.update({
|
|
311
|
+
objectId
|
|
312
|
+
}, update);
|
|
308
313
|
};
|
|
309
|
-
|
|
310
314
|
const rval = {
|
|
311
315
|
setInitial,
|
|
312
316
|
setRunning,
|
|
@@ -316,10 +320,9 @@ function pushStatusHandler(config, existingObjectId) {
|
|
|
316
320
|
};
|
|
317
321
|
|
|
318
322
|
// define objectId to be dynamic
|
|
319
|
-
Object.defineProperty(rval,
|
|
323
|
+
Object.defineProperty(rval, 'objectId', {
|
|
320
324
|
get: () => objectId
|
|
321
325
|
});
|
|
322
|
-
|
|
323
326
|
return Object.freeze(rval);
|
|
324
327
|
}
|
|
325
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/StatusHandler.js"],"names":["flatten","jobStatusHandler","pushStatusHandler","PUSH_STATUS_COLLECTION","JOB_STATUS_COLLECTION","incrementOp","object","key","amount","__op","array","flattened","i","length","Array","isArray","concat","push","statusHandler","className","database","lastPromise","Promise","resolve","create","then","update","where","Object","freeze","restStatusHandler","config","auth","Auth","master","rest","response","assign","objectId","jobStatus","objectIdSize","handler","setRunning","jobName","params","now","Date","status","source","createdAt","ACL","setMessage","message","setSucceeded","setFinalStatus","setFailed","undefined","finishedAt","existingObjectId","pushStatus","setInitial","body","options","pushTime","toISOString","hasOwnProperty","hasPushScheduledSupport","push_time","logger","warn","data","payloadString","JSON","stringify","pushHash","alert","query","payload","title","expiry","expiration_time","expiration_interval","numSent","result","batches","verbose","count","trackSent","results","UTCOffset","cleanupInstallations","process","env","PARSE_SERVER_CLEANUP_INVALID_INSTALLATIONS","numFailed","devicesToRemove","reduce","memo","device","deviceType","transmitted","offsetKey","error","deviceToken","token","forEach","info","acl","many","res","complete","fail","err","errorMessage","rval","defineProperty","get"],"mappings":"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","file":"StatusHandler.js","sourcesContent":["import { md5Hash, newObjectId } from './cryptoUtils';\nimport { logger }               from './logger';\nimport rest                     from './rest';\nimport Auth                     from './Auth';\n\nconst PUSH_STATUS_COLLECTION = '_PushStatus';\nconst JOB_STATUS_COLLECTION = '_JobStatus';\n\nconst incrementOp = function(object = {}, key, amount = 1) {\n  if (!object[key]) {\n    object[key] = {__op: 'Increment', amount: amount}\n  } else {\n    object[key].amount += amount;\n  }\n  return object[key];\n}\n\nexport function flatten(array) {\n  var flattened = [];\n  for(var i = 0; i < array.length; i++) {\n    if(Array.isArray(array[i])) {\n      flattened = flattened.concat(flatten(array[i]));\n    } else {\n      flattened.push(array[i]);\n    }\n  }\n  return flattened;\n}\n\nfunction statusHandler(className, database) {\n  let lastPromise = Promise.resolve();\n\n  function create(object) {\n    lastPromise = lastPromise.then(() => {\n      return database.create(className, object).then(() => {\n        return Promise.resolve(object);\n      });\n    });\n    return lastPromise;\n  }\n\n  function update(where, object) {\n    lastPromise = lastPromise.then(() => {\n      return database.update(className, where, object);\n    });\n    return lastPromise;\n  }\n\n  return Object.freeze({\n    create,\n    update\n  })\n}\n\nfunction restStatusHandler(className, config) {\n  let lastPromise = Promise.resolve();\n  const auth = Auth.master(config);\n  function create(object) {\n    lastPromise = lastPromise.then(() => {\n      return rest.create(config, auth, className, object)\n        .then(({ response }) => {\n          // merge the objects\n          return Promise.resolve(Object.assign({}, object, response));\n        });\n    });\n    return lastPromise;\n  }\n\n  function update(where, object) {\n    // TODO: when we have updateWhere, use that for proper interfacing\n    lastPromise = lastPromise.then(() => {\n      return rest.update(config, auth, className, { objectId: where.objectId }, object)\n        .then(({ response }) => {\n          // merge the objects\n          return Promise.resolve(Object.assign({}, object, response));\n        });\n    });\n    return lastPromise;\n  }\n\n  return Object.freeze({\n    create,\n    update\n  })\n}\n\nexport function jobStatusHandler(config) {\n  let jobStatus;\n  const objectId = newObjectId(config.objectIdSize);\n  const database = config.database;\n  const handler = statusHandler(JOB_STATUS_COLLECTION, database);\n  const setRunning = function(jobName, params) {\n    const now = new Date();\n    jobStatus = {\n      objectId,\n      jobName,\n      params,\n      status: 'running',\n      source: 'api',\n      createdAt: now,\n      // lockdown!\n      ACL: {}\n    }\n\n    return handler.create(jobStatus);\n  }\n\n  const setMessage = function(message) {\n    if (!message || typeof message !== 'string') {\n      return Promise.resolve();\n    }\n    return handler.update({ objectId }, { message });\n  }\n\n  const setSucceeded = function(message) {\n    return setFinalStatus('succeeded', message);\n  }\n\n  const setFailed = function(message) {\n    return setFinalStatus('failed', message);\n  }\n\n  const setFinalStatus = function(status, message = undefined) {\n    const finishedAt = new Date();\n    const update = { status, finishedAt };\n    if (message && typeof message === 'string') {\n      update.message = message;\n    }\n    return handler.update({ objectId }, update);\n  }\n\n  return Object.freeze({\n    setRunning,\n    setSucceeded,\n    setMessage,\n    setFailed\n  });\n}\n\nexport function pushStatusHandler(config, existingObjectId) {\n\n  let pushStatus;\n  const database = config.database;\n  const handler = restStatusHandler(PUSH_STATUS_COLLECTION, config);\n  let objectId = existingObjectId;\n  const setInitial = function(body = {}, where, options = {source: 'rest'}) {\n    const now = new Date();\n    let pushTime = now.toISOString();\n    let status = 'pending';\n    if (body.hasOwnProperty('push_time')) {\n      if (config.hasPushScheduledSupport) {\n        pushTime = body.push_time;\n        status = 'scheduled';\n      } else {\n        logger.warn('Trying to schedule a push while server is not configured.');\n        logger.warn('Push will be sent immediately');\n      }\n    }\n\n    const data =  body.data || {};\n    const payloadString = JSON.stringify(data);\n    let pushHash;\n    if (typeof data.alert === 'string') {\n      pushHash = md5Hash(data.alert);\n    } else if (typeof data.alert === 'object') {\n      pushHash = md5Hash(JSON.stringify(data.alert));\n    } else {\n      pushHash = 'd41d8cd98f00b204e9800998ecf8427e';\n    }\n    const object = {\n      pushTime,\n      query: JSON.stringify(where),\n      payload: payloadString,\n      source: options.source,\n      title: options.title,\n      expiry: body.expiration_time,\n      expiration_interval: body.expiration_interval,\n      status: status,\n      numSent: 0,\n      pushHash,\n      // lockdown!\n      ACL: {}\n    }\n    return handler.create(object).then((result) => {\n      objectId = result.objectId;\n      pushStatus = {\n        objectId\n      };\n      return Promise.resolve(pushStatus);\n    });\n  }\n\n  const setRunning = function(batches) {\n    logger.verbose(`_PushStatus ${objectId}: sending push to installations with %d batches`, batches);\n    return handler.update(\n      {\n        status:\"pending\",\n        objectId: objectId\n      },\n      {\n        status: \"running\",\n        count: batches\n      }\n    );\n  }\n\n  const trackSent = function(results, UTCOffset, cleanupInstallations = process.env.PARSE_SERVER_CLEANUP_INVALID_INSTALLATIONS) {\n    const update = {\n      numSent: 0,\n      numFailed: 0\n    };\n    const devicesToRemove = [];\n    if (Array.isArray(results)) {\n      results = flatten(results);\n      results.reduce((memo, result) => {\n        // Cannot handle that\n        if (!result || !result.device || !result.device.deviceType) {\n          return memo;\n        }\n        const deviceType = result.device.deviceType;\n        const key = result.transmitted ? `sentPerType.${deviceType}` : `failedPerType.${deviceType}`;\n        memo[key] = incrementOp(memo, key);\n        if (typeof UTCOffset !== 'undefined') {\n          const offsetKey = result.transmitted ? `sentPerUTCOffset.${UTCOffset}` : `failedPerUTCOffset.${UTCOffset}`;\n          memo[offsetKey] = incrementOp(memo, offsetKey);\n        }\n        if (result.transmitted) {\n          memo.numSent++;\n        } else {\n          if (result && result.response && result.response.error && result.device && result.device.deviceToken) {\n            const token = result.device.deviceToken;\n            const error = result.response.error;\n            // GCM errors\n            if (error === 'NotRegistered' || error === 'InvalidRegistration') {\n              devicesToRemove.push(token);\n            }\n            // APNS errors\n            if (error === 'Unregistered' || error === 'BadDeviceToken') {\n              devicesToRemove.push(token);\n            }\n          }\n          memo.numFailed++;\n        }\n        return memo;\n      }, update);\n    }\n\n    logger.verbose(`_PushStatus ${objectId}: sent push! %d success, %d failures`, update.numSent, update.numFailed);\n    logger.verbose(`_PushStatus ${objectId}: needs cleanup`, { devicesToRemove });\n    ['numSent', 'numFailed'].forEach((key) => {\n      if (update[key] > 0) {\n        update[key] = {\n          __op: 'Increment',\n          amount: update[key]\n        };\n      } else {\n        delete update[key];\n      }\n    });\n\n    if (devicesToRemove.length > 0 && cleanupInstallations) {\n      logger.info(`Removing device tokens on ${devicesToRemove.length} _Installations`);\n      database.update('_Installation', { deviceToken: { '$in': devicesToRemove }}, { deviceToken: {\"__op\": \"Delete\"} }, {\n        acl: undefined,\n        many: true\n      });\n    }\n\n    // indicate this batch is complete\n    incrementOp(update, 'count', -1);\n\n    return handler.update({ objectId }, update).then((res) => {\n      if (res && res.count === 0) {\n        return this.complete();\n      }\n    })\n  }\n\n  const complete = function() {\n    return handler.update({ objectId }, {\n      status: 'succeeded',\n      count: {__op: 'Delete'}\n    });\n  }\n\n  const fail = function(err) {\n    if (typeof err === 'string') {\n      err = { message: err };\n    }\n    const update = {\n      errorMessage: err,\n      status: 'failed'\n    }\n    return handler.update({ objectId }, update);\n  }\n\n  const rval = {\n    setInitial,\n    setRunning,\n    trackSent,\n    complete,\n    fail\n  };\n\n  // define objectId to be dynamic\n  Object.defineProperty(rval, \"objectId\", {\n    get: () => objectId\n  });\n\n  return Object.freeze(rval);\n}\n"]}
|
|
328
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_cryptoUtils","require","_KeyPromiseQueue","_logger","_rest","_interopRequireDefault","_Auth","e","__esModule","default","PUSH_STATUS_COLLECTION","JOB_STATUS_COLLECTION","pushPromiseQueue","KeyPromiseQueue","jobPromiseQueue","incrementOp","object","key","amount","__op","flatten","array","flattened","i","length","Array","isArray","concat","push","statusHandler","className","database","create","then","Promise","resolve","update","where","enqueue","objectId","Object","freeze","restStatusHandler","config","auth","Auth","master","rest","response","jobStatusHandler","jobStatus","newObjectId","objectIdSize","handler","setRunning","jobName","now","Date","status","source","createdAt","ACL","setMessage","message","setSucceeded","setFinalStatus","setFailed","undefined","finishedAt","Error","pushStatusHandler","existingObjectId","pushStatus","setInitial","body","options","pushTime","toISOString","prototype","hasOwnProperty","call","hasPushScheduledSupport","push_time","logger","warn","data","payloadString","JSON","stringify","pushHash","alert","md5Hash","query","payload","title","expiry","expiration_time","expiration_interval","numSent","result","batches","verbose","count","trackSent","results","UTCOffset","cleanupInstallations","process","env","PARSE_SERVER_CLEANUP_INVALID_INSTALLATIONS","numFailed","devicesToRemove","reduce","memo","device","deviceType","transmitted","offsetKey","error","deviceToken","token","code","forEach","info","$in","acl","many","res","complete","fail","err","errorMessage","rval","defineProperty","get"],"sources":["../src/StatusHandler.js"],"sourcesContent":["import { md5Hash, newObjectId } from './cryptoUtils';\nimport { KeyPromiseQueue } from './KeyPromiseQueue';\nimport { logger } from './logger';\nimport rest from './rest';\nimport Auth from './Auth';\n\nconst PUSH_STATUS_COLLECTION = '_PushStatus';\nconst JOB_STATUS_COLLECTION = '_JobStatus';\n\nconst pushPromiseQueue = new KeyPromiseQueue();\nconst jobPromiseQueue = new KeyPromiseQueue();\n\nconst incrementOp = function (object = {}, key, amount = 1) {\n  if (!object[key]) {\n    object[key] = { __op: 'Increment', amount: amount };\n  } else {\n    object[key].amount += amount;\n  }\n  return object[key];\n};\n\nexport function flatten(array) {\n  var flattened = [];\n  for (var i = 0; i < array.length; i++) {\n    if (Array.isArray(array[i])) {\n      flattened = flattened.concat(flatten(array[i]));\n    } else {\n      flattened.push(array[i]);\n    }\n  }\n  return flattened;\n}\n\nfunction statusHandler(className, database) {\n  function create(object) {\n    return database.create(className, object).then(() => {\n      return Promise.resolve(object);\n    });\n  }\n\n  function update(where, object) {\n    return jobPromiseQueue.enqueue(where.objectId, () => database.update(className, where, object));\n  }\n\n  return Object.freeze({\n    create,\n    update,\n  });\n}\n\nfunction restStatusHandler(className, config) {\n  const auth = Auth.master(config);\n  function create(object) {\n    return rest.create(config, auth, className, object).then(({ response }) => {\n      return { ...object, ...response };\n    });\n  }\n\n  function update(where, object) {\n    return pushPromiseQueue.enqueue(where.objectId, () =>\n      rest\n        .update(config, auth, className, { objectId: where.objectId }, object)\n        .then(({ response }) => {\n          return { ...object, ...response };\n        })\n    );\n  }\n\n  return Object.freeze({\n    create,\n    update,\n  });\n}\n\nexport function jobStatusHandler(config) {\n  let jobStatus;\n  const objectId = newObjectId(config.objectIdSize);\n  const database = config.database;\n  const handler = statusHandler(JOB_STATUS_COLLECTION, database);\n  const setRunning = function (jobName) {\n    const now = new Date();\n    jobStatus = {\n      objectId,\n      jobName,\n      status: 'running',\n      source: 'api',\n      createdAt: now,\n      // lockdown!\n      ACL: {},\n    };\n\n    return handler.create(jobStatus);\n  };\n\n  const setMessage = function (message) {\n    if (!message || typeof message !== 'string') {\n      return Promise.resolve();\n    }\n    return handler.update({ objectId }, { message });\n  };\n\n  const setSucceeded = function (message) {\n    return setFinalStatus('succeeded', message);\n  };\n\n  const setFailed = function (message) {\n    return setFinalStatus('failed', message);\n  };\n\n  const setFinalStatus = function (status, message = undefined) {\n    const finishedAt = new Date();\n    const update = { status, finishedAt };\n    if (message && typeof message === 'string') {\n      update.message = message;\n    }\n    if (message instanceof Error && typeof message.message === 'string') {\n      update.message = message.message;\n    }\n    return handler.update({ objectId }, update);\n  };\n\n  return Object.freeze({\n    setRunning,\n    setSucceeded,\n    setMessage,\n    setFailed,\n  });\n}\n\nexport function pushStatusHandler(config, existingObjectId) {\n  let pushStatus;\n  const database = config.database;\n  const handler = restStatusHandler(PUSH_STATUS_COLLECTION, config);\n  let objectId = existingObjectId;\n  const setInitial = function (body = {}, where, options = { source: 'rest' }) {\n    const now = new Date();\n    let pushTime = now.toISOString();\n    let status = 'pending';\n    if (Object.prototype.hasOwnProperty.call(body, 'push_time')) {\n      if (config.hasPushScheduledSupport) {\n        pushTime = body.push_time;\n        status = 'scheduled';\n      } else {\n        logger.warn('Trying to schedule a push while server is not configured.');\n        logger.warn('Push will be sent immediately');\n      }\n    }\n\n    const data = body.data || {};\n    const payloadString = JSON.stringify(data);\n    let pushHash;\n    if (typeof data.alert === 'string') {\n      pushHash = md5Hash(data.alert);\n    } else if (typeof data.alert === 'object') {\n      pushHash = md5Hash(JSON.stringify(data.alert));\n    } else {\n      pushHash = 'd41d8cd98f00b204e9800998ecf8427e';\n    }\n    const object = {\n      pushTime,\n      query: JSON.stringify(where),\n      payload: payloadString,\n      source: options.source,\n      title: options.title,\n      expiry: body.expiration_time,\n      expiration_interval: body.expiration_interval,\n      status: status,\n      numSent: 0,\n      pushHash,\n      // lockdown!\n      ACL: {},\n    };\n    return handler.create(object).then(result => {\n      objectId = result.objectId;\n      pushStatus = {\n        objectId,\n      };\n      return Promise.resolve(pushStatus);\n    });\n  };\n\n  const setRunning = function (batches) {\n    logger.verbose(\n      `_PushStatus ${objectId}: sending push to installations with %d batches`,\n      batches\n    );\n    return handler.update(\n      {\n        status: 'pending',\n        objectId: objectId,\n      },\n      {\n        status: 'running',\n        count: batches,\n      }\n    );\n  };\n\n  const trackSent = function (\n    results,\n    UTCOffset,\n    cleanupInstallations = process.env.PARSE_SERVER_CLEANUP_INVALID_INSTALLATIONS\n  ) {\n    const update = {\n      numSent: 0,\n      numFailed: 0,\n    };\n    const devicesToRemove = [];\n    if (Array.isArray(results)) {\n      results = flatten(results);\n      results.reduce((memo, result) => {\n        // Cannot handle that\n        if (!result || !result.device || !result.device.deviceType) {\n          return memo;\n        }\n        const deviceType = result.device.deviceType;\n        const key = result.transmitted\n          ? `sentPerType.${deviceType}`\n          : `failedPerType.${deviceType}`;\n        memo[key] = incrementOp(memo, key);\n        if (typeof UTCOffset !== 'undefined') {\n          const offsetKey = result.transmitted\n            ? `sentPerUTCOffset.${UTCOffset}`\n            : `failedPerUTCOffset.${UTCOffset}`;\n          memo[offsetKey] = incrementOp(memo, offsetKey);\n        }\n        if (result.transmitted) {\n          memo.numSent++;\n        } else {\n          if (\n            result &&\n            result.response &&\n            result.response.error &&\n            result.device &&\n            result.device.deviceToken\n          ) {\n            const token = result.device.deviceToken;\n            const error = result.response.error;\n            // GCM / FCM HTTP v1 API errors; see:\n            // https://firebase.google.com/docs/reference/fcm/rest/v1/ErrorCode\n            if (error === 'NotRegistered' || error === 'InvalidRegistration') {\n              devicesToRemove.push(token);\n            }\n            // FCM API v2 errors; see:\n            // https://firebase.google.com/docs/cloud-messaging/manage-tokens\n            // https://github.com/firebase/functions-samples/blob/703c0359eacf07a551751d1319d34f912a2cd828/Node/fcm-notifications/functions/index.js#L89-L93C16\n            if (\n              error?.code === 'messaging/registration-token-not-registered' ||\n              error?.code === 'messaging/invalid-registration-token' ||\n              (error?.code === 'messaging/invalid-argument' && error?.message === 'The registration token is not a valid FCM registration token')\n            ) {\n              devicesToRemove.push(token);\n            }\n            // APNS errors; see:\n            // https://developer.apple.com/documentation/usernotifications/handling-notification-responses-from-apns\n            if (error === 'Unregistered' || error === 'BadDeviceToken') {\n              devicesToRemove.push(token);\n            }\n          }\n          memo.numFailed++;\n        }\n        return memo;\n      }, update);\n    }\n\n    logger.verbose(\n      `_PushStatus ${objectId}: sent push! %d success, %d failures`,\n      update.numSent,\n      update.numFailed\n    );\n    logger.verbose(`_PushStatus ${objectId}: needs cleanup`, {\n      devicesToRemove,\n    });\n    ['numSent', 'numFailed'].forEach(key => {\n      if (update[key] > 0) {\n        update[key] = {\n          __op: 'Increment',\n          amount: update[key],\n        };\n      } else {\n        delete update[key];\n      }\n    });\n\n    if (devicesToRemove.length > 0 && cleanupInstallations) {\n      logger.info(`Removing device tokens on ${devicesToRemove.length} _Installations`);\n      database.update(\n        '_Installation',\n        { deviceToken: { $in: devicesToRemove } },\n        { deviceToken: { __op: 'Delete' } },\n        {\n          acl: undefined,\n          many: true,\n        }\n      );\n    }\n    incrementOp(update, 'count', -1);\n    update.status = 'running';\n\n    return handler.update({ objectId }, update).then(res => {\n      if (res && res.count === 0) {\n        return this.complete();\n      }\n    });\n  };\n\n  const complete = function () {\n    return handler.update(\n      { objectId },\n      {\n        status: 'succeeded',\n        count: { __op: 'Delete' },\n      }\n    );\n  };\n\n  const fail = function (err) {\n    if (typeof err === 'string') {\n      err = { message: err };\n    }\n    const update = {\n      errorMessage: err,\n      status: 'failed',\n    };\n    return handler.update({ objectId }, update);\n  };\n\n  const rval = {\n    setInitial,\n    setRunning,\n    trackSent,\n    complete,\n    fail,\n  };\n\n  // define objectId to be dynamic\n  Object.defineProperty(rval, 'objectId', {\n    get: () => objectId,\n  });\n\n  return Object.freeze(rval);\n}\n"],"mappings":"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","ignoreList":[]}
|
package/lib/TestUtils.js
CHANGED
|
@@ -1,16 +1,16 @@
|
|
|
1
|
-
|
|
1
|
+
"use strict";
|
|
2
2
|
|
|
3
3
|
Object.defineProperty(exports, "__esModule", {
|
|
4
4
|
value: true
|
|
5
5
|
});
|
|
6
|
+
exports.Connections = void 0;
|
|
6
7
|
exports.destroyAllDataPermanently = destroyAllDataPermanently;
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
var
|
|
11
|
-
|
|
12
|
-
function _interopRequireDefault(
|
|
13
|
-
|
|
8
|
+
exports.getConnectionsCount = getConnectionsCount;
|
|
9
|
+
exports.resolvingPromise = resolvingPromise;
|
|
10
|
+
exports.sleep = sleep;
|
|
11
|
+
var _cache = _interopRequireDefault(require("./cache"));
|
|
12
|
+
var _SchemaCache = _interopRequireDefault(require("./Adapters/Cache/SchemaCache"));
|
|
13
|
+
function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
|
|
14
14
|
/**
|
|
15
15
|
* Destroys all data in the database
|
|
16
16
|
* @param {boolean} fast set to true if it's ok to just drop objects and not indexes.
|
|
@@ -19,13 +19,69 @@ function destroyAllDataPermanently(fast) {
|
|
|
19
19
|
if (!process.env.TESTING) {
|
|
20
20
|
throw 'Only supported in test environment';
|
|
21
21
|
}
|
|
22
|
-
return Promise.all(Object.keys(
|
|
23
|
-
const app =
|
|
22
|
+
return Promise.all(Object.keys(_cache.default.cache).map(appId => {
|
|
23
|
+
const app = _cache.default.get(appId);
|
|
24
|
+
const deletePromises = [];
|
|
25
|
+
if (app.cacheAdapter && app.cacheAdapter.clear) {
|
|
26
|
+
deletePromises.push(app.cacheAdapter.clear());
|
|
27
|
+
}
|
|
24
28
|
if (app.databaseController) {
|
|
25
|
-
|
|
26
|
-
} else {
|
|
27
|
-
|
|
29
|
+
deletePromises.push(app.databaseController.deleteEverything(fast));
|
|
30
|
+
} else if (app.databaseAdapter) {
|
|
31
|
+
_SchemaCache.default.clear();
|
|
32
|
+
deletePromises.push(app.databaseAdapter.deleteAllClasses(fast));
|
|
28
33
|
}
|
|
34
|
+
return Promise.all(deletePromises);
|
|
29
35
|
}));
|
|
30
36
|
}
|
|
31
|
-
|
|
37
|
+
function resolvingPromise() {
|
|
38
|
+
let res;
|
|
39
|
+
let rej;
|
|
40
|
+
const promise = new Promise((resolve, reject) => {
|
|
41
|
+
res = resolve;
|
|
42
|
+
rej = reject;
|
|
43
|
+
});
|
|
44
|
+
promise.resolve = res;
|
|
45
|
+
promise.reject = rej;
|
|
46
|
+
return promise;
|
|
47
|
+
}
|
|
48
|
+
function sleep(ms) {
|
|
49
|
+
return new Promise(resolve => setTimeout(resolve, ms));
|
|
50
|
+
}
|
|
51
|
+
function getConnectionsCount(server) {
|
|
52
|
+
return new Promise((resolve, reject) => {
|
|
53
|
+
server.getConnections((err, count) => {
|
|
54
|
+
/* istanbul ignore next */
|
|
55
|
+
if (err) {
|
|
56
|
+
reject(err);
|
|
57
|
+
} else {
|
|
58
|
+
resolve(count);
|
|
59
|
+
}
|
|
60
|
+
});
|
|
61
|
+
});
|
|
62
|
+
}
|
|
63
|
+
;
|
|
64
|
+
class Connections {
|
|
65
|
+
constructor() {
|
|
66
|
+
this.sockets = new Set();
|
|
67
|
+
}
|
|
68
|
+
track(server) {
|
|
69
|
+
server.on('connection', socket => {
|
|
70
|
+
this.sockets.add(socket);
|
|
71
|
+
socket.on('close', () => {
|
|
72
|
+
this.sockets.delete(socket);
|
|
73
|
+
});
|
|
74
|
+
});
|
|
75
|
+
}
|
|
76
|
+
destroyAll() {
|
|
77
|
+
for (const socket of this.sockets.values()) {
|
|
78
|
+
socket.destroy();
|
|
79
|
+
}
|
|
80
|
+
this.sockets.clear();
|
|
81
|
+
}
|
|
82
|
+
count() {
|
|
83
|
+
return this.sockets.size;
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
exports.Connections = Connections;
|
|
87
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJfY2FjaGUiLCJfaW50ZXJvcFJlcXVpcmVEZWZhdWx0IiwicmVxdWlyZSIsIl9TY2hlbWFDYWNoZSIsImUiLCJfX2VzTW9kdWxlIiwiZGVmYXVsdCIsImRlc3Ryb3lBbGxEYXRhUGVybWFuZW50bHkiLCJmYXN0IiwicHJvY2VzcyIsImVudiIsIlRFU1RJTkciLCJQcm9taXNlIiwiYWxsIiwiT2JqZWN0Iiwia2V5cyIsIkFwcENhY2hlIiwiY2FjaGUiLCJtYXAiLCJhcHBJZCIsImFwcCIsImdldCIsImRlbGV0ZVByb21pc2VzIiwiY2FjaGVBZGFwdGVyIiwiY2xlYXIiLCJwdXNoIiwiZGF0YWJhc2VDb250cm9sbGVyIiwiZGVsZXRlRXZlcnl0aGluZyIsImRhdGFiYXNlQWRhcHRlciIsIlNjaGVtYUNhY2hlIiwiZGVsZXRlQWxsQ2xhc3NlcyIsInJlc29sdmluZ1Byb21pc2UiLCJyZXMiLCJyZWoiLCJwcm9taXNlIiwicmVzb2x2ZSIsInJlamVjdCIsInNsZWVwIiwibXMiLCJzZXRUaW1lb3V0IiwiZ2V0Q29ubmVjdGlvbnNDb3VudCIsInNlcnZlciIsImdldENvbm5lY3Rpb25zIiwiZXJyIiwiY291bnQiLCJDb25uZWN0aW9ucyIsImNvbnN0cnVjdG9yIiwic29ja2V0cyIsIlNldCIsInRyYWNrIiwib24iLCJzb2NrZXQiLCJhZGQiLCJkZWxldGUiLCJkZXN0cm95QWxsIiwidmFsdWVzIiwiZGVzdHJveSIsInNpemUiLCJleHBvcnRzIl0sInNvdXJjZXMiOlsiLi4vc3JjL1Rlc3RVdGlscy5qcyJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgQXBwQ2FjaGUgZnJvbSAnLi9jYWNoZSc7XG5pbXBvcnQgU2NoZW1hQ2FjaGUgZnJvbSAnLi9BZGFwdGVycy9DYWNoZS9TY2hlbWFDYWNoZSc7XG5cbi8qKlxuICogRGVzdHJveXMgYWxsIGRhdGEgaW4gdGhlIGRhdGFiYXNlXG4gKiBAcGFyYW0ge2Jvb2xlYW59IGZhc3Qgc2V0IHRvIHRydWUgaWYgaXQncyBvayB0byBqdXN0IGRyb3Agb2JqZWN0cyBhbmQgbm90IGluZGV4ZXMuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBkZXN0cm95QWxsRGF0YVBlcm1hbmVudGx5KGZhc3QpIHtcbiAgaWYgKCFwcm9jZXNzLmVudi5URVNUSU5HKSB7XG4gICAgdGhyb3cgJ09ubHkgc3VwcG9ydGVkIGluIHRlc3QgZW52aXJvbm1lbnQnO1xuICB9XG4gIHJldHVybiBQcm9taXNlLmFsbChcbiAgICBPYmplY3Qua2V5cyhBcHBDYWNoZS5jYWNoZSkubWFwKGFwcElkID0+IHtcbiAgICAgIGNvbnN0IGFwcCA9IEFwcENhY2hlLmdldChhcHBJZCk7XG4gICAgICBjb25zdCBkZWxldGVQcm9taXNlcyA9IFtdO1xuICAgICAgaWYgKGFwcC5jYWNoZUFkYXB0ZXIgJiYgYXBwLmNhY2hlQWRhcHRlci5jbGVhcikge1xuICAgICAgICBkZWxldGVQcm9taXNlcy5wdXNoKGFwcC5jYWNoZUFkYXB0ZXIuY2xlYXIoKSk7XG4gICAgICB9XG4gICAgICBpZiAoYXBwLmRhdGFiYXNlQ29udHJvbGxlcikge1xuICAgICAgICBkZWxldGVQcm9taXNlcy5wdXNoKGFwcC5kYXRhYmFzZUNvbnRyb2xsZXIuZGVsZXRlRXZlcnl0aGluZyhmYXN0KSk7XG4gICAgICB9IGVsc2UgaWYgKGFwcC5kYXRhYmFzZUFkYXB0ZXIpIHtcbiAgICAgICAgU2NoZW1hQ2FjaGUuY2xlYXIoKTtcbiAgICAgICAgZGVsZXRlUHJvbWlzZXMucHVzaChhcHAuZGF0YWJhc2VBZGFwdGVyLmRlbGV0ZUFsbENsYXNzZXMoZmFzdCkpO1xuICAgICAgfVxuICAgICAgcmV0dXJuIFByb21pc2UuYWxsKGRlbGV0ZVByb21pc2VzKTtcbiAgICB9KVxuICApO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gcmVzb2x2aW5nUHJvbWlzZSgpIHtcbiAgbGV0IHJlcztcbiAgbGV0IHJlajtcbiAgY29uc3QgcHJvbWlzZSA9IG5ldyBQcm9taXNlKChyZXNvbHZlLCByZWplY3QpID0+IHtcbiAgICByZXMgPSByZXNvbHZlO1xuICAgIHJlaiA9IHJlamVjdDtcbiAgfSk7XG4gIHByb21pc2UucmVzb2x2ZSA9IHJlcztcbiAgcHJvbWlzZS5yZWplY3QgPSByZWo7XG4gIHJldHVybiBwcm9taXNlO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gc2xlZXAobXMpIHtcbiAgcmV0dXJuIG5ldyBQcm9taXNlKChyZXNvbHZlKSA9PiBzZXRUaW1lb3V0KHJlc29sdmUsIG1zKSk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBnZXRDb25uZWN0aW9uc0NvdW50KHNlcnZlcikge1xuICByZXR1cm4gbmV3IFByb21pc2UoKHJlc29sdmUsIHJlamVjdCkgPT4ge1xuICAgIHNlcnZlci5nZXRDb25uZWN0aW9ucygoZXJyLCBjb3VudCkgPT4ge1xuICAgICAgLyogaXN0YW5idWwgaWdub3JlIG5leHQgKi9cbiAgICAgIGlmIChlcnIpIHtcbiAgICAgICAgcmVqZWN0KGVycik7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICByZXNvbHZlKGNvdW50KTtcbiAgICAgIH1cbiAgICB9KTtcbiAgfSk7XG59O1xuXG5leHBvcnQgY2xhc3MgQ29ubmVjdGlvbnMge1xuICBjb25zdHJ1Y3RvcigpIHtcbiAgICB0aGlzLnNvY2tldHMgPSBuZXcgU2V0KCk7XG4gIH1cblxuICB0cmFjayhzZXJ2ZXIpIHtcbiAgICBzZXJ2ZXIub24oJ2Nvbm5lY3Rpb24nLCBzb2NrZXQgPT4ge1xuICAgICAgdGhpcy5zb2NrZXRzLmFkZChzb2NrZXQpO1xuICAgICAgc29ja2V0Lm9uKCdjbG9zZScsICgpID0+IHtcbiAgICAgICAgdGhpcy5zb2NrZXRzLmRlbGV0ZShzb2NrZXQpO1xuICAgICAgfSk7XG4gICAgfSk7XG4gIH1cblxuICBkZXN0cm95QWxsKCkge1xuICAgIGZvciAoY29uc3Qgc29ja2V0IG9mIHRoaXMuc29ja2V0cy52YWx1ZXMoKSkge1xuICAgICAgc29ja2V0LmRlc3Ryb3koKTtcbiAgICB9XG4gICAgdGhpcy5zb2NrZXRzLmNsZWFyKCk7XG4gIH1c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
|