serverless-spy 0.0.32 → 0.0.34

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (49) hide show
  1. package/.jsii +9 -9
  2. package/common/publishSpyEvent.ts +269 -0
  3. package/dist/releasetag.txt +1 -1
  4. package/extension/interceptor.ts +24 -13
  5. package/functions/onConnect.ts +2 -1
  6. package/functions/onDisconnect.ts +2 -1
  7. package/functions/sendMessage.ts +3 -268
  8. package/lib/common/publishSpyEvent.d.ts +4 -0
  9. package/lib/common/publishSpyEvent.js +211 -0
  10. package/lib/common/publishSpyEvent.mjs +205 -0
  11. package/lib/extension/dist/layer/nodejs/node_modules/interceptor.js +13890 -13
  12. package/lib/extension/dist/layer/nodejs/node_modules/interceptor.js.map +4 -4
  13. package/lib/listener/{SpyListener.d.ts → ServerlessSpyListener.d.ts} +12 -12
  14. package/lib/listener/ServerlessSpyListener.js +3 -0
  15. package/lib/listener/ServerlessSpyListener.mjs +2 -0
  16. package/lib/listener/ServerlessSpyListenerParams.d.ts +5 -0
  17. package/lib/listener/ServerlessSpyListenerParams.js +3 -0
  18. package/lib/listener/ServerlessSpyListenerParams.mjs +2 -0
  19. package/lib/listener/SpyHandlers.ts.d.ts +1 -1
  20. package/lib/listener/SpyHandlers.ts.js +1 -1
  21. package/lib/listener/SpyHandlers.ts.mjs +1 -1
  22. package/lib/listener/WaitForParams.d.ts +5 -0
  23. package/lib/listener/WaitForParams.js +3 -0
  24. package/lib/listener/WaitForParams.mjs +2 -0
  25. package/lib/listener/WsListener.d.ts +38 -0
  26. package/lib/listener/WsListener.js +185 -0
  27. package/lib/listener/WsListener.mjs +181 -0
  28. package/lib/listener/createServerlessSpyListener.d.ts +2 -13
  29. package/lib/listener/createServerlessSpyListener.js +5 -163
  30. package/lib/listener/createServerlessSpyListener.mjs +5 -163
  31. package/lib/listener/index.d.ts +1 -1
  32. package/lib/listener/index.js +2 -2
  33. package/lib/listener/index.mjs +2 -2
  34. package/lib/src/ServerlessSpy.d.ts +16 -8
  35. package/lib/src/ServerlessSpy.js +94 -53
  36. package/lib/src/ServerlessSpy.mjs +93 -52
  37. package/lib/src/common/envVariableNames.d.ts +8 -0
  38. package/lib/src/common/envVariableNames.js +15 -0
  39. package/lib/src/common/envVariableNames.mjs +12 -0
  40. package/listener/{SpyListener.ts → ServerlessSpyListener.ts} +12 -13
  41. package/listener/ServerlessSpyListenerParams.ts +6 -0
  42. package/listener/SpyHandlers.ts.ts +1 -1
  43. package/listener/WaitForParams.ts +6 -0
  44. package/listener/WsListener.ts +273 -0
  45. package/listener/createServerlessSpyListener.ts +5 -265
  46. package/listener/index.ts +1 -1
  47. package/package.json +1 -1
  48. package/lib/listener/SpyListener.js +0 -3
  49. package/lib/listener/SpyListener.mjs +0 -2
