@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.
Files changed (95) hide show
  1. package/.github/workflows/pull_request.yml +32 -32
  2. package/.github/workflows/release.yml +129 -129
  3. package/.prettierignore +4 -4
  4. package/CHANGELOG.md +14 -0
  5. package/DOCS_GUIDE.md +151 -0
  6. package/README.md +248 -0
  7. package/dist/core/__tests__/logger-types.test.d.ts +1 -0
  8. package/dist/core/__tests__/logger-types.test.js +325 -0
  9. package/dist/core/__tests__/logger.test.d.ts +1 -0
  10. package/dist/core/__tests__/logger.test.js +337 -0
  11. package/dist/core/__tests__/tracer.test.d.ts +1 -0
  12. package/dist/core/__tests__/tracer.test.js +330 -0
  13. package/dist/core/index.d.ts +4 -0
  14. package/dist/core/index.js +8 -0
  15. package/dist/core/logger-types.d.ts +43 -0
  16. package/dist/core/logger-types.js +3 -0
  17. package/dist/core/logger.d.ts +13 -0
  18. package/dist/core/logger.js +123 -0
  19. package/dist/core/tracer-types.d.ts +50 -0
  20. package/dist/core/tracer-types.js +3 -0
  21. package/dist/core/tracer.d.ts +10 -0
  22. package/dist/core/tracer.js +114 -0
  23. package/dist/index.d.ts +3 -1
  24. package/dist/index.js +4 -2
  25. package/dist/sst/__tests__/telemetry.test.d.ts +1 -0
  26. package/dist/sst/__tests__/telemetry.test.js +138 -0
  27. package/dist/sst/index.d.ts +1 -0
  28. package/dist/sst/index.js +18 -0
  29. package/dist/sst/middy/index.d.ts +1 -0
  30. package/dist/sst/middy/index.js +18 -0
  31. package/dist/sst/middy/middleware.d.ts +5 -0
  32. package/dist/sst/middy/middleware.js +157 -0
  33. package/dist/sst/telemetry.d.ts +4 -0
  34. package/dist/sst/telemetry.js +121 -0
  35. package/dist/telemetry/core/__tests__/logger-types.test.d.ts +1 -0
  36. package/dist/telemetry/core/__tests__/logger-types.test.js +325 -0
  37. package/dist/telemetry/core/__tests__/logger.test.d.ts +1 -0
  38. package/dist/telemetry/core/__tests__/logger.test.js +337 -0
  39. package/dist/telemetry/core/__tests__/tracer.test.d.ts +1 -0
  40. package/dist/telemetry/core/__tests__/tracer.test.js +330 -0
  41. package/dist/telemetry/core/index.d.ts +4 -0
  42. package/dist/telemetry/core/index.js +8 -0
  43. package/dist/telemetry/core/logger-types.d.ts +43 -0
  44. package/dist/telemetry/core/logger-types.js +3 -0
  45. package/dist/telemetry/core/logger.d.ts +13 -0
  46. package/dist/telemetry/core/logger.js +123 -0
  47. package/dist/telemetry/core/tracer-types.d.ts +50 -0
  48. package/dist/telemetry/core/tracer-types.js +3 -0
  49. package/dist/telemetry/core/tracer.d.ts +10 -0
  50. package/dist/telemetry/core/tracer.js +114 -0
  51. package/dist/telemetry/index.d.ts +3 -0
  52. package/dist/telemetry/index.js +20 -0
  53. package/dist/telemetry/sst/__tests__/telemetry.test.d.ts +1 -0
  54. package/dist/telemetry/sst/__tests__/telemetry.test.js +138 -0
  55. package/dist/telemetry/sst/index.d.ts +1 -0
  56. package/dist/telemetry/sst/index.js +18 -0
  57. package/dist/telemetry/sst/middy/index.d.ts +1 -0
  58. package/dist/telemetry/sst/middy/index.js +18 -0
  59. package/dist/telemetry/sst/middy/middleware.d.ts +5 -0
  60. package/dist/telemetry/sst/middy/middleware.js +157 -0
  61. package/dist/telemetry/sst/telemetry.d.ts +4 -0
  62. package/dist/telemetry/sst/telemetry.js +121 -0
  63. package/dist/telemetry/tsed/__tests__/config.test.d.ts +1 -0
  64. package/dist/telemetry/tsed/__tests__/config.test.js +146 -0
  65. package/dist/telemetry/tsed/__tests__/service.test.d.ts +1 -0
  66. package/dist/telemetry/tsed/__tests__/service.test.js +63 -0
  67. package/dist/telemetry/tsed/config.d.ts +26 -0
  68. package/dist/telemetry/tsed/config.js +166 -0
  69. package/dist/telemetry/tsed/index.d.ts +4 -0
  70. package/dist/telemetry/tsed/index.js +21 -0
  71. package/dist/telemetry/tsed/log-telemetry.d.ts +1 -0
  72. package/dist/telemetry/tsed/log-telemetry.js +196 -0
  73. package/dist/telemetry/tsed/service.d.ts +26 -0
  74. package/dist/telemetry/tsed/service.js +150 -0
  75. package/dist/telemetry/tsed/sync-log-record-processor.d.ts +11 -0
  76. package/dist/telemetry/tsed/sync-log-record-processor.js +74 -0
  77. package/dist/tsed/__tests__/config.test.d.ts +1 -0
  78. package/dist/tsed/__tests__/config.test.js +146 -0
  79. package/dist/tsed/__tests__/service.test.d.ts +1 -0
  80. package/dist/tsed/__tests__/service.test.js +63 -0
  81. package/dist/tsed/config.d.ts +26 -0
  82. package/dist/tsed/config.js +166 -0
  83. package/dist/tsed/index.d.ts +4 -0
  84. package/dist/tsed/index.js +21 -0
  85. package/dist/tsed/log-telemetry.d.ts +1 -0
  86. package/dist/tsed/log-telemetry.js +196 -0
  87. package/dist/tsed/service.d.ts +26 -0
  88. package/dist/tsed/service.js +150 -0
  89. package/dist/tsed/sync-log-record-processor.d.ts +11 -0
  90. package/dist/tsed/sync-log-record-processor.js +74 -0
  91. package/package.json +72 -56
  92. package/release.config.js +23 -23
  93. package/vitest.config.ts +22 -0
  94. package/dist/check-if-is-working.d.ts +0 -1
  95. package/dist/check-if-is-working.js +0 -8
