@serve.zone/dcrouter 13.1.3 → 13.3.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 (69) hide show
  1. package/dist_serve/bundle.js +1202 -1133
  2. package/dist_ts/00_commitinfo_data.js +1 -1
  3. package/dist_ts_web/00_commitinfo_data.js +1 -1
  4. package/dist_ts_web/appstate.d.ts +1 -0
  5. package/dist_ts_web/appstate.js +9 -22
  6. package/dist_ts_web/elements/index.d.ts +2 -6
  7. package/dist_ts_web/elements/index.js +3 -7
  8. package/dist_ts_web/elements/network/index.d.ts +6 -0
  9. package/dist_ts_web/elements/network/index.js +7 -0
  10. package/dist_ts_web/elements/{ops-view-network.d.ts → network/ops-view-network-activity.d.ts} +3 -3
  11. package/dist_ts_web/elements/{ops-view-network.js → network/ops-view-network-activity.js} +20 -30
  12. package/dist_ts_web/elements/network/ops-view-network.d.ts +24 -0
  13. package/dist_ts_web/elements/network/ops-view-network.js +151 -0
  14. package/dist_ts_web/elements/{ops-view-networktargets.d.ts → network/ops-view-networktargets.d.ts} +1 -1
  15. package/dist_ts_web/elements/{ops-view-networktargets.js → network/ops-view-networktargets.js} +6 -6
  16. package/dist_ts_web/elements/{ops-view-routes.d.ts → network/ops-view-routes.d.ts} +1 -1
  17. package/dist_ts_web/elements/{ops-view-routes.js → network/ops-view-routes.js} +5 -6
  18. package/dist_ts_web/elements/{ops-view-sourceprofiles.d.ts → network/ops-view-sourceprofiles.d.ts} +1 -1
  19. package/dist_ts_web/elements/{ops-view-sourceprofiles.js → network/ops-view-sourceprofiles.js} +6 -6
  20. package/dist_ts_web/elements/{ops-view-targetprofiles.d.ts → network/ops-view-targetprofiles.d.ts} +2 -2
  21. package/dist_ts_web/elements/{ops-view-targetprofiles.js → network/ops-view-targetprofiles.js} +7 -7
  22. package/dist_ts_web/elements/ops-dashboard.js +4 -27
  23. package/dist_ts_web/elements/ops-view-apitokens.js +2 -1
  24. package/dist_ts_web/elements/ops-view-certificates.js +2 -1
  25. package/dist_ts_web/elements/ops-view-config.js +3 -3
  26. package/dist_ts_web/elements/ops-view-remoteingress.js +2 -1
  27. package/dist_ts_web/elements/ops-view-vpn.js +2 -1
  28. package/dist_ts_web/elements/security/index.d.ts +5 -0
  29. package/dist_ts_web/elements/security/index.js +6 -0
  30. package/dist_ts_web/elements/security/ops-view-security-authentication.d.ts +13 -0
  31. package/dist_ts_web/elements/security/ops-view-security-authentication.js +156 -0
  32. package/dist_ts_web/elements/security/ops-view-security-blocked.d.ts +15 -0
  33. package/dist_ts_web/elements/security/ops-view-security-blocked.js +152 -0
  34. package/dist_ts_web/elements/security/ops-view-security-emailsecurity.d.ts +14 -0
  35. package/dist_ts_web/elements/security/ops-view-security-emailsecurity.js +196 -0
  36. package/dist_ts_web/elements/security/ops-view-security-overview.d.ts +16 -0
  37. package/dist_ts_web/elements/security/ops-view-security-overview.js +204 -0
  38. package/dist_ts_web/elements/security/ops-view-security.d.ts +23 -0
  39. package/dist_ts_web/elements/security/ops-view-security.js +146 -0
  40. package/dist_ts_web/router.d.ts +5 -3
  41. package/dist_ts_web/router.js +69 -17
  42. package/package.json +1 -1
  43. package/ts/00_commitinfo_data.ts +1 -1
  44. package/ts_web/00_commitinfo_data.ts +1 -1
  45. package/ts_web/appstate.ts +10 -24
  46. package/ts_web/elements/index.ts +2 -6
  47. package/ts_web/elements/network/index.ts +6 -0
  48. package/ts_web/elements/{ops-view-network.ts → network/ops-view-network-activity.ts} +43 -53
  49. package/ts_web/elements/network/ops-view-network.ts +119 -0
  50. package/ts_web/elements/{ops-view-networktargets.ts → network/ops-view-networktargets.ts} +5 -5
  51. package/ts_web/elements/{ops-view-routes.ts → network/ops-view-routes.ts} +4 -5
  52. package/ts_web/elements/{ops-view-sourceprofiles.ts → network/ops-view-sourceprofiles.ts} +5 -5
  53. package/ts_web/elements/{ops-view-targetprofiles.ts → network/ops-view-targetprofiles.ts} +6 -6
  54. package/ts_web/elements/ops-dashboard.ts +3 -26
  55. package/ts_web/elements/ops-view-apitokens.ts +1 -0
  56. package/ts_web/elements/ops-view-certificates.ts +1 -0
  57. package/ts_web/elements/ops-view-config.ts +2 -2
  58. package/ts_web/elements/ops-view-remoteingress.ts +1 -0
  59. package/ts_web/elements/ops-view-vpn.ts +1 -0
  60. package/ts_web/elements/security/index.ts +5 -0
  61. package/ts_web/elements/security/ops-view-security-authentication.ts +120 -0
  62. package/ts_web/elements/security/ops-view-security-blocked.ts +117 -0
  63. package/ts_web/elements/security/ops-view-security-emailsecurity.ts +159 -0
  64. package/ts_web/elements/security/ops-view-security-overview.ts +171 -0
  65. package/ts_web/elements/security/ops-view-security.ts +114 -0
  66. package/ts_web/router.ts +75 -17
  67. package/dist_ts_web/elements/ops-view-security.d.ts +0 -24
  68. package/dist_ts_web/elements/ops-view-security.js +0 -481
  69. package/ts_web/elements/ops-view-security.ts +0 -453
