@mecanizou/telemetry-hub 1.4.0 → 1.4.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +14 -0
- package/CLAUDE.md +78 -0
- package/dist/core/logger.js +4 -2
- package/dist/sst/extraction.js +3 -4
- package/docs/c4-container.md +123 -0
- package/docs/service-profile.md +179 -0
- package/package.json +1 -1
package/CHANGELOG.md
CHANGED
|
@@ -1,3 +1,17 @@
|
|
|
1
|
+
## [1.4.2](https://github.com/mecanizou-eco/telemetry-hub/compare/v1.4.1...v1.4.2) (2026-03-17)
|
|
2
|
+
|
|
3
|
+
|
|
4
|
+
### Bug Fixes
|
|
5
|
+
|
|
6
|
+
* getEnvironmentStage on extractHttpAttributes ([1386a99](https://github.com/mecanizou-eco/telemetry-hub/commit/1386a99a14d6dff7d2cd57e23921cbac8f6ec165))
|
|
7
|
+
|
|
8
|
+
## [1.4.1](https://github.com/mecanizou-eco/telemetry-hub/compare/v1.4.0...v1.4.1) (2026-03-16)
|
|
9
|
+
|
|
10
|
+
|
|
11
|
+
### Bug Fixes
|
|
12
|
+
|
|
13
|
+
* disableEmiter meme logic ([e5908cb](https://github.com/mecanizou-eco/telemetry-hub/commit/e5908cb2c6b5c0eeb993f281f4fcfa9dd18c9359))
|
|
14
|
+
|
|
1
15
|
# [1.4.0](https://github.com/mecanizou-eco/telemetry-hub/compare/v1.3.15...v1.4.0) (2026-01-30)
|
|
2
16
|
|
|
3
17
|
|
package/CLAUDE.md
ADDED
|
@@ -0,0 +1,78 @@
|
|
|
1
|
+
# CLAUDE.md
|
|
2
|
+
|
|
3
|
+
This file provides guidance to Claude Code (claude.ai/code) when working with code in this repository.
|
|
4
|
+
|
|
5
|
+
## What Is This
|
|
6
|
+
|
|
7
|
+
`@mecanizou/telemetry-hub` is a shared npm package that provides standardized OpenTelemetry-based logging, tracing, and metrics for Mecanizou services. It supports two framework integrations: **SST/Middy** (Lambda handlers) and **Ts.ED** (Express-based serverless). Published to npm via semantic-release on push to `master`.
|
|
8
|
+
|
|
9
|
+
## Commands
|
|
10
|
+
|
|
11
|
+
```bash
|
|
12
|
+
npm install # install dependencies
|
|
13
|
+
npm run build # compile TypeScript to dist/
|
|
14
|
+
npm test # run all tests (vitest, single run)
|
|
15
|
+
npm run test:watch # vitest in watch mode
|
|
16
|
+
npm run test:coverage # vitest with v8 coverage report
|
|
17
|
+
npm run test:tsc # type-check without emitting
|
|
18
|
+
npm run lint # eslint
|
|
19
|
+
npm run check # prettier check
|
|
20
|
+
npm run fix # prettier auto-fix
|
|
21
|
+
```
|
|
22
|
+
|
|
23
|
+
Run a single test file:
|
|
24
|
+
```bash
|
|
25
|
+
npx vitest run src/core/__tests__/logger.test.ts
|
|
26
|
+
```
|
|
27
|
+
|
|
28
|
+
## Architecture
|
|
29
|
+
|
|
30
|
+
```
|
|
31
|
+
src/
|
|
32
|
+
├── index.ts # Package entry point — re-exports core + sst (tsed is commented out)
|
|
33
|
+
├── core/ # Framework-agnostic base classes
|
|
34
|
+
│ ├── logger.ts # StandardLogger — structured log emitter via OTel LoggerProvider
|
|
35
|
+
│ ├── logger-types.ts # StandardLogData interface, LogParams, IStandardLogger
|
|
36
|
+
│ ├── tracer.ts # StandardTracer — span lifecycle with standardized attributes
|
|
37
|
+
│ ├── tracer-types.ts # StandardTraceData, StandardSpanResult, IStandardTracer
|
|
38
|
+
│ └── environment.ts # getEnvironmentStage() — reads STAGE env var, normalizes "prod" → "production"
|
|
39
|
+
├── sst/ # SST/Middy integration (actively exported)
|
|
40
|
+
│ ├── telemetry.ts # Module-level OTel SDK init, singleton getStandardLogger/getStandardTracer, forceFlush
|
|
41
|
+
│ ├── extraction.ts # Extract HTTP attrs, logged user, default attrs from Lambda event/context
|
|
42
|
+
│ └── middy/
|
|
43
|
+
│ └── middleware.ts # middyTelemetryMiddleware — before/after/onError hooks with spans, metrics, structured logs
|
|
44
|
+
└── tsed/ # Ts.ED integration (exists but NOT exported from package entry — deployment issues)
|
|
45
|
+
├── config.ts # TsedTelemetryProvider — OTel provider lifecycle (init, flush, shutdown)
|
|
46
|
+
├── service.ts # TsedTelemetryService — @Injectable service wrapping StandardLogger
|
|
47
|
+
├── log-telemetry.ts # @TsedLogTelemetry decorator
|
|
48
|
+
└── sync-log-record-processor.ts # Custom sync processor for Lambda flush guarantees
|
|
49
|
+
```
|
|
50
|
+
|
|
51
|
+
### Key Design Decisions
|
|
52
|
+
|
|
53
|
+
- **SST telemetry module initializes at import time** (`src/sst/telemetry.ts`): OTel SDK, exporters, and providers are created at module scope. Logger/tracer are singletons accessed via `getStandardLogger()`/`getStandardTracer()`.
|
|
54
|
+
- **Log emission is disabled in development**: `StandardLogger` checks `STAGE` env var and skips OTel emit when `environment === 'development'`, falling back to `console.log`.
|
|
55
|
+
- **forceFlush is critical**: Lambda environments freeze after handler return. Both SST middleware and Ts.ED service call `forceFlush()` after every request to ensure telemetry is exported before freeze.
|
|
56
|
+
- **Tsed integration is commented out** in `src/index.ts` due to deployment issues with `@tsed/di` and `@tsed/platform-serverless` in SST builds. The code exists and is tested but not shipped.
|
|
57
|
+
|
|
58
|
+
### Environment Variables
|
|
59
|
+
|
|
60
|
+
| Variable | Required | Description |
|
|
61
|
+
|---|---|---|
|
|
62
|
+
| `OTEL_EXPORTER_OTLP_ENDPOINT` | Yes | Base URL for OTel collector (appends `/alloy/v1`) |
|
|
63
|
+
| `OTEL_EXPORTER_OTLP_USER` | Yes | Basic auth username |
|
|
64
|
+
| `OTEL_EXPORTER_OTLP_PASS` | Yes | Basic auth password |
|
|
65
|
+
| `SERVICE_NAME` | No | Defaults to `sst-service` (SST) or `unknown-service` (core/Ts.ED) |
|
|
66
|
+
| `STAGE` | No | `prod`/`production`/`staging`/`development`. Defaults to `development` |
|
|
67
|
+
|
|
68
|
+
## Release Process
|
|
69
|
+
|
|
70
|
+
- Commits must follow [Conventional Commits](https://www.conventionalcommits.org/) (`feat:`, `fix:`, `chore:`, etc.) — enforced by commitlint.
|
|
71
|
+
- Pushing to `master`, `alpha`, or `beta` triggers semantic-release which bumps version, updates CHANGELOG.md, publishes to npm, and creates a GitHub release.
|
|
72
|
+
- CI runs build + tests on PRs to `master`/`staging`.
|
|
73
|
+
|
|
74
|
+
## Code Style
|
|
75
|
+
|
|
76
|
+
- ESLint + Prettier: single quotes, semicolons, unix line endings, `no-console` (error level — use OTel logger instead).
|
|
77
|
+
- `eqeqeq` strict, `prefer-const`, `no-var`.
|
|
78
|
+
- Documentation and code comments are in Portuguese (Brazilian).
|
package/dist/core/logger.js
CHANGED
|
@@ -26,7 +26,9 @@ class StandardLogger {
|
|
|
26
26
|
this.defineDefaultAttributes(scopeAttributes);
|
|
27
27
|
this.environment = (0, environment_1.getEnvironmentStage)();
|
|
28
28
|
this.disableEmitter =
|
|
29
|
-
disableEmitter
|
|
29
|
+
disableEmitter === false
|
|
30
|
+
? false
|
|
31
|
+
: disableEmitter || this.environment === 'development';
|
|
30
32
|
}
|
|
31
33
|
defineDefaultAttributes(scopeAttributes) {
|
|
32
34
|
var _a, _b, _c;
|
|
@@ -132,4 +134,4 @@ class StandardLogger {
|
|
|
132
134
|
}
|
|
133
135
|
}
|
|
134
136
|
exports.StandardLogger = StandardLogger;
|
|
135
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"logger.js","sourceRoot":"","sources":["../../src/core/logger.ts"],"names":[],"mappings":"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","sourcesContent":["import { SeverityNumber, Logger } from '@opentelemetry/api-logs';\nimport { IStandardLogger, StandardLogData, LogParams } from './logger-types';\nimport { IStandardTracer } from './tracer-types';\nimport { Attributes, Span } from '@opentelemetry/api';\nimport { getEnvironmentStage } from './environment';\n\n/**\n * Logger core que implementa o padrão de logs estruturados.\n * Recebe um LoggerProvider do OpenTelemetry para emitir os logs.\n */\nexport class StandardLogger implements IStandardLogger {\n  public defaultAttributes: Record<string, any> = {};\n  public scopeAttributes?: Attributes;\n  private readonly logger: Logger;\n  private readonly tracer: IStandardTracer;\n  private readonly spanContext: Span;\n  private readonly environment: string;\n  private readonly defaultServiceName: string;\n  private readonly disableEmitter: boolean = false;\n\n  constructor(\n    logger: Logger,\n    tracer: IStandardTracer,\n    scopeAttributes?: Attributes,\n    disableEmitter?: boolean\n  ) {\n    this.logger = logger;\n    this.tracer = tracer;\n    this.spanContext = this.tracer.getTracer().startSpan('init-logger');\n\n    this.defaultServiceName = process.env.SERVICE_NAME\n      ? process.env.SERVICE_NAME\n      : 'unknown-service';\n\n    this.scopeAttributes = scopeAttributes;\n    this.defineDefaultAttributes(scopeAttributes);\n    this.environment = getEnvironmentStage();\n    this.disableEmitter =\n      disableEmitter || this.environment === 'development' || false;\n  }\n\n  public defineDefaultAttributes(scopeAttributes?: any) {\n    this.defaultAttributes = {\n      trace: {\n        id: this.spanContext.spanContext().traceId,\n      },\n      span: {\n        id: this.spanContext.spanContext().spanId,\n      },\n      function: {\n        name:\n          scopeAttributes?.['function']?.['name'] ||\n          scopeAttributes?.functionName ||\n          'unknown',\n        invocationId:\n          scopeAttributes?.['function']?.['invocationId'] ||\n          scopeAttributes?.['execution']?.['awsRequestId'] ||\n          scopeAttributes?.invocationId ||\n          'unknown',\n      },\n      request: {\n        id:\n          scopeAttributes?.['requestId'] ||\n          scopeAttributes?.requestId ||\n          'unknown',\n      },\n    };\n  }\n\n  async logError(data: LogParams): Promise<void> {\n    await this.log({\n      ...data,\n      severity: 'ERROR',\n    });\n  }\n\n  async logInfo(data: LogParams): Promise<void> {\n    await this.log({\n      ...data,\n      severity: 'INFO',\n    });\n  }\n\n  async logWarn(data: LogParams): Promise<void> {\n    await this.log({\n      ...data,\n      severity: 'WARN',\n    });\n  }\n\n  async logDebug(data: LogParams): Promise<void> {\n    await this.log({\n      ...data,\n      severity: 'DEBUG',\n    });\n  }\n\n  async log(\n    data: Omit<LogParams, 'timestamp' | 'environment'> & {\n      severity: StandardLogData['severity'];\n    }\n  ): Promise<void> {\n    const fullData: StandardLogData = {\n      ...data,\n      serviceName: data.serviceName || this.defaultServiceName,\n      environment: this.environment,\n      timestamp: new Date().toISOString(),\n    };\n\n    const severityMap = {\n      ERROR: SeverityNumber.ERROR,\n      WARN: SeverityNumber.WARN,\n      INFO: SeverityNumber.INFO,\n      DEBUG: SeverityNumber.DEBUG,\n    };\n\n    const logRecord = {\n      message: fullData.message || '',\n      severityNumber: severityMap[fullData.severity],\n      severityText: fullData.severity,\n      body: fullData.body || fullData.message || '',\n      attributes: this.buildAttributes(fullData),\n    };\n\n    if (this.disableEmitter) {\n      console.log('[Telemetry] Log emission disabled. Log record:', logRecord);\n    } else {\n      this.logger.emit(logRecord);\n    }\n  }\n\n  private buildAttributes(data: StandardLogData): Record<string, any> {\n    const attributes: Record<string, any> = {\n      duration: {\n        timestamp: data.timestamp,\n      },\n      service: {\n        name: data.serviceName || this.defaultServiceName,\n      },\n      deployment: {\n        environmentName: data.environment,\n      },\n      execution: {\n        ...data.execution,\n      },\n    };\n\n    // Informações do erro\n    if (data.error) {\n      attributes['error'] = {\n        type: data.error.name || '',\n        message: data.error.message || '',\n        stack: data.error.stack || '',\n      };\n    }\n\n    // Performance\n    if (data.performance) {\n      if (data.performance.durationMs)\n        attributes['duration']['ms'] = data.performance.durationMs;\n      if (data.performance.success !== undefined)\n        attributes['execution']['success'] = data.performance.success;\n    }\n\n    // Contexto adicional\n    if (data.context) {\n      attributes['context'] = JSON.stringify(data.context);\n    }\n\n    const entries = {\n      ...data,\n      ...this.defaultAttributes,\n      ...attributes,\n    };\n\n    delete entries.body;\n\n    // Filtrar valores undefined/null\n    return Object.fromEntries(\n      Object.entries(entries).filter(\n        ([, value]) => value !== undefined && value !== null\n      )\n    );\n  }\n}\n"]}
|
|
137
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"logger.js","sourceRoot":"","sources":["../../src/core/logger.ts"],"names":[],"mappings":"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","sourcesContent":["import { SeverityNumber, Logger } from '@opentelemetry/api-logs';\nimport { IStandardLogger, StandardLogData, LogParams } from './logger-types';\nimport { IStandardTracer } from './tracer-types';\nimport { Attributes, Span } from '@opentelemetry/api';\nimport { getEnvironmentStage } from './environment';\n\n/**\n * Logger core que implementa o padrão de logs estruturados.\n * Recebe um LoggerProvider do OpenTelemetry para emitir os logs.\n */\nexport class StandardLogger implements IStandardLogger {\n  public defaultAttributes: Record<string, any> = {};\n  public scopeAttributes?: Attributes;\n  private readonly logger: Logger;\n  private readonly tracer: IStandardTracer;\n  private readonly spanContext: Span;\n  private readonly environment: string;\n  private readonly defaultServiceName: string;\n  private readonly disableEmitter: boolean = false;\n\n  constructor(\n    logger: Logger,\n    tracer: IStandardTracer,\n    scopeAttributes?: Attributes,\n    disableEmitter?: boolean\n  ) {\n    this.logger = logger;\n    this.tracer = tracer;\n    this.spanContext = this.tracer.getTracer().startSpan('init-logger');\n\n    this.defaultServiceName = process.env.SERVICE_NAME\n      ? process.env.SERVICE_NAME\n      : 'unknown-service';\n\n    this.scopeAttributes = scopeAttributes;\n    this.defineDefaultAttributes(scopeAttributes);\n    this.environment = getEnvironmentStage();\n    this.disableEmitter =\n      disableEmitter === false\n        ? false\n        : disableEmitter || this.environment === 'development';\n  }\n\n  public defineDefaultAttributes(scopeAttributes?: any) {\n    this.defaultAttributes = {\n      trace: {\n        id: this.spanContext.spanContext().traceId,\n      },\n      span: {\n        id: this.spanContext.spanContext().spanId,\n      },\n      function: {\n        name:\n          scopeAttributes?.['function']?.['name'] ||\n          scopeAttributes?.functionName ||\n          'unknown',\n        invocationId:\n          scopeAttributes?.['function']?.['invocationId'] ||\n          scopeAttributes?.['execution']?.['awsRequestId'] ||\n          scopeAttributes?.invocationId ||\n          'unknown',\n      },\n      request: {\n        id:\n          scopeAttributes?.['requestId'] ||\n          scopeAttributes?.requestId ||\n          'unknown',\n      },\n    };\n  }\n\n  async logError(data: LogParams): Promise<void> {\n    await this.log({\n      ...data,\n      severity: 'ERROR',\n    });\n  }\n\n  async logInfo(data: LogParams): Promise<void> {\n    await this.log({\n      ...data,\n      severity: 'INFO',\n    });\n  }\n\n  async logWarn(data: LogParams): Promise<void> {\n    await this.log({\n      ...data,\n      severity: 'WARN',\n    });\n  }\n\n  async logDebug(data: LogParams): Promise<void> {\n    await this.log({\n      ...data,\n      severity: 'DEBUG',\n    });\n  }\n\n  async log(\n    data: Omit<LogParams, 'timestamp' | 'environment'> & {\n      severity: StandardLogData['severity'];\n    }\n  ): Promise<void> {\n    const fullData: StandardLogData = {\n      ...data,\n      serviceName: data.serviceName || this.defaultServiceName,\n      environment: this.environment,\n      timestamp: new Date().toISOString(),\n    };\n\n    const severityMap = {\n      ERROR: SeverityNumber.ERROR,\n      WARN: SeverityNumber.WARN,\n      INFO: SeverityNumber.INFO,\n      DEBUG: SeverityNumber.DEBUG,\n    };\n\n    const logRecord = {\n      message: fullData.message || '',\n      severityNumber: severityMap[fullData.severity],\n      severityText: fullData.severity,\n      body: fullData.body || fullData.message || '',\n      attributes: this.buildAttributes(fullData),\n    };\n\n    if (this.disableEmitter) {\n      console.log('[Telemetry] Log emission disabled. Log record:', logRecord);\n    } else {\n      this.logger.emit(logRecord);\n    }\n  }\n\n  private buildAttributes(data: StandardLogData): Record<string, any> {\n    const attributes: Record<string, any> = {\n      duration: {\n        timestamp: data.timestamp,\n      },\n      service: {\n        name: data.serviceName || this.defaultServiceName,\n      },\n      deployment: {\n        environmentName: data.environment,\n      },\n      execution: {\n        ...data.execution,\n      },\n    };\n\n    // Informações do erro\n    if (data.error) {\n      attributes['error'] = {\n        type: data.error.name || '',\n        message: data.error.message || '',\n        stack: data.error.stack || '',\n      };\n    }\n\n    // Performance\n    if (data.performance) {\n      if (data.performance.durationMs)\n        attributes['duration']['ms'] = data.performance.durationMs;\n      if (data.performance.success !== undefined)\n        attributes['execution']['success'] = data.performance.success;\n    }\n\n    // Contexto adicional\n    if (data.context) {\n      attributes['context'] = JSON.stringify(data.context);\n    }\n\n    const entries = {\n      ...data,\n      ...this.defaultAttributes,\n      ...attributes,\n    };\n\n    delete entries.body;\n\n    // Filtrar valores undefined/null\n    return Object.fromEntries(\n      Object.entries(entries).filter(\n        ([, value]) => value !== undefined && value !== null\n      )\n    );\n  }\n}\n"]}
|
package/dist/sst/extraction.js
CHANGED
|
@@ -3,6 +3,7 @@ Object.defineProperty(exports, "__esModule", { value: true });
|
|
|
3
3
|
exports.extractHttpAttributes = extractHttpAttributes;
|
|
4
4
|
exports.extractLoggedUser = extractLoggedUser;
|
|
5
5
|
exports.extractDefaultAttributes = extractDefaultAttributes;
|
|
6
|
+
const environment_1 = require("../core/environment");
|
|
6
7
|
function extractHttpAttributes(event) {
|
|
7
8
|
var _a, _b, _c, _d, _e, _f;
|
|
8
9
|
if (event === null || event === void 0 ? void 0 : event.requestContext) {
|
|
@@ -60,9 +61,7 @@ function extractDefaultAttributes(request) {
|
|
|
60
61
|
const routeKey = (event === null || event === void 0 ? void 0 : event.routeKey) || ((_a = event === null || event === void 0 ? void 0 : event.requestContext) === null || _a === void 0 ? void 0 : _a.routeKey) || 'unknown';
|
|
61
62
|
const requestId = (event === null || event === void 0 ? void 0 : event.requestId) || ((_b = event === null || event === void 0 ? void 0 : event.requestContext) === null || _b === void 0 ? void 0 : _b.requestId) || 'unknown';
|
|
62
63
|
const loggedUser = extractLoggedUser(event);
|
|
63
|
-
|
|
64
|
-
if (stage === 'prod')
|
|
65
|
-
stage = 'production';
|
|
64
|
+
const stage = (0, environment_1.getEnvironmentStage)();
|
|
66
65
|
return {
|
|
67
66
|
functionName,
|
|
68
67
|
invocationId,
|
|
@@ -90,4 +89,4 @@ function extractDefaultAttributes(request) {
|
|
|
90
89
|
};
|
|
91
90
|
}
|
|
92
91
|
}
|
|
93
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
92
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"extraction.js","sourceRoot":"","sources":["../../src/sst/extraction.ts"],"names":[],"mappings":"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","sourcesContent":["import { getEnvironmentStage } from '../core/environment';\n\nexport function extractHttpAttributes(event: any): {\n  method: string;\n  url: string;\n  routeKey?: string;\n  user_agent: string;\n  path_parameters: Record<string, string>;\n  query_string_parameters: Record<string, string>;\n  body: any;\n} {\n  if (event?.requestContext) {\n    return {\n      method: event.requestContext.http?.method || 'unknown',\n      routeKey: event.routeKey || event.requestContext?.routeKey || 'unknown',\n      url: event.requestContext.domainName && event.requestContext.http?.path,\n      user_agent: event.headers?.['user-agent'] || 'unknown',\n      path_parameters: event.pathParameters || {},\n      query_string_parameters: event.queryStringParameters || {},\n      body: event.body || {},\n    };\n  }\n\n  return {\n    method: event?.httpMethod || 'unknown',\n    routeKey: event?.routeKey || event.requestContext?.routeKey || 'unknown',\n    url: event?.path || 'unknown',\n    user_agent: event?.headers?.['User-Agent'] || 'unknown',\n    path_parameters: event?.pathParameters || {},\n    query_string_parameters: event?.queryStringParameters || {},\n    body: event?.body || {},\n  };\n}\n\nexport function extractLoggedUser(\n  event: any\n): { email?: string; name?: string; uid?: string } | undefined {\n  let loggedUser = event?.requestContext?.authorizer?.lambda?.loggedUser;\n\n  if (loggedUser) {\n    loggedUser =\n      typeof loggedUser === 'string' ? JSON.parse(loggedUser) : loggedUser;\n\n    const accountUser = loggedUser?.user?.account_user;\n    const firstAccountUser = Array.isArray(accountUser)\n      ? accountUser[0]\n      : accountUser;\n\n    if (loggedUser?.user && accountUser && firstAccountUser?.account) {\n      return {\n        email: loggedUser.user.email,\n        uid: firstAccountUser.account.uid,\n        name: firstAccountUser.account.name,\n      };\n    }\n  }\n\n  return undefined;\n}\n\nexport function extractDefaultAttributes(request: any): Record<string, any> {\n  try {\n    const lambdaContext = request?.context || request?.lambdaContext;\n    const functionName = lambdaContext?.functionName || 'unknown';\n    const invocationId = lambdaContext?.awsRequestId || 'unknown';\n    const serviceName = process.env.SERVICE_NAME || 'sst-service';\n    const event = request?.event || request;\n    const queryStringParameters = event?.queryStringParameters || {};\n    const origin = queryStringParameters.origin || 'unknown';\n    const http = extractHttpAttributes(event);\n    const routeKey =\n      event?.routeKey || event?.requestContext?.routeKey || 'unknown';\n    const requestId =\n      event?.requestId || event?.requestContext?.requestId || 'unknown';\n    const loggedUser = extractLoggedUser(event);\n    const stage = getEnvironmentStage();\n\n    return {\n      functionName,\n      invocationId,\n      serviceName,\n      origin,\n      routeKey,\n      requestId,\n      stage,\n      http,\n      loggedUser,\n    };\n  } catch (error) {\n    console.error('[Telemetry] Error extracting default attributes:', error);\n    return {\n      functionName: 'unknown',\n      invocationId: 'unknown',\n      serviceName: 'unknown',\n      origin: 'unknown',\n      routeKey: 'unknown',\n      requestId: 'unknown',\n      stage: 'unknown',\n      http: {},\n      loggedUser: undefined,\n    };\n  }\n}\n"]}
|
|
@@ -0,0 +1,123 @@
|
|
|
1
|
+
# C4 Container Diagram — @mecanizou/telemetry-hub
|
|
2
|
+
|
|
3
|
+
## Descricao do Servico
|
|
4
|
+
|
|
5
|
+
`@mecanizou/telemetry-hub` e uma biblioteca npm compartilhada que fornece logging, tracing e metricas padronizados baseados em OpenTelemetry para todos os servicos Mecanizou. Suporta duas integracoes: **SST/Middy** (Lambda handlers — ativa) e **Ts.ED** (Express serverless — presente mas nao exportada). Nao e um servico deployado — e consumido como dependencia por servicos Lambda e Ts.ED, que enviam sinais de telemetria para o Grafana Alloy (OTel Collector).
|
|
6
|
+
|
|
7
|
+
## Diagrama C4 Container
|
|
8
|
+
|
|
9
|
+
```mermaid
|
|
10
|
+
C4Container
|
|
11
|
+
title C4 Container — @mecanizou/telemetry-hub
|
|
12
|
+
|
|
13
|
+
Person(developer, "Desenvolvedor", "Desenvolve e publica o pacote")
|
|
14
|
+
|
|
15
|
+
System_Boundary(telemetry_hub, "@mecanizou/telemetry-hub (Biblioteca npm)") {
|
|
16
|
+
|
|
17
|
+
Container(core_logger, "StandardLogger", "TypeScript, OTel API Logs", "Emite log records estruturados via OTel LoggerProvider. Desabilitado em development (fallback console.log)")
|
|
18
|
+
Container(core_tracer, "StandardTracer", "TypeScript, OTel API", "Gerencia ciclo de vida de spans com SpanKind.SERVER e atributos padronizados")
|
|
19
|
+
|
|
20
|
+
Container(sst_telemetry, "SST Telemetry Module", "TypeScript, OTel SDK Node", "Inicializa OTel SDK no import (module scope). Cria exporters OTLP HTTP para traces, metrics e logs. Singletons getStandardLogger/getStandardTracer. forceFlush apos cada request")
|
|
21
|
+
Container(middy_middleware, "middyTelemetryMiddleware", "TypeScript, Middy", "Middleware before/after/onError: cria span, registra metricas (duration histogram, success/failure counters), emite log estruturado, chama forceFlush")
|
|
22
|
+
|
|
23
|
+
Container(sst_extraction, "SST Extraction Utils", "TypeScript", "Extrai atributos HTTP, loggedUser e contexto Lambda de APIGatewayProxyEvent v1/v2")
|
|
24
|
+
|
|
25
|
+
Container(tsed_provider, "TsedTelemetryProvider", "TypeScript, OTel SDK, Ts.ED DI", "Gerencia lifecycle OTel para Ts.ED (init/flush/shutdown). Usa SyncLogRecordProcessor customizado para Lambda")
|
|
26
|
+
Container(tsed_service, "TsedTelemetryService", "TypeScript, Ts.ED @Injectable", "Servico DI que wrapa StandardLogger + StandardTracer para controllers Ts.ED")
|
|
27
|
+
Container(tsed_decorator, "@TsedLogTelemetry", "TypeScript, Ts.ED Decorator", "Method decorator: cria span, registra metricas tsed_*, emite log, forceFlush. Equivalente ao Middy middleware")
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
System_Boundary(consumers, "Servicos Consumidores Mecanizou") {
|
|
31
|
+
|
|
32
|
+
Container(sst_lambdas, "SST Lambda Services", "Node.js 18-20, SST v2, Middy", "order-service, quote-service, pricing-service, cart-service, catalog-service, payment-service, etc. Importam middyTelemetryMiddleware")
|
|
33
|
+
Container(tsed_services, "Ts.ED Services", "Node.js, Ts.ED, Express", "crm-service e outros. Usam TsedTelemetryProvider + @TsedLogTelemetry (integracao desabilitada no momento)")
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
System_Boundary(aws, "AWS Infrastructure") {
|
|
37
|
+
|
|
38
|
+
Container(api_gateway, "API Gateway", "AWS API Gateway v1/v2", "Recebe requests HTTP e invoca Lambda handlers. Injeta requestContext com authorizer data")
|
|
39
|
+
Container(authorizer, "mecanizou-authorizer", "AWS Lambda, Node.js", "Lambda Authorizer que valida JWT/Cognito e injeta loggedUser no requestContext")
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
System_Ext(grafana_alloy, "Grafana Alloy", "OTel Collector (OTLP HTTP)", "Recebe traces, metrics e logs via OTLP HTTP com Basic Auth. Endpoint: $OTEL_EXPORTER_OTLP_ENDPOINT/alloy/v1")
|
|
43
|
+
System_Ext(grafana_cloud, "Grafana Cloud", "Loki, Tempo, Mimir", "Backend de observabilidade: armazena e visualiza logs, traces e metricas")
|
|
44
|
+
System_Ext(npm_registry, "npm Registry", "registry.npmjs.org", "Hospeda o pacote @mecanizou/telemetry-hub publicado")
|
|
45
|
+
System_Ext(github, "GitHub", "GitHub Actions, Releases", "Repositorio de codigo, CI/CD (build, test, semantic-release), GitHub Releases")
|
|
46
|
+
System_Ext(slack, "Slack", "Webhook", "Recebe notificacoes de build/release via webhook")
|
|
47
|
+
|
|
48
|
+
Rel(developer, github, "Push commits / Cria PRs")
|
|
49
|
+
Rel(github, npm_registry, "Publica pacote via semantic-release", "npm publish")
|
|
50
|
+
Rel(github, slack, "Envia notificacao de build", "HTTPS Webhook")
|
|
51
|
+
|
|
52
|
+
Rel(sst_lambdas, middy_middleware, "Importa e usa como Middy middleware", "npm dependency")
|
|
53
|
+
Rel(tsed_services, tsed_decorator, "Importa e usa como decorator", "npm dependency (desabilitado)")
|
|
54
|
+
Rel(tsed_services, tsed_provider, "Injeta via Ts.ED DI", "npm dependency (desabilitado)")
|
|
55
|
+
|
|
56
|
+
Rel(middy_middleware, sst_telemetry, "Usa singletons e forceFlush")
|
|
57
|
+
Rel(middy_middleware, sst_extraction, "Extrai atributos do evento Lambda")
|
|
58
|
+
Rel(middy_middleware, core_logger, "Emite logs estruturados (INFO/ERROR)")
|
|
59
|
+
Rel(middy_middleware, core_tracer, "Cria e finaliza spans")
|
|
60
|
+
|
|
61
|
+
Rel(tsed_decorator, tsed_service, "Obtem logger/tracer via DI")
|
|
62
|
+
Rel(tsed_service, tsed_provider, "Obtem LoggerProvider/TracerProvider")
|
|
63
|
+
Rel(tsed_service, core_logger, "Emite logs estruturados")
|
|
64
|
+
Rel(tsed_service, core_tracer, "Cria e finaliza spans")
|
|
65
|
+
|
|
66
|
+
Rel(sst_telemetry, grafana_alloy, "Envia traces via OTLP HTTP", "POST /alloy/v1/traces (Basic Auth)")
|
|
67
|
+
Rel(sst_telemetry, grafana_alloy, "Envia metrics via OTLP HTTP", "POST /alloy/v1/metrics (Basic Auth)")
|
|
68
|
+
Rel(sst_telemetry, grafana_alloy, "Envia logs via OTLP HTTP", "POST /alloy/v1/logs (Basic Auth)")
|
|
69
|
+
Rel(tsed_provider, grafana_alloy, "Envia traces, metrics e logs via OTLP HTTP", "POST /alloy/v1/* (Basic Auth)")
|
|
70
|
+
|
|
71
|
+
Rel(grafana_alloy, grafana_cloud, "Encaminha sinais de telemetria", "Remote Write / OTLP")
|
|
72
|
+
|
|
73
|
+
Rel(api_gateway, authorizer, "Invoca antes de cada request autenticado", "Lambda Authorizer")
|
|
74
|
+
Rel(authorizer, api_gateway, "Retorna loggedUser no requestContext", "Authorizer Response")
|
|
75
|
+
Rel(api_gateway, sst_lambdas, "Invoca Lambda handler com evento enriquecido", "AWS Lambda Invoke")
|
|
76
|
+
Rel(sst_extraction, api_gateway, "Le requestContext.authorizer.lambda.loggedUser", "Dados do evento Lambda")
|
|
77
|
+
```
|
|
78
|
+
|
|
79
|
+
## Tabela de Containers
|
|
80
|
+
|
|
81
|
+
| Container | Tecnologia | Responsabilidade | Owner |
|
|
82
|
+
|---|---|---|---|
|
|
83
|
+
| **StandardLogger** | TypeScript, `@opentelemetry/api-logs` | Emite log records estruturados (ERROR/WARN/INFO/DEBUG) via OTel LoggerProvider. Desabilitado em development (fallback `console.log`). Enriquece logs com trace ID, span ID, service name, HTTP context e user identity | telemetry-hub (core) |
|
|
84
|
+
| **StandardTracer** | TypeScript, `@opentelemetry/api` | Gerencia ciclo de vida de spans (`startSpan`/`end`/`setSuccess`/`setError`/`recordException`) com `SpanKind.SERVER` e atributos padronizados de servico, ambiente, execucao e performance | telemetry-hub (core) |
|
|
85
|
+
| **SST Telemetry Module** | TypeScript, `@opentelemetry/sdk-node`, OTLP HTTP Exporters | Inicializa OTel SDK no module scope (import time). Cria 3 exporters OTLP HTTP (traces, metrics, logs) com Basic Auth. Expoe singletons `getStandardLogger`/`getStandardTracer` e `forceFlush` para garantir envio antes do Lambda freeze | telemetry-hub (sst) |
|
|
86
|
+
| **middyTelemetryMiddleware** | TypeScript, `@middy/core` | Middleware Middy com hooks before/after/onError. Cria span por invocacao, registra metricas (`lambda_execution_duration` histogram, `total_successful`/`total_failed` counters), emite log estruturado e chama `forceFlush` | telemetry-hub (sst) |
|
|
87
|
+
| **SST Extraction Utils** | TypeScript | Extrai atributos HTTP (method, URL, route key, user agent, body, params, query), `loggedUser` do authorizer e atributos padrao (function name, AWS request ID, stage, service name) de eventos API Gateway v1/v2 | telemetry-hub (sst) |
|
|
88
|
+
| **TsedTelemetryProvider** | TypeScript, `@opentelemetry/sdk-*`, `@tsed/di` | Gerencia lifecycle completo OTel para Ts.ED: init, flush, shutdown. Usa `SyncLogRecordProcessor` customizado que rastreia promises de export para garantir flush em Lambda. **Nao exportado atualmente** | telemetry-hub (tsed) |
|
|
89
|
+
| **TsedTelemetryService** | TypeScript, `@tsed/di @Injectable` | Servico injetavel via DI que wrapa StandardLogger + StandardTracer. Expoe `logError`/`logException` com `forceFlush` automatico. **Nao exportado atualmente** | telemetry-hub (tsed) |
|
|
90
|
+
| **@TsedLogTelemetry** | TypeScript, Ts.ED Decorator | Method decorator que automatiza span creation, metricas (`tsed_execution_duration`, `tsed_successful_requests`, `tsed_failed_requests`), logging e flush. Equivalente funcional do middyTelemetryMiddleware para Ts.ED. **Nao exportado atualmente** | telemetry-hub (tsed) |
|
|
91
|
+
| **SST Lambda Services** | Node.js 18-20, SST v2, Middy | Servicos backend Mecanizou (order, quote, pricing, cart, catalog, payment, etc.) que importam `middyTelemetryMiddleware` como dependencia npm | Equipes de produto |
|
|
92
|
+
| **Ts.ED Services** | Node.js, Ts.ED, Express Serverless | Servicos como crm-service que usariam TsedTelemetryProvider + @TsedLogTelemetry (integracao desabilitada por conflito de build com SST) | Equipes de produto |
|
|
93
|
+
| **API Gateway** | AWS API Gateway v1/v2 | Recebe requests HTTP externos, executa Lambda Authorizer para autenticacao e invoca Lambda handlers com evento enriquecido (requestContext com loggedUser) | Infra |
|
|
94
|
+
| **mecanizou-authorizer** | AWS Lambda, Node.js, JWT/Cognito | Lambda Authorizer que valida tokens JWT (RS256) e Cognito, injeta `loggedUser` (email, uid, name, roles) no `requestContext.authorizer.lambda` | Auth |
|
|
95
|
+
| **Grafana Alloy** | OTel Collector, OTLP HTTP | Recebe sinais de telemetria (traces, metrics, logs) via OTLP HTTP com Basic Auth no endpoint `/alloy/v1/*`. Encaminha para backends Grafana Cloud | Observabilidade |
|
|
96
|
+
| **Grafana Cloud** | Loki (logs), Tempo (traces), Mimir (metrics) | Backend de observabilidade: armazena, indexa e visualiza logs, traces e metricas de todos os servicos Mecanizou | Observabilidade |
|
|
97
|
+
| **npm Registry** | registry.npmjs.org | Hospeda o pacote `@mecanizou/telemetry-hub` publicado. Autenticacao via `NPM_PUBLISH_TOKEN` | Externo |
|
|
98
|
+
| **GitHub** | GitHub Actions, Releases, semantic-release | Repositorio de codigo, CI/CD (build, test em PRs; semantic-release em push to master/alpha/beta), GitHub Releases automaticos | Externo |
|
|
99
|
+
| **Slack** | Webhook HTTPS | Recebe notificacoes de build/release do GitHub Actions via `SLACK_WEBHOOK_URL` | Externo |
|
|
100
|
+
|
|
101
|
+
## Sinais de Telemetria Emitidos
|
|
102
|
+
|
|
103
|
+
### Metricas (OTel Metrics)
|
|
104
|
+
|
|
105
|
+
| Metrica | Tipo | Framework | Labels |
|
|
106
|
+
|---|---|---|---|
|
|
107
|
+
| `lambda_execution_duration` | Histogram (ms) | SST/Middy | `origin`, `status`, `functionName`, `requestId`, `routeKey`, `serviceName`, `environmentName` |
|
|
108
|
+
| `total_successful` | Counter | SST/Middy | `origin`, `functionName`, `requestId`, `routeKey`, `serviceName`, `environmentName` |
|
|
109
|
+
| `total_failed` | Counter | SST/Middy | `origin`, `error_type`, `functionName`, `requestId`, `routeKey`, `serviceName`, `environmentName` |
|
|
110
|
+
| `tsed_execution_duration` | Histogram (ms) | Ts.ED | `origin`, `status`, `controller`, `method`, `serviceName`, `environmentName` |
|
|
111
|
+
| `tsed_successful_requests` | Counter | Ts.ED | `origin`, `controller`, `method`, `serviceName`, `environmentName` |
|
|
112
|
+
| `tsed_failed_requests` | Counter | Ts.ED | `origin`, `error_type`, `controller`, `method`, `serviceName`, `environmentName` |
|
|
113
|
+
|
|
114
|
+
### Variaveis de Ambiente
|
|
115
|
+
|
|
116
|
+
| Variavel | Obrigatoria | Default | Descricao |
|
|
117
|
+
|---|---|---|---|
|
|
118
|
+
| `OTEL_EXPORTER_OTLP_ENDPOINT` | Sim (prod) | — | URL base do OTel Collector (Grafana Alloy) |
|
|
119
|
+
| `OTEL_EXPORTER_OTLP_USER` | Sim (prod) | — | Username para Basic Auth do OTLP |
|
|
120
|
+
| `OTEL_EXPORTER_OTLP_PASS` | Sim (prod) | — | Password para Basic Auth do OTLP |
|
|
121
|
+
| `SERVICE_NAME` | Nao | `sst-service` / `unknown-service` | Nome do servico no OTel resource |
|
|
122
|
+
| `STAGE` | Nao | `development` | Ambiente (`prod` normalizado para `production`) |
|
|
123
|
+
| `OTEL_ENABLED` | Nao | `true` | Habilita/desabilita OTel (somente Ts.ED) |
|
|
@@ -0,0 +1,179 @@
|
|
|
1
|
+
# Service Profile: telemetry-hub
|
|
2
|
+
|
|
3
|
+
## 1. Nome e Propósito
|
|
4
|
+
|
|
5
|
+
**Nome:** `@mecanizou/telemetry-hub`
|
|
6
|
+
|
|
7
|
+
**Tipo:** Biblioteca npm compartilhada (não é um serviço deployado).
|
|
8
|
+
|
|
9
|
+
**Propósito:** Fornecer logging estruturado, tracing distribuído e métricas padronizadas via OpenTelemetry para os serviços da Mecanizou. Abstrai a configuração dos exporters OTLP e expõe classes/middlewares prontos para dois frameworks: SST/Middy (Lambdas) e Ts.ED (Express serverless).
|
|
10
|
+
|
|
11
|
+
## 2. Stack Tecnológica
|
|
12
|
+
|
|
13
|
+
| Camada | Tecnologia | Versão |
|
|
14
|
+
|---|---|---|
|
|
15
|
+
| Linguagem | TypeScript | ^5.9.3 |
|
|
16
|
+
| Runtime | Node.js | 24.x (CI release), 18.x (CI PR) |
|
|
17
|
+
| Módulo | CommonJS (target ES6) | |
|
|
18
|
+
| Testes | Vitest | ^4.0.13 |
|
|
19
|
+
| Cobertura | @vitest/coverage-v8 | ^4.0.15 |
|
|
20
|
+
| Linting | ESLint + Prettier | ^9.39 / ^3.6 |
|
|
21
|
+
| Release | semantic-release | ^25.0.2 |
|
|
22
|
+
| Commits | Conventional Commits (commitlint) | @commitlint/config-conventional ^20.0 |
|
|
23
|
+
|
|
24
|
+
### Dependências de Produção
|
|
25
|
+
|
|
26
|
+
| Pacote | Versão | Papel |
|
|
27
|
+
|---|---|---|
|
|
28
|
+
| `@opentelemetry/api` | ^1.9.0 | API de tracing e métricas |
|
|
29
|
+
| `@opentelemetry/api-logs` | ^0.56.0 | API de logs |
|
|
30
|
+
| `@opentelemetry/sdk-node` | ^0.56.0 | SDK Node.js OTel |
|
|
31
|
+
| `@opentelemetry/sdk-logs` | ^0.56.0 | LoggerProvider, BatchLogRecordProcessor |
|
|
32
|
+
| `@opentelemetry/sdk-metrics` | ^1.30.0 | MeterProvider, PeriodicExportingMetricReader |
|
|
33
|
+
| `@opentelemetry/exporter-logs-otlp-http` | ^0.56.0 | Exportador de logs via HTTP/OTLP |
|
|
34
|
+
| `@opentelemetry/exporter-trace-otlp-http` | ^0.56.0 | Exportador de traces via HTTP/OTLP |
|
|
35
|
+
| `@opentelemetry/exporter-metrics-otlp-http` | ^0.56.0 | Exportador de métricas via HTTP/OTLP |
|
|
36
|
+
| `@opentelemetry/resources` | ^1.30.0 | Definição de Resource (service.name) |
|
|
37
|
+
| `@opentelemetry/semantic-conventions` | ^1.30.0 | Constantes semânticas (ATTR_SERVICE_NAME) |
|
|
38
|
+
| `@middy/core` | ^6.4.5 | Framework de middleware para Lambda |
|
|
39
|
+
| `@tsed/di` | 8.19.5 | Injeção de dependência Ts.ED |
|
|
40
|
+
| `@tsed/platform-serverless` | 8.19.5 | Plataforma serverless Ts.ED |
|
|
41
|
+
| `find-my-way` | 8.1.0 | Router HTTP (dependência declarada) |
|
|
42
|
+
|
|
43
|
+
## 3. Endpoints/Rotas Expostos
|
|
44
|
+
|
|
45
|
+
**Nenhum.** Este repositório é uma biblioteca npm, não um serviço HTTP. Não possui rotas, handlers Lambda, ou endpoints próprios.
|
|
46
|
+
|
|
47
|
+
O que ele expõe são **middlewares e decorators** para serem usados por outros serviços:
|
|
48
|
+
|
|
49
|
+
| Export | Tipo | Usado por |
|
|
50
|
+
|---|---|---|
|
|
51
|
+
| `middyTelemetryMiddleware()` | Middleware Middy (before/after/onError) | Serviços SST/Lambda |
|
|
52
|
+
| `TsedTelemetryProvider` | Classe de configuração OTel | Serviços Ts.ED |
|
|
53
|
+
| `TsedTelemetryService` | Serviço injetável `@Injectable()` | Serviços Ts.ED |
|
|
54
|
+
| `@TsedLogTelemetry()` | Decorator de método | Controllers Ts.ED |
|
|
55
|
+
|
|
56
|
+
**Nota:** As exportações Ts.ED estão comentadas em `src/index.ts` e não são incluídas no pacote publicado atualmente, devido a problemas de compatibilidade com deploy SST.
|
|
57
|
+
|
|
58
|
+
## 4. Dependências Externas
|
|
59
|
+
|
|
60
|
+
### 4.1. Coletor OpenTelemetry (OTLP HTTP)
|
|
61
|
+
|
|
62
|
+
A biblioteca faz requisições HTTP para um coletor OpenTelemetry compatível. As URLs são construídas a partir da variável de ambiente base:
|
|
63
|
+
|
|
64
|
+
| Variável | Obrigatória | Descrição |
|
|
65
|
+
|---|---|---|
|
|
66
|
+
| `OTEL_EXPORTER_OTLP_ENDPOINT` | Sim | URL base do coletor (ex: `https://seu-endpoint-otel.com`) |
|
|
67
|
+
| `OTEL_EXPORTER_OTLP_USER` | Sim | Usuário para autenticação Basic Auth |
|
|
68
|
+
| `OTEL_EXPORTER_OTLP_PASS` | Sim | Senha para autenticação Basic Auth |
|
|
69
|
+
|
|
70
|
+
**Endpoints construídos:**
|
|
71
|
+
|
|
72
|
+
- **SST** (`src/sst/telemetry.ts`):
|
|
73
|
+
- `{OTEL_EXPORTER_OTLP_ENDPOINT}/alloy/v1/traces`
|
|
74
|
+
- `{OTEL_EXPORTER_OTLP_ENDPOINT}/alloy/v1/metrics`
|
|
75
|
+
- `{OTEL_EXPORTER_OTLP_ENDPOINT}/alloy/v1/logs`
|
|
76
|
+
|
|
77
|
+
- **Ts.ED** (`src/tsed/config.ts`):
|
|
78
|
+
- `{OTEL_EXPORTER_OTLP_ENDPOINT}/alloy/v1/v1/traces`
|
|
79
|
+
- `{OTEL_EXPORTER_OTLP_ENDPOINT}/alloy/v1/v1/metrics`
|
|
80
|
+
- `{OTEL_EXPORTER_OTLP_ENDPOINT}/alloy/v1/v1/logs`
|
|
81
|
+
|
|
82
|
+
> **Observação:** Na implementação Ts.ED, o endpoint recebe o sufixo `/alloy/v1` no construtor do `TsedTelemetryProvider` e depois `/v1/logs` etc. no exporter, resultando em `/alloy/v1/v1/...`. Isso diverge do padrão SST que usa `/alloy/v1/...`. Pode ser intencional ou um bug.
|
|
83
|
+
|
|
84
|
+
**Autenticação:** Basic Auth via header `Authorization: Basic <base64(user:password)>`.
|
|
85
|
+
|
|
86
|
+
### 4.2. Variáveis de Ambiente Adicionais
|
|
87
|
+
|
|
88
|
+
| Variável | Obrigatória | Default | Uso |
|
|
89
|
+
|---|---|---|---|
|
|
90
|
+
| `SERVICE_NAME` | Não | `sst-service` (SST), `unknown-service` (core/Ts.ED) | Atributo `service.name` nos sinais OTel |
|
|
91
|
+
| `STAGE` | Não | `development` | Ambiente de execução. `prod` é normalizado para `production` |
|
|
92
|
+
| `OTEL_ENABLED` | Não | `true` | Apenas Ts.ED: `false` desabilita a inicialização OTel |
|
|
93
|
+
| `NPM_PUBLISH_TOKEN` | Sim (CI) | — | Token de autenticação para publicar no registry npm |
|
|
94
|
+
|
|
95
|
+
### 4.3. Bancos de Dados
|
|
96
|
+
|
|
97
|
+
**Nenhum.** A biblioteca não acessa bancos de dados diretamente.
|
|
98
|
+
|
|
99
|
+
### 4.4. Filas / Mensageria
|
|
100
|
+
|
|
101
|
+
**Nenhum.** Não há produção ou consumo de mensagens SQS, SNS ou EventBridge.
|
|
102
|
+
|
|
103
|
+
### 4.5. APIs Externas / Outros Serviços Internos
|
|
104
|
+
|
|
105
|
+
Nenhuma chamada direta a APIs externas ou serviços internos. A única comunicação de rede é com o coletor OTLP descrito acima.
|
|
106
|
+
|
|
107
|
+
## 5. Mecanismos de Autenticação/Autorização
|
|
108
|
+
|
|
109
|
+
A biblioteca **não implementa** autenticação ou autorização própria.
|
|
110
|
+
|
|
111
|
+
No entanto, o módulo de extração SST (`src/sst/extraction.ts`) **lê** dados do usuário autenticado a partir do evento Lambda:
|
|
112
|
+
|
|
113
|
+
```typescript
|
|
114
|
+
// Extrai loggedUser injetado pelo mecanizou-authorizer
|
|
115
|
+
event.requestContext.authorizer.lambda.loggedUser
|
|
116
|
+
```
|
|
117
|
+
|
|
118
|
+
Campos extraídos: `email`, `uid` (account), `name` (account). Esses dados são incluídos como atributos nos logs e traces para correlação, mas não são usados para decisões de autorização.
|
|
119
|
+
|
|
120
|
+
O decorator Ts.ED (`src/tsed/log-telemetry.ts`) tenta acessar `LoggedUserIdentifier.use($ctx)` para extrair `accountUserUid`, `accountUid` e `applicationUid`, mas faz isso de forma defensiva (try/catch) pois é uma dependência da aplicação consumidora.
|
|
121
|
+
|
|
122
|
+
## 6. Eventos Produzidos e Consumidos
|
|
123
|
+
|
|
124
|
+
**Nenhum.** A biblioteca não produz nem consome eventos de mensageria (SQS, SNS, EventBridge, DMS).
|
|
125
|
+
|
|
126
|
+
Os "eventos" que ela produz são sinais de **observabilidade** enviados ao coletor OTLP:
|
|
127
|
+
|
|
128
|
+
### Sinais OTel Emitidos
|
|
129
|
+
|
|
130
|
+
**Logs estruturados** (via `StandardLogger`):
|
|
131
|
+
- Severidades: `ERROR`, `WARN`, `INFO`, `DEBUG`
|
|
132
|
+
- Atributos padronizados: `service.name`, `deployment.environmentName`, `duration.timestamp`, `error.*`, `execution.*`, `http.*`, `context`, `function.*`, `request.*`, `trace.*`, `span.*`
|
|
133
|
+
|
|
134
|
+
**Traces** (via `StandardTracer`):
|
|
135
|
+
- Spans com `SpanKind.SERVER`
|
|
136
|
+
- Atributos: `service.name`, `deployment.environmentName`, `execution.*`, `duration.*`, `error.*`
|
|
137
|
+
- Status: `OK` ou `ERROR`
|
|
138
|
+
|
|
139
|
+
**Métricas** (via OpenTelemetry Metrics API):
|
|
140
|
+
|
|
141
|
+
| Métrica | Tipo | Framework | Descrição |
|
|
142
|
+
|---|---|---|---|
|
|
143
|
+
| `lambda_execution_duration` | Histogram (ms) | SST | Duração de execução do Lambda |
|
|
144
|
+
| `total_successful` | Counter | SST | Total de execuções bem-sucedidas |
|
|
145
|
+
| `total_failed` | Counter | SST | Total de execuções com falha |
|
|
146
|
+
| `tsed_execution_duration` | Histogram (ms) | Ts.ED | Duração de execução do método |
|
|
147
|
+
| `tsed_successful_requests` | Counter | Ts.ED | Total de requisições bem-sucedidas |
|
|
148
|
+
| `tsed_failed_requests` | Counter | Ts.ED | Total de requisições com falha |
|
|
149
|
+
|
|
150
|
+
Labels das métricas: `origin`, `status`, `functionName`/`controller`, `method`, `routeKey`, `requestId`, `serviceName`, `environmentName`, `error_type` (em falhas).
|
|
151
|
+
|
|
152
|
+
## 7. Dados Armazenados
|
|
153
|
+
|
|
154
|
+
**Nenhum.** A biblioteca não gerencia entidades, tabelas ou qualquer armazenamento persistente. Ela apenas emite sinais de telemetria para um coletor externo.
|
|
155
|
+
|
|
156
|
+
## 8. Infraestrutura
|
|
157
|
+
|
|
158
|
+
**Não possui infraestrutura própria.** Não há `sst.config.ts`, `serverless.yml`, `template.yaml`, diretório `infra/`, `Dockerfile`, ou `docker-compose.yml`.
|
|
159
|
+
|
|
160
|
+
O repositório é uma biblioteca npm pura. A infraestrutura relevante é:
|
|
161
|
+
|
|
162
|
+
### CI/CD (GitHub Actions)
|
|
163
|
+
|
|
164
|
+
| Workflow | Trigger | Ações |
|
|
165
|
+
|---|---|---|
|
|
166
|
+
| `release.yml` | Push em `master`, `alpha`, `beta` | Install → Build → semantic-release (publica no npm) → Slack notify |
|
|
167
|
+
| `pull_request.yml` | PR para `master`, `staging` | Install → Build → Test |
|
|
168
|
+
|
|
169
|
+
### Publicação npm
|
|
170
|
+
|
|
171
|
+
- Registry: `registry.npmjs.org` (configurado em `.npmrc`)
|
|
172
|
+
- Autenticação: token via `NPM_PUBLISH_TOKEN`
|
|
173
|
+
- Artefato publicado: diretório `dist/` (CommonJS compilado com declarations)
|
|
174
|
+
- Versionamento: automático via semantic-release baseado em conventional commits
|
|
175
|
+
- Branches de release: `master` (estável), `alpha` e `beta` (pre-release)
|
|
176
|
+
|
|
177
|
+
### Destino do pacote publicado
|
|
178
|
+
|
|
179
|
+
O pacote `@mecanizou/telemetry-hub` é consumido como dependência por outros serviços Mecanizou (SST Lambdas e Ts.ED services) que o instalam via `npm install @mecanizou/telemetry-hub`.
|