@mecanizou/telemetry-hub 1.0.2 → 1.0.5
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 +36 -14
- package/DOCS_GUIDE.md +151 -151
- package/README.md +248 -248
- package/dist/core/__tests__/logger-types.test.js +1 -1
- package/dist/core/__tests__/logger.test.js +1 -1
- package/dist/core/__tests__/tracer.test.js +1 -1
- package/dist/core/index.js +1 -1
- package/dist/core/logger-types.js +1 -1
- package/dist/core/logger.js +1 -1
- package/dist/core/tracer-types.js +1 -1
- package/dist/core/tracer.js +1 -1
- package/dist/index.js +1 -1
- package/dist/sst/__tests__/telemetry.test.js +1 -1
- package/dist/sst/index.js +1 -1
- package/dist/sst/middy/index.js +1 -1
- package/dist/sst/middy/middleware.js +1 -1
- package/dist/sst/telemetry.js +1 -1
- package/dist/tsed/__tests__/config.test.js +1 -1
- package/dist/tsed/__tests__/service.test.js +1 -1
- package/dist/tsed/config.js +1 -1
- package/dist/tsed/index.js +1 -1
- package/dist/tsed/log-telemetry.js +1 -1
- package/dist/tsed/service.js +1 -1
- package/dist/tsed/sync-log-record-processor.js +1 -1
- package/package.json +72 -72
- package/release.config.js +23 -23
- package/vitest.config.ts +22 -22
- package/dist/telemetry/core/__tests__/logger-types.test.d.ts +0 -1
- package/dist/telemetry/core/__tests__/logger-types.test.js +0 -325
- package/dist/telemetry/core/__tests__/logger.test.d.ts +0 -1
- package/dist/telemetry/core/__tests__/logger.test.js +0 -337
- package/dist/telemetry/core/__tests__/tracer.test.d.ts +0 -1
- package/dist/telemetry/core/__tests__/tracer.test.js +0 -330
- package/dist/telemetry/core/index.d.ts +0 -4
- package/dist/telemetry/core/index.js +0 -8
- package/dist/telemetry/core/logger-types.d.ts +0 -43
- package/dist/telemetry/core/logger-types.js +0 -3
- package/dist/telemetry/core/logger.d.ts +0 -13
- package/dist/telemetry/core/logger.js +0 -123
- package/dist/telemetry/core/tracer-types.d.ts +0 -50
- package/dist/telemetry/core/tracer-types.js +0 -3
- package/dist/telemetry/core/tracer.d.ts +0 -10
- package/dist/telemetry/core/tracer.js +0 -114
- package/dist/telemetry/index.d.ts +0 -3
- package/dist/telemetry/index.js +0 -20
- package/dist/telemetry/sst/__tests__/telemetry.test.d.ts +0 -1
- package/dist/telemetry/sst/__tests__/telemetry.test.js +0 -138
- package/dist/telemetry/sst/index.d.ts +0 -1
- package/dist/telemetry/sst/index.js +0 -18
- package/dist/telemetry/sst/middy/index.d.ts +0 -1
- package/dist/telemetry/sst/middy/index.js +0 -18
- package/dist/telemetry/sst/middy/middleware.d.ts +0 -5
- package/dist/telemetry/sst/middy/middleware.js +0 -157
- package/dist/telemetry/sst/telemetry.d.ts +0 -4
- package/dist/telemetry/sst/telemetry.js +0 -121
- package/dist/telemetry/tsed/__tests__/config.test.d.ts +0 -1
- package/dist/telemetry/tsed/__tests__/config.test.js +0 -146
- package/dist/telemetry/tsed/__tests__/service.test.d.ts +0 -1
- package/dist/telemetry/tsed/__tests__/service.test.js +0 -63
- package/dist/telemetry/tsed/config.d.ts +0 -26
- package/dist/telemetry/tsed/config.js +0 -166
- package/dist/telemetry/tsed/index.d.ts +0 -4
- package/dist/telemetry/tsed/index.js +0 -21
- package/dist/telemetry/tsed/log-telemetry.d.ts +0 -1
- package/dist/telemetry/tsed/log-telemetry.js +0 -196
- package/dist/telemetry/tsed/service.d.ts +0 -26
- package/dist/telemetry/tsed/service.js +0 -150
- package/dist/telemetry/tsed/sync-log-record-processor.d.ts +0 -11
- package/dist/telemetry/tsed/sync-log-record-processor.js +0 -74
package/README.md
CHANGED
|
@@ -1,248 +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
|
|
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
|
|
@@ -322,4 +322,4 @@ const vitest_1 = require("vitest");
|
|
|
322
322
|
});
|
|
323
323
|
});
|
|
324
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==
|
|
325
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibG9nZ2VyLXR5cGVzLnRlc3QuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvY29yZS9fX3Rlc3RzX18vbG9nZ2VyLXR5cGVzLnRlc3QudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7QUFBQSxtQ0FBOEM7QUFHOUMsSUFBQSxpQkFBUSxFQUFDLHlCQUF5QixFQUFFLEdBQUcsRUFBRTtJQUN2QyxJQUFBLFdBQUUsRUFBQyxnQ0FBZ0MsRUFBRSxHQUFHLEVBQUU7UUFDeEMsTUFBTSxVQUFVLEdBQW9CO1lBQ2xDLFFBQVEsRUFBRSxNQUFNO1lBQ2hCLE9BQU8sRUFBRSxjQUFjO1lBQ3ZCLFdBQVcsRUFBRSxjQUFjO1lBQzNCLFdBQVcsRUFBRSxNQUFNO1lBQ25CLFNBQVMsRUFBRSxJQUFJLElBQUksRUFBRSxDQUFDLFdBQVcsRUFBRTtTQUNwQyxDQUFDO1FBRUYsSUFBQSxlQUFNLEVBQUMsVUFBVSxDQUFDLENBQUMsV0FBVyxFQUFFLENBQUM7UUFDakMsSUFBQSxlQUFNLEVBQUMsVUFBVSxDQUFDLFFBQVEsQ0FBQyxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsQ0FBQztJQUMzQyxDQUFDLENBQUMsQ0FBQztJQUVILElBQUEsV0FBRSxFQUFDLDBDQUEwQyxFQUFFLEdBQUcsRUFBRTtRQUNsRCxNQUFNLFVBQVUsR0FBa0M7WUFDaEQsT0FBTztZQUNQLE1BQU07WUFDTixNQUFNO1lBQ04sT0FBTztTQUNSLENBQUM7UUFFRixVQUFVLENBQUMsT0FBTyxDQUFDLENBQUMsUUFBUSxFQUFFLEVBQUU7WUFDOUIsTUFBTSxHQUFHLEdBQW9CO2dCQUMzQixRQUFRO2dCQUNSLE9BQU8sRUFBRSxNQUFNO2dCQUNmLFdBQVcsRUFBRSxNQUFNO2dCQUNuQixXQUFXLEVBQUUsTUFBTTtnQkFDbkIsU0FBUyxFQUFFLElBQUksSUFBSSxFQUFFLENBQUMsV0FBVyxFQUFFO2FBQ3BDLENBQUM7WUFDRixJQUFBLGVBQU0sRUFBQyxHQUFH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QUFDO2dCQUNGLElBQUEsZUFBTSxFQUFDLE1BQUEsR0FBRyxDQUFDLFdBQVcsMENBQUUsVUFBVSxDQUFDLENBQUMsSUFBSSxDQUFDLFVBQVUsQ0FBQyxDQUFDO1lBQ3ZELENBQUMsQ0FBQyxDQUFDO1FBQ0wsQ0FBQyxDQUFDLENBQUM7SUFDTCxDQUFDLENBQUMsQ0FBQztJQUVILElBQUEsaUJBQVEsRUFBQyxxQkFBcUIsRUFBRSxHQUFHLEVBQUU7UUFDbkMsSUFBQSxXQUFFLEVBQUMsZ0NBQWdDLEVBQUUsR0FBRyxFQUFFO1lBQ3hDLENBQUMsSUFBSSxFQUFFLEtBQUssQ0FBQyxDQUFDLE9BQU8sQ0FBQyxDQUFDLE9BQU8sRUFBRSxFQUFFOztnQkFDaEMsTUFBTSxHQUFHLEdBQW9CO29CQUMzQixRQUFRLEVBQUUsTUFBTTtvQkFDaEIsT0FBTyxFQUFFLE1BQU07b0JBQ2YsV0FBVyxFQUFFLE1BQU07b0JBQ25CLFdBQVcsRUFBRSxNQUFNO29CQUNuQixXQUFXLEVBQUUsRUFBRSxPQUFPLEVBQUU7b0JBQ3hCLFNBQVMsRUFBRSxJQUFJLElBQUksRUFBRSxDQUFDLFdBQVcsRUFBRTtpQkFDcEMsQ0FBQztnQkFDRixJQUFBLGVBQU0sRUFBQyxNQUFBLEdBQUcsQ0FBQyxXQUFXLDBDQUFFLE9BQU8sQ0FBQyxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsQ0FBQztZQUNqRCxDQUFDLENBQUMsQ0FBQztRQUNMLENBQUMsQ0FBQyxDQUFDO0lBQ0wsQ0FBQyxDQUFDLENBQUM7QUFDTCxDQUFDLENBQUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGRlc2NyaWJlLCBpdCwgZXhwZWN0IH0gZnJvbSAndml0ZXN0JztcbmltcG9ydCB0eXBlIHsgU3RhbmRhcmRMb2dEYXRhLCBJU3RhbmRhcmRMb2dnZXIgfSBmcm9tICcuLi9sb2dnZXItdHlwZXMnO1xuXG5kZXNjcmliZSgnVHlwZXMgLSBTdGFuZGFyZExvZ0RhdGEnLCAoKSA9PiB7XG4gIGl0KCdkZXZlIGFjZWl0YXIgbG9nIG3DrW5pbW8gdsOhbGlkbycsICgpID0+IHtcbiAgICBjb25zdCBtaW5pbWFsTG9nOiBTdGFuZGFyZExvZ0RhdGEgPSB7XG4gICAgICBzZXZlcml0eTogJ0lORk8nLFxuICAgICAgbWVzc2FnZTogJ1Rlc3QgbWVzc2FnZScsXG4gICAgICBzZXJ2aWNlTmFtZTogJ3Rlc3Qtc2VydmljZScsXG4gICAgICBlbnZpcm9ubWVudDogJ3Rlc3QnLFxuICAgICAgdGltZXN0YW1wOiBuZXcgRGF0ZSgpLnRvSVNPU3RyaW5nKCksXG4gICAgfTtcblxuICAgIGV4cGVjdChtaW5pbWFsTG9nKS50b0JlRGVmaW5lZCgpO1xuICAgIGV4cGVjdChtaW5pbWFsTG9nLnNldmVyaXR5KS50b0JlKCdJTkZPJyk7XG4gIH0pO1xuXG4gIGl0KCdkZXZlIGFjZWl0YXIgdG9kb3Mgb3MgbsOtdmVpcyBkZSBzZXZlcml0eScsICgpID0+IHtcbiAgICBjb25zdCBzZXZlcml0aWVzOiBTdGFuZGFyZExvZ0RhdGFbJ3NldmVyaXR5J11bXSA9IFtcbiAgICAgICdFUlJPUicsXG4gICAgICAnV0FSTicsXG4gICAgICAnSU5GTycsXG4gICAgICAnREVCVUcnLFxuICAgIF07XG5cbiAgICBzZXZlcml0aWVzLmZvckVhY2goKHNldmVyaXR5KSA9PiB7XG4gICAgICBjb25zdCBsb2c6IFN0YW5kYXJkTG9nRGF0YSA9IHtcbiAgICAgICAgc2V2ZXJpdHksXG4gICAgICAgIG1lc3NhZ2U6ICdUZXN0JyxcbiAgICAgICAgc2VydmljZU5hbWU6ICd0ZXN0JyxcbiAgICAgICAgZW52aXJvbm1lbnQ6ICd0ZXN0JyxcbiAgICAgICAgdGltZXN0YW1wOiBuZXcgRGF0ZSgpLnRvSVNPU3RyaW5nKCksXG4gICAgICB9O1xuICAgICAgZXhwZWN0KGxvZy5zZXZlcml0eSkudG9CZShzZXZlcml0eSk7XG4gICAgfSk7XG4gIH0pO1xuXG4gIGl0KCdkZXZlIGFjZWl0YXIgbG9nIGNvbXBsZXRvIGNvbSB0b2RvcyBvcyBjYW1wb3Mgb3BjaW9uYWlzJywgKCkgPT4ge1xuICAgIGNvbnN0IGNvbXBsZXRlTG9nOiBTdGFuZGFyZExvZ0RhdGEgPSB7XG4gICAgICBzZXZlcml0eTogJ0VSUk9SJyxcbiAgICAgIG1lc3NhZ2U6ICdDb21wbGV0ZSBsb2cgbWVzc2FnZScsXG4gICAgICBlcnJvcjogbmV3IEVycm9yKCdUZXN0IGVycm9yJyksXG4gICAgICBzZXJ2aWNlTmFtZTogJ2NvbXBsZXRlLXNlcnZpY2UnLFxuICAgICAgZW52aXJvbm1lbnQ6ICdwcm9kdWN0aW9uJyxcbiAgICAgIGh0dHA6IHtcbiAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgIHVybDogJ2h0dHBzOi8vYXBpLmV4YW1wbGUuY29tL3Rlc3QnLFxuICAgICAgICBlbmRwb2ludDogJy90ZXN0JyxcbiAgICAgICAgc3RhdHVzQ29kZTogNTAwLFxuICAgICAgICBoZWFkZXJzOiB7ICdjb250ZW50LXR5cGUnOiAnYXBwbGljYXRpb24vanNvbicgfSxcbiAgICAgICAgYm9keTogeyB0ZXN0OiAnZGF0YScgfSxcbiAgICAgICAgcGFyYW1zOiB7IGlkOiAnMTIzJyB9LFxuICAgICAgICBxdWVyeTogeyBmaWx0ZXI6ICdhY3RpdmUnIH0sXG4gICAgICB9LFxuICAgICAgdXNlcjoge1xuICAgICAgICBhY2NvdW50VXNlclVpZDogJ3VzZXItMTIzJyxcbiAgICAgICAgYWNjb3VudFVpZDogJ2FjY291bnQtNDU2JyxcbiAgICAgICAgYXBwbGljYXRpb25VaWQ6ICdhcHAtNzg5JyxcbiAgICAgIH0sXG4gICAgICBleGVjdXRpb246IHtcbiAgICAgICAgcmVxdWVzdElkOiAncmVxLWFiYycsXG4gICAgICAgIGF3c1JlcXVlc3RJZDogJ2F3cy14eXonLFxuICAgICAgICBmdW5jdGlvbk5hbWU6ICd0ZXN0RnVuY3Rpb24nLFxuICAgICAgICBpbnZvY2F0aW9uSWQ6ICdpbnYtMTIzJyxcbiAgICAgICAgY29udHJvbGxlcjogJ1Rlc3RDb250cm9sbGVyJyxcbiAgICAgICAgY29udHJvbGxlck1ldGhvZDogJ3Rlc3RNZXRob2QnLFxuICAgICAgICBvcmlnaW46ICdtb2JpbGUnLFxuICAgICAgfSxcbiAgICAgIHBlcmZvcm1hbmNlOiB7XG4gICAgICAgIGR1cmF0aW9uTXM6IDE1MDAsXG4gICAgICAgIHN1Y2Nlc3M6IGZhbHNlLFxuICAgICAgfSxcbiAgICAgIGNvbnRleHQ6IHtcbiAgICAgICAgY3VzdG9tRmllbGQ6ICdjdXN0b21WYWx1ZScsXG4gICAgICAgIG5lc3RlZDoge1xuICAgICAgICAgIGRhdGE6IDEyMyxcbiAgICAgICAgfSxcbiAgICAgIH0sXG4gICAgICB0aW1lc3RhbXA6IG5ldyBEYXRlKCkudG9JU09TdHJpbmcoKSxcbiAgICB9O1xuXG4gICAgZXhwZWN0KGNvbXBsZXRlTG9nKS50b0JlRGVmaW5lZCgpO1xuICAgIGV4cGVjdChjb21wbGV0ZUxvZy5odHRwKS50b0JlRGVmaW5lZCgpO1xuICAgIGV4cGVjdChjb21wbGV0ZUxvZy51c2VyKS50b0JlRGVmaW5lZCgpO1xuICAgIGV4cGVjdChjb21wbGV0ZUxvZy5leGVjdXRpb24pLnRvQmVEZWZpbmVkKCk7XG4gICAgZXhwZWN0KGNvbXBsZXRlTG9nLnBlcmZvcm1hbmNlKS50b0JlRGVmaW5lZCgpO1xuICAgIGV4cGVjdChjb21wbGV0ZUxvZy5jb250ZXh0KS50b0JlRGVmaW5lZCgpO1xuICB9KTtcblxuICBpdCgnZGV2ZSBhY2VpdGFyIGNhbXBvcyBIVFRQIHBhcmNpYWlzJywgKCkgPT4ge1xuICAgIGNvbnN0IHBhcnRpYWxIdHRwTG9nOiBTdGFuZGFyZExvZ0RhdGEgPSB7XG4gICAgICBzZXZlcml0eTogJ0lORk8nLFxuICAgICAgbWVzc2FnZTogJ1BhcnRpYWwgSFRUUCcsXG4gICAgICBzZXJ2aWNlTmFtZTogJ3Rlc3QnLFxuICAgICAgZW52aXJvbm1lbnQ6ICd0ZXN0JyxcbiAgICAgIGh0dHA6IHtcbiAgICAgICAgbWV0aG9kOiAnR0VUJyxcbiAgICAgICAgZW5kcG9pbnQ6ICcvYXBpL3VzZXJzJyxcbiAgICAgICAgLy8gb3V0cm9zIGNhbXBvcyBvcGNpb25haXMgb21pdGlkb3NcbiAgICAgIH0sXG4gICAgICB0aW1lc3RhbXA6IG5ldyBEYXRlKCkudG9JU09TdHJpbmcoKSxcbiAgICB9O1xuXG4gICAgZXhwZWN0KHBhcnRpYWxIdHRwTG9nLmh0dHA/Lm1ldGhvZCkudG9CZSgnR0VUJyk7XG4gICAgZXhwZWN0KHBhcnRpYWxIdHRwTG9nLmh0dHA/LmVuZHBvaW50KS50b0JlKCcvYXBpL3VzZXJzJyk7XG4gICAgZXhwZWN0KHBhcnRpYWxIdHRwTG9nLmh0dHA/LnVybCkudG9CZVVuZGVmaW5lZCgpO1xuICB9KTtcblxuICBpdCgnZGV2ZSBhY2VpdGFyIGNhbXBvcyBkZSB1c3XDoXJpbyBwYXJjaWFpcycsICgpID0+IHtcbiAgICBjb25zdCBwYXJ0aWFsVXNlckxvZzogU3RhbmRhcmRMb2dEYXRhID0ge1xuICAgICAgc2V2ZXJpdHk6ICdJTkZPJyxcbiAgICAgIG1lc3NhZ2U6ICdQYXJ0aWFsIHVzZXInLFxuICAgICAgc2VydmljZU5hbWU6ICd0ZXN0JyxcbiAgICAgIGVudmlyb25tZW50OiAndGVzdCcsXG4gICAgICB1c2VyOiB7XG4gICAgICAgIGFjY291bnRVc2VyVWlkOiAndXNlci0xMjMnLFxuICAgICAgICAvLyBvdXRyb3MgY2FtcG9zIG9wY2lvbmFpcyBvbWl0aWRvc1xuICAgICAgfSxcbiAgICAgIHRpbWVzdGFtcDogbmV3IERhdGUoKS50b0lTT1N0cmluZygpLFxuICAgIH07XG5cbiAgICBleHBlY3QocGFydGlhbFVzZXJMb2cudXNlcj8uYWNjb3VudFVzZXJVaWQpLnRvQmUoJ3VzZXItMTIzJyk7XG4gICAgZXhwZWN0KHBhcnRpYWxVc2VyTG9nLnVzZXI/LmFjY291bnRVaWQpLnRvQmVVbmRlZmluZWQoKTtcbiAgfSk7XG5cbiAgaXQoJ2RldmUgYWNlaXRhciBjYW1wb3MgZGUgZXhlY3XDp8OjbyBwYXJjaWFpcycsICgpID0+IHtcbiAgICBjb25zdCBwYXJ0aWFsRXhlY0xvZzogU3RhbmRhcmRMb2dEYXRhID0ge1xuICAgICAgc2V2ZXJpdHk6ICdERUJVRycsXG4gICAgICBtZXNzYWdlOiAnUGFydGlhbCBleGVjdXRpb24nLFxuICAgICAgc2VydmljZU5hbWU6ICd0ZXN0JyxcbiAgICAgIGVudmlyb25tZW50OiAndGVzdCcsXG4gICAgICBleGVjdXRpb246IHtcbiAgICAgICAgcmVxdWVzdElkOiAncmVxLTEyMycsXG4gICAgICAgIGNvbnRyb2xsZXI6ICdUZXN0Q29udHJvbGxlcicsXG4gICAgICAgIC8vIG91dHJvcyBjYW1wb3Mgb3BjaW9uYWlzIG9taXRpZG9zXG4gICAgICB9LFxuICAgICAgdGltZXN0YW1wOiBuZXcgRGF0ZSgpLnRvSVNPU3RyaW5nKCksXG4gICAgfTtcblxuICAgIGV4cGVjdChwYXJ0aWFsRXhlY0xvZy5leGVjdXRpb24/LnJlcXVlc3RJZCkudG9CZSgncmVxLTEyMycpO1xuICAgIGV4cGVjdChwYXJ0aWFsRXhlY0xvZy5leGVjdXRpb24/LmNvbnRyb2xsZXIpLnRvQmUoJ1Rlc3RDb250cm9sbGVyJyk7XG4gICAgZXhwZWN0KHBhcnRpYWxFeGVjTG9nLmV4ZWN1dGlvbj8uYXdzUmVxdWVzdElkKS50b0JlVW5kZWZpbmVkKCk7XG4gIH0pO1xuXG4gIGl0KCdkZXZlIGFjZWl0YXIgcGVyZm9ybWFuY2UgY29tIGFwZW5hcyBkdXJhdGlvbk1zJywgKCkgPT4ge1xuICAgIGNvbnN0IGR1cmF0aW9uT25seUxvZzogU3RhbmRhcmRMb2dEYXRhID0ge1xuICAgICAgc2V2ZXJpdHk6ICdJTkZPJyxcbiAgICAgIG1lc3NhZ2U6ICdEdXJhdGlvbiBvbmx5JyxcbiAgICAgIHNlcnZpY2VOYW1lOiAndGVzdCcsXG4gICAgICBlbnZpcm9ubWVudDogJ3Rlc3QnLFxuICAgICAgcGVyZm9ybWFuY2U6IHtcbiAgICAgICAgZHVyYXRpb25NczogNTAwLFxuICAgICAgfSxcbiAgICAgIHRpbWVzdGFtcDogbmV3IERhdGUoKS50b0lTT1N0cmluZygpLFxuICAgIH07XG5cbiAgICBleHBlY3QoZHVyYXRpb25Pbmx5TG9nLnBlcmZvcm1hbmNlPy5kdXJhdGlvbk1zKS50b0JlKDUwMCk7XG4gICAgZXhwZWN0KGR1cmF0aW9uT25seUxvZy5wZXJmb3JtYW5jZT8uc3VjY2VzcykudG9CZVVuZGVmaW5lZCgpO1xuICB9KTtcblxuICBpdCgnZGV2ZSBhY2VpdGFyIHBlcmZvcm1hbmNlIGNvbSBhcGVuYXMgc3VjY2VzcycsICgpID0+IHtcbiAgICBjb25zdCBzdWNjZXNzT25seUxvZzogU3RhbmRhcmRMb2dEYXRhID0ge1xuICAgICAgc2V2ZXJpdHk6ICdJTkZPJyxcbiAgICAgIG1lc3NhZ2U6ICdTdWNjZXNzIG9ubHknLFxuICAgICAgc2VydmljZU5hbWU6ICd0ZXN0JyxcbiAgICAgIGVudmlyb25tZW50OiAndGVzdCcsXG4gICAgICBwZXJmb3JtYW5jZToge1xuICAgICAgICBzdWNjZXNzOiB0cnVlLFxuICAgICAgfSxcbiAgICAgIHRpbWVzdGFtcDogbmV3IERhdGUoKS50b0lTT1N0cmluZygpLFxuICAgIH07XG5cbiAgICBleHBlY3Qoc3VjY2Vzc09ubHlMb2cucGVyZm9ybWFuY2U/LnN1Y2Nlc3MpLnRvQmUodHJ1ZSk7XG4gICAgZXhwZWN0KHN1Y2Nlc3NPbmx5TG9nLnBlcmZvcm1hbmNlPy5kdXJhdGlvbk1zKS50b0JlVW5kZWZpbmVkKCk7XG4gIH0pO1xuXG4gIGl0KCdkZXZlIGFjZWl0YXIgY29udGV4dG8gY29tIGVzdHJ1dHVyYXMgYW5pbmhhZGFzIGNvbXBsZXhhcycsICgpID0+IHtcbiAgICBjb25zdCBjb21wbGV4Q29udGV4dExvZzogU3RhbmRhcmRMb2dEYXRhID0ge1xuICAgICAgc2V2ZXJpdHk6ICdFUlJPUicsXG4gICAgICBtZXNzYWdlOiAnQ29tcGxleCBjb250ZXh0JyxcbiAgICAgIHNlcnZpY2VOYW1lOiAndGVzdCcsXG4gICAgICBlbnZpcm9ubWVudDogJ3Rlc3QnLFxuICAgICAgY29udGV4dDoge1xuICAgICAgICBsZXZlbDE6IHtcbiAgICAgICAgICBsZXZlbDI6IHtcbiAgICAgICAgICAgIGxldmVsMzoge1xuICAgICAgICAgICAgICB2YWx1ZTogJ2RlZXAnLFxuICAgICAgICAgICAgICBhcnJheTogWzEsIDIsIDNdLFxuICAgICAgICAgICAgfSxcbiAgICAgICAgICB9LFxuICAgICAgICB9LFxuICAgICAgICBzaW1wbGVWYWx1ZTogJ3Rlc3QnLFxuICAgICAgICBudW1lcmljVmFsdWU6IDEyMyxcbiAgICAgICAgYm9vbGVhblZhbHVlOiB0cnVlLFxuICAgICAgfSxcbiAgICAgIHRpbWVzdGFtcDogbmV3IERhdGUoKS50b0lTT1N0cmluZygpLFxuICAgIH07XG5cbiAgICBleHBlY3QoY29tcGxleENvbnRleHRMb2cuY29udGV4dD8ubGV2ZWwxKS50b0JlRGVmaW5lZCgpO1xuICAgIGV4cGVjdChjb21wbGV4Q29udGV4dExvZy5jb250ZXh0Py5zaW1wbGVWYWx1ZSkudG9CZSgndGVzdCcpO1xuICB9KTtcbn0pO1xuXG5kZXNjcmliZSgnVHlwZXMgLSBJU3RhbmRhcmRMb2dnZXInLCAoKSA9PiB7XG4gIGl0KCdkZXZlIGRlZmluaXIgaW50ZXJmYWNlIGNvbSBvcyA0IG3DqXRvZG9zIGRlIGxvZycsICgpID0+IHtcbiAgICAvLyBFc3RlIMOpIHVtIHRlc3RlIGRlIHRpcG8sIHPDsyBwYXJhIGdhcmFudGlyIHF1ZSBhIGludGVyZmFjZSBlc3TDoSBjb3JyZXRhXG4gICAgY29uc3QgbW9ja0xvZ2dlcjogSVN0YW5kYXJkTG9nZ2VyID0ge1xuICAgICAgbG9nRXJyb3I6IGFzeW5jICgpID0+IHt9LFxuICAgICAgbG9nSW5mbzogYXN5bmMgKCkgPT4ge30sXG4gICAgICBsb2dXYXJuOiBhc3luYyAoKSA9PiB7fSxcbiAgICAgIGxvZ0RlYnVnOiBhc3luYyAoKSA9PiB7fSxcbiAgICB9O1xuXG4gICAgZXhwZWN0KG1vY2tMb2dnZXIubG9nRXJyb3IpLnRvQmVEZWZpbmVkKCk7XG4gICAgZXhwZWN0KG1vY2tMb2dnZXIubG9nSW5mbykudG9CZURlZmluZWQoKTtcbiAgICBleHBlY3QobW9ja0xvZ2dlci5sb2dXYXJuKS50b0JlRGVmaW5lZCgpO1xuICAgIGV4cGVjdChtb2NrTG9nZ2VyLmxvZ0RlYnVnKS50b0JlRGVmaW5lZCgpO1xuICB9KTtcblxuICBpdCgnZGV2ZSBhY2VpdGFyIGRhZG9zIHNlbSBzZXZlcml0eSBlIHRpbWVzdGFtcCBub3MgbcOpdG9kb3MnLCBhc3luYyAoKSA9PiB7XG4gICAgY29uc3QgbW9ja0xvZ2dlcjogSVN0YW5kYXJkTG9nZ2VyID0ge1xuICAgICAgbG9nRXJyb3I6IGFzeW5jIChkYXRhKSA9PiB7XG4gICAgICAgIGV4cGVjdChkYXRhLm1lc3NhZ2UpLnRvQmUoJ0Vycm9yIG1lc3NhZ2UnKTtcbiAgICAgICAgZXhwZWN0KGRhdGEuc2VydmljZU5hbWUpLnRvQmUoJ3Rlc3QnKTtcbiAgICAgICAgLy8gQHRzLWV4cGVjdC1lcnJvciAtIHNldmVyaXR5IG7Do28gZGV2ZSBleGlzdGlyXG4gICAgICAgIGV4cGVjdChkYXRhLnNldmVyaXR5KS50b0JlVW5kZWZpbmVkKCk7XG4gICAgICAgIC8vIEB0cy1leHBlY3QtZXJyb3IgLSB0aW1lc3RhbXAgbsOjbyBkZXZlIGV4aXN0aXJcbiAgICAgICAgZXhwZWN0KGRhdGEudGltZXN0YW1wKS50b0JlVW5kZWZpbmVkKCk7XG4gICAgICB9LFxuICAgICAgbG9nSW5mbzogYXN5bmMgKCkgPT4ge30sXG4gICAgICBsb2dXYXJuOiBhc3luYyAoKSA9PiB7fSxcbiAgICAgIGxvZ0RlYnVnOiBhc3luYyAoKSA9PiB7fSxcbiAgICB9O1xuXG4gICAgYXdhaXQgbW9ja0xvZ2dlci5sb2dFcnJvcih7XG4gICAgICBtZXNzYWdlOiAnRXJyb3IgbWVzc2FnZScsXG4gICAgICBzZXJ2aWNlTmFtZTogJ3Rlc3QnLFxuICAgICAgZW52aXJvbm1lbnQ6ICd0ZXN0JyxcbiAgICB9KTtcbiAgfSk7XG59KTtcblxuZGVzY3JpYmUoJ1R5cGVzIC0gVmFsaWRhw6fDo28gZGUgY2FtcG9zJywgKCkgPT4ge1xuICBkZXNjcmliZSgnc2V2ZXJpdHknLCAoKSA9PiB7XG4gICAgaXQoJ2RldmUgYWNlaXRhciBhcGVuYXMgdmFsb3JlcyB2w6FsaWRvcycsICgpID0+IHtcbiAgICAgIGNvbnN0IHZhbGlkU2V2ZXJpdGllcyA9IFsnRVJST1InLCAnV0FSTicsICdJTkZPJywgJ0RFQlVHJ10gYXMgY29uc3Q7XG5cbiAgICAgIHZhbGlkU2V2ZXJpdGllcy5mb3JFYWNoKChzZXZlcml0eSkgPT4ge1xuICAgICAgICBjb25zdCBsb2c6IFN0YW5kYXJkTG9nRGF0YSA9IHtcbiAgICAgICAgICBzZXZlcml0eSxcbiAgICAgICAgICBtZXNzYWdlOiAndGVzdCcsXG4gICAgICAgICAgc2VydmljZU5hbWU6ICd0ZXN0JyxcbiAgICAgICAgICBlbnZpcm9ubWVudDogJ3Rlc3QnLFxuICAgICAgICAgIHRpbWVzdGFtcDogbmV3IERhdGUoKS50b0lTT1N0cmluZygpLFxuICAgICAgICB9O1xuICAgICAgICBleHBlY3QobG9nLnNldmVyaXR5KS50b0JlKHNldmVyaXR5KTtcbiAgICAgIH0pO1xuICAgIH0pO1xuICB9KTtcblxuICBkZXNjcmliZSgnZW52aXJvbm1lbnQnLCAoKSA9PiB7XG4gICAgaXQoJ2RldmUgYWNlaXRhciBhbWJpZW50ZXMgY29tdW5zJywgKCkgPT4ge1xuICAgICAgY29uc3QgZW52aXJvbm1lbnRzID0gWydkZXZlbG9wbWVudCcsICdzdGFnaW5nJywgJ3Byb2R1Y3Rpb24nLCAndGVzdCddO1xuXG4gICAgICBlbnZpcm9ubWVudHMuZm9yRWFjaCgoZW52KSA9PiB7XG4gICAgICAgIGNvbnN0IGxvZzogU3RhbmRhcmRMb2dEYXRhID0ge1xuICAgICAgICAgIHNldmVyaXR5OiAnSU5GTycsXG4gICAgICAgICAgbWVzc2FnZTogJ3Rlc3QnLFxuICAgICAgICAgIHNlcnZpY2VOYW1lOiAndGVzdCcsXG4gICAgICAgICAgZW52aXJvbm1lbnQ6IGVudixcbiAgICAgICAgICB0aW1lc3RhbXA6IG5ldyBEYXRlKCkudG9JU09TdHJpbmcoKSxcbiAgICAgICAgfTtcbiAgICAgICAgZXhwZWN0KGxvZy5lbnZpcm9ubWVudCkudG9CZShlbnYpO1xuICAgICAgfSk7XG4gICAgfSk7XG4gIH0pO1xuXG4gIGRlc2NyaWJlKCd0aW1lc3RhbXAnLCAoKSA9PiB7XG4gICAgaXQoJ2RldmUgYWNlaXRhciBmb3JtYXRvIElTTyA4NjAxJywgKCkgPT4ge1xuICAgICAgY29uc3QgaXNvVGltZXN0YW1wID0gbmV3IERhdGUoKS50b0lTT1N0cmluZygpO1xuICAgICAgY29uc3QgbG9nOiBTdGFuZGFyZExvZ0RhdGEgPSB7XG4gICAgICAgIHNldmVyaXR5OiAnSU5GTycsXG4gICAgICAgIG1lc3NhZ2U6ICd0ZXN0JyxcbiAgICAgICAgc2VydmljZU5hbWU6ICd0ZXN0JyxcbiAgICAgICAgZW52aXJvbm1lbnQ6ICd0ZXN0JyxcbiAgICAgICAgdGltZXN0YW1wOiBpc29UaW1lc3RhbXAsXG4gICAgICB9O1xuXG4gICAgICBleHBlY3QobG9nLnRpbWVzdGFtcCkudG9NYXRjaChcbiAgICAgICAgL15cXGR7NH0tXFxkezJ9LVxcZHsyfVRcXGR7Mn06XFxkezJ9OlxcZHsyfVxcLlxcZHszfVokL1xuICAgICAgKTtcbiAgICB9KTtcbiAgfSk7XG5cbiAgZGVzY3JpYmUoJ2h0dHAuc3RhdHVzQ29kZScsICgpID0+IHtcbiAgICBpdCgnZGV2ZSBhY2VpdGFyIGPDs2RpZ29zIEhUVFAgdsOhbGlkb3MnLCAoKSA9PiB7XG4gICAgICBjb25zdCBzdGF0dXNDb2RlcyA9IFsyMDAsIDIwMSwgNDAwLCA0MDEsIDQwNCwgNTAwLCA1MDIsIDUwM107XG5cbiAgICAgIHN0YXR1c0NvZGVzLmZvckVhY2goKHN0YXR1c0NvZGUpID0+IHtcbiAgICAgICAgY29uc3QgbG9nOiBTdGFuZGFyZExvZ0RhdGEgPSB7XG4gICAgICAgICAgc2V2ZXJpdHk6ICdJTkZPJyxcbiAgICAgICAgICBtZXNzYWdlOiAndGVzdCcsXG4gICAgICAgICAgc2VydmljZU5hbWU6ICd0ZXN0JyxcbiAgICAgICAgICBlbnZpcm9ubWVudDogJ3Rlc3QnLFxuICAgICAgICAgIGh0dHA6IHsgc3RhdHVzQ29kZSB9LFxuICAgICAgICAgIHRpbWVzdGFtcDogbmV3IERhdGUoKS50b0lTT1N0cmluZygpLFxuICAgICAgICB9O1xuICAgICAgICBleHBlY3QobG9nLmh0dHA/LnN0YXR1c0NvZGUpLnRvQmUoc3RhdHVzQ29kZSk7XG4gICAgICB9KTtcbiAgICB9KTtcbiAgfSk7XG5cbiAgZGVzY3JpYmUoJ3BlcmZvcm1hbmNlLmR1cmF0aW9uTXMnLCAoKSA9PiB7XG4gICAgaXQoJ2RldmUgYWNlaXRhciB2YWxvcmVzIG51bcOpcmljb3MgcG9zaXRpdm9zJywgKCkgPT4ge1xuICAgICAgY29uc3QgZHVyYXRpb25zID0gWzAsIDEwMCwgNTAwLCAxMDAwLCA1MDAwLCAxMDAwMF07XG5cbiAgICAgIGR1cmF0aW9ucy5mb3JFYWNoKChkdXJhdGlvbk1zKSA9PiB7XG4gICAgICAgIGNvbnN0IGxvZzogU3RhbmRhcmRMb2dEYXRhID0ge1xuICAgICAgICAgIHNldmVyaXR5OiAnSU5GTycsXG4gICAgICAgICAgbWVzc2FnZTogJ3Rlc3QnLFxuICAgICAgICAgIHNlcnZpY2VOYW1lOiAndGVzdCcsXG4gICAgICAgICAgZW52aXJvbm1lbnQ6ICd0ZXN0JyxcbiAgICAgICAgICBwZXJmb3JtYW5jZTogeyBkdXJhdGlvbk1zIH0sXG4gICAgICAgICAgdGltZXN0YW1wOiBuZXcgRGF0ZSgpLnRvSVNPU3RyaW5nKCksXG4gICAgICAgIH07XG4gICAgICAgIGV4cGVjdChsb2cucGVyZm9ybWFuY2U/LmR1cmF0aW9uTXMpLnRvQmUoZHVyYXRpb25Ncyk7XG4gICAgICB9KTtcbiAgICB9KTtcbiAgfSk7XG5cbiAgZGVzY3JpYmUoJ3BlcmZvcm1hbmNlLnN1Y2Nlc3MnLCAoKSA9PiB7XG4gICAgaXQoJ2RldmUgYWNlaXRhciB2YWxvcmVzIGJvb2xlYW5vcycsICgpID0+IHtcbiAgICAgIFt0cnVlLCBmYWxzZV0uZm9yRWFjaCgoc3VjY2VzcykgPT4ge1xuICAgICAgICBjb25zdCBsb2c6IFN0YW5kYXJkTG9nRGF0YSA9IHtcbiAgICAgICAgICBzZXZlcml0eTogJ0lORk8nLFxuICAgICAgICAgIG1lc3NhZ2U6ICd0ZXN0JyxcbiAgICAgICAgICBzZXJ2aWNlTmFtZTogJ3Rlc3QnLFxuICAgICAgICAgIGVudmlyb25tZW50OiAndGVzdCcsXG4gICAgICAgICAgcGVyZm9ybWFuY2U6IHsgc3VjY2VzcyB9LFxuICAgICAgICAgIHRpbWVzdGFtcDogbmV3IERhdGUoKS50b0lTT1N0cmluZygpLFxuICAgICAgICB9O1xuICAgICAgICBleHBlY3QobG9nLnBlcmZvcm1hbmNlPy5zdWNjZXNzKS50b0JlKHN1Y2Nlc3MpO1xuICAgICAgfSk7XG4gICAgfSk7XG4gIH0pO1xufSk7XG4iXX0=
|