@@ -1,481 +0,0 @@
1
- var __esDecorate = (this && this.__esDecorate) || function (ctor, descriptorIn, decorators, contextIn, initializers, extraInitializers) {
2
- function accept(f) { if (f !== void 0 && typeof f !== "function") throw new TypeError("Function expected"); return f; }
3
- var kind = contextIn.kind, key = kind === "getter" ? "get" : kind === "setter" ? "set" : "value";
4
- var target = !descriptorIn && ctor ? contextIn["static"] ? ctor : ctor.prototype : null;
5
- var descriptor = descriptorIn || (target ? Object.getOwnPropertyDescriptor(target, contextIn.name) : {});
6
- var _, done = false;
7
- for (var i = decorators.length - 1; i >= 0; i--) {
8
- var context = {};
9
- for (var p in contextIn) context[p] = p === "access" ? {} : contextIn[p];
10
- for (var p in contextIn.access) context.access[p] = contextIn.access[p];
11
- context.addInitializer = function (f) { if (done) throw new TypeError("Cannot add initializers after decoration has completed"); extraInitializers.push(accept(f || null)); };
12
- var result = (0, decorators[i])(kind === "accessor" ? { get: descriptor.get, set: descriptor.set } : descriptor[key], context);
13
- if (kind === "accessor") {
14
- if (result === void 0) continue;
15
- if (result === null || typeof result !== "object") throw new TypeError("Object expected");
16
- if (_ = accept(result.get)) descriptor.get = _;
17
- if (_ = accept(result.set)) descriptor.set = _;
18
- if (_ = accept(result.init)) initializers.unshift(_);
19
- }
20
- else if (_ = accept(result)) {
21
- if (kind === "field") initializers.unshift(_);
22
- else descriptor[key] = _;
23
- }
24
- }
25
- if (target) Object.defineProperty(target, contextIn.name, descriptor);
26
- done = true;
27
- };
28
- var __runInitializers = (this && this.__runInitializers) || function (thisArg, initializers, value) {
29
- var useValue = arguments.length > 2;
30
- for (var i = 0; i < initializers.length; i++) {
31
- value = useValue ? initializers[i].call(thisArg, value) : initializers[i].call(thisArg);
32
- }
33
- return useValue ? value : void 0;
34
- };
35
- import * as plugins from '../plugins.js';
36
- import * as shared from './shared/index.js';
37
- import * as appstate from '../appstate.js';
38
- import { DeesElement, customElement, html, state, css, cssManager, } from '@design.estate/dees-element';
39
- import {} from '@design.estate/dees-catalog';
40
- let OpsViewSecurity = (() => {
41
- let _classDecorators = [customElement('ops-view-security')];
42
- let _classDescriptor;
43
- let _classExtraInitializers = [];
44
- let _classThis;
45
- let _classSuper = DeesElement;
46
- let _statsState_decorators;
47
- let _statsState_initializers = [];
48
- let _statsState_extraInitializers = [];
49
- let _selectedTab_decorators;
50
- let _selectedTab_initializers = [];
51
- let _selectedTab_extraInitializers = [];
52
- var OpsViewSecurity = class extends _classSuper {
53
- static { _classThis = this; }
54
- static {
55
- const _metadata = typeof Symbol === "function" && Symbol.metadata ? Object.create(_classSuper[Symbol.metadata] ?? null) : void 0;
56
- _statsState_decorators = [state()];
57
- _selectedTab_decorators = [state()];
58
- __esDecorate(this, null, _statsState_decorators, { kind: "accessor", name: "statsState", static: false, private: false, access: { has: obj => "statsState" in obj, get: obj => obj.statsState, set: (obj, value) => { obj.statsState = value; } }, metadata: _metadata }, _statsState_initializers, _statsState_extraInitializers);
59
- __esDecorate(this, null, _selectedTab_decorators, { kind: "accessor", name: "selectedTab", static: false, private: false, access: { has: obj => "selectedTab" in obj, get: obj => obj.selectedTab, set: (obj, value) => { obj.selectedTab = value; } }, metadata: _metadata }, _selectedTab_initializers, _selectedTab_extraInitializers);
60
- __esDecorate(null, _classDescriptor = { value: _classThis }, _classDecorators, { kind: "class", name: _classThis.name, metadata: _metadata }, null, _classExtraInitializers);
61
- OpsViewSecurity = _classThis = _classDescriptor.value;
62
- if (_metadata) Object.defineProperty(_classThis, Symbol.metadata, { enumerable: true, configurable: true, writable: true, value: _metadata });
63
- }
64
- #statsState_accessor_storage = __runInitializers(this, _statsState_initializers, {
65
- serverStats: null,
66
- emailStats: null,
67
- dnsStats: null,
68
- securityMetrics: null,
69
- radiusStats: null,
70
- vpnStats: null,
71
- lastUpdated: 0,
72
- isLoading: false,
73
- error: null,
74
- });
75
- get statsState() { return this.#statsState_accessor_storage; }
76
- set statsState(value) { this.#statsState_accessor_storage = value; }
77
- #selectedTab_accessor_storage = (__runInitializers(this, _statsState_extraInitializers), __runInitializers(this, _selectedTab_initializers, 'overview'));
78
- get selectedTab() { return this.#selectedTab_accessor_storage; }
79
- set selectedTab(value) { this.#selectedTab_accessor_storage = value; }
80
- tabLabelMap = (__runInitializers(this, _selectedTab_extraInitializers), {
81
- 'overview': 'Overview',
82
- 'blocked': 'Blocked IPs',
83
- 'authentication': 'Authentication',
84
- 'email-security': 'Email Security',
85
- });
86
- labelToTab = {
87
- 'Overview': 'overview',
88
- 'Blocked IPs': 'blocked',
89
- 'Authentication': 'authentication',
90
- 'Email Security': 'email-security',
91
- };
92
- constructor() {
93
- super();
94
- const subscription = appstate.statsStatePart
95
- .select((stateArg) => stateArg)
96
- .subscribe((statsState) => {
97
- this.statsState = statsState;
98
- });
99
- this.rxSubscriptions.push(subscription);
100
- }
101
- async firstUpdated() {
102
- const toggle = this.shadowRoot.querySelector('dees-input-multitoggle');
103
- if (toggle) {
104
- const sub = toggle.changeSubject.subscribe(() => {
105
- const tab = this.labelToTab[toggle.selectedOption];
106
- if (tab)
107
- this.selectedTab = tab;
108
- });
109
- this.rxSubscriptions.push(sub);
110
- }
111
- }
112
- static styles = [
113
- cssManager.defaultStyles,
114
- shared.viewHostCss,
115
- css `
116
- dees-input-multitoggle {
117
- margin-bottom: 24px;
118
- }
119
-
120
- h2 {
121
- margin: 32px 0 16px 0;
122
- font-size: 24px;
123
- font-weight: 600;
124
- color: ${cssManager.bdTheme('#333', '#ccc')};
125
- }
126
-
127
- dees-statsgrid {
128
- margin-bottom: 32px;
129
- }
130
-
131
- .securityCard {
132
- background: ${cssManager.bdTheme('#fff', '#222')};
133
- border: 1px solid ${cssManager.bdTheme('#e9ecef', '#333')};
134
- border-radius: 8px;
135
- padding: 24px;
136
- position: relative;
137
- overflow: hidden;
138
- }
139
-
140
- .actionButton {
141
- margin-top: 16px;
142
- }
143
-
144
- `,
145
- ];
146
- render() {
147
- return html `
148
- <dees-heading level="2">Security</dees-heading>
149
-
150
- <dees-input-multitoggle
151
- .type=${'single'}
152
- .options=${['Overview', 'Blocked IPs', 'Authentication', 'Email Security']}
153
- .selectedOption=${this.tabLabelMap[this.selectedTab]}
154
- ></dees-input-multitoggle>
155
-
156
- ${this.renderTabContent()}
157
- `;
158
- }
159
- renderTabContent() {
160
- const metrics = this.statsState.securityMetrics;
161
- if (!metrics) {
162
- return html `
163
- <div class="loadingMessage">
164
- <p>Loading security metrics...</p>
165
- </div>
166
- `;
167
- }
168
- switch (this.selectedTab) {
169
- case 'overview':
170
- return this.renderOverview(metrics);
171
- case 'blocked':
172
- return this.renderBlockedIPs(metrics);
173
- case 'authentication':
174
- return this.renderAuthentication(metrics);
175
- case 'email-security':
176
- return this.renderEmailSecurity(metrics);
177
- }
178
- }
179
- renderOverview(metrics) {
180
- const threatLevel = this.calculateThreatLevel(metrics);
181
- const threatScore = this.getThreatScore(metrics);
182
- // Derive active sessions from recent successful auth events (last hour)
183
- const allEvents = metrics.recentEvents || [];
184
- const oneHourAgo = Date.now() - 3600000;
185
- const recentAuthSuccesses = allEvents.filter((evt) => evt.type === 'authentication' && evt.success === true && evt.timestamp >= oneHourAgo).length;
186
- const tiles = [
187
- {
188
- id: 'threatLevel',
189
- title: 'Threat Level',
190
- value: threatScore,
191
- type: 'gauge',
192
- icon: 'lucide:Shield',
193
- gaugeOptions: {
194
- min: 0,
195
- max: 100,
196
- thresholds: [
197
- { value: 0, color: '#ef4444' },
198
- { value: 30, color: '#f59e0b' },
199
- { value: 70, color: '#22c55e' },
200
- ],
201
- },
202
- description: `Status: ${threatLevel.toUpperCase()}`,
203
- },
204
- {
205
- id: 'blockedThreats',
206
- title: 'Blocked Threats',
207
- value: (metrics.blockedIPs?.length || 0) + metrics.spamDetected,
208
- type: 'number',
209
- icon: 'lucide:ShieldCheck',
210
- color: '#ef4444',
211
- description: 'Total threats blocked today',
212
- },
213
- {
214
- id: 'activeSessions',
215
- title: 'Active Sessions',
216
- value: recentAuthSuccesses,
217
- type: 'number',
218
- icon: 'lucide:Users',
219
- color: '#22c55e',
220
- description: 'Authenticated in last hour',
221
- },
222
- {
223
- id: 'authFailures',
224
- title: 'Auth Failures',
225
- value: metrics.authenticationFailures,
226
- type: 'number',
227
- icon: 'lucide:LockOpen',
228
- color: metrics.authenticationFailures > 10 ? '#ef4444' : '#f59e0b',
229
- description: 'Failed login attempts today',
230
- },
231
- ];
232
- return html `
233
- <dees-statsgrid
234
- .tiles=${tiles}
235
- .minTileWidth=${200}
236
- ></dees-statsgrid>
237
-
238
- <h2>Recent Security Events</h2>
239
- <dees-table
240
- .heading1=${'Security Events'}
241
- .heading2=${'Last 24 hours'}
242
- .data=${this.getSecurityEvents(metrics)}
243
- .displayFunction=${(item) => ({
244
- 'Time': new Date(item.timestamp).toLocaleTimeString(),
245
- 'Event': item.event,
246
- 'Severity': item.severity,
247
- 'Details': item.details,
248
- })}
249
- ></dees-table>
250
- `;
251
- }
252
- renderBlockedIPs(metrics) {
253
- const blockedIPs = metrics.blockedIPs || [];
254
- const tiles = [
255
- {
256
- id: 'totalBlocked',
257
- title: 'Blocked IPs',
258
- value: blockedIPs.length,
259
- type: 'number',
260
- icon: 'lucide:ShieldBan',
261
- color: blockedIPs.length > 0 ? '#ef4444' : '#22c55e',
262
- description: 'Currently blocked addresses',
263
- },
264
- ];
265
- return html `
266
- <dees-statsgrid
267
- .tiles=${tiles}
268
- .minTileWidth=${200}
269
- ></dees-statsgrid>
270
-
271
- <dees-table
272
- .heading1=${'Blocked IP Addresses'}
273
- .heading2=${'IPs blocked due to suspicious activity'}
274
- .data=${blockedIPs.map((ip) => ({ ip }))}
275
- .displayFunction=${(item) => ({
276
- 'IP Address': item.ip,
277
- 'Reason': 'Suspicious activity',
278
- })}
279
- .dataActions=${[
280
- {
281
- name: 'Unblock',
282
- iconName: 'lucide:shield-off',
283
- type: ['contextmenu'],
284
- actionFunc: async (item) => {
285
- await this.unblockIP(item.ip);
286
- },
287
- },
288
- {
289
- name: 'Clear All',
290
- iconName: 'lucide:trash-2',
291
- type: ['header'],
292
- actionFunc: async () => {
293
- await this.clearBlockedIPs();
294
- },
295
- },
296
- ]}
297
- ></dees-table>
298
- `;
299
- }
300
- renderAuthentication(metrics) {
301
- // Derive auth events from recentEvents
302
- const allEvents = metrics.recentEvents || [];
303
- const authEvents = allEvents.filter((evt) => evt.type === 'authentication');
304
- const successfulLogins = authEvents.filter((evt) => evt.success === true).length;
305
- const tiles = [
306
- {
307
- id: 'authFailures',
308
- title: 'Authentication Failures',
309
- value: metrics.authenticationFailures,
310
- type: 'number',
311
- icon: 'lucide:LockOpen',
312
- color: metrics.authenticationFailures > 10 ? '#ef4444' : '#f59e0b',
313
- description: 'Failed authentication attempts today',
314
- },
315
- {
316
- id: 'successfulLogins',
317
- title: 'Successful Logins',
318
- value: successfulLogins,
319
- type: 'number',
320
- icon: 'lucide:Lock',
321
- color: '#22c55e',
322
- description: 'Successful logins today',
323
- },
324
- ];
325
- // Map auth events to login history table data
326
- const loginHistory = authEvents.map((evt) => ({
327
- timestamp: evt.timestamp,
328
- username: evt.details?.username || 'unknown',
329
- ipAddress: evt.ipAddress || 'unknown',
330
- success: evt.success ?? false,
331
- reason: evt.success ? '' : evt.message || 'Authentication failed',
332
- }));
333
- return html `
334
- <dees-statsgrid
335
- .tiles=${tiles}
336
- .minTileWidth=${200}
337
- ></dees-statsgrid>
338
-
339
- <h2>Recent Login Attempts</h2>
340
- <dees-table
341
- .heading1=${'Login History'}
342
- .heading2=${'Recent authentication attempts'}
343
- .data=${loginHistory}
344
- .displayFunction=${(item) => ({
345
- 'Time': new Date(item.timestamp).toLocaleString(),
346
- 'Username': item.username,
347
- 'IP Address': item.ipAddress,
348
- 'Status': item.success ? 'Success' : 'Failed',
349
- 'Reason': item.reason || '-',
350
- })}
351
- ></dees-table>
352
- `;
353
- }
354
- renderEmailSecurity(metrics) {
355
- const tiles = [
356
- {
357
- id: 'malware',
358
- title: 'Malware Detection',
359
- value: metrics.malwareDetected,
360
- type: 'number',
361
- icon: 'lucide:BugOff',
362
- color: metrics.malwareDetected > 0 ? '#ef4444' : '#22c55e',
363
- description: 'Malware detected',
364
- },
365
- {
366
- id: 'phishing',
367
- title: 'Phishing Detection',
368
- value: metrics.phishingDetected,
369
- type: 'number',
370
- icon: 'lucide:Fish',
371
- color: metrics.phishingDetected > 0 ? '#ef4444' : '#22c55e',
372
- description: 'Phishing attempts detected',
373
- },
374
- {
375
- id: 'suspicious',
376
- title: 'Suspicious Activities',
377
- value: metrics.suspiciousActivities,
378
- type: 'number',
379
- icon: 'lucide:TriangleAlert',
380
- color: metrics.suspiciousActivities > 5 ? '#ef4444' : '#f59e0b',
381
- description: 'Suspicious activities detected',
382
- },
383
- {
384
- id: 'spam',
385
- title: 'Spam Detection',
386
- value: metrics.spamDetected,
387
- type: 'number',
388
- icon: 'lucide:Ban',
389
- color: '#f59e0b',
390
- description: 'Spam emails blocked',
391
- },
392
- ];
393
- return html `
394
- <dees-statsgrid
395
- .tiles=${tiles}
396
- .minTileWidth=${200}
397
- ></dees-statsgrid>
398
-
399
- <h2>Email Security Configuration</h2>
400
- <div class="securityCard">
401
- <dees-form>
402
- <dees-input-checkbox
403
- .key=${'enableSPF'}
404
- .label=${'Enable SPF checking'}
405
- .value=${true}
406
- ></dees-input-checkbox>
407
- <dees-input-checkbox
408
- .key=${'enableDKIM'}
409
- .label=${'Enable DKIM validation'}
410
- .value=${true}
411
- ></dees-input-checkbox>
412
- <dees-input-checkbox
413
- .key=${'enableDMARC'}
414
- .label=${'Enable DMARC policy enforcement'}
415
- .value=${true}
416
- ></dees-input-checkbox>
417
- <dees-input-checkbox
418
- .key=${'enableSpamFilter'}
419
- .label=${'Enable spam filtering'}
420
- .value=${true}
421
- ></dees-input-checkbox>
422
- </dees-form>
423
- <dees-button
424
- class="actionButton"
425
- type="highlighted"
426
- @click=${() => this.saveEmailSecuritySettings()}
427
- >
428
- Save Settings
429
- </dees-button>
430
- </div>
431
- `;
432
- }
433
- calculateThreatLevel(metrics) {
434
- const score = this.getThreatScore(metrics);
435
- if (score < 30)
436
- return 'alert';
437
- if (score < 70)
438
- return 'warning';
439
- return 'success';
440
- }
441
- getThreatScore(metrics) {
442
- // Simple scoring algorithm
443
- let score = 100;
444
- const blockedCount = Array.isArray(metrics.blockedIPs) ? metrics.blockedIPs.length : (metrics.blockedIPs || 0);
445
- score -= blockedCount * 2;
446
- score -= (metrics.authenticationFailures || 0) * 1;
447
- score -= (metrics.spamDetected || 0) * 0.5;
448
- score -= (metrics.malwareDetected || 0) * 3;
449
- score -= (metrics.phishingDetected || 0) * 3;
450
- score -= (metrics.suspiciousActivities || 0) * 2;
451
- return Math.max(0, Math.min(100, Math.round(score)));
452
- }
453
- getSecurityEvents(metrics) {
454
- const events = metrics.recentEvents || [];
455
- return events.map((evt) => ({
456
- timestamp: evt.timestamp,
457
- event: evt.message,
458
- severity: evt.level === 'critical' ? 'critical' : evt.level === 'error' ? 'high' : evt.level === 'warn' ? 'warning' : 'info',
459
- details: evt.ipAddress ? `IP: ${evt.ipAddress}` : evt.domain ? `Domain: ${evt.domain}` : evt.type,
460
- }));
461
- }
462
- async clearBlockedIPs() {
463
- // SmartProxy manages IP blocking — not yet exposed via API
464
- alert('Clearing blocked IPs is not yet supported from the UI.');
465
- }
466
- async unblockIP(ip) {
467
- // SmartProxy manages IP blocking — not yet exposed via API
468
- alert(`Unblocking IP ${ip} is not yet supported from the UI.`);
469
- }
470
- async saveEmailSecuritySettings() {
471
- // Config is read-only from the UI for now
472
- alert('Email security settings are read-only. Update the dcrouter configuration file to change these settings.');
473
- }
474
- static {
475
- __runInitializers(_classThis, _classExtraInitializers);
476
- }
477
- };
478
- return OpsViewSecurity = _classThis;
479
- })();
480
- export { OpsViewSecurity };
481
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoib3BzLXZpZXctc2VjdXJpdHkuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi90c193ZWIvZWxlbWVudHMvb3BzLXZpZXctc2VjdXJpdHkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQUFBLE9BQU8sS0FBSyxPQUFPLE1BQU0sZUFBZSxDQUFDO0FBQ3pDLE9BQU8sS0FBSyxNQUFNLE1BQU0sbUJBQW1CLENBQUM7QUFDNUMsT0FBTyxLQUFLLFFBQVEsTUFBTSxnQkFBZ0IsQ0FBQztBQUUzQyxPQUFPLEVBQ0wsV0FBVyxFQUNYLGFBQWEsRUFDYixJQUFJLEVBQ0osS0FBSyxFQUNMLEdBQUcsRUFDSCxVQUFVLEdBQ1gsTUFBTSw2QkFBNkIsQ0FBQztBQUNyQyxPQUFPLEVBQW1CLE1BQU0sNkJBQTZCLENBQUM7SUFHakQsZUFBZTs0QkFEM0IsYUFBYSxDQUFDLG1CQUFtQixDQUFDOzs7O3NCQUNFLFdBQVc7Ozs7Ozs7K0JBQW5CLFNBQVEsV0FBVzs7OztzQ0FDN0MsS0FBSyxFQUFFO3VDQWFQLEtBQUssRUFBRTtZQVpSLG1MQUFTLFVBQVUsNkJBQVYsVUFBVSwrRkFVakI7WUFHRixzTEFBUyxXQUFXLDZCQUFYLFdBQVcsaUdBQTRFO1lBZmxHLDZLQXFiQzs7OztRQW5iQyxpRkFBNEM7WUFDMUMsV0FBVyxFQUFFLElBQUk7WUFDakIsVUFBVSxFQUFFLElBQUk7WUFDaEIsUUFBUSxFQUFFLElBQUk7WUFDZCxlQUFlLEVBQUUsSUFBSTtZQUNyQixXQUFXLEVBQUUsSUFBSTtZQUNqQixRQUFRLEVBQUUsSUFBSTtZQUNkLFdBQVcsRUFBRSxDQUFDO1lBQ2QsU0FBUyxFQUFFLEtBQUs7WUFDaEIsS0FBSyxFQUFFLElBQUk7U0FDWixFQUFDO1FBVkYsSUFBUyxVQUFVLGdEQVVqQjtRQVZGLElBQVMsVUFBVSxzREFVakI7UUFHRiw0SUFBcUYsVUFBVSxHQUFDO1FBQWhHLElBQVMsV0FB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