@@ -0,0 +1,205 @@
1
+ import { ApiGatewayManagementApi, PostToConnectionCommand, } from '@aws-sdk/client-apigatewaymanagementapi';
2
+ import { DeleteItemCommand, DynamoDBClient, ScanCommand, } from '@aws-sdk/client-dynamodb';
3
+ import { unmarshall } from '@aws-sdk/util-dynamodb';
4
+ import { envVariableNames } from '../src/common/envVariableNames';
5
+ const ddb = new DynamoDBClient({
6
+ region: process.env.AWS_REGION,
7
+ });
8
+ const apigwManagementApi = new ApiGatewayManagementApi({
9
+ apiVersion: '2018-11-29',
10
+ endpoint: process.env[envVariableNames.WS_ENDPOINT],
11
+ });
12
+ let connections;
13
+ export async function publishSpyEvent(event) {
14
+ console.log('EVENT', JSON.stringify(event));
15
+ const mapping = JSON.parse(process.env[envVariableNames.INFRA_MAPPING]);
16
+ console.log('mapping', JSON.stringify(mapping));
17
+ let connectionData;
18
+ const scanParams = new ScanCommand({
19
+ TableName: process.env[envVariableNames.TABLE_NAME],
20
+ ProjectionExpression: 'connectionId',
21
+ });
22
+ connectionData = await ddb.send(scanParams);
23
+ connections = connectionData.Items;
24
+ console.log(`connections at ${new Date().toISOString()}`, JSON.stringify(connections));
25
+ const postDataPromises = [];
26
+ if (event?.Records && event.Records[0]?.Sns) {
27
+ console.log('*** SNS ***');
28
+ const eventSns = event;
29
+ for (const record of eventSns.Records) {
30
+ const subscriptionArn = record.EventSubscriptionArn;
31
+ let serviceKey;
32
+ if (mapping[subscriptionArn]) {
33
+ // subscription event that could contain filter based on existing subscription
34
+ serviceKey = mapping[subscriptionArn];
35
+ }
36
+ else {
37
+ // catch all subscription
38
+ const topicArn = record.Sns.TopicArn;
39
+ serviceKey = mapping[topicArn];
40
+ }
41
+ let message;
42
+ try {
43
+ message = JSON.parse(record.Sns.Message);
44
+ }
45
+ catch {
46
+ message = record.Sns.Message;
47
+ }
48
+ const spyEventType = getSpyEventType(serviceKey);
49
+ const data = {
50
+ spyEventType,
51
+ message,
52
+ subject: record.Sns.Subject,
53
+ timestamp: record.Sns.Timestamp,
54
+ topicArn: record.Sns.TopicArn,
55
+ messageId: record.Sns.MessageId,
56
+ messageAttributes: record.Sns.MessageAttributes,
57
+ };
58
+ const fluentEvent = {
59
+ data,
60
+ serviceKey,
61
+ };
62
+ postDataPromises.push(postData(fluentEvent));
63
+ }
64
+ }
65
+ else if (event?.Records && event.Records[0]?.eventSource === 'aws:sqs') {
66
+ console.log('*** SQS ***');
67
+ const eventSqs = event;
68
+ for (const record of eventSqs.Records) {
69
+ const subscriptionArn = record.eventSourceARN;
70
+ const serviceKey = mapping[subscriptionArn];
71
+ let body;
72
+ try {
73
+ body = JSON.parse(record.body);
74
+ }
75
+ catch {
76
+ body = record.body;
77
+ }
78
+ const data = {
79
+ spyEventType: 'Sqs',
80
+ body,
81
+ messageAttributes: record.messageAttributes,
82
+ };
83
+ const fluentEvent = {
84
+ data,
85
+ serviceKey,
86
+ };
87
+ postDataPromises.push(postData(fluentEvent));
88
+ }
89
+ }
90
+ else if (event?.Records && event.Records[0]?.s3) {
91
+ console.log('*** S3 ***');
92
+ const eventS3 = event;
93
+ for (const record of eventS3.Records) {
94
+ const bucketArn = record.s3.bucket.arn;
95
+ const data = {
96
+ spyEventType: 'S3',
97
+ eventName: record.eventName,
98
+ eventTime: record.eventTime,
99
+ bucket: record.s3.bucket.name,
100
+ key: record.s3.object.key,
101
+ };
102
+ const fluentEvent = {
103
+ data,
104
+ serviceKey: mapping[bucketArn],
105
+ };
106
+ postDataPromises.push(postData(fluentEvent));
107
+ }
108
+ }
109
+ else if (event.Records && event.Records[0]?.dynamodb) {
110
+ console.log('*** DYNAMODB ***');
111
+ const eventDynamoDB = event;
112
+ for (const record of eventDynamoDB.Records) {
113
+ let arn = record.eventSourceARN;
114
+ arn = arn.substring(0, arn.indexOf('/stream/'));
115
+ const data = {
116
+ spyEventType: 'DynamoDB',
117
+ eventName: record.eventName,
118
+ newImage: unmarshall(record.dynamodb?.NewImage),
119
+ keys: unmarshall(record.dynamodb?.Keys),
120
+ oldImage: record.dynamodb?.OldImage
121
+ ? unmarshall(record.dynamodb?.OldImage)
122
+ : undefined,
123
+ };
124
+ const fluentEvent = {
125
+ data,
126
+ serviceKey: mapping[arn],
127
+ };
128
+ postDataPromises.push(postData(fluentEvent));
129
+ }
130
+ }
131
+ else if (event.detail &&
132
+ event['detail-type'] &&
133
+ event.version &&
134
+ event.source) {
135
+ console.log('*** EventBridge ***');
136
+ const eventEb = event;
137
+ const serviceKey = mapping.eventBridge; // the is new lambda for each subscription
138
+ const spyEventType = getSpyEventType(serviceKey);
139
+ const message = eventEb.detail;
140
+ const data = {
141
+ spyEventType,
142
+ detail: message,
143
+ detailType: eventEb['detail-type'],
144
+ source: eventEb.source,
145
+ time: eventEb.time,
146
+ account: eventEb.account,
147
+ };
148
+ const fluentEvent = {
149
+ data,
150
+ serviceKey,
151
+ };
152
+ postDataPromises.push(postData(fluentEvent));
153
+ }
154
+ else {
155
+ console.log('*** OTHER ***');
156
+ const fluentEvent = event;
157
+ postDataPromises.push(postData(fluentEvent));
158
+ }
159
+ await Promise.all(postDataPromises);
160
+ }
161
+ export async function postData(spyMessage) {
162
+ // const postData = JSON.parse(event.body!).data;
163
+ console.log('postData', JSON.stringify(spyMessage));
164
+ if (!connections) {
165
+ return;
166
+ }
167
+ const postCalls = connections.map(async ({ connectionId }) => {
168
+ console.log(`Sending message to: ${connectionId.S}`);
169
+ try {
170
+ const postToConnectionCommand = new PostToConnectionCommand({
171
+ ConnectionId: connectionId.S,
172
+ Data: JSON.stringify({
173
+ timestamp: new Date().toISOString(),
174
+ serviceKey: spyMessage.serviceKey,
175
+ data: spyMessage.data,
176
+ }),
177
+ });
178
+ await apigwManagementApi.send(postToConnectionCommand);
179
+ }
180
+ catch (e) {
181
+ console.error(`Fails sending message to: ${connectionId.S}`, e);
182
+ console.error(`Status code: ${e.statusCode}`);
183
+ if (e.$metadata.httpStatusCode === 410) {
184
+ console.log(`Found stale connection, deleting ${connectionId}`);
185
+ const deleteParams = new DeleteItemCommand({
186
+ TableName: process.env[envVariableNames.TABLE_NAME],
187
+ Key: { connectionId },
188
+ });
189
+ await ddb.send(deleteParams);
190
+ }
191
+ else {
192
+ throw e;
193
+ }
194
+ }
195
+ });
196
+ await Promise.all(postCalls);
197
+ console.log('Send message finish');
198
+ }
199
+ export function getSpyEventType(serviceKey) {
200
+ if (!serviceKey) {
201
+ throw new Error('Missing serviceKey');
202
+ }
203
+ return serviceKey.substring(0, serviceKey.indexOf('#'));
204
+ }
205
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGlzaFNweUV2ZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vY29tbW9uL3B1Ymxpc2hTcHlFdmVudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQ0wsdUJBQXVCLEVBQ3ZCLHVCQUF1QixHQUN4QixNQUFNLHlDQUF5QyxDQUFDO0FBQ2pELE9BQU8sRUFFTCxpQkFBaUIsRUFDakIsY0FBYyxFQUNkLFdBQVcsR0FDWixNQUFNLDBCQUEwQixDQUFDO0FBRWxDLE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSx3QkFBd0IsQ0FBQztBQVFwRCxPQUFPLEVBQUUsZ0JBQWdCLEVBQUUsTUFBTSxnQ0FBZ0MsQ0FBQztBQVVsRSxNQUFNLEdBQUcsR0FBRyxJQUFJLGNBQWMsQ0FBQztJQUM3QixNQUFNLEVBQUUsT0FBTyxDQUFDLEdBQUcsQ0FBQyxVQUFVO0NBQy9CLENBQUMsQ0FBQztBQUVILE1BQU0sa0JBQWtCLEdBQUcsSUFBSSx1QkFBdUIsQ0FBQztJQUNyRCxVQUFVLEVBQUUsWUFBWTtJQUN4QixRQUFRLEVBQUUsT0FBTyxDQUFDLEdBQUcsQ0FBQyxnQkFBZ0IsQ0FBQyxXQUFXLENBQUU7Q0FDckQsQ0FBQyxDQUFDO0FBRUgsSUFBSSxXQUF5RCxDQUFDO0FBRTlELE1BQU0sQ0FBQyxLQUFLLFVBQVUsZUFBZSxDQUFDLEtBQVU7SUFDOUMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxPQUFPLEVBQUUsSUFBSSxDQUFDLFNBQVMsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDO0lBRTVDLE1BQU0sT0FBTyxHQUFHLElBQUksQ0FBQyxLQUFLLENBQUMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxnQkFBZ0IsQ0FBQyxhQUFhLENBQUUsQ0FBQyxDQUFDO0lBQ3pFLE9BQU8sQ0FBQyxHQUFHLENBQUMsU0FBUyxFQUFFLElBQUksQ0FBQyxTQUFTLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQztJQUVoRCxJQUFJLGNBQWMsQ0FBQztJQUVuQixNQUFNLFVBQVUsR0FBRyxJQUFJLFdBQVcsQ0FBQztRQUNqQyxTQUFTLEVBQUUsT0FBTyxDQUFDLEdBQUcsQ0FBQyxnQkFBZ0IsQ0FBQyxVQUFVLENBQVc7UUFDN0Qsb0JBQW9CLEVBQUUsY0FBYztLQUNyQyxDQUFDLENBQUM7SUFFSCxjQUFjLEdBQUcsTUFBTSxHQUFHLENBQUMsSUFBSSxDQUFDLFVBQVUsQ0FBQyxDQUFDO0lBRTVDLFdBQVcsR0FBRyxjQUFjLENBQUMsS0FBSyxDQUFDO0lBRW5DLE9BQU8sQ0FBQyxHQUFHLENBQ1Qsa0JBQWtCLElBQUksSUFBSSxFQUFFLENBQUMsV0FBVyxFQUFFLEVBQUUsRUFDNUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxXQUFXLENBQUMsQ0FDNUIsQ0FBQztJQUVGLE1BQU0sZ0JBQWdCLEdBQW1CLEVBQUUsQ0FBQztJQUU1QyxJQUFJLEtBQUssRUFBRSxPQUFPLElBQUksS0FBSyxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsRUFBRSxHQUFHLEVBQUU7UUFDM0MsT0FBTyxDQUFDLEdBQUcsQ0FBQyxhQUFhLENBQUMsQ0FBQztRQUMzQixNQUFNLFFBQVEsR0FBRyxLQUFpQixDQUFDO1FBQ25DLEtBQUssTUFBTSxNQUFNLElBQUksUUFBUSxDQUFDLE9BQU8sRUFBRTtZQUNyQyxNQUFNLGVBQWUsR0FBRyxNQUFNLENBQUMsb0JBQW9CLENBQUM7WUFFcEQsSUFBSSxVQUFrQixDQUFDO1lBQ3ZCLElBQUksT0FBTyxDQUFDLGVBQWUsQ0FBQyxFQUFFO2dCQUM1Qiw4RUFBOEU7Z0JBQzlFLFVBQVUsR0FBRyxPQUFPLENBQUMsZUFBZSxDQUFDLENBQUM7YUFDdkM7aUJBQU07Z0JBQ0wseUJBQXlCO2dCQUN6QixNQUFNLFFBQVEsR0FBRyxNQUFNLENBQUMsR0FBRyxDQUFDLFFBQVEsQ0FBQztnQkFDckMsVUFBVSxHQUFHLE9BQU8sQ0FBQyxRQUFRLENBQUMsQ0FBQzthQUNoQztZQUVELElBQUksT0FBZSxDQUFDO1lBRXBCLElBQUk7Z0JBQ0YsT0FBTyxHQUFHLElBQUksQ0FBQyxLQUFLLENBQUMsTUFBTSxDQUFDLEdBQUcsQ0FBQyxPQUFPLENBQUMsQ0FBQzthQUMxQztZQUFDLE1BQU07Z0JBQ04sT0FBTyxHQUFHLE1BQU0sQ0FBQyxHQUFHLENBQUMsT0FBTyxDQUFDO2FBQzlCO1lBRUQsTUFBTSxZQUFZLEdBQUcsZUFBZSxDQUFDLFVBQVUsQ0FFbEIsQ0FBQztZQUU5QixNQUFNLElBQUksR0FBK0M7Z0JBQ3ZELFlBQVk7Z0JBQ1osT0FBTztnQkFDUCxPQUFPLEVBQUUsTUFBTSxDQUFDLEdBQUcsQ0FBQyxPQUFPO2dCQUMzQixTQUFTLEVBQUUsTUFBTSxDQUFDLEdBQUcsQ0FBQyxTQUFTO2dCQUMvQixRQUFRLEVBQUUsTUFBTSxDQUFDLEdBQUcsQ0FBQyxRQUFRO2dCQUM3QixTQUFTLEVBQUUsTUFBTSxDQUFDLEdBQUcsQ0FBQyxTQUFTO2dCQUMvQixpQkFBaUIsRUFBRSxNQUFNLENBQUMsR0FBRyxDQUFDLGlCQUFpQjthQUNoRCxDQUFDO1lBRUYsTUFBTSxXQUFXLEdBQWtDO2dCQUNqRCxJQUFJO2dCQUNKLFVBQVU7YUFDWCxDQUFDO1lBQ0YsZ0JBQWdCLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxXQUFXLENBQUMsQ0FBQyxDQUFDO1NBQzlDO0tBQ0Y7U0FBTSxJQUFJLEtBQUssRUFBRSxPQUFPLElBQUksS0FBSyxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsRUFBRSxXQUFXLEtBQUssU0FBUyxFQUFFO1FBQ3hFLE9BQU8sQ0FBQyxHQUFHLENBQUMsYUFBYSxDQUFDLENBQUM7UUFDM0IsTUFBTSxRQUFRLEdBQUcsS0FBaUIsQ0FBQztRQUNuQyxLQUFLLE1BQU0sTUFBTSxJQUFJLFFBQVEsQ0FBQyxPQUFPLEVBQUU7WUFDckMsTUFBTSxlQUFlLEdBQUcsTUFBTSxDQUFDLGNBQWMsQ0FBQztZQUU5QyxNQUFNLFVBQVUsR0FBRyxPQUFPLENBQUMsZUFBZSxDQUFDLENBQUM7WUFDNUMsSUFBSSxJQUFZLENBQUM7WUFFakIsSUFBSTtnQkFDRixJQUFJLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLENBQUM7YUFDaEM7WUFBQyxNQUFNO2dCQUNOLElBQUksR0FBRyxNQUFNLENBQUMsSUFBSSxDQUFDO2FBQ3BCO1lBRUQsTUFBTSxJQUFJLEdBQWdCO2dCQUN4QixZQUFZLEVBQUUsS0FBSztnQkFDbkIsSUFBSTtnQkFDSixpQkFBaUIsRUFBRSxNQUFNLENBQUMsaUJBQWlCO2FBQzVDLENBQUM7WUFFRixNQUFNLFdBQVcsR0FBa0M7Z0JBQ2pELElBQUk7Z0JBQ0osVUFBVTthQUNYLENBQUM7WUFDRixnQkFBZ0IsQ0FBQyxJQUFJLENBQUMsUUFBUSxDQUFDLFdBQVcsQ0FBQyxDQUFDLENBQUM7U0FDOUM7S0FDRjtTQUFNLElBQUksS0FBSyxFQUFFLE9BQU8sSUFBSSxLQUFLLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQyxFQUFFLEVBQUUsRUFBRTtRQUNqRCxPQUFPLENBQUMsR0FBRyxDQUFDLFlBQVksQ0FBQyxDQUFDO1FBQzFCLE1BQU0sT0FBTyxHQUFHLEtBQWdCLENBQUM7UUFDakMsS0FBSyxNQUFNLE1BQU0sSUFBSSxPQUFPLENBQUMsT0FBTyxFQUFFO1lBQ3BDLE1BQU0sU0FBUyxHQUFHLE1BQU0sQ0FBQyxFQUFFLENBQUMsTUFBTSxDQUFDLEdBQUcsQ0FBQztZQUV2QyxNQUFNLElBQUksR0FBZTtnQkFDdkIsWUFBWSxFQUFFLElBQUk7Z0JBQ2xCLFNBQVMsRUFBRSxNQUFNLENBQUMsU0FBUztnQkFDM0IsU0FBUyxFQUFFLE1BQU0sQ0FBQyxTQUFTO2dCQUMzQixNQUFNLEVBQUUsTUFBTSxDQUFDLEVBQUUsQ0FBQyxNQUFNLENBQUMsSUFBSTtnQkFDN0IsR0FBRyxFQUFFLE1BQU0sQ0FBQyxFQUFFLENBQUMsTUFBTSxDQUFDLEdBQUc7YUFDMUIsQ0FBQztZQUVGLE1BQU0sV0FBVyxHQUFrQztnQkFDakQsSUFBSTtnQkFDSixVQUFVLEVBQUUsT0FBTyxDQUFDLFNBQVMsQ0FBQzthQUMvQixDQUFDO1lBQ0YsZ0JBQWdCLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxXQUFXLENBQUMsQ0FBQyxDQUFDO1NBQzlDO0tBQ0Y7U0FBTSxJQUFJLEtBQUssQ0FBQyxPQUFPLElBQUksS0FBSyxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsRUFBRSxRQUFRLEVBQUU7UUFDdEQsT0FBTyxDQUFDLEdBQUcsQ0FBQyxrQkFBa0IsQ0FBQyxDQUFDO1FBQ2hDLE1BQU0sYUFBYSxHQUFHLEtBQTRCLENBQUM7UUFDbkQsS0FBSyxNQUFNLE1BQU0sSUFBSSxhQUFhLENBQUMsT0FBTyxFQUFFO1lBQzFDLElBQUksR0FBRyxHQUFHLE1BQU0sQ0FBQyxjQUFlLENBQUM7WUFDakMsR0FBRyxHQUFHLEdBQUcsQ0FBQyxTQUFTLENBQUMsQ0FBQyxFQUFFLEdBQUcsQ0FBQyxPQUFPLENBQUMsVUFBVSxDQUFDLENBQUMsQ0FBQztZQUVoRCxNQUFNLElBQUksR0FBcUI7Z0JBQzdCLFlBQVksRUFBRSxVQUFVO2dCQUN4QixTQUFTLEVBQUUsTUFBTSxDQUFDLFNBQVM7Z0JBQzNCLFFBQVEsRUFBRSxVQUFVLENBQUMsTUFBTSxDQUFDLFFBQVEsRUFBRSxRQUFlLENBQUM7Z0JBQ3RELElBQUksRUFBRSxVQUFVLENBQUMsTUFBTSxDQUFDLFFBQVEsRUFBRSxJQUFXLENBQUM7Z0JBQzlDLFFBQVEsRUFBRSxNQUFNLENBQUMsUUFBUSxFQUFFLFFBQVE7b0JBQ2pDLENBQUMsQ0FBQyxVQUFVLENBQUMsTUFBTSxDQUFDLFFBQVEsRUFBRSxRQUFlLENBQUM7b0JBQzlDLENBQUMsQ0FBQyxTQUFTO2FBQ2QsQ0FBQztZQUVGLE1BQU0sV0FBVyxHQUFrQztnQkFDakQsSUFBSTtnQkFDSixVQUFVLEVBQUUsT0FBTyxDQUFDLEdBQUcsQ0FBQzthQUN6QixDQUFDO1lBQ0YsZ0JBQWdCLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxXQUFXLENBQUMsQ0FBQyxDQUFDO1NBQzlDO0tBQ0Y7U0FBTSxJQUNMLEtBQUssQ0FBQyxNQUFNO1FBQ1osS0FBSyxDQUFDLGFBQWEsQ0FBQztRQUNwQixLQUFLLENBQUMsT0FBTztRQUNiLEtBQUssQ0FBQyxNQUFNLEVBQ1o7UUFDQSxPQUFPLENBQUMsR0FBRyxDQUFDLHFCQUFxQixDQUFDLENBQUM7UUFDbkMsTUFBTSxPQUFPLEdBQUcsS0FBbUMsQ0FBQztRQUVwRCxNQUFNLFVBQVUsR0FBRyxPQUFPLENBQUMsV0FBVyxDQUFDLENBQUMsMENBQTBDO1FBRWxGLE1BQU0sWUFBWSxHQUFHLGVBQWUsQ0FBQyxVQUFVLENBRTFCLENBQUM7UUFFdEIsTUFBTSxPQUFPLEdBQUcsT0FBTyxDQUFDLE1BQU0sQ0FBQztRQUUvQixNQUFNLElBQUksR0FBa0Q7WUFDMUQsWUFBWTtZQUNaLE1BQU0sRUFBRSxPQUFPO1lBQ2YsVUFBVSxFQUFFLE9BQU8sQ0FBQyxhQUFhLENBQUM7WUFDbEMsTUFBTSxFQUFFLE9BQU8sQ0FBQyxNQUFNO1lBQ3RCLElBQUksRUFBRSxPQUFPLENBQUMsSUFBSTtZQUNsQixPQUFPLEVBQUUsT0FBTyxDQUFDLE9BQU87U0FDekIsQ0FBQztRQUVGLE1BQU0sV0FBVyxHQUFrQztZQUNqRCxJQUFJO1lBQ0osVUFBVTtTQUNYLENBQUM7UUFDRixnQkFBZ0IsQ0FBQyxJQUFJLENBQUMsUUFBUSxDQUFDLFdBQVcsQ0FBQyxDQUFDLENBQUM7S0FDOUM7U0FBTTtRQUNMLE9BQU8sQ0FBQyxHQUFHLENBQUMsZUFBZSxDQUFDLENBQUM7UUFDN0IsTUFBTSxXQUFXLEdBQWtDLEtBQUssQ0FBQztRQUN6RCxnQkFBZ0IsQ0FBQyxJQUFJLENBQUMsUUFBUSxDQUFDLFdBQVcsQ0FBQyxDQUFDLENBQUM7S0FDOUM7SUFFRCxNQUFNLE9BQU8sQ0FBQyxHQUFHLENBQUMsZ0JBQWdCLENBQUMsQ0FBQztBQUN0QyxDQUFDO0FBRUQsTUFBTSxDQUFDLEtBQUssVUFBVSxRQUFRLENBQUMsVUFBeUM7SUFDdEUsaURBQWlEO0lBQ2pELE9BQU8sQ0FBQyxHQUFHLENBQUMsVUFBVSxFQUFFLElBQUksQ0FBQyxTQUFTLENBQUMsVUFBVSxDQUFDLENBQUMsQ0FBQztJQUVwRCxJQUFJLENBQUMsV0FBVyxFQUFFO1FBQ2hCLE9BQU87S0FDUjtJQUVELE1BQU0sU0FBUyxHQUFHLFdBQVcsQ0FBQyxHQUFHLENBQUMsS0FBSyxFQUFFLEVBQUUsWUFBWSxFQUFFLEVBQUUsRUFBRTtRQUMzRCxPQUFPLENBQUMsR0FBRyxDQUFDLHVCQUF1QixZQUFZLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQztRQUVyRCxJQUFJO1lBQ0YsTUFBTSx1QkFBdUIsR0FBRyxJQUFJLHVCQUF1QixDQUFDO2dCQUMxRCxZQUFZLEVBQUUsWUFBWSxDQUFDLENBQUM7Z0JBQzVCLElBQUksRUFBRSxJQUFJLENBQUMsU0FBUyxDQUFDO29CQUNuQixTQUFTLEVBQUUsSUFBSSxJQUFJLEVBQUUsQ0FBQyxXQUFXLEVBQUU7b0JBQ25DLFVBQVUsRUFBRSxVQUFVLENBQUMsVUFBVTtvQkFDakMsSUFBSSxFQUFFLFVBQVUsQ0FBQyxJQUFJO2lCQUN0QixDQUFRO2FBQ1YsQ0FBQyxDQUFDO1lBRUgsTUFBTSxrQkFBa0IsQ0FBQyxJQUFJLENBQUMsdUJBQXVCLENBQUMsQ0FBQztTQUN4RDtRQUFDLE9BQU8sQ0FBQyxFQUFFO1lBQ1YsT0FBTyxDQUFDLEtBQUssQ0FBQyw2QkFBNkIsWUFBWSxDQUFDLENBQUMsRUFBRSxFQUFFLENBQUMsQ0FBQyxDQUFDO1lBQ2hFLE9BQU8sQ0FBQyxLQUFLLENBQUMsZ0JBQWlCLENBQVMsQ0FBQyxVQUFVLEVBQUUsQ0FBQyxDQUFDO1lBQ3ZELElBQUssQ0FBUyxDQUFDLFNBQVMsQ0FBQyxjQUFjLEtBQUssR0FBRyxFQUFFO2dCQUMvQyxPQUFPLENBQUMsR0FBRyxDQUFDLG9DQUFvQyxZQUFZLEVBQUUsQ0FBQyxDQUFDO2dCQUVoRSxNQUFNLFlBQVksR0FBRyxJQUFJLGlCQUFpQixDQUFDO29CQUN6QyxTQUFTLEVBQUUsT0FBTyxDQUFDLEdBQUcsQ0FBQyxnQkFBZ0IsQ0FBQyxVQUFVLENBQUM7b0JBQ25ELEdBQUcsRUFBRSxFQUFFLFlBQVksRUFBRTtpQkFDdEIsQ0FBQyxDQUFDO2dCQUVILE1BQU0sR0FBRyxDQUFDLElBQUksQ0FBQyxZQUFZLENBQUMsQ0FBQzthQUM5QjtpQkFBTTtnQkFDTCxNQUFNLENBQUMsQ0FBQzthQUNUO1NBQ0Y7SUFDSCxDQUFDLENBQUMsQ0FBQztJQUVILE1BQU0sT0FBTyxDQUFDLEdBQUcsQ0FBQyxTQUFTLENBQUMsQ0FBQztJQUM3QixPQUFPLENBQUMsR0FBRyxDQUFDLHFCQUFxQixDQUFDLENBQUM7QUFDckMsQ0FBQztBQUVELE1BQU0sVUFBVSxlQUFlLENBQUMsVUFBa0I7SUFDaEQsSUFBSSxDQUFDLFVBQVUsRUFBRTtRQUNmLE1BQU0sSUFBSSxLQUFLLENBQUMsb0JBQW9CLENBQUMsQ0FBQztLQUN2QztJQUVELE9BQU8sVUFBVSxDQUFDLFNBQVMsQ0FBQyxDQUFDLEVBQUUsVUFBVSxDQUFDLE9BQU8sQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDO0FBQzFELENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge1xuICBBcGlHYXRld2F5TWFuYWdlbWVudEFwaSxcbiAgUG9zdFRvQ29ubmVjdGlvbkNvbW1hbmQsXG59IGZyb20gJ0Bhd3Mtc2RrL2NsaWVudC1hcGlnYXRld2F5bWFuYWdlbWVudGFwaSc7XG5pbXBvcnQge1xuICBBdHRyaWJ1dGVWYWx1ZSxcbiAgRGVsZXRlSXRlbUNvbW1hbmQsXG4gIER5bmFtb0RCQ2xpZW50LFxuICBTY2FuQ29tbWFuZCxcbn0gZnJvbSAnQGF3cy1zZGsvY2xpZW50LWR5bmFtb2RiJztcblxuaW1wb3J0IHsgdW5tYXJzaGFsbCB9IGZyb20gJ0Bhd3Mtc2RrL3V0aWwtZHluYW1vZGInO1xuaW1wb3J0IHtcbiAgRHluYW1vREJTdHJlYW1FdmVudCxcbiAgUzNFdmVudCxcbiAgU05TRXZlbnQsXG4gIEV2ZW50QnJpZGdlRXZlbnQsXG4gIFNRU0V2ZW50LFxufSBmcm9tICdhd3MtbGFtYmRhJztcbmltcG9ydCB7IGVudlZhcmlhYmxlTmFtZXMgfSBmcm9tICcuLi9zcmMvY29tbW9uL2VudlZhcmlhYmxlTmFtZXMnO1xuaW1wb3J0IHsgRHluYW1vREJTcHlFdmVudCB9IGZyb20gJy4vc3B5RXZlbnRzL0R5bmFtb0RCU3B5RXZlbnQnO1xuaW1wb3J0IHsgRXZlbnRCcmlkZ2VSdWxlU3B5RXZlbnQgfSBmcm9tICcuL3NweUV2ZW50cy9FdmVudEJyaWRnZVJ1bGVTcHlFdmVudCc7XG5pbXBvcnQgeyBFdmVudEJyaWRnZVNweUV2ZW50IH0gZnJvbSAnLi9zcHlFdmVudHMvRXZlbnRCcmlkZ2VTcHlFdmVudCc7XG5pbXBvcnQgeyBTM1NweUV2ZW50IH0gZnJvbSAnLi9zcHlFdmVudHMvUzNTcHlFdmVudCc7XG5pbXBvcnQgeyBTbnNTdWJzY3JpcHRpb25TcHlFdmVudCB9IGZyb20gJy4vc3B5RXZlbnRzL1Nuc1N1YnNjcmlwdGlvblNweUV2ZW50JztcbmltcG9ydCB7IFNuc1RvcGljU3B5RXZlbnQgfSBmcm9tICcuL3NweUV2ZW50cy9TbnNUb3BpY1NweUV2ZW50JztcbmltcG9ydCB7IFNweU1lc3NhZ2UgfSBmcm9tICcuL3NweUV2ZW50cy9TcHlNZXNzYWdlJztcbmltcG9ydCB7IFNxc1NweUV2ZW50IH0gZnJvbSAnLi9zcHlFdmVudHMvU3FzU3B5RXZlbnQnO1xuXG5jb25zdCBkZGIgPSBuZXcgRHluYW1vREJDbGllbnQoe1xuICByZWdpb246IHByb2Nlc3MuZW52LkFXU19SRUdJT04sXG59KTtcblxuY29uc3QgYXBpZ3dNYW5hZ2VtZW50QXBpID0gbmV3IEFwaUdhdGV3YXlNYW5hZ2VtZW50QXBpKHtcbiAgYXBpVmVyc2lvbjogJzIwMTgtMTEtMjknLFxuICBlbmRwb2ludDogcHJvY2Vzcy5lbnZbZW52VmFyaWFibGVOYW1lcy5XU19FTkRQT0lOVF0hLFxufSk7XG5cbmxldCBjb25uZWN0aW9uczogUmVjb3JkPHN0cmluZywgQXR0cmlidXRlVmFsdWU+W10gfCB1bmRlZmluZWQ7XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBwdWJsaXNoU3B5RXZlbnQoZXZlbnQ6IGFueSkge1xuICBjb25zb2xlLmxvZygnRVZFTlQnLCBKU09OLnN0cmluZ2lmeShldmVudCkpO1xuXG4gIGNvbnN0IG1hcHBpbmcgPSBKU09OLnBhcnNlKHByb2Nlc3MuZW52W2VudlZhcmlhYmxlTmFtZXMuSU5GUkFfTUFQUElOR10hKTtcbiAgY29uc29sZS5sb2coJ21hcHBpbmcnLCBKU09OLnN0cmluZ2lmeShtYXBwaW5nKSk7XG5cbiAgbGV0IGNvbm5lY3Rpb25EYXRhO1xuXG4gIGNvbnN0IHNjYW5QYXJhbXMgPSBuZXcgU2NhbkNvbW1hbmQoe1xuICAgIFRhYmxlTmFtZTogcHJvY2Vzcy5lbnZbZW52VmFyaWFibGVOYW1lcy5UQUJMRV9OQU1FXSBhcyBzdHJpbmcsXG4gICAgUHJvamVjdGlvbkV4cHJlc3Npb246ICdjb25uZWN0aW9uSWQnLFxuICB9KTtcblxuICBjb25uZWN0aW9uRGF0YSA9IGF3YWl0IGRkYi5zZW5kKHNjYW5QYXJhbXMpO1xuXG4gIGNvbm5lY3Rpb25zID0gY29ubmVjdGlvbkRhdGEuSXRlbXM7XG5cbiAgY29uc29sZS5sb2coXG4gICAgYGNvbm5lY3Rpb25zIGF0ICR7bmV3IERhdGUoKS50b0lTT1N0cmluZygpfWAsXG4gICAgSlNPTi5zdHJpbmdpZnkoY29ubmVjdGlvbnMpXG4gICk7XG5cbiAgY29uc3QgcG9zdERhdGFQcm9taXNlczogUHJvbWlzZTxhbnk+W10gPSBbXTtcblxuICBpZiAoZXZlbnQ/LlJlY29yZHMgJiYgZXZlbnQuUmVjb3Jkc1swXT8uU25zKSB7XG4gICAgY29uc29sZS5sb2coJyoqKiBTTlMgKioqJyk7XG4gICAgY29uc3QgZXZlbnRTbnMgPSBldmVudCBhcyBTTlNFdmVudDtcbiAgICBmb3IgKGNvbnN0IHJlY29yZCBvZiBldmVudFNucy5SZWNvcmRzKSB7XG4gICAgICBjb25zdCBzdWJzY3JpcHRpb25Bcm4gPSByZWNvcmQuRXZlbnRTdWJzY3JpcHRpb25Bcm47XG5cbiAgICAgIGxldCBzZXJ2aWNlS2V5OiBzdHJpbmc7XG4gICAgICBpZiAobWFwcGluZ1tzdWJzY3JpcHRpb25Bcm5dKSB7XG4gICAgICAgIC8vIHN1YnNjcmlwdGlvbiBldmVudCB0aGF0IGNvdWxkIGNvbnRhaW4gZmlsdGVyIGJhc2VkIG9uIGV4aXN0aW5nIHN1YnNjcmlwdGlvblxuICAgICAgICBzZXJ2aWNlS2V5ID0gbWFwcGluZ1tzdWJzY3JpcHRpb25Bcm5dO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgLy8gY2F0Y2ggYWxsIHN1YnNjcmlwdGlvblxuICAgICAgICBjb25zdCB0b3BpY0FybiA9IHJlY29yZC5TbnMuVG9waWNBcm47XG4gICAgICAgIHNlcnZpY2VLZXkgPSBtYXBwaW5nW3RvcGljQXJuXTtcbiAgICAgIH1cblxuICAgICAgbGV0IG1lc3NhZ2U6IHN0cmluZztcblxuICAgICAgdHJ5IHtcbiAgICAgICAgbWVzc2FnZSA9IEpTT04ucGFyc2UocmVjb3JkLlNucy5NZXNzYWdlKTtcbiAgICAgIH0gY2F0Y2gge1xuICAgICAgICBtZXNzYWdlID0gcmVjb3JkLlNucy5NZXNzYWdlO1xuICAgICAgfVxuXG4gICAgICBjb25zdCBzcHlFdmVudFR5cGUgPSBnZXRTcHlFdmVudFR5cGUoc2VydmljZUtleSkgYXNcbiAgICAgICAgfCAnRnVuY3Rpb25TbnNUb3BpYydcbiAgICAgICAgfCAnRnVuY3Rpb25TbnNTdWJzY3JpcHRpb24nO1xuXG4gICAgICBjb25zdCBkYXRhOiBTbnNUb3BpY1NweUV2ZW50IHwgU25zU3Vic2NyaXB0aW9uU3B5RXZlbnQgPSB7XG4gICAgICAgIHNweUV2ZW50VHlwZSxcbiAgICAgICAgbWVzc2FnZSxcbiAgICAgICAgc3ViamVjdDogcmVjb3JkLlNucy5TdWJqZWN0LFxuICAgICAgICB0aW1lc3RhbXA6IHJlY29yZC5TbnMuVGltZXN0YW1wLFxuICAgICAgICB0b3BpY0FybjogcmVjb3JkLlNucy5Ub3BpY0FybixcbiAgICAgICAgbWVzc2FnZUlkOiByZWNvcmQuU25zLk1lc3NhZ2VJZCxcbiAgICAgICAgbWVzc2FnZUF0dHJpYnV0ZXM6IHJlY29yZC5TbnMuTWVzc2FnZUF0dHJpYnV0ZXMsXG4gICAgICB9O1xuXG4gICAgICBjb25zdCBmbHVlbnRFdmVudDogT21pdDxTcHlNZXNzYWdlLCAndGltZXN0YW1wJz4gPSB7XG4gICAgICAgIGRhdGEsXG4gICAgICAgIHNlcnZpY2VLZXksXG4gICAgICB9O1xuICAgICAgcG9zdERhdGFQcm9taXNlcy5wdXNoKHBvc3REYXRhKGZsdWVudEV2ZW50KSk7XG4gICAgfVxuICB9IGVsc2UgaWYgKGV2ZW50Py5SZWNvcmRzICYmIGV2ZW50LlJlY29yZHNbMF0/LmV2ZW50U291cmNlID09PSAnYXdzOnNxcycpIHtcbiAgICBjb25zb2xlLmxvZygnKioqIFNRUyAqKionKTtcbiAgICBjb25zdCBldmVudFNxcyA9IGV2ZW50IGFzIFNRU0V2ZW50O1xuICAgIGZvciAoY29uc3QgcmVjb3JkIG9mIGV2ZW50U3FzLlJlY29yZHMpIHtcbiAgICAgIGNvbnN0IHN1YnNjcmlwdGlvbkFybiA9IHJlY29yZC5ldmVudFNvdXJjZUFSTjtcblxuICAgICAgY29uc3Qgc2VydmljZUtleSA9IG1hcHBpbmdbc3Vic2NyaXB0aW9uQXJuXTtcbiAgICAgIGxldCBib2R5OiBzdHJpbmc7XG5cbiAgICAgIHRyeSB7XG4gICAgICAgIGJvZHkgPSBKU09OLnBhcnNlKHJlY29yZC5ib2R5KTtcbiAgICAgIH0gY2F0Y2gge1xuICAgICAgICBib2R5ID0gcmVjb3JkLmJvZHk7XG4gICAgICB9XG5cbiAgICAgIGNvbnN0IGRhdGE6IFNxc1NweUV2ZW50ID0ge1xuICAgICAgICBzcHlFdmVudFR5cGU6ICdTcXMnLFxuICAgICAgICBib2R5LFxuICAgICAgICBtZXNzYWdlQXR0cmlidXRlczogcmVjb3JkLm1lc3NhZ2VBdHRyaWJ1dGVzLFxuICAgICAgfTtcblxuICAgICAgY29uc3QgZmx1ZW50RXZlbnQ6IE9taXQ8U3B5TWVzc2FnZSwgJ3RpbWVzdGFtcCc+ID0ge1xuICAgICAgICBkYXRhLFxuICAgICAgICBzZXJ2aWNlS2V5LFxuICAgICAgfTtcbiAgICAgIHBvc3REYXRhUHJvbWlzZXMucHVzaChwb3N0RGF0YShmbHVlbnRFdmVudCkpO1xuICAgIH1cbiAgfSBlbHNlIGlmIChldmVudD8uUmVjb3JkcyAmJiBldmVudC5SZWNvcmRzWzBdPy5zMykge1xuICAgIGNvbnNvbGUubG9nKCcqKiogUzMgKioqJyk7XG4gICAgY29uc3QgZXZlbnRTMyA9IGV2ZW50IGFzIFMzRXZlbnQ7XG4gICAgZm9yIChjb25zdCByZWNvcmQgb2YgZXZlbnRTMy5SZWNvcmRzKSB7XG4gICAgICBjb25zdCBidWNrZXRBcm4gPSByZWNvcmQuczMuYnVja2V0LmFybjtcblxuICAgICAgY29uc3QgZGF0YTogUzNTcHlFdmVudCA9IHtcbiAgICAgICAgc3B5RXZlbnRUeXBlOiAnUzMnLFxuICAgICAgICBldmVudE5hbWU6IHJlY29yZC5ldmVudE5hbWUsXG4gICAgICAgIGV2ZW50VGltZTogcmVjb3JkLmV2ZW50VGltZSxcbiAgICAgICAgYnVja2V0OiByZWNvcmQuczMuYnVja2V0Lm5hbWUsXG4gICAgICAgIGtleTogcmVjb3JkLnMzLm9iamVjdC5rZXksXG4gICAgICB9O1xuXG4gICAgICBjb25zdCBmbHVlbnRFdmVudDogT21pdDxTcHlNZXNzYWdlLCAndGltZXN0YW1wJz4gPSB7XG4gICAgICAgIGRhdGEsXG4gICAgICAgIHNlcnZpY2VLZXk6IG1hcHBpbmdbYnVja2V0QXJuXSxcbiAgICAgIH07XG4gICAgICBwb3N0RGF0YVByb21pc2VzLnB1c2gocG9zdERhdGEoZmx1ZW50RXZlbnQpKTtcbiAgICB9XG4gIH0gZWxzZSBpZiAoZXZlbnQuUmVjb3JkcyAmJiBldmVudC5SZWNvcmRzWzBdPy5keW5hbW9kYikge1xuICAgIGNvbnNvbGUubG9nKCcqKiogRFlOQU1PREIgKioqJyk7XG4gICAgY29uc3QgZXZlbnREeW5hbW9EQiA9IGV2ZW50IGFzIER5bmFtb0RCU3RyZWFtRXZlbnQ7XG4gICAgZm9yIChjb25zdCByZWNvcmQgb2YgZXZlbnREeW5hbW9EQi5SZWNvcmRzKSB7XG4gICAgICBsZXQgYXJuID0gcmVjb3JkLmV2ZW50U291cmNlQVJOITtcbiAgICAgIGFybiA9IGFybi5zdWJzdHJpbmcoMCwgYXJuLmluZGV4T2YoJy9zdHJlYW0vJykpO1xuXG4gICAgICBjb25zdCBkYXRhOiBEeW5hbW9EQlNweUV2ZW50ID0ge1xuICAgICAgICBzcHlFdmVudFR5cGU6ICdEeW5hbW9EQicsXG4gICAgICAgIGV2ZW50TmFtZTogcmVjb3JkLmV2ZW50TmFtZSxcbiAgICAgICAgbmV3SW1hZ2U6IHVubWFyc2hhbGwocmVjb3JkLmR5bmFtb2RiPy5OZXdJbWFnZSBhcyBhbnkpLFxuICAgICAgICBrZXlzOiB1bm1hcnNoYWxsKHJlY29yZC5keW5hbW9kYj8uS2V5cyBhcyBhbnkpLFxuICAgICAgICBvbGRJbWFnZTogcmVjb3JkLmR5bmFtb2RiPy5PbGRJbWFnZVxuICAgICAgICAgID8gdW5tYXJzaGFsbChyZWNvcmQuZHluYW1vZGI/Lk9sZEltYWdlIGFzIGFueSlcbiAgICAgICAgICA6IHVuZGVmaW5lZCxcbiAgICAgIH07XG5cbiAgICAgIGNvbnN0IGZsdWVudEV2ZW50OiBPbWl0PFNweU1lc3NhZ2UsICd0aW1lc3RhbXAnPiA9IHtcbiAgICAgICAgZGF0YSxcbiAgICAgICAgc2VydmljZUtleTogbWFwcGluZ1thcm5dLFxuICAgICAgfTtcbiAgICAgIHBvc3REYXRhUHJvbWlzZXMucHVzaChwb3N0RGF0YShmbHVlbnRFdmVudCkpO1xuICAgIH1cbiAgfSBlbHNlIGlmIChcbiAgICBldmVudC5kZXRhaWwgJiZcbiAgICBldmVudFsnZGV0YWlsLXR5cGUnXSAmJlxuICAgIGV2ZW50LnZlcnNpb24gJiZcbiAgICBldmVudC5zb3VyY2VcbiAgKSB7XG4gICAgY29uc29sZS5sb2coJyoqKiBFdmVudEJyaWRnZSAqKionKTtcbiAgICBjb25zdCBldmVudEViID0gZXZlbnQgYXMgRXZlbnRCcmlkZ2VFdmVudDxhbnksIGFueT47XG5cbiAgICBjb25zdCBzZXJ2aWNlS2V5ID0gbWFwcGluZy5ldmVudEJyaWRnZTsgLy8gdGhlIGlzIG5ldyBsYW1iZGEgZm9yIGVhY2ggc3Vic2NyaXB0aW9uXG5cbiAgICBjb25zdCBzcHlFdmVudFR5cGUgPSBnZXRTcHlFdmVudFR5cGUoc2VydmljZUtleSkgYXNcbiAgICAgIHwgJ0V2ZW50QnJpZGdlJ1xuICAgICAgfCAnRXZlbnRCcmlkZ2VSdWxlJztcblxuICAgIGNvbnN0IG1lc3NhZ2UgPSBldmVudEViLmRldGFpbDtcblxuICAgIGNvbnN0IGRhdGE6IEV2ZW50QnJpZGdlU3B5RXZlbnQgfCBFdmVudEJyaWRnZVJ1bGVTcHlFdmVudCA9IHtcbiAgICAgIHNweUV2ZW50VHlwZSxcbiAgICAgIGRldGFpbDogbWVzc2FnZSxcbiAgICAgIGRldGFpbFR5cGU6IGV2ZW50RWJbJ2RldGFpbC10eXBlJ10sXG4gICAgICBzb3VyY2U6IGV2ZW50RWIuc291cmNlLFxuICAgICAgdGltZTogZXZlbnRFYi50aW1lLFxuICAgICAgYWNjb3VudDogZXZlbnRFYi5hY2NvdW50LFxuICAgIH07XG5cbiAgICBjb25zdCBmbHVlbnRFdmVudDogT21pdDxTcHlNZXNzYWdlLCAndGltZXN0YW1wJz4gPSB7XG4gICAgICBkYXRhLFxuICAgICAgc2VydmljZUtleSxcbiAgICB9O1xuICAgIHBvc3REYXRhUHJvbWlzZXMucHVzaChwb3N0RGF0YShmbHVlbnRFdmVudCkpO1xuICB9IGVsc2Uge1xuICAgIGNvbnNvbGUubG9nKCcqKiogT1RIRVIgKioqJyk7XG4gICAgY29uc3QgZmx1ZW50RXZlbnQ6IE9taXQ8U3B5TWVzc2FnZSwgJ3RpbWVzdGFtcCc+ID0gZXZlbnQ7XG4gICAgcG9zdERhdGFQcm9taXNlcy5wdXNoKHBvc3REYXRhKGZsdWVudEV2ZW50KSk7XG4gIH1cblxuICBhd2FpdCBQcm9taXNlLmFsbChwb3N0RGF0YVByb21pc2VzKTtcbn1cblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIHBvc3REYXRhKHNweU1lc3NhZ2U6IE9taXQ8U3B5TWVzc2FnZSwgJ3RpbWVzdGFtcCc+KSB7XG4gIC8vIGNvbnN0IHBvc3REYXRhID0gSlNPTi5wYXJzZShldmVudC5ib2R5ISkuZGF0YTtcbiAgY29uc29sZS5sb2coJ3Bvc3REYXRhJywgSlNPTi5zdHJpbmdpZnkoc3B5TWVzc2FnZSkpO1xuXG4gIGlmICghY29ubmVjdGlvbnMpIHtcbiAgICByZXR1cm47XG4gIH1cblxuICBjb25zdCBwb3N0Q2FsbHMgPSBjb25uZWN0aW9ucy5tYXAoYXN5bmMgKHsgY29ubmVjdGlvbklkIH0pID0+IHtcbiAgICBjb25zb2xlLmxvZyhgU2VuZGluZyBtZXNzYWdlIHRvOiAke2Nvbm5lY3Rpb25JZC5TfWApO1xuXG4gICAgdHJ5IHtcbiAgICAgIGNvbnN0IHBvc3RUb0Nvbm5lY3Rpb25Db21tYW5kID0gbmV3IFBvc3RUb0Nvbm5lY3Rpb25Db21tYW5kKHtcbiAgICAgICAgQ29ubmVjdGlvbklkOiBjb25uZWN0aW9uSWQuUyxcbiAgICAgICAgRGF0YTogSlNPTi5zdHJpbmdpZnkoe1xuICAgICAgICAgIHRpbWVzdGFtcDogbmV3IERhdGUoKS50b0lTT1N0cmluZygpLFxuICAgICAgICAgIHNlcnZpY2VLZXk6IHNweU1lc3NhZ2Uuc2VydmljZUtleSxcbiAgICAgICAgICBkYXRhOiBzcHlNZXNzYWdlLmRhdGEsXG4gICAgICAgIH0pIGFzIGFueSxcbiAgICAgIH0pO1xuXG4gICAgICBhd2FpdCBhcGlnd01hbmFnZW1lbnRBcGkuc2VuZChwb3N0VG9Db25uZWN0aW9uQ29tbWFuZCk7XG4gICAgfSBjYXRjaCAoZSkge1xuICAgICAgY29uc29sZS5lcnJvcihgRmFpbHMgc2VuZGluZyBtZXNzYWdlIHRvOiAke2Nvbm5lY3Rpb25JZC5TfWAsIGUpO1xuICAgICAgY29uc29sZS5lcnJvcihgU3RhdHVzIGNvZGU6ICR7KGUgYXMgYW55KS5zdGF0dXNDb2RlfWApO1xuICAgICAgaWYgKChlIGFzIGFueSkuJG1ldGFkYXRhLmh0dHBTdGF0dXNDb2RlID09PSA0MTApIHtcbiAgICAgICAgY29uc29sZS5sb2coYEZvdW5kIHN0YWxlIGNvbm5lY3Rpb24sIGRlbGV0aW5nICR7Y29ubmVjdGlvbklkfWApO1xuXG4gICAgICAgIGNvbnN0IGRlbGV0ZVBhcmFtcyA9IG5ldyBEZWxldGVJdGVtQ29tbWFuZCh7XG4gICAgICAgICAgVGFibGVOYW1lOiBwcm9jZXNzLmVudltlbnZWYXJpYWJsZU5hbWVzLlRBQkxFX05BTUVdLFxuICAgICAgICAgIEtleTogeyBjb25uZWN0aW9uSWQgfSxcbiAgICAgICAgfSk7XG5cbiAgICAgICAgYXdhaXQgZGRiLnNlbmQoZGVsZXRlUGFyYW1zKTtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIHRocm93IGU7XG4gICAgICB9XG4gICAgfVxuICB9KTtcblxuICBhd2FpdCBQcm9taXNlLmFsbChwb3N0Q2FsbHMpO1xuICBjb25zb2xlLmxvZygnU2VuZCBtZXNzYWdlIGZpbmlzaCcpO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gZ2V0U3B5RXZlbnRUeXBlKHNlcnZpY2VLZXk6IHN0cmluZykge1xuICBpZiAoIXNlcnZpY2VLZXkpIHtcbiAgICB0aHJvdyBuZXcgRXJyb3IoJ01pc3Npbmcgc2VydmljZUtleScpO1xuICB9XG5cbiAgcmV0dXJuIHNlcnZpY2VLZXkuc3Vic3RyaW5nKDAsIHNlcnZpY2VLZXkuaW5kZXhPZignIycpKTtcbn1cbiJdfQ==