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