@serve.zone/dcrouter 13.9.2 → 13.11.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 (56) hide show
  1. package/dist_serve/bundle.js +1306 -1180
  2. package/dist_ts/00_commitinfo_data.js +2 -2
  3. package/dist_ts/classes.dcrouter.d.ts +2 -0
  4. package/dist_ts/classes.dcrouter.js +15 -1
  5. package/dist_ts/db/documents/classes.email-domain.doc.d.ts +16 -0
  6. package/dist_ts/db/documents/classes.email-domain.doc.js +118 -0
  7. package/dist_ts/db/documents/index.d.ts +1 -0
  8. package/dist_ts/db/documents/index.js +3 -1
  9. package/dist_ts/email/classes.email-domain.manager.d.ts +45 -0
  10. package/dist_ts/email/classes.email-domain.manager.js +272 -0
  11. package/dist_ts/email/index.d.ts +1 -0
  12. package/dist_ts/email/index.js +2 -0
  13. package/dist_ts/opsserver/classes.opsserver.d.ts +1 -0
  14. package/dist_ts/opsserver/classes.opsserver.js +3 -1
  15. package/dist_ts/opsserver/handlers/email-domain.handler.d.ts +16 -0
  16. package/dist_ts/opsserver/handlers/email-domain.handler.js +149 -0
  17. package/dist_ts/opsserver/handlers/index.d.ts +1 -0
  18. package/dist_ts/opsserver/handlers/index.js +2 -1
  19. package/dist_ts_interfaces/data/email-domain.d.ts +68 -0
  20. package/dist_ts_interfaces/data/email-domain.js +2 -0
  21. package/dist_ts_interfaces/data/index.d.ts +1 -0
  22. package/dist_ts_interfaces/data/index.js +2 -1
  23. package/dist_ts_interfaces/requests/email-domains.d.ts +140 -0
  24. package/dist_ts_interfaces/requests/email-domains.js +2 -0
  25. package/dist_ts_interfaces/requests/index.d.ts +1 -0
  26. package/dist_ts_interfaces/requests/index.js +2 -1
  27. package/dist_ts_web/00_commitinfo_data.js +2 -2
  28. package/dist_ts_web/appstate.d.ts +20 -0
  29. package/dist_ts_web/appstate.js +81 -1
  30. package/dist_ts_web/elements/domains/ops-view-certificates.js +17 -96
  31. package/dist_ts_web/elements/email/index.d.ts +1 -0
  32. package/dist_ts_web/elements/email/index.js +2 -1
  33. package/dist_ts_web/elements/email/ops-view-email-domains.d.ts +19 -0
  34. package/dist_ts_web/elements/email/ops-view-email-domains.js +403 -0
  35. package/dist_ts_web/elements/email/ops-view-email-security.d.ts +1 -1
  36. package/dist_ts_web/elements/email/ops-view-email-security.js +39 -58
  37. package/dist_ts_web/elements/ops-dashboard.js +3 -1
  38. package/dist_ts_web/router.js +2 -2
  39. package/package.json +2 -2
  40. package/ts/00_commitinfo_data.ts +1 -1
  41. package/ts/classes.dcrouter.ts +17 -0
  42. package/ts/db/documents/classes.email-domain.doc.ts +53 -0
  43. package/ts/db/documents/index.ts +3 -0
  44. package/ts/email/classes.email-domain.manager.ts +316 -0
  45. package/ts/email/index.ts +1 -0
  46. package/ts/opsserver/classes.opsserver.ts +2 -0
  47. package/ts/opsserver/handlers/email-domain.handler.ts +194 -0
  48. package/ts/opsserver/handlers/index.ts +2 -1
  49. package/ts_web/00_commitinfo_data.ts +1 -1
  50. package/ts_web/appstate.ts +123 -0
  51. package/ts_web/elements/domains/ops-view-certificates.ts +16 -95
  52. package/ts_web/elements/email/index.ts +1 -0
  53. package/ts_web/elements/email/ops-view-email-domains.ts +389 -0
  54. package/ts_web/elements/email/ops-view-email-security.ts +38 -57
  55. package/ts_web/elements/ops-dashboard.ts +2 -0
  56. package/ts_web/router.ts +1 -1
