@mecanizou/telemetry-hub 1.0.0 → 1.0.2
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/.github/workflows/pull_request.yml +32 -32
- package/.github/workflows/release.yml +129 -129
- package/.prettierignore +4 -4
- package/CHANGELOG.md +14 -0
- package/DOCS_GUIDE.md +151 -0
- package/README.md +248 -0
- package/dist/core/__tests__/logger-types.test.d.ts +1 -0
- package/dist/core/__tests__/logger-types.test.js +325 -0
- package/dist/core/__tests__/logger.test.d.ts +1 -0
- package/dist/core/__tests__/logger.test.js +337 -0
- package/dist/core/__tests__/tracer.test.d.ts +1 -0
- package/dist/core/__tests__/tracer.test.js +330 -0
- package/dist/core/index.d.ts +4 -0
- package/dist/core/index.js +8 -0
- package/dist/core/logger-types.d.ts +43 -0
- package/dist/core/logger-types.js +3 -0
- package/dist/core/logger.d.ts +13 -0
- package/dist/core/logger.js +123 -0
- package/dist/core/tracer-types.d.ts +50 -0
- package/dist/core/tracer-types.js +3 -0
- package/dist/core/tracer.d.ts +10 -0
- package/dist/core/tracer.js +114 -0
- package/dist/index.d.ts +3 -1
- package/dist/index.js +4 -2
- package/dist/sst/__tests__/telemetry.test.d.ts +1 -0
- package/dist/sst/__tests__/telemetry.test.js +138 -0
- package/dist/sst/index.d.ts +1 -0
- package/dist/sst/index.js +18 -0
- package/dist/sst/middy/index.d.ts +1 -0
- package/dist/sst/middy/index.js +18 -0
- package/dist/sst/middy/middleware.d.ts +5 -0
- package/dist/sst/middy/middleware.js +157 -0
- package/dist/sst/telemetry.d.ts +4 -0
- package/dist/sst/telemetry.js +121 -0
- package/dist/telemetry/core/__tests__/logger-types.test.d.ts +1 -0
- package/dist/telemetry/core/__tests__/logger-types.test.js +325 -0
- package/dist/telemetry/core/__tests__/logger.test.d.ts +1 -0
- package/dist/telemetry/core/__tests__/logger.test.js +337 -0
- package/dist/telemetry/core/__tests__/tracer.test.d.ts +1 -0
- package/dist/telemetry/core/__tests__/tracer.test.js +330 -0
- package/dist/telemetry/core/index.d.ts +4 -0
- package/dist/telemetry/core/index.js +8 -0
- package/dist/telemetry/core/logger-types.d.ts +43 -0
- package/dist/telemetry/core/logger-types.js +3 -0
- package/dist/telemetry/core/logger.d.ts +13 -0
- package/dist/telemetry/core/logger.js +123 -0
- package/dist/telemetry/core/tracer-types.d.ts +50 -0
- package/dist/telemetry/core/tracer-types.js +3 -0
- package/dist/telemetry/core/tracer.d.ts +10 -0
- package/dist/telemetry/core/tracer.js +114 -0
- package/dist/telemetry/index.d.ts +3 -0
- package/dist/telemetry/index.js +20 -0
- package/dist/telemetry/sst/__tests__/telemetry.test.d.ts +1 -0
- package/dist/telemetry/sst/__tests__/telemetry.test.js +138 -0
- package/dist/telemetry/sst/index.d.ts +1 -0
- package/dist/telemetry/sst/index.js +18 -0
- package/dist/telemetry/sst/middy/index.d.ts +1 -0
- package/dist/telemetry/sst/middy/index.js +18 -0
- package/dist/telemetry/sst/middy/middleware.d.ts +5 -0
- package/dist/telemetry/sst/middy/middleware.js +157 -0
- package/dist/telemetry/sst/telemetry.d.ts +4 -0
- package/dist/telemetry/sst/telemetry.js +121 -0
- package/dist/telemetry/tsed/__tests__/config.test.d.ts +1 -0
- package/dist/telemetry/tsed/__tests__/config.test.js +146 -0
- package/dist/telemetry/tsed/__tests__/service.test.d.ts +1 -0
- package/dist/telemetry/tsed/__tests__/service.test.js +63 -0
- package/dist/telemetry/tsed/config.d.ts +26 -0
- package/dist/telemetry/tsed/config.js +166 -0
- package/dist/telemetry/tsed/index.d.ts +4 -0
- package/dist/telemetry/tsed/index.js +21 -0
- package/dist/telemetry/tsed/log-telemetry.d.ts +1 -0
- package/dist/telemetry/tsed/log-telemetry.js +196 -0
- package/dist/telemetry/tsed/service.d.ts +26 -0
- package/dist/telemetry/tsed/service.js +150 -0
- package/dist/telemetry/tsed/sync-log-record-processor.d.ts +11 -0
- package/dist/telemetry/tsed/sync-log-record-processor.js +74 -0
- package/dist/tsed/__tests__/config.test.d.ts +1 -0
- package/dist/tsed/__tests__/config.test.js +146 -0
- package/dist/tsed/__tests__/service.test.d.ts +1 -0
- package/dist/tsed/__tests__/service.test.js +63 -0
- package/dist/tsed/config.d.ts +26 -0
- package/dist/tsed/config.js +166 -0
- package/dist/tsed/index.d.ts +4 -0
- package/dist/tsed/index.js +21 -0
- package/dist/tsed/log-telemetry.d.ts +1 -0
- package/dist/tsed/log-telemetry.js +196 -0
- package/dist/tsed/service.d.ts +26 -0
- package/dist/tsed/service.js +150 -0
- package/dist/tsed/sync-log-record-processor.d.ts +11 -0
- package/dist/tsed/sync-log-record-processor.js +74 -0
- package/package.json +72 -56
- package/release.config.js +23 -23
- package/vitest.config.ts +22 -0
- package/dist/check-if-is-working.d.ts +0 -1
- package/dist/check-if-is-working.js +0 -8
|
@@ -0,0 +1,196 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
|
|
3
|
+
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
|
|
4
|
+
return new (P || (P = Promise))(function (resolve, reject) {
|
|
5
|
+
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
|
6
|
+
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
|
7
|
+
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
|
|
8
|
+
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
9
|
+
});
|
|
10
|
+
};
|
|
11
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
12
|
+
exports.TsedLogTelemetry = TsedLogTelemetry;
|
|
13
|
+
const service_1 = require("./service");
|
|
14
|
+
const api_1 = require("@opentelemetry/api");
|
|
15
|
+
const perf_hooks_1 = require("perf_hooks");
|
|
16
|
+
function TsedLogTelemetry() {
|
|
17
|
+
return function (_target, _propertyKey, descriptor) {
|
|
18
|
+
const originalMethod = descriptor.value;
|
|
19
|
+
const methodName = _propertyKey;
|
|
20
|
+
descriptor.value = function (...args) {
|
|
21
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
22
|
+
var _a, _b, _c, _d, _e, _f, _g, _h, _j, _k, _l, _m;
|
|
23
|
+
const startTime = perf_hooks_1.performance.now();
|
|
24
|
+
const $ctx = args.find((arg) => (arg === null || arg === void 0 ? void 0 : arg.request) && (arg === null || arg === void 0 ? void 0 : arg.response));
|
|
25
|
+
let controllerName = 'UnknownController';
|
|
26
|
+
if (((_a = this === null || this === void 0 ? void 0 : this.constructor) === null || _a === void 0 ? void 0 : _a.name) && this.constructor.name !== 'Object') {
|
|
27
|
+
controllerName = this.constructor.name;
|
|
28
|
+
}
|
|
29
|
+
if (!$ctx) {
|
|
30
|
+
console.warn('[TsedLogTelemetry] Context not found, executing without telemetry');
|
|
31
|
+
return yield originalMethod.apply(this, args);
|
|
32
|
+
}
|
|
33
|
+
let telemetryService;
|
|
34
|
+
try {
|
|
35
|
+
if ($ctx.injector) {
|
|
36
|
+
telemetryService = $ctx.injector.get(service_1.TsedTelemetryService);
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
catch (error) {
|
|
40
|
+
(_b = $ctx.logger) === null || _b === void 0 ? void 0 : _b.warn({
|
|
41
|
+
'TsedLogTelemetry - Failed to get TelemetryService from injector': error,
|
|
42
|
+
});
|
|
43
|
+
}
|
|
44
|
+
if (!telemetryService) {
|
|
45
|
+
(_c = $ctx.logger) === null || _c === void 0 ? void 0 : _c.warn('[TsedLogTelemetry] TelemetryService not available, executing without telemetry');
|
|
46
|
+
return yield originalMethod.apply(this, args);
|
|
47
|
+
}
|
|
48
|
+
const request = $ctx.request;
|
|
49
|
+
const serviceName = process.env.SERVICE_NAME || 'tsed-service';
|
|
50
|
+
let stage = process.env.STAGE || 'development';
|
|
51
|
+
if (stage === 'prod')
|
|
52
|
+
stage = 'production';
|
|
53
|
+
const queryStringParameters = request.query || {};
|
|
54
|
+
const origin = queryStringParameters.origin || 'web';
|
|
55
|
+
const tracer = telemetryService.getTracer();
|
|
56
|
+
let spanResult = null;
|
|
57
|
+
if (tracer) {
|
|
58
|
+
spanResult = tracer.startSpan({
|
|
59
|
+
spanName: `${controllerName}.${methodName}`,
|
|
60
|
+
serviceName,
|
|
61
|
+
environment: stage,
|
|
62
|
+
execution: {
|
|
63
|
+
controller: controllerName,
|
|
64
|
+
controllerMethod: methodName,
|
|
65
|
+
requestId: $ctx.id,
|
|
66
|
+
awsRequestId: (_d = $ctx.context) === null || _d === void 0 ? void 0 : _d.awsRequestId,
|
|
67
|
+
origin,
|
|
68
|
+
},
|
|
69
|
+
http: {
|
|
70
|
+
method: request.method,
|
|
71
|
+
url: request.url,
|
|
72
|
+
endpoint: request.url,
|
|
73
|
+
},
|
|
74
|
+
});
|
|
75
|
+
}
|
|
76
|
+
const meter = api_1.metrics.getMeter(origin);
|
|
77
|
+
const executionTimeHistogram = meter.createHistogram('tsed_execution_duration', {
|
|
78
|
+
description: 'Tempo total de execução do método Tsed em ms',
|
|
79
|
+
unit: 'ms',
|
|
80
|
+
valueType: api_1.ValueType.DOUBLE,
|
|
81
|
+
});
|
|
82
|
+
const successCounter = meter.createCounter('tsed_successful_requests', {
|
|
83
|
+
description: 'Total de requisições bem-sucedidas',
|
|
84
|
+
});
|
|
85
|
+
const failureCounter = meter.createCounter('tsed_failed_requests', {
|
|
86
|
+
description: 'Total de requisições com falha',
|
|
87
|
+
});
|
|
88
|
+
try {
|
|
89
|
+
const result = yield originalMethod.apply(this, args);
|
|
90
|
+
const durationMs = perf_hooks_1.performance.now() - startTime;
|
|
91
|
+
if (spanResult) {
|
|
92
|
+
spanResult.setSuccess(true);
|
|
93
|
+
}
|
|
94
|
+
successCounter.add(1, {
|
|
95
|
+
origin,
|
|
96
|
+
controller: controllerName,
|
|
97
|
+
method: methodName,
|
|
98
|
+
'service.name': serviceName,
|
|
99
|
+
'deployment.environment.name': stage,
|
|
100
|
+
});
|
|
101
|
+
executionTimeHistogram.record(durationMs, {
|
|
102
|
+
origin,
|
|
103
|
+
status: 'success',
|
|
104
|
+
controller: controllerName,
|
|
105
|
+
method: methodName,
|
|
106
|
+
'service.name': serviceName,
|
|
107
|
+
'deployment.environment.name': stage,
|
|
108
|
+
});
|
|
109
|
+
const logger = telemetryService.getLogger();
|
|
110
|
+
if (logger) {
|
|
111
|
+
yield logger.logInfo({
|
|
112
|
+
message: 'Request completed successfully',
|
|
113
|
+
serviceName,
|
|
114
|
+
environment: stage,
|
|
115
|
+
execution: {
|
|
116
|
+
controller: controllerName,
|
|
117
|
+
controllerMethod: methodName,
|
|
118
|
+
requestId: $ctx.id,
|
|
119
|
+
awsRequestId: (_e = $ctx.context) === null || _e === void 0 ? void 0 : _e.awsRequestId,
|
|
120
|
+
origin,
|
|
121
|
+
},
|
|
122
|
+
http: {
|
|
123
|
+
method: request.method,
|
|
124
|
+
url: request.url,
|
|
125
|
+
endpoint: request.url,
|
|
126
|
+
statusCode: ((_f = $ctx.response) === null || _f === void 0 ? void 0 : _f.statusCode) || 200,
|
|
127
|
+
},
|
|
128
|
+
performance: {
|
|
129
|
+
durationMs,
|
|
130
|
+
success: true,
|
|
131
|
+
},
|
|
132
|
+
});
|
|
133
|
+
}
|
|
134
|
+
if (spanResult) {
|
|
135
|
+
spanResult.end();
|
|
136
|
+
}
|
|
137
|
+
yield telemetryService.telemetryProvider.forceFlush();
|
|
138
|
+
return result;
|
|
139
|
+
}
|
|
140
|
+
catch (error) {
|
|
141
|
+
const durationMs = perf_hooks_1.performance.now() - startTime;
|
|
142
|
+
(_g = $ctx.logger) === null || _g === void 0 ? void 0 : _g.info('Error caught, sending to telemetry');
|
|
143
|
+
if (spanResult) {
|
|
144
|
+
spanResult.setError(error);
|
|
145
|
+
}
|
|
146
|
+
failureCounter.add(1, {
|
|
147
|
+
origin,
|
|
148
|
+
error_type: ((_h = error === null || error === void 0 ? void 0 : error.constructor) === null || _h === void 0 ? void 0 : _h.name) || 'UnknownError',
|
|
149
|
+
controller: controllerName,
|
|
150
|
+
method: methodName,
|
|
151
|
+
'service.name': serviceName,
|
|
152
|
+
'deployment.environment.name': stage,
|
|
153
|
+
});
|
|
154
|
+
executionTimeHistogram.record(durationMs, {
|
|
155
|
+
origin,
|
|
156
|
+
status: 'error',
|
|
157
|
+
controller: controllerName,
|
|
158
|
+
method: methodName,
|
|
159
|
+
'service.name': serviceName,
|
|
160
|
+
'deployment.environment.name': stage,
|
|
161
|
+
});
|
|
162
|
+
let userInfo = {};
|
|
163
|
+
try {
|
|
164
|
+
if (typeof LoggedUserIdentifier !== 'undefined') {
|
|
165
|
+
const { loggedUser } = LoggedUserIdentifier.use($ctx);
|
|
166
|
+
userInfo = {
|
|
167
|
+
accountUserUid: loggedUser === null || loggedUser === void 0 ? void 0 : loggedUser.uid,
|
|
168
|
+
accountUid: (_j = loggedUser === null || loggedUser === void 0 ? void 0 : loggedUser.account) === null || _j === void 0 ? void 0 : _j.uid,
|
|
169
|
+
applicationUid: (_k = loggedUser === null || loggedUser === void 0 ? void 0 : loggedUser.application) === null || _k === void 0 ? void 0 : _k.uid,
|
|
170
|
+
};
|
|
171
|
+
}
|
|
172
|
+
}
|
|
173
|
+
catch (e) {
|
|
174
|
+
}
|
|
175
|
+
yield telemetryService.logError(Object.assign(Object.assign({ error: error, context: {
|
|
176
|
+
statusCode: error.status || 500,
|
|
177
|
+
url: request.url,
|
|
178
|
+
headers: request.headers,
|
|
179
|
+
body: request.body,
|
|
180
|
+
params: request.params,
|
|
181
|
+
query: request.query,
|
|
182
|
+
errorType: (_l = error === null || error === void 0 ? void 0 : error.constructor) === null || _l === void 0 ? void 0 : _l.name,
|
|
183
|
+
} }, userInfo), { requestId: $ctx.id, awsRequestId: (_m = $ctx.context) === null || _m === void 0 ? void 0 : _m.awsRequestId, endpoint: request.url, method: request.method, controller: controllerName, controllerMethod: methodName }));
|
|
184
|
+
if (spanResult) {
|
|
185
|
+
spanResult.end();
|
|
186
|
+
}
|
|
187
|
+
yield telemetryService.telemetryProvider.forceFlush();
|
|
188
|
+
$ctx.logger.info('Error sent to telemetry');
|
|
189
|
+
throw error;
|
|
190
|
+
}
|
|
191
|
+
});
|
|
192
|
+
};
|
|
193
|
+
return descriptor;
|
|
194
|
+
};
|
|
195
|
+
}
|
|
196
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"log-telemetry.js","sourceRoot":"","sources":["../../src/tsed/log-telemetry.ts"],"names":[],"mappings":"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","sourcesContent":["/**\r\n * Decorator que captura erros automaticamente, cria spans e envia para telemetria\r\n * Alinhado com a implementação do SST middleware para garantir consistência\r\n *\r\n * Uso no controller:\r\n *\r\n * @TsedLogTelemetry()\r\n * async getCheckout(@Context() $ctx: ServerlessContext, ...) {\r\n *   return await this.service.execute(params);\r\n * }\r\n */\r\nimport { ServerlessContext } from '@tsed/platform-serverless';\r\nimport { TsedTelemetryService } from './service';\r\nimport { metrics, ValueType } from '@opentelemetry/api';\r\nimport { performance } from 'perf_hooks';\r\n\r\nexport function TsedLogTelemetry() {\r\n  return function (\r\n    _target: any,\r\n    _propertyKey: string,\r\n    descriptor: PropertyDescriptor\r\n  ) {\r\n    const originalMethod = descriptor.value;\r\n    const methodName = _propertyKey;\r\n\r\n    descriptor.value = async function (this: any, ...args: any[]) {\r\n      const startTime = performance.now();\r\n\r\n      // Encontrar o contexto\r\n      const $ctx: ServerlessContext = args.find(\r\n        (arg) => arg?.request && arg?.response\r\n      );\r\n\r\n      // Capturar o nome do controller\r\n      let controllerName = 'UnknownController';\r\n      if (this?.constructor?.name && this.constructor.name !== 'Object') {\r\n        controllerName = this.constructor.name;\r\n      }\r\n\r\n      if (!$ctx) {\r\n        console.warn(\r\n          '[TsedLogTelemetry] Context not found, executing without telemetry'\r\n        );\r\n        return await originalMethod.apply(this, args);\r\n      }\r\n\r\n      // Buscar o TelemetryService do injector\r\n      let telemetryService: TsedTelemetryService | undefined;\r\n      try {\r\n        if ($ctx.injector) {\r\n          telemetryService = $ctx.injector.get(\r\n            TsedTelemetryService\r\n          ) as TsedTelemetryService;\r\n        }\r\n      } catch (error) {\r\n        $ctx.logger?.warn({\r\n          'TsedLogTelemetry - Failed to get TelemetryService from injector':\r\n            error,\r\n        });\r\n      }\r\n\r\n      if (!telemetryService) {\r\n        $ctx.logger?.warn(\r\n          '[TsedLogTelemetry] TelemetryService not available, executing without telemetry'\r\n        );\r\n        return await originalMethod.apply(this, args);\r\n      }\r\n\r\n      // Extrair informações básicas\r\n      const request = $ctx.request;\r\n      const serviceName = process.env.SERVICE_NAME || 'tsed-service';\r\n      let stage = process.env.STAGE || 'development';\r\n      if (stage === 'prod') stage = 'production';\r\n\r\n      const queryStringParameters = request.query || {};\r\n      const origin = queryStringParameters.origin || 'web';\r\n\r\n      // Criar span usando StandardTracer (similar ao SST middleware)\r\n      const tracer = (telemetryService as any).getTracer();\r\n      let spanResult: any = null;\r\n\r\n      if (tracer) {\r\n        spanResult = tracer.startSpan({\r\n          spanName: `${controllerName}.${methodName}`,\r\n          serviceName,\r\n          environment: stage,\r\n          execution: {\r\n            controller: controllerName,\r\n            controllerMethod: methodName,\r\n            requestId: $ctx.id,\r\n            awsRequestId: $ctx.context?.awsRequestId,\r\n            origin,\r\n          },\r\n          http: {\r\n            method: request.method,\r\n            url: request.url,\r\n            endpoint: request.url,\r\n          },\r\n        });\r\n      }\r\n\r\n      // Criar métricas (similar ao SST middleware)\r\n      const meter = metrics.getMeter(origin);\r\n      const executionTimeHistogram = meter.createHistogram(\r\n        'tsed_execution_duration',\r\n        {\r\n          description: 'Tempo total de execução do método Tsed em ms',\r\n          unit: 'ms',\r\n          valueType: ValueType.DOUBLE,\r\n        }\r\n      );\r\n      const successCounter = meter.createCounter('tsed_successful_requests', {\r\n        description: 'Total de requisições bem-sucedidas',\r\n      });\r\n      const failureCounter = meter.createCounter('tsed_failed_requests', {\r\n        description: 'Total de requisições com falha',\r\n      });\r\n\r\n      try {\r\n        const result = await originalMethod.apply(this, args);\r\n\r\n        // SUCESSO - Similar ao 'after' do SST middleware\r\n        const durationMs = performance.now() - startTime;\r\n\r\n        // Marcar span como sucesso\r\n        if (spanResult) {\r\n          spanResult.setSuccess(true);\r\n        }\r\n\r\n        // Métricas de sucesso\r\n        successCounter.add(1, {\r\n          origin,\r\n          controller: controllerName,\r\n          method: methodName,\r\n          'service.name': serviceName,\r\n          'deployment.environment.name': stage,\r\n        });\r\n\r\n        executionTimeHistogram.record(durationMs, {\r\n          origin,\r\n          status: 'success',\r\n          controller: controllerName,\r\n          method: methodName,\r\n          'service.name': serviceName,\r\n          'deployment.environment.name': stage,\r\n        });\r\n\r\n        // Log de sucesso (similar ao SST)\r\n        const logger = (telemetryService as any).getLogger();\r\n        if (logger) {\r\n          await logger.logInfo({\r\n            message: 'Request completed successfully',\r\n            serviceName,\r\n            environment: stage,\r\n            execution: {\r\n              controller: controllerName,\r\n              controllerMethod: methodName,\r\n              requestId: $ctx.id,\r\n              awsRequestId: $ctx.context?.awsRequestId,\r\n              origin,\r\n            },\r\n            http: {\r\n              method: request.method,\r\n              url: request.url,\r\n              endpoint: request.url,\r\n              statusCode: $ctx.response?.statusCode || 200,\r\n            },\r\n            performance: {\r\n              durationMs,\r\n              success: true,\r\n            },\r\n          });\r\n        }\r\n\r\n        // Finalizar span e force flush\r\n        if (spanResult) {\r\n          spanResult.end();\r\n        }\r\n        await (telemetryService as any).telemetryProvider.forceFlush();\r\n\r\n        return result;\r\n      } catch (error: any) {\r\n        // ERRO - Similar ao 'onError' do SST middleware\r\n        const durationMs = performance.now() - startTime;\r\n\r\n        $ctx.logger?.info('Error caught, sending to telemetry');\r\n\r\n        // Marcar span como erro\r\n        if (spanResult) {\r\n          spanResult.setError(error);\r\n        }\r\n\r\n        // Métricas de erro\r\n        failureCounter.add(1, {\r\n          origin,\r\n          error_type: error?.constructor?.name || 'UnknownError',\r\n          controller: controllerName,\r\n          method: methodName,\r\n          'service.name': serviceName,\r\n          'deployment.environment.name': stage,\r\n        });\r\n\r\n        executionTimeHistogram.record(durationMs, {\r\n          origin,\r\n          status: 'error',\r\n          controller: controllerName,\r\n          method: methodName,\r\n          'service.name': serviceName,\r\n          'deployment.environment.name': stage,\r\n        });\r\n\r\n        // Extrair informações do usuário\r\n        let userInfo: any = {};\r\n        try {\r\n          // @ts-ignore - LoggedUserIdentifier é específico da aplicação\r\n          if (typeof LoggedUserIdentifier !== 'undefined') {\r\n            // @ts-ignore\r\n            const { loggedUser } = LoggedUserIdentifier.use($ctx);\r\n            userInfo = {\r\n              accountUserUid: loggedUser?.uid,\r\n              accountUid: loggedUser?.account?.uid,\r\n              applicationUid: loggedUser?.application?.uid,\r\n            };\r\n          }\r\n        } catch (e) {\r\n          // LoggedUserIdentifier não está disponível\r\n        }\r\n\r\n        // Log de erro (similar ao SST)\r\n        await telemetryService.logError({\r\n          error: error,\r\n          context: {\r\n            statusCode: error.status || 500,\r\n            url: request.url,\r\n            headers: request.headers,\r\n            body: request.body,\r\n            params: request.params,\r\n            query: request.query,\r\n            errorType: error?.constructor?.name,\r\n          },\r\n          ...userInfo,\r\n          requestId: $ctx.id,\r\n          awsRequestId: $ctx.context?.awsRequestId,\r\n          endpoint: request.url,\r\n          method: request.method,\r\n          controller: controllerName,\r\n          controllerMethod: methodName,\r\n        });\r\n\r\n        // Finalizar span e force flush\r\n        if (spanResult) {\r\n          spanResult.end();\r\n        }\r\n        await (telemetryService as any).telemetryProvider.forceFlush();\r\n\r\n        $ctx.logger.info('Error sent to telemetry');\r\n\r\n        // Re-throw para manter o comportamento normal de erro\r\n        throw error;\r\n      }\r\n    };\r\n\r\n    return descriptor;\r\n  };\r\n}\r\n"]}
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
import { ServerlessContext } from '@tsed/platform-serverless';
|
|
2
|
+
import { TsedTelemetryProvider } from './config';
|
|
3
|
+
export interface ErrorLogData {
|
|
4
|
+
error: Error;
|
|
5
|
+
context?: Record<string, any>;
|
|
6
|
+
accountUserUid?: string;
|
|
7
|
+
accountUid?: string;
|
|
8
|
+
requestId?: string;
|
|
9
|
+
endpoint?: string;
|
|
10
|
+
method?: string;
|
|
11
|
+
controller?: string;
|
|
12
|
+
controllerMethod?: string;
|
|
13
|
+
applicationUid?: string;
|
|
14
|
+
awsRequestId?: string;
|
|
15
|
+
}
|
|
16
|
+
export declare class TsedTelemetryService {
|
|
17
|
+
private readonly telemetryProvider;
|
|
18
|
+
protected $ctx: ServerlessContext;
|
|
19
|
+
private standardLogger;
|
|
20
|
+
private standardTracer;
|
|
21
|
+
constructor(telemetryProvider: TsedTelemetryProvider);
|
|
22
|
+
private getLogger;
|
|
23
|
+
private getTracer;
|
|
24
|
+
logError(data: ErrorLogData): Promise<void>;
|
|
25
|
+
logException(error: Error, context?: Record<string, any>): Promise<void>;
|
|
26
|
+
}
|
|
@@ -0,0 +1,150 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
|
|
3
|
+
var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
|
|
4
|
+
if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
|
|
5
|
+
else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
|
|
6
|
+
return c > 3 && r && Object.defineProperty(target, key, r), r;
|
|
7
|
+
};
|
|
8
|
+
var __metadata = (this && this.__metadata) || function (k, v) {
|
|
9
|
+
if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
|
|
10
|
+
};
|
|
11
|
+
var __param = (this && this.__param) || function (paramIndex, decorator) {
|
|
12
|
+
return function (target, key) { decorator(target, key, paramIndex); }
|
|
13
|
+
};
|
|
14
|
+
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
|
|
15
|
+
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
|
|
16
|
+
return new (P || (P = Promise))(function (resolve, reject) {
|
|
17
|
+
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
|
18
|
+
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
|
19
|
+
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
|
|
20
|
+
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
21
|
+
});
|
|
22
|
+
};
|
|
23
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
24
|
+
exports.TsedTelemetryService = void 0;
|
|
25
|
+
const di_1 = require("@tsed/di");
|
|
26
|
+
const platform_serverless_1 = require("@tsed/platform-serverless");
|
|
27
|
+
const config_1 = require("./config");
|
|
28
|
+
const core_1 = require("../core");
|
|
29
|
+
const api_1 = require("@opentelemetry/api");
|
|
30
|
+
let TsedTelemetryService = class TsedTelemetryService {
|
|
31
|
+
constructor(telemetryProvider) {
|
|
32
|
+
this.telemetryProvider = telemetryProvider;
|
|
33
|
+
this.standardLogger = null;
|
|
34
|
+
this.standardTracer = null;
|
|
35
|
+
}
|
|
36
|
+
getLogger() {
|
|
37
|
+
var _a;
|
|
38
|
+
if (!this.telemetryProvider.isInitialized()) {
|
|
39
|
+
(_a = this.$ctx) === null || _a === void 0 ? void 0 : _a.logger.warn('[Telemetry] Telemetry not initialized, skipping log');
|
|
40
|
+
return null;
|
|
41
|
+
}
|
|
42
|
+
if (!this.standardLogger) {
|
|
43
|
+
const loggerProvider = this.telemetryProvider.getLoggerProvider();
|
|
44
|
+
if (!loggerProvider) {
|
|
45
|
+
return null;
|
|
46
|
+
}
|
|
47
|
+
const otelLogger = loggerProvider.getLogger('tsed-service-logger', '1.0.0');
|
|
48
|
+
const serviceName = process.env.SERVICE_NAME || 'tsed-service';
|
|
49
|
+
this.standardLogger = new core_1.StandardLogger(otelLogger, serviceName);
|
|
50
|
+
}
|
|
51
|
+
return this.standardLogger;
|
|
52
|
+
}
|
|
53
|
+
getTracer() {
|
|
54
|
+
var _a;
|
|
55
|
+
if (!this.telemetryProvider.isInitialized()) {
|
|
56
|
+
(_a = this.$ctx) === null || _a === void 0 ? void 0 : _a.logger.warn('[Telemetry] Telemetry not initialized, skipping trace');
|
|
57
|
+
return null;
|
|
58
|
+
}
|
|
59
|
+
if (!this.standardTracer) {
|
|
60
|
+
const tracerProvider = this.telemetryProvider.getTracerProvider();
|
|
61
|
+
if (!tracerProvider) {
|
|
62
|
+
return null;
|
|
63
|
+
}
|
|
64
|
+
const serviceName = process.env.SERVICE_NAME || 'tsed-service';
|
|
65
|
+
const otelTracer = api_1.trace.getTracer(serviceName, '1.0.0');
|
|
66
|
+
this.standardTracer = new core_1.StandardTracer(otelTracer, serviceName);
|
|
67
|
+
}
|
|
68
|
+
return this.standardTracer;
|
|
69
|
+
}
|
|
70
|
+
logError(data) {
|
|
71
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
72
|
+
var _a, _b, _c, _d;
|
|
73
|
+
const logger = this.getLogger();
|
|
74
|
+
if (!logger) {
|
|
75
|
+
return;
|
|
76
|
+
}
|
|
77
|
+
try {
|
|
78
|
+
let stage = process.env.STAGE || 'development';
|
|
79
|
+
if (stage === 'prod')
|
|
80
|
+
stage = 'production';
|
|
81
|
+
(_a = this.$ctx) === null || _a === void 0 ? void 0 : _a.logger.info('[Telemetry] Logging error...');
|
|
82
|
+
yield logger.logError({
|
|
83
|
+
message: data.error.message,
|
|
84
|
+
error: data.error,
|
|
85
|
+
serviceName: process.env.SERVICE_NAME || 'tsed-service',
|
|
86
|
+
environment: stage,
|
|
87
|
+
http: {
|
|
88
|
+
endpoint: data.endpoint,
|
|
89
|
+
method: data.method,
|
|
90
|
+
statusCode: data.error.status || 500,
|
|
91
|
+
},
|
|
92
|
+
user: {
|
|
93
|
+
accountUserUid: data.accountUserUid,
|
|
94
|
+
accountUid: data.accountUid,
|
|
95
|
+
applicationUid: data.applicationUid,
|
|
96
|
+
},
|
|
97
|
+
execution: {
|
|
98
|
+
requestId: data.requestId,
|
|
99
|
+
awsRequestId: data.awsRequestId,
|
|
100
|
+
controller: data.controller,
|
|
101
|
+
controllerMethod: data.controllerMethod,
|
|
102
|
+
},
|
|
103
|
+
context: data.context,
|
|
104
|
+
});
|
|
105
|
+
(_b = this.$ctx) === null || _b === void 0 ? void 0 : _b.logger.info('[Telemetry] Forcing flush...');
|
|
106
|
+
const flushStartTime = Date.now();
|
|
107
|
+
yield this.telemetryProvider.forceFlush();
|
|
108
|
+
const flushDuration = Date.now() - flushStartTime;
|
|
109
|
+
(_c = this.$ctx) === null || _c === void 0 ? void 0 : _c.logger.info({
|
|
110
|
+
'Telemetry - Log sent successfully': {
|
|
111
|
+
errorType: data.error.name,
|
|
112
|
+
message: data.error.message,
|
|
113
|
+
endpoint: data.endpoint,
|
|
114
|
+
controller: data.controller,
|
|
115
|
+
method: data.controllerMethod,
|
|
116
|
+
flushDurationMs: flushDuration,
|
|
117
|
+
},
|
|
118
|
+
});
|
|
119
|
+
}
|
|
120
|
+
catch (error) {
|
|
121
|
+
(_d = this.$ctx) === null || _d === void 0 ? void 0 : _d.logger.error({
|
|
122
|
+
'Telemetry - CRITICAL ERROR - Failed to log error': {
|
|
123
|
+
originalError: data.error.message,
|
|
124
|
+
telemetryError: error instanceof Error ? error.message : String(error),
|
|
125
|
+
stack: error instanceof Error ? error.stack : undefined,
|
|
126
|
+
},
|
|
127
|
+
});
|
|
128
|
+
}
|
|
129
|
+
});
|
|
130
|
+
}
|
|
131
|
+
logException(error, context) {
|
|
132
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
133
|
+
yield this.logError({
|
|
134
|
+
error,
|
|
135
|
+
context,
|
|
136
|
+
});
|
|
137
|
+
});
|
|
138
|
+
}
|
|
139
|
+
};
|
|
140
|
+
exports.TsedTelemetryService = TsedTelemetryService;
|
|
141
|
+
__decorate([
|
|
142
|
+
(0, di_1.InjectContext)(),
|
|
143
|
+
__metadata("design:type", platform_serverless_1.ServerlessContext)
|
|
144
|
+
], TsedTelemetryService.prototype, "$ctx", void 0);
|
|
145
|
+
exports.TsedTelemetryService = TsedTelemetryService = __decorate([
|
|
146
|
+
(0, di_1.Injectable)(),
|
|
147
|
+
__param(0, (0, di_1.Inject)()),
|
|
148
|
+
__metadata("design:paramtypes", [config_1.TsedTelemetryProvider])
|
|
149
|
+
], TsedTelemetryService);
|
|
150
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"service.js","sourceRoot":"","sources":["../../src/tsed/service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, Inject, InjectContext } from '@tsed/di';\r\nimport { ServerlessContext } from '@tsed/platform-serverless';\r\nimport { TsedTelemetryProvider } from './config';\r\nimport { StandardLogger, StandardTracer } from '../core';\r\nimport { trace, metrics } from '@opentelemetry/api';\r\n\r\nexport interface ErrorLogData {\r\n  error: Error;\r\n  context?: Record<string, any>;\r\n  accountUserUid?: string;\r\n  accountUid?: string;\r\n  requestId?: string;\r\n  endpoint?: string;\r\n  method?: string;\r\n  controller?: string;\r\n  controllerMethod?: string;\r\n  applicationUid?: string;\r\n  awsRequestId?: string;\r\n}\r\n\r\n@Injectable()\r\nexport class TsedTelemetryService {\r\n  @InjectContext() protected $ctx: ServerlessContext;\r\n  private standardLogger: StandardLogger | null = null;\r\n  private standardTracer: StandardTracer | null = null;\r\n\r\n  constructor(\r\n    @Inject()\r\n    private readonly telemetryProvider: TsedTelemetryProvider\r\n  ) {}\r\n\r\n  private getLogger(): StandardLogger | null {\r\n    if (!this.telemetryProvider.isInitialized()) {\r\n      this.$ctx?.logger.warn(\r\n        '[Telemetry] Telemetry not initialized, skipping log'\r\n      );\r\n      return null;\r\n    }\r\n\r\n    if (!this.standardLogger) {\r\n      const loggerProvider = this.telemetryProvider.getLoggerProvider();\r\n      if (!loggerProvider) {\r\n        return null;\r\n      }\r\n\r\n      const otelLogger = loggerProvider.getLogger(\r\n        'tsed-service-logger',\r\n        '1.0.0'\r\n      );\r\n\r\n      // Pega o service name do ambiente ou usa default\r\n      const serviceName = process.env.SERVICE_NAME || 'tsed-service';\r\n      this.standardLogger = new StandardLogger(otelLogger, serviceName);\r\n    }\r\n\r\n    return this.standardLogger;\r\n  }\r\n\r\n  private getTracer(): StandardTracer | null {\r\n    if (!this.telemetryProvider.isInitialized()) {\r\n      this.$ctx?.logger.warn(\r\n        '[Telemetry] Telemetry not initialized, skipping trace'\r\n      );\r\n      return null;\r\n    }\r\n\r\n    if (!this.standardTracer) {\r\n      const tracerProvider = this.telemetryProvider.getTracerProvider();\r\n      if (!tracerProvider) {\r\n        return null;\r\n      }\r\n\r\n      const serviceName = process.env.SERVICE_NAME || 'tsed-service';\r\n      const otelTracer = trace.getTracer(serviceName, '1.0.0');\r\n      this.standardTracer = new StandardTracer(otelTracer, serviceName);\r\n    }\r\n\r\n    return this.standardTracer;\r\n  }\r\n\r\n  public async logError(data: ErrorLogData): Promise<void> {\r\n    const logger = this.getLogger();\r\n    if (!logger) {\r\n      return;\r\n    }\r\n\r\n    try {\r\n      let stage = process.env.STAGE || 'development';\r\n      if (stage === 'prod') stage = 'production';\r\n\r\n      this.$ctx?.logger.info('[Telemetry] Logging error...');\r\n\r\n      await logger.logError({\r\n        message: data.error.message,\r\n        error: data.error,\r\n        serviceName: process.env.SERVICE_NAME || 'tsed-service',\r\n        environment: stage,\r\n        http: {\r\n          endpoint: data.endpoint,\r\n          method: data.method,\r\n          statusCode: (data.error as any).status || 500,\r\n        },\r\n        user: {\r\n          accountUserUid: data.accountUserUid,\r\n          accountUid: data.accountUid,\r\n          applicationUid: data.applicationUid,\r\n        },\r\n        execution: {\r\n          requestId: data.requestId,\r\n          awsRequestId: data.awsRequestId,\r\n          controller: data.controller,\r\n          controllerMethod: data.controllerMethod,\r\n        },\r\n        context: data.context,\r\n      });\r\n\r\n      // CRITICAL: Force flush to ensure log is sent before Lambda freezes\r\n      this.$ctx?.logger.info('[Telemetry] Forcing flush...');\r\n      const flushStartTime = Date.now();\r\n\r\n      await this.telemetryProvider.forceFlush();\r\n\r\n      const flushDuration = Date.now() - flushStartTime;\r\n\r\n      this.$ctx?.logger.info({\r\n        'Telemetry - Log sent successfully': {\r\n          errorType: data.error.name,\r\n          message: data.error.message,\r\n          endpoint: data.endpoint,\r\n          controller: data.controller,\r\n          method: data.controllerMethod,\r\n          flushDurationMs: flushDuration,\r\n        },\r\n      });\r\n    } catch (error) {\r\n      this.$ctx?.logger.error({\r\n        'Telemetry - CRITICAL ERROR - Failed to log error': {\r\n          originalError: data.error.message,\r\n          telemetryError:\r\n            error instanceof Error ? error.message : String(error),\r\n          stack: error instanceof Error ? error.stack : undefined,\r\n        },\r\n      });\r\n      // Don't re-throw to avoid breaking the application flow\r\n    }\r\n  }\r\n\r\n  public async logException(\r\n    error: Error,\r\n    context?: Record<string, any>\r\n  ): Promise<void> {\r\n    await this.logError({\r\n      error,\r\n      context,\r\n    });\r\n  }\r\n}\r\n"]}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
import { LogRecordProcessor, ReadableLogRecord } from '@opentelemetry/sdk-logs';
|
|
2
|
+
import { OTLPLogExporter } from '@opentelemetry/exporter-logs-otlp-http';
|
|
3
|
+
import { Context } from '@opentelemetry/api';
|
|
4
|
+
export declare class TsedSyncLogRecordProcessor implements LogRecordProcessor {
|
|
5
|
+
private readonly exporter;
|
|
6
|
+
private readonly pendingExports;
|
|
7
|
+
constructor(exporter: OTLPLogExporter);
|
|
8
|
+
onEmit(logRecord: ReadableLogRecord, _context?: Context): void;
|
|
9
|
+
forceFlush(): Promise<void>;
|
|
10
|
+
shutdown(): Promise<void>;
|
|
11
|
+
}
|
|
@@ -0,0 +1,74 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
|
|
3
|
+
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
|
|
4
|
+
return new (P || (P = Promise))(function (resolve, reject) {
|
|
5
|
+
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
|
6
|
+
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
|
7
|
+
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
|
|
8
|
+
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
9
|
+
});
|
|
10
|
+
};
|
|
11
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
12
|
+
exports.TsedSyncLogRecordProcessor = void 0;
|
|
13
|
+
class TsedSyncLogRecordProcessor {
|
|
14
|
+
constructor(exporter) {
|
|
15
|
+
this.pendingExports = [];
|
|
16
|
+
this.exporter = exporter;
|
|
17
|
+
}
|
|
18
|
+
onEmit(logRecord, _context) {
|
|
19
|
+
const exportPromise = new Promise((resolve, reject) => {
|
|
20
|
+
this.exporter.export([logRecord], (result) => {
|
|
21
|
+
if (result.code === 0) {
|
|
22
|
+
resolve();
|
|
23
|
+
}
|
|
24
|
+
else {
|
|
25
|
+
console.error('[SyncLogRecordProcessor] Export failed:', result.error);
|
|
26
|
+
reject(result.error instanceof Error
|
|
27
|
+
? result.error
|
|
28
|
+
: new Error(String(result.error)));
|
|
29
|
+
}
|
|
30
|
+
});
|
|
31
|
+
})
|
|
32
|
+
.then(() => {
|
|
33
|
+
const index = this.pendingExports.indexOf(exportPromise);
|
|
34
|
+
if (index > -1) {
|
|
35
|
+
this.pendingExports.splice(index, 1);
|
|
36
|
+
}
|
|
37
|
+
})
|
|
38
|
+
.catch((error) => {
|
|
39
|
+
console.error('[SyncLogRecordProcessor] Export error:', error);
|
|
40
|
+
const index = this.pendingExports.indexOf(exportPromise);
|
|
41
|
+
if (index > -1) {
|
|
42
|
+
this.pendingExports.splice(index, 1);
|
|
43
|
+
}
|
|
44
|
+
});
|
|
45
|
+
this.pendingExports.push(exportPromise);
|
|
46
|
+
}
|
|
47
|
+
forceFlush() {
|
|
48
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
49
|
+
console.log(`[SyncLogRecordProcessor] forceFlush called with ${this.pendingExports.length} pending exports`);
|
|
50
|
+
if (this.pendingExports.length === 0) {
|
|
51
|
+
console.log('[SyncLogRecordProcessor] No pending exports to flush');
|
|
52
|
+
return;
|
|
53
|
+
}
|
|
54
|
+
try {
|
|
55
|
+
yield Promise.all(this.pendingExports);
|
|
56
|
+
console.log('[SyncLogRecordProcessor] All pending exports completed');
|
|
57
|
+
}
|
|
58
|
+
catch (error) {
|
|
59
|
+
console.error('[SyncLogRecordProcessor] Error during forceFlush:', error);
|
|
60
|
+
throw error;
|
|
61
|
+
}
|
|
62
|
+
});
|
|
63
|
+
}
|
|
64
|
+
shutdown() {
|
|
65
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
66
|
+
console.log('[SyncLogRecordProcessor] Shutting down...');
|
|
67
|
+
yield this.forceFlush();
|
|
68
|
+
yield this.exporter.shutdown();
|
|
69
|
+
console.log('[SyncLogRecordProcessor] Shutdown complete');
|
|
70
|
+
});
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
exports.TsedSyncLogRecordProcessor = TsedSyncLogRecordProcessor;
|
|
74
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3luYy1sb2ctcmVjb3JkLXByb2Nlc3Nvci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy90c2VkL3N5bmMtbG9nLXJlY29yZC1wcm9jZXNzb3IudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7O0FBWUEsTUFBYSwwQkFBMEI7SUFJckMsWUFBWSxRQUF5QjtRQUZwQixtQkFBYyxHQUFvQixFQUFFLENBQUM7UUFHcEQsSUFBSSxDQUFDLFFBQVEsR0FBRyxRQUFRLENBQUM7SUFDM0IsQ0FBQztJQUVELE1BQU0sQ0FBQyxTQUE0QixFQUFFLFFBQWtCO1FBSXJELE1BQU0sYUFBYSxHQUFHLElBQUksT0FBTyxDQUFPLENBQUMsT0FBTyxFQUFFLE1BQU0sRUFBRSxFQUFFO1lBQzFELElBQUksQ0FBQyxRQUFRLENBQUMsTUFBTSxDQUFDLENBQUMsU0FBUyxDQUFDLEVBQUUsQ0FBQyxNQUFNLEVBQUUsRUFBRTtnQkFDM0MsSUFBSSxNQUFNLENBQUMsSUFBSSxLQUFLLENBQUMsRUFBRSxDQUFDO29CQUN0QixPQUFPLEVBQUUsQ0FBQztnQkFDWixDQUFDO3FCQUFNLENBQUM7b0JBQ04sT0FBTyxDQUFDLEtBQUssQ0FDWCx5Q0FBeUMsRUFDekMsTUFBTSxDQUFDLEtBQUssQ0FDYixDQUFDO29CQUNGLE1BQU0sQ0FDSixNQUFNLENBQUMsS0FBSyxZQUFZLEtBQUs7d0JBQzNCLENBQUMsQ0FBQyxNQUFNLENBQUMsS0FBSzt3QkFDZCxDQUFDLENBQUMsSUFBSSxLQUFLLENBQUMsTUFBTSxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUNwQyxDQUFDO2dCQUNKLENBQUM7WUFDSCxDQUFDLENBQUMsQ0FBQztRQUNMLENBQUMsQ0FBQzthQUNDLElBQUksQ0FBQyxHQUFHLEVBQUU7WUFDVCxNQUFNLEtBQUssR0FBRyxJQUFJLENBQUMsY0FBYyxDQUFDLE9BQU8sQ0FBQyxhQUFhLENBQUMsQ0FBQztZQUN6RCxJQUFJLEtBQUssR0FBRyxDQUFDLENBQUMsRUFBRSxDQUFDO2dCQUNmLElBQUksQ0FBQyxjQUFjLENBQUMsTUFBTSxDQUFDLEtBQUssRUFBRSxDQUFDLENBQUMsQ0FBQztZQUN2QyxDQUFDO1FBQ0gsQ0FBQyxDQUFDO2FBQ0QsS0FBSyxDQUFDLENBQUMsS0FBYyxFQUFFLEVBQUU7WUFDeEIsT0FBTyxDQUFDLEtBQUssQ0FBQyx3Q0FBd0MsRUFBRSxLQUFLLENBQUMsQ0FBQztZQUMvRCxNQUFNLEtBQUssR0FBRyxJQUFJLENBQUMsY0FBYyxDQUFDLE9BQU8sQ0FBQyxhQUFhLENBQUMsQ0FBQztZQUN6RCxJQUFJLEtBQUssR0FBRyxDQUFDLENBQUMsRUFBRSxDQUFDO2dCQUNmLElBQUksQ0FBQyxjQUFjLENBQUMsTUFBTSxDQUFDLEtBQUssRUFBRSxDQUFDLENBQUMsQ0FBQztZQUN2QyxDQUFDO1FBQ0gsQ0FBQyxDQUFDLENBQUM7UUFFTCxJQUFJLENBQUMsY0FBYyxDQUFDLElBQUksQ0FBQyxhQUFhLENBQUMsQ0FBQztJQUMxQyxDQUFDO0lBRUssVUFBVTs7WUFDZCxPQUFPLENBQUMsR0FBRyxDQUNULG1EQUFtRCxJQUFJLENBQUMsY0FBYyxDQUFDLE1BQU0sa0JBQWtCLENBQ2hHLENBQUM7WUFFRixJQUFJLElBQUksQ0FBQyxjQUFjLENBQUMsTUFBTSxLQUFLLENBQUMsRUFBRSxDQUFDO2dCQUNyQyxPQUFPLENBQUMsR0FBRyxDQUFDLHNEQUFzRCxDQUFDLENBQUM7Z0JBQ3BFLE9BQU87WUFDVCxDQUFDO1lBRUQsSUFBSSxDQUFDO2dCQUVILE1BQU0sT0FBTyxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsY0FBYyxDQUFDLENBQUM7Z0JBQ3ZDLE9BQU8sQ0FBQyxHQUFHLENBQUMsd0RBQXdELENBQUMsQ0FBQztZQUN4RSxDQUFDO1lBQUMsT0FBTyxLQUFLLEVBQUUsQ0FBQztnQkFDZixPQUFPLENBQUMsS0FBSyxDQUFDLG1EQUFtRCxFQUFFLEtBQUssQ0FBQyxDQUFDO2dCQUMxRSxNQUFNLEtBQUssQ0FBQztZQUNkLENBQUM7UUFDSCxDQUFDO0tBQUE7SUFFSyxRQUFROztZQUNaLE9BQU8sQ0FBQyxHQUFHLENBQUMsMkNBQTJDLENBQUMsQ0FBQztZQUN6RCxNQUFNLElBQUksQ0FBQyxVQUFVLEVBQUUsQ0FBQztZQUN4QixNQUFNLElBQUksQ0FBQyxRQUFRLENBQUMsUUFBUSxFQUFFLENBQUM7WUFDL0IsT0FBTyxDQUFDLEdBQUcsQ0FBQyw0Q0FBNEMsQ0FBQyxDQUFDO1FBQzVELENBQUM7S0FBQTtDQUNGO0FBeEVELGdFQXdFQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IExvZ1JlY29yZFByb2Nlc3NvciwgUmVhZGFibGVMb2dSZWNvcmQgfSBmcm9tICdAb3BlbnRlbGVtZXRyeS9zZGstbG9ncyc7XHJcbmltcG9ydCB7IE9UTFBMb2dFeHBvcnRlciB9IGZyb20gJ0BvcGVudGVsZW1ldHJ5L2V4cG9ydGVyLWxvZ3Mtb3RscC1odHRwJztcclxuaW1wb3J0IHsgQ29udGV4dCB9IGZyb20gJ0BvcGVudGVsZW1ldHJ5L2FwaSc7XHJcblxyXG4vKipcclxuICogQ3VzdG9tIExvZ1JlY29yZFByb2Nlc3NvciB0aGF0IGVuc3VyZXMgc3luY2hyb25vdXMgZXhwb3J0IGZvciBBV1MgTGFtYmRhXHJcbiAqXHJcbiAqIFRoaXMgcHJvY2Vzc29yIHdyYXBzIHRoZSBPVExQIGV4cG9ydGVyIGFuZCBlbnN1cmVzIHRoYXQgbG9ncyBhcmUgYWN0dWFsbHlcclxuICogZXhwb3J0ZWQgc3luY2hyb25vdXNseSBieSBpbW1lZGlhdGVseSBjYWxsaW5nIGV4cG9ydCgpIGFuZCB3YWl0aW5nIGZvciBpdFxyXG4gKiB0byBjb21wbGV0ZSwgcmF0aGVyIHRoYW4gcmVseWluZyBvbiBTaW1wbGVMb2dSZWNvcmRQcm9jZXNzb3Igd2hpY2ggbWF5IG5vdFxyXG4gKiBwcm9wZXJseSBhd2FpdCB0aGUgYXN5bmMgSFRUUCByZXF1ZXN0IGluIExhbWJkYSBlbnZpcm9ubWVudHMuXHJcbiAqL1xyXG5leHBvcnQgY2xhc3MgVHNlZFN5bmNMb2dSZWNvcmRQcm9jZXNzb3IgaW1wbGVtZW50cyBMb2dSZWNvcmRQcm9jZXNzb3Ige1xyXG4gIHByaXZhdGUgcmVhZG9ubHkgZXhwb3J0ZXI6IE9UTFBMb2dFeHBvcnRlcjtcclxuICBwcml2YXRlIHJlYWRvbmx5IHBlbmRpbmdFeHBvcnRzOiBQcm9taXNlPHZvaWQ+W10gPSBbXTtcclxuXHJcbiAgY29uc3RydWN0b3IoZXhwb3J0ZXI6IE9UTFBMb2dFeHBvcnRlcikge1xyXG4gICAgdGhpcy5leHBvcnRlciA9IGV4cG9ydGVyO1xyXG4gIH1cclxuXHJcbiAgb25FbWl0KGxvZ1JlY29yZDogUmVhZGFibGVMb2dSZWNvcmQsIF9jb250ZXh0PzogQ29udGV4dCk6IHZvaWQge1xyXG4gICAgLy8gRXhwb3J0IGltbWVkaWF0ZWx5IGFuZCB0cmFjayB0aGUgcHJvbWlzZVxyXG4gICAgLy8gVGhlIGV4cG9ydCBtZXRob2QgcmV0dXJucyB2b2lkLCBidXQgaW50ZXJuYWxseSB0cmlnZ2VycyBhc3luYyBIVFRQIHJlcXVlc3RcclxuICAgIC8vIFdlIG5lZWQgdG8gd3JhcCBpdCB0byB0cmFjayBjb21wbGV0aW9uXHJcbiAgICBjb25zdCBleHBvcnRQcm9taXNlID0gbmV3IFByb21pc2U8dm9pZD4oKHJlc29sdmUsIHJlamVjdCkgPT4ge1xyXG4gICAgICB0aGlzLmV4cG9ydGVyLmV4cG9ydChbbG9nUmVjb3JkXSwgKHJlc3VsdCkgPT4ge1xyXG4gICAgICAgIGlmIChyZXN1bHQuY29kZSA9PT0gMCkge1xyXG4gICAgICAgICAgcmVzb2x2ZSgpO1xyXG4gICAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgICBjb25zb2xlLmVycm9yKFxyXG4gICAgICAgICAgICAnW1N5bmNMb2dSZWNvcmRQcm9jZXNzb3JdIEV4cG9ydCBmYWlsZWQ6JyxcclxuICAgICAgICAgICAgcmVzdWx0LmVycm9yXHJcbiAgICAgICAgICApO1xyXG4gICAgICAgICAgcmVqZWN0KFxyXG4gICAgICAgICAgICByZXN1bHQuZXJyb3IgaW5zdGFuY2VvZiBFcnJvclxyXG4gICAgICAgICAgICAgID8gcmVzdWx0LmVycm9yXHJcbiAgICAgICAgICAgICAgOiBuZXcgRXJyb3IoU3RyaW5nKHJlc3VsdC5lcnJvcikpXHJcbiAgICAgICAgICApO1xyXG4gICAgICAgIH1cclxuICAgICAgfSk7XHJcbiAgICB9KVxyXG4gICAgICAudGhlbigoKSA9PiB7XHJcbiAgICAgICAgY29uc3QgaW5kZXggPSB0aGlzLnBlbmRpbmdFeHBvcnRzLmluZGV4T2YoZXhwb3J0UHJvbWlzZSk7XHJcbiAgICAgICAgaWYgKGluZGV4ID4gLTEpIHtcclxuICAgICAgICAgIHRoaXMucGVuZGluZ0V4cG9ydHMuc3BsaWNlKGluZGV4LCAxKTtcclxuICAgICAgICB9XHJcbiAgICAgIH0pXHJcbiAgICAgIC5jYXRjaCgoZXJyb3I6IHVua25vd24pID0+IHtcclxuICAgICAgICBjb25zb2xlLmVycm9yKCdbU3luY0xvZ1JlY29yZFByb2Nlc3Nvcl0gRXhwb3J0IGVycm9yOicsIGVycm9yKTtcclxuICAgICAgICBjb25zdCBpbmRleCA9IHRoaXMucGVuZGluZ0V4cG9ydHMuaW5kZXhPZihleHBvcnRQcm9taXNlKTtcclxuICAgICAgICBpZiAoaW5kZXggPiAtMSkge1xyXG4gICAgICAgICAgdGhpcy5wZW5kaW5nRXhwb3J0cy5zcGxpY2UoaW5kZXgsIDEpO1xyXG4gICAgICAgIH1cclxuICAgICAgfSk7XHJcblxyXG4gICAgdGhpcy5wZW5kaW5nRXhwb3J0cy5wdXNoKGV4cG9ydFByb21pc2UpO1xyXG4gIH1cclxuXHJcbiAgYXN5bmMgZm9yY2VGbHVzaCgpOiBQcm9taXNlPHZvaWQ+IHtcclxuICAgIGNvbnNvbGUubG9nKFxyXG4gICAgICBgW1N5bmNMb2dSZWNvcmRQcm9jZXNzb3JdIGZvcmNlRmx1c2ggY2FsbGVkIHdpdGggJHt0aGlzLnBlbmRpbmdFeHBvcnRzLmxlbmd0aH0gcGVuZGluZyBleHBvcnRzYFxyXG4gICAgKTtcclxuXHJcbiAgICBpZiAodGhpcy5wZW5kaW5nRXhwb3J0cy5sZW5ndGggPT09IDApIHtcclxuICAgICAgY29uc29sZS5sb2coJ1tTeW5jTG9nUmVjb3JkUHJvY2Vzc29yXSBObyBwZW5kaW5nIGV4cG9ydHMgdG8gZmx1c2gnKTtcclxuICAgICAgcmV0dXJuO1xyXG4gICAgfVxyXG5cclxuICAgIHRyeSB7XHJcbiAgICAgIC8vIFdhaXQgZm9yIGFsbCBwZW5kaW5nIGV4cG9ydHMgdG8gY29tcGxldGVcclxuICAgICAgYXdhaXQgUHJvbWlzZS5hbGwodGhpcy5wZW5kaW5nRXhwb3J0cyk7XHJcbiAgICAgIGNvbnNvbGUubG9nKCdbU3luY0xvZ1JlY29yZFByb2Nlc3Nvcl0gQWxsIHBlbmRpbmcgZXhwb3J0cyBjb21wbGV0ZWQnKTtcclxuICAgIH0gY2F0Y2ggKGVycm9yKSB7XHJcbiAgICAgIGNvbnNvbGUuZXJyb3IoJ1tTeW5jTG9nUmVjb3JkUHJvY2Vzc29yXSBFcnJvciBkdXJpbmcgZm9yY2VGbHVzaDonLCBlcnJvcik7XHJcbiAgICAgIHRocm93IGVycm9yO1xyXG4gICAgfVxyXG4gIH1cclxuXHJcbiAgYXN5bmMgc2h1dGRvd24oKTogUHJvbWlzZTx2b2lkPiB7XHJcbiAgICBjb25zb2xlLmxvZygnW1N5bmNMb2dSZWNvcmRQcm9jZXNzb3JdIFNodXR0aW5nIGRvd24uLi4nKTtcclxuICAgIGF3YWl0IHRoaXMuZm9yY2VGbHVzaCgpO1xyXG4gICAgYXdhaXQgdGhpcy5leHBvcnRlci5zaHV0ZG93bigpO1xyXG4gICAgY29uc29sZS5sb2coJ1tTeW5jTG9nUmVjb3JkUHJvY2Vzc29yXSBTaHV0ZG93biBjb21wbGV0ZScpO1xyXG4gIH1cclxufVxyXG4iXX0=
|