@victor-studio/monitor 0.2.0 → 0.5.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (71) hide show
  1. package/dist/adapters/bullmq/bullmq.cjs +85 -0
  2. package/dist/adapters/bullmq/bullmq.cjs.map +1 -0
  3. package/dist/adapters/bullmq/bullmq.d.cts +176 -0
  4. package/dist/adapters/bullmq/bullmq.d.ts +176 -0
  5. package/dist/adapters/bullmq/bullmq.js +82 -0
  6. package/dist/adapters/bullmq/bullmq.js.map +1 -0
  7. package/dist/adapters/gemini/gemini.cjs +65 -0
  8. package/dist/adapters/gemini/gemini.cjs.map +1 -0
  9. package/dist/adapters/gemini/gemini.d.cts +173 -0
  10. package/dist/adapters/gemini/gemini.d.ts +173 -0
  11. package/dist/adapters/gemini/gemini.js +62 -0
  12. package/dist/adapters/gemini/gemini.js.map +1 -0
  13. package/dist/adapters/index.cjs +56 -0
  14. package/dist/adapters/index.cjs.map +1 -0
  15. package/dist/adapters/index.d.cts +154 -0
  16. package/dist/adapters/index.d.ts +154 -0
  17. package/dist/adapters/index.js +53 -0
  18. package/dist/adapters/index.js.map +1 -0
  19. package/dist/adapters/neon/neon.cjs +36 -0
  20. package/dist/adapters/neon/neon.cjs.map +1 -0
  21. package/dist/adapters/neon/neon.d.cts +151 -0
  22. package/dist/adapters/neon/neon.d.ts +151 -0
  23. package/dist/adapters/neon/neon.js +34 -0
  24. package/dist/adapters/neon/neon.js.map +1 -0
  25. package/dist/adapters/railway/railway.cjs +20 -0
  26. package/dist/adapters/railway/railway.cjs.map +1 -0
  27. package/dist/adapters/railway/railway.d.cts +23 -0
  28. package/dist/adapters/railway/railway.d.ts +23 -0
  29. package/dist/adapters/railway/railway.js +18 -0
  30. package/dist/adapters/railway/railway.js.map +1 -0
  31. package/dist/adapters/resend/resend.cjs +36 -0
  32. package/dist/adapters/resend/resend.cjs.map +1 -0
  33. package/dist/adapters/resend/resend.d.cts +157 -0
  34. package/dist/adapters/resend/resend.d.ts +157 -0
  35. package/dist/adapters/resend/resend.js +34 -0
  36. package/dist/adapters/resend/resend.js.map +1 -0
  37. package/dist/adapters/upstash/upstash.cjs +39 -0
  38. package/dist/adapters/upstash/upstash.cjs.map +1 -0
  39. package/dist/adapters/upstash/upstash.d.cts +150 -0
  40. package/dist/adapters/upstash/upstash.d.ts +150 -0
  41. package/dist/adapters/upstash/upstash.js +37 -0
  42. package/dist/adapters/upstash/upstash.js.map +1 -0
  43. package/dist/adapters/vercel/vercel.cjs +18 -0
  44. package/dist/adapters/vercel/vercel.cjs.map +1 -0
  45. package/dist/adapters/vercel/vercel.d.cts +23 -0
  46. package/dist/adapters/vercel/vercel.d.ts +23 -0
  47. package/dist/adapters/vercel/vercel.js +16 -0
  48. package/dist/adapters/vercel/vercel.js.map +1 -0
  49. package/dist/deploys/index.cjs +53 -0
  50. package/dist/deploys/index.cjs.map +1 -0
  51. package/dist/deploys/index.d.cts +174 -0
  52. package/dist/deploys/index.d.ts +174 -0
  53. package/dist/deploys/index.js +50 -0
  54. package/dist/deploys/index.js.map +1 -0
  55. package/dist/errors/index.cjs +93 -0
  56. package/dist/errors/index.cjs.map +1 -0
  57. package/dist/errors/index.d.cts +183 -0
  58. package/dist/errors/index.d.ts +183 -0
  59. package/dist/errors/index.js +88 -0
  60. package/dist/errors/index.js.map +1 -0
  61. package/dist/index.cjs +96 -0
  62. package/dist/index.cjs.map +1 -1
  63. package/dist/index.d.cts +41 -1
  64. package/dist/index.d.ts +41 -1
  65. package/dist/index.js +95 -1
  66. package/dist/index.js.map +1 -1
  67. package/dist/next/index.d.cts +5 -0
  68. package/dist/next/index.d.ts +5 -0
  69. package/dist/react/index.d.cts +5 -0
  70. package/dist/react/index.d.ts +5 -0
  71. package/package.json +51 -1
@@ -0,0 +1,173 @@
1
+ interface HeartbeatData {
2
+ status: 'online' | 'offline';
3
+ latencyMs: number;
4
+ }
5
+ interface RequestData {
6
+ method: string;
7
+ route: string;
8
+ statusCode: number;
9
+ responseTimeMs: number;
10
+ userAgent?: string;
11
+ region?: string;
12
+ }
13
+ interface VitalData {
14
+ name: 'LCP' | 'INP' | 'CLS' | 'FCP' | 'TTFB';
15
+ value: number;
16
+ rating: 'good' | 'needs-improvement' | 'poor';
17
+ page?: string;
18
+ deviceType?: string;
19
+ browser?: string;
20
+ }
21
+ interface ErrorData {
22
+ type: 'unhandled' | 'caught' | 'promise';
23
+ message: string;
24
+ stack?: string;
25
+ groupingKey: string;
26
+ route?: string;
27
+ method?: string;
28
+ statusCode?: number;
29
+ environment?: string;
30
+ commitHash?: string;
31
+ extra?: Record<string, string>;
32
+ }
33
+ interface DeployData {
34
+ commitHash: string;
35
+ branch?: string;
36
+ author?: string;
37
+ status: 'started' | 'succeeded' | 'failed';
38
+ buildDurationMs?: number;
39
+ url?: string;
40
+ environment?: string;
41
+ provider?: string;
42
+ }
43
+ interface AdapterData {
44
+ adapter: string;
45
+ operation: string;
46
+ durationMs: number;
47
+ success: boolean;
48
+ error?: string;
49
+ meta?: Record<string, string>;
50
+ }
51
+ type MonitorEvent = {
52
+ type: 'heartbeat';
53
+ data: HeartbeatData;
54
+ timestamp: string;
55
+ } | {
56
+ type: 'request';
57
+ data: RequestData;
58
+ timestamp: string;
59
+ } | {
60
+ type: 'vital';
61
+ data: VitalData;
62
+ timestamp: string;
63
+ } | {
64
+ type: 'error';
65
+ data: ErrorData;
66
+ timestamp: string;
67
+ } | {
68
+ type: 'deploy';
69
+ data: DeployData;
70
+ timestamp: string;
71
+ } | {
72
+ type: 'adapter';
73
+ data: AdapterData;
74
+ timestamp: string;
75
+ };
76
+ /** Hook para filtrar/modificar eventos antes do envio */
77
+ type BeforeSendHook = (event: MonitorEvent) => MonitorEvent | null;
78
+
79
+ interface MonitorConfig {
80
+ /** ID do projeto no Nuvio */
81
+ projectId: string;
82
+ /** API key gerada no Nuvio */
83
+ apiKey: string;
84
+ /** URL do endpoint de ingest */
85
+ endpoint: string;
86
+ /** Intervalo do heartbeat em ms (default: 60000) */
87
+ heartbeatInterval?: number;
88
+ /** Intervalo do flush do buffer em ms (default: 30000) */
89
+ flushInterval?: number;
90
+ /** Timeout do fetch em ms (default: 10000) */
91
+ timeout?: number;
92
+ /** Max tentativas de retry (default: 3) */
93
+ maxRetries?: number;
94
+ /** Desabilitar em desenvolvimento (default: true) */
95
+ disableInDev?: boolean;
96
+ /** Taxa de amostragem 0.0-1.0 (default: 1.0). Heartbeats nunca são amostrados */
97
+ sampleRate?: number;
98
+ /** Hook chamado antes de enfileirar cada evento. Retorna null para dropar */
99
+ beforeSend?: BeforeSendHook;
100
+ /** Habilitar logs de debug no console (default: false) */
101
+ debug?: boolean;
102
+ /** Capturar erros globais automaticamente — window.onerror (default: false) */
103
+ captureErrors?: boolean;
104
+ /** Capturar unhandled promise rejections automaticamente (default: false) */
105
+ captureUnhandledRejections?: boolean;
106
+ }
107
+ declare class MonitorClient {
108
+ readonly config: MonitorConfig;
109
+ private readonly collector;
110
+ private readonly logger;
111
+ private heartbeatTimer;
112
+ private globalHandlersCleanup;
113
+ private active;
114
+ constructor(config: MonitorConfig);
115
+ /** Inicia o monitoring (heartbeat + collector) */
116
+ start(): void;
117
+ /** Para o monitoring */
118
+ stop(): void;
119
+ /** Força um flush imediato do buffer */
120
+ flush(): void;
121
+ /** Verifica se o monitoring está ativo */
122
+ get isActive(): boolean;
123
+ /** Registra um evento de request HTTP (usado pelo middleware) */
124
+ trackRequest(data: RequestData): void;
125
+ /** Registra um Web Vital (usado pelo MonitorScript) */
126
+ trackVital(data: VitalData): void;
127
+ /** Registra um evento de adapter (DB, cache, AI, queue, email) */
128
+ trackAdapter(data: AdapterData): void;
129
+ /** Registra um erro capturado */
130
+ captureError(data: ErrorData): void;
131
+ /** Registra um evento de deploy */
132
+ trackDeploy(data: DeployData): void;
133
+ private startHeartbeat;
134
+ private sendHeartbeat;
135
+ }
136
+
137
+ /**
138
+ * Instrumenta uma chamada ao Gemini API com timing e token tracking.
139
+ *
140
+ * @example
141
+ * ```ts
142
+ * import { monitorAI } from '@victor-studio/monitor/adapters/gemini'
143
+ *
144
+ * const result = await monitorAI(monitor, 'generate', () =>
145
+ * model.generateContent(prompt),
146
+ * { model: 'gemini-2.5-flash' },
147
+ * )
148
+ * ```
149
+ */
150
+ declare function monitorAI<T>(monitor: MonitorClient, operation: string, fn: () => Promise<T>, meta?: Record<string, string>): Promise<T>;
151
+ /**
152
+ * Instrumenta uma chamada AI e extrai token usage do resultado.
153
+ * Específico para o formato de resposta do Google Generative AI SDK.
154
+ *
155
+ * @example
156
+ * ```ts
157
+ * const result = await monitorAIWithTokens(monitor, 'generate', async () => {
158
+ * const res = await model.generateContent(prompt)
159
+ * return {
160
+ * result: res,
161
+ * inputTokens: res.response.usageMetadata?.promptTokenCount,
162
+ * outputTokens: res.response.usageMetadata?.candidatesTokenCount,
163
+ * }
164
+ * }, { model: 'gemini-2.5-flash' })
165
+ * ```
166
+ */
167
+ declare function monitorAIWithTokens<T>(monitor: MonitorClient, operation: string, fn: () => Promise<{
168
+ result: T;
169
+ inputTokens?: number;
170
+ outputTokens?: number;
171
+ }>, meta?: Record<string, string>): Promise<T>;
172
+
173
+ export { monitorAI, monitorAIWithTokens };
@@ -0,0 +1,173 @@
1
+ interface HeartbeatData {
2
+ status: 'online' | 'offline';
3
+ latencyMs: number;
4
+ }
5
+ interface RequestData {
6
+ method: string;
7
+ route: string;
8
+ statusCode: number;
9
+ responseTimeMs: number;
10
+ userAgent?: string;
11
+ region?: string;
12
+ }
13
+ interface VitalData {
14
+ name: 'LCP' | 'INP' | 'CLS' | 'FCP' | 'TTFB';
15
+ value: number;
16
+ rating: 'good' | 'needs-improvement' | 'poor';
17
+ page?: string;
18
+ deviceType?: string;
19
+ browser?: string;
20
+ }
21
+ interface ErrorData {
22
+ type: 'unhandled' | 'caught' | 'promise';
23
+ message: string;
24
+ stack?: string;
25
+ groupingKey: string;
26
+ route?: string;
27
+ method?: string;
28
+ statusCode?: number;
29
+ environment?: string;
30
+ commitHash?: string;
31
+ extra?: Record<string, string>;
32
+ }
33
+ interface DeployData {
34
+ commitHash: string;
35
+ branch?: string;
36
+ author?: string;
37
+ status: 'started' | 'succeeded' | 'failed';
38
+ buildDurationMs?: number;
39
+ url?: string;
40
+ environment?: string;
41
+ provider?: string;
42
+ }
43
+ interface AdapterData {
44
+ adapter: string;
45
+ operation: string;
46
+ durationMs: number;
47
+ success: boolean;
48
+ error?: string;
49
+ meta?: Record<string, string>;
50
+ }
51
+ type MonitorEvent = {
52
+ type: 'heartbeat';
53
+ data: HeartbeatData;
54
+ timestamp: string;
55
+ } | {
56
+ type: 'request';
57
+ data: RequestData;
58
+ timestamp: string;
59
+ } | {
60
+ type: 'vital';
61
+ data: VitalData;
62
+ timestamp: string;
63
+ } | {
64
+ type: 'error';
65
+ data: ErrorData;
66
+ timestamp: string;
67
+ } | {
68
+ type: 'deploy';
69
+ data: DeployData;
70
+ timestamp: string;
71
+ } | {
72
+ type: 'adapter';
73
+ data: AdapterData;
74
+ timestamp: string;
75
+ };
76
+ /** Hook para filtrar/modificar eventos antes do envio */
77
+ type BeforeSendHook = (event: MonitorEvent) => MonitorEvent | null;
78
+
79
+ interface MonitorConfig {
80
+ /** ID do projeto no Nuvio */
81
+ projectId: string;
82
+ /** API key gerada no Nuvio */
83
+ apiKey: string;
84
+ /** URL do endpoint de ingest */
85
+ endpoint: string;
86
+ /** Intervalo do heartbeat em ms (default: 60000) */
87
+ heartbeatInterval?: number;
88
+ /** Intervalo do flush do buffer em ms (default: 30000) */
89
+ flushInterval?: number;
90
+ /** Timeout do fetch em ms (default: 10000) */
91
+ timeout?: number;
92
+ /** Max tentativas de retry (default: 3) */
93
+ maxRetries?: number;
94
+ /** Desabilitar em desenvolvimento (default: true) */
95
+ disableInDev?: boolean;
96
+ /** Taxa de amostragem 0.0-1.0 (default: 1.0). Heartbeats nunca são amostrados */
97
+ sampleRate?: number;
98
+ /** Hook chamado antes de enfileirar cada evento. Retorna null para dropar */
99
+ beforeSend?: BeforeSendHook;
100
+ /** Habilitar logs de debug no console (default: false) */
101
+ debug?: boolean;
102
+ /** Capturar erros globais automaticamente — window.onerror (default: false) */
103
+ captureErrors?: boolean;
104
+ /** Capturar unhandled promise rejections automaticamente (default: false) */
105
+ captureUnhandledRejections?: boolean;
106
+ }
107
+ declare class MonitorClient {
108
+ readonly config: MonitorConfig;
109
+ private readonly collector;
110
+ private readonly logger;
111
+ private heartbeatTimer;
112
+ private globalHandlersCleanup;
113
+ private active;
114
+ constructor(config: MonitorConfig);
115
+ /** Inicia o monitoring (heartbeat + collector) */
116
+ start(): void;
117
+ /** Para o monitoring */
118
+ stop(): void;
119
+ /** Força um flush imediato do buffer */
120
+ flush(): void;
121
+ /** Verifica se o monitoring está ativo */
122
+ get isActive(): boolean;
123
+ /** Registra um evento de request HTTP (usado pelo middleware) */
124
+ trackRequest(data: RequestData): void;
125
+ /** Registra um Web Vital (usado pelo MonitorScript) */
126
+ trackVital(data: VitalData): void;
127
+ /** Registra um evento de adapter (DB, cache, AI, queue, email) */
128
+ trackAdapter(data: AdapterData): void;
129
+ /** Registra um erro capturado */
130
+ captureError(data: ErrorData): void;
131
+ /** Registra um evento de deploy */
132
+ trackDeploy(data: DeployData): void;
133
+ private startHeartbeat;
134
+ private sendHeartbeat;
135
+ }
136
+
137
+ /**
138
+ * Instrumenta uma chamada ao Gemini API com timing e token tracking.
139
+ *
140
+ * @example
141
+ * ```ts
142
+ * import { monitorAI } from '@victor-studio/monitor/adapters/gemini'
143
+ *
144
+ * const result = await monitorAI(monitor, 'generate', () =>
145
+ * model.generateContent(prompt),
146
+ * { model: 'gemini-2.5-flash' },
147
+ * )
148
+ * ```
149
+ */
150
+ declare function monitorAI<T>(monitor: MonitorClient, operation: string, fn: () => Promise<T>, meta?: Record<string, string>): Promise<T>;
151
+ /**
152
+ * Instrumenta uma chamada AI e extrai token usage do resultado.
153
+ * Específico para o formato de resposta do Google Generative AI SDK.
154
+ *
155
+ * @example
156
+ * ```ts
157
+ * const result = await monitorAIWithTokens(monitor, 'generate', async () => {
158
+ * const res = await model.generateContent(prompt)
159
+ * return {
160
+ * result: res,
161
+ * inputTokens: res.response.usageMetadata?.promptTokenCount,
162
+ * outputTokens: res.response.usageMetadata?.candidatesTokenCount,
163
+ * }
164
+ * }, { model: 'gemini-2.5-flash' })
165
+ * ```
166
+ */
167
+ declare function monitorAIWithTokens<T>(monitor: MonitorClient, operation: string, fn: () => Promise<{
168
+ result: T;
169
+ inputTokens?: number;
170
+ outputTokens?: number;
171
+ }>, meta?: Record<string, string>): Promise<T>;
172
+
173
+ export { monitorAI, monitorAIWithTokens };
@@ -0,0 +1,62 @@
1
+ // src/adapters/core.ts
2
+ async function withTiming(monitor, adapter, operation, fn, meta) {
3
+ const start = performance.now();
4
+ try {
5
+ const result = await fn();
6
+ monitor.trackAdapter({
7
+ adapter,
8
+ operation,
9
+ durationMs: Math.round(performance.now() - start),
10
+ success: true,
11
+ meta
12
+ });
13
+ return result;
14
+ } catch (error) {
15
+ monitor.trackAdapter({
16
+ adapter,
17
+ operation,
18
+ durationMs: Math.round(performance.now() - start),
19
+ success: false,
20
+ error: error instanceof Error ? error.message : String(error),
21
+ meta
22
+ });
23
+ throw error;
24
+ }
25
+ }
26
+
27
+ // src/adapters/gemini.ts
28
+ function monitorAI(monitor, operation, fn, meta) {
29
+ return withTiming(monitor, "gemini", operation, fn, meta);
30
+ }
31
+ async function monitorAIWithTokens(monitor, operation, fn, meta) {
32
+ const start = performance.now();
33
+ try {
34
+ const { result, inputTokens, outputTokens } = await fn();
35
+ monitor.trackAdapter({
36
+ adapter: "gemini",
37
+ operation,
38
+ durationMs: Math.round(performance.now() - start),
39
+ success: true,
40
+ meta: {
41
+ ...meta,
42
+ ...inputTokens !== void 0 ? { inputTokens: String(inputTokens) } : {},
43
+ ...outputTokens !== void 0 ? { outputTokens: String(outputTokens) } : {}
44
+ }
45
+ });
46
+ return result;
47
+ } catch (error) {
48
+ monitor.trackAdapter({
49
+ adapter: "gemini",
50
+ operation,
51
+ durationMs: Math.round(performance.now() - start),
52
+ success: false,
53
+ error: error instanceof Error ? error.message : String(error),
54
+ meta
55
+ });
56
+ throw error;
57
+ }
58
+ }
59
+
60
+ export { monitorAI, monitorAIWithTokens };
61
+ //# sourceMappingURL=gemini.js.map
62
+ //# sourceMappingURL=gemini.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../../src/adapters/core.ts","../../../src/adapters/gemini.ts"],"names":[],"mappings":";AAgBA,eAAsB,UAAA,CACpB,OAAA,EACA,OAAA,EACA,SAAA,EACA,IACA,IAAA,EACY;AACZ,EAAA,MAAM,KAAA,GAAQ,YAAY,GAAA,EAAI;AAE9B,EAAA,IAAI;AACF,IAAA,MAAM,MAAA,GAAS,MAAM,EAAA,EAAG;AAExB,IAAA,OAAA,CAAQ,YAAA,CAAa;AAAA,MACnB,OAAA;AAAA,MACA,SAAA;AAAA,MACA,YAAY,IAAA,CAAK,KAAA,CAAM,WAAA,CAAY,GAAA,KAAQ,KAAK,CAAA;AAAA,MAChD,OAAA,EAAS,IAAA;AAAA,MACT;AAAA,KACqB,CAAA;AAEvB,IAAA,OAAO,MAAA;AAAA,EACT,SAAS,KAAA,EAAO;AACd,IAAA,OAAA,CAAQ,YAAA,CAAa;AAAA,MACnB,OAAA;AAAA,MACA,SAAA;AAAA,MACA,YAAY,IAAA,CAAK,KAAA,CAAM,WAAA,CAAY,GAAA,KAAQ,KAAK,CAAA;AAAA,MAChD,OAAA,EAAS,KAAA;AAAA,MACT,OAAO,KAAA,YAAiB,KAAA,GAAQ,KAAA,CAAM,OAAA,GAAU,OAAO,KAAK,CAAA;AAAA,MAC5D;AAAA,KACqB,CAAA;AAEvB,IAAA,MAAM,KAAA;AAAA,EACR;AACF;;;AC/BO,SAAS,SAAA,CACd,OAAA,EACA,SAAA,EACA,EAAA,EACA,IAAA,EACY;AACZ,EAAA,OAAO,UAAA,CAAW,OAAA,EAAS,QAAA,EAAU,SAAA,EAAW,IAAI,IAAI,CAAA;AAC1D;AAkBA,eAAsB,mBAAA,CACpB,OAAA,EACA,SAAA,EACA,EAAA,EACA,IAAA,EACY;AACZ,EAAA,MAAM,KAAA,GAAQ,YAAY,GAAA,EAAI;AAE9B,EAAA,IAAI;AACF,IAAA,MAAM,EAAE,MAAA,EAAQ,WAAA,EAAa,YAAA,EAAa,GAAI,MAAM,EAAA,EAAG;AAEvD,IAAA,OAAA,CAAQ,YAAA,CAAa;AAAA,MACnB,OAAA,EAAS,QAAA;AAAA,MACT,SAAA;AAAA,MACA,YAAY,IAAA,CAAK,KAAA,CAAM,WAAA,CAAY,GAAA,KAAQ,KAAK,CAAA;AAAA,MAChD,OAAA,EAAS,IAAA;AAAA,MACT,IAAA,EAAM;AAAA,QACJ,GAAG,IAAA;AAAA,QACH,GAAI,gBAAgB,KAAA,CAAA,GAAY,EAAE,aAAa,MAAA,CAAO,WAAW,CAAA,EAAE,GAAI,EAAC;AAAA,QACxE,GAAI,iBAAiB,KAAA,CAAA,GAAY,EAAE,cAAc,MAAA,CAAO,YAAY,CAAA,EAAE,GAAI;AAAC;AAC7E,KACD,CAAA;AAED,IAAA,OAAO,MAAA;AAAA,EACT,SAAS,KAAA,EAAO;AACd,IAAA,OAAA,CAAQ,YAAA,CAAa;AAAA,MACnB,OAAA,EAAS,QAAA;AAAA,MACT,SAAA;AAAA,MACA,YAAY,IAAA,CAAK,KAAA,CAAM,WAAA,CAAY,GAAA,KAAQ,KAAK,CAAA;AAAA,MAChD,OAAA,EAAS,KAAA;AAAA,MACT,OAAO,KAAA,YAAiB,KAAA,GAAQ,KAAA,CAAM,OAAA,GAAU,OAAO,KAAK,CAAA;AAAA,MAC5D;AAAA,KACD,CAAA;AAED,IAAA,MAAM,KAAA;AAAA,EACR;AACF","file":"gemini.js","sourcesContent":["// Adapter core — utilitário genérico de instrumentação\n\nimport type { MonitorClient } from '../core/client'\nimport type { AdapterData } from '../types'\n\n/**\n * Mede a duração de uma operação assíncrona e reporta ao monitor.\n * Utilitário base usado por todos os adapters.\n *\n * @example\n * ```ts\n * const result = await withTiming(monitor, 'neon', 'query', () =>\n * db.select().from(users),\n * )\n * ```\n */\nexport async function withTiming<T>(\n monitor: MonitorClient,\n adapter: string,\n operation: string,\n fn: () => Promise<T>,\n meta?: Record<string, string>,\n): Promise<T> {\n const start = performance.now()\n\n try {\n const result = await fn()\n\n monitor.trackAdapter({\n adapter,\n operation,\n durationMs: Math.round(performance.now() - start),\n success: true,\n meta,\n } satisfies AdapterData)\n\n return result\n } catch (error) {\n monitor.trackAdapter({\n adapter,\n operation,\n durationMs: Math.round(performance.now() - start),\n success: false,\n error: error instanceof Error ? error.message : String(error),\n meta,\n } satisfies AdapterData)\n\n throw error // nunca engolir erros\n }\n}\n\n/**\n * Versão sync do withTiming para operações síncronas.\n */\nexport function withTimingSync<T>(\n monitor: MonitorClient,\n adapter: string,\n operation: string,\n fn: () => T,\n meta?: Record<string, string>,\n): T {\n const start = performance.now()\n\n try {\n const result = fn()\n\n monitor.trackAdapter({\n adapter,\n operation,\n durationMs: Math.round(performance.now() - start),\n success: true,\n meta,\n } satisfies AdapterData)\n\n return result\n } catch (error) {\n monitor.trackAdapter({\n adapter,\n operation,\n durationMs: Math.round(performance.now() - start),\n success: false,\n error: error instanceof Error ? error.message : String(error),\n meta,\n } satisfies AdapterData)\n\n throw error\n }\n}\n","// Adapter Gemini — instrumentação de chamadas AI\n\nimport type { MonitorClient } from '../core/client'\nimport { withTiming } from './core'\n\n/**\n * Instrumenta uma chamada ao Gemini API com timing e token tracking.\n *\n * @example\n * ```ts\n * import { monitorAI } from '@victor-studio/monitor/adapters/gemini'\n *\n * const result = await monitorAI(monitor, 'generate', () =>\n * model.generateContent(prompt),\n * { model: 'gemini-2.5-flash' },\n * )\n * ```\n */\nexport function monitorAI<T>(\n monitor: MonitorClient,\n operation: string,\n fn: () => Promise<T>,\n meta?: Record<string, string>,\n): Promise<T> {\n return withTiming(monitor, 'gemini', operation, fn, meta)\n}\n\n/**\n * Instrumenta uma chamada AI e extrai token usage do resultado.\n * Específico para o formato de resposta do Google Generative AI SDK.\n *\n * @example\n * ```ts\n * const result = await monitorAIWithTokens(monitor, 'generate', async () => {\n * const res = await model.generateContent(prompt)\n * return {\n * result: res,\n * inputTokens: res.response.usageMetadata?.promptTokenCount,\n * outputTokens: res.response.usageMetadata?.candidatesTokenCount,\n * }\n * }, { model: 'gemini-2.5-flash' })\n * ```\n */\nexport async function monitorAIWithTokens<T>(\n monitor: MonitorClient,\n operation: string,\n fn: () => Promise<{ result: T; inputTokens?: number; outputTokens?: number }>,\n meta?: Record<string, string>,\n): Promise<T> {\n const start = performance.now()\n\n try {\n const { result, inputTokens, outputTokens } = await fn()\n\n monitor.trackAdapter({\n adapter: 'gemini',\n operation,\n durationMs: Math.round(performance.now() - start),\n success: true,\n meta: {\n ...meta,\n ...(inputTokens !== undefined ? { inputTokens: String(inputTokens) } : {}),\n ...(outputTokens !== undefined ? { outputTokens: String(outputTokens) } : {}),\n },\n })\n\n return result\n } catch (error) {\n monitor.trackAdapter({\n adapter: 'gemini',\n operation,\n durationMs: Math.round(performance.now() - start),\n success: false,\n error: error instanceof Error ? error.message : String(error),\n meta,\n })\n\n throw error\n }\n}\n"]}
@@ -0,0 +1,56 @@
1
+ 'use strict';
2
+
3
+ // src/adapters/core.ts
4
+ async function withTiming(monitor, adapter, operation, fn, meta) {
5
+ const start = performance.now();
6
+ try {
7
+ const result = await fn();
8
+ monitor.trackAdapter({
9
+ adapter,
10
+ operation,
11
+ durationMs: Math.round(performance.now() - start),
12
+ success: true,
13
+ meta
14
+ });
15
+ return result;
16
+ } catch (error) {
17
+ monitor.trackAdapter({
18
+ adapter,
19
+ operation,
20
+ durationMs: Math.round(performance.now() - start),
21
+ success: false,
22
+ error: error instanceof Error ? error.message : String(error),
23
+ meta
24
+ });
25
+ throw error;
26
+ }
27
+ }
28
+ function withTimingSync(monitor, adapter, operation, fn, meta) {
29
+ const start = performance.now();
30
+ try {
31
+ const result = fn();
32
+ monitor.trackAdapter({
33
+ adapter,
34
+ operation,
35
+ durationMs: Math.round(performance.now() - start),
36
+ success: true,
37
+ meta
38
+ });
39
+ return result;
40
+ } catch (error) {
41
+ monitor.trackAdapter({
42
+ adapter,
43
+ operation,
44
+ durationMs: Math.round(performance.now() - start),
45
+ success: false,
46
+ error: error instanceof Error ? error.message : String(error),
47
+ meta
48
+ });
49
+ throw error;
50
+ }
51
+ }
52
+
53
+ exports.withTiming = withTiming;
54
+ exports.withTimingSync = withTimingSync;
55
+ //# sourceMappingURL=index.cjs.map
56
+ //# sourceMappingURL=index.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../src/adapters/core.ts"],"names":[],"mappings":";;;AAgBA,eAAsB,UAAA,CACpB,OAAA,EACA,OAAA,EACA,SAAA,EACA,IACA,IAAA,EACY;AACZ,EAAA,MAAM,KAAA,GAAQ,YAAY,GAAA,EAAI;AAE9B,EAAA,IAAI;AACF,IAAA,MAAM,MAAA,GAAS,MAAM,EAAA,EAAG;AAExB,IAAA,OAAA,CAAQ,YAAA,CAAa;AAAA,MACnB,OAAA;AAAA,MACA,SAAA;AAAA,MACA,YAAY,IAAA,CAAK,KAAA,CAAM,WAAA,CAAY,GAAA,KAAQ,KAAK,CAAA;AAAA,MAChD,OAAA,EAAS,IAAA;AAAA,MACT;AAAA,KACqB,CAAA;AAEvB,IAAA,OAAO,MAAA;AAAA,EACT,SAAS,KAAA,EAAO;AACd,IAAA,OAAA,CAAQ,YAAA,CAAa;AAAA,MACnB,OAAA;AAAA,MACA,SAAA;AAAA,MACA,YAAY,IAAA,CAAK,KAAA,CAAM,WAAA,CAAY,GAAA,KAAQ,KAAK,CAAA;AAAA,MAChD,OAAA,EAAS,KAAA;AAAA,MACT,OAAO,KAAA,YAAiB,KAAA,GAAQ,KAAA,CAAM,OAAA,GAAU,OAAO,KAAK,CAAA;AAAA,MAC5D;AAAA,KACqB,CAAA;AAEvB,IAAA,MAAM,KAAA;AAAA,EACR;AACF;AAKO,SAAS,cAAA,CACd,OAAA,EACA,OAAA,EACA,SAAA,EACA,IACA,IAAA,EACG;AACH,EAAA,MAAM,KAAA,GAAQ,YAAY,GAAA,EAAI;AAE9B,EAAA,IAAI;AACF,IAAA,MAAM,SAAS,EAAA,EAAG;AAElB,IAAA,OAAA,CAAQ,YAAA,CAAa;AAAA,MACnB,OAAA;AAAA,MACA,SAAA;AAAA,MACA,YAAY,IAAA,CAAK,KAAA,CAAM,WAAA,CAAY,GAAA,KAAQ,KAAK,CAAA;AAAA,MAChD,OAAA,EAAS,IAAA;AAAA,MACT;AAAA,KACqB,CAAA;AAEvB,IAAA,OAAO,MAAA;AAAA,EACT,SAAS,KAAA,EAAO;AACd,IAAA,OAAA,CAAQ,YAAA,CAAa;AAAA,MACnB,OAAA;AAAA,MACA,SAAA;AAAA,MACA,YAAY,IAAA,CAAK,KAAA,CAAM,WAAA,CAAY,GAAA,KAAQ,KAAK,CAAA;AAAA,MAChD,OAAA,EAAS,KAAA;AAAA,MACT,OAAO,KAAA,YAAiB,KAAA,GAAQ,KAAA,CAAM,OAAA,GAAU,OAAO,KAAK,CAAA;AAAA,MAC5D;AAAA,KACqB,CAAA;AAEvB,IAAA,MAAM,KAAA;AAAA,EACR;AACF","file":"index.cjs","sourcesContent":["// Adapter core — utilitário genérico de instrumentação\n\nimport type { MonitorClient } from '../core/client'\nimport type { AdapterData } from '../types'\n\n/**\n * Mede a duração de uma operação assíncrona e reporta ao monitor.\n * Utilitário base usado por todos os adapters.\n *\n * @example\n * ```ts\n * const result = await withTiming(monitor, 'neon', 'query', () =>\n * db.select().from(users),\n * )\n * ```\n */\nexport async function withTiming<T>(\n monitor: MonitorClient,\n adapter: string,\n operation: string,\n fn: () => Promise<T>,\n meta?: Record<string, string>,\n): Promise<T> {\n const start = performance.now()\n\n try {\n const result = await fn()\n\n monitor.trackAdapter({\n adapter,\n operation,\n durationMs: Math.round(performance.now() - start),\n success: true,\n meta,\n } satisfies AdapterData)\n\n return result\n } catch (error) {\n monitor.trackAdapter({\n adapter,\n operation,\n durationMs: Math.round(performance.now() - start),\n success: false,\n error: error instanceof Error ? error.message : String(error),\n meta,\n } satisfies AdapterData)\n\n throw error // nunca engolir erros\n }\n}\n\n/**\n * Versão sync do withTiming para operações síncronas.\n */\nexport function withTimingSync<T>(\n monitor: MonitorClient,\n adapter: string,\n operation: string,\n fn: () => T,\n meta?: Record<string, string>,\n): T {\n const start = performance.now()\n\n try {\n const result = fn()\n\n monitor.trackAdapter({\n adapter,\n operation,\n durationMs: Math.round(performance.now() - start),\n success: true,\n meta,\n } satisfies AdapterData)\n\n return result\n } catch (error) {\n monitor.trackAdapter({\n adapter,\n operation,\n durationMs: Math.round(performance.now() - start),\n success: false,\n error: error instanceof Error ? error.message : String(error),\n meta,\n } satisfies AdapterData)\n\n throw error\n }\n}\n"]}
@@ -0,0 +1,154 @@
1
+ interface HeartbeatData {
2
+ status: 'online' | 'offline';
3
+ latencyMs: number;
4
+ }
5
+ interface RequestData {
6
+ method: string;
7
+ route: string;
8
+ statusCode: number;
9
+ responseTimeMs: number;
10
+ userAgent?: string;
11
+ region?: string;
12
+ }
13
+ interface VitalData {
14
+ name: 'LCP' | 'INP' | 'CLS' | 'FCP' | 'TTFB';
15
+ value: number;
16
+ rating: 'good' | 'needs-improvement' | 'poor';
17
+ page?: string;
18
+ deviceType?: string;
19
+ browser?: string;
20
+ }
21
+ interface ErrorData {
22
+ type: 'unhandled' | 'caught' | 'promise';
23
+ message: string;
24
+ stack?: string;
25
+ groupingKey: string;
26
+ route?: string;
27
+ method?: string;
28
+ statusCode?: number;
29
+ environment?: string;
30
+ commitHash?: string;
31
+ extra?: Record<string, string>;
32
+ }
33
+ interface DeployData {
34
+ commitHash: string;
35
+ branch?: string;
36
+ author?: string;
37
+ status: 'started' | 'succeeded' | 'failed';
38
+ buildDurationMs?: number;
39
+ url?: string;
40
+ environment?: string;
41
+ provider?: string;
42
+ }
43
+ interface AdapterData {
44
+ adapter: string;
45
+ operation: string;
46
+ durationMs: number;
47
+ success: boolean;
48
+ error?: string;
49
+ meta?: Record<string, string>;
50
+ }
51
+ type MonitorEvent = {
52
+ type: 'heartbeat';
53
+ data: HeartbeatData;
54
+ timestamp: string;
55
+ } | {
56
+ type: 'request';
57
+ data: RequestData;
58
+ timestamp: string;
59
+ } | {
60
+ type: 'vital';
61
+ data: VitalData;
62
+ timestamp: string;
63
+ } | {
64
+ type: 'error';
65
+ data: ErrorData;
66
+ timestamp: string;
67
+ } | {
68
+ type: 'deploy';
69
+ data: DeployData;
70
+ timestamp: string;
71
+ } | {
72
+ type: 'adapter';
73
+ data: AdapterData;
74
+ timestamp: string;
75
+ };
76
+ /** Hook para filtrar/modificar eventos antes do envio */
77
+ type BeforeSendHook = (event: MonitorEvent) => MonitorEvent | null;
78
+
79
+ interface MonitorConfig {
80
+ /** ID do projeto no Nuvio */
81
+ projectId: string;
82
+ /** API key gerada no Nuvio */
83
+ apiKey: string;
84
+ /** URL do endpoint de ingest */
85
+ endpoint: string;
86
+ /** Intervalo do heartbeat em ms (default: 60000) */
87
+ heartbeatInterval?: number;
88
+ /** Intervalo do flush do buffer em ms (default: 30000) */
89
+ flushInterval?: number;
90
+ /** Timeout do fetch em ms (default: 10000) */
91
+ timeout?: number;
92
+ /** Max tentativas de retry (default: 3) */
93
+ maxRetries?: number;
94
+ /** Desabilitar em desenvolvimento (default: true) */
95
+ disableInDev?: boolean;
96
+ /** Taxa de amostragem 0.0-1.0 (default: 1.0). Heartbeats nunca são amostrados */
97
+ sampleRate?: number;
98
+ /** Hook chamado antes de enfileirar cada evento. Retorna null para dropar */
99
+ beforeSend?: BeforeSendHook;
100
+ /** Habilitar logs de debug no console (default: false) */
101
+ debug?: boolean;
102
+ /** Capturar erros globais automaticamente — window.onerror (default: false) */
103
+ captureErrors?: boolean;
104
+ /** Capturar unhandled promise rejections automaticamente (default: false) */
105
+ captureUnhandledRejections?: boolean;
106
+ }
107
+ declare class MonitorClient {
108
+ readonly config: MonitorConfig;
109
+ private readonly collector;
110
+ private readonly logger;
111
+ private heartbeatTimer;
112
+ private globalHandlersCleanup;
113
+ private active;
114
+ constructor(config: MonitorConfig);
115
+ /** Inicia o monitoring (heartbeat + collector) */
116
+ start(): void;
117
+ /** Para o monitoring */
118
+ stop(): void;
119
+ /** Força um flush imediato do buffer */
120
+ flush(): void;
121
+ /** Verifica se o monitoring está ativo */
122
+ get isActive(): boolean;
123
+ /** Registra um evento de request HTTP (usado pelo middleware) */
124
+ trackRequest(data: RequestData): void;
125
+ /** Registra um Web Vital (usado pelo MonitorScript) */
126
+ trackVital(data: VitalData): void;
127
+ /** Registra um evento de adapter (DB, cache, AI, queue, email) */
128
+ trackAdapter(data: AdapterData): void;
129
+ /** Registra um erro capturado */
130
+ captureError(data: ErrorData): void;
131
+ /** Registra um evento de deploy */
132
+ trackDeploy(data: DeployData): void;
133
+ private startHeartbeat;
134
+ private sendHeartbeat;
135
+ }
136
+
137
+ /**
138
+ * Mede a duração de uma operação assíncrona e reporta ao monitor.
139
+ * Utilitário base usado por todos os adapters.
140
+ *
141
+ * @example
142
+ * ```ts
143
+ * const result = await withTiming(monitor, 'neon', 'query', () =>
144
+ * db.select().from(users),
145
+ * )
146
+ * ```
147
+ */
148
+ declare function withTiming<T>(monitor: MonitorClient, adapter: string, operation: string, fn: () => Promise<T>, meta?: Record<string, string>): Promise<T>;
149
+ /**
150
+ * Versão sync do withTiming para operações síncronas.
151
+ */
152
+ declare function withTimingSync<T>(monitor: MonitorClient, adapter: string, operation: string, fn: () => T, meta?: Record<string, string>): T;
153
+
154
+ export { withTiming, withTimingSync };