@@ -0,0 +1,403 @@
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 { DeesElement, html, customElement, css, state, cssManager, } from '@design.estate/dees-element';
36
+ import * as appstate from '../../appstate.js';
37
+ import * as interfaces from '../../../dist_ts_interfaces/index.js';
38
+ import { viewHostCss } from '../shared/css.js';
39
+ import {} from '@design.estate/dees-catalog';
40
+ let OpsViewEmailDomains = (() => {
41
+ let _classDecorators = [customElement('ops-view-email-domains')];
42
+ let _classDescriptor;
43
+ let _classExtraInitializers = [];
44
+ let _classThis;
45
+ let _classSuper = DeesElement;
46
+ let _emailDomainsState_decorators;
47
+ let _emailDomainsState_initializers = [];
48
+ let _emailDomainsState_extraInitializers = [];
49
+ let _domainsState_decorators;
50
+ let _domainsState_initializers = [];
51
+ let _domainsState_extraInitializers = [];
52
+ var OpsViewEmailDomains = 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
+ _emailDomainsState_decorators = [state()];
57
+ _domainsState_decorators = [state()];
58
+ __esDecorate(this, null, _emailDomainsState_decorators, { kind: "accessor", name: "emailDomainsState", static: false, private: false, access: { has: obj => "emailDomainsState" in obj, get: obj => obj.emailDomainsState, set: (obj, value) => { obj.emailDomainsState = value; } }, metadata: _metadata }, _emailDomainsState_initializers, _emailDomainsState_extraInitializers);
59
+ __esDecorate(this, null, _domainsState_decorators, { kind: "accessor", name: "domainsState", static: false, private: false, access: { has: obj => "domainsState" in obj, get: obj => obj.domainsState, set: (obj, value) => { obj.domainsState = value; } }, metadata: _metadata }, _domainsState_initializers, _domainsState_extraInitializers);
60
+ __esDecorate(null, _classDescriptor = { value: _classThis }, _classDecorators, { kind: "class", name: _classThis.name, metadata: _metadata }, null, _classExtraInitializers);
61
+ OpsViewEmailDomains = _classThis = _classDescriptor.value;
62
+ if (_metadata) Object.defineProperty(_classThis, Symbol.metadata, { enumerable: true, configurable: true, writable: true, value: _metadata });
63
+ }
64
+ #emailDomainsState_accessor_storage = __runInitializers(this, _emailDomainsState_initializers, appstate.emailDomainsStatePart.getState());
65
+ get emailDomainsState() { return this.#emailDomainsState_accessor_storage; }
66
+ set emailDomainsState(value) { this.#emailDomainsState_accessor_storage = value; }
67
+ #domainsState_accessor_storage = (__runInitializers(this, _emailDomainsState_extraInitializers), __runInitializers(this, _domainsState_initializers, appstate.domainsStatePart.getState()));
68
+ get domainsState() { return this.#domainsState_accessor_storage; }
69
+ set domainsState(value) { this.#domainsState_accessor_storage = value; }
70
+ constructor() {
71
+ super();
72
+ __runInitializers(this, _domainsState_extraInitializers);
73
+ const sub = appstate.emailDomainsStatePart.select().subscribe((s) => {
74
+ this.emailDomainsState = s;
75
+ });
76
+ this.rxSubscriptions.push(sub);
77
+ const domSub = appstate.domainsStatePart.select().subscribe((s) => {
78
+ this.domainsState = s;
79
+ });
80
+ this.rxSubscriptions.push(domSub);
81
+ }
82
+ async connectedCallback() {
83
+ await super.connectedCallback();
84
+ await appstate.emailDomainsStatePart.dispatchAction(appstate.fetchEmailDomainsAction, null);
85
+ await appstate.domainsStatePart.dispatchAction(appstate.fetchDomainsAndProvidersAction, null);
86
+ }
87
+ static styles = [
88
+ cssManager.defaultStyles,
89
+ viewHostCss,
90
+ css `
91
+ .emailDomainsContainer {
92
+ display: flex;
93
+ flex-direction: column;
94
+ gap: 24px;
95
+ }
96
+
97
+ .statusBadge {
98
+ display: inline-flex;
99
+ align-items: center;
100
+ padding: 3px 10px;
101
+ border-radius: 12px;
102
+ font-size: 12px;
103
+ font-weight: 600;
104
+ letter-spacing: 0.02em;
105
+ text-transform: uppercase;
106
+ }
107
+
108
+ .statusBadge.valid {
109
+ background: ${cssManager.bdTheme('#dcfce7', '#14532d')};
110
+ color: ${cssManager.bdTheme('#166534', '#4ade80')};
111
+ }
112
+
113
+ .statusBadge.missing {
114
+ background: ${cssManager.bdTheme('#fef2f2', '#450a0a')};
115
+ color: ${cssManager.bdTheme('#991b1b', '#f87171')};
116
+ }
117
+
118
+ .statusBadge.invalid {
119
+ background: ${cssManager.bdTheme('#fff7ed', '#431407')};
120
+ color: ${cssManager.bdTheme('#9a3412', '#fb923c')};
121
+ }
122
+
123
+ .statusBadge.unchecked {
124
+ background: ${cssManager.bdTheme('#f3f4f6', '#1f2937')};
125
+ color: ${cssManager.bdTheme('#4b5563', '#9ca3af')};
126
+ }
127
+
128
+ .sourceBadge {
129
+ display: inline-flex;
130
+ align-items: center;
131
+ padding: 3px 8px;
132
+ border-radius: 4px;
133
+ font-size: 11px;
134
+ font-weight: 500;
135
+ background: ${cssManager.bdTheme('#f3f4f6', '#1f2937')};
136
+ color: ${cssManager.bdTheme('#374151', '#d1d5db')};
137
+ }
138
+ `,
139
+ ];
140
+ render() {
141
+ const domains = this.emailDomainsState.domains;
142
+ const validCount = domains.filter((d) => d.dnsStatus.mx === 'valid' &&
143
+ d.dnsStatus.spf === 'valid' &&
144
+ d.dnsStatus.dkim === 'valid' &&
145
+ d.dnsStatus.dmarc === 'valid').length;
146
+ const issueCount = domains.length - validCount;
147
+ const tiles = [
148
+ {
149
+ id: 'total',
150
+ title: 'Total Domains',
151
+ value: domains.length,
152
+ type: 'number',
153
+ icon: 'lucide:globe',
154
+ color: '#3b82f6',
155
+ },
156
+ {
157
+ id: 'valid',
158
+ title: 'Valid DNS',
159
+ value: validCount,
160
+ type: 'number',
161
+ icon: 'lucide:Check',
162
+ color: '#22c55e',
163
+ },
164
+ {
165
+ id: 'issues',
166
+ title: 'Issues',
167
+ value: issueCount,
168
+ type: 'number',
169
+ icon: 'lucide:TriangleAlert',
170
+ color: issueCount > 0 ? '#ef4444' : '#22c55e',
171
+ },
172
+ {
173
+ id: 'dkim',
174
+ title: 'DKIM Active',
175
+ value: domains.filter((d) => d.dkim.publicKey).length,
176
+ type: 'number',
177
+ icon: 'lucide:KeyRound',
178
+ color: '#8b5cf6',
179
+ },
180
+ ];
181
+ return html `
182
+ <dees-heading level="3">Email Domains</dees-heading>
183
+
184
+ <div class="emailDomainsContainer">
185
+ <dees-statsgrid
186
+ .tiles=${tiles}
187
+ .minTileWidth=${200}
188
+ .gridActions=${[
189
+ {
190
+ name: 'Refresh',
191
+ iconName: 'lucide:RefreshCw',
192
+ action: async () => {
193
+ await appstate.emailDomainsStatePart.dispatchAction(appstate.fetchEmailDomainsAction, null);
194
+ },
195
+ },
196
+ ]}
197
+ ></dees-statsgrid>
198
+
199
+ <dees-table
200
+ .heading1=${'Email Domains'}
201
+ .heading2=${'DKIM, SPF, DMARC and MX management'}
202
+ .data=${domains}
203
+ .showColumnFilters=${true}
204
+ .displayFunction=${(d) => ({
205
+ Domain: d.domain,
206
+ Source: this.renderSourceBadge(d.linkedDomainId),
207
+ MX: this.renderDnsStatus(d.dnsStatus.mx),
208
+ SPF: this.renderDnsStatus(d.dnsStatus.spf),
209
+ DKIM: this.renderDnsStatus(d.dnsStatus.dkim),
210
+ DMARC: this.renderDnsStatus(d.dnsStatus.dmarc),
211
+ })}
212
+ .dataActions=${[
213
+ {
214
+ name: 'Add Email Domain',
215
+ iconName: 'lucide:plus',
216
+ type: ['header'],
217
+ actionFunc: async () => {
218
+ await this.showCreateDialog();
219
+ },
220
+ },
221
+ {
222
+ name: 'Validate DNS',
223
+ iconName: 'lucide:search-check',
224
+ type: ['inRow', 'contextmenu'],
225
+ actionFunc: async (actionData) => {
226
+ const d = actionData.item;
227
+ await appstate.emailDomainsStatePart.dispatchAction(appstate.validateEmailDomainAction, d.id);
228
+ const { DeesToast } = await import('@design.estate/dees-catalog');
229
+ DeesToast.show({ message: `DNS validated for ${d.domain}`, type: 'success', duration: 2500 });
230
+ },
231
+ },
232
+ {
233
+ name: 'Provision DNS',
234
+ iconName: 'lucide:wand-sparkles',
235
+ type: ['inRow', 'contextmenu'],
236
+ actionFunc: async (actionData) => {
237
+ const d = actionData.item;
238
+ await appstate.emailDomainsStatePart.dispatchAction(appstate.provisionEmailDomainDnsAction, d.id);
239
+ const { DeesToast } = await import('@design.estate/dees-catalog');
240
+ DeesToast.show({ message: `DNS records provisioned for ${d.domain}`, type: 'success', duration: 2500 });
241
+ },
242
+ },
243
+ {
244
+ name: 'View DNS Records',
245
+ iconName: 'lucide:list',
246
+ type: ['inRow', 'contextmenu'],
247
+ actionFunc: async (actionData) => {
248
+ const d = actionData.item;
249
+ await this.showDnsRecordsDialog(d);
250
+ },
251
+ },
252
+ {
253
+ name: 'Delete',
254
+ iconName: 'lucide:trash2',
255
+ type: ['inRow', 'contextmenu'],
256
+ actionFunc: async (actionData) => {
257
+ const d = actionData.item;
258
+ await appstate.emailDomainsStatePart.dispatchAction(appstate.deleteEmailDomainAction, d.id);
259
+ },
260
+ },
261
+ ]}
262
+ dataName="email domain"
263
+ ></dees-table>
264
+ </div>
265
+ `;
266
+ }
267
+ renderDnsStatus(status) {
268
+ return html `<span class="statusBadge ${status}">${status}</span>`;
269
+ }
270
+ renderSourceBadge(linkedDomainId) {
271
+ const domain = this.domainsState.domains.find((d) => d.id === linkedDomainId);
272
+ if (!domain)
273
+ return html `<span class="sourceBadge">unknown</span>`;
274
+ const label = domain.source === 'dcrouter'
275
+ ? 'dcrouter'
276
+ : this.domainsState.providers.find((p) => p.id === domain.providerId)?.name || 'provider';
277
+ return html `<span class="sourceBadge">${label}</span>`;
278
+ }
279
+ async showCreateDialog() {
280
+ const { DeesModal } = await import('@design.estate/dees-catalog');
281
+ const domainOptions = this.domainsState.domains.map((d) => ({
282
+ option: `${d.name} (${d.source})`,
283
+ key: d.id,
284
+ }));
285
+ DeesModal.createAndShow({
286
+ heading: 'Add Email Domain',
287
+ content: html `
288
+ <dees-form>
289
+ <dees-input-dropdown
290
+ .key=${'linkedDomainId'}
291
+ .label=${'Domain'}
292
+ .description=${'Select an existing DNS domain'}
293
+ .options=${domainOptions}
294
+ .required=${true}
295
+ ></dees-input-dropdown>
296
+ <dees-input-text
297
+ .key=${'dkimSelector'}
298
+ .label=${'DKIM Selector'}
299
+ .description=${'Identifier used in DNS record name'}
300
+ .value=${'default'}
301
+ ></dees-input-text>
302
+ <dees-input-dropdown
303
+ .key=${'dkimKeySize'}
304
+ .label=${'DKIM Key Size'}
305
+ .options=${[
306
+ { option: '2048 (recommended)', key: '2048' },
307
+ { option: '1024', key: '1024' },
308
+ { option: '4096', key: '4096' },
309
+ ]}
310
+ .selectedOption=${{ option: '2048 (recommended)', key: '2048' }}
311
+ ></dees-input-dropdown>
312
+ <dees-input-checkbox
313
+ .key=${'rotateKeys'}
314
+ .label=${'Auto-rotate DKIM keys'}
315
+ .value=${false}
316
+ ></dees-input-checkbox>
317
+ </dees-form>
318
+ `,
319
+ menuOptions: [
320
+ { name: 'Cancel', action: async (m) => m.destroy() },
321
+ {
322
+ name: 'Create',
323
+ action: async (m) => {
324
+ const form = m.shadowRoot?.querySelector('.content')?.querySelector('dees-form');
325
+ if (!form)
326
+ return;
327
+ const data = await form.collectFormData();
328
+ const linkedDomainId = typeof data.linkedDomainId === 'object'
329
+ ? data.linkedDomainId.key
330
+ : data.linkedDomainId;
331
+ const keySize = typeof data.dkimKeySize === 'object'
332
+ ? parseInt(data.dkimKeySize.key, 10)
333
+ : parseInt(data.dkimKeySize || '2048', 10);
334
+ await appstate.emailDomainsStatePart.dispatchAction(appstate.createEmailDomainAction, {
335
+ linkedDomainId,
336
+ dkimSelector: data.dkimSelector || 'default',
337
+ dkimKeySize: keySize,
338
+ rotateKeys: Boolean(data.rotateKeys),
339
+ });
340
+ m.destroy();
341
+ },
342
+ },
343
+ ],
344
+ });
345
+ }
346
+ async showDnsRecordsDialog(emailDomain) {
347
+ const { DeesModal, DeesToast } = await import('@design.estate/dees-catalog');
348
+ // Fetch required DNS records
349
+ let records = [];
350
+ try {
351
+ const response = await appstate.fetchEmailDomainDnsRecords(emailDomain.id);
352
+ records = response.records;
353
+ }
354
+ catch {
355
+ records = [];
356
+ }
357
+ DeesModal.createAndShow({
358
+ heading: `DNS Records: ${emailDomain.domain}`,
359
+ content: html `
360
+ <dees-table
361
+ .data=${records}
362
+ .displayFunction=${(r) => ({
363
+ Type: r.type,
364
+ Name: r.name,
365
+ Value: r.value,
366
+ Status: html `<span class="statusBadge ${r.status}">${r.status}</span>`,
367
+ })}
368
+ .dataActions=${[
369
+ {
370
+ name: 'Copy Value',
371
+ iconName: 'lucide:copy',
372
+ type: ['inRow'],
373
+ actionFunc: async (actionData) => {
374
+ const rec = actionData.item;
375
+ await navigator.clipboard.writeText(rec.value);
376
+ DeesToast.show({ message: 'Copied to clipboard', type: 'success', duration: 1500 });
377
+ },
378
+ },
379
+ ]}
380
+ dataName="DNS record"
381
+ ></dees-table>
382
+ `,
383
+ menuOptions: [
384
+ {
385
+ name: 'Auto-Provision All',
386
+ action: async (m) => {
387
+ await appstate.emailDomainsStatePart.dispatchAction(appstate.provisionEmailDomainDnsAction, emailDomain.id);
388
+ DeesToast.show({ message: 'DNS records provisioned', type: 'success', duration: 2500 });
389
+ m.destroy();
390
+ },
391
+ },
392
+ { name: 'Close', action: async (m) => m.destroy() },
393
+ ],
394
+ });
395
+ }
396
+ static {
397
+ __runInitializers(_classThis, _classExtraInitializers);
398
+ }
399
+ };
400
+ return OpsViewEmailDomains = _classThis;
401
+ })();
402
+ export { OpsViewEmailDomains };
403
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoib3BzLXZpZXctZW1haWwtZG9tYWlucy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3RzX3dlYi9lbGVtZW50cy9lbWFpbC9vcHMtdmlldy1lbWFpbC1kb21haW5zLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQSxPQUFPLEVBQ0wsV0FBVyxFQUNYLElBQUksRUFDSixhQUFhLEVBRWIsR0FBRyxFQUNILEtBQUssRUFDTCxVQUFVLEdBQ1gsTUFBTSw2QkFBNkIsQ0FBQztBQUNyQyxPQUFPLEtBQUssUUFBUSxNQUFNLG1CQUFtQixDQUFDO0FBQzlDLE9BQU8sS0FBSyxVQUFVLE1BQU0sc0NBQXNDLENBQUM7QUFDbkUsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLGtCQUFrQixDQUFDO0FBQy9DLE9BQU8sRUFBbUIsTUFBTSw2QkFBNkIsQ0FBQztJQVNqRCxtQkFBbUI7NEJBRC9CLGFBQWEsQ0FBQyx3QkFBd0IsQ0FBQzs7OztzQkFDQyxXQUFXOzs7Ozs7O21DQUFuQixTQUFRLFdBQVc7Ozs7NkNBQ2pELEtBQUssRUFBRTt3Q0FJUCxLQUFLLEVBQUU7WUFIUix3TUFBUyxpQkFBaUIsNkJBQWpCLGlCQUFpQiw2R0FDbUI7WUFHN0MseUxBQVMsWUFBWSw2QkFBWixZQUFZLG1HQUFpRTtZQU54Riw2S0ErV0M7Ozs7UUE3V0MsK0ZBQ0UsUUFBUSxDQUFDLHFCQUFxQixDQUFDLFFBQVEsRUFBRyxFQUFDO1FBRDdDLElBQVMsaUJBQWlCLHVEQUNtQjtRQUQ3QyxJQUFTLGlCQUFpQiw2REFDbUI7UUFHN0MscUpBQWdELFFBQVEsQ0FBQyxnQkFBZ0IsQ0FBQyxRQUFRLEVBQUcsR0FBQztRQUF0RixJQUFTLFlBQVksa0RBQWlFO1FBQXRGLElBQVMsWUFBWSx3REFBaUU7UUFFdEY7WUFDRSxLQUFLLEVBQUUsQ0FBQzs7WUFDUixNQUFNLEdBQUcsR0FBRyxRQUFRLENBQUMscUJB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
@@ -10,5 +10,5 @@ export declare class OpsViewEmailSecurity extends DeesElement {
10
10
  constructor();
11
11
  static styles: import("@design.estate/dees-element").CSSResult[];
12
12
  render(): TemplateResult;
13
- private saveEmailSecuritySettings;
13
+ private showEditSecurityDialog;
14
14
  }
@@ -72,25 +72,10 @@ let OpsViewEmailSecurity = (() => {
72
72
  cssManager.defaultStyles,
73
73
  viewHostCss,
74
74
  css `
75
- h2 {
76
- margin: 32px 0 16px 0;
77
- font-size: 24px;
78
- font-weight: 600;
79
- color: ${cssManager.bdTheme('#333', '#ccc')};
80
- }
81
- dees-statsgrid {
82
- margin-bottom: 32px;
83
- }
84
- .securityCard {
85
- background: ${cssManager.bdTheme('#fff', '#222')};
86
- border: 1px solid ${cssManager.bdTheme('#e9ecef', '#333')};
87
- border-radius: 8px;
88
- padding: 24px;
89
- position: relative;
90
- overflow: hidden;
91
- }
92
- .actionButton {
93
- margin-top: 16px;
75
+ .securityContainer {
76
+ display: flex;
77
+ flex-direction: column;
78
+ gap: 24px;
94
79
  }
95
80
  `,
96
81
  ];
@@ -144,48 +129,44 @@ let OpsViewEmailSecurity = (() => {
144
129
  return html `
145
130
  <dees-heading level="3">Email Security</dees-heading>
146
131
 
147
- <dees-statsgrid
148
- .tiles=${tiles}
149
- .minTileWidth=${200}
150
- ></dees-statsgrid>
132
+ <div class="securityContainer">
133
+ <dees-statsgrid
134
+ .tiles=${tiles}
135
+ .minTileWidth=${200}
136
+ ></dees-statsgrid>
151
137
 
152
- <h2>Email Security Configuration</h2>
153
- <div class="securityCard">
154
- <dees-form>
155
- <dees-input-checkbox
156
- .key=${'enableSPF'}
157
- .label=${'Enable SPF checking'}
158
- .value=${true}
159
- ></dees-input-checkbox>
160
- <dees-input-checkbox
161
- .key=${'enableDKIM'}
162
- .label=${'Enable DKIM validation'}
163
- .value=${true}
164
- ></dees-input-checkbox>
165
- <dees-input-checkbox
166
- .key=${'enableDMARC'}
167
- .label=${'Enable DMARC policy enforcement'}
168
- .value=${true}
169
- ></dees-input-checkbox>
170
- <dees-input-checkbox
171
- .key=${'enableSpamFilter'}
172
- .label=${'Enable spam filtering'}
173
- .value=${true}
174
- ></dees-input-checkbox>
175
- </dees-form>
176
- <dees-button
177
- class="actionButton"
178
- type="highlighted"
179
- @click=${() => this.saveEmailSecuritySettings()}
180
- >
181
- Save Settings
182
- </dees-button>
138
+ <dees-settings
139
+ .heading=${'Security Configuration'}
140
+ .settingsFields=${[
141
+ { key: 'spf', label: 'SPF checking', value: 'enabled' },
142
+ { key: 'dkim', label: 'DKIM validation', value: 'enabled' },
143
+ { key: 'dmarc', label: 'DMARC policy', value: 'enabled' },
144
+ { key: 'spam', label: 'Spam filtering', value: 'enabled' },
145
+ ]}
146
+ .actions=${[{ name: 'Edit', action: () => this.showEditSecurityDialog() }]}
147
+ ></dees-settings>
183
148
  </div>
184
149
  `;
185
150
  }
186
- async saveEmailSecuritySettings() {
187
- // Config is read-only from the UI for now
188
- alert('Email security settings are read-only. Update the dcrouter configuration file to change these settings.');
151
+ async showEditSecurityDialog() {
152
+ const { DeesModal } = await import('@design.estate/dees-catalog');
153
+ DeesModal.createAndShow({
154
+ heading: 'Edit Security Configuration',
155
+ content: html `
156
+ <dees-form>
157
+ <dees-input-checkbox .key=${'enableSPF'} .label=${'SPF checking'} .value=${true}></dees-input-checkbox>
158
+ <dees-input-checkbox .key=${'enableDKIM'} .label=${'DKIM validation'} .value=${true}></dees-input-checkbox>
159
+ <dees-input-checkbox .key=${'enableDMARC'} .label=${'DMARC policy enforcement'} .value=${true}></dees-input-checkbox>
160
+ <dees-input-checkbox .key=${'enableSpamFilter'} .label=${'Spam filtering'} .value=${true}></dees-input-checkbox>
161
+ </dees-form>
162
+ <p style="margin-top: 12px; font-size: 12px; opacity: 0.7;">
163
+ These settings are read-only for now. Update the dcrouter configuration to change them.
164
+ </p>
165
+ `,
166
+ menuOptions: [
167
+ { name: 'Close', action: async (modalArg) => modalArg.destroy() },
168
+ ],
169
+ });
189
170
  }
190
171
  static {
191
172
  __runInitializers(_classThis, _classExtraInitializers);
@@ -194,4 +175,4 @@ let OpsViewEmailSecurity = (() => {
194
175
  return OpsViewEmailSecurity = _classThis;
195
176
  })();
196
177
  export { OpsViewEmailSecurity };
197
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoib3BzLXZpZXctZW1haWwtc2VjdXJpdHkuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi90c193ZWIvZWxlbWVudHMvZW1haWwvb3BzLXZpZXctZW1haWwtc2VjdXJpdHkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQUFBLE9BQU8sS0FBSyxRQUFRLE1BQU0sbUJBQW1CLENBQUM7QUFDOUMsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLGtCQUFrQixDQUFDO0FBRS9DLE9BQU8sRUFDTCxXQUFXLEVBQ1gsYUFBYSxFQUNiLElBQUksRUFDSixLQUFLLEVBQ0wsR0FBRyxFQUNILFVBQVUsR0FFWCxNQUFNLDZCQUE2QixDQUFDO0FBQ3JDLE9BQU8sRUFBbUIsTUFBTSw2QkFBNkIsQ0FBQztJQVNqRCxvQkFBb0I7NEJBRGhDLGFBQWEsQ0FBQyx5QkFBeUIsQ0FBQzs7OztzQkFDQyxXQUFXOzs7O29DQUFuQixTQUFRLFdBQVc7Ozs7c0NBQ2xELEtBQUssRUFBRTtZQUNSLG1MQUFTLFVBQVUsNkJBQVYsVUFBVSwrRkFBNkQ7WUFGbEYsNktBMElDOzs7O1FBeElDLGlGQUE0QyxRQUFRLENBQUMsY0FBYyxDQUFDLFFBQVEsRUFBRyxFQUFDO1FBQWhGLElBQVMsVUFBVSxnREFBNkQ7UUFBaEYsSUFBUyxVQUFVLHNEQUE2RDtRQUVoRjtZQUNFLEtBQUssRUFBRSxDQUFDOztZQUNSLE1BQU0sR0FBRyxHQUFHLFFBQVEsQ0FBQyxjQUFjO2lCQUNoQyxNQUFNLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQztpQkFDaEIsU0FBUyxDQUFDLENBQUMsQ0FBQyxFQUFFLEVBQUU7Z0JBQ2YsSUFBSSxDQUFDLFVBQVUsR0FBRyxDQUFDLENBQUM7WUFDdEIsQ0FBQyxDQUFDLENBQUM7WUFDTCxJQUFJLENBQUMsZUFBZSxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsQ0FB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
178
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoib3BzLXZpZXctZW1haWwtc2VjdXJpdHkuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi90c193ZWIvZWxlbWVudHMvZW1haWwvb3BzLXZpZXctZW1haWwtc2VjdXJpdHkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQUFBLE9BQU8sS0FBSyxRQUFRLE1BQU0sbUJBQW1CLENBQUM7QUFDOUMsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLGtCQUFrQixDQUFDO0FBRS9DLE9BQU8sRUFDTCxXQUFXLEVBQ1gsYUFBYSxFQUNiLElBQUksRUFDSixLQUFLLEVBQ0wsR0FBRyxFQUNILFVBQVUsR0FFWCxNQUFNLDZCQUE2QixDQUFDO0FBQ3JDLE9BQU8sRUFBbUIsTUFBTSw2QkFBNkIsQ0FBQztJQVNqRCxvQkFBb0I7NEJBRGhDLGFBQWEsQ0FBQyx5QkFBeUIsQ0FBQzs7OztzQkFDQyxXQUFXOzs7O29DQUFuQixTQUFRLFdBQVc7Ozs7c0NBQ2xELEtBQUssRUFBRTtZQUNSLG1MQUFTLFVBQVUsNkJBQVYsVUFBVSwrRkFBNkQ7WUFGbEYsNktBdUhDOzs7O1FBckhDLGlGQUE0QyxRQUFRLENBQUMsY0FBYyxDQUFDLFFBQVEsRUFBRyxFQUFDO1FBQWhGLElBQVMsVUFBVSxnREFBNkQ7UUFBaEYsSUFBUyxVQUFVLHNEQUE2RDtRQUVoRjtZQUNFLEtBQUssRUFBRSxDQUFDOztZQUNSLE1BQU0sR0FBRyxHQUFHLFFBQVEsQ0FBQyxjQUFjO2lCQUNoQyxNQUFNLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQztpQkFDaEIsU0FBUyxDQUFDLENBQUMsQ0FBQyxFQUFFLEVBQUU7Z0JBQ2YsSUFBSSxDQUFDLFVBQVUsR0FBRyxDQUFDLENBQUM7WUFDdEIsQ0FBQyxDQUFDLENBQUM7WUFDTCxJQUFJLENBQUMsZUFBZSxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsQ0FB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