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
|
@@ -1,30 +1,17 @@
|
|
|
1
|
-
|
|
1
|
+
"use strict";
|
|
2
2
|
|
|
3
3
|
Object.defineProperty(exports, "__esModule", {
|
|
4
4
|
value: true
|
|
5
5
|
});
|
|
6
|
-
exports.PushController =
|
|
7
|
-
|
|
8
|
-
var
|
|
9
|
-
|
|
10
|
-
var
|
|
11
|
-
|
|
12
|
-
var
|
|
13
|
-
|
|
14
|
-
var _RestWrite = require('../RestWrite');
|
|
15
|
-
|
|
16
|
-
var _RestWrite2 = _interopRequireDefault(_RestWrite);
|
|
17
|
-
|
|
18
|
-
var _Auth = require('../Auth');
|
|
19
|
-
|
|
20
|
-
var _StatusHandler = require('../StatusHandler');
|
|
21
|
-
|
|
22
|
-
var _utils = require('../Push/utils');
|
|
23
|
-
|
|
24
|
-
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
25
|
-
|
|
6
|
+
exports.default = exports.PushController = void 0;
|
|
7
|
+
var _node = require("parse/node");
|
|
8
|
+
var _RestQuery = _interopRequireDefault(require("../RestQuery"));
|
|
9
|
+
var _RestWrite = _interopRequireDefault(require("../RestWrite"));
|
|
10
|
+
var _Auth = require("../Auth");
|
|
11
|
+
var _StatusHandler = require("../StatusHandler");
|
|
12
|
+
var _utils = require("../Push/utils");
|
|
13
|
+
function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
|
|
26
14
|
class PushController {
|
|
27
|
-
|
|
28
15
|
sendPush(body = {}, where = {}, config, auth, onPushStatusSaved = () => {}, now = new Date()) {
|
|
29
16
|
if (!config.hasPushSupport) {
|
|
30
17
|
throw new _node.Parse.Error(_node.Parse.Error.PUSH_MISCONFIGURED, 'Missing push configuration');
|
|
@@ -38,11 +25,10 @@ class PushController {
|
|
|
38
25
|
}
|
|
39
26
|
|
|
40
27
|
// Immediate push
|
|
41
|
-
if (body.expiration_interval && !
|
|
28
|
+
if (body.expiration_interval && !Object.prototype.hasOwnProperty.call(body, 'push_time')) {
|
|
42
29
|
const ttlMs = body.expiration_interval * 1000;
|
|
43
30
|
body.expiration_time = new Date(now.valueOf() + ttlMs).valueOf();
|
|
44
31
|
}
|
|
45
|
-
|
|
46
32
|
const pushTime = PushController.getPushTime(body);
|
|
47
33
|
if (pushTime && pushTime.date !== 'undefined') {
|
|
48
34
|
body['push_time'] = PushController.formatPushTime(pushTime);
|
|
@@ -53,27 +39,45 @@ class PushController {
|
|
|
53
39
|
let badgeUpdate = () => {
|
|
54
40
|
return Promise.resolve();
|
|
55
41
|
};
|
|
56
|
-
|
|
57
42
|
if (body.data && body.data.badge) {
|
|
58
43
|
const badge = body.data.badge;
|
|
59
44
|
let restUpdate = {};
|
|
60
45
|
if (typeof badge == 'string' && badge.toLowerCase() === 'increment') {
|
|
61
|
-
restUpdate = {
|
|
46
|
+
restUpdate = {
|
|
47
|
+
badge: {
|
|
48
|
+
__op: 'Increment',
|
|
49
|
+
amount: 1
|
|
50
|
+
}
|
|
51
|
+
};
|
|
62
52
|
} else if (typeof badge == 'object' && typeof badge.__op == 'string' && badge.__op.toLowerCase() == 'increment' && Number(badge.amount)) {
|
|
63
|
-
restUpdate = {
|
|
53
|
+
restUpdate = {
|
|
54
|
+
badge: {
|
|
55
|
+
__op: 'Increment',
|
|
56
|
+
amount: badge.amount
|
|
57
|
+
}
|
|
58
|
+
};
|
|
64
59
|
} else if (Number(badge)) {
|
|
65
|
-
restUpdate = {
|
|
60
|
+
restUpdate = {
|
|
61
|
+
badge: badge
|
|
62
|
+
};
|
|
66
63
|
} else {
|
|
67
64
|
throw "Invalid value for badge, expected number or 'Increment' or {increment: number}";
|
|
68
65
|
}
|
|
69
66
|
|
|
70
67
|
// Force filtering on only valid device tokens
|
|
71
68
|
const updateWhere = (0, _utils.applyDeviceTokenExists)(where);
|
|
72
|
-
badgeUpdate = () => {
|
|
69
|
+
badgeUpdate = async () => {
|
|
73
70
|
// Build a real RestQuery so we can use it in RestWrite
|
|
74
|
-
const restQuery =
|
|
71
|
+
const restQuery = await (0, _RestQuery.default)({
|
|
72
|
+
method: _RestQuery.default.Method.find,
|
|
73
|
+
config,
|
|
74
|
+
runBeforeFind: false,
|
|
75
|
+
auth: (0, _Auth.master)(config),
|
|
76
|
+
className: '_Installation',
|
|
77
|
+
restWhere: updateWhere
|
|
78
|
+
});
|
|
75
79
|
return restQuery.buildRestWhere().then(() => {
|
|
76
|
-
const write = new
|
|
80
|
+
const write = new _RestWrite.default(config, (0, _Auth.master)(config), '_Installation', restQuery.restWhere, restUpdate);
|
|
77
81
|
write.runOptions.many = true;
|
|
78
82
|
return write.execute();
|
|
79
83
|
});
|
|
@@ -89,18 +93,25 @@ class PushController {
|
|
|
89
93
|
// Update audience lastUsed and timesUsed
|
|
90
94
|
if (body.audience_id) {
|
|
91
95
|
const audienceId = body.audience_id;
|
|
92
|
-
|
|
93
96
|
var updateAudience = {
|
|
94
|
-
lastUsed: {
|
|
95
|
-
|
|
97
|
+
lastUsed: {
|
|
98
|
+
__type: 'Date',
|
|
99
|
+
iso: new Date().toISOString()
|
|
100
|
+
},
|
|
101
|
+
timesUsed: {
|
|
102
|
+
__op: 'Increment',
|
|
103
|
+
amount: 1
|
|
104
|
+
}
|
|
96
105
|
};
|
|
97
|
-
const write = new
|
|
106
|
+
const write = new _RestWrite.default(config, (0, _Auth.master)(config), '_Audience', {
|
|
107
|
+
objectId: audienceId
|
|
108
|
+
}, updateAudience);
|
|
98
109
|
write.execute();
|
|
99
110
|
}
|
|
100
111
|
// Don't wait for the audience update promise to resolve.
|
|
101
112
|
return Promise.resolve();
|
|
102
113
|
}).then(() => {
|
|
103
|
-
if (
|
|
114
|
+
if (Object.prototype.hasOwnProperty.call(body, 'push_time') && config.hasPushScheduledSupport) {
|
|
104
115
|
return Promise.resolve();
|
|
105
116
|
}
|
|
106
117
|
return config.pushControllerQueue.enqueue(body, where, config, auth, pushStatus);
|
|
@@ -117,7 +128,7 @@ class PushController {
|
|
|
117
128
|
* @returns {Number|undefined} The expiration time if it exists in the request
|
|
118
129
|
*/
|
|
119
130
|
static getExpirationTime(body = {}) {
|
|
120
|
-
var hasExpirationTime =
|
|
131
|
+
var hasExpirationTime = Object.prototype.hasOwnProperty.call(body, 'expiration_time');
|
|
121
132
|
if (!hasExpirationTime) {
|
|
122
133
|
return;
|
|
123
134
|
}
|
|
@@ -136,13 +147,11 @@ class PushController {
|
|
|
136
147
|
}
|
|
137
148
|
return expirationTime.valueOf();
|
|
138
149
|
}
|
|
139
|
-
|
|
140
150
|
static getExpirationInterval(body = {}) {
|
|
141
|
-
const hasExpirationInterval =
|
|
151
|
+
const hasExpirationInterval = Object.prototype.hasOwnProperty.call(body, 'expiration_interval');
|
|
142
152
|
if (!hasExpirationInterval) {
|
|
143
153
|
return;
|
|
144
154
|
}
|
|
145
|
-
|
|
146
155
|
var expirationIntervalParam = body['expiration_interval'];
|
|
147
156
|
if (typeof expirationIntervalParam !== 'number' || expirationIntervalParam <= 0) {
|
|
148
157
|
throw new _node.Parse.Error(_node.Parse.Error.PUSH_MISCONFIGURED, `expiration_interval must be a number greater than 0`);
|
|
@@ -156,14 +165,13 @@ class PushController {
|
|
|
156
165
|
* @returns {Number|undefined} The push time if it exists in the request
|
|
157
166
|
*/
|
|
158
167
|
static getPushTime(body = {}) {
|
|
159
|
-
var hasPushTime =
|
|
168
|
+
var hasPushTime = Object.prototype.hasOwnProperty.call(body, 'push_time');
|
|
160
169
|
if (!hasPushTime) {
|
|
161
170
|
return;
|
|
162
171
|
}
|
|
163
172
|
var pushTimeParam = body['push_time'];
|
|
164
173
|
var date;
|
|
165
174
|
var isLocalTime = true;
|
|
166
|
-
|
|
167
175
|
if (typeof pushTimeParam === 'number') {
|
|
168
176
|
date = new Date(pushTimeParam * 1000);
|
|
169
177
|
} else if (typeof pushTimeParam === 'string') {
|
|
@@ -176,7 +184,6 @@ class PushController {
|
|
|
176
184
|
if (!isFinite(date)) {
|
|
177
185
|
throw new _node.Parse.Error(_node.Parse.Error.PUSH_MISCONFIGURED, body['push_time'] + ' is not valid time.');
|
|
178
186
|
}
|
|
179
|
-
|
|
180
187
|
return {
|
|
181
188
|
date,
|
|
182
189
|
isLocalTime
|
|
@@ -190,8 +197,8 @@ class PushController {
|
|
|
190
197
|
*/
|
|
191
198
|
static pushTimeHasTimezoneComponent(pushTimeParam) {
|
|
192
199
|
const offsetPattern = /(.+)([+-])\d\d:\d\d$/;
|
|
193
|
-
return pushTimeParam.indexOf('Z') === pushTimeParam.length - 1 // 2007-04-05T12:30Z
|
|
194
|
-
|
|
200
|
+
return pushTimeParam.indexOf('Z') === pushTimeParam.length - 1 || offsetPattern.test(pushTimeParam) // 2007-04-05T12:30Z
|
|
201
|
+
; // 2007-04-05T12:30.000+02:00, 2007-04-05T12:30.000-02:00
|
|
195
202
|
}
|
|
196
203
|
|
|
197
204
|
/**
|
|
@@ -200,7 +207,10 @@ class PushController {
|
|
|
200
207
|
* @param isLocalTime {boolean}
|
|
201
208
|
* @returns {string}
|
|
202
209
|
*/
|
|
203
|
-
static formatPushTime({
|
|
210
|
+
static formatPushTime({
|
|
211
|
+
date,
|
|
212
|
+
isLocalTime
|
|
213
|
+
}) {
|
|
204
214
|
if (isLocalTime) {
|
|
205
215
|
// Strip 'Z'
|
|
206
216
|
const isoString = date.toISOString();
|
|
@@ -209,7 +219,6 @@ class PushController {
|
|
|
209
219
|
return date.toISOString();
|
|
210
220
|
}
|
|
211
221
|
}
|
|
212
|
-
|
|
213
222
|
exports.PushController = PushController;
|
|
214
|
-
exports.default = PushController;
|
|
215
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/Controllers/PushController.js"],"names":["PushController","sendPush","body","where","config","auth","onPushStatusSaved","now","Date","hasPushSupport","Parse","Error","PUSH_MISCONFIGURED","expiration_time","getExpirationTime","expiration_interval","getExpirationInterval","hasOwnProperty","ttlMs","valueOf","pushTime","getPushTime","date","formatPushTime","badgeUpdate","Promise","resolve","data","badge","restUpdate","toLowerCase","__op","amount","Number","updateWhere","restQuery","RestQuery","buildRestWhere","then","write","RestWrite","restWhere","runOptions","many","execute","pushStatus","setInitial","objectId","audience_id","audienceId","updateAudience","lastUsed","__type","iso","toISOString","timesUsed","hasPushScheduledSupport","pushControllerQueue","enqueue","catch","err","fail","hasExpirationTime","expirationTimeParam","expirationTime","isFinite","hasExpirationInterval","expirationIntervalParam","hasPushTime","pushTimeParam","isLocalTime","pushTimeHasTimezoneComponent","offsetPattern","indexOf","length","test","isoString","substring"],"mappings":"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","file":"PushController.js","sourcesContent":["import { Parse }              from 'parse/node';\nimport RestQuery              from '../RestQuery';\nimport RestWrite              from '../RestWrite';\nimport { master }             from '../Auth';\nimport { pushStatusHandler }  from '../StatusHandler';\nimport { applyDeviceTokenExists } from '../Push/utils';\n\nexport class PushController {\n\n  sendPush(body = {}, where = {}, config, auth, onPushStatusSaved = () => {}, now = new Date()) {\n    if (!config.hasPushSupport) {\n      throw new Parse.Error(Parse.Error.PUSH_MISCONFIGURED,\n        'Missing push configuration');\n    }\n\n    // Replace the expiration_time and push_time with a valid Unix epoch milliseconds time\n    body.expiration_time = PushController.getExpirationTime(body);\n    body.expiration_interval = PushController.getExpirationInterval(body);\n    if (body.expiration_time && body.expiration_interval) {\n      throw new Parse.Error(\n        Parse.Error.PUSH_MISCONFIGURED,\n        'Both expiration_time and expiration_interval cannot be set');\n    }\n\n    // Immediate push\n    if (body.expiration_interval && !body.hasOwnProperty('push_time')) {\n      const ttlMs = body.expiration_interval * 1000;\n      body.expiration_time = (new Date(now.valueOf() + ttlMs)).valueOf();\n    }\n\n    const pushTime = PushController.getPushTime(body);\n    if (pushTime && pushTime.date !== 'undefined') {\n      body['push_time'] = PushController.formatPushTime(pushTime);\n    }\n\n    // TODO: If the req can pass the checking, we return immediately instead of waiting\n    // pushes to be sent. We probably change this behaviour in the future.\n    let badgeUpdate = () => {\n      return Promise.resolve();\n    }\n\n    if (body.data && body.data.badge) {\n      const badge = body.data.badge;\n      let restUpdate = {};\n      if (typeof badge == 'string' && badge.toLowerCase() === 'increment') {\n        restUpdate = { badge: { __op: 'Increment', amount: 1 } }\n      } else if (typeof badge == 'object' && typeof badge.__op == 'string' &&\n                 badge.__op.toLowerCase() == 'increment' && Number(badge.amount)) {\n        restUpdate = { badge: { __op: 'Increment', amount: badge.amount } }\n      } else if (Number(badge)) {\n        restUpdate = { badge: badge }\n      } else {\n        throw \"Invalid value for badge, expected number or 'Increment' or {increment: number}\";\n      }\n\n      // Force filtering on only valid device tokens\n      const updateWhere = applyDeviceTokenExists(where);\n      badgeUpdate = () => {\n        // Build a real RestQuery so we can use it in RestWrite\n        const restQuery = new RestQuery(config, master(config), '_Installation', updateWhere);\n        return restQuery.buildRestWhere().then(() => {\n          const write = new RestWrite(config, master(config), '_Installation', restQuery.restWhere, restUpdate);\n          write.runOptions.many = true;\n          return write.execute();\n        });\n      }\n    }\n    const pushStatus = pushStatusHandler(config);\n    return Promise.resolve().then(() => {\n      return pushStatus.setInitial(body, where);\n    }).then(() => {\n      onPushStatusSaved(pushStatus.objectId);\n      return badgeUpdate();\n    }).then(() => {\n      // Update audience lastUsed and timesUsed\n      if (body.audience_id) {\n        const audienceId = body.audience_id;\n\n        var updateAudience = {\n          lastUsed: { __type: \"Date\", iso: new Date().toISOString() },\n          timesUsed: { __op: \"Increment\", \"amount\": 1 }\n        };\n        const write = new RestWrite(config, master(config), '_Audience', {objectId: audienceId}, updateAudience);\n        write.execute();\n      }\n      // Don't wait for the audience update promise to resolve.\n      return Promise.resolve();\n    }).then(() => {\n      if (body.hasOwnProperty('push_time') && config.hasPushScheduledSupport) {\n        return Promise.resolve();\n      }\n      return config.pushControllerQueue.enqueue(body, where, config, auth, pushStatus);\n    }).catch((err) => {\n      return pushStatus.fail(err).then(() => {\n        throw err;\n      });\n    });\n  }\n\n  /**\n   * Get expiration time from the request body.\n   * @param {Object} request A request object\n   * @returns {Number|undefined} The expiration time if it exists in the request\n   */\n  static getExpirationTime(body = {}) {\n    var hasExpirationTime = body.hasOwnProperty('expiration_time');\n    if (!hasExpirationTime) {\n      return;\n    }\n    var expirationTimeParam = body['expiration_time'];\n    var expirationTime;\n    if (typeof expirationTimeParam === 'number') {\n      expirationTime = new Date(expirationTimeParam * 1000);\n    } else if (typeof expirationTimeParam === 'string') {\n      expirationTime = new Date(expirationTimeParam);\n    } else {\n      throw new Parse.Error(Parse.Error.PUSH_MISCONFIGURED,\n        body['expiration_time'] + ' is not valid time.');\n    }\n    // Check expirationTime is valid or not, if it is not valid, expirationTime is NaN\n    if (!isFinite(expirationTime)) {\n      throw new Parse.Error(Parse.Error.PUSH_MISCONFIGURED,\n        body['expiration_time'] + ' is not valid time.');\n    }\n    return expirationTime.valueOf();\n  }\n\n  static getExpirationInterval(body = {}) {\n    const hasExpirationInterval = body.hasOwnProperty('expiration_interval');\n    if (!hasExpirationInterval) {\n      return;\n    }\n\n    var expirationIntervalParam = body['expiration_interval'];\n    if (typeof expirationIntervalParam !== 'number' || expirationIntervalParam <= 0) {\n      throw new Parse.Error(Parse.Error.PUSH_MISCONFIGURED,\n        `expiration_interval must be a number greater than 0`);\n    }\n    return expirationIntervalParam;\n  }\n\n  /**\n   * Get push time from the request body.\n   * @param {Object} request A request object\n   * @returns {Number|undefined} The push time if it exists in the request\n   */\n  static getPushTime(body = {}) {\n    var hasPushTime = body.hasOwnProperty('push_time');\n    if (!hasPushTime) {\n      return;\n    }\n    var pushTimeParam = body['push_time'];\n    var date;\n    var isLocalTime = true;\n\n    if (typeof pushTimeParam === 'number') {\n      date = new Date(pushTimeParam * 1000);\n    } else if (typeof pushTimeParam === 'string') {\n      isLocalTime = !PushController.pushTimeHasTimezoneComponent(pushTimeParam);\n      date = new Date(pushTimeParam);\n    } else {\n      throw new Parse.Error(Parse.Error.PUSH_MISCONFIGURED,\n        body['push_time'] + ' is not valid time.');\n    }\n    // Check pushTime is valid or not, if it is not valid, pushTime is NaN\n    if (!isFinite(date)) {\n      throw new Parse.Error(Parse.Error.PUSH_MISCONFIGURED,\n        body['push_time'] + ' is not valid time.');\n    }\n\n    return {\n      date,\n      isLocalTime,\n    };\n  }\n\n  /**\n   * Checks if a ISO8601 formatted date contains a timezone component\n   * @param pushTimeParam {string}\n   * @returns {boolean}\n   */\n  static pushTimeHasTimezoneComponent(pushTimeParam: string): boolean {\n    const offsetPattern = /(.+)([+-])\\d\\d:\\d\\d$/;\n    return pushTimeParam.indexOf('Z') === pushTimeParam.length - 1 // 2007-04-05T12:30Z\n      || offsetPattern.test(pushTimeParam); // 2007-04-05T12:30.000+02:00, 2007-04-05T12:30.000-02:00\n  }\n\n  /**\n   * Converts a date to ISO format in UTC time and strips the timezone if `isLocalTime` is true\n   * @param date {Date}\n   * @param isLocalTime {boolean}\n   * @returns {string}\n   */\n  static formatPushTime({ date, isLocalTime }: { date: Date, isLocalTime: boolean }) {\n    if (isLocalTime) { // Strip 'Z'\n      const isoString = date.toISOString();\n      return isoString.substring(0, isoString.indexOf('Z'));\n    }\n    return date.toISOString();\n  }\n}\n\nexport default PushController;\n"]}
|
|
223
|
+
var _default = exports.default = PushController;
|
|
224
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_node","require","_RestQuery","_interopRequireDefault","_RestWrite","_Auth","_StatusHandler","_utils","e","__esModule","default","PushController","sendPush","body","where","config","auth","onPushStatusSaved","now","Date","hasPushSupport","Parse","Error","PUSH_MISCONFIGURED","expiration_time","getExpirationTime","expiration_interval","getExpirationInterval","Object","prototype","hasOwnProperty","call","ttlMs","valueOf","pushTime","getPushTime","date","formatPushTime","badgeUpdate","Promise","resolve","data","badge","restUpdate","toLowerCase","__op","amount","Number","updateWhere","applyDeviceTokenExists","restQuery","RestQuery","method","Method","find","runBeforeFind","master","className","restWhere","buildRestWhere","then","write","RestWrite","runOptions","many","execute","pushStatus","pushStatusHandler","setInitial","objectId","audience_id","audienceId","updateAudience","lastUsed","__type","iso","toISOString","timesUsed","hasPushScheduledSupport","pushControllerQueue","enqueue","catch","err","fail","hasExpirationTime","expirationTimeParam","expirationTime","isFinite","hasExpirationInterval","expirationIntervalParam","hasPushTime","pushTimeParam","isLocalTime","pushTimeHasTimezoneComponent","offsetPattern","indexOf","length","test","isoString","substring","exports","_default"],"sources":["../../src/Controllers/PushController.js"],"sourcesContent":["import { Parse } from 'parse/node';\nimport RestQuery from '../RestQuery';\nimport RestWrite from '../RestWrite';\nimport { master } from '../Auth';\nimport { pushStatusHandler } from '../StatusHandler';\nimport { applyDeviceTokenExists } from '../Push/utils';\n\nexport class PushController {\n  sendPush(body = {}, where = {}, config, auth, onPushStatusSaved = () => {}, now = new Date()) {\n    if (!config.hasPushSupport) {\n      throw new Parse.Error(Parse.Error.PUSH_MISCONFIGURED, 'Missing push configuration');\n    }\n\n    // Replace the expiration_time and push_time with a valid Unix epoch milliseconds time\n    body.expiration_time = PushController.getExpirationTime(body);\n    body.expiration_interval = PushController.getExpirationInterval(body);\n    if (body.expiration_time && body.expiration_interval) {\n      throw new Parse.Error(\n        Parse.Error.PUSH_MISCONFIGURED,\n        'Both expiration_time and expiration_interval cannot be set'\n      );\n    }\n\n    // Immediate push\n    if (body.expiration_interval && !Object.prototype.hasOwnProperty.call(body, 'push_time')) {\n      const ttlMs = body.expiration_interval * 1000;\n      body.expiration_time = new Date(now.valueOf() + ttlMs).valueOf();\n    }\n\n    const pushTime = PushController.getPushTime(body);\n    if (pushTime && pushTime.date !== 'undefined') {\n      body['push_time'] = PushController.formatPushTime(pushTime);\n    }\n\n    // TODO: If the req can pass the checking, we return immediately instead of waiting\n    // pushes to be sent. We probably change this behaviour in the future.\n    let badgeUpdate = () => {\n      return Promise.resolve();\n    };\n\n    if (body.data && body.data.badge) {\n      const badge = body.data.badge;\n      let restUpdate = {};\n      if (typeof badge == 'string' && badge.toLowerCase() === 'increment') {\n        restUpdate = { badge: { __op: 'Increment', amount: 1 } };\n      } else if (\n        typeof badge == 'object' &&\n        typeof badge.__op == 'string' &&\n        badge.__op.toLowerCase() == 'increment' &&\n        Number(badge.amount)\n      ) {\n        restUpdate = { badge: { __op: 'Increment', amount: badge.amount } };\n      } else if (Number(badge)) {\n        restUpdate = { badge: badge };\n      } else {\n        throw \"Invalid value for badge, expected number or 'Increment' or {increment: number}\";\n      }\n\n      // Force filtering on only valid device tokens\n      const updateWhere = applyDeviceTokenExists(where);\n      badgeUpdate = async () => {\n        // Build a real RestQuery so we can use it in RestWrite\n        const restQuery = await RestQuery({\n          method: RestQuery.Method.find,\n          config,\n          runBeforeFind: false,\n          auth: master(config),\n          className: '_Installation',\n          restWhere: updateWhere,\n        });\n        return restQuery.buildRestWhere().then(() => {\n          const write = new RestWrite(\n            config,\n            master(config),\n            '_Installation',\n            restQuery.restWhere,\n            restUpdate\n          );\n          write.runOptions.many = true;\n          return write.execute();\n        });\n      };\n    }\n    const pushStatus = pushStatusHandler(config);\n    return Promise.resolve()\n      .then(() => {\n        return pushStatus.setInitial(body, where);\n      })\n      .then(() => {\n        onPushStatusSaved(pushStatus.objectId);\n        return badgeUpdate();\n      })\n      .then(() => {\n        // Update audience lastUsed and timesUsed\n        if (body.audience_id) {\n          const audienceId = body.audience_id;\n\n          var updateAudience = {\n            lastUsed: { __type: 'Date', iso: new Date().toISOString() },\n            timesUsed: { __op: 'Increment', amount: 1 },\n          };\n          const write = new RestWrite(\n            config,\n            master(config),\n            '_Audience',\n            { objectId: audienceId },\n            updateAudience\n          );\n          write.execute();\n        }\n        // Don't wait for the audience update promise to resolve.\n        return Promise.resolve();\n      })\n      .then(() => {\n        if (\n          Object.prototype.hasOwnProperty.call(body, 'push_time') &&\n          config.hasPushScheduledSupport\n        ) {\n          return Promise.resolve();\n        }\n        return config.pushControllerQueue.enqueue(body, where, config, auth, pushStatus);\n      })\n      .catch(err => {\n        return pushStatus.fail(err).then(() => {\n          throw err;\n        });\n      });\n  }\n\n  /**\n   * Get expiration time from the request body.\n   * @param {Object} request A request object\n   * @returns {Number|undefined} The expiration time if it exists in the request\n   */\n  static getExpirationTime(body = {}) {\n    var hasExpirationTime = Object.prototype.hasOwnProperty.call(body, 'expiration_time');\n    if (!hasExpirationTime) {\n      return;\n    }\n    var expirationTimeParam = body['expiration_time'];\n    var expirationTime;\n    if (typeof expirationTimeParam === 'number') {\n      expirationTime = new Date(expirationTimeParam * 1000);\n    } else if (typeof expirationTimeParam === 'string') {\n      expirationTime = new Date(expirationTimeParam);\n    } else {\n      throw new Parse.Error(\n        Parse.Error.PUSH_MISCONFIGURED,\n        body['expiration_time'] + ' is not valid time.'\n      );\n    }\n    // Check expirationTime is valid or not, if it is not valid, expirationTime is NaN\n    if (!isFinite(expirationTime)) {\n      throw new Parse.Error(\n        Parse.Error.PUSH_MISCONFIGURED,\n        body['expiration_time'] + ' is not valid time.'\n      );\n    }\n    return expirationTime.valueOf();\n  }\n\n  static getExpirationInterval(body = {}) {\n    const hasExpirationInterval = Object.prototype.hasOwnProperty.call(body, 'expiration_interval');\n    if (!hasExpirationInterval) {\n      return;\n    }\n\n    var expirationIntervalParam = body['expiration_interval'];\n    if (typeof expirationIntervalParam !== 'number' || expirationIntervalParam <= 0) {\n      throw new Parse.Error(\n        Parse.Error.PUSH_MISCONFIGURED,\n        `expiration_interval must be a number greater than 0`\n      );\n    }\n    return expirationIntervalParam;\n  }\n\n  /**\n   * Get push time from the request body.\n   * @param {Object} request A request object\n   * @returns {Number|undefined} The push time if it exists in the request\n   */\n  static getPushTime(body = {}) {\n    var hasPushTime = Object.prototype.hasOwnProperty.call(body, 'push_time');\n    if (!hasPushTime) {\n      return;\n    }\n    var pushTimeParam = body['push_time'];\n    var date;\n    var isLocalTime = true;\n\n    if (typeof pushTimeParam === 'number') {\n      date = new Date(pushTimeParam * 1000);\n    } else if (typeof pushTimeParam === 'string') {\n      isLocalTime = !PushController.pushTimeHasTimezoneComponent(pushTimeParam);\n      date = new Date(pushTimeParam);\n    } else {\n      throw new Parse.Error(\n        Parse.Error.PUSH_MISCONFIGURED,\n        body['push_time'] + ' is not valid time.'\n      );\n    }\n    // Check pushTime is valid or not, if it is not valid, pushTime is NaN\n    if (!isFinite(date)) {\n      throw new Parse.Error(\n        Parse.Error.PUSH_MISCONFIGURED,\n        body['push_time'] + ' is not valid time.'\n      );\n    }\n\n    return {\n      date,\n      isLocalTime,\n    };\n  }\n\n  /**\n   * Checks if a ISO8601 formatted date contains a timezone component\n   * @param pushTimeParam {string}\n   * @returns {boolean}\n   */\n  static pushTimeHasTimezoneComponent(pushTimeParam: string): boolean {\n    const offsetPattern = /(.+)([+-])\\d\\d:\\d\\d$/;\n    return (\n      pushTimeParam.indexOf('Z') === pushTimeParam.length - 1 || offsetPattern.test(pushTimeParam) // 2007-04-05T12:30Z\n    ); // 2007-04-05T12:30.000+02:00, 2007-04-05T12:30.000-02:00\n  }\n\n  /**\n   * Converts a date to ISO format in UTC time and strips the timezone if `isLocalTime` is true\n   * @param date {Date}\n   * @param isLocalTime {boolean}\n   * @returns {string}\n   */\n  static formatPushTime({ date, isLocalTime }: { date: Date, isLocalTime: boolean }) {\n    if (isLocalTime) {\n      // Strip 'Z'\n      const isoString = date.toISOString();\n      return isoString.substring(0, isoString.indexOf('Z'));\n    }\n    return date.toISOString();\n  }\n}\n\nexport default PushController;\n"],"mappings":"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","ignoreList":[]}
|