package/README.md ADDED
@@ -0,0 +1,248 @@
1
+ # 📊 Telemetry Hub
2
+
3
+ Biblioteca centralizada para telemetria e logging estruturado padronizado, compatível com múltiplos frameworks (Tsed, SST/Middy).
4
+
5
+ ## 🔗 Quick Links
6
+
7
+ - **📘 Guias por Framework:**
8
+ - [Tsed](./src/telemetry/tsed/README.md)
9
+ - [SST/Middy](./src/telemetry/sst/README.md)
10
+ - **📖 Conceitos Core:**
11
+ - [Logging Padronizado](./src/telemetry/core/LOGGING.md)
12
+ - [Tracing Padronizado](./src/telemetry/core/TRACING.md)
13
+ - [Testes](./src/telemetry/core/__tests__/README.md)
14
+
15
+ ## 🎯 Objetivo
16
+
17
+ Padronizar a forma como logs são gerados em diferentes tipos de projetos, garantindo:
18
+ - **Consistência** nos campos de log entre diferentes implementações
19
+ - **Observabilidade** através do OpenTelemetry
20
+ - **Facilidade de manutenção** com uma única fonte de verdade
21
+
22
+ ## 📦 Instalação
23
+
24
+ ```bash
25
+ npm install @mecanizou/telemetry-hub
26
+ ```
27
+
28
+ ## 🏗️ Arquitetura
29
+
30
+ ```
31
+ src/telemetry/
32
+ ├── core/ # 🎯 Implementação base (StandardLogger, StandardTracer)
33
+ │ ├── types.ts # Interface StandardLogData
34
+ │ ├── logger.ts # StandardLogger
35
+ │ ├── tracer-types.ts # Interface StandardTraceData
36
+ │ ├── tracer.ts # StandardTracer
37
+ │ ├── index.ts # Exports
38
+ │ ├── README.md # 📚 Visão geral do Core
39
+ │ ├── LOGGING.md # 📝 Guia de logging
40
+ │ ├── TRACING.md # 🔍 Guia de tracing
41
+ │ └── __tests__/ # Testes (57 testes, 100% cobertura)
42
+ │ ├── logger.test.ts
43
+ │ ├── tracer.test.ts
44
+ │ ├── logger-types.test.ts
45
+ │ └── README.md # 📚 Documentação dos testes
46
+
47
+ ├── tsed/ # 🔷 Implementação para Tsed
48
+ │ ├── config.ts # TsedTelemetryProvider (logs + traces + métricas)
49
+ │ ├── service.ts # TsedTelemetryService
50
+ │ ├── log-telemetry.ts # Decorator @TsedLogTelemetry
51
+ │ ├── sync-log-record-processor.ts
52
+ │ ├── index.ts
53
+ │ ├── README.md # 📚 Guia de uso Tsed
54
+ │ ├── IMPLEMENTATION_SUMMARY.md # 📝 Resumo da implementação
55
+ │ └── SST_COMPARISON.md # 🔄 Comparação Tsed vs SST
56
+
57
+ └── sst/ # 🟦 Implementação para SST/Middy
58
+ ├── telemetry.ts # Configuração OpenTelemetry + getStandardLogger/Tracer
59
+ ├── index.ts
60
+ ├── README.md # 📚 Guia de uso SST
61
+ └── middy/
62
+ ├── middleware.ts # Middleware Middy (usa StandardLogger + StandardTracer)
63
+ └── index.ts
64
+ ```
65
+
66
+ ### Componentes Principais
67
+
68
+ #### 1. **StandardLogger** (Core)
69
+ - Classe base que formata logs no padrão OpenTelemetry
70
+ - Garante que todos os campos seguem a mesma estrutura
71
+ - Implementa os métodos: `logError`, `logInfo`, `logWarn`, `logDebug`
72
+
73
+ #### 2. **StandardLogData** (Interface)
74
+ Define o contrato padrão de log com campos:
75
+ - **severity**, **message**, **error**
76
+ - **serviceName**, **environment**
77
+ - **http** (method, url, endpoint, statusCode, etc.)
78
+ - **user** (accountUserUid, accountUid, applicationUid)
79
+ - **execution** (requestId, awsRequestId, functionName, controller, etc.)
80
+ - **performance** (durationMs, success)
81
+ - **context** (dados adicionais)
82
+
83
+ ## 🚀 Uso
84
+
85
+ ### Para Projetos Tsed
86
+
87
+ ```typescript
88
+ import {
89
+ TsedTelemetryProvider,
90
+ TsedTelemetryService,
91
+ TsedLogTelemetry
92
+ } from '@mecanizou/telemetry-hub';
93
+
94
+ // 1. Configurar provider (no módulo principal)
95
+ @Configuration({
96
+ // ...
97
+ })
98
+ export class Server {
99
+ @Inject()
100
+ protected telemetryProvider: TsedTelemetryProvider;
101
+
102
+ async $afterInit() {
103
+ await this.telemetryProvider.initialize();
104
+ }
105
+ }
106
+
107
+ // 2. Usar decorator nos controllers
108
+ export class CheckoutController {
109
+ @TsedLogTelemetry()
110
+ async getCheckout(@Context() $ctx: ServerlessContext) {
111
+ // Erros são capturados e logados automaticamente
112
+ return await this.service.execute();
113
+ }
114
+ }
115
+ ```
116
+
117
+ ### Para Projetos SST (Lambdas com Middy)
118
+
119
+ ```typescript
120
+ import { middyMiddleware } from '@mecanizou/telemetry-hub';
121
+ import middy from '@middy/core';
122
+
123
+ export const handler = middy(async (event, context) => {
124
+ // Sua lógica aqui
125
+ return { statusCode: 200, body: 'OK' };
126
+ }).use(middyMiddleware());
127
+ ```
128
+
129
+ O middleware automaticamente:
130
+ - Registra logs de INFO em caso de sucesso
131
+ - Registra logs de ERROR em caso de falha
132
+ - Mantém traces e métricas do OpenTelemetry
133
+
134
+ ## ⚙️ Configuração
135
+
136
+ ### Variáveis de Ambiente Obrigatórias
137
+
138
+ ```bash
139
+ OTEL_EXPORTER_OTLP_ENDPOINT=https://seu-endpoint-otel.com
140
+ OTEL_EXPORTER_OTLP_USER=seu-usuario
141
+ OTEL_EXPORTER_OTLP_PASS=sua-senha
142
+ ```
143
+
144
+ ### Variáveis de Ambiente Opcionais
145
+
146
+ ```bash
147
+ SERVICE_NAME=my-service # Default: 'unknown-service' (Tsed) ou 'sst-service' (SST)
148
+ STAGE=production # Default: 'development'
149
+ ```
150
+
151
+ ## ✅ Testes
152
+
153
+ O projeto possui **79 testes** com 100% de cobertura:
154
+
155
+ - **Core**: 57 testes (logger, tracer, tipos)
156
+ - **SST**: 9 testes (integração, singleton, forceFlush)
157
+ - **Tsed**: 13 testes (provider, service, integração)
158
+
159
+ Para executar:
160
+ ```bash
161
+ npm test # Executar todos os testes
162
+ npm run test:watch # Modo watch
163
+ npm run test:ui # UI interativa
164
+ npm run test:coverage # Relatório de cobertura
165
+ ```
166
+
167
+ ## 📝 Campos Padronizados de Log
168
+
169
+ Todos os logs exportam os seguintes atributos OpenTelemetry:
170
+
171
+ ```typescript
172
+ {
173
+ // Identificação
174
+ timestamp: "2024-12-05T10:30:00.000Z",
175
+ "service.name": "cart-service",
176
+ "deployment.environment.name": "production",
177
+
178
+ // Erro (quando aplicável)
179
+ "error.type": "ValidationError",
180
+ "error.message": "Invalid cart item",
181
+ "error.stack": "...",
182
+
183
+ // HTTP
184
+ "http.method": "POST",
185
+ "http.url": "/api/cart/checkout",
186
+ "http.endpoint": "/api/cart/checkout",
187
+ "http.status_code": 400,
188
+
189
+ // Usuário
190
+ "accountUser.uid": "user-123",
191
+ "account.uid": "account-456",
192
+ "application.uid": "app-789",
193
+
194
+ // Execução
195
+ "request.id": "req-abc",
196
+ "aws.request.id": "aws-xyz",
197
+ "faas.name": "checkoutHandler",
198
+ "faas.invocation_id": "inv-123",
199
+ "controller.name": "CheckoutController",
200
+ "controller.method": "getCheckout",
201
+ "origin": "mobile",
202
+
203
+ // Performance
204
+ "duration.ms": 234,
205
+ "execution.success": true,
206
+
207
+ // Contexto adicional
208
+ "context": "{...}"
209
+ }
210
+ ```
211
+
212
+ ## 🔄 Compatibilidade
213
+
214
+ | Framework | Versão Testada | Status |
215
+ |-----------|---------------|--------|
216
+ | Tsed | ^8.19.4 | ✅ |
217
+ | SST/Middy | ^6.4.5 | ✅ |
218
+ | OpenTelemetry | ^0.56.0 | ✅ |
219
+
220
+ ## 📚 Documentação Adicional
221
+
222
+ > 📖 **[Guia de Navegação da Documentação](./DOCS_GUIDE.md)** - Mapa completo de toda documentação disponível
223
+
224
+ ### Core (Conceitos e Implementação Base)
225
+ - [Core README](./src/telemetry/core/README.md) - Visão geral do StandardLogger e StandardTracer
226
+ - [LOGGING.md](./src/telemetry/core/LOGGING.md) - Guia detalhado de logging estruturado
227
+ - [TRACING.md](./src/telemetry/core/TRACING.md) - Guia detalhado de traces padronizados
228
+ - [Testes](./src/telemetry/core/__tests__/README.md) - Documentação dos testes (57 tests, 100% coverage)
229
+
230
+ ### Por Framework
231
+ - **Tsed:** [src/telemetry/tsed/README.md](./src/telemetry/tsed/README.md)
232
+ - [Resumo da Implementação](./src/telemetry/tsed/IMPLEMENTATION_SUMMARY.md)
233
+ - [Comparação Tsed vs SST](./src/telemetry/tsed/SST_COMPARISON.md)
234
+ - **SST:** [src/telemetry/sst/README.md](./src/telemetry/sst/README.md)
235
+
236
+ ## 🤝 Contribuindo
237
+
238
+ Esta é uma lib interna da Mecanizou. Para contribuir:
239
+
240
+ 1. Clone o repositório
241
+ 2. Instale dependências: `npm install`
242
+ 3. Execute testes: `npm test`
243
+ 4. Execute verificação de tipo: `npm run test:tsc`
244
+ 5. Formate código: `npm run fix`
245
+
246
+ ## 📄 Licença
247
+
248
+ ISC
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,325 @@
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
+ const vitest_1 = require("vitest");
13
+ (0, vitest_1.describe)('Types - StandardLogData', () => {
14
+ (0, vitest_1.it)('deve aceitar log mínimo válido', () => {
15
+ const minimalLog = {
16
+ severity: 'INFO',
17
+ message: 'Test message',
18
+ serviceName: 'test-service',
19
+ environment: 'test',
20
+ timestamp: new Date().toISOString(),
21
+ };
22
+ (0, vitest_1.expect)(minimalLog).toBeDefined();
23
+ (0, vitest_1.expect)(minimalLog.severity).toBe('INFO');
24
+ });
25
+ (0, vitest_1.it)('deve aceitar todos os níveis de severity', () => {
26
+ const severities = [
27
+ 'ERROR',
28
+ 'WARN',
29
+ 'INFO',
30
+ 'DEBUG',
31
+ ];
32
+ severities.forEach((severity) => {
33
+ const log = {
34
+ severity,
35
+ message: 'Test',
36
+ serviceName: 'test',
37
+ environment: 'test',
38
+ timestamp: new Date().toISOString(),
39
+ };
40
+ (0, vitest_1.expect)(log.severity).toBe(severity);
41
+ });
42
+ });
43
+ (0, vitest_1.it)('deve aceitar log completo com todos os campos opcionais', () => {
44
+ const completeLog = {
45
+ severity: 'ERROR',
46
+ message: 'Complete log message',
47
+ error: new Error('Test error'),
48
+ serviceName: 'complete-service',
49
+ environment: 'production',
50
+ http: {
51
+ method: 'POST',
52
+ url: 'https://api.example.com/test',
53
+ endpoint: '/test',
54
+ statusCode: 500,
55
+ headers: { 'content-type': 'application/json' },
56
+ body: { test: 'data' },
57
+ params: { id: '123' },
58
+ query: { filter: 'active' },
59
+ },
60
+ user: {
61
+ accountUserUid: 'user-123',
62
+ accountUid: 'account-456',
63
+ applicationUid: 'app-789',
64
+ },
65
+ execution: {
66
+ requestId: 'req-abc',
67
+ awsRequestId: 'aws-xyz',
68
+ functionName: 'testFunction',
69
+ invocationId: 'inv-123',
70
+ controller: 'TestController',
71
+ controllerMethod: 'testMethod',
72
+ origin: 'mobile',
73
+ },
74
+ performance: {
75
+ durationMs: 1500,
76
+ success: false,
77
+ },
78
+ context: {
79
+ customField: 'customValue',
80
+ nested: {
81
+ data: 123,
82
+ },
83
+ },
84
+ timestamp: new Date().toISOString(),
85
+ };
86
+ (0, vitest_1.expect)(completeLog).toBeDefined();
87
+ (0, vitest_1.expect)(completeLog.http).toBeDefined();
88
+ (0, vitest_1.expect)(completeLog.user).toBeDefined();
89
+ (0, vitest_1.expect)(completeLog.execution).toBeDefined();
90
+ (0, vitest_1.expect)(completeLog.performance).toBeDefined();
91
+ (0, vitest_1.expect)(completeLog.context).toBeDefined();
92
+ });
93
+ (0, vitest_1.it)('deve aceitar campos HTTP parciais', () => {
94
+ var _a, _b, _c;
95
+ const partialHttpLog = {
96
+ severity: 'INFO',
97
+ message: 'Partial HTTP',
98
+ serviceName: 'test',
99
+ environment: 'test',
100
+ http: {
101
+ method: 'GET',
102
+ endpoint: '/api/users',
103
+ },
104
+ timestamp: new Date().toISOString(),
105
+ };
106
+ (0, vitest_1.expect)((_a = partialHttpLog.http) === null || _a === void 0 ? void 0 : _a.method).toBe('GET');
107
+ (0, vitest_1.expect)((_b = partialHttpLog.http) === null || _b === void 0 ? void 0 : _b.endpoint).toBe('/api/users');
108
+ (0, vitest_1.expect)((_c = partialHttpLog.http) === null || _c === void 0 ? void 0 : _c.url).toBeUndefined();
109
+ });
110
+ (0, vitest_1.it)('deve aceitar campos de usuário parciais', () => {
111
+ var _a, _b;
112
+ const partialUserLog = {
113
+ severity: 'INFO',
114
+ message: 'Partial user',
115
+ serviceName: 'test',
116
+ environment: 'test',
117
+ user: {
118
+ accountUserUid: 'user-123',
119
+ },
120
+ timestamp: new Date().toISOString(),
121
+ };
122
+ (0, vitest_1.expect)((_a = partialUserLog.user) === null || _a === void 0 ? void 0 : _a.accountUserUid).toBe('user-123');
123
+ (0, vitest_1.expect)((_b = partialUserLog.user) === null || _b === void 0 ? void 0 : _b.accountUid).toBeUndefined();
124
+ });
125
+ (0, vitest_1.it)('deve aceitar campos de execução parciais', () => {
126
+ var _a, _b, _c;
127
+ const partialExecLog = {
128
+ severity: 'DEBUG',
129
+ message: 'Partial execution',
130
+ serviceName: 'test',
131
+ environment: 'test',
132
+ execution: {
133
+ requestId: 'req-123',
134
+ controller: 'TestController',
135
+ },
136
+ timestamp: new Date().toISOString(),
137
+ };
138
+ (0, vitest_1.expect)((_a = partialExecLog.execution) === null || _a === void 0 ? void 0 : _a.requestId).toBe('req-123');
139
+ (0, vitest_1.expect)((_b = partialExecLog.execution) === null || _b === void 0 ? void 0 : _b.controller).toBe('TestController');
140
+ (0, vitest_1.expect)((_c = partialExecLog.execution) === null || _c === void 0 ? void 0 : _c.awsRequestId).toBeUndefined();
141
+ });
142
+ (0, vitest_1.it)('deve aceitar performance com apenas durationMs', () => {
143
+ var _a, _b;
144
+ const durationOnlyLog = {
145
+ severity: 'INFO',
146
+ message: 'Duration only',
147
+ serviceName: 'test',
148
+ environment: 'test',
149
+ performance: {
150
+ durationMs: 500,
151
+ },
152
+ timestamp: new Date().toISOString(),
153
+ };
154
+ (0, vitest_1.expect)((_a = durationOnlyLog.performance) === null || _a === void 0 ? void 0 : _a.durationMs).toBe(500);
155
+ (0, vitest_1.expect)((_b = durationOnlyLog.performance) === null || _b === void 0 ? void 0 : _b.success).toBeUndefined();
156
+ });
157
+ (0, vitest_1.it)('deve aceitar performance com apenas success', () => {
158
+ var _a, _b;
159
+ const successOnlyLog = {
160
+ severity: 'INFO',
161
+ message: 'Success only',
162
+ serviceName: 'test',
163
+ environment: 'test',
164
+ performance: {
165
+ success: true,
166
+ },
167
+ timestamp: new Date().toISOString(),
168
+ };
169
+ (0, vitest_1.expect)((_a = successOnlyLog.performance) === null || _a === void 0 ? void 0 : _a.success).toBe(true);
170
+ (0, vitest_1.expect)((_b = successOnlyLog.performance) === null || _b === void 0 ? void 0 : _b.durationMs).toBeUndefined();
171
+ });
172
+ (0, vitest_1.it)('deve aceitar contexto com estruturas aninhadas complexas', () => {
173
+ var _a, _b;
174
+ const complexContextLog = {
175
+ severity: 'ERROR',
176
+ message: 'Complex context',
177
+ serviceName: 'test',
178
+ environment: 'test',
179
+ context: {
180
+ level1: {
181
+ level2: {
182
+ level3: {
183
+ value: 'deep',
184
+ array: [1, 2, 3],
185
+ },
186
+ },
187
+ },
188
+ simpleValue: 'test',
189
+ numericValue: 123,
190
+ booleanValue: true,
191
+ },
192
+ timestamp: new Date().toISOString(),
193
+ };
194
+ (0, vitest_1.expect)((_a = complexContextLog.context) === null || _a === void 0 ? void 0 : _a.level1).toBeDefined();
195
+ (0, vitest_1.expect)((_b = complexContextLog.context) === null || _b === void 0 ? void 0 : _b.simpleValue).toBe('test');
196
+ });
197
+ });
198
+ (0, vitest_1.describe)('Types - IStandardLogger', () => {
199
+ (0, vitest_1.it)('deve definir interface com os 4 métodos de log', () => {
200
+ const mockLogger = {
201
+ logError: () => __awaiter(void 0, void 0, void 0, function* () { }),
202
+ logInfo: () => __awaiter(void 0, void 0, void 0, function* () { }),
203
+ logWarn: () => __awaiter(void 0, void 0, void 0, function* () { }),
204
+ logDebug: () => __awaiter(void 0, void 0, void 0, function* () { }),
205
+ };
206
+ (0, vitest_1.expect)(mockLogger.logError).toBeDefined();
207
+ (0, vitest_1.expect)(mockLogger.logInfo).toBeDefined();
208
+ (0, vitest_1.expect)(mockLogger.logWarn).toBeDefined();
209
+ (0, vitest_1.expect)(mockLogger.logDebug).toBeDefined();
210
+ });
211
+ (0, vitest_1.it)('deve aceitar dados sem severity e timestamp nos métodos', () => __awaiter(void 0, void 0, void 0, function* () {
212
+ const mockLogger = {
213
+ logError: (data) => __awaiter(void 0, void 0, void 0, function* () {
214
+ (0, vitest_1.expect)(data.message).toBe('Error message');
215
+ (0, vitest_1.expect)(data.serviceName).toBe('test');
216
+ (0, vitest_1.expect)(data.severity).toBeUndefined();
217
+ (0, vitest_1.expect)(data.timestamp).toBeUndefined();
218
+ }),
219
+ logInfo: () => __awaiter(void 0, void 0, void 0, function* () { }),
220
+ logWarn: () => __awaiter(void 0, void 0, void 0, function* () { }),
221
+ logDebug: () => __awaiter(void 0, void 0, void 0, function* () { }),
222
+ };
223
+ yield mockLogger.logError({
224
+ message: 'Error message',
225
+ serviceName: 'test',
226
+ environment: 'test',
227
+ });
228
+ }));
229
+ });
230
+ (0, vitest_1.describe)('Types - Validação de campos', () => {
231
+ (0, vitest_1.describe)('severity', () => {
232
+ (0, vitest_1.it)('deve aceitar apenas valores válidos', () => {
233
+ const validSeverities = ['ERROR', 'WARN', 'INFO', 'DEBUG'];
234
+ validSeverities.forEach((severity) => {
235
+ const log = {
236
+ severity,
237
+ message: 'test',
238
+ serviceName: 'test',
239
+ environment: 'test',
240
+ timestamp: new Date().toISOString(),
241
+ };
242
+ (0, vitest_1.expect)(log.severity).toBe(severity);
243
+ });
244
+ });
245
+ });
246
+ (0, vitest_1.describe)('environment', () => {
247
+ (0, vitest_1.it)('deve aceitar ambientes comuns', () => {
248
+ const environments = ['development', 'staging', 'production', 'test'];
249
+ environments.forEach((env) => {
250
+ const log = {
251
+ severity: 'INFO',
252
+ message: 'test',
253
+ serviceName: 'test',
254
+ environment: env,
255
+ timestamp: new Date().toISOString(),
256
+ };
257
+ (0, vitest_1.expect)(log.environment).toBe(env);
258
+ });
259
+ });
260
+ });
261
+ (0, vitest_1.describe)('timestamp', () => {
262
+ (0, vitest_1.it)('deve aceitar formato ISO 8601', () => {
263
+ const isoTimestamp = new Date().toISOString();
264
+ const log = {
265
+ severity: 'INFO',
266
+ message: 'test',
267
+ serviceName: 'test',
268
+ environment: 'test',
269
+ timestamp: isoTimestamp,
270
+ };
271
+ (0, vitest_1.expect)(log.timestamp).toMatch(/^\d{4}-\d{2}-\d{2}T\d{2}:\d{2}:\d{2}\.\d{3}Z$/);
272
+ });
273
+ });
274
+ (0, vitest_1.describe)('http.statusCode', () => {
275
+ (0, vitest_1.it)('deve aceitar códigos HTTP válidos', () => {
276
+ const statusCodes = [200, 201, 400, 401, 404, 500, 502, 503];
277
+ statusCodes.forEach((statusCode) => {
278
+ var _a;
279
+ const log = {
280
+ severity: 'INFO',
281
+ message: 'test',
282
+ serviceName: 'test',
283
+ environment: 'test',
284
+ http: { statusCode },
285
+ timestamp: new Date().toISOString(),
286
+ };
287
+ (0, vitest_1.expect)((_a = log.http) === null || _a === void 0 ? void 0 : _a.statusCode).toBe(statusCode);
288
+ });
289
+ });
290
+ });
291
+ (0, vitest_1.describe)('performance.durationMs', () => {
292
+ (0, vitest_1.it)('deve aceitar valores numéricos positivos', () => {
293
+ const durations = [0, 100, 500, 1000, 5000, 10000];
294
+ durations.forEach((durationMs) => {
295
+ var _a;
296
+ const log = {
297
+ severity: 'INFO',
298
+ message: 'test',
299
+ serviceName: 'test',
300
+ environment: 'test',
301
+ performance: { durationMs },
302
+ timestamp: new Date().toISOString(),
303
+ };
304
+ (0, vitest_1.expect)((_a = log.performance) === null || _a === void 0 ? void 0 : _a.durationMs).toBe(durationMs);
305
+ });
306
+ });
307
+ });
308
+ (0, vitest_1.describe)('performance.success', () => {
309
+ (0, vitest_1.it)('deve aceitar valores booleanos', () => {
310
+ [true, false].forEach((success) => {
311
+ var _a;
312
+ const log = {
313
+ severity: 'INFO',
314
+ message: 'test',
315
+ serviceName: 'test',
316
+ environment: 'test',
317
+ performance: { success },
318
+ timestamp: new Date().toISOString(),
319
+ };
320
+ (0, vitest_1.expect)((_a = log.performance) === null || _a === void 0 ? void 0 : _a.success).toBe(success);
321
+ });
322
+ });
323
+ });
324
+ });
325
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibG9nZ2VyLXR5cGVzLnRlc3QuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvY29yZS9fX3Rlc3RzX18vbG9nZ2VyLXR5cGVzLnRlc3QudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7QUFBQSxtQ0FBOEM7QUFHOUMsSUFBQSxpQkFBUSxFQUFDLHlCQUF5QixFQUFFLEdBQUcsRUFBRTtJQUN2QyxJQUFBLFdBQUUsRUFBQyxnQ0FBZ0MsRUFBRSxHQUFHLEVBQUU7UUFDeEMsTUFBTSxVQUFVLEdBQW9CO1lBQ2xDLFFBQVEsRUFBRSxNQUFNO1lBQ2hCLE9BQU8sRUFBRSxjQUFjO1lBQ3ZCLFdBQVcsRUFBRSxjQUFjO1lBQzNCLFdBQVcsRUFBRSxNQUFNO1lBQ25CLFNBQVMsRUFBRSxJQUFJLElBQUksRUFBRSxDQUFDLFdBQVcsRUFBRTtTQUNwQyxDQUFDO1FBRUYsSUFBQSxlQUFNLEVBQUMsVUFBVSxDQUFDLENBQUMsV0FBVyxFQUFFLENBQUM7UUFDakMsSUFBQSxlQUFNLEVBQUMsVUFBVSxDQUFDLFFBQVEsQ0FBQyxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsQ0FBQztJQUMzQyxDQUFDLENBQUMsQ0FBQztJQUVILElBQUEsV0FBRSxFQUFDLDBDQUEwQyxFQUFFLEdBQUcsRUFBRTtRQUNsRCxNQUFNLFVBQVUsR0FBa0M7WUFDaEQsT0FBTztZQUNQLE1BQU07WUFDTixNQUFNO1lBQ04sT0FBTztTQUNSLENBQUM7UUFFRixVQUFVLENBQUMsT0FBTyxDQUFDLENBQUMsUUFBUSxFQUFFLEVBQUU7WUFDOUIsTUFBTSxHQUFHLEdBQW9CO2dCQUMzQixRQUFRO2dCQUNSLE9BQU8sRUFBRSxNQUFNO2dCQUNmLFdBQVcsRUFBRSxNQUFNO2dCQUNuQixXQUFXLEVBQUUsTUFBTTtnQkFDbkIsU0FBUyxFQUFFLElBQUksSUFBSSxFQUFFLENBQUMsV0FBVyxFQUFFO2FBQ3BDLENBQUM7WUFDRixJQUFBLGVBQU0sRUFBQyxHQUFH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QUFDO2dCQUNGLElBQUEsZUFBTSxFQUFDLE1BQUEsR0FBRyxDQUFDLFdBQVcsMENBQUUsVUFBVSxDQUFDLENBQUMsSUFBSSxDQUFDLFVBQVUsQ0FBQyxDQUFDO1lBQ3ZELENBQUMsQ0FBQyxDQUFDO1FBQ0wsQ0FBQyxDQUFDLENBQUM7SUFDTCxDQUFDLENBQUMsQ0FBQztJQUVILElBQUEsaUJBQVEsRUFBQyxxQkFBcUIsRUFBRSxHQUFHLEVBQUU7UUFDbkMsSUFBQSxXQUFFLEVBQUMsZ0NBQWdDLEVBQUUsR0FBRyxFQUFFO1lBQ3hDLENBQUMsSUFBSSxFQUFFLEtBQUssQ0FBQyxDQUFDLE9BQU8sQ0FBQyxDQUFDLE9BQU8sRUFBRSxFQUFFOztnQkFDaEMsTUFBTSxHQUFHLEdBQW9CO29CQUMzQixRQUFRLEVBQUUsTUFBTTtvQkFDaEIsT0FBTyxFQUFFLE1BQU07b0JBQ2YsV0FBVyxFQUFFLE1BQU07b0JBQ25CLFdBQVcsRUFBRSxNQUFNO29CQUNuQixXQUFXLEVBQUUsRUFBRSxPQUFPLEVBQUU7b0JBQ3hCLFNBQVMsRUFBRSxJQUFJLElBQUksRUFBRSxDQUFDLFdBQVcsRUFBRTtpQkFDcEMsQ0FBQztnQkFDRixJQUFBLGVBQU0sRUFBQyxNQUFBLEdBQUcsQ0FBQyxXQUFXLDBDQUFFLE9BQU8sQ0FBQyxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsQ0FBQztZQUNqRCxDQUFDLENBQUMsQ0FBQztRQUNMLENBQUMsQ0FBQyxDQUFDO0lBQ0wsQ0FBQyxDQUFDLENBQUM7QUFDTCxDQUFDLENBQUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGRlc2NyaWJlLCBpdCwgZXhwZWN0IH0gZnJvbSAndml0ZXN0JztcclxuaW1wb3J0IHR5cGUgeyBTdGFuZGFyZExvZ0RhdGEsIElTdGFuZGFyZExvZ2dlciB9IGZyb20gJy4uL2xvZ2dlci10eXBlcyc7XHJcblxyXG5kZXNjcmliZSgnVHlwZXMgLSBTdGFuZGFyZExvZ0RhdGEnLCAoKSA9PiB7XHJcbiAgaXQoJ2RldmUgYWNlaXRhciBsb2cgbcOtbmltbyB2w6FsaWRvJywgKCkgPT4ge1xyXG4gICAgY29uc3QgbWluaW1hbExvZzogU3RhbmRhcmRMb2dEYXRhID0ge1xyXG4gICAgICBzZXZlcml0eTogJ0lORk8nLFxyXG4gICAgICBtZXNzYWdlOiAnVGVzdCBtZXNzYWdlJyxcclxuICAgICAgc2VydmljZU5hbWU6ICd0ZXN0LXNlcnZpY2UnLFxyXG4gICAgICBlbnZpcm9ubWVudDogJ3Rlc3QnLFxyXG4gICAgICB0aW1lc3RhbXA6IG5ldyBEYXRlKCkudG9JU09TdHJpbmcoKSxcclxuICAgIH07XHJcblxyXG4gICAgZXhwZWN0KG1pbmltYWxMb2cpLnRvQmVEZWZpbmVkKCk7XHJcbiAgICBleHBlY3QobWluaW1hbExvZy5zZXZlcml0eSkudG9CZSgnSU5GTycpO1xyXG4gIH0pO1xyXG5cclxuICBpdCgnZGV2ZSBhY2VpdGFyIHRvZG9zIG9zIG7DrXZlaXMgZGUgc2V2ZXJpdHknLCAoKSA9PiB7XHJcbiAgICBjb25zdCBzZXZlcml0aWVzOiBTdGFuZGFyZExvZ0RhdGFbJ3NldmVyaXR5J11bXSA9IFtcclxuICAgICAgJ0VSUk9SJyxcclxuICAgICAgJ1dBUk4nLFxyXG4gICAgICAnSU5GTycsXHJcbiAgICAgICdERUJVRycsXHJcbiAgICBdO1xyXG5cclxuICAgIHNldmVyaXRpZXMuZm9yRWFjaCgoc2V2ZXJpdHkpID0+IHtcclxuICAgICAgY29uc3QgbG9nOiBTdGFuZGFyZExvZ0RhdGEgPSB7XHJcbiAgICAgICAgc2V2ZXJpdHksXHJcbiAgICAgICAgbWVzc2FnZTogJ1Rlc3QnLFxyXG4gICAgICAgIHNlcnZpY2VOYW1lOiAndGVzdCcsXHJcbiAgICAgICAgZW52aXJvbm1lbnQ6ICd0ZXN0JyxcclxuICAgICAgICB0aW1lc3RhbXA6IG5ldyBEYXRlKCkudG9JU09TdHJpbmcoKSxcclxuICAgICAgfTtcclxuICAgICAgZXhwZWN0KGxvZy5zZXZlcml0eSkudG9CZShzZXZlcml0eSk7XHJcbiAgICB9KTtcclxuICB9KTtcclxuXHJcbiAgaXQoJ2RldmUgYWNlaXRhciBsb2cgY29tcGxldG8gY29tIHRvZG9zIG9zIGNhbXBvcyBvcGNpb25haXMnLCAoKSA9PiB7XHJcbiAgICBjb25zdCBjb21wbGV0ZUxvZzogU3RhbmRhcmRMb2dEYXRhID0ge1xyXG4gICAgICBzZXZlcml0eTogJ0VSUk9SJyxcclxuICAgICAgbWVzc2FnZTogJ0NvbXBsZXRlIGxvZyBtZXNzYWdlJyxcclxuICAgICAgZXJyb3I6IG5ldyBFcnJvcignVGVzdCBlcnJvcicpLFxyXG4gICAgICBzZXJ2aWNlTmFtZTogJ2NvbXBsZXRlLXNlcnZpY2UnLFxyXG4gICAgICBlbnZpcm9ubWVudDogJ3Byb2R1Y3Rpb24nLFxyXG4gICAgICBodHRwOiB7XHJcbiAgICAgICAgbWV0aG9kOiAnUE9TVCcsXHJcbiAgICAgICAgdXJsOiAnaHR0cHM6Ly9hcGkuZXhhbXBsZS5jb20vdGVzdCcsXHJcbiAgICAgICAgZW5kcG9pbnQ6ICcvdGVzdCcsXHJcbiAgICAgICAgc3RhdHVzQ29kZTogNTAwLFxyXG4gICAgICAgIGhlYWRlcnM6IHsgJ2NvbnRlbnQtdHlwZSc6ICdhcHBsaWNhdGlvbi9qc29uJyB9LFxyXG4gICAgICAgIGJvZHk6IHsgdGVzdDogJ2RhdGEnIH0sXHJcbiAgICAgICAgcGFyYW1zOiB7IGlkOiAnMTIzJyB9LFxyXG4gICAgICAgIHF1ZXJ5OiB7IGZpbHRlcjogJ2FjdGl2ZScgfSxcclxuICAgICAgfSxcclxuICAgICAgdXNlcjoge1xyXG4gICAgICAgIGFjY291bnRVc2VyVWlkOiAndXNlci0xMjMnLFxyXG4gICAgICAgIGFjY291bnRVaWQ6ICdhY2NvdW50LTQ1NicsXHJcbiAgICAgICAgYXBwbGljYXRpb25VaWQ6ICdhcHAtNzg5JyxcclxuICAgICAgfSxcclxuICAgICAgZXhlY3V0aW9uOiB7XHJcbiAgICAgICAgcmVxdWVzdElkOiAncmVxLWFiYycsXHJcbiAgICAgICAgYXdzUmVxdWVzdElkOiAnYXdzLXh5eicsXHJcbiAgICAgICAgZnVuY3Rpb25OYW1lOiAndGVzdEZ1bmN0aW9uJyxcclxuICAgICAgICBpbnZvY2F0aW9uSWQ6ICdpbnYtMTIzJyxcclxuICAgICAgICBjb250cm9sbGVyOiAnVGVzdENvbnRyb2xsZXInLFxyXG4gICAgICAgIGNvbnRyb2xsZXJNZXRob2Q6ICd0ZXN0TWV0aG9kJyxcclxuICAgICAgICBvcmlnaW46ICdtb2JpbGUnLFxyXG4gICAgICB9LFxyXG4gICAgICBwZXJmb3JtYW5jZToge1xyXG4gICAgICAgIGR1cmF0aW9uTXM6IDE1MDAsXHJcbiAgICAgICAgc3VjY2VzczogZmFsc2UsXHJcbiAgICAgIH0sXHJcbiAgICAgIGNvbnRleHQ6IHtcclxuICAgICAgICBjdXN0b21GaWVsZDogJ2N1c3RvbVZhbHVlJyxcclxuICAgICAgICBuZXN0ZWQ6IHtcclxuICAgICAgICAgIGRhdGE6IDEyMyxcclxuICAgICAgICB9LFxyXG4gICAgICB9LFxyXG4gICAgICB0aW1lc3RhbXA6IG5ldyBEYXRlKCkudG9JU09TdHJpbmcoKSxcclxuICAgIH07XHJcblxyXG4gICAgZXhwZWN0KGNvbXBsZXRlTG9nKS50b0JlRGVmaW5lZCgpO1xyXG4gICAgZXhwZWN0KGNvbXBsZXRlTG9nLmh0dHApLnRvQmVEZWZpbmVkKCk7XHJcbiAgICBleHBlY3QoY29tcGxldGVMb2cudXNlcikudG9CZURlZmluZWQoKTtcclxuICAgIGV4cGVjdChjb21wbGV0ZUxvZy5leGVjdXRpb24pLnRvQmVEZWZpbmVkKCk7XHJcbiAgICBleHBlY3QoY29tcGxldGVMb2cucGVyZm9ybWFuY2UpLnRvQmVEZWZpbmVkKCk7XHJcbiAgICBleHBlY3QoY29tcGxldGVMb2cuY29udGV4dCkudG9CZURlZmluZWQoKTtcclxuICB9KTtcclxuXHJcbiAgaXQoJ2RldmUgYWNlaXRhciBjYW1wb3MgSFRUUCBwYXJjaWFpcycsICgpID0+IHtcclxuICAgIGNvbnN0IHBhcnRpYWxIdHRwTG9nOiBTdGFuZGFyZExvZ0RhdGEgPSB7XHJcbiAgICAgIHNldmVyaXR5OiAnSU5GTycsXHJcbiAgICAgIG1lc3NhZ2U6ICdQYXJ0aWFsIEhUVFAnLFxyXG4gICAgICBzZXJ2aWNlTmFtZTogJ3Rlc3QnLFxyXG4gICAgICBlbnZpcm9ubWVudDogJ3Rlc3QnLFxyXG4gICAgICBodHRwOiB7XHJcbiAgICAgICAgbWV0aG9kOiAnR0VUJyxcclxuICAgICAgICBlbmRwb2ludDogJy9hcGkvdXNlcnMnLFxyXG4gICAgICAgIC8vIG91dHJvcyBjYW1wb3Mgb3BjaW9uYWlzIG9taXRpZG9zXHJcbiAgICAgIH0sXHJcbiAgICAgIHRpbWVzdGFtcDogbmV3IERhdGUoKS50b0lTT1N0cmluZygpLFxyXG4gICAgfTtcclxuXHJcbiAgICBleHBlY3QocGFydGlhbEh0dHBMb2cuaHR0cD8ubWV0aG9kKS50b0JlKCdHRVQnKTtcclxuICAgIGV4cGVjdChwYXJ0aWFsSHR0cExvZy5odHRwPy5lbmRwb2ludCkudG9CZSgnL2FwaS91c2VycycpO1xyXG4gICAgZXhwZWN0KHBhcnRpYWxIdHRwTG9nLmh0dHA/LnVybCkudG9CZVVuZGVmaW5lZCgpO1xyXG4gIH0pO1xyXG5cclxuICBpdCgnZGV2ZSBhY2VpdGFyIGNhbXBvcyBkZSB1c3XDoXJpbyBwYXJjaWFpcycsICgpID0+IHtcclxuICAgIGNvbnN0IHBhcnRpYWxVc2VyTG9nOiBTdGFuZGFyZExvZ0RhdGEgPSB7XHJcbiAgICAgIHNldmVyaXR5OiAnSU5GTycsXHJcbiAgICAgIG1lc3NhZ2U6ICdQYXJ0aWFsIHVzZXInLFxyXG4gICAgICBzZXJ2aWNlTmFtZTogJ3Rlc3QnLFxyXG4gICAgICBlbnZpcm9ubWVudDogJ3Rlc3QnLFxyXG4gICAgICB1c2VyOiB7XHJcbiAgICAgICAgYWNjb3VudFVzZXJVaWQ6ICd1c2VyLTEyMycsXHJcbiAgICAgICAgLy8gb3V0cm9zIGNhbXBvcyBvcGNpb25haXMgb21pdGlkb3NcclxuICAgICAgfSxcclxuICAgICAgdGltZXN0YW1wOiBuZXcgRGF0ZSgpLnRvSVNPU3RyaW5nKCksXHJcbiAgICB9O1xyXG5cclxuICAgIGV4cGVjdChwYXJ0aWFsVXNlckxvZy51c2VyPy5hY2NvdW50VXNlclVpZCkudG9CZSgndXNlci0xMjMnKTtcclxuICAgIGV4cGVjdChwYXJ0aWFsVXNlckxvZy51c2VyPy5hY2NvdW50VWlkKS50b0JlVW5kZWZpbmVkKCk7XHJcbiAgfSk7XHJcblxyXG4gIGl0KCdkZXZlIGFjZWl0YXIgY2FtcG9zIGRlIGV4ZWN1w6fDo28gcGFyY2lhaXMnLCAoKSA9PiB7XHJcbiAgICBjb25zdCBwYXJ0aWFsRXhlY0xvZzogU3RhbmRhcmRMb2dEYXRhID0ge1xyXG4gICAgICBzZXZlcml0eTogJ0RFQlVHJyxcclxuICAgICAgbWVzc2FnZTogJ1BhcnRpYWwgZXhlY3V0aW9uJyxcclxuICAgICAgc2VydmljZU5hbWU6ICd0ZXN0JyxcclxuICAgICAgZW52aXJvbm1lbnQ6ICd0ZXN0JyxcclxuICAgICAgZXhlY3V0aW9uOiB7XHJcbiAgICAgICAgcmVxdWVzdElkOiAncmVxLTEyMycsXHJcbiAgICAgICAgY29udHJvbGxlcjogJ1Rlc3RDb250cm9sbGVyJyxcclxuICAgICAgICAvLyBvdXRyb3MgY2FtcG9zIG9wY2lvbmFpcyBvbWl0aWRvc1xyXG4gICAgICB9LFxyXG4gICAgICB0aW1lc3RhbXA6IG5ldyBEYXRlKCkudG9JU09TdHJpbmcoKSxcclxuICAgIH07XHJcblxyXG4gICAgZXhwZWN0KHBhcnRpYWxFeGVjTG9nLmV4ZWN1dGlvbj8ucmVxdWVzdElkKS50b0JlKCdyZXEtMTIzJyk7XHJcbiAgICBleHBlY3QocGFydGlhbEV4ZWNMb2cuZXhlY3V0aW9uPy5jb250cm9sbGVyKS50b0JlKCdUZXN0Q29udHJvbGxlcicpO1xyXG4gICAgZXhwZWN0KHBhcnRpYWxFeGVjTG9nLmV4ZWN1dGlvbj8uYXdzUmVxdWVzdElkKS50b0JlVW5kZWZpbmVkKCk7XHJcbiAgfSk7XHJcblxyXG4gIGl0KCdkZXZlIGFjZWl0YXIgcGVyZm9ybWFuY2UgY29tIGFwZW5hcyBkdXJhdGlvbk1zJywgKCkgPT4ge1xyXG4gICAgY29uc3QgZHVyYXRpb25Pbmx5TG9nOiBTdGFuZGFyZExvZ0RhdGEgPSB7XHJcbiAgICAgIHNldmVyaXR5OiAnSU5GTycsXHJcbiAgICAgIG1lc3NhZ2U6ICdEdXJhdGlvbiBvbmx5JyxcclxuICAgICAgc2VydmljZU5hbWU6ICd0ZXN0JyxcclxuICAgICAgZW52aXJvbm1lbnQ6ICd0ZXN0JyxcclxuICAgICAgcGVyZm9ybWFuY2U6IHtcclxuICAgICAgICBkdXJhdGlvbk1zOiA1MDAsXHJcbiAgICAgIH0sXHJcbiAgICAgIHRpbWVzdGFtcDogbmV3IERhdGUoKS50b0lTT1N0cmluZygpLFxyXG4gICAgfTtcclxuXHJcbiAgICBleHBlY3QoZHVyYXRpb25Pbmx5TG9nLnBlcmZvcm1hbmNlPy5kdXJhdGlvbk1zKS50b0JlKDUwMCk7XHJcbiAgICBleHBlY3QoZHVyYXRpb25Pbmx5TG9nLnBlcmZvcm1hbmNlPy5zdWNjZXNzKS50b0JlVW5kZWZpbmVkKCk7XHJcbiAgfSk7XHJcblxyXG4gIGl0KCdkZXZlIGFjZWl0YXIgcGVyZm9ybWFuY2UgY29tIGFwZW5hcyBzdWNjZXNzJywgKCkgPT4ge1xyXG4gICAgY29uc3Qgc3VjY2Vzc09ubHlMb2c6IFN0YW5kYXJkTG9nRGF0YSA9IHtcclxuICAgICAgc2V2ZXJpdHk6ICdJTkZPJyxcclxuICAgICAgbWVzc2FnZTogJ1N1Y2Nlc3Mgb25seScsXHJcbiAgICAgIHNlcnZpY2VOYW1lOiAndGVzdCcsXHJcbiAgICAgIGVudmlyb25tZW50OiAndGVzdCcsXHJcbiAgICAgIHBlcmZvcm1hbmNlOiB7XHJcbiAgICAgICAgc3VjY2VzczogdHJ1ZSxcclxuICAgICAgfSxcclxuICAgICAgdGltZXN0YW1wOiBuZXcgRGF0ZSgpLnRvSVNPU3RyaW5nKCksXHJcbiAgICB9O1xyXG5cclxuICAgIGV4cGVjdChzdWNjZXNzT25seUxvZy5wZXJmb3JtYW5jZT8uc3VjY2VzcykudG9CZSh0cnVlKTtcclxuICAgIGV4cGVjdChzdWNjZXNzT25seUxvZy5wZXJmb3JtYW5jZT8uZHVyYXRpb25NcykudG9CZVVuZGVmaW5lZCgpO1xyXG4gIH0pO1xyXG5cclxuICBpdCgnZGV2ZSBhY2VpdGFyIGNvbnRleHRvIGNvbSBlc3RydXR1cmFzIGFuaW5oYWRhcyBjb21wbGV4YXMnLCAoKSA9PiB7XHJcbiAgICBjb25zdCBjb21wbGV4Q29udGV4dExvZzogU3RhbmRhcmRMb2dEYXRhID0ge1xyXG4gICAgICBzZXZlcml0eTogJ0VSUk9SJyxcclxuICAgICAgbWVzc2FnZTogJ0NvbXBsZXggY29udGV4dCcsXHJcbiAgICAgIHNlcnZpY2VOYW1lOiAndGVzdCcsXHJcbiAgICAgIGVudmlyb25tZW50OiAndGVzdCcsXHJcbiAgICAgIGNvbnRleHQ6IHtcclxuICAgICAgICBsZXZlbDE6IHtcclxuICAgICAgICAgIGxldmVsMjoge1xyXG4gICAgICAgICAgICBsZXZlbDM6IHtcclxuICAgICAgICAgICAgICB2YWx1ZTogJ2RlZXAnLFxyXG4gICAgICAgICAgICAgIGFycmF5OiBbMSwgMiwgM10sXHJcbiAgICAgICAgICAgIH0sXHJcbiAgICAgICAgICB9LFxyXG4gICAgICAgIH0sXHJcbiAgICAgICAgc2ltcGxlVmFsdWU6ICd0ZXN0JyxcclxuICAgICAgICBudW1lcmljVmFsdWU6IDEyMyxcclxuICAgICAgICBib29sZWFuVmFsdWU6IHRydWUsXHJcbiAgICAgIH0sXHJcbiAgICAgIHRpbWVzdGFtcDogbmV3IERhdGUoKS50b0lTT1N0cmluZygpLFxyXG4gICAgfTtcclxuXHJcbiAgICBleHBlY3QoY29tcGxleENvbnRleHRMb2cuY29udGV4dD8ubGV2ZWwxKS50b0JlRGVmaW5lZCgpO1xyXG4gICAgZXhwZWN0KGNvbXBsZXhDb250ZXh0TG9nLmNvbnRleHQ/LnNpbXBsZVZhbHVlKS50b0JlKCd0ZXN0Jyk7XHJcbiAgfSk7XHJcbn0pO1xyXG5cclxuZGVzY3JpYmUoJ1R5cGVzIC0gSVN0YW5kYXJkTG9nZ2VyJywgKCkgPT4ge1xyXG4gIGl0KCdkZXZlIGRlZmluaXIgaW50ZXJmYWNlIGNvbSBvcyA0IG3DqXRvZG9zIGRlIGxvZycsICgpID0+IHtcclxuICAgIC8vIEVzdGUgw6kgdW0gdGVzdGUgZGUgdGlwbywgc8OzIHBhcmEgZ2FyYW50aXIgcXVlIGEgaW50ZXJmYWNlIGVzdMOhIGNvcnJldGFcclxuICAgIGNvbnN0IG1vY2tMb2dnZXI6IElTdGFuZGFyZExvZ2dlciA9IHtcclxuICAgICAgbG9nRXJyb3I6IGFzeW5jICgpID0+IHt9LFxyXG4gICAgICBsb2dJbmZvOiBhc3luYyAoKSA9PiB7fSxcclxuICAgICAgbG9nV2FybjogYXN5bmMgKCkgPT4ge30sXHJcbiAgICAgIGxvZ0RlYnVnOiBhc3luYyAoKSA9PiB7fSxcclxuICAgIH07XHJcblxyXG4gICAgZXhwZWN0KG1vY2tMb2dnZXIubG9nRXJyb3IpLnRvQmVEZWZpbmVkKCk7XHJcbiAgICBleHBlY3QobW9ja0xvZ2dlci5sb2dJbmZvKS50b0JlRGVmaW5lZCgpO1xyXG4gICAgZXhwZWN0KG1vY2tMb2dnZXIubG9nV2FybikudG9CZURlZmluZWQoKTtcclxuICAgIGV4cGVjdChtb2NrTG9nZ2VyLmxvZ0RlYnVnKS50b0JlRGVmaW5lZCgpO1xyXG4gIH0pO1xyXG5cclxuICBpdCgnZGV2ZSBhY2VpdGFyIGRhZG9zIHNlbSBzZXZlcml0eSBlIHRpbWVzdGFtcCBub3MgbcOpdG9kb3MnLCBhc3luYyAoKSA9PiB7XHJcbiAgICBjb25zdCBtb2NrTG9nZ2VyOiBJU3RhbmRhcmRMb2dnZXIgPSB7XHJcbiAgICAgIGxvZ0Vycm9yOiBhc3luYyAoZGF0YSkgPT4ge1xyXG4gICAgICAgIGV4cGVjdChkYXRhLm1lc3NhZ2UpLnRvQmUoJ0Vycm9yIG1lc3NhZ2UnKTtcclxuICAgICAgICBleHBlY3QoZGF0YS5zZXJ2aWNlTmFtZSkudG9CZSgndGVzdCcpO1xyXG4gICAgICAgIC8vIEB0cy1leHBlY3QtZXJyb3IgLSBzZXZlcml0eSBuw6NvIGRldmUgZXhpc3RpclxyXG4gICAgICAgIGV4cGVjdChkYXRhLnNldmVyaXR5KS50b0JlVW5kZWZpbmVkKCk7XHJcbiAgICAgICAgLy8gQHRzLWV4cGVjdC1lcnJvciAtIHRpbWVzdGFtcCBuw6NvIGRldmUgZXhpc3RpclxyXG4gICAgICAgIGV4cGVjdChkYXRhLnRpbWVzdGFtcCkudG9CZVVuZGVmaW5lZCgpO1xyXG4gICAgICB9LFxyXG4gICAgICBsb2dJbmZvOiBhc3luYyAoKSA9PiB7fSxcclxuICAgICAgbG9nV2FybjogYXN5bmMgKCkgPT4ge30sXHJcbiAgICAgIGxvZ0RlYnVnOiBhc3luYyAoKSA9PiB7fSxcclxuICAgIH07XHJcblxyXG4gICAgYXdhaXQgbW9ja0xvZ2dlci5sb2dFcnJvcih7XHJcbiAgICAgIG1lc3NhZ2U6ICdFcnJvciBtZXNzYWdlJyxcclxuICAgICAgc2VydmljZU5hbWU6ICd0ZXN0JyxcclxuICAgICAgZW52aXJvbm1lbnQ6ICd0ZXN0JyxcclxuICAgIH0pO1xyXG4gIH0pO1xyXG59KTtcclxuXHJcbmRlc2NyaWJlKCdUeXBlcyAtIFZhbGlkYcOnw6NvIGRlIGNhbXBvcycsICgpID0+IHtcclxuICBkZXNjcmliZSgnc2V2ZXJpdHknLCAoKSA9PiB7XHJcbiAgICBpdCgnZGV2ZSBhY2VpdGFyIGFwZW5hcyB2YWxvcmVzIHbDoWxpZG9zJywgKCkgPT4ge1xyXG4gICAgICBjb25zdCB2YWxpZFNldmVyaXRpZXMgPSBbJ0VSUk9SJywgJ1dBUk4nLCAnSU5GTycsICdERUJVRyddIGFzIGNvbnN0O1xyXG5cclxuICAgICAgdmFsaWRTZXZlcml0aWVzLmZvckVhY2goKHNldmVyaXR5KSA9PiB7XHJcbiAgICAgICAgY29uc3QgbG9nOiBTdGFuZGFyZExvZ0RhdGEgPSB7XHJcbiAgICAgICAgICBzZXZlcml0eSxcclxuICAgICAgICAgIG1lc3NhZ2U6ICd0ZXN0JyxcclxuICAgICAgICAgIHNlcnZpY2VOYW1lOiAndGVzdCcsXHJcbiAgICAgICAgICBlbnZpcm9ubWVudDogJ3Rlc3QnLFxyXG4gICAgICAgICAgdGltZXN0YW1wOiBuZXcgRGF0ZSgpLnRvSVNPU3RyaW5nKCksXHJcbiAgICAgICAgfTtcclxuICAgICAgICBleHBlY3QobG9nLnNldmVyaXR5KS50b0JlKHNldmVyaXR5KTtcclxuICAgICAgfSk7XHJcbiAgICB9KTtcclxuICB9KTtcclxuXHJcbiAgZGVzY3JpYmUoJ2Vudmlyb25tZW50JywgKCkgPT4ge1xyXG4gICAgaXQoJ2RldmUgYWNlaXRhciBhbWJpZW50ZXMgY29tdW5zJywgKCkgPT4ge1xyXG4gICAgICBjb25zdCBlbnZpcm9ubWVudHMgPSBbJ2RldmVsb3BtZW50JywgJ3N0YWdpbmcnLCAncHJvZHVjdGlvbicsICd0ZXN0J107XHJcblxyXG4gICAgICBlbnZpcm9ubWVudHMuZm9yRWFjaCgoZW52KSA9PiB7XHJcbiAgICAgICAgY29uc3QgbG9nOiBTdGFuZGFyZExvZ0RhdGEgPSB7XHJcbiAgICAgICAgICBzZXZlcml0eTogJ0lORk8nLFxyXG4gICAgICAgICAgbWVzc2FnZTogJ3Rlc3QnLFxyXG4gICAgICAgICAgc2VydmljZU5hbWU6ICd0ZXN0JyxcclxuICAgICAgICAgIGVudmlyb25tZW50OiBlbnYsXHJcbiAgICAgICAgICB0aW1lc3RhbXA6IG5ldyBEYXRlKCkudG9JU09TdHJpbmcoKSxcclxuICAgICAgICB9O1xyXG4gICAgICAgIGV4cGVjdChsb2cuZW52aXJvbm1lbnQpLnRvQmUoZW52KTtcclxuICAgICAgfSk7XHJcbiAgICB9KTtcclxuICB9KTtcclxuXHJcbiAgZGVzY3JpYmUoJ3RpbWVzdGFtcCcsICgpID0+IHtcclxuICAgIGl0KCdkZXZlIGFjZWl0YXIgZm9ybWF0byBJU08gODYwMScsICgpID0+IHtcclxuICAgICAgY29uc3QgaXNvVGltZXN0YW1wID0gbmV3IERhdGUoKS50b0lTT1N0cmluZygpO1xyXG4gICAgICBjb25zdCBsb2c6IFN0YW5kYXJkTG9nRGF0YSA9IHtcclxuICAgICAgICBzZXZlcml0eTogJ0lORk8nLFxyXG4gICAgICAgIG1lc3NhZ2U6ICd0ZXN0JyxcclxuICAgICAgICBzZXJ2aWNlTmFtZTogJ3Rlc3QnLFxyXG4gICAgICAgIGVudmlyb25tZW50OiAndGVzdCcsXHJcbiAgICAgICAgdGltZXN0YW1wOiBpc29UaW1lc3RhbXAsXHJcbiAgICAgIH07XHJcblxyXG4gICAgICBleHBlY3QobG9nLnRpbWVzdGFtcCkudG9NYXRjaChcclxuICAgICAgICAvXlxcZHs0fS1cXGR7Mn0tXFxkezJ9VFxcZHsyfTpcXGR7Mn06XFxkezJ9XFwuXFxkezN9WiQvXHJcbiAgICAgICk7XHJcbiAgICB9KTtcclxuICB9KTtcclxuXHJcbiAgZGVzY3JpYmUoJ2h0dHAuc3RhdHVzQ29kZScsICgpID0+IHtcclxuICAgIGl0KCdkZXZlIGFjZWl0YXIgY8OzZGlnb3MgSFRUUCB2w6FsaWRvcycsICgpID0+IHtcclxuICAgICAgY29uc3Qgc3RhdHVzQ29kZXMgPSBbMjAwLCAyMDEsIDQwMCwgNDAxLCA0MDQsIDUwMCwgNTAyLCA1MDNdO1xyXG5cclxuICAgICAgc3RhdHVzQ29kZXMuZm9yRWFjaCgoc3RhdHVzQ29kZSkgPT4ge1xyXG4gICAgICAgIGNvbnN0IGxvZzogU3RhbmRhcmRMb2dEYXRhID0ge1xyXG4gICAgICAgICAgc2V2ZXJpdHk6ICdJTkZPJyxcclxuICAgICAgICAgIG1lc3NhZ2U6ICd0ZXN0JyxcclxuICAgICAgICAgIHNlcnZpY2VOYW1lOiAndGVzdCcsXHJcbiAgICAgICAgICBlbnZpcm9ubWVudDogJ3Rlc3QnLFxyXG4gICAgICAgICAgaHR0cDogeyBzdGF0dXNDb2RlIH0sXHJcbiAgICAgICAgICB0aW1lc3RhbXA6IG5ldyBEYXRlKCkudG9JU09TdHJpbmcoKSxcclxuICAgICAgICB9O1xyXG4gICAgICAgIGV4cGVjdChsb2cuaHR0cD8uc3RhdHVzQ29kZSkudG9CZShzdGF0dXNDb2RlKTtcclxuICAgICAgfSk7XHJcbiAgICB9KTtcclxuICB9KTtcclxuXHJcbiAgZGVzY3JpYmUoJ3BlcmZvcm1hbmNlLmR1cmF0aW9uTXMnLCAoKSA9PiB7XHJcbiAgICBpdCgnZGV2ZSBhY2VpdGFyIHZhbG9yZXMgbnVtw6lyaWNvcyBwb3NpdGl2b3MnLCAoKSA9PiB7XHJcbiAgICAgIGNvbnN0IGR1cmF0aW9ucyA9IFswLCAxMDAsIDUwMCwgMTAwMCwgNTAwMCwgMTAwMDBdO1xyXG5cclxuICAgICAgZHVyYXRpb25zLmZvckVhY2goKGR1cmF0aW9uTXMpID0+IHtcclxuICAgICAgICBjb25zdCBsb2c6IFN0YW5kYXJkTG9nRGF0YSA9IHtcclxuICAgICAgICAgIHNldmVyaXR5OiAnSU5GTycsXHJcbiAgICAgICAgICBtZXNzYWdlOiAndGVzdCcsXHJcbiAgICAgICAgICBzZXJ2aWNlTmFtZTogJ3Rlc3QnLFxyXG4gICAgICAgICAgZW52aXJvbm1lbnQ6ICd0ZXN0JyxcclxuICAgICAgICAgIHBlcmZvcm1hbmNlOiB7IGR1cmF0aW9uTXMgfSxcclxuICAgICAgICAgIHRpbWVzdGFtcDogbmV3IERhdGUoKS50b0lTT1N0cmluZygpLFxyXG4gICAgICAgIH07XHJcbiAgICAgICAgZXhwZWN0KGxvZy5wZXJmb3JtYW5jZT8uZHVyYXRpb25NcykudG9CZShkdXJhdGlvbk1zKTtcclxuICAgICAgfSk7XHJcbiAgICB9KTtcclxuICB9KTtcclxuXHJcbiAgZGVzY3JpYmUoJ3BlcmZvcm1hbmNlLnN1Y2Nlc3MnLCAoKSA9PiB7XHJcbiAgICBpdCgnZGV2ZSBhY2VpdGFyIHZhbG9yZXMgYm9vbGVhbm9zJywgKCkgPT4ge1xyXG4gICAgICBbdHJ1ZSwgZmFsc2VdLmZvckVhY2goKHN1Y2Nlc3MpID0+IHtcclxuICAgICAgICBjb25zdCBsb2c6IFN0YW5kYXJkTG9nRGF0YSA9IHtcclxuICAgICAgICAgIHNldmVyaXR5OiAnSU5GTycsXHJcbiAgICAgICAgICBtZXNzYWdlOiAndGVzdCcsXHJcbiAgICAgICAgICBzZXJ2aWNlTmFtZTogJ3Rlc3QnLFxyXG4gICAgICAgICAgZW52aXJvbm1lbnQ6ICd0ZXN0JyxcclxuICAgICAgICAgIHBlcmZvcm1hbmNlOiB7IHN1Y2Nlc3MgfSxcclxuICAgICAgICAgIHRpbWVzdGFtcDogbmV3IERhdGUoKS50b0lTT1N0cmluZygpLFxyXG4gICAgICAgIH07XHJcbiAgICAgICAgZXhwZWN0KGxvZy5wZXJmb3JtYW5jZT8uc3VjY2VzcykudG9CZShzdWNjZXNzKTtcclxuICAgICAgfSk7XHJcbiAgICB9KTtcclxuICB9KTtcclxufSk7XHJcbiJdfQ==
@@ -0,0 +1 @@
1
+ export {};