@serve.zone/dcrouter 11.0.39 → 11.0.40

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 (50) hide show
  1. package/dist_serve/bundle.js +1 -1
  2. package/dist_ts/errors/base.errors.js +320 -0
  3. package/dist_ts/errors/error.codes.d.ts +115 -0
  4. package/dist_ts/errors/error.codes.js +136 -0
  5. package/dist_ts/monitoring/classes.metricsmanager.d.ts +178 -0
  6. package/dist_ts/monitoring/classes.metricsmanager.js +642 -0
  7. package/dist_ts/monitoring/index.d.ts +1 -0
  8. package/dist_ts/monitoring/index.js +2 -0
  9. package/dist_ts/opsserver/classes.opsserver.d.ts +37 -0
  10. package/dist_ts/opsserver/classes.opsserver.js +85 -0
  11. package/dist_ts/opsserver/handlers/api-token.handler.d.ts +6 -0
  12. package/dist_ts/opsserver/handlers/api-token.handler.js +62 -0
  13. package/dist_ts/opsserver/handlers/certificate.handler.d.ts +32 -0
  14. package/dist_ts/opsserver/handlers/certificate.handler.js +421 -0
  15. package/dist_ts/opsserver/handlers/email-ops.handler.d.ts +30 -0
  16. package/dist_ts/opsserver/handlers/email-ops.handler.js +227 -0
  17. package/dist_ts/opsserver/handlers/index.d.ts +11 -0
  18. package/dist_ts/opsserver/handlers/index.js +12 -0
  19. package/dist_ts/opsserver/handlers/radius.handler.d.ts +6 -0
  20. package/dist_ts/opsserver/handlers/radius.handler.js +295 -0
  21. package/dist_ts/opsserver/handlers/remoteingress.handler.d.ts +6 -0
  22. package/dist_ts/opsserver/handlers/remoteingress.handler.js +156 -0
  23. package/dist_ts/opsserver/handlers/route-management.handler.d.ts +14 -0
  24. package/dist_ts/opsserver/handlers/route-management.handler.js +117 -0
  25. package/dist_ts/opsserver/handlers/security.handler.d.ts +9 -0
  26. package/dist_ts/opsserver/handlers/security.handler.js +231 -0
  27. package/dist_ts/opsserver/handlers/stats.handler.d.ts +11 -0
  28. package/dist_ts/opsserver/handlers/stats.handler.js +399 -0
  29. package/dist_ts/opsserver/helpers/guards.d.ts +27 -0
  30. package/dist_ts/opsserver/helpers/guards.js +43 -0
  31. package/dist_ts/opsserver/index.d.ts +1 -0
  32. package/dist_ts/opsserver/index.js +2 -0
  33. package/dist_ts/radius/classes.accounting.manager.d.ts +218 -0
  34. package/dist_ts/radius/classes.accounting.manager.js +417 -0
  35. package/dist_ts/radius/classes.radius.server.d.ts +171 -0
  36. package/dist_ts/radius/classes.radius.server.js +385 -0
  37. package/dist_ts/radius/classes.vlan.manager.d.ts +128 -0
  38. package/dist_ts/radius/classes.vlan.manager.js +279 -0
  39. package/dist_ts/radius/index.d.ts +13 -0
  40. package/dist_ts/radius/index.js +14 -0
  41. package/dist_ts/remoteingress/classes.remoteingress-manager.d.ts +82 -0
  42. package/dist_ts/remoteingress/classes.remoteingress-manager.js +227 -0
  43. package/dist_ts/remoteingress/classes.tunnel-manager.d.ts +59 -0
  44. package/dist_ts/remoteingress/classes.tunnel-manager.js +165 -0
  45. package/dist_ts/remoteingress/index.d.ts +2 -0
  46. package/dist_ts/remoteingress/index.js +3 -0
  47. package/dist_ts_web/00_commitinfo_data.js +1 -1
  48. package/package.json +2 -2
  49. package/ts/00_commitinfo_data.ts +1 -1
  50. package/ts_web/00_commitinfo_data.ts +1 -1
