@nlabs/reaktor 0.1.6 → 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.
- package/lib/config.js +20 -11
- package/lib/data/conversations.js +69 -39
- package/lib/data/dynamodb.js +60 -27
- package/lib/data/email.js +50 -19
- package/lib/data/files.js +103 -46
- package/lib/data/groups.js +114 -67
- package/lib/data/images.js +168 -100
- package/lib/data/index.js +234 -24
- package/lib/data/ios.js +48 -23
- package/lib/data/locations.js +45 -29
- package/lib/data/messages.d.ts +2 -2
- package/lib/data/messages.js +105 -72
- package/lib/data/notifications.js +31 -14
- package/lib/data/payments.js +164 -114
- package/lib/data/posts.js +144 -87
- package/lib/data/reactions.js +58 -33
- package/lib/data/s3.js +54 -24
- package/lib/data/search.js +32 -19
- package/lib/data/sms.js +35 -15
- package/lib/data/subscription.js +78 -51
- package/lib/data/tags.js +90 -51
- package/lib/data/users.js +118 -70
- package/lib/index.js +42 -8
- package/lib/types/apps.js +1 -1
- package/lib/types/arangodb.js +1 -1
- package/lib/types/auth.js +1 -1
- package/lib/types/conversations.js +1 -1
- package/lib/types/email.js +1 -1
- package/lib/types/files.js +1 -1
- package/lib/types/google.js +1 -1
- package/lib/types/groups.js +1 -1
- package/lib/types/images.js +1 -1
- package/lib/types/index.js +210 -22
- package/lib/types/locations.js +1 -1
- package/lib/types/messages.d.ts +12 -7
- package/lib/types/messages.js +1 -1
- package/lib/types/notifications.js +1 -1
- package/lib/types/payments.js +1 -1
- package/lib/types/posts.js +1 -1
- package/lib/types/reactions.js +1 -1
- package/lib/types/tags.js +1 -1
- package/lib/types/users.d.ts +0 -1
- package/lib/types/users.js +1 -1
- package/lib/utils/analytics.js +23 -11
- package/lib/utils/arangodb.js +48 -19
- package/lib/utils/auth.js +31 -15
- package/lib/utils/graphql.js +16 -4
- package/lib/utils/index.js +78 -11
- package/lib/utils/objects.js +30 -10
- package/lib/utils/redis.js +21 -8
- package/package.json +5 -5
- package/index.d.ts +0 -1
package/lib/data/payments.js
CHANGED
|
@@ -1,3 +1,34 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
Object.defineProperty(exports, "__esModule", {
|
|
4
|
+
value: true
|
|
5
|
+
});
|
|
6
|
+
exports.createPaymentHold = exports.createPaymentTransfer = exports.deletePaymentAccountBank = exports.deleteCreditCard = exports.getCreditCards = exports.updateCreditCard = exports.addCreditCard = exports.addPaymentAccountCard = exports.addPaymentAccountBank = exports.addCustomerAccount = void 0;
|
|
7
|
+
|
|
8
|
+
var _utils = require("@nlabs/utils");
|
|
9
|
+
|
|
10
|
+
var _arangojs = require("arangojs");
|
|
11
|
+
|
|
12
|
+
var _graphqlErrors = require("graphql-errors");
|
|
13
|
+
|
|
14
|
+
var _isEmpty = _interopRequireDefault(require("lodash/isEmpty"));
|
|
15
|
+
|
|
16
|
+
var _luxon = require("luxon");
|
|
17
|
+
|
|
18
|
+
var stripe = _interopRequireWildcard(require("stripe"));
|
|
19
|
+
|
|
20
|
+
var _config = require("../config");
|
|
21
|
+
|
|
22
|
+
var _analytics = require("../utils/analytics");
|
|
23
|
+
|
|
24
|
+
var _arangodb = require("../utils/arangodb");
|
|
25
|
+
|
|
26
|
+
var _users = require("./users");
|
|
27
|
+
|
|
28
|
+
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; } }
|
|
29
|
+
|
|
30
|
+
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
31
|
+
|
|
1
32
|
function _templateObject11() {
|
|
2
33
|
var data = _taggedTemplateLiteral(["INSERT ", " IN payments RETURN NEW"]);
|
|
3
34
|
|
|
@@ -114,29 +145,19 @@ function _templateObject() {
|
|
|
114
145
|
|
|
115
146
|
function _taggedTemplateLiteral(strings, raw) { if (!raw) { raw = strings.slice(0); } return Object.freeze(Object.defineProperties(strings, { raw: { value: Object.freeze(raw) } })); }
|
|
116
147
|
|
|
117
|
-
import { createHash, parseChar, parseId, parseNum, parseString, parseVarChar } from '@nlabs/utils';
|
|
118
|
-
import { aql } from 'arangojs';
|
|
119
|
-
import { UserError } from 'graphql-errors';
|
|
120
|
-
import isEmpty from 'lodash/isEmpty';
|
|
121
|
-
import { DateTime } from 'luxon';
|
|
122
|
-
import * as stripe from 'stripe';
|
|
123
|
-
import { Config } from '../config';
|
|
124
|
-
import { logError } from '../utils/analytics';
|
|
125
|
-
import { useDb } from '../utils/arangodb';
|
|
126
|
-
import { getUser } from './users';
|
|
127
148
|
/**
|
|
128
149
|
* Copyright (c) 2019-Present, Nitrogen Labs, Inc.
|
|
129
150
|
* Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.
|
|
130
151
|
*/
|
|
131
|
-
|
|
132
152
|
var eventCategory = 'payments';
|
|
133
|
-
|
|
153
|
+
|
|
154
|
+
var addCustomerAccount = function addCustomerAccount(context) {
|
|
134
155
|
var action = 'addCustomerAccount';
|
|
135
156
|
var database = context.database,
|
|
136
157
|
sessionId = context.userId,
|
|
137
158
|
username = context.username; // Stripe
|
|
138
159
|
|
|
139
|
-
var stripeClient = stripe(Config.get('stripe.token'));
|
|
160
|
+
var stripeClient = stripe(_config.Config.get('stripe.token'));
|
|
140
161
|
return stripeClient.customers.create({
|
|
141
162
|
metadata: {
|
|
142
163
|
userId: sessionId,
|
|
@@ -149,14 +170,14 @@ export var addCustomerAccount = function addCustomerAccount(context) {
|
|
|
149
170
|
modified: now,
|
|
150
171
|
stripeCustomerId: customer.id
|
|
151
172
|
};
|
|
152
|
-
var aqlQry = aql(_templateObject(), sessionId, update);
|
|
153
|
-
return useDb(database).query(aqlQry).then(function (cursor) {
|
|
173
|
+
var aqlQry = (0, _arangojs.aql)(_templateObject(), sessionId, update);
|
|
174
|
+
return (0, _arangodb.useDb)(database).query(aqlQry).then(function (cursor) {
|
|
154
175
|
return cursor.next();
|
|
155
176
|
}).then(function () {
|
|
156
177
|
var updatedUser = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
|
|
157
|
-
return !
|
|
178
|
+
return !(0, _isEmpty.default)(updatedUser);
|
|
158
179
|
}).catch(function (error) {
|
|
159
|
-
return logError({
|
|
180
|
+
return (0, _analytics.logError)({
|
|
160
181
|
action: action,
|
|
161
182
|
category: eventCategory,
|
|
162
183
|
label: 'db_error'
|
|
@@ -166,7 +187,10 @@ export var addCustomerAccount = function addCustomerAccount(context) {
|
|
|
166
187
|
});
|
|
167
188
|
});
|
|
168
189
|
};
|
|
169
|
-
|
|
190
|
+
|
|
191
|
+
exports.addCustomerAccount = addCustomerAccount;
|
|
192
|
+
|
|
193
|
+
var addPaymentAccountBank = function addPaymentAccountBank(context, bankAccount) {
|
|
170
194
|
var action = 'addPaymentAccountBank';
|
|
171
195
|
var database = context.database,
|
|
172
196
|
sessionId = context.userId; // Params
|
|
@@ -174,27 +198,27 @@ export var addPaymentAccountBank = function addPaymentAccountBank(context, bankA
|
|
|
174
198
|
var accountNumber = bankAccount.accountNumber,
|
|
175
199
|
fullName = bankAccount.fullName,
|
|
176
200
|
routing = bankAccount.routing;
|
|
177
|
-
var formatAccount = parseString(accountNumber, 32);
|
|
201
|
+
var formatAccount = (0, _utils.parseString)(accountNumber, 32);
|
|
178
202
|
|
|
179
203
|
if (formatAccount === '') {
|
|
180
|
-
throw new UserError('required_account_number');
|
|
204
|
+
throw new _graphqlErrors.UserError('required_account_number');
|
|
181
205
|
}
|
|
182
206
|
|
|
183
|
-
var formatFullName = parseVarChar(fullName, 128);
|
|
207
|
+
var formatFullName = (0, _utils.parseVarChar)(fullName, 128);
|
|
184
208
|
|
|
185
209
|
if (formatFullName === '') {
|
|
186
|
-
throw new UserError('required_full_name');
|
|
210
|
+
throw new _graphqlErrors.UserError('required_full_name');
|
|
187
211
|
}
|
|
188
212
|
|
|
189
|
-
var formatRouting = parseString(routing, 32);
|
|
213
|
+
var formatRouting = (0, _utils.parseString)(routing, 32);
|
|
190
214
|
|
|
191
215
|
if (formatRouting === '') {
|
|
192
|
-
throw new UserError('required_routing_number');
|
|
216
|
+
throw new _graphqlErrors.UserError('required_routing_number');
|
|
193
217
|
}
|
|
194
218
|
|
|
195
|
-
return getUser(context, sessionId).then(function (user) {
|
|
219
|
+
return (0, _users.getUser)(context, sessionId).then(function (user) {
|
|
196
220
|
// Stripe
|
|
197
|
-
var stripeClient = stripe(Config.get('stripe.token'));
|
|
221
|
+
var stripeClient = stripe(_config.Config.get('stripe.token'));
|
|
198
222
|
return stripeClient.customers.createSource(user.stripeAccountId, {
|
|
199
223
|
source: {
|
|
200
224
|
account_holder_name: formatFullName,
|
|
@@ -213,8 +237,8 @@ export var addPaymentAccountBank = function addPaymentAccountBank(context, bankA
|
|
|
213
237
|
bankRouting: account.routing_number,
|
|
214
238
|
modified: Date.now()
|
|
215
239
|
};
|
|
216
|
-
var aqlQry = aql(_templateObject2(), sessionId, update);
|
|
217
|
-
return useDb(database).query(aqlQry).then(function (cursor) {
|
|
240
|
+
var aqlQry = (0, _arangojs.aql)(_templateObject2(), sessionId, update);
|
|
241
|
+
return (0, _arangodb.useDb)(database).query(aqlQry).then(function (cursor) {
|
|
218
242
|
return cursor.next();
|
|
219
243
|
}).then(function () {
|
|
220
244
|
var updatedUser = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
|
|
@@ -224,7 +248,7 @@ export var addPaymentAccountBank = function addPaymentAccountBank(context, bankA
|
|
|
224
248
|
var msg = error.message;
|
|
225
249
|
|
|
226
250
|
if (msg === 'A bank account with that routing number and account number ' + 'already exists for this customer.') {
|
|
227
|
-
return logError({
|
|
251
|
+
return (0, _analytics.logError)({
|
|
228
252
|
action: action,
|
|
229
253
|
category: eventCategory,
|
|
230
254
|
label: 'bank_account_exists'
|
|
@@ -233,7 +257,7 @@ export var addPaymentAccountBank = function addPaymentAccountBank(context, bankA
|
|
|
233
257
|
});
|
|
234
258
|
}
|
|
235
259
|
|
|
236
|
-
return logError({
|
|
260
|
+
return (0, _analytics.logError)({
|
|
237
261
|
action: action,
|
|
238
262
|
category: eventCategory,
|
|
239
263
|
label: 'payment_error'
|
|
@@ -243,11 +267,14 @@ export var addPaymentAccountBank = function addPaymentAccountBank(context, bankA
|
|
|
243
267
|
});
|
|
244
268
|
});
|
|
245
269
|
};
|
|
246
|
-
|
|
270
|
+
|
|
271
|
+
exports.addPaymentAccountBank = addPaymentAccountBank;
|
|
272
|
+
|
|
273
|
+
var addPaymentAccountCard = function addPaymentAccountCard(context, card, requestData) {
|
|
247
274
|
var action = 'addPaymentAccountCard';
|
|
248
275
|
var database = context.database,
|
|
249
276
|
sessionId = context.userId;
|
|
250
|
-
return getUser(context, sessionId).then(function (user) {
|
|
277
|
+
return (0, _users.getUser)(context, sessionId).then(function (user) {
|
|
251
278
|
// User
|
|
252
279
|
var userKey = user._key,
|
|
253
280
|
_user$country = user.country,
|
|
@@ -261,9 +288,9 @@ export var addPaymentAccountCard = function addPaymentAccountCard(context, card,
|
|
|
261
288
|
last = user.last,
|
|
262
289
|
username = user.username; // Birthday
|
|
263
290
|
|
|
264
|
-
var dobDay = +DateTime.fromMillis(dob).toFormat('d');
|
|
265
|
-
var dobMonth = +DateTime.fromMillis(dob).toFormat('M');
|
|
266
|
-
var dobYear = +DateTime.fromMillis(dob).toFormat('y'); // Card
|
|
291
|
+
var dobDay = +_luxon.DateTime.fromMillis(dob).toFormat('d');
|
|
292
|
+
var dobMonth = +_luxon.DateTime.fromMillis(dob).toFormat('M');
|
|
293
|
+
var dobYear = +_luxon.DateTime.fromMillis(dob).toFormat('y'); // Card
|
|
267
294
|
|
|
268
295
|
var accountNumber = card.accountNumber,
|
|
269
296
|
acceptedTerms = card.acceptedTerms,
|
|
@@ -273,9 +300,9 @@ export var addPaymentAccountCard = function addPaymentAccountCard(context, card,
|
|
|
273
300
|
street1 = card.street1,
|
|
274
301
|
zip = card.zip; // Acceptance
|
|
275
302
|
|
|
276
|
-
var accepted = acceptedTerms ? DateTime.local().millisecond : 0; // Stripe
|
|
303
|
+
var accepted = acceptedTerms ? _luxon.DateTime.local().millisecond : 0; // Stripe
|
|
277
304
|
|
|
278
|
-
var stripeClient = stripe(Config.get('stripe.token'));
|
|
305
|
+
var stripeClient = stripe(_config.Config.get('stripe.token'));
|
|
279
306
|
return stripeClient.accounts.create({
|
|
280
307
|
country: country,
|
|
281
308
|
email: email,
|
|
@@ -321,14 +348,14 @@ export var addPaymentAccountCard = function addPaymentAccountCard(context, card,
|
|
|
321
348
|
modified: now,
|
|
322
349
|
stripeAccountId: customer.id
|
|
323
350
|
};
|
|
324
|
-
var aqlQry = aql(_templateObject3(), userKey, update);
|
|
325
|
-
return useDb(database).query(aqlQry).then(function (cursor) {
|
|
351
|
+
var aqlQry = (0, _arangojs.aql)(_templateObject3(), userKey, update);
|
|
352
|
+
return (0, _arangodb.useDb)(database).query(aqlQry).then(function (cursor) {
|
|
326
353
|
return cursor.next();
|
|
327
354
|
}).then(function () {
|
|
328
355
|
var updatedUser = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
|
|
329
|
-
return !
|
|
356
|
+
return !(0, _isEmpty.default)(updatedUser);
|
|
330
357
|
}).catch(function (error) {
|
|
331
|
-
return logError({
|
|
358
|
+
return (0, _analytics.logError)({
|
|
332
359
|
action: action,
|
|
333
360
|
category: eventCategory,
|
|
334
361
|
label: 'db_error'
|
|
@@ -339,7 +366,10 @@ export var addPaymentAccountCard = function addPaymentAccountCard(context, card,
|
|
|
339
366
|
});
|
|
340
367
|
});
|
|
341
368
|
};
|
|
342
|
-
|
|
369
|
+
|
|
370
|
+
exports.addPaymentAccountCard = addPaymentAccountCard;
|
|
371
|
+
|
|
372
|
+
var addCreditCard = function addCreditCard(context, card) {
|
|
343
373
|
var action = 'addCreditCard';
|
|
344
374
|
var database = context.database,
|
|
345
375
|
sessionId = context.userId;
|
|
@@ -353,72 +383,72 @@ export var addCreditCard = function addCreditCard(context, card) {
|
|
|
353
383
|
street1 = card.street1,
|
|
354
384
|
state = card.state,
|
|
355
385
|
zip = card.zip;
|
|
356
|
-
var formatNumber = parseNum(accountNumber, 16);
|
|
386
|
+
var formatNumber = (0, _utils.parseNum)(accountNumber, 16);
|
|
357
387
|
|
|
358
388
|
if (!formatNumber) {
|
|
359
|
-
throw new UserError('required_credit_card_number');
|
|
389
|
+
throw new _graphqlErrors.UserError('required_credit_card_number');
|
|
360
390
|
}
|
|
361
391
|
|
|
362
|
-
var formatExpMonth = parseNum(expMonth, 2);
|
|
392
|
+
var formatExpMonth = (0, _utils.parseNum)(expMonth, 2);
|
|
363
393
|
|
|
364
394
|
if (!formatExpMonth) {
|
|
365
|
-
throw new UserError('required_credit_card_exp_month');
|
|
395
|
+
throw new _graphqlErrors.UserError('required_credit_card_exp_month');
|
|
366
396
|
}
|
|
367
397
|
|
|
368
|
-
var formatExpYear = parseNum(expYear, 2);
|
|
398
|
+
var formatExpYear = (0, _utils.parseNum)(expYear, 2);
|
|
369
399
|
|
|
370
400
|
if (!formatExpYear) {
|
|
371
|
-
throw new UserError('required_credit_card_exp_year');
|
|
401
|
+
throw new _graphqlErrors.UserError('required_credit_card_exp_year');
|
|
372
402
|
}
|
|
373
403
|
|
|
374
|
-
var formatCvc = parseNum(cvc, 3);
|
|
404
|
+
var formatCvc = (0, _utils.parseNum)(cvc, 3);
|
|
375
405
|
|
|
376
406
|
if (!formatCvc) {
|
|
377
|
-
throw new UserError('required_credit_card_cvc');
|
|
407
|
+
throw new _graphqlErrors.UserError('required_credit_card_cvc');
|
|
378
408
|
}
|
|
379
409
|
|
|
380
410
|
var paymentCard = {};
|
|
381
|
-
var formatCity = parseVarChar(city, 32);
|
|
411
|
+
var formatCity = (0, _utils.parseVarChar)(city, 32);
|
|
382
412
|
|
|
383
413
|
if (formatCity) {
|
|
384
414
|
paymentCard.city = formatCity;
|
|
385
415
|
}
|
|
386
416
|
|
|
387
|
-
var formatCountry = parseVarChar(country, 32);
|
|
417
|
+
var formatCountry = (0, _utils.parseVarChar)(country, 32);
|
|
388
418
|
|
|
389
419
|
if (formatCountry) {
|
|
390
420
|
paymentCard.country = formatCountry;
|
|
391
421
|
}
|
|
392
422
|
|
|
393
|
-
var formatFullName = parseVarChar(fullName, 32);
|
|
423
|
+
var formatFullName = (0, _utils.parseVarChar)(fullName, 32);
|
|
394
424
|
|
|
395
425
|
if (formatFullName) {
|
|
396
426
|
paymentCard.fullName = formatFullName;
|
|
397
427
|
}
|
|
398
428
|
|
|
399
|
-
var formatStreet = parseVarChar(street1, 32);
|
|
429
|
+
var formatStreet = (0, _utils.parseVarChar)(street1, 32);
|
|
400
430
|
|
|
401
431
|
if (formatStreet) {
|
|
402
432
|
paymentCard.street1 = formatStreet;
|
|
403
433
|
}
|
|
404
434
|
|
|
405
|
-
var formatState = parseVarChar(state, 32);
|
|
435
|
+
var formatState = (0, _utils.parseVarChar)(state, 32);
|
|
406
436
|
|
|
407
437
|
if (formatState) {
|
|
408
438
|
paymentCard.state = formatState;
|
|
409
439
|
}
|
|
410
440
|
|
|
411
|
-
var formatZip = parseVarChar(zip, 32);
|
|
441
|
+
var formatZip = (0, _utils.parseVarChar)(zip, 32);
|
|
412
442
|
|
|
413
443
|
if (formatZip) {
|
|
414
444
|
paymentCard.zip = formatZip;
|
|
415
445
|
} // Get stripe id
|
|
416
446
|
|
|
417
447
|
|
|
418
|
-
return getUser(context, sessionId).then(function () {
|
|
448
|
+
return (0, _users.getUser)(context, sessionId).then(function () {
|
|
419
449
|
var user = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
|
|
420
450
|
// Stripe
|
|
421
|
-
var stripeClient = stripe(Config.get('stripe.token'));
|
|
451
|
+
var stripeClient = stripe(_config.Config.get('stripe.token'));
|
|
422
452
|
return stripeClient.customers.createSource(user.stripeCustomerId, {
|
|
423
453
|
source: {
|
|
424
454
|
address_city: formatCity,
|
|
@@ -438,7 +468,7 @@ export var addCreditCard = function addCreditCard(context, card) {
|
|
|
438
468
|
var now = Date.now();
|
|
439
469
|
|
|
440
470
|
var insert = _objectSpread({}, paymentCard, {
|
|
441
|
-
_key: createHash("user-payment-".concat(sessionId)),
|
|
471
|
+
_key: (0, _utils.createHash)("user-payment-".concat(sessionId)),
|
|
442
472
|
accountNumber: stripeObj.last4,
|
|
443
473
|
added: now,
|
|
444
474
|
brand: stripeObj.brand,
|
|
@@ -449,18 +479,18 @@ export var addCreditCard = function addCreditCard(context, card) {
|
|
|
449
479
|
userId: sessionId
|
|
450
480
|
});
|
|
451
481
|
|
|
452
|
-
var insertAqlQry = aql(_templateObject4(), insert);
|
|
453
|
-
return useDb(database).query(insertAqlQry).then(function (cursor) {
|
|
482
|
+
var insertAqlQry = (0, _arangojs.aql)(_templateObject4(), insert);
|
|
483
|
+
return (0, _arangodb.useDb)(database).query(insertAqlQry).then(function (cursor) {
|
|
454
484
|
return cursor.next();
|
|
455
485
|
}).then(function () {
|
|
456
486
|
var newCard = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
|
|
457
487
|
|
|
458
|
-
if (!
|
|
488
|
+
if (!(0, _isEmpty.default)(newCard)) {
|
|
459
489
|
// Add linked edge
|
|
460
490
|
var cardId = card._id,
|
|
461
491
|
cardKey = card._key;
|
|
462
|
-
var edgeCollection = useDb(database).edgeCollection('hasPayment');
|
|
463
|
-
var edgeId = createHash("payment-".concat(cardKey));
|
|
492
|
+
var edgeCollection = (0, _arangodb.useDb)(database).edgeCollection('hasPayment');
|
|
493
|
+
var edgeId = (0, _utils.createHash)("payment-".concat(cardKey));
|
|
464
494
|
var edge = {
|
|
465
495
|
_key: edgeId
|
|
466
496
|
};
|
|
@@ -472,7 +502,7 @@ export var addCreditCard = function addCreditCard(context, card) {
|
|
|
472
502
|
return newCard;
|
|
473
503
|
});
|
|
474
504
|
}).catch(function (error) {
|
|
475
|
-
return logError({
|
|
505
|
+
return (0, _analytics.logError)({
|
|
476
506
|
action: action,
|
|
477
507
|
category: eventCategory,
|
|
478
508
|
label: 'payment_error'
|
|
@@ -482,7 +512,10 @@ export var addCreditCard = function addCreditCard(context, card) {
|
|
|
482
512
|
});
|
|
483
513
|
});
|
|
484
514
|
};
|
|
485
|
-
|
|
515
|
+
|
|
516
|
+
exports.addCreditCard = addCreditCard;
|
|
517
|
+
|
|
518
|
+
var updateCreditCard = function updateCreditCard(context, card) {
|
|
486
519
|
var database = context.database,
|
|
487
520
|
sessionId = context.userId;
|
|
488
521
|
var city = card.city,
|
|
@@ -494,21 +527,21 @@ export var updateCreditCard = function updateCreditCard(context, card) {
|
|
|
494
527
|
street1 = card.street1,
|
|
495
528
|
state = card.state,
|
|
496
529
|
zip = card.zip;
|
|
497
|
-
var formatId = parseId(id);
|
|
530
|
+
var formatId = (0, _utils.parseId)(id);
|
|
498
531
|
|
|
499
532
|
if (formatId) {
|
|
500
|
-
throw new UserError('required_credit_card_id');
|
|
533
|
+
throw new _graphqlErrors.UserError('required_credit_card_id');
|
|
501
534
|
}
|
|
502
535
|
|
|
503
536
|
var paymentCard = {};
|
|
504
|
-
var formatExpMonth = parseNum(expMonth, 2);
|
|
505
|
-
var formatExpYear = parseNum(expYear, 2);
|
|
506
|
-
var formatCity = parseVarChar(city, 32);
|
|
507
|
-
var formatCountry = parseVarChar(country, 2);
|
|
508
|
-
var formatFullName = parseVarChar(fullName, 32);
|
|
509
|
-
var formatStreet1 = parseString(street1, 32);
|
|
510
|
-
var formatState = parseVarChar(state, 2);
|
|
511
|
-
var formatZip = parseVarChar(zip, 10);
|
|
537
|
+
var formatExpMonth = (0, _utils.parseNum)(expMonth, 2);
|
|
538
|
+
var formatExpYear = (0, _utils.parseNum)(expYear, 2);
|
|
539
|
+
var formatCity = (0, _utils.parseVarChar)(city, 32);
|
|
540
|
+
var formatCountry = (0, _utils.parseVarChar)(country, 2);
|
|
541
|
+
var formatFullName = (0, _utils.parseVarChar)(fullName, 32);
|
|
542
|
+
var formatStreet1 = (0, _utils.parseString)(street1, 32);
|
|
543
|
+
var formatState = (0, _utils.parseVarChar)(state, 2);
|
|
544
|
+
var formatZip = (0, _utils.parseVarChar)(zip, 10);
|
|
512
545
|
|
|
513
546
|
if (formatExpMonth) {
|
|
514
547
|
paymentCard.expMonth = formatExpMonth;
|
|
@@ -543,8 +576,8 @@ export var updateCreditCard = function updateCreditCard(context, card) {
|
|
|
543
576
|
}
|
|
544
577
|
|
|
545
578
|
var update = paymentCard;
|
|
546
|
-
var aqlQry = aql(_templateObject5(), formatId, sessionId, update, sessionId);
|
|
547
|
-
return useDb(database).query(aqlQry).then(function (cursor) {
|
|
579
|
+
var aqlQry = (0, _arangojs.aql)(_templateObject5(), formatId, sessionId, update, sessionId);
|
|
580
|
+
return (0, _arangodb.useDb)(database).query(aqlQry).then(function (cursor) {
|
|
548
581
|
return cursor.next();
|
|
549
582
|
}).then(function () {
|
|
550
583
|
var results = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {
|
|
@@ -554,12 +587,12 @@ export var updateCreditCard = function updateCreditCard(context, card) {
|
|
|
554
587
|
var updatedCard = results.card;
|
|
555
588
|
var user = results.user;
|
|
556
589
|
|
|
557
|
-
if (
|
|
558
|
-
throw new UserError('not_found');
|
|
590
|
+
if ((0, _isEmpty.default)(updatedCard)) {
|
|
591
|
+
throw new _graphqlErrors.UserError('not_found');
|
|
559
592
|
} // Stripe
|
|
560
593
|
|
|
561
594
|
|
|
562
|
-
var stripeClient = stripe(Config.get('stripe.token'));
|
|
595
|
+
var stripeClient = stripe(_config.Config.get('stripe.token'));
|
|
563
596
|
return stripeClient.customers.updateCard(user.stripeCustomerId, card.stripeId, {
|
|
564
597
|
address_city: formatCity,
|
|
565
598
|
address_country: formatCountry,
|
|
@@ -573,22 +606,25 @@ export var updateCreditCard = function updateCreditCard(context, card) {
|
|
|
573
606
|
return card;
|
|
574
607
|
}).catch(function (error) {
|
|
575
608
|
console.log('payments::updateCard::error', error);
|
|
576
|
-
throw new UserError('payment_error');
|
|
609
|
+
throw new _graphqlErrors.UserError('payment_error');
|
|
577
610
|
});
|
|
578
611
|
});
|
|
579
612
|
};
|
|
580
|
-
|
|
613
|
+
|
|
614
|
+
exports.updateCreditCard = updateCreditCard;
|
|
615
|
+
|
|
616
|
+
var getCreditCards = function getCreditCards(context) {
|
|
581
617
|
var action = 'getCreditCards';
|
|
582
618
|
var database = context.database,
|
|
583
619
|
sessionId = context.userId;
|
|
584
|
-
var aqlQry = aql(_templateObject6(), sessionId);
|
|
585
|
-
return useDb(database).query(aqlQry).then(function (cursor) {
|
|
620
|
+
var aqlQry = (0, _arangojs.aql)(_templateObject6(), sessionId);
|
|
621
|
+
return (0, _arangodb.useDb)(database).query(aqlQry).then(function (cursor) {
|
|
586
622
|
return cursor.all();
|
|
587
623
|
}).then(function () {
|
|
588
624
|
var list = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : [];
|
|
589
625
|
return list;
|
|
590
626
|
}).catch(function (error) {
|
|
591
|
-
return logError({
|
|
627
|
+
return (0, _analytics.logError)({
|
|
592
628
|
action: action,
|
|
593
629
|
category: eventCategory,
|
|
594
630
|
label: 'db_error'
|
|
@@ -597,12 +633,15 @@ export var getCreditCards = function getCreditCards(context) {
|
|
|
597
633
|
});
|
|
598
634
|
});
|
|
599
635
|
};
|
|
600
|
-
|
|
636
|
+
|
|
637
|
+
exports.getCreditCards = getCreditCards;
|
|
638
|
+
|
|
639
|
+
var deleteCreditCard = function deleteCreditCard(context, cardId) {
|
|
601
640
|
var database = context.database,
|
|
602
641
|
sessionId = context.userId;
|
|
603
|
-
var formatCardId = parseId(cardId);
|
|
604
|
-
var aqlQry = aql(_templateObject7(), formatCardId, sessionId, sessionId);
|
|
605
|
-
return useDb(database).query(aqlQry).then(function (cursor) {
|
|
642
|
+
var formatCardId = (0, _utils.parseId)(cardId);
|
|
643
|
+
var aqlQry = (0, _arangojs.aql)(_templateObject7(), formatCardId, sessionId, sessionId);
|
|
644
|
+
return (0, _arangodb.useDb)(database).query(aqlQry).then(function (cursor) {
|
|
606
645
|
return cursor.next();
|
|
607
646
|
}).then(function () {
|
|
608
647
|
var result = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {
|
|
@@ -610,7 +649,7 @@ export var deleteCreditCard = function deleteCreditCard(context, cardId) {
|
|
|
610
649
|
user: {}
|
|
611
650
|
};
|
|
612
651
|
|
|
613
|
-
if (
|
|
652
|
+
if ((0, _isEmpty.default)(result)) {
|
|
614
653
|
return false;
|
|
615
654
|
}
|
|
616
655
|
|
|
@@ -618,21 +657,21 @@ export var deleteCreditCard = function deleteCreditCard(context, cardId) {
|
|
|
618
657
|
user = result.user;
|
|
619
658
|
var cardKey = card._key; // Remove linked edges
|
|
620
659
|
|
|
621
|
-
var edgeCollection = useDb(database).edgeCollection('hasPayment');
|
|
660
|
+
var edgeCollection = (0, _arangodb.useDb)(database).edgeCollection('hasPayment');
|
|
622
661
|
return edgeCollection.outEdges(cardKey).then(function (edges) {
|
|
623
662
|
if (edges.length) {
|
|
624
663
|
return Promise.all(edges.map(function (edge) {
|
|
625
664
|
var edgeKey = edge._key;
|
|
626
|
-
var removeAqlQry = aql(_templateObject8(), edgeKey);
|
|
627
|
-
return useDb(database).query(removeAqlQry);
|
|
665
|
+
var removeAqlQry = (0, _arangojs.aql)(_templateObject8(), edgeKey);
|
|
666
|
+
return (0, _arangodb.useDb)(database).query(removeAqlQry);
|
|
628
667
|
})).then(function () {
|
|
629
668
|
// Stripe
|
|
630
|
-
var stripeClient = stripe(Config.get('stripe.token'));
|
|
669
|
+
var stripeClient = stripe(_config.Config.get('stripe.token'));
|
|
631
670
|
return stripeClient.customers.deleteCard(user.stripeCustomerId, card.stripeId).then(function () {
|
|
632
671
|
return true;
|
|
633
672
|
}).catch(function (error) {
|
|
634
673
|
console.log('payments::deleteCard::error', error);
|
|
635
|
-
throw new UserError('payment_error');
|
|
674
|
+
throw new _graphqlErrors.UserError('payment_error');
|
|
636
675
|
});
|
|
637
676
|
});
|
|
638
677
|
}
|
|
@@ -641,7 +680,10 @@ export var deleteCreditCard = function deleteCreditCard(context, cardId) {
|
|
|
641
680
|
});
|
|
642
681
|
});
|
|
643
682
|
};
|
|
644
|
-
|
|
683
|
+
|
|
684
|
+
exports.deleteCreditCard = deleteCreditCard;
|
|
685
|
+
|
|
686
|
+
var deletePaymentAccountBank = function deletePaymentAccountBank(context, bankId) {
|
|
645
687
|
var database = context.database,
|
|
646
688
|
sessionId = context.userId; // Clean db
|
|
647
689
|
|
|
@@ -652,13 +694,13 @@ export var deletePaymentAccountBank = function deletePaymentAccountBank(context,
|
|
|
652
694
|
bankRouting: '',
|
|
653
695
|
modified: Date.now()
|
|
654
696
|
};
|
|
655
|
-
var aqlQry = aql(_templateObject9(), sessionId, update);
|
|
656
|
-
return useDb(database).query(aqlQry).then(function (cursor) {
|
|
697
|
+
var aqlQry = (0, _arangojs.aql)(_templateObject9(), sessionId, update);
|
|
698
|
+
return (0, _arangodb.useDb)(database).query(aqlQry).then(function (cursor) {
|
|
657
699
|
return cursor.next();
|
|
658
700
|
}).then(function () {
|
|
659
701
|
var user = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
|
|
660
702
|
// Stripe
|
|
661
|
-
var stripeClient = stripe(Config.get('stripe.token'));
|
|
703
|
+
var stripeClient = stripe(_config.Config.get('stripe.token'));
|
|
662
704
|
return stripeClient.customers.deleteSource(user.stripeAccountId, bankId).then(function () {
|
|
663
705
|
var response = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {
|
|
664
706
|
deleted: false
|
|
@@ -669,16 +711,19 @@ export var deletePaymentAccountBank = function deletePaymentAccountBank(context,
|
|
|
669
711
|
});
|
|
670
712
|
});
|
|
671
713
|
};
|
|
672
|
-
|
|
714
|
+
|
|
715
|
+
exports.deletePaymentAccountBank = deletePaymentAccountBank;
|
|
716
|
+
|
|
717
|
+
var createPaymentTransfer = function createPaymentTransfer(context, transfer) {
|
|
673
718
|
var database = context.database,
|
|
674
719
|
sessionId = context.userId;
|
|
675
720
|
var amount = transfer.amount,
|
|
676
721
|
currency = transfer.currency;
|
|
677
|
-
var formatAmount = parseNum(amount);
|
|
678
|
-
var formatCurrency = parseChar(currency, 3, 'USD').toUpperCase();
|
|
679
|
-
return getUser(context, sessionId).then(function (user) {
|
|
722
|
+
var formatAmount = (0, _utils.parseNum)(amount);
|
|
723
|
+
var formatCurrency = (0, _utils.parseChar)(currency, 3, 'USD').toUpperCase();
|
|
724
|
+
return (0, _users.getUser)(context, sessionId).then(function (user) {
|
|
680
725
|
// Stripe
|
|
681
|
-
var stripeClient = stripe(Config.get('stripe.token'));
|
|
726
|
+
var stripeClient = stripe(_config.Config.get('stripe.token'));
|
|
682
727
|
return stripeClient.transfers.create({
|
|
683
728
|
amount: formatAmount,
|
|
684
729
|
currency: formatCurrency,
|
|
@@ -693,8 +738,8 @@ export var createPaymentTransfer = function createPaymentTransfer(context, trans
|
|
|
693
738
|
modified: now,
|
|
694
739
|
userId: sessionId
|
|
695
740
|
};
|
|
696
|
-
var aqlQry = aql(_templateObject10(), insert);
|
|
697
|
-
return useDb(database).query(aqlQry).then(function (cursor) {
|
|
741
|
+
var aqlQry = (0, _arangojs.aql)(_templateObject10(), insert);
|
|
742
|
+
return (0, _arangodb.useDb)(database).query(aqlQry).then(function (cursor) {
|
|
698
743
|
return cursor.next();
|
|
699
744
|
}).then(function (newTransfer) {
|
|
700
745
|
return newTransfer;
|
|
@@ -702,7 +747,10 @@ export var createPaymentTransfer = function createPaymentTransfer(context, trans
|
|
|
702
747
|
});
|
|
703
748
|
});
|
|
704
749
|
};
|
|
705
|
-
|
|
750
|
+
|
|
751
|
+
exports.createPaymentTransfer = createPaymentTransfer;
|
|
752
|
+
|
|
753
|
+
var createPaymentHold = function createPaymentHold(context, payment) {
|
|
706
754
|
var database = context.database,
|
|
707
755
|
sessionId = context.userId;
|
|
708
756
|
var amount = payment.amount,
|
|
@@ -710,8 +758,8 @@ export var createPaymentHold = function createPaymentHold(context, payment) {
|
|
|
710
758
|
cardId = payment.cardId,
|
|
711
759
|
currency = payment.currency,
|
|
712
760
|
description = payment.description;
|
|
713
|
-
var formatCurrency = parseChar(currency, 3, 'USD').toUpperCase();
|
|
714
|
-
var stripeClient = stripe(Config.get('stripe.token'));
|
|
761
|
+
var formatCurrency = (0, _utils.parseChar)(currency, 3, 'USD').toUpperCase();
|
|
762
|
+
var stripeClient = stripe(_config.Config.get('stripe.token'));
|
|
715
763
|
return stripeClient.charges.create({
|
|
716
764
|
amount: amount,
|
|
717
765
|
capture: capture,
|
|
@@ -734,15 +782,17 @@ export var createPaymentHold = function createPaymentHold(context, payment) {
|
|
|
734
782
|
modified: now,
|
|
735
783
|
userId: sessionId
|
|
736
784
|
};
|
|
737
|
-
var aqlQry = aql(_templateObject11(), insert);
|
|
738
|
-
return useDb(database).query(aqlQry).then(function (cursor) {
|
|
785
|
+
var aqlQry = (0, _arangojs.aql)(_templateObject11(), insert);
|
|
786
|
+
return (0, _arangodb.useDb)(database).query(aqlQry).then(function (cursor) {
|
|
739
787
|
return cursor.next();
|
|
740
788
|
}).then(function (newPayment) {
|
|
741
789
|
return newPayment;
|
|
742
790
|
});
|
|
743
791
|
}).catch(function (error) {
|
|
744
792
|
console.log('payments::createHold::error', error);
|
|
745
|
-
throw new UserError('payment_error');
|
|
793
|
+
throw new _graphqlErrors.UserError('payment_error');
|
|
746
794
|
});
|
|
747
795
|
};
|
|
748
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/data/payments.ts"],"names":["createHash","parseChar","parseId","parseNum","parseString","parseVarChar","aql","UserError","isEmpty","DateTime","stripe","Config","logError","useDb","getUser","eventCategory","addCustomerAccount","context","action","database","sessionId","userId","username","stripeClient","get","customers","create","metadata","then","customer","now","Date","update","modified","stripeCustomerId","id","aqlQry","query","cursor","next","updatedUser","catch","error","category","label","addPaymentAccountBank","bankAccount","accountNumber","fullName","routing","formatAccount","formatFullName","formatRouting","user","createSource","stripeAccountId","source","account_holder_name","account_holder_type","account_number","country","currency","object","routing_number","account","last4","bankFullName","bankId","bankRouting","msg","message","addPaymentAccountCard","card","requestData","userKey","_key","city","dob","email","first","last","dobDay","fromMillis","toFormat","dobMonth","dobYear","acceptedTerms","expMonth","expYear","state","street1","zip","accepted","local","millisecond","accounts","external_account","exp_month","exp_year","number","legal_entity","address","line1","postal_code","day","month","year","first_name","last_name","type","managed","tos_acceptance","date","ip","user_agent","userAgent","addCreditCard","cvc","formatNumber","formatExpMonth","formatExpYear","formatCvc","paymentCard","formatCity","formatCountry","formatStreet","formatState","formatZip","address_city","address_country","address_line1","address_state","address_zip","name","stripeObj","insert","added","brand","cvc_check","insertAqlQry","newCard","cardId","_id","cardKey","edgeCollection","edgeId","edge","save","updateCreditCard","formatId","formatStreet1","results","updatedCard","updateCard","stripeId","console","log","getCreditCards","all","list","deleteCreditCard","formatCardId","result","outEdges","edges","length","Promise","map","edgeKey","removeAqlQry","deleteCard","deletePaymentAccountBank","deleteSource","response","deleted","resolve","createPaymentTransfer","transfer","amount","formatAmount","formatCurrency","toUpperCase","transfers","destination","stripeTransfer","newTransfer","createPaymentHold","payment","capture","description","charges","stripeCharge","chargeFailCode","failure_code","chargeFailMsg","failure_message","chargeId","chargeStatus","status","newPayment"],"mappings":"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","sourcesContent":["import {createHash, parseChar, parseId, parseNum, parseString, parseVarChar} from '@nlabs/utils';\nimport {aql} from 'arangojs';\nimport {AqlQuery} from 'arangojs/lib/cjs/aql-query';\nimport {ArrayCursor} from 'arangojs/lib/cjs/cursor';\nimport {UserError} from 'graphql-errors';\nimport isEmpty from 'lodash/isEmpty';\nimport {DateTime} from 'luxon';\nimport * as stripe from 'stripe';\n\nimport {Config} from '../config';\nimport {ApiContext, PaymentBankAccount, PaymentCardType, PaymentCharge, PaymentTransfer, UserType} from '../types';\nimport {logError} from '../utils/analytics';\nimport {useDb} from '../utils/arangodb';\nimport {getUser} from './users';\n\n/**\n * Copyright (c) 2019-Present, Nitrogen Labs, Inc.\n * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.\n */\nconst eventCategory: string = 'payments';\n\nexport const addCustomerAccount = (context: ApiContext): Promise<boolean> => {\n  const action: string = 'addCustomerAccount';\n  const {database, userId: sessionId, username} = context;\n\n  // Stripe\n  const stripeClient = stripe(Config.get('stripe.token'));\n\n  return stripeClient.customers\n    .create({\n      metadata: {\n        userId: sessionId,\n        username\n      }\n    })\n    .then((customer) => {\n      // Create session\n      const now: number = Date.now();\n      const update: UserType = {\n        modified: now,\n        stripeCustomerId: customer.id\n      };\n\n      const aqlQry: AqlQuery = aql`UPDATE ${sessionId} WITH ${update} IN users LIMIT 1 RETURN NEW`;\n\n      return useDb(database).query(aqlQry)\n        .then((cursor: ArrayCursor) => cursor.next())\n        .then((updatedUser: UserType = {}) => !isEmpty(updatedUser))\n        .catch((error: Error) => logError({\n          action,\n          category: eventCategory,\n          label: 'db_error'\n        }, error, context).then(() => null));\n    });\n};\n\nexport const addPaymentAccountBank = (context: ApiContext, bankAccount: PaymentBankAccount): Promise<boolean> => {\n  const action: string = 'addPaymentAccountBank';\n  const {database, userId: sessionId} = context;\n\n  // Params\n  const {\n    accountNumber,\n    fullName,\n    routing\n  } = bankAccount;\n\n  const formatAccount: string = parseString(accountNumber, 32);\n\n  if(formatAccount === '') {\n    throw new UserError('required_account_number');\n  }\n\n  const formatFullName: string = parseVarChar(fullName, 128);\n\n  if(formatFullName === '') {\n    throw new UserError('required_full_name');\n  }\n\n  const formatRouting: string = parseString(routing, 32);\n\n  if(formatRouting === '') {\n    throw new UserError('required_routing_number');\n  }\n\n  return getUser(context, sessionId)\n    .then((user: UserType) => {\n      // Stripe\n      const stripeClient = stripe(Config.get('stripe.token'));\n\n      return stripeClient.customers\n        .createSource(user.stripeAccountId, {\n          source: {\n            account_holder_name: formatFullName,\n            account_holder_type: 'individual',\n            account_number: formatAccount,\n            country: 'US',\n            currency: 'USD',\n            object: 'bank_account',\n            routing_number: formatRouting\n          }\n        })\n        .then((account) => {\n          const update: any = {\n            bankAccount: account.last4,\n            bankFullName: formatFullName,\n            bankId: account.id,\n            bankRouting: account.routing_number,\n            modified: Date.now()\n          };\n\n          const aqlQry: AqlQuery = aql`UPDATE ${sessionId} WITH ${update} IN users LIMIT 1 RETURN NEW`;\n\n          return useDb(database).query(aqlQry)\n            .then((cursor: ArrayCursor) => cursor.next())\n            .then((updatedUser: UserType = {}) => updatedUser);\n        })\n        .catch((error: Error) => {\n          const msg = error.message;\n\n          if(msg === 'A bank account with that routing number and account number ' +\n            'already exists for this customer.') {\n            return logError({\n              action,\n              category: eventCategory,\n              label: 'bank_account_exists'\n            }, error, context).then(() => null);\n          }\n          return logError({\n            action,\n            category: eventCategory,\n            label: 'payment_error'\n          }, error, context).then(() => null);\n        });\n    });\n};\n\nexport const addPaymentAccountCard = (context: ApiContext, card: PaymentCardType, requestData): Promise<UserType> => {\n  const action: string = 'addPaymentAccountCard';\n  const {database, userId: sessionId} = context;\n\n  return getUser(context, sessionId)\n    .then((user: UserType) => {\n      // User\n      const {\n        _key: userKey,\n        country = 'US',\n        city,\n        currency = 'USD',\n        dob,\n        email,\n        first,\n        last,\n        username\n      }: UserType = user;\n\n      // Birthday\n      const dobDay: number = +(DateTime.fromMillis(dob).toFormat('d'));\n      const dobMonth: number = +(DateTime.fromMillis(dob).toFormat('M'));\n      const dobYear: number = +(DateTime.fromMillis(dob).toFormat('y'));\n\n      // Card\n      const {\n        accountNumber,\n        acceptedTerms,\n        expMonth,\n        expYear,\n        state,\n        street1,\n        zip\n      }: PaymentCardType = card;\n\n      // Acceptance\n      const accepted: number = acceptedTerms ? DateTime.local().millisecond : 0;\n\n      // Stripe\n      const stripeClient = stripe(Config.get('stripe.token'));\n\n      return stripeClient.accounts\n        .create({\n          country,\n          email,\n          external_account: {\n            currency,\n            exp_month: expMonth,\n            exp_year: expYear,\n            number: accountNumber,\n            object: 'card'\n          },\n          legal_entity: {\n            address: {\n              city,\n              line1: street1,\n              postal_code: zip,\n              state\n            },\n            dob: {\n              day: dobDay,\n              month: dobMonth,\n              year: dobYear\n            },\n            first_name: first,\n            last_name: last,\n            // ssn_last_4: ssn.substr(-4),\n            // personal_id_number: ssn,\n            type: 'individual'\n          },\n          managed: true,\n          metadata: {\n            userId: userKey,\n            username\n          },\n          tos_acceptance: {\n            date: accepted,\n            ip: requestData.ip,\n            user_agent: requestData.userAgent.source\n          }\n        })\n        .then((customer) => {\n          // Create session\n          const now: number = Date.now();\n          const update: UserType = {\n            modified: now,\n            stripeAccountId: customer.id\n          };\n\n          const aqlQry: AqlQuery = aql`UPDATE ${userKey} WITH ${update} IN users LIMIT 1 RETURN NEW`;\n\n          return useDb(database).query(aqlQry)\n            .then((cursor: ArrayCursor) => cursor.next())\n            .then((updatedUser: UserType = {}) => !isEmpty(updatedUser))\n            .catch((error: Error) => logError({\n              action,\n              category: eventCategory,\n              label: 'db_error'\n            }, error, context).then(() => null));\n        });\n    });\n};\n\nexport const addCreditCard = (context: ApiContext, card: PaymentCardType): Promise<PaymentCardType> => {\n  const action: string = 'addCreditCard';\n  const {database, userId: sessionId} = context;\n\n  const {\n    accountNumber,\n    city,\n    country,\n    cvc,\n    expMonth,\n    expYear,\n    fullName,\n    street1,\n    state,\n    zip\n  }: PaymentCardType = card;\n  const formatNumber: number = parseNum(accountNumber, 16);\n\n  if(!formatNumber) {\n    throw new UserError('required_credit_card_number');\n  }\n\n  const formatExpMonth: number = parseNum(expMonth, 2);\n\n  if(!formatExpMonth) {\n    throw new UserError('required_credit_card_exp_month');\n  }\n\n  const formatExpYear: number = parseNum(expYear, 2);\n\n  if(!formatExpYear) {\n    throw new UserError('required_credit_card_exp_year');\n  }\n\n  const formatCvc: number = parseNum(cvc, 3);\n\n  if(!formatCvc) {\n    throw new UserError('required_credit_card_cvc');\n  }\n\n  const paymentCard: PaymentCardType = {};\n  const formatCity: string = parseVarChar(city, 32);\n\n  if(formatCity) {\n    paymentCard.city = formatCity;\n  }\n\n  const formatCountry: string = parseVarChar(country, 32);\n\n  if(formatCountry) {\n    paymentCard.country = formatCountry;\n  }\n\n  const formatFullName: string = parseVarChar(fullName, 32);\n\n  if(formatFullName) {\n    paymentCard.fullName = formatFullName;\n  }\n\n  const formatStreet: string = parseVarChar(street1, 32);\n\n  if(formatStreet) {\n    paymentCard.street1 = formatStreet;\n  }\n\n  const formatState: string = parseVarChar(state, 32);\n\n  if(formatState) {\n    paymentCard.state = formatState;\n  }\n\n  const formatZip: string = parseVarChar(zip, 32);\n\n  if(formatZip) {\n    paymentCard.zip = formatZip;\n  }\n\n  // Get stripe id\n  return getUser(context, sessionId)\n    .then((user: UserType = {}) => {\n      // Stripe\n      const stripeClient = stripe(Config.get('stripe.token'));\n\n      return stripeClient.customers\n        .createSource(user.stripeCustomerId, {\n          source: {\n            address_city: formatCity,\n            address_country: formatCountry,\n            address_line1: formatStreet,\n            address_state: formatState,\n            address_zip: formatZip,\n            currency: 'usd',\n            cvc: formatCvc,\n            exp_month: formatExpMonth,\n            exp_year: formatExpYear,\n            name: formatFullName,\n            number: formatNumber,\n            object: 'card'\n          }\n        })\n        .then((stripeObj) => {\n          const now: number = Date.now();\n          const insert = {\n            ...paymentCard,\n            _key: createHash(`user-payment-${sessionId}`),\n            accountNumber: stripeObj.last4,\n            added: now,\n            brand: stripeObj.brand,\n            cvc: stripeObj.cvc_check,\n            expMonth,\n            expYear,\n            modified: now,\n            userId: sessionId\n          };\n          const insertAqlQry: AqlQuery = aql`INSERT ${insert} IN creditCards RETURN NEW`;\n\n          return useDb(database).query(insertAqlQry)\n            .then((cursor: ArrayCursor) => cursor.next())\n            .then((newCard: PaymentCardType = {}) => {\n              if(!isEmpty(newCard)) {\n                // Add linked edge\n                const {_id: cardId, _key: cardKey} = card;\n                const edgeCollection = useDb(database).edgeCollection('hasPayment');\n                const edgeId = createHash(`payment-${cardKey}`);\n                const edge: any = {\n                  _key: edgeId\n                };\n\n                return edgeCollection.save(edge, `users/${sessionId}`, cardId).then(() => card);\n              }\n\n              return newCard;\n            });\n        })\n        .catch((error: Error) => logError({\n          action,\n          category: eventCategory,\n          label: 'payment_error'\n        }, error, context).then(() => null));\n    });\n};\n\nexport const updateCreditCard = (context: ApiContext, card: PaymentCardType): Promise<PaymentCardType> => {\n  const {database, userId: sessionId} = context;\n\n  const {\n    city,\n    country,\n    expMonth,\n    expYear,\n    fullName,\n    id,\n    street1,\n    state,\n    zip\n  }: PaymentCardType = card;\n\n  const formatId: string = parseId(id);\n\n  if(formatId) {\n    throw new UserError('required_credit_card_id');\n  }\n\n  const paymentCard: PaymentCardType = {};\n  const formatExpMonth: number = parseNum(expMonth, 2);\n  const formatExpYear: number = parseNum(expYear, 2);\n  const formatCity: string = parseVarChar(city, 32);\n  const formatCountry: string = parseVarChar(country, 2);\n  const formatFullName: string = parseVarChar(fullName, 32);\n  const formatStreet1: string = parseString(street1, 32);\n  const formatState: string = parseVarChar(state, 2);\n  const formatZip: string = parseVarChar(zip, 10);\n\n  if(formatExpMonth) {\n    paymentCard.expMonth = formatExpMonth;\n  }\n\n  if(formatExpYear) {\n    paymentCard.expYear = formatExpYear;\n  }\n\n  if(formatCity) {\n    paymentCard.city = formatCity;\n  }\n\n  if(formatCountry) {\n    paymentCard.country = formatCountry;\n  }\n\n  if(formatFullName) {\n    paymentCard.fullName = formatFullName;\n  }\n\n  if(formatStreet1) {\n    paymentCard.street1 = formatStreet1;\n  }\n\n  if(formatState) {\n    paymentCard.state = formatState;\n  }\n\n  if(formatZip) {\n    paymentCard.zip = formatZip;\n  }\n\n  const update: any = paymentCard;\n  const aqlQry: AqlQuery = aql`\n      LET updatedCard = FIRST(\n        FOR c IN creditCards\n        FILTER c._key == ${formatId} && c.userId == ${sessionId}\n        UPDATE c WITH ${update} IN creditCards\n        LIMIT 1\n        RETURN NEW\n      )\n      LET user = FIRST(\n        FOR u IN users\n        FILTER u._key == ${sessionId}\n        LIMIT 1\n        RETURN u\n      )\n      RETURN {user: user, card: updatedCard}`;\n\n  return useDb(database).query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((results = {card: {}, user: {}}) => {\n      const updatedCard: PaymentCardType = results.card;\n      const {user} = results;\n\n      if(isEmpty(updatedCard)) {\n        throw new UserError('not_found');\n      }\n\n      // Stripe\n      const stripeClient = stripe(Config.get('stripe.token'));\n\n      return stripeClient.customers\n        .updateCard(user.stripeCustomerId, card.stripeId, {\n          address_city: formatCity,\n          address_country: formatCountry,\n          address_line1: formatStreet1,\n          address_state: formatState,\n          address_zip: formatZip,\n          exp_month: formatExpMonth,\n          exp_year: formatExpYear,\n          name: formatFullName\n        })\n        .then(() => card)\n        .catch((error: Error) => {\n          console.log('payments::updateCard::error', error);\n          throw new UserError('payment_error');\n        });\n    });\n};\n\nexport const getCreditCards = (context: ApiContext): Promise<PaymentCardType[]> => {\n  const action: string = 'getCreditCards';\n  const {database, userId: sessionId} = context;\n  const aqlQry: AqlQuery = aql`FOR c IN creditCards\n        FILTER c.userId == ${sessionId}\n        RETURN c`;\n\n  return useDb(database).query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .then((list: PaymentCardType[] = []) => list)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const deleteCreditCard = (context: ApiContext, cardId: string): Promise<boolean> => {\n  const {database, userId: sessionId} = context;\n  const formatCardId: string = parseId(cardId);\n  const aqlQry: AqlQuery = aql`\n      LET card = FIRST(\n        FOR c IN creditCards\n        FILTER c._key == ${formatCardId} && c.userId == ${sessionId}\n        LIMIT 1\n        REMOVE c IN creditCards\n        RETURN OLD\n      )\n      LET user = FIRST(\n        FOR u IN users\n        FILTER u._key == ${sessionId}\n        LIMIT 1\n        RETURN u\n      )\n      RETURN {user: user, card: card}`;\n\n  return useDb(database).query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((result = {card: {}, user: {}}) => {\n      if(isEmpty(result)) {\n        return false;\n      }\n\n      const {card, user} = result;\n      const {_key: cardKey} = card;\n\n      // Remove linked edges\n      const edgeCollection = useDb(database).edgeCollection('hasPayment');\n\n      return edgeCollection.outEdges(cardKey)\n        .then((edges) => {\n          if(edges.length) {\n            return Promise.all(\n              edges.map((edge) => {\n                const {_key: edgeKey} = edge;\n                const removeAqlQry: AqlQuery = aql`REMOVE {_key:${edgeKey}} IN hasPayment`;\n                return useDb(database).query(removeAqlQry);\n              }))\n              .then(() => {\n                // Stripe\n                const stripeClient = stripe(Config.get('stripe.token'));\n\n                return stripeClient.customers\n                  .deleteCard(user.stripeCustomerId, card.stripeId)\n                  .then(() => true)\n                  .catch((error: Error) => {\n                    console.log('payments::deleteCard::error', error);\n                    throw new UserError('payment_error');\n                  });\n              });\n          }\n\n          return false;\n        });\n    });\n};\n\nexport const deletePaymentAccountBank = (context: ApiContext, bankId: string): Promise<boolean> => {\n  const {database, userId: sessionId} = context;\n\n  // Clean db\n  const update: UserType = {\n    bankAccount: '',\n    bankFullName: '',\n    bankId: '',\n    bankRouting: '',\n    modified: Date.now()\n  };\n  const aqlQry: AqlQuery = aql`UPDATE ${sessionId} WITH ${update} IN users LIMIT 1 RETURN NEW`;\n\n  return useDb(database).query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((user: UserType = {}) => {\n      // Stripe\n      const stripeClient = stripe(Config.get('stripe.token'));\n\n      return stripeClient.customers\n        .deleteSource(user.stripeAccountId, bankId)\n        .then((response = {deleted: false}) => response.deleted)\n        .catch(() => Promise.resolve(false));\n    });\n};\n\nexport const createPaymentTransfer = (context: ApiContext, transfer: PaymentTransfer): Promise<PaymentTransfer> => {\n  const {database, userId: sessionId} = context;\n  const {amount, currency} = transfer;\n  const formatAmount: number = parseNum(amount);\n  const formatCurrency: string = parseChar(currency, 3, 'USD').toUpperCase();\n\n  return getUser(context, sessionId)\n    .then((user: UserType) => {\n      // Stripe\n      const stripeClient = stripe(Config.get('stripe.token'));\n\n      return stripeClient.transfers\n        .create({\n          amount: formatAmount,\n          currency: formatCurrency,\n          destination: user.stripeAccountId\n        })\n        .then((stripeTransfer) => {\n          console.log(stripeTransfer);\n          const now: number = Date.now();\n          const insert: PaymentTransfer = {\n            added: now,\n            amount: formatAmount,\n            currency: formatCurrency,\n            modified: now,\n            userId: sessionId\n          };\n          const aqlQry: AqlQuery = aql`INSERT ${insert} IN transfers RETURN NEW`;\n\n          return useDb(database).query(aqlQry)\n            .then((cursor: ArrayCursor) => cursor.next())\n            .then((newTransfer: PaymentTransfer) => newTransfer);\n        });\n    });\n};\n\nexport const createPaymentHold = (context: ApiContext, payment: PaymentCharge): Promise<PaymentCharge> => {\n  const {database, userId: sessionId} = context;\n  const {amount, capture, cardId, currency, description} = payment;\n  const formatCurrency = parseChar(currency, 3, 'USD').toUpperCase();\n\n  const stripeClient = stripe(Config.get('stripe.token'));\n\n  return stripeClient.charges\n    .create({\n      amount,\n      capture,\n      currency: formatCurrency,\n      description,\n      source: cardId\n    })\n    .then((stripeCharge) => {\n      const now: number = Date.now();\n      const insert: PaymentCharge = {\n        added: now,\n        amount,\n        capture,\n        cardId,\n        chargeFailCode: stripeCharge.failure_code,\n        chargeFailMsg: stripeCharge.failure_message,\n        chargeId: stripeCharge.id,\n        chargeStatus: stripeCharge.status,\n        currency: formatCurrency,\n        description,\n        modified: now,\n        userId: sessionId\n      };\n      const aqlQry: AqlQuery = aql`INSERT ${insert} IN payments RETURN NEW`;\n\n      return useDb(database).query(aqlQry)\n        .then((cursor: ArrayCursor) => cursor.next())\n        .then((newPayment: PaymentCharge) => newPayment);\n    })\n    .catch((error: Error) => {\n      console.log('payments::createHold::error', error);\n      throw new UserError('payment_error');\n    });\n};\n"]}
|
|
796
|
+
|
|
797
|
+
exports.createPaymentHold = createPaymentHold;
|
|
798
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/data/payments.ts"],"names":["eventCategory","addCustomerAccount","context","action","database","sessionId","userId","username","stripeClient","stripe","Config","get","customers","create","metadata","then","customer","now","Date","update","modified","stripeCustomerId","id","aqlQry","aql","query","cursor","next","updatedUser","catch","error","category","label","addPaymentAccountBank","bankAccount","accountNumber","fullName","routing","formatAccount","UserError","formatFullName","formatRouting","user","createSource","stripeAccountId","source","account_holder_name","account_holder_type","account_number","country","currency","object","routing_number","account","last4","bankFullName","bankId","bankRouting","msg","message","addPaymentAccountCard","card","requestData","userKey","_key","city","dob","email","first","last","dobDay","DateTime","fromMillis","toFormat","dobMonth","dobYear","acceptedTerms","expMonth","expYear","state","street1","zip","accepted","local","millisecond","accounts","external_account","exp_month","exp_year","number","legal_entity","address","line1","postal_code","day","month","year","first_name","last_name","type","managed","tos_acceptance","date","ip","user_agent","userAgent","addCreditCard","cvc","formatNumber","formatExpMonth","formatExpYear","formatCvc","paymentCard","formatCity","formatCountry","formatStreet","formatState","formatZip","address_city","address_country","address_line1","address_state","address_zip","name","stripeObj","insert","added","brand","cvc_check","insertAqlQry","newCard","cardId","_id","cardKey","edgeCollection","edgeId","edge","save","updateCreditCard","formatId","formatStreet1","results","updatedCard","updateCard","stripeId","console","log","getCreditCards","all","list","deleteCreditCard","formatCardId","result","outEdges","edges","length","Promise","map","edgeKey","removeAqlQry","deleteCard","deletePaymentAccountBank","deleteSource","response","deleted","resolve","createPaymentTransfer","transfer","amount","formatAmount","formatCurrency","toUpperCase","transfers","destination","stripeTransfer","newTransfer","createPaymentHold","payment","capture","description","charges","stripeCharge","chargeFailCode","failure_code","chargeFailMsg","failure_message","chargeId","chargeStatus","status","newPayment"],"mappings":"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","sourcesContent":["import {createHash, parseChar, parseId, parseNum, parseString, parseVarChar} from '@nlabs/utils';\nimport {aql} from 'arangojs';\nimport {AqlQuery} from 'arangojs/lib/cjs/aql-query';\nimport {ArrayCursor} from 'arangojs/lib/cjs/cursor';\nimport {UserError} from 'graphql-errors';\nimport isEmpty from 'lodash/isEmpty';\nimport {DateTime} from 'luxon';\nimport * as stripe from 'stripe';\n\nimport {Config} from '../config';\nimport {ApiContext, PaymentBankAccount, PaymentCardType, PaymentCharge, PaymentTransfer, UserType} from '../types';\nimport {logError} from '../utils/analytics';\nimport {useDb} from '../utils/arangodb';\nimport {getUser} from './users';\n\n/**\n * Copyright (c) 2019-Present, Nitrogen Labs, Inc.\n * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.\n */\nconst eventCategory: string = 'payments';\n\nexport const addCustomerAccount = (context: ApiContext): Promise<boolean> => {\n  const action: string = 'addCustomerAccount';\n  const {database, userId: sessionId, username} = context;\n\n  // Stripe\n  const stripeClient = stripe(Config.get('stripe.token'));\n\n  return stripeClient.customers\n    .create({\n      metadata: {\n        userId: sessionId,\n        username\n      }\n    })\n    .then((customer) => {\n      // Create session\n      const now: number = Date.now();\n      const update: UserType = {\n        modified: now,\n        stripeCustomerId: customer.id\n      };\n\n      const aqlQry: AqlQuery = aql`UPDATE ${sessionId} WITH ${update} IN users LIMIT 1 RETURN NEW`;\n\n      return useDb(database).query(aqlQry)\n        .then((cursor: ArrayCursor) => cursor.next())\n        .then((updatedUser: UserType = {}) => !isEmpty(updatedUser))\n        .catch((error: Error) => logError({\n          action,\n          category: eventCategory,\n          label: 'db_error'\n        }, error, context).then(() => null));\n    });\n};\n\nexport const addPaymentAccountBank = (context: ApiContext, bankAccount: PaymentBankAccount): Promise<boolean> => {\n  const action: string = 'addPaymentAccountBank';\n  const {database, userId: sessionId} = context;\n\n  // Params\n  const {\n    accountNumber,\n    fullName,\n    routing\n  } = bankAccount;\n\n  const formatAccount: string = parseString(accountNumber, 32);\n\n  if(formatAccount === '') {\n    throw new UserError('required_account_number');\n  }\n\n  const formatFullName: string = parseVarChar(fullName, 128);\n\n  if(formatFullName === '') {\n    throw new UserError('required_full_name');\n  }\n\n  const formatRouting: string = parseString(routing, 32);\n\n  if(formatRouting === '') {\n    throw new UserError('required_routing_number');\n  }\n\n  return getUser(context, sessionId)\n    .then((user: UserType) => {\n      // Stripe\n      const stripeClient = stripe(Config.get('stripe.token'));\n\n      return stripeClient.customers\n        .createSource(user.stripeAccountId, {\n          source: {\n            account_holder_name: formatFullName,\n            account_holder_type: 'individual',\n            account_number: formatAccount,\n            country: 'US',\n            currency: 'USD',\n            object: 'bank_account',\n            routing_number: formatRouting\n          }\n        })\n        .then((account) => {\n          const update: any = {\n            bankAccount: account.last4,\n            bankFullName: formatFullName,\n            bankId: account.id,\n            bankRouting: account.routing_number,\n            modified: Date.now()\n          };\n\n          const aqlQry: AqlQuery = aql`UPDATE ${sessionId} WITH ${update} IN users LIMIT 1 RETURN NEW`;\n\n          return useDb(database).query(aqlQry)\n            .then((cursor: ArrayCursor) => cursor.next())\n            .then((updatedUser: UserType = {}) => updatedUser);\n        })\n        .catch((error: Error) => {\n          const msg = error.message;\n\n          if(msg === 'A bank account with that routing number and account number ' +\n            'already exists for this customer.') {\n            return logError({\n              action,\n              category: eventCategory,\n              label: 'bank_account_exists'\n            }, error, context).then(() => null);\n          }\n          return logError({\n            action,\n            category: eventCategory,\n            label: 'payment_error'\n          }, error, context).then(() => null);\n        });\n    });\n};\n\nexport const addPaymentAccountCard = (context: ApiContext, card: PaymentCardType, requestData): Promise<UserType> => {\n  const action: string = 'addPaymentAccountCard';\n  const {database, userId: sessionId} = context;\n\n  return getUser(context, sessionId)\n    .then((user: UserType) => {\n      // User\n      const {\n        _key: userKey,\n        country = 'US',\n        city,\n        currency = 'USD',\n        dob,\n        email,\n        first,\n        last,\n        username\n      }: UserType = user;\n\n      // Birthday\n      const dobDay: number = +(DateTime.fromMillis(dob).toFormat('d'));\n      const dobMonth: number = +(DateTime.fromMillis(dob).toFormat('M'));\n      const dobYear: number = +(DateTime.fromMillis(dob).toFormat('y'));\n\n      // Card\n      const {\n        accountNumber,\n        acceptedTerms,\n        expMonth,\n        expYear,\n        state,\n        street1,\n        zip\n      }: PaymentCardType = card;\n\n      // Acceptance\n      const accepted: number = acceptedTerms ? DateTime.local().millisecond : 0;\n\n      // Stripe\n      const stripeClient = stripe(Config.get('stripe.token'));\n\n      return stripeClient.accounts\n        .create({\n          country,\n          email,\n          external_account: {\n            currency,\n            exp_month: expMonth,\n            exp_year: expYear,\n            number: accountNumber,\n            object: 'card'\n          },\n          legal_entity: {\n            address: {\n              city,\n              line1: street1,\n              postal_code: zip,\n              state\n            },\n            dob: {\n              day: dobDay,\n              month: dobMonth,\n              year: dobYear\n            },\n            first_name: first,\n            last_name: last,\n            // ssn_last_4: ssn.substr(-4),\n            // personal_id_number: ssn,\n            type: 'individual'\n          },\n          managed: true,\n          metadata: {\n            userId: userKey,\n            username\n          },\n          tos_acceptance: {\n            date: accepted,\n            ip: requestData.ip,\n            user_agent: requestData.userAgent.source\n          }\n        })\n        .then((customer) => {\n          // Create session\n          const now: number = Date.now();\n          const update: UserType = {\n            modified: now,\n            stripeAccountId: customer.id\n          };\n\n          const aqlQry: AqlQuery = aql`UPDATE ${userKey} WITH ${update} IN users LIMIT 1 RETURN NEW`;\n\n          return useDb(database).query(aqlQry)\n            .then((cursor: ArrayCursor) => cursor.next())\n            .then((updatedUser: UserType = {}) => !isEmpty(updatedUser))\n            .catch((error: Error) => logError({\n              action,\n              category: eventCategory,\n              label: 'db_error'\n            }, error, context).then(() => null));\n        });\n    });\n};\n\nexport const addCreditCard = (context: ApiContext, card: PaymentCardType): Promise<PaymentCardType> => {\n  const action: string = 'addCreditCard';\n  const {database, userId: sessionId} = context;\n\n  const {\n    accountNumber,\n    city,\n    country,\n    cvc,\n    expMonth,\n    expYear,\n    fullName,\n    street1,\n    state,\n    zip\n  }: PaymentCardType = card;\n  const formatNumber: number = parseNum(accountNumber, 16);\n\n  if(!formatNumber) {\n    throw new UserError('required_credit_card_number');\n  }\n\n  const formatExpMonth: number = parseNum(expMonth, 2);\n\n  if(!formatExpMonth) {\n    throw new UserError('required_credit_card_exp_month');\n  }\n\n  const formatExpYear: number = parseNum(expYear, 2);\n\n  if(!formatExpYear) {\n    throw new UserError('required_credit_card_exp_year');\n  }\n\n  const formatCvc: number = parseNum(cvc, 3);\n\n  if(!formatCvc) {\n    throw new UserError('required_credit_card_cvc');\n  }\n\n  const paymentCard: PaymentCardType = {};\n  const formatCity: string = parseVarChar(city, 32);\n\n  if(formatCity) {\n    paymentCard.city = formatCity;\n  }\n\n  const formatCountry: string = parseVarChar(country, 32);\n\n  if(formatCountry) {\n    paymentCard.country = formatCountry;\n  }\n\n  const formatFullName: string = parseVarChar(fullName, 32);\n\n  if(formatFullName) {\n    paymentCard.fullName = formatFullName;\n  }\n\n  const formatStreet: string = parseVarChar(street1, 32);\n\n  if(formatStreet) {\n    paymentCard.street1 = formatStreet;\n  }\n\n  const formatState: string = parseVarChar(state, 32);\n\n  if(formatState) {\n    paymentCard.state = formatState;\n  }\n\n  const formatZip: string = parseVarChar(zip, 32);\n\n  if(formatZip) {\n    paymentCard.zip = formatZip;\n  }\n\n  // Get stripe id\n  return getUser(context, sessionId)\n    .then((user: UserType = {}) => {\n      // Stripe\n      const stripeClient = stripe(Config.get('stripe.token'));\n\n      return stripeClient.customers\n        .createSource(user.stripeCustomerId, {\n          source: {\n            address_city: formatCity,\n            address_country: formatCountry,\n            address_line1: formatStreet,\n            address_state: formatState,\n            address_zip: formatZip,\n            currency: 'usd',\n            cvc: formatCvc,\n            exp_month: formatExpMonth,\n            exp_year: formatExpYear,\n            name: formatFullName,\n            number: formatNumber,\n            object: 'card'\n          }\n        })\n        .then((stripeObj) => {\n          const now: number = Date.now();\n          const insert = {\n            ...paymentCard,\n            _key: createHash(`user-payment-${sessionId}`),\n            accountNumber: stripeObj.last4,\n            added: now,\n            brand: stripeObj.brand,\n            cvc: stripeObj.cvc_check,\n            expMonth,\n            expYear,\n            modified: now,\n            userId: sessionId\n          };\n          const insertAqlQry: AqlQuery = aql`INSERT ${insert} IN creditCards RETURN NEW`;\n\n          return useDb(database).query(insertAqlQry)\n            .then((cursor: ArrayCursor) => cursor.next())\n            .then((newCard: PaymentCardType = {}) => {\n              if(!isEmpty(newCard)) {\n                // Add linked edge\n                const {_id: cardId, _key: cardKey} = card;\n                const edgeCollection = useDb(database).edgeCollection('hasPayment');\n                const edgeId = createHash(`payment-${cardKey}`);\n                const edge: any = {\n                  _key: edgeId\n                };\n\n                return edgeCollection.save(edge, `users/${sessionId}`, cardId).then(() => card);\n              }\n\n              return newCard;\n            });\n        })\n        .catch((error: Error) => logError({\n          action,\n          category: eventCategory,\n          label: 'payment_error'\n        }, error, context).then(() => null));\n    });\n};\n\nexport const updateCreditCard = (context: ApiContext, card: PaymentCardType): Promise<PaymentCardType> => {\n  const {database, userId: sessionId} = context;\n\n  const {\n    city,\n    country,\n    expMonth,\n    expYear,\n    fullName,\n    id,\n    street1,\n    state,\n    zip\n  }: PaymentCardType = card;\n\n  const formatId: string = parseId(id);\n\n  if(formatId) {\n    throw new UserError('required_credit_card_id');\n  }\n\n  const paymentCard: PaymentCardType = {};\n  const formatExpMonth: number = parseNum(expMonth, 2);\n  const formatExpYear: number = parseNum(expYear, 2);\n  const formatCity: string = parseVarChar(city, 32);\n  const formatCountry: string = parseVarChar(country, 2);\n  const formatFullName: string = parseVarChar(fullName, 32);\n  const formatStreet1: string = parseString(street1, 32);\n  const formatState: string = parseVarChar(state, 2);\n  const formatZip: string = parseVarChar(zip, 10);\n\n  if(formatExpMonth) {\n    paymentCard.expMonth = formatExpMonth;\n  }\n\n  if(formatExpYear) {\n    paymentCard.expYear = formatExpYear;\n  }\n\n  if(formatCity) {\n    paymentCard.city = formatCity;\n  }\n\n  if(formatCountry) {\n    paymentCard.country = formatCountry;\n  }\n\n  if(formatFullName) {\n    paymentCard.fullName = formatFullName;\n  }\n\n  if(formatStreet1) {\n    paymentCard.street1 = formatStreet1;\n  }\n\n  if(formatState) {\n    paymentCard.state = formatState;\n  }\n\n  if(formatZip) {\n    paymentCard.zip = formatZip;\n  }\n\n  const update: any = paymentCard;\n  const aqlQry: AqlQuery = aql`\n      LET updatedCard = FIRST(\n        FOR c IN creditCards\n        FILTER c._key == ${formatId} && c.userId == ${sessionId}\n        UPDATE c WITH ${update} IN creditCards\n        LIMIT 1\n        RETURN NEW\n      )\n      LET user = FIRST(\n        FOR u IN users\n        FILTER u._key == ${sessionId}\n        LIMIT 1\n        RETURN u\n      )\n      RETURN {user: user, card: updatedCard}`;\n\n  return useDb(database).query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((results = {card: {}, user: {}}) => {\n      const updatedCard: PaymentCardType = results.card;\n      const {user} = results;\n\n      if(isEmpty(updatedCard)) {\n        throw new UserError('not_found');\n      }\n\n      // Stripe\n      const stripeClient = stripe(Config.get('stripe.token'));\n\n      return stripeClient.customers\n        .updateCard(user.stripeCustomerId, card.stripeId, {\n          address_city: formatCity,\n          address_country: formatCountry,\n          address_line1: formatStreet1,\n          address_state: formatState,\n          address_zip: formatZip,\n          exp_month: formatExpMonth,\n          exp_year: formatExpYear,\n          name: formatFullName\n        })\n        .then(() => card)\n        .catch((error: Error) => {\n          console.log('payments::updateCard::error', error);\n          throw new UserError('payment_error');\n        });\n    });\n};\n\nexport const getCreditCards = (context: ApiContext): Promise<PaymentCardType[]> => {\n  const action: string = 'getCreditCards';\n  const {database, userId: sessionId} = context;\n  const aqlQry: AqlQuery = aql`FOR c IN creditCards\n        FILTER c.userId == ${sessionId}\n        RETURN c`;\n\n  return useDb(database).query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .then((list: PaymentCardType[] = []) => list)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const deleteCreditCard = (context: ApiContext, cardId: string): Promise<boolean> => {\n  const {database, userId: sessionId} = context;\n  const formatCardId: string = parseId(cardId);\n  const aqlQry: AqlQuery = aql`\n      LET card = FIRST(\n        FOR c IN creditCards\n        FILTER c._key == ${formatCardId} && c.userId == ${sessionId}\n        LIMIT 1\n        REMOVE c IN creditCards\n        RETURN OLD\n      )\n      LET user = FIRST(\n        FOR u IN users\n        FILTER u._key == ${sessionId}\n        LIMIT 1\n        RETURN u\n      )\n      RETURN {user: user, card: card}`;\n\n  return useDb(database).query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((result = {card: {}, user: {}}) => {\n      if(isEmpty(result)) {\n        return false;\n      }\n\n      const {card, user} = result;\n      const {_key: cardKey} = card;\n\n      // Remove linked edges\n      const edgeCollection = useDb(database).edgeCollection('hasPayment');\n\n      return edgeCollection.outEdges(cardKey)\n        .then((edges) => {\n          if(edges.length) {\n            return Promise.all(\n              edges.map((edge) => {\n                const {_key: edgeKey} = edge;\n                const removeAqlQry: AqlQuery = aql`REMOVE {_key:${edgeKey}} IN hasPayment`;\n                return useDb(database).query(removeAqlQry);\n              }))\n              .then(() => {\n                // Stripe\n                const stripeClient = stripe(Config.get('stripe.token'));\n\n                return stripeClient.customers\n                  .deleteCard(user.stripeCustomerId, card.stripeId)\n                  .then(() => true)\n                  .catch((error: Error) => {\n                    console.log('payments::deleteCard::error', error);\n                    throw new UserError('payment_error');\n                  });\n              });\n          }\n\n          return false;\n        });\n    });\n};\n\nexport const deletePaymentAccountBank = (context: ApiContext, bankId: string): Promise<boolean> => {\n  const {database, userId: sessionId} = context;\n\n  // Clean db\n  const update: UserType = {\n    bankAccount: '',\n    bankFullName: '',\n    bankId: '',\n    bankRouting: '',\n    modified: Date.now()\n  };\n  const aqlQry: AqlQuery = aql`UPDATE ${sessionId} WITH ${update} IN users LIMIT 1 RETURN NEW`;\n\n  return useDb(database).query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((user: UserType = {}) => {\n      // Stripe\n      const stripeClient = stripe(Config.get('stripe.token'));\n\n      return stripeClient.customers\n        .deleteSource(user.stripeAccountId, bankId)\n        .then((response = {deleted: false}) => response.deleted)\n        .catch(() => Promise.resolve(false));\n    });\n};\n\nexport const createPaymentTransfer = (context: ApiContext, transfer: PaymentTransfer): Promise<PaymentTransfer> => {\n  const {database, userId: sessionId} = context;\n  const {amount, currency} = transfer;\n  const formatAmount: number = parseNum(amount);\n  const formatCurrency: string = parseChar(currency, 3, 'USD').toUpperCase();\n\n  return getUser(context, sessionId)\n    .then((user: UserType) => {\n      // Stripe\n      const stripeClient = stripe(Config.get('stripe.token'));\n\n      return stripeClient.transfers\n        .create({\n          amount: formatAmount,\n          currency: formatCurrency,\n          destination: user.stripeAccountId\n        })\n        .then((stripeTransfer) => {\n          console.log(stripeTransfer);\n          const now: number = Date.now();\n          const insert: PaymentTransfer = {\n            added: now,\n            amount: formatAmount,\n            currency: formatCurrency,\n            modified: now,\n            userId: sessionId\n          };\n          const aqlQry: AqlQuery = aql`INSERT ${insert} IN transfers RETURN NEW`;\n\n          return useDb(database).query(aqlQry)\n            .then((cursor: ArrayCursor) => cursor.next())\n            .then((newTransfer: PaymentTransfer) => newTransfer);\n        });\n    });\n};\n\nexport const createPaymentHold = (context: ApiContext, payment: PaymentCharge): Promise<PaymentCharge> => {\n  const {database, userId: sessionId} = context;\n  const {amount, capture, cardId, currency, description} = payment;\n  const formatCurrency = parseChar(currency, 3, 'USD').toUpperCase();\n\n  const stripeClient = stripe(Config.get('stripe.token'));\n\n  return stripeClient.charges\n    .create({\n      amount,\n      capture,\n      currency: formatCurrency,\n      description,\n      source: cardId\n    })\n    .then((stripeCharge) => {\n      const now: number = Date.now();\n      const insert: PaymentCharge = {\n        added: now,\n        amount,\n        capture,\n        cardId,\n        chargeFailCode: stripeCharge.failure_code,\n        chargeFailMsg: stripeCharge.failure_message,\n        chargeId: stripeCharge.id,\n        chargeStatus: stripeCharge.status,\n        currency: formatCurrency,\n        description,\n        modified: now,\n        userId: sessionId\n      };\n      const aqlQry: AqlQuery = aql`INSERT ${insert} IN payments RETURN NEW`;\n\n      return useDb(database).query(aqlQry)\n        .then((cursor: ArrayCursor) => cursor.next())\n        .then((newPayment: PaymentCharge) => newPayment);\n    })\n    .catch((error: Error) => {\n      console.log('payments::createHold::error', error);\n      throw new UserError('payment_error');\n    });\n};\n"]}
|