parse-server 6.0.0-alpha.2 → 6.0.0-alpha.21
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +45 -17
- package/lib/AccountLockout.js +11 -26
- package/lib/Adapters/AdapterLoader.js +8 -14
- package/lib/Adapters/Analytics/AnalyticsAdapter.js +2 -8
- package/lib/Adapters/Auth/AuthAdapter.js +7 -16
- package/lib/Adapters/Auth/OAuth1Client.js +32 -57
- package/lib/Adapters/Auth/apple.js +6 -22
- package/lib/Adapters/Auth/facebook.js +7 -37
- package/lib/Adapters/Auth/gcenter.js +8 -37
- package/lib/Adapters/Auth/github.js +7 -10
- package/lib/Adapters/Auth/google.js +11 -34
- package/lib/Adapters/Auth/gpgames.js +5 -8
- package/lib/Adapters/Auth/httpsRequest.js +1 -7
- package/lib/Adapters/Auth/index.js +20 -65
- package/lib/Adapters/Auth/instagram.js +5 -9
- package/lib/Adapters/Auth/janraincapture.js +8 -12
- package/lib/Adapters/Auth/janrainengage.js +7 -11
- package/lib/Adapters/Auth/keycloak.js +5 -19
- package/lib/Adapters/Auth/ldap.js +1 -15
- package/lib/Adapters/Auth/line.js +7 -10
- package/lib/Adapters/Auth/linkedin.js +7 -12
- package/lib/Adapters/Auth/meetup.js +7 -10
- package/lib/Adapters/Auth/microsoft.js +7 -10
- package/lib/Adapters/Auth/oauth2.js +6 -18
- package/lib/Adapters/Auth/phantauth.js +8 -10
- package/lib/Adapters/Auth/qq.js +7 -13
- package/lib/Adapters/Auth/spotify.js +7 -14
- package/lib/Adapters/Auth/twitter.js +5 -15
- package/lib/Adapters/Auth/vkontakte.js +9 -15
- package/lib/Adapters/Auth/wechat.js +7 -10
- package/lib/Adapters/Auth/weibo.js +7 -11
- package/lib/Adapters/Cache/CacheAdapter.js +4 -12
- package/lib/Adapters/Cache/InMemoryCache.js +5 -19
- package/lib/Adapters/Cache/InMemoryCacheAdapter.js +1 -11
- package/lib/Adapters/Cache/LRUCache.js +1 -11
- package/lib/Adapters/Cache/NullCacheAdapter.js +1 -8
- package/lib/Adapters/Cache/RedisCacheAdapter.js +46 -87
- package/lib/Adapters/Cache/SchemaCache.js +1 -6
- package/lib/Adapters/Email/MailAdapter.js +2 -7
- package/lib/Adapters/Files/FilesAdapter.js +7 -21
- package/lib/Adapters/Files/GridFSBucketAdapter.js +6 -44
- package/lib/Adapters/Files/GridStoreAdapter.js +1 -1
- package/lib/Adapters/Logger/LoggerAdapter.js +2 -11
- package/lib/Adapters/Logger/WinstonLogger.js +3 -30
- package/lib/Adapters/Logger/WinstonLoggerAdapter.js +5 -16
- package/lib/Adapters/MessageQueue/EventEmitterMQ.js +3 -20
- package/lib/Adapters/PubSub/EventEmitterPubSub.js +1 -16
- package/lib/Adapters/PubSub/PubSubAdapter.js +2 -9
- package/lib/Adapters/PubSub/RedisPubSub.js +13 -10
- package/lib/Adapters/Push/PushAdapter.js +2 -8
- package/lib/Adapters/Storage/Mongo/MongoCollection.js +12 -37
- package/lib/Adapters/Storage/Mongo/MongoSchemaCollection.js +26 -79
- package/lib/Adapters/Storage/Mongo/MongoStorageAdapter.js +78 -209
- package/lib/Adapters/Storage/Mongo/MongoTransform.js +82 -371
- package/lib/Adapters/Storage/Postgres/PostgresClient.js +1 -13
- package/lib/Adapters/Storage/Postgres/PostgresConfigParser.js +1 -20
- package/lib/Adapters/Storage/Postgres/PostgresStorageAdapter.js +119 -446
- package/lib/Adapters/Storage/Postgres/sql/index.js +4 -7
- package/lib/Adapters/Storage/StorageAdapter.js +1 -1
- package/lib/Adapters/WebSocketServer/WSAdapter.js +3 -12
- package/lib/Adapters/WebSocketServer/WSSAdapter.js +7 -12
- package/lib/Auth.js +54 -121
- package/lib/ClientSDK.js +3 -11
- package/lib/Config.js +69 -113
- package/lib/Controllers/AdaptableController.js +6 -18
- package/lib/Controllers/AnalyticsController.js +1 -9
- package/lib/Controllers/CacheController.js +3 -23
- package/lib/Controllers/DatabaseController.js +147 -345
- package/lib/Controllers/FilesController.js +5 -34
- package/lib/Controllers/HooksController.js +1 -51
- package/lib/Controllers/LiveQueryController.js +4 -23
- package/lib/Controllers/LoggerController.js +15 -54
- package/lib/Controllers/ParseGraphQLController.js +49 -104
- package/lib/Controllers/PushController.js +20 -59
- package/lib/Controllers/SchemaController.js +154 -344
- package/lib/Controllers/UserController.js +11 -72
- package/lib/Controllers/index.js +19 -68
- package/lib/Controllers/types.js +1 -1
- package/lib/Deprecator/Deprecations.js +1 -8
- package/lib/Deprecator/Deprecator.js +9 -18
- package/lib/GraphQL/ParseGraphQLSchema.js +16 -100
- package/lib/GraphQL/ParseGraphQLServer.js +2 -29
- package/lib/GraphQL/helpers/objectsMutations.js +2 -12
- package/lib/GraphQL/helpers/objectsQueries.js +18 -76
- package/lib/GraphQL/loaders/defaultGraphQLMutations.js +1 -9
- package/lib/GraphQL/loaders/defaultGraphQLQueries.js +1 -8
- package/lib/GraphQL/loaders/defaultGraphQLTypes.js +9 -115
- package/lib/GraphQL/loaders/defaultRelaySchema.js +6 -18
- package/lib/GraphQL/loaders/filesMutations.js +2 -19
- package/lib/GraphQL/loaders/functionsMutations.js +6 -17
- package/lib/GraphQL/loaders/parseClassMutations.js +6 -44
- package/lib/GraphQL/loaders/parseClassQueries.js +1 -26
- package/lib/GraphQL/loaders/parseClassTypes.js +10 -64
- package/lib/GraphQL/loaders/schemaDirectives.js +1 -17
- package/lib/GraphQL/loaders/schemaMutations.js +1 -20
- package/lib/GraphQL/loaders/schemaQueries.js +1 -14
- package/lib/GraphQL/loaders/schemaTypes.js +2 -6
- package/lib/GraphQL/loaders/usersMutations.js +6 -28
- package/lib/GraphQL/loaders/usersQueries.js +4 -26
- package/lib/GraphQL/parseGraphQLUtils.js +6 -19
- package/lib/GraphQL/transformers/className.js +1 -4
- package/lib/GraphQL/transformers/constraintType.js +1 -20
- package/lib/GraphQL/transformers/inputType.js +1 -20
- package/lib/GraphQL/transformers/mutation.js +6 -51
- package/lib/GraphQL/transformers/outputType.js +1 -20
- package/lib/GraphQL/transformers/query.js +6 -42
- package/lib/GraphQL/transformers/schemaFields.js +7 -34
- package/lib/KeyPromiseQueue.js +1 -12
- package/lib/LiveQuery/Client.js +1 -25
- package/lib/LiveQuery/Id.js +1 -7
- package/lib/LiveQuery/ParseCloudCodePublisher.js +13 -19
- package/lib/LiveQuery/ParseLiveQueryServer.js +92 -306
- package/lib/LiveQuery/ParsePubSub.js +1 -12
- package/lib/LiveQuery/ParseWebSocketServer.js +4 -26
- package/lib/LiveQuery/QueryTools.js +14 -116
- package/lib/LiveQuery/RequestSchema.js +1 -1
- package/lib/LiveQuery/SessionTokenCache.js +1 -17
- package/lib/LiveQuery/Subscription.js +4 -18
- package/lib/LiveQuery/equalObjects.js +2 -14
- package/lib/Options/Definitions.js +79 -10
- package/lib/Options/docs.js +23 -3
- package/lib/Options/index.js +4 -12
- package/lib/Options/parsers.js +1 -18
- package/lib/Page.js +1 -9
- package/lib/ParseMessageQueue.js +1 -10
- package/lib/ParseServer.js +144 -182
- package/lib/ParseServerRESTController.js +6 -33
- package/lib/PromiseRouter.js +16 -50
- package/lib/Push/PushQueue.js +3 -15
- package/lib/Push/PushWorker.js +7 -32
- package/lib/Push/utils.js +9 -38
- package/lib/RestQuery.js +105 -242
- package/lib/RestWrite.js +212 -377
- package/lib/Routers/AggregateRouter.js +14 -51
- package/lib/Routers/AnalyticsRouter.js +2 -8
- package/lib/Routers/AudiencesRouter.js +1 -15
- package/lib/Routers/ClassesRouter.js +3 -53
- package/lib/Routers/CloudCodeRouter.js +1 -19
- package/lib/Routers/FeaturesRouter.js +1 -10
- package/lib/Routers/FilesRouter.js +29 -76
- package/lib/Routers/FunctionsRouter.js +5 -28
- package/lib/Routers/GlobalConfigRouter.js +4 -18
- package/lib/Routers/GraphQLRouter.js +1 -14
- package/lib/Routers/HooksRouter.js +1 -29
- package/lib/Routers/IAPValidationRouter.js +6 -29
- package/lib/Routers/InstallationsRouter.js +2 -12
- package/lib/Routers/LogsRouter.js +4 -16
- package/lib/Routers/PagesRouter.js +69 -129
- package/lib/Routers/PublicAPIRouter.js +3 -62
- package/lib/Routers/PurgeRouter.js +1 -15
- package/lib/Routers/PushRouter.js +2 -18
- package/lib/Routers/RolesRouter.js +1 -7
- package/lib/Routers/SchemasRouter.js +4 -34
- package/lib/Routers/SecurityRouter.js +1 -12
- package/lib/Routers/SessionsRouter.js +3 -19
- package/lib/Routers/UsersRouter.js +58 -155
- package/lib/SchemaMigrations/DefinedSchemas.js +56 -115
- package/lib/SchemaMigrations/Migrations.js +2 -8
- package/lib/Security/Check.js +8 -16
- package/lib/Security/CheckGroup.js +4 -11
- package/lib/Security/CheckGroups/CheckGroupDatabase.js +8 -18
- package/lib/Security/CheckGroups/CheckGroupServerConfig.js +5 -15
- package/lib/Security/CheckGroups/CheckGroups.js +1 -4
- package/lib/Security/CheckRunner.js +22 -41
- package/lib/StatusHandler.js +12 -69
- package/lib/TestUtils.js +1 -6
- package/lib/Utils.js +27 -66
- package/lib/batch.js +17 -28
- package/lib/cache.js +1 -3
- package/lib/cli/definitions/parse-live-query-server.js +1 -3
- package/lib/cli/definitions/parse-server.js +1 -3
- package/lib/cli/parse-live-query-server.js +1 -6
- package/lib/cli/parse-server.js +11 -21
- package/lib/cli/utils/commander.js +13 -51
- package/lib/cli/utils/runner.js +1 -14
- package/lib/cloud-code/Parse.Cloud.js +71 -92
- package/lib/cryptoUtils.js +11 -19
- package/lib/defaults.js +2 -14
- package/lib/deprecated.js +1 -2
- package/lib/index.js +16 -34
- package/lib/logger.js +6 -13
- package/lib/middlewares.js +147 -151
- package/lib/password.js +6 -10
- package/lib/request.js +173 -2
- package/lib/requiredParameter.js +1 -3
- package/lib/rest.js +19 -41
- package/lib/triggers.js +54 -252
- package/lib/vendor/mongodbUrl.js +125 -305
- package/package.json +22 -19
- package/lib/cloud-code/HTTPResponse.js +0 -73
- package/lib/cloud-code/httpRequest.js +0 -192
package/lib/Utils.js
CHANGED
|
@@ -5,14 +5,13 @@
|
|
|
5
5
|
* @file General purpose utilities
|
|
6
6
|
* @description General purpose utilities.
|
|
7
7
|
*/
|
|
8
|
-
const path = require('path');
|
|
9
8
|
|
|
9
|
+
const path = require('path');
|
|
10
10
|
const fs = require('fs').promises;
|
|
11
|
+
|
|
11
12
|
/**
|
|
12
13
|
* The general purpose utilities.
|
|
13
14
|
*/
|
|
14
|
-
|
|
15
|
-
|
|
16
15
|
class Utils {
|
|
17
16
|
/**
|
|
18
17
|
* @function getLocalizedPath
|
|
@@ -45,50 +44,52 @@ class Utils {
|
|
|
45
44
|
static async getLocalizedPath(defaultPath, locale) {
|
|
46
45
|
// Get file name and paths
|
|
47
46
|
const file = path.basename(defaultPath);
|
|
48
|
-
const basePath = path.dirname(defaultPath);
|
|
47
|
+
const basePath = path.dirname(defaultPath);
|
|
49
48
|
|
|
49
|
+
// If locale is not set return default file
|
|
50
50
|
if (!locale) {
|
|
51
51
|
return {
|
|
52
52
|
path: defaultPath
|
|
53
53
|
};
|
|
54
|
-
}
|
|
55
|
-
|
|
54
|
+
}
|
|
56
55
|
|
|
56
|
+
// Check file for locale exists
|
|
57
57
|
const localePath = path.join(basePath, locale, file);
|
|
58
|
-
const localeFileExists = await Utils.fileExists(localePath);
|
|
58
|
+
const localeFileExists = await Utils.fileExists(localePath);
|
|
59
59
|
|
|
60
|
+
// If file for locale exists return file
|
|
60
61
|
if (localeFileExists) {
|
|
61
62
|
return {
|
|
62
63
|
path: localePath,
|
|
63
64
|
subdir: locale
|
|
64
65
|
};
|
|
65
|
-
}
|
|
66
|
-
|
|
66
|
+
}
|
|
67
67
|
|
|
68
|
+
// Check file for language exists
|
|
68
69
|
const language = locale.split('-')[0];
|
|
69
70
|
const languagePath = path.join(basePath, language, file);
|
|
70
|
-
const languageFileExists = await Utils.fileExists(languagePath);
|
|
71
|
+
const languageFileExists = await Utils.fileExists(languagePath);
|
|
71
72
|
|
|
73
|
+
// If file for language exists return file
|
|
72
74
|
if (languageFileExists) {
|
|
73
75
|
return {
|
|
74
76
|
path: languagePath,
|
|
75
77
|
subdir: language
|
|
76
78
|
};
|
|
77
|
-
}
|
|
78
|
-
|
|
79
|
+
}
|
|
79
80
|
|
|
81
|
+
// Return default file
|
|
80
82
|
return {
|
|
81
83
|
path: defaultPath
|
|
82
84
|
};
|
|
83
85
|
}
|
|
86
|
+
|
|
84
87
|
/**
|
|
85
88
|
* @function fileExists
|
|
86
89
|
* @description Checks whether a file exists.
|
|
87
90
|
* @param {String} path The file path.
|
|
88
91
|
* @returns {Promise<Boolean>} Is true if the file can be accessed, false otherwise.
|
|
89
92
|
*/
|
|
90
|
-
|
|
91
|
-
|
|
92
93
|
static async fileExists(path) {
|
|
93
94
|
try {
|
|
94
95
|
await fs.access(path);
|
|
@@ -97,17 +98,17 @@ class Utils {
|
|
|
97
98
|
return false;
|
|
98
99
|
}
|
|
99
100
|
}
|
|
101
|
+
|
|
100
102
|
/**
|
|
101
103
|
* @function isPath
|
|
102
104
|
* @description Evaluates whether a string is a file path (as opposed to a URL for example).
|
|
103
105
|
* @param {String} s The string to evaluate.
|
|
104
106
|
* @returns {Boolean} Returns true if the evaluated string is a path.
|
|
105
107
|
*/
|
|
106
|
-
|
|
107
|
-
|
|
108
108
|
static isPath(s) {
|
|
109
109
|
return /(^\/)|(^\.\/)|(^\.\.\/)/.test(s);
|
|
110
110
|
}
|
|
111
|
+
|
|
111
112
|
/**
|
|
112
113
|
* Flattens an object and crates new keys with custom delimiters.
|
|
113
114
|
* @param {Object} obj The object to flatten.
|
|
@@ -115,13 +116,10 @@ class Utils {
|
|
|
115
116
|
* @param {Object} result
|
|
116
117
|
* @returns {Object} The flattened object.
|
|
117
118
|
**/
|
|
118
|
-
|
|
119
|
-
|
|
120
119
|
static flattenObject(obj, parentKey, delimiter = '.', result = {}) {
|
|
121
120
|
for (const key in obj) {
|
|
122
121
|
if (Object.prototype.hasOwnProperty.call(obj, key)) {
|
|
123
122
|
const newKey = parentKey ? parentKey + delimiter + key : key;
|
|
124
|
-
|
|
125
123
|
if (typeof obj[key] === 'object' && obj[key] !== null) {
|
|
126
124
|
this.flattenObject(obj[key], newKey, delimiter, result);
|
|
127
125
|
} else {
|
|
@@ -129,19 +127,18 @@ class Utils {
|
|
|
129
127
|
}
|
|
130
128
|
}
|
|
131
129
|
}
|
|
132
|
-
|
|
133
130
|
return result;
|
|
134
131
|
}
|
|
132
|
+
|
|
135
133
|
/**
|
|
136
134
|
* Determines whether an object is a Promise.
|
|
137
135
|
* @param {any} object The object to validate.
|
|
138
136
|
* @returns {Boolean} Returns true if the object is a promise.
|
|
139
137
|
*/
|
|
140
|
-
|
|
141
|
-
|
|
142
138
|
static isPromise(object) {
|
|
143
139
|
return object instanceof Promise;
|
|
144
140
|
}
|
|
141
|
+
|
|
145
142
|
/**
|
|
146
143
|
* Creates an object with all permutations of the original keys.
|
|
147
144
|
* For example, this definition:
|
|
@@ -166,17 +163,13 @@ class Utils {
|
|
|
166
163
|
* @param {Object} [current={}] The current result entry being composed.
|
|
167
164
|
* @param {Array} [results=[]] The resulting array of permutations.
|
|
168
165
|
*/
|
|
169
|
-
|
|
170
|
-
|
|
171
166
|
static getObjectKeyPermutations(object, index = 0, current = {}, results = []) {
|
|
172
167
|
const keys = Object.keys(object);
|
|
173
168
|
const key = keys[index];
|
|
174
169
|
const values = object[key];
|
|
175
|
-
|
|
176
170
|
for (const value of values) {
|
|
177
171
|
current[key] = value;
|
|
178
172
|
const nextIndex = index + 1;
|
|
179
|
-
|
|
180
173
|
if (nextIndex < keys.length) {
|
|
181
174
|
Utils.getObjectKeyPermutations(object, nextIndex, current, results);
|
|
182
175
|
} else {
|
|
@@ -184,9 +177,9 @@ class Utils {
|
|
|
184
177
|
results.push(result);
|
|
185
178
|
}
|
|
186
179
|
}
|
|
187
|
-
|
|
188
180
|
return results;
|
|
189
181
|
}
|
|
182
|
+
|
|
190
183
|
/**
|
|
191
184
|
* Validates parameters and throws if a parameter is invalid.
|
|
192
185
|
* Example parameter types syntax:
|
|
@@ -206,128 +199,106 @@ class Utils {
|
|
|
206
199
|
* @param {Object} types.v The function to validate the parameter value.
|
|
207
200
|
* @param {Boolean} [types.o=false] Is true if the parameter is optional.
|
|
208
201
|
*/
|
|
209
|
-
|
|
210
|
-
|
|
211
202
|
static validateParams(params, types) {
|
|
212
203
|
for (const key of Object.keys(params)) {
|
|
213
204
|
const type = types[key];
|
|
214
205
|
const isOptional = !!type.o;
|
|
215
206
|
const param = params[key];
|
|
216
|
-
|
|
217
207
|
if (!(isOptional && param == null) && !type.v(param)) {
|
|
218
208
|
throw `Invalid parameter ${key} must be of type ${type.t} but is ${typeof param}`;
|
|
219
209
|
}
|
|
220
210
|
}
|
|
221
211
|
}
|
|
212
|
+
|
|
222
213
|
/**
|
|
223
214
|
* Computes the relative date based on a string.
|
|
224
215
|
* @param {String} text The string to interpret the date from.
|
|
225
216
|
* @param {Date} now The date the string is comparing against.
|
|
226
217
|
* @returns {Object} The relative date object.
|
|
227
218
|
**/
|
|
228
|
-
|
|
229
|
-
|
|
230
219
|
static relativeTimeToDate(text, now = new Date()) {
|
|
231
220
|
text = text.toLowerCase();
|
|
232
|
-
let parts = text.split(' ');
|
|
221
|
+
let parts = text.split(' ');
|
|
233
222
|
|
|
223
|
+
// Filter out whitespace
|
|
234
224
|
parts = parts.filter(part => part !== '');
|
|
235
225
|
const future = parts[0] === 'in';
|
|
236
226
|
const past = parts[parts.length - 1] === 'ago';
|
|
237
|
-
|
|
238
227
|
if (!future && !past && text !== 'now') {
|
|
239
228
|
return {
|
|
240
229
|
status: 'error',
|
|
241
230
|
info: "Time should either start with 'in' or end with 'ago'"
|
|
242
231
|
};
|
|
243
232
|
}
|
|
244
|
-
|
|
245
233
|
if (future && past) {
|
|
246
234
|
return {
|
|
247
235
|
status: 'error',
|
|
248
236
|
info: "Time cannot have both 'in' and 'ago'"
|
|
249
237
|
};
|
|
250
|
-
}
|
|
251
|
-
|
|
238
|
+
}
|
|
252
239
|
|
|
240
|
+
// strip the 'ago' or 'in'
|
|
253
241
|
if (future) {
|
|
254
242
|
parts = parts.slice(1);
|
|
255
243
|
} else {
|
|
256
244
|
// past
|
|
257
245
|
parts = parts.slice(0, parts.length - 1);
|
|
258
246
|
}
|
|
259
|
-
|
|
260
247
|
if (parts.length % 2 !== 0 && text !== 'now') {
|
|
261
248
|
return {
|
|
262
249
|
status: 'error',
|
|
263
250
|
info: 'Invalid time string. Dangling unit or number.'
|
|
264
251
|
};
|
|
265
252
|
}
|
|
266
|
-
|
|
267
253
|
const pairs = [];
|
|
268
|
-
|
|
269
254
|
while (parts.length) {
|
|
270
255
|
pairs.push([parts.shift(), parts.shift()]);
|
|
271
256
|
}
|
|
272
|
-
|
|
273
257
|
let seconds = 0;
|
|
274
|
-
|
|
275
258
|
for (const [num, interval] of pairs) {
|
|
276
259
|
const val = Number(num);
|
|
277
|
-
|
|
278
260
|
if (!Number.isInteger(val)) {
|
|
279
261
|
return {
|
|
280
262
|
status: 'error',
|
|
281
263
|
info: `'${num}' is not an integer.`
|
|
282
264
|
};
|
|
283
265
|
}
|
|
284
|
-
|
|
285
266
|
switch (interval) {
|
|
286
267
|
case 'yr':
|
|
287
268
|
case 'yrs':
|
|
288
269
|
case 'year':
|
|
289
270
|
case 'years':
|
|
290
271
|
seconds += val * 31536000; // 365 * 24 * 60 * 60
|
|
291
|
-
|
|
292
272
|
break;
|
|
293
|
-
|
|
294
273
|
case 'wk':
|
|
295
274
|
case 'wks':
|
|
296
275
|
case 'week':
|
|
297
276
|
case 'weeks':
|
|
298
277
|
seconds += val * 604800; // 7 * 24 * 60 * 60
|
|
299
|
-
|
|
300
278
|
break;
|
|
301
|
-
|
|
302
279
|
case 'd':
|
|
303
280
|
case 'day':
|
|
304
281
|
case 'days':
|
|
305
282
|
seconds += val * 86400; // 24 * 60 * 60
|
|
306
|
-
|
|
307
283
|
break;
|
|
308
|
-
|
|
309
284
|
case 'hr':
|
|
310
285
|
case 'hrs':
|
|
311
286
|
case 'hour':
|
|
312
287
|
case 'hours':
|
|
313
288
|
seconds += val * 3600; // 60 * 60
|
|
314
|
-
|
|
315
289
|
break;
|
|
316
|
-
|
|
317
290
|
case 'min':
|
|
318
291
|
case 'mins':
|
|
319
292
|
case 'minute':
|
|
320
293
|
case 'minutes':
|
|
321
294
|
seconds += val * 60;
|
|
322
295
|
break;
|
|
323
|
-
|
|
324
296
|
case 'sec':
|
|
325
297
|
case 'secs':
|
|
326
298
|
case 'second':
|
|
327
299
|
case 'seconds':
|
|
328
300
|
seconds += val;
|
|
329
301
|
break;
|
|
330
|
-
|
|
331
302
|
default:
|
|
332
303
|
return {
|
|
333
304
|
status: 'error',
|
|
@@ -335,9 +306,7 @@ class Utils {
|
|
|
335
306
|
};
|
|
336
307
|
}
|
|
337
308
|
}
|
|
338
|
-
|
|
339
309
|
const milliseconds = seconds * 1000;
|
|
340
|
-
|
|
341
310
|
if (future) {
|
|
342
311
|
return {
|
|
343
312
|
status: 'success',
|
|
@@ -358,6 +327,7 @@ class Utils {
|
|
|
358
327
|
};
|
|
359
328
|
}
|
|
360
329
|
}
|
|
330
|
+
|
|
361
331
|
/**
|
|
362
332
|
* Deep-scans an object for a matching key/value definition.
|
|
363
333
|
* @param {Object} obj The object to scan.
|
|
@@ -365,15 +335,10 @@ class Utils {
|
|
|
365
335
|
* @param {any | undefined} value The value to match, or undefined if only the key should be matched.
|
|
366
336
|
* @returns {Boolean} True if a match was found, false otherwise.
|
|
367
337
|
*/
|
|
368
|
-
|
|
369
|
-
|
|
370
338
|
static objectContainsKeyValue(obj, key, value) {
|
|
371
339
|
const isMatch = (a, b) => typeof a === 'string' && new RegExp(b).test(a) || a === b;
|
|
372
|
-
|
|
373
340
|
const isKeyMatch = k => isMatch(k, key);
|
|
374
|
-
|
|
375
341
|
const isValueMatch = v => isMatch(v, value);
|
|
376
|
-
|
|
377
342
|
for (const [k, v] of Object.entries(obj)) {
|
|
378
343
|
if (key !== undefined && value === undefined && isKeyMatch(k)) {
|
|
379
344
|
return true;
|
|
@@ -382,16 +347,12 @@ class Utils {
|
|
|
382
347
|
} else if (key !== undefined && value !== undefined && isKeyMatch(k) && isValueMatch(v)) {
|
|
383
348
|
return true;
|
|
384
349
|
}
|
|
385
|
-
|
|
386
350
|
if (['[object Object]', '[object Array]'].includes(Object.prototype.toString.call(v))) {
|
|
387
351
|
return Utils.objectContainsKeyValue(v, key, value);
|
|
388
352
|
}
|
|
389
353
|
}
|
|
390
|
-
|
|
391
354
|
return false;
|
|
392
355
|
}
|
|
393
|
-
|
|
394
356
|
}
|
|
395
|
-
|
|
396
357
|
module.exports = Utils;
|
|
397
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/Utils.js"],"names":["path","require","fs","promises","Utils","getLocalizedPath","defaultPath","locale","file","basename","basePath","dirname","localePath","join","localeFileExists","fileExists","subdir","language","split","languagePath","languageFileExists","access","e","isPath","s","test","flattenObject","obj","parentKey","delimiter","result","key","Object","prototype","hasOwnProperty","call","newKey","isPromise","object","Promise","getObjectKeyPermutations","index","current","results","keys","values","value","nextIndex","length","assign","push","validateParams","params","types","type","isOptional","o","param","v","t","relativeTimeToDate","text","now","Date","toLowerCase","parts","filter","part","future","past","status","info","slice","pairs","shift","seconds","num","interval","val","Number","isInteger","milliseconds","valueOf","objectContainsKeyValue","isMatch","a","b","RegExp","isKeyMatch","k","isValueMatch","entries","undefined","includes","toString","module","exports"],"mappings":"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","sourcesContent":["/**\n * utils.js\n * @file General purpose utilities\n * @description General purpose utilities.\n */\n\nconst path = require('path');\nconst fs = require('fs').promises;\n\n/**\n * The general purpose utilities.\n */\nclass Utils {\n  /**\n   * @function getLocalizedPath\n   * @description Returns a localized file path accoring to the locale.\n   *\n   * Localized files are searched in subfolders of a given path, e.g.\n   *\n   * root/\n   * ├── base/                    // base path to files\n   * │   ├── example.html         // default file\n   * │   └── de/                  // de language folder\n   * │   │   └── example.html     // de localized file\n   * │   └── de-AT/               // de-AT locale folder\n   * │   │   └── example.html     // de-AT localized file\n   *\n   * Files are matched with the locale in the following order:\n   * 1. Locale match, e.g. locale `de-AT` matches file in folder `de-AT`.\n   * 2. Language match, e.g. locale `de-AT` matches file in folder `de`.\n   * 3. Default; file in base folder is returned.\n   *\n   * @param {String} defaultPath The absolute file path, which is also\n   * the default path returned if localization is not available.\n   * @param {String} locale The locale.\n   * @returns {Promise<Object>} The object contains:\n   * - `path`: The path to the localized file, or the original path if\n   *   localization is not available.\n   * - `subdir`: The subdirectory of the localized file, or undefined if\n   *   there is no matching localized file.\n   */\n  static async getLocalizedPath(defaultPath, locale) {\n    // Get file name and paths\n    const file = path.basename(defaultPath);\n    const basePath = path.dirname(defaultPath);\n\n    // If locale is not set return default file\n    if (!locale) {\n      return { path: defaultPath };\n    }\n\n    // Check file for locale exists\n    const localePath = path.join(basePath, locale, file);\n    const localeFileExists = await Utils.fileExists(localePath);\n\n    // If file for locale exists return file\n    if (localeFileExists) {\n      return { path: localePath, subdir: locale };\n    }\n\n    // Check file for language exists\n    const language = locale.split('-')[0];\n    const languagePath = path.join(basePath, language, file);\n    const languageFileExists = await Utils.fileExists(languagePath);\n\n    // If file for language exists return file\n    if (languageFileExists) {\n      return { path: languagePath, subdir: language };\n    }\n\n    // Return default file\n    return { path: defaultPath };\n  }\n\n  /**\n   * @function fileExists\n   * @description Checks whether a file exists.\n   * @param {String} path The file path.\n   * @returns {Promise<Boolean>} Is true if the file can be accessed, false otherwise.\n   */\n  static async fileExists(path) {\n    try {\n      await fs.access(path);\n      return true;\n    } catch (e) {\n      return false;\n    }\n  }\n\n  /**\n   * @function isPath\n   * @description Evaluates whether a string is a file path (as opposed to a URL for example).\n   * @param {String} s The string to evaluate.\n   * @returns {Boolean} Returns true if the evaluated string is a path.\n   */\n  static isPath(s) {\n    return /(^\\/)|(^\\.\\/)|(^\\.\\.\\/)/.test(s);\n  }\n\n  /**\n   * Flattens an object and crates new keys with custom delimiters.\n   * @param {Object} obj The object to flatten.\n   * @param {String} [delimiter='.'] The delimiter of the newly generated keys.\n   * @param {Object} result\n   * @returns {Object} The flattened object.\n   **/\n  static flattenObject(obj, parentKey, delimiter = '.', result = {}) {\n    for (const key in obj) {\n      if (Object.prototype.hasOwnProperty.call(obj, key)) {\n        const newKey = parentKey ? parentKey + delimiter + key : key;\n\n        if (typeof obj[key] === 'object' && obj[key] !== null) {\n          this.flattenObject(obj[key], newKey, delimiter, result);\n        } else {\n          result[newKey] = obj[key];\n        }\n      }\n    }\n    return result;\n  }\n\n  /**\n   * Determines whether an object is a Promise.\n   * @param {any} object The object to validate.\n   * @returns {Boolean} Returns true if the object is a promise.\n   */\n  static isPromise(object) {\n    return object instanceof Promise;\n  }\n\n  /**\n   * Creates an object with all permutations of the original keys.\n   * For example, this definition:\n   * ```\n   * {\n   *   a: [true, false],\n   *   b: [1, 2],\n   *   c: ['x']\n   * }\n   * ```\n   * permutates to:\n   * ```\n   * [\n   *   { a: true, b: 1, c: 'x' },\n   *   { a: true, b: 2, c: 'x' },\n   *   { a: false, b: 1, c: 'x' },\n   *   { a: false, b: 2, c: 'x' }\n   * ]\n   * ```\n   * @param {Object} object The object to permutate.\n   * @param {Integer} [index=0] The current key index.\n   * @param {Object} [current={}] The current result entry being composed.\n   * @param {Array} [results=[]] The resulting array of permutations.\n   */\n  static getObjectKeyPermutations(object, index = 0, current = {}, results = []) {\n    const keys = Object.keys(object);\n    const key = keys[index];\n    const values = object[key];\n\n    for (const value of values) {\n      current[key] = value;\n      const nextIndex = index + 1;\n\n      if (nextIndex < keys.length) {\n        Utils.getObjectKeyPermutations(object, nextIndex, current, results);\n      } else {\n        const result = Object.assign({}, current);\n        results.push(result);\n      }\n    }\n    return results;\n  }\n\n  /**\n   * Validates parameters and throws if a parameter is invalid.\n   * Example parameter types syntax:\n   * ```\n   * {\n   *   parameterName: {\n   *      t: 'boolean',\n   *      v: isBoolean,\n   *      o: true\n   *   },\n   *   ...\n   * }\n   * ```\n   * @param {Object} params The parameters to validate.\n   * @param {Array<Object>} types The parameter types used for validation.\n   * @param {Object} types.t The parameter type; used for error message, not for validation.\n   * @param {Object} types.v The function to validate the parameter value.\n   * @param {Boolean} [types.o=false] Is true if the parameter is optional.\n   */\n  static validateParams(params, types) {\n    for (const key of Object.keys(params)) {\n      const type = types[key];\n      const isOptional = !!type.o;\n      const param = params[key];\n      if (!(isOptional && param == null) && !type.v(param)) {\n        throw `Invalid parameter ${key} must be of type ${type.t} but is ${typeof param}`;\n      }\n    }\n  }\n\n  /**\n   * Computes the relative date based on a string.\n   * @param {String} text The string to interpret the date from.\n   * @param {Date} now The date the string is comparing against.\n   * @returns {Object} The relative date object.\n   **/\n  static relativeTimeToDate(text, now = new Date()) {\n    text = text.toLowerCase();\n    let parts = text.split(' ');\n\n    // Filter out whitespace\n    parts = parts.filter(part => part !== '');\n\n    const future = parts[0] === 'in';\n    const past = parts[parts.length - 1] === 'ago';\n\n    if (!future && !past && text !== 'now') {\n      return {\n        status: 'error',\n        info: \"Time should either start with 'in' or end with 'ago'\",\n      };\n    }\n\n    if (future && past) {\n      return {\n        status: 'error',\n        info: \"Time cannot have both 'in' and 'ago'\",\n      };\n    }\n\n    // strip the 'ago' or 'in'\n    if (future) {\n      parts = parts.slice(1);\n    } else {\n      // past\n      parts = parts.slice(0, parts.length - 1);\n    }\n\n    if (parts.length % 2 !== 0 && text !== 'now') {\n      return {\n        status: 'error',\n        info: 'Invalid time string. Dangling unit or number.',\n      };\n    }\n\n    const pairs = [];\n    while (parts.length) {\n      pairs.push([parts.shift(), parts.shift()]);\n    }\n\n    let seconds = 0;\n    for (const [num, interval] of pairs) {\n      const val = Number(num);\n      if (!Number.isInteger(val)) {\n        return {\n          status: 'error',\n          info: `'${num}' is not an integer.`,\n        };\n      }\n\n      switch (interval) {\n        case 'yr':\n        case 'yrs':\n        case 'year':\n        case 'years':\n          seconds += val * 31536000; // 365 * 24 * 60 * 60\n          break;\n\n        case 'wk':\n        case 'wks':\n        case 'week':\n        case 'weeks':\n          seconds += val * 604800; // 7 * 24 * 60 * 60\n          break;\n\n        case 'd':\n        case 'day':\n        case 'days':\n          seconds += val * 86400; // 24 * 60 * 60\n          break;\n\n        case 'hr':\n        case 'hrs':\n        case 'hour':\n        case 'hours':\n          seconds += val * 3600; // 60 * 60\n          break;\n\n        case 'min':\n        case 'mins':\n        case 'minute':\n        case 'minutes':\n          seconds += val * 60;\n          break;\n\n        case 'sec':\n        case 'secs':\n        case 'second':\n        case 'seconds':\n          seconds += val;\n          break;\n\n        default:\n          return {\n            status: 'error',\n            info: `Invalid interval: '${interval}'`,\n          };\n      }\n    }\n\n    const milliseconds = seconds * 1000;\n    if (future) {\n      return {\n        status: 'success',\n        info: 'future',\n        result: new Date(now.valueOf() + milliseconds),\n      };\n    } else if (past) {\n      return {\n        status: 'success',\n        info: 'past',\n        result: new Date(now.valueOf() - milliseconds),\n      };\n    } else {\n      return {\n        status: 'success',\n        info: 'present',\n        result: new Date(now.valueOf()),\n      };\n    }\n  }\n\n  /**\n   * Deep-scans an object for a matching key/value definition.\n   * @param {Object} obj The object to scan.\n   * @param {String | undefined} key The key to match, or undefined if only the value should be matched.\n   * @param {any | undefined} value The value to match, or undefined if only the key should be matched.\n   * @returns {Boolean} True if a match was found, false otherwise.\n   */\n  static objectContainsKeyValue(obj, key, value) {\n    const isMatch = (a, b) => (typeof a === 'string' && new RegExp(b).test(a)) || a === b;\n    const isKeyMatch = k => isMatch(k, key);\n    const isValueMatch = v => isMatch(v, value);\n    for (const [k, v] of Object.entries(obj)) {\n      if (key !== undefined && value === undefined && isKeyMatch(k)) {\n        return true;\n      } else if (key === undefined && value !== undefined && isValueMatch(v)) {\n        return true;\n      } else if (key !== undefined && value !== undefined && isKeyMatch(k) && isValueMatch(v)) {\n        return true;\n      }\n      if (['[object Object]', '[object Array]'].includes(Object.prototype.toString.call(v))) {\n        return Utils.objectContainsKeyValue(v, key, value);\n      }\n    }\n    return false;\n  }\n}\n\nmodule.exports = Utils;\n"]}
|
|
358
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["path","require","fs","promises","Utils","getLocalizedPath","defaultPath","locale","file","basename","basePath","dirname","localePath","join","localeFileExists","fileExists","subdir","language","split","languagePath","languageFileExists","access","e","isPath","s","test","flattenObject","obj","parentKey","delimiter","result","key","Object","prototype","hasOwnProperty","call","newKey","isPromise","object","Promise","getObjectKeyPermutations","index","current","results","keys","values","value","nextIndex","length","assign","push","validateParams","params","types","type","isOptional","o","param","v","t","relativeTimeToDate","text","now","Date","toLowerCase","parts","filter","part","future","past","status","info","slice","pairs","shift","seconds","num","interval","val","Number","isInteger","milliseconds","valueOf","objectContainsKeyValue","isMatch","a","b","RegExp","isKeyMatch","k","isValueMatch","entries","undefined","includes","toString","module","exports"],"sources":["../src/Utils.js"],"sourcesContent":["/**\n * utils.js\n * @file General purpose utilities\n * @description General purpose utilities.\n */\n\nconst path = require('path');\nconst fs = require('fs').promises;\n\n/**\n * The general purpose utilities.\n */\nclass Utils {\n  /**\n   * @function getLocalizedPath\n   * @description Returns a localized file path accoring to the locale.\n   *\n   * Localized files are searched in subfolders of a given path, e.g.\n   *\n   * root/\n   * ├── base/                    // base path to files\n   * │   ├── example.html         // default file\n   * │   └── de/                  // de language folder\n   * │   │   └── example.html     // de localized file\n   * │   └── de-AT/               // de-AT locale folder\n   * │   │   └── example.html     // de-AT localized file\n   *\n   * Files are matched with the locale in the following order:\n   * 1. Locale match, e.g. locale `de-AT` matches file in folder `de-AT`.\n   * 2. Language match, e.g. locale `de-AT` matches file in folder `de`.\n   * 3. Default; file in base folder is returned.\n   *\n   * @param {String} defaultPath The absolute file path, which is also\n   * the default path returned if localization is not available.\n   * @param {String} locale The locale.\n   * @returns {Promise<Object>} The object contains:\n   * - `path`: The path to the localized file, or the original path if\n   *   localization is not available.\n   * - `subdir`: The subdirectory of the localized file, or undefined if\n   *   there is no matching localized file.\n   */\n  static async getLocalizedPath(defaultPath, locale) {\n    // Get file name and paths\n    const file = path.basename(defaultPath);\n    const basePath = path.dirname(defaultPath);\n\n    // If locale is not set return default file\n    if (!locale) {\n      return { path: defaultPath };\n    }\n\n    // Check file for locale exists\n    const localePath = path.join(basePath, locale, file);\n    const localeFileExists = await Utils.fileExists(localePath);\n\n    // If file for locale exists return file\n    if (localeFileExists) {\n      return { path: localePath, subdir: locale };\n    }\n\n    // Check file for language exists\n    const language = locale.split('-')[0];\n    const languagePath = path.join(basePath, language, file);\n    const languageFileExists = await Utils.fileExists(languagePath);\n\n    // If file for language exists return file\n    if (languageFileExists) {\n      return { path: languagePath, subdir: language };\n    }\n\n    // Return default file\n    return { path: defaultPath };\n  }\n\n  /**\n   * @function fileExists\n   * @description Checks whether a file exists.\n   * @param {String} path The file path.\n   * @returns {Promise<Boolean>} Is true if the file can be accessed, false otherwise.\n   */\n  static async fileExists(path) {\n    try {\n      await fs.access(path);\n      return true;\n    } catch (e) {\n      return false;\n    }\n  }\n\n  /**\n   * @function isPath\n   * @description Evaluates whether a string is a file path (as opposed to a URL for example).\n   * @param {String} s The string to evaluate.\n   * @returns {Boolean} Returns true if the evaluated string is a path.\n   */\n  static isPath(s) {\n    return /(^\\/)|(^\\.\\/)|(^\\.\\.\\/)/.test(s);\n  }\n\n  /**\n   * Flattens an object and crates new keys with custom delimiters.\n   * @param {Object} obj The object to flatten.\n   * @param {String} [delimiter='.'] The delimiter of the newly generated keys.\n   * @param {Object} result\n   * @returns {Object} The flattened object.\n   **/\n  static flattenObject(obj, parentKey, delimiter = '.', result = {}) {\n    for (const key in obj) {\n      if (Object.prototype.hasOwnProperty.call(obj, key)) {\n        const newKey = parentKey ? parentKey + delimiter + key : key;\n\n        if (typeof obj[key] === 'object' && obj[key] !== null) {\n          this.flattenObject(obj[key], newKey, delimiter, result);\n        } else {\n          result[newKey] = obj[key];\n        }\n      }\n    }\n    return result;\n  }\n\n  /**\n   * Determines whether an object is a Promise.\n   * @param {any} object The object to validate.\n   * @returns {Boolean} Returns true if the object is a promise.\n   */\n  static isPromise(object) {\n    return object instanceof Promise;\n  }\n\n  /**\n   * Creates an object with all permutations of the original keys.\n   * For example, this definition:\n   * ```\n   * {\n   *   a: [true, false],\n   *   b: [1, 2],\n   *   c: ['x']\n   * }\n   * ```\n   * permutates to:\n   * ```\n   * [\n   *   { a: true, b: 1, c: 'x' },\n   *   { a: true, b: 2, c: 'x' },\n   *   { a: false, b: 1, c: 'x' },\n   *   { a: false, b: 2, c: 'x' }\n   * ]\n   * ```\n   * @param {Object} object The object to permutate.\n   * @param {Integer} [index=0] The current key index.\n   * @param {Object} [current={}] The current result entry being composed.\n   * @param {Array} [results=[]] The resulting array of permutations.\n   */\n  static getObjectKeyPermutations(object, index = 0, current = {}, results = []) {\n    const keys = Object.keys(object);\n    const key = keys[index];\n    const values = object[key];\n\n    for (const value of values) {\n      current[key] = value;\n      const nextIndex = index + 1;\n\n      if (nextIndex < keys.length) {\n        Utils.getObjectKeyPermutations(object, nextIndex, current, results);\n      } else {\n        const result = Object.assign({}, current);\n        results.push(result);\n      }\n    }\n    return results;\n  }\n\n  /**\n   * Validates parameters and throws if a parameter is invalid.\n   * Example parameter types syntax:\n   * ```\n   * {\n   *   parameterName: {\n   *      t: 'boolean',\n   *      v: isBoolean,\n   *      o: true\n   *   },\n   *   ...\n   * }\n   * ```\n   * @param {Object} params The parameters to validate.\n   * @param {Array<Object>} types The parameter types used for validation.\n   * @param {Object} types.t The parameter type; used for error message, not for validation.\n   * @param {Object} types.v The function to validate the parameter value.\n   * @param {Boolean} [types.o=false] Is true if the parameter is optional.\n   */\n  static validateParams(params, types) {\n    for (const key of Object.keys(params)) {\n      const type = types[key];\n      const isOptional = !!type.o;\n      const param = params[key];\n      if (!(isOptional && param == null) && !type.v(param)) {\n        throw `Invalid parameter ${key} must be of type ${type.t} but is ${typeof param}`;\n      }\n    }\n  }\n\n  /**\n   * Computes the relative date based on a string.\n   * @param {String} text The string to interpret the date from.\n   * @param {Date} now The date the string is comparing against.\n   * @returns {Object} The relative date object.\n   **/\n  static relativeTimeToDate(text, now = new Date()) {\n    text = text.toLowerCase();\n    let parts = text.split(' ');\n\n    // Filter out whitespace\n    parts = parts.filter(part => part !== '');\n\n    const future = parts[0] === 'in';\n    const past = parts[parts.length - 1] === 'ago';\n\n    if (!future && !past && text !== 'now') {\n      return {\n        status: 'error',\n        info: \"Time should either start with 'in' or end with 'ago'\",\n      };\n    }\n\n    if (future && past) {\n      return {\n        status: 'error',\n        info: \"Time cannot have both 'in' and 'ago'\",\n      };\n    }\n\n    // strip the 'ago' or 'in'\n    if (future) {\n      parts = parts.slice(1);\n    } else {\n      // past\n      parts = parts.slice(0, parts.length - 1);\n    }\n\n    if (parts.length % 2 !== 0 && text !== 'now') {\n      return {\n        status: 'error',\n        info: 'Invalid time string. Dangling unit or number.',\n      };\n    }\n\n    const pairs = [];\n    while (parts.length) {\n      pairs.push([parts.shift(), parts.shift()]);\n    }\n\n    let seconds = 0;\n    for (const [num, interval] of pairs) {\n      const val = Number(num);\n      if (!Number.isInteger(val)) {\n        return {\n          status: 'error',\n          info: `'${num}' is not an integer.`,\n        };\n      }\n\n      switch (interval) {\n        case 'yr':\n        case 'yrs':\n        case 'year':\n        case 'years':\n          seconds += val * 31536000; // 365 * 24 * 60 * 60\n          break;\n\n        case 'wk':\n        case 'wks':\n        case 'week':\n        case 'weeks':\n          seconds += val * 604800; // 7 * 24 * 60 * 60\n          break;\n\n        case 'd':\n        case 'day':\n        case 'days':\n          seconds += val * 86400; // 24 * 60 * 60\n          break;\n\n        case 'hr':\n        case 'hrs':\n        case 'hour':\n        case 'hours':\n          seconds += val * 3600; // 60 * 60\n          break;\n\n        case 'min':\n        case 'mins':\n        case 'minute':\n        case 'minutes':\n          seconds += val * 60;\n          break;\n\n        case 'sec':\n        case 'secs':\n        case 'second':\n        case 'seconds':\n          seconds += val;\n          break;\n\n        default:\n          return {\n            status: 'error',\n            info: `Invalid interval: '${interval}'`,\n          };\n      }\n    }\n\n    const milliseconds = seconds * 1000;\n    if (future) {\n      return {\n        status: 'success',\n        info: 'future',\n        result: new Date(now.valueOf() + milliseconds),\n      };\n    } else if (past) {\n      return {\n        status: 'success',\n        info: 'past',\n        result: new Date(now.valueOf() - milliseconds),\n      };\n    } else {\n      return {\n        status: 'success',\n        info: 'present',\n        result: new Date(now.valueOf()),\n      };\n    }\n  }\n\n  /**\n   * Deep-scans an object for a matching key/value definition.\n   * @param {Object} obj The object to scan.\n   * @param {String | undefined} key The key to match, or undefined if only the value should be matched.\n   * @param {any | undefined} value The value to match, or undefined if only the key should be matched.\n   * @returns {Boolean} True if a match was found, false otherwise.\n   */\n  static objectContainsKeyValue(obj, key, value) {\n    const isMatch = (a, b) => (typeof a === 'string' && new RegExp(b).test(a)) || a === b;\n    const isKeyMatch = k => isMatch(k, key);\n    const isValueMatch = v => isMatch(v, value);\n    for (const [k, v] of Object.entries(obj)) {\n      if (key !== undefined && value === undefined && isKeyMatch(k)) {\n        return true;\n      } else if (key === undefined && value !== undefined && isValueMatch(v)) {\n        return true;\n      } else if (key !== undefined && value !== undefined && isKeyMatch(k) && isValueMatch(v)) {\n        return true;\n      }\n      if (['[object Object]', '[object Array]'].includes(Object.prototype.toString.call(v))) {\n        return Utils.objectContainsKeyValue(v, key, value);\n      }\n    }\n    return false;\n  }\n}\n\nmodule.exports = Utils;\n"],"mappings":"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"}
|
package/lib/batch.js
CHANGED
|
@@ -1,18 +1,16 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
|
|
3
3
|
const Parse = require('parse/node').Parse;
|
|
4
|
+
const path = require('path');
|
|
5
|
+
// These methods handle batch requests.
|
|
6
|
+
const batchPath = '/batch';
|
|
4
7
|
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
const batchPath = '/batch'; // Mounts a batch-handler onto a PromiseRouter.
|
|
9
|
-
|
|
8
|
+
// Mounts a batch-handler onto a PromiseRouter.
|
|
10
9
|
function mountOnto(router) {
|
|
11
10
|
router.route('POST', batchPath, req => {
|
|
12
11
|
return handleBatch(router, req);
|
|
13
12
|
});
|
|
14
13
|
}
|
|
15
|
-
|
|
16
14
|
function parseURL(urlString) {
|
|
17
15
|
try {
|
|
18
16
|
return new URL(urlString);
|
|
@@ -20,26 +18,23 @@ function parseURL(urlString) {
|
|
|
20
18
|
return undefined;
|
|
21
19
|
}
|
|
22
20
|
}
|
|
23
|
-
|
|
24
21
|
function makeBatchRoutingPathFunction(originalUrl, serverURL, publicServerURL) {
|
|
25
22
|
serverURL = serverURL ? parseURL(serverURL) : undefined;
|
|
26
23
|
publicServerURL = publicServerURL ? parseURL(publicServerURL) : undefined;
|
|
27
24
|
const apiPrefixLength = originalUrl.length - batchPath.length;
|
|
28
25
|
let apiPrefix = originalUrl.slice(0, apiPrefixLength);
|
|
29
|
-
|
|
30
26
|
const makeRoutablePath = function (requestPath) {
|
|
31
27
|
// The routablePath is the path minus the api prefix
|
|
32
28
|
if (requestPath.slice(0, apiPrefix.length) != apiPrefix) {
|
|
33
29
|
throw new Parse.Error(Parse.Error.INVALID_JSON, 'cannot route batch path ' + requestPath);
|
|
34
30
|
}
|
|
35
|
-
|
|
36
31
|
return path.posix.join('/', requestPath.slice(apiPrefix.length));
|
|
37
32
|
};
|
|
38
|
-
|
|
39
33
|
if (serverURL && publicServerURL && serverURL.pathname != publicServerURL.pathname) {
|
|
40
34
|
const localPath = serverURL.pathname;
|
|
41
|
-
const publicPath = publicServerURL.pathname;
|
|
35
|
+
const publicPath = publicServerURL.pathname;
|
|
42
36
|
|
|
37
|
+
// Override the api prefix
|
|
43
38
|
apiPrefix = localPath;
|
|
44
39
|
return function (requestPath) {
|
|
45
40
|
// Figure out which server url was used by figuring out which
|
|
@@ -47,44 +42,41 @@ function makeBatchRoutingPathFunction(originalUrl, serverURL, publicServerURL) {
|
|
|
47
42
|
const startsWithLocal = requestPath.startsWith(localPath);
|
|
48
43
|
const startsWithPublic = requestPath.startsWith(publicPath);
|
|
49
44
|
const pathLengthToUse = startsWithLocal && startsWithPublic ? Math.max(localPath.length, publicPath.length) : startsWithLocal ? localPath.length : publicPath.length;
|
|
50
|
-
const newPath = path.posix.join('/', localPath, '/', requestPath.slice(pathLengthToUse));
|
|
45
|
+
const newPath = path.posix.join('/', localPath, '/', requestPath.slice(pathLengthToUse));
|
|
51
46
|
|
|
47
|
+
// Use the method for local routing
|
|
52
48
|
return makeRoutablePath(newPath);
|
|
53
49
|
};
|
|
54
50
|
}
|
|
55
|
-
|
|
56
51
|
return makeRoutablePath;
|
|
57
|
-
}
|
|
58
|
-
// TODO: pass along auth correctly
|
|
59
|
-
|
|
52
|
+
}
|
|
60
53
|
|
|
54
|
+
// Returns a promise for a {response} object.
|
|
55
|
+
// TODO: pass along auth correctly
|
|
61
56
|
function handleBatch(router, req) {
|
|
62
57
|
if (!Array.isArray(req.body.requests)) {
|
|
63
58
|
throw new Parse.Error(Parse.Error.INVALID_JSON, 'requests must be an array');
|
|
64
|
-
}
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
// The batch paths are all from the root of our domain.
|
|
65
62
|
// That means they include the API prefix, that the API is mounted
|
|
66
63
|
// to. However, our promise router does not route the api prefix. So
|
|
67
64
|
// we need to figure out the API prefix, so that we can strip it
|
|
68
65
|
// from all the subrequests.
|
|
69
|
-
|
|
70
|
-
|
|
71
66
|
if (!req.originalUrl.endsWith(batchPath)) {
|
|
72
67
|
throw 'internal routing problem - expected url to end with batch';
|
|
73
68
|
}
|
|
74
|
-
|
|
75
69
|
const makeRoutablePath = makeBatchRoutingPathFunction(req.originalUrl, req.config.serverURL, req.config.publicServerURL);
|
|
76
|
-
|
|
77
70
|
const batch = transactionRetries => {
|
|
78
71
|
let initialPromise = Promise.resolve();
|
|
79
|
-
|
|
80
72
|
if (req.body.transaction === true) {
|
|
81
73
|
initialPromise = req.config.database.createTransactionalSession();
|
|
82
74
|
}
|
|
83
|
-
|
|
84
75
|
return initialPromise.then(() => {
|
|
85
76
|
const promises = req.body.requests.map(restRequest => {
|
|
86
|
-
const routablePath = makeRoutablePath(restRequest.path);
|
|
77
|
+
const routablePath = makeRoutablePath(restRequest.path);
|
|
87
78
|
|
|
79
|
+
// Construct a request that we can send to a handler
|
|
88
80
|
const request = {
|
|
89
81
|
body: restRequest.body,
|
|
90
82
|
config: req.config,
|
|
@@ -128,17 +120,14 @@ function handleBatch(router, req) {
|
|
|
128
120
|
if (error && error.response && error.response.find(errorItem => typeof errorItem.error === 'object' && errorItem.error.code === 251) && transactionRetries > 0) {
|
|
129
121
|
return batch(transactionRetries - 1);
|
|
130
122
|
}
|
|
131
|
-
|
|
132
123
|
throw error;
|
|
133
124
|
});
|
|
134
125
|
});
|
|
135
126
|
};
|
|
136
|
-
|
|
137
127
|
return batch(5);
|
|
138
128
|
}
|
|
139
|
-
|
|
140
129
|
module.exports = {
|
|
141
130
|
mountOnto,
|
|
142
131
|
makeBatchRoutingPathFunction
|
|
143
132
|
};
|
|
144
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/batch.js"],"names":["Parse","require","path","batchPath","mountOnto","router","route","req","handleBatch","parseURL","urlString","URL","error","undefined","makeBatchRoutingPathFunction","originalUrl","serverURL","publicServerURL","apiPrefixLength","length","apiPrefix","slice","makeRoutablePath","requestPath","Error","INVALID_JSON","posix","join","pathname","localPath","publicPath","startsWithLocal","startsWith","startsWithPublic","pathLengthToUse","Math","max","newPath","Array","isArray","body","requests","endsWith","config","batch","transactionRetries","initialPromise","Promise","resolve","transaction","database","createTransactionalSession","then","promises","map","restRequest","routablePath","request","auth","info","tryRouteRequest","method","response","success","code","message","all","results","find","result","abortTransactionalSession","reject","commitTransactionalSession","catch","errorItem","module","exports"],"mappings":"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","sourcesContent":["const Parse = require('parse/node').Parse;\nconst path = require('path');\n// These methods handle batch requests.\nconst batchPath = '/batch';\n\n// Mounts a batch-handler onto a PromiseRouter.\nfunction mountOnto(router) {\n  router.route('POST', batchPath, req => {\n    return handleBatch(router, req);\n  });\n}\n\nfunction parseURL(urlString) {\n  try {\n    return new URL(urlString);\n  } catch (error) {\n    return undefined;\n  }\n}\n\nfunction makeBatchRoutingPathFunction(originalUrl, serverURL, publicServerURL) {\n  serverURL = serverURL ? parseURL(serverURL) : undefined;\n  publicServerURL = publicServerURL ? parseURL(publicServerURL) : undefined;\n\n  const apiPrefixLength = originalUrl.length - batchPath.length;\n  let apiPrefix = originalUrl.slice(0, apiPrefixLength);\n\n  const makeRoutablePath = function (requestPath) {\n    // The routablePath is the path minus the api prefix\n    if (requestPath.slice(0, apiPrefix.length) != apiPrefix) {\n      throw new Parse.Error(Parse.Error.INVALID_JSON, 'cannot route batch path ' + requestPath);\n    }\n    return path.posix.join('/', requestPath.slice(apiPrefix.length));\n  };\n\n  if (serverURL && publicServerURL && serverURL.pathname != publicServerURL.pathname) {\n    const localPath = serverURL.pathname;\n    const publicPath = publicServerURL.pathname;\n\n    // Override the api prefix\n    apiPrefix = localPath;\n    return function (requestPath) {\n      // Figure out which server url was used by figuring out which\n      // path more closely matches requestPath\n      const startsWithLocal = requestPath.startsWith(localPath);\n      const startsWithPublic = requestPath.startsWith(publicPath);\n      const pathLengthToUse =\n        startsWithLocal && startsWithPublic\n          ? Math.max(localPath.length, publicPath.length)\n          : startsWithLocal\n            ? localPath.length\n            : publicPath.length;\n\n      const newPath = path.posix.join('/', localPath, '/', requestPath.slice(pathLengthToUse));\n\n      // Use the method for local routing\n      return makeRoutablePath(newPath);\n    };\n  }\n\n  return makeRoutablePath;\n}\n\n// Returns a promise for a {response} object.\n// TODO: pass along auth correctly\nfunction handleBatch(router, req) {\n  if (!Array.isArray(req.body.requests)) {\n    throw new Parse.Error(Parse.Error.INVALID_JSON, 'requests must be an array');\n  }\n\n  // The batch paths are all from the root of our domain.\n  // That means they include the API prefix, that the API is mounted\n  // to. However, our promise router does not route the api prefix. So\n  // we need to figure out the API prefix, so that we can strip it\n  // from all the subrequests.\n  if (!req.originalUrl.endsWith(batchPath)) {\n    throw 'internal routing problem - expected url to end with batch';\n  }\n\n  const makeRoutablePath = makeBatchRoutingPathFunction(\n    req.originalUrl,\n    req.config.serverURL,\n    req.config.publicServerURL\n  );\n\n  const batch = transactionRetries => {\n    let initialPromise = Promise.resolve();\n    if (req.body.transaction === true) {\n      initialPromise = req.config.database.createTransactionalSession();\n    }\n\n    return initialPromise.then(() => {\n      const promises = req.body.requests.map(restRequest => {\n        const routablePath = makeRoutablePath(restRequest.path);\n\n        // Construct a request that we can send to a handler\n        const request = {\n          body: restRequest.body,\n          config: req.config,\n          auth: req.auth,\n          info: req.info,\n        };\n\n        return router.tryRouteRequest(restRequest.method, routablePath, request).then(\n          response => {\n            return { success: response.response };\n          },\n          error => {\n            return { error: { code: error.code, error: error.message } };\n          }\n        );\n      });\n\n      return Promise.all(promises)\n        .then(results => {\n          if (req.body.transaction === true) {\n            if (results.find(result => typeof result.error === 'object')) {\n              return req.config.database.abortTransactionalSession().then(() => {\n                return Promise.reject({ response: results });\n              });\n            } else {\n              return req.config.database.commitTransactionalSession().then(() => {\n                return { response: results };\n              });\n            }\n          } else {\n            return { response: results };\n          }\n        })\n        .catch(error => {\n          if (\n            error &&\n            error.response &&\n            error.response.find(\n              errorItem => typeof errorItem.error === 'object' && errorItem.error.code === 251\n            ) &&\n            transactionRetries > 0\n          ) {\n            return batch(transactionRetries - 1);\n          }\n          throw error;\n        });\n    });\n  };\n  return batch(5);\n}\n\nmodule.exports = {\n  mountOnto,\n  makeBatchRoutingPathFunction,\n};\n"]}
|
|
133
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Parse","require","path","batchPath","mountOnto","router","route","req","handleBatch","parseURL","urlString","URL","error","undefined","makeBatchRoutingPathFunction","originalUrl","serverURL","publicServerURL","apiPrefixLength","length","apiPrefix","slice","makeRoutablePath","requestPath","Error","INVALID_JSON","posix","join","pathname","localPath","publicPath","startsWithLocal","startsWith","startsWithPublic","pathLengthToUse","Math","max","newPath","Array","isArray","body","requests","endsWith","config","batch","transactionRetries","initialPromise","Promise","resolve","transaction","database","createTransactionalSession","then","promises","map","restRequest","routablePath","request","auth","info","tryRouteRequest","method","response","success","code","message","all","results","find","result","abortTransactionalSession","reject","commitTransactionalSession","catch","errorItem","module","exports"],"sources":["../src/batch.js"],"sourcesContent":["const Parse = require('parse/node').Parse;\nconst path = require('path');\n// These methods handle batch requests.\nconst batchPath = '/batch';\n\n// Mounts a batch-handler onto a PromiseRouter.\nfunction mountOnto(router) {\n  router.route('POST', batchPath, req => {\n    return handleBatch(router, req);\n  });\n}\n\nfunction parseURL(urlString) {\n  try {\n    return new URL(urlString);\n  } catch (error) {\n    return undefined;\n  }\n}\n\nfunction makeBatchRoutingPathFunction(originalUrl, serverURL, publicServerURL) {\n  serverURL = serverURL ? parseURL(serverURL) : undefined;\n  publicServerURL = publicServerURL ? parseURL(publicServerURL) : undefined;\n\n  const apiPrefixLength = originalUrl.length - batchPath.length;\n  let apiPrefix = originalUrl.slice(0, apiPrefixLength);\n\n  const makeRoutablePath = function (requestPath) {\n    // The routablePath is the path minus the api prefix\n    if (requestPath.slice(0, apiPrefix.length) != apiPrefix) {\n      throw new Parse.Error(Parse.Error.INVALID_JSON, 'cannot route batch path ' + requestPath);\n    }\n    return path.posix.join('/', requestPath.slice(apiPrefix.length));\n  };\n\n  if (serverURL && publicServerURL && serverURL.pathname != publicServerURL.pathname) {\n    const localPath = serverURL.pathname;\n    const publicPath = publicServerURL.pathname;\n\n    // Override the api prefix\n    apiPrefix = localPath;\n    return function (requestPath) {\n      // Figure out which server url was used by figuring out which\n      // path more closely matches requestPath\n      const startsWithLocal = requestPath.startsWith(localPath);\n      const startsWithPublic = requestPath.startsWith(publicPath);\n      const pathLengthToUse =\n        startsWithLocal && startsWithPublic\n          ? Math.max(localPath.length, publicPath.length)\n          : startsWithLocal\n            ? localPath.length\n            : publicPath.length;\n\n      const newPath = path.posix.join('/', localPath, '/', requestPath.slice(pathLengthToUse));\n\n      // Use the method for local routing\n      return makeRoutablePath(newPath);\n    };\n  }\n\n  return makeRoutablePath;\n}\n\n// Returns a promise for a {response} object.\n// TODO: pass along auth correctly\nfunction handleBatch(router, req) {\n  if (!Array.isArray(req.body.requests)) {\n    throw new Parse.Error(Parse.Error.INVALID_JSON, 'requests must be an array');\n  }\n\n  // The batch paths are all from the root of our domain.\n  // That means they include the API prefix, that the API is mounted\n  // to. However, our promise router does not route the api prefix. So\n  // we need to figure out the API prefix, so that we can strip it\n  // from all the subrequests.\n  if (!req.originalUrl.endsWith(batchPath)) {\n    throw 'internal routing problem - expected url to end with batch';\n  }\n\n  const makeRoutablePath = makeBatchRoutingPathFunction(\n    req.originalUrl,\n    req.config.serverURL,\n    req.config.publicServerURL\n  );\n\n  const batch = transactionRetries => {\n    let initialPromise = Promise.resolve();\n    if (req.body.transaction === true) {\n      initialPromise = req.config.database.createTransactionalSession();\n    }\n\n    return initialPromise.then(() => {\n      const promises = req.body.requests.map(restRequest => {\n        const routablePath = makeRoutablePath(restRequest.path);\n\n        // Construct a request that we can send to a handler\n        const request = {\n          body: restRequest.body,\n          config: req.config,\n          auth: req.auth,\n          info: req.info,\n        };\n\n        return router.tryRouteRequest(restRequest.method, routablePath, request).then(\n          response => {\n            return { success: response.response };\n          },\n          error => {\n            return { error: { code: error.code, error: error.message } };\n          }\n        );\n      });\n\n      return Promise.all(promises)\n        .then(results => {\n          if (req.body.transaction === true) {\n            if (results.find(result => typeof result.error === 'object')) {\n              return req.config.database.abortTransactionalSession().then(() => {\n                return Promise.reject({ response: results });\n              });\n            } else {\n              return req.config.database.commitTransactionalSession().then(() => {\n                return { response: results };\n              });\n            }\n          } else {\n            return { response: results };\n          }\n        })\n        .catch(error => {\n          if (\n            error &&\n            error.response &&\n            error.response.find(\n              errorItem => typeof errorItem.error === 'object' && errorItem.error.code === 251\n            ) &&\n            transactionRetries > 0\n          ) {\n            return batch(transactionRetries - 1);\n          }\n          throw error;\n        });\n    });\n  };\n  return batch(5);\n}\n\nmodule.exports = {\n  mountOnto,\n  makeBatchRoutingPathFunction,\n};\n"],"mappings":";;AAAA,MAAMA,KAAK,GAAGC,OAAO,CAAC,YAAY,CAAC,CAACD,KAAK;AACzC,MAAME,IAAI,GAAGD,OAAO,CAAC,MAAM,CAAC;AAC5B;AACA,MAAME,SAAS,GAAG,QAAQ;;AAE1B;AACA,SAASC,SAAS,CAACC,MAAM,EAAE;EACzBA,MAAM,CAACC,KAAK,CAAC,MAAM,EAAEH,SAAS,EAAEI,GAAG,IAAI;IACrC,OAAOC,WAAW,CAACH,MAAM,EAAEE,GAAG,CAAC;EACjC,CAAC,CAAC;AACJ;AAEA,SAASE,QAAQ,CAACC,SAAS,EAAE;EAC3B,IAAI;IACF,OAAO,IAAIC,GAAG,CAACD,SAAS,CAAC;EAC3B,CAAC,CAAC,OAAOE,KAAK,EAAE;IACd,OAAOC,SAAS;EAClB;AACF;AAEA,SAASC,4BAA4B,CAACC,WAAW,EAAEC,SAAS,EAAEC,eAAe,EAAE;EAC7ED,SAAS,GAAGA,SAAS,GAAGP,QAAQ,CAACO,SAAS,CAAC,GAAGH,SAAS;EACvDI,eAAe,GAAGA,eAAe,GAAGR,QAAQ,CAACQ,eAAe,CAAC,GAAGJ,SAAS;EAEzE,MAAMK,eAAe,GAAGH,WAAW,CAACI,MAAM,GAAGhB,SAAS,CAACgB,MAAM;EAC7D,IAAIC,SAAS,GAAGL,WAAW,CAACM,KAAK,CAAC,CAAC,EAAEH,eAAe,CAAC;EAErD,MAAMI,gBAAgB,GAAG,UAAUC,WAAW,EAAE;IAC9C;IACA,IAAIA,WAAW,CAACF,KAAK,CAAC,CAAC,EAAED,SAAS,CAACD,MAAM,CAAC,IAAIC,SAAS,EAAE;MACvD,MAAM,IAAIpB,KAAK,CAACwB,KAAK,CAACxB,KAAK,CAACwB,KAAK,CAACC,YAAY,EAAE,0BAA0B,GAAGF,WAAW,CAAC;IAC3F;IACA,OAAOrB,IAAI,CAACwB,KAAK,CAACC,IAAI,CAAC,GAAG,EAAEJ,WAAW,CAACF,KAAK,CAACD,SAAS,CAACD,MAAM,CAAC,CAAC;EAClE,CAAC;EAED,IAAIH,SAAS,IAAIC,eAAe,IAAID,SAAS,CAACY,QAAQ,IAAIX,eAAe,CAACW,QAAQ,EAAE;IAClF,MAAMC,SAAS,GAAGb,SAAS,CAACY,QAAQ;IACpC,MAAME,UAAU,GAAGb,eAAe,CAACW,QAAQ;;IAE3C;IACAR,SAAS,GAAGS,SAAS;IACrB,OAAO,UAAUN,WAAW,EAAE;MAC5B;MACA;MACA,MAAMQ,eAAe,GAAGR,WAAW,CAACS,UAAU,CAACH,SAAS,CAAC;MACzD,MAAMI,gBAAgB,GAAGV,WAAW,CAACS,UAAU,CAACF,UAAU,CAAC;MAC3D,MAAMI,eAAe,GACnBH,eAAe,IAAIE,gBAAgB,GAC/BE,IAAI,CAACC,GAAG,CAACP,SAAS,CAACV,MAAM,EAAEW,UAAU,CAACX,MAAM,CAAC,GAC7CY,eAAe,GACbF,SAAS,CAACV,MAAM,GAChBW,UAAU,CAACX,MAAM;MAEzB,MAAMkB,OAAO,GAAGnC,IAAI,CAACwB,KAAK,CAACC,IAAI,CAAC,GAAG,EAAEE,SAAS,EAAE,GAAG,EAAEN,WAAW,CAACF,KAAK,CAACa,eAAe,CAAC,CAAC;;MAExF;MACA,OAAOZ,gBAAgB,CAACe,OAAO,CAAC;IAClC,CAAC;EACH;EAEA,OAAOf,gBAAgB;AACzB;;AAEA;AACA;AACA,SAASd,WAAW,CAACH,MAAM,EAAEE,GAAG,EAAE;EAChC,IAAI,CAAC+B,KAAK,CAACC,OAAO,CAAChC,GAAG,CAACiC,IAAI,CAACC,QAAQ,CAAC,EAAE;IACrC,MAAM,IAAIzC,KAAK,CAACwB,KAAK,CAACxB,KAAK,CAACwB,KAAK,CAACC,YAAY,EAAE,2BAA2B,CAAC;EAC9E;;EAEA;EACA;EACA;EACA;EACA;EACA,IAAI,CAAClB,GAAG,CAACQ,WAAW,CAAC2B,QAAQ,CAACvC,SAAS,CAAC,EAAE;IACxC,MAAM,2DAA2D;EACnE;EAEA,MAAMmB,gBAAgB,GAAGR,4BAA4B,CACnDP,GAAG,CAACQ,WAAW,EACfR,GAAG,CAACoC,MAAM,CAAC3B,SAAS,EACpBT,GAAG,CAACoC,MAAM,CAAC1B,eAAe,CAC3B;EAED,MAAM2B,KAAK,GAAGC,kBAAkB,IAAI;IAClC,IAAIC,cAAc,GAAGC,OAAO,CAACC,OAAO,EAAE;IACtC,IAAIzC,GAAG,CAACiC,IAAI,CAACS,WAAW,KAAK,IAAI,EAAE;MACjCH,cAAc,GAAGvC,GAAG,CAACoC,MAAM,CAACO,QAAQ,CAACC,0BAA0B,EAAE;IACnE;IAEA,OAAOL,cAAc,CAACM,IAAI,CAAC,MAAM;MAC/B,MAAMC,QAAQ,GAAG9C,GAAG,CAACiC,IAAI,CAACC,QAAQ,CAACa,GAAG,CAACC,WAAW,IAAI;QACpD,MAAMC,YAAY,GAAGlC,gBAAgB,CAACiC,WAAW,CAACrD,IAAI,CAAC;;QAEvD;QACA,MAAMuD,OAAO,GAAG;UACdjB,IAAI,EAAEe,WAAW,CAACf,IAAI;UACtBG,MAAM,EAAEpC,GAAG,CAACoC,MAAM;UAClBe,IAAI,EAAEnD,GAAG,CAACmD,IAAI;UACdC,IAAI,EAAEpD,GAAG,CAACoD;QACZ,CAAC;QAED,OAAOtD,MAAM,CAACuD,eAAe,CAACL,WAAW,CAACM,MAAM,EAAEL,YAAY,EAAEC,OAAO,CAAC,CAACL,IAAI,CAC3EU,QAAQ,IAAI;UACV,OAAO;YAAEC,OAAO,EAAED,QAAQ,CAACA;UAAS,CAAC;QACvC,CAAC,EACDlD,KAAK,IAAI;UACP,OAAO;YAAEA,KAAK,EAAE;cAAEoD,IAAI,EAAEpD,KAAK,CAACoD,IAAI;cAAEpD,KAAK,EAAEA,KAAK,CAACqD;YAAQ;UAAE,CAAC;QAC9D,CAAC,CACF;MACH,CAAC,CAAC;MAEF,OAAOlB,OAAO,CAACmB,GAAG,CAACb,QAAQ,CAAC,CACzBD,IAAI,CAACe,OAAO,IAAI;QACf,IAAI5D,GAAG,CAACiC,IAAI,CAACS,WAAW,KAAK,IAAI,EAAE;UACjC,IAAIkB,OAAO,CAACC,IAAI,CAACC,MAAM,IAAI,OAAOA,MAAM,CAACzD,KAAK,KAAK,QAAQ,CAAC,EAAE;YAC5D,OAAOL,GAAG,CAACoC,MAAM,CAACO,QAAQ,CAACoB,yBAAyB,EAAE,CAAClB,IAAI,CAAC,MAAM;cAChE,OAAOL,OAAO,CAACwB,MAAM,CAAC;gBAAET,QAAQ,EAAEK;cAAQ,CAAC,CAAC;YAC9C,CAAC,CAAC;UACJ,CAAC,MAAM;YACL,OAAO5D,GAAG,CAACoC,MAAM,CAACO,QAAQ,CAACsB,0BAA0B,EAAE,CAACpB,IAAI,CAAC,MAAM;cACjE,OAAO;gBAAEU,QAAQ,EAAEK;cAAQ,CAAC;YAC9B,CAAC,CAAC;UACJ;QACF,CAAC,MAAM;UACL,OAAO;YAAEL,QAAQ,EAAEK;UAAQ,CAAC;QAC9B;MACF,CAAC,CAAC,CACDM,KAAK,CAAC7D,KAAK,IAAI;QACd,IACEA,KAAK,IACLA,KAAK,CAACkD,QAAQ,IACdlD,KAAK,CAACkD,QAAQ,CAACM,IAAI,CACjBM,SAAS,IAAI,OAAOA,SAAS,CAAC9D,KAAK,KAAK,QAAQ,IAAI8D,SAAS,CAAC9D,KAAK,CAACoD,IAAI,KAAK,GAAG,CACjF,IACDnB,kBAAkB,GAAG,CAAC,EACtB;UACA,OAAOD,KAAK,CAACC,kBAAkB,GAAG,CAAC,CAAC;QACtC;QACA,MAAMjC,KAAK;MACb,CAAC,CAAC;IACN,CAAC,CAAC;EACJ,CAAC;EACD,OAAOgC,KAAK,CAAC,CAAC,CAAC;AACjB;AAEA+B,MAAM,CAACC,OAAO,GAAG;EACfxE,SAAS;EACTU;AACF,CAAC"}
|
package/lib/cache.js
CHANGED
|
@@ -4,13 +4,11 @@ Object.defineProperty(exports, "__esModule", {
|
|
|
4
4
|
value: true
|
|
5
5
|
});
|
|
6
6
|
exports.default = exports.AppCache = void 0;
|
|
7
|
-
|
|
8
7
|
var _InMemoryCache = require("./Adapters/Cache/InMemoryCache");
|
|
9
|
-
|
|
10
8
|
var AppCache = new _InMemoryCache.InMemoryCache({
|
|
11
9
|
ttl: NaN
|
|
12
10
|
});
|
|
13
11
|
exports.AppCache = AppCache;
|
|
14
12
|
var _default = AppCache;
|
|
15
13
|
exports.default = _default;
|
|
16
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
14
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJBcHBDYWNoZSIsIkluTWVtb3J5Q2FjaGUiLCJ0dGwiLCJOYU4iXSwic291cmNlcyI6WyIuLi9zcmMvY2FjaGUuanMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSW5NZW1vcnlDYWNoZSB9IGZyb20gJy4vQWRhcHRlcnMvQ2FjaGUvSW5NZW1vcnlDYWNoZSc7XG5cbmV4cG9ydCB2YXIgQXBwQ2FjaGUgPSBuZXcgSW5NZW1vcnlDYWNoZSh7IHR0bDogTmFOIH0pO1xuZXhwb3J0IGRlZmF1bHQgQXBwQ2FjaGU7XG4iXSwibWFwcGluZ3MiOiI7Ozs7OztBQUFBO0FBRU8sSUFBSUEsUUFBUSxHQUFHLElBQUlDLDRCQUFhLENBQUM7RUFBRUMsR0FBRyxFQUFFQztBQUFJLENBQUMsQ0FBQztBQUFDO0FBQUEsZUFDdkNILFFBQVE7QUFBQSJ9
|