@@ -0,0 +1,642 @@
1
+ import * as plugins from '../plugins.js';
2
+ import { DcRouter } from '../classes.dcrouter.js';
3
+ import { MetricsCache } from './classes.metricscache.js';
4
+ import { SecurityLogger, SecurityEventType } from '../security/classes.securitylogger.js';
5
+ import { logger } from '../logger.js';
6
+ export class MetricsManager {
7
+ metricsLogger;
8
+ smartMetrics;
9
+ dcRouter;
10
+ resetInterval;
11
+ metricsCache;
12
+ // Constants
13
+ MAX_TOP_DOMAINS = 1000; // Limit topDomains Map size
14
+ // Track email-specific metrics
15
+ emailMetrics = {
16
+ sentToday: 0,
17
+ receivedToday: 0,
18
+ failedToday: 0,
19
+ bouncedToday: 0,
20
+ queueSize: 0,
21
+ lastResetDate: new Date().toDateString(),
22
+ deliveryTimes: [], // Track delivery times in ms
23
+ recipients: new Map(), // Track email count by recipient
24
+ recentActivity: [],
25
+ };
26
+ // Track DNS-specific metrics
27
+ dnsMetrics = {
28
+ totalQueries: 0,
29
+ cacheHits: 0,
30
+ cacheMisses: 0,
31
+ queryTypes: {},
32
+ topDomains: new Map(),
33
+ lastResetDate: new Date().toDateString(),
34
+ // Per-second query count ring buffer (300 entries = 5 minutes)
35
+ queryRing: new Int32Array(300),
36
+ queryRingLastSecond: 0, // last epoch second that was written
37
+ responseTimes: [], // Track response times in ms
38
+ recentQueries: [],
39
+ };
40
+ // Per-minute time-series buckets for charts
41
+ emailMinuteBuckets = new Map();
42
+ dnsMinuteBuckets = new Map();
43
+ // Track security-specific metrics
44
+ securityMetrics = {
45
+ blockedIPs: 0,
46
+ authFailures: 0,
47
+ spamDetected: 0,
48
+ malwareDetected: 0,
49
+ phishingDetected: 0,
50
+ lastResetDate: new Date().toDateString(),
51
+ incidents: [],
52
+ };
53
+ constructor(dcRouter) {
54
+ this.dcRouter = dcRouter;
55
+ // Create a Smartlog instance for SmartMetrics (requires its own instance)
56
+ this.metricsLogger = new plugins.smartlog.Smartlog({
57
+ logContext: {
58
+ environment: 'production',
59
+ runtime: 'node',
60
+ zone: 'dcrouter-metrics',
61
+ }
62
+ });
63
+ this.smartMetrics = new plugins.smartmetrics.SmartMetrics(this.metricsLogger, 'dcrouter');
64
+ // Initialize metrics cache with 500ms TTL
65
+ this.metricsCache = new MetricsCache(500);
66
+ }
67
+ async start() {
68
+ // Start SmartMetrics collection
69
+ this.smartMetrics.start();
70
+ // Reset daily counters at midnight
71
+ this.resetInterval = setInterval(() => {
72
+ const currentDate = new Date().toDateString();
73
+ if (currentDate !== this.emailMetrics.lastResetDate) {
74
+ this.emailMetrics.sentToday = 0;
75
+ this.emailMetrics.receivedToday = 0;
76
+ this.emailMetrics.failedToday = 0;
77
+ this.emailMetrics.bouncedToday = 0;
78
+ this.emailMetrics.deliveryTimes = [];
79
+ this.emailMetrics.recipients.clear();
80
+ this.emailMetrics.recentActivity = [];
81
+ this.emailMetrics.lastResetDate = currentDate;
82
+ }
83
+ if (currentDate !== this.dnsMetrics.lastResetDate) {
84
+ this.dnsMetrics.totalQueries = 0;
85
+ this.dnsMetrics.cacheHits = 0;
86
+ this.dnsMetrics.cacheMisses = 0;
87
+ this.dnsMetrics.queryTypes = {};
88
+ this.dnsMetrics.topDomains.clear();
89
+ this.dnsMetrics.queryRing.fill(0);
90
+ this.dnsMetrics.queryRingLastSecond = 0;
91
+ this.dnsMetrics.responseTimes = [];
92
+ this.dnsMetrics.recentQueries = [];
93
+ this.dnsMetrics.lastResetDate = currentDate;
94
+ }
95
+ if (currentDate !== this.securityMetrics.lastResetDate) {
96
+ this.securityMetrics.blockedIPs = 0;
97
+ this.securityMetrics.authFailures = 0;
98
+ this.securityMetrics.spamDetected = 0;
99
+ this.securityMetrics.malwareDetected = 0;
100
+ this.securityMetrics.phishingDetected = 0;
101
+ this.securityMetrics.incidents = [];
102
+ this.securityMetrics.lastResetDate = currentDate;
103
+ }
104
+ // Prune old time-series buckets every minute (don't wait for lazy query)
105
+ this.pruneOldBuckets();
106
+ }, 60000); // Check every minute
107
+ logger.log('info', 'MetricsManager started');
108
+ }
109
+ async stop() {
110
+ // Clear the reset interval
111
+ if (this.resetInterval) {
112
+ clearInterval(this.resetInterval);
113
+ this.resetInterval = undefined;
114
+ }
115
+ this.smartMetrics.stop();
116
+ // Clear caches and time-series buckets on shutdown
117
+ this.metricsCache.clear();
118
+ this.emailMinuteBuckets.clear();
119
+ this.dnsMinuteBuckets.clear();
120
+ logger.log('info', 'MetricsManager stopped');
121
+ }
122
+ // Get server metrics from SmartMetrics and SmartProxy
123
+ async getServerStats() {
124
+ return this.metricsCache.get('serverStats', async () => {
125
+ const smartMetricsData = await this.smartMetrics.getMetrics();
126
+ const proxyMetrics = this.dcRouter.smartProxy ? this.dcRouter.smartProxy.getMetrics() : null;
127
+ const proxyStats = this.dcRouter.smartProxy ? await this.dcRouter.smartProxy.getStatistics() : null;
128
+ const { heapUsed, heapTotal, external, rss } = process.memoryUsage();
129
+ return {
130
+ uptime: process.uptime(),
131
+ startTime: Date.now() - (process.uptime() * 1000),
132
+ memoryUsage: {
133
+ heapUsed,
134
+ heapTotal,
135
+ external,
136
+ rss,
137
+ maxMemoryMB: this.smartMetrics.maxMemoryMB,
138
+ actualUsageBytes: smartMetricsData.memoryUsageBytes,
139
+ actualUsagePercentage: smartMetricsData.memoryPercentage,
140
+ },
141
+ cpuUsage: {
142
+ user: smartMetricsData.cpuPercentage,
143
+ system: 0,
144
+ },
145
+ activeConnections: proxyStats ? proxyStats.activeConnections : 0,
146
+ totalConnections: proxyMetrics ? proxyMetrics.totals.connections() : 0,
147
+ requestsPerSecond: proxyMetrics ? proxyMetrics.requests.perSecond() : 0,
148
+ throughput: proxyMetrics ? {
149
+ bytesIn: proxyMetrics.totals.bytesIn(),
150
+ bytesOut: proxyMetrics.totals.bytesOut(),
151
+ bytesInPerSecond: proxyMetrics.throughput.instant().in,
152
+ bytesOutPerSecond: proxyMetrics.throughput.instant().out,
153
+ } : { bytesIn: 0, bytesOut: 0, bytesInPerSecond: 0, bytesOutPerSecond: 0 },
154
+ };
155
+ });
156
+ }
157
+ // Get email metrics
158
+ async getEmailStats() {
159
+ return this.metricsCache.get('emailStats', () => {
160
+ // Calculate average delivery time
161
+ const avgDeliveryTime = this.emailMetrics.deliveryTimes.length > 0
162
+ ? this.emailMetrics.deliveryTimes.reduce((a, b) => a + b, 0) / this.emailMetrics.deliveryTimes.length
163
+ : 0;
164
+ // Get top recipients
165
+ const topRecipients = Array.from(this.emailMetrics.recipients.entries())
166
+ .sort((a, b) => b[1] - a[1])
167
+ .slice(0, 10)
168
+ .map(([email, count]) => ({ email, count }));
169
+ // Get recent activity (last 50 entries)
170
+ const recentActivity = this.emailMetrics.recentActivity.slice(-50);
171
+ return {
172
+ sentToday: this.emailMetrics.sentToday,
173
+ receivedToday: this.emailMetrics.receivedToday,
174
+ failedToday: this.emailMetrics.failedToday,
175
+ bounceRate: this.emailMetrics.bouncedToday > 0
176
+ ? (this.emailMetrics.bouncedToday / this.emailMetrics.sentToday) * 100
177
+ : 0,
178
+ deliveryRate: this.emailMetrics.sentToday > 0
179
+ ? ((this.emailMetrics.sentToday - this.emailMetrics.failedToday) / this.emailMetrics.sentToday) * 100
180
+ : 100,
181
+ queueSize: this.emailMetrics.queueSize,
182
+ averageDeliveryTime: Math.round(avgDeliveryTime),
183
+ topRecipients,
184
+ recentActivity,
185
+ };
186
+ });
187
+ }
188
+ // Get DNS metrics
189
+ async getDnsStats() {
190
+ return this.metricsCache.get('dnsStats', () => {
191
+ const cacheHitRate = this.dnsMetrics.totalQueries > 0
192
+ ? (this.dnsMetrics.cacheHits / this.dnsMetrics.totalQueries) * 100
193
+ : 0;
194
+ const topDomains = Array.from(this.dnsMetrics.topDomains.entries())
195
+ .sort((a, b) => b[1] - a[1])
196
+ .slice(0, 10)
197
+ .map(([domain, count]) => ({ domain, count }));
198
+ // Calculate queries per second from ring buffer (sum last 60 seconds)
199
+ const queriesPerSecond = this.getQueryRingSum(60) / 60;
200
+ // Calculate average response time
201
+ const avgResponseTime = this.dnsMetrics.responseTimes.length > 0
202
+ ? this.dnsMetrics.responseTimes.reduce((a, b) => a + b, 0) / this.dnsMetrics.responseTimes.length
203
+ : 0;
204
+ return {
205
+ queriesPerSecond: Math.round(queriesPerSecond * 10) / 10,
206
+ totalQueries: this.dnsMetrics.totalQueries,
207
+ cacheHits: this.dnsMetrics.cacheHits,
208
+ cacheMisses: this.dnsMetrics.cacheMisses,
209
+ cacheHitRate: cacheHitRate,
210
+ topDomains: topDomains,
211
+ queryTypes: this.dnsMetrics.queryTypes,
212
+ averageResponseTime: Math.round(avgResponseTime),
213
+ activeDomains: this.dnsMetrics.topDomains.size,
214
+ recentQueries: this.dnsMetrics.recentQueries.slice(),
215
+ };
216
+ });
217
+ }
218
+ /**
219
+ * Sync security metrics from the SecurityLogger singleton (last 24h).
220
+ * Called before returning security stats so counters reflect real events.
221
+ */
222
+ syncFromSecurityLogger() {
223
+ try {
224
+ const securityLogger = SecurityLogger.getInstance();
225
+ const summary = securityLogger.getEventsSummary(86400000); // last 24h
226
+ this.securityMetrics.spamDetected = summary.byType[SecurityEventType.SPAM] || 0;
227
+ this.securityMetrics.malwareDetected = summary.byType[SecurityEventType.MALWARE] || 0;
228
+ this.securityMetrics.phishingDetected = summary.byType[SecurityEventType.DMARC] || 0; // phishing via DMARC
229
+ this.securityMetrics.authFailures =
230
+ summary.byType[SecurityEventType.AUTHENTICATION] || 0;
231
+ this.securityMetrics.blockedIPs =
232
+ (summary.byType[SecurityEventType.IP_REPUTATION] || 0) +
233
+ (summary.byType[SecurityEventType.REJECTED_CONNECTION] || 0);
234
+ }
235
+ catch {
236
+ // SecurityLogger may not be initialized yet — ignore
237
+ }
238
+ }
239
+ // Get security metrics
240
+ async getSecurityStats() {
241
+ return this.metricsCache.get('securityStats', () => {
242
+ // Sync counters from the real SecurityLogger events
243
+ this.syncFromSecurityLogger();
244
+ // Get recent incidents (last 20)
245
+ const recentIncidents = this.securityMetrics.incidents.slice(-20);
246
+ return {
247
+ blockedIPs: this.securityMetrics.blockedIPs,
248
+ authFailures: this.securityMetrics.authFailures,
249
+ spamDetected: this.securityMetrics.spamDetected,
250
+ malwareDetected: this.securityMetrics.malwareDetected,
251
+ phishingDetected: this.securityMetrics.phishingDetected,
252
+ totalThreatsBlocked: this.securityMetrics.spamDetected +
253
+ this.securityMetrics.malwareDetected +
254
+ this.securityMetrics.phishingDetected,
255
+ recentIncidents,
256
+ };
257
+ });
258
+ }
259
+ // Get connection info from SmartProxy
260
+ async getConnectionInfo() {
261
+ return this.metricsCache.get('connectionInfo', () => {
262
+ const proxyMetrics = this.dcRouter.smartProxy ? this.dcRouter.smartProxy.getMetrics() : null;
263
+ if (!proxyMetrics) {
264
+ return [];
265
+ }
266
+ const connectionsByRoute = proxyMetrics.connections.byRoute();
267
+ const connectionInfo = [];
268
+ for (const [routeName, count] of connectionsByRoute) {
269
+ connectionInfo.push({
270
+ type: 'https',
271
+ count,
272
+ source: routeName,
273
+ lastActivity: new Date(),
274
+ });
275
+ }
276
+ return connectionInfo;
277
+ });
278
+ }
279
+ // Email event tracking methods
280
+ trackEmailSent(recipient, deliveryTimeMs) {
281
+ this.emailMetrics.sentToday++;
282
+ this.incrementEmailBucket('sent');
283
+ if (recipient) {
284
+ const count = this.emailMetrics.recipients.get(recipient) || 0;
285
+ this.emailMetrics.recipients.set(recipient, count + 1);
286
+ // Cap recipients map to prevent unbounded growth within a day
287
+ if (this.emailMetrics.recipients.size > this.MAX_TOP_DOMAINS) {
288
+ const sorted = Array.from(this.emailMetrics.recipients.entries())
289
+ .sort((a, b) => b[1] - a[1])
290
+ .slice(0, Math.floor(this.MAX_TOP_DOMAINS * 0.8));
291
+ this.emailMetrics.recipients = new Map(sorted);
292
+ }
293
+ }
294
+ if (deliveryTimeMs) {
295
+ this.emailMetrics.deliveryTimes.push(deliveryTimeMs);
296
+ // Keep only last 1000 delivery times
297
+ if (this.emailMetrics.deliveryTimes.length > 1000) {
298
+ this.emailMetrics.deliveryTimes.shift();
299
+ }
300
+ }
301
+ this.emailMetrics.recentActivity.push({
302
+ timestamp: Date.now(),
303
+ type: 'sent',
304
+ details: recipient || 'unknown',
305
+ });
306
+ // Keep only last 1000 activities
307
+ if (this.emailMetrics.recentActivity.length > 1000) {
308
+ this.emailMetrics.recentActivity.shift();
309
+ }
310
+ }
311
+ trackEmailReceived(sender) {
312
+ this.emailMetrics.receivedToday++;
313
+ this.incrementEmailBucket('received');
314
+ this.emailMetrics.recentActivity.push({
315
+ timestamp: Date.now(),
316
+ type: 'received',
317
+ details: sender || 'unknown',
318
+ });
319
+ // Keep only last 1000 activities
320
+ if (this.emailMetrics.recentActivity.length > 1000) {
321
+ this.emailMetrics.recentActivity.shift();
322
+ }
323
+ }
324
+ trackEmailFailed(recipient, reason) {
325
+ this.emailMetrics.failedToday++;
326
+ this.incrementEmailBucket('failed');
327
+ this.emailMetrics.recentActivity.push({
328
+ timestamp: Date.now(),
329
+ type: 'failed',
330
+ details: `${recipient || 'unknown'}: ${reason || 'unknown error'}`,
331
+ });
332
+ // Keep only last 1000 activities
333
+ if (this.emailMetrics.recentActivity.length > 1000) {
334
+ this.emailMetrics.recentActivity.shift();
335
+ }
336
+ }
337
+ trackEmailBounced(recipient) {
338
+ this.emailMetrics.bouncedToday++;
339
+ this.emailMetrics.recentActivity.push({
340
+ timestamp: Date.now(),
341
+ type: 'bounced',
342
+ details: recipient || 'unknown',
343
+ });
344
+ // Keep only last 1000 activities
345
+ if (this.emailMetrics.recentActivity.length > 1000) {
346
+ this.emailMetrics.recentActivity.shift();
347
+ }
348
+ }
349
+ updateQueueSize(size) {
350
+ this.emailMetrics.queueSize = size;
351
+ }
352
+ // DNS event tracking methods
353
+ trackDnsQuery(queryType, domain, cacheHit, responseTimeMs, answered) {
354
+ this.dnsMetrics.totalQueries++;
355
+ this.incrementDnsBucket();
356
+ // Store recent query entry
357
+ this.dnsMetrics.recentQueries.push({
358
+ timestamp: Date.now(),
359
+ domain,
360
+ type: queryType,
361
+ answered: answered ?? true,
362
+ responseTimeMs: responseTimeMs ?? 0,
363
+ });
364
+ if (this.dnsMetrics.recentQueries.length > 100) {
365
+ this.dnsMetrics.recentQueries.shift();
366
+ }
367
+ if (cacheHit) {
368
+ this.dnsMetrics.cacheHits++;
369
+ }
370
+ else {
371
+ this.dnsMetrics.cacheMisses++;
372
+ }
373
+ // Increment per-second query counter in ring buffer
374
+ this.incrementQueryRing();
375
+ // Track response time if provided
376
+ if (responseTimeMs) {
377
+ this.dnsMetrics.responseTimes.push(responseTimeMs);
378
+ // Keep only last 1000 response times
379
+ if (this.dnsMetrics.responseTimes.length > 1000) {
380
+ this.dnsMetrics.responseTimes.shift();
381
+ }
382
+ }
383
+ // Track query types
384
+ this.dnsMetrics.queryTypes[queryType] = (this.dnsMetrics.queryTypes[queryType] || 0) + 1;
385
+ // Track top domains with size limit
386
+ const currentCount = this.dnsMetrics.topDomains.get(domain) || 0;
387
+ this.dnsMetrics.topDomains.set(domain, currentCount + 1);
388
+ // If we've exceeded the limit, remove the least accessed domains
389
+ if (this.dnsMetrics.topDomains.size > this.MAX_TOP_DOMAINS) {
390
+ // Convert to array, sort by count, and keep only top domains
391
+ const sortedDomains = Array.from(this.dnsMetrics.topDomains.entries())
392
+ .sort((a, b) => b[1] - a[1])
393
+ .slice(0, Math.floor(this.MAX_TOP_DOMAINS * 0.8)); // Keep 80% to avoid frequent cleanup
394
+ // Clear and repopulate with top domains
395
+ this.dnsMetrics.topDomains.clear();
396
+ sortedDomains.forEach(([domain, count]) => {
397
+ this.dnsMetrics.topDomains.set(domain, count);
398
+ });
399
+ }
400
+ }
401
+ // Security event tracking methods
402
+ trackBlockedIP(ip, reason) {
403
+ this.securityMetrics.blockedIPs++;
404
+ this.securityMetrics.incidents.push({
405
+ timestamp: Date.now(),
406
+ type: 'ip_blocked',
407
+ severity: 'medium',
408
+ details: `IP ${ip || 'unknown'} blocked: ${reason || 'security policy'}`,
409
+ });
410
+ // Keep only last 1000 incidents
411
+ if (this.securityMetrics.incidents.length > 1000) {
412
+ this.securityMetrics.incidents.shift();
413
+ }
414
+ }
415
+ trackAuthFailure(username, ip) {
416
+ this.securityMetrics.authFailures++;
417
+ this.securityMetrics.incidents.push({
418
+ timestamp: Date.now(),
419
+ type: 'auth_failure',
420
+ severity: 'low',
421
+ details: `Authentication failed for ${username || 'unknown'} from ${ip || 'unknown'}`,
422
+ });
423
+ // Keep only last 1000 incidents
424
+ if (this.securityMetrics.incidents.length > 1000) {
425
+ this.securityMetrics.incidents.shift();
426
+ }
427
+ }
428
+ trackSpamDetected(sender) {
429
+ this.securityMetrics.spamDetected++;
430
+ this.securityMetrics.incidents.push({
431
+ timestamp: Date.now(),
432
+ type: 'spam_detected',
433
+ severity: 'low',
434
+ details: `Spam detected from ${sender || 'unknown'}`,
435
+ });
436
+ // Keep only last 1000 incidents
437
+ if (this.securityMetrics.incidents.length > 1000) {
438
+ this.securityMetrics.incidents.shift();
439
+ }
440
+ }
441
+ trackMalwareDetected(source) {
442
+ this.securityMetrics.malwareDetected++;
443
+ this.securityMetrics.incidents.push({
444
+ timestamp: Date.now(),
445
+ type: 'malware_detected',
446
+ severity: 'high',
447
+ details: `Malware detected from ${source || 'unknown'}`,
448
+ });
449
+ // Keep only last 1000 incidents
450
+ if (this.securityMetrics.incidents.length > 1000) {
451
+ this.securityMetrics.incidents.shift();
452
+ }
453
+ }
454
+ trackPhishingDetected(source) {
455
+ this.securityMetrics.phishingDetected++;
456
+ this.securityMetrics.incidents.push({
457
+ timestamp: Date.now(),
458
+ type: 'phishing_detected',
459
+ severity: 'high',
460
+ details: `Phishing attempt from ${source || 'unknown'}`,
461
+ });
462
+ // Keep only last 1000 incidents
463
+ if (this.securityMetrics.incidents.length > 1000) {
464
+ this.securityMetrics.incidents.shift();
465
+ }
466
+ }
467
+ // Get network metrics from SmartProxy
468
+ async getNetworkStats() {
469
+ // Use shorter cache TTL for network stats to ensure real-time updates
470
+ return this.metricsCache.get('networkStats', () => {
471
+ const proxyMetrics = this.dcRouter.smartProxy ? this.dcRouter.smartProxy.getMetrics() : null;
472
+ if (!proxyMetrics) {
473
+ return {
474
+ connectionsByIP: new Map(),
475
+ throughputRate: { bytesInPerSecond: 0, bytesOutPerSecond: 0 },
476
+ topIPs: [],
477
+ totalDataTransferred: { bytesIn: 0, bytesOut: 0 },
478
+ throughputHistory: [],
479
+ throughputByIP: new Map(),
480
+ requestsPerSecond: 0,
481
+ requestsTotal: 0,
482
+ };
483
+ }
484
+ // Get metrics using the new API
485
+ const connectionsByIP = proxyMetrics.connections.byIP();
486
+ const instantThroughput = proxyMetrics.throughput.instant();
487
+ // Get throughput rate
488
+ const throughputRate = {
489
+ bytesInPerSecond: instantThroughput.in,
490
+ bytesOutPerSecond: instantThroughput.out
491
+ };
492
+ // Get top IPs
493
+ const topIPs = proxyMetrics.connections.topIPs(10);
494
+ // Get total data transferred
495
+ const totalDataTransferred = {
496
+ bytesIn: proxyMetrics.totals.bytesIn(),
497
+ bytesOut: proxyMetrics.totals.bytesOut()
498
+ };
499
+ // Get throughput history from Rust engine (up to 300 seconds)
500
+ const throughputHistory = proxyMetrics.throughput.history(300);
501
+ // Get per-IP throughput
502
+ const throughputByIP = proxyMetrics.throughput.byIP();
503
+ // Get HTTP request rates
504
+ const requestsPerSecond = proxyMetrics.requests.perSecond();
505
+ const requestsTotal = proxyMetrics.requests.total();
506
+ return {
507
+ connectionsByIP,
508
+ throughputRate,
509
+ topIPs,
510
+ totalDataTransferred,
511
+ throughputHistory,
512
+ throughputByIP,
513
+ requestsPerSecond,
514
+ requestsTotal,
515
+ };
516
+ }, 1000); // 1s cache — matches typical dashboard poll interval
517
+ }
518
+ // --- Time-series helpers ---
519
+ static minuteKey(ts = Date.now()) {
520
+ return Math.floor(ts / 60000) * 60000;
521
+ }
522
+ incrementEmailBucket(field) {
523
+ const key = MetricsManager.minuteKey();
524
+ let bucket = this.emailMinuteBuckets.get(key);
525
+ if (!bucket) {
526
+ bucket = { sent: 0, received: 0, failed: 0 };
527
+ this.emailMinuteBuckets.set(key, bucket);
528
+ }
529
+ bucket[field]++;
530
+ }
531
+ incrementDnsBucket() {
532
+ const key = MetricsManager.minuteKey();
533
+ let bucket = this.dnsMinuteBuckets.get(key);
534
+ if (!bucket) {
535
+ bucket = { queries: 0 };
536
+ this.dnsMinuteBuckets.set(key, bucket);
537
+ }
538
+ bucket.queries++;
539
+ }
540
+ /**
541
+ * Increment the per-second query counter in the ring buffer.
542
+ * Zeros any stale slots between the last write and the current second.
543
+ */
544
+ incrementQueryRing() {
545
+ const currentSecond = Math.floor(Date.now() / 1000);
546
+ const ring = this.dnsMetrics.queryRing;
547
+ const last = this.dnsMetrics.queryRingLastSecond;
548
+ if (last === 0) {
549
+ // First call — zero and anchor
550
+ ring.fill(0);
551
+ this.dnsMetrics.queryRingLastSecond = currentSecond;
552
+ ring[currentSecond % ring.length] = 1;
553
+ return;
554
+ }
555
+ const gap = currentSecond - last;
556
+ if (gap >= ring.length) {
557
+ // Entire ring is stale — clear all
558
+ ring.fill(0);
559
+ }
560
+ else if (gap > 0) {
561
+ // Zero slots from (last+1) to currentSecond (inclusive)
562
+ for (let s = last + 1; s <= currentSecond; s++) {
563
+ ring[s % ring.length] = 0;
564
+ }
565
+ }
566
+ this.dnsMetrics.queryRingLastSecond = currentSecond;
567
+ ring[currentSecond % ring.length]++;
568
+ }
569
+ /**
570
+ * Sum query counts from the ring buffer for the last N seconds.
571
+ */
572
+ getQueryRingSum(seconds) {
573
+ const currentSecond = Math.floor(Date.now() / 1000);
574
+ const ring = this.dnsMetrics.queryRing;
575
+ const last = this.dnsMetrics.queryRingLastSecond;
576
+ if (last === 0)
577
+ return 0;
578
+ // First, zero stale slots so reads are accurate even without writes
579
+ const gap = currentSecond - last;
580
+ if (gap >= ring.length)
581
+ return 0; // all data is stale
582
+ let sum = 0;
583
+ const limit = Math.min(seconds, ring.length);
584
+ for (let i = 0; i < limit; i++) {
585
+ const sec = currentSecond - i;
586
+ if (sec < last - (ring.length - 1))
587
+ break; // slot is from older cycle
588
+ if (sec > last)
589
+ continue; // no writes yet for this second
590
+ sum += ring[sec % ring.length];
591
+ }
592
+ return sum;
593
+ }
594
+ pruneOldBuckets() {
595
+ const cutoff = Date.now() - 86400000; // 24h
596
+ for (const key of this.emailMinuteBuckets.keys()) {
597
+ if (key < cutoff)
598
+ this.emailMinuteBuckets.delete(key);
599
+ }
600
+ for (const key of this.dnsMinuteBuckets.keys()) {
601
+ if (key < cutoff)
602
+ this.dnsMinuteBuckets.delete(key);
603
+ }
604
+ }
605
+ /**
606
+ * Get email time-series data for the last N hours, aggregated per minute.
607
+ */
608
+ getEmailTimeSeries(hours = 24) {
609
+ this.pruneOldBuckets();
610
+ const cutoff = Date.now() - hours * 3600000;
611
+ const sent = [];
612
+ const received = [];
613
+ const failed = [];
614
+ const sortedKeys = Array.from(this.emailMinuteBuckets.keys())
615
+ .filter((k) => k >= cutoff)
616
+ .sort((a, b) => a - b);
617
+ for (const key of sortedKeys) {
618
+ const bucket = this.emailMinuteBuckets.get(key);
619
+ sent.push({ timestamp: key, value: bucket.sent });
620
+ received.push({ timestamp: key, value: bucket.received });
621
+ failed.push({ timestamp: key, value: bucket.failed });
622
+ }
623
+ return { sent, received, failed };
624
+ }
625
+ /**
626
+ * Get DNS time-series data for the last N hours, aggregated per minute.
627
+ */
628
+ getDnsTimeSeries(hours = 24) {
629
+ this.pruneOldBuckets();
630
+ const cutoff = Date.now() - hours * 3600000;
631
+ const queries = [];
632
+ const sortedKeys = Array.from(this.dnsMinuteBuckets.keys())
633
+ .filter((k) => k >= cutoff)
634
+ .sort((a, b) => a - b);
635
+ for (const key of sortedKeys) {
636
+ const bucket = this.dnsMinuteBuckets.get(key);
637
+ queries.push({ timestamp: key, value: bucket.queries });
638
+ }
639
+ return { queries };
640
+ }
641
+ }
642
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"classes.metricsmanager.js","sourceRoot":"","sources":["../../../ts/monitoring/classes.metricsmanager.ts"],"names":[],"mappings":"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"}