web-mojo 2.2.71 → 2.2.72

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.
@@ -1 +1 @@
1
- {"version":3,"file":"admin.es.js","sources":["../src/extensions/admin/account/AdminDashboardPage.js","../src/core/views/navigation/SideNavView.js","../src/extensions/admin/account/users/sections/AdminProfileSection.js","../src/extensions/admin/account/users/sections/AdminPersonalSection.js","../src/extensions/admin/account/users/sections/AdminSecuritySection.js","../src/extensions/admin/account/users/sections/AdminConnectedSection.js","../src/extensions/admin/account/users/sections/AdminNotificationsSection.js","../src/extensions/admin/account/users/sections/AdminApiKeysSection.js","../src/extensions/admin/account/users/UserView.js","../src/extensions/admin/account/users/UserTablePage.js","../src/extensions/admin/account/users/MemberView.js","../src/extensions/admin/account/users/MemberTablePage.js","../src/extensions/admin/account/groups/GroupView.js","../src/extensions/admin/account/groups/GroupTablePage.js","../src/extensions/admin/account/devices/DeviceView.js","../src/extensions/admin/account/devices/UserDeviceTablePage.js","../src/extensions/admin/account/devices/UserDeviceLocationTablePage.js","../src/extensions/admin/account/devices/GeoIPView.js","../src/extensions/admin/account/devices/GeoLocatedIPTablePage.js","../src/core/models/ApiKey.js","../src/extensions/admin/account/api_keys/ApiKeyView.js","../src/extensions/admin/account/api_keys/ApiKeyTablePage.js","../src/extensions/admin/aws/CloudWatchChart.js","../src/extensions/admin/aws/CloudWatchDashboardPage.js","../src/extensions/admin/incidents/IncidentDashboardPage.js","../src/core/views/data/StackTraceView.js","../src/extensions/admin/incidents/adapters/IncidentHistoryAdapter.js","../src/extensions/admin/incidents/IncidentView.js","../src/extensions/admin/incidents/IncidentTablePage.js","../src/extensions/admin/incidents/EventView.js","../src/extensions/admin/incidents/EventTablePage.js","../src/extensions/admin/incidents/adapters/TicketNoteAdapter.js","../src/extensions/admin/incidents/TicketView.js","../src/extensions/admin/incidents/TicketTablePage.js","../src/extensions/admin/incidents/RuleSetView.js","../src/extensions/admin/incidents/RuleSetTablePage.js","../src/extensions/admin/messaging/email/EmailDomainTablePage.js","../src/extensions/admin/messaging/email/EmailMailboxTablePage.js","../src/extensions/admin/messaging/email/EmailTemplateView.js","../src/extensions/admin/messaging/email/EmailTemplateTablePage.js","../src/extensions/admin/messaging/email/EmailView.js","../src/extensions/admin/messaging/email/SentMessageTablePage.js","../src/core/models/Phonehub.js","../src/extensions/admin/messaging/sms/PhoneNumberView.js","../src/extensions/admin/messaging/sms/PhoneNumberTablePage.js","../src/extensions/admin/messaging/sms/SMSView.js","../src/extensions/admin/messaging/sms/SMSTablePage.js","../src/extensions/admin/messaging/push/PushDashboardPage.js","../src/extensions/admin/messaging/push/PushConfigTablePage.js","../src/extensions/admin/messaging/push/PushTemplateTablePage.js","../src/extensions/admin/messaging/push/PushDeliveryView.js","../src/extensions/admin/messaging/push/PushDeliveryTablePage.js","../src/extensions/admin/messaging/push/PushDeviceView.js","../src/extensions/admin/messaging/push/PushDeviceTablePage.js","../src/extensions/admin/jobs/JobStatsView.js","../src/extensions/admin/jobs/JobHealthView.js","../src/extensions/admin/jobs/JobDetailsView.js","../src/extensions/admin/jobs/JobsAdminPage.js","../src/extensions/admin/jobs/TaskDetailsView.js","../src/extensions/admin/jobs/RunnerDetailsView.js","../src/extensions/admin/jobs/TaskManagementPage.js","../src/extensions/admin/monitoring/LogView.js","../src/extensions/admin/monitoring/LogTablePage.js","../src/extensions/admin/monitoring/MetricsPermissionsView.js","../src/extensions/admin/monitoring/MetricsPermissionsTablePage.js","../src/core/models/Settings.js","../src/extensions/admin/settings/SettingView.js","../src/extensions/admin/settings/SettingTablePage.js","../src/extensions/admin/storage/FileManagerTablePage.js","../src/extensions/admin/storage/FileView.js","../src/extensions/admin/storage/FileTablePage.js","../src/extensions/admin/storage/S3BucketTablePage.js","../src/extensions/admin/account/devices/UserDeviceLocationView.js","../src/extensions/admin/aws/CloudWatchResourceView.js","../src/admin.js"],"sourcesContent":["/**\n * AdminDashboardPage - Administrative dashboard with system metrics and charts\n */\n\nimport Page from '@core/Page.js';\nimport View from '@core/View.js';\nimport { MetricsChart, MetricsMiniChart, MetricsMiniChartWidget } from '@ext/charts/index.js';\n\n// Embedded HeaderView for dashboard statistics\nclass AdminHeaderView extends View {\n constructor(options = {}) {\n super({\n title: 'Dashboard',\n ...options,\n headerActions: [\n {\n label: 'Export',\n icon: 'bi-download',\n action: 'export',\n buttonClass: 'btn-primary'\n }\n ],\n className: 'admin-header-section'\n });\n\n // Mock data - replace with real API calls\n this.stats = {\n user_activity_day: 0,\n total_users: 0,\n group_activity_day: 0,\n total_groups: 0,\n api_calls: 0,\n apiChange: '',\n incidents: 0,\n incidentsChange: ''\n };\n\n // Prepare formatted data for template\n this.prepareStatsForTemplate();\n }\n\n async getTemplate() {\n return `\n <div class=\"admin-stats-header mb-4\">\n <div class=\"row\">\n <div class=\"col-xl-3 col-lg-6 col-12 mb-3\">\n <div data-container=\"user_activity_day\"></div>\n </div>\n\n <div class=\"col-xl-3 col-lg-6 col-12 mb-3\">\n <div data-container=\"group_activity_day\"></div>\n </div>\n\n <div class=\"col-xl-3 col-lg-6 col-12 mb-3\">\n <div data-container=\"api_activity_day\"></div>\n </div>\n\n <div class=\"col-xl-3 col-lg-6 col-12 mb-3\">\n <div data-container=\"incident_activity_day\"></div>\n </div>\n </div>\n </div>\n `;\n }\n\n async onInit() {\n // TODO: Replace with actual API calls to fetch real statistics\n // this.loadStats();\n // slug: user_activity_day, group_activity_day\n this.userActivity = new MetricsMiniChartWidget({\n icon: \"bi bi-people fs-2\",\n title: 'User Activity',\n subtitle: '{{now_value}} <span class=\"subtitle-label\">{{now_label}}</span> {{total_users}} <span class=\"subtitle-label\">Total</span>',\n background: \"#5388D6\",\n textColor: \"#FFFFFF\",\n granularity: 'days',\n trendRange: 4,\n trendOffset: 0,\n slugs: ['user_activity_day'],\n account: 'global',\n chartType: 'bar',\n showTooltip: true,\n showXAxis: true,\n height: 50,\n chartWidth: '100%',\n color: 'rgba(245, 245, 255, 0.8)',\n fill: true,\n fillColor: 'rgba(245, 245, 255, 0.6)',\n smoothing: 0.3,\n showTrending: true,\n showSettings: true,\n showDateRange: true,\n containerId: 'user_activity_day'\n });\n this.addChild(this.userActivity);\n\n this.groupActivity = new MetricsMiniChartWidget({\n icon: \"bi bi-collection fs-2\",\n title: 'Group Activity',\n subtitle: '{{now_value}} <span class=\"subtitle-label\">{{now_label}}</span> {{total_groups}} <span class=\"subtitle-label\">Total</span>',\n background: \"#1f6a7a\",\n textColor: \"#FFFFFF\",\n granularity: 'days',\n trendRange: 4,\n trendOffset: 0,\n slugs: ['group_activity_day'],\n account: 'global',\n chartType: 'bar',\n showTooltip: true,\n showXAxis: true,\n height: 50,\n chartWidth: '100%',\n color: 'rgba(245, 245, 255, 0.8)',\n fill: true,\n fillColor: 'rgba(245, 245, 255, 0.6)',\n smoothing: 0.3,\n showTrending: true,\n containerId: 'group_activity_day'\n });\n this.addChild(this.groupActivity);\n\n this.apiActivity = new MetricsMiniChartWidget({\n icon: \"bi bi-graph-up fs-2\",\n title: 'API Requests',\n subtitle: '{{now_value}} <span class=\"subtitle-label\">{{now_label}}</span> {{total}} <span class=\"subtitle-label\">Total</span>',\n background: \"#50A079\",\n textColor: \"#FFFFFF\",\n endpoint: '/api/metrics/fetch',\n trendRange: 4,\n trendOffset: 0,\n granularity: 'days',\n slugs: ['api_calls'],\n account: 'global',\n chartType: 'line',\n showTooltip: true,\n showXAxis: true,\n height: 50,\n chartWidth: '100%',\n color: 'rgba(245, 245, 255, 0.8)',\n fill: true,\n fillColor: 'rgba(245, 245, 255, 0.6)',\n smoothing: 0.3,\n showTrending: true,\n containerId: 'api_activity_day'\n });\n this.addChild(this.apiActivity);\n\n this.incidentActivity = new MetricsMiniChartWidget({\n icon: \"bi bi-exclamation-triangle fs-2\",\n title: 'Incidents',\n subtitle: '{{now_value}} <span class=\"subtitle-label\">{{now_label}}</span> {{total}} <span class=\"subtitle-label\">Total</span>',\n background: \"#B14545\",\n textColor: \"#FFFFFF\",\n endpoint: '/api/metrics/fetch',\n trendRange: 4,\n trendOffset: 0,\n granularity: 'days',\n slugs: ['incidents'],\n account: 'incident',\n chartType: 'line',\n showTooltip: true,\n showXAxis: true,\n height: 50,\n chartWidth: '100%',\n color: 'rgba(245, 245, 255, 0.8)',\n fill: true,\n fillColor: 'rgba(245, 245, 255, 0.6)',\n smoothing: 0.3,\n showTrending: true,\n containerId: 'incident_activity_day'\n });\n this.addChild(this.incidentActivity);\n }\n\n async onBeforeRender() {\n\n }\n\n prepareStatsForTemplate() {\n // Determine badge and icon classes based on incidents change\n // const isDecreasing = this.stats.incidentsChange.startsWith('-');\n // this.stats.incidentsBadgeClass = isDecreasing ? 'bg-success-subtle text-success' : 'bg-warning-subtle text-warning';\n // this.stats.incidentsIconClass = isDecreasing ? 'arrow-down' : 'arrow-up';\n\n }\n\n async loadValues() {\n try {\n const response = await this.getApp().rest.GET('/api/metrics/value/get', {\n slugs: ['total_users', 'total_groups'],\n account: \"global\"\n });\n if (response.success && response.data.status) {\n Object.assign(this.stats, response.data.data);\n }\n if (this.groupActivity) {\n this.groupActivity.header.total_groups = this.stats.total_groups || 0;\n this.groupActivity.header.render();\n }\n if (this.userActivity) {\n this.userActivity.header.total_users = this.stats.total_users || 0;\n this.userActivity.header.render();\n }\n } catch (error) {\n console.error('Failed to load admin stats:', error);\n }\n }\n\n async loadStats() {\n // Example of how to load real data\n try {\n const response = await this.getApp().rest.GET('/api/metrics/series', {\n slugs: ['user_created', 'user_activity_day', \"incidents\",\n \"api_calls\", \"api_errors\", \"group_activity_day\"],\n account: \"global\",\n granularity: \"days\"\n });\n if (response.success && response.data.status) {\n Object.assign(this.stats, response.data.data);\n this.prepareStatsForTemplate();\n }\n } catch (error) {\n console.error('Failed to load admin stats:', error);\n }\n }\n}\n\nexport default class AdminDashboardPage extends Page {\n constructor(options = {}) {\n super({\n ...options,\n title: 'Admin Dashboard',\n className: 'admin-dashboard-page'\n });\n\n // Page data\n this.pageTitle = 'Admin Dashboard';\n this.pageSubtitle = 'System monitoring and metrics overview';\n }\n\n async getTemplate() {\n return `\n <div class=\"admin-dashboard-container container-lg\">\n <!-- Page Header -->\n <div class=\"d-flex justify-content-between align-items-center mb-2\">\n <div>\n <p class=\"text-muted mb-0\">{{pageSubtitle}}</p>\n <small class=\"text-info\">\n <i class=\"bi bi-shield-check me-1\"></i>\n Real-time system metrics and performance monitoring\n </small>\n </div>\n <div class=\"btn-group\" role=\"group\">\n <button type=\"button\" class=\"btn btn-outline-secondary btn-sm\"\n data-action=\"refresh-all\" title=\"Refresh All Charts\">\n <i class=\"bi bi-arrow-clockwise\"></i> Refresh\n </button>\n <button type=\"button\" class=\"btn btn-outline-primary btn-sm\"\n data-action=\"export-metrics\" title=\"Export Metrics Data\">\n <i class=\"bi bi-download\"></i> Export\n </button>\n <button type=\"button\" class=\"btn btn-outline-warning btn-sm\"\n data-action=\"view-alerts\" title=\"View System Alerts\">\n <i class=\"bi bi-bell\"></i> Alerts\n </button>\n </div>\n </div>\n\n <!-- Stats Header -->\n <div data-container=\"admin-header\"></div>\n <div data-container=\"example-chart\"></div>\n <!-- Charts Section -->\n <div class=\"row\">\n <!-- Full Width API Metrics Chart -->\n <div class=\"col-12 mb-4\">\n <div data-container=\"api-metrics-chart\"></div>\n </div>\n </div>\n\n <!-- System Status Footer -->\n <div class=\"row\">\n <div class=\"col-12\">\n <div class=\"alert alert-success border-0\" role=\"alert\">\n <div class=\"d-flex align-items-center\">\n <i class=\"bi bi-check-circle-fill me-2\"></i>\n <div>\n <strong>System Status:</strong> All systems operational.\n Last updated: <span class=\"text-muted\">{{lastUpdated}}</span>\n </div>\n <div class=\"ms-auto\">\n <button class=\"btn btn-sm btn-outline-success\" data-action=\"view-system-status\">\n <i class=\"bi bi-info-circle\"></i> Details\n </button>\n </div>\n </div>\n </div>\n </div>\n </div>\n </div>\n `;\n }\n\n async onInit() {\n // Set last updated time\n this.lastUpdated = new Date().toLocaleString();\n\n // Create and add header view\n this.headerView = new AdminHeaderView({\n containerId: 'admin-header'\n });\n this.addChild(this.headerView);\n\n // Create API Metrics Chart\n this.apiMetricsChart = new MetricsChart({\n title: `<i class=\"bi bi-graph-up me-2\"></i> API Metrics`,\n endpoint: '/api/metrics/fetch',\n height: 250,\n granularity: 'hours',\n slugs: ['api_calls', 'api_errors'],\n account: 'global',\n chartType: 'line',\n showDateRange: false,\n yAxis: {\n label: 'Count',\n beginAtZero: true\n },\n tooltip: {\n y: 'number'\n },\n containerId: 'api-metrics-chart'\n });\n this.addChild(this.apiMetricsChart);\n\n }\n\n // Action Handlers\n async onActionRefreshAll(event, element) {\n try {\n // Show loading state\n const button = element || event?.currentTarget || null;\n const icon = button?.querySelector?.('i');\n icon?.classList.add('bi-spin');\n if (button) button.disabled = true;\n\n // Refresh all charts\n const promises = [\n this.headerView?.loadValues(),\n this.apiMetricsChart?.refresh()\n ].filter(Boolean);\n\n await Promise.allSettled(promises);\n\n\n // Update last updated time\n this.lastUpdated = new Date().toLocaleString();\n\n // Emit refresh event\n const eventBus = this.getApp()?.events;\n if (eventBus) {\n eventBus.emit('admin:dashboard-refreshed', {\n page: this,\n timestamp: this.lastUpdated\n });\n }\n\n } catch (error) {\n console.error('Failed to refresh dashboard:', error);\n // Show error feedback\n const alert = this.element.querySelector('.alert-success');\n if (alert) {\n alert.className = 'alert alert-danger border-0';\n alert.innerHTML = `\n <div class=\"d-flex align-items-center\">\n <i class=\"bi bi-exclamation-triangle-fill me-2\"></i>\n <div>\n <strong>Error:</strong> Failed to refresh dashboard data.\n </div>\n </div>\n `;\n\n // Reset after 5 seconds\n setTimeout(() => {\n alert.className = 'alert alert-success border-0';\n alert.innerHTML = `\n <div class=\"d-flex align-items-center\">\n <i class=\"bi bi-check-circle-fill me-2\"></i>\n <div>\n <strong>System Status:</strong> All systems operational.\n Last updated: <span class=\"text-muted\">${this.lastUpdated}</span>\n </div>\n </div>\n `;\n }, 5000);\n }\n } finally {\n // Reset button state\n const icon = element.querySelector('i');\n icon?.classList.remove('bi-spin');\n if (button) button.disabled = false;\n }\n }\n\n async onActionExportMetrics(event, element) {\n try {\n // Export all charts as PNG\n await this.apiMetricsChart?.export('png');\n\n // Show success feedback\n const eventBus = this.getApp()?.events;\n if (eventBus) {\n eventBus.emit('admin:metrics-exported', {\n page: this,\n charts: ['api-metrics']\n });\n }\n\n } catch (error) {\n console.error('Failed to export metrics:', error);\n }\n }\n\n async onActionViewAlerts(event, element) {\n // Navigate to alerts page or show alerts modal\n const router = this.getApp()?.router;\n if (router) {\n router.navigateTo('/admin/alerts');\n }\n }\n\n async onActionViewSystemStatus(event, element) {\n // Navigate to system status page\n const router = this.getApp()?.router;\n if (router) {\n router.navigateTo('/admin/system-status');\n }\n }\n\n // Public API\n async refreshDashboard() {\n return this.onActionRefreshAll(null, null, { disabled: false, querySelector: () => null });\n }\n\n getCharts() {\n return {\n apiMetrics: this.apiMetricsChart\n };\n }\n\n getStats() {\n return this.headerView?.stats || {};\n }\n\n async onAfterRender() {\n this.headerView?.loadValues();\n }\n}\n","/**\n * SideNavView - Left sidebar navigation with content panel\n *\n * A reusable navigation component that displays a vertical sidebar with\n * nav links, optional group labels, and icons. The content panel mounts\n * one child view at a time, switching on nav click.\n *\n * Features:\n * - Left sidebar with nav links, icons, and group dividers\n * - Active state with accent border\n * - Mount/unmount child views on section switch\n * - Responsive: collapses to dropdown on narrow containers\n * - Permission-aware: skips sections the user lacks permission for\n * - Configurable nav width and content padding\n * - Smooth fade transitions between sections\n *\n * Example Usage:\n * ```javascript\n * const sideNav = new SideNavView({\n * sections: [\n * { key: 'profile', label: 'Profile', icon: 'bi-person', view: profileView },\n * { key: 'security', label: 'Security', icon: 'bi-shield-lock', view: securityView },\n * { type: 'divider', label: 'Activity' },\n * { key: 'sessions', label: 'Sessions', icon: 'bi-clock-history', view: sessionsView },\n * ],\n * activeSection: 'profile',\n * navWidth: 200,\n * contentPadding: '1.5rem 2.5rem',\n * enableResponsive: true\n * });\n * ```\n */\n\nimport View from '@core/View.js';\n\nclass SideNavView extends View {\n constructor(options = {}) {\n const {\n sections = [],\n activeSection,\n navWidth,\n contentPadding,\n enableResponsive,\n minWidth,\n ...viewOptions\n } = options;\n\n super({\n tagName: 'div',\n className: 'side-nav-view',\n ...viewOptions\n });\n\n // Configuration\n this.navWidth = navWidth || 200;\n this.contentPadding = contentPadding || '1.5rem 2.5rem';\n this.enableResponsive = enableResponsive !== false;\n this.minWidth = minWidth || 500;\n\n // State\n this.sectionConfigs = []; // Full config array (including dividers)\n this.sectionViews = {}; // key → view instance\n this.sectionKeys = []; // Ordered navigable section keys\n this.activeSection = null;\n this.currentMode = 'sidebar'; // 'sidebar' or 'dropdown'\n this.resizeObserver = null;\n this.lastContainerWidth = 0;\n\n // Process sections config\n for (const config of sections) {\n this._addSectionConfig(config);\n }\n\n // Set initial active section\n this.activeSection = activeSection || this.sectionKeys[0] || null;\n\n // Bind resize handler\n this.handleResize = this.handleResize.bind(this);\n }\n\n /**\n * Process and store a section config entry\n * @param {object} config - Section config (navigable or divider)\n * @private\n */\n _addSectionConfig(config) {\n if (config.type === 'divider') {\n this.sectionConfigs.push({ type: 'divider', label: config.label });\n return;\n }\n\n // Skip if user lacks required permission\n if (config.permissions && !this._hasPermission(config.permissions)) {\n return;\n }\n\n this.sectionConfigs.push(config);\n this.sectionKeys.push(config.key);\n\n if (config.view) {\n this.sectionViews[config.key] = config.view;\n config.view.parent = this;\n }\n }\n\n /**\n * Check if the current user has a permission\n * @param {string} perm - Permission string\n * @returns {boolean}\n * @private\n */\n _hasPermission(perm) {\n try {\n return this.getApp().activeUser.hasPerm(perm);\n } catch {\n return true; // If app isn't available yet, allow — will be checked at render\n }\n }\n\n // ───────────────────────────────────────────────\n // Template\n // ───────────────────────────────────────────────\n\n async renderTemplate() {\n const nav = this.currentMode === 'dropdown'\n ? this._buildDropdownNav()\n : this._buildSidebarNav();\n\n return `\n <style>\n .snv-layout { display: flex; height: 100%; min-height: 0; }\n .snv-nav {\n width: ${this.navWidth}px;\n background: #f8f9fc;\n border-right: 1px solid #e9ecef;\n padding: 0.75rem 0;\n flex-shrink: 0;\n overflow-y: auto;\n }\n .snv-nav a {\n color: #495057;\n padding: 0.45rem 1.25rem;\n font-size: 0.85rem;\n display: flex;\n align-items: center;\n gap: 0.5rem;\n text-decoration: none;\n cursor: pointer;\n }\n .snv-nav a:hover { background: #e9ecef; }\n .snv-nav a.active {\n background: #e7f1ff;\n color: #0d6efd;\n font-weight: 600;\n border-right: 2px solid #0d6efd;\n }\n .snv-nav a i { width: 18px; text-align: center; font-size: 0.9rem; }\n .snv-nav-label {\n font-size: 0.65rem;\n font-weight: 700;\n text-transform: uppercase;\n letter-spacing: 0.06em;\n color: #adb5bd;\n padding: 0.75rem 1.25rem 0.25rem;\n }\n .snv-content {\n flex: 1;\n overflow-y: auto;\n padding: ${this.contentPadding};\n min-width: 0;\n }\n .snv-content > .snv-section { display: none; }\n .snv-content > .snv-section.snv-active { display: block; }\n .snv-dropdown { margin-bottom: 0.75rem; }\n .snv-select-btn {\n display: flex;\n align-items: center;\n gap: 0.5rem;\n width: 100%;\n padding: 0.5rem 1rem;\n background: #f8f9fc;\n border: 1px solid #dee2e6;\n border-radius: 0.375rem;\n font-size: 0.85rem;\n color: #495057;\n cursor: pointer;\n }\n .snv-select-btn:hover { background: #e9ecef; }\n .snv-select-btn::after {\n content: '';\n display: inline-block;\n margin-left: auto;\n border-top: 0.3em solid;\n border-right: 0.3em solid transparent;\n border-left: 0.3em solid transparent;\n }\n @media (max-width: 576px) {\n .snv-nav { display: none; }\n .snv-content { padding: 1.25rem; }\n }\n </style>\n ${this.currentMode === 'dropdown' ? `\n <div class=\"snv-dropdown\">${nav}</div>\n <div class=\"snv-content\" data-container=\"snv-content\"></div>\n ` : `\n <div class=\"snv-layout\">\n <nav class=\"snv-nav\">${nav}</nav>\n <div class=\"snv-content\" data-container=\"snv-content\"></div>\n </div>\n `}\n `;\n }\n\n /**\n * Build sidebar navigation HTML\n * @returns {string}\n * @private\n */\n _buildSidebarNav() {\n return this.sectionConfigs.map(config => {\n if (config.type === 'divider') {\n return `<div class=\"snv-nav-label\">${this.escapeHtml(config.label)}</div>`;\n }\n const isActive = config.key === this.activeSection;\n const icon = config.icon ? `<i class=\"bi ${config.icon}\"></i>` : '';\n return `<a href=\"#\" class=\"${isActive ? 'active' : ''}\" data-action=\"navigate\" data-section=\"${config.key}\">${icon} ${this.escapeHtml(config.label)}</a>`;\n }).join('');\n }\n\n /**\n * Build dropdown navigation HTML (responsive mode)\n * @returns {string}\n * @private\n */\n _buildDropdownNav() {\n const activeConfig = this.sectionConfigs.find(c => c.key === this.activeSection);\n const activeLabel = activeConfig ? activeConfig.label : this.sectionKeys[0];\n\n const items = this.sectionConfigs\n .filter(c => c.type !== 'divider')\n .map(config => {\n const isActive = config.key === this.activeSection;\n return `\n <li>\n <button class=\"dropdown-item ${isActive ? 'active' : ''}\"\n data-action=\"navigate\"\n data-section=\"${config.key}\"\n type=\"button\">\n ${config.icon ? `<i class=\"bi ${config.icon} me-2\"></i>` : ''}\n ${this.escapeHtml(config.label)}\n ${isActive ? '<i class=\"bi bi-check-lg ms-2\"></i>' : ''}\n </button>\n </li>\n `;\n }).join('');\n\n return `\n <div class=\"dropdown\">\n <button class=\"snv-select-btn\" type=\"button\"\n data-bs-toggle=\"dropdown\" aria-expanded=\"false\">\n ${activeConfig?.icon ? `<i class=\"bi ${activeConfig.icon}\"></i>` : ''}\n <span>${this.escapeHtml(activeLabel)}</span>\n </button>\n <ul class=\"dropdown-menu w-100\">${items}</ul>\n </div>\n `;\n }\n\n // ───────────────────────────────────────────────\n // Lifecycle\n // ───────────────────────────────────────────────\n\n async onAfterRender() {\n await super.onAfterRender();\n\n // Mount the active section\n if (this.activeSection) {\n await this._mountSection(this.activeSection);\n }\n\n // Set up responsive behavior\n if (this.enableResponsive) {\n this._setupResponsive();\n }\n }\n\n async onBeforeDestroy() {\n await super.onBeforeDestroy();\n\n // Clean up resize observer\n if (this.resizeObserver) {\n this.resizeObserver.disconnect();\n this.resizeObserver = null;\n }\n\n if (typeof window !== 'undefined') {\n window.removeEventListener('resize', this.handleResize);\n }\n\n // Destroy all section views\n for (const view of Object.values(this.sectionViews)) {\n if (view && typeof view.destroy === 'function') {\n await view.destroy();\n }\n }\n }\n\n // ───────────────────────────────────────────────\n // Section switching\n // ───────────────────────────────────────────────\n\n /**\n * Navigate to a section\n * @param {string} key - Section key\n * @returns {Promise<boolean>}\n */\n async showSection(key) {\n if (!this.sectionViews[key]) {\n console.warn(`SideNavView: Section \"${key}\" does not exist`);\n return false;\n }\n\n if (key === this.activeSection) {\n // Already active — but ensure it's mounted\n const view = this.sectionViews[key];\n if (view && view.isMounted() && this.element?.contains(view.element)) {\n return true;\n }\n }\n\n const previousSection = this.activeSection;\n this.activeSection = key;\n\n // Unmount previous section\n if (previousSection && previousSection !== key) {\n await this._unmountSection(previousSection);\n }\n\n // Mount new section\n await this._mountSection(key);\n\n // Update nav visual state\n this._updateNavState(key);\n\n this.emit('section:changed', {\n activeSection: key,\n previousSection\n });\n\n return true;\n }\n\n /**\n * Mount a section view into the content area\n * @param {string} key - Section key\n * @private\n */\n async _mountSection(key) {\n const view = this.sectionViews[key];\n if (!view) return;\n\n const container = this.element?.querySelector('[data-container=\"snv-content\"]');\n if (!container) return;\n\n if (!view.isMounted()) {\n await view.render(true, container);\n }\n }\n\n /**\n * Unmount a section view\n * @param {string} key - Section key\n * @private\n */\n async _unmountSection(key) {\n const view = this.sectionViews[key];\n if (!view || !view.isMounted()) return;\n\n await view.unmount();\n }\n\n /**\n * Update nav link active state\n * @param {string} activeKey - Active section key\n * @private\n */\n _updateNavState(activeKey) {\n if (!this.element) return;\n\n // Update sidebar links\n this.element.querySelectorAll('.snv-nav a, .dropdown-item').forEach(link => {\n const section = link.dataset.section;\n if (section) {\n link.classList.toggle('active', section === activeKey);\n }\n });\n\n // Update dropdown button label\n const selectBtn = this.element.querySelector('.snv-select-btn span');\n if (selectBtn) {\n const config = this.sectionConfigs.find(c => c.key === activeKey);\n if (config) {\n selectBtn.textContent = config.label;\n }\n }\n }\n\n // ───────────────────────────────────────────────\n // Action handlers\n // ───────────────────────────────────────────────\n\n async onActionNavigate(event, el) {\n event.preventDefault();\n const section = el.dataset.section;\n if (section) {\n await this.showSection(section);\n }\n return true;\n }\n\n // ───────────────────────────────────────────────\n // Responsive\n // ───────────────────────────────────────────────\n\n /**\n * Set up responsive width detection\n * @private\n */\n _setupResponsive() {\n if (!this.element || !this.enableResponsive) return;\n\n this._updateMode();\n\n if (typeof ResizeObserver !== 'undefined') {\n this.resizeObserver = new ResizeObserver(() => {\n this.handleResize();\n });\n const container = this.element.parentElement || this.element;\n this.resizeObserver.observe(container);\n } else {\n window.addEventListener('resize', this.handleResize);\n }\n }\n\n /**\n * Handle resize events\n */\n async handleResize() {\n const containerWidth = this._getContainerWidth();\n if (Math.abs(containerWidth - this.lastContainerWidth) > 50) {\n this.lastContainerWidth = containerWidth;\n await this._updateMode();\n }\n }\n\n /**\n * Get the container width\n * @returns {number}\n * @private\n */\n _getContainerWidth() {\n if (!this.element) return this.minWidth;\n const container = this.element.parentElement || this.element;\n return container.offsetWidth || this.minWidth;\n }\n\n /**\n * Check and switch between sidebar and dropdown modes\n * @private\n */\n async _updateMode() {\n const containerWidth = this._getContainerWidth();\n const newMode = containerWidth < this.minWidth ? 'dropdown' : 'sidebar';\n\n if (newMode !== this.currentMode) {\n this.currentMode = newMode;\n if (this.isMounted()) {\n await this.render();\n }\n this.emit('navigation:modeChanged', {\n mode: this.currentMode,\n containerWidth\n });\n }\n }\n\n // ───────────────────────────────────────────────\n // Public API\n // ───────────────────────────────────────────────\n\n /**\n * Get the active section key\n * @returns {string|null}\n */\n getActiveSection() {\n return this.activeSection;\n }\n\n /**\n * Get all navigable section keys\n * @returns {string[]}\n */\n getSectionKeys() {\n return [...this.sectionKeys];\n }\n\n /**\n * Get a section's view by key\n * @param {string} key - Section key\n * @returns {View|null}\n */\n getSection(key) {\n return this.sectionViews[key] || null;\n }\n\n /**\n * Add a section dynamically\n * @param {object} config - Section config\n * @param {boolean} makeActive - Whether to activate the section\n * @returns {Promise<boolean>}\n */\n async addSection(config, makeActive = false) {\n if (config.key && this.sectionViews[config.key]) {\n console.warn(`SideNavView: Section \"${config.key}\" already exists`);\n return false;\n }\n\n this._addSectionConfig(config);\n\n if (this.isMounted()) {\n await this.render();\n if (makeActive && config.key) {\n await this.showSection(config.key);\n }\n }\n\n this.emit('section:added', { config });\n return true;\n }\n\n /**\n * Remove a section dynamically\n * @param {string} key - Section key to remove\n * @returns {Promise<boolean>}\n */\n async removeSection(key) {\n const view = this.sectionViews[key];\n if (!view) {\n console.warn(`SideNavView: Section \"${key}\" does not exist`);\n return false;\n }\n\n // Destroy the view\n if (typeof view.destroy === 'function') {\n await view.destroy();\n }\n\n // Remove from data structures\n delete this.sectionViews[key];\n this.sectionKeys = this.sectionKeys.filter(k => k !== key);\n this.sectionConfigs = this.sectionConfigs.filter(c => c.key !== key);\n\n // Handle active section removal\n if (this.activeSection === key) {\n this.activeSection = this.sectionKeys[0] || null;\n }\n\n if (this.isMounted()) {\n await this.render();\n }\n\n this.emit('section:removed', { key });\n return true;\n }\n\n /**\n * Prevent model changes from triggering a full re-render.\n * Section views manage their own model reactivity.\n */\n _onModelChange() {\n // no-op — same pattern as UserView\n }\n\n static create(options = {}) {\n return new SideNavView(options);\n }\n}\n\nexport default SideNavView;\n","/**\n * AdminProfileSection - Contact, verification overrides, and account overview\n *\n * Admin can view/edit contact info (email, phone), override verification\n * status, and see account metadata (username, status, role, MFA, dates).\n * Uses model.save() against /api/user/<id> (admin endpoint).\n */\nimport View from '@core/View.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\n\nexport default class AdminProfileSection extends View {\n constructor(options = {}) {\n super({\n className: 'admin-profile-section',\n template: `\n <style>\n .ap-section-label { font-size: 0.7rem; font-weight: 700; text-transform: uppercase; letter-spacing: 0.06em; color: #adb5bd; margin-bottom: 0.5rem; margin-top: 1.75rem; }\n .ap-section-label:first-child { margin-top: 0; }\n .ap-field-row { display: flex; align-items: center; padding: 0.6rem 0; border-bottom: 1px solid #f0f0f0; }\n .ap-field-row:last-child { border-bottom: none; }\n .ap-field-label { width: 140px; font-size: 0.8rem; color: #6c757d; flex-shrink: 0; }\n .ap-field-value { flex: 1; font-size: 0.88rem; color: #212529; display: flex; align-items: center; gap: 0.4rem; }\n .ap-field-action { color: #6c757d; cursor: pointer; font-size: 0.8rem; margin-left: auto; padding: 0.15rem 0.4rem; border-radius: 4px; background: none; border: none; }\n .ap-field-action:hover { background: #f0f0f0; color: #0d6efd; }\n .ap-badge-ok { font-size: 0.65rem; padding: 0.15em 0.45em; background: #d1e7dd; color: #0f5132; border-radius: 3px; }\n .ap-badge-warn { font-size: 0.65rem; padding: 0.15em 0.45em; background: #fff3cd; color: #856404; border-radius: 3px; }\n .ap-badge-muted { font-size: 0.65rem; padding: 0.15em 0.45em; background: #f0f0f0; color: #6c757d; border-radius: 3px; }\n .ap-not-set { color: #adb5bd; font-style: italic; font-size: 0.85rem; }\n </style>\n\n <!-- Contact & Verification -->\n <div class=\"ap-section-label\">Contact & Verification</div>\n <div class=\"ap-field-row\">\n <div class=\"ap-field-label\">Email</div>\n <div class=\"ap-field-value\">\n {{model.email}}\n {{#model.is_email_verified|bool}}\n <span class=\"ap-badge-ok\">Verified</span>\n {{/model.is_email_verified|bool}}\n {{^model.is_email_verified|bool}}\n <span class=\"ap-badge-warn\">Unverified</span>\n {{/model.is_email_verified|bool}}\n </div>\n {{#model.is_email_verified|bool}}\n <button type=\"button\" class=\"ap-field-action\" data-action=\"unverify-email\" title=\"Mark as unverified\"><i class=\"bi bi-x-circle\"></i></button>\n {{/model.is_email_verified|bool}}\n {{^model.is_email_verified|bool}}\n <button type=\"button\" class=\"ap-field-action\" data-action=\"force-verify-email\" title=\"Force verify\"><i class=\"bi bi-patch-check\"></i></button>\n {{/model.is_email_verified|bool}}\n <button type=\"button\" class=\"ap-field-action\" data-action=\"change-email\" title=\"Change email\"><i class=\"bi bi-pencil\"></i></button>\n </div>\n <div class=\"ap-field-row\">\n <div class=\"ap-field-label\">Phone</div>\n <div class=\"ap-field-value\">\n {{#hasPhone|bool}}\n {{model.phone_number}}\n {{#model.is_phone_verified|bool}}\n <span class=\"ap-badge-ok\">Verified</span>\n {{/model.is_phone_verified|bool}}\n {{^model.is_phone_verified|bool}}\n <span class=\"ap-badge-warn\">Unverified</span>\n {{/model.is_phone_verified|bool}}\n {{/hasPhone|bool}}\n {{^hasPhone|bool}}\n <span class=\"ap-not-set\">Not set</span>\n {{/hasPhone|bool}}\n </div>\n {{#hasPhone|bool}}\n {{#model.is_phone_verified|bool}}\n <button type=\"button\" class=\"ap-field-action\" data-action=\"unverify-phone\" title=\"Mark as unverified\"><i class=\"bi bi-x-circle\"></i></button>\n {{/model.is_phone_verified|bool}}\n {{^model.is_phone_verified|bool}}\n <button type=\"button\" class=\"ap-field-action\" data-action=\"force-verify-phone\" title=\"Force verify\"><i class=\"bi bi-patch-check\"></i></button>\n {{/model.is_phone_verified|bool}}\n <button type=\"button\" class=\"ap-field-action\" data-action=\"change-phone\" title=\"Change phone\"><i class=\"bi bi-pencil\"></i></button>\n <button type=\"button\" class=\"ap-field-action\" data-action=\"remove-phone\" title=\"Remove phone\"><i class=\"bi bi-x-lg\"></i></button>\n {{/hasPhone|bool}}\n {{^hasPhone|bool}}\n <button type=\"button\" class=\"ap-field-action\" data-action=\"set-phone\" title=\"Set phone number\"><i class=\"bi bi-plus\"></i></button>\n {{/hasPhone|bool}}\n </div>\n\n <!-- Account -->\n <div class=\"ap-section-label\">Account</div>\n <div class=\"ap-field-row\">\n <div class=\"ap-field-label\">Username</div>\n <div class=\"ap-field-value\">{{model.username}}</div>\n <button type=\"button\" class=\"ap-field-action\" data-action=\"edit-username\" title=\"Edit\"><i class=\"bi bi-pencil\"></i></button>\n </div>\n <div class=\"ap-field-row\">\n <div class=\"ap-field-label\">Status</div>\n <div class=\"ap-field-value\">\n {{#model.is_active|bool}}<span class=\"ap-badge-ok\">Active</span>{{/model.is_active|bool}}\n {{^model.is_active|bool}}<span class=\"ap-badge-warn\">Inactive</span>{{/model.is_active|bool}}\n </div>\n </div>\n <div class=\"ap-field-row\">\n <div class=\"ap-field-label\">Role</div>\n <div class=\"ap-field-value\">\n {{roleLabel}}\n {{#model.is_staff|bool}}<span class=\"ap-badge-muted\">Staff</span>{{/model.is_staff|bool}}\n </div>\n </div>\n <div class=\"ap-field-row\">\n <div class=\"ap-field-label\">MFA</div>\n <div class=\"ap-field-value\">\n {{#model.requires_mfa|bool}}<span class=\"ap-badge-ok\">Required</span>{{/model.requires_mfa|bool}}\n {{^model.requires_mfa|bool}}<span class=\"ap-badge-muted\">Not required</span>{{/model.requires_mfa|bool}}\n </div>\n </div>\n <div class=\"ap-field-row\">\n <div class=\"ap-field-label\">Member Since</div>\n <div class=\"ap-field-value\">{{model.date_joined|date}}</div>\n </div>\n <div class=\"ap-field-row\">\n <div class=\"ap-field-label\">Last Login</div>\n <div class=\"ap-field-value\">{{model.last_login|relative}}</div>\n </div>\n `,\n ...options\n });\n }\n\n // ── Computed properties ─────────────────────────\n\n get hasPhone() {\n return !!(this.model && this.model.get('phone_number'));\n }\n\n get roleLabel() {\n if (!this.model) return 'User';\n if (this.model.get('is_superuser')) return 'Superuser';\n return 'User';\n }\n\n // ── Verification overrides ──────────────────────\n\n async onActionForceVerifyEmail() {\n const confirmed = await Dialog.confirm(\n `Mark <strong>${this.model.get('email')}</strong> as verified? This bypasses the normal verification flow.`,\n 'Force Verify Email'\n );\n if (!confirmed) return true;\n await this._saveField({ is_email_verified: true }, 'Email marked as verified');\n return true;\n }\n\n async onActionUnverifyEmail() {\n const confirmed = await Dialog.confirm(\n 'Mark email as unverified? The user will need to re-verify their email.',\n 'Unverify Email'\n );\n if (!confirmed) return true;\n await this._saveField({ is_email_verified: false }, 'Email marked as unverified');\n return true;\n }\n\n async onActionForceVerifyPhone() {\n const confirmed = await Dialog.confirm(\n `Mark <strong>${this.model.get('phone_number')}</strong> as verified? This bypasses the normal verification flow.`,\n 'Force Verify Phone'\n );\n if (!confirmed) return true;\n await this._saveField({ is_phone_verified: true }, 'Phone marked as verified');\n return true;\n }\n\n async onActionUnverifyPhone() {\n const confirmed = await Dialog.confirm(\n 'Mark phone as unverified? The user will need to re-verify their phone number.',\n 'Unverify Phone'\n );\n if (!confirmed) return true;\n await this._saveField({ is_phone_verified: false }, 'Phone marked as unverified');\n return true;\n }\n\n // ── Contact editing ─────────────────────────────\n\n async onActionChangeEmail() {\n const email = await Dialog.prompt(\n 'Enter the new email address for this user:',\n 'Change Email',\n { defaultValue: this.model.get('email') || '' }\n );\n if (email === null || !email.trim()) return true;\n await this._saveField({ email: email.trim() }, 'Email updated');\n return true;\n }\n\n async onActionChangePhone() {\n const phone = await Dialog.prompt(\n 'Enter the new phone number for this user:',\n 'Change Phone',\n { defaultValue: this.model.get('phone_number') || '' }\n );\n if (phone === null || !phone.trim()) return true;\n await this._saveField({ phone_number: phone.trim() }, 'Phone number updated');\n return true;\n }\n\n async onActionSetPhone() {\n const phone = await Dialog.prompt(\n 'Enter a phone number for this user:',\n 'Set Phone Number',\n { placeholder: '(415) 555-0123' }\n );\n if (!phone || !phone.trim()) return true;\n await this._saveField({ phone_number: phone.trim() }, 'Phone number added');\n return true;\n }\n\n async onActionRemovePhone() {\n const confirmed = await Dialog.confirm(\n 'Remove this user\\'s phone number?',\n 'Remove Phone'\n );\n if (!confirmed) return true;\n\n const resp = await this.model.save({ phone_number: null });\n if (resp.status === 200) {\n this.model.set('is_phone_verified', false);\n this.getApp()?.toast?.success('Phone number removed');\n await this.render();\n } else {\n this.getApp()?.toast?.error(resp.message || 'Failed to remove phone number');\n }\n return true;\n }\n\n async onActionEditUsername() {\n const username = await Dialog.prompt(\n 'Username:',\n 'Edit Username',\n { defaultValue: this.model.get('username') || '' }\n );\n if (username !== null && username.trim()) {\n await this._saveField({ username: username.trim() }, 'Username updated');\n }\n return true;\n }\n\n // ── Helpers ─────────────────────────────────────\n\n async _saveField(fields, successMsg) {\n const resp = await this.model.save(fields);\n if (resp.status === 200) {\n this.getApp()?.toast?.success(successMsg);\n await this.render();\n } else {\n this.getApp()?.toast?.error(resp.message || 'Failed to save');\n }\n }\n}\n","/**\n * AdminPersonalSection - Personal information editing\n *\n * Admin can view/edit name fields, DOB, timezone, and address.\n * Uses model.save() against /api/user/<id> (admin endpoint).\n */\nimport View from '@core/View.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\n\nexport default class AdminPersonalSection extends View {\n constructor(options = {}) {\n super({\n className: 'admin-personal-section',\n template: `\n <style>\n .aps-section-label { font-size: 0.7rem; font-weight: 700; text-transform: uppercase; letter-spacing: 0.06em; color: #adb5bd; margin-bottom: 0.5rem; margin-top: 1.75rem; }\n .aps-section-label:first-child { margin-top: 0; }\n .aps-field-row { display: flex; align-items: center; padding: 0.6rem 0; border-bottom: 1px solid #f0f0f0; }\n .aps-field-row:last-child { border-bottom: none; }\n .aps-field-label { width: 140px; font-size: 0.8rem; color: #6c757d; flex-shrink: 0; }\n .aps-field-value { flex: 1; font-size: 0.88rem; color: #212529; display: flex; align-items: center; gap: 0.4rem; }\n .aps-field-action { color: #6c757d; cursor: pointer; font-size: 0.8rem; margin-left: auto; padding: 0.15rem 0.4rem; border-radius: 4px; background: none; border: none; }\n .aps-field-action:hover { background: #f0f0f0; color: #0d6efd; }\n .aps-badge-ok { font-size: 0.65rem; padding: 0.15em 0.45em; background: #d1e7dd; color: #0f5132; border-radius: 3px; }\n .aps-badge-warn { font-size: 0.65rem; padding: 0.15em 0.45em; background: #fff3cd; color: #856404; border-radius: 3px; }\n .aps-not-set { color: #adb5bd; font-style: italic; font-size: 0.85rem; }\n </style>\n\n <!-- Name -->\n <div class=\"aps-section-label\">Name</div>\n <div class=\"aps-field-row\">\n <div class=\"aps-field-label\">Display Name</div>\n <div class=\"aps-field-value\">\n {{#model.display_name}}{{model.display_name}}{{/model.display_name}}\n {{^model.display_name}}<span class=\"aps-not-set\">Not set</span>{{/model.display_name}}\n </div>\n <button type=\"button\" class=\"aps-field-action\" data-action=\"edit-display-name\" title=\"Edit\"><i class=\"bi bi-pencil\"></i></button>\n </div>\n <div class=\"aps-field-row\">\n <div class=\"aps-field-label\">First Name</div>\n <div class=\"aps-field-value\">\n {{#model.first_name}}{{model.first_name}}{{/model.first_name}}\n {{^model.first_name}}<span class=\"aps-not-set\">Not set</span>{{/model.first_name}}\n </div>\n <button type=\"button\" class=\"aps-field-action\" data-action=\"edit-first-name\" title=\"Edit\"><i class=\"bi bi-pencil\"></i></button>\n </div>\n <div class=\"aps-field-row\">\n <div class=\"aps-field-label\">Last Name</div>\n <div class=\"aps-field-value\">\n {{#model.last_name}}{{model.last_name}}{{/model.last_name}}\n {{^model.last_name}}<span class=\"aps-not-set\">Not set</span>{{/model.last_name}}\n </div>\n <button type=\"button\" class=\"aps-field-action\" data-action=\"edit-last-name\" title=\"Edit\"><i class=\"bi bi-pencil\"></i></button>\n </div>\n\n <!-- Details -->\n <div class=\"aps-section-label\">Details</div>\n <div class=\"aps-field-row\">\n <div class=\"aps-field-label\">Date of Birth</div>\n <div class=\"aps-field-value\">\n {{#hasDob|bool}}\n {{dobFormatted}}\n {{#model.is_dob_verified|bool}}<span class=\"aps-badge-ok\">Verified</span>{{/model.is_dob_verified|bool}}\n {{^model.is_dob_verified|bool}}<span class=\"aps-badge-warn\">Unverified</span>{{/model.is_dob_verified|bool}}\n {{/hasDob|bool}}\n {{^hasDob|bool}}<span class=\"aps-not-set\">Not set</span>{{/hasDob|bool}}\n </div>\n {{#hasDob|bool}}\n {{#model.is_dob_verified|bool}}\n <button type=\"button\" class=\"aps-field-action\" data-action=\"unverify-dob\" title=\"Mark as unverified\"><i class=\"bi bi-x-circle\"></i></button>\n {{/model.is_dob_verified|bool}}\n {{^model.is_dob_verified|bool}}\n <button type=\"button\" class=\"aps-field-action\" data-action=\"force-verify-dob\" title=\"Force verify\"><i class=\"bi bi-patch-check\"></i></button>\n {{/model.is_dob_verified|bool}}\n {{/hasDob|bool}}\n <button type=\"button\" class=\"aps-field-action\" data-action=\"edit-dob\" title=\"Edit\"><i class=\"bi bi-pencil\"></i></button>\n </div>\n <div class=\"aps-field-row\">\n <div class=\"aps-field-label\">Timezone</div>\n <div class=\"aps-field-value\">{{timezoneDisplay}}</div>\n <button type=\"button\" class=\"aps-field-action\" data-action=\"edit-timezone\" title=\"Edit\"><i class=\"bi bi-pencil\"></i></button>\n </div>\n\n <!-- Address -->\n <div class=\"aps-section-label\">Address</div>\n <div class=\"aps-field-row\">\n <div class=\"aps-field-label\">Address</div>\n <div class=\"aps-field-value\">\n {{#hasAddress|bool}}{{addressSummary}}{{/hasAddress|bool}}\n {{^hasAddress|bool}}<span class=\"aps-not-set\">Not set</span>{{/hasAddress|bool}}\n </div>\n <button type=\"button\" class=\"aps-field-action\" data-action=\"edit-address\" title=\"Edit\"><i class=\"bi bi-pencil\"></i></button>\n </div>\n `,\n ...options\n });\n }\n\n // ── Computed properties ─────────────────────────\n\n get hasDob() {\n return !!this.model?.get('dob');\n }\n\n get dobFormatted() {\n const dob = this.model?.get('dob');\n if (!dob) return '';\n try {\n const [year, month, day] = dob.split('-');\n return `${month}/${day}/${year}`;\n } catch {\n return dob;\n }\n }\n\n get timezoneDisplay() {\n const meta = this.model?.get('metadata') || {};\n return meta.timezone || 'Not set';\n }\n\n get hasAddress() {\n const meta = this.model?.get('metadata') || {};\n return !!(meta.street || meta.city || meta.state || meta.zip || meta.country);\n }\n\n get addressSummary() {\n const meta = this.model?.get('metadata') || {};\n return [meta.street, meta.city, meta.state, meta.zip, meta.country].filter(Boolean).join(', ');\n }\n\n // ── Verification overrides ────────────────────────\n\n async onActionForceVerifyDob() {\n const confirmed = await Dialog.confirm(\n 'Mark date of birth as verified?',\n 'Force Verify DOB'\n );\n if (!confirmed) return true;\n await this._saveField({ is_dob_verified: true }, 'DOB marked as verified');\n return true;\n }\n\n async onActionUnverifyDob() {\n const confirmed = await Dialog.confirm(\n 'Mark date of birth as unverified?',\n 'Unverify DOB'\n );\n if (!confirmed) return true;\n await this._saveField({ is_dob_verified: false }, 'DOB marked as unverified');\n return true;\n }\n\n // ── Action handlers ─────────────────────────────\n\n async onActionEditDisplayName() {\n const name = await Dialog.prompt(\n 'Display name:',\n 'Edit Display Name',\n { defaultValue: this.model.get('display_name') || '' }\n );\n if (name !== null && name.trim()) {\n await this._saveField({ display_name: name.trim() }, 'Display name');\n }\n return true;\n }\n\n async onActionEditFirstName() {\n const name = await Dialog.prompt(\n 'First name:',\n 'Edit First Name',\n { defaultValue: this.model.get('first_name') || '' }\n );\n if (name !== null) {\n await this._saveField({ first_name: name.trim() }, 'First name');\n }\n return true;\n }\n\n async onActionEditLastName() {\n const name = await Dialog.prompt(\n 'Last name:',\n 'Edit Last Name',\n { defaultValue: this.model.get('last_name') || '' }\n );\n if (name !== null) {\n await this._saveField({ last_name: name.trim() }, 'Last name');\n }\n return true;\n }\n\n async onActionEditDob() {\n const data = await Dialog.showForm({\n title: 'Date of Birth',\n size: 'sm',\n fields: [{ name: 'dob', type: 'date', label: 'Date of Birth', cols: 12 }],\n data: { dob: this.model.get('dob') || '' }\n });\n if (!data) return true;\n await this._saveField({ dob: data.dob || null }, 'Date of birth');\n return true;\n }\n\n async onActionEditTimezone() {\n const meta = this.model.get('metadata') || {};\n const data = await Dialog.showForm({\n title: 'Change Timezone',\n size: 'sm',\n fields: [{\n name: 'timezone',\n type: 'select',\n label: 'Timezone',\n cols: 12,\n options: [\n { value: 'America/New_York', text: 'Eastern Time (ET)' },\n { value: 'America/Chicago', text: 'Central Time (CT)' },\n { value: 'America/Denver', text: 'Mountain Time (MT)' },\n { value: 'America/Los_Angeles', text: 'Pacific Time (PT)' },\n { value: 'America/Anchorage', text: 'Alaska Time (AKT)' },\n { value: 'Pacific/Honolulu', text: 'Hawaii Time (HT)' },\n { value: 'UTC', text: 'UTC' },\n { value: 'Europe/London', text: 'London (GMT/BST)' },\n { value: 'Europe/Paris', text: 'Paris (CET/CEST)' },\n { value: 'Europe/Berlin', text: 'Berlin (CET/CEST)' },\n { value: 'Asia/Tokyo', text: 'Tokyo (JST)' },\n { value: 'Asia/Shanghai', text: 'Shanghai (CST)' },\n { value: 'Australia/Sydney', text: 'Sydney (AEST)' }\n ]\n }],\n data: { timezone: meta.timezone || '' }\n });\n if (!data) return true;\n await this._saveField({ metadata: { ...meta, timezone: data.timezone } }, 'Timezone');\n return true;\n }\n\n async onActionEditAddress() {\n const meta = this.model.get('metadata') || {};\n const data = await Dialog.showForm({\n title: 'Edit Address',\n size: 'md',\n fields: [\n { name: 'street', type: 'text', label: 'Street', placeholder: '123 Main St', cols: 12 },\n { name: 'city', type: 'text', label: 'City', cols: 6 },\n { name: 'state', type: 'text', label: 'State / Province', cols: 6 },\n { name: 'zip', type: 'text', label: 'Zip / Postal Code', cols: 6 },\n { name: 'country', type: 'text', label: 'Country', cols: 6 }\n ],\n data: { street: meta.street || '', city: meta.city || '', state: meta.state || '', zip: meta.zip || '', country: meta.country || '' }\n });\n if (!data) return true;\n\n const updatedMeta = { ...meta, street: data.street || '', city: data.city || '', state: data.state || '', zip: data.zip || '', country: data.country || '' };\n await this._saveField({ metadata: updatedMeta }, 'Address');\n return true;\n }\n\n // ── Helpers ─────────────────────────────────────\n\n async _saveField(fields, label) {\n const resp = await this.model.save(fields);\n if (resp.status === 200) {\n this.getApp()?.toast?.success(`${label} updated`);\n await this.render();\n } else {\n this.getApp()?.toast?.error(resp.message || `Failed to update ${label.toLowerCase()}`);\n }\n }\n}\n","/**\n * AdminSecuritySection - Admin security management for a user\n *\n * Admin powers: force password reset, enable/disable MFA,\n * revoke all sessions, view/delete passkeys, view recovery codes.\n * Uses admin endpoints and model.save() against /api/user/<id>.\n */\nimport View from '@core/View.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\nimport rest from '@core/Rest.js';\nimport { PasskeyList, PasskeyForms } from '@core/models/Passkeys.js';\n\nexport default class AdminSecuritySection extends View {\n constructor(options = {}) {\n super({\n className: 'admin-security-section',\n template: `\n <style>\n .as-section-label { font-size: 0.7rem; font-weight: 700; text-transform: uppercase; letter-spacing: 0.06em; color: #adb5bd; margin-bottom: 0.5rem; margin-top: 1.75rem; }\n .as-section-label:first-child { margin-top: 0; }\n .as-item { display: flex; align-items: center; gap: 0.85rem; padding: 0.85rem 1rem; border: 1px solid #f0f0f0; border-radius: 8px; margin-bottom: 0.5rem; cursor: pointer; transition: border-color 0.15s, background 0.15s; }\n .as-item:hover { border-color: #dee2e6; background: #fafbfd; }\n .as-icon { width: 36px; height: 36px; border-radius: 8px; display: flex; align-items: center; justify-content: center; font-size: 1rem; flex-shrink: 0; }\n .as-info { flex: 1; min-width: 0; }\n .as-title { font-weight: 600; font-size: 0.88rem; }\n .as-desc { font-size: 0.78rem; color: #6c757d; }\n .as-badge { font-size: 0.72rem; padding: 0.15em 0.5em; border-radius: 3px; flex-shrink: 0; }\n .as-chevron { color: #ced4da; font-size: 0.8rem; flex-shrink: 0; }\n </style>\n\n <div class=\"as-section-label\">Authentication</div>\n\n <div class=\"as-item\" data-action=\"send-password-reset\">\n <div class=\"as-icon bg-primary bg-opacity-10 text-primary\"><i class=\"bi bi-envelope\"></i></div>\n <div class=\"as-info\">\n <div class=\"as-title\">Send Password Reset</div>\n <div class=\"as-desc\">Send a password reset email to {{model.email}}</div>\n </div>\n <span class=\"as-badge bg-light text-muted border\">Send</span>\n </div>\n\n <div class=\"as-item\" data-action=\"send-magic-link\">\n <div class=\"as-icon\" style=\"background: rgba(13,110,253,0.1); color: #0d6efd;\"><i class=\"bi bi-link-45deg\"></i></div>\n <div class=\"as-info\">\n <div class=\"as-title\">Send Magic Login Link</div>\n <div class=\"as-desc\">Send a one-click login link to {{model.email}}</div>\n </div>\n <span class=\"as-badge bg-light text-muted border\">Send</span>\n </div>\n\n {{^model.is_email_verified|bool}}\n <div class=\"as-item\" data-action=\"send-email-verification\">\n <div class=\"as-icon\" style=\"background: rgba(25,135,84,0.1); color: #198754;\"><i class=\"bi bi-envelope-check\"></i></div>\n <div class=\"as-info\">\n <div class=\"as-title\">Send Email Verification</div>\n <div class=\"as-desc\">Send a verification email to {{model.email}}</div>\n </div>\n <span class=\"as-badge bg-light text-muted border\">Send</span>\n </div>\n {{/model.is_email_verified|bool}}\n\n <div class=\"as-item\" data-action=\"set-password\">\n <div class=\"as-icon bg-warning bg-opacity-10 text-warning\"><i class=\"bi bi-key\"></i></div>\n <div class=\"as-info\">\n <div class=\"as-title\">Set Password</div>\n <div class=\"as-desc\">Set a new password directly for this user</div>\n </div>\n <span class=\"as-badge bg-light text-muted border\">Set</span>\n </div>\n\n <div class=\"as-section-label\">Multi-Factor Authentication</div>\n\n <div class=\"as-item\" data-action=\"toggle-mfa\">\n <div class=\"as-icon\" style=\"background: rgba(111,66,193,0.1); color: #6f42c1;\"><i class=\"bi bi-shield-lock\"></i></div>\n <div class=\"as-info\">\n <div class=\"as-title\">MFA Requirement</div>\n <div class=\"as-desc\">\n {{#model.requires_mfa|bool}}User is required to use MFA{{/model.requires_mfa|bool}}\n {{^model.requires_mfa|bool}}MFA is not required for this user{{/model.requires_mfa|bool}}\n </div>\n </div>\n {{#model.requires_mfa|bool}}\n <span class=\"as-badge bg-success bg-opacity-10 text-success border\">Enabled</span>\n {{/model.requires_mfa|bool}}\n {{^model.requires_mfa|bool}}\n <span class=\"as-badge bg-light text-muted border\">Disabled</span>\n {{/model.requires_mfa|bool}}\n </div>\n\n <div class=\"as-item\" data-action=\"manage-passkeys\">\n <div class=\"as-icon bg-success bg-opacity-10 text-success\"><i class=\"bi bi-fingerprint\"></i></div>\n <div class=\"as-info\">\n <div class=\"as-title\">Passkeys</div>\n <div class=\"as-desc\">View and manage registered passkeys</div>\n </div>\n <i class=\"bi bi-chevron-right as-chevron\"></i>\n </div>\n\n {{#model.requires_mfa|bool}}\n <div class=\"as-item\" data-action=\"view-recovery-codes\">\n <div class=\"as-icon\" style=\"background: rgba(111,66,193,0.1); color: #6f42c1;\"><i class=\"bi bi-file-earmark-lock\"></i></div>\n <div class=\"as-info\">\n <div class=\"as-title\">Recovery Codes</div>\n <div class=\"as-desc\">View remaining recovery codes</div>\n </div>\n <i class=\"bi bi-chevron-right as-chevron\"></i>\n </div>\n\n <div class=\"as-item\" data-action=\"disable-totp\">\n <div class=\"as-icon\" style=\"background: rgba(220,53,69,0.1); color: #dc3545;\"><i class=\"bi bi-shield-x\"></i></div>\n <div class=\"as-info\">\n <div class=\"as-title\">Disable Authenticator</div>\n <div class=\"as-desc\">Remove TOTP requirement for this user</div>\n </div>\n </div>\n {{/model.requires_mfa|bool}}\n\n <div class=\"as-section-label\">Sessions</div>\n\n <div class=\"as-item\" data-action=\"revoke-all-sessions\">\n <div class=\"as-icon\" style=\"background: rgba(220,53,69,0.1); color: #dc3545;\"><i class=\"bi bi-box-arrow-right\"></i></div>\n <div class=\"as-info\">\n <div class=\"as-title\">Revoke All Sessions</div>\n <div class=\"as-desc\">Force sign-out from all devices</div>\n </div>\n </div>\n `,\n ...options\n });\n }\n\n // ── Password actions ────────────────────────────\n\n async onActionSendPasswordReset() {\n const app = this.getApp();\n const email = this.model.get('email');\n\n const confirmed = await Dialog.confirm(\n `Send a password reset email to <strong>${email}</strong>?`,\n 'Send Password Reset'\n );\n if (!confirmed) return true;\n\n const resp = await rest.POST('/api/auth/password/reset', { email });\n if (resp.success) {\n app?.toast?.success('Password reset email sent');\n } else {\n app?.toast?.error(resp.message || 'Failed to send password reset');\n }\n return true;\n }\n\n async onActionSendEmailVerification() {\n const app = this.getApp();\n const email = this.model.get('email');\n\n const confirmed = await Dialog.confirm(\n `Send a verification email to <strong>${email}</strong>?`,\n 'Send Email Verification'\n );\n if (!confirmed) return true;\n\n const resp = await rest.POST('/api/auth/email/verify', { email });\n if (resp.success) {\n app?.toast?.success('Verification email sent');\n } else {\n app?.toast?.error(resp.message || 'Failed to send verification email');\n }\n return true;\n }\n\n async onActionSendMagicLink() {\n const app = this.getApp();\n const email = this.model.get('email');\n\n const confirmed = await Dialog.confirm(\n `Send a magic login link to <strong>${email}</strong>? They will be able to sign in with one click.`,\n 'Send Magic Login Link'\n );\n if (!confirmed) return true;\n\n const resp = await rest.POST('/api/auth/magic-link', { email });\n if (resp.success) {\n app?.toast?.success('Magic login link sent');\n } else {\n app?.toast?.error(resp.message || 'Failed to send magic link');\n }\n return true;\n }\n\n async onActionSetPassword() {\n const app = this.getApp();\n const data = await Dialog.showForm({\n title: 'Set Password',\n size: 'sm',\n fields: [\n { name: 'password', type: 'password', label: 'New Password', required: true, cols: 12, help: 'Set a new password for this user.' },\n { name: 'confirm', type: 'password', label: 'Confirm Password', required: true, cols: 12 }\n ]\n });\n if (!data) return true;\n\n if (data.password !== data.confirm) {\n app?.toast?.error('Passwords do not match');\n return true;\n }\n\n const resp = await this.model.save({ password: data.password });\n if (resp.status === 200) {\n app?.toast?.success('Password updated');\n } else {\n app?.toast?.error(resp.message || 'Failed to set password');\n }\n return true;\n }\n\n // ── MFA actions ─────────────────────────────────\n\n async onActionToggleMfa() {\n const app = this.getApp();\n const currentMfa = this.model.get('requires_mfa');\n const action = currentMfa ? 'disable' : 'enable';\n\n const confirmed = await Dialog.confirm(\n `${currentMfa ? 'Disable' : 'Enable'} MFA requirement for this user?`,\n `${currentMfa ? 'Disable' : 'Enable'} MFA`\n );\n if (!confirmed) return true;\n\n const resp = await this.model.save({ requires_mfa: !currentMfa });\n if (resp.status === 200) {\n app?.toast?.success(`MFA ${action}d`);\n await this.render();\n } else {\n app?.toast?.error(resp.message || `Failed to ${action} MFA`);\n }\n return true;\n }\n\n async onActionDisableTotp() {\n const app = this.getApp();\n const confirmed = await Dialog.confirm(\n 'Disable the authenticator app for this user? They will no longer need a TOTP code to sign in.',\n 'Disable Authenticator'\n );\n if (!confirmed) return true;\n\n const resp = await rest.DELETE(`/api/user/${this.model.id}/totp`);\n if (resp.success) {\n this.model.set('requires_mfa', false);\n app?.toast?.success('Authenticator disabled');\n await this.render();\n } else {\n app?.toast?.error(resp.message || 'Failed to disable authenticator');\n }\n return true;\n }\n\n async onActionManagePasskeys() {\n const collection = new PasskeyList({ params: { user: this.model.id } });\n try {\n await collection.fetch();\n } catch (e) {\n // ignore\n }\n\n const items = collection.models || [];\n const view = new View({\n template: `\n <style>\n .pk-row { display: flex; align-items: center; gap: 0.75rem; padding: 0.65rem 0.75rem; border: 1px solid #f0f0f0; border-radius: 8px; margin-bottom: 0.4rem; }\n .pk-icon { width: 32px; height: 32px; background: #e7f1ff; border-radius: 6px; display: flex; align-items: center; justify-content: center; color: #0d6efd; font-size: 0.9rem; flex-shrink: 0; }\n .pk-info { flex: 1; min-width: 0; }\n .pk-name { font-weight: 600; font-size: 0.85rem; }\n .pk-meta { font-size: 0.73rem; color: #6c757d; }\n .pk-actions .btn { padding: 0.2rem 0.4rem; font-size: 0.75rem; }\n .pk-empty { text-align: center; padding: 2rem 1rem; color: #6c757d; }\n .pk-empty i { font-size: 2rem; color: #ced4da; display: block; margin-bottom: 0.5rem; }\n </style>\n {{#passkeys}}\n <div class=\"pk-row\">\n <div class=\"pk-icon\"><i class=\"bi bi-fingerprint\"></i></div>\n <div class=\"pk-info\">\n <div class=\"pk-name\">{{.friendly_name|default:'Unnamed Passkey'}}</div>\n <div class=\"pk-meta\">Created {{.created|date}} &middot; Last used {{.last_used|relative|default:'never'}} &middot; {{.sign_count}} uses</div>\n </div>\n <div class=\"pk-actions\">\n <button type=\"button\" class=\"btn btn-outline-secondary\" data-action=\"edit-passkey\" data-id=\"{{.id}}\" title=\"Edit\"><i class=\"bi bi-pencil\"></i></button>\n <button type=\"button\" class=\"btn btn-outline-danger\" data-action=\"delete-passkey\" data-id=\"{{.id}}\" title=\"Delete\"><i class=\"bi bi-trash\"></i></button>\n </div>\n </div>\n {{/passkeys}}\n {{^passkeys|bool}}\n <div class=\"pk-empty\">\n <i class=\"bi bi-fingerprint\"></i>\n No passkeys registered\n </div>\n {{/passkeys|bool}}\n `\n });\n view.passkeys = items.map(p => p.toJSON ? p.toJSON() : p);\n\n view.onActionEditPasskey = async (event, el) => {\n const id = el.dataset.id;\n const passkey = items.find(p => String(p.id) === String(id));\n if (passkey) {\n await Dialog.showModelForm({ title: 'Edit Passkey', model: passkey, fields: PasskeyForms.edit.fields, size: 'sm' });\n }\n return true;\n };\n\n view.onActionDeletePasskey = async (event, el) => {\n const id = el.dataset.id;\n const confirmed = await Dialog.confirm('Delete this passkey?', 'Delete Passkey');\n if (confirmed) {\n const passkey = items.find(p => String(p.id) === String(id));\n if (passkey) {\n await passkey.destroy();\n this.getApp()?.toast?.success('Passkey deleted');\n }\n }\n return true;\n };\n\n await Dialog.showDialog({\n title: 'Passkeys',\n body: view,\n size: 'md',\n buttons: [{ text: 'Close', class: 'btn-outline-secondary', dismiss: true }]\n });\n return true;\n }\n\n async onActionViewRecoveryCodes() {\n const app = this.getApp();\n const resp = await rest.GET(`/api/user/${this.model.id}/totp/recovery-codes`, {}, { dataOnly: true });\n if (!resp.success || !resp.data) {\n app?.toast?.error(resp.message || 'Failed to load recovery codes');\n return true;\n }\n\n const { remaining, codes } = resp.data;\n const view = new View({\n template: `\n <style>\n .rc-info { font-size: 0.82rem; color: #6c757d; margin-bottom: 1rem; }\n .rc-remaining { font-weight: 600; color: #495057; }\n .rc-list { display: grid; grid-template-columns: 1fr 1fr; gap: 0.3rem; }\n .rc-code { font-family: monospace; font-size: 0.85rem; padding: 0.35rem 0.6rem; background: #f8f9fa; border: 1px solid #e9ecef; border-radius: 4px; text-align: center; }\n </style>\n <div class=\"rc-info\"><span class=\"rc-remaining\">{{remaining}}</span> recovery codes remaining</div>\n <div class=\"rc-list\">\n {{#codes}}<div class=\"rc-code\">{{.}}</div>{{/codes}}\n </div>\n `\n });\n view.remaining = remaining;\n view.codes = codes || [];\n\n await Dialog.showDialog({\n title: 'Recovery Codes',\n body: view,\n size: 'sm',\n buttons: [{ text: 'Close', class: 'btn-outline-secondary', dismiss: true }]\n });\n return true;\n }\n\n // ── Session actions ─────────────────────────────\n\n async onActionRevokeAllSessions() {\n const app = this.getApp();\n const confirmed = await Dialog.confirm(\n 'Revoke all sessions for this user? They will be signed out of all devices immediately.',\n 'Revoke All Sessions'\n );\n if (!confirmed) return true;\n\n const resp = await rest.POST(`/api/user/${this.model.id}/sessions/revoke`);\n if (resp.success) {\n app?.toast?.success('All sessions revoked');\n } else {\n app?.toast?.error(resp.message || 'Failed to revoke sessions');\n }\n return true;\n }\n}\n","/**\n * AdminConnectedSection - Admin view of a user's OAuth connections\n *\n * View linked providers and unlink on behalf of user.\n * Uses admin-scoped endpoints.\n */\nimport View from '@core/View.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\nimport rest from '@core/Rest.js';\n\nconst PROVIDER_ICONS = {\n google: 'bi-google',\n github: 'bi-github',\n microsoft: 'bi-microsoft',\n apple: 'bi-apple',\n facebook: 'bi-facebook',\n twitter: 'bi-twitter-x',\n linkedin: 'bi-linkedin'\n};\n\nexport default class AdminConnectedSection extends View {\n constructor(options = {}) {\n super({\n className: 'admin-connected-section',\n template: `\n <style>\n .ac-row { display: flex; align-items: center; gap: 0.85rem; padding: 0.85rem 1rem; border: 1px solid #f0f0f0; border-radius: 8px; margin-bottom: 0.5rem; }\n .ac-icon { width: 36px; height: 36px; border-radius: 8px; display: flex; align-items: center; justify-content: center; font-size: 1rem; flex-shrink: 0; background: #f0f0f0; color: #495057; }\n .ac-info { flex: 1; min-width: 0; }\n .ac-provider { font-weight: 600; font-size: 0.88rem; text-transform: capitalize; }\n .ac-meta { font-size: 0.78rem; color: #6c757d; }\n .ac-actions .btn { font-size: 0.75rem; padding: 0.25rem 0.5rem; }\n .ac-empty { text-align: center; padding: 2rem 1rem; color: #6c757d; }\n .ac-empty i { font-size: 2rem; color: #ced4da; display: block; margin-bottom: 0.5rem; }\n </style>\n\n {{#connections}}\n <div class=\"ac-row\">\n <div class=\"ac-icon\"><i class=\"bi {{.icon}}\"></i></div>\n <div class=\"ac-info\">\n <div class=\"ac-provider\">{{.provider}}</div>\n <div class=\"ac-meta\">{{.email}} &middot; Connected {{.created|relative}}</div>\n </div>\n <div class=\"ac-actions\">\n <button type=\"button\" class=\"btn btn-outline-danger\" data-action=\"unlink\" data-id=\"{{.id}}\" title=\"Unlink\"><i class=\"bi bi-x-lg me-1\"></i>Unlink</button>\n </div>\n </div>\n {{/connections}}\n {{^connections|bool}}\n <div class=\"ac-empty\">\n <i class=\"bi bi-plug\"></i>\n No connected accounts\n </div>\n {{/connections|bool}}\n `,\n ...options\n });\n this.connections = [];\n }\n\n async onBeforeRender() {\n try {\n const resp = await rest.GET(`/api/user/${this.model.id}/oauth_connection`);\n const results = resp?.data?.results || resp?.data || [];\n this.connections = results.map(c => ({\n ...c,\n icon: PROVIDER_ICONS[c.provider] || 'bi-link-45deg'\n }));\n } catch (e) {\n this.connections = [];\n }\n }\n\n async onActionUnlink(event, el) {\n const id = el.dataset.id;\n const connection = this.connections.find(c => String(c.id) === String(id));\n const provider = connection?.provider || 'this account';\n\n const confirmed = await Dialog.confirm(\n `Unlink ${provider} for this user?`,\n 'Unlink Account'\n );\n if (!confirmed) return true;\n\n const resp = await rest.DELETE(`/api/user/${this.model.id}/oauth_connection/${id}`);\n if (resp.success) {\n this.getApp()?.toast?.success(`${provider} account unlinked`);\n await this.render();\n } else {\n this.getApp()?.toast?.error(resp.message || 'Failed to unlink account');\n }\n return true;\n }\n}\n","/**\n * AdminNotificationsSection - Admin view/edit of user notification preferences\n *\n * Per-kind, per-channel toggle grid. Uses admin-scoped endpoints.\n */\nimport View from '@core/View.js';\nimport rest from '@core/Rest.js';\n\nconst CHANNEL_LABELS = {\n in_app: 'In-App',\n email: 'Email',\n push: 'Push'\n};\n\nconst CHANNELS = ['in_app', 'email', 'push'];\n\nexport default class AdminNotificationsSection extends View {\n constructor(options = {}) {\n super({\n className: 'admin-notifications-section',\n template: `\n <style>\n .an-table { width: 100%; border-collapse: collapse; }\n .an-table th { font-size: 0.7rem; font-weight: 700; text-transform: uppercase; letter-spacing: 0.06em; color: #adb5bd; padding: 0.5rem 0.75rem; border-bottom: 2px solid #e9ecef; }\n .an-table th:first-child { text-align: left; }\n .an-table th:not(:first-child) { text-align: center; width: 80px; }\n .an-table td { padding: 0.65rem 0.75rem; border-bottom: 1px solid #f0f0f0; }\n .an-table td:first-child { font-size: 0.88rem; font-weight: 500; text-transform: capitalize; }\n .an-table td:not(:first-child) { text-align: center; }\n .an-table tr:last-child td { border-bottom: none; }\n .an-empty { text-align: center; padding: 2rem 1rem; color: #6c757d; }\n .an-empty i { font-size: 2rem; color: #ced4da; display: block; margin-bottom: 0.5rem; }\n </style>\n\n {{#hasPreferences|bool}}\n <table class=\"an-table\">\n <thead>\n <tr>\n <th>Type</th>\n {{#channels}}\n <th>{{.label}}</th>\n {{/channels}}\n </tr>\n </thead>\n <tbody>\n {{#preferenceRows}}\n <tr>\n <td>{{.kindLabel}}</td>\n {{#.toggles}}\n <td>\n <input type=\"checkbox\" class=\"form-check-input\"\n data-action=\"toggle-pref\"\n data-kind=\"{{.kind}}\"\n data-channel=\"{{.channel}}\"\n {{#.checked}}checked{{/.checked}}>\n </td>\n {{/.toggles}}\n </tr>\n {{/preferenceRows}}\n </tbody>\n </table>\n {{/hasPreferences|bool}}\n {{^hasPreferences|bool}}\n <div class=\"an-empty\">\n <i class=\"bi bi-bell\"></i>\n No notification preferences configured\n </div>\n {{/hasPreferences|bool}}\n `,\n ...options\n });\n this.preferences = {};\n }\n\n get channels() {\n return CHANNELS.map(ch => ({ key: ch, label: CHANNEL_LABELS[ch] || ch }));\n }\n\n get hasPreferences() {\n return Object.keys(this.preferences).length > 0;\n }\n\n get preferenceRows() {\n return Object.keys(this.preferences).sort().map(kind => ({\n kind,\n kindLabel: kind.replace(/[_-]/g, ' ').replace(/\\b\\w/g, c => c.toUpperCase()),\n toggles: CHANNELS.map(channel => ({\n kind,\n channel,\n checked: this.preferences[kind]?.[channel] !== false\n }))\n }));\n }\n\n async onBeforeRender() {\n try {\n const resp = await rest.GET(`/api/user/${this.model.id}/notification/preferences`, {}, { dataOnly: true });\n this.preferences = resp?.data?.preferences || resp?.data || {};\n } catch (e) {\n this.preferences = {};\n }\n }\n\n async onActionTogglePref(event, el) {\n const kind = el.dataset.kind;\n const channel = el.dataset.channel;\n const checked = el.checked;\n\n if (!this.preferences[kind]) {\n this.preferences[kind] = {};\n }\n this.preferences[kind][channel] = checked;\n\n try {\n const resp = await rest.POST(`/api/user/${this.model.id}/notification/preferences`, {\n preferences: { [kind]: { [channel]: checked } }\n });\n if (!resp.success) {\n this.getApp()?.toast?.error(resp.message || 'Failed to update preference');\n el.checked = !checked;\n }\n } catch (e) {\n this.getApp()?.toast?.error('Failed to update preference');\n el.checked = !checked;\n }\n return true;\n }\n}\n","/**\n * AdminApiKeysSection - Admin view/revoke of a user's API keys\n *\n * View-only table of API keys with revoke/delete capability.\n * Admin cannot generate keys on behalf of users — only view and revoke.\n * Uses admin-scoped endpoints.\n */\nimport View from '@core/View.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\nimport rest from '@core/Rest.js';\n\nexport default class AdminApiKeysSection extends View {\n constructor(options = {}) {\n super({\n className: 'admin-api-keys-section',\n template: `\n <style>\n .aak-list { border: 1px solid #e9ecef; border-radius: 8px; overflow: hidden; }\n .aak-item { display: flex; align-items: center; padding: 0.75rem 1rem; border-bottom: 1px solid #f0f0f0; gap: 1rem; }\n .aak-item:last-child { border-bottom: none; }\n .aak-item-icon { color: #6c757d; font-size: 1.1rem; flex-shrink: 0; }\n .aak-item-info { flex: 1; min-width: 0; }\n .aak-item-name { font-weight: 600; font-size: 0.85rem; display: flex; align-items: center; gap: 0.5rem; }\n .aak-item-meta { font-size: 0.75rem; color: #6c757d; display: flex; gap: 1rem; flex-wrap: wrap; margin-top: 0.15rem; }\n .aak-item-meta i { margin-right: 0.2rem; }\n .aak-empty { padding: 2rem; text-align: center; color: #6c757d; font-size: 0.85rem; }\n .aak-empty i { font-size: 1.5rem; display: block; margin-bottom: 0.5rem; }\n </style>\n\n <div id=\"aak-keys-list\"></div>\n `,\n ...options\n });\n this.apiKeys = [];\n }\n\n async onBeforeRender() {\n await this._loadKeys();\n }\n\n async _loadKeys() {\n try {\n const resp = await rest.GET(`/api/user/${this.model.id}/api_keys`, {}, {}, { dataOnly: true });\n this.apiKeys = resp.success && Array.isArray(resp.data) ? resp.data : [];\n } catch (e) {\n this.apiKeys = [];\n }\n }\n\n onAfterRender() {\n this._renderKeysList();\n }\n\n _renderKeysList() {\n const container = this.element?.querySelector('#aak-keys-list');\n if (!container) return;\n\n if (!this.apiKeys.length) {\n container.innerHTML = `\n <div class=\"aak-list\">\n <div class=\"aak-empty\">\n <i class=\"bi bi-key\"></i>\n No API keys for this user\n </div>\n </div>`;\n return;\n }\n\n const rows = this.apiKeys.map(key => {\n const name = key.name || 'API Key';\n const created = key.created ? new Date(key.created * 1000).toLocaleDateString() : '';\n const expires = key.expires ? new Date(key.expires * 1000).toLocaleDateString() : 'Never';\n const lastUsed = key.last_used ? new Date(key.last_used * 1000).toLocaleDateString() : 'Never';\n const ips = key.allowed_ips?.length ? key.allowed_ips.join(', ') : 'Any';\n const isActive = key.is_active !== false;\n const statusBadge = isActive\n ? '<span class=\"badge bg-success\">Active</span>'\n : '<span class=\"badge bg-secondary\">Inactive</span>';\n const tokenPreview = key.token_prefix ? `${key.token_prefix}...` : '••••••••';\n\n return `\n <div class=\"aak-item\">\n <div class=\"aak-item-icon\"><i class=\"bi bi-key\"></i></div>\n <div class=\"aak-item-info\">\n <div class=\"aak-item-name\">${name} ${statusBadge}</div>\n <div class=\"aak-item-meta\">\n <span><i class=\"bi bi-code-square\"></i>${tokenPreview}</span>\n <span><i class=\"bi bi-calendar\"></i>Created ${created}</span>\n <span><i class=\"bi bi-clock\"></i>Expires ${expires}</span>\n <span><i class=\"bi bi-activity\"></i>Last used ${lastUsed}</span>\n <span><i class=\"bi bi-globe\"></i>IPs: ${ips}</span>\n </div>\n </div>\n <div>\n <button type=\"button\" class=\"btn btn-outline-danger btn-sm\" data-action=\"revoke-key\" data-id=\"${key.id}\" title=\"Revoke\">\n <i class=\"bi bi-trash\"></i>\n </button>\n </div>\n </div>`;\n }).join('');\n\n container.innerHTML = `<div class=\"aak-list\">${rows}</div>`;\n }\n\n async onActionRevokeKey(event, el) {\n const id = el.dataset.id;\n if (!id) return true;\n\n const confirmed = await Dialog.confirm(\n 'Revoke this API key? Any applications using it will lose access immediately.',\n 'Revoke API Key'\n );\n if (!confirmed) return true;\n\n const resp = await rest.DELETE(`/api/user/${this.model.id}/api_keys/${id}`, {}, {}, { dataOnly: true });\n if (resp.success) {\n this.getApp()?.toast?.success('API key revoked');\n await this._loadKeys();\n this._renderKeysList();\n } else {\n this.getApp()?.toast?.error(resp.message || 'Failed to revoke API key');\n }\n return true;\n }\n}\n","/**\n * UserView - Comprehensive user management interface\n *\n * Features:\n * - Header with avatar, name, contact info, status, and context menu\n * - Left-nav sidebar for section switching (Profile, Permissions, Groups, etc.)\n * - Integrated with DataView, TableView, FormView, and SideNavView\n * - Clean Bootstrap 5 styling\n */\n\nimport View from '@core/View.js';\nimport SideNavView from '@core/views/navigation/SideNavView.js';\nimport DataView from '@core/views/data/DataView.js';\nimport TableView from '@core/views/table/TableView.js';\nimport TableRow from '@core/views/table/TableRow.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport rest from '@core/Rest.js';\nimport { User, UserDataView, UserForms, UserDeviceList, UserDeviceLocationList } from '@core/models/User.js';\nimport { LogList } from '@core/models/Log.js';\nimport { IncidentEventList } from '@core/models/Incident.js';\nimport { MemberList } from '@core/models/Member.js';\nimport { PushDeviceList } from '@core/models/Push.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\nimport FormView from '@core/forms/FormView.js';\nimport AdminProfileSection from './sections/AdminProfileSection.js';\nimport AdminPersonalSection from './sections/AdminPersonalSection.js';\nimport AdminSecuritySection from './sections/AdminSecuritySection.js';\nimport AdminConnectedSection from './sections/AdminConnectedSection.js';\nimport AdminNotificationsSection from './sections/AdminNotificationsSection.js';\nimport AdminApiKeysSection from './sections/AdminApiKeysSection.js';\n// DeviceView and GeoIPView are opened automatically via VIEW_CLASS on row click\n\n\n// ── Custom TableRow classes for rich multiline rows ──\n\nclass DeviceRow extends TableRow {\n get deviceIcon() {\n const dev = this.model?.get('device_info')?.device || {};\n const os = this.model?.get('device_info')?.os || {};\n const isMobile = ['iPhone', 'Android'].some(m =>\n (dev.family || '').includes(m) || (os.family || '').includes(m)\n );\n return isMobile ? 'bi-phone' : 'bi-laptop';\n }\n get deviceName() {\n const dev = this.model?.get('device_info')?.device || {};\n return `${dev.brand || ''} ${dev.family || ''}`.trim() || 'Unknown Device';\n }\n get deviceModel() {\n return this.model?.get('device_info')?.device?.model || '';\n }\n get browserName() {\n const ua = this.model?.get('device_info')?.user_agent || {};\n return ua.family ? `${ua.family} ${ua.major || ''}`.trim() : '';\n }\n get osName() {\n const os = this.model?.get('device_info')?.os || {};\n return os.family ? `${os.family} ${os.major || ''}`.trim() : '';\n }\n get deviceMeta() {\n return [this.browserName, this.osName].filter(Boolean).join(' · ') || '—';\n }\n}\n\nclass LocationRow extends TableRow {\n get deviceIcon() {\n const dev = this.model?.get('user_device')?.device_info?.device || {};\n const os = this.model?.get('user_device')?.device_info?.os || {};\n const isMobile = ['iPhone', 'Android'].some(m =>\n (dev.family || '').includes(m) || (os.family || '').includes(m)\n );\n return isMobile ? 'bi-phone' : 'bi-laptop';\n }\n get browserName() {\n const ua = this.model?.get('user_device')?.device_info?.user_agent || {};\n return ua.family ? `${ua.family} ${ua.major || ''}`.trim() : 'Unknown';\n }\n get deviceName() {\n const dev = this.model?.get('user_device')?.device_info?.device || {};\n return `${dev.brand || ''} ${dev.family || ''}`.trim() || 'Unknown';\n }\n get locationText() {\n const geo = this.model?.get('geolocation') || {};\n return [geo.city, geo.region].filter(Boolean).join(', ') || geo.country_name || '—';\n }\n get countryName() {\n return this.model?.get('geolocation')?.country_name || '';\n }\n get threatFlags() {\n const geo = this.model?.get('geolocation') || {};\n const flags = [];\n if (geo.is_vpn) flags.push('<span class=\"badge bg-warning text-dark\" style=\"font-size:0.6rem;\">VPN</span>');\n if (geo.is_tor) flags.push('<span class=\"badge bg-danger\" style=\"font-size:0.6rem;\">Tor</span>');\n if (geo.is_proxy) flags.push('<span class=\"badge bg-warning text-dark\" style=\"font-size:0.6rem;\">Proxy</span>');\n return flags.join(' ');\n }\n get hasThreatFlags() {\n const geo = this.model?.get('geolocation') || {};\n return !!(geo.is_vpn || geo.is_tor || geo.is_proxy);\n }\n}\n\nclass UserView extends View {\n constructor(options = {}) {\n super({\n className: 'user-view',\n ...options\n });\n\n // User model instance\n this.model = options.model || new User(options.data || {});\n\n // Section views\n this.sideNavView = null;\n\n // Set template\n this.template = `\n <div class=\"user-view-container\">\n <!-- User Header + Context Menu -->\n <div class=\"d-flex justify-content-between align-items-start mb-4\">\n <div data-container=\"user-header\" style=\"flex: 1;\"></div>\n <div data-container=\"user-context-menu\" class=\"ms-3 flex-shrink-0\"></div>\n </div>\n <!-- Side Nav Container -->\n <div data-container=\"user-sidenav\" style=\"min-height: 400px;\"></div>\n </div>\n `;\n }\n\n async onInit() {\n // ── Header ──────────────────────────────────\n this.header = new View({\n containerId: 'user-header',\n template: `\n <div class=\"d-flex justify-content-between align-items-start\">\n <!-- Left Side: Primary Identity -->\n <div class=\"d-flex align-items-center gap-3\">\n {{{model.avatar|avatar('md','rounded-circle')}}}\n <div>\n <h3 class=\"mb-0\">{{model.display_name|default('Unnamed User')}}</h3>\n <a href=\"mailto:{{model.email}}\" class=\"text-decoration-none text-body\">{{model.email}}</a>{{{model.email|clipboard('icon-only')}}}\n {{#model.phone_number}}\n <div class=\"text-muted small mt-1\">{{{model.phone_number|phone(false)}}}</div>\n {{/model.phone_number}}\n </div>\n </div>\n\n <!-- Right Side: Status -->\n <div class=\"text-end\">\n <div class=\"d-flex align-items-center justify-content-end gap-3\">\n <span class=\"d-inline-flex align-items-center gap-1\" title=\"{{model.is_online|boolean('Online','Offline')}}\">\n <i class=\"bi bi-circle-fill {{model.is_online|boolean('text-success','text-secondary')}}\" style=\"font-size: 0.5rem;\"></i>\n <span class=\"small\">{{model.is_online|boolean('Online','Offline')}}</span>\n </span>\n <span class=\"d-inline-flex align-items-center gap-1\" style=\"cursor: pointer;\"\n data-action=\"toggle-active\"\n title=\"{{model.is_active|boolean('Click to deactivate','Click to activate')}}\">\n <i class=\"bi {{model.is_active|boolean('bi-toggle-on text-success','bi-toggle-off text-secondary')}}\" style=\"font-size: 1.1rem;\"></i>\n <span class=\"small\">{{model.is_active|boolean('Active','Inactive')}}</span>\n </span>\n </div>\n {{#model.last_activity}}\n <div class=\"text-muted small mt-1\">Last active {{model.last_activity|relative}}</div>\n {{/model.last_activity}}\n </div>\n </div>`\n });\n\n this.header.setModel(this.model);\n this.addChild(this.header);\n\n // ── Section views ───────────────────────────\n\n const profileView = new AdminProfileSection({ model: this.model });\n const personalView = new AdminPersonalSection({ model: this.model });\n const securityView = new AdminSecuritySection({ model: this.model });\n const connectedView = new AdminConnectedSection({ model: this.model });\n const notificationsView = new AdminNotificationsSection({ model: this.model });\n const apiKeysView = new AdminApiKeysSection({ model: this.model });\n\n const permsView = new FormView({\n fields: User.PERMISSION_FIELDS,\n model: this.model,\n autosaveModelField: true\n });\n\n // Groups\n const membersCollection = new MemberList({\n params: { user: this.model.get('id'), size: 5 }\n });\n const groupsView = new TableView({\n collection: membersCollection,\n hideActivePillNames: ['user'],\n columns: [\n { key: 'created', label: 'Date Joined', formatter: 'date', sortable: true },\n { key: 'group.name', label: 'Group Name', sortable: true },\n { key: 'permissions|keys|badge', label: 'Permissions' }\n ]\n });\n\n // Events\n const eventsCollection = new IncidentEventList({\n params: { size: 5, model_name: \"account.User\", model_id: this.model.get('id') }\n });\n const eventsView = new TableView({\n collection: eventsCollection,\n hideActivePillNames: ['model_name', 'model_id'],\n columns: [\n { key: 'id', label: 'ID', sortable: true, width: '40px' },\n { key: 'created', label: 'Date', formatter: 'datetime', sortable: true, width: '150px' },\n { key: 'category|badge', label: 'Category' },\n { key: 'title', label: 'Title' }\n ]\n });\n\n // Devices — multiline rows with detail dialog\n const devicesView = new TableView({\n collection: new UserDeviceList({ params: { size: 10, user: this.model.get('id') } }),\n hideActivePillNames: ['user'],\n clickAction: 'view',\n itemClass: DeviceRow,\n columns: [\n {\n key: 'device_info',\n label: 'Device',\n template: `\n <div style=\"font-size:0.85rem; font-weight:500;\">\n <i class=\"bi {{deviceIcon}} text-muted me-1\" style=\"font-size:1.1rem; vertical-align:middle;\"></i>{{deviceName}}\n {{#deviceModel}} <span class=\"text-muted fw-normal\">({{deviceModel}})</span>{{/deviceModel}}\n </div>\n <div style=\"font-size:0.73rem; color:#6c757d; margin-top:0.15rem;\">\n {{deviceMeta}}\n {{#model.last_ip}} <span class=\"text-muted mx-1\">&middot;</span> {{model.last_ip}}{{/model.last_ip}}\n </div>`\n },\n { key: 'first_seen', label: 'First Seen', formatter: 'epoch|relative', width: '120px' },\n { key: 'last_seen', label: 'Last Seen', formatter: 'epoch|relative', width: '120px' }\n ]\n });\n\n // Locations — multiline rows with detail dialog\n const locationsView = new TableView({\n collection: new UserDeviceLocationList({ params: { size: 10, user: this.model.get('id') } }),\n hideActivePillNames: ['user'],\n clickAction: 'view',\n itemClass: LocationRow,\n columns: [\n {\n key: 'user_device',\n label: 'Session',\n template: `\n <div style=\"font-size:0.85rem; font-weight:500;\">\n <i class=\"bi {{deviceIcon}} text-muted me-1\" style=\"font-size:1.1rem; vertical-align:middle;\"></i>{{browserName}} <span class=\"text-muted fw-normal\">on</span> {{deviceName}}\n </div>\n <div style=\"font-size:0.73rem; color:#6c757d; margin-top:0.15rem;\">\n <i class=\"bi bi-geo-alt me-1\"></i>{{locationText}}\n {{#countryName}} <span class=\"text-muted mx-1\">&middot;</span> {{countryName}}{{/countryName}}\n {{#model.ip_address}} <span class=\"text-muted mx-1\">&middot;</span> {{model.ip_address}}{{/model.ip_address}}\n {{#hasThreatFlags|bool}} <span class=\"ms-1\">{{{threatFlags}}}</span>{{/hasThreatFlags|bool}}\n </div>`\n },\n { key: 'last_seen', label: 'Last Seen', formatter: 'epoch|relative', width: '120px' }\n ]\n });\n\n // Push Devices\n const pushDevices = new PushDeviceList({\n params: { size: 5, user: this.model.get('id') }\n });\n const pushDevicesView = new TableView({\n collection: pushDevices,\n hideActivePillNames: ['user'],\n columns: [\n { key: 'duid|truncate_middle(16)', label: 'Device ID', sortable: true },\n { key: 'device_info.user_agent.family', label: 'Browser', formatter: \"default('—')\" },\n { key: 'device_info.os.family', label: 'OS', formatter: \"default('—')\" },\n { key: 'first_seen', label: 'First Seen', formatter: \"epoch|datetime\" },\n { key: 'last_seen', label: 'Last Seen', formatter: \"epoch|datetime\" }\n ],\n size: 5\n });\n\n // Logs (model-scoped)\n const logsCollection = new LogList({\n params: { size: 5, model_name: \"account.User\", model_id: this.model.get('id') }\n });\n const logsView = new TableView({\n collection: logsCollection,\n permissions: 'view_logs',\n hideActivePillNames: ['model_name', 'model_id'],\n columns: [\n {\n key: 'created', label: 'Timestamp', sortable: true, formatter: \"epoch|datetime\",\n filter: { name: \"created\", type: 'daterange', startName: 'dr_start', endName: 'dr_end', fieldName: 'dr_field', label: 'Date Range', format: 'YYYY-MM-DD', displayFormat: 'MMM DD, YYYY', separator: ' to ' }\n },\n {\n key: 'level', label: 'Level', sortable: true,\n filter: { type: 'select', options: [{ value: 'info', label: 'Info' }, { value: 'warning', label: 'Warning' }, { value: 'error', label: 'Error' }] }\n },\n { key: 'kind', label: 'Kind', filter: { type: 'text' } },\n { name: 'log', label: 'Log' }\n ]\n });\n\n // Activity (user-scoped logs)\n const activityCollection = new LogList({\n params: { size: 5, uid: this.model.get('id') }\n });\n const activityView = new TableView({\n collection: activityCollection,\n hideActivePillNames: ['uid'],\n permissions: 'view_logs',\n columns: [\n {\n key: 'created', label: 'Timestamp', sortable: true, formatter: \"epoch|datetime\",\n filter: { name: \"created\", type: 'daterange', startName: 'dr_start', endName: 'dr_end', fieldName: 'dr_field', label: 'Date Range', format: 'YYYY-MM-DD', displayFormat: 'MMM DD, YYYY', separator: ' to ' }\n },\n {\n key: 'level', label: 'Level', sortable: true,\n filter: { type: 'select', options: [{ value: 'info', label: 'Info' }, { value: 'warning', label: 'Warning' }, { value: 'error', label: 'Error' }] }\n },\n { key: 'kind', label: 'Kind', filter: { type: 'text' } },\n { name: 'path', label: 'Path' }\n ]\n });\n\n // ── SideNavView ─────────────────────────────\n this.sideNavView = new SideNavView({\n containerId: 'user-sidenav',\n activeSection: 'profile',\n navWidth: 200,\n contentPadding: '1.25rem 2rem',\n enableResponsive: true,\n minWidth: 500,\n sections: [\n { key: 'profile', label: 'Profile', icon: 'bi-person', view: profileView },\n { key: 'personal', label: 'Personal', icon: 'bi-person-vcard', view: personalView },\n { key: 'security', label: 'Security', icon: 'bi-shield-lock', view: securityView },\n { key: 'connected', label: 'Connected', icon: 'bi-plug', view: connectedView },\n { type: 'divider', label: 'Access' },\n { key: 'permissions', label: 'Permissions', icon: 'bi-shield-check', view: permsView },\n { key: 'groups', label: 'Groups', icon: 'bi-people', view: groupsView },\n { key: 'api_keys', label: 'API Keys', icon: 'bi-key', view: apiKeysView },\n { type: 'divider', label: 'Activity' },\n { key: 'events', label: 'Events', icon: 'bi-calendar-event', view: eventsView },\n { key: 'activity', label: 'Activity Log', icon: 'bi-clock-history', view: activityView, permissions: 'view_logs' },\n { key: 'logs', label: 'Object Logs', icon: 'bi-journal-text', view: logsView, permissions: 'view_logs' },\n { type: 'divider', label: 'Devices' },\n { key: 'devices', label: 'Devices', icon: 'bi-laptop', view: devicesView },\n { key: 'locations', label: 'Locations', icon: 'bi-geo-alt', view: locationsView },\n { key: 'push_devices', label: 'Push Devices', icon: 'bi-phone', view: pushDevicesView },\n { type: 'divider', label: 'Settings' },\n { key: 'notifications', label: 'Notifications', icon: 'bi-bell', view: notificationsView }\n ]\n });\n this.addChild(this.sideNavView);\n\n // ── Context Menu ────────────────────────────\n const userMenu = new ContextMenu({\n containerId: 'user-context-menu',\n className: \"context-menu-view header-menu-absolute\",\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: [\n { label: 'Edit User', action: 'edit-user', icon: 'bi-pencil' },\n ...(this.model.get('avatar')\n ? [{ label: 'Clear Avatar', action: 'clear-avatar', icon: 'bi-person-x' }]\n : []),\n { type: 'divider' },\n { label: 'Send Password Reset', action: 'send-password-reset', icon: 'bi-envelope' },\n { label: 'Send Magic Login Link', action: 'send-magic-link', icon: 'bi-link-45deg' },\n { label: 'Revoke All Sessions', action: 'revoke-all-sessions', icon: 'bi-box-arrow-right' },\n { type: 'divider' },\n ...(this.model.get('is_email_verified')\n ? []\n : [\n { label: 'Send Email Verification', action: 'send-email-verification', icon: 'bi-envelope-check' },\n { label: 'Force Verify Email', action: 'force-verify-email', icon: 'bi-patch-check' },\n ]),\n ...(this.model.get('phone_number') && !this.model.get('is_phone_verified')\n ? [{ label: 'Force Verify Phone', action: 'force-verify-phone', icon: 'bi-patch-check' }]\n : []),\n { type: 'divider' },\n this.model.get('is_active')\n ? { label: 'Deactivate User', action: 'deactivate-user', icon: 'bi-person-dash' }\n : { label: 'Activate User', action: 'activate-user', icon: 'bi-person-check' },\n ]\n }\n });\n this.addChild(userMenu);\n }\n\n // ── Context menu actions ────────────────────────\n\n async onActionEditUser() {\n await Dialog.showModelForm({\n title: `EDIT - #${this.model.id} ${this.options.modelName}`,\n model: this.model,\n formConfig: UserForms.edit,\n });\n }\n\n async onActionClearAvatar() {\n const confirmed = await Dialog.confirm(\n 'Remove this user\\'s avatar? They will see the default placeholder.',\n 'Clear Avatar'\n );\n if (!confirmed) return true;\n\n const resp = await this.model.save({ avatar: null });\n if (resp.status === 200) {\n this.getApp().toast.success('Avatar cleared');\n } else {\n this.getApp().toast.error('Failed to clear avatar');\n }\n return true;\n }\n\n async onActionSendPasswordReset() {\n const email = this.model.get('email');\n const confirmed = await Dialog.confirm(\n `Send a password reset email to <strong>${email}</strong>?`,\n 'Send Password Reset'\n );\n if (!confirmed) return true;\n\n const resp = await rest.POST('/api/auth/password/reset', { email });\n if (resp.success) {\n this.getApp().toast.success('Password reset email sent');\n } else {\n this.getApp().toast.error(resp.message || 'Failed to send password reset');\n }\n return true;\n }\n\n async onActionSendMagicLink() {\n const email = this.model.get('email');\n const confirmed = await Dialog.confirm(\n `Send a magic login link to <strong>${email}</strong>?`,\n 'Send Magic Login Link'\n );\n if (!confirmed) return true;\n\n const resp = await rest.POST('/api/auth/magic-link', { email });\n if (resp.success) {\n this.getApp().toast.success('Magic login link sent');\n } else {\n this.getApp().toast.error(resp.message || 'Failed to send magic link');\n }\n return true;\n }\n\n async onActionRevokeAllSessions() {\n const confirmed = await Dialog.confirm(\n 'Revoke all sessions? The user will be signed out of all devices immediately.',\n 'Revoke All Sessions'\n );\n if (!confirmed) return true;\n\n const resp = await rest.POST(`/api/user/${this.model.id}/sessions/revoke`);\n if (resp.success) {\n this.getApp().toast.success('All sessions revoked');\n } else {\n this.getApp().toast.error(resp.message || 'Failed to revoke sessions');\n }\n return true;\n }\n\n async onActionSendEmailVerification() {\n const email = this.model.get('email');\n const confirmed = await Dialog.confirm(\n `Send a verification email to <strong>${email}</strong>?`,\n 'Send Email Verification'\n );\n if (!confirmed) return true;\n\n const resp = await rest.POST('/api/auth/email/verify', { email });\n if (resp.success) {\n this.getApp().toast.success('Verification email sent');\n } else {\n this.getApp().toast.error(resp.message || 'Failed to send verification email');\n }\n return true;\n }\n\n async onActionForceVerifyEmail() {\n const confirmed = await Dialog.confirm(\n `Mark <strong>${this.model.get('email')}</strong> as verified?`,\n 'Force Verify Email'\n );\n if (!confirmed) return true;\n\n const resp = await this.model.save({ is_email_verified: true });\n if (resp.status === 200) {\n this.getApp().toast.success('Email marked as verified');\n } else {\n this.getApp().toast.error('Failed to verify email');\n }\n return true;\n }\n\n async onActionForceVerifyPhone() {\n const confirmed = await Dialog.confirm(\n `Mark <strong>${this.model.get('phone_number')}</strong> as verified?`,\n 'Force Verify Phone'\n );\n if (!confirmed) return true;\n\n const resp = await this.model.save({ is_phone_verified: true });\n if (resp.status === 200) {\n this.getApp().toast.success('Phone marked as verified');\n } else {\n this.getApp().toast.error('Failed to verify phone');\n }\n return true;\n }\n\n async onActionToggleActive() {\n if (this.model.get('is_active')) {\n return this.onActionDeactivateUser();\n } else {\n return this.onActionActivateUser();\n }\n }\n\n async onActionDeactivateUser() {\n const confirmed = await Dialog.confirm(\"Are you sure you want to deactivate this user?\");\n if (!confirmed) return true;\n\n const resp = await this.model.save({ is_active: false });\n if (resp.status === 200) {\n this.getApp().toast.success(\"User deactivated\");\n } else {\n this.getApp().toast.error(\"Failed to deactivate user\");\n }\n return true;\n }\n\n async onActionActivateUser() {\n const confirmed = await Dialog.confirm(\"Are you sure you want to activate this user?\");\n if (!confirmed) return true;\n\n const resp = await this.model.save({ is_active: true });\n if (resp.status === 200) {\n this.getApp().toast.success(\"User activated\");\n } else {\n this.getApp().toast.error(\"Failed to activate user\");\n }\n return true;\n }\n\n async showSection(sectionName) {\n if (this.sideNavView) {\n await this.sideNavView.showSection(sectionName);\n }\n }\n\n getActiveSection() {\n return this.sideNavView ? this.sideNavView.getActiveSection() : null;\n }\n\n // Legacy API compatibility\n async showTab(tabName) {\n return this.showSection(tabName);\n }\n\n getActiveTab() {\n return this.getActiveSection();\n }\n\n _onModelChange() {\n // do nothing, we do not want model changes to render this entire view\n }\n\n // Static factory method\n static create(options = {}) {\n return new UserView(options);\n }\n}\n\nUser.VIEW_CLASS = UserView;\n\nexport default UserView;\n","/**\n * UsersPage - Simple Todo List using TablePage component\n * Demonstrates clean usage of TablePage framework features\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport {UserList, UserForms} from '@core/models/User.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\nimport MOJOUtils from '@core/utils/MOJOUtils.js';\nimport UserView from './UserView.js';\n\n\nclass UserTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_users',\n pageName: 'Manage Users',\n router: \"admin/users\",\n Collection: UserList,\n\n viewDialogOptions: { header: false },\n\n defaultQuery: {\n sort: '-last_activity',\n is_active: true\n },\n\n // Column definitions\n columns: [\n {\n key: 'id',\n label: 'ID',\n sortable: true,\n class: 'text-muted'\n },\n // {\n // label: 'Avatar',\n // key: 'avatar|avatar(\"sm\")',\n // sortable: false,\n // visibility: 'md'\n // },\n {\n key: 'display_name|tooltip:model.username',\n label: 'Display Name',\n },\n {\n label: 'Info',\n key: 'permissions.manage_users',\n template: `\n {{^model.is_active}}<span class=\"text-danger\">DISABLED</span> {{/model.is_active}}\n {{#model.permissions.manage_users}}{{{model.permissions.manage_users|yesnoicon('bi bi-person-gear text-danger')|tooltip('Manage Users')}}} {{/model.permissions.manage_users}}\n {{#model.permissions.manage_groups}}{{{model.permissions.manage_groups|yesnoicon('bi bi-building-gear text-primary')|tooltip('Manage Groups')}}} {{/model.permissions.manage_groups}}\n {{#model.permissions.view_global}}{{{model.permissions.view_global|yesnoicon('bi bi-globe text-secondary')|tooltip('View Global Menu')}}} {{/model.permissions.view_global}}\n {{#model.permissions.view_admin}}{{{model.permissions.view_admin|yesnoicon('bi bi-wrench text-secondary')|tooltip('View Admin Menu')}}} {{/model.permissions.view_admin}}\n `,\n sortable: false,\n },\n {\n key: 'email',\n label: 'Email',\n visibility: 'xl',\n className: 'text-muted fs-8',\n },\n // {\n // key: 'username',\n // label: 'Username',\n // visibility: 'xl',\n // className: 'text-muted fs-8',\n // },\n {\n key: 'last_activity',\n label: 'Last Activity',\n formatter: \"relative\",\n className: 'text-muted fs-8',\n }\n ],\n\n filters: [\n {\n key: 'is_active',\n label: 'Active',\n type: 'boolean',\n defaultValue: true,\n },\n {\n key: 'email',\n label: 'Email',\n type: 'text',\n defaultValue: '',\n },\n {\n key: 'username',\n label: 'Username',\n type: 'text',\n defaultValue: '',\n },\n {\n key: 'locations__ip_address',\n label: 'IP Address',\n type: 'text',\n defaultValue: '',\n },\n {\n key: 'last_activity',\n type: 'daterange',\n startName: 'dr_start',\n endName: 'dr_end',\n fieldName: 'dr_field',\n label: 'Date Range',\n format: 'YYYY-MM-DD',\n displayFormat: 'MMM DD, YYYY',\n separator: ' to '\n }\n ],\n\n // Table features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // TablePage toolbar\n showRefresh: true,\n showAdd: true,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No users found. Click \"Add\" to create a new user.',\n\n // Context menu configuration\n contextMenu: [\n {\n icon: 'bi-pencil',\n action: 'edit',\n label: \"Edit Profile\"\n },\n {\n icon: 'bi-shield-check',\n action: 'edit-permissions',\n label: \"Edit Permissions\"\n },\n {\n icon: 'bi-shield',\n action: 'change-password',\n label: \"Change Password\",\n },\n { separator: true },\n {\n icon: 'bi-envelope',\n action: 'send-invite',\n label: \"Send Invite\"\n }\n ],\n\n // Table display options (for HTML table styling)\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n }\n });\n }\n\n async onActionEditPermissions(event, element) {\n\n event.preventDefault();\n const item = this.collection.get(element.dataset.id);\n const result = await Dialog.showModelForm({\n model: item,\n size: 'lg',\n title: `Edit Permissions for \"${item._.username}\"`,\n fields: UserForms.permissions.fields\n });\n }\n\n async onActionChangePassword(event, element) {\n // Implement password change logic here\n const item = this.collection.get(element.dataset.id);\n const data = await Dialog.showForm({\n title: `Change Password for \"${item._.username}\"`,\n fields: [\n {\n type: 'text', // Change from 'hidden' to 'text'\n name: 'username',\n value: item.get('email') || item.get('username'),\n attributes: {\n autocomplete: 'username',\n readonly: 'readonly',\n tabindex: '-1',\n style: 'position: absolute; left: -9999px; opacity: 0; height: 0; width: 0;'\n }\n },\n {\n name: 'new_password',\n label: 'New Password',\n type: 'password',\n passwordUsage: 'new',\n required: true,\n showToggle: true,\n attributes: {\n autocomplete: 'new-password' // Make sure this isn't being overridden\n }\n }\n ]\n });\n\n if (data && data.new_password) {\n // Basic password validation\n const result = MOJOUtils.checkPasswordStrength(data.new_password);\n if (result.score < 5) {\n this.getApp().toast.error('Password must be at least 6 characters long and contain at least 2 of the following: uppercase letter, lowercase letter, or number');\n await this.onActionChangePassword(event, element);\n return;\n }\n const resp = await item.save({new_password: data.new_password});\n if (!this.onPasswordChange(resp)) {\n await this.onActionChangePassword(event, element);\n }\n }\n }\n\n onPasswordChange(resp) {\n if (resp.success) {\n this.getApp().toast.success('Password changed successfully');\n return true;\n } else {\n if (resp.data && resp.data.error) {\n this.getApp().toast.error(resp.data.error);\n } else {\n this.getApp().toast.error('Failed to change password');\n }\n }\n return false;\n }\n\n async onActionSendInvite(event, element) {\n const item = this.collection.get(element.dataset.id);\n const resp = await item.save({send_invite: true});\n if (resp.success) {\n this.getApp().toast.success('Invite sent successfully');\n return true;\n } else {\n if (resp.data && resp.data.error) {\n this.getApp().toast.error(resp.data.error);\n } else {\n this.getApp().toast.error('Failed to send invite');\n }\n }\n return false;\n }\n\n}\n\nexport default UserTablePage;\n","/**\n * MemberView - Modern membership detail view\n *\n * Features:\n * - Header with avatar, user name, group name, role badge, active toggle\n * - SideNavView: Details (with clickable user/group), Permissions, Logs\n * - Context menu: Edit, View User, View Group, Remove\n * - Click user → opens UserView, click group → opens GroupView\n */\n\nimport View from '@core/View.js';\nimport SideNavView from '@core/views/navigation/SideNavView.js';\nimport TableView from '@core/views/table/TableView.js';\nimport FormView from '@core/forms/FormView.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport { Member, MemberForms } from '@core/models/Member.js';\nimport { LogList } from '@core/models/Log.js';\nimport Modal from '@core/views/feedback/Modal.js';\n\nclass MemberView extends View {\n constructor(options = {}) {\n super({\n className: 'member-view',\n ...options\n });\n\n this.model = options.model || new Member(options.data || {});\n\n this.template = `\n <div class=\"member-view-container\">\n <!-- Header + Context Menu -->\n <div class=\"d-flex justify-content-between align-items-start mb-4\">\n <div data-container=\"member-header\" style=\"flex: 1;\"></div>\n <div data-container=\"member-context-menu\" class=\"ms-3 flex-shrink-0\"></div>\n </div>\n <!-- Side Nav -->\n <div data-container=\"member-sidenav\" style=\"min-height: 300px;\"></div>\n </div>\n `;\n }\n\n async onInit() {\n // ── Header ──────────────────────────────────\n this.header = new View({\n containerId: 'member-header',\n template: `\n <div class=\"d-flex justify-content-between align-items-start\">\n <!-- Left: Avatar + Identity -->\n <div class=\"d-flex align-items-center gap-3\">\n {{{model.user.avatar|avatar('md','rounded-circle')}}}\n <div>\n <h4 class=\"mb-0\">\n <a href=\"#\" data-action=\"view-user\" class=\"text-decoration-none text-body\">{{model.user.display_name}}</a>\n </h4>\n <div class=\"text-muted small mt-1\">\n <i class=\"bi bi-people me-1\"></i>\n <a href=\"#\" data-action=\"view-group\" class=\"text-decoration-none\">{{model.group.name}}</a>\n {{#model.group.kind}}\n <span class=\"badge bg-light text-muted border ms-1\" style=\"font-size: 0.65rem;\">{{model.group.kind|capitalize}}</span>\n {{/model.group.kind}}\n </div>\n </div>\n </div>\n\n <!-- Right: Status -->\n <div class=\"text-end\">\n <div class=\"d-flex align-items-center gap-2\">\n {{#model.metadata.role}}\n <span class=\"badge bg-primary bg-opacity-10 text-primary\" style=\"font-size: 0.72rem;\">{{model.metadata.role}}</span>\n {{/model.metadata.role}}\n <span class=\"d-inline-flex align-items-center gap-1\" style=\"cursor: pointer;\"\n data-action=\"toggle-active\"\n title=\"{{model.is_active|boolean('Click to deactivate','Click to activate')}}\">\n <i class=\"bi {{model.is_active|boolean('bi-toggle-on text-success','bi-toggle-off text-secondary')}}\" style=\"font-size: 1.1rem;\"></i>\n <span class=\"small\">{{model.is_active|boolean('Active','Inactive')}}</span>\n </span>\n </div>\n {{#model.created}}\n <div class=\"text-muted small mt-1\">Joined {{model.created|date}}</div>\n {{/model.created}}\n </div>\n </div>`\n });\n this.header.setModel(this.model);\n this.addChild(this.header);\n\n // ── Details section ─────────────────────────\n const detailsView = new View({\n model: this.model,\n template: `\n <style>\n .mv-section-label { font-size: 0.7rem; font-weight: 700; text-transform: uppercase; letter-spacing: 0.06em; color: #adb5bd; margin-bottom: 0.5rem; margin-top: 1.5rem; }\n .mv-section-label:first-child { margin-top: 0; }\n .mv-field-row { display: flex; align-items: baseline; padding: 0.5rem 0; border-bottom: 1px solid #f0f0f0; }\n .mv-field-row:last-child { border-bottom: none; }\n .mv-field-label { width: 130px; font-size: 0.78rem; color: #6c757d; flex-shrink: 0; }\n .mv-field-value { flex: 1; font-size: 0.88rem; color: #212529; }\n .mv-field-action { color: #6c757d; cursor: pointer; font-size: 0.8rem; margin-left: auto; padding: 0.15rem 0.4rem; border-radius: 4px; background: none; border: none; }\n .mv-field-action:hover { background: #f0f0f0; color: #0d6efd; }\n </style>\n\n <div class=\"mv-section-label\">User</div>\n <div class=\"mv-field-row\">\n <div class=\"mv-field-label\">Name</div>\n <div class=\"mv-field-value\">\n <a href=\"#\" data-action=\"view-user\" class=\"text-decoration-none\">{{model.user.display_name}}</a>\n </div>\n </div>\n <div class=\"mv-field-row\">\n <div class=\"mv-field-label\">Email</div>\n <div class=\"mv-field-value\">{{model.user.email}}</div>\n </div>\n\n <div class=\"mv-section-label\">Group</div>\n <div class=\"mv-field-row\">\n <div class=\"mv-field-label\">Name</div>\n <div class=\"mv-field-value\">\n <a href=\"#\" data-action=\"view-group\" class=\"text-decoration-none\">{{model.group.name}}</a>\n </div>\n </div>\n {{#model.group.kind}}\n <div class=\"mv-field-row\">\n <div class=\"mv-field-label\">Kind</div>\n <div class=\"mv-field-value\"><span class=\"badge bg-primary bg-opacity-10 text-primary\">{{model.group.kind|capitalize}}</span></div>\n </div>\n {{/model.group.kind}}\n\n <div class=\"mv-section-label\">Membership</div>\n <div class=\"mv-field-row\">\n <div class=\"mv-field-label\">Role</div>\n <div class=\"mv-field-value\">{{model.metadata.role|default('—')}}</div>\n <button type=\"button\" class=\"mv-field-action\" data-action=\"edit-membership\" title=\"Edit\"><i class=\"bi bi-pencil\"></i></button>\n </div>\n <div class=\"mv-field-row\">\n <div class=\"mv-field-label\">Status</div>\n <div class=\"mv-field-value\">\n {{#model.is_active|bool}}<span style=\"font-size:0.65rem; padding:0.15em 0.45em; background:#d1e7dd; color:#0f5132; border-radius:3px;\">Active</span>{{/model.is_active|bool}}\n {{^model.is_active|bool}}<span style=\"font-size:0.65rem; padding:0.15em 0.45em; background:#fff3cd; color:#856404; border-radius:3px;\">Inactive</span>{{/model.is_active|bool}}\n </div>\n </div>\n <div class=\"mv-field-row\">\n <div class=\"mv-field-label\">Member ID</div>\n <div class=\"mv-field-value\" style=\"font-family: ui-monospace, monospace; font-size: 0.82rem;\">{{model.id}}</div>\n </div>\n <div class=\"mv-field-row\">\n <div class=\"mv-field-label\">Joined</div>\n <div class=\"mv-field-value\">{{model.created|datetime|default('—')}}</div>\n </div>\n `\n });\n\n // ── Permissions section — editable switches ──\n const permissionsView = new FormView({\n fields: Member.PERMISSION_FIELDS,\n model: this.model,\n autosaveModelField: true\n });\n\n // ── Logs section ────────────────────────────\n const logsView = new TableView({\n collection: new LogList({\n params: { size: 10, model_name: 'account.Member', model_id: this.model.get('id') }\n }),\n permissions: 'view_logs',\n hideActivePillNames: ['model_name', 'model_id'],\n columns: [\n {\n key: 'created', label: 'Timestamp', sortable: true, formatter: 'epoch|datetime',\n filter: { name: 'created', type: 'daterange', startName: 'dr_start', endName: 'dr_end', fieldName: 'dr_field', label: 'Date Range', format: 'YYYY-MM-DD', displayFormat: 'MMM DD, YYYY', separator: ' to ' }\n },\n { key: 'level', label: 'Level', sortable: true },\n { key: 'kind', label: 'Kind' },\n { name: 'log', label: 'Log' }\n ]\n });\n\n // ── SideNavView ─────────────────────────────\n this.sideNavView = new SideNavView({\n containerId: 'member-sidenav',\n activeSection: 'details',\n navWidth: 160,\n contentPadding: '1rem 1.5rem',\n enableResponsive: true,\n minWidth: 450,\n sections: [\n { key: 'details', label: 'Details', icon: 'bi-info-circle', view: detailsView },\n { key: 'permissions', label: 'Permissions', icon: 'bi-shield-check', view: permissionsView },\n { type: 'divider', label: 'Activity' },\n { key: 'logs', label: 'Logs', icon: 'bi-journal-text', view: logsView, permissions: 'view_logs' }\n ]\n });\n this.addChild(this.sideNavView);\n\n // ── Context Menu ────────────────────────────\n const memberMenu = new ContextMenu({\n containerId: 'member-context-menu',\n className: 'context-menu-view header-menu-absolute',\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: [\n { label: 'Edit Membership', action: 'edit-membership', icon: 'bi-pencil' },\n { type: 'divider' },\n { label: 'View User', action: 'view-user', icon: 'bi-person' },\n { label: 'View Group', action: 'view-group', icon: 'bi-people' },\n { type: 'divider' },\n this.model.get('is_active')\n ? { label: 'Deactivate Member', action: 'deactivate-member', icon: 'bi-toggle-off' }\n : { label: 'Activate Member', action: 'activate-member', icon: 'bi-toggle-on' },\n { label: 'Remove From Group', action: 'remove-member', icon: 'bi-person-dash', danger: true }\n ]\n }\n });\n this.addChild(memberMenu);\n }\n\n // ── Actions ─────────────────────────────────\n\n async onActionEditMembership() {\n await Modal.modelForm({\n title: 'Edit Membership',\n model: this.model,\n formConfig: MemberForms.edit,\n });\n }\n\n async onActionViewUser() {\n const userId = this.model.get('user')?.id;\n if (!userId) return true;\n const { User } = await import('@core/models/User.js');\n await Modal.showModelById(User, userId);\n return true;\n }\n\n async onActionViewGroup() {\n const groupId = this.model.get('group')?.id;\n if (!groupId) return true;\n\n const { Group } = await import('@core/models/Group.js');\n await Modal.showModelById(Group, groupId);\n return true;\n }\n\n async onActionToggleActive() {\n if (this.model.get('is_active')) {\n return this.onActionDeactivateMember();\n } else {\n return this.onActionActivateMember();\n }\n }\n\n async onActionDeactivateMember() {\n const confirmed = await Modal.confirm(\n `Deactivate <strong>${this.model.get('user.display_name')}</strong>'s membership in <strong>${this.model.get('group.name')}</strong>?`,\n 'Deactivate Member'\n );\n if (!confirmed) return true;\n\n const resp = await this.model.save({ is_active: false });\n if (resp.status === 200) {\n this.getApp()?.toast?.success('Member deactivated');\n } else {\n this.getApp()?.toast?.error('Failed to deactivate member');\n }\n return true;\n }\n\n async onActionActivateMember() {\n const confirmed = await Modal.confirm(\n `Activate <strong>${this.model.get('user.display_name')}</strong>'s membership in <strong>${this.model.get('group.name')}</strong>?`,\n 'Activate Member'\n );\n if (!confirmed) return true;\n\n const resp = await this.model.save({ is_active: true });\n if (resp.status === 200) {\n this.getApp()?.toast?.success('Member activated');\n } else {\n this.getApp()?.toast?.error('Failed to activate member');\n }\n return true;\n }\n\n async onActionRemoveMember() {\n const confirmed = await Modal.confirm(\n `Remove <strong>${this.model.get('user.display_name')}</strong> from <strong>${this.model.get('group.name')}</strong>? This cannot be undone.`,\n 'Remove Member'\n );\n if (!confirmed) return true;\n\n const resp = await this.model.destroy();\n if (resp.success) {\n this.getApp()?.toast?.success('Member removed');\n this.emit('member:removed', { model: this.model });\n } else {\n this.getApp()?.toast?.error('Failed to remove member');\n }\n return true;\n }\n\n _onModelChange() {\n // Prevent full re-render on model changes\n }\n\n static create(options = {}) {\n return new MemberView(options);\n }\n}\n\nMember.VIEW_CLASS = MemberView;\n\nexport default MemberView;\n","/**\n * MemberTablePage - Member management using TablePage component\n * Clean implementation using TablePage with minimal overrides\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { MemberList, MemberForms } from '@core/models/Member.js';\nimport MemberView from './MemberView.js';\n\nclass MemberTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_members',\n pageName: 'Manage Members',\n router: \"admin/members\",\n Collection: MemberList,\n \n formEdit: MemberForms.edit,\n itemViewClass: MemberView,\n \n viewDialogOptions: {\n header: false,\n size: 'lg'\n },\n \n // Column definitions\n columns: [\n {\n key: 'id',\n label: 'ID',\n width: '60px',\n sortable: true,\n class: 'text-muted'\n },\n {\n key: 'user.display_name',\n label: 'User',\n formatter: \"default('Unknown User')\"\n },\n {\n key: 'user.email',\n label: 'Email',\n formatter: \"default('No Email')\"\n },\n {\n key: 'group.name',\n label: 'Group',\n formatter: \"default('Unknown Group')\"\n },\n {\n key: 'role',\n label: 'Role',\n formatter: \"badge\"\n },\n {\n key: 'status',\n label: 'Status',\n formatter: \"badge\"\n },\n {\n key: 'created',\n label: 'Added',\n formatter: \"epoch|datetime\"\n }\n ],\n\n // Table features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Toolbar\n showRefresh: true,\n showAdd: true,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No members found. Click \"Add Member\" to add users to groups.',\n\n // Batch actions\n batchBarLocation: 'top',\n batchActions: [\n { label: \"Remove\", icon: \"bi bi-person-dash\", action: \"batch-remove\" },\n { label: \"Export\", icon: \"bi bi-download\", action: \"batch-export\" },\n { label: \"Change Role\", icon: \"bi bi-person-gear\", action: \"batch-role\" },\n { label: \"Activate\", icon: \"bi bi-check-circle\", action: \"batch-activate\" },\n { label: \"Deactivate\", icon: \"bi bi-x-circle\", action: \"batch-deactivate\" }\n ],\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n }\n });\n }\n}\n\nexport default MemberTablePage;","/**\n * GroupView - Modern group management interface\n *\n * Features:\n * - Clean header with avatar, name, kind badge, parent link, active/online status\n * - SideNavView with: Details, Members, Children, Events, Logs\n * - Expanded context menu with quick actions\n * - Clean Bootstrap 5 styling matching UserView patterns\n */\n\nimport View from '@core/View.js';\nimport SideNavView from '@core/views/navigation/SideNavView.js';\nimport TableView from '@core/views/table/TableView.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport { Group, GroupList, GroupForms } from '@core/models/Group.js';\nimport { MemberList } from '@core/models/Member.js';\nimport { IncidentEventList } from '@core/models/Incident.js';\nimport { LogList } from '@core/models/Log.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\n\nclass GroupView extends View {\n constructor(options = {}) {\n super({\n className: 'group-view',\n ...options\n });\n\n this.model = options.model || new Group(options.data || {});\n\n this.template = `\n <div class=\"group-view-container\">\n <!-- Header -->\n <div data-container=\"group-header\"></div>\n <!-- Side Nav -->\n <div data-container=\"group-sidenav\" style=\"min-height: 400px;\"></div>\n </div>\n `;\n }\n\n async onInit() {\n // ── Header ──────────────────────────────────\n this.header = new View({\n containerId: 'group-header',\n template: `\n <div class=\"d-flex justify-content-between align-items-start mb-4\">\n <!-- Left Side: Identity -->\n <div class=\"d-flex align-items-center gap-3\">\n {{#model.avatar}}\n {{{model.avatar|avatar('md','rounded')}}}\n {{/model.avatar}}\n {{^model.avatar}}\n <div class=\"d-flex align-items-center justify-content-center rounded bg-light\" style=\"width: 56px; height: 56px;\">\n <i class=\"bi bi-people text-secondary\" style=\"font-size: 1.5rem;\"></i>\n </div>\n {{/model.avatar}}\n <div>\n <h3 class=\"mb-0\">{{model.name|default('Unnamed Group')}}</h3>\n <div class=\"d-flex align-items-center gap-2 mt-1\">\n <span class=\"badge bg-primary bg-opacity-10 text-primary\" style=\"font-size: 0.72rem;\">{{model.kind|capitalize}}</span>\n {{#model.parent}}\n <span class=\"text-muted small\">\n <i class=\"bi bi-diagram-3 me-1\"></i>\n <a href=\"#\" data-action=\"view-parent\" data-id=\"{{model.parent.id}}\" class=\"text-decoration-none\">{{model.parent.name}}</a>\n </span>\n {{/model.parent}}\n </div>\n {{#model.metadata.timezone}}\n <div class=\"text-muted small mt-1\"><i class=\"bi bi-clock me-1\"></i>{{model.metadata.timezone}}</div>\n {{/model.metadata.timezone}}\n </div>\n </div>\n\n <!-- Right Side: Status & Actions -->\n <div class=\"d-flex align-items-start gap-4\">\n <div class=\"text-end\">\n <div class=\"d-flex align-items-center justify-content-end gap-3\">\n <span class=\"d-inline-flex align-items-center gap-1\" title=\"{{model.is_active|boolean('Group Active','Group Inactive')}}\">\n <i class=\"bi {{model.is_active|boolean('bi-toggle-on text-success','bi-toggle-off text-secondary')}}\" style=\"font-size: 1.1rem;\"></i>\n <span class=\"small\">{{model.is_active|boolean('Active','Inactive')}}</span>\n </span>\n </div>\n {{#model.last_activity}}\n <div class=\"text-muted small mt-1\">Last active {{model.last_activity|relative}}</div>\n {{/model.last_activity}}\n {{#model.created}}\n <div class=\"text-muted small mt-1\">Created {{model.created|date}}</div>\n {{/model.created}}\n </div>\n <div data-container=\"group-context-menu\"></div>\n </div>\n </div>`\n });\n this.header.setModel(this.model);\n this.addChild(this.header);\n\n // ── Details section ─────────────────────────\n const detailsView = new View({\n model: this.model,\n template: `\n <style>\n .gv-section-label { font-size: 0.7rem; font-weight: 700; text-transform: uppercase; letter-spacing: 0.06em; color: #adb5bd; margin-bottom: 0.5rem; margin-top: 1.5rem; }\n .gv-section-label:first-child { margin-top: 0; }\n .gv-field-row { display: flex; align-items: baseline; padding: 0.5rem 0; border-bottom: 1px solid #f0f0f0; }\n .gv-field-row:last-child { border-bottom: none; }\n .gv-field-label { width: 140px; font-size: 0.78rem; color: #6c757d; flex-shrink: 0; }\n .gv-field-value { flex: 1; font-size: 0.88rem; color: #212529; }\n .gv-field-action { color: #6c757d; cursor: pointer; font-size: 0.8rem; margin-left: auto; padding: 0.15rem 0.4rem; border-radius: 4px; background: none; border: none; }\n .gv-field-action:hover { background: #f0f0f0; color: #0d6efd; }\n </style>\n\n <div class=\"gv-section-label\">Group</div>\n <div class=\"gv-field-row\">\n <div class=\"gv-field-label\">Name</div>\n <div class=\"gv-field-value\">{{model.name}}</div>\n <button type=\"button\" class=\"gv-field-action\" data-action=\"edit-group\" title=\"Edit\"><i class=\"bi bi-pencil\"></i></button>\n </div>\n <div class=\"gv-field-row\">\n <div class=\"gv-field-label\">Kind</div>\n <div class=\"gv-field-value\"><span class=\"badge bg-primary bg-opacity-10 text-primary\">{{model.kind|capitalize}}</span></div>\n </div>\n <div class=\"gv-field-row\">\n <div class=\"gv-field-label\">Status</div>\n <div class=\"gv-field-value\">\n {{#model.is_active|bool}}<span style=\"font-size:0.65rem; padding:0.15em 0.45em; background:#d1e7dd; color:#0f5132; border-radius:3px;\">Active</span>{{/model.is_active|bool}}\n {{^model.is_active|bool}}<span style=\"font-size:0.65rem; padding:0.15em 0.45em; background:#fff3cd; color:#856404; border-radius:3px;\">Inactive</span>{{/model.is_active|bool}}\n </div>\n </div>\n <div class=\"gv-field-row\">\n <div class=\"gv-field-label\">ID</div>\n <div class=\"gv-field-value\" style=\"font-family: ui-monospace, monospace; font-size: 0.82rem;\">{{model.id}}</div>\n </div>\n\n <div class=\"gv-section-label\">Hierarchy</div>\n <div class=\"gv-field-row\">\n <div class=\"gv-field-label\">Parent</div>\n <div class=\"gv-field-value\">\n {{#model.parent}}\n <a href=\"#\" data-action=\"view-parent\" data-id=\"{{model.parent.id}}\" class=\"text-decoration-none\">{{model.parent.name}}</a>\n <span class=\"text-muted small ms-1\">({{model.parent.kind|capitalize}})</span>\n {{/model.parent}}\n {{^model.parent}}<span style=\"color:#adb5bd; font-style:italic; font-size:0.85rem;\">None — top-level group</span>{{/model.parent}}\n </div>\n </div>\n\n <div class=\"gv-section-label\">Settings</div>\n {{#model.metadata.timezone}}\n <div class=\"gv-field-row\">\n <div class=\"gv-field-label\">Timezone</div>\n <div class=\"gv-field-value\">{{model.metadata.timezone}}</div>\n </div>\n {{/model.metadata.timezone}}\n {{#model.metadata.domain}}\n <div class=\"gv-field-row\">\n <div class=\"gv-field-label\">Domain</div>\n <div class=\"gv-field-value\">{{model.metadata.domain}}</div>\n </div>\n {{/model.metadata.domain}}\n {{#model.metadata.portal}}\n <div class=\"gv-field-row\">\n <div class=\"gv-field-label\">Portal URL</div>\n <div class=\"gv-field-value\"><a href=\"{{model.metadata.portal}}\" target=\"_blank\" class=\"text-decoration-none\">{{model.metadata.portal}}</a></div>\n </div>\n {{/model.metadata.portal}}\n {{#model.metadata.eod_hour}}\n <div class=\"gv-field-row\">\n <div class=\"gv-field-label\">End of Day</div>\n <div class=\"gv-field-value\">{{model.metadata.eod_hour}}:00</div>\n </div>\n {{/model.metadata.eod_hour}}\n\n <div class=\"gv-section-label\">Dates</div>\n <div class=\"gv-field-row\">\n <div class=\"gv-field-label\">Created</div>\n <div class=\"gv-field-value\">{{model.created|datetime|default('—')}}</div>\n </div>\n <div class=\"gv-field-row\">\n <div class=\"gv-field-label\">Modified</div>\n <div class=\"gv-field-value\">{{model.modified|datetime|default('—')}}</div>\n </div>\n `\n });\n\n // ── Members ─────────────────────────────────\n const membersView = new TableView({\n collection: new MemberList({ params: { group: this.model.get('id'), size: 10 } }),\n hideActivePillNames: ['group'],\n clickAction: 'view',\n showAdd: true,\n addButtonLabel: 'Invite',\n onAdd: (event) => this.onInviteClick(event),\n columns: [\n { key: 'user.display_name', label: 'User', sortable: true },\n { key: 'user.email', label: 'Email', sortable: true },\n { key: 'permissions|keys|badge', label: 'Permissions' },\n { key: 'created', label: 'Joined', formatter: 'date', sortable: true }\n ]\n });\n\n // ── Children (sub-groups) ───────────────────\n const childrenView = new TableView({\n collection: new GroupList({ params: { parent: this.model.get('id'), size: 10 } }),\n hideActivePillNames: ['parent'],\n clickAction: 'view',\n showAdd: true,\n addButtonLabel: 'Add Group',\n onAdd: () => this.onActionAddChildGroup(),\n columns: [\n { key: 'name', label: 'Name', sortable: true },\n { key: 'kind', label: 'Kind', formatter: 'badge' },\n {\n key: 'is_active', label: 'Status', width: '80px',\n template: `\n {{#model.is_active|bool}}<span class=\"badge bg-success bg-opacity-10 text-success\">Active</span>{{/model.is_active|bool}}\n {{^model.is_active|bool}}<span class=\"badge bg-secondary bg-opacity-10 text-secondary\">Inactive</span>{{/model.is_active|bool}}`\n },\n { key: 'created', label: 'Created', formatter: 'date', sortable: true }\n ]\n });\n\n // ── Events ──────────────────────────────────\n const eventsView = new TableView({\n collection: new IncidentEventList({\n params: { size: 10, model_name: 'account.Group', model_id: this.model.get('id') }\n }),\n hideActivePillNames: ['model_name', 'model_id'],\n columns: [\n { key: 'created', label: 'Date', formatter: 'datetime', sortable: true, width: '150px' },\n { key: 'category|badge', label: 'Category' },\n { key: 'title', label: 'Title' }\n ]\n });\n\n // ── Logs ────────────────────────────────────\n const logsView = new TableView({\n collection: new LogList({\n params: { size: 10, model_name: 'account.Group', model_id: this.model.get('id') }\n }),\n permissions: 'view_logs',\n hideActivePillNames: ['model_name', 'model_id'],\n columns: [\n {\n key: 'created', label: 'Timestamp', sortable: true, formatter: 'epoch|datetime',\n filter: { name: 'created', type: 'daterange', startName: 'dr_start', endName: 'dr_end', fieldName: 'dr_field', label: 'Date Range', format: 'YYYY-MM-DD', displayFormat: 'MMM DD, YYYY', separator: ' to ' }\n },\n {\n key: 'level', label: 'Level', sortable: true,\n filter: { type: 'select', options: [{ value: 'info', label: 'Info' }, { value: 'warning', label: 'Warning' }, { value: 'error', label: 'Error' }] }\n },\n { key: 'kind', label: 'Kind', filter: { type: 'text' } },\n { name: 'log', label: 'Log' }\n ]\n });\n\n // ── SideNavView ─────────────────────────────\n this.sideNavView = new SideNavView({\n containerId: 'group-sidenav',\n activeSection: 'details',\n navWidth: 180,\n contentPadding: '1.25rem 2rem',\n enableResponsive: true,\n minWidth: 500,\n sections: [\n { key: 'details', label: 'Details', icon: 'bi-info-circle', view: detailsView },\n { key: 'members', label: 'Members', icon: 'bi-people', view: membersView },\n { key: 'children', label: 'Sub-Groups', icon: 'bi-diagram-3', view: childrenView },\n { type: 'divider', label: 'Activity' },\n { key: 'events', label: 'Events', icon: 'bi-calendar-event', view: eventsView },\n { key: 'logs', label: 'Logs', icon: 'bi-journal-text', view: logsView, permissions: 'view_logs' }\n ]\n });\n this.addChild(this.sideNavView);\n\n // ── Context Menu ────────────────────────────\n const groupMenu = new ContextMenu({\n containerId: 'group-context-menu',\n className: 'context-menu-view header-menu-absolute',\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: [\n { label: 'Edit Group', action: 'edit-group', icon: 'bi-pencil' },\n { type: 'divider' },\n { label: 'Invite Member', action: 'invite-member', icon: 'bi-person-plus' },\n { label: 'Add Sub-Group', action: 'add-child-group', icon: 'bi-diagram-3' },\n { type: 'divider' },\n this.model.get('is_active')\n ? { label: 'Deactivate Group', action: 'deactivate-group', icon: 'bi-toggle-off' }\n : { label: 'Activate Group', action: 'activate-group', icon: 'bi-toggle-on' },\n ]\n }\n });\n this.addChild(groupMenu);\n }\n\n // ── Actions ─────────────────────────────────\n\n async onActionEditGroup() {\n const resp = await Dialog.showModelForm({\n title: `Edit Group — ${this.model.get('name')}`,\n model: this.model,\n size: 'lg',\n formConfig: GroupForms.detailed,\n });\n if (resp) {\n await this.render();\n }\n }\n\n async onActionInviteMember() {\n return this.onInviteClick(new Event('click'));\n }\n\n async onInviteClick(event) {\n if (event?.preventDefault) {\n event.preventDefault();\n event.stopPropagation();\n }\n const data = await Dialog.showForm({\n title: `Invite User to ${this.model.get('name')}`,\n size: 'sm',\n fields: [\n { type: 'email', name: 'email', label: 'Email', required: true, cols: 12 }\n ]\n });\n if (!data?.email) return true;\n\n const app = this.getApp();\n const resp = await app.rest.POST('/api/group/member/invite', {\n group: this.model.id,\n email: data.email\n });\n if (resp.success) {\n app.toast.success('User invited successfully');\n // Refresh members if on that section\n if (this.sideNavView?.getActiveSection() === 'members') {\n await this.sideNavView.showSection('members');\n }\n } else {\n app.toast.error(resp.message || 'Failed to invite user');\n }\n return true;\n }\n\n async onActionAddChildGroup() {\n const data = await Dialog.showForm({\n title: `Add Sub-Group to ${this.model.get('name')}`,\n size: 'sm',\n fields: GroupForms.create.fields.filter(f => f.name !== 'parent')\n });\n if (!data) return true;\n\n data.parent = this.model.id;\n const newGroup = new Group(data);\n const resp = await newGroup.save();\n if (resp.status === 200 || resp.status === 201) {\n this.getApp()?.toast?.success('Sub-group created');\n if (this.sideNavView?.getActiveSection() === 'children') {\n await this.sideNavView.showSection('children');\n }\n } else {\n this.getApp()?.toast?.error(resp.message || 'Failed to create sub-group');\n }\n return true;\n }\n\n async onActionDeactivateGroup() {\n const confirmed = await Dialog.confirm(\n `Are you sure you want to deactivate <strong>${this.model.get('name')}</strong>?`,\n 'Deactivate Group'\n );\n if (!confirmed) return true;\n\n const resp = await this.model.save({ is_active: false });\n if (resp.status === 200) {\n this.getApp()?.toast?.success('Group deactivated');\n await this.render();\n } else {\n this.getApp()?.toast?.error('Failed to deactivate group');\n }\n return true;\n }\n\n async onActionActivateGroup() {\n const confirmed = await Dialog.confirm(\n `Are you sure you want to activate <strong>${this.model.get('name')}</strong>?`,\n 'Activate Group'\n );\n if (!confirmed) return true;\n\n const resp = await this.model.save({ is_active: true });\n if (resp.status === 200) {\n this.getApp()?.toast?.success('Group activated');\n await this.render();\n } else {\n this.getApp()?.toast?.error('Failed to activate group');\n }\n return true;\n }\n\n async onActionViewParent(event, element) {\n const parentId = element?.dataset?.id;\n if (!parentId) return true;\n\n const parent = new Group({ id: parentId });\n await parent.fetch();\n if (parent.id) {\n Dialog.showDialog({\n title: false,\n size: 'lg',\n body: new GroupView({ model: parent }),\n buttons: [{ text: 'Close', class: 'btn-secondary', dismiss: true }]\n });\n }\n return true;\n }\n\n // ── Navigation helpers ──────────────────────\n\n async showSection(sectionName) {\n if (this.sideNavView) {\n await this.sideNavView.showSection(sectionName);\n }\n }\n\n getActiveSection() {\n return this.sideNavView ? this.sideNavView.getActiveSection() : null;\n }\n\n _onModelChange() {\n // Prevent full re-render on model changes\n }\n\n static create(options = {}) {\n return new GroupView(options);\n }\n}\n\nGroup.VIEW_CLASS = GroupView;\n\nexport default GroupView;\n","/**\n * GroupTablePage - Group management using TablePage component\n * Clean implementation using TablePage with minimal overrides\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { GroupList, GroupForms, Group } from '@core/models/Group.js';\nimport GroupView from './GroupView.js';\n\nclass GroupTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_groups',\n pageName: 'Manage Groups',\n router: \"admin/groups\",\n Collection: GroupList,\n\n formCreate: GroupForms.create,\n formEdit: GroupForms.edit,\n itemViewClass: GroupView,\n\n viewDialogOptions: {\n header: false\n },\n\n defaultQuery: {\n sort: '-id',\n is_active: 1,\n },\n\n // Column definitions\n columns: [\n {\n key: 'id',\n label: 'ID',\n sortable: true,\n class: 'text-muted'\n },\n\n {\n key: 'name',\n label: 'Display Name'\n },\n {\n key: 'kind|badge',\n label: 'Kind',\n filter: {\n type: \"select\",\n options: Group.GroupKindOptions\n }\n },\n {\n key: 'is_active|yesnoicon',\n label: 'Enabled',\n visibility: 'lg'\n },\n {\n key: 'parent.name',\n label: 'Parent',\n formatter: \"default('-')\",\n visibility: 'md',\n class: 'text-muted fs-8'\n },\n {\n key: 'created',\n label: 'Created',\n className: 'text-muted fs-8',\n formatter: \"epoch|datetime\",\n visibility: 'lg'\n },\n {\n key: 'last_activity',\n label: 'Activity',\n className: 'text-muted fs-8',\n formatter: \"relative\",\n visibility: 'lg'\n }\n ],\n\n filters: [\n {\n key: 'is_active',\n label: 'Active',\n type: 'select',\n options: [\n { label: 'Active', value: true },\n { label: 'Inactive', value: false }\n ]\n }\n ],\n\n contextMenu: [\n {\n icon: 'bi-pencil',\n action: 'edit',\n label: \"Edit Group\"\n },\n {\n icon: 'bi-bullseye',\n action: 'make-active',\n label: \"Make Active Group\"\n }\n ],\n\n // Table features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Toolbar\n showRefresh: true,\n showAdd: true,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No groups found. Click \"Add Group\" to create your first one.',\n\n // Batch actions\n batchBarLocation: 'top',\n batchActions: [\n { label: \"Delete\", icon: \"bi bi-trash\", action: \"batch-delete\" },\n { label: \"Export\", icon: \"bi bi-download\", action: \"batch-export\" },\n { label: \"Activate\", icon: \"bi bi-check-circle\", action: \"batch-activate\" },\n { label: \"Deactivate\", icon: \"bi bi-x-circle\", action: \"batch-deactivate\" },\n { label: \"Move\", icon: \"bi bi-arrow-right\", action: \"batch-move\" }\n ],\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n },\n });\n }\n\n onActionMakeActive(event, element) {\n const item = this.collection.get(element.dataset.id);\n this.getApp().setActiveGroup(item);\n }\n\n}\n\nexport default GroupTablePage;\n","/**\n * DeviceView - Clean, modern user device detail view\n *\n * SideNavView layout with:\n * - Details: browser, OS, device, user agent in clean field rows\n * - Locations: multiline table of all locations this device connected from\n * - Header: smart browser/OS icon, device summary, last seen, context menu\n */\n\nimport View from '@core/View.js';\nimport SideNavView from '@core/views/navigation/SideNavView.js';\nimport TableView from '@core/views/table/TableView.js';\nimport TableRow from '@core/views/table/TableRow.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport { UserDevice, UserDeviceLocationList } from '@core/models/User.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\n\n// ── Location row for multiline table ──\n\nclass DeviceLocationRow extends TableRow {\n get locationText() {\n const geo = this.model?.get('geolocation') || {};\n return [geo.city, geo.region].filter(Boolean).join(', ') || geo.country_name || '—';\n }\n get countryName() {\n return this.model?.get('geolocation')?.country_name || '';\n }\n get ispName() {\n const geo = this.model?.get('geolocation') || {};\n return geo.isp || geo.asn_org || '';\n }\n get threatFlags() {\n const geo = this.model?.get('geolocation') || {};\n const flags = [];\n if (geo.is_vpn) flags.push('<span class=\"badge bg-warning text-dark\" style=\"font-size:0.6rem;\">VPN</span>');\n if (geo.is_tor) flags.push('<span class=\"badge bg-danger\" style=\"font-size:0.6rem;\">Tor</span>');\n if (geo.is_proxy) flags.push('<span class=\"badge bg-warning text-dark\" style=\"font-size:0.6rem;\">Proxy</span>');\n return flags.join(' ');\n }\n get hasThreatFlags() {\n const geo = this.model?.get('geolocation') || {};\n return !!(geo.is_vpn || geo.is_tor || geo.is_proxy);\n }\n}\n\nclass DeviceView extends View {\n constructor(options = {}) {\n super({\n className: 'device-view',\n ...options\n });\n\n this.model = options.model || new UserDevice(options.data || {});\n this.deviceInfo = this.model.get('device_info') || {};\n this.deviceIcon = this._getIcon(this.deviceInfo);\n\n // Computed properties for template\n this.browserFull = this._getBrowser();\n this.osFull = this._getOS();\n this.deviceFull = this._getDevice();\n this.isMobile = this._isMobile();\n\n this.template = `\n <style>\n .dv-header { display: flex; justify-content: space-between; align-items: flex-start; margin-bottom: 1.5rem; }\n .dv-identity { display: flex; align-items: center; gap: 1rem; }\n .dv-icon-wrap { width: 48px; height: 48px; border-radius: 12px; display: flex; align-items: center; justify-content: center; font-size: 1.4rem; flex-shrink: 0; }\n .dv-title { font-size: 1.15rem; font-weight: 600; margin: 0; line-height: 1.3; }\n .dv-subtitle { font-size: 0.8rem; color: #6c757d; margin-top: 0.15rem; }\n .dv-status { text-align: right; display: flex; align-items: flex-start; gap: 0.75rem; }\n .dv-last-seen-label { font-size: 0.7rem; color: #adb5bd; text-transform: uppercase; letter-spacing: 0.04em; }\n .dv-last-seen-value { font-size: 0.88rem; font-weight: 500; }\n .dv-last-seen-ip { font-size: 0.75rem; color: #6c757d; margin-top: 0.1rem; }\n </style>\n\n <div class=\"dv-header\">\n <div class=\"dv-identity\">\n <div class=\"dv-icon-wrap bg-primary bg-opacity-10 text-primary\">\n <i class=\"bi {{deviceIcon}}\"></i>\n </div>\n <div>\n <h4 class=\"dv-title\">{{browserFull}} <span class=\"fw-normal text-muted\">on</span> {{osFull}}</h4>\n <div class=\"dv-subtitle\">\n {{deviceFull}}\n {{#model.user.display_name}}\n <span class=\"text-muted mx-1\">&middot;</span>\n <a href=\"#\" data-action=\"view-user\" class=\"text-decoration-none\">{{model.user.display_name}}</a>\n {{/model.user.display_name}}\n </div>\n </div>\n </div>\n <div class=\"dv-status\">\n <div>\n <div class=\"dv-last-seen-label\">Last Seen</div>\n <div class=\"dv-last-seen-value\">{{model.last_seen|relative}}</div>\n {{#model.last_ip}}<div class=\"dv-last-seen-ip\">{{model.last_ip}}</div>{{/model.last_ip}}\n </div>\n <div data-container=\"device-context-menu\"></div>\n </div>\n </div>\n\n <div data-container=\"device-sidenav\" style=\"min-height: 300px;\"></div>\n `;\n }\n\n // ── Computed getters ──────────────────────────\n\n _getBrowser() {\n const ua = this.deviceInfo?.user_agent || {};\n const parts = [ua.family, ua.major].filter(Boolean);\n return parts.length ? parts.join(' ') : 'Unknown Browser';\n }\n\n _getOS() {\n const os = this.deviceInfo?.os || {};\n const ver = [os.major, os.minor].filter(Boolean).join('.');\n return os.family ? `${os.family} ${ver}`.trim() : 'Unknown OS';\n }\n\n _getDevice() {\n const dev = this.deviceInfo?.device || {};\n const parts = [dev.brand, dev.family].filter(Boolean);\n const name = parts.length ? parts.join(' ') : 'Unknown Device';\n return dev.model ? `${name} (${dev.model})` : name;\n }\n\n _isMobile() {\n const dev = this.deviceInfo?.device || {};\n const os = this.deviceInfo?.os || {};\n return ['iPhone', 'Android', 'iPad'].some(m =>\n (dev.family || '').includes(m) || (os.family || '').includes(m)\n );\n }\n\n _getIcon(deviceInfo) {\n const os = deviceInfo?.os?.family?.toLowerCase() || '';\n const browser = deviceInfo?.user_agent?.family?.toLowerCase() || '';\n const device = deviceInfo?.device?.family?.toLowerCase() || '';\n\n if (browser.includes('chrome')) return 'bi-browser-chrome';\n if (browser.includes('firefox')) return 'bi-browser-firefox';\n if (browser.includes('safari')) return 'bi-browser-safari';\n if (browser.includes('edge')) return 'bi-browser-edge';\n if (os.includes('mac') || os.includes('ios')) return 'bi-apple';\n if (os.includes('windows')) return 'bi-windows';\n if (os.includes('android')) return 'bi-android2';\n if (os.includes('linux')) return 'bi-ubuntu';\n if (device.includes('iphone')) return 'bi-phone';\n if (device.includes('ipad')) return 'bi-tablet';\n return 'bi-laptop';\n }\n\n async onInit() {\n // ── Details section ─────────────────────────\n const detailsView = new View({\n model: this.model,\n template: `\n <style>\n .dv-section-label { font-size: 0.7rem; font-weight: 700; text-transform: uppercase; letter-spacing: 0.06em; color: #adb5bd; margin-bottom: 0.5rem; margin-top: 1.5rem; }\n .dv-section-label:first-child { margin-top: 0; }\n .dv-field-row { display: flex; align-items: baseline; padding: 0.5rem 0; border-bottom: 1px solid #f0f0f0; }\n .dv-field-row:last-child { border-bottom: none; }\n .dv-field-label { width: 130px; font-size: 0.78rem; color: #6c757d; flex-shrink: 0; }\n .dv-field-value { flex: 1; font-size: 0.88rem; color: #212529; }\n .dv-ua-string { font-family: ui-monospace, monospace; font-size: 0.73rem; color: #6c757d; word-break: break-all; line-height: 1.5; padding: 0.5rem 0.75rem; background: #f8f9fa; border-radius: 6px; margin-top: 0.25rem; }\n </style>\n\n <div class=\"dv-section-label\">Browser</div>\n <div class=\"dv-field-row\">\n <div class=\"dv-field-label\">Name</div>\n <div class=\"dv-field-value\">{{model.device_info.user_agent.family|default('—')}}</div>\n </div>\n <div class=\"dv-field-row\">\n <div class=\"dv-field-label\">Version</div>\n <div class=\"dv-field-value\">{{model.device_info.user_agent.major|default('—')}}{{#model.device_info.user_agent.minor}}.{{model.device_info.user_agent.minor}}{{/model.device_info.user_agent.minor}}{{#model.device_info.user_agent.patch}}.{{model.device_info.user_agent.patch}}{{/model.device_info.user_agent.patch}}</div>\n </div>\n\n <div class=\"dv-section-label\">Operating System</div>\n <div class=\"dv-field-row\">\n <div class=\"dv-field-label\">Name</div>\n <div class=\"dv-field-value\">{{model.device_info.os.family|default('—')}}</div>\n </div>\n <div class=\"dv-field-row\">\n <div class=\"dv-field-label\">Version</div>\n <div class=\"dv-field-value\">{{model.device_info.os.major|default('—')}}{{#model.device_info.os.minor}}.{{model.device_info.os.minor}}{{/model.device_info.os.minor}}{{#model.device_info.os.patch}}.{{model.device_info.os.patch}}{{/model.device_info.os.patch}}</div>\n </div>\n\n <div class=\"dv-section-label\">Hardware</div>\n <div class=\"dv-field-row\">\n <div class=\"dv-field-label\">Brand</div>\n <div class=\"dv-field-value\">{{model.device_info.device.brand|default('—')}}</div>\n </div>\n <div class=\"dv-field-row\">\n <div class=\"dv-field-label\">Family</div>\n <div class=\"dv-field-value\">{{model.device_info.device.family|default('—')}}</div>\n </div>\n <div class=\"dv-field-row\">\n <div class=\"dv-field-label\">Model</div>\n <div class=\"dv-field-value\">{{model.device_info.device.model|default('—')}}</div>\n </div>\n\n <div class=\"dv-section-label\">Identification</div>\n <div class=\"dv-field-row\">\n <div class=\"dv-field-label\">Device ID</div>\n <div class=\"dv-field-value\" style=\"font-family: ui-monospace, monospace; font-size: 0.78rem;\">{{model.duid|truncate_middle(32)}}</div>\n </div>\n <div class=\"dv-field-row\">\n <div class=\"dv-field-label\">Last IP</div>\n <div class=\"dv-field-value\">{{model.last_ip|default('—')}}</div>\n </div>\n <div class=\"dv-field-row\">\n <div class=\"dv-field-label\">First Seen</div>\n <div class=\"dv-field-value\">{{model.first_seen|epoch|datetime|default('—')}}</div>\n </div>\n <div class=\"dv-field-row\">\n <div class=\"dv-field-label\">Last Seen</div>\n <div class=\"dv-field-value\">{{model.last_seen|epoch|datetime|default('—')}}</div>\n </div>\n\n {{#model.device_info.string}}\n <div class=\"dv-section-label\">User Agent String</div>\n <div class=\"dv-ua-string\">{{model.device_info.string}}</div>\n {{/model.device_info.string}}\n `\n });\n\n // ── Locations section — multiline rows ──────\n const locationsView = new TableView({\n collection: new UserDeviceLocationList({\n params: { user_device: this.model.get('id'), size: 10 }\n }),\n hideActivePillNames: ['user_device'],\n clickAction: 'view',\n itemClass: DeviceLocationRow,\n selectable: false,\n columns: [\n {\n key: 'ip_address',\n label: 'Location',\n template: `\n <div style=\"font-size:0.85rem; font-weight:500;\">\n <i class=\"bi bi-geo-alt text-muted me-1\" style=\"font-size:0.95rem; vertical-align:middle;\"></i>{{locationText}}\n {{#countryName}} <span class=\"text-muted fw-normal\">&middot; {{countryName}}</span>{{/countryName}}\n </div>\n <div style=\"font-size:0.73rem; color:#6c757d; margin-top:0.15rem;\">\n {{model.ip_address}}\n {{#ispName}} <span class=\"text-muted mx-1\">&middot;</span> {{ispName}}{{/ispName}}\n {{#hasThreatFlags|bool}} <span class=\"ms-1\">{{{threatFlags}}}</span>{{/hasThreatFlags|bool}}\n </div>`\n },\n { key: 'first_seen', label: 'First Seen', formatter: 'epoch|relative', width: '110px' },\n { key: 'last_seen', label: 'Last Seen', formatter: 'epoch|relative', width: '110px' }\n ]\n });\n\n // ── SideNavView ─────────────────────────────\n this.sideNavView = new SideNavView({\n containerId: 'device-sidenav',\n activeSection: 'details',\n navWidth: 160,\n contentPadding: '1rem 1.5rem',\n enableResponsive: true,\n minWidth: 450,\n sections: [\n { key: 'details', label: 'Details', icon: 'bi-info-circle', view: detailsView },\n { key: 'locations', label: 'Locations', icon: 'bi-geo-alt', view: locationsView }\n ]\n });\n this.addChild(this.sideNavView);\n\n // ── Context Menu ────────────────────────────\n const deviceMenu = new ContextMenu({\n containerId: 'device-context-menu',\n className: \"context-menu-view header-menu-absolute\",\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: [\n { label: 'View User', action: 'view-user', icon: 'bi-person' },\n { label: 'Block Device', action: 'block-device', icon: 'bi-shield-slash', disabled: true },\n { type: 'divider' },\n { label: 'Delete Record', action: 'delete-device', icon: 'bi-trash', danger: true }\n ]\n }\n });\n this.addChild(deviceMenu);\n }\n\n async onActionViewUser() {\n this.emit('view-user', { userId: this.model.get('user')?.id });\n }\n\n async onActionDeleteDevice() {\n const confirmed = await Dialog.confirm(\n 'Are you sure you want to delete this device record?',\n 'Delete Device'\n );\n if (!confirmed) return true;\n\n const resp = await this.model.destroy();\n if (resp.success) {\n this.emit('device:deleted', { model: this.model });\n }\n return true;\n }\n\n static async show(duid) {\n const model = await UserDevice.getByDuid(duid);\n if (model) {\n return Dialog.showDialog({\n title: false,\n size: 'lg',\n body: new DeviceView({ model }),\n buttons: [{ text: 'Close', class: 'btn-secondary', dismiss: true }]\n });\n }\n Dialog.alert({ message: `Could not find device with DUID: ${duid}`, type: 'warning' });\n return null;\n }\n}\n\nUserDevice.VIEW_CLASS = DeviceView;\nexport default DeviceView;\n","/**\n * UserDeviceTablePage - User device management using TablePage component\n * Clean implementation using TablePage with minimal overrides\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { UserDeviceList } from '@core/models/User.js';\nimport DeviceView from './DeviceView.js';\n\nclass UserDeviceTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_user_devices',\n pageName: 'User Devices',\n router: \"admin/user/devices\",\n Collection: UserDeviceList,\n\n // itemViewClass: DeviceView,\n viewDialogOptions: {\n header: false,\n size: 'lg'\n },\n\n // Column definitions\n columns: [\n { key: 'id', label: 'ID', width: '70px', sortable: true, class: 'text-muted' },\n { key: 'duid', label: 'Device ID', sortable: true, formatter: 'truncate_middle(16)' },\n { key: 'user.display_name', label: 'User', sortable: true, formatter: \"default('—')\" },\n { key: 'device_info.user_agent.family', label: 'Browser', formatter: \"default('—')\" },\n { key: 'device_info.os.family', label: 'OS', formatter: \"default('—')\" },\n { key: 'last_ip', label: 'Last IP', sortable: true },\n { key: 'first_seen', label: 'First Seen', formatter: \"epoch|datetime\" },\n { key: 'last_seen', label: 'Last Seen', formatter: \"epoch|datetime\" }\n ],\n\n // Table features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Toolbar\n showRefresh: true,\n showAdd: false,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No user devices found.',\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n }\n });\n }\n}\n\nexport default UserDeviceTablePage;\n","/**\n * UserDeviceLocationTablePage - Device location tracking using TablePage component\n * Clean implementation using TablePage with minimal overrides\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { UserDeviceLocationList } from '@core/models/User.js';\n\nclass UserDeviceLocationTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_user_device_locations',\n pageName: 'Device Locations',\n router: \"admin/user/device-locations\",\n Collection: UserDeviceLocationList,\n\n // Column definitions\n columns: [\n { key: 'id', label: 'ID', width: '70px', sortable: true, class: 'text-muted' },\n { key: 'user.display_name', label: 'User', sortable: true },\n { key: 'user_device', label: 'Device', template: '{{user_device.device_info.user_agent.family}} on {{user_device.device_info.os.family}}', sortable: true },\n { key: 'ip_address', label: 'IP Address', sortable: true },\n { key: 'geolocation.city', label: 'City', formatter: \"default('—')\" },\n { key: 'geolocation.region', label: 'Region', formatter: \"default('—')\" },\n { key: 'geolocation.country_name', label: 'Country', formatter: \"default('—')\" },\n { key: 'last_seen', label: 'Last Seen', formatter: \"epoch|datetime\" }\n ],\n\n // Table features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Toolbar\n showRefresh: true,\n showAdd: false,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No device locations found.',\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n }\n });\n }\n}\n\nexport default UserDeviceLocationTablePage;","/**\n * GeoIPView - Detailed view for a GeoLocatedIP record\n */\n\nimport View from '@core/View.js';\nimport TabView from '@core/views/navigation/TabView.js';\nimport DataView from '@core/views/data/DataView.js';\nimport TableView from '@core/views/table/TableView.js';\nimport MapView from '@ext/map/MapView.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport { GeoLocatedIP } from '@core/models/System.js';\nimport { IncidentEventList } from '@core/models/Incident.js';\nimport { LogList } from '@core/models/Log.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\n\nclass GeoIPView extends View {\n constructor(options = {}) {\n super({\n className: 'geoip-view',\n ...options\n });\n\n this.model = options.model || new GeoLocatedIP(options.data || {});\n this.hasCoordinates = this.model.get('latitude') && this.model.get('longitude');\n\n this.template = `\n <div class=\"geoip-view-container\">\n <!-- Header -->\n <div class=\"d-flex justify-content-between align-items-center mb-4\">\n <!-- Left Side: Icon & Info -->\n <div class=\"d-flex align-items-center gap-3\">\n <div class=\"fs-1 text-primary\">\n <i class=\"bi bi-globe-americas\"></i>\n </div>\n <div>\n <h3 class=\"mb-1\">{{model.ip_address}}</h3>\n <div class=\"text-muted small\">\n {{model.city|default('Unknown Location')}}, {{model.country_name|default('Unknown Location')}}\n </div>\n <div class=\"text-muted small mt-1\">\n ISP: {{model.isp|capitalize}}\n </div>\n </div>\n </div>\n\n <!-- Right Side: Risk Summary + Actions -->\n <div class=\"d-flex align-items-start gap-4\">\n <!-- Risk summary -->\n <div class=\"text-end\">\n <div class=\"d-flex align-items-baseline justify-content-end gap-2\">\n <span class=\"text-muted\">Risk:</span>\n <span class=\"fw-bold fs-4\n {{#model.is_threat}} text-danger {{/model.is_threat}}\n {{#model.is_suspicious}} text-warning {{/model.is_suspicious}}\n {{^model.is_threat}}{{^model.is_suspicious}} text-success {{/model.is_suspicious}}{{/model.is_threat}}\n \">{{#model.threat_level}}{{model.threat_level|capitalize}}{{/model.threat_level}}{{^model.threat_level}}Unknown{{/model.threat_level}}</span>\n </div>\n <div class=\"mt-1 small d-flex align-items-center justify-content-end gap-2\">\n <span class=\"text-muted\">Score:</span>\n <span class=\"fw-semibold\">{{model.risk_score|default('—')}}</span>\n </div>\n <div class=\"mt-1 d-flex align-items-center justify-content-end gap-2\">\n <i class=\"bi bi-shield-lock {{#model.is_tor}}fs-4 text-success{{/model.is_tor}}{{^model.is_tor}}text-muted{{/model.is_tor}}\" data-bs-toggle=\"tooltip\" title=\"TOR exit\"></i>\n <i class=\"bi bi-shield {{#model.is_vpn}}fs-4 text-success{{/model.is_vpn}}{{^model.is_vpn}}text-muted{{/model.is_vpn}}\" data-bs-toggle=\"tooltip\" title=\"VPN detected\"></i>\n <i class=\"bi bi-cloud {{#model.is_cloud}}fs-4 text-success{{/model.is_cloud}}{{^model.is_cloud}}text-muted{{/model.is_cloud}}\" data-bs-toggle=\"tooltip\" title=\"Cloud provider\"></i>\n <i class=\"bi bi-hdd-stack {{#model.is_datacenter}}fs-4 text-success{{/model.is_datacenter}}{{^model.is_datacenter}}text-muted{{/model.is_datacenter}}\" data-bs-toggle=\"tooltip\" title=\"Datacenter\"></i>\n <i class=\"bi bi-phone {{#model.is_mobile}}fs-4 text-success{{/model.is_mobile}}{{^model.is_mobile}}text-muted{{/model.is_mobile}}\" data-bs-toggle=\"tooltip\" title=\"Mobile connection\"></i>\n <i class=\"bi bi-diagram-3 {{#model.is_proxy}}fs-4 text-success{{/model.is_proxy}}{{^model.is_proxy}}text-muted{{/model.is_proxy}}\" data-bs-toggle=\"tooltip\" title=\"Proxy\"></i>\n </div>\n </div>\n <!-- Actions: context menu aligned to top (not vertically centered) -->\n <div class=\"d-flex align-items-start\">\n <div data-container=\"geoip-context-menu\"></div>\n </div>\n </div>\n </div>\n\n <!-- Tabs -->\n <div data-container=\"geoip-tabs\"></div>\n </div>\n `;\n }\n\n async onInit() {\n // Location Details Tab\n this.detailsView = new DataView({\n model: this.model,\n className: \"p-3\",\n showEmptyValues: true,\n emptyValueText: '—',\n columns: 2,\n fields: [\n { name: 'ip_address', label: 'IP Address', cols: 4 },\n { name: 'subnet', label: 'Subnet', cols: 4 },\n { name: 'country_name', label: 'Country', cols: 4 },\n { name: 'country_code', label: 'Country Code', cols: 4 },\n { name: 'region', label: 'Region', cols: 4 },\n { name: 'city', label: 'City', cols: 4 },\n { name: 'postal_code', label: 'Postal Code', cols: 4 },\n { name: 'timezone', label: 'Timezone', cols: 4 },\n { name: 'latitude', label: 'Latitude', cols: 4 },\n { name: 'longitude', label: 'Longitude', cols: 4 },\n ]\n });\n\n // Network Tab\n this.networkView = new DataView({\n model: this.model,\n className: \"p-3\",\n showEmptyValues: true,\n emptyValueText: '—',\n columns: 2,\n fields: [\n { name: 'is_tor', label: 'TOR Exit Node', formatter: 'yesnoicon', cols: 4 },\n { name: 'is_vpn', label: 'VPN', formatter: 'yesnoicon', cols: 4 },\n { name: 'is_proxy', label: 'Proxy', formatter: 'yesnoicon', cols: 4 },\n { name: 'is_cloud', label: 'Cloud Provider', formatter: 'yesnoicon', cols: 4 },\n { name: 'is_datacenter', label: 'Datacenter', formatter: 'yesnoicon', cols: 4 },\n { name: 'is_mobile', label: 'Mobile', formatter: 'yesnoicon', cols: 4 },\n { name: 'mobile_carrier', label: 'Mobile Carrier', cols: 8 },\n { name: 'asn', label: 'ASN', cols: 4 },\n { name: 'asn_org', label: 'ASN Organization', cols: 8 },\n { name: 'isp', label: 'ISP', cols: 12 },\n { name: 'connection_type', label: 'Connection Type', cols: 6 }\n ]\n });\n\n // Risk & Reputation Tab\n this.riskView = new DataView({\n model: this.model,\n className: \"p-3\",\n showEmptyValues: true,\n emptyValueText: '—',\n columns: 2,\n fields: [\n { name: 'threat_level', label: 'Threat Level', cols: 6 },\n { name: 'risk_score', label: 'Risk Score', cols: 6 },\n { name: 'is_threat', label: 'Threat', formatter: 'yesnoicon', cols: 6 },\n { name: 'is_suspicious', label: 'Suspicious', formatter: 'yesnoicon', cols: 6 },\n { name: 'is_known_attacker', label: 'Known Attacker', formatter: 'yesnoicon', cols: 6 },\n { name: 'is_known_abuser', label: 'Known Abuser', formatter: 'yesnoicon', cols: 6 }\n ]\n });\n\n // Metadata Tab\n this.metadataView = new DataView({\n model: this.model,\n className: \"p-3\",\n showEmptyValues: true,\n emptyValueText: '—',\n columns: 2,\n fields: [\n { name: 'id', label: 'Record ID', cols: 6 },\n { name: 'provider', label: 'Data Provider', formatter: 'capitalize', cols: 6 },\n { name: 'created', label: 'Created', formatter: 'datetime', cols: 6 },\n { name: 'modified', label: 'Last Modified', formatter: 'datetime', cols: 6 },\n { name: 'last_seen', label: 'Last Seen', formatter: 'datetime', cols: 6 },\n { name: 'expires_at', label: 'Expires', formatter: 'datetime', cols: 6 }\n ]\n });\n\n // Create Events table with IncidentEventList collection\n const eventsCollection = new IncidentEventList({\n params: {\n size: 5,\n source_ip: this.model.get(\"ip_address\")\n }\n });\n this.eventsView = new TableView({\n collection: eventsCollection,\n hideActivePillNames: ['source_ip'],\n columns: [\n { key: 'id', label: 'ID', sortable: true, width: '40px' },\n { key: 'created', label: 'Date', formatter: 'datetime', sortable: true, width: '150px' },\n { key: 'category|badge', label: 'Category' },\n { key: 'title', label: 'Title' }\n ]\n });\n\n // Create Logs table with LogList collection\n const logsCollection = new LogList({\n params: {\n size: 5,\n ip: this.model.get('ip_address')\n }\n });\n this.logsView = new TableView({\n collection: logsCollection,\n permissions: 'view_logs',\n hideActivePillNames: ['ip'],\n columns: [\n {\n key: 'created',\n label: 'Timestamp',\n sortable: true,\n formatter: \"epoch|datetime\",\n filter: {\n name: \"created\",\n type: 'daterange',\n startName: 'dr_start',\n endName: 'dr_end',\n fieldName: 'dr_field',\n label: 'Date Range',\n format: 'YYYY-MM-DD',\n displayFormat: 'MMM DD, YYYY',\n separator: ' to '\n }\n },\n {\n key: 'level',\n label: 'Level',\n sortable: true,\n filter: {\n type: 'select',\n options: [\n { value: 'info', label: 'Info' },\n { value: 'warning', label: 'Warning' },\n { value: 'error', label: 'Error' }\n ]\n }\n },\n { key: 'kind', label: 'Kind', filter: { type: 'text' } },\n { name: 'log', label: 'Log' }\n ]\n });\n\n const tabs = {\n 'Location': this.detailsView,\n 'Network': this.networkView,\n 'Risk & Reputation': this.riskView,\n 'Events': this.eventsView,\n 'Logs': this.logsView,\n 'Metadata': this.metadataView\n };\n\n // Add Map tab if coordinates exist\n if (this.hasCoordinates) {\n const lat = this.model.get('latitude');\n const lng = this.model.get('longitude');\n const city = this.model.get('city') || 'Unknown';\n const region = this.model.get('region') || '';\n const country = this.model.get('country_name') || '';\n\n const locationStr = [city, region, country].filter(Boolean).join(', ');\n\n this.mapView = new MapView({\n markers: [{\n lat: lat,\n lng: lng,\n popup: `<strong>${this.model.get('ip_address')}</strong><br>${locationStr}`\n }],\n tileLayer: \"light\",\n zoom: 4,\n height: 450\n });\n tabs['Map'] = this.mapView;\n }\n\n this.tabView = new TabView({\n containerId: 'geoip-tabs',\n tabs: tabs,\n activeTab: this.hasCoordinates ? 'Map' : 'Location'\n });\n this.addChild(this.tabView);\n\n // ContextMenu\n const menuItems = [\n { label: 'Edit Location', action: 'edit-location', icon: 'bi-geo-alt' },\n { label: 'Edit Security', action: 'edit-security', icon: 'bi-shield-lock' },\n { label: 'Edit Network', action: 'edit-network', icon: 'bi-diagram-3' },\n { type: 'divider' },\n { label: 'Refresh Geolocation', action: 'refresh-geoip', icon: 'bi-arrow-clockwise' },\n ];\n\n if (this.hasCoordinates) {\n menuItems.push({\n label: 'View on Map',\n action: 'view-on-map',\n icon: 'bi-map'\n });\n }\n\n menuItems.push(\n { type: 'divider' },\n { label: 'Delete Record', action: 'delete-geoip', icon: 'bi-trash', danger: true }\n );\n\n const geoIPMenu = new ContextMenu({\n containerId: 'geoip-context-menu',\n className: \"context-menu-view header-menu-absolute\",\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: menuItems\n }\n });\n this.addChild(geoIPMenu);\n }\n\n async onAfterRender() {\n await super.onAfterRender();\n\n // Initialize Bootstrap tooltips for header icons/badges\n if (window.bootstrap && window.bootstrap.Tooltip && this.element) {\n const tooltipTriggerList = this.element.querySelectorAll('[data-bs-toggle=\"tooltip\"]');\n tooltipTriggerList.forEach(el => {\n // Dispose any existing instance (in case of re-render)\n const existing = window.bootstrap.Tooltip.getInstance(el);\n if (existing && typeof existing.dispose === 'function') {\n existing.dispose();\n }\n new window.bootstrap.Tooltip(el);\n });\n }\n }\n\n async onActionEditLocation() {\n const resp = await Dialog.showModelForm({\n title: `Edit Location - ${this.model.get('ip_address')}`,\n model: this.model,\n formConfig: GeoLocatedIP.EDIT_LOCATION_FORM,\n });\n\n if (resp) {\n await this.render();\n this.getApp()?.toast?.success('Location updated successfully');\n }\n }\n\n async onActionEditSecurity() {\n const resp = await Dialog.showModelForm({\n title: `Edit Security - ${this.model.get('ip_address')}`,\n model: this.model,\n formConfig: GeoLocatedIP.EDIT_SECURITY_FORM,\n });\n\n if (resp) {\n await this.render();\n this.getApp()?.toast?.success('Security settings updated successfully');\n }\n }\n\n async onActionEditNetwork() {\n const resp = await Dialog.showModelForm({\n title: `Edit Network - ${this.model.get('ip_address')}`,\n model: this.model,\n formConfig: GeoLocatedIP.EDIT_NETWORK_FORM,\n });\n\n if (resp) {\n await this.render();\n this.getApp()?.toast?.success('Network information updated successfully');\n }\n }\n\n async onActionRefreshGeoip() {\n // Placeholder for refresh logic, e.g., a POST request to a refresh endpoint\n await this.model.save({ refresh: true });\n this.getApp()?.toast?.info('Refresh request sent for ' + this.model.get('ip_address'));\n }\n\n async onActionThreatAnalysis() {\n // Placeholder for refresh logic, e.g., a POST request to a refresh endpoint\n await this.model.save({ threat_analysis: true });\n this.getApp()?.toast?.info('Requesting threat analysis for ' + this.model.get('ip_address'));\n }\n\n async onActionViewOnMap() {\n if (this.hasCoordinates) {\n const lat = this.model.get('latitude');\n const lon = this.model.get('longitude');\n const url = `https://www.google.com/maps/search/?api=1&query=${lat},${lon}`;\n window.open(url, '_blank');\n }\n }\n\n async onActionDeleteGeoip() {\n const confirmed = await Dialog.confirm(\n `Are you sure you want to delete the GeoIP record for \"${this.model.get('ip_address')}\"?`,\n 'Confirm Deletion',\n { confirmClass: 'btn-danger', confirmText: 'Delete' }\n );\n if (confirmed) {\n const resp = await this.model.destroy();\n if (resp.success) {\n this.emit('geoip:deleted', { model: this.model });\n }\n }\n }\n\n static async show(ip) {\n const model = await GeoLocatedIP.lookup(ip);\n if (model) {\n const view = new GeoIPView({ model });\n const dialog = new Dialog({\n header: false,\n size: 'lg',\n body: view,\n buttons: [{ text: 'Close', class: 'btn-secondary', dismiss: true }]\n });\n await dialog.render(true, document.body);\n dialog.show();\n return dialog;\n }\n Dialog.alert({ message: `Could not find geolocation data for IP: ${ip}`, type: 'warning' });\n return null;\n }\n}\n\nGeoLocatedIP.VIEW_CLASS = GeoIPView;\n\nexport default GeoIPView;\n","/**\n * GeoLocatedIPTablePage - GeoIP cache management using TablePage component\n * Clean implementation using TablePage with minimal overrides\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { GeoLocatedIPList, GeoLocatedIP } from '@core/models/System.js';\nimport GeoIPView from './GeoIPView.js';\n\nclass GeoLocatedIPTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_system_geoip',\n pageName: 'GeoIP Cache',\n router: \"admin/system/geoip\",\n Collection: GeoLocatedIPList,\n\n itemView: GeoIPView,\n viewDialogOptions: {\n header: false,\n size: 'xl'\n },\n\n // Column definitions\n columns: [\n { key: 'ip_address', label: 'IP Address', sortable: true },\n { key: 'city', label: 'City', sortable: true, formatter: \"default('—')\" },\n { key: 'region', label: 'Region', sortable: true, formatter: \"default('—')\" },\n { key: 'country_name', label: 'Country', sortable: true, formatter: \"default('—')\" },\n { key: 'isp', label: 'ISP', sortable: true, formatter: \"default('—')\" },\n { key: 'threat_level', label: 'Threat', formatter: \"default('—')\"}\n ],\n\n // Table features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Actions\n // actions: ['view', 'edit', 'delete'],\n clickAction: 'view',\n\n // Toolbar\n showRefresh: true,\n showAdd: true,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No GeoIP records found.',\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n },\n tableViewOptions: {\n addButtonLabel: \"Lookup IP\",\n onAdd: (evt) => {\n evt.preventDefault();\n // Implement the logic for adding a new record\n this.onLookup();\n }\n }\n });\n }\n\n async onLookup() {\n // Implement the logic for adding a new record\n const data = await this.getApp().showForm({\n title: \"Lookup IP\",\n fields: [\n {\n name: 'ip',\n type: 'text',\n required: true\n }\n ]\n });\n if (data && data.ip) {\n const model = await GeoLocatedIP.lookup(data.ip);\n if (model) {\n this.tableView._onRowView({ model });\n }\n }\n }\n}\n\nexport default GeoLocatedIPTablePage;\n","import Collection from '@core/Collection.js';\nimport Model from '@core/Model.js';\n\n/**\n * ApiKey - Group-scoped API key for external integrations and services.\n * Maps to REST endpoints under /api/group/apikey\n *\n * Key properties:\n * - Scoped to a single group\n * - Carries only explicitly granted permissions (least-privilege)\n * - sys.* permissions always denied\n * - No IP restriction (unlike User Auth Tokens)\n * - Header format: Authorization: apikey <token>\n *\n * The raw token is only returned at creation time — it is never shown again.\n *\n * Endpoints:\n * GET /api/group/apikey - List keys (filter by ?group=<id>)\n * POST /api/group/apikey - Create a key\n * GET /api/group/apikey/<id> - Get key details\n * POST /api/group/apikey/<id> - Update name, permissions, limits, is_active\n * DELETE /api/group/apikey/<id> - Delete key\n */\nclass ApiKey extends Model {\n constructor(data = {}, options = {}) {\n super(data, {\n endpoint: '/api/group/apikey',\n ...options\n });\n }\n}\n\n/**\n * ApiKeyList - Collection of ApiKey records.\n * Filter by group: new ApiKeyList({ params: { group: groupId } })\n */\nclass ApiKeyList extends Collection {\n constructor(options = {}) {\n super({\n ModelClass: ApiKey,\n endpoint: '/api/group/apikey',\n size: 25,\n ...options\n });\n }\n}\n\n/**\n * Forms configuration for ApiKey\n */\nconst ApiKeyForms = {\n create: {\n title: 'Create API Key',\n fields: [\n {\n name: 'name',\n type: 'text',\n label: 'Name',\n placeholder: 'Mobile App v2',\n required: true,\n columns: 12,\n help: 'A descriptive name to identify this key.'\n },\n {\n name: 'group',\n type: 'number',\n label: 'Group ID',\n required: true,\n columns: 12,\n help: 'The group this key is scoped to.'\n },\n {\n name: 'permissions',\n type: 'textarea',\n label: 'Permissions (JSON)',\n placeholder: '{\"view_orders\": true, \"create_orders\": true}',\n columns: 12,\n help: 'JSON dict of permissions to grant. Leave empty for no permissions.'\n }\n ]\n },\n\n edit: {\n title: 'Edit API Key',\n fields: [\n {\n name: 'name',\n type: 'text',\n label: 'Name',\n required: true,\n columns: 12\n },\n {\n name: 'is_active',\n type: 'switch',\n label: 'Active',\n columns: 12,\n help: 'Deactivate to revoke access without deleting the key.'\n },\n {\n name: 'permissions',\n type: 'textarea',\n label: 'Permissions (JSON)',\n columns: 12,\n help: 'JSON dict of granted permissions.'\n }\n ]\n }\n};\n\nexport { ApiKey, ApiKeyList, ApiKeyForms };\n","/**\n * ApiKeyView - Group-scoped API key detail and management interface\n *\n * Shows key metadata, permissions, and provides actions to edit, toggle\n * active state, and delete. The raw token is only displayed at creation\n * time and is not shown here.\n */\n\nimport View from '@core/View.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport { ApiKey, ApiKeyForms } from '@core/models/ApiKey.js';\n\nclass ApiKeyView extends View {\n constructor(options = {}) {\n super({\n className: 'api-key-view',\n ...options\n });\n\n this.model = options.model || new ApiKey(options.data || {});\n\n this.template = `\n <div class=\"api-key-view-container\">\n <!-- Header -->\n <div class=\"d-flex justify-content-between align-items-start mb-4\">\n <!-- Left: Icon & Identity -->\n <div class=\"d-flex align-items-center gap-3\">\n <div class=\"fs-1 text-primary\">\n <i class=\"bi bi-key\"></i>\n </div>\n <div>\n <h3 class=\"mb-1\">{{model.name|default('Unnamed Key')}}</h3>\n <div class=\"text-muted small\">\n ID: {{model.id}}\n <span class=\"mx-2\">|</span>\n Group: {{model.group.name|default(model.group)}}\n </div>\n <div class=\"mt-1\">\n <span class=\"badge {{model.is_active|boolean('bg-success','bg-secondary')}}\">\n {{model.is_active|boolean('Active','Inactive')}}\n </span>\n </div>\n </div>\n </div>\n\n <!-- Right: Meta & Actions -->\n <div class=\"d-flex align-items-start gap-4\">\n <div class=\"text-end\">\n <div class=\"text-muted small\">Created</div>\n <div>{{model.created|datetime}}</div>\n </div>\n <div data-container=\"apikey-context-menu\"></div>\n </div>\n </div>\n\n <!-- Details -->\n <div class=\"list-group mb-3\">\n <div class=\"list-group-item\">\n <h6 class=\"mb-1 text-muted\">Token Preview</h6>\n <p class=\"mb-1 font-monospace small text-muted\">\n The raw token is only shown once at creation time.\n </p>\n </div>\n <div class=\"list-group-item\">\n <h6 class=\"mb-1 text-muted\">Permissions</h6>\n {{#model.permissions}}\n <pre class=\"mb-0 small\">{{model.permissions|json}}</pre>\n {{/model.permissions}}\n {{^model.permissions}}\n <span class=\"text-muted small\">No permissions granted</span>\n {{/model.permissions}}\n </div>\n {{#model.limits}}\n <div class=\"list-group-item\">\n <h6 class=\"mb-1 text-muted\">Rate Limit Overrides</h6>\n <pre class=\"mb-0 small\">{{model.limits|json}}</pre>\n </div>\n {{/model.limits}}\n <div class=\"list-group-item\">\n <h6 class=\"mb-1 text-muted\">Usage</h6>\n <p class=\"mb-0 small text-muted\">\n Include in requests as:\n <code>Authorization: apikey &lt;token&gt;</code>\n </p>\n </div>\n </div>\n </div>\n `;\n }\n\n async onInit() {\n const isActive = this.model.get('is_active');\n\n const apiKeyMenu = new ContextMenu({\n containerId: 'apikey-context-menu',\n className: 'context-menu-view header-menu-absolute',\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: [\n { label: 'Edit', action: 'edit-key', icon: 'bi-pencil' },\n isActive\n ? { label: 'Deactivate', action: 'deactivate-key', icon: 'bi-x-circle' }\n : { label: 'Activate', action: 'activate-key', icon: 'bi-check-circle' },\n { type: 'divider' },\n { label: 'Delete Key', action: 'delete-key', icon: 'bi-trash', danger: true }\n ]\n }\n });\n this.addChild(apiKeyMenu);\n }\n\n async onActionEditKey() {\n const app = this.getApp();\n const resp = await app.showModelForm({\n title: `Edit API Key — ${this.model.get('name')}`,\n model: this.model,\n formConfig: ApiKeyForms.edit,\n });\n if (resp) {\n this.render();\n }\n }\n\n async onActionDeactivateKey() {\n const app = this.getApp();\n const confirmed = await app.confirm({\n title: 'Deactivate API Key',\n message: `Deactivate \"${this.model.get('name')}\"? Requests using this key will be rejected.`,\n confirmLabel: 'Deactivate',\n confirmClass: 'btn-warning'\n });\n if (!confirmed) return;\n\n app.showLoading();\n const resp = await this.model.save({ is_active: false });\n app.hideLoading();\n if (resp && resp.success !== false) {\n app.toast.success('API key deactivated');\n this.render();\n } else {\n app.toast.error('Failed to deactivate key');\n }\n }\n\n async onActionActivateKey() {\n const app = this.getApp();\n app.showLoading();\n const resp = await this.model.save({ is_active: true });\n app.hideLoading();\n if (resp && resp.success !== false) {\n app.toast.success('API key activated');\n this.render();\n } else {\n app.toast.error('Failed to activate key');\n }\n }\n\n async onActionDeleteKey() {\n const app = this.getApp();\n const confirmed = await app.confirm({\n title: 'Delete API Key',\n message: `Permanently delete \"${this.model.get('name')}\"? This cannot be undone.`,\n confirmLabel: 'Delete',\n confirmClass: 'btn-danger'\n });\n if (!confirmed) return;\n\n app.showLoading();\n const resp = await this.model.delete();\n app.hideLoading();\n if (resp && resp.success !== false) {\n app.toast.success('API key deleted');\n this.emit('deleted', { model: this.model });\n } else {\n app.toast.error('Failed to delete key');\n }\n }\n}\n\nApiKey.VIEW_CLASS = ApiKeyView;\nexport default ApiKeyView;\n","/**\n * ApiKeyTablePage - Group API key management using TablePage component\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { ApiKey, ApiKeyList, ApiKeyForms } from '@core/models/ApiKey.js';\nimport ApiKeyView from './ApiKeyView.js';\n\n// Register the add/edit forms on the model class so TableView can find them automatically\nApiKey.ADD_FORM = ApiKeyForms.create;\nApiKey.EDIT_FORM = ApiKeyForms.edit;\n\nclass ApiKeyTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_api_keys',\n pageName: 'API Keys',\n router: 'admin/api-keys',\n Collection: ApiKeyList,\n\n itemViewClass: ApiKeyView,\n viewDialogOptions: {\n header: false,\n size: 'lg'\n },\n\n columns: [\n { key: 'id', label: 'ID', width: '70px', sortable: true, class: 'text-muted' },\n { key: 'name', label: 'Name', sortable: true },\n { key: 'group.name', label: 'Group', sortable: true, formatter: \"default('—')\" },\n {\n key: 'is_active',\n label: 'Status',\n formatter: \"boolean('Active|bg-success','Inactive|bg-secondary')|badge\",\n width: '100px'\n },\n { key: 'created', label: 'Created', formatter: 'datetime', sortable: true }\n ],\n\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n showRefresh: true,\n showAdd: true,\n showExport: false,\n\n addButtonLabel: 'New API Key',\n\n emptyMessage: 'No API keys found.',\n\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n }\n });\n }\n\n // Override to intercept and show the one-time token after model.save()\n async onActionAdd() {\n const app = this.getApp();\n const model = new ApiKey();\n const result = await app.showForm({\n model,\n ...ApiKeyForms.create\n });\n if (!result) return;\n\n const resp = await model.save(result);\n if (!resp?.data?.status) {\n app.showError(resp?.data?.error || 'Failed to create API key');\n return;\n }\n\n // Token is only present in the creation response — show it once\n const token = resp.data?.data?.token;\n await app.showAlert({\n title: 'API Key Created — Save Your Token',\n message: token\n ? `Copy this token now. It will not be shown again.\\n\\n${token}`\n : 'API key created successfully.',\n type: token ? 'warning' : 'success',\n size: 'lg'\n });\n\n this.collection.add(model);\n this.tableView?.refresh();\n }\n}\n\nexport default ApiKeyTablePage;\n","/**\n * CloudWatchChart - MetricsChart configured for CloudWatch endpoints\n *\n * Extends MetricsChart with:\n * - CloudWatch endpoint defaults (/api/aws/cloudwatch/fetch)\n * - `stat` parameter support (avg, max, min, sum)\n * - Response format normalization (periods → labels, {slug,values} → {slug: values})\n *\n * NOTE: The CloudWatch API currently returns a slightly different format\n * than the standard metrics API (periods vs labels, array vs dict).\n * processMetricsData normalizes this until the backend aligns.\n */\nimport MetricsChart from '@ext/charts/MetricsChart.js';\n\nexport default class CloudWatchChart extends MetricsChart {\n constructor(options = {}) {\n super({\n endpoint: '/api/aws/cloudwatch/fetch',\n account: options.resourceType || options.account || 'ec2',\n category: options.category || null,\n slugs: options.slugs || (options.slug ? [options.slug] : null),\n granularity: options.granularity || 'hours',\n title: options.title || 'CloudWatch',\n defaultDateRange: options.defaultDateRange || '24h',\n showDateRange: false,\n ...options\n });\n\n this.stat = options.stat || 'avg';\n this.resourceType = options.resourceType || options.account || 'ec2';\n }\n\n buildApiParams() {\n const params = super.buildApiParams();\n // CloudWatch uses 'stat' parameter\n params.stat = this.stat;\n return params;\n }\n\n setStat(stat) {\n this.stat = stat;\n return this.fetchData();\n }\n}\n","/**\n * CloudWatchDashboardPage - AWS CloudWatch monitoring dashboard\n *\n * 2-column grid of MetricsCharts showing key metrics across all resources.\n * Each chart auto-plots all instances for its account+category.\n * Uses /api/aws/cloudwatch/fetch via MetricsChart.\n */\nimport Page from '@core/Page.js';\nimport CloudWatchChart from './CloudWatchChart.js';\n\nconst DASHBOARD_CHARTS = [\n { account: 'ec2', category: 'cpu', title: 'EC2 CPU', unit: '%' },\n { account: 'ec2', category: 'net_out', title: 'EC2 Network Out', unit: 'bytes' },\n { account: 'ec2', category: 'memory', title: 'EC2 Memory', unit: '%' },\n { account: 'ec2', category: 'disk', title: 'EC2 Disk', unit: '%' },\n { account: 'rds', category: 'cpu', title: 'RDS CPU', unit: '%' },\n { account: 'rds', category: 'conns', title: 'RDS Connections', unit: '' },\n { account: 'rds', category: 'read_latency', title: 'RDS Read Latency', unit: 's' },\n { account: 'rds', category: 'write_latency', title: 'RDS Write Latency', unit: 's' },\n { account: 'redis', category: 'cpu', title: 'Redis CPU', unit: '%' },\n { account: 'redis', category: 'conns', title: 'Redis Connections', unit: '' },\n { account: 'redis', category: 'cache_misses', title: 'Redis Cache Misses', unit: '' },\n { account: 'redis', category: 'cache_hits', title: 'Redis Cache Hits', unit: '' }\n];\n\nfunction yAxisForUnit(unit) {\n if (unit === '%') return { label: '%', beginAtZero: true, max: 100 };\n if (unit === 'bytes') return { label: 'Bytes', beginAtZero: true };\n if (unit === 's') return { label: 'Seconds', beginAtZero: true };\n return { beginAtZero: true };\n}\n\nexport default class CloudWatchDashboardPage extends Page {\n constructor(options = {}) {\n super({\n ...options,\n title: 'CloudWatch Monitoring',\n className: 'cloudwatch-dashboard-page'\n });\n }\n\n async getTemplate() {\n return `\n <style>\n .cw-grid { display: grid; grid-template-columns: 1fr 1fr; gap: 1rem; }\n @media (max-width: 992px) { .cw-grid { grid-template-columns: 1fr; } }\n </style>\n <div class=\"container-fluid\">\n <p class=\"text-muted mb-3\">AWS CloudWatch resource monitoring</p>\n <div class=\"cw-grid\" id=\"cw-grid\">\n ${DASHBOARD_CHARTS.map((_, i) => `<div id=\"cw-chart-${i}\"></div>`).join('')}\n </div>\n </div>\n `;\n }\n\n async onInit() {\n this.getApp()?.showLoading('Loading CloudWatch...');\n try {\n for (let i = 0; i < DASHBOARD_CHARTS.length; i++) {\n const def = DASHBOARD_CHARTS[i];\n const chart = new CloudWatchChart({\n containerId: `cw-chart-${i}`,\n account: def.account,\n category: def.category,\n title: def.title,\n height: 160,\n yAxis: yAxisForUnit(def.unit),\n responsive: true,\n showGranularity: true,\n showDateRange: true,\n defaultDateRange: '24h',\n granularity: 'hours'\n });\n this.addChild(chart);\n }\n } finally {\n this.getApp()?.hideLoading();\n }\n }\n}\n","import Page from '@core/Page.js';\nimport View from '@core/View.js';\nimport {\n MetricsChart,\n MetricsMiniChartWidget\n} from '@ext/charts/index.js';\nimport TableView from '@core/views/table/TableView.js';\nimport {\n IncidentList,\n IncidentStats\n} from '@core/models/Incident.js';\nimport {\n TicketList\n} from '@core/models/Tickets.js';\nimport { MetricsCountryMapView } from '@ext/map/index.js';\n\nclass IncidentDashboardHeader extends View {\n constructor(options = {}) {\n super({\n ...options,\n className: 'incident-dashboard-header'\n });\n\n this.model = new IncidentStats();\n }\n\n async getTemplate() {\n return `\n <div class=\"row\">\n <div class=\"col-xl-3 col-lg-6 col-12 mb-3\">\n <div class=\"card h-100 border-0 shadow-sm\">\n <div class=\"card-body\">\n <div class=\"d-flex justify-content-between align-items-start\">\n <div>\n <h6 class=\"card-title text-muted mb-2\">Open Incidents</h6>\n <h3 class=\"mb-1 fw-bold\">{{model.incidents.open}}</h3>\n <span class=\"badge bg-danger-subtle text-danger\">{{model.incidents.new}} New</span>\n </div>\n <div class=\"text-danger\">\n <i class=\"bi bi-exclamation-triangle fs-2\"></i>\n </div>\n </div>\n </div>\n </div>\n </div>\n <div class=\"col-xl-3 col-lg-6 col-12 mb-3\">\n <div class=\"card h-100 border-0 shadow-sm\">\n <div class=\"card-body\">\n <div class=\"d-flex justify-content-between align-items-start\">\n <div>\n <h6 class=\"card-title text-muted mb-2\">Open Tickets</h6>\n <h3 class=\"mb-1 fw-bold\">{{model.tickets.open}}</h3>\n <span class=\"badge bg-warning-subtle text-warning\">{{model.tickets.new}} New</span>\n </div>\n <div class=\"text-warning\">\n <i class=\"bi bi-ticket-perforated fs-2\"></i>\n </div>\n </div>\n </div>\n </div>\n </div>\n <div class=\"col-xl-3 col-lg-6 col-12 mb-3\">\n <div class=\"card h-100 border-0 shadow-sm\">\n <div class=\"card-body\">\n <div class=\"d-flex justify-content-between align-items-start\">\n <div>\n <h6 class=\"card-title text-muted mb-2\">Recent Events</h6>\n <h3 class=\"mb-1 fw-bold\">{{model.events.recent}}</h3>\n <span class=\"badge bg-info-subtle text-info\">{{model.events.critical}} Critical</span>\n </div>\n <div class=\"text-info\">\n <i class=\"bi bi-activity fs-2\"></i>\n </div>\n </div>\n </div>\n </div>\n </div>\n <div class=\"col-xl-3 col-lg-6 col-12 mb-3\">\n <div class=\"card h-100 border-0 shadow-sm\">\n <div class=\"card-body\">\n <div class=\"d-flex justify-content-between align-items-start\">\n <div>\n <h6 class=\"card-title text-muted mb-2\">Recent Incidents</h6>\n <h3 class=\"mb-1 fw-bold\">{{model.incidents.recent}}</h3>\n <span class=\"badge bg-secondary-subtle text-secondary\">Last 24h</span>\n </div>\n <div class=\"text-secondary\">\n <i class=\"bi bi-clock-history fs-2\"></i>\n </div>\n </div>\n </div>\n </div>\n </div>\n </div>\n `;\n }\n\n async onBeforeRender() {\n await this.model.fetch();\n }\n}\n\n\nclass IncidentDashboardPage extends Page {\n constructor(options = {}) {\n super({\n ...options,\n title: 'Incidents Dashboard',\n className: 'incident-dashboard-page'\n });\n }\n\n async getTemplate() {\n return `\n <div class=\"container-fluid incident-dashboard\">\n <div class=\"d-flex justify-content-between align-items-center mb-3\">\n <div>\n <p class=\"text-muted mb-0\">Incident Intelligence Hub</p>\n <small class=\"text-info\">\n <i class=\"bi bi-activity me-1\"></i>\n Real-time visibility into incidents, tickets, and correlated events\n </small>\n </div>\n <div class=\"btn-group\" role=\"group\">\n <button type=\"button\"\n class=\"btn btn-outline-secondary btn-sm\"\n data-action=\"refresh-all\"\n title=\"Refresh dashboards\">\n <i class=\"bi bi-arrow-clockwise\"></i> Refresh\n </button>\n </div>\n </div>\n\n <div data-container=\"header\" class=\"mb-4\"></div>\n\n <div class=\"row g-4\">\n <div class=\"col-xl-6 col-lg-12\" data-container=\"events-widget\"></div>\n <div class=\"col-xl-6 col-lg-12\" data-container=\"incidents-widget\"></div>\n </div>\n\n\n <div class=\"row g-4 mt-1\">\n <div class=\"col-12\">\n <div class=\"card shadow-sm h-100\">\n <div class=\"card-header border-0 bg-transparent d-flex align-items-center justify-content-between\">\n <div>\n <h6 class=\"mb-0 text-uppercase small text-muted\">Global Event Hotspots</h6>\n <span class=\"text-muted small\">Clusters sized by total events</span>\n </div>\n <span class=\"badge bg-info-subtle text-info\">Interactive</span>\n </div>\n <div class=\"card-body p-0\" data-container=\"events-country-map\"></div>\n </div>\n </div>\n </div>\n\n\n <div class=\"row g-4 mt-1\">\n <div class=\"col-lg-6\">\n <div class=\"card shadow-sm h-100\">\n <div class=\"card-header border-0 bg-transparent d-flex align-items-center justify-content-between\">\n <div>\n <h6 class=\"mb-0 text-uppercase small text-muted\">Events by Country</h6>\n <span class=\"text-muted small\">Hotspots from the last 24 hours</span>\n </div>\n <span class=\"badge bg-success-subtle text-success\">Live</span>\n </div>\n <div class=\"card-body p-3\">\n <div data-container=\"events-by-country-chart\"></div>\n </div>\n </div>\n </div>\n <div class=\"col-lg-6\">\n <div class=\"card shadow-sm h-100\">\n <div class=\"card-header border-0 bg-transparent d-flex align-items-center justify-content-between\">\n <div>\n <h6 class=\"mb-0 text-uppercase small text-muted\">Incidents by Country</h6>\n <span class=\"text-muted small\">Highest volume regions</span>\n </div>\n <span class=\"badge bg-warning-subtle text-warning\">24h</span>\n </div>\n <div class=\"card-body p-3\">\n <div data-container=\"incidents-by-country-chart\"></div>\n </div>\n </div>\n </div>\n </div>\n\n <div class=\"row g-4 mt-1\">\n <div class=\"col-lg-6\">\n <div class=\"card shadow-sm h-100\">\n <div class=\"card-header border-0 bg-transparent d-flex align-items-center justify-content-between\">\n <div>\n <h6 class=\"mb-0 text-uppercase small text-muted\">New Tickets</h6>\n <span class=\"text-muted small\">Fresh tickets awaiting triage</span>\n </div>\n <i class=\"bi bi-ticket-perforated text-muted\"></i>\n </div>\n <div class=\"card-body\" data-container=\"my-tickets-table\"></div>\n </div>\n </div>\n <div class=\"col-lg-6\">\n <div class=\"card shadow-sm h-100\">\n <div class=\"card-header border-0 bg-transparent d-flex align-items-center justify-content-between\">\n <div>\n <h6 class=\"mb-0 text-uppercase small text-muted\">New Incidents</h6>\n <span class=\"text-muted small\">Newest incidents in the queue</span>\n </div>\n <i class=\"bi bi-flag text-warning\"></i>\n </div>\n <div class=\"card-body\" data-container=\"high-priority-incidents-table\"></div>\n </div>\n </div>\n </div>\n </div>\n `;\n }\n\n async onInit() {\n this.header = new IncidentDashboardHeader({\n containerId: 'header'\n });\n this.addChild(this.header);\n\n this.eventsWidget = new MetricsMiniChartWidget({\n containerId: 'events-widget',\n icon: 'bi bi-activity fs-2',\n title: 'System Events',\n subtitle: '{{now_value}} <span class=\"subtitle-label\">{{now_label}}</span>',\n background: '#154360',\n textColor: '#FFFFFF',\n endpoint: '/api/metrics/fetch',\n granularity: 'days',\n slugs: ['incident_events'],\n account: 'incident',\n chartType: 'line',\n showTooltip: true,\n showXAxis: false,\n height: 140,\n color: 'rgba(255,255,255,0.9)',\n fill: true,\n fillColor: 'rgba(255,255,255,0.2)',\n smoothing: 0.3,\n defaultDateRange: '7d',\n valueFormat: 'number',\n showTrending: true,\n showSettings: true,\n settingsKey: 'incident-dashboard-events'\n });\n this.addChild(this.eventsWidget);\n\n this.incidentsWidget = new MetricsMiniChartWidget({\n containerId: 'incidents-widget',\n icon: 'bi bi-exclamation-triangle fs-2',\n title: 'System Incidents',\n subtitle: '{{now_value}} <span class=\"subtitle-label\">{{now_label}}</span>',\n background: '#7D6608',\n textColor: '#FFFFFF',\n endpoint: '/api/metrics/fetch',\n granularity: 'days',\n slugs: ['incidents'],\n account: 'incident',\n chartType: 'line',\n showTooltip: true,\n showXAxis: false,\n height: 140,\n color: 'rgba(255,255,255,0.9)',\n fill: true,\n fillColor: 'rgba(255,255,255,0.25)',\n smoothing: 0.3,\n defaultDateRange: '7d',\n valueFormat: 'number',\n showTrending: true,\n showSettings: true,\n settingsKey: 'incident-dashboard-incidents'\n });\n this.addChild(this.incidentsWidget);\n\n this.eventsByCountryChart = new MetricsChart({\n title: '<i class=\"bi bi-globe-central-south-asia me-2\"></i> Events by Country',\n endpoint: '/api/metrics/fetch',\n account: 'incident',\n category: 'incident_events_by_country',\n granularity: 'days',\n chartType: 'line',\n showDateRange: false,\n showMetricsFilter: false,\n height: 220,\n maxDatasets: 10,\n colors: [\n 'rgba(32, 201, 151, 0.85)'\n ],\n yAxis: {\n label: 'Events',\n beginAtZero: true\n },\n tooltip: { y: 'number:0' },\n containerId: 'events-by-country-chart'\n });\n this.addChild(this.eventsByCountryChart);\n\n this.incidentsByCountryChart = new MetricsChart({\n title: '<i class=\"bi bi-geo-alt me-2\"></i> Incidents by Country',\n endpoint: '/api/metrics/fetch',\n account: 'incident',\n category: 'incidents_by_country',\n granularity: 'days',\n chartType: 'line',\n showDateRange: false,\n showMetricsFilter: false,\n height: 220,\n maxDatasets: 10,\n colors: [\n 'rgba(255, 193, 7, 0.85)'\n ],\n yAxis: {\n label: 'Incidents',\n beginAtZero: true\n },\n tooltip: { y: 'number:0' },\n containerId: 'incidents-by-country-chart'\n });\n this.addChild(this.incidentsByCountryChart);\n\n this.eventsCountryMap = new MetricsCountryMapView({\n containerId: 'events-country-map',\n category: 'incident_events_by_country',\n account: 'incident',\n maxCountries: 20,\n metricLabel: 'Events',\n height: 360,\n mapStyle: 'dark'\n });\n this.addChild(this.eventsCountryMap);\n\n const myTicketsCollection = new TicketList({\n params: {\n status: 'new'\n }\n });\n this.myTicketsTable = new TableView({\n containerId: 'my-tickets-table',\n title: 'New Tickets',\n collection: myTicketsCollection,\n columns: [\n { key: 'id', label: 'ID' },\n { key: 'title', label: 'Title' },\n { key: 'priority', label: 'Priority' }\n ]\n });\n this.addChild(this.myTicketsTable);\n\n const newIncidentsCollection = new IncidentList({\n params: {\n status: 'new'\n }\n });\n this.highPriorityIncidentsTable = new TableView({\n containerId: 'high-priority-incidents-table',\n title: 'New Incidents',\n collection: newIncidentsCollection,\n columns: [\n { key: 'id', label: 'ID' },\n { key: 'title', label: 'Title' },\n { key: 'status', label: 'Status', formatter: 'badge' }\n ]\n });\n this.addChild(this.highPriorityIncidentsTable);\n }\n\n async onActionRefreshAll(event, element) {\n const button = element || event?.currentTarget || null;\n const icon = button?.querySelector?.('i');\n icon?.classList.add('bi-spin');\n if (button) button.disabled = true;\n\n const refreshTasks = [\n this.header?.model?.fetch()?.then(() => this.header.render()),\n this.eventsWidget?.refresh(),\n this.incidentsWidget?.refresh(),\n this.eventsByCountryChart?.refresh(),\n this.incidentsByCountryChart?.refresh(),\n this.eventsCountryMap?.refresh(),\n this.myTicketsTable?.collection?.fetch(),\n this.highPriorityIncidentsTable?.collection?.fetch()\n ].filter(Boolean);\n\n await Promise.allSettled(refreshTasks);\n\n icon?.classList.remove('bi-spin');\n if (button) button.disabled = false;\n }\n}\n\nexport default IncidentDashboardPage;\n","/**\n * StackTraceView - Display formatted and color-coded stack traces\n */\n\nimport View from '@core/View.js';\n\nclass StackTraceView extends View {\n constructor(options = {}) {\n super({\n className: 'stack-trace-view',\n ...options\n });\n\n this.stackTrace = options.stackTrace || '';\n \n this.template = `\n <div class=\"stack-trace-container p-3\">\n <style>\n .stack-trace-line {\n font-family: 'Monaco', 'Menlo', 'Ubuntu Mono', 'Consolas', monospace;\n font-size: 13px;\n line-height: 1.6;\n padding: 4px 8px;\n margin: 0;\n border-left: 3px solid transparent;\n }\n .stack-trace-line:hover {\n background-color: rgba(0, 0, 0, 0.05);\n }\n .stack-trace-error {\n color: #dc3545;\n font-weight: 600;\n border-left-color: #dc3545;\n background-color: rgba(220, 53, 69, 0.05);\n }\n .stack-trace-file {\n color: #0d6efd;\n font-weight: 500;\n border-left-color: #0d6efd;\n }\n .stack-trace-function {\n color: #6610f2;\n font-weight: 500;\n }\n .stack-trace-location {\n color: #6c757d;\n font-size: 12px;\n }\n .stack-trace-line-number {\n color: #fd7e14;\n font-weight: 600;\n }\n .stack-trace-context {\n color: #495057;\n background-color: rgba(0, 0, 0, 0.02);\n }\n .stack-trace-container {\n background-color: #f8f9fa;\n border: 1px solid #dee2e6;\n border-radius: 0.375rem;\n max-height: 600px;\n overflow-y: auto;\n }\n </style>\n <div class=\"stack-trace-content\">\n {{{formattedStackTrace}}}\n </div>\n </div>\n `;\n }\n\n async onBeforeRender() {\n this.formattedStackTrace = this.formatStackTrace(this.stackTrace);\n }\n\n formatStackTrace(stackTrace) {\n if (!stackTrace) {\n return '<div class=\"text-muted p-3\">No stack trace available</div>';\n }\n\n // Convert to string if it's an object\n const traceStr = typeof stackTrace === 'string' ? stackTrace : JSON.stringify(stackTrace, null, 2);\n \n const lines = traceStr.split('\\n');\n let html = '';\n\n lines.forEach((line, index) => {\n if (!line.trim()) {\n html += '<div class=\"stack-trace-line\">&nbsp;</div>';\n return;\n }\n\n // Detect error message (usually the first line)\n if (index === 0 && (line.includes('Error:') || line.includes('Exception:'))) {\n html += `<div class=\"stack-trace-line stack-trace-error\">${this.escapeHtml(line)}</div>`;\n return;\n }\n\n // Detect file paths with line numbers\n // Pattern: at functionName (file.js:123:45) or file.js:123:45\n const filePattern = /(.+?)\\s*\\(([^:]+):(\\d+):(\\d+)\\)/;\n const simpleFilePattern = /^\\s*at\\s+([^:]+):(\\d+):(\\d+)/;\n \n let match = line.match(filePattern);\n if (match) {\n const [, funcName, filePath, lineNum, colNum] = match;\n html += `<div class=\"stack-trace-line stack-trace-file\">\n <span class=\"stack-trace-function\">${this.escapeHtml(funcName.trim())}</span>\n <span class=\"stack-trace-location\"> (${this.escapeHtml(filePath)}:<span class=\"stack-trace-line-number\">${lineNum}</span>:${colNum})</span>\n </div>`;\n return;\n }\n\n match = line.match(simpleFilePattern);\n if (match) {\n const [, filePath, lineNum, colNum] = match;\n html += `<div class=\"stack-trace-line stack-trace-file\">\n <span class=\"stack-trace-location\">at ${this.escapeHtml(filePath)}:<span class=\"stack-trace-line-number\">${lineNum}</span>:${colNum}</span>\n </div>`;\n return;\n }\n\n // Python-style stack trace: File \"...\", line X, in function\n const pythonPattern = /File\\s+\"([^\"]+)\",\\s+line\\s+(\\d+),\\s+in\\s+(.+)/;\n match = line.match(pythonPattern);\n if (match) {\n const [, filePath, lineNum, funcName] = match;\n html += `<div class=\"stack-trace-line stack-trace-file\">\n <span class=\"stack-trace-location\">File \"${this.escapeHtml(filePath)}\", line <span class=\"stack-trace-line-number\">${lineNum}</span>, in </span>\n <span class=\"stack-trace-function\">${this.escapeHtml(funcName)}</span>\n </div>`;\n return;\n }\n\n // Check if line starts with \"at \" (stack frame indicator)\n if (line.trim().startsWith('at ')) {\n html += `<div class=\"stack-trace-line stack-trace-file\">${this.escapeHtml(line)}</div>`;\n return;\n }\n\n // Default: context line\n html += `<div class=\"stack-trace-line stack-trace-context\">${this.escapeHtml(line)}</div>`;\n });\n\n return html;\n }\n\n updateStackTrace(newStackTrace) {\n this.stackTrace = newStackTrace;\n this.render();\n }\n}\n\nexport default StackTraceView;\n","import { IncidentHistory, IncidentHistoryList } from '@core/models/Incident.js';\n\nclass IncidentHistoryAdapter {\n constructor(incidentId) {\n this.incidentId = incidentId;\n this.collection = new IncidentHistoryList({ params: { incident: this.incidentId } });\n }\n\n async fetch() {\n await this.collection.fetch();\n return this.collection.models.map(item => this.transform(item));\n }\n\n transform(item) {\n return {\n id: item.get('id'),\n type: item.get('kind') === 'comment' ? 'user_comment' : 'system_event',\n author: {\n name: item.get('by.display_name') || 'System',\n avatarUrl: item.get('by.avatar.url')\n },\n timestamp: item.get('created'),\n content: item.get('note'),\n attachments: [] // Incident history doesn't have attachments in this phase\n };\n }\n\n async addNote(data) {\n const history = new IncidentHistory({\n incident: this.incidentId,\n note: data.text,\n kind: 'comment'\n });\n const resp = await history.save();\n if (resp.success) {\n await this.collection.fetch();\n }\n return resp;\n }\n}\n\nexport default IncidentHistoryAdapter;\n","/**\n * IncidentView - Detailed view for an Incident record\n */\n\nimport View from '@core/View.js';\nimport TabView from '@core/views/navigation/TabView.js';\nimport DataView from '@core/views/data/DataView.js';\nimport TableView from '@core/views/table/TableView.js';\nimport StackTraceView from '@core/views/data/StackTraceView.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport { Incident, IncidentForms, IncidentEventList } from '@core/models/Incident.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\nimport IncidentHistoryAdapter from './adapters/IncidentHistoryAdapter.js';\nimport ChatView from '@core/views/chat/ChatView.js';\n\nclass IncidentView extends View {\n constructor(options = {}) {\n super({\n className: 'incident-view',\n ...options\n });\n\n this.model = options.model || new Incident(options.data || {});\n this.incidentIcon = this.getIconForIncident(this.model.get('state'));\n\n this.template = `\n <div class=\"incident-view-container\">\n <!-- Header -->\n <div class=\"d-flex justify-content-between align-items-center mb-4\">\n <div class=\"d-flex align-items-center gap-3\">\n <div class=\"fs-1 {{incidentIcon.color}}\">\n <i class=\"bi {{incidentIcon.icon}}\"></i>\n </div>\n <div>\n <h3 class=\"mb-1\">Incident #{{model.id}}</h3>\n <div class=\"text-muted small\">\n Category: {{model.category|capitalize}}\n </div>\n <div class=\"text-muted small mt-1\">\n Created: {{model.created|datetime}}\n </div>\n </div>\n </div>\n <div class=\"d-flex align-items-center gap-4\">\n <div class=\"text-end\">\n <div>State: <span class=\"badge bg-primary\">{{model.state|capitalize}}</span></div>\n <div class=\"text-muted small mt-1\">Priority: {{model.priority}}</div>\n </div>\n <div data-container=\"incident-context-menu\"></div>\n </div>\n </div>\n\n <!-- Tabs -->\n <div data-container=\"incident-tabs\"></div>\n </div>\n `;\n }\n\n getIconForIncident(state) {\n const s = state?.toLowerCase();\n if (s === 'resolved' || s === 'closed') return { icon: 'bi-check-circle-fill', color: 'text-success' };\n if (s === 'new' || s === 'opened') return { icon: 'bi-exclamation-triangle-fill', color: 'text-danger' };\n if (s === 'paused' || s === 'ignore') return { icon: 'bi-pause-circle-fill', color: 'text-warning' };\n return { icon: 'bi-shield-exclamation', color: 'text-secondary' };\n }\n\n async onInit() {\n // Overview Tab\n this.overviewView = new DataView({\n model: this.model,\n className: \"p-3\",\n columns: 2,\n fields: [\n { name: 'id', label: 'Incident ID' },\n { name: 'state', label: 'State', format: 'badge' },\n { name: 'priority', label: 'Priority' },\n { name: 'category', label: 'Category' },\n { name: 'model_name', label: 'Related Model' },\n { name: 'model_id', label: 'Related Model ID' },\n { name: 'details', label: 'Details', columns: 12, format: 'pre' },\n ]\n });\n\n // Events Tab\n const eventsCollection = new IncidentEventList({\n params: { incident: this.model.get('id') }\n });\n this.eventsView = new TableView({\n collection: eventsCollection,\n hideActivePillNames: ['incident'],\n columns: [\n { key: 'id', label: 'ID', width: '70px', sortable: true },\n { key: 'created', label: 'Date', formatter: 'datetime', sortable: true, width: '180px' },\n { key: 'category', label: 'Category', formatter: 'badge', sortable: true },\n { key: 'title', label: 'Title', sortable: true },\n { key: 'level', label: 'Level', sortable: true, width: '80px' },\n ],\n showAdd: false,\n actions: ['view'],\n paginated: true,\n size: 10\n });\n\n // History & Comments Tab\n const adapter = new IncidentHistoryAdapter(this.model.get('id'));\n this.historyView = new ChatView({ adapter });\n\n const tabs = { \n 'Overview': this.overviewView,\n 'Events': this.eventsView,\n 'History & Comments': this.historyView\n };\n \n const metadata = this.model.get('metadata') || {};\n \n // Add Stack Trace tab if present\n if (metadata.stack_trace) {\n this.stackTraceView = new StackTraceView({\n stackTrace: metadata.stack_trace\n });\n tabs['Stack Trace'] = this.stackTraceView;\n }\n \n // Add Metadata tab if there's metadata\n if (Object.keys(metadata).length > 0) {\n this.metadataView = new View({\n model: this.model,\n template: `<pre class=\"bg-light p-3 border rounded\"><code>{{{model.metadata|json}}}</code></pre>`\n });\n tabs['Metadata'] = this.metadataView;\n }\n\n this.tabView = new TabView({\n containerId: 'incident-tabs',\n tabs: tabs,\n activeTab: 'Overview'\n });\n this.addChild(this.tabView);\n\n // ContextMenu\n const incidentMenu = new ContextMenu({\n containerId: 'incident-context-menu',\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: [\n { label: 'Edit Incident', action: 'edit-incident', icon: 'bi-pencil' },\n { label: 'Resolve', action: 'resolve-incident', icon: 'bi-check-circle' },\n { type: 'divider' },\n { label: 'Delete Incident', action: 'delete-incident', icon: 'bi-trash', danger: true }\n ]\n }\n });\n this.addChild(incidentMenu);\n }\n\n async onActionEditIncident() {\n const resp = await Dialog.showModelForm({\n title: `Edit Incident #${this.model.id}`,\n model: this.model,\n formConfig: IncidentForms.edit,\n });\n if (resp) {\n this.render();\n }\n }\n \n async onActionResolveIncident() {\n await this.model.save({ state: 'resolved' });\n this.render();\n this.emit('incident:updated', { model: this.model });\n }\n\n async onActionDeleteIncident() {\n const confirmed = await Dialog.confirm(\n `Are you sure you want to delete this incident?`,\n 'Confirm Deletion',\n { confirmClass: 'btn-danger', confirmText: 'Delete' }\n );\n if (confirmed) {\n const resp = await this.model.destroy();\n if (resp.success) {\n this.emit('incident:deleted', { model: this.model });\n }\n }\n }\n}\n\nIncident.VIEW_CLASS = IncidentView;\n\nexport default IncidentView;\n","/**\n * IncidentTablePage - Incident management using TablePage component\n * Clean implementation using TablePage with minimal overrides\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { IncidentList, IncidentForms } from '@core/models/Incident.js';\nimport IncidentView from './IncidentView.js';\n\nclass IncidentTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_incidents',\n pageName: 'Manage Incidents',\n router: \"admin/incidents\",\n Collection: IncidentList,\n\n formCreate: IncidentForms.create,\n formEdit: IncidentForms.edit,\n itemViewClass: IncidentView,\n\n viewDialogOptions: {\n header: false,\n size: 'xl'\n },\n\n defaultQuery: {\n sort: '-id',\n status: \"new\",\n },\n\n // Column definitions\n columns: [\n {\n key: 'id',\n label: 'ID',\n width: '60px',\n sortable: true,\n class: 'text-muted'\n },\n {\n key: 'status', label: \"Status\",\n filter: {\n type: 'multiselect',\n options: [\"new\", \"open\", \"paused\", \"resolved\", \"qa\", \"ignored\"],\n }\n },\n {\n key: 'created',\n label: 'Created',\n formatter: \"epoch|datetime\",\n filter: {\n type: 'daterange',\n }\n },\n {\n key: 'scope',\n label: 'Scope',\n sortable: true,\n filter: {type:\"text\"}\n },\n {\n key: 'category',\n label: 'Category',\n sortable: true,\n filter: {type:\"text\"}\n },\n {\n key: 'priority',\n label: 'Priority',\n filter: {type:\"text\"}\n },\n {\n key: 'title',\n label: 'title',\n formatter: \"truncate(100)|default('No description')\"\n }\n ],\n\n filters: [\n {\n key: 'category__not',\n label: 'Not Category',\n filter: {type:\"text\"}\n },\n ],\n\n // Table features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Toolbar\n showRefresh: true,\n showAdd: true,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No incidents found. Click \"Add Incident\" to create your first incident.',\n\n // Batch actions\n batchBarLocation: 'top',\n batchActions: [\n { label: \"Open\", icon: \"bi bi-folder2-open\", action: \"open\" },\n { label: \"Resolve\", icon: \"bi bi-check-circle\", action: \"resolve\" },\n { label: \"Pause\", icon: \"bi bi-pause-circle\", action: \"pause\" },\n { label: \"Ignore\", icon: \"bi bi-x-circle\", action: \"ignore\" },\n { label: \"Merge\", icon: \"bi bi-merge\", action: \"merge\" }\n ],\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n }\n });\n }\n\n async onActionBatchResolve(event, element) {\n const selected = this.tableView.getSelectedItems();\n if (!selected.length) return;\n\n const app = this.getApp();\n const result = await app.confirm(`Are you sure you want to close ${selected.length} incidents?`);\n if (!result) return;\n await Promise.all(selected.map(item => item.model.save({status: 'resolved'})));\n this.tableView.collection.fetch();\n }\n\n async onActionBatchOpen(event, element) {\n const selected = this.tableView.getSelectedItems();\n if (!selected.length) return;\n\n const app = this.getApp();\n const result = await app.confirm(`Are you sure you want to open ${selected.length} incidents?`);\n if (!result) return;\n await Promise.all(selected.map(item => item.model.save({status: 'open'})));\n this.tableView.collection.fetch();\n }\n\n async onActionBatchPause(event, element) {\n const selected = this.tableView.getSelectedItems();\n if (!selected.length) return;\n\n const app = this.getApp();\n const result = await app.confirm(`Are you sure you want to pause ${selected.length} incidents?`);\n if (!result) return;\n await Promise.all(selected.map(item => item.model.save({status: 'paused'})));\n this.tableView.collection.fetch();\n }\n\n async onActionBatchIgnore(event, element) {\n const selected = this.tableView.getSelectedItems();\n if (!selected.length) return;\n\n const app = this.getApp();\n const result = await app.confirm(`Are you sure you want to ignore ${selected.length} incidents?`);\n if (!result) return;\n await Promise.all(selected.map(item => item.model.save({status: 'ignored'})));\n this.tableView.collection.fetch();\n }\n\n async onActionBatchMerge(_event, _element) {\n const selected = this.tableView.getSelectedItems();\n if (!selected.length) return;\n\n const app = this.getApp();\n const result = await app.showForm({\n title: `Merge ${selected.length} incidents`,\n fields: [\n {\n name: 'merge',\n type: 'select',\n label: 'Select Parent Incident',\n options: selected.map(item => ({value: item.model.id, label: item.model.id})),\n required: true\n }\n ]\n });\n if (!result) return;\n\n // Find the parent model from selected items\n const parentModel = selected.find(item => item.model.id == result.merge)?.model;\n if (!parentModel) return;\n\n // Get list of all IDs to merge (excluding the parent)\n const mergeIds = selected\n .map(item => item.model.id)\n .filter(id => id != result.merge);\n\n // Save the merge operation to the parent model\n await parentModel.save({ merge: mergeIds });\n this.tableView.collection.fetch();\n }\n}\n\nexport default IncidentTablePage;\n","/**\n * EventView - Detailed view for an IncidentEvent record\n */\n\nimport View from '@core/View.js';\nimport TabView from '@core/views/navigation/TabView.js';\nimport DataView from '@core/views/data/DataView.js';\nimport StackTraceView from '@core/views/data/StackTraceView.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport { IncidentEvent } from '@core/models/Incident.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\n\nclass EventView extends View {\n constructor(options = {}) {\n super({\n className: 'event-view',\n ...options\n });\n\n this.model = options.model || new IncidentEvent(options.data || {});\n this.eventIcon = this.getIconForEvent(this.model.get('level'));\n\n this.template = `\n <div class=\"event-view-container\">\n <!-- Header -->\n <div class=\"d-flex justify-content-between align-items-center mb-4\">\n <div class=\"d-flex align-items-center gap-3\">\n <div class=\"fs-1 {{eventIcon.color}}\">\n <i class=\"bi {{eventIcon.icon}}\"></i>\n </div>\n <div>\n <h3 class=\"mb-1\">{{model.title|default('System Event')}}</h3>\n <div class=\"text-muted small\">\n Category: {{model.category|capitalize}}\n </div>\n <div class=\"text-muted small mt-1\">\n {{model.created|datetime}} from {{model.source_ip|default('Unknown IP')}}\n </div>\n </div>\n </div>\n <div data-container=\"event-context-menu\"></div>\n </div>\n\n <!-- Body -->\n <div data-container=\"event-tabs\"></div>\n </div>\n `;\n }\n\n getIconForEvent(level) {\n if (level >= 40) return { icon: 'bi-exclamation-octagon-fill', color: 'text-danger' }; // Error\n if (level >= 30) return { icon: 'bi-exclamation-triangle-fill', color: 'text-warning' }; // Warning\n if (level >= 20) return { icon: 'bi-info-circle-fill', color: 'text-info' }; // Info\n return { icon: 'bi-bell-fill', color: 'text-secondary' }; // Debug/Default\n }\n\n async onInit() {\n // Overview Tab\n this.overviewView = new DataView({\n model: this.model,\n className: \"p-3\",\n columns: 2,\n fields: [\n { name: 'id', label: 'Event ID' },\n { name: 'level', label: 'Level' },\n { name: 'hostname', label: 'Hostname' },\n { name: 'incident', label: 'Incident ID' },\n { name: 'model_name', label: 'Related Model' },\n { name: 'model_id', label: 'Related Model ID' },\n { name: 'details', label: 'Details', columns: 12 },\n ]\n });\n\n const tabs = { 'Overview': this.overviewView };\n \n const metadata = this.model.get('metadata') || {};\n \n // Add Stack Trace tab if present\n if (metadata.stack_trace) {\n this.stackTraceView = new StackTraceView({\n stackTrace: metadata.stack_trace\n });\n tabs['Stack Trace'] = this.stackTraceView;\n }\n \n // Add Metadata tab if there's metadata\n if (Object.keys(metadata).length > 0) {\n this.metadataView = new View({\n model: this.model,\n template: `<pre class=\"bg-light p-3 border rounded\"><code>{{{model.metadata|json}}}</code></pre>`\n });\n tabs['Metadata'] = this.metadataView;\n }\n\n this.tabView = new TabView({\n containerId: 'event-tabs',\n tabs: tabs,\n activeTab: 'Overview'\n });\n this.addChild(this.tabView);\n\n // ContextMenu\n const menuItems = [\n { label: 'View Incident', action: 'view-incident', icon: 'bi-shield-exclamation', disabled: !this.model.get('incident') },\n { label: 'View Related Model', action: 'view-model', icon: 'bi-box-arrow-up-right', disabled: !this.model.get('model_id') },\n { type: 'divider' },\n { label: 'Delete Event', action: 'delete-event', icon: 'bi-trash', danger: true }\n ];\n\n const eventMenu = new ContextMenu({\n containerId: 'event-context-menu',\n className: \"context-menu-view header-menu-absolute\",\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: menuItems\n }\n });\n this.addChild(eventMenu);\n }\n\n async onActionViewIncident() {\n console.log(\"TODO: View incident\", this.model.get('incident'));\n }\n\n async onActionViewModel() {\n console.log(\"TODO: View model\", this.model.get('model_name'), this.model.get('model_id'));\n }\n\n async onActionDeleteEvent() {\n const confirmed = await Dialog.confirm(\n `Are you sure you want to delete this event? This action cannot be undone.`,\n 'Confirm Deletion',\n { confirmClass: 'btn-danger', confirmText: 'Delete' }\n );\n if (confirmed) {\n const resp = await this.model.destroy();\n if (resp.success) {\n this.emit('event:deleted', { model: this.model });\n }\n }\n }\n}\n\nIncidentEvent.VIEW_CLASS = EventView;\n\nexport default EventView;\n","/**\n * EventTablePage - System events management using TablePage component\n * Clean implementation using TablePage with minimal overrides\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { IncidentEventList, IncidentEventForms } from '@core/models/Incident.js';\nimport EventView from './EventView.js';\n\nclass EventTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_events',\n pageName: 'System Events',\n router: \"admin/events\",\n Collection: IncidentEventList,\n\n formEdit: IncidentEventForms.edit,\n itemViewClass: EventView,\n\n viewDialogOptions: {\n header: false,\n size: 'lg'\n },\n\n defaultQuery: {\n sort: '-id',\n category__not: \"ossec\",\n },\n\n // Column definitions\n columns: [\n {\n key: 'created', label: 'Timestamp',\n sortable: true, formatter: 'datetime',\n filter: {\n type: 'daterange',\n }\n },\n {\n key: 'level', label: 'Level',\n sortable: true, formatter: 'badge',\n filter: {\n type: \"select\",\n options: [\n { value: '5', label: 'Critical' },\n { value: '4', label: 'Warning' },\n { value: '3', label: 'Info' },\n { value: '2', label: 'Debug' },\n { value: '1', label: 'Trace' }\n ]\n }\n },\n {\n key: 'scope',\n label: 'Scope',\n sortable: true, formatter: 'badge',\n filter: {\n type: \"combobox\",\n options: [\n { value: 'account', label: 'Account' },\n { value: 'incident', label: 'Incident' },\n { value: 'ossec', label: 'OSSEC' },\n { value: 'fileman', label: 'File Manager' },\n { value: 'metrics', label: 'Metrics' },\n { value: 'jobs', label: 'Jobs' },\n { value: 'aws', label: 'AWS' }\n\n ]\n }\n },\n {\n key: 'category',\n label: 'Category',\n sortable: true, formatter: 'badge',\n filter: {\n type: \"combobox\",\n options: [\n { value: 'rest_error', label: 'Rest Error' },\n { value: 'api_error', label: 'API Error' },\n { value: 'auth', label: 'Auth' },\n { value: 'database', label: 'Database' }\n ]\n }\n },\n { key: 'title', label: 'Title', sortable: true, formatter: 'truncate(50)' },\n {\n key: 'source_ip', label: 'Source IP', sortable: true,\n filter: {\n type: \"text\"\n }\n },\n {\n key: 'metadata.server', label: 'Server',\n sortable: true,\n filter: {\n type: \"text\"\n }\n },\n ],\n\n filters: [\n {\n key: 'category__not',\n label: 'Not Category',\n filter: {type:\"text\"}\n },\n {\n key: 'metadata__http_url__icontains',\n label: 'URL Contains',\n filter: {type:\"text\"}\n },\n {\n key: 'metadata__http_path__icontains',\n label: 'Path Contains',\n filter: {type:\"text\"}\n },\n {\n key: 'metadata__http_query_string__icontains',\n label: 'Query String Contains',\n filter: {type:\"text\"}\n },\n {\n key: 'metadata__rule_id',\n label: 'Rule ID',\n filter: {type:\"text\"}\n },\n {\n key: 'metadata__country_code',\n label: 'Country',\n filter: {type:\"text\"}\n },\n {\n key: 'metadata__region',\n label: 'Region',\n filter: {type:\"text\"}\n },\n {\n key: 'metadata__city__icontains',\n label: 'City',\n filter: {type:\"text\"}\n },\n {\n key: 'metadata__http_status',\n label: 'HTTP Status',\n filter: {type:\"text\"}\n },\n {\n key: 'model_name',\n label: 'Model Name',\n filter: {type:\"text\"}\n },\n {\n key: 'model_id',\n label: 'Model ID',\n filter: {type:\"text\"}\n },\n {\n key: 'metadata__user_email',\n label: 'User Email',\n filter: {type:\"text\"}\n },\n {\n key: 'metadata__http_user_agent__icontains',\n label: 'User Agent Contains',\n filter: {type:\"text\"}\n },\n ],\n\n // Table features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Toolbar\n showRefresh: true,\n showAdd: false,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No events found.',\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n }\n });\n }\n}\n\nexport default EventTablePage;\n","import { TicketNote, TicketNoteList } from '@core/models/Tickets.js';\n\nclass TicketNoteAdapter {\n constructor(ticketId) {\n this.ticketId = ticketId;\n this.collection = new TicketNoteList({ params: { parent: this.ticketId, sort: 'created', size: 100} });\n }\n\n async fetch() {\n await this.collection.fetch();\n return this.collection.models.map(note => this.transform(note));\n }\n\n transform(note) {\n return {\n id: note.get('id'),\n type: 'user_comment', // Ticket notes are always user comments\n author: {\n id: note.get('user.id'),\n name: note.get('user.display_name') || 'System',\n avatarUrl: note.get('user.avatar.url')\n },\n timestamp: note.get('created'),\n content: note.get('note'),\n attachments: note.get('media') ? [note.get('media')] : []\n };\n }\n\n async addNote(data) {\n const note = new TicketNote();\n const resp = await note.save({\n parent: this.ticketId,\n note: data.text,\n media: data.files && data.files.length > 0 ? data.files[0].id : null\n });\n if (resp.success) {\n await this.collection.fetch(); // Refresh the collection\n }\n return resp;\n }\n}\n\nexport default TicketNoteAdapter;\n","import View from '@core/View.js';\nimport DataView from '@core/views/data/DataView.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport { Ticket, TicketForms } from '@core/models/Tickets.js';\nimport ChatView from '@core/views/chat/ChatView.js';\nimport TicketNoteAdapter from './adapters/TicketNoteAdapter.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\n\nclass TicketView extends View {\n constructor(options = {}) {\n super({\n className: 'ticket-view',\n ...options\n });\n\n this.model = options.model || new Ticket(options.data || {});\n\n this.template = `\n <div class=\"ticket-view-container d-flex flex-column h-100\">\n <!-- Ticket Header -->\n <div class=\"d-flex justify-content-between align-items-start mb-3 flex-shrink-0\">\n <!-- Left Side: Primary Identity -->\n <div class=\"d-flex align-items-center gap-3\">\n <div class=\"avatar-placeholder rounded-circle bg-light d-flex align-items-center justify-content-center\" style=\"width: 80px; height: 80px;\">\n <i class=\"bi bi-ticket-perforated text-secondary\" style=\"font-size: 40px;\"></i>\n </div>\n <div>\n <h3 class=\"mb-1\">{{model.title|truncate(50)|default('Untitled Ticket')}}</h3>\n <div class=\"text-muted small\">\n <span>Ticket #{{model.id}}</span>\n <span class=\"mx-2\">|</span>\n <span>Priority: {{model.priority|capitalize}}</span>\n {{#model.assignee}}\n <span class=\"mx-2\">|</span>\n <span>Assigned to: {{model.assignee.display_name}}</span>\n {{/model.assignee}}\n </div>\n {{#model.incident}}\n <div class=\"text-muted small mt-1\">\n <i class=\"bi bi-exclamation-triangle\"></i> Related to incident: {{model.incident}}\n </div>\n {{/model.incident}}\n </div>\n </div>\n\n <!-- Right Side: Status & Actions -->\n <div class=\"d-flex align-items-start gap-4\">\n <div class=\"text-end\">\n <div class=\"d-flex align-items-center gap-2\">\n <span class=\"badge {{model.status|badgeClass}}\">{{model.status|capitalize}}</span>\n </div>\n {{#model.created}}\n <div class=\"text-muted small mt-1\">Created {{model.created|relative}}</div>\n {{/model.created}}\n {{#model.modified}}\n <div class=\"text-muted small\">Updated {{model.modified|relative}}</div>\n {{/model.modified}}\n </div>\n <div data-container=\"ticket-context-menu\"></div>\n </div>\n </div>\n\n <!-- Chat View (Full height) -->\n <div class=\"flex-grow-1\" style=\"min-height: 0;\" data-container=\"chat-view\"></div>\n </div>\n `;\n }\n\n async onInit() {\n // Chat View with compact theme (Option 4)\n const adapter = new TicketNoteAdapter(this.model.get('id'));\n this.chatView = new ChatView({\n containerId: 'chat-view',\n adapter: adapter,\n theme: 'compact', // Use compact admin-style theme\n currentUserId: this.getCurrentUserId(),\n inputPlaceholder: 'Add a note...',\n inputButtonText: 'Add Note'\n });\n this.addChild(this.chatView);\n\n // Context Menu\n const ticketMenu = new ContextMenu({\n containerId: 'ticket-context-menu',\n className: \"context-menu-view header-menu-absolute\",\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: [\n { label: 'Edit Ticket', action: 'edit-ticket', icon: 'bi-pencil' },\n { label: 'Change Status', action: 'change-status', icon: 'bi-tag' },\n { label: 'Set Priority', action: 'set-priority', icon: 'bi-flag' },\n { label: 'Assign User', action: 'assign-user', icon: 'bi-person' },\n { type: 'divider' },\n { label: 'Close Ticket', action: 'close-ticket', icon: 'bi-x-circle' },\n ]\n }\n });\n this.addChild(ticketMenu);\n }\n\n /**\n * Get current user ID for chat message positioning\n * @returns {number|null}\n */\n getCurrentUserId() {\n // Get from WebApp state or wherever your app stores current user\n const currentUser = window.app?.state?.user;\n return currentUser?.id || null;\n }\n\n // Context Menu Action Handlers\n async onActionEditTicket() {\n const resp = await Dialog.showModelForm({\n title: `Edit Ticket #${this.model.get('id')} - ${this.model.get('title')}`,\n model: this.model,\n size: 'lg',\n fields: TicketForms.edit.fields\n });\n if (resp) {\n this.render(); // Re-render to show updated data in header\n }\n }\n\n async onActionChangeStatus() {\n const statuses = ['new', 'open', 'in_progress', 'pending', 'resolved', 'closed', 'ignored'];\n const currentStatus = this.model.get('status');\n\n const result = await Dialog.showForm({\n title: 'Change Ticket Status',\n size: 'sm',\n fields: [\n {\n name: 'status',\n label: 'New Status',\n type: 'select',\n options: statuses.map(s => ({ value: s, label: s.replace('_', ' ').toUpperCase() })),\n value: currentStatus,\n required: true\n }\n ]\n });\n\n if (result) {\n try {\n await this.model.save({ status: result.status });\n this.render();\n } catch (error) {\n Dialog.alert({\n type: 'error',\n title: 'Error',\n message: 'Failed to update ticket status: ' + error.message\n });\n }\n }\n }\n\n async onActionSetPriority() {\n const priorities = ['low', 'normal', 'high', 'urgent'];\n const currentPriority = this.model.get('priority');\n\n const result = await Dialog.showForm({\n title: 'Set Ticket Priority',\n size: 'sm',\n fields: [\n {\n name: 'priority',\n label: 'Priority Level',\n type: 'select',\n options: priorities.map(p => ({ value: p, label: p.toUpperCase() })),\n value: currentPriority,\n required: true\n }\n ]\n });\n\n if (result) {\n try {\n await this.model.save({ priority: result.priority });\n this.render();\n } catch (error) {\n Dialog.alert({\n type: 'error',\n title: 'Error',\n message: 'Failed to update ticket priority: ' + error.message\n });\n }\n }\n }\n\n async onActionAssignUser() {\n console.log(\"TODO: Implement assign user dialog with user selector\");\n Dialog.alert({\n title: 'Coming Soon',\n message: 'User assignment feature will be implemented soon.'\n });\n }\n\n async onActionCloseTicket() {\n const confirmed = await Dialog.confirm({\n title: 'Close Ticket',\n message: `Are you sure you want to close ticket #${this.model.get('id')}?`,\n confirmText: 'Close Ticket',\n confirmClass: 'btn-warning'\n });\n\n if (confirmed) {\n try {\n await this.model.save({ status: 'closed' });\n this.render();\n Dialog.alert({\n type: 'success',\n title: 'Success',\n message: 'Ticket has been closed successfully.'\n });\n } catch (error) {\n Dialog.alert({\n type: 'error',\n title: 'Error',\n message: 'Failed to close ticket: ' + error.message\n });\n }\n }\n }\n}\n\nTicket.VIEW_CLASS = TicketView;\n\nexport default TicketView;\n","/**\n * TicketTablePage - Ticket management using TablePage component\n * Clean implementation using TablePage with minimal overrides\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { TicketList, TicketForms, TicketCategories } from '@core/models/Tickets.js';\nimport TicketView from './TicketView.js';\n\nclass TicketTablePage extends TablePage {\n constructor(options = {}) {\n super({\n name: 'admin_tickets',\n pageName: 'Tickets',\n router: \"admin/tickets\",\n Collection: TicketList,\n\n formCreate: TicketForms.create,\n formEdit: TicketForms.edit,\n itemViewClass: TicketView,\n\n viewDialogOptions: {\n header: false\n },\n\n defaultQuery: {\n sort: '-priority',\n status: \"open\"\n },\n\n // Column definitions\n columns: [\n { key: 'id', label: 'ID', width: '70px', sortable: true, class: 'text-muted' },\n { key: 'title', label: 'Title', sortable: true},\n {\n key: 'status', label: 'Status', sortable: true,\n editable: true,\n editableOptions: {\n type: \"select\",\n options: [\n \"new\", \"open\", \"paused\", \"resolved\", \"qa\", \"ignored\"\n ]\n },\n filter: {\n type: \"multiselect\",\n placeHolder: \"Select Status\",\n options: [\n \"new\", \"open\", \"paused\", \"resolved\", \"qa\", \"ignored\"\n ]\n }\n },\n { key: 'priority', label: 'Priority', sortable: true },\n {\n key: 'category', label: 'Category', sortable: true,\n editable: true,\n editableOptions: {\n type: \"select\",\n options: [\n ... Object.keys(TicketCategories)\n ]\n },\n filter: {\n type: \"multiselect\",\n placeHolder: \"Select Category\",\n options: [\n ... Object.keys(TicketCategories)\n ]\n }\n },\n { key: 'assignee.display_name', label: 'Assignee', sortable: true, formatter: \"default('Unassigned')\" },\n { key: 'incident.id', label: 'Incident ID', sortable: true },\n { key: 'created', label: 'Created', sortable: true, formatter: 'datetime' }\n ],\n\n // Table features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Toolbar\n showRefresh: true,\n showAdd: true,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No tickets found.',\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n },\n ...options,\n });\n }\n}\n\nexport default TicketTablePage;\n","import View from '@core/View.js';\nimport TabView from '@core/views/navigation/TabView.js';\nimport DataView from '@core/views/data/DataView.js';\nimport TableView from '@core/views/table/TableView.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport { RuleSet, RuleList, BundleByOptions, MatchByOptions } from '@core/models/Incident.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\n\nclass RuleSetView extends View {\n constructor(options = {}) {\n super({\n className: 'ruleset-view',\n ...options\n });\n\n this.model = options.model || new RuleSet(options.data || {});\n\n this.template = `\n <div class=\"ruleset-view-container\">\n <!-- Header -->\n <div class=\"d-flex justify-content-between align-items-center mb-4\">\n <div class=\"d-flex align-items-center gap-3\">\n <div class=\"fs-1 text-primary\"><i class=\"bi bi-gear-wide-connected\"></i></div>\n <div>\n <h3 class=\"mb-1\">{{model.name}}</h3>\n <div class=\"text-muted small\">Scope: {{model.category}} | Priority: {{model.priority}}</div>\n </div>\n </div>\n <div data-container=\"ruleset-context-menu\"></div>\n </div>\n\n <!-- Tabs -->\n <div data-container=\"ruleset-tabs\"></div>\n </div>\n `;\n }\n\n async onInit() {\n // Get labels for current values\n const matchByValue = this.model.get('match_by');\n const matchByOption = MatchByOptions.find(opt => opt.value === matchByValue);\n const matchByLabel = matchByOption ? matchByOption.label : String(matchByValue);\n\n const bundleByValue = this.model.get('bundle_by');\n const bundleByOption = BundleByOptions.find(opt => opt.value === bundleByValue);\n const bundleByLabel = bundleByOption ? bundleByOption.label : String(bundleByValue);\n\n // Config Tab\n this.configView = new DataView({\n model: this.model,\n className: \"p-3\",\n columns: 2,\n fields: [\n { name: 'name', label: 'Name', cols: 4 },\n { name: 'category', label: 'Scope', formatter: 'badge', cols: 4 },\n { name: 'is_active', label: 'Is Active', formatter: 'yesno_icon', cols: 4 },\n { name: 'priority', label: 'Priority', cols: 4 },\n { name: 'id', label: 'RuleSet ID', cols: 4 },\n\n {\n name: 'match_by',\n label: 'Match Logic',\n template: matchByLabel,\n cols: 4\n },\n {\n name: 'bundle_by',\n label: 'Bundle By',\n template: bundleByLabel,\n cols: 4\n },\n { name: 'bundle_minutes', label: 'Bundle Minutes', cols: 4 },\n { name: 'bundle_by_rule_set', label: 'Bundle By Rule Set', formatter: 'yesno_icon', cols: 4 },\n { name: 'handler', label: 'Handler', cols: 12 },\n ]\n });\n\n // Rules Tab\n const rulesCollection = new RuleList({\n params: { parent: this.model.get('id') }\n });\n this.rulesView = new TableView({\n collection: rulesCollection,\n hideActivePillNames: ['parent'],\n columns: [\n { key: 'id', label: 'ID', width: '70px' },\n { key: 'name', label: 'Name' },\n { key: 'field_name', label: 'Field' },\n { key: 'comparator', label: 'Comparator', width: '120px' },\n { key: 'value', label: 'Value' },\n { key: 'value_type', label: 'Type', width: '100px' },\n ],\n showAdd: true,\n clickAction: 'edit',\n actions: ['edit', 'delete'],\n contextMenu: [\n { label: 'Edit Rule', action: 'edit', icon: 'bi-pencil' },\n { label: 'Duplicate Rule', action: 'duplicate', icon: 'bi-files' },\n { divider: true },\n { label: 'Delete Rule', action: 'delete', icon: 'bi-trash', danger: true }\n ],\n // Pass the parent ID so new rules get associated with this ruleset\n addFormDefaults: {\n parent: this.model.get('id')\n }\n });\n\n this.tabView = new TabView({\n containerId: 'ruleset-tabs',\n tabs: {\n 'Configuration': this.configView,\n 'Rules': this.rulesView\n },\n activeTab: 'Configuration'\n });\n this.addChild(this.tabView);\n\n const contextMenu = new ContextMenu({\n containerId: 'ruleset-context-menu',\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: [\n { label: 'Edit RuleSet', action: 'edit-ruleset', icon: 'bi-pencil' },\n { label: 'Disable', action: 'disable-ruleset', icon: 'bi-toggle-off' },\n { type: 'divider' },\n { label: 'Delete RuleSet', action: 'delete-ruleset', icon: 'bi-trash', danger: true }\n ]\n }\n });\n this.addChild(contextMenu);\n }\n\n /**\n * Action handler: Edit RuleSet\n */\n async onActionEditRuleset() {\n const resp = await Dialog.showModelForm({\n title: `Edit RuleSet - ${this.model.get('name')}`,\n model: this.model,\n formConfig: RuleSet.EDIT_FORM,\n });\n if (resp) {\n await this.render();\n }\n }\n\n /**\n * Action handler: Disable/Enable RuleSet\n */\n async onActionDisableRuleset() {\n const isActive = this.model.get('is_active');\n const newStatus = !isActive;\n\n try {\n this.model.set('is_active', newStatus);\n await this.model.save();\n await this.render();\n\n Dialog.showToast({\n message: `RuleSet ${newStatus ? 'enabled' : 'disabled'} successfully`,\n type: 'success'\n });\n } catch (error) {\n Dialog.showToast({\n message: `Failed to update RuleSet: ${error.message}`,\n type: 'error'\n });\n }\n }\n\n /**\n * Action handler: Delete RuleSet\n */\n async onActionDeleteRuleset() {\n const confirmed = await Dialog.confirm({\n title: 'Delete RuleSet',\n message: `Are you sure you want to delete the ruleset \"${this.model.get('name')}\"? This action cannot be undone.`,\n confirmText: 'Delete',\n confirmClass: 'btn-danger'\n });\n\n if (confirmed) {\n try {\n await this.model.destroy();\n\n Dialog.showToast({\n message: 'RuleSet deleted successfully',\n type: 'success'\n });\n\n // Close the dialog\n const dialog = this.element?.closest('.modal');\n if (dialog) {\n const bsModal = bootstrap.Modal.getInstance(dialog);\n if (bsModal) {\n bsModal.hide();\n }\n }\n\n // Emit event to parent to refresh the table\n this.emit('ruleset:deleted', { model: this.model });\n } catch (error) {\n Dialog.showToast({\n message: `Failed to delete RuleSet: ${error.message}`,\n type: 'error'\n });\n }\n }\n }\n}\n\nRuleSetView.VIEW_CLASS = RuleSetView;\n\nexport default RuleSetView;\n","import TablePage from '@core/pages/TablePage.js';\nimport { RuleSetList } from '@core/models/Incident.js';\nimport RuleSetView from './RuleSetView.js';\n\nclass RuleSetTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_rulesets',\n pageName: 'Rule Engine',\n router: \"admin/rulesets\",\n Collection: RuleSetList,\n itemView: RuleSetView,\n viewDialogOptions: {\n header: false,\n size: 'xl'\n },\n\n columns: [\n { key: 'id', label: 'ID', width: '70px', sortable: true, class: 'text-muted' },\n { key: 'name', label: 'Name', sortable: true },\n { key: 'category', label: 'Scope', sortable: true, formatter: 'badge' },\n { key: 'priority', label: 'Priority', sortable: true },\n { key: 'match_by', label: 'Match Logic', formatter: (v) => v === 0 ? 'ALL' : 'ANY' }\n ],\n\n selectable: true,\n searchable: true,\n sortable: true,\n paginated: true,\n showRefresh: true,\n showAdd: true,\n showExport: true,\n\n tableOptions: {\n pageSizes: [10, 25, 50],\n defaultPageSize: 25,\n emptyMessage: 'No rule sets found.',\n emptyIcon: 'bi-gear',\n actions: [\"view\", \"edit\", \"delete\"],\n }\n });\n }\n}\n\nexport default RuleSetTablePage;\n","/**\n * EmailDomainTablePage - Admin page for managing SES/SNS Email Domains\n * Clean implementation following simplified TablePage pattern\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\nimport View from '@core/View.js';\nimport { EmailDomain, EmailDomainList, EmailDomainForms } from '@core/models/Email.js';\n\nclass EmailDomainTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_email_domains',\n pageName: 'Email Domains',\n router: 'admin/email/domains',\n Collection: EmailDomainList,\n formCreate: EmailDomainForms.create,\n formEdit: EmailDomainForms.edit,\n\n // Column definitions\n columns: [\n {\n key: 'id',\n label: 'ID',\n width: '70px',\n sortable: true,\n class: 'text-muted'\n },\n {\n key: 'name',\n label: 'Domain',\n sortable: true\n },\n {\n key: 'region',\n label: 'Region',\n sortable: true,\n formatter: \"default('—')\"\n },\n {\n key: 'receiving_enabled',\n label: 'Receiving',\n formatter: \"boolean|badge\"\n },\n {\n key: 'can_send',\n label: 'Send Verified',\n formatter: \"boolean|badge\"\n },\n {\n key: 'can_recv',\n label: 'Recv Verified',\n formatter: \"boolean|badge\"\n },\n {\n key: 'created',\n label: 'Created',\n formatter: \"epoch|datetime\"\n }\n ],\n\n // Table features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Toolbar\n showRefresh: true,\n showAdd: true,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No domains found. Click \"Add Domain\" to get started.',\n\n // Context menu configuration\n contextMenu: [\n {\n icon: 'bi-shield-check',\n action: 'edit-aws-creds',\n label: 'Edit AWS Credentials'\n },\n {\n icon: 'bi-rocket-takeoff',\n action: 'onboard',\n label: 'Onboard'\n },\n {\n icon: 'bi-shield-check',\n action: 'audit',\n label: 'Audit'\n },\n {\n icon: 'bi-arrow-repeat',\n action: 'reconcile',\n label: 'Reconcile'\n }\n ],\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n }\n });\n }\n\n async onActionEditAwsCreds(event, element) {\n const item = this.collection.get(element.dataset.id);\n const result = await Dialog.showModelForm({\n model: item,\n formConfig: EmailDomainForms.credentials,\n }\n );\n return true;\n }\n\n async onActionOnboard(event, element) {\n const item = this.collection.get(element.dataset.id);\n const model = new EmailDomain({ id: item.id });\n\n const formData = await Dialog.showForm(EmailDomainForms.onboard);\n if (!formData) return;\n\n try {\n const resp = await model.onboard(formData);\n if (!resp.success) {\n throw new Error(resp.message || 'Network error during onboarding');\n }\n if (!resp.data?.status) {\n throw new Error(resp.data?.error || 'Onboarding failed');\n }\n\n this.getApp()?.toast?.success('Domain onboarding completed successfully');\n await this.refresh();\n } catch (err) {\n console.error('Onboard error:', err);\n this.showError(err.message || 'Failed to onboard domain');\n }\n }\n\n async onActionAudit(event, element) {\n const item = this.collection.get(element.dataset.id);\n const model = new EmailDomain({ id: item.id });\n\n try {\n const resp = await model.audit();\n if (!resp.success) {\n throw new Error(resp.message || 'Network error during audit');\n }\n if (!resp.data?.status) {\n throw new Error(resp.data?.error || 'Audit failed');\n }\n\n const result = resp.data?.data || {};\n await Dialog.showDialog({\n title: `Audit Report - ${item.name}`,\n body: new View({\n template: `\n <div>\n <p class=\"text-muted\">Drift report and status:</p>\n <pre class=\"bg-light p-3 rounded small\"><code>{{{data.result|json}}}</code></pre>\n </div>\n `,\n data: { result }\n }),\n size: 'lg'\n });\n } catch (err) {\n console.error('Audit error:', err);\n this.showError(err.message || 'Failed to audit domain');\n }\n }\n\n async onActionReconcile(event, element) {\n const item = this.collection.get(element.dataset.id);\n const model = new EmailDomain({ id: item.id });\n\n try {\n const resp = await model.reconcile();\n if (!resp.success) {\n throw new Error(resp.message || 'Network error during reconcile');\n }\n if (!resp.data?.status) {\n throw new Error(resp.data?.error || 'Reconcile failed');\n }\n\n this.getApp()?.toast?.success('Reconcile completed successfully');\n await this.refresh();\n } catch (err) {\n console.error('Reconcile error:', err);\n this.showError(err.message || 'Failed to reconcile domain');\n }\n }\n}\n\nexport default EmailDomainTablePage;\n","/**\n * EmailMailboxTablePage - Admin page for managing Mailboxes\n * Clean implementation using TablePage with minimal overrides\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { MailboxList, MailboxForms, Mailbox } from '@core/models/Email.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\n\nclass EmailMailboxTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_email_mailboxes',\n pageName: 'Mailboxes',\n router: 'admin/email/mailboxes',\n Collection: MailboxList,\n\n formCreate: MailboxForms.create,\n formEdit: MailboxForms.edit,\n clickAction: \"edit\",\n\n // Table columns\n columns: [\n { key: 'id', label: 'ID', width: '70px', sortable: true, class: 'text-muted' },\n { key: 'email', label: 'Email', sortable: true },\n { key: 'domain.name', label: 'Domain', sortable: true, formatter: \"default('—')\" },\n { key: 'allow_inbound', label: 'Inbound', formatter: \"boolean|badge\" },\n { key: 'allow_outbound', label: 'Outbound', formatter: \"boolean|badge\" },\n { key: 'is_system_default', label: 'System Default', formatter: \"boolean|badge\" },\n { key: 'is_domain_default', label: 'Domain Default', formatter: \"boolean|badge\" }\n ],\n\n // Features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Toolbar\n showRefresh: true,\n showAdd: true,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No mailboxes found. Click \"Add Mailbox\" to create one.',\n\n // Context menu configuration\n contextMenu: [\n { icon: 'bi-envelope', action: 'send-email', label: 'Send Email' },\n { icon: 'bi-envelope', action: 'send-template-email', label: 'Send Template Email' }\n ],\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n }\n });\n }\n\n async onActionSendEmail(event, element) {\n const item = this.collection.get(element.dataset.id);\n // Implement send email action\n const data = await Dialog.showForm({\n title: 'Send Email',\n fields: [\n { name: 'to', label: 'To', type: 'email', required: true },\n { name: 'subject', label: 'Subject', type: 'text', required: true },\n { name: 'body_html', label: 'Body', type: 'textarea', required: true }\n ]\n });\n data.from_email = item.get('email');\n const result = await Mailbox.sendEmail(data);\n if (result.success) {\n this.getApp().toast.success('Email sent successfully');\n } else {\n let msg = \"Failed to send email\";\n if (result.data.details) {\n msg = result.data.details;\n } else if (result.data.error) {\n msg = result.data.error;\n }\n console.log(result);\n this.getApp().toast.error(msg);\n }\n }\n\n async onActionSendTemplateEmail(event, element) {\n const item = this.collection.get(element.dataset.id);\n // Implement send email action\n const data = await Dialog.showForm({\n title: 'Send Email',\n fields: [\n { name: 'to', label: 'To', type: 'email', required: true },\n { name: 'subject', label: 'Subject', type: 'text', required: true },\n { name: 'template_name', label: 'Template', type: 'text', required: true },\n { name: 'template_context', label: 'Context', type: 'textarea', required: true }\n ]\n });\n data.from_email = item.get('email');\n const result = await Mailbox.sendEmail(data);\n if (result.success) {\n this.getApp().toast.success('Email sent successfully');\n } else {\n let msg = \"Failed to send email\";\n if (result.data.details) {\n msg = result.data.details;\n } else if (result.data.error) {\n msg = result.data.error;\n }\n console.log(result);\n this.getApp().toast.error(msg);\n }\n }\n}\n\nexport default EmailMailboxTablePage;\n","/**\n * EmailTemplateView - A clean view for previewing email templates\n */\n\nimport View from '@core/View.js';\nimport TabView from '@core/views/navigation/TabView.js';\nimport { EmailTemplate } from '@core/models/Email.js';\n\n/**\n * EmailHtmlPreviewView - Renders HTML email template in a sandboxed iframe\n */\nclass EmailHtmlPreviewView extends View {\n constructor(options = {}) {\n super({\n className: 'email-html-preview',\n template: `\n <div class=\"email-html-preview-container\">\n <div class=\"d-flex justify-content-between align-items-center mb-2\">\n <small class=\"text-muted\">HTML Preview (sandboxed)</small>\n <button type=\"button\" class=\"btn btn-sm btn-outline-secondary\" data-action=\"refresh-preview\">\n <i class=\"bi bi-arrow-clockwise\"></i> Refresh\n </button>\n </div>\n <iframe\n id=\"email-preview-frame\"\n class=\"border rounded w-100\"\n style=\"height: 500px; background: white;\"\n sandbox=\"allow-same-origin\"\n frameborder=\"0\"\n ></iframe>\n </div>\n `,\n ...options\n });\n }\n\n async onAfterRender() {\n await super.onAfterRender();\n this.renderHtmlInIframe();\n }\n\n renderHtmlInIframe() {\n const iframe = this.element.querySelector('#email-preview-frame');\n if (!iframe) return;\n\n const htmlContent = this.model.get('html_template') || '';\n\n // Get iframe document\n const iframeDoc = iframe.contentDocument || iframe.contentWindow.document;\n\n // Write HTML content\n iframeDoc.open();\n iframeDoc.write(htmlContent);\n iframeDoc.close();\n }\n\n async onActionRefreshPreview(event, element) {\n this.renderHtmlInIframe();\n }\n}\n\nclass EmailTemplateView extends View {\n constructor(options = {}) {\n super({\n className: 'email-template-view',\n ...options\n });\n\n this.model = options.model || new EmailTemplate(options.data || {});\n this.hasHtml = !!this.model.get('html_template');\n this.hasText = !!this.model.get('text_template');\n this.hasMetadata = this.model.get('metadata') && Object.keys(this.model.get('metadata')).length > 0;\n\n this.template = `\n <div class=\"email-template-view-container p-3\">\n <!-- Header -->\n <div class=\"template-header border-bottom pb-3 mb-3\">\n <h4 class=\"mb-1\">{{model.name}}</h4>\n <div class=\"text-muted\">\n <strong>Subject:</strong> {{model.subject_template|default('Not set')}}\n </div>\n </div>\n\n <!-- Tabs -->\n <div data-container=\"template-tabs\"></div>\n </div>\n `;\n }\n\n async onInit() {\n const tabs = {};\n\n if (this.hasHtml) {\n tabs['HTML Preview'] = new EmailHtmlPreviewView({\n model: this.model\n });\n }\n\n if (this.hasText) {\n tabs['Text Version'] = new View({\n model: this.model,\n template: `<pre class=\"email-text-content p-3 bg-light border rounded\" style=\"white-space: pre-wrap; word-wrap: break-word;\">{{model.text_template}}</pre>`\n });\n }\n\n if (this.hasMetadata) {\n tabs['Metadata'] = new View({\n model: this.model,\n template: `<pre class=\"email-metadata-content p-3 bg-light border rounded\"><code>{{model.metadata|json}}</code></pre>`\n });\n }\n\n this.tabView = new TabView({\n containerId: 'template-tabs',\n tabs: tabs,\n activeTab: Object.keys(tabs)[0] || ''\n });\n this.addChild(this.tabView);\n }\n}\n\nexport default EmailTemplateView;\n","/**\n * EmailTemplateTablePage - Admin page for managing Email Templates\n * Clean implementation using TablePage with minimal overrides\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { EmailTemplateList, EmailTemplateForms } from '@core/models/Email.js';\nimport EmailTemplateView from './EmailTemplateView.js';\n\nclass EmailTemplateTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_email_templates',\n pageName: 'Email Templates',\n router: 'admin/email/templates',\n Collection: EmailTemplateList,\n\n formCreate: EmailTemplateForms.create,\n formEdit: EmailTemplateForms.edit,\n itemViewClass: EmailTemplateView,\n clickAction: \"edit\",\n\n viewDialogOptions: {\n header: false,\n size: 'xl',\n scrollable: true\n },\n\n formDialogConfig: { // Dialog options for forms\n size: 'fullscreen',\n },\n\n // Table columns\n columns: [\n { key: 'id', label: 'ID', width: '70px', sortable: true, class: 'text-muted' },\n { key: 'name', label: 'Name', sortable: true },\n { key: 'created', label: 'Created', formatter: 'datetime' },\n { key: 'modified', label: 'Modified', formatter: 'datetime' }\n ],\n\n // Features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Toolbar\n showRefresh: true,\n showAdd: true,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No email templates found. Click \"Add Template\" to create your first one.',\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n }\n });\n }\n}\n\nexport default EmailTemplateTablePage;\n","/**\n * EmailView - A slick, email-client-like view for sent messages\n */\n\nimport View from '@core/View.js';\nimport TabView from '@core/views/navigation/TabView.js';\nimport { SentMessage } from '@core/models/Email.js';\n\nclass EmailView extends View {\n constructor(options = {}) {\n super({\n className: 'email-view',\n ...options\n });\n\n this.model = options.model || new SentMessage(options.data || {});\n this.hasHtml = !!this.model.get('body_html');\n this.hasText = !!this.model.get('body_text');\n this.hasContext = this.model.get('template_context') && Object.keys(this.model.get('template_context')).length > 0;\n\n this.template = `\n <div class=\"email-view-container p-3\">\n <!-- Email Header -->\n <div class=\"email-header border-bottom pb-3 mb-3\">\n <h4 class=\"mb-2\">{{model.subject}}</h4>\n <div class=\"d-flex justify-content-between align-items-center\">\n <div class=\"d-flex align-items-center\">\n <div class=\"flex-shrink-0\">\n <i class=\"bi bi-person-circle fs-2 text-secondary\"></i>\n </div>\n <div class=\"ms-3\">\n <div class=\"fw-bold\">{{model.mailbox.email}}</div>\n <div class=\"text-muted small\">To: {{model.to_addresses|list}}</div>\n </div>\n </div>\n <div class=\"text-end\">\n <div class=\"text-muted small\">{{model.created|datetime}}</div>\n <span class=\"badge {{model.status|badge}} mt-1\">{{model.status|capitalize}}</span>\n </div>\n </div>\n </div>\n\n <!-- Email Body Tabs -->\n <div data-container=\"email-tabs\"></div>\n </div>\n `;\n }\n\n async onInit() {\n const tabs = {};\n\n if (this.hasHtml) {\n tabs['HTML'] = new View({\n model: this.model,\n template: `<div class=\"email-html-content border rounded p-3\" style=\"height: 500px; overflow-y: auto;\">{{{model.body_html}}}</div>`\n });\n }\n\n if (this.hasText) {\n tabs['Text'] = new View({\n model: this.model,\n template: `<pre class=\"email-text-content p-3 bg-light border rounded\" style=\"white-space: pre-wrap; word-wrap: break-word;\">{{model.body_text}}</pre>`\n });\n }\n\n if (this.hasContext) {\n tabs['Context'] = new View({\n model: this.model,\n template: `<pre class=\"email-context-content p-3 bg-light border rounded\"><code>{{model.template_context|json}}</code></pre>`\n });\n }\n\n this.tabView = new TabView({\n containerId: 'email-tabs',\n tabs: tabs,\n activeTab: this.hasHtml ? 'HTML' : (this.hasText ? 'Text' : 'Context')\n });\n this.addChild(this.tabView);\n }\n}\n\nSentMessage.VIEW_CLASS = EmailView;\n\n\nexport default EmailView;\n","/**\n * SentMessageTablePage - Admin page for viewing sent emails\n * Clean implementation using TablePage with minimal overrides\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { SentMessageList } from '@core/models/Email.js';\nimport EmailView from './EmailView.js';\n\nclass SentMessageTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_email_sent',\n pageName: 'Sent Messages',\n router: 'admin/email/sent',\n Collection: SentMessageList,\n \n itemViewClass: EmailView,\n viewDialogOptions: {\n header: false,\n size: 'xl',\n scrollable: true\n },\n\n // Table columns\n columns: [\n { key: 'id', label: 'ID', width: '70px', sortable: true, class: 'text-muted' },\n { key: 'mailbox.email', label: 'From', sortable: true },\n { key: 'to_addresses', label: 'To', sortable: false, formatter: \"list\" },\n { key: 'subject', label: 'Subject', sortable: true },\n { key: 'status', label: 'Status', formatter: 'badge' },\n { key: 'status_reason', label: 'Reason', formatter: \"truncate(80)|default('—')\" },\n { key: 'created', label: 'Created', formatter: 'datetime' }\n ],\n\n // Features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Toolbar\n showRefresh: true,\n showAdd: false,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No sent messages found.',\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n }\n });\n }\n}\n\nexport default SentMessageTablePage;","/**\n * Phonehub.js\n * Models and Collections for PhoneHub phone numbers and SMS with helper actions.\n *\n * Endpoints (all require authentication unless noted):\n * - Phone Number\n * - POST /api/phonehub/number/normalize\n * - POST /api/phonehub/number/lookup\n * - GET /api/phonehub/number\n * - GET /api/phonehub/number/:id\n * - PUT /api/phonehub/number/:id\n * - DELETE /api/phonehub/number/:id\n *\n * - SMS\n * - POST /api/phonehub/sms/send\n * - GET /api/phonehub/sms\n * - GET /api/phonehub/sms/:id\n *\n * Notes on response shapes:\n * - List endpoints typically return { status: true, data: [...], count, size, start }\n * - Single item endpoints typically return { status: true, data: {...} }\n * - Action endpoints (normalize, lookup, send) return { status: true, data: {...} } or { status: false, error }\n */\n\nimport Model from '@core/Model.js';\nimport Collection from '@core/Collection.js';\nimport rest from '@core/Rest.js';\n\n// ======================================================\n// PhoneNumber Model & Collection\n// ======================================================\n\n/**\n * PhoneNumber - Represents a cached phone number lookup result.\n */\nclass PhoneNumber extends Model {\n constructor(data = {}, options = {}) {\n super(data, {\n endpoint: '/api/phonehub/number',\n ...options\n });\n }\n\n /**\n * Normalize an arbitrary phone_number string to E.164 format.\n * @param {string} phoneNumber - Raw phone number input.\n * @param {string} [countryCode='US'] - Optional country code (default US).\n * @returns {Promise<{success: boolean, phone_number?: string, data?: object, error?: string, response?: any}>}\n */\n static async normalize(phoneNumber, countryCode = 'US') {\n const url = '/api/phonehub/number/normalize';\n const payload = {\n phone_number: phoneNumber\n };\n if (countryCode) payload.country_code = countryCode;\n\n const resp = await rest.POST(url, payload);\n const body = resp?.data ?? resp; // rest wrapper or raw\n const ok = body?.status === true || body?.success === true;\n\n if (ok) {\n const normalized = body?.data?.phone_number ?? body?.phone_number;\n return { success: true, phone_number: normalized, data: body?.data ?? body, response: resp };\n }\n return { success: false, error: body?.error || 'Normalization failed', response: resp };\n }\n\n /**\n * Lookup phone number details (carrier, line_type, owner, etc.)\n * @param {string} phoneNumber - E.164 or raw phone number.\n * @param {object} [options] - { force_refresh?: boolean, group?: number }\n * @returns {Promise<{success: boolean, model?: PhoneNumber, data?: object, error?: string, response?: any}>}\n */\n static async lookup(phoneNumber, options = {}) {\n const url = '/api/phonehub/number/lookup';\n const resp = await rest.POST(url, {\n phone_number: phoneNumber,\n ...options\n });\n const body = resp?.data ?? resp;\n const ok = body?.status === true || body?.success === true;\n\n if (ok) {\n const data = body?.data ?? {};\n const model = new PhoneNumber(data, { endpoint: '/api/phonehub/number' });\n return { success: true, model, data, response: resp };\n }\n return { success: false, error: body?.error || 'Phone lookup failed', response: resp };\n }\n\n\n}\n\n/**\n * PhoneNumberList - Paginated collection of PhoneNumber models.\n */\nclass PhoneNumberList extends Collection {\n constructor(options = {}) {\n super({\n ModelClass: PhoneNumber,\n endpoint: '/api/phonehub/number',\n size: 10,\n ...options\n });\n }\n\n\n}\n\n// ======================================================\n// SMS Model & Collection\n// ======================================================\n\n/**\n * SMS - Represents an SMS message (sent or received).\n */\nclass SMS extends Model {\n constructor(data = {}, options = {}) {\n super(data, {\n endpoint: '/api/phonehub/sms',\n ...options\n });\n }\n\n /**\n * Send SMS via PhoneHub (Twilio under the hood).\n * @param {object} params - { to_number, body, from_number?, group?, metadata? }\n * @returns {Promise<{success: boolean, model?: SMS, data?: object, error?: string, response?: any}>}\n */\n static async send(params = {}) {\n const url = '/api/phonehub/sms/send';\n const resp = await rest.POST(url, params);\n const body = resp?.data ?? resp;\n const ok = body?.status === true || body?.success === true;\n\n if (ok) {\n const data = body?.data ?? {};\n const model = new SMS(data, { endpoint: '/api/phonehub/sms' });\n return { success: true, model, data, response: resp };\n }\n return { success: false, error: body?.error || 'Failed to send SMS', response: resp };\n }\n\n\n\n\n}\n\n/**\n * SMSList - Paginated collection of SMS models.\n */\nclass SMSList extends Collection {\n constructor(options = {}) {\n super({\n ModelClass: SMS,\n endpoint: '/api/phonehub/sms',\n size: 10,\n ...options\n });\n }\n\n\n}\n\n// Exported API\nexport {\n PhoneNumber,\n PhoneNumberList,\n SMS,\n SMSList\n};\n\nexport default {\n PhoneNumber,\n PhoneNumberList,\n SMS,\n SMSList\n};","/**\n * PhoneNumberView - Detailed view for a PhoneHub PhoneNumber record\n *\n * Mirrors GeoIPView structure:\n * - Header with primary identifier and quick info\n * - Tabbed content using TabView and DataView sections\n * - Minimal context menu for refresh and delete\n */\n\nimport View from '@core/View.js';\nimport TabView from '@core/views/navigation/TabView.js';\nimport DataView from '@core/views/data/DataView.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\nimport { PhoneNumber } from '@core/models/Phonehub.js';\n\nclass PhoneNumberView extends View {\n constructor(options = {}) {\n super({\n className: 'phone-number-view',\n ...options\n });\n\n this.model = options.model || new PhoneNumber(options.data || {});\n\n this.template = `\n <div class=\"phone-number-view-container\">\n <!-- Header -->\n <div class=\"d-flex justify-content-between align-items-center mb-4\">\n <!-- Left Side: Icon & Info -->\n <div class=\"d-flex align-items-center gap-3\">\n <div class=\"fs-1 text-primary\">\n <i class=\"bi bi-telephone\"></i>\n </div>\n <div>\n <h3 class=\"mb-1\">{{model.phone_number|default('Unknown Number')}}</h3>\n <div class=\"text-muted small\">\n {{model.carrier|default('—')}} {{#model.line_type}}· {{model.line_type|capitalize}}{{/model.line_type}}\n </div>\n <div class=\"text-muted small mt-1\">\n {{#model.country_code}}Country: {{model.country_code}}{{/model.country_code}}\n {{#model.region}} · Region: {{model.region}}{{/model.region}}\n {{#model.state}} · State: {{model.state}}{{/model.state}}\n </div>\n </div>\n </div>\n\n <!-- Right Side: Actions -->\n <div class=\"d-flex align-items-center gap-4\">\n <div data-container=\"phone-context-menu\"></div>\n </div>\n </div>\n\n <!-- Tabs -->\n <div data-container=\"phone-tabs\"></div>\n </div>\n `;\n }\n\n async onInit() {\n // Overview Tab\n this.overviewView = new DataView({\n model: this.model,\n className: 'p-3',\n showEmptyValues: true,\n emptyValueText: '—',\n columns: 2,\n fields: [\n { name: 'phone_number', label: 'Phone Number', cols: 6 },\n { name: 'country_code', label: 'Country Code', cols: 6 },\n { name: 'region', label: 'Region', cols: 6 },\n { name: 'state', label: 'State', cols: 6 },\n { name: 'registered_owner', label: 'Registered Owner', cols: 6 },\n { name: 'owner_type', label: 'Owner Type', formatter: 'capitalize', cols: 6 },\n { name: 'is_valid', label: 'Valid', formatter: 'yesnoicon', cols: 4 },\n { name: 'is_mobile', label: 'Mobile', formatter: 'yesnoicon', cols: 4 },\n { name: 'is_voip', label: 'VOIP', formatter: 'yesnoicon', cols: 4 },\n ]\n });\n\n // Carrier & Status Tab\n this.carrierView = new DataView({\n model: this.model,\n className: 'p-3',\n showEmptyValues: true,\n emptyValueText: '—',\n columns: 2,\n fields: [\n { name: 'carrier', label: 'Carrier', cols: 6 },\n { name: 'line_type', label: 'Line Type', formatter: 'capitalize', cols: 6 },\n { name: 'lookup_provider', label: 'Lookup Provider', formatter: 'capitalize', cols: 6 },\n { name: 'lookup_count', label: 'Lookup Count', cols: 6 },\n { name: 'last_lookup_at', label: 'Last Lookup', formatter: 'datetime', cols: 6 },\n { name: 'lookup_expires_at', label: 'Cache Expires', formatter: 'datetime', cols: 6 }\n ]\n });\n\n // Address Tab\n this.addressView = new DataView({\n model: this.model,\n className: 'p-3',\n showEmptyValues: true,\n emptyValueText: '—',\n columns: 2,\n fields: [\n { name: 'address_line1', label: 'Address Line 1', cols: 12 },\n { name: 'address_city', label: 'City', cols: 4 },\n { name: 'address_state', label: 'State', cols: 4 },\n { name: 'address_zip', label: 'ZIP', cols: 4 },\n { name: 'address_country', label: 'Country', cols: 6 }\n ]\n });\n\n // Metadata Tab\n this.metadataView = new DataView({\n model: this.model,\n className: 'p-3',\n showEmptyValues: true,\n emptyValueText: '—',\n columns: 2,\n fields: [\n { name: 'id', label: 'Record ID', cols: 6 },\n { name: 'created', label: 'Created', formatter: 'datetime', cols: 6 },\n { name: 'modified', label: 'Last Modified', formatter: 'datetime', cols: 6 }\n ]\n });\n\n const tabs = {\n 'Overview': this.overviewView,\n 'Carrier': this.carrierView,\n 'Address': this.addressView,\n 'Metadata': this.metadataView\n };\n\n this.tabView = new TabView({\n containerId: 'phone-tabs',\n tabs,\n activeTab: 'Overview'\n });\n this.addChild(this.tabView);\n\n // Minimal ContextMenu\n const menuItems = [\n { label: 'Refresh Lookup', action: 'refresh-lookup', icon: 'bi-arrow-repeat' },\n { type: 'divider' },\n { label: 'Delete Record', action: 'delete-phone', icon: 'bi-trash', danger: true }\n ];\n\n const ctxMenu = new ContextMenu({\n containerId: 'phone-context-menu',\n className: 'context-menu-view header-menu-absolute',\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: menuItems\n }\n });\n this.addChild(ctxMenu);\n }\n\n // Actions\n\n async onActionRefreshLookup() {\n const number = this.model.get('phone_number');\n if (!number) {\n this.getApp()?.toast?.warning?.('No phone number to lookup');\n return;\n }\n\n try {\n this.getApp()?.toast?.info?.('Refreshing lookup...');\n // Force refresh lookup and update current model\n const resp = await PhoneNumber.lookup(number, { force_refresh: true });\n if (resp.success && resp.data) {\n this.model.set(resp.data);\n await this.render();\n this.getApp()?.toast?.success?.('Lookup refreshed');\n } else {\n const msg = resp.error || 'Lookup failed';\n this.getApp()?.toast?.error?.(msg);\n }\n } catch (e) {\n this.getApp()?.toast?.error?.(e.message || 'Lookup failed');\n }\n }\n\n async onActionDeletePhone() {\n const confirmed = await Dialog.confirm(\n `Are you sure you want to delete the record for \"${this.model.get('phone_number') || 'this number'}\"?`,\n 'Confirm Deletion',\n { confirmClass: 'btn-danger', confirmText: 'Delete' }\n );\n\n if (!confirmed) return;\n\n try {\n const resp = await this.model.destroy();\n if (resp?.success) {\n this.emit('phone:deleted', { model: this.model });\n } else {\n this.getApp()?.toast?.error?.('Delete failed');\n }\n } catch (e) {\n this.getApp()?.toast?.error?.(e.message || 'Delete failed');\n }\n }\n\n static async show(phone_number) {\n const resp = await PhoneNumber.lookup(phone_number);\n if (resp?.model) {\n const view = new PhoneNumberView({ model: resp.model });\n const dialog = new Dialog({\n header: false,\n size: 'lg',\n body: view,\n buttons: [{ text: 'Close', class: 'btn-secondary', dismiss: true }]\n });\n await dialog.render(true, document.body);\n dialog.show();\n return dialog;\n }\n Dialog.alert({ message: `Could not find phone data for number: ${phone_number}`, type: 'warning' });\n return null;\n }\n}\n\nPhoneNumberView.MODEL_CLASS = PhoneNumber;\n\nexport default PhoneNumberView;\n","/**\n * PhoneNumberTablePage - PhoneHub numbers management using TablePage component\n * Mirrors the GeoLocatedIPTablePage pattern with minimal, consistent configuration.\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { PhoneNumberList, PhoneNumber } from '@core/models/Phonehub.js';\nimport PhoneNumberView from './PhoneNumberView.js';\n\nclass PhoneNumberTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n\n // Identity\n name: 'admin_phonehub_numbers',\n pageName: 'Phone Numbers',\n router: 'admin/phonehub/numbers',\n\n // Data source\n Collection: PhoneNumberList,\n\n // Item view configuration\n itemView: PhoneNumberView,\n viewDialogOptions: {\n header: false,\n // size: 'xl'\n },\n\n // Column definitions\n columns: [\n { key: 'phone_number', label: 'Phone Number', sortable: true },\n { key: 'carrier', label: 'Carrier', sortable: true, formatter: \"default('—')\" },\n { key: 'line_type', label: 'Line Type', sortable: true, formatter: \"capitalize\" },\n { key: 'is_mobile', label: 'Mobile', formatter: 'yesnoicon' },\n { key: 'is_voip', label: 'VOIP', formatter: 'yesnoicon' },\n { key: 'is_valid', label: 'Valid', formatter: 'yesnoicon' },\n { key: 'registered_owner', label: 'Owner', sortable: true, formatter: \"default('—')\" },\n { key: 'owner_type', label: 'Owner Type', formatter: \"capitalize\" },\n { key: 'last_lookup_at|relative', label: 'Last Lookup', sortable: true},\n ],\n\n // Table features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Row action\n clickAction: 'view',\n\n // Toolbar\n showRefresh: true,\n showAdd: true,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No phone numbers found.',\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n },\n\n tableViewOptions: {\n addButtonLabel: \"Lookup\",\n addButtonIcon: 'bi-search',\n onAdd: (evt) => {\n evt.preventDefault();\n // Implement the logic for adding a new record\n this.onLookup();\n }\n }\n });\n }\n\n async onLookup() {\n // Implement the logic for adding a new record\n const data = await this.getApp().showForm({\n title: \"Lookup Phone Number\",\n fields: [\n {\n name: 'number',\n type: 'text',\n required: true\n }\n ]\n });\n if (data && data.number) {\n const resp = await PhoneNumber.lookup(data.number);\n if (resp.model) {\n this.tableView._onRowView(resp);\n }\n }\n }\n}\n\nexport default PhoneNumberTablePage;\n","/**\n * SMSView - Detailed view for a PhoneHub SMS record\n *\n * Mirrors GeoIPView structure:\n * - Header with key info (direction, from/to, status)\n * - Tabbed content using TabView and DataView sections\n * - Minimal context menu for refresh and delete\n */\n\nimport View from '@core/View.js';\nimport TabView from '@core/views/navigation/TabView.js';\nimport DataView from '@core/views/data/DataView.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\nimport { SMS } from '@core/models/Phonehub.js';\n\nclass SMSView extends View {\n constructor(options = {}) {\n super({\n className: 'sms-view',\n ...options\n });\n\n this.model = options.model || new SMS(options.data || {});\n\n this.template = `\n <div class=\"sms-view-container\">\n <!-- Header -->\n <div class=\"d-flex justify-content-between align-items-center mb-4\">\n <!-- Left Side: Icon & Info -->\n <div class=\"d-flex align-items-center gap-3\">\n <div class=\"fs-1 text-primary\">\n <i class=\"bi bi-chat-dots\"></i>\n </div>\n <div>\n <h3 class=\"mb-1\">\n {{#model.direction}}{{model.direction|capitalize}}{{/model.direction}}\n {{^model.direction}}Message{{/model.direction}}\n <small class=\"text-muted ms-2\">\n {{#model.status}}[{{model.status|capitalize}}]{{/model.status}}\n </small>\n </h3>\n <div class=\"text-muted small\">\n {{#model.from_number}}From: {{model.from_number}}{{/model.from_number}}\n {{#model.to_number}} · To: {{model.to_number}}{{/model.to_number}}\n </div>\n <div class=\"text-muted small mt-1\">\n {{#model.provider}}Provider: {{model.provider|capitalize}}{{/model.provider}}\n {{#model.provider_message_id}} · SID: {{model.provider_message_id}}{{/model.provider_message_id}}\n </div>\n </div>\n </div>\n\n <!-- Right Side: Actions -->\n <div class=\"d-flex align-items-center gap-4\">\n <div data-container=\"sms-context-menu\"></div>\n </div>\n </div>\n\n <!-- Tabs -->\n <div data-container=\"sms-tabs\"></div>\n </div>\n `;\n }\n\n async onInit() {\n // Message Tab\n this.messageView = new DataView({\n model: this.model,\n className: 'p-3',\n showEmptyValues: true,\n emptyValueText: '—',\n columns: 2,\n fields: [\n { name: 'direction', label: 'Direction', formatter: 'capitalize', cols: 4 },\n { name: 'status', label: 'Status', formatter: 'capitalize', cols: 4 },\n { name: 'from_number', label: 'From', cols: 6 },\n { name: 'to_number', label: 'To', cols: 6 },\n { name: 'body', label: 'Message Body', cols: 12 }\n ]\n });\n\n // Delivery Tab\n this.deliveryView = new DataView({\n model: this.model,\n className: 'p-3',\n showEmptyValues: true,\n emptyValueText: '—',\n columns: 2,\n fields: [\n { name: 'provider', label: 'Provider', formatter: 'capitalize', cols: 6 },\n { name: 'provider_message_id', label: 'Provider Message ID', cols: 6 },\n { name: 'sent_at', label: 'Sent At', formatter: 'datetime', cols: 6 },\n { name: 'delivered_at', label: 'Delivered At', formatter: 'datetime', cols: 6 },\n { name: 'error_code', label: 'Error Code', cols: 6 },\n { name: 'error_message', label: 'Error Message', cols: 12 }\n ]\n });\n\n // Metadata Tab\n this.metadataView = new DataView({\n model: this.model,\n className: 'p-3',\n showEmptyValues: true,\n emptyValueText: '—',\n columns: 2,\n fields: [\n { name: 'id', label: 'Record ID', cols: 6 },\n { name: 'created', label: 'Created', formatter: 'datetime', cols: 6 },\n { name: 'modified', label: 'Last Modified', formatter: 'datetime', cols: 6 }\n ]\n });\n\n const tabs = {\n 'Message': this.messageView,\n 'Delivery': this.deliveryView,\n 'Metadata': this.metadataView\n };\n\n this.tabView = new TabView({\n containerId: 'sms-tabs',\n tabs,\n activeTab: 'Message'\n });\n this.addChild(this.tabView);\n\n // ContextMenu (minimal)\n const menuItems = [\n { label: 'Refresh', action: 'refresh-sms', icon: 'bi-arrow-repeat' },\n { type: 'divider' },\n { label: 'Delete Message', action: 'delete-sms', icon: 'bi-trash', danger: true }\n ];\n\n const ctxMenu = new ContextMenu({\n containerId: 'sms-context-menu',\n className: 'context-menu-view header-menu-absolute',\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: menuItems\n }\n });\n this.addChild(ctxMenu);\n }\n\n // Actions\n\n async onActionRefreshSms() {\n try {\n this.getApp()?.toast?.info?.('Refreshing message...');\n await this.model.fetch();\n await this.render();\n this.getApp()?.toast?.success?.('Message refreshed');\n } catch (e) {\n this.getApp()?.toast?.error?.(e.message || 'Refresh failed');\n }\n }\n\n async onActionDeleteSms() {\n const title = 'Confirm Deletion';\n const msg = `Are you sure you want to delete this message?`;\n const confirmed = await Dialog.confirm(msg, title, {\n confirmClass: 'btn-danger',\n confirmText: 'Delete'\n });\n\n if (!confirmed) return;\n\n try {\n const resp = await this.model.destroy();\n if (resp?.success) {\n this.emit('sms:deleted', { model: this.model });\n } else {\n this.getApp()?.toast?.error?.('Delete failed');\n }\n } catch (e) {\n this.getApp()?.toast?.error?.(e.message || 'Delete failed');\n }\n }\n}\n\nSMSView.MODEL_CLASS = SMS;\n\nexport default SMSView;","/**\n * SMSTablePage - PhoneHub SMS management using TablePage component\n * Mirrors the GeoLocatedIPTablePage pattern with minimal, consistent configuration.\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { SMSList } from '@core/models/Phonehub.js';\nimport SMSView from './SMSView.js';\n\nclass SMSTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n\n // Identity\n name: 'admin_phonehub_sms',\n pageName: 'SMS Messages',\n router: 'admin/phonehub/sms',\n\n // Data source\n Collection: SMSList,\n\n // Item view configuration\n itemView: SMSView,\n viewDialogOptions: {\n header: false,\n size: 'xl'\n },\n\n // Column definitions\n columns: [\n { key: 'direction', label: 'Direction', sortable: true },\n { key: 'from_number', label: 'From', sortable: true, formatter: \"default('—')\" },\n { key: 'to_number', label: 'To', sortable: true, formatter: \"default('—')\" },\n { key: 'status', label: 'Status', sortable: true },\n { key: 'provider', label: 'Provider', sortable: true, formatter: \"default('—')\" },\n { key: 'body', label: 'Message', formatter: \"default('—')\" },\n { key: 'sent_at', label: 'Sent At', sortable: true, formatter: 'datetime' },\n { key: 'delivered_at', label: 'Delivered At', sortable: true, formatter: 'datetime' },\n { key: 'created', label: 'Created', sortable: true, formatter: 'datetime' }\n ],\n\n // Table features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Row action\n clickAction: 'view',\n\n // Toolbar\n showRefresh: true,\n showAdd: false,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No SMS messages found.',\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n }\n });\n }\n}\n\nexport default SMSTablePage;","import Page from '@core/Page.js';\nimport { MetricsChart, PieChart } from '@ext/charts/index.js';\nimport TableView from '@core/views/table/TableView.js';\nimport { PushDeliveryList } from '@core/models/Push.js';\n\nclass PushDashboardPage extends Page {\n constructor(options = {}) {\n super({\n ...options,\n title: 'Push Notifications Dashboard',\n className: 'push-dashboard-page'\n });\n }\n\n async getTemplate() {\n return `\n <div class=\"container-fluid\">\n <h1 class=\"h3 mb-4\">Push Notifications</h1>\n <div class=\"row\">\n <!-- Stat cards -->\n </div>\n <div class=\"row\">\n <div class=\"col-xl-8 col-lg-7\">\n <div class=\"card shadow mb-4\">\n <div class=\"card-header\">Notifications Over Time</div>\n <div class=\"card-body\" data-container=\"deliveries-chart\"></div>\n </div>\n </div>\n <div class=\"col-xl-4 col-lg-5\">\n <div class=\"card shadow mb-4\">\n <div class=\"card-header\">Delivery Status</div>\n <div class=\"card-body\" data-container=\"status-chart\"></div>\n </div>\n </div>\n </div>\n <div class=\"row\">\n <div class=\"col-lg-6 mb-4\" data-container=\"recent-deliveries\"></div>\n <div class=\"col-lg-6 mb-4\" data-container=\"failed-deliveries\"></div>\n </div>\n </div>\n `;\n }\n\n async onInit() {\n this.deliveriesChart = new MetricsChart({\n containerId: 'deliveries-chart',\n endpoint: '/api/metrics/fetch',\n slugs: ['push_sent', 'push_failed'],\n chartType: 'line'\n });\n this.addChild(this.deliveriesChart);\n\n this.statusChart = new PieChart({\n containerId: 'status-chart',\n endpoint: '/api/account/devices/push/stats' // Assuming this returns data for pie chart\n });\n this.addChild(this.statusChart);\n\n this.recentDeliveries = new TableView({\n containerId: 'recent-deliveries',\n title: 'Recent Deliveries',\n Collection: new PushDeliveryList({ params: { _sort: '-created', _limit: 5 } }),\n columns: [{ key: 'title', label: 'Title' }, { key: 'status', label: 'Status', formatter: 'badge' }]\n });\n this.addChild(this.recentDeliveries);\n\n this.failedDeliveries = new TableView({\n containerId: 'failed-deliveries',\n title: 'Failed Deliveries',\n Collection: new PushDeliveryList({ params: { status: 'failed', _sort: '-created', _limit: 5 } }),\n columns: [{ key: 'title', label: 'Title' }, { key: 'error_message', label: 'Error' }]\n });\n this.addChild(this.failedDeliveries);\n }\n}\n\nexport default PushDashboardPage;\n","import TablePage from '@core/pages/TablePage.js';\nimport { PushConfigList, PushConfigForms } from '@core/models/Push.js';\n\nclass PushConfigTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_push_configs',\n pageName: 'Push Configurations',\n router: \"admin/push/configs\",\n Collection: PushConfigList,\n formCreate: PushConfigForms.create,\n formEdit: PushConfigForms.edit,\n\n columns: [\n { key: 'id', label: 'ID', width: '70px' },\n { key: 'name', label: 'Name' },\n { key: 'group.name', label: 'Group', formatter: \"default('Default')\" },\n { key: 'fcm_sender_id', label: 'Project ID' },\n { key: 'is_active', label: 'Active', format: 'boolean' },\n ],\n\n searchable: true,\n sortable: true,\n paginated: true,\n showRefresh: true,\n showAdd: true,\n showExport: true,\n actions: [\"edit\", \"delete\"],\n tableOptions: {\n pageSizes: [10, 25, 50],\n defaultPageSize: 25,\n emptyMessage: 'No push configurations found.',\n emptyIcon: 'bi-gear',\n\n }\n });\n }\n}\n\nexport default PushConfigTablePage;\n","import TablePage from '@core/pages/TablePage.js';\nimport { PushTemplateList, PushTemplateForms } from '@core/models/Push.js';\n\nclass PushTemplateTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_push_templates',\n pageName: 'Push Templates',\n router: \"admin/push/templates\",\n Collection: PushTemplateList,\n formCreate: PushTemplateForms.create,\n formEdit: PushTemplateForms.edit,\n\n columns: [\n { key: 'id', label: 'ID', width: '70px' },\n { key: 'name', label: 'Name' },\n { key: 'category', label: 'Category' },\n { key: 'group.name', label: 'Group', formatter: \"default('Default')\" },\n { key: 'priority', label: 'Priority' },\n { key: 'is_active', label: 'Active', format: 'boolean' },\n ],\n\n searchable: true,\n sortable: true,\n paginated: true,\n showRefresh: true,\n showAdd: true,\n showExport: true,\n\n tableOptions: {\n pageSizes: [10, 25, 50],\n defaultPageSize: 25,\n emptyMessage: 'No push templates found.',\n emptyIcon: 'bi-file-earmark-text',\n actions: [\"edit\", \"delete\"],\n }\n });\n }\n}\n\nexport default PushTemplateTablePage;\n","import View from '@core/View.js';\n\nclass PushDeliveryView extends View {\n constructor(options = {}) {\n super({\n className: 'push-delivery-view',\n ...options\n });\n this.model = options.model;\n }\n\n getTemplate() {\n return `\n <div class=\"p-3\">\n <div class=\"phone-mockup\">\n <div class=\"phone-screen\">\n <div class=\"notification\">\n <div class=\"notification-header\">\n <i class=\"bi bi-app-indicator\"></i>\n <strong>Your App</strong>\n <span class=\"ms-auto small text-muted\">now</span>\n </div>\n <div class=\"notification-body\">\n <div class=\"fw-bold\">{{model.title}}</div>\n <div>{{model.body}}</div>\n </div>\n </div>\n </div>\n </div>\n <div class=\"mt-3\">\n <h5>Delivery Details</h5>\n <p><strong>Status:</strong> <span class=\"badge {{model.status|badge}}\">{{model.status}}</span></p>\n <p><strong>Error:</strong> {{model.error_message|default('None')}}</p>\n </div>\n </div>\n `;\n }\n}\n\nexport default PushDeliveryView;\n","import TablePage from '@core/pages/TablePage.js';\nimport { PushDeliveryList } from '@core/models/Push.js';\nimport PushDeliveryView from './PushDeliveryView.js';\n\nclass PushDeliveryTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_push_deliveries',\n pageName: 'Push Deliveries',\n router: \"admin/push/deliveries\",\n Collection: PushDeliveryList,\n itemViewClass: PushDeliveryView,\n viewDialogOptions: {\n header: false,\n size: 'md'\n },\n\n columns: [\n { key: 'id', label: 'ID', width: '70px' },\n { key: 'created', label: 'Timestamp', formatter: 'datetime' },\n { key: 'user.display_name', label: 'User' },\n { key: 'device.device_name', label: 'Device' },\n { key: 'title', label: 'Title' },\n { key: 'category', label: 'Category' },\n { key: 'status', label: 'Status', formatter: 'badge' },\n ],\n \n searchable: true,\n sortable: true,\n paginated: true,\n showRefresh: true,\n\n tableOptions: {\n pageSizes: [10, 25, 50],\n defaultPageSize: 25,\n emptyMessage: 'No deliveries found.',\n emptyIcon: 'bi-send',\n actions: [\"view\"],\n }\n });\n }\n}\n\nexport default PushDeliveryTablePage;\n","import View from '@core/View.js';\nimport DataView from '@core/views/data/DataView.js';\n\nclass PushDeviceView extends View {\n constructor(options = {}) {\n super({\n className: 'push-device-view',\n ...options\n });\n this.model = options.model;\n }\n\n getTemplate() {\n return `\n <div class=\"p-3\">\n <h3>{{model.device_name}}</h3>\n <p class=\"text-muted\">{{model.user.display_name}}</p>\n <div data-container=\"data-view\"></div>\n </div>\n `;\n }\n\n onInit() {\n this.dataView = new DataView({\n containerId: 'data-view',\n model: this.model,\n fields: [\n { name: 'platform', label: 'Platform', format: 'badge' },\n { name: 'push_enabled', label: 'Push Enabled', format: 'boolean' },\n { name: 'app_version', label: 'App Version' },\n { name: 'os_version', label: 'OS Version' },\n { name: 'last_seen', label: 'Last Seen', format: 'datetime' },\n { name: 'push_preferences', label: 'Preferences', format: 'json' },\n ]\n });\n this.addChild(this.dataView);\n }\n}\n\nexport default PushDeviceView;\n","import TablePage from '@core/pages/TablePage.js';\nimport { PushDeviceList } from '@core/models/Push.js';\nimport PushDeviceView from './PushDeviceView.js';\n\nclass PushDeviceTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_push_devices',\n pageName: 'Registered Devices',\n router: \"admin/push/devices\",\n Collection: PushDeviceList,\n itemViewClass: PushDeviceView,\n viewDialogOptions: {\n header: false,\n size: 'lg'\n },\n\n columns: [\n { key: 'id', label: 'ID', width: '70px' },\n { key: 'user.display_name', label: 'User' },\n { key: 'device_name', label: 'Device Name' },\n { key: 'platform', label: 'Platform', formatter: 'badge' },\n { key: 'app_version', label: 'App Version' },\n { key: 'push_enabled', label: 'Push Enabled', format: 'boolean' },\n { key: 'last_seen', label: 'Last Seen', formatter: 'datetime' },\n ],\n \n searchable: true,\n sortable: true,\n paginated: true,\n showRefresh: true,\n\n tableOptions: {\n pageSizes: [10, 25, 50],\n defaultPageSize: 25,\n emptyMessage: 'No devices found.',\n emptyIcon: 'bi-phone',\n actions: [\"view\", \"delete\"],\n }\n });\n }\n}\n\nexport default PushDeviceTablePage;\n","/**\n * JobStatsView - Job statistics overview cards\n */\n\nimport View from '@core/View.js';\nimport { Job } from '@core/models/Job.js';\n\nexport default class JobStatsView extends View {\n constructor(options = {}) {\n super({\n className: 'job-stats-section',\n ...options\n });\n\n this.stats = {\n pending: 0,\n running: 0,\n completed: 0,\n failed: 0,\n scheduled: 0\n };\n\n this.template = `\n <div class=\"job-stats-header mb-4\">\n <div class=\"row\">\n <div class=\"col-xl-2 col-lg-4 col-md-6 col-12 mb-3\">\n <div class=\"card h-100 border-0 shadow\">\n <div class=\"card-body\">\n <div class=\"d-flex justify-content-between align-items-start\">\n <div>\n <h6 class=\"card-title text-muted mb-2\">Pending</h6>\n <h3 class=\"mb-1 fw-bold\">{{stats.pending}}</h3>\n <span class=\"badge bg-primary-subtle text-primary\">\n <i class=\"bi bi-hourglass\"></i> Queued\n </span>\n </div>\n <div class=\"text-primary\">\n <i class=\"bi bi-clock fs-2\"></i>\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <div class=\"col-xl-2 col-lg-4 col-md-6 col-12 mb-3\">\n <div class=\"card h-100 border-0 shadow\">\n <div class=\"card-body\">\n <div class=\"d-flex justify-content-between align-items-start\">\n <div>\n <h6 class=\"card-title text-muted mb-2\">Running</h6>\n <h3 class=\"mb-1 fw-bold\">{{stats.running}}</h3>\n <span class=\"badge bg-success-subtle text-success\">\n <i class=\"bi bi-arrow-repeat\"></i> Active\n </span>\n </div>\n <div class=\"text-success\">\n <i class=\"bi bi-play-circle fs-2\"></i>\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <div class=\"col-xl-2 col-lg-4 col-md-6 col-12 mb-3\">\n <div class=\"card h-100 border-0 shadow\">\n <div class=\"card-body\">\n <div class=\"d-flex justify-content-between align-items-start\">\n <div>\n <h6 class=\"card-title text-muted mb-2\">Scheduled</h6>\n <h3 class=\"mb-1 fw-bold\">{{stats.scheduled}}</h3>\n <span class=\"badge bg-warning-subtle text-warning\">\n <i class=\"bi bi-calendar-event\"></i> Planned\n </span>\n </div>\n <div class=\"text-warning\">\n <i class=\"bi bi-calendar3 fs-2\"></i>\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <div class=\"col-xl-3 col-lg-6 col-md-6 col-12 mb-3\">\n <div class=\"card h-100 border-0 shadow\">\n <div class=\"card-body\">\n <div class=\"d-flex justify-content-between align-items-start\">\n <div>\n <h6 class=\"card-title text-muted mb-2\">Completed</h6>\n <h3 class=\"mb-1 fw-bold\">{{stats.completed}}</h3>\n <span class=\"badge bg-info-subtle text-info\">\n <i class=\"bi bi-check-circle\"></i> Done\n </span>\n </div>\n <div class=\"text-info\">\n <i class=\"bi bi-check-square fs-2\"></i>\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <div class=\"col-xl-3 col-lg-6 col-md-6 col-12 mb-3\">\n <div class=\"card h-100 border-0 shadow\">\n <div class=\"card-body\">\n <div class=\"d-flex justify-content-between align-items-start\">\n <div>\n <h6 class=\"card-title text-muted mb-2\">Failed</h6>\n <h3 class=\"mb-1 fw-bold\">{{stats.failed}}</h3>\n <span class=\"badge bg-danger-subtle text-danger\">\n <i class=\"bi bi-x-octagon\"></i> Errors\n </span>\n </div>\n <div class=\"text-danger\">\n <i class=\"bi bi-exclamation-triangle fs-2\"></i>\n </div>\n </div>\n </div>\n </div>\n </div>\n </div>\n </div>\n `;\n }\n\n _onModelChange() {\n this.loadStats();\n if (this.isMounted()) {\n this.render();\n }\n }\n\n async loadStats() {\n this.stats = this.model.attributes.totals;\n }\n}\n","/**\n * JobHealthView - System health overview\n */\n\nimport View from '@core/View.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\nimport { Job } from '@core/models/Job.js';\n\nexport default class JobHealthView extends View {\n constructor(options = {}) {\n super({\n className: 'job-health-section',\n ...options\n });\n\n this.health = {\n status: 'unknown',\n runners: { active: 0, total: 0 },\n channels: []\n };\n\n this.template = `\n <div class=\"job-health-header mb-4\">\n <div class=\"card border-0 shadow\">\n <div class=\"card-body\">\n <div class=\"row align-items-center\">\n <div class=\"col-md-6\">\n <div class=\"d-flex align-items-center\">\n <div class=\"health-indicator me-3\">\n <i class=\"bi bi-circle-fill fs-4 {{healthStatusClass}}\"></i>\n </div>\n <div>\n <h5 class=\"mb-1\">Service Health: {{health.overall_status|capitalize}}</h5>\n <small class=\"text-muted d-block\">\n Workers: {{health.runners.active}}/{{health.runners.total}} active\n </small>\n <small class=\"text-muted d-block\">\n Scheduler:\n <span class=\"{{schedulerStatusClass}} fw-bold\">\n <i class=\"bi {{schedulerIcon}} me-1\"></i>{{schedulerStatusText}}\n </span>\n </small>\n </div>\n </div>\n </div>\n <div class=\"col-md-6\">\n <div class=\"d-flex justify-content-end\">\n <div class=\"btn-group\">\n <button class=\"btn btn-sm btn-outline-primary\" data-action=\"refresh-health\">\n <i class=\"bi bi-arrow-clockwise\"></i> Refresh\n </button>\n <button class=\"btn btn-sm btn-outline-secondary\" data-action=\"system-settings\">\n <i class=\"bi bi-gear\"></i> Settings\n </button>\n </div>\n </div>\n </div>\n </div>\n {{#health.channelsArray.length}}\n <div class=\"row mt-3\">\n <div class=\"col-12\">\n <small class=\"text-muted d-block mb-2\">Channel Status:</small>\n <div class=\"d-flex flex-wrap gap-2\">\n {{#health.channelsArray}}\n <span class=\"badge {{statusBadgeClass}} d-flex align-items-center\">\n <i class=\"bi {{statusIcon}} me-1\"></i>\n {{channel}} ({{queued}} queued, {{inflight}} inflight)\n </span>\n {{/health.channelsArray}}\n </div>\n </div>\n </div>\n {{/health.channelsArray.length}}\n </div>\n </div>\n </div>\n `;\n }\n\n _onModelChange() {\n this.loadHealth();\n if (this.isMounted()) {\n this.render();\n }\n }\n\n async loadHealth() {\n if (!this.model._.totals) return;\n const data = this.model.attributes\n // Determine overall health status\n let overall_status = 'healthy';\n if (data.totals.runners_active === 0) {\n overall_status = 'critical';\n } else if (!data.scheduler.active) {\n overall_status = 'warning';\n }\n this.health = {\n overall_status,\n channels: data.channels,\n runners: {\n active: data.totals.runners_active,\n total: data.runners.length\n },\n totals: data.totals,\n scheduler: data.scheduler\n };\n\n this.healthStatusClass = this.getHealthStatusClass(this.health.overall_status);\n this.setupChannelDisplay();\n this.setupSchedulerDisplay();\n\n }\n\n setupChannelDisplay() {\n if (!this.health.channels) return;\n\n this.health.channelsArray = Object.values(this.health.channels).map(channel => {\n let channelStatus = 'healthy';\n\n // Determine status based on queue backlog and runner availability\n const totalJobs = (channel.queued_count || 0) + (channel.inflight_count || 0);\n if (totalJobs > 50) {\n channelStatus = 'warning';\n }\n if (totalJobs > 100 || channel.runners === 0) {\n channelStatus = 'critical';\n }\n\n return {\n ...channel,\n status: channelStatus,\n statusBadgeClass: this.getChannelBadgeClass(channelStatus),\n statusIcon: this.getChannelIcon(channelStatus),\n queued: channel.queued_count || 0,\n inflight: channel.inflight_count || 0,\n // For backward compatibility, map to old names\n pending: channel.queued_count || 0,\n running: channel.inflight_count || 0\n };\n });\n }\n\n setupSchedulerDisplay() {\n if (!this.health.scheduler) {\n this.schedulerStatusText = 'Unknown';\n this.schedulerStatusClass = 'text-muted';\n this.schedulerIcon = 'bi-question-circle-fill';\n return;\n }\n\n if (this.health.scheduler.active) {\n this.schedulerStatusText = 'Running';\n this.schedulerStatusClass = 'text-success';\n this.schedulerIcon = 'bi-check-circle-fill';\n } else {\n this.schedulerStatusText = 'Stopped';\n this.schedulerStatusClass = 'text-danger';\n this.schedulerIcon = 'bi-x-octagon-fill';\n }\n }\n\n getHealthStatusClass(status) {\n const classes = {\n 'healthy': 'text-success',\n 'warning': 'text-warning',\n 'critical': 'text-danger',\n 'unknown': 'text-muted'\n };\n return classes[status] || 'text-muted';\n }\n\n getChannelBadgeClass(status) {\n const classes = {\n 'healthy': 'bg-success',\n 'warning': 'bg-warning',\n 'critical': 'bg-danger'\n };\n return classes[status] || 'bg-secondary';\n }\n\n getChannelIcon(status) {\n const icons = {\n 'healthy': 'bi-check-circle-fill',\n 'warning': 'bi-exclamation-triangle-fill',\n 'critical': 'bi-x-octagon-fill'\n };\n return icons[status] || 'bi-dash-circle-fill';\n }\n\n async onActionRefreshHealth(event, element) {\n try {\n element.disabled = true;\n await this.model.fetch();\n } catch (error) {\n console.error('Failed to refresh health:', error);\n } finally {\n element.disabled = false;\n }\n }\n\n async onActionSystemSettings() {\n await Dialog.showAlert({\n title: 'System Settings',\n message: 'System settings interface coming soon!',\n type: 'info'\n });\n }\n}\n","/**\n * JobDetailsView - Comprehensive job details interface\n *\n * Features:\n * - Clean header with job status, timing, and function info\n * - Tabbed interface for Overview, Payload, Events, and Logs\n * - Integrated with Table and ContextMenu components\n * - Clean Bootstrap 5 styling\n */\n\nimport View from '@core/View.js';\nimport TabView from '@core/views/navigation/TabView.js';\nimport TableView from '@core/views/table/TableView.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport { Job, JobEventList, JobLogList, JobForms } from '@core/models/Job.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\n\nclass JobDetailsView extends View {\n constructor(options = {}) {\n super({\n className: 'job-details-view',\n ...options\n });\n\n // Job model instance\n this.model = options.model || new Job(options.data || {});\n\n // Tab views\n this.tabView = null;\n this.overviewView = null;\n this.payloadView = null;\n this.eventsView = null;\n this.logsView = null;\n\n // Auto refresh\n this.autoRefreshInterval = null;\n\n // Set template\n this.template = `\n <div class=\"job-details-container\">\n <!-- Job Header -->\n <div class=\"d-flex justify-content-between align-items-start mb-4\">\n <!-- Left Side: Primary Identity -->\n <div class=\"d-flex align-items-center gap-3\">\n <div class=\"avatar-placeholder rounded-circle bg-light d-flex align-items-center justify-content-center\" style=\"width: 80px; height: 80px;\">\n <i class=\"bi {{model.statusIcon}} text-secondary\" style=\"font-size: 40px;\"></i>\n </div>\n <div>\n <h3 class=\"mb-1\">{{model.func|truncate(32)|default('Unknown Function')}}</h3>\n <div class=\"text-muted small\">\n <span>ID: {{model.id}}</span>\n <span class=\"mx-2\">|</span>\n <span>Channel: <span class=\"badge bg-primary\">{{model.channel}}</span></span>\n {{#model.runner_id}}\n <span class=\"mx-2\">|</span>\n <span>Runner: {{model.runner_id|truncate(16)}}</span>\n {{/model.runner_id}}\n </div>\n <div class=\"text-muted small mt-2\">\n <div>Created: {{model.created|datetime}}</div>\n {{#model.started_at}}\n <div>Started: {{model.started_at|datetime}}</div>\n {{/model.started_at}}\n {{#model.finished_at}}\n <div>Finished: {{model.finished_at|datetime}}</div>\n {{/model.finished_at}}\n </div>\n </div>\n </div>\n\n <!-- Right Side: Status & Actions -->\n <div class=\"d-flex align-items-start gap-4\">\n <div class=\"text-end\">\n <div class=\"d-flex align-items-center gap-2\">\n <span class=\"badge {{model.statusBadgeClass}} fs-6\">\n <i class=\"bi {{model.statusIcon}}\"></i> {{model.status|uppercase}}\n </span>\n {{#model.cancel_requested}}\n <span class=\"badge bg-warning ms-1\">\n <i class=\"bi bi-exclamation-triangle\"></i> Cancel Requested\n </span>\n {{/model.cancel_requested}}\n </div>\n {{#model.formattedDuration}}\n <div class=\"text-muted small mt-1\">Duration: {{model.formattedDuration}}</div>\n {{/model.formattedDuration}}\n </div>\n <div data-container=\"job-context-menu\"></div>\n </div>\n </div>\n\n <!-- Tab Container -->\n <div data-container=\"job-details-tabs\"></div>\n </div>\n `;\n }\n\n async onInit() {\n // Create Overview view\n this.overviewView = new View({\n template: `\n <div class=\"job-overview-tab\">\n <div class=\"card border-0 bg-light mb-3\">\n <div class=\"card-body\">\n <div class=\"row\">\n <div class=\"col-md-6\">\n <div class=\"mb-3\">\n <label class=\"form-label fw-bold text-muted small\">Job ID</label>\n <div class=\"font-monospace\">{{model.id}}</div>\n </div>\n <div class=\"mb-3\">\n <label class=\"form-label fw-bold text-muted small\">Function</label>\n <div class=\"font-monospace\">{{model.func}}</div>\n </div>\n <div class=\"mb-3\">\n <label class=\"form-label fw-bold text-muted small\">Channel</label>\n <div>\n <span class=\"badge bg-primary\">{{model.channel}}</span>\n </div>\n </div>\n {{#model.runner_id}}\n <div class=\"mb-3\">\n <label class=\"form-label fw-bold text-muted small\">Runner</label>\n <div class=\"font-monospace small\">{{model.runner_id}}</div>\n </div>\n {{/model.runner_id}}\n </div>\n <div class=\"col-md-6\">\n <div class=\"mb-3\">\n <label class=\"form-label fw-bold text-muted small\">Status</label>\n <div>\n <span class=\"badge {{model.statusBadgeClass}} fs-6\">\n <i class=\"bi {{model.statusIcon}}\"></i> {{model.status|uppercase}}\n </span>\n {{#model.cancel_requested}}\n <span class=\"badge bg-warning ms-1\">\n <i class=\"bi bi-exclamation-triangle\"></i> Cancel Requested\n </span>\n {{/model.cancel_requested}}\n </div>\n </div>\n <div class=\"mb-3\">\n <label class=\"form-label fw-bold text-muted small\">Created</label>\n <div>{{model.created|datetime}}</div>\n </div>\n {{#model.started_at}}\n <div class=\"mb-3\">\n <label class=\"form-label fw-bold text-muted small\">Started</label>\n <div>{{model.started_at|datetime}}</div>\n </div>\n {{/model.started_at}}\n {{#model.finished_at}}\n <div class=\"mb-3\">\n <label class=\"form-label fw-bold text-muted small\">Finished</label>\n <div>{{model.finished_at|datetime}}</div>\n </div>\n {{/model.finished_at}}\n {{#model.duration_ms}}\n <div class=\"mb-3\">\n <label class=\"form-label fw-bold text-muted small\">Duration</label>\n <div>{{model.formattedDuration}}</div>\n </div>\n {{/model.duration_ms}}\n </div>\n </div>\n </div>\n </div>\n </div>\n `,\n model: this.model\n });\n\n // Create Payload view\n this.payloadView = new View({\n template: `\n <div class=\"job-payload-tab\">\n <pre class=\"bg-light p-3 rounded\"><code>{{{model.payload|json}}}</code></pre>\n </div>\n `,\n model: this.model,\n });\n\n // Create Events table\n const eventsCollection = new JobEventList({\n params: { job: this.model.get('id'), size: 10 }\n });\n this.eventsView = new TableView({\n collection: eventsCollection,\n hideActivePillNames: ['job'],\n columns: [\n { key: 'at', label: 'Timestamp', formatter: 'datetime', sortable: true },\n { key: 'event', label: 'Event', formatter: 'badge' },\n { key: 'details|json', label: 'Details' }\n ]\n });\n\n // Create Logs table\n const logsCollection = new JobLogList({\n params: { job: this.model.get('id'), size: 10 }\n });\n this.logsView = new TableView({\n collection: logsCollection,\n hideActivePillNames: ['job'],\n columns: [\n { key: 'created|datetime', label: 'Created', sortable: true },\n { key: 'kind', label: 'Kind', formatter: 'badge' },\n { key: 'message', label: 'Message' }\n ]\n });\n\n // Create TabView\n this.tabView = new TabView({\n tabs: {\n 'Overview': this.overviewView,\n 'Payload': this.payloadView,\n 'Events': this.eventsView,\n 'Logs': this.logsView\n },\n activeTab: 'Overview',\n containerId: 'job-details-tabs'\n });\n this.addChild(this.tabView);\n\n // Create ContextMenu\n const contextMenuItems = [\n { label: 'Refresh', action: 'refresh-job', icon: 'bi-arrow-clockwise' }\n ];\n\n if (this.model.canCancel && this.model.canCancel()) {\n contextMenuItems.push({\n label: 'Cancel Job',\n action: 'cancel-job',\n icon: 'bi-x-circle',\n class: 'text-danger'\n });\n }\n\n if (this.model.canRetry && this.model.canRetry()) {\n contextMenuItems.push({\n label: 'Retry Job',\n action: 'retry-job',\n icon: 'bi-arrow-repeat',\n class: 'text-primary'\n });\n }\n\n const jobMenu = new ContextMenu({\n containerId: 'job-context-menu',\n className: \"context-menu-view header-menu-absolute\",\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: contextMenuItems\n }\n });\n this.addChild(jobMenu);\n\n await this.model.fetch({ params: { graph: \"detail\" } });\n // Start auto-refresh if job is active\n // this.startAutoRefresh();\n }\n\n async onBeforeRender() {\n await this.prepareJobData();\n }\n\n async prepareJobData() {\n if (!this.model) return;\n\n // Status styling\n this.model._.statusBadgeClass = this.model.getStatusBadgeClass ? this.model.getStatusBadgeClass() : 'bg-secondary';\n this.model._.statusIcon = this.model.getStatusIcon ? this.model.getStatusIcon() : 'bi-question-circle';\n this.model._.formattedDuration = this.model.getFormattedDuration ? this.model.getFormattedDuration() : 'N/A';\n }\n\n async loadJobDetails() {\n if (!this.model?.get('id')) return;\n\n try {\n if (this.model.getDetailedStatus) {\n await this.model.getDetailedStatus();\n }\n await this.prepareJobData();\n } catch (error) {\n console.error('Failed to load job details:', error);\n }\n }\n\n async onActionRefreshJob() {\n await this.model.fetch({ params: { graph: \"detail\" } });\n }\n\n async onActionCancelJob() {\n if (confirm('Are you sure you want to cancel this job?')) {\n try {\n const response = await this.model.cancel();\n if (response.success) {\n await this.loadJobDetails();\n await this.render();\n this.emit('job-cancelled', { job: this.model });\n } else {\n alert('Failed to cancel job: ' + (response.data?.error || 'Unknown error'));\n }\n } catch (error) {\n console.error('Failed to cancel job:', error);\n alert('Failed to cancel job: ' + error.message);\n }\n }\n }\n\n async onActionRetryJob() {\n const retryData = await Dialog.showForm({\n title: 'Retry Job',\n formConfig: JobForms.retry\n });\n\n if (retryData) {\n try {\n const response = await this.model.retry(retryData.delay || 0);\n if (response.success) {\n this.emit('job-retried', { job: this.model, newJobId: response.newJobId });\n } else {\n alert('Failed to retry job: ' + (response.data?.error || 'Unknown error'));\n }\n } catch (error) {\n console.error('Failed to retry job:', error);\n alert('Failed to retry job: ' + error.message);\n }\n }\n }\n\n startAutoRefresh() {\n if (this.autoRefreshInterval) clearInterval(this.autoRefreshInterval);\n\n if (this.model?.isActive && this.model.isActive()) {\n this.autoRefreshInterval = setInterval(async () => {\n try {\n await this.loadJobDetails();\n if (this.isMounted()) {\n await this.render();\n }\n } catch (error) {\n console.error('Auto-refresh failed:', error);\n }\n }, 5000);\n }\n }\n\n stopAutoRefresh() {\n if (this.autoRefreshInterval) {\n clearInterval(this.autoRefreshInterval);\n this.autoRefreshInterval = null;\n }\n }\n\n async onDestroy() {\n this.stopAutoRefresh();\n await super.onDestroy();\n }\n\n // Static show method for backward compatibility (if needed)\n static async show(job, options = {}) {\n const view = new JobDetailsView({ model: job });\n\n return await Dialog.showDialog({\n title: `<i class=\"bi bi-info-circle me-2\"></i>Job Details - ${job.get('id')}`,\n body: view,\n size: 'xl',\n scrollable: true,\n buttons: [{ text: 'Close', class: 'btn-secondary', dismiss: true }],\n onHide: () => view.stopAutoRefresh(),\n ...options\n });\n }\n}\n\nJob.VIEW_CLASS = JobDetailsView;\n\nexport default JobDetailsView;\n","/**\n * JobsAdminPage - Async Job Engine management dashboard\n * Main page orchestrating job monitoring and management components\n */\n\nimport Page from '@core/Page.js';\nimport View from '@core/View.js';\nimport TableView from '@core/views/table/TableView.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\nimport { Job, JobList, JobsEngineStats } from '@core/models/Job.js';\nimport { JobRunner, JobRunnerForms, JobRunnerList } from '@core/models/JobRunner.js';\nimport { MetricsChart, MetricsMiniChartWidget } from '@ext/charts/index.js';\n\n// Import component views\nimport JobStatsView from './JobStatsView.js';\nimport JobHealthView from './JobHealthView.js';\nimport JobDetailsView from './JobDetailsView.js';\n\nclass JobMetricsModalView extends View {\n constructor(options = {}) {\n super({\n className: 'job-metrics-modal-view',\n ...options\n });\n\n this.template = `\n <div data-container=\"job-metrics-modal-chart\" style=\"min-height:320px;\"></div>\n `;\n }\n\n async onInit() {\n this.chart = new MetricsChart({\n containerId: 'job-metrics-modal-chart',\n title: '<i class=\"bi bi-graph-up me-2\"></i> Job Channel Metrics',\n endpoint: '/api/metrics/fetch',\n height: 320,\n granularity: 'hours',\n category: 'jobs_channels',\n account: 'global',\n chartType: 'bar',\n showDateRange: true,\n yAxis: {\n label: 'Count',\n beginAtZero: true\n },\n tooltip: {\n y: 'number:0'\n }\n });\n\n this.addChild(this.chart);\n }\n}\n\nexport default class JobsAdminPage extends Page {\n constructor(options = {}) {\n super({\n title: 'Jobs Management',\n className: 'jobs-admin-page',\n ...options\n });\n\n this.pageTitle = 'Jobs Management';\n this.pageSubtitle = 'Async job monitoring and runner management';\n this.lastUpdated = new Date().toLocaleString();\n this.autoRefreshInterval = null;\n this.refreshRate = 30000; // 30 seconds default\n\n this.template = `\n <div class=\"jobs-admin-container\">\n <!-- Page Header -->\n <div class=\"d-flex justify-content-between align-items-center mb-4\">\n <div>\n <h1 class=\"h3 mb-1\">{{pageTitle}}</h1>\n <p class=\"text-muted mb-0\">{{pageSubtitle}}</p>\n <small class=\"text-info\">\n <i class=\"bi bi-arrow-clockwise me-1\"></i>\n Auto-refresh: {{refreshRateLabel}} | Last updated: {{lastUpdated}}\n </small>\n </div>\n <div class=\"btn-group\" role=\"group\">\n <button type=\"button\" class=\"btn btn-outline-secondary btn-sm\"\n data-action=\"refresh-all\" title=\"Refresh All Data\">\n <i class=\"bi bi-arrow-clockwise\"></i> Refresh\n </button>\n <button type=\"button\" class=\"btn btn-outline-primary btn-sm\"\n data-action=\"view-all-jobs\" title=\"View All Jobs\">\n <i class=\"bi bi-table\"></i> View All\n </button>\n <div class=\"dropdown\">\n <button class=\"btn btn-outline-secondary btn-sm dropdown-toggle\"\n type=\"button\" data-bs-toggle=\"dropdown\">\n <i class=\"bi bi-gear\"></i> Settings\n </button>\n <ul class=\"dropdown-menu dropdown-menu-end\">\n <li><h6 class=\"dropdown-header\">Auto Refresh</h6></li>\n <li><button class=\"dropdown-item\" data-action=\"set-refresh-rate\" data-rate=\"5\">5 seconds</button></li>\n <li><button class=\"dropdown-item\" data-action=\"set-refresh-rate\" data-rate=\"10\">10 seconds</button></li>\n <li><button class=\"dropdown-item\" data-action=\"set-refresh-rate\" data-rate=\"30\">30 seconds</button></li>\n <li><button class=\"dropdown-item\" data-action=\"set-refresh-rate\" data-rate=\"0\">Off</button></li>\n <li><hr class=\"dropdown-divider\"></li>\n <li><button class=\"dropdown-item\" data-action=\"runner-broadcast\">Broadcast Command</button></li>\n </ul>\n </div>\n <div class=\"dropdown\">\n <button class=\"btn btn-danger btn-sm dropdown-toggle\"\n type=\"button\" data-bs-toggle=\"dropdown\">\n <i class=\"bi bi-wrench\"></i> Manage\n </button>\n <ul class=\"dropdown-menu dropdown-menu-end\">\n <li><button class=\"dropdown-item\" data-action=\"run-simple-job\"><i class=\"bi bi-play-circle me-2\"></i>Run Simple Job</button></li>\n <li><button class=\"dropdown-item\" data-action=\"run-test-jobs\"><i class=\"bi bi-robot me-2\"></i>Run Test Jobs</button></li>\n <li><hr class=\"dropdown-divider\"></li>\n <li><button class=\"dropdown-item\" data-action=\"clear-stuck\"><i class=\"bi bi-wrench me-2\"></i>Clear Stuck Jobs</button></li>\n <li><button class=\"dropdown-item\" data-action=\"clear-channel\"><i class=\"bi bi-eraser me-2\"></i>Clear Channel</button></li>\n <li><button class=\"dropdown-item\" data-action=\"purge-jobs\"><i class=\"bi bi-trash me-2\"></i>Purge Jobs</button></li>\n <li><button class=\"dropdown-item\" data-action=\"cleanup-consumers\"><i class=\"bi bi-people me-2\"></i>Cleanup Consumers</button></li>\n </ul>\n </div>\n </div>\n </div>\n\n <div data-container=\"job-stats\"></div>\n\n <div class=\"row mb-4 g-3 align-items-stretch\">\n <div class=\"col-lg-6\" data-container=\"jobs-published-chart\"></div>\n <div class=\"col-lg-6 position-relative\">\n <div data-container=\"jobs-failed-chart\"></div>\n </div>\n </div>\n\n <div class=\"row\">\n <div class=\"col-xxl-6 col-lg-6 mb-4\">\n <div data-container=\"job-health\"></div>\n </div>\n <div class=\"col-xxl-6 col-lg-6 mb-4\">\n <div class=\"card shadow-sm h-100\">\n <div class=\"card-header d-flex justify-content-between align-items-center\">\n <h5 class=\"mb-0\"><i class=\"bi bi-cpu me-2\"></i>Job Runners</h5>\n <small class=\"text-muted\">Heartbeat &amp; status</small>\n </div>\n <div class=\"card-body p-0\" data-container=\"runner-table\"></div>\n </div>\n </div>\n </div>\n\n <div class=\"row\">\n <div class=\"col-xl-6 mb-4\">\n <div class=\"card shadow-sm h-100\">\n <div class=\"card-header d-flex justify-content-between align-items-center\">\n <h5 class=\"mb-0\"><i class=\"bi bi-play-circle me-2\"></i>Running Jobs</h5>\n <small class=\"text-muted\">Currently executing</small>\n </div>\n <div class=\"card-body p-0\" data-container=\"running-jobs-table\"></div>\n </div>\n </div>\n <div class=\"col-xl-6 mb-4\">\n <div class=\"card shadow-sm h-100\">\n <div class=\"card-header d-flex justify-content-between align-items-center\">\n <h5 class=\"mb-0\"><i class=\"bi bi-hourglass-split me-2\"></i>Pending Jobs</h5>\n <small class=\"text-muted\">Waiting in queue</small>\n </div>\n <div class=\"card-body p-0\" data-container=\"pending-jobs-table\"></div>\n </div>\n </div>\n </div>\n\n <div class=\"row\">\n <div class=\"col-xl-6 mb-4\">\n <div class=\"card shadow-sm h-100\">\n <div class=\"card-header\">\n <h5 class=\"mb-0\"><i class=\"bi bi-calendar-event me-2\"></i>Scheduled Jobs</h5>\n </div>\n <div class=\"card-body p-0\" data-container=\"scheduled-jobs-table\"></div>\n </div>\n </div>\n <div class=\"col-xl-6 mb-4\">\n <div class=\"card shadow-sm h-100\">\n <div class=\"card-header d-flex justify-content-between align-items-center\">\n <h5 class=\"mb-0\"><i class=\"bi bi-bug me-2\"></i>Failed Jobs</h5>\n <small class=\"text-muted\">Latest errors</small>\n </div>\n <div class=\"card-body p-0\" data-container=\"failed-jobs-table\"></div>\n </div>\n </div>\n </div>\n\n <div class=\"card shadow-sm mb-5\">\n <div class=\"card-header d-flex justify-content-between align-items-center\">\n <h5 class=\"mb-0\"><i class=\"bi bi-tools me-2\"></i>Operations</h5>\n </div>\n <div class=\"card-body\">\n <div class=\"d-flex flex-wrap gap-2\">\n <button class=\"btn btn-outline-primary\" data-action=\"run-simple-job\">\n <i class=\"bi bi-play-circle me-2\"></i>Run Simple Job\n </button>\n <button class=\"btn btn-outline-primary\" data-action=\"run-test-jobs\">\n <i class=\"bi bi-robot me-2\"></i>Run Test Jobs\n </button>\n <button class=\"btn btn-outline-warning\" data-action=\"clear-stuck\">\n <i class=\"bi bi-wrench me-2\"></i>Clear Stuck\n </button>\n <button class=\"btn btn-outline-warning\" data-action=\"clear-channel\">\n <i class=\"bi bi-eraser me-2\"></i>Clear Channel\n </button>\n <button class=\"btn btn-outline-danger\" data-action=\"purge-jobs\">\n <i class=\"bi bi-trash me-2\"></i>Purge Jobs\n </button>\n <button class=\"btn btn-outline-info\" data-action=\"cleanup-consumers\">\n <i class=\"bi bi-people me-2\"></i>Cleanup Consumers\n </button>\n <button class=\"btn btn-outline-secondary\" data-action=\"runner-broadcast\">\n <i class=\"bi bi-wifi me-2\"></i>Broadcast Command\n </button>\n </div>\n </div>\n </div>\n </div>\n `;\n }\n\n async onInit() {\n\n this.jobStats = new JobsEngineStats();\n // Create child views\n this.jobStatsView = new JobStatsView({\n containerId: 'job-stats',\n model: this.jobStats\n });\n this.addChild(this.jobStatsView);\n\n this.jobHealthView = new JobHealthView({\n containerId: 'job-health',\n model: this.jobStats\n });\n this.addChild(this.jobHealthView);\n\n this.jobsPublishedChart = new MetricsMiniChartWidget({\n containerId: 'jobs-published-chart',\n icon: 'bi bi-upload',\n title: 'Jobs Published',\n subtitle: '{{now_value}} {{now_label}}',\n granularity: 'days',\n slugs: ['jobs.published'],\n account: 'global',\n chartType: 'line',\n height: 90,\n showSettings: true,\n showTrending: true,\n showDateRange: false\n });\n this.addChild(this.jobsPublishedChart);\n\n this.jobsFailedChart = new MetricsMiniChartWidget({\n containerId: 'jobs-failed-chart',\n icon: 'bi bi-exclamation-octagon',\n title: 'Jobs Failed',\n subtitle: '{{now_value}} {{now_label}}',\n granularity: 'days',\n slugs: ['jobs.failed'],\n account: 'global',\n chartType: 'line',\n height: 90,\n showSettings: true,\n showTrending: true,\n showDateRange: false\n });\n this.addChild(this.jobsFailedChart);\n\n this.runningJobsTable = new TableView({\n containerId: 'running-jobs-table',\n Collection: JobList,\n collectionParams: {\n size: 5,\n sort: '-created',\n status: 'running'\n },\n searchable: true,\n filterable: false,\n paginated: true,\n itemView: JobDetailsView,\n hideActivePills: ['status'],\n viewDialogOptions: {\n title: 'Job Details',\n size: 'xl',\n scrollable: true\n },\n tableOptions: {\n striped: false,\n hover: true,\n size: 'sm'\n },\n columns: [\n {\n key: 'id',\n label: 'Job',\n template: `\n <div class=\"fw-semibold font-monospace\">{{model.id|truncate_middle(12)}}</div>\n <div class=\"text-muted small\">{{model.channel}} &middot; {{model.func|truncate_middle(28)|default('n/a')}}</div>\n `\n },\n {\n key: 'runner_id',\n label: 'Runner',\n template: `\n <span class=\"font-monospace\">{{model.runner_id|truncate_middle(12)|default('n/a')}}</span>\n `\n },\n {\n key: 'status',\n label: 'State',\n formatter: (value, context) => {\n const job = context.row;\n const badgeClass = job.getStatusBadgeClass ? job.getStatusBadgeClass() : 'bg-secondary';\n const icon = job.getStatusIcon ? job.getStatusIcon() : 'bi-question';\n return `<span class=\"badge ${badgeClass}\"><i class=\"${icon} me-1\"></i>${value.toUpperCase()}</span>`;\n }\n },\n {\n key: 'created',\n label: 'Started',\n formatter: 'datetime'\n }\n ]\n });\n this.addChild(this.runningJobsTable);\n\n this.pendingJobsTable = new TableView({\n containerId: 'pending-jobs-table',\n Collection: JobList,\n collectionParams: {\n size: 5,\n sort: '-created',\n status: 'pending',\n run_at__isnull: true\n },\n searchable: true,\n filterable: false,\n paginated: true,\n selectable: true,\n batchBarLocation: 'top',\n batchActions: [\n {\n icon: 'bi-x-circle-fill',\n label: 'Cancel Jobs',\n action: \"cancel-jobs\"\n }\n ],\n itemView: JobDetailsView,\n hideActivePills: ['status'],\n viewDialogOptions: {\n title: 'Job Details',\n size: 'xl',\n scrollable: true\n },\n tableOptions: {\n striped: false,\n hover: true,\n size: 'sm'\n },\n columns: [\n {\n key: 'id',\n label: 'Job',\n template: `\n <div class=\"fw-semibold font-monospace\">{{model.id|truncate_middle(12)}}</div>\n <div class=\"text-muted small\">{{model.channel}} &middot; {{model.func|truncate_middle(28)|default('n/a')}}</div>\n `\n },\n {\n key: 'priority',\n label: 'Priority',\n formatter: (value = 0) => {\n const badge = value >= 8 ? 'bg-danger' : value >= 5 ? 'bg-warning' : 'bg-secondary';\n return `<span class=\"badge ${badge}\">${value}</span>`;\n }\n },\n {\n key: 'modified',\n label: 'Queued',\n formatter: 'relative'\n }\n ]\n });\n this.pendingJobsTable.on(\"action:batch-cancel-jobs\", async (action, event, element) => {\n const items = this.pendingJobsTable.getSelectedItems();\n await Promise.all(items.map(item => item.model.cancel()));\n this.getApp().toast.success(\"Jobs cancelled successfully\");\n this.pendingJobsTable.collection.fetch();\n });\n this.addChild(this.pendingJobsTable);\n\n this.failedJobsTable = new TableView({\n containerId: 'failed-jobs-table',\n Collection: JobList,\n collectionParams: {\n size: 5,\n sort: '-finished_at',\n status: 'failed'\n },\n searchable: true,\n filterable: false,\n paginated: true,\n itemView: JobDetailsView,\n viewDialogOptions: {\n title: 'Job Details',\n size: 'xl',\n scrollable: true\n },\n hideActivePills: ['status'],\n tableOptions: {\n striped: false,\n hover: true,\n size: 'sm'\n },\n columns: [\n {\n key: 'id',\n label: 'Job',\n template: `\n <div class=\"fw-semibold font-monospace\">{{model.id|truncate_middle(12)}}</div>\n <div class=\"text-muted small\">{{model.channel}} &middot; {{model.func|truncate_middle(28)|default('n/a')}}</div>\n `\n },\n {\n key: 'last_error',\n label: 'Error',\n template: `\n <div class=\"text-danger small\">{{model.last_error|truncate(80)|default('Unknown error')}}</div>\n `\n },\n {\n key: 'modified',\n label: 'Failed',\n formatter: 'relative'\n }\n ]\n });\n this.addChild(this.failedJobsTable);\n\n this.scheduledJobsTable = new TableView({\n containerId: 'scheduled-jobs-table',\n Collection: JobList,\n collectionParams: {\n size: 5,\n sort: 'run_at',\n run_at__isnull: false,\n status: 'pending'\n },\n searchable: true,\n filterable: false,\n paginated: true,\n itemView: JobDetailsView,\n hideActivePills: ['status'],\n viewDialogOptions: {\n title: 'Job Details',\n size: 'xl',\n scrollable: true\n },\n tableOptions: {\n striped: false,\n hover: true,\n size: 'sm'\n },\n columns: [\n {\n key: 'id',\n label: 'Job',\n formatter: 'truncate_middle(12)'\n },\n {\n key: 'run_at',\n label: 'Scheduled For',\n formatter: 'datetime'\n },\n {\n key: 'channel',\n label: 'Channel',\n formatter: 'badge'\n }\n ],\n selectable: true,\n batchBarLocation: 'top',\n batchActions: [\n {\n icon: 'bi-x-circle-fill',\n label: 'Cancel Jobs',\n action: \"cancel-jobs\"\n }\n ],\n });\n this.scheduledJobsTable.on(\"action:batch-cancel-jobs\", async (action, event, element) => {\n const items = this.scheduledJobsTable.getSelectedItems();\n await Promise.all(items.map(item => item.model.cancel()));\n this.getApp().toast.success(\"Jobs cancelled successfully\");\n this.scheduledJobsTable.collection.fetch();\n });\n this.addChild(this.scheduledJobsTable);\n\n this.runnersTable = new TableView({\n containerId: 'runner-table',\n Collection: JobRunnerList,\n searchable: true,\n filterable: false,\n paginated: true,\n tableOptions: {\n striped: false,\n hover: true,\n size: 'sm'\n },\n columns: [\n {\n key: 'runner_id',\n label: 'Runner',\n formatter: 'truncate_middle(16)'\n },\n {\n key: 'alive',\n label: 'Status',\n formatter: (value) => {\n const isAlive = value === true;\n const badgeClass = isAlive ? 'bg-success' : 'bg-danger';\n const icon = isAlive ? 'bi-check-circle-fill' : 'bi-x-octagon-fill';\n const text = isAlive ? 'ALIVE' : 'DEAD';\n return `<span class=\"badge ${badgeClass}\"><i class=\"${icon} me-1\"></i>${text}</span>`;\n }\n },\n {\n key: 'last_heartbeat',\n label: 'Heartbeat',\n formatter: (value) => {\n if (!value) return 'Never';\n const heartbeatTime = new Date(value);\n const now = new Date();\n const diffMs = now - heartbeatTime;\n const diffSeconds = Math.floor(diffMs / 1000);\n if (diffSeconds < 60) return `${diffSeconds}s ago`;\n if (diffSeconds < 3600) return `${Math.floor(diffSeconds / 60)}m ago`;\n return `${Math.floor(diffSeconds / 3600)}h ago`;\n }\n }\n ]\n });\n this.addChild(this.runnersTable);\n\n await this.refreshData();\n\n }\n\n // Auto-refresh management\n startAutoRefresh() {\n if (this.autoRefreshInterval) {\n clearInterval(this.autoRefreshInterval);\n }\n\n if (this.refreshRate > 0) {\n this.autoRefreshInterval = setInterval(async () => {\n await this.refreshData();\n }, this.refreshRate);\n }\n }\n\n async refreshData() {\n try {\n const tasks = [\n this.jobStats.fetch()\n ];\n\n if (this.jobsPublishedChart) {\n tasks.push(this.jobsPublishedChart.refresh());\n }\n if (this.jobsFailedChart) {\n tasks.push(this.jobsFailedChart.refresh());\n }\n if (this.runningJobsTable?.collection?.fetch) {\n tasks.push(this.runningJobsTable.collection.fetch());\n }\n if (this.pendingJobsTable?.collection?.fetch) {\n tasks.push(this.pendingJobsTable.collection.fetch());\n }\n if (this.failedJobsTable?.collection?.fetch) {\n tasks.push(this.failedJobsTable.collection.fetch());\n }\n if (this.scheduledJobsTable?.collection?.fetch) {\n tasks.push(this.scheduledJobsTable.collection.fetch());\n }\n if (this.runnersTable?.collection?.fetch) {\n tasks.push(this.runnersTable.collection.fetch());\n }\n\n await Promise.all(tasks);\n\n this.lastUpdated = new Date().toLocaleString();\n this.updateHeaderTimestamp();\n\n } catch (error) {\n console.error('Failed to refresh jobs dashboard:', error);\n }\n }\n\n updateHeaderTimestamp() {\n const timestampElement = this.element?.querySelector('.text-info');\n if (timestampElement) {\n const rateLabel = this.refreshRate === 0 ? 'Off' : `${this.refreshRate / 1000}s`;\n timestampElement.innerHTML = `\n <i class=\"bi bi-arrow-clockwise me-1\"></i>\n Auto-refresh: ${rateLabel} | Last updated: ${this.lastUpdated}\n `;\n }\n }\n\n get refreshRateSeconds() {\n return this.refreshRate / 1000;\n }\n\n get refreshRateLabel() {\n return this.refreshRate === 0 ? 'Off' : `${this.refreshRateSeconds}s`;\n }\n\n // Action handlers\n async onActionRefreshAll(event, element) {\n try {\n const icon = element.querySelector('i');\n icon?.classList.add('spinning');\n element.disabled = true;\n\n await this.refreshData();\n\n } catch (error) {\n console.error('Failed to refresh jobs dashboard:', error);\n } finally {\n const icon = element.querySelector('i');\n icon?.classList.remove('spinning');\n element.disabled = false;\n }\n }\n\n async onActionSetRefreshRate(event, element) {\n const rate = parseInt(element.getAttribute('data-rate')) * 1000;\n this.refreshRate = rate;\n this.startAutoRefresh();\n\n const rateText = rate === 0 ? 'Off' : `${rate / 1000}s`;\n this.getApp().toast.success(`Auto-refresh set to ${rateText}`);\n }\n\n async onActionExportData() {\n await Dialog.showAlert({\n title: 'Export Data',\n message: 'Data export functionality coming soon!',\n type: 'info'\n });\n }\n\n // Job management actions\n async onActionRunSimpleJob(event, element) {\n const confirmed = await Dialog.showConfirm({\n title: 'Run Simple Job',\n message: 'This will run a simple test job to verify the job system is working correctly.',\n confirmText: 'Run Test',\n confirmClass: 'btn-success'\n });\n\n if (confirmed) {\n await this.executeJobAction(element, () => Job.test(), 'Test job started successfully');\n }\n }\n\n async onActionRunTestJobs(event, element) {\n const confirmed = await Dialog.showConfirm({\n title: 'Run Test Jobs',\n message: 'This will run a suite of test jobs to verify all job functionalities.',\n confirmText: 'Run Tests',\n confirmClass: 'btn-success'\n });\n\n if (confirmed) {\n await this.executeJobAction(element, () => Job.tests(), 'Test suite started successfully');\n }\n }\n\n async onActionClearStuck(event, element) {\n const channels = this.jobHealthView?.health?.channelsArray || [];\n const channelOptions = [\n { value: '', label: 'All Channels' },\n ...channels.map(ch => ({ value: ch.channel, label: ch.channel }))\n ];\n\n const result = await Dialog.showForm({\n title: 'Clear Stuck Jobs',\n formConfig: {\n fields: [\n {\n name: 'channel',\n type: 'select',\n label: 'Channel',\n options: channelOptions,\n value: '',\n help: 'Select specific channel or leave empty for all channels'\n }\n ]\n }\n });\n\n if (result) {\n await this.executeJobAction(\n element,\n () => Job.clearStuck(result.channel || null),\n (response) => {\n const count = response.data.count || 0;\n const channelText = result.channel ? ` from channel \"${result.channel}\"` : '';\n return `Cleared ${count} stuck job${count !== 1 ? 's' : ''}${channelText}`;\n }\n );\n }\n }\n\n async onActionClearChannel(event, element) {\n const channels = this.jobHealthView?.health?.channelsArray || [];\n const channelOptions = channels.map(ch => ({ value: ch.channel, label: ch.channel }));\n\n const result = await Dialog.showForm({\n title: 'Clear Channel',\n formConfig: {\n fields: [\n {\n name: 'channel',\n type: 'select',\n label: 'Channel',\n options: channelOptions,\n required: true,\n help: 'Select the channel to clear.'\n }\n ]\n }\n });\n\n if (result) {\n await this.executeJobAction(\n element,\n () => Job.clearChannel(result.channel),\n `Channel \"${result.channel}\" cleared successfully.`\n );\n }\n }\n\n async onActionPurgeJobs(event, element) {\n const result = await Dialog.showForm({\n title: 'Purge Old Jobs',\n formConfig: {\n fields: [\n {\n name: 'days_old',\n type: 'number',\n label: 'Days Old',\n value: 30,\n required: true,\n help: 'Delete jobs older than this many days.'\n }\n ]\n }\n });\n\n if (result) {\n await this.executeJobAction(\n element,\n () => Job.purgeJobs(result.days_old),\n (response) => {\n const count = response.data.count || 0;\n return `Purged ${count} old job(s).`;\n }\n );\n }\n }\n\n async onActionCleanupConsumers(event, element) {\n const confirmed = await Dialog.showConfirm({\n title: 'Cleanup Consumers',\n message: 'This will remove stale consumer records from the system. This is generally safe.',\n confirmText: 'Cleanup',\n confirmClass: 'btn-warning'\n });\n\n if (confirmed) {\n await this.executeJobAction(\n element,\n () => Job.cleanConsumers(),\n (response) => {\n const count = response.data.count || 0;\n return `Cleaned up ${count} consumer(s).`;\n }\n );\n }\n }\n\n async onActionRunnerBroadcast() {\n const result = await Dialog.showForm({\n title: 'Broadcast Command to All Runners',\n formConfig: JobRunnerForms.broadcast\n });\n\n if (result) {\n try {\n const broadcastResult = await JobRunner.broadcast(\n result.command,\n {},\n result.timeout\n );\n\n if (broadcastResult.success) {\n this.getApp().toast.success(`Broadcast command \"${result.command}\" sent successfully`);\n await this.refreshData();\n } else {\n this.getApp().toast.error(broadcastResult.data?.error || 'Failed to broadcast command');\n }\n } catch (error) {\n console.error('Failed to broadcast command:', error);\n this.getApp().toast.error('Error broadcasting command: ' + error.message);\n }\n }\n }\n\n // Helper method to execute job actions with consistent error handling\n async executeJobAction(element, actionFn, successMessage) {\n try {\n element.disabled = true;\n const icon = element.querySelector('i');\n icon?.classList.add('spinning');\n\n const result = await actionFn();\n if (result.success && result.data?.status) {\n const message = typeof successMessage === 'function'\n ? successMessage(result)\n : successMessage;\n this.getApp().toast.success(message);\n await this.refreshData();\n } else {\n this.getApp().toast.error(result.data?.error || 'Operation failed');\n }\n } catch (error) {\n console.error('Job action failed:', error);\n this.getApp().toast.error('Error: ' + error.message);\n } finally {\n element.disabled = false;\n const icon = element.querySelector('i');\n icon?.classList.remove('spinning');\n }\n }\n\n async onEnter() {\n // Start auto-refresh\n this.startAutoRefresh();\n }\n\n async onExit() {\n if (this.autoRefreshInterval) {\n clearInterval(this.autoRefreshInterval);\n this.autoRefreshInterval = null;\n }\n }\n\n // Public API\n async refreshDashboard() {\n return await this.refreshData();\n }\n\n getStats() {\n return this.jobStatsView?.stats || {};\n }\n\n getHealth() {\n return this.jobHealthView?.health || {};\n }\n\n async onActionOpenJobMetricsModal() {\n const modalView = new JobMetricsModalView();\n await Dialog.showDialog({\n title: '<i class=\"bi bi-graph-up me-2\"></i>Job Channel Metrics',\n body: modalView,\n size: 'xl',\n scrollable: true,\n buttons: [\n { text: 'Close', class: 'btn-secondary', dismiss: true }\n ]\n });\n }\n\n async onActionViewAllJobs() {\n const jobList = new JobList({\n params: {\n sort: '-created',\n size: 25\n }\n });\n\n const view = new TableView({\n collection: jobList,\n fetchOnMount: true,\n columns: [\n {\n key: 'id',\n label: 'Job',\n template: `\n <div class=\"fw-semibold font-monospace\">{{model.id}}</div>\n <div class=\"text-muted small\">{{model.func|default('Unknown')}}</div>\n `\n },\n {\n key: 'channel',\n label: 'Channel',\n formatter: 'badge'\n },\n {\n key: 'status',\n label: 'Status',\n formatter: (value, context) => {\n const job = context.row;\n const badgeClass = job.getStatusBadgeClass ? job.getStatusBadgeClass() : 'bg-secondary';\n const icon = job.getStatusIcon ? job.getStatusIcon() : 'bi-question';\n return `<span class=\"badge ${badgeClass}\"><i class=\"${icon} me-1\"></i>${value?.toUpperCase() || 'UNKNOWN'}</span>`;\n }\n },\n {\n key: 'created',\n label: 'Created',\n formatter: 'datetime'\n },\n {\n key: 'finished_at',\n label: 'Finished',\n formatter: 'datetime'\n },\n {\n key: 'duration_ms',\n label: 'Duration',\n formatter: 'duration'\n }\n ],\n filters: [\n {\n key: 'func__icontains',\n label: 'Function',\n type: 'text'\n },\n {\n key: 'status',\n label: 'Status',\n type: 'select',\n options: [\n { label: 'Pending', value: 'pending' },\n { label: 'Running', value: 'running' },\n { label: 'Completed', value: 'completed' },\n { label: 'Failed', value: 'failed' },\n { label: 'Canceled', value: 'canceled' },\n { label: 'Expired', value: 'expired' }\n ]\n },\n {\n key: 'channel',\n label: 'Channel',\n type: 'text'\n }\n ],\n itemView: JobDetailsView,\n viewDialogOptions: {\n title: 'Job Details',\n size: 'xl',\n scrollable: true\n },\n searchable: true,\n filterable: true,\n paginated: true,\n tableOptions: {\n striped: true,\n hover: true,\n responsive: true\n }\n });\n\n await this.getApp().showDialog({\n title: '<i class=\"bi bi-table me-2\"></i>All Jobs',\n body: view,\n size: 'xl',\n scrollable: true,\n buttons: [\n { text: 'Close', class: 'btn-secondary', dismiss: true }\n ]\n });\n }\n}\n","/**\n * TaskDetailsView - Detailed task information view for Dialog display\n * Shows comprehensive task data, logs, metrics, and available actions\n */\n\nimport View from '@core/View.js';\n\nexport default class TaskDetailsView extends View {\n constructor(options = {}) {\n super({\n ...options,\n className: 'mojo-task-details-view'\n });\n\n this.task = options.task || null;\n this.logs = [];\n this.metrics = null;\n }\n\n async getTemplate() {\n return `\n <div class=\"mojo-task-details-container\">\n {{#task}}\n <!-- Task Overview -->\n <div class=\"card border-0 bg-light mb-3\">\n <div class=\"card-body\">\n <div class=\"row\">\n <div class=\"col-md-6\">\n <div class=\"mb-3\">\n <label class=\"form-label fw-bold text-muted small\">Task ID</label>\n <div class=\"font-monospace\">{{id}}</div>\n </div>\n <div class=\"mb-3\">\n <label class=\"form-label fw-bold text-muted small\">Function</label>\n <div>{{function}}</div>\n </div>\n <div class=\"mb-3\">\n <label class=\"form-label fw-bold text-muted small\">Channel</label>\n <div>\n <span class=\"badge bg-primary\">{{channel}}</span>\n </div>\n </div>\n </div>\n <div class=\"col-md-6\">\n <div class=\"mb-3\">\n <label class=\"form-label fw-bold text-muted small\">Status</label>\n <div>\n <span class=\"badge {{statusBadgeClass}} fs-6\">\n <i class=\"bi {{statusIcon}}\"></i> {{status|uppercase}}\n </span>\n </div>\n </div>\n <div class=\"mb-3\">\n <label class=\"form-label fw-bold text-muted small\">Created</label>\n <div>{{created|datetime}}</div>\n </div>\n {{#completed_at}}\n <div class=\"mb-3\">\n <label class=\"form-label fw-bold text-muted small\">Completed</label>\n <div>{{completed_at|datetime}}</div>\n </div>\n {{/completed_at}}\n {{#expires}}\n <div class=\"mb-3\">\n <label class=\"form-label fw-bold text-muted small\">Expires</label>\n <div class=\"{{expiresClass}}\">{{expires|datetime}}</div>\n </div>\n {{/expires}}\n </div>\n </div>\n </div>\n </div>\n\n <!-- Task Data -->\n {{#data}}\n <div class=\"card mb-3\">\n <div class=\"card-header py-2\">\n <h6 class=\"mb-0\">\n <i class=\"bi bi-database me-2\"></i>Task Data\n </h6>\n </div>\n <div class=\"card-body\">\n <pre class=\"bg-light p-3 rounded mb-0\"><code>{{dataFormatted}}</code></pre>\n </div>\n </div>\n {{/data}}\n\n <!-- Error Information -->\n {{#error}}\n <div class=\"card border-danger mb-3\">\n <div class=\"card-header bg-danger-subtle py-2\">\n <h6 class=\"mb-0 text-danger\">\n <i class=\"bi bi-exclamation-triangle me-2\"></i>Error Details\n </h6>\n </div>\n <div class=\"card-body\">\n <div class=\"alert alert-danger mb-0\">\n <strong>Error:</strong> {{error}}\n </div>\n {{#errorDetails}}\n <div class=\"mt-3\">\n <label class=\"form-label fw-bold small\">Stack Trace:</label>\n <pre class=\"bg-light p-3 rounded small mb-0\"><code>{{errorDetails}}</code></pre>\n </div>\n {{/errorDetails}}\n </div>\n </div>\n {{/error}}\n\n <!-- Performance Metrics -->\n {{#metrics}}\n <div class=\"card mb-3\">\n <div class=\"card-header py-2\">\n <h6 class=\"mb-0\">\n <i class=\"bi bi-speedometer2 me-2\"></i>Performance Metrics\n </h6>\n </div>\n <div class=\"card-body\">\n <div class=\"row\">\n <div class=\"col-md-3 col-6 mb-3\">\n <div class=\"text-center\">\n <div class=\"h5 mb-1 text-primary\">{{executionTime}}ms</div>\n <small class=\"text-muted\">Execution Time</small>\n </div>\n </div>\n <div class=\"col-md-3 col-6 mb-3\">\n <div class=\"text-center\">\n <div class=\"h5 mb-1 text-info\">{{memoryUsage}}MB</div>\n <small class=\"text-muted\">Memory Usage</small>\n </div>\n </div>\n <div class=\"col-md-3 col-6 mb-3\">\n <div class=\"text-center\">\n <div class=\"h5 mb-1 text-warning\">{{cpuUsage}}%</div>\n <small class=\"text-muted\">CPU Usage</small>\n </div>\n </div>\n <div class=\"col-md-3 col-6 mb-3\">\n <div class=\"text-center\">\n <div class=\"h5 mb-1 text-secondary\">{{retryCount}}</div>\n <small class=\"text-muted\">Retry Count</small>\n </div>\n </div>\n </div>\n </div>\n </div>\n {{/metrics}}\n\n <!-- Task Logs -->\n <div class=\"card\">\n <div class=\"card-header py-2 d-flex justify-content-between align-items-center\">\n <h6 class=\"mb-0\">\n <i class=\"bi bi-journal-text me-2\"></i>Task Logs\n </h6>\n <button class=\"btn btn-sm btn-outline-primary\" data-action=\"refresh-logs\">\n <i class=\"bi bi-arrow-clockwise\"></i> Refresh\n </button>\n </div>\n <div class=\"card-body p-0\">\n <div class=\"mojo-task-logs-container\" style=\"max-height: 300px; overflow-y: auto;\">\n {{#logs.length}}\n {{#logs}}\n <div class=\"log-entry p-3 border-bottom\">\n <div class=\"d-flex justify-content-between align-items-start\">\n <div class=\"log-message flex-grow-1\">\n <span class=\"badge bg-{{levelClass}} me-2\">{{level|uppercase}}</span>\n {{message}}\n </div>\n <small class=\"text-muted ms-3\">{{timestamp|time}}</small>\n </div>\n </div>\n {{/logs}}\n {{/logs.length}}\n {{^logs.length}}\n <div class=\"text-center text-muted py-5\">\n <i class=\"bi bi-journal fs-1 opacity-50\"></i>\n <p class=\"mb-0 mt-2\">No logs available for this task</p>\n </div>\n {{/logs.length}}\n </div>\n </div>\n </div>\n {{/task}}\n\n {{^task}}\n <div class=\"alert alert-warning\">\n <i class=\"bi bi-exclamation-triangle me-2\"></i>\n No task data available\n </div>\n {{/task}}\n </div>\n `;\n }\n\n async onInit() {\n if (this.task) {\n await this.prepareTaskData();\n await this.loadTaskLogs();\n await this.loadTaskMetrics();\n }\n }\n\n async prepareTaskData() {\n if (!this.task) return;\n\n // Status styling\n this.task.statusBadgeClass = this.getStatusBadgeClass(this.task.status);\n this.task.statusIcon = this.getStatusIcon(this.task.status);\n\n // Format data for display\n if (this.task.data && typeof this.task.data === 'object') {\n this.task.dataFormatted = JSON.stringify(this.task.data, null, 2);\n }\n\n // Check if task has expired\n if (this.task.expires) {\n this.task.expiresClass = this.task.expires * 1000 < Date.now() ? 'text-danger' : 'text-muted';\n }\n }\n\n getStatusBadgeClass(status) {\n const classes = {\n 'pending': 'bg-primary',\n 'running': 'bg-success', \n 'completed': 'bg-info',\n 'error': 'bg-danger',\n 'cancelled': 'bg-secondary',\n 'expired': 'bg-warning'\n };\n return classes[status] || 'bg-secondary';\n }\n\n getStatusIcon(status) {\n const icons = {\n 'pending': 'bi-hourglass',\n 'running': 'bi-arrow-repeat',\n 'completed': 'bi-check-circle',\n 'error': 'bi-x-octagon',\n 'cancelled': 'bi-x-circle',\n 'expired': 'bi-clock'\n };\n return icons[status] || 'bi-question-circle';\n }\n\n getLogLevelClass(level) {\n const classes = {\n 'debug': 'secondary',\n 'info': 'primary',\n 'warning': 'warning',\n 'error': 'danger'\n };\n return classes[level] || 'secondary';\n }\n\n async loadTaskLogs() {\n if (!this.task?.id) return;\n\n try {\n const response = await this.getApp().rest.GET(`/api/tasks/${this.task.id}/logs`);\n if (response.success && response.data.status) {\n this.logs = response.data.data.map(log => ({\n ...log,\n levelClass: this.getLogLevelClass(log.level)\n }));\n } else {\n this.logs = [];\n }\n } catch (error) {\n console.error('Failed to load task logs:', error);\n this.logs = [];\n }\n }\n\n async loadTaskMetrics() {\n if (!this.task?.id) return;\n\n try {\n const response = await this.getApp().rest.GET(`/api/tasks/${this.task.id}/metrics`);\n if (response.success && response.data.status) {\n this.metrics = response.data.data;\n }\n } catch (error) {\n console.error('Failed to load task metrics:', error);\n this.metrics = null;\n }\n }\n\n async setTask(task) {\n this.task = task;\n await this.prepareTaskData();\n await this.loadTaskLogs();\n await this.loadTaskMetrics();\n \n if (this.isMounted()) {\n await this.render();\n }\n }\n\n async onActionRefreshLogs(action, event, element) {\n if (!this.task?.id) return;\n\n try {\n element.disabled = true;\n const icon = element.querySelector('i');\n if (icon) icon.classList.add('spinning');\n\n await this.loadTaskLogs();\n await this.render();\n \n } catch (error) {\n console.error('Failed to refresh logs:', error);\n this.showError('Failed to refresh logs: ' + error.message);\n } finally {\n element.disabled = false;\n const icon = element.querySelector('i');\n if (icon) icon.classList.remove('spinning');\n }\n }\n\n // Static method for easy Dialog integration\n static async show(task, options = {}) {\n const view = new TaskDetailsView({ task });\n await view.onInit();\n\n const buttons = [];\n\n // Add action buttons based on task status\n if (['pending', 'running'].includes(task.status)) {\n buttons.push({\n text: 'Cancel Task',\n class: 'btn-outline-danger',\n action: async () => {\n if (confirm('Are you sure you want to cancel this task?')) {\n try {\n const response = await view.getApp().rest.POST(`/api/tasks/${task.id}/cancel`);\n if (response.success && response.data.status) {\n view.showSuccess('Task cancelled successfully');\n return { action: 'cancelled', task };\n } else {\n view.showError(response.data.error || 'Failed to cancel task');\n }\n } catch (error) {\n view.showError('Failed to cancel task: ' + error.message);\n }\n }\n return null;\n }\n });\n }\n\n if (task.status === 'error') {\n buttons.push({\n text: 'Retry Task',\n class: 'btn-outline-primary',\n action: async () => {\n try {\n const response = await view.getApp().rest.POST(`/api/tasks/${task.id}/retry`);\n if (response.success && response.data.status) {\n view.showSuccess('Task queued for retry');\n return { action: 'retried', task };\n } else {\n view.showError(response.data.error || 'Failed to retry task');\n }\n } catch (error) {\n view.showError('Failed to retry task: ' + error.message);\n }\n return null;\n }\n });\n }\n\n buttons.push({\n text: 'Clone Task',\n class: 'btn-outline-info',\n action: async () => {\n try {\n const response = await view.getApp().rest.POST(`/api/tasks/${task.id}/clone`);\n if (response.success && response.data.status) {\n view.showSuccess('Task cloned successfully');\n return { action: 'cloned', originalTask: task, newTask: response.data.data };\n } else {\n view.showError(response.data.error || 'Failed to clone task');\n }\n } catch (error) {\n view.showError('Failed to clone task: ' + error.message);\n }\n return null;\n }\n });\n\n buttons.push({\n text: 'Export',\n class: 'btn-outline-secondary',\n action: () => {\n try {\n const exportData = {\n task: task,\n logs: view.logs,\n metrics: view.metrics,\n exported_at: new Date().toISOString(),\n exported_by: 'task-management-system'\n };\n\n const blob = new Blob([JSON.stringify(exportData, null, 2)], {\n type: 'application/json'\n });\n \n const url = URL.createObjectURL(blob);\n const a = document.createElement('a');\n a.href = url;\n a.download = `task-${task.id}-${Date.now()}.json`;\n document.body.appendChild(a);\n a.click();\n document.body.removeChild(a);\n URL.revokeObjectURL(url);\n\n view.showSuccess('Task data exported successfully');\n return null;\n } catch (error) {\n view.showError('Failed to export task data');\n return null;\n }\n }\n });\n\n buttons.push({\n text: 'Close',\n class: 'btn-secondary',\n dismiss: true\n });\n\n return await Dialog.showDialog({\n title: `<i class=\"bi bi-info-circle me-2\"></i>Task Details - ${task.id}`,\n body: view,\n size: 'lg',\n scrollable: true,\n buttons: buttons,\n ...options\n });\n }\n}","/**\n * RunnerDetailsView\n *\n * Tabbed runner inspector opened via RunnerDetailsView.show(runner).\n * All tab view classes are internal to this module.\n *\n * Tabs:\n * Overview — identity, channels, uptime (runner object only, no fetch)\n * System Info — OS, CPU, memory, disk, network (GET /api/jobs/runners/sysinfo/<id>)\n * Running Jobs — jobs on this runner (GET /api/jobs/job?runner_id=&status=running)\n * Logs — aggregated logs from running jobs (GET /api/jobs/logs?job_id=)\n * Actions — ping, shutdown, broadcast, export\n */\n\nimport View from '@core/View.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\nimport TabView from '@core/views/navigation/TabView.js';\nimport { JobRunner } from '@core/models/JobRunner.js';\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Shared helpers\n// ─────────────────────────────────────────────────────────────────────────────\n\nfunction formatUptime(seconds) {\n const d = Math.floor(seconds / 86400);\n const h = Math.floor((seconds % 86400) / 3600);\n const m = Math.floor((seconds % 3600) / 60);\n if (d > 0) return `${d}d ${h}h ${m}m`;\n if (h > 0) return `${h}h ${m}m`;\n return `${m}m`;\n}\n\nfunction formatDuration(seconds) {\n if (seconds < 60) return `${Math.round(seconds)}s`;\n if (seconds < 3600) return `${Math.round(seconds / 60)}m ${Math.round(seconds % 60)}s`;\n return `${Math.round(seconds / 3600)}h ${Math.round((seconds % 3600) / 60)}m`;\n}\n\nfunction formatBytes(bytes) {\n if (bytes == null) return 'N/A';\n if (bytes >= 1e9) return (bytes / 1e9).toFixed(2) + ' GB';\n if (bytes >= 1e6) return (bytes / 1e6).toFixed(2) + ' MB';\n if (bytes >= 1e3) return (bytes / 1e3).toFixed(2) + ' KB';\n return bytes + ' B';\n}\n\nfunction resourceBadgeClass(pct) {\n if (pct >= 80) return 'bg-danger-subtle text-danger';\n if (pct >= 60) return 'bg-warning-subtle text-warning';\n return 'bg-success-subtle text-success';\n}\n\nfunction progressBarClass(pct) {\n if (pct >= 80) return 'bg-danger';\n if (pct >= 60) return 'bg-warning';\n return 'bg-success';\n}\n\nfunction heartbeatAgeSec(isoString) {\n if (!isoString) return null;\n return (Date.now() - new Date(isoString).getTime()) / 1000;\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// RunnerOverviewTab\n// ─────────────────────────────────────────────────────────────────────────────\n\nclass RunnerOverviewTab extends View {\n constructor(options = {}) {\n super({ className: 'runner-overview-tab', ...options });\n this.model = options.model || null;\n }\n\n async onBeforeRender() {\n const r = this.model;\n if (!r) return;\n\n this.aliveBadgeClass = r.get('alive') ? 'bg-success-subtle text-success' : 'bg-danger-subtle text-danger';\n this.aliveIcon = r.get('alive') ? 'bi-check-circle-fill' : 'bi-x-circle-fill';\n this.aliveText = r.get('alive') ? 'Alive' : 'Dead';\n\n const started = r.get('started');\n this.startedText = started ? new Date(started).toLocaleString() : 'N/A';\n\n if (started) {\n const sec = (Date.now() - new Date(started).getTime()) / 1000;\n this.uptimeText = formatUptime(sec);\n } else {\n this.uptimeText = 'N/A';\n }\n\n const ageSec = heartbeatAgeSec(r.get('last_heartbeat'));\n if (ageSec !== null) {\n this.heartbeatText = new Date(r.get('last_heartbeat')).toLocaleString();\n this.heartbeatAgeText = `${Math.round(ageSec)}s ago`;\n this.heartbeatClass = ageSec < 30 ? 'text-success' : ageSec < 120 ? 'text-warning' : 'text-danger';\n } else {\n this.heartbeatText = 'N/A';\n this.heartbeatAgeText = '';\n this.heartbeatClass = 'text-muted';\n }\n\n const jobsProcessed = r.get('jobs_processed') || 0;\n const jobsFailed = r.get('jobs_failed') || 0;\n this.errorRate = (jobsProcessed > 0)\n ? ((jobsFailed / jobsProcessed) * 100).toFixed(2) + '%'\n : '0.00%';\n }\n\n async getTemplate() {\n return `\n {{^model}}\n <div class=\"alert alert-warning\"><i class=\"bi bi-exclamation-triangle me-2\"></i>No runner data available.</div>\n {{/model}}\n\n {{#model}}\n <div class=\"card border-0 shadow-sm mb-3\">\n <div class=\"card-header bg-white border-bottom py-2 d-flex justify-content-between align-items-center\">\n <h6 class=\"mb-0 fw-semibold\">\n <i class=\"bi bi-info-circle text-primary me-2\"></i>Identity\n </h6>\n <span class=\"badge {{aliveBadgeClass}}\">\n <i class=\"bi {{aliveIcon}} me-1\"></i>{{aliveText}}\n </span>\n </div>\n <div class=\"card-body\">\n <div class=\"row g-3\">\n <div class=\"col-md-6\">\n <table class=\"table table-sm table-borderless mb-0\">\n <tbody>\n <tr>\n <td class=\"text-muted small fw-semibold text-uppercase pe-3\" style=\"width:38%;white-space:nowrap;font-size:0.72rem;\">Runner ID</td>\n <td class=\"font-monospace small\">{{model.runner_id}}</td>\n </tr>\n <tr>\n <td class=\"text-muted small fw-semibold text-uppercase\" style=\"font-size:0.72rem;\">Started</td>\n <td class=\"small\">{{startedText}}</td>\n </tr>\n <tr>\n <td class=\"text-muted small fw-semibold text-uppercase\" style=\"font-size:0.72rem;\">Uptime</td>\n <td class=\"small fw-semibold\">{{uptimeText}}</td>\n </tr>\n <tr>\n <td class=\"text-muted small fw-semibold text-uppercase\" style=\"font-size:0.72rem;\">Heartbeat</td>\n <td class=\"small {{heartbeatClass}}\">\n {{heartbeatText}}\n {{#heartbeatAgeText}}\n <span class=\"text-muted\">({{heartbeatAgeText}})</span>\n {{/heartbeatAgeText}}\n </td>\n </tr>\n </tbody>\n </table>\n </div>\n <div class=\"col-md-6\">\n <table class=\"table table-sm table-borderless mb-0\">\n <tbody>\n <tr>\n <td class=\"text-muted small fw-semibold text-uppercase pe-3\" style=\"width:38%;white-space:nowrap;font-size:0.72rem;\">Jobs Done</td>\n <td class=\"small fw-bold text-success\">{{model.jobs_processed|number}}</td>\n </tr>\n <tr>\n <td class=\"text-muted small fw-semibold text-uppercase\" style=\"font-size:0.72rem;\">Jobs Failed</td>\n <td class=\"small fw-bold text-danger\">{{model.jobs_failed|number}}</td>\n </tr>\n <tr>\n <td class=\"text-muted small fw-semibold text-uppercase\" style=\"font-size:0.72rem;\">Error Rate</td>\n <td class=\"small\">{{errorRate}}</td>\n </tr>\n </tbody>\n </table>\n </div>\n </div>\n\n {{#model.channels.length}}\n <div class=\"mt-3 pt-3 border-top\">\n <div class=\"text-muted small fw-semibold text-uppercase mb-2\" style=\"font-size:0.72rem;\">Assigned Channels</div>\n <div class=\"d-flex flex-wrap gap-2\">\n {{#model.channels}}\n <span class=\"badge bg-primary-subtle text-primary px-3 py-2\">\n <i class=\"bi bi-circle-fill me-1\" style=\"font-size:0.4rem;vertical-align:middle;\"></i>{{.}}\n </span>\n {{/model.channels}}\n </div>\n </div>\n {{/model.channels.length}}\n </div>\n </div>\n\n <div class=\"alert alert-light border d-flex align-items-center gap-2 mb-0\" style=\"font-size:0.83rem;\">\n <i class=\"bi bi-cpu text-primary flex-shrink-0\"></i>\n <span>CPU, memory, disk, and network detail is in the <strong>System Info</strong> tab.</span>\n </div>\n {{/model}}\n `;\n }\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// RunnerSysinfoTab\n// ─────────────────────────────────────────────────────────────────────────────\n\nclass RunnerSysinfoTab extends View {\n constructor(options = {}) {\n super({ className: 'runner-sysinfo-tab', ...options });\n this.model = options.model || null;\n this.sysinfo = null;\n this.sysinfoError = null;\n this.loading = false;\n this.loaded = false;\n }\n\n async onTabActivated() {\n if (this.loaded) return;\n this.loaded = true;\n this.loading = true;\n this.sysinfoError = null;\n await this.render();\n await this.loadSysinfo();\n this.loading = false;\n await this.render();\n }\n\n async loadSysinfo() {\n try {\n const resp = await this.getApp().rest.GET(\n `/api/jobs/runners/sysinfo/${this.model.get('runner_id')}`\n );\n if (resp.success && resp.data) {\n const payload = resp.data.data || resp.data;\n if (payload && payload.status === 'error') {\n this.sysinfoError = payload.error || 'Runner reported an error collecting sysinfo.';\n return;\n }\n if (!resp.data.status) {\n this.sysinfoError = resp.data.error || 'Could not load system info.';\n return;\n }\n this.sysinfo = payload.result || payload;\n this.enrichSysinfo();\n } else {\n this.sysinfoError = 'Could not load system info.';\n }\n } catch (e) {\n this.sysinfoError = e.message || 'Request failed.';\n }\n }\n\n enrichSysinfo() {\n const s = this.sysinfo;\n if (!s) return;\n\n if (s.memory) {\n s.memory.totalFmt = formatBytes(s.memory.total);\n s.memory.usedFmt = formatBytes(s.memory.used);\n s.memory.availableFmt = formatBytes(s.memory.available);\n s.memory.barClass = progressBarClass(s.memory.percent);\n s.memory.badgeClass = resourceBadgeClass(s.memory.percent);\n }\n\n if (s.disk) {\n s.disk.totalFmt = formatBytes(s.disk.total);\n s.disk.usedFmt = formatBytes(s.disk.used);\n s.disk.freeFmt = formatBytes(s.disk.free);\n s.disk.barClass = progressBarClass(s.disk.percent);\n s.disk.badgeClass = resourceBadgeClass(s.disk.percent);\n }\n\n if (s.network) {\n s.network.bytesRecvFmt = formatBytes(s.network.bytes_recv);\n s.network.bytesSentFmt = formatBytes(s.network.bytes_sent);\n s.network.errClass = (s.network.errin > 0 || s.network.errout > 0)\n ? 'text-danger fw-bold' : 'text-success';\n s.network.dropClass = (s.network.dropin > 0 || s.network.dropout > 0)\n ? 'text-warning fw-bold' : 'text-success';\n }\n\n const cpuPct = s.cpu_load || 0;\n s.cpuLoadBarClass = progressBarClass(cpuPct);\n s.cpuLoadBadgeClass = resourceBadgeClass(cpuPct);\n\n if (s.cpu && s.cpu.freq) {\n s.cpu.freqText = `${Math.round(s.cpu.freq.current).toLocaleString()} MHz current`\n + ` · ${Math.round(s.cpu.freq.max).toLocaleString()} MHz max`;\n } else if (s.cpu) {\n s.cpu.freqText = null;\n }\n\n if (s.cpus_load && s.cpus_load.length) {\n s.cpuCores = s.cpus_load.map((pct, i) => ({\n index: i,\n pct: pct.toFixed(1),\n barClass: progressBarClass(pct)\n }));\n } else {\n s.cpuCores = [];\n }\n\n s.bootDatetime = s.boot_time ? new Date(s.boot_time * 1000).toLocaleString() : null;\n s.collectedText = s.datetime || null;\n }\n\n async onActionRefreshSysinfo() {\n this.loaded = false;\n this.sysinfo = null;\n this.sysinfoError = null;\n await this.onTabActivated();\n }\n\n async getTemplate() {\n return `\n {{#loading|bool}}\n <div class=\"text-center py-5\">\n <div class=\"spinner-border text-primary\" role=\"status\"></div>\n <div class=\"mt-2 text-muted small\">Loading system info…</div>\n </div>\n {{/loading|bool}}\n\n {{^loading|bool}}\n\n {{#sysinfoError|bool}}\n <div class=\"alert alert-warning d-flex align-items-start gap-2\">\n <i class=\"bi bi-exclamation-triangle flex-shrink-0 mt-1\"></i>\n <div>\n <strong>Could not load system info</strong><br>\n <span class=\"small\">{{sysinfoError}}</span><br>\n <button class=\"btn btn-sm btn-outline-warning mt-2\" data-action=\"refresh-sysinfo\">\n <i class=\"bi bi-arrow-clockwise me-1\"></i>Retry\n </button>\n </div>\n </div>\n {{/sysinfoError|bool}}\n\n {{#sysinfo|bool}}\n\n <div class=\"d-flex justify-content-end align-items-center gap-3 mb-3\">\n {{#sysinfo.collectedText}}\n <small class=\"text-muted\">Collected {{sysinfo.collectedText}}</small>\n {{/sysinfo.collectedText}}\n <button class=\"btn btn-sm btn-outline-secondary\" data-action=\"refresh-sysinfo\">\n <i class=\"bi bi-arrow-clockwise me-1\"></i>Refresh\n </button>\n </div>\n\n <!-- OS -->\n <div class=\"card border-0 shadow-sm mb-3\">\n <div class=\"card-header bg-white border-bottom py-2\">\n <h6 class=\"mb-0 fw-semibold\"><i class=\"bi bi-hdd-rack text-primary me-2\"></i>Operating System</h6>\n </div>\n <div class=\"card-body p-0\">\n <table class=\"table table-sm table-borderless mb-0\">\n <tbody>\n {{#sysinfo.os}}\n <tr>\n <td class=\"text-muted small fw-semibold text-uppercase ps-3 pe-2\" style=\"width:20%;font-size:0.72rem;white-space:nowrap;\">Hostname</td>\n <td class=\"font-monospace small\">{{.hostname}}</td>\n <td class=\"text-muted small fw-semibold text-uppercase pe-2\" style=\"width:15%;font-size:0.72rem;white-space:nowrap;\">OS</td>\n <td class=\"small\">{{.system}}</td>\n </tr>\n <tr>\n <td class=\"text-muted small fw-semibold text-uppercase ps-3 pe-2\" style=\"font-size:0.72rem;\">Release</td>\n <td class=\"font-monospace small\">{{.release}}</td>\n <td class=\"text-muted small fw-semibold text-uppercase pe-2\" style=\"font-size:0.72rem;\">Machine</td>\n <td class=\"font-monospace small\">{{.machine}}</td>\n </tr>\n <tr>\n <td class=\"text-muted small fw-semibold text-uppercase ps-3 pe-2\" style=\"font-size:0.72rem;\">Version</td>\n <td colspan=\"3\" class=\"font-monospace small text-muted\" style=\"font-size:0.76rem;\">{{.version}}</td>\n </tr>\n {{/sysinfo.os}}\n {{#sysinfo.bootDatetime}}\n <tr>\n <td class=\"text-muted small fw-semibold text-uppercase ps-3 pe-2\" style=\"font-size:0.72rem;\">Boot Time</td>\n <td colspan=\"3\" class=\"small\">{{sysinfo.bootDatetime}}</td>\n </tr>\n {{/sysinfo.bootDatetime}}\n </tbody>\n </table>\n </div>\n </div>\n\n <!-- CPU -->\n <div class=\"card border-0 shadow-sm mb-3\">\n <div class=\"card-header bg-white border-bottom py-2 d-flex justify-content-between align-items-center\">\n <h6 class=\"mb-0 fw-semibold\"><i class=\"bi bi-cpu text-primary me-2\"></i>CPU</h6>\n <span class=\"badge {{sysinfo.cpuLoadBadgeClass}}\">{{sysinfo.cpu_load}}% overall</span>\n </div>\n <div class=\"card-body\">\n <div class=\"d-flex justify-content-between mb-1\">\n <small class=\"fw-semibold text-muted\">Overall Load</small>\n <small class=\"fw-bold\">{{sysinfo.cpu_load}}%</small>\n </div>\n <div class=\"progress mb-2\" style=\"height:8px;\">\n <div class=\"progress-bar {{sysinfo.cpuLoadBarClass}}\" style=\"width:{{sysinfo.cpu_load}}%;\"></div>\n </div>\n {{#sysinfo.cpu}}\n <div class=\"text-muted small mb-3\">\n {{.count}} logical cores\n {{#.freqText}}&nbsp;·&nbsp;{{.freqText}}{{/.freqText}}\n </div>\n {{/sysinfo.cpu}}\n\n {{#sysinfo.cpuCores.length}}\n <div class=\"row g-2\">\n {{#sysinfo.cpuCores}}\n <div class=\"col-6 col-md-3\">\n <div class=\"border rounded p-2 bg-light text-center\">\n <div class=\"text-muted fw-semibold text-uppercase mb-1\" style=\"font-size:0.65rem;\">Core {{.index}}</div>\n <div class=\"fw-bold small\">{{.pct}}%</div>\n <div class=\"progress mt-1\" style=\"height:4px;\">\n <div class=\"progress-bar {{.barClass}}\" style=\"width:{{.pct}}%;\"></div>\n </div>\n </div>\n </div>\n {{/sysinfo.cpuCores}}\n </div>\n {{/sysinfo.cpuCores.length}}\n </div>\n </div>\n\n <!-- Memory -->\n {{#sysinfo.memory}}\n <div class=\"card border-0 shadow-sm mb-3\">\n <div class=\"card-header bg-white border-bottom py-2 d-flex justify-content-between align-items-center\">\n <h6 class=\"mb-0 fw-semibold\"><i class=\"bi bi-memory text-primary me-2\"></i>Memory</h6>\n <span class=\"badge {{.badgeClass}}\">{{.percent}}% used</span>\n </div>\n <div class=\"card-body\">\n <div class=\"d-flex justify-content-between mb-1\">\n <small class=\"fw-semibold text-muted\">RAM Usage</small>\n <small class=\"fw-bold\">{{.usedFmt}} / {{.totalFmt}}</small>\n </div>\n <div class=\"progress mb-3\" style=\"height:8px;\">\n <div class=\"progress-bar {{.barClass}}\" style=\"width:{{.percent}}%;\"></div>\n </div>\n <div class=\"row g-2 text-center\">\n <div class=\"col-6 col-md-3\">\n <div class=\"text-muted fw-semibold text-uppercase\" style=\"font-size:0.7rem;\">Total</div>\n <div class=\"fw-semibold small\">{{.totalFmt}}</div>\n </div>\n <div class=\"col-6 col-md-3\">\n <div class=\"text-muted fw-semibold text-uppercase\" style=\"font-size:0.7rem;\">Used</div>\n <div class=\"fw-semibold small\">{{.usedFmt}}</div>\n </div>\n <div class=\"col-6 col-md-3\">\n <div class=\"text-muted fw-semibold text-uppercase\" style=\"font-size:0.7rem;\">Available</div>\n <div class=\"fw-semibold small text-success\">{{.availableFmt}}</div>\n </div>\n <div class=\"col-6 col-md-3\">\n <div class=\"text-muted fw-semibold text-uppercase\" style=\"font-size:0.7rem;\">Percent</div>\n <div class=\"fw-semibold small\">{{.percent}}%</div>\n </div>\n </div>\n </div>\n </div>\n {{/sysinfo.memory}}\n\n <!-- Disk -->\n {{#sysinfo.disk}}\n <div class=\"card border-0 shadow-sm mb-3\">\n <div class=\"card-header bg-white border-bottom py-2 d-flex justify-content-between align-items-center\">\n <h6 class=\"mb-0 fw-semibold\"><i class=\"bi bi-hdd text-primary me-2\"></i>Disk (Root)</h6>\n <span class=\"badge {{.badgeClass}}\">{{.percent}}% used</span>\n </div>\n <div class=\"card-body\">\n <div class=\"d-flex justify-content-between mb-1\">\n <small class=\"fw-semibold text-muted\">Disk Usage</small>\n <small class=\"fw-bold\">{{.usedFmt}} / {{.totalFmt}}</small>\n </div>\n <div class=\"progress mb-3\" style=\"height:8px;\">\n <div class=\"progress-bar {{.barClass}}\" style=\"width:{{.percent}}%;\"></div>\n </div>\n <div class=\"row g-2 text-center\">\n <div class=\"col-6 col-md-3\">\n <div class=\"text-muted fw-semibold text-uppercase\" style=\"font-size:0.7rem;\">Total</div>\n <div class=\"fw-semibold small\">{{.totalFmt}}</div>\n </div>\n <div class=\"col-6 col-md-3\">\n <div class=\"text-muted fw-semibold text-uppercase\" style=\"font-size:0.7rem;\">Used</div>\n <div class=\"fw-semibold small\">{{.usedFmt}}</div>\n </div>\n <div class=\"col-6 col-md-3\">\n <div class=\"text-muted fw-semibold text-uppercase\" style=\"font-size:0.7rem;\">Free</div>\n <div class=\"fw-semibold small text-success\">{{.freeFmt}}</div>\n </div>\n <div class=\"col-6 col-md-3\">\n <div class=\"text-muted fw-semibold text-uppercase\" style=\"font-size:0.7rem;\">Percent</div>\n <div class=\"fw-semibold small\">{{.percent}}%</div>\n </div>\n </div>\n </div>\n </div>\n {{/sysinfo.disk}}\n\n <!-- Network -->\n {{#sysinfo.network}}\n <div class=\"card border-0 shadow-sm mb-3\">\n <div class=\"card-header bg-white border-bottom py-2\">\n <h6 class=\"mb-0 fw-semibold\"><i class=\"bi bi-diagram-3 text-primary me-2\"></i>Network</h6>\n </div>\n <div class=\"card-body\">\n <div class=\"row g-2\">\n <div class=\"col-6 col-md-4\">\n <div class=\"border rounded p-2 bg-light\">\n <div class=\"text-muted fw-semibold text-uppercase mb-1\" style=\"font-size:0.67rem;\"><i class=\"bi bi-arrow-down text-primary me-1\"></i>Bytes Recv</div>\n <div class=\"fw-bold font-monospace small\">{{.bytesRecvFmt}}</div>\n </div>\n </div>\n <div class=\"col-6 col-md-4\">\n <div class=\"border rounded p-2 bg-light\">\n <div class=\"text-muted fw-semibold text-uppercase mb-1\" style=\"font-size:0.67rem;\"><i class=\"bi bi-arrow-up text-primary me-1\"></i>Bytes Sent</div>\n <div class=\"fw-bold font-monospace small\">{{.bytesSentFmt}}</div>\n </div>\n </div>\n <div class=\"col-6 col-md-4\">\n <div class=\"border rounded p-2 bg-light\">\n <div class=\"text-muted fw-semibold text-uppercase mb-1\" style=\"font-size:0.67rem;\"><i class=\"bi bi-share text-primary me-1\"></i>TCP Connections</div>\n <div class=\"fw-bold font-monospace small\">{{.tcp_cons|number}}</div>\n </div>\n </div>\n <div class=\"col-6 col-md-4\">\n <div class=\"border rounded p-2 bg-light\">\n <div class=\"text-muted fw-semibold text-uppercase mb-1\" style=\"font-size:0.67rem;\"><i class=\"bi bi-arrow-down me-1\"></i>Packets Recv</div>\n <div class=\"fw-bold font-monospace small\">{{.packets_recv|number}}</div>\n </div>\n </div>\n <div class=\"col-6 col-md-4\">\n <div class=\"border rounded p-2 bg-light\">\n <div class=\"text-muted fw-semibold text-uppercase mb-1\" style=\"font-size:0.67rem;\"><i class=\"bi bi-arrow-up me-1\"></i>Packets Sent</div>\n <div class=\"fw-bold font-monospace small\">{{.packets_sent|number}}</div>\n </div>\n </div>\n <div class=\"col-6 col-md-4\">\n <div class=\"border rounded p-2 bg-light\">\n <div class=\"text-muted fw-semibold text-uppercase mb-1\" style=\"font-size:0.67rem;\"><i class=\"bi bi-exclamation-triangle me-1\"></i>Errors In / Out</div>\n <div class=\"fw-bold font-monospace small {{.errClass}}\">{{.errin}} / {{.errout}}</div>\n </div>\n </div>\n <div class=\"col-6 col-md-4\">\n <div class=\"border rounded p-2 bg-light\">\n <div class=\"text-muted fw-semibold text-uppercase mb-1\" style=\"font-size:0.67rem;\"><i class=\"bi bi-x-circle me-1\"></i>Drops In</div>\n <div class=\"fw-bold font-monospace small {{.dropClass}}\">{{.dropin}}</div>\n </div>\n </div>\n <div class=\"col-6 col-md-4\">\n <div class=\"border rounded p-2 bg-light\">\n <div class=\"text-muted fw-semibold text-uppercase mb-1\" style=\"font-size:0.67rem;\"><i class=\"bi bi-x-circle me-1\"></i>Drops Out</div>\n <div class=\"fw-bold font-monospace small {{.dropClass}}\">{{.dropout}}</div>\n </div>\n </div>\n </div>\n </div>\n </div>\n {{/sysinfo.network}}\n\n <!-- Logged-in Users -->\n <div class=\"card border-0 shadow-sm\">\n <div class=\"card-header bg-white border-bottom py-2\">\n <h6 class=\"mb-0 fw-semibold\"><i class=\"bi bi-person-badge text-primary me-2\"></i>Logged-in Users</h6>\n </div>\n {{#sysinfo.users.length}}\n <ul class=\"list-group list-group-flush\">\n {{#sysinfo.users}}\n <li class=\"list-group-item font-monospace small\">{{.name|default:'unknown'}}\n {{#.terminal}}<span class=\"text-muted ms-2\">{{.terminal}}</span>{{/.terminal}}\n </li>\n {{/sysinfo.users}}\n </ul>\n {{/sysinfo.users.length}}\n {{^sysinfo.users.length}}\n <div class=\"card-body text-center text-muted py-4\">\n <i class=\"bi bi-person-x fs-2 d-block mb-2 opacity-50\"></i>\n <div class=\"small\">No users currently logged in</div>\n </div>\n {{/sysinfo.users.length}}\n </div>\n\n {{/sysinfo|bool}}\n {{/loading|bool}}\n `;\n }\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// RunnerJobsTab\n// ─────────────────────────────────────────────────────────────────────────────\n\nclass RunnerJobsTab extends View {\n constructor(options = {}) {\n super({ className: 'runner-jobs-tab', ...options });\n this.model = options.model || null;\n this.jobs = [];\n this.loading = false;\n this.loaded = false;\n }\n\n async onTabActivated() {\n if (this.loaded) return;\n this.loaded = true;\n this.loading = true;\n await this.render();\n await this.loadJobs();\n this.loading = false;\n await this.render();\n }\n\n async loadJobs() {\n try {\n const resp = await this.getApp().rest.GET(\n `/api/jobs/job?runner_id=${this.model.get('runner_id')}&status=running&size=50`\n );\n if (resp.success && resp.data && resp.data.status) {\n const now = Date.now() / 1000;\n this.jobs = (resp.data.data || []).map(job => ({\n ...job,\n durationText: job.started_at\n ? formatDuration(now - new Date(job.started_at).getTime() / 1000)\n : 'N/A',\n startedText: job.started_at\n ? new Date(job.started_at).toLocaleTimeString()\n : 'N/A',\n attemptBadgeClass: job.attempt > 1\n ? 'bg-danger-subtle text-danger'\n : 'bg-warning-subtle text-warning'\n }));\n } else {\n this.jobs = [];\n }\n } catch (e) {\n this.jobs = [];\n this.showError('Could not load running jobs: ' + e.message);\n }\n }\n\n async onActionRefreshJobs() {\n this.loaded = false;\n this.jobs = [];\n await this.onTabActivated();\n }\n\n async onActionViewJob(event, element) {\n const jobId = element.dataset.jobId;\n this.emit('job:view', { jobId, runner: this.model });\n }\n\n async onActionCancelJob(event, element) {\n const jobId = element.dataset.jobId;\n const ok = await Dialog.confirm(\n 'Cancel this job? The runner will receive a cooperative cancel signal.',\n 'Cancel Job',\n { confirmText: 'Cancel Job', confirmClass: 'btn-warning' }\n );\n if (!ok) return;\n\n try {\n const resp = await this.getApp().rest.POST(\n `/api/jobs/job/${jobId}`, { cancel_request: true }\n );\n if (resp.success && resp.data && resp.data.status) {\n this.showSuccess('Cancel signal sent.');\n this.loaded = false;\n await this.onTabActivated();\n } else {\n this.showError((resp.data && resp.data.error) || 'Could not cancel job.');\n }\n } catch (e) {\n this.showError('Could not cancel job: ' + e.message);\n }\n }\n\n async getTemplate() {\n return `\n {{#loading|bool}}\n <div class=\"text-center py-5\">\n <div class=\"spinner-border text-primary\" role=\"status\"></div>\n <div class=\"mt-2 text-muted small\">Loading running jobs…</div>\n </div>\n {{/loading|bool}}\n\n {{^loading|bool}}\n <div class=\"d-flex justify-content-between align-items-center mb-3\">\n <small class=\"text-muted\">{{jobs.length}} job(s) currently executing on this runner</small>\n <button class=\"btn btn-sm btn-outline-secondary\" data-action=\"refresh-jobs\">\n <i class=\"bi bi-arrow-clockwise me-1\"></i>Refresh\n </button>\n </div>\n\n {{#jobs.length}}\n <div class=\"card border-0 shadow-sm\">\n <div class=\"table-responsive\">\n <table class=\"table table-sm table-hover align-middle mb-0\">\n <thead class=\"table-light\">\n <tr>\n <th class=\"ps-3 border-0 text-muted fw-semibold text-uppercase\" style=\"font-size:0.72rem;letter-spacing:0.04em;\">Job ID</th>\n <th class=\"border-0 text-muted fw-semibold text-uppercase\" style=\"font-size:0.72rem;letter-spacing:0.04em;\">Function</th>\n <th class=\"border-0 text-muted fw-semibold text-uppercase\" style=\"font-size:0.72rem;letter-spacing:0.04em;\">Channel</th>\n <th class=\"border-0 text-muted fw-semibold text-uppercase\" style=\"font-size:0.72rem;letter-spacing:0.04em;\">Started</th>\n <th class=\"border-0 text-muted fw-semibold text-uppercase\" style=\"font-size:0.72rem;letter-spacing:0.04em;\">Duration</th>\n <th class=\"border-0 text-muted fw-semibold text-uppercase\" style=\"font-size:0.72rem;letter-spacing:0.04em;\">Attempt</th>\n <th class=\"border-0 text-end pe-3 text-muted fw-semibold text-uppercase\" style=\"font-size:0.72rem;letter-spacing:0.04em;\">Actions</th>\n </tr>\n </thead>\n <tbody>\n {{#jobs}}\n <tr>\n <td class=\"ps-3\">\n <span class=\"font-monospace text-primary small\" title=\"{{.id}}\">{{.id|truncate:12}}</span>\n </td>\n <td>\n <span class=\"font-monospace text-muted small\" title=\"{{.func}}\">{{.func|truncate:42}}</span>\n </td>\n <td>\n <span class=\"badge bg-primary-subtle text-primary\">{{.channel}}</span>\n </td>\n <td><small class=\"text-muted\">{{.startedText}}</small></td>\n <td><span class=\"badge bg-light text-secondary border\">{{.durationText}}</span></td>\n <td><span class=\"badge {{.attemptBadgeClass}}\">{{.attempt}}</span></td>\n <td class=\"text-end pe-3\">\n <div class=\"btn-group btn-group-sm\">\n <button class=\"btn btn-outline-primary btn-sm\" data-action=\"view-job\"\n data-job-id=\"{{.id}}\" title=\"View job details\">\n <i class=\"bi bi-eye\"></i>\n </button>\n <button class=\"btn btn-outline-warning btn-sm\" data-action=\"cancel-job\"\n data-job-id=\"{{.id}}\" title=\"Cancel job\">\n <i class=\"bi bi-x-circle\"></i>\n </button>\n </div>\n </td>\n </tr>\n {{/jobs}}\n </tbody>\n </table>\n </div>\n </div>\n {{/jobs.length}}\n\n {{^jobs.length}}\n <div class=\"text-center text-muted py-5\">\n <i class=\"bi bi-list-task fs-2 d-block mb-2 opacity-50\"></i>\n <div class=\"small\">No jobs currently executing on this runner</div>\n </div>\n {{/jobs.length}}\n {{/loading|bool}}\n `;\n }\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// RunnerLogsTab\n// ─────────────────────────────────────────────────────────────────────────────\n\nclass RunnerLogsTab extends View {\n constructor(options = {}) {\n super({ className: 'runner-logs-tab', ...options });\n this.model = options.model || null;\n this.logs = [];\n this.filteredLogs = [];\n this.logFilter = 'all';\n this.loading = false;\n this.loaded = false;\n // precomputed filter button classes\n this.filterAllClass = 'btn-primary';\n this.filterDebugClass = 'btn-outline-secondary';\n this.filterInfoClass = 'btn-outline-primary';\n this.filterWarnClass = 'btn-outline-warning';\n this.filterErrorClass = 'btn-outline-danger';\n }\n\n async onTabActivated() {\n if (this.loaded) return;\n this.loaded = true;\n this.loading = true;\n await this.render();\n await this.loadLogs();\n this.loading = false;\n await this.render();\n }\n\n async loadLogs() {\n try {\n // Get running job IDs for this runner first\n const jobsResp = await this.getApp().rest.GET(\n `/api/jobs/job?runner_id=${this.model.get('runner_id')}&status=running&size=50`\n );\n\n const jobIds = [];\n if (jobsResp.success && jobsResp.data && jobsResp.data.status) {\n (jobsResp.data.data || []).forEach(j => jobIds.push(j.id));\n }\n\n if (!jobIds.length) {\n this.logs = [];\n return;\n }\n\n // Fetch logs for each job in parallel (cap at 5 jobs)\n const promises = jobIds.slice(0, 5).map(id =>\n this.getApp().rest.GET(`/api/jobs/logs?job_id=${id}&sort=-created&size=20`)\n .then(r => (r.success && r.data && r.data.status) ? (r.data.data || []) : [])\n .catch(() => [])\n );\n\n const results = await Promise.all(promises);\n const all = [].concat(...results);\n\n all.sort((a, b) => new Date(b.created) - new Date(a.created));\n this.logs = all.slice(0, 50).map(log => ({\n ...log,\n levelBadgeClass: this.getLogLevelClass(log.kind),\n kindDisplay: (log.kind || 'info').toUpperCase(),\n createdText: new Date(log.created).toLocaleTimeString()\n }));\n } catch (e) {\n this.logs = [];\n this.showError('Could not load logs: ' + e.message);\n }\n }\n\n getLogLevelClass(kind) {\n const map = {\n debug: 'bg-secondary-subtle text-secondary',\n info: 'bg-primary-subtle text-primary',\n warn: 'bg-warning-subtle text-warning',\n error: 'bg-danger-subtle text-danger'\n };\n return map[kind] || 'bg-secondary-subtle text-secondary';\n }\n\n async onBeforeRender() {\n this.filteredLogs = this.logFilter === 'all'\n ? this.logs\n : this.logs.filter(l => l.kind === this.logFilter);\n\n this.filterAllClass = this.logFilter === 'all' ? 'btn-primary' : 'btn-outline-secondary';\n this.filterDebugClass = this.logFilter === 'debug' ? 'btn-secondary' : 'btn-outline-secondary';\n this.filterInfoClass = this.logFilter === 'info' ? 'btn-primary' : 'btn-outline-primary';\n this.filterWarnClass = this.logFilter === 'warn' ? 'btn-warning' : 'btn-outline-warning';\n this.filterErrorClass = this.logFilter === 'error' ? 'btn-danger' : 'btn-outline-danger';\n }\n\n async onActionFilterLogs(event, element) {\n this.logFilter = element.dataset.kind || 'all';\n await this.render();\n }\n\n async onActionRefreshLogs() {\n this.loaded = false;\n this.logs = [];\n this.logFilter = 'all';\n await this.onTabActivated();\n }\n\n async getTemplate() {\n return `\n {{#loading|bool}}\n <div class=\"text-center py-5\">\n <div class=\"spinner-border text-primary\" role=\"status\"></div>\n <div class=\"mt-2 text-muted small\">Loading logs…</div>\n </div>\n {{/loading|bool}}\n\n {{^loading|bool}}\n <div class=\"card border-0 shadow-sm\">\n <div class=\"card-header bg-white border-bottom py-2 d-flex align-items-center gap-2 flex-wrap\">\n <small class=\"text-muted fw-semibold me-1\">Filter:</small>\n <button class=\"btn btn-sm {{filterAllClass}}\" data-action=\"filter-logs\" data-kind=\"all\">All</button>\n <button class=\"btn btn-sm {{filterDebugClass}}\" data-action=\"filter-logs\" data-kind=\"debug\">Debug</button>\n <button class=\"btn btn-sm {{filterInfoClass}}\" data-action=\"filter-logs\" data-kind=\"info\">Info</button>\n <button class=\"btn btn-sm {{filterWarnClass}}\" data-action=\"filter-logs\" data-kind=\"warn\">Warning</button>\n <button class=\"btn btn-sm {{filterErrorClass}}\" data-action=\"filter-logs\" data-kind=\"error\">Error</button>\n <div class=\"ms-auto d-flex align-items-center gap-2\">\n <small class=\"text-muted\">{{filteredLogs.length}} entries</small>\n <button class=\"btn btn-sm btn-outline-secondary\" data-action=\"refresh-logs\">\n <i class=\"bi bi-arrow-clockwise\"></i>\n </button>\n </div>\n </div>\n\n <div style=\"max-height:420px;overflow-y:auto;\">\n {{#filteredLogs.length}}\n {{#filteredLogs}}\n <div class=\"d-flex align-items-start gap-2 px-3 py-2 border-bottom font-monospace\" style=\"font-size:0.78rem;\">\n <span class=\"text-muted flex-shrink-0 pt-1\" style=\"min-width:65px;\">{{.createdText}}</span>\n <span class=\"badge {{.levelBadgeClass}} flex-shrink-0\" style=\"margin-top:1px;\">{{.kindDisplay}}</span>\n <span class=\"flex-grow-1 text-break\">{{.message}}</span>\n </div>\n {{/filteredLogs}}\n {{/filteredLogs.length}}\n\n {{^filteredLogs.length}}\n <div class=\"text-center text-muted py-5\">\n <i class=\"bi bi-journal fs-2 d-block mb-2 opacity-50\"></i>\n <div class=\"small\">No log entries</div>\n </div>\n {{/filteredLogs.length}}\n </div>\n </div>\n {{/loading|bool}}\n `;\n }\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// RunnerActionsTab\n// ─────────────────────────────────────────────────────────────────────────────\n\nclass RunnerActionsTab extends View {\n constructor(options = {}) {\n super({ className: 'runner-actions-tab', ...options });\n this.model = options.model || null;\n this.pingResult = null;\n }\n\n async onActionPing() {\n this.pingResult = null;\n await this.render();\n try {\n const resp = await this.getApp().rest.POST('/api/jobs/runners/ping', {\n runner_id: this.model.get('runner_id'),\n timeout: 2.0\n });\n if (resp.success && resp.data) {\n this.pingResult = resp.data.responsive\n ? '<span class=\"text-success\"><i class=\"bi bi-check-circle-fill me-1\"></i>Runner is responsive</span>'\n : '<span class=\"text-warning\"><i class=\"bi bi-exclamation-triangle-fill me-1\"></i>Runner did not respond within 2s</span>';\n } else {\n this.pingResult = '<span class=\"text-danger\"><i class=\"bi bi-x-circle-fill me-1\"></i>Ping request failed</span>';\n }\n } catch (e) {\n this.pingResult = `<span class=\"text-danger\"><i class=\"bi bi-x-circle-fill me-1\"></i>${e.message}</span>`;\n }\n await this.render();\n }\n\n async onActionShutdown() {\n const ok = await Dialog.confirm(\n `Send a graceful shutdown to <strong class=\"font-monospace\">${this.model.get('runner_id')}</strong>?`\n + '<br><br>The runner will finish its current job then exit. This is fire-and-forget.',\n 'Shutdown Runner',\n { confirmText: 'Shutdown', confirmClass: 'btn-danger' }\n );\n if (!ok) return;\n\n try {\n const resp = await this.getApp().rest.POST('/api/jobs/runners/shutdown', {\n runner_id: this.model.get('runner_id'),\n graceful: true\n });\n if (resp.success && resp.data && resp.data.status) {\n this.showSuccess('Shutdown command sent to runner.');\n this.emit('runner:shutdown', { runner: this.model });\n } else {\n this.showError((resp.data && resp.data.error) || 'Shutdown command failed.');\n }\n } catch (e) {\n this.showError('Shutdown failed: ' + e.message);\n }\n }\n\n async onActionBroadcast() {\n const commandEl = this.element && this.element.querySelector('[data-field=\"broadcast-command\"]');\n const timeoutEl = this.element && this.element.querySelector('[data-field=\"broadcast-timeout\"]');\n const command = commandEl ? commandEl.value : 'status';\n const timeout = timeoutEl ? (parseFloat(timeoutEl.value) || 2.0) : 2.0;\n\n Dialog.showBusy({ message: `Broadcasting \"${command}\" to all runners…` });\n try {\n const resp = await this.getApp().rest.POST('/api/jobs/runners/broadcast', {\n command,\n timeout\n });\n Dialog.hideBusy();\n\n if (resp.success && resp.data) {\n await Dialog.showCode(\n JSON.stringify(resp.data, null, 2),\n 'json',\n { title: `Broadcast Response — ${command}`, size: 'lg' }\n );\n } else {\n this.showError((resp.data && resp.data.error) || 'Broadcast failed.');\n }\n } catch (e) {\n Dialog.hideBusy();\n this.showError('Broadcast failed: ' + e.message);\n }\n }\n\n async onActionExport() {\n try {\n const exportData = {\n runner: this.model.toJSON ? this.model.toJSON() : this.model,\n exported_at: new Date().toISOString()\n };\n const blob = new Blob([JSON.stringify(exportData, null, 2)], { type: 'application/json' });\n const url = URL.createObjectURL(blob);\n const a = Object.assign(document.createElement('a'), {\n href: url,\n download: `runner-${this.model.get('runner_id')}-${Date.now()}.json`\n });\n document.body.appendChild(a);\n a.click();\n document.body.removeChild(a);\n URL.revokeObjectURL(url);\n this.showSuccess('Runner data exported.');\n } catch (e) {\n this.showError('Export failed: ' + e.message);\n }\n }\n\n async getTemplate() {\n return `\n <p class=\"text-muted small mb-3\">\n <i class=\"bi bi-info-circle me-1\"></i>\n Actions operate on runner <strong class=\"font-monospace\">{{model.runner_id}}</strong> unless otherwise noted.\n </p>\n\n <div class=\"d-flex align-items-center gap-2 mb-3\">\n <span class=\"text-muted fw-semibold text-uppercase\" style=\"font-size:0.7rem;letter-spacing:0.09em;white-space:nowrap;\">Runner Control</span>\n <hr class=\"flex-grow-1 my-0\">\n </div>\n\n <div class=\"row g-3 mb-4\">\n\n <!-- Ping -->\n <div class=\"col-md-4\">\n <div class=\"card border-0 shadow-sm h-100\">\n <div class=\"card-body d-flex flex-column gap-3\">\n <div class=\"d-flex gap-3 align-items-start\">\n <div class=\"d-flex align-items-center justify-content-center rounded bg-success-subtle text-success flex-shrink-0\"\n style=\"width:40px;height:40px;font-size:1.1rem;\">\n <i class=\"bi bi-broadcast-pin\"></i>\n </div>\n <div>\n <div class=\"fw-semibold mb-1\">Ping Runner</div>\n <div class=\"text-muted small\">Verify this runner is truly responsive, not just alive on paper.</div>\n </div>\n </div>\n {{#pingResult|bool}}\n <div class=\"small\">{{{pingResult}}}</div>\n {{/pingResult|bool}}\n <button class=\"btn btn-sm btn-outline-success mt-auto\" data-action=\"ping\">\n <i class=\"bi bi-broadcast-pin me-1\"></i>Ping Now\n </button>\n </div>\n </div>\n </div>\n\n <!-- Shutdown -->\n <div class=\"col-md-4\">\n <div class=\"card border-0 shadow-sm h-100\">\n <div class=\"card-body d-flex flex-column gap-3\">\n <div class=\"d-flex gap-3 align-items-start\">\n <div class=\"d-flex align-items-center justify-content-center rounded bg-danger-subtle text-danger flex-shrink-0\"\n style=\"width:40px;height:40px;font-size:1.1rem;\">\n <i class=\"bi bi-power\"></i>\n </div>\n <div>\n <div class=\"fw-semibold mb-1\">Graceful Shutdown</div>\n <div class=\"text-muted small\">Runner finishes its current job then exits. Fire-and-forget.</div>\n </div>\n </div>\n <button class=\"btn btn-sm btn-outline-danger mt-auto\" data-action=\"shutdown\">\n <i class=\"bi bi-power me-1\"></i>Shutdown\n </button>\n </div>\n </div>\n </div>\n\n <!-- Export -->\n <div class=\"col-md-4\">\n <div class=\"card border-0 shadow-sm h-100\">\n <div class=\"card-body d-flex flex-column gap-3\">\n <div class=\"d-flex gap-3 align-items-start\">\n <div class=\"d-flex align-items-center justify-content-center rounded bg-secondary-subtle text-secondary flex-shrink-0\"\n style=\"width:40px;height:40px;font-size:1.1rem;\">\n <i class=\"bi bi-download\"></i>\n </div>\n <div>\n <div class=\"fw-semibold mb-1\">Export Snapshot</div>\n <div class=\"text-muted small\">Download runner identity data as a JSON file.</div>\n </div>\n </div>\n <button class=\"btn btn-sm btn-outline-secondary mt-auto\" data-action=\"export\">\n <i class=\"bi bi-download me-1\"></i>Export JSON\n </button>\n </div>\n </div>\n </div>\n\n </div>\n\n <div class=\"d-flex align-items-center gap-2 mb-3\">\n <span class=\"text-muted fw-semibold text-uppercase\" style=\"font-size:0.7rem;letter-spacing:0.09em;white-space:nowrap;\">Broadcast Command</span>\n <hr class=\"flex-grow-1 my-0\">\n </div>\n\n <div class=\"card border-0 shadow-sm\">\n <div class=\"card-body\">\n <p class=\"text-muted small mb-3\">\n Send a command to <strong>all active runners</strong> simultaneously and collect replies within the timeout window.\n </p>\n <div class=\"row g-2 align-items-end\">\n <div class=\"col-md-4\">\n <label class=\"form-label fw-semibold small text-muted mb-1\">Command</label>\n <select class=\"form-select form-select-sm\" data-field=\"broadcast-command\">\n <option value=\"status\">status</option>\n <option value=\"pause\">pause</option>\n <option value=\"resume\">resume</option>\n <option value=\"reload\">reload</option>\n <option value=\"shutdown\">shutdown</option>\n </select>\n </div>\n <div class=\"col-md-3\">\n <label class=\"form-label fw-semibold small text-muted mb-1\">Timeout (s)</label>\n <input type=\"number\" class=\"form-control form-control-sm\"\n data-field=\"broadcast-timeout\" value=\"2.0\" min=\"0.5\" step=\"0.5\" />\n </div>\n <div class=\"col-md-5\">\n <button class=\"btn btn-primary btn-sm w-100\" data-action=\"broadcast\">\n <i class=\"bi bi-megaphone me-1\"></i>Broadcast to All Runners\n </button>\n </div>\n </div>\n </div>\n </div>\n `;\n }\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// RunnerDetailsView — shell (default export)\n// ─────────────────────────────────────────────────────────────────────────────\n\nexport default class RunnerDetailsView extends View {\n constructor(options = {}) {\n super({ className: 'runner-details-view', ...options });\n this.model = options.model instanceof JobRunner\n ? options.model\n : new JobRunner(options.model || options.data || {});\n }\n\n async onInit() {\n if (!this.model) return;\n\n const tabView = new TabView({\n containerId: 'runner-tabs',\n tabs: {\n 'Overview': new RunnerOverviewTab({ model: this.model }),\n 'System Info': new RunnerSysinfoTab({ model: this.model }),\n 'Running Jobs': new RunnerJobsTab({ model: this.model }),\n 'Logs': new RunnerLogsTab({ model: this.model }),\n 'Actions': new RunnerActionsTab({ model: this.model })\n }\n });\n\n this.addChild(tabView);\n }\n\n async getTemplate() {\n return `<div data-container=\"runner-tabs\"></div>`;\n }\n\n /**\n * Open this view in a Dialog.\n *\n * @param {object} runner — runner object from GET /api/jobs/runners\n * @param {object} options — extra options forwarded to Dialog.showDialog()\n * @returns {Promise<any>}\n */\n static async show(runner, options = {}) {\n const model = runner instanceof JobRunner ? runner : new JobRunner(runner);\n const view = new RunnerDetailsView({ model });\n\n return await Dialog.showDialog({\n title: `<i class=\"bi bi-cpu me-2\"></i><span class=\"font-monospace\">${model.get('runner_id')}</span>`,\n body: view,\n size: 'xl',\n scrollable: true,\n buttons: [\n { text: 'Close', class: 'btn-secondary', dismiss: true }\n ],\n ...options\n });\n }\n}\n\n\nJobRunner.VIEW_CLASS = RunnerDetailsView;\n","/**\n * TaskManagementPage - Async task monitoring and management dashboard\n */\n\nimport Page from '@core/Page.js';\nimport View from '@core/View.js';\nimport Collection from '@core/Collection.js';\nimport TabView from '@core/views/navigation/TabView.js';\nimport TableView from '@core/views/table/TableView.js';\nimport { MetricsChart } from '@ext/charts/index.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\nimport TaskDetailsView from './TaskDetailsView.js';\nimport RunnerDetailsView from './RunnerDetailsView.js';\n\n// Task Stats Header View\nclass TaskStatsView extends View {\n constructor(options = {}) {\n super({\n ...options,\n className: 'mojo-task-stats-section'\n });\n\n this.stats = {\n pending: 0,\n running: 0,\n completed: 0,\n errors: 0\n };\n }\n\n async getTemplate() {\n return `\n <div class=\"mojo-task-stats-header mb-4\">\n <div class=\"row\">\n <div class=\"col-xl-3 col-lg-6 col-12 mb-3\">\n <div class=\"card h-100 border-0 shadow-sm\">\n <div class=\"card-body\">\n <div class=\"d-flex justify-content-between align-items-start\">\n <div>\n <h6 class=\"card-title text-muted mb-2\">Pending Tasks</h6>\n <h3 class=\"mb-1 fw-bold\">{{stats.pending}}</h3>\n <span class=\"badge bg-primary-subtle text-primary\">\n <i class=\"bi bi-clock\"></i> Queued\n </span>\n </div>\n <div class=\"text-primary\">\n <i class=\"bi bi-hourglass fs-2\"></i>\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <div class=\"col-xl-3 col-lg-6 col-12 mb-3\">\n <div class=\"card h-100 border-0 shadow-sm\">\n <div class=\"card-body\">\n <div class=\"d-flex justify-content-between align-items-start\">\n <div>\n <h6 class=\"card-title text-muted mb-2\">Running Tasks</h6>\n <h3 class=\"mb-1 fw-bold\">{{stats.running}}</h3>\n <span class=\"badge bg-success-subtle text-success\">\n <i class=\"bi bi-play-circle\"></i> Active\n </span>\n </div>\n <div class=\"text-success\">\n <i class=\"bi bi-arrow-repeat fs-2\"></i>\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <div class=\"col-xl-3 col-lg-6 col-12 mb-3\">\n <div class=\"card h-100 border-0 shadow-sm\">\n <div class=\"card-body\">\n <div class=\"d-flex justify-content-between align-items-start\">\n <div>\n <h6 class=\"card-title text-muted mb-2\">Completed Tasks</h6>\n <h3 class=\"mb-1 fw-bold\">{{stats.completed}}</h3>\n <span class=\"badge bg-info-subtle text-info\">\n <i class=\"bi bi-check-circle\"></i> Done\n </span>\n </div>\n <div class=\"text-info\">\n <i class=\"bi bi-check-square fs-2\"></i>\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <div class=\"col-xl-3 col-lg-6 col-12 mb-3\">\n <div class=\"card h-100 border-0 shadow-sm\">\n <div class=\"card-body\">\n <div class=\"d-flex justify-content-between align-items-start\">\n <div>\n <h6 class=\"card-title text-muted mb-2\">Failed Tasks</h6>\n <h3 class=\"mb-1 fw-bold\">{{stats.errors}}</h3>\n <span class=\"badge bg-danger-subtle text-danger\">\n <i class=\"bi bi-exclamation-circle\"></i> Errors\n </span>\n </div>\n <div class=\"text-danger\">\n <i class=\"bi bi-x-octagon fs-2\"></i>\n </div>\n </div>\n </div>\n </div>\n </div>\n </div>\n </div>\n `;\n }\n\n async loadStats() {\n try {\n const response = await this.getApp().rest.GET('/api/tasks/status');\n if (response.success && response.data.status) {\n this.stats = response.data.data;\n }\n } catch (error) {\n console.error('Failed to load task stats:', error);\n }\n }\n\n async onInit() {\n await this.loadStats();\n }\n}\n\n// Task Runners Status View\nclass TaskRunnersView extends View {\n constructor(options = {}) {\n super({\n ...options,\n className: 'mojo-task-runners-section'\n });\n\n this.runners = [];\n }\n\n async getTemplate() {\n return `\n <div class=\"card border shadow-sm mb-4\">\n <div class=\"card-header d-flex justify-content-between align-items-center\">\n <h5 class=\"card-title mb-0\">\n <i class=\"bi bi-cpu me-2\"></i>Task Runners\n </h5>\n <button class=\"btn btn-sm btn-outline-primary\" data-action=\"refresh-runners\">\n <i class=\"bi bi-arrow-clockwise\"></i> Refresh\n </button>\n </div>\n <div class=\"card-body\">\n {{#runners.length}}\n <div class=\"mojo-task-runner-list\">\n {{#runners}}\n <div class=\"mojo-task-runner-item p-3 mb-2 bg-light rounded\">\n <div class=\"row align-items-center\">\n <div class=\"col-md-8 col-lg-9\">\n <div class=\"d-flex align-items-center\">\n <div class=\"mojo-task-runner-status me-3\">\n <span class=\"badge {{statusBadge}}\">\n <i class=\"bi {{statusIcon}}\"></i> {{status}}\n </span>\n </div>\n <div class=\"mojo-task-runner-info\">\n <div class=\"mojo-task-runner-name\">\n <strong>{{hostname}}</strong>\n {{#max_workers}}<span class=\"text-muted ms-2\">• {{max_workers}} workers</span>{{/max_workers}}\n </div>\n <div class=\"mojo-task-runner-channels\">\n <small class=\"text-muted\">\n {{#channels.length}}Channels: {{#channels}}{{.}}{{^last}}, {{/last}}{{/channels}}{{/channels.length}}\n {{^channels.length}}No channels assigned{{/channels.length}}\n </small>\n </div>\n </div>\n </div>\n </div>\n <div class=\"col-md-4 col-lg-3\">\n <div class=\"mojo-task-runner-actions d-flex justify-content-end align-items-center\">\n <small class=\"text-muted me-2 d-none d-sm-inline\">{{pingAgeText}}</small>\n <div class=\"dropdown\">\n <button class=\"btn btn-sm btn-outline-secondary dropdown-toggle\"\n data-bs-toggle=\"dropdown\" aria-expanded=\"false\">\n <i class=\"bi bi-three-dots-vertical\"></i>\n </button>\n <ul class=\"dropdown-menu dropdown-menu-end\">\n <li><button class=\"dropdown-item\" data-action=\"view-runner-details\" data-runner-id=\"{{id}}\">\n <i class=\"bi bi-info-circle me-2\"></i>View Details\n </button></li>\n {{#isActive}}\n <li><button class=\"dropdown-item text-warning\" data-action=\"pause-runner\" data-runner-id=\"{{id}}\">\n <i class=\"bi bi-pause me-2\"></i>Pause Runner\n </button></li>\n {{/isActive}}\n {{^isActive}}\n <li><button class=\"dropdown-item text-success\" data-action=\"restart-runner\" data-runner-id=\"{{id}}\">\n <i class=\"bi bi-play me-2\"></i>Restart Runner\n </button></li>\n {{/isActive}}\n <li><hr class=\"dropdown-divider\"></li>\n <li><button class=\"dropdown-item text-danger\" data-action=\"remove-runner\" data-runner-id=\"{{id}}\">\n <i class=\"bi bi-trash me-2\"></i>Remove Runner\n </button></li>\n </ul>\n </div>\n </div>\n </div>\n </div>\n <div class=\"row mt-2 d-sm-none\">\n <div class=\"col-12\">\n <small class=\"text-muted\">Last ping: {{pingAgeText}}</small>\n </div>\n </div>\n </div>\n {{/runners}}\n </div>\n {{/runners.length}}\n {{^runners.length}}\n <div class=\"text-center text-muted py-4\">\n <i class=\"bi bi-cpu fs-1\"></i>\n <p class=\"mt-2\">No task runners found</p>\n </div>\n {{/runners.length}}\n </div>\n </div>\n `;\n }\n\n async loadRunners() {\n try {\n const response = await this.getApp().rest.GET('/api/tasks/runners');\n if (response.success && response.data.status) {\n this.runners = response.data.data.map(runner => {\n const isActive = runner.status === 'active';\n const pingAge = runner.ping_age || 0;\n\n return {\n ...runner,\n isActive,\n statusBadge: isActive ? 'bg-success' : 'bg-warning',\n statusIcon: isActive ? 'bi-check-circle-fill' : 'bi-exclamation-triangle-fill',\n pingAgeText: this.formatPingAge(pingAge)\n };\n });\n }\n } catch (error) {\n console.error('Failed to load runners:', error);\n }\n }\n\n formatPingAge(seconds) {\n if (seconds < 60) return `${Math.round(seconds)}s ago`;\n if (seconds < 3600) return `${Math.round(seconds / 60)}m ago`;\n return `${Math.round(seconds / 3600)}h ago`;\n }\n\n async onInit() {\n await this.loadRunners();\n }\n\n async onActionRefreshRunners(event, element) {\n await this.loadRunners();\n }\n\n async onActionViewRunnerDetails(event, element) {\n const runnerId = element.getAttribute('data-runner-id');\n const runner = this.runners.find(r => r.id === runnerId);\n\n if (runner) {\n const result = await RunnerDetailsView.show(runner);\n\n // Refresh runners list if any action was taken\n if (result?.action) {\n await this.loadRunners();\n this.emit('runner:' + result.action, result);\n }\n }\n }\n\n async onActionPauseRunner(event, element) {\n const runnerId = element.getAttribute('data-runner-id');\n const runner = this.runners.find(r => r.id === runnerId);\n\n if (!runner || !confirm(`Are you sure you want to pause runner \"${runner.hostname}\"?`)) {\n return;\n }\n\n try {\n element.disabled = true;\n const response = await this.getApp().rest.POST(`/api/runners/${runnerId}/pause`);\n\n if (response.success && response.data.status) {\n this.showSuccess('Runner paused successfully');\n await this.loadRunners();\n } else {\n this.showError(response.data.error || 'Failed to pause runner');\n }\n } catch (error) {\n console.error('Failed to pause runner:', error);\n this.showError('Failed to pause runner: ' + error.message);\n } finally {\n element.disabled = false;\n }\n }\n\n async onActionRestartRunner(event, element) {\n const runnerId = element.getAttribute('data-runner-id');\n const runner = this.runners.find(r => r.id === runnerId);\n\n if (!runner || !confirm(`Are you sure you want to restart runner \"${runner.hostname}\"?`)) {\n return;\n }\n\n try {\n element.disabled = true;\n const response = await this.getApp().rest.POST(`/api/runners/${runnerId}/restart`);\n\n if (response.success && response.data.status) {\n this.showSuccess('Runner restart initiated');\n await this.loadRunners();\n } else {\n this.showError(response.data.error || 'Failed to restart runner');\n }\n } catch (error) {\n console.error('Failed to restart runner:', error);\n this.showError('Failed to restart runner: ' + error.message);\n } finally {\n element.disabled = false;\n }\n }\n\n async onActionRemoveRunner(event, element) {\n const runnerId = element.getAttribute('data-runner-id');\n const runner = this.runners.find(r => r.id === runnerId);\n\n if (!runner) return;\n\n const confirmMessage = `Are you sure you want to remove runner \"${runner.hostname}\"? This action cannot be undone.`;\n if (!confirm(confirmMessage)) {\n return;\n }\n\n try {\n element.disabled = true;\n const response = await this.getApp().rest.DELETE(`/api/runners/${runnerId}`);\n\n if (response.success && response.data.status) {\n this.showSuccess('Runner removed successfully');\n await this.loadRunners();\n } else {\n this.showError(response.data.error || 'Failed to remove runner');\n }\n } catch (error) {\n console.error('Failed to remove runner:', error);\n this.showError('Failed to remove runner: ' + error.message);\n } finally {\n element.disabled = false;\n }\n }\n}\n\n// Task Charts View\nclass TaskChartsView extends View {\n constructor(options = {}) {\n super({\n ...options,\n className: 'mojo-task-charts-section'\n });\n }\n\n async getTemplate() {\n return `\n <div class=\"row mb-4\">\n <div class=\"col-xl-6 col-lg-12 mb-4\">\n <div class=\"card border shadow-sm\">\n <div class=\"card-body\" style=\"min-height: 300px;\">\n <div data-container=\"task-flow-chart\"></div>\n </div>\n </div>\n </div>\n <div class=\"col-xl-6 col-lg-12 mb-4\">\n <div class=\"card border shadow-sm\">\n <div class=\"card-body\" style=\"min-height: 300px;\">\n <div data-container=\"task-errors-chart\"></div>\n </div>\n </div>\n </div>\n </div>\n `;\n }\n\n async onInit() {\n // Task Flow Chart (Published vs Completed)\n this.taskFlowChart = new MetricsChart({\n title: '<i class=\"bi bi-graph-up me-2\"></i>Task Flow',\n endpoint: '/api/metrics/fetch',\n height: 280,\n granularity: 'hours',\n slugs: ['tasks_pub', 'tasks_completed'],\n account: 'global',\n chartType: 'line',\n showDateRange: false,\n colors: [\n 'rgba(13, 110, 253, 0.8)', // Primary blue for published\n 'rgba(25, 135, 84, 0.8)' // Success green for completed\n ],\n yAxis: {\n label: 'Count',\n beginAtZero: true\n },\n tooltip: {\n y: 'number'\n },\n containerId: 'task-flow-chart'\n });\n this.addChild(this.taskFlowChart);\n\n // Task Errors Chart\n this.taskErrorsChart = new MetricsChart({\n title: '<i class=\"bi bi-exclamation-triangle me-2\"></i>Task Issues',\n endpoint: '/api/metrics/fetch',\n height: 280,\n granularity: 'hours',\n slugs: ['tasks_errors', 'tasks_expired'],\n account: 'global',\n chartType: 'line',\n showDateRange: false,\n colors: [\n 'rgba(220, 53, 69, 0.8)', // Danger red for errors\n 'rgba(255, 193, 7, 0.8)' // Warning yellow for expired\n ],\n yAxis: {\n label: 'Count',\n beginAtZero: true\n },\n tooltip: {\n y: 'number'\n },\n containerId: 'task-errors-chart'\n });\n this.addChild(this.taskErrorsChart);\n }\n}\n\n// Task Table Views\nclass BaseTaskTable extends TableView {\n constructor(options = {}) {\n super({\n showPagination: true,\n showSearch: true,\n showRefresh: true,\n pageSize: 10,\n columns: [\n { key: 'id', label: 'Task ID', sortable: true },\n { key: 'function', label: 'Function', sortable: true },\n { key: 'channel', label: 'Channel', sortable: true },\n { key: 'created', label: 'Created', sortable: true, formatter: 'datetime' },\n { key: 'status', label: 'Status', formatter: 'badge' }\n ],\n ...options\n });\n }\n\n buildActionCell(item) {\n const actions = [];\n\n if (item.status === 'pending') {\n actions.push(`\n <button class=\"btn btn-sm btn-outline-danger\" data-action=\"cancel-task\" data-task-id=\"${item.id}\" title=\"Cancel Task\">\n <i class=\"bi bi-x-circle\"></i>\n </button>\n `);\n }\n\n if (item.status === 'error') {\n actions.push(`\n <button class=\"btn btn-sm btn-outline-primary\" data-action=\"retry-task\" data-task-id=\"${item.id}\" title=\"Retry Task\">\n <i class=\"bi bi-arrow-clockwise\"></i>\n </button>\n `);\n }\n\n actions.push(`\n <button class=\"btn btn-sm btn-outline-info\" data-action=\"view-task-details\" data-task-id=\"${item.id}\" title=\"View Details\">\n <i class=\"bi bi-info-circle\"></i>\n </button>\n `);\n\n return actions.join(' ');\n }\n\n async onActionCancelTask(action, event, element) {\n const taskId = element.getAttribute('data-task-id');\n\n if (!confirm('Are you sure you want to cancel this task?')) {\n return;\n }\n\n try {\n element.disabled = true;\n const response = await this.getApp().rest.POST(`/api/tasks/${taskId}/cancel`);\n\n if (response.success && response.data.status) {\n this.showSuccess('Task cancelled successfully');\n // Refresh the active table\n const activeTab = this.getParentPage()?.taskTablesView?.getActiveTab();\n if (activeTab) {\n const activeTable = this.getParentPage()?.taskTablesView?.getTab(activeTab);\n if (activeTable?.refresh) {\n await activeTable.refresh();\n }\n }\n } else {\n this.showError(response.data.error || 'Failed to cancel task');\n }\n } catch (error) {\n console.error('Failed to cancel task:', error);\n this.showError('Failed to cancel task: ' + error.message);\n } finally {\n element.disabled = false;\n }\n }\n\n async onActionRetryTask(action, event, element) {\n const taskId = element.getAttribute('data-task-id');\n\n try {\n element.disabled = true;\n const response = await this.getApp().rest.POST(`/api/tasks/${taskId}/retry`);\n\n if (response.success && response.data.status) {\n this.showSuccess('Task queued for retry');\n // Refresh the active table\n const activeTab = this.getParentPage()?.taskTablesView?.getActiveTab();\n if (activeTab) {\n const activeTable = this.getParentPage()?.taskTablesView?.getTab(activeTab);\n if (activeTable?.refresh) {\n await activeTable.refresh();\n }\n }\n } else {\n this.showError(response.data.error || 'Failed to retry task');\n }\n } catch (error) {\n console.error('Failed to retry task:', error);\n this.showError('Failed to retry task: ' + error.message);\n } finally {\n element.disabled = false;\n }\n }\n\n async onActionViewTaskDetails(action, event, element) {\n const taskId = element.getAttribute('data-task-id');\n\n try {\n // Fetch detailed task data\n const response = await this.getApp().rest.GET(`/api/tasks/${taskId}/details`);\n\n if (response.success && response.data.status) {\n const task = response.data.data;\n const result = await TaskDetailsView.show(task);\n\n // Refresh tables if any action was taken\n if (result?.action) {\n await this.refreshActiveTables();\n this.emit('task:' + result.action, result);\n }\n } else {\n this.showError(response.data.error || 'Failed to load task details');\n }\n } catch (error) {\n console.error('Failed to load task details:', error);\n this.showError('Failed to load task details: ' + error.message);\n }\n }\n\n getParentPage() {\n // Helper to get the parent TaskManagementPage\n let parent = this.parent;\n while (parent && parent.constructor.name !== 'TaskManagementPage') {\n parent = parent.parent;\n }\n return parent;\n }\n\n async refreshActiveTables() {\n const parentPage = this.getParentPage();\n if (parentPage?.taskTablesView) {\n const activeTab = parentPage.taskTablesView.getActiveTab();\n if (activeTab) {\n const activeTable = parentPage.taskTablesView.getTab(activeTab);\n if (activeTable?.refresh) {\n await activeTable.refresh();\n }\n }\n }\n }\n}\n\nclass PendingTasksTable extends TableView {\n constructor(options = {}) {\n super({\n ...options,\n title: 'Pending Tasks',\n collection: new Collection({\n endpoint: '/api/tasks/pending'\n }),\n columns: [\n { key: 'id', label: 'Task ID', sortable: true },\n { key: 'function', label: 'Function', sortable: true },\n { key: 'channel', label: 'Channel', sortable: true },\n { key: 'created', label: 'Created', sortable: true, formatter: 'datetime' },\n { key: 'status', label: 'Status', formatter: 'badge' }\n ],\n });\n }\n}\n\nclass RunningTasksTable extends TableView {\n constructor(options = {}) {\n super({\n ...options,\n title: 'Running Tasks',\n collection: new Collection({\n endpoint: '/api/tasks/running'\n }),\n columns: [\n { key: 'id', label: 'Task ID', sortable: true },\n { key: 'function', label: 'Function', sortable: true },\n { key: 'channel', label: 'Channel', sortable: true },\n { key: 'created', label: 'Created', sortable: true, formatter: 'datetime' },\n { key: 'status', label: 'Status', formatter: 'badge' }\n ],\n });\n }\n}\n\nclass CompletedTasksTable extends TableView {\n constructor(options = {}) {\n super({\n ...options,\n title: 'Completed Tasks',\n collection: new Collection({\n endpoint: '/api/tasks/completed'\n }),\n columns: [\n { key: 'id', label: 'Task ID', sortable: true },\n { key: 'function', label: 'Function', sortable: true },\n { key: 'channel', label: 'Channel', sortable: true },\n { key: 'created', label: 'Created', sortable: true, formatter: 'datetime' },\n { key: 'completed_at', label: 'Completed', sortable: true, formatter: 'datetime' },\n { key: 'status', label: 'Status', formatter: 'badge' }\n ]\n });\n }\n}\n\nclass ErrorTasksTable extends TableView {\n constructor(options = {}) {\n super({\n ...options,\n title: 'Failed Tasks',\n collection: new Collection({\n endpoint: '/api/tasks/errors'\n }),\n columns: [\n { key: 'id', label: 'Task ID', sortable: true },\n { key: 'function', label: 'Function', sortable: true },\n { key: 'channel', label: 'Channel', sortable: true },\n { key: 'created', label: 'Created', sortable: true, formatter: 'datetime' },\n { key: 'error', label: 'Error', sortable: false },\n { key: 'status', label: 'Status', formatter: 'badge' }\n ]\n });\n }\n}\n\n// Main Task Management Page\nexport default class TaskManagementPage extends Page {\n constructor(options = {}) {\n super({\n ...options,\n title: 'Task Management',\n className: 'mojo-task-management-page'\n });\n\n this.pageTitle = 'Task Management';\n this.pageSubtitle = 'Async task monitoring and runner management';\n this.lastUpdated = new Date().toLocaleString();\n }\n\n async getTemplate() {\n return `\n <div class=\"mojo-task-management-container\">\n <!-- Page Header -->\n <div class=\"d-flex justify-content-between align-items-center mb-4\">\n <div>\n <h1 class=\"h3 mb-1\">{{pageTitle}}</h1>\n <p class=\"text-muted mb-0\">{{pageSubtitle}}</p>\n <small class=\"text-info\">\n <i class=\"bi bi-cpu me-1\"></i>\n Real-time task processing and runner monitoring\n </small>\n </div>\n <div class=\"btn-group\" role=\"group\">\n <button type=\"button\" class=\"btn btn-outline-secondary btn-sm\"\n data-action=\"refresh-all\" title=\"Refresh All Data\">\n <i class=\"bi bi-arrow-clockwise\"></i> Refresh\n </button>\n <button type=\"button\" class=\"btn btn-outline-primary btn-sm\"\n data-action=\"export-tasks\" title=\"Export Task Data\">\n <i class=\"bi bi-download\"></i> Export\n </button>\n <button type=\"button\" class=\"btn btn-outline-success btn-sm\"\n data-action=\"manage-channels\" title=\"Manage Channels\">\n <i class=\"bi bi-collection\"></i> Channels\n </button>\n </div>\n </div>\n\n <!-- Task Stats -->\n <div data-container=\"task-stats\"></div>\n\n <!-- Task Runners -->\n <div data-container=\"task-runners\"></div>\n\n <!-- Task Charts -->\n <div data-container=\"task-charts\"></div>\n\n <!-- Task Tables -->\n <div class=\"card border shadow-sm\">\n <div class=\"card-header\">\n <h5 class=\"card-title mb-0\">\n <i class=\"bi bi-list-task me-2\"></i>Task Management\n </h5>\n </div>\n <div class=\"card-body\">\n <div data-container=\"task-tables\"></div>\n </div>\n </div>\n\n <!-- System Status Footer -->\n <div class=\"row mt-4\">\n <div class=\"col-12\">\n <div class=\"alert alert-info border-0\" role=\"alert\">\n <div class=\"d-flex align-items-center\">\n <i class=\"bi bi-info-circle-fill me-2\"></i>\n <div>\n <strong>Task System Status:</strong> Monitoring active.\n Last updated: <span class=\"text-muted\">{{lastUpdated}}</span>\n </div>\n <div class=\"ms-auto\">\n <button class=\"btn btn-sm btn-outline-info\" data-action=\"view-system-logs\">\n <i class=\"bi bi-journal-text\"></i> View Logs\n </button>\n </div>\n </div>\n </div>\n </div>\n </div>\n </div>\n `;\n }\n\n async onInit() {\n // Create Task Stats View\n this.taskStatsView = new TaskStatsView({\n containerId: 'task-stats'\n });\n this.addChild(this.taskStatsView);\n\n // Create Task Runners View\n this.taskRunnersView = new TaskRunnersView({\n containerId: 'task-runners'\n });\n this.addChild(this.taskRunnersView);\n\n // Create Task Charts View\n this.taskChartsView = new TaskChartsView({\n containerId: 'task-charts'\n });\n this.addChild(this.taskChartsView);\n\n // Create Task Tables with TabView\n this.taskTablesView = new TabView({\n containerId: 'task-tables',\n tabs: {\n 'Pending': new PendingTasksTable(),\n 'Running': new RunningTasksTable(),\n 'Completed': new CompletedTasksTable(),\n 'Errors': new ErrorTasksTable()\n },\n activeTab: 'Pending'\n });\n this.addChild(this.taskTablesView);\n }\n\n async onActionRefreshAll(action, event, element) {\n try {\n // Show loading state\n const icon = element.querySelector('i');\n icon?.classList.add('bi-spin');\n element.disabled = true;\n\n // Refresh all components\n const promises = [\n this.taskStatsView?.loadStats(),\n this.taskRunnersView?.loadRunners(),\n this.taskChartsView?.taskFlowChart?.refresh(),\n this.taskChartsView?.taskErrorsChart?.refresh()\n ].filter(Boolean);\n\n // Refresh active table\n const activeTab = this.taskTablesView?.getActiveTab();\n if (activeTab) {\n const activeTable = this.taskTablesView.getTab(activeTab);\n if (activeTable?.refresh) {\n promises.push(activeTable.refresh());\n }\n }\n\n await Promise.allSettled(promises);\n\n // Update timestamp\n this.lastUpdated = new Date().toLocaleString();\n\n // Emit refresh event\n const eventBus = this.getApp()?.events;\n if (eventBus) {\n eventBus.emit('tasks:dashboard-refreshed', {\n page: this,\n timestamp: this.lastUpdated\n });\n }\n\n } catch (error) {\n console.error('Failed to refresh task dashboard:', error);\n } finally {\n // Reset button state\n const icon = element.querySelector('i');\n icon?.classList.remove('bi-spin');\n element.disabled = false;\n }\n }\n\n async onActionExportTasks(action, event, element) {\n try {\n // Export charts\n await this.taskChartsView?.taskFlowChart?.export('png');\n await this.taskChartsView?.taskErrorsChart?.export('png');\n\n // Export active table\n const activeTab = this.taskTablesView?.getActiveTab();\n if (activeTab) {\n const activeTable = this.taskTablesView.getTab(activeTab);\n if (activeTable?.exportToCSV) {\n activeTable.exportToCSV();\n }\n }\n\n } catch (error) {\n console.error('Failed to export task data:', error);\n }\n }\n\n async onActionManageChannels(event, element) {\n try {\n // For now, show a simple info modal with channel information\n const response = await this.getApp().rest.GET('/api/tasks/channels');\n\n if (response.success && response.data.status) {\n const channels = response.data.data;\n\n // Create a simple info alert\n const channelList = channels.map(channel =>\n `${channel.name} (${channel.pending} pending, ${channel.running} running)`\n ).join('\\n');\n\n alert(`Task Channels:\\n\\n${channelList}\\n\\nFull channel management interface coming soon!`);\n } else {\n this.showError('Failed to load channel information');\n }\n } catch (error) {\n console.error('Failed to load channels:', error);\n // Navigate to dedicated channels page as fallback\n const router = this.getApp()?.router;\n if (router) {\n router.navigateTo('/admin/task-channels');\n }\n }\n }\n\n async onActionViewSystemLogs(event, element) {\n try {\n // Try to get recent system logs related to tasks\n const response = await this.getApp().rest.GET('/api/tasks/logs?limit=50');\n\n if (response.success && response.data.status) {\n const logs = response.data.data;\n\n // Create a simple log preview\n const logPreview = logs.slice(0, 10).map(log =>\n `[${new Date(log.timestamp * 1000).toLocaleString()}] ${log.level.toUpperCase()}: ${log.message}`\n ).join('\\n');\n\n const viewFullLogs = confirm(`Recent Task System Logs:\\n\\n${logPreview}\\n\\n... and ${logs.length - 10} more entries.\\n\\nView full logs?`);\n\n if (viewFullLogs) {\n const router = this.getApp()?.router;\n if (router) {\n router.navigateTo('/admin/logs?filter=tasks');\n }\n }\n } else {\n // Navigate to system logs as fallback\n const router = this.getApp()?.router;\n if (router) {\n router.navigateTo('/admin/logs');\n }\n }\n } catch (error) {\n console.error('Failed to load system logs:', error);\n // Navigate to system logs as fallback\n const router = this.getApp()?.router;\n if (router) {\n router.navigateTo('/admin/logs');\n }\n }\n }\n\n // Public API\n async refreshDashboard() {\n return this.onActionRefreshAll(null, null, { disabled: false, querySelector: () => null });\n }\n\n getStats() {\n return this.taskStatsView?.stats || {};\n }\n\n getRunners() {\n return this.taskRunnersView?.runners || [];\n }\n\n getCharts() {\n return {\n taskFlow: this.taskChartsView?.taskFlowChart,\n taskErrors: this.taskChartsView?.taskErrorsChart\n };\n }\n\n\n}\n","/**\n * LogView - Detailed view for a Log record\n */\n\nimport View from '@core/View.js';\nimport TabView from '@core/views/navigation/TabView.js';\nimport DataView from '@core/views/data/DataView.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport { Log } from '@core/models/Log.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\nimport DeviceView from '../account/devices/DeviceView.js';\nimport GeoIPView from '../account/devices/GeoIPView.js';\n\nclass LogView extends View {\n constructor(options = {}) {\n super({\n className: 'log-view',\n ...options\n });\n\n this.model = options.model || new Log(options.data || {});\n this.logIcon = this.getIconForLog(this.model.get('level'));\n\n this.template = `\n <div class=\"log-view-container\">\n <!-- Header -->\n <div class=\"d-flex justify-content-between align-items-center mb-4\">\n <div class=\"d-flex align-items-center gap-3\">\n <div class=\"fs-1 {{logIcon.color}}\">\n <i class=\"bi {{logIcon.icon}}\"></i>\n </div>\n <div>\n <h4 class=\"mb-1\">\n <span class=\"badge bg-secondary\">{{model.method}}</span> {{model.path}}\n </h4>\n <div class=\"text-muted small\">\n {{model.created|datetime}}\n </div>\n </div>\n </div>\n <div data-container=\"log-context-menu\"></div>\n </div>\n\n <!-- Tabs -->\n <div data-container=\"log-tabs\"></div>\n </div>\n `;\n }\n\n getIconForLog(level) {\n const lvl = level?.toLowerCase();\n if (lvl === 'error' || lvl === 'critical') return { icon: 'bi-x-octagon-fill', color: 'text-danger' };\n if (lvl === 'warning') return { icon: 'bi-exclamation-triangle-fill', color: 'text-warning' };\n if (lvl === 'info') return { icon: 'bi-info-circle-fill', color: 'text-info' };\n return { icon: 'bi-journal-text', color: 'text-secondary' };\n }\n\n async onInit() {\n // Overview Tab\n this.overviewView = new DataView({\n model: this.model,\n className: \"p-3\",\n columns: 2,\n fields: [\n { name: 'id', label: 'Log ID' },\n { name: 'level', label: 'Level', format: 'badge' },\n { name: 'kind', label: 'Kind' },\n { name: 'ip', label: 'IP Address', template: '<a href=\"#\" data-action=\"view-ip\">{{model.ip}}</a>' },\n { name: 'uid', label: 'User ID' },\n { name: 'username', label: 'Username' },\n { name: 'duid', label: 'Device ID', template: '<a href=\"#\" data-action=\"view-device\">{{model.duid|truncate_middle(32)}}</a>' },\n { name: 'model_name', label: 'Related Model' },\n { name: 'model_id', label: 'Related Model ID' },\n { name: 'user_agent', label: 'User Agent', columns: 12 },\n ]\n });\n\n // Log Content Tab\n const logContent = this.model.get('log');\n let formattedLog = logContent;\n try {\n // Attempt to parse and prettify if it's a JSON string\n const parsed = JSON.parse(logContent);\n formattedLog = JSON.stringify(parsed, null, 2);\n } catch (e) {\n // Not a valid JSON string, display as is\n }\n\n this.logContentView = new View({\n template: `\n <div class=\"position-relative\">\n <button class=\"btn btn-sm btn-outline-secondary position-absolute top-0 end-0 mt-2 me-2\" data-action=\"copy-log\">\n <i class=\"bi bi-clipboard\"></i> Copy\n </button>\n <pre class=\"bg-light p-3 border rounded\" style=\"max-height: 600px; overflow-y: auto;\"><code>${formattedLog}</code></pre>\n </div>\n `,\n onActionCopyLog: () => {\n navigator.clipboard.writeText(formattedLog);\n this.getApp()?.toast?.success('Log content copied to clipboard.');\n }\n });\n\n // TabView\n this.tabView = new TabView({\n containerId: 'log-tabs',\n tabs: {\n 'Log': this.logContentView,\n 'Details': this.overviewView,\n\n },\n activeTab: 'Log'\n });\n this.addChild(this.tabView);\n\n // ContextMenu\n const logMenu = new ContextMenu({\n containerId: 'log-context-menu',\n className: \"context-menu-view header-menu-absolute\",\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: [\n { label: 'View User', action: 'view-user', icon: 'bi-person', disabled: !this.model.get('uid') },\n { label: 'View Device', action: 'view-device', icon: 'bi-phone', disabled: !this.model.get('duid') },\n { type: 'divider' },\n { label: 'Delete Log', action: 'delete-log', icon: 'bi-trash', danger: true }\n ]\n }\n });\n this.addChild(logMenu);\n }\n\n async onActionViewIp(event) {\n event.preventDefault();\n const ip = this.model.get('ip');\n if (ip) {\n GeoIPView.show(ip);\n }\n }\n\n async onActionViewDevice(event) {\n event.preventDefault();\n const duid = this.model.get('duid');\n if (duid) {\n DeviceView.show(duid);\n }\n }\n\n async onActionViewUser() {\n console.log(\"TODO: View user\", this.model.get('uid'));\n }\n\n async onActionDeleteLog() {\n const confirmed = await Dialog.confirm(\n `Are you sure you want to delete this log entry? This action cannot be undone.`,\n 'Confirm Deletion',\n { confirmClass: 'btn-danger', confirmText: 'Delete' }\n );\n if (confirmed) {\n const resp = await this.model.destroy();\n if (resp.success) {\n this.emit('log:deleted', { model: this.model });\n }\n }\n }\n}\n\nLog.VIEW_CLASS = LogView;\n\nexport default LogView;\n","/**\n * LogTablePage - Log management using TablePage component\n * Clean implementation using TablePage with minimal overrides\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { LogList } from '@core/models/Log.js';\nimport LogView from './LogView.js';\n\nclass LogTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_logs',\n pageName: 'Manage Logs',\n router: \"admin/logs\",\n Collection: LogList,\n\n itemViewClass: LogView,\n viewDialogOptions: {\n header: false,\n size: 'xl'\n },\n\n // Column definitions\n columns: [\n {\n key: 'created|epoch|datetime',\n label: 'Timestamp',\n sortable: true,\n filter: {\n type: 'daterange',\n }\n },\n {\n key: 'level',\n label: 'Level',\n sortable: true,\n formatter: 'badge',\n filter: {\n type: 'select',\n options: [\n { value: 'info', label: 'Info' },\n { value: 'warning', label: 'Warning' },\n { value: 'error', label: 'Error' }\n ]\n }\n },\n {\n key: 'kind',\n label: 'Kind',\n filter: {\n type: \"text\"\n }\n },\n {\n key: 'method',\n label: 'Method',\n filter: {\n type: \"text\"\n }\n },\n {\n key: 'path',\n label: 'Path',\n filter: {\n type: \"text\"\n }\n },\n {\n key: 'username',\n label: 'User',\n filter: {\n type: \"text\"\n }\n },\n {\n key: 'ip',\n label: 'IP',\n filter: {\n type: \"text\"\n }\n },\n {\n key: 'duid',\n label: 'Browser ID',\n formatter: 'truncate_middle(16)',\n filter: {\n type: \"text\"\n }\n }\n ],\n\n // Default sort by timestamp descending (newest first)\n defaultQuery: {\n sort: '-created'\n },\n\n // Table features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Toolbar\n showRefresh: true,\n showAdd: false, // Logs are typically not manually created\n showExport: true,\n\n // Empty state\n emptyMessage: 'No log entries found.',\n\n // Batch actions\n batchBarLocation: 'top',\n batchActions: [\n { label: \"Export\", icon: \"bi bi-download\", action: \"batch-export\" },\n { label: \"Archive\", icon: \"bi bi-archive\", action: \"batch-archive\" },\n { label: \"Delete\", icon: \"bi bi-trash\", action: \"batch-delete\" },\n { label: \"Mark as Reviewed\", icon: \"bi bi-check2\", action: \"batch-reviewed\" }\n ],\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n }\n });\n }\n}\n\nexport default LogTablePage;\n","import View from '@core/View.js';\nimport DataView from '@core/views/data/DataView.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport { MetricsPermission, MetricsForms } from '@core/models/Metrics.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\n\nclass MetricsPermissionsView extends View {\n constructor(options = {}) {\n super({\n className: 'metrics-permissions-view',\n ...options\n });\n\n this.model = options.model || new MetricsPermission(options.data || {});\n\n this.template = `\n <div class=\"container p-3\">\n <div class=\"d-flex justify-content-between align-items-center mb-4\">\n <div>\n <h3 class=\"mb-1\">Permissions for {{model.account}}</h3>\n </div>\n <div data-container=\"context-menu\"></div>\n </div>\n <div data-container=\"data-view\"></div>\n </div>\n `;\n }\n\n async onInit() {\n this.dataView = new DataView({\n containerId: 'data-view',\n model: this.model,\n fields: [\n { name: 'view_permissions', label: 'View Permissions', format: 'list|badge' },\n { name: 'write_permissions', label: 'Write Permissions', format: 'list|badge' },\n ]\n });\n this.addChild(this.dataView);\n\n const contextMenu = new ContextMenu({\n containerId: 'context-menu',\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: [\n { label: 'Edit', action: 'edit', icon: 'bi-pencil' },\n { label: 'Delete', action: 'delete', icon: 'bi-trash', danger: true },\n ]\n }\n });\n this.addChild(contextMenu);\n }\n\n async onActionEdit() {\n const resp = await Dialog.showModelForm({\n title: `Edit Permissions for ${this.model.get('account')}`,\n model: this.model,\n formConfig: MetricsForms.edit\n });\n if (resp) {\n this.model.set(resp.data.data);\n this.render();\n }\n }\n\n async onActionDelete() {\n const confirmed = await Dialog.confirm(`Are you sure you want to delete all permissions for ${this.model.get('account')}?`);\n if (confirmed) {\n await this.model.destroy();\n this.emit('deleted', this.model);\n }\n }\n}\n\nMetricsPermission.VIEW_CLASS = MetricsPermissionsView;\n\nexport default MetricsPermissionsView;\n","import TablePage from '@core/pages/TablePage.js';\nimport { MetricsPermissionList, MetricsForms } from '@core/models/Metrics.js';\nimport MetricsPermissionsView from './MetricsPermissionsView.js';\n\nclass MetricsPermissionsTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_metrics_permissions',\n pageName: 'Metrics Permissions',\n router: \"admin/metrics/permissions\",\n Collection: MetricsPermissionList,\n formEdit: MetricsForms.edit,\n itemViewClass: MetricsPermissionsView,\n viewDialogOptions: {\n header: false,\n size: 'lg'\n },\n\n columns: [\n { key: 'account', label: 'Account', sortable: true },\n { key: 'view_permissions', label: 'View Permissions', formatter: 'list|badge' },\n { key: 'write_permissions', label: 'Write Permissions', formatter: 'list|badge' },\n ],\n\n selectable: true,\n searchable: true,\n sortable: true,\n paginated: true,\n showRefresh: true,\n showAdd: true,\n showExport: true,\n\n tableOptions: {\n pageSizes: [10, 25, 50],\n defaultPageSize: 25,\n emptyMessage: 'No metrics permissions found.',\n emptyIcon: 'bi-bar-chart-line',\n actions: [\"view\", \"edit\", \"delete\"],\n }\n });\n }\n}\n\nexport default MetricsPermissionsTablePage;\n","import Collection from '@core/Collection.js';\nimport Model from '@core/Model.js';\n\n/**\n * Setting - Global or group-scoped configuration key-value pair.\n * Maps to REST endpoints under /api/settings\n *\n * Key properties:\n * - Can be global or scoped to a group (via `group` field)\n * - Supports secret values (`is_secret: true`) — API masks display_value as \"******\"\n * - Secret values are write-only; the raw value is never returned after creation\n *\n * Endpoints:\n * GET /api/settings - List settings (requires manage_settings)\n * POST /api/settings - Create a setting\n * GET /api/settings/<id> - Get one setting\n * POST /api/settings/<id> - Update a setting\n * DELETE /api/settings/<id> - Delete a setting\n */\nclass Setting extends Model {\n constructor(data = {}, options = {}) {\n super(data, {\n endpoint: '/api/settings',\n ...options\n });\n }\n}\n\n/**\n * SettingList - Collection of Setting records.\n * Supports search and sort query params.\n */\nclass SettingList extends Collection {\n constructor(options = {}) {\n super({\n ModelClass: Setting,\n endpoint: '/api/settings',\n size: 25,\n ...options\n });\n }\n}\n\n/**\n * Forms configuration for Setting\n */\nconst SettingForms = {\n create: {\n title: 'Create Setting',\n fields: [\n {\n name: 'key',\n type: 'text',\n label: 'Key',\n placeholder: 'WEBHOOK_SECRET',\n required: true,\n columns: 12,\n help: 'A unique configuration key name.'\n },\n {\n name: 'value',\n type: 'textarea',\n label: 'Value',\n required: true,\n columns: 12,\n help: 'The configuration value. For secrets, this will be masked after creation.'\n },\n {\n name: 'group',\n type: 'number',\n label: 'Group ID',\n columns: 6,\n help: 'Optional. Leave empty for a global setting, or enter a group ID to scope it.'\n },\n {\n name: 'is_secret',\n type: 'switch',\n label: 'Secret',\n columns: 6,\n help: 'Mark as secret to mask the value in API responses.'\n }\n ]\n },\n\n edit: {\n title: 'Edit Setting',\n fields: [\n {\n name: 'key',\n type: 'text',\n label: 'Key',\n columns: 12,\n disabled: true\n },\n {\n name: 'value',\n type: 'textarea',\n label: 'Value',\n columns: 12,\n help: 'Enter a new value to replace the current one.'\n },\n {\n name: 'is_secret',\n type: 'switch',\n label: 'Secret',\n columns: 12,\n help: 'Mark as secret to mask the value in API responses.'\n }\n ]\n }\n};\n\nexport { Setting, SettingList, SettingForms };\n","/**\n * SettingView - Secure setting detail and management interface\n *\n * Shows setting metadata (key, value, group scope, secret status)\n * and provides actions to edit and delete. Secret values display\n * the masked display_value from the API.\n */\n\nimport View from '@core/View.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport { Setting, SettingForms } from '@core/models/Settings.js';\n\nclass SettingView extends View {\n constructor(options = {}) {\n super({\n className: 'setting-view',\n ...options\n });\n\n this.model = options.model || new Setting(options.data || {});\n\n this.template = `\n <div class=\"setting-view-container\">\n <!-- Header -->\n <div class=\"d-flex justify-content-between align-items-start mb-4\">\n <!-- Left: Icon & Identity -->\n <div class=\"d-flex align-items-center gap-3\">\n <div class=\"fs-1 text-primary\">\n <i class=\"bi bi-gear\"></i>\n </div>\n <div>\n <h3 class=\"mb-1 font-monospace\">{{model.key|default('Unnamed Setting')}}</h3>\n <div class=\"text-muted small\">\n ID: {{model.id}}\n <span class=\"mx-2\">|</span>\n Scope: {{model.group.name|default('Global')}}\n </div>\n <div class=\"mt-1\">\n <span class=\"badge {{model.is_secret|boolean('bg-warning text-dark','bg-secondary')}}\">\n {{model.is_secret|boolean('Secret','Plain')}}\n </span>\n </div>\n </div>\n </div>\n\n <!-- Right: Meta & Actions -->\n <div class=\"d-flex align-items-start gap-4\">\n <div class=\"text-end\">\n <div class=\"text-muted small\">Created</div>\n <div>{{model.created|datetime}}</div>\n </div>\n <div data-container=\"setting-context-menu\"></div>\n </div>\n </div>\n\n <!-- Details -->\n <div class=\"list-group mb-3\">\n <div class=\"list-group-item\">\n <h6 class=\"mb-1 text-muted\">Key</h6>\n <p class=\"mb-0 font-monospace\">{{model.key}}</p>\n </div>\n <div class=\"list-group-item\">\n <h6 class=\"mb-1 text-muted\">Value</h6>\n {{#model.is_secret|bool}}\n <p class=\"mb-0 text-muted font-monospace\">{{model.display_value|default('******')}}</p>\n <small class=\"text-muted\">This is a secret value. Enter a new value to replace it.</small>\n {{/model.is_secret|bool}}\n {{^model.is_secret|bool}}\n <p class=\"mb-0 font-monospace\">{{model.value|default(model.display_value)|default('—')}}</p>\n {{/model.is_secret|bool}}\n </div>\n {{#model.group}}\n <div class=\"list-group-item\">\n <h6 class=\"mb-1 text-muted\">Group</h6>\n <p class=\"mb-0\">{{model.group.name|default(model.group)}}</p>\n </div>\n {{/model.group}}\n </div>\n </div>\n `;\n }\n\n async onInit() {\n const settingMenu = new ContextMenu({\n containerId: 'setting-context-menu',\n className: 'context-menu-view header-menu-absolute',\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: [\n { label: 'Edit', action: 'edit-setting', icon: 'bi-pencil' },\n { type: 'divider' },\n { label: 'Delete Setting', action: 'delete-setting', icon: 'bi-trash', danger: true }\n ]\n }\n });\n this.addChild(settingMenu);\n }\n\n async onActionEditSetting() {\n const app = this.getApp();\n const resp = await app.showModelForm({\n title: `Edit Setting — ${this.model.get('key')}`,\n model: this.model,\n formConfig: SettingForms.edit,\n });\n if (resp) {\n this.render();\n }\n }\n\n async onActionDeleteSetting() {\n const app = this.getApp();\n const confirmed = await app.confirm({\n title: 'Delete Setting',\n message: `Permanently delete \"${this.model.get('key')}\"? This cannot be undone.`,\n confirmLabel: 'Delete',\n confirmClass: 'btn-danger'\n });\n if (!confirmed) return;\n\n app.showLoading();\n const resp = await this.model.delete();\n app.hideLoading();\n if (resp && resp.success !== false) {\n app.toast.success('Setting deleted');\n this.emit('deleted', { model: this.model });\n } else {\n app.toast.error('Failed to delete setting');\n }\n }\n}\n\nSetting.VIEW_CLASS = SettingView;\nexport default SettingView;\n","/**\n * SettingTablePage - Secure settings management using TablePage component\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { Setting, SettingList, SettingForms } from '@core/models/Settings.js';\nimport SettingView from './SettingView.js';\n\n// Register the add/edit forms on the model class so TableView can find them automatically\nSetting.ADD_FORM = SettingForms.create;\nSetting.EDIT_FORM = SettingForms.edit;\n\nclass SettingTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_settings',\n pageName: 'Settings',\n router: 'admin/settings',\n Collection: SettingList,\n\n itemViewClass: SettingView,\n viewDialogOptions: {\n header: false,\n size: 'lg'\n },\n\n columns: [\n { key: 'id', label: 'ID', width: '70px', sortable: true, class: 'text-muted' },\n { key: 'key', label: 'Key', sortable: true },\n { key: 'display_value', label: 'Value', formatter: \"default('—')\" },\n { key: 'group.name', label: 'Group', sortable: true, formatter: \"default('Global')\" },\n {\n key: 'is_secret',\n label: 'Secret',\n formatter: \"boolean('Secret|bg-warning text-dark','Plain|bg-secondary')|badge\",\n width: '100px'\n },\n { key: 'created', label: 'Created', formatter: 'datetime', sortable: true }\n ],\n\n defaultQuery: {\n sort: 'key'\n },\n\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n showRefresh: true,\n showAdd: true,\n showExport: false,\n\n addButtonLabel: 'New Setting',\n\n emptyMessage: 'No settings found.',\n\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n }\n });\n }\n}\n\nexport default SettingTablePage;\n","/**\n * FileManagerTablePage - File Manager backend management using TablePage component\n * Clean implementation using TablePage with minimal overrides\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { FileManagerList, FileManagerForms } from '@core/models/Files.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\n\nclass FileManagerTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_file_managers',\n pageName: 'Manage Storage Backends',\n router: \"admin/file-managers\",\n Collection: FileManagerList,\n\n formCreate: FileManagerForms.create,\n formEdit: FileManagerForms.edit,\n\n // Column definitions\n columns: [\n {\n key: 'id',\n label: 'ID',\n sortable: true,\n class: 'text-muted'\n },\n {\n key: 'name',\n label: 'Name',\n formatter: \"default('Unnamed Backend')\"\n },\n {\n key: 'backend_url',\n label: 'Backend URL',\n sortable: true\n },\n {\n key: 'is_default',\n label: 'Default',\n formatter: \"boolean|badge\"\n },\n {\n key: 'is_active',\n label: 'Active',\n formatter: \"boolean|badge\"\n },\n {\n key: 'is_public',\n label: 'Public',\n formatter: \"boolean|badge\"\n },\n {\n key: 'backend_type',\n label: 'Type',\n formatter: \"default('Unknown')\"\n },\n {\n key: 'created',\n label: 'Created',\n formatter: \"epoch|datetime\"\n }\n ],\n\n contextMenu: [\n { icon: 'bi-pencil', action: 'edit', label: 'Edit Name' },\n { icon: 'bi-shield', action: 'edit-credentials', label: 'Edit Credentials' },\n { icon: 'bi-person', action: 'edit-owners', label: 'Edit Owners' },\n { divider: true },\n { icon: 'bi-copy', action: 'clone', label: 'Clone Manager' },\n { divider: true },\n { icon: 'bi-check', action: 'test-connection', label: 'Test Connection' },\n { icon: 'bi-question-circle', action: 'check-cors', label: 'Check CORS' },\n { icon: 'bi-wrench', action: 'fix-cors', label: 'Fix CORS' },\n ],\n\n // Table features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Toolbar\n showRefresh: true,\n showAdd: true,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No storage backends found. Click \"Add Storage Backend\" to configure your first backend.',\n\n // Batch actions\n batchBarLocation: 'top',\n batchActions: [\n { label: \"Delete\", icon: \"bi bi-trash\", action: \"batch-delete\" },\n { label: \"Export\", icon: \"bi bi-download\", action: \"batch-export\" },\n { label: \"Activate\", icon: \"bi bi-check-circle\", action: \"batch-activate\" },\n { label: \"Deactivate\", icon: \"bi bi-x-circle\", action: \"batch-deactivate\" }\n ],\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n }\n });\n }\n\n async onActionEditOwners(event, element) {\n const item = this.collection.get(element.dataset.id);\n const result = await Dialog.showModelForm({\n title: 'Edit Owners',\n model: item,\n fields: FileManagerForms.owners.fields\n });\n if (!result) return true;\n if (result.success) {\n this.getApp().toast.success(\"Owners Updated successfully\");\n } else {\n this.getApp().toast.error(\"Owners update failed\");\n }\n }\n\n async onActionCheckCors(event, element) {\n const item = this.collection.get(element.dataset.id);\n const result = await item.save({check_cors: true});\n if (result.success && result.data.status) {\n // this.getApp().toast.success(result.data.result);\n await Dialog.showData({\n title: `Audit Report - ${item._.name}`,\n data: result.data,\n size: 'lg'\n });\n } else {\n this.getApp().toast.error(\"Connection test failed\");\n }\n return true;\n }\n\n async onActionTestConnection(event, element) {\n const item = this.collection.get(element.dataset.id);\n const result = await item.save({test_connection: true});\n if (result.success && result.data.status) {\n this.getApp().toast.success(\"Connection test successful\");\n } else {\n this.getApp().toast.error(\"Connection test failed\");\n }\n return true;\n }\n\n async onActionEditCredentials(event, element) {\n const item = this.collection.get(element.dataset.id);\n const result = await Dialog.showModelForm({\n title: 'Edit Credentials',\n model: item,\n fields: FileManagerForms.credentials.fields\n });\n\n if (!result) return true;\n\n if (result.success && result.data.status) {\n this.getApp().toast.success(\"Credentials updated successfully\");\n } else {\n this.getApp().toast.error(\"Failed to update credentials\");\n }\n return true;\n }\n\n async onActionClone(event, element) {\n\n const confirmed = await Dialog.showConfirm({\n title: 'Clone File Manager',\n message: 'This will create a clone with the same credentials.',\n });\n\n if (!confirmed) {\n return true;\n }\n\n const item = this.collection.get(element.dataset.id);\n const result = await item.save({clone: true});\n if (result.success && result.data.status) {\n this.getApp().toast.success(\"Connection cloned successfully\");\n this.collection.fetch();\n } else {\n this.getApp().toast.error(\"Failed to clone connection\");\n }\n return true;\n }\n}\n\nexport default FileManagerTablePage;\n","/**\n * FileView - Comprehensive file management interface\n *\n * Features:\n * - Clean header with file thumbnail/icon, name, size, and status\n * - Tabbed interface for Info (metadata) and Renditions\n * - Integrated with DataView, Table, and ContextMenu components\n */\n\nimport View from '@core/View.js';\nimport TabView from '@core/views/navigation/TabView.js';\nimport DataView from '@core/views/data/DataView.js';\nimport TableView from '@core/views/table/TableView.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport Collection from '@core/Collection.js';\nimport { File, FileForms } from '@core/models/Files.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\nimport LightboxGallery from '@ext/lightbox/LightboxGallery.js';\nimport PDFViewer from '@ext/lightbox/PDFViewer.js';\n\nclass FileView extends View {\n constructor(options = {}) {\n super({\n className: 'file-view',\n ...options\n });\n\n this.model = options.model || new File(options.data || {});\n this.isImage = this.model.get('category') === 'image';\n\n // Prepare renditions for the table by converting the renditions object to an array\n const renditionsData = this.model.get('renditions') || {};\n this.renditionsCollection = new Collection(Object.values(renditionsData));\n\n this.template = `\n <div class=\"file-view-container\">\n <!-- Header -->\n <div class=\"d-flex justify-content-between align-items-center mb-4\">\n <!-- Left Side: Thumbnail & Info -->\n <div class=\"d-flex align-items-center gap-3\">\n <div class=\"file-thumbnail\" style=\"width: 80px; height: 80px;\">\n {{#isImage}}\n <a href=\"{{model.url}}\" target=\"_blank\" title=\"View original file\">\n <img src=\"{{model.renditions.thumbnail.url|default(model.url)}}\" class=\"img-fluid rounded\" style=\"width: 80px; height: 80px; object-fit: cover;\">\n </a>\n {{/isImage}}\n {{^isImage}}\n <div class=\"avatar-placeholder rounded bg-light d-flex align-items-center justify-content-center\" style=\"width: 80px; height: 80px;\">\n <i class=\"bi bi-file-earmark-text text-secondary\" style=\"font-size: 40px;\"></i>\n </div>\n {{/isImage}}\n </div>\n <div>\n <h3 class=\"mb-1\" style=\"word-break: break-all;\">{{model.filename|truncate(40)}}</h3>\n <div class=\"text-muted small\">\n <span><i class=\"bi bi-hdd\"></i> {{model.file_size|filesize}}</span>\n <span class=\"mx-2\">|</span>\n <span>{{model.content_type}}</span>\n </div>\n <div class=\"text-muted small mt-1\">\n Uploaded: {{model.created|datetime}}\n </div>\n </div>\n </div>\n\n <!-- Right Side: Status & Actions -->\n <div class=\"d-flex align-items-center gap-4\">\n <div class=\"text-end\">\n <div class=\"d-flex align-items-center gap-2 justify-content-end\">\n <span class=\"badge {{model.upload_status|badge}}\">{{model.upload_status|capitalize}}</span>\n </div>\n <div class=\"text-muted small mt-1\">\n Public: {{{model.is_public|yesnoicon}}}\n </div>\n </div>\n <div data-container=\"file-context-menu\"></div>\n </div>\n </div>\n\n <!-- Tab Container -->\n <div data-container=\"file-tabs\"></div>\n </div>\n `;\n }\n\n async onInit() {\n // Info Tab using DataView\n this.infoView = new DataView({\n model: this.model,\n className: \"p-3\",\n showEmptyValues: true,\n emptyValueText: '—',\n columns: 2,\n fields: [\n { name: 'id', label: 'ID' },\n { name: 'filename', label: 'Filename' },\n { name: 'storage_filename', label: 'Storage Filename' },\n { name: 'content_type', label: 'Content Type' },\n { name: 'file_size', label: 'File Size', format: 'filesize' },\n { name: 'category', label: 'Category' },\n { name: 'upload_status', label: 'Status', format: 'badge' },\n { name: 'created', label: 'Created', format: 'datetime' },\n { name: 'modified', label: 'Modified', format: 'datetime' },\n { name: 'user.display_name', label: 'Uploaded By' },\n { name: 'file_manager.name', label: 'Storage Backend' },\n { name: 'storage_file_path', label: 'Storage Path' },\n { name: 'url', label: 'Public URL', format: 'url' },\n { name: 'is_public', label: 'Is Public', format: 'boolean' },\n ]\n });\n\n // Renditions Tab using TableView\n this.renditionsView = new TableView({\n collection: this.renditionsCollection,\n columns: [\n { key: 'role', label: 'Role', formatter: 'badge' },\n { key: 'filename', label: 'Filename', formatter: 'truncate(40)' },\n { key: 'file_size', label: 'Size', formatter: 'filesize' },\n { key: 'content_type', label: 'Content Type' },\n {\n key: 'actions',\n label: 'Actions',\n template: `\n <a href=\"{{url}}\" target=\"_blank\" class=\"btn btn-sm btn-outline-primary\" title=\"View\">\n <i class=\"bi bi-eye\"></i>\n </a>\n <a href=\"{{url}}\" download=\"{{filename}}\" class=\"btn btn-sm btn-outline-secondary\" title=\"Download\">\n <i class=\"bi bi-download\"></i>\n </a>\n `\n }\n ]\n });\n\n // Create TabView, only showing Renditions tab if they exist\n const tabs = { 'Info': this.infoView };\n tabs['Renditions'] = this.renditionsView;\n\n this.tabView = new TabView({\n tabs: tabs,\n activeTab: 'Info',\n containerId: 'file-tabs'\n });\n this.addChild(this.tabView);\n\n // Create ContextMenu\n const fileMenu = new ContextMenu({\n containerId: 'file-context-menu',\n className: \"context-menu-view header-menu-absolute\",\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: [\n { label: 'View', action: 'view-file', icon: 'bi-eye' },\n { label: 'Download', action: 'download-file', icon: 'bi bi-download' },\n { label: 'Edit Details', action: 'edit-file', icon: 'bi bi-pencil' },\n { type: 'divider' },\n this.model.get('is_public')\n ? { label: 'Make Private', action: 'make-private', icon: 'bi bi-lock' }\n : { label: 'Make Public', action: 'make-public', icon: 'bi bi-unlock' },\n { type: 'divider' },\n { label: 'Delete File', action: 'delete-file', icon: 'bi bi-trash', danger: true }\n ]\n }\n });\n this.addChild(fileMenu);\n }\n\n async onActionViewFile() {\n const contentType = this.model.get('content_type');\n const fileUrl = this.model.get('url');\n\n if (contentType.startsWith('image/')) {\n const renditions = this.model.get('renditions') || {};\n const images = [\n { src: fileUrl, alt: 'Original' },\n ...Object.values(renditions).map(r => ({ src: r.url, alt: r.role }))\n ];\n LightboxGallery.show(images, { fitToScreen: false });\n } else if (contentType === 'application/pdf') {\n PDFViewer.showDialog(fileUrl, { title: this.model.get('filename') });\n } else {\n window.open(fileUrl, '_blank');\n }\n }\n\n async onActionDownloadFile() {\n const url = this.model.get('url');\n if (url) {\n const a = document.createElement('a');\n a.href = url;\n a.download = this.model.get('filename');\n document.body.appendChild(a);\n a.click();\n document.body.removeChild(a);\n }\n }\n\n async onActionEditFile() {\n const resp = await Dialog.showModelForm({\n title: `Edit File - ${this.model.get('filename')}`,\n model: this.model,\n formConfig: FileForms.edit,\n });\n if (resp) {\n this.render();\n }\n }\n\n async onActionMakePublic() {\n await this.model.save({ is_public: true });\n this.render();\n }\n\n async onActionMakePrivate() {\n await this.model.save({ is_public: false });\n this.render();\n }\n\n async onActionDeleteFile() {\n const confirmed = await Dialog.confirm(\n `Are you sure you want to delete the file \"${this.model.get('filename')}\"? This action cannot be undone.`,\n 'Confirm Deletion',\n { confirmClass: 'btn-danger', confirmText: 'Delete' }\n );\n if (confirmed) {\n const resp = await this.model.destroy();\n if (resp.success) {\n this.emit('file:deleted', { model: this.model });\n }\n }\n }\n}\n\nFile.VIEW_CLASS = FileView;\n\nexport default FileView;\n","/**\n * FileTablePage - File management using TablePage component\n * Clean implementation with file drop support\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { File, FileList, FileForms } from '@core/models/Files.js';\nimport FileView from './FileView.js';\nimport applyFileDropMixin from '@core/mixins/FileDropMixin.js';\n\nclass FileTablePage extends TablePage {\n constructor(options = {}) {\n super({\n name: 'admin_files',\n pageName: 'Manage Files',\n router: \"admin/files\",\n Collection: FileList,\n\n // Don't use formCreate - we have custom file upload handling\n formEdit: FileForms.edit,\n itemViewClass: FileView,\n \n // Custom add handler for file uploads\n onAdd: async (event) => {\n await this.handleFileUpload(event);\n },\n\n viewDialogOptions: {\n header: false,\n size: 'xl'\n },\n\n // Column definitions\n columns: [\n {\n key: 'id',\n label: 'ID',\n sortable: true,\n class: 'text-muted'\n },\n {\n key: 'filename',\n label: 'Filename'\n },\n {\n key: 'content_type',\n label: 'Type',\n formatter: \"default('Unknown')\"\n },\n {\n key: 'file_size',\n label: 'Size',\n formatter: \"filesize\"\n },\n {\n key: 'group.name',\n label: 'Group',\n formatter: \"default('No Group')\"\n },\n {\n key: 'upload_status',\n label: 'Status',\n formatter: \"badge\"\n },\n {\n key: 'created',\n label: 'Uploaded',\n formatter: \"epoch|datetime\"\n }\n ],\n\n // Table features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Toolbar\n showRefresh: true,\n showAdd: true,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No files found. Click \"Add File\" to upload your first file.',\n\n // Batch actions\n batchBarLocation: 'top',\n batchActions: [\n { label: \"Download\", icon: \"bi bi-download\", action: \"batch-download\" },\n { label: \"Delete\", icon: \"bi bi-trash\", action: \"batch-delete\" },\n { label: \"Move to Group\", icon: \"bi bi-folder\", action: \"batch-move\" }\n ],\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n },\n ...options,\n });\n\n // Enable file drop support\n this.enableFileDrop({\n acceptedTypes: ['*/*'],\n maxFileSize: 100 * 1024 * 1024, // 100MB\n multiple: false,\n validateOnDrop: true\n });\n }\n\n /**\n * Handle file upload via file picker\n * Opens native file picker and uses same upload flow as drag-drop\n */\n async handleFileUpload(event) {\n if (event) event.preventDefault();\n\n // Create hidden file input element\n const fileInput = document.createElement('input');\n fileInput.type = 'file';\n fileInput.accept = '*/*';\n fileInput.multiple = false;\n fileInput.style.display = 'none';\n\n // Handle file selection\n fileInput.addEventListener('change', async (e) => {\n const file = e.target.files[0];\n \n if (!file) {\n return;\n }\n\n // Validate file size (same as FileDropMixin config)\n const maxSize = 100 * 1024 * 1024; // 100MB\n if (file.size > maxSize) {\n this.showError(`File size (${this._formatFileSize(file.size)}) exceeds maximum (${this._formatFileSize(maxSize)})`);\n return;\n }\n\n // Use the same upload flow as drag-drop\n try {\n const fileModel = new File();\n let extra = {};\n if (this.options.requiresGroup && this.getApp().activeGroup) {\n extra.group = this.getApp().activeGroup.id;\n }\n\n const upload = fileModel.upload({\n file: file,\n name: file.name,\n description: `File uploaded on ${new Date().toLocaleDateString()}`,\n showToast: true,\n onProgress: (progressInfo) => {\n console.log(`Upload progress: ${progressInfo.percentage}%`);\n },\n onComplete: (result) => {\n console.log('Upload completed:', result);\n this.refresh();\n },\n onError: (error) => {\n console.error('Upload failed:', error);\n this.showError('Upload failed: ' + error.message);\n },\n ...extra\n });\n\n await upload;\n } catch (error) {\n console.error('Error starting file upload:', error);\n this.showError('Failed to start file upload: ' + error.message);\n } finally {\n // Clean up file input\n fileInput.remove();\n }\n });\n\n // Trigger file picker\n document.body.appendChild(fileInput);\n fileInput.click();\n }\n\n /**\n * Format file size for display\n * @param {number} bytes - File size in bytes\n * @returns {string} Formatted file size\n * @private\n */\n _formatFileSize(bytes) {\n if (bytes === 0) return '0 Bytes';\n const k = 1024;\n const sizes = ['Bytes', 'KB', 'MB', 'GB'];\n const i = Math.floor(Math.log(bytes) / Math.log(k));\n return parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + ' ' + sizes[i];\n }\n\n async onFileDrop(files, event, validation) {\n const file = files[0];\n console.log(`File Dropped: ${file.name} (${file.type}) (${file.size} bytes)`);\n\n try {\n // Create new File model instance\n const fileModel = new File();\n let extra = {};\n if (this.options.requiresGroup && this.getApp().activeGroup) {\n extra.group = this.getApp().activeGroup.id;\n }\n\n // Start upload with progress tracking\n const upload = fileModel.upload({\n file: file,\n name: file.name,\n description: `File uploaded via drag & drop on ${new Date().toLocaleDateString()}`,\n showToast: true,\n onProgress: (progressInfo) => {\n console.log(`Upload progress: ${progressInfo.percentage}%`);\n },\n onComplete: (result) => {\n console.log('Upload completed:', result);\n this.refresh();\n },\n onError: (error) => {\n console.error('Upload failed:', error);\n this.showError('Upload failed: ' + error.message);\n },\n ...extra\n });\n\n await upload;\n } catch (error) {\n console.error('Error starting file upload:', error);\n this.showError('Failed to start file upload: ' + error.message);\n }\n }\n}\n\n// Apply file drop mixin\napplyFileDropMixin(FileTablePage);\n\nexport default FileTablePage;\n","/**\n * S3BucketTablePage - S3 Bucket management using TablePage component\n * Clean implementation using TablePage with minimal overrides\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { S3BucketList, S3BucketForms } from '@core/models/AWS.js';\n\nclass S3BucketTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_s3_buckets',\n pageName: 'Manage S3 Buckets',\n router: \"admin/s3-buckets\",\n Collection: S3BucketList,\n \n formCreate: S3BucketForms.create,\n formEdit: S3BucketForms.edit,\n\n // Column definitions\n columns: [\n {\n key: 'id',\n label: 'ID',\n width: '60px',\n sortable: true,\n class: 'text-muted'\n },\n {\n key: 'name',\n label: 'Bucket Name',\n sortable: true\n },\n {\n key: 'created',\n label: 'Created',\n formatter: \"epoch|datetime\"\n }\n ],\n\n // Table features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Toolbar\n showRefresh: true,\n showAdd: true,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No S3 buckets found. Click \"Add S3 Bucket\" to create your first bucket.',\n\n // Batch actions\n batchBarLocation: 'top',\n batchActions: [\n { label: \"Delete\", icon: \"bi bi-trash\", action: \"batch-delete\" },\n { label: \"Export\", icon: \"bi bi-download\", action: \"batch-export\" },\n { label: \"Make Public\", icon: \"bi bi-unlock\", action: \"batch-public\" },\n { label: \"Make Private\", icon: \"bi bi-lock\", action: \"batch-private\" },\n { label: \"Empty Bucket\", icon: \"bi bi-bucket\", action: \"batch-empty\" }\n ],\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n }\n });\n }\n}\n\nexport default S3BucketTablePage;","/**\n * UserDeviceLocationView - Clean, modern session location detail view\n *\n * Shows a device-location record with:\n * - Header: device icon, browser/device summary, location, IP, threat badges\n * - SideNavView: Location, Device, Network & Risk, Map (if coords), Events\n *\n * Opened when clicking a location row in admin UserView or DeviceView.\n */\n\nimport View from '@core/View.js';\nimport SideNavView from '@core/views/navigation/SideNavView.js';\nimport TableView from '@core/views/table/TableView.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport { UserDeviceLocation } from '@core/models/User.js';\nimport { IncidentEventList } from '@core/models/Incident.js';\nimport { LogList } from '@core/models/Log.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\n\nclass UserDeviceLocationView extends View {\n constructor(options = {}) {\n super({\n className: 'udl-view',\n ...options\n });\n\n this.model = options.model || new UserDeviceLocation(options.data || {});\n\n // Extract nested data\n this._ud = this.model.get('user_device') || {};\n this._di = this._ud.device_info || {};\n this._geo = this.model.get('geolocation') || {};\n\n // Computed for template\n this.deviceIcon = this._getDeviceIcon();\n this.browserFull = this._getBrowser();\n this.osFull = this._getOS();\n this.deviceFull = this._getDevice();\n this.locationSummary = this._getLocationSummary();\n this.countryFlag = this._geo.country_code || '';\n this.threatLevel = this._geo.threat_level || 'unknown';\n this.threatColor = this._getThreatColor();\n this.hasCoordinates = !!(this._geo.latitude && this._geo.longitude);\n\n this.template = `\n <style>\n .udl-header { display: flex; justify-content: space-between; align-items: flex-start; margin-bottom: 1.5rem; }\n .udl-identity { display: flex; align-items: center; gap: 1rem; }\n .udl-icon-wrap { width: 48px; height: 48px; border-radius: 12px; display: flex; align-items: center; justify-content: center; font-size: 1.4rem; flex-shrink: 0; }\n .udl-title { font-size: 1.15rem; font-weight: 600; margin: 0; line-height: 1.3; }\n .udl-subtitle { font-size: 0.8rem; color: #6c757d; margin-top: 0.15rem; }\n .udl-right { display: flex; align-items: flex-start; gap: 0.75rem; }\n .udl-threat-label { font-size: 0.7rem; color: #adb5bd; text-transform: uppercase; letter-spacing: 0.04em; }\n .udl-threat-value { font-size: 1rem; font-weight: 600; }\n .udl-threat-flags { display: flex; gap: 0.35rem; margin-top: 0.35rem; }\n .udl-flag { font-size: 0.65rem; padding: 0.15em 0.45em; border-radius: 3px; font-weight: 600; }\n </style>\n\n <div class=\"udl-header\">\n <div class=\"udl-identity\">\n <div class=\"udl-icon-wrap bg-primary bg-opacity-10 text-primary\">\n <i class=\"bi {{deviceIcon}}\"></i>\n </div>\n <div>\n <h4 class=\"udl-title\">{{locationSummary}}</h4>\n <div class=\"udl-subtitle\">\n {{browserFull}} <span class=\"text-muted\">on</span> {{deviceFull}}\n <span class=\"text-muted mx-1\">&middot;</span>\n {{model.ip_address}}\n </div>\n </div>\n </div>\n <div class=\"udl-right\">\n <div class=\"text-end\">\n <div class=\"udl-threat-label\">Threat Level</div>\n <div class=\"udl-threat-value {{threatColor}}\">{{threatLevel|capitalize}}</div>\n <div class=\"udl-threat-flags\">\n {{#_geo.is_vpn}}<span class=\"udl-flag bg-warning text-dark\">VPN</span>{{/_geo.is_vpn}}\n {{#_geo.is_tor}}<span class=\"udl-flag bg-danger text-white\">Tor</span>{{/_geo.is_tor}}\n {{#_geo.is_proxy}}<span class=\"udl-flag bg-warning text-dark\">Proxy</span>{{/_geo.is_proxy}}\n {{#_geo.is_datacenter}}<span class=\"udl-flag bg-secondary text-white\">DC</span>{{/_geo.is_datacenter}}\n {{#_geo.is_cloud}}<span class=\"udl-flag bg-info text-white\">Cloud</span>{{/_geo.is_cloud}}\n </div>\n </div>\n <div data-container=\"udl-context-menu\"></div>\n </div>\n </div>\n\n <div data-container=\"udl-sidenav\" style=\"min-height: 300px;\"></div>\n `;\n }\n\n // ── Computed helpers ─────────────────────────\n\n _getDeviceIcon() {\n const browser = this._di?.user_agent?.family?.toLowerCase() || '';\n const os = this._di?.os?.family?.toLowerCase() || '';\n const device = this._di?.device?.family?.toLowerCase() || '';\n if (browser.includes('chrome')) return 'bi-browser-chrome';\n if (browser.includes('firefox')) return 'bi-browser-firefox';\n if (browser.includes('safari')) return 'bi-browser-safari';\n if (browser.includes('edge')) return 'bi-browser-edge';\n if (os.includes('mac') || os.includes('ios')) return 'bi-apple';\n if (os.includes('windows')) return 'bi-windows';\n if (os.includes('android')) return 'bi-android2';\n if (device.includes('iphone')) return 'bi-phone';\n if (device.includes('ipad')) return 'bi-tablet';\n return 'bi-geo-alt';\n }\n\n _getBrowser() {\n const ua = this._di?.user_agent || {};\n return ua.family ? `${ua.family} ${ua.major || ''}`.trim() : 'Unknown Browser';\n }\n\n _getOS() {\n const os = this._di?.os || {};\n const ver = [os.major, os.minor].filter(Boolean).join('.');\n return os.family ? `${os.family} ${ver}`.trim() : 'Unknown OS';\n }\n\n _getDevice() {\n const dev = this._di?.device || {};\n const parts = [dev.brand, dev.family].filter(Boolean);\n return parts.length ? parts.join(' ') : 'Unknown Device';\n }\n\n _getLocationSummary() {\n const parts = [this._geo.city, this._geo.region, this._geo.country_name].filter(Boolean);\n return parts.length ? parts.join(', ') : 'Unknown Location';\n }\n\n _getThreatColor() {\n const level = (this._geo.threat_level || '').toLowerCase();\n if (level === 'high' || this._geo.is_threat) return 'text-danger';\n if (level === 'medium' || this._geo.is_suspicious) return 'text-warning';\n if (level === 'low') return 'text-success';\n return 'text-muted';\n }\n\n async onInit() {\n const geo = this._geo;\n const di = this._di;\n\n // ── Location section ────────────────────────\n const locationView = new View({\n model: this.model,\n template: `\n <style>\n .udl-section-label { font-size: 0.7rem; font-weight: 700; text-transform: uppercase; letter-spacing: 0.06em; color: #adb5bd; margin-bottom: 0.5rem; margin-top: 1.5rem; }\n .udl-section-label:first-child { margin-top: 0; }\n .udl-field-row { display: flex; align-items: baseline; padding: 0.5rem 0; border-bottom: 1px solid #f0f0f0; }\n .udl-field-row:last-child { border-bottom: none; }\n .udl-field-label { width: 130px; font-size: 0.78rem; color: #6c757d; flex-shrink: 0; }\n .udl-field-value { flex: 1; font-size: 0.88rem; color: #212529; }\n </style>\n\n <div class=\"udl-section-label\">Geography</div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">City</div>\n <div class=\"udl-field-value\">${geo.city || '—'}</div>\n </div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">Region</div>\n <div class=\"udl-field-value\">${geo.region || '—'}</div>\n </div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">Country</div>\n <div class=\"udl-field-value\">${geo.country_name || '—'} ${geo.country_code ? `<span class=\"text-muted\">(${geo.country_code})</span>` : ''}</div>\n </div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">Postal Code</div>\n <div class=\"udl-field-value\">${geo.postal_code || '—'}</div>\n </div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">Timezone</div>\n <div class=\"udl-field-value\">${geo.timezone || '—'}</div>\n </div>\n ${geo.latitude ? `\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">Coordinates</div>\n <div class=\"udl-field-value\">${geo.latitude}, ${geo.longitude}</div>\n </div>` : ''}\n\n <div class=\"udl-section-label\">Network</div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">IP Address</div>\n <div class=\"udl-field-value\" style=\"font-family: ui-monospace, monospace; font-size: 0.82rem;\">{{model.ip_address}}</div>\n </div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">ISP</div>\n <div class=\"udl-field-value\">${geo.isp || '—'}</div>\n </div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">ASN</div>\n <div class=\"udl-field-value\">${geo.asn || '—'} ${geo.asn_org ? `<span class=\"text-muted small\">(${geo.asn_org})</span>` : ''}</div>\n </div>\n\n <div class=\"udl-section-label\">Timestamps</div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">First Seen</div>\n <div class=\"udl-field-value\">{{model.first_seen|epoch|datetime|default('—')}}</div>\n </div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">Last Seen</div>\n <div class=\"udl-field-value\">{{model.last_seen|epoch|datetime|default('—')}}</div>\n </div>\n `\n });\n\n // ── Device section ──────────────────────────\n const deviceView = new View({\n model: this.model,\n template: `\n <style>\n .udl-section-label { font-size: 0.7rem; font-weight: 700; text-transform: uppercase; letter-spacing: 0.06em; color: #adb5bd; margin-bottom: 0.5rem; margin-top: 1.5rem; }\n .udl-section-label:first-child { margin-top: 0; }\n .udl-field-row { display: flex; align-items: baseline; padding: 0.5rem 0; border-bottom: 1px solid #f0f0f0; }\n .udl-field-row:last-child { border-bottom: none; }\n .udl-field-label { width: 130px; font-size: 0.78rem; color: #6c757d; flex-shrink: 0; }\n .udl-field-value { flex: 1; font-size: 0.88rem; color: #212529; }\n .udl-ua-string { font-family: ui-monospace, monospace; font-size: 0.73rem; color: #6c757d; word-break: break-all; line-height: 1.5; padding: 0.5rem 0.75rem; background: #f8f9fa; border-radius: 6px; margin-top: 0.25rem; }\n </style>\n\n <div class=\"udl-section-label\">Browser</div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">Name</div>\n <div class=\"udl-field-value\">${di?.user_agent?.family || '—'}</div>\n </div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">Version</div>\n <div class=\"udl-field-value\">${[di?.user_agent?.major, di?.user_agent?.minor, di?.user_agent?.patch].filter(Boolean).join('.') || '—'}</div>\n </div>\n\n <div class=\"udl-section-label\">Operating System</div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">Name</div>\n <div class=\"udl-field-value\">${di?.os?.family || '—'}</div>\n </div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">Version</div>\n <div class=\"udl-field-value\">${[di?.os?.major, di?.os?.minor, di?.os?.patch].filter(Boolean).join('.') || '—'}</div>\n </div>\n\n <div class=\"udl-section-label\">Hardware</div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">Brand</div>\n <div class=\"udl-field-value\">${di?.device?.brand || '—'}</div>\n </div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">Family</div>\n <div class=\"udl-field-value\">${di?.device?.family || '—'}</div>\n </div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">Model</div>\n <div class=\"udl-field-value\">${di?.device?.model || '—'}</div>\n </div>\n\n ${di?.string ? `\n <div class=\"udl-section-label\">User Agent String</div>\n <div class=\"udl-ua-string\">${di.string}</div>` : ''}\n `\n });\n\n // ── Risk & Reputation section ───────────────\n const riskView = new View({\n model: this.model,\n template: `\n <style>\n .udl-section-label { font-size: 0.7rem; font-weight: 700; text-transform: uppercase; letter-spacing: 0.06em; color: #adb5bd; margin-bottom: 0.5rem; margin-top: 1.5rem; }\n .udl-section-label:first-child { margin-top: 0; }\n .udl-field-row { display: flex; align-items: baseline; padding: 0.5rem 0; border-bottom: 1px solid #f0f0f0; }\n .udl-field-row:last-child { border-bottom: none; }\n .udl-field-label { width: 130px; font-size: 0.78rem; color: #6c757d; flex-shrink: 0; }\n .udl-field-value { flex: 1; font-size: 0.88rem; color: #212529; }\n .udl-risk-icon { font-size: 0.85rem; margin-right: 0.35rem; }\n .udl-risk-yes { color: #dc3545; }\n .udl-risk-no { color: #adb5bd; }\n </style>\n\n <div class=\"udl-section-label\">Threat Assessment</div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">Threat Level</div>\n <div class=\"udl-field-value ${this.threatColor}\" style=\"font-weight: 600;\">${(geo.threat_level || 'Unknown')}</div>\n </div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">Risk Score</div>\n <div class=\"udl-field-value\">${geo.risk_score != null ? geo.risk_score : '—'}</div>\n </div>\n\n <div class=\"udl-section-label\">Detection Flags</div>\n ${this._riskRow('VPN', 'bi-shield', geo.is_vpn)}\n ${this._riskRow('Tor Exit Node', 'bi-shield-lock', geo.is_tor)}\n ${this._riskRow('Proxy', 'bi-diagram-3', geo.is_proxy)}\n ${this._riskRow('Cloud Provider', 'bi-cloud', geo.is_cloud)}\n ${this._riskRow('Datacenter', 'bi-hdd-stack', geo.is_datacenter)}\n ${this._riskRow('Mobile', 'bi-phone', geo.is_mobile)}\n\n <div class=\"udl-section-label\">Reputation</div>\n ${this._riskRow('Known Attacker', 'bi-exclamation-triangle', geo.is_known_attacker)}\n ${this._riskRow('Known Abuser', 'bi-flag', geo.is_known_abuser)}\n ${this._riskRow('Threat', 'bi-shield-exclamation', geo.is_threat)}\n ${this._riskRow('Suspicious', 'bi-question-circle', geo.is_suspicious)}\n `\n });\n\n // ── Build sections array ────────────────────\n const sections = [\n { key: 'location', label: 'Location', icon: 'bi-geo-alt', view: locationView },\n { key: 'device', label: 'Device', icon: 'bi-laptop', view: deviceView },\n { key: 'risk', label: 'Risk', icon: 'bi-shield-exclamation', view: riskView }\n ];\n\n // Map (if coordinates)\n if (this.hasCoordinates) {\n try {\n const MapView = (await import('@ext/map/MapView.js')).default;\n const mapView = new MapView({\n markers: [{\n lat: this._geo.latitude,\n lng: this._geo.longitude,\n popup: `<strong>${this.model.get('ip_address')}</strong><br>${this.locationSummary}`\n }],\n tileLayer: 'light',\n zoom: 6,\n height: 400\n });\n sections.push({ key: 'map', label: 'Map', icon: 'bi-map', view: mapView });\n } catch (e) {\n // MapView extension not available — skip\n }\n }\n\n // Events (by IP)\n const ip = this.model.get('ip_address');\n if (ip) {\n const eventsView = new TableView({\n collection: new IncidentEventList({\n params: { size: 10, source_ip: ip }\n }),\n hideActivePillNames: ['source_ip'],\n columns: [\n { key: 'created', label: 'Date', formatter: 'datetime', sortable: true, width: '150px' },\n { key: 'category|badge', label: 'Category' },\n { key: 'title', label: 'Title' }\n ]\n });\n sections.push({ type: 'divider', label: 'Activity' });\n sections.push({ key: 'events', label: 'Events', icon: 'bi-calendar-event', view: eventsView });\n\n const logsView = new TableView({\n collection: new LogList({\n params: { size: 10, ip }\n }),\n permissions: 'view_logs',\n hideActivePillNames: ['ip'],\n columns: [\n { key: 'created', label: 'Timestamp', sortable: true, formatter: 'epoch|datetime' },\n { key: 'level', label: 'Level', sortable: true },\n { key: 'kind', label: 'Kind' },\n { name: 'log', label: 'Log' }\n ]\n });\n sections.push({ key: 'logs', label: 'Logs', icon: 'bi-journal-text', view: logsView, permissions: 'view_logs' });\n }\n\n // ── SideNavView ─────────────────────────────\n this.sideNavView = new SideNavView({\n containerId: 'udl-sidenav',\n activeSection: this.hasCoordinates ? 'map' : 'location',\n navWidth: 160,\n contentPadding: '1rem 1.5rem',\n enableResponsive: true,\n minWidth: 450,\n sections\n });\n this.addChild(this.sideNavView);\n\n // ── Context Menu ────────────────────────────\n const menu = new ContextMenu({\n containerId: 'udl-context-menu',\n className: 'context-menu-view header-menu-absolute',\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: [\n ...(this._ud?.user ? [{ label: 'View User', action: 'view-user', icon: 'bi-person' }] : []),\n ...(this._ud?.id ? [{ label: 'View Device', action: 'view-device', icon: 'bi-laptop' }] : []),\n ...(this.hasCoordinates ? [{\n label: 'Open in Maps',\n action: 'open-in-maps',\n icon: 'bi-box-arrow-up-right'\n }] : []),\n { type: 'divider' },\n { label: 'Delete Record', action: 'delete-record', icon: 'bi-trash', danger: true }\n ]\n }\n });\n this.addChild(menu);\n }\n\n // ── Helpers ──────────────────────────────────\n\n _riskRow(label, icon, value) {\n const cls = value ? 'udl-risk-yes' : 'udl-risk-no';\n const indicator = value\n ? '<i class=\"bi bi-check-circle-fill udl-risk-icon udl-risk-yes\"></i>Yes'\n : '<i class=\"bi bi-dash-circle udl-risk-icon udl-risk-no\"></i>No';\n return `\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\"><i class=\"bi ${icon} me-1 ${cls}\"></i>${label}</div>\n <div class=\"udl-field-value\">${indicator}</div>\n </div>`;\n }\n\n // ── Actions ──────────────────────────────────\n\n async onActionViewUser() {\n const userId = this._ud?.user?.id || this._ud?.user;\n if (userId) this.emit('view-user', { userId });\n }\n\n async onActionViewDevice() {\n const deviceId = this._ud?.id;\n if (deviceId) this.emit('view-device', { deviceId });\n }\n\n async onActionOpenInMaps() {\n if (this.hasCoordinates) {\n window.open(`https://www.google.com/maps/search/?api=1&query=${this._geo.latitude},${this._geo.longitude}`, '_blank');\n }\n }\n\n async onActionDeleteRecord() {\n const confirmed = await Dialog.confirm(\n 'Are you sure you want to delete this location record?',\n 'Delete Location Record'\n );\n if (!confirmed) return true;\n\n const resp = await this.model.destroy();\n if (resp.success) {\n this.emit('location:deleted', { model: this.model });\n }\n return true;\n }\n\n static async show(id) {\n const model = new UserDeviceLocation({ id });\n await model.fetch();\n if (model.id) {\n return Dialog.showDialog({\n title: false,\n size: 'lg',\n body: new UserDeviceLocationView({ model }),\n buttons: [{ text: 'Close', class: 'btn-secondary', dismiss: true }]\n });\n }\n Dialog.alert({ message: `Could not find location record: ${id}`, type: 'warning' });\n return null;\n }\n}\n\nUserDeviceLocation.VIEW_CLASS = UserDeviceLocationView;\nexport default UserDeviceLocationView;\n","/**\n * CloudWatchResourceView - Detailed metric view for a single AWS resource\n *\n * Shows all available metric categories for an EC2, RDS, or Redis resource\n * using CloudWatchChart (MetricsChart) instances in a responsive grid.\n * Each chart has its own granularity/date controls.\n * Can be opened in a Dialog via the static show() method.\n */\nimport View from '@core/View.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\nimport CloudWatchChart from './CloudWatchChart.js';\n\nconst METRIC_CATEGORIES = {\n ec2: [\n { key: 'cpu', label: 'CPU Utilization', unit: '%' },\n { key: 'memory', label: 'Memory Usage', unit: '%' },\n { key: 'disk', label: 'Disk Usage', unit: '%' },\n { key: 'net_in', label: 'Network In', unit: 'bytes' },\n { key: 'net_out', label: 'Network Out', unit: 'bytes' },\n { key: 'disk_read', label: 'Disk Read Ops', unit: 'ops' },\n { key: 'disk_write', label: 'Disk Write Ops', unit: 'ops' },\n { key: 'status_check', label: 'Status Check', unit: '' }\n ],\n rds: [\n { key: 'cpu', label: 'CPU Utilization', unit: '%' },\n { key: 'conns', label: 'Active Connections', unit: '' },\n { key: 'free_storage', label: 'Free Storage', unit: 'bytes' },\n { key: 'free_memory', label: 'Freeable Memory', unit: 'bytes' },\n { key: 'read_iops', label: 'Read IOPS', unit: 'ops/s' },\n { key: 'write_iops', label: 'Write IOPS', unit: 'ops/s' },\n { key: 'read_latency', label: 'Read Latency', unit: 's' },\n { key: 'write_latency', label: 'Write Latency', unit: 's' },\n { key: 'net_in', label: 'Network In', unit: 'bytes' },\n { key: 'net_out', label: 'Network Out', unit: 'bytes' }\n ],\n redis: [\n { key: 'cpu', label: 'CPU Utilization', unit: '%' },\n { key: 'conns', label: 'Current Connections', unit: '' },\n { key: 'cache_memory', label: 'Cache Memory Used', unit: 'bytes' },\n { key: 'cache_hits', label: 'Cache Hits', unit: '' },\n { key: 'cache_misses', label: 'Cache Misses', unit: '' },\n { key: 'replication_lag', label: 'Replication Lag', unit: 's' },\n { key: 'net_in', label: 'Network In', unit: 'bytes' },\n { key: 'net_out', label: 'Network Out', unit: 'bytes' }\n ]\n};\n\nconst TYPE_ICONS = { ec2: 'bi-pc-display', rds: 'bi-database', redis: 'bi-lightning-charge' };\nconst TYPE_LABELS = { ec2: 'EC2 Instance', rds: 'RDS Database', redis: 'ElastiCache Redis' };\n\nfunction yAxisForUnit(unit) {\n if (unit === '%') return { label: '%', beginAtZero: true, max: 100 };\n if (unit === 'bytes') return { label: 'Bytes', beginAtZero: true };\n if (unit === 's') return { label: 'Seconds', beginAtZero: true };\n return { beginAtZero: true };\n}\n\nexport default class CloudWatchResourceView extends View {\n constructor(options = {}) {\n super({\n className: 'cloudwatch-resource-view',\n ...options\n });\n\n this.resourceType = options.resourceType || 'ec2';\n this.slug = options.slug || '';\n this.resource = options.resource || {};\n }\n\n async getTemplate() {\n const categories = METRIC_CATEGORIES[this.resourceType] || [];\n const icon = TYPE_ICONS[this.resourceType] || 'bi-cloud';\n const typeLabel = TYPE_LABELS[this.resourceType] || 'Resource';\n\n const state = this.resource.state || this.resource.status || 'unknown';\n const metaItems = this._buildMetaItems();\n const metaHtml = metaItems.map(m => `<span class=\"me-3\" style=\"font-size: 0.78rem; color: #6c757d;\">${m}</span>`).join('');\n\n return `\n <style>\n .cwrv-header { padding: 1rem 0; border-bottom: 1px solid #e9ecef; margin-bottom: 1rem; }\n .cwrv-name { font-size: 1.15rem; font-weight: 700; }\n .cwrv-grid { display: grid; grid-template-columns: 1fr 1fr; gap: 1rem; }\n @media (max-width: 768px) { .cwrv-grid { grid-template-columns: 1fr; } }\n </style>\n\n <div class=\"cwrv-header\">\n <div class=\"cwrv-name\">\n <i class=\"bi ${icon} me-2\"></i>${this.slug}\n <span class=\"badge bg-secondary ms-2\" style=\"font-size: 0.7rem;\">${typeLabel}</span>\n </div>\n <div class=\"mt-1\">${metaHtml}</div>\n </div>\n\n <div class=\"cwrv-grid\">\n ${categories.map((_, i) => `<div id=\"cwrv-chart-${i}\"></div>`).join('')}\n </div>\n `;\n }\n\n async onInit() {\n const categories = METRIC_CATEGORIES[this.resourceType] || [];\n\n for (let i = 0; i < categories.length; i++) {\n const cat = categories[i];\n const chart = new CloudWatchChart({\n containerId: `cwrv-chart-${i}`,\n account: this.resourceType,\n category: cat.key,\n slug: this.slug,\n title: cat.label,\n height: 200,\n yAxis: yAxisForUnit(cat.unit),\n showGranularity: true,\n showDateRange: false,\n defaultDateRange: '24h',\n granularity: 'hours'\n });\n this.addChild(chart);\n }\n }\n\n _buildMetaItems() {\n const r = this.resource;\n switch (this.resourceType) {\n case 'ec2':\n return [r.instance_type, r.private_ip, r.public_ip].filter(Boolean)\n .map((v, i) => {\n const icons = ['bi-cpu', 'bi-hdd-network', 'bi-globe'];\n return `<i class=\"bi ${icons[i]} me-1\"></i>${v}`;\n });\n case 'rds':\n return [r.engine, r.instance_class].filter(Boolean)\n .map((v, i) => {\n const icons = ['bi-database', 'bi-cpu'];\n return `<i class=\"bi ${icons[i]} me-1\"></i>${v}`;\n });\n case 'redis':\n return [r.engine, r.node_type, r.num_nodes ? `${r.num_nodes} node${r.num_nodes > 1 ? 's' : ''}` : ''].filter(Boolean)\n .map((v, i) => {\n const icons = ['bi-lightning', 'bi-cpu', 'bi-diagram-3'];\n return `<i class=\"bi ${icons[i]} me-1\"></i>${v}`;\n });\n default:\n return [];\n }\n }\n\n static async show(resourceType, slug, resource = {}, options = {}) {\n const view = new CloudWatchResourceView({\n resourceType, slug, resource\n });\n\n const icon = TYPE_ICONS[resourceType] || 'bi-cloud';\n const typeLabel = TYPE_LABELS[resourceType] || 'Resource';\n\n await Dialog.showDialog(view, {\n header: `<i class=\"bi ${icon} me-2\"></i>${slug} <small class=\"text-muted\">— ${typeLabel}</small>`,\n size: 'xl',\n scrollable: true\n });\n }\n}\n","/**\n * MOJO Admin Extension - Entry (2.1.0)\n */\n\n// Bundle admin CSS\nimport '@ext/admin/css/admin.css';\n\n// Admin Pages\nexport { default as AdminDashboardPage } from '@ext/admin/account/AdminDashboardPage.js';\nexport { default as UserTablePage } from '@ext/admin/account/users/UserTablePage.js';\nexport { default as MemberTablePage } from '@ext/admin/account/users/MemberTablePage.js';\nexport { default as GroupTablePage } from '@ext/admin/account/groups/GroupTablePage.js';\nexport { default as UserDeviceTablePage } from '@ext/admin/account/devices/UserDeviceTablePage.js';\nexport { default as UserDeviceLocationTablePage } from '@ext/admin/account/devices/UserDeviceLocationTablePage.js';\nexport { default as GeoLocatedIPTablePage } from '@ext/admin/account/devices/GeoLocatedIPTablePage.js';\nexport { default as ApiKeyTablePage } from '@ext/admin/account/api_keys/ApiKeyTablePage.js';\n\nexport { default as CloudWatchDashboardPage } from '@ext/admin/aws/CloudWatchDashboardPage.js';\nexport { default as CloudWatchChart } from '@ext/admin/aws/CloudWatchChart.js';\n\nexport { default as IncidentDashboardPage } from '@ext/admin/incidents/IncidentDashboardPage.js';\nexport { default as IncidentTablePage } from '@ext/admin/incidents/IncidentTablePage.js';\nexport { default as EventTablePage } from '@ext/admin/incidents/EventTablePage.js';\nexport { default as TicketTablePage } from '@ext/admin/incidents/TicketTablePage.js';\nexport { default as RuleSetTablePage } from '@ext/admin/incidents/RuleSetTablePage.js';\n\nexport { default as EmailDomainTablePage } from '@ext/admin/messaging/email/EmailDomainTablePage.js';\nexport { default as EmailMailboxTablePage } from '@ext/admin/messaging/email/EmailMailboxTablePage.js';\nexport { default as EmailTemplateTablePage } from '@ext/admin/messaging/email/EmailTemplateTablePage.js';\nexport { default as SentMessageTablePage } from '@ext/admin/messaging/email/SentMessageTablePage.js';\nexport { default as PhoneNumberTablePage } from '@ext/admin/messaging/sms/PhoneNumberTablePage.js';\nexport { default as SMSTablePage } from '@ext/admin/messaging/sms/SMSTablePage.js';\nexport { default as PushDashboardPage } from '@ext/admin/messaging/push/PushDashboardPage.js';\nexport { default as PushConfigTablePage } from '@ext/admin/messaging/push/PushConfigTablePage.js';\nexport { default as PushTemplateTablePage } from '@ext/admin/messaging/push/PushTemplateTablePage.js';\nexport { default as PushDeliveryTablePage } from '@ext/admin/messaging/push/PushDeliveryTablePage.js';\nexport { default as PushDeviceTablePage } from '@ext/admin/messaging/push/PushDeviceTablePage.js';\n\nexport { default as JobsAdminPage } from '@ext/admin/jobs/JobsAdminPage.js';\nexport { default as TaskManagementPage } from '@ext/admin/jobs/TaskManagementPage.js';\n\nexport { default as LogTablePage } from '@ext/admin/monitoring/LogTablePage.js';\nexport { default as MetricsPermissionsTablePage } from '@ext/admin/monitoring/MetricsPermissionsTablePage.js';\n\nexport { default as SettingTablePage } from '@ext/admin/settings/SettingTablePage.js';\n\nexport { default as FileManagerTablePage } from '@ext/admin/storage/FileManagerTablePage.js';\nexport { default as FileTablePage } from '@ext/admin/storage/FileTablePage.js';\nexport { default as S3BucketTablePage } from '@ext/admin/storage/S3BucketTablePage.js';\n\n// Admin Views\nexport { default as DeviceView } from '@ext/admin/account/devices/DeviceView.js';\nexport { default as GeoIPView } from '@ext/admin/account/devices/GeoIPView.js';\nexport { default as UserDeviceLocationView } from '@ext/admin/account/devices/UserDeviceLocationView.js';\nexport { default as GroupView } from '@ext/admin/account/groups/GroupView.js';\nexport { default as ApiKeyView } from '@ext/admin/account/api_keys/ApiKeyView.js';\nexport { default as CloudWatchResourceView } from '@ext/admin/aws/CloudWatchResourceView.js';\nexport { default as MemberView } from '@ext/admin/account/users/MemberView.js';\nexport { default as UserView } from '@ext/admin/account/users/UserView.js';\n\nexport { default as IncidentView } from '@ext/admin/incidents/IncidentView.js';\nexport { default as EventView } from '@ext/admin/incidents/EventView.js';\nexport { default as TicketView } from '@ext/admin/incidents/TicketView.js';\nexport { default as RuleSetView } from '@ext/admin/incidents/RuleSetView.js';\n\nexport { default as EmailTemplateView } from '@ext/admin/messaging/email/EmailTemplateView.js';\nexport { default as EmailView } from '@ext/admin/messaging/email/EmailView.js';\nexport { default as PhoneNumberView } from '@ext/admin/messaging/sms/PhoneNumberView.js';\nexport { default as PushDeliveryView } from '@ext/admin/messaging/push/PushDeliveryView.js';\nexport { default as PushDeviceView } from '@ext/admin/messaging/push/PushDeviceView.js';\n\nexport { default as JobDetailsView } from '@ext/admin/jobs/JobDetailsView.js';\nexport { default as JobHealthView } from '@ext/admin/jobs/JobHealthView.js';\nexport { default as JobStatsView } from '@ext/admin/jobs/JobStatsView.js';\n\nexport { default as LogView } from '@ext/admin/monitoring/LogView.js';\nexport { default as MetricsPermissionsView } from '@ext/admin/monitoring/MetricsPermissionsView.js';\n\nexport { default as SettingView } from '@ext/admin/settings/SettingView.js';\n\nexport { default as FileView } from '@ext/admin/storage/FileView.js';\n\n// Admin Components\nexport { default as RunnerDetailsView } from '@ext/admin/jobs/RunnerDetailsView.js';\nexport { default as TaskDetailsView } from '@ext/admin/jobs/TaskDetailsView.js';\n// Convenience\nexport { default as WebApp } from '@core/WebApp.js';\n\n// Version info passthrough\nexport {\n VERSION_INFO,\n VERSION,\n VERSION_MAJOR,\n VERSION_MINOR,\n VERSION_REVISION,\n BUILD_TIME\n} from './version.js';\n\n\n\n// Import all admin page classes for the register function\nimport AdminDashboardPageClass from '@ext/admin/account/AdminDashboardPage.js';\nimport UserTablePageClass from '@ext/admin/account/users/UserTablePage.js';\nimport MemberTablePageClass from '@ext/admin/account/users/MemberTablePage.js';\nimport GroupTablePageClass from '@ext/admin/account/groups/GroupTablePage.js';\nimport UserDeviceTablePageClass from '@ext/admin/account/devices/UserDeviceTablePage.js';\nimport UserDeviceLocationTablePageClass from '@ext/admin/account/devices/UserDeviceLocationTablePage.js';\nimport GeoLocatedIPTablePageClass from '@ext/admin/account/devices/GeoLocatedIPTablePage.js';\nimport ApiKeyTablePageClass from '@ext/admin/account/api_keys/ApiKeyTablePage.js';\nimport CloudWatchDashboardPageClass from '@ext/admin/aws/CloudWatchDashboardPage.js';\n\nimport IncidentDashboardPageClass from '@ext/admin/incidents/IncidentDashboardPage.js';\nimport IncidentTablePageClass from '@ext/admin/incidents/IncidentTablePage.js';\nimport EventTablePageClass from '@ext/admin/incidents/EventTablePage.js';\nimport TicketTablePageClass from '@ext/admin/incidents/TicketTablePage.js';\nimport RuleSetTablePageClass from '@ext/admin/incidents/RuleSetTablePage.js';\n\nimport EmailDomainTablePageClass from '@ext/admin/messaging/email/EmailDomainTablePage.js';\nimport EmailMailboxTablePageClass from '@ext/admin/messaging/email/EmailMailboxTablePage.js';\nimport EmailTemplateTablePageClass from '@ext/admin/messaging/email/EmailTemplateTablePage.js';\nimport SentMessageTablePageClass from '@ext/admin/messaging/email/SentMessageTablePage.js';\nimport PhoneNumberTablePageClass from '@ext/admin/messaging/sms/PhoneNumberTablePage.js';\nimport SMSTablePageClass from '@ext/admin/messaging/sms/SMSTablePage.js';\nimport PushDashboardPageClass from '@ext/admin/messaging/push/PushDashboardPage.js';\nimport PushConfigTablePageClass from '@ext/admin/messaging/push/PushConfigTablePage.js';\nimport PushTemplateTablePageClass from '@ext/admin/messaging/push/PushTemplateTablePage.js';\nimport PushDeliveryTablePageClass from '@ext/admin/messaging/push/PushDeliveryTablePage.js';\nimport PushDeviceTablePageClass from '@ext/admin/messaging/push/PushDeviceTablePage.js';\n\nimport JobsAdminPageClass from '@ext/admin/jobs/JobsAdminPage.js';\nimport TaskManagementPageClass from '@ext/admin/jobs/TaskManagementPage.js';\n\nimport LogTablePageClass from '@ext/admin/monitoring/LogTablePage.js';\nimport MetricsPermissionsTablePageClass from '@ext/admin/monitoring/MetricsPermissionsTablePage.js';\n\nimport SettingTablePageClass from '@ext/admin/settings/SettingTablePage.js';\n\nimport FileManagerTablePageClass from '@ext/admin/storage/FileManagerTablePage.js';\nimport FileTablePageClass from '@ext/admin/storage/FileTablePage.js';\nimport S3BucketTablePageClass from '@ext/admin/storage/S3BucketTablePage.js';\n\n/**\n * Register all admin pages to a WebApp instance\n * @param {WebApp} app - The WebApp instance to register pages to\n * @returns {void}\n */\nexport function registerSystemPages(app, addToMenu = true) {\n // Register all admin pages with consistent naming\n app.registerPage('system/dashboard', AdminDashboardPageClass, {permissions: [\"view_admin\"]});\n app.registerPage('system/jobs', JobsAdminPageClass, {permissions: [\"view_jobs\", \"manage_jobs\"]});\n app.registerPage('system/users', UserTablePageClass, {permissions: [\"manage_users\"]});\n app.registerPage('system/groups', GroupTablePageClass, {permissions: [\"manage_groups\"]});\n app.registerPage('system/members', MemberTablePageClass, {permissions: [\"manage_members\"]});\n app.registerPage('system/s3buckets', S3BucketTablePageClass, {permissions: [\"manage_aws\"]});\n app.registerPage('system/filemanagers', FileManagerTablePageClass, {permissions: [\"manage_files\"]});\n app.registerPage('system/files', FileTablePageClass, {permissions: [\"manage_files\"]});\n app.registerPage('system/incidents', IncidentTablePageClass, {permissions: [\"view_incidents\"]});\n app.registerPage('system/events', EventTablePageClass, {permissions: [\"view_incidents\"]});\n app.registerPage('system/logs', LogTablePageClass, {permissions: [\"view_logs\"]});\n app.registerPage('system/user/devices', UserDeviceTablePageClass, {permissions: [\"manage_users\"]});\n app.registerPage('system/user/device-locations', UserDeviceLocationTablePageClass, {permissions: [\"manage_users\"]});\n app.registerPage('system/system/geoip', GeoLocatedIPTablePageClass, {permissions: [\"manage_users\"]});\n app.registerPage('system/email/mailboxes', EmailMailboxTablePageClass, {permissions: [\"manage_aws\"]});\n app.registerPage('system/email/domains', EmailDomainTablePageClass, {permissions: [\"manage_aws\"]});\n app.registerPage('system/email/sent', SentMessageTablePageClass, {permissions: [\"manage_aws\"]});\n app.registerPage('system/email/templates', EmailTemplateTablePageClass, {permissions: [\"manage_aws\"]});\n app.registerPage('system/incident-dashboard', IncidentDashboardPageClass, { permissions: [\"view_incidents\"] });\n app.registerPage('system/rulesets', RuleSetTablePageClass, { permissions: [\"manage_incidents\"] });\n app.registerPage('system/tickets', TicketTablePageClass, { permissions: [\"manage_incidents\"] });\n app.registerPage('system/metrics/permissions', MetricsPermissionsTablePageClass, { permissions: [\"manage_metrics\"] });\n app.registerPage('system/push/dashboard', PushDashboardPageClass, { permissions: [\"manage_users\"] });\n app.registerPage('system/push/configs', PushConfigTablePageClass, { permissions: [\"manage_users\"] });\n app.registerPage('system/push/templates', PushTemplateTablePageClass, { permissions: [\"manage_users\"] });\n app.registerPage('system/push/deliveries', PushDeliveryTablePageClass, { permissions: [\"manage_users\"] });\n app.registerPage('system/push/devices', PushDeviceTablePageClass, { permissions: [\"manage_users\"] });\n app.registerPage('system/phonehub/numbers', PhoneNumberTablePageClass, { permissions: [\"manage_users\"] });\n app.registerPage('system/phonehub/sms', SMSTablePageClass, { permissions: [\"manage_users\"] });\n app.registerPage('system/api-keys', ApiKeyTablePageClass, { permissions: [\"manage_groups\", \"manage_group\"] });\n app.registerPage('system/settings', SettingTablePageClass, { permissions: [\"manage_settings\"] });\n app.registerPage('system/cloudwatch', CloudWatchDashboardPageClass, { permissions: [\"view_admin\"] });\n\n // Check if sidebar exists and has an admin menu config\n if (addToMenu && app.sidebar && app.sidebar.getMenuConfig) {\n const adminMenuConfig = app.sidebar.getMenuConfig('system');\n if (adminMenuConfig && adminMenuConfig.items) {\n // Add admin pages to sidebar menu\n const adminMenuItems = [\n {\n text: 'Dashboard',\n route: '?page=system/dashboard',\n icon: 'bi-speedometer2',\n permissions: [\"view_admin\"]\n },\n {\n text: 'Jobs Management',\n route: '?page=system/jobs',\n icon: 'bi-gear-wide-connected',\n permissions: [\"view_jobs\", \"manage_jobs\"]\n },\n {\n text: 'Users',\n route: '?page=system/users',\n icon: 'bi-people',\n permissions: [\"manage_users\"]\n },\n {\n text: 'Groups',\n route: '?page=system/groups',\n icon: 'bi-diagram-3',\n permissions: [\"manage_groups\"]\n },\n {\n text: 'Incidents & Tickets',\n route: null,\n icon: 'bi-shield-exclamation',\n permissions: [\"view_incidents\"],\n children: [\n {\n text: 'Dashboard',\n route: '?page=system/incident-dashboard',\n icon: 'bi-bar-chart-line',\n permissions: [\"view_incidents\"]\n },\n {\n text: 'Incidents',\n route: '?page=system/incidents',\n icon: 'bi-exclamation-triangle',\n permissions: [\"view_incidents\"]\n },\n {\n text: 'Tickets',\n route: '?page=system/tickets',\n icon: 'bi-ticket-detailed',\n permissions: [\"manage_incidents\"]\n },\n {\n text: 'Events',\n route: '?page=system/events',\n icon: 'bi-bell',\n permissions: [\"view_incidents\"]\n },\n {\n text: 'Rule Engine',\n route: '?page=system/rulesets',\n icon: 'bi-gear-wide-connected',\n permissions: [\"manage_incidents\"]\n },\n ]\n },\n {\n text: 'Security',\n route: null,\n icon: 'bi-shield',\n permissions: [\"manage_groups\"],\n children: [\n {\n text: 'Logs',\n route: '?page=system/logs',\n icon: 'bi-journal-text',\n permissions: [\"view_logs\"]\n },\n {\n text: 'User Devices',\n route: '?page=system/user/devices',\n icon: 'bi-phone',\n permissions: [\"manage_users\"]\n },\n {\n text: 'Device Locations',\n route: '?page=system/user/device-locations',\n icon: 'bi-geo-alt',\n permissions: [\"manage_users\"]\n },\n {\n text: 'GeoIP Cache',\n route: '?page=system/system/geoip',\n icon: 'bi-globe',\n permissions: [\"manage_users\"]\n },\n {\n text: 'Metrics Permissions',\n route: '?page=system/metrics/permissions',\n icon: 'bi-bar-chart-line',\n permissions: [\"manage_metrics\"]\n },\n {\n text: 'API Keys',\n route: '?page=system/api-keys',\n icon: 'bi-key',\n permissions: [\"manage_groups\", \"manage_group\"]\n },\n {\n text: 'Settings',\n route: '?page=system/settings',\n icon: 'bi-gear',\n permissions: [\"manage_settings\"]\n }\n ]\n },\n {\n text: 'Storage',\n route: null,\n icon: 'bi-folder',\n permissions: [\"manage_files\", \"manage_aws\"],\n children: [\n {\n text: 'S3 Buckets',\n route: '?page=system/s3buckets',\n icon: 'bi-bucket',\n permissions: [\"manage_aws\"]\n },\n {\n text: 'Storage Backends',\n route: '?page=system/filemanagers',\n icon: 'bi-hdd-stack',\n permissions: [\"manage_aws\"]\n },\n {\n text: 'Files',\n route: '?page=system/files',\n icon: 'bi-file-earmark',\n permissions: [\"manage_files\"]\n },\n ]\n },\n {\n text: 'Push Notifications',\n route: null,\n icon: 'bi-broadcast',\n permissions: [\"manage_users\"],\n children: [\n { text: 'Dashboard', route: '?page=system/push/dashboard', icon: 'bi-bar-chart-line' },\n { text: 'Configurations', route: '?page=system/push/configs', icon: 'bi-gear' },\n { text: 'Templates', route: '?page=system/push/templates', icon: 'bi-file-earmark-text' },\n { text: 'Deliveries', route: '?page=system/push/deliveries', icon: 'bi-send' },\n { text: 'Devices', route: '?page=system/push/devices', icon: 'bi-phone' },\n ]\n },\n {\n text: 'Email Admin',\n route: null,\n icon: 'bi-envelope',\n permissions: [\"manage_aws\"],\n children: [\n {\n text: 'Domains',\n route: '?page=system/email/domains',\n icon: 'bi-globe',\n permissions: [\"manage_aws\"]\n },\n {\n text: 'Mailboxes',\n route: '?page=system/email/mailboxes',\n icon: 'bi-inbox',\n permissions: [\"manage_aws\"]\n },\n {\n text: 'Sent',\n route: '?page=system/email/sent',\n icon: 'bi-send-check',\n permissions: [\"manage_aws\"]\n },\n {\n text: 'Templates',\n route: '?page=system/email/templates',\n icon: 'bi-file-text',\n permissions: [\"manage_aws\"]\n }\n ]\n },\n {\n text: 'AWS',\n route: null,\n icon: 'bi-cloud',\n permissions: [\"view_admin\"],\n children: [\n {\n text: 'CloudWatch',\n route: '?page=system/cloudwatch',\n icon: 'bi-bar-chart-line',\n permissions: [\"view_admin\"]\n }\n ]\n },\n {\n text: 'Phone Hub',\n route: null,\n icon: 'bi-telephone',\n permissions: [\"manage_users\"],\n children: [\n {\n text: 'Numbers',\n route: '?page=system/phonehub/numbers',\n icon: 'bi-collection',\n permissions: [\"manage_users\"]\n },\n {\n text: 'SMS',\n route: '?page=system/phonehub/sms',\n icon: 'bi-chat-dots',\n permissions: [\"manage_users\"]\n }\n ]\n }\n ];\n\n // Add items to existing admin menu\n adminMenuConfig.items.unshift(...adminMenuItems);\n }\n }\n\n}\n\nexport { registerSystemPages as registerAdminPages };\n"],"names":["AdminHeaderView","View","constructor","options","super","title","headerActions","label","icon","action","buttonClass","className","this","stats","user_activity_day","total_users","group_activity_day","total_groups","api_calls","apiChange","incidents","incidentsChange","prepareStatsForTemplate","getTemplate","onInit","userActivity","MetricsMiniChartWidget","subtitle","background","textColor","granularity","trendRange","trendOffset","slugs","account","chartType","showTooltip","showXAxis","height","chartWidth","color","fill","fillColor","smoothing","showTrending","showSettings","showDateRange","containerId","addChild","groupActivity","apiActivity","endpoint","incidentActivity","onBeforeRender","loadValues","response","getApp","rest","GET","success","data","status","Object","assign","header","render","error","console","loadStats","AdminDashboardPage","Page","pageTitle","pageSubtitle","lastUpdated","Date","toLocaleString","headerView","apiMetricsChart","MetricsChart","yAxis","beginAtZero","tooltip","y","onActionRefreshAll","event","element","button","currentTarget","querySelector","classList","add","disabled","promises","refresh","filter","Boolean","Promise","allSettled","eventBus","events","emit","page","timestamp","alert","innerHTML","setTimeout","remove","onActionExportMetrics","export","charts","onActionViewAlerts","router","navigateTo","onActionViewSystemStatus","refreshDashboard","getCharts","apiMetrics","getStats","onAfterRender","SideNavView","sections","activeSection","navWidth","contentPadding","enableResponsive","minWidth","viewOptions","tagName","sectionConfigs","sectionViews","sectionKeys","currentMode","resizeObserver","lastContainerWidth","config","_addSectionConfig","handleResize","bind","type","permissions","_hasPermission","push","key","view","parent","perm","activeUser","hasPerm","renderTemplate","nav","_buildDropdownNav","_buildSidebarNav","map","escapeHtml","isActive","join","activeConfig","find","c","activeLabel","items","_mountSection","_setupResponsive","onBeforeDestroy","disconnect","window","removeEventListener","values","destroy","showSection","warn","isMounted","contains","previousSection","_unmountSection","_updateNavState","container","unmount","activeKey","querySelectorAll","forEach","link","section","dataset","toggle","selectBtn","textContent","onActionNavigate","el","preventDefault","_updateMode","ResizeObserver","parentElement","observe","addEventListener","containerWidth","_getContainerWidth","Math","abs","offsetWidth","newMode","mode","getActiveSection","getSectionKeys","getSection","addSection","makeActive","removeSection","k","_onModelChange","create","AdminProfileSection","template","hasPhone","model","get","roleLabel","onActionForceVerifyEmail","Dialog","confirm","_saveField","is_email_verified","onActionUnverifyEmail","onActionForceVerifyPhone","is_phone_verified","onActionUnverifyPhone","onActionChangeEmail","email","prompt","defaultValue","trim","onActionChangePhone","phone","phone_number","onActionSetPhone","placeholder","onActionRemovePhone","resp","save","set","toast","message","onActionEditUsername","username","fields","successMsg","AdminPersonalSection","hasDob","dobFormatted","dob","year","month","day","split","timezoneDisplay","timezone","hasAddress","meta","street","city","state","zip","country","addressSummary","onActionForceVerifyDob","is_dob_verified","onActionUnverifyDob","onActionEditDisplayName","name","display_name","onActionEditFirstName","first_name","onActionEditLastName","last_name","onActionEditDob","showForm","size","cols","onActionEditTimezone","value","text","metadata","onActionEditAddress","updatedMeta","toLowerCase","AdminSecuritySection","onActionSendPasswordReset","app","POST","onActionSendEmailVerification","onActionSendMagicLink","onActionSetPassword","required","help","password","onActionToggleMfa","currentMfa","requires_mfa","onActionDisableTotp","DELETE","id","onActionManagePasskeys","collection","PasskeyList","params","user","fetch","e","models","passkeys","p","toJSON","onActionEditPasskey","async","passkey","String","showModelForm","PasskeyForms","edit","onActionDeletePasskey","showDialog","body","buttons","class","dismiss","onActionViewRecoveryCodes","dataOnly","remaining","codes","onActionRevokeAllSessions","PROVIDER_ICONS","google","github","microsoft","apple","facebook","twitter","linkedin","AdminConnectedSection","connections","results","provider","onActionUnlink","connection","CHANNEL_LABELS","in_app","CHANNELS","AdminNotificationsSection","preferences","channels","ch","hasPreferences","keys","length","preferenceRows","sort","kind","kindLabel","replace","toUpperCase","toggles","channel","checked","onActionTogglePref","AdminApiKeysSection","apiKeys","_loadKeys","Array","isArray","_renderKeysList","rows","created","toLocaleDateString","expires","lastUsed","last_used","ips","allowed_ips","is_active","token_prefix","onActionRevokeKey","DeviceRow","TableRow","deviceIcon","dev","device","os","some","m","family","includes","deviceName","brand","deviceModel","browserName","ua","user_agent","major","osName","deviceMeta","LocationRow","device_info","locationText","geo","region","country_name","countryName","threatFlags","flags","is_vpn","is_tor","is_proxy","hasThreatFlags","UserView","User","sideNavView","setModel","profileView","personalView","securityView","connectedView","notificationsView","apiKeysView","permsView","FormView","PERMISSION_FIELDS","autosaveModelField","membersCollection","MemberList","groupsView","TableView","hideActivePillNames","columns","formatter","sortable","eventsCollection","IncidentEventList","model_name","model_id","eventsView","width","devicesView","UserDeviceList","clickAction","itemClass","locationsView","UserDeviceLocationList","pushDevices","PushDeviceList","pushDevicesView","logsCollection","LogList","logsView","startName","endName","fieldName","format","displayFormat","separator","activityCollection","uid","activityView","userMenu","ContextMenu","context","onActionEditUser","modelName","formConfig","UserForms","onActionClearAvatar","avatar","onActionToggleActive","onActionDeactivateUser","onActionActivateUser","sectionName","showTab","tabName","getActiveTab","VIEW_CLASS","UserTablePage","TablePage","pageName","Collection","UserList","viewDialogOptions","defaultQuery","visibility","filters","selectable","searchable","filterable","paginated","showRefresh","showAdd","showExport","emptyMessage","contextMenu","tableOptions","striped","bordered","hover","responsive","onActionEditPermissions","item","_","onActionChangePassword","attributes","autocomplete","readonly","tabindex","style","passwordUsage","showToggle","new_password","MOJOUtils","checkPasswordStrength","score","onPasswordChange","onActionSendInvite","send_invite","MemberView","Member","detailsView","permissionsView","memberMenu","danger","onActionEditMembership","Modal","modelForm","MemberForms","onActionViewUser","userId","import","then","n","q","showModelById","onActionViewGroup","groupId","Group","onActionDeactivateMember","onActionActivateMember","onActionRemoveMember","MemberTablePage","formEdit","itemViewClass","batchBarLocation","batchActions","GroupView","membersView","group","addButtonLabel","onAdd","onInviteClick","childrenView","GroupList","onActionAddChildGroup","groupMenu","onActionEditGroup","GroupForms","detailed","onActionInviteMember","Event","stopPropagation","f","newGroup","onActionDeactivateGroup","onActionActivateGroup","onActionViewParent","parentId","GroupTablePage","formCreate","GroupKindOptions","onActionMakeActive","setActiveGroup","DeviceLocationRow","ispName","isp","asn_org","DeviceView","UserDevice","deviceInfo","_getIcon","browserFull","_getBrowser","osFull","_getOS","deviceFull","_getDevice","isMobile","_isMobile","parts","ver","minor","browser","user_device","deviceMenu","onActionDeleteDevice","show","duid","getByDuid","UserDeviceTablePage","UserDeviceLocationTablePage","GeoIPView","GeoLocatedIP","hasCoordinates","DataView","showEmptyValues","emptyValueText","networkView","riskView","metadataView","source_ip","ip","tabs","Location","Network","Events","Logs","Metadata","lat","lng","locationStr","mapView","MapView","markers","popup","tileLayer","zoom","tabView","TabView","activeTab","menuItems","geoIPMenu","bootstrap","Tooltip","existing","getInstance","dispose","onActionEditLocation","EDIT_LOCATION_FORM","onActionEditSecurity","EDIT_SECURITY_FORM","onActionEditNetwork","EDIT_NETWORK_FORM","onActionRefreshGeoip","info","onActionThreatAnalysis","threat_analysis","onActionViewOnMap","url","open","onActionDeleteGeoip","confirmClass","confirmText","lookup","dialog","document","GeoLocatedIPTablePage","GeoLocatedIPList","itemView","tableViewOptions","evt","onLookup","tableView","_onRowView","ApiKey","Model","ApiKeyList","ModelClass","ApiKeyForms","ApiKeyView","apiKeyMenu","onActionEditKey","onActionDeactivateKey","confirmLabel","showLoading","hideLoading","onActionActivateKey","onActionDeleteKey","delete","ADD_FORM","EDIT_FORM","ApiKeyTablePage","onActionAdd","result","showError","token","showAlert","CloudWatchChart","resourceType","category","slug","defaultDateRange","stat","buildApiParams","setStat","fetchData","DASHBOARD_CHARTS","unit","yAxisForUnit","max","CloudWatchDashboardPage","i","def","chart","showGranularity","IncidentDashboardHeader","IncidentStats","IncidentDashboardPage","eventsWidget","valueFormat","settingsKey","incidentsWidget","eventsByCountryChart","showMetricsFilter","maxDatasets","colors","incidentsByCountryChart","eventsCountryMap","MetricsCountryMapView","maxCountries","metricLabel","mapStyle","myTicketsCollection","TicketList","myTicketsTable","newIncidentsCollection","IncidentList","highPriorityIncidentsTable","refreshTasks","StackTraceView","stackTrace","formattedStackTrace","formatStackTrace","lines","JSON","stringify","html","line","index","match","funcName","filePath","lineNum","colNum","startsWith","updateStackTrace","newStackTrace","IncidentHistoryAdapter","incidentId","IncidentHistoryList","incident","transform","author","avatarUrl","content","attachments","addNote","history","IncidentHistory","note","IncidentView","Incident","incidentIcon","getIconForIncident","s","overviewView","actions","adapter","historyView","ChatView","Overview","stack_trace","stackTraceView","incidentMenu","onActionEditIncident","IncidentForms","onActionResolveIncident","onActionDeleteIncident","IncidentTablePage","onActionBatchResolve","selected","getSelectedItems","all","onActionBatchOpen","onActionBatchPause","onActionBatchIgnore","onActionBatchMerge","_event","_element","parentModel","merge","mergeIds","EventView","IncidentEvent","eventIcon","getIconForEvent","level","eventMenu","onActionViewIncident","onActionViewModel","onActionDeleteEvent","EventTablePage","IncidentEventForms","category__not","TicketNoteAdapter","ticketId","TicketNoteList","TicketNote","media","files","TicketView","Ticket","chatView","theme","currentUserId","getCurrentUserId","inputPlaceholder","inputButtonText","ticketMenu","currentUser","onActionEditTicket","TicketForms","onActionChangeStatus","currentStatus","onActionSetPriority","currentPriority","priority","onActionAssignUser","onActionCloseTicket","TicketTablePage","editable","editableOptions","placeHolder","TicketCategories","RuleSetView","RuleSet","matchByValue","matchByOption","MatchByOptions","opt","matchByLabel","bundleByValue","bundleByOption","BundleByOptions","bundleByLabel","configView","rulesCollection","RuleList","rulesView","divider","addFormDefaults","Configuration","Rules","onActionEditRuleset","onActionDisableRuleset","newStatus","showToast","onActionDeleteRuleset","closest","bsModal","hide","RuleSetTablePage","RuleSetList","v","pageSizes","defaultPageSize","emptyIcon","EmailDomainTablePage","EmailDomainList","EmailDomainForms","onActionEditAwsCreds","credentials","onActionOnboard","EmailDomain","formData","onboard","Error","err","onActionAudit","audit","onActionReconcile","reconcile","EmailMailboxTablePage","MailboxList","MailboxForms","onActionSendEmail","from_email","Mailbox","sendEmail","msg","details","onActionSendTemplateEmail","EmailHtmlPreviewView","renderHtmlInIframe","iframe","htmlContent","iframeDoc","contentDocument","contentWindow","write","close","onActionRefreshPreview","EmailTemplateView","EmailTemplate","hasHtml","hasText","hasMetadata","EmailTemplateTablePage","EmailTemplateList","EmailTemplateForms","scrollable","formDialogConfig","EmailView","SentMessage","hasContext","SentMessageTablePage","SentMessageList","PhoneNumber","normalize","phoneNumber","countryCode","payload","country_code","PhoneNumberList","SMS","send","SMSList","PhoneNumberView","carrierView","addressView","Carrier","Address","ctxMenu","onActionRefreshLookup","number","force_refresh","warning","onActionDeletePhone","MODEL_CLASS","PhoneNumberTablePage","addButtonIcon","SMSView","messageView","deliveryView","Message","Delivery","onActionRefreshSms","onActionDeleteSms","SMSTablePage","PushDashboardPage","deliveriesChart","statusChart","PieChart","recentDeliveries","PushDeliveryList","_sort","_limit","failedDeliveries","PushConfigTablePage","PushConfigList","PushConfigForms","PushTemplateTablePage","PushTemplateList","PushTemplateForms","PushDeliveryView","PushDeliveryTablePage","PushDeviceView","dataView","PushDeviceTablePage","JobStatsView","pending","running","completed","failed","scheduled","totals","JobHealthView","health","runners","active","total","loadHealth","overall_status","runners_active","scheduler","healthStatusClass","getHealthStatusClass","setupChannelDisplay","setupSchedulerDisplay","channelsArray","channelStatus","totalJobs","queued_count","inflight_count","statusBadgeClass","getChannelBadgeClass","statusIcon","getChannelIcon","queued","inflight","schedulerStatusText","schedulerStatusClass","schedulerIcon","healthy","critical","unknown","onActionRefreshHealth","onActionSystemSettings","JobDetailsView","Job","payloadView","autoRefreshInterval","JobEventList","job","JobLogList","Payload","contextMenuItems","canCancel","canRetry","jobMenu","graph","prepareJobData","getStatusBadgeClass","getStatusIcon","formattedDuration","getFormattedDuration","loadJobDetails","getDetailedStatus","onActionRefreshJob","onActionCancelJob","cancel","onActionRetryJob","retryData","JobForms","retry","delay","newJobId","startAutoRefresh","clearInterval","setInterval","stopAutoRefresh","onDestroy","onHide","JobMetricsModalView","JobsAdminPage","refreshRate","jobStats","JobsEngineStats","jobStatsView","jobHealthView","jobsPublishedChart","jobsFailedChart","runningJobsTable","JobList","collectionParams","hideActivePills","row","pendingJobsTable","run_at__isnull","on","failedJobsTable","scheduledJobsTable","runnersTable","JobRunnerList","isAlive","heartbeatTime","diffMs","diffSeconds","floor","refreshData","tasks","updateHeaderTimestamp","timestampElement","rateLabel","refreshRateSeconds","refreshRateLabel","onActionSetRefreshRate","rate","parseInt","getAttribute","rateText","onActionExportData","onActionRunSimpleJob","showConfirm","executeJobAction","test","onActionRunTestJobs","tests","onActionClearStuck","channelOptions","clearStuck","count","onActionClearChannel","clearChannel","onActionPurgeJobs","purgeJobs","days_old","onActionCleanupConsumers","cleanConsumers","onActionRunnerBroadcast","JobRunnerForms","broadcast","broadcastResult","JobRunner","command","timeout","actionFn","successMessage","onEnter","onExit","getHealth","onActionOpenJobMetricsModal","modalView","onActionViewAllJobs","jobList","fetchOnMount","TaskDetailsView","task","logs","metrics","prepareTaskData","loadTaskLogs","loadTaskMetrics","dataFormatted","expiresClass","now","cancelled","expired","getLogLevelClass","debug","log","levelClass","setTask","onActionRefreshLogs","showSuccess","originalTask","newTask","exportData","exported_at","toISOString","exported_by","blob","Blob","URL","createObjectURL","a","createElement","href","download","appendChild","click","removeChild","revokeObjectURL","formatBytes","bytes","toFixed","resourceBadgeClass","pct","progressBarClass","RunnerOverviewTab","r","aliveBadgeClass","aliveIcon","aliveText","started","startedText","sec","getTime","uptimeText","seconds","d","h","formatUptime","ageSec","isoString","heartbeatText","heartbeatAgeText","round","heartbeatClass","jobsProcessed","jobsFailed","errorRate","RunnerSysinfoTab","sysinfo","sysinfoError","loading","loaded","onTabActivated","loadSysinfo","enrichSysinfo","memory","totalFmt","usedFmt","used","availableFmt","available","barClass","percent","badgeClass","disk","freeFmt","free","network","bytesRecvFmt","bytes_recv","bytesSentFmt","bytes_sent","errClass","errin","errout","dropClass","dropin","dropout","cpuPct","cpu_load","cpuLoadBarClass","cpuLoadBadgeClass","cpu","freq","freqText","current","cpus_load","cpuCores","bootDatetime","boot_time","collectedText","datetime","onActionRefreshSysinfo","RunnerJobsTab","jobs","loadJobs","durationText","started_at","toLocaleTimeString","attemptBadgeClass","attempt","onActionRefreshJobs","onActionViewJob","jobId","runner","cancel_request","RunnerLogsTab","filteredLogs","logFilter","filterAllClass","filterDebugClass","filterInfoClass","filterWarnClass","filterErrorClass","loadLogs","jobsResp","jobIds","j","slice","catch","concat","b","levelBadgeClass","kindDisplay","createdText","l","onActionFilterLogs","RunnerActionsTab","pingResult","onActionPing","runner_id","onActionShutdown","graceful","onActionBroadcast","commandEl","timeoutEl","parseFloat","showBusy","hideBusy","showCode","onActionExport","RunnerDetailsView","Actions","TaskStatsView","errors","TaskRunnersView","loadRunners","pingAge","ping_age","statusBadge","pingAgeText","formatPingAge","onActionRefreshRunners","onActionViewRunnerDetails","runnerId","onActionPauseRunner","hostname","onActionRestartRunner","onActionRemoveRunner","confirmMessage","TaskChartsView","taskFlowChart","taskErrorsChart","PendingTasksTable","RunningTasksTable","CompletedTasksTable","ErrorTasksTable","TaskManagementPage","taskStatsView","taskRunnersView","taskChartsView","taskTablesView","Pending","Running","Completed","Errors","activeTable","getTab","onActionExportTasks","exportToCSV","onActionManageChannels","channelList","onActionViewSystemLogs","logPreview","getRunners","taskFlow","taskErrors","LogView","Log","logIcon","getIconForLog","lvl","logContent","formattedLog","parsed","parse","logContentView","onActionCopyLog","navigator","clipboard","writeText","Details","logMenu","onActionViewIp","onActionViewDevice","onActionDeleteLog","LogTablePage","MetricsPermissionsView","MetricsPermission","onActionEdit","MetricsForms","onActionDelete","MetricsPermissionsTablePage","MetricsPermissionList","Setting","SettingList","SettingForms","SettingView","settingMenu","onActionEditSetting","onActionDeleteSetting","SettingTablePage","FileManagerTablePage","FileManagerList","FileManagerForms","onActionEditOwners","owners","onActionCheckCors","check_cors","showData","onActionTestConnection","test_connection","onActionEditCredentials","onActionClone","clone","FileView","File","isImage","renditionsData","renditionsCollection","infoView","renditionsView","Info","fileMenu","onActionViewFile","contentType","fileUrl","renditions","images","src","alt","role","LightboxGallery","fitToScreen","PDFViewer","onActionDownloadFile","onActionEditFile","FileForms","onActionMakePublic","is_public","onActionMakePrivate","onActionDeleteFile","FileTablePage","FileList","handleFileUpload","enableFileDrop","acceptedTypes","maxFileSize","multiple","validateOnDrop","fileInput","accept","display","file","target","maxSize","_formatFileSize","fileModel","extra","requiresGroup","activeGroup","upload","description","onProgress","progressInfo","percentage","onComplete","onError","pow","onFileDrop","validation","applyFileDropMixin","S3BucketTablePage","S3BucketList","S3BucketForms","UserDeviceLocationView","UserDeviceLocation","_ud","_di","_geo","_getDeviceIcon","locationSummary","_getLocationSummary","countryFlag","threatLevel","threat_level","threatColor","_getThreatColor","latitude","longitude","is_threat","is_suspicious","di","postal_code","asn","patch","string","risk_score","_riskRow","is_cloud","is_datacenter","is_mobile","is_known_attacker","is_known_abuser","default","menu","deviceId","onActionOpenInMaps","onActionDeleteRecord","METRIC_CATEGORIES","ec2","rds","redis","TYPE_ICONS","TYPE_LABELS","CloudWatchResourceView","resource","categories","typeLabel","metaHtml","_buildMetaItems","cat","instance_type","private_ip","public_ip","engine","instance_class","node_type","num_nodes","registerSystemPages","addToMenu","registerPage","AdminDashboardPageClass","JobsAdminPageClass","UserTablePageClass","GroupTablePageClass","MemberTablePageClass","S3BucketTablePageClass","FileManagerTablePageClass","FileTablePageClass","IncidentTablePageClass","EventTablePageClass","LogTablePageClass","UserDeviceTablePageClass","UserDeviceLocationTablePageClass","GeoLocatedIPTablePageClass","EmailMailboxTablePageClass","EmailDomainTablePageClass","SentMessageTablePageClass","EmailTemplateTablePageClass","IncidentDashboardPageClass","RuleSetTablePageClass","TicketTablePageClass","MetricsPermissionsTablePageClass","PushDashboardPageClass","PushConfigTablePageClass","PushTemplateTablePageClass","PushDeliveryTablePageClass","PushDeviceTablePageClass","PhoneNumberTablePageClass","SMSTablePageClass","ApiKeyTablePageClass","SettingTablePageClass","CloudWatchDashboardPageClass","sidebar","getMenuConfig","adminMenuConfig","adminMenuItems","route","children","unshift"],"mappings":"s5CASA,MAAMA,wBAAwBC,EAC5B,WAAAC,CAAYC,EAAU,IACpBC,MAAM,CACJC,MAAO,eACJF,EACHG,cAAe,CACX,CACIC,MAAO,SACPC,KAAM,cACNC,OAAQ,SACRC,YAAa,gBAGrBC,UAAW,yBAIbC,KAAKC,MAAQ,CACXC,kBAAmB,EACnBC,YAAa,EACbC,mBAAoB,EACpBC,aAAc,EACdC,UAAW,EACXC,UAAW,GACXC,UAAW,EACXC,gBAAiB,IAInBT,KAAKU,yBACP,CAEA,iBAAMC,GACJ,MAAO,4oBAqBT,CAEA,YAAMC,GAIJZ,KAAKa,aAAe,IAAIC,EAAuB,CAC7ClB,KAAM,oBACNH,MAAO,gBACPsB,SAAU,4HACVC,WAAY,UACZC,UAAW,UACXC,YAAa,OACbC,WAAY,EACZC,YAAa,EACbC,MAAO,CAAC,qBACRC,QAAS,SACTC,UAAW,MACXC,aAAa,EACbC,WAAW,EACXC,OAAQ,GACRC,WAAY,OACZC,MAAO,2BACPC,MAAM,EACNC,UAAW,2BACXC,UAAW,GACXC,cAAc,EACdC,cAAc,EACdC,eAAe,EACfC,YAAa,sBAEfnC,KAAKoC,SAASpC,KAAKa,cAEnBb,KAAKqC,cAAgB,IAAIvB,EAAuB,CAC5ClB,KAAM,wBACNH,MAAO,iBACPsB,SAAU,6HACVC,WAAY,UACZC,UAAW,UACbC,YAAa,OACbC,WAAY,EACZC,YAAa,EACbC,MAAO,CAAC,sBACRC,QAAS,SACTC,UAAW,MACXC,aAAa,EACbC,WAAW,EACXC,OAAQ,GACRC,WAAY,OACZC,MAAO,2BACPC,MAAM,EACNC,UAAW,2BACXC,UAAW,GACXC,cAAc,EACdG,YAAa,uBAEfnC,KAAKoC,SAASpC,KAAKqC,eAEnBrC,KAAKsC,YAAc,IAAIxB,EAAuB,CAC1ClB,KAAM,sBACNH,MAAO,eACPsB,SAAU,sHACVC,WAAY,UACZC,UAAW,UACbsB,SAAU,qBACVpB,WAAY,EACZC,YAAa,EACbF,YAAa,OACbG,MAAO,CAAC,aACRC,QAAS,SACTC,UAAW,OACXC,aAAa,EACbC,WAAW,EACXC,OAAQ,GACRC,WAAY,OACZC,MAAO,2BACPC,MAAM,EACNC,UAAW,2BACXC,UAAW,GACXC,cAAc,EACdG,YAAa,qBAEfnC,KAAKoC,SAASpC,KAAKsC,aAEnBtC,KAAKwC,iBAAmB,IAAI1B,EAAuB,CAC/ClB,KAAM,kCACNH,MAAO,YACPsB,SAAU,sHACVC,WAAY,UACZC,UAAW,UACbsB,SAAU,qBACVpB,WAAY,EACZC,YAAa,EACbF,YAAa,OACbG,MAAO,CAAC,aACRC,QAAS,WACTC,UAAW,OACXC,aAAa,EACbC,WAAW,EACXC,OAAQ,GACRC,WAAY,OACZC,MAAO,2BACPC,MAAM,EACNC,UAAW,2BACXC,UAAW,GACXC,cAAc,EACdG,YAAa,0BAEfnC,KAAKoC,SAASpC,KAAKwC,iBACrB,CAEA,oBAAMC,GAEN,CAEA,uBAAA/B,GAMA,CAEA,gBAAMgC,GACF,IACE,MAAMC,QAAiB3C,KAAK4C,SAASC,KAAKC,IAAI,yBAA0B,CACtEzB,MAAO,CAAC,cAAe,gBACvBC,QAAS,WAEPqB,EAASI,SAAWJ,EAASK,KAAKC,QACpCC,OAAOC,OAAOnD,KAAKC,MAAO0C,EAASK,KAAKA,MAEtChD,KAAKqC,gBACLrC,KAAKqC,cAAce,OAAO/C,aAAeL,KAAKC,MAAMI,cAAgB,EACpEL,KAAKqC,cAAce,OAAOC,UAE1BrD,KAAKa,eACLb,KAAKa,aAAauC,OAAOjD,YAAcH,KAAKC,MAAME,aAAe,EACjEH,KAAKa,aAAauC,OAAOC,SAE/B,OAASC,GACPC,QAAQD,MAAM,8BAA+BA,EAC/C,CACJ,CAEA,eAAME,GAEJ,IACE,MAAMb,QAAiB3C,KAAK4C,SAASC,KAAKC,IAAI,sBAAuB,CACnEzB,MAAO,CAAC,eAAgB,oBAAqB,YACzC,YAAa,aAAc,sBAC/BC,QAAS,SACTJ,YAAa,SAEXyB,EAASI,SAAWJ,EAASK,KAAKC,SACpCC,OAAOC,OAAOnD,KAAKC,MAAO0C,EAASK,KAAKA,MACxChD,KAAKU,0BAET,OAAS4C,GACPC,QAAQD,MAAM,8BAA+BA,EAC/C,CACF,EAGa,MAAMG,2BAA2BC,EAC9C,WAAApE,CAAYC,EAAU,IACpBC,MAAM,IACDD,EACHE,MAAO,kBACPM,UAAW,yBAIbC,KAAK2D,UAAY,kBACjB3D,KAAK4D,aAAe,wCACtB,CAEA,iBAAMjD,GACJ,MAAO,+5EA2DT,CAEA,YAAMC,GAEJZ,KAAK6D,4BAAA,IAAkBC,MAAOC,iBAG9B/D,KAAKgE,WAAa,IAAI5E,gBAAgB,CACpC+C,YAAa,iBAEfnC,KAAKoC,SAASpC,KAAKgE,YAGnBhE,KAAKiE,gBAAkB,IAAIC,EAAa,CACtCzE,MAAO,kDACP8C,SAAU,qBACVb,OAAQ,IACRR,YAAa,QACbG,MAAO,CAAC,YAAa,cACrBC,QAAS,SACTC,UAAW,OACXW,eAAe,EACfiC,MAAO,CACLxE,MAAO,QACPyE,aAAa,GAEfC,QAAS,CACPC,EAAG,UAELnC,YAAa,sBAEfnC,KAAKoC,SAASpC,KAAKiE,gBAErB,CAGA,wBAAMM,CAAmBC,EAAOC,GAC9B,IAEE,MAAMC,EAASD,GAAWD,GAAOG,eAAiB,KAC5C/E,EAAO8E,GAAQE,gBAAgB,KACrChF,GAAMiF,UAAUC,IAAI,WAChBJ,IAAQA,EAAOK,UAAW,GAG9B,MAAMC,EAAW,CACfhF,KAAKgE,YAAYtB,aACjB1C,KAAKiE,iBAAiBgB,WACtBC,OAAOC,eAEHC,QAAQC,WAAWL,GAIzBhF,KAAK6D,4BAAA,IAAkBC,MAAOC,iBAG9B,MAAMuB,EAAWtF,KAAK4C,UAAU2C,OAC5BD,GACFA,EAASE,KAAK,4BAA6B,CACzCC,KAAMzF,KACN0F,UAAW1F,KAAK6D,aAItB,OAASP,GACPC,QAAQD,MAAM,+BAAgCA,GAE9C,MAAMqC,EAAQ3F,KAAKyE,QAAQG,cAAc,kBACrCe,IACFA,EAAM5F,UAAY,8BAClB4F,EAAMC,UAAY,oQAUlBC,WAAW,KACTF,EAAM5F,UAAY,+BAClB4F,EAAMC,UAAY,4QAK6B5F,KAAK6D,4EAInD,KAEP,CAAA,QAEE,MAAMjE,EAAO6E,EAAQG,cAAc,KACnChF,GAAMiF,UAAUiB,OAAO,WACnBpB,gBAAeK,UAAW,EAChC,CACF,CAEA,2BAAMgB,CAAsBvB,EAAOC,GACjC,UAEQzE,KAAKiE,iBAAiB+B,OAAO,QAGnC,MAAMV,EAAWtF,KAAK4C,UAAU2C,OAC5BD,GACFA,EAASE,KAAK,yBAA0B,CACtCC,KAAMzF,KACNiG,OAAQ,CAAC,gBAIf,OAAS3C,GACPC,QAAQD,MAAM,4BAA6BA,EAC7C,CACF,CAEA,wBAAM4C,CAAmB1B,EAAOC,GAE9B,MAAM0B,EAASnG,KAAK4C,UAAUuD,OAC1BA,GACFA,EAAOC,WAAW,gBAEtB,CAEA,8BAAMC,CAAyB7B,EAAOC,GAEpC,MAAM0B,EAASnG,KAAK4C,UAAUuD,OAC1BA,GACFA,EAAOC,WAAW,uBAEtB,CAGA,sBAAME,GACJ,OAAOtG,KAAKuE,mBAAmB,KAAM,KAAM,CAAEQ,UAAU,EAAOH,cAAe,IAAM,MACrF,CAEA,SAAA2B,GACE,MAAO,CACLC,WAAYxG,KAAKiE,gBAErB,CAEA,QAAAwC,GACE,OAAOzG,KAAKgE,YAAY/D,OAAS,CAAA,CACnC,CAEA,mBAAMyG,GACJ1G,KAAKgE,YAAYtB,YACnB,ECnaF,MAAMiE,oBAAoBtH,EACtB,WAAAC,CAAYC,EAAU,IAClB,MAAMqH,SACFA,EAAW,GAAAC,cACXA,EAAAC,SACAA,EAAAC,eACAA,EAAAC,iBACAA,EAAAC,SACAA,KACGC,GACH3H,EAEJC,MAAM,CACF2H,QAAS,MACTpH,UAAW,mBACRmH,IAIPlH,KAAK8G,SAAWA,GAAY,IAC5B9G,KAAK+G,eAAiBA,GAAkB,gBACxC/G,KAAKgH,kBAAwC,IAArBA,EACxBhH,KAAKiH,SAAWA,GAAY,IAG5BjH,KAAKoH,eAAiB,GACtBpH,KAAKqH,aAAe,GACpBrH,KAAKsH,YAAc,GACnBtH,KAAK6G,cAAgB,KACrB7G,KAAKuH,YAAc,UACnBvH,KAAKwH,eAAiB,KACtBxH,KAAKyH,mBAAqB,EAG1B,IAAA,MAAWC,KAAUd,EACjB5G,KAAK2H,kBAAkBD,GAI3B1H,KAAK6G,cAAgBA,GAAiB7G,KAAKsH,YAAY,IAAM,KAG7DtH,KAAK4H,aAAe5H,KAAK4H,aAAaC,KAAK7H,KAC/C,CAOA,iBAAA2H,CAAkBD,GACM,YAAhBA,EAAOI,KAMPJ,EAAOK,cAAgB/H,KAAKgI,eAAeN,EAAOK,eAItD/H,KAAKoH,eAAea,KAAKP,GACzB1H,KAAKsH,YAAYW,KAAKP,EAAOQ,KAEzBR,EAAOS,OACPnI,KAAKqH,aAAaK,EAAOQ,KAAOR,EAAOS,KACvCT,EAAOS,KAAKC,OAASpI,OAdrBA,KAAKoH,eAAea,KAAK,CAAEH,KAAM,UAAWnI,MAAO+H,EAAO/H,OAgBlE,CAQA,cAAAqI,CAAeK,GACX,IACI,OAAOrI,KAAK4C,SAAS0F,WAAWC,QAAQF,EAC5C,CAAA,MACI,OAAO,CACX,CACJ,CAMA,oBAAMG,GACF,MAAMC,EAA2B,aAArBzI,KAAKuH,YACXvH,KAAK0I,oBACL1I,KAAK2I,mBAEX,MAAO,8JAIc3I,KAAK8G,65CAoCH9G,KAAK+G,m2CAiCD,aAArB/G,KAAKuH,YAA6B,+CACJkB,sGAE5B,wFAE2BA,6IAKvC,CAOA,gBAAAE,GACI,OAAO3I,KAAKoH,eAAewB,IAAIlB,IAC3B,GAAoB,YAAhBA,EAAOI,KACP,MAAO,8BAA8B9H,KAAK6I,WAAWnB,EAAO/H,eAEhE,MAAMmJ,EAAWpB,EAAOQ,MAAQlI,KAAK6G,cAC/BjH,EAAO8H,EAAO9H,KAAO,gBAAgB8H,EAAO9H,aAAe,GACjE,MAAO,sBAAsBkJ,EAAW,SAAW,4CAA4CpB,EAAOQ,QAAQtI,KAAQI,KAAK6I,WAAWnB,EAAO/H,eAC9IoJ,KAAK,GACZ,CAOA,iBAAAL,GACI,MAAMM,EAAehJ,KAAKoH,eAAe6B,KAAKC,GAAKA,EAAEhB,MAAQlI,KAAK6G,eAC5DsC,EAAcH,EAAeA,EAAarJ,MAAQK,KAAKsH,YAAY,GAEnE8B,EAAQpJ,KAAKoH,eACdlC,OAAOgE,GAAgB,YAAXA,EAAEpB,MACdc,IAAIlB,IACD,MAAMoB,EAAWpB,EAAOQ,MAAQlI,KAAK6G,cACrC,MAAO,oFAEgCiC,EAAW,SAAW,8GAE7BpB,EAAOQ,qFAEzBR,EAAO9H,KAAO,gBAAgB8H,EAAO9H,kBAAoB,mCACzDI,KAAK6I,WAAWnB,EAAO/H,uCACvBmJ,EAAW,sCAAwC,uFAIlEC,KAAK,IAEZ,MAAO,qMAIOC,GAAcpJ,KAAO,gBAAgBoJ,EAAapJ,aAAe,iCAC3DI,KAAK6I,WAAWM,yFAEMC,sCAG9C,CAMA,mBAAM1C,SACIlH,MAAMkH,gBAGR1G,KAAK6G,qBACC7G,KAAKqJ,cAAcrJ,KAAK6G,eAI9B7G,KAAKgH,kBACLhH,KAAKsJ,kBAEb,CAEA,qBAAMC,SACI/J,MAAM+J,kBAGRvJ,KAAKwH,iBACLxH,KAAKwH,eAAegC,aACpBxJ,KAAKwH,eAAiB,MAGJ,oBAAXiC,QACPA,OAAOC,oBAAoB,SAAU1J,KAAK4H,cAI9C,IAAA,MAAWO,KAAQjF,OAAOyG,OAAO3J,KAAKqH,cAC9Bc,GAAgC,mBAAjBA,EAAKyB,eACdzB,EAAKyB,SAGvB,CAWA,iBAAMC,CAAY3B,GACd,IAAKlI,KAAKqH,aAAaa,GAEnB,OADA3E,QAAQuG,KAAK,yBAAyB5B,sBAC/B,EAGX,GAAIA,IAAQlI,KAAK6G,cAAe,CAE5B,MAAMsB,EAAOnI,KAAKqH,aAAaa,GAC/B,GAAIC,GAAQA,EAAK4B,aAAe/J,KAAKyE,SAASuF,SAAS7B,EAAK1D,SACxD,OAAO,CAEf,CAEA,MAAMwF,EAAkBjK,KAAK6G,cAmB7B,OAlBA7G,KAAK6G,cAAgBqB,EAGjB+B,GAAmBA,IAAoB/B,SACjClI,KAAKkK,gBAAgBD,SAIzBjK,KAAKqJ,cAAcnB,GAGzBlI,KAAKmK,gBAAgBjC,GAErBlI,KAAKwF,KAAK,kBAAmB,CACzBqB,cAAeqB,EACf+B,qBAGG,CACX,CAOA,mBAAMZ,CAAcnB,GAChB,MAAMC,EAAOnI,KAAKqH,aAAaa,GAC/B,IAAKC,EAAM,OAEX,MAAMiC,EAAYpK,KAAKyE,SAASG,cAAc,kCACzCwF,IAEAjC,EAAK4B,mBACA5B,EAAK9E,QAAO,EAAM+G,GAEhC,CAOA,qBAAMF,CAAgBhC,GAClB,MAAMC,EAAOnI,KAAKqH,aAAaa,GAC1BC,GAASA,EAAK4B,mBAEb5B,EAAKkC,SACf,CAOA,eAAAF,CAAgBG,GACZ,IAAKtK,KAAKyE,QAAS,OAGnBzE,KAAKyE,QAAQ8F,iBAAiB,8BAA8BC,QAAQC,IAChE,MAAMC,EAAUD,EAAKE,QAAQD,QACzBA,GACAD,EAAK5F,UAAU+F,OAAO,SAAUF,IAAYJ,KAKpD,MAAMO,EAAY7K,KAAKyE,QAAQG,cAAc,wBAC7C,GAAIiG,EAAW,CACX,MAAMnD,EAAS1H,KAAKoH,eAAe6B,KAAKC,GAAKA,EAAEhB,MAAQoC,GACnD5C,IACAmD,EAAUC,YAAcpD,EAAO/H,MAEvC,CACJ,CAMA,sBAAMoL,CAAiBvG,EAAOwG,GAC1BxG,EAAMyG,iBACN,MAAMP,EAAUM,EAAGL,QAAQD,QAI3B,OAHIA,SACM1K,KAAK6J,YAAYa,IAEpB,CACX,CAUA,gBAAApB,GACI,GAAKtJ,KAAKyE,SAAYzE,KAAKgH,iBAI3B,GAFAhH,KAAKkL,cAEyB,oBAAnBC,eAAgC,CACvCnL,KAAKwH,eAAiB,IAAI2D,eAAe,KACrCnL,KAAK4H,iBAET,MAAMwC,EAAYpK,KAAKyE,QAAQ2G,eAAiBpL,KAAKyE,QACrDzE,KAAKwH,eAAe6D,QAAQjB,EAChC,MACIX,OAAO6B,iBAAiB,SAAUtL,KAAK4H,aAE/C,CAKA,kBAAMA,GACF,MAAM2D,EAAiBvL,KAAKwL,qBACxBC,KAAKC,IAAIH,EAAiBvL,KAAKyH,oBAAsB,KACrDzH,KAAKyH,mBAAqB8D,QACpBvL,KAAKkL,cAEnB,CAOA,kBAAAM,GACI,OAAKxL,KAAKyE,UACQzE,KAAKyE,QAAQ2G,eAAiBpL,KAAKyE,SACpCkH,aAFS3L,KAAKiH,QAGnC,CAMA,iBAAMiE,GACF,MAAMK,EAAiBvL,KAAKwL,qBACtBI,EAAUL,EAAiBvL,KAAKiH,SAAW,WAAa,UAE1D2E,IAAY5L,KAAKuH,cACjBvH,KAAKuH,YAAcqE,EACf5L,KAAK+J,mBACC/J,KAAKqD,SAEfrD,KAAKwF,KAAK,yBAA0B,CAChCqG,KAAM7L,KAAKuH,YACXgE,mBAGZ,CAUA,gBAAAO,GACI,OAAO9L,KAAK6G,aAChB,CAMA,cAAAkF,GACI,MAAO,IAAI/L,KAAKsH,YACpB,CAOA,UAAA0E,CAAW9D,GACP,OAAOlI,KAAKqH,aAAaa,IAAQ,IACrC,CAQA,gBAAM+D,CAAWvE,EAAQwE,GAAa,GAClC,OAAIxE,EAAOQ,KAAOlI,KAAKqH,aAAaK,EAAOQ,MACvC3E,QAAQuG,KAAK,yBAAyBpC,EAAOQ,wBACtC,IAGXlI,KAAK2H,kBAAkBD,GAEnB1H,KAAK+J,oBACC/J,KAAKqD,SACP6I,GAAcxE,EAAOQ,WACflI,KAAK6J,YAAYnC,EAAOQ,MAItClI,KAAKwF,KAAK,gBAAiB,CAAEkC,YACtB,EACX,CAOA,mBAAMyE,CAAcjE,GAChB,MAAMC,EAAOnI,KAAKqH,aAAaa,GAC/B,OAAKC,GAMuB,mBAAjBA,EAAKyB,eACNzB,EAAKyB,iBAIR5J,KAAKqH,aAAaa,GACzBlI,KAAKsH,YAActH,KAAKsH,YAAYpC,OAAOkH,GAAKA,IAAMlE,GACtDlI,KAAKoH,eAAiBpH,KAAKoH,eAAelC,OAAOgE,GAAKA,EAAEhB,MAAQA,GAG5DlI,KAAK6G,gBAAkBqB,IACvBlI,KAAK6G,cAAgB7G,KAAKsH,YAAY,IAAM,MAG5CtH,KAAK+J,mBACC/J,KAAKqD,SAGfrD,KAAKwF,KAAK,kBAAmB,CAAE0C,SACxB,IAxBH3E,QAAQuG,KAAK,yBAAyB5B,sBAC/B,EAwBf,CAMA,cAAAmE,GAEA,CAEA,aAAOC,CAAO/M,EAAU,IACpB,OAAO,IAAIoH,YAAYpH,EAC3B,EC/jBW,MAAMgN,4BAA4BlN,EAC7C,WAAAC,CAAYC,EAAU,IAClBC,MAAM,CACFO,UAAW,wBACXyM,SAAU,soOAyGPjN,GAEX,CAIA,YAAIkN,GACA,SAAUzM,KAAK0M,QAAS1M,KAAK0M,MAAMC,IAAI,gBAC3C,CAEA,aAAIC,GACA,OAAK5M,KAAK0M,OACN1M,KAAK0M,MAAMC,IAAI,gBAAwB,YADnB,MAG5B,CAIA,8BAAME,GAKF,cAJwBC,EAAOC,QAC3B,gBAAgB/M,KAAK0M,MAAMC,IAAI,6EAC/B,+BAGE3M,KAAKgN,WAAW,CAAEC,mBAAmB,GAAQ,6BAC5C,EACX,CAEA,2BAAMC,GAKF,cAJwBJ,EAAOC,QAC3B,yEACA,2BAGE/M,KAAKgN,WAAW,CAAEC,mBAAmB,GAAS,+BAC7C,EACX,CAEA,8BAAME,GAKF,cAJwBL,EAAOC,QAC3B,gBAAgB/M,KAAK0M,MAAMC,IAAI,oFAC/B,+BAGE3M,KAAKgN,WAAW,CAAEI,mBAAmB,GAAQ,6BAC5C,EACX,CAEA,2BAAMC,GAKF,cAJwBP,EAAOC,QAC3B,gFACA,2BAGE/M,KAAKgN,WAAW,CAAEI,mBAAmB,GAAS,+BAC7C,EACX,CAIA,yBAAME,GACF,MAAMC,QAAcT,EAAOU,OACvB,6CACA,eACA,CAAEC,aAAczN,KAAK0M,MAAMC,IAAI,UAAY,KAE/C,OAAc,OAAVY,IAAmBA,EAAMG,eACvB1N,KAAKgN,WAAW,CAAEO,MAAOA,EAAMG,QAAU,kBACxC,EACX,CAEA,yBAAMC,GACF,MAAMC,QAAcd,EAAOU,OACvB,4CACA,eACA,CAAEC,aAAczN,KAAK0M,MAAMC,IAAI,iBAAmB,KAEtD,OAAc,OAAViB,IAAmBA,EAAMF,eACvB1N,KAAKgN,WAAW,CAAEa,aAAcD,EAAMF,QAAU,yBAC/C,EACX,CAEA,sBAAMI,GACF,MAAMF,QAAcd,EAAOU,OACvB,sCACA,mBACA,CAAEO,YAAa,mBAEnB,OAAKH,IAAUA,EAAMF,eACf1N,KAAKgN,WAAW,CAAEa,aAAcD,EAAMF,QAAU,uBAC/C,EACX,CAEA,yBAAMM,GAKF,WAJwBlB,EAAOC,QAC3B,mCACA,iBAEY,OAAO,EAEvB,MAAMkB,QAAajO,KAAK0M,MAAMwB,KAAK,CAAEL,aAAc,OAQnD,OAPoB,MAAhBI,EAAKhL,QACLjD,KAAK0M,MAAMyB,IAAI,qBAAqB,GACpCnO,KAAK4C,UAAUwL,OAAOrL,QAAQ,8BACxB/C,KAAKqD,UAEXrD,KAAK4C,UAAUwL,OAAO9K,MAAM2K,EAAKI,SAAW,kCAEzC,CACX,CAEA,0BAAMC,GACF,MAAMC,QAAiBzB,EAAOU,OAC1B,YACA,gBACA,CAAEC,aAAczN,KAAK0M,MAAMC,IAAI,aAAe,KAKlD,OAHiB,OAAb4B,GAAqBA,EAASb,cACxB1N,KAAKgN,WAAW,CAAEuB,SAAUA,EAASb,QAAU,qBAElD,CACX,CAIA,gBAAMV,CAAWwB,EAAQC,GACrB,MAAMR,QAAajO,KAAK0M,MAAMwB,KAAKM,GACf,MAAhBP,EAAKhL,QACLjD,KAAK4C,UAAUwL,OAAOrL,QAAQ0L,SACxBzO,KAAKqD,UAEXrD,KAAK4C,UAAUwL,OAAO9K,MAAM2K,EAAKI,SAAW,iBAEpD,ECnPW,MAAMK,6BAA6BrP,EAC9C,WAAAC,CAAYC,EAAU,IAClBC,MAAM,CACFO,UAAW,yBACXyM,SAAU,0sLAiFPjN,GAEX,CAIA,UAAIoP,GACA,QAAS3O,KAAK0M,OAAOC,IAAI,MAC7B,CAEA,gBAAIiC,GACA,MAAMC,EAAM7O,KAAK0M,OAAOC,IAAI,OAC5B,IAAKkC,EAAK,MAAO,GACjB,IACI,MAAOC,EAAMC,EAAOC,GAAOH,EAAII,MAAM,KACrC,MAAO,GAAGF,KAASC,KAAOF,GAC9B,CAAA,MACI,OAAOD,CACX,CACJ,CAEA,mBAAIK,GAEA,OADalP,KAAK0M,OAAOC,IAAI,aAAe,CAAA,GAChCwC,UAAY,SAC5B,CAEA,cAAIC,GACA,MAAMC,EAAOrP,KAAK0M,OAAOC,IAAI,aAAe,CAAA,EAC5C,SAAU0C,EAAKC,QAAUD,EAAKE,MAAQF,EAAKG,OAASH,EAAKI,KAAOJ,EAAKK,QACzE,CAEA,kBAAIC,GACA,MAAMN,EAAOrP,KAAK0M,OAAOC,IAAI,aAAe,CAAA,EAC5C,MAAO,CAAC0C,EAAKC,OAAQD,EAAKE,KAAMF,EAAKG,MAAOH,EAAKI,IAAKJ,EAAKK,SAASxK,OAAOC,SAAS4D,KAAK,KAC7F,CAIA,4BAAM6G,GAKF,cAJwB9C,EAAOC,QAC3B,kCACA,6BAGE/M,KAAKgN,WAAW,CAAE6C,iBAAiB,GAAQ,2BAC1C,EACX,CAEA,yBAAMC,GAKF,cAJwBhD,EAAOC,QAC3B,oCACA,yBAGE/M,KAAKgN,WAAW,CAAE6C,iBAAiB,GAAS,6BAC3C,EACX,CAIA,6BAAME,GACF,MAAMC,QAAalD,EAAOU,OACtB,gBACA,oBACA,CAAEC,aAAczN,KAAK0M,MAAMC,IAAI,iBAAmB,KAKtD,OAHa,OAATqD,GAAiBA,EAAKtC,cAChB1N,KAAKgN,WAAW,CAAEiD,aAAcD,EAAKtC,QAAU,iBAElD,CACX,CAEA,2BAAMwC,GACF,MAAMF,QAAalD,EAAOU,OACtB,cACA,kBACA,CAAEC,aAAczN,KAAK0M,MAAMC,IAAI,eAAiB,KAKpD,OAHa,OAATqD,SACMhQ,KAAKgN,WAAW,CAAEmD,WAAYH,EAAKtC,QAAU,eAEhD,CACX,CAEA,0BAAM0C,GACF,MAAMJ,QAAalD,EAAOU,OACtB,aACA,iBACA,CAAEC,aAAczN,KAAK0M,MAAMC,IAAI,cAAgB,KAKnD,OAHa,OAATqD,SACMhQ,KAAKgN,WAAW,CAAEqD,UAAWL,EAAKtC,QAAU,cAE/C,CACX,CAEA,qBAAM4C,GACF,MAAMtN,QAAa8J,EAAOyD,SAAS,CAC/B9Q,MAAO,gBACP+Q,KAAM,KACNhC,OAAQ,CAAC,CAAEwB,KAAM,MAAOlI,KAAM,OAAQnI,MAAO,gBAAiB8Q,KAAM,KACpEzN,KAAM,CAAE6L,IAAK7O,KAAK0M,MAAMC,IAAI,QAAU,MAE1C,OAAK3J,UACChD,KAAKgN,WAAW,CAAE6B,IAAK7L,EAAK6L,KAAO,MAAQ,kBAC1C,EACX,CAEA,0BAAM6B,GACF,MAAMrB,EAAOrP,KAAK0M,MAAMC,IAAI,aAAe,CAAA,EACrC3J,QAAa8J,EAAOyD,SAAS,CAC/B9Q,MAAO,kBACP+Q,KAAM,KACNhC,OAAQ,CAAC,CACLwB,KAAM,WACNlI,KAAM,SACNnI,MAAO,WACP8Q,KAAM,GACNlR,QAAS,CACL,CAAEoR,MAAO,mBAAoBC,KAAM,qBACnC,CAAED,MAAO,kBAAmBC,KAAM,qBAClC,CAAED,MAAO,iBAAkBC,KAAM,sBACjC,CAAED,MAAO,sBAAuBC,KAAM,qBACtC,CAAED,MAAO,oBAAqBC,KAAM,qBACpC,CAAED,MAAO,mBAAoBC,KAAM,oBACnC,CAAED,MAAO,MAAOC,KAAM,OACtB,CAAED,MAAO,gBAAiBC,KAAM,oBAChC,CAAED,MAAO,eAAgBC,KAAM,oBAC/B,CAAED,MAAO,gBAAiBC,KAAM,qBAChC,CAAED,MAAO,aAAcC,KAAM,eAC7B,CAAED,MAAO,gBAAiBC,KAAM,kBAChC,CAAED,MAAO,mBAAoBC,KAAM,oBAG3C5N,KAAM,CAAEmM,SAAUE,EAAKF,UAAY,MAEvC,OAAKnM,UACChD,KAAKgN,WAAW,CAAE6D,SAAU,IAAKxB,EAAMF,SAAUnM,EAAKmM,WAAc,aACnE,EACX,CAEA,yBAAM2B,GACF,MAAMzB,EAAOrP,KAAK0M,MAAMC,IAAI,aAAe,CAAA,EACrC3J,QAAa8J,EAAOyD,SAAS,CAC/B9Q,MAAO,eACP+Q,KAAM,KACNhC,OAAQ,CACJ,CAAEwB,KAAM,SAAUlI,KAAM,OAAQnI,MAAO,SAAUoO,YAAa,cAAe0C,KAAM,IACnF,CAAET,KAAM,OAAQlI,KAAM,OAAQnI,MAAO,OAAQ8Q,KAAM,GACnD,CAAET,KAAM,QAASlI,KAAM,OAAQnI,MAAO,mBAAoB8Q,KAAM,GAChE,CAAET,KAAM,MAAOlI,KAAM,OAAQnI,MAAO,oBAAqB8Q,KAAM,GAC/D,CAAET,KAAM,UAAWlI,KAAM,OAAQnI,MAAO,UAAW8Q,KAAM,IAE7DzN,KAAM,CAAEsM,OAAQD,EAAKC,QAAU,GAAIC,KAAMF,EAAKE,MAAQ,GAAIC,MAAOH,EAAKG,OAAS,GAAIC,IAAKJ,EAAKI,KAAO,GAAIC,QAASL,EAAKK,SAAW,MAErI,IAAK1M,EAAM,OAAO,EAElB,MAAM+N,EAAc,IAAK1B,EAAMC,OAAQtM,EAAKsM,QAAU,GAAIC,KAAMvM,EAAKuM,MAAQ,GAAIC,MAAOxM,EAAKwM,OAAS,GAAIC,IAAKzM,EAAKyM,KAAO,GAAIC,QAAS1M,EAAK0M,SAAW,IAExJ,aADM1P,KAAKgN,WAAW,CAAE6D,SAAUE,GAAe,YAC1C,CACX,CAIA,gBAAM/D,CAAWwB,EAAQ7O,GACrB,MAAMsO,QAAajO,KAAK0M,MAAMwB,KAAKM,GACf,MAAhBP,EAAKhL,QACLjD,KAAK4C,UAAUwL,OAAOrL,QAAQ,GAAGpD,mBAC3BK,KAAKqD,UAEXrD,KAAK4C,UAAUwL,OAAO9K,MAAM2K,EAAKI,SAAW,oBAAoB1O,EAAMqR,gBAE9E,EC9PW,MAAMC,6BAA6B5R,EAC9C,WAAAC,CAAYC,EAAU,IAClBC,MAAM,CACFO,UAAW,yBACXyM,SAAU,8uNA+GPjN,GAEX,CAIA,+BAAM2R,GACF,MAAMC,EAAMnR,KAAK4C,SACX2K,EAAQvN,KAAK0M,MAAMC,IAAI,SAM7B,WAJwBG,EAAOC,QAC3B,0CAA0CQ,cAC1C,wBAEY,OAAO,EAEvB,MAAMU,QAAapL,EAAKuO,KAAK,2BAA4B,CAAE7D,UAM3D,OALIU,EAAKlL,QACLoO,GAAK/C,OAAOrL,QAAQ,6BAEpBoO,GAAK/C,OAAO9K,MAAM2K,EAAKI,SAAW,kCAE/B,CACX,CAEA,mCAAMgD,GACF,MAAMF,EAAMnR,KAAK4C,SACX2K,EAAQvN,KAAK0M,MAAMC,IAAI,SAM7B,WAJwBG,EAAOC,QAC3B,wCAAwCQ,cACxC,4BAEY,OAAO,EAEvB,MAAMU,QAAapL,EAAKuO,KAAK,yBAA0B,CAAE7D,UAMzD,OALIU,EAAKlL,QACLoO,GAAK/C,OAAOrL,QAAQ,2BAEpBoO,GAAK/C,OAAO9K,MAAM2K,EAAKI,SAAW,sCAE/B,CACX,CAEA,2BAAMiD,GACF,MAAMH,EAAMnR,KAAK4C,SACX2K,EAAQvN,KAAK0M,MAAMC,IAAI,SAM7B,WAJwBG,EAAOC,QAC3B,sCAAsCQ,2DACtC,0BAEY,OAAO,EAEvB,MAAMU,QAAapL,EAAKuO,KAAK,uBAAwB,CAAE7D,UAMvD,OALIU,EAAKlL,QACLoO,GAAK/C,OAAOrL,QAAQ,yBAEpBoO,GAAK/C,OAAO9K,MAAM2K,EAAKI,SAAW,8BAE/B,CACX,CAEA,yBAAMkD,GACF,MAAMJ,EAAMnR,KAAK4C,SACXI,QAAa8J,EAAOyD,SAAS,CAC/B9Q,MAAO,eACP+Q,KAAM,KACNhC,OAAQ,CACJ,CAAEwB,KAAM,WAAYlI,KAAM,WAAYnI,MAAO,eAAgB6R,UAAU,EAAMf,KAAM,GAAIgB,KAAM,qCAC7F,CAAEzB,KAAM,UAAWlI,KAAM,WAAYnI,MAAO,mBAAoB6R,UAAU,EAAMf,KAAM,OAG9F,IAAKzN,EAAM,OAAO,EAElB,GAAIA,EAAK0O,WAAa1O,EAAK+J,QAEvB,OADAoE,GAAK/C,OAAO9K,MAAM,2BACX,EAGX,MAAM2K,QAAajO,KAAK0M,MAAMwB,KAAK,CAAEwD,SAAU1O,EAAK0O,WAMpD,OALoB,MAAhBzD,EAAKhL,OACLkO,GAAK/C,OAAOrL,QAAQ,oBAEpBoO,GAAK/C,OAAO9K,MAAM2K,EAAKI,SAAW,2BAE/B,CACX,CAIA,uBAAMsD,GACF,MAAMR,EAAMnR,KAAK4C,SACXgP,EAAa5R,KAAK0M,MAAMC,IAAI,gBAC5B9M,EAAS+R,EAAa,UAAY,SAMxC,WAJwB9E,EAAOC,SACxB6E,EAAa,UAAY,UAA5B,mCACGA,EAAa,UAAY,UAA5B,SAEY,OAAO,EAEvB,MAAM3D,QAAajO,KAAK0M,MAAMwB,KAAK,CAAE2D,cAAeD,IAOpD,OANoB,MAAhB3D,EAAKhL,QACLkO,GAAK/C,OAAOrL,QAAQ,OAAOlD,YACrBG,KAAKqD,UAEX8N,GAAK/C,OAAO9K,MAAM2K,EAAKI,SAAW,aAAaxO,UAE5C,CACX,CAEA,yBAAMiS,GACF,MAAMX,EAAMnR,KAAK4C,SAKjB,WAJwBkK,EAAOC,QAC3B,gGACA,0BAEY,OAAO,EAEvB,MAAMkB,QAAapL,EAAKkP,OAAO,aAAa/R,KAAK0M,MAAMsF,WAQvD,OAPI/D,EAAKlL,SACL/C,KAAK0M,MAAMyB,IAAI,gBAAgB,GAC/BgD,GAAK/C,OAAOrL,QAAQ,gCACd/C,KAAKqD,UAEX8N,GAAK/C,OAAO9K,MAAM2K,EAAKI,SAAW,oCAE/B,CACX,CAEA,4BAAM4D,GACF,MAAMC,EAAa,IAAIC,GAAY,CAAEC,OAAQ,CAAEC,KAAMrS,KAAK0M,MAAMsF,MAChE,UACUE,EAAWI,OACrB,OAASC,GAET,CAEA,MAAMnJ,EAAQ8I,EAAWM,QAAU,GAC7BrK,EAAO,IAAI9I,EAAK,CAClBmN,SAAU,wqEA8Dd,OA9BArE,EAAKsK,SAAWrJ,EAAMR,IAAI8J,GAAKA,EAAEC,OAASD,EAAEC,SAAWD,GAEvDvK,EAAKyK,oBAAsBC,MAAOrO,EAAOwG,KACrC,MAAMgH,EAAKhH,EAAGL,QAAQqH,GAChBc,EAAU1J,EAAMH,KAAKyJ,GAAKK,OAAOL,EAAEV,MAAQe,OAAOf,IAIxD,OAHIc,SACMhG,EAAOkG,cAAc,CAAEvT,MAAO,eAAgBiN,MAAOoG,EAAStE,OAAQyE,GAAaC,KAAK1E,OAAQgC,KAAM,QAEzG,GAGXrI,EAAKgL,sBAAwBN,MAAOrO,EAAOwG,KACvC,MAAMgH,EAAKhH,EAAGL,QAAQqH,GAEtB,SADwBlF,EAAOC,QAAQ,uBAAwB,kBAChD,CACX,MAAM+F,EAAU1J,EAAMH,KAAKyJ,GAAKK,OAAOL,EAAEV,MAAQe,OAAOf,IACpDc,UACMA,EAAQlJ,UACd5J,KAAK4C,UAAUwL,OAAOrL,QAAQ,mBAEtC,CACA,OAAO,SAGL+J,EAAOsG,WAAW,CACpB3T,MAAO,WACP4T,KAAMlL,EACNqI,KAAM,KACN8C,QAAS,CAAC,CAAE1C,KAAM,QAAS2C,MAAO,wBAAyBC,SAAS,OAEjE,CACX,CAEA,+BAAMC,GACF,MAAMtC,EAAMnR,KAAK4C,SACXqL,QAAapL,EAAKC,IAAI,aAAa9C,KAAK0M,MAAMsF,yBAA0B,CAAA,EAAI,CAAE0B,UAAU,IAC9F,IAAKzF,EAAKlL,UAAYkL,EAAKjL,KAEvB,OADAmO,GAAK/C,OAAO9K,MAAM2K,EAAKI,SAAW,kCAC3B,EAGX,MAAMsF,UAAEA,EAAAC,MAAWA,GAAU3F,EAAKjL,KAC5BmF,EAAO,IAAI9I,EAAK,CAClBmN,SAAU,qwBAsBd,OATArE,EAAKwL,UAAYA,EACjBxL,EAAKyL,MAAQA,GAAS,SAEhB9G,EAAOsG,WAAW,CACpB3T,MAAO,iBACP4T,KAAMlL,EACNqI,KAAM,KACN8C,QAAS,CAAC,CAAE1C,KAAM,QAAS2C,MAAO,wBAAyBC,SAAS,OAEjE,CACX,CAIA,+BAAMK,GACF,MAAM1C,EAAMnR,KAAK4C,SAKjB,WAJwBkK,EAAOC,QAC3B,yFACA,wBAEY,OAAO,EAEvB,MAAMkB,QAAapL,EAAKuO,KAAK,aAAapR,KAAK0M,MAAMsF,sBAMrD,OALI/D,EAAKlL,QACLoO,GAAK/C,OAAOrL,QAAQ,wBAEpBoO,GAAK/C,OAAO9K,MAAM2K,EAAKI,SAAW,8BAE/B,CACX,ECvXJ,MAAMyF,GAAiB,CACnBC,OAAQ,YACRC,OAAQ,YACRC,UAAW,eACXC,MAAO,WACPC,SAAU,cACVC,QAAS,eACTC,SAAU,eAGC,MAAMC,8BAA8BjV,EAC/C,WAAAC,CAAYC,EAAU,IAClBC,MAAM,CACFO,UAAW,0BACXyM,SAAU,q7DA+BPjN,IAEPS,KAAKuU,YAAc,EACvB,CAEA,oBAAM9R,GACF,IACI,MAAMwL,QAAapL,EAAKC,IAAI,aAAa9C,KAAK0M,MAAMsF,uBAC9CwC,EAAUvG,GAAMjL,MAAMwR,SAAWvG,GAAMjL,MAAQ,GACrDhD,KAAKuU,YAAcC,EAAQ5L,IAAIM,IAAAA,IACxBA,EACHtJ,KAAMkU,GAAe5K,EAAEuL,WAAa,kBAE5C,OAASlC,GACLvS,KAAKuU,YAAc,EACvB,CACJ,CAEA,oBAAMG,CAAelQ,EAAOwG,GACxB,MAAMgH,EAAKhH,EAAGL,QAAQqH,GAChB2C,EAAa3U,KAAKuU,YAAYtL,KAAKC,GAAK6J,OAAO7J,EAAE8I,MAAQe,OAAOf,IAChEyC,EAAWE,GAAYF,UAAY,eAMzC,WAJwB3H,EAAOC,QAC3B,UAAU0H,mBACV,mBAEY,OAAO,EAEvB,MAAMxG,QAAapL,EAAKkP,OAAO,aAAa/R,KAAK0M,MAAMsF,uBAAuBA,KAO9E,OANI/D,EAAKlL,SACL/C,KAAK4C,UAAUwL,OAAOrL,QAAQ,GAAG0R,4BAC3BzU,KAAKqD,UAEXrD,KAAK4C,UAAUwL,OAAO9K,MAAM2K,EAAKI,SAAW,6BAEzC,CACX,ECpFJ,MAAMuG,GAAiB,CACnBC,OAAQ,SACRtH,MAAO,QACPtF,KAAM,QAGJ6M,GAAW,CAAC,SAAU,QAAS,QAEtB,MAAMC,kCAAkC1V,EACnD,WAAAC,CAAYC,EAAU,IAClBC,MAAM,CACFO,UAAW,8BACXyM,SAAU,qgFAiDPjN,IAEPS,KAAKgV,YAAc,CAAA,CACvB,CAEA,YAAIC,GACA,OAAOH,GAASlM,IAAIsM,IAAA,CAAShN,IAAKgN,EAAIvV,MAAOiV,GAAeM,IAAOA,IACvE,CAEA,kBAAIC,GACA,OAAOjS,OAAOkS,KAAKpV,KAAKgV,aAAaK,OAAS,CAClD,CAEA,kBAAIC,GACA,OAAOpS,OAAOkS,KAAKpV,KAAKgV,aAAaO,OAAO3M,IAAI4M,IAAA,CAC5CA,OACAC,UAAWD,EAAKE,QAAQ,QAAS,KAAKA,QAAQ,QAASxM,GAAKA,EAAEyM,eAC9DC,QAASd,GAASlM,IAAIiN,IAAA,CAClBL,OACAK,UACAC,SAA+C,IAAtC9V,KAAKgV,YAAYQ,KAAQK,QAG9C,CAEA,oBAAMpT,GACF,IACI,MAAMwL,QAAapL,EAAKC,IAAI,aAAa9C,KAAK0M,MAAMsF,8BAA+B,CAAA,EAAI,CAAE0B,UAAU,IACnG1T,KAAKgV,YAAc/G,GAAMjL,MAAMgS,aAAe/G,GAAMjL,MAAQ,CAAA,CAChE,OAASuP,GACLvS,KAAKgV,YAAc,CAAA,CACvB,CACJ,CAEA,wBAAMe,CAAmBvR,EAAOwG,GAC5B,MAAMwK,EAAOxK,EAAGL,QAAQ6K,KAClBK,EAAU7K,EAAGL,QAAQkL,QACrBC,EAAU9K,EAAG8K,QAEd9V,KAAKgV,YAAYQ,KAClBxV,KAAKgV,YAAYQ,GAAQ,CAAA,GAE7BxV,KAAKgV,YAAYQ,GAAMK,GAAWC,EAElC,IACI,MAAM7H,QAAapL,EAAKuO,KAAK,aAAapR,KAAK0M,MAAMsF,8BAA+B,CAChFgD,YAAa,CAAEQ,CAACA,GAAO,CAAEK,CAACA,GAAUC,MAEnC7H,EAAKlL,UACN/C,KAAK4C,UAAUwL,OAAO9K,MAAM2K,EAAKI,SAAW,+BAC5CrD,EAAG8K,SAAWA,EAEtB,OAASvD,GACLvS,KAAK4C,UAAUwL,OAAO9K,MAAM,+BAC5B0H,EAAG8K,SAAWA,CAClB,CACA,OAAO,CACX,ECnHW,MAAME,4BAA4B3W,EAC7C,WAAAC,CAAYC,EAAU,IAClBC,MAAM,CACFO,UAAW,yBACXyM,SAAU,2lCAgBPjN,IAEPS,KAAKiW,QAAU,EACnB,CAEA,oBAAMxT,SACIzC,KAAKkW,WACf,CAEA,eAAMA,GACF,IACI,MAAMjI,QAAapL,EAAKC,IAAI,aAAa9C,KAAK0M,MAAMsF,cAAe,CAAA,EAAI,CAAA,EAAI,CAAE0B,UAAU,IACvF1T,KAAKiW,QAAUhI,EAAKlL,SAAWoT,MAAMC,QAAQnI,EAAKjL,MAAQiL,EAAKjL,KAAO,EAC1E,OAASuP,GACLvS,KAAKiW,QAAU,EACnB,CACJ,CAEA,aAAAvP,GACI1G,KAAKqW,iBACT,CAEA,eAAAA,GACI,MAAMjM,EAAYpK,KAAKyE,SAASG,cAAc,kBAC9C,IAAKwF,EAAW,OAEhB,IAAKpK,KAAKiW,QAAQZ,OAQd,YAPAjL,EAAUxE,UAAY,mPAU1B,MAAM0Q,EAAOtW,KAAKiW,QAAQrN,IAAIV,IAC1B,MAAM8H,EAAO9H,EAAI8H,MAAQ,UACnBuG,EAAUrO,EAAIqO,QAAU,IAAIzS,KAAmB,IAAdoE,EAAIqO,SAAgBC,qBAAuB,GAC5EC,EAAUvO,EAAIuO,QAAU,IAAI3S,KAAmB,IAAdoE,EAAIuO,SAAgBD,qBAAuB,QAC5EE,EAAWxO,EAAIyO,UAAY,IAAI7S,KAAqB,IAAhBoE,EAAIyO,WAAkBH,qBAAuB,QACjFI,EAAM1O,EAAI2O,aAAaxB,OAASnN,EAAI2O,YAAY9N,KAAK,MAAQ,MAOnE,MAAO,iOAIkCiH,MAVN,IAAlB9H,EAAI4O,UAEf,+CACA,qLACe5O,EAAI6O,aAAe,GAAG7O,EAAI6O,kBAAoB,8FASLR,kFACHE,uFACKC,+EACRE,0NAIoD1O,EAAI8J,wKAKjHjJ,KAAK,IAERqB,EAAUxE,UAAY,yBAAyB0Q,SACnD,CAEA,uBAAMU,CAAkBxS,EAAOwG,GAC3B,MAAMgH,EAAKhH,EAAGL,QAAQqH,GACtB,IAAKA,EAAI,OAAO,EAMhB,WAJwBlF,EAAOC,QAC3B,+EACA,mBAEY,OAAO,EAEvB,MAAMkB,QAAapL,EAAKkP,OAAO,aAAa/R,KAAK0M,MAAMsF,eAAeA,IAAM,CAAA,EAAI,CAAA,EAAI,CAAE0B,UAAU,IAQhG,OAPIzF,EAAKlL,SACL/C,KAAK4C,UAAUwL,OAAOrL,QAAQ,yBACxB/C,KAAKkW,YACXlW,KAAKqW,mBAELrW,KAAK4C,UAAUwL,OAAO9K,MAAM2K,EAAKI,SAAW,6BAEzC,CACX,ECxFJ,MAAM4I,kBAAkBC,GACpB,cAAIC,GACA,MAAMC,EAAMpX,KAAK0M,OAAOC,IAAI,gBAAgB0K,QAAU,CAAA,EAChDC,EAAKtX,KAAK0M,OAAOC,IAAI,gBAAgB2K,IAAM,CAAA,EAIjD,MAHiB,CAAC,SAAU,WAAWC,KAAKC,IACvCJ,EAAIK,QAAU,IAAIC,SAASF,KAAOF,EAAGG,QAAU,IAAIC,SAASF,IAE/C,WAAa,WACnC,CACA,cAAIG,GACA,MAAMP,EAAMpX,KAAK0M,OAAOC,IAAI,gBAAgB0K,QAAU,CAAA,EACtD,MAAO,GAAGD,EAAIQ,OAAS,MAAMR,EAAIK,QAAU,KAAK/J,QAAU,gBAC9D,CACA,eAAImK,GACA,OAAO7X,KAAK0M,OAAOC,IAAI,gBAAgB0K,QAAQ3K,OAAS,EAC5D,CACA,eAAIoL,GACA,MAAMC,EAAK/X,KAAK0M,OAAOC,IAAI,gBAAgBqL,YAAc,CAAA,EACzD,OAAOD,EAAGN,OAAS,GAAGM,EAAGN,UAAUM,EAAGE,OAAS,KAAKvK,OAAS,EACjE,CACA,UAAIwK,GACA,MAAMZ,EAAKtX,KAAK0M,OAAOC,IAAI,gBAAgB2K,IAAM,CAAA,EACjD,OAAOA,EAAGG,OAAS,GAAGH,EAAGG,UAAUH,EAAGW,OAAS,KAAKvK,OAAS,EACjE,CACA,cAAIyK,GACA,MAAO,CAACnY,KAAK8X,YAAa9X,KAAKkY,QAAQhT,OAAOC,SAAS4D,KAAK,QAAU,GAC1E,EAGJ,MAAMqP,oBAAoBlB,GACtB,cAAIC,GACA,MAAMC,EAAMpX,KAAK0M,OAAOC,IAAI,gBAAgB0L,aAAahB,QAAU,CAAA,EAC7DC,EAAKtX,KAAK0M,OAAOC,IAAI,gBAAgB0L,aAAaf,IAAM,CAAA,EAI9D,MAHiB,CAAC,SAAU,WAAWC,KAAKC,IACvCJ,EAAIK,QAAU,IAAIC,SAASF,KAAOF,EAAGG,QAAU,IAAIC,SAASF,IAE/C,WAAa,WACnC,CACA,eAAIM,GACA,MAAMC,EAAK/X,KAAK0M,OAAOC,IAAI,gBAAgB0L,aAAaL,YAAc,CAAA,EACtE,OAAOD,EAAGN,OAAS,GAAGM,EAAGN,UAAUM,EAAGE,OAAS,KAAKvK,OAAS,SACjE,CACA,cAAIiK,GACA,MAAMP,EAAMpX,KAAK0M,OAAOC,IAAI,gBAAgB0L,aAAahB,QAAU,CAAA,EACnE,MAAO,GAAGD,EAAIQ,OAAS,MAAMR,EAAIK,QAAU,KAAK/J,QAAU,SAC9D,CACA,gBAAI4K,GACA,MAAMC,EAAMvY,KAAK0M,OAAOC,IAAI,gBAAkB,CAAA,EAC9C,MAAO,CAAC4L,EAAIhJ,KAAMgJ,EAAIC,QAAQtT,OAAOC,SAAS4D,KAAK,OAASwP,EAAIE,cAAgB,GACpF,CACA,eAAIC,GACA,OAAO1Y,KAAK0M,OAAOC,IAAI,gBAAgB8L,cAAgB,EAC3D,CACA,eAAIE,GACA,MAAMJ,EAAMvY,KAAK0M,OAAOC,IAAI,gBAAkB,CAAA,EACxCiM,EAAQ,GAId,OAHIL,EAAIM,QAAQD,EAAM3Q,KAAK,iFACvBsQ,EAAIO,QAAQF,EAAM3Q,KAAK,sEACvBsQ,EAAIQ,UAAUH,EAAM3Q,KAAK,mFACtB2Q,EAAM7P,KAAK,IACtB,CACA,kBAAIiQ,GACA,MAAMT,EAAMvY,KAAK0M,OAAOC,IAAI,gBAAkB,CAAA,EAC9C,SAAU4L,EAAIM,QAAUN,EAAIO,QAAUP,EAAIQ,SAC9C,EAGJ,MAAME,iBAAiB5Z,EACnB,WAAAC,CAAYC,EAAU,IAClBC,MAAM,CACFO,UAAW,eACRR,IAIPS,KAAK0M,MAAQnN,EAAQmN,OAAS,IAAIwM,EAAK3Z,EAAQyD,MAAQ,IAGvDhD,KAAKmZ,YAAc,KAGnBnZ,KAAKwM,SAAW,8iBAWpB,CAEA,YAAM5L,GAEFZ,KAAKoD,OAAS,IAAI/D,EAAK,CACnB8C,YAAa,cACbqK,SAAU,svEAmCdxM,KAAKoD,OAAOgW,SAASpZ,KAAK0M,OAC1B1M,KAAKoC,SAASpC,KAAKoD,QAInB,MAAMiW,EAAc,IAAI9M,oBAAoB,CAAEG,MAAO1M,KAAK0M,QACpD4M,EAAe,IAAI5K,qBAAqB,CAAEhC,MAAO1M,KAAK0M,QACtD6M,EAAe,IAAItI,qBAAqB,CAAEvE,MAAO1M,KAAK0M,QACtD8M,EAAgB,IAAIlF,sBAAsB,CAAE5H,MAAO1M,KAAK0M,QACxD+M,EAAoB,IAAI1E,0BAA0B,CAAErI,MAAO1M,KAAK0M,QAChEgN,EAAc,IAAI1D,oBAAoB,CAAEtJ,MAAO1M,KAAK0M,QAEpDiN,EAAY,IAAIC,GAAS,CAC3BpL,OAAQ0K,EAAKW,kBACbnN,MAAO1M,KAAK0M,MACZoN,oBAAoB,IAIlBC,EAAoB,IAAIC,GAAW,CACrC5H,OAAQ,CAAEC,KAAMrS,KAAK0M,MAAMC,IAAI,MAAO6D,KAAM,KAE1CyJ,EAAa,IAAIC,GAAU,CAC7BhI,WAAY6H,EACZI,oBAAqB,CAAC,QACtBC,QAAS,CACL,CAAElS,IAAK,UAAWvI,MAAO,cAAe0a,UAAW,OAAQC,UAAU,GACrE,CAAEpS,IAAK,aAAcvI,MAAO,aAAc2a,UAAU,GACpD,CAAEpS,IAAK,yBAA0BvI,MAAO,kBAK1C4a,EAAmB,IAAIC,EAAkB,CAC3CpI,OAAQ,CAAE5B,KAAM,EAAGiK,WAAY,eAAgBC,SAAU1a,KAAK0M,MAAMC,IAAI,SAEtEgO,EAAa,IAAIT,GAAU,CAC7BhI,WAAYqI,EACZJ,oBAAqB,CAAC,aAAc,YACpCC,QAAS,CACL,CAAElS,IAAK,KAAMvI,MAAO,KAAM2a,UAAU,EAAMM,MAAO,QACjD,CAAE1S,IAAK,UAAWvI,MAAO,OAAQ0a,UAAW,WAAYC,UAAU,EAAMM,MAAO,SAC/E,CAAE1S,IAAK,iBAAkBvI,MAAO,YAChC,CAAEuI,IAAK,QAASvI,MAAO,YAKzBkb,EAAc,IAAIX,GAAU,CAC9BhI,WAAY,IAAI4I,EAAe,CAAE1I,OAAQ,CAAE5B,KAAM,GAAI6B,KAAMrS,KAAK0M,MAAMC,IAAI,SAC1EwN,oBAAqB,CAAC,QACtBY,YAAa,OACbC,UAAW/D,UACXmD,QAAS,CACL,CACIlS,IAAK,cACLvI,MAAO,SACP6M,SAAU,kqBAUd,CAAEtE,IAAK,aAAcvI,MAAO,aAAc0a,UAAW,iBAAkBO,MAAO,SAC9E,CAAE1S,IAAK,YAAavI,MAAO,YAAa0a,UAAW,iBAAkBO,MAAO,YAK9EK,EAAgB,IAAIf,GAAU,CAChChI,WAAY,IAAIgJ,EAAuB,CAAE9I,OAAQ,CAAE5B,KAAM,GAAI6B,KAAMrS,KAAK0M,MAAMC,IAAI,SAClFwN,oBAAqB,CAAC,QACtBY,YAAa,OACbC,UAAW5C,YACXgC,QAAS,CACL,CACIlS,IAAK,cACLvI,MAAO,UACP6M,SAAU,w4BAWd,CAAEtE,IAAK,YAAavI,MAAO,YAAa0a,UAAW,iBAAkBO,MAAO,YAK9EO,EAAc,IAAIC,EAAe,CACnChJ,OAAQ,CAAE5B,KAAM,EAAG6B,KAAMrS,KAAK0M,MAAMC,IAAI,SAEtC0O,EAAkB,IAAInB,GAAU,CAClChI,WAAYiJ,EACZhB,oBAAqB,CAAC,QACtBC,QAAS,CACL,CAAElS,IAAK,2BAA4BvI,MAAO,YAAa2a,UAAU,GACjE,CAAEpS,IAAK,gCAAiCvI,MAAO,UAAW0a,UAAW,gBACrE,CAAEnS,IAAK,wBAAyBvI,MAAO,KAAM0a,UAAW,gBACxD,CAAEnS,IAAK,aAAcvI,MAAO,aAAc0a,UAAW,kBACrD,CAAEnS,IAAK,YAAavI,MAAO,YAAa0a,UAAW,mBAEvD7J,KAAM,IAIJ8K,EAAiB,IAAIC,GAAQ,CAC/BnJ,OAAQ,CAAE5B,KAAM,EAAGiK,WAAY,eAAgBC,SAAU1a,KAAK0M,MAAMC,IAAI,SAEtE6O,EAAW,IAAItB,GAAU,CAC3BhI,WAAYoJ,EACZvT,YAAa,YACboS,oBAAqB,CAAC,aAAc,YACpCC,QAAS,CACL,CACIlS,IAAK,UAAWvI,MAAO,YAAa2a,UAAU,EAAMD,UAAW,iBAC/DnV,OAAQ,CAAE8K,KAAM,UAAWlI,KAAM,YAAa2T,UAAW,WAAYC,QAAS,SAAUC,UAAW,WAAYhc,MAAO,aAAcic,OAAQ,aAAcC,cAAe,eAAgBC,UAAW,SAExM,CACI5T,IAAK,QAASvI,MAAO,QAAS2a,UAAU,EACxCpV,OAAQ,CAAE4C,KAAM,SAAUvI,QAAS,CAAC,CAAEoR,MAAO,OAAQhR,MAAO,QAAU,CAAEgR,MAAO,UAAWhR,MAAO,WAAa,CAAEgR,MAAO,QAAShR,MAAO,YAE3I,CAAEuI,IAAK,OAAQvI,MAAO,OAAQuF,OAAQ,CAAE4C,KAAM,SAC9C,CAAEkI,KAAM,MAAOrQ,MAAO,UAKxBoc,EAAqB,IAAIR,GAAQ,CACnCnJ,OAAQ,CAAE5B,KAAM,EAAGwL,IAAKhc,KAAK0M,MAAMC,IAAI,SAErCsP,EAAe,IAAI/B,GAAU,CAC/BhI,WAAY6J,EACZ5B,oBAAqB,CAAC,OACtBpS,YAAa,YACbqS,QAAS,CACL,CACIlS,IAAK,UAAWvI,MAAO,YAAa2a,UAAU,EAAMD,UAAW,iBAC/DnV,OAAQ,CAAE8K,KAAM,UAAWlI,KAAM,YAAa2T,UAAW,WAAYC,QAAS,SAAUC,UAAW,WAAYhc,MAAO,aAAcic,OAAQ,aAAcC,cAAe,eAAgBC,UAAW,SAExM,CACI5T,IAAK,QAASvI,MAAO,QAAS2a,UAAU,EACxCpV,OAAQ,CAAE4C,KAAM,SAAUvI,QAAS,CAAC,CAAEoR,MAAO,OAAQhR,MAAO,QAAU,CAAEgR,MAAO,UAAWhR,MAAO,WAAa,CAAEgR,MAAO,QAAShR,MAAO,YAE3I,CAAEuI,IAAK,OAAQvI,MAAO,OAAQuF,OAAQ,CAAE4C,KAAM,SAC9C,CAAEkI,KAAM,OAAQrQ,MAAO,WAK/BK,KAAKmZ,YAAc,IAAIxS,YAAY,CAC/BxE,YAAa,eACb0E,cAAe,UACfC,SAAU,IACVC,eAAgB,eAChBC,kBAAkB,EAClBC,SAAU,IACVL,SAAU,CACN,CAAEsB,IAAK,UAAWvI,MAAO,UAAWC,KAAM,YAAauI,KAAMkR,GAC7D,CAAEnR,IAAK,WAAYvI,MAAO,WAAYC,KAAM,kBAAmBuI,KAAMmR,GACrE,CAAEpR,IAAK,WAAYvI,MAAO,WAAYC,KAAM,iBAAkBuI,KAAMoR,GACpE,CAAErR,IAAK,YAAavI,MAAO,YAAaC,KAAM,UAAWuI,KAAMqR,GAC/D,CAAE1R,KAAM,UAAWnI,MAAO,UAC1B,CAAEuI,IAAK,cAAevI,MAAO,cAAeC,KAAM,kBAAmBuI,KAAMwR,GAC3E,CAAEzR,IAAK,SAAUvI,MAAO,SAAUC,KAAM,YAAauI,KAAM8R,GAC3D,CAAE/R,IAAK,WAAYvI,MAAO,WAAYC,KAAM,SAAUuI,KAAMuR,GAC5D,CAAE5R,KAAM,UAAWnI,MAAO,YAC1B,CAAEuI,IAAK,SAAUvI,MAAO,SAAUC,KAAM,oBAAqBuI,KAAMwS,GACnE,CAAEzS,IAAK,WAAYvI,MAAO,eAAgBC,KAAM,mBAAoBuI,KAAM8T,EAAclU,YAAa,aACrG,CAAEG,IAAK,OAAQvI,MAAO,cAAeC,KAAM,kBAAmBuI,KAAMqT,EAAUzT,YAAa,aAC3F,CAAED,KAAM,UAAWnI,MAAO,WAC1B,CAAEuI,IAAK,UAAWvI,MAAO,UAAWC,KAAM,YAAauI,KAAM0S,GAC7D,CAAE3S,IAAK,YAAavI,MAAO,YAAaC,KAAM,aAAcuI,KAAM8S,GAClE,CAAE/S,IAAK,eAAgBvI,MAAO,eAAgBC,KAAM,WAAYuI,KAAMkT,GACtE,CAAEvT,KAAM,UAAWnI,MAAO,YAC1B,CAAEuI,IAAK,gBAAiBvI,MAAO,gBAAiBC,KAAM,UAAWuI,KAAMsR,MAG/EzZ,KAAKoC,SAASpC,KAAKmZ,aAGnB,MAAM+C,EAAW,IAAIC,EAAY,CAC7Bha,YAAa,oBACbpC,UAAW,yCACXqc,QAASpc,KAAK0M,MACdhF,OAAQ,CACJ9H,KAAM,yBACNwJ,MAAO,CACH,CAAEzJ,MAAO,YAAaE,OAAQ,YAAaD,KAAM,gBAC7CI,KAAK0M,MAAMC,IAAI,UACb,CAAC,CAAEhN,MAAO,eAAgBE,OAAQ,eAAgBD,KAAM,gBACxD,GACN,CAAEkI,KAAM,WACR,CAAEnI,MAAO,sBAAuBE,OAAQ,sBAAuBD,KAAM,eACrE,CAAED,MAAO,wBAAyBE,OAAQ,kBAAmBD,KAAM,iBACnE,CAAED,MAAO,sBAAuBE,OAAQ,sBAAuBD,KAAM,sBACrE,CAAEkI,KAAM,cACJ9H,KAAK0M,MAAMC,IAAI,qBACb,GACA,CACE,CAAEhN,MAAO,0BAA2BE,OAAQ,0BAA2BD,KAAM,qBAC7E,CAAED,MAAO,qBAAsBE,OAAQ,qBAAsBD,KAAM,sBAEvEI,KAAK0M,MAAMC,IAAI,kBAAoB3M,KAAK0M,MAAMC,IAAI,qBAChD,CAAC,CAAEhN,MAAO,qBAAsBE,OAAQ,qBAAsBD,KAAM,mBACpE,GACN,CAAEkI,KAAM,WACR9H,KAAK0M,MAAMC,IAAI,aACT,CAAEhN,MAAO,kBAAmBE,OAAQ,kBAAmBD,KAAM,kBAC7D,CAAED,MAAO,gBAAiBE,OAAQ,gBAAiBD,KAAM,uBAI3EI,KAAKoC,SAAS8Z,EAClB,CAIA,sBAAMG,SACIvP,EAAOkG,cAAc,CACvBvT,MAAO,WAAWO,KAAK0M,MAAMsF,MAAMhS,KAAKT,QAAQ+c,YAChD5P,MAAO1M,KAAK0M,MACZ6P,WAAYC,EAAUtJ,MAE9B,CAEA,yBAAMuJ,GAKF,cAJwB3P,EAAOC,QAC3B,oEACA,mBAKgB,aADD/M,KAAK0M,MAAMwB,KAAK,CAAEwO,OAAQ,QACpCzZ,OACLjD,KAAK4C,SAASwL,MAAMrL,QAAQ,kBAE5B/C,KAAK4C,SAASwL,MAAM9K,MAAM,2BAEvB,EACX,CAEA,+BAAM4N,GACF,MAAM3D,EAAQvN,KAAK0M,MAAMC,IAAI,SAK7B,WAJwBG,EAAOC,QAC3B,0CAA0CQ,cAC1C,wBAEY,OAAO,EAEvB,MAAMU,QAAapL,EAAKuO,KAAK,2BAA4B,CAAE7D,UAM3D,OALIU,EAAKlL,QACL/C,KAAK4C,SAASwL,MAAMrL,QAAQ,6BAE5B/C,KAAK4C,SAASwL,MAAM9K,MAAM2K,EAAKI,SAAW,kCAEvC,CACX,CAEA,2BAAMiD,GACF,MAAM/D,EAAQvN,KAAK0M,MAAMC,IAAI,SAK7B,WAJwBG,EAAOC,QAC3B,sCAAsCQ,cACtC,0BAEY,OAAO,EAEvB,MAAMU,QAAapL,EAAKuO,KAAK,uBAAwB,CAAE7D,UAMvD,OALIU,EAAKlL,QACL/C,KAAK4C,SAASwL,MAAMrL,QAAQ,yBAE5B/C,KAAK4C,SAASwL,MAAM9K,MAAM2K,EAAKI,SAAW,8BAEvC,CACX,CAEA,+BAAMwF,GAKF,WAJwB/G,EAAOC,QAC3B,+EACA,wBAEY,OAAO,EAEvB,MAAMkB,QAAapL,EAAKuO,KAAK,aAAapR,KAAK0M,MAAMsF,sBAMrD,OALI/D,EAAKlL,QACL/C,KAAK4C,SAASwL,MAAMrL,QAAQ,wBAE5B/C,KAAK4C,SAASwL,MAAM9K,MAAM2K,EAAKI,SAAW,8BAEvC,CACX,CAEA,mCAAMgD,GACF,MAAM9D,EAAQvN,KAAK0M,MAAMC,IAAI,SAK7B,WAJwBG,EAAOC,QAC3B,wCAAwCQ,cACxC,4BAEY,OAAO,EAEvB,MAAMU,QAAapL,EAAKuO,KAAK,yBAA0B,CAAE7D,UAMzD,OALIU,EAAKlL,QACL/C,KAAK4C,SAASwL,MAAMrL,QAAQ,2BAE5B/C,KAAK4C,SAASwL,MAAM9K,MAAM2K,EAAKI,SAAW,sCAEvC,CACX,CAEA,8BAAMxB,GAKF,cAJwBC,EAAOC,QAC3B,gBAAgB/M,KAAK0M,MAAMC,IAAI,iCAC/B,yBAKgB,aADD3M,KAAK0M,MAAMwB,KAAK,CAAEjB,mBAAmB,KAC/ChK,OACLjD,KAAK4C,SAASwL,MAAMrL,QAAQ,4BAE5B/C,KAAK4C,SAASwL,MAAM9K,MAAM,2BAEvB,EACX,CAEA,8BAAM6J,GAKF,cAJwBL,EAAOC,QAC3B,gBAAgB/M,KAAK0M,MAAMC,IAAI,wCAC/B,yBAKgB,aADD3M,KAAK0M,MAAMwB,KAAK,CAAEd,mBAAmB,KAC/CnK,OACLjD,KAAK4C,SAASwL,MAAMrL,QAAQ,4BAE5B/C,KAAK4C,SAASwL,MAAM9K,MAAM,2BAEvB,EACX,CAEA,0BAAMqZ,GACF,OAAI3c,KAAK0M,MAAMC,IAAI,aACR3M,KAAK4c,yBAEL5c,KAAK6c,sBAEpB,CAEA,4BAAMD,GAEF,cADwB9P,EAAOC,QAAQ,qDAInB,aADD/M,KAAK0M,MAAMwB,KAAK,CAAE4I,WAAW,KACvC7T,OACLjD,KAAK4C,SAASwL,MAAMrL,QAAQ,oBAE5B/C,KAAK4C,SAASwL,MAAM9K,MAAM,8BAEvB,EACX,CAEA,0BAAMuZ,GAEF,cADwB/P,EAAOC,QAAQ,mDAInB,aADD/M,KAAK0M,MAAMwB,KAAK,CAAE4I,WAAW,KACvC7T,OACLjD,KAAK4C,SAASwL,MAAMrL,QAAQ,kBAE5B/C,KAAK4C,SAASwL,MAAM9K,MAAM,4BAEvB,EACX,CAEA,iBAAMuG,CAAYiT,GACV9c,KAAKmZ,mBACCnZ,KAAKmZ,YAAYtP,YAAYiT,EAE3C,CAEA,gBAAAhR,GACI,OAAO9L,KAAKmZ,YAAcnZ,KAAKmZ,YAAYrN,mBAAqB,IACpE,CAGA,aAAMiR,CAAQC,GACV,OAAOhd,KAAK6J,YAAYmT,EAC5B,CAEA,YAAAC,GACI,OAAOjd,KAAK8L,kBAChB,CAEA,cAAAO,GAEA,CAGA,aAAOC,CAAO/M,EAAU,IACpB,OAAO,IAAI0Z,SAAS1Z,EACxB,EAGJ2Z,EAAKgE,WAAajE,SCzjBlB,MAAMkE,sBAAsBC,EACxB,WAAA9d,CAAYC,EAAU,IAClBC,MAAM,IACCD,EACHyQ,KAAM,cACNqN,SAAU,eACVlX,OAAQ,cACRmX,WAAYC,EAEZC,kBAAmB,CAAEpa,QAAQ,GAE7Bqa,aAAc,CACVlI,KAAM,iBACNuB,WAAW,GAIfsD,QAAS,CACL,CACIlS,IAAK,KACLvI,MAAO,KACP2a,UAAU,EACV/G,MAAO,cAQX,CACIrL,IAAK,sCACLvI,MAAO,gBAEX,CACIA,MAAO,OACPuI,IAAK,2BACLsE,SAAU,m5BAOV8N,UAAU,GAEd,CACIpS,IAAK,QACLvI,MAAO,QACP+d,WAAY,KACZ3d,UAAW,mBAQf,CACImI,IAAK,gBACLvI,MAAO,gBACP0a,UAAW,WACXta,UAAW,oBAInB4d,QAAS,CACL,CACIzV,IAAK,YACLvI,MAAO,SACPmI,KAAM,UACN2F,cAAc,GAElB,CACIvF,IAAK,QACLvI,MAAO,QACPmI,KAAM,OACN2F,aAAc,IAElB,CACIvF,IAAK,WACLvI,MAAO,WACPmI,KAAM,OACN2F,aAAc,IAElB,CACIvF,IAAK,wBACLvI,MAAO,aACPmI,KAAM,OACN2F,aAAc,IAElB,CACIvF,IAAK,gBACLJ,KAAM,YACN2T,UAAW,WACXC,QAAS,SACTC,UAAW,WACXhc,MAAO,aACPic,OAAQ,aACRC,cAAe,eACfC,UAAW,SAKnB8B,YAAY,EACZC,YAAY,EACZvD,UAAU,EACVwD,YAAY,EACZC,WAAW,EAGXC,aAAa,EACbC,SAAS,EACTC,YAAY,EAGZC,aAAc,oDAGdC,YAAa,CACT,CACIxe,KAAM,YACNC,OAAQ,OACRF,MAAO,gBAEX,CACIC,KAAM,kBACNC,OAAQ,mBACRF,MAAO,oBAEX,CACIC,KAAM,YACNC,OAAQ,kBACRF,MAAO,mBAEX,CAAEmc,WAAW,GACb,CACIlc,KAAM,cACNC,OAAQ,cACRF,MAAO,gBAKf0e,aAAc,CACVC,SAAS,EACTC,UAAU,EACVC,OAAO,EACPC,YAAY,IAGxB,CAEA,6BAAMC,CAAwBla,EAAOC,GAEjCD,EAAMyG,iBACN,MAAM0T,EAAO3e,KAAKkS,WAAWvF,IAAIlI,EAAQkG,QAAQqH,UAC5BlF,EAAOkG,cAAc,CACxCtG,MAAOiS,EACPnO,KAAM,KACN/Q,MAAO,yBAAyBkf,EAAKC,EAAErQ,YACvCC,OAAQgO,EAAUzU,YAAYyG,QAEpC,CAEA,4BAAMqQ,CAAuBra,EAAOC,GAEhC,MAAMka,EAAO3e,KAAKkS,WAAWvF,IAAIlI,EAAQkG,QAAQqH,IAC3ChP,QAAa8J,EAAOyD,SAAS,CAC/B9Q,MAAO,wBAAwBkf,EAAKC,EAAErQ,YACtCC,OAAQ,CACJ,CACI1G,KAAM,OACNkI,KAAM,WACNW,MAAOgO,EAAKhS,IAAI,UAAYgS,EAAKhS,IAAI,YACrCmS,WAAY,CACRC,aAAc,WACdC,SAAU,WACVC,SAAU,KACVC,MAAO,wEAGf,CACIlP,KAAM,eACNrQ,MAAO,eACPmI,KAAM,WACNqX,cAAe,MACf3N,UAAU,EACV4N,YAAY,EACZN,WAAY,CACRC,aAAc,oBAM9B,GAAI/b,GAAQA,EAAKqc,aAAc,CAG3B,GADeC,EAAUC,sBAAsBvc,EAAKqc,cACzCG,MAAQ,EAGf,OAFAxf,KAAK4C,SAASwL,MAAM9K,MAAM,iJACpBtD,KAAK6e,uBAAuBra,EAAOC,IAG7C,MAAMwJ,QAAa0Q,EAAKzQ,KAAK,CAACmR,aAAcrc,EAAKqc,eAC5Crf,KAAKyf,iBAAiBxR,UACjBjO,KAAK6e,uBAAuBra,EAAOC,EAEjD,CACJ,CAEA,gBAAAgb,CAAiBxR,GACb,OAAIA,EAAKlL,SACL/C,KAAK4C,SAASwL,MAAMrL,QAAQ,kCACrB,IAEHkL,EAAKjL,MAAQiL,EAAKjL,KAAKM,MACvBtD,KAAK4C,SAASwL,MAAM9K,MAAM2K,EAAKjL,KAAKM,OAEpCtD,KAAK4C,SAASwL,MAAM9K,MAAM,8BAG3B,EACX,CAEA,wBAAMoc,CAAmBlb,EAAOC,GAC5B,MAAMka,EAAO3e,KAAKkS,WAAWvF,IAAIlI,EAAQkG,QAAQqH,IAC3C/D,QAAa0Q,EAAKzQ,KAAK,CAACyR,aAAa,IAC3C,OAAI1R,EAAKlL,SACL/C,KAAK4C,SAASwL,MAAMrL,QAAQ,6BACrB,IAEHkL,EAAKjL,MAAQiL,EAAKjL,KAAKM,MACvBtD,KAAK4C,SAASwL,MAAM9K,MAAM2K,EAAKjL,KAAKM,OAEpCtD,KAAK4C,SAASwL,MAAM9K,MAAM,0BAG3B,EACX,ECzOJ,MAAMsc,mBAAmBvgB,EACrB,WAAAC,CAAYC,EAAU,IAClBC,MAAM,CACFO,UAAW,iBACRR,IAGPS,KAAK0M,MAAQnN,EAAQmN,OAAS,IAAImT,GAAOtgB,EAAQyD,MAAQ,IAEzDhD,KAAKwM,SAAW,uiBAWpB,CAEA,YAAM5L,GAEFZ,KAAKoD,OAAS,IAAI/D,EAAK,CACnB8C,YAAa,gBACbqK,SAAU,6xEAsCdxM,KAAKoD,OAAOgW,SAASpZ,KAAK0M,OAC1B1M,KAAKoC,SAASpC,KAAKoD,QAGnB,MAAM0c,EAAc,IAAIzgB,EAAK,CACzBqN,MAAO1M,KAAK0M,MACZF,SAAU,+xHA+DRuT,EAAkB,IAAInG,GAAS,CACjCpL,OAAQqR,GAAOhG,kBACfnN,MAAO1M,KAAK0M,MACZoN,oBAAoB,IAIlB0B,EAAW,IAAItB,GAAU,CAC3BhI,WAAY,IAAIqJ,GAAQ,CACpBnJ,OAAQ,CAAE5B,KAAM,GAAIiK,WAAY,iBAAkBC,SAAU1a,KAAK0M,MAAMC,IAAI,SAE/E5E,YAAa,YACboS,oBAAqB,CAAC,aAAc,YACpCC,QAAS,CACL,CACIlS,IAAK,UAAWvI,MAAO,YAAa2a,UAAU,EAAMD,UAAW,iBAC/DnV,OAAQ,CAAE8K,KAAM,UAAWlI,KAAM,YAAa2T,UAAW,WAAYC,QAAS,SAAUC,UAAW,WAAYhc,MAAO,aAAcic,OAAQ,aAAcC,cAAe,eAAgBC,UAAW,SAExM,CAAE5T,IAAK,QAASvI,MAAO,QAAS2a,UAAU,GAC1C,CAAEpS,IAAK,OAAQvI,MAAO,QACtB,CAAEqQ,KAAM,MAAOrQ,MAAO,UAK9BK,KAAKmZ,YAAc,IAAIxS,YAAY,CAC/BxE,YAAa,iBACb0E,cAAe,UACfC,SAAU,IACVC,eAAgB,cAChBC,kBAAkB,EAClBC,SAAU,IACVL,SAAU,CACN,CAAEsB,IAAK,UAAWvI,MAAO,UAAWC,KAAM,iBAAkBuI,KAAM2X,GAClE,CAAE5X,IAAK,cAAevI,MAAO,cAAeC,KAAM,kBAAmBuI,KAAM4X,GAC3E,CAAEjY,KAAM,UAAWnI,MAAO,YAC1B,CAAEuI,IAAK,OAAQvI,MAAO,OAAQC,KAAM,kBAAmBuI,KAAMqT,EAAUzT,YAAa,gBAG5F/H,KAAKoC,SAASpC,KAAKmZ,aAGnB,MAAM6G,EAAa,IAAI7D,EAAY,CAC/Bha,YAAa,sBACbpC,UAAW,yCACXqc,QAASpc,KAAK0M,MACdhF,OAAQ,CACJ9H,KAAM,yBACNwJ,MAAO,CACH,CAAEzJ,MAAO,kBAAmBE,OAAQ,kBAAmBD,KAAM,aAC7D,CAAEkI,KAAM,WACR,CAAEnI,MAAO,YAAaE,OAAQ,YAAaD,KAAM,aACjD,CAAED,MAAO,aAAcE,OAAQ,aAAcD,KAAM,aACnD,CAAEkI,KAAM,WACR9H,KAAK0M,MAAMC,IAAI,aACT,CAAEhN,MAAO,oBAAqBE,OAAQ,oBAAqBD,KAAM,iBACjE,CAAED,MAAO,kBAAmBE,OAAQ,kBAAmBD,KAAM,gBACnE,CAAED,MAAO,oBAAqBE,OAAQ,gBAAiBD,KAAM,iBAAkBqgB,QAAQ,OAInGjgB,KAAKoC,SAAS4d,EAClB,CAIA,4BAAME,SACIC,GAAMC,UAAU,CAClB3gB,MAAO,kBACPiN,MAAO1M,KAAK0M,MACZ6P,WAAY8D,GAAYnN,MAEhC,CAEA,sBAAMoN,GACF,MAAMC,EAASvgB,KAAK0M,MAAMC,IAAI,SAASqF,GACvC,IAAKuO,EAAQ,OAAO,EACpB,MAAQrH,KAAAA,SAAesH,OAAO,+BAAsBC,KAAAC,GAAAA,EAAAC,GAEpD,aADMR,GAAMS,cAAc1H,EAAMqH,IACzB,CACX,CAEA,uBAAMM,GACF,MAAMC,EAAU9gB,KAAK0M,MAAMC,IAAI,UAAUqF,GACzC,IAAK8O,EAAS,OAAO,EAErB,MAAQC,MAAAA,SAAgBP,OAAO,+BAAuBC,KAAAC,GAAAA,EAAAhO,GAEtD,aADMyN,GAAMS,cAAcG,EAAOD,IAC1B,CACX,CAEA,0BAAMnE,GACF,OAAI3c,KAAK0M,MAAMC,IAAI,aACR3M,KAAKghB,2BAELhhB,KAAKihB,wBAEpB,CAEA,8BAAMD,GAKF,cAJwBb,GAAMpT,QAC1B,sBAAsB/M,KAAK0M,MAAMC,IAAI,yDAAyD3M,KAAK0M,MAAMC,IAAI,0BAC7G,wBAKgB,aADD3M,KAAK0M,MAAMwB,KAAK,CAAE4I,WAAW,KACvC7T,OACLjD,KAAK4C,UAAUwL,OAAOrL,QAAQ,sBAE9B/C,KAAK4C,UAAUwL,OAAO9K,MAAM,gCAEzB,EACX,CAEA,4BAAM2d,GAKF,cAJwBd,GAAMpT,QAC1B,oBAAoB/M,KAAK0M,MAAMC,IAAI,yDAAyD3M,KAAK0M,MAAMC,IAAI,0BAC3G,sBAKgB,aADD3M,KAAK0M,MAAMwB,KAAK,CAAE4I,WAAW,KACvC7T,OACLjD,KAAK4C,UAAUwL,OAAOrL,QAAQ,oBAE9B/C,KAAK4C,UAAUwL,OAAO9K,MAAM,8BAEzB,EACX,CAEA,0BAAM4d,GAKF,cAJwBf,GAAMpT,QAC1B,kBAAkB/M,KAAK0M,MAAMC,IAAI,8CAA8C3M,KAAK0M,MAAMC,IAAI,iDAC9F,2BAIe3M,KAAK0M,MAAM9C,WACrB7G,SACL/C,KAAK4C,UAAUwL,OAAOrL,QAAQ,kBAC9B/C,KAAKwF,KAAK,iBAAkB,CAAEkH,MAAO1M,KAAK0M,SAE1C1M,KAAK4C,UAAUwL,OAAO9K,MAAM,4BAEzB,EACX,CAEA,cAAA+I,GAEA,CAEA,aAAOC,CAAO/M,EAAU,IACpB,OAAO,IAAIqgB,WAAWrgB,EAC1B,EAGJsgB,GAAO3C,WAAa0C,WC5SpB,MAAMuB,wBAAwB/D,EAC1B,WAAA9d,CAAYC,EAAU,IAClBC,MAAM,IACCD,EACHyQ,KAAM,gBACNqN,SAAU,iBACVlX,OAAQ,gBACRmX,WAAYtD,GAEZoH,SAAUf,GAAYnN,KACtBmO,cAAezB,WAEfpC,kBAAmB,CACfpa,QAAQ,EACRoN,KAAM,MAIV4J,QAAS,CACL,CACIlS,IAAK,KACLvI,MAAO,KACPib,MAAO,OACPN,UAAU,EACV/G,MAAO,cAEX,CACIrL,IAAK,oBACLvI,MAAO,OACP0a,UAAW,2BAEf,CACInS,IAAK,aACLvI,MAAO,QACP0a,UAAW,uBAEf,CACInS,IAAK,aACLvI,MAAO,QACP0a,UAAW,4BAEf,CACInS,IAAK,OACLvI,MAAO,OACP0a,UAAW,SAEf,CACInS,IAAK,SACLvI,MAAO,SACP0a,UAAW,SAEf,CACInS,IAAK,UACLvI,MAAO,QACP0a,UAAW,mBAKnBuD,YAAY,EACZC,YAAY,EACZvD,UAAU,EACVwD,YAAY,EACZC,WAAW,EAGXC,aAAa,EACbC,SAAS,EACTC,YAAY,EAGZC,aAAc,+DAGdmD,iBAAkB,MAClBC,aAAc,CACV,CAAE5hB,MAAO,SAAUC,KAAM,oBAAqBC,OAAQ,gBACtD,CAAEF,MAAO,SAAUC,KAAM,iBAAkBC,OAAQ,gBACnD,CAAEF,MAAO,cAAeC,KAAM,oBAAqBC,OAAQ,cAC3D,CAAEF,MAAO,WAAYC,KAAM,qBAAsBC,OAAQ,kBACzD,CAAEF,MAAO,aAAcC,KAAM,iBAAkBC,OAAQ,qBAI3Dwe,aAAc,CACVC,SAAS,EACTC,UAAU,EACVC,OAAO,EACPC,YAAY,IAGxB,EChFJ,MAAM+C,kBAAkBniB,EACpB,WAAAC,CAAYC,EAAU,IAClBC,MAAM,CACFO,UAAW,gBACRR,IAGPS,KAAK0M,MAAQnN,EAAQmN,OAAS,IAAIqU,EAAMxhB,EAAQyD,MAAQ,IAExDhD,KAAKwM,SAAW,kTAQpB,CAEA,YAAM5L,GAEFZ,KAAKoD,OAAS,IAAI/D,EAAK,CACnB8C,YAAa,eACbqK,SAAU,siGAiDdxM,KAAKoD,OAAOgW,SAASpZ,KAAK0M,OAC1B1M,KAAKoC,SAASpC,KAAKoD,QAGnB,MAAM0c,EAAc,IAAIzgB,EAAK,CACzBqN,MAAO1M,KAAK0M,MACZF,SAAU,slKAqFRiV,EAAc,IAAIvH,GAAU,CAC9BhI,WAAY,IAAI8H,GAAW,CAAE5H,OAAQ,CAAEsP,MAAO1hB,KAAK0M,MAAMC,IAAI,MAAO6D,KAAM,MAC1E2J,oBAAqB,CAAC,SACtBY,YAAa,OACbkD,SAAS,EACT0D,eAAgB,SAChBC,MAAQpd,GAAUxE,KAAK6hB,cAAcrd,GACrC4V,QAAS,CACL,CAAElS,IAAK,oBAAqBvI,MAAO,OAAQ2a,UAAU,GACrD,CAAEpS,IAAK,aAAcvI,MAAO,QAAS2a,UAAU,GAC/C,CAAEpS,IAAK,yBAA0BvI,MAAO,eACxC,CAAEuI,IAAK,UAAWvI,MAAO,SAAU0a,UAAW,OAAQC,UAAU,MAKlEwH,EAAe,IAAI5H,GAAU,CAC/BhI,WAAY,IAAI6P,EAAU,CAAE3P,OAAQ,CAAEhK,OAAQpI,KAAK0M,MAAMC,IAAI,MAAO6D,KAAM,MAC1E2J,oBAAqB,CAAC,UACtBY,YAAa,OACbkD,SAAS,EACT0D,eAAgB,YAChBC,MAAO,IAAM5hB,KAAKgiB,wBAClB5H,QAAS,CACL,CAAElS,IAAK,OAAQvI,MAAO,OAAQ2a,UAAU,GACxC,CAAEpS,IAAK,OAAQvI,MAAO,OAAQ0a,UAAW,SACzC,CACInS,IAAK,YAAavI,MAAO,SAAUib,MAAO,OAC1CpO,SAAU,gTAId,CAAEtE,IAAK,UAAWvI,MAAO,UAAW0a,UAAW,OAAQC,UAAU,MAKnEK,EAAa,IAAIT,GAAU,CAC7BhI,WAAY,IAAIsI,EAAkB,CAC9BpI,OAAQ,CAAE5B,KAAM,GAAIiK,WAAY,gBAAiBC,SAAU1a,KAAK0M,MAAMC,IAAI,SAE9EwN,oBAAqB,CAAC,aAAc,YACpCC,QAAS,CACL,CAAElS,IAAK,UAAWvI,MAAO,OAAQ0a,UAAW,WAAYC,UAAU,EAAMM,MAAO,SAC/E,CAAE1S,IAAK,iBAAkBvI,MAAO,YAChC,CAAEuI,IAAK,QAASvI,MAAO,YAKzB6b,EAAW,IAAItB,GAAU,CAC3BhI,WAAY,IAAIqJ,GAAQ,CACpBnJ,OAAQ,CAAE5B,KAAM,GAAIiK,WAAY,gBAAiBC,SAAU1a,KAAK0M,MAAMC,IAAI,SAE9E5E,YAAa,YACboS,oBAAqB,CAAC,aAAc,YACpCC,QAAS,CACL,CACIlS,IAAK,UAAWvI,MAAO,YAAa2a,UAAU,EAAMD,UAAW,iBAC/DnV,OAAQ,CAAE8K,KAAM,UAAWlI,KAAM,YAAa2T,UAAW,WAAYC,QAAS,SAAUC,UAAW,WAAYhc,MAAO,aAAcic,OAAQ,aAAcC,cAAe,eAAgBC,UAAW,SAExM,CACI5T,IAAK,QAASvI,MAAO,QAAS2a,UAAU,EACxCpV,OAAQ,CAAE4C,KAAM,SAAUvI,QAAS,CAAC,CAAEoR,MAAO,OAAQhR,MAAO,QAAU,CAAEgR,MAAO,UAAWhR,MAAO,WAAa,CAAEgR,MAAO,QAAShR,MAAO,YAE3I,CAAEuI,IAAK,OAAQvI,MAAO,OAAQuF,OAAQ,CAAE4C,KAAM,SAC9C,CAAEkI,KAAM,MAAOrQ,MAAO,UAK9BK,KAAKmZ,YAAc,IAAIxS,YAAY,CAC/BxE,YAAa,gBACb0E,cAAe,UACfC,SAAU,IACVC,eAAgB,eAChBC,kBAAkB,EAClBC,SAAU,IACVL,SAAU,CACN,CAAEsB,IAAK,UAAWvI,MAAO,UAAWC,KAAM,iBAAkBuI,KAAM2X,GAClE,CAAE5X,IAAK,UAAWvI,MAAO,UAAWC,KAAM,YAAauI,KAAMsZ,GAC7D,CAAEvZ,IAAK,WAAYvI,MAAO,aAAcC,KAAM,eAAgBuI,KAAM2Z,GACpE,CAAEha,KAAM,UAAWnI,MAAO,YAC1B,CAAEuI,IAAK,SAAUvI,MAAO,SAAUC,KAAM,oBAAqBuI,KAAMwS,GACnE,CAAEzS,IAAK,OAAQvI,MAAO,OAAQC,KAAM,kBAAmBuI,KAAMqT,EAAUzT,YAAa,gBAG5F/H,KAAKoC,SAASpC,KAAKmZ,aAGnB,MAAM8I,EAAY,IAAI9F,EAAY,CAC9Bha,YAAa,qBACbpC,UAAW,yCACXqc,QAASpc,KAAK0M,MACdhF,OAAQ,CACJ9H,KAAM,yBACNwJ,MAAO,CACH,CAAEzJ,MAAO,aAAcE,OAAQ,aAAcD,KAAM,aACnD,CAAEkI,KAAM,WACR,CAAEnI,MAAO,gBAAiBE,OAAQ,gBAAiBD,KAAM,kBACzD,CAAED,MAAO,gBAAiBE,OAAQ,kBAAmBD,KAAM,gBAC3D,CAAEkI,KAAM,WACR9H,KAAK0M,MAAMC,IAAI,aACT,CAAEhN,MAAO,mBAAoBE,OAAQ,mBAAoBD,KAAM,iBAC/D,CAAED,MAAO,iBAAkBE,OAAQ,iBAAkBD,KAAM,oBAI7EI,KAAKoC,SAAS6f,EAClB,CAIA,uBAAMC,SACiBpV,EAAOkG,cAAc,CACpCvT,MAAO,gBAAgBO,KAAK0M,MAAMC,IAAI,UACtCD,MAAO1M,KAAK0M,MACZ8D,KAAM,KACN+L,WAAY4F,EAAWC,kBAGjBpiB,KAAKqD,QAEnB,CAEA,0BAAMgf,GACF,OAAOriB,KAAK6hB,cAAc,IAAIS,MAAM,SACxC,CAEA,mBAAMT,CAAcrd,GACZA,GAAOyG,iBACPzG,EAAMyG,iBACNzG,EAAM+d,mBAEV,MAAMvf,QAAa8J,EAAOyD,SAAS,CAC/B9Q,MAAO,kBAAkBO,KAAK0M,MAAMC,IAAI,UACxC6D,KAAM,KACNhC,OAAQ,CACJ,CAAE1G,KAAM,QAASkI,KAAM,QAASrQ,MAAO,QAAS6R,UAAU,EAAMf,KAAM,OAG9E,IAAKzN,GAAMuK,MAAO,OAAO,EAEzB,MAAM4D,EAAMnR,KAAK4C,SACXqL,QAAakD,EAAItO,KAAKuO,KAAK,2BAA4B,CACzDsQ,MAAO1hB,KAAK0M,MAAMsF,GAClBzE,MAAOvK,EAAKuK,QAWhB,OATIU,EAAKlL,SACLoO,EAAI/C,MAAMrL,QAAQ,6BAE2B,YAAzC/C,KAAKmZ,aAAarN,0BACZ9L,KAAKmZ,YAAYtP,YAAY,YAGvCsH,EAAI/C,MAAM9K,MAAM2K,EAAKI,SAAW,0BAE7B,CACX,CAEA,2BAAM2T,GACF,MAAMhf,QAAa8J,EAAOyD,SAAS,CAC/B9Q,MAAO,oBAAoBO,KAAK0M,MAAMC,IAAI,UAC1C6D,KAAM,KACNhC,OAAQ2T,EAAW7V,OAAOkC,OAAOtJ,OAAOsd,GAAgB,WAAXA,EAAExS,QAEnD,IAAKhN,EAAM,OAAO,EAElBA,EAAKoF,OAASpI,KAAK0M,MAAMsF,GACzB,MAAMyQ,EAAW,IAAI1B,EAAM/d,GACrBiL,QAAawU,EAASvU,OAS5B,OARoB,MAAhBD,EAAKhL,QAAkC,MAAhBgL,EAAKhL,QAC5BjD,KAAK4C,UAAUwL,OAAOrL,QAAQ,qBACe,aAAzC/C,KAAKmZ,aAAarN,0BACZ9L,KAAKmZ,YAAYtP,YAAY,aAGvC7J,KAAK4C,UAAUwL,OAAO9K,MAAM2K,EAAKI,SAAW,+BAEzC,CACX,CAEA,6BAAMqU,GAKF,cAJwB5V,EAAOC,QAC3B,+CAA+C/M,KAAK0M,MAAMC,IAAI,oBAC9D,uBAKgB,aADD3M,KAAK0M,MAAMwB,KAAK,CAAE4I,WAAW,KACvC7T,QACLjD,KAAK4C,UAAUwL,OAAOrL,QAAQ,2BACxB/C,KAAKqD,UAEXrD,KAAK4C,UAAUwL,OAAO9K,MAAM,+BAEzB,EACX,CAEA,2BAAMqf,GAKF,cAJwB7V,EAAOC,QAC3B,6CAA6C/M,KAAK0M,MAAMC,IAAI,oBAC5D,qBAKgB,aADD3M,KAAK0M,MAAMwB,KAAK,CAAE4I,WAAW,KACvC7T,QACLjD,KAAK4C,UAAUwL,OAAOrL,QAAQ,yBACxB/C,KAAKqD,UAEXrD,KAAK4C,UAAUwL,OAAO9K,MAAM,6BAEzB,EACX,CAEA,wBAAMsf,CAAmBpe,EAAOC,GAC5B,MAAMoe,EAAWpe,GAASkG,SAASqH,GACnC,IAAK6Q,EAAU,OAAO,EAEtB,MAAMza,EAAS,IAAI2Y,EAAM,CAAE/O,GAAI6Q,IAU/B,aATMza,EAAOkK,QACTlK,EAAO4J,IACPlF,EAAOsG,WAAW,CACd3T,OAAO,EACP+Q,KAAM,KACN6C,KAAM,IAAImO,UAAU,CAAE9U,MAAOtE,IAC7BkL,QAAS,CAAC,CAAE1C,KAAM,QAAS2C,MAAO,gBAAiBC,SAAS,OAG7D,CACX,CAIA,iBAAM3J,CAAYiT,GACV9c,KAAKmZ,mBACCnZ,KAAKmZ,YAAYtP,YAAYiT,EAE3C,CAEA,gBAAAhR,GACI,OAAO9L,KAAKmZ,YAAcnZ,KAAKmZ,YAAYrN,mBAAqB,IACpE,CAEA,cAAAO,GAEA,CAEA,aAAOC,CAAO/M,EAAU,IACpB,OAAO,IAAIiiB,UAAUjiB,EACzB,EAGJwhB,EAAM7D,WAAasE,UC5anB,MAAMsB,uBAAuB1F,EACzB,WAAA9d,CAAYC,EAAU,IAClBC,MAAM,IACCD,EACHyQ,KAAM,eACNqN,SAAU,gBACVlX,OAAQ,eACRmX,WAAYyE,EAEZgB,WAAYZ,EAAW7V,OACvB8U,SAAUe,EAAWjP,KACrBmO,cAAeG,UAEfhE,kBAAmB,CACfpa,QAAQ,GAGZqa,aAAc,CACVlI,KAAM,MACNuB,UAAW,GAIfsD,QAAS,CACL,CACIlS,IAAK,KACLvI,MAAO,KACP2a,UAAU,EACV/G,MAAO,cAGX,CACIrL,IAAK,OACLvI,MAAO,gBAEX,CACIuI,IAAK,aACLvI,MAAO,OACPuF,OAAQ,CACJ4C,KAAM,SACNvI,QAASwhB,EAAMiC,mBAGvB,CACI9a,IAAK,sBACLvI,MAAO,UACP+d,WAAY,MAEhB,CACIxV,IAAK,cACLvI,MAAO,SACP0a,UAAW,eACXqD,WAAY,KACZnK,MAAO,mBAEX,CACIrL,IAAK,UACLvI,MAAO,UACPI,UAAW,kBACXsa,UAAW,iBACXqD,WAAY,MAEhB,CACIxV,IAAK,gBACLvI,MAAO,WACPI,UAAW,kBACXsa,UAAW,WACXqD,WAAY,OAIpBC,QAAS,CACL,CACIzV,IAAK,YACLvI,MAAO,SACPmI,KAAM,SACNvI,QAAS,CACL,CAAEI,MAAO,SAAUgR,OAAO,GAC1B,CAAEhR,MAAO,WAAYgR,OAAO,MAKxCyN,YAAa,CACT,CACIxe,KAAM,YACNC,OAAQ,OACRF,MAAO,cAEX,CACIC,KAAM,cACNC,OAAQ,cACRF,MAAO,sBAKfie,YAAY,EACZC,YAAY,EACZvD,UAAU,EACVwD,YAAY,EACZC,WAAW,EAGXC,aAAa,EACbC,SAAS,EACTC,YAAY,EAGZC,aAAc,+DAGdmD,iBAAkB,MAClBC,aAAc,CACV,CAAE5hB,MAAO,SAAUC,KAAM,cAAeC,OAAQ,gBAChD,CAAEF,MAAO,SAAUC,KAAM,iBAAkBC,OAAQ,gBACnD,CAAEF,MAAO,WAAYC,KAAM,qBAAsBC,OAAQ,kBACzD,CAAEF,MAAO,aAAcC,KAAM,iBAAkBC,OAAQ,oBACvD,CAAEF,MAAO,OAAQC,KAAM,oBAAqBC,OAAQ,eAIxDwe,aAAc,CACVC,SAAS,EACTC,UAAU,EACVC,OAAO,EACPC,YAAY,IAGxB,CAEA,kBAAAwE,CAAmBze,EAAOC,GACtB,MAAMka,EAAO3e,KAAKkS,WAAWvF,IAAIlI,EAAQkG,QAAQqH,IACjDhS,KAAK4C,SAASsgB,eAAevE,EACjC,EC5HJ,MAAMwE,0BAA0BjM,GAC5B,gBAAIoB,GACA,MAAMC,EAAMvY,KAAK0M,OAAOC,IAAI,gBAAkB,CAAA,EAC9C,MAAO,CAAC4L,EAAIhJ,KAAMgJ,EAAIC,QAAQtT,OAAOC,SAAS4D,KAAK,OAASwP,EAAIE,cAAgB,GACpF,CACA,eAAIC,GACA,OAAO1Y,KAAK0M,OAAOC,IAAI,gBAAgB8L,cAAgB,EAC3D,CACA,WAAI2K,GACA,MAAM7K,EAAMvY,KAAK0M,OAAOC,IAAI,gBAAkB,CAAA,EAC9C,OAAO4L,EAAI8K,KAAO9K,EAAI+K,SAAW,EACrC,CACA,eAAI3K,GACA,MAAMJ,EAAMvY,KAAK0M,OAAOC,IAAI,gBAAkB,CAAA,EACxCiM,EAAQ,GAId,OAHIL,EAAIM,QAAQD,EAAM3Q,KAAK,iFACvBsQ,EAAIO,QAAQF,EAAM3Q,KAAK,sEACvBsQ,EAAIQ,UAAUH,EAAM3Q,KAAK,mFACtB2Q,EAAM7P,KAAK,IACtB,CACA,kBAAIiQ,GACA,MAAMT,EAAMvY,KAAK0M,OAAOC,IAAI,gBAAkB,CAAA,EAC9C,SAAU4L,EAAIM,QAAUN,EAAIO,QAAUP,EAAIQ,SAC9C,EAGJ,MAAMwK,mBAAmBlkB,EACrB,WAAAC,CAAYC,EAAU,IAClBC,MAAM,CACFO,UAAW,iBACRR,IAGPS,KAAK0M,MAAQnN,EAAQmN,OAAS,IAAI8W,EAAWjkB,EAAQyD,MAAQ,IAC7DhD,KAAKyjB,WAAazjB,KAAK0M,MAAMC,IAAI,gBAAkB,CAAA,EACnD3M,KAAKmX,WAAanX,KAAK0jB,SAAS1jB,KAAKyjB,YAGrCzjB,KAAK2jB,YAAc3jB,KAAK4jB,cACxB5jB,KAAK6jB,OAAS7jB,KAAK8jB,SACnB9jB,KAAK+jB,WAAa/jB,KAAKgkB,aACvBhkB,KAAKikB,SAAWjkB,KAAKkkB,YAErBlkB,KAAKwM,SAAW,u9EAyCpB,CAIA,WAAAoX,GACI,MAAM7L,EAAK/X,KAAKyjB,YAAYzL,YAAc,CAAA,EACpCmM,EAAQ,CAACpM,EAAGN,OAAQM,EAAGE,OAAO/S,OAAOC,SAC3C,OAAOgf,EAAM9O,OAAS8O,EAAMpb,KAAK,KAAO,iBAC5C,CAEA,MAAA+a,GACI,MAAMxM,EAAKtX,KAAKyjB,YAAYnM,IAAM,CAAA,EAC5B8M,EAAM,CAAC9M,EAAGW,MAAOX,EAAG+M,OAAOnf,OAAOC,SAAS4D,KAAK,KACtD,OAAOuO,EAAGG,OAAS,GAAGH,EAAGG,UAAU2M,IAAM1W,OAAS,YACtD,CAEA,UAAAsW,GACI,MAAM5M,EAAMpX,KAAKyjB,YAAYpM,QAAU,CAAA,EACjC8M,EAAQ,CAAC/M,EAAIQ,MAAOR,EAAIK,QAAQvS,OAAOC,SACvC6K,EAAOmU,EAAM9O,OAAS8O,EAAMpb,KAAK,KAAO,iBAC9C,OAAOqO,EAAI1K,MAAQ,GAAGsD,MAASoH,EAAI1K,SAAWsD,CAClD,CAEA,SAAAkU,GACI,MAAM9M,EAAMpX,KAAKyjB,YAAYpM,QAAU,CAAA,EACjCC,EAAKtX,KAAKyjB,YAAYnM,IAAM,CAAA,EAClC,MAAO,CAAC,SAAU,UAAW,QAAQC,KAAKC,IACrCJ,EAAIK,QAAU,IAAIC,SAASF,KAAOF,EAAGG,QAAU,IAAIC,SAASF,GAErE,CAEA,QAAAkM,CAASD,GACL,MAAMnM,EAAKmM,GAAYnM,IAAIG,QAAQzG,eAAiB,GAC9CsT,EAAUb,GAAYzL,YAAYP,QAAQzG,eAAiB,GAC3DqG,EAASoM,GAAYpM,QAAQI,QAAQzG,eAAiB,GAE5D,OAAIsT,EAAQ5M,SAAS,UAAkB,oBACnC4M,EAAQ5M,SAAS,WAAmB,qBACpC4M,EAAQ5M,SAAS,UAAkB,oBACnC4M,EAAQ5M,SAAS,QAAgB,kBACjCJ,EAAGI,SAAS,QAAUJ,EAAGI,SAAS,OAAe,WACjDJ,EAAGI,SAAS,WAAmB,aAC/BJ,EAAGI,SAAS,WAAmB,cAC/BJ,EAAGI,SAAS,SAAiB,YAC7BL,EAAOK,SAAS,UAAkB,WAClCL,EAAOK,SAAS,QAAgB,YAC7B,WACX,CAEA,YAAM9W,GAEF,MAAMkf,EAAc,IAAIzgB,EAAK,CACzBqN,MAAO1M,KAAK0M,MACZF,SAAU,i7IAuERyO,EAAgB,IAAIf,GAAU,CAChChI,WAAY,IAAIgJ,EAAuB,CACnC9I,OAAQ,CAAEmS,YAAavkB,KAAK0M,MAAMC,IAAI,MAAO6D,KAAM,MAEvD2J,oBAAqB,CAAC,eACtBY,YAAa,OACbC,UAAWmI,kBACXvF,YAAY,EACZxD,QAAS,CACL,CACIlS,IAAK,aACLvI,MAAO,WACP6M,SAAU,sxBAWd,CAAEtE,IAAK,aAAcvI,MAAO,aAAc0a,UAAW,iBAAkBO,MAAO,SAC9E,CAAE1S,IAAK,YAAavI,MAAO,YAAa0a,UAAW,iBAAkBO,MAAO,YAKpF5a,KAAKmZ,YAAc,IAAIxS,YAAY,CAC/BxE,YAAa,iBACb0E,cAAe,UACfC,SAAU,IACVC,eAAgB,cAChBC,kBAAkB,EAClBC,SAAU,IACVL,SAAU,CACN,CAAEsB,IAAK,UAAWvI,MAAO,UAAWC,KAAM,iBAAkBuI,KAAM2X,GAClE,CAAE5X,IAAK,YAAavI,MAAO,YAAaC,KAAM,aAAcuI,KAAM8S,MAG1Ejb,KAAKoC,SAASpC,KAAKmZ,aAGnB,MAAMqL,EAAa,IAAIrI,EAAY,CAC/Bha,YAAa,sBACbpC,UAAW,yCACXqc,QAASpc,KAAK0M,MACdhF,OAAQ,CACJ9H,KAAM,yBACNwJ,MAAO,CACH,CAAEzJ,MAAO,YAAaE,OAAQ,YAAaD,KAAM,aACjD,CAAED,MAAO,eAAgBE,OAAQ,eAAgBD,KAAM,kBAAmBmF,UAAU,GACpF,CAAE+C,KAAM,WACR,CAAEnI,MAAO,gBAAiBE,OAAQ,gBAAiBD,KAAM,WAAYqgB,QAAQ,OAIzFjgB,KAAKoC,SAASoiB,EAClB,CAEA,sBAAMlE,GACFtgB,KAAKwF,KAAK,YAAa,CAAE+a,OAAQvgB,KAAK0M,MAAMC,IAAI,SAASqF,IAC7D,CAEA,0BAAMyS,GAKF,cAJwB3X,EAAOC,QAC3B,sDACA,2BAIe/M,KAAK0M,MAAM9C,WACrB7G,SACL/C,KAAKwF,KAAK,iBAAkB,CAAEkH,MAAO1M,KAAK0M,SAEvC,EACX,CAEA,iBAAagY,CAAKC,GACd,MAAMjY,QAAc8W,EAAWoB,UAAUD,GACzC,OAAIjY,EACOI,EAAOsG,WAAW,CACrB3T,OAAO,EACP+Q,KAAM,KACN6C,KAAM,IAAIkQ,WAAW,CAAE7W,UACvB4G,QAAS,CAAC,CAAE1C,KAAM,QAAS2C,MAAO,gBAAiBC,SAAS,OAGpE1G,EAAOnH,MAAM,CAAE0I,QAAS,oCAAoCsW,IAAQ7c,KAAM,YACnE,KACX,EAGJ0b,EAAWtG,WAAaqG,WCxTxB,MAAMsB,4BAA4BzH,EAC9B,WAAA9d,CAAYC,EAAU,IAClBC,MAAM,IACCD,EACHyQ,KAAM,qBACNqN,SAAU,eACVlX,OAAQ,qBACRmX,WAAYxC,EAGZ0C,kBAAmB,CACfpa,QAAQ,EACRoN,KAAM,MAIV4J,QAAS,CACL,CAAElS,IAAK,KAAMvI,MAAO,KAAMib,MAAO,OAAQN,UAAU,EAAM/G,MAAO,cAChE,CAAErL,IAAK,OAAQvI,MAAO,YAAa2a,UAAU,EAAMD,UAAW,uBAC9D,CAAEnS,IAAK,oBAAqBvI,MAAO,OAAQ2a,UAAU,EAAMD,UAAW,gBACtE,CAAEnS,IAAK,gCAAiCvI,MAAO,UAAW0a,UAAW,gBACrE,CAAEnS,IAAK,wBAAyBvI,MAAO,KAAM0a,UAAW,gBACxD,CAAEnS,IAAK,UAAWvI,MAAO,UAAW2a,UAAU,GAC9C,CAAEpS,IAAK,aAAcvI,MAAO,aAAc0a,UAAW,kBACrD,CAAEnS,IAAK,YAAavI,MAAO,YAAa0a,UAAW,mBAIvDuD,YAAY,EACZC,YAAY,EACZvD,UAAU,EACVwD,YAAY,EACZC,WAAW,EAGXC,aAAa,EACbC,SAAS,EACTC,YAAY,EAGZC,aAAc,yBAGdE,aAAc,CACVC,SAAS,EACTC,UAAU,EACVC,OAAO,EACPC,YAAY,IAGxB,ECnDJ,MAAMqG,oCAAoC1H,EACtC,WAAA9d,CAAYC,EAAU,IAClBC,MAAM,IACCD,EACHyQ,KAAM,8BACNqN,SAAU,mBACVlX,OAAQ,8BACRmX,WAAYpC,EAGZd,QAAS,CACL,CAAElS,IAAK,KAAMvI,MAAO,KAAMib,MAAO,OAAQN,UAAU,EAAM/G,MAAO,cAChE,CAAErL,IAAK,oBAAqBvI,MAAO,OAAQ2a,UAAU,GACrD,CAAEpS,IAAK,cAAevI,MAAO,SAAU6M,SAAU,yFAA0F8N,UAAU,GACrJ,CAAEpS,IAAK,aAAcvI,MAAO,aAAc2a,UAAU,GACpD,CAAEpS,IAAK,mBAAoBvI,MAAO,OAAQ0a,UAAW,gBACrD,CAAEnS,IAAK,qBAAsBvI,MAAO,SAAU0a,UAAW,gBACzD,CAAEnS,IAAK,2BAA4BvI,MAAO,UAAW0a,UAAW,gBAChE,CAAEnS,IAAK,YAAavI,MAAO,YAAa0a,UAAW,mBAIvDuD,YAAY,EACZC,YAAY,EACZvD,UAAU,EACVwD,YAAY,EACZC,WAAW,EAGXC,aAAa,EACbC,SAAS,EACTC,YAAY,EAGZC,aAAc,6BAGdE,aAAc,CACVC,SAAS,EACTC,UAAU,EACVC,OAAO,EACPC,YAAY,IAGxB,ECrCJ,MAAMsG,kBAAkB1lB,EACpB,WAAAC,CAAYC,EAAU,IAClBC,MAAM,CACFO,UAAW,gBACRR,IAGPS,KAAK0M,MAAQnN,EAAQmN,OAAS,IAAIsY,EAAazlB,EAAQyD,MAAQ,IAC/DhD,KAAKilB,eAAiBjlB,KAAK0M,MAAMC,IAAI,aAAe3M,KAAK0M,MAAMC,IAAI,aAEnE3M,KAAKwM,SAAW,ipIAwDpB,CAEA,YAAM5L,GAEFZ,KAAK8f,YAAc,IAAIoF,GAAS,CAC5BxY,MAAO1M,KAAK0M,MACZ3M,UAAW,MACXolB,iBAAiB,EACjBC,eAAgB,IAChBhL,QAAS,EACT5L,OAAQ,CACJ,CAAEwB,KAAM,aAAcrQ,MAAO,aAAc8Q,KAAM,GACjD,CAAET,KAAM,SAAUrQ,MAAO,SAAU8Q,KAAM,GACzC,CAAET,KAAM,eAAgBrQ,MAAO,UAAW8Q,KAAM,GAChD,CAAET,KAAM,eAAgBrQ,MAAO,eAAgB8Q,KAAM,GACrD,CAAET,KAAM,SAAUrQ,MAAO,SAAU8Q,KAAM,GACzC,CAAET,KAAM,OAAQrQ,MAAO,OAAQ8Q,KAAM,GACrC,CAAET,KAAM,cAAerQ,MAAO,cAAe8Q,KAAM,GACnD,CAAET,KAAM,WAAYrQ,MAAO,WAAY8Q,KAAM,GAC7C,CAAET,KAAM,WAAYrQ,MAAO,WAAY8Q,KAAM,GAC7C,CAAET,KAAM,YAAarQ,MAAO,YAAa8Q,KAAM,MAK/CzQ,KAAKqlB,YAAc,IAAIH,GAAS,CAC5BxY,MAAO1M,KAAK0M,MACZ3M,UAAW,MACXolB,iBAAiB,EACjBC,eAAgB,IAChBhL,QAAS,EACT5L,OAAQ,CACJ,CAAEwB,KAAM,SAAUrQ,MAAO,gBAAiB0a,UAAW,YAAa5J,KAAM,GACxE,CAAET,KAAM,SAAUrQ,MAAO,MAAO0a,UAAW,YAAa5J,KAAM,GAC9D,CAAET,KAAM,WAAYrQ,MAAO,QAAS0a,UAAW,YAAa5J,KAAM,GAClE,CAAET,KAAM,WAAYrQ,MAAO,iBAAkB0a,UAAW,YAAa5J,KAAM,GAC3E,CAAET,KAAM,gBAAiBrQ,MAAO,aAAc0a,UAAW,YAAa5J,KAAM,GAC5E,CAAET,KAAM,YAAarQ,MAAO,SAAU0a,UAAW,YAAa5J,KAAM,GACpE,CAAET,KAAM,iBAAkBrQ,MAAO,iBAAkB8Q,KAAM,GACzD,CAAET,KAAM,MAAOrQ,MAAO,MAAO8Q,KAAM,GACnC,CAAET,KAAM,UAAWrQ,MAAO,mBAAoB8Q,KAAM,GACpD,CAAET,KAAM,MAAOrQ,MAAO,MAAO8Q,KAAM,IACnC,CAAET,KAAM,kBAAmBrQ,MAAO,kBAAmB8Q,KAAM,MAKnEzQ,KAAKslB,SAAW,IAAIJ,GAAS,CACzBxY,MAAO1M,KAAK0M,MACZ3M,UAAW,MACXolB,iBAAiB,EACjBC,eAAgB,IAChBhL,QAAS,EACT5L,OAAQ,CACJ,CAAEwB,KAAM,eAAgBrQ,MAAO,eAAgB8Q,KAAM,GACrD,CAAET,KAAM,aAAcrQ,MAAO,aAAc8Q,KAAM,GACjD,CAAET,KAAM,YAAarQ,MAAO,SAAU0a,UAAW,YAAa5J,KAAM,GACpE,CAAET,KAAM,gBAAiBrQ,MAAO,aAAc0a,UAAW,YAAa5J,KAAM,GAC5E,CAAET,KAAM,oBAAqBrQ,MAAO,iBAAkB0a,UAAW,YAAa5J,KAAM,GACpF,CAAET,KAAM,kBAAmBrQ,MAAO,eAAgB0a,UAAW,YAAa5J,KAAM,MAKhGzQ,KAAKulB,aAAe,IAAIL,GAAS,CAC7BxY,MAAO1M,KAAK0M,MACZ3M,UAAW,MACXolB,iBAAiB,EACjBC,eAAgB,IAChBhL,QAAS,EACT5L,OAAQ,CACJ,CAAEwB,KAAM,KAAMrQ,MAAO,YAAa8Q,KAAM,GACxC,CAAET,KAAM,WAAYrQ,MAAO,gBAAiB0a,UAAW,aAAc5J,KAAM,GAC3E,CAAET,KAAM,UAAWrQ,MAAO,UAAW0a,UAAW,WAAY5J,KAAM,GAClE,CAAET,KAAM,WAAYrQ,MAAO,gBAAiB0a,UAAW,WAAY5J,KAAM,GACzE,CAAET,KAAM,YAAarQ,MAAO,YAAa0a,UAAW,WAAY5J,KAAM,GACtE,CAAET,KAAM,aAAcrQ,MAAO,UAAW0a,UAAW,WAAY5J,KAAM,MAK7E,MAAM8J,EAAmB,IAAIC,EAAkB,CAC3CpI,OAAQ,CACJ5B,KAAM,EACNgV,UAAWxlB,KAAK0M,MAAMC,IAAI,iBAGlC3M,KAAK2a,WAAa,IAAIT,GAAU,CAC5BhI,WAAYqI,EACZJ,oBAAqB,CAAC,aACtBC,QAAS,CACL,CAAElS,IAAK,KAAMvI,MAAO,KAAM2a,UAAU,EAAMM,MAAO,QACjD,CAAE1S,IAAK,UAAWvI,MAAO,OAAQ0a,UAAW,WAAYC,UAAU,EAAMM,MAAO,SAC/E,CAAE1S,IAAK,iBAAkBvI,MAAO,YAChC,CAAEuI,IAAK,QAASvI,MAAO,YAK/B,MAAM2b,EAAiB,IAAIC,GAAQ,CAC/BnJ,OAAQ,CACJ5B,KAAM,EACNiV,GAAIzlB,KAAK0M,MAAMC,IAAI,iBAG3B3M,KAAKwb,SAAW,IAAItB,GAAU,CAC1BhI,WAAYoJ,EACZvT,YAAa,YACboS,oBAAqB,CAAC,MACtBC,QAAS,CACL,CACIlS,IAAK,UACLvI,MAAO,YACP2a,UAAU,EACVD,UAAW,iBACXnV,OAAQ,CACJ8K,KAAM,UACNlI,KAAM,YACN2T,UAAW,WACXC,QAAS,SACTC,UAAW,WACXhc,MAAO,aACPic,OAAQ,aACRC,cAAe,eACfC,UAAW,SAGnB,CACI5T,IAAK,QACLvI,MAAO,QACP2a,UAAU,EACVpV,OAAQ,CACJ4C,KAAM,SACNvI,QAAS,CACL,CAAEoR,MAAO,OAAQhR,MAAO,QACxB,CAAEgR,MAAO,UAAWhR,MAAO,WAC3B,CAAEgR,MAAO,QAAShR,MAAO,YAIrC,CAAEuI,IAAK,OAAQvI,MAAO,OAAQuF,OAAQ,CAAE4C,KAAM,SAC9C,CAAEkI,KAAM,MAAOrQ,MAAO,UAI9B,MAAM+lB,EAAO,CACTC,SAAY3lB,KAAK8f,YACjB8F,QAAW5lB,KAAKqlB,YAChB,oBAAqBrlB,KAAKslB,SAC1BO,OAAU7lB,KAAK2a,WACfmL,KAAQ9lB,KAAKwb,SACbuK,SAAY/lB,KAAKulB,cAIrB,GAAIvlB,KAAKilB,eAAgB,CACrB,MAAMe,EAAMhmB,KAAK0M,MAAMC,IAAI,YACrBsZ,EAAMjmB,KAAK0M,MAAMC,IAAI,aAKrBuZ,EAAc,CAJPlmB,KAAK0M,MAAMC,IAAI,SAAW,UACxB3M,KAAK0M,MAAMC,IAAI,WAAa,GAC3B3M,KAAK0M,MAAMC,IAAI,iBAAmB,IAENzH,OAAOC,SAAS4D,KAAK,MAEjE/I,KAAKmmB,QAAU,IAAIC,GAAQ,CACvBC,QAAS,CAAC,CACNL,MACAC,MACAK,MAAO,WAAWtmB,KAAK0M,MAAMC,IAAI,6BAA6BuZ,MAElEK,UAAW,QACXC,KAAM,EACN9kB,OAAQ,MAEZgkB,EAAU,IAAI1lB,KAAKmmB,OACvB,CAEAnmB,KAAKymB,QAAU,IAAIC,EAAQ,CACvBvkB,YAAa,aACbujB,OACAiB,UAAW3mB,KAAKilB,eAAiB,MAAQ,aAE7CjlB,KAAKoC,SAASpC,KAAKymB,SAGnB,MAAMG,EAAY,CACd,CAAEjnB,MAAO,gBAAiBE,OAAQ,gBAAiBD,KAAM,cACzD,CAAED,MAAO,gBAAiBE,OAAQ,gBAAiBD,KAAM,kBACzD,CAAED,MAAO,eAAgBE,OAAQ,eAAgBD,KAAM,gBACvD,CAAEkI,KAAM,WACR,CAAEnI,MAAO,sBAAuBE,OAAQ,gBAAiBD,KAAM,uBAG/DI,KAAKilB,gBACL2B,EAAU3e,KAAK,CACXtI,MAAO,cACPE,OAAQ,cACRD,KAAM,WAIdgnB,EAAU3e,KACN,CAAEH,KAAM,WACR,CAAEnI,MAAO,gBAAiBE,OAAQ,eAAgBD,KAAM,WAAYqgB,QAAQ,IAGhF,MAAM4G,EAAY,IAAI1K,EAAY,CAC9Bha,YAAa,qBACbpC,UAAW,yCACXqc,QAASpc,KAAK0M,MACdhF,OAAQ,CACJ9H,KAAM,yBACNwJ,MAAOwd,KAGf5mB,KAAKoC,SAASykB,EAClB,CAEA,mBAAMngB,SACIlH,MAAMkH,gBAGR+C,OAAOqd,WAAard,OAAOqd,UAAUC,SAAW/mB,KAAKyE,SAC1BzE,KAAKyE,QAAQ8F,iBAAiB,8BACtCC,QAAQQ,IAEvB,MAAMgc,EAAWvd,OAAOqd,UAAUC,QAAQE,YAAYjc,GAClDgc,GAAwC,mBAArBA,EAASE,SAC5BF,EAASE,UAEb,IAAIzd,OAAOqd,UAAUC,QAAQ/b,IAGzC,CAEA,0BAAMmc,SACiBra,EAAOkG,cAAc,CACpCvT,MAAO,mBAAmBO,KAAK0M,MAAMC,IAAI,gBACzCD,MAAO1M,KAAK0M,MACZ6P,WAAYyI,EAAaoC,6BAInBpnB,KAAKqD,SACXrD,KAAK4C,UAAUwL,OAAOrL,QAAQ,iCAEtC,CAEA,0BAAMskB,SACiBva,EAAOkG,cAAc,CACpCvT,MAAO,mBAAmBO,KAAK0M,MAAMC,IAAI,gBACzCD,MAAO1M,KAAK0M,MACZ6P,WAAYyI,EAAasC,6BAInBtnB,KAAKqD,SACXrD,KAAK4C,UAAUwL,OAAOrL,QAAQ,0CAEtC,CAEA,yBAAMwkB,SACiBza,EAAOkG,cAAc,CACpCvT,MAAO,kBAAkBO,KAAK0M,MAAMC,IAAI,gBACxCD,MAAO1M,KAAK0M,MACZ6P,WAAYyI,EAAawC,4BAInBxnB,KAAKqD,SACXrD,KAAK4C,UAAUwL,OAAOrL,QAAQ,4CAEtC,CAEA,0BAAM0kB,SAEIznB,KAAK0M,MAAMwB,KAAK,CAAEjJ,SAAS,IACjCjF,KAAK4C,UAAUwL,OAAOsZ,KAAK,4BAA8B1nB,KAAK0M,MAAMC,IAAI,cAC5E,CAEA,4BAAMgb,SAEI3nB,KAAK0M,MAAMwB,KAAK,CAAE0Z,iBAAiB,IACzC5nB,KAAK4C,UAAUwL,OAAOsZ,KAAK,kCAAoC1nB,KAAK0M,MAAMC,IAAI,cAClF,CAEA,uBAAMkb,GACF,GAAI7nB,KAAKilB,eAAgB,CACrB,MAEM6C,EAAM,mDAFA9nB,KAAK0M,MAAMC,IAAI,eACf3M,KAAK0M,MAAMC,IAAI,eAE3BlD,OAAOse,KAAKD,EAAK,SACrB,CACJ,CAEA,yBAAME,SACsBlb,EAAOC,QAC3B,yDAAyD/M,KAAK0M,MAAMC,IAAI,kBACxE,mBACA,CAAEsb,aAAc,aAAcC,YAAa,mBAGxBloB,KAAK0M,MAAM9C,WACrB7G,SACL/C,KAAKwF,KAAK,gBAAiB,CAAEkH,MAAO1M,KAAK0M,OAGrD,CAEA,iBAAagY,CAAKe,GACd,MAAM/Y,QAAcsY,EAAamD,OAAO1C,GACxC,GAAI/Y,EAAO,CACP,MAAMvE,EAAO,IAAI4c,UAAU,CAAErY,UACvB0b,EAAS,IAAItb,EAAO,CACtB1J,QAAQ,EACRoN,KAAM,KACN6C,KAAMlL,EACNmL,QAAS,CAAC,CAAE1C,KAAM,QAAS2C,MAAO,gBAAiBC,SAAS,MAIhE,aAFM4U,EAAO/kB,QAAO,EAAMglB,SAAShV,MACnC+U,EAAO1D,OACA0D,CACX,CAEA,OADAtb,EAAOnH,MAAM,CAAE0I,QAAS,2CAA2CoX,IAAM3d,KAAM,YACxE,IACX,EAGJkd,EAAa9H,WAAa6H,UChZ1B,MAAMuD,8BAA8BlL,EAChC,WAAA9d,CAAYC,EAAU,IAClBC,MAAM,IACCD,EACHyQ,KAAM,qBACNqN,SAAU,cACVlX,OAAQ,qBACRmX,WAAYiL,EAEZC,SAAUzD,UACVvH,kBAAmB,CACfpa,QAAQ,EACRoN,KAAM,MAIV4J,QAAS,CACL,CAAElS,IAAK,aAAcvI,MAAO,aAAc2a,UAAU,GACpD,CAAEpS,IAAK,OAAQvI,MAAO,OAAQ2a,UAAU,EAAMD,UAAW,gBACzD,CAAEnS,IAAK,SAAUvI,MAAO,SAAU2a,UAAU,EAAMD,UAAW,gBAC7D,CAAEnS,IAAK,eAAgBvI,MAAO,UAAW2a,UAAU,EAAMD,UAAW,gBACpE,CAAEnS,IAAK,MAAOvI,MAAO,MAAO2a,UAAU,EAAMD,UAAW,gBACvD,CAAEnS,IAAK,eAAgBvI,MAAO,SAAU0a,UAAW,iBAIvDuD,YAAY,EACZC,YAAY,EACZvD,UAAU,EACVwD,YAAY,EACZC,WAAW,EAIXhD,YAAa,OAGbiD,aAAa,EACbC,SAAS,EACTC,YAAY,EAGZC,aAAc,0BAGdE,aAAc,CACVC,SAAS,EACTC,UAAU,EACVC,OAAO,EACPC,YAAY,GAEhBgK,iBAAkB,CACd9G,eAAgB,YAChBC,MAAQ8G,IACJA,EAAIzd,iBAEJjL,KAAK2oB,cAIrB,CAEA,cAAMA,GAEF,MAAM3lB,QAAahD,KAAK4C,SAAS2N,SAAS,CACtC9Q,MAAO,YACP+O,OAAQ,CACJ,CACIwB,KAAM,KACNlI,KAAM,OACN0J,UAAU,MAItB,GAAIxO,GAAQA,EAAKyiB,GAAI,CACjB,MAAM/Y,QAAcsY,EAAamD,OAAOnlB,EAAKyiB,IACzC/Y,GACA1M,KAAK4oB,UAAUC,WAAW,CAAEnc,SAEpC,CACJ,EClEJ,MAAMoc,eAAeC,EACjB,WAAAzpB,CAAY0D,EAAO,GAAIzD,EAAU,CAAA,GAC7BC,MAAMwD,EAAM,CACRT,SAAU,uBACPhD,GAEX,EAOJ,MAAMypB,mBAAmB1L,EACrB,WAAAhe,CAAYC,EAAU,IAClBC,MAAM,CACFypB,WAAYH,OACZvmB,SAAU,oBACViO,KAAM,MACHjR,GAEX,EAMJ,MAAM2pB,GAAc,CAChB5c,OAAQ,CACJ7M,MAAO,iBACP+O,OAAQ,CACJ,CACIwB,KAAM,OACNlI,KAAM,OACNnI,MAAO,OACPoO,YAAa,gBACbyD,UAAU,EACV4I,QAAS,GACT3I,KAAM,4CAEV,CACIzB,KAAM,QACNlI,KAAM,SACNnI,MAAO,WACP6R,UAAU,EACV4I,QAAS,GACT3I,KAAM,oCAEV,CACIzB,KAAM,cACNlI,KAAM,WACNnI,MAAO,qBACPoO,YAAa,+CACbqM,QAAS,GACT3I,KAAM,wEAKlByB,KAAM,CACFzT,MAAO,eACP+O,OAAQ,CACJ,CACIwB,KAAM,OACNlI,KAAM,OACNnI,MAAO,OACP6R,UAAU,EACV4I,QAAS,IAEb,CACIpK,KAAM,YACNlI,KAAM,SACNnI,MAAO,SACPya,QAAS,GACT3I,KAAM,yDAEV,CACIzB,KAAM,cACNlI,KAAM,WACNnI,MAAO,qBACPya,QAAS,GACT3I,KAAM,wCC5FtB,MAAM0X,mBAAmB9pB,EACrB,WAAAC,CAAYC,EAAU,IAClBC,MAAM,CACFO,UAAW,kBACRR,IAGPS,KAAK0M,MAAQnN,EAAQmN,OAAS,IAAIoc,OAAOvpB,EAAQyD,MAAQ,IAEzDhD,KAAKwM,SAAW,6zGAmEpB,CAEA,YAAM5L,GACF,MAAMkI,EAAW9I,KAAK0M,MAAMC,IAAI,aAE1Byc,EAAa,IAAIjN,EAAY,CAC/Bha,YAAa,sBACbpC,UAAW,yCACXqc,QAASpc,KAAK0M,MACdhF,OAAQ,CACJ9H,KAAM,yBACNwJ,MAAO,CACH,CAAEzJ,MAAO,OAAQE,OAAQ,WAAYD,KAAM,aAC3CkJ,EACM,CAAEnJ,MAAO,aAAcE,OAAQ,iBAAkBD,KAAM,eACvD,CAAED,MAAO,WAAYE,OAAQ,eAAgBD,KAAM,mBACzD,CAAEkI,KAAM,WACR,CAAEnI,MAAO,aAAcE,OAAQ,aAAcD,KAAM,WAAYqgB,QAAQ,OAInFjgB,KAAKoC,SAASgnB,EAClB,CAEA,qBAAMC,GACF,MAAMlY,EAAMnR,KAAK4C,eACEuO,EAAI6B,cAAc,CACjCvT,MAAO,kBAAkBO,KAAK0M,MAAMC,IAAI,UACxCD,MAAO1M,KAAK0M,MACZ6P,WAAY2M,GAAYhW,QAGxBlT,KAAKqD,QAEb,CAEA,2BAAMimB,GACF,MAAMnY,EAAMnR,KAAK4C,SAOjB,WANwBuO,EAAIpE,QAAQ,CAChCtN,MAAO,qBACP4O,QAAS,eAAerO,KAAK0M,MAAMC,IAAI,sDACvC4c,aAAc,aACdtB,aAAc,iBAEF,OAEhB9W,EAAIqY,cACJ,MAAMvb,QAAajO,KAAK0M,MAAMwB,KAAK,CAAE4I,WAAW,IAChD3F,EAAIsY,cACAxb,IAAyB,IAAjBA,EAAKlL,SACboO,EAAI/C,MAAMrL,QAAQ,uBAClB/C,KAAKqD,UAEL8N,EAAI/C,MAAM9K,MAAM,2BAExB,CAEA,yBAAMomB,GACF,MAAMvY,EAAMnR,KAAK4C,SACjBuO,EAAIqY,cACJ,MAAMvb,QAAajO,KAAK0M,MAAMwB,KAAK,CAAE4I,WAAW,IAChD3F,EAAIsY,cACAxb,IAAyB,IAAjBA,EAAKlL,SACboO,EAAI/C,MAAMrL,QAAQ,qBAClB/C,KAAKqD,UAEL8N,EAAI/C,MAAM9K,MAAM,yBAExB,CAEA,uBAAMqmB,GACF,MAAMxY,EAAMnR,KAAK4C,SAOjB,WANwBuO,EAAIpE,QAAQ,CAChCtN,MAAO,iBACP4O,QAAS,uBAAuBrO,KAAK0M,MAAMC,IAAI,mCAC/C4c,aAAc,SACdtB,aAAc,gBAEF,OAEhB9W,EAAIqY,cACJ,MAAMvb,QAAajO,KAAK0M,MAAMkd,SAC9BzY,EAAIsY,cACAxb,IAAyB,IAAjBA,EAAKlL,SACboO,EAAI/C,MAAMrL,QAAQ,mBAClB/C,KAAKwF,KAAK,UAAW,CAAEkH,MAAO1M,KAAK0M,SAEnCyE,EAAI/C,MAAM9K,MAAM,uBAExB,EAGJwlB,OAAO5L,WAAaiM,WC3KpBL,OAAOe,SAAWX,GAAY5c,OAC9Bwc,OAAOgB,UAAYZ,GAAYhW,KAE/B,MAAM6W,wBAAwB3M,EAC1B,WAAA9d,CAAYC,EAAU,IAClBC,MAAM,IACCD,EACHyQ,KAAM,iBACNqN,SAAU,WACVlX,OAAQ,iBACRmX,WAAY0L,WAEZ3H,cAAe8H,WACf3L,kBAAmB,CACfpa,QAAQ,EACRoN,KAAM,MAGV4J,QAAS,CACL,CAAElS,IAAK,KAAMvI,MAAO,KAAMib,MAAO,OAAQN,UAAU,EAAM/G,MAAO,cAChE,CAAErL,IAAK,OAAQvI,MAAO,OAAQ2a,UAAU,GACxC,CAAEpS,IAAK,aAAcvI,MAAO,QAAS2a,UAAU,EAAMD,UAAW,gBAChE,CACInS,IAAK,YACLvI,MAAO,SACP0a,UAAW,6DACXO,MAAO,SAEX,CAAE1S,IAAK,UAAWvI,MAAO,UAAW0a,UAAW,WAAYC,UAAU,IAGzEsD,YAAY,EACZC,YAAY,EACZvD,UAAU,EACVwD,YAAY,EACZC,WAAW,EAEXC,aAAa,EACbC,SAAS,EACTC,YAAY,EAEZyD,eAAgB,cAEhBxD,aAAc,qBAEdE,aAAc,CACVC,SAAS,EACTC,UAAU,EACVC,OAAO,EACPC,YAAY,IAGxB,CAGA,iBAAMuL,GACF,MAAM7Y,EAAMnR,KAAK4C,SACX8J,EAAQ,IAAIoc,OACZmB,QAAe9Y,EAAIZ,SAAS,CAC9B7D,WACGwc,GAAY5c,SAEnB,IAAK2d,EAAQ,OAEb,MAAMhc,QAAavB,EAAMwB,KAAK+b,GAC9B,IAAKhc,GAAMjL,MAAMC,OAEb,YADAkO,EAAI+Y,UAAUjc,GAAMjL,MAAMM,OAAS,4BAKvC,MAAM6mB,EAAQlc,EAAKjL,MAAMA,MAAMmnB,YACzBhZ,EAAIiZ,UAAU,CAChB3qB,MAAO,oCACP4O,QAAS8b,EACH,uDAAuDA,IACvD,gCACNriB,KAAMqiB,EAAQ,UAAY,UAC1B3Z,KAAM,OAGVxQ,KAAKkS,WAAWpN,IAAI4H,GACpB1M,KAAK4oB,WAAW3jB,SACpB,EC9EW,MAAMolB,wBAAwBnmB,EACzC,WAAA5E,CAAYC,EAAU,IAClBC,MAAM,CACF+C,SAAU,4BACVjB,QAAS/B,EAAQ+qB,cAAgB/qB,EAAQ+B,SAAW,MACpDipB,SAAUhrB,EAAQgrB,UAAY,KAC9BlpB,MAAO9B,EAAQ8B,QAAU9B,EAAQirB,KAAO,CAACjrB,EAAQirB,MAAQ,MACzDtpB,YAAa3B,EAAQ2B,aAAe,QACpCzB,MAAOF,EAAQE,OAAS,aACxBgrB,iBAAkBlrB,EAAQkrB,kBAAoB,MAC9CvoB,eAAe,KACZ3C,IAGPS,KAAK0qB,KAAOnrB,EAAQmrB,MAAQ,MAC5B1qB,KAAKsqB,aAAe/qB,EAAQ+qB,cAAgB/qB,EAAQ+B,SAAW,KACnE,CAEA,cAAAqpB,GACI,MAAMvY,EAAS5S,MAAMmrB,iBAGrB,OADAvY,EAAOsY,KAAO1qB,KAAK0qB,KACZtY,CACX,CAEA,OAAAwY,CAAQF,GAEJ,OADA1qB,KAAK0qB,KAAOA,EACL1qB,KAAK6qB,WAChB,EChCJ,MAAMC,GAAmB,CACrB,CAAExpB,QAAS,MAAOipB,SAAU,MAAiB9qB,MAAO,UAAsBsrB,KAAM,KAChF,CAAEzpB,QAAS,MAAOipB,SAAU,UAAkB9qB,MAAO,kBAAsBsrB,KAAM,SACjF,CAAEzpB,QAAS,MAAOipB,SAAU,SAAkB9qB,MAAO,aAAsBsrB,KAAM,KACjF,CAAEzpB,QAAS,MAAOipB,SAAU,OAAkB9qB,MAAO,WAAsBsrB,KAAM,KACjF,CAAEzpB,QAAS,MAAOipB,SAAU,MAAkB9qB,MAAO,UAAsBsrB,KAAM,KACjF,CAAEzpB,QAAS,MAAOipB,SAAU,QAAkB9qB,MAAO,kBAAsBsrB,KAAM,IACjF,CAAEzpB,QAAS,MAAOipB,SAAU,eAAkB9qB,MAAO,mBAAsBsrB,KAAM,KACjF,CAAEzpB,QAAS,MAAOipB,SAAU,gBAAkB9qB,MAAO,oBAAsBsrB,KAAM,KACjF,CAAEzpB,QAAS,QAASipB,SAAU,MAAgB9qB,MAAO,YAAsBsrB,KAAM,KACjF,CAAEzpB,QAAS,QAASipB,SAAU,QAAgB9qB,MAAO,oBAAsBsrB,KAAM,IACjF,CAAEzpB,QAAS,QAASipB,SAAU,eAAgB9qB,MAAO,qBAAsBsrB,KAAM,IACjF,CAAEzpB,QAAS,QAASipB,SAAU,aAAgB9qB,MAAO,mBAAsBsrB,KAAM,KAGrF,SAASC,GAAaD,GAClB,MAAa,MAATA,EAAyB,CAAEprB,MAAO,IAAKyE,aAAa,EAAM6mB,IAAK,KACtD,UAATF,EAAyB,CAAEprB,MAAO,QAASyE,aAAa,GAC/C,MAAT2mB,EAAyB,CAAEprB,MAAO,UAAWyE,aAAa,GACvD,CAAEA,aAAa,EAC1B,CAEe,MAAM8mB,gCAAgCxnB,EACjD,WAAApE,CAAYC,EAAU,IAClBC,MAAM,IACCD,EACHE,MAAO,wBACPM,UAAW,6BAEnB,CAEA,iBAAMY,GACF,MAAO,saAQOmqB,GAAiBliB,IAAI,CAACgW,EAAGuM,IAAM,qBAAqBA,aAAapiB,KAAK,2DAIxF,CAEA,YAAMnI,GACFZ,KAAK4C,UAAU4mB,YAAY,yBAC3B,IACI,IAAA,IAAS2B,EAAI,EAAGA,EAAIL,GAAiBzV,OAAQ8V,IAAK,CAC9C,MAAMC,EAAMN,GAAiBK,GACvBE,EAAQ,IAAIhB,gBAAgB,CAC9BloB,YAAa,YAAYgpB,IACzB7pB,QAAS8pB,EAAI9pB,QACbipB,SAAUa,EAAIb,SACd9qB,MAAO2rB,EAAI3rB,MACXiC,OAAQ,IACRyC,MAAO6mB,GAAaI,EAAIL,MACxBtM,YAAY,EACZ6M,iBAAiB,EACjBppB,eAAe,EACfuoB,iBAAkB,MAClBvpB,YAAa,UAEjBlB,KAAKoC,SAASipB,EAClB,CACJ,CAAA,QACIrrB,KAAK4C,UAAU6mB,aACnB,CACJ,EC/DJ,MAAM8B,gCAAgClsB,EAClC,WAAAC,CAAYC,EAAU,IAClBC,MAAM,IACCD,EACHQ,UAAW,8BAGfC,KAAK0M,MAAQ,IAAI8e,CACrB,CAEA,iBAAM7qB,GACF,MAAO,2xHAoEX,CAEA,oBAAM8B,SACIzC,KAAK0M,MAAM4F,OACrB,EAIJ,MAAMmZ,8BAA8B/nB,EAChC,WAAApE,CAAYC,EAAU,IAClBC,MAAM,IACCD,EACHE,MAAO,sBACPM,UAAW,2BAEnB,CAEA,iBAAMY,GACF,MAAO,koLAuGX,CAEA,YAAMC,GACFZ,KAAKoD,OAAS,IAAImoB,wBAAwB,CACtCppB,YAAa,WAEjBnC,KAAKoC,SAASpC,KAAKoD,QAEnBpD,KAAK0rB,aAAe,IAAI5qB,EAAuB,CAC3CqB,YAAa,gBACbvC,KAAM,sBACNH,MAAO,gBACPsB,SAAU,kEACVC,WAAY,UACZC,UAAW,UACXsB,SAAU,qBACVrB,YAAa,OACbG,MAAO,CAAC,mBACRC,QAAS,WACTC,UAAW,OACXC,aAAa,EACbC,WAAW,EACXC,OAAQ,IACRE,MAAO,wBACPC,MAAM,EACNC,UAAW,wBACXC,UAAW,GACX0oB,iBAAkB,KAClBkB,YAAa,SACb3pB,cAAc,EACdC,cAAc,EACd2pB,YAAa,8BAEjB5rB,KAAKoC,SAASpC,KAAK0rB,cAEnB1rB,KAAK6rB,gBAAkB,IAAI/qB,EAAuB,CAC9CqB,YAAa,mBACbvC,KAAM,kCACNH,MAAO,mBACPsB,SAAU,kEACVC,WAAY,UACZC,UAAW,UACXsB,SAAU,qBACVrB,YAAa,OACbG,MAAO,CAAC,aACRC,QAAS,WACTC,UAAW,OACXC,aAAa,EACbC,WAAW,EACXC,OAAQ,IACRE,MAAO,wBACPC,MAAM,EACNC,UAAW,yBACXC,UAAW,GACX0oB,iBAAkB,KAClBkB,YAAa,SACb3pB,cAAc,EACdC,cAAc,EACd2pB,YAAa,iCAEjB5rB,KAAKoC,SAASpC,KAAK6rB,iBAEnB7rB,KAAK8rB,qBAAuB,IAAI5nB,EAAa,CACzCzE,MAAO,wEACP8C,SAAU,qBACVjB,QAAS,WACTipB,SAAU,6BACVrpB,YAAa,OACbK,UAAW,OACXW,eAAe,EACf6pB,mBAAmB,EACnBrqB,OAAQ,IACRsqB,YAAa,GACbC,OAAQ,CACJ,4BAEJ9nB,MAAO,CACHxE,MAAO,SACPyE,aAAa,GAEjBC,QAAS,CAAEC,EAAG,YACdnC,YAAa,4BAEjBnC,KAAKoC,SAASpC,KAAK8rB,sBAEnB9rB,KAAKksB,wBAA0B,IAAIhoB,EAAa,CAC5CzE,MAAO,0DACP8C,SAAU,qBACVjB,QAAS,WACTipB,SAAU,uBACVrpB,YAAa,OACbK,UAAW,OACXW,eAAe,EACf6pB,mBAAmB,EACnBrqB,OAAQ,IACRsqB,YAAa,GACbC,OAAQ,CACJ,2BAEJ9nB,MAAO,CACHxE,MAAO,YACPyE,aAAa,GAEjBC,QAAS,CAAEC,EAAG,YACdnC,YAAa,+BAEjBnC,KAAKoC,SAASpC,KAAKksB,yBAEnBlsB,KAAKmsB,iBAAmB,IAAIC,GAAsB,CAC9CjqB,YAAa,qBACbooB,SAAU,6BACVjpB,QAAS,WACT+qB,aAAc,GACdC,YAAa,SACb5qB,OAAQ,IACR6qB,SAAU,SAEdvsB,KAAKoC,SAASpC,KAAKmsB,kBAEnB,MAAMK,EAAsB,IAAIC,EAAW,CACvCra,OAAQ,CACJnP,OAAQ,SAGhBjD,KAAK0sB,eAAiB,IAAIxS,GAAU,CAChC/X,YAAa,mBACb1C,MAAO,cACPyS,WAAYsa,EACZpS,QAAS,CACL,CAAElS,IAAK,KAAMvI,MAAO,MACpB,CAAEuI,IAAK,QAASvI,MAAO,SACvB,CAAEuI,IAAK,WAAYvI,MAAO,eAGlCK,KAAKoC,SAASpC,KAAK0sB,gBAEnB,MAAMC,EAAyB,IAAIC,EAAa,CAC5Cxa,OAAQ,CACJnP,OAAQ,SAGhBjD,KAAK6sB,2BAA6B,IAAI3S,GAAU,CAC5C/X,YAAa,gCACb1C,MAAO,gBACPyS,WAAYya,EACZvS,QAAS,CACL,CAAElS,IAAK,KAAMvI,MAAO,MACpB,CAAEuI,IAAK,QAASvI,MAAO,SACvB,CAAEuI,IAAK,SAAUvI,MAAO,SAAU0a,UAAW,YAGrDra,KAAKoC,SAASpC,KAAK6sB,2BACvB,CAEA,wBAAMtoB,CAAmBC,EAAOC,GAC5B,MAAMC,EAASD,GAAWD,GAAOG,eAAiB,KAC5C/E,EAAO8E,GAAQE,gBAAgB,KACrChF,GAAMiF,UAAUC,IAAI,WAChBJ,IAAQA,EAAOK,UAAW,GAE9B,MAAM+nB,EAAe,CACjB9sB,KAAKoD,QAAQsJ,OAAO4F,SAASmO,KAAK,IAAMzgB,KAAKoD,OAAOC,UACpDrD,KAAK0rB,cAAczmB,UACnBjF,KAAK6rB,iBAAiB5mB,UACtBjF,KAAK8rB,sBAAsB7mB,UAC3BjF,KAAKksB,yBAAyBjnB,UAC9BjF,KAAKmsB,kBAAkBlnB,UACvBjF,KAAK0sB,gBAAgBxa,YAAYI,QACjCtS,KAAK6sB,4BAA4B3a,YAAYI,SAC/CpN,OAAOC,eAEHC,QAAQC,WAAWynB,GAEzBltB,GAAMiF,UAAUiB,OAAO,WACnBpB,IAAQA,EAAOK,UAAW,EAClC,ECjYJ,MAAMgoB,uBAAuB1tB,EACzB,WAAAC,CAAYC,EAAU,IAClBC,MAAM,CACFO,UAAW,sBACRR,IAGPS,KAAKgtB,WAAaztB,EAAQytB,YAAc,GAExChtB,KAAKwM,SAAW,+oEAsDpB,CAEA,oBAAM/J,GACFzC,KAAKitB,oBAAsBjtB,KAAKktB,iBAAiBltB,KAAKgtB,WAC1D,CAEA,gBAAAE,CAAiBF,GACb,IAAKA,EACD,MAAO,6DAIX,MAEMG,GAFiC,iBAAfH,EAA0BA,EAAaI,KAAKC,UAAUL,EAAY,KAAM,IAEzE/d,MAAM,MAC7B,IAAIqe,EAAO,GA4DX,OA1DAH,EAAM3iB,QAAQ,CAAC+iB,EAAMC,KACjB,IAAKD,EAAK7f,OAEN,YADA4f,GAAQ,8CAKZ,GAAc,IAAVE,IAAgBD,EAAK7V,SAAS,WAAa6V,EAAK7V,SAAS,eAEzD,YADA4V,GAAQ,mDAAmDttB,KAAK6I,WAAW0kB,YAS/E,IAAIE,EAAQF,EAAKE,MAHG,mCAIpB,GAAIA,EAAO,CACP,MAAM,CAAGC,EAAUC,EAAUC,EAASC,GAAUJ,EAKhD,YAJAH,GAAQ,2GACiCttB,KAAK6I,WAAW6kB,EAAShgB,4EACvB1N,KAAK6I,WAAW8kB,4CAAmDC,YAAkBC,oCAGpI,CAGA,GADAJ,EAAQF,EAAKE,MAZa,gCAatBA,EAAO,CACP,OAASE,EAAUC,EAASC,GAAUJ,EAItC,YAHAH,GAAQ,8GACoCttB,KAAK6I,WAAW8kB,4CAAmDC,YAAkBC,mCAGrI,CAKA,GADAJ,EAAQF,EAAKE,MADS,iDAElBA,EAAO,CACP,OAASE,EAAUC,EAASF,GAAYD,EAKxC,YAJAH,GAAQ,iHACuCttB,KAAK6I,WAAW8kB,mDAA0DC,gFAChF5tB,KAAK6I,WAAW6kB,oCAG7D,CAGIH,EAAK7f,OAAOogB,WAAW,OACvBR,GAAQ,kDAAkDttB,KAAK6I,WAAW0kB,WAK9ED,GAAQ,qDAAqDttB,KAAK6I,WAAW0kB,aAG1ED,CACX,CAEA,gBAAAS,CAAiBC,GACbhuB,KAAKgtB,WAAagB,EAClBhuB,KAAKqD,QACT,ECpJJ,MAAM4qB,uBACF,WAAA3uB,CAAY4uB,GACRluB,KAAKkuB,WAAaA,EAClBluB,KAAKkS,WAAa,IAAIic,EAAoB,CAAE/b,OAAQ,CAAEgc,SAAUpuB,KAAKkuB,aACzE,CAEA,WAAM5b,GAEF,aADMtS,KAAKkS,WAAWI,QACftS,KAAKkS,WAAWM,OAAO5J,OAAY5I,KAAKquB,UAAU1P,GAC7D,CAEA,SAAA0P,CAAU1P,GACN,MAAO,CACH3M,GAAI2M,EAAKhS,IAAI,MACb7E,KAA2B,YAArB6W,EAAKhS,IAAI,QAAwB,eAAiB,eACxD2hB,OAAQ,CACJte,KAAM2O,EAAKhS,IAAI,oBAAsB,SACrC4hB,UAAW5P,EAAKhS,IAAI,kBAExBjH,UAAWiZ,EAAKhS,IAAI,WACpB6hB,QAAS7P,EAAKhS,IAAI,QAClB8hB,YAAa,GAErB,CAEA,aAAMC,CAAQ1rB,GACV,MAAM2rB,EAAU,IAAIC,EAAgB,CAChCR,SAAUpuB,KAAKkuB,WACfW,KAAM7rB,EAAK4N,KACX4E,KAAM,YAEJvH,QAAa0gB,EAAQzgB,OAI3B,OAHID,EAAKlL,eACC/C,KAAKkS,WAAWI,QAEnBrE,CACX,ECvBJ,MAAM6gB,qBAAqBzvB,EACvB,WAAAC,CAAYC,EAAU,IAClBC,MAAM,CACFO,UAAW,mBACRR,IAGPS,KAAK0M,MAAQnN,EAAQmN,OAAS,IAAIqiB,EAASxvB,EAAQyD,MAAQ,IAC3DhD,KAAKgvB,aAAehvB,KAAKivB,mBAAmBjvB,KAAK0M,MAAMC,IAAI,UAE3D3M,KAAKwM,SAAW,8/CA+BpB,CAEA,kBAAAyiB,CAAmBzf,GACf,MAAM0f,EAAI1f,GAAOwB,cACjB,MAAU,aAANke,GAA0B,WAANA,EAAuB,CAAEtvB,KAAM,uBAAwBgC,MAAO,gBAC5E,QAANstB,GAAqB,WAANA,EAAuB,CAAEtvB,KAAM,+BAAgCgC,MAAO,eAC/E,WAANstB,GAAwB,WAANA,EAAuB,CAAEtvB,KAAM,uBAAwBgC,MAAO,gBAC7E,CAAEhC,KAAM,wBAAyBgC,MAAO,iBACnD,CAEA,YAAMhB,GAEFZ,KAAKmvB,aAAe,IAAIjK,GAAS,CAC7BxY,MAAO1M,KAAK0M,MACZ3M,UAAW,MACXqa,QAAS,EACT5L,OAAQ,CACJ,CAAEwB,KAAM,KAAMrQ,MAAO,eACrB,CAAEqQ,KAAM,QAASrQ,MAAO,QAASic,OAAQ,SACzC,CAAE5L,KAAM,WAAYrQ,MAAO,YAC3B,CAAEqQ,KAAM,WAAYrQ,MAAO,YAC3B,CAAEqQ,KAAM,aAAcrQ,MAAO,iBAC7B,CAAEqQ,KAAM,WAAYrQ,MAAO,oBAC3B,CAAEqQ,KAAM,UAAWrQ,MAAO,UAAWya,QAAS,GAAIwB,OAAQ,UAKlE,MAAMrB,EAAmB,IAAIC,EAAkB,CAC3CpI,OAAQ,CAAEgc,SAAUpuB,KAAK0M,MAAMC,IAAI,SAEvC3M,KAAK2a,WAAa,IAAIT,GAAU,CAC5BhI,WAAYqI,EACZJ,oBAAqB,CAAC,YACtBC,QAAS,CACL,CAAElS,IAAK,KAAMvI,MAAO,KAAMib,MAAO,OAAQN,UAAU,GACnD,CAAEpS,IAAK,UAAWvI,MAAO,OAAQ0a,UAAW,WAAYC,UAAU,EAAMM,MAAO,SAC/E,CAAE1S,IAAK,WAAYvI,MAAO,WAAY0a,UAAW,QAASC,UAAU,GACpE,CAAEpS,IAAK,QAASvI,MAAO,QAAS2a,UAAU,GAC1C,CAAEpS,IAAK,QAASvI,MAAO,QAAS2a,UAAU,EAAMM,MAAO,SAE3DqD,SAAS,EACTmR,QAAS,CAAC,QACVrR,WAAW,EACXvN,KAAM,KAIV,MAAM6e,EAAU,IAAIpB,uBAAuBjuB,KAAK0M,MAAMC,IAAI,OAC1D3M,KAAKsvB,YAAc,IAAIC,EAAS,CAAEF,YAElC,MAAM3J,EAAO,CACT8J,SAAYxvB,KAAKmvB,aACjBtJ,OAAU7lB,KAAK2a,WACf,qBAAsB3a,KAAKsvB,aAGzBze,EAAW7Q,KAAK0M,MAAMC,IAAI,aAAe,CAAA,EAG3CkE,EAAS4e,cACTzvB,KAAK0vB,eAAiB,IAAI3C,eAAe,CACrCC,WAAYnc,EAAS4e,cAEzB/J,EAAK,eAAiB1lB,KAAK0vB,gBAI3BxsB,OAAOkS,KAAKvE,GAAUwE,OAAS,IAC/BrV,KAAKulB,aAAe,IAAIlmB,EAAK,CACzBqN,MAAO1M,KAAK0M,MACZF,SAAU,0FAEdkZ,EAAe,SAAI1lB,KAAKulB,cAG5BvlB,KAAKymB,QAAU,IAAIC,EAAQ,CACvBvkB,YAAa,gBACbujB,OACAiB,UAAW,aAEf3mB,KAAKoC,SAASpC,KAAKymB,SAGnB,MAAMkJ,EAAe,IAAIxT,EAAY,CACjCha,YAAa,wBACbia,QAASpc,KAAK0M,MACdhF,OAAQ,CACJ9H,KAAM,yBACNwJ,MAAO,CACH,CAAEzJ,MAAO,gBAAiBE,OAAQ,gBAAiBD,KAAM,aACzD,CAAED,MAAO,UAAWE,OAAQ,mBAAoBD,KAAM,mBACtD,CAAEkI,KAAM,WACR,CAAEnI,MAAO,kBAAmBE,OAAQ,kBAAmBD,KAAM,WAAYqgB,QAAQ,OAI7FjgB,KAAKoC,SAASutB,EAClB,CAEA,0BAAMC,SACiB9iB,EAAOkG,cAAc,CACpCvT,MAAO,kBAAkBO,KAAK0M,MAAMsF,KACpCtF,MAAO1M,KAAK0M,MACZ6P,WAAYsT,EAAc3c,QAG1BlT,KAAKqD,QAEb,CAEA,6BAAMysB,SACI9vB,KAAK0M,MAAMwB,KAAK,CAAEsB,MAAO,aAC/BxP,KAAKqD,SACLrD,KAAKwF,KAAK,mBAAoB,CAAEkH,MAAO1M,KAAK0M,OAChD,CAEA,4BAAMqjB,SACsBjjB,EAAOC,QAC3B,iDACA,mBACA,CAAEkb,aAAc,aAAcC,YAAa,mBAGxBloB,KAAK0M,MAAM9C,WACrB7G,SACL/C,KAAKwF,KAAK,mBAAoB,CAAEkH,MAAO1M,KAAK0M,OAGxD,EAGJqiB,EAAS7R,WAAa4R,aCnLtB,MAAMkB,0BAA0B5S,EAC5B,WAAA9d,CAAYC,EAAU,IAClBC,MAAM,IACCD,EACHyQ,KAAM,kBACNqN,SAAU,mBACVlX,OAAQ,kBACRmX,WAAYsP,EAEZ7J,WAAY8M,EAAcvjB,OAC1B8U,SAAUyO,EAAc3c,KACxBmO,cAAeyN,aAEftR,kBAAmB,CACfpa,QAAQ,EACRoN,KAAM,MAGViN,aAAc,CACVlI,KAAM,MACNtS,OAAQ,OAIZmX,QAAS,CACL,CACIlS,IAAK,KACLvI,MAAO,KACPib,MAAO,OACPN,UAAU,EACV/G,MAAO,cAEX,CACIrL,IAAK,SAAUvI,MAAO,SACtBuF,OAAQ,CACJ4C,KAAM,cACNvI,QAAS,CAAC,MAAO,OAAQ,SAAU,WAAY,KAAM,aAG7D,CACI2I,IAAK,UACLvI,MAAO,UACP0a,UAAW,iBACXnV,OAAQ,CACJ4C,KAAM,cAGd,CACII,IAAK,QACLvI,MAAO,QACP2a,UAAU,EACVpV,OAAQ,CAAC4C,KAAK,SAElB,CACII,IAAK,WACLvI,MAAO,WACP2a,UAAU,EACVpV,OAAQ,CAAC4C,KAAK,SAElB,CACII,IAAK,WACLvI,MAAO,WACPuF,OAAQ,CAAC4C,KAAK,SAElB,CACII,IAAK,QACLvI,MAAO,QACP0a,UAAW,4CAInBsD,QAAS,CACL,CACIzV,IAAK,gBACLvI,MAAO,eACPuF,OAAQ,CAAC4C,KAAK,UAKtB8V,YAAY,EACZC,YAAY,EACZvD,UAAU,EACVwD,YAAY,EACZC,WAAW,EAGXC,aAAa,EACbC,SAAS,EACTC,YAAY,EAGZC,aAAc,0EAGdmD,iBAAkB,MAClBC,aAAc,CACV,CAAE5hB,MAAO,OAAQC,KAAM,qBAAsBC,OAAQ,QACrD,CAAEF,MAAO,UAAWC,KAAM,qBAAsBC,OAAQ,WACxD,CAAEF,MAAO,QAASC,KAAM,qBAAsBC,OAAQ,SACtD,CAAEF,MAAO,SAAUC,KAAM,iBAAkBC,OAAQ,UACnD,CAAEF,MAAO,QAASC,KAAM,cAAeC,OAAQ,UAInDwe,aAAc,CACVC,SAAS,EACTC,UAAU,EACVC,OAAO,EACPC,YAAY,IAGxB,CAEA,0BAAMwR,CAAqBzrB,EAAOC,GAC9B,MAAMyrB,EAAWlwB,KAAK4oB,UAAUuH,mBAChC,IAAKD,EAAS7a,OAAQ,OAEtB,MAAMlE,EAAMnR,KAAK4C,eACIuO,EAAIpE,QAAQ,kCAAkCmjB,EAAS7a,6BAEtEjQ,QAAQgrB,IAAIF,EAAStnB,IAAI+V,GAAQA,EAAKjS,MAAMwB,KAAK,CAACjL,OAAQ,eAChEjD,KAAK4oB,UAAU1W,WAAWI,QAC9B,CAEA,uBAAM+d,CAAkB7rB,EAAOC,GAC3B,MAAMyrB,EAAWlwB,KAAK4oB,UAAUuH,mBAChC,IAAKD,EAAS7a,OAAQ,OAEtB,MAAMlE,EAAMnR,KAAK4C,eACIuO,EAAIpE,QAAQ,iCAAiCmjB,EAAS7a,6BAErEjQ,QAAQgrB,IAAIF,EAAStnB,IAAI+V,GAAQA,EAAKjS,MAAMwB,KAAK,CAACjL,OAAQ,WAChEjD,KAAK4oB,UAAU1W,WAAWI,QAC9B,CAEA,wBAAMge,CAAmB9rB,EAAOC,GAC5B,MAAMyrB,EAAWlwB,KAAK4oB,UAAUuH,mBAChC,IAAKD,EAAS7a,OAAQ,OAEtB,MAAMlE,EAAMnR,KAAK4C,eACIuO,EAAIpE,QAAQ,kCAAkCmjB,EAAS7a,6BAEtEjQ,QAAQgrB,IAAIF,EAAStnB,IAAI+V,GAAQA,EAAKjS,MAAMwB,KAAK,CAACjL,OAAQ,aAChEjD,KAAK4oB,UAAU1W,WAAWI,QAC9B,CAEA,yBAAMie,CAAoB/rB,EAAOC,GAC7B,MAAMyrB,EAAWlwB,KAAK4oB,UAAUuH,mBAChC,IAAKD,EAAS7a,OAAQ,OAEtB,MAAMlE,EAAMnR,KAAK4C,eACIuO,EAAIpE,QAAQ,mCAAmCmjB,EAAS7a,6BAEvEjQ,QAAQgrB,IAAIF,EAAStnB,IAAI+V,GAAQA,EAAKjS,MAAMwB,KAAK,CAACjL,OAAQ,cAChEjD,KAAK4oB,UAAU1W,WAAWI,QAC9B,CAEA,wBAAMke,CAAmBC,EAAQC,GAC7B,MAAMR,EAAWlwB,KAAK4oB,UAAUuH,mBAChC,IAAKD,EAAS7a,OAAQ,OAEtB,MAAMlE,EAAMnR,KAAK4C,SACXqnB,QAAe9Y,EAAIZ,SAAS,CAC9B9Q,MAAO,SAASywB,EAAS7a,mBACzB7G,OAAQ,CACJ,CACIwB,KAAM,QACNlI,KAAM,SACNnI,MAAO,yBACPJ,QAAS2wB,EAAStnB,IAAI+V,IAAA,CAAUhO,MAAOgO,EAAKjS,MAAMsF,GAAIrS,MAAOgf,EAAKjS,MAAMsF,MACxER,UAAU,MAItB,IAAKyY,EAAQ,OAGb,MAAM0G,EAAcT,EAASjnB,KAAK0V,GAAQA,EAAKjS,MAAMsF,IAAMiY,EAAO2G,QAAQlkB,MAC1E,IAAKikB,EAAa,OAGlB,MAAME,EAAWX,EACZtnB,IAAI+V,GAAQA,EAAKjS,MAAMsF,IACvB9M,OAAO8M,GAAMA,GAAMiY,EAAO2G,aAGzBD,EAAYziB,KAAK,CAAE0iB,MAAOC,IAChC7wB,KAAK4oB,UAAU1W,WAAWI,OAC9B,EC1LJ,MAAMwe,kBAAkBzxB,EACpB,WAAAC,CAAYC,EAAU,IAClBC,MAAM,CACFO,UAAW,gBACRR,IAGPS,KAAK0M,MAAQnN,EAAQmN,OAAS,IAAIqkB,EAAcxxB,EAAQyD,MAAQ,IAChEhD,KAAKgxB,UAAYhxB,KAAKixB,gBAAgBjxB,KAAK0M,MAAMC,IAAI,UAErD3M,KAAKwM,SAAW,mqCAyBpB,CAEA,eAAAykB,CAAgBC,GACZ,OAAIA,GAAS,GAAW,CAAEtxB,KAAM,8BAA+BgC,MAAO,eAClEsvB,GAAS,GAAW,CAAEtxB,KAAM,+BAAgCgC,MAAO,gBACnEsvB,GAAS,GAAW,CAAEtxB,KAAM,sBAAuBgC,MAAO,aACvD,CAAEhC,KAAM,eAAgBgC,MAAO,iBAC1C,CAEA,YAAMhB,GAEFZ,KAAKmvB,aAAe,IAAIjK,GAAS,CAC7BxY,MAAO1M,KAAK0M,MACZ3M,UAAW,MACXqa,QAAS,EACT5L,OAAQ,CACJ,CAAEwB,KAAM,KAAMrQ,MAAO,YACrB,CAAEqQ,KAAM,QAASrQ,MAAO,SACxB,CAAEqQ,KAAM,WAAYrQ,MAAO,YAC3B,CAAEqQ,KAAM,WAAYrQ,MAAO,eAC3B,CAAEqQ,KAAM,aAAcrQ,MAAO,iBAC7B,CAAEqQ,KAAM,WAAYrQ,MAAO,oBAC3B,CAAEqQ,KAAM,UAAWrQ,MAAO,UAAWya,QAAS,OAItD,MAAMsL,EAAO,CAAE8J,SAAYxvB,KAAKmvB,cAE1Bte,EAAW7Q,KAAK0M,MAAMC,IAAI,aAAe,CAAA,EAG3CkE,EAAS4e,cACTzvB,KAAK0vB,eAAiB,IAAI3C,eAAe,CACrCC,WAAYnc,EAAS4e,cAEzB/J,EAAK,eAAiB1lB,KAAK0vB,gBAI3BxsB,OAAOkS,KAAKvE,GAAUwE,OAAS,IAC/BrV,KAAKulB,aAAe,IAAIlmB,EAAK,CACzBqN,MAAO1M,KAAK0M,MACZF,SAAU,0FAEdkZ,EAAe,SAAI1lB,KAAKulB,cAG5BvlB,KAAKymB,QAAU,IAAIC,EAAQ,CACvBvkB,YAAa,aACbujB,OACAiB,UAAW,aAEf3mB,KAAKoC,SAASpC,KAAKymB,SAGnB,MAAMG,EAAY,CACd,CAAEjnB,MAAO,gBAAiBE,OAAQ,gBAAiBD,KAAM,wBAAyBmF,UAAW/E,KAAK0M,MAAMC,IAAI,aAC5G,CAAEhN,MAAO,qBAAsBE,OAAQ,aAAcD,KAAM,wBAAyBmF,UAAW/E,KAAK0M,MAAMC,IAAI,aAC9G,CAAE7E,KAAM,WACR,CAAEnI,MAAO,eAAgBE,OAAQ,eAAgBD,KAAM,WAAYqgB,QAAQ,IAGzEkR,EAAY,IAAIhV,EAAY,CAC9Bha,YAAa,qBACbpC,UAAW,yCACXqc,QAASpc,KAAK0M,MACdhF,OAAQ,CACJ9H,KAAM,yBACNwJ,MAAOwd,KAGf5mB,KAAKoC,SAAS+uB,EAClB,CAEA,0BAAMC,GACiCpxB,KAAK0M,MAAMC,IAAI,WACtD,CAEA,uBAAM0kB,GAC8BrxB,KAAK0M,MAAMC,IAAI,cAAe3M,KAAK0M,MAAMC,IAAI,WACjF,CAEA,yBAAM2kB,SACsBxkB,EAAOC,QAC3B,4EACA,mBACA,CAAEkb,aAAc,aAAcC,YAAa,mBAGxBloB,KAAK0M,MAAM9C,WACrB7G,SACL/C,KAAKwF,KAAK,gBAAiB,CAAEkH,MAAO1M,KAAK0M,OAGrD,EAGJqkB,EAAc7T,WAAa4T,UCvI3B,MAAMS,uBAAuBnU,EACzB,WAAA9d,CAAYC,EAAU,IAClBC,MAAM,IACCD,EACHyQ,KAAM,eACNqN,SAAU,gBACVlX,OAAQ,eACRmX,WAAY9C,EAEZ4G,SAAUoQ,EAAmBte,KAC7BmO,cAAeyP,UAEftT,kBAAmB,CACfpa,QAAQ,EACRoN,KAAM,MAGViN,aAAc,CACVlI,KAAM,MACNkc,cAAe,SAInBrX,QAAS,CACL,CACIlS,IAAK,UAAWvI,MAAO,YACvB2a,UAAU,EAAMD,UAAW,WAC3BnV,OAAQ,CACJ4C,KAAM,cAGd,CACII,IAAK,QAASvI,MAAO,QACrB2a,UAAU,EAAMD,UAAW,QAC3BnV,OAAQ,CACJ4C,KAAM,SACNvI,QAAS,CACL,CAAEoR,MAAO,IAAKhR,MAAO,YACrB,CAAEgR,MAAO,IAAKhR,MAAO,WACrB,CAAEgR,MAAO,IAAKhR,MAAO,QACrB,CAAEgR,MAAO,IAAKhR,MAAO,SACrB,CAAEgR,MAAO,IAAKhR,MAAO,YAIjC,CACIuI,IAAK,QACLvI,MAAO,QACP2a,UAAU,EAAMD,UAAW,QAC3BnV,OAAQ,CACJ4C,KAAM,WACNvI,QAAS,CACL,CAAEoR,MAAO,UAAWhR,MAAO,WAC3B,CAAEgR,MAAO,WAAYhR,MAAO,YAC5B,CAAEgR,MAAO,QAAShR,MAAO,SACzB,CAAEgR,MAAO,UAAWhR,MAAO,gBAC3B,CAAEgR,MAAO,UAAWhR,MAAO,WAC3B,CAAEgR,MAAO,OAAQhR,MAAO,QACxB,CAAEgR,MAAO,MAAOhR,MAAO,UAKnC,CACIuI,IAAK,WACLvI,MAAO,WACP2a,UAAU,EAAMD,UAAW,QAC3BnV,OAAQ,CACJ4C,KAAM,WACNvI,QAAS,CACL,CAAEoR,MAAO,aAAchR,MAAO,cAC9B,CAAEgR,MAAO,YAAahR,MAAO,aAC7B,CAAEgR,MAAO,OAAQhR,MAAO,QACxB,CAAEgR,MAAO,WAAYhR,MAAO,eAIxC,CAAEuI,IAAK,QAASvI,MAAO,QAAS2a,UAAU,EAAMD,UAAW,gBAC3D,CACInS,IAAK,YAAavI,MAAO,YAAa2a,UAAU,EAChDpV,OAAQ,CACJ4C,KAAM,SAGd,CACII,IAAK,kBAAmBvI,MAAO,SAC/B2a,UAAU,EACVpV,OAAQ,CACJ4C,KAAM,UAKlB6V,QAAS,CACL,CACIzV,IAAK,gBACLvI,MAAO,eACPuF,OAAQ,CAAC4C,KAAK,SAElB,CACII,IAAK,gCACLvI,MAAO,eACPuF,OAAQ,CAAC4C,KAAK,SAElB,CACII,IAAK,iCACLvI,MAAO,gBACPuF,OAAQ,CAAC4C,KAAK,SAElB,CACII,IAAK,yCACLvI,MAAO,wBACPuF,OAAQ,CAAC4C,KAAK,SAElB,CACII,IAAK,oBACLvI,MAAO,UACPuF,OAAQ,CAAC4C,KAAK,SAElB,CACII,IAAK,yBACLvI,MAAO,UACPuF,OAAQ,CAAC4C,KAAK,SAElB,CACII,IAAK,mBACLvI,MAAO,SACPuF,OAAQ,CAAC4C,KAAK,SAElB,CACII,IAAK,4BACLvI,MAAO,OACPuF,OAAQ,CAAC4C,KAAK,SAElB,CACII,IAAK,wBACLvI,MAAO,cACPuF,OAAQ,CAAC4C,KAAK,SAElB,CACII,IAAK,aACLvI,MAAO,aACPuF,OAAQ,CAAC4C,KAAK,SAElB,CACII,IAAK,WACLvI,MAAO,WACPuF,OAAQ,CAAC4C,KAAK,SAElB,CACII,IAAK,uBACLvI,MAAO,aACPuF,OAAQ,CAAC4C,KAAK,SAElB,CACII,IAAK,uCACLvI,MAAO,sBACPuF,OAAQ,CAAC4C,KAAK,UAKtB8V,YAAY,EACZC,YAAY,EACZvD,UAAU,EACVwD,YAAY,EACZC,WAAW,EAGXC,aAAa,EACbC,SAAS,EACTC,YAAY,EAGZC,aAAc,mBAGdE,aAAc,CACVC,SAAS,EACTC,UAAU,EACVC,OAAO,EACPC,YAAY,IAGxB,EC/LJ,MAAMiT,kBACF,WAAApyB,CAAYqyB,GACR3xB,KAAK2xB,SAAWA,EAChB3xB,KAAKkS,WAAa,IAAI0f,EAAe,CAAExf,OAAQ,CAAEhK,OAAQpI,KAAK2xB,SAAUpc,KAAM,UAAW/E,KAAM,MACnG,CAEA,WAAM8B,GAEF,aADMtS,KAAKkS,WAAWI,QACftS,KAAKkS,WAAWM,OAAO5J,OAAY5I,KAAKquB,UAAUQ,GAC7D,CAEA,SAAAR,CAAUQ,GACN,MAAO,CACH7c,GAAI6c,EAAKliB,IAAI,MACb7E,KAAM,eACNwmB,OAAQ,CACJtc,GAAI6c,EAAKliB,IAAI,WACbqD,KAAM6e,EAAKliB,IAAI,sBAAwB,SACvC4hB,UAAWM,EAAKliB,IAAI,oBAExBjH,UAAWmpB,EAAKliB,IAAI,WACpB6hB,QAASK,EAAKliB,IAAI,QAClB8hB,YAAaI,EAAKliB,IAAI,SAAW,CAACkiB,EAAKliB,IAAI,UAAY,GAE/D,CAEA,aAAM+hB,CAAQ1rB,GACV,MAAM6rB,EAAO,IAAIgD,EACX5jB,QAAa4gB,EAAK3gB,KAAK,CACzB9F,OAAQpI,KAAK2xB,SACb9C,KAAM7rB,EAAK4N,KACXkhB,MAAO9uB,EAAK+uB,OAAS/uB,EAAK+uB,MAAM1c,OAAS,EAAIrS,EAAK+uB,MAAM,GAAG/f,GAAK,OAKpE,OAHI/D,EAAKlL,eACC/C,KAAKkS,WAAWI,QAEnBrE,CACX,EC/BJ,MAAM+jB,mBAAmB3yB,EACrB,WAAAC,CAAYC,EAAU,IAClBC,MAAM,CACFO,UAAW,iBACRR,IAGPS,KAAK0M,MAAQnN,EAAQmN,OAAS,IAAIulB,EAAO1yB,EAAQyD,MAAQ,IAEzDhD,KAAKwM,SAAW,22FAiDpB,CAEA,YAAM5L,GAEF,MAAMyuB,EAAU,IAAIqC,kBAAkB1xB,KAAK0M,MAAMC,IAAI,OACrD3M,KAAKkyB,SAAW,IAAI3C,EAAS,CACzBptB,YAAa,YACbktB,UACA8C,MAAO,UACPC,cAAepyB,KAAKqyB,mBACpBC,iBAAkB,gBAClBC,gBAAiB,aAErBvyB,KAAKoC,SAASpC,KAAKkyB,UAGnB,MAAMM,EAAa,IAAIrW,EAAY,CAC/Bha,YAAa,sBACbpC,UAAW,yCACXqc,QAASpc,KAAK0M,MACdhF,OAAQ,CACJ9H,KAAM,yBACNwJ,MAAO,CACH,CAAEzJ,MAAO,cAAeE,OAAQ,cAAeD,KAAM,aACrD,CAAED,MAAO,gBAAiBE,OAAQ,gBAAiBD,KAAM,UACzD,CAAED,MAAO,eAAgBE,OAAQ,eAAgBD,KAAM,WACvD,CAAED,MAAO,cAAeE,OAAQ,cAAeD,KAAM,aACrD,CAAEkI,KAAM,WACR,CAAEnI,MAAO,eAAgBE,OAAQ,eAAgBD,KAAM,mBAInEI,KAAKoC,SAASowB,EAClB,CAMA,gBAAAH,GAEI,MAAMI,EAAchpB,OAAO0H,KAAK3B,OAAO6C,KACvC,OAAOogB,GAAazgB,IAAM,IAC9B,CAGA,wBAAM0gB,SACiB5lB,EAAOkG,cAAc,CACpCvT,MAAO,gBAAgBO,KAAK0M,MAAMC,IAAI,WAAW3M,KAAK0M,MAAMC,IAAI,WAChED,MAAO1M,KAAK0M,MACZ8D,KAAM,KACNhC,OAAQmkB,EAAYzf,KAAK1E,UAGzBxO,KAAKqD,QAEb,CAEA,0BAAMuvB,GACF,MACMC,EAAgB7yB,KAAK0M,MAAMC,IAAI,UAE/Bsd,QAAend,EAAOyD,SAAS,CACjC9Q,MAAO,uBACP+Q,KAAM,KACNhC,OAAQ,CACJ,CACIwB,KAAM,SACNrQ,MAAO,aACPmI,KAAM,SACNvI,QAXK,CAAC,MAAO,OAAQ,cAAe,UAAW,WAAY,SAAU,WAWnDqJ,IAAIsmB,IAAA,CAAQve,MAAOue,EAAGvvB,MAAOuvB,EAAExZ,QAAQ,IAAK,KAAKC,iBACnEhF,MAAOkiB,EACPrhB,UAAU,MAKtB,GAAIyY,EACA,UACUjqB,KAAK0M,MAAMwB,KAAK,CAAEjL,OAAQgnB,EAAOhnB,SACvCjD,KAAKqD,QACT,OAASC,GACLwJ,EAAOnH,MAAM,CACTmC,KAAM,QACNrI,MAAO,QACP4O,QAAS,mCAAqC/K,EAAM+K,SAE5D,CAER,CAEA,yBAAMykB,GACF,MACMC,EAAkB/yB,KAAK0M,MAAMC,IAAI,YAEjCsd,QAAend,EAAOyD,SAAS,CACjC9Q,MAAO,sBACP+Q,KAAM,KACNhC,OAAQ,CACJ,CACIwB,KAAM,WACNrQ,MAAO,iBACPmI,KAAM,SACNvI,QAXO,CAAC,MAAO,SAAU,OAAQ,UAWbqJ,IAAI8J,IAAA,CAAQ/B,MAAO+B,EAAG/S,MAAO+S,EAAEiD,iBACnDhF,MAAOoiB,EACPvhB,UAAU,MAKtB,GAAIyY,EACA,UACUjqB,KAAK0M,MAAMwB,KAAK,CAAE8kB,SAAU/I,EAAO+I,WACzChzB,KAAKqD,QACT,OAASC,GACLwJ,EAAOnH,MAAM,CACTmC,KAAM,QACNrI,MAAO,QACP4O,QAAS,qCAAuC/K,EAAM+K,SAE9D,CAER,CAEA,wBAAM4kB,GAEFnmB,EAAOnH,MAAM,CACTlG,MAAO,cACP4O,QAAS,qDAEjB,CAEA,yBAAM6kB,GAQF,SAPwBpmB,EAAOC,QAAQ,CACnCtN,MAAO,eACP4O,QAAS,0CAA0CrO,KAAK0M,MAAMC,IAAI,SAClEub,YAAa,eACbD,aAAc,gBAId,UACUjoB,KAAK0M,MAAMwB,KAAK,CAAEjL,OAAQ,WAChCjD,KAAKqD,SACLyJ,EAAOnH,MAAM,CACTmC,KAAM,UACNrI,MAAO,UACP4O,QAAS,wCAEjB,OAAS/K,GACLwJ,EAAOnH,MAAM,CACTmC,KAAM,QACNrI,MAAO,QACP4O,QAAS,2BAA6B/K,EAAM+K,SAEpD,CAER,EAGJ4jB,EAAO/U,WAAa8U,WCzNpB,MAAMmB,wBAAwB/V,EAC1B,WAAA9d,CAAYC,EAAU,IAClBC,MAAM,CACFwQ,KAAM,gBACNqN,SAAU,UACVlX,OAAQ,gBACRmX,WAAYmP,EAEZ1J,WAAY4P,EAAYrmB,OACxB8U,SAAUuR,EAAYzf,KACtBmO,cAAe2Q,WAEfxU,kBAAmB,CACfpa,QAAQ,GAGZqa,aAAc,CACVlI,KAAM,YACNtS,OAAQ,QAIZmX,QAAS,CACL,CAAElS,IAAK,KAAMvI,MAAO,KAAMib,MAAO,OAAQN,UAAU,EAAM/G,MAAO,cAChE,CAAErL,IAAK,QAASvI,MAAO,QAAS2a,UAAU,GAC1C,CACIpS,IAAK,SAAUvI,MAAO,SAAU2a,UAAU,EAC1C8Y,UAAU,EACVC,gBAAiB,CACfvrB,KAAM,SACNvI,QAAS,CACL,MAAO,OAAQ,SAAU,WAAY,KAAM,YAGjD2F,OAAQ,CACN4C,KAAM,cACNwrB,YAAa,gBACb/zB,QAAS,CACL,MAAO,OAAQ,SAAU,WAAY,KAAM,aAIrD,CAAE2I,IAAK,WAAYvI,MAAO,WAAY2a,UAAU,GAChD,CACIpS,IAAK,WAAYvI,MAAO,WAAY2a,UAAU,EAC9C8Y,UAAU,EACVC,gBAAiB,CACfvrB,KAAM,SACNvI,QAAS,IACD2D,OAAOkS,KAAKme,KAGtBruB,OAAQ,CACN4C,KAAM,cACNwrB,YAAa,kBACb/zB,QAAS,IACD2D,OAAOkS,KAAKme,MAI1B,CAAErrB,IAAK,wBAAyBvI,MAAO,WAAY2a,UAAU,EAAMD,UAAW,yBAC9E,CAAEnS,IAAK,cAAevI,MAAO,cAAe2a,UAAU,GACtD,CAAEpS,IAAK,UAAWvI,MAAO,UAAW2a,UAAU,EAAMD,UAAW,aAInEuD,YAAY,EACZC,YAAY,EACZvD,UAAU,EACVwD,YAAY,EACZC,WAAW,EAGXC,aAAa,EACbC,SAAS,EACTC,YAAY,EAGZC,aAAc,oBAGdE,aAAc,CACVC,SAAS,EACTC,UAAU,EACVC,OAAO,EACPC,YAAY,MAEblf,GAEX,EC1FJ,MAAMi0B,oBAAoBn0B,EACtB,WAAAC,CAAYC,EAAU,IAClBC,MAAM,CACFO,UAAW,kBACRR,IAGPS,KAAK0M,MAAQnN,EAAQmN,OAAS,IAAI+mB,EAAQl0B,EAAQyD,MAAQ,IAE1DhD,KAAKwM,SAAW,k1BAkBpB,CAEA,YAAM5L,GAEF,MAAM8yB,EAAe1zB,KAAK0M,MAAMC,IAAI,YAC9BgnB,EAAgBC,EAAe3qB,KAAK4qB,GAAOA,EAAIljB,QAAU+iB,GACzDI,EAAeH,EAAgBA,EAAch0B,MAAQoT,OAAO2gB,GAE5DK,EAAgB/zB,KAAK0M,MAAMC,IAAI,aAC/BqnB,EAAiBC,EAAgBhrB,KAAK4qB,GAAOA,EAAIljB,QAAUojB,GAC3DG,EAAgBF,EAAiBA,EAAer0B,MAAQoT,OAAOghB,GAGrE/zB,KAAKm0B,WAAa,IAAIjP,GAAS,CAC3BxY,MAAO1M,KAAK0M,MACZ3M,UAAW,MACXqa,QAAS,EACT5L,OAAQ,CACJ,CAAEwB,KAAM,OAAQrQ,MAAO,OAAQ8Q,KAAM,GACrC,CAAET,KAAM,WAAYrQ,MAAO,QAAS0a,UAAW,QAAS5J,KAAM,GAC9D,CAAET,KAAM,YAAarQ,MAAO,YAAa0a,UAAW,aAAc5J,KAAM,GACxE,CAAET,KAAM,WAAYrQ,MAAO,WAAY8Q,KAAM,GAC7C,CAAET,KAAM,KAAMrQ,MAAO,aAAc8Q,KAAM,GAEzC,CACIT,KAAM,WACNrQ,MAAO,cACP6M,SAAUsnB,EACVrjB,KAAM,GAEV,CACIT,KAAM,YACNrQ,MAAO,YACP6M,SAAU0nB,EACVzjB,KAAM,GAEV,CAAET,KAAM,iBAAkBrQ,MAAO,iBAAkB8Q,KAAM,GACzD,CAAET,KAAM,qBAAsBrQ,MAAO,qBAAsB0a,UAAW,aAAc5J,KAAM,GAC1F,CAAET,KAAM,UAAWrQ,MAAO,UAAW8Q,KAAM,OAKnD,MAAM2jB,EAAkB,IAAIC,EAAS,CACjCjiB,OAAQ,CAAEhK,OAAQpI,KAAK0M,MAAMC,IAAI,SAErC3M,KAAKs0B,UAAY,IAAIpa,GAAU,CAC3BhI,WAAYkiB,EACZja,oBAAqB,CAAC,UACtBC,QAAS,CACL,CAAElS,IAAK,KAAMvI,MAAO,KAAMib,MAAO,QACjC,CAAE1S,IAAK,OAAQvI,MAAO,QACtB,CAAEuI,IAAK,aAAcvI,MAAO,SAC5B,CAAEuI,IAAK,aAAcvI,MAAO,aAAcib,MAAO,SACjD,CAAE1S,IAAK,QAASvI,MAAO,SACvB,CAAEuI,IAAK,aAAcvI,MAAO,OAAQib,MAAO,UAE/CqD,SAAS,EACTlD,YAAa,OACbqU,QAAS,CAAC,OAAQ,UAClBhR,YAAa,CACT,CAAEze,MAAO,YAAaE,OAAQ,OAAQD,KAAM,aAC5C,CAAED,MAAO,iBAAkBE,OAAQ,YAAaD,KAAM,YACtD,CAAE20B,SAAS,GACX,CAAE50B,MAAO,cAAeE,OAAQ,SAAUD,KAAM,WAAYqgB,QAAQ,IAGxEuU,gBAAiB,CACbpsB,OAAQpI,KAAK0M,MAAMC,IAAI,SAI/B3M,KAAKymB,QAAU,IAAIC,EAAQ,CACvBvkB,YAAa,eACbujB,KAAM,CACF+O,cAAiBz0B,KAAKm0B,WACtBO,MAAS10B,KAAKs0B,WAElB3N,UAAW,kBAEf3mB,KAAKoC,SAASpC,KAAKymB,SAEnB,MAAMrI,EAAc,IAAIjC,EAAY,CAChCha,YAAa,uBACbia,QAASpc,KAAK0M,MACdhF,OAAQ,CACJ9H,KAAM,yBACNwJ,MAAO,CACH,CAAEzJ,MAAO,eAAgBE,OAAQ,eAAgBD,KAAM,aACvD,CAAED,MAAO,UAAWE,OAAQ,kBAAmBD,KAAM,iBACrD,CAAEkI,KAAM,WACR,CAAEnI,MAAO,iBAAkBE,OAAQ,iBAAkBD,KAAM,WAAYqgB,QAAQ,OAI3FjgB,KAAKoC,SAASgc,EAClB,CAKA,yBAAMuW,SACiB7nB,EAAOkG,cAAc,CACpCvT,MAAO,kBAAkBO,KAAK0M,MAAMC,IAAI,UACxCD,MAAO1M,KAAK0M,MACZ6P,WAAYkX,EAAQ3J,mBAGd9pB,KAAKqD,QAEnB,CAKA,4BAAMuxB,GACF,MACMC,GADW70B,KAAK0M,MAAMC,IAAI,aAGhC,IACI3M,KAAK0M,MAAMyB,IAAI,YAAa0mB,SACtB70B,KAAK0M,MAAMwB,aACXlO,KAAKqD,SAEXyJ,EAAOgoB,UAAU,CACbzmB,QAAS,WAAWwmB,EAAY,UAAY,0BAC5C/sB,KAAM,WAEd,OAASxE,GACLwJ,EAAOgoB,UAAU,CACbzmB,QAAS,6BAA6B/K,EAAM+K,UAC5CvG,KAAM,SAEd,CACJ,CAKA,2BAAMitB,GAQF,SAPwBjoB,EAAOC,QAAQ,CACnCtN,MAAO,iBACP4O,QAAS,gDAAgDrO,KAAK0M,MAAMC,IAAI,0CACxEub,YAAa,SACbD,aAAc,eAId,UACUjoB,KAAK0M,MAAM9C,UAEjBkD,EAAOgoB,UAAU,CACbzmB,QAAS,+BACTvG,KAAM,YAIV,MAAMsgB,EAASpoB,KAAKyE,SAASuwB,QAAQ,UACrC,GAAI5M,EAAQ,CACR,MAAM6M,EAAUnO,UAAU3G,MAAM8G,YAAYmB,GACxC6M,GACAA,EAAQC,MAEhB,CAGAl1B,KAAKwF,KAAK,kBAAmB,CAAEkH,MAAO1M,KAAK0M,OAC/C,OAASpJ,GACLwJ,EAAOgoB,UAAU,CACbzmB,QAAS,6BAA6B/K,EAAM+K,UAC5CvG,KAAM,SAEd,CAER,EAGJ0rB,YAAYtW,WAAasW,YChNzB,MAAM2B,yBAAyB/X,EAC3B,WAAA9d,CAAYC,EAAU,IAClBC,MAAM,IACCD,EACHyQ,KAAM,iBACNqN,SAAU,cACVlX,OAAQ,iBACRmX,WAAY8X,EACZ5M,SAAUgL,YACVhW,kBAAmB,CACfpa,QAAQ,EACRoN,KAAM,MAGV4J,QAAS,CACL,CAAElS,IAAK,KAAMvI,MAAO,KAAMib,MAAO,OAAQN,UAAU,EAAM/G,MAAO,cAChE,CAAErL,IAAK,OAAQvI,MAAO,OAAQ2a,UAAU,GACxC,CAAEpS,IAAK,WAAYvI,MAAO,QAAS2a,UAAU,EAAMD,UAAW,SAC9D,CAAEnS,IAAK,WAAYvI,MAAO,WAAY2a,UAAU,GAChD,CAAEpS,IAAK,WAAYvI,MAAO,cAAe0a,UAAYgb,GAAY,IAANA,EAAU,MAAQ,QAGjFzX,YAAY,EACZC,YAAY,EACZvD,UAAU,EACVyD,WAAW,EACXC,aAAa,EACbC,SAAS,EACTC,YAAY,EAEZG,aAAc,CACViX,UAAW,CAAC,GAAI,GAAI,IACpBC,gBAAiB,GACjBpX,aAAc,sBACdqX,UAAW,UACXpG,QAAS,CAAC,OAAQ,OAAQ,YAGtC,EChCJ,MAAMqG,6BAA6BrY,EACjC,WAAA9d,CAAYC,EAAU,IACpBC,MAAM,IACDD,EACHyQ,KAAM,sBACNqN,SAAU,gBACVlX,OAAQ,sBACRmX,WAAYoY,EACZ3S,WAAY4S,EAAiBrpB,OAC7B8U,SAAUuU,EAAiBziB,KAG3BkH,QAAS,CACP,CACElS,IAAK,KACLvI,MAAO,KACPib,MAAO,OACPN,UAAU,EACV/G,MAAO,cAET,CACErL,IAAK,OACLvI,MAAO,SACP2a,UAAU,GAEZ,CACEpS,IAAK,SACLvI,MAAO,SACP2a,UAAU,EACVD,UAAW,gBAEb,CACEnS,IAAK,oBACLvI,MAAO,YACP0a,UAAW,iBAEb,CACEnS,IAAK,WACLvI,MAAO,gBACP0a,UAAW,iBAEb,CACEnS,IAAK,WACLvI,MAAO,gBACP0a,UAAW,iBAEb,CACEnS,IAAK,UACLvI,MAAO,UACP0a,UAAW,mBAKfuD,YAAY,EACZC,YAAY,EACZvD,UAAU,EACVwD,YAAY,EACZC,WAAW,EAGXC,aAAa,EACbC,SAAS,EACTC,YAAY,EAGZC,aAAc,uDAGdC,YAAa,CACT,CACExe,KAAM,kBACNC,OAAQ,iBACRF,MAAO,wBAEX,CACEC,KAAM,oBACNC,OAAQ,UACRF,MAAO,WAET,CACEC,KAAM,kBACNC,OAAQ,QACRF,MAAO,SAET,CACEC,KAAM,kBACNC,OAAQ,YACRF,MAAO,cAKX0e,aAAc,CACZC,SAAS,EACTC,UAAU,EACVC,OAAO,EACPC,YAAY,IAGlB,CAEE,0BAAMmX,CAAqBpxB,EAAOC,GAC9B,MAAMka,EAAO3e,KAAKkS,WAAWvF,IAAIlI,EAAQkG,QAAQqH,IAMjD,aALqBlF,EAAOkG,cAAc,CAClCtG,MAAOiS,EACPpC,WAAYoZ,EAAiBE,eAG9B,CACX,CAEF,qBAAMC,CAAgBtxB,EAAOC,GAC3B,MAAMka,EAAO3e,KAAKkS,WAAWvF,IAAIlI,EAAQkG,QAAQqH,IAC3CtF,EAAQ,IAAIqpB,GAAY,CAAE/jB,GAAI2M,EAAK3M,KAEnCgkB,QAAiBlpB,EAAOyD,SAASolB,EAAiBM,SACxD,GAAKD,EAEL,IACE,MAAM/nB,QAAavB,EAAMupB,QAAQD,GACjC,IAAK/nB,EAAKlL,QACR,MAAM,IAAImzB,MAAMjoB,EAAKI,SAAW,mCAElC,IAAKJ,EAAKjL,MAAMC,OACd,MAAM,IAAIizB,MAAMjoB,EAAKjL,MAAMM,OAAS,qBAGtCtD,KAAK4C,UAAUwL,OAAOrL,QAAQ,kDACxB/C,KAAKiF,SACb,OAASkxB,GACP5yB,QAAQD,MAAM,iBAAkB6yB,GAChCn2B,KAAKkqB,UAAUiM,EAAI9nB,SAAW,2BAChC,CACF,CAEA,mBAAM+nB,CAAc5xB,EAAOC,GACzB,MAAMka,EAAO3e,KAAKkS,WAAWvF,IAAIlI,EAAQkG,QAAQqH,IAC3CtF,EAAQ,IAAIqpB,GAAY,CAAE/jB,GAAI2M,EAAK3M,KAEzC,IACE,MAAM/D,QAAavB,EAAM2pB,QACzB,IAAKpoB,EAAKlL,QACR,MAAM,IAAImzB,MAAMjoB,EAAKI,SAAW,8BAElC,IAAKJ,EAAKjL,MAAMC,OACd,MAAM,IAAIizB,MAAMjoB,EAAKjL,MAAMM,OAAS,gBAGtC,MAAM2mB,EAAShc,EAAKjL,MAAMA,MAAQ,CAAA,QAC5B8J,EAAOsG,WAAW,CACtB3T,MAAO,kBAAkBkf,EAAK3O,OAC9BqD,KAAM,IAAIhU,EAAK,CACbmN,SAAU,yNAMVxJ,KAAM,CAAEinB,YAEVzZ,KAAM,MAEV,OAAS2lB,GACP5yB,QAAQD,MAAM,eAAgB6yB,GAC9Bn2B,KAAKkqB,UAAUiM,EAAI9nB,SAAW,yBAChC,CACF,CAEA,uBAAMioB,CAAkB9xB,EAAOC,GAC7B,MAAMka,EAAO3e,KAAKkS,WAAWvF,IAAIlI,EAAQkG,QAAQqH,IAC3CtF,EAAQ,IAAIqpB,GAAY,CAAE/jB,GAAI2M,EAAK3M,KAEzC,IACE,MAAM/D,QAAavB,EAAM6pB,YACzB,IAAKtoB,EAAKlL,QACR,MAAM,IAAImzB,MAAMjoB,EAAKI,SAAW,kCAElC,IAAKJ,EAAKjL,MAAMC,OACd,MAAM,IAAIizB,MAAMjoB,EAAKjL,MAAMM,OAAS,oBAGtCtD,KAAK4C,UAAUwL,OAAOrL,QAAQ,0CACxB/C,KAAKiF,SACb,OAASkxB,GACP5yB,QAAQD,MAAM,mBAAoB6yB,GAClCn2B,KAAKkqB,UAAUiM,EAAI9nB,SAAW,6BAChC,CACF,EC7LF,MAAMmoB,8BAA8BpZ,EAChC,WAAA9d,CAAYC,EAAU,IAClBC,MAAM,IACCD,EACHyQ,KAAM,wBACNqN,SAAU,YACVlX,OAAQ,wBACRmX,WAAYmZ,GAEZ1T,WAAY2T,GAAapqB,OACzB8U,SAAUsV,GAAaxjB,KACvB6H,YAAa,OAGbX,QAAS,CACL,CAAElS,IAAK,KAAMvI,MAAO,KAAMib,MAAO,OAAQN,UAAU,EAAM/G,MAAO,cAChE,CAAErL,IAAK,QAASvI,MAAO,QAAS2a,UAAU,GAC1C,CAAEpS,IAAK,cAAevI,MAAO,SAAU2a,UAAU,EAAMD,UAAW,gBAClE,CAAEnS,IAAK,gBAAiBvI,MAAO,UAAW0a,UAAW,iBACrD,CAAEnS,IAAK,iBAAkBvI,MAAO,WAAY0a,UAAW,iBACvD,CAAEnS,IAAK,oBAAqBvI,MAAO,iBAAkB0a,UAAW,iBAChE,CAAEnS,IAAK,oBAAqBvI,MAAO,iBAAkB0a,UAAW,kBAIpEuD,YAAY,EACZC,YAAY,EACZvD,UAAU,EACVwD,YAAY,EACZC,WAAW,EAGXC,aAAa,EACbC,SAAS,EACTC,YAAY,EAGZC,aAAc,yDAGdC,YAAa,CACT,CAAExe,KAAM,cAAeC,OAAQ,aAAcF,MAAO,cACpD,CAAEC,KAAM,cAAeC,OAAQ,sBAAuBF,MAAO,wBAIjE0e,aAAc,CACVC,SAAS,EACTC,UAAU,EACVC,OAAO,EACPC,YAAY,IAGxB,CAEA,uBAAMkY,CAAkBnyB,EAAOC,GAC3B,MAAMka,EAAO3e,KAAKkS,WAAWvF,IAAIlI,EAAQkG,QAAQqH,IAE3ChP,QAAa8J,EAAOyD,SAAS,CAC/B9Q,MAAO,aACP+O,OAAQ,CACJ,CAAEwB,KAAM,KAAMrQ,MAAO,KAAMmI,KAAM,QAAS0J,UAAU,GACpD,CAAExB,KAAM,UAAWrQ,MAAO,UAAWmI,KAAM,OAAQ0J,UAAU,GAC7D,CAAExB,KAAM,YAAarQ,MAAO,OAAQmI,KAAM,WAAY0J,UAAU,MAGxExO,EAAK4zB,WAAajY,EAAKhS,IAAI,SAC3B,MAAMsd,QAAe4M,GAAQC,UAAU9zB,GACvC,GAAIinB,EAAOlnB,QACP/C,KAAK4C,SAASwL,MAAMrL,QAAQ,+BACzB,CACH,IAAIg0B,EAAM,uBACN9M,EAAOjnB,KAAKg0B,QACZD,EAAM9M,EAAOjnB,KAAKg0B,QACX/M,EAAOjnB,KAAKM,QACnByzB,EAAM9M,EAAOjnB,KAAKM,OAGtBtD,KAAK4C,SAASwL,MAAM9K,MAAMyzB,EAC9B,CACJ,CAEA,+BAAME,CAA0BzyB,EAAOC,GACnC,MAAMka,EAAO3e,KAAKkS,WAAWvF,IAAIlI,EAAQkG,QAAQqH,IAE3ChP,QAAa8J,EAAOyD,SAAS,CAC/B9Q,MAAO,aACP+O,OAAQ,CACJ,CAAEwB,KAAM,KAAMrQ,MAAO,KAAMmI,KAAM,QAAS0J,UAAU,GACpD,CAAExB,KAAM,UAAWrQ,MAAO,UAAWmI,KAAM,OAAQ0J,UAAU,GAC7D,CAAExB,KAAM,gBAAiBrQ,MAAO,WAAYmI,KAAM,OAAQ0J,UAAU,GACpE,CAAExB,KAAM,mBAAoBrQ,MAAO,UAAWmI,KAAM,WAAY0J,UAAU,MAGlFxO,EAAK4zB,WAAajY,EAAKhS,IAAI,SAC3B,MAAMsd,QAAe4M,GAAQC,UAAU9zB,GACvC,GAAIinB,EAAOlnB,QACP/C,KAAK4C,SAASwL,MAAMrL,QAAQ,+BACzB,CACH,IAAIg0B,EAAM,uBACN9M,EAAOjnB,KAAKg0B,QACZD,EAAM9M,EAAOjnB,KAAKg0B,QACX/M,EAAOjnB,KAAKM,QACnByzB,EAAM9M,EAAOjnB,KAAKM,OAGtBtD,KAAK4C,SAASwL,MAAM9K,MAAMyzB,EAC9B,CACJ,EC1GJ,MAAMG,6BAA6B73B,EAC/B,WAAAC,CAAYC,EAAU,IAClBC,MAAM,CACFO,UAAW,qBACXyM,SAAU,61BAiBPjN,GAEX,CAEA,mBAAMmH,SACIlH,MAAMkH,gBACZ1G,KAAKm3B,oBACT,CAEA,kBAAAA,GACI,MAAMC,EAASp3B,KAAKyE,QAAQG,cAAc,wBAC1C,IAAKwyB,EAAQ,OAEb,MAAMC,EAAcr3B,KAAK0M,MAAMC,IAAI,kBAAoB,GAGjD2qB,EAAYF,EAAOG,iBAAmBH,EAAOI,cAAcnP,SAGjEiP,EAAUvP,OACVuP,EAAUG,MAAMJ,GAChBC,EAAUI,OACd,CAEA,4BAAMC,CAAuBnzB,EAAOC,GAChCzE,KAAKm3B,oBACT,EAGJ,MAAMS,0BAA0Bv4B,EAC5B,WAAAC,CAAYC,EAAU,IAClBC,MAAM,CACFO,UAAW,yBACRR,IAGPS,KAAK0M,MAAQnN,EAAQmN,OAAS,IAAImrB,GAAct4B,EAAQyD,MAAQ,IAChEhD,KAAK83B,UAAY93B,KAAK0M,MAAMC,IAAI,iBAChC3M,KAAK+3B,UAAY/3B,KAAK0M,MAAMC,IAAI,iBAChC3M,KAAKg4B,YAAch4B,KAAK0M,MAAMC,IAAI,aAAezJ,OAAOkS,KAAKpV,KAAK0M,MAAMC,IAAI,aAAa0I,OAAS,EAElGrV,KAAKwM,SAAW,6iBAcpB,CAEA,YAAM5L,GACF,MAAM8kB,EAAO,CAAA,EAET1lB,KAAK83B,UACLpS,EAAK,gBAAkB,IAAIwR,qBAAqB,CAC5CxqB,MAAO1M,KAAK0M,SAIhB1M,KAAK+3B,UACLrS,EAAK,gBAAkB,IAAIrmB,EAAK,CAC5BqN,MAAO1M,KAAK0M,MACZF,SAAU,qJAIdxM,KAAKg4B,cACLtS,EAAe,SAAI,IAAIrmB,EAAK,CACxBqN,MAAO1M,KAAK0M,MACZF,SAAU,gHAIlBxM,KAAKymB,QAAU,IAAIC,EAAQ,CACvBvkB,YAAa,gBACbujB,OACAiB,UAAWzjB,OAAOkS,KAAKsQ,GAAM,IAAM,KAEvC1lB,KAAKoC,SAASpC,KAAKymB,QACvB,EC7GJ,MAAMwR,+BAA+B7a,EACjC,WAAA9d,CAAYC,EAAU,IAClBC,MAAM,IACCD,EACHyQ,KAAM,wBACNqN,SAAU,kBACVlX,OAAQ,wBACRmX,WAAY4a,GAEZnV,WAAYoV,GAAmB7rB,OAC/B8U,SAAU+W,GAAmBjlB,KAC7BmO,cAAeuW,kBACf7c,YAAa,OAEbyC,kBAAmB,CACfpa,QAAQ,EACRoN,KAAM,KACN4nB,YAAY,GAGhBC,iBAAkB,CACd7nB,KAAM,cAIV4J,QAAS,CACL,CAAElS,IAAK,KAAMvI,MAAO,KAAMib,MAAO,OAAQN,UAAU,EAAM/G,MAAO,cAChE,CAAErL,IAAK,OAAQvI,MAAO,OAAQ2a,UAAU,GACxC,CAAEpS,IAAK,UAAWvI,MAAO,UAAW0a,UAAW,YAC/C,CAAEnS,IAAK,WAAYvI,MAAO,WAAY0a,UAAW,aAIrDuD,YAAY,EACZC,YAAY,EACZvD,UAAU,EACVwD,YAAY,EACZC,WAAW,EAGXC,aAAa,EACbC,SAAS,EACTC,YAAY,EAGZC,aAAc,2EAGdE,aAAc,CACVC,SAAS,EACTC,UAAU,EACVC,OAAO,EACPC,YAAY,IAGxB,ECxDJ,MAAM6Z,kBAAkBj5B,EACpB,WAAAC,CAAYC,EAAU,IAClBC,MAAM,CACFO,UAAW,gBACRR,IAGPS,KAAK0M,MAAQnN,EAAQmN,OAAS,IAAI6rB,GAAYh5B,EAAQyD,MAAQ,IAC9DhD,KAAK83B,UAAY93B,KAAK0M,MAAMC,IAAI,aAChC3M,KAAK+3B,UAAY/3B,KAAK0M,MAAMC,IAAI,aAChC3M,KAAKw4B,WAAax4B,KAAK0M,MAAMC,IAAI,qBAAuBzJ,OAAOkS,KAAKpV,KAAK0M,MAAMC,IAAI,qBAAqB0I,OAAS,EAEjHrV,KAAKwM,SAAW,szCA0BpB,CAEA,YAAM5L,GACF,MAAM8kB,EAAO,CAAA,EAET1lB,KAAK83B,UACLpS,EAAW,KAAI,IAAIrmB,EAAK,CACpBqN,MAAO1M,KAAK0M,MACZF,SAAU,6HAIdxM,KAAK+3B,UACLrS,EAAW,KAAI,IAAIrmB,EAAK,CACpBqN,MAAO1M,KAAK0M,MACZF,SAAU,iJAIdxM,KAAKw4B,aACL9S,EAAc,QAAI,IAAIrmB,EAAK,CACvBqN,MAAO1M,KAAK0M,MACZF,SAAU,uHAIlBxM,KAAKymB,QAAU,IAAIC,EAAQ,CACvBvkB,YAAa,aACbujB,OACAiB,UAAW3mB,KAAK83B,QAAU,OAAU93B,KAAK+3B,QAAU,OAAS,YAEhE/3B,KAAKoC,SAASpC,KAAKymB,QACvB,EAGJ8R,GAAYrb,WAAaob,UCxEzB,MAAMG,6BAA6Brb,EAC/B,WAAA9d,CAAYC,EAAU,IAClBC,MAAM,IACCD,EACHyQ,KAAM,mBACNqN,SAAU,gBACVlX,OAAQ,mBACRmX,WAAYob,GAEZrX,cAAeiX,UACf9a,kBAAmB,CACfpa,QAAQ,EACRoN,KAAM,KACN4nB,YAAY,GAIhBhe,QAAS,CACL,CAAElS,IAAK,KAAMvI,MAAO,KAAMib,MAAO,OAAQN,UAAU,EAAM/G,MAAO,cAChE,CAAErL,IAAK,gBAAiBvI,MAAO,OAAQ2a,UAAU,GACjD,CAAEpS,IAAK,eAAgBvI,MAAO,KAAM2a,UAAU,EAAOD,UAAW,QAChE,CAAEnS,IAAK,UAAWvI,MAAO,UAAW2a,UAAU,GAC9C,CAAEpS,IAAK,SAAUvI,MAAO,SAAU0a,UAAW,SAC7C,CAAEnS,IAAK,gBAAiBvI,MAAO,SAAU0a,UAAW,6BACpD,CAAEnS,IAAK,UAAWvI,MAAO,UAAW0a,UAAW,aAInDuD,YAAY,EACZC,YAAY,EACZvD,UAAU,EACVwD,YAAY,EACZC,WAAW,EAGXC,aAAa,EACbC,SAAS,EACTC,YAAY,EAGZC,aAAc,0BAGdE,aAAc,CACVC,SAAS,EACTC,UAAU,EACVC,OAAO,EACPC,YAAY,IAGxB,ECxBJ,MAAMka,oBAAoB5P,EACxB,WAAAzpB,CAAY0D,EAAO,GAAIzD,EAAU,CAAA,GAC/BC,MAAMwD,EAAM,CACVT,SAAU,0BACPhD,GAEP,CAQA,sBAAaq5B,CAAUC,EAAaC,EAAc,MAChD,MACMC,EAAU,CACdlrB,aAAcgrB,GAEZC,MAAqBE,aAAeF,GAExC,MAAM7qB,QAAapL,EAAKuO,KANZ,iCAMsB2nB,GAC5B1lB,EAAOpF,GAAMjL,MAAQiL,EAG3B,OAF4B,IAAjBoF,GAAMpQ,SAAqC,IAAlBoQ,GAAMtQ,QAIjC,CAAEA,SAAS,EAAM8K,aADLwF,GAAMrQ,MAAM6K,cAAgBwF,GAAMxF,aACH7K,KAAMqQ,GAAMrQ,MAAQqQ,EAAM1Q,SAAUsL,GAEjF,CAAElL,SAAS,EAAOO,MAAO+P,GAAM/P,OAAS,uBAAwBX,SAAUsL,EACnF,CAQA,mBAAaka,CAAO0Q,EAAat5B,EAAU,IACzC,MACM0O,QAAapL,EAAKuO,KADZ,8BACsB,CAChCvD,aAAcgrB,KACXt5B,IAEC8T,EAAOpF,GAAMjL,MAAQiL,EAG3B,IAF4B,IAAjBoF,GAAMpQ,SAAqC,IAAlBoQ,GAAMtQ,QAElC,CACN,MAAMC,EAAOqQ,GAAMrQ,MAAQ,CAAA,EAE3B,MAAO,CAAED,SAAS,EAAM2J,MADV,IAAIisB,YAAY31B,EAAM,CAAET,SAAU,yBACjBS,OAAML,SAAUsL,EACjD,CACA,MAAO,CAAElL,SAAS,EAAOO,MAAO+P,GAAM/P,OAAS,sBAAuBX,SAAUsL,EAClF,EAQF,MAAMgrB,wBAAwB3b,EAC5B,WAAAhe,CAAYC,EAAU,IACpBC,MAAM,CACJypB,WAAY0P,YACZp2B,SAAU,uBACViO,KAAM,MACHjR,GAEP,EAYF,MAAM25B,YAAYnQ,EAChB,WAAAzpB,CAAY0D,EAAO,GAAIzD,EAAU,CAAA,GAC/BC,MAAMwD,EAAM,CACVT,SAAU,uBACPhD,GAEP,CAOA,iBAAa45B,CAAK/mB,EAAS,IACzB,MACMnE,QAAapL,EAAKuO,KADZ,yBACsBgB,GAC5BiB,EAAOpF,GAAMjL,MAAQiL,EAG3B,IAF4B,IAAjBoF,GAAMpQ,SAAqC,IAAlBoQ,GAAMtQ,QAElC,CACN,MAAMC,EAAOqQ,GAAMrQ,MAAQ,CAAA,EAE3B,MAAO,CAAED,SAAS,EAAM2J,MADV,IAAIwsB,IAAIl2B,EAAM,CAAET,SAAU,sBACTS,OAAML,SAAUsL,EACjD,CACA,MAAO,CAAElL,SAAS,EAAOO,MAAO+P,GAAM/P,OAAS,qBAAsBX,SAAUsL,EACjF,EAUF,MAAMmrB,gBAAgB9b,EACpB,WAAAhe,CAAYC,EAAU,IACpBC,MAAM,CACJypB,WAAYiQ,IACZ32B,SAAU,oBACViO,KAAM,MACHjR,GAEP,EC/IF,MAAM85B,wBAAwBh6B,EAC5B,WAAAC,CAAYC,EAAU,IACpBC,MAAM,CACJO,UAAW,uBACRR,IAGLS,KAAK0M,MAAQnN,EAAQmN,OAAS,IAAIisB,YAAYp5B,EAAQyD,MAAQ,IAE9DhD,KAAKwM,SAAW,wzCAgClB,CAEA,YAAM5L,GAEJZ,KAAKmvB,aAAe,IAAIjK,GAAS,CAC/BxY,MAAO1M,KAAK0M,MACZ3M,UAAW,MACXolB,iBAAiB,EACjBC,eAAgB,IAChBhL,QAAS,EACT5L,OAAQ,CACN,CAAEwB,KAAM,eAAgBrQ,MAAO,eAAgB8Q,KAAM,GACrD,CAAET,KAAM,eAAgBrQ,MAAO,eAAgB8Q,KAAM,GACrD,CAAET,KAAM,SAAUrQ,MAAO,SAAU8Q,KAAM,GACzC,CAAET,KAAM,QAASrQ,MAAO,QAAS8Q,KAAM,GACvC,CAAET,KAAM,mBAAoBrQ,MAAO,mBAAoB8Q,KAAM,GAC7D,CAAET,KAAM,aAAcrQ,MAAO,aAAc0a,UAAW,aAAc5J,KAAM,GAC1E,CAAET,KAAM,WAAYrQ,MAAO,QAAS0a,UAAW,YAAa5J,KAAM,GAClE,CAAET,KAAM,YAAarQ,MAAO,SAAU0a,UAAW,YAAa5J,KAAM,GACpE,CAAET,KAAM,UAAWrQ,MAAO,OAAQ0a,UAAW,YAAa5J,KAAM,MAKpEzQ,KAAKs5B,YAAc,IAAIpU,GAAS,CAC9BxY,MAAO1M,KAAK0M,MACZ3M,UAAW,MACXolB,iBAAiB,EACjBC,eAAgB,IAChBhL,QAAS,EACT5L,OAAQ,CACN,CAAEwB,KAAM,UAAWrQ,MAAO,UAAW8Q,KAAM,GAC3C,CAAET,KAAM,YAAarQ,MAAO,YAAa0a,UAAW,aAAc5J,KAAM,GACxE,CAAET,KAAM,kBAAmBrQ,MAAO,kBAAmB0a,UAAW,aAAc5J,KAAM,GACpF,CAAET,KAAM,eAAgBrQ,MAAO,eAAgB8Q,KAAM,GACrD,CAAET,KAAM,iBAAkBrQ,MAAO,cAAe0a,UAAW,WAAY5J,KAAM,GAC7E,CAAET,KAAM,oBAAqBrQ,MAAO,gBAAiB0a,UAAW,WAAY5J,KAAM,MAKtFzQ,KAAKu5B,YAAc,IAAIrU,GAAS,CAC9BxY,MAAO1M,KAAK0M,MACZ3M,UAAW,MACXolB,iBAAiB,EACjBC,eAAgB,IAChBhL,QAAS,EACT5L,OAAQ,CACN,CAAEwB,KAAM,gBAAiBrQ,MAAO,iBAAkB8Q,KAAM,IACxD,CAAET,KAAM,eAAgBrQ,MAAO,OAAQ8Q,KAAM,GAC7C,CAAET,KAAM,gBAAiBrQ,MAAO,QAAS8Q,KAAM,GAC/C,CAAET,KAAM,cAAerQ,MAAO,MAAO8Q,KAAM,GAC3C,CAAET,KAAM,kBAAmBrQ,MAAO,UAAW8Q,KAAM,MAKvDzQ,KAAKulB,aAAe,IAAIL,GAAS,CAC/BxY,MAAO1M,KAAK0M,MACZ3M,UAAW,MACXolB,iBAAiB,EACjBC,eAAgB,IAChBhL,QAAS,EACT5L,OAAQ,CACN,CAAEwB,KAAM,KAAMrQ,MAAO,YAAa8Q,KAAM,GACxC,CAAET,KAAM,UAAWrQ,MAAO,UAAW0a,UAAW,WAAY5J,KAAM,GAClE,CAAET,KAAM,WAAYrQ,MAAO,gBAAiB0a,UAAW,WAAY5J,KAAM,MAI7E,MAAMiV,EAAO,CACX8J,SAAYxvB,KAAKmvB,aACjBqK,QAAWx5B,KAAKs5B,YAChBG,QAAWz5B,KAAKu5B,YAChBxT,SAAY/lB,KAAKulB,cAGnBvlB,KAAKymB,QAAU,IAAIC,EAAQ,CACzBvkB,YAAa,aACbujB,OACAiB,UAAW,aAEb3mB,KAAKoC,SAASpC,KAAKymB,SAGnB,MAMMiT,EAAU,IAAIvd,EAAY,CAC9Bha,YAAa,qBACbpC,UAAW,yCACXqc,QAASpc,KAAK0M,MACdhF,OAAQ,CACN9H,KAAM,yBACNwJ,MAZc,CAChB,CAAEzJ,MAAO,iBAAkBE,OAAQ,iBAAkBD,KAAM,mBAC3D,CAAEkI,KAAM,WACR,CAAEnI,MAAO,gBAAiBE,OAAQ,eAAgBD,KAAM,WAAYqgB,QAAQ,OAY9EjgB,KAAKoC,SAASs3B,EAChB,CAIA,2BAAMC,GACJ,MAAMC,EAAS55B,KAAK0M,MAAMC,IAAI,gBAC9B,GAAKitB,EAKL,IACE55B,KAAK4C,UAAUwL,OAAOsZ,OAAO,wBAE7B,MAAMzZ,QAAa0qB,YAAYxQ,OAAOyR,EAAQ,CAAEC,eAAe,IAC/D,GAAI5rB,EAAKlL,SAAWkL,EAAKjL,KACvBhD,KAAK0M,MAAMyB,IAAIF,EAAKjL,YACdhD,KAAKqD,SACXrD,KAAK4C,UAAUwL,OAAOrL,UAAU,wBAC3B,CACL,MAAMg0B,EAAM9oB,EAAK3K,OAAS,gBAC1BtD,KAAK4C,UAAUwL,OAAO9K,QAAQyzB,EAChC,CACF,OAASxkB,GACPvS,KAAK4C,UAAUwL,OAAO9K,QAAQiP,EAAElE,SAAW,gBAC7C,MAlBErO,KAAK4C,UAAUwL,OAAO0rB,UAAU,4BAmBpC,CAEA,yBAAMC,GAOJ,SANwBjtB,EAAOC,QAC7B,mDAAmD/M,KAAK0M,MAAMC,IAAI,iBAAmB,kBACrF,mBACA,CAAEsb,aAAc,aAAcC,YAAa,WAK7C,IACE,MAAMja,QAAajO,KAAK0M,MAAM9C,UAC1BqE,GAAMlL,QACR/C,KAAKwF,KAAK,gBAAiB,CAAEkH,MAAO1M,KAAK0M,QAEzC1M,KAAK4C,UAAUwL,OAAO9K,QAAQ,gBAElC,OAASiP,GACPvS,KAAK4C,UAAUwL,OAAO9K,QAAQiP,EAAElE,SAAW,gBAC7C,CACF,CAEA,iBAAaqW,CAAK7W,GACd,MAAMI,QAAa0qB,YAAYxQ,OAAOta,GACtC,GAAII,GAAMvB,MAAO,CACb,MAAMvE,EAAO,IAAIkxB,gBAAgB,CAAE3sB,MAAOuB,EAAKvB,QACzC0b,EAAS,IAAItb,EAAO,CACtB1J,QAAQ,EACRoN,KAAM,KACN6C,KAAMlL,EACNmL,QAAS,CAAC,CAAE1C,KAAM,QAAS2C,MAAO,gBAAiBC,SAAS,MAIhE,aAFM4U,EAAO/kB,QAAO,EAAMglB,SAAShV,MACnC+U,EAAO1D,OACA0D,CACX,CAEA,OADAtb,EAAOnH,MAAM,CAAE0I,QAAS,yCAAyCR,IAAgB/F,KAAM,YAChF,IACX,EAGFuxB,gBAAgBW,YAAcrB,YCzN9B,MAAMsB,6BAA6B7c,EACjC,WAAA9d,CAAYC,EAAU,IACpBC,MAAM,IACDD,EAGHyQ,KAAM,yBACNqN,SAAU,gBACVlX,OAAQ,yBAGRmX,WAAY2b,gBAGZzQ,SAAU6Q,gBACV7b,kBAAmB,CACjBpa,QAAQ,GAKVgX,QAAS,CACP,CAAElS,IAAK,eAAgBvI,MAAO,eAAgB2a,UAAU,GACxD,CAAEpS,IAAK,UAAWvI,MAAO,UAAW2a,UAAU,EAAMD,UAAW,gBAC/D,CAAEnS,IAAK,YAAavI,MAAO,YAAa2a,UAAU,EAAMD,UAAW,cACnE,CAAEnS,IAAK,YAAavI,MAAO,SAAU0a,UAAW,aAChD,CAAEnS,IAAK,UAAWvI,MAAO,OAAQ0a,UAAW,aAC5C,CAAEnS,IAAK,WAAYvI,MAAO,QAAS0a,UAAW,aAC9C,CAAEnS,IAAK,mBAAoBvI,MAAO,QAAS2a,UAAU,EAAMD,UAAW,gBACtE,CAAEnS,IAAK,aAAcvI,MAAO,aAAc0a,UAAW,cACrD,CAAEnS,IAAK,0BAA2BvI,MAAO,cAAe2a,UAAU,IAIpEsD,YAAY,EACZC,YAAY,EACZvD,UAAU,EACVwD,YAAY,EACZC,WAAW,EAGXhD,YAAa,OAGbiD,aAAa,EACbC,SAAS,EACTC,YAAY,EAGZC,aAAc,0BAGdE,aAAc,CACZC,SAAS,EACTC,UAAU,EACVC,OAAO,EACPC,YAAY,GAGdgK,iBAAkB,CACd9G,eAAgB,SAChBuY,cAAe,YACftY,MAAQ8G,IACJA,EAAIzd,iBAEJjL,KAAK2oB,cAIjB,CAEA,cAAMA,GAEF,MAAM3lB,QAAahD,KAAK4C,SAAS2N,SAAS,CACtC9Q,MAAO,sBACP+O,OAAQ,CACJ,CACIwB,KAAM,SACNlI,KAAM,OACN0J,UAAU,MAItB,GAAIxO,GAAQA,EAAK42B,OAAQ,CACrB,MAAM3rB,QAAa0qB,YAAYxQ,OAAOnlB,EAAK42B,QACvC3rB,EAAKvB,OACL1M,KAAK4oB,UAAUC,WAAW5a,EAElC,CACJ,EClFF,MAAMksB,gBAAgB96B,EACpB,WAAAC,CAAYC,EAAU,IACpBC,MAAM,CACJO,UAAW,cACRR,IAGLS,KAAK0M,MAAQnN,EAAQmN,OAAS,IAAIwsB,IAAI35B,EAAQyD,MAAQ,IAEtDhD,KAAKwM,SAAW,2kDAsClB,CAEA,YAAM5L,GAEJZ,KAAKo6B,YAAc,IAAIlV,GAAS,CAC9BxY,MAAO1M,KAAK0M,MACZ3M,UAAW,MACXolB,iBAAiB,EACjBC,eAAgB,IAChBhL,QAAS,EACT5L,OAAQ,CACN,CAAEwB,KAAM,YAAarQ,MAAO,YAAa0a,UAAW,aAAc5J,KAAM,GACxE,CAAET,KAAM,SAAUrQ,MAAO,SAAU0a,UAAW,aAAc5J,KAAM,GAClE,CAAET,KAAM,cAAerQ,MAAO,OAAQ8Q,KAAM,GAC5C,CAAET,KAAM,YAAarQ,MAAO,KAAM8Q,KAAM,GACxC,CAAET,KAAM,OAAQrQ,MAAO,eAAgB8Q,KAAM,OAKjDzQ,KAAKq6B,aAAe,IAAInV,GAAS,CAC/BxY,MAAO1M,KAAK0M,MACZ3M,UAAW,MACXolB,iBAAiB,EACjBC,eAAgB,IAChBhL,QAAS,EACT5L,OAAQ,CACN,CAAEwB,KAAM,WAAYrQ,MAAO,WAAY0a,UAAW,aAAc5J,KAAM,GACtE,CAAET,KAAM,sBAAuBrQ,MAAO,sBAAuB8Q,KAAM,GACnE,CAAET,KAAM,UAAWrQ,MAAO,UAAW0a,UAAW,WAAY5J,KAAM,GAClE,CAAET,KAAM,eAAgBrQ,MAAO,eAAgB0a,UAAW,WAAY5J,KAAM,GAC5E,CAAET,KAAM,aAAcrQ,MAAO,aAAc8Q,KAAM,GACjD,CAAET,KAAM,gBAAiBrQ,MAAO,gBAAiB8Q,KAAM,OAK3DzQ,KAAKulB,aAAe,IAAIL,GAAS,CAC/BxY,MAAO1M,KAAK0M,MACZ3M,UAAW,MACXolB,iBAAiB,EACjBC,eAAgB,IAChBhL,QAAS,EACT5L,OAAQ,CACN,CAAEwB,KAAM,KAAMrQ,MAAO,YAAa8Q,KAAM,GACxC,CAAET,KAAM,UAAWrQ,MAAO,UAAW0a,UAAW,WAAY5J,KAAM,GAClE,CAAET,KAAM,WAAYrQ,MAAO,gBAAiB0a,UAAW,WAAY5J,KAAM,MAI7E,MAAMiV,EAAO,CACX4U,QAAWt6B,KAAKo6B,YAChBG,SAAYv6B,KAAKq6B,aACjBtU,SAAY/lB,KAAKulB,cAGnBvlB,KAAKymB,QAAU,IAAIC,EAAQ,CACzBvkB,YAAa,WACbujB,OACAiB,UAAW,YAEb3mB,KAAKoC,SAASpC,KAAKymB,SAGnB,MAMMiT,EAAU,IAAIvd,EAAY,CAC9Bha,YAAa,mBACbpC,UAAW,yCACXqc,QAASpc,KAAK0M,MACdhF,OAAQ,CACN9H,KAAM,yBACNwJ,MAZc,CAChB,CAAEzJ,MAAO,UAAWE,OAAQ,cAAeD,KAAM,mBACjD,CAAEkI,KAAM,WACR,CAAEnI,MAAO,iBAAkBE,OAAQ,aAAcD,KAAM,WAAYqgB,QAAQ,OAY7EjgB,KAAKoC,SAASs3B,EAChB,CAIA,wBAAMc,GACJ,IACEx6B,KAAK4C,UAAUwL,OAAOsZ,OAAO,+BACvB1nB,KAAK0M,MAAM4F,cACXtS,KAAKqD,SACXrD,KAAK4C,UAAUwL,OAAOrL,UAAU,oBAClC,OAASwP,GACPvS,KAAK4C,UAAUwL,OAAO9K,QAAQiP,EAAElE,SAAW,iBAC7C,CACF,CAEA,uBAAMosB,GAQJ,SALwB3tB,EAAOC,QADnB,gDADE,mBAEqC,CACjDkb,aAAc,aACdC,YAAa,WAKf,IACE,MAAMja,QAAajO,KAAK0M,MAAM9C,UAC1BqE,GAAMlL,QACR/C,KAAKwF,KAAK,cAAe,CAAEkH,MAAO1M,KAAK0M,QAEvC1M,KAAK4C,UAAUwL,OAAO9K,QAAQ,gBAElC,OAASiP,GACPvS,KAAK4C,UAAUwL,OAAO9K,QAAQiP,EAAElE,SAAW,gBAC7C,CACF,EAGF8rB,QAAQH,YAAcd,IC5KtB,MAAMwB,qBAAqBtd,EACzB,WAAA9d,CAAYC,EAAU,IACpBC,MAAM,IACDD,EAGHyQ,KAAM,qBACNqN,SAAU,eACVlX,OAAQ,qBAGRmX,WAAY8b,QAGZ5Q,SAAU2R,QACV3c,kBAAmB,CACjBpa,QAAQ,EACRoN,KAAM,MAIR4J,QAAS,CACP,CAAElS,IAAK,YAAavI,MAAO,YAAa2a,UAAU,GAClD,CAAEpS,IAAK,cAAevI,MAAO,OAAQ2a,UAAU,EAAMD,UAAW,gBAChE,CAAEnS,IAAK,YAAavI,MAAO,KAAM2a,UAAU,EAAMD,UAAW,gBAC5D,CAAEnS,IAAK,SAAUvI,MAAO,SAAU2a,UAAU,GAC5C,CAAEpS,IAAK,WAAYvI,MAAO,WAAY2a,UAAU,EAAMD,UAAW,gBACjE,CAAEnS,IAAK,OAAQvI,MAAO,UAAW0a,UAAW,gBAC5C,CAAEnS,IAAK,UAAWvI,MAAO,UAAW2a,UAAU,EAAMD,UAAW,YAC/D,CAAEnS,IAAK,eAAgBvI,MAAO,eAAgB2a,UAAU,EAAMD,UAAW,YACzE,CAAEnS,IAAK,UAAWvI,MAAO,UAAW2a,UAAU,EAAMD,UAAW,aAIjEuD,YAAY,EACZC,YAAY,EACZvD,UAAU,EACVwD,YAAY,EACZC,WAAW,EAGXhD,YAAa,OAGbiD,aAAa,EACbC,SAAS,EACTC,YAAY,EAGZC,aAAc,yBAGdE,aAAc,CACZC,SAAS,EACTC,UAAU,EACVC,OAAO,EACPC,YAAY,IAGlB,EC/DF,MAAMkc,0BAA0Bj3B,EAC5B,WAAApE,CAAYC,EAAU,IAClBC,MAAM,IACCD,EACHE,MAAO,+BACPM,UAAW,uBAEnB,CAEA,iBAAMY,GACF,MAAO,osCA0BX,CAEA,YAAMC,GACFZ,KAAK46B,gBAAkB,IAAI12B,EAAa,CACpC/B,YAAa,mBACbI,SAAU,qBACVlB,MAAO,CAAC,YAAa,eACrBE,UAAW,SAEfvB,KAAKoC,SAASpC,KAAK46B,iBAEnB56B,KAAK66B,YAAc,IAAIC,EAAS,CAC5B34B,YAAa,eACbI,SAAU,oCAEdvC,KAAKoC,SAASpC,KAAK66B,aAEnB76B,KAAK+6B,iBAAmB,IAAI7gB,GAAU,CAClC/X,YAAa,oBACb1C,MAAO,oBACP6d,WAAY,IAAI0d,GAAiB,CAAE5oB,OAAQ,CAAE6oB,MAAO,WAAYC,OAAQ,KACxE9gB,QAAS,CAAC,CAAElS,IAAK,QAASvI,MAAO,SAAW,CAAEuI,IAAK,SAAUvI,MAAO,SAAU0a,UAAW,YAE7Fra,KAAKoC,SAASpC,KAAK+6B,kBAEnB/6B,KAAKm7B,iBAAmB,IAAIjhB,GAAU,CAClC/X,YAAa,oBACb1C,MAAO,oBACP6d,WAAY,IAAI0d,GAAiB,CAAE5oB,OAAQ,CAAEnP,OAAQ,SAAUg4B,MAAO,WAAYC,OAAQ,KAC1F9gB,QAAS,CAAC,CAAElS,IAAK,QAASvI,MAAO,SAAW,CAAEuI,IAAK,gBAAiBvI,MAAO,YAE/EK,KAAKoC,SAASpC,KAAKm7B,iBACvB,ECtEJ,MAAMC,4BAA4Bhe,EAC9B,WAAA9d,CAAYC,EAAU,IAClBC,MAAM,IACCD,EACHyQ,KAAM,qBACNqN,SAAU,sBACVlX,OAAQ,qBACRmX,WAAY+d,GACZtY,WAAYuY,GAAgBhvB,OAC5B8U,SAAUka,GAAgBpoB,KAE1BkH,QAAS,CACL,CAAElS,IAAK,KAAMvI,MAAO,KAAMib,MAAO,QACjC,CAAE1S,IAAK,OAAQvI,MAAO,QACtB,CAAEuI,IAAK,aAAcvI,MAAO,QAAS0a,UAAW,sBAChD,CAAEnS,IAAK,gBAAiBvI,MAAO,cAC/B,CAAEuI,IAAK,YAAavI,MAAO,SAAUic,OAAQ,YAGjDiC,YAAY,EACZvD,UAAU,EACVyD,WAAW,EACXC,aAAa,EACbC,SAAS,EACTC,YAAY,EACZkR,QAAS,CAAC,OAAQ,UAClB/Q,aAAc,CACViX,UAAW,CAAC,GAAI,GAAI,IACpBC,gBAAiB,GACjBpX,aAAc,gCACdqX,UAAW,YAIvB,EClCJ,MAAM+F,8BAA8Bne,EAChC,WAAA9d,CAAYC,EAAU,IAClBC,MAAM,IACCD,EACHyQ,KAAM,uBACNqN,SAAU,iBACVlX,OAAQ,uBACRmX,WAAYke,GACZzY,WAAY0Y,GAAkBnvB,OAC9B8U,SAAUqa,GAAkBvoB,KAE5BkH,QAAS,CACL,CAAElS,IAAK,KAAMvI,MAAO,KAAMib,MAAO,QACjC,CAAE1S,IAAK,OAAQvI,MAAO,QACtB,CAAEuI,IAAK,WAAYvI,MAAO,YAC1B,CAAEuI,IAAK,aAAcvI,MAAO,QAAS0a,UAAW,sBAChD,CAAEnS,IAAK,WAAYvI,MAAO,YAC1B,CAAEuI,IAAK,YAAavI,MAAO,SAAUic,OAAQ,YAGjDiC,YAAY,EACZvD,UAAU,EACVyD,WAAW,EACXC,aAAa,EACbC,SAAS,EACTC,YAAY,EAEZG,aAAc,CACViX,UAAW,CAAC,GAAI,GAAI,IACpBC,gBAAiB,GACjBpX,aAAc,2BACdqX,UAAW,uBACXpG,QAAS,CAAC,OAAQ,YAG9B,ECpCJ,MAAMsM,yBAAyBr8B,EAC3B,WAAAC,CAAYC,EAAU,IAClBC,MAAM,CACFO,UAAW,wBACRR,IAEPS,KAAK0M,MAAQnN,EAAQmN,KACzB,CAEA,WAAA/L,GACI,MAAO,koCAwBX,EChCJ,MAAMg7B,8BAA8Bve,EAChC,WAAA9d,CAAYC,EAAU,IAClBC,MAAM,IACCD,EACHyQ,KAAM,wBACNqN,SAAU,kBACVlX,OAAQ,wBACRmX,WAAY0d,GACZ3Z,cAAeqa,iBACfle,kBAAmB,CACfpa,QAAQ,EACRoN,KAAM,MAGV4J,QAAS,CACL,CAAElS,IAAK,KAAMvI,MAAO,KAAMib,MAAO,QACjC,CAAE1S,IAAK,UAAWvI,MAAO,YAAa0a,UAAW,YACjD,CAAEnS,IAAK,oBAAqBvI,MAAO,QACnC,CAAEuI,IAAK,qBAAsBvI,MAAO,UACpC,CAAEuI,IAAK,QAASvI,MAAO,SACvB,CAAEuI,IAAK,WAAYvI,MAAO,YAC1B,CAAEuI,IAAK,SAAUvI,MAAO,SAAU0a,UAAW,UAGjDwD,YAAY,EACZvD,UAAU,EACVyD,WAAW,EACXC,aAAa,EAEbK,aAAc,CACViX,UAAW,CAAC,GAAI,GAAI,IACpBC,gBAAiB,GACjBpX,aAAc,uBACdqX,UAAW,UACXpG,QAAS,CAAC,UAGtB,ECtCJ,MAAMwM,uBAAuBv8B,EACzB,WAAAC,CAAYC,EAAU,IAClBC,MAAM,CACFO,UAAW,sBACRR,IAEPS,KAAK0M,MAAQnN,EAAQmN,KACzB,CAEA,WAAA/L,GACI,MAAO,8OAOX,CAEA,MAAAC,GACIZ,KAAK67B,SAAW,IAAI3W,GAAS,CACzB/iB,YAAa,YACbuK,MAAO1M,KAAK0M,MACZ8B,OAAQ,CACJ,CAAEwB,KAAM,WAAYrQ,MAAO,WAAYic,OAAQ,SAC/C,CAAE5L,KAAM,eAAgBrQ,MAAO,eAAgBic,OAAQ,WACvD,CAAE5L,KAAM,cAAerQ,MAAO,eAC9B,CAAEqQ,KAAM,aAAcrQ,MAAO,cAC7B,CAAEqQ,KAAM,YAAarQ,MAAO,YAAaic,OAAQ,YACjD,CAAE5L,KAAM,mBAAoBrQ,MAAO,cAAeic,OAAQ,WAGlE5b,KAAKoC,SAASpC,KAAK67B,SACvB,EChCJ,MAAMC,4BAA4B1e,EAC9B,WAAA9d,CAAYC,EAAU,IAClBC,MAAM,IACCD,EACHyQ,KAAM,qBACNqN,SAAU,qBACVlX,OAAQ,qBACRmX,WAAYlC,EACZiG,cAAeua,eACfpe,kBAAmB,CACfpa,QAAQ,EACRoN,KAAM,MAGV4J,QAAS,CACL,CAAElS,IAAK,KAAMvI,MAAO,KAAMib,MAAO,QACjC,CAAE1S,IAAK,oBAAqBvI,MAAO,QACnC,CAAEuI,IAAK,cAAevI,MAAO,eAC7B,CAAEuI,IAAK,WAAYvI,MAAO,WAAY0a,UAAW,SACjD,CAAEnS,IAAK,cAAevI,MAAO,eAC7B,CAAEuI,IAAK,eAAgBvI,MAAO,eAAgBic,OAAQ,WACtD,CAAE1T,IAAK,YAAavI,MAAO,YAAa0a,UAAW,aAGvDwD,YAAY,EACZvD,UAAU,EACVyD,WAAW,EACXC,aAAa,EAEbK,aAAc,CACViX,UAAW,CAAC,GAAI,GAAI,IACpBC,gBAAiB,GACjBpX,aAAc,oBACdqX,UAAW,WACXpG,QAAS,CAAC,OAAQ,YAG9B,EClCW,MAAM2M,qBAAqB18B,EACtC,WAAAC,CAAYC,EAAU,IAClBC,MAAM,CACFO,UAAW,uBACRR,IAGPS,KAAKC,MAAQ,CACT+7B,QAAS,EACTC,QAAS,EACTC,UAAW,EACXC,OAAQ,EACRC,UAAW,GAGfp8B,KAAKwM,SAAW,8jLAoGpB,CAEA,cAAAH,GACErM,KAAKwD,YACDxD,KAAK+J,aACL/J,KAAKqD,QAEX,CAEA,eAAMG,GACFxD,KAAKC,MAAQD,KAAK0M,MAAMoS,WAAWud,MACvC,EC7HW,MAAMC,sBAAsBj9B,EACvC,WAAAC,CAAYC,EAAU,IAClBC,MAAM,CACFO,UAAW,wBACRR,IAGPS,KAAKu8B,OAAS,CACVt5B,OAAQ,UACRu5B,QAAS,CAAEC,OAAQ,EAAGC,MAAO,GAC7BznB,SAAU,IAGdjV,KAAKwM,SAAW,4zGAwDpB,CAEA,cAAAH,GACErM,KAAK28B,aACD38B,KAAK+J,aACL/J,KAAKqD,QAEX,CAEA,gBAAMs5B,GACF,IAAK38B,KAAK0M,MAAMkS,EAAEyd,OAAQ,OAC1B,MAAMr5B,EAAOhD,KAAK0M,MAAMoS,WAExB,IAAI8d,EAAiB,UACc,IAA/B55B,EAAKq5B,OAAOQ,eACZD,EAAiB,WACT55B,EAAK85B,UAAUL,SACvBG,EAAiB,WAErB58B,KAAKu8B,OAAS,CACVK,iBACA3nB,SAAUjS,EAAKiS,SACfunB,QAAS,CACLC,OAAQz5B,EAAKq5B,OAAOQ,eACpBH,MAAO15B,EAAKw5B,QAAQnnB,QAExBgnB,OAAQr5B,EAAKq5B,OACbS,UAAW95B,EAAK85B,WAGpB98B,KAAK+8B,kBAAoB/8B,KAAKg9B,qBAAqBh9B,KAAKu8B,OAAOK,gBAC/D58B,KAAKi9B,sBACLj9B,KAAKk9B,uBAET,CAEA,mBAAAD,GACSj9B,KAAKu8B,OAAOtnB,WAEjBjV,KAAKu8B,OAAOY,cAAgBj6B,OAAOyG,OAAO3J,KAAKu8B,OAAOtnB,UAAUrM,IAAIiN,IAChE,IAAIunB,EAAgB,UAGpB,MAAMC,GAAaxnB,EAAQynB,cAAgB,IAAMznB,EAAQ0nB,gBAAkB,GAQ3E,OAPIF,EAAY,KACZD,EAAgB,YAEhBC,EAAY,KAA2B,IAApBxnB,EAAQ2mB,WAC3BY,EAAgB,YAGb,IACAvnB,EACH5S,OAAQm6B,EACRI,iBAAkBx9B,KAAKy9B,qBAAqBL,GAC5CM,WAAY19B,KAAK29B,eAAeP,GAChCQ,OAAQ/nB,EAAQynB,cAAgB,EAChCO,SAAUhoB,EAAQ0nB,gBAAkB,EAEpCvB,QAASnmB,EAAQynB,cAAgB,EACjCrB,QAASpmB,EAAQ0nB,gBAAkB,KAG/C,CAEA,qBAAAL,GACI,IAAKl9B,KAAKu8B,OAAOO,UAIb,OAHA98B,KAAK89B,oBAAsB,UAC3B99B,KAAK+9B,qBAAuB,kBAC5B/9B,KAAKg+B,cAAgB,2BAIrBh+B,KAAKu8B,OAAOO,UAAUL,QACtBz8B,KAAK89B,oBAAsB,UAC3B99B,KAAK+9B,qBAAuB,eAC5B/9B,KAAKg+B,cAAgB,yBAErBh+B,KAAK89B,oBAAsB,UAC3B99B,KAAK+9B,qBAAuB,cAC5B/9B,KAAKg+B,cAAgB,oBAE7B,CAEA,oBAAAhB,CAAqB/5B,GAOjB,MANgB,CACZg7B,QAAW,eACXnE,QAAW,eACXoE,SAAY,cACZC,QAAW,cAEAl7B,IAAW,YAC9B,CAEA,oBAAAw6B,CAAqBx6B,GAMjB,MALgB,CACZg7B,QAAW,aACXnE,QAAW,aACXoE,SAAY,aAEDj7B,IAAW,cAC9B,CAEA,cAAA06B,CAAe16B,GAMX,MALc,CACVg7B,QAAW,uBACXnE,QAAW,+BACXoE,SAAY,qBAEHj7B,IAAW,qBAC5B,CAEA,2BAAMm7B,CAAsB55B,EAAOC,GAC/B,IACIA,EAAQM,UAAW,QACb/E,KAAK0M,MAAM4F,OACrB,OAAShP,GACLC,QAAQD,MAAM,4BAA6BA,EAC/C,CAAA,QACImB,EAAQM,UAAW,CACvB,CACJ,CAEA,4BAAMs5B,SACIvxB,EAAOsd,UAAU,CACnB3qB,MAAO,kBACP4O,QAAS,yCACTvG,KAAM,QAEd,EC7LJ,MAAMw2B,uBAAuBj/B,EACzB,WAAAC,CAAYC,EAAU,IAClBC,MAAM,CACFO,UAAW,sBACRR,IAIPS,KAAK0M,MAAQnN,EAAQmN,OAAS,IAAI6xB,GAAIh/B,EAAQyD,MAAQ,IAGtDhD,KAAKymB,QAAU,KACfzmB,KAAKmvB,aAAe,KACpBnvB,KAAKw+B,YAAc,KACnBx+B,KAAK2a,WAAa,KAClB3a,KAAKwb,SAAW,KAGhBxb,KAAKy+B,oBAAsB,KAG3Bz+B,KAAKwM,SAAW,wyGAyDpB,CAEA,YAAM5L,GAEFZ,KAAKmvB,aAAe,IAAI9vB,EAAK,CACzBmN,SAAU,60IAqEVE,MAAO1M,KAAK0M,QAIhB1M,KAAKw+B,YAAc,IAAIn/B,EAAK,CACxBmN,SAAU,2LAKVE,MAAO1M,KAAK0M,QAIhB,MAAM6N,EAAmB,IAAImkB,GAAa,CACtCtsB,OAAQ,CAAEusB,IAAK3+B,KAAK0M,MAAMC,IAAI,MAAO6D,KAAM,MAE/CxQ,KAAK2a,WAAa,IAAIT,GAAU,CAC5BhI,WAAYqI,EACZJ,oBAAqB,CAAC,OACtBC,QAAS,CACL,CAAElS,IAAK,KAAMvI,MAAO,YAAa0a,UAAW,WAAYC,UAAU,GAClE,CAAEpS,IAAK,QAASvI,MAAO,QAAS0a,UAAW,SAC3C,CAAEnS,IAAK,eAAgBvI,MAAO,cAKtC,MAAM2b,EAAiB,IAAIsjB,GAAW,CAClCxsB,OAAQ,CAAEusB,IAAK3+B,KAAK0M,MAAMC,IAAI,MAAO6D,KAAM,MAE/CxQ,KAAKwb,SAAW,IAAItB,GAAU,CAC1BhI,WAAYoJ,EACZnB,oBAAqB,CAAC,OACtBC,QAAS,CACL,CAAElS,IAAK,mBAAoBvI,MAAO,UAAW2a,UAAU,GACvD,CAAEpS,IAAK,OAAQvI,MAAO,OAAQ0a,UAAW,SACzC,CAAEnS,IAAK,UAAWvI,MAAO,cAKjCK,KAAKymB,QAAU,IAAIC,EAAQ,CACvBhB,KAAM,CACF8J,SAAYxvB,KAAKmvB,aACjB0P,QAAW7+B,KAAKw+B,YAChB3Y,OAAU7lB,KAAK2a,WACfmL,KAAQ9lB,KAAKwb,UAEjBmL,UAAW,WACXxkB,YAAa,qBAEjBnC,KAAKoC,SAASpC,KAAKymB,SAGnB,MAAMqY,EAAmB,CACrB,CAAEn/B,MAAO,UAAWE,OAAQ,cAAeD,KAAM,uBAGjDI,KAAK0M,MAAMqyB,WAAa/+B,KAAK0M,MAAMqyB,aACnCD,EAAiB72B,KAAK,CAClBtI,MAAO,aACPE,OAAQ,aACRD,KAAM,cACN2T,MAAO,gBAIXvT,KAAK0M,MAAMsyB,UAAYh/B,KAAK0M,MAAMsyB,YAClCF,EAAiB72B,KAAK,CAClBtI,MAAO,YACPE,OAAQ,YACRD,KAAM,kBACN2T,MAAO,iBAIf,MAAM0rB,EAAU,IAAI9iB,EAAY,CAC5Bha,YAAa,mBACbpC,UAAW,yCACXqc,QAASpc,KAAK0M,MACdhF,OAAQ,CACJ9H,KAAM,yBACNwJ,MAAO01B,KAGf9+B,KAAKoC,SAAS68B,SAERj/B,KAAK0M,MAAM4F,MAAM,CAAEF,OAAQ,CAAE8sB,MAAO,WAG9C,CAEA,oBAAMz8B,SACIzC,KAAKm/B,gBACf,CAEA,oBAAMA,GACGn/B,KAAK0M,QAGV1M,KAAK0M,MAAMkS,EAAE4e,iBAAmBx9B,KAAK0M,MAAM0yB,oBAAsBp/B,KAAK0M,MAAM0yB,sBAAwB,eACpGp/B,KAAK0M,MAAMkS,EAAE8e,WAAa19B,KAAK0M,MAAM2yB,cAAgBr/B,KAAK0M,MAAM2yB,gBAAkB,qBAClFr/B,KAAK0M,MAAMkS,EAAE0gB,kBAAoBt/B,KAAK0M,MAAM6yB,qBAAuBv/B,KAAK0M,MAAM6yB,uBAAyB,MAC3G,CAEA,oBAAMC,GACF,GAAKx/B,KAAK0M,OAAOC,IAAI,MAErB,IACQ3M,KAAK0M,MAAM+yB,yBACLz/B,KAAK0M,MAAM+yB,0BAEfz/B,KAAKm/B,gBACf,OAAS77B,GACLC,QAAQD,MAAM,8BAA+BA,EACjD,CACJ,CAEA,wBAAMo8B,SACI1/B,KAAK0M,MAAM4F,MAAM,CAAEF,OAAQ,CAAE8sB,MAAO,WAC9C,CAEA,uBAAMS,GACF,GAAI5yB,QAAQ,6CACR,IACI,MAAMpK,QAAiB3C,KAAK0M,MAAMkzB,SAC9Bj9B,EAASI,eACH/C,KAAKw/B,uBACLx/B,KAAKqD,SACXrD,KAAKwF,KAAK,gBAAiB,CAAEm5B,IAAK3+B,KAAK0M,SAEvC/G,MAAM,0BAA4BhD,EAASK,MAAMM,OAAS,iBAElE,OAASA,GACLC,QAAQD,MAAM,wBAAyBA,GACvCqC,MAAM,yBAA2BrC,EAAM+K,QAC3C,CAER,CAEA,sBAAMwxB,GACF,MAAMC,QAAkBhzB,EAAOyD,SAAS,CACpC9Q,MAAO,YACP8c,WAAYwjB,GAASC,QAGzB,GAAIF,EACA,IACI,MAAMn9B,QAAiB3C,KAAK0M,MAAMszB,MAAMF,EAAUG,OAAS,GACvDt9B,EAASI,QACT/C,KAAKwF,KAAK,cAAe,CAAEm5B,IAAK3+B,KAAK0M,MAAOwzB,SAAUv9B,EAASu9B,WAE/Dv6B,MAAM,yBAA2BhD,EAASK,MAAMM,OAAS,iBAEjE,OAASA,GACLC,QAAQD,MAAM,uBAAwBA,GACtCqC,MAAM,wBAA0BrC,EAAM+K,QAC1C,CAER,CAEA,gBAAA8xB,GACQngC,KAAKy+B,qBAAqB2B,cAAcpgC,KAAKy+B,qBAE7Cz+B,KAAK0M,OAAO5D,UAAY9I,KAAK0M,MAAM5D,aACnC9I,KAAKy+B,oBAAsB4B,YAAYxtB,UACnC,UACU7S,KAAKw/B,iBACPx/B,KAAK+J,mBACC/J,KAAKqD,QAEnB,OAASC,GACLC,QAAQD,MAAM,uBAAwBA,EAC1C,GACD,KAEX,CAEA,eAAAg9B,GACQtgC,KAAKy+B,sBACL2B,cAAcpgC,KAAKy+B,qBACnBz+B,KAAKy+B,oBAAsB,KAEnC,CAEA,eAAM8B,GACFvgC,KAAKsgC,wBACC9gC,MAAM+gC,WAChB,CAGA,iBAAa7b,CAAKia,EAAKp/B,EAAU,IAC7B,MAAM4I,EAAO,IAAIm2B,eAAe,CAAE5xB,MAAOiyB,IAEzC,aAAa7xB,EAAOsG,WAAW,CAC3B3T,MAAO,uDAAuDk/B,EAAIhyB,IAAI,QACtE0G,KAAMlL,EACNqI,KAAM,KACN4nB,YAAY,EACZ9kB,QAAS,CAAC,CAAE1C,KAAM,QAAS2C,MAAO,gBAAiBC,SAAS,IAC5DgtB,OAAQ,IAAMr4B,EAAKm4B,qBAChB/gC,GAEX,EAGJg/B,GAAIrhB,WAAaohB,eCtWjB,MAAMmC,4BAA4BphC,EAC9B,WAAAC,CAAYC,EAAU,IAClBC,MAAM,CACFO,UAAW,4BACRR,IAGPS,KAAKwM,SAAW,wGAGpB,CAEA,YAAM5L,GACFZ,KAAKqrB,MAAQ,IAAInnB,EAAa,CAC1B/B,YAAa,0BACb1C,MAAO,0DACP8C,SAAU,qBACVb,OAAQ,IACRR,YAAa,QACbqpB,SAAU,gBACVjpB,QAAS,SACTC,UAAW,MACXW,eAAe,EACfiC,MAAO,CACHxE,MAAO,QACPyE,aAAa,GAEjBC,QAAS,CACLC,EAAG,cAIXtE,KAAKoC,SAASpC,KAAKqrB,MACvB,EAGW,MAAMqV,sBAAsBh9B,EACvC,WAAApE,CAAYC,EAAU,IAClBC,MAAM,CACFC,MAAO,kBACPM,UAAW,qBACRR,IAGPS,KAAK2D,UAAY,kBACjB3D,KAAK4D,aAAe,6CACpB5D,KAAK6D,4BAAA,IAAkBC,MAAOC,iBAC9B/D,KAAKy+B,oBAAsB,KAC3Bz+B,KAAK2gC,YAAc,IAEnB3gC,KAAKwM,SAAW,88SAuJpB,CAEA,YAAM5L,GAEFZ,KAAK4gC,SAAW,IAAIC,GAEpB7gC,KAAK8gC,aAAe,IAAI/E,aAAa,CACjC55B,YAAa,YACbuK,MAAO1M,KAAK4gC,WAEhB5gC,KAAKoC,SAASpC,KAAK8gC,cAEnB9gC,KAAK+gC,cAAgB,IAAIzE,cAAc,CACnCn6B,YAAa,aACbuK,MAAO1M,KAAK4gC,WAEhB5gC,KAAKoC,SAASpC,KAAK+gC,eAEnB/gC,KAAKghC,mBAAqB,IAAIlgC,EAAuB,CACjDqB,YAAa,uBACbvC,KAAM,eACNH,MAAO,iBACPsB,SAAU,8BACVG,YAAa,OACbG,MAAO,CAAC,kBACRC,QAAS,SACTC,UAAW,OACXG,OAAQ,GACRO,cAAc,EACdD,cAAc,EACdE,eAAe,IAEnBlC,KAAKoC,SAASpC,KAAKghC,oBAEnBhhC,KAAKihC,gBAAkB,IAAIngC,EAAuB,CAC9CqB,YAAa,oBACbvC,KAAM,4BACNH,MAAO,cACPsB,SAAU,8BACVG,YAAa,OACbG,MAAO,CAAC,eACRC,QAAS,SACTC,UAAW,OACXG,OAAQ,GACRO,cAAc,EACdD,cAAc,EACdE,eAAe,IAEnBlC,KAAKoC,SAASpC,KAAKihC,iBAEnBjhC,KAAKkhC,iBAAmB,IAAIhnB,GAAU,CAClC/X,YAAa,qBACbmb,WAAY6jB,GACZC,iBAAkB,CACd5wB,KAAM,EACN+E,KAAM,WACNtS,OAAQ,WAEZ4a,YAAY,EACZC,YAAY,EACZC,WAAW,EACXyK,SAAU8V,eACV+C,gBAAiB,CAAC,UAClB7jB,kBAAmB,CACf/d,MAAO,cACP+Q,KAAM,KACN4nB,YAAY,GAEhB/Z,aAAc,CACVC,SAAS,EACTE,OAAO,EACPhO,KAAM,MAEV4J,QAAS,CACL,CACIlS,IAAK,KACLvI,MAAO,MACP6M,SAAU,8QAKd,CACItE,IAAK,YACLvI,MAAO,SACP6M,SAAU,gJAId,CACItE,IAAK,SACLvI,MAAO,QACP0a,UAAW,CAAC1J,EAAOyL,KACf,MAAMuiB,EAAMviB,EAAQklB,IAGpB,MAAO,sBAFY3C,EAAIS,oBAAsBT,EAAIS,sBAAwB,6BAC5DT,EAAIU,cAAgBV,EAAIU,gBAAkB,2BACiB1uB,EAAMgF,yBAGtF,CACIzN,IAAK,UACLvI,MAAO,UACP0a,UAAW,eAIvBra,KAAKoC,SAASpC,KAAKkhC,kBAEnBlhC,KAAKuhC,iBAAmB,IAAIrnB,GAAU,CAClC/X,YAAa,qBACbmb,WAAY6jB,GACZC,iBAAkB,CACd5wB,KAAM,EACN+E,KAAM,WACNtS,OAAQ,UACRu+B,gBAAgB,GAEpB3jB,YAAY,EACZC,YAAY,EACZC,WAAW,EACXH,YAAY,EACZ0D,iBAAkB,MAClBC,aAAc,CACV,CACI3hB,KAAM,mBACND,MAAO,cACPE,OAAQ,gBAGhB2oB,SAAU8V,eACV+C,gBAAiB,CAAC,UAClB7jB,kBAAmB,CACf/d,MAAO,cACP+Q,KAAM,KACN4nB,YAAY,GAEhB/Z,aAAc,CACVC,SAAS,EACTE,OAAO,EACPhO,KAAM,MAEV4J,QAAS,CACL,CACIlS,IAAK,KACLvI,MAAO,MACP6M,SAAU,8QAKd,CACItE,IAAK,WACLvI,MAAO,WACP0a,UAAW,CAAC1J,EAAQ,IAET,sBADOA,GAAS,EAAI,YAAcA,GAAS,EAAI,aAAe,mBAC9BA,YAG/C,CACIzI,IAAK,WACLvI,MAAO,SACP0a,UAAW,eAIvBra,KAAKuhC,iBAAiBE,GAAG,2BAA4B5uB,MAAOhT,EAAQ2E,EAAOC,KACvE,MAAM2E,EAAQpJ,KAAKuhC,iBAAiBpR,yBAC9B/qB,QAAQgrB,IAAIhnB,EAAMR,OAAY+V,EAAKjS,MAAMkzB,WAC/C5/B,KAAK4C,SAASwL,MAAMrL,QAAQ,+BAC5B/C,KAAKuhC,iBAAiBrvB,WAAWI,UAErCtS,KAAKoC,SAASpC,KAAKuhC,kBAEnBvhC,KAAK0hC,gBAAkB,IAAIxnB,GAAU,CACjC/X,YAAa,oBACbmb,WAAY6jB,GACZC,iBAAkB,CACd5wB,KAAM,EACN+E,KAAM,eACNtS,OAAQ,UAEZ4a,YAAY,EACZC,YAAY,EACZC,WAAW,EACXyK,SAAU8V,eACV9gB,kBAAmB,CACf/d,MAAO,cACP+Q,KAAM,KACN4nB,YAAY,GAEhBiJ,gBAAiB,CAAC,UAClBhjB,aAAc,CACVC,SAAS,EACTE,OAAO,EACPhO,KAAM,MAEV4J,QAAS,CACL,CACIlS,IAAK,KACLvI,MAAO,MACP6M,SAAU,8QAKd,CACItE,IAAK,aACLvI,MAAO,QACP6M,SAAU,qJAId,CACItE,IAAK,WACLvI,MAAO,SACP0a,UAAW,eAIvBra,KAAKoC,SAASpC,KAAK0hC,iBAEnB1hC,KAAK2hC,mBAAqB,IAAIznB,GAAU,CACpC/X,YAAa,uBACbmb,WAAY6jB,GACZC,iBAAkB,CACd5wB,KAAM,EACN+E,KAAM,SACNisB,gBAAgB,EAChBv+B,OAAQ,WAEZ4a,YAAY,EACZC,YAAY,EACZC,WAAW,EACXyK,SAAU8V,eACV+C,gBAAiB,CAAC,UAClB7jB,kBAAmB,CACf/d,MAAO,cACP+Q,KAAM,KACN4nB,YAAY,GAEhB/Z,aAAc,CACVC,SAAS,EACTE,OAAO,EACPhO,KAAM,MAEV4J,QAAS,CACL,CACIlS,IAAK,KACLvI,MAAO,MACP0a,UAAW,uBAEf,CACInS,IAAK,SACLvI,MAAO,gBACP0a,UAAW,YAEf,CACInS,IAAK,UACLvI,MAAO,UACP0a,UAAW,UAGnBuD,YAAY,EACZ0D,iBAAkB,MAClBC,aAAc,CACV,CACI3hB,KAAM,mBACND,MAAO,cACPE,OAAQ,kBAIpBG,KAAK2hC,mBAAmBF,GAAG,2BAA4B5uB,MAAOhT,EAAQ2E,EAAOC,KACzE,MAAM2E,EAAQpJ,KAAK2hC,mBAAmBxR,yBAChC/qB,QAAQgrB,IAAIhnB,EAAMR,OAAY+V,EAAKjS,MAAMkzB,WAC/C5/B,KAAK4C,SAASwL,MAAMrL,QAAQ,+BAC5B/C,KAAK2hC,mBAAmBzvB,WAAWI,UAEvCtS,KAAKoC,SAASpC,KAAK2hC,oBAEnB3hC,KAAK4hC,aAAe,IAAI1nB,GAAU,CAC9B/X,YAAa,eACbmb,WAAYukB,GACZhkB,YAAY,EACZC,YAAY,EACZC,WAAW,EACXM,aAAc,CACVC,SAAS,EACTE,OAAO,EACPhO,KAAM,MAEV4J,QAAS,CACL,CACIlS,IAAK,YACLvI,MAAO,SACP0a,UAAW,uBAEf,CACInS,IAAK,QACLvI,MAAO,SACP0a,UAAY1J,IACR,MAAMmxB,GAAoB,IAAVnxB,EAIhB,MAAO,sBAHYmxB,EAAU,aAAe,0BAC/BA,EAAU,uBAAyB,iCACnCA,EAAU,QAAU,kBAIzC,CACI55B,IAAK,iBACLvI,MAAO,YACP0a,UAAY1J,IACR,IAAKA,EAAO,MAAO,QACnB,MAAMoxB,EAAgB,IAAIj+B,KAAK6M,GAEzBqxB,qBADUl+B,KACKi+B,EACfE,EAAcx2B,KAAKy2B,MAAMF,EAAS,KACxC,OAAIC,EAAc,GAAW,GAAGA,SAC5BA,EAAc,KAAa,GAAGx2B,KAAKy2B,MAAMD,EAAc,WACpD,GAAGx2B,KAAKy2B,MAAMD,EAAc,kBAKnDjiC,KAAKoC,SAASpC,KAAK4hC,oBAEb5hC,KAAKmiC,aAEf,CAGA,gBAAAhC,GACQngC,KAAKy+B,qBACL2B,cAAcpgC,KAAKy+B,qBAGnBz+B,KAAK2gC,YAAc,IACnB3gC,KAAKy+B,oBAAsB4B,YAAYxtB,gBAC7B7S,KAAKmiC,eACZniC,KAAK2gC,aAEhB,CAEA,iBAAMwB,GACF,IACI,MAAMC,EAAQ,CACVpiC,KAAK4gC,SAAStuB,SAGdtS,KAAKghC,oBACLoB,EAAMn6B,KAAKjI,KAAKghC,mBAAmB/7B,WAEnCjF,KAAKihC,iBACLmB,EAAMn6B,KAAKjI,KAAKihC,gBAAgBh8B,WAEhCjF,KAAKkhC,kBAAkBhvB,YAAYI,OACnC8vB,EAAMn6B,KAAKjI,KAAKkhC,iBAAiBhvB,WAAWI,SAE5CtS,KAAKuhC,kBAAkBrvB,YAAYI,OACnC8vB,EAAMn6B,KAAKjI,KAAKuhC,iBAAiBrvB,WAAWI,SAE5CtS,KAAK0hC,iBAAiBxvB,YAAYI,OAClC8vB,EAAMn6B,KAAKjI,KAAK0hC,gBAAgBxvB,WAAWI,SAE3CtS,KAAK2hC,oBAAoBzvB,YAAYI,OACrC8vB,EAAMn6B,KAAKjI,KAAK2hC,mBAAmBzvB,WAAWI,SAE9CtS,KAAK4hC,cAAc1vB,YAAYI,OAC/B8vB,EAAMn6B,KAAKjI,KAAK4hC,aAAa1vB,WAAWI,eAGtClN,QAAQgrB,IAAIgS,GAElBpiC,KAAK6D,4BAAA,IAAkBC,MAAOC,iBAC9B/D,KAAKqiC,uBAET,OAAS/+B,GACLC,QAAQD,MAAM,oCAAqCA,EACvD,CACJ,CAEA,qBAAA++B,GACI,MAAMC,EAAmBtiC,KAAKyE,SAASG,cAAc,cACrD,GAAI09B,EAAkB,CAClB,MAAMC,EAAiC,IAArBviC,KAAK2gC,YAAoB,MAAW3gC,KAAK2gC,YAAc,IAAtB,IACnD2B,EAAiB18B,UAAY,+FAET28B,qBAA6BviC,KAAK6D,2BAE1D,CACJ,CAEA,sBAAI2+B,GACA,OAAOxiC,KAAK2gC,YAAc,GAC9B,CAEA,oBAAI8B,GACA,OAA4B,IAArBziC,KAAK2gC,YAAoB,MAAQ,GAAG3gC,KAAKwiC,qBACpD,CAGA,wBAAMj+B,CAAmBC,EAAOC,GAC5B,IACI,MAAM7E,EAAO6E,EAAQG,cAAc,KACnChF,GAAMiF,UAAUC,IAAI,YACpBL,EAAQM,UAAW,QAEb/E,KAAKmiC,aAEf,OAAS7+B,GACLC,QAAQD,MAAM,oCAAqCA,EACvD,CAAA,QACI,MAAM1D,EAAO6E,EAAQG,cAAc,KACnChF,GAAMiF,UAAUiB,OAAO,YACvBrB,EAAQM,UAAW,CACvB,CACJ,CAEA,4BAAM29B,CAAuBl+B,EAAOC,GAChC,MAAMk+B,EAAqD,IAA9CC,SAASn+B,EAAQo+B,aAAa,cAC3C7iC,KAAK2gC,YAAcgC,EACnB3iC,KAAKmgC,mBAEL,MAAM2C,EAAoB,IAATH,EAAa,MAAWA,EAAO,IAAV,IACtC3iC,KAAK4C,SAASwL,MAAMrL,QAAQ,uBAAuB+/B,IACvD,CAEA,wBAAMC,SACIj2B,EAAOsd,UAAU,CACnB3qB,MAAO,cACP4O,QAAS,yCACTvG,KAAM,QAEd,CAGA,0BAAMk7B,CAAqBx+B,EAAOC,SACNqI,EAAOm2B,YAAY,CACvCxjC,MAAO,iBACP4O,QAAS,iFACT6Z,YAAa,WACbD,aAAc,uBAIRjoB,KAAKkjC,iBAAiBz+B,EAAS,IAAM85B,GAAI4E,OAAQ,gCAE/D,CAEA,yBAAMC,CAAoB5+B,EAAOC,SACLqI,EAAOm2B,YAAY,CACvCxjC,MAAO,gBACP4O,QAAS,wEACT6Z,YAAa,YACbD,aAAc,uBAIRjoB,KAAKkjC,iBAAiBz+B,EAAS,IAAM85B,GAAI8E,QAAS,kCAEhE,CAEA,wBAAMC,CAAmB9+B,EAAOC,GAC5B,MACM8+B,EAAiB,CACnB,CAAE5yB,MAAO,GAAIhR,MAAO,oBAFPK,KAAK+gC,eAAexE,QAAQY,eAAiB,IAG9Cv0B,IAAIsM,IAAA,CAASvE,MAAOuE,EAAGW,QAASlW,MAAOuV,EAAGW,YAGpDoU,QAAend,EAAOyD,SAAS,CACjC9Q,MAAO,mBACP8c,WAAY,CACR/N,OAAQ,CACJ,CACIwB,KAAM,UACNlI,KAAM,SACNnI,MAAO,UACPJ,QAASgkC,EACT5yB,MAAO,GACPc,KAAM,+DAMlBwY,SACMjqB,KAAKkjC,iBACPz+B,EACA,IAAM85B,GAAIiF,WAAWvZ,EAAOpU,SAAW,MACtClT,IACG,MAAM8gC,EAAQ9gC,EAASK,KAAKygC,OAAS,EAErC,MAAO,WAAWA,cAA4B,IAAVA,EAAc,IAAM,KADpCxZ,EAAOpU,QAAU,kBAAkBoU,EAAOpU,WAAa,MAK3F,CAEA,0BAAM6tB,CAAqBl/B,EAAOC,GAC9B,MACM8+B,GADWvjC,KAAK+gC,eAAexE,QAAQY,eAAiB,IAC9Bv0B,IAAIsM,IAAA,CAASvE,MAAOuE,EAAGW,QAASlW,MAAOuV,EAAGW,WAEpEoU,QAAend,EAAOyD,SAAS,CACjC9Q,MAAO,gBACP8c,WAAY,CACR/N,OAAQ,CACJ,CACIwB,KAAM,UACNlI,KAAM,SACNnI,MAAO,UACPJ,QAASgkC,EACT/xB,UAAU,EACVC,KAAM,oCAMlBwY,SACMjqB,KAAKkjC,iBACPz+B,EACA,IAAM85B,GAAIoF,aAAa1Z,EAAOpU,SAC9B,YAAYoU,EAAOpU,iCAG/B,CAEA,uBAAM+tB,CAAkBp/B,EAAOC,GAC3B,MAAMwlB,QAAend,EAAOyD,SAAS,CACjC9Q,MAAO,iBACP8c,WAAY,CACR/N,OAAQ,CACJ,CACIwB,KAAM,WACNlI,KAAM,SACNnI,MAAO,WACPgR,MAAO,GACPa,UAAU,EACVC,KAAM,8CAMlBwY,SACMjqB,KAAKkjC,iBACPz+B,EACA,IAAM85B,GAAIsF,UAAU5Z,EAAO6Z,UAC1BnhC,GAEU,UADOA,EAASK,KAAKygC,OAAS,gBAKrD,CAEA,8BAAMM,CAAyBv/B,EAAOC,SACVqI,EAAOm2B,YAAY,CACvCxjC,MAAO,oBACP4O,QAAS,mFACT6Z,YAAa,UACbD,aAAc,uBAIRjoB,KAAKkjC,iBACPz+B,EACA,IAAM85B,GAAIyF,iBACTrhC,GAEU,cADOA,EAASK,KAAKygC,OAAS,iBAKrD,CAEA,6BAAMQ,GACF,MAAMha,QAAend,EAAOyD,SAAS,CACjC9Q,MAAO,mCACP8c,WAAY2nB,GAAeC,YAG/B,GAAIla,EACA,IACI,MAAMma,QAAwBC,GAAUF,UACpCla,EAAOqa,QACP,CAAA,EACAra,EAAOsa,SAGPH,EAAgBrhC,SAChB/C,KAAK4C,SAASwL,MAAMrL,QAAQ,sBAAsBknB,EAAOqa,oCACnDtkC,KAAKmiC,eAEXniC,KAAK4C,SAASwL,MAAM9K,MAAM8gC,EAAgBphC,MAAMM,OAAS,8BAEjE,OAASA,GACLC,QAAQD,MAAM,+BAAgCA,GAC9CtD,KAAK4C,SAASwL,MAAM9K,MAAM,+BAAiCA,EAAM+K,QACrE,CAER,CAGA,sBAAM60B,CAAiBz+B,EAAS+/B,EAAUC,GACtC,IACIhgC,EAAQM,UAAW,EACnB,MAAMnF,EAAO6E,EAAQG,cAAc,KACnChF,GAAMiF,UAAUC,IAAI,YAEpB,MAAMmlB,QAAeua,IACrB,GAAIva,EAAOlnB,SAAWknB,EAAOjnB,MAAMC,OAAQ,CACvC,MAAMoL,EAAoC,mBAAnBo2B,EACjBA,EAAexa,GACfwa,EACNzkC,KAAK4C,SAASwL,MAAMrL,QAAQsL,SACtBrO,KAAKmiC,aACf,MACIniC,KAAK4C,SAASwL,MAAM9K,MAAM2mB,EAAOjnB,MAAMM,OAAS,mBAExD,OAASA,GACLC,QAAQD,MAAM,qBAAsBA,GACpCtD,KAAK4C,SAASwL,MAAM9K,MAAM,UAAYA,EAAM+K,QAChD,CAAA,QACI5J,EAAQM,UAAW,EACnB,MAAMnF,EAAO6E,EAAQG,cAAc,KACnChF,GAAMiF,UAAUiB,OAAO,WAC3B,CACJ,CAEA,aAAM4+B,GAEF1kC,KAAKmgC,kBACT,CAEA,YAAMwE,GACE3kC,KAAKy+B,sBACL2B,cAAcpgC,KAAKy+B,qBACnBz+B,KAAKy+B,oBAAsB,KAEnC,CAGA,sBAAMn4B,GACF,aAAatG,KAAKmiC,aACtB,CAEA,QAAA17B,GACI,OAAOzG,KAAK8gC,cAAc7gC,OAAS,CAAA,CACvC,CAEA,SAAA2kC,GACI,OAAO5kC,KAAK+gC,eAAexE,QAAU,CAAA,CACzC,CAEA,iCAAMsI,GACF,MAAMC,EAAY,IAAIrE,0BAChB3zB,EAAOsG,WAAW,CACpB3T,MAAO,yDACP4T,KAAMyxB,EACNt0B,KAAM,KACN4nB,YAAY,EACZ9kB,QAAS,CACL,CAAE1C,KAAM,QAAS2C,MAAO,gBAAiBC,SAAS,KAG9D,CAEA,yBAAMuxB,GACF,MAAMC,EAAU,IAAI7D,GAAQ,CACxB/uB,OAAQ,CACJmD,KAAM,WACN/E,KAAM,MAIRrI,EAAO,IAAI+R,GAAU,CACvBhI,WAAY8yB,EACZC,cAAc,EACd7qB,QAAS,CACL,CACIlS,IAAK,KACLvI,MAAO,MACP6M,SAAU,+MAKd,CACItE,IAAK,UACLvI,MAAO,UACP0a,UAAW,SAEf,CACInS,IAAK,SACLvI,MAAO,SACP0a,UAAW,CAAC1J,EAAOyL,KACf,MAAMuiB,EAAMviB,EAAQklB,IAGpB,MAAO,sBAFY3C,EAAIS,oBAAsBT,EAAIS,sBAAwB,6BAC5DT,EAAIU,cAAgBV,EAAIU,gBAAkB,2BACiB1uB,GAAOgF,eAAiB,qBAGxG,CACIzN,IAAK,UACLvI,MAAO,UACP0a,UAAW,YAEf,CACInS,IAAK,cACLvI,MAAO,WACP0a,UAAW,YAEf,CACInS,IAAK,cACLvI,MAAO,WACP0a,UAAW,aAGnBsD,QAAS,CACL,CACIzV,IAAK,kBACLvI,MAAO,WACPmI,KAAM,QAEV,CACII,IAAK,SACLvI,MAAO,SACPmI,KAAM,SACNvI,QAAS,CACL,CAAEI,MAAO,UAAWgR,MAAO,WAC3B,CAAEhR,MAAO,UAAWgR,MAAO,WAC3B,CAAEhR,MAAO,YAAagR,MAAO,aAC7B,CAAEhR,MAAO,SAAUgR,MAAO,UAC1B,CAAEhR,MAAO,WAAYgR,MAAO,YAC5B,CAAEhR,MAAO,UAAWgR,MAAO,aAGnC,CACIzI,IAAK,UACLvI,MAAO,UACPmI,KAAM,SAGd0gB,SAAU8V,eACV9gB,kBAAmB,CACf/d,MAAO,cACP+Q,KAAM,KACN4nB,YAAY,GAEhBva,YAAY,EACZC,YAAY,EACZC,WAAW,EACXM,aAAc,CACVC,SAAS,EACTE,OAAO,EACPC,YAAY,WAIdze,KAAK4C,SAASwQ,WAAW,CAC3B3T,MAAO,2CACP4T,KAAMlL,EACNqI,KAAM,KACN4nB,YAAY,EACZ9kB,QAAS,CACL,CAAE1C,KAAM,QAAS2C,MAAO,gBAAiBC,SAAS,KAG9D,ECr9BW,MAAM0xB,wBAAwB7lC,EAC3C,WAAAC,CAAYC,EAAU,IACpBC,MAAM,IACDD,EACHQ,UAAW,2BAGbC,KAAKmlC,KAAO5lC,EAAQ4lC,MAAQ,KAC5BnlC,KAAKolC,KAAO,GACZplC,KAAKqlC,QAAU,IACjB,CAEA,iBAAM1kC,GACJ,MAAO,smNA4KT,CAEA,YAAMC,GACAZ,KAAKmlC,aACDnlC,KAAKslC,wBACLtlC,KAAKulC,qBACLvlC,KAAKwlC,kBAEf,CAEA,qBAAMF,GACCtlC,KAAKmlC,OAGVnlC,KAAKmlC,KAAK3H,iBAAmBx9B,KAAKo/B,oBAAoBp/B,KAAKmlC,KAAKliC,QAChEjD,KAAKmlC,KAAKzH,WAAa19B,KAAKq/B,cAAcr/B,KAAKmlC,KAAKliC,QAGhDjD,KAAKmlC,KAAKniC,MAAkC,iBAAnBhD,KAAKmlC,KAAKniC,OACrChD,KAAKmlC,KAAKM,cAAgBrY,KAAKC,UAAUrtB,KAAKmlC,KAAKniC,KAAM,KAAM,IAI7DhD,KAAKmlC,KAAK1uB,UACZzW,KAAKmlC,KAAKO,aAAmC,IAApB1lC,KAAKmlC,KAAK1uB,QAAiB3S,KAAK6hC,MAAQ,cAAgB,cAErF,CAEA,mBAAAvG,CAAoBn8B,GASlB,MARgB,CACd+4B,QAAW,aACXC,QAAW,aACXC,UAAa,UACb54B,MAAS,YACTsiC,UAAa,eACbC,QAAW,cAEE5iC,IAAW,cAC5B,CAEA,aAAAo8B,CAAcp8B,GASZ,MARc,CACZ+4B,QAAW,eACXC,QAAW,kBACXC,UAAa,kBACb54B,MAAS,eACTsiC,UAAa,cACbC,QAAW,YAEA5iC,IAAW,oBAC1B,CAEA,gBAAA6iC,CAAiB5U,GAOf,MANgB,CACd6U,MAAS,YACTre,KAAQ,UACRoS,QAAW,UACXx2B,MAAS,UAEI4tB,IAAU,WAC3B,CAEA,kBAAMqU,GACJ,GAAKvlC,KAAKmlC,MAAMnzB,GAEhB,IACE,MAAMrP,QAAiB3C,KAAK4C,SAASC,KAAKC,IAAI,cAAc9C,KAAKmlC,KAAKnzB,WAClErP,EAASI,SAAWJ,EAASK,KAAKC,OACpCjD,KAAKolC,KAAOziC,EAASK,KAAKA,KAAK4F,IAAIo9B,IAAA,IAC9BA,EACHC,WAAYjmC,KAAK8lC,iBAAiBE,EAAI9U,UAGxClxB,KAAKolC,KAAO,EAEhB,OAAS9hC,GACPC,QAAQD,MAAM,4BAA6BA,GAC3CtD,KAAKolC,KAAO,EACd,CACF,CAEA,qBAAMI,GACJ,GAAKxlC,KAAKmlC,MAAMnzB,GAEhB,IACE,MAAMrP,QAAiB3C,KAAK4C,SAASC,KAAKC,IAAI,cAAc9C,KAAKmlC,KAAKnzB,cAClErP,EAASI,SAAWJ,EAASK,KAAKC,SACpCjD,KAAKqlC,QAAU1iC,EAASK,KAAKA,KAEjC,OAASM,GACPC,QAAQD,MAAM,+BAAgCA,GAC9CtD,KAAKqlC,QAAU,IACjB,CACF,CAEA,aAAMa,CAAQf,GACZnlC,KAAKmlC,KAAOA,QACNnlC,KAAKslC,wBACLtlC,KAAKulC,qBACLvlC,KAAKwlC,kBAEPxlC,KAAK+J,mBACD/J,KAAKqD,QAEf,CAEA,yBAAM8iC,CAAoBtmC,EAAQ2E,EAAOC,GACvC,GAAKzE,KAAKmlC,MAAMnzB,GAEhB,IACEvN,EAAQM,UAAW,EACnB,MAAMnF,EAAO6E,EAAQG,cAAc,KAC/BhF,GAAMA,EAAKiF,UAAUC,IAAI,kBAEvB9E,KAAKulC,qBACLvlC,KAAKqD,QAEb,OAASC,GACPC,QAAQD,MAAM,0BAA2BA,GACzCtD,KAAKkqB,UAAU,2BAA6B5mB,EAAM+K,QACpD,CAAA,QACE5J,EAAQM,UAAW,EACnB,MAAMnF,EAAO6E,EAAQG,cAAc,KAC/BhF,GAAMA,EAAKiF,UAAUiB,OAAO,WAClC,CACF,CAGA,iBAAa4e,CAAKygB,EAAM5lC,EAAU,IAChC,MAAM4I,EAAO,IAAI+8B,gBAAgB,CAAEC,eAC7Bh9B,EAAKvH,SAEX,MAAM0S,EAAU,GA2GhB,MAxGI,CAAC,UAAW,WAAWoE,SAASytB,EAAKliC,SACvCqQ,EAAQrL,KAAK,CACX2I,KAAM,cACN2C,MAAO,qBACP1T,OAAQgT,UACN,GAAI9F,QAAQ,8CACV,IACE,MAAMpK,QAAiBwF,EAAKvF,SAASC,KAAKuO,KAAK,cAAc+zB,EAAKnzB,aAClE,GAAIrP,EAASI,SAAWJ,EAASK,KAAKC,OAEpC,OADAkF,EAAKi+B,YAAY,+BACV,CAAEvmC,OAAQ,YAAaslC,QAE9Bh9B,EAAK+hB,UAAUvnB,EAASK,KAAKM,OAAS,wBAE1C,OAASA,GACP6E,EAAK+hB,UAAU,0BAA4B5mB,EAAM+K,QACnD,CAEF,OAAO,QAKO,UAAhB82B,EAAKliC,QACPqQ,EAAQrL,KAAK,CACX2I,KAAM,aACN2C,MAAO,sBACP1T,OAAQgT,UACN,IACE,MAAMlQ,QAAiBwF,EAAKvF,SAASC,KAAKuO,KAAK,cAAc+zB,EAAKnzB,YAClE,GAAIrP,EAASI,SAAWJ,EAASK,KAAKC,OAEpC,OADAkF,EAAKi+B,YAAY,yBACV,CAAEvmC,OAAQ,UAAWslC,QAE5Bh9B,EAAK+hB,UAAUvnB,EAASK,KAAKM,OAAS,uBAE1C,OAASA,GACP6E,EAAK+hB,UAAU,yBAA2B5mB,EAAM+K,QAClD,CACA,OAAO,QAKbiF,EAAQrL,KAAK,CACX2I,KAAM,aACN2C,MAAO,mBACP1T,OAAQgT,UACN,IACE,MAAMlQ,QAAiBwF,EAAKvF,SAASC,KAAKuO,KAAK,cAAc+zB,EAAKnzB,YAClE,GAAIrP,EAASI,SAAWJ,EAASK,KAAKC,OAEpC,OADAkF,EAAKi+B,YAAY,4BACV,CAAEvmC,OAAQ,SAAUwmC,aAAclB,EAAMmB,QAAS3jC,EAASK,KAAKA,MAEtEmF,EAAK+hB,UAAUvnB,EAASK,KAAKM,OAAS,uBAE1C,OAASA,GACP6E,EAAK+hB,UAAU,yBAA2B5mB,EAAM+K,QAClD,CACA,OAAO,QAIXiF,EAAQrL,KAAK,CACX2I,KAAM,SACN2C,MAAO,wBACP1T,OAAQ,KACN,IACE,MAAM0mC,EAAa,CACjBpB,OACAC,KAAMj9B,EAAKi9B,KACXC,QAASl9B,EAAKk9B,QACdmB,4BAAA,IAAiB1iC,MAAO2iC,cACxBC,YAAa,0BAGTC,EAAO,IAAIC,KAAK,CAACxZ,KAAKC,UAAUkZ,EAAY,KAAM,IAAK,CAC3Dz+B,KAAM,qBAGFggB,EAAM+e,IAAIC,gBAAgBH,GAC1BI,EAAI1e,SAAS2e,cAAc,KASjC,OARAD,EAAEE,KAAOnf,EACTif,EAAEG,SAAW,QAAQ/B,EAAKnzB,MAAMlO,KAAK6hC,aACrCtd,SAAShV,KAAK8zB,YAAYJ,GAC1BA,EAAEK,QACF/e,SAAShV,KAAKg0B,YAAYN,GAC1BF,IAAIS,gBAAgBxf,GAEpB3f,EAAKi+B,YAAY,mCACV,IACT,OAAS9iC,GAEP,OADA6E,EAAK+hB,UAAU,8BACR,IACT,KAIJ5W,EAAQrL,KAAK,CACX2I,KAAM,QACN2C,MAAO,gBACPC,SAAS,UAGE1G,OAAOsG,WAAW,CAC7B3T,MAAO,wDAAwD0lC,EAAKnzB,KACpEqB,KAAMlL,EACNqI,KAAM,KACN4nB,YAAY,EACZ9kB,aACG/T,GAEP,ECjZF,SAASgoC,GAAYC,GACnB,OAAa,MAATA,EAAsB,MACtBA,GAAS,KAAaA,EAAQ,KAAKC,QAAQ,GAAK,MAChDD,GAAS,KAAaA,EAAQ,KAAKC,QAAQ,GAAK,MAChDD,GAAS,KAAaA,EAAQ,KAAKC,QAAQ,GAAK,MAC7CD,EAAQ,IACjB,CAEA,SAASE,GAAmBC,GAC1B,OAAIA,GAAO,GAAW,+BAClBA,GAAO,GAAW,iCACf,gCACT,CAEA,SAASC,GAAiBD,GACxB,OAAIA,GAAO,GAAW,YAClBA,GAAO,GAAW,aACf,YACT,CAWA,MAAME,0BAA0BxoC,EAC9B,WAAAC,CAAYC,EAAU,IACpBC,MAAM,CAAEO,UAAW,yBAA0BR,IAC7CS,KAAK0M,MAAQnN,EAAQmN,OAAS,IAChC,CAEA,oBAAMjK,GACJ,MAAMqlC,EAAI9nC,KAAK0M,MACf,IAAKo7B,EAAG,OAER9nC,KAAK+nC,gBAAkBD,EAAEn7B,IAAI,SAAW,iCAAmC,+BAC3E3M,KAAKgoC,UAAYF,EAAEn7B,IAAI,SAAW,uBAAyB,mBAC3D3M,KAAKioC,UAAYH,EAAEn7B,IAAI,SAAW,QAAU,OAE5C,MAAMu7B,EAAUJ,EAAEn7B,IAAI,WAGtB,GAFA3M,KAAKmoC,YAAcD,EAAU,IAAIpkC,KAAKokC,GAASnkC,iBAAmB,MAE9DmkC,EAAS,CACX,MAAME,GAAOtkC,KAAK6hC,MAAQ,IAAI7hC,KAAKokC,GAASG,WAAa,IACzDroC,KAAKsoC,WA/DX,SAAsBC,GACpB,MAAMC,EAAI/8B,KAAKy2B,MAAMqG,EAAU,OACzBE,EAAIh9B,KAAKy2B,MAAOqG,EAAU,MAAS,MACnC/wB,EAAI/L,KAAKy2B,MAAOqG,EAAU,KAAQ,IACxC,OAAIC,EAAI,EAAU,GAAGA,MAAMC,MAAMjxB,KAC7BixB,EAAI,EAAU,GAAGA,MAAMjxB,KACpB,GAAGA,IACZ,CAwDwBkxB,CAAaN,EACjC,MACEpoC,KAAKsoC,WAAa,MAGpB,MAAMK,GAjCeC,EAiCUd,EAAEn7B,IAAI,oBA/B/B7I,KAAK6hC,MAAQ,IAAI7hC,KAAK8kC,GAAWP,WAAa,IAD/B,KADzB,IAAyBO,EAkCN,OAAXD,GACF3oC,KAAK6oC,cAAgB,IAAI/kC,KAAKgkC,EAAEn7B,IAAI,mBAAmB5I,iBACvD/D,KAAK8oC,iBAAmB,GAAGr9B,KAAKs9B,MAAMJ,UACtC3oC,KAAKgpC,eAAiBL,EAAS,GAAK,eAAiBA,EAAS,IAAM,eAAiB,gBAErF3oC,KAAK6oC,cAAgB,MACrB7oC,KAAK8oC,iBAAmB,GACxB9oC,KAAKgpC,eAAiB,cAGxB,MAAMC,EAAgBnB,EAAEn7B,IAAI,mBAAqB,EAC3Cu8B,EAAapB,EAAEn7B,IAAI,gBAAkB,EAC3C3M,KAAKmpC,UAAaF,EAAgB,GAC5BC,EAAaD,EAAiB,KAAKxB,QAAQ,GAAK,IAClD,OACN,CAEA,iBAAM9mC,GACJ,MAAO,89HAqFT,EAOF,MAAMyoC,yBAAyB/pC,EAC7B,WAAAC,CAAYC,EAAU,IACpBC,MAAM,CAAEO,UAAW,wBAAyBR,IAC5CS,KAAK0M,MAAQnN,EAAQmN,OAAS,KAC9B1M,KAAKqpC,QAAU,KACfrpC,KAAKspC,aAAe,KACpBtpC,KAAKupC,SAAU,EACfvpC,KAAKwpC,QAAS,CAChB,CAEA,oBAAMC,GACAzpC,KAAKwpC,SACTxpC,KAAKwpC,QAAS,EACdxpC,KAAKupC,SAAU,EACfvpC,KAAKspC,aAAe,WACdtpC,KAAKqD,eACLrD,KAAK0pC,cACX1pC,KAAKupC,SAAU,QACTvpC,KAAKqD,SACb,CAEA,iBAAMqmC,GACJ,IACE,MAAMz7B,QAAajO,KAAK4C,SAASC,KAAKC,IACpC,6BAA6B9C,KAAK0M,MAAMC,IAAI,gBAE9C,GAAIsB,EAAKlL,SAAWkL,EAAKjL,KAAM,CAC7B,MAAM+1B,EAAU9qB,EAAKjL,KAAKA,MAAQiL,EAAKjL,KACvC,GAAI+1B,GAA8B,UAAnBA,EAAQ91B,OAErB,YADAjD,KAAKspC,aAAevQ,EAAQz1B,OAAS,gDAGvC,IAAK2K,EAAKjL,KAAKC,OAEb,YADAjD,KAAKspC,aAAer7B,EAAKjL,KAAKM,OAAS,+BAGzCtD,KAAKqpC,QAAUtQ,EAAQ9O,QAAU8O,EACjC/4B,KAAK2pC,eACP,MACE3pC,KAAKspC,aAAe,6BAExB,OAAS/2B,GACPvS,KAAKspC,aAAe/2B,EAAElE,SAAW,iBACnC,CACF,CAEA,aAAAs7B,GACE,MAAMza,EAAIlvB,KAAKqpC,QACf,IAAKna,EAAG,OAEJA,EAAE0a,SACJ1a,EAAE0a,OAAOC,SAAetC,GAAYrY,EAAE0a,OAAOlN,OAC7CxN,EAAE0a,OAAOE,QAAevC,GAAYrY,EAAE0a,OAAOG,MAC7C7a,EAAE0a,OAAOI,aAAezC,GAAYrY,EAAE0a,OAAOK,WAC7C/a,EAAE0a,OAAOM,SAAetC,GAAiB1Y,EAAE0a,OAAOO,SAClDjb,EAAE0a,OAAOQ,WAAe1C,GAAmBxY,EAAE0a,OAAOO,UAGlDjb,EAAEmb,OACJnb,EAAEmb,KAAKR,SAAatC,GAAYrY,EAAEmb,KAAK3N,OACvCxN,EAAEmb,KAAKP,QAAavC,GAAYrY,EAAEmb,KAAKN,MACvC7a,EAAEmb,KAAKC,QAAa/C,GAAYrY,EAAEmb,KAAKE,MACvCrb,EAAEmb,KAAKH,SAAatC,GAAiB1Y,EAAEmb,KAAKF,SAC5Cjb,EAAEmb,KAAKD,WAAa1C,GAAmBxY,EAAEmb,KAAKF,UAG5Cjb,EAAEsb,UACJtb,EAAEsb,QAAQC,aAAelD,GAAYrY,EAAEsb,QAAQE,YAC/Cxb,EAAEsb,QAAQG,aAAepD,GAAYrY,EAAEsb,QAAQI,YAC/C1b,EAAEsb,QAAQK,SAAgB3b,EAAEsb,QAAQM,MAAQ,GAAK5b,EAAEsb,QAAQO,OAAS,EAChE,sBAAwB,eAC5B7b,EAAEsb,QAAQQ,UAAgB9b,EAAEsb,QAAQS,OAAS,GAAK/b,EAAEsb,QAAQU,QAAU,EAClE,uBAAyB,gBAG/B,MAAMC,EAASjc,EAAEkc,UAAY,EAC7Blc,EAAEmc,gBAAoBzD,GAAiBuD,GACvCjc,EAAEoc,kBAAoB5D,GAAmByD,GAErCjc,EAAEqc,KAAOrc,EAAEqc,IAAIC,KACjBtc,EAAEqc,IAAIE,SAAW,GAAGhgC,KAAKs9B,MAAM7Z,EAAEqc,IAAIC,KAAKE,SAAS3nC,kCACzC0H,KAAKs9B,MAAM7Z,EAAEqc,IAAIC,KAAKvgB,KAAKlnB,2BAC5BmrB,EAAEqc,MACXrc,EAAEqc,IAAIE,SAAW,MAGfvc,EAAEyc,WAAazc,EAAEyc,UAAUt2B,OAC7B6Z,EAAE0c,SAAW1c,EAAEyc,UAAU/iC,IAAI,CAAC++B,EAAKxc,KAAA,CACjCqC,MAAOrC,EACPwc,IAAKA,EAAIF,QAAQ,GACjByC,SAAUtC,GAAiBD,MAG7BzY,EAAE0c,SAAW,GAGf1c,EAAE2c,aAAgB3c,EAAE4c,UAAY,IAAIhoC,KAAmB,IAAdorB,EAAE4c,WAAkB/nC,iBAAmB,KAChFmrB,EAAE6c,cAAgB7c,EAAE8c,UAAa,IACnC,CAEA,4BAAMC,GACJjsC,KAAKwpC,QAAS,EACdxpC,KAAKqpC,QAAU,KACfrpC,KAAKspC,aAAe,WACdtpC,KAAKypC,gBACb,CAEA,iBAAM9oC,GACJ,MAAO,goaA8QT,EAOF,MAAMurC,sBAAsB7sC,EAC1B,WAAAC,CAAYC,EAAU,IACpBC,MAAM,CAAEO,UAAW,qBAAsBR,IACzCS,KAAK0M,MAAQnN,EAAQmN,OAAS,KAC9B1M,KAAKmsC,KAAO,GACZnsC,KAAKupC,SAAU,EACfvpC,KAAKwpC,QAAS,CAChB,CAEA,oBAAMC,GACAzpC,KAAKwpC,SACTxpC,KAAKwpC,QAAS,EACdxpC,KAAKupC,SAAU,QACTvpC,KAAKqD,eACLrD,KAAKosC,WACXpsC,KAAKupC,SAAU,QACTvpC,KAAKqD,SACb,CAEA,cAAM+oC,GACJ,IACE,MAAMn+B,QAAajO,KAAK4C,SAASC,KAAKC,IACpC,2BAA2B9C,KAAK0M,MAAMC,IAAI,uCAE5C,GAAIsB,EAAKlL,SAAWkL,EAAKjL,MAAQiL,EAAKjL,KAAKC,OAAQ,CACjD,MAAM0iC,EAAM7hC,KAAK6hC,MAAQ,IACzB3lC,KAAKmsC,MAAQl+B,EAAKjL,KAAKA,MAAQ,IAAI4F,IAAI+1B,IAAA,UAClCA,EACH0N,aAAc1N,EAAI2N,YAvkBJ/D,EAwkBK5C,EAAM,IAAI7hC,KAAK66B,EAAI2N,YAAYjE,UAAY,IAvkBlEE,EAAU,GAAW,GAAG98B,KAAKs9B,MAAMR,MACnCA,EAAU,KAAa,GAAG98B,KAAKs9B,MAAMR,EAAU,QAAQ98B,KAAKs9B,MAAMR,EAAU,OACzE,GAAG98B,KAAKs9B,MAAMR,EAAU,UAAU98B,KAAKs9B,MAAOR,EAAU,KAAQ,QAskB3D,MACJJ,YAAaxJ,EAAI2N,WACb,IAAIxoC,KAAK66B,EAAI2N,YAAYC,qBACzB,MACJC,kBAAmB7N,EAAI8N,QAAU,EAC7B,+BACA,kCA/kBd,IAAwBlE,GAilBlB,MACEvoC,KAAKmsC,KAAO,EAEhB,OAAS55B,GACPvS,KAAKmsC,KAAO,GACZnsC,KAAKkqB,UAAU,gCAAkC3X,EAAElE,QACrD,CACF,CAEA,yBAAMq+B,GACJ1sC,KAAKwpC,QAAS,EACdxpC,KAAKmsC,KAAO,SACNnsC,KAAKypC,gBACb,CAEA,qBAAMkD,CAAgBnoC,EAAOC,GAC3B,MAAMmoC,EAAQnoC,EAAQkG,QAAQiiC,MAC9B5sC,KAAKwF,KAAK,WAAY,CAAEonC,QAAOC,OAAQ7sC,KAAK0M,OAC9C,CAEA,uBAAMizB,CAAkBn7B,EAAOC,GAC7B,MAAMmoC,EAAQnoC,EAAQkG,QAAQiiC,MAM9B,SALiB9/B,EAAOC,QACtB,wEACA,aACA,CAAEmb,YAAa,aAAcD,aAAc,gBAI7C,IACE,MAAMha,QAAajO,KAAK4C,SAASC,KAAKuO,KACpC,iBAAiBw7B,IAAS,CAAEE,gBAAgB,IAE1C7+B,EAAKlL,SAAWkL,EAAKjL,MAAQiL,EAAKjL,KAAKC,QACzCjD,KAAKomC,YAAY,uBACjBpmC,KAAKwpC,QAAS,QACRxpC,KAAKypC,kBAEXzpC,KAAKkqB,UAAWjc,EAAKjL,MAAQiL,EAAKjL,KAAKM,OAAU,wBAErD,OAASiP,GACPvS,KAAKkqB,UAAU,yBAA2B3X,EAAElE,QAC9C,CACF,CAEA,iBAAM1N,GACJ,MAAO,qrHA0ET,EAOF,MAAMosC,sBAAsB1tC,EAC1B,WAAAC,CAAYC,EAAU,IACpBC,MAAM,CAAEO,UAAW,qBAAsBR,IACzCS,KAAK0M,MAAQnN,EAAQmN,OAAS,KAC9B1M,KAAKolC,KAAO,GACZplC,KAAKgtC,aAAe,GACpBhtC,KAAKitC,UAAY,MACjBjtC,KAAKupC,SAAU,EACfvpC,KAAKwpC,QAAS,EAEdxpC,KAAKktC,eAAmB,cACxBltC,KAAKmtC,iBAAmB,wBACxBntC,KAAKotC,gBAAmB,sBACxBptC,KAAKqtC,gBAAmB,sBACxBrtC,KAAKstC,iBAAmB,oBAC1B,CAEA,oBAAM7D,GACAzpC,KAAKwpC,SACTxpC,KAAKwpC,QAAS,EACdxpC,KAAKupC,SAAU,QACTvpC,KAAKqD,eACLrD,KAAKutC,WACXvtC,KAAKupC,SAAU,QACTvpC,KAAKqD,SACb,CAEA,cAAMkqC,GACJ,IAEE,MAAMC,QAAiBxtC,KAAK4C,SAASC,KAAKC,IACxC,2BAA2B9C,KAAK0M,MAAMC,IAAI,uCAGtC8gC,EAAS,GAKf,GAJID,EAASzqC,SAAWyqC,EAASxqC,MAAQwqC,EAASxqC,KAAKC,SACpDuqC,EAASxqC,KAAKA,MAAQ,IAAIwH,QAAQkjC,GAAKD,EAAOxlC,KAAKylC,EAAE17B,MAGnDy7B,EAAOp4B,OAEV,YADArV,KAAKolC,KAAO,IAKd,MAAMpgC,EAAWyoC,EAAOE,MAAM,EAAG,GAAG/kC,IAAIoJ,GACtChS,KAAK4C,SAASC,KAAKC,IAAI,yBAAyBkP,2BAC7CyO,KAAKqnB,GAAMA,EAAE/kC,SAAW+kC,EAAE9kC,MAAQ8kC,EAAE9kC,KAAKC,QAAW6kC,EAAE9kC,KAAKA,MAAc,IACzE4qC,MAAM,IAAM,KAGXp5B,QAAgBpP,QAAQgrB,IAAIprB,GAC5BorB,EAAM,GAAGyd,UAAUr5B,GAEzB4b,EAAI7a,KAAK,CAACwxB,EAAG+G,IAAM,IAAIhqC,KAAKgqC,EAAEv3B,SAAW,IAAIzS,KAAKijC,EAAExwB,UACpDvW,KAAKolC,KAAOhV,EAAIud,MAAM,EAAG,IAAI/kC,IAAIo9B,IAAA,IAC5BA,EACH+H,gBAAiB/tC,KAAK8lC,iBAAiBE,EAAIxwB,MAC3Cw4B,aAAchI,EAAIxwB,MAAQ,QAAQG,cAClCs4B,YAAa,IAAInqC,KAAKkiC,EAAIzvB,SAASg2B,uBAEvC,OAASh6B,GACPvS,KAAKolC,KAAO,GACZplC,KAAKkqB,UAAU,wBAA0B3X,EAAElE,QAC7C,CACF,CAEA,gBAAAy3B,CAAiBtwB,GAOf,MANY,CACVuwB,MAAO,qCACPre,KAAO,iCACP5d,KAAO,iCACPxG,MAAO,gCAEEkS,IAAS,oCACtB,CAEA,oBAAM/S,GACJzC,KAAKgtC,aAAkC,QAAnBhtC,KAAKitC,UACrBjtC,KAAKolC,KACLplC,KAAKolC,KAAKlgC,OAAOgpC,GAAKA,EAAE14B,OAASxV,KAAKitC,WAE1CjtC,KAAKktC,eAAsC,QAAnBltC,KAAKitC,UAAwB,cAAyB,wBAC9EjtC,KAAKmtC,iBAAsC,UAAnBntC,KAAKitC,UAAwB,gBAA0B,wBAC/EjtC,KAAKotC,gBAAsC,SAAnBptC,KAAKitC,UAAwB,cAA0B,sBAC/EjtC,KAAKqtC,gBAAsC,SAAnBrtC,KAAKitC,UAAwB,cAA0B,sBAC/EjtC,KAAKstC,iBAAsC,UAAnBttC,KAAKitC,UAAwB,aAA0B,oBACjF,CAEA,wBAAMkB,CAAmB3pC,EAAOC,GAC9BzE,KAAKitC,UAAYxoC,EAAQkG,QAAQ6K,MAAQ,YACnCxV,KAAKqD,QACb,CAEA,yBAAM8iC,GACJnmC,KAAKwpC,QAAS,EACdxpC,KAAKolC,KAAO,GACZplC,KAAKitC,UAAY,YACXjtC,KAAKypC,gBACb,CAEA,iBAAM9oC,GACJ,MAAO,8zEA8CT,EAOF,MAAMytC,yBAAyB/uC,EAC7B,WAAAC,CAAYC,EAAU,IACpBC,MAAM,CAAEO,UAAW,wBAAyBR,IAC5CS,KAAK0M,MAAQnN,EAAQmN,OAAS,KAC9B1M,KAAKquC,WAAa,IACpB,CAEA,kBAAMC,GACJtuC,KAAKquC,WAAa,WACZruC,KAAKqD,SACX,IACE,MAAM4K,QAAajO,KAAK4C,SAASC,KAAKuO,KAAK,yBAA0B,CACnEm9B,UAAWvuC,KAAK0M,MAAMC,IAAI,aAC1B43B,QAAS,IAEPt2B,EAAKlL,SAAWkL,EAAKjL,KACvBhD,KAAKquC,WAAapgC,EAAKjL,KAAKyb,WACxB,qGACA,yHAEJze,KAAKquC,WAAa,8FAEtB,OAAS97B,GACPvS,KAAKquC,WAAa,qEAAqE97B,EAAElE,gBAC3F,OACMrO,KAAKqD,QACb,CAEA,sBAAMmrC,GAOJ,SANiB1hC,EAAOC,QACtB,8DAA8D/M,KAAK0M,MAAMC,IAAI,2GAE7E,kBACA,CAAEub,YAAa,WAAYD,aAAc,eAI3C,IACE,MAAMha,QAAajO,KAAK4C,SAASC,KAAKuO,KAAK,6BAA8B,CACvEm9B,UAAWvuC,KAAK0M,MAAMC,IAAI,aAC1B8hC,UAAU,IAERxgC,EAAKlL,SAAWkL,EAAKjL,MAAQiL,EAAKjL,KAAKC,QACzCjD,KAAKomC,YAAY,oCACjBpmC,KAAKwF,KAAK,kBAAmB,CAAEqnC,OAAQ7sC,KAAK0M,SAE5C1M,KAAKkqB,UAAWjc,EAAKjL,MAAQiL,EAAKjL,KAAKM,OAAU,2BAErD,OAASiP,GACPvS,KAAKkqB,UAAU,oBAAsB3X,EAAElE,QACzC,CACF,CAEA,uBAAMqgC,GACJ,MAAMC,EAAY3uC,KAAKyE,SAAWzE,KAAKyE,QAAQG,cAAc,oCACvDgqC,EAAY5uC,KAAKyE,SAAWzE,KAAKyE,QAAQG,cAAc,oCACvD0/B,EAAUqK,EAAYA,EAAUh+B,MAAQ,SACxC4zB,EAAUqK,GAAaC,WAAWD,EAAUj+B,QAAiB,EAEnE7D,EAAOgiC,SAAS,CAAEzgC,QAAS,iBAAiBi2B,uBAC5C,IACE,MAAMr2B,QAAajO,KAAK4C,SAASC,KAAKuO,KAAK,8BAA+B,CACxEkzB,UACAC,YAEFz3B,EAAOiiC,WAEH9gC,EAAKlL,SAAWkL,EAAKjL,WACjB8J,EAAOkiC,SACX5hB,KAAKC,UAAUpf,EAAKjL,KAAM,KAAM,GAChC,OACA,CAAEvD,MAAO,wBAAwB6kC,IAAW9zB,KAAM,OAGpDxQ,KAAKkqB,UAAWjc,EAAKjL,MAAQiL,EAAKjL,KAAKM,OAAU,oBAErD,OAASiP,GACPzF,EAAOiiC,WACP/uC,KAAKkqB,UAAU,qBAAuB3X,EAAElE,QAC1C,CACF,CAEA,oBAAM4gC,GACJ,IACE,MAAM1I,EAAa,CACjBsG,OAAQ7sC,KAAK0M,MAAMiG,OAAS3S,KAAK0M,MAAMiG,SAAW3S,KAAK0M,MACvD85B,4BAAA,IAAiB1iC,MAAO2iC,eAEpBE,EAAO,IAAIC,KAAK,CAACxZ,KAAKC,UAAUkZ,EAAY,KAAM,IAAK,CAAEz+B,KAAM,qBAC/DggB,EAAM+e,IAAIC,gBAAgBH,GAC1BI,EAAI7jC,OAAOC,OAAOklB,SAAS2e,cAAc,KAAM,CACnDC,KAAMnf,EACNof,SAAU,UAAUlnC,KAAK0M,MAAMC,IAAI,gBAAgB7I,KAAK6hC,eAE1Dtd,SAAShV,KAAK8zB,YAAYJ,GAC1BA,EAAEK,QACF/e,SAAShV,KAAKg0B,YAAYN,GAC1BF,IAAIS,gBAAgBxf,GACpB9nB,KAAKomC,YAAY,wBACnB,OAAS7zB,GACPvS,KAAKkqB,UAAU,kBAAoB3X,EAAElE,QACvC,CACF,CAEA,iBAAM1N,GACJ,MAAO,qyKAoHT,EAOa,MAAMuuC,0BAA0B7vC,EAC7C,WAAAC,CAAYC,EAAU,IACpBC,MAAM,CAAEO,UAAW,yBAA0BR,IAC7CS,KAAK0M,MAAQnN,EAAQmN,iBAAiB23B,GAClC9kC,EAAQmN,MACR,IAAI23B,GAAU9kC,EAAQmN,OAASnN,EAAQyD,MAAQ,CAAA,EACrD,CAEA,YAAMpC,GACJ,IAAKZ,KAAK0M,MAAO,OAEjB,MAAM+Z,EAAU,IAAIC,EAAQ,CAC1BvkB,YAAa,cACbujB,KAAM,CACJ8J,SAAgB,IAAIqY,kBAAkB,CAAEn7B,MAAO1M,KAAK0M,QACpD,cAAgB,IAAI08B,iBAAiB,CAAE18B,MAAO1M,KAAK0M,QACnD,eAAgB,IAAIw/B,cAAc,CAAEx/B,MAAO1M,KAAK0M,QAChDoZ,KAAgB,IAAIinB,cAAc,CAAErgC,MAAO1M,KAAK0M,QAChDyiC,QAAgB,IAAIf,iBAAiB,CAAE1hC,MAAO1M,KAAK0M,WAIvD1M,KAAKoC,SAASqkB,EAChB,CAEA,iBAAM9lB,GACJ,MAAO,0CACT,CASA,iBAAa+jB,CAAKmoB,EAAQttC,EAAU,IAClC,MAAMmN,EAAQmgC,aAAkBxI,GAAYwI,EAAS,IAAIxI,GAAUwI,GAC7D1kC,EAAO,IAAI+mC,kBAAkB,CAAExiC,UAErC,aAAaI,EAAOsG,WAAW,CAC7B3T,MAAO,8DAA8DiN,EAAMC,IAAI,sBAC/E0G,KAAMlL,EACNqI,KAAM,KACN4nB,YAAY,EACZ9kB,QAAS,CACP,CAAE1C,KAAM,QAAS2C,MAAO,gBAAiBC,SAAS,OAEjDjU,GAEP,EAIF8kC,GAAUnnB,WAAagyB,kBCtpCvB,MAAME,sBAAsB/vC,EAC1B,WAAAC,CAAYC,EAAU,IACpBC,MAAM,IACDD,EACHQ,UAAW,4BAGbC,KAAKC,MAAQ,CACX+7B,QAAS,EACTC,QAAS,EACTC,UAAW,EACXmT,OAAQ,EAEZ,CAEA,iBAAM1uC,GACJ,MAAO,4xGAiFT,CAEA,eAAM6C,GACJ,IACE,MAAMb,QAAiB3C,KAAK4C,SAASC,KAAKC,IAAI,qBAC1CH,EAASI,SAAWJ,EAASK,KAAKC,SACpCjD,KAAKC,MAAQ0C,EAASK,KAAKA,KAE/B,OAASM,GACPC,QAAQD,MAAM,6BAA8BA,EAC9C,CACF,CAEA,YAAM1C,SACEZ,KAAKwD,WACb,EAIF,MAAM8rC,wBAAwBjwC,EAC5B,WAAAC,CAAYC,EAAU,IACpBC,MAAM,IACDD,EACHQ,UAAW,8BAGbC,KAAKw8B,QAAU,EACjB,CAEA,iBAAM77B,GACJ,MAAO,igJAsFT,CAEA,iBAAM4uC,GACJ,IACE,MAAM5sC,QAAiB3C,KAAK4C,SAASC,KAAKC,IAAI,sBAC1CH,EAASI,SAAWJ,EAASK,KAAKC,SACpCjD,KAAKw8B,QAAU75B,EAASK,KAAKA,KAAK4F,IAAIikC,IACpC,MAAM/jC,EAA6B,WAAlB+jC,EAAO5pC,OAClBusC,EAAU3C,EAAO4C,UAAY,EAEnC,MAAO,IACF5C,EACH/jC,WACA4mC,YAAa5mC,EAAW,aAAe,aACvC40B,WAAY50B,EAAW,uBAAyB,+BAChD6mC,YAAa3vC,KAAK4vC,cAAcJ,MAIxC,OAASlsC,GACPC,QAAQD,MAAM,0BAA2BA,EAC3C,CACF,CAEA,aAAAssC,CAAcrH,GACZ,OAAIA,EAAU,GAAW,GAAG98B,KAAKs9B,MAAMR,UACnCA,EAAU,KAAa,GAAG98B,KAAKs9B,MAAMR,EAAU,WAC5C,GAAG98B,KAAKs9B,MAAMR,EAAU,YACjC,CAEA,YAAM3nC,SACEZ,KAAKuvC,aACb,CAEA,4BAAMM,CAAuBrrC,EAAOC,SAC5BzE,KAAKuvC,aACb,CAEA,+BAAMO,CAA0BtrC,EAAOC,GACrC,MAAMsrC,EAAWtrC,EAAQo+B,aAAa,kBAChCgK,EAAS7sC,KAAKw8B,QAAQvzB,KAAK6+B,GAAKA,EAAE91B,KAAO+9B,GAE/C,GAAIlD,EAAQ,CACV,MAAM5iB,QAAeilB,kBAAkBxqB,KAAKmoB,GAGxC5iB,GAAQpqB,eACJG,KAAKuvC,cACXvvC,KAAKwF,KAAK,UAAYykB,EAAOpqB,OAAQoqB,GAEzC,CACF,CAEA,yBAAM+lB,CAAoBxrC,EAAOC,GAC/B,MAAMsrC,EAAWtrC,EAAQo+B,aAAa,kBAChCgK,EAAS7sC,KAAKw8B,QAAQvzB,KAAK6+B,GAAKA,EAAE91B,KAAO+9B,GAE/C,GAAKlD,GAAW9/B,QAAQ,0CAA0C8/B,EAAOoD,cAIzE,IACExrC,EAAQM,UAAW,EACnB,MAAMpC,QAAiB3C,KAAK4C,SAASC,KAAKuO,KAAK,gBAAgB2+B,WAE3DptC,EAASI,SAAWJ,EAASK,KAAKC,QACpCjD,KAAKomC,YAAY,oCACXpmC,KAAKuvC,eAEXvvC,KAAKkqB,UAAUvnB,EAASK,KAAKM,OAAS,yBAE1C,OAASA,GACPC,QAAQD,MAAM,0BAA2BA,GACzCtD,KAAKkqB,UAAU,2BAA6B5mB,EAAM+K,QACpD,CAAA,QACE5J,EAAQM,UAAW,CACrB,CACF,CAEA,2BAAMmrC,CAAsB1rC,EAAOC,GACjC,MAAMsrC,EAAWtrC,EAAQo+B,aAAa,kBAChCgK,EAAS7sC,KAAKw8B,QAAQvzB,KAAK6+B,GAAKA,EAAE91B,KAAO+9B,GAE/C,GAAKlD,GAAW9/B,QAAQ,4CAA4C8/B,EAAOoD,cAI3E,IACExrC,EAAQM,UAAW,EACnB,MAAMpC,QAAiB3C,KAAK4C,SAASC,KAAKuO,KAAK,gBAAgB2+B,aAE3DptC,EAASI,SAAWJ,EAASK,KAAKC,QACpCjD,KAAKomC,YAAY,kCACXpmC,KAAKuvC,eAEXvvC,KAAKkqB,UAAUvnB,EAASK,KAAKM,OAAS,2BAE1C,OAASA,GACPC,QAAQD,MAAM,4BAA6BA,GAC3CtD,KAAKkqB,UAAU,6BAA+B5mB,EAAM+K,QACtD,CAAA,QACE5J,EAAQM,UAAW,CACrB,CACF,CAEA,0BAAMorC,CAAqB3rC,EAAOC,GAChC,MAAMsrC,EAAWtrC,EAAQo+B,aAAa,kBAChCgK,EAAS7sC,KAAKw8B,QAAQvzB,KAAK6+B,GAAKA,EAAE91B,KAAO+9B,GAE/C,IAAKlD,EAAQ,OAEb,MAAMuD,EAAiB,2CAA2CvD,EAAOoD,2CACzE,GAAKljC,QAAQqjC,GAIb,IACE3rC,EAAQM,UAAW,EACnB,MAAMpC,QAAiB3C,KAAK4C,SAASC,KAAKkP,OAAO,gBAAgBg+B,KAE7DptC,EAASI,SAAWJ,EAASK,KAAKC,QACpCjD,KAAKomC,YAAY,qCACXpmC,KAAKuvC,eAEXvvC,KAAKkqB,UAAUvnB,EAASK,KAAKM,OAAS,0BAE1C,OAASA,GACPC,QAAQD,MAAM,2BAA4BA,GAC1CtD,KAAKkqB,UAAU,4BAA8B5mB,EAAM+K,QACrD,CAAA,QACE5J,EAAQM,UAAW,CACrB,CACF,EAIF,MAAMsrC,uBAAuBhxC,EAC3B,WAAAC,CAAYC,EAAU,IACpBC,MAAM,IACDD,EACHQ,UAAW,4BAEf,CAEA,iBAAMY,GACJ,MAAO,slBAkBT,CAEA,YAAMC,GAEJZ,KAAKswC,cAAgB,IAAIpsC,EAAa,CACpCzE,MAAO,+CACP8C,SAAU,qBACVb,OAAQ,IACRR,YAAa,QACbG,MAAO,CAAC,YAAa,mBACrBC,QAAS,SACTC,UAAW,OACXW,eAAe,EACf+pB,OAAQ,CACN,0BACA,0BAEF9nB,MAAO,CACLxE,MAAO,QACPyE,aAAa,GAEfC,QAAS,CACPC,EAAG,UAELnC,YAAa,oBAEfnC,KAAKoC,SAASpC,KAAKswC,eAGnBtwC,KAAKuwC,gBAAkB,IAAIrsC,EAAa,CACtCzE,MAAO,6DACP8C,SAAU,qBACVb,OAAQ,IACRR,YAAa,QACbG,MAAO,CAAC,eAAgB,iBACxBC,QAAS,SACTC,UAAW,OACXW,eAAe,EACf+pB,OAAQ,CACN,yBACA,0BAEF9nB,MAAO,CACLxE,MAAO,QACPyE,aAAa,GAEfC,QAAS,CACPC,EAAG,UAELnC,YAAa,sBAEfnC,KAAKoC,SAASpC,KAAKuwC,gBACrB,EA8JF,MAAMC,0BAA0Bt2B,GAC9B,WAAA5a,CAAYC,EAAU,IACpBC,MAAM,IACDD,EACHE,MAAO,gBACPyS,WAAY,IAAIoL,EAAW,CACzB/a,SAAU,uBAEZ6X,QAAS,CACP,CAAElS,IAAK,KAAMvI,MAAO,UAAW2a,UAAU,GACzC,CAAEpS,IAAK,WAAYvI,MAAO,WAAY2a,UAAU,GAChD,CAAEpS,IAAK,UAAWvI,MAAO,UAAW2a,UAAU,GAC9C,CAAEpS,IAAK,UAAWvI,MAAO,UAAW2a,UAAU,EAAMD,UAAW,YAC/D,CAAEnS,IAAK,SAAUvI,MAAO,SAAU0a,UAAW,WAGnD,EAGF,MAAMo2B,0BAA0Bv2B,GAC9B,WAAA5a,CAAYC,EAAU,IACpBC,MAAM,IACDD,EACHE,MAAO,gBACPyS,WAAY,IAAIoL,EAAW,CACzB/a,SAAU,uBAEZ6X,QAAS,CACP,CAAElS,IAAK,KAAMvI,MAAO,UAAW2a,UAAU,GACzC,CAAEpS,IAAK,WAAYvI,MAAO,WAAY2a,UAAU,GAChD,CAAEpS,IAAK,UAAWvI,MAAO,UAAW2a,UAAU,GAC9C,CAAEpS,IAAK,UAAWvI,MAAO,UAAW2a,UAAU,EAAMD,UAAW,YAC/D,CAAEnS,IAAK,SAAUvI,MAAO,SAAU0a,UAAW,WAGnD,EAGF,MAAMq2B,4BAA4Bx2B,GAChC,WAAA5a,CAAYC,EAAU,IACpBC,MAAM,IACDD,EACHE,MAAO,kBACPyS,WAAY,IAAIoL,EAAW,CACzB/a,SAAU,yBAEZ6X,QAAS,CACP,CAAElS,IAAK,KAAMvI,MAAO,UAAW2a,UAAU,GACzC,CAAEpS,IAAK,WAAYvI,MAAO,WAAY2a,UAAU,GAChD,CAAEpS,IAAK,UAAWvI,MAAO,UAAW2a,UAAU,GAC9C,CAAEpS,IAAK,UAAWvI,MAAO,UAAW2a,UAAU,EAAMD,UAAW,YAC/D,CAAEnS,IAAK,eAAgBvI,MAAO,YAAa2a,UAAU,EAAMD,UAAW,YACtE,CAAEnS,IAAK,SAAUvI,MAAO,SAAU0a,UAAW,WAGnD,EAGF,MAAMs2B,wBAAwBz2B,GAC5B,WAAA5a,CAAYC,EAAU,IACpBC,MAAM,IACDD,EACHE,MAAO,eACPyS,WAAY,IAAIoL,EAAW,CACzB/a,SAAU,sBAEZ6X,QAAS,CACP,CAAElS,IAAK,KAAMvI,MAAO,UAAW2a,UAAU,GACzC,CAAEpS,IAAK,WAAYvI,MAAO,WAAY2a,UAAU,GAChD,CAAEpS,IAAK,UAAWvI,MAAO,UAAW2a,UAAU,GAC9C,CAAEpS,IAAK,UAAWvI,MAAO,UAAW2a,UAAU,EAAMD,UAAW,YAC/D,CAAEnS,IAAK,QAASvI,MAAO,QAAS2a,UAAU,GAC1C,CAAEpS,IAAK,SAAUvI,MAAO,SAAU0a,UAAW,WAGnD,EAIa,MAAMu2B,2BAA2BltC,EAC9C,WAAApE,CAAYC,EAAU,IACpBC,MAAM,IACDD,EACHE,MAAO,kBACPM,UAAW,8BAGbC,KAAK2D,UAAY,kBACjB3D,KAAK4D,aAAe,8CACpB5D,KAAK6D,4BAAA,IAAkBC,MAAOC,gBAChC,CAEA,iBAAMpD,GACJ,MAAO,isFAsET,CAEA,YAAMC,GAEJZ,KAAK6wC,cAAgB,IAAIzB,cAAc,CACrCjtC,YAAa,eAEfnC,KAAKoC,SAASpC,KAAK6wC,eAGnB7wC,KAAK8wC,gBAAkB,IAAIxB,gBAAgB,CACzCntC,YAAa,iBAEfnC,KAAKoC,SAASpC,KAAK8wC,iBAGnB9wC,KAAK+wC,eAAiB,IAAIV,eAAe,CACvCluC,YAAa,gBAEfnC,KAAKoC,SAASpC,KAAK+wC,gBAGnB/wC,KAAKgxC,eAAiB,IAAItqB,EAAQ,CAChCvkB,YAAa,cACbujB,KAAM,CACJurB,QAAW,IAAIT,kBACfU,QAAW,IAAIT,kBACfU,UAAa,IAAIT,oBACjBU,OAAU,IAAIT,iBAEhBhqB,UAAW,YAEb3mB,KAAKoC,SAASpC,KAAKgxC,eACrB,CAEA,wBAAMzsC,CAAmB1E,EAAQ2E,EAAOC,GACtC,IAEE,MAAM7E,EAAO6E,EAAQG,cAAc,KACnChF,GAAMiF,UAAUC,IAAI,WACpBL,EAAQM,UAAW,EAGnB,MAAMC,EAAW,CACfhF,KAAK6wC,eAAertC,YACpBxD,KAAK8wC,iBAAiBvB,cACtBvvC,KAAK+wC,gBAAgBT,eAAerrC,UACpCjF,KAAK+wC,gBAAgBR,iBAAiBtrC,WACtCC,OAAOC,SAGHwhB,EAAY3mB,KAAKgxC,gBAAgB/zB,eACvC,GAAI0J,EAAW,CACb,MAAM0qB,EAAcrxC,KAAKgxC,eAAeM,OAAO3qB,GAC3C0qB,GAAapsC,SACfD,EAASiD,KAAKopC,EAAYpsC,UAE9B,OAEMG,QAAQC,WAAWL,GAGzBhF,KAAK6D,4BAAA,IAAkBC,MAAOC,iBAG9B,MAAMuB,EAAWtF,KAAK4C,UAAU2C,OAC5BD,GACFA,EAASE,KAAK,4BAA6B,CACzCC,KAAMzF,KACN0F,UAAW1F,KAAK6D,aAItB,OAASP,GACPC,QAAQD,MAAM,oCAAqCA,EACrD,CAAA,QAEE,MAAM1D,EAAO6E,EAAQG,cAAc,KACnChF,GAAMiF,UAAUiB,OAAO,WACvBrB,EAAQM,UAAW,CACrB,CACF,CAEA,yBAAMwsC,CAAoB1xC,EAAQ2E,EAAOC,GACvC,UAEQzE,KAAK+wC,gBAAgBT,eAAetqC,OAAO,cAC3ChG,KAAK+wC,gBAAgBR,iBAAiBvqC,OAAO,QAGnD,MAAM2gB,EAAY3mB,KAAKgxC,gBAAgB/zB,eACvC,GAAI0J,EAAW,CACb,MAAM0qB,EAAcrxC,KAAKgxC,eAAeM,OAAO3qB,GAC3C0qB,GAAaG,aACfH,EAAYG,aAEhB,CAEF,OAASluC,GACPC,QAAQD,MAAM,8BAA+BA,EAC/C,CACF,CAEA,4BAAMmuC,CAAuBjtC,EAAOC,GAClC,IAEE,MAAM9B,QAAiB3C,KAAK4C,SAASC,KAAKC,IAAI,uBAE9C,GAAIH,EAASI,SAAWJ,EAASK,KAAKC,OAAQ,CAC5C,MAGMyuC,EAHW/uC,EAASK,KAAKA,KAGF4F,IAAIiN,GAC/B,GAAGA,EAAQ7F,SAAS6F,EAAQmmB,oBAAoBnmB,EAAQomB,oBACxDlzB,KAAK,MAEPpD,MAAM,qBAAqB+rC,sDAC7B,MACE1xC,KAAKkqB,UAAU,qCAEnB,OAAS5mB,GACPC,QAAQD,MAAM,2BAA4BA,GAE1C,MAAM6C,EAASnG,KAAK4C,UAAUuD,OAC1BA,GACFA,EAAOC,WAAW,uBAEtB,CACF,CAEA,4BAAMurC,CAAuBntC,EAAOC,GAClC,IAEE,MAAM9B,QAAiB3C,KAAK4C,SAASC,KAAKC,IAAI,4BAE9C,GAAIH,EAASI,SAAWJ,EAASK,KAAKC,OAAQ,CAC5C,MAAMmiC,EAAOziC,EAASK,KAAKA,KAGrB4uC,EAAaxM,EAAKuI,MAAM,EAAG,IAAI/kC,OACnC,IAAI,IAAI9E,KAAqB,IAAhBkiC,EAAItgC,WAAkB3B,qBAAqBiiC,EAAI9U,MAAMvb,kBAAkBqwB,EAAI33B,WACxFtF,KAAK,MAIP,GAFqBgE,QAAQ,+BAA+B6kC,gBAAyBxM,EAAK/vB,OAAS,uCAEjF,CAChB,MAAMlP,EAASnG,KAAK4C,UAAUuD,OAC1BA,GACFA,EAAOC,WAAW,2BAEtB,CACF,KAAO,CAEL,MAAMD,EAASnG,KAAK4C,UAAUuD,OAC1BA,GACFA,EAAOC,WAAW,cAEtB,CACF,OAAS9C,GACPC,QAAQD,MAAM,8BAA+BA,GAE7C,MAAM6C,EAASnG,KAAK4C,UAAUuD,OAC1BA,GACFA,EAAOC,WAAW,cAEtB,CACF,CAGA,sBAAME,GACJ,OAAOtG,KAAKuE,mBAAmB,KAAM,KAAM,CAAEQ,UAAU,EAAOH,cAAe,IAAM,MACrF,CAEA,QAAA6B,GACE,OAAOzG,KAAK6wC,eAAe5wC,OAAS,CAAA,CACtC,CAEA,UAAA4xC,GACE,OAAO7xC,KAAK8wC,iBAAiBtU,SAAW,EAC1C,CAEA,SAAAj2B,GACE,MAAO,CACLurC,SAAU9xC,KAAK+wC,gBAAgBT,cAC/ByB,WAAY/xC,KAAK+wC,gBAAgBR,gBAErC,ECz6BF,MAAMyB,gBAAgB3yC,EAClB,WAAAC,CAAYC,EAAU,IAClBC,MAAM,CACFO,UAAW,cACRR,IAGPS,KAAK0M,MAAQnN,EAAQmN,OAAS,IAAIulC,GAAI1yC,EAAQyD,MAAQ,IACtDhD,KAAKkyC,QAAUlyC,KAAKmyC,cAAcnyC,KAAK0M,MAAMC,IAAI,UAEjD3M,KAAKwM,SAAW,whCAwBpB,CAEA,aAAA2lC,CAAcjhB,GACV,MAAMkhB,EAAMlhB,GAAOlgB,cACnB,MAAY,UAARohC,GAA2B,aAARA,EAA2B,CAAExyC,KAAM,oBAAqBgC,MAAO,eAC1E,YAARwwC,EAA0B,CAAExyC,KAAM,+BAAgCgC,MAAO,gBACjE,SAARwwC,EAAuB,CAAExyC,KAAM,sBAAuBgC,MAAO,aAC1D,CAAEhC,KAAM,kBAAmBgC,MAAO,iBAC7C,CAEA,YAAMhB,GAEFZ,KAAKmvB,aAAe,IAAIjK,GAAS,CAC7BxY,MAAO1M,KAAK0M,MACZ3M,UAAW,MACXqa,QAAS,EACT5L,OAAQ,CACJ,CAAEwB,KAAM,KAAMrQ,MAAO,UACrB,CAAEqQ,KAAM,QAASrQ,MAAO,QAASic,OAAQ,SACzC,CAAE5L,KAAM,OAAQrQ,MAAO,QACvB,CAAEqQ,KAAM,KAAMrQ,MAAO,aAAc6M,SAAU,sDAC7C,CAAEwD,KAAM,MAAOrQ,MAAO,WACtB,CAAEqQ,KAAM,WAAYrQ,MAAO,YAC3B,CAAEqQ,KAAM,OAAQrQ,MAAO,YAAa6M,SAAU,gFAC9C,CAAEwD,KAAM,aAAcrQ,MAAO,iBAC7B,CAAEqQ,KAAM,WAAYrQ,MAAO,oBAC3B,CAAEqQ,KAAM,aAAcrQ,MAAO,aAAcya,QAAS,OAK5D,MAAMi4B,EAAaryC,KAAK0M,MAAMC,IAAI,OAClC,IAAI2lC,EAAeD,EACnB,IAEI,MAAME,EAASnlB,KAAKolB,MAAMH,GAC1BC,EAAellB,KAAKC,UAAUklB,EAAQ,KAAM,EAChD,OAAShgC,GAET,CAEAvS,KAAKyyC,eAAiB,IAAIpzC,EAAK,CAC3BmN,SAAU,yYAK4F8lC,uDAGtGI,gBAAiB,KACbC,UAAUC,UAAUC,UAAUP,GAC9BtyC,KAAK4C,UAAUwL,OAAOrL,QAAQ,uCAKtC/C,KAAKymB,QAAU,IAAIC,EAAQ,CACvBvkB,YAAa,WACbujB,KAAM,CACFusB,IAAOjyC,KAAKyyC,eACZK,QAAW9yC,KAAKmvB,cAGpBxI,UAAW,QAEf3mB,KAAKoC,SAASpC,KAAKymB,SAGnB,MAAMssB,EAAU,IAAI52B,EAAY,CAC5Bha,YAAa,mBACbpC,UAAW,yCACXqc,QAASpc,KAAK0M,MACdhF,OAAQ,CACJ9H,KAAM,yBACNwJ,MAAO,CACH,CAAEzJ,MAAO,YAAaE,OAAQ,YAAaD,KAAM,YAAamF,UAAW/E,KAAK0M,MAAMC,IAAI,QACxF,CAAEhN,MAAO,cAAeE,OAAQ,cAAeD,KAAM,WAAYmF,UAAW/E,KAAK0M,MAAMC,IAAI,SAC3F,CAAE7E,KAAM,WACR,CAAEnI,MAAO,aAAcE,OAAQ,aAAcD,KAAM,WAAYqgB,QAAQ,OAInFjgB,KAAKoC,SAAS2wC,EAClB,CAEA,oBAAMC,CAAexuC,GACjBA,EAAMyG,iBACN,MAAMwa,EAAKzlB,KAAK0M,MAAMC,IAAI,MACtB8Y,GACAV,UAAUL,KAAKe,EAEvB,CAEA,wBAAMwtB,CAAmBzuC,GACrBA,EAAMyG,iBACN,MAAM0Z,EAAO3kB,KAAK0M,MAAMC,IAAI,QACxBgY,GACApB,WAAWmB,KAAKC,EAExB,CAEA,sBAAMrE,GAC6BtgB,KAAK0M,MAAMC,IAAI,MAClD,CAEA,uBAAMumC,SACsBpmC,EAAOC,QAC3B,gFACA,mBACA,CAAEkb,aAAc,aAAcC,YAAa,mBAGxBloB,KAAK0M,MAAM9C,WACrB7G,SACL/C,KAAKwF,KAAK,cAAe,CAAEkH,MAAO1M,KAAK0M,OAGnD,EAGJulC,GAAI/0B,WAAa80B,QC/JjB,MAAMmB,qBAAqB/1B,EACvB,WAAA9d,CAAYC,EAAU,IAClBC,MAAM,IACCD,EACHyQ,KAAM,aACNqN,SAAU,cACVlX,OAAQ,aACRmX,WAAY/B,GAEZ8F,cAAe2wB,QACfx0B,kBAAmB,CACfpa,QAAQ,EACRoN,KAAM,MAIV4J,QAAS,CACL,CACIlS,IAAK,yBACLvI,MAAO,YACP2a,UAAU,EACVpV,OAAQ,CACJ4C,KAAM,cAGd,CACII,IAAK,QACLvI,MAAO,QACP2a,UAAU,EACVD,UAAW,QACXnV,OAAQ,CACJ4C,KAAM,SACNvI,QAAS,CACL,CAAEoR,MAAO,OAAQhR,MAAO,QACxB,CAAEgR,MAAO,UAAWhR,MAAO,WAC3B,CAAEgR,MAAO,QAAShR,MAAO,YAIrC,CACIuI,IAAK,OACLvI,MAAO,OACPuF,OAAQ,CACJ4C,KAAM,SAGd,CACII,IAAK,SACLvI,MAAO,SACPuF,OAAQ,CACJ4C,KAAM,SAGd,CACII,IAAK,OACLvI,MAAO,OACPuF,OAAQ,CACJ4C,KAAM,SAGd,CACII,IAAK,WACLvI,MAAO,OACPuF,OAAQ,CACJ4C,KAAM,SAGd,CACII,IAAK,KACLvI,MAAO,KACPuF,OAAQ,CACJ4C,KAAM,SAGd,CACII,IAAK,OACLvI,MAAO,aACP0a,UAAW,sBACXnV,OAAQ,CACJ4C,KAAM,UAMlB2V,aAAc,CACVlI,KAAM,YAIVqI,YAAY,EACZC,YAAY,EACZvD,UAAU,EACVwD,YAAY,EACZC,WAAW,EAGXC,aAAa,EACbC,SAAS,EACTC,YAAY,EAGZC,aAAc,wBAGdmD,iBAAkB,MAClBC,aAAc,CACV,CAAE5hB,MAAO,SAAUC,KAAM,iBAAkBC,OAAQ,gBACnD,CAAEF,MAAO,UAAWC,KAAM,gBAAiBC,OAAQ,iBACnD,CAAEF,MAAO,SAAUC,KAAM,cAAeC,OAAQ,gBAChD,CAAEF,MAAO,mBAAoBC,KAAM,eAAgBC,OAAQ,mBAI/Dwe,aAAc,CACVC,SAAS,EACTC,UAAU,EACVC,OAAO,EACPC,YAAY,IAGxB,EC5HJ,MAAM20B,+BAA+B/zC,EACjC,WAAAC,CAAYC,EAAU,IAClBC,MAAM,CACFO,UAAW,8BACRR,IAGPS,KAAK0M,MAAQnN,EAAQmN,OAAS,IAAI2mC,GAAkB9zC,EAAQyD,MAAQ,IAEpEhD,KAAKwM,SAAW,sbAWpB,CAEA,YAAM5L,GACFZ,KAAK67B,SAAW,IAAI3W,GAAS,CACzB/iB,YAAa,YACbuK,MAAO1M,KAAK0M,MACZ8B,OAAQ,CACJ,CAAEwB,KAAM,mBAAoBrQ,MAAO,mBAAoBic,OAAQ,cAC/D,CAAE5L,KAAM,oBAAqBrQ,MAAO,oBAAqBic,OAAQ,iBAGzE5b,KAAKoC,SAASpC,KAAK67B,UAEnB,MAAMzd,EAAc,IAAIjC,EAAY,CAChCha,YAAa,eACbia,QAASpc,KAAK0M,MACdhF,OAAQ,CACJ9H,KAAM,yBACNwJ,MAAO,CACH,CAAEzJ,MAAO,OAAQE,OAAQ,OAAQD,KAAM,aACvC,CAAED,MAAO,SAAUE,OAAQ,SAAUD,KAAM,WAAYqgB,QAAQ,OAI3EjgB,KAAKoC,SAASgc,EAClB,CAEA,kBAAMk1B,GACF,MAAMrlC,QAAanB,EAAOkG,cAAc,CACpCvT,MAAO,wBAAwBO,KAAK0M,MAAMC,IAAI,aAC9CD,MAAO1M,KAAK0M,MACZ6P,WAAYg3B,GAAargC,OAEzBjF,IACAjO,KAAK0M,MAAMyB,IAAIF,EAAKjL,KAAKA,MACzBhD,KAAKqD,SAEb,CAEA,oBAAMmwC,SACsB1mC,EAAOC,QAAQ,uDAAuD/M,KAAK0M,MAAMC,IAAI,uBAEnG3M,KAAK0M,MAAM9C,UACjB5J,KAAKwF,KAAK,UAAWxF,KAAK0M,OAElC,EAGJ2mC,GAAkBn2B,WAAak2B,uBCtE/B,MAAMK,oCAAoCr2B,EACtC,WAAA9d,CAAYC,EAAU,IAClBC,MAAM,IACCD,EACHyQ,KAAM,4BACNqN,SAAU,sBACVlX,OAAQ,4BACRmX,WAAYo2B,GACZtyB,SAAUmyB,GAAargC,KACvBmO,cAAe+xB,uBACf51B,kBAAmB,CACfpa,QAAQ,EACRoN,KAAM,MAGV4J,QAAS,CACL,CAAElS,IAAK,UAAWvI,MAAO,UAAW2a,UAAU,GAC9C,CAAEpS,IAAK,mBAAoBvI,MAAO,mBAAoB0a,UAAW,cACjE,CAAEnS,IAAK,oBAAqBvI,MAAO,oBAAqB0a,UAAW,eAGvEuD,YAAY,EACZC,YAAY,EACZvD,UAAU,EACVyD,WAAW,EACXC,aAAa,EACbC,SAAS,EACTC,YAAY,EAEZG,aAAc,CACViX,UAAW,CAAC,GAAI,GAAI,IACpBC,gBAAiB,GACjBpX,aAAc,gCACdqX,UAAW,oBACXpG,QAAS,CAAC,OAAQ,OAAQ,YAGtC,ECtBJ,MAAMukB,gBAAgB5qB,EAClB,WAAAzpB,CAAY0D,EAAO,GAAIzD,EAAU,CAAA,GAC7BC,MAAMwD,EAAM,CACRT,SAAU,mBACPhD,GAEX,EAOJ,MAAMq0C,oBAAoBt2B,EACtB,WAAAhe,CAAYC,EAAU,IAClBC,MAAM,CACFypB,WAAY0qB,QACZpxC,SAAU,gBACViO,KAAM,MACHjR,GAEX,EAMJ,MAAMs0C,GAAe,CACjBvnC,OAAQ,CACJ7M,MAAO,iBACP+O,OAAQ,CACJ,CACIwB,KAAM,MACNlI,KAAM,OACNnI,MAAO,MACPoO,YAAa,iBACbyD,UAAU,EACV4I,QAAS,GACT3I,KAAM,oCAEV,CACIzB,KAAM,QACNlI,KAAM,WACNnI,MAAO,QACP6R,UAAU,EACV4I,QAAS,GACT3I,KAAM,6EAEV,CACIzB,KAAM,QACNlI,KAAM,SACNnI,MAAO,WACPya,QAAS,EACT3I,KAAM,gFAEV,CACIzB,KAAM,YACNlI,KAAM,SACNnI,MAAO,SACPya,QAAS,EACT3I,KAAM,wDAKlByB,KAAM,CACFzT,MAAO,eACP+O,OAAQ,CACJ,CACIwB,KAAM,MACNlI,KAAM,OACNnI,MAAO,MACPya,QAAS,GACTrV,UAAU,GAEd,CACIiL,KAAM,QACNlI,KAAM,WACNnI,MAAO,QACPya,QAAS,GACT3I,KAAM,iDAEV,CACIzB,KAAM,YACNlI,KAAM,SACNnI,MAAO,SACPya,QAAS,GACT3I,KAAM,yDC9FtB,MAAMqiC,oBAAoBz0C,EACtB,WAAAC,CAAYC,EAAU,IAClBC,MAAM,CACFO,UAAW,kBACRR,IAGPS,KAAK0M,MAAQnN,EAAQmN,OAAS,IAAIinC,QAAQp0C,EAAQyD,MAAQ,IAE1DhD,KAAKwM,SAAW,8iGA2DpB,CAEA,YAAM5L,GACF,MAAMmzC,EAAc,IAAI53B,EAAY,CAChCha,YAAa,uBACbpC,UAAW,yCACXqc,QAASpc,KAAK0M,MACdhF,OAAQ,CACJ9H,KAAM,yBACNwJ,MAAO,CACH,CAAEzJ,MAAO,OAAQE,OAAQ,eAAgBD,KAAM,aAC/C,CAAEkI,KAAM,WACR,CAAEnI,MAAO,iBAAkBE,OAAQ,iBAAkBD,KAAM,WAAYqgB,QAAQ,OAI3FjgB,KAAKoC,SAAS2xC,EAClB,CAEA,yBAAMC,GACF,MAAM7iC,EAAMnR,KAAK4C,eACEuO,EAAI6B,cAAc,CACjCvT,MAAO,kBAAkBO,KAAK0M,MAAMC,IAAI,SACxCD,MAAO1M,KAAK0M,MACZ6P,WAAYs3B,GAAa3gC,QAGzBlT,KAAKqD,QAEb,CAEA,2BAAM4wC,GACF,MAAM9iC,EAAMnR,KAAK4C,SAOjB,WANwBuO,EAAIpE,QAAQ,CAChCtN,MAAO,iBACP4O,QAAS,uBAAuBrO,KAAK0M,MAAMC,IAAI,kCAC/C4c,aAAc,SACdtB,aAAc,gBAEF,OAEhB9W,EAAIqY,cACJ,MAAMvb,QAAajO,KAAK0M,MAAMkd,SAC9BzY,EAAIsY,cACAxb,IAAyB,IAAjBA,EAAKlL,SACboO,EAAI/C,MAAMrL,QAAQ,mBAClB/C,KAAKwF,KAAK,UAAW,CAAEkH,MAAO1M,KAAK0M,SAEnCyE,EAAI/C,MAAM9K,MAAM,2BAExB,EAGJqwC,QAAQz2B,WAAa42B,YC5HrBH,QAAQ9pB,SAAWgqB,GAAavnC,OAChCqnC,QAAQ7pB,UAAY+pB,GAAa3gC,KAEjC,MAAMghC,yBAAyB92B,EAC3B,WAAA9d,CAAYC,EAAU,IAClBC,MAAM,IACCD,EACHyQ,KAAM,iBACNqN,SAAU,WACVlX,OAAQ,iBACRmX,WAAYs2B,YAEZvyB,cAAeyyB,YACft2B,kBAAmB,CACfpa,QAAQ,EACRoN,KAAM,MAGV4J,QAAS,CACL,CAAElS,IAAK,KAAMvI,MAAO,KAAMib,MAAO,OAAQN,UAAU,EAAM/G,MAAO,cAChE,CAAErL,IAAK,MAAOvI,MAAO,MAAO2a,UAAU,GACtC,CAAEpS,IAAK,gBAAiBvI,MAAO,QAAS0a,UAAW,gBACnD,CAAEnS,IAAK,aAAcvI,MAAO,QAAS2a,UAAU,EAAMD,UAAW,qBAChE,CACInS,IAAK,YACLvI,MAAO,SACP0a,UAAW,oEACXO,MAAO,SAEX,CAAE1S,IAAK,UAAWvI,MAAO,UAAW0a,UAAW,WAAYC,UAAU,IAGzEmD,aAAc,CACVlI,KAAM,OAGVqI,YAAY,EACZC,YAAY,EACZvD,UAAU,EACVwD,YAAY,EACZC,WAAW,EAEXC,aAAa,EACbC,SAAS,EACTC,YAAY,EAEZyD,eAAgB,cAEhBxD,aAAc,qBAEdE,aAAc,CACVC,SAAS,EACTC,UAAU,EACVC,OAAO,EACPC,YAAY,IAGxB,ECzDJ,MAAM01B,6BAA6B/2B,EAC/B,WAAA9d,CAAYC,EAAU,IAClBC,MAAM,IACCD,EACHyQ,KAAM,sBACNqN,SAAU,0BACVlX,OAAQ,sBACRmX,WAAY82B,EAEZrxB,WAAYsxB,EAAiB/nC,OAC7B8U,SAAUizB,EAAiBnhC,KAG3BkH,QAAS,CACL,CACIlS,IAAK,KACLvI,MAAO,KACP2a,UAAU,EACV/G,MAAO,cAEX,CACIrL,IAAK,OACLvI,MAAO,OACP0a,UAAW,8BAEf,CACInS,IAAK,cACLvI,MAAO,cACP2a,UAAU,GAEd,CACIpS,IAAK,aACLvI,MAAO,UACP0a,UAAW,iBAEf,CACInS,IAAK,YACLvI,MAAO,SACP0a,UAAW,iBAEf,CACInS,IAAK,YACLvI,MAAO,SACP0a,UAAW,iBAEf,CACInS,IAAK,eACLvI,MAAO,OACP0a,UAAW,sBAEf,CACInS,IAAK,UACLvI,MAAO,UACP0a,UAAW,mBAInB+D,YAAa,CACT,CAAExe,KAAM,YAAaC,OAAQ,OAAQF,MAAO,aAC5C,CAAEC,KAAM,YAAaC,OAAQ,mBAAoBF,MAAO,oBACxD,CAAEC,KAAM,YAAaC,OAAQ,cAAeF,MAAO,eACnD,CAAE40B,SAAS,GACX,CAAE30B,KAAM,UAAWC,OAAQ,QAASF,MAAO,iBAC3C,CAAE40B,SAAS,GACX,CAAE30B,KAAM,WAAYC,OAAQ,kBAAmBF,MAAO,mBACtD,CAAEC,KAAM,qBAAsBC,OAAQ,aAAcF,MAAO,cAC3D,CAAEC,KAAM,YAAaC,OAAQ,WAAYF,MAAO,aAIpDie,YAAY,EACZC,YAAY,EACZvD,UAAU,EACVwD,YAAY,EACZC,WAAW,EAGXC,aAAa,EACbC,SAAS,EACTC,YAAY,EAGZC,aAAc,0FAGdmD,iBAAkB,MAClBC,aAAc,CACV,CAAE5hB,MAAO,SAAUC,KAAM,cAAeC,OAAQ,gBAChD,CAAEF,MAAO,SAAUC,KAAM,iBAAkBC,OAAQ,gBACnD,CAAEF,MAAO,WAAYC,KAAM,qBAAsBC,OAAQ,kBACzD,CAAEF,MAAO,aAAcC,KAAM,iBAAkBC,OAAQ,qBAI3Dwe,aAAc,CACVC,SAAS,EACTC,UAAU,EACVC,OAAO,EACPC,YAAY,IAGxB,CAEA,wBAAM61B,CAAmB9vC,EAAOC,GAC5B,MAAMka,EAAO3e,KAAKkS,WAAWvF,IAAIlI,EAAQkG,QAAQqH,IAC3CiY,QAAend,EAAOkG,cAAc,CACtCvT,MAAO,cACPiN,MAAOiS,EACPnQ,OAAQ6lC,EAAiBE,OAAO/lC,SAEpC,IAAKyb,EAAQ,OAAO,EAChBA,EAAOlnB,QACP/C,KAAK4C,SAASwL,MAAMrL,QAAQ,+BAE5B/C,KAAK4C,SAASwL,MAAM9K,MAAM,uBAElC,CAEA,uBAAMkxC,CAAkBhwC,EAAOC,GAC3B,MAAMka,EAAO3e,KAAKkS,WAAWvF,IAAIlI,EAAQkG,QAAQqH,IAC3CiY,QAAetL,EAAKzQ,KAAK,CAACumC,YAAY,IAW5C,OAVIxqB,EAAOlnB,SAAWknB,EAAOjnB,KAAKC,aAExB6J,EAAO4nC,SAAS,CAClBj1C,MAAO,kBAAkBkf,EAAKC,EAAE5O,OAChChN,KAAMinB,EAAOjnB,KACbwN,KAAM,OAGVxQ,KAAK4C,SAASwL,MAAM9K,MAAM,2BAEvB,CACX,CAEA,4BAAMqxC,CAAuBnwC,EAAOC,GAChC,MAAMka,EAAO3e,KAAKkS,WAAWvF,IAAIlI,EAAQkG,QAAQqH,IAC3CiY,QAAetL,EAAKzQ,KAAK,CAAC0mC,iBAAiB,IAMjD,OALI3qB,EAAOlnB,SAAWknB,EAAOjnB,KAAKC,OAC9BjD,KAAK4C,SAASwL,MAAMrL,QAAQ,8BAE5B/C,KAAK4C,SAASwL,MAAM9K,MAAM,2BAEvB,CACX,CAEA,6BAAMuxC,CAAwBrwC,EAAOC,GACjC,MAAMka,EAAO3e,KAAKkS,WAAWvF,IAAIlI,EAAQkG,QAAQqH,IAC3CiY,QAAend,EAAOkG,cAAc,CACtCvT,MAAO,mBACPiN,MAAOiS,EACPnQ,OAAQ6lC,EAAiBxe,YAAYrnB,SAGzC,OAAKyb,IAEDA,EAAOlnB,SAAWknB,EAAOjnB,KAAKC,OAC9BjD,KAAK4C,SAASwL,MAAMrL,QAAQ,oCAE5B/C,KAAK4C,SAASwL,MAAM9K,MAAM,iCAEvB,EACX,CAEA,mBAAMwxC,CAActwC,EAAOC,GAOvB,WALwBqI,EAAOm2B,YAAY,CACvCxjC,MAAO,qBACP4O,QAAS,yDAIT,OAAO,EAGX,MAAMsQ,EAAO3e,KAAKkS,WAAWvF,IAAIlI,EAAQkG,QAAQqH,IAC3CiY,QAAetL,EAAKzQ,KAAK,CAAC6mC,OAAO,IAOvC,OANI9qB,EAAOlnB,SAAWknB,EAAOjnB,KAAKC,QAC9BjD,KAAK4C,SAASwL,MAAMrL,QAAQ,kCAC5B/C,KAAKkS,WAAWI,SAEhBtS,KAAK4C,SAASwL,MAAM9K,MAAM,+BAEvB,CACX,EC5KJ,MAAM0xC,iBAAiB31C,EACnB,WAAAC,CAAYC,EAAU,IAClBC,MAAM,CACFO,UAAW,eACRR,IAGPS,KAAK0M,MAAQnN,EAAQmN,OAAS,IAAIuoC,EAAK11C,EAAQyD,MAAQ,IACvDhD,KAAKk1C,QAAyC,UAA/Bl1C,KAAK0M,MAAMC,IAAI,YAG9B,MAAMwoC,EAAiBn1C,KAAK0M,MAAMC,IAAI,eAAiB,CAAA,EACvD3M,KAAKo1C,qBAAuB,IAAI93B,EAAWpa,OAAOyG,OAAOwrC,IAEzDn1C,KAAKwM,SAAW,8xFAiDpB,CAEA,YAAM5L,GAEFZ,KAAKq1C,SAAW,IAAInwB,GAAS,CACzBxY,MAAO1M,KAAK0M,MACZ3M,UAAW,MACXolB,iBAAiB,EACjBC,eAAgB,IAChBhL,QAAS,EACT5L,OAAQ,CACJ,CAAEwB,KAAM,KAAMrQ,MAAO,MACrB,CAAEqQ,KAAM,WAAYrQ,MAAO,YAC3B,CAAEqQ,KAAM,mBAAoBrQ,MAAO,oBACnC,CAAEqQ,KAAM,eAAgBrQ,MAAO,gBAC/B,CAAEqQ,KAAM,YAAarQ,MAAO,YAAaic,OAAQ,YACjD,CAAE5L,KAAM,WAAYrQ,MAAO,YAC3B,CAAEqQ,KAAM,gBAAiBrQ,MAAO,SAAUic,OAAQ,SAClD,CAAE5L,KAAM,UAAWrQ,MAAO,UAAWic,OAAQ,YAC7C,CAAE5L,KAAM,WAAYrQ,MAAO,WAAYic,OAAQ,YAC/C,CAAE5L,KAAM,oBAAqBrQ,MAAO,eACpC,CAAEqQ,KAAM,oBAAqBrQ,MAAO,mBACpC,CAAEqQ,KAAM,oBAAqBrQ,MAAO,gBACpC,CAAEqQ,KAAM,MAAOrQ,MAAO,aAAcic,OAAQ,OAC5C,CAAE5L,KAAM,YAAarQ,MAAO,YAAaic,OAAQ,cAKzD5b,KAAKs1C,eAAiB,IAAIp7B,GAAU,CAChChI,WAAYlS,KAAKo1C,qBACjBh7B,QAAS,CACL,CAAElS,IAAK,OAAQvI,MAAO,OAAQ0a,UAAW,SACzC,CAAEnS,IAAK,WAAYvI,MAAO,WAAY0a,UAAW,gBACjD,CAAEnS,IAAK,YAAavI,MAAO,OAAQ0a,UAAW,YAC9C,CAAEnS,IAAK,eAAgBvI,MAAO,gBAC9B,CACIuI,IAAK,UACLvI,MAAO,UACP6M,SAAU,0bAatB,MAAMkZ,EAAO,CAAE6vB,KAAQv1C,KAAKq1C,UAC5B3vB,EAAiB,WAAI1lB,KAAKs1C,eAE1Bt1C,KAAKymB,QAAU,IAAIC,EAAQ,CACvBhB,OACAiB,UAAW,OACXxkB,YAAa,cAEjBnC,KAAKoC,SAASpC,KAAKymB,SAGnB,MAAM+uB,EAAW,IAAIr5B,EAAY,CAC7Bha,YAAa,oBACbpC,UAAW,yCACXqc,QAASpc,KAAK0M,MACdhF,OAAQ,CACJ9H,KAAM,yBACNwJ,MAAO,CACH,CAAEzJ,MAAO,OAAQE,OAAQ,YAAaD,KAAM,UAC5C,CAAED,MAAO,WAAYE,OAAQ,gBAAiBD,KAAM,kBACpD,CAAED,MAAO,eAAgBE,OAAQ,YAAaD,KAAM,gBACpD,CAAEkI,KAAM,WACR9H,KAAK0M,MAAMC,IAAI,aACT,CAAEhN,MAAO,eAAgBE,OAAQ,eAAgBD,KAAM,cACvD,CAAED,MAAO,cAAeE,OAAQ,cAAeD,KAAM,gBAC3D,CAAEkI,KAAM,WACR,CAAEnI,MAAO,cAAeE,OAAQ,cAAeD,KAAM,cAAeqgB,QAAQ,OAIxFjgB,KAAKoC,SAASozC,EAClB,CAEA,sBAAMC,GACF,MAAMC,EAAc11C,KAAK0M,MAAMC,IAAI,gBAC7BgpC,EAAU31C,KAAK0M,MAAMC,IAAI,OAE/B,GAAI+oC,EAAY5nB,WAAW,UAAW,CAClC,MAAM8nB,EAAa51C,KAAK0M,MAAMC,IAAI,eAAiB,CAAA,EAC7CkpC,EAAS,CACX,CAAEC,IAAKH,EAASI,IAAK,eAClB7yC,OAAOyG,OAAOisC,GAAYhtC,IAAIk/B,IAAA,CAAQgO,IAAKhO,EAAEhgB,IAAKiuB,IAAKjO,EAAEkO,SAEhEC,GAAgBvxB,KAAKmxB,EAAQ,CAAEK,aAAa,GAChD,KAA2B,oBAAhBR,EACPS,GAAU/iC,WAAWuiC,EAAS,CAAEl2C,MAAOO,KAAK0M,MAAMC,IAAI,cAEtDlD,OAAOse,KAAK4tB,EAAS,SAE7B,CAEA,0BAAMS,GACF,MAAMtuB,EAAM9nB,KAAK0M,MAAMC,IAAI,OAC3B,GAAImb,EAAK,CACL,MAAMif,EAAI1e,SAAS2e,cAAc,KACjCD,EAAEE,KAAOnf,EACTif,EAAEG,SAAWlnC,KAAK0M,MAAMC,IAAI,YAC5B0b,SAAShV,KAAK8zB,YAAYJ,GAC1BA,EAAEK,QACF/e,SAAShV,KAAKg0B,YAAYN,EAC9B,CACJ,CAEA,sBAAMsP,SACiBvpC,EAAOkG,cAAc,CACpCvT,MAAO,eAAeO,KAAK0M,MAAMC,IAAI,cACrCD,MAAO1M,KAAK0M,MACZ6P,WAAY+5B,EAAUpjC,QAGtBlT,KAAKqD,QAEb,CAEA,wBAAMkzC,SACIv2C,KAAK0M,MAAMwB,KAAK,CAAEsoC,WAAW,IACnCx2C,KAAKqD,QACT,CAEA,yBAAMozC,SACIz2C,KAAK0M,MAAMwB,KAAK,CAAEsoC,WAAW,IACnCx2C,KAAKqD,QACT,CAEA,wBAAMqzC,SACsB5pC,EAAOC,QAC3B,6CAA6C/M,KAAK0M,MAAMC,IAAI,8CAC5D,mBACA,CAAEsb,aAAc,aAAcC,YAAa,mBAGxBloB,KAAK0M,MAAM9C,WACrB7G,SACL/C,KAAKwF,KAAK,eAAgB,CAAEkH,MAAO1M,KAAK0M,OAGpD,EAGJuoC,EAAK/3B,WAAa83B,SChOlB,MAAM2B,sBAAsBv5B,EACxB,WAAA9d,CAAYC,EAAU,IAClBC,MAAM,CACFwQ,KAAM,cACNqN,SAAU,eACVlX,OAAQ,cACRmX,WAAYs5B,EAGZx1B,SAAUk1B,EAAUpjC,KACpBmO,cAAe2zB,SAGfpzB,MAAO/O,MAAOrO,UACJxE,KAAK62C,iBAAiBryC,IAGhCgZ,kBAAmB,CACfpa,QAAQ,EACRoN,KAAM,MAIV4J,QAAS,CACL,CACIlS,IAAK,KACLvI,MAAO,KACP2a,UAAU,EACV/G,MAAO,cAEX,CACIrL,IAAK,WACLvI,MAAO,YAEX,CACIuI,IAAK,eACLvI,MAAO,OACP0a,UAAW,sBAEf,CACInS,IAAK,YACLvI,MAAO,OACP0a,UAAW,YAEf,CACInS,IAAK,aACLvI,MAAO,QACP0a,UAAW,uBAEf,CACInS,IAAK,gBACLvI,MAAO,SACP0a,UAAW,SAEf,CACInS,IAAK,UACLvI,MAAO,WACP0a,UAAW,mBAKnBuD,YAAY,EACZC,YAAY,EACZvD,UAAU,EACVwD,YAAY,EACZC,WAAW,EAGXC,aAAa,EACbC,SAAS,EACTC,YAAY,EAGZC,aAAc,8DAGdmD,iBAAkB,MAClBC,aAAc,CACV,CAAE5hB,MAAO,WAAYC,KAAM,iBAAkBC,OAAQ,kBACrD,CAAEF,MAAO,SAAUC,KAAM,cAAeC,OAAQ,gBAChD,CAAEF,MAAO,gBAAiBC,KAAM,eAAgBC,OAAQ,eAI5Dwe,aAAc,CACVC,SAAS,EACTC,UAAU,EACVC,OAAO,EACPC,YAAY,MAEblf,IAIPS,KAAK82C,eAAe,CAChBC,cAAe,CAAC,OAChBC,YAAa,UACbC,UAAU,EACVC,gBAAgB,GAExB,CAMA,sBAAML,CAAiBryC,GACfA,KAAayG,iBAGjB,MAAMksC,EAAY9uB,SAAS2e,cAAc,SACzCmQ,EAAUrvC,KAAO,OACjBqvC,EAAUC,OAAS,MACnBD,EAAUF,UAAW,EACrBE,EAAUj4B,MAAMm4B,QAAU,OAG1BF,EAAU7rC,iBAAiB,SAAUuH,MAAON,IACxC,MAAM+kC,EAAO/kC,EAAEglC,OAAOxlB,MAAM,GAE5B,IAAKulB,EACD,OAIJ,MAAME,EAAU,UAChB,GAAIF,EAAK9mC,KAAOgnC,EACZx3C,KAAKkqB,UAAU,cAAclqB,KAAKy3C,gBAAgBH,EAAK9mC,2BAA2BxQ,KAAKy3C,gBAAgBD,YAK3G,IACI,MAAME,EAAY,IAAIzC,EACtB,IAAI0C,EAAQ,CAAA,EACR33C,KAAKT,QAAQq4C,eAAiB53C,KAAK4C,SAASi1C,cAC5CF,EAAMj2B,MAAQ1hB,KAAK4C,SAASi1C,YAAY7lC,IAG5C,MAAM8lC,EAASJ,EAAUI,OAAO,CAC5BR,OACAtnC,KAAMsnC,EAAKtnC,KACX+nC,YAAa,oCAAA,IAAwBj0C,MAAO0S,uBAC5Cse,WAAW,EACXkjB,WAAaC,IACuBA,EAAaC,YAEjDC,WAAaluB,IAETjqB,KAAKiF,WAETmzC,QAAU90C,IACNC,QAAQD,MAAM,iBAAkBA,GAChCtD,KAAKkqB,UAAU,kBAAoB5mB,EAAM+K,aAE1CspC,UAGDG,CACV,OAASx0C,GACLC,QAAQD,MAAM,8BAA+BA,GAC7CtD,KAAKkqB,UAAU,gCAAkC5mB,EAAM+K,QAC3D,CAAA,QAEI8oC,EAAUrxC,QACd,IAIJuiB,SAAShV,KAAK8zB,YAAYgQ,GAC1BA,EAAU/P,OACd,CAQA,eAAAqQ,CAAgBjQ,GACZ,GAAc,IAAVA,EAAa,MAAO,UACxB,MAEMrc,EAAI1f,KAAKy2B,MAAMz2B,KAAKu6B,IAAIwB,GAAS/7B,KAAKu6B,IAFlC,OAGV,OAAO6I,YAAYrH,EAAQ/7B,KAAK4sC,IAHtB,KAG6BltB,IAAIsc,QAAQ,IAAM,IAF3C,CAAC,QAAS,KAAM,KAAM,MAEiCtc,EACzE,CAEA,gBAAMmtB,CAAWvmB,EAAOvtB,EAAO+zC,GAC3B,MAAMjB,EAAOvlB,EAAM,GACUulB,EAAKtnC,KAASsnC,EAAKxvC,KAAUwvC,EAAK9mC,KAE/D,IAEI,MAAMknC,EAAY,IAAIzC,EACtB,IAAI0C,EAAQ,CAAA,EACR33C,KAAKT,QAAQq4C,eAAiB53C,KAAK4C,SAASi1C,cAC5CF,EAAMj2B,MAAQ1hB,KAAK4C,SAASi1C,YAAY7lC,IAI5C,MAAM8lC,EAASJ,EAAUI,OAAO,CAC5BR,OACAtnC,KAAMsnC,EAAKtnC,KACX+nC,YAAa,oDAAA,IAAwCj0C,MAAO0S,uBAC5Dse,WAAW,EACXkjB,WAAaC,IACuBA,EAAaC,YAEjDC,WAAaluB,IAETjqB,KAAKiF,WAETmzC,QAAU90C,IACNC,QAAQD,MAAM,iBAAkBA,GAChCtD,KAAKkqB,UAAU,kBAAoB5mB,EAAM+K,aAE1CspC,UAGDG,CACV,OAASx0C,GACLC,QAAQD,MAAM,8BAA+BA,GAC7CtD,KAAKkqB,UAAU,gCAAkC5mB,EAAM+K,QAC3D,CACJ,EAIJmqC,GAAmB7B,eCvOnB,MAAM8B,0BAA0Br7B,EAC5B,WAAA9d,CAAYC,EAAU,IAClBC,MAAM,IACCD,EACHyQ,KAAM,mBACNqN,SAAU,oBACVlX,OAAQ,mBACRmX,WAAYo7B,GAEZ31B,WAAY41B,GAAcrsC,OAC1B8U,SAAUu3B,GAAczlC,KAGxBkH,QAAS,CACL,CACIlS,IAAK,KACLvI,MAAO,KACPib,MAAO,OACPN,UAAU,EACV/G,MAAO,cAEX,CACIrL,IAAK,OACLvI,MAAO,cACP2a,UAAU,GAEd,CACIpS,IAAK,UACLvI,MAAO,UACP0a,UAAW,mBAKnBuD,YAAY,EACZC,YAAY,EACZvD,UAAU,EACVwD,YAAY,EACZC,WAAW,EAGXC,aAAa,EACbC,SAAS,EACTC,YAAY,EAGZC,aAAc,0EAGdmD,iBAAkB,MAClBC,aAAc,CACV,CAAE5hB,MAAO,SAAUC,KAAM,cAAeC,OAAQ,gBAChD,CAAEF,MAAO,SAAUC,KAAM,iBAAkBC,OAAQ,gBACnD,CAAEF,MAAO,cAAeC,KAAM,eAAgBC,OAAQ,gBACtD,CAAEF,MAAO,eAAgBC,KAAM,aAAcC,OAAQ,iBACrD,CAAEF,MAAO,eAAgBC,KAAM,eAAgBC,OAAQ,gBAI3Dwe,aAAc,CACVC,SAAS,EACTC,UAAU,EACVC,OAAO,EACPC,YAAY,IAGxB,ECvDJ,MAAMm6B,+BAA+Bv5C,EACjC,WAAAC,CAAYC,EAAU,IAClBC,MAAM,CACFO,UAAW,cACRR,IAGPS,KAAK0M,MAAQnN,EAAQmN,OAAS,IAAImsC,EAAmBt5C,EAAQyD,MAAQ,IAGrEhD,KAAK84C,IAAM94C,KAAK0M,MAAMC,IAAI,gBAAkB,CAAA,EAC5C3M,KAAK+4C,IAAM/4C,KAAK84C,IAAIzgC,aAAe,CAAA,EACnCrY,KAAKg5C,KAAOh5C,KAAK0M,MAAMC,IAAI,gBAAkB,CAAA,EAG7C3M,KAAKmX,WAAanX,KAAKi5C,iBACvBj5C,KAAK2jB,YAAc3jB,KAAK4jB,cACxB5jB,KAAK6jB,OAAS7jB,KAAK8jB,SACnB9jB,KAAK+jB,WAAa/jB,KAAKgkB,aACvBhkB,KAAKk5C,gBAAkBl5C,KAAKm5C,sBAC5Bn5C,KAAKo5C,YAAcp5C,KAAKg5C,KAAKhgB,cAAgB,GAC7Ch5B,KAAKq5C,YAAcr5C,KAAKg5C,KAAKM,cAAgB,UAC7Ct5C,KAAKu5C,YAAcv5C,KAAKw5C,kBACxBx5C,KAAKilB,kBAAoBjlB,KAAKg5C,KAAKS,WAAYz5C,KAAKg5C,KAAKU,WAEzD15C,KAAKwM,SAAW,s8FA8CpB,CAIA,cAAAysC,GACI,MAAM30B,EAAUtkB,KAAK+4C,KAAK/gC,YAAYP,QAAQzG,eAAiB,GACzDsG,EAAKtX,KAAK+4C,KAAKzhC,IAAIG,QAAQzG,eAAiB,GAC5CqG,EAASrX,KAAK+4C,KAAK1hC,QAAQI,QAAQzG,eAAiB,GAC1D,OAAIsT,EAAQ5M,SAAS,UAAkB,oBACnC4M,EAAQ5M,SAAS,WAAmB,qBACpC4M,EAAQ5M,SAAS,UAAkB,oBACnC4M,EAAQ5M,SAAS,QAAgB,kBACjCJ,EAAGI,SAAS,QAAUJ,EAAGI,SAAS,OAAe,WACjDJ,EAAGI,SAAS,WAAmB,aAC/BJ,EAAGI,SAAS,WAAmB,cAC/BL,EAAOK,SAAS,UAAkB,WAClCL,EAAOK,SAAS,QAAgB,YAC7B,YACX,CAEA,WAAAkM,GACI,MAAM7L,EAAK/X,KAAK+4C,KAAK/gC,YAAc,CAAA,EACnC,OAAOD,EAAGN,OAAS,GAAGM,EAAGN,UAAUM,EAAGE,OAAS,KAAKvK,OAAS,iBACjE,CAEA,MAAAoW,GACI,MAAMxM,EAAKtX,KAAK+4C,KAAKzhC,IAAM,CAAA,EACrB8M,EAAM,CAAC9M,EAAGW,MAAOX,EAAG+M,OAAOnf,OAAOC,SAAS4D,KAAK,KACtD,OAAOuO,EAAGG,OAAS,GAAGH,EAAGG,UAAU2M,IAAM1W,OAAS,YACtD,CAEA,UAAAsW,GACI,MAAM5M,EAAMpX,KAAK+4C,KAAK1hC,QAAU,CAAA,EAC1B8M,EAAQ,CAAC/M,EAAIQ,MAAOR,EAAIK,QAAQvS,OAAOC,SAC7C,OAAOgf,EAAM9O,OAAS8O,EAAMpb,KAAK,KAAO,gBAC5C,CAEA,mBAAAowC,GACI,MAAMh1B,EAAQ,CAACnkB,KAAKg5C,KAAKzpC,KAAMvP,KAAKg5C,KAAKxgC,OAAQxY,KAAKg5C,KAAKvgC,cAAcvT,OAAOC,SAChF,OAAOgf,EAAM9O,OAAS8O,EAAMpb,KAAK,MAAQ,kBAC7C,CAEA,eAAAywC,GACI,MAAMtoB,GAASlxB,KAAKg5C,KAAKM,cAAgB,IAAItoC,cAC7C,MAAc,SAAVkgB,GAAoBlxB,KAAKg5C,KAAKW,UAAkB,cACtC,WAAVzoB,GAAsBlxB,KAAKg5C,KAAKY,cAAsB,eAC5C,QAAV1oB,EAAwB,eACrB,YACX,CAEA,YAAMtwB,GACF,MAAM2X,EAAMvY,KAAKg5C,KACXa,EAAK75C,KAAK+4C,IAqKVnyC,EAAW,CACb,CAAEsB,IAAK,WAAYvI,MAAO,WAAYC,KAAM,aAAcuI,KAnKzC,IAAI9I,EAAK,CAC1BqN,MAAO1M,KAAK0M,MACZF,SAAU,y6BAa6B+L,EAAIhJ,MAAQ,mMAIZgJ,EAAIC,QAAU,oMAIdD,EAAIE,cAAgB,OAAOF,EAAIygB,aAAe,6BAA6BzgB,EAAIygB,uBAAyB,uMAIxGzgB,EAAIuhC,aAAe,qMAInBvhC,EAAIpJ,UAAY,sDAEjDoJ,EAAIkhC,SAAW,uKAGkBlhC,EAAIkhC,aAAalhC,EAAImhC,0CAC9C,wfASyBnhC,EAAI8K,KAAO,gMAIX9K,EAAIwhC,KAAO,OAAOxhC,EAAI+K,QAAU,mCAAmC/K,EAAI+K,kBAAoB,ulBAkHlI,CAAEpb,IAAK,SAAUvI,MAAO,SAAUC,KAAM,YAAauI,KAlGtC,IAAI9I,EAAK,CACxBqN,MAAO1M,KAAK0M,MACZF,SAAU,ypCAc6BqtC,GAAI7hC,YAAYP,QAAU,oMAI1B,CAACoiC,GAAI7hC,YAAYC,MAAO4hC,GAAI7hC,YAAYqM,MAAOw1B,GAAI7hC,YAAYgiC,OAAO90C,OAAOC,SAAS4D,KAAK,MAAQ,0QAMnG8wC,GAAIviC,IAAIG,QAAU,oMAIlB,CAACoiC,GAAIviC,IAAIW,MAAO4hC,GAAIviC,IAAI+M,MAAOw1B,GAAIviC,IAAI0iC,OAAO90C,OAAOC,SAAS4D,KAAK,MAAQ,mQAM3E8wC,GAAIxiC,QAAQO,OAAS,mMAIrBiiC,GAAIxiC,QAAQI,QAAU,kMAItBoiC,GAAIxiC,QAAQ3K,OAAS,wDAGtDmtC,GAAII,OAAS,wHAEcJ,EAAGI,eAAiB,sBAkDrD,CAAE/xC,IAAK,OAAQvI,MAAO,OAAQC,KAAM,wBAAyBuI,KA7ChD,IAAI9I,EAAK,CACtBqN,MAAO1M,KAAK0M,MACZF,SAAU,wnCAgB4BxM,KAAKu5C,0CAA2ChhC,EAAI+gC,cAAgB,6MAIjD,MAAlB/gC,EAAI2hC,WAAqB3hC,EAAI2hC,WAAa,8HAI3El6C,KAAKm6C,SAAS,MAAO,YAAa5hC,EAAIM,4BACtC7Y,KAAKm6C,SAAS,gBAAiB,iBAAkB5hC,EAAIO,4BACrD9Y,KAAKm6C,SAAS,QAAS,eAAgB5hC,EAAIQ,8BAC3C/Y,KAAKm6C,SAAS,iBAAkB,WAAY5hC,EAAI6hC,8BAChDp6C,KAAKm6C,SAAS,aAAc,eAAgB5hC,EAAI8hC,mCAChDr6C,KAAKm6C,SAAS,SAAU,WAAY5hC,EAAI+hC,kGAGxCt6C,KAAKm6C,SAAS,iBAAkB,0BAA2B5hC,EAAIgiC,uCAC/Dv6C,KAAKm6C,SAAS,eAAgB,UAAW5hC,EAAIiiC,qCAC7Cx6C,KAAKm6C,SAAS,SAAU,wBAAyB5hC,EAAIohC,+BACrD35C,KAAKm6C,SAAS,aAAc,qBAAsB5hC,EAAIqhC,mCAYhE,GAAI55C,KAAKilB,eACL,IACI,MACMkB,EAAU,IAAIC,SADG5F,OAAO,8CAAqBC,KAAAC,GAAAA,EAAAxX,IAAGuxC,SAC1B,CACxBp0B,QAAS,CAAC,CACNL,IAAKhmB,KAAKg5C,KAAKS,SACfxzB,IAAKjmB,KAAKg5C,KAAKU,UACfpzB,MAAO,WAAWtmB,KAAK0M,MAAMC,IAAI,6BAA6B3M,KAAKk5C,oBAEvE3yB,UAAW,QACXC,KAAM,EACN9kB,OAAQ,MAEZkF,EAASqB,KAAK,CAAEC,IAAK,MAAOvI,MAAO,MAAOC,KAAM,SAAUuI,KAAMge,GACpE,OAAS5T,GAET,CAIJ,MAAMkT,EAAKzlB,KAAK0M,MAAMC,IAAI,cAC1B,GAAI8Y,EAAI,CACJ,MAAM9K,EAAa,IAAIT,GAAU,CAC7BhI,WAAY,IAAIsI,EAAkB,CAC9BpI,OAAQ,CAAE5B,KAAM,GAAIgV,UAAWC,KAEnCtL,oBAAqB,CAAC,aACtBC,QAAS,CACL,CAAElS,IAAK,UAAWvI,MAAO,OAAQ0a,UAAW,WAAYC,UAAU,EAAMM,MAAO,SAC/E,CAAE1S,IAAK,iBAAkBvI,MAAO,YAChC,CAAEuI,IAAK,QAASvI,MAAO,YAG/BiH,EAASqB,KAAK,CAAEH,KAAM,UAAWnI,MAAO,aACxCiH,EAASqB,KAAK,CAAEC,IAAK,SAAUvI,MAAO,SAAUC,KAAM,oBAAqBuI,KAAMwS,IAEjF,MAAMa,EAAW,IAAItB,GAAU,CAC3BhI,WAAY,IAAIqJ,GAAQ,CACpBnJ,OAAQ,CAAE5B,KAAM,GAAIiV,QAExB1d,YAAa,YACboS,oBAAqB,CAAC,MACtBC,QAAS,CACL,CAAElS,IAAK,UAAWvI,MAAO,YAAa2a,UAAU,EAAMD,UAAW,kBACjE,CAAEnS,IAAK,QAASvI,MAAO,QAAS2a,UAAU,GAC1C,CAAEpS,IAAK,OAAQvI,MAAO,QACtB,CAAEqQ,KAAM,MAAOrQ,MAAO,UAG9BiH,EAASqB,KAAK,CAAEC,IAAK,OAAQvI,MAAO,OAAQC,KAAM,kBAAmBuI,KAAMqT,EAAUzT,YAAa,aACtG,CAGA/H,KAAKmZ,YAAc,IAAIxS,YAAY,CAC/BxE,YAAa,cACb0E,cAAe7G,KAAKilB,eAAiB,MAAQ,WAC7Cne,SAAU,IACVC,eAAgB,cAChBC,kBAAkB,EAClBC,SAAU,IACVL,aAEJ5G,KAAKoC,SAASpC,KAAKmZ,aAGnB,MAAMuhC,EAAO,IAAIv+B,EAAY,CACzBha,YAAa,mBACbpC,UAAW,yCACXqc,QAASpc,KAAK0M,MACdhF,OAAQ,CACJ9H,KAAM,yBACNwJ,MAAO,IACCpJ,KAAK84C,KAAKzmC,KAAO,CAAC,CAAE1S,MAAO,YAAaE,OAAQ,YAAaD,KAAM,cAAiB,MACpFI,KAAK84C,KAAK9mC,GAAK,CAAC,CAAErS,MAAO,cAAeE,OAAQ,cAAeD,KAAM,cAAiB,MACtFI,KAAKilB,eAAiB,CAAC,CACvBtlB,MAAO,eACPE,OAAQ,eACRD,KAAM,0BACL,GACL,CAAEkI,KAAM,WACR,CAAEnI,MAAO,gBAAiBE,OAAQ,gBAAiBD,KAAM,WAAYqgB,QAAQ,OAIzFjgB,KAAKoC,SAASs4C,EAClB,CAIA,QAAAP,CAASx6C,EAAOC,EAAM+Q,GAKlB,MAAO,wGAE6C/Q,UANxC+Q,EAAQ,eAAiB,sBAMwChR,yDAL3DgR,EACZ,wEACA,2FAMV,CAIA,sBAAM2P,GACF,MAAMC,EAASvgB,KAAK84C,KAAKzmC,MAAML,IAAMhS,KAAK84C,KAAKzmC,KAC3CkO,GAAQvgB,KAAKwF,KAAK,YAAa,CAAE+a,UACzC,CAEA,wBAAM0yB,GACF,MAAM0H,EAAW36C,KAAK84C,KAAK9mC,GACvB2oC,GAAU36C,KAAKwF,KAAK,cAAe,CAAEm1C,YAC7C,CAEA,wBAAMC,GACE56C,KAAKilB,gBACLxb,OAAOse,KAAK,mDAAmD/nB,KAAKg5C,KAAKS,YAAYz5C,KAAKg5C,KAAKU,YAAa,SAEpH,CAEA,0BAAMmB,GAKF,cAJwB/tC,EAAOC,QAC3B,wDACA,oCAIe/M,KAAK0M,MAAM9C,WACrB7G,SACL/C,KAAKwF,KAAK,mBAAoB,CAAEkH,MAAO1M,KAAK0M,SAEzC,EACX,CAEA,iBAAagY,CAAK1S,GACd,MAAMtF,EAAQ,IAAImsC,EAAmB,CAAE7mC,OAEvC,aADMtF,EAAM4F,QACR5F,EAAMsF,GACClF,EAAOsG,WAAW,CACrB3T,OAAO,EACP+Q,KAAM,KACN6C,KAAM,IAAIulC,uBAAuB,CAAElsC,UACnC4G,QAAS,CAAC,CAAE1C,KAAM,QAAS2C,MAAO,gBAAiBC,SAAS,OAGpE1G,EAAOnH,MAAM,CAAE0I,QAAS,mCAAmC2D,IAAMlK,KAAM,YAChE,KACX,EAGJ+wC,EAAmB37B,WAAa07B,uBCnchC,MAAMkC,GAAoB,CACtBC,IAAK,CACD,CAAE7yC,IAAK,MAAgBvI,MAAO,kBAAoBorB,KAAM,KACxD,CAAE7iB,IAAK,SAAgBvI,MAAO,eAAoBorB,KAAM,KACxD,CAAE7iB,IAAK,OAAgBvI,MAAO,aAAoBorB,KAAM,KACxD,CAAE7iB,IAAK,SAAgBvI,MAAO,aAAoBorB,KAAM,SACxD,CAAE7iB,IAAK,UAAgBvI,MAAO,cAAoBorB,KAAM,SACxD,CAAE7iB,IAAK,YAAgBvI,MAAO,gBAAoBorB,KAAM,OACxD,CAAE7iB,IAAK,aAAgBvI,MAAO,iBAAoBorB,KAAM,OACxD,CAAE7iB,IAAK,eAAgBvI,MAAO,eAAoBorB,KAAM,KAE5DiwB,IAAK,CACD,CAAE9yC,IAAK,MAAkBvI,MAAO,kBAAqBorB,KAAM,KAC3D,CAAE7iB,IAAK,QAAkBvI,MAAO,qBAAsBorB,KAAM,IAC5D,CAAE7iB,IAAK,eAAkBvI,MAAO,eAAqBorB,KAAM,SAC3D,CAAE7iB,IAAK,cAAkBvI,MAAO,kBAAqBorB,KAAM,SAC3D,CAAE7iB,IAAK,YAAkBvI,MAAO,YAAqBorB,KAAM,SAC3D,CAAE7iB,IAAK,aAAkBvI,MAAO,aAAqBorB,KAAM,SAC3D,CAAE7iB,IAAK,eAAkBvI,MAAO,eAAqBorB,KAAM,KAC3D,CAAE7iB,IAAK,gBAAkBvI,MAAO,gBAAqBorB,KAAM,KAC3D,CAAE7iB,IAAK,SAAkBvI,MAAO,aAAqBorB,KAAM,SAC3D,CAAE7iB,IAAK,UAAkBvI,MAAO,cAAqBorB,KAAM,UAE/DkwB,MAAO,CACH,CAAE/yC,IAAK,MAAmBvI,MAAO,kBAAsBorB,KAAM,KAC7D,CAAE7iB,IAAK,QAAmBvI,MAAO,sBAAuBorB,KAAM,IAC9D,CAAE7iB,IAAK,eAAmBvI,MAAO,oBAAsBorB,KAAM,SAC7D,CAAE7iB,IAAK,aAAmBvI,MAAO,aAAsBorB,KAAM,IAC7D,CAAE7iB,IAAK,eAAmBvI,MAAO,eAAsBorB,KAAM,IAC7D,CAAE7iB,IAAK,kBAAmBvI,MAAO,kBAAuBorB,KAAM,KAC9D,CAAE7iB,IAAK,SAAmBvI,MAAO,aAAsBorB,KAAM,SAC7D,CAAE7iB,IAAK,UAAmBvI,MAAO,cAAsBorB,KAAM,WAI/DmwB,GAAa,CAAEH,IAAK,gBAAiBC,IAAK,cAAeC,MAAO,uBAChEE,GAAc,CAAEJ,IAAK,eAAgBC,IAAK,eAAgBC,MAAO,qBAEvE,SAASjwB,GAAaD,GAClB,MAAa,MAATA,EAAyB,CAAEprB,MAAO,IAAKyE,aAAa,EAAM6mB,IAAK,KACtD,UAATF,EAAyB,CAAEprB,MAAO,QAASyE,aAAa,GAC/C,MAAT2mB,EAAyB,CAAEprB,MAAO,UAAWyE,aAAa,GACvD,CAAEA,aAAa,EAC1B,CAEe,MAAMg3C,+BAA+B/7C,EAChD,WAAAC,CAAYC,EAAU,IAClBC,MAAM,CACFO,UAAW,8BACRR,IAGPS,KAAKsqB,aAAe/qB,EAAQ+qB,cAAgB,MAC5CtqB,KAAKwqB,KAAOjrB,EAAQirB,MAAQ,GAC5BxqB,KAAKq7C,SAAW97C,EAAQ87C,UAAY,CAAA,CACxC,CAEA,iBAAM16C,GACF,MAAM26C,EAAaR,GAAkB96C,KAAKsqB,eAAiB,GACrD1qB,EAAOs7C,GAAWl7C,KAAKsqB,eAAiB,WACxCixB,EAAYJ,GAAYn7C,KAAKsqB,eAAiB,WAEtCtqB,KAAKq7C,SAAS7rC,OAASxP,KAAKq7C,SAASp4C,OACnD,MACMu4C,EADYx7C,KAAKy7C,kBACI7yC,IAAI4O,GAAK,kEAAkEA,YAAYzO,KAAK,IAEvH,MAAO,ugBAUoBnJ,eAAkBI,KAAKwqB,8FAC6B+wB,uEAEnDC,uFAIlBF,EAAW1yC,IAAI,CAACgW,EAAGuM,IAAM,uBAAuBA,aAAapiB,KAAK,mCAGhF,CAEA,YAAMnI,GACF,MAAM06C,EAAaR,GAAkB96C,KAAKsqB,eAAiB,GAE3D,IAAA,IAASa,EAAI,EAAGA,EAAImwB,EAAWjmC,OAAQ8V,IAAK,CACxC,MAAMuwB,EAAMJ,EAAWnwB,GACjBE,EAAQ,IAAIhB,gBAAgB,CAC9BloB,YAAa,cAAcgpB,IAC3B7pB,QAAStB,KAAKsqB,aACdC,SAAUmxB,EAAIxzC,IACdsiB,KAAMxqB,KAAKwqB,KACX/qB,MAAOi8C,EAAI/7C,MACX+B,OAAQ,IACRyC,MAAO6mB,GAAa0wB,EAAI3wB,MACxBO,iBAAiB,EACjBppB,eAAe,EACfuoB,iBAAkB,MAClBvpB,YAAa,UAEjBlB,KAAKoC,SAASipB,EAClB,CACJ,CAEA,eAAAowB,GACI,MAAM3T,EAAI9nC,KAAKq7C,SACf,OAAQr7C,KAAKsqB,cACT,IAAK,MACD,MAAO,CAACwd,EAAE6T,cAAe7T,EAAE8T,WAAY9T,EAAE+T,WAAW32C,OAAOC,SACtDyD,IAAI,CAACysB,EAAGlK,IAEE,gBADO,CAAC,SAAU,iBAAkB,YACdA,gBAAgBkK,KAEzD,IAAK,MACD,MAAO,CAACyS,EAAEgU,OAAQhU,EAAEiU,gBAAgB72C,OAAOC,SACtCyD,IAAI,CAACysB,EAAGlK,IAEE,gBADO,CAAC,cAAe,UACDA,gBAAgBkK,KAEzD,IAAK,QACD,MAAO,CAACyS,EAAEgU,OAAQhU,EAAEkU,UAAWlU,EAAEmU,UAAY,GAAGnU,EAAEmU,iBAAiBnU,EAAEmU,UAAY,EAAI,IAAM,KAAO,IAAI/2C,OAAOC,SACxGyD,IAAI,CAACysB,EAAGlK,IAEE,gBADO,CAAC,eAAgB,SAAU,gBACZA,gBAAgBkK,KAEzD,QACI,MAAO,GAEnB,CAEA,iBAAa3Q,CAAK4F,EAAcE,EAAM6wB,EAAW,CAAA,EAAI97C,EAAU,IAC3D,MAAM4I,EAAO,IAAIizC,uBAAuB,CACpC9wB,eAAcE,OAAM6wB,aAGlBz7C,EAAOs7C,GAAW5wB,IAAiB,WACnCixB,EAAYJ,GAAY7wB,IAAiB,iBAEzCxd,EAAOsG,WAAWjL,EAAM,CAC1B/E,OAAQ,gBAAgBxD,eAAkB4qB,iCAAoC+wB,YAC9E/qC,KAAM,KACN4nB,YAAY,GAEpB,ECfG,SAAS8jB,GAAoB/qC,EAAKgrC,GAAY,GAoCjD,GAlCAhrC,EAAIirC,aAAa,mBAAoBC,mBAAyB,CAACt0C,YAAa,CAAC,gBAC7EoJ,EAAIirC,aAAa,cAAeE,cAAoB,CAACv0C,YAAa,CAAC,YAAa,iBAChFoJ,EAAIirC,aAAa,eAAgBG,cAAoB,CAACx0C,YAAa,CAAC,kBACpEoJ,EAAIirC,aAAa,gBAAiBI,eAAqB,CAACz0C,YAAa,CAAC,mBACtEoJ,EAAIirC,aAAa,iBAAkBK,gBAAsB,CAAC10C,YAAa,CAAC,oBACxEoJ,EAAIirC,aAAa,mBAAoBM,kBAAwB,CAAC30C,YAAa,CAAC,gBAC5EoJ,EAAIirC,aAAa,sBAAuBO,qBAA2B,CAAC50C,YAAa,CAAC,kBAClFoJ,EAAIirC,aAAa,eAAgBQ,cAAoB,CAAC70C,YAAa,CAAC,kBACpEoJ,EAAIirC,aAAa,mBAAoBS,kBAAwB,CAAC90C,YAAa,CAAC,oBAC5EoJ,EAAIirC,aAAa,gBAAiBU,eAAqB,CAAC/0C,YAAa,CAAC,oBACtEoJ,EAAIirC,aAAa,cAAeW,aAAmB,CAACh1C,YAAa,CAAC,eAClEoJ,EAAIirC,aAAa,sBAAuBY,oBAA0B,CAACj1C,YAAa,CAAC,kBACjFoJ,EAAIirC,aAAa,+BAAgCa,4BAAkC,CAACl1C,YAAa,CAAC,kBAClGoJ,EAAIirC,aAAa,sBAAuBc,sBAA4B,CAACn1C,YAAa,CAAC,kBACnFoJ,EAAIirC,aAAa,yBAA0Be,sBAA4B,CAACp1C,YAAa,CAAC,gBACtFoJ,EAAIirC,aAAa,uBAAwBgB,qBAA2B,CAACr1C,YAAa,CAAC,gBACnFoJ,EAAIirC,aAAa,oBAAqBiB,qBAA2B,CAACt1C,YAAa,CAAC,gBAChFoJ,EAAIirC,aAAa,yBAA0BkB,uBAA6B,CAACv1C,YAAa,CAAC,gBACvFoJ,EAAIirC,aAAa,4BAA6BmB,sBAA4B,CAAEx1C,YAAa,CAAC,oBAC1FoJ,EAAIirC,aAAa,kBAAmBoB,iBAAuB,CAAEz1C,YAAa,CAAC,sBAC3EoJ,EAAIirC,aAAa,iBAAkBqB,gBAAsB,CAAE11C,YAAa,CAAC,sBACzEoJ,EAAIirC,aAAa,6BAA8BsB,4BAAkC,CAAE31C,YAAa,CAAC,oBACjGoJ,EAAIirC,aAAa,wBAAyBuB,kBAAwB,CAAE51C,YAAa,CAAC,kBAClFoJ,EAAIirC,aAAa,sBAAuBwB,oBAA0B,CAAE71C,YAAa,CAAC,kBAClFoJ,EAAIirC,aAAa,wBAAyByB,sBAA4B,CAAE91C,YAAa,CAAC,kBACtFoJ,EAAIirC,aAAa,yBAA0B0B,sBAA4B,CAAE/1C,YAAa,CAAC,kBACvFoJ,EAAIirC,aAAa,sBAAuB2B,oBAA0B,CAAEh2C,YAAa,CAAC,kBAClFoJ,EAAIirC,aAAa,0BAA2B4B,qBAA2B,CAAEj2C,YAAa,CAAC,kBACvFoJ,EAAIirC,aAAa,sBAAuB6B,aAAmB,CAAEl2C,YAAa,CAAC,kBAC3EoJ,EAAIirC,aAAa,kBAAmB8B,gBAAsB,CAAEn2C,YAAa,CAAC,gBAAiB,kBAC3FoJ,EAAIirC,aAAa,kBAAmB+B,iBAAuB,CAAEp2C,YAAa,CAAC,qBAC3EoJ,EAAIirC,aAAa,oBAAqBgC,wBAA8B,CAAEr2C,YAAa,CAAC,gBAGhFo0C,GAAahrC,EAAIktC,SAAWltC,EAAIktC,QAAQC,cAAe,CACvD,MAAMC,EAAkBptC,EAAIktC,QAAQC,cAAc,UAClD,GAAIC,GAAmBA,EAAgBn1C,MAAO,CAE1C,MAAMo1C,EAAiB,CACnB,CACI5tC,KAAM,YACN6tC,MAAO,yBACP7+C,KAAM,kBACNmI,YAAa,CAAC,eAElB,CACI6I,KAAM,kBACN6tC,MAAO,oBACP7+C,KAAM,yBACNmI,YAAa,CAAC,YAAa,gBAE/B,CACI6I,KAAM,QACN6tC,MAAO,qBACP7+C,KAAM,YACNmI,YAAa,CAAC,iBAElB,CACI6I,KAAM,SACN6tC,MAAO,sBACP7+C,KAAM,eACNmI,YAAa,CAAC,kBAElB,CACI6I,KAAM,sBACN6tC,MAAO,KACP7+C,KAAM,wBACNmI,YAAa,CAAC,kBACd22C,SAAU,CACN,CACI9tC,KAAM,YACN6tC,MAAO,kCACP7+C,KAAM,oBACNmI,YAAa,CAAC,mBAElB,CACI6I,KAAM,YACN6tC,MAAO,yBACP7+C,KAAM,0BACNmI,YAAa,CAAC,mBAElB,CACI6I,KAAM,UACN6tC,MAAO,uBACP7+C,KAAM,qBACNmI,YAAa,CAAC,qBAElB,CACI6I,KAAM,SACN6tC,MAAO,sBACP7+C,KAAM,UACNmI,YAAa,CAAC,mBAElB,CACI6I,KAAM,cACN6tC,MAAO,wBACP7+C,KAAM,yBACNmI,YAAa,CAAC,uBAI1B,CACI6I,KAAM,WACN6tC,MAAO,KACP7+C,KAAM,YACNmI,YAAa,CAAC,iBACd22C,SAAU,CACN,CACI9tC,KAAM,OACN6tC,MAAO,oBACP7+C,KAAM,kBACNmI,YAAa,CAAC,cAElB,CACI6I,KAAM,eACN6tC,MAAO,4BACP7+C,KAAM,WACNmI,YAAa,CAAC,iBAElB,CACI6I,KAAM,mBACN6tC,MAAO,qCACP7+C,KAAM,aACNmI,YAAa,CAAC,iBAElB,CACI6I,KAAM,cACN6tC,MAAO,4BACP7+C,KAAM,WACNmI,YAAa,CAAC,iBAElB,CACI6I,KAAM,sBACN6tC,MAAO,mCACP7+C,KAAM,oBACNmI,YAAa,CAAC,mBAElB,CACI6I,KAAM,WACN6tC,MAAO,wBACP7+C,KAAM,SACNmI,YAAa,CAAC,gBAAiB,iBAEnC,CACI6I,KAAM,WACN6tC,MAAO,wBACP7+C,KAAM,UACNmI,YAAa,CAAC,sBAI1B,CACI6I,KAAM,UACN6tC,MAAO,KACP7+C,KAAM,YACNmI,YAAa,CAAC,eAAgB,cAC9B22C,SAAU,CACN,CACI9tC,KAAM,aACN6tC,MAAO,yBACP7+C,KAAM,YACNmI,YAAa,CAAC,eAElB,CACI6I,KAAM,mBACN6tC,MAAO,4BACP7+C,KAAM,eACNmI,YAAa,CAAC,eAElB,CACI6I,KAAM,QACN6tC,MAAO,qBACP7+C,KAAM,kBACNmI,YAAa,CAAC,mBAI1B,CACI6I,KAAM,qBACN6tC,MAAO,KACP7+C,KAAM,eACNmI,YAAa,CAAC,gBACd22C,SAAU,CACN,CAAE9tC,KAAM,YAAa6tC,MAAO,8BAA+B7+C,KAAM,qBACjE,CAAEgR,KAAM,iBAAkB6tC,MAAO,4BAA6B7+C,KAAM,WACpE,CAAEgR,KAAM,YAAa6tC,MAAO,8BAA+B7+C,KAAM,wBACjE,CAAEgR,KAAM,aAAc6tC,MAAO,+BAAgC7+C,KAAM,WACnE,CAAEgR,KAAM,UAAW6tC,MAAO,4BAA6B7+C,KAAM,cAGrE,CACIgR,KAAM,cACN6tC,MAAO,KACP7+C,KAAM,cACNmI,YAAa,CAAC,cACd22C,SAAU,CACN,CACI9tC,KAAM,UACN6tC,MAAO,6BACP7+C,KAAM,WACNmI,YAAa,CAAC,eAElB,CACI6I,KAAM,YACN6tC,MAAO,+BACP7+C,KAAM,WACNmI,YAAa,CAAC,eAElB,CACI6I,KAAM,OACN6tC,MAAO,0BACP7+C,KAAM,gBACNmI,YAAa,CAAC,eAElB,CACI6I,KAAM,YACN6tC,MAAO,+BACP7+C,KAAM,eACNmI,YAAa,CAAC,iBAI1B,CACI6I,KAAM,MACN6tC,MAAO,KACP7+C,KAAM,WACNmI,YAAa,CAAC,cACd22C,SAAU,CACN,CACI9tC,KAAM,aACN6tC,MAAO,0BACP7+C,KAAM,oBACNmI,YAAa,CAAC,iBAI1B,CACI6I,KAAM,YACN6tC,MAAO,KACP7+C,KAAM,eACNmI,YAAa,CAAC,gBACd22C,SAAU,CACN,CACI9tC,KAAM,UACN6tC,MAAO,gCACP7+C,KAAM,gBACNmI,YAAa,CAAC,iBAElB,CACI6I,KAAM,MACN6tC,MAAO,4BACP7+C,KAAM,eACNmI,YAAa,CAAC,oBAO9Bw2C,EAAgBn1C,MAAMu1C,WAAWH,EACrC,CACJ,CAEJ"}
1
+ {"version":3,"file":"admin.es.js","sources":["../src/extensions/admin/account/AdminDashboardPage.js","../src/core/views/navigation/SideNavView.js","../src/extensions/admin/account/users/sections/AdminProfileSection.js","../src/extensions/admin/account/users/sections/AdminPersonalSection.js","../src/extensions/admin/account/users/sections/AdminSecuritySection.js","../src/extensions/admin/account/users/sections/AdminConnectedSection.js","../src/extensions/admin/account/users/sections/AdminNotificationsSection.js","../src/extensions/admin/account/users/sections/AdminApiKeysSection.js","../src/extensions/admin/account/users/UserView.js","../src/extensions/admin/account/users/UserTablePage.js","../src/extensions/admin/account/users/MemberView.js","../src/extensions/admin/account/users/MemberTablePage.js","../src/extensions/admin/account/groups/GroupView.js","../src/extensions/admin/account/groups/GroupTablePage.js","../src/extensions/admin/account/devices/DeviceView.js","../src/extensions/admin/account/devices/UserDeviceTablePage.js","../src/extensions/admin/account/devices/UserDeviceLocationTablePage.js","../src/extensions/admin/account/devices/GeoIPView.js","../src/extensions/admin/account/devices/GeoLocatedIPTablePage.js","../src/core/models/ApiKey.js","../src/extensions/admin/account/api_keys/ApiKeyView.js","../src/extensions/admin/account/api_keys/ApiKeyTablePage.js","../src/extensions/admin/aws/CloudWatchChart.js","../src/extensions/admin/aws/CloudWatchDashboardPage.js","../src/extensions/admin/incidents/IncidentDashboardPage.js","../src/core/views/data/StackTraceView.js","../src/extensions/admin/incidents/adapters/IncidentHistoryAdapter.js","../src/extensions/admin/incidents/IncidentView.js","../src/extensions/admin/incidents/IncidentTablePage.js","../src/extensions/admin/incidents/EventView.js","../src/extensions/admin/incidents/EventTablePage.js","../src/extensions/admin/incidents/adapters/TicketNoteAdapter.js","../src/extensions/admin/incidents/TicketView.js","../src/extensions/admin/incidents/TicketTablePage.js","../src/extensions/admin/incidents/RuleSetView.js","../src/extensions/admin/incidents/RuleSetTablePage.js","../src/extensions/admin/messaging/email/EmailDomainTablePage.js","../src/extensions/admin/messaging/email/EmailMailboxTablePage.js","../src/extensions/admin/messaging/email/EmailTemplateView.js","../src/extensions/admin/messaging/email/EmailTemplateTablePage.js","../src/extensions/admin/messaging/email/EmailView.js","../src/extensions/admin/messaging/email/SentMessageTablePage.js","../src/core/models/Phonehub.js","../src/extensions/admin/messaging/sms/PhoneNumberView.js","../src/extensions/admin/messaging/sms/PhoneNumberTablePage.js","../src/extensions/admin/messaging/sms/SMSView.js","../src/extensions/admin/messaging/sms/SMSTablePage.js","../src/extensions/admin/messaging/push/PushDashboardPage.js","../src/extensions/admin/messaging/push/PushConfigTablePage.js","../src/extensions/admin/messaging/push/PushTemplateTablePage.js","../src/extensions/admin/messaging/push/PushDeliveryView.js","../src/extensions/admin/messaging/push/PushDeliveryTablePage.js","../src/extensions/admin/messaging/push/PushDeviceView.js","../src/extensions/admin/messaging/push/PushDeviceTablePage.js","../src/extensions/admin/jobs/JobStatsView.js","../src/extensions/admin/jobs/JobHealthView.js","../src/extensions/admin/jobs/JobDetailsView.js","../src/extensions/admin/jobs/JobsAdminPage.js","../src/extensions/admin/jobs/TaskDetailsView.js","../src/extensions/admin/jobs/RunnerDetailsView.js","../src/extensions/admin/jobs/TaskManagementPage.js","../src/extensions/admin/monitoring/LogView.js","../src/extensions/admin/monitoring/LogTablePage.js","../src/extensions/admin/monitoring/MetricsPermissionsView.js","../src/extensions/admin/monitoring/MetricsPermissionsTablePage.js","../src/core/models/Settings.js","../src/extensions/admin/settings/SettingView.js","../src/extensions/admin/settings/SettingTablePage.js","../src/extensions/admin/storage/FileManagerTablePage.js","../src/extensions/admin/storage/FileView.js","../src/extensions/admin/storage/FileTablePage.js","../src/extensions/admin/storage/S3BucketTablePage.js","../src/extensions/admin/account/devices/UserDeviceLocationView.js","../src/extensions/admin/aws/CloudWatchResourceView.js","../src/admin.js"],"sourcesContent":["/**\n * AdminDashboardPage - Administrative dashboard with system metrics and charts\n */\n\nimport Page from '@core/Page.js';\nimport View from '@core/View.js';\nimport { MetricsChart, MetricsMiniChart, MetricsMiniChartWidget } from '@ext/charts/index.js';\n\n// Embedded HeaderView for dashboard statistics\nclass AdminHeaderView extends View {\n constructor(options = {}) {\n super({\n title: 'Dashboard',\n ...options,\n headerActions: [\n {\n label: 'Export',\n icon: 'bi-download',\n action: 'export',\n buttonClass: 'btn-primary'\n }\n ],\n className: 'admin-header-section'\n });\n\n // Mock data - replace with real API calls\n this.stats = {\n user_activity_day: 0,\n total_users: 0,\n group_activity_day: 0,\n total_groups: 0,\n api_calls: 0,\n apiChange: '',\n incidents: 0,\n incidentsChange: ''\n };\n\n // Prepare formatted data for template\n this.prepareStatsForTemplate();\n }\n\n async getTemplate() {\n return `\n <div class=\"admin-stats-header mb-4\">\n <div class=\"row\">\n <div class=\"col-xl-3 col-lg-6 col-12 mb-3\">\n <div data-container=\"user_activity_day\"></div>\n </div>\n\n <div class=\"col-xl-3 col-lg-6 col-12 mb-3\">\n <div data-container=\"group_activity_day\"></div>\n </div>\n\n <div class=\"col-xl-3 col-lg-6 col-12 mb-3\">\n <div data-container=\"api_activity_day\"></div>\n </div>\n\n <div class=\"col-xl-3 col-lg-6 col-12 mb-3\">\n <div data-container=\"incident_activity_day\"></div>\n </div>\n </div>\n </div>\n `;\n }\n\n async onInit() {\n // TODO: Replace with actual API calls to fetch real statistics\n // this.loadStats();\n // slug: user_activity_day, group_activity_day\n this.userActivity = new MetricsMiniChartWidget({\n icon: \"bi bi-people fs-2\",\n title: 'User Activity',\n subtitle: '{{now_value}} <span class=\"subtitle-label\">{{now_label}}</span> {{total_users}} <span class=\"subtitle-label\">Total</span>',\n background: \"#5388D6\",\n textColor: \"#FFFFFF\",\n granularity: 'days',\n trendRange: 4,\n trendOffset: 0,\n slugs: ['user_activity_day'],\n account: 'global',\n chartType: 'bar',\n showTooltip: true,\n showXAxis: true,\n height: 50,\n chartWidth: '100%',\n color: 'rgba(245, 245, 255, 0.8)',\n fill: true,\n fillColor: 'rgba(245, 245, 255, 0.6)',\n smoothing: 0.3,\n showTrending: true,\n showSettings: true,\n showDateRange: true,\n containerId: 'user_activity_day'\n });\n this.addChild(this.userActivity);\n\n this.groupActivity = new MetricsMiniChartWidget({\n icon: \"bi bi-collection fs-2\",\n title: 'Group Activity',\n subtitle: '{{now_value}} <span class=\"subtitle-label\">{{now_label}}</span> {{total_groups}} <span class=\"subtitle-label\">Total</span>',\n background: \"#1f6a7a\",\n textColor: \"#FFFFFF\",\n granularity: 'days',\n trendRange: 4,\n trendOffset: 0,\n slugs: ['group_activity_day'],\n account: 'global',\n chartType: 'bar',\n showTooltip: true,\n showXAxis: true,\n height: 50,\n chartWidth: '100%',\n color: 'rgba(245, 245, 255, 0.8)',\n fill: true,\n fillColor: 'rgba(245, 245, 255, 0.6)',\n smoothing: 0.3,\n showTrending: true,\n containerId: 'group_activity_day'\n });\n this.addChild(this.groupActivity);\n\n this.apiActivity = new MetricsMiniChartWidget({\n icon: \"bi bi-graph-up fs-2\",\n title: 'API Requests',\n subtitle: '{{now_value}} <span class=\"subtitle-label\">{{now_label}}</span> {{total}} <span class=\"subtitle-label\">Total</span>',\n background: \"#50A079\",\n textColor: \"#FFFFFF\",\n endpoint: '/api/metrics/fetch',\n trendRange: 4,\n trendOffset: 0,\n granularity: 'days',\n slugs: ['api_calls'],\n account: 'global',\n chartType: 'line',\n showTooltip: true,\n showXAxis: true,\n height: 50,\n chartWidth: '100%',\n color: 'rgba(245, 245, 255, 0.8)',\n fill: true,\n fillColor: 'rgba(245, 245, 255, 0.6)',\n smoothing: 0.3,\n showTrending: true,\n containerId: 'api_activity_day'\n });\n this.addChild(this.apiActivity);\n\n this.incidentActivity = new MetricsMiniChartWidget({\n icon: \"bi bi-exclamation-triangle fs-2\",\n title: 'Incidents',\n subtitle: '{{now_value}} <span class=\"subtitle-label\">{{now_label}}</span> {{total}} <span class=\"subtitle-label\">Total</span>',\n background: \"#B14545\",\n textColor: \"#FFFFFF\",\n endpoint: '/api/metrics/fetch',\n trendRange: 4,\n trendOffset: 0,\n granularity: 'days',\n slugs: ['incidents'],\n account: 'incident',\n chartType: 'line',\n showTooltip: true,\n showXAxis: true,\n height: 50,\n chartWidth: '100%',\n color: 'rgba(245, 245, 255, 0.8)',\n fill: true,\n fillColor: 'rgba(245, 245, 255, 0.6)',\n smoothing: 0.3,\n showTrending: true,\n containerId: 'incident_activity_day'\n });\n this.addChild(this.incidentActivity);\n }\n\n async onBeforeRender() {\n\n }\n\n prepareStatsForTemplate() {\n // Determine badge and icon classes based on incidents change\n // const isDecreasing = this.stats.incidentsChange.startsWith('-');\n // this.stats.incidentsBadgeClass = isDecreasing ? 'bg-success-subtle text-success' : 'bg-warning-subtle text-warning';\n // this.stats.incidentsIconClass = isDecreasing ? 'arrow-down' : 'arrow-up';\n\n }\n\n async loadValues() {\n try {\n const response = await this.getApp().rest.GET('/api/metrics/value/get', {\n slugs: ['total_users', 'total_groups'],\n account: \"global\"\n });\n if (response.success && response.data.status) {\n Object.assign(this.stats, response.data.data);\n }\n if (this.groupActivity) {\n this.groupActivity.header.total_groups = this.stats.total_groups || 0;\n this.groupActivity.header.render();\n }\n if (this.userActivity) {\n this.userActivity.header.total_users = this.stats.total_users || 0;\n this.userActivity.header.render();\n }\n } catch (error) {\n console.error('Failed to load admin stats:', error);\n }\n }\n\n async loadStats() {\n // Example of how to load real data\n try {\n const response = await this.getApp().rest.GET('/api/metrics/series', {\n slugs: ['user_created', 'user_activity_day', \"incidents\",\n \"api_calls\", \"api_errors\", \"group_activity_day\"],\n account: \"global\",\n granularity: \"days\"\n });\n if (response.success && response.data.status) {\n Object.assign(this.stats, response.data.data);\n this.prepareStatsForTemplate();\n }\n } catch (error) {\n console.error('Failed to load admin stats:', error);\n }\n }\n}\n\nexport default class AdminDashboardPage extends Page {\n constructor(options = {}) {\n super({\n ...options,\n title: 'Admin Dashboard',\n className: 'admin-dashboard-page'\n });\n\n // Page data\n this.pageTitle = 'Admin Dashboard';\n this.pageSubtitle = 'System monitoring and metrics overview';\n }\n\n async getTemplate() {\n return `\n <div class=\"admin-dashboard-container container-lg\">\n <!-- Page Header -->\n <div class=\"d-flex justify-content-between align-items-center mb-2\">\n <div>\n <p class=\"text-muted mb-0\">{{pageSubtitle}}</p>\n <small class=\"text-info\">\n <i class=\"bi bi-shield-check me-1\"></i>\n Real-time system metrics and performance monitoring\n </small>\n </div>\n <div class=\"btn-group\" role=\"group\">\n <button type=\"button\" class=\"btn btn-outline-secondary btn-sm\"\n data-action=\"refresh-all\" title=\"Refresh All Charts\">\n <i class=\"bi bi-arrow-clockwise\"></i> Refresh\n </button>\n <button type=\"button\" class=\"btn btn-outline-primary btn-sm\"\n data-action=\"export-metrics\" title=\"Export Metrics Data\">\n <i class=\"bi bi-download\"></i> Export\n </button>\n <button type=\"button\" class=\"btn btn-outline-warning btn-sm\"\n data-action=\"view-alerts\" title=\"View System Alerts\">\n <i class=\"bi bi-bell\"></i> Alerts\n </button>\n </div>\n </div>\n\n <!-- Stats Header -->\n <div data-container=\"admin-header\"></div>\n <div data-container=\"example-chart\"></div>\n <!-- Charts Section -->\n <div class=\"row\">\n <!-- Full Width API Metrics Chart -->\n <div class=\"col-12 mb-4\">\n <div data-container=\"api-metrics-chart\"></div>\n </div>\n </div>\n\n <!-- System Status Footer -->\n <div class=\"row\">\n <div class=\"col-12\">\n <div class=\"alert alert-success border-0\" role=\"alert\">\n <div class=\"d-flex align-items-center\">\n <i class=\"bi bi-check-circle-fill me-2\"></i>\n <div>\n <strong>System Status:</strong> All systems operational.\n Last updated: <span class=\"text-muted\">{{lastUpdated}}</span>\n </div>\n <div class=\"ms-auto\">\n <button class=\"btn btn-sm btn-outline-success\" data-action=\"view-system-status\">\n <i class=\"bi bi-info-circle\"></i> Details\n </button>\n </div>\n </div>\n </div>\n </div>\n </div>\n </div>\n `;\n }\n\n async onInit() {\n // Set last updated time\n this.lastUpdated = new Date().toLocaleString();\n\n // Create and add header view\n this.headerView = new AdminHeaderView({\n containerId: 'admin-header'\n });\n this.addChild(this.headerView);\n\n // Create API Metrics Chart\n this.apiMetricsChart = new MetricsChart({\n title: `<i class=\"bi bi-graph-up me-2\"></i> API Metrics`,\n endpoint: '/api/metrics/fetch',\n height: 250,\n granularity: 'hours',\n slugs: ['api_calls', 'api_errors'],\n account: 'global',\n chartType: 'line',\n showDateRange: false,\n yAxis: {\n label: 'Count',\n beginAtZero: true\n },\n tooltip: {\n y: 'number'\n },\n containerId: 'api-metrics-chart'\n });\n this.addChild(this.apiMetricsChart);\n\n }\n\n // Action Handlers\n async onActionRefreshAll(event, element) {\n try {\n // Show loading state\n const button = element || event?.currentTarget || null;\n const icon = button?.querySelector?.('i');\n icon?.classList.add('bi-spin');\n if (button) button.disabled = true;\n\n // Refresh all charts\n const promises = [\n this.headerView?.loadValues(),\n this.apiMetricsChart?.refresh()\n ].filter(Boolean);\n\n await Promise.allSettled(promises);\n\n\n // Update last updated time\n this.lastUpdated = new Date().toLocaleString();\n\n // Emit refresh event\n const eventBus = this.getApp()?.events;\n if (eventBus) {\n eventBus.emit('admin:dashboard-refreshed', {\n page: this,\n timestamp: this.lastUpdated\n });\n }\n\n } catch (error) {\n console.error('Failed to refresh dashboard:', error);\n // Show error feedback\n const alert = this.element.querySelector('.alert-success');\n if (alert) {\n alert.className = 'alert alert-danger border-0';\n alert.innerHTML = `\n <div class=\"d-flex align-items-center\">\n <i class=\"bi bi-exclamation-triangle-fill me-2\"></i>\n <div>\n <strong>Error:</strong> Failed to refresh dashboard data.\n </div>\n </div>\n `;\n\n // Reset after 5 seconds\n setTimeout(() => {\n alert.className = 'alert alert-success border-0';\n alert.innerHTML = `\n <div class=\"d-flex align-items-center\">\n <i class=\"bi bi-check-circle-fill me-2\"></i>\n <div>\n <strong>System Status:</strong> All systems operational.\n Last updated: <span class=\"text-muted\">${this.lastUpdated}</span>\n </div>\n </div>\n `;\n }, 5000);\n }\n } finally {\n // Reset button state\n const icon = element.querySelector('i');\n icon?.classList.remove('bi-spin');\n if (button) button.disabled = false;\n }\n }\n\n async onActionExportMetrics(event, element) {\n try {\n // Export all charts as PNG\n await this.apiMetricsChart?.export('png');\n\n // Show success feedback\n const eventBus = this.getApp()?.events;\n if (eventBus) {\n eventBus.emit('admin:metrics-exported', {\n page: this,\n charts: ['api-metrics']\n });\n }\n\n } catch (error) {\n console.error('Failed to export metrics:', error);\n }\n }\n\n async onActionViewAlerts(event, element) {\n // Navigate to alerts page or show alerts modal\n const router = this.getApp()?.router;\n if (router) {\n router.navigateTo('/admin/alerts');\n }\n }\n\n async onActionViewSystemStatus(event, element) {\n // Navigate to system status page\n const router = this.getApp()?.router;\n if (router) {\n router.navigateTo('/admin/system-status');\n }\n }\n\n // Public API\n async refreshDashboard() {\n return this.onActionRefreshAll(null, null, { disabled: false, querySelector: () => null });\n }\n\n getCharts() {\n return {\n apiMetrics: this.apiMetricsChart\n };\n }\n\n getStats() {\n return this.headerView?.stats || {};\n }\n\n async onAfterRender() {\n this.headerView?.loadValues();\n }\n}\n","/**\n * SideNavView - Left sidebar navigation with content panel\n *\n * A reusable navigation component that displays a vertical sidebar with\n * nav links, optional group labels, and icons. The content panel mounts\n * one child view at a time, switching on nav click.\n *\n * Features:\n * - Left sidebar with nav links, icons, and group dividers\n * - Active state with accent border\n * - Mount/unmount child views on section switch\n * - Responsive: collapses to dropdown on narrow containers\n * - Permission-aware: skips sections the user lacks permission for\n * - Configurable nav width and content padding\n * - Smooth fade transitions between sections\n *\n * Example Usage:\n * ```javascript\n * const sideNav = new SideNavView({\n * sections: [\n * { key: 'profile', label: 'Profile', icon: 'bi-person', view: profileView },\n * { key: 'security', label: 'Security', icon: 'bi-shield-lock', view: securityView },\n * { type: 'divider', label: 'Activity' },\n * { key: 'sessions', label: 'Sessions', icon: 'bi-clock-history', view: sessionsView },\n * ],\n * activeSection: 'profile',\n * navWidth: 200,\n * contentPadding: '1.5rem 2.5rem',\n * enableResponsive: true\n * });\n * ```\n */\n\nimport View from '@core/View.js';\n\nclass SideNavView extends View {\n constructor(options = {}) {\n const {\n sections = [],\n activeSection,\n navWidth,\n contentPadding,\n enableResponsive,\n minWidth,\n ...viewOptions\n } = options;\n\n super({\n tagName: 'div',\n className: 'side-nav-view',\n ...viewOptions\n });\n\n // Configuration\n this.navWidth = navWidth || 200;\n this.contentPadding = contentPadding || '1.5rem 2.5rem';\n this.enableResponsive = enableResponsive !== false;\n this.minWidth = minWidth || 500;\n\n // State\n this.sectionConfigs = []; // Full config array (including dividers)\n this.sectionViews = {}; // key → view instance\n this.sectionKeys = []; // Ordered navigable section keys\n this.activeSection = null;\n this.currentMode = 'sidebar'; // 'sidebar' or 'dropdown'\n this.resizeObserver = null;\n this.lastContainerWidth = 0;\n\n // Process sections config\n for (const config of sections) {\n this._addSectionConfig(config);\n }\n\n // Set initial active section\n this.activeSection = activeSection || this.sectionKeys[0] || null;\n\n // Bind resize handler\n this.handleResize = this.handleResize.bind(this);\n }\n\n /**\n * Process and store a section config entry\n * @param {object} config - Section config (navigable or divider)\n * @private\n */\n _addSectionConfig(config) {\n if (config.type === 'divider') {\n this.sectionConfigs.push({ type: 'divider', label: config.label });\n return;\n }\n\n // Skip if user lacks required permission\n if (config.permissions && !this._hasPermission(config.permissions)) {\n return;\n }\n\n this.sectionConfigs.push(config);\n this.sectionKeys.push(config.key);\n\n if (config.view) {\n this.sectionViews[config.key] = config.view;\n config.view.parent = this;\n }\n }\n\n /**\n * Check if the current user has a permission\n * @param {string} perm - Permission string\n * @returns {boolean}\n * @private\n */\n _hasPermission(perm) {\n try {\n return this.getApp().activeUser.hasPerm(perm);\n } catch {\n return true; // If app isn't available yet, allow — will be checked at render\n }\n }\n\n // ───────────────────────────────────────────────\n // Template\n // ───────────────────────────────────────────────\n\n async renderTemplate() {\n const nav = this.currentMode === 'dropdown'\n ? this._buildDropdownNav()\n : this._buildSidebarNav();\n\n return `\n <style>\n .snv-layout { display: flex; height: 100%; min-height: 0; }\n .snv-nav {\n width: ${this.navWidth}px;\n background: #f8f9fc;\n border-right: 1px solid #e9ecef;\n padding: 0.75rem 0;\n flex-shrink: 0;\n overflow-y: auto;\n }\n .snv-nav a {\n color: #495057;\n padding: 0.45rem 1.25rem;\n font-size: 0.85rem;\n display: flex;\n align-items: center;\n gap: 0.5rem;\n text-decoration: none;\n cursor: pointer;\n }\n .snv-nav a:hover { background: #e9ecef; }\n .snv-nav a.active {\n background: #e7f1ff;\n color: #0d6efd;\n font-weight: 600;\n border-right: 2px solid #0d6efd;\n }\n .snv-nav a i { width: 18px; text-align: center; font-size: 0.9rem; }\n .snv-nav-label {\n font-size: 0.65rem;\n font-weight: 700;\n text-transform: uppercase;\n letter-spacing: 0.06em;\n color: #adb5bd;\n padding: 0.75rem 1.25rem 0.25rem;\n }\n .snv-content {\n flex: 1;\n overflow-y: auto;\n padding: ${this.contentPadding};\n min-width: 0;\n }\n .snv-content > .snv-section { display: none; }\n .snv-content > .snv-section.snv-active { display: block; }\n .snv-dropdown { margin-bottom: 0.75rem; }\n .snv-select-btn {\n display: flex;\n align-items: center;\n gap: 0.5rem;\n width: 100%;\n padding: 0.5rem 1rem;\n background: #f8f9fc;\n border: 1px solid #dee2e6;\n border-radius: 0.375rem;\n font-size: 0.85rem;\n color: #495057;\n cursor: pointer;\n }\n .snv-select-btn:hover { background: #e9ecef; }\n .snv-select-btn::after {\n content: '';\n display: inline-block;\n margin-left: auto;\n border-top: 0.3em solid;\n border-right: 0.3em solid transparent;\n border-left: 0.3em solid transparent;\n }\n @media (max-width: 576px) {\n .snv-nav { display: none; }\n .snv-content { padding: 1.25rem; }\n }\n </style>\n ${this.currentMode === 'dropdown' ? `\n <div class=\"snv-dropdown\">${nav}</div>\n <div class=\"snv-content\" data-container=\"snv-content\"></div>\n ` : `\n <div class=\"snv-layout\">\n <nav class=\"snv-nav\">${nav}</nav>\n <div class=\"snv-content\" data-container=\"snv-content\"></div>\n </div>\n `}\n `;\n }\n\n /**\n * Build sidebar navigation HTML\n * @returns {string}\n * @private\n */\n _buildSidebarNav() {\n return this.sectionConfigs.map(config => {\n if (config.type === 'divider') {\n return `<div class=\"snv-nav-label\">${this.escapeHtml(config.label)}</div>`;\n }\n const isActive = config.key === this.activeSection;\n const icon = config.icon ? `<i class=\"bi ${config.icon}\"></i>` : '';\n return `<a href=\"#\" class=\"${isActive ? 'active' : ''}\" data-action=\"navigate\" data-section=\"${config.key}\">${icon} ${this.escapeHtml(config.label)}</a>`;\n }).join('');\n }\n\n /**\n * Build dropdown navigation HTML (responsive mode)\n * @returns {string}\n * @private\n */\n _buildDropdownNav() {\n const activeConfig = this.sectionConfigs.find(c => c.key === this.activeSection);\n const activeLabel = activeConfig ? activeConfig.label : this.sectionKeys[0];\n\n const items = this.sectionConfigs\n .filter(c => c.type !== 'divider')\n .map(config => {\n const isActive = config.key === this.activeSection;\n return `\n <li>\n <button class=\"dropdown-item ${isActive ? 'active' : ''}\"\n data-action=\"navigate\"\n data-section=\"${config.key}\"\n type=\"button\">\n ${config.icon ? `<i class=\"bi ${config.icon} me-2\"></i>` : ''}\n ${this.escapeHtml(config.label)}\n ${isActive ? '<i class=\"bi bi-check-lg ms-2\"></i>' : ''}\n </button>\n </li>\n `;\n }).join('');\n\n return `\n <div class=\"dropdown\">\n <button class=\"snv-select-btn\" type=\"button\"\n data-bs-toggle=\"dropdown\" aria-expanded=\"false\">\n ${activeConfig?.icon ? `<i class=\"bi ${activeConfig.icon}\"></i>` : ''}\n <span>${this.escapeHtml(activeLabel)}</span>\n </button>\n <ul class=\"dropdown-menu w-100\">${items}</ul>\n </div>\n `;\n }\n\n // ───────────────────────────────────────────────\n // Lifecycle\n // ───────────────────────────────────────────────\n\n async onAfterRender() {\n await super.onAfterRender();\n\n // Mount the active section\n if (this.activeSection) {\n await this._mountSection(this.activeSection);\n }\n\n // Set up responsive behavior\n if (this.enableResponsive) {\n this._setupResponsive();\n }\n }\n\n async onBeforeDestroy() {\n await super.onBeforeDestroy();\n\n // Clean up resize observer\n if (this.resizeObserver) {\n this.resizeObserver.disconnect();\n this.resizeObserver = null;\n }\n\n if (typeof window !== 'undefined') {\n window.removeEventListener('resize', this.handleResize);\n }\n\n // Destroy all section views\n for (const view of Object.values(this.sectionViews)) {\n if (view && typeof view.destroy === 'function') {\n await view.destroy();\n }\n }\n }\n\n // ───────────────────────────────────────────────\n // Section switching\n // ───────────────────────────────────────────────\n\n /**\n * Navigate to a section\n * @param {string} key - Section key\n * @returns {Promise<boolean>}\n */\n async showSection(key) {\n if (!this.sectionViews[key]) {\n console.warn(`SideNavView: Section \"${key}\" does not exist`);\n return false;\n }\n\n if (key === this.activeSection) {\n // Already active — but ensure it's mounted\n const view = this.sectionViews[key];\n if (view && view.isMounted() && this.element?.contains(view.element)) {\n return true;\n }\n }\n\n const previousSection = this.activeSection;\n this.activeSection = key;\n\n // Unmount previous section\n if (previousSection && previousSection !== key) {\n await this._unmountSection(previousSection);\n }\n\n // Mount new section\n await this._mountSection(key);\n\n // Update nav visual state\n this._updateNavState(key);\n\n this.emit('section:changed', {\n activeSection: key,\n previousSection\n });\n\n return true;\n }\n\n /**\n * Mount a section view into the content area\n * @param {string} key - Section key\n * @private\n */\n async _mountSection(key) {\n const view = this.sectionViews[key];\n if (!view) return;\n\n const container = this.element?.querySelector('[data-container=\"snv-content\"]');\n if (!container) return;\n\n if (!view.isMounted()) {\n await view.render(true, container);\n }\n }\n\n /**\n * Unmount a section view\n * @param {string} key - Section key\n * @private\n */\n async _unmountSection(key) {\n const view = this.sectionViews[key];\n if (!view || !view.isMounted()) return;\n\n await view.unmount();\n }\n\n /**\n * Update nav link active state\n * @param {string} activeKey - Active section key\n * @private\n */\n _updateNavState(activeKey) {\n if (!this.element) return;\n\n // Update sidebar links\n this.element.querySelectorAll('.snv-nav a, .dropdown-item').forEach(link => {\n const section = link.dataset.section;\n if (section) {\n link.classList.toggle('active', section === activeKey);\n }\n });\n\n // Update dropdown button label\n const selectBtn = this.element.querySelector('.snv-select-btn span');\n if (selectBtn) {\n const config = this.sectionConfigs.find(c => c.key === activeKey);\n if (config) {\n selectBtn.textContent = config.label;\n }\n }\n }\n\n // ───────────────────────────────────────────────\n // Action handlers\n // ───────────────────────────────────────────────\n\n async onActionNavigate(event, el) {\n event.preventDefault();\n const section = el.dataset.section;\n if (section) {\n await this.showSection(section);\n }\n return true;\n }\n\n // ───────────────────────────────────────────────\n // Responsive\n // ───────────────────────────────────────────────\n\n /**\n * Set up responsive width detection\n * @private\n */\n _setupResponsive() {\n if (!this.element || !this.enableResponsive) return;\n\n this._updateMode();\n\n if (typeof ResizeObserver !== 'undefined') {\n this.resizeObserver = new ResizeObserver(() => {\n this.handleResize();\n });\n const container = this.element.parentElement || this.element;\n this.resizeObserver.observe(container);\n } else {\n window.addEventListener('resize', this.handleResize);\n }\n }\n\n /**\n * Handle resize events\n */\n async handleResize() {\n const containerWidth = this._getContainerWidth();\n if (Math.abs(containerWidth - this.lastContainerWidth) > 50) {\n this.lastContainerWidth = containerWidth;\n await this._updateMode();\n }\n }\n\n /**\n * Get the container width\n * @returns {number}\n * @private\n */\n _getContainerWidth() {\n if (!this.element) return this.minWidth;\n const container = this.element.parentElement || this.element;\n return container.offsetWidth || this.minWidth;\n }\n\n /**\n * Check and switch between sidebar and dropdown modes\n * @private\n */\n async _updateMode() {\n const containerWidth = this._getContainerWidth();\n const newMode = containerWidth < this.minWidth ? 'dropdown' : 'sidebar';\n\n if (newMode !== this.currentMode) {\n this.currentMode = newMode;\n if (this.isMounted()) {\n await this.render();\n }\n this.emit('navigation:modeChanged', {\n mode: this.currentMode,\n containerWidth\n });\n }\n }\n\n // ───────────────────────────────────────────────\n // Public API\n // ───────────────────────────────────────────────\n\n /**\n * Get the active section key\n * @returns {string|null}\n */\n getActiveSection() {\n return this.activeSection;\n }\n\n /**\n * Get all navigable section keys\n * @returns {string[]}\n */\n getSectionKeys() {\n return [...this.sectionKeys];\n }\n\n /**\n * Get a section's view by key\n * @param {string} key - Section key\n * @returns {View|null}\n */\n getSection(key) {\n return this.sectionViews[key] || null;\n }\n\n /**\n * Add a section dynamically\n * @param {object} config - Section config\n * @param {boolean} makeActive - Whether to activate the section\n * @returns {Promise<boolean>}\n */\n async addSection(config, makeActive = false) {\n if (config.key && this.sectionViews[config.key]) {\n console.warn(`SideNavView: Section \"${config.key}\" already exists`);\n return false;\n }\n\n this._addSectionConfig(config);\n\n if (this.isMounted()) {\n await this.render();\n if (makeActive && config.key) {\n await this.showSection(config.key);\n }\n }\n\n this.emit('section:added', { config });\n return true;\n }\n\n /**\n * Remove a section dynamically\n * @param {string} key - Section key to remove\n * @returns {Promise<boolean>}\n */\n async removeSection(key) {\n const view = this.sectionViews[key];\n if (!view) {\n console.warn(`SideNavView: Section \"${key}\" does not exist`);\n return false;\n }\n\n // Destroy the view\n if (typeof view.destroy === 'function') {\n await view.destroy();\n }\n\n // Remove from data structures\n delete this.sectionViews[key];\n this.sectionKeys = this.sectionKeys.filter(k => k !== key);\n this.sectionConfigs = this.sectionConfigs.filter(c => c.key !== key);\n\n // Handle active section removal\n if (this.activeSection === key) {\n this.activeSection = this.sectionKeys[0] || null;\n }\n\n if (this.isMounted()) {\n await this.render();\n }\n\n this.emit('section:removed', { key });\n return true;\n }\n\n /**\n * Prevent model changes from triggering a full re-render.\n * Section views manage their own model reactivity.\n */\n _onModelChange() {\n // no-op — same pattern as UserView\n }\n\n static create(options = {}) {\n return new SideNavView(options);\n }\n}\n\nexport default SideNavView;\n","/**\n * AdminProfileSection - Contact, verification overrides, and account overview\n *\n * Admin can view/edit contact info (email, phone), override verification\n * status, and see account metadata (username, status, role, MFA, dates).\n * Uses model.save() against /api/user/<id> (admin endpoint).\n */\nimport View from '@core/View.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\n\nexport default class AdminProfileSection extends View {\n constructor(options = {}) {\n super({\n className: 'admin-profile-section',\n template: `\n <style>\n .ap-section-label { font-size: 0.7rem; font-weight: 700; text-transform: uppercase; letter-spacing: 0.06em; color: #adb5bd; margin-bottom: 0.5rem; margin-top: 1.75rem; }\n .ap-section-label:first-child { margin-top: 0; }\n .ap-field-row { display: flex; align-items: center; padding: 0.6rem 0; border-bottom: 1px solid #f0f0f0; }\n .ap-field-row:last-child { border-bottom: none; }\n .ap-field-label { width: 140px; font-size: 0.8rem; color: #6c757d; flex-shrink: 0; }\n .ap-field-value { flex: 1; font-size: 0.88rem; color: #212529; display: flex; align-items: center; gap: 0.4rem; }\n .ap-field-action { color: #6c757d; cursor: pointer; font-size: 0.8rem; margin-left: auto; padding: 0.15rem 0.4rem; border-radius: 4px; background: none; border: none; }\n .ap-field-action:hover { background: #f0f0f0; color: #0d6efd; }\n .ap-badge-ok { font-size: 0.65rem; padding: 0.15em 0.45em; background: #d1e7dd; color: #0f5132; border-radius: 3px; }\n .ap-badge-warn { font-size: 0.65rem; padding: 0.15em 0.45em; background: #fff3cd; color: #856404; border-radius: 3px; }\n .ap-badge-muted { font-size: 0.65rem; padding: 0.15em 0.45em; background: #f0f0f0; color: #6c757d; border-radius: 3px; }\n .ap-not-set { color: #adb5bd; font-style: italic; font-size: 0.85rem; }\n </style>\n\n <!-- Contact & Verification -->\n <div class=\"ap-section-label\">Contact & Verification</div>\n <div class=\"ap-field-row\">\n <div class=\"ap-field-label\">Email</div>\n <div class=\"ap-field-value\">\n {{model.email}}\n {{#model.is_email_verified|bool}}\n <span class=\"ap-badge-ok\">Verified</span>\n {{/model.is_email_verified|bool}}\n {{^model.is_email_verified|bool}}\n <span class=\"ap-badge-warn\">Unverified</span>\n {{/model.is_email_verified|bool}}\n </div>\n {{#model.is_email_verified|bool}}\n <button type=\"button\" class=\"ap-field-action\" data-action=\"unverify-email\" title=\"Mark as unverified\"><i class=\"bi bi-x-circle\"></i></button>\n {{/model.is_email_verified|bool}}\n {{^model.is_email_verified|bool}}\n <button type=\"button\" class=\"ap-field-action\" data-action=\"force-verify-email\" title=\"Force verify\"><i class=\"bi bi-patch-check\"></i></button>\n {{/model.is_email_verified|bool}}\n <button type=\"button\" class=\"ap-field-action\" data-action=\"change-email\" title=\"Change email\"><i class=\"bi bi-pencil\"></i></button>\n </div>\n <div class=\"ap-field-row\">\n <div class=\"ap-field-label\">Phone</div>\n <div class=\"ap-field-value\">\n {{#hasPhone|bool}}\n {{model.phone_number}}\n {{#model.is_phone_verified|bool}}\n <span class=\"ap-badge-ok\">Verified</span>\n {{/model.is_phone_verified|bool}}\n {{^model.is_phone_verified|bool}}\n <span class=\"ap-badge-warn\">Unverified</span>\n {{/model.is_phone_verified|bool}}\n {{/hasPhone|bool}}\n {{^hasPhone|bool}}\n <span class=\"ap-not-set\">Not set</span>\n {{/hasPhone|bool}}\n </div>\n {{#hasPhone|bool}}\n {{#model.is_phone_verified|bool}}\n <button type=\"button\" class=\"ap-field-action\" data-action=\"unverify-phone\" title=\"Mark as unverified\"><i class=\"bi bi-x-circle\"></i></button>\n {{/model.is_phone_verified|bool}}\n {{^model.is_phone_verified|bool}}\n <button type=\"button\" class=\"ap-field-action\" data-action=\"force-verify-phone\" title=\"Force verify\"><i class=\"bi bi-patch-check\"></i></button>\n {{/model.is_phone_verified|bool}}\n <button type=\"button\" class=\"ap-field-action\" data-action=\"change-phone\" title=\"Change phone\"><i class=\"bi bi-pencil\"></i></button>\n <button type=\"button\" class=\"ap-field-action\" data-action=\"remove-phone\" title=\"Remove phone\"><i class=\"bi bi-x-lg\"></i></button>\n {{/hasPhone|bool}}\n {{^hasPhone|bool}}\n <button type=\"button\" class=\"ap-field-action\" data-action=\"set-phone\" title=\"Set phone number\"><i class=\"bi bi-plus\"></i></button>\n {{/hasPhone|bool}}\n </div>\n\n <!-- Account -->\n <div class=\"ap-section-label\">Account</div>\n <div class=\"ap-field-row\">\n <div class=\"ap-field-label\">Username</div>\n <div class=\"ap-field-value\">{{model.username}}</div>\n <button type=\"button\" class=\"ap-field-action\" data-action=\"edit-username\" title=\"Edit\"><i class=\"bi bi-pencil\"></i></button>\n </div>\n <div class=\"ap-field-row\">\n <div class=\"ap-field-label\">Status</div>\n <div class=\"ap-field-value\">\n {{#model.is_active|bool}}<span class=\"ap-badge-ok\">Active</span>{{/model.is_active|bool}}\n {{^model.is_active|bool}}<span class=\"ap-badge-warn\">Inactive</span>{{/model.is_active|bool}}\n </div>\n </div>\n <div class=\"ap-field-row\">\n <div class=\"ap-field-label\">Role</div>\n <div class=\"ap-field-value\">\n {{roleLabel}}\n {{#model.is_staff|bool}}<span class=\"ap-badge-muted\">Staff</span>{{/model.is_staff|bool}}\n </div>\n </div>\n <div class=\"ap-field-row\">\n <div class=\"ap-field-label\">MFA</div>\n <div class=\"ap-field-value\">\n {{#model.requires_mfa|bool}}<span class=\"ap-badge-ok\">Required</span>{{/model.requires_mfa|bool}}\n {{^model.requires_mfa|bool}}<span class=\"ap-badge-muted\">Not required</span>{{/model.requires_mfa|bool}}\n </div>\n </div>\n <div class=\"ap-field-row\">\n <div class=\"ap-field-label\">Member Since</div>\n <div class=\"ap-field-value\">{{model.date_joined|date}}</div>\n </div>\n <div class=\"ap-field-row\">\n <div class=\"ap-field-label\">Last Login</div>\n <div class=\"ap-field-value\">{{model.last_login|relative}}</div>\n </div>\n `,\n ...options\n });\n }\n\n // ── Computed properties ─────────────────────────\n\n get hasPhone() {\n return !!(this.model && this.model.get('phone_number'));\n }\n\n get roleLabel() {\n if (!this.model) return 'User';\n if (this.model.get('is_superuser')) return 'Superuser';\n return 'User';\n }\n\n // ── Verification overrides ──────────────────────\n\n async onActionForceVerifyEmail() {\n const confirmed = await Dialog.confirm(\n `Mark <strong>${this.model.get('email')}</strong> as verified? This bypasses the normal verification flow.`,\n 'Force Verify Email'\n );\n if (!confirmed) return true;\n await this._saveField({ is_email_verified: true }, 'Email marked as verified');\n return true;\n }\n\n async onActionUnverifyEmail() {\n const confirmed = await Dialog.confirm(\n 'Mark email as unverified? The user will need to re-verify their email.',\n 'Unverify Email'\n );\n if (!confirmed) return true;\n await this._saveField({ is_email_verified: false }, 'Email marked as unverified');\n return true;\n }\n\n async onActionForceVerifyPhone() {\n const confirmed = await Dialog.confirm(\n `Mark <strong>${this.model.get('phone_number')}</strong> as verified? This bypasses the normal verification flow.`,\n 'Force Verify Phone'\n );\n if (!confirmed) return true;\n await this._saveField({ is_phone_verified: true }, 'Phone marked as verified');\n return true;\n }\n\n async onActionUnverifyPhone() {\n const confirmed = await Dialog.confirm(\n 'Mark phone as unverified? The user will need to re-verify their phone number.',\n 'Unverify Phone'\n );\n if (!confirmed) return true;\n await this._saveField({ is_phone_verified: false }, 'Phone marked as unverified');\n return true;\n }\n\n // ── Contact editing ─────────────────────────────\n\n async onActionChangeEmail() {\n const email = await Dialog.prompt(\n 'Enter the new email address for this user:',\n 'Change Email',\n { defaultValue: this.model.get('email') || '' }\n );\n if (email === null || !email.trim()) return true;\n await this._saveField({ email: email.trim() }, 'Email updated');\n return true;\n }\n\n async onActionChangePhone() {\n const phone = await Dialog.prompt(\n 'Enter the new phone number for this user:',\n 'Change Phone',\n { defaultValue: this.model.get('phone_number') || '' }\n );\n if (phone === null || !phone.trim()) return true;\n await this._saveField({ phone_number: phone.trim() }, 'Phone number updated');\n return true;\n }\n\n async onActionSetPhone() {\n const phone = await Dialog.prompt(\n 'Enter a phone number for this user:',\n 'Set Phone Number',\n { placeholder: '(415) 555-0123' }\n );\n if (!phone || !phone.trim()) return true;\n await this._saveField({ phone_number: phone.trim() }, 'Phone number added');\n return true;\n }\n\n async onActionRemovePhone() {\n const confirmed = await Dialog.confirm(\n 'Remove this user\\'s phone number?',\n 'Remove Phone'\n );\n if (!confirmed) return true;\n\n const resp = await this.model.save({ phone_number: null });\n if (resp.status === 200) {\n this.model.set('is_phone_verified', false);\n this.getApp()?.toast?.success('Phone number removed');\n await this.render();\n } else {\n this.getApp()?.toast?.error(resp.message || 'Failed to remove phone number');\n }\n return true;\n }\n\n async onActionEditUsername() {\n const username = await Dialog.prompt(\n 'Username:',\n 'Edit Username',\n { defaultValue: this.model.get('username') || '' }\n );\n if (username !== null && username.trim()) {\n await this._saveField({ username: username.trim() }, 'Username updated');\n }\n return true;\n }\n\n // ── Helpers ─────────────────────────────────────\n\n async _saveField(fields, successMsg) {\n const resp = await this.model.save(fields);\n if (resp.status === 200) {\n this.getApp()?.toast?.success(successMsg);\n await this.render();\n } else {\n this.getApp()?.toast?.error(resp.message || 'Failed to save');\n }\n }\n}\n","/**\n * AdminPersonalSection - Personal information editing\n *\n * Admin can view/edit name fields, DOB, timezone, and address.\n * Uses model.save() against /api/user/<id> (admin endpoint).\n */\nimport View from '@core/View.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\n\nexport default class AdminPersonalSection extends View {\n constructor(options = {}) {\n super({\n className: 'admin-personal-section',\n template: `\n <style>\n .aps-section-label { font-size: 0.7rem; font-weight: 700; text-transform: uppercase; letter-spacing: 0.06em; color: #adb5bd; margin-bottom: 0.5rem; margin-top: 1.75rem; }\n .aps-section-label:first-child { margin-top: 0; }\n .aps-field-row { display: flex; align-items: center; padding: 0.6rem 0; border-bottom: 1px solid #f0f0f0; }\n .aps-field-row:last-child { border-bottom: none; }\n .aps-field-label { width: 140px; font-size: 0.8rem; color: #6c757d; flex-shrink: 0; }\n .aps-field-value { flex: 1; font-size: 0.88rem; color: #212529; display: flex; align-items: center; gap: 0.4rem; }\n .aps-field-action { color: #6c757d; cursor: pointer; font-size: 0.8rem; margin-left: auto; padding: 0.15rem 0.4rem; border-radius: 4px; background: none; border: none; }\n .aps-field-action:hover { background: #f0f0f0; color: #0d6efd; }\n .aps-badge-ok { font-size: 0.65rem; padding: 0.15em 0.45em; background: #d1e7dd; color: #0f5132; border-radius: 3px; }\n .aps-badge-warn { font-size: 0.65rem; padding: 0.15em 0.45em; background: #fff3cd; color: #856404; border-radius: 3px; }\n .aps-not-set { color: #adb5bd; font-style: italic; font-size: 0.85rem; }\n </style>\n\n <!-- Name -->\n <div class=\"aps-section-label\">Name</div>\n <div class=\"aps-field-row\">\n <div class=\"aps-field-label\">Display Name</div>\n <div class=\"aps-field-value\">\n {{#model.display_name}}{{model.display_name}}{{/model.display_name}}\n {{^model.display_name}}<span class=\"aps-not-set\">Not set</span>{{/model.display_name}}\n </div>\n <button type=\"button\" class=\"aps-field-action\" data-action=\"edit-display-name\" title=\"Edit\"><i class=\"bi bi-pencil\"></i></button>\n </div>\n <div class=\"aps-field-row\">\n <div class=\"aps-field-label\">First Name</div>\n <div class=\"aps-field-value\">\n {{#model.first_name}}{{model.first_name}}{{/model.first_name}}\n {{^model.first_name}}<span class=\"aps-not-set\">Not set</span>{{/model.first_name}}\n </div>\n <button type=\"button\" class=\"aps-field-action\" data-action=\"edit-first-name\" title=\"Edit\"><i class=\"bi bi-pencil\"></i></button>\n </div>\n <div class=\"aps-field-row\">\n <div class=\"aps-field-label\">Last Name</div>\n <div class=\"aps-field-value\">\n {{#model.last_name}}{{model.last_name}}{{/model.last_name}}\n {{^model.last_name}}<span class=\"aps-not-set\">Not set</span>{{/model.last_name}}\n </div>\n <button type=\"button\" class=\"aps-field-action\" data-action=\"edit-last-name\" title=\"Edit\"><i class=\"bi bi-pencil\"></i></button>\n </div>\n\n <!-- Details -->\n <div class=\"aps-section-label\">Details</div>\n <div class=\"aps-field-row\">\n <div class=\"aps-field-label\">Date of Birth</div>\n <div class=\"aps-field-value\">\n {{#hasDob|bool}}\n {{dobFormatted}}\n {{#model.is_dob_verified|bool}}<span class=\"aps-badge-ok\">Verified</span>{{/model.is_dob_verified|bool}}\n {{^model.is_dob_verified|bool}}<span class=\"aps-badge-warn\">Unverified</span>{{/model.is_dob_verified|bool}}\n {{/hasDob|bool}}\n {{^hasDob|bool}}<span class=\"aps-not-set\">Not set</span>{{/hasDob|bool}}\n </div>\n {{#hasDob|bool}}\n {{#model.is_dob_verified|bool}}\n <button type=\"button\" class=\"aps-field-action\" data-action=\"unverify-dob\" title=\"Mark as unverified\"><i class=\"bi bi-x-circle\"></i></button>\n {{/model.is_dob_verified|bool}}\n {{^model.is_dob_verified|bool}}\n <button type=\"button\" class=\"aps-field-action\" data-action=\"force-verify-dob\" title=\"Force verify\"><i class=\"bi bi-patch-check\"></i></button>\n {{/model.is_dob_verified|bool}}\n {{/hasDob|bool}}\n <button type=\"button\" class=\"aps-field-action\" data-action=\"edit-dob\" title=\"Edit\"><i class=\"bi bi-pencil\"></i></button>\n </div>\n <div class=\"aps-field-row\">\n <div class=\"aps-field-label\">Timezone</div>\n <div class=\"aps-field-value\">{{timezoneDisplay}}</div>\n <button type=\"button\" class=\"aps-field-action\" data-action=\"edit-timezone\" title=\"Edit\"><i class=\"bi bi-pencil\"></i></button>\n </div>\n\n <!-- Address -->\n <div class=\"aps-section-label\">Address</div>\n <div class=\"aps-field-row\">\n <div class=\"aps-field-label\">Address</div>\n <div class=\"aps-field-value\">\n {{#hasAddress|bool}}{{addressSummary}}{{/hasAddress|bool}}\n {{^hasAddress|bool}}<span class=\"aps-not-set\">Not set</span>{{/hasAddress|bool}}\n </div>\n <button type=\"button\" class=\"aps-field-action\" data-action=\"edit-address\" title=\"Edit\"><i class=\"bi bi-pencil\"></i></button>\n </div>\n `,\n ...options\n });\n }\n\n // ── Computed properties ─────────────────────────\n\n get hasDob() {\n return !!this.model?.get('dob');\n }\n\n get dobFormatted() {\n const dob = this.model?.get('dob');\n if (!dob) return '';\n try {\n const [year, month, day] = dob.split('-');\n return `${month}/${day}/${year}`;\n } catch {\n return dob;\n }\n }\n\n get timezoneDisplay() {\n const meta = this.model?.get('metadata') || {};\n return meta.timezone || 'Not set';\n }\n\n get hasAddress() {\n const meta = this.model?.get('metadata') || {};\n return !!(meta.street || meta.city || meta.state || meta.zip || meta.country);\n }\n\n get addressSummary() {\n const meta = this.model?.get('metadata') || {};\n return [meta.street, meta.city, meta.state, meta.zip, meta.country].filter(Boolean).join(', ');\n }\n\n // ── Verification overrides ────────────────────────\n\n async onActionForceVerifyDob() {\n const confirmed = await Dialog.confirm(\n 'Mark date of birth as verified?',\n 'Force Verify DOB'\n );\n if (!confirmed) return true;\n await this._saveField({ is_dob_verified: true }, 'DOB marked as verified');\n return true;\n }\n\n async onActionUnverifyDob() {\n const confirmed = await Dialog.confirm(\n 'Mark date of birth as unverified?',\n 'Unverify DOB'\n );\n if (!confirmed) return true;\n await this._saveField({ is_dob_verified: false }, 'DOB marked as unverified');\n return true;\n }\n\n // ── Action handlers ─────────────────────────────\n\n async onActionEditDisplayName() {\n const name = await Dialog.prompt(\n 'Display name:',\n 'Edit Display Name',\n { defaultValue: this.model.get('display_name') || '' }\n );\n if (name !== null && name.trim()) {\n await this._saveField({ display_name: name.trim() }, 'Display name');\n }\n return true;\n }\n\n async onActionEditFirstName() {\n const name = await Dialog.prompt(\n 'First name:',\n 'Edit First Name',\n { defaultValue: this.model.get('first_name') || '' }\n );\n if (name !== null) {\n await this._saveField({ first_name: name.trim() }, 'First name');\n }\n return true;\n }\n\n async onActionEditLastName() {\n const name = await Dialog.prompt(\n 'Last name:',\n 'Edit Last Name',\n { defaultValue: this.model.get('last_name') || '' }\n );\n if (name !== null) {\n await this._saveField({ last_name: name.trim() }, 'Last name');\n }\n return true;\n }\n\n async onActionEditDob() {\n const data = await Dialog.showForm({\n title: 'Date of Birth',\n size: 'sm',\n fields: [{ name: 'dob', type: 'date', label: 'Date of Birth', cols: 12 }],\n data: { dob: this.model.get('dob') || '' }\n });\n if (!data) return true;\n await this._saveField({ dob: data.dob || null }, 'Date of birth');\n return true;\n }\n\n async onActionEditTimezone() {\n const meta = this.model.get('metadata') || {};\n const data = await Dialog.showForm({\n title: 'Change Timezone',\n size: 'sm',\n fields: [{\n name: 'timezone',\n type: 'select',\n label: 'Timezone',\n cols: 12,\n options: [\n { value: 'America/New_York', text: 'Eastern Time (ET)' },\n { value: 'America/Chicago', text: 'Central Time (CT)' },\n { value: 'America/Denver', text: 'Mountain Time (MT)' },\n { value: 'America/Los_Angeles', text: 'Pacific Time (PT)' },\n { value: 'America/Anchorage', text: 'Alaska Time (AKT)' },\n { value: 'Pacific/Honolulu', text: 'Hawaii Time (HT)' },\n { value: 'UTC', text: 'UTC' },\n { value: 'Europe/London', text: 'London (GMT/BST)' },\n { value: 'Europe/Paris', text: 'Paris (CET/CEST)' },\n { value: 'Europe/Berlin', text: 'Berlin (CET/CEST)' },\n { value: 'Asia/Tokyo', text: 'Tokyo (JST)' },\n { value: 'Asia/Shanghai', text: 'Shanghai (CST)' },\n { value: 'Australia/Sydney', text: 'Sydney (AEST)' }\n ]\n }],\n data: { timezone: meta.timezone || '' }\n });\n if (!data) return true;\n await this._saveField({ metadata: { ...meta, timezone: data.timezone } }, 'Timezone');\n return true;\n }\n\n async onActionEditAddress() {\n const meta = this.model.get('metadata') || {};\n const data = await Dialog.showForm({\n title: 'Edit Address',\n size: 'md',\n fields: [\n { name: 'street', type: 'text', label: 'Street', placeholder: '123 Main St', cols: 12 },\n { name: 'city', type: 'text', label: 'City', cols: 6 },\n { name: 'state', type: 'text', label: 'State / Province', cols: 6 },\n { name: 'zip', type: 'text', label: 'Zip / Postal Code', cols: 6 },\n { name: 'country', type: 'text', label: 'Country', cols: 6 }\n ],\n data: { street: meta.street || '', city: meta.city || '', state: meta.state || '', zip: meta.zip || '', country: meta.country || '' }\n });\n if (!data) return true;\n\n const updatedMeta = { ...meta, street: data.street || '', city: data.city || '', state: data.state || '', zip: data.zip || '', country: data.country || '' };\n await this._saveField({ metadata: updatedMeta }, 'Address');\n return true;\n }\n\n // ── Helpers ─────────────────────────────────────\n\n async _saveField(fields, label) {\n const resp = await this.model.save(fields);\n if (resp.status === 200) {\n this.getApp()?.toast?.success(`${label} updated`);\n await this.render();\n } else {\n this.getApp()?.toast?.error(resp.message || `Failed to update ${label.toLowerCase()}`);\n }\n }\n}\n","/**\n * AdminSecuritySection - Admin security management for a user\n *\n * Admin powers: force password reset, enable/disable MFA,\n * revoke all sessions, view/delete passkeys, view recovery codes.\n * Uses admin endpoints and model.save() against /api/user/<id>.\n */\nimport View from '@core/View.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\nimport rest from '@core/Rest.js';\nimport { PasskeyList, PasskeyForms } from '@core/models/Passkeys.js';\n\nexport default class AdminSecuritySection extends View {\n constructor(options = {}) {\n super({\n className: 'admin-security-section',\n template: `\n <style>\n .as-section-label { font-size: 0.7rem; font-weight: 700; text-transform: uppercase; letter-spacing: 0.06em; color: #adb5bd; margin-bottom: 0.5rem; margin-top: 1.75rem; }\n .as-section-label:first-child { margin-top: 0; }\n .as-item { display: flex; align-items: center; gap: 0.85rem; padding: 0.85rem 1rem; border: 1px solid #f0f0f0; border-radius: 8px; margin-bottom: 0.5rem; cursor: pointer; transition: border-color 0.15s, background 0.15s; }\n .as-item:hover { border-color: #dee2e6; background: #fafbfd; }\n .as-icon { width: 36px; height: 36px; border-radius: 8px; display: flex; align-items: center; justify-content: center; font-size: 1rem; flex-shrink: 0; }\n .as-info { flex: 1; min-width: 0; }\n .as-title { font-weight: 600; font-size: 0.88rem; }\n .as-desc { font-size: 0.78rem; color: #6c757d; }\n .as-badge { font-size: 0.72rem; padding: 0.15em 0.5em; border-radius: 3px; flex-shrink: 0; }\n .as-chevron { color: #ced4da; font-size: 0.8rem; flex-shrink: 0; }\n </style>\n\n <div class=\"as-section-label\">Authentication</div>\n\n <div class=\"as-item\" data-action=\"send-password-reset\">\n <div class=\"as-icon bg-primary bg-opacity-10 text-primary\"><i class=\"bi bi-envelope\"></i></div>\n <div class=\"as-info\">\n <div class=\"as-title\">Send Password Reset</div>\n <div class=\"as-desc\">Send a password reset email to {{model.email}}</div>\n </div>\n <span class=\"as-badge bg-light text-muted border\">Send</span>\n </div>\n\n <div class=\"as-item\" data-action=\"send-magic-link\">\n <div class=\"as-icon\" style=\"background: rgba(13,110,253,0.1); color: #0d6efd;\"><i class=\"bi bi-link-45deg\"></i></div>\n <div class=\"as-info\">\n <div class=\"as-title\">Send Magic Login Link</div>\n <div class=\"as-desc\">Send a one-click login link to {{model.email}}</div>\n </div>\n <span class=\"as-badge bg-light text-muted border\">Send</span>\n </div>\n\n {{^model.is_email_verified|bool}}\n <div class=\"as-item\" data-action=\"send-email-verification\">\n <div class=\"as-icon\" style=\"background: rgba(25,135,84,0.1); color: #198754;\"><i class=\"bi bi-envelope-check\"></i></div>\n <div class=\"as-info\">\n <div class=\"as-title\">Send Email Verification</div>\n <div class=\"as-desc\">Send a verification email to {{model.email}}</div>\n </div>\n <span class=\"as-badge bg-light text-muted border\">Send</span>\n </div>\n {{/model.is_email_verified|bool}}\n\n <div class=\"as-item\" data-action=\"set-password\">\n <div class=\"as-icon bg-warning bg-opacity-10 text-warning\"><i class=\"bi bi-key\"></i></div>\n <div class=\"as-info\">\n <div class=\"as-title\">Set Password</div>\n <div class=\"as-desc\">Set a new password directly for this user</div>\n </div>\n <span class=\"as-badge bg-light text-muted border\">Set</span>\n </div>\n\n <div class=\"as-section-label\">Multi-Factor Authentication</div>\n\n <div class=\"as-item\" data-action=\"toggle-mfa\">\n <div class=\"as-icon\" style=\"background: rgba(111,66,193,0.1); color: #6f42c1;\"><i class=\"bi bi-shield-lock\"></i></div>\n <div class=\"as-info\">\n <div class=\"as-title\">MFA Requirement</div>\n <div class=\"as-desc\">\n {{#model.requires_mfa|bool}}User is required to use MFA{{/model.requires_mfa|bool}}\n {{^model.requires_mfa|bool}}MFA is not required for this user{{/model.requires_mfa|bool}}\n </div>\n </div>\n {{#model.requires_mfa|bool}}\n <span class=\"as-badge bg-success bg-opacity-10 text-success border\">Enabled</span>\n {{/model.requires_mfa|bool}}\n {{^model.requires_mfa|bool}}\n <span class=\"as-badge bg-light text-muted border\">Disabled</span>\n {{/model.requires_mfa|bool}}\n </div>\n\n <div class=\"as-item\" data-action=\"manage-passkeys\">\n <div class=\"as-icon bg-success bg-opacity-10 text-success\"><i class=\"bi bi-fingerprint\"></i></div>\n <div class=\"as-info\">\n <div class=\"as-title\">Passkeys</div>\n <div class=\"as-desc\">View and manage registered passkeys</div>\n </div>\n <i class=\"bi bi-chevron-right as-chevron\"></i>\n </div>\n\n {{#model.requires_mfa|bool}}\n <div class=\"as-item\" data-action=\"view-recovery-codes\">\n <div class=\"as-icon\" style=\"background: rgba(111,66,193,0.1); color: #6f42c1;\"><i class=\"bi bi-file-earmark-lock\"></i></div>\n <div class=\"as-info\">\n <div class=\"as-title\">Recovery Codes</div>\n <div class=\"as-desc\">View remaining recovery codes</div>\n </div>\n <i class=\"bi bi-chevron-right as-chevron\"></i>\n </div>\n\n <div class=\"as-item\" data-action=\"disable-totp\">\n <div class=\"as-icon\" style=\"background: rgba(220,53,69,0.1); color: #dc3545;\"><i class=\"bi bi-shield-x\"></i></div>\n <div class=\"as-info\">\n <div class=\"as-title\">Disable Authenticator</div>\n <div class=\"as-desc\">Remove TOTP requirement for this user</div>\n </div>\n </div>\n {{/model.requires_mfa|bool}}\n\n <div class=\"as-section-label\">Sessions</div>\n\n <div class=\"as-item\" data-action=\"revoke-all-sessions\">\n <div class=\"as-icon\" style=\"background: rgba(220,53,69,0.1); color: #dc3545;\"><i class=\"bi bi-box-arrow-right\"></i></div>\n <div class=\"as-info\">\n <div class=\"as-title\">Revoke All Sessions</div>\n <div class=\"as-desc\">Force sign-out from all devices</div>\n </div>\n </div>\n `,\n ...options\n });\n }\n\n // ── Password actions ────────────────────────────\n\n async onActionSendPasswordReset() {\n const app = this.getApp();\n const email = this.model.get('email');\n\n const confirmed = await Dialog.confirm(\n `Send a password reset email to <strong>${email}</strong>?`,\n 'Send Password Reset'\n );\n if (!confirmed) return true;\n\n const resp = await rest.POST('/api/auth/password/reset', { email });\n if (resp.success) {\n app?.toast?.success('Password reset email sent');\n } else {\n app?.toast?.error(resp.message || 'Failed to send password reset');\n }\n return true;\n }\n\n async onActionSendEmailVerification() {\n const app = this.getApp();\n const email = this.model.get('email');\n\n const confirmed = await Dialog.confirm(\n `Send a verification email to <strong>${email}</strong>?`,\n 'Send Email Verification'\n );\n if (!confirmed) return true;\n\n const resp = await rest.POST('/api/auth/email/verify', { email });\n if (resp.success) {\n app?.toast?.success('Verification email sent');\n } else {\n app?.toast?.error(resp.message || 'Failed to send verification email');\n }\n return true;\n }\n\n async onActionSendMagicLink() {\n const app = this.getApp();\n const email = this.model.get('email');\n\n const confirmed = await Dialog.confirm(\n `Send a magic login link to <strong>${email}</strong>? They will be able to sign in with one click.`,\n 'Send Magic Login Link'\n );\n if (!confirmed) return true;\n\n const resp = await rest.POST('/api/auth/magic-link', { email });\n if (resp.success) {\n app?.toast?.success('Magic login link sent');\n } else {\n app?.toast?.error(resp.message || 'Failed to send magic link');\n }\n return true;\n }\n\n async onActionSetPassword() {\n const app = this.getApp();\n const data = await Dialog.showForm({\n title: 'Set Password',\n size: 'sm',\n fields: [\n { name: 'password', type: 'password', label: 'New Password', required: true, cols: 12, help: 'Set a new password for this user.' },\n { name: 'confirm', type: 'password', label: 'Confirm Password', required: true, cols: 12 }\n ]\n });\n if (!data) return true;\n\n if (data.password !== data.confirm) {\n app?.toast?.error('Passwords do not match');\n return true;\n }\n\n const resp = await this.model.save({ password: data.password });\n if (resp.status === 200) {\n app?.toast?.success('Password updated');\n } else {\n app?.toast?.error(resp.message || 'Failed to set password');\n }\n return true;\n }\n\n // ── MFA actions ─────────────────────────────────\n\n async onActionToggleMfa() {\n const app = this.getApp();\n const currentMfa = this.model.get('requires_mfa');\n const action = currentMfa ? 'disable' : 'enable';\n\n const confirmed = await Dialog.confirm(\n `${currentMfa ? 'Disable' : 'Enable'} MFA requirement for this user?`,\n `${currentMfa ? 'Disable' : 'Enable'} MFA`\n );\n if (!confirmed) return true;\n\n const resp = await this.model.save({ requires_mfa: !currentMfa });\n if (resp.status === 200) {\n app?.toast?.success(`MFA ${action}d`);\n await this.render();\n } else {\n app?.toast?.error(resp.message || `Failed to ${action} MFA`);\n }\n return true;\n }\n\n async onActionDisableTotp() {\n const app = this.getApp();\n const confirmed = await Dialog.confirm(\n 'Disable the authenticator app for this user? They will no longer need a TOTP code to sign in.',\n 'Disable Authenticator'\n );\n if (!confirmed) return true;\n\n const resp = await rest.DELETE(`/api/user/${this.model.id}/totp`);\n if (resp.success) {\n this.model.set('requires_mfa', false);\n app?.toast?.success('Authenticator disabled');\n await this.render();\n } else {\n app?.toast?.error(resp.message || 'Failed to disable authenticator');\n }\n return true;\n }\n\n async onActionManagePasskeys() {\n const collection = new PasskeyList({ params: { user: this.model.id } });\n try {\n await collection.fetch();\n } catch (e) {\n // ignore\n }\n\n const items = collection.models || [];\n const view = new View({\n template: `\n <style>\n .pk-row { display: flex; align-items: center; gap: 0.75rem; padding: 0.65rem 0.75rem; border: 1px solid #f0f0f0; border-radius: 8px; margin-bottom: 0.4rem; }\n .pk-icon { width: 32px; height: 32px; background: #e7f1ff; border-radius: 6px; display: flex; align-items: center; justify-content: center; color: #0d6efd; font-size: 0.9rem; flex-shrink: 0; }\n .pk-info { flex: 1; min-width: 0; }\n .pk-name { font-weight: 600; font-size: 0.85rem; }\n .pk-meta { font-size: 0.73rem; color: #6c757d; }\n .pk-actions .btn { padding: 0.2rem 0.4rem; font-size: 0.75rem; }\n .pk-empty { text-align: center; padding: 2rem 1rem; color: #6c757d; }\n .pk-empty i { font-size: 2rem; color: #ced4da; display: block; margin-bottom: 0.5rem; }\n </style>\n {{#passkeys}}\n <div class=\"pk-row\">\n <div class=\"pk-icon\"><i class=\"bi bi-fingerprint\"></i></div>\n <div class=\"pk-info\">\n <div class=\"pk-name\">{{.friendly_name|default:'Unnamed Passkey'}}</div>\n <div class=\"pk-meta\">Created {{.created|date}} &middot; Last used {{.last_used|relative|default:'never'}} &middot; {{.sign_count}} uses</div>\n </div>\n <div class=\"pk-actions\">\n <button type=\"button\" class=\"btn btn-outline-secondary\" data-action=\"edit-passkey\" data-id=\"{{.id}}\" title=\"Edit\"><i class=\"bi bi-pencil\"></i></button>\n <button type=\"button\" class=\"btn btn-outline-danger\" data-action=\"delete-passkey\" data-id=\"{{.id}}\" title=\"Delete\"><i class=\"bi bi-trash\"></i></button>\n </div>\n </div>\n {{/passkeys}}\n {{^passkeys|bool}}\n <div class=\"pk-empty\">\n <i class=\"bi bi-fingerprint\"></i>\n No passkeys registered\n </div>\n {{/passkeys|bool}}\n `\n });\n view.passkeys = items.map(p => p.toJSON ? p.toJSON() : p);\n\n view.onActionEditPasskey = async (event, el) => {\n const id = el.dataset.id;\n const passkey = items.find(p => String(p.id) === String(id));\n if (passkey) {\n await Dialog.showModelForm({ title: 'Edit Passkey', model: passkey, fields: PasskeyForms.edit.fields, size: 'sm' });\n }\n return true;\n };\n\n view.onActionDeletePasskey = async (event, el) => {\n const id = el.dataset.id;\n const confirmed = await Dialog.confirm('Delete this passkey?', 'Delete Passkey');\n if (confirmed) {\n const passkey = items.find(p => String(p.id) === String(id));\n if (passkey) {\n await passkey.destroy();\n this.getApp()?.toast?.success('Passkey deleted');\n }\n }\n return true;\n };\n\n await Dialog.showDialog({\n title: 'Passkeys',\n body: view,\n size: 'md',\n buttons: [{ text: 'Close', class: 'btn-outline-secondary', dismiss: true }]\n });\n return true;\n }\n\n async onActionViewRecoveryCodes() {\n const app = this.getApp();\n const resp = await rest.GET(`/api/user/${this.model.id}/totp/recovery-codes`, {}, { dataOnly: true });\n if (!resp.success || !resp.data) {\n app?.toast?.error(resp.message || 'Failed to load recovery codes');\n return true;\n }\n\n const { remaining, codes } = resp.data;\n const view = new View({\n template: `\n <style>\n .rc-info { font-size: 0.82rem; color: #6c757d; margin-bottom: 1rem; }\n .rc-remaining { font-weight: 600; color: #495057; }\n .rc-list { display: grid; grid-template-columns: 1fr 1fr; gap: 0.3rem; }\n .rc-code { font-family: monospace; font-size: 0.85rem; padding: 0.35rem 0.6rem; background: #f8f9fa; border: 1px solid #e9ecef; border-radius: 4px; text-align: center; }\n </style>\n <div class=\"rc-info\"><span class=\"rc-remaining\">{{remaining}}</span> recovery codes remaining</div>\n <div class=\"rc-list\">\n {{#codes}}<div class=\"rc-code\">{{.}}</div>{{/codes}}\n </div>\n `\n });\n view.remaining = remaining;\n view.codes = codes || [];\n\n await Dialog.showDialog({\n title: 'Recovery Codes',\n body: view,\n size: 'sm',\n buttons: [{ text: 'Close', class: 'btn-outline-secondary', dismiss: true }]\n });\n return true;\n }\n\n // ── Session actions ─────────────────────────────\n\n async onActionRevokeAllSessions() {\n const app = this.getApp();\n const confirmed = await Dialog.confirm(\n 'Revoke all sessions for this user? They will be signed out of all devices immediately.',\n 'Revoke All Sessions'\n );\n if (!confirmed) return true;\n\n const resp = await rest.POST(`/api/user/${this.model.id}/sessions/revoke`);\n if (resp.success) {\n app?.toast?.success('All sessions revoked');\n } else {\n app?.toast?.error(resp.message || 'Failed to revoke sessions');\n }\n return true;\n }\n}\n","/**\n * AdminConnectedSection - Admin view of a user's OAuth connections\n *\n * View linked providers and unlink on behalf of user.\n * Uses admin-scoped endpoints.\n */\nimport View from '@core/View.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\nimport rest from '@core/Rest.js';\n\nconst PROVIDER_ICONS = {\n google: 'bi-google',\n github: 'bi-github',\n microsoft: 'bi-microsoft',\n apple: 'bi-apple',\n facebook: 'bi-facebook',\n twitter: 'bi-twitter-x',\n linkedin: 'bi-linkedin'\n};\n\nexport default class AdminConnectedSection extends View {\n constructor(options = {}) {\n super({\n className: 'admin-connected-section',\n template: `\n <style>\n .ac-row { display: flex; align-items: center; gap: 0.85rem; padding: 0.85rem 1rem; border: 1px solid #f0f0f0; border-radius: 8px; margin-bottom: 0.5rem; }\n .ac-icon { width: 36px; height: 36px; border-radius: 8px; display: flex; align-items: center; justify-content: center; font-size: 1rem; flex-shrink: 0; background: #f0f0f0; color: #495057; }\n .ac-info { flex: 1; min-width: 0; }\n .ac-provider { font-weight: 600; font-size: 0.88rem; text-transform: capitalize; }\n .ac-meta { font-size: 0.78rem; color: #6c757d; }\n .ac-actions .btn { font-size: 0.75rem; padding: 0.25rem 0.5rem; }\n .ac-empty { text-align: center; padding: 2rem 1rem; color: #6c757d; }\n .ac-empty i { font-size: 2rem; color: #ced4da; display: block; margin-bottom: 0.5rem; }\n </style>\n\n {{#connections}}\n <div class=\"ac-row\">\n <div class=\"ac-icon\"><i class=\"bi {{.icon}}\"></i></div>\n <div class=\"ac-info\">\n <div class=\"ac-provider\">{{.provider}}</div>\n <div class=\"ac-meta\">{{.email}} &middot; Connected {{.created|relative}}</div>\n </div>\n <div class=\"ac-actions\">\n <button type=\"button\" class=\"btn btn-outline-danger\" data-action=\"unlink\" data-id=\"{{.id}}\" title=\"Unlink\"><i class=\"bi bi-x-lg me-1\"></i>Unlink</button>\n </div>\n </div>\n {{/connections}}\n {{^connections|bool}}\n <div class=\"ac-empty\">\n <i class=\"bi bi-plug\"></i>\n No connected accounts\n </div>\n {{/connections|bool}}\n `,\n ...options\n });\n this.connections = [];\n }\n\n async onBeforeRender() {\n try {\n const resp = await rest.GET(`/api/user/${this.model.id}/oauth_connection`);\n const results = resp?.data?.results || resp?.data || [];\n this.connections = results.map(c => ({\n ...c,\n icon: PROVIDER_ICONS[c.provider] || 'bi-link-45deg'\n }));\n } catch (e) {\n this.connections = [];\n }\n }\n\n async onActionUnlink(event, el) {\n const id = el.dataset.id;\n const connection = this.connections.find(c => String(c.id) === String(id));\n const provider = connection?.provider || 'this account';\n\n const confirmed = await Dialog.confirm(\n `Unlink ${provider} for this user?`,\n 'Unlink Account'\n );\n if (!confirmed) return true;\n\n const resp = await rest.DELETE(`/api/user/${this.model.id}/oauth_connection/${id}`);\n if (resp.success) {\n this.getApp()?.toast?.success(`${provider} account unlinked`);\n await this.render();\n } else {\n this.getApp()?.toast?.error(resp.message || 'Failed to unlink account');\n }\n return true;\n }\n}\n","/**\n * AdminNotificationsSection - Admin view/edit of user notification preferences\n *\n * Per-kind, per-channel toggle grid. Uses admin-scoped endpoints.\n */\nimport View from '@core/View.js';\nimport rest from '@core/Rest.js';\n\nconst CHANNEL_LABELS = {\n in_app: 'In-App',\n email: 'Email',\n push: 'Push'\n};\n\nconst CHANNELS = ['in_app', 'email', 'push'];\n\nexport default class AdminNotificationsSection extends View {\n constructor(options = {}) {\n super({\n className: 'admin-notifications-section',\n template: `\n <style>\n .an-table { width: 100%; border-collapse: collapse; }\n .an-table th { font-size: 0.7rem; font-weight: 700; text-transform: uppercase; letter-spacing: 0.06em; color: #adb5bd; padding: 0.5rem 0.75rem; border-bottom: 2px solid #e9ecef; }\n .an-table th:first-child { text-align: left; }\n .an-table th:not(:first-child) { text-align: center; width: 80px; }\n .an-table td { padding: 0.65rem 0.75rem; border-bottom: 1px solid #f0f0f0; }\n .an-table td:first-child { font-size: 0.88rem; font-weight: 500; text-transform: capitalize; }\n .an-table td:not(:first-child) { text-align: center; }\n .an-table tr:last-child td { border-bottom: none; }\n .an-empty { text-align: center; padding: 2rem 1rem; color: #6c757d; }\n .an-empty i { font-size: 2rem; color: #ced4da; display: block; margin-bottom: 0.5rem; }\n </style>\n\n {{#hasPreferences|bool}}\n <table class=\"an-table\">\n <thead>\n <tr>\n <th>Type</th>\n {{#channels}}\n <th>{{.label}}</th>\n {{/channels}}\n </tr>\n </thead>\n <tbody>\n {{#preferenceRows}}\n <tr>\n <td>{{.kindLabel}}</td>\n {{#.toggles}}\n <td>\n <input type=\"checkbox\" class=\"form-check-input\"\n data-action=\"toggle-pref\"\n data-kind=\"{{.kind}}\"\n data-channel=\"{{.channel}}\"\n {{#.checked}}checked{{/.checked}}>\n </td>\n {{/.toggles}}\n </tr>\n {{/preferenceRows}}\n </tbody>\n </table>\n {{/hasPreferences|bool}}\n {{^hasPreferences|bool}}\n <div class=\"an-empty\">\n <i class=\"bi bi-bell\"></i>\n No notification preferences configured\n </div>\n {{/hasPreferences|bool}}\n `,\n ...options\n });\n this.preferences = {};\n }\n\n get channels() {\n return CHANNELS.map(ch => ({ key: ch, label: CHANNEL_LABELS[ch] || ch }));\n }\n\n get hasPreferences() {\n return Object.keys(this.preferences).length > 0;\n }\n\n get preferenceRows() {\n return Object.keys(this.preferences).sort().map(kind => ({\n kind,\n kindLabel: kind.replace(/[_-]/g, ' ').replace(/\\b\\w/g, c => c.toUpperCase()),\n toggles: CHANNELS.map(channel => ({\n kind,\n channel,\n checked: this.preferences[kind]?.[channel] !== false\n }))\n }));\n }\n\n async onBeforeRender() {\n try {\n const resp = await rest.GET(`/api/user/${this.model.id}/notification/preferences`, {}, { dataOnly: true });\n this.preferences = resp?.data?.preferences || resp?.data || {};\n } catch (e) {\n this.preferences = {};\n }\n }\n\n async onActionTogglePref(event, el) {\n const kind = el.dataset.kind;\n const channel = el.dataset.channel;\n const checked = el.checked;\n\n if (!this.preferences[kind]) {\n this.preferences[kind] = {};\n }\n this.preferences[kind][channel] = checked;\n\n try {\n const resp = await rest.POST(`/api/user/${this.model.id}/notification/preferences`, {\n preferences: { [kind]: { [channel]: checked } }\n });\n if (!resp.success) {\n this.getApp()?.toast?.error(resp.message || 'Failed to update preference');\n el.checked = !checked;\n }\n } catch (e) {\n this.getApp()?.toast?.error('Failed to update preference');\n el.checked = !checked;\n }\n return true;\n }\n}\n","/**\n * AdminApiKeysSection - Admin view/revoke of a user's API keys\n *\n * View-only table of API keys with revoke/delete capability.\n * Admin cannot generate keys on behalf of users — only view and revoke.\n * Uses admin-scoped endpoints.\n */\nimport View from '@core/View.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\nimport rest from '@core/Rest.js';\n\nexport default class AdminApiKeysSection extends View {\n constructor(options = {}) {\n super({\n className: 'admin-api-keys-section',\n template: `\n <style>\n .aak-list { border: 1px solid #e9ecef; border-radius: 8px; overflow: hidden; }\n .aak-item { display: flex; align-items: center; padding: 0.75rem 1rem; border-bottom: 1px solid #f0f0f0; gap: 1rem; }\n .aak-item:last-child { border-bottom: none; }\n .aak-item-icon { color: #6c757d; font-size: 1.1rem; flex-shrink: 0; }\n .aak-item-info { flex: 1; min-width: 0; }\n .aak-item-name { font-weight: 600; font-size: 0.85rem; display: flex; align-items: center; gap: 0.5rem; }\n .aak-item-meta { font-size: 0.75rem; color: #6c757d; display: flex; gap: 1rem; flex-wrap: wrap; margin-top: 0.15rem; }\n .aak-item-meta i { margin-right: 0.2rem; }\n .aak-empty { padding: 2rem; text-align: center; color: #6c757d; font-size: 0.85rem; }\n .aak-empty i { font-size: 1.5rem; display: block; margin-bottom: 0.5rem; }\n </style>\n\n <div id=\"aak-keys-list\"></div>\n `,\n ...options\n });\n this.apiKeys = [];\n }\n\n async onBeforeRender() {\n await this._loadKeys();\n }\n\n async _loadKeys() {\n try {\n const resp = await rest.GET(`/api/user/${this.model.id}/api_keys`, {}, {}, { dataOnly: true });\n this.apiKeys = resp.success && Array.isArray(resp.data) ? resp.data : [];\n } catch (e) {\n this.apiKeys = [];\n }\n }\n\n onAfterRender() {\n this._renderKeysList();\n }\n\n _renderKeysList() {\n const container = this.element?.querySelector('#aak-keys-list');\n if (!container) return;\n\n if (!this.apiKeys.length) {\n container.innerHTML = `\n <div class=\"aak-list\">\n <div class=\"aak-empty\">\n <i class=\"bi bi-key\"></i>\n No API keys for this user\n </div>\n </div>`;\n return;\n }\n\n const rows = this.apiKeys.map(key => {\n const name = key.name || 'API Key';\n const created = key.created ? new Date(key.created * 1000).toLocaleDateString() : '';\n const expires = key.expires ? new Date(key.expires * 1000).toLocaleDateString() : 'Never';\n const lastUsed = key.last_used ? new Date(key.last_used * 1000).toLocaleDateString() : 'Never';\n const ips = key.allowed_ips?.length ? key.allowed_ips.join(', ') : 'Any';\n const isActive = key.is_active !== false;\n const statusBadge = isActive\n ? '<span class=\"badge bg-success\">Active</span>'\n : '<span class=\"badge bg-secondary\">Inactive</span>';\n const tokenPreview = key.token_prefix ? `${key.token_prefix}...` : '••••••••';\n\n return `\n <div class=\"aak-item\">\n <div class=\"aak-item-icon\"><i class=\"bi bi-key\"></i></div>\n <div class=\"aak-item-info\">\n <div class=\"aak-item-name\">${name} ${statusBadge}</div>\n <div class=\"aak-item-meta\">\n <span><i class=\"bi bi-code-square\"></i>${tokenPreview}</span>\n <span><i class=\"bi bi-calendar\"></i>Created ${created}</span>\n <span><i class=\"bi bi-clock\"></i>Expires ${expires}</span>\n <span><i class=\"bi bi-activity\"></i>Last used ${lastUsed}</span>\n <span><i class=\"bi bi-globe\"></i>IPs: ${ips}</span>\n </div>\n </div>\n <div>\n <button type=\"button\" class=\"btn btn-outline-danger btn-sm\" data-action=\"revoke-key\" data-id=\"${key.id}\" title=\"Revoke\">\n <i class=\"bi bi-trash\"></i>\n </button>\n </div>\n </div>`;\n }).join('');\n\n container.innerHTML = `<div class=\"aak-list\">${rows}</div>`;\n }\n\n async onActionRevokeKey(event, el) {\n const id = el.dataset.id;\n if (!id) return true;\n\n const confirmed = await Dialog.confirm(\n 'Revoke this API key? Any applications using it will lose access immediately.',\n 'Revoke API Key'\n );\n if (!confirmed) return true;\n\n const resp = await rest.DELETE(`/api/user/${this.model.id}/api_keys/${id}`, {}, {}, { dataOnly: true });\n if (resp.success) {\n this.getApp()?.toast?.success('API key revoked');\n await this._loadKeys();\n this._renderKeysList();\n } else {\n this.getApp()?.toast?.error(resp.message || 'Failed to revoke API key');\n }\n return true;\n }\n}\n","/**\n * UserView - Comprehensive user management interface\n *\n * Features:\n * - Header with avatar, name, contact info, status, and context menu\n * - Left-nav sidebar for section switching (Profile, Permissions, Groups, etc.)\n * - Integrated with DataView, TableView, FormView, and SideNavView\n * - Clean Bootstrap 5 styling\n */\n\nimport View from '@core/View.js';\nimport SideNavView from '@core/views/navigation/SideNavView.js';\nimport DataView from '@core/views/data/DataView.js';\nimport TableView from '@core/views/table/TableView.js';\nimport TableRow from '@core/views/table/TableRow.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport rest from '@core/Rest.js';\nimport { User, UserDataView, UserForms, UserDeviceList, UserDeviceLocationList } from '@core/models/User.js';\nimport { LogList } from '@core/models/Log.js';\nimport { IncidentEventList } from '@core/models/Incident.js';\nimport { MemberList } from '@core/models/Member.js';\nimport { PushDeviceList } from '@core/models/Push.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\nimport FormView from '@core/forms/FormView.js';\nimport AdminProfileSection from './sections/AdminProfileSection.js';\nimport AdminPersonalSection from './sections/AdminPersonalSection.js';\nimport AdminSecuritySection from './sections/AdminSecuritySection.js';\nimport AdminConnectedSection from './sections/AdminConnectedSection.js';\nimport AdminNotificationsSection from './sections/AdminNotificationsSection.js';\nimport AdminApiKeysSection from './sections/AdminApiKeysSection.js';\n// DeviceView and GeoIPView are opened automatically via VIEW_CLASS on row click\n\n\n// ── Custom TableRow classes for rich multiline rows ──\n\nclass DeviceRow extends TableRow {\n get deviceIcon() {\n const dev = this.model?.get('device_info')?.device || {};\n const os = this.model?.get('device_info')?.os || {};\n const isMobile = ['iPhone', 'Android'].some(m =>\n (dev.family || '').includes(m) || (os.family || '').includes(m)\n );\n return isMobile ? 'bi-phone' : 'bi-laptop';\n }\n get deviceName() {\n const dev = this.model?.get('device_info')?.device || {};\n return `${dev.brand || ''} ${dev.family || ''}`.trim() || 'Unknown Device';\n }\n get deviceModel() {\n return this.model?.get('device_info')?.device?.model || '';\n }\n get browserName() {\n const ua = this.model?.get('device_info')?.user_agent || {};\n return ua.family ? `${ua.family} ${ua.major || ''}`.trim() : '';\n }\n get osName() {\n const os = this.model?.get('device_info')?.os || {};\n return os.family ? `${os.family} ${os.major || ''}`.trim() : '';\n }\n get deviceMeta() {\n return [this.browserName, this.osName].filter(Boolean).join(' · ') || '—';\n }\n}\n\nclass LocationRow extends TableRow {\n get deviceIcon() {\n const dev = this.model?.get('user_device')?.device_info?.device || {};\n const os = this.model?.get('user_device')?.device_info?.os || {};\n const isMobile = ['iPhone', 'Android'].some(m =>\n (dev.family || '').includes(m) || (os.family || '').includes(m)\n );\n return isMobile ? 'bi-phone' : 'bi-laptop';\n }\n get browserName() {\n const ua = this.model?.get('user_device')?.device_info?.user_agent || {};\n return ua.family ? `${ua.family} ${ua.major || ''}`.trim() : 'Unknown';\n }\n get deviceName() {\n const dev = this.model?.get('user_device')?.device_info?.device || {};\n return `${dev.brand || ''} ${dev.family || ''}`.trim() || 'Unknown';\n }\n get locationText() {\n const geo = this.model?.get('geolocation') || {};\n return [geo.city, geo.region].filter(Boolean).join(', ') || geo.country_name || '—';\n }\n get countryName() {\n return this.model?.get('geolocation')?.country_name || '';\n }\n get threatFlags() {\n const geo = this.model?.get('geolocation') || {};\n const flags = [];\n if (geo.is_vpn) flags.push('<span class=\"badge bg-warning text-dark\" style=\"font-size:0.6rem;\">VPN</span>');\n if (geo.is_tor) flags.push('<span class=\"badge bg-danger\" style=\"font-size:0.6rem;\">Tor</span>');\n if (geo.is_proxy) flags.push('<span class=\"badge bg-warning text-dark\" style=\"font-size:0.6rem;\">Proxy</span>');\n return flags.join(' ');\n }\n get hasThreatFlags() {\n const geo = this.model?.get('geolocation') || {};\n return !!(geo.is_vpn || geo.is_tor || geo.is_proxy);\n }\n}\n\nclass UserView extends View {\n constructor(options = {}) {\n super({\n className: 'user-view',\n ...options\n });\n\n // User model instance\n this.model = options.model || new User(options.data || {});\n\n // Section views\n this.sideNavView = null;\n\n // Set template\n this.template = `\n <div class=\"user-view-container\">\n <!-- User Header + Context Menu -->\n <div class=\"d-flex justify-content-between align-items-start mb-4\">\n <div data-container=\"user-header\" style=\"flex: 1;\"></div>\n <div data-container=\"user-context-menu\" class=\"ms-3 flex-shrink-0\"></div>\n </div>\n <!-- Side Nav Container -->\n <div data-container=\"user-sidenav\" style=\"min-height: 400px;\"></div>\n </div>\n `;\n }\n\n async onInit() {\n // ── Header ──────────────────────────────────\n this.header = new View({\n containerId: 'user-header',\n template: `\n <div class=\"d-flex justify-content-between align-items-start\">\n <!-- Left Side: Primary Identity -->\n <div class=\"d-flex align-items-center gap-3\">\n {{{model.avatar|avatar('md','rounded-circle')}}}\n <div>\n <h3 class=\"mb-0\">{{model.display_name|default('Unnamed User')}}</h3>\n <a href=\"mailto:{{model.email}}\" class=\"text-decoration-none text-body\">{{model.email}}</a>{{{model.email|clipboard('icon-only')}}}\n {{#model.phone_number}}\n <div class=\"text-muted small mt-1\">{{{model.phone_number|phone(false)}}}</div>\n {{/model.phone_number}}\n </div>\n </div>\n\n <!-- Right Side: Status -->\n <div class=\"text-end\">\n <div class=\"d-flex align-items-center justify-content-end gap-3\">\n <span class=\"d-inline-flex align-items-center gap-1\" title=\"{{model.is_online|boolean('Online','Offline')}}\">\n <i class=\"bi bi-circle-fill {{model.is_online|boolean('text-success','text-secondary')}}\" style=\"font-size: 0.5rem;\"></i>\n <span class=\"small\">{{model.is_online|boolean('Online','Offline')}}</span>\n </span>\n <span class=\"d-inline-flex align-items-center gap-1\" style=\"cursor: pointer;\"\n data-action=\"toggle-active\"\n title=\"{{model.is_active|boolean('Click to deactivate','Click to activate')}}\">\n <i class=\"bi {{model.is_active|boolean('bi-toggle-on text-success','bi-toggle-off text-secondary')}}\" style=\"font-size: 1.1rem;\"></i>\n <span class=\"small\">{{model.is_active|boolean('Active','Inactive')}}</span>\n </span>\n </div>\n {{#model.last_activity}}\n <div class=\"text-muted small mt-1\">Last active {{model.last_activity|relative}}</div>\n {{/model.last_activity}}\n </div>\n </div>`\n });\n\n this.header.setModel(this.model);\n this.addChild(this.header);\n\n // ── Section views ───────────────────────────\n\n const profileView = new AdminProfileSection({ model: this.model });\n const personalView = new AdminPersonalSection({ model: this.model });\n const securityView = new AdminSecuritySection({ model: this.model });\n const connectedView = new AdminConnectedSection({ model: this.model });\n const notificationsView = new AdminNotificationsSection({ model: this.model });\n const apiKeysView = new AdminApiKeysSection({ model: this.model });\n\n const permsView = new FormView({\n fields: User.PERMISSION_FIELDS,\n model: this.model,\n autosaveModelField: true\n });\n\n // Groups\n const membersCollection = new MemberList({\n params: { user: this.model.get('id'), size: 5 }\n });\n const groupsView = new TableView({\n collection: membersCollection,\n hideActivePillNames: ['user'],\n columns: [\n { key: 'created', label: 'Date Joined', formatter: 'date', sortable: true },\n { key: 'group.name', label: 'Group Name', sortable: true },\n { key: 'permissions|keys|badge', label: 'Permissions' }\n ]\n });\n\n // Events\n const eventsCollection = new IncidentEventList({\n params: { size: 5, model_name: \"account.User\", model_id: this.model.get('id') }\n });\n const eventsView = new TableView({\n collection: eventsCollection,\n hideActivePillNames: ['model_name', 'model_id'],\n columns: [\n { key: 'id', label: 'ID', sortable: true, width: '40px' },\n { key: 'created', label: 'Date', formatter: 'datetime', sortable: true, width: '150px' },\n { key: 'category|badge', label: 'Category' },\n { key: 'title', label: 'Title' }\n ]\n });\n\n // Devices — multiline rows with detail dialog\n const devicesView = new TableView({\n collection: new UserDeviceList({ params: { size: 10, user: this.model.get('id') } }),\n hideActivePillNames: ['user'],\n clickAction: 'view',\n itemClass: DeviceRow,\n columns: [\n {\n key: 'device_info',\n label: 'Device',\n template: `\n <div style=\"font-size:0.85rem; font-weight:500;\">\n <i class=\"bi {{deviceIcon}} text-muted me-1\" style=\"font-size:1.1rem; vertical-align:middle;\"></i>{{deviceName}}\n {{#deviceModel}} <span class=\"text-muted fw-normal\">({{deviceModel}})</span>{{/deviceModel}}\n </div>\n <div style=\"font-size:0.73rem; color:#6c757d; margin-top:0.15rem;\">\n {{deviceMeta}}\n {{#model.last_ip}} <span class=\"text-muted mx-1\">&middot;</span> {{model.last_ip}}{{/model.last_ip}}\n </div>`\n },\n { key: 'first_seen', label: 'First Seen', formatter: 'epoch|relative', width: '120px' },\n { key: 'last_seen', label: 'Last Seen', formatter: 'epoch|relative', width: '120px' }\n ]\n });\n\n // Locations — multiline rows with detail dialog\n const locationsView = new TableView({\n collection: new UserDeviceLocationList({ params: { size: 10, user: this.model.get('id') } }),\n hideActivePillNames: ['user'],\n clickAction: 'view',\n itemClass: LocationRow,\n columns: [\n {\n key: 'user_device',\n label: 'Session',\n template: `\n <div style=\"font-size:0.85rem; font-weight:500;\">\n <i class=\"bi {{deviceIcon}} text-muted me-1\" style=\"font-size:1.1rem; vertical-align:middle;\"></i>{{browserName}} <span class=\"text-muted fw-normal\">on</span> {{deviceName}}\n </div>\n <div style=\"font-size:0.73rem; color:#6c757d; margin-top:0.15rem;\">\n <i class=\"bi bi-geo-alt me-1\"></i>{{locationText}}\n {{#countryName}} <span class=\"text-muted mx-1\">&middot;</span> {{countryName}}{{/countryName}}\n {{#model.ip_address}} <span class=\"text-muted mx-1\">&middot;</span> {{model.ip_address}}{{/model.ip_address}}\n {{#hasThreatFlags|bool}} <span class=\"ms-1\">{{{threatFlags}}}</span>{{/hasThreatFlags|bool}}\n </div>`\n },\n { key: 'last_seen', label: 'Last Seen', formatter: 'epoch|relative', width: '120px' }\n ]\n });\n\n // Push Devices\n const pushDevices = new PushDeviceList({\n params: { size: 5, user: this.model.get('id') }\n });\n const pushDevicesView = new TableView({\n collection: pushDevices,\n hideActivePillNames: ['user'],\n columns: [\n { key: 'duid|truncate_middle(16)', label: 'Device ID', sortable: true },\n { key: 'device_info.user_agent.family', label: 'Browser', formatter: \"default('—')\" },\n { key: 'device_info.os.family', label: 'OS', formatter: \"default('—')\" },\n { key: 'first_seen', label: 'First Seen', formatter: \"epoch|datetime\" },\n { key: 'last_seen', label: 'Last Seen', formatter: \"epoch|datetime\" }\n ],\n size: 5\n });\n\n // Logs (model-scoped)\n const logsCollection = new LogList({\n params: { size: 5, model_name: \"account.User\", model_id: this.model.get('id') }\n });\n const logsView = new TableView({\n collection: logsCollection,\n permissions: 'view_logs',\n hideActivePillNames: ['model_name', 'model_id'],\n columns: [\n {\n key: 'created', label: 'Timestamp', sortable: true, formatter: \"epoch|datetime\",\n filter: { name: \"created\", type: 'daterange', startName: 'dr_start', endName: 'dr_end', fieldName: 'dr_field', label: 'Date Range', format: 'YYYY-MM-DD', displayFormat: 'MMM DD, YYYY', separator: ' to ' }\n },\n {\n key: 'level', label: 'Level', sortable: true,\n filter: { type: 'select', options: [{ value: 'info', label: 'Info' }, { value: 'warning', label: 'Warning' }, { value: 'error', label: 'Error' }] }\n },\n { key: 'kind', label: 'Kind', filter: { type: 'text' } },\n { name: 'log', label: 'Log' }\n ]\n });\n\n // Activity (user-scoped logs)\n const activityCollection = new LogList({\n params: { size: 5, uid: this.model.get('id') }\n });\n const activityView = new TableView({\n collection: activityCollection,\n hideActivePillNames: ['uid'],\n permissions: 'view_logs',\n columns: [\n {\n key: 'created', label: 'Timestamp', sortable: true, formatter: \"epoch|datetime\",\n filter: { name: \"created\", type: 'daterange', startName: 'dr_start', endName: 'dr_end', fieldName: 'dr_field', label: 'Date Range', format: 'YYYY-MM-DD', displayFormat: 'MMM DD, YYYY', separator: ' to ' }\n },\n {\n key: 'level', label: 'Level', sortable: true,\n filter: { type: 'select', options: [{ value: 'info', label: 'Info' }, { value: 'warning', label: 'Warning' }, { value: 'error', label: 'Error' }] }\n },\n { key: 'kind', label: 'Kind', filter: { type: 'text' } },\n { name: 'path', label: 'Path' }\n ]\n });\n\n // ── SideNavView ─────────────────────────────\n this.sideNavView = new SideNavView({\n containerId: 'user-sidenav',\n activeSection: 'profile',\n navWidth: 200,\n contentPadding: '1.25rem 2rem',\n enableResponsive: true,\n minWidth: 500,\n sections: [\n { key: 'profile', label: 'Profile', icon: 'bi-person', view: profileView },\n { key: 'personal', label: 'Personal', icon: 'bi-person-vcard', view: personalView },\n { key: 'security', label: 'Security', icon: 'bi-shield-lock', view: securityView },\n { key: 'connected', label: 'Connected', icon: 'bi-plug', view: connectedView },\n { type: 'divider', label: 'Access' },\n { key: 'permissions', label: 'Permissions', icon: 'bi-shield-check', view: permsView },\n { key: 'groups', label: 'Groups', icon: 'bi-people', view: groupsView },\n { key: 'api_keys', label: 'API Keys', icon: 'bi-key', view: apiKeysView },\n { type: 'divider', label: 'Activity' },\n { key: 'events', label: 'Events', icon: 'bi-calendar-event', view: eventsView },\n { key: 'activity', label: 'Activity Log', icon: 'bi-clock-history', view: activityView, permissions: 'view_logs' },\n { key: 'logs', label: 'Object Logs', icon: 'bi-journal-text', view: logsView, permissions: 'view_logs' },\n { type: 'divider', label: 'Devices' },\n { key: 'devices', label: 'Devices', icon: 'bi-laptop', view: devicesView },\n { key: 'locations', label: 'Locations', icon: 'bi-geo-alt', view: locationsView },\n { key: 'push_devices', label: 'Push Devices', icon: 'bi-phone', view: pushDevicesView },\n { type: 'divider', label: 'Settings' },\n { key: 'notifications', label: 'Notifications', icon: 'bi-bell', view: notificationsView }\n ]\n });\n this.addChild(this.sideNavView);\n\n // ── Context Menu ────────────────────────────\n const userMenu = new ContextMenu({\n containerId: 'user-context-menu',\n className: \"context-menu-view header-menu-absolute\",\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: [\n { label: 'Edit User', action: 'edit-user', icon: 'bi-pencil' },\n ...(this.model.get('avatar')\n ? [{ label: 'Clear Avatar', action: 'clear-avatar', icon: 'bi-person-x' }]\n : []),\n { type: 'divider' },\n { label: 'Send Password Reset', action: 'send-password-reset', icon: 'bi-envelope' },\n { label: 'Send Magic Login Link', action: 'send-magic-link', icon: 'bi-link-45deg' },\n { label: 'Revoke All Sessions', action: 'revoke-all-sessions', icon: 'bi-box-arrow-right' },\n { type: 'divider' },\n ...(this.model.get('is_email_verified')\n ? []\n : [\n { label: 'Send Email Verification', action: 'send-email-verification', icon: 'bi-envelope-check' },\n { label: 'Force Verify Email', action: 'force-verify-email', icon: 'bi-patch-check' },\n ]),\n ...(this.model.get('phone_number') && !this.model.get('is_phone_verified')\n ? [{ label: 'Force Verify Phone', action: 'force-verify-phone', icon: 'bi-patch-check' }]\n : []),\n { type: 'divider' },\n this.model.get('is_active')\n ? { label: 'Deactivate User', action: 'deactivate-user', icon: 'bi-person-dash' }\n : { label: 'Activate User', action: 'activate-user', icon: 'bi-person-check' },\n ]\n }\n });\n this.addChild(userMenu);\n }\n\n // ── Context menu actions ────────────────────────\n\n async onActionEditUser() {\n await Dialog.showModelForm({\n title: `EDIT - #${this.model.id} ${this.options.modelName}`,\n model: this.model,\n formConfig: UserForms.edit,\n });\n }\n\n async onActionClearAvatar() {\n const confirmed = await Dialog.confirm(\n 'Remove this user\\'s avatar? They will see the default placeholder.',\n 'Clear Avatar'\n );\n if (!confirmed) return true;\n\n const resp = await this.model.save({ avatar: null });\n if (resp.status === 200) {\n this.getApp().toast.success('Avatar cleared');\n } else {\n this.getApp().toast.error('Failed to clear avatar');\n }\n return true;\n }\n\n async onActionSendPasswordReset() {\n const email = this.model.get('email');\n const confirmed = await Dialog.confirm(\n `Send a password reset email to <strong>${email}</strong>?`,\n 'Send Password Reset'\n );\n if (!confirmed) return true;\n\n const resp = await rest.POST('/api/auth/password/reset', { email });\n if (resp.success) {\n this.getApp().toast.success('Password reset email sent');\n } else {\n this.getApp().toast.error(resp.message || 'Failed to send password reset');\n }\n return true;\n }\n\n async onActionSendMagicLink() {\n const email = this.model.get('email');\n const confirmed = await Dialog.confirm(\n `Send a magic login link to <strong>${email}</strong>?`,\n 'Send Magic Login Link'\n );\n if (!confirmed) return true;\n\n const resp = await rest.POST('/api/auth/magic-link', { email });\n if (resp.success) {\n this.getApp().toast.success('Magic login link sent');\n } else {\n this.getApp().toast.error(resp.message || 'Failed to send magic link');\n }\n return true;\n }\n\n async onActionRevokeAllSessions() {\n const confirmed = await Dialog.confirm(\n 'Revoke all sessions? The user will be signed out of all devices immediately.',\n 'Revoke All Sessions'\n );\n if (!confirmed) return true;\n\n const resp = await rest.POST(`/api/user/${this.model.id}/sessions/revoke`);\n if (resp.success) {\n this.getApp().toast.success('All sessions revoked');\n } else {\n this.getApp().toast.error(resp.message || 'Failed to revoke sessions');\n }\n return true;\n }\n\n async onActionSendEmailVerification() {\n const email = this.model.get('email');\n const confirmed = await Dialog.confirm(\n `Send a verification email to <strong>${email}</strong>?`,\n 'Send Email Verification'\n );\n if (!confirmed) return true;\n\n const resp = await rest.POST('/api/auth/email/verify', { email });\n if (resp.success) {\n this.getApp().toast.success('Verification email sent');\n } else {\n this.getApp().toast.error(resp.message || 'Failed to send verification email');\n }\n return true;\n }\n\n async onActionForceVerifyEmail() {\n const confirmed = await Dialog.confirm(\n `Mark <strong>${this.model.get('email')}</strong> as verified?`,\n 'Force Verify Email'\n );\n if (!confirmed) return true;\n\n const resp = await this.model.save({ is_email_verified: true });\n if (resp.status === 200) {\n this.getApp().toast.success('Email marked as verified');\n } else {\n this.getApp().toast.error('Failed to verify email');\n }\n return true;\n }\n\n async onActionForceVerifyPhone() {\n const confirmed = await Dialog.confirm(\n `Mark <strong>${this.model.get('phone_number')}</strong> as verified?`,\n 'Force Verify Phone'\n );\n if (!confirmed) return true;\n\n const resp = await this.model.save({ is_phone_verified: true });\n if (resp.status === 200) {\n this.getApp().toast.success('Phone marked as verified');\n } else {\n this.getApp().toast.error('Failed to verify phone');\n }\n return true;\n }\n\n async onActionToggleActive() {\n if (this.model.get('is_active')) {\n return this.onActionDeactivateUser();\n } else {\n return this.onActionActivateUser();\n }\n }\n\n async onActionDeactivateUser() {\n const confirmed = await Dialog.confirm(\"Are you sure you want to deactivate this user?\");\n if (!confirmed) return true;\n\n const resp = await this.model.save({ is_active: false });\n if (resp.status === 200) {\n this.getApp().toast.success(\"User deactivated\");\n } else {\n this.getApp().toast.error(\"Failed to deactivate user\");\n }\n return true;\n }\n\n async onActionActivateUser() {\n const confirmed = await Dialog.confirm(\"Are you sure you want to activate this user?\");\n if (!confirmed) return true;\n\n const resp = await this.model.save({ is_active: true });\n if (resp.status === 200) {\n this.getApp().toast.success(\"User activated\");\n } else {\n this.getApp().toast.error(\"Failed to activate user\");\n }\n return true;\n }\n\n async showSection(sectionName) {\n if (this.sideNavView) {\n await this.sideNavView.showSection(sectionName);\n }\n }\n\n getActiveSection() {\n return this.sideNavView ? this.sideNavView.getActiveSection() : null;\n }\n\n // Legacy API compatibility\n async showTab(tabName) {\n return this.showSection(tabName);\n }\n\n getActiveTab() {\n return this.getActiveSection();\n }\n\n _onModelChange() {\n // do nothing, we do not want model changes to render this entire view\n }\n\n // Static factory method\n static create(options = {}) {\n return new UserView(options);\n }\n}\n\nUser.VIEW_CLASS = UserView;\n\nexport default UserView;\n","/**\n * UsersPage - Simple Todo List using TablePage component\n * Demonstrates clean usage of TablePage framework features\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport {UserList, UserForms} from '@core/models/User.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\nimport MOJOUtils from '@core/utils/MOJOUtils.js';\nimport UserView from './UserView.js';\n\n\nclass UserTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_users',\n pageName: 'Manage Users',\n router: \"admin/users\",\n Collection: UserList,\n\n viewDialogOptions: { header: false },\n\n defaultQuery: {\n sort: '-last_activity',\n is_active: true\n },\n\n // Column definitions\n columns: [\n {\n key: 'id',\n label: 'ID',\n sortable: true,\n class: 'text-muted'\n },\n // {\n // label: 'Avatar',\n // key: 'avatar|avatar(\"sm\")',\n // sortable: false,\n // visibility: 'md'\n // },\n {\n key: 'display_name|tooltip:model.username',\n label: 'Display Name',\n },\n {\n label: 'Info',\n key: 'permissions.manage_users',\n template: `\n {{^model.is_active}}<span class=\"text-danger\">DISABLED</span> {{/model.is_active}}\n {{#model.permissions.manage_users}}{{{model.permissions.manage_users|yesnoicon('bi bi-person-gear text-danger')|tooltip('Manage Users')}}} {{/model.permissions.manage_users}}\n {{#model.permissions.manage_groups}}{{{model.permissions.manage_groups|yesnoicon('bi bi-building-gear text-primary')|tooltip('Manage Groups')}}} {{/model.permissions.manage_groups}}\n {{#model.permissions.view_global}}{{{model.permissions.view_global|yesnoicon('bi bi-globe text-secondary')|tooltip('View Global Menu')}}} {{/model.permissions.view_global}}\n {{#model.permissions.view_admin}}{{{model.permissions.view_admin|yesnoicon('bi bi-wrench text-secondary')|tooltip('View Admin Menu')}}} {{/model.permissions.view_admin}}\n `,\n sortable: false,\n },\n {\n key: 'email',\n label: 'Email',\n visibility: 'xl',\n className: 'text-muted fs-8',\n },\n // {\n // key: 'username',\n // label: 'Username',\n // visibility: 'xl',\n // className: 'text-muted fs-8',\n // },\n {\n key: 'last_activity',\n label: 'Last Activity',\n formatter: \"relative\",\n className: 'text-muted fs-8',\n }\n ],\n\n filters: [\n {\n key: 'is_active',\n label: 'Active',\n type: 'boolean',\n defaultValue: true,\n },\n {\n key: 'email',\n label: 'Email',\n type: 'text',\n defaultValue: '',\n },\n {\n key: 'username',\n label: 'Username',\n type: 'text',\n defaultValue: '',\n },\n {\n key: 'locations__ip_address',\n label: 'IP Address',\n type: 'text',\n defaultValue: '',\n },\n {\n key: 'last_activity',\n type: 'daterange',\n startName: 'dr_start',\n endName: 'dr_end',\n fieldName: 'dr_field',\n label: 'Date Range',\n format: 'YYYY-MM-DD',\n displayFormat: 'MMM DD, YYYY',\n separator: ' to '\n }\n ],\n\n // Table features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // TablePage toolbar\n showRefresh: true,\n showAdd: true,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No users found. Click \"Add\" to create a new user.',\n\n // Context menu configuration\n contextMenu: [\n {\n icon: 'bi-pencil',\n action: 'edit',\n label: \"Edit Profile\"\n },\n {\n icon: 'bi-shield-check',\n action: 'edit-permissions',\n label: \"Edit Permissions\"\n },\n {\n icon: 'bi-shield',\n action: 'change-password',\n label: \"Change Password\",\n },\n { separator: true },\n {\n icon: 'bi-envelope',\n action: 'send-invite',\n label: \"Send Invite\"\n }\n ],\n\n // Table display options (for HTML table styling)\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n }\n });\n }\n\n async onActionEditPermissions(event, element) {\n\n event.preventDefault();\n const item = this.collection.get(element.dataset.id);\n const result = await Dialog.showModelForm({\n model: item,\n size: 'lg',\n title: `Edit Permissions for \"${item._.username}\"`,\n fields: UserForms.permissions.fields\n });\n }\n\n async onActionChangePassword(event, element) {\n // Implement password change logic here\n const item = this.collection.get(element.dataset.id);\n const data = await Dialog.showForm({\n title: `Change Password for \"${item._.username}\"`,\n fields: [\n {\n type: 'text', // Change from 'hidden' to 'text'\n name: 'username',\n value: item.get('email') || item.get('username'),\n attributes: {\n autocomplete: 'username',\n readonly: 'readonly',\n tabindex: '-1',\n style: 'position: absolute; left: -9999px; opacity: 0; height: 0; width: 0;'\n }\n },\n {\n name: 'new_password',\n label: 'New Password',\n type: 'password',\n passwordUsage: 'new',\n required: true,\n showToggle: true,\n attributes: {\n autocomplete: 'new-password' // Make sure this isn't being overridden\n }\n }\n ]\n });\n\n if (data && data.new_password) {\n // Basic password validation\n const result = MOJOUtils.checkPasswordStrength(data.new_password);\n if (result.score < 5) {\n this.getApp().toast.error('Password must be at least 6 characters long and contain at least 2 of the following: uppercase letter, lowercase letter, or number');\n await this.onActionChangePassword(event, element);\n return;\n }\n const resp = await item.save({new_password: data.new_password});\n if (!this.onPasswordChange(resp)) {\n await this.onActionChangePassword(event, element);\n }\n }\n }\n\n onPasswordChange(resp) {\n if (resp.success) {\n this.getApp().toast.success('Password changed successfully');\n return true;\n } else {\n if (resp.data && resp.data.error) {\n this.getApp().toast.error(resp.data.error);\n } else {\n this.getApp().toast.error('Failed to change password');\n }\n }\n return false;\n }\n\n async onActionSendInvite(event, element) {\n const item = this.collection.get(element.dataset.id);\n const resp = await item.save({send_invite: true});\n if (resp.success) {\n this.getApp().toast.success('Invite sent successfully');\n return true;\n } else {\n if (resp.data && resp.data.error) {\n this.getApp().toast.error(resp.data.error);\n } else {\n this.getApp().toast.error('Failed to send invite');\n }\n }\n return false;\n }\n\n}\n\nexport default UserTablePage;\n","/**\n * MemberView - Modern membership detail view\n *\n * Features:\n * - Header with avatar, user name, group name, role badge, active toggle\n * - SideNavView: Details (with clickable user/group), Permissions, Logs\n * - Context menu: Edit, View User, View Group, Remove\n * - Click user → opens UserView, click group → opens GroupView\n */\n\nimport View from '@core/View.js';\nimport SideNavView from '@core/views/navigation/SideNavView.js';\nimport TableView from '@core/views/table/TableView.js';\nimport FormView from '@core/forms/FormView.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport { Member, MemberForms } from '@core/models/Member.js';\nimport { LogList } from '@core/models/Log.js';\nimport Modal from '@core/views/feedback/Modal.js';\n\nclass MemberView extends View {\n constructor(options = {}) {\n super({\n className: 'member-view',\n ...options\n });\n\n this.model = options.model || new Member(options.data || {});\n\n this.template = `\n <div class=\"member-view-container\">\n <!-- Header + Context Menu -->\n <div class=\"d-flex justify-content-between align-items-start mb-4\">\n <div data-container=\"member-header\" style=\"flex: 1;\"></div>\n <div data-container=\"member-context-menu\" class=\"ms-3 flex-shrink-0\"></div>\n </div>\n <!-- Side Nav -->\n <div data-container=\"member-sidenav\" style=\"min-height: 300px;\"></div>\n </div>\n `;\n }\n\n async onInit() {\n // ── Header ──────────────────────────────────\n this.header = new View({\n containerId: 'member-header',\n template: `\n <div class=\"d-flex justify-content-between align-items-start\">\n <!-- Left: Avatar + Identity -->\n <div class=\"d-flex align-items-center gap-3\">\n {{{model.user.avatar|avatar('md','rounded-circle')}}}\n <div>\n <h4 class=\"mb-0\">\n <a href=\"#\" data-action=\"view-user\" class=\"text-decoration-none text-body\">{{model.user.display_name}}</a>\n </h4>\n <div class=\"text-muted small mt-1\">\n <i class=\"bi bi-people me-1\"></i>\n <a href=\"#\" data-action=\"view-group\" class=\"text-decoration-none\">{{model.group.name}}</a>\n {{#model.group.kind}}\n <span class=\"badge bg-light text-muted border ms-1\" style=\"font-size: 0.65rem;\">{{model.group.kind|capitalize}}</span>\n {{/model.group.kind}}\n </div>\n </div>\n </div>\n\n <!-- Right: Status -->\n <div class=\"text-end\">\n <div class=\"d-flex align-items-center gap-2\">\n {{#model.metadata.role}}\n <span class=\"badge bg-primary bg-opacity-10 text-primary\" style=\"font-size: 0.72rem;\">{{model.metadata.role}}</span>\n {{/model.metadata.role}}\n <span class=\"d-inline-flex align-items-center gap-1\" style=\"cursor: pointer;\"\n data-action=\"toggle-active\"\n title=\"{{model.is_active|boolean('Click to deactivate','Click to activate')}}\">\n <i class=\"bi {{model.is_active|boolean('bi-toggle-on text-success','bi-toggle-off text-secondary')}}\" style=\"font-size: 1.1rem;\"></i>\n <span class=\"small\">{{model.is_active|boolean('Active','Inactive')}}</span>\n </span>\n </div>\n {{#model.created}}\n <div class=\"text-muted small mt-1\">Joined {{model.created|date}}</div>\n {{/model.created}}\n </div>\n </div>`\n });\n this.header.setModel(this.model);\n this.addChild(this.header);\n\n // ── Details section ─────────────────────────\n const detailsView = new View({\n model: this.model,\n template: `\n <style>\n .mv-section-label { font-size: 0.7rem; font-weight: 700; text-transform: uppercase; letter-spacing: 0.06em; color: #adb5bd; margin-bottom: 0.5rem; margin-top: 1.5rem; }\n .mv-section-label:first-child { margin-top: 0; }\n .mv-field-row { display: flex; align-items: baseline; padding: 0.5rem 0; border-bottom: 1px solid #f0f0f0; }\n .mv-field-row:last-child { border-bottom: none; }\n .mv-field-label { width: 130px; font-size: 0.78rem; color: #6c757d; flex-shrink: 0; }\n .mv-field-value { flex: 1; font-size: 0.88rem; color: #212529; }\n .mv-field-action { color: #6c757d; cursor: pointer; font-size: 0.8rem; margin-left: auto; padding: 0.15rem 0.4rem; border-radius: 4px; background: none; border: none; }\n .mv-field-action:hover { background: #f0f0f0; color: #0d6efd; }\n </style>\n\n <div class=\"mv-section-label\">User</div>\n <div class=\"mv-field-row\">\n <div class=\"mv-field-label\">Name</div>\n <div class=\"mv-field-value\">\n <a href=\"#\" data-action=\"view-user\" class=\"text-decoration-none\">{{model.user.display_name}}</a>\n </div>\n </div>\n <div class=\"mv-field-row\">\n <div class=\"mv-field-label\">Email</div>\n <div class=\"mv-field-value\">{{model.user.email}}</div>\n </div>\n\n <div class=\"mv-section-label\">Group</div>\n <div class=\"mv-field-row\">\n <div class=\"mv-field-label\">Name</div>\n <div class=\"mv-field-value\">\n <a href=\"#\" data-action=\"view-group\" class=\"text-decoration-none\">{{model.group.name}}</a>\n </div>\n </div>\n {{#model.group.kind}}\n <div class=\"mv-field-row\">\n <div class=\"mv-field-label\">Kind</div>\n <div class=\"mv-field-value\"><span class=\"badge bg-primary bg-opacity-10 text-primary\">{{model.group.kind|capitalize}}</span></div>\n </div>\n {{/model.group.kind}}\n\n <div class=\"mv-section-label\">Membership</div>\n <div class=\"mv-field-row\">\n <div class=\"mv-field-label\">Role</div>\n <div class=\"mv-field-value\">{{model.metadata.role|default('—')}}</div>\n <button type=\"button\" class=\"mv-field-action\" data-action=\"edit-membership\" title=\"Edit\"><i class=\"bi bi-pencil\"></i></button>\n </div>\n <div class=\"mv-field-row\">\n <div class=\"mv-field-label\">Status</div>\n <div class=\"mv-field-value\">\n {{#model.is_active|bool}}<span style=\"font-size:0.65rem; padding:0.15em 0.45em; background:#d1e7dd; color:#0f5132; border-radius:3px;\">Active</span>{{/model.is_active|bool}}\n {{^model.is_active|bool}}<span style=\"font-size:0.65rem; padding:0.15em 0.45em; background:#fff3cd; color:#856404; border-radius:3px;\">Inactive</span>{{/model.is_active|bool}}\n </div>\n </div>\n <div class=\"mv-field-row\">\n <div class=\"mv-field-label\">Member ID</div>\n <div class=\"mv-field-value\" style=\"font-family: ui-monospace, monospace; font-size: 0.82rem;\">{{model.id}}</div>\n </div>\n <div class=\"mv-field-row\">\n <div class=\"mv-field-label\">Joined</div>\n <div class=\"mv-field-value\">{{model.created|datetime|default('—')}}</div>\n </div>\n `\n });\n\n // ── Permissions section — editable switches ──\n const permissionsView = new FormView({\n fields: Member.PERMISSION_FIELDS,\n model: this.model,\n autosaveModelField: true\n });\n\n // ── Logs section ────────────────────────────\n const logsView = new TableView({\n collection: new LogList({\n params: { size: 10, model_name: 'account.Member', model_id: this.model.get('id') }\n }),\n permissions: 'view_logs',\n hideActivePillNames: ['model_name', 'model_id'],\n columns: [\n {\n key: 'created', label: 'Timestamp', sortable: true, formatter: 'epoch|datetime',\n filter: { name: 'created', type: 'daterange', startName: 'dr_start', endName: 'dr_end', fieldName: 'dr_field', label: 'Date Range', format: 'YYYY-MM-DD', displayFormat: 'MMM DD, YYYY', separator: ' to ' }\n },\n { key: 'level', label: 'Level', sortable: true },\n { key: 'kind', label: 'Kind' },\n { name: 'log', label: 'Log' }\n ]\n });\n\n // ── SideNavView ─────────────────────────────\n this.sideNavView = new SideNavView({\n containerId: 'member-sidenav',\n activeSection: 'details',\n navWidth: 160,\n contentPadding: '1rem 1.5rem',\n enableResponsive: true,\n minWidth: 450,\n sections: [\n { key: 'details', label: 'Details', icon: 'bi-info-circle', view: detailsView },\n { key: 'permissions', label: 'Permissions', icon: 'bi-shield-check', view: permissionsView },\n { type: 'divider', label: 'Activity' },\n { key: 'logs', label: 'Logs', icon: 'bi-journal-text', view: logsView, permissions: 'view_logs' }\n ]\n });\n this.addChild(this.sideNavView);\n\n // ── Context Menu ────────────────────────────\n const memberMenu = new ContextMenu({\n containerId: 'member-context-menu',\n className: 'context-menu-view header-menu-absolute',\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: [\n { label: 'Edit Membership', action: 'edit-membership', icon: 'bi-pencil' },\n { type: 'divider' },\n { label: 'View User', action: 'view-user', icon: 'bi-person' },\n { label: 'View Group', action: 'view-group', icon: 'bi-people' },\n { type: 'divider' },\n this.model.get('is_active')\n ? { label: 'Deactivate Member', action: 'deactivate-member', icon: 'bi-toggle-off' }\n : { label: 'Activate Member', action: 'activate-member', icon: 'bi-toggle-on' },\n { label: 'Remove From Group', action: 'remove-member', icon: 'bi-person-dash', danger: true }\n ]\n }\n });\n this.addChild(memberMenu);\n }\n\n // ── Actions ─────────────────────────────────\n\n async onActionEditMembership() {\n await Modal.modelForm({\n title: 'Edit Membership',\n model: this.model,\n formConfig: MemberForms.edit,\n });\n }\n\n async onActionViewUser() {\n const userId = this.model.get('user')?.id;\n if (!userId) return true;\n const { User } = await import('@core/models/User.js');\n await Modal.showModelById(User, userId);\n return true;\n }\n\n async onActionViewGroup() {\n const groupId = this.model.get('group')?.id;\n if (!groupId) return true;\n\n const { Group } = await import('@core/models/Group.js');\n await Modal.showModelById(Group, groupId);\n return true;\n }\n\n async onActionToggleActive() {\n if (this.model.get('is_active')) {\n return this.onActionDeactivateMember();\n } else {\n return this.onActionActivateMember();\n }\n }\n\n async onActionDeactivateMember() {\n const confirmed = await Modal.confirm(\n `Deactivate <strong>${this.model.get('user.display_name')}</strong>'s membership in <strong>${this.model.get('group.name')}</strong>?`,\n 'Deactivate Member'\n );\n if (!confirmed) return true;\n\n const resp = await this.model.save({ is_active: false });\n if (resp.status === 200) {\n this.getApp()?.toast?.success('Member deactivated');\n } else {\n this.getApp()?.toast?.error('Failed to deactivate member');\n }\n return true;\n }\n\n async onActionActivateMember() {\n const confirmed = await Modal.confirm(\n `Activate <strong>${this.model.get('user.display_name')}</strong>'s membership in <strong>${this.model.get('group.name')}</strong>?`,\n 'Activate Member'\n );\n if (!confirmed) return true;\n\n const resp = await this.model.save({ is_active: true });\n if (resp.status === 200) {\n this.getApp()?.toast?.success('Member activated');\n } else {\n this.getApp()?.toast?.error('Failed to activate member');\n }\n return true;\n }\n\n async onActionRemoveMember() {\n const confirmed = await Modal.confirm(\n `Remove <strong>${this.model.get('user.display_name')}</strong> from <strong>${this.model.get('group.name')}</strong>? This cannot be undone.`,\n 'Remove Member'\n );\n if (!confirmed) return true;\n\n const resp = await this.model.destroy();\n if (resp.success) {\n this.getApp()?.toast?.success('Member removed');\n this.emit('member:removed', { model: this.model });\n } else {\n this.getApp()?.toast?.error('Failed to remove member');\n }\n return true;\n }\n\n _onModelChange() {\n // Prevent full re-render on model changes\n }\n\n static create(options = {}) {\n return new MemberView(options);\n }\n}\n\nMember.VIEW_CLASS = MemberView;\n\nexport default MemberView;\n","/**\n * MemberTablePage - Member management using TablePage component\n * Clean implementation using TablePage with minimal overrides\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { MemberList, MemberForms } from '@core/models/Member.js';\nimport MemberView from './MemberView.js';\n\nclass MemberTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_members',\n pageName: 'Manage Members',\n router: \"admin/members\",\n Collection: MemberList,\n \n formEdit: MemberForms.edit,\n itemViewClass: MemberView,\n \n viewDialogOptions: {\n header: false,\n size: 'lg'\n },\n \n // Column definitions\n columns: [\n {\n key: 'id',\n label: 'ID',\n width: '60px',\n sortable: true,\n class: 'text-muted'\n },\n {\n key: 'user.display_name',\n label: 'User',\n formatter: \"default('Unknown User')\"\n },\n {\n key: 'user.email',\n label: 'Email',\n formatter: \"default('No Email')\"\n },\n {\n key: 'group.name',\n label: 'Group',\n formatter: \"default('Unknown Group')\"\n },\n {\n key: 'role',\n label: 'Role',\n formatter: \"badge\"\n },\n {\n key: 'status',\n label: 'Status',\n formatter: \"badge\"\n },\n {\n key: 'created',\n label: 'Added',\n formatter: \"epoch|datetime\"\n }\n ],\n\n // Table features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Toolbar\n showRefresh: true,\n showAdd: true,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No members found. Click \"Add Member\" to add users to groups.',\n\n // Batch actions\n batchBarLocation: 'top',\n batchActions: [\n { label: \"Remove\", icon: \"bi bi-person-dash\", action: \"batch-remove\" },\n { label: \"Export\", icon: \"bi bi-download\", action: \"batch-export\" },\n { label: \"Change Role\", icon: \"bi bi-person-gear\", action: \"batch-role\" },\n { label: \"Activate\", icon: \"bi bi-check-circle\", action: \"batch-activate\" },\n { label: \"Deactivate\", icon: \"bi bi-x-circle\", action: \"batch-deactivate\" }\n ],\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n }\n });\n }\n}\n\nexport default MemberTablePage;","/**\n * GroupView - Modern group management interface\n *\n * Features:\n * - Clean header with avatar, name, kind badge, parent link, active/online status\n * - SideNavView with: Details, Members, Children, Events, Logs\n * - Expanded context menu with quick actions\n * - Clean Bootstrap 5 styling matching UserView patterns\n */\n\nimport View from '@core/View.js';\nimport SideNavView from '@core/views/navigation/SideNavView.js';\nimport TableView from '@core/views/table/TableView.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport { Group, GroupList, GroupForms } from '@core/models/Group.js';\nimport { MemberList } from '@core/models/Member.js';\nimport { IncidentEventList } from '@core/models/Incident.js';\nimport { LogList } from '@core/models/Log.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\n\nclass GroupView extends View {\n constructor(options = {}) {\n super({\n className: 'group-view',\n ...options\n });\n\n this.model = options.model || new Group(options.data || {});\n\n this.template = `\n <div class=\"group-view-container\">\n <!-- Header -->\n <div data-container=\"group-header\"></div>\n <!-- Side Nav -->\n <div data-container=\"group-sidenav\" style=\"min-height: 400px;\"></div>\n </div>\n `;\n }\n\n async onInit() {\n // ── Header ──────────────────────────────────\n this.header = new View({\n containerId: 'group-header',\n template: `\n <div class=\"d-flex justify-content-between align-items-start mb-4\">\n <!-- Left Side: Identity -->\n <div class=\"d-flex align-items-center gap-3\">\n {{#model.avatar}}\n {{{model.avatar|avatar('md','rounded')}}}\n {{/model.avatar}}\n {{^model.avatar}}\n <div class=\"d-flex align-items-center justify-content-center rounded bg-light\" style=\"width: 56px; height: 56px;\">\n <i class=\"bi bi-people text-secondary\" style=\"font-size: 1.5rem;\"></i>\n </div>\n {{/model.avatar}}\n <div>\n <h3 class=\"mb-0\">{{model.name|default('Unnamed Group')}}</h3>\n <div class=\"d-flex align-items-center gap-2 mt-1\">\n <span class=\"badge bg-primary bg-opacity-10 text-primary\" style=\"font-size: 0.72rem;\">{{model.kind|capitalize}}</span>\n {{#model.parent}}\n <span class=\"text-muted small\">\n <i class=\"bi bi-diagram-3 me-1\"></i>\n <a href=\"#\" data-action=\"view-parent\" data-id=\"{{model.parent.id}}\" class=\"text-decoration-none\">{{model.parent.name}}</a>\n </span>\n {{/model.parent}}\n </div>\n {{#model.metadata.timezone}}\n <div class=\"text-muted small mt-1\"><i class=\"bi bi-clock me-1\"></i>{{model.metadata.timezone}}</div>\n {{/model.metadata.timezone}}\n </div>\n </div>\n\n <!-- Right Side: Status & Actions -->\n <div class=\"d-flex align-items-start gap-4\">\n <div class=\"text-end\">\n <div class=\"d-flex align-items-center justify-content-end gap-3\">\n <span class=\"d-inline-flex align-items-center gap-1\" title=\"{{model.is_active|boolean('Group Active','Group Inactive')}}\">\n <i class=\"bi {{model.is_active|boolean('bi-toggle-on text-success','bi-toggle-off text-secondary')}}\" style=\"font-size: 1.1rem;\"></i>\n <span class=\"small\">{{model.is_active|boolean('Active','Inactive')}}</span>\n </span>\n </div>\n {{#model.last_activity}}\n <div class=\"text-muted small mt-1\">Last active {{model.last_activity|relative}}</div>\n {{/model.last_activity}}\n {{#model.created}}\n <div class=\"text-muted small mt-1\">Created {{model.created|date}}</div>\n {{/model.created}}\n </div>\n <div data-container=\"group-context-menu\"></div>\n </div>\n </div>`\n });\n this.header.setModel(this.model);\n this.addChild(this.header);\n\n // ── Details section ─────────────────────────\n const detailsView = new View({\n model: this.model,\n template: `\n <style>\n .gv-section-label { font-size: 0.7rem; font-weight: 700; text-transform: uppercase; letter-spacing: 0.06em; color: #adb5bd; margin-bottom: 0.5rem; margin-top: 1.5rem; }\n .gv-section-label:first-child { margin-top: 0; }\n .gv-field-row { display: flex; align-items: baseline; padding: 0.5rem 0; border-bottom: 1px solid #f0f0f0; }\n .gv-field-row:last-child { border-bottom: none; }\n .gv-field-label { width: 140px; font-size: 0.78rem; color: #6c757d; flex-shrink: 0; }\n .gv-field-value { flex: 1; font-size: 0.88rem; color: #212529; }\n .gv-field-action { color: #6c757d; cursor: pointer; font-size: 0.8rem; margin-left: auto; padding: 0.15rem 0.4rem; border-radius: 4px; background: none; border: none; }\n .gv-field-action:hover { background: #f0f0f0; color: #0d6efd; }\n </style>\n\n <div class=\"gv-section-label\">Group</div>\n <div class=\"gv-field-row\">\n <div class=\"gv-field-label\">Name</div>\n <div class=\"gv-field-value\">{{model.name}}</div>\n <button type=\"button\" class=\"gv-field-action\" data-action=\"edit-group\" title=\"Edit\"><i class=\"bi bi-pencil\"></i></button>\n </div>\n <div class=\"gv-field-row\">\n <div class=\"gv-field-label\">Kind</div>\n <div class=\"gv-field-value\"><span class=\"badge bg-primary bg-opacity-10 text-primary\">{{model.kind|capitalize}}</span></div>\n </div>\n <div class=\"gv-field-row\">\n <div class=\"gv-field-label\">Status</div>\n <div class=\"gv-field-value\">\n {{#model.is_active|bool}}<span style=\"font-size:0.65rem; padding:0.15em 0.45em; background:#d1e7dd; color:#0f5132; border-radius:3px;\">Active</span>{{/model.is_active|bool}}\n {{^model.is_active|bool}}<span style=\"font-size:0.65rem; padding:0.15em 0.45em; background:#fff3cd; color:#856404; border-radius:3px;\">Inactive</span>{{/model.is_active|bool}}\n </div>\n </div>\n <div class=\"gv-field-row\">\n <div class=\"gv-field-label\">ID</div>\n <div class=\"gv-field-value\" style=\"font-family: ui-monospace, monospace; font-size: 0.82rem;\">{{model.id}}</div>\n </div>\n\n <div class=\"gv-section-label\">Hierarchy</div>\n <div class=\"gv-field-row\">\n <div class=\"gv-field-label\">Parent</div>\n <div class=\"gv-field-value\">\n {{#model.parent}}\n <a href=\"#\" data-action=\"view-parent\" data-id=\"{{model.parent.id}}\" class=\"text-decoration-none\">{{model.parent.name}}</a>\n <span class=\"text-muted small ms-1\">({{model.parent.kind|capitalize}})</span>\n {{/model.parent}}\n {{^model.parent}}<span style=\"color:#adb5bd; font-style:italic; font-size:0.85rem;\">None — top-level group</span>{{/model.parent}}\n </div>\n </div>\n\n <div class=\"gv-section-label\">Settings</div>\n {{#model.metadata.timezone}}\n <div class=\"gv-field-row\">\n <div class=\"gv-field-label\">Timezone</div>\n <div class=\"gv-field-value\">{{model.metadata.timezone}}</div>\n </div>\n {{/model.metadata.timezone}}\n {{#model.metadata.domain}}\n <div class=\"gv-field-row\">\n <div class=\"gv-field-label\">Domain</div>\n <div class=\"gv-field-value\">{{model.metadata.domain}}</div>\n </div>\n {{/model.metadata.domain}}\n {{#model.metadata.portal}}\n <div class=\"gv-field-row\">\n <div class=\"gv-field-label\">Portal URL</div>\n <div class=\"gv-field-value\"><a href=\"{{model.metadata.portal}}\" target=\"_blank\" class=\"text-decoration-none\">{{model.metadata.portal}}</a></div>\n </div>\n {{/model.metadata.portal}}\n {{#model.metadata.eod_hour}}\n <div class=\"gv-field-row\">\n <div class=\"gv-field-label\">End of Day</div>\n <div class=\"gv-field-value\">{{model.metadata.eod_hour}}:00</div>\n </div>\n {{/model.metadata.eod_hour}}\n\n <div class=\"gv-section-label\">Dates</div>\n <div class=\"gv-field-row\">\n <div class=\"gv-field-label\">Created</div>\n <div class=\"gv-field-value\">{{model.created|datetime|default('—')}}</div>\n </div>\n <div class=\"gv-field-row\">\n <div class=\"gv-field-label\">Modified</div>\n <div class=\"gv-field-value\">{{model.modified|datetime|default('—')}}</div>\n </div>\n `\n });\n\n // ── Members ─────────────────────────────────\n const membersView = new TableView({\n collection: new MemberList({ params: { group: this.model.get('id'), size: 10 } }),\n hideActivePillNames: ['group'],\n clickAction: 'view',\n showAdd: true,\n addButtonLabel: 'Invite',\n onAdd: (event) => this.onInviteClick(event),\n columns: [\n { key: 'user.display_name', label: 'User', sortable: true },\n { key: 'user.email', label: 'Email', sortable: true },\n { key: 'permissions|keys|badge', label: 'Permissions' },\n { key: 'created', label: 'Joined', formatter: 'date', sortable: true }\n ]\n });\n\n // ── Children (sub-groups) ───────────────────\n const childrenView = new TableView({\n collection: new GroupList({ params: { parent: this.model.get('id'), size: 10 } }),\n hideActivePillNames: ['parent'],\n clickAction: 'view',\n showAdd: true,\n addButtonLabel: 'Add Group',\n onAdd: () => this.onActionAddChildGroup(),\n columns: [\n { key: 'name', label: 'Name', sortable: true },\n { key: 'kind', label: 'Kind', formatter: 'badge' },\n {\n key: 'is_active', label: 'Status', width: '80px',\n template: `\n {{#model.is_active|bool}}<span class=\"badge bg-success bg-opacity-10 text-success\">Active</span>{{/model.is_active|bool}}\n {{^model.is_active|bool}}<span class=\"badge bg-secondary bg-opacity-10 text-secondary\">Inactive</span>{{/model.is_active|bool}}`\n },\n { key: 'created', label: 'Created', formatter: 'date', sortable: true }\n ]\n });\n\n // ── Events ──────────────────────────────────\n const eventsView = new TableView({\n collection: new IncidentEventList({\n params: { size: 10, model_name: 'account.Group', model_id: this.model.get('id') }\n }),\n hideActivePillNames: ['model_name', 'model_id'],\n columns: [\n { key: 'created', label: 'Date', formatter: 'datetime', sortable: true, width: '150px' },\n { key: 'category|badge', label: 'Category' },\n { key: 'title', label: 'Title' }\n ]\n });\n\n // ── Logs ────────────────────────────────────\n const logsView = new TableView({\n collection: new LogList({\n params: { size: 10, model_name: 'account.Group', model_id: this.model.get('id') }\n }),\n permissions: 'view_logs',\n hideActivePillNames: ['model_name', 'model_id'],\n columns: [\n {\n key: 'created', label: 'Timestamp', sortable: true, formatter: 'epoch|datetime',\n filter: { name: 'created', type: 'daterange', startName: 'dr_start', endName: 'dr_end', fieldName: 'dr_field', label: 'Date Range', format: 'YYYY-MM-DD', displayFormat: 'MMM DD, YYYY', separator: ' to ' }\n },\n {\n key: 'level', label: 'Level', sortable: true,\n filter: { type: 'select', options: [{ value: 'info', label: 'Info' }, { value: 'warning', label: 'Warning' }, { value: 'error', label: 'Error' }] }\n },\n { key: 'kind', label: 'Kind', filter: { type: 'text' } },\n { name: 'log', label: 'Log' }\n ]\n });\n\n // ── SideNavView ─────────────────────────────\n this.sideNavView = new SideNavView({\n containerId: 'group-sidenav',\n activeSection: 'details',\n navWidth: 180,\n contentPadding: '1.25rem 2rem',\n enableResponsive: true,\n minWidth: 500,\n sections: [\n { key: 'details', label: 'Details', icon: 'bi-info-circle', view: detailsView },\n { key: 'members', label: 'Members', icon: 'bi-people', view: membersView },\n { key: 'children', label: 'Sub-Groups', icon: 'bi-diagram-3', view: childrenView },\n { type: 'divider', label: 'Activity' },\n { key: 'events', label: 'Events', icon: 'bi-calendar-event', view: eventsView },\n { key: 'logs', label: 'Logs', icon: 'bi-journal-text', view: logsView, permissions: 'view_logs' }\n ]\n });\n this.addChild(this.sideNavView);\n\n // ── Context Menu ────────────────────────────\n const groupMenu = new ContextMenu({\n containerId: 'group-context-menu',\n className: 'context-menu-view header-menu-absolute',\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: [\n { label: 'Edit Group', action: 'edit-group', icon: 'bi-pencil' },\n { type: 'divider' },\n { label: 'Invite Member', action: 'invite-member', icon: 'bi-person-plus' },\n { label: 'Add Sub-Group', action: 'add-child-group', icon: 'bi-diagram-3' },\n { type: 'divider' },\n this.model.get('is_active')\n ? { label: 'Deactivate Group', action: 'deactivate-group', icon: 'bi-toggle-off' }\n : { label: 'Activate Group', action: 'activate-group', icon: 'bi-toggle-on' },\n ]\n }\n });\n this.addChild(groupMenu);\n }\n\n // ── Actions ─────────────────────────────────\n\n async onActionEditGroup() {\n const resp = await Dialog.showModelForm({\n title: `Edit Group — ${this.model.get('name')}`,\n model: this.model,\n size: 'lg',\n formConfig: GroupForms.detailed,\n });\n if (resp) {\n await this.render();\n }\n }\n\n async onActionInviteMember() {\n return this.onInviteClick(new Event('click'));\n }\n\n async onInviteClick(event) {\n if (event?.preventDefault) {\n event.preventDefault();\n event.stopPropagation();\n }\n const data = await Dialog.showForm({\n title: `Invite User to ${this.model.get('name')}`,\n size: 'sm',\n fields: [\n { type: 'email', name: 'email', label: 'Email', required: true, cols: 12 }\n ]\n });\n if (!data?.email) return true;\n\n const app = this.getApp();\n const resp = await app.rest.POST('/api/group/member/invite', {\n group: this.model.id,\n email: data.email\n });\n if (resp.success) {\n app.toast.success('User invited successfully');\n // Refresh members if on that section\n if (this.sideNavView?.getActiveSection() === 'members') {\n await this.sideNavView.showSection('members');\n }\n } else {\n app.toast.error(resp.message || 'Failed to invite user');\n }\n return true;\n }\n\n async onActionAddChildGroup() {\n const data = await Dialog.showForm({\n title: `Add Sub-Group to ${this.model.get('name')}`,\n size: 'sm',\n fields: GroupForms.create.fields.filter(f => f.name !== 'parent')\n });\n if (!data) return true;\n\n data.parent = this.model.id;\n const newGroup = new Group(data);\n const resp = await newGroup.save();\n if (resp.status === 200 || resp.status === 201) {\n this.getApp()?.toast?.success('Sub-group created');\n if (this.sideNavView?.getActiveSection() === 'children') {\n await this.sideNavView.showSection('children');\n }\n } else {\n this.getApp()?.toast?.error(resp.message || 'Failed to create sub-group');\n }\n return true;\n }\n\n async onActionDeactivateGroup() {\n const confirmed = await Dialog.confirm(\n `Are you sure you want to deactivate <strong>${this.model.get('name')}</strong>?`,\n 'Deactivate Group'\n );\n if (!confirmed) return true;\n\n const resp = await this.model.save({ is_active: false });\n if (resp.status === 200) {\n this.getApp()?.toast?.success('Group deactivated');\n await this.render();\n } else {\n this.getApp()?.toast?.error('Failed to deactivate group');\n }\n return true;\n }\n\n async onActionActivateGroup() {\n const confirmed = await Dialog.confirm(\n `Are you sure you want to activate <strong>${this.model.get('name')}</strong>?`,\n 'Activate Group'\n );\n if (!confirmed) return true;\n\n const resp = await this.model.save({ is_active: true });\n if (resp.status === 200) {\n this.getApp()?.toast?.success('Group activated');\n await this.render();\n } else {\n this.getApp()?.toast?.error('Failed to activate group');\n }\n return true;\n }\n\n async onActionViewParent(event, element) {\n const parentId = element?.dataset?.id;\n if (!parentId) return true;\n\n const parent = new Group({ id: parentId });\n await parent.fetch();\n if (parent.id) {\n Dialog.showDialog({\n title: false,\n size: 'lg',\n body: new GroupView({ model: parent }),\n buttons: [{ text: 'Close', class: 'btn-secondary', dismiss: true }]\n });\n }\n return true;\n }\n\n // ── Navigation helpers ──────────────────────\n\n async showSection(sectionName) {\n if (this.sideNavView) {\n await this.sideNavView.showSection(sectionName);\n }\n }\n\n getActiveSection() {\n return this.sideNavView ? this.sideNavView.getActiveSection() : null;\n }\n\n _onModelChange() {\n // Prevent full re-render on model changes\n }\n\n static create(options = {}) {\n return new GroupView(options);\n }\n}\n\nGroup.VIEW_CLASS = GroupView;\n\nexport default GroupView;\n","/**\n * GroupTablePage - Group management using TablePage component\n * Clean implementation using TablePage with minimal overrides\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { GroupList, GroupForms, Group } from '@core/models/Group.js';\nimport GroupView from './GroupView.js';\n\nclass GroupTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_groups',\n pageName: 'Manage Groups',\n router: \"admin/groups\",\n Collection: GroupList,\n\n formCreate: GroupForms.create,\n formEdit: GroupForms.edit,\n itemViewClass: GroupView,\n\n viewDialogOptions: {\n header: false\n },\n\n defaultQuery: {\n sort: '-id',\n is_active: 1,\n },\n\n // Column definitions\n columns: [\n {\n key: 'id',\n label: 'ID',\n sortable: true,\n class: 'text-muted'\n },\n\n {\n key: 'name',\n label: 'Display Name'\n },\n {\n key: 'kind|badge',\n label: 'Kind',\n filter: {\n type: \"select\",\n options: Group.GroupKindOptions\n }\n },\n {\n key: 'is_active|yesnoicon',\n label: 'Enabled',\n visibility: 'lg'\n },\n {\n key: 'parent.name',\n label: 'Parent',\n formatter: \"default('-')\",\n visibility: 'md',\n class: 'text-muted fs-8'\n },\n {\n key: 'created',\n label: 'Created',\n className: 'text-muted fs-8',\n formatter: \"epoch|datetime\",\n visibility: 'lg'\n },\n {\n key: 'last_activity',\n label: 'Activity',\n className: 'text-muted fs-8',\n formatter: \"relative\",\n visibility: 'lg'\n }\n ],\n\n filters: [\n {\n key: 'is_active',\n label: 'Active',\n type: 'select',\n options: [\n { label: 'Active', value: true },\n { label: 'Inactive', value: false }\n ]\n }\n ],\n\n contextMenu: [\n {\n icon: 'bi-pencil',\n action: 'edit',\n label: \"Edit Group\"\n },\n {\n icon: 'bi-bullseye',\n action: 'make-active',\n label: \"Make Active Group\"\n }\n ],\n\n // Table features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Toolbar\n showRefresh: true,\n showAdd: true,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No groups found. Click \"Add Group\" to create your first one.',\n\n // Batch actions\n batchBarLocation: 'top',\n batchActions: [\n { label: \"Delete\", icon: \"bi bi-trash\", action: \"batch-delete\" },\n { label: \"Export\", icon: \"bi bi-download\", action: \"batch-export\" },\n { label: \"Activate\", icon: \"bi bi-check-circle\", action: \"batch-activate\" },\n { label: \"Deactivate\", icon: \"bi bi-x-circle\", action: \"batch-deactivate\" },\n { label: \"Move\", icon: \"bi bi-arrow-right\", action: \"batch-move\" }\n ],\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n },\n });\n }\n\n onActionMakeActive(event, element) {\n const item = this.collection.get(element.dataset.id);\n this.getApp().setActiveGroup(item);\n }\n\n}\n\nexport default GroupTablePage;\n","/**\n * DeviceView - Clean, modern user device detail view\n *\n * SideNavView layout with:\n * - Details: browser, OS, device, user agent in clean field rows\n * - Locations: multiline table of all locations this device connected from\n * - Header: smart browser/OS icon, device summary, last seen, context menu\n */\n\nimport View from '@core/View.js';\nimport SideNavView from '@core/views/navigation/SideNavView.js';\nimport TableView from '@core/views/table/TableView.js';\nimport TableRow from '@core/views/table/TableRow.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport { UserDevice, UserDeviceLocationList } from '@core/models/User.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\n\n// ── Location row for multiline table ──\n\nclass DeviceLocationRow extends TableRow {\n get locationText() {\n const geo = this.model?.get('geolocation') || {};\n return [geo.city, geo.region].filter(Boolean).join(', ') || geo.country_name || '—';\n }\n get countryName() {\n return this.model?.get('geolocation')?.country_name || '';\n }\n get ispName() {\n const geo = this.model?.get('geolocation') || {};\n return geo.isp || geo.asn_org || '';\n }\n get threatFlags() {\n const geo = this.model?.get('geolocation') || {};\n const flags = [];\n if (geo.is_vpn) flags.push('<span class=\"badge bg-warning text-dark\" style=\"font-size:0.6rem;\">VPN</span>');\n if (geo.is_tor) flags.push('<span class=\"badge bg-danger\" style=\"font-size:0.6rem;\">Tor</span>');\n if (geo.is_proxy) flags.push('<span class=\"badge bg-warning text-dark\" style=\"font-size:0.6rem;\">Proxy</span>');\n return flags.join(' ');\n }\n get hasThreatFlags() {\n const geo = this.model?.get('geolocation') || {};\n return !!(geo.is_vpn || geo.is_tor || geo.is_proxy);\n }\n}\n\nclass DeviceView extends View {\n constructor(options = {}) {\n super({\n className: 'device-view',\n ...options\n });\n\n this.model = options.model || new UserDevice(options.data || {});\n this.deviceInfo = this.model.get('device_info') || {};\n this.deviceIcon = this._getIcon(this.deviceInfo);\n\n // Computed properties for template\n this.browserFull = this._getBrowser();\n this.osFull = this._getOS();\n this.deviceFull = this._getDevice();\n this.isMobile = this._isMobile();\n\n this.template = `\n <style>\n .dv-header { display: flex; justify-content: space-between; align-items: flex-start; margin-bottom: 1.5rem; }\n .dv-identity { display: flex; align-items: center; gap: 1rem; }\n .dv-icon-wrap { width: 48px; height: 48px; border-radius: 12px; display: flex; align-items: center; justify-content: center; font-size: 1.4rem; flex-shrink: 0; }\n .dv-title { font-size: 1.15rem; font-weight: 600; margin: 0; line-height: 1.3; }\n .dv-subtitle { font-size: 0.8rem; color: #6c757d; margin-top: 0.15rem; }\n .dv-status { text-align: right; display: flex; align-items: flex-start; gap: 0.75rem; }\n .dv-last-seen-label { font-size: 0.7rem; color: #adb5bd; text-transform: uppercase; letter-spacing: 0.04em; }\n .dv-last-seen-value { font-size: 0.88rem; font-weight: 500; }\n .dv-last-seen-ip { font-size: 0.75rem; color: #6c757d; margin-top: 0.1rem; }\n </style>\n\n <div class=\"dv-header\">\n <div class=\"dv-identity\">\n <div class=\"dv-icon-wrap bg-primary bg-opacity-10 text-primary\">\n <i class=\"bi {{deviceIcon}}\"></i>\n </div>\n <div>\n <h4 class=\"dv-title\">{{browserFull}} <span class=\"fw-normal text-muted\">on</span> {{osFull}}</h4>\n <div class=\"dv-subtitle\">\n {{deviceFull}}\n {{#model.user.display_name}}\n <span class=\"text-muted mx-1\">&middot;</span>\n <a href=\"#\" data-action=\"view-user\" class=\"text-decoration-none\">{{model.user.display_name}}</a>\n {{/model.user.display_name}}\n </div>\n </div>\n </div>\n <div class=\"dv-status\">\n <div>\n <div class=\"dv-last-seen-label\">Last Seen</div>\n <div class=\"dv-last-seen-value\">{{model.last_seen|relative}}</div>\n {{#model.last_ip}}<div class=\"dv-last-seen-ip\">{{model.last_ip}}</div>{{/model.last_ip}}\n </div>\n <div data-container=\"device-context-menu\"></div>\n </div>\n </div>\n\n <div data-container=\"device-sidenav\" style=\"min-height: 300px;\"></div>\n `;\n }\n\n // ── Computed getters ──────────────────────────\n\n _getBrowser() {\n const ua = this.deviceInfo?.user_agent || {};\n const parts = [ua.family, ua.major].filter(Boolean);\n return parts.length ? parts.join(' ') : 'Unknown Browser';\n }\n\n _getOS() {\n const os = this.deviceInfo?.os || {};\n const ver = [os.major, os.minor].filter(Boolean).join('.');\n return os.family ? `${os.family} ${ver}`.trim() : 'Unknown OS';\n }\n\n _getDevice() {\n const dev = this.deviceInfo?.device || {};\n const parts = [dev.brand, dev.family].filter(Boolean);\n const name = parts.length ? parts.join(' ') : 'Unknown Device';\n return dev.model ? `${name} (${dev.model})` : name;\n }\n\n _isMobile() {\n const dev = this.deviceInfo?.device || {};\n const os = this.deviceInfo?.os || {};\n return ['iPhone', 'Android', 'iPad'].some(m =>\n (dev.family || '').includes(m) || (os.family || '').includes(m)\n );\n }\n\n _getIcon(deviceInfo) {\n const os = deviceInfo?.os?.family?.toLowerCase() || '';\n const browser = deviceInfo?.user_agent?.family?.toLowerCase() || '';\n const device = deviceInfo?.device?.family?.toLowerCase() || '';\n\n if (browser.includes('chrome')) return 'bi-browser-chrome';\n if (browser.includes('firefox')) return 'bi-browser-firefox';\n if (browser.includes('safari')) return 'bi-browser-safari';\n if (browser.includes('edge')) return 'bi-browser-edge';\n if (os.includes('mac') || os.includes('ios')) return 'bi-apple';\n if (os.includes('windows')) return 'bi-windows';\n if (os.includes('android')) return 'bi-android2';\n if (os.includes('linux')) return 'bi-ubuntu';\n if (device.includes('iphone')) return 'bi-phone';\n if (device.includes('ipad')) return 'bi-tablet';\n return 'bi-laptop';\n }\n\n async onInit() {\n // ── Details section ─────────────────────────\n const detailsView = new View({\n model: this.model,\n template: `\n <style>\n .dv-section-label { font-size: 0.7rem; font-weight: 700; text-transform: uppercase; letter-spacing: 0.06em; color: #adb5bd; margin-bottom: 0.5rem; margin-top: 1.5rem; }\n .dv-section-label:first-child { margin-top: 0; }\n .dv-field-row { display: flex; align-items: baseline; padding: 0.5rem 0; border-bottom: 1px solid #f0f0f0; }\n .dv-field-row:last-child { border-bottom: none; }\n .dv-field-label { width: 130px; font-size: 0.78rem; color: #6c757d; flex-shrink: 0; }\n .dv-field-value { flex: 1; font-size: 0.88rem; color: #212529; }\n .dv-ua-string { font-family: ui-monospace, monospace; font-size: 0.73rem; color: #6c757d; word-break: break-all; line-height: 1.5; padding: 0.5rem 0.75rem; background: #f8f9fa; border-radius: 6px; margin-top: 0.25rem; }\n </style>\n\n <div class=\"dv-section-label\">Browser</div>\n <div class=\"dv-field-row\">\n <div class=\"dv-field-label\">Name</div>\n <div class=\"dv-field-value\">{{model.device_info.user_agent.family|default('—')}}</div>\n </div>\n <div class=\"dv-field-row\">\n <div class=\"dv-field-label\">Version</div>\n <div class=\"dv-field-value\">{{model.device_info.user_agent.major|default('—')}}{{#model.device_info.user_agent.minor}}.{{model.device_info.user_agent.minor}}{{/model.device_info.user_agent.minor}}{{#model.device_info.user_agent.patch}}.{{model.device_info.user_agent.patch}}{{/model.device_info.user_agent.patch}}</div>\n </div>\n\n <div class=\"dv-section-label\">Operating System</div>\n <div class=\"dv-field-row\">\n <div class=\"dv-field-label\">Name</div>\n <div class=\"dv-field-value\">{{model.device_info.os.family|default('—')}}</div>\n </div>\n <div class=\"dv-field-row\">\n <div class=\"dv-field-label\">Version</div>\n <div class=\"dv-field-value\">{{model.device_info.os.major|default('—')}}{{#model.device_info.os.minor}}.{{model.device_info.os.minor}}{{/model.device_info.os.minor}}{{#model.device_info.os.patch}}.{{model.device_info.os.patch}}{{/model.device_info.os.patch}}</div>\n </div>\n\n <div class=\"dv-section-label\">Hardware</div>\n <div class=\"dv-field-row\">\n <div class=\"dv-field-label\">Brand</div>\n <div class=\"dv-field-value\">{{model.device_info.device.brand|default('—')}}</div>\n </div>\n <div class=\"dv-field-row\">\n <div class=\"dv-field-label\">Family</div>\n <div class=\"dv-field-value\">{{model.device_info.device.family|default('—')}}</div>\n </div>\n <div class=\"dv-field-row\">\n <div class=\"dv-field-label\">Model</div>\n <div class=\"dv-field-value\">{{model.device_info.device.model|default('—')}}</div>\n </div>\n\n <div class=\"dv-section-label\">Identification</div>\n <div class=\"dv-field-row\">\n <div class=\"dv-field-label\">Device ID</div>\n <div class=\"dv-field-value\" style=\"font-family: ui-monospace, monospace; font-size: 0.78rem;\">{{model.duid|truncate_middle(32)}}</div>\n </div>\n <div class=\"dv-field-row\">\n <div class=\"dv-field-label\">Last IP</div>\n <div class=\"dv-field-value\">{{model.last_ip|default('—')}}</div>\n </div>\n <div class=\"dv-field-row\">\n <div class=\"dv-field-label\">First Seen</div>\n <div class=\"dv-field-value\">{{model.first_seen|epoch|datetime|default('—')}}</div>\n </div>\n <div class=\"dv-field-row\">\n <div class=\"dv-field-label\">Last Seen</div>\n <div class=\"dv-field-value\">{{model.last_seen|epoch|datetime|default('—')}}</div>\n </div>\n\n {{#model.device_info.string}}\n <div class=\"dv-section-label\">User Agent String</div>\n <div class=\"dv-ua-string\">{{model.device_info.string}}</div>\n {{/model.device_info.string}}\n `\n });\n\n // ── Locations section — multiline rows ──────\n const locationsView = new TableView({\n collection: new UserDeviceLocationList({\n params: { user_device: this.model.get('id'), size: 10 }\n }),\n hideActivePillNames: ['user_device'],\n clickAction: 'view',\n itemClass: DeviceLocationRow,\n selectable: false,\n columns: [\n {\n key: 'ip_address',\n label: 'Location',\n template: `\n <div style=\"font-size:0.85rem; font-weight:500;\">\n <i class=\"bi bi-geo-alt text-muted me-1\" style=\"font-size:0.95rem; vertical-align:middle;\"></i>{{locationText}}\n {{#countryName}} <span class=\"text-muted fw-normal\">&middot; {{countryName}}</span>{{/countryName}}\n </div>\n <div style=\"font-size:0.73rem; color:#6c757d; margin-top:0.15rem;\">\n {{model.ip_address}}\n {{#ispName}} <span class=\"text-muted mx-1\">&middot;</span> {{ispName}}{{/ispName}}\n {{#hasThreatFlags|bool}} <span class=\"ms-1\">{{{threatFlags}}}</span>{{/hasThreatFlags|bool}}\n </div>`\n },\n { key: 'first_seen', label: 'First Seen', formatter: 'epoch|relative', width: '110px' },\n { key: 'last_seen', label: 'Last Seen', formatter: 'epoch|relative', width: '110px' }\n ]\n });\n\n // ── SideNavView ─────────────────────────────\n this.sideNavView = new SideNavView({\n containerId: 'device-sidenav',\n activeSection: 'details',\n navWidth: 160,\n contentPadding: '1rem 1.5rem',\n enableResponsive: true,\n minWidth: 450,\n sections: [\n { key: 'details', label: 'Details', icon: 'bi-info-circle', view: detailsView },\n { key: 'locations', label: 'Locations', icon: 'bi-geo-alt', view: locationsView }\n ]\n });\n this.addChild(this.sideNavView);\n\n // ── Context Menu ────────────────────────────\n const deviceMenu = new ContextMenu({\n containerId: 'device-context-menu',\n className: \"context-menu-view header-menu-absolute\",\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: [\n { label: 'View User', action: 'view-user', icon: 'bi-person' },\n { label: 'Block Device', action: 'block-device', icon: 'bi-shield-slash', disabled: true },\n { type: 'divider' },\n { label: 'Delete Record', action: 'delete-device', icon: 'bi-trash', danger: true }\n ]\n }\n });\n this.addChild(deviceMenu);\n }\n\n async onActionViewUser() {\n this.emit('view-user', { userId: this.model.get('user')?.id });\n }\n\n async onActionDeleteDevice() {\n const confirmed = await Dialog.confirm(\n 'Are you sure you want to delete this device record?',\n 'Delete Device'\n );\n if (!confirmed) return true;\n\n const resp = await this.model.destroy();\n if (resp.success) {\n this.emit('device:deleted', { model: this.model });\n }\n return true;\n }\n\n static async show(duid) {\n const model = await UserDevice.getByDuid(duid);\n if (model) {\n return Dialog.showDialog({\n title: false,\n size: 'lg',\n body: new DeviceView({ model }),\n buttons: [{ text: 'Close', class: 'btn-secondary', dismiss: true }]\n });\n }\n Dialog.alert({ message: `Could not find device with DUID: ${duid}`, type: 'warning' });\n return null;\n }\n}\n\nUserDevice.VIEW_CLASS = DeviceView;\nexport default DeviceView;\n","/**\n * UserDeviceTablePage - User device management using TablePage component\n * Clean implementation using TablePage with minimal overrides\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { UserDeviceList } from '@core/models/User.js';\nimport DeviceView from './DeviceView.js';\n\nclass UserDeviceTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_user_devices',\n pageName: 'User Devices',\n router: \"admin/user/devices\",\n Collection: UserDeviceList,\n\n // itemViewClass: DeviceView,\n viewDialogOptions: {\n header: false,\n size: 'lg'\n },\n\n // Column definitions\n columns: [\n { key: 'id', label: 'ID', width: '70px', sortable: true, class: 'text-muted' },\n { key: 'duid', label: 'Device ID', sortable: true, formatter: 'truncate_middle(16)' },\n { key: 'user.display_name', label: 'User', sortable: true, formatter: \"default('—')\" },\n { key: 'device_info.user_agent.family', label: 'Browser', formatter: \"default('—')\" },\n { key: 'device_info.os.family', label: 'OS', formatter: \"default('—')\" },\n { key: 'last_ip', label: 'Last IP', sortable: true },\n { key: 'first_seen', label: 'First Seen', formatter: \"epoch|datetime\" },\n { key: 'last_seen', label: 'Last Seen', formatter: \"epoch|datetime\" }\n ],\n\n // Table features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Toolbar\n showRefresh: true,\n showAdd: false,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No user devices found.',\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n }\n });\n }\n}\n\nexport default UserDeviceTablePage;\n","/**\n * UserDeviceLocationTablePage - Device location tracking using TablePage component\n * Clean implementation using TablePage with minimal overrides\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { UserDeviceLocationList } from '@core/models/User.js';\n\nclass UserDeviceLocationTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_user_device_locations',\n pageName: 'Device Locations',\n router: \"admin/user/device-locations\",\n Collection: UserDeviceLocationList,\n\n // Column definitions\n columns: [\n { key: 'id', label: 'ID', width: '70px', sortable: true, class: 'text-muted' },\n { key: 'user.display_name', label: 'User', sortable: true },\n { key: 'user_device', label: 'Device', template: '{{user_device.device_info.user_agent.family}} on {{user_device.device_info.os.family}}', sortable: true },\n { key: 'ip_address', label: 'IP Address', sortable: true },\n { key: 'geolocation.city', label: 'City', formatter: \"default('—')\" },\n { key: 'geolocation.region', label: 'Region', formatter: \"default('—')\" },\n { key: 'geolocation.country_name', label: 'Country', formatter: \"default('—')\" },\n { key: 'last_seen', label: 'Last Seen', formatter: \"epoch|datetime\" }\n ],\n\n // Table features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Toolbar\n showRefresh: true,\n showAdd: false,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No device locations found.',\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n }\n });\n }\n}\n\nexport default UserDeviceLocationTablePage;","/**\n * GeoIPView - Detailed view for a GeoLocatedIP record\n */\n\nimport View from '@core/View.js';\nimport TabView from '@core/views/navigation/TabView.js';\nimport DataView from '@core/views/data/DataView.js';\nimport TableView from '@core/views/table/TableView.js';\nimport MapView from '@ext/map/MapView.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport { GeoLocatedIP } from '@core/models/System.js';\nimport { IncidentEventList } from '@core/models/Incident.js';\nimport { LogList } from '@core/models/Log.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\n\nclass GeoIPView extends View {\n constructor(options = {}) {\n super({\n className: 'geoip-view',\n ...options\n });\n\n this.model = options.model || new GeoLocatedIP(options.data || {});\n this.hasCoordinates = this.model.get('latitude') && this.model.get('longitude');\n\n this.template = `\n <div class=\"geoip-view-container\">\n <!-- Header -->\n <div class=\"d-flex justify-content-between align-items-center mb-4\">\n <!-- Left Side: Icon & Info -->\n <div class=\"d-flex align-items-center gap-3\">\n <div class=\"fs-1 text-primary\">\n <i class=\"bi bi-globe-americas\"></i>\n </div>\n <div>\n <h3 class=\"mb-1\">{{model.ip_address}}</h3>\n <div class=\"text-muted small\">\n {{model.city|default('Unknown Location')}}, {{model.country_name|default('Unknown Location')}}\n </div>\n <div class=\"text-muted small mt-1\">\n ISP: {{model.isp|capitalize}}\n </div>\n </div>\n </div>\n\n <!-- Right Side: Risk Summary + Actions -->\n <div class=\"d-flex align-items-start gap-4\">\n <!-- Risk summary -->\n <div class=\"text-end\">\n <div class=\"d-flex align-items-baseline justify-content-end gap-2\">\n <span class=\"text-muted\">Risk:</span>\n <span class=\"fw-bold fs-4\n {{#model.is_threat}} text-danger {{/model.is_threat}}\n {{#model.is_suspicious}} text-warning {{/model.is_suspicious}}\n {{^model.is_threat}}{{^model.is_suspicious}} text-success {{/model.is_suspicious}}{{/model.is_threat}}\n \">{{#model.threat_level}}{{model.threat_level|capitalize}}{{/model.threat_level}}{{^model.threat_level}}Unknown{{/model.threat_level}}</span>\n </div>\n <div class=\"mt-1 small d-flex align-items-center justify-content-end gap-2\">\n <span class=\"text-muted\">Score:</span>\n <span class=\"fw-semibold\">{{model.risk_score|default('—')}}</span>\n </div>\n <div class=\"mt-1 d-flex align-items-center justify-content-end gap-2\">\n <i class=\"bi bi-shield-lock {{#model.is_tor}}fs-4 text-success{{/model.is_tor}}{{^model.is_tor}}text-muted{{/model.is_tor}}\" data-bs-toggle=\"tooltip\" title=\"TOR exit\"></i>\n <i class=\"bi bi-shield {{#model.is_vpn}}fs-4 text-success{{/model.is_vpn}}{{^model.is_vpn}}text-muted{{/model.is_vpn}}\" data-bs-toggle=\"tooltip\" title=\"VPN detected\"></i>\n <i class=\"bi bi-cloud {{#model.is_cloud}}fs-4 text-success{{/model.is_cloud}}{{^model.is_cloud}}text-muted{{/model.is_cloud}}\" data-bs-toggle=\"tooltip\" title=\"Cloud provider\"></i>\n <i class=\"bi bi-hdd-stack {{#model.is_datacenter}}fs-4 text-success{{/model.is_datacenter}}{{^model.is_datacenter}}text-muted{{/model.is_datacenter}}\" data-bs-toggle=\"tooltip\" title=\"Datacenter\"></i>\n <i class=\"bi bi-phone {{#model.is_mobile}}fs-4 text-success{{/model.is_mobile}}{{^model.is_mobile}}text-muted{{/model.is_mobile}}\" data-bs-toggle=\"tooltip\" title=\"Mobile connection\"></i>\n <i class=\"bi bi-diagram-3 {{#model.is_proxy}}fs-4 text-success{{/model.is_proxy}}{{^model.is_proxy}}text-muted{{/model.is_proxy}}\" data-bs-toggle=\"tooltip\" title=\"Proxy\"></i>\n </div>\n </div>\n <!-- Actions: context menu aligned to top (not vertically centered) -->\n <div class=\"d-flex align-items-start\">\n <div data-container=\"geoip-context-menu\"></div>\n </div>\n </div>\n </div>\n\n <!-- Tabs -->\n <div data-container=\"geoip-tabs\"></div>\n </div>\n `;\n }\n\n async onInit() {\n // Location Details Tab\n this.detailsView = new DataView({\n model: this.model,\n className: \"p-3\",\n showEmptyValues: true,\n emptyValueText: '—',\n columns: 2,\n fields: [\n { name: 'ip_address', label: 'IP Address', cols: 4 },\n { name: 'subnet', label: 'Subnet', cols: 4 },\n { name: 'country_name', label: 'Country', cols: 4 },\n { name: 'country_code', label: 'Country Code', cols: 4 },\n { name: 'region', label: 'Region', cols: 4 },\n { name: 'city', label: 'City', cols: 4 },\n { name: 'postal_code', label: 'Postal Code', cols: 4 },\n { name: 'timezone', label: 'Timezone', cols: 4 },\n { name: 'latitude', label: 'Latitude', cols: 4 },\n { name: 'longitude', label: 'Longitude', cols: 4 },\n ]\n });\n\n // Network Tab\n this.networkView = new DataView({\n model: this.model,\n className: \"p-3\",\n showEmptyValues: true,\n emptyValueText: '—',\n columns: 2,\n fields: [\n { name: 'is_tor', label: 'TOR Exit Node', formatter: 'yesnoicon', cols: 4 },\n { name: 'is_vpn', label: 'VPN', formatter: 'yesnoicon', cols: 4 },\n { name: 'is_proxy', label: 'Proxy', formatter: 'yesnoicon', cols: 4 },\n { name: 'is_cloud', label: 'Cloud Provider', formatter: 'yesnoicon', cols: 4 },\n { name: 'is_datacenter', label: 'Datacenter', formatter: 'yesnoicon', cols: 4 },\n { name: 'is_mobile', label: 'Mobile', formatter: 'yesnoicon', cols: 4 },\n { name: 'mobile_carrier', label: 'Mobile Carrier', cols: 8 },\n { name: 'asn', label: 'ASN', cols: 4 },\n { name: 'asn_org', label: 'ASN Organization', cols: 8 },\n { name: 'isp', label: 'ISP', cols: 12 },\n { name: 'connection_type', label: 'Connection Type', cols: 6 }\n ]\n });\n\n // Risk & Reputation Tab\n this.riskView = new DataView({\n model: this.model,\n className: \"p-3\",\n showEmptyValues: true,\n emptyValueText: '—',\n columns: 2,\n fields: [\n { name: 'threat_level', label: 'Threat Level', cols: 6 },\n { name: 'risk_score', label: 'Risk Score', cols: 6 },\n { name: 'is_threat', label: 'Threat', formatter: 'yesnoicon', cols: 6 },\n { name: 'is_suspicious', label: 'Suspicious', formatter: 'yesnoicon', cols: 6 },\n { name: 'is_known_attacker', label: 'Known Attacker', formatter: 'yesnoicon', cols: 6 },\n { name: 'is_known_abuser', label: 'Known Abuser', formatter: 'yesnoicon', cols: 6 }\n ]\n });\n\n // Metadata Tab\n this.metadataView = new DataView({\n model: this.model,\n className: \"p-3\",\n showEmptyValues: true,\n emptyValueText: '—',\n columns: 2,\n fields: [\n { name: 'id', label: 'Record ID', cols: 6 },\n { name: 'provider', label: 'Data Provider', formatter: 'capitalize', cols: 6 },\n { name: 'created', label: 'Created', formatter: 'datetime', cols: 6 },\n { name: 'modified', label: 'Last Modified', formatter: 'datetime', cols: 6 },\n { name: 'last_seen', label: 'Last Seen', formatter: 'datetime', cols: 6 },\n { name: 'expires_at', label: 'Expires', formatter: 'datetime', cols: 6 }\n ]\n });\n\n // Create Events table with IncidentEventList collection\n const eventsCollection = new IncidentEventList({\n params: {\n size: 5,\n source_ip: this.model.get(\"ip_address\")\n }\n });\n this.eventsView = new TableView({\n collection: eventsCollection,\n hideActivePillNames: ['source_ip'],\n columns: [\n { key: 'id', label: 'ID', sortable: true, width: '40px' },\n { key: 'created', label: 'Date', formatter: 'datetime', sortable: true, width: '150px' },\n { key: 'category|badge', label: 'Category' },\n { key: 'title', label: 'Title' }\n ]\n });\n\n // Create Logs table with LogList collection\n const logsCollection = new LogList({\n params: {\n size: 5,\n ip: this.model.get('ip_address')\n }\n });\n this.logsView = new TableView({\n collection: logsCollection,\n permissions: 'view_logs',\n hideActivePillNames: ['ip'],\n columns: [\n {\n key: 'created',\n label: 'Timestamp',\n sortable: true,\n formatter: \"epoch|datetime\",\n filter: {\n name: \"created\",\n type: 'daterange',\n startName: 'dr_start',\n endName: 'dr_end',\n fieldName: 'dr_field',\n label: 'Date Range',\n format: 'YYYY-MM-DD',\n displayFormat: 'MMM DD, YYYY',\n separator: ' to '\n }\n },\n {\n key: 'level',\n label: 'Level',\n sortable: true,\n filter: {\n type: 'select',\n options: [\n { value: 'info', label: 'Info' },\n { value: 'warning', label: 'Warning' },\n { value: 'error', label: 'Error' }\n ]\n }\n },\n { key: 'kind', label: 'Kind', filter: { type: 'text' } },\n { name: 'log', label: 'Log' }\n ]\n });\n\n const tabs = {\n 'Location': this.detailsView,\n 'Network': this.networkView,\n 'Risk & Reputation': this.riskView,\n 'Events': this.eventsView,\n 'Logs': this.logsView,\n 'Metadata': this.metadataView\n };\n\n // Add Map tab if coordinates exist\n if (this.hasCoordinates) {\n const lat = this.model.get('latitude');\n const lng = this.model.get('longitude');\n const city = this.model.get('city') || 'Unknown';\n const region = this.model.get('region') || '';\n const country = this.model.get('country_name') || '';\n\n const locationStr = [city, region, country].filter(Boolean).join(', ');\n\n this.mapView = new MapView({\n markers: [{\n lat: lat,\n lng: lng,\n popup: `<strong>${this.model.get('ip_address')}</strong><br>${locationStr}`\n }],\n tileLayer: \"light\",\n zoom: 4,\n height: 450\n });\n tabs['Map'] = this.mapView;\n }\n\n this.tabView = new TabView({\n containerId: 'geoip-tabs',\n tabs: tabs,\n activeTab: this.hasCoordinates ? 'Map' : 'Location'\n });\n this.addChild(this.tabView);\n\n // ContextMenu\n const menuItems = [\n { label: 'Edit Location', action: 'edit-location', icon: 'bi-geo-alt' },\n { label: 'Edit Security', action: 'edit-security', icon: 'bi-shield-lock' },\n { label: 'Edit Network', action: 'edit-network', icon: 'bi-diagram-3' },\n { type: 'divider' },\n { label: 'Refresh Geolocation', action: 'refresh-geoip', icon: 'bi-arrow-clockwise' },\n ];\n\n if (this.hasCoordinates) {\n menuItems.push({\n label: 'View on Map',\n action: 'view-on-map',\n icon: 'bi-map'\n });\n }\n\n menuItems.push(\n { type: 'divider' },\n { label: 'Delete Record', action: 'delete-geoip', icon: 'bi-trash', danger: true }\n );\n\n const geoIPMenu = new ContextMenu({\n containerId: 'geoip-context-menu',\n className: \"context-menu-view header-menu-absolute\",\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: menuItems\n }\n });\n this.addChild(geoIPMenu);\n }\n\n async onAfterRender() {\n await super.onAfterRender();\n\n // Initialize Bootstrap tooltips for header icons/badges\n if (window.bootstrap && window.bootstrap.Tooltip && this.element) {\n const tooltipTriggerList = this.element.querySelectorAll('[data-bs-toggle=\"tooltip\"]');\n tooltipTriggerList.forEach(el => {\n // Dispose any existing instance (in case of re-render)\n const existing = window.bootstrap.Tooltip.getInstance(el);\n if (existing && typeof existing.dispose === 'function') {\n existing.dispose();\n }\n new window.bootstrap.Tooltip(el);\n });\n }\n }\n\n async onActionEditLocation() {\n const resp = await Dialog.showModelForm({\n title: `Edit Location - ${this.model.get('ip_address')}`,\n model: this.model,\n formConfig: GeoLocatedIP.EDIT_LOCATION_FORM,\n });\n\n if (resp) {\n await this.render();\n this.getApp()?.toast?.success('Location updated successfully');\n }\n }\n\n async onActionEditSecurity() {\n const resp = await Dialog.showModelForm({\n title: `Edit Security - ${this.model.get('ip_address')}`,\n model: this.model,\n formConfig: GeoLocatedIP.EDIT_SECURITY_FORM,\n });\n\n if (resp) {\n await this.render();\n this.getApp()?.toast?.success('Security settings updated successfully');\n }\n }\n\n async onActionEditNetwork() {\n const resp = await Dialog.showModelForm({\n title: `Edit Network - ${this.model.get('ip_address')}`,\n model: this.model,\n formConfig: GeoLocatedIP.EDIT_NETWORK_FORM,\n });\n\n if (resp) {\n await this.render();\n this.getApp()?.toast?.success('Network information updated successfully');\n }\n }\n\n async onActionRefreshGeoip() {\n // Placeholder for refresh logic, e.g., a POST request to a refresh endpoint\n await this.model.save({ refresh: true });\n this.getApp()?.toast?.info('Refresh request sent for ' + this.model.get('ip_address'));\n }\n\n async onActionThreatAnalysis() {\n // Placeholder for refresh logic, e.g., a POST request to a refresh endpoint\n await this.model.save({ threat_analysis: true });\n this.getApp()?.toast?.info('Requesting threat analysis for ' + this.model.get('ip_address'));\n }\n\n async onActionViewOnMap() {\n if (this.hasCoordinates) {\n const lat = this.model.get('latitude');\n const lon = this.model.get('longitude');\n const url = `https://www.google.com/maps/search/?api=1&query=${lat},${lon}`;\n window.open(url, '_blank');\n }\n }\n\n async onActionDeleteGeoip() {\n const confirmed = await Dialog.confirm(\n `Are you sure you want to delete the GeoIP record for \"${this.model.get('ip_address')}\"?`,\n 'Confirm Deletion',\n { confirmClass: 'btn-danger', confirmText: 'Delete' }\n );\n if (confirmed) {\n const resp = await this.model.destroy();\n if (resp.success) {\n this.emit('geoip:deleted', { model: this.model });\n }\n }\n }\n\n static async show(ip) {\n const model = await GeoLocatedIP.lookup(ip);\n if (model) {\n const view = new GeoIPView({ model });\n const dialog = new Dialog({\n header: false,\n size: 'lg',\n body: view,\n buttons: [{ text: 'Close', class: 'btn-secondary', dismiss: true }]\n });\n await dialog.render(true, document.body);\n dialog.show();\n return dialog;\n }\n Dialog.alert({ message: `Could not find geolocation data for IP: ${ip}`, type: 'warning' });\n return null;\n }\n}\n\nGeoLocatedIP.VIEW_CLASS = GeoIPView;\n\nexport default GeoIPView;\n","/**\n * GeoLocatedIPTablePage - GeoIP cache management using TablePage component\n * Clean implementation using TablePage with minimal overrides\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { GeoLocatedIPList, GeoLocatedIP } from '@core/models/System.js';\nimport GeoIPView from './GeoIPView.js';\n\nclass GeoLocatedIPTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_system_geoip',\n pageName: 'GeoIP Cache',\n router: \"admin/system/geoip\",\n Collection: GeoLocatedIPList,\n\n itemView: GeoIPView,\n viewDialogOptions: {\n header: false,\n size: 'xl'\n },\n\n // Column definitions\n columns: [\n { key: 'ip_address', label: 'IP Address', sortable: true },\n { key: 'city', label: 'City', sortable: true, formatter: \"default('—')\" },\n { key: 'region', label: 'Region', sortable: true, formatter: \"default('—')\" },\n { key: 'country_name', label: 'Country', sortable: true, formatter: \"default('—')\" },\n { key: 'isp', label: 'ISP', sortable: true, formatter: \"default('—')\" },\n { key: 'threat_level', label: 'Threat', formatter: \"default('—')\"}\n ],\n\n // Table features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Actions\n // actions: ['view', 'edit', 'delete'],\n clickAction: 'view',\n\n // Toolbar\n showRefresh: true,\n showAdd: true,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No GeoIP records found.',\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n },\n tableViewOptions: {\n addButtonLabel: \"Lookup IP\",\n onAdd: (evt) => {\n evt.preventDefault();\n // Implement the logic for adding a new record\n this.onLookup();\n }\n }\n });\n }\n\n async onLookup() {\n // Implement the logic for adding a new record\n const data = await this.getApp().showForm({\n title: \"Lookup IP\",\n fields: [\n {\n name: 'ip',\n type: 'text',\n required: true\n }\n ]\n });\n if (data && data.ip) {\n const model = await GeoLocatedIP.lookup(data.ip);\n if (model) {\n this.tableView._onRowView({ model });\n }\n }\n }\n}\n\nexport default GeoLocatedIPTablePage;\n","import Collection from '@core/Collection.js';\nimport Model from '@core/Model.js';\n\n/**\n * ApiKey - Group-scoped API key for external integrations and services.\n * Maps to REST endpoints under /api/group/apikey\n *\n * Key properties:\n * - Scoped to a single group\n * - Carries only explicitly granted permissions (least-privilege)\n * - sys.* permissions always denied\n * - No IP restriction (unlike User Auth Tokens)\n * - Header format: Authorization: apikey <token>\n *\n * The raw token is only returned at creation time — it is never shown again.\n *\n * Endpoints:\n * GET /api/group/apikey - List keys (filter by ?group=<id>)\n * POST /api/group/apikey - Create a key\n * GET /api/group/apikey/<id> - Get key details\n * POST /api/group/apikey/<id> - Update name, permissions, limits, is_active\n * DELETE /api/group/apikey/<id> - Delete key\n */\nclass ApiKey extends Model {\n constructor(data = {}, options = {}) {\n super(data, {\n endpoint: '/api/group/apikey',\n ...options\n });\n }\n}\n\n/**\n * ApiKeyList - Collection of ApiKey records.\n * Filter by group: new ApiKeyList({ params: { group: groupId } })\n */\nclass ApiKeyList extends Collection {\n constructor(options = {}) {\n super({\n ModelClass: ApiKey,\n endpoint: '/api/group/apikey',\n size: 25,\n ...options\n });\n }\n}\n\n/**\n * Forms configuration for ApiKey\n */\nconst ApiKeyForms = {\n create: {\n title: 'Create API Key',\n fields: [\n {\n name: 'name',\n type: 'text',\n label: 'Name',\n placeholder: 'Mobile App v2',\n required: true,\n columns: 12,\n help: 'A descriptive name to identify this key.'\n },\n {\n name: 'group',\n type: 'number',\n label: 'Group ID',\n required: true,\n columns: 12,\n help: 'The group this key is scoped to.'\n },\n {\n name: 'permissions',\n type: 'textarea',\n label: 'Permissions (JSON)',\n placeholder: '{\"view_orders\": true, \"create_orders\": true}',\n columns: 12,\n help: 'JSON dict of permissions to grant. Leave empty for no permissions.'\n }\n ]\n },\n\n edit: {\n title: 'Edit API Key',\n fields: [\n {\n name: 'name',\n type: 'text',\n label: 'Name',\n required: true,\n columns: 12\n },\n {\n name: 'is_active',\n type: 'switch',\n label: 'Active',\n columns: 12,\n help: 'Deactivate to revoke access without deleting the key.'\n },\n {\n name: 'permissions',\n type: 'textarea',\n label: 'Permissions (JSON)',\n columns: 12,\n help: 'JSON dict of granted permissions.'\n }\n ]\n }\n};\n\nexport { ApiKey, ApiKeyList, ApiKeyForms };\n","/**\n * ApiKeyView - Group-scoped API key detail and management interface\n *\n * Shows key metadata, permissions, and provides actions to edit, toggle\n * active state, and delete. The raw token is only displayed at creation\n * time and is not shown here.\n */\n\nimport View from '@core/View.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport { ApiKey, ApiKeyForms } from '@core/models/ApiKey.js';\n\nclass ApiKeyView extends View {\n constructor(options = {}) {\n super({\n className: 'api-key-view',\n ...options\n });\n\n this.model = options.model || new ApiKey(options.data || {});\n\n this.template = `\n <div class=\"api-key-view-container\">\n <!-- Header -->\n <div class=\"d-flex justify-content-between align-items-start mb-4\">\n <!-- Left: Icon & Identity -->\n <div class=\"d-flex align-items-center gap-3\">\n <div class=\"fs-1 text-primary\">\n <i class=\"bi bi-key\"></i>\n </div>\n <div>\n <h3 class=\"mb-1\">{{model.name|default('Unnamed Key')}}</h3>\n <div class=\"text-muted small\">\n ID: {{model.id}}\n <span class=\"mx-2\">|</span>\n Group: {{model.group.name|default(model.group)}}\n </div>\n <div class=\"mt-1\">\n <span class=\"badge {{model.is_active|boolean('bg-success','bg-secondary')}}\">\n {{model.is_active|boolean('Active','Inactive')}}\n </span>\n </div>\n </div>\n </div>\n\n <!-- Right: Meta & Actions -->\n <div class=\"d-flex align-items-start gap-4\">\n <div class=\"text-end\">\n <div class=\"text-muted small\">Created</div>\n <div>{{model.created|datetime}}</div>\n </div>\n <div data-container=\"apikey-context-menu\"></div>\n </div>\n </div>\n\n <!-- Details -->\n <div class=\"list-group mb-3\">\n <div class=\"list-group-item\">\n <h6 class=\"mb-1 text-muted\">Token Preview</h6>\n <p class=\"mb-1 font-monospace small text-muted\">\n The raw token is only shown once at creation time.\n </p>\n </div>\n <div class=\"list-group-item\">\n <h6 class=\"mb-1 text-muted\">Permissions</h6>\n {{#model.permissions}}\n <pre class=\"mb-0 small\">{{model.permissions|json}}</pre>\n {{/model.permissions}}\n {{^model.permissions}}\n <span class=\"text-muted small\">No permissions granted</span>\n {{/model.permissions}}\n </div>\n {{#model.limits}}\n <div class=\"list-group-item\">\n <h6 class=\"mb-1 text-muted\">Rate Limit Overrides</h6>\n <pre class=\"mb-0 small\">{{model.limits|json}}</pre>\n </div>\n {{/model.limits}}\n <div class=\"list-group-item\">\n <h6 class=\"mb-1 text-muted\">Usage</h6>\n <p class=\"mb-0 small text-muted\">\n Include in requests as:\n <code>Authorization: apikey &lt;token&gt;</code>\n </p>\n </div>\n </div>\n </div>\n `;\n }\n\n async onInit() {\n const isActive = this.model.get('is_active');\n\n const apiKeyMenu = new ContextMenu({\n containerId: 'apikey-context-menu',\n className: 'context-menu-view header-menu-absolute',\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: [\n { label: 'Edit', action: 'edit-key', icon: 'bi-pencil' },\n isActive\n ? { label: 'Deactivate', action: 'deactivate-key', icon: 'bi-x-circle' }\n : { label: 'Activate', action: 'activate-key', icon: 'bi-check-circle' },\n { type: 'divider' },\n { label: 'Delete Key', action: 'delete-key', icon: 'bi-trash', danger: true }\n ]\n }\n });\n this.addChild(apiKeyMenu);\n }\n\n async onActionEditKey() {\n const app = this.getApp();\n const resp = await app.showModelForm({\n title: `Edit API Key — ${this.model.get('name')}`,\n model: this.model,\n formConfig: ApiKeyForms.edit,\n });\n if (resp) {\n this.render();\n }\n }\n\n async onActionDeactivateKey() {\n const app = this.getApp();\n const confirmed = await app.confirm({\n title: 'Deactivate API Key',\n message: `Deactivate \"${this.model.get('name')}\"? Requests using this key will be rejected.`,\n confirmLabel: 'Deactivate',\n confirmClass: 'btn-warning'\n });\n if (!confirmed) return;\n\n app.showLoading();\n const resp = await this.model.save({ is_active: false });\n app.hideLoading();\n if (resp && resp.success !== false) {\n app.toast.success('API key deactivated');\n this.render();\n } else {\n app.toast.error('Failed to deactivate key');\n }\n }\n\n async onActionActivateKey() {\n const app = this.getApp();\n app.showLoading();\n const resp = await this.model.save({ is_active: true });\n app.hideLoading();\n if (resp && resp.success !== false) {\n app.toast.success('API key activated');\n this.render();\n } else {\n app.toast.error('Failed to activate key');\n }\n }\n\n async onActionDeleteKey() {\n const app = this.getApp();\n const confirmed = await app.confirm({\n title: 'Delete API Key',\n message: `Permanently delete \"${this.model.get('name')}\"? This cannot be undone.`,\n confirmLabel: 'Delete',\n confirmClass: 'btn-danger'\n });\n if (!confirmed) return;\n\n app.showLoading();\n const resp = await this.model.delete();\n app.hideLoading();\n if (resp && resp.success !== false) {\n app.toast.success('API key deleted');\n this.emit('deleted', { model: this.model });\n } else {\n app.toast.error('Failed to delete key');\n }\n }\n}\n\nApiKey.VIEW_CLASS = ApiKeyView;\nexport default ApiKeyView;\n","/**\n * ApiKeyTablePage - Group API key management using TablePage component\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { ApiKey, ApiKeyList, ApiKeyForms } from '@core/models/ApiKey.js';\nimport ApiKeyView from './ApiKeyView.js';\n\n// Register the add/edit forms on the model class so TableView can find them automatically\nApiKey.ADD_FORM = ApiKeyForms.create;\nApiKey.EDIT_FORM = ApiKeyForms.edit;\n\nclass ApiKeyTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_api_keys',\n pageName: 'API Keys',\n router: 'admin/api-keys',\n Collection: ApiKeyList,\n\n itemViewClass: ApiKeyView,\n viewDialogOptions: {\n header: false,\n size: 'lg'\n },\n\n columns: [\n { key: 'id', label: 'ID', width: '70px', sortable: true, class: 'text-muted' },\n { key: 'name', label: 'Name', sortable: true },\n { key: 'group.name', label: 'Group', sortable: true, formatter: \"default('—')\" },\n {\n key: 'is_active',\n label: 'Status',\n formatter: \"boolean('Active|bg-success','Inactive|bg-secondary')|badge\",\n width: '100px'\n },\n { key: 'created', label: 'Created', formatter: 'datetime', sortable: true }\n ],\n\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n showRefresh: true,\n showAdd: true,\n showExport: false,\n\n addButtonLabel: 'New API Key',\n\n emptyMessage: 'No API keys found.',\n\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n }\n });\n }\n\n // Override to intercept and show the one-time token after model.save()\n async onActionAdd() {\n const app = this.getApp();\n const model = new ApiKey();\n const result = await app.showForm({\n model,\n ...ApiKeyForms.create\n });\n if (!result) return;\n\n const resp = await model.save(result);\n if (!resp?.data?.status) {\n app.showError(resp?.data?.error || 'Failed to create API key');\n return;\n }\n\n // Token is only present in the creation response — show it once\n const token = resp.data?.data?.token;\n await app.showAlert({\n title: 'API Key Created — Save Your Token',\n message: token\n ? `Copy this token now. It will not be shown again.\\n\\n${token}`\n : 'API key created successfully.',\n type: token ? 'warning' : 'success',\n size: 'lg'\n });\n\n this.collection.add(model);\n this.tableView?.refresh();\n }\n}\n\nexport default ApiKeyTablePage;\n","/**\n * CloudWatchChart - MetricsChart configured for CloudWatch endpoints\n *\n * Extends MetricsChart with:\n * - CloudWatch endpoint defaults (/api/aws/cloudwatch/fetch)\n * - `stat` parameter support (avg, max, min, sum)\n * - Response format normalization (periods → labels, {slug,values} → {slug: values})\n *\n * NOTE: The CloudWatch API currently returns a slightly different format\n * than the standard metrics API (periods vs labels, array vs dict).\n * processMetricsData normalizes this until the backend aligns.\n */\nimport MetricsChart from '@ext/charts/MetricsChart.js';\n\nexport default class CloudWatchChart extends MetricsChart {\n constructor(options = {}) {\n super({\n endpoint: '/api/aws/cloudwatch/fetch',\n account: options.resourceType || options.account || 'ec2',\n category: options.category || null,\n slugs: options.slugs || (options.slug ? [options.slug] : null),\n granularity: options.granularity || 'hours',\n title: options.title || 'CloudWatch',\n defaultDateRange: options.defaultDateRange || '24h',\n showDateRange: false,\n ...options\n });\n\n this.stat = options.stat || 'avg';\n this.resourceType = options.resourceType || options.account || 'ec2';\n }\n\n buildApiParams() {\n const params = super.buildApiParams();\n // CloudWatch uses 'stat' parameter\n params.stat = this.stat;\n return params;\n }\n\n setStat(stat) {\n this.stat = stat;\n return this.fetchData();\n }\n}\n","/**\n * CloudWatchDashboardPage - AWS CloudWatch monitoring dashboard\n *\n * 2-column grid of MetricsCharts showing key metrics across all resources.\n * Each chart auto-plots all instances for its account+category.\n * Uses /api/aws/cloudwatch/fetch via MetricsChart.\n */\nimport Page from '@core/Page.js';\nimport CloudWatchChart from './CloudWatchChart.js';\n\nconst DASHBOARD_CHARTS = [\n { account: 'ec2', category: 'cpu', title: 'EC2 CPU', unit: '%' },\n { account: 'ec2', category: 'net_out', title: 'EC2 Network Out', unit: 'bytes' },\n { account: 'ec2', category: 'memory', title: 'EC2 Memory', unit: '%' },\n { account: 'ec2', category: 'disk', title: 'EC2 Disk', unit: '%' },\n { account: 'rds', category: 'cpu', title: 'RDS CPU', unit: '%' },\n { account: 'rds', category: 'conns', title: 'RDS Connections', unit: '' },\n { account: 'rds', category: 'read_latency', title: 'RDS Read Latency', unit: 's' },\n { account: 'rds', category: 'write_latency', title: 'RDS Write Latency', unit: 's' },\n { account: 'redis', category: 'cpu', title: 'Redis CPU', unit: '%' },\n { account: 'redis', category: 'conns', title: 'Redis Connections', unit: '' },\n { account: 'redis', category: 'cache_misses', title: 'Redis Cache Misses', unit: '' },\n { account: 'redis', category: 'cache_hits', title: 'Redis Cache Hits', unit: '' }\n];\n\nfunction yAxisForUnit(unit) {\n if (unit === '%') return { label: '%', beginAtZero: true, max: 100 };\n if (unit === 'bytes') return { label: 'Bytes', beginAtZero: true };\n if (unit === 's') return { label: 'Seconds', beginAtZero: true };\n return { beginAtZero: true };\n}\n\nexport default class CloudWatchDashboardPage extends Page {\n constructor(options = {}) {\n super({\n ...options,\n title: 'CloudWatch Monitoring',\n className: 'cloudwatch-dashboard-page'\n });\n }\n\n async getTemplate() {\n return `\n <style>\n .cw-grid { display: grid; grid-template-columns: 1fr 1fr; gap: 1rem; }\n @media (max-width: 992px) { .cw-grid { grid-template-columns: 1fr; } }\n </style>\n <div class=\"container-fluid\">\n <p class=\"text-muted mb-3\">AWS CloudWatch resource monitoring</p>\n <div class=\"cw-grid\" id=\"cw-grid\">\n ${DASHBOARD_CHARTS.map((_, i) => `<div id=\"cw-chart-${i}\"></div>`).join('')}\n </div>\n </div>\n `;\n }\n\n async onInit() {\n this.getApp()?.showLoading('Loading CloudWatch...');\n try {\n for (let i = 0; i < DASHBOARD_CHARTS.length; i++) {\n const def = DASHBOARD_CHARTS[i];\n const chart = new CloudWatchChart({\n containerId: `cw-chart-${i}`,\n account: def.account,\n category: def.category,\n title: def.title,\n height: 160,\n yAxis: yAxisForUnit(def.unit),\n responsive: true,\n showGranularity: true,\n showDateRange: true,\n defaultDateRange: '24h',\n granularity: 'hours'\n });\n this.addChild(chart);\n }\n } finally {\n this.getApp()?.hideLoading();\n }\n }\n}\n","import Page from '@core/Page.js';\nimport View from '@core/View.js';\nimport {\n MetricsChart,\n MetricsMiniChartWidget\n} from '@ext/charts/index.js';\nimport TableView from '@core/views/table/TableView.js';\nimport {\n IncidentList,\n IncidentStats\n} from '@core/models/Incident.js';\nimport {\n TicketList\n} from '@core/models/Tickets.js';\nimport { MetricsCountryMapView } from '@ext/map/index.js';\n\nclass IncidentDashboardHeader extends View {\n constructor(options = {}) {\n super({\n ...options,\n className: 'incident-dashboard-header'\n });\n\n this.model = new IncidentStats();\n }\n\n async getTemplate() {\n return `\n <div class=\"row\">\n <div class=\"col-xl-3 col-lg-6 col-12 mb-3\">\n <div class=\"card h-100 border-0 shadow-sm\">\n <div class=\"card-body\">\n <div class=\"d-flex justify-content-between align-items-start\">\n <div>\n <h6 class=\"card-title text-muted mb-2\">Open Incidents</h6>\n <h3 class=\"mb-1 fw-bold\">{{model.incidents.open}}</h3>\n <span class=\"badge bg-danger-subtle text-danger\">{{model.incidents.new}} New</span>\n </div>\n <div class=\"text-danger\">\n <i class=\"bi bi-exclamation-triangle fs-2\"></i>\n </div>\n </div>\n </div>\n </div>\n </div>\n <div class=\"col-xl-3 col-lg-6 col-12 mb-3\">\n <div class=\"card h-100 border-0 shadow-sm\">\n <div class=\"card-body\">\n <div class=\"d-flex justify-content-between align-items-start\">\n <div>\n <h6 class=\"card-title text-muted mb-2\">Open Tickets</h6>\n <h3 class=\"mb-1 fw-bold\">{{model.tickets.open}}</h3>\n <span class=\"badge bg-warning-subtle text-warning\">{{model.tickets.new}} New</span>\n </div>\n <div class=\"text-warning\">\n <i class=\"bi bi-ticket-perforated fs-2\"></i>\n </div>\n </div>\n </div>\n </div>\n </div>\n <div class=\"col-xl-3 col-lg-6 col-12 mb-3\">\n <div class=\"card h-100 border-0 shadow-sm\">\n <div class=\"card-body\">\n <div class=\"d-flex justify-content-between align-items-start\">\n <div>\n <h6 class=\"card-title text-muted mb-2\">Recent Events</h6>\n <h3 class=\"mb-1 fw-bold\">{{model.events.recent}}</h3>\n <span class=\"badge bg-info-subtle text-info\">{{model.events.critical}} Critical</span>\n </div>\n <div class=\"text-info\">\n <i class=\"bi bi-activity fs-2\"></i>\n </div>\n </div>\n </div>\n </div>\n </div>\n <div class=\"col-xl-3 col-lg-6 col-12 mb-3\">\n <div class=\"card h-100 border-0 shadow-sm\">\n <div class=\"card-body\">\n <div class=\"d-flex justify-content-between align-items-start\">\n <div>\n <h6 class=\"card-title text-muted mb-2\">Recent Incidents</h6>\n <h3 class=\"mb-1 fw-bold\">{{model.incidents.recent}}</h3>\n <span class=\"badge bg-secondary-subtle text-secondary\">Last 24h</span>\n </div>\n <div class=\"text-secondary\">\n <i class=\"bi bi-clock-history fs-2\"></i>\n </div>\n </div>\n </div>\n </div>\n </div>\n </div>\n `;\n }\n\n async onBeforeRender() {\n await this.model.fetch();\n }\n}\n\n\nclass IncidentDashboardPage extends Page {\n constructor(options = {}) {\n super({\n ...options,\n title: 'Incidents Dashboard',\n className: 'incident-dashboard-page'\n });\n }\n\n async getTemplate() {\n return `\n <div class=\"container-fluid incident-dashboard\">\n <div class=\"d-flex justify-content-between align-items-center mb-3\">\n <div>\n <p class=\"text-muted mb-0\">Incident Intelligence Hub</p>\n <small class=\"text-info\">\n <i class=\"bi bi-activity me-1\"></i>\n Real-time visibility into incidents, tickets, and correlated events\n </small>\n </div>\n <div class=\"btn-group\" role=\"group\">\n <button type=\"button\"\n class=\"btn btn-outline-secondary btn-sm\"\n data-action=\"refresh-all\"\n title=\"Refresh dashboards\">\n <i class=\"bi bi-arrow-clockwise\"></i> Refresh\n </button>\n </div>\n </div>\n\n <div data-container=\"header\" class=\"mb-4\"></div>\n\n <div class=\"row g-4\">\n <div class=\"col-xl-6 col-lg-12\" data-container=\"events-widget\"></div>\n <div class=\"col-xl-6 col-lg-12\" data-container=\"incidents-widget\"></div>\n </div>\n\n\n <div class=\"row g-4 mt-1\">\n <div class=\"col-12\">\n <div class=\"card shadow-sm h-100\">\n <div class=\"card-header border-0 bg-transparent d-flex align-items-center justify-content-between\">\n <div>\n <h6 class=\"mb-0 text-uppercase small text-muted\">Global Event Hotspots</h6>\n <span class=\"text-muted small\">Clusters sized by total events</span>\n </div>\n <span class=\"badge bg-info-subtle text-info\">Interactive</span>\n </div>\n <div class=\"card-body p-0\" data-container=\"events-country-map\"></div>\n </div>\n </div>\n </div>\n\n\n <div class=\"row g-4 mt-1\">\n <div class=\"col-lg-6\">\n <div class=\"card shadow-sm h-100\">\n <div class=\"card-header border-0 bg-transparent d-flex align-items-center justify-content-between\">\n <div>\n <h6 class=\"mb-0 text-uppercase small text-muted\">Events by Country</h6>\n <span class=\"text-muted small\">Hotspots from the last 24 hours</span>\n </div>\n <span class=\"badge bg-success-subtle text-success\">Live</span>\n </div>\n <div class=\"card-body p-3\">\n <div data-container=\"events-by-country-chart\"></div>\n </div>\n </div>\n </div>\n <div class=\"col-lg-6\">\n <div class=\"card shadow-sm h-100\">\n <div class=\"card-header border-0 bg-transparent d-flex align-items-center justify-content-between\">\n <div>\n <h6 class=\"mb-0 text-uppercase small text-muted\">Incidents by Country</h6>\n <span class=\"text-muted small\">Highest volume regions</span>\n </div>\n <span class=\"badge bg-warning-subtle text-warning\">24h</span>\n </div>\n <div class=\"card-body p-3\">\n <div data-container=\"incidents-by-country-chart\"></div>\n </div>\n </div>\n </div>\n </div>\n\n <div class=\"row g-4 mt-1\">\n <div class=\"col-lg-6\">\n <div class=\"card shadow-sm h-100\">\n <div class=\"card-header border-0 bg-transparent d-flex align-items-center justify-content-between\">\n <div>\n <h6 class=\"mb-0 text-uppercase small text-muted\">New Tickets</h6>\n <span class=\"text-muted small\">Fresh tickets awaiting triage</span>\n </div>\n <i class=\"bi bi-ticket-perforated text-muted\"></i>\n </div>\n <div class=\"card-body\" data-container=\"my-tickets-table\"></div>\n </div>\n </div>\n <div class=\"col-lg-6\">\n <div class=\"card shadow-sm h-100\">\n <div class=\"card-header border-0 bg-transparent d-flex align-items-center justify-content-between\">\n <div>\n <h6 class=\"mb-0 text-uppercase small text-muted\">New Incidents</h6>\n <span class=\"text-muted small\">Newest incidents in the queue</span>\n </div>\n <i class=\"bi bi-flag text-warning\"></i>\n </div>\n <div class=\"card-body\" data-container=\"high-priority-incidents-table\"></div>\n </div>\n </div>\n </div>\n </div>\n `;\n }\n\n async onInit() {\n this.header = new IncidentDashboardHeader({\n containerId: 'header'\n });\n this.addChild(this.header);\n\n this.eventsWidget = new MetricsMiniChartWidget({\n containerId: 'events-widget',\n icon: 'bi bi-activity fs-2',\n title: 'System Events',\n subtitle: '{{now_value}} <span class=\"subtitle-label\">{{now_label}}</span>',\n background: '#154360',\n textColor: '#FFFFFF',\n endpoint: '/api/metrics/fetch',\n granularity: 'days',\n slugs: ['incident_events'],\n account: 'incident',\n chartType: 'line',\n showTooltip: true,\n showXAxis: false,\n height: 140,\n color: 'rgba(255,255,255,0.9)',\n fill: true,\n fillColor: 'rgba(255,255,255,0.2)',\n smoothing: 0.3,\n defaultDateRange: '7d',\n valueFormat: 'number',\n showTrending: true,\n showSettings: true,\n settingsKey: 'incident-dashboard-events'\n });\n this.addChild(this.eventsWidget);\n\n this.incidentsWidget = new MetricsMiniChartWidget({\n containerId: 'incidents-widget',\n icon: 'bi bi-exclamation-triangle fs-2',\n title: 'System Incidents',\n subtitle: '{{now_value}} <span class=\"subtitle-label\">{{now_label}}</span>',\n background: '#7D6608',\n textColor: '#FFFFFF',\n endpoint: '/api/metrics/fetch',\n granularity: 'days',\n slugs: ['incidents'],\n account: 'incident',\n chartType: 'line',\n showTooltip: true,\n showXAxis: false,\n height: 140,\n color: 'rgba(255,255,255,0.9)',\n fill: true,\n fillColor: 'rgba(255,255,255,0.25)',\n smoothing: 0.3,\n defaultDateRange: '7d',\n valueFormat: 'number',\n showTrending: true,\n showSettings: true,\n settingsKey: 'incident-dashboard-incidents'\n });\n this.addChild(this.incidentsWidget);\n\n this.eventsByCountryChart = new MetricsChart({\n title: '<i class=\"bi bi-globe-central-south-asia me-2\"></i> Events by Country',\n endpoint: '/api/metrics/fetch',\n account: 'incident',\n category: 'incident_events_by_country',\n granularity: 'days',\n chartType: 'line',\n showDateRange: false,\n showMetricsFilter: false,\n height: 220,\n maxDatasets: 10,\n colors: [\n 'rgba(32, 201, 151, 0.85)'\n ],\n yAxis: {\n label: 'Events',\n beginAtZero: true\n },\n tooltip: { y: 'number:0' },\n containerId: 'events-by-country-chart'\n });\n this.addChild(this.eventsByCountryChart);\n\n this.incidentsByCountryChart = new MetricsChart({\n title: '<i class=\"bi bi-geo-alt me-2\"></i> Incidents by Country',\n endpoint: '/api/metrics/fetch',\n account: 'incident',\n category: 'incidents_by_country',\n granularity: 'days',\n chartType: 'line',\n showDateRange: false,\n showMetricsFilter: false,\n height: 220,\n maxDatasets: 10,\n colors: [\n 'rgba(255, 193, 7, 0.85)'\n ],\n yAxis: {\n label: 'Incidents',\n beginAtZero: true\n },\n tooltip: { y: 'number:0' },\n containerId: 'incidents-by-country-chart'\n });\n this.addChild(this.incidentsByCountryChart);\n\n this.eventsCountryMap = new MetricsCountryMapView({\n containerId: 'events-country-map',\n category: 'incident_events_by_country',\n account: 'incident',\n maxCountries: 20,\n metricLabel: 'Events',\n height: 360,\n mapStyle: 'dark'\n });\n this.addChild(this.eventsCountryMap);\n\n const myTicketsCollection = new TicketList({\n params: {\n status: 'new'\n }\n });\n this.myTicketsTable = new TableView({\n containerId: 'my-tickets-table',\n title: 'New Tickets',\n collection: myTicketsCollection,\n columns: [\n { key: 'id', label: 'ID' },\n { key: 'title', label: 'Title' },\n { key: 'priority', label: 'Priority' }\n ]\n });\n this.addChild(this.myTicketsTable);\n\n const newIncidentsCollection = new IncidentList({\n params: {\n status: 'new'\n }\n });\n this.highPriorityIncidentsTable = new TableView({\n containerId: 'high-priority-incidents-table',\n title: 'New Incidents',\n collection: newIncidentsCollection,\n columns: [\n { key: 'id', label: 'ID' },\n { key: 'title', label: 'Title' },\n { key: 'status', label: 'Status', formatter: 'badge' }\n ]\n });\n this.addChild(this.highPriorityIncidentsTable);\n }\n\n async onActionRefreshAll(event, element) {\n const button = element || event?.currentTarget || null;\n const icon = button?.querySelector?.('i');\n icon?.classList.add('bi-spin');\n if (button) button.disabled = true;\n\n const refreshTasks = [\n this.header?.model?.fetch()?.then(() => this.header.render()),\n this.eventsWidget?.refresh(),\n this.incidentsWidget?.refresh(),\n this.eventsByCountryChart?.refresh(),\n this.incidentsByCountryChart?.refresh(),\n this.eventsCountryMap?.refresh(),\n this.myTicketsTable?.collection?.fetch(),\n this.highPriorityIncidentsTable?.collection?.fetch()\n ].filter(Boolean);\n\n await Promise.allSettled(refreshTasks);\n\n icon?.classList.remove('bi-spin');\n if (button) button.disabled = false;\n }\n}\n\nexport default IncidentDashboardPage;\n","/**\n * StackTraceView - Display formatted and color-coded stack traces\n */\n\nimport View from '@core/View.js';\n\nclass StackTraceView extends View {\n constructor(options = {}) {\n super({\n className: 'stack-trace-view',\n ...options\n });\n\n this.stackTrace = options.stackTrace || '';\n \n this.template = `\n <div class=\"stack-trace-container p-3\">\n <style>\n .stack-trace-line {\n font-family: 'Monaco', 'Menlo', 'Ubuntu Mono', 'Consolas', monospace;\n font-size: 13px;\n line-height: 1.6;\n padding: 4px 8px;\n margin: 0;\n border-left: 3px solid transparent;\n }\n .stack-trace-line:hover {\n background-color: rgba(0, 0, 0, 0.05);\n }\n .stack-trace-error {\n color: #dc3545;\n font-weight: 600;\n border-left-color: #dc3545;\n background-color: rgba(220, 53, 69, 0.05);\n }\n .stack-trace-file {\n color: #0d6efd;\n font-weight: 500;\n border-left-color: #0d6efd;\n }\n .stack-trace-function {\n color: #6610f2;\n font-weight: 500;\n }\n .stack-trace-location {\n color: #6c757d;\n font-size: 12px;\n }\n .stack-trace-line-number {\n color: #fd7e14;\n font-weight: 600;\n }\n .stack-trace-context {\n color: #495057;\n background-color: rgba(0, 0, 0, 0.02);\n }\n .stack-trace-container {\n background-color: #f8f9fa;\n border: 1px solid #dee2e6;\n border-radius: 0.375rem;\n max-height: 600px;\n overflow-y: auto;\n }\n </style>\n <div class=\"stack-trace-content\">\n {{{formattedStackTrace}}}\n </div>\n </div>\n `;\n }\n\n async onBeforeRender() {\n this.formattedStackTrace = this.formatStackTrace(this.stackTrace);\n }\n\n formatStackTrace(stackTrace) {\n if (!stackTrace) {\n return '<div class=\"text-muted p-3\">No stack trace available</div>';\n }\n\n // Convert to string if it's an object\n const traceStr = typeof stackTrace === 'string' ? stackTrace : JSON.stringify(stackTrace, null, 2);\n \n const lines = traceStr.split('\\n');\n let html = '';\n\n lines.forEach((line, index) => {\n if (!line.trim()) {\n html += '<div class=\"stack-trace-line\">&nbsp;</div>';\n return;\n }\n\n // Detect error message (usually the first line)\n if (index === 0 && (line.includes('Error:') || line.includes('Exception:'))) {\n html += `<div class=\"stack-trace-line stack-trace-error\">${this.escapeHtml(line)}</div>`;\n return;\n }\n\n // Detect file paths with line numbers\n // Pattern: at functionName (file.js:123:45) or file.js:123:45\n const filePattern = /(.+?)\\s*\\(([^:]+):(\\d+):(\\d+)\\)/;\n const simpleFilePattern = /^\\s*at\\s+([^:]+):(\\d+):(\\d+)/;\n \n let match = line.match(filePattern);\n if (match) {\n const [, funcName, filePath, lineNum, colNum] = match;\n html += `<div class=\"stack-trace-line stack-trace-file\">\n <span class=\"stack-trace-function\">${this.escapeHtml(funcName.trim())}</span>\n <span class=\"stack-trace-location\"> (${this.escapeHtml(filePath)}:<span class=\"stack-trace-line-number\">${lineNum}</span>:${colNum})</span>\n </div>`;\n return;\n }\n\n match = line.match(simpleFilePattern);\n if (match) {\n const [, filePath, lineNum, colNum] = match;\n html += `<div class=\"stack-trace-line stack-trace-file\">\n <span class=\"stack-trace-location\">at ${this.escapeHtml(filePath)}:<span class=\"stack-trace-line-number\">${lineNum}</span>:${colNum}</span>\n </div>`;\n return;\n }\n\n // Python-style stack trace: File \"...\", line X, in function\n const pythonPattern = /File\\s+\"([^\"]+)\",\\s+line\\s+(\\d+),\\s+in\\s+(.+)/;\n match = line.match(pythonPattern);\n if (match) {\n const [, filePath, lineNum, funcName] = match;\n html += `<div class=\"stack-trace-line stack-trace-file\">\n <span class=\"stack-trace-location\">File \"${this.escapeHtml(filePath)}\", line <span class=\"stack-trace-line-number\">${lineNum}</span>, in </span>\n <span class=\"stack-trace-function\">${this.escapeHtml(funcName)}</span>\n </div>`;\n return;\n }\n\n // Check if line starts with \"at \" (stack frame indicator)\n if (line.trim().startsWith('at ')) {\n html += `<div class=\"stack-trace-line stack-trace-file\">${this.escapeHtml(line)}</div>`;\n return;\n }\n\n // Default: context line\n html += `<div class=\"stack-trace-line stack-trace-context\">${this.escapeHtml(line)}</div>`;\n });\n\n return html;\n }\n\n updateStackTrace(newStackTrace) {\n this.stackTrace = newStackTrace;\n this.render();\n }\n}\n\nexport default StackTraceView;\n","import { IncidentHistory, IncidentHistoryList } from '@core/models/Incident.js';\n\nclass IncidentHistoryAdapter {\n constructor(incidentId) {\n this.incidentId = incidentId;\n this.collection = new IncidentHistoryList({ params: { incident: this.incidentId } });\n }\n\n async fetch() {\n await this.collection.fetch();\n return this.collection.models.map(item => this.transform(item));\n }\n\n transform(item) {\n return {\n id: item.get('id'),\n type: item.get('kind') === 'comment' ? 'user_comment' : 'system_event',\n author: {\n name: item.get('by.display_name') || 'System',\n avatarUrl: item.get('by.avatar.url')\n },\n timestamp: item.get('created'),\n content: item.get('note'),\n attachments: [] // Incident history doesn't have attachments in this phase\n };\n }\n\n async addNote(data) {\n const history = new IncidentHistory({\n incident: this.incidentId,\n note: data.text,\n kind: 'comment'\n });\n const resp = await history.save();\n if (resp.success) {\n await this.collection.fetch();\n }\n return resp;\n }\n}\n\nexport default IncidentHistoryAdapter;\n","/**\n * IncidentView - Detailed view for an Incident record\n */\n\nimport View from '@core/View.js';\nimport TabView from '@core/views/navigation/TabView.js';\nimport DataView from '@core/views/data/DataView.js';\nimport TableView from '@core/views/table/TableView.js';\nimport StackTraceView from '@core/views/data/StackTraceView.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport { Incident, IncidentForms, IncidentEventList } from '@core/models/Incident.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\nimport IncidentHistoryAdapter from './adapters/IncidentHistoryAdapter.js';\nimport ChatView from '@core/views/chat/ChatView.js';\n\nclass IncidentView extends View {\n constructor(options = {}) {\n super({\n className: 'incident-view',\n ...options\n });\n\n this.model = options.model || new Incident(options.data || {});\n this.incidentIcon = this.getIconForIncident(this.model.get('state'));\n\n this.template = `\n <div class=\"incident-view-container\">\n <!-- Header -->\n <div class=\"d-flex justify-content-between align-items-center mb-4\">\n <div class=\"d-flex align-items-center gap-3\">\n <div class=\"fs-1 {{incidentIcon.color}}\">\n <i class=\"bi {{incidentIcon.icon}}\"></i>\n </div>\n <div>\n <h3 class=\"mb-1\">Incident #{{model.id}}</h3>\n <div class=\"text-muted small\">\n Category: {{model.category|capitalize}}\n </div>\n <div class=\"text-muted small mt-1\">\n Created: {{model.created|datetime}}\n </div>\n </div>\n </div>\n <div class=\"d-flex align-items-center gap-4\">\n <div class=\"text-end\">\n <div>State: <span class=\"badge bg-primary\">{{model.state|capitalize}}</span></div>\n <div class=\"text-muted small mt-1\">Priority: {{model.priority}}</div>\n </div>\n <div data-container=\"incident-context-menu\"></div>\n </div>\n </div>\n\n <!-- Tabs -->\n <div data-container=\"incident-tabs\"></div>\n </div>\n `;\n }\n\n getIconForIncident(state) {\n const s = state?.toLowerCase();\n if (s === 'resolved' || s === 'closed') return { icon: 'bi-check-circle-fill', color: 'text-success' };\n if (s === 'new' || s === 'opened') return { icon: 'bi-exclamation-triangle-fill', color: 'text-danger' };\n if (s === 'paused' || s === 'ignore') return { icon: 'bi-pause-circle-fill', color: 'text-warning' };\n return { icon: 'bi-shield-exclamation', color: 'text-secondary' };\n }\n\n async onInit() {\n // Overview Tab\n this.overviewView = new DataView({\n model: this.model,\n className: \"p-3\",\n columns: 2,\n fields: [\n { name: 'id', label: 'Incident ID' },\n { name: 'state', label: 'State', format: 'badge' },\n { name: 'priority', label: 'Priority' },\n { name: 'category', label: 'Category' },\n { name: 'model_name', label: 'Related Model' },\n { name: 'model_id', label: 'Related Model ID' },\n { name: 'details', label: 'Details', columns: 12, format: 'pre' },\n ]\n });\n\n // Events Tab\n const eventsCollection = new IncidentEventList({\n params: { incident: this.model.get('id') }\n });\n this.eventsView = new TableView({\n collection: eventsCollection,\n hideActivePillNames: ['incident'],\n columns: [\n { key: 'id', label: 'ID', width: '70px', sortable: true },\n { key: 'created', label: 'Date', formatter: 'datetime', sortable: true, width: '180px' },\n { key: 'category', label: 'Category', formatter: 'badge', sortable: true },\n { key: 'title', label: 'Title', sortable: true },\n { key: 'level', label: 'Level', sortable: true, width: '80px' },\n ],\n showAdd: false,\n actions: ['view'],\n paginated: true,\n size: 10\n });\n\n // History & Comments Tab\n const adapter = new IncidentHistoryAdapter(this.model.get('id'));\n this.historyView = new ChatView({ adapter });\n\n const tabs = { \n 'Overview': this.overviewView,\n 'Events': this.eventsView,\n 'History & Comments': this.historyView\n };\n \n const metadata = this.model.get('metadata') || {};\n \n // Add Stack Trace tab if present\n if (metadata.stack_trace) {\n this.stackTraceView = new StackTraceView({\n stackTrace: metadata.stack_trace\n });\n tabs['Stack Trace'] = this.stackTraceView;\n }\n \n // Add Metadata tab if there's metadata\n if (Object.keys(metadata).length > 0) {\n this.metadataView = new View({\n model: this.model,\n template: `<pre class=\"bg-light p-3 border rounded\"><code>{{{model.metadata|json}}}</code></pre>`\n });\n tabs['Metadata'] = this.metadataView;\n }\n\n this.tabView = new TabView({\n containerId: 'incident-tabs',\n tabs: tabs,\n activeTab: 'Overview'\n });\n this.addChild(this.tabView);\n\n // ContextMenu\n const incidentMenu = new ContextMenu({\n containerId: 'incident-context-menu',\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: [\n { label: 'Edit Incident', action: 'edit-incident', icon: 'bi-pencil' },\n { label: 'Resolve', action: 'resolve-incident', icon: 'bi-check-circle' },\n { type: 'divider' },\n { label: 'Delete Incident', action: 'delete-incident', icon: 'bi-trash', danger: true }\n ]\n }\n });\n this.addChild(incidentMenu);\n }\n\n async onActionEditIncident() {\n const resp = await Dialog.showModelForm({\n title: `Edit Incident #${this.model.id}`,\n model: this.model,\n formConfig: IncidentForms.edit,\n });\n if (resp) {\n this.render();\n }\n }\n \n async onActionResolveIncident() {\n await this.model.save({ state: 'resolved' });\n this.render();\n this.emit('incident:updated', { model: this.model });\n }\n\n async onActionDeleteIncident() {\n const confirmed = await Dialog.confirm(\n `Are you sure you want to delete this incident?`,\n 'Confirm Deletion',\n { confirmClass: 'btn-danger', confirmText: 'Delete' }\n );\n if (confirmed) {\n const resp = await this.model.destroy();\n if (resp.success) {\n this.emit('incident:deleted', { model: this.model });\n }\n }\n }\n}\n\nIncident.VIEW_CLASS = IncidentView;\n\nexport default IncidentView;\n","/**\n * IncidentTablePage - Incident management using TablePage component\n * Clean implementation using TablePage with minimal overrides\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { IncidentList, IncidentForms } from '@core/models/Incident.js';\nimport IncidentView from './IncidentView.js';\n\nclass IncidentTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_incidents',\n pageName: 'Manage Incidents',\n router: \"admin/incidents\",\n Collection: IncidentList,\n\n formCreate: IncidentForms.create,\n formEdit: IncidentForms.edit,\n itemViewClass: IncidentView,\n\n viewDialogOptions: {\n header: false,\n size: 'xl'\n },\n\n defaultQuery: {\n sort: '-id',\n status: \"new\",\n },\n\n // Column definitions\n columns: [\n {\n key: 'id',\n label: 'ID',\n width: '60px',\n sortable: true,\n class: 'text-muted'\n },\n {\n key: 'status', label: \"Status\",\n filter: {\n type: 'multiselect',\n options: [\"new\", \"open\", \"paused\", \"resolved\", \"qa\", \"ignored\"],\n }\n },\n {\n key: 'created',\n label: 'Created',\n formatter: \"epoch|datetime\",\n filter: {\n type: 'daterange',\n }\n },\n {\n key: 'scope',\n label: 'Scope',\n sortable: true,\n filter: {type:\"text\"}\n },\n {\n key: 'category',\n label: 'Category',\n sortable: true,\n filter: {type:\"text\"}\n },\n {\n key: 'priority',\n label: 'Priority',\n filter: {type:\"text\"}\n },\n {\n key: 'title',\n label: 'title',\n formatter: \"truncate(100)|default('No description')\"\n }\n ],\n\n filters: [\n {\n key: 'category__not',\n label: 'Not Category',\n filter: {type:\"text\"}\n },\n ],\n\n // Table features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Toolbar\n showRefresh: true,\n showAdd: true,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No incidents found. Click \"Add Incident\" to create your first incident.',\n\n // Batch actions\n batchBarLocation: 'top',\n batchActions: [\n { label: \"Open\", icon: \"bi bi-folder2-open\", action: \"open\" },\n { label: \"Resolve\", icon: \"bi bi-check-circle\", action: \"resolve\" },\n { label: \"Pause\", icon: \"bi bi-pause-circle\", action: \"pause\" },\n { label: \"Ignore\", icon: \"bi bi-x-circle\", action: \"ignore\" },\n { label: \"Merge\", icon: \"bi bi-merge\", action: \"merge\" }\n ],\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n }\n });\n }\n\n async onActionBatchResolve(event, element) {\n const selected = this.tableView.getSelectedItems();\n if (!selected.length) return;\n\n const app = this.getApp();\n const result = await app.confirm(`Are you sure you want to close ${selected.length} incidents?`);\n if (!result) return;\n await Promise.all(selected.map(item => item.model.save({status: 'resolved'})));\n this.tableView.collection.fetch();\n }\n\n async onActionBatchOpen(event, element) {\n const selected = this.tableView.getSelectedItems();\n if (!selected.length) return;\n\n const app = this.getApp();\n const result = await app.confirm(`Are you sure you want to open ${selected.length} incidents?`);\n if (!result) return;\n await Promise.all(selected.map(item => item.model.save({status: 'open'})));\n this.tableView.collection.fetch();\n }\n\n async onActionBatchPause(event, element) {\n const selected = this.tableView.getSelectedItems();\n if (!selected.length) return;\n\n const app = this.getApp();\n const result = await app.confirm(`Are you sure you want to pause ${selected.length} incidents?`);\n if (!result) return;\n await Promise.all(selected.map(item => item.model.save({status: 'paused'})));\n this.tableView.collection.fetch();\n }\n\n async onActionBatchIgnore(event, element) {\n const selected = this.tableView.getSelectedItems();\n if (!selected.length) return;\n\n const app = this.getApp();\n const result = await app.confirm(`Are you sure you want to ignore ${selected.length} incidents?`);\n if (!result) return;\n await Promise.all(selected.map(item => item.model.save({status: 'ignored'})));\n this.tableView.collection.fetch();\n }\n\n async onActionBatchMerge(_event, _element) {\n const selected = this.tableView.getSelectedItems();\n if (!selected.length) return;\n\n const app = this.getApp();\n const result = await app.showForm({\n title: `Merge ${selected.length} incidents`,\n fields: [\n {\n name: 'merge',\n type: 'select',\n label: 'Select Parent Incident',\n options: selected.map(item => ({value: item.model.id, label: item.model.id})),\n required: true\n }\n ]\n });\n if (!result) return;\n\n // Find the parent model from selected items\n const parentModel = selected.find(item => item.model.id == result.merge)?.model;\n if (!parentModel) return;\n\n // Get list of all IDs to merge (excluding the parent)\n const mergeIds = selected\n .map(item => item.model.id)\n .filter(id => id != result.merge);\n\n // Save the merge operation to the parent model\n await parentModel.save({ merge: mergeIds });\n this.tableView.collection.fetch();\n }\n}\n\nexport default IncidentTablePage;\n","/**\n * EventView - Detailed view for an IncidentEvent record\n */\n\nimport View from '@core/View.js';\nimport TabView from '@core/views/navigation/TabView.js';\nimport DataView from '@core/views/data/DataView.js';\nimport StackTraceView from '@core/views/data/StackTraceView.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport { IncidentEvent } from '@core/models/Incident.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\n\nclass EventView extends View {\n constructor(options = {}) {\n super({\n className: 'event-view',\n ...options\n });\n\n this.model = options.model || new IncidentEvent(options.data || {});\n this.eventIcon = this.getIconForEvent(this.model.get('level'));\n\n this.template = `\n <div class=\"event-view-container\">\n <!-- Header -->\n <div class=\"d-flex justify-content-between align-items-center mb-4\">\n <div class=\"d-flex align-items-center gap-3\">\n <div class=\"fs-1 {{eventIcon.color}}\">\n <i class=\"bi {{eventIcon.icon}}\"></i>\n </div>\n <div>\n <h3 class=\"mb-1\">{{model.title|default('System Event')}}</h3>\n <div class=\"text-muted small\">\n Category: {{model.category|capitalize}}\n </div>\n <div class=\"text-muted small mt-1\">\n {{model.created|datetime}} from {{model.source_ip|default('Unknown IP')}}\n </div>\n </div>\n </div>\n <div data-container=\"event-context-menu\"></div>\n </div>\n\n <!-- Body -->\n <div data-container=\"event-tabs\"></div>\n </div>\n `;\n }\n\n getIconForEvent(level) {\n if (level >= 40) return { icon: 'bi-exclamation-octagon-fill', color: 'text-danger' }; // Error\n if (level >= 30) return { icon: 'bi-exclamation-triangle-fill', color: 'text-warning' }; // Warning\n if (level >= 20) return { icon: 'bi-info-circle-fill', color: 'text-info' }; // Info\n return { icon: 'bi-bell-fill', color: 'text-secondary' }; // Debug/Default\n }\n\n async onInit() {\n // Overview Tab\n this.overviewView = new DataView({\n model: this.model,\n className: \"p-3\",\n columns: 2,\n fields: [\n { name: 'id', label: 'Event ID' },\n { name: 'level', label: 'Level' },\n { name: 'hostname', label: 'Hostname' },\n { name: 'incident', label: 'Incident ID' },\n { name: 'model_name', label: 'Related Model' },\n { name: 'model_id', label: 'Related Model ID' },\n { name: 'details', label: 'Details', columns: 12 },\n ]\n });\n\n const tabs = { 'Overview': this.overviewView };\n \n const metadata = this.model.get('metadata') || {};\n \n // Add Stack Trace tab if present\n if (metadata.stack_trace) {\n this.stackTraceView = new StackTraceView({\n stackTrace: metadata.stack_trace\n });\n tabs['Stack Trace'] = this.stackTraceView;\n }\n \n // Add Metadata tab if there's metadata\n if (Object.keys(metadata).length > 0) {\n this.metadataView = new View({\n model: this.model,\n template: `<pre class=\"bg-light p-3 border rounded\"><code>{{{model.metadata|json}}}</code></pre>`\n });\n tabs['Metadata'] = this.metadataView;\n }\n\n this.tabView = new TabView({\n containerId: 'event-tabs',\n tabs: tabs,\n activeTab: 'Overview'\n });\n this.addChild(this.tabView);\n\n // ContextMenu\n const menuItems = [\n { label: 'View Incident', action: 'view-incident', icon: 'bi-shield-exclamation', disabled: !this.model.get('incident') },\n { label: 'View Related Model', action: 'view-model', icon: 'bi-box-arrow-up-right', disabled: !this.model.get('model_id') },\n { type: 'divider' },\n { label: 'Delete Event', action: 'delete-event', icon: 'bi-trash', danger: true }\n ];\n\n const eventMenu = new ContextMenu({\n containerId: 'event-context-menu',\n className: \"context-menu-view header-menu-absolute\",\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: menuItems\n }\n });\n this.addChild(eventMenu);\n }\n\n async onActionViewIncident() {\n console.log(\"TODO: View incident\", this.model.get('incident'));\n }\n\n async onActionViewModel() {\n console.log(\"TODO: View model\", this.model.get('model_name'), this.model.get('model_id'));\n }\n\n async onActionDeleteEvent() {\n const confirmed = await Dialog.confirm(\n `Are you sure you want to delete this event? This action cannot be undone.`,\n 'Confirm Deletion',\n { confirmClass: 'btn-danger', confirmText: 'Delete' }\n );\n if (confirmed) {\n const resp = await this.model.destroy();\n if (resp.success) {\n this.emit('event:deleted', { model: this.model });\n }\n }\n }\n}\n\nIncidentEvent.VIEW_CLASS = EventView;\n\nexport default EventView;\n","/**\n * EventTablePage - System events management using TablePage component\n * Clean implementation using TablePage with minimal overrides\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { IncidentEventList, IncidentEventForms } from '@core/models/Incident.js';\nimport EventView from './EventView.js';\n\nclass EventTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_events',\n pageName: 'System Events',\n router: \"admin/events\",\n Collection: IncidentEventList,\n\n formEdit: IncidentEventForms.edit,\n itemViewClass: EventView,\n\n viewDialogOptions: {\n header: false,\n size: 'lg'\n },\n\n defaultQuery: {\n sort: '-id',\n category__not: \"ossec\",\n },\n\n // Column definitions\n columns: [\n {\n key: 'created', label: 'Timestamp',\n sortable: true, formatter: 'datetime',\n filter: {\n type: 'daterange',\n }\n },\n {\n key: 'level', label: 'Level',\n sortable: true, formatter: 'badge',\n filter: {\n type: \"select\",\n options: [\n { value: '5', label: 'Critical' },\n { value: '4', label: 'Warning' },\n { value: '3', label: 'Info' },\n { value: '2', label: 'Debug' },\n { value: '1', label: 'Trace' }\n ]\n }\n },\n {\n key: 'scope',\n label: 'Scope',\n sortable: true, formatter: 'badge',\n filter: {\n type: \"combobox\",\n options: [\n { value: 'account', label: 'Account' },\n { value: 'incident', label: 'Incident' },\n { value: 'ossec', label: 'OSSEC' },\n { value: 'fileman', label: 'File Manager' },\n { value: 'metrics', label: 'Metrics' },\n { value: 'jobs', label: 'Jobs' },\n { value: 'aws', label: 'AWS' }\n\n ]\n }\n },\n {\n key: 'category',\n label: 'Category',\n sortable: true, formatter: 'badge',\n filter: {\n type: \"combobox\",\n options: [\n { value: 'rest_error', label: 'Rest Error' },\n { value: 'api_error', label: 'API Error' },\n { value: 'auth', label: 'Auth' },\n { value: 'database', label: 'Database' }\n ]\n }\n },\n { key: 'title', label: 'Title', sortable: true, formatter: 'truncate(50)' },\n {\n key: 'source_ip', label: 'Source IP', sortable: true,\n filter: {\n type: \"text\"\n }\n },\n {\n key: 'metadata.server', label: 'Server',\n sortable: true,\n filter: {\n type: \"text\"\n }\n },\n ],\n\n filters: [\n {\n key: 'category__not',\n label: 'Not Category',\n filter: {type:\"text\"}\n },\n {\n key: 'metadata__http_url__icontains',\n label: 'URL Contains',\n filter: {type:\"text\"}\n },\n {\n key: 'metadata__http_path__icontains',\n label: 'Path Contains',\n filter: {type:\"text\"}\n },\n {\n key: 'metadata__http_query_string__icontains',\n label: 'Query String Contains',\n filter: {type:\"text\"}\n },\n {\n key: 'metadata__rule_id',\n label: 'Rule ID',\n filter: {type:\"text\"}\n },\n {\n key: 'metadata__country_code',\n label: 'Country',\n filter: {type:\"text\"}\n },\n {\n key: 'metadata__region',\n label: 'Region',\n filter: {type:\"text\"}\n },\n {\n key: 'metadata__city__icontains',\n label: 'City',\n filter: {type:\"text\"}\n },\n {\n key: 'metadata__http_status',\n label: 'HTTP Status',\n filter: {type:\"text\"}\n },\n {\n key: 'model_name',\n label: 'Model Name',\n filter: {type:\"text\"}\n },\n {\n key: 'model_id',\n label: 'Model ID',\n filter: {type:\"text\"}\n },\n {\n key: 'metadata__user_email',\n label: 'User Email',\n filter: {type:\"text\"}\n },\n {\n key: 'metadata__http_user_agent__icontains',\n label: 'User Agent Contains',\n filter: {type:\"text\"}\n },\n ],\n\n // Table features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Toolbar\n showRefresh: true,\n showAdd: false,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No events found.',\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n }\n });\n }\n}\n\nexport default EventTablePage;\n","import { TicketNote, TicketNoteList } from '@core/models/Tickets.js';\n\nclass TicketNoteAdapter {\n constructor(ticketId) {\n this.ticketId = ticketId;\n this.collection = new TicketNoteList({ params: { parent: this.ticketId, sort: 'created', size: 100} });\n }\n\n async fetch() {\n await this.collection.fetch();\n return this.collection.models.map(note => this.transform(note));\n }\n\n transform(note) {\n return {\n id: note.get('id'),\n type: 'user_comment', // Ticket notes are always user comments\n author: {\n id: note.get('user.id'),\n name: note.get('user.display_name') || 'System',\n avatarUrl: note.get('user.avatar.url')\n },\n timestamp: note.get('created'),\n content: note.get('note'),\n attachments: note.get('media') ? [note.get('media')] : []\n };\n }\n\n async addNote(data) {\n const note = new TicketNote();\n const resp = await note.save({\n parent: this.ticketId,\n note: data.text,\n media: data.files && data.files.length > 0 ? data.files[0].id : null\n });\n if (resp.success) {\n await this.collection.fetch(); // Refresh the collection\n }\n return resp;\n }\n}\n\nexport default TicketNoteAdapter;\n","import View from '@core/View.js';\nimport DataView from '@core/views/data/DataView.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport { Ticket, TicketForms } from '@core/models/Tickets.js';\nimport ChatView from '@core/views/chat/ChatView.js';\nimport TicketNoteAdapter from './adapters/TicketNoteAdapter.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\n\nclass TicketView extends View {\n constructor(options = {}) {\n super({\n className: 'ticket-view',\n ...options\n });\n\n this.model = options.model || new Ticket(options.data || {});\n\n this.template = `\n <div class=\"ticket-view-container d-flex flex-column h-100\">\n <!-- Ticket Header -->\n <div class=\"d-flex justify-content-between align-items-start mb-3 flex-shrink-0\">\n <!-- Left Side: Primary Identity -->\n <div class=\"d-flex align-items-center gap-3\">\n <div class=\"avatar-placeholder rounded-circle bg-light d-flex align-items-center justify-content-center\" style=\"width: 80px; height: 80px;\">\n <i class=\"bi bi-ticket-perforated text-secondary\" style=\"font-size: 40px;\"></i>\n </div>\n <div>\n <h3 class=\"mb-1\">{{model.title|truncate(50)|default('Untitled Ticket')}}</h3>\n <div class=\"text-muted small\">\n <span>Ticket #{{model.id}}</span>\n <span class=\"mx-2\">|</span>\n <span>Priority: {{model.priority|capitalize}}</span>\n {{#model.assignee}}\n <span class=\"mx-2\">|</span>\n <span>Assigned to: {{model.assignee.display_name}}</span>\n {{/model.assignee}}\n </div>\n {{#model.incident}}\n <div class=\"text-muted small mt-1\">\n <i class=\"bi bi-exclamation-triangle\"></i> Related to incident: {{model.incident}}\n </div>\n {{/model.incident}}\n </div>\n </div>\n\n <!-- Right Side: Status & Actions -->\n <div class=\"d-flex align-items-start gap-4\">\n <div class=\"text-end\">\n <div class=\"d-flex align-items-center gap-2\">\n <span class=\"badge {{model.status|badgeClass}}\">{{model.status|capitalize}}</span>\n </div>\n {{#model.created}}\n <div class=\"text-muted small mt-1\">Created {{model.created|relative}}</div>\n {{/model.created}}\n {{#model.modified}}\n <div class=\"text-muted small\">Updated {{model.modified|relative}}</div>\n {{/model.modified}}\n </div>\n <div data-container=\"ticket-context-menu\"></div>\n </div>\n </div>\n\n <!-- Chat View (Full height) -->\n <div class=\"flex-grow-1\" style=\"min-height: 0;\" data-container=\"chat-view\"></div>\n </div>\n `;\n }\n\n async onInit() {\n // Chat View with compact theme (Option 4)\n const adapter = new TicketNoteAdapter(this.model.get('id'));\n this.chatView = new ChatView({\n containerId: 'chat-view',\n adapter: adapter,\n theme: 'compact', // Use compact admin-style theme\n currentUserId: this.getCurrentUserId(),\n inputPlaceholder: 'Add a note...',\n inputButtonText: 'Add Note'\n });\n this.addChild(this.chatView);\n\n // Context Menu\n const ticketMenu = new ContextMenu({\n containerId: 'ticket-context-menu',\n className: \"context-menu-view header-menu-absolute\",\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: [\n { label: 'Edit Ticket', action: 'edit-ticket', icon: 'bi-pencil' },\n { label: 'Change Status', action: 'change-status', icon: 'bi-tag' },\n { label: 'Set Priority', action: 'set-priority', icon: 'bi-flag' },\n { label: 'Assign User', action: 'assign-user', icon: 'bi-person' },\n { type: 'divider' },\n { label: 'Close Ticket', action: 'close-ticket', icon: 'bi-x-circle' },\n ]\n }\n });\n this.addChild(ticketMenu);\n }\n\n /**\n * Get current user ID for chat message positioning\n * @returns {number|null}\n */\n getCurrentUserId() {\n // Get from WebApp state or wherever your app stores current user\n const currentUser = window.app?.state?.user;\n return currentUser?.id || null;\n }\n\n // Context Menu Action Handlers\n async onActionEditTicket() {\n const resp = await Dialog.showModelForm({\n title: `Edit Ticket #${this.model.get('id')} - ${this.model.get('title')}`,\n model: this.model,\n size: 'lg',\n fields: TicketForms.edit.fields\n });\n if (resp) {\n this.render(); // Re-render to show updated data in header\n }\n }\n\n async onActionChangeStatus() {\n const statuses = ['new', 'open', 'in_progress', 'pending', 'resolved', 'closed', 'ignored'];\n const currentStatus = this.model.get('status');\n\n const result = await Dialog.showForm({\n title: 'Change Ticket Status',\n size: 'sm',\n fields: [\n {\n name: 'status',\n label: 'New Status',\n type: 'select',\n options: statuses.map(s => ({ value: s, label: s.replace('_', ' ').toUpperCase() })),\n value: currentStatus,\n required: true\n }\n ]\n });\n\n if (result) {\n try {\n await this.model.save({ status: result.status });\n this.render();\n } catch (error) {\n Dialog.alert({\n type: 'error',\n title: 'Error',\n message: 'Failed to update ticket status: ' + error.message\n });\n }\n }\n }\n\n async onActionSetPriority() {\n const priorities = ['low', 'normal', 'high', 'urgent'];\n const currentPriority = this.model.get('priority');\n\n const result = await Dialog.showForm({\n title: 'Set Ticket Priority',\n size: 'sm',\n fields: [\n {\n name: 'priority',\n label: 'Priority Level',\n type: 'select',\n options: priorities.map(p => ({ value: p, label: p.toUpperCase() })),\n value: currentPriority,\n required: true\n }\n ]\n });\n\n if (result) {\n try {\n await this.model.save({ priority: result.priority });\n this.render();\n } catch (error) {\n Dialog.alert({\n type: 'error',\n title: 'Error',\n message: 'Failed to update ticket priority: ' + error.message\n });\n }\n }\n }\n\n async onActionAssignUser() {\n console.log(\"TODO: Implement assign user dialog with user selector\");\n Dialog.alert({\n title: 'Coming Soon',\n message: 'User assignment feature will be implemented soon.'\n });\n }\n\n async onActionCloseTicket() {\n const confirmed = await Dialog.confirm({\n title: 'Close Ticket',\n message: `Are you sure you want to close ticket #${this.model.get('id')}?`,\n confirmText: 'Close Ticket',\n confirmClass: 'btn-warning'\n });\n\n if (confirmed) {\n try {\n await this.model.save({ status: 'closed' });\n this.render();\n Dialog.alert({\n type: 'success',\n title: 'Success',\n message: 'Ticket has been closed successfully.'\n });\n } catch (error) {\n Dialog.alert({\n type: 'error',\n title: 'Error',\n message: 'Failed to close ticket: ' + error.message\n });\n }\n }\n }\n}\n\nTicket.VIEW_CLASS = TicketView;\n\nexport default TicketView;\n","/**\n * TicketTablePage - Ticket management using TablePage component\n * Clean implementation using TablePage with minimal overrides\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { TicketList, TicketForms, TicketCategories } from '@core/models/Tickets.js';\nimport TicketView from './TicketView.js';\n\nclass TicketTablePage extends TablePage {\n constructor(options = {}) {\n super({\n name: 'admin_tickets',\n pageName: 'Tickets',\n router: \"admin/tickets\",\n Collection: TicketList,\n\n formCreate: TicketForms.create,\n formEdit: TicketForms.edit,\n itemViewClass: TicketView,\n\n viewDialogOptions: {\n header: false\n },\n\n defaultQuery: {\n sort: '-priority',\n status: \"open\"\n },\n\n // Column definitions\n columns: [\n { key: 'id', label: 'ID', width: '70px', sortable: true, class: 'text-muted' },\n { key: 'title', label: 'Title', sortable: true},\n {\n key: 'status', label: 'Status', sortable: true,\n editable: true,\n editableOptions: {\n type: \"select\",\n options: [\n \"new\", \"open\", \"paused\", \"resolved\", \"qa\", \"ignored\"\n ]\n },\n filter: {\n type: \"multiselect\",\n placeHolder: \"Select Status\",\n options: [\n \"new\", \"open\", \"paused\", \"resolved\", \"qa\", \"ignored\"\n ]\n }\n },\n { key: 'priority', label: 'Priority', sortable: true },\n {\n key: 'category', label: 'Category', sortable: true,\n editable: true,\n editableOptions: {\n type: \"select\",\n options: [\n ... Object.keys(TicketCategories)\n ]\n },\n filter: {\n type: \"multiselect\",\n placeHolder: \"Select Category\",\n options: [\n ... Object.keys(TicketCategories)\n ]\n }\n },\n { key: 'assignee.display_name', label: 'Assignee', sortable: true, formatter: \"default('Unassigned')\" },\n { key: 'incident.id', label: 'Incident ID', sortable: true },\n { key: 'created', label: 'Created', sortable: true, formatter: 'datetime' }\n ],\n\n // Table features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Toolbar\n showRefresh: true,\n showAdd: true,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No tickets found.',\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n },\n ...options,\n });\n }\n}\n\nexport default TicketTablePage;\n","import View from '@core/View.js';\nimport TabView from '@core/views/navigation/TabView.js';\nimport DataView from '@core/views/data/DataView.js';\nimport TableView from '@core/views/table/TableView.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport { RuleSet, RuleList, BundleByOptions, MatchByOptions } from '@core/models/Incident.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\n\nclass RuleSetView extends View {\n constructor(options = {}) {\n super({\n className: 'ruleset-view',\n ...options\n });\n\n this.model = options.model || new RuleSet(options.data || {});\n\n this.template = `\n <div class=\"ruleset-view-container\">\n <!-- Header -->\n <div class=\"d-flex justify-content-between align-items-center mb-4\">\n <div class=\"d-flex align-items-center gap-3\">\n <div class=\"fs-1 text-primary\"><i class=\"bi bi-gear-wide-connected\"></i></div>\n <div>\n <h3 class=\"mb-1\">{{model.name}}</h3>\n <div class=\"text-muted small\">Scope: {{model.category}} | Priority: {{model.priority}}</div>\n </div>\n </div>\n <div data-container=\"ruleset-context-menu\"></div>\n </div>\n\n <!-- Tabs -->\n <div data-container=\"ruleset-tabs\"></div>\n </div>\n `;\n }\n\n async onInit() {\n // Get labels for current values\n const matchByValue = this.model.get('match_by');\n const matchByOption = MatchByOptions.find(opt => opt.value === matchByValue);\n const matchByLabel = matchByOption ? matchByOption.label : String(matchByValue);\n\n const bundleByValue = this.model.get('bundle_by');\n const bundleByOption = BundleByOptions.find(opt => opt.value === bundleByValue);\n const bundleByLabel = bundleByOption ? bundleByOption.label : String(bundleByValue);\n\n // Config Tab\n this.configView = new DataView({\n model: this.model,\n className: \"p-3\",\n columns: 2,\n fields: [\n { name: 'name', label: 'Name', cols: 4 },\n { name: 'category', label: 'Scope', formatter: 'badge', cols: 4 },\n { name: 'is_active', label: 'Is Active', formatter: 'yesno_icon', cols: 4 },\n { name: 'priority', label: 'Priority', cols: 4 },\n { name: 'id', label: 'RuleSet ID', cols: 4 },\n\n {\n name: 'match_by',\n label: 'Match Logic',\n template: matchByLabel,\n cols: 4\n },\n {\n name: 'bundle_by',\n label: 'Bundle By',\n template: bundleByLabel,\n cols: 4\n },\n { name: 'bundle_minutes', label: 'Bundle Minutes', cols: 4 },\n { name: 'bundle_by_rule_set', label: 'Bundle By Rule Set', formatter: 'yesno_icon', cols: 4 },\n { name: 'handler', label: 'Handler', cols: 12 },\n ]\n });\n\n // Rules Tab\n const rulesCollection = new RuleList({\n params: { parent: this.model.get('id') }\n });\n this.rulesView = new TableView({\n collection: rulesCollection,\n hideActivePillNames: ['parent'],\n columns: [\n { key: 'id', label: 'ID', width: '70px' },\n { key: 'name', label: 'Name' },\n { key: 'field_name', label: 'Field' },\n { key: 'comparator', label: 'Comparator', width: '120px' },\n { key: 'value', label: 'Value' },\n { key: 'value_type', label: 'Type', width: '100px' },\n ],\n showAdd: true,\n clickAction: 'edit',\n actions: ['edit', 'delete'],\n contextMenu: [\n { label: 'Edit Rule', action: 'edit', icon: 'bi-pencil' },\n { label: 'Duplicate Rule', action: 'duplicate', icon: 'bi-files' },\n { divider: true },\n { label: 'Delete Rule', action: 'delete', icon: 'bi-trash', danger: true }\n ],\n // Pass the parent ID so new rules get associated with this ruleset\n addFormDefaults: {\n parent: this.model.get('id')\n }\n });\n\n this.tabView = new TabView({\n containerId: 'ruleset-tabs',\n tabs: {\n 'Configuration': this.configView,\n 'Rules': this.rulesView\n },\n activeTab: 'Configuration'\n });\n this.addChild(this.tabView);\n\n const contextMenu = new ContextMenu({\n containerId: 'ruleset-context-menu',\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: [\n { label: 'Edit RuleSet', action: 'edit-ruleset', icon: 'bi-pencil' },\n { label: 'Disable', action: 'disable-ruleset', icon: 'bi-toggle-off' },\n { type: 'divider' },\n { label: 'Delete RuleSet', action: 'delete-ruleset', icon: 'bi-trash', danger: true }\n ]\n }\n });\n this.addChild(contextMenu);\n }\n\n /**\n * Action handler: Edit RuleSet\n */\n async onActionEditRuleset() {\n const resp = await Dialog.showModelForm({\n title: `Edit RuleSet - ${this.model.get('name')}`,\n model: this.model,\n formConfig: RuleSet.EDIT_FORM,\n });\n if (resp) {\n await this.render();\n }\n }\n\n /**\n * Action handler: Disable/Enable RuleSet\n */\n async onActionDisableRuleset() {\n const isActive = this.model.get('is_active');\n const newStatus = !isActive;\n\n try {\n this.model.set('is_active', newStatus);\n await this.model.save();\n await this.render();\n\n Dialog.showToast({\n message: `RuleSet ${newStatus ? 'enabled' : 'disabled'} successfully`,\n type: 'success'\n });\n } catch (error) {\n Dialog.showToast({\n message: `Failed to update RuleSet: ${error.message}`,\n type: 'error'\n });\n }\n }\n\n /**\n * Action handler: Delete RuleSet\n */\n async onActionDeleteRuleset() {\n const confirmed = await Dialog.confirm({\n title: 'Delete RuleSet',\n message: `Are you sure you want to delete the ruleset \"${this.model.get('name')}\"? This action cannot be undone.`,\n confirmText: 'Delete',\n confirmClass: 'btn-danger'\n });\n\n if (confirmed) {\n try {\n await this.model.destroy();\n\n Dialog.showToast({\n message: 'RuleSet deleted successfully',\n type: 'success'\n });\n\n // Close the dialog\n const dialog = this.element?.closest('.modal');\n if (dialog) {\n const bsModal = bootstrap.Modal.getInstance(dialog);\n if (bsModal) {\n bsModal.hide();\n }\n }\n\n // Emit event to parent to refresh the table\n this.emit('ruleset:deleted', { model: this.model });\n } catch (error) {\n Dialog.showToast({\n message: `Failed to delete RuleSet: ${error.message}`,\n type: 'error'\n });\n }\n }\n }\n}\n\nRuleSetView.VIEW_CLASS = RuleSetView;\n\nexport default RuleSetView;\n","import TablePage from '@core/pages/TablePage.js';\nimport { RuleSetList } from '@core/models/Incident.js';\nimport RuleSetView from './RuleSetView.js';\n\nclass RuleSetTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_rulesets',\n pageName: 'Rule Engine',\n router: \"admin/rulesets\",\n Collection: RuleSetList,\n itemView: RuleSetView,\n viewDialogOptions: {\n header: false,\n size: 'xl'\n },\n\n columns: [\n { key: 'id', label: 'ID', width: '70px', sortable: true, class: 'text-muted' },\n { key: 'name', label: 'Name', sortable: true },\n { key: 'category', label: 'Scope', sortable: true, formatter: 'badge' },\n { key: 'priority', label: 'Priority', sortable: true },\n { key: 'match_by', label: 'Match Logic', formatter: (v) => v === 0 ? 'ALL' : 'ANY' }\n ],\n\n selectable: true,\n searchable: true,\n sortable: true,\n paginated: true,\n showRefresh: true,\n showAdd: true,\n showExport: true,\n\n tableOptions: {\n pageSizes: [10, 25, 50],\n defaultPageSize: 25,\n emptyMessage: 'No rule sets found.',\n emptyIcon: 'bi-gear',\n actions: [\"view\", \"edit\", \"delete\"],\n }\n });\n }\n}\n\nexport default RuleSetTablePage;\n","/**\n * EmailDomainTablePage - Admin page for managing SES/SNS Email Domains\n * Clean implementation following simplified TablePage pattern\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\nimport View from '@core/View.js';\nimport { EmailDomain, EmailDomainList, EmailDomainForms } from '@core/models/Email.js';\n\nclass EmailDomainTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_email_domains',\n pageName: 'Email Domains',\n router: 'admin/email/domains',\n Collection: EmailDomainList,\n formCreate: EmailDomainForms.create,\n formEdit: EmailDomainForms.edit,\n\n // Column definitions\n columns: [\n {\n key: 'id',\n label: 'ID',\n width: '70px',\n sortable: true,\n class: 'text-muted'\n },\n {\n key: 'name',\n label: 'Domain',\n sortable: true\n },\n {\n key: 'region',\n label: 'Region',\n sortable: true,\n formatter: \"default('—')\"\n },\n {\n key: 'receiving_enabled',\n label: 'Receiving',\n formatter: \"boolean|badge\"\n },\n {\n key: 'can_send',\n label: 'Send Verified',\n formatter: \"boolean|badge\"\n },\n {\n key: 'can_recv',\n label: 'Recv Verified',\n formatter: \"boolean|badge\"\n },\n {\n key: 'created',\n label: 'Created',\n formatter: \"epoch|datetime\"\n }\n ],\n\n // Table features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Toolbar\n showRefresh: true,\n showAdd: true,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No domains found. Click \"Add Domain\" to get started.',\n\n // Context menu configuration\n contextMenu: [\n {\n icon: 'bi-shield-check',\n action: 'edit-aws-creds',\n label: 'Edit AWS Credentials'\n },\n {\n icon: 'bi-rocket-takeoff',\n action: 'onboard',\n label: 'Onboard'\n },\n {\n icon: 'bi-shield-check',\n action: 'audit',\n label: 'Audit'\n },\n {\n icon: 'bi-arrow-repeat',\n action: 'reconcile',\n label: 'Reconcile'\n }\n ],\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n }\n });\n }\n\n async onActionEditAwsCreds(event, element) {\n const item = this.collection.get(element.dataset.id);\n const result = await Dialog.showModelForm({\n model: item,\n formConfig: EmailDomainForms.credentials,\n }\n );\n return true;\n }\n\n async onActionOnboard(event, element) {\n const item = this.collection.get(element.dataset.id);\n const model = new EmailDomain({ id: item.id });\n\n const formData = await Dialog.showForm(EmailDomainForms.onboard);\n if (!formData) return;\n\n try {\n const resp = await model.onboard(formData);\n if (!resp.success) {\n throw new Error(resp.message || 'Network error during onboarding');\n }\n if (!resp.data?.status) {\n throw new Error(resp.data?.error || 'Onboarding failed');\n }\n\n this.getApp()?.toast?.success('Domain onboarding completed successfully');\n await this.refresh();\n } catch (err) {\n console.error('Onboard error:', err);\n this.showError(err.message || 'Failed to onboard domain');\n }\n }\n\n async onActionAudit(event, element) {\n const item = this.collection.get(element.dataset.id);\n const model = new EmailDomain({ id: item.id });\n\n try {\n const resp = await model.audit();\n if (!resp.success) {\n throw new Error(resp.message || 'Network error during audit');\n }\n if (!resp.data?.status) {\n throw new Error(resp.data?.error || 'Audit failed');\n }\n\n const result = resp.data?.data || {};\n await Dialog.showDialog({\n title: `Audit Report - ${item.name}`,\n body: new View({\n template: `\n <div>\n <p class=\"text-muted\">Drift report and status:</p>\n <pre class=\"bg-light p-3 rounded small\"><code>{{{data.result|json}}}</code></pre>\n </div>\n `,\n data: { result }\n }),\n size: 'lg'\n });\n } catch (err) {\n console.error('Audit error:', err);\n this.showError(err.message || 'Failed to audit domain');\n }\n }\n\n async onActionReconcile(event, element) {\n const item = this.collection.get(element.dataset.id);\n const model = new EmailDomain({ id: item.id });\n\n try {\n const resp = await model.reconcile();\n if (!resp.success) {\n throw new Error(resp.message || 'Network error during reconcile');\n }\n if (!resp.data?.status) {\n throw new Error(resp.data?.error || 'Reconcile failed');\n }\n\n this.getApp()?.toast?.success('Reconcile completed successfully');\n await this.refresh();\n } catch (err) {\n console.error('Reconcile error:', err);\n this.showError(err.message || 'Failed to reconcile domain');\n }\n }\n}\n\nexport default EmailDomainTablePage;\n","/**\n * EmailMailboxTablePage - Admin page for managing Mailboxes\n * Clean implementation using TablePage with minimal overrides\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { MailboxList, MailboxForms, Mailbox } from '@core/models/Email.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\n\nclass EmailMailboxTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_email_mailboxes',\n pageName: 'Mailboxes',\n router: 'admin/email/mailboxes',\n Collection: MailboxList,\n\n formCreate: MailboxForms.create,\n formEdit: MailboxForms.edit,\n clickAction: \"edit\",\n\n // Table columns\n columns: [\n { key: 'id', label: 'ID', width: '70px', sortable: true, class: 'text-muted' },\n { key: 'email', label: 'Email', sortable: true },\n { key: 'domain.name', label: 'Domain', sortable: true, formatter: \"default('—')\" },\n { key: 'allow_inbound', label: 'Inbound', formatter: \"boolean|badge\" },\n { key: 'allow_outbound', label: 'Outbound', formatter: \"boolean|badge\" },\n { key: 'is_system_default', label: 'System Default', formatter: \"boolean|badge\" },\n { key: 'is_domain_default', label: 'Domain Default', formatter: \"boolean|badge\" }\n ],\n\n // Features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Toolbar\n showRefresh: true,\n showAdd: true,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No mailboxes found. Click \"Add Mailbox\" to create one.',\n\n // Context menu configuration\n contextMenu: [\n { icon: 'bi-envelope', action: 'send-email', label: 'Send Email' },\n { icon: 'bi-envelope', action: 'send-template-email', label: 'Send Template Email' }\n ],\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n }\n });\n }\n\n async onActionSendEmail(event, element) {\n const item = this.collection.get(element.dataset.id);\n // Implement send email action\n const data = await Dialog.showForm({\n title: 'Send Email',\n fields: [\n { name: 'to', label: 'To', type: 'email', required: true },\n { name: 'subject', label: 'Subject', type: 'text', required: true },\n { name: 'body_html', label: 'Body', type: 'textarea', required: true }\n ]\n });\n data.from_email = item.get('email');\n const result = await Mailbox.sendEmail(data);\n if (result.success) {\n this.getApp().toast.success('Email sent successfully');\n } else {\n let msg = \"Failed to send email\";\n if (result.data.details) {\n msg = result.data.details;\n } else if (result.data.error) {\n msg = result.data.error;\n }\n console.log(result);\n this.getApp().toast.error(msg);\n }\n }\n\n async onActionSendTemplateEmail(event, element) {\n const item = this.collection.get(element.dataset.id);\n // Implement send email action\n const data = await Dialog.showForm({\n title: 'Send Email',\n fields: [\n { name: 'to', label: 'To', type: 'email', required: true },\n { name: 'subject', label: 'Subject', type: 'text', required: true },\n { name: 'template_name', label: 'Template', type: 'text', required: true },\n { name: 'template_context', label: 'Context', type: 'textarea', required: true }\n ]\n });\n data.from_email = item.get('email');\n const result = await Mailbox.sendEmail(data);\n if (result.success) {\n this.getApp().toast.success('Email sent successfully');\n } else {\n let msg = \"Failed to send email\";\n if (result.data.details) {\n msg = result.data.details;\n } else if (result.data.error) {\n msg = result.data.error;\n }\n console.log(result);\n this.getApp().toast.error(msg);\n }\n }\n}\n\nexport default EmailMailboxTablePage;\n","/**\n * EmailTemplateView - A clean view for previewing email templates\n */\n\nimport View from '@core/View.js';\nimport TabView from '@core/views/navigation/TabView.js';\nimport { EmailTemplate } from '@core/models/Email.js';\n\n/**\n * EmailHtmlPreviewView - Renders HTML email template in a sandboxed iframe\n */\nclass EmailHtmlPreviewView extends View {\n constructor(options = {}) {\n super({\n className: 'email-html-preview',\n template: `\n <div class=\"email-html-preview-container\">\n <div class=\"d-flex justify-content-between align-items-center mb-2\">\n <small class=\"text-muted\">HTML Preview (sandboxed)</small>\n <button type=\"button\" class=\"btn btn-sm btn-outline-secondary\" data-action=\"refresh-preview\">\n <i class=\"bi bi-arrow-clockwise\"></i> Refresh\n </button>\n </div>\n <iframe\n id=\"email-preview-frame\"\n class=\"border rounded w-100\"\n style=\"height: 500px; background: white;\"\n sandbox=\"allow-same-origin\"\n frameborder=\"0\"\n ></iframe>\n </div>\n `,\n ...options\n });\n }\n\n async onAfterRender() {\n await super.onAfterRender();\n this.renderHtmlInIframe();\n }\n\n renderHtmlInIframe() {\n const iframe = this.element.querySelector('#email-preview-frame');\n if (!iframe) return;\n\n const htmlContent = this.model.get('html_template') || '';\n\n // Get iframe document\n const iframeDoc = iframe.contentDocument || iframe.contentWindow.document;\n\n // Write HTML content\n iframeDoc.open();\n iframeDoc.write(htmlContent);\n iframeDoc.close();\n }\n\n async onActionRefreshPreview(event, element) {\n this.renderHtmlInIframe();\n }\n}\n\nclass EmailTemplateView extends View {\n constructor(options = {}) {\n super({\n className: 'email-template-view',\n ...options\n });\n\n this.model = options.model || new EmailTemplate(options.data || {});\n this.hasHtml = !!this.model.get('html_template');\n this.hasText = !!this.model.get('text_template');\n this.hasMetadata = this.model.get('metadata') && Object.keys(this.model.get('metadata')).length > 0;\n\n this.template = `\n <div class=\"email-template-view-container p-3\">\n <!-- Header -->\n <div class=\"template-header border-bottom pb-3 mb-3\">\n <h4 class=\"mb-1\">{{model.name}}</h4>\n <div class=\"text-muted\">\n <strong>Subject:</strong> {{model.subject_template|default('Not set')}}\n </div>\n </div>\n\n <!-- Tabs -->\n <div data-container=\"template-tabs\"></div>\n </div>\n `;\n }\n\n async onInit() {\n const tabs = {};\n\n if (this.hasHtml) {\n tabs['HTML Preview'] = new EmailHtmlPreviewView({\n model: this.model\n });\n }\n\n if (this.hasText) {\n tabs['Text Version'] = new View({\n model: this.model,\n template: `<pre class=\"email-text-content p-3 bg-light border rounded\" style=\"white-space: pre-wrap; word-wrap: break-word;\">{{model.text_template}}</pre>`\n });\n }\n\n if (this.hasMetadata) {\n tabs['Metadata'] = new View({\n model: this.model,\n template: `<pre class=\"email-metadata-content p-3 bg-light border rounded\"><code>{{model.metadata|json}}</code></pre>`\n });\n }\n\n this.tabView = new TabView({\n containerId: 'template-tabs',\n tabs: tabs,\n activeTab: Object.keys(tabs)[0] || ''\n });\n this.addChild(this.tabView);\n }\n}\n\nexport default EmailTemplateView;\n","/**\n * EmailTemplateTablePage - Admin page for managing Email Templates\n * Clean implementation using TablePage with minimal overrides\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { EmailTemplateList, EmailTemplateForms } from '@core/models/Email.js';\nimport EmailTemplateView from './EmailTemplateView.js';\n\nclass EmailTemplateTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_email_templates',\n pageName: 'Email Templates',\n router: 'admin/email/templates',\n Collection: EmailTemplateList,\n\n formCreate: EmailTemplateForms.create,\n formEdit: EmailTemplateForms.edit,\n itemViewClass: EmailTemplateView,\n clickAction: \"edit\",\n\n viewDialogOptions: {\n header: false,\n size: 'xl',\n scrollable: true\n },\n\n formDialogConfig: { // Dialog options for forms\n size: 'fullscreen',\n },\n\n // Table columns\n columns: [\n { key: 'id', label: 'ID', width: '70px', sortable: true, class: 'text-muted' },\n { key: 'name', label: 'Name', sortable: true },\n { key: 'created', label: 'Created', formatter: 'datetime' },\n { key: 'modified', label: 'Modified', formatter: 'datetime' }\n ],\n\n // Features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Toolbar\n showRefresh: true,\n showAdd: true,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No email templates found. Click \"Add Template\" to create your first one.',\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n }\n });\n }\n}\n\nexport default EmailTemplateTablePage;\n","/**\n * EmailView - A slick, email-client-like view for sent messages\n */\n\nimport View from '@core/View.js';\nimport TabView from '@core/views/navigation/TabView.js';\nimport { SentMessage } from '@core/models/Email.js';\n\nclass EmailView extends View {\n constructor(options = {}) {\n super({\n className: 'email-view',\n ...options\n });\n\n this.model = options.model || new SentMessage(options.data || {});\n this.hasHtml = !!this.model.get('body_html');\n this.hasText = !!this.model.get('body_text');\n this.hasContext = this.model.get('template_context') && Object.keys(this.model.get('template_context')).length > 0;\n\n this.template = `\n <div class=\"email-view-container p-3\">\n <!-- Email Header -->\n <div class=\"email-header border-bottom pb-3 mb-3\">\n <h4 class=\"mb-2\">{{model.subject}}</h4>\n <div class=\"d-flex justify-content-between align-items-center\">\n <div class=\"d-flex align-items-center\">\n <div class=\"flex-shrink-0\">\n <i class=\"bi bi-person-circle fs-2 text-secondary\"></i>\n </div>\n <div class=\"ms-3\">\n <div class=\"fw-bold\">{{model.mailbox.email}}</div>\n <div class=\"text-muted small\">To: {{model.to_addresses|list}}</div>\n </div>\n </div>\n <div class=\"text-end\">\n <div class=\"text-muted small\">{{model.created|datetime}}</div>\n <span class=\"badge {{model.status|badge}} mt-1\">{{model.status|capitalize}}</span>\n </div>\n </div>\n </div>\n\n <!-- Email Body Tabs -->\n <div data-container=\"email-tabs\"></div>\n </div>\n `;\n }\n\n async onInit() {\n const tabs = {};\n\n if (this.hasHtml) {\n tabs['HTML'] = new View({\n model: this.model,\n template: `<div class=\"email-html-content border rounded p-3\" style=\"height: 500px; overflow-y: auto;\">{{{model.body_html}}}</div>`\n });\n }\n\n if (this.hasText) {\n tabs['Text'] = new View({\n model: this.model,\n template: `<pre class=\"email-text-content p-3 bg-light border rounded\" style=\"white-space: pre-wrap; word-wrap: break-word;\">{{model.body_text}}</pre>`\n });\n }\n\n if (this.hasContext) {\n tabs['Context'] = new View({\n model: this.model,\n template: `<pre class=\"email-context-content p-3 bg-light border rounded\"><code>{{model.template_context|json}}</code></pre>`\n });\n }\n\n this.tabView = new TabView({\n containerId: 'email-tabs',\n tabs: tabs,\n activeTab: this.hasHtml ? 'HTML' : (this.hasText ? 'Text' : 'Context')\n });\n this.addChild(this.tabView);\n }\n}\n\nSentMessage.VIEW_CLASS = EmailView;\n\n\nexport default EmailView;\n","/**\n * SentMessageTablePage - Admin page for viewing sent emails\n * Clean implementation using TablePage with minimal overrides\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { SentMessageList } from '@core/models/Email.js';\nimport EmailView from './EmailView.js';\n\nclass SentMessageTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_email_sent',\n pageName: 'Sent Messages',\n router: 'admin/email/sent',\n Collection: SentMessageList,\n \n itemViewClass: EmailView,\n viewDialogOptions: {\n header: false,\n size: 'xl',\n scrollable: true\n },\n\n // Table columns\n columns: [\n { key: 'id', label: 'ID', width: '70px', sortable: true, class: 'text-muted' },\n { key: 'mailbox.email', label: 'From', sortable: true },\n { key: 'to_addresses', label: 'To', sortable: false, formatter: \"list\" },\n { key: 'subject', label: 'Subject', sortable: true },\n { key: 'status', label: 'Status', formatter: 'badge' },\n { key: 'status_reason', label: 'Reason', formatter: \"truncate(80)|default('—')\" },\n { key: 'created', label: 'Created', formatter: 'datetime' }\n ],\n\n // Features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Toolbar\n showRefresh: true,\n showAdd: false,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No sent messages found.',\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n }\n });\n }\n}\n\nexport default SentMessageTablePage;","/**\n * Phonehub.js\n * Models and Collections for PhoneHub phone numbers and SMS with helper actions.\n *\n * Endpoints (all require authentication unless noted):\n * - Phone Number\n * - POST /api/phonehub/number/normalize\n * - POST /api/phonehub/number/lookup\n * - GET /api/phonehub/number\n * - GET /api/phonehub/number/:id\n * - PUT /api/phonehub/number/:id\n * - DELETE /api/phonehub/number/:id\n *\n * - SMS\n * - POST /api/phonehub/sms/send\n * - GET /api/phonehub/sms\n * - GET /api/phonehub/sms/:id\n *\n * Notes on response shapes:\n * - List endpoints typically return { status: true, data: [...], count, size, start }\n * - Single item endpoints typically return { status: true, data: {...} }\n * - Action endpoints (normalize, lookup, send) return { status: true, data: {...} } or { status: false, error }\n */\n\nimport Model from '@core/Model.js';\nimport Collection from '@core/Collection.js';\nimport rest from '@core/Rest.js';\n\n// ======================================================\n// PhoneNumber Model & Collection\n// ======================================================\n\n/**\n * PhoneNumber - Represents a cached phone number lookup result.\n */\nclass PhoneNumber extends Model {\n constructor(data = {}, options = {}) {\n super(data, {\n endpoint: '/api/phonehub/number',\n ...options\n });\n }\n\n /**\n * Normalize an arbitrary phone_number string to E.164 format.\n * @param {string} phoneNumber - Raw phone number input.\n * @param {string} [countryCode='US'] - Optional country code (default US).\n * @returns {Promise<{success: boolean, phone_number?: string, data?: object, error?: string, response?: any}>}\n */\n static async normalize(phoneNumber, countryCode = 'US') {\n const url = '/api/phonehub/number/normalize';\n const payload = {\n phone_number: phoneNumber\n };\n if (countryCode) payload.country_code = countryCode;\n\n const resp = await rest.POST(url, payload);\n const body = resp?.data ?? resp; // rest wrapper or raw\n const ok = body?.status === true || body?.success === true;\n\n if (ok) {\n const normalized = body?.data?.phone_number ?? body?.phone_number;\n return { success: true, phone_number: normalized, data: body?.data ?? body, response: resp };\n }\n return { success: false, error: body?.error || 'Normalization failed', response: resp };\n }\n\n /**\n * Lookup phone number details (carrier, line_type, owner, etc.)\n * @param {string} phoneNumber - E.164 or raw phone number.\n * @param {object} [options] - { force_refresh?: boolean, group?: number }\n * @returns {Promise<{success: boolean, model?: PhoneNumber, data?: object, error?: string, response?: any}>}\n */\n static async lookup(phoneNumber, options = {}) {\n const url = '/api/phonehub/number/lookup';\n const resp = await rest.POST(url, {\n phone_number: phoneNumber,\n ...options\n });\n const body = resp?.data ?? resp;\n const ok = body?.status === true || body?.success === true;\n\n if (ok) {\n const data = body?.data ?? {};\n const model = new PhoneNumber(data, { endpoint: '/api/phonehub/number' });\n return { success: true, model, data, response: resp };\n }\n return { success: false, error: body?.error || 'Phone lookup failed', response: resp };\n }\n\n\n}\n\n/**\n * PhoneNumberList - Paginated collection of PhoneNumber models.\n */\nclass PhoneNumberList extends Collection {\n constructor(options = {}) {\n super({\n ModelClass: PhoneNumber,\n endpoint: '/api/phonehub/number',\n size: 10,\n ...options\n });\n }\n\n\n}\n\n// ======================================================\n// SMS Model & Collection\n// ======================================================\n\n/**\n * SMS - Represents an SMS message (sent or received).\n */\nclass SMS extends Model {\n constructor(data = {}, options = {}) {\n super(data, {\n endpoint: '/api/phonehub/sms',\n ...options\n });\n }\n\n /**\n * Send SMS via PhoneHub (Twilio under the hood).\n * @param {object} params - { to_number, body, from_number?, group?, metadata? }\n * @returns {Promise<{success: boolean, model?: SMS, data?: object, error?: string, response?: any}>}\n */\n static async send(params = {}) {\n const url = '/api/phonehub/sms/send';\n const resp = await rest.POST(url, params);\n const body = resp?.data ?? resp;\n const ok = body?.status === true || body?.success === true;\n\n if (ok) {\n const data = body?.data ?? {};\n const model = new SMS(data, { endpoint: '/api/phonehub/sms' });\n return { success: true, model, data, response: resp };\n }\n return { success: false, error: body?.error || 'Failed to send SMS', response: resp };\n }\n\n\n\n\n}\n\n/**\n * SMSList - Paginated collection of SMS models.\n */\nclass SMSList extends Collection {\n constructor(options = {}) {\n super({\n ModelClass: SMS,\n endpoint: '/api/phonehub/sms',\n size: 10,\n ...options\n });\n }\n\n\n}\n\n// Exported API\nexport {\n PhoneNumber,\n PhoneNumberList,\n SMS,\n SMSList\n};\n\nexport default {\n PhoneNumber,\n PhoneNumberList,\n SMS,\n SMSList\n};","/**\n * PhoneNumberView - Detailed view for a PhoneHub PhoneNumber record\n *\n * Mirrors GeoIPView structure:\n * - Header with primary identifier and quick info\n * - Tabbed content using TabView and DataView sections\n * - Minimal context menu for refresh and delete\n */\n\nimport View from '@core/View.js';\nimport TabView from '@core/views/navigation/TabView.js';\nimport DataView from '@core/views/data/DataView.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\nimport { PhoneNumber } from '@core/models/Phonehub.js';\n\nclass PhoneNumberView extends View {\n constructor(options = {}) {\n super({\n className: 'phone-number-view',\n ...options\n });\n\n this.model = options.model || new PhoneNumber(options.data || {});\n\n this.template = `\n <div class=\"phone-number-view-container\">\n <!-- Header -->\n <div class=\"d-flex justify-content-between align-items-center mb-4\">\n <!-- Left Side: Icon & Info -->\n <div class=\"d-flex align-items-center gap-3\">\n <div class=\"fs-1 text-primary\">\n <i class=\"bi bi-telephone\"></i>\n </div>\n <div>\n <h3 class=\"mb-1\">{{model.phone_number|default('Unknown Number')}}</h3>\n <div class=\"text-muted small\">\n {{model.carrier|default('—')}} {{#model.line_type}}· {{model.line_type|capitalize}}{{/model.line_type}}\n </div>\n <div class=\"text-muted small mt-1\">\n {{#model.country_code}}Country: {{model.country_code}}{{/model.country_code}}\n {{#model.region}} · Region: {{model.region}}{{/model.region}}\n {{#model.state}} · State: {{model.state}}{{/model.state}}\n </div>\n </div>\n </div>\n\n <!-- Right Side: Actions -->\n <div class=\"d-flex align-items-center gap-4\">\n <div data-container=\"phone-context-menu\"></div>\n </div>\n </div>\n\n <!-- Tabs -->\n <div data-container=\"phone-tabs\"></div>\n </div>\n `;\n }\n\n async onInit() {\n // Overview Tab\n this.overviewView = new DataView({\n model: this.model,\n className: 'p-3',\n showEmptyValues: true,\n emptyValueText: '—',\n columns: 2,\n fields: [\n { name: 'phone_number', label: 'Phone Number', cols: 6 },\n { name: 'country_code', label: 'Country Code', cols: 6 },\n { name: 'region', label: 'Region', cols: 6 },\n { name: 'state', label: 'State', cols: 6 },\n { name: 'registered_owner', label: 'Registered Owner', cols: 6 },\n { name: 'owner_type', label: 'Owner Type', formatter: 'capitalize', cols: 6 },\n { name: 'is_valid', label: 'Valid', formatter: 'yesnoicon', cols: 4 },\n { name: 'is_mobile', label: 'Mobile', formatter: 'yesnoicon', cols: 4 },\n { name: 'is_voip', label: 'VOIP', formatter: 'yesnoicon', cols: 4 },\n ]\n });\n\n // Carrier & Status Tab\n this.carrierView = new DataView({\n model: this.model,\n className: 'p-3',\n showEmptyValues: true,\n emptyValueText: '—',\n columns: 2,\n fields: [\n { name: 'carrier', label: 'Carrier', cols: 6 },\n { name: 'line_type', label: 'Line Type', formatter: 'capitalize', cols: 6 },\n { name: 'lookup_provider', label: 'Lookup Provider', formatter: 'capitalize', cols: 6 },\n { name: 'lookup_count', label: 'Lookup Count', cols: 6 },\n { name: 'last_lookup_at', label: 'Last Lookup', formatter: 'datetime', cols: 6 },\n { name: 'lookup_expires_at', label: 'Cache Expires', formatter: 'datetime', cols: 6 }\n ]\n });\n\n // Address Tab\n this.addressView = new DataView({\n model: this.model,\n className: 'p-3',\n showEmptyValues: true,\n emptyValueText: '—',\n columns: 2,\n fields: [\n { name: 'address_line1', label: 'Address Line 1', cols: 12 },\n { name: 'address_city', label: 'City', cols: 4 },\n { name: 'address_state', label: 'State', cols: 4 },\n { name: 'address_zip', label: 'ZIP', cols: 4 },\n { name: 'address_country', label: 'Country', cols: 6 }\n ]\n });\n\n // Metadata Tab\n this.metadataView = new DataView({\n model: this.model,\n className: 'p-3',\n showEmptyValues: true,\n emptyValueText: '—',\n columns: 2,\n fields: [\n { name: 'id', label: 'Record ID', cols: 6 },\n { name: 'created', label: 'Created', formatter: 'datetime', cols: 6 },\n { name: 'modified', label: 'Last Modified', formatter: 'datetime', cols: 6 }\n ]\n });\n\n const tabs = {\n 'Overview': this.overviewView,\n 'Carrier': this.carrierView,\n 'Address': this.addressView,\n 'Metadata': this.metadataView\n };\n\n this.tabView = new TabView({\n containerId: 'phone-tabs',\n tabs,\n activeTab: 'Overview'\n });\n this.addChild(this.tabView);\n\n // Minimal ContextMenu\n const menuItems = [\n { label: 'Refresh Lookup', action: 'refresh-lookup', icon: 'bi-arrow-repeat' },\n { type: 'divider' },\n { label: 'Delete Record', action: 'delete-phone', icon: 'bi-trash', danger: true }\n ];\n\n const ctxMenu = new ContextMenu({\n containerId: 'phone-context-menu',\n className: 'context-menu-view header-menu-absolute',\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: menuItems\n }\n });\n this.addChild(ctxMenu);\n }\n\n // Actions\n\n async onActionRefreshLookup() {\n const number = this.model.get('phone_number');\n if (!number) {\n this.getApp()?.toast?.warning?.('No phone number to lookup');\n return;\n }\n\n try {\n this.getApp()?.toast?.info?.('Refreshing lookup...');\n // Force refresh lookup and update current model\n const resp = await PhoneNumber.lookup(number, { force_refresh: true });\n if (resp.success && resp.data) {\n this.model.set(resp.data);\n await this.render();\n this.getApp()?.toast?.success?.('Lookup refreshed');\n } else {\n const msg = resp.error || 'Lookup failed';\n this.getApp()?.toast?.error?.(msg);\n }\n } catch (e) {\n this.getApp()?.toast?.error?.(e.message || 'Lookup failed');\n }\n }\n\n async onActionDeletePhone() {\n const confirmed = await Dialog.confirm(\n `Are you sure you want to delete the record for \"${this.model.get('phone_number') || 'this number'}\"?`,\n 'Confirm Deletion',\n { confirmClass: 'btn-danger', confirmText: 'Delete' }\n );\n\n if (!confirmed) return;\n\n try {\n const resp = await this.model.destroy();\n if (resp?.success) {\n this.emit('phone:deleted', { model: this.model });\n } else {\n this.getApp()?.toast?.error?.('Delete failed');\n }\n } catch (e) {\n this.getApp()?.toast?.error?.(e.message || 'Delete failed');\n }\n }\n\n static async show(phone_number) {\n const resp = await PhoneNumber.lookup(phone_number);\n if (resp?.model) {\n const view = new PhoneNumberView({ model: resp.model });\n const dialog = new Dialog({\n header: false,\n size: 'lg',\n body: view,\n buttons: [{ text: 'Close', class: 'btn-secondary', dismiss: true }]\n });\n await dialog.render(true, document.body);\n dialog.show();\n return dialog;\n }\n Dialog.alert({ message: `Could not find phone data for number: ${phone_number}`, type: 'warning' });\n return null;\n }\n}\n\nPhoneNumberView.MODEL_CLASS = PhoneNumber;\n\nexport default PhoneNumberView;\n","/**\n * PhoneNumberTablePage - PhoneHub numbers management using TablePage component\n * Mirrors the GeoLocatedIPTablePage pattern with minimal, consistent configuration.\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { PhoneNumberList, PhoneNumber } from '@core/models/Phonehub.js';\nimport PhoneNumberView from './PhoneNumberView.js';\n\nclass PhoneNumberTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n\n // Identity\n name: 'admin_phonehub_numbers',\n pageName: 'Phone Numbers',\n router: 'admin/phonehub/numbers',\n\n // Data source\n Collection: PhoneNumberList,\n\n // Item view configuration\n itemView: PhoneNumberView,\n viewDialogOptions: {\n header: false,\n // size: 'xl'\n },\n\n // Column definitions\n columns: [\n { key: 'phone_number', label: 'Phone Number', sortable: true },\n { key: 'carrier', label: 'Carrier', sortable: true, formatter: \"default('—')\" },\n { key: 'line_type', label: 'Line Type', sortable: true, formatter: \"capitalize\" },\n { key: 'is_mobile', label: 'Mobile', formatter: 'yesnoicon' },\n { key: 'is_voip', label: 'VOIP', formatter: 'yesnoicon' },\n { key: 'is_valid', label: 'Valid', formatter: 'yesnoicon' },\n { key: 'registered_owner', label: 'Owner', sortable: true, formatter: \"default('—')\" },\n { key: 'owner_type', label: 'Owner Type', formatter: \"capitalize\" },\n { key: 'last_lookup_at|relative', label: 'Last Lookup', sortable: true},\n ],\n\n // Table features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Row action\n clickAction: 'view',\n\n // Toolbar\n showRefresh: true,\n showAdd: true,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No phone numbers found.',\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n },\n\n tableViewOptions: {\n addButtonLabel: \"Lookup\",\n addButtonIcon: 'bi-search',\n onAdd: (evt) => {\n evt.preventDefault();\n // Implement the logic for adding a new record\n this.onLookup();\n }\n }\n });\n }\n\n async onLookup() {\n // Implement the logic for adding a new record\n const data = await this.getApp().showForm({\n title: \"Lookup Phone Number\",\n fields: [\n {\n name: 'number',\n type: 'text',\n required: true\n }\n ]\n });\n if (data && data.number) {\n const resp = await PhoneNumber.lookup(data.number);\n if (resp.model) {\n this.tableView._onRowView(resp);\n }\n }\n }\n}\n\nexport default PhoneNumberTablePage;\n","/**\n * SMSView - Detailed view for a PhoneHub SMS record\n *\n * Mirrors GeoIPView structure:\n * - Header with key info (direction, from/to, status)\n * - Tabbed content using TabView and DataView sections\n * - Minimal context menu for refresh and delete\n */\n\nimport View from '@core/View.js';\nimport TabView from '@core/views/navigation/TabView.js';\nimport DataView from '@core/views/data/DataView.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\nimport { SMS } from '@core/models/Phonehub.js';\n\nclass SMSView extends View {\n constructor(options = {}) {\n super({\n className: 'sms-view',\n ...options\n });\n\n this.model = options.model || new SMS(options.data || {});\n\n this.template = `\n <div class=\"sms-view-container\">\n <!-- Header -->\n <div class=\"d-flex justify-content-between align-items-center mb-4\">\n <!-- Left Side: Icon & Info -->\n <div class=\"d-flex align-items-center gap-3\">\n <div class=\"fs-1 text-primary\">\n <i class=\"bi bi-chat-dots\"></i>\n </div>\n <div>\n <h3 class=\"mb-1\">\n {{#model.direction}}{{model.direction|capitalize}}{{/model.direction}}\n {{^model.direction}}Message{{/model.direction}}\n <small class=\"text-muted ms-2\">\n {{#model.status}}[{{model.status|capitalize}}]{{/model.status}}\n </small>\n </h3>\n <div class=\"text-muted small\">\n {{#model.from_number}}From: {{model.from_number}}{{/model.from_number}}\n {{#model.to_number}} · To: {{model.to_number}}{{/model.to_number}}\n </div>\n <div class=\"text-muted small mt-1\">\n {{#model.provider}}Provider: {{model.provider|capitalize}}{{/model.provider}}\n {{#model.provider_message_id}} · SID: {{model.provider_message_id}}{{/model.provider_message_id}}\n </div>\n </div>\n </div>\n\n <!-- Right Side: Actions -->\n <div class=\"d-flex align-items-center gap-4\">\n <div data-container=\"sms-context-menu\"></div>\n </div>\n </div>\n\n <!-- Tabs -->\n <div data-container=\"sms-tabs\"></div>\n </div>\n `;\n }\n\n async onInit() {\n // Message Tab\n this.messageView = new DataView({\n model: this.model,\n className: 'p-3',\n showEmptyValues: true,\n emptyValueText: '—',\n columns: 2,\n fields: [\n { name: 'direction', label: 'Direction', formatter: 'capitalize', cols: 4 },\n { name: 'status', label: 'Status', formatter: 'capitalize', cols: 4 },\n { name: 'from_number', label: 'From', cols: 6 },\n { name: 'to_number', label: 'To', cols: 6 },\n { name: 'body', label: 'Message Body', cols: 12 }\n ]\n });\n\n // Delivery Tab\n this.deliveryView = new DataView({\n model: this.model,\n className: 'p-3',\n showEmptyValues: true,\n emptyValueText: '—',\n columns: 2,\n fields: [\n { name: 'provider', label: 'Provider', formatter: 'capitalize', cols: 6 },\n { name: 'provider_message_id', label: 'Provider Message ID', cols: 6 },\n { name: 'sent_at', label: 'Sent At', formatter: 'datetime', cols: 6 },\n { name: 'delivered_at', label: 'Delivered At', formatter: 'datetime', cols: 6 },\n { name: 'error_code', label: 'Error Code', cols: 6 },\n { name: 'error_message', label: 'Error Message', cols: 12 }\n ]\n });\n\n // Metadata Tab\n this.metadataView = new DataView({\n model: this.model,\n className: 'p-3',\n showEmptyValues: true,\n emptyValueText: '—',\n columns: 2,\n fields: [\n { name: 'id', label: 'Record ID', cols: 6 },\n { name: 'created', label: 'Created', formatter: 'datetime', cols: 6 },\n { name: 'modified', label: 'Last Modified', formatter: 'datetime', cols: 6 }\n ]\n });\n\n const tabs = {\n 'Message': this.messageView,\n 'Delivery': this.deliveryView,\n 'Metadata': this.metadataView\n };\n\n this.tabView = new TabView({\n containerId: 'sms-tabs',\n tabs,\n activeTab: 'Message'\n });\n this.addChild(this.tabView);\n\n // ContextMenu (minimal)\n const menuItems = [\n { label: 'Refresh', action: 'refresh-sms', icon: 'bi-arrow-repeat' },\n { type: 'divider' },\n { label: 'Delete Message', action: 'delete-sms', icon: 'bi-trash', danger: true }\n ];\n\n const ctxMenu = new ContextMenu({\n containerId: 'sms-context-menu',\n className: 'context-menu-view header-menu-absolute',\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: menuItems\n }\n });\n this.addChild(ctxMenu);\n }\n\n // Actions\n\n async onActionRefreshSms() {\n try {\n this.getApp()?.toast?.info?.('Refreshing message...');\n await this.model.fetch();\n await this.render();\n this.getApp()?.toast?.success?.('Message refreshed');\n } catch (e) {\n this.getApp()?.toast?.error?.(e.message || 'Refresh failed');\n }\n }\n\n async onActionDeleteSms() {\n const title = 'Confirm Deletion';\n const msg = `Are you sure you want to delete this message?`;\n const confirmed = await Dialog.confirm(msg, title, {\n confirmClass: 'btn-danger',\n confirmText: 'Delete'\n });\n\n if (!confirmed) return;\n\n try {\n const resp = await this.model.destroy();\n if (resp?.success) {\n this.emit('sms:deleted', { model: this.model });\n } else {\n this.getApp()?.toast?.error?.('Delete failed');\n }\n } catch (e) {\n this.getApp()?.toast?.error?.(e.message || 'Delete failed');\n }\n }\n}\n\nSMSView.MODEL_CLASS = SMS;\n\nexport default SMSView;","/**\n * SMSTablePage - PhoneHub SMS management using TablePage component\n * Mirrors the GeoLocatedIPTablePage pattern with minimal, consistent configuration.\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { SMSList } from '@core/models/Phonehub.js';\nimport SMSView from './SMSView.js';\n\nclass SMSTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n\n // Identity\n name: 'admin_phonehub_sms',\n pageName: 'SMS Messages',\n router: 'admin/phonehub/sms',\n\n // Data source\n Collection: SMSList,\n\n // Item view configuration\n itemView: SMSView,\n viewDialogOptions: {\n header: false,\n size: 'xl'\n },\n\n // Column definitions\n columns: [\n { key: 'direction', label: 'Direction', sortable: true },\n { key: 'from_number', label: 'From', sortable: true, formatter: \"default('—')\" },\n { key: 'to_number', label: 'To', sortable: true, formatter: \"default('—')\" },\n { key: 'status', label: 'Status', sortable: true },\n { key: 'provider', label: 'Provider', sortable: true, formatter: \"default('—')\" },\n { key: 'body', label: 'Message', formatter: \"default('—')\" },\n { key: 'sent_at', label: 'Sent At', sortable: true, formatter: 'datetime' },\n { key: 'delivered_at', label: 'Delivered At', sortable: true, formatter: 'datetime' },\n { key: 'created', label: 'Created', sortable: true, formatter: 'datetime' }\n ],\n\n // Table features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Row action\n clickAction: 'view',\n\n // Toolbar\n showRefresh: true,\n showAdd: false,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No SMS messages found.',\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n }\n });\n }\n}\n\nexport default SMSTablePage;","import Page from '@core/Page.js';\nimport { MetricsChart, PieChart } from '@ext/charts/index.js';\nimport TableView from '@core/views/table/TableView.js';\nimport { PushDeliveryList } from '@core/models/Push.js';\n\nclass PushDashboardPage extends Page {\n constructor(options = {}) {\n super({\n ...options,\n title: 'Push Notifications Dashboard',\n className: 'push-dashboard-page'\n });\n }\n\n async getTemplate() {\n return `\n <div class=\"container-fluid\">\n <h1 class=\"h3 mb-4\">Push Notifications</h1>\n <div class=\"row\">\n <!-- Stat cards -->\n </div>\n <div class=\"row\">\n <div class=\"col-xl-8 col-lg-7\">\n <div class=\"card shadow mb-4\">\n <div class=\"card-header\">Notifications Over Time</div>\n <div class=\"card-body\" data-container=\"deliveries-chart\"></div>\n </div>\n </div>\n <div class=\"col-xl-4 col-lg-5\">\n <div class=\"card shadow mb-4\">\n <div class=\"card-header\">Delivery Status</div>\n <div class=\"card-body\" data-container=\"status-chart\"></div>\n </div>\n </div>\n </div>\n <div class=\"row\">\n <div class=\"col-lg-6 mb-4\" data-container=\"recent-deliveries\"></div>\n <div class=\"col-lg-6 mb-4\" data-container=\"failed-deliveries\"></div>\n </div>\n </div>\n `;\n }\n\n async onInit() {\n this.deliveriesChart = new MetricsChart({\n containerId: 'deliveries-chart',\n endpoint: '/api/metrics/fetch',\n slugs: ['push_sent', 'push_failed'],\n chartType: 'line'\n });\n this.addChild(this.deliveriesChart);\n\n this.statusChart = new PieChart({\n containerId: 'status-chart',\n endpoint: '/api/account/devices/push/stats' // Assuming this returns data for pie chart\n });\n this.addChild(this.statusChart);\n\n this.recentDeliveries = new TableView({\n containerId: 'recent-deliveries',\n title: 'Recent Deliveries',\n Collection: new PushDeliveryList({ params: { _sort: '-created', _limit: 5 } }),\n columns: [{ key: 'title', label: 'Title' }, { key: 'status', label: 'Status', formatter: 'badge' }]\n });\n this.addChild(this.recentDeliveries);\n\n this.failedDeliveries = new TableView({\n containerId: 'failed-deliveries',\n title: 'Failed Deliveries',\n Collection: new PushDeliveryList({ params: { status: 'failed', _sort: '-created', _limit: 5 } }),\n columns: [{ key: 'title', label: 'Title' }, { key: 'error_message', label: 'Error' }]\n });\n this.addChild(this.failedDeliveries);\n }\n}\n\nexport default PushDashboardPage;\n","import TablePage from '@core/pages/TablePage.js';\nimport { PushConfigList, PushConfigForms } from '@core/models/Push.js';\n\nclass PushConfigTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_push_configs',\n pageName: 'Push Configurations',\n router: \"admin/push/configs\",\n Collection: PushConfigList,\n formCreate: PushConfigForms.create,\n formEdit: PushConfigForms.edit,\n\n columns: [\n { key: 'id', label: 'ID', width: '70px' },\n { key: 'name', label: 'Name' },\n { key: 'group.name', label: 'Group', formatter: \"default('Default')\" },\n { key: 'fcm_sender_id', label: 'Project ID' },\n { key: 'is_active', label: 'Active', format: 'boolean' },\n ],\n\n searchable: true,\n sortable: true,\n paginated: true,\n showRefresh: true,\n showAdd: true,\n showExport: true,\n actions: [\"edit\", \"delete\"],\n tableOptions: {\n pageSizes: [10, 25, 50],\n defaultPageSize: 25,\n emptyMessage: 'No push configurations found.',\n emptyIcon: 'bi-gear',\n\n }\n });\n }\n}\n\nexport default PushConfigTablePage;\n","import TablePage from '@core/pages/TablePage.js';\nimport { PushTemplateList, PushTemplateForms } from '@core/models/Push.js';\n\nclass PushTemplateTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_push_templates',\n pageName: 'Push Templates',\n router: \"admin/push/templates\",\n Collection: PushTemplateList,\n formCreate: PushTemplateForms.create,\n formEdit: PushTemplateForms.edit,\n\n columns: [\n { key: 'id', label: 'ID', width: '70px' },\n { key: 'name', label: 'Name' },\n { key: 'category', label: 'Category' },\n { key: 'group.name', label: 'Group', formatter: \"default('Default')\" },\n { key: 'priority', label: 'Priority' },\n { key: 'is_active', label: 'Active', format: 'boolean' },\n ],\n\n searchable: true,\n sortable: true,\n paginated: true,\n showRefresh: true,\n showAdd: true,\n showExport: true,\n\n tableOptions: {\n pageSizes: [10, 25, 50],\n defaultPageSize: 25,\n emptyMessage: 'No push templates found.',\n emptyIcon: 'bi-file-earmark-text',\n actions: [\"edit\", \"delete\"],\n }\n });\n }\n}\n\nexport default PushTemplateTablePage;\n","import View from '@core/View.js';\n\nclass PushDeliveryView extends View {\n constructor(options = {}) {\n super({\n className: 'push-delivery-view',\n ...options\n });\n this.model = options.model;\n }\n\n getTemplate() {\n return `\n <div class=\"p-3\">\n <div class=\"phone-mockup\">\n <div class=\"phone-screen\">\n <div class=\"notification\">\n <div class=\"notification-header\">\n <i class=\"bi bi-app-indicator\"></i>\n <strong>Your App</strong>\n <span class=\"ms-auto small text-muted\">now</span>\n </div>\n <div class=\"notification-body\">\n <div class=\"fw-bold\">{{model.title}}</div>\n <div>{{model.body}}</div>\n </div>\n </div>\n </div>\n </div>\n <div class=\"mt-3\">\n <h5>Delivery Details</h5>\n <p><strong>Status:</strong> <span class=\"badge {{model.status|badge}}\">{{model.status}}</span></p>\n <p><strong>Error:</strong> {{model.error_message|default('None')}}</p>\n </div>\n </div>\n `;\n }\n}\n\nexport default PushDeliveryView;\n","import TablePage from '@core/pages/TablePage.js';\nimport { PushDeliveryList } from '@core/models/Push.js';\nimport PushDeliveryView from './PushDeliveryView.js';\n\nclass PushDeliveryTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_push_deliveries',\n pageName: 'Push Deliveries',\n router: \"admin/push/deliveries\",\n Collection: PushDeliveryList,\n itemViewClass: PushDeliveryView,\n viewDialogOptions: {\n header: false,\n size: 'md'\n },\n\n columns: [\n { key: 'id', label: 'ID', width: '70px' },\n { key: 'created', label: 'Timestamp', formatter: 'datetime' },\n { key: 'user.display_name', label: 'User' },\n { key: 'device.device_name', label: 'Device' },\n { key: 'title', label: 'Title' },\n { key: 'category', label: 'Category' },\n { key: 'status', label: 'Status', formatter: 'badge' },\n ],\n \n searchable: true,\n sortable: true,\n paginated: true,\n showRefresh: true,\n\n tableOptions: {\n pageSizes: [10, 25, 50],\n defaultPageSize: 25,\n emptyMessage: 'No deliveries found.',\n emptyIcon: 'bi-send',\n actions: [\"view\"],\n }\n });\n }\n}\n\nexport default PushDeliveryTablePage;\n","import View from '@core/View.js';\nimport DataView from '@core/views/data/DataView.js';\n\nclass PushDeviceView extends View {\n constructor(options = {}) {\n super({\n className: 'push-device-view',\n ...options\n });\n this.model = options.model;\n }\n\n getTemplate() {\n return `\n <div class=\"p-3\">\n <h3>{{model.device_name}}</h3>\n <p class=\"text-muted\">{{model.user.display_name}}</p>\n <div data-container=\"data-view\"></div>\n </div>\n `;\n }\n\n onInit() {\n this.dataView = new DataView({\n containerId: 'data-view',\n model: this.model,\n fields: [\n { name: 'platform', label: 'Platform', format: 'badge' },\n { name: 'push_enabled', label: 'Push Enabled', format: 'boolean' },\n { name: 'app_version', label: 'App Version' },\n { name: 'os_version', label: 'OS Version' },\n { name: 'last_seen', label: 'Last Seen', format: 'datetime' },\n { name: 'push_preferences', label: 'Preferences', format: 'json' },\n ]\n });\n this.addChild(this.dataView);\n }\n}\n\nexport default PushDeviceView;\n","import TablePage from '@core/pages/TablePage.js';\nimport { PushDeviceList } from '@core/models/Push.js';\nimport PushDeviceView from './PushDeviceView.js';\n\nclass PushDeviceTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_push_devices',\n pageName: 'Registered Devices',\n router: \"admin/push/devices\",\n Collection: PushDeviceList,\n itemViewClass: PushDeviceView,\n viewDialogOptions: {\n header: false,\n size: 'lg'\n },\n\n columns: [\n { key: 'id', label: 'ID', width: '70px' },\n { key: 'user.display_name', label: 'User' },\n { key: 'device_name', label: 'Device Name' },\n { key: 'platform', label: 'Platform', formatter: 'badge' },\n { key: 'app_version', label: 'App Version' },\n { key: 'push_enabled', label: 'Push Enabled', format: 'boolean' },\n { key: 'last_seen', label: 'Last Seen', formatter: 'datetime' },\n ],\n \n searchable: true,\n sortable: true,\n paginated: true,\n showRefresh: true,\n\n tableOptions: {\n pageSizes: [10, 25, 50],\n defaultPageSize: 25,\n emptyMessage: 'No devices found.',\n emptyIcon: 'bi-phone',\n actions: [\"view\", \"delete\"],\n }\n });\n }\n}\n\nexport default PushDeviceTablePage;\n","/**\n * JobStatsView - Job statistics overview cards\n */\n\nimport View from '@core/View.js';\nimport { Job } from '@core/models/Job.js';\n\nexport default class JobStatsView extends View {\n constructor(options = {}) {\n super({\n className: 'job-stats-section',\n ...options\n });\n\n this.stats = {\n pending: 0,\n running: 0,\n completed: 0,\n failed: 0,\n scheduled: 0\n };\n\n this.template = `\n <div class=\"job-stats-header mb-4\">\n <div class=\"row\">\n <div class=\"col-xl-2 col-lg-4 col-md-6 col-12 mb-3\">\n <div class=\"card h-100 border-0 shadow\">\n <div class=\"card-body\">\n <div class=\"d-flex justify-content-between align-items-start\">\n <div>\n <h6 class=\"card-title text-muted mb-2\">Pending</h6>\n <h3 class=\"mb-1 fw-bold\">{{stats.pending}}</h3>\n <span class=\"badge bg-primary-subtle text-primary\">\n <i class=\"bi bi-hourglass\"></i> Queued\n </span>\n </div>\n <div class=\"text-primary\">\n <i class=\"bi bi-clock fs-2\"></i>\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <div class=\"col-xl-2 col-lg-4 col-md-6 col-12 mb-3\">\n <div class=\"card h-100 border-0 shadow\">\n <div class=\"card-body\">\n <div class=\"d-flex justify-content-between align-items-start\">\n <div>\n <h6 class=\"card-title text-muted mb-2\">Running</h6>\n <h3 class=\"mb-1 fw-bold\">{{stats.running}}</h3>\n <span class=\"badge bg-success-subtle text-success\">\n <i class=\"bi bi-arrow-repeat\"></i> Active\n </span>\n </div>\n <div class=\"text-success\">\n <i class=\"bi bi-play-circle fs-2\"></i>\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <div class=\"col-xl-2 col-lg-4 col-md-6 col-12 mb-3\">\n <div class=\"card h-100 border-0 shadow\">\n <div class=\"card-body\">\n <div class=\"d-flex justify-content-between align-items-start\">\n <div>\n <h6 class=\"card-title text-muted mb-2\">Scheduled</h6>\n <h3 class=\"mb-1 fw-bold\">{{stats.scheduled}}</h3>\n <span class=\"badge bg-warning-subtle text-warning\">\n <i class=\"bi bi-calendar-event\"></i> Planned\n </span>\n </div>\n <div class=\"text-warning\">\n <i class=\"bi bi-calendar3 fs-2\"></i>\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <div class=\"col-xl-3 col-lg-6 col-md-6 col-12 mb-3\">\n <div class=\"card h-100 border-0 shadow\">\n <div class=\"card-body\">\n <div class=\"d-flex justify-content-between align-items-start\">\n <div>\n <h6 class=\"card-title text-muted mb-2\">Completed</h6>\n <h3 class=\"mb-1 fw-bold\">{{stats.completed}}</h3>\n <span class=\"badge bg-info-subtle text-info\">\n <i class=\"bi bi-check-circle\"></i> Done\n </span>\n </div>\n <div class=\"text-info\">\n <i class=\"bi bi-check-square fs-2\"></i>\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <div class=\"col-xl-3 col-lg-6 col-md-6 col-12 mb-3\">\n <div class=\"card h-100 border-0 shadow\">\n <div class=\"card-body\">\n <div class=\"d-flex justify-content-between align-items-start\">\n <div>\n <h6 class=\"card-title text-muted mb-2\">Failed</h6>\n <h3 class=\"mb-1 fw-bold\">{{stats.failed}}</h3>\n <span class=\"badge bg-danger-subtle text-danger\">\n <i class=\"bi bi-x-octagon\"></i> Errors\n </span>\n </div>\n <div class=\"text-danger\">\n <i class=\"bi bi-exclamation-triangle fs-2\"></i>\n </div>\n </div>\n </div>\n </div>\n </div>\n </div>\n </div>\n `;\n }\n\n _onModelChange() {\n this.loadStats();\n if (this.isMounted()) {\n this.render();\n }\n }\n\n async loadStats() {\n this.stats = this.model.attributes.totals;\n }\n}\n","/**\n * JobHealthView - System health overview\n */\n\nimport View from '@core/View.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\nimport { Job } from '@core/models/Job.js';\n\nexport default class JobHealthView extends View {\n constructor(options = {}) {\n super({\n className: 'job-health-section',\n ...options\n });\n\n this.health = {\n status: 'unknown',\n runners: { active: 0, total: 0 },\n channels: []\n };\n\n this.template = `\n <div class=\"job-health-header mb-4\">\n <div class=\"card border-0 shadow\">\n <div class=\"card-body\">\n <div class=\"row align-items-center\">\n <div class=\"col-md-6\">\n <div class=\"d-flex align-items-center\">\n <div class=\"health-indicator me-3\">\n <i class=\"bi bi-circle-fill fs-4 {{healthStatusClass}}\"></i>\n </div>\n <div>\n <h5 class=\"mb-1\">Service Health: {{health.overall_status|capitalize}}</h5>\n <small class=\"text-muted d-block\">\n Workers: {{health.runners.active}}/{{health.runners.total}} active\n </small>\n <small class=\"text-muted d-block\">\n Scheduler:\n <span class=\"{{schedulerStatusClass}} fw-bold\">\n <i class=\"bi {{schedulerIcon}} me-1\"></i>{{schedulerStatusText}}\n </span>\n </small>\n </div>\n </div>\n </div>\n <div class=\"col-md-6\">\n <div class=\"d-flex justify-content-end\">\n <div class=\"btn-group\">\n <button class=\"btn btn-sm btn-outline-primary\" data-action=\"refresh-health\">\n <i class=\"bi bi-arrow-clockwise\"></i> Refresh\n </button>\n <button class=\"btn btn-sm btn-outline-secondary\" data-action=\"system-settings\">\n <i class=\"bi bi-gear\"></i> Settings\n </button>\n </div>\n </div>\n </div>\n </div>\n {{#health.channelsArray.length}}\n <div class=\"row mt-3\">\n <div class=\"col-12\">\n <small class=\"text-muted d-block mb-2\">Channel Status:</small>\n <div class=\"d-flex flex-wrap gap-2\">\n {{#health.channelsArray}}\n <span class=\"badge {{statusBadgeClass}} d-flex align-items-center\">\n <i class=\"bi {{statusIcon}} me-1\"></i>\n {{channel}} ({{queued}} queued, {{inflight}} inflight)\n </span>\n {{/health.channelsArray}}\n </div>\n </div>\n </div>\n {{/health.channelsArray.length}}\n </div>\n </div>\n </div>\n `;\n }\n\n _onModelChange() {\n this.loadHealth();\n if (this.isMounted()) {\n this.render();\n }\n }\n\n async loadHealth() {\n if (!this.model._.totals) return;\n const data = this.model.attributes\n // Determine overall health status\n let overall_status = 'healthy';\n if (data.totals.runners_active === 0) {\n overall_status = 'critical';\n } else if (!data.scheduler.active) {\n overall_status = 'warning';\n }\n this.health = {\n overall_status,\n channels: data.channels,\n runners: {\n active: data.totals.runners_active,\n total: data.runners.length\n },\n totals: data.totals,\n scheduler: data.scheduler\n };\n\n this.healthStatusClass = this.getHealthStatusClass(this.health.overall_status);\n this.setupChannelDisplay();\n this.setupSchedulerDisplay();\n\n }\n\n setupChannelDisplay() {\n if (!this.health.channels) return;\n\n this.health.channelsArray = Object.values(this.health.channels).map(channel => {\n let channelStatus = 'healthy';\n\n // Determine status based on queue backlog and runner availability\n const totalJobs = (channel.queued_count || 0) + (channel.inflight_count || 0);\n if (totalJobs > 50) {\n channelStatus = 'warning';\n }\n if (totalJobs > 100 || channel.runners === 0) {\n channelStatus = 'critical';\n }\n\n return {\n ...channel,\n status: channelStatus,\n statusBadgeClass: this.getChannelBadgeClass(channelStatus),\n statusIcon: this.getChannelIcon(channelStatus),\n queued: channel.queued_count || 0,\n inflight: channel.inflight_count || 0,\n // For backward compatibility, map to old names\n pending: channel.queued_count || 0,\n running: channel.inflight_count || 0\n };\n });\n }\n\n setupSchedulerDisplay() {\n if (!this.health.scheduler) {\n this.schedulerStatusText = 'Unknown';\n this.schedulerStatusClass = 'text-muted';\n this.schedulerIcon = 'bi-question-circle-fill';\n return;\n }\n\n if (this.health.scheduler.active) {\n this.schedulerStatusText = 'Running';\n this.schedulerStatusClass = 'text-success';\n this.schedulerIcon = 'bi-check-circle-fill';\n } else {\n this.schedulerStatusText = 'Stopped';\n this.schedulerStatusClass = 'text-danger';\n this.schedulerIcon = 'bi-x-octagon-fill';\n }\n }\n\n getHealthStatusClass(status) {\n const classes = {\n 'healthy': 'text-success',\n 'warning': 'text-warning',\n 'critical': 'text-danger',\n 'unknown': 'text-muted'\n };\n return classes[status] || 'text-muted';\n }\n\n getChannelBadgeClass(status) {\n const classes = {\n 'healthy': 'bg-success',\n 'warning': 'bg-warning',\n 'critical': 'bg-danger'\n };\n return classes[status] || 'bg-secondary';\n }\n\n getChannelIcon(status) {\n const icons = {\n 'healthy': 'bi-check-circle-fill',\n 'warning': 'bi-exclamation-triangle-fill',\n 'critical': 'bi-x-octagon-fill'\n };\n return icons[status] || 'bi-dash-circle-fill';\n }\n\n async onActionRefreshHealth(event, element) {\n try {\n element.disabled = true;\n await this.model.fetch();\n } catch (error) {\n console.error('Failed to refresh health:', error);\n } finally {\n element.disabled = false;\n }\n }\n\n async onActionSystemSettings() {\n await Dialog.showAlert({\n title: 'System Settings',\n message: 'System settings interface coming soon!',\n type: 'info'\n });\n }\n}\n","/**\n * JobDetailsView - Comprehensive job details interface\n *\n * Features:\n * - Clean header with job status, timing, and function info\n * - Tabbed interface for Overview, Payload, Events, and Logs\n * - Integrated with Table and ContextMenu components\n * - Clean Bootstrap 5 styling\n */\n\nimport View from '@core/View.js';\nimport TabView from '@core/views/navigation/TabView.js';\nimport TableView from '@core/views/table/TableView.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport { Job, JobEventList, JobLogList, JobForms } from '@core/models/Job.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\n\nclass JobDetailsView extends View {\n constructor(options = {}) {\n super({\n className: 'job-details-view',\n ...options\n });\n\n // Job model instance\n this.model = options.model || new Job(options.data || {});\n\n // Tab views\n this.tabView = null;\n this.overviewView = null;\n this.payloadView = null;\n this.eventsView = null;\n this.logsView = null;\n\n // Auto refresh\n this.autoRefreshInterval = null;\n\n // Set template\n this.template = `\n <div class=\"job-details-container\">\n <!-- Job Header -->\n <div class=\"d-flex justify-content-between align-items-start mb-4\">\n <!-- Left Side: Primary Identity -->\n <div class=\"d-flex align-items-center gap-3\">\n <div class=\"avatar-placeholder rounded-circle bg-light d-flex align-items-center justify-content-center\" style=\"width: 80px; height: 80px;\">\n <i class=\"bi {{model.statusIcon}} text-secondary\" style=\"font-size: 40px;\"></i>\n </div>\n <div>\n <h3 class=\"mb-1\">{{model.func|truncate(32)|default('Unknown Function')}}</h3>\n <div class=\"text-muted small\">\n <span>ID: {{model.id}}</span>\n <span class=\"mx-2\">|</span>\n <span>Channel: <span class=\"badge bg-primary\">{{model.channel}}</span></span>\n {{#model.runner_id}}\n <span class=\"mx-2\">|</span>\n <span>Runner: {{model.runner_id|truncate(16)}}</span>\n {{/model.runner_id}}\n </div>\n <div class=\"text-muted small mt-2\">\n <div>Created: {{model.created|datetime}}</div>\n {{#model.started_at}}\n <div>Started: {{model.started_at|datetime}}</div>\n {{/model.started_at}}\n {{#model.finished_at}}\n <div>Finished: {{model.finished_at|datetime}}</div>\n {{/model.finished_at}}\n </div>\n </div>\n </div>\n\n <!-- Right Side: Status & Actions -->\n <div class=\"d-flex align-items-start gap-4\">\n <div class=\"text-end\">\n <div class=\"d-flex align-items-center gap-2\">\n <span class=\"badge {{model.statusBadgeClass}} fs-6\">\n <i class=\"bi {{model.statusIcon}}\"></i> {{model.status|uppercase}}\n </span>\n {{#model.cancel_requested}}\n <span class=\"badge bg-warning ms-1\">\n <i class=\"bi bi-exclamation-triangle\"></i> Cancel Requested\n </span>\n {{/model.cancel_requested}}\n </div>\n {{#model.formattedDuration}}\n <div class=\"text-muted small mt-1\">Duration: {{model.formattedDuration}}</div>\n {{/model.formattedDuration}}\n </div>\n <div data-container=\"job-context-menu\"></div>\n </div>\n </div>\n\n <!-- Tab Container -->\n <div data-container=\"job-details-tabs\"></div>\n </div>\n `;\n }\n\n async onInit() {\n // Create Overview view\n this.overviewView = new View({\n template: `\n <div class=\"job-overview-tab\">\n <div class=\"card border-0 bg-light mb-3\">\n <div class=\"card-body\">\n <div class=\"row\">\n <div class=\"col-md-6\">\n <div class=\"mb-3\">\n <label class=\"form-label fw-bold text-muted small\">Job ID</label>\n <div class=\"font-monospace\">{{model.id}}</div>\n </div>\n <div class=\"mb-3\">\n <label class=\"form-label fw-bold text-muted small\">Function</label>\n <div class=\"font-monospace\">{{model.func}}</div>\n </div>\n <div class=\"mb-3\">\n <label class=\"form-label fw-bold text-muted small\">Channel</label>\n <div>\n <span class=\"badge bg-primary\">{{model.channel}}</span>\n </div>\n </div>\n {{#model.runner_id}}\n <div class=\"mb-3\">\n <label class=\"form-label fw-bold text-muted small\">Runner</label>\n <div class=\"font-monospace small\">{{model.runner_id}}</div>\n </div>\n {{/model.runner_id}}\n </div>\n <div class=\"col-md-6\">\n <div class=\"mb-3\">\n <label class=\"form-label fw-bold text-muted small\">Status</label>\n <div>\n <span class=\"badge {{model.statusBadgeClass}} fs-6\">\n <i class=\"bi {{model.statusIcon}}\"></i> {{model.status|uppercase}}\n </span>\n {{#model.cancel_requested}}\n <span class=\"badge bg-warning ms-1\">\n <i class=\"bi bi-exclamation-triangle\"></i> Cancel Requested\n </span>\n {{/model.cancel_requested}}\n </div>\n </div>\n <div class=\"mb-3\">\n <label class=\"form-label fw-bold text-muted small\">Created</label>\n <div>{{model.created|datetime}}</div>\n </div>\n {{#model.started_at}}\n <div class=\"mb-3\">\n <label class=\"form-label fw-bold text-muted small\">Started</label>\n <div>{{model.started_at|datetime}}</div>\n </div>\n {{/model.started_at}}\n {{#model.finished_at}}\n <div class=\"mb-3\">\n <label class=\"form-label fw-bold text-muted small\">Finished</label>\n <div>{{model.finished_at|datetime}}</div>\n </div>\n {{/model.finished_at}}\n {{#model.duration_ms}}\n <div class=\"mb-3\">\n <label class=\"form-label fw-bold text-muted small\">Duration</label>\n <div>{{model.formattedDuration}}</div>\n </div>\n {{/model.duration_ms}}\n </div>\n </div>\n </div>\n </div>\n </div>\n `,\n model: this.model\n });\n\n // Create Payload view\n this.payloadView = new View({\n template: `\n <div class=\"job-payload-tab\">\n <pre class=\"bg-light p-3 rounded\"><code>{{{model.payload|json}}}</code></pre>\n </div>\n `,\n model: this.model,\n });\n\n // Create Events table\n const eventsCollection = new JobEventList({\n params: { job: this.model.get('id'), size: 10 }\n });\n this.eventsView = new TableView({\n collection: eventsCollection,\n hideActivePillNames: ['job'],\n columns: [\n { key: 'at', label: 'Timestamp', formatter: 'datetime', sortable: true },\n { key: 'event', label: 'Event', formatter: 'badge' },\n { key: 'details|json', label: 'Details' }\n ]\n });\n\n // Create Logs table\n const logsCollection = new JobLogList({\n params: { job: this.model.get('id'), size: 10 }\n });\n this.logsView = new TableView({\n collection: logsCollection,\n hideActivePillNames: ['job'],\n columns: [\n { key: 'created|datetime', label: 'Created', sortable: true },\n { key: 'kind', label: 'Kind', formatter: 'badge' },\n { key: 'message', label: 'Message' }\n ]\n });\n\n // Create TabView\n this.tabView = new TabView({\n tabs: {\n 'Overview': this.overviewView,\n 'Payload': this.payloadView,\n 'Events': this.eventsView,\n 'Logs': this.logsView\n },\n activeTab: 'Overview',\n containerId: 'job-details-tabs'\n });\n this.addChild(this.tabView);\n\n // Create ContextMenu\n const contextMenuItems = [\n { label: 'Refresh', action: 'refresh-job', icon: 'bi-arrow-clockwise' }\n ];\n\n if (this.model.canCancel && this.model.canCancel()) {\n contextMenuItems.push({\n label: 'Cancel Job',\n action: 'cancel-job',\n icon: 'bi-x-circle',\n class: 'text-danger'\n });\n }\n\n if (this.model.canRetry && this.model.canRetry()) {\n contextMenuItems.push({\n label: 'Retry Job',\n action: 'retry-job',\n icon: 'bi-arrow-repeat',\n class: 'text-primary'\n });\n }\n\n const jobMenu = new ContextMenu({\n containerId: 'job-context-menu',\n className: \"context-menu-view header-menu-absolute\",\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: contextMenuItems\n }\n });\n this.addChild(jobMenu);\n\n await this.model.fetch({ params: { graph: \"detail\" } });\n // Start auto-refresh if job is active\n // this.startAutoRefresh();\n }\n\n async onBeforeRender() {\n await this.prepareJobData();\n }\n\n async prepareJobData() {\n if (!this.model) return;\n\n // Status styling\n this.model._.statusBadgeClass = this.model.getStatusBadgeClass ? this.model.getStatusBadgeClass() : 'bg-secondary';\n this.model._.statusIcon = this.model.getStatusIcon ? this.model.getStatusIcon() : 'bi-question-circle';\n this.model._.formattedDuration = this.model.getFormattedDuration ? this.model.getFormattedDuration() : 'N/A';\n }\n\n async loadJobDetails() {\n if (!this.model?.get('id')) return;\n\n try {\n if (this.model.getDetailedStatus) {\n await this.model.getDetailedStatus();\n }\n await this.prepareJobData();\n } catch (error) {\n console.error('Failed to load job details:', error);\n }\n }\n\n async onActionRefreshJob() {\n await this.model.fetch({ params: { graph: \"detail\" } });\n }\n\n async onActionCancelJob() {\n if (confirm('Are you sure you want to cancel this job?')) {\n try {\n const response = await this.model.cancel();\n if (response.success) {\n await this.loadJobDetails();\n await this.render();\n this.emit('job-cancelled', { job: this.model });\n } else {\n alert('Failed to cancel job: ' + (response.data?.error || 'Unknown error'));\n }\n } catch (error) {\n console.error('Failed to cancel job:', error);\n alert('Failed to cancel job: ' + error.message);\n }\n }\n }\n\n async onActionRetryJob() {\n const retryData = await Dialog.showForm({\n title: 'Retry Job',\n formConfig: JobForms.retry\n });\n\n if (retryData) {\n try {\n const response = await this.model.retry(retryData.delay || 0);\n if (response.success) {\n this.emit('job-retried', { job: this.model, newJobId: response.newJobId });\n } else {\n alert('Failed to retry job: ' + (response.data?.error || 'Unknown error'));\n }\n } catch (error) {\n console.error('Failed to retry job:', error);\n alert('Failed to retry job: ' + error.message);\n }\n }\n }\n\n startAutoRefresh() {\n if (this.autoRefreshInterval) clearInterval(this.autoRefreshInterval);\n\n if (this.model?.isActive && this.model.isActive()) {\n this.autoRefreshInterval = setInterval(async () => {\n try {\n await this.loadJobDetails();\n if (this.isMounted()) {\n await this.render();\n }\n } catch (error) {\n console.error('Auto-refresh failed:', error);\n }\n }, 5000);\n }\n }\n\n stopAutoRefresh() {\n if (this.autoRefreshInterval) {\n clearInterval(this.autoRefreshInterval);\n this.autoRefreshInterval = null;\n }\n }\n\n async onDestroy() {\n this.stopAutoRefresh();\n await super.onDestroy();\n }\n\n // Static show method for backward compatibility (if needed)\n static async show(job, options = {}) {\n const view = new JobDetailsView({ model: job });\n\n return await Dialog.showDialog({\n title: `<i class=\"bi bi-info-circle me-2\"></i>Job Details - ${job.get('id')}`,\n body: view,\n size: 'xl',\n scrollable: true,\n buttons: [{ text: 'Close', class: 'btn-secondary', dismiss: true }],\n onHide: () => view.stopAutoRefresh(),\n ...options\n });\n }\n}\n\nJob.VIEW_CLASS = JobDetailsView;\n\nexport default JobDetailsView;\n","/**\n * JobsAdminPage - Async Job Engine management dashboard\n * Main page orchestrating job monitoring and management components\n */\n\nimport Page from '@core/Page.js';\nimport View from '@core/View.js';\nimport TableView from '@core/views/table/TableView.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\nimport { Job, JobList, JobsEngineStats } from '@core/models/Job.js';\nimport { JobRunner, JobRunnerForms, JobRunnerList } from '@core/models/JobRunner.js';\nimport { MetricsChart, MetricsMiniChartWidget } from '@ext/charts/index.js';\n\n// Import component views\nimport JobStatsView from './JobStatsView.js';\nimport JobHealthView from './JobHealthView.js';\nimport JobDetailsView from './JobDetailsView.js';\n\nclass JobMetricsModalView extends View {\n constructor(options = {}) {\n super({\n className: 'job-metrics-modal-view',\n ...options\n });\n\n this.template = `\n <div data-container=\"job-metrics-modal-chart\" style=\"min-height:320px;\"></div>\n `;\n }\n\n async onInit() {\n this.chart = new MetricsChart({\n containerId: 'job-metrics-modal-chart',\n title: '<i class=\"bi bi-graph-up me-2\"></i> Job Channel Metrics',\n endpoint: '/api/metrics/fetch',\n height: 320,\n granularity: 'hours',\n category: 'jobs_channels',\n account: 'global',\n chartType: 'bar',\n showDateRange: true,\n yAxis: {\n label: 'Count',\n beginAtZero: true\n },\n tooltip: {\n y: 'number:0'\n }\n });\n\n this.addChild(this.chart);\n }\n}\n\nexport default class JobsAdminPage extends Page {\n constructor(options = {}) {\n super({\n title: 'Jobs Management',\n className: 'jobs-admin-page',\n ...options\n });\n\n this.pageTitle = 'Jobs Management';\n this.pageSubtitle = 'Async job monitoring and runner management';\n this.lastUpdated = new Date().toLocaleString();\n this.autoRefreshInterval = null;\n this.refreshRate = 30000; // 30 seconds default\n\n this.template = `\n <div class=\"jobs-admin-container\">\n <!-- Page Header -->\n <div class=\"d-flex justify-content-between align-items-center mb-4\">\n <div>\n <h1 class=\"h3 mb-1\">{{pageTitle}}</h1>\n <p class=\"text-muted mb-0\">{{pageSubtitle}}</p>\n <small class=\"text-info\">\n <i class=\"bi bi-arrow-clockwise me-1\"></i>\n Auto-refresh: {{refreshRateLabel}} | Last updated: {{lastUpdated}}\n </small>\n </div>\n <div class=\"btn-group\" role=\"group\">\n <button type=\"button\" class=\"btn btn-outline-secondary btn-sm\"\n data-action=\"refresh-all\" title=\"Refresh All Data\">\n <i class=\"bi bi-arrow-clockwise\"></i> Refresh\n </button>\n <button type=\"button\" class=\"btn btn-outline-primary btn-sm\"\n data-action=\"view-all-jobs\" title=\"View All Jobs\">\n <i class=\"bi bi-table\"></i> View All\n </button>\n <div class=\"dropdown\">\n <button class=\"btn btn-outline-secondary btn-sm dropdown-toggle\"\n type=\"button\" data-bs-toggle=\"dropdown\">\n <i class=\"bi bi-gear\"></i> Settings\n </button>\n <ul class=\"dropdown-menu dropdown-menu-end\">\n <li><h6 class=\"dropdown-header\">Auto Refresh</h6></li>\n <li><button class=\"dropdown-item\" data-action=\"set-refresh-rate\" data-rate=\"5\">5 seconds</button></li>\n <li><button class=\"dropdown-item\" data-action=\"set-refresh-rate\" data-rate=\"10\">10 seconds</button></li>\n <li><button class=\"dropdown-item\" data-action=\"set-refresh-rate\" data-rate=\"30\">30 seconds</button></li>\n <li><button class=\"dropdown-item\" data-action=\"set-refresh-rate\" data-rate=\"0\">Off</button></li>\n <li><hr class=\"dropdown-divider\"></li>\n <li><button class=\"dropdown-item\" data-action=\"runner-broadcast\">Broadcast Command</button></li>\n </ul>\n </div>\n <div class=\"dropdown\">\n <button class=\"btn btn-danger btn-sm dropdown-toggle\"\n type=\"button\" data-bs-toggle=\"dropdown\">\n <i class=\"bi bi-wrench\"></i> Manage\n </button>\n <ul class=\"dropdown-menu dropdown-menu-end\">\n <li><button class=\"dropdown-item\" data-action=\"run-simple-job\"><i class=\"bi bi-play-circle me-2\"></i>Run Simple Job</button></li>\n <li><button class=\"dropdown-item\" data-action=\"run-test-jobs\"><i class=\"bi bi-robot me-2\"></i>Run Test Jobs</button></li>\n <li><hr class=\"dropdown-divider\"></li>\n <li><button class=\"dropdown-item\" data-action=\"clear-stuck\"><i class=\"bi bi-wrench me-2\"></i>Clear Stuck Jobs</button></li>\n <li><button class=\"dropdown-item\" data-action=\"clear-channel\"><i class=\"bi bi-eraser me-2\"></i>Clear Channel</button></li>\n <li><button class=\"dropdown-item\" data-action=\"purge-jobs\"><i class=\"bi bi-trash me-2\"></i>Purge Jobs</button></li>\n <li><button class=\"dropdown-item\" data-action=\"cleanup-consumers\"><i class=\"bi bi-people me-2\"></i>Cleanup Consumers</button></li>\n </ul>\n </div>\n </div>\n </div>\n\n <div data-container=\"job-stats\"></div>\n\n <div class=\"row mb-4 g-3 align-items-stretch\">\n <div class=\"col-lg-6\" data-container=\"jobs-published-chart\"></div>\n <div class=\"col-lg-6 position-relative\">\n <div data-container=\"jobs-failed-chart\"></div>\n </div>\n </div>\n\n <div class=\"row\">\n <div class=\"col-xxl-6 col-lg-6 mb-4\">\n <div data-container=\"job-health\"></div>\n </div>\n <div class=\"col-xxl-6 col-lg-6 mb-4\">\n <div class=\"card shadow-sm h-100\">\n <div class=\"card-header d-flex justify-content-between align-items-center\">\n <h5 class=\"mb-0\"><i class=\"bi bi-cpu me-2\"></i>Job Runners</h5>\n <small class=\"text-muted\">Heartbeat &amp; status</small>\n </div>\n <div class=\"card-body p-0\" data-container=\"runner-table\"></div>\n </div>\n </div>\n </div>\n\n <div class=\"row\">\n <div class=\"col-xl-6 mb-4\">\n <div class=\"card shadow-sm h-100\">\n <div class=\"card-header d-flex justify-content-between align-items-center\">\n <h5 class=\"mb-0\"><i class=\"bi bi-play-circle me-2\"></i>Running Jobs</h5>\n <small class=\"text-muted\">Currently executing</small>\n </div>\n <div class=\"card-body p-0\" data-container=\"running-jobs-table\"></div>\n </div>\n </div>\n <div class=\"col-xl-6 mb-4\">\n <div class=\"card shadow-sm h-100\">\n <div class=\"card-header d-flex justify-content-between align-items-center\">\n <h5 class=\"mb-0\"><i class=\"bi bi-hourglass-split me-2\"></i>Pending Jobs</h5>\n <small class=\"text-muted\">Waiting in queue</small>\n </div>\n <div class=\"card-body p-0\" data-container=\"pending-jobs-table\"></div>\n </div>\n </div>\n </div>\n\n <div class=\"row\">\n <div class=\"col-xl-6 mb-4\">\n <div class=\"card shadow-sm h-100\">\n <div class=\"card-header\">\n <h5 class=\"mb-0\"><i class=\"bi bi-calendar-event me-2\"></i>Scheduled Jobs</h5>\n </div>\n <div class=\"card-body p-0\" data-container=\"scheduled-jobs-table\"></div>\n </div>\n </div>\n <div class=\"col-xl-6 mb-4\">\n <div class=\"card shadow-sm h-100\">\n <div class=\"card-header d-flex justify-content-between align-items-center\">\n <h5 class=\"mb-0\"><i class=\"bi bi-bug me-2\"></i>Failed Jobs</h5>\n <small class=\"text-muted\">Latest errors</small>\n </div>\n <div class=\"card-body p-0\" data-container=\"failed-jobs-table\"></div>\n </div>\n </div>\n </div>\n\n <div class=\"card shadow-sm mb-5\">\n <div class=\"card-header d-flex justify-content-between align-items-center\">\n <h5 class=\"mb-0\"><i class=\"bi bi-tools me-2\"></i>Operations</h5>\n </div>\n <div class=\"card-body\">\n <div class=\"d-flex flex-wrap gap-2\">\n <button class=\"btn btn-outline-primary\" data-action=\"run-simple-job\">\n <i class=\"bi bi-play-circle me-2\"></i>Run Simple Job\n </button>\n <button class=\"btn btn-outline-primary\" data-action=\"run-test-jobs\">\n <i class=\"bi bi-robot me-2\"></i>Run Test Jobs\n </button>\n <button class=\"btn btn-outline-warning\" data-action=\"clear-stuck\">\n <i class=\"bi bi-wrench me-2\"></i>Clear Stuck\n </button>\n <button class=\"btn btn-outline-warning\" data-action=\"clear-channel\">\n <i class=\"bi bi-eraser me-2\"></i>Clear Channel\n </button>\n <button class=\"btn btn-outline-danger\" data-action=\"purge-jobs\">\n <i class=\"bi bi-trash me-2\"></i>Purge Jobs\n </button>\n <button class=\"btn btn-outline-info\" data-action=\"cleanup-consumers\">\n <i class=\"bi bi-people me-2\"></i>Cleanup Consumers\n </button>\n <button class=\"btn btn-outline-secondary\" data-action=\"runner-broadcast\">\n <i class=\"bi bi-wifi me-2\"></i>Broadcast Command\n </button>\n </div>\n </div>\n </div>\n </div>\n `;\n }\n\n async onInit() {\n\n this.jobStats = new JobsEngineStats();\n // Create child views\n this.jobStatsView = new JobStatsView({\n containerId: 'job-stats',\n model: this.jobStats\n });\n this.addChild(this.jobStatsView);\n\n this.jobHealthView = new JobHealthView({\n containerId: 'job-health',\n model: this.jobStats\n });\n this.addChild(this.jobHealthView);\n\n this.jobsPublishedChart = new MetricsMiniChartWidget({\n containerId: 'jobs-published-chart',\n icon: 'bi bi-upload',\n title: 'Jobs Published',\n subtitle: '{{now_value}} {{now_label}}',\n granularity: 'days',\n slugs: ['jobs.published'],\n account: 'global',\n chartType: 'line',\n height: 90,\n showSettings: true,\n showTrending: true,\n showDateRange: false\n });\n this.addChild(this.jobsPublishedChart);\n\n this.jobsFailedChart = new MetricsMiniChartWidget({\n containerId: 'jobs-failed-chart',\n icon: 'bi bi-exclamation-octagon',\n title: 'Jobs Failed',\n subtitle: '{{now_value}} {{now_label}}',\n granularity: 'days',\n slugs: ['jobs.failed'],\n account: 'global',\n chartType: 'line',\n height: 90,\n showSettings: true,\n showTrending: true,\n showDateRange: false\n });\n this.addChild(this.jobsFailedChart);\n\n this.runningJobsTable = new TableView({\n containerId: 'running-jobs-table',\n Collection: JobList,\n collectionParams: {\n size: 5,\n sort: '-created',\n status: 'running'\n },\n searchable: true,\n filterable: false,\n paginated: true,\n itemView: JobDetailsView,\n hideActivePills: ['status'],\n viewDialogOptions: {\n title: 'Job Details',\n size: 'xl',\n scrollable: true\n },\n tableOptions: {\n striped: false,\n hover: true,\n size: 'sm'\n },\n columns: [\n {\n key: 'id',\n label: 'Job',\n template: `\n <div class=\"fw-semibold font-monospace\">{{model.id|truncate_middle(12)}}</div>\n <div class=\"text-muted small\">{{model.channel}} &middot; {{model.func|truncate_middle(28)|default('n/a')}}</div>\n `\n },\n {\n key: 'runner_id',\n label: 'Runner',\n template: `\n <span class=\"font-monospace\">{{model.runner_id|truncate_middle(12)|default('n/a')}}</span>\n `\n },\n {\n key: 'status',\n label: 'State',\n formatter: (value, context) => {\n const job = context.row;\n const badgeClass = job.getStatusBadgeClass ? job.getStatusBadgeClass() : 'bg-secondary';\n const icon = job.getStatusIcon ? job.getStatusIcon() : 'bi-question';\n return `<span class=\"badge ${badgeClass}\"><i class=\"${icon} me-1\"></i>${value.toUpperCase()}</span>`;\n }\n },\n {\n key: 'created',\n label: 'Started',\n formatter: 'datetime'\n }\n ]\n });\n this.addChild(this.runningJobsTable);\n\n this.pendingJobsTable = new TableView({\n containerId: 'pending-jobs-table',\n Collection: JobList,\n collectionParams: {\n size: 5,\n sort: '-created',\n status: 'pending',\n run_at__isnull: true\n },\n searchable: true,\n filterable: false,\n paginated: true,\n selectable: true,\n batchBarLocation: 'top',\n batchActions: [\n {\n icon: 'bi-x-circle-fill',\n label: 'Cancel Jobs',\n action: \"cancel-jobs\"\n }\n ],\n itemView: JobDetailsView,\n hideActivePills: ['status'],\n viewDialogOptions: {\n title: 'Job Details',\n size: 'xl',\n scrollable: true\n },\n tableOptions: {\n striped: false,\n hover: true,\n size: 'sm'\n },\n columns: [\n {\n key: 'id',\n label: 'Job',\n template: `\n <div class=\"fw-semibold font-monospace\">{{model.id|truncate_middle(12)}}</div>\n <div class=\"text-muted small\">{{model.channel}} &middot; {{model.func|truncate_middle(28)|default('n/a')}}</div>\n `\n },\n {\n key: 'priority',\n label: 'Priority',\n formatter: (value = 0) => {\n const badge = value >= 8 ? 'bg-danger' : value >= 5 ? 'bg-warning' : 'bg-secondary';\n return `<span class=\"badge ${badge}\">${value}</span>`;\n }\n },\n {\n key: 'modified',\n label: 'Queued',\n formatter: 'relative'\n }\n ]\n });\n this.pendingJobsTable.on(\"action:batch-cancel-jobs\", async (action, event, element) => {\n const items = this.pendingJobsTable.getSelectedItems();\n await Promise.all(items.map(item => item.model.cancel()));\n this.getApp().toast.success(\"Jobs cancelled successfully\");\n this.pendingJobsTable.collection.fetch();\n });\n this.addChild(this.pendingJobsTable);\n\n this.failedJobsTable = new TableView({\n containerId: 'failed-jobs-table',\n Collection: JobList,\n collectionParams: {\n size: 5,\n sort: '-finished_at',\n status: 'failed'\n },\n searchable: true,\n filterable: false,\n paginated: true,\n itemView: JobDetailsView,\n viewDialogOptions: {\n title: 'Job Details',\n size: 'xl',\n scrollable: true\n },\n hideActivePills: ['status'],\n tableOptions: {\n striped: false,\n hover: true,\n size: 'sm'\n },\n columns: [\n {\n key: 'id',\n label: 'Job',\n template: `\n <div class=\"fw-semibold font-monospace\">{{model.id|truncate_middle(12)}}</div>\n <div class=\"text-muted small\">{{model.channel}} &middot; {{model.func|truncate_middle(28)|default('n/a')}}</div>\n `\n },\n {\n key: 'last_error',\n label: 'Error',\n template: `\n <div class=\"text-danger small\">{{model.last_error|truncate(80)|default('Unknown error')}}</div>\n `\n },\n {\n key: 'modified',\n label: 'Failed',\n formatter: 'relative'\n }\n ]\n });\n this.addChild(this.failedJobsTable);\n\n this.scheduledJobsTable = new TableView({\n containerId: 'scheduled-jobs-table',\n Collection: JobList,\n collectionParams: {\n size: 5,\n sort: 'run_at',\n run_at__isnull: false,\n status: 'pending'\n },\n searchable: true,\n filterable: false,\n paginated: true,\n itemView: JobDetailsView,\n hideActivePills: ['status'],\n viewDialogOptions: {\n title: 'Job Details',\n size: 'xl',\n scrollable: true\n },\n tableOptions: {\n striped: false,\n hover: true,\n size: 'sm'\n },\n columns: [\n {\n key: 'id',\n label: 'Job',\n formatter: 'truncate_middle(12)'\n },\n {\n key: 'run_at',\n label: 'Scheduled For',\n formatter: 'datetime'\n },\n {\n key: 'channel',\n label: 'Channel',\n formatter: 'badge'\n }\n ],\n selectable: true,\n batchBarLocation: 'top',\n batchActions: [\n {\n icon: 'bi-x-circle-fill',\n label: 'Cancel Jobs',\n action: \"cancel-jobs\"\n }\n ],\n });\n this.scheduledJobsTable.on(\"action:batch-cancel-jobs\", async (action, event, element) => {\n const items = this.scheduledJobsTable.getSelectedItems();\n await Promise.all(items.map(item => item.model.cancel()));\n this.getApp().toast.success(\"Jobs cancelled successfully\");\n this.scheduledJobsTable.collection.fetch();\n });\n this.addChild(this.scheduledJobsTable);\n\n this.runnersTable = new TableView({\n containerId: 'runner-table',\n Collection: JobRunnerList,\n searchable: true,\n filterable: false,\n paginated: true,\n tableOptions: {\n striped: false,\n hover: true,\n size: 'sm'\n },\n columns: [\n {\n key: 'runner_id',\n label: 'Runner',\n formatter: 'truncate_middle(16)'\n },\n {\n key: 'alive',\n label: 'Status',\n formatter: (value) => {\n const isAlive = value === true;\n const badgeClass = isAlive ? 'bg-success' : 'bg-danger';\n const icon = isAlive ? 'bi-check-circle-fill' : 'bi-x-octagon-fill';\n const text = isAlive ? 'ALIVE' : 'DEAD';\n return `<span class=\"badge ${badgeClass}\"><i class=\"${icon} me-1\"></i>${text}</span>`;\n }\n },\n {\n key: 'last_heartbeat',\n label: 'Heartbeat',\n formatter: (value) => {\n if (!value) return 'Never';\n const heartbeatTime = new Date(value);\n const now = new Date();\n const diffMs = now - heartbeatTime;\n const diffSeconds = Math.floor(diffMs / 1000);\n if (diffSeconds < 60) return `${diffSeconds}s ago`;\n if (diffSeconds < 3600) return `${Math.floor(diffSeconds / 60)}m ago`;\n return `${Math.floor(diffSeconds / 3600)}h ago`;\n }\n }\n ]\n });\n this.addChild(this.runnersTable);\n\n await this.refreshData();\n\n }\n\n // Auto-refresh management\n startAutoRefresh() {\n if (this.autoRefreshInterval) {\n clearInterval(this.autoRefreshInterval);\n }\n\n if (this.refreshRate > 0) {\n this.autoRefreshInterval = setInterval(async () => {\n await this.refreshData();\n }, this.refreshRate);\n }\n }\n\n async refreshData() {\n try {\n const tasks = [\n this.jobStats.fetch()\n ];\n\n if (this.jobsPublishedChart) {\n tasks.push(this.jobsPublishedChart.refresh());\n }\n if (this.jobsFailedChart) {\n tasks.push(this.jobsFailedChart.refresh());\n }\n if (this.runningJobsTable?.collection?.fetch) {\n tasks.push(this.runningJobsTable.collection.fetch());\n }\n if (this.pendingJobsTable?.collection?.fetch) {\n tasks.push(this.pendingJobsTable.collection.fetch());\n }\n if (this.failedJobsTable?.collection?.fetch) {\n tasks.push(this.failedJobsTable.collection.fetch());\n }\n if (this.scheduledJobsTable?.collection?.fetch) {\n tasks.push(this.scheduledJobsTable.collection.fetch());\n }\n if (this.runnersTable?.collection?.fetch) {\n tasks.push(this.runnersTable.collection.fetch());\n }\n\n await Promise.all(tasks);\n\n this.lastUpdated = new Date().toLocaleString();\n this.updateHeaderTimestamp();\n\n } catch (error) {\n console.error('Failed to refresh jobs dashboard:', error);\n }\n }\n\n updateHeaderTimestamp() {\n const timestampElement = this.element?.querySelector('.text-info');\n if (timestampElement) {\n const rateLabel = this.refreshRate === 0 ? 'Off' : `${this.refreshRate / 1000}s`;\n timestampElement.innerHTML = `\n <i class=\"bi bi-arrow-clockwise me-1\"></i>\n Auto-refresh: ${rateLabel} | Last updated: ${this.lastUpdated}\n `;\n }\n }\n\n get refreshRateSeconds() {\n return this.refreshRate / 1000;\n }\n\n get refreshRateLabel() {\n return this.refreshRate === 0 ? 'Off' : `${this.refreshRateSeconds}s`;\n }\n\n // Action handlers\n async onActionRefreshAll(event, element) {\n try {\n const icon = element.querySelector('i');\n icon?.classList.add('spinning');\n element.disabled = true;\n\n await this.refreshData();\n\n } catch (error) {\n console.error('Failed to refresh jobs dashboard:', error);\n } finally {\n const icon = element.querySelector('i');\n icon?.classList.remove('spinning');\n element.disabled = false;\n }\n }\n\n async onActionSetRefreshRate(event, element) {\n const rate = parseInt(element.getAttribute('data-rate')) * 1000;\n this.refreshRate = rate;\n this.startAutoRefresh();\n\n const rateText = rate === 0 ? 'Off' : `${rate / 1000}s`;\n this.getApp().toast.success(`Auto-refresh set to ${rateText}`);\n }\n\n async onActionExportData() {\n await Dialog.showAlert({\n title: 'Export Data',\n message: 'Data export functionality coming soon!',\n type: 'info'\n });\n }\n\n // Job management actions\n async onActionRunSimpleJob(event, element) {\n const confirmed = await Dialog.showConfirm({\n title: 'Run Simple Job',\n message: 'This will run a simple test job to verify the job system is working correctly.',\n confirmText: 'Run Test',\n confirmClass: 'btn-success'\n });\n\n if (confirmed) {\n await this.executeJobAction(element, () => Job.test(), 'Test job started successfully');\n }\n }\n\n async onActionRunTestJobs(event, element) {\n const confirmed = await Dialog.showConfirm({\n title: 'Run Test Jobs',\n message: 'This will run a suite of test jobs to verify all job functionalities.',\n confirmText: 'Run Tests',\n confirmClass: 'btn-success'\n });\n\n if (confirmed) {\n await this.executeJobAction(element, () => Job.tests(), 'Test suite started successfully');\n }\n }\n\n async onActionClearStuck(event, element) {\n const channels = this.jobHealthView?.health?.channelsArray || [];\n const channelOptions = [\n { value: '', label: 'All Channels' },\n ...channels.map(ch => ({ value: ch.channel, label: ch.channel }))\n ];\n\n const result = await Dialog.showForm({\n title: 'Clear Stuck Jobs',\n formConfig: {\n fields: [\n {\n name: 'channel',\n type: 'select',\n label: 'Channel',\n options: channelOptions,\n value: '',\n help: 'Select specific channel or leave empty for all channels'\n }\n ]\n }\n });\n\n if (result) {\n await this.executeJobAction(\n element,\n () => Job.clearStuck(result.channel || null),\n (response) => {\n const count = response.data.count || 0;\n const channelText = result.channel ? ` from channel \"${result.channel}\"` : '';\n return `Cleared ${count} stuck job${count !== 1 ? 's' : ''}${channelText}`;\n }\n );\n }\n }\n\n async onActionClearChannel(event, element) {\n const channels = this.jobHealthView?.health?.channelsArray || [];\n const channelOptions = channels.map(ch => ({ value: ch.channel, label: ch.channel }));\n\n const result = await Dialog.showForm({\n title: 'Clear Channel',\n formConfig: {\n fields: [\n {\n name: 'channel',\n type: 'select',\n label: 'Channel',\n options: channelOptions,\n required: true,\n help: 'Select the channel to clear.'\n }\n ]\n }\n });\n\n if (result) {\n await this.executeJobAction(\n element,\n () => Job.clearChannel(result.channel),\n `Channel \"${result.channel}\" cleared successfully.`\n );\n }\n }\n\n async onActionPurgeJobs(event, element) {\n const result = await Dialog.showForm({\n title: 'Purge Old Jobs',\n formConfig: {\n fields: [\n {\n name: 'days_old',\n type: 'number',\n label: 'Days Old',\n value: 30,\n required: true,\n help: 'Delete jobs older than this many days.'\n }\n ]\n }\n });\n\n if (result) {\n await this.executeJobAction(\n element,\n () => Job.purgeJobs(result.days_old),\n (response) => {\n const count = response.data.count || 0;\n return `Purged ${count} old job(s).`;\n }\n );\n }\n }\n\n async onActionCleanupConsumers(event, element) {\n const confirmed = await Dialog.showConfirm({\n title: 'Cleanup Consumers',\n message: 'This will remove stale consumer records from the system. This is generally safe.',\n confirmText: 'Cleanup',\n confirmClass: 'btn-warning'\n });\n\n if (confirmed) {\n await this.executeJobAction(\n element,\n () => Job.cleanConsumers(),\n (response) => {\n const count = response.data.count || 0;\n return `Cleaned up ${count} consumer(s).`;\n }\n );\n }\n }\n\n async onActionRunnerBroadcast() {\n const result = await Dialog.showForm({\n title: 'Broadcast Command to All Runners',\n formConfig: JobRunnerForms.broadcast\n });\n\n if (result) {\n try {\n const broadcastResult = await JobRunner.broadcast(\n result.command,\n {},\n result.timeout\n );\n\n if (broadcastResult.success) {\n this.getApp().toast.success(`Broadcast command \"${result.command}\" sent successfully`);\n await this.refreshData();\n } else {\n this.getApp().toast.error(broadcastResult.data?.error || 'Failed to broadcast command');\n }\n } catch (error) {\n console.error('Failed to broadcast command:', error);\n this.getApp().toast.error('Error broadcasting command: ' + error.message);\n }\n }\n }\n\n // Helper method to execute job actions with consistent error handling\n async executeJobAction(element, actionFn, successMessage) {\n try {\n element.disabled = true;\n const icon = element.querySelector('i');\n icon?.classList.add('spinning');\n\n const result = await actionFn();\n if (result.success && result.data?.status) {\n const message = typeof successMessage === 'function'\n ? successMessage(result)\n : successMessage;\n this.getApp().toast.success(message);\n await this.refreshData();\n } else {\n this.getApp().toast.error(result.data?.error || 'Operation failed');\n }\n } catch (error) {\n console.error('Job action failed:', error);\n this.getApp().toast.error('Error: ' + error.message);\n } finally {\n element.disabled = false;\n const icon = element.querySelector('i');\n icon?.classList.remove('spinning');\n }\n }\n\n async onEnter() {\n // Start auto-refresh\n this.startAutoRefresh();\n }\n\n async onExit() {\n if (this.autoRefreshInterval) {\n clearInterval(this.autoRefreshInterval);\n this.autoRefreshInterval = null;\n }\n }\n\n // Public API\n async refreshDashboard() {\n return await this.refreshData();\n }\n\n getStats() {\n return this.jobStatsView?.stats || {};\n }\n\n getHealth() {\n return this.jobHealthView?.health || {};\n }\n\n async onActionOpenJobMetricsModal() {\n const modalView = new JobMetricsModalView();\n await Dialog.showDialog({\n title: '<i class=\"bi bi-graph-up me-2\"></i>Job Channel Metrics',\n body: modalView,\n size: 'xl',\n scrollable: true,\n buttons: [\n { text: 'Close', class: 'btn-secondary', dismiss: true }\n ]\n });\n }\n\n async onActionViewAllJobs() {\n const jobList = new JobList({\n params: {\n sort: '-created',\n size: 25\n }\n });\n\n const view = new TableView({\n collection: jobList,\n fetchOnMount: true,\n columns: [\n {\n key: 'id',\n label: 'Job',\n template: `\n <div class=\"fw-semibold font-monospace\">{{model.id}}</div>\n <div class=\"text-muted small\">{{model.func|default('Unknown')}}</div>\n `\n },\n {\n key: 'channel',\n label: 'Channel',\n formatter: 'badge'\n },\n {\n key: 'status',\n label: 'Status',\n formatter: (value, context) => {\n const job = context.row;\n const badgeClass = job.getStatusBadgeClass ? job.getStatusBadgeClass() : 'bg-secondary';\n const icon = job.getStatusIcon ? job.getStatusIcon() : 'bi-question';\n return `<span class=\"badge ${badgeClass}\"><i class=\"${icon} me-1\"></i>${value?.toUpperCase() || 'UNKNOWN'}</span>`;\n }\n },\n {\n key: 'created',\n label: 'Created',\n formatter: 'datetime'\n },\n {\n key: 'finished_at',\n label: 'Finished',\n formatter: 'datetime'\n },\n {\n key: 'duration_ms',\n label: 'Duration',\n formatter: 'duration'\n }\n ],\n filters: [\n {\n key: 'func__icontains',\n label: 'Function',\n type: 'text'\n },\n {\n key: 'status',\n label: 'Status',\n type: 'select',\n options: [\n { label: 'Pending', value: 'pending' },\n { label: 'Running', value: 'running' },\n { label: 'Completed', value: 'completed' },\n { label: 'Failed', value: 'failed' },\n { label: 'Canceled', value: 'canceled' },\n { label: 'Expired', value: 'expired' }\n ]\n },\n {\n key: 'channel',\n label: 'Channel',\n type: 'text'\n }\n ],\n itemView: JobDetailsView,\n viewDialogOptions: {\n title: 'Job Details',\n size: 'xl',\n scrollable: true\n },\n searchable: true,\n filterable: true,\n paginated: true,\n tableOptions: {\n striped: true,\n hover: true,\n responsive: true\n }\n });\n\n await this.getApp().showDialog({\n title: '<i class=\"bi bi-table me-2\"></i>All Jobs',\n body: view,\n size: 'xl',\n scrollable: true,\n buttons: [\n { text: 'Close', class: 'btn-secondary', dismiss: true }\n ]\n });\n }\n}\n","/**\n * TaskDetailsView - Detailed task information view for Dialog display\n * Shows comprehensive task data, logs, metrics, and available actions\n */\n\nimport View from '@core/View.js';\n\nexport default class TaskDetailsView extends View {\n constructor(options = {}) {\n super({\n ...options,\n className: 'mojo-task-details-view'\n });\n\n this.task = options.task || null;\n this.logs = [];\n this.metrics = null;\n }\n\n async getTemplate() {\n return `\n <div class=\"mojo-task-details-container\">\n {{#task}}\n <!-- Task Overview -->\n <div class=\"card border-0 bg-light mb-3\">\n <div class=\"card-body\">\n <div class=\"row\">\n <div class=\"col-md-6\">\n <div class=\"mb-3\">\n <label class=\"form-label fw-bold text-muted small\">Task ID</label>\n <div class=\"font-monospace\">{{id}}</div>\n </div>\n <div class=\"mb-3\">\n <label class=\"form-label fw-bold text-muted small\">Function</label>\n <div>{{function}}</div>\n </div>\n <div class=\"mb-3\">\n <label class=\"form-label fw-bold text-muted small\">Channel</label>\n <div>\n <span class=\"badge bg-primary\">{{channel}}</span>\n </div>\n </div>\n </div>\n <div class=\"col-md-6\">\n <div class=\"mb-3\">\n <label class=\"form-label fw-bold text-muted small\">Status</label>\n <div>\n <span class=\"badge {{statusBadgeClass}} fs-6\">\n <i class=\"bi {{statusIcon}}\"></i> {{status|uppercase}}\n </span>\n </div>\n </div>\n <div class=\"mb-3\">\n <label class=\"form-label fw-bold text-muted small\">Created</label>\n <div>{{created|datetime}}</div>\n </div>\n {{#completed_at}}\n <div class=\"mb-3\">\n <label class=\"form-label fw-bold text-muted small\">Completed</label>\n <div>{{completed_at|datetime}}</div>\n </div>\n {{/completed_at}}\n {{#expires}}\n <div class=\"mb-3\">\n <label class=\"form-label fw-bold text-muted small\">Expires</label>\n <div class=\"{{expiresClass}}\">{{expires|datetime}}</div>\n </div>\n {{/expires}}\n </div>\n </div>\n </div>\n </div>\n\n <!-- Task Data -->\n {{#data}}\n <div class=\"card mb-3\">\n <div class=\"card-header py-2\">\n <h6 class=\"mb-0\">\n <i class=\"bi bi-database me-2\"></i>Task Data\n </h6>\n </div>\n <div class=\"card-body\">\n <pre class=\"bg-light p-3 rounded mb-0\"><code>{{dataFormatted}}</code></pre>\n </div>\n </div>\n {{/data}}\n\n <!-- Error Information -->\n {{#error}}\n <div class=\"card border-danger mb-3\">\n <div class=\"card-header bg-danger-subtle py-2\">\n <h6 class=\"mb-0 text-danger\">\n <i class=\"bi bi-exclamation-triangle me-2\"></i>Error Details\n </h6>\n </div>\n <div class=\"card-body\">\n <div class=\"alert alert-danger mb-0\">\n <strong>Error:</strong> {{error}}\n </div>\n {{#errorDetails}}\n <div class=\"mt-3\">\n <label class=\"form-label fw-bold small\">Stack Trace:</label>\n <pre class=\"bg-light p-3 rounded small mb-0\"><code>{{errorDetails}}</code></pre>\n </div>\n {{/errorDetails}}\n </div>\n </div>\n {{/error}}\n\n <!-- Performance Metrics -->\n {{#metrics}}\n <div class=\"card mb-3\">\n <div class=\"card-header py-2\">\n <h6 class=\"mb-0\">\n <i class=\"bi bi-speedometer2 me-2\"></i>Performance Metrics\n </h6>\n </div>\n <div class=\"card-body\">\n <div class=\"row\">\n <div class=\"col-md-3 col-6 mb-3\">\n <div class=\"text-center\">\n <div class=\"h5 mb-1 text-primary\">{{executionTime}}ms</div>\n <small class=\"text-muted\">Execution Time</small>\n </div>\n </div>\n <div class=\"col-md-3 col-6 mb-3\">\n <div class=\"text-center\">\n <div class=\"h5 mb-1 text-info\">{{memoryUsage}}MB</div>\n <small class=\"text-muted\">Memory Usage</small>\n </div>\n </div>\n <div class=\"col-md-3 col-6 mb-3\">\n <div class=\"text-center\">\n <div class=\"h5 mb-1 text-warning\">{{cpuUsage}}%</div>\n <small class=\"text-muted\">CPU Usage</small>\n </div>\n </div>\n <div class=\"col-md-3 col-6 mb-3\">\n <div class=\"text-center\">\n <div class=\"h5 mb-1 text-secondary\">{{retryCount}}</div>\n <small class=\"text-muted\">Retry Count</small>\n </div>\n </div>\n </div>\n </div>\n </div>\n {{/metrics}}\n\n <!-- Task Logs -->\n <div class=\"card\">\n <div class=\"card-header py-2 d-flex justify-content-between align-items-center\">\n <h6 class=\"mb-0\">\n <i class=\"bi bi-journal-text me-2\"></i>Task Logs\n </h6>\n <button class=\"btn btn-sm btn-outline-primary\" data-action=\"refresh-logs\">\n <i class=\"bi bi-arrow-clockwise\"></i> Refresh\n </button>\n </div>\n <div class=\"card-body p-0\">\n <div class=\"mojo-task-logs-container\" style=\"max-height: 300px; overflow-y: auto;\">\n {{#logs.length}}\n {{#logs}}\n <div class=\"log-entry p-3 border-bottom\">\n <div class=\"d-flex justify-content-between align-items-start\">\n <div class=\"log-message flex-grow-1\">\n <span class=\"badge bg-{{levelClass}} me-2\">{{level|uppercase}}</span>\n {{message}}\n </div>\n <small class=\"text-muted ms-3\">{{timestamp|time}}</small>\n </div>\n </div>\n {{/logs}}\n {{/logs.length}}\n {{^logs.length}}\n <div class=\"text-center text-muted py-5\">\n <i class=\"bi bi-journal fs-1 opacity-50\"></i>\n <p class=\"mb-0 mt-2\">No logs available for this task</p>\n </div>\n {{/logs.length}}\n </div>\n </div>\n </div>\n {{/task}}\n\n {{^task}}\n <div class=\"alert alert-warning\">\n <i class=\"bi bi-exclamation-triangle me-2\"></i>\n No task data available\n </div>\n {{/task}}\n </div>\n `;\n }\n\n async onInit() {\n if (this.task) {\n await this.prepareTaskData();\n await this.loadTaskLogs();\n await this.loadTaskMetrics();\n }\n }\n\n async prepareTaskData() {\n if (!this.task) return;\n\n // Status styling\n this.task.statusBadgeClass = this.getStatusBadgeClass(this.task.status);\n this.task.statusIcon = this.getStatusIcon(this.task.status);\n\n // Format data for display\n if (this.task.data && typeof this.task.data === 'object') {\n this.task.dataFormatted = JSON.stringify(this.task.data, null, 2);\n }\n\n // Check if task has expired\n if (this.task.expires) {\n this.task.expiresClass = this.task.expires * 1000 < Date.now() ? 'text-danger' : 'text-muted';\n }\n }\n\n getStatusBadgeClass(status) {\n const classes = {\n 'pending': 'bg-primary',\n 'running': 'bg-success', \n 'completed': 'bg-info',\n 'error': 'bg-danger',\n 'cancelled': 'bg-secondary',\n 'expired': 'bg-warning'\n };\n return classes[status] || 'bg-secondary';\n }\n\n getStatusIcon(status) {\n const icons = {\n 'pending': 'bi-hourglass',\n 'running': 'bi-arrow-repeat',\n 'completed': 'bi-check-circle',\n 'error': 'bi-x-octagon',\n 'cancelled': 'bi-x-circle',\n 'expired': 'bi-clock'\n };\n return icons[status] || 'bi-question-circle';\n }\n\n getLogLevelClass(level) {\n const classes = {\n 'debug': 'secondary',\n 'info': 'primary',\n 'warning': 'warning',\n 'error': 'danger'\n };\n return classes[level] || 'secondary';\n }\n\n async loadTaskLogs() {\n if (!this.task?.id) return;\n\n try {\n const response = await this.getApp().rest.GET(`/api/tasks/${this.task.id}/logs`);\n if (response.success && response.data.status) {\n this.logs = response.data.data.map(log => ({\n ...log,\n levelClass: this.getLogLevelClass(log.level)\n }));\n } else {\n this.logs = [];\n }\n } catch (error) {\n console.error('Failed to load task logs:', error);\n this.logs = [];\n }\n }\n\n async loadTaskMetrics() {\n if (!this.task?.id) return;\n\n try {\n const response = await this.getApp().rest.GET(`/api/tasks/${this.task.id}/metrics`);\n if (response.success && response.data.status) {\n this.metrics = response.data.data;\n }\n } catch (error) {\n console.error('Failed to load task metrics:', error);\n this.metrics = null;\n }\n }\n\n async setTask(task) {\n this.task = task;\n await this.prepareTaskData();\n await this.loadTaskLogs();\n await this.loadTaskMetrics();\n \n if (this.isMounted()) {\n await this.render();\n }\n }\n\n async onActionRefreshLogs(action, event, element) {\n if (!this.task?.id) return;\n\n try {\n element.disabled = true;\n const icon = element.querySelector('i');\n if (icon) icon.classList.add('spinning');\n\n await this.loadTaskLogs();\n await this.render();\n \n } catch (error) {\n console.error('Failed to refresh logs:', error);\n this.showError('Failed to refresh logs: ' + error.message);\n } finally {\n element.disabled = false;\n const icon = element.querySelector('i');\n if (icon) icon.classList.remove('spinning');\n }\n }\n\n // Static method for easy Dialog integration\n static async show(task, options = {}) {\n const view = new TaskDetailsView({ task });\n await view.onInit();\n\n const buttons = [];\n\n // Add action buttons based on task status\n if (['pending', 'running'].includes(task.status)) {\n buttons.push({\n text: 'Cancel Task',\n class: 'btn-outline-danger',\n action: async () => {\n if (confirm('Are you sure you want to cancel this task?')) {\n try {\n const response = await view.getApp().rest.POST(`/api/tasks/${task.id}/cancel`);\n if (response.success && response.data.status) {\n view.showSuccess('Task cancelled successfully');\n return { action: 'cancelled', task };\n } else {\n view.showError(response.data.error || 'Failed to cancel task');\n }\n } catch (error) {\n view.showError('Failed to cancel task: ' + error.message);\n }\n }\n return null;\n }\n });\n }\n\n if (task.status === 'error') {\n buttons.push({\n text: 'Retry Task',\n class: 'btn-outline-primary',\n action: async () => {\n try {\n const response = await view.getApp().rest.POST(`/api/tasks/${task.id}/retry`);\n if (response.success && response.data.status) {\n view.showSuccess('Task queued for retry');\n return { action: 'retried', task };\n } else {\n view.showError(response.data.error || 'Failed to retry task');\n }\n } catch (error) {\n view.showError('Failed to retry task: ' + error.message);\n }\n return null;\n }\n });\n }\n\n buttons.push({\n text: 'Clone Task',\n class: 'btn-outline-info',\n action: async () => {\n try {\n const response = await view.getApp().rest.POST(`/api/tasks/${task.id}/clone`);\n if (response.success && response.data.status) {\n view.showSuccess('Task cloned successfully');\n return { action: 'cloned', originalTask: task, newTask: response.data.data };\n } else {\n view.showError(response.data.error || 'Failed to clone task');\n }\n } catch (error) {\n view.showError('Failed to clone task: ' + error.message);\n }\n return null;\n }\n });\n\n buttons.push({\n text: 'Export',\n class: 'btn-outline-secondary',\n action: () => {\n try {\n const exportData = {\n task: task,\n logs: view.logs,\n metrics: view.metrics,\n exported_at: new Date().toISOString(),\n exported_by: 'task-management-system'\n };\n\n const blob = new Blob([JSON.stringify(exportData, null, 2)], {\n type: 'application/json'\n });\n \n const url = URL.createObjectURL(blob);\n const a = document.createElement('a');\n a.href = url;\n a.download = `task-${task.id}-${Date.now()}.json`;\n document.body.appendChild(a);\n a.click();\n document.body.removeChild(a);\n URL.revokeObjectURL(url);\n\n view.showSuccess('Task data exported successfully');\n return null;\n } catch (error) {\n view.showError('Failed to export task data');\n return null;\n }\n }\n });\n\n buttons.push({\n text: 'Close',\n class: 'btn-secondary',\n dismiss: true\n });\n\n return await Dialog.showDialog({\n title: `<i class=\"bi bi-info-circle me-2\"></i>Task Details - ${task.id}`,\n body: view,\n size: 'lg',\n scrollable: true,\n buttons: buttons,\n ...options\n });\n }\n}","/**\n * RunnerDetailsView\n *\n * Tabbed runner inspector opened via RunnerDetailsView.show(runner).\n * All tab view classes are internal to this module.\n *\n * Tabs:\n * Overview — identity, channels, uptime (runner object only, no fetch)\n * System Info — OS, CPU, memory, disk, network (GET /api/jobs/runners/sysinfo/<id>)\n * Running Jobs — jobs on this runner (GET /api/jobs/job?runner_id=&status=running)\n * Logs — aggregated logs from running jobs (GET /api/jobs/logs?job_id=)\n * Actions — ping, shutdown, broadcast, export\n */\n\nimport View from '@core/View.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\nimport TabView from '@core/views/navigation/TabView.js';\nimport { JobRunner } from '@core/models/JobRunner.js';\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Shared helpers\n// ─────────────────────────────────────────────────────────────────────────────\n\nfunction formatUptime(seconds) {\n const d = Math.floor(seconds / 86400);\n const h = Math.floor((seconds % 86400) / 3600);\n const m = Math.floor((seconds % 3600) / 60);\n if (d > 0) return `${d}d ${h}h ${m}m`;\n if (h > 0) return `${h}h ${m}m`;\n return `${m}m`;\n}\n\nfunction formatDuration(seconds) {\n if (seconds < 60) return `${Math.round(seconds)}s`;\n if (seconds < 3600) return `${Math.round(seconds / 60)}m ${Math.round(seconds % 60)}s`;\n return `${Math.round(seconds / 3600)}h ${Math.round((seconds % 3600) / 60)}m`;\n}\n\nfunction formatBytes(bytes) {\n if (bytes == null) return 'N/A';\n if (bytes >= 1e9) return (bytes / 1e9).toFixed(2) + ' GB';\n if (bytes >= 1e6) return (bytes / 1e6).toFixed(2) + ' MB';\n if (bytes >= 1e3) return (bytes / 1e3).toFixed(2) + ' KB';\n return bytes + ' B';\n}\n\nfunction resourceBadgeClass(pct) {\n if (pct >= 80) return 'bg-danger-subtle text-danger';\n if (pct >= 60) return 'bg-warning-subtle text-warning';\n return 'bg-success-subtle text-success';\n}\n\nfunction progressBarClass(pct) {\n if (pct >= 80) return 'bg-danger';\n if (pct >= 60) return 'bg-warning';\n return 'bg-success';\n}\n\nfunction heartbeatAgeSec(isoString) {\n if (!isoString) return null;\n return (Date.now() - new Date(isoString).getTime()) / 1000;\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// RunnerOverviewTab\n// ─────────────────────────────────────────────────────────────────────────────\n\nclass RunnerOverviewTab extends View {\n constructor(options = {}) {\n super({ className: 'runner-overview-tab', ...options });\n this.model = options.model || null;\n }\n\n async onBeforeRender() {\n const r = this.model;\n if (!r) return;\n\n this.aliveBadgeClass = r.get('alive') ? 'bg-success-subtle text-success' : 'bg-danger-subtle text-danger';\n this.aliveIcon = r.get('alive') ? 'bi-check-circle-fill' : 'bi-x-circle-fill';\n this.aliveText = r.get('alive') ? 'Alive' : 'Dead';\n\n const started = r.get('started');\n this.startedText = started ? new Date(started).toLocaleString() : 'N/A';\n\n if (started) {\n const sec = (Date.now() - new Date(started).getTime()) / 1000;\n this.uptimeText = formatUptime(sec);\n } else {\n this.uptimeText = 'N/A';\n }\n\n const ageSec = heartbeatAgeSec(r.get('last_heartbeat'));\n if (ageSec !== null) {\n this.heartbeatText = new Date(r.get('last_heartbeat')).toLocaleString();\n this.heartbeatAgeText = `${Math.round(ageSec)}s ago`;\n this.heartbeatClass = ageSec < 30 ? 'text-success' : ageSec < 120 ? 'text-warning' : 'text-danger';\n } else {\n this.heartbeatText = 'N/A';\n this.heartbeatAgeText = '';\n this.heartbeatClass = 'text-muted';\n }\n\n const jobsProcessed = r.get('jobs_processed') || 0;\n const jobsFailed = r.get('jobs_failed') || 0;\n this.errorRate = (jobsProcessed > 0)\n ? ((jobsFailed / jobsProcessed) * 100).toFixed(2) + '%'\n : '0.00%';\n }\n\n async getTemplate() {\n return `\n {{^model}}\n <div class=\"alert alert-warning\"><i class=\"bi bi-exclamation-triangle me-2\"></i>No runner data available.</div>\n {{/model}}\n\n {{#model}}\n <div class=\"card border-0 shadow-sm mb-3\">\n <div class=\"card-header bg-white border-bottom py-2 d-flex justify-content-between align-items-center\">\n <h6 class=\"mb-0 fw-semibold\">\n <i class=\"bi bi-info-circle text-primary me-2\"></i>Identity\n </h6>\n <span class=\"badge {{aliveBadgeClass}}\">\n <i class=\"bi {{aliveIcon}} me-1\"></i>{{aliveText}}\n </span>\n </div>\n <div class=\"card-body\">\n <div class=\"row g-3\">\n <div class=\"col-md-6\">\n <table class=\"table table-sm table-borderless mb-0\">\n <tbody>\n <tr>\n <td class=\"text-muted small fw-semibold text-uppercase pe-3\" style=\"width:38%;white-space:nowrap;font-size:0.72rem;\">Runner ID</td>\n <td class=\"font-monospace small\">{{model.runner_id}}</td>\n </tr>\n <tr>\n <td class=\"text-muted small fw-semibold text-uppercase\" style=\"font-size:0.72rem;\">Started</td>\n <td class=\"small\">{{startedText}}</td>\n </tr>\n <tr>\n <td class=\"text-muted small fw-semibold text-uppercase\" style=\"font-size:0.72rem;\">Uptime</td>\n <td class=\"small fw-semibold\">{{uptimeText}}</td>\n </tr>\n <tr>\n <td class=\"text-muted small fw-semibold text-uppercase\" style=\"font-size:0.72rem;\">Heartbeat</td>\n <td class=\"small {{heartbeatClass}}\">\n {{heartbeatText}}\n {{#heartbeatAgeText}}\n <span class=\"text-muted\">({{heartbeatAgeText}})</span>\n {{/heartbeatAgeText}}\n </td>\n </tr>\n </tbody>\n </table>\n </div>\n <div class=\"col-md-6\">\n <table class=\"table table-sm table-borderless mb-0\">\n <tbody>\n <tr>\n <td class=\"text-muted small fw-semibold text-uppercase pe-3\" style=\"width:38%;white-space:nowrap;font-size:0.72rem;\">Jobs Done</td>\n <td class=\"small fw-bold text-success\">{{model.jobs_processed|number}}</td>\n </tr>\n <tr>\n <td class=\"text-muted small fw-semibold text-uppercase\" style=\"font-size:0.72rem;\">Jobs Failed</td>\n <td class=\"small fw-bold text-danger\">{{model.jobs_failed|number}}</td>\n </tr>\n <tr>\n <td class=\"text-muted small fw-semibold text-uppercase\" style=\"font-size:0.72rem;\">Error Rate</td>\n <td class=\"small\">{{errorRate}}</td>\n </tr>\n </tbody>\n </table>\n </div>\n </div>\n\n {{#model.channels.length}}\n <div class=\"mt-3 pt-3 border-top\">\n <div class=\"text-muted small fw-semibold text-uppercase mb-2\" style=\"font-size:0.72rem;\">Assigned Channels</div>\n <div class=\"d-flex flex-wrap gap-2\">\n {{#model.channels}}\n <span class=\"badge bg-primary-subtle text-primary px-3 py-2\">\n <i class=\"bi bi-circle-fill me-1\" style=\"font-size:0.4rem;vertical-align:middle;\"></i>{{.}}\n </span>\n {{/model.channels}}\n </div>\n </div>\n {{/model.channels.length}}\n </div>\n </div>\n\n <div class=\"alert alert-light border d-flex align-items-center gap-2 mb-0\" style=\"font-size:0.83rem;\">\n <i class=\"bi bi-cpu text-primary flex-shrink-0\"></i>\n <span>CPU, memory, disk, and network detail is in the <strong>System Info</strong> tab.</span>\n </div>\n {{/model}}\n `;\n }\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// RunnerSysinfoTab\n// ─────────────────────────────────────────────────────────────────────────────\n\nclass RunnerSysinfoTab extends View {\n constructor(options = {}) {\n super({ className: 'runner-sysinfo-tab', ...options });\n this.model = options.model || null;\n this.sysinfo = null;\n this.sysinfoError = null;\n this.loading = false;\n this.loaded = false;\n }\n\n async onTabActivated() {\n if (this.loaded) return;\n this.loaded = true;\n this.loading = true;\n this.sysinfoError = null;\n await this.render();\n await this.loadSysinfo();\n this.loading = false;\n await this.render();\n }\n\n async loadSysinfo() {\n try {\n const resp = await this.getApp().rest.GET(\n `/api/jobs/runners/sysinfo/${this.model.get('runner_id')}`\n );\n if (resp.success && resp.data) {\n const payload = resp.data.data || resp.data;\n if (payload && payload.status === 'error') {\n this.sysinfoError = payload.error || 'Runner reported an error collecting sysinfo.';\n return;\n }\n if (!resp.data.status) {\n this.sysinfoError = resp.data.error || 'Could not load system info.';\n return;\n }\n this.sysinfo = payload.result || payload;\n this.enrichSysinfo();\n } else {\n this.sysinfoError = 'Could not load system info.';\n }\n } catch (e) {\n this.sysinfoError = e.message || 'Request failed.';\n }\n }\n\n enrichSysinfo() {\n const s = this.sysinfo;\n if (!s) return;\n\n if (s.memory) {\n s.memory.totalFmt = formatBytes(s.memory.total);\n s.memory.usedFmt = formatBytes(s.memory.used);\n s.memory.availableFmt = formatBytes(s.memory.available);\n s.memory.barClass = progressBarClass(s.memory.percent);\n s.memory.badgeClass = resourceBadgeClass(s.memory.percent);\n }\n\n if (s.disk) {\n s.disk.totalFmt = formatBytes(s.disk.total);\n s.disk.usedFmt = formatBytes(s.disk.used);\n s.disk.freeFmt = formatBytes(s.disk.free);\n s.disk.barClass = progressBarClass(s.disk.percent);\n s.disk.badgeClass = resourceBadgeClass(s.disk.percent);\n }\n\n if (s.network) {\n s.network.bytesRecvFmt = formatBytes(s.network.bytes_recv);\n s.network.bytesSentFmt = formatBytes(s.network.bytes_sent);\n s.network.errClass = (s.network.errin > 0 || s.network.errout > 0)\n ? 'text-danger fw-bold' : 'text-success';\n s.network.dropClass = (s.network.dropin > 0 || s.network.dropout > 0)\n ? 'text-warning fw-bold' : 'text-success';\n }\n\n const cpuPct = s.cpu_load || 0;\n s.cpuLoadBarClass = progressBarClass(cpuPct);\n s.cpuLoadBadgeClass = resourceBadgeClass(cpuPct);\n\n if (s.cpu && s.cpu.freq) {\n s.cpu.freqText = `${Math.round(s.cpu.freq.current).toLocaleString()} MHz current`\n + ` · ${Math.round(s.cpu.freq.max).toLocaleString()} MHz max`;\n } else if (s.cpu) {\n s.cpu.freqText = null;\n }\n\n if (s.cpus_load && s.cpus_load.length) {\n s.cpuCores = s.cpus_load.map((pct, i) => ({\n index: i,\n pct: pct.toFixed(1),\n barClass: progressBarClass(pct)\n }));\n } else {\n s.cpuCores = [];\n }\n\n s.bootDatetime = s.boot_time ? new Date(s.boot_time * 1000).toLocaleString() : null;\n s.collectedText = s.datetime || null;\n }\n\n async onActionRefreshSysinfo() {\n this.loaded = false;\n this.sysinfo = null;\n this.sysinfoError = null;\n await this.onTabActivated();\n }\n\n async getTemplate() {\n return `\n {{#loading|bool}}\n <div class=\"text-center py-5\">\n <div class=\"spinner-border text-primary\" role=\"status\"></div>\n <div class=\"mt-2 text-muted small\">Loading system info…</div>\n </div>\n {{/loading|bool}}\n\n {{^loading|bool}}\n\n {{#sysinfoError|bool}}\n <div class=\"alert alert-warning d-flex align-items-start gap-2\">\n <i class=\"bi bi-exclamation-triangle flex-shrink-0 mt-1\"></i>\n <div>\n <strong>Could not load system info</strong><br>\n <span class=\"small\">{{sysinfoError}}</span><br>\n <button class=\"btn btn-sm btn-outline-warning mt-2\" data-action=\"refresh-sysinfo\">\n <i class=\"bi bi-arrow-clockwise me-1\"></i>Retry\n </button>\n </div>\n </div>\n {{/sysinfoError|bool}}\n\n {{#sysinfo|bool}}\n\n <div class=\"d-flex justify-content-end align-items-center gap-3 mb-3\">\n {{#sysinfo.collectedText}}\n <small class=\"text-muted\">Collected {{sysinfo.collectedText}}</small>\n {{/sysinfo.collectedText}}\n <button class=\"btn btn-sm btn-outline-secondary\" data-action=\"refresh-sysinfo\">\n <i class=\"bi bi-arrow-clockwise me-1\"></i>Refresh\n </button>\n </div>\n\n <!-- OS -->\n <div class=\"card border-0 shadow-sm mb-3\">\n <div class=\"card-header bg-white border-bottom py-2\">\n <h6 class=\"mb-0 fw-semibold\"><i class=\"bi bi-hdd-rack text-primary me-2\"></i>Operating System</h6>\n </div>\n <div class=\"card-body p-0\">\n <table class=\"table table-sm table-borderless mb-0\">\n <tbody>\n {{#sysinfo.os}}\n <tr>\n <td class=\"text-muted small fw-semibold text-uppercase ps-3 pe-2\" style=\"width:20%;font-size:0.72rem;white-space:nowrap;\">Hostname</td>\n <td class=\"font-monospace small\">{{.hostname}}</td>\n <td class=\"text-muted small fw-semibold text-uppercase pe-2\" style=\"width:15%;font-size:0.72rem;white-space:nowrap;\">OS</td>\n <td class=\"small\">{{.system}}</td>\n </tr>\n <tr>\n <td class=\"text-muted small fw-semibold text-uppercase ps-3 pe-2\" style=\"font-size:0.72rem;\">Release</td>\n <td class=\"font-monospace small\">{{.release}}</td>\n <td class=\"text-muted small fw-semibold text-uppercase pe-2\" style=\"font-size:0.72rem;\">Machine</td>\n <td class=\"font-monospace small\">{{.machine}}</td>\n </tr>\n <tr>\n <td class=\"text-muted small fw-semibold text-uppercase ps-3 pe-2\" style=\"font-size:0.72rem;\">Version</td>\n <td colspan=\"3\" class=\"font-monospace small text-muted\" style=\"font-size:0.76rem;\">{{.version}}</td>\n </tr>\n {{/sysinfo.os}}\n {{#sysinfo.bootDatetime}}\n <tr>\n <td class=\"text-muted small fw-semibold text-uppercase ps-3 pe-2\" style=\"font-size:0.72rem;\">Boot Time</td>\n <td colspan=\"3\" class=\"small\">{{sysinfo.bootDatetime}}</td>\n </tr>\n {{/sysinfo.bootDatetime}}\n </tbody>\n </table>\n </div>\n </div>\n\n <!-- CPU -->\n <div class=\"card border-0 shadow-sm mb-3\">\n <div class=\"card-header bg-white border-bottom py-2 d-flex justify-content-between align-items-center\">\n <h6 class=\"mb-0 fw-semibold\"><i class=\"bi bi-cpu text-primary me-2\"></i>CPU</h6>\n <span class=\"badge {{sysinfo.cpuLoadBadgeClass}}\">{{sysinfo.cpu_load}}% overall</span>\n </div>\n <div class=\"card-body\">\n <div class=\"d-flex justify-content-between mb-1\">\n <small class=\"fw-semibold text-muted\">Overall Load</small>\n <small class=\"fw-bold\">{{sysinfo.cpu_load}}%</small>\n </div>\n <div class=\"progress mb-2\" style=\"height:8px;\">\n <div class=\"progress-bar {{sysinfo.cpuLoadBarClass}}\" style=\"width:{{sysinfo.cpu_load}}%;\"></div>\n </div>\n {{#sysinfo.cpu}}\n <div class=\"text-muted small mb-3\">\n {{.count}} logical cores\n {{#.freqText}}&nbsp;·&nbsp;{{.freqText}}{{/.freqText}}\n </div>\n {{/sysinfo.cpu}}\n\n {{#sysinfo.cpuCores.length}}\n <div class=\"row g-2\">\n {{#sysinfo.cpuCores}}\n <div class=\"col-6 col-md-3\">\n <div class=\"border rounded p-2 bg-light text-center\">\n <div class=\"text-muted fw-semibold text-uppercase mb-1\" style=\"font-size:0.65rem;\">Core {{.index}}</div>\n <div class=\"fw-bold small\">{{.pct}}%</div>\n <div class=\"progress mt-1\" style=\"height:4px;\">\n <div class=\"progress-bar {{.barClass}}\" style=\"width:{{.pct}}%;\"></div>\n </div>\n </div>\n </div>\n {{/sysinfo.cpuCores}}\n </div>\n {{/sysinfo.cpuCores.length}}\n </div>\n </div>\n\n <!-- Memory -->\n {{#sysinfo.memory}}\n <div class=\"card border-0 shadow-sm mb-3\">\n <div class=\"card-header bg-white border-bottom py-2 d-flex justify-content-between align-items-center\">\n <h6 class=\"mb-0 fw-semibold\"><i class=\"bi bi-memory text-primary me-2\"></i>Memory</h6>\n <span class=\"badge {{.badgeClass}}\">{{.percent}}% used</span>\n </div>\n <div class=\"card-body\">\n <div class=\"d-flex justify-content-between mb-1\">\n <small class=\"fw-semibold text-muted\">RAM Usage</small>\n <small class=\"fw-bold\">{{.usedFmt}} / {{.totalFmt}}</small>\n </div>\n <div class=\"progress mb-3\" style=\"height:8px;\">\n <div class=\"progress-bar {{.barClass}}\" style=\"width:{{.percent}}%;\"></div>\n </div>\n <div class=\"row g-2 text-center\">\n <div class=\"col-6 col-md-3\">\n <div class=\"text-muted fw-semibold text-uppercase\" style=\"font-size:0.7rem;\">Total</div>\n <div class=\"fw-semibold small\">{{.totalFmt}}</div>\n </div>\n <div class=\"col-6 col-md-3\">\n <div class=\"text-muted fw-semibold text-uppercase\" style=\"font-size:0.7rem;\">Used</div>\n <div class=\"fw-semibold small\">{{.usedFmt}}</div>\n </div>\n <div class=\"col-6 col-md-3\">\n <div class=\"text-muted fw-semibold text-uppercase\" style=\"font-size:0.7rem;\">Available</div>\n <div class=\"fw-semibold small text-success\">{{.availableFmt}}</div>\n </div>\n <div class=\"col-6 col-md-3\">\n <div class=\"text-muted fw-semibold text-uppercase\" style=\"font-size:0.7rem;\">Percent</div>\n <div class=\"fw-semibold small\">{{.percent}}%</div>\n </div>\n </div>\n </div>\n </div>\n {{/sysinfo.memory}}\n\n <!-- Disk -->\n {{#sysinfo.disk}}\n <div class=\"card border-0 shadow-sm mb-3\">\n <div class=\"card-header bg-white border-bottom py-2 d-flex justify-content-between align-items-center\">\n <h6 class=\"mb-0 fw-semibold\"><i class=\"bi bi-hdd text-primary me-2\"></i>Disk (Root)</h6>\n <span class=\"badge {{.badgeClass}}\">{{.percent}}% used</span>\n </div>\n <div class=\"card-body\">\n <div class=\"d-flex justify-content-between mb-1\">\n <small class=\"fw-semibold text-muted\">Disk Usage</small>\n <small class=\"fw-bold\">{{.usedFmt}} / {{.totalFmt}}</small>\n </div>\n <div class=\"progress mb-3\" style=\"height:8px;\">\n <div class=\"progress-bar {{.barClass}}\" style=\"width:{{.percent}}%;\"></div>\n </div>\n <div class=\"row g-2 text-center\">\n <div class=\"col-6 col-md-3\">\n <div class=\"text-muted fw-semibold text-uppercase\" style=\"font-size:0.7rem;\">Total</div>\n <div class=\"fw-semibold small\">{{.totalFmt}}</div>\n </div>\n <div class=\"col-6 col-md-3\">\n <div class=\"text-muted fw-semibold text-uppercase\" style=\"font-size:0.7rem;\">Used</div>\n <div class=\"fw-semibold small\">{{.usedFmt}}</div>\n </div>\n <div class=\"col-6 col-md-3\">\n <div class=\"text-muted fw-semibold text-uppercase\" style=\"font-size:0.7rem;\">Free</div>\n <div class=\"fw-semibold small text-success\">{{.freeFmt}}</div>\n </div>\n <div class=\"col-6 col-md-3\">\n <div class=\"text-muted fw-semibold text-uppercase\" style=\"font-size:0.7rem;\">Percent</div>\n <div class=\"fw-semibold small\">{{.percent}}%</div>\n </div>\n </div>\n </div>\n </div>\n {{/sysinfo.disk}}\n\n <!-- Network -->\n {{#sysinfo.network}}\n <div class=\"card border-0 shadow-sm mb-3\">\n <div class=\"card-header bg-white border-bottom py-2\">\n <h6 class=\"mb-0 fw-semibold\"><i class=\"bi bi-diagram-3 text-primary me-2\"></i>Network</h6>\n </div>\n <div class=\"card-body\">\n <div class=\"row g-2\">\n <div class=\"col-6 col-md-4\">\n <div class=\"border rounded p-2 bg-light\">\n <div class=\"text-muted fw-semibold text-uppercase mb-1\" style=\"font-size:0.67rem;\"><i class=\"bi bi-arrow-down text-primary me-1\"></i>Bytes Recv</div>\n <div class=\"fw-bold font-monospace small\">{{.bytesRecvFmt}}</div>\n </div>\n </div>\n <div class=\"col-6 col-md-4\">\n <div class=\"border rounded p-2 bg-light\">\n <div class=\"text-muted fw-semibold text-uppercase mb-1\" style=\"font-size:0.67rem;\"><i class=\"bi bi-arrow-up text-primary me-1\"></i>Bytes Sent</div>\n <div class=\"fw-bold font-monospace small\">{{.bytesSentFmt}}</div>\n </div>\n </div>\n <div class=\"col-6 col-md-4\">\n <div class=\"border rounded p-2 bg-light\">\n <div class=\"text-muted fw-semibold text-uppercase mb-1\" style=\"font-size:0.67rem;\"><i class=\"bi bi-share text-primary me-1\"></i>TCP Connections</div>\n <div class=\"fw-bold font-monospace small\">{{.tcp_cons|number}}</div>\n </div>\n </div>\n <div class=\"col-6 col-md-4\">\n <div class=\"border rounded p-2 bg-light\">\n <div class=\"text-muted fw-semibold text-uppercase mb-1\" style=\"font-size:0.67rem;\"><i class=\"bi bi-arrow-down me-1\"></i>Packets Recv</div>\n <div class=\"fw-bold font-monospace small\">{{.packets_recv|number}}</div>\n </div>\n </div>\n <div class=\"col-6 col-md-4\">\n <div class=\"border rounded p-2 bg-light\">\n <div class=\"text-muted fw-semibold text-uppercase mb-1\" style=\"font-size:0.67rem;\"><i class=\"bi bi-arrow-up me-1\"></i>Packets Sent</div>\n <div class=\"fw-bold font-monospace small\">{{.packets_sent|number}}</div>\n </div>\n </div>\n <div class=\"col-6 col-md-4\">\n <div class=\"border rounded p-2 bg-light\">\n <div class=\"text-muted fw-semibold text-uppercase mb-1\" style=\"font-size:0.67rem;\"><i class=\"bi bi-exclamation-triangle me-1\"></i>Errors In / Out</div>\n <div class=\"fw-bold font-monospace small {{.errClass}}\">{{.errin}} / {{.errout}}</div>\n </div>\n </div>\n <div class=\"col-6 col-md-4\">\n <div class=\"border rounded p-2 bg-light\">\n <div class=\"text-muted fw-semibold text-uppercase mb-1\" style=\"font-size:0.67rem;\"><i class=\"bi bi-x-circle me-1\"></i>Drops In</div>\n <div class=\"fw-bold font-monospace small {{.dropClass}}\">{{.dropin}}</div>\n </div>\n </div>\n <div class=\"col-6 col-md-4\">\n <div class=\"border rounded p-2 bg-light\">\n <div class=\"text-muted fw-semibold text-uppercase mb-1\" style=\"font-size:0.67rem;\"><i class=\"bi bi-x-circle me-1\"></i>Drops Out</div>\n <div class=\"fw-bold font-monospace small {{.dropClass}}\">{{.dropout}}</div>\n </div>\n </div>\n </div>\n </div>\n </div>\n {{/sysinfo.network}}\n\n <!-- Logged-in Users -->\n <div class=\"card border-0 shadow-sm\">\n <div class=\"card-header bg-white border-bottom py-2\">\n <h6 class=\"mb-0 fw-semibold\"><i class=\"bi bi-person-badge text-primary me-2\"></i>Logged-in Users</h6>\n </div>\n {{#sysinfo.users.length}}\n <ul class=\"list-group list-group-flush\">\n {{#sysinfo.users}}\n <li class=\"list-group-item font-monospace small\">{{.name|default:'unknown'}}\n {{#.terminal}}<span class=\"text-muted ms-2\">{{.terminal}}</span>{{/.terminal}}\n </li>\n {{/sysinfo.users}}\n </ul>\n {{/sysinfo.users.length}}\n {{^sysinfo.users.length}}\n <div class=\"card-body text-center text-muted py-4\">\n <i class=\"bi bi-person-x fs-2 d-block mb-2 opacity-50\"></i>\n <div class=\"small\">No users currently logged in</div>\n </div>\n {{/sysinfo.users.length}}\n </div>\n\n {{/sysinfo|bool}}\n {{/loading|bool}}\n `;\n }\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// RunnerJobsTab\n// ─────────────────────────────────────────────────────────────────────────────\n\nclass RunnerJobsTab extends View {\n constructor(options = {}) {\n super({ className: 'runner-jobs-tab', ...options });\n this.model = options.model || null;\n this.jobs = [];\n this.loading = false;\n this.loaded = false;\n }\n\n async onTabActivated() {\n if (this.loaded) return;\n this.loaded = true;\n this.loading = true;\n await this.render();\n await this.loadJobs();\n this.loading = false;\n await this.render();\n }\n\n async loadJobs() {\n try {\n const resp = await this.getApp().rest.GET(\n `/api/jobs/job?runner_id=${this.model.get('runner_id')}&status=running&size=50`\n );\n if (resp.success && resp.data && resp.data.status) {\n const now = Date.now() / 1000;\n this.jobs = (resp.data.data || []).map(job => ({\n ...job,\n durationText: job.started_at\n ? formatDuration(now - new Date(job.started_at).getTime() / 1000)\n : 'N/A',\n startedText: job.started_at\n ? new Date(job.started_at).toLocaleTimeString()\n : 'N/A',\n attemptBadgeClass: job.attempt > 1\n ? 'bg-danger-subtle text-danger'\n : 'bg-warning-subtle text-warning'\n }));\n } else {\n this.jobs = [];\n }\n } catch (e) {\n this.jobs = [];\n this.showError('Could not load running jobs: ' + e.message);\n }\n }\n\n async onActionRefreshJobs() {\n this.loaded = false;\n this.jobs = [];\n await this.onTabActivated();\n }\n\n async onActionViewJob(event, element) {\n const jobId = element.dataset.jobId;\n this.emit('job:view', { jobId, runner: this.model });\n }\n\n async onActionCancelJob(event, element) {\n const jobId = element.dataset.jobId;\n const ok = await Dialog.confirm(\n 'Cancel this job? The runner will receive a cooperative cancel signal.',\n 'Cancel Job',\n { confirmText: 'Cancel Job', confirmClass: 'btn-warning' }\n );\n if (!ok) return;\n\n try {\n const resp = await this.getApp().rest.POST(\n `/api/jobs/job/${jobId}`, { cancel_request: true }\n );\n if (resp.success && resp.data && resp.data.status) {\n this.showSuccess('Cancel signal sent.');\n this.loaded = false;\n await this.onTabActivated();\n } else {\n this.showError((resp.data && resp.data.error) || 'Could not cancel job.');\n }\n } catch (e) {\n this.showError('Could not cancel job: ' + e.message);\n }\n }\n\n async getTemplate() {\n return `\n {{#loading|bool}}\n <div class=\"text-center py-5\">\n <div class=\"spinner-border text-primary\" role=\"status\"></div>\n <div class=\"mt-2 text-muted small\">Loading running jobs…</div>\n </div>\n {{/loading|bool}}\n\n {{^loading|bool}}\n <div class=\"d-flex justify-content-between align-items-center mb-3\">\n <small class=\"text-muted\">{{jobs.length}} job(s) currently executing on this runner</small>\n <button class=\"btn btn-sm btn-outline-secondary\" data-action=\"refresh-jobs\">\n <i class=\"bi bi-arrow-clockwise me-1\"></i>Refresh\n </button>\n </div>\n\n {{#jobs.length}}\n <div class=\"card border-0 shadow-sm\">\n <div class=\"table-responsive\">\n <table class=\"table table-sm table-hover align-middle mb-0\">\n <thead class=\"table-light\">\n <tr>\n <th class=\"ps-3 border-0 text-muted fw-semibold text-uppercase\" style=\"font-size:0.72rem;letter-spacing:0.04em;\">Job ID</th>\n <th class=\"border-0 text-muted fw-semibold text-uppercase\" style=\"font-size:0.72rem;letter-spacing:0.04em;\">Function</th>\n <th class=\"border-0 text-muted fw-semibold text-uppercase\" style=\"font-size:0.72rem;letter-spacing:0.04em;\">Channel</th>\n <th class=\"border-0 text-muted fw-semibold text-uppercase\" style=\"font-size:0.72rem;letter-spacing:0.04em;\">Started</th>\n <th class=\"border-0 text-muted fw-semibold text-uppercase\" style=\"font-size:0.72rem;letter-spacing:0.04em;\">Duration</th>\n <th class=\"border-0 text-muted fw-semibold text-uppercase\" style=\"font-size:0.72rem;letter-spacing:0.04em;\">Attempt</th>\n <th class=\"border-0 text-end pe-3 text-muted fw-semibold text-uppercase\" style=\"font-size:0.72rem;letter-spacing:0.04em;\">Actions</th>\n </tr>\n </thead>\n <tbody>\n {{#jobs}}\n <tr>\n <td class=\"ps-3\">\n <span class=\"font-monospace text-primary small\" title=\"{{.id}}\">{{.id|truncate:12}}</span>\n </td>\n <td>\n <span class=\"font-monospace text-muted small\" title=\"{{.func}}\">{{.func|truncate:42}}</span>\n </td>\n <td>\n <span class=\"badge bg-primary-subtle text-primary\">{{.channel}}</span>\n </td>\n <td><small class=\"text-muted\">{{.startedText}}</small></td>\n <td><span class=\"badge bg-light text-secondary border\">{{.durationText}}</span></td>\n <td><span class=\"badge {{.attemptBadgeClass}}\">{{.attempt}}</span></td>\n <td class=\"text-end pe-3\">\n <div class=\"btn-group btn-group-sm\">\n <button class=\"btn btn-outline-primary btn-sm\" data-action=\"view-job\"\n data-job-id=\"{{.id}}\" title=\"View job details\">\n <i class=\"bi bi-eye\"></i>\n </button>\n <button class=\"btn btn-outline-warning btn-sm\" data-action=\"cancel-job\"\n data-job-id=\"{{.id}}\" title=\"Cancel job\">\n <i class=\"bi bi-x-circle\"></i>\n </button>\n </div>\n </td>\n </tr>\n {{/jobs}}\n </tbody>\n </table>\n </div>\n </div>\n {{/jobs.length}}\n\n {{^jobs.length}}\n <div class=\"text-center text-muted py-5\">\n <i class=\"bi bi-list-task fs-2 d-block mb-2 opacity-50\"></i>\n <div class=\"small\">No jobs currently executing on this runner</div>\n </div>\n {{/jobs.length}}\n {{/loading|bool}}\n `;\n }\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// RunnerLogsTab\n// ─────────────────────────────────────────────────────────────────────────────\n\nclass RunnerLogsTab extends View {\n constructor(options = {}) {\n super({ className: 'runner-logs-tab', ...options });\n this.model = options.model || null;\n this.logs = [];\n this.filteredLogs = [];\n this.logFilter = 'all';\n this.loading = false;\n this.loaded = false;\n // precomputed filter button classes\n this.filterAllClass = 'btn-primary';\n this.filterDebugClass = 'btn-outline-secondary';\n this.filterInfoClass = 'btn-outline-primary';\n this.filterWarnClass = 'btn-outline-warning';\n this.filterErrorClass = 'btn-outline-danger';\n }\n\n async onTabActivated() {\n if (this.loaded) return;\n this.loaded = true;\n this.loading = true;\n await this.render();\n await this.loadLogs();\n this.loading = false;\n await this.render();\n }\n\n async loadLogs() {\n try {\n // Get running job IDs for this runner first\n const jobsResp = await this.getApp().rest.GET(\n `/api/jobs/job?runner_id=${this.model.get('runner_id')}&status=running&size=50`\n );\n\n const jobIds = [];\n if (jobsResp.success && jobsResp.data && jobsResp.data.status) {\n (jobsResp.data.data || []).forEach(j => jobIds.push(j.id));\n }\n\n if (!jobIds.length) {\n this.logs = [];\n return;\n }\n\n // Fetch logs for each job in parallel (cap at 5 jobs)\n const promises = jobIds.slice(0, 5).map(id =>\n this.getApp().rest.GET(`/api/jobs/logs?job_id=${id}&sort=-created&size=20`)\n .then(r => (r.success && r.data && r.data.status) ? (r.data.data || []) : [])\n .catch(() => [])\n );\n\n const results = await Promise.all(promises);\n const all = [].concat(...results);\n\n all.sort((a, b) => new Date(b.created) - new Date(a.created));\n this.logs = all.slice(0, 50).map(log => ({\n ...log,\n levelBadgeClass: this.getLogLevelClass(log.kind),\n kindDisplay: (log.kind || 'info').toUpperCase(),\n createdText: new Date(log.created).toLocaleTimeString()\n }));\n } catch (e) {\n this.logs = [];\n this.showError('Could not load logs: ' + e.message);\n }\n }\n\n getLogLevelClass(kind) {\n const map = {\n debug: 'bg-secondary-subtle text-secondary',\n info: 'bg-primary-subtle text-primary',\n warn: 'bg-warning-subtle text-warning',\n error: 'bg-danger-subtle text-danger'\n };\n return map[kind] || 'bg-secondary-subtle text-secondary';\n }\n\n async onBeforeRender() {\n this.filteredLogs = this.logFilter === 'all'\n ? this.logs\n : this.logs.filter(l => l.kind === this.logFilter);\n\n this.filterAllClass = this.logFilter === 'all' ? 'btn-primary' : 'btn-outline-secondary';\n this.filterDebugClass = this.logFilter === 'debug' ? 'btn-secondary' : 'btn-outline-secondary';\n this.filterInfoClass = this.logFilter === 'info' ? 'btn-primary' : 'btn-outline-primary';\n this.filterWarnClass = this.logFilter === 'warn' ? 'btn-warning' : 'btn-outline-warning';\n this.filterErrorClass = this.logFilter === 'error' ? 'btn-danger' : 'btn-outline-danger';\n }\n\n async onActionFilterLogs(event, element) {\n this.logFilter = element.dataset.kind || 'all';\n await this.render();\n }\n\n async onActionRefreshLogs() {\n this.loaded = false;\n this.logs = [];\n this.logFilter = 'all';\n await this.onTabActivated();\n }\n\n async getTemplate() {\n return `\n {{#loading|bool}}\n <div class=\"text-center py-5\">\n <div class=\"spinner-border text-primary\" role=\"status\"></div>\n <div class=\"mt-2 text-muted small\">Loading logs…</div>\n </div>\n {{/loading|bool}}\n\n {{^loading|bool}}\n <div class=\"card border-0 shadow-sm\">\n <div class=\"card-header bg-white border-bottom py-2 d-flex align-items-center gap-2 flex-wrap\">\n <small class=\"text-muted fw-semibold me-1\">Filter:</small>\n <button class=\"btn btn-sm {{filterAllClass}}\" data-action=\"filter-logs\" data-kind=\"all\">All</button>\n <button class=\"btn btn-sm {{filterDebugClass}}\" data-action=\"filter-logs\" data-kind=\"debug\">Debug</button>\n <button class=\"btn btn-sm {{filterInfoClass}}\" data-action=\"filter-logs\" data-kind=\"info\">Info</button>\n <button class=\"btn btn-sm {{filterWarnClass}}\" data-action=\"filter-logs\" data-kind=\"warn\">Warning</button>\n <button class=\"btn btn-sm {{filterErrorClass}}\" data-action=\"filter-logs\" data-kind=\"error\">Error</button>\n <div class=\"ms-auto d-flex align-items-center gap-2\">\n <small class=\"text-muted\">{{filteredLogs.length}} entries</small>\n <button class=\"btn btn-sm btn-outline-secondary\" data-action=\"refresh-logs\">\n <i class=\"bi bi-arrow-clockwise\"></i>\n </button>\n </div>\n </div>\n\n <div style=\"max-height:420px;overflow-y:auto;\">\n {{#filteredLogs.length}}\n {{#filteredLogs}}\n <div class=\"d-flex align-items-start gap-2 px-3 py-2 border-bottom font-monospace\" style=\"font-size:0.78rem;\">\n <span class=\"text-muted flex-shrink-0 pt-1\" style=\"min-width:65px;\">{{.createdText}}</span>\n <span class=\"badge {{.levelBadgeClass}} flex-shrink-0\" style=\"margin-top:1px;\">{{.kindDisplay}}</span>\n <span class=\"flex-grow-1 text-break\">{{.message}}</span>\n </div>\n {{/filteredLogs}}\n {{/filteredLogs.length}}\n\n {{^filteredLogs.length}}\n <div class=\"text-center text-muted py-5\">\n <i class=\"bi bi-journal fs-2 d-block mb-2 opacity-50\"></i>\n <div class=\"small\">No log entries</div>\n </div>\n {{/filteredLogs.length}}\n </div>\n </div>\n {{/loading|bool}}\n `;\n }\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// RunnerActionsTab\n// ─────────────────────────────────────────────────────────────────────────────\n\nclass RunnerActionsTab extends View {\n constructor(options = {}) {\n super({ className: 'runner-actions-tab', ...options });\n this.model = options.model || null;\n this.pingResult = null;\n }\n\n async onActionPing() {\n this.pingResult = null;\n await this.render();\n try {\n const resp = await this.getApp().rest.POST('/api/jobs/runners/ping', {\n runner_id: this.model.get('runner_id'),\n timeout: 2.0\n });\n if (resp.success && resp.data) {\n this.pingResult = resp.data.responsive\n ? '<span class=\"text-success\"><i class=\"bi bi-check-circle-fill me-1\"></i>Runner is responsive</span>'\n : '<span class=\"text-warning\"><i class=\"bi bi-exclamation-triangle-fill me-1\"></i>Runner did not respond within 2s</span>';\n } else {\n this.pingResult = '<span class=\"text-danger\"><i class=\"bi bi-x-circle-fill me-1\"></i>Ping request failed</span>';\n }\n } catch (e) {\n this.pingResult = `<span class=\"text-danger\"><i class=\"bi bi-x-circle-fill me-1\"></i>${e.message}</span>`;\n }\n await this.render();\n }\n\n async onActionShutdown() {\n const ok = await Dialog.confirm(\n `Send a graceful shutdown to <strong class=\"font-monospace\">${this.model.get('runner_id')}</strong>?`\n + '<br><br>The runner will finish its current job then exit. This is fire-and-forget.',\n 'Shutdown Runner',\n { confirmText: 'Shutdown', confirmClass: 'btn-danger' }\n );\n if (!ok) return;\n\n try {\n const resp = await this.getApp().rest.POST('/api/jobs/runners/shutdown', {\n runner_id: this.model.get('runner_id'),\n graceful: true\n });\n if (resp.success && resp.data && resp.data.status) {\n this.showSuccess('Shutdown command sent to runner.');\n this.emit('runner:shutdown', { runner: this.model });\n } else {\n this.showError((resp.data && resp.data.error) || 'Shutdown command failed.');\n }\n } catch (e) {\n this.showError('Shutdown failed: ' + e.message);\n }\n }\n\n async onActionBroadcast() {\n const commandEl = this.element && this.element.querySelector('[data-field=\"broadcast-command\"]');\n const timeoutEl = this.element && this.element.querySelector('[data-field=\"broadcast-timeout\"]');\n const command = commandEl ? commandEl.value : 'status';\n const timeout = timeoutEl ? (parseFloat(timeoutEl.value) || 2.0) : 2.0;\n\n Dialog.showBusy({ message: `Broadcasting \"${command}\" to all runners…` });\n try {\n const resp = await this.getApp().rest.POST('/api/jobs/runners/broadcast', {\n command,\n timeout\n });\n Dialog.hideBusy();\n\n if (resp.success && resp.data) {\n await Dialog.showCode(\n JSON.stringify(resp.data, null, 2),\n 'json',\n { title: `Broadcast Response — ${command}`, size: 'lg' }\n );\n } else {\n this.showError((resp.data && resp.data.error) || 'Broadcast failed.');\n }\n } catch (e) {\n Dialog.hideBusy();\n this.showError('Broadcast failed: ' + e.message);\n }\n }\n\n async onActionExport() {\n try {\n const exportData = {\n runner: this.model.toJSON ? this.model.toJSON() : this.model,\n exported_at: new Date().toISOString()\n };\n const blob = new Blob([JSON.stringify(exportData, null, 2)], { type: 'application/json' });\n const url = URL.createObjectURL(blob);\n const a = Object.assign(document.createElement('a'), {\n href: url,\n download: `runner-${this.model.get('runner_id')}-${Date.now()}.json`\n });\n document.body.appendChild(a);\n a.click();\n document.body.removeChild(a);\n URL.revokeObjectURL(url);\n this.showSuccess('Runner data exported.');\n } catch (e) {\n this.showError('Export failed: ' + e.message);\n }\n }\n\n async getTemplate() {\n return `\n <p class=\"text-muted small mb-3\">\n <i class=\"bi bi-info-circle me-1\"></i>\n Actions operate on runner <strong class=\"font-monospace\">{{model.runner_id}}</strong> unless otherwise noted.\n </p>\n\n <div class=\"d-flex align-items-center gap-2 mb-3\">\n <span class=\"text-muted fw-semibold text-uppercase\" style=\"font-size:0.7rem;letter-spacing:0.09em;white-space:nowrap;\">Runner Control</span>\n <hr class=\"flex-grow-1 my-0\">\n </div>\n\n <div class=\"row g-3 mb-4\">\n\n <!-- Ping -->\n <div class=\"col-md-4\">\n <div class=\"card border-0 shadow-sm h-100\">\n <div class=\"card-body d-flex flex-column gap-3\">\n <div class=\"d-flex gap-3 align-items-start\">\n <div class=\"d-flex align-items-center justify-content-center rounded bg-success-subtle text-success flex-shrink-0\"\n style=\"width:40px;height:40px;font-size:1.1rem;\">\n <i class=\"bi bi-broadcast-pin\"></i>\n </div>\n <div>\n <div class=\"fw-semibold mb-1\">Ping Runner</div>\n <div class=\"text-muted small\">Verify this runner is truly responsive, not just alive on paper.</div>\n </div>\n </div>\n {{#pingResult|bool}}\n <div class=\"small\">{{{pingResult}}}</div>\n {{/pingResult|bool}}\n <button class=\"btn btn-sm btn-outline-success mt-auto\" data-action=\"ping\">\n <i class=\"bi bi-broadcast-pin me-1\"></i>Ping Now\n </button>\n </div>\n </div>\n </div>\n\n <!-- Shutdown -->\n <div class=\"col-md-4\">\n <div class=\"card border-0 shadow-sm h-100\">\n <div class=\"card-body d-flex flex-column gap-3\">\n <div class=\"d-flex gap-3 align-items-start\">\n <div class=\"d-flex align-items-center justify-content-center rounded bg-danger-subtle text-danger flex-shrink-0\"\n style=\"width:40px;height:40px;font-size:1.1rem;\">\n <i class=\"bi bi-power\"></i>\n </div>\n <div>\n <div class=\"fw-semibold mb-1\">Graceful Shutdown</div>\n <div class=\"text-muted small\">Runner finishes its current job then exits. Fire-and-forget.</div>\n </div>\n </div>\n <button class=\"btn btn-sm btn-outline-danger mt-auto\" data-action=\"shutdown\">\n <i class=\"bi bi-power me-1\"></i>Shutdown\n </button>\n </div>\n </div>\n </div>\n\n <!-- Export -->\n <div class=\"col-md-4\">\n <div class=\"card border-0 shadow-sm h-100\">\n <div class=\"card-body d-flex flex-column gap-3\">\n <div class=\"d-flex gap-3 align-items-start\">\n <div class=\"d-flex align-items-center justify-content-center rounded bg-secondary-subtle text-secondary flex-shrink-0\"\n style=\"width:40px;height:40px;font-size:1.1rem;\">\n <i class=\"bi bi-download\"></i>\n </div>\n <div>\n <div class=\"fw-semibold mb-1\">Export Snapshot</div>\n <div class=\"text-muted small\">Download runner identity data as a JSON file.</div>\n </div>\n </div>\n <button class=\"btn btn-sm btn-outline-secondary mt-auto\" data-action=\"export\">\n <i class=\"bi bi-download me-1\"></i>Export JSON\n </button>\n </div>\n </div>\n </div>\n\n </div>\n\n <div class=\"d-flex align-items-center gap-2 mb-3\">\n <span class=\"text-muted fw-semibold text-uppercase\" style=\"font-size:0.7rem;letter-spacing:0.09em;white-space:nowrap;\">Broadcast Command</span>\n <hr class=\"flex-grow-1 my-0\">\n </div>\n\n <div class=\"card border-0 shadow-sm\">\n <div class=\"card-body\">\n <p class=\"text-muted small mb-3\">\n Send a command to <strong>all active runners</strong> simultaneously and collect replies within the timeout window.\n </p>\n <div class=\"row g-2 align-items-end\">\n <div class=\"col-md-4\">\n <label class=\"form-label fw-semibold small text-muted mb-1\">Command</label>\n <select class=\"form-select form-select-sm\" data-field=\"broadcast-command\">\n <option value=\"status\">status</option>\n <option value=\"pause\">pause</option>\n <option value=\"resume\">resume</option>\n <option value=\"reload\">reload</option>\n <option value=\"shutdown\">shutdown</option>\n </select>\n </div>\n <div class=\"col-md-3\">\n <label class=\"form-label fw-semibold small text-muted mb-1\">Timeout (s)</label>\n <input type=\"number\" class=\"form-control form-control-sm\"\n data-field=\"broadcast-timeout\" value=\"2.0\" min=\"0.5\" step=\"0.5\" />\n </div>\n <div class=\"col-md-5\">\n <button class=\"btn btn-primary btn-sm w-100\" data-action=\"broadcast\">\n <i class=\"bi bi-megaphone me-1\"></i>Broadcast to All Runners\n </button>\n </div>\n </div>\n </div>\n </div>\n `;\n }\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// RunnerDetailsView — shell (default export)\n// ─────────────────────────────────────────────────────────────────────────────\n\nexport default class RunnerDetailsView extends View {\n constructor(options = {}) {\n super({ className: 'runner-details-view', ...options });\n this.model = options.model instanceof JobRunner\n ? options.model\n : new JobRunner(options.model || options.data || {});\n }\n\n async onInit() {\n if (!this.model) return;\n\n const tabView = new TabView({\n containerId: 'runner-tabs',\n tabs: {\n 'Overview': new RunnerOverviewTab({ model: this.model }),\n 'System Info': new RunnerSysinfoTab({ model: this.model }),\n 'Running Jobs': new RunnerJobsTab({ model: this.model }),\n 'Logs': new RunnerLogsTab({ model: this.model }),\n 'Actions': new RunnerActionsTab({ model: this.model })\n }\n });\n\n this.addChild(tabView);\n }\n\n async getTemplate() {\n return `<div data-container=\"runner-tabs\"></div>`;\n }\n\n /**\n * Open this view in a Dialog.\n *\n * @param {object} runner — runner object from GET /api/jobs/runners\n * @param {object} options — extra options forwarded to Dialog.showDialog()\n * @returns {Promise<any>}\n */\n static async show(runner, options = {}) {\n const model = runner instanceof JobRunner ? runner : new JobRunner(runner);\n const view = new RunnerDetailsView({ model });\n\n return await Dialog.showDialog({\n title: `<i class=\"bi bi-cpu me-2\"></i><span class=\"font-monospace\">${model.get('runner_id')}</span>`,\n body: view,\n size: 'xl',\n scrollable: true,\n buttons: [\n { text: 'Close', class: 'btn-secondary', dismiss: true }\n ],\n ...options\n });\n }\n}\n\n\nJobRunner.VIEW_CLASS = RunnerDetailsView;\n","/**\n * TaskManagementPage - Async task monitoring and management dashboard\n */\n\nimport Page from '@core/Page.js';\nimport View from '@core/View.js';\nimport Collection from '@core/Collection.js';\nimport TabView from '@core/views/navigation/TabView.js';\nimport TableView from '@core/views/table/TableView.js';\nimport { MetricsChart } from '@ext/charts/index.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\nimport TaskDetailsView from './TaskDetailsView.js';\nimport RunnerDetailsView from './RunnerDetailsView.js';\n\n// Task Stats Header View\nclass TaskStatsView extends View {\n constructor(options = {}) {\n super({\n ...options,\n className: 'mojo-task-stats-section'\n });\n\n this.stats = {\n pending: 0,\n running: 0,\n completed: 0,\n errors: 0\n };\n }\n\n async getTemplate() {\n return `\n <div class=\"mojo-task-stats-header mb-4\">\n <div class=\"row\">\n <div class=\"col-xl-3 col-lg-6 col-12 mb-3\">\n <div class=\"card h-100 border-0 shadow-sm\">\n <div class=\"card-body\">\n <div class=\"d-flex justify-content-between align-items-start\">\n <div>\n <h6 class=\"card-title text-muted mb-2\">Pending Tasks</h6>\n <h3 class=\"mb-1 fw-bold\">{{stats.pending}}</h3>\n <span class=\"badge bg-primary-subtle text-primary\">\n <i class=\"bi bi-clock\"></i> Queued\n </span>\n </div>\n <div class=\"text-primary\">\n <i class=\"bi bi-hourglass fs-2\"></i>\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <div class=\"col-xl-3 col-lg-6 col-12 mb-3\">\n <div class=\"card h-100 border-0 shadow-sm\">\n <div class=\"card-body\">\n <div class=\"d-flex justify-content-between align-items-start\">\n <div>\n <h6 class=\"card-title text-muted mb-2\">Running Tasks</h6>\n <h3 class=\"mb-1 fw-bold\">{{stats.running}}</h3>\n <span class=\"badge bg-success-subtle text-success\">\n <i class=\"bi bi-play-circle\"></i> Active\n </span>\n </div>\n <div class=\"text-success\">\n <i class=\"bi bi-arrow-repeat fs-2\"></i>\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <div class=\"col-xl-3 col-lg-6 col-12 mb-3\">\n <div class=\"card h-100 border-0 shadow-sm\">\n <div class=\"card-body\">\n <div class=\"d-flex justify-content-between align-items-start\">\n <div>\n <h6 class=\"card-title text-muted mb-2\">Completed Tasks</h6>\n <h3 class=\"mb-1 fw-bold\">{{stats.completed}}</h3>\n <span class=\"badge bg-info-subtle text-info\">\n <i class=\"bi bi-check-circle\"></i> Done\n </span>\n </div>\n <div class=\"text-info\">\n <i class=\"bi bi-check-square fs-2\"></i>\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <div class=\"col-xl-3 col-lg-6 col-12 mb-3\">\n <div class=\"card h-100 border-0 shadow-sm\">\n <div class=\"card-body\">\n <div class=\"d-flex justify-content-between align-items-start\">\n <div>\n <h6 class=\"card-title text-muted mb-2\">Failed Tasks</h6>\n <h3 class=\"mb-1 fw-bold\">{{stats.errors}}</h3>\n <span class=\"badge bg-danger-subtle text-danger\">\n <i class=\"bi bi-exclamation-circle\"></i> Errors\n </span>\n </div>\n <div class=\"text-danger\">\n <i class=\"bi bi-x-octagon fs-2\"></i>\n </div>\n </div>\n </div>\n </div>\n </div>\n </div>\n </div>\n `;\n }\n\n async loadStats() {\n try {\n const response = await this.getApp().rest.GET('/api/tasks/status');\n if (response.success && response.data.status) {\n this.stats = response.data.data;\n }\n } catch (error) {\n console.error('Failed to load task stats:', error);\n }\n }\n\n async onInit() {\n await this.loadStats();\n }\n}\n\n// Task Runners Status View\nclass TaskRunnersView extends View {\n constructor(options = {}) {\n super({\n ...options,\n className: 'mojo-task-runners-section'\n });\n\n this.runners = [];\n }\n\n async getTemplate() {\n return `\n <div class=\"card border shadow-sm mb-4\">\n <div class=\"card-header d-flex justify-content-between align-items-center\">\n <h5 class=\"card-title mb-0\">\n <i class=\"bi bi-cpu me-2\"></i>Task Runners\n </h5>\n <button class=\"btn btn-sm btn-outline-primary\" data-action=\"refresh-runners\">\n <i class=\"bi bi-arrow-clockwise\"></i> Refresh\n </button>\n </div>\n <div class=\"card-body\">\n {{#runners.length}}\n <div class=\"mojo-task-runner-list\">\n {{#runners}}\n <div class=\"mojo-task-runner-item p-3 mb-2 bg-light rounded\">\n <div class=\"row align-items-center\">\n <div class=\"col-md-8 col-lg-9\">\n <div class=\"d-flex align-items-center\">\n <div class=\"mojo-task-runner-status me-3\">\n <span class=\"badge {{statusBadge}}\">\n <i class=\"bi {{statusIcon}}\"></i> {{status}}\n </span>\n </div>\n <div class=\"mojo-task-runner-info\">\n <div class=\"mojo-task-runner-name\">\n <strong>{{hostname}}</strong>\n {{#max_workers}}<span class=\"text-muted ms-2\">• {{max_workers}} workers</span>{{/max_workers}}\n </div>\n <div class=\"mojo-task-runner-channels\">\n <small class=\"text-muted\">\n {{#channels.length}}Channels: {{#channels}}{{.}}{{^last}}, {{/last}}{{/channels}}{{/channels.length}}\n {{^channels.length}}No channels assigned{{/channels.length}}\n </small>\n </div>\n </div>\n </div>\n </div>\n <div class=\"col-md-4 col-lg-3\">\n <div class=\"mojo-task-runner-actions d-flex justify-content-end align-items-center\">\n <small class=\"text-muted me-2 d-none d-sm-inline\">{{pingAgeText}}</small>\n <div class=\"dropdown\">\n <button class=\"btn btn-sm btn-outline-secondary dropdown-toggle\"\n data-bs-toggle=\"dropdown\" aria-expanded=\"false\">\n <i class=\"bi bi-three-dots-vertical\"></i>\n </button>\n <ul class=\"dropdown-menu dropdown-menu-end\">\n <li><button class=\"dropdown-item\" data-action=\"view-runner-details\" data-runner-id=\"{{id}}\">\n <i class=\"bi bi-info-circle me-2\"></i>View Details\n </button></li>\n {{#isActive}}\n <li><button class=\"dropdown-item text-warning\" data-action=\"pause-runner\" data-runner-id=\"{{id}}\">\n <i class=\"bi bi-pause me-2\"></i>Pause Runner\n </button></li>\n {{/isActive}}\n {{^isActive}}\n <li><button class=\"dropdown-item text-success\" data-action=\"restart-runner\" data-runner-id=\"{{id}}\">\n <i class=\"bi bi-play me-2\"></i>Restart Runner\n </button></li>\n {{/isActive}}\n <li><hr class=\"dropdown-divider\"></li>\n <li><button class=\"dropdown-item text-danger\" data-action=\"remove-runner\" data-runner-id=\"{{id}}\">\n <i class=\"bi bi-trash me-2\"></i>Remove Runner\n </button></li>\n </ul>\n </div>\n </div>\n </div>\n </div>\n <div class=\"row mt-2 d-sm-none\">\n <div class=\"col-12\">\n <small class=\"text-muted\">Last ping: {{pingAgeText}}</small>\n </div>\n </div>\n </div>\n {{/runners}}\n </div>\n {{/runners.length}}\n {{^runners.length}}\n <div class=\"text-center text-muted py-4\">\n <i class=\"bi bi-cpu fs-1\"></i>\n <p class=\"mt-2\">No task runners found</p>\n </div>\n {{/runners.length}}\n </div>\n </div>\n `;\n }\n\n async loadRunners() {\n try {\n const response = await this.getApp().rest.GET('/api/tasks/runners');\n if (response.success && response.data.status) {\n this.runners = response.data.data.map(runner => {\n const isActive = runner.status === 'active';\n const pingAge = runner.ping_age || 0;\n\n return {\n ...runner,\n isActive,\n statusBadge: isActive ? 'bg-success' : 'bg-warning',\n statusIcon: isActive ? 'bi-check-circle-fill' : 'bi-exclamation-triangle-fill',\n pingAgeText: this.formatPingAge(pingAge)\n };\n });\n }\n } catch (error) {\n console.error('Failed to load runners:', error);\n }\n }\n\n formatPingAge(seconds) {\n if (seconds < 60) return `${Math.round(seconds)}s ago`;\n if (seconds < 3600) return `${Math.round(seconds / 60)}m ago`;\n return `${Math.round(seconds / 3600)}h ago`;\n }\n\n async onInit() {\n await this.loadRunners();\n }\n\n async onActionRefreshRunners(event, element) {\n await this.loadRunners();\n }\n\n async onActionViewRunnerDetails(event, element) {\n const runnerId = element.getAttribute('data-runner-id');\n const runner = this.runners.find(r => r.id === runnerId);\n\n if (runner) {\n const result = await RunnerDetailsView.show(runner);\n\n // Refresh runners list if any action was taken\n if (result?.action) {\n await this.loadRunners();\n this.emit('runner:' + result.action, result);\n }\n }\n }\n\n async onActionPauseRunner(event, element) {\n const runnerId = element.getAttribute('data-runner-id');\n const runner = this.runners.find(r => r.id === runnerId);\n\n if (!runner || !confirm(`Are you sure you want to pause runner \"${runner.hostname}\"?`)) {\n return;\n }\n\n try {\n element.disabled = true;\n const response = await this.getApp().rest.POST(`/api/runners/${runnerId}/pause`);\n\n if (response.success && response.data.status) {\n this.showSuccess('Runner paused successfully');\n await this.loadRunners();\n } else {\n this.showError(response.data.error || 'Failed to pause runner');\n }\n } catch (error) {\n console.error('Failed to pause runner:', error);\n this.showError('Failed to pause runner: ' + error.message);\n } finally {\n element.disabled = false;\n }\n }\n\n async onActionRestartRunner(event, element) {\n const runnerId = element.getAttribute('data-runner-id');\n const runner = this.runners.find(r => r.id === runnerId);\n\n if (!runner || !confirm(`Are you sure you want to restart runner \"${runner.hostname}\"?`)) {\n return;\n }\n\n try {\n element.disabled = true;\n const response = await this.getApp().rest.POST(`/api/runners/${runnerId}/restart`);\n\n if (response.success && response.data.status) {\n this.showSuccess('Runner restart initiated');\n await this.loadRunners();\n } else {\n this.showError(response.data.error || 'Failed to restart runner');\n }\n } catch (error) {\n console.error('Failed to restart runner:', error);\n this.showError('Failed to restart runner: ' + error.message);\n } finally {\n element.disabled = false;\n }\n }\n\n async onActionRemoveRunner(event, element) {\n const runnerId = element.getAttribute('data-runner-id');\n const runner = this.runners.find(r => r.id === runnerId);\n\n if (!runner) return;\n\n const confirmMessage = `Are you sure you want to remove runner \"${runner.hostname}\"? This action cannot be undone.`;\n if (!confirm(confirmMessage)) {\n return;\n }\n\n try {\n element.disabled = true;\n const response = await this.getApp().rest.DELETE(`/api/runners/${runnerId}`);\n\n if (response.success && response.data.status) {\n this.showSuccess('Runner removed successfully');\n await this.loadRunners();\n } else {\n this.showError(response.data.error || 'Failed to remove runner');\n }\n } catch (error) {\n console.error('Failed to remove runner:', error);\n this.showError('Failed to remove runner: ' + error.message);\n } finally {\n element.disabled = false;\n }\n }\n}\n\n// Task Charts View\nclass TaskChartsView extends View {\n constructor(options = {}) {\n super({\n ...options,\n className: 'mojo-task-charts-section'\n });\n }\n\n async getTemplate() {\n return `\n <div class=\"row mb-4\">\n <div class=\"col-xl-6 col-lg-12 mb-4\">\n <div class=\"card border shadow-sm\">\n <div class=\"card-body\" style=\"min-height: 300px;\">\n <div data-container=\"task-flow-chart\"></div>\n </div>\n </div>\n </div>\n <div class=\"col-xl-6 col-lg-12 mb-4\">\n <div class=\"card border shadow-sm\">\n <div class=\"card-body\" style=\"min-height: 300px;\">\n <div data-container=\"task-errors-chart\"></div>\n </div>\n </div>\n </div>\n </div>\n `;\n }\n\n async onInit() {\n // Task Flow Chart (Published vs Completed)\n this.taskFlowChart = new MetricsChart({\n title: '<i class=\"bi bi-graph-up me-2\"></i>Task Flow',\n endpoint: '/api/metrics/fetch',\n height: 280,\n granularity: 'hours',\n slugs: ['tasks_pub', 'tasks_completed'],\n account: 'global',\n chartType: 'line',\n showDateRange: false,\n colors: [\n 'rgba(13, 110, 253, 0.8)', // Primary blue for published\n 'rgba(25, 135, 84, 0.8)' // Success green for completed\n ],\n yAxis: {\n label: 'Count',\n beginAtZero: true\n },\n tooltip: {\n y: 'number'\n },\n containerId: 'task-flow-chart'\n });\n this.addChild(this.taskFlowChart);\n\n // Task Errors Chart\n this.taskErrorsChart = new MetricsChart({\n title: '<i class=\"bi bi-exclamation-triangle me-2\"></i>Task Issues',\n endpoint: '/api/metrics/fetch',\n height: 280,\n granularity: 'hours',\n slugs: ['tasks_errors', 'tasks_expired'],\n account: 'global',\n chartType: 'line',\n showDateRange: false,\n colors: [\n 'rgba(220, 53, 69, 0.8)', // Danger red for errors\n 'rgba(255, 193, 7, 0.8)' // Warning yellow for expired\n ],\n yAxis: {\n label: 'Count',\n beginAtZero: true\n },\n tooltip: {\n y: 'number'\n },\n containerId: 'task-errors-chart'\n });\n this.addChild(this.taskErrorsChart);\n }\n}\n\n// Task Table Views\nclass BaseTaskTable extends TableView {\n constructor(options = {}) {\n super({\n showPagination: true,\n showSearch: true,\n showRefresh: true,\n pageSize: 10,\n columns: [\n { key: 'id', label: 'Task ID', sortable: true },\n { key: 'function', label: 'Function', sortable: true },\n { key: 'channel', label: 'Channel', sortable: true },\n { key: 'created', label: 'Created', sortable: true, formatter: 'datetime' },\n { key: 'status', label: 'Status', formatter: 'badge' }\n ],\n ...options\n });\n }\n\n buildActionCell(item) {\n const actions = [];\n\n if (item.status === 'pending') {\n actions.push(`\n <button class=\"btn btn-sm btn-outline-danger\" data-action=\"cancel-task\" data-task-id=\"${item.id}\" title=\"Cancel Task\">\n <i class=\"bi bi-x-circle\"></i>\n </button>\n `);\n }\n\n if (item.status === 'error') {\n actions.push(`\n <button class=\"btn btn-sm btn-outline-primary\" data-action=\"retry-task\" data-task-id=\"${item.id}\" title=\"Retry Task\">\n <i class=\"bi bi-arrow-clockwise\"></i>\n </button>\n `);\n }\n\n actions.push(`\n <button class=\"btn btn-sm btn-outline-info\" data-action=\"view-task-details\" data-task-id=\"${item.id}\" title=\"View Details\">\n <i class=\"bi bi-info-circle\"></i>\n </button>\n `);\n\n return actions.join(' ');\n }\n\n async onActionCancelTask(action, event, element) {\n const taskId = element.getAttribute('data-task-id');\n\n if (!confirm('Are you sure you want to cancel this task?')) {\n return;\n }\n\n try {\n element.disabled = true;\n const response = await this.getApp().rest.POST(`/api/tasks/${taskId}/cancel`);\n\n if (response.success && response.data.status) {\n this.showSuccess('Task cancelled successfully');\n // Refresh the active table\n const activeTab = this.getParentPage()?.taskTablesView?.getActiveTab();\n if (activeTab) {\n const activeTable = this.getParentPage()?.taskTablesView?.getTab(activeTab);\n if (activeTable?.refresh) {\n await activeTable.refresh();\n }\n }\n } else {\n this.showError(response.data.error || 'Failed to cancel task');\n }\n } catch (error) {\n console.error('Failed to cancel task:', error);\n this.showError('Failed to cancel task: ' + error.message);\n } finally {\n element.disabled = false;\n }\n }\n\n async onActionRetryTask(action, event, element) {\n const taskId = element.getAttribute('data-task-id');\n\n try {\n element.disabled = true;\n const response = await this.getApp().rest.POST(`/api/tasks/${taskId}/retry`);\n\n if (response.success && response.data.status) {\n this.showSuccess('Task queued for retry');\n // Refresh the active table\n const activeTab = this.getParentPage()?.taskTablesView?.getActiveTab();\n if (activeTab) {\n const activeTable = this.getParentPage()?.taskTablesView?.getTab(activeTab);\n if (activeTable?.refresh) {\n await activeTable.refresh();\n }\n }\n } else {\n this.showError(response.data.error || 'Failed to retry task');\n }\n } catch (error) {\n console.error('Failed to retry task:', error);\n this.showError('Failed to retry task: ' + error.message);\n } finally {\n element.disabled = false;\n }\n }\n\n async onActionViewTaskDetails(action, event, element) {\n const taskId = element.getAttribute('data-task-id');\n\n try {\n // Fetch detailed task data\n const response = await this.getApp().rest.GET(`/api/tasks/${taskId}/details`);\n\n if (response.success && response.data.status) {\n const task = response.data.data;\n const result = await TaskDetailsView.show(task);\n\n // Refresh tables if any action was taken\n if (result?.action) {\n await this.refreshActiveTables();\n this.emit('task:' + result.action, result);\n }\n } else {\n this.showError(response.data.error || 'Failed to load task details');\n }\n } catch (error) {\n console.error('Failed to load task details:', error);\n this.showError('Failed to load task details: ' + error.message);\n }\n }\n\n getParentPage() {\n // Helper to get the parent TaskManagementPage\n let parent = this.parent;\n while (parent && parent.constructor.name !== 'TaskManagementPage') {\n parent = parent.parent;\n }\n return parent;\n }\n\n async refreshActiveTables() {\n const parentPage = this.getParentPage();\n if (parentPage?.taskTablesView) {\n const activeTab = parentPage.taskTablesView.getActiveTab();\n if (activeTab) {\n const activeTable = parentPage.taskTablesView.getTab(activeTab);\n if (activeTable?.refresh) {\n await activeTable.refresh();\n }\n }\n }\n }\n}\n\nclass PendingTasksTable extends TableView {\n constructor(options = {}) {\n super({\n ...options,\n title: 'Pending Tasks',\n collection: new Collection({\n endpoint: '/api/tasks/pending'\n }),\n columns: [\n { key: 'id', label: 'Task ID', sortable: true },\n { key: 'function', label: 'Function', sortable: true },\n { key: 'channel', label: 'Channel', sortable: true },\n { key: 'created', label: 'Created', sortable: true, formatter: 'datetime' },\n { key: 'status', label: 'Status', formatter: 'badge' }\n ],\n });\n }\n}\n\nclass RunningTasksTable extends TableView {\n constructor(options = {}) {\n super({\n ...options,\n title: 'Running Tasks',\n collection: new Collection({\n endpoint: '/api/tasks/running'\n }),\n columns: [\n { key: 'id', label: 'Task ID', sortable: true },\n { key: 'function', label: 'Function', sortable: true },\n { key: 'channel', label: 'Channel', sortable: true },\n { key: 'created', label: 'Created', sortable: true, formatter: 'datetime' },\n { key: 'status', label: 'Status', formatter: 'badge' }\n ],\n });\n }\n}\n\nclass CompletedTasksTable extends TableView {\n constructor(options = {}) {\n super({\n ...options,\n title: 'Completed Tasks',\n collection: new Collection({\n endpoint: '/api/tasks/completed'\n }),\n columns: [\n { key: 'id', label: 'Task ID', sortable: true },\n { key: 'function', label: 'Function', sortable: true },\n { key: 'channel', label: 'Channel', sortable: true },\n { key: 'created', label: 'Created', sortable: true, formatter: 'datetime' },\n { key: 'completed_at', label: 'Completed', sortable: true, formatter: 'datetime' },\n { key: 'status', label: 'Status', formatter: 'badge' }\n ]\n });\n }\n}\n\nclass ErrorTasksTable extends TableView {\n constructor(options = {}) {\n super({\n ...options,\n title: 'Failed Tasks',\n collection: new Collection({\n endpoint: '/api/tasks/errors'\n }),\n columns: [\n { key: 'id', label: 'Task ID', sortable: true },\n { key: 'function', label: 'Function', sortable: true },\n { key: 'channel', label: 'Channel', sortable: true },\n { key: 'created', label: 'Created', sortable: true, formatter: 'datetime' },\n { key: 'error', label: 'Error', sortable: false },\n { key: 'status', label: 'Status', formatter: 'badge' }\n ]\n });\n }\n}\n\n// Main Task Management Page\nexport default class TaskManagementPage extends Page {\n constructor(options = {}) {\n super({\n ...options,\n title: 'Task Management',\n className: 'mojo-task-management-page'\n });\n\n this.pageTitle = 'Task Management';\n this.pageSubtitle = 'Async task monitoring and runner management';\n this.lastUpdated = new Date().toLocaleString();\n }\n\n async getTemplate() {\n return `\n <div class=\"mojo-task-management-container\">\n <!-- Page Header -->\n <div class=\"d-flex justify-content-between align-items-center mb-4\">\n <div>\n <h1 class=\"h3 mb-1\">{{pageTitle}}</h1>\n <p class=\"text-muted mb-0\">{{pageSubtitle}}</p>\n <small class=\"text-info\">\n <i class=\"bi bi-cpu me-1\"></i>\n Real-time task processing and runner monitoring\n </small>\n </div>\n <div class=\"btn-group\" role=\"group\">\n <button type=\"button\" class=\"btn btn-outline-secondary btn-sm\"\n data-action=\"refresh-all\" title=\"Refresh All Data\">\n <i class=\"bi bi-arrow-clockwise\"></i> Refresh\n </button>\n <button type=\"button\" class=\"btn btn-outline-primary btn-sm\"\n data-action=\"export-tasks\" title=\"Export Task Data\">\n <i class=\"bi bi-download\"></i> Export\n </button>\n <button type=\"button\" class=\"btn btn-outline-success btn-sm\"\n data-action=\"manage-channels\" title=\"Manage Channels\">\n <i class=\"bi bi-collection\"></i> Channels\n </button>\n </div>\n </div>\n\n <!-- Task Stats -->\n <div data-container=\"task-stats\"></div>\n\n <!-- Task Runners -->\n <div data-container=\"task-runners\"></div>\n\n <!-- Task Charts -->\n <div data-container=\"task-charts\"></div>\n\n <!-- Task Tables -->\n <div class=\"card border shadow-sm\">\n <div class=\"card-header\">\n <h5 class=\"card-title mb-0\">\n <i class=\"bi bi-list-task me-2\"></i>Task Management\n </h5>\n </div>\n <div class=\"card-body\">\n <div data-container=\"task-tables\"></div>\n </div>\n </div>\n\n <!-- System Status Footer -->\n <div class=\"row mt-4\">\n <div class=\"col-12\">\n <div class=\"alert alert-info border-0\" role=\"alert\">\n <div class=\"d-flex align-items-center\">\n <i class=\"bi bi-info-circle-fill me-2\"></i>\n <div>\n <strong>Task System Status:</strong> Monitoring active.\n Last updated: <span class=\"text-muted\">{{lastUpdated}}</span>\n </div>\n <div class=\"ms-auto\">\n <button class=\"btn btn-sm btn-outline-info\" data-action=\"view-system-logs\">\n <i class=\"bi bi-journal-text\"></i> View Logs\n </button>\n </div>\n </div>\n </div>\n </div>\n </div>\n </div>\n `;\n }\n\n async onInit() {\n // Create Task Stats View\n this.taskStatsView = new TaskStatsView({\n containerId: 'task-stats'\n });\n this.addChild(this.taskStatsView);\n\n // Create Task Runners View\n this.taskRunnersView = new TaskRunnersView({\n containerId: 'task-runners'\n });\n this.addChild(this.taskRunnersView);\n\n // Create Task Charts View\n this.taskChartsView = new TaskChartsView({\n containerId: 'task-charts'\n });\n this.addChild(this.taskChartsView);\n\n // Create Task Tables with TabView\n this.taskTablesView = new TabView({\n containerId: 'task-tables',\n tabs: {\n 'Pending': new PendingTasksTable(),\n 'Running': new RunningTasksTable(),\n 'Completed': new CompletedTasksTable(),\n 'Errors': new ErrorTasksTable()\n },\n activeTab: 'Pending'\n });\n this.addChild(this.taskTablesView);\n }\n\n async onActionRefreshAll(action, event, element) {\n try {\n // Show loading state\n const icon = element.querySelector('i');\n icon?.classList.add('bi-spin');\n element.disabled = true;\n\n // Refresh all components\n const promises = [\n this.taskStatsView?.loadStats(),\n this.taskRunnersView?.loadRunners(),\n this.taskChartsView?.taskFlowChart?.refresh(),\n this.taskChartsView?.taskErrorsChart?.refresh()\n ].filter(Boolean);\n\n // Refresh active table\n const activeTab = this.taskTablesView?.getActiveTab();\n if (activeTab) {\n const activeTable = this.taskTablesView.getTab(activeTab);\n if (activeTable?.refresh) {\n promises.push(activeTable.refresh());\n }\n }\n\n await Promise.allSettled(promises);\n\n // Update timestamp\n this.lastUpdated = new Date().toLocaleString();\n\n // Emit refresh event\n const eventBus = this.getApp()?.events;\n if (eventBus) {\n eventBus.emit('tasks:dashboard-refreshed', {\n page: this,\n timestamp: this.lastUpdated\n });\n }\n\n } catch (error) {\n console.error('Failed to refresh task dashboard:', error);\n } finally {\n // Reset button state\n const icon = element.querySelector('i');\n icon?.classList.remove('bi-spin');\n element.disabled = false;\n }\n }\n\n async onActionExportTasks(action, event, element) {\n try {\n // Export charts\n await this.taskChartsView?.taskFlowChart?.export('png');\n await this.taskChartsView?.taskErrorsChart?.export('png');\n\n // Export active table\n const activeTab = this.taskTablesView?.getActiveTab();\n if (activeTab) {\n const activeTable = this.taskTablesView.getTab(activeTab);\n if (activeTable?.exportToCSV) {\n activeTable.exportToCSV();\n }\n }\n\n } catch (error) {\n console.error('Failed to export task data:', error);\n }\n }\n\n async onActionManageChannels(event, element) {\n try {\n // For now, show a simple info modal with channel information\n const response = await this.getApp().rest.GET('/api/tasks/channels');\n\n if (response.success && response.data.status) {\n const channels = response.data.data;\n\n // Create a simple info alert\n const channelList = channels.map(channel =>\n `${channel.name} (${channel.pending} pending, ${channel.running} running)`\n ).join('\\n');\n\n alert(`Task Channels:\\n\\n${channelList}\\n\\nFull channel management interface coming soon!`);\n } else {\n this.showError('Failed to load channel information');\n }\n } catch (error) {\n console.error('Failed to load channels:', error);\n // Navigate to dedicated channels page as fallback\n const router = this.getApp()?.router;\n if (router) {\n router.navigateTo('/admin/task-channels');\n }\n }\n }\n\n async onActionViewSystemLogs(event, element) {\n try {\n // Try to get recent system logs related to tasks\n const response = await this.getApp().rest.GET('/api/tasks/logs?limit=50');\n\n if (response.success && response.data.status) {\n const logs = response.data.data;\n\n // Create a simple log preview\n const logPreview = logs.slice(0, 10).map(log =>\n `[${new Date(log.timestamp * 1000).toLocaleString()}] ${log.level.toUpperCase()}: ${log.message}`\n ).join('\\n');\n\n const viewFullLogs = confirm(`Recent Task System Logs:\\n\\n${logPreview}\\n\\n... and ${logs.length - 10} more entries.\\n\\nView full logs?`);\n\n if (viewFullLogs) {\n const router = this.getApp()?.router;\n if (router) {\n router.navigateTo('/admin/logs?filter=tasks');\n }\n }\n } else {\n // Navigate to system logs as fallback\n const router = this.getApp()?.router;\n if (router) {\n router.navigateTo('/admin/logs');\n }\n }\n } catch (error) {\n console.error('Failed to load system logs:', error);\n // Navigate to system logs as fallback\n const router = this.getApp()?.router;\n if (router) {\n router.navigateTo('/admin/logs');\n }\n }\n }\n\n // Public API\n async refreshDashboard() {\n return this.onActionRefreshAll(null, null, { disabled: false, querySelector: () => null });\n }\n\n getStats() {\n return this.taskStatsView?.stats || {};\n }\n\n getRunners() {\n return this.taskRunnersView?.runners || [];\n }\n\n getCharts() {\n return {\n taskFlow: this.taskChartsView?.taskFlowChart,\n taskErrors: this.taskChartsView?.taskErrorsChart\n };\n }\n\n\n}\n","/**\n * LogView - Detailed view for a Log record\n */\n\nimport View from '@core/View.js';\nimport TabView from '@core/views/navigation/TabView.js';\nimport DataView from '@core/views/data/DataView.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport { Log } from '@core/models/Log.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\nimport DeviceView from '../account/devices/DeviceView.js';\nimport GeoIPView from '../account/devices/GeoIPView.js';\n\nclass LogView extends View {\n constructor(options = {}) {\n super({\n className: 'log-view',\n ...options\n });\n\n this.model = options.model || new Log(options.data || {});\n this.logIcon = this.getIconForLog(this.model.get('level'));\n\n this.template = `\n <div class=\"log-view-container\">\n <!-- Header -->\n <div class=\"d-flex justify-content-between align-items-center mb-4\">\n <div class=\"d-flex align-items-center gap-3\">\n <div class=\"fs-1 {{logIcon.color}}\">\n <i class=\"bi {{logIcon.icon}}\"></i>\n </div>\n <div>\n <h4 class=\"mb-1\">\n <span class=\"badge bg-secondary\">{{model.method}}</span> {{model.path}}\n </h4>\n <div class=\"text-muted small\">\n {{model.created|datetime}}\n </div>\n </div>\n </div>\n <div data-container=\"log-context-menu\"></div>\n </div>\n\n <!-- Tabs -->\n <div data-container=\"log-tabs\"></div>\n </div>\n `;\n }\n\n getIconForLog(level) {\n const lvl = level?.toLowerCase();\n if (lvl === 'error' || lvl === 'critical') return { icon: 'bi-x-octagon-fill', color: 'text-danger' };\n if (lvl === 'warning') return { icon: 'bi-exclamation-triangle-fill', color: 'text-warning' };\n if (lvl === 'info') return { icon: 'bi-info-circle-fill', color: 'text-info' };\n return { icon: 'bi-journal-text', color: 'text-secondary' };\n }\n\n async onInit() {\n // Overview Tab\n this.overviewView = new DataView({\n model: this.model,\n className: \"p-3\",\n columns: 2,\n fields: [\n { name: 'id', label: 'Log ID' },\n { name: 'level', label: 'Level', format: 'badge' },\n { name: 'kind', label: 'Kind' },\n { name: 'ip', label: 'IP Address', template: '<a href=\"#\" data-action=\"view-ip\">{{model.ip}}</a>' },\n { name: 'uid', label: 'User ID' },\n { name: 'username', label: 'Username' },\n { name: 'duid', label: 'Device ID', template: '<a href=\"#\" data-action=\"view-device\">{{model.duid|truncate_middle(32)}}</a>' },\n { name: 'model_name', label: 'Related Model' },\n { name: 'model_id', label: 'Related Model ID' },\n { name: 'user_agent', label: 'User Agent', columns: 12 },\n ]\n });\n\n // Log Content Tab\n const logContent = this.model.get('log');\n let formattedLog = logContent;\n try {\n // Attempt to parse and prettify if it's a JSON string\n const parsed = JSON.parse(logContent);\n formattedLog = JSON.stringify(parsed, null, 2);\n } catch (e) {\n // Not a valid JSON string, display as is\n }\n\n this.logContentView = new View({\n template: `\n <div class=\"position-relative\">\n <button class=\"btn btn-sm btn-outline-secondary position-absolute top-0 end-0 mt-2 me-2\" data-action=\"copy-log\">\n <i class=\"bi bi-clipboard\"></i> Copy\n </button>\n <pre class=\"bg-light p-3 border rounded\" style=\"max-height: 600px; overflow-y: auto;\"><code>${formattedLog}</code></pre>\n </div>\n `,\n onActionCopyLog: () => {\n navigator.clipboard.writeText(formattedLog);\n this.getApp()?.toast?.success('Log content copied to clipboard.');\n }\n });\n\n // TabView\n this.tabView = new TabView({\n containerId: 'log-tabs',\n tabs: {\n 'Log': this.logContentView,\n 'Details': this.overviewView,\n\n },\n activeTab: 'Log'\n });\n this.addChild(this.tabView);\n\n // ContextMenu\n const logMenu = new ContextMenu({\n containerId: 'log-context-menu',\n className: \"context-menu-view header-menu-absolute\",\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: [\n { label: 'View User', action: 'view-user', icon: 'bi-person', disabled: !this.model.get('uid') },\n { label: 'View Device', action: 'view-device', icon: 'bi-phone', disabled: !this.model.get('duid') },\n { type: 'divider' },\n { label: 'Delete Log', action: 'delete-log', icon: 'bi-trash', danger: true }\n ]\n }\n });\n this.addChild(logMenu);\n }\n\n async onActionViewIp(event) {\n event.preventDefault();\n const ip = this.model.get('ip');\n if (ip) {\n GeoIPView.show(ip);\n }\n }\n\n async onActionViewDevice(event) {\n event.preventDefault();\n const duid = this.model.get('duid');\n if (duid) {\n DeviceView.show(duid);\n }\n }\n\n async onActionViewUser() {\n console.log(\"TODO: View user\", this.model.get('uid'));\n }\n\n async onActionDeleteLog() {\n const confirmed = await Dialog.confirm(\n `Are you sure you want to delete this log entry? This action cannot be undone.`,\n 'Confirm Deletion',\n { confirmClass: 'btn-danger', confirmText: 'Delete' }\n );\n if (confirmed) {\n const resp = await this.model.destroy();\n if (resp.success) {\n this.emit('log:deleted', { model: this.model });\n }\n }\n }\n}\n\nLog.VIEW_CLASS = LogView;\n\nexport default LogView;\n","/**\n * LogTablePage - Log management using TablePage component\n * Clean implementation using TablePage with minimal overrides\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { LogList } from '@core/models/Log.js';\nimport LogView from './LogView.js';\n\nclass LogTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_logs',\n pageName: 'Manage Logs',\n router: \"admin/logs\",\n Collection: LogList,\n\n itemViewClass: LogView,\n viewDialogOptions: {\n header: false,\n size: 'xl'\n },\n\n // Column definitions\n columns: [\n {\n key: 'created|epoch|datetime',\n label: 'Timestamp',\n sortable: true,\n filter: {\n type: 'daterange',\n }\n },\n {\n key: 'level',\n label: 'Level',\n sortable: true,\n formatter: 'badge',\n filter: {\n type: 'select',\n options: [\n { value: 'info', label: 'Info' },\n { value: 'warning', label: 'Warning' },\n { value: 'error', label: 'Error' }\n ]\n }\n },\n {\n key: 'kind',\n label: 'Kind',\n filter: {\n type: \"text\"\n }\n },\n {\n key: 'method',\n label: 'Method',\n filter: {\n type: \"text\"\n }\n },\n {\n key: 'path',\n label: 'Path',\n filter: {\n type: \"text\"\n }\n },\n {\n key: 'username',\n label: 'User',\n filter: {\n type: \"text\"\n }\n },\n {\n key: 'ip',\n label: 'IP',\n filter: {\n type: \"text\"\n }\n },\n {\n key: 'duid',\n label: 'Browser ID',\n formatter: 'truncate_middle(16)',\n filter: {\n type: \"text\"\n }\n }\n ],\n\n // Default sort by timestamp descending (newest first)\n defaultQuery: {\n sort: '-created'\n },\n\n // Table features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Toolbar\n showRefresh: true,\n showAdd: false, // Logs are typically not manually created\n showExport: true,\n\n // Empty state\n emptyMessage: 'No log entries found.',\n\n // Batch actions\n batchBarLocation: 'top',\n batchActions: [\n { label: \"Export\", icon: \"bi bi-download\", action: \"batch-export\" },\n { label: \"Archive\", icon: \"bi bi-archive\", action: \"batch-archive\" },\n { label: \"Delete\", icon: \"bi bi-trash\", action: \"batch-delete\" },\n { label: \"Mark as Reviewed\", icon: \"bi bi-check2\", action: \"batch-reviewed\" }\n ],\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n }\n });\n }\n}\n\nexport default LogTablePage;\n","import View from '@core/View.js';\nimport DataView from '@core/views/data/DataView.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport { MetricsPermission, MetricsForms } from '@core/models/Metrics.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\n\nclass MetricsPermissionsView extends View {\n constructor(options = {}) {\n super({\n className: 'metrics-permissions-view',\n ...options\n });\n\n this.model = options.model || new MetricsPermission(options.data || {});\n\n this.template = `\n <div class=\"container p-3\">\n <div class=\"d-flex justify-content-between align-items-center mb-4\">\n <div>\n <h3 class=\"mb-1\">Permissions for {{model.account}}</h3>\n </div>\n <div data-container=\"context-menu\"></div>\n </div>\n <div data-container=\"data-view\"></div>\n </div>\n `;\n }\n\n async onInit() {\n this.dataView = new DataView({\n containerId: 'data-view',\n model: this.model,\n fields: [\n { name: 'view_permissions', label: 'View Permissions', format: 'list|badge' },\n { name: 'write_permissions', label: 'Write Permissions', format: 'list|badge' },\n ]\n });\n this.addChild(this.dataView);\n\n const contextMenu = new ContextMenu({\n containerId: 'context-menu',\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: [\n { label: 'Edit', action: 'edit', icon: 'bi-pencil' },\n { label: 'Delete', action: 'delete', icon: 'bi-trash', danger: true },\n ]\n }\n });\n this.addChild(contextMenu);\n }\n\n async onActionEdit() {\n const resp = await Dialog.showModelForm({\n title: `Edit Permissions for ${this.model.get('account')}`,\n model: this.model,\n formConfig: MetricsForms.edit\n });\n if (resp) {\n this.model.set(resp.data.data);\n this.render();\n }\n }\n\n async onActionDelete() {\n const confirmed = await Dialog.confirm(`Are you sure you want to delete all permissions for ${this.model.get('account')}?`);\n if (confirmed) {\n await this.model.destroy();\n this.emit('deleted', this.model);\n }\n }\n}\n\nMetricsPermission.VIEW_CLASS = MetricsPermissionsView;\n\nexport default MetricsPermissionsView;\n","import TablePage from '@core/pages/TablePage.js';\nimport { MetricsPermissionList, MetricsForms } from '@core/models/Metrics.js';\nimport MetricsPermissionsView from './MetricsPermissionsView.js';\n\nclass MetricsPermissionsTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_metrics_permissions',\n pageName: 'Metrics Permissions',\n router: \"admin/metrics/permissions\",\n Collection: MetricsPermissionList,\n formEdit: MetricsForms.edit,\n itemViewClass: MetricsPermissionsView,\n viewDialogOptions: {\n header: false,\n size: 'lg'\n },\n\n columns: [\n { key: 'account', label: 'Account', sortable: true },\n { key: 'view_permissions', label: 'View Permissions', formatter: 'list|badge' },\n { key: 'write_permissions', label: 'Write Permissions', formatter: 'list|badge' },\n ],\n\n selectable: true,\n searchable: true,\n sortable: true,\n paginated: true,\n showRefresh: true,\n showAdd: true,\n showExport: true,\n\n tableOptions: {\n pageSizes: [10, 25, 50],\n defaultPageSize: 25,\n emptyMessage: 'No metrics permissions found.',\n emptyIcon: 'bi-bar-chart-line',\n actions: [\"view\", \"edit\", \"delete\"],\n }\n });\n }\n}\n\nexport default MetricsPermissionsTablePage;\n","import Collection from '@core/Collection.js';\nimport Model from '@core/Model.js';\nimport { GroupList } from '@core/models/Group.js';\n\n/**\n * Setting - Global or group-scoped configuration key-value pair.\n * Maps to REST endpoints under /api/settings\n *\n * Key properties:\n * - Can be global or scoped to a group (via `group` field)\n * - Supports secret values (`is_secret: true`) — API masks display_value as \"******\"\n * - Secret values are write-only; the raw value is never returned after creation\n *\n * Endpoints:\n * GET /api/settings - List settings (requires manage_settings)\n * POST /api/settings - Create a setting\n * GET /api/settings/<id> - Get one setting\n * POST /api/settings/<id> - Update a setting\n * DELETE /api/settings/<id> - Delete a setting\n */\nclass Setting extends Model {\n constructor(data = {}, options = {}) {\n super(data, {\n endpoint: '/api/settings',\n ...options\n });\n }\n}\n\n/**\n * SettingList - Collection of Setting records.\n * Supports search and sort query params.\n */\nclass SettingList extends Collection {\n constructor(options = {}) {\n super({\n ModelClass: Setting,\n endpoint: '/api/settings',\n size: 25,\n ...options\n });\n }\n}\n\n/**\n * Forms configuration for Setting\n */\nconst SettingForms = {\n create: {\n title: 'Create Setting',\n fields: [\n {\n name: 'key',\n type: 'text',\n label: 'Key',\n placeholder: 'WEBHOOK_SECRET',\n required: true,\n columns: 12,\n help: 'A unique configuration key name.'\n },\n {\n name: 'value',\n type: 'textarea',\n label: 'Value',\n required: true,\n columns: 12,\n help: 'The configuration value. For secrets, this will be masked after creation.'\n },\n {\n type: 'collection',\n name: 'parent',\n label: 'Parent Group',\n Collection: GroupList, // Collection class\n labelField: 'name', // Field to display in dropdown\n valueField: 'id', // Field to use as value\n maxItems: 10, // Max items to show in dropdown\n placeholder: 'Search groups...',\n emptyFetch: false,\n debounceMs: 300, // Search debounce delay\n columns: 12\n },\n {\n name: 'is_secret',\n type: 'switch',\n label: 'Secret',\n columns: 6,\n help: 'Mark as secret to mask the value in API responses.'\n }\n ]\n },\n\n edit: {\n title: 'Edit Setting',\n fields: [\n {\n name: 'key',\n type: 'text',\n label: 'Key',\n columns: 12,\n disabled: true\n },\n {\n name: 'value',\n type: 'textarea',\n label: 'Value',\n columns: 12,\n help: 'Enter a new value to replace the current one.'\n },\n {\n name: 'is_secret',\n type: 'switch',\n label: 'Secret',\n columns: 12,\n help: 'Mark as secret to mask the value in API responses.'\n },\n {\n type: 'collection',\n name: 'parent',\n label: 'Parent Group',\n Collection: GroupList, // Collection class\n labelField: 'name', // Field to display in dropdown\n valueField: 'id', // Field to use as value\n maxItems: 10, // Max items to show in dropdown\n placeholder: 'Search groups...',\n emptyFetch: false,\n debounceMs: 300, // Search debounce delay\n columns: 12\n }\n ]\n }\n};\n\nexport { Setting, SettingList, SettingForms };\n","/**\n * SettingView - Secure setting detail and management interface\n *\n * Shows setting metadata (key, value, group scope, secret status)\n * and provides actions to edit and delete. Secret values display\n * the masked display_value from the API.\n */\n\nimport View from '@core/View.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport { Setting, SettingForms } from '@core/models/Settings.js';\n\nclass SettingView extends View {\n constructor(options = {}) {\n super({\n className: 'setting-view',\n ...options\n });\n\n this.model = options.model || new Setting(options.data || {});\n\n this.template = `\n <div class=\"setting-view-container\">\n <!-- Header -->\n <div class=\"d-flex justify-content-between align-items-start mb-4\">\n <!-- Left: Icon & Identity -->\n <div class=\"d-flex align-items-center gap-3\">\n <div class=\"fs-1 text-primary\">\n <i class=\"bi bi-gear\"></i>\n </div>\n <div>\n <h3 class=\"mb-1 font-monospace\">{{model.key|default('Unnamed Setting')}}</h3>\n <div class=\"text-muted small\">\n ID: {{model.id}}\n <span class=\"mx-2\">|</span>\n Scope: {{model.group.name|default('Global')}}\n </div>\n <div class=\"mt-1\">\n <span class=\"badge {{model.is_secret|boolean('bg-warning text-dark','bg-secondary')}}\">\n {{model.is_secret|boolean('Secret','Plain')}}\n </span>\n </div>\n </div>\n </div>\n\n <!-- Right: Meta & Actions -->\n <div class=\"d-flex align-items-start gap-4\">\n <div class=\"text-end\">\n <div class=\"text-muted small\">Created</div>\n <div>{{model.created|datetime}}</div>\n </div>\n <div data-container=\"setting-context-menu\"></div>\n </div>\n </div>\n\n <!-- Details -->\n <div class=\"list-group mb-3\">\n <div class=\"list-group-item\">\n <h6 class=\"mb-1 text-muted\">Key</h6>\n <p class=\"mb-0 font-monospace\">{{model.key}}</p>\n </div>\n <div class=\"list-group-item\">\n <h6 class=\"mb-1 text-muted\">Value</h6>\n {{#model.is_secret|bool}}\n <p class=\"mb-0 text-muted font-monospace\">{{model.display_value|default('******')}}</p>\n <small class=\"text-muted\">This is a secret value. Enter a new value to replace it.</small>\n {{/model.is_secret|bool}}\n {{^model.is_secret|bool}}\n <p class=\"mb-0 font-monospace\">{{model.value|default(model.display_value)|default('—')}}</p>\n {{/model.is_secret|bool}}\n </div>\n {{#model.group}}\n <div class=\"list-group-item\">\n <h6 class=\"mb-1 text-muted\">Group</h6>\n <p class=\"mb-0\">{{model.group.name|default(model.group)}}</p>\n </div>\n {{/model.group}}\n </div>\n </div>\n `;\n }\n\n async onInit() {\n const settingMenu = new ContextMenu({\n containerId: 'setting-context-menu',\n className: 'context-menu-view header-menu-absolute',\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: [\n { label: 'Edit', action: 'edit-setting', icon: 'bi-pencil' },\n { type: 'divider' },\n { label: 'Delete Setting', action: 'delete-setting', icon: 'bi-trash', danger: true }\n ]\n }\n });\n this.addChild(settingMenu);\n }\n\n async onActionEditSetting() {\n const app = this.getApp();\n const resp = await app.showModelForm({\n title: `Edit Setting — ${this.model.get('key')}`,\n model: this.model,\n formConfig: SettingForms.edit,\n });\n if (resp) {\n this.render();\n }\n }\n\n async onActionDeleteSetting() {\n const app = this.getApp();\n const confirmed = await app.confirm({\n title: 'Delete Setting',\n message: `Permanently delete \"${this.model.get('key')}\"? This cannot be undone.`,\n confirmLabel: 'Delete',\n confirmClass: 'btn-danger'\n });\n if (!confirmed) return;\n\n app.showLoading();\n const resp = await this.model.delete();\n app.hideLoading();\n if (resp && resp.success !== false) {\n app.toast.success('Setting deleted');\n this.emit('deleted', { model: this.model });\n } else {\n app.toast.error('Failed to delete setting');\n }\n }\n}\n\nSetting.VIEW_CLASS = SettingView;\nexport default SettingView;\n","/**\n * SettingTablePage - Secure settings management using TablePage component\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { Setting, SettingList, SettingForms } from '@core/models/Settings.js';\nimport SettingView from './SettingView.js';\n\n// Register the add/edit forms on the model class so TableView can find them automatically\nSetting.ADD_FORM = SettingForms.create;\nSetting.EDIT_FORM = SettingForms.edit;\n\nclass SettingTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_settings',\n pageName: 'Settings',\n router: 'admin/settings',\n Collection: SettingList,\n\n itemViewClass: SettingView,\n viewDialogOptions: {\n header: false,\n size: 'lg'\n },\n\n columns: [\n { key: 'id', label: 'ID', width: '70px', sortable: true, class: 'text-muted' },\n { key: 'key', label: 'Key', sortable: true },\n { key: 'display_value', label: 'Value', formatter: \"default('—')\" },\n { key: 'group.name', label: 'Group', sortable: true, formatter: \"default('Global')\" },\n {\n key: 'is_secret',\n label: 'Secret',\n formatter: \"boolean('Secret|bg-warning text-dark','Plain|bg-secondary')|badge\",\n width: '100px'\n },\n { key: 'created', label: 'Created', formatter: 'datetime', sortable: true }\n ],\n\n defaultQuery: {\n sort: 'key'\n },\n\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n showRefresh: true,\n showAdd: true,\n showExport: false,\n\n addButtonLabel: 'New Setting',\n\n emptyMessage: 'No settings found.',\n\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n }\n });\n }\n}\n\nexport default SettingTablePage;\n","/**\n * FileManagerTablePage - File Manager backend management using TablePage component\n * Clean implementation using TablePage with minimal overrides\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { FileManagerList, FileManagerForms } from '@core/models/Files.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\n\nclass FileManagerTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_file_managers',\n pageName: 'Manage Storage Backends',\n router: \"admin/file-managers\",\n Collection: FileManagerList,\n\n formCreate: FileManagerForms.create,\n formEdit: FileManagerForms.edit,\n\n // Column definitions\n columns: [\n {\n key: 'id',\n label: 'ID',\n sortable: true,\n class: 'text-muted'\n },\n {\n key: 'name',\n label: 'Name',\n formatter: \"default('Unnamed Backend')\"\n },\n {\n key: 'backend_url',\n label: 'Backend URL',\n sortable: true\n },\n {\n key: 'is_default',\n label: 'Default',\n formatter: \"boolean|badge\"\n },\n {\n key: 'is_active',\n label: 'Active',\n formatter: \"boolean|badge\"\n },\n {\n key: 'is_public',\n label: 'Public',\n formatter: \"boolean|badge\"\n },\n {\n key: 'backend_type',\n label: 'Type',\n formatter: \"default('Unknown')\"\n },\n {\n key: 'created',\n label: 'Created',\n formatter: \"epoch|datetime\"\n }\n ],\n\n contextMenu: [\n { icon: 'bi-pencil', action: 'edit', label: 'Edit Name' },\n { icon: 'bi-shield', action: 'edit-credentials', label: 'Edit Credentials' },\n { icon: 'bi-person', action: 'edit-owners', label: 'Edit Owners' },\n { divider: true },\n { icon: 'bi-copy', action: 'clone', label: 'Clone Manager' },\n { divider: true },\n { icon: 'bi-check', action: 'test-connection', label: 'Test Connection' },\n { icon: 'bi-question-circle', action: 'check-cors', label: 'Check CORS' },\n { icon: 'bi-wrench', action: 'fix-cors', label: 'Fix CORS' },\n ],\n\n // Table features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Toolbar\n showRefresh: true,\n showAdd: true,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No storage backends found. Click \"Add Storage Backend\" to configure your first backend.',\n\n // Batch actions\n batchBarLocation: 'top',\n batchActions: [\n { label: \"Delete\", icon: \"bi bi-trash\", action: \"batch-delete\" },\n { label: \"Export\", icon: \"bi bi-download\", action: \"batch-export\" },\n { label: \"Activate\", icon: \"bi bi-check-circle\", action: \"batch-activate\" },\n { label: \"Deactivate\", icon: \"bi bi-x-circle\", action: \"batch-deactivate\" }\n ],\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n }\n });\n }\n\n async onActionEditOwners(event, element) {\n const item = this.collection.get(element.dataset.id);\n const result = await Dialog.showModelForm({\n title: 'Edit Owners',\n model: item,\n fields: FileManagerForms.owners.fields\n });\n if (!result) return true;\n if (result.success) {\n this.getApp().toast.success(\"Owners Updated successfully\");\n } else {\n this.getApp().toast.error(\"Owners update failed\");\n }\n }\n\n async onActionCheckCors(event, element) {\n const item = this.collection.get(element.dataset.id);\n const result = await item.save({check_cors: true});\n if (result.success && result.data.status) {\n // this.getApp().toast.success(result.data.result);\n await Dialog.showData({\n title: `Audit Report - ${item._.name}`,\n data: result.data,\n size: 'lg'\n });\n } else {\n this.getApp().toast.error(\"Connection test failed\");\n }\n return true;\n }\n\n async onActionTestConnection(event, element) {\n const item = this.collection.get(element.dataset.id);\n const result = await item.save({test_connection: true});\n if (result.success && result.data.status) {\n this.getApp().toast.success(\"Connection test successful\");\n } else {\n this.getApp().toast.error(\"Connection test failed\");\n }\n return true;\n }\n\n async onActionEditCredentials(event, element) {\n const item = this.collection.get(element.dataset.id);\n const result = await Dialog.showModelForm({\n title: 'Edit Credentials',\n model: item,\n fields: FileManagerForms.credentials.fields\n });\n\n if (!result) return true;\n\n if (result.success && result.data.status) {\n this.getApp().toast.success(\"Credentials updated successfully\");\n } else {\n this.getApp().toast.error(\"Failed to update credentials\");\n }\n return true;\n }\n\n async onActionClone(event, element) {\n\n const confirmed = await Dialog.showConfirm({\n title: 'Clone File Manager',\n message: 'This will create a clone with the same credentials.',\n });\n\n if (!confirmed) {\n return true;\n }\n\n const item = this.collection.get(element.dataset.id);\n const result = await item.save({clone: true});\n if (result.success && result.data.status) {\n this.getApp().toast.success(\"Connection cloned successfully\");\n this.collection.fetch();\n } else {\n this.getApp().toast.error(\"Failed to clone connection\");\n }\n return true;\n }\n}\n\nexport default FileManagerTablePage;\n","/**\n * FileView - Comprehensive file management interface\n *\n * Features:\n * - Clean header with file thumbnail/icon, name, size, and status\n * - Tabbed interface for Info (metadata) and Renditions\n * - Integrated with DataView, Table, and ContextMenu components\n */\n\nimport View from '@core/View.js';\nimport TabView from '@core/views/navigation/TabView.js';\nimport DataView from '@core/views/data/DataView.js';\nimport TableView from '@core/views/table/TableView.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport Collection from '@core/Collection.js';\nimport { File, FileForms } from '@core/models/Files.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\nimport LightboxGallery from '@ext/lightbox/LightboxGallery.js';\nimport PDFViewer from '@ext/lightbox/PDFViewer.js';\n\nclass FileView extends View {\n constructor(options = {}) {\n super({\n className: 'file-view',\n ...options\n });\n\n this.model = options.model || new File(options.data || {});\n this.isImage = this.model.get('category') === 'image';\n\n // Prepare renditions for the table by converting the renditions object to an array\n const renditionsData = this.model.get('renditions') || {};\n this.renditionsCollection = new Collection(Object.values(renditionsData));\n\n this.template = `\n <div class=\"file-view-container\">\n <!-- Header -->\n <div class=\"d-flex justify-content-between align-items-center mb-4\">\n <!-- Left Side: Thumbnail & Info -->\n <div class=\"d-flex align-items-center gap-3\">\n <div class=\"file-thumbnail\" style=\"width: 80px; height: 80px;\">\n {{#isImage}}\n <a href=\"{{model.url}}\" target=\"_blank\" title=\"View original file\">\n <img src=\"{{model.renditions.thumbnail.url|default(model.url)}}\" class=\"img-fluid rounded\" style=\"width: 80px; height: 80px; object-fit: cover;\">\n </a>\n {{/isImage}}\n {{^isImage}}\n <div class=\"avatar-placeholder rounded bg-light d-flex align-items-center justify-content-center\" style=\"width: 80px; height: 80px;\">\n <i class=\"bi bi-file-earmark-text text-secondary\" style=\"font-size: 40px;\"></i>\n </div>\n {{/isImage}}\n </div>\n <div>\n <h3 class=\"mb-1\" style=\"word-break: break-all;\">{{model.filename|truncate(40)}}</h3>\n <div class=\"text-muted small\">\n <span><i class=\"bi bi-hdd\"></i> {{model.file_size|filesize}}</span>\n <span class=\"mx-2\">|</span>\n <span>{{model.content_type}}</span>\n </div>\n <div class=\"text-muted small mt-1\">\n Uploaded: {{model.created|datetime}}\n </div>\n </div>\n </div>\n\n <!-- Right Side: Status & Actions -->\n <div class=\"d-flex align-items-center gap-4\">\n <div class=\"text-end\">\n <div class=\"d-flex align-items-center gap-2 justify-content-end\">\n <span class=\"badge {{model.upload_status|badge}}\">{{model.upload_status|capitalize}}</span>\n </div>\n <div class=\"text-muted small mt-1\">\n Public: {{{model.is_public|yesnoicon}}}\n </div>\n </div>\n <div data-container=\"file-context-menu\"></div>\n </div>\n </div>\n\n <!-- Tab Container -->\n <div data-container=\"file-tabs\"></div>\n </div>\n `;\n }\n\n async onInit() {\n // Info Tab using DataView\n this.infoView = new DataView({\n model: this.model,\n className: \"p-3\",\n showEmptyValues: true,\n emptyValueText: '—',\n columns: 2,\n fields: [\n { name: 'id', label: 'ID' },\n { name: 'filename', label: 'Filename' },\n { name: 'storage_filename', label: 'Storage Filename' },\n { name: 'content_type', label: 'Content Type' },\n { name: 'file_size', label: 'File Size', format: 'filesize' },\n { name: 'category', label: 'Category' },\n { name: 'upload_status', label: 'Status', format: 'badge' },\n { name: 'created', label: 'Created', format: 'datetime' },\n { name: 'modified', label: 'Modified', format: 'datetime' },\n { name: 'user.display_name', label: 'Uploaded By' },\n { name: 'file_manager.name', label: 'Storage Backend' },\n { name: 'storage_file_path', label: 'Storage Path' },\n { name: 'url', label: 'Public URL', format: 'url' },\n { name: 'is_public', label: 'Is Public', format: 'boolean' },\n ]\n });\n\n // Renditions Tab using TableView\n this.renditionsView = new TableView({\n collection: this.renditionsCollection,\n columns: [\n { key: 'role', label: 'Role', formatter: 'badge' },\n { key: 'filename', label: 'Filename', formatter: 'truncate(40)' },\n { key: 'file_size', label: 'Size', formatter: 'filesize' },\n { key: 'content_type', label: 'Content Type' },\n {\n key: 'actions',\n label: 'Actions',\n template: `\n <a href=\"{{url}}\" target=\"_blank\" class=\"btn btn-sm btn-outline-primary\" title=\"View\">\n <i class=\"bi bi-eye\"></i>\n </a>\n <a href=\"{{url}}\" download=\"{{filename}}\" class=\"btn btn-sm btn-outline-secondary\" title=\"Download\">\n <i class=\"bi bi-download\"></i>\n </a>\n `\n }\n ]\n });\n\n // Create TabView, only showing Renditions tab if they exist\n const tabs = { 'Info': this.infoView };\n tabs['Renditions'] = this.renditionsView;\n\n this.tabView = new TabView({\n tabs: tabs,\n activeTab: 'Info',\n containerId: 'file-tabs'\n });\n this.addChild(this.tabView);\n\n // Create ContextMenu\n const fileMenu = new ContextMenu({\n containerId: 'file-context-menu',\n className: \"context-menu-view header-menu-absolute\",\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: [\n { label: 'View', action: 'view-file', icon: 'bi-eye' },\n { label: 'Download', action: 'download-file', icon: 'bi bi-download' },\n { label: 'Edit Details', action: 'edit-file', icon: 'bi bi-pencil' },\n { type: 'divider' },\n this.model.get('is_public')\n ? { label: 'Make Private', action: 'make-private', icon: 'bi bi-lock' }\n : { label: 'Make Public', action: 'make-public', icon: 'bi bi-unlock' },\n { type: 'divider' },\n { label: 'Delete File', action: 'delete-file', icon: 'bi bi-trash', danger: true }\n ]\n }\n });\n this.addChild(fileMenu);\n }\n\n async onActionViewFile() {\n const contentType = this.model.get('content_type');\n const fileUrl = this.model.get('url');\n\n if (contentType.startsWith('image/')) {\n const renditions = this.model.get('renditions') || {};\n const images = [\n { src: fileUrl, alt: 'Original' },\n ...Object.values(renditions).map(r => ({ src: r.url, alt: r.role }))\n ];\n LightboxGallery.show(images, { fitToScreen: false });\n } else if (contentType === 'application/pdf') {\n PDFViewer.showDialog(fileUrl, { title: this.model.get('filename') });\n } else {\n window.open(fileUrl, '_blank');\n }\n }\n\n async onActionDownloadFile() {\n const url = this.model.get('url');\n if (url) {\n const a = document.createElement('a');\n a.href = url;\n a.download = this.model.get('filename');\n document.body.appendChild(a);\n a.click();\n document.body.removeChild(a);\n }\n }\n\n async onActionEditFile() {\n const resp = await Dialog.showModelForm({\n title: `Edit File - ${this.model.get('filename')}`,\n model: this.model,\n formConfig: FileForms.edit,\n });\n if (resp) {\n this.render();\n }\n }\n\n async onActionMakePublic() {\n await this.model.save({ is_public: true });\n this.render();\n }\n\n async onActionMakePrivate() {\n await this.model.save({ is_public: false });\n this.render();\n }\n\n async onActionDeleteFile() {\n const confirmed = await Dialog.confirm(\n `Are you sure you want to delete the file \"${this.model.get('filename')}\"? This action cannot be undone.`,\n 'Confirm Deletion',\n { confirmClass: 'btn-danger', confirmText: 'Delete' }\n );\n if (confirmed) {\n const resp = await this.model.destroy();\n if (resp.success) {\n this.emit('file:deleted', { model: this.model });\n }\n }\n }\n}\n\nFile.VIEW_CLASS = FileView;\n\nexport default FileView;\n","/**\n * FileTablePage - File management using TablePage component\n * Clean implementation with file drop support\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { File, FileList, FileForms } from '@core/models/Files.js';\nimport FileView from './FileView.js';\nimport applyFileDropMixin from '@core/mixins/FileDropMixin.js';\n\nclass FileTablePage extends TablePage {\n constructor(options = {}) {\n super({\n name: 'admin_files',\n pageName: 'Manage Files',\n router: \"admin/files\",\n Collection: FileList,\n\n // Don't use formCreate - we have custom file upload handling\n formEdit: FileForms.edit,\n itemViewClass: FileView,\n \n // Custom add handler for file uploads\n onAdd: async (event) => {\n await this.handleFileUpload(event);\n },\n\n viewDialogOptions: {\n header: false,\n size: 'xl'\n },\n\n // Column definitions\n columns: [\n {\n key: 'id',\n label: 'ID',\n sortable: true,\n class: 'text-muted'\n },\n {\n key: 'filename',\n label: 'Filename'\n },\n {\n key: 'content_type',\n label: 'Type',\n formatter: \"default('Unknown')\"\n },\n {\n key: 'file_size',\n label: 'Size',\n formatter: \"filesize\"\n },\n {\n key: 'group.name',\n label: 'Group',\n formatter: \"default('No Group')\"\n },\n {\n key: 'upload_status',\n label: 'Status',\n formatter: \"badge\"\n },\n {\n key: 'created',\n label: 'Uploaded',\n formatter: \"epoch|datetime\"\n }\n ],\n\n // Table features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Toolbar\n showRefresh: true,\n showAdd: true,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No files found. Click \"Add File\" to upload your first file.',\n\n // Batch actions\n batchBarLocation: 'top',\n batchActions: [\n { label: \"Download\", icon: \"bi bi-download\", action: \"batch-download\" },\n { label: \"Delete\", icon: \"bi bi-trash\", action: \"batch-delete\" },\n { label: \"Move to Group\", icon: \"bi bi-folder\", action: \"batch-move\" }\n ],\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n },\n ...options,\n });\n\n // Enable file drop support\n this.enableFileDrop({\n acceptedTypes: ['*/*'],\n maxFileSize: 100 * 1024 * 1024, // 100MB\n multiple: false,\n validateOnDrop: true\n });\n }\n\n /**\n * Handle file upload via file picker\n * Opens native file picker and uses same upload flow as drag-drop\n */\n async handleFileUpload(event) {\n if (event) event.preventDefault();\n\n // Create hidden file input element\n const fileInput = document.createElement('input');\n fileInput.type = 'file';\n fileInput.accept = '*/*';\n fileInput.multiple = false;\n fileInput.style.display = 'none';\n\n // Handle file selection\n fileInput.addEventListener('change', async (e) => {\n const file = e.target.files[0];\n \n if (!file) {\n return;\n }\n\n // Validate file size (same as FileDropMixin config)\n const maxSize = 100 * 1024 * 1024; // 100MB\n if (file.size > maxSize) {\n this.showError(`File size (${this._formatFileSize(file.size)}) exceeds maximum (${this._formatFileSize(maxSize)})`);\n return;\n }\n\n // Use the same upload flow as drag-drop\n try {\n const fileModel = new File();\n let extra = {};\n if (this.options.requiresGroup && this.getApp().activeGroup) {\n extra.group = this.getApp().activeGroup.id;\n }\n\n const upload = fileModel.upload({\n file: file,\n name: file.name,\n description: `File uploaded on ${new Date().toLocaleDateString()}`,\n showToast: true,\n onProgress: (progressInfo) => {\n console.log(`Upload progress: ${progressInfo.percentage}%`);\n },\n onComplete: (result) => {\n console.log('Upload completed:', result);\n this.refresh();\n },\n onError: (error) => {\n console.error('Upload failed:', error);\n this.showError('Upload failed: ' + error.message);\n },\n ...extra\n });\n\n await upload;\n } catch (error) {\n console.error('Error starting file upload:', error);\n this.showError('Failed to start file upload: ' + error.message);\n } finally {\n // Clean up file input\n fileInput.remove();\n }\n });\n\n // Trigger file picker\n document.body.appendChild(fileInput);\n fileInput.click();\n }\n\n /**\n * Format file size for display\n * @param {number} bytes - File size in bytes\n * @returns {string} Formatted file size\n * @private\n */\n _formatFileSize(bytes) {\n if (bytes === 0) return '0 Bytes';\n const k = 1024;\n const sizes = ['Bytes', 'KB', 'MB', 'GB'];\n const i = Math.floor(Math.log(bytes) / Math.log(k));\n return parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + ' ' + sizes[i];\n }\n\n async onFileDrop(files, event, validation) {\n const file = files[0];\n console.log(`File Dropped: ${file.name} (${file.type}) (${file.size} bytes)`);\n\n try {\n // Create new File model instance\n const fileModel = new File();\n let extra = {};\n if (this.options.requiresGroup && this.getApp().activeGroup) {\n extra.group = this.getApp().activeGroup.id;\n }\n\n // Start upload with progress tracking\n const upload = fileModel.upload({\n file: file,\n name: file.name,\n description: `File uploaded via drag & drop on ${new Date().toLocaleDateString()}`,\n showToast: true,\n onProgress: (progressInfo) => {\n console.log(`Upload progress: ${progressInfo.percentage}%`);\n },\n onComplete: (result) => {\n console.log('Upload completed:', result);\n this.refresh();\n },\n onError: (error) => {\n console.error('Upload failed:', error);\n this.showError('Upload failed: ' + error.message);\n },\n ...extra\n });\n\n await upload;\n } catch (error) {\n console.error('Error starting file upload:', error);\n this.showError('Failed to start file upload: ' + error.message);\n }\n }\n}\n\n// Apply file drop mixin\napplyFileDropMixin(FileTablePage);\n\nexport default FileTablePage;\n","/**\n * S3BucketTablePage - S3 Bucket management using TablePage component\n * Clean implementation using TablePage with minimal overrides\n */\n\nimport TablePage from '@core/pages/TablePage.js';\nimport { S3BucketList, S3BucketForms } from '@core/models/AWS.js';\n\nclass S3BucketTablePage extends TablePage {\n constructor(options = {}) {\n super({\n ...options,\n name: 'admin_s3_buckets',\n pageName: 'Manage S3 Buckets',\n router: \"admin/s3-buckets\",\n Collection: S3BucketList,\n \n formCreate: S3BucketForms.create,\n formEdit: S3BucketForms.edit,\n\n // Column definitions\n columns: [\n {\n key: 'id',\n label: 'ID',\n width: '60px',\n sortable: true,\n class: 'text-muted'\n },\n {\n key: 'name',\n label: 'Bucket Name',\n sortable: true\n },\n {\n key: 'created',\n label: 'Created',\n formatter: \"epoch|datetime\"\n }\n ],\n\n // Table features\n selectable: true,\n searchable: true,\n sortable: true,\n filterable: true,\n paginated: true,\n\n // Toolbar\n showRefresh: true,\n showAdd: true,\n showExport: true,\n\n // Empty state\n emptyMessage: 'No S3 buckets found. Click \"Add S3 Bucket\" to create your first bucket.',\n\n // Batch actions\n batchBarLocation: 'top',\n batchActions: [\n { label: \"Delete\", icon: \"bi bi-trash\", action: \"batch-delete\" },\n { label: \"Export\", icon: \"bi bi-download\", action: \"batch-export\" },\n { label: \"Make Public\", icon: \"bi bi-unlock\", action: \"batch-public\" },\n { label: \"Make Private\", icon: \"bi bi-lock\", action: \"batch-private\" },\n { label: \"Empty Bucket\", icon: \"bi bi-bucket\", action: \"batch-empty\" }\n ],\n\n // Table display options\n tableOptions: {\n striped: true,\n bordered: false,\n hover: true,\n responsive: false\n }\n });\n }\n}\n\nexport default S3BucketTablePage;","/**\n * UserDeviceLocationView - Clean, modern session location detail view\n *\n * Shows a device-location record with:\n * - Header: device icon, browser/device summary, location, IP, threat badges\n * - SideNavView: Location, Device, Network & Risk, Map (if coords), Events\n *\n * Opened when clicking a location row in admin UserView or DeviceView.\n */\n\nimport View from '@core/View.js';\nimport SideNavView from '@core/views/navigation/SideNavView.js';\nimport TableView from '@core/views/table/TableView.js';\nimport ContextMenu from '@core/views/feedback/ContextMenu.js';\nimport { UserDeviceLocation } from '@core/models/User.js';\nimport { IncidentEventList } from '@core/models/Incident.js';\nimport { LogList } from '@core/models/Log.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\n\nclass UserDeviceLocationView extends View {\n constructor(options = {}) {\n super({\n className: 'udl-view',\n ...options\n });\n\n this.model = options.model || new UserDeviceLocation(options.data || {});\n\n // Extract nested data\n this._ud = this.model.get('user_device') || {};\n this._di = this._ud.device_info || {};\n this._geo = this.model.get('geolocation') || {};\n\n // Computed for template\n this.deviceIcon = this._getDeviceIcon();\n this.browserFull = this._getBrowser();\n this.osFull = this._getOS();\n this.deviceFull = this._getDevice();\n this.locationSummary = this._getLocationSummary();\n this.countryFlag = this._geo.country_code || '';\n this.threatLevel = this._geo.threat_level || 'unknown';\n this.threatColor = this._getThreatColor();\n this.hasCoordinates = !!(this._geo.latitude && this._geo.longitude);\n\n this.template = `\n <style>\n .udl-header { display: flex; justify-content: space-between; align-items: flex-start; margin-bottom: 1.5rem; }\n .udl-identity { display: flex; align-items: center; gap: 1rem; }\n .udl-icon-wrap { width: 48px; height: 48px; border-radius: 12px; display: flex; align-items: center; justify-content: center; font-size: 1.4rem; flex-shrink: 0; }\n .udl-title { font-size: 1.15rem; font-weight: 600; margin: 0; line-height: 1.3; }\n .udl-subtitle { font-size: 0.8rem; color: #6c757d; margin-top: 0.15rem; }\n .udl-right { display: flex; align-items: flex-start; gap: 0.75rem; }\n .udl-threat-label { font-size: 0.7rem; color: #adb5bd; text-transform: uppercase; letter-spacing: 0.04em; }\n .udl-threat-value { font-size: 1rem; font-weight: 600; }\n .udl-threat-flags { display: flex; gap: 0.35rem; margin-top: 0.35rem; }\n .udl-flag { font-size: 0.65rem; padding: 0.15em 0.45em; border-radius: 3px; font-weight: 600; }\n </style>\n\n <div class=\"udl-header\">\n <div class=\"udl-identity\">\n <div class=\"udl-icon-wrap bg-primary bg-opacity-10 text-primary\">\n <i class=\"bi {{deviceIcon}}\"></i>\n </div>\n <div>\n <h4 class=\"udl-title\">{{locationSummary}}</h4>\n <div class=\"udl-subtitle\">\n {{browserFull}} <span class=\"text-muted\">on</span> {{deviceFull}}\n <span class=\"text-muted mx-1\">&middot;</span>\n {{model.ip_address}}\n </div>\n </div>\n </div>\n <div class=\"udl-right\">\n <div class=\"text-end\">\n <div class=\"udl-threat-label\">Threat Level</div>\n <div class=\"udl-threat-value {{threatColor}}\">{{threatLevel|capitalize}}</div>\n <div class=\"udl-threat-flags\">\n {{#_geo.is_vpn}}<span class=\"udl-flag bg-warning text-dark\">VPN</span>{{/_geo.is_vpn}}\n {{#_geo.is_tor}}<span class=\"udl-flag bg-danger text-white\">Tor</span>{{/_geo.is_tor}}\n {{#_geo.is_proxy}}<span class=\"udl-flag bg-warning text-dark\">Proxy</span>{{/_geo.is_proxy}}\n {{#_geo.is_datacenter}}<span class=\"udl-flag bg-secondary text-white\">DC</span>{{/_geo.is_datacenter}}\n {{#_geo.is_cloud}}<span class=\"udl-flag bg-info text-white\">Cloud</span>{{/_geo.is_cloud}}\n </div>\n </div>\n <div data-container=\"udl-context-menu\"></div>\n </div>\n </div>\n\n <div data-container=\"udl-sidenav\" style=\"min-height: 300px;\"></div>\n `;\n }\n\n // ── Computed helpers ─────────────────────────\n\n _getDeviceIcon() {\n const browser = this._di?.user_agent?.family?.toLowerCase() || '';\n const os = this._di?.os?.family?.toLowerCase() || '';\n const device = this._di?.device?.family?.toLowerCase() || '';\n if (browser.includes('chrome')) return 'bi-browser-chrome';\n if (browser.includes('firefox')) return 'bi-browser-firefox';\n if (browser.includes('safari')) return 'bi-browser-safari';\n if (browser.includes('edge')) return 'bi-browser-edge';\n if (os.includes('mac') || os.includes('ios')) return 'bi-apple';\n if (os.includes('windows')) return 'bi-windows';\n if (os.includes('android')) return 'bi-android2';\n if (device.includes('iphone')) return 'bi-phone';\n if (device.includes('ipad')) return 'bi-tablet';\n return 'bi-geo-alt';\n }\n\n _getBrowser() {\n const ua = this._di?.user_agent || {};\n return ua.family ? `${ua.family} ${ua.major || ''}`.trim() : 'Unknown Browser';\n }\n\n _getOS() {\n const os = this._di?.os || {};\n const ver = [os.major, os.minor].filter(Boolean).join('.');\n return os.family ? `${os.family} ${ver}`.trim() : 'Unknown OS';\n }\n\n _getDevice() {\n const dev = this._di?.device || {};\n const parts = [dev.brand, dev.family].filter(Boolean);\n return parts.length ? parts.join(' ') : 'Unknown Device';\n }\n\n _getLocationSummary() {\n const parts = [this._geo.city, this._geo.region, this._geo.country_name].filter(Boolean);\n return parts.length ? parts.join(', ') : 'Unknown Location';\n }\n\n _getThreatColor() {\n const level = (this._geo.threat_level || '').toLowerCase();\n if (level === 'high' || this._geo.is_threat) return 'text-danger';\n if (level === 'medium' || this._geo.is_suspicious) return 'text-warning';\n if (level === 'low') return 'text-success';\n return 'text-muted';\n }\n\n async onInit() {\n const geo = this._geo;\n const di = this._di;\n\n // ── Location section ────────────────────────\n const locationView = new View({\n model: this.model,\n template: `\n <style>\n .udl-section-label { font-size: 0.7rem; font-weight: 700; text-transform: uppercase; letter-spacing: 0.06em; color: #adb5bd; margin-bottom: 0.5rem; margin-top: 1.5rem; }\n .udl-section-label:first-child { margin-top: 0; }\n .udl-field-row { display: flex; align-items: baseline; padding: 0.5rem 0; border-bottom: 1px solid #f0f0f0; }\n .udl-field-row:last-child { border-bottom: none; }\n .udl-field-label { width: 130px; font-size: 0.78rem; color: #6c757d; flex-shrink: 0; }\n .udl-field-value { flex: 1; font-size: 0.88rem; color: #212529; }\n </style>\n\n <div class=\"udl-section-label\">Geography</div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">City</div>\n <div class=\"udl-field-value\">${geo.city || '—'}</div>\n </div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">Region</div>\n <div class=\"udl-field-value\">${geo.region || '—'}</div>\n </div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">Country</div>\n <div class=\"udl-field-value\">${geo.country_name || '—'} ${geo.country_code ? `<span class=\"text-muted\">(${geo.country_code})</span>` : ''}</div>\n </div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">Postal Code</div>\n <div class=\"udl-field-value\">${geo.postal_code || '—'}</div>\n </div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">Timezone</div>\n <div class=\"udl-field-value\">${geo.timezone || '—'}</div>\n </div>\n ${geo.latitude ? `\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">Coordinates</div>\n <div class=\"udl-field-value\">${geo.latitude}, ${geo.longitude}</div>\n </div>` : ''}\n\n <div class=\"udl-section-label\">Network</div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">IP Address</div>\n <div class=\"udl-field-value\" style=\"font-family: ui-monospace, monospace; font-size: 0.82rem;\">{{model.ip_address}}</div>\n </div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">ISP</div>\n <div class=\"udl-field-value\">${geo.isp || '—'}</div>\n </div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">ASN</div>\n <div class=\"udl-field-value\">${geo.asn || '—'} ${geo.asn_org ? `<span class=\"text-muted small\">(${geo.asn_org})</span>` : ''}</div>\n </div>\n\n <div class=\"udl-section-label\">Timestamps</div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">First Seen</div>\n <div class=\"udl-field-value\">{{model.first_seen|epoch|datetime|default('—')}}</div>\n </div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">Last Seen</div>\n <div class=\"udl-field-value\">{{model.last_seen|epoch|datetime|default('—')}}</div>\n </div>\n `\n });\n\n // ── Device section ──────────────────────────\n const deviceView = new View({\n model: this.model,\n template: `\n <style>\n .udl-section-label { font-size: 0.7rem; font-weight: 700; text-transform: uppercase; letter-spacing: 0.06em; color: #adb5bd; margin-bottom: 0.5rem; margin-top: 1.5rem; }\n .udl-section-label:first-child { margin-top: 0; }\n .udl-field-row { display: flex; align-items: baseline; padding: 0.5rem 0; border-bottom: 1px solid #f0f0f0; }\n .udl-field-row:last-child { border-bottom: none; }\n .udl-field-label { width: 130px; font-size: 0.78rem; color: #6c757d; flex-shrink: 0; }\n .udl-field-value { flex: 1; font-size: 0.88rem; color: #212529; }\n .udl-ua-string { font-family: ui-monospace, monospace; font-size: 0.73rem; color: #6c757d; word-break: break-all; line-height: 1.5; padding: 0.5rem 0.75rem; background: #f8f9fa; border-radius: 6px; margin-top: 0.25rem; }\n </style>\n\n <div class=\"udl-section-label\">Browser</div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">Name</div>\n <div class=\"udl-field-value\">${di?.user_agent?.family || '—'}</div>\n </div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">Version</div>\n <div class=\"udl-field-value\">${[di?.user_agent?.major, di?.user_agent?.minor, di?.user_agent?.patch].filter(Boolean).join('.') || '—'}</div>\n </div>\n\n <div class=\"udl-section-label\">Operating System</div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">Name</div>\n <div class=\"udl-field-value\">${di?.os?.family || '—'}</div>\n </div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">Version</div>\n <div class=\"udl-field-value\">${[di?.os?.major, di?.os?.minor, di?.os?.patch].filter(Boolean).join('.') || '—'}</div>\n </div>\n\n <div class=\"udl-section-label\">Hardware</div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">Brand</div>\n <div class=\"udl-field-value\">${di?.device?.brand || '—'}</div>\n </div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">Family</div>\n <div class=\"udl-field-value\">${di?.device?.family || '—'}</div>\n </div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">Model</div>\n <div class=\"udl-field-value\">${di?.device?.model || '—'}</div>\n </div>\n\n ${di?.string ? `\n <div class=\"udl-section-label\">User Agent String</div>\n <div class=\"udl-ua-string\">${di.string}</div>` : ''}\n `\n });\n\n // ── Risk & Reputation section ───────────────\n const riskView = new View({\n model: this.model,\n template: `\n <style>\n .udl-section-label { font-size: 0.7rem; font-weight: 700; text-transform: uppercase; letter-spacing: 0.06em; color: #adb5bd; margin-bottom: 0.5rem; margin-top: 1.5rem; }\n .udl-section-label:first-child { margin-top: 0; }\n .udl-field-row { display: flex; align-items: baseline; padding: 0.5rem 0; border-bottom: 1px solid #f0f0f0; }\n .udl-field-row:last-child { border-bottom: none; }\n .udl-field-label { width: 130px; font-size: 0.78rem; color: #6c757d; flex-shrink: 0; }\n .udl-field-value { flex: 1; font-size: 0.88rem; color: #212529; }\n .udl-risk-icon { font-size: 0.85rem; margin-right: 0.35rem; }\n .udl-risk-yes { color: #dc3545; }\n .udl-risk-no { color: #adb5bd; }\n </style>\n\n <div class=\"udl-section-label\">Threat Assessment</div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">Threat Level</div>\n <div class=\"udl-field-value ${this.threatColor}\" style=\"font-weight: 600;\">${(geo.threat_level || 'Unknown')}</div>\n </div>\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\">Risk Score</div>\n <div class=\"udl-field-value\">${geo.risk_score != null ? geo.risk_score : '—'}</div>\n </div>\n\n <div class=\"udl-section-label\">Detection Flags</div>\n ${this._riskRow('VPN', 'bi-shield', geo.is_vpn)}\n ${this._riskRow('Tor Exit Node', 'bi-shield-lock', geo.is_tor)}\n ${this._riskRow('Proxy', 'bi-diagram-3', geo.is_proxy)}\n ${this._riskRow('Cloud Provider', 'bi-cloud', geo.is_cloud)}\n ${this._riskRow('Datacenter', 'bi-hdd-stack', geo.is_datacenter)}\n ${this._riskRow('Mobile', 'bi-phone', geo.is_mobile)}\n\n <div class=\"udl-section-label\">Reputation</div>\n ${this._riskRow('Known Attacker', 'bi-exclamation-triangle', geo.is_known_attacker)}\n ${this._riskRow('Known Abuser', 'bi-flag', geo.is_known_abuser)}\n ${this._riskRow('Threat', 'bi-shield-exclamation', geo.is_threat)}\n ${this._riskRow('Suspicious', 'bi-question-circle', geo.is_suspicious)}\n `\n });\n\n // ── Build sections array ────────────────────\n const sections = [\n { key: 'location', label: 'Location', icon: 'bi-geo-alt', view: locationView },\n { key: 'device', label: 'Device', icon: 'bi-laptop', view: deviceView },\n { key: 'risk', label: 'Risk', icon: 'bi-shield-exclamation', view: riskView }\n ];\n\n // Map (if coordinates)\n if (this.hasCoordinates) {\n try {\n const MapView = (await import('@ext/map/MapView.js')).default;\n const mapView = new MapView({\n markers: [{\n lat: this._geo.latitude,\n lng: this._geo.longitude,\n popup: `<strong>${this.model.get('ip_address')}</strong><br>${this.locationSummary}`\n }],\n tileLayer: 'light',\n zoom: 6,\n height: 400\n });\n sections.push({ key: 'map', label: 'Map', icon: 'bi-map', view: mapView });\n } catch (e) {\n // MapView extension not available — skip\n }\n }\n\n // Events (by IP)\n const ip = this.model.get('ip_address');\n if (ip) {\n const eventsView = new TableView({\n collection: new IncidentEventList({\n params: { size: 10, source_ip: ip }\n }),\n hideActivePillNames: ['source_ip'],\n columns: [\n { key: 'created', label: 'Date', formatter: 'datetime', sortable: true, width: '150px' },\n { key: 'category|badge', label: 'Category' },\n { key: 'title', label: 'Title' }\n ]\n });\n sections.push({ type: 'divider', label: 'Activity' });\n sections.push({ key: 'events', label: 'Events', icon: 'bi-calendar-event', view: eventsView });\n\n const logsView = new TableView({\n collection: new LogList({\n params: { size: 10, ip }\n }),\n permissions: 'view_logs',\n hideActivePillNames: ['ip'],\n columns: [\n { key: 'created', label: 'Timestamp', sortable: true, formatter: 'epoch|datetime' },\n { key: 'level', label: 'Level', sortable: true },\n { key: 'kind', label: 'Kind' },\n { name: 'log', label: 'Log' }\n ]\n });\n sections.push({ key: 'logs', label: 'Logs', icon: 'bi-journal-text', view: logsView, permissions: 'view_logs' });\n }\n\n // ── SideNavView ─────────────────────────────\n this.sideNavView = new SideNavView({\n containerId: 'udl-sidenav',\n activeSection: this.hasCoordinates ? 'map' : 'location',\n navWidth: 160,\n contentPadding: '1rem 1.5rem',\n enableResponsive: true,\n minWidth: 450,\n sections\n });\n this.addChild(this.sideNavView);\n\n // ── Context Menu ────────────────────────────\n const menu = new ContextMenu({\n containerId: 'udl-context-menu',\n className: 'context-menu-view header-menu-absolute',\n context: this.model,\n config: {\n icon: 'bi-three-dots-vertical',\n items: [\n ...(this._ud?.user ? [{ label: 'View User', action: 'view-user', icon: 'bi-person' }] : []),\n ...(this._ud?.id ? [{ label: 'View Device', action: 'view-device', icon: 'bi-laptop' }] : []),\n ...(this.hasCoordinates ? [{\n label: 'Open in Maps',\n action: 'open-in-maps',\n icon: 'bi-box-arrow-up-right'\n }] : []),\n { type: 'divider' },\n { label: 'Delete Record', action: 'delete-record', icon: 'bi-trash', danger: true }\n ]\n }\n });\n this.addChild(menu);\n }\n\n // ── Helpers ──────────────────────────────────\n\n _riskRow(label, icon, value) {\n const cls = value ? 'udl-risk-yes' : 'udl-risk-no';\n const indicator = value\n ? '<i class=\"bi bi-check-circle-fill udl-risk-icon udl-risk-yes\"></i>Yes'\n : '<i class=\"bi bi-dash-circle udl-risk-icon udl-risk-no\"></i>No';\n return `\n <div class=\"udl-field-row\">\n <div class=\"udl-field-label\"><i class=\"bi ${icon} me-1 ${cls}\"></i>${label}</div>\n <div class=\"udl-field-value\">${indicator}</div>\n </div>`;\n }\n\n // ── Actions ──────────────────────────────────\n\n async onActionViewUser() {\n const userId = this._ud?.user?.id || this._ud?.user;\n if (userId) this.emit('view-user', { userId });\n }\n\n async onActionViewDevice() {\n const deviceId = this._ud?.id;\n if (deviceId) this.emit('view-device', { deviceId });\n }\n\n async onActionOpenInMaps() {\n if (this.hasCoordinates) {\n window.open(`https://www.google.com/maps/search/?api=1&query=${this._geo.latitude},${this._geo.longitude}`, '_blank');\n }\n }\n\n async onActionDeleteRecord() {\n const confirmed = await Dialog.confirm(\n 'Are you sure you want to delete this location record?',\n 'Delete Location Record'\n );\n if (!confirmed) return true;\n\n const resp = await this.model.destroy();\n if (resp.success) {\n this.emit('location:deleted', { model: this.model });\n }\n return true;\n }\n\n static async show(id) {\n const model = new UserDeviceLocation({ id });\n await model.fetch();\n if (model.id) {\n return Dialog.showDialog({\n title: false,\n size: 'lg',\n body: new UserDeviceLocationView({ model }),\n buttons: [{ text: 'Close', class: 'btn-secondary', dismiss: true }]\n });\n }\n Dialog.alert({ message: `Could not find location record: ${id}`, type: 'warning' });\n return null;\n }\n}\n\nUserDeviceLocation.VIEW_CLASS = UserDeviceLocationView;\nexport default UserDeviceLocationView;\n","/**\n * CloudWatchResourceView - Detailed metric view for a single AWS resource\n *\n * Shows all available metric categories for an EC2, RDS, or Redis resource\n * using CloudWatchChart (MetricsChart) instances in a responsive grid.\n * Each chart has its own granularity/date controls.\n * Can be opened in a Dialog via the static show() method.\n */\nimport View from '@core/View.js';\nimport Dialog from '@core/views/feedback/Dialog.js';\nimport CloudWatchChart from './CloudWatchChart.js';\n\nconst METRIC_CATEGORIES = {\n ec2: [\n { key: 'cpu', label: 'CPU Utilization', unit: '%' },\n { key: 'memory', label: 'Memory Usage', unit: '%' },\n { key: 'disk', label: 'Disk Usage', unit: '%' },\n { key: 'net_in', label: 'Network In', unit: 'bytes' },\n { key: 'net_out', label: 'Network Out', unit: 'bytes' },\n { key: 'disk_read', label: 'Disk Read Ops', unit: 'ops' },\n { key: 'disk_write', label: 'Disk Write Ops', unit: 'ops' },\n { key: 'status_check', label: 'Status Check', unit: '' }\n ],\n rds: [\n { key: 'cpu', label: 'CPU Utilization', unit: '%' },\n { key: 'conns', label: 'Active Connections', unit: '' },\n { key: 'free_storage', label: 'Free Storage', unit: 'bytes' },\n { key: 'free_memory', label: 'Freeable Memory', unit: 'bytes' },\n { key: 'read_iops', label: 'Read IOPS', unit: 'ops/s' },\n { key: 'write_iops', label: 'Write IOPS', unit: 'ops/s' },\n { key: 'read_latency', label: 'Read Latency', unit: 's' },\n { key: 'write_latency', label: 'Write Latency', unit: 's' },\n { key: 'net_in', label: 'Network In', unit: 'bytes' },\n { key: 'net_out', label: 'Network Out', unit: 'bytes' }\n ],\n redis: [\n { key: 'cpu', label: 'CPU Utilization', unit: '%' },\n { key: 'conns', label: 'Current Connections', unit: '' },\n { key: 'cache_memory', label: 'Cache Memory Used', unit: 'bytes' },\n { key: 'cache_hits', label: 'Cache Hits', unit: '' },\n { key: 'cache_misses', label: 'Cache Misses', unit: '' },\n { key: 'replication_lag', label: 'Replication Lag', unit: 's' },\n { key: 'net_in', label: 'Network In', unit: 'bytes' },\n { key: 'net_out', label: 'Network Out', unit: 'bytes' }\n ]\n};\n\nconst TYPE_ICONS = { ec2: 'bi-pc-display', rds: 'bi-database', redis: 'bi-lightning-charge' };\nconst TYPE_LABELS = { ec2: 'EC2 Instance', rds: 'RDS Database', redis: 'ElastiCache Redis' };\n\nfunction yAxisForUnit(unit) {\n if (unit === '%') return { label: '%', beginAtZero: true, max: 100 };\n if (unit === 'bytes') return { label: 'Bytes', beginAtZero: true };\n if (unit === 's') return { label: 'Seconds', beginAtZero: true };\n return { beginAtZero: true };\n}\n\nexport default class CloudWatchResourceView extends View {\n constructor(options = {}) {\n super({\n className: 'cloudwatch-resource-view',\n ...options\n });\n\n this.resourceType = options.resourceType || 'ec2';\n this.slug = options.slug || '';\n this.resource = options.resource || {};\n }\n\n async getTemplate() {\n const categories = METRIC_CATEGORIES[this.resourceType] || [];\n const icon = TYPE_ICONS[this.resourceType] || 'bi-cloud';\n const typeLabel = TYPE_LABELS[this.resourceType] || 'Resource';\n\n const state = this.resource.state || this.resource.status || 'unknown';\n const metaItems = this._buildMetaItems();\n const metaHtml = metaItems.map(m => `<span class=\"me-3\" style=\"font-size: 0.78rem; color: #6c757d;\">${m}</span>`).join('');\n\n return `\n <style>\n .cwrv-header { padding: 1rem 0; border-bottom: 1px solid #e9ecef; margin-bottom: 1rem; }\n .cwrv-name { font-size: 1.15rem; font-weight: 700; }\n .cwrv-grid { display: grid; grid-template-columns: 1fr 1fr; gap: 1rem; }\n @media (max-width: 768px) { .cwrv-grid { grid-template-columns: 1fr; } }\n </style>\n\n <div class=\"cwrv-header\">\n <div class=\"cwrv-name\">\n <i class=\"bi ${icon} me-2\"></i>${this.slug}\n <span class=\"badge bg-secondary ms-2\" style=\"font-size: 0.7rem;\">${typeLabel}</span>\n </div>\n <div class=\"mt-1\">${metaHtml}</div>\n </div>\n\n <div class=\"cwrv-grid\">\n ${categories.map((_, i) => `<div id=\"cwrv-chart-${i}\"></div>`).join('')}\n </div>\n `;\n }\n\n async onInit() {\n const categories = METRIC_CATEGORIES[this.resourceType] || [];\n\n for (let i = 0; i < categories.length; i++) {\n const cat = categories[i];\n const chart = new CloudWatchChart({\n containerId: `cwrv-chart-${i}`,\n account: this.resourceType,\n category: cat.key,\n slug: this.slug,\n title: cat.label,\n height: 200,\n yAxis: yAxisForUnit(cat.unit),\n showGranularity: true,\n showDateRange: false,\n defaultDateRange: '24h',\n granularity: 'hours'\n });\n this.addChild(chart);\n }\n }\n\n _buildMetaItems() {\n const r = this.resource;\n switch (this.resourceType) {\n case 'ec2':\n return [r.instance_type, r.private_ip, r.public_ip].filter(Boolean)\n .map((v, i) => {\n const icons = ['bi-cpu', 'bi-hdd-network', 'bi-globe'];\n return `<i class=\"bi ${icons[i]} me-1\"></i>${v}`;\n });\n case 'rds':\n return [r.engine, r.instance_class].filter(Boolean)\n .map((v, i) => {\n const icons = ['bi-database', 'bi-cpu'];\n return `<i class=\"bi ${icons[i]} me-1\"></i>${v}`;\n });\n case 'redis':\n return [r.engine, r.node_type, r.num_nodes ? `${r.num_nodes} node${r.num_nodes > 1 ? 's' : ''}` : ''].filter(Boolean)\n .map((v, i) => {\n const icons = ['bi-lightning', 'bi-cpu', 'bi-diagram-3'];\n return `<i class=\"bi ${icons[i]} me-1\"></i>${v}`;\n });\n default:\n return [];\n }\n }\n\n static async show(resourceType, slug, resource = {}, options = {}) {\n const view = new CloudWatchResourceView({\n resourceType, slug, resource\n });\n\n const icon = TYPE_ICONS[resourceType] || 'bi-cloud';\n const typeLabel = TYPE_LABELS[resourceType] || 'Resource';\n\n await Dialog.showDialog(view, {\n header: `<i class=\"bi ${icon} me-2\"></i>${slug} <small class=\"text-muted\">— ${typeLabel}</small>`,\n size: 'xl',\n scrollable: true\n });\n }\n}\n","/**\n * MOJO Admin Extension - Entry (2.1.0)\n */\n\n// Bundle admin CSS\nimport '@ext/admin/css/admin.css';\n\n// Admin Pages\nexport { default as AdminDashboardPage } from '@ext/admin/account/AdminDashboardPage.js';\nexport { default as UserTablePage } from '@ext/admin/account/users/UserTablePage.js';\nexport { default as MemberTablePage } from '@ext/admin/account/users/MemberTablePage.js';\nexport { default as GroupTablePage } from '@ext/admin/account/groups/GroupTablePage.js';\nexport { default as UserDeviceTablePage } from '@ext/admin/account/devices/UserDeviceTablePage.js';\nexport { default as UserDeviceLocationTablePage } from '@ext/admin/account/devices/UserDeviceLocationTablePage.js';\nexport { default as GeoLocatedIPTablePage } from '@ext/admin/account/devices/GeoLocatedIPTablePage.js';\nexport { default as ApiKeyTablePage } from '@ext/admin/account/api_keys/ApiKeyTablePage.js';\n\nexport { default as CloudWatchDashboardPage } from '@ext/admin/aws/CloudWatchDashboardPage.js';\nexport { default as CloudWatchChart } from '@ext/admin/aws/CloudWatchChart.js';\n\nexport { default as IncidentDashboardPage } from '@ext/admin/incidents/IncidentDashboardPage.js';\nexport { default as IncidentTablePage } from '@ext/admin/incidents/IncidentTablePage.js';\nexport { default as EventTablePage } from '@ext/admin/incidents/EventTablePage.js';\nexport { default as TicketTablePage } from '@ext/admin/incidents/TicketTablePage.js';\nexport { default as RuleSetTablePage } from '@ext/admin/incidents/RuleSetTablePage.js';\n\nexport { default as EmailDomainTablePage } from '@ext/admin/messaging/email/EmailDomainTablePage.js';\nexport { default as EmailMailboxTablePage } from '@ext/admin/messaging/email/EmailMailboxTablePage.js';\nexport { default as EmailTemplateTablePage } from '@ext/admin/messaging/email/EmailTemplateTablePage.js';\nexport { default as SentMessageTablePage } from '@ext/admin/messaging/email/SentMessageTablePage.js';\nexport { default as PhoneNumberTablePage } from '@ext/admin/messaging/sms/PhoneNumberTablePage.js';\nexport { default as SMSTablePage } from '@ext/admin/messaging/sms/SMSTablePage.js';\nexport { default as PushDashboardPage } from '@ext/admin/messaging/push/PushDashboardPage.js';\nexport { default as PushConfigTablePage } from '@ext/admin/messaging/push/PushConfigTablePage.js';\nexport { default as PushTemplateTablePage } from '@ext/admin/messaging/push/PushTemplateTablePage.js';\nexport { default as PushDeliveryTablePage } from '@ext/admin/messaging/push/PushDeliveryTablePage.js';\nexport { default as PushDeviceTablePage } from '@ext/admin/messaging/push/PushDeviceTablePage.js';\n\nexport { default as JobsAdminPage } from '@ext/admin/jobs/JobsAdminPage.js';\nexport { default as TaskManagementPage } from '@ext/admin/jobs/TaskManagementPage.js';\n\nexport { default as LogTablePage } from '@ext/admin/monitoring/LogTablePage.js';\nexport { default as MetricsPermissionsTablePage } from '@ext/admin/monitoring/MetricsPermissionsTablePage.js';\n\nexport { default as SettingTablePage } from '@ext/admin/settings/SettingTablePage.js';\n\nexport { default as FileManagerTablePage } from '@ext/admin/storage/FileManagerTablePage.js';\nexport { default as FileTablePage } from '@ext/admin/storage/FileTablePage.js';\nexport { default as S3BucketTablePage } from '@ext/admin/storage/S3BucketTablePage.js';\n\n// Admin Views\nexport { default as DeviceView } from '@ext/admin/account/devices/DeviceView.js';\nexport { default as GeoIPView } from '@ext/admin/account/devices/GeoIPView.js';\nexport { default as UserDeviceLocationView } from '@ext/admin/account/devices/UserDeviceLocationView.js';\nexport { default as GroupView } from '@ext/admin/account/groups/GroupView.js';\nexport { default as ApiKeyView } from '@ext/admin/account/api_keys/ApiKeyView.js';\nexport { default as CloudWatchResourceView } from '@ext/admin/aws/CloudWatchResourceView.js';\nexport { default as MemberView } from '@ext/admin/account/users/MemberView.js';\nexport { default as UserView } from '@ext/admin/account/users/UserView.js';\n\nexport { default as IncidentView } from '@ext/admin/incidents/IncidentView.js';\nexport { default as EventView } from '@ext/admin/incidents/EventView.js';\nexport { default as TicketView } from '@ext/admin/incidents/TicketView.js';\nexport { default as RuleSetView } from '@ext/admin/incidents/RuleSetView.js';\n\nexport { default as EmailTemplateView } from '@ext/admin/messaging/email/EmailTemplateView.js';\nexport { default as EmailView } from '@ext/admin/messaging/email/EmailView.js';\nexport { default as PhoneNumberView } from '@ext/admin/messaging/sms/PhoneNumberView.js';\nexport { default as PushDeliveryView } from '@ext/admin/messaging/push/PushDeliveryView.js';\nexport { default as PushDeviceView } from '@ext/admin/messaging/push/PushDeviceView.js';\n\nexport { default as JobDetailsView } from '@ext/admin/jobs/JobDetailsView.js';\nexport { default as JobHealthView } from '@ext/admin/jobs/JobHealthView.js';\nexport { default as JobStatsView } from '@ext/admin/jobs/JobStatsView.js';\n\nexport { default as LogView } from '@ext/admin/monitoring/LogView.js';\nexport { default as MetricsPermissionsView } from '@ext/admin/monitoring/MetricsPermissionsView.js';\n\nexport { default as SettingView } from '@ext/admin/settings/SettingView.js';\n\nexport { default as FileView } from '@ext/admin/storage/FileView.js';\n\n// Admin Components\nexport { default as RunnerDetailsView } from '@ext/admin/jobs/RunnerDetailsView.js';\nexport { default as TaskDetailsView } from '@ext/admin/jobs/TaskDetailsView.js';\n// Convenience\nexport { default as WebApp } from '@core/WebApp.js';\n\n// Version info passthrough\nexport {\n VERSION_INFO,\n VERSION,\n VERSION_MAJOR,\n VERSION_MINOR,\n VERSION_REVISION,\n BUILD_TIME\n} from './version.js';\n\n\n\n// Import all admin page classes for the register function\nimport AdminDashboardPageClass from '@ext/admin/account/AdminDashboardPage.js';\nimport UserTablePageClass from '@ext/admin/account/users/UserTablePage.js';\nimport MemberTablePageClass from '@ext/admin/account/users/MemberTablePage.js';\nimport GroupTablePageClass from '@ext/admin/account/groups/GroupTablePage.js';\nimport UserDeviceTablePageClass from '@ext/admin/account/devices/UserDeviceTablePage.js';\nimport UserDeviceLocationTablePageClass from '@ext/admin/account/devices/UserDeviceLocationTablePage.js';\nimport GeoLocatedIPTablePageClass from '@ext/admin/account/devices/GeoLocatedIPTablePage.js';\nimport ApiKeyTablePageClass from '@ext/admin/account/api_keys/ApiKeyTablePage.js';\nimport CloudWatchDashboardPageClass from '@ext/admin/aws/CloudWatchDashboardPage.js';\n\nimport IncidentDashboardPageClass from '@ext/admin/incidents/IncidentDashboardPage.js';\nimport IncidentTablePageClass from '@ext/admin/incidents/IncidentTablePage.js';\nimport EventTablePageClass from '@ext/admin/incidents/EventTablePage.js';\nimport TicketTablePageClass from '@ext/admin/incidents/TicketTablePage.js';\nimport RuleSetTablePageClass from '@ext/admin/incidents/RuleSetTablePage.js';\n\nimport EmailDomainTablePageClass from '@ext/admin/messaging/email/EmailDomainTablePage.js';\nimport EmailMailboxTablePageClass from '@ext/admin/messaging/email/EmailMailboxTablePage.js';\nimport EmailTemplateTablePageClass from '@ext/admin/messaging/email/EmailTemplateTablePage.js';\nimport SentMessageTablePageClass from '@ext/admin/messaging/email/SentMessageTablePage.js';\nimport PhoneNumberTablePageClass from '@ext/admin/messaging/sms/PhoneNumberTablePage.js';\nimport SMSTablePageClass from '@ext/admin/messaging/sms/SMSTablePage.js';\nimport PushDashboardPageClass from '@ext/admin/messaging/push/PushDashboardPage.js';\nimport PushConfigTablePageClass from '@ext/admin/messaging/push/PushConfigTablePage.js';\nimport PushTemplateTablePageClass from '@ext/admin/messaging/push/PushTemplateTablePage.js';\nimport PushDeliveryTablePageClass from '@ext/admin/messaging/push/PushDeliveryTablePage.js';\nimport PushDeviceTablePageClass from '@ext/admin/messaging/push/PushDeviceTablePage.js';\n\nimport JobsAdminPageClass from '@ext/admin/jobs/JobsAdminPage.js';\nimport TaskManagementPageClass from '@ext/admin/jobs/TaskManagementPage.js';\n\nimport LogTablePageClass from '@ext/admin/monitoring/LogTablePage.js';\nimport MetricsPermissionsTablePageClass from '@ext/admin/monitoring/MetricsPermissionsTablePage.js';\n\nimport SettingTablePageClass from '@ext/admin/settings/SettingTablePage.js';\n\nimport FileManagerTablePageClass from '@ext/admin/storage/FileManagerTablePage.js';\nimport FileTablePageClass from '@ext/admin/storage/FileTablePage.js';\nimport S3BucketTablePageClass from '@ext/admin/storage/S3BucketTablePage.js';\n\n/**\n * Register all admin pages to a WebApp instance\n * @param {WebApp} app - The WebApp instance to register pages to\n * @returns {void}\n */\nexport function registerSystemPages(app, addToMenu = true) {\n // Register all admin pages with consistent naming\n app.registerPage('system/dashboard', AdminDashboardPageClass, {permissions: [\"view_admin\"]});\n app.registerPage('system/jobs', JobsAdminPageClass, {permissions: [\"view_jobs\", \"manage_jobs\"]});\n app.registerPage('system/users', UserTablePageClass, {permissions: [\"manage_users\"]});\n app.registerPage('system/groups', GroupTablePageClass, {permissions: [\"manage_groups\"]});\n app.registerPage('system/members', MemberTablePageClass, {permissions: [\"manage_members\"]});\n app.registerPage('system/s3buckets', S3BucketTablePageClass, {permissions: [\"manage_aws\"]});\n app.registerPage('system/filemanagers', FileManagerTablePageClass, {permissions: [\"manage_files\"]});\n app.registerPage('system/files', FileTablePageClass, {permissions: [\"manage_files\"]});\n app.registerPage('system/incidents', IncidentTablePageClass, {permissions: [\"view_incidents\"]});\n app.registerPage('system/events', EventTablePageClass, {permissions: [\"view_incidents\"]});\n app.registerPage('system/logs', LogTablePageClass, {permissions: [\"view_logs\"]});\n app.registerPage('system/user/devices', UserDeviceTablePageClass, {permissions: [\"manage_users\"]});\n app.registerPage('system/user/device-locations', UserDeviceLocationTablePageClass, {permissions: [\"manage_users\"]});\n app.registerPage('system/system/geoip', GeoLocatedIPTablePageClass, {permissions: [\"manage_users\"]});\n app.registerPage('system/email/mailboxes', EmailMailboxTablePageClass, {permissions: [\"manage_aws\"]});\n app.registerPage('system/email/domains', EmailDomainTablePageClass, {permissions: [\"manage_aws\"]});\n app.registerPage('system/email/sent', SentMessageTablePageClass, {permissions: [\"manage_aws\"]});\n app.registerPage('system/email/templates', EmailTemplateTablePageClass, {permissions: [\"manage_aws\"]});\n app.registerPage('system/incident-dashboard', IncidentDashboardPageClass, { permissions: [\"view_incidents\"] });\n app.registerPage('system/rulesets', RuleSetTablePageClass, { permissions: [\"manage_incidents\"] });\n app.registerPage('system/tickets', TicketTablePageClass, { permissions: [\"manage_incidents\"] });\n app.registerPage('system/metrics/permissions', MetricsPermissionsTablePageClass, { permissions: [\"manage_metrics\"] });\n app.registerPage('system/push/dashboard', PushDashboardPageClass, { permissions: [\"manage_users\"] });\n app.registerPage('system/push/configs', PushConfigTablePageClass, { permissions: [\"manage_users\"] });\n app.registerPage('system/push/templates', PushTemplateTablePageClass, { permissions: [\"manage_users\"] });\n app.registerPage('system/push/deliveries', PushDeliveryTablePageClass, { permissions: [\"manage_users\"] });\n app.registerPage('system/push/devices', PushDeviceTablePageClass, { permissions: [\"manage_users\"] });\n app.registerPage('system/phonehub/numbers', PhoneNumberTablePageClass, { permissions: [\"manage_users\"] });\n app.registerPage('system/phonehub/sms', SMSTablePageClass, { permissions: [\"manage_users\"] });\n app.registerPage('system/api-keys', ApiKeyTablePageClass, { permissions: [\"manage_groups\", \"manage_group\"] });\n app.registerPage('system/settings', SettingTablePageClass, { permissions: [\"manage_settings\"] });\n app.registerPage('system/cloudwatch', CloudWatchDashboardPageClass, { permissions: [\"view_admin\"] });\n\n // Check if sidebar exists and has an admin menu config\n if (addToMenu && app.sidebar && app.sidebar.getMenuConfig) {\n const adminMenuConfig = app.sidebar.getMenuConfig('system');\n if (adminMenuConfig && adminMenuConfig.items) {\n // Add admin pages to sidebar menu\n const adminMenuItems = [\n {\n text: 'Dashboard',\n route: '?page=system/dashboard',\n icon: 'bi-speedometer2',\n permissions: [\"view_admin\"]\n },\n {\n text: 'Jobs Management',\n route: '?page=system/jobs',\n icon: 'bi-gear-wide-connected',\n permissions: [\"view_jobs\", \"manage_jobs\"]\n },\n {\n text: 'Users',\n route: '?page=system/users',\n icon: 'bi-people',\n permissions: [\"manage_users\"]\n },\n {\n text: 'Groups',\n route: '?page=system/groups',\n icon: 'bi-diagram-3',\n permissions: [\"manage_groups\"]\n },\n {\n text: 'Incidents & Tickets',\n route: null,\n icon: 'bi-shield-exclamation',\n permissions: [\"view_incidents\"],\n children: [\n {\n text: 'Dashboard',\n route: '?page=system/incident-dashboard',\n icon: 'bi-bar-chart-line',\n permissions: [\"view_incidents\"]\n },\n {\n text: 'Incidents',\n route: '?page=system/incidents',\n icon: 'bi-exclamation-triangle',\n permissions: [\"view_incidents\"]\n },\n {\n text: 'Tickets',\n route: '?page=system/tickets',\n icon: 'bi-ticket-detailed',\n permissions: [\"manage_incidents\"]\n },\n {\n text: 'Events',\n route: '?page=system/events',\n icon: 'bi-bell',\n permissions: [\"view_incidents\"]\n },\n {\n text: 'Rule Engine',\n route: '?page=system/rulesets',\n icon: 'bi-gear-wide-connected',\n permissions: [\"manage_incidents\"]\n },\n ]\n },\n {\n text: 'Security',\n route: null,\n icon: 'bi-shield',\n permissions: [\"manage_groups\"],\n children: [\n {\n text: 'Logs',\n route: '?page=system/logs',\n icon: 'bi-journal-text',\n permissions: [\"view_logs\"]\n },\n {\n text: 'User Devices',\n route: '?page=system/user/devices',\n icon: 'bi-phone',\n permissions: [\"manage_users\"]\n },\n {\n text: 'Device Locations',\n route: '?page=system/user/device-locations',\n icon: 'bi-geo-alt',\n permissions: [\"manage_users\"]\n },\n {\n text: 'GeoIP Cache',\n route: '?page=system/system/geoip',\n icon: 'bi-globe',\n permissions: [\"manage_users\"]\n },\n {\n text: 'Metrics Permissions',\n route: '?page=system/metrics/permissions',\n icon: 'bi-bar-chart-line',\n permissions: [\"manage_metrics\"]\n },\n {\n text: 'API Keys',\n route: '?page=system/api-keys',\n icon: 'bi-key',\n permissions: [\"manage_groups\", \"manage_group\"]\n },\n {\n text: 'Settings',\n route: '?page=system/settings',\n icon: 'bi-gear',\n permissions: [\"manage_settings\"]\n }\n ]\n },\n {\n text: 'Storage',\n route: null,\n icon: 'bi-folder',\n permissions: [\"manage_files\", \"manage_aws\"],\n children: [\n {\n text: 'S3 Buckets',\n route: '?page=system/s3buckets',\n icon: 'bi-bucket',\n permissions: [\"manage_aws\"]\n },\n {\n text: 'Storage Backends',\n route: '?page=system/filemanagers',\n icon: 'bi-hdd-stack',\n permissions: [\"manage_aws\"]\n },\n {\n text: 'Files',\n route: '?page=system/files',\n icon: 'bi-file-earmark',\n permissions: [\"manage_files\"]\n },\n ]\n },\n {\n text: 'Push Notifications',\n route: null,\n icon: 'bi-broadcast',\n permissions: [\"manage_users\"],\n children: [\n { text: 'Dashboard', route: '?page=system/push/dashboard', icon: 'bi-bar-chart-line' },\n { text: 'Configurations', route: '?page=system/push/configs', icon: 'bi-gear' },\n { text: 'Templates', route: '?page=system/push/templates', icon: 'bi-file-earmark-text' },\n { text: 'Deliveries', route: '?page=system/push/deliveries', icon: 'bi-send' },\n { text: 'Devices', route: '?page=system/push/devices', icon: 'bi-phone' },\n ]\n },\n {\n text: 'Email Admin',\n route: null,\n icon: 'bi-envelope',\n permissions: [\"manage_aws\"],\n children: [\n {\n text: 'Domains',\n route: '?page=system/email/domains',\n icon: 'bi-globe',\n permissions: [\"manage_aws\"]\n },\n {\n text: 'Mailboxes',\n route: '?page=system/email/mailboxes',\n icon: 'bi-inbox',\n permissions: [\"manage_aws\"]\n },\n {\n text: 'Sent',\n route: '?page=system/email/sent',\n icon: 'bi-send-check',\n permissions: [\"manage_aws\"]\n },\n {\n text: 'Templates',\n route: '?page=system/email/templates',\n icon: 'bi-file-text',\n permissions: [\"manage_aws\"]\n }\n ]\n },\n {\n text: 'AWS',\n route: null,\n icon: 'bi-cloud',\n permissions: [\"view_admin\"],\n children: [\n {\n text: 'CloudWatch',\n route: '?page=system/cloudwatch',\n icon: 'bi-bar-chart-line',\n permissions: [\"view_admin\"]\n }\n ]\n },\n {\n text: 'Phone Hub',\n route: null,\n icon: 'bi-telephone',\n permissions: [\"manage_users\"],\n children: [\n {\n text: 'Numbers',\n route: '?page=system/phonehub/numbers',\n icon: 'bi-collection',\n permissions: [\"manage_users\"]\n },\n {\n text: 'SMS',\n route: '?page=system/phonehub/sms',\n icon: 'bi-chat-dots',\n permissions: [\"manage_users\"]\n }\n ]\n }\n ];\n\n // Add items to existing admin menu\n adminMenuConfig.items.unshift(...adminMenuItems);\n }\n }\n\n}\n\nexport { registerSystemPages as registerAdminPages };\n"],"names":["AdminHeaderView","View","constructor","options","super","title","headerActions","label","icon","action","buttonClass","className","this","stats","user_activity_day","total_users","group_activity_day","total_groups","api_calls","apiChange","incidents","incidentsChange","prepareStatsForTemplate","getTemplate","onInit","userActivity","MetricsMiniChartWidget","subtitle","background","textColor","granularity","trendRange","trendOffset","slugs","account","chartType","showTooltip","showXAxis","height","chartWidth","color","fill","fillColor","smoothing","showTrending","showSettings","showDateRange","containerId","addChild","groupActivity","apiActivity","endpoint","incidentActivity","onBeforeRender","loadValues","response","getApp","rest","GET","success","data","status","Object","assign","header","render","error","console","loadStats","AdminDashboardPage","Page","pageTitle","pageSubtitle","lastUpdated","Date","toLocaleString","headerView","apiMetricsChart","MetricsChart","yAxis","beginAtZero","tooltip","y","onActionRefreshAll","event","element","button","currentTarget","querySelector","classList","add","disabled","promises","refresh","filter","Boolean","Promise","allSettled","eventBus","events","emit","page","timestamp","alert","innerHTML","setTimeout","remove","onActionExportMetrics","export","charts","onActionViewAlerts","router","navigateTo","onActionViewSystemStatus","refreshDashboard","getCharts","apiMetrics","getStats","onAfterRender","SideNavView","sections","activeSection","navWidth","contentPadding","enableResponsive","minWidth","viewOptions","tagName","sectionConfigs","sectionViews","sectionKeys","currentMode","resizeObserver","lastContainerWidth","config","_addSectionConfig","handleResize","bind","type","permissions","_hasPermission","push","key","view","parent","perm","activeUser","hasPerm","renderTemplate","nav","_buildDropdownNav","_buildSidebarNav","map","escapeHtml","isActive","join","activeConfig","find","c","activeLabel","items","_mountSection","_setupResponsive","onBeforeDestroy","disconnect","window","removeEventListener","values","destroy","showSection","warn","isMounted","contains","previousSection","_unmountSection","_updateNavState","container","unmount","activeKey","querySelectorAll","forEach","link","section","dataset","toggle","selectBtn","textContent","onActionNavigate","el","preventDefault","_updateMode","ResizeObserver","parentElement","observe","addEventListener","containerWidth","_getContainerWidth","Math","abs","offsetWidth","newMode","mode","getActiveSection","getSectionKeys","getSection","addSection","makeActive","removeSection","k","_onModelChange","create","AdminProfileSection","template","hasPhone","model","get","roleLabel","onActionForceVerifyEmail","Dialog","confirm","_saveField","is_email_verified","onActionUnverifyEmail","onActionForceVerifyPhone","is_phone_verified","onActionUnverifyPhone","onActionChangeEmail","email","prompt","defaultValue","trim","onActionChangePhone","phone","phone_number","onActionSetPhone","placeholder","onActionRemovePhone","resp","save","set","toast","message","onActionEditUsername","username","fields","successMsg","AdminPersonalSection","hasDob","dobFormatted","dob","year","month","day","split","timezoneDisplay","timezone","hasAddress","meta","street","city","state","zip","country","addressSummary","onActionForceVerifyDob","is_dob_verified","onActionUnverifyDob","onActionEditDisplayName","name","display_name","onActionEditFirstName","first_name","onActionEditLastName","last_name","onActionEditDob","showForm","size","cols","onActionEditTimezone","value","text","metadata","onActionEditAddress","updatedMeta","toLowerCase","AdminSecuritySection","onActionSendPasswordReset","app","POST","onActionSendEmailVerification","onActionSendMagicLink","onActionSetPassword","required","help","password","onActionToggleMfa","currentMfa","requires_mfa","onActionDisableTotp","DELETE","id","onActionManagePasskeys","collection","PasskeyList","params","user","fetch","e","models","passkeys","p","toJSON","onActionEditPasskey","async","passkey","String","showModelForm","PasskeyForms","edit","onActionDeletePasskey","showDialog","body","buttons","class","dismiss","onActionViewRecoveryCodes","dataOnly","remaining","codes","onActionRevokeAllSessions","PROVIDER_ICONS","google","github","microsoft","apple","facebook","twitter","linkedin","AdminConnectedSection","connections","results","provider","onActionUnlink","connection","CHANNEL_LABELS","in_app","CHANNELS","AdminNotificationsSection","preferences","channels","ch","hasPreferences","keys","length","preferenceRows","sort","kind","kindLabel","replace","toUpperCase","toggles","channel","checked","onActionTogglePref","AdminApiKeysSection","apiKeys","_loadKeys","Array","isArray","_renderKeysList","rows","created","toLocaleDateString","expires","lastUsed","last_used","ips","allowed_ips","is_active","token_prefix","onActionRevokeKey","DeviceRow","TableRow","deviceIcon","dev","device","os","some","m","family","includes","deviceName","brand","deviceModel","browserName","ua","user_agent","major","osName","deviceMeta","LocationRow","device_info","locationText","geo","region","country_name","countryName","threatFlags","flags","is_vpn","is_tor","is_proxy","hasThreatFlags","UserView","User","sideNavView","setModel","profileView","personalView","securityView","connectedView","notificationsView","apiKeysView","permsView","FormView","PERMISSION_FIELDS","autosaveModelField","membersCollection","MemberList","groupsView","TableView","hideActivePillNames","columns","formatter","sortable","eventsCollection","IncidentEventList","model_name","model_id","eventsView","width","devicesView","UserDeviceList","clickAction","itemClass","locationsView","UserDeviceLocationList","pushDevices","PushDeviceList","pushDevicesView","logsCollection","LogList","logsView","startName","endName","fieldName","format","displayFormat","separator","activityCollection","uid","activityView","userMenu","ContextMenu","context","onActionEditUser","modelName","formConfig","UserForms","onActionClearAvatar","avatar","onActionToggleActive","onActionDeactivateUser","onActionActivateUser","sectionName","showTab","tabName","getActiveTab","VIEW_CLASS","UserTablePage","TablePage","pageName","Collection","UserList","viewDialogOptions","defaultQuery","visibility","filters","selectable","searchable","filterable","paginated","showRefresh","showAdd","showExport","emptyMessage","contextMenu","tableOptions","striped","bordered","hover","responsive","onActionEditPermissions","item","_","onActionChangePassword","attributes","autocomplete","readonly","tabindex","style","passwordUsage","showToggle","new_password","MOJOUtils","checkPasswordStrength","score","onPasswordChange","onActionSendInvite","send_invite","MemberView","Member","detailsView","permissionsView","memberMenu","danger","onActionEditMembership","Modal","modelForm","MemberForms","onActionViewUser","userId","import","then","n","q","showModelById","onActionViewGroup","groupId","Group","onActionDeactivateMember","onActionActivateMember","onActionRemoveMember","MemberTablePage","formEdit","itemViewClass","batchBarLocation","batchActions","GroupView","membersView","group","addButtonLabel","onAdd","onInviteClick","childrenView","GroupList","onActionAddChildGroup","groupMenu","onActionEditGroup","GroupForms","detailed","onActionInviteMember","Event","stopPropagation","f","newGroup","onActionDeactivateGroup","onActionActivateGroup","onActionViewParent","parentId","GroupTablePage","formCreate","GroupKindOptions","onActionMakeActive","setActiveGroup","DeviceLocationRow","ispName","isp","asn_org","DeviceView","UserDevice","deviceInfo","_getIcon","browserFull","_getBrowser","osFull","_getOS","deviceFull","_getDevice","isMobile","_isMobile","parts","ver","minor","browser","user_device","deviceMenu","onActionDeleteDevice","show","duid","getByDuid","UserDeviceTablePage","UserDeviceLocationTablePage","GeoIPView","GeoLocatedIP","hasCoordinates","DataView","showEmptyValues","emptyValueText","networkView","riskView","metadataView","source_ip","ip","tabs","Location","Network","Events","Logs","Metadata","lat","lng","locationStr","mapView","MapView","markers","popup","tileLayer","zoom","tabView","TabView","activeTab","menuItems","geoIPMenu","bootstrap","Tooltip","existing","getInstance","dispose","onActionEditLocation","EDIT_LOCATION_FORM","onActionEditSecurity","EDIT_SECURITY_FORM","onActionEditNetwork","EDIT_NETWORK_FORM","onActionRefreshGeoip","info","onActionThreatAnalysis","threat_analysis","onActionViewOnMap","url","open","onActionDeleteGeoip","confirmClass","confirmText","lookup","dialog","document","GeoLocatedIPTablePage","GeoLocatedIPList","itemView","tableViewOptions","evt","onLookup","tableView","_onRowView","ApiKey","Model","ApiKeyList","ModelClass","ApiKeyForms","ApiKeyView","apiKeyMenu","onActionEditKey","onActionDeactivateKey","confirmLabel","showLoading","hideLoading","onActionActivateKey","onActionDeleteKey","delete","ADD_FORM","EDIT_FORM","ApiKeyTablePage","onActionAdd","result","showError","token","showAlert","CloudWatchChart","resourceType","category","slug","defaultDateRange","stat","buildApiParams","setStat","fetchData","DASHBOARD_CHARTS","unit","yAxisForUnit","max","CloudWatchDashboardPage","i","def","chart","showGranularity","IncidentDashboardHeader","IncidentStats","IncidentDashboardPage","eventsWidget","valueFormat","settingsKey","incidentsWidget","eventsByCountryChart","showMetricsFilter","maxDatasets","colors","incidentsByCountryChart","eventsCountryMap","MetricsCountryMapView","maxCountries","metricLabel","mapStyle","myTicketsCollection","TicketList","myTicketsTable","newIncidentsCollection","IncidentList","highPriorityIncidentsTable","refreshTasks","StackTraceView","stackTrace","formattedStackTrace","formatStackTrace","lines","JSON","stringify","html","line","index","match","funcName","filePath","lineNum","colNum","startsWith","updateStackTrace","newStackTrace","IncidentHistoryAdapter","incidentId","IncidentHistoryList","incident","transform","author","avatarUrl","content","attachments","addNote","history","IncidentHistory","note","IncidentView","Incident","incidentIcon","getIconForIncident","s","overviewView","actions","adapter","historyView","ChatView","Overview","stack_trace","stackTraceView","incidentMenu","onActionEditIncident","IncidentForms","onActionResolveIncident","onActionDeleteIncident","IncidentTablePage","onActionBatchResolve","selected","getSelectedItems","all","onActionBatchOpen","onActionBatchPause","onActionBatchIgnore","onActionBatchMerge","_event","_element","parentModel","merge","mergeIds","EventView","IncidentEvent","eventIcon","getIconForEvent","level","eventMenu","onActionViewIncident","onActionViewModel","onActionDeleteEvent","EventTablePage","IncidentEventForms","category__not","TicketNoteAdapter","ticketId","TicketNoteList","TicketNote","media","files","TicketView","Ticket","chatView","theme","currentUserId","getCurrentUserId","inputPlaceholder","inputButtonText","ticketMenu","currentUser","onActionEditTicket","TicketForms","onActionChangeStatus","currentStatus","onActionSetPriority","currentPriority","priority","onActionAssignUser","onActionCloseTicket","TicketTablePage","editable","editableOptions","placeHolder","TicketCategories","RuleSetView","RuleSet","matchByValue","matchByOption","MatchByOptions","opt","matchByLabel","bundleByValue","bundleByOption","BundleByOptions","bundleByLabel","configView","rulesCollection","RuleList","rulesView","divider","addFormDefaults","Configuration","Rules","onActionEditRuleset","onActionDisableRuleset","newStatus","showToast","onActionDeleteRuleset","closest","bsModal","hide","RuleSetTablePage","RuleSetList","v","pageSizes","defaultPageSize","emptyIcon","EmailDomainTablePage","EmailDomainList","EmailDomainForms","onActionEditAwsCreds","credentials","onActionOnboard","EmailDomain","formData","onboard","Error","err","onActionAudit","audit","onActionReconcile","reconcile","EmailMailboxTablePage","MailboxList","MailboxForms","onActionSendEmail","from_email","Mailbox","sendEmail","msg","details","onActionSendTemplateEmail","EmailHtmlPreviewView","renderHtmlInIframe","iframe","htmlContent","iframeDoc","contentDocument","contentWindow","write","close","onActionRefreshPreview","EmailTemplateView","EmailTemplate","hasHtml","hasText","hasMetadata","EmailTemplateTablePage","EmailTemplateList","EmailTemplateForms","scrollable","formDialogConfig","EmailView","SentMessage","hasContext","SentMessageTablePage","SentMessageList","PhoneNumber","normalize","phoneNumber","countryCode","payload","country_code","PhoneNumberList","SMS","send","SMSList","PhoneNumberView","carrierView","addressView","Carrier","Address","ctxMenu","onActionRefreshLookup","number","force_refresh","warning","onActionDeletePhone","MODEL_CLASS","PhoneNumberTablePage","addButtonIcon","SMSView","messageView","deliveryView","Message","Delivery","onActionRefreshSms","onActionDeleteSms","SMSTablePage","PushDashboardPage","deliveriesChart","statusChart","PieChart","recentDeliveries","PushDeliveryList","_sort","_limit","failedDeliveries","PushConfigTablePage","PushConfigList","PushConfigForms","PushTemplateTablePage","PushTemplateList","PushTemplateForms","PushDeliveryView","PushDeliveryTablePage","PushDeviceView","dataView","PushDeviceTablePage","JobStatsView","pending","running","completed","failed","scheduled","totals","JobHealthView","health","runners","active","total","loadHealth","overall_status","runners_active","scheduler","healthStatusClass","getHealthStatusClass","setupChannelDisplay","setupSchedulerDisplay","channelsArray","channelStatus","totalJobs","queued_count","inflight_count","statusBadgeClass","getChannelBadgeClass","statusIcon","getChannelIcon","queued","inflight","schedulerStatusText","schedulerStatusClass","schedulerIcon","healthy","critical","unknown","onActionRefreshHealth","onActionSystemSettings","JobDetailsView","Job","payloadView","autoRefreshInterval","JobEventList","job","JobLogList","Payload","contextMenuItems","canCancel","canRetry","jobMenu","graph","prepareJobData","getStatusBadgeClass","getStatusIcon","formattedDuration","getFormattedDuration","loadJobDetails","getDetailedStatus","onActionRefreshJob","onActionCancelJob","cancel","onActionRetryJob","retryData","JobForms","retry","delay","newJobId","startAutoRefresh","clearInterval","setInterval","stopAutoRefresh","onDestroy","onHide","JobMetricsModalView","JobsAdminPage","refreshRate","jobStats","JobsEngineStats","jobStatsView","jobHealthView","jobsPublishedChart","jobsFailedChart","runningJobsTable","JobList","collectionParams","hideActivePills","row","pendingJobsTable","run_at__isnull","on","failedJobsTable","scheduledJobsTable","runnersTable","JobRunnerList","isAlive","heartbeatTime","diffMs","diffSeconds","floor","refreshData","tasks","updateHeaderTimestamp","timestampElement","rateLabel","refreshRateSeconds","refreshRateLabel","onActionSetRefreshRate","rate","parseInt","getAttribute","rateText","onActionExportData","onActionRunSimpleJob","showConfirm","executeJobAction","test","onActionRunTestJobs","tests","onActionClearStuck","channelOptions","clearStuck","count","onActionClearChannel","clearChannel","onActionPurgeJobs","purgeJobs","days_old","onActionCleanupConsumers","cleanConsumers","onActionRunnerBroadcast","JobRunnerForms","broadcast","broadcastResult","JobRunner","command","timeout","actionFn","successMessage","onEnter","onExit","getHealth","onActionOpenJobMetricsModal","modalView","onActionViewAllJobs","jobList","fetchOnMount","TaskDetailsView","task","logs","metrics","prepareTaskData","loadTaskLogs","loadTaskMetrics","dataFormatted","expiresClass","now","cancelled","expired","getLogLevelClass","debug","log","levelClass","setTask","onActionRefreshLogs","showSuccess","originalTask","newTask","exportData","exported_at","toISOString","exported_by","blob","Blob","URL","createObjectURL","a","createElement","href","download","appendChild","click","removeChild","revokeObjectURL","formatBytes","bytes","toFixed","resourceBadgeClass","pct","progressBarClass","RunnerOverviewTab","r","aliveBadgeClass","aliveIcon","aliveText","started","startedText","sec","getTime","uptimeText","seconds","d","h","formatUptime","ageSec","isoString","heartbeatText","heartbeatAgeText","round","heartbeatClass","jobsProcessed","jobsFailed","errorRate","RunnerSysinfoTab","sysinfo","sysinfoError","loading","loaded","onTabActivated","loadSysinfo","enrichSysinfo","memory","totalFmt","usedFmt","used","availableFmt","available","barClass","percent","badgeClass","disk","freeFmt","free","network","bytesRecvFmt","bytes_recv","bytesSentFmt","bytes_sent","errClass","errin","errout","dropClass","dropin","dropout","cpuPct","cpu_load","cpuLoadBarClass","cpuLoadBadgeClass","cpu","freq","freqText","current","cpus_load","cpuCores","bootDatetime","boot_time","collectedText","datetime","onActionRefreshSysinfo","RunnerJobsTab","jobs","loadJobs","durationText","started_at","toLocaleTimeString","attemptBadgeClass","attempt","onActionRefreshJobs","onActionViewJob","jobId","runner","cancel_request","RunnerLogsTab","filteredLogs","logFilter","filterAllClass","filterDebugClass","filterInfoClass","filterWarnClass","filterErrorClass","loadLogs","jobsResp","jobIds","j","slice","catch","concat","b","levelBadgeClass","kindDisplay","createdText","l","onActionFilterLogs","RunnerActionsTab","pingResult","onActionPing","runner_id","onActionShutdown","graceful","onActionBroadcast","commandEl","timeoutEl","parseFloat","showBusy","hideBusy","showCode","onActionExport","RunnerDetailsView","Actions","TaskStatsView","errors","TaskRunnersView","loadRunners","pingAge","ping_age","statusBadge","pingAgeText","formatPingAge","onActionRefreshRunners","onActionViewRunnerDetails","runnerId","onActionPauseRunner","hostname","onActionRestartRunner","onActionRemoveRunner","confirmMessage","TaskChartsView","taskFlowChart","taskErrorsChart","PendingTasksTable","RunningTasksTable","CompletedTasksTable","ErrorTasksTable","TaskManagementPage","taskStatsView","taskRunnersView","taskChartsView","taskTablesView","Pending","Running","Completed","Errors","activeTable","getTab","onActionExportTasks","exportToCSV","onActionManageChannels","channelList","onActionViewSystemLogs","logPreview","getRunners","taskFlow","taskErrors","LogView","Log","logIcon","getIconForLog","lvl","logContent","formattedLog","parsed","parse","logContentView","onActionCopyLog","navigator","clipboard","writeText","Details","logMenu","onActionViewIp","onActionViewDevice","onActionDeleteLog","LogTablePage","MetricsPermissionsView","MetricsPermission","onActionEdit","MetricsForms","onActionDelete","MetricsPermissionsTablePage","MetricsPermissionList","Setting","SettingList","SettingForms","labelField","valueField","maxItems","emptyFetch","debounceMs","SettingView","settingMenu","onActionEditSetting","onActionDeleteSetting","SettingTablePage","FileManagerTablePage","FileManagerList","FileManagerForms","onActionEditOwners","owners","onActionCheckCors","check_cors","showData","onActionTestConnection","test_connection","onActionEditCredentials","onActionClone","clone","FileView","File","isImage","renditionsData","renditionsCollection","infoView","renditionsView","Info","fileMenu","onActionViewFile","contentType","fileUrl","renditions","images","src","alt","role","LightboxGallery","fitToScreen","PDFViewer","onActionDownloadFile","onActionEditFile","FileForms","onActionMakePublic","is_public","onActionMakePrivate","onActionDeleteFile","FileTablePage","FileList","handleFileUpload","enableFileDrop","acceptedTypes","maxFileSize","multiple","validateOnDrop","fileInput","accept","display","file","target","maxSize","_formatFileSize","fileModel","extra","requiresGroup","activeGroup","upload","description","onProgress","progressInfo","percentage","onComplete","onError","pow","onFileDrop","validation","applyFileDropMixin","S3BucketTablePage","S3BucketList","S3BucketForms","UserDeviceLocationView","UserDeviceLocation","_ud","_di","_geo","_getDeviceIcon","locationSummary","_getLocationSummary","countryFlag","threatLevel","threat_level","threatColor","_getThreatColor","latitude","longitude","is_threat","is_suspicious","di","postal_code","asn","patch","string","risk_score","_riskRow","is_cloud","is_datacenter","is_mobile","is_known_attacker","is_known_abuser","default","menu","deviceId","onActionOpenInMaps","onActionDeleteRecord","METRIC_CATEGORIES","ec2","rds","redis","TYPE_ICONS","TYPE_LABELS","CloudWatchResourceView","resource","categories","typeLabel","metaHtml","_buildMetaItems","cat","instance_type","private_ip","public_ip","engine","instance_class","node_type","num_nodes","registerSystemPages","addToMenu","registerPage","AdminDashboardPageClass","JobsAdminPageClass","UserTablePageClass","GroupTablePageClass","MemberTablePageClass","S3BucketTablePageClass","FileManagerTablePageClass","FileTablePageClass","IncidentTablePageClass","EventTablePageClass","LogTablePageClass","UserDeviceTablePageClass","UserDeviceLocationTablePageClass","GeoLocatedIPTablePageClass","EmailMailboxTablePageClass","EmailDomainTablePageClass","SentMessageTablePageClass","EmailTemplateTablePageClass","IncidentDashboardPageClass","RuleSetTablePageClass","TicketTablePageClass","MetricsPermissionsTablePageClass","PushDashboardPageClass","PushConfigTablePageClass","PushTemplateTablePageClass","PushDeliveryTablePageClass","PushDeviceTablePageClass","PhoneNumberTablePageClass","SMSTablePageClass","ApiKeyTablePageClass","SettingTablePageClass","CloudWatchDashboardPageClass","sidebar","getMenuConfig","adminMenuConfig","adminMenuItems","route","children","unshift"],"mappings":"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"}