@nlabs/reaktor 0.1.5 → 0.1.7

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.
Files changed (52) hide show
  1. package/lib/config.js +20 -11
  2. package/lib/data/conversations.js +69 -39
  3. package/lib/data/dynamodb.js +60 -27
  4. package/lib/data/email.js +50 -19
  5. package/lib/data/files.js +103 -46
  6. package/lib/data/groups.js +114 -67
  7. package/lib/data/images.js +169 -102
  8. package/lib/data/index.js +234 -24
  9. package/lib/data/ios.js +48 -23
  10. package/lib/data/locations.js +45 -29
  11. package/lib/data/messages.d.ts +2 -2
  12. package/lib/data/messages.js +105 -72
  13. package/lib/data/notifications.js +31 -14
  14. package/lib/data/payments.js +164 -114
  15. package/lib/data/posts.js +144 -87
  16. package/lib/data/reactions.js +58 -33
  17. package/lib/data/s3.js +54 -24
  18. package/lib/data/search.js +32 -19
  19. package/lib/data/sms.js +35 -15
  20. package/lib/data/subscription.js +78 -51
  21. package/lib/data/tags.js +90 -51
  22. package/lib/data/users.js +119 -72
  23. package/lib/index.js +42 -8
  24. package/lib/types/apps.js +1 -1
  25. package/lib/types/arangodb.js +1 -1
  26. package/lib/types/auth.js +1 -1
  27. package/lib/types/conversations.js +1 -1
  28. package/lib/types/email.js +1 -1
  29. package/lib/types/files.js +1 -1
  30. package/lib/types/google.js +1 -1
  31. package/lib/types/groups.js +1 -1
  32. package/lib/types/images.js +1 -1
  33. package/lib/types/index.js +210 -22
  34. package/lib/types/locations.js +1 -1
  35. package/lib/types/messages.d.ts +12 -7
  36. package/lib/types/messages.js +1 -1
  37. package/lib/types/notifications.js +1 -1
  38. package/lib/types/payments.js +1 -1
  39. package/lib/types/posts.js +1 -1
  40. package/lib/types/reactions.js +1 -1
  41. package/lib/types/tags.js +1 -1
  42. package/lib/types/users.d.ts +0 -1
  43. package/lib/types/users.js +1 -1
  44. package/lib/utils/analytics.js +23 -11
  45. package/lib/utils/arangodb.js +48 -19
  46. package/lib/utils/auth.js +31 -15
  47. package/lib/utils/graphql.js +16 -4
  48. package/lib/utils/index.js +78 -11
  49. package/lib/utils/objects.js +30 -10
  50. package/lib/utils/redis.js +21 -8
  51. package/package.json +5 -5
  52. package/index.d.ts +0 -1
package/lib/data/users.js CHANGED
@@ -1,3 +1,24 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.getDisplayName = exports.getRelationship = exports.getRelationsByUser = exports.getUsersByRelations = exports.deleteUserRelation = exports.addUserRelation = exports.getUserList = exports.getUser = exports.getSessionUser = exports.deactivateUser = exports.deleteUser = exports.addUser = void 0;
7
+
8
+ var _utils = require("@nlabs/utils");
9
+
10
+ var _arangojs = require("arangojs");
11
+
12
+ var _isEmpty = _interopRequireDefault(require("lodash/isEmpty"));
13
+
14
+ var stripe = _interopRequireWildcard(require("stripe"));
15
+
16
+ var _utils2 = require("../utils");
17
+
18
+ function _interopRequireWildcard(obj) { if (obj && obj.__esModule) { return obj; } else { var newObj = {}; if (obj != null) { for (var key in obj) { if (Object.prototype.hasOwnProperty.call(obj, key)) { var desc = Object.defineProperty && Object.getOwnPropertyDescriptor ? Object.getOwnPropertyDescriptor(obj, key) : {}; if (desc.get || desc.set) { Object.defineProperty(newObj, key, desc); } else { newObj[key] = obj[key]; } } } } newObj.default = obj; return newObj; } }
19
+
20
+ function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
21
+
1
22
  function _templateObject8() {
2
23
  var data = _taggedTemplateLiteral(["FOR u, r IN INBOUND ", " hasRelation\n FILTER r._from == ", "\n LIMIT 1\n RETURN r"]);
3
24
 
@@ -80,17 +101,9 @@ function _templateObject() {
80
101
 
81
102
  function _taggedTemplateLiteral(strings, raw) { if (!raw) { raw = strings.slice(0); } return Object.freeze(Object.defineProperties(strings, { raw: { value: Object.freeze(raw) } })); }
82
103
 
83
- /**
84
- * Copyright (c) 2019-Present, Nitrogen Labs, Inc.
85
- * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.
86
- */
87
- import { createHash, parseChar, parseEmail, parseId, parseUsername, parseVarChar } from '@nlabs/utils';
88
- import { aql } from 'arangojs';
89
- import isEmpty from 'lodash/isEmpty';
90
- import * as stripe from 'stripe';
91
- import { getLimit, logError, logException, useDb } from '../utils';
92
104
  var eventCategory = 'users';
93
- export var addUser = function addUser(context, user) {
105
+
106
+ var addUser = function addUser(context, user) {
94
107
  var database = context.database;
95
108
  var email = user.email,
96
109
  phone = user.phone,
@@ -98,21 +111,24 @@ export var addUser = function addUser(context, user) {
98
111
  username = user.username; // Add new user properties
99
112
 
100
113
  var insert = {
101
- _key: createHash(sub, null),
114
+ _key: (0, _utils.createHash)(sub, null),
102
115
  added: Date.now(),
103
- email: email !== undefined ? parseEmail(email) : undefined,
116
+ email: email !== undefined ? (0, _utils.parseEmail)(email) : undefined,
104
117
  modified: Date.now(),
105
- phone: phone !== undefined ? parseVarChar(phone, 15) : undefined,
106
- sub: parseVarChar(sub, 32),
107
- username: parseUsername(username)
118
+ phone: phone !== undefined ? (0, _utils.parseVarChar)(phone, 15) : undefined,
119
+ sub: (0, _utils.parseVarChar)(sub, 32),
120
+ username: (0, _utils.parseUsername)(username)
108
121
  }; // Add new user in ArangoDB
109
122
 
110
- var aqlQry = aql(_templateObject(), insert);
111
- return useDb(database).query(aqlQry).then(function (cursor) {
123
+ var aqlQry = (0, _arangojs.aql)(_templateObject(), insert);
124
+ return (0, _utils2.useDb)(database).query(aqlQry).then(function (cursor) {
112
125
  return cursor.next();
113
126
  });
114
127
  };
115
- export var deleteUser = function deleteUser(context, userId) {
128
+
129
+ exports.addUser = addUser;
130
+
131
+ var deleteUser = function deleteUser(context, userId) {
116
132
  var action = 'delete';
117
133
  var database = context.database,
118
134
  sessionId = context.userId,
@@ -120,7 +136,7 @@ export var deleteUser = function deleteUser(context, userId) {
120
136
  var isAdmin = sessionType > 2;
121
137
 
122
138
  if (!isAdmin && sessionId !== userId) {
123
- logException({
139
+ (0, _utils2.logException)({
124
140
  action: action,
125
141
  category: eventCategory,
126
142
  label: 'unauthorized',
@@ -129,8 +145,8 @@ export var deleteUser = function deleteUser(context, userId) {
129
145
  return null;
130
146
  }
131
147
 
132
- var aqlQry = aql(_templateObject2(), userId);
133
- return useDb(database).query(aqlQry).then(function (cursor) {
148
+ var aqlQry = (0, _arangojs.aql)(_templateObject2(), userId);
149
+ return (0, _utils2.useDb)(database).query(aqlQry).then(function (cursor) {
134
150
  return cursor.next();
135
151
  }).then(function () {
136
152
  var user = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
@@ -143,7 +159,10 @@ export var deleteUser = function deleteUser(context, userId) {
143
159
  throw error;
144
160
  });
145
161
  };
146
- export var deactivateUser = function deactivateUser(context, userId) {
162
+
163
+ exports.deleteUser = deleteUser;
164
+
165
+ var deactivateUser = function deactivateUser(context, userId) {
147
166
  var action = 'delete';
148
167
  var database = context.database,
149
168
  sessionId = context.userId,
@@ -151,7 +170,7 @@ export var deactivateUser = function deactivateUser(context, userId) {
151
170
  var isAdmin = sessionType > 2;
152
171
 
153
172
  if (!isAdmin && sessionId !== userId) {
154
- logException({
173
+ (0, _utils2.logException)({
155
174
  action: action,
156
175
  category: eventCategory,
157
176
  label: 'unauthorized',
@@ -163,64 +182,72 @@ export var deactivateUser = function deactivateUser(context, userId) {
163
182
  var updated = {
164
183
  userType: 0
165
184
  };
166
- var aqlQry = aql(_templateObject3(), userId, updated);
167
- return useDb(database).query(aqlQry).then(function (cursor) {
185
+ var aqlQry = (0, _arangojs.aql)(_templateObject3(), userId, updated);
186
+ return (0, _utils2.useDb)(database).query(aqlQry).then(function (cursor) {
168
187
  return cursor.next();
169
188
  }).catch(function (error) {
170
189
  throw error;
171
190
  });
172
191
  };
173
- export var getSessionUser = function getSessionUser(context) {
192
+
193
+ exports.deactivateUser = deactivateUser;
194
+
195
+ var getSessionUser = function getSessionUser(context) {
174
196
  var action = 'getSessionUser';
175
197
  var database = context.database,
176
- sessionId = context.userId;
177
- console.log('getSessionUser::sessionId', sessionId); // Get data from database
198
+ sessionId = context.userId; // Get data from database
178
199
 
179
- var aqlQry = aql(_templateObject4(), sessionId);
180
- return useDb(database).query(aqlQry).then(function (cursor) {
200
+ var aqlQry = (0, _arangojs.aql)(_templateObject4(), sessionId);
201
+ return (0, _utils2.useDb)(database).query(aqlQry).then(function (cursor) {
181
202
  return cursor.next();
182
203
  }).then(function () {
183
204
  var user = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
184
205
  return user;
185
206
  }).catch(function (error) {
186
- return logError({
207
+ return (0, _utils2.logError)({
187
208
  action: action,
188
209
  category: eventCategory,
189
210
  label: 'db_error'
190
211
  }, error, context).then(function () {});
191
212
  });
192
213
  };
193
- export var getUser = function getUser(context, userId) {
214
+
215
+ exports.getSessionUser = getSessionUser;
216
+
217
+ var getUser = function getUser(context, userId) {
194
218
  var action = 'getItem';
195
- var formatUserId = parseId(userId);
219
+ var formatUserId = (0, _utils.parseId)(userId);
196
220
  var database = context.database; // Get data from database
197
221
 
198
- var aqlQry = aql(_templateObject5(), formatUserId);
199
- return useDb(database).query(aqlQry).then(function (cursor) {
222
+ var aqlQry = (0, _arangojs.aql)(_templateObject5(), formatUserId);
223
+ return (0, _utils2.useDb)(database).query(aqlQry).then(function (cursor) {
200
224
  return cursor.next();
201
225
  }).then(function () {
202
226
  var user = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
203
227
  return user;
204
228
  }).catch(function (error) {
205
- return logError({
229
+ return (0, _utils2.logError)({
206
230
  action: action,
207
231
  category: eventCategory,
208
232
  label: 'db_error'
209
233
  }, error, context).then(function () {});
210
234
  });
211
235
  };
212
- export var getUserList = function getUserList(context) {
236
+
237
+ exports.getUser = getUser;
238
+
239
+ var getUserList = function getUserList(context) {
213
240
  var from = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 0;
214
241
  var to = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : 15;
215
242
  var action = 'getItem';
216
243
  var database = context.database;
217
- var limit = getLimit(from, to); // Get data from database
244
+ var limit = (0, _utils2.getLimit)(from, to); // Get data from database
218
245
 
219
246
  var aqlQry = "FOR u IN users\n ".concat(limit.aql, "\n SORT u.username\n RETURN u");
220
- return useDb(database).query(aqlQry).then(function (cursor) {
247
+ return (0, _utils2.useDb)(database).query(aqlQry).then(function (cursor) {
221
248
  return cursor.all();
222
249
  }).catch(function (error) {
223
- return logError({
250
+ return (0, _utils2.logError)({
224
251
  action: action,
225
252
  category: eventCategory,
226
253
  label: 'db_error'
@@ -229,14 +256,17 @@ export var getUserList = function getUserList(context) {
229
256
  });
230
257
  });
231
258
  };
232
- export var addUserRelation = function addUserRelation(context, userId, value) {
259
+
260
+ exports.getUserList = getUserList;
261
+
262
+ var addUserRelation = function addUserRelation(context, userId, value) {
233
263
  var action = 'addRelation';
234
264
  var database = context.database,
235
265
  sessionId = context.userId;
236
- var formatUserId = parseId(userId);
237
- var formatValue = parseChar(value, 32);
238
- var db = useDb(database);
239
- var aqlQry = aql(_templateObject6(), formatValue, sessionId, formatUserId);
266
+ var formatUserId = (0, _utils.parseId)(userId);
267
+ var formatValue = (0, _utils.parseChar)(value, 32);
268
+ var db = (0, _utils2.useDb)(database);
269
+ var aqlQry = (0, _arangojs.aql)(_templateObject6(), formatValue, sessionId, formatUserId);
240
270
  return db.query(aqlQry).then(function (cursor) {
241
271
  return cursor.next();
242
272
  }).then(function (relation) {
@@ -246,7 +276,7 @@ export var addUserRelation = function addUserRelation(context, userId, value) {
246
276
  } // Otherwise, create the new relationship
247
277
 
248
278
 
249
- var edgeId = createHash("relation-".concat(sessionId, "-").concat(formatUserId, "-").concat(formatValue));
279
+ var edgeId = (0, _utils.createHash)("relation-".concat(sessionId, "-").concat(formatUserId, "-").concat(formatValue));
250
280
  var edge = {
251
281
  _key: edgeId,
252
282
  added: Date.now(),
@@ -259,7 +289,7 @@ export var addUserRelation = function addUserRelation(context, userId, value) {
259
289
  throw error;
260
290
  });
261
291
  }).catch(function (error) {
262
- return logError({
292
+ return (0, _utils2.logError)({
263
293
  action: action,
264
294
  category: eventCategory,
265
295
  label: 'db_error'
@@ -268,20 +298,23 @@ export var addUserRelation = function addUserRelation(context, userId, value) {
268
298
  });
269
299
  });
270
300
  };
271
- export var deleteUserRelation = function deleteUserRelation(context, userId, value) {
301
+
302
+ exports.addUserRelation = addUserRelation;
303
+
304
+ var deleteUserRelation = function deleteUserRelation(context, userId, value) {
272
305
  var action = 'getRelationsByUser';
273
306
  var database = context.database,
274
307
  sessionId = context.userId;
275
- var formatUserId = parseId(userId);
276
- var formatValue = parseChar(value, 32);
308
+ var formatUserId = (0, _utils.parseId)(userId);
309
+ var formatValue = (0, _utils.parseChar)(value, 32);
277
310
  var aqlQry = "FOR u, r IN INBOUND \"".concat("users/".concat(formatUserId), "\" hasRelation\n FILTER r._from == \"", "users/".concat(sessionId), "\" && r.value == \"", formatValue, "\"\n REMOVE r IN hasRelation\n RETURN OLD");
278
- return useDb(database).query(aqlQry).then(function (cursor) {
311
+ return (0, _utils2.useDb)(database).query(aqlQry).then(function (cursor) {
279
312
  return cursor.next();
280
313
  }).then(function () {
281
314
  var relation = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
282
315
  return relation;
283
316
  }).catch(function (error) {
284
- return logError({
317
+ return (0, _utils2.logError)({
285
318
  action: action,
286
319
  category: eventCategory,
287
320
  label: 'db_error'
@@ -290,17 +323,20 @@ export var deleteUserRelation = function deleteUserRelation(context, userId, val
290
323
  });
291
324
  });
292
325
  };
293
- export var getUsersByRelations = function getUsersByRelations(context, userId, type, from, to) {
326
+
327
+ exports.deleteUserRelation = deleteUserRelation;
328
+
329
+ var getUsersByRelations = function getUsersByRelations(context, userId, type, from, to) {
294
330
  var action = 'getUsersByRelations';
295
331
  var database = context.database;
296
- var formatUserId = parseId(userId);
297
- var formatType = parseChar(type, 32);
298
- var limit = getLimit(from, to);
332
+ var formatUserId = (0, _utils.parseId)(userId);
333
+ var formatType = (0, _utils.parseChar)(type, 32);
334
+ var limit = (0, _utils2.getLimit)(from, to);
299
335
  var aqlQry = "FOR u, r IN INBOUND \"".concat("users/".concat(formatUserId), "\" hasRelation\n FILTER r.type == \"", formatType, "\"\n ").concat(limit.aql, "\n RETURN u");
300
- return useDb(database).query(aqlQry).then(function (cursor) {
336
+ return (0, _utils2.useDb)(database).query(aqlQry).then(function (cursor) {
301
337
  return cursor.all();
302
338
  }).catch(function (error) {
303
- return logError({
339
+ return (0, _utils2.logError)({
304
340
  action: action,
305
341
  category: eventCategory,
306
342
  label: 'db_error'
@@ -309,16 +345,19 @@ export var getUsersByRelations = function getUsersByRelations(context, userId, t
309
345
  });
310
346
  });
311
347
  };
312
- export var getRelationsByUser = function getRelationsByUser(context, userId) {
348
+
349
+ exports.getUsersByRelations = getUsersByRelations;
350
+
351
+ var getRelationsByUser = function getRelationsByUser(context, userId) {
313
352
  var action = 'getRelationsByUser';
314
353
  var database = context.database,
315
354
  sessionId = context.userId;
316
- var formatUserId = parseId(userId);
317
- var aqlQry = aql(_templateObject7(), "users/".concat(formatUserId), "users/".concat(sessionId));
318
- return useDb(database).query(aqlQry).then(function (cursor) {
355
+ var formatUserId = (0, _utils.parseId)(userId);
356
+ var aqlQry = (0, _arangojs.aql)(_templateObject7(), "users/".concat(formatUserId), "users/".concat(sessionId));
357
+ return (0, _utils2.useDb)(database).query(aqlQry).then(function (cursor) {
319
358
  return cursor.all();
320
359
  }).catch(function (error) {
321
- return logError({
360
+ return (0, _utils2.logError)({
322
361
  action: action,
323
362
  category: eventCategory,
324
363
  label: 'db_error'
@@ -327,16 +366,19 @@ export var getRelationsByUser = function getRelationsByUser(context, userId) {
327
366
  });
328
367
  });
329
368
  };
330
- export var getRelationship = function getRelationship(context, userId) {
369
+
370
+ exports.getRelationsByUser = getRelationsByUser;
371
+
372
+ var getRelationship = function getRelationship(context, userId) {
331
373
  var action = 'getRelationship';
332
374
  var database = context.database,
333
375
  sessionId = context.userId;
334
- var formatUserId = parseId(userId);
335
- var aqlQry = aql(_templateObject8(), "users/".concat(formatUserId), "users/".concat(sessionId));
336
- return useDb(database).query(aqlQry).then(function (cursor) {
376
+ var formatUserId = (0, _utils.parseId)(userId);
377
+ var aqlQry = (0, _arangojs.aql)(_templateObject8(), "users/".concat(formatUserId), "users/".concat(sessionId));
378
+ return (0, _utils2.useDb)(database).query(aqlQry).then(function (cursor) {
337
379
  return cursor.next();
338
380
  }).catch(function (error) {
339
- return logError({
381
+ return (0, _utils2.logError)({
340
382
  action: action,
341
383
  category: eventCategory,
342
384
  label: 'db_error'
@@ -345,7 +387,10 @@ export var getRelationship = function getRelationship(context, userId) {
345
387
  });
346
388
  });
347
389
  };
348
- export var getDisplayName = function getDisplayName() {
390
+
391
+ exports.getRelationship = getRelationship;
392
+
393
+ var getDisplayName = function getDisplayName() {
349
394
  var user = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
350
395
  var first = user.first,
351
396
  last = user.last,
@@ -355,14 +400,16 @@ export var getDisplayName = function getDisplayName() {
355
400
  username = _user$username === void 0 ? '' : _user$username;
356
401
  var fullname = [first, last].join(' ').trim();
357
402
 
358
- if (!isEmpty(name)) {
403
+ if (!(0, _isEmpty.default)(name)) {
359
404
  return name;
360
405
  } else if (fullname !== '') {
361
406
  return fullname;
362
- } else if (!isEmpty(username)) {
407
+ } else if (!(0, _isEmpty.default)(username)) {
363
408
  return username;
364
409
  }
365
410
 
366
411
  return 'Unknown';
367
412
  };
368
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/data/users.ts"],"names":["createHash","parseChar","parseEmail","parseId","parseUsername","parseVarChar","aql","isEmpty","stripe","getLimit","logError","logException","useDb","eventCategory","addUser","context","user","database","email","phone","sub","username","insert","_key","added","Date","now","undefined","modified","aqlQry","query","then","cursor","next","deleteUser","userId","action","sessionId","sessionType","userType","isAdmin","category","label","value","customers","del","stripeCustomerId","accounts","stripeAccountId","catch","error","deactivateUser","updated","getSessionUser","console","log","getUser","formatUserId","getUserList","from","to","limit","all","addUserRelation","formatValue","db","relation","edgeId","edge","edgeCollection","save","deleteUserRelation","getUsersByRelations","type","formatType","getRelationsByUser","getRelationship","getDisplayName","first","last","name","fullname","join","trim"],"mappings":"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","sourcesContent":["/**\n * Copyright (c) 2019-Present, Nitrogen Labs, Inc.\n * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.\n */\nimport {createHash, parseChar, parseEmail, parseId, parseUsername, parseVarChar} from '@nlabs/utils';\nimport {aql, Database, EdgeCollection} from 'arangojs';\nimport {AqlQuery} from 'arangojs/lib/cjs/aql-query';\nimport {ArrayCursor} from 'arangojs/lib/cjs/cursor';\nimport isEmpty from 'lodash/isEmpty';\nimport * as stripe from 'stripe';\n\nimport {ApiContext} from '../types/auth';\nimport {UserRelationType, UserType} from '../types/users';\nimport {getLimit, logError, logException, useDb} from '../utils';\n\nconst eventCategory: string = 'users';\n\nexport const addUser = (context: ApiContext, user: UserType): Promise<UserType> => {\n  const {database} = context;\n  const {email, phone, sub, username} = user;\n\n  // Add new user properties\n  const insert: UserType = {\n    _key: createHash(sub, null),\n    added: Date.now(),\n    email: email !== undefined ? parseEmail(email) : undefined,\n    modified: Date.now(),\n    phone: phone !== undefined ? parseVarChar(phone, 15) : undefined,\n    sub: parseVarChar(sub, 32),\n    username: parseUsername(username)\n  };\n\n  // Add new user in ArangoDB\n  const aqlQry: AqlQuery = aql`INSERT ${insert} IN users RETURN NEW`;\n  return useDb(database).query(aqlQry).then((cursor: ArrayCursor) => cursor.next());\n};\n\nexport const deleteUser = (context: ApiContext, userId: string): Promise<UserType> => {\n  const action: string = 'delete';\n  const {database, userId: sessionId, userType: sessionType} = context;\n  const isAdmin: boolean = sessionType > 2;\n\n  if(!isAdmin && (sessionId !== userId)) {\n    logException({\n      action,\n      category: eventCategory,\n      label: 'unauthorized',\n      value: 'invalid_session'\n    }, context);\n    return null;\n  }\n\n  const aqlQry: AqlQuery = aql`FOR u IN users\n      FILTER u._key == ${userId}\n      LIMIT 1\n      REMOVE u IN users\n      RETURN OLD`;\n\n  return useDb(database).query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((user: UserType = {}) => stripe.customers.del(user.stripeCustomerId)\n      .then(() => stripe.accounts.del(user.stripeAccountId))\n      .then(() => user))\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const deactivateUser = (context: ApiContext, userId: string): Promise<UserType> => {\n  const action: string = 'delete';\n  const {database, userId: sessionId, userType: sessionType} = context;\n  const isAdmin: boolean = sessionType > 2;\n\n  if(!isAdmin && (sessionId !== userId)) {\n    logException({\n      action,\n      category: eventCategory,\n      label: 'unauthorized',\n      value: 'invalid_session'\n    }, context);\n    return null;\n  }\n\n  const updated: UserType = {\n    userType: 0\n  };\n  const aqlQry: AqlQuery = aql`UPDATE ${userId} WITH ${updated} IN users LIMIT 1 RETURN NEW`;\n\n  return useDb(database).query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const getSessionUser = (context: ApiContext): Promise<UserType> => {\n  const action: string = 'getSessionUser';\n  const {database, userId: sessionId} = context;\n\n  console.log('getSessionUser::sessionId', sessionId);\n  // Get data from database\n  const aqlQry = aql`FOR u IN users\n    FILTER u._key == ${sessionId}\n    LIMIT 1\n    RETURN u`;\n\n  return useDb(database).query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((user = {}) => user)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => {}));\n};\n\nexport const getUser = (context: ApiContext, userId): Promise<UserType> => {\n  const action: string = 'getItem';\n  const formatUserId: string = parseId(userId);\n  const {database} = context;\n\n  // Get data from database\n  const aqlQry = aql`FOR u IN users\n    FILTER u._key == ${formatUserId}\n    LIMIT 1\n    RETURN u`;\n\n  return useDb(database).query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((user = {}) => user)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => {}));\n};\n\nexport const getUserList = (context: ApiContext, from: number = 0, to: number = 15): Promise<UserType> => {\n  const action: string = 'getItem';\n  const {database} = context;\n  const limit = getLimit(from, to);\n\n  // Get data from database\n  const aqlQry: string = `FOR u IN users\n    ${limit.aql}\n    SORT u.username\n    RETURN u`;\n\n  return useDb(database).query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => []));\n};\n\nexport const addUserRelation = (context: ApiContext, userId, value): Promise<UserRelationType> => {\n  const action: string = 'addRelation';\n  const {database, userId: sessionId} = context;\n  const formatUserId: string = parseId(userId);\n  const formatValue: string = parseChar(value, 32);\n  const db: Database = useDb(database);\n  const aqlQry = aql`FOR r IN hasRelation\n    FILTER r.value == ${formatValue} && r._from == ${sessionId} && r._to == ${formatUserId}\n    LIMIT 1\n    RETURN r`;\n\n  return db.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((relation: UserRelationType) => {\n      // If a relationship between two users with the value exists, just return\n      if(relation) {\n        return relation;\n      }\n\n      // Otherwise, create the new relationship\n      const edgeId = createHash(`relation-${sessionId}-${formatUserId}-${formatValue}`);\n      const edge = {\n        _key: edgeId,\n        added: Date.now(),\n        value: formatValue\n      };\n      const edgeCollection: EdgeCollection = db.edgeCollection('hasRelation') as any;\n\n      return edgeCollection.save(edge, `users/${sessionId}`, `users/${formatUserId}`)\n        .then(() => edgeCollection.edge(edgeId))\n        .catch((error: Error) => {\n          throw error;\n        });\n    })\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const deleteUserRelation = (context: ApiContext, userId: string, value: string): Promise<UserRelationType> => {\n  const action: string = 'getRelationsByUser';\n  const {database, userId: sessionId} = context;\n  const formatUserId: string = parseId(userId);\n  const formatValue: string = parseChar(value, 32);\n  const aqlQry = `FOR u, r IN INBOUND \"${`users/${formatUserId}`}\" hasRelation\n    FILTER r._from == \"${`users/${sessionId}`}\" && r.value == \"${formatValue}\"\n    REMOVE r IN hasRelation\n    RETURN OLD`;\n\n  return useDb(database).query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((relation = {}) => relation)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const getUsersByRelations = (\n  context: ApiContext,\n  userId: string,\n  type: string,\n  from: number,\n  to: number\n): Promise<UserType[]> => {\n  const action: string = 'getUsersByRelations';\n  const {database} = context;\n  const formatUserId: string = parseId(userId);\n  const formatType: string = parseChar(type, 32);\n  const limit = getLimit(from, to);\n  const aqlQry: string = `FOR u, r IN INBOUND \"${`users/${formatUserId}`}\" hasRelation\n    FILTER r.type == \"${formatType}\"\n    ${limit.aql}\n    RETURN u`;\n\n  return useDb(database).query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const getRelationsByUser = (context: ApiContext, userId: string): Promise<UserRelationType[]> => {\n  const action: string = 'getRelationsByUser';\n  const {database, userId: sessionId} = context;\n  const formatUserId: string = parseId(userId);\n  const aqlQry: AqlQuery = aql`FOR u, r IN INBOUND ${`users/${formatUserId}`} hasRelation\n      FILTER r._from == ${`users/${sessionId}`}\n      RETURN r`;\n\n  return useDb(database).query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const getRelationship = (context: ApiContext, userId: string): Promise<UserRelationType> => {\n  const action: string = 'getRelationship';\n  const {database, userId: sessionId} = context;\n  const formatUserId: string = parseId(userId);\n  const aqlQry: AqlQuery = aql`FOR u, r IN INBOUND ${`users/${formatUserId}`} hasRelation\n      FILTER r._from == ${`users/${sessionId}`}\n      LIMIT 1\n      RETURN r`;\n\n  return useDb(database).query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const getDisplayName = (user: UserType = {}): string => {\n  const {first, last, name = '', username = ''} = user;\n  const fullname: string = ([first, last]).join(' ').trim();\n\n  if(!isEmpty(name)) {\n    return name;\n  } else if(fullname !== '') {\n    return fullname;\n  } else if(!isEmpty(username)) {\n    return username;\n  }\n\n  return 'Unknown';\n};\n"]}
413
+
414
+ exports.getDisplayName = getDisplayName;
415
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/data/users.ts"],"names":["eventCategory","addUser","context","user","database","email","phone","sub","username","insert","_key","added","Date","now","undefined","modified","aqlQry","aql","query","then","cursor","next","deleteUser","userId","action","sessionId","sessionType","userType","isAdmin","category","label","value","stripe","customers","del","stripeCustomerId","accounts","stripeAccountId","catch","error","deactivateUser","updated","getSessionUser","getUser","formatUserId","getUserList","from","to","limit","all","addUserRelation","formatValue","db","relation","edgeId","edge","edgeCollection","save","deleteUserRelation","getUsersByRelations","type","formatType","getRelationsByUser","getRelationship","getDisplayName","first","last","name","fullname","join","trim"],"mappings":"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","sourcesContent":["/**\n * Copyright (c) 2019-Present, Nitrogen Labs, Inc.\n * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.\n */\nimport {createHash, parseChar, parseEmail, parseId, parseUsername, parseVarChar} from '@nlabs/utils';\nimport {aql, Database, EdgeCollection} from 'arangojs';\nimport {AqlQuery} from 'arangojs/lib/cjs/aql-query';\nimport {ArrayCursor} from 'arangojs/lib/cjs/cursor';\nimport isEmpty from 'lodash/isEmpty';\nimport * as stripe from 'stripe';\n\nimport {ApiContext} from '../types/auth';\nimport {UserRelationType, UserType} from '../types/users';\nimport {getLimit, logError, logException, useDb} from '../utils';\n\nconst eventCategory: string = 'users';\n\nexport const addUser = (context: ApiContext, user: UserType): Promise<UserType> => {\n  const {database} = context;\n  const {email, phone, sub, username} = user;\n\n  // Add new user properties\n  const insert: UserType = {\n    _key: createHash(sub, null),\n    added: Date.now(),\n    email: email !== undefined ? parseEmail(email) : undefined,\n    modified: Date.now(),\n    phone: phone !== undefined ? parseVarChar(phone, 15) : undefined,\n    sub: parseVarChar(sub, 32),\n    username: parseUsername(username)\n  };\n\n  // Add new user in ArangoDB\n  const aqlQry: AqlQuery = aql`INSERT ${insert} IN users RETURN NEW`;\n  return useDb(database).query(aqlQry).then((cursor: ArrayCursor) => cursor.next());\n};\n\nexport const deleteUser = (context: ApiContext, userId: string): Promise<UserType> => {\n  const action: string = 'delete';\n  const {database, userId: sessionId, userType: sessionType} = context;\n  const isAdmin: boolean = sessionType > 2;\n\n  if(!isAdmin && (sessionId !== userId)) {\n    logException({\n      action,\n      category: eventCategory,\n      label: 'unauthorized',\n      value: 'invalid_session'\n    }, context);\n    return null;\n  }\n\n  const aqlQry: AqlQuery = aql`FOR u IN users\n      FILTER u._key == ${userId}\n      LIMIT 1\n      REMOVE u IN users\n      RETURN OLD`;\n\n  return useDb(database).query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((user: UserType = {}) => stripe.customers.del(user.stripeCustomerId)\n      .then(() => stripe.accounts.del(user.stripeAccountId))\n      .then(() => user))\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const deactivateUser = (context: ApiContext, userId: string): Promise<UserType> => {\n  const action: string = 'delete';\n  const {database, userId: sessionId, userType: sessionType} = context;\n  const isAdmin: boolean = sessionType > 2;\n\n  if(!isAdmin && (sessionId !== userId)) {\n    logException({\n      action,\n      category: eventCategory,\n      label: 'unauthorized',\n      value: 'invalid_session'\n    }, context);\n    return null;\n  }\n\n  const updated: UserType = {\n    userType: 0\n  };\n  const aqlQry: AqlQuery = aql`UPDATE ${userId} WITH ${updated} IN users LIMIT 1 RETURN NEW`;\n\n  return useDb(database).query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const getSessionUser = (context: ApiContext): Promise<UserType> => {\n  const action: string = 'getSessionUser';\n  const {database, userId: sessionId} = context;\n\n  // Get data from database\n  const aqlQry = aql`FOR u IN users\n    FILTER u._key == ${sessionId}\n    LIMIT 1\n    RETURN u`;\n\n  return useDb(database).query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((user = {}) => user)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => {}));\n};\n\nexport const getUser = (context: ApiContext, userId): Promise<UserType> => {\n  const action: string = 'getItem';\n  const formatUserId: string = parseId(userId);\n  const {database} = context;\n\n  // Get data from database\n  const aqlQry = aql`FOR u IN users\n    FILTER u._key == ${formatUserId}\n    LIMIT 1\n    RETURN u`;\n\n  return useDb(database).query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((user = {}) => user)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => {}));\n};\n\nexport const getUserList = (context: ApiContext, from: number = 0, to: number = 15): Promise<UserType> => {\n  const action: string = 'getItem';\n  const {database} = context;\n  const limit = getLimit(from, to);\n\n  // Get data from database\n  const aqlQry: string = `FOR u IN users\n    ${limit.aql}\n    SORT u.username\n    RETURN u`;\n\n  return useDb(database).query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => []));\n};\n\nexport const addUserRelation = (context: ApiContext, userId, value): Promise<UserRelationType> => {\n  const action: string = 'addRelation';\n  const {database, userId: sessionId} = context;\n  const formatUserId: string = parseId(userId);\n  const formatValue: string = parseChar(value, 32);\n  const db: Database = useDb(database);\n  const aqlQry = aql`FOR r IN hasRelation\n    FILTER r.value == ${formatValue} && r._from == ${sessionId} && r._to == ${formatUserId}\n    LIMIT 1\n    RETURN r`;\n\n  return db.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((relation: UserRelationType) => {\n      // If a relationship between two users with the value exists, just return\n      if(relation) {\n        return relation;\n      }\n\n      // Otherwise, create the new relationship\n      const edgeId = createHash(`relation-${sessionId}-${formatUserId}-${formatValue}`);\n      const edge = {\n        _key: edgeId,\n        added: Date.now(),\n        value: formatValue\n      };\n      const edgeCollection: EdgeCollection = db.edgeCollection('hasRelation') as any;\n\n      return edgeCollection.save(edge, `users/${sessionId}`, `users/${formatUserId}`)\n        .then(() => edgeCollection.edge(edgeId))\n        .catch((error: Error) => {\n          throw error;\n        });\n    })\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const deleteUserRelation = (context: ApiContext, userId: string, value: string): Promise<UserRelationType> => {\n  const action: string = 'getRelationsByUser';\n  const {database, userId: sessionId} = context;\n  const formatUserId: string = parseId(userId);\n  const formatValue: string = parseChar(value, 32);\n  const aqlQry = `FOR u, r IN INBOUND \"${`users/${formatUserId}`}\" hasRelation\n    FILTER r._from == \"${`users/${sessionId}`}\" && r.value == \"${formatValue}\"\n    REMOVE r IN hasRelation\n    RETURN OLD`;\n\n  return useDb(database).query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((relation = {}) => relation)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const getUsersByRelations = (\n  context: ApiContext,\n  userId: string,\n  type: string,\n  from: number,\n  to: number\n): Promise<UserType[]> => {\n  const action: string = 'getUsersByRelations';\n  const {database} = context;\n  const formatUserId: string = parseId(userId);\n  const formatType: string = parseChar(type, 32);\n  const limit = getLimit(from, to);\n  const aqlQry: string = `FOR u, r IN INBOUND \"${`users/${formatUserId}`}\" hasRelation\n    FILTER r.type == \"${formatType}\"\n    ${limit.aql}\n    RETURN u`;\n\n  return useDb(database).query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const getRelationsByUser = (context: ApiContext, userId: string): Promise<UserRelationType[]> => {\n  const action: string = 'getRelationsByUser';\n  const {database, userId: sessionId} = context;\n  const formatUserId: string = parseId(userId);\n  const aqlQry: AqlQuery = aql`FOR u, r IN INBOUND ${`users/${formatUserId}`} hasRelation\n      FILTER r._from == ${`users/${sessionId}`}\n      RETURN r`;\n\n  return useDb(database).query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const getRelationship = (context: ApiContext, userId: string): Promise<UserRelationType> => {\n  const action: string = 'getRelationship';\n  const {database, userId: sessionId} = context;\n  const formatUserId: string = parseId(userId);\n  const aqlQry: AqlQuery = aql`FOR u, r IN INBOUND ${`users/${formatUserId}`} hasRelation\n      FILTER r._from == ${`users/${sessionId}`}\n      LIMIT 1\n      RETURN r`;\n\n  return useDb(database).query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const getDisplayName = (user: UserType = {}): string => {\n  const {first, last, name = '', username = ''} = user;\n  const fullname: string = ([first, last]).join(' ').trim();\n\n  if(!isEmpty(name)) {\n    return name;\n  } else if(fullname !== '') {\n    return fullname;\n  } else if(!isEmpty(username)) {\n    return username;\n  }\n\n  return 'Unknown';\n};\n"]}
package/lib/index.js CHANGED
@@ -1,8 +1,42 @@
1
- /**
2
- * Copyright (c) 2019-Present, Nitrogen Labs, Inc.
3
- * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.
4
- */
5
- export * from './data';
6
- export * from './utils';
7
- export * from './types';
8
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uL3NyYy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7OztBQUlBLGNBQWMsUUFBZDtBQUNBLGNBQWMsU0FBZDtBQUNBLGNBQWMsU0FBZCIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQ29weXJpZ2h0IChjKSAyMDE5LVByZXNlbnQsIE5pdHJvZ2VuIExhYnMsIEluYy5cbiAqIENvcHlyaWdodHMgbGljZW5zZWQgdW5kZXIgdGhlIE1JVCBMaWNlbnNlLiBTZWUgdGhlIGFjY29tcGFueWluZyBMSUNFTlNFIGZpbGUgZm9yIHRlcm1zLlxuICovXG5leHBvcnQgKiBmcm9tICcuL2RhdGEnO1xuZXhwb3J0ICogZnJvbSAnLi91dGlscyc7XG5leHBvcnQgKiBmcm9tICcuL3R5cGVzJztcbiJdfQ==
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+
7
+ var _data = require("./data");
8
+
9
+ Object.keys(_data).forEach(function (key) {
10
+ if (key === "default" || key === "__esModule") return;
11
+ Object.defineProperty(exports, key, {
12
+ enumerable: true,
13
+ get: function get() {
14
+ return _data[key];
15
+ }
16
+ });
17
+ });
18
+
19
+ var _utils = require("./utils");
20
+
21
+ Object.keys(_utils).forEach(function (key) {
22
+ if (key === "default" || key === "__esModule") return;
23
+ Object.defineProperty(exports, key, {
24
+ enumerable: true,
25
+ get: function get() {
26
+ return _utils[key];
27
+ }
28
+ });
29
+ });
30
+
31
+ var _types = require("./types");
32
+
33
+ Object.keys(_types).forEach(function (key) {
34
+ if (key === "default" || key === "__esModule") return;
35
+ Object.defineProperty(exports, key, {
36
+ enumerable: true,
37
+ get: function get() {
38
+ return _types[key];
39
+ }
40
+ });
41
+ });
42
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uL3NyYy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7QUFJQTs7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7O0FBQ0E7O0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBOztBQUNBOztBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQSIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQ29weXJpZ2h0IChjKSAyMDE5LVByZXNlbnQsIE5pdHJvZ2VuIExhYnMsIEluYy5cbiAqIENvcHlyaWdodHMgbGljZW5zZWQgdW5kZXIgdGhlIE1JVCBMaWNlbnNlLiBTZWUgdGhlIGFjY29tcGFueWluZyBMSUNFTlNFIGZpbGUgZm9yIHRlcm1zLlxuICovXG5leHBvcnQgKiBmcm9tICcuL2RhdGEnO1xuZXhwb3J0ICogZnJvbSAnLi91dGlscyc7XG5leHBvcnQgKiBmcm9tICcuL3R5cGVzJztcbiJdfQ==
package/lib/types/apps.js CHANGED
@@ -1,2 +1,2 @@
1
-
1
+ "use strict";
2
2
  //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbXX0=
@@ -1,2 +1,2 @@
1
-
1
+ "use strict";
2
2
  //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbXX0=
package/lib/types/auth.js CHANGED
@@ -1,2 +1,2 @@
1
-
1
+ "use strict";
2
2
  //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbXX0=
@@ -1,2 +1,2 @@
1
-
1
+ "use strict";
2
2
  //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbXX0=
@@ -1,2 +1,2 @@
1
-
1
+ "use strict";
2
2
  //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbXX0=
@@ -1,2 +1,2 @@
1
-
1
+ "use strict";
2
2
  //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbXX0=
@@ -1,2 +1,2 @@
1
-
1
+ "use strict";
2
2
  //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbXX0=
@@ -1,2 +1,2 @@
1
-
1
+ "use strict";
2
2
  //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbXX0=
@@ -1,2 +1,2 @@
1
-
1
+ "use strict";
2
2
  //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbXX0=