@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,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9kYXRhL3BheW1lbnRzLnRzIl0sIm5hbWVzIjpbImNyZWF0ZUhhc2giLCJwYXJzZUNoYXIiLCJwYXJzZUlkIiwicGFyc2VOdW0iLCJwYXJzZVN0cmluZyIsInBhcnNlVmFyQ2hhciIsImFxbCIsIlVzZXJFcnJvciIsImlzRW1wdHkiLCJEYXRlVGltZSIsInN0cmlwZSIsIkNvbmZpZyIsImxvZ0Vycm9yIiwidXNlRGIiLCJnZXRVc2VyIiwiZXZlbnRDYXRlZ29yeSIsImFkZEN1c3RvbWVyQWNjb3VudCIsImNvbnRleHQiLCJhY3Rpb24iLCJkYXRhYmFzZSIsInNlc3Npb25JZCIsInVzZXJJZCIsInVzZXJuYW1lIiwic3RyaXBlQ2xpZW50IiwiZ2V0IiwiY3VzdG9tZXJzIiwiY3JlYXRlIiwibWV0YWRhdGEiLCJ0aGVuIiwiY3VzdG9tZXIiLCJub3ciLCJEYXRlIiwidXBkYXRlIiwibW9kaWZpZWQiLCJzdHJpcGVDdXN0b21lcklkIiwiaWQiLCJhcWxRcnkiLCJxdWVyeSIsImN1cnNvciIsIm5leHQiLCJ1cGRhdGVkVXNlciIsImNhdGNoIiwiZXJyb3IiLCJjYXRlZ29yeSIsImxhYmVsIiwiYWRkUGF5bWVudEFjY291bnRCYW5rIiwiYmFua0FjY291bnQiLCJhY2NvdW50TnVtYmVyIiwiZnVsbE5hbWUiLCJyb3V0aW5nIiwiZm9ybWF0QWNjb3VudCIsImZvcm1hdEZ1bGxOYW1lIiwiZm9ybWF0Um91dGluZyIsInVzZXIiLCJjcmVhdGVTb3VyY2UiLCJzdHJpcGVBY2NvdW50SWQiLCJzb3VyY2UiLCJhY2NvdW50X2hvbGRlcl9uYW1lIiwiYWNjb3VudF9ob2xkZXJfdHlwZSIsImFjY291bnRfbnVtYmVyIiwiY291bnRyeSIsImN1cnJlbmN5Iiwib2JqZWN0Iiwicm91dGluZ19udW1iZXIiLCJhY2NvdW50IiwibGFzdDQiLCJiYW5rRnVsbE5hbWUiLCJiYW5rSWQiLCJiYW5rUm91dGluZyIsIm1zZyIsIm1lc3NhZ2UiLCJhZGRQYXltZW50QWNjb3VudENhcmQiLCJjYXJkIiwicmVxdWVzdERhdGEiLCJ1c2VyS2V5IiwiX2tleSIsImNpdHkiLCJkb2IiLCJlbWFpbCIsImZpcnN0IiwibGFzdCIsImRvYkRheSIsImZyb21NaWxsaXMiLCJ0b0Zvcm1hdCIsImRvYk1vbnRoIiwiZG9iWWVhciIsImFjY2VwdGVkVGVybXMiLCJleHBNb250aCIsImV4cFllYXIiLCJzdGF0ZSIsInN0cmVldDEiLCJ6aXAiLCJhY2NlcHRlZCIsImxvY2FsIiwibWlsbGlzZWNvbmQiLCJhY2NvdW50cyIsImV4dGVybmFsX2FjY291bnQiLCJleHBfbW9udGgiLCJleHBfeWVhciIsIm51bWJlciIsImxlZ2FsX2VudGl0eSIsImFkZHJlc3MiLCJsaW5lMSIsInBvc3RhbF9jb2RlIiwiZGF5IiwibW9udGgiLCJ5ZWFyIiwiZmlyc3RfbmFtZSIsImxhc3RfbmFtZSIsInR5cGUiLCJtYW5hZ2VkIiwidG9zX2FjY2VwdGFuY2UiLCJkYXRlIiwiaXAiLCJ1c2VyX2FnZW50IiwidXNlckFnZW50IiwiYWRkQ3JlZGl0Q2FyZCIsImN2YyIsImZvcm1hdE51bWJlciIsImZvcm1hdEV4cE1vbnRoIiwiZm9ybWF0RXhwWWVhciIsImZvcm1hdEN2YyIsInBheW1lbnRDYXJkIiwiZm9ybWF0Q2l0eSIsImZvcm1hdENvdW50cnkiLCJmb3JtYXRTdHJlZXQiLCJmb3JtYXRTdGF0ZSIsImZvcm1hdFppcCIsImFkZHJlc3NfY2l0eSIsImFkZHJlc3NfY291bnRyeSIsImFkZHJlc3NfbGluZTEiLCJhZGRyZXNzX3N0YXRlIiwiYWRkcmVzc196aXAiLCJuYW1lIiwic3RyaXBlT2JqIiwiaW5zZXJ0IiwiYWRkZWQiLCJicmFuZCIsImN2Y19jaGVjayIsImluc2VydEFxbFFyeSIsIm5ld0NhcmQiLCJjYXJkSWQiLCJfaWQiLCJjYXJkS2V5IiwiZWRnZUNvbGxlY3Rpb24iLCJlZGdlSWQiLCJlZGdlIiwic2F2ZSIsInVwZGF0ZUNyZWRpdENhcmQiLCJmb3JtYXRJZCIsImZvcm1hdFN0cmVldDEiLCJyZXN1bHRzIiwidXBkYXRlZENhcmQiLCJ1cGRhdGVDYXJkIiwic3RyaXBlSWQiLCJjb25zb2xlIiwibG9nIiwiZ2V0Q3JlZGl0Q2FyZHMiLCJhbGwiLCJsaXN0IiwiZGVsZXRlQ3JlZGl0Q2FyZCIsImZvcm1hdENhcmRJZCIsInJlc3VsdCIsIm91dEVkZ2VzIiwiZWRnZXMiLCJsZW5ndGgiLCJQcm9taXNlIiwibWFwIiwiZWRnZUtleSIsInJlbW92ZUFxbFFyeSIsImRlbGV0ZUNhcmQiLCJkZWxldGVQYXltZW50QWNjb3VudEJhbmsiLCJkZWxldGVTb3VyY2UiLCJyZXNwb25zZSIsImRlbGV0ZWQiLCJyZXNvbHZlIiwiY3JlYXRlUGF5bWVudFRyYW5zZmVyIiwidHJhbnNmZXIiLCJhbW91bnQiLCJmb3JtYXRBbW91bnQiLCJmb3JtYXRDdXJyZW5jeSIsInRvVXBwZXJDYXNlIiwidHJhbnNmZXJzIiwiZGVzdGluYXRpb24iLCJzdHJpcGVUcmFuc2ZlciIsIm5ld1RyYW5zZmVyIiwiY3JlYXRlUGF5bWVudEhvbGQiLCJwYXltZW50IiwiY2FwdHVyZSIsImRlc2NyaXB0aW9uIiwiY2hhcmdlcyIsInN0cmlwZUNoYXJnZSIsImNoYXJnZUZhaWxDb2RlIiwiZmFpbHVyZV9jb2RlIiwiY2hhcmdlRmFpbE1zZyIsImZhaWx1cmVfbWVzc2FnZSIsImNoYXJnZUlkIiwiY2hhcmdlU3RhdHVzIiwic3RhdHVzIiwibmV3UGF5bWVudCJdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQSxTQUFRQSxVQUFSLEVBQW9CQyxTQUFwQixFQUErQkMsT0FBL0IsRUFBd0NDLFFBQXhDLEVBQWtEQyxXQUFsRCxFQUErREMsWUFBL0QsUUFBa0YsY0FBbEY7QUFDQSxTQUFRQyxHQUFSLFFBQWtCLFVBQWxCO0FBR0EsU0FBUUMsU0FBUixRQUF3Qixn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O0FBY0EsUUFBTWdCLE1BQWdCLEdBQUc5QixHQUFILHNCQUFnQmlJLE1BQWhCLENBQXRCO0FBRUEsV0FBTzFILEtBQUssQ0FBQ00sUUFBRCxDQUFMLENBQWdCa0IsS0FBaEIsQ0FBc0JELE1BQXRCLEVBQ0pSLElBREksQ0FDQyxVQUFDVSxNQUFEO0FBQUEsYUFBeUJBLE1BQU0sQ0FBQ0MsSUFBUCxFQUF6QjtBQUFBLEtBREQsRUFFSlgsSUFGSSxDQUVDLFVBQUMySyxVQUFEO0FBQUEsYUFBK0JBLFVBQS9CO0FBQUEsS0FGRCxDQUFQO0FBR0QsR0E3QkksRUE4Qko5SixLQTlCSSxDQThCRSxVQUFDQyxLQUFELEVBQWtCO0FBQ3ZCaUgsSUFBQUEsT0FBTyxDQUFDQyxHQUFSLENBQVksNkJBQVosRUFBMkNsSCxLQUEzQztBQUNBLFVBQU0sSUFBSW5DLFNBQUosQ0FBYyxlQUFkLENBQU47QUFDRCxHQWpDSSxDQUFQO0FBa0NELENBekNNIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtjcmVhdGVIYXNoLCBwYXJzZUNoYXIsIHBhcnNlSWQsIHBhcnNlTnVtLCBwYXJzZVN0cmluZywgcGFyc2VWYXJDaGFyfSBmcm9tICdAbmxhYnMvdXRpbHMnO1xuaW1wb3J0IHthcWx9IGZyb20gJ2FyYW5nb2pzJztcbmltcG9ydCB7QXFsUXVlcnl9IGZyb20gJ2FyYW5nb2pzL2xpYi9janMvYXFsLXF1ZXJ5JztcbmltcG9ydCB7QXJyYXlDdXJzb3J9IGZyb20gJ2FyYW5nb2pzL2xpYi9janMvY3Vyc29yJztcbmltcG9ydCB7VXNlckVycm9yfSBmcm9tICdncmFwaHFsLWVycm9ycyc7XG5pbXBvcnQgaXNFbXB0eSBmcm9tICdsb2Rhc2gvaXNFbXB0eSc7XG5pbXBvcnQge0RhdGVUaW1lfSBmcm9tICdsdXhvbic7XG5pbXBvcnQgKiBhcyBzdHJpcGUgZnJvbSAnc3RyaXBlJztcblxuaW1wb3J0IHtDb25maWd9IGZyb20gJy4uL2NvbmZpZyc7XG5pbXBvcnQge0FwaUNvbnRleHQsIFBheW1lbnRCYW5rQWNjb3VudCwgUGF5bWVudENhcmRUeXBlLCBQYXltZW50Q2hhcmdlLCBQYXltZW50VHJhbnNmZXIsIFVzZXJUeXBlfSBmcm9tICcuLi90eXBlcyc7XG5pbXBvcnQge2xvZ0Vycm9yfSBmcm9tICcuLi91dGlscy9hbmFseXRpY3MnO1xuaW1wb3J0IHt1c2VEYn0gZnJvbSAnLi4vdXRpbHMvYXJhbmdvZGInO1xuaW1wb3J0IHtnZXRVc2VyfSBmcm9tICcuL3VzZXJzJztcblxuLyoqXG4gKiBDb3B5cmlnaHQgKGMpIDIwMTktUHJlc2VudCwgTml0cm9nZW4gTGFicywgSW5jLlxuICogQ29weXJpZ2h0cyBsaWNlbnNlZCB1bmRlciB0aGUgTUlUIExpY2Vuc2UuIFNlZSB0aGUgYWNjb21wYW55aW5nIExJQ0VOU0UgZmlsZSBmb3IgdGVybXMuXG4gKi9cbmNvbnN0IGV2ZW50Q2F0ZWdvcnk6IHN0cmluZyA9ICdwYXltZW50cyc7XG5cbmV4cG9ydCBjb25zdCBhZGRDdXN0b21lckFjY291bnQgPSAoY29udGV4dDogQXBpQ29udGV4dCk6IFByb21pc2U8Ym9vbGVhbj4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdhZGRDdXN0b21lckFjY291bnQnO1xuICBjb25zdCB7ZGF0YWJhc2UsIHVzZXJJZDogc2Vzc2lvbklkLCB1c2VybmFtZX0gPSBjb250ZXh0O1xuXG4gIC8vIFN0cmlwZVxuICBjb25zdCBzdHJpcGVDbGllbnQgPSBzdHJpcGUoQ29uZmlnLmdldCgnc3RyaXBlLnRva2VuJykpO1xuXG4gIHJldHVybiBzdHJpcGVDbGllbnQuY3VzdG9tZXJzXG4gICAgLmNyZWF0ZSh7XG4gICAgICBtZXRhZGF0YToge1xuICAgICAgICB1c2VySWQ6IHNlc3Npb25JZCxcbiAgICAgICAgdXNlcm5hbWVcbiAgICAgIH1cbiAgICB9KVxuICAgIC50aGVuKChjdXN0b21lcikgPT4ge1xuICAgICAgLy8gQ3JlYXRlIHNlc3Npb25cbiAgICAgIGNvbnN0IG5vdzogbnVtYmVyID0gRGF0ZS5ub3coKTtcbiAgICAgIGNvbnN0IHVwZGF0ZTogVXNlclR5cGUgPSB7XG4gICAgICAgIG1vZGlmaWVkOiBub3csXG4gICAgICAgIHN0cmlwZUN1c3RvbWVySWQ6IGN1c3RvbWVyLmlkXG4gICAgICB9O1xuXG4gICAgICBjb25zdCBhcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYFVQREFURSAke3Nlc3Npb25JZH0gV0lUSCAke3VwZGF0ZX0gSU4gdXNlcnMgTElNSVQgMSBSRVRVUk4gTkVXYDtcblxuICAgICAgcmV0dXJuIHVzZURiKGRhdGFiYXNlKS5xdWVyeShhcWxRcnkpXG4gICAgICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgICAgICAudGhlbigodXBkYXRlZFVzZXI6IFVzZXJUeXBlID0ge30pID0+ICFpc0VtcHR5KHVwZGF0ZWRVc2VyKSlcbiAgICAgICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgICAgICBhY3Rpb24sXG4gICAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgICAgbGFiZWw6ICdkYl9lcnJvcidcbiAgICAgICAgfSwgZXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCkpO1xuICAgIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IGFkZFBheW1lbnRBY2NvdW50QmFuayA9IChjb250ZXh0OiBBcGlDb250ZXh0LCBiYW5rQWNjb3VudDogUGF5bWVudEJhbmtBY2NvdW50KTogUHJvbWlzZTxib29sZWFuPiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2FkZFBheW1lbnRBY2NvdW50QmFuayc7XG4gIGNvbnN0IHtkYXRhYmFzZSwgdXNlcklkOiBzZXNzaW9uSWR9ID0gY29udGV4dDtcblxuICAvLyBQYXJhbXNcbiAgY29uc3Qge1xuICAgIGFjY291bnROdW1iZXIsXG4gICAgZnVsbE5hbWUsXG4gICAgcm91dGluZ1xuICB9ID0gYmFua0FjY291bnQ7XG5cbiAgY29uc3QgZm9ybWF0QWNjb3VudDogc3RyaW5nID0gcGFyc2VTdHJpbmcoYWNjb3VudE51bWJlciwgMzIpO1xuXG4gIGlmKGZvcm1hdEFjY291bnQgPT09ICcnKSB7XG4gICAgdGhyb3cgbmV3IFVzZXJFcnJvcigncmVxdWlyZWRfYWNjb3VudF9udW1iZXInKTtcbiAgfVxuXG4gIGNvbnN0IGZvcm1hdEZ1bGxOYW1lOiBzdHJpbmcgPSBwYXJzZVZhckNoYXIoZnVsbE5hbWUsIDEyOCk7XG5cbiAgaWYoZm9ybWF0RnVsbE5hbWUgPT09ICcnKSB7XG4gICAgdGhyb3cgbmV3IFVzZXJFcnJvcigncmVxdWlyZWRfZnVsbF9uYW1lJyk7XG4gIH1cblxuICBjb25zdCBmb3JtYXRSb3V0aW5nOiBzdHJpbmcgPSBwYXJzZVN0cmluZyhyb3V0aW5nLCAzMik7XG5cbiAgaWYoZm9ybWF0Um91dGluZyA9PT0gJycpIHtcbiAgICB0aHJvdyBuZXcgVXNlckVycm9yKCdyZXF1aXJlZF9yb3V0aW5nX251bWJlcicpO1xuICB9XG5cbiAgcmV0dXJuIGdldFVzZXIoY29udGV4dCwgc2Vzc2lvbklkKVxuICAgIC50aGVuKCh1c2VyOiBVc2VyVHlwZSkgPT4ge1xuICAgICAgLy8gU3RyaXBlXG4gICAgICBjb25zdCBzdHJpcGVDbGllbnQgPSBzdHJpcGUoQ29uZmlnLmdldCgnc3RyaXBlLnRva2VuJykpO1xuXG4gICAgICByZXR1cm4gc3RyaXBlQ2xpZW50LmN1c3RvbWVyc1xuICAgICAgICAuY3JlYXRlU291cmNlKHVzZXIuc3RyaXBlQWNjb3VudElkLCB7XG4gICAgICAgICAgc291cmNlOiB7XG4gICAgICAgICAgICBhY2NvdW50X2hvbGRlcl9uYW1lOiBmb3JtYXRGdWxsTmFtZSxcbiAgICAgICAgICAgIGFjY291bnRfaG9sZGVyX3R5cGU6ICdpbmRpdmlkdWFsJyxcbiAgICAgICAgICAgIGFjY291bnRfbnVtYmVyOiBmb3JtYXRBY2NvdW50LFxuICAgICAgICAgICAgY291bnRyeTogJ1VTJyxcbiAgICAgICAgICAgIGN1cnJlbmN5OiAnVVNEJyxcbiAgICAgICAgICAgIG9iamVjdDogJ2JhbmtfYWNjb3VudCcsXG4gICAgICAgICAgICByb3V0aW5nX251bWJlcjogZm9ybWF0Um91dGluZ1xuICAgICAgICAgIH1cbiAgICAgICAgfSlcbiAgICAgICAgLnRoZW4oKGFjY291bnQpID0+IHtcbiAgICAgICAgICBjb25zdCB1cGRhdGU6IGFueSA9IHtcbiAgICAgICAgICAgIGJhbmtBY2NvdW50OiBhY2NvdW50Lmxhc3Q0LFxuICAgICAgICAgICAgYmFua0Z1bGxOYW1lOiBmb3JtYXRGdWxsTmFtZSxcbiAgICAgICAgICAgIGJhbmtJZDogYWNjb3VudC5pZCxcbiAgICAgICAgICAgIGJhbmtSb3V0aW5nOiBhY2NvdW50LnJvdXRpbmdfbnVtYmVyLFxuICAgICAgICAgICAgbW9kaWZpZWQ6IERhdGUubm93KClcbiAgICAgICAgICB9O1xuXG4gICAgICAgICAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBVUERBVEUgJHtzZXNzaW9uSWR9IFdJVEggJHt1cGRhdGV9IElOIHVzZXJzIExJTUlUIDEgUkVUVVJOIE5FV2A7XG5cbiAgICAgICAgICByZXR1cm4gdXNlRGIoZGF0YWJhc2UpLnF1ZXJ5KGFxbFFyeSlcbiAgICAgICAgICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgICAgICAgICAgLnRoZW4oKHVwZGF0ZWRVc2VyOiBVc2VyVHlwZSA9IHt9KSA9PiB1cGRhdGVkVXNlcik7XG4gICAgICAgIH0pXG4gICAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiB7XG4gICAgICAgICAgY29uc3QgbXNnID0gZXJyb3IubWVzc2FnZTtcblxuICAgICAgICAgIGlmKG1zZyA9PT0gJ0EgYmFuayBhY2NvdW50IHdpdGggdGhhdCByb3V0aW5nIG51bWJlciBhbmQgYWNjb3VudCBudW1iZXIgJyArXG4gICAgICAgICAgICAnYWxyZWFkeSBleGlzdHMgZm9yIHRoaXMgY3VzdG9tZXIuJykge1xuICAgICAgICAgICAgcmV0dXJuIGxvZ0Vycm9yKHtcbiAgICAgICAgICAgICAgYWN0aW9uLFxuICAgICAgICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgICAgICAgbGFiZWw6ICdiYW5rX2FjY291bnRfZXhpc3RzJ1xuICAgICAgICAgICAgfSwgZXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCk7XG4gICAgICAgICAgfVxuICAgICAgICAgIHJldHVybiBsb2dFcnJvcih7XG4gICAgICAgICAgICBhY3Rpb24sXG4gICAgICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgICAgIGxhYmVsOiAncGF5bWVudF9lcnJvcidcbiAgICAgICAgICB9LCBlcnJvciwgY29udGV4dCkudGhlbigoKSA9PiBudWxsKTtcbiAgICAgICAgfSk7XG4gICAgfSk7XG59O1xuXG5leHBvcnQgY29uc3QgYWRkUGF5bWVudEFjY291bnRDYXJkID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGNhcmQ6IFBheW1lbnRDYXJkVHlwZSwgcmVxdWVzdERhdGEpOiBQcm9taXNlPFVzZXJUeXBlPiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2FkZFBheW1lbnRBY2NvdW50Q2FyZCc7XG4gIGNvbnN0IHtkYXRhYmFzZSwgdXNlcklkOiBzZXNzaW9uSWR9ID0gY29udGV4dDtcblxuICByZXR1cm4gZ2V0VXNlcihjb250ZXh0LCBzZXNzaW9uSWQpXG4gICAgLnRoZW4oKHVzZXI6IFVzZXJUeXBlKSA9PiB7XG4gICAgICAvLyBVc2VyXG4gICAgICBjb25zdCB7XG4gICAgICAgIF9rZXk6IHVzZXJLZXksXG4gICAgICAgIGNvdW50cnkgPSAnVVMnLFxuICAgICAgICBjaXR5LFxuICAgICAgICBjdXJyZW5jeSA9ICdVU0QnLFxuICAgICAgICBkb2IsXG4gICAgICAgIGVtYWlsLFxuICAgICAgICBmaXJzdCxcbiAgICAgICAgbGFzdCxcbiAgICAgICAgdXNlcm5hbWVcbiAgICAgIH06IFVzZXJUeXBlID0gdXNlcjtcblxuICAgICAgLy8gQmlydGhkYXlcbiAgICAgIGNvbnN0IGRvYkRheTogbnVtYmVyID0gKyhEYXRlVGltZS5mcm9tTWlsbGlzKGRvYikudG9Gb3JtYXQoJ2QnKSk7XG4gICAgICBjb25zdCBkb2JNb250aDogbnVtYmVyID0gKyhEYXRlVGltZS5mcm9tTWlsbGlzKGRvYikudG9Gb3JtYXQoJ00nKSk7XG4gICAgICBjb25zdCBkb2JZZWFyOiBudW1iZXIgPSArKERhdGVUaW1lLmZyb21NaWxsaXMoZG9iKS50b0Zvcm1hdCgneScpKTtcblxuICAgICAgLy8gQ2FyZFxuICAgICAgY29uc3Qge1xuICAgICAgICBhY2NvdW50TnVtYmVyLFxuICAgICAgICBhY2NlcHRlZFRlcm1zLFxuICAgICAgICBleHBNb250aCxcbiAgICAgICAgZXhwWWVhcixcbiAgICAgICAgc3RhdGUsXG4gICAgICAgIHN0cmVldDEsXG4gICAgICAgIHppcFxuICAgICAgfTogUGF5bWVudENhcmRUeXBlID0gY2FyZDtcblxuICAgICAgLy8gQWNjZXB0YW5jZVxuICAgICAgY29uc3QgYWNjZXB0ZWQ6IG51bWJlciA9IGFjY2VwdGVkVGVybXMgPyBEYXRlVGltZS5sb2NhbCgpLm1pbGxpc2Vjb25kIDogMDtcblxuICAgICAgLy8gU3RyaXBlXG4gICAgICBjb25zdCBzdHJpcGVDbGllbnQgPSBzdHJpcGUoQ29uZmlnLmdldCgnc3RyaXBlLnRva2VuJykpO1xuXG4gICAgICByZXR1cm4gc3RyaXBlQ2xpZW50LmFjY291bnRzXG4gICAgICAgIC5jcmVhdGUoe1xuICAgICAgICAgIGNvdW50cnksXG4gICAgICAgICAgZW1haWwsXG4gICAgICAgICAgZXh0ZXJuYWxfYWNjb3VudDoge1xuICAgICAgICAgICAgY3VycmVuY3ksXG4gICAgICAgICAgICBleHBfbW9udGg6IGV4cE1vbnRoLFxuICAgICAgICAgICAgZXhwX3llYXI6IGV4cFllYXIsXG4gICAgICAgICAgICBudW1iZXI6IGFjY291bnROdW1iZXIsXG4gICAgICAgICAgICBvYmplY3Q6ICdjYXJkJ1xuICAgICAgICAgIH0sXG4gICAgICAgICAgbGVnYWxfZW50aXR5OiB7XG4gICAgICAgICAgICBhZGRyZXNzOiB7XG4gICAgICAgICAgICAgIGNpdHksXG4gICAgICAgICAgICAgIGxpbmUxOiBzdHJlZXQxLFxuICAgICAgICAgICAgICBwb3N0YWxfY29kZTogemlwLFxuICAgICAgICAgICAgICBzdGF0ZVxuICAgICAgICAgICAgfSxcbiAgICAgICAgICAgIGRvYjoge1xuICAgICAgICAgICAgICBkYXk6IGRvYkRheSxcbiAgICAgICAgICAgICAgbW9udGg6IGRvYk1vbnRoLFxuICAgICAgICAgICAgICB5ZWFyOiBkb2JZZWFyXG4gICAgICAgICAgICB9LFxuICAgICAgICAgICAgZmlyc3RfbmFtZTogZmlyc3QsXG4gICAgICAgICAgICBsYXN0X25hbWU6IGxhc3QsXG4gICAgICAgICAgICAvLyBzc25fbGFzdF80OiBzc24uc3Vic3RyKC00KSxcbiAgICAgICAgICAgIC8vIHBlcnNvbmFsX2lkX251bWJlcjogc3NuLFxuICAgICAgICAgICAgdHlwZTogJ2luZGl2aWR1YWwnXG4gICAgICAgICAgfSxcbiAgICAgICAgICBtYW5hZ2VkOiB0cnVlLFxuICAgICAgICAgIG1ldGFkYXRhOiB7XG4gICAgICAgICAgICB1c2VySWQ6IHVzZXJLZXksXG4gICAgICAgICAgICB1c2VybmFtZVxuICAgICAgICAgIH0sXG4gICAgICAgICAgdG9zX2FjY2VwdGFuY2U6IHtcbiAgICAgICAgICAgIGRhdGU6IGFjY2VwdGVkLFxuICAgICAgICAgICAgaXA6IHJlcXVlc3REYXRhLmlwLFxuICAgICAgICAgICAgdXNlcl9hZ2VudDogcmVxdWVzdERhdGEudXNlckFnZW50LnNvdXJjZVxuICAgICAgICAgIH1cbiAgICAgICAgfSlcbiAgICAgICAgLnRoZW4oKGN1c3RvbWVyKSA9PiB7XG4gICAgICAgICAgLy8gQ3JlYXRlIHNlc3Npb25cbiAgICAgICAgICBjb25zdCBub3c6IG51bWJlciA9IERhdGUubm93KCk7XG4gICAgICAgICAgY29uc3QgdXBkYXRlOiBVc2VyVHlwZSA9IHtcbiAgICAgICAgICAgIG1vZGlmaWVkOiBub3csXG4gICAgICAgICAgICBzdHJpcGVBY2NvdW50SWQ6IGN1c3RvbWVyLmlkXG4gICAgICAgICAgfTtcblxuICAgICAgICAgIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgVVBEQVRFICR7dXNlcktleX0gV0lUSCAke3VwZGF0ZX0gSU4gdXNlcnMgTElNSVQgMSBSRVRVUk4gTkVXYDtcblxuICAgICAgICAgIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkoYXFsUXJ5KVxuICAgICAgICAgICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5uZXh0KCkpXG4gICAgICAgICAgICAudGhlbigodXBkYXRlZFVzZXI6IFVzZXJUeXBlID0ge30pID0+ICFpc0VtcHR5KHVwZGF0ZWRVc2VyKSlcbiAgICAgICAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICAgICAgICAgIGFjdGlvbixcbiAgICAgICAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgICAgICAgIGxhYmVsOiAnZGJfZXJyb3InXG4gICAgICAgICAgICB9LCBlcnJvciwgY29udGV4dCkudGhlbigoKSA9PiBudWxsKSk7XG4gICAgICAgIH0pO1xuICAgIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IGFkZENyZWRpdENhcmQgPSAoY29udGV4dDogQXBpQ29udGV4dCwgY2FyZDogUGF5bWVudENhcmRUeXBlKTogUHJvbWlzZTxQYXltZW50Q2FyZFR5cGU+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnYWRkQ3JlZGl0Q2FyZCc7XG4gIGNvbnN0IHtkYXRhYmFzZSwgdXNlcklkOiBzZXNzaW9uSWR9ID0gY29udGV4dDtcblxuICBjb25zdCB7XG4gICAgYWNjb3VudE51bWJlcixcbiAgICBjaXR5LFxuICAgIGNvdW50cnksXG4gICAgY3ZjLFxuICAgIGV4cE1vbnRoLFxuICAgIGV4cFllYXIsXG4gICAgZnVsbE5hbWUsXG4gICAgc3RyZWV0MSxcbiAgICBzdGF0ZSxcbiAgICB6aXBcbiAgfTogUGF5bWVudENhcmRUeXBlID0gY2FyZDtcbiAgY29uc3QgZm9ybWF0TnVtYmVyOiBudW1iZXIgPSBwYXJzZU51bShhY2NvdW50TnVtYmVyLCAxNik7XG5cbiAgaWYoIWZvcm1hdE51bWJlcikge1xuICAgIHRocm93IG5ldyBVc2VyRXJyb3IoJ3JlcXVpcmVkX2NyZWRpdF9jYXJkX251bWJlcicpO1xuICB9XG5cbiAgY29uc3QgZm9ybWF0RXhwTW9udGg6IG51bWJlciA9IHBhcnNlTnVtKGV4cE1vbnRoLCAyKTtcblxuICBpZighZm9ybWF0RXhwTW9udGgpIHtcbiAgICB0aHJvdyBuZXcgVXNlckVycm9yKCdyZXF1aXJlZF9jcmVkaXRfY2FyZF9leHBfbW9udGgnKTtcbiAgfVxuXG4gIGNvbnN0IGZvcm1hdEV4cFllYXI6IG51bWJlciA9IHBhcnNlTnVtKGV4cFllYXIsIDIpO1xuXG4gIGlmKCFmb3JtYXRFeHBZZWFyKSB7XG4gICAgdGhyb3cgbmV3IFVzZXJFcnJvcigncmVxdWlyZWRfY3JlZGl0X2NhcmRfZXhwX3llYXInKTtcbiAgfVxuXG4gIGNvbnN0IGZvcm1hdEN2YzogbnVtYmVyID0gcGFyc2VOdW0oY3ZjLCAzKTtcblxuICBpZighZm9ybWF0Q3ZjKSB7XG4gICAgdGhyb3cgbmV3IFVzZXJFcnJvcigncmVxdWlyZWRfY3JlZGl0X2NhcmRfY3ZjJyk7XG4gIH1cblxuICBjb25zdCBwYXltZW50Q2FyZDogUGF5bWVudENhcmRUeXBlID0ge307XG4gIGNvbnN0IGZvcm1hdENpdHk6IHN0cmluZyA9IHBhcnNlVmFyQ2hhcihjaXR5LCAzMik7XG5cbiAgaWYoZm9ybWF0Q2l0eSkge1xuICAgIHBheW1lbnRDYXJkLmNpdHkgPSBmb3JtYXRDaXR5O1xuICB9XG5cbiAgY29uc3QgZm9ybWF0Q291bnRyeTogc3RyaW5nID0gcGFyc2VWYXJDaGFyKGNvdW50cnksIDMyKTtcblxuICBpZihmb3JtYXRDb3VudHJ5KSB7XG4gICAgcGF5bWVudENhcmQuY291bnRyeSA9IGZvcm1hdENvdW50cnk7XG4gIH1cblxuICBjb25zdCBmb3JtYXRGdWxsTmFtZTogc3RyaW5nID0gcGFyc2VWYXJDaGFyKGZ1bGxOYW1lLCAzMik7XG5cbiAgaWYoZm9ybWF0RnVsbE5hbWUpIHtcbiAgICBwYXltZW50Q2FyZC5mdWxsTmFtZSA9IGZvcm1hdEZ1bGxOYW1lO1xuICB9XG5cbiAgY29uc3QgZm9ybWF0U3RyZWV0OiBzdHJpbmcgPSBwYXJzZVZhckNoYXIoc3RyZWV0MSwgMzIpO1xuXG4gIGlmKGZvcm1hdFN0cmVldCkge1xuICAgIHBheW1lbnRDYXJkLnN0cmVldDEgPSBmb3JtYXRTdHJlZXQ7XG4gIH1cblxuICBjb25zdCBmb3JtYXRTdGF0ZTogc3RyaW5nID0gcGFyc2VWYXJDaGFyKHN0YXRlLCAzMik7XG5cbiAgaWYoZm9ybWF0U3RhdGUpIHtcbiAgICBwYXltZW50Q2FyZC5zdGF0ZSA9IGZvcm1hdFN0YXRlO1xuICB9XG5cbiAgY29uc3QgZm9ybWF0WmlwOiBzdHJpbmcgPSBwYXJzZVZhckNoYXIoemlwLCAzMik7XG5cbiAgaWYoZm9ybWF0WmlwKSB7XG4gICAgcGF5bWVudENhcmQuemlwID0gZm9ybWF0WmlwO1xuICB9XG5cbiAgLy8gR2V0IHN0cmlwZSBpZFxuICByZXR1cm4gZ2V0VXNlcihjb250ZXh0LCBzZXNzaW9uSWQpXG4gICAgLnRoZW4oKHVzZXI6IFVzZXJUeXBlID0ge30pID0+IHtcbiAgICAgIC8vIFN0cmlwZVxuICAgICAgY29uc3Qgc3RyaXBlQ2xpZW50ID0gc3RyaXBlKENvbmZpZy5nZXQoJ3N0cmlwZS50b2tlbicpKTtcblxuICAgICAgcmV0dXJuIHN0cmlwZUNsaWVudC5jdXN0b21lcnNcbiAgICAgICAgLmNyZWF0ZVNvdXJjZSh1c2VyLnN0cmlwZUN1c3RvbWVySWQsIHtcbiAgICAgICAgICBzb3VyY2U6IHtcbiAgICAgICAgICAgIGFkZHJlc3NfY2l0eTogZm9ybWF0Q2l0eSxcbiAgICAgICAgICAgIGFkZHJlc3NfY291bnRyeTogZm9ybWF0Q291bnRyeSxcbiAgICAgICAgICAgIGFkZHJlc3NfbGluZTE6IGZvcm1hdFN0cmVldCxcbiAgICAgICAgICAgIGFkZHJlc3Nfc3RhdGU6IGZvcm1hdFN0YXRlLFxuICAgICAgICAgICAgYWRkcmVzc196aXA6IGZvcm1hdFppcCxcbiAgICAgICAgICAgIGN1cnJlbmN5OiAndXNkJyxcbiAgICAgICAgICAgIGN2YzogZm9ybWF0Q3ZjLFxuICAgICAgICAgICAgZXhwX21vbnRoOiBmb3JtYXRFeHBNb250aCxcbiAgICAgICAgICAgIGV4cF95ZWFyOiBmb3JtYXRFeHBZZWFyLFxuICAgICAgICAgICAgbmFtZTogZm9ybWF0RnVsbE5hbWUsXG4gICAgICAgICAgICBudW1iZXI6IGZvcm1hdE51bWJlcixcbiAgICAgICAgICAgIG9iamVjdDogJ2NhcmQnXG4gICAgICAgICAgfVxuICAgICAgICB9KVxuICAgICAgICAudGhlbigoc3RyaXBlT2JqKSA9PiB7XG4gICAgICAgICAgY29uc3Qgbm93OiBudW1iZXIgPSBEYXRlLm5vdygpO1xuICAgICAgICAgIGNvbnN0IGluc2VydCA9IHtcbiAgICAgICAgICAgIC4uLnBheW1lbnRDYXJkLFxuICAgICAgICAgICAgX2tleTogY3JlYXRlSGFzaChgdXNlci1wYXltZW50LSR7c2Vzc2lvbklkfWApLFxuICAgICAgICAgICAgYWNjb3VudE51bWJlcjogc3RyaXBlT2JqLmxhc3Q0LFxuICAgICAgICAgICAgYWRkZWQ6IG5vdyxcbiAgICAgICAgICAgIGJyYW5kOiBzdHJpcGVPYmouYnJhbmQsXG4gICAgICAgICAgICBjdmM6IHN0cmlwZU9iai5jdmNfY2hlY2ssXG4gICAgICAgICAgICBleHBNb250aCxcbiAgICAgICAgICAgIGV4cFllYXIsXG4gICAgICAgICAgICBtb2RpZmllZDogbm93LFxuICAgICAgICAgICAgdXNlcklkOiBzZXNzaW9uSWRcbiAgICAgICAgICB9O1xuICAgICAgICAgIGNvbnN0IGluc2VydEFxbFFyeTogQXFsUXVlcnkgPSBhcWxgSU5TRVJUICR7aW5zZXJ0fSBJTiBjcmVkaXRDYXJkcyBSRVRVUk4gTkVXYDtcblxuICAgICAgICAgIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkoaW5zZXJ0QXFsUXJ5KVxuICAgICAgICAgICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5uZXh0KCkpXG4gICAgICAgICAgICAudGhlbigobmV3Q2FyZDogUGF5bWVudENhcmRUeXBlID0ge30pID0+IHtcbiAgICAgICAgICAgICAgaWYoIWlzRW1wdHkobmV3Q2FyZCkpIHtcbiAgICAgICAgICAgICAgICAvLyBBZGQgbGlua2VkIGVkZ2VcbiAgICAgICAgICAgICAgICBjb25zdCB7X2lkOiBjYXJkSWQsIF9rZXk6IGNhcmRLZXl9ID0gY2FyZDtcbiAgICAgICAgICAgICAgICBjb25zdCBlZGdlQ29sbGVjdGlvbiA9IHVzZURiKGRhdGFiYXNlKS5lZGdlQ29sbGVjdGlvbignaGFzUGF5bWVudCcpO1xuICAgICAgICAgICAgICAgIGNvbnN0IGVkZ2VJZCA9IGNyZWF0ZUhhc2goYHBheW1lbnQtJHtjYXJkS2V5fWApO1xuICAgICAgICAgICAgICAgIGNvbnN0IGVkZ2U6IGFueSA9IHtcbiAgICAgICAgICAgICAgICAgIF9rZXk6IGVkZ2VJZFxuICAgICAgICAgICAgICAgIH07XG5cbiAgICAgICAgICAgICAgICByZXR1cm4gZWRnZUNvbGxlY3Rpb24uc2F2ZShlZGdlLCBgdXNlcnMvJHtzZXNzaW9uSWR9YCwgY2FyZElkKS50aGVuKCgpID0+IGNhcmQpO1xuICAgICAgICAgICAgICB9XG5cbiAgICAgICAgICAgICAgcmV0dXJuIG5ld0NhcmQ7XG4gICAgICAgICAgICB9KTtcbiAgICAgICAgfSlcbiAgICAgICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgICAgICBhY3Rpb24sXG4gICAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgICAgbGFiZWw6ICdwYXltZW50X2Vycm9yJ1xuICAgICAgICB9LCBlcnJvciwgY29udGV4dCkudGhlbigoKSA9PiBudWxsKSk7XG4gICAgfSk7XG59O1xuXG5leHBvcnQgY29uc3QgdXBkYXRlQ3JlZGl0Q2FyZCA9IChjb250ZXh0OiBBcGlDb250ZXh0LCBjYXJkOiBQYXltZW50Q2FyZFR5cGUpOiBQcm9taXNlPFBheW1lbnRDYXJkVHlwZT4gPT4ge1xuICBjb25zdCB7ZGF0YWJhc2UsIHVzZXJJZDogc2Vzc2lvbklkfSA9IGNvbnRleHQ7XG5cbiAgY29uc3Qge1xuICAgIGNpdHksXG4gICAgY291bnRyeSxcbiAgICBleHBNb250aCxcbiAgICBleHBZZWFyLFxuICAgIGZ1bGxOYW1lLFxuICAgIGlkLFxuICAgIHN0cmVldDEsXG4gICAgc3RhdGUsXG4gICAgemlwXG4gIH06IFBheW1lbnRDYXJkVHlwZSA9IGNhcmQ7XG5cbiAgY29uc3QgZm9ybWF0SWQ6IHN0cmluZyA9IHBhcnNlSWQoaWQpO1xuXG4gIGlmKGZvcm1hdElkKSB7XG4gICAgdGhyb3cgbmV3IFVzZXJFcnJvcigncmVxdWlyZWRfY3JlZGl0X2NhcmRfaWQnKTtcbiAgfVxuXG4gIGNvbnN0IHBheW1lbnRDYXJkOiBQYXltZW50Q2FyZFR5cGUgPSB7fTtcbiAgY29uc3QgZm9ybWF0RXhwTW9udGg6IG51bWJlciA9IHBhcnNlTnVtKGV4cE1vbnRoLCAyKTtcbiAgY29uc3QgZm9ybWF0RXhwWWVhcjogbnVtYmVyID0gcGFyc2VOdW0oZXhwWWVhciwgMik7XG4gIGNvbnN0IGZvcm1hdENpdHk6IHN0cmluZyA9IHBhcnNlVmFyQ2hhcihjaXR5LCAzMik7XG4gIGNvbnN0IGZvcm1hdENvdW50cnk6IHN0cmluZyA9IHBhcnNlVmFyQ2hhcihjb3VudHJ5LCAyKTtcbiAgY29uc3QgZm9ybWF0RnVsbE5hbWU6IHN0cmluZyA9IHBhcnNlVmFyQ2hhcihmdWxsTmFtZSwgMzIpO1xuICBjb25zdCBmb3JtYXRTdHJlZXQxOiBzdHJpbmcgPSBwYXJzZVN0cmluZyhzdHJlZXQxLCAzMik7XG4gIGNvbnN0IGZvcm1hdFN0YXRlOiBzdHJpbmcgPSBwYXJzZVZhckNoYXIoc3RhdGUsIDIpO1xuICBjb25zdCBmb3JtYXRaaXA6IHN0cmluZyA9IHBhcnNlVmFyQ2hhcih6aXAsIDEwKTtcblxuICBpZihmb3JtYXRFeHBNb250aCkge1xuICAgIHBheW1lbnRDYXJkLmV4cE1vbnRoID0gZm9ybWF0RXhwTW9udGg7XG4gIH1cblxuICBpZihmb3JtYXRFeHBZZWFyKSB7XG4gICAgcGF5bWVudENhcmQuZXhwWWVhciA9IGZvcm1hdEV4cFllYXI7XG4gIH1cblxuICBpZihmb3JtYXRDaXR5KSB7XG4gICAgcGF5bWVudENhcmQuY2l0eSA9IGZvcm1hdENpdHk7XG4gIH1cblxuICBpZihmb3JtYXRDb3VudHJ5KSB7XG4gICAgcGF5bWVudENhcmQuY291bnRyeSA9IGZvcm1hdENvdW50cnk7XG4gIH1cblxuICBpZihmb3JtYXRGdWxsTmFtZSkge1xuICAgIHBheW1lbnRDYXJkLmZ1bGxOYW1lID0gZm9ybWF0RnVsbE5hbWU7XG4gIH1cblxuICBpZihmb3JtYXRTdHJlZXQxKSB7XG4gICAgcGF5bWVudENhcmQuc3RyZWV0MSA9IGZvcm1hdFN0cmVldDE7XG4gIH1cblxuICBpZihmb3JtYXRTdGF0ZSkge1xuICAgIHBheW1lbnRDYXJkLnN0YXRlID0gZm9ybWF0U3RhdGU7XG4gIH1cblxuICBpZihmb3JtYXRaaXApIHtcbiAgICBwYXltZW50Q2FyZC56aXAgPSBmb3JtYXRaaXA7XG4gIH1cblxuICBjb25zdCB1cGRhdGU6IGFueSA9IHBheW1lbnRDYXJkO1xuICBjb25zdCBhcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYFxuICAgICAgTEVUIHVwZGF0ZWRDYXJkID0gRklSU1QoXG4gICAgICAgIEZPUiBjIElOIGNyZWRpdENhcmRzXG4gICAgICAgIEZJTFRFUiBjLl9rZXkgPT0gJHtmb3JtYXRJZH0gJiYgYy51c2VySWQgPT0gJHtzZXNzaW9uSWR9XG4gICAgICAgIFVQREFURSBjIFdJVEggJHt1cGRhdGV9IElOIGNyZWRpdENhcmRzXG4gICAgICAgIExJTUlUIDFcbiAgICAgICAgUkVUVVJOIE5FV1xuICAgICAgKVxuICAgICAgTEVUIHVzZXIgPSBGSVJTVChcbiAgICAgICAgRk9SIHUgSU4gdXNlcnNcbiAgICAgICAgRklMVEVSIHUuX2tleSA9PSAke3Nlc3Npb25JZH1cbiAgICAgICAgTElNSVQgMVxuICAgICAgICBSRVRVUk4gdVxuICAgICAgKVxuICAgICAgUkVUVVJOIHt1c2VyOiB1c2VyLCBjYXJkOiB1cGRhdGVkQ2FyZH1gO1xuXG4gIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgIC50aGVuKChyZXN1bHRzID0ge2NhcmQ6IHt9LCB1c2VyOiB7fX0pID0+IHtcbiAgICAgIGNvbnN0IHVwZGF0ZWRDYXJkOiBQYXltZW50Q2FyZFR5cGUgPSByZXN1bHRzLmNhcmQ7XG4gICAgICBjb25zdCB7dXNlcn0gPSByZXN1bHRzO1xuXG4gICAgICBpZihpc0VtcHR5KHVwZGF0ZWRDYXJkKSkge1xuICAgICAgICB0aHJvdyBuZXcgVXNlckVycm9yKCdub3RfZm91bmQnKTtcbiAgICAgIH1cblxuICAgICAgLy8gU3RyaXBlXG4gICAgICBjb25zdCBzdHJpcGVDbGllbnQgPSBzdHJpcGUoQ29uZmlnLmdldCgnc3RyaXBlLnRva2VuJykpO1xuXG4gICAgICByZXR1cm4gc3RyaXBlQ2xpZW50LmN1c3RvbWVyc1xuICAgICAgICAudXBkYXRlQ2FyZCh1c2VyLnN0cmlwZUN1c3RvbWVySWQsIGNhcmQuc3RyaXBlSWQsIHtcbiAgICAgICAgICBhZGRyZXNzX2NpdHk6IGZvcm1hdENpdHksXG4gICAgICAgICAgYWRkcmVzc19jb3VudHJ5OiBmb3JtYXRDb3VudHJ5LFxuICAgICAgICAgIGFkZHJlc3NfbGluZTE6IGZvcm1hdFN0cmVldDEsXG4gICAgICAgICAgYWRkcmVzc19zdGF0ZTogZm9ybWF0U3RhdGUsXG4gICAgICAgICAgYWRkcmVzc196aXA6IGZvcm1hdFppcCxcbiAgICAgICAgICBleHBfbW9udGg6IGZvcm1hdEV4cE1vbnRoLFxuICAgICAgICAgIGV4cF95ZWFyOiBmb3JtYXRFeHBZZWFyLFxuICAgICAgICAgIG5hbWU6IGZvcm1hdEZ1bGxOYW1lXG4gICAgICAgIH0pXG4gICAgICAgIC50aGVuKCgpID0+IGNhcmQpXG4gICAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiB7XG4gICAgICAgICAgY29uc29sZS5sb2coJ3BheW1lbnRzOjp1cGRhdGVDYXJkOjplcnJvcicsIGVycm9yKTtcbiAgICAgICAgICB0aHJvdyBuZXcgVXNlckVycm9yKCdwYXltZW50X2Vycm9yJyk7XG4gICAgICAgIH0pO1xuICAgIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IGdldENyZWRpdENhcmRzID0gKGNvbnRleHQ6IEFwaUNvbnRleHQpOiBQcm9taXNlPFBheW1lbnRDYXJkVHlwZVtdPiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2dldENyZWRpdENhcmRzJztcbiAgY29uc3Qge2RhdGFiYXNlLCB1c2VySWQ6IHNlc3Npb25JZH0gPSBjb250ZXh0O1xuICBjb25zdCBhcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYEZPUiBjIElOIGNyZWRpdENhcmRzXG4gICAgICAgIEZJTFRFUiBjLnVzZXJJZCA9PSAke3Nlc3Npb25JZH1cbiAgICAgICAgUkVUVVJOIGNgO1xuXG4gIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IuYWxsKCkpXG4gICAgLnRoZW4oKGxpc3Q6IFBheW1lbnRDYXJkVHlwZVtdID0gW10pID0+IGxpc3QpXG4gICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgIGFjdGlvbixcbiAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgbGFiZWw6ICdkYl9lcnJvcidcbiAgICB9LCBlcnJvciwgY29udGV4dCkudGhlbigoKSA9PiBudWxsKSk7XG59O1xuXG5leHBvcnQgY29uc3QgZGVsZXRlQ3JlZGl0Q2FyZCA9IChjb250ZXh0OiBBcGlDb250ZXh0LCBjYXJkSWQ6IHN0cmluZyk6IFByb21pc2U8Ym9vbGVhbj4gPT4ge1xuICBjb25zdCB7ZGF0YWJhc2UsIHVzZXJJZDogc2Vzc2lvbklkfSA9IGNvbnRleHQ7XG4gIGNvbnN0IGZvcm1hdENhcmRJZDogc3RyaW5nID0gcGFyc2VJZChjYXJkSWQpO1xuICBjb25zdCBhcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYFxuICAgICAgTEVUIGNhcmQgPSBGSVJTVChcbiAgICAgICAgRk9SIGMgSU4gY3JlZGl0Q2FyZHNcbiAgICAgICAgRklMVEVSIGMuX2tleSA9PSAke2Zvcm1hdENhcmRJZH0gJiYgYy51c2VySWQgPT0gJHtzZXNzaW9uSWR9XG4gICAgICAgIExJTUlUIDFcbiAgICAgICAgUkVNT1ZFIGMgSU4gY3JlZGl0Q2FyZHNcbiAgICAgICAgUkVUVVJOIE9MRFxuICAgICAgKVxuICAgICAgTEVUIHVzZXIgPSBGSVJTVChcbiAgICAgICAgRk9SIHUgSU4gdXNlcnNcbiAgICAgICAgRklMVEVSIHUuX2tleSA9PSAke3Nlc3Npb25JZH1cbiAgICAgICAgTElNSVQgMVxuICAgICAgICBSRVRVUk4gdVxuICAgICAgKVxuICAgICAgUkVUVVJOIHt1c2VyOiB1c2VyLCBjYXJkOiBjYXJkfWA7XG5cbiAgcmV0dXJuIHVzZURiKGRhdGFiYXNlKS5xdWVyeShhcWxRcnkpXG4gICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5uZXh0KCkpXG4gICAgLnRoZW4oKHJlc3VsdCA9IHtjYXJkOiB7fSwgdXNlcjoge319KSA9PiB7XG4gICAgICBpZihpc0VtcHR5KHJlc3VsdCkpIHtcbiAgICAgICAgcmV0dXJuIGZhbHNlO1xuICAgICAgfVxuXG4gICAgICBjb25zdCB7Y2FyZCwgdXNlcn0gPSByZXN1bHQ7XG4gICAgICBjb25zdCB7X2tleTogY2FyZEtleX0gPSBjYXJkO1xuXG4gICAgICAvLyBSZW1vdmUgbGlua2VkIGVkZ2VzXG4gICAgICBjb25zdCBlZGdlQ29sbGVjdGlvbiA9IHVzZURiKGRhdGFiYXNlKS5lZGdlQ29sbGVjdGlvbignaGFzUGF5bWVudCcpO1xuXG4gICAgICByZXR1cm4gZWRnZUNvbGxlY3Rpb24ub3V0RWRnZXMoY2FyZEtleSlcbiAgICAgICAgLnRoZW4oKGVkZ2VzKSA9PiB7XG4gICAgICAgICAgaWYoZWRnZXMubGVuZ3RoKSB7XG4gICAgICAgICAgICByZXR1cm4gUHJvbWlzZS5hbGwoXG4gICAgICAgICAgICAgIGVkZ2VzLm1hcCgoZWRnZSkgPT4ge1xuICAgICAgICAgICAgICAgIGNvbnN0IHtfa2V5OiBlZGdlS2V5fSA9IGVkZ2U7XG4gICAgICAgICAgICAgICAgY29uc3QgcmVtb3ZlQXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBSRU1PVkUge19rZXk6JHtlZGdlS2V5fX0gSU4gaGFzUGF5bWVudGA7XG4gICAgICAgICAgICAgICAgcmV0dXJuIHVzZURiKGRhdGFiYXNlKS5xdWVyeShyZW1vdmVBcWxRcnkpO1xuICAgICAgICAgICAgICB9KSlcbiAgICAgICAgICAgICAgLnRoZW4oKCkgPT4ge1xuICAgICAgICAgICAgICAgIC8vIFN0cmlwZVxuICAgICAgICAgICAgICAgIGNvbnN0IHN0cmlwZUNsaWVudCA9IHN0cmlwZShDb25maWcuZ2V0KCdzdHJpcGUudG9rZW4nKSk7XG5cbiAgICAgICAgICAgICAgICByZXR1cm4gc3RyaXBlQ2xpZW50LmN1c3RvbWVyc1xuICAgICAgICAgICAgICAgICAgLmRlbGV0ZUNhcmQodXNlci5zdHJpcGVDdXN0b21lcklkLCBjYXJkLnN0cmlwZUlkKVxuICAgICAgICAgICAgICAgICAgLnRoZW4oKCkgPT4gdHJ1ZSlcbiAgICAgICAgICAgICAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiB7XG4gICAgICAgICAgICAgICAgICAgIGNvbnNvbGUubG9nKCdwYXltZW50czo6ZGVsZXRlQ2FyZDo6ZXJyb3InLCBlcnJvcik7XG4gICAgICAgICAgICAgICAgICAgIHRocm93IG5ldyBVc2VyRXJyb3IoJ3BheW1lbnRfZXJyb3InKTtcbiAgICAgICAgICAgICAgICAgIH0pO1xuICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICB9XG5cbiAgICAgICAgICByZXR1cm4gZmFsc2U7XG4gICAgICAgIH0pO1xuICAgIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IGRlbGV0ZVBheW1lbnRBY2NvdW50QmFuayA9IChjb250ZXh0OiBBcGlDb250ZXh0LCBiYW5rSWQ6IHN0cmluZyk6IFByb21pc2U8Ym9vbGVhbj4gPT4ge1xuICBjb25zdCB7ZGF0YWJhc2UsIHVzZXJJZDogc2Vzc2lvbklkfSA9IGNvbnRleHQ7XG5cbiAgLy8gQ2xlYW4gZGJcbiAgY29uc3QgdXBkYXRlOiBVc2VyVHlwZSA9IHtcbiAgICBiYW5rQWNjb3VudDogJycsXG4gICAgYmFua0Z1bGxOYW1lOiAnJyxcbiAgICBiYW5rSWQ6ICcnLFxuICAgIGJhbmtSb3V0aW5nOiAnJyxcbiAgICBtb2RpZmllZDogRGF0ZS5ub3coKVxuICB9O1xuICBjb25zdCBhcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYFVQREFURSAke3Nlc3Npb25JZH0gV0lUSCAke3VwZGF0ZX0gSU4gdXNlcnMgTElNSVQgMSBSRVRVUk4gTkVXYDtcblxuICByZXR1cm4gdXNlRGIoZGF0YWJhc2UpLnF1ZXJ5KGFxbFFyeSlcbiAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAudGhlbigodXNlcjogVXNlclR5cGUgPSB7fSkgPT4ge1xuICAgICAgLy8gU3RyaXBlXG4gICAgICBjb25zdCBzdHJpcGVDbGllbnQgPSBzdHJpcGUoQ29uZmlnLmdldCgnc3RyaXBlLnRva2VuJykpO1xuXG4gICAgICByZXR1cm4gc3RyaXBlQ2xpZW50LmN1c3RvbWVyc1xuICAgICAgICAuZGVsZXRlU291cmNlKHVzZXIuc3RyaXBlQWNjb3VudElkLCBiYW5rSWQpXG4gICAgICAgIC50aGVuKChyZXNwb25zZSA9IHtkZWxldGVkOiBmYWxzZX0pID0+IHJlc3BvbnNlLmRlbGV0ZWQpXG4gICAgICAgIC5jYXRjaCgoKSA9PiBQcm9taXNlLnJlc29sdmUoZmFsc2UpKTtcbiAgICB9KTtcbn07XG5cbmV4cG9ydCBjb25zdCBjcmVhdGVQYXltZW50VHJhbnNmZXIgPSAoY29udGV4dDogQXBpQ29udGV4dCwgdHJhbnNmZXI6IFBheW1lbnRUcmFuc2Zlcik6IFByb21pc2U8UGF5bWVudFRyYW5zZmVyPiA9PiB7XG4gIGNvbnN0IHtkYXRhYmFzZSwgdXNlcklkOiBzZXNzaW9uSWR9ID0gY29udGV4dDtcbiAgY29uc3Qge2Ftb3VudCwgY3VycmVuY3l9ID0gdHJhbnNmZXI7XG4gIGNvbnN0IGZvcm1hdEFtb3VudDogbnVtYmVyID0gcGFyc2VOdW0oYW1vdW50KTtcbiAgY29uc3QgZm9ybWF0Q3VycmVuY3k6IHN0cmluZyA9IHBhcnNlQ2hhcihjdXJyZW5jeSwgMywgJ1VTRCcpLnRvVXBwZXJDYXNlKCk7XG5cbiAgcmV0dXJuIGdldFVzZXIoY29udGV4dCwgc2Vzc2lvbklkKVxuICAgIC50aGVuKCh1c2VyOiBVc2VyVHlwZSkgPT4ge1xuICAgICAgLy8gU3RyaXBlXG4gICAgICBjb25zdCBzdHJpcGVDbGllbnQgPSBzdHJpcGUoQ29uZmlnLmdldCgnc3RyaXBlLnRva2VuJykpO1xuXG4gICAgICByZXR1cm4gc3RyaXBlQ2xpZW50LnRyYW5zZmVyc1xuICAgICAgICAuY3JlYXRlKHtcbiAgICAgICAgICBhbW91bnQ6IGZvcm1hdEFtb3VudCxcbiAgICAgICAgICBjdXJyZW5jeTogZm9ybWF0Q3VycmVuY3ksXG4gICAgICAgICAgZGVzdGluYXRpb246IHVzZXIuc3RyaXBlQWNjb3VudElkXG4gICAgICAgIH0pXG4gICAgICAgIC50aGVuKChzdHJpcGVUcmFuc2ZlcikgPT4ge1xuICAgICAgICAgIGNvbnNvbGUubG9nKHN0cmlwZVRyYW5zZmVyKTtcbiAgICAgICAgICBjb25zdCBub3c6IG51bWJlciA9IERhdGUubm93KCk7XG4gICAgICAgICAgY29uc3QgaW5zZXJ0OiBQYXltZW50VHJhbnNmZXIgPSB7XG4gICAgICAgICAgICBhZGRlZDogbm93LFxuICAgICAgICAgICAgYW1vdW50OiBmb3JtYXRBbW91bnQsXG4gICAgICAgICAgICBjdXJyZW5jeTogZm9ybWF0Q3VycmVuY3ksXG4gICAgICAgICAgICBtb2RpZmllZDogbm93LFxuICAgICAgICAgICAgdXNlcklkOiBzZXNzaW9uSWRcbiAgICAgICAgICB9O1xuICAgICAgICAgIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgSU5TRVJUICR7aW5zZXJ0fSBJTiB0cmFuc2ZlcnMgUkVUVVJOIE5FV2A7XG5cbiAgICAgICAgICByZXR1cm4gdXNlRGIoZGF0YWJhc2UpLnF1ZXJ5KGFxbFFyeSlcbiAgICAgICAgICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgICAgICAgICAgLnRoZW4oKG5ld1RyYW5zZmVyOiBQYXltZW50VHJhbnNmZXIpID0+IG5ld1RyYW5zZmVyKTtcbiAgICAgICAgfSk7XG4gICAgfSk7XG59O1xuXG5leHBvcnQgY29uc3QgY3JlYXRlUGF5bWVudEhvbGQgPSAoY29udGV4dDogQXBpQ29udGV4dCwgcGF5bWVudDogUGF5bWVudENoYXJnZSk6IFByb21pc2U8UGF5bWVudENoYXJnZT4gPT4ge1xuICBjb25zdCB7ZGF0YWJhc2UsIHVzZXJJZDogc2Vzc2lvbklkfSA9IGNvbnRleHQ7XG4gIGNvbnN0IHthbW91bnQsIGNhcHR1cmUsIGNhcmRJZCwgY3VycmVuY3ksIGRlc2NyaXB0aW9ufSA9IHBheW1lbnQ7XG4gIGNvbnN0IGZvcm1hdEN1cnJlbmN5ID0gcGFyc2VDaGFyKGN1cnJlbmN5LCAzLCAnVVNEJykudG9VcHBlckNhc2UoKTtcblxuICBjb25zdCBzdHJpcGVDbGllbnQgPSBzdHJpcGUoQ29uZmlnLmdldCgnc3RyaXBlLnRva2VuJykpO1xuXG4gIHJldHVybiBzdHJpcGVDbGllbnQuY2hhcmdlc1xuICAgIC5jcmVhdGUoe1xuICAgICAgYW1vdW50LFxuICAgICAgY2FwdHVyZSxcbiAgICAgIGN1cnJlbmN5OiBmb3JtYXRDdXJyZW5jeSxcbiAgICAgIGRlc2NyaXB0aW9uLFxuICAgICAgc291cmNlOiBjYXJkSWRcbiAgICB9KVxuICAgIC50aGVuKChzdHJpcGVDaGFyZ2UpID0+IHtcbiAgICAgIGNvbnN0IG5vdzogbnVtYmVyID0gRGF0ZS5ub3coKTtcbiAgICAgIGNvbnN0IGluc2VydDogUGF5bWVudENoYXJnZSA9IHtcbiAgICAgICAgYWRkZWQ6IG5vdyxcbiAgICAgICAgYW1vdW50LFxuICAgICAgICBjYXB0dXJlLFxuICAgICAgICBjYXJkSWQsXG4gICAgICAgIGNoYXJnZUZhaWxDb2RlOiBzdHJpcGVDaGFyZ2UuZmFpbHVyZV9jb2RlLFxuICAgICAgICBjaGFyZ2VGYWlsTXNnOiBzdHJpcGVDaGFyZ2UuZmFpbHVyZV9tZXNzYWdlLFxuICAgICAgICBjaGFyZ2VJZDogc3RyaXBlQ2hhcmdlLmlkLFxuICAgICAgICBjaGFyZ2VTdGF0dXM6IHN0cmlwZUNoYXJnZS5zdGF0dXMsXG4gICAgICAgIGN1cnJlbmN5OiBmb3JtYXRDdXJyZW5jeSxcbiAgICAgICAgZGVzY3JpcHRpb24sXG4gICAgICAgIG1vZGlmaWVkOiBub3csXG4gICAgICAgIHVzZXJJZDogc2Vzc2lvbklkXG4gICAgICB9O1xuICAgICAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBJTlNFUlQgJHtpbnNlcnR9IElOIHBheW1lbnRzIFJFVFVSTiBORVdgO1xuXG4gICAgICByZXR1cm4gdXNlRGIoZGF0YWJhc2UpLnF1ZXJ5KGFxbFFyeSlcbiAgICAgICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5uZXh0KCkpXG4gICAgICAgIC50aGVuKChuZXdQYXltZW50OiBQYXltZW50Q2hhcmdlKSA9PiBuZXdQYXltZW50KTtcbiAgICB9KVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiB7XG4gICAgICBjb25zb2xlLmxvZygncGF5bWVudHM6OmNyZWF0ZUhvbGQ6OmVycm9yJywgZXJyb3IpO1xuICAgICAgdGhyb3cgbmV3IFVzZXJFcnJvcigncGF5bWVudF9lcnJvcicpO1xuICAgIH0pO1xufTtcbiJdfQ==
|