flow-debugger 1.0.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.
- package/PORTFOLIO_README_SECTION.md +177 -0
- package/README.md +251 -0
- package/dashboard/app.js +339 -0
- package/dashboard/index.html +168 -0
- package/dashboard/style.css +846 -0
- package/dist/cjs/core/Analytics.js +174 -0
- package/dist/cjs/core/Analytics.js.map +1 -0
- package/dist/cjs/core/Classifier.js +66 -0
- package/dist/cjs/core/Classifier.js.map +1 -0
- package/dist/cjs/core/HealthMonitor.js +79 -0
- package/dist/cjs/core/HealthMonitor.js.map +1 -0
- package/dist/cjs/core/RootCause.js +89 -0
- package/dist/cjs/core/RootCause.js.map +1 -0
- package/dist/cjs/core/Sampler.js +34 -0
- package/dist/cjs/core/Sampler.js.map +1 -0
- package/dist/cjs/core/Timeline.js +90 -0
- package/dist/cjs/core/Timeline.js.map +1 -0
- package/dist/cjs/core/TraceEngine.js +222 -0
- package/dist/cjs/core/TraceEngine.js.map +1 -0
- package/dist/cjs/core/types.js +21 -0
- package/dist/cjs/core/types.js.map +1 -0
- package/dist/cjs/index.js +46 -0
- package/dist/cjs/index.js.map +1 -0
- package/dist/cjs/integrations/axios.js +136 -0
- package/dist/cjs/integrations/axios.js.map +1 -0
- package/dist/cjs/integrations/fetch.js +153 -0
- package/dist/cjs/integrations/fetch.js.map +1 -0
- package/dist/cjs/integrations/mongo.js +111 -0
- package/dist/cjs/integrations/mongo.js.map +1 -0
- package/dist/cjs/integrations/mysql.js +212 -0
- package/dist/cjs/integrations/mysql.js.map +1 -0
- package/dist/cjs/integrations/postgres.js +182 -0
- package/dist/cjs/integrations/postgres.js.map +1 -0
- package/dist/cjs/integrations/redis.js +105 -0
- package/dist/cjs/integrations/redis.js.map +1 -0
- package/dist/cjs/middleware/express.js +255 -0
- package/dist/cjs/middleware/express.js.map +1 -0
- package/dist/esm/core/Analytics.js +170 -0
- package/dist/esm/core/Analytics.js.map +1 -0
- package/dist/esm/core/Classifier.js +61 -0
- package/dist/esm/core/Classifier.js.map +1 -0
- package/dist/esm/core/HealthMonitor.js +75 -0
- package/dist/esm/core/HealthMonitor.js.map +1 -0
- package/dist/esm/core/RootCause.js +86 -0
- package/dist/esm/core/RootCause.js.map +1 -0
- package/dist/esm/core/Sampler.js +30 -0
- package/dist/esm/core/Sampler.js.map +1 -0
- package/dist/esm/core/Timeline.js +86 -0
- package/dist/esm/core/Timeline.js.map +1 -0
- package/dist/esm/core/TraceEngine.js +217 -0
- package/dist/esm/core/TraceEngine.js.map +1 -0
- package/dist/esm/core/types.js +18 -0
- package/dist/esm/core/types.js.map +1 -0
- package/dist/esm/index.js +22 -0
- package/dist/esm/index.js.map +1 -0
- package/dist/esm/integrations/axios.js +133 -0
- package/dist/esm/integrations/axios.js.map +1 -0
- package/dist/esm/integrations/fetch.js +149 -0
- package/dist/esm/integrations/fetch.js.map +1 -0
- package/dist/esm/integrations/mongo.js +107 -0
- package/dist/esm/integrations/mongo.js.map +1 -0
- package/dist/esm/integrations/mysql.js +209 -0
- package/dist/esm/integrations/mysql.js.map +1 -0
- package/dist/esm/integrations/postgres.js +179 -0
- package/dist/esm/integrations/postgres.js.map +1 -0
- package/dist/esm/integrations/redis.js +102 -0
- package/dist/esm/integrations/redis.js.map +1 -0
- package/dist/esm/middleware/express.js +219 -0
- package/dist/esm/middleware/express.js.map +1 -0
- package/dist/types/core/Analytics.d.ts +35 -0
- package/dist/types/core/Analytics.d.ts.map +1 -0
- package/dist/types/core/Classifier.d.ts +21 -0
- package/dist/types/core/Classifier.d.ts.map +1 -0
- package/dist/types/core/HealthMonitor.d.ts +14 -0
- package/dist/types/core/HealthMonitor.d.ts.map +1 -0
- package/dist/types/core/RootCause.d.ts +12 -0
- package/dist/types/core/RootCause.d.ts.map +1 -0
- package/dist/types/core/Sampler.d.ts +13 -0
- package/dist/types/core/Sampler.d.ts.map +1 -0
- package/dist/types/core/Timeline.d.ts +22 -0
- package/dist/types/core/Timeline.d.ts.map +1 -0
- package/dist/types/core/TraceEngine.d.ts +47 -0
- package/dist/types/core/TraceEngine.d.ts.map +1 -0
- package/dist/types/core/types.d.ts +118 -0
- package/dist/types/core/types.d.ts.map +1 -0
- package/dist/types/index.d.ts +18 -0
- package/dist/types/index.d.ts.map +1 -0
- package/dist/types/integrations/axios.d.ts +22 -0
- package/dist/types/integrations/axios.d.ts.map +1 -0
- package/dist/types/integrations/fetch.d.ts +25 -0
- package/dist/types/integrations/fetch.d.ts.map +1 -0
- package/dist/types/integrations/mongo.d.ts +26 -0
- package/dist/types/integrations/mongo.d.ts.map +1 -0
- package/dist/types/integrations/mysql.d.ts +20 -0
- package/dist/types/integrations/mysql.d.ts.map +1 -0
- package/dist/types/integrations/postgres.d.ts +20 -0
- package/dist/types/integrations/postgres.d.ts.map +1 -0
- package/dist/types/integrations/redis.d.ts +20 -0
- package/dist/types/integrations/redis.d.ts.map +1 -0
- package/dist/types/middleware/express.d.ts +39 -0
- package/dist/types/middleware/express.d.ts.map +1 -0
- package/example/server.ts +234 -0
- package/jest.config.js +8 -0
- package/package.json +110 -0
- package/portfolio-repo/APIRESPONSE DASH.png +0 -0
- package/portfolio-repo/PAYLOAD.png +0 -0
- package/portfolio-repo/README.md +182 -0
- package/src/core/Analytics.ts +209 -0
- package/src/core/Classifier.ts +82 -0
- package/src/core/HealthMonitor.ts +92 -0
- package/src/core/RootCause.ts +105 -0
- package/src/core/Sampler.ts +35 -0
- package/src/core/Timeline.ts +108 -0
- package/src/core/TraceEngine.ts +266 -0
- package/src/core/types.ts +170 -0
- package/src/index.ts +42 -0
- package/src/integrations/axios.ts +164 -0
- package/src/integrations/fetch.ts +172 -0
- package/src/integrations/mongo.ts +130 -0
- package/src/integrations/mysql.ts +239 -0
- package/src/integrations/postgres.ts +217 -0
- package/src/integrations/redis.ts +122 -0
- package/src/middleware/express.ts +264 -0
- package/tests/Analytics.test.ts +136 -0
- package/tests/Classifier.test.ts +57 -0
- package/tests/RootCause.test.ts +69 -0
- package/tests/TraceEngine.test.ts +110 -0
- package/tsconfig.cjs.json +9 -0
- package/tsconfig.esm.json +9 -0
- package/tsconfig.json +31 -0
- package/tsconfig.types.json +8 -0
|
@@ -0,0 +1,149 @@
|
|
|
1
|
+
// ─────────────────────────────────────────────────────────────
|
|
2
|
+
// flow-debugger — Fetch API Auto-Instrument
|
|
3
|
+
// Wraps global fetch to trace all HTTP requests
|
|
4
|
+
// ─────────────────────────────────────────────────────────────
|
|
5
|
+
import { DEFAULT_CONFIG } from '../core/types';
|
|
6
|
+
import { classifyQuery } from '../core/Classifier';
|
|
7
|
+
const DEFAULT_SERVICE_MAP = {
|
|
8
|
+
'stripe.com': 'stripe',
|
|
9
|
+
'api.stripe.com': 'stripe',
|
|
10
|
+
'razorpay.com': 'razorpay',
|
|
11
|
+
'api.razorpay.com': 'razorpay',
|
|
12
|
+
'sendgrid.com': 'sendgrid',
|
|
13
|
+
'api.sendgrid.com': 'sendgrid',
|
|
14
|
+
'twilio.com': 'twilio',
|
|
15
|
+
'api.twilio.com': 'twilio',
|
|
16
|
+
};
|
|
17
|
+
let originalFetch;
|
|
18
|
+
/**
|
|
19
|
+
* Auto-instrument global fetch to trace all HTTP requests.
|
|
20
|
+
*
|
|
21
|
+
* Usage:
|
|
22
|
+
* fetchTracer({ getTracer: () => currentTracer, serviceMap: { 'myapi.com': 'external' } })
|
|
23
|
+
*
|
|
24
|
+
* Output:
|
|
25
|
+
* Fetch POST https://api.stripe.com/v1/charges → 234ms ✔ [stripe]
|
|
26
|
+
* Fetch GET https://api.razorpay.com/orders → timeout ❌ [razorpay]
|
|
27
|
+
*/
|
|
28
|
+
export function fetchTracer(options) {
|
|
29
|
+
try {
|
|
30
|
+
if (typeof globalThis.fetch === 'undefined')
|
|
31
|
+
return;
|
|
32
|
+
if (globalThis.fetch.__flowDebuggerPatched)
|
|
33
|
+
return;
|
|
34
|
+
const config = { ...DEFAULT_CONFIG, ...options?.config };
|
|
35
|
+
const serviceMap = { ...DEFAULT_SERVICE_MAP, ...options?.serviceMap };
|
|
36
|
+
originalFetch = globalThis.fetch;
|
|
37
|
+
globalThis.fetch = async function tracedFetch(input, init) {
|
|
38
|
+
const tracer = options?.getTracer?.();
|
|
39
|
+
if (!tracer) {
|
|
40
|
+
return originalFetch(input, init);
|
|
41
|
+
}
|
|
42
|
+
const url = typeof input === 'string' ? input : input instanceof URL ? input.href : input.url;
|
|
43
|
+
const method = init?.method || 'GET';
|
|
44
|
+
const service = getServiceFromUrl(url, serviceMap);
|
|
45
|
+
const stepName = `Fetch ${method} ${getDomain(url)}`;
|
|
46
|
+
const startTime = performance.now();
|
|
47
|
+
let status = 'success';
|
|
48
|
+
let error;
|
|
49
|
+
let stackTrace;
|
|
50
|
+
let response;
|
|
51
|
+
try {
|
|
52
|
+
response = await originalFetch(input, init);
|
|
53
|
+
// HTTP errors (4xx, 5xx) are not thrown by fetch, but we should mark them
|
|
54
|
+
if (!response.ok) {
|
|
55
|
+
status = 'error';
|
|
56
|
+
error = `HTTP ${response.status} ${response.statusText}`;
|
|
57
|
+
}
|
|
58
|
+
}
|
|
59
|
+
catch (err) {
|
|
60
|
+
status = 'error';
|
|
61
|
+
const e = err instanceof Error ? err : new Error(String(err));
|
|
62
|
+
error = e.message;
|
|
63
|
+
stackTrace = e.stack;
|
|
64
|
+
// Detect network errors as timeouts
|
|
65
|
+
if (error.includes('fetch failed') || error.includes('network') || error.includes('ECONNREFUSED')) {
|
|
66
|
+
status = 'timeout';
|
|
67
|
+
}
|
|
68
|
+
const endTime = performance.now();
|
|
69
|
+
const duration = endTime - startTime;
|
|
70
|
+
const classification = classifyQuery(duration, status, config);
|
|
71
|
+
tracer.addStep({
|
|
72
|
+
name: stepName,
|
|
73
|
+
service,
|
|
74
|
+
status,
|
|
75
|
+
classification,
|
|
76
|
+
startTime,
|
|
77
|
+
endTime,
|
|
78
|
+
duration,
|
|
79
|
+
error,
|
|
80
|
+
stackTrace,
|
|
81
|
+
metadata: { url, method, service },
|
|
82
|
+
});
|
|
83
|
+
throw err;
|
|
84
|
+
}
|
|
85
|
+
const endTime = performance.now();
|
|
86
|
+
const duration = endTime - startTime;
|
|
87
|
+
const classification = classifyQuery(duration, status, config);
|
|
88
|
+
tracer.addStep({
|
|
89
|
+
name: stepName,
|
|
90
|
+
service,
|
|
91
|
+
status,
|
|
92
|
+
classification,
|
|
93
|
+
startTime,
|
|
94
|
+
endTime,
|
|
95
|
+
duration,
|
|
96
|
+
error,
|
|
97
|
+
metadata: { url, method, statusCode: response.status },
|
|
98
|
+
});
|
|
99
|
+
return response;
|
|
100
|
+
};
|
|
101
|
+
globalThis.fetch.__flowDebuggerPatched = true;
|
|
102
|
+
}
|
|
103
|
+
catch (_) {
|
|
104
|
+
// Production-safe: never crash
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
/**
|
|
108
|
+
* Remove fetch tracer (for testing/cleanup)
|
|
109
|
+
*/
|
|
110
|
+
export function removeFetchTracer() {
|
|
111
|
+
try {
|
|
112
|
+
if (originalFetch) {
|
|
113
|
+
globalThis.fetch = originalFetch;
|
|
114
|
+
delete globalThis.fetch.__flowDebuggerPatched;
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
catch (_) { }
|
|
118
|
+
}
|
|
119
|
+
/** Extract service tag from URL */
|
|
120
|
+
function getServiceFromUrl(url, serviceMap) {
|
|
121
|
+
try {
|
|
122
|
+
const urlObj = new URL(url);
|
|
123
|
+
const hostname = urlObj.hostname;
|
|
124
|
+
// Check exact match
|
|
125
|
+
if (serviceMap[hostname])
|
|
126
|
+
return serviceMap[hostname];
|
|
127
|
+
// Check partial match
|
|
128
|
+
for (const [pattern, service] of Object.entries(serviceMap)) {
|
|
129
|
+
if (hostname.includes(pattern))
|
|
130
|
+
return service;
|
|
131
|
+
}
|
|
132
|
+
// Default to fetch
|
|
133
|
+
return 'fetch';
|
|
134
|
+
}
|
|
135
|
+
catch {
|
|
136
|
+
return 'fetch';
|
|
137
|
+
}
|
|
138
|
+
}
|
|
139
|
+
/** Extract domain from URL for display */
|
|
140
|
+
function getDomain(url) {
|
|
141
|
+
try {
|
|
142
|
+
const urlObj = new URL(url);
|
|
143
|
+
return urlObj.hostname + urlObj.pathname.substring(0, 30);
|
|
144
|
+
}
|
|
145
|
+
catch {
|
|
146
|
+
return url.substring(0, 50);
|
|
147
|
+
}
|
|
148
|
+
}
|
|
149
|
+
//# sourceMappingURL=fetch.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"fetch.js","sourceRoot":"","sources":["../../../src/integrations/fetch.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,107 @@
|
|
|
1
|
+
// ─────────────────────────────────────────────────────────────
|
|
2
|
+
// flow-debugger — MongoDB/Mongoose Auto-Instrument
|
|
3
|
+
// Patches Mongoose to automatically trace all DB queries
|
|
4
|
+
// ─────────────────────────────────────────────────────────────
|
|
5
|
+
import { DEFAULT_CONFIG } from '../core/types';
|
|
6
|
+
import { classifyQuery } from '../core/Classifier';
|
|
7
|
+
/**
|
|
8
|
+
* Auto-instrument Mongoose to trace all queries.
|
|
9
|
+
*
|
|
10
|
+
* Usage:
|
|
11
|
+
* mongoTracer(mongoose, { getTracer: () => currentTracer })
|
|
12
|
+
*
|
|
13
|
+
* Output:
|
|
14
|
+
* Mongo users.findOne → 22ms ✔
|
|
15
|
+
* Mongo payments.insertMany → error ❌
|
|
16
|
+
*/
|
|
17
|
+
export function mongoTracer(mongoose, options) {
|
|
18
|
+
try {
|
|
19
|
+
const config = { ...DEFAULT_CONFIG, ...options?.config };
|
|
20
|
+
const queryProto = mongoose.Query?.prototype;
|
|
21
|
+
if (!queryProto || queryProto.__flowDebuggerPatched)
|
|
22
|
+
return;
|
|
23
|
+
queryProto.__flowDebuggerPatched = true;
|
|
24
|
+
const originalExec = queryProto.exec;
|
|
25
|
+
queryProto.exec = async function (...args) {
|
|
26
|
+
const tracer = options?.getTracer?.();
|
|
27
|
+
if (!tracer) {
|
|
28
|
+
return originalExec.apply(this, args);
|
|
29
|
+
}
|
|
30
|
+
const op = this.op || 'query';
|
|
31
|
+
const collection = this.mongooseCollection?.name || this.model?.collection?.name || 'unknown';
|
|
32
|
+
const stepName = `Mongo ${collection}.${op}`;
|
|
33
|
+
const startTime = performance.now();
|
|
34
|
+
let status = 'success';
|
|
35
|
+
let error;
|
|
36
|
+
let stackTrace;
|
|
37
|
+
let result;
|
|
38
|
+
try {
|
|
39
|
+
result = await originalExec.apply(this, args);
|
|
40
|
+
}
|
|
41
|
+
catch (err) {
|
|
42
|
+
status = 'error';
|
|
43
|
+
const e = err instanceof Error ? err : new Error(String(err));
|
|
44
|
+
error = e.message;
|
|
45
|
+
stackTrace = e.stack;
|
|
46
|
+
// Record step before re-throwing
|
|
47
|
+
const endTime = performance.now();
|
|
48
|
+
const duration = endTime - startTime;
|
|
49
|
+
const classification = classifyQuery(duration, status, config);
|
|
50
|
+
const step = {
|
|
51
|
+
name: stepName,
|
|
52
|
+
service: 'mongo',
|
|
53
|
+
status,
|
|
54
|
+
classification,
|
|
55
|
+
startTime: startTime,
|
|
56
|
+
endTime,
|
|
57
|
+
duration,
|
|
58
|
+
error,
|
|
59
|
+
stackTrace,
|
|
60
|
+
metadata: { operation: op, collection },
|
|
61
|
+
};
|
|
62
|
+
tracer.addStep(step);
|
|
63
|
+
throw err;
|
|
64
|
+
}
|
|
65
|
+
const endTime = performance.now();
|
|
66
|
+
const duration = endTime - startTime;
|
|
67
|
+
const classification = classifyQuery(duration, status, config);
|
|
68
|
+
const step = {
|
|
69
|
+
name: stepName,
|
|
70
|
+
service: 'mongo',
|
|
71
|
+
status,
|
|
72
|
+
classification,
|
|
73
|
+
startTime: startTime,
|
|
74
|
+
endTime,
|
|
75
|
+
duration,
|
|
76
|
+
error,
|
|
77
|
+
stackTrace,
|
|
78
|
+
metadata: { operation: op, collection },
|
|
79
|
+
};
|
|
80
|
+
tracer.addStep(step);
|
|
81
|
+
// Log slow query warning
|
|
82
|
+
if (duration > (config.slowQueryThreshold ?? 300)) {
|
|
83
|
+
try {
|
|
84
|
+
(config.logger || console.log)(`\x1b[33m⚠ Slow MongoDB query: ${stepName} (${Math.round(duration)}ms)\x1b[0m`);
|
|
85
|
+
}
|
|
86
|
+
catch (_) { }
|
|
87
|
+
}
|
|
88
|
+
return result;
|
|
89
|
+
};
|
|
90
|
+
}
|
|
91
|
+
catch (_) {
|
|
92
|
+
// Production-safe: never crash the app
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
/**
|
|
96
|
+
* Remove the monkey-patch from Mongoose (for testing/cleanup).
|
|
97
|
+
*/
|
|
98
|
+
export function removeMongoTracer(mongoose) {
|
|
99
|
+
try {
|
|
100
|
+
const queryProto = mongoose.Query?.prototype;
|
|
101
|
+
if (queryProto) {
|
|
102
|
+
delete queryProto.__flowDebuggerPatched;
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
catch (_) { }
|
|
106
|
+
}
|
|
107
|
+
//# sourceMappingURL=mongo.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"mongo.js","sourceRoot":"","sources":["../../../src/integrations/mongo.ts"],"names":[],"mappings":"AAAA,gEAAgE;AAChE,mDAAmD;AACnD,yDAAyD;AACzD,gEAAgE;AAGhE,OAAO,EAAyC,cAAc,EAAE,MAAM,eAAe,CAAC;AACtF,OAAO,EAAE,aAAa,EAAE,MAAM,oBAAoB,CAAC;AAWnD;;;;;;;;;GASG;AACH,MAAM,UAAU,WAAW,CAAC,QAAqB,EAAE,OAA4B;IAC3E,IAAI,CAAC;QACD,MAAM,MAAM,GAAG,EAAE,GAAG,cAAc,EAAE,GAAG,OAAO,EAAE,MAAM,EAAE,CAAC;QACzD,MAAM,UAAU,GAAG,QAAQ,CAAC,KAAK,EAAE,SAAS,CAAC;QAE7C,IAAI,CAAC,UAAU,IAAI,UAAU,CAAC,qBAAqB;YAAE,OAAO;QAC5D,UAAU,CAAC,qBAAqB,GAAG,IAAI,CAAC;QAExC,MAAM,YAAY,GAAG,UAAU,CAAC,IAAI,CAAC;QAErC,UAAU,CAAC,IAAI,GAAG,KAAK,WAAsB,GAAG,IAAW;YACvD,MAAM,MAAM,GAAG,OAAO,EAAE,SAAS,EAAE,EAAE,CAAC;YACtC,IAAI,CAAC,MAAM,EAAE,CAAC;gBACV,OAAO,YAAY,CAAC,KAAK,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;YAC1C,CAAC;YAED,MAAM,EAAE,GAAG,IAAI,CAAC,EAAE,IAAI,OAAO,CAAC;YAC9B,MAAM,UAAU,GAAG,IAAI,CAAC,kBAAkB,EAAE,IAAI,IAAI,IAAI,CAAC,KAAK,EAAE,UAAU,EAAE,IAAI,IAAI,SAAS,CAAC;YAC9F,MAAM,QAAQ,GAAG,SAAS,UAAU,IAAI,EAAE,EAAE,CAAC;YAC7C,MAAM,SAAS,GAAG,WAAW,CAAC,GAAG,EAAE,CAAC;YAEpC,IAAI,MAAM,GAAe,SAAS,CAAC;YACnC,IAAI,KAAyB,CAAC;YAC9B,IAAI,UAA8B,CAAC;YACnC,IAAI,MAAW,CAAC;YAEhB,IAAI,CAAC;gBACD,MAAM,GAAG,MAAM,YAAY,CAAC,KAAK,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;YAClD,CAAC;YAAC,OAAO,GAAY,EAAE,CAAC;gBACpB,MAAM,GAAG,OAAO,CAAC;gBACjB,MAAM,CAAC,GAAG,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,KAAK,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC;gBAC9D,KAAK,GAAG,CAAC,CAAC,OAAO,CAAC;gBAClB,UAAU,GAAG,CAAC,CAAC,KAAK,CAAC;gBAErB,iCAAiC;gBACjC,MAAM,OAAO,GAAG,WAAW,CAAC,GAAG,EAAE,CAAC;gBAClC,MAAM,QAAQ,GAAG,OAAO,GAAG,SAAS,CAAC;gBACrC,MAAM,cAAc,GAAG,aAAa,CAAC,QAAQ,EAAE,MAAM,EAAE,MAAM,CAAC,CAAC;gBAE/D,MAAM,IAAI,GAAc;oBACpB,IAAI,EAAE,QAAQ;oBACd,OAAO,EAAE,OAAO;oBAChB,MAAM;oBACN,cAAc;oBACd,SAAS,EAAE,SAAS;oBACpB,OAAO;oBACP,QAAQ;oBACR,KAAK;oBACL,UAAU;oBACV,QAAQ,EAAE,EAAE,SAAS,EAAE,EAAE,EAAE,UAAU,EAAE;iBAC1C,CAAC;gBAEF,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;gBACrB,MAAM,GAAG,CAAC;YACd,CAAC;YAED,MAAM,OAAO,GAAG,WAAW,CAAC,GAAG,EAAE,CAAC;YAClC,MAAM,QAAQ,GAAG,OAAO,GAAG,SAAS,CAAC;YACrC,MAAM,cAAc,GAAG,aAAa,CAAC,QAAQ,EAAE,MAAM,EAAE,MAAM,CAAC,CAAC;YAE/D,MAAM,IAAI,GAAc;gBACpB,IAAI,EAAE,QAAQ;gBACd,OAAO,EAAE,OAAO;gBAChB,MAAM;gBACN,cAAc;gBACd,SAAS,EAAE,SAAS;gBACpB,OAAO;gBACP,QAAQ;gBACR,KAAK;gBACL,UAAU;gBACV,QAAQ,EAAE,EAAE,SAAS,EAAE,EAAE,EAAE,UAAU,EAAE;aAC1C,CAAC;YAEF,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;YAErB,yBAAyB;YACzB,IAAI,QAAQ,GAAG,CAAC,MAAM,CAAC,kBAAkB,IAAI,GAAG,CAAC,EAAE,CAAC;gBAChD,IAAI,CAAC;oBACD,CAAC,MAAM,CAAC,MAAM,IAAI,OAAO,CAAC,GAAG,CAAC,CAC1B,iCAAiC,QAAQ,KAAK,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,YAAY,CACjF,CAAC;gBACN,CAAC;gBAAC,OAAO,CAAC,EAAE,CAAC,CAAC,CAAC;YACnB,CAAC;YAED,OAAO,MAAM,CAAC;QAClB,CAAC,CAAC;IACN,CAAC;IAAC,OAAO,CAAC,EAAE,CAAC;QACT,uCAAuC;IAC3C,CAAC;AACL,CAAC;AAED;;GAEG;AACH,MAAM,UAAU,iBAAiB,CAAC,QAAqB;IACnD,IAAI,CAAC;QACD,MAAM,UAAU,GAAG,QAAQ,CAAC,KAAK,EAAE,SAAS,CAAC;QAC7C,IAAI,UAAU,EAAE,CAAC;YACb,OAAO,UAAU,CAAC,qBAAqB,CAAC;QAC5C,CAAC;IACL,CAAC;IAAC,OAAO,CAAC,EAAE,CAAC,CAAC,CAAC;AACnB,CAAC"}
|
|
@@ -0,0 +1,209 @@
|
|
|
1
|
+
// ─────────────────────────────────────────────────────────────
|
|
2
|
+
// flow-debugger — MySQL Auto-Instrument (mysql2)
|
|
3
|
+
// Patches mysql2 pool to trace all queries
|
|
4
|
+
// ─────────────────────────────────────────────────────────────
|
|
5
|
+
import { DEFAULT_CONFIG } from '../core/types';
|
|
6
|
+
import { classifyQuery } from '../core/Classifier';
|
|
7
|
+
/**
|
|
8
|
+
* Auto-instrument mysql2 pool to trace all queries.
|
|
9
|
+
*
|
|
10
|
+
* Usage:
|
|
11
|
+
* mysqlTracer(pool, { getTracer: () => currentTracer })
|
|
12
|
+
*
|
|
13
|
+
* Output:
|
|
14
|
+
* MySQL SELECT users → 12ms ✔
|
|
15
|
+
* MySQL INSERT payments → error ❌
|
|
16
|
+
*/
|
|
17
|
+
export function mysqlTracer(pool, options) {
|
|
18
|
+
try {
|
|
19
|
+
if (pool.__flowDebuggerPatched)
|
|
20
|
+
return;
|
|
21
|
+
pool.__flowDebuggerPatched = true;
|
|
22
|
+
const config = { ...DEFAULT_CONFIG, ...options?.config };
|
|
23
|
+
// Patch pool.query
|
|
24
|
+
const originalQuery = pool.query.bind(pool);
|
|
25
|
+
pool.query = function tracedQuery(...args) {
|
|
26
|
+
const tracer = options?.getTracer?.();
|
|
27
|
+
if (!tracer) {
|
|
28
|
+
return originalQuery(...args);
|
|
29
|
+
}
|
|
30
|
+
const sql = typeof args[0] === 'string' ? args[0] : args[0]?.sql || 'unknown';
|
|
31
|
+
const operation = extractSqlOperation(sql);
|
|
32
|
+
const stepName = `MySQL ${operation}`;
|
|
33
|
+
const startTime = performance.now();
|
|
34
|
+
// Check if using callback or promise style
|
|
35
|
+
const lastArg = args[args.length - 1];
|
|
36
|
+
if (typeof lastArg === 'function') {
|
|
37
|
+
// Callback style
|
|
38
|
+
const callback = lastArg;
|
|
39
|
+
args[args.length - 1] = function (err, results, fields) {
|
|
40
|
+
const endTime = performance.now();
|
|
41
|
+
const duration = endTime - startTime;
|
|
42
|
+
const status = err ? 'error' : 'success';
|
|
43
|
+
const classification = classifyQuery(duration, status, config);
|
|
44
|
+
const step = {
|
|
45
|
+
name: stepName,
|
|
46
|
+
service: 'mysql',
|
|
47
|
+
status,
|
|
48
|
+
classification,
|
|
49
|
+
startTime,
|
|
50
|
+
endTime,
|
|
51
|
+
duration,
|
|
52
|
+
error: err?.message,
|
|
53
|
+
stackTrace: err?.stack,
|
|
54
|
+
metadata: { sql: sql.substring(0, 200), operation },
|
|
55
|
+
};
|
|
56
|
+
tracer.addStep(step);
|
|
57
|
+
if (duration > (config.slowQueryThreshold ?? 300)) {
|
|
58
|
+
try {
|
|
59
|
+
(config.logger || console.log)(`\x1b[33m⚠ Slow MySQL query: ${stepName} (${Math.round(duration)}ms)\n ${sql.substring(0, 100)}\x1b[0m`);
|
|
60
|
+
}
|
|
61
|
+
catch (_) { }
|
|
62
|
+
}
|
|
63
|
+
callback(err, results, fields);
|
|
64
|
+
};
|
|
65
|
+
return originalQuery(...args);
|
|
66
|
+
}
|
|
67
|
+
else {
|
|
68
|
+
// Promise style
|
|
69
|
+
const promise = originalQuery(...args);
|
|
70
|
+
if (promise && typeof promise.then === 'function') {
|
|
71
|
+
return promise.then((result) => {
|
|
72
|
+
const endTime = performance.now();
|
|
73
|
+
const duration = endTime - startTime;
|
|
74
|
+
const classification = classifyQuery(duration, 'success', config);
|
|
75
|
+
const step = {
|
|
76
|
+
name: stepName,
|
|
77
|
+
service: 'mysql',
|
|
78
|
+
status: 'success',
|
|
79
|
+
classification,
|
|
80
|
+
startTime,
|
|
81
|
+
endTime,
|
|
82
|
+
duration,
|
|
83
|
+
metadata: { sql: sql.substring(0, 200), operation },
|
|
84
|
+
};
|
|
85
|
+
tracer.addStep(step);
|
|
86
|
+
if (duration > (config.slowQueryThreshold ?? 300)) {
|
|
87
|
+
try {
|
|
88
|
+
(config.logger || console.log)(`\x1b[33m⚠ Slow MySQL query: ${stepName} (${Math.round(duration)}ms)\n ${sql.substring(0, 100)}\x1b[0m`);
|
|
89
|
+
}
|
|
90
|
+
catch (_) { }
|
|
91
|
+
}
|
|
92
|
+
return result;
|
|
93
|
+
}, (err) => {
|
|
94
|
+
const endTime = performance.now();
|
|
95
|
+
const duration = endTime - startTime;
|
|
96
|
+
const classification = classifyQuery(duration, 'error', config);
|
|
97
|
+
const step = {
|
|
98
|
+
name: stepName,
|
|
99
|
+
service: 'mysql',
|
|
100
|
+
status: 'error',
|
|
101
|
+
classification,
|
|
102
|
+
startTime,
|
|
103
|
+
endTime,
|
|
104
|
+
duration,
|
|
105
|
+
error: err?.message,
|
|
106
|
+
stackTrace: err?.stack,
|
|
107
|
+
metadata: { sql: sql.substring(0, 200), operation },
|
|
108
|
+
};
|
|
109
|
+
tracer.addStep(step);
|
|
110
|
+
throw err;
|
|
111
|
+
});
|
|
112
|
+
}
|
|
113
|
+
return promise;
|
|
114
|
+
}
|
|
115
|
+
};
|
|
116
|
+
// Patch pool.execute (mysql2 specific)
|
|
117
|
+
if (pool.execute) {
|
|
118
|
+
const originalExecute = pool.execute.bind(pool);
|
|
119
|
+
pool.execute = function tracedExecute(...args) {
|
|
120
|
+
const tracer = options?.getTracer?.();
|
|
121
|
+
if (!tracer) {
|
|
122
|
+
return originalExecute(...args);
|
|
123
|
+
}
|
|
124
|
+
const sql = typeof args[0] === 'string' ? args[0] : args[0]?.sql || 'unknown';
|
|
125
|
+
const operation = extractSqlOperation(sql);
|
|
126
|
+
const stepName = `MySQL ${operation}`;
|
|
127
|
+
const startTime = performance.now();
|
|
128
|
+
const lastArg = args[args.length - 1];
|
|
129
|
+
if (typeof lastArg === 'function') {
|
|
130
|
+
const callback = lastArg;
|
|
131
|
+
args[args.length - 1] = function (err, results, fields) {
|
|
132
|
+
const endTime = performance.now();
|
|
133
|
+
const duration = endTime - startTime;
|
|
134
|
+
const status = err ? 'error' : 'success';
|
|
135
|
+
const classification = classifyQuery(duration, status, config);
|
|
136
|
+
tracer.addStep({
|
|
137
|
+
name: stepName,
|
|
138
|
+
service: 'mysql',
|
|
139
|
+
status,
|
|
140
|
+
classification,
|
|
141
|
+
startTime,
|
|
142
|
+
endTime,
|
|
143
|
+
duration,
|
|
144
|
+
error: err?.message,
|
|
145
|
+
stackTrace: err?.stack,
|
|
146
|
+
metadata: { sql: sql.substring(0, 200), operation },
|
|
147
|
+
});
|
|
148
|
+
callback(err, results, fields);
|
|
149
|
+
};
|
|
150
|
+
return originalExecute(...args);
|
|
151
|
+
}
|
|
152
|
+
else {
|
|
153
|
+
const promise = originalExecute(...args);
|
|
154
|
+
if (promise && typeof promise.then === 'function') {
|
|
155
|
+
return promise.then((result) => {
|
|
156
|
+
const endTime = performance.now();
|
|
157
|
+
const duration = endTime - startTime;
|
|
158
|
+
tracer.addStep({
|
|
159
|
+
name: stepName,
|
|
160
|
+
service: 'mysql',
|
|
161
|
+
status: 'success',
|
|
162
|
+
classification: classifyQuery(duration, 'success', config),
|
|
163
|
+
startTime,
|
|
164
|
+
endTime,
|
|
165
|
+
duration,
|
|
166
|
+
metadata: { sql: sql.substring(0, 200), operation },
|
|
167
|
+
});
|
|
168
|
+
return result;
|
|
169
|
+
}, (err) => {
|
|
170
|
+
const endTime = performance.now();
|
|
171
|
+
const duration = endTime - startTime;
|
|
172
|
+
tracer.addStep({
|
|
173
|
+
name: stepName,
|
|
174
|
+
service: 'mysql',
|
|
175
|
+
status: 'error',
|
|
176
|
+
classification: classifyQuery(duration, 'error', config),
|
|
177
|
+
startTime,
|
|
178
|
+
endTime,
|
|
179
|
+
duration,
|
|
180
|
+
error: err?.message,
|
|
181
|
+
stackTrace: err?.stack,
|
|
182
|
+
metadata: { sql: sql.substring(0, 200), operation },
|
|
183
|
+
});
|
|
184
|
+
throw err;
|
|
185
|
+
});
|
|
186
|
+
}
|
|
187
|
+
return promise;
|
|
188
|
+
}
|
|
189
|
+
};
|
|
190
|
+
}
|
|
191
|
+
}
|
|
192
|
+
catch (_) {
|
|
193
|
+
// Production-safe: never crash
|
|
194
|
+
}
|
|
195
|
+
}
|
|
196
|
+
/** Extract the SQL operation type from a query string */
|
|
197
|
+
function extractSqlOperation(sql) {
|
|
198
|
+
const trimmed = sql.trim().toUpperCase();
|
|
199
|
+
const match = trimmed.match(/^(SELECT|INSERT|UPDATE|DELETE|CREATE|ALTER|DROP|TRUNCATE|REPLACE|CALL)\b/);
|
|
200
|
+
if (match) {
|
|
201
|
+
const op = match[1];
|
|
202
|
+
// Try to extract table name
|
|
203
|
+
const tableMatch = sql.match(/(?:FROM|INTO|UPDATE|TABLE|JOIN)\s+[`"]?(\w+)[`"]?/i);
|
|
204
|
+
const table = tableMatch ? tableMatch[1] : '';
|
|
205
|
+
return table ? `${op} ${table}` : op;
|
|
206
|
+
}
|
|
207
|
+
return sql.substring(0, 30);
|
|
208
|
+
}
|
|
209
|
+
//# sourceMappingURL=mysql.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"mysql.js","sourceRoot":"","sources":["../../../src/integrations/mysql.ts"],"names":[],"mappings":"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"}
|