@nlabs/reaktor 0.1.2 → 0.1.3
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/.DS_Store +0 -0
- package/lib/config.d.ts +21 -0
- package/lib/config.js +130 -0
- package/lib/data/conversations.d.ts +6 -0
- package/lib/data/conversations.js +197 -0
- package/lib/data/dynamodb.d.ts +8 -0
- package/lib/data/dynamodb.js +139 -0
- package/lib/data/email.d.ts +7 -0
- package/lib/data/email.js +163 -0
- package/lib/data/files.d.ts +16 -0
- package/lib/data/files.js +406 -0
- package/lib/data/groups.d.ts +13 -0
- package/lib/data/groups.js +354 -0
- package/lib/data/images.d.ts +12 -0
- package/lib/data/images.js +667 -0
- package/{src/data/index.ts → lib/data/index.d.ts} +1 -5
- package/lib/data/index.js +24 -0
- package/lib/data/ios.d.ts +6 -0
- package/lib/data/ios.js +302 -0
- package/lib/data/locations.d.ts +3 -0
- package/lib/data/locations.js +132 -0
- package/lib/data/messages.d.ts +9 -0
- package/lib/data/messages.js +248 -0
- package/lib/data/notifications.d.ts +5 -0
- package/lib/data/notifications.js +42 -0
- package/lib/data/payments.d.ts +11 -0
- package/lib/data/payments.js +748 -0
- package/lib/data/posts.d.ts +22 -0
- package/lib/data/posts.js +578 -0
- package/lib/data/reactions.d.ts +6 -0
- package/lib/data/reactions.js +218 -0
- package/lib/data/s3.d.ts +6 -0
- package/lib/data/s3.js +103 -0
- package/lib/data/search.d.ts +3 -0
- package/lib/data/search.js +98 -0
- package/lib/data/sms.d.ts +3 -0
- package/lib/data/sms.js +59 -0
- package/lib/data/subscription.d.ts +7 -0
- package/lib/data/subscription.js +284 -0
- package/lib/data/tags.d.ts +14 -0
- package/lib/data/tags.js +304 -0
- package/lib/data/users.d.ts +12 -0
- package/lib/data/users.js +310 -0
- package/lib/index.d.ts +3 -0
- package/lib/index.js +8 -0
- package/lib/types/apps.d.ts +43 -0
- package/lib/types/apps.js +2 -0
- package/lib/types/arangodb.d.ts +17 -0
- package/lib/types/arangodb.js +2 -0
- package/lib/types/auth.d.ts +10 -0
- package/lib/types/auth.js +2 -0
- package/lib/types/conversations.d.ts +6 -0
- package/lib/types/conversations.js +2 -0
- package/lib/types/email.d.ts +12 -0
- package/lib/types/email.js +2 -0
- package/lib/types/files.d.ts +26 -0
- package/lib/types/files.js +2 -0
- package/lib/types/google.d.ts +27 -0
- package/lib/types/google.js +2 -0
- package/lib/types/groups.d.ts +21 -0
- package/lib/types/groups.js +2 -0
- package/lib/types/images.d.ts +24 -0
- package/lib/types/images.js +2 -0
- package/{src/types/index.ts → lib/types/index.d.ts} +0 -4
- package/lib/types/index.js +22 -0
- package/lib/types/locations.d.ts +20 -0
- package/lib/types/locations.js +2 -0
- package/lib/types/messages.d.ts +12 -0
- package/lib/types/messages.js +2 -0
- package/lib/types/notifications.d.ts +19 -0
- package/lib/types/notifications.js +2 -0
- package/lib/types/payments.d.ts +114 -0
- package/lib/types/payments.js +2 -0
- package/lib/types/posts.d.ts +28 -0
- package/lib/types/posts.js +2 -0
- package/lib/types/reactions.d.ts +4 -0
- package/lib/types/reactions.js +2 -0
- package/lib/types/tags.d.ts +9 -0
- package/lib/types/tags.js +2 -0
- package/lib/types/users.d.ts +78 -0
- package/lib/types/users.js +2 -0
- package/lib/utils/analytics.d.ts +3 -0
- package/lib/utils/analytics.js +47 -0
- package/lib/utils/arangodb.d.ts +9 -0
- package/lib/utils/arangodb.js +98 -0
- package/lib/utils/auth.d.ts +7 -0
- package/lib/utils/auth.js +80 -0
- package/lib/utils/graphql.d.ts +1 -0
- package/lib/utils/graphql.js +7 -0
- package/{src/utils/index.ts → lib/utils/index.d.ts} +0 -4
- package/lib/utils/index.js +11 -0
- package/lib/utils/objects.d.ts +3 -0
- package/lib/utils/objects.js +34 -0
- package/lib/utils/redis.d.ts +1 -0
- package/lib/utils/redis.js +15 -0
- package/package.json +5 -5
- package/.vscode/extensions.json +0 -15
- package/.vscode/settings.json +0 -82
- package/lex.config.js +0 -4
- package/src/config.ts +0 -127
- package/src/data/conversations.ts +0 -181
- package/src/data/dynamodb.ts +0 -157
- package/src/data/email.ts +0 -163
- package/src/data/files.ts +0 -352
- package/src/data/groups.ts +0 -308
- package/src/data/images.ts +0 -606
- package/src/data/ios.ts +0 -249
- package/src/data/locations.ts +0 -114
- package/src/data/messages.ts +0 -237
- package/src/data/notifications.ts +0 -48
- package/src/data/payments.ts +0 -675
- package/src/data/posts.ts +0 -580
- package/src/data/reactions.ts +0 -186
- package/src/data/s3.ts +0 -117
- package/src/data/search.ts +0 -74
- package/src/data/sms.ts +0 -60
- package/src/data/subscription.ts +0 -228
- package/src/data/tags.ts +0 -230
- package/src/data/users.ts +0 -254
- package/src/index.ts +0 -7
- package/src/types/apps.ts +0 -56
- package/src/types/arangodb.ts +0 -23
- package/src/types/auth.ts +0 -20
- package/src/types/conversations.ts +0 -11
- package/src/types/email.ts +0 -17
- package/src/types/files.ts +0 -31
- package/src/types/google.ts +0 -37
- package/src/types/groups.ts +0 -27
- package/src/types/images.ts +0 -32
- package/src/types/locations.ts +0 -24
- package/src/types/messages.ts +0 -16
- package/src/types/notifications.ts +0 -26
- package/src/types/payments.ts +0 -129
- package/src/types/posts.ts +0 -34
- package/src/types/reactions.ts +0 -8
- package/src/types/tags.ts +0 -13
- package/src/types/users.ts +0 -89
- package/src/utils/analytics.ts +0 -41
- package/src/utils/arangodb.ts +0 -100
- package/src/utils/auth.ts +0 -61
- package/src/utils/graphql.ts +0 -7
- package/src/utils/objects.ts +0 -34
- package/src/utils/redis.ts +0 -17
- package/tsconfig.json +0 -45
|
@@ -0,0 +1,748 @@
|
|
|
1
|
+
function _templateObject11() {
|
|
2
|
+
var data = _taggedTemplateLiteral(["INSERT ", " IN payments RETURN NEW"]);
|
|
3
|
+
|
|
4
|
+
_templateObject11 = function _templateObject11() {
|
|
5
|
+
return data;
|
|
6
|
+
};
|
|
7
|
+
|
|
8
|
+
return data;
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
function _templateObject10() {
|
|
12
|
+
var data = _taggedTemplateLiteral(["INSERT ", " IN transfers RETURN NEW"]);
|
|
13
|
+
|
|
14
|
+
_templateObject10 = function _templateObject10() {
|
|
15
|
+
return data;
|
|
16
|
+
};
|
|
17
|
+
|
|
18
|
+
return data;
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
function _templateObject9() {
|
|
22
|
+
var data = _taggedTemplateLiteral(["UPDATE ", " WITH ", " IN users LIMIT 1 RETURN NEW"]);
|
|
23
|
+
|
|
24
|
+
_templateObject9 = function _templateObject9() {
|
|
25
|
+
return data;
|
|
26
|
+
};
|
|
27
|
+
|
|
28
|
+
return data;
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
function _templateObject8() {
|
|
32
|
+
var data = _taggedTemplateLiteral(["REMOVE {_key:", "} IN hasPayment"]);
|
|
33
|
+
|
|
34
|
+
_templateObject8 = function _templateObject8() {
|
|
35
|
+
return data;
|
|
36
|
+
};
|
|
37
|
+
|
|
38
|
+
return data;
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
function _templateObject7() {
|
|
42
|
+
var data = _taggedTemplateLiteral(["\n LET card = FIRST(\n FOR c IN creditCards\n FILTER c._key == ", " && c.userId == ", "\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 == ", "\n LIMIT 1\n RETURN u\n )\n RETURN {user: user, card: card}"]);
|
|
43
|
+
|
|
44
|
+
_templateObject7 = function _templateObject7() {
|
|
45
|
+
return data;
|
|
46
|
+
};
|
|
47
|
+
|
|
48
|
+
return data;
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
function _templateObject6() {
|
|
52
|
+
var data = _taggedTemplateLiteral(["FOR c IN creditCards\n FILTER c.userId == ", "\n RETURN c"]);
|
|
53
|
+
|
|
54
|
+
_templateObject6 = function _templateObject6() {
|
|
55
|
+
return data;
|
|
56
|
+
};
|
|
57
|
+
|
|
58
|
+
return data;
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
function _templateObject5() {
|
|
62
|
+
var data = _taggedTemplateLiteral(["\n LET updatedCard = FIRST(\n FOR c IN creditCards\n FILTER c._key == ", " && c.userId == ", "\n UPDATE c WITH ", " IN creditCards\n LIMIT 1\n RETURN NEW\n )\n LET user = FIRST(\n FOR u IN users\n FILTER u._key == ", "\n LIMIT 1\n RETURN u\n )\n RETURN {user: user, card: updatedCard}"]);
|
|
63
|
+
|
|
64
|
+
_templateObject5 = function _templateObject5() {
|
|
65
|
+
return data;
|
|
66
|
+
};
|
|
67
|
+
|
|
68
|
+
return data;
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
function _templateObject4() {
|
|
72
|
+
var data = _taggedTemplateLiteral(["INSERT ", " IN creditCards RETURN NEW"]);
|
|
73
|
+
|
|
74
|
+
_templateObject4 = function _templateObject4() {
|
|
75
|
+
return data;
|
|
76
|
+
};
|
|
77
|
+
|
|
78
|
+
return data;
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i] != null ? arguments[i] : {}; var ownKeys = Object.keys(source); if (typeof Object.getOwnPropertySymbols === 'function') { ownKeys = ownKeys.concat(Object.getOwnPropertySymbols(source).filter(function (sym) { return Object.getOwnPropertyDescriptor(source, sym).enumerable; })); } ownKeys.forEach(function (key) { _defineProperty(target, key, source[key]); }); } return target; }
|
|
82
|
+
|
|
83
|
+
function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
|
|
84
|
+
|
|
85
|
+
function _templateObject3() {
|
|
86
|
+
var data = _taggedTemplateLiteral(["UPDATE ", " WITH ", " IN users LIMIT 1 RETURN NEW"]);
|
|
87
|
+
|
|
88
|
+
_templateObject3 = function _templateObject3() {
|
|
89
|
+
return data;
|
|
90
|
+
};
|
|
91
|
+
|
|
92
|
+
return data;
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
function _templateObject2() {
|
|
96
|
+
var data = _taggedTemplateLiteral(["UPDATE ", " WITH ", " IN users LIMIT 1 RETURN NEW"]);
|
|
97
|
+
|
|
98
|
+
_templateObject2 = function _templateObject2() {
|
|
99
|
+
return data;
|
|
100
|
+
};
|
|
101
|
+
|
|
102
|
+
return data;
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
function _templateObject() {
|
|
106
|
+
var data = _taggedTemplateLiteral(["UPDATE ", " WITH ", " IN users LIMIT 1 RETURN NEW"]);
|
|
107
|
+
|
|
108
|
+
_templateObject = function _templateObject() {
|
|
109
|
+
return data;
|
|
110
|
+
};
|
|
111
|
+
|
|
112
|
+
return data;
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
function _taggedTemplateLiteral(strings, raw) { if (!raw) { raw = strings.slice(0); } return Object.freeze(Object.defineProperties(strings, { raw: { value: Object.freeze(raw) } })); }
|
|
116
|
+
|
|
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
|
+
/**
|
|
128
|
+
* Copyright (c) 2019-Present, Nitrogen Labs, Inc.
|
|
129
|
+
* Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.
|
|
130
|
+
*/
|
|
131
|
+
|
|
132
|
+
var eventCategory = 'payments';
|
|
133
|
+
export var addCustomerAccount = function addCustomerAccount(context) {
|
|
134
|
+
var action = 'addCustomerAccount';
|
|
135
|
+
var database = context.database,
|
|
136
|
+
sessionId = context.userId,
|
|
137
|
+
username = context.username; // Stripe
|
|
138
|
+
|
|
139
|
+
var stripeClient = stripe(Config.get('stripe.token'));
|
|
140
|
+
return stripeClient.customers.create({
|
|
141
|
+
metadata: {
|
|
142
|
+
userId: sessionId,
|
|
143
|
+
username: username
|
|
144
|
+
}
|
|
145
|
+
}).then(function (customer) {
|
|
146
|
+
// Create session
|
|
147
|
+
var now = Date.now();
|
|
148
|
+
var update = {
|
|
149
|
+
modified: now,
|
|
150
|
+
stripeCustomerId: customer.id
|
|
151
|
+
};
|
|
152
|
+
var aqlQry = aql(_templateObject(), sessionId, update);
|
|
153
|
+
return useDb(database).query(aqlQry).then(function (cursor) {
|
|
154
|
+
return cursor.next();
|
|
155
|
+
}).then(function () {
|
|
156
|
+
var updatedUser = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
|
|
157
|
+
return !isEmpty(updatedUser);
|
|
158
|
+
}).catch(function (error) {
|
|
159
|
+
return logError({
|
|
160
|
+
action: action,
|
|
161
|
+
category: eventCategory,
|
|
162
|
+
label: 'db_error'
|
|
163
|
+
}, error, context).then(function () {
|
|
164
|
+
return null;
|
|
165
|
+
});
|
|
166
|
+
});
|
|
167
|
+
});
|
|
168
|
+
};
|
|
169
|
+
export var addPaymentAccountBank = function addPaymentAccountBank(context, bankAccount) {
|
|
170
|
+
var action = 'addPaymentAccountBank';
|
|
171
|
+
var database = context.database,
|
|
172
|
+
sessionId = context.userId; // Params
|
|
173
|
+
|
|
174
|
+
var accountNumber = bankAccount.accountNumber,
|
|
175
|
+
fullName = bankAccount.fullName,
|
|
176
|
+
routing = bankAccount.routing;
|
|
177
|
+
var formatAccount = parseString(accountNumber, 32);
|
|
178
|
+
|
|
179
|
+
if (formatAccount === '') {
|
|
180
|
+
throw new UserError('required_account_number');
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
var formatFullName = parseVarChar(fullName, 128);
|
|
184
|
+
|
|
185
|
+
if (formatFullName === '') {
|
|
186
|
+
throw new UserError('required_full_name');
|
|
187
|
+
}
|
|
188
|
+
|
|
189
|
+
var formatRouting = parseString(routing, 32);
|
|
190
|
+
|
|
191
|
+
if (formatRouting === '') {
|
|
192
|
+
throw new UserError('required_routing_number');
|
|
193
|
+
}
|
|
194
|
+
|
|
195
|
+
return getUser(context, sessionId).then(function (user) {
|
|
196
|
+
// Stripe
|
|
197
|
+
var stripeClient = stripe(Config.get('stripe.token'));
|
|
198
|
+
return stripeClient.customers.createSource(user.stripeAccountId, {
|
|
199
|
+
source: {
|
|
200
|
+
account_holder_name: formatFullName,
|
|
201
|
+
account_holder_type: 'individual',
|
|
202
|
+
account_number: formatAccount,
|
|
203
|
+
country: 'US',
|
|
204
|
+
currency: 'USD',
|
|
205
|
+
object: 'bank_account',
|
|
206
|
+
routing_number: formatRouting
|
|
207
|
+
}
|
|
208
|
+
}).then(function (account) {
|
|
209
|
+
var update = {
|
|
210
|
+
bankAccount: account.last4,
|
|
211
|
+
bankFullName: formatFullName,
|
|
212
|
+
bankId: account.id,
|
|
213
|
+
bankRouting: account.routing_number,
|
|
214
|
+
modified: Date.now()
|
|
215
|
+
};
|
|
216
|
+
var aqlQry = aql(_templateObject2(), sessionId, update);
|
|
217
|
+
return useDb(database).query(aqlQry).then(function (cursor) {
|
|
218
|
+
return cursor.next();
|
|
219
|
+
}).then(function () {
|
|
220
|
+
var updatedUser = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
|
|
221
|
+
return updatedUser;
|
|
222
|
+
});
|
|
223
|
+
}).catch(function (error) {
|
|
224
|
+
var msg = error.message;
|
|
225
|
+
|
|
226
|
+
if (msg === 'A bank account with that routing number and account number ' + 'already exists for this customer.') {
|
|
227
|
+
return logError({
|
|
228
|
+
action: action,
|
|
229
|
+
category: eventCategory,
|
|
230
|
+
label: 'bank_account_exists'
|
|
231
|
+
}, error, context).then(function () {
|
|
232
|
+
return null;
|
|
233
|
+
});
|
|
234
|
+
}
|
|
235
|
+
|
|
236
|
+
return logError({
|
|
237
|
+
action: action,
|
|
238
|
+
category: eventCategory,
|
|
239
|
+
label: 'payment_error'
|
|
240
|
+
}, error, context).then(function () {
|
|
241
|
+
return null;
|
|
242
|
+
});
|
|
243
|
+
});
|
|
244
|
+
});
|
|
245
|
+
};
|
|
246
|
+
export var addPaymentAccountCard = function addPaymentAccountCard(context, card, requestData) {
|
|
247
|
+
var action = 'addPaymentAccountCard';
|
|
248
|
+
var database = context.database,
|
|
249
|
+
sessionId = context.userId;
|
|
250
|
+
return getUser(context, sessionId).then(function (user) {
|
|
251
|
+
// User
|
|
252
|
+
var userKey = user._key,
|
|
253
|
+
_user$country = user.country,
|
|
254
|
+
country = _user$country === void 0 ? 'US' : _user$country,
|
|
255
|
+
city = user.city,
|
|
256
|
+
_user$currency = user.currency,
|
|
257
|
+
currency = _user$currency === void 0 ? 'USD' : _user$currency,
|
|
258
|
+
dob = user.dob,
|
|
259
|
+
email = user.email,
|
|
260
|
+
first = user.first,
|
|
261
|
+
last = user.last,
|
|
262
|
+
username = user.username; // Birthday
|
|
263
|
+
|
|
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
|
|
267
|
+
|
|
268
|
+
var accountNumber = card.accountNumber,
|
|
269
|
+
acceptedTerms = card.acceptedTerms,
|
|
270
|
+
expMonth = card.expMonth,
|
|
271
|
+
expYear = card.expYear,
|
|
272
|
+
state = card.state,
|
|
273
|
+
street1 = card.street1,
|
|
274
|
+
zip = card.zip; // Acceptance
|
|
275
|
+
|
|
276
|
+
var accepted = acceptedTerms ? DateTime.local().millisecond : 0; // Stripe
|
|
277
|
+
|
|
278
|
+
var stripeClient = stripe(Config.get('stripe.token'));
|
|
279
|
+
return stripeClient.accounts.create({
|
|
280
|
+
country: country,
|
|
281
|
+
email: email,
|
|
282
|
+
external_account: {
|
|
283
|
+
currency: currency,
|
|
284
|
+
exp_month: expMonth,
|
|
285
|
+
exp_year: expYear,
|
|
286
|
+
number: accountNumber,
|
|
287
|
+
object: 'card'
|
|
288
|
+
},
|
|
289
|
+
legal_entity: {
|
|
290
|
+
address: {
|
|
291
|
+
city: city,
|
|
292
|
+
line1: street1,
|
|
293
|
+
postal_code: zip,
|
|
294
|
+
state: state
|
|
295
|
+
},
|
|
296
|
+
dob: {
|
|
297
|
+
day: dobDay,
|
|
298
|
+
month: dobMonth,
|
|
299
|
+
year: dobYear
|
|
300
|
+
},
|
|
301
|
+
first_name: first,
|
|
302
|
+
last_name: last,
|
|
303
|
+
// ssn_last_4: ssn.substr(-4),
|
|
304
|
+
// personal_id_number: ssn,
|
|
305
|
+
type: 'individual'
|
|
306
|
+
},
|
|
307
|
+
managed: true,
|
|
308
|
+
metadata: {
|
|
309
|
+
userId: userKey,
|
|
310
|
+
username: username
|
|
311
|
+
},
|
|
312
|
+
tos_acceptance: {
|
|
313
|
+
date: accepted,
|
|
314
|
+
ip: requestData.ip,
|
|
315
|
+
user_agent: requestData.userAgent.source
|
|
316
|
+
}
|
|
317
|
+
}).then(function (customer) {
|
|
318
|
+
// Create session
|
|
319
|
+
var now = Date.now();
|
|
320
|
+
var update = {
|
|
321
|
+
modified: now,
|
|
322
|
+
stripeAccountId: customer.id
|
|
323
|
+
};
|
|
324
|
+
var aqlQry = aql(_templateObject3(), userKey, update);
|
|
325
|
+
return useDb(database).query(aqlQry).then(function (cursor) {
|
|
326
|
+
return cursor.next();
|
|
327
|
+
}).then(function () {
|
|
328
|
+
var updatedUser = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
|
|
329
|
+
return !isEmpty(updatedUser);
|
|
330
|
+
}).catch(function (error) {
|
|
331
|
+
return logError({
|
|
332
|
+
action: action,
|
|
333
|
+
category: eventCategory,
|
|
334
|
+
label: 'db_error'
|
|
335
|
+
}, error, context).then(function () {
|
|
336
|
+
return null;
|
|
337
|
+
});
|
|
338
|
+
});
|
|
339
|
+
});
|
|
340
|
+
});
|
|
341
|
+
};
|
|
342
|
+
export var addCreditCard = function addCreditCard(context, card) {
|
|
343
|
+
var action = 'addCreditCard';
|
|
344
|
+
var database = context.database,
|
|
345
|
+
sessionId = context.userId;
|
|
346
|
+
var accountNumber = card.accountNumber,
|
|
347
|
+
city = card.city,
|
|
348
|
+
country = card.country,
|
|
349
|
+
cvc = card.cvc,
|
|
350
|
+
expMonth = card.expMonth,
|
|
351
|
+
expYear = card.expYear,
|
|
352
|
+
fullName = card.fullName,
|
|
353
|
+
street1 = card.street1,
|
|
354
|
+
state = card.state,
|
|
355
|
+
zip = card.zip;
|
|
356
|
+
var formatNumber = parseNum(accountNumber, 16);
|
|
357
|
+
|
|
358
|
+
if (!formatNumber) {
|
|
359
|
+
throw new UserError('required_credit_card_number');
|
|
360
|
+
}
|
|
361
|
+
|
|
362
|
+
var formatExpMonth = parseNum(expMonth, 2);
|
|
363
|
+
|
|
364
|
+
if (!formatExpMonth) {
|
|
365
|
+
throw new UserError('required_credit_card_exp_month');
|
|
366
|
+
}
|
|
367
|
+
|
|
368
|
+
var formatExpYear = parseNum(expYear, 2);
|
|
369
|
+
|
|
370
|
+
if (!formatExpYear) {
|
|
371
|
+
throw new UserError('required_credit_card_exp_year');
|
|
372
|
+
}
|
|
373
|
+
|
|
374
|
+
var formatCvc = parseNum(cvc, 3);
|
|
375
|
+
|
|
376
|
+
if (!formatCvc) {
|
|
377
|
+
throw new UserError('required_credit_card_cvc');
|
|
378
|
+
}
|
|
379
|
+
|
|
380
|
+
var paymentCard = {};
|
|
381
|
+
var formatCity = parseVarChar(city, 32);
|
|
382
|
+
|
|
383
|
+
if (formatCity) {
|
|
384
|
+
paymentCard.city = formatCity;
|
|
385
|
+
}
|
|
386
|
+
|
|
387
|
+
var formatCountry = parseVarChar(country, 32);
|
|
388
|
+
|
|
389
|
+
if (formatCountry) {
|
|
390
|
+
paymentCard.country = formatCountry;
|
|
391
|
+
}
|
|
392
|
+
|
|
393
|
+
var formatFullName = parseVarChar(fullName, 32);
|
|
394
|
+
|
|
395
|
+
if (formatFullName) {
|
|
396
|
+
paymentCard.fullName = formatFullName;
|
|
397
|
+
}
|
|
398
|
+
|
|
399
|
+
var formatStreet = parseVarChar(street1, 32);
|
|
400
|
+
|
|
401
|
+
if (formatStreet) {
|
|
402
|
+
paymentCard.street1 = formatStreet;
|
|
403
|
+
}
|
|
404
|
+
|
|
405
|
+
var formatState = parseVarChar(state, 32);
|
|
406
|
+
|
|
407
|
+
if (formatState) {
|
|
408
|
+
paymentCard.state = formatState;
|
|
409
|
+
}
|
|
410
|
+
|
|
411
|
+
var formatZip = parseVarChar(zip, 32);
|
|
412
|
+
|
|
413
|
+
if (formatZip) {
|
|
414
|
+
paymentCard.zip = formatZip;
|
|
415
|
+
} // Get stripe id
|
|
416
|
+
|
|
417
|
+
|
|
418
|
+
return getUser(context, sessionId).then(function () {
|
|
419
|
+
var user = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
|
|
420
|
+
// Stripe
|
|
421
|
+
var stripeClient = stripe(Config.get('stripe.token'));
|
|
422
|
+
return stripeClient.customers.createSource(user.stripeCustomerId, {
|
|
423
|
+
source: {
|
|
424
|
+
address_city: formatCity,
|
|
425
|
+
address_country: formatCountry,
|
|
426
|
+
address_line1: formatStreet,
|
|
427
|
+
address_state: formatState,
|
|
428
|
+
address_zip: formatZip,
|
|
429
|
+
currency: 'usd',
|
|
430
|
+
cvc: formatCvc,
|
|
431
|
+
exp_month: formatExpMonth,
|
|
432
|
+
exp_year: formatExpYear,
|
|
433
|
+
name: formatFullName,
|
|
434
|
+
number: formatNumber,
|
|
435
|
+
object: 'card'
|
|
436
|
+
}
|
|
437
|
+
}).then(function (stripeObj) {
|
|
438
|
+
var now = Date.now();
|
|
439
|
+
|
|
440
|
+
var insert = _objectSpread({}, paymentCard, {
|
|
441
|
+
_key: createHash("user-payment-".concat(sessionId)),
|
|
442
|
+
accountNumber: stripeObj.last4,
|
|
443
|
+
added: now,
|
|
444
|
+
brand: stripeObj.brand,
|
|
445
|
+
cvc: stripeObj.cvc_check,
|
|
446
|
+
expMonth: expMonth,
|
|
447
|
+
expYear: expYear,
|
|
448
|
+
modified: now,
|
|
449
|
+
userId: sessionId
|
|
450
|
+
});
|
|
451
|
+
|
|
452
|
+
var insertAqlQry = aql(_templateObject4(), insert);
|
|
453
|
+
return useDb(database).query(insertAqlQry).then(function (cursor) {
|
|
454
|
+
return cursor.next();
|
|
455
|
+
}).then(function () {
|
|
456
|
+
var newCard = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
|
|
457
|
+
|
|
458
|
+
if (!isEmpty(newCard)) {
|
|
459
|
+
// Add linked edge
|
|
460
|
+
var cardId = card._id,
|
|
461
|
+
cardKey = card._key;
|
|
462
|
+
var edgeCollection = useDb(database).edgeCollection('hasPayment');
|
|
463
|
+
var edgeId = createHash("payment-".concat(cardKey));
|
|
464
|
+
var edge = {
|
|
465
|
+
_key: edgeId
|
|
466
|
+
};
|
|
467
|
+
return edgeCollection.save(edge, "users/".concat(sessionId), cardId).then(function () {
|
|
468
|
+
return card;
|
|
469
|
+
});
|
|
470
|
+
}
|
|
471
|
+
|
|
472
|
+
return newCard;
|
|
473
|
+
});
|
|
474
|
+
}).catch(function (error) {
|
|
475
|
+
return logError({
|
|
476
|
+
action: action,
|
|
477
|
+
category: eventCategory,
|
|
478
|
+
label: 'payment_error'
|
|
479
|
+
}, error, context).then(function () {
|
|
480
|
+
return null;
|
|
481
|
+
});
|
|
482
|
+
});
|
|
483
|
+
});
|
|
484
|
+
};
|
|
485
|
+
export var updateCreditCard = function updateCreditCard(context, card) {
|
|
486
|
+
var database = context.database,
|
|
487
|
+
sessionId = context.userId;
|
|
488
|
+
var city = card.city,
|
|
489
|
+
country = card.country,
|
|
490
|
+
expMonth = card.expMonth,
|
|
491
|
+
expYear = card.expYear,
|
|
492
|
+
fullName = card.fullName,
|
|
493
|
+
id = card.id,
|
|
494
|
+
street1 = card.street1,
|
|
495
|
+
state = card.state,
|
|
496
|
+
zip = card.zip;
|
|
497
|
+
var formatId = parseId(id);
|
|
498
|
+
|
|
499
|
+
if (formatId) {
|
|
500
|
+
throw new UserError('required_credit_card_id');
|
|
501
|
+
}
|
|
502
|
+
|
|
503
|
+
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);
|
|
512
|
+
|
|
513
|
+
if (formatExpMonth) {
|
|
514
|
+
paymentCard.expMonth = formatExpMonth;
|
|
515
|
+
}
|
|
516
|
+
|
|
517
|
+
if (formatExpYear) {
|
|
518
|
+
paymentCard.expYear = formatExpYear;
|
|
519
|
+
}
|
|
520
|
+
|
|
521
|
+
if (formatCity) {
|
|
522
|
+
paymentCard.city = formatCity;
|
|
523
|
+
}
|
|
524
|
+
|
|
525
|
+
if (formatCountry) {
|
|
526
|
+
paymentCard.country = formatCountry;
|
|
527
|
+
}
|
|
528
|
+
|
|
529
|
+
if (formatFullName) {
|
|
530
|
+
paymentCard.fullName = formatFullName;
|
|
531
|
+
}
|
|
532
|
+
|
|
533
|
+
if (formatStreet1) {
|
|
534
|
+
paymentCard.street1 = formatStreet1;
|
|
535
|
+
}
|
|
536
|
+
|
|
537
|
+
if (formatState) {
|
|
538
|
+
paymentCard.state = formatState;
|
|
539
|
+
}
|
|
540
|
+
|
|
541
|
+
if (formatZip) {
|
|
542
|
+
paymentCard.zip = formatZip;
|
|
543
|
+
}
|
|
544
|
+
|
|
545
|
+
var update = paymentCard;
|
|
546
|
+
var aqlQry = aql(_templateObject5(), formatId, sessionId, update, sessionId);
|
|
547
|
+
return useDb(database).query(aqlQry).then(function (cursor) {
|
|
548
|
+
return cursor.next();
|
|
549
|
+
}).then(function () {
|
|
550
|
+
var results = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {
|
|
551
|
+
card: {},
|
|
552
|
+
user: {}
|
|
553
|
+
};
|
|
554
|
+
var updatedCard = results.card;
|
|
555
|
+
var user = results.user;
|
|
556
|
+
|
|
557
|
+
if (isEmpty(updatedCard)) {
|
|
558
|
+
throw new UserError('not_found');
|
|
559
|
+
} // Stripe
|
|
560
|
+
|
|
561
|
+
|
|
562
|
+
var stripeClient = stripe(Config.get('stripe.token'));
|
|
563
|
+
return stripeClient.customers.updateCard(user.stripeCustomerId, card.stripeId, {
|
|
564
|
+
address_city: formatCity,
|
|
565
|
+
address_country: formatCountry,
|
|
566
|
+
address_line1: formatStreet1,
|
|
567
|
+
address_state: formatState,
|
|
568
|
+
address_zip: formatZip,
|
|
569
|
+
exp_month: formatExpMonth,
|
|
570
|
+
exp_year: formatExpYear,
|
|
571
|
+
name: formatFullName
|
|
572
|
+
}).then(function () {
|
|
573
|
+
return card;
|
|
574
|
+
}).catch(function (error) {
|
|
575
|
+
console.log('payments::updateCard::error', error);
|
|
576
|
+
throw new UserError('payment_error');
|
|
577
|
+
});
|
|
578
|
+
});
|
|
579
|
+
};
|
|
580
|
+
export var getCreditCards = function getCreditCards(context) {
|
|
581
|
+
var action = 'getCreditCards';
|
|
582
|
+
var database = context.database,
|
|
583
|
+
sessionId = context.userId;
|
|
584
|
+
var aqlQry = aql(_templateObject6(), sessionId);
|
|
585
|
+
return useDb(database).query(aqlQry).then(function (cursor) {
|
|
586
|
+
return cursor.all();
|
|
587
|
+
}).then(function () {
|
|
588
|
+
var list = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : [];
|
|
589
|
+
return list;
|
|
590
|
+
}).catch(function (error) {
|
|
591
|
+
return logError({
|
|
592
|
+
action: action,
|
|
593
|
+
category: eventCategory,
|
|
594
|
+
label: 'db_error'
|
|
595
|
+
}, error, context).then(function () {
|
|
596
|
+
return null;
|
|
597
|
+
});
|
|
598
|
+
});
|
|
599
|
+
};
|
|
600
|
+
export var deleteCreditCard = function deleteCreditCard(context, cardId) {
|
|
601
|
+
var database = context.database,
|
|
602
|
+
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) {
|
|
606
|
+
return cursor.next();
|
|
607
|
+
}).then(function () {
|
|
608
|
+
var result = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {
|
|
609
|
+
card: {},
|
|
610
|
+
user: {}
|
|
611
|
+
};
|
|
612
|
+
|
|
613
|
+
if (isEmpty(result)) {
|
|
614
|
+
return false;
|
|
615
|
+
}
|
|
616
|
+
|
|
617
|
+
var card = result.card,
|
|
618
|
+
user = result.user;
|
|
619
|
+
var cardKey = card._key; // Remove linked edges
|
|
620
|
+
|
|
621
|
+
var edgeCollection = useDb(database).edgeCollection('hasPayment');
|
|
622
|
+
return edgeCollection.outEdges(cardKey).then(function (edges) {
|
|
623
|
+
if (edges.length) {
|
|
624
|
+
return Promise.all(edges.map(function (edge) {
|
|
625
|
+
var edgeKey = edge._key;
|
|
626
|
+
var removeAqlQry = aql(_templateObject8(), edgeKey);
|
|
627
|
+
return useDb(database).query(removeAqlQry);
|
|
628
|
+
})).then(function () {
|
|
629
|
+
// Stripe
|
|
630
|
+
var stripeClient = stripe(Config.get('stripe.token'));
|
|
631
|
+
return stripeClient.customers.deleteCard(user.stripeCustomerId, card.stripeId).then(function () {
|
|
632
|
+
return true;
|
|
633
|
+
}).catch(function (error) {
|
|
634
|
+
console.log('payments::deleteCard::error', error);
|
|
635
|
+
throw new UserError('payment_error');
|
|
636
|
+
});
|
|
637
|
+
});
|
|
638
|
+
}
|
|
639
|
+
|
|
640
|
+
return false;
|
|
641
|
+
});
|
|
642
|
+
});
|
|
643
|
+
};
|
|
644
|
+
export var deletePaymentAccountBank = function deletePaymentAccountBank(context, bankId) {
|
|
645
|
+
var database = context.database,
|
|
646
|
+
sessionId = context.userId; // Clean db
|
|
647
|
+
|
|
648
|
+
var update = {
|
|
649
|
+
bankAccount: '',
|
|
650
|
+
bankFullName: '',
|
|
651
|
+
bankId: '',
|
|
652
|
+
bankRouting: '',
|
|
653
|
+
modified: Date.now()
|
|
654
|
+
};
|
|
655
|
+
var aqlQry = aql(_templateObject9(), sessionId, update);
|
|
656
|
+
return useDb(database).query(aqlQry).then(function (cursor) {
|
|
657
|
+
return cursor.next();
|
|
658
|
+
}).then(function () {
|
|
659
|
+
var user = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
|
|
660
|
+
// Stripe
|
|
661
|
+
var stripeClient = stripe(Config.get('stripe.token'));
|
|
662
|
+
return stripeClient.customers.deleteSource(user.stripeAccountId, bankId).then(function () {
|
|
663
|
+
var response = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {
|
|
664
|
+
deleted: false
|
|
665
|
+
};
|
|
666
|
+
return response.deleted;
|
|
667
|
+
}).catch(function () {
|
|
668
|
+
return Promise.resolve(false);
|
|
669
|
+
});
|
|
670
|
+
});
|
|
671
|
+
};
|
|
672
|
+
export var createPaymentTransfer = function createPaymentTransfer(context, transfer) {
|
|
673
|
+
var database = context.database,
|
|
674
|
+
sessionId = context.userId;
|
|
675
|
+
var amount = transfer.amount,
|
|
676
|
+
currency = transfer.currency;
|
|
677
|
+
var formatAmount = parseNum(amount);
|
|
678
|
+
var formatCurrency = parseChar(currency, 3, 'USD').toUpperCase();
|
|
679
|
+
return getUser(context, sessionId).then(function (user) {
|
|
680
|
+
// Stripe
|
|
681
|
+
var stripeClient = stripe(Config.get('stripe.token'));
|
|
682
|
+
return stripeClient.transfers.create({
|
|
683
|
+
amount: formatAmount,
|
|
684
|
+
currency: formatCurrency,
|
|
685
|
+
destination: user.stripeAccountId
|
|
686
|
+
}).then(function (stripeTransfer) {
|
|
687
|
+
console.log(stripeTransfer);
|
|
688
|
+
var now = Date.now();
|
|
689
|
+
var insert = {
|
|
690
|
+
added: now,
|
|
691
|
+
amount: formatAmount,
|
|
692
|
+
currency: formatCurrency,
|
|
693
|
+
modified: now,
|
|
694
|
+
userId: sessionId
|
|
695
|
+
};
|
|
696
|
+
var aqlQry = aql(_templateObject10(), insert);
|
|
697
|
+
return useDb(database).query(aqlQry).then(function (cursor) {
|
|
698
|
+
return cursor.next();
|
|
699
|
+
}).then(function (newTransfer) {
|
|
700
|
+
return newTransfer;
|
|
701
|
+
});
|
|
702
|
+
});
|
|
703
|
+
});
|
|
704
|
+
};
|
|
705
|
+
export var createPaymentHold = function createPaymentHold(context, payment) {
|
|
706
|
+
var database = context.database,
|
|
707
|
+
sessionId = context.userId;
|
|
708
|
+
var amount = payment.amount,
|
|
709
|
+
capture = payment.capture,
|
|
710
|
+
cardId = payment.cardId,
|
|
711
|
+
currency = payment.currency,
|
|
712
|
+
description = payment.description;
|
|
713
|
+
var formatCurrency = parseChar(currency, 3, 'USD').toUpperCase();
|
|
714
|
+
var stripeClient = stripe(Config.get('stripe.token'));
|
|
715
|
+
return stripeClient.charges.create({
|
|
716
|
+
amount: amount,
|
|
717
|
+
capture: capture,
|
|
718
|
+
currency: formatCurrency,
|
|
719
|
+
description: description,
|
|
720
|
+
source: cardId
|
|
721
|
+
}).then(function (stripeCharge) {
|
|
722
|
+
var now = Date.now();
|
|
723
|
+
var insert = {
|
|
724
|
+
added: now,
|
|
725
|
+
amount: amount,
|
|
726
|
+
capture: capture,
|
|
727
|
+
cardId: cardId,
|
|
728
|
+
chargeFailCode: stripeCharge.failure_code,
|
|
729
|
+
chargeFailMsg: stripeCharge.failure_message,
|
|
730
|
+
chargeId: stripeCharge.id,
|
|
731
|
+
chargeStatus: stripeCharge.status,
|
|
732
|
+
currency: formatCurrency,
|
|
733
|
+
description: description,
|
|
734
|
+
modified: now,
|
|
735
|
+
userId: sessionId
|
|
736
|
+
};
|
|
737
|
+
var aqlQry = aql(_templateObject11(), insert);
|
|
738
|
+
return useDb(database).query(aqlQry).then(function (cursor) {
|
|
739
|
+
return cursor.next();
|
|
740
|
+
}).then(function (newPayment) {
|
|
741
|
+
return newPayment;
|
|
742
|
+
});
|
|
743
|
+
}).catch(function (error) {
|
|
744
|
+
console.log('payments::createHold::error', error);
|
|
745
|
+
throw new UserError('payment_error');
|
|
746
|
+
});
|
|
747
|
+
};
|
|
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"]}
|