Compare commits

..

40 Commits

Author SHA1 Message Date
jkunz 8fa78bf40f v2.12.4 2026-04-13 22:48:14 +00:00
jkunz 6cb99c0ab6 fix(deps): bump @design.estate/dees-catalog to ^3.70.0 2026-04-13 22:48:14 +00:00
jkunz 7b9ddb2dc5 v2.12.3 2026-04-07 22:45:27 +00:00
jkunz 00ded5d12e fix(route-card): align route card with dees-tile layout and update header and footer styling 2026-04-07 22:45:27 +00:00
jkunz 23fb25dcd5 v2.12.2 2026-04-07 22:41:35 +00:00
jkunz 8ddcb56f90 fix(ts_web): adjust route card section background color in dark theme 2026-04-07 22:41:35 +00:00
jkunz 9d186ece0a v2.12.1 2026-04-07 22:34:02 +00:00
jkunz 55cb4880ce fix(ts_web): handle slotted config section content visibility and field row borders correctly 2026-04-07 22:34:02 +00:00
jkunz 171aae7095 v2.12.0 2026-04-07 22:27:23 +00:00
jkunz 9aad9c0c2a feat(elements): standardize dashboard and detail views on dees tile and stats grid components 2026-04-07 22:27:23 +00:00
jkunz b91d3a9341 update 2026-04-07 21:55:54 +00:00
jkunz 590e4bb039 v2.11.2 2026-04-05 02:37:20 +00:00
jkunz c4bef0d443 fix(route-card): align route card with source profile metadata and vpnOnly route configuration 2026-04-05 02:37:20 +00:00
jkunz 7f05b3b52d v2.11.1 2026-04-04 20:47:05 +00:00
jkunz 1fc1de6691 fix(route-card): clarify VPN mode badge labels in route cards 2026-04-04 20:47:05 +00:00
jkunz 70c43ceae3 v2.11.0 2026-04-02 21:16:01 +00:00
jkunz cf247117d4 feat(route-ui): add VPN details and conditional card actions to route cards 2026-04-02 21:16:01 +00:00
jkunz 35ffde9253 v2.10.0 2026-04-02 18:48:22 +00:00
jkunz 531ec5a2d0 feat(docs): document newly available catalog components and updated build configuration details 2026-04-02 18:48:22 +00:00
jkunz cde6ccd841 v2.9.1 2026-04-02 17:15:47 +00:00
jkunz f181f662d8 fix(build): migrate build configuration to .smartconfig and update toolchain dependencies 2026-04-02 17:15:47 +00:00
jkunz 85f26c69d6 feat(sz-route-card): add Linked section for security profile and network target refs 2026-04-02 16:45:08 +00:00
jkunz 9d1471a363 v2.9.0 2026-03-19 16:01:34 +00:00
jkunz 6db2e3ff4f feat(app-store-view): add app details action to store cards 2026-03-19 16:01:34 +00:00
jkunz bbbc0958f4 v2.8.0 2026-03-18 02:34:14 +00:00
jkunz 52c9c7251e feat(elements): add app store view component for browsing and deploying app templates 2026-03-18 02:34:14 +00:00
jkunz a1409a4d57 v2.7.0 2026-03-17 19:29:47 +00:00
jkunz 61359bc712 feat(sz-service-detail-view): replace the custom logs panel with dees-chart-log in the service detail view 2026-03-17 19:29:47 +00:00
jkunz 889f84d666 v2.6.2 2026-03-16 14:53:08 +00:00
jkunz a188fcbe85 fix(platform-service-detail-view): wrap service logs chart in a full-width container to preserve layout 2026-03-16 14:53:08 +00:00
jkunz 53b730914c v2.6.1 2026-03-16 14:19:49 +00:00
jkunz 51b4312cc0 fix(platform-service-detail-view): replace custom service log markup with dees-chart-log in the platform service detail view 2026-03-16 14:19:49 +00:00
jkunz 668839887f v2.6.0 2026-03-16 11:44:44 +00:00
jkunz 567551b544 feat(service-create-view): add platform service toggles for MongoDB, S3, and ClickHouse provisioning 2026-03-16 11:44:44 +00:00
jkunz d5c265860c v2.5.0 2026-02-23 21:23:35 +00:00
jkunz 0f6bfe45aa feat(sz-config-section): add header action buttons to sz-config-section allowing configurable actions/events 2026-02-23 21:23:35 +00:00
jkunz 33a97b410e v2.4.0 2026-02-23 21:07:26 +00:00
jkunz bd6bce04c9 feat(elements): add configuration overview and section components with demo view and index exports 2026-02-23 21:07:26 +00:00
jkunz 204253f78c v2.3.0 2026-02-22 01:05:56 +00:00
jkunz d1c19389d7 feat(routes): add route UI components and demo view with list/card and app-shell integration 2026-02-22 01:05:56 +00:00
30 changed files with 6307 additions and 2793 deletions
+11
View File
@@ -2,6 +2,17 @@
"@git.zone/tswatch": { "@git.zone/tswatch": {
"preset": "element" "preset": "element"
}, },
"@git.zone/tsbundle": {
"bundles": [
{
"from": "./ts_web/index.ts",
"to": "./dist_bundle/bundle.js",
"outputMode": "bundle",
"bundler": "esbuild",
"production": true
}
]
},
"@git.zone/cli": { "@git.zone/cli": {
"projectType": "wcc", "projectType": "wcc",
"module": { "module": {
+129
View File
@@ -1,5 +1,134 @@
# Changelog # Changelog
## 2026-04-13 - 2.12.4 - fix(deps)
bump @design.estate/dees-catalog to ^3.70.0
- Updates @design.estate/dees-catalog from ^3.67.1 to ^3.70.0 in package.json
## 2026-04-07 - 2.12.3 - fix(route-card)
align route card with dees-tile layout and update header and footer styling
- replace the custom card wrapper with dees-tile and move content into header, body, and footer slots
- restyle header, route name, and action buttons to match shared tile design tokens and interaction states
- wrap empty and populated route states consistently inside the tile component
## 2026-04-07 - 2.12.2 - fix(ts_web)
adjust route card section background color in dark theme
- Updates the dark theme background for route card sections from #0a0a0a to #101010 for improved visual consistency.
## 2026-04-07 - 2.12.1 - fix(ts_web)
handle slotted config section content visibility and field row borders correctly
- track whether the default slot has assigned content and hide the slot container when empty
- wrap rendered fields in a dedicated list so the last field row border is removed correctly when slot content is present
## 2026-04-07 - 2.12.0 - feat(elements)
standardize dashboard and detail views on dees tile and stats grid components
- replace custom stat cards with dees-statsgrid across status and network views and remove the obsolete sz-stat-card element
- migrate multiple detail and settings views to dees-tile headers and footers for consistent action placement and styling
- extend sz-config-section with footer links and actions plus improved collapsed tile behavior
## 2026-04-05 - 2.11.2 - fix(route-card)
align route card with source profile metadata and vpnOnly route configuration
- rename linked route metadata fields from security profile to source profile in rendering and feature detection
- simplify VPN display logic to use the boolean vpnOnly flag instead of the previous nested VPN configuration object
## 2026-04-04 - 2.11.1 - fix(route-card)
clarify VPN mode badge labels in route cards
- Renames VPN badge text from "VPN Only"/"VPN + Public" to "VPN Mandatory"/"VPN Voluntary" for clearer route mode descriptions.
## 2026-04-02 - 2.11.0 - feat(route-ui)
add VPN details and conditional card actions to route cards
- Extend route card data and rendering to display VPN access mode and allowed client tags.
- Add optional Edit and Delete action buttons that emit route-edit and route-delete events.
- Allow the route list view to control action visibility per route via a showActionsFilter callback.
- Include VPN as a visible route feature indicator in the card summary.
## 2026-04-02 - 2.10.0 - feat(docs)
document newly available catalog components and updated build configuration details
- Update component counts and add documentation for App Store, Routes, MTA/Email, and Configuration views
- Expand the README with new component tables and exported TypeScript types
- Refresh build notes to reference .smartconfig.json and the renamed config file
## 2026-04-02 - 2.9.1 - fix(build)
migrate build configuration to .smartconfig and update toolchain dependencies
- replace npmextra.json with .smartconfig.json and add tsbundle bundle configuration
- simplify the build script to use the centralized tsbundle configuration
- update build and UI package dependencies and switch TypeScript to explicit node types
## 2026-03-19 - 2.9.0 - feat(app-store-view)
add app details action to store cards
- adds a new "View Details" button alongside the existing deploy action
- dispatches a bubbling "view-app" event with the selected app payload for parent handlers
- updates card action layout and button styling to support multiple actions
## 2026-03-18 - 2.8.0 - feat(elements)
add app store view component for browsing and deploying app templates
- introduces a new sz-app-store-view element with app card rendering, search/filter empty states, and deploy action events
- exports the new app store view from the elements index for public consumption
## 2026-03-17 - 2.7.0 - feat(sz-service-detail-view)
replace the custom logs panel with dees-chart-log in the service detail view
- Removes the bespoke log streaming UI styles and markup in favor of a shared log chart component.
- Maps service log entries to structured timestamp, level, and message data for the new component.
- Enables auto-scrolling, metrics display, and a higher log entry limit in the embedded log viewer.
## 2026-03-16 - 2.6.2 - fix(platform-service-detail-view)
wrap service logs chart in a full-width container to preserve layout
- Places the logs component inside a container spanning all grid columns.
- Keeps the service logs view aligned correctly within the detail page layout.
## 2026-03-16 - 2.6.1 - fix(platform-service-detail-view)
replace custom service log markup with dees-chart-log in the platform service detail view
- Removes bespoke log container styles and rendering logic in favor of the shared dees-chart-log component
- Normalizes log timestamps to ISO format before passing entries to the chart component
- Enables log auto-scroll, entry limits, and metrics display for service logs
## 2026-03-16 - 2.6.0 - feat(service-create-view)
add platform service toggles for MongoDB, S3, and ClickHouse provisioning
- Adds a new Platform Services section to the service creation view with dedicated toggles for managed infrastructure dependencies.
- Includes MongoDB, S3-compatible storage, and ClickHouse selections in the emitted create-service configuration payload.
- Resets selected platform services after form submission to keep create flow state consistent.
## 2026-02-23 - 2.5.0 - feat(sz-config-section)
add header action buttons to sz-config-section allowing configurable actions/events
- Introduce IConfigSectionAction interface (label, icon, event, detail).
- Add actions property to SzConfigSection and render header action buttons in the component template.
- Add styles for .header-action and hover state to match design system.
- Dispatch CustomEvent when an action is clicked, using action.event (defaults to 'action') and action.detail.
- Update demo (sz-demo-view-config) to include a sample 'View Routes' action showing usage.
## 2026-02-23 - 2.4.0 - feat(elements)
add configuration overview and section components with demo view and index exports
- Adds new sz-config-section component (IConfigField interface, rich renderers for boolean, pills, badge, code, link and 'Not configured' handling).
- Adds new sz-config-overview wrapper component with heading/info banner and slot styling.
- Adds demo view sz-demo-view-config that supplies example configuration groups and fields for System, Proxy, Email, DNS, TLS, Cache, RADIUS and Remote Ingress.
- Exports new components from ts_web/elements/index.ts so they are available to the element registry.
## 2026-02-22 - 2.3.0 - feat(routes)
add route UI components and demo view with list/card and app-shell integration
- Add new route UI components: sz-route-card, sz-route-list-view, and sz-demo-view-routes under ts_web/elements
- Export new components from ts_web/elements/index.ts and register demo view in the demo app shell menu (ts_web/pages/sz-demo-app-shell.ts)
- sz-route-card introduces route types/interfaces (IRouteConfig, IRouteAction, IRouteMatch, IRouteTls, IRouteSecurity) and rich rendering (ports/domains formatting, feature icons, security and headers display)
- sz-route-list-view provides demo data, search, filtering (by action and enabled state), results count, grid rendering of sz-route-card, and emits a 'route-click' event
- Demo view integrates with app UI secondary menu (actions and statistics) and wires up route-click handling for interactivity
## 2026-02-21 - 2.2.0 - feat(demo-mta) ## 2026-02-21 - 2.2.0 - feat(demo-mta)
add MTA / Email demo views and components and integrate into demo app shell add MTA / Email demo views and components and integrate into demo app shell
+21
View File
@@ -0,0 +1,21 @@
MIT License
Copyright (c) 2024 Task Venture Capital GmbH
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.
+13 -14
View File
@@ -1,6 +1,6 @@
{ {
"name": "@serve.zone/catalog", "name": "@serve.zone/catalog",
"version": "2.2.0", "version": "2.12.4",
"private": false, "private": false,
"description": "UI component catalog for serve.zone", "description": "UI component catalog for serve.zone",
"main": "dist_ts_web/index.js", "main": "dist_ts_web/index.js",
@@ -8,35 +8,34 @@
"type": "module", "type": "module",
"scripts": { "scripts": {
"test": "tstest test/", "test": "tstest test/",
"build": "tsbuild tsfolders --allowimplicitany && tsbundle element --production", "build": "tsbuild tsfolders --allowimplicitany && tsbundle",
"watch": "tswatch" "watch": "tswatch"
}, },
"author": "Lossless GmbH", "author": "Lossless GmbH",
"license": "MIT", "license": "MIT",
"dependencies": { "dependencies": {
"@design.estate/dees-catalog": "^3.43.0", "@design.estate/dees-catalog": "^3.70.0",
"@design.estate/dees-domtools": "^2.3.8", "@design.estate/dees-domtools": "^2.5.4",
"@design.estate/dees-element": "^2.1.6", "@design.estate/dees-element": "^2.2.4",
"@design.estate/dees-wcctools": "^3.8.0" "@design.estate/dees-wcctools": "^3.8.0"
}, },
"devDependencies": { "devDependencies": {
"@git.zone/tsbuild": "^4.1.2", "@git.zone/tsbuild": "^4.4.0",
"@git.zone/tsbundle": "^2.8.3", "@git.zone/tsbundle": "^2.10.0",
"@git.zone/tsrun": "^2.0.1", "@git.zone/tsrun": "^2.0.2",
"@git.zone/tstest": "^3.1.8", "@git.zone/tstest": "^3.6.3",
"@git.zone/tswatch": "^3.1.0", "@git.zone/tswatch": "^3.3.2",
"@push.rocks/projectinfo": "^5.0.2", "@push.rocks/projectinfo": "^5.1.0",
"@types/node": "^25.3.0" "@types/node": "^25.5.2"
}, },
"files": [ "files": [
"ts/**/*",
"ts_web/**/*", "ts_web/**/*",
"dist/**/*", "dist/**/*",
"dist_*/**/*", "dist_*/**/*",
"dist_ts/**/*", "dist_ts/**/*",
"dist_ts_web/**/*", "dist_ts_web/**/*",
"assets/**/*", "assets/**/*",
"npmextra.json", ".smartconfig.json",
"readme.md" "readme.md"
], ],
"browserslist": [ "browserslist": [
+1727 -1693
View File
File diff suppressed because it is too large Load Diff
+8 -3
View File
@@ -2,7 +2,7 @@
## Project Structure ## Project Structure
- `html/index.ts` - WccTools setup with sections for Pages and Elements - `html/index.ts` - WccTools setup with sections for Pages and Elements
- `ts_web/elements/` - All web components (27 elements + 6 demo-view wrappers) - `ts_web/elements/` - All web components (33 elements + 9 demo-view wrappers)
- `ts_web/elements/index.ts` - Barrel export for all element components - `ts_web/elements/index.ts` - Barrel export for all element components
- `ts_web/pages/` - Page components - `ts_web/pages/` - Page components
@@ -16,13 +16,18 @@
## Demo Groups ## Demo Groups
| Group | Elements | | Group | Elements |
|-------|----------| |-------|----------|
| Dashboard | sz-dashboard-view, sz-stat-card, sz-resource-usage-card, sz-traffic-card, sz-quick-actions-card | | Dashboard | sz-dashboard-view, sz-resource-usage-card, sz-traffic-card, sz-quick-actions-card |
| Dashboard Grids | sz-status-grid-cluster, sz-status-grid-services, sz-status-grid-network, sz-status-grid-infra | | Dashboard Grids | sz-status-grid-cluster, sz-status-grid-services, sz-status-grid-network, sz-status-grid-infra |
| Platform | sz-platform-services-card, sz-platform-service-detail-view | | Platform | sz-platform-services-card, sz-platform-service-detail-view |
| Network | sz-network-proxy-view, sz-network-dns-view, sz-network-domains-view, sz-reverse-proxy-card, sz-dns-ssl-card, sz-certificates-card, sz-domain-detail-view | | Network | sz-network-proxy-view, sz-network-dns-view, sz-network-domains-view, sz-reverse-proxy-card, sz-dns-ssl-card, sz-certificates-card, sz-domain-detail-view |
| Routes | sz-route-list-view, sz-route-card |
| Services | sz-services-list-view, sz-services-backups-view, sz-service-detail-view, sz-service-create-view | | Services | sz-services-list-view, sz-services-backups-view, sz-service-detail-view, sz-service-create-view |
| App Store | sz-app-store-view |
| MTA / Email | sz-mta-list-view, sz-mta-detail-view |
| Configuration | sz-config-overview, sz-config-section |
| Auth & Settings | sz-login-view, sz-tokens-view, sz-settings-view, sz-registry-advertisement, sz-registry-external-view | | Auth & Settings | sz-login-view, sz-tokens-view, sz-settings-view, sz-registry-advertisement, sz-registry-external-view |
## Build ## Build
- `pnpm run build` - tsbuild tsfolders + tsbundle element --production - `pnpm run build` - tsbuild tsfolders + tsbundle (reads from .smartconfig.json)
- `pnpm run watch` - starts wcctools dev server - `pnpm run watch` - starts wcctools dev server
- Config file: `.smartconfig.json` (renamed from npmextra.json)
+45 -2
View File
@@ -14,15 +14,19 @@ For reporting bugs, issues, or security vulnerabilities, please visit [community
## 🚀 What It Does ## 🚀 What It Does
`@serve.zone/catalog` provides **30+ production-ready web components** covering every aspect of server management: `@serve.zone/catalog` provides **34 production-ready web components** covering every aspect of server management:
- 📊 **Dashboard** — Real-time cluster overview, resource usage, traffic metrics, quick actions - 📊 **Dashboard** — Real-time cluster overview, resource usage, traffic metrics, quick actions
- 🐳 **Services** — Docker container management, deployment, logs, live stats, backups, and an integrated IDE workspace - 🐳 **Services** — Docker container management, deployment, logs, live stats, backups, and an integrated IDE workspace
- 🛒 **App Store** — Browse and deploy pre-configured application templates (WordPress, Gitea, etc.)
- 🌐 **Network** — Reverse proxy configuration, DNS record management, domain & SSL certificate monitoring - 🌐 **Network** — Reverse proxy configuration, DNS record management, domain & SSL certificate monitoring
- 🔀 **Routes** — SmartProxy route configuration, match criteria, TLS modes, security profiles, forwarding targets
- 📧 **MTA / Email** — Inbound and outbound email management, SMTP transaction logs, authentication results
- 📦 **Registries** — Container registry management (onebox + external registries like Docker Hub, GHCR, ECR) - 📦 **Registries** — Container registry management (onebox + external registries like Docker Hub, GHCR, ECR)
- 🔑 **Auth** — Login view, API token management (global + CI tokens) - 🔑 **Auth** — Login view, API token management (global + CI tokens)
- ⚙️ **Settings** — Appearance, Cloudflare integration, SSL/TLS config, network settings, account management - ⚙️ **Settings** — Appearance, Cloudflare integration, SSL/TLS config, network settings, account management
- 🏗️ **Platform Services** — MongoDB, MinIO, ClickHouse, Redis, Caddy monitoring and control - 🏗️ **Platform Services** — MongoDB, MinIO, ClickHouse, Redis, Caddy monitoring and control
- 📋 **Configuration** — Read-only overview of the running server configuration with collapsible sections
Every component supports **light and dark themes** out of the box and communicates via standard `CustomEvent` dispatching. Every component supports **light and dark themes** out of the box and communicates via standard `CustomEvent` dispatching.
@@ -85,6 +89,12 @@ import '@serve.zone/catalog';
| `SzServiceCreateView` | `<sz-service-create-view>` | Service deployment form — image, ports, env vars, volumes, resource limits | | `SzServiceCreateView` | `<sz-service-create-view>` | Service deployment form — image, ports, env vars, volumes, resource limits |
| `SzServicesBackupsView` | `<sz-services-backups-view>` | Backup schedule and backup history management | | `SzServicesBackupsView` | `<sz-services-backups-view>` | Backup schedule and backup history management |
### App Store
| Component | Tag | Description |
|-----------|-----|-------------|
| `SzAppStoreView` | `<sz-app-store-view>` | App marketplace for deploying pre-configured templates (WordPress, Gitea, etc.) with category filtering |
### Platform Services ### Platform Services
| Component | Tag | Description | | Component | Tag | Description |
@@ -104,6 +114,20 @@ import '@serve.zone/catalog';
| `SzDnsSslCard` | `<sz-dns-ssl-card>` | Cloudflare DNS and ACME config status | | `SzDnsSslCard` | `<sz-dns-ssl-card>` | Cloudflare DNS and ACME config status |
| `SzCertificatesCard` | `<sz-certificates-card>` | Certificate status counts — valid, expiring, expired | | `SzCertificatesCard` | `<sz-certificates-card>` | Certificate status counts — valid, expiring, expired |
### Routes
| Component | Tag | Description |
|-----------|-----|-------------|
| `SzRouteListView` | `<sz-route-list-view>` | Route configuration list with type filtering (HTTPS, email, DNS, etc.) |
| `SzRouteCard` | `<sz-route-card>` | Single route card — match criteria, action type, TLS mode, targets, security profile |
### MTA / Email
| Component | Tag | Description |
|-----------|-----|-------------|
| `SzMtaListView` | `<sz-mta-list-view>` | Email management — inbound/outbound messages with status badges and filtering |
| `SzMtaDetailView` | `<sz-mta-detail-view>` | Email detail — SMTP transaction log, TLS info, SPF/DKIM/DMARC results, headers, body |
### Registries ### Registries
| Component | Tag | Description | | Component | Tag | Description |
@@ -119,6 +143,13 @@ import '@serve.zone/catalog';
| `SzTokensView` | `<sz-tokens-view>` | API token management — global and CI tokens with copy/regenerate/delete | | `SzTokensView` | `<sz-tokens-view>` | API token management — global and CI tokens with copy/regenerate/delete |
| `SzSettingsView` | `<sz-settings-view>` | Full settings panel — appearance, Cloudflare, SSL/TLS, network, account | | `SzSettingsView` | `<sz-settings-view>` | Full settings panel — appearance, Cloudflare, SSL/TLS, network, account |
### Configuration
| Component | Tag | Description |
|-----------|-----|-------------|
| `SzConfigOverview` | `<sz-config-overview>` | Top-level configuration overview with informational banner |
| `SzConfigSection` | `<sz-config-section>` | Collapsible config section — icon, enabled/disabled badge, key-value fields, action buttons |
## 🏗️ Architecture ## 🏗️ Architecture
### Component Pattern ### Component Pattern
@@ -186,8 +217,20 @@ import type { IServiceDetail, IServiceStats, ILogEntry, IServiceBackup } from '@
// Network // Network
import type { IDomainDetail, ICertificateDetail, IDnsRecord, ITrafficTarget } from '@serve.zone/catalog'; import type { IDomainDetail, ICertificateDetail, IDnsRecord, ITrafficTarget } from '@serve.zone/catalog';
// Routes
import type { IRouteConfig, IRouteMatch, IRouteAction, IRouteTls, IRouteSecurity } from '@serve.zone/catalog';
// MTA / Email
import type { IEmail, IEmailDetail, ISmtpLogEntry, IConnectionInfo, IAuthenticationResults } from '@serve.zone/catalog';
// Configuration
import type { IConfigField, IConfigSectionAction } from '@serve.zone/catalog';
// Settings & Auth // Settings & Auth
import type { ISettings, IToken, IExternalRegistry } from '@serve.zone/catalog'; import type { ISettings, IToken, IExternalRegistry } from '@serve.zone/catalog';
// App Store
import type { IAppTemplate } from '@serve.zone/catalog';
``` ```
## 🛠️ Development ## 🛠️ Development
@@ -210,7 +253,7 @@ The **wcctools dev server** provides an interactive dashboard where every compon
## License and Legal Information ## License and Legal Information
This repository contains open-source code licensed under the MIT License. A copy of the license can be found in the [LICENSE](./LICENSE) file. This repository contains open-source code licensed under the MIT License. A copy of the license can be found in the [license](./license) file.
**Please note:** The MIT License does not grant permission to use the trade names, trademarks, service marks, or product names of the project, except as required for reasonable and customary use in describing the origin of the work and reproducing the content of the NOTICE file. **Please note:** The MIT License does not grant permission to use the trade names, trademarks, service marks, or product names of the project, except as required for reasonable and customary use in describing the origin of the work and reproducing the content of the NOTICE file.
+1 -1
View File
@@ -3,6 +3,6 @@
*/ */
export const commitinfo = { export const commitinfo = {
name: '@serve.zone/catalog', name: '@serve.zone/catalog',
version: '2.2.0', version: '2.12.4',
description: 'UI component catalog for serve.zone' description: 'UI component catalog for serve.zone'
} }
+11 -1
View File
@@ -1,5 +1,4 @@
// Dashboard Cards // Dashboard Cards
export * from './sz-stat-card.js';
export * from './sz-resource-usage-card.js'; export * from './sz-resource-usage-card.js';
export * from './sz-traffic-card.js'; export * from './sz-traffic-card.js';
export * from './sz-platform-services-card.js'; export * from './sz-platform-services-card.js';
@@ -28,6 +27,7 @@ export * from './sz-registry-external-view.js';
export * from './sz-services-list-view.js'; export * from './sz-services-list-view.js';
export * from './sz-services-backups-view.js'; export * from './sz-services-backups-view.js';
export * from './sz-service-detail-view.js'; export * from './sz-service-detail-view.js';
export * from './sz-app-store-view.js';
// Tokens View // Tokens View
export * from './sz-tokens-view.js'; export * from './sz-tokens-view.js';
@@ -49,6 +49,14 @@ export * from './sz-domain-detail-view.js';
export * from './sz-mta-list-view.js'; export * from './sz-mta-list-view.js';
export * from './sz-mta-detail-view.js'; export * from './sz-mta-detail-view.js';
// Route Configuration Views
export * from './sz-route-card.js';
export * from './sz-route-list-view.js';
// Config Views
export * from './sz-config-section.js';
export * from './sz-config-overview.js';
// Demo Views // Demo Views
export * from './sz-demo-view-dashboard.js'; export * from './sz-demo-view-dashboard.js';
export * from './sz-demo-view-services.js'; export * from './sz-demo-view-services.js';
@@ -57,3 +65,5 @@ export * from './sz-demo-view-registries.js';
export * from './sz-demo-view-tokens.js'; export * from './sz-demo-view-tokens.js';
export * from './sz-demo-view-settings.js'; export * from './sz-demo-view-settings.js';
export * from './sz-demo-view-mta.js'; export * from './sz-demo-view-mta.js';
export * from './sz-demo-view-routes.js';
export * from './sz-demo-view-config.js';
+611
View File
@@ -0,0 +1,611 @@
import {
DeesElement,
customElement,
html,
css,
cssManager,
property,
state,
type TemplateResult,
} from '@design.estate/dees-element';
declare global {
interface HTMLElementTagNameMap {
'sz-app-store-view': SzAppStoreView;
}
}
export interface IAppTemplate {
id: string;
name: string;
description: string;
category: string;
iconUrl?: string;
iconName?: string;
image: string;
port: number;
envVars?: Array<{ key: string; value: string; description: string; required?: boolean }>;
volumes?: string[];
enableMongoDB?: boolean;
enableS3?: boolean;
enableClickHouse?: boolean;
}
@customElement('sz-app-store-view')
export class SzAppStoreView extends DeesElement {
public static demo = () => html`
<div style="padding: 24px; max-width: 1200px;">
<sz-app-store-view
.apps=${[
{
id: 'wordpress',
name: 'WordPress',
description: 'The world\'s most popular open-source content management system for building websites and blogs.',
category: 'CMS',
iconName: 'file-text',
image: 'wordpress:latest',
port: 80,
envVars: [
{ key: 'WORDPRESS_DB_HOST', value: '', description: 'Database host address', required: true },
{ key: 'WORDPRESS_DB_USER', value: 'wordpress', description: 'Database username', required: true },
{ key: 'WORDPRESS_DB_PASSWORD', value: '', description: 'Database password', required: true },
{ key: 'WORDPRESS_DB_NAME', value: 'wordpress', description: 'Database name', required: true },
],
volumes: ['/var/www/html'],
enableMongoDB: false,
enableS3: false,
},
{
id: 'gitea',
name: 'Gitea',
description: 'A lightweight, self-hosted Git service. Painless setup for your own code hosting platform.',
category: 'Development',
iconName: 'git-branch',
image: 'gitea/gitea:latest',
port: 3000,
envVars: [
{ key: 'GITEA__database__DB_TYPE', value: 'sqlite3', description: 'Database type', required: true },
{ key: 'GITEA__server__ROOT_URL', value: '', description: 'Public URL of the Gitea instance', required: false },
],
volumes: ['/data'],
},
{
id: 'ghost',
name: 'Ghost',
description: 'A powerful open-source publishing platform for professional bloggers and content creators.',
category: 'CMS',
iconName: 'book-open',
image: 'ghost:latest',
port: 2368,
envVars: [
{ key: 'url', value: '', description: 'Public URL for the Ghost site', required: true },
{ key: 'database__client', value: 'sqlite3', description: 'Database client type', required: false },
],
volumes: ['/var/lib/ghost/content'],
},
{
id: 'nginx',
name: 'Nginx',
description: 'High-performance HTTP server and reverse proxy with low resource consumption.',
category: 'Web Server',
iconName: 'globe',
image: 'nginx:alpine',
port: 80,
volumes: ['/usr/share/nginx/html', '/etc/nginx/conf.d'],
},
{
id: 'redis',
name: 'Redis',
description: 'In-memory data store used as a database, cache, streaming engine, and message broker.',
category: 'Database',
iconName: 'database',
image: 'redis:alpine',
port: 6379,
},
{
id: 'postgres',
name: 'PostgreSQL',
description: 'Advanced open-source relational database with strong reliability and feature set.',
category: 'Database',
iconName: 'database',
image: 'postgres:16-alpine',
port: 5432,
envVars: [
{ key: 'POSTGRES_USER', value: 'postgres', description: 'Superuser username', required: true },
{ key: 'POSTGRES_PASSWORD', value: '', description: 'Superuser password', required: true },
{ key: 'POSTGRES_DB', value: 'postgres', description: 'Default database name', required: false },
],
volumes: ['/var/lib/postgresql/data'],
},
]}
></sz-app-store-view>
</div>
`;
public static demoGroups = ['Services'];
@property({ type: Array })
public accessor apps: IAppTemplate[] = [];
@state()
private accessor selectedCategory: string = 'All';
@state()
private accessor searchQuery: string = '';
public static styles = [
cssManager.defaultStyles,
css`
:host {
display: block;
}
.header {
margin-bottom: 24px;
}
.header-title {
font-size: 20px;
font-weight: 600;
color: ${cssManager.bdTheme('#18181b', '#fafafa')};
}
.header-subtitle {
font-size: 14px;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')};
margin-top: 4px;
}
.filter-bar {
display: flex;
flex-wrap: wrap;
align-items: center;
gap: 12px;
margin-bottom: 24px;
}
.category-pills {
display: flex;
flex-wrap: wrap;
gap: 8px;
flex: 1;
min-width: 0;
}
.category-pill {
padding: 6px 14px;
border-radius: 9999px;
font-size: 13px;
font-weight: 500;
cursor: pointer;
transition: all 200ms ease;
border: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')};
background: ${cssManager.bdTheme('#ffffff', '#09090b')};
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')};
white-space: nowrap;
}
.category-pill:hover {
border-color: ${cssManager.bdTheme('#d4d4d8', '#3f3f46')};
color: ${cssManager.bdTheme('#18181b', '#fafafa')};
}
.category-pill.active {
background: ${cssManager.bdTheme('#18181b', '#fafafa')};
color: ${cssManager.bdTheme('#fafafa', '#18181b')};
border-color: ${cssManager.bdTheme('#18181b', '#fafafa')};
}
.search-input {
width: 240px;
padding: 8px 12px 8px 36px;
background: ${cssManager.bdTheme('#ffffff', '#18181b')};
border: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')};
border-radius: 6px;
font-size: 14px;
color: ${cssManager.bdTheme('#18181b', '#fafafa')};
outline: none;
transition: border-color 200ms ease;
box-sizing: border-box;
}
.search-input:focus {
border-color: ${cssManager.bdTheme('#3b82f6', '#60a5fa')};
}
.search-input::placeholder {
color: ${cssManager.bdTheme('#a1a1aa', '#52525b')};
}
.search-wrapper {
position: relative;
flex-shrink: 0;
}
.search-icon {
position: absolute;
left: 10px;
top: 50%;
transform: translateY(-50%);
width: 16px;
height: 16px;
color: ${cssManager.bdTheme('#a1a1aa', '#52525b')};
pointer-events: none;
}
.app-grid {
display: grid;
grid-template-columns: repeat(3, 1fr);
gap: 16px;
}
@media (max-width: 1024px) {
.app-grid {
grid-template-columns: repeat(2, 1fr);
}
}
@media (max-width: 640px) {
.app-grid {
grid-template-columns: 1fr;
}
.filter-bar {
flex-direction: column;
align-items: stretch;
}
.search-input {
width: 100%;
}
}
.app-card {
background: ${cssManager.bdTheme('#ffffff', '#09090b')};
border: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')};
border-radius: 8px;
padding: 20px;
display: flex;
flex-direction: column;
gap: 12px;
transition: all 200ms ease;
cursor: default;
}
.app-card:hover {
border-color: ${cssManager.bdTheme('#d4d4d8', '#3f3f46')};
box-shadow: 0 4px 12px ${cssManager.bdTheme('rgba(0,0,0,0.05)', 'rgba(0,0,0,0.2)')};
transform: translateY(-1px);
}
.card-top {
display: flex;
align-items: flex-start;
gap: 14px;
}
.app-icon {
width: 44px;
height: 44px;
border-radius: 10px;
background: ${cssManager.bdTheme('#f4f4f5', '#18181b')};
display: flex;
align-items: center;
justify-content: center;
flex-shrink: 0;
overflow: hidden;
}
.app-icon svg {
width: 22px;
height: 22px;
color: ${cssManager.bdTheme('#18181b', '#fafafa')};
}
.app-icon img {
width: 100%;
height: 100%;
object-fit: cover;
}
.app-icon .letter-fallback {
font-size: 20px;
font-weight: 600;
color: ${cssManager.bdTheme('#18181b', '#fafafa')};
user-select: none;
}
.card-info {
flex: 1;
min-width: 0;
}
.app-name {
font-size: 15px;
font-weight: 600;
color: ${cssManager.bdTheme('#18181b', '#fafafa')};
line-height: 1.3;
}
.category-badge {
display: inline-flex;
align-items: center;
padding: 2px 8px;
border-radius: 9999px;
font-size: 11px;
font-weight: 500;
margin-top: 4px;
background: ${cssManager.bdTheme('#eff6ff', 'rgba(59, 130, 246, 0.15)')};
color: ${cssManager.bdTheme('#2563eb', '#60a5fa')};
}
.app-description {
font-size: 13px;
line-height: 1.5;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')};
display: -webkit-box;
-webkit-line-clamp: 2;
-webkit-box-orient: vertical;
overflow: hidden;
text-overflow: ellipsis;
flex: 1;
}
.card-footer {
display: flex;
align-items: center;
justify-content: space-between;
padding-top: 12px;
border-top: 1px solid ${cssManager.bdTheme('#f4f4f5', '#1a1a1e')};
margin-top: auto;
}
.image-tag {
font-family: monospace;
font-size: 12px;
color: ${cssManager.bdTheme('#a1a1aa', '#52525b')};
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
max-width: 160px;
}
.deploy-button {
display: inline-flex;
align-items: center;
gap: 6px;
padding: 7px 16px;
border-radius: 6px;
font-size: 13px;
font-weight: 500;
cursor: pointer;
transition: all 200ms ease;
border: none;
background: ${cssManager.bdTheme('#18181b', '#fafafa')};
color: ${cssManager.bdTheme('#fafafa', '#18181b')};
white-space: nowrap;
flex-shrink: 0;
}
.deploy-button:hover {
opacity: 0.85;
}
.deploy-button svg {
width: 14px;
height: 14px;
}
.details-button {
display: inline-flex;
align-items: center;
gap: 6px;
padding: 7px 12px;
border-radius: 6px;
font-size: 13px;
font-weight: 500;
cursor: pointer;
transition: all 200ms ease;
border: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')};
background: transparent;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')};
white-space: nowrap;
flex-shrink: 0;
}
.details-button:hover {
border-color: ${cssManager.bdTheme('#d4d4d8', '#3f3f46')};
color: ${cssManager.bdTheme('#18181b', '#fafafa')};
}
.card-actions {
display: flex;
align-items: center;
gap: 8px;
}
.empty-state {
grid-column: 1 / -1;
padding: 64px 24px;
text-align: center;
}
.empty-icon {
width: 48px;
height: 48px;
color: ${cssManager.bdTheme('#d4d4d8', '#3f3f46')};
margin: 0 auto 16px;
}
.empty-title {
font-size: 16px;
font-weight: 600;
color: ${cssManager.bdTheme('#18181b', '#fafafa')};
margin-bottom: 4px;
}
.empty-description {
font-size: 14px;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')};
}
`,
];
private get categories(): string[] {
const cats = new Set(this.apps.map((app) => app.category));
return ['All', ...Array.from(cats).sort()];
}
private get filteredApps(): IAppTemplate[] {
let result = this.apps;
if (this.selectedCategory !== 'All') {
result = result.filter((app) => app.category === this.selectedCategory);
}
if (this.searchQuery.trim()) {
const query = this.searchQuery.trim().toLowerCase();
result = result.filter(
(app) =>
app.name.toLowerCase().includes(query) ||
app.description.toLowerCase().includes(query) ||
app.category.toLowerCase().includes(query)
);
}
return result;
}
public render(): TemplateResult {
const filtered = this.filteredApps;
return html`
<div class="header">
<div class="header-title">App Store</div>
<div class="header-subtitle">Deploy popular applications with one click</div>
</div>
<div class="filter-bar">
<div class="category-pills">
${this.categories.map(
(cat) => html`
<button
class="category-pill ${this.selectedCategory === cat ? 'active' : ''}"
@click=${() => { this.selectedCategory = cat; }}
>
${cat}
</button>
`
)}
</div>
<div class="search-wrapper">
<svg class="search-icon" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<circle cx="11" cy="11" r="8"></circle>
<line x1="21" y1="21" x2="16.65" y2="16.65"></line>
</svg>
<input
type="text"
class="search-input"
placeholder="Search apps..."
.value=${this.searchQuery}
@input=${(e: Event) => { this.searchQuery = (e.target as HTMLInputElement).value; }}
>
</div>
</div>
<div class="app-grid">
${filtered.length > 0
? filtered.map((app) => this.renderAppCard(app))
: html`
<div class="empty-state">
<svg class="empty-icon" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.5">
<circle cx="11" cy="11" r="8"></circle>
<line x1="21" y1="21" x2="16.65" y2="16.65"></line>
<line x1="8" y1="11" x2="14" y2="11"></line>
</svg>
<div class="empty-title">No apps found</div>
<div class="empty-description">
Try adjusting your search or filter to find what you're looking for.
</div>
</div>
`}
</div>
`;
}
private renderAppCard(app: IAppTemplate): TemplateResult {
return html`
<div class="app-card">
<div class="card-top">
<div class="app-icon">
${app.iconUrl
? html`<img src="${app.iconUrl}" alt="${app.name}">`
: app.iconName
? this.renderIconByName(app.iconName)
: html`<span class="letter-fallback">${app.name.charAt(0).toUpperCase()}</span>`}
</div>
<div class="card-info">
<div class="app-name">${app.name}</div>
<div class="category-badge">${app.category}</div>
</div>
</div>
<div class="app-description">${app.description}</div>
<div class="card-footer">
<span class="image-tag" title="${app.image}">${app.image}</span>
<div class="card-actions">
<button class="details-button" @click=${() => this.handleViewDetails(app)}>
View Details
</button>
<button class="deploy-button" @click=${() => this.handleDeploy(app)}>
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<polyline points="20 12 20 22 4 22 4 12"></polyline>
<rect x="2" y="7" width="20" height="5"></rect>
<line x1="12" y1="22" x2="12" y2="7"></line>
<path d="M12 7H7.5a2.5 2.5 0 0 1 0-5C11 2 12 7 12 7z"></path>
<path d="M12 7h4.5a2.5 2.5 0 0 0 0-5C13 2 12 7 12 7z"></path>
</svg>
Deploy
</button>
</div>
</div>
</div>
`;
}
private renderIconByName(name: string): TemplateResult {
const icons: Record<string, TemplateResult> = {
'file-text': html`<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M14 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V8z"></path><polyline points="14 2 14 8 20 8"></polyline><line x1="16" y1="13" x2="8" y2="13"></line><line x1="16" y1="17" x2="8" y2="17"></line><polyline points="10 9 9 9 8 9"></polyline></svg>`,
'git-branch': html`<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><line x1="6" y1="3" x2="6" y2="15"></line><circle cx="18" cy="6" r="3"></circle><circle cx="6" cy="18" r="3"></circle><path d="M18 9a9 9 0 0 1-9 9"></path></svg>`,
'book-open': html`<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M2 3h6a4 4 0 0 1 4 4v14a3 3 0 0 0-3-3H2z"></path><path d="M22 3h-6a4 4 0 0 0-4 4v14a3 3 0 0 1 3-3h7z"></path></svg>`,
'globe': html`<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><circle cx="12" cy="12" r="10"></circle><line x1="2" y1="12" x2="22" y2="12"></line><path d="M12 2a15.3 15.3 0 0 1 4 10 15.3 15.3 0 0 1-4 10 15.3 15.3 0 0 1-4-10 15.3 15.3 0 0 1 4-10z"></path></svg>`,
'database': html`<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><ellipse cx="12" cy="5" rx="9" ry="3"></ellipse><path d="M21 12c0 1.66-4 3-9 3s-9-1.34-9-3"></path><path d="M3 5v14c0 1.66 4 3 9 3s9-1.34 9-3V5"></path></svg>`,
'server': html`<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><rect x="2" y="2" width="20" height="8" rx="2" ry="2"></rect><rect x="2" y="14" width="20" height="8" rx="2" ry="2"></rect><line x1="6" y1="6" x2="6.01" y2="6"></line><line x1="6" y1="18" x2="6.01" y2="18"></line></svg>`,
'package': html`<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><line x1="16.5" y1="9.4" x2="7.5" y2="4.21"></line><path d="M21 16V8a2 2 0 0 0-1-1.73l-7-4a2 2 0 0 0-2 0l-7 4A2 2 0 0 0 3 8v8a2 2 0 0 0 1 1.73l7 4a2 2 0 0 0 2 0l7-4A2 2 0 0 0 21 16z"></path><polyline points="3.27 6.96 12 12.01 20.73 6.96"></polyline><line x1="12" y1="22.08" x2="12" y2="12"></line></svg>`,
'mail': html`<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M4 4h16c1.1 0 2 .9 2 2v12c0 1.1-.9 2-2 2H4c-1.1 0-2-.9-2-2V6c0-1.1.9-2 2-2z"></path><polyline points="22,6 12,13 2,6"></polyline></svg>`,
'shield': html`<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M12 22s8-4 8-10V5l-8-3-8 3v7c0 6 8 10 8 10z"></path></svg>`,
'monitor': html`<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><rect x="2" y="3" width="20" height="14" rx="2" ry="2"></rect><line x1="8" y1="21" x2="16" y2="21"></line><line x1="12" y1="17" x2="12" y2="21"></line></svg>`,
};
return icons[name] || html`<span class="letter-fallback">?</span>`;
}
private handleViewDetails(app: IAppTemplate) {
this.dispatchEvent(
new CustomEvent('view-app', {
detail: { app },
bubbles: true,
composed: true,
})
);
}
private handleDeploy(app: IAppTemplate) {
this.dispatchEvent(
new CustomEvent('deploy-app', {
detail: { app },
bubbles: true,
composed: true,
})
);
}
}
+92
View File
@@ -0,0 +1,92 @@
import {
DeesElement,
customElement,
html,
css,
cssManager,
property,
type TemplateResult,
} from '@design.estate/dees-element';
declare global {
interface HTMLElementTagNameMap {
'sz-config-overview': SzConfigOverview;
}
}
@customElement('sz-config-overview')
export class SzConfigOverview extends DeesElement {
public static demo = () => html`<sz-config-overview
heading="Configuration"
infoText="This is a read-only view of the current running configuration."
>
<div style="padding: 20px; text-align: center; color: #71717a; font-size: 14px;">
Place &lt;sz-config-section&gt; elements here
</div>
</sz-config-overview>`;
public static demoGroups = ['Configuration'];
@property({ type: String })
public accessor heading: string = '';
@property({ type: String })
public accessor infoText: string = '';
public static styles = [
cssManager.defaultStyles,
css`
:host {
display: block;
}
.heading {
font-size: 20px;
font-weight: 600;
color: ${cssManager.bdTheme('#18181b', '#fafafa')};
margin-bottom: 16px;
}
.info-banner {
display: flex;
align-items: flex-start;
gap: 12px;
padding: 14px 18px;
margin-bottom: 20px;
border-radius: 8px;
background: ${cssManager.bdTheme('#eff6ff', 'rgba(59,130,246,0.08)')};
border: 1px solid ${cssManager.bdTheme('#bfdbfe', 'rgba(59,130,246,0.2)')};
color: ${cssManager.bdTheme('#1e40af', '#93c5fd')};
font-size: 13px;
line-height: 1.5;
}
.info-banner dees-icon {
flex-shrink: 0;
font-size: 18px;
margin-top: 1px;
}
::slotted(sz-config-section) {
margin-bottom: 12px;
}
::slotted(sz-config-section:last-child) {
margin-bottom: 0;
}
`,
];
public render(): TemplateResult {
return html`
${this.heading ? html`<div class="heading">${this.heading}</div>` : ''}
${this.infoText ? html`
<div class="info-banner">
<dees-icon .icon=${'lucide:info'}></dees-icon>
<span>${this.infoText}</span>
</div>
` : ''}
<slot></slot>
`;
}
}
+681
View File
@@ -0,0 +1,681 @@
import {
DeesElement,
customElement,
html,
css,
cssManager,
property,
state,
type TemplateResult,
} from '@design.estate/dees-element';
export interface IConfigField {
key: string;
value: string | number | boolean | string[] | null;
type?: 'text' | 'boolean' | 'badge' | 'pills' | 'code' | 'link';
description?: string;
linkTo?: string;
}
export interface IConfigSectionAction {
label: string;
icon?: string;
event?: string;
detail?: any;
}
export interface IConfigSectionLink {
label: string;
href: string;
icon?: string;
external?: boolean;
}
declare global {
interface HTMLElementTagNameMap {
'sz-config-section': SzConfigSection;
}
}
@customElement('sz-config-section')
export class SzConfigSection extends DeesElement {
public static demo = () => html`
<sz-config-section
title="SmartProxy"
subtitle="HTTP/HTTPS and TCP/SNI reverse proxy"
icon="lucide:network"
status="enabled"
.fields=${[
{ key: 'Route Count', value: 12 },
{ key: 'ACME Enabled', value: true, type: 'boolean' },
{ key: 'Account Email', value: 'admin@example.com' },
{ key: 'Use Production', value: true, type: 'boolean' },
{ key: 'Auto Renew', value: true, type: 'boolean' },
{ key: 'Renew Threshold', value: '30 days' },
] as IConfigField[]}
.links=${[
{ label: 'Docs', href: 'https://code.foss.global/serve.zone/smartproxy', icon: 'lucide:bookOpen', external: true },
] as IConfigSectionLink[]}
.actions=${[
{ label: 'Configure', icon: 'lucide:settings', event: 'configure' },
] as IConfigSectionAction[]}
></sz-config-section>
<sz-config-section
title="Email Server"
subtitle="SMTP email handling with smartmta"
icon="lucide:mail"
status="disabled"
.fields=${[
{ key: 'Ports', value: ['25', '465', '587'], type: 'pills' },
{ key: 'Hostname', value: null },
{ key: 'Domains', value: ['example.com', 'mail.example.com'], type: 'pills' },
] as IConfigField[]}
.links=${[
{ label: 'Docs', href: 'https://code.foss.global/serve.zone/smartmta', icon: 'lucide:bookOpen', external: true },
{ label: 'Source', href: 'https://code.foss.global/serve.zone/smartmta', icon: 'lucide:github', external: true },
] as IConfigSectionLink[]}
.actions=${[
{ label: 'Enable', icon: 'lucide:power', event: 'enable' },
] as IConfigSectionAction[]}
></sz-config-section>
<sz-config-section
title="DNS Server"
subtitle="Authoritative DNS with smartdns"
icon="lucide:globe"
status="not-configured"
collapsible
.fields=${[
{ key: 'Port', value: 53 },
{ key: 'NS Domains', value: ['ns1.example.com', 'ns2.example.com'], type: 'pills' },
] as IConfigField[]}
.links=${[
{ label: 'Getting Started', href: 'https://docs.example.com/dns', icon: 'lucide:bookOpen', external: true },
] as IConfigSectionLink[]}
></sz-config-section>
`;
public static demoGroups = ['Configuration'];
@property({ type: String })
public accessor title: string = '';
@property({ type: String })
public accessor subtitle: string = '';
@property({ type: String })
public accessor icon: string = '';
@property({ type: String })
public accessor status: 'enabled' | 'disabled' | 'not-configured' | 'warning' = 'enabled';
@property({ type: Array })
public accessor fields: IConfigField[] = [];
@property({ type: Array })
public accessor actions: IConfigSectionAction[] = [];
@property({ type: Array })
public accessor links: IConfigSectionLink[] = [];
@property({ type: Boolean })
public accessor collapsible: boolean = false;
@property({ type: Boolean })
public accessor collapsed: boolean = false;
@state()
accessor isCollapsed: boolean = false;
@state()
accessor hasSlottedContent: boolean = false;
public static styles = [
cssManager.defaultStyles,
css`
:host {
display: block;
margin-bottom: 16px;
}
dees-tile {
display: block;
}
:host([collapsed]) dees-tile::part(content) {
display: none;
}
:host([collapsed]) dees-tile::part(footer) {
display: none;
}
.section-header {
display: flex;
align-items: center;
padding: 10px 16px;
gap: 12px;
width: 100%;
box-sizing: border-box;
cursor: default;
user-select: none;
}
:host([collapsible]) .section-header {
cursor: pointer;
}
:host([collapsible]) .section-header:hover {
background: var(--dees-color-hover);
}
.header-left {
display: flex;
align-items: center;
gap: 10px;
min-width: 0;
flex: 1;
}
.header-icon {
display: flex;
align-items: center;
justify-content: center;
width: 28px;
height: 28px;
background: var(--dees-color-border-default);
border-radius: 6px;
flex-shrink: 0;
}
.header-icon dees-icon {
font-size: 14px;
color: var(--dees-color-text-muted);
}
.header-text {
min-width: 0;
}
.header-title {
font-size: 13px;
font-weight: 500;
letter-spacing: -0.01em;
color: var(--dees-color-text-secondary);
line-height: 1.3;
}
.header-subtitle {
font-size: 11px;
color: var(--dees-color-text-muted);
letter-spacing: -0.01em;
line-height: 1.3;
margin-top: 1px;
}
.header-right {
display: flex;
align-items: center;
gap: 10px;
flex-shrink: 0;
}
/* Status badge */
.status-badge {
display: inline-flex;
align-items: center;
gap: 6px;
padding: 3px 10px;
border-radius: 9999px;
font-size: 12px;
font-weight: 500;
white-space: nowrap;
}
.status-badge.enabled {
background: ${cssManager.bdTheme('#dcfce7', 'rgba(34,197,94,0.2)')};
color: ${cssManager.bdTheme('#16a34a', '#22c55e')};
}
.status-badge.disabled {
background: ${cssManager.bdTheme('#fee2e2', 'rgba(239,68,68,0.2)')};
color: ${cssManager.bdTheme('#dc2626', '#ef4444')};
}
.status-badge.not-configured {
background: ${cssManager.bdTheme('#f4f4f5', 'rgba(113,113,122,0.2)')};
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')};
}
.status-badge.warning {
background: ${cssManager.bdTheme('#fef3c7', 'rgba(245,158,11,0.15)')};
color: ${cssManager.bdTheme('#92400e', '#fbbf24')};
}
.status-dot {
width: 7px;
height: 7px;
border-radius: 50%;
}
.status-badge.enabled .status-dot {
background: ${cssManager.bdTheme('#16a34a', '#22c55e')};
}
.status-badge.disabled .status-dot {
background: ${cssManager.bdTheme('#dc2626', '#ef4444')};
}
.status-badge.not-configured .status-dot {
background: ${cssManager.bdTheme('#a1a1aa', '#52525b')};
}
.status-badge.warning .status-dot {
background: ${cssManager.bdTheme('#f59e0b', '#fbbf24')};
}
/* Footer action buttons — canonical dees-modal / dees-tile pattern */
.section-footer {
display: flex;
flex-direction: row;
justify-content: flex-end;
align-items: center;
gap: 0;
height: 36px;
width: 100%;
box-sizing: border-box;
}
.tile-button {
padding: 0 16px;
height: 100%;
text-align: center;
font-size: 12px;
font-weight: 500;
cursor: pointer;
user-select: none;
transition: all 0.15s ease;
background: transparent;
border: none;
border-left: 1px solid var(--dees-color-border-subtle);
color: var(--dees-color-text-muted);
white-space: nowrap;
display: flex;
align-items: center;
gap: 6px;
font-family: inherit;
text-decoration: none;
}
.tile-button:first-child {
border-left: none;
}
.tile-button:hover {
background: var(--dees-color-hover);
color: var(--dees-color-text-primary);
}
.tile-button.primary {
color: ${cssManager.bdTheme('hsl(217.2 91.2% 59.8%)', 'hsl(213.1 93.9% 67.8%)')};
font-weight: 600;
}
.tile-button.primary:hover {
background: ${cssManager.bdTheme('hsl(217.2 91.2% 59.8% / 0.08)', 'hsl(213.1 93.9% 67.8% / 0.08)')};
color: ${cssManager.bdTheme('hsl(217.2 91.2% 50%)', 'hsl(213.1 93.9% 75%)')};
}
.tile-button dees-icon {
font-size: 12px;
}
/* Chevron */
.chevron {
display: flex;
align-items: center;
transition: transform 200ms ease;
}
.chevron.collapsed {
transform: rotate(-90deg);
}
.chevron dees-icon {
font-size: 14px;
color: var(--dees-color-text-muted);
}
/* Content */
.section-content {
padding: 0;
}
.fields-list {
display: flex;
flex-direction: column;
}
/* Field rows */
.field-row {
display: flex;
align-items: flex-start;
justify-content: space-between;
padding: 10px 20px;
border-bottom: 1px solid ${cssManager.bdTheme('#f4f4f5', '#1a1a1e')};
gap: 16px;
}
.fields-list .field-row:last-child {
border-bottom: none;
}
.field-key {
font-size: 13px;
font-weight: 500;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')};
flex-shrink: 0;
min-width: 140px;
padding-top: 1px;
}
.field-value {
font-size: 13px;
font-family: 'SF Mono', Monaco, 'Cascadia Code', monospace;
color: ${cssManager.bdTheme('#18181b', '#fafafa')};
text-align: right;
word-break: break-all;
}
.field-value.null-value {
color: ${cssManager.bdTheme('#a1a1aa', '#52525b')};
font-style: italic;
font-family: inherit;
}
/* Boolean display */
.bool-value {
display: inline-flex;
align-items: center;
gap: 5px;
font-family: inherit;
font-size: 13px;
font-weight: 500;
}
.bool-value.true {
color: ${cssManager.bdTheme('#16a34a', '#22c55e')};
}
.bool-value.false {
color: ${cssManager.bdTheme('#dc2626', '#ef4444')};
}
.bool-dot {
width: 6px;
height: 6px;
border-radius: 50%;
}
.bool-value.true .bool-dot {
background: ${cssManager.bdTheme('#16a34a', '#22c55e')};
}
.bool-value.false .bool-dot {
background: ${cssManager.bdTheme('#dc2626', '#ef4444')};
}
/* Pills */
.pills {
display: flex;
flex-wrap: wrap;
gap: 5px;
justify-content: flex-end;
}
.pill {
display: inline-flex;
align-items: center;
padding: 2px 9px;
border-radius: 9999px;
font-size: 12px;
font-weight: 500;
font-family: 'SF Mono', Monaco, 'Cascadia Code', monospace;
background: ${cssManager.bdTheme('#eff6ff', 'rgba(59,130,246,0.1)')};
color: ${cssManager.bdTheme('#2563eb', '#60a5fa')};
}
/* Code value */
.code-value {
font-family: 'SF Mono', Monaco, 'Cascadia Code', monospace;
font-size: 12px;
background: ${cssManager.bdTheme('#f4f4f5', '#18181b')};
padding: 2px 8px;
border-radius: 4px;
color: ${cssManager.bdTheme('#18181b', '#fafafa')};
}
/* Link value */
.link-value {
color: ${cssManager.bdTheme('#2563eb', '#60a5fa')};
cursor: pointer;
text-decoration: none;
font-family: inherit;
font-size: 13px;
}
.link-value:hover {
text-decoration: underline;
}
/* Description hint */
.field-description {
font-size: 11px;
color: ${cssManager.bdTheme('#a1a1aa', '#52525b')};
margin-top: 3px;
text-align: right;
}
/* Slot for custom content */
.slot-content {
border-top: 1px solid ${cssManager.bdTheme('#f4f4f5', '#1a1a1e')};
}
.slot-content.empty {
display: none;
}
/* Badge type */
.badge-value {
display: inline-flex;
align-items: center;
padding: 2px 9px;
border-radius: 9999px;
font-size: 12px;
font-weight: 500;
background: ${cssManager.bdTheme('#f4f4f5', '#27272a')};
color: ${cssManager.bdTheme('#52525b', '#a1a1aa')};
}
`,
];
async connectedCallback() {
await super.connectedCallback();
this.isCollapsed = this.collapsed;
if (this.collapsible) {
this.setAttribute('collapsible', '');
}
}
updated(changedProperties: Map<string, any>) {
super.updated(changedProperties);
if (changedProperties.has('isCollapsed')) {
this.toggleAttribute('collapsed', this.isCollapsed);
}
}
private onSlotChange(e: Event) {
const slot = e.target as HTMLSlotElement;
this.hasSlottedContent = slot.assignedNodes({ flatten: true }).length > 0;
}
public render(): TemplateResult {
const statusLabels: Record<string, string> = {
'enabled': 'Enabled',
'disabled': 'Disabled',
'not-configured': 'Not Configured',
'warning': 'Warning',
};
return html`
<dees-tile>
<div
slot="header"
class="section-header"
@click=${() => {
if (this.collapsible) {
this.isCollapsed = !this.isCollapsed;
}
}}
>
<div class="header-left">
${this.icon ? html`
<div class="header-icon">
<dees-icon .icon=${this.icon}></dees-icon>
</div>
` : ''}
<div class="header-text">
<div class="header-title">${this.title}</div>
${this.subtitle ? html`<div class="header-subtitle">${this.subtitle}</div>` : ''}
</div>
</div>
<div class="header-right">
${this.status ? html`
<span class="status-badge ${this.status}">
<span class="status-dot"></span>
${statusLabels[this.status] || this.status}
</span>
` : ''}
${this.collapsible ? html`
<span class="chevron ${this.isCollapsed ? 'collapsed' : ''}">
<dees-icon .icon=${'lucide:chevronDown'}></dees-icon>
</span>
` : ''}
</div>
</div>
<div class="section-content">
${this.fields.length > 0 ? html`
<div class="fields-list">
${this.fields.map(field => this.renderField(field))}
</div>
` : ''}
<div class="slot-content ${!this.hasSlottedContent ? 'empty' : ''}">
<slot @slotchange=${this.onSlotChange}></slot>
</div>
</div>
${this.links.length > 0 || this.actions.length > 0 ? html`
<div slot="footer" class="section-footer">
${this.links.map(link => html`
<a
class="tile-button"
href=${link.href}
target=${link.external ? '_blank' : '_self'}
rel=${link.external ? 'noopener noreferrer' : ''}
@click=${(e: Event) => e.stopPropagation()}
>
${link.icon ? html`<dees-icon .icon=${link.icon}></dees-icon>` : ''}
${link.label}
${link.external ? html`<dees-icon .icon=${'lucide:externalLink'}></dees-icon>` : ''}
</a>
`)}
${this.actions.map(action => html`
<button class="tile-button primary" @click=${(e: Event) => {
e.stopPropagation();
this.dispatchEvent(new CustomEvent(action.event || 'action', {
detail: action.detail || { label: action.label },
bubbles: true,
composed: true,
}));
}}>
${action.icon ? html`<dees-icon .icon=${action.icon}></dees-icon>` : ''}
${action.label}
</button>
`)}
</div>
` : ''}
</dees-tile>
`;
}
private renderField(field: IConfigField): TemplateResult {
return html`
<div class="field-row">
<div class="field-key">${field.key}</div>
<div>
${this.renderFieldValue(field)}
${field.description ? html`<div class="field-description">${field.description}</div>` : ''}
</div>
</div>
`;
}
private renderFieldValue(field: IConfigField): TemplateResult {
const value = field.value;
const type = field.type || this.inferType(value);
// Null / undefined
if (value === null || value === undefined) {
return html`<span class="field-value null-value">Not configured</span>`;
}
switch (type) {
case 'boolean':
return html`
<span class="bool-value ${value ? 'true' : 'false'}">
<span class="bool-dot"></span>
${value ? 'Enabled' : 'Disabled'}
</span>
`;
case 'pills':
if (Array.isArray(value) && value.length === 0) {
return html`<span class="field-value null-value">None</span>`;
}
return html`
<div class="pills">
${(value as string[]).map(v => html`<span class="pill">${v}</span>`)}
</div>
`;
case 'code':
return html`<span class="code-value">${String(value)}</span>`;
case 'badge':
return html`<span class="badge-value">${String(value)}</span>`;
case 'link':
return html`
<span
class="link-value"
@click=${() => {
if (field.linkTo) {
this.dispatchEvent(new CustomEvent('navigate', {
detail: { target: field.linkTo },
bubbles: true,
composed: true,
}));
}
}}
>${String(value)}</span>
`;
default:
return html`<span class="field-value">${String(value)}</span>`;
}
}
private inferType(value: unknown): string {
if (typeof value === 'boolean') return 'boolean';
if (Array.isArray(value)) return 'pills';
return 'text';
}
}
+165
View File
@@ -0,0 +1,165 @@
import {
DeesElement,
customElement,
html,
css,
cssManager,
type TemplateResult,
} from '@design.estate/dees-element';
import type { IConfigField, IConfigSectionAction } from './sz-config-section.js';
import './index.js';
declare global {
interface HTMLElementTagNameMap {
'sz-demo-view-config': SzDemoViewConfig;
}
}
@customElement('sz-demo-view-config')
export class SzDemoViewConfig extends DeesElement {
public static styles = [
cssManager.defaultStyles,
css`
:host {
display: block;
padding: 24px;
height: 100%;
overflow-y: auto;
box-sizing: border-box;
}
`,
];
public render(): TemplateResult {
const systemFields: IConfigField[] = [
{ key: 'Base Directory', value: '/home/user/.serve.zone/dcrouter' },
{ key: 'Data Directory', value: '/home/user/.serve.zone/dcrouter/data' },
{ key: 'Public IP', value: '203.0.113.50' },
{ key: 'Proxy IPs', value: ['203.0.113.10', '203.0.113.11'], type: 'pills' },
{ key: 'Uptime', value: '3d 14h 22m' },
{ key: 'Storage Backend', value: 'filesystem', type: 'badge' },
];
const proxyFields: IConfigField[] = [
{ key: 'Route Count', value: 12 },
{ key: 'ACME Enabled', value: true, type: 'boolean' },
{ key: 'Account Email', value: 'admin@serve.zone' },
{ key: 'Use Production', value: true, type: 'boolean' },
{ key: 'Auto Renew', value: true, type: 'boolean' },
{ key: 'Renew Threshold', value: '30 days' },
];
const emailFields: IConfigField[] = [
{ key: 'Ports', value: ['25', '465', '587'], type: 'pills' },
{ key: 'Hostname', value: 'mail.serve.zone' },
{ key: 'Domains', value: ['serve.zone', 'mail.serve.zone'], type: 'pills' },
{ key: 'Email Routes', value: 5 },
{ key: 'Received Path', value: '/data/emails' },
];
const dnsFields: IConfigField[] = [
{ key: 'Port', value: 53 },
{ key: 'NS Domains', value: ['ns1.serve.zone', 'ns2.serve.zone'], type: 'pills' },
{ key: 'Scopes', value: ['serve.zone', 'example.com'], type: 'pills' },
{ key: 'Record Count', value: 24 },
{ key: 'DNS Challenge', value: true, type: 'boolean' },
];
const tlsFields: IConfigField[] = [
{ key: 'Contact Email', value: 'admin@serve.zone' },
{ key: 'Domain', value: 'serve.zone' },
{ key: 'Source', value: 'acme', type: 'badge' },
{ key: 'Certificate Path', value: null },
{ key: 'Key Path', value: null },
];
const cacheFields: IConfigField[] = [
{ key: 'Storage Path', value: '/home/user/.serve.zone/dcrouter/tsmdb' },
{ key: 'DB Name', value: 'dcrouter' },
{ key: 'Default TTL', value: '30 days' },
{ key: 'Cleanup Interval', value: '1 hour' },
];
const radiusFields: IConfigField[] = [
{ key: 'Auth Port', value: null },
{ key: 'Accounting Port', value: null },
];
const remoteIngressFields: IConfigField[] = [
{ key: 'Tunnel Port', value: 8443 },
{ key: 'Hub Domain', value: 'hub.serve.zone' },
{ key: 'TLS Configured', value: true, type: 'boolean' },
];
return html`
<sz-config-overview
infoText="This view displays the current running configuration. DcRouter is configured through code or remote management."
>
<sz-config-section
title="System"
subtitle="Base paths and infrastructure"
icon="lucide:server"
status="enabled"
.fields=${systemFields}
></sz-config-section>
<sz-config-section
title="SmartProxy"
subtitle="HTTP/HTTPS and TCP/SNI reverse proxy"
icon="lucide:network"
status="enabled"
.fields=${proxyFields}
.actions=${[{ label: 'View Routes', icon: 'lucide:arrow-right', event: 'navigate', detail: { view: 'routes' } }] as IConfigSectionAction[]}
></sz-config-section>
<sz-config-section
title="Email Server"
subtitle="SMTP email handling with smartmta"
icon="lucide:mail"
status="enabled"
.fields=${emailFields}
></sz-config-section>
<sz-config-section
title="DNS Server"
subtitle="Authoritative DNS with smartdns"
icon="lucide:globe"
status="enabled"
.fields=${dnsFields}
></sz-config-section>
<sz-config-section
title="TLS / Certificates"
subtitle="Certificate management and ACME"
icon="lucide:shield-check"
status="enabled"
.fields=${tlsFields}
></sz-config-section>
<sz-config-section
title="Cache Database"
subtitle="Persistent caching with smartdata"
icon="lucide:database"
status="enabled"
.fields=${cacheFields}
></sz-config-section>
<sz-config-section
title="RADIUS Server"
subtitle="Network authentication and VLAN assignment"
icon="lucide:wifi"
status="not-configured"
.fields=${radiusFields}
></sz-config-section>
<sz-config-section
title="Remote Ingress"
subtitle="Edge tunnel nodes"
icon="lucide:cloud"
status="enabled"
.fields=${remoteIngressFields}
></sz-config-section>
</sz-config-overview>
`;
}
}
+362
View File
@@ -0,0 +1,362 @@
import {
DeesElement,
customElement,
html,
css,
cssManager,
state,
type TemplateResult,
} from '@design.estate/dees-element';
import type { DeesAppui } from '@design.estate/dees-catalog';
import type { IRouteConfig } from './sz-route-card.js';
import './index.js';
declare global {
interface HTMLElementTagNameMap {
'sz-demo-view-routes': SzDemoViewRoutes;
}
}
@customElement('sz-demo-view-routes')
export class SzDemoViewRoutes extends DeesElement {
private appui: DeesAppui | null = null;
@state()
private accessor currentTab: 'all' | 'https' | 'email' | 'dns' = 'all';
private demoRoutes: IRouteConfig[] = [
// 1. HTTPS with TLS termination + auto cert
{
id: 'route-1',
name: 'Web Frontend',
description: 'Main website with TLS termination and automatic certificates',
enabled: true,
priority: 10,
tags: ['web', 'https', 'production'],
match: {
ports: 443,
domains: ['serve.zone', 'www.serve.zone'],
protocol: 'http',
},
action: {
type: 'forward',
targets: [{ host: '10.0.0.10', port: 3000 }],
tls: { mode: 'terminate', certificate: 'auto' },
},
},
// 2. HTTP to HTTPS redirect
{
id: 'route-2',
name: 'HTTP Redirect',
description: 'Redirects all HTTP traffic to HTTPS',
enabled: true,
priority: 100,
tags: ['web', 'http', 'redirect'],
match: {
ports: 80,
domains: ['serve.zone', 'www.serve.zone'],
protocol: 'http',
},
action: {
type: 'socket-handler',
},
},
// 3. Email SMTP route
{
id: 'route-3',
name: 'SMTP Inbound',
description: 'Inbound email relay for serve.zone domain',
enabled: true,
tags: ['email', 'smtp', 'production'],
match: {
ports: 25,
domains: 'mail.serve.zone',
},
action: {
type: 'forward',
targets: [{ host: '10.0.1.5', port: 25 }],
tls: { mode: 'terminate', certificate: 'auto' },
},
},
// 4. API gateway with path matching, rate limiting, CORS
{
id: 'route-4',
name: 'API Gateway',
description: 'API gateway with rate limiting, CORS headers, and load balancing',
enabled: true,
priority: 20,
tags: ['web', 'api', 'https', 'production'],
match: {
ports: 443,
domains: 'api.serve.zone',
path: '/v2/*',
protocol: 'http',
},
action: {
type: 'forward',
targets: [
{ host: ['10.0.0.20', '10.0.0.21', '10.0.0.22'], port: 8080 },
],
tls: { mode: 'terminate', certificate: 'auto' },
loadBalancing: { algorithm: 'round-robin' },
},
security: {
rateLimit: { enabled: true, maxRequests: 200, window: 60 },
maxConnections: 5000,
},
headers: {
response: {
'Access-Control-Allow-Origin': '*',
'X-Request-Id': '{{requestId}}',
},
},
},
// 5. WebSocket route
{
id: 'route-5',
name: 'WebSocket Realtime',
description: 'Real-time WebSocket connections for live updates',
enabled: true,
tags: ['web', 'https', 'websocket'],
match: {
ports: 443,
domains: 'ws.serve.zone',
path: '/ws/*',
protocol: 'http',
},
action: {
type: 'forward',
targets: [{ host: '10.0.0.30', port: 9090 }],
tls: { mode: 'terminate', certificate: 'auto' },
websocket: { enabled: true },
},
},
// 6. Wildcard domain route
{
id: 'route-6',
name: 'Tenant Wildcard',
description: 'Multi-tenant wildcard routing for customer subdomains',
enabled: true,
priority: 50,
tags: ['web', 'https', 'multi-tenant'],
match: {
ports: 443,
domains: '*.customers.serve.zone',
protocol: 'http',
},
action: {
type: 'forward',
targets: [{ host: '10.0.0.40', port: 8080 }],
tls: { mode: 'terminate', certificate: 'auto' },
},
security: {
ipAllowList: ['10.0.0.0/8', '172.16.0.0/12'],
},
},
// 7. Load-balanced route with health check
{
id: 'route-7',
name: 'Microservices LB',
description: 'Load-balanced microservices backend with IP-hash affinity',
enabled: true,
tags: ['web', 'https', 'production'],
match: {
ports: [443, 8443],
domains: 'services.serve.zone',
protocol: 'http',
},
action: {
type: 'forward',
targets: [
{ host: ['10.0.2.1', '10.0.2.2', '10.0.2.3', '10.0.2.4'], port: 3000 },
],
tls: { mode: 'terminate-and-reencrypt', certificate: 'auto' },
loadBalancing: { algorithm: 'ip-hash' },
},
},
// 8. DNS-over-HTTPS route
{
id: 'route-8',
name: 'DNS over HTTPS',
description: 'DNS-over-HTTPS resolver endpoint',
enabled: true,
tags: ['dns', 'https'],
match: {
ports: 443,
domains: 'dns.serve.zone',
path: '/dns-query',
protocol: 'http',
},
action: {
type: 'forward',
targets: [{ host: '10.0.3.1', port: 8053 }],
tls: { mode: 'terminate', certificate: 'auto' },
},
},
// 9. NFTables high-performance route
{
id: 'route-9',
name: 'High-Perf TCP Proxy',
description: 'NFTables-accelerated TCP proxy for game servers',
enabled: true,
tags: ['tcp', 'nftables', 'production'],
match: {
ports: [{ from: 27000, to: 27050 }],
protocol: 'tcp',
},
action: {
type: 'forward',
targets: [{ host: '10.0.4.1', port: 'preserve' }],
forwardingEngine: 'nftables',
},
},
// 10. Disabled maintenance route
{
id: 'route-10',
name: 'Legacy Admin Panel',
description: 'Deprecated admin panel — disabled for maintenance',
enabled: false,
tags: ['web', 'https', 'deprecated'],
match: {
ports: 443,
domains: 'admin-old.serve.zone',
protocol: 'http',
},
action: {
type: 'socket-handler',
},
security: {
ipBlockList: ['0.0.0.0/0'],
},
},
];
private get filteredRoutes(): IRouteConfig[] {
if (this.currentTab === 'all') return this.demoRoutes;
if (this.currentTab === 'https') {
return this.demoRoutes.filter((r) =>
r.tags?.some((t) => ['web', 'https', 'http'].includes(t))
);
}
if (this.currentTab === 'email') {
return this.demoRoutes.filter((r) =>
r.tags?.some((t) => ['email', 'smtp'].includes(t))
);
}
if (this.currentTab === 'dns') {
return this.demoRoutes.filter((r) =>
r.tags?.some((t) => ['dns'].includes(t))
);
}
return this.demoRoutes;
}
async onActivate(context: { appui: DeesAppui; viewId: string }) {
this.appui = context.appui;
this.appui.setContentTabs([
{
key: 'All Routes',
action: () => {
this.currentTab = 'all';
this.updateSecondaryMenu();
},
},
{
key: 'HTTP/S',
action: () => {
this.currentTab = 'https';
this.updateSecondaryMenu();
},
},
{
key: 'Email',
action: () => {
this.currentTab = 'email';
this.updateSecondaryMenu();
},
},
{
key: 'DNS',
action: () => {
this.currentTab = 'dns';
this.updateSecondaryMenu();
},
},
]);
this.updateSecondaryMenu();
}
private updateSecondaryMenu() {
if (!this.appui) return;
const total = this.demoRoutes.length;
const active = this.demoRoutes.filter((r) => r.enabled !== false).length;
const forwardCount = this.demoRoutes.filter((r) => r.action.type === 'forward').length;
this.appui.setSecondaryMenu({
heading: 'Routes',
groups: [
{
name: 'Actions',
items: [
{
type: 'action',
key: 'Refresh',
iconName: 'lucide:RefreshCw',
action: () => {
console.log('Refresh routes');
},
},
{
type: 'action',
key: 'Add Route',
iconName: 'lucide:Plus',
action: () => {
console.log('Add route');
},
},
],
},
{
name: 'Statistics',
items: [
{ type: 'header' as const, label: `${total} Total Routes` },
{ type: 'header' as const, label: `${active} Active` },
{ type: 'header' as const, label: `${forwardCount} Forward` },
{ type: 'header' as const, label: `${total - forwardCount} Socket Handler` },
],
},
],
});
}
onDeactivate() {
// Cleanup if needed
}
public static styles = [
cssManager.defaultStyles,
css`
:host {
display: block;
padding: 24px;
height: 100%;
overflow-y: auto;
box-sizing: border-box;
}
`,
];
public render(): TemplateResult {
return html`
<sz-route-list-view
.routes=${this.filteredRoutes}
@route-click=${(e: CustomEvent<IRouteConfig>) => {
console.log('Route clicked:', e.detail.name);
}}
></sz-route-list-view>
`;
}
}
+79 -45
View File
@@ -229,57 +229,87 @@ export class SzDomainDetailView extends DeesElement {
} }
} }
.section { dees-tile.full-width {
background: ${cssManager.bdTheme('#ffffff', '#09090b')};
border: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')};
border-radius: 8px;
overflow: hidden;
}
.section.full-width {
grid-column: 1 / -1; grid-column: 1 / -1;
} }
.section-header { .section-header {
height: 36px;
display: flex; display: flex;
justify-content: space-between;
align-items: center; align-items: center;
padding: 14px 16px; padding: 0 16px;
border-bottom: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')}; width: 100%;
background: ${cssManager.bdTheme('#f4f4f5', '#18181b')}; box-sizing: border-box;
} }
.section-title { .section-title {
font-size: 14px; flex: 1;
font-weight: 600;
color: ${cssManager.bdTheme('#18181b', '#fafafa')};
display: flex; display: flex;
align-items: center; align-items: center;
gap: 8px; gap: 8px;
font-weight: 500;
font-size: 13px;
letter-spacing: -0.01em;
color: var(--dees-color-text-secondary);
} }
.section-title svg { .section-title svg {
width: 16px; width: 14px;
height: 16px; height: 14px;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')}; flex-shrink: 0;
color: var(--dees-color-text-secondary);
} }
.section-action { .section-footer {
padding: 6px 10px; display: flex;
background: transparent; flex-direction: row;
border: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')}; justify-content: flex-end;
border-radius: 4px; align-items: center;
gap: 0;
height: 36px;
width: 100%;
box-sizing: border-box;
}
.tile-button {
padding: 0 16px;
height: 100%;
text-align: center;
font-size: 12px; font-size: 12px;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')}; font-weight: 500;
cursor: pointer; cursor: pointer;
transition: all 200ms ease; user-select: none;
transition: all 0.15s ease;
background: transparent;
border: none;
border-left: 1px solid var(--dees-color-border-subtle);
color: var(--dees-color-text-muted);
white-space: nowrap;
display: flex;
align-items: center;
gap: 6px;
} }
.section-action:hover { .tile-button:first-child {
background: ${cssManager.bdTheme('#ffffff', '#09090b')}; border-left: none;
color: ${cssManager.bdTheme('#18181b', '#fafafa')};
} }
.tile-button:hover {
background: var(--dees-color-hover);
color: var(--dees-color-text-primary);
}
.tile-button.primary {
color: ${cssManager.bdTheme('hsl(217.2 91.2% 59.8%)', 'hsl(213.1 93.9% 67.8%)')};
font-weight: 600;
}
.tile-button.primary:hover {
background: ${cssManager.bdTheme('hsl(217.2 91.2% 59.8% / 0.08)', 'hsl(213.1 93.9% 67.8% / 0.08)')};
color: ${cssManager.bdTheme('hsl(217.2 91.2% 50%)', 'hsl(213.1 93.9% 75%)')};
}
.section-content { .section-content {
padding: 16px; padding: 16px;
} }
@@ -582,8 +612,8 @@ export class SzDomainDetailView extends DeesElement {
<div class="grid"> <div class="grid">
<!-- Certificate Section --> <!-- Certificate Section -->
<div class="section"> <dees-tile>
<div class="section-header"> <div slot="header" class="section-header">
<div class="section-title"> <div class="section-title">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"> <svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<rect x="3" y="11" width="18" height="11" rx="2" ry="2"></rect> <rect x="3" y="11" width="18" height="11" rx="2" ry="2"></rect>
@@ -591,9 +621,6 @@ export class SzDomainDetailView extends DeesElement {
</svg> </svg>
SSL Certificate SSL Certificate
</div> </div>
${this.certificate ? html`
<button class="section-action" @click=${() => this.handleRenewCertificate()}>Renew</button>
` : ''}
</div> </div>
<div class="section-content"> <div class="section-content">
${this.certificate ? html` ${this.certificate ? html`
@@ -652,11 +679,16 @@ export class SzDomainDetailView extends DeesElement {
<div class="empty-state">No certificate configured</div> <div class="empty-state">No certificate configured</div>
`} `}
</div> </div>
${this.certificate ? html`
<div slot="footer" class="section-footer">
<button class="tile-button primary" @click=${() => this.handleRenewCertificate()}>Renew</button>
</div> </div>
` : ''}
</dees-tile>
<!-- Proxy Routes Section --> <!-- Proxy Routes Section -->
<div class="section"> <dees-tile>
<div class="section-header"> <div slot="header" class="section-header">
<div class="section-title"> <div class="section-title">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"> <svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<polyline points="16 3 21 3 21 8"></polyline> <polyline points="16 3 21 3 21 8"></polyline>
@@ -679,11 +711,11 @@ export class SzDomainDetailView extends DeesElement {
<div class="empty-state">No proxy routes configured</div> <div class="empty-state">No proxy routes configured</div>
`} `}
</div> </div>
</div> </dees-tile>
<!-- DNS Records Section --> <!-- DNS Records Section -->
<div class="section full-width"> <dees-tile class="full-width">
<div class="section-header"> <div slot="header" class="section-header">
<div class="section-title"> <div class="section-title">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"> <svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<circle cx="12" cy="12" r="10"></circle> <circle cx="12" cy="12" r="10"></circle>
@@ -692,13 +724,6 @@ export class SzDomainDetailView extends DeesElement {
</svg> </svg>
DNS Records DNS Records
</div> </div>
<button class="section-action" @click=${() => this.handleAddDnsRecord()}>
<svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" style="margin-right: 4px;">
<line x1="12" y1="5" x2="12" y2="19"></line>
<line x1="5" y1="12" x2="19" y2="12"></line>
</svg>
Add Record
</button>
</div> </div>
<div class="section-content"> <div class="section-content">
${this.dnsRecords.length > 0 ? html` ${this.dnsRecords.length > 0 ? html`
@@ -737,7 +762,16 @@ export class SzDomainDetailView extends DeesElement {
<div class="empty-state">No DNS records configured</div> <div class="empty-state">No DNS records configured</div>
`} `}
</div> </div>
<div slot="footer" class="section-footer">
<button class="tile-button primary" @click=${() => this.handleAddDnsRecord()}>
<svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<line x1="12" y1="5" x2="12" y2="19"></line>
<line x1="5" y1="12" x2="19" y2="12"></line>
</svg>
Add Record
</button>
</div> </div>
</dees-tile>
</div> </div>
`; `;
} }
+125 -81
View File
@@ -256,31 +256,89 @@ export class SzMtaDetailView extends DeesElement {
gap: 24px; gap: 24px;
} }
.card { .card-header {
background: ${cssManager.bdTheme('#ffffff', '#09090b')}; height: 36px;
border: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')}; display: flex;
border-radius: 8px; align-items: center;
overflow: hidden; padding: 0 16px;
width: 100%;
box-sizing: border-box;
} }
.card-header { .card-heading {
flex: 1;
display: flex; display: flex;
justify-content: space-between; align-items: baseline;
align-items: center; gap: 8px;
padding: 16px; min-width: 0;
border-bottom: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')};
} }
.card-title { .card-title {
font-size: 16px; font-weight: 500;
font-weight: 600; font-size: 13px;
color: ${cssManager.bdTheme('#18181b', '#fafafa')}; letter-spacing: -0.01em;
color: var(--dees-color-text-secondary);
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
} }
.card-subtitle { .card-subtitle {
font-size: 13px; font-size: 12px;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')}; color: var(--dees-color-text-muted);
margin-top: 2px; letter-spacing: -0.01em;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
.card-footer {
display: flex;
flex-direction: row;
justify-content: flex-end;
align-items: center;
gap: 0;
height: 36px;
width: 100%;
box-sizing: border-box;
}
.tile-button {
padding: 0 16px;
height: 100%;
text-align: center;
font-size: 12px;
font-weight: 500;
cursor: pointer;
user-select: none;
transition: all 0.15s ease;
background: transparent;
border: none;
border-left: 1px solid var(--dees-color-border-subtle);
color: var(--dees-color-text-muted);
white-space: nowrap;
display: flex;
align-items: center;
gap: 6px;
}
.tile-button:first-child {
border-left: none;
}
.tile-button:hover {
background: var(--dees-color-hover);
color: var(--dees-color-text-primary);
}
.tile-button.primary {
color: ${cssManager.bdTheme('hsl(217.2 91.2% 59.8%)', 'hsl(213.1 93.9% 67.8%)')};
font-weight: 600;
}
.tile-button.primary:hover {
background: ${cssManager.bdTheme('hsl(217.2 91.2% 59.8% / 0.08)', 'hsl(213.1 93.9% 67.8% / 0.08)')};
color: ${cssManager.bdTheme('hsl(217.2 91.2% 50%)', 'hsl(213.1 93.9% 75%)')};
} }
.card-content { .card-content {
@@ -451,37 +509,13 @@ export class SzMtaDetailView extends DeesElement {
color: ${cssManager.bdTheme('#dc2626', '#ef4444')}; color: ${cssManager.bdTheme('#dc2626', '#ef4444')};
} }
/* Copy button */ /* SMTP metadata banner — sits inside content, above the log */
.smtp-copy-button {
display: inline-flex;
align-items: center;
gap: 6px;
padding: 6px 12px;
border-radius: 6px;
border: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')};
background: transparent;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')};
font-size: 12px;
font-weight: 500;
cursor: pointer;
transition: all 150ms ease;
}
.smtp-copy-button:hover {
background: ${cssManager.bdTheme('#f4f4f5', '#27272a')};
color: ${cssManager.bdTheme('#18181b', '#fafafa')};
border-color: ${cssManager.bdTheme('#d4d4d8', '#3f3f46')};
}
/* Header subtitle enhancements */
.smtp-header-subtitle { .smtp-header-subtitle {
font-size: 13px; padding: 10px 16px;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')}; font-size: 12px;
margin-top: 2px; color: var(--dees-color-text-muted);
display: flex; border-bottom: 1px solid var(--dees-color-border-subtle);
align-items: center; font-family: monospace;
gap: 8px;
flex-wrap: wrap;
} }
.smtp-direction-badge { .smtp-direction-badge {
@@ -578,7 +612,7 @@ export class SzMtaDetailView extends DeesElement {
color: ${cssManager.bdTheme('#ca8a04', '#facc15')}; color: ${cssManager.bdTheme('#ca8a04', '#facc15')};
} }
.rejection-card { dees-tile.rejection-card::part(outer) {
border-color: ${cssManager.bdTheme('#fecaca', 'rgba(239, 68, 68, 0.3)')}; border-color: ${cssManager.bdTheme('#fecaca', 'rgba(239, 68, 68, 0.3)')};
} }
@@ -646,9 +680,11 @@ export class SzMtaDetailView extends DeesElement {
<div class="content"> <div class="content">
<div class="main-content"> <div class="main-content">
<!-- Email Metadata --> <!-- Email Metadata -->
<div class="card"> <dees-tile>
<div class="card-header"> <div slot="header" class="card-header">
<div class="card-title">Email Metadata</div> <div class="card-heading">
<span class="card-title">Email Metadata</span>
</div>
</div> </div>
<div class="card-content"> <div class="card-content">
<div class="detail-list"> <div class="detail-list">
@@ -684,49 +720,53 @@ export class SzMtaDetailView extends DeesElement {
</div> </div>
</div> </div>
</div> </div>
</div> </dees-tile>
<!-- SMTP Transaction Log --> <!-- SMTP Transaction Log -->
<div class="card"> <dees-tile>
<div class="card-header"> <div slot="header" class="card-header">
<div> <div class="card-heading">
<div class="card-title">SMTP Transaction Log</div> <span class="card-title">SMTP Transaction Log</span>
<div class="smtp-header-subtitle">
<span class="smtp-direction-badge ${email.direction}">${email.direction}</span> <span class="smtp-direction-badge ${email.direction}">${email.direction}</span>
<span>${email.direction === 'outbound' </div>
</div>
<div class="smtp-header-subtitle">
${email.direction === 'outbound'
? `${email.connectionInfo.sourceHostname}${email.connectionInfo.destinationIp}:${email.connectionInfo.destinationPort}` ? `${email.connectionInfo.sourceHostname}${email.connectionInfo.destinationIp}:${email.connectionInfo.destinationPort}`
: `${email.connectionInfo.sourceIp}${email.connectionInfo.sourceHostname}:${email.connectionInfo.destinationPort}` : `${email.connectionInfo.sourceIp}${email.connectionInfo.sourceHostname}:${email.connectionInfo.destinationPort}`
}</span> }
</div> </div>
</div> ${this.renderSmtpLog(email)}
<button class="smtp-copy-button" @click=${() => this.copySmtpLog()}> <div slot="footer" class="card-footer">
<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"> <button class="tile-button" @click=${() => this.copySmtpLog()}>
<svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
<rect x="9" y="9" width="13" height="13" rx="2" ry="2"></rect> <rect x="9" y="9" width="13" height="13" rx="2" ry="2"></rect>
<path d="M5 15H4a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h9a2 2 0 0 1 2 2v1"></path> <path d="M5 15H4a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h9a2 2 0 0 1 2 2v1"></path>
</svg> </svg>
Copy Log Copy Log
</button> </button>
</div> </div>
${this.renderSmtpLog(email)} </dees-tile>
</div>
<!-- Email Body --> <!-- Email Body -->
<div class="card"> <dees-tile>
<div class="card-header"> <div slot="header" class="card-header">
<div> <div class="card-heading">
<div class="card-title">Email Body (Escaped)</div> <span class="card-title">Email Body (Escaped)</span>
<div class="card-subtitle">Raw content — HTML is not rendered</div> <span class="card-subtitle">Raw content — HTML is not rendered</span>
</div> </div>
</div> </div>
<pre class="email-body-container">${email.body}</pre> <pre class="email-body-container">${email.body}</pre>
</div> </dees-tile>
</div> </div>
<div class="sidebar"> <div class="sidebar">
<!-- Connection Info --> <!-- Connection Info -->
<div class="card"> <dees-tile>
<div class="card-header"> <div slot="header" class="card-header">
<div class="card-title">Connection Info</div> <div class="card-heading">
<span class="card-title">Connection Info</span>
</div>
</div> </div>
<div class="card-content"> <div class="card-content">
<div class="detail-list"> <div class="detail-list">
@@ -772,12 +812,14 @@ export class SzMtaDetailView extends DeesElement {
` : ''} ` : ''}
</div> </div>
</div> </div>
</div> </dees-tile>
<!-- Authentication Results --> <!-- Authentication Results -->
<div class="card"> <dees-tile>
<div class="card-header"> <div slot="header" class="card-header">
<div class="card-title">Authentication Results</div> <div class="card-heading">
<span class="card-title">Authentication Results</span>
</div>
</div> </div>
<div class="card-content"> <div class="card-content">
<div class="auth-row"> <div class="auth-row">
@@ -802,13 +844,15 @@ export class SzMtaDetailView extends DeesElement {
<span class="auth-badge ${email.authenticationResults.dmarc}">${email.authenticationResults.dmarc}</span> <span class="auth-badge ${email.authenticationResults.dmarc}">${email.authenticationResults.dmarc}</span>
</div> </div>
</div> </div>
</div> </dees-tile>
<!-- Rejection Details (conditional) --> <!-- Rejection Details (conditional) -->
${email.status === 'rejected' || email.status === 'bounced' ? html` ${email.status === 'rejected' || email.status === 'bounced' ? html`
<div class="card rejection-card"> <dees-tile class="rejection-card">
<div class="card-header"> <div slot="header" class="card-header">
<div class="card-title">Rejection Details</div> <div class="card-heading">
<span class="card-title">Rejection Details</span>
</div>
</div> </div>
<div class="card-content"> <div class="card-content">
${email.rejectionReason ? html` ${email.rejectionReason ? html`
@@ -820,7 +864,7 @@ export class SzMtaDetailView extends DeesElement {
<div class="rejection-text">${email.bounceMessage}</div> <div class="rejection-text">${email.bounceMessage}</div>
` : ''} ` : ''}
</div> </div>
</div> </dees-tile>
` : ''} ` : ''}
</div> </div>
</div> </div>
+43 -37
View File
@@ -7,8 +7,7 @@ import {
property, property,
type TemplateResult, type TemplateResult,
} from '@design.estate/dees-element'; } from '@design.estate/dees-element';
import type { IStatsTile } from '@design.estate/dees-catalog';
import './sz-stat-card.js';
declare global { declare global {
interface HTMLElementTagNameMap { interface HTMLElementTagNameMap {
@@ -91,19 +90,11 @@ export class SzNetworkDomainsView extends DeesElement {
border-color: ${cssManager.bdTheme('#d4d4d8', '#3f3f46')}; border-color: ${cssManager.bdTheme('#d4d4d8', '#3f3f46')};
} }
.stats-grid { dees-statsgrid {
display: grid; display: block;
grid-template-columns: repeat(2, 1fr);
gap: 16px;
margin-bottom: 24px; margin-bottom: 24px;
} }
@media (min-width: 768px) {
.stats-grid {
grid-template-columns: repeat(4, 1fr);
}
}
.table-container { .table-container {
background: ${cssManager.bdTheme('#ffffff', '#09090b')}; background: ${cssManager.bdTheme('#ffffff', '#09090b')};
border: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')}; border: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')};
@@ -205,6 +196,42 @@ export class SzNetworkDomainsView extends DeesElement {
`, `,
]; ];
private get tiles(): IStatsTile[] {
return [
{
id: 'total',
title: 'Total Domains',
value: this.stats.total,
type: 'number',
icon: 'lucide:globe',
},
{
id: 'valid',
title: 'Valid Certificates',
value: this.stats.valid,
type: 'number',
icon: 'lucide:shieldCheck',
color: '#22c55e',
},
{
id: 'expiring',
title: 'Expiring Soon',
value: this.stats.expiring,
type: 'number',
icon: 'lucide:shieldAlert',
color: this.stats.expiring > 0 ? '#f59e0b' : undefined,
},
{
id: 'expired',
title: 'Expired/Pending',
value: this.stats.expired,
type: 'number',
icon: 'lucide:circleOff',
color: this.stats.expired > 0 ? '#ef4444' : undefined,
},
];
}
public render(): TemplateResult { public render(): TemplateResult {
return html` return html`
<div class="header"> <div class="header">
@@ -212,31 +239,10 @@ export class SzNetworkDomainsView extends DeesElement {
<button class="sync-button" @click=${() => this.handleSync()}>Sync Cloudflare</button> <button class="sync-button" @click=${() => this.handleSync()}>Sync Cloudflare</button>
</div> </div>
<div class="stats-grid"> <dees-statsgrid
<sz-stat-card .tiles=${this.tiles}
label="Total Domains" .minTileWidth=${200}
value="${this.stats.total}" ></dees-statsgrid>
icon="server"
></sz-stat-card>
<sz-stat-card
label="Valid Certificates"
value="${this.stats.valid}"
icon="check"
variant="success"
></sz-stat-card>
<sz-stat-card
label="Expiring Soon"
value="${this.stats.expiring}"
icon="stop"
variant="${this.stats.expiring > 0 ? 'warning' : 'default'}"
></sz-stat-card>
<sz-stat-card
label="Expired/Pending"
value="${this.stats.expired}"
icon="stop"
variant="${this.stats.expired > 0 ? 'error' : 'default'}"
></sz-stat-card>
</div>
<div class="table-container"> <div class="table-container">
<div class="table-header"> <div class="table-header">
+156 -131
View File
@@ -7,8 +7,7 @@ import {
property, property,
type TemplateResult, type TemplateResult,
} from '@design.estate/dees-element'; } from '@design.estate/dees-element';
import type { IStatsTile } from '@design.estate/dees-catalog';
import './sz-stat-card.js';
declare global { declare global {
interface HTMLElementTagNameMap { interface HTMLElementTagNameMap {
@@ -113,42 +112,107 @@ export class SzNetworkProxyView extends DeesElement {
background: ${cssManager.bdTheme('#f4f4f5', '#18181b')}; background: ${cssManager.bdTheme('#f4f4f5', '#18181b')};
} }
.stats-grid { dees-statsgrid {
display: grid; display: block;
grid-template-columns: repeat(2, 1fr);
gap: 16px;
margin-bottom: 24px; margin-bottom: 24px;
} }
@media (min-width: 768px) { dees-tile {
.stats-grid { display: block;
grid-template-columns: repeat(4, 1fr);
}
}
.section {
background: ${cssManager.bdTheme('#ffffff', '#09090b')};
border: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')};
border-radius: 8px;
margin-bottom: 24px; margin-bottom: 24px;
overflow: hidden;
} }
.section-header { .section-header {
padding: 16px; height: 36px;
border-bottom: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')}; display: flex;
align-items: center;
padding: 0 16px;
width: 100%;
box-sizing: border-box;
}
.section-heading {
flex: 1;
display: flex;
align-items: baseline;
gap: 8px;
min-width: 0;
} }
.section-title { .section-title {
font-size: 16px; font-weight: 500;
font-weight: 600; font-size: 13px;
color: ${cssManager.bdTheme('#18181b', '#fafafa')}; letter-spacing: -0.01em;
color: var(--dees-color-text-secondary);
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
} }
.section-subtitle { .section-subtitle {
font-size: 13px; font-size: 12px;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')}; color: var(--dees-color-text-muted);
margin-top: 2px; letter-spacing: -0.01em;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
.section-footer {
display: flex;
flex-direction: row;
justify-content: flex-end;
align-items: center;
gap: 0;
height: 36px;
width: 100%;
box-sizing: border-box;
}
.tile-button {
padding: 0 16px;
height: 100%;
text-align: center;
font-size: 12px;
font-weight: 500;
cursor: pointer;
user-select: none;
transition: all 0.15s ease;
background: transparent;
border: none;
border-left: 1px solid var(--dees-color-border-subtle);
color: var(--dees-color-text-muted);
white-space: nowrap;
display: flex;
align-items: center;
gap: 6px;
}
.tile-button:first-child {
border-left: none;
}
.tile-button:hover {
background: var(--dees-color-hover);
color: var(--dees-color-text-primary);
}
.tile-button.primary {
color: ${cssManager.bdTheme('hsl(217.2 91.2% 59.8%)', 'hsl(213.1 93.9% 67.8%)')};
font-weight: 600;
}
.tile-button.primary:hover {
background: ${cssManager.bdTheme('hsl(217.2 91.2% 59.8% / 0.08)', 'hsl(213.1 93.9% 67.8% / 0.08)')};
color: ${cssManager.bdTheme('hsl(217.2 91.2% 50%)', 'hsl(213.1 93.9% 75%)')};
}
.tile-button.danger {
color: ${cssManager.bdTheme('#dc2626', '#ef4444')};
}
.tile-button.danger:hover {
background: ${cssManager.bdTheme('#fee2e2', 'rgba(239, 68, 68, 0.1)')};
} }
.table-header { .table-header {
@@ -234,61 +298,6 @@ export class SzNetworkProxyView extends DeesElement {
color: ${cssManager.bdTheme('#dc2626', '#ef4444')}; color: ${cssManager.bdTheme('#dc2626', '#ef4444')};
} }
.logs-header {
display: flex;
justify-content: space-between;
align-items: center;
padding: 16px;
border-bottom: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')};
}
.logs-actions {
display: flex;
gap: 8px;
}
.stream-button {
display: inline-flex;
align-items: center;
gap: 6px;
padding: 6px 12px;
background: ${cssManager.bdTheme('#2563eb', '#3b82f6')};
border: none;
border-radius: 4px;
font-size: 13px;
font-weight: 500;
color: white;
cursor: pointer;
transition: all 200ms ease;
}
.stream-button:hover {
background: ${cssManager.bdTheme('#1d4ed8', '#2563eb')};
}
.stream-button.streaming {
background: ${cssManager.bdTheme('#dc2626', '#ef4444')};
}
.stream-button.streaming:hover {
background: ${cssManager.bdTheme('#b91c1c', '#dc2626')};
}
.clear-button {
padding: 6px 12px;
background: transparent;
border: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')};
border-radius: 4px;
font-size: 13px;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')};
cursor: pointer;
transition: all 200ms ease;
}
.clear-button:hover {
background: ${cssManager.bdTheme('#f4f4f5', '#18181b')};
}
.logs-container { .logs-container {
padding: 16px; padding: 16px;
font-family: monospace; font-family: monospace;
@@ -336,41 +345,57 @@ export class SzNetworkProxyView extends DeesElement {
`, `,
]; ];
private get tiles(): IStatsTile[] {
return [
{
id: 'proxy-status',
title: 'Proxy Status',
value: this.proxyStatus === 'running' ? 'Running' : 'Stopped',
type: 'text',
icon: 'lucide:server',
color: this.proxyStatus === 'running' ? '#22c55e' : '#ef4444',
},
{
id: 'routes',
title: 'Routes',
value: this.routeCount,
type: 'number',
icon: 'lucide:server',
},
{
id: 'certificates',
title: 'Certificates',
value: this.certificateCount,
type: 'number',
icon: 'lucide:check',
},
{
id: 'targets',
title: 'Targets',
value: this.targetCount,
type: 'number',
icon: 'lucide:server',
},
];
}
public render(): TemplateResult { public render(): TemplateResult {
return html` return html`
<div class="actions"> <div class="actions">
<button class="refresh-button" @click=${() => this.handleRefresh()}>Refresh</button> <button class="refresh-button" @click=${() => this.handleRefresh()}>Refresh</button>
</div> </div>
<div class="stats-grid"> <dees-statsgrid
<sz-stat-card .tiles=${this.tiles}
label="Proxy Status" .minTileWidth=${200}
value="${this.proxyStatus === 'running' ? 'Running' : 'Stopped'}" ></dees-statsgrid>
icon="server"
variant="${this.proxyStatus === 'running' ? 'success' : 'error'}"
valueBadge
></sz-stat-card>
<sz-stat-card
label="Routes"
value="${this.routeCount}"
icon="server"
></sz-stat-card>
<sz-stat-card
label="Certificates"
value="${this.certificateCount}"
icon="check"
></sz-stat-card>
<sz-stat-card
label="Targets"
value="${this.targetCount}"
icon="server"
></sz-stat-card>
</div>
<div class="section"> <dees-tile>
<div class="section-header"> <div slot="header" class="section-header">
<div class="section-title">Traffic Targets</div> <div class="section-heading">
<div class="section-subtitle">Services, registry, and platform services with their routing info</div> <span class="section-title">Traffic Targets</span>
<span class="section-subtitle">Services, registry, and platform services with their routing info</span>
</div>
</div> </div>
<div class="table-header"> <div class="table-header">
<span>Type</span> <span>Type</span>
@@ -388,30 +413,13 @@ export class SzNetworkProxyView extends DeesElement {
<span><span class="status-badge ${target.status}">${target.status}</span></span> <span><span class="status-badge ${target.status}">${target.status}</span></span>
</div> </div>
`)} `)}
</div> </dees-tile>
<div class="section"> <dees-tile>
<div class="logs-header"> <div slot="header" class="section-header">
<div> <div class="section-heading">
<div class="section-title">Access Logs</div> <span class="section-title">Access Logs</span>
<div class="section-subtitle">Real-time Caddy access logs</div> <span class="section-subtitle">Real-time Caddy access logs</span>
</div>
<div class="logs-actions">
<button class="stream-button ${this.streaming ? 'streaming' : ''}" @click=${() => this.toggleStreaming()}>
<svg width="14" height="14" viewBox="0 0 24 24" fill="currentColor">
${this.streaming
? html`<rect x="6" y="6" width="12" height="12" rx="1"/>`
: html`<polygon points="5,3 19,12 5,21"/>`
}
</svg>
${this.streaming ? 'Stop' : 'Stream'}
</button>
<button class="clear-button" @click=${() => this.handleClearLogs()}>
<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<polyline points="3,6 5,6 21,6"/><path d="M19,6v14a2,2,0,0,1-2,2H7a2,2,0,0,1-2-2V6m3,0V4a2,2,0,0,1,2-2h4a2,2,0,0,1,2,2v2"/>
</svg>
Clear logs
</button>
</div> </div>
</div> </div>
<div class="logs-container"> <div class="logs-container">
@@ -428,7 +436,24 @@ export class SzNetworkProxyView extends DeesElement {
<div class="empty-logs">Click "Stream" to start live access log streaming</div> <div class="empty-logs">Click "Stream" to start live access log streaming</div>
`} `}
</div> </div>
<div slot="footer" class="section-footer">
<button class="tile-button ${this.streaming ? 'danger' : 'primary'}" @click=${() => this.toggleStreaming()}>
<svg width="12" height="12" viewBox="0 0 24 24" fill="currentColor">
${this.streaming
? html`<rect x="6" y="6" width="12" height="12" rx="1"/>`
: html`<polygon points="5,3 19,12 5,21"/>`
}
</svg>
${this.streaming ? 'Stop' : 'Stream'}
</button>
<button class="tile-button" @click=${() => this.handleClearLogs()}>
<svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<polyline points="3,6 5,6 21,6"/><path d="M19,6v14a2,2,0,0,1-2,2H7a2,2,0,0,1-2-2V6m3,0V4a2,2,0,0,1,2-2h4a2,2,0,0,1,2,2v2"/>
</svg>
Clear logs
</button>
</div> </div>
</dees-tile>
`; `;
} }
@@ -225,39 +225,35 @@ export class SzPlatformServiceDetailView extends DeesElement {
} }
} }
.section { dees-tile.full-width {
background: ${cssManager.bdTheme('#ffffff', '#09090b')};
border: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')};
border-radius: 8px;
overflow: hidden;
}
.section.full-width {
grid-column: 1 / -1; grid-column: 1 / -1;
} }
.section-header { .section-header {
height: 36px;
display: flex; display: flex;
justify-content: space-between;
align-items: center; align-items: center;
padding: 14px 16px; padding: 0 16px;
border-bottom: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')}; width: 100%;
background: ${cssManager.bdTheme('#f4f4f5', '#18181b')}; box-sizing: border-box;
} }
.section-title { .section-title {
font-size: 14px; flex: 1;
font-weight: 600;
color: ${cssManager.bdTheme('#18181b', '#fafafa')};
display: flex; display: flex;
align-items: center; align-items: center;
gap: 8px; gap: 8px;
font-weight: 500;
font-size: 13px;
letter-spacing: -0.01em;
color: var(--dees-color-text-secondary);
} }
.section-title svg { .section-title svg {
width: 16px; width: 14px;
height: 16px; height: 14px;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')}; flex-shrink: 0;
color: var(--dees-color-text-secondary);
} }
.section-content { .section-content {
@@ -363,56 +359,6 @@ export class SzPlatformServiceDetailView extends DeesElement {
background: ${cssManager.bdTheme('#ef4444', '#ef4444')}; background: ${cssManager.bdTheme('#ef4444', '#ef4444')};
} }
.log-container {
background: ${cssManager.bdTheme('#18181b', '#09090b')};
border-radius: 6px;
padding: 12px;
max-height: 300px;
overflow-y: auto;
font-family: 'SF Mono', Monaco, 'Cascadia Code', monospace;
font-size: 12px;
line-height: 1.6;
}
.log-entry {
display: flex;
gap: 12px;
padding: 4px 0;
}
.log-timestamp {
color: #71717a;
flex-shrink: 0;
}
.log-level {
flex-shrink: 0;
width: 50px;
text-transform: uppercase;
font-weight: 500;
}
.log-level.info {
color: #60a5fa;
}
.log-level.warn {
color: #fbbf24;
}
.log-level.error {
color: #f87171;
}
.log-level.debug {
color: #a1a1aa;
}
.log-message {
color: #fafafa;
word-break: break-word;
}
.config-item { .config-item {
display: flex; display: flex;
justify-content: space-between; justify-content: space-between;
@@ -502,8 +448,8 @@ export class SzPlatformServiceDetailView extends DeesElement {
<div class="grid"> <div class="grid">
<!-- Connection Info --> <!-- Connection Info -->
<div class="section"> <dees-tile>
<div class="section-header"> <div slot="header" class="section-header">
<div class="section-title"> <div class="section-title">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"> <svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<path d="M10 13a5 5 0 0 0 7.54.54l3-3a5 5 0 0 0-7.07-7.07l-1.72 1.71"></path> <path d="M10 13a5 5 0 0 0 7.54.54l3-3a5 5 0 0 0-7.07-7.07l-1.72 1.71"></path>
@@ -566,11 +512,11 @@ export class SzPlatformServiceDetailView extends DeesElement {
</div> </div>
` : ''} ` : ''}
</div> </div>
</div> </dees-tile>
<!-- Configuration --> <!-- Configuration -->
<div class="section"> <dees-tile>
<div class="section-header"> <div slot="header" class="section-header">
<div class="section-title"> <div class="section-title">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"> <svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<circle cx="12" cy="12" r="3"></circle> <circle cx="12" cy="12" r="3"></circle>
@@ -587,12 +533,12 @@ export class SzPlatformServiceDetailView extends DeesElement {
</div> </div>
`)} `)}
</div> </div>
</div> </dees-tile>
<!-- Metrics --> <!-- Metrics -->
${this.service.metrics ? html` ${this.service.metrics ? html`
<div class="section full-width"> <dees-tile class="full-width">
<div class="section-header"> <div slot="header" class="section-header">
<div class="section-title"> <div class="section-title">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"> <svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<line x1="18" y1="20" x2="18" y2="10"></line> <line x1="18" y1="20" x2="18" y2="10"></line>
@@ -633,33 +579,22 @@ export class SzPlatformServiceDetailView extends DeesElement {
` : ''} ` : ''}
</div> </div>
</div> </div>
</div> </dees-tile>
` : ''} ` : ''}
<!-- Logs --> <!-- Logs -->
<div class="section full-width"> <div style="grid-column: 1 / -1;">
<div class="section-header"> <dees-chart-log
<div class="section-title"> .label=${'Service Logs'}
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"> .logEntries=${this.logs.map(log => ({
<polyline points="4 17 10 11 4 5"></polyline> timestamp: log.timestamp.includes('T') ? log.timestamp : new Date(log.timestamp).toISOString(),
<line x1="12" y1="19" x2="20" y2="19"></line> level: log.level as 'debug' | 'info' | 'warn' | 'error',
</svg> message: log.message,
Logs }))}
</div> .autoScroll=${true}
</div> .maxEntries=${2000}
<div class="section-content"> .showMetrics=${true}
<div class="log-container"> ></dees-chart-log>
${this.logs.length > 0 ? this.logs.map(log => html`
<div class="log-entry">
<span class="log-timestamp">${log.timestamp}</span>
<span class="log-level ${log.level}">${log.level}</span>
<span class="log-message">${log.message}</span>
</div>
`) : html`
<div style="color: #71717a; text-align: center; padding: 20px;">No logs available</div>
`}
</div>
</div>
</div> </div>
</div> </div>
`; `;
+879
View File
@@ -0,0 +1,879 @@
import {
DeesElement,
customElement,
html,
css,
cssManager,
property,
type TemplateResult,
} from '@design.estate/dees-element';
declare global {
interface HTMLElementTagNameMap {
'sz-route-card': SzRouteCard;
}
}
// Simplified route types for display purposes
export type TRouteActionType = 'forward' | 'socket-handler';
export type TTlsMode = 'passthrough' | 'terminate' | 'terminate-and-reencrypt';
export type TPortRange = number | number[] | Array<{ from: number; to: number }>;
export interface IRouteMatch {
ports: TPortRange;
domains?: string | string[];
path?: string;
clientIp?: string[];
tlsVersion?: string[];
headers?: Record<string, string>;
protocol?: 'http' | 'tcp';
}
export interface IRouteTarget {
host: string | string[];
port: number | 'preserve';
}
export interface IRouteTls {
mode: TTlsMode;
certificate?: 'auto' | { key: string; cert: string };
}
export interface IRouteAction {
type: TRouteActionType;
targets?: IRouteTarget[];
tls?: IRouteTls;
websocket?: { enabled: boolean };
loadBalancing?: { algorithm: 'round-robin' | 'least-connections' | 'ip-hash' };
forwardingEngine?: 'node' | 'nftables';
}
export interface IRouteSecurity {
ipAllowList?: string[];
ipBlockList?: string[];
maxConnections?: number;
rateLimit?: { enabled: boolean; maxRequests: number; window: number };
}
export interface IRouteMetadata {
sourceProfileRef?: string;
networkTargetRef?: string;
sourceProfileName?: string;
networkTargetName?: string;
lastResolvedAt?: number;
}
export interface IRouteConfig {
id?: string;
match: IRouteMatch;
action: IRouteAction;
security?: IRouteSecurity;
headers?: { request?: Record<string, string>; response?: Record<string, string> };
metadata?: IRouteMetadata;
/** When true, only VPN clients whose TargetProfile matches this route get access */
vpnOnly?: boolean;
name?: string;
description?: string;
priority?: number;
tags?: string[];
enabled?: boolean;
}
function formatPorts(ports: TPortRange): string {
if (typeof ports === 'number') return String(ports);
if (Array.isArray(ports)) {
return ports
.map((p) => {
if (typeof p === 'number') return String(p);
return `${p.from}\u2013${p.to}`;
})
.join(', ');
}
return String(ports);
}
function formatTargets(targets: IRouteTarget[]): string[] {
const result: string[] = [];
for (const t of targets) {
const hosts = Array.isArray(t.host) ? t.host : [t.host];
const portStr = t.port === 'preserve' ? '(preserve)' : String(t.port);
for (const h of hosts) {
result.push(`${h}:${portStr}`);
}
}
return result;
}
@customElement('sz-route-card')
export class SzRouteCard extends DeesElement {
public static demo = () => html`
<div style="padding: 24px; max-width: 520px;">
<sz-route-card
.route=${{
name: 'API Gateway',
description: 'Main API gateway with TLS termination and load balancing',
enabled: true,
priority: 10,
tags: ['web', 'api', 'production'],
match: {
ports: [443, 8443],
domains: ['api.example.com', '*.api.serve.zone'],
path: '/api/*',
protocol: 'http' as const,
clientIp: ['10.0.0.0/8'],
},
action: {
type: 'forward' as const,
targets: [
{ host: ['10.0.0.1', '10.0.0.2'], port: 8080 },
],
tls: { mode: 'terminate' as const, certificate: 'auto' as const },
websocket: { enabled: true },
loadBalancing: { algorithm: 'round-robin' as const },
forwardingEngine: 'nftables' as const,
},
security: {
ipAllowList: ['10.0.0.0/8'],
ipBlockList: ['192.168.100.0/24'],
rateLimit: { enabled: true, maxRequests: 100, window: 60 },
maxConnections: 1000,
},
vpnOnly: true,
metadata: {
sourceProfileName: 'STANDARD',
networkTargetName: 'LOSSLESS_INFRA',
},
} satisfies IRouteConfig}
></sz-route-card>
</div>
`;
public static demoGroups = ['Routes'];
@property({ type: Object })
public accessor route: IRouteConfig | null = null;
@property({ type: Boolean })
public accessor showActions: boolean = false;
public static styles = [
cssManager.defaultStyles,
css`
:host {
display: block;
}
dees-tile::part(outer) {
transition: border-color 200ms ease, box-shadow 200ms ease;
}
dees-tile:hover::part(outer) {
border-color: var(--dees-color-border-strong);
box-shadow: 0 2px 8px ${cssManager.bdTheme('rgba(0,0,0,0.06)', 'rgba(0,0,0,0.2)')};
}
/* Header */
.header {
display: flex;
align-items: center;
justify-content: space-between;
gap: 12px;
height: 40px;
padding: 0 16px;
width: 100%;
box-sizing: border-box;
}
.header-left {
display: flex;
align-items: center;
gap: 8px;
min-width: 0;
flex: 1;
}
.status-dot {
width: 8px;
height: 8px;
border-radius: 50%;
flex-shrink: 0;
}
.status-dot.enabled {
background: ${cssManager.bdTheme('#22c55e', '#22c55e')};
box-shadow: 0 0 6px ${cssManager.bdTheme('rgba(34,197,94,0.4)', 'rgba(34,197,94,0.3)')};
}
.status-dot.disabled {
background: ${cssManager.bdTheme('#a1a1aa', '#52525b')};
}
.route-name {
font-size: 14px;
font-weight: 500;
letter-spacing: -0.01em;
color: var(--dees-color-text-secondary);
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
.header-badges {
display: flex;
gap: 6px;
flex-shrink: 0;
}
.badge {
display: inline-flex;
align-items: center;
padding: 2px 8px;
border-radius: 9999px;
font-size: 11px;
font-weight: 500;
white-space: nowrap;
}
.badge.forward {
background: ${cssManager.bdTheme('#dbeafe', 'rgba(59, 130, 246, 0.2)')};
color: ${cssManager.bdTheme('#2563eb', '#60a5fa')};
}
.badge.socket-handler {
background: ${cssManager.bdTheme('#ede9fe', 'rgba(139, 92, 246, 0.2)')};
color: ${cssManager.bdTheme('#7c3aed', '#a78bfa')};
}
.badge.enabled {
background: ${cssManager.bdTheme('#dcfce7', 'rgba(34, 197, 94, 0.2)')};
color: ${cssManager.bdTheme('#16a34a', '#22c55e')};
}
.badge.disabled {
background: ${cssManager.bdTheme('#f4f4f5', 'rgba(113, 113, 122, 0.2)')};
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')};
}
.card-body {
padding: 16px 20px;
}
.description {
font-size: 13px;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')};
margin-bottom: 8px;
line-height: 1.4;
}
.meta-row {
display: flex;
align-items: center;
justify-content: space-between;
flex-wrap: wrap;
gap: 6px;
margin-bottom: 16px;
}
.tags {
display: flex;
flex-wrap: wrap;
gap: 4px;
}
.tag {
padding: 2px 8px;
background: ${cssManager.bdTheme('#f4f4f5', '#27272a')};
border-radius: 4px;
font-size: 11px;
font-weight: 500;
color: ${cssManager.bdTheme('#52525b', '#a1a1aa')};
}
.priority {
font-size: 11px;
color: ${cssManager.bdTheme('#a1a1aa', '#71717a')};
font-weight: 500;
}
/* Sections */
.section {
border-left: 3px solid;
padding: 10px 14px;
margin-bottom: 12px;
border-radius: 0 6px 6px 0;
background: ${cssManager.bdTheme('#fafafa', '#101010')};
}
.section:last-of-type {
margin-bottom: 0;
}
.section.match {
border-left-color: ${cssManager.bdTheme('#3b82f6', '#3b82f6')};
}
.section.action {
border-left-color: ${cssManager.bdTheme('#22c55e', '#22c55e')};
}
.section.security {
border-left-color: ${cssManager.bdTheme('#f59e0b', '#f59e0b')};
}
.section.linked {
border-left-color: ${cssManager.bdTheme('#8b5cf6', '#8b5cf6')};
}
.linked-name {
display: inline-flex;
padding: 1px 6px;
border-radius: 3px;
font-size: 12px;
font-weight: 600;
font-family: monospace;
background: ${cssManager.bdTheme('#ede9fe', 'rgba(139, 92, 246, 0.15)')};
color: ${cssManager.bdTheme('#6d28d9', '#a78bfa')};
}
.section-label {
font-size: 10px;
font-weight: 600;
text-transform: uppercase;
letter-spacing: 0.08em;
color: ${cssManager.bdTheme('#a1a1aa', '#71717a')};
margin-bottom: 8px;
}
.field-row {
display: flex;
gap: 8px;
margin-bottom: 5px;
font-size: 13px;
line-height: 1.5;
}
.field-row:last-child {
margin-bottom: 0;
}
.field-key {
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')};
min-width: 64px;
flex-shrink: 0;
font-weight: 500;
}
.field-value {
color: ${cssManager.bdTheme('#18181b', '#e4e4e7')};
word-break: break-all;
}
.domain-chip {
display: inline-flex;
padding: 1px 6px;
background: ${cssManager.bdTheme('#eff6ff', 'rgba(59, 130, 246, 0.1)')};
border-radius: 3px;
font-size: 12px;
margin-right: 4px;
margin-bottom: 2px;
font-family: monospace;
}
.domain-chip.glob {
background: ${cssManager.bdTheme('#fef3c7', 'rgba(245, 158, 11, 0.15)')};
color: ${cssManager.bdTheme('#92400e', '#fbbf24')};
}
.mono {
font-family: monospace;
font-size: 12px;
}
.protocol-badge {
display: inline-flex;
padding: 1px 6px;
border-radius: 3px;
font-size: 11px;
font-weight: 500;
}
.protocol-badge.http {
background: ${cssManager.bdTheme('#dbeafe', 'rgba(59, 130, 246, 0.2)')};
color: ${cssManager.bdTheme('#2563eb', '#60a5fa')};
}
.protocol-badge.tcp {
background: ${cssManager.bdTheme('#dcfce7', 'rgba(34, 197, 94, 0.2)')};
color: ${cssManager.bdTheme('#16a34a', '#22c55e')};
}
.tls-badge {
display: inline-flex;
padding: 1px 6px;
border-radius: 3px;
font-size: 11px;
font-weight: 500;
}
.tls-badge.auto {
background: ${cssManager.bdTheme('#dcfce7', 'rgba(34, 197, 94, 0.2)')};
color: ${cssManager.bdTheme('#16a34a', '#22c55e')};
}
.tls-badge.custom {
background: ${cssManager.bdTheme('#ffedd5', 'rgba(249, 115, 22, 0.2)')};
color: ${cssManager.bdTheme('#c2410c', '#fb923c')};
}
.engine-badge {
display: inline-flex;
padding: 1px 6px;
border-radius: 3px;
font-size: 11px;
font-weight: 500;
background: ${cssManager.bdTheme('#fae8ff', 'rgba(168, 85, 247, 0.2)')};
color: ${cssManager.bdTheme('#7e22ce', '#c084fc')};
}
.header-pair {
display: inline;
font-family: monospace;
font-size: 12px;
}
/* Feature icons */
.features-row {
display: flex;
flex-wrap: wrap;
gap: 10px;
margin-top: 14px;
padding-top: 12px;
border-top: 1px solid ${cssManager.bdTheme('#f4f4f5', '#1a1a1a')};
}
.feature {
display: flex;
align-items: center;
gap: 4px;
font-size: 11px;
font-weight: 500;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')};
}
.feature-icon {
font-size: 13px;
}
.no-route {
text-align: center;
padding: 24px;
color: ${cssManager.bdTheme('#a1a1aa', '#52525b')};
font-size: 13px;
}
.section.vpn {
border-left-color: ${cssManager.bdTheme('#0891b2', '#06b6d4')};
}
.vpn-badge {
display: inline-flex;
align-items: center;
padding: 2px 8px;
border-radius: 9999px;
font-size: 11px;
font-weight: 600;
white-space: nowrap;
}
.vpn-badge.mandatory {
background: ${cssManager.bdTheme('#fff7ed', 'rgba(249, 115, 22, 0.2)')};
color: ${cssManager.bdTheme('#c2410c', '#fb923c')};
}
.vpn-badge.optional {
background: ${cssManager.bdTheme('#ecfdf5', 'rgba(16, 185, 129, 0.2)')};
color: ${cssManager.bdTheme('#047857', '#34d399')};
}
.vpn-tag {
display: inline-flex;
padding: 1px 6px;
border-radius: 3px;
font-size: 12px;
font-weight: 600;
font-family: monospace;
margin-right: 4px;
margin-bottom: 2px;
background: ${cssManager.bdTheme('#ecfeff', 'rgba(6, 182, 212, 0.15)')};
color: ${cssManager.bdTheme('#0e7490', '#22d3ee')};
}
.card-footer {
display: flex;
flex-direction: row;
justify-content: flex-end;
align-items: center;
gap: 0;
height: 36px;
width: 100%;
box-sizing: border-box;
}
.tile-button {
padding: 0 16px;
height: 100%;
text-align: center;
font-size: 12px;
font-weight: 500;
cursor: pointer;
user-select: none;
transition: all 0.15s ease;
background: transparent;
border: none;
border-left: 1px solid var(--dees-color-border-subtle);
color: var(--dees-color-text-muted);
white-space: nowrap;
display: flex;
align-items: center;
gap: 6px;
font-family: inherit;
}
.tile-button:first-child {
border-left: none;
}
.tile-button:hover {
background: var(--dees-color-hover);
color: var(--dees-color-text-primary);
}
.tile-button.edit:hover {
color: ${cssManager.bdTheme('hsl(217.2 91.2% 59.8%)', 'hsl(213.1 93.9% 67.8%)')};
}
.tile-button.delete:hover {
background: ${cssManager.bdTheme('#fee2e2', 'rgba(239, 68, 68, 0.1)')};
color: ${cssManager.bdTheme('#dc2626', '#f87171')};
}
`,
];
public render(): TemplateResult {
if (!this.route) {
return html`
<dees-tile>
<div class="card-body"><div class="no-route">No route data</div></div>
</dees-tile>
`;
}
const r = this.route;
const isEnabled = r.enabled !== false;
const match = r.match;
const action = r.action;
const security = r.security;
return html`
<dees-tile>
<!-- Header -->
<div slot="header" class="header">
<div class="header-left">
<span class="status-dot ${isEnabled ? 'enabled' : 'disabled'}"></span>
<span class="route-name">${r.name || r.id || 'Unnamed Route'}</span>
</div>
<div class="header-badges">
<span class="badge ${action.type}">${action.type}</span>
<span class="badge ${isEnabled ? 'enabled' : 'disabled'}">${isEnabled ? 'enabled' : 'disabled'}</span>
</div>
</div>
<div class="card-body">
${r.description ? html`<div class="description">${r.description}</div>` : ''}
<div class="meta-row">
${r.tags && r.tags.length > 0
? html`<div class="tags">${r.tags.map((t) => html`<span class="tag">${t}</span>`)}</div>`
: html`<div></div>`}
${r.priority != null ? html`<span class="priority">Priority: ${r.priority}</span>` : ''}
</div>
<!-- Match Section -->
<div class="section match">
<div class="section-label">Match</div>
<div class="field-row">
<span class="field-key">Ports</span>
<span class="field-value mono">${formatPorts(match.ports)}</span>
</div>
${match.domains
? html`
<div class="field-row">
<span class="field-key">Domains</span>
<span class="field-value">${this.renderDomains(match.domains)}</span>
</div>
`
: ''}
${match.path
? html`
<div class="field-row">
<span class="field-key">Path</span>
<span class="field-value mono">${match.path}</span>
</div>
`
: ''}
${match.protocol
? html`
<div class="field-row">
<span class="field-key">Protocol</span>
<span class="field-value">
<span class="protocol-badge ${match.protocol}">${match.protocol}</span>
</span>
</div>
`
: ''}
${match.clientIp && match.clientIp.length > 0
? html`
<div class="field-row">
<span class="field-key">Client</span>
<span class="field-value mono">${match.clientIp.join(', ')}</span>
</div>
`
: ''}
${match.tlsVersion && match.tlsVersion.length > 0
? html`
<div class="field-row">
<span class="field-key">TLS Ver</span>
<span class="field-value">${match.tlsVersion.join(', ')}</span>
</div>
`
: ''}
${match.headers
? html`
<div class="field-row">
<span class="field-key">Headers</span>
<span class="field-value">
${Object.entries(match.headers).map(
([k, v]) => html`<span class="header-pair">${k}=${v}</span> `
)}
</span>
</div>
`
: ''}
</div>
<!-- Action Section -->
<div class="section action">
<div class="section-label">Action</div>
${action.targets && action.targets.length > 0
? html`
<div class="field-row">
<span class="field-key">Targets</span>
<span class="field-value mono">${formatTargets(action.targets).join(', ')}</span>
</div>
`
: ''}
${action.tls
? html`
<div class="field-row">
<span class="field-key">TLS</span>
<span class="field-value">
${action.tls.mode}
${action.tls.certificate
? action.tls.certificate === 'auto'
? html` <span class="tls-badge auto">auto cert</span>`
: html` <span class="tls-badge custom">custom cert</span>`
: ''}
</span>
</div>
`
: ''}
${action.forwardingEngine
? html`
<div class="field-row">
<span class="field-key">Engine</span>
<span class="field-value"><span class="engine-badge">${action.forwardingEngine}</span></span>
</div>
`
: ''}
${action.loadBalancing
? html`
<div class="field-row">
<span class="field-key">LB</span>
<span class="field-value">${action.loadBalancing.algorithm}</span>
</div>
`
: ''}
${action.websocket?.enabled
? html`
<div class="field-row">
<span class="field-key">WS</span>
<span class="field-value"><span class="badge enabled">enabled</span></span>
</div>
`
: ''}
</div>
<!-- Security Section -->
${security
? html`
<div class="section security">
<div class="section-label">Security</div>
${security.ipAllowList && security.ipAllowList.length > 0
? html`
<div class="field-row">
<span class="field-key">Allow</span>
<span class="field-value mono">${security.ipAllowList.join(', ')}</span>
</div>
`
: ''}
${security.ipBlockList && security.ipBlockList.length > 0
? html`
<div class="field-row">
<span class="field-key">Block</span>
<span class="field-value mono">${security.ipBlockList.join(', ')}</span>
</div>
`
: ''}
${security.rateLimit?.enabled
? html`
<div class="field-row">
<span class="field-key">Rate</span>
<span class="field-value">${security.rateLimit.maxRequests} req / ${security.rateLimit.window}s</span>
</div>
`
: ''}
${security.maxConnections
? html`
<div class="field-row">
<span class="field-key">Max Conn</span>
<span class="field-value">${security.maxConnections}</span>
</div>
`
: ''}
</div>
`
: ''}
<!-- VPN Section -->
${this.renderVpn()}
<!-- Linked References Section -->
${this.renderLinked()}
<!-- Feature Icons Row -->
${this.renderFeatures()}
</div>
<!-- Action Buttons -->
${this.showActions ? html`
<div slot="footer" class="card-footer">
<button class="tile-button edit" @click=${(e: Event) => {
e.stopPropagation();
this.dispatchEvent(new CustomEvent('route-edit', {
detail: this.route,
bubbles: true,
composed: true,
}));
}}>Edit</button>
<button class="tile-button delete" @click=${(e: Event) => {
e.stopPropagation();
this.dispatchEvent(new CustomEvent('route-delete', {
detail: this.route,
bubbles: true,
composed: true,
}));
}}>Delete</button>
</div>
` : ''}
</dees-tile>
`;
}
private renderDomains(domains: string | string[]): TemplateResult {
const list = Array.isArray(domains) ? domains : [domains];
return html`${list.map(
(d) =>
html`<span class="domain-chip ${d.includes('*') ? 'glob' : ''}">${d}</span>`
)}`;
}
private renderVpn(): TemplateResult {
if (!this.route?.vpnOnly) return html``;
return html`
<div class="section vpn">
<div class="section-label">VPN Access</div>
<div class="field-row">
<span class="field-key">Mode</span>
<span class="field-value">
<span class="vpn-badge mandatory">VPN Only</span>
</span>
</div>
</div>
`;
}
private renderLinked(): TemplateResult {
const meta = this.route?.metadata;
if (!meta) return html``;
const hasProfile = !!meta.sourceProfileName;
const hasTarget = !!meta.networkTargetName;
if (!hasProfile && !hasTarget) return html``;
return html`
<div class="section linked">
<div class="section-label">Linked</div>
${hasProfile
? html`
<div class="field-row">
<span class="field-key">Profile</span>
<span class="field-value"><span class="linked-name">${meta.sourceProfileName}</span></span>
</div>
`
: ''}
${hasTarget
? html`
<div class="field-row">
<span class="field-key">Target</span>
<span class="field-value"><span class="linked-name">${meta.networkTargetName}</span></span>
</div>
`
: ''}
</div>
`;
}
private renderFeatures(): TemplateResult {
if (!this.route) return html``;
const features: TemplateResult[] = [];
const action = this.route.action;
const security = this.route.security;
const headers = this.route.headers;
const meta = this.route.metadata;
if (action.tls) {
features.push(html`<span class="feature"><span class="feature-icon">&#x1f512;</span>TLS</span>`);
}
if (action.websocket?.enabled) {
features.push(html`<span class="feature"><span class="feature-icon">&#x2194;</span>WS</span>`);
}
if (action.loadBalancing) {
features.push(html`<span class="feature"><span class="feature-icon">&#x2696;</span>LB</span>`);
}
if (security) {
features.push(html`<span class="feature"><span class="feature-icon">&#x1f6e1;</span>Security</span>`);
}
if (headers) {
features.push(html`<span class="feature"><span class="feature-icon">&#x2699;</span>Headers</span>`);
}
if (this.route?.vpnOnly) {
features.push(html`<span class="feature"><span class="feature-icon">&#x1f510;</span>VPN</span>`);
}
if (meta?.sourceProfileName || meta?.networkTargetName) {
features.push(html`<span class="feature"><span class="feature-icon">&#x1f517;</span>Linked</span>`);
}
if (features.length === 0) return html``;
return html`<div class="features-row">${features}</div>`;
}
}
+330
View File
@@ -0,0 +1,330 @@
import {
DeesElement,
customElement,
html,
css,
cssManager,
property,
state,
type TemplateResult,
} from '@design.estate/dees-element';
import type { IRouteConfig, TRouteActionType } from './sz-route-card.js';
declare global {
interface HTMLElementTagNameMap {
'sz-route-list-view': SzRouteListView;
}
}
@customElement('sz-route-list-view')
export class SzRouteListView extends DeesElement {
public static demo = () => html`
<div style="padding: 24px; max-width: 1200px;">
<sz-route-list-view
.routes=${[
{
name: 'HTTPS Gateway',
description: 'Main web gateway with TLS termination',
enabled: true,
tags: ['web', 'https', 'production'],
match: { ports: 443, domains: ['*.example.com', 'serve.zone'], protocol: 'http' as const },
action: {
type: 'forward' as const,
targets: [{ host: '10.0.0.1', port: 8080 }],
tls: { mode: 'terminate' as const, certificate: 'auto' as const },
},
},
{
name: 'SMTP Inbound',
description: 'Email relay for incoming mail',
enabled: true,
tags: ['email', 'smtp'],
match: { ports: 25, domains: 'mail.serve.zone' },
action: {
type: 'forward' as const,
targets: [{ host: '10.0.1.5', port: 25 }],
},
},
{
name: 'WebSocket API',
description: 'Real-time WebSocket connections',
enabled: true,
tags: ['web', 'api'],
match: { ports: 443, domains: 'ws.example.com', path: '/ws/*' },
action: {
type: 'forward' as const,
targets: [{ host: '10.0.0.3', port: 9090 }],
websocket: { enabled: true },
tls: { mode: 'terminate' as const, certificate: 'auto' as const },
},
},
{
name: 'Maintenance Page',
enabled: false,
tags: ['web'],
match: { ports: [80, 443], domains: 'old.example.com' },
action: { type: 'socket-handler' as const },
},
] satisfies IRouteConfig[]}
></sz-route-list-view>
</div>
`;
public static demoGroups = ['Routes'];
@property({ type: Array })
public accessor routes: IRouteConfig[] = [];
@property({ attribute: false })
public accessor showActionsFilter: ((route: IRouteConfig) => boolean) | null = null;
@state()
private accessor searchQuery: string = '';
@state()
private accessor actionFilter: TRouteActionType | 'all' = 'all';
@state()
private accessor enabledFilter: 'all' | 'enabled' | 'disabled' = 'all';
private get filteredRoutes(): IRouteConfig[] {
return this.routes.filter((route) => {
// Action type filter
if (this.actionFilter !== 'all' && route.action.type !== this.actionFilter) return false;
// Enabled/disabled filter
if (this.enabledFilter === 'enabled' && route.enabled === false) return false;
if (this.enabledFilter === 'disabled' && route.enabled !== false) return false;
// Search query
if (this.searchQuery) {
const q = this.searchQuery.toLowerCase();
return this.routeMatchesSearch(route, q);
}
return true;
});
}
private routeMatchesSearch(route: IRouteConfig, q: string): boolean {
// Name and description
if (route.name?.toLowerCase().includes(q)) return true;
if (route.description?.toLowerCase().includes(q)) return true;
// Domains
if (route.match.domains) {
const domains = Array.isArray(route.match.domains) ? route.match.domains : [route.match.domains];
if (domains.some((d) => d.toLowerCase().includes(q))) return true;
}
// Ports
const portsStr = this.formatPortsForSearch(route.match.ports);
if (portsStr.includes(q)) return true;
// Path
if (route.match.path?.toLowerCase().includes(q)) return true;
// Client IPs
if (route.match.clientIp?.some((ip) => ip.includes(q))) return true;
// Targets
if (route.action.targets) {
for (const t of route.action.targets) {
const hosts = Array.isArray(t.host) ? t.host : [t.host];
if (hosts.some((h) => h.toLowerCase().includes(q))) return true;
}
}
// Tags
if (route.tags?.some((t) => t.toLowerCase().includes(q))) return true;
return false;
}
private formatPortsForSearch(ports: import('./sz-route-card.js').TPortRange): string {
if (typeof ports === 'number') return String(ports);
if (Array.isArray(ports)) {
return ports
.map((p) => (typeof p === 'number' ? String(p) : `${p.from}-${p.to}`))
.join(' ');
}
return String(ports);
}
public static styles = [
cssManager.defaultStyles,
css`
:host {
display: block;
}
.filter-bar {
display: flex;
flex-wrap: wrap;
gap: 12px;
align-items: center;
margin-bottom: 12px;
}
.search-input {
flex: 1;
min-width: 200px;
padding: 8px 12px;
background: ${cssManager.bdTheme('#ffffff', '#09090b')};
border: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')};
border-radius: 6px;
font-size: 14px;
color: ${cssManager.bdTheme('#18181b', '#fafafa')};
outline: none;
transition: border-color 200ms ease;
}
.search-input::placeholder {
color: ${cssManager.bdTheme('#a1a1aa', '#52525b')};
}
.search-input:focus {
border-color: ${cssManager.bdTheme('#2563eb', '#3b82f6')};
}
.chip-group {
display: flex;
gap: 4px;
}
.chip {
padding: 6px 12px;
background: transparent;
border: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')};
border-radius: 9999px;
font-size: 12px;
font-weight: 500;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')};
cursor: pointer;
transition: all 200ms ease;
white-space: nowrap;
}
.chip:hover {
background: ${cssManager.bdTheme('#f4f4f5', '#18181b')};
color: ${cssManager.bdTheme('#18181b', '#fafafa')};
}
.chip.active {
background: ${cssManager.bdTheme('#18181b', '#fafafa')};
color: ${cssManager.bdTheme('#fafafa', '#18181b')};
border-color: ${cssManager.bdTheme('#18181b', '#fafafa')};
}
.results-count {
font-size: 13px;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')};
margin-bottom: 16px;
}
.grid {
display: grid;
grid-template-columns: repeat(auto-fill, minmax(420px, 1fr));
gap: 16px;
}
.grid sz-route-card {
cursor: pointer;
}
.empty-state {
text-align: center;
padding: 48px 24px;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')};
font-size: 14px;
}
.empty-state-icon {
font-size: 32px;
margin-bottom: 12px;
opacity: 0.5;
}
`,
];
public render(): TemplateResult {
const filtered = this.filteredRoutes;
return html`
<div class="filter-bar">
<input
class="search-input"
type="text"
placeholder="Search routes by domain, IP, port, path, or tag..."
.value=${this.searchQuery}
@input=${(e: InputEvent) => {
this.searchQuery = (e.target as HTMLInputElement).value;
}}
/>
<div class="chip-group">
${(['all', 'forward', 'socket-handler'] as const).map(
(type) => html`
<button
class="chip ${this.actionFilter === type ? 'active' : ''}"
@click=${() => {
this.actionFilter = type;
}}
>
${type === 'all' ? 'All' : type === 'forward' ? 'Forward' : 'Socket Handler'}
</button>
`
)}
</div>
<div class="chip-group">
${(['all', 'enabled', 'disabled'] as const).map(
(status) => html`
<button
class="chip ${this.enabledFilter === status ? 'active' : ''}"
@click=${() => {
this.enabledFilter = status;
}}
>
${status.charAt(0).toUpperCase() + status.slice(1)}
</button>
`
)}
</div>
</div>
<div class="results-count">
Showing ${filtered.length} of ${this.routes.length} routes
</div>
${filtered.length > 0
? html`
<div class="grid">
${filtered.map(
(route) => html`
<sz-route-card
.route=${route}
.showActions=${this.showActionsFilter?.(route) ?? false}
@click=${() => this.handleRouteClick(route)}
></sz-route-card>
`
)}
</div>
`
: html`
<div class="empty-state">
<div class="empty-state-icon">&#x1f50d;</div>
<div>No routes match your filters</div>
</div>
`}
`;
}
private handleRouteClick(route: IRouteConfig) {
this.dispatchEvent(
new CustomEvent('route-click', {
detail: route,
bubbles: true,
composed: true,
})
);
}
}
+192
View File
@@ -48,6 +48,9 @@ export interface IServiceConfig {
memoryLimit: string; memoryLimit: string;
restartPolicy: 'always' | 'on-failure' | 'never'; restartPolicy: 'always' | 'on-failure' | 'never';
networkMode: string; networkMode: string;
enableMongoDB: boolean;
enableS3: boolean;
enableClickHouse: boolean;
} }
@customElement('sz-service-create-view') @customElement('sz-service-create-view')
@@ -104,6 +107,15 @@ export class SzServiceCreateView extends DeesElement {
@state() @state()
private accessor showAdvanced: boolean = false; private accessor showAdvanced: boolean = false;
@state()
private accessor enableMongoDB: boolean = false;
@state()
private accessor enableS3: boolean = false;
@state()
private accessor enableClickHouse: boolean = false;
public static styles = [ public static styles = [
cssManager.defaultStyles, cssManager.defaultStyles,
css` css`
@@ -312,6 +324,105 @@ export class SzServiceCreateView extends DeesElement {
accent-color: ${cssManager.bdTheme('#3b82f6', '#60a5fa')}; accent-color: ${cssManager.bdTheme('#3b82f6', '#60a5fa')};
} }
.platform-toggle-list {
display: flex;
flex-direction: column;
gap: 12px;
}
.platform-toggle-item {
display: flex;
align-items: center;
justify-content: space-between;
padding: 12px 16px;
background: ${cssManager.bdTheme('#f4f4f5', '#18181b')};
border-radius: 8px;
transition: background 200ms ease;
}
.platform-toggle-item:has(input:checked) {
background: ${cssManager.bdTheme('#eff6ff', 'rgba(59, 130, 246, 0.1)')};
}
.platform-toggle-info {
display: flex;
align-items: center;
gap: 12px;
}
.platform-toggle-icon {
width: 36px;
height: 36px;
border-radius: 8px;
background: ${cssManager.bdTheme('#ffffff', '#27272a')};
display: flex;
align-items: center;
justify-content: center;
}
.platform-toggle-icon svg {
width: 20px;
height: 20px;
color: ${cssManager.bdTheme('#18181b', '#fafafa')};
}
.platform-toggle-name {
font-size: 14px;
font-weight: 500;
color: ${cssManager.bdTheme('#18181b', '#fafafa')};
}
.platform-toggle-desc {
font-size: 12px;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')};
margin-top: 2px;
}
.toggle-switch {
position: relative;
width: 44px;
height: 24px;
flex-shrink: 0;
}
.toggle-switch input {
opacity: 0;
width: 0;
height: 0;
}
.toggle-slider {
position: absolute;
cursor: pointer;
top: 0;
left: 0;
right: 0;
bottom: 0;
background: ${cssManager.bdTheme('#d4d4d8', '#3f3f46')};
border-radius: 12px;
transition: background 200ms ease;
}
.toggle-slider::before {
content: '';
position: absolute;
height: 18px;
width: 18px;
left: 3px;
bottom: 3px;
background: white;
border-radius: 50%;
transition: transform 200ms ease;
}
.toggle-switch input:checked + .toggle-slider {
background: ${cssManager.bdTheme('#3b82f6', '#60a5fa')};
}
.toggle-switch input:checked + .toggle-slider::before {
transform: translateX(20px);
}
.actions { .actions {
display: flex; display: flex;
justify-content: flex-end; justify-content: flex-end;
@@ -536,6 +647,81 @@ export class SzServiceCreateView extends DeesElement {
</button> </button>
</div> </div>
<!-- Platform Services -->
<div class="section">
<div class="section-title">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<rect x="2" y="2" width="20" height="8" rx="2" ry="2"></rect>
<rect x="2" y="14" width="20" height="8" rx="2" ry="2"></rect>
<line x1="6" y1="6" x2="6.01" y2="6"></line>
<line x1="6" y1="18" x2="6.01" y2="18"></line>
</svg>
Platform Services
</div>
<div class="form-hint" style="margin-bottom: 12px;">
Enable managed infrastructure services for this deployment. Resources are automatically provisioned and connection details injected as environment variables.
</div>
<div class="platform-toggle-list">
<label class="platform-toggle-item">
<div class="platform-toggle-info">
<div class="platform-toggle-icon">
<svg viewBox="0 0 24 24" fill="currentColor"><path d="M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm-1 17.93c-3.95-.49-7-3.85-7-7.93 0-.62.08-1.21.21-1.79L9 15v1c0 1.1.9 2 2 2v1.93zm6.9-2.54c-.26-.81-1-1.39-1.9-1.39h-1v-3c0-.55-.45-1-1-1H8v-2h2c.55 0 1-.45 1-1V7h2c1.1 0 2-.9 2-2v-.41c2.93 1.19 5 4.06 5 7.41 0 2.08-.8 3.97-2.1 5.39z"/></svg>
</div>
<div>
<div class="platform-toggle-name">MongoDB</div>
<div class="platform-toggle-desc">Document database with auto-provisioned credentials</div>
</div>
</div>
<div class="toggle-switch">
<input
type="checkbox"
?checked=${this.enableMongoDB}
@change=${(e: Event) => this.enableMongoDB = (e.target as HTMLInputElement).checked}
>
<span class="toggle-slider"></span>
</div>
</label>
<label class="platform-toggle-item">
<div class="platform-toggle-info">
<div class="platform-toggle-icon">
<svg viewBox="0 0 24 24" fill="currentColor"><path d="M21 16.5c0 .38-.21.71-.53.88l-7.9 4.44c-.16.12-.36.18-.57.18-.21 0-.41-.06-.57-.18l-7.9-4.44A.991.991 0 0 1 3 16.5v-9c0-.38.21-.71.53-.88l7.9-4.44c.16-.12.36-.18.57-.18.21 0 .41.06.57.18l7.9 4.44c.32.17.53.5.53.88v9z"/></svg>
</div>
<div>
<div class="platform-toggle-name">S3 Storage (MinIO)</div>
<div class="platform-toggle-desc">Object storage bucket with auto-provisioned access keys</div>
</div>
</div>
<div class="toggle-switch">
<input
type="checkbox"
?checked=${this.enableS3}
@change=${(e: Event) => this.enableS3 = (e.target as HTMLInputElement).checked}
>
<span class="toggle-slider"></span>
</div>
</label>
<label class="platform-toggle-item">
<div class="platform-toggle-info">
<div class="platform-toggle-icon">
<svg viewBox="0 0 24 24" fill="currentColor"><rect x="2" y="2" width="6" height="20"/><rect x="9" y="7" width="6" height="15"/><rect x="16" y="12" width="6" height="10"/></svg>
</div>
<div>
<div class="platform-toggle-name">ClickHouse</div>
<div class="platform-toggle-desc">Analytics database with auto-provisioned credentials</div>
</div>
</div>
<div class="toggle-switch">
<input
type="checkbox"
?checked=${this.enableClickHouse}
@change=${(e: Event) => this.enableClickHouse = (e.target as HTMLInputElement).checked}
>
<span class="toggle-slider"></span>
</div>
</label>
</div>
</div>
<!-- Advanced Options Toggle --> <!-- Advanced Options Toggle -->
<button <button
class="toggle-advanced ${this.showAdvanced ? 'open' : ''}" class="toggle-advanced ${this.showAdvanced ? 'open' : ''}"
@@ -750,6 +936,9 @@ export class SzServiceCreateView extends DeesElement {
memoryLimit: this.memoryLimit, memoryLimit: this.memoryLimit,
restartPolicy: this.restartPolicy, restartPolicy: this.restartPolicy,
networkMode: this.networkMode, networkMode: this.networkMode,
enableMongoDB: this.enableMongoDB,
enableS3: this.enableS3,
enableClickHouse: this.enableClickHouse,
}; };
this.dispatchEvent(new CustomEvent('create-service', { this.dispatchEvent(new CustomEvent('create-service', {
@@ -771,5 +960,8 @@ export class SzServiceCreateView extends DeesElement {
this.restartPolicy = 'always'; this.restartPolicy = 'always';
this.networkMode = 'bridge'; this.networkMode = 'bridge';
this.showAdvanced = false; this.showAdvanced = false;
this.enableMongoDB = false;
this.enableS3 = false;
this.enableClickHouse = false;
} }
} }
+132 -142
View File
@@ -11,8 +11,6 @@ import {
import type { IExecutionEnvironment } from '@design.estate/dees-catalog'; import type { IExecutionEnvironment } from '@design.estate/dees-catalog';
import './sz-stat-card.js';
declare global { declare global {
interface HTMLElementTagNameMap { interface HTMLElementTagNameMap {
'sz-service-detail-view': SzServiceDetailView; 'sz-service-detail-view': SzServiceDetailView;
@@ -218,31 +216,97 @@ export class SzServiceDetailView extends DeesElement {
gap: 24px; gap: 24px;
} }
.card { .card-header {
background: ${cssManager.bdTheme('#ffffff', '#09090b')}; height: 36px;
border: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')}; display: flex;
border-radius: 8px; align-items: center;
overflow: hidden; padding: 0 8px 0 16px;
width: 100%;
box-sizing: border-box;
} }
.card-header { .card-heading {
flex: 1;
display: flex; display: flex;
justify-content: space-between; align-items: baseline;
align-items: center; gap: 8px;
padding: 16px; min-width: 0;
border-bottom: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')};
} }
.card-title { .card-title {
font-size: 16px; font-weight: 500;
font-weight: 600; font-size: 13px;
color: ${cssManager.bdTheme('#18181b', '#fafafa')}; letter-spacing: -0.01em;
color: var(--dees-color-text-secondary);
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
} }
.card-subtitle { .card-subtitle {
font-size: 13px; font-size: 12px;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')}; color: var(--dees-color-text-muted);
margin-top: 2px; letter-spacing: -0.01em;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
.card-footer {
display: flex;
flex-direction: row;
justify-content: flex-end;
align-items: center;
gap: 0;
height: 36px;
width: 100%;
box-sizing: border-box;
}
.tile-button {
padding: 0 16px;
height: 100%;
text-align: center;
font-size: 12px;
font-weight: 500;
cursor: pointer;
user-select: none;
transition: all 0.15s ease;
background: transparent;
border: none;
border-left: 1px solid var(--dees-color-border-subtle);
color: var(--dees-color-text-muted);
white-space: nowrap;
display: flex;
align-items: center;
gap: 6px;
}
.tile-button:first-child {
border-left: none;
}
.tile-button:hover {
background: var(--dees-color-hover);
color: var(--dees-color-text-primary);
}
.tile-button.primary {
color: ${cssManager.bdTheme('hsl(217.2 91.2% 59.8%)', 'hsl(213.1 93.9% 67.8%)')};
font-weight: 600;
}
.tile-button.primary:hover {
background: ${cssManager.bdTheme('hsl(217.2 91.2% 59.8% / 0.08)', 'hsl(213.1 93.9% 67.8% / 0.08)')};
color: ${cssManager.bdTheme('hsl(217.2 91.2% 50%)', 'hsl(213.1 93.9% 75%)')};
}
.tile-button.danger {
color: ${cssManager.bdTheme('#dc2626', '#ef4444')};
}
.tile-button.danger:hover {
background: ${cssManager.bdTheme('#fee2e2', 'rgba(239, 68, 68, 0.1)')};
} }
.card-content { .card-content {
@@ -396,68 +460,6 @@ export class SzServiceDetailView extends DeesElement {
color: ${cssManager.bdTheme('#18181b', '#fafafa')}; color: ${cssManager.bdTheme('#18181b', '#fafafa')};
} }
.logs-header {
display: flex;
justify-content: space-between;
align-items: center;
}
.logs-actions {
display: flex;
gap: 8px;
align-items: center;
}
.stream-button {
display: inline-flex;
align-items: center;
gap: 6px;
padding: 6px 12px;
background: ${cssManager.bdTheme('#2563eb', '#3b82f6')};
border: none;
border-radius: 4px;
font-size: 13px;
font-weight: 500;
color: white;
cursor: pointer;
}
.stream-button.streaming {
background: ${cssManager.bdTheme('#dc2626', '#ef4444')};
}
.clear-button {
padding: 6px 12px;
background: transparent;
border: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')};
border-radius: 4px;
font-size: 13px;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')};
cursor: pointer;
}
.logs-container {
padding: 16px;
font-family: monospace;
font-size: 12px;
max-height: 300px;
overflow-y: auto;
background: ${cssManager.bdTheme('#fafafa', '#0a0a0a')};
}
.log-entry {
padding: 2px 0;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')};
white-space: pre-wrap;
word-break: break-all;
}
.empty-logs {
padding: 24px;
text-align: center;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')};
}
.tag-badge { .tag-badge {
display: inline-flex; display: inline-flex;
padding: 2px 8px; padding: 2px 8px;
@@ -580,12 +582,11 @@ export class SzServiceDetailView extends DeesElement {
<div class="content"> <div class="content">
<div class="main-content"> <div class="main-content">
<div class="card"> <dees-tile>
<div class="card-header"> <div slot="header" class="card-header">
<div> <div class="card-heading">
<div class="card-title">Service Details</div> <span class="card-title">Service Details</span>
</div> </div>
<button class="action-button" style="width: auto; padding: 6px 12px;" @click=${() => this.handleEdit()}>Edit</button>
</div> </div>
<div class="card-content"> <div class="card-content">
<div class="detail-list"> <div class="detail-list">
@@ -619,43 +620,30 @@ export class SzServiceDetailView extends DeesElement {
</div> </div>
</div> </div>
</div> </div>
<div slot="footer" class="card-footer">
<button class="tile-button primary" @click=${() => this.handleEdit()}>Edit</button>
</div> </div>
</dees-tile>
<div class="card"> <dees-chart-log
<div class="card-header"> .label=${'Service Logs'}
<div class="logs-header" style="width: 100%;"> .logEntries=${this.logs.map(log => ({
<div> timestamp: log.timestamp?.includes?.('T') ? log.timestamp : new Date(log.timestamp || Date.now()).toISOString(),
<div class="card-title">Logs</div> level: (log.level || 'info') as 'debug' | 'info' | 'warn' | 'error',
<div class="card-subtitle">Container logs</div> message: log.message,
</div> }))}
<div class="logs-actions"> .autoScroll=${true}
<button class="stream-button ${this.streaming ? 'streaming' : ''}" @click=${() => this.toggleStreaming()}> .maxEntries=${2000}
<svg width="14" height="14" viewBox="0 0 24 24" fill="currentColor"> .showMetrics=${true}
${this.streaming ></dees-chart-log>
? html`<rect x="6" y="6" width="12" height="12" rx="1"/>`
: html`<polygon points="5,3 19,12 5,21"/>`
}
</svg>
${this.streaming ? 'Stop' : 'Stream'}
</button>
<button class="clear-button" @click=${() => this.handleClearLogs()}>Clear logs</button>
</div>
</div>
</div>
<div class="logs-container">
${this.logs.length > 0 ? this.logs.map(log => html`
<div class="log-entry">${log.timestamp} ${log.message}</div>
`) : html`
<div class="empty-logs">Click "Stream" to start live log streaming</div>
`}
</div>
</div>
</div> </div>
<div class="sidebar"> <div class="sidebar">
<div class="card"> <dees-tile>
<div class="card-header"> <div slot="header" class="card-header">
<div class="card-title">Live stats</div> <div class="card-heading">
<span class="card-title">Live stats</span>
</div>
</div> </div>
<div class="card-content"> <div class="card-content">
<div class="stats-grid"> <div class="stats-grid">
@@ -678,13 +666,13 @@ export class SzServiceDetailView extends DeesElement {
</div> </div>
</div> </div>
</div> </div>
</div> </dees-tile>
<div class="card"> <dees-tile>
<div class="card-header"> <div slot="header" class="card-header">
<div> <div class="card-heading">
<div class="card-title">Actions</div> <span class="card-title">Actions</span>
<div class="card-subtitle">Manage service state</div> <span class="card-subtitle">Manage service state</span>
</div> </div>
</div> </div>
<div class="card-content"> <div class="card-content">
@@ -704,13 +692,13 @@ export class SzServiceDetailView extends DeesElement {
<button class="action-button danger" @click=${() => this.handleAction('delete')}>Delete Service</button> <button class="action-button danger" @click=${() => this.handleAction('delete')}>Delete Service</button>
</div> </div>
</div> </div>
</div> </dees-tile>
<div class="card"> <dees-tile>
<div class="card-header"> <div slot="header" class="card-header">
<div> <div class="card-heading">
<div class="card-title">Image Source</div> <span class="card-title">Image Source</span>
<div class="card-subtitle">${this.service.registry === 'Docker Hub' ? 'External container registry' : 'Onebox registry'}</div> <span class="card-subtitle">${this.service.registry === 'Docker Hub' ? 'External container registry' : 'Onebox registry'}</span>
</div> </div>
</div> </div>
<div class="card-content"> <div class="card-content">
@@ -729,21 +717,14 @@ export class SzServiceDetailView extends DeesElement {
</div> </div>
</div> </div>
</div> </div>
</div> </dees-tile>
<div class="card"> <dees-tile>
<div class="card-header"> <div slot="header" class="card-header">
<div> <div class="card-heading">
<div class="card-title">Backups</div> <span class="card-title">Backups</span>
<div class="card-subtitle">Create and manage service backups</div> <span class="card-subtitle">Create and manage service backups</span>
</div> </div>
<button class="action-button" style="width: auto; padding: 6px 12px;" @click=${() => this.handleCreateBackup()}>
<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" style="margin-right: 4px;">
<line x1="12" y1="5" x2="12" y2="19"></line>
<line x1="5" y1="12" x2="19" y2="12"></line>
</svg>
Create Backup
</button>
</div> </div>
<div class="card-content"> <div class="card-content">
<div class="backup-list"> <div class="backup-list">
@@ -778,7 +759,16 @@ export class SzServiceDetailView extends DeesElement {
`)} `)}
</div> </div>
</div> </div>
<div slot="footer" class="card-footer">
<button class="tile-button primary" @click=${() => this.handleCreateBackup()}>
<svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<line x1="12" y1="5" x2="12" y2="19"></line>
<line x1="5" y1="12" x2="19" y2="12"></line>
</svg>
Create Backup
</button>
</div> </div>
</dees-tile>
</div> </div>
</div> </div>
`; `;
+89 -66
View File
@@ -65,71 +65,94 @@ export class SzServicesBackupsView extends DeesElement {
display: block; display: block;
} }
.section { dees-tile {
background: ${cssManager.bdTheme('#ffffff', '#09090b')}; display: block;
border: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')};
border-radius: 8px;
margin-bottom: 24px; margin-bottom: 24px;
overflow: hidden;
} }
.section-header { .section-header {
height: 36px;
display: flex; display: flex;
justify-content: space-between; align-items: center;
align-items: flex-start; padding: 0 16px;
padding: 16px; width: 100%;
border-bottom: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')}; box-sizing: border-box;
} }
.section-info { .section-heading {
flex: 1;
display: flex; display: flex;
flex-direction: column; align-items: baseline;
gap: 4px; gap: 8px;
min-width: 0;
} }
.section-title { .section-title {
font-size: 16px; font-weight: 500;
font-weight: 600; font-size: 13px;
color: ${cssManager.bdTheme('#18181b', '#fafafa')}; letter-spacing: -0.01em;
color: var(--dees-color-text-secondary);
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
} }
.section-subtitle { .section-subtitle {
font-size: 13px; font-size: 12px;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')}; color: var(--dees-color-text-muted);
letter-spacing: -0.01em;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
} }
.header-actions { .section-footer {
display: flex; display: flex;
gap: 8px; flex-direction: row;
justify-content: flex-end;
align-items: center;
gap: 0;
height: 36px;
width: 100%;
box-sizing: border-box;
} }
.action-button { .tile-button {
display: inline-flex; padding: 0 16px;
height: 100%;
text-align: center;
font-size: 12px;
font-weight: 500;
cursor: pointer;
user-select: none;
transition: all 0.15s ease;
background: transparent;
border: none;
border-left: 1px solid var(--dees-color-border-subtle);
color: var(--dees-color-text-muted);
white-space: nowrap;
display: flex;
align-items: center; align-items: center;
gap: 6px; gap: 6px;
padding: 8px 14px;
background: ${cssManager.bdTheme('#ffffff', '#09090b')};
border: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')};
border-radius: 6px;
font-size: 13px;
font-weight: 500;
color: ${cssManager.bdTheme('#18181b', '#fafafa')};
cursor: pointer;
transition: all 200ms ease;
} }
.action-button:hover { .tile-button:first-child {
background: ${cssManager.bdTheme('#f4f4f5', '#18181b')}; border-left: none;
} }
.action-button.primary { .tile-button:hover {
background: ${cssManager.bdTheme('#18181b', '#fafafa')}; background: var(--dees-color-hover);
color: ${cssManager.bdTheme('#fafafa', '#18181b')}; color: var(--dees-color-text-primary);
border: none;
} }
.action-button.primary:hover { .tile-button.primary {
opacity: 0.9; color: ${cssManager.bdTheme('hsl(217.2 91.2% 59.8%)', 'hsl(213.1 93.9% 67.8%)')};
font-weight: 600;
}
.tile-button.primary:hover {
background: ${cssManager.bdTheme('hsl(217.2 91.2% 59.8% / 0.08)', 'hsl(213.1 93.9% 67.8% / 0.08)')};
color: ${cssManager.bdTheme('hsl(217.2 91.2% 50%)', 'hsl(213.1 93.9% 75%)')};
} }
.table-header { .table-header {
@@ -253,28 +276,11 @@ export class SzServicesBackupsView extends DeesElement {
public render(): TemplateResult { public render(): TemplateResult {
return html` return html`
<div class="section"> <dees-tile>
<div class="section-header"> <div slot="header" class="section-header">
<div class="section-info"> <div class="section-heading">
<div class="section-title">Backup Schedules</div> <span class="section-title">Backup Schedules</span>
<div class="section-subtitle">Configure automated backup schedules for your services</div> <span class="section-subtitle">Configure automated backup schedules for your services</span>
</div>
<div class="header-actions">
<button class="action-button" @click=${() => this.handleImport()}>
<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<path d="M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4"/>
<polyline points="17 8 12 3 7 8"/>
<line x1="12" y1="3" x2="12" y2="15"/>
</svg>
Import Backup
</button>
<button class="action-button primary" @click=${() => this.handleCreateSchedule()}>
<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<line x1="12" y1="5" x2="12" y2="19"></line>
<line x1="5" y1="12" x2="19" y2="12"></line>
</svg>
Create Schedule
</button>
</div> </div>
</div> </div>
<div class="table-header schedules-header"> <div class="table-header schedules-header">
@@ -318,13 +324,30 @@ export class SzServicesBackupsView extends DeesElement {
</span> </span>
</div> </div>
`)} `)}
<div slot="footer" class="section-footer">
<button class="tile-button" @click=${() => this.handleImport()}>
<svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<path d="M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4"/>
<polyline points="17 8 12 3 7 8"/>
<line x1="12" y1="3" x2="12" y2="15"/>
</svg>
Import Backup
</button>
<button class="tile-button primary" @click=${() => this.handleCreateSchedule()}>
<svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<line x1="12" y1="5" x2="12" y2="19"></line>
<line x1="5" y1="12" x2="19" y2="12"></line>
</svg>
Create Schedule
</button>
</div> </div>
</dees-tile>
<div class="section"> <dees-tile>
<div class="section-header"> <div slot="header" class="section-header">
<div class="section-info"> <div class="section-heading">
<div class="section-title">All Backups</div> <span class="section-title">All Backups</span>
<div class="section-subtitle">Browse and manage all backups across services</div> <span class="section-subtitle">Browse and manage all backups across services</span>
</div> </div>
</div> </div>
<div class="table-header backups-header"> <div class="table-header backups-header">
@@ -358,7 +381,7 @@ export class SzServicesBackupsView extends DeesElement {
</span> </span>
</div> </div>
`)} `)}
</div> </dees-tile>
`; `;
} }
+133 -73
View File
@@ -72,28 +72,98 @@ export class SzSettingsView extends DeesElement {
display: block; display: block;
} }
.section { dees-tile {
background: ${cssManager.bdTheme('#ffffff', '#09090b')}; display: block;
border: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')};
border-radius: 8px;
padding: 20px;
margin-bottom: 24px; margin-bottom: 24px;
} }
.section-header { .section-header {
margin-bottom: 16px; height: 36px;
display: flex;
align-items: center;
padding: 0 16px;
width: 100%;
box-sizing: border-box;
}
.section-heading {
flex: 1;
display: flex;
align-items: baseline;
gap: 8px;
min-width: 0;
} }
.section-title { .section-title {
font-size: 16px; font-weight: 500;
font-weight: 600; font-size: 13px;
color: ${cssManager.bdTheme('#18181b', '#fafafa')}; letter-spacing: -0.01em;
color: var(--dees-color-text-secondary);
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
} }
.section-subtitle { .section-subtitle {
font-size: 13px; font-size: 12px;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')}; color: var(--dees-color-text-muted);
margin-top: 2px; letter-spacing: -0.01em;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
.section-content {
padding: 20px;
}
.section-footer {
display: flex;
flex-direction: row;
justify-content: flex-end;
align-items: center;
gap: 0;
height: 36px;
width: 100%;
box-sizing: border-box;
}
.tile-button {
padding: 0 16px;
height: 100%;
text-align: center;
font-size: 12px;
font-weight: 500;
cursor: pointer;
user-select: none;
transition: all 0.15s ease;
background: transparent;
border: none;
border-left: 1px solid var(--dees-color-border-subtle);
color: var(--dees-color-text-muted);
white-space: nowrap;
display: flex;
align-items: center;
gap: 6px;
}
.tile-button:first-child {
border-left: none;
}
.tile-button:hover {
background: var(--dees-color-hover);
color: var(--dees-color-text-primary);
}
.tile-button.primary {
color: ${cssManager.bdTheme('hsl(217.2 91.2% 59.8%)', 'hsl(213.1 93.9% 67.8%)')};
font-weight: 600;
}
.tile-button.primary:hover {
background: ${cssManager.bdTheme('hsl(217.2 91.2% 59.8% / 0.08)', 'hsl(213.1 93.9% 67.8% / 0.08)')};
color: ${cssManager.bdTheme('hsl(217.2 91.2% 50%)', 'hsl(213.1 93.9% 75%)')};
} }
.form-group { .form-group {
@@ -224,53 +294,19 @@ export class SzSettingsView extends DeesElement {
margin-bottom: 4px; margin-bottom: 4px;
} }
.actions {
display: flex;
justify-content: flex-end;
gap: 12px;
padding-top: 16px;
border-top: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')};
margin-top: 24px;
}
.button {
padding: 10px 20px;
border-radius: 6px;
font-size: 14px;
font-weight: 500;
cursor: pointer;
transition: all 200ms ease;
}
.button.secondary {
background: ${cssManager.bdTheme('#ffffff', '#09090b')};
border: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')};
color: ${cssManager.bdTheme('#18181b', '#fafafa')};
}
.button.secondary:hover {
background: ${cssManager.bdTheme('#f4f4f5', '#18181b')};
}
.button.primary {
background: ${cssManager.bdTheme('#18181b', '#fafafa')};
border: none;
color: ${cssManager.bdTheme('#fafafa', '#18181b')};
}
.button.primary:hover {
opacity: 0.9;
}
`, `,
]; ];
public render(): TemplateResult { public render(): TemplateResult {
return html` return html`
<div class="section"> <dees-tile>
<div class="section-header"> <div slot="header" class="section-header">
<div class="section-title">Appearance</div> <div class="section-heading">
<div class="section-subtitle">Customize the look and feel</div> <span class="section-title">Appearance</span>
<span class="section-subtitle">Customize the look and feel</span>
</div> </div>
</div>
<div class="section-content">
<div class="form-row"> <div class="form-row">
<div class="form-label-group"> <div class="form-label-group">
<span class="form-label">Dark Mode</span> <span class="form-label">Dark Mode</span>
@@ -279,12 +315,16 @@ export class SzSettingsView extends DeesElement {
<div class="toggle-switch ${this.settings.darkMode ? 'active' : ''}" @click=${() => this.toggleDarkMode()}></div> <div class="toggle-switch ${this.settings.darkMode ? 'active' : ''}" @click=${() => this.toggleDarkMode()}></div>
</div> </div>
</div> </div>
</dees-tile>
<div class="section"> <dees-tile>
<div class="section-header"> <div slot="header" class="section-header">
<div class="section-title">Cloudflare Integration</div> <div class="section-heading">
<div class="section-subtitle">Configure Cloudflare API for DNS management</div> <span class="section-title">Cloudflare Integration</span>
<span class="section-subtitle">Configure Cloudflare API for DNS management</span>
</div> </div>
</div>
<div class="section-content">
<div class="input-group"> <div class="input-group">
<div class="form-group"> <div class="form-group">
<div class="field-label">API Token</div> <div class="field-label">API Token</div>
@@ -297,12 +337,16 @@ export class SzSettingsView extends DeesElement {
<div class="form-hint">Get your API token from the Cloudflare dashboard with DNS edit permissions.</div> <div class="form-hint">Get your API token from the Cloudflare dashboard with DNS edit permissions.</div>
</div> </div>
</div> </div>
</dees-tile>
<div class="section"> <dees-tile>
<div class="section-header"> <div slot="header" class="section-header">
<div class="section-title">SSL/TLS Settings</div> <div class="section-heading">
<div class="section-subtitle">Configure certificate management</div> <span class="section-title">SSL/TLS Settings</span>
<span class="section-subtitle">Configure certificate management</span>
</div> </div>
</div>
<div class="section-content">
<div class="form-row"> <div class="form-row">
<div class="form-label-group"> <div class="form-label-group">
<span class="form-label">Auto-Renew Certificates</span> <span class="form-label">Auto-Renew Certificates</span>
@@ -321,12 +365,16 @@ export class SzSettingsView extends DeesElement {
<div class="form-hint">Email address for Let's Encrypt notifications.</div> <div class="form-hint">Email address for Let's Encrypt notifications.</div>
</div> </div>
</div> </div>
</dees-tile>
<div class="section"> <dees-tile>
<div class="section-header"> <div slot="header" class="section-header">
<div class="section-title">Network Settings</div> <div class="section-heading">
<div class="section-subtitle">Configure network and proxy settings</div> <span class="section-title">Network Settings</span>
<span class="section-subtitle">Configure network and proxy settings</span>
</div> </div>
</div>
<div class="section-content">
<div class="input-row"> <div class="input-row">
<div class="form-group"> <div class="form-group">
<div class="field-label">HTTP Port</div> <div class="field-label">HTTP Port</div>
@@ -345,12 +393,16 @@ export class SzSettingsView extends DeesElement {
<div class="toggle-switch ${this.settings.forceHttps ? 'active' : ''}" @click=${() => this.toggleSetting('forceHttps')}></div> <div class="toggle-switch ${this.settings.forceHttps ? 'active' : ''}" @click=${() => this.toggleSetting('forceHttps')}></div>
</div> </div>
</div> </div>
</dees-tile>
<div class="section"> <dees-tile>
<div class="section-header"> <div slot="header" class="section-header">
<div class="section-title">Account</div> <div class="section-heading">
<div class="section-subtitle">Manage your account settings</div> <span class="section-title">Account</span>
<span class="section-subtitle">Manage your account settings</span>
</div> </div>
</div>
<div class="section-content">
<div class="form-group"> <div class="form-group">
<div class="field-label">Current User</div> <div class="field-label">Current User</div>
<div style="font-size: 14px; color: ${cssManager.bdTheme('#18181b', '#fafafa')};">${this.currentUser || 'Unknown'}</div> <div style="font-size: 14px; color: ${cssManager.bdTheme('#18181b', '#fafafa')};">${this.currentUser || 'Unknown'}</div>
@@ -370,15 +422,23 @@ export class SzSettingsView extends DeesElement {
<div class="field-label">Confirm Password</div> <div class="field-label">Confirm Password</div>
<input type="password" id="confirmPassword"> <input type="password" id="confirmPassword">
</div> </div>
<button class="button secondary" style="width: fit-content;" @click=${() => this.handleChangePassword()}>Update Password</button>
</div> </div>
</div> </div>
</div> </div>
<div slot="footer" class="section-footer">
<button class="tile-button" @click=${() => this.handleChangePassword()}>Update Password</button>
</div>
</dees-tile>
<div class="actions"> <dees-tile>
<button class="button secondary" @click=${() => this.handleReset()}>Reset</button> <div class="section-content" style="padding: 12px 16px; text-align: center; color: var(--dees-color-text-muted); font-size: 12px;">
<button class="button primary" @click=${() => this.handleSave()}>Save Settings</button> Save your changes or reset to defaults.
</div> </div>
<div slot="footer" class="section-footer">
<button class="tile-button" @click=${() => this.handleReset()}>Reset</button>
<button class="tile-button primary" @click=${() => this.handleSave()}>Save Settings</button>
</div>
</dees-tile>
`; `;
} }
-189
View File
@@ -1,189 +0,0 @@
import {
DeesElement,
customElement,
html,
css,
cssManager,
property,
type TemplateResult,
} from '@design.estate/dees-element';
declare global {
interface HTMLElementTagNameMap {
'sz-stat-card': SzStatCard;
}
}
@customElement('sz-stat-card')
export class SzStatCard extends DeesElement {
public static demo = () => html`
<style>
.demo-grid {
display: grid;
grid-template-columns: repeat(2, 1fr);
gap: 16px;
padding: 24px;
max-width: 800px;
}
</style>
<div class="demo-grid">
<sz-stat-card
label="Total Services"
value="7"
icon="server"
></sz-stat-card>
<sz-stat-card
label="Running"
value="7"
icon="check"
variant="success"
></sz-stat-card>
<sz-stat-card
label="Stopped"
value="0"
icon="stop"
></sz-stat-card>
<sz-stat-card
label="Docker"
value="Running"
icon="container"
variant="success"
valueBadge
></sz-stat-card>
</div>
`;
public static demoGroups = ['Dashboard'];
@property({ type: String })
public accessor label: string = '';
@property({ type: String })
public accessor value: string = '';
@property({ type: String })
public accessor icon: string = '';
@property({ type: String })
public accessor variant: 'default' | 'success' | 'warning' | 'error' = 'default';
@property({ type: Boolean })
public accessor valueBadge: boolean = false;
public static styles = [
cssManager.defaultStyles,
css`
:host {
display: block;
height: 100%;
}
.card {
background: ${cssManager.bdTheme('#ffffff', '#09090b')};
border: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')};
border-radius: 8px;
padding: 20px;
transition: all 200ms ease;
height: 100%;
box-sizing: border-box;
}
.card:hover {
border-color: ${cssManager.bdTheme('#d4d4d8', '#3f3f46')};
box-shadow: 0 4px 12px ${cssManager.bdTheme('rgba(0,0,0,0.05)', 'rgba(0,0,0,0.2)')};
}
.header {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 8px;
}
.label {
font-size: 14px;
font-weight: 500;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')};
}
.icon {
width: 20px;
height: 20px;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')};
}
.value {
font-size: 28px;
font-weight: 700;
color: ${cssManager.bdTheme('#18181b', '#fafafa')};
line-height: 1.2;
}
.value.success {
color: ${cssManager.bdTheme('#16a34a', '#22c55e')};
}
.value.warning {
color: ${cssManager.bdTheme('#ca8a04', '#facc15')};
}
.value.error {
color: ${cssManager.bdTheme('#dc2626', '#ef4444')};
}
.badge {
display: inline-flex;
align-items: center;
padding: 4px 12px;
border-radius: 9999px;
font-size: 14px;
font-weight: 500;
}
.badge.success {
background: ${cssManager.bdTheme('#dcfce7', 'rgba(34, 197, 94, 0.2)')};
color: ${cssManager.bdTheme('#16a34a', '#22c55e')};
}
.badge.warning {
background: ${cssManager.bdTheme('#fef9c3', 'rgba(250, 204, 21, 0.2)')};
color: ${cssManager.bdTheme('#ca8a04', '#facc15')};
}
.badge.error {
background: ${cssManager.bdTheme('#fee2e2', 'rgba(239, 68, 68, 0.2)')};
color: ${cssManager.bdTheme('#dc2626', '#ef4444')};
}
.badge.default {
background: ${cssManager.bdTheme('#f4f4f5', '#27272a')};
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')};
}
`,
];
public render(): TemplateResult {
const valueClass = this.valueBadge ? `badge ${this.variant}` : `value ${this.variant}`;
return html`
<div class="card">
<div class="header">
<span class="label">${this.label}</span>
${this.renderIcon()}
</div>
<div class="${valueClass}">${this.value}</div>
</div>
`;
}
private renderIcon(): TemplateResult {
const icons: Record<string, TemplateResult> = {
server: html`<svg class="icon" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><rect x="2" y="2" width="20" height="8" rx="2" ry="2"></rect><rect x="2" y="14" width="20" height="8" rx="2" ry="2"></rect><line x1="6" y1="6" x2="6.01" y2="6"></line><line x1="6" y1="18" x2="6.01" y2="18"></line></svg>`,
check: html`<svg class="icon" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><polyline points="20 6 9 17 4 12"></polyline></svg>`,
stop: html`<svg class="icon" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><circle cx="12" cy="12" r="10"></circle><line x1="10" y1="15" x2="10" y2="9"></line><line x1="14" y1="15" x2="14" y2="9"></line></svg>`,
container: html`<svg class="icon" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2"><path d="M21 16V8a2 2 0 0 0-1-1.73l-7-4a2 2 0 0 0-2 0l-7 4A2 2 0 0 0 3 8v8a2 2 0 0 0 1 1.73l7 4a2 2 0 0 0 2 0l7-4A2 2 0 0 0 21 16z"></path></svg>`,
};
return icons[this.icon] || html``;
}
}
+41 -45
View File
@@ -7,8 +7,7 @@ import {
property, property,
type TemplateResult, type TemplateResult,
} from '@design.estate/dees-element'; } from '@design.estate/dees-element';
import type { IStatsTile } from '@design.estate/dees-catalog';
import './sz-stat-card.js';
declare global { declare global {
interface HTMLElementTagNameMap { interface HTMLElementTagNameMap {
@@ -54,54 +53,51 @@ export class SzStatusGridCluster extends DeesElement {
:host { :host {
display: block; display: block;
} }
.grid {
display: grid;
grid-template-columns: repeat(2, 1fr);
gap: 16px;
align-items: stretch;
}
.grid > * {
height: 100%;
}
@media (min-width: 768px) {
.grid {
grid-template-columns: repeat(4, 1fr);
}
}
`, `,
]; ];
private get tiles(): IStatsTile[] {
return [
{
id: 'total',
title: 'Total Services',
value: this.stats.totalServices,
type: 'number',
icon: 'lucide:server',
},
{
id: 'running',
title: 'Running',
value: this.stats.running,
type: 'number',
icon: 'lucide:check',
color: '#22c55e',
},
{
id: 'stopped',
title: 'Stopped',
value: this.stats.stopped,
type: 'number',
icon: 'lucide:circleStop',
color: this.stats.stopped > 0 ? '#f59e0b' : undefined,
},
{
id: 'docker',
title: 'Docker',
value: this.stats.dockerStatus === 'running' ? 'Running' : 'Stopped',
type: 'text',
icon: 'lucide:container',
color: this.stats.dockerStatus === 'running' ? '#22c55e' : '#ef4444',
},
];
}
public render(): TemplateResult { public render(): TemplateResult {
return html` return html`
<div class="grid"> <dees-statsgrid
<sz-stat-card .tiles=${this.tiles}
label="Total Services" .minTileWidth=${200}
value="${this.stats.totalServices}" ></dees-statsgrid>
icon="server"
></sz-stat-card>
<sz-stat-card
label="Running"
value="${this.stats.running}"
icon="check"
variant="success"
></sz-stat-card>
<sz-stat-card
label="Stopped"
value="${this.stats.stopped}"
icon="stop"
variant="${this.stats.stopped > 0 ? 'warning' : 'default'}"
></sz-stat-card>
<sz-stat-card
label="Docker"
value="${this.stats.dockerStatus === 'running' ? 'Running' : 'Stopped'}"
icon="container"
variant="${this.stats.dockerStatus === 'running' ? 'success' : 'error'}"
valueBadge
></sz-stat-card>
</div>
`; `;
} }
} }
+94 -74
View File
@@ -55,56 +55,94 @@ export class SzTokensView extends DeesElement {
display: block; display: block;
} }
.section { dees-tile {
background: ${cssManager.bdTheme('#ffffff', '#09090b')}; display: block;
border: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')};
border-radius: 8px;
margin-bottom: 24px; margin-bottom: 24px;
overflow: hidden;
} }
.section-header { .section-header {
height: 36px;
display: flex; display: flex;
justify-content: space-between; align-items: center;
align-items: flex-start; padding: 0 16px;
padding: 16px; width: 100%;
border-bottom: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')}; box-sizing: border-box;
} }
.section-info { .section-heading {
flex: 1;
display: flex; display: flex;
flex-direction: column; align-items: baseline;
gap: 4px; gap: 8px;
min-width: 0;
} }
.section-title { .section-title {
font-size: 16px; font-weight: 500;
font-weight: 600; font-size: 13px;
color: ${cssManager.bdTheme('#18181b', '#fafafa')}; letter-spacing: -0.01em;
color: var(--dees-color-text-secondary);
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
} }
.section-subtitle { .section-subtitle {
font-size: 13px; font-size: 12px;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')}; color: var(--dees-color-text-muted);
letter-spacing: -0.01em;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
} }
.create-button { .section-footer {
display: inline-flex; display: flex;
flex-direction: row;
justify-content: flex-end;
align-items: center;
gap: 0;
height: 36px;
width: 100%;
box-sizing: border-box;
}
.tile-button {
padding: 0 16px;
height: 100%;
text-align: center;
font-size: 12px;
font-weight: 500;
cursor: pointer;
user-select: none;
transition: all 0.15s ease;
background: transparent;
border: none;
border-left: 1px solid var(--dees-color-border-subtle);
color: var(--dees-color-text-muted);
white-space: nowrap;
display: flex;
align-items: center; align-items: center;
gap: 6px; gap: 6px;
padding: 8px 14px;
background: ${cssManager.bdTheme('#18181b', '#fafafa')};
border: none;
border-radius: 6px;
font-size: 13px;
font-weight: 500;
color: ${cssManager.bdTheme('#fafafa', '#18181b')};
cursor: pointer;
transition: all 200ms ease;
} }
.create-button:hover { .tile-button:first-child {
opacity: 0.9; border-left: none;
}
.tile-button:hover {
background: var(--dees-color-hover);
color: var(--dees-color-text-primary);
}
.tile-button.primary {
color: ${cssManager.bdTheme('hsl(217.2 91.2% 59.8%)', 'hsl(213.1 93.9% 67.8%)')};
font-weight: 600;
}
.tile-button.primary:hover {
background: ${cssManager.bdTheme('hsl(217.2 91.2% 59.8% / 0.08)', 'hsl(213.1 93.9% 67.8% / 0.08)')};
color: ${cssManager.bdTheme('hsl(217.2 91.2% 50%)', 'hsl(213.1 93.9% 75%)')};
} }
.token-list { .token-list {
@@ -192,45 +230,18 @@ export class SzTokensView extends DeesElement {
.empty-text { .empty-text {
font-size: 14px; font-size: 14px;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')}; color: ${cssManager.bdTheme('#71717a', '#a1a1aa')};
margin-bottom: 16px;
}
.empty-button {
display: inline-flex;
align-items: center;
gap: 6px;
padding: 8px 14px;
background: ${cssManager.bdTheme('#ffffff', '#09090b')};
border: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')};
border-radius: 6px;
font-size: 13px;
font-weight: 500;
color: ${cssManager.bdTheme('#18181b', '#fafafa')};
cursor: pointer;
transition: all 200ms ease;
}
.empty-button:hover {
background: ${cssManager.bdTheme('#f4f4f5', '#18181b')};
} }
`, `,
]; ];
public render(): TemplateResult { public render(): TemplateResult {
return html` return html`
<div class="section"> <dees-tile>
<div class="section-header"> <div slot="header" class="section-header">
<div class="section-info"> <div class="section-heading">
<div class="section-title">Global Tokens</div> <span class="section-title">Global Tokens</span>
<div class="section-subtitle">Tokens that can push images to multiple services</div> <span class="section-subtitle">Tokens that can push images to multiple services</span>
</div> </div>
<button class="create-button" @click=${() => this.handleCreate('global')}>
<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<line x1="12" y1="5" x2="12" y2="19"></line>
<line x1="5" y1="12" x2="19" y2="12"></line>
</svg>
Create Token
</button>
</div> </div>
${this.globalTokens.length > 0 ? html` ${this.globalTokens.length > 0 ? html`
<div class="token-list"> <div class="token-list">
@@ -239,25 +250,26 @@ export class SzTokensView extends DeesElement {
` : html` ` : html`
<div class="empty-state"> <div class="empty-state">
<div class="empty-text">No global tokens created</div> <div class="empty-text">No global tokens created</div>
<button class="empty-button" @click=${() => this.handleCreate('global')}>Create Global Token</button>
</div> </div>
`} `}
</div> <div slot="footer" class="section-footer">
<button class="tile-button primary" @click=${() => this.handleCreate('global')}>
<div class="section"> <svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<div class="section-header">
<div class="section-info">
<div class="section-title">CI Tokens (Service-specific)</div>
<div class="section-subtitle">Tokens tied to individual services for CI/CD pipelines</div>
</div>
<button class="create-button" @click=${() => this.handleCreate('ci')}>
<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<line x1="12" y1="5" x2="12" y2="19"></line> <line x1="12" y1="5" x2="12" y2="19"></line>
<line x1="5" y1="12" x2="19" y2="12"></line> <line x1="5" y1="12" x2="19" y2="12"></line>
</svg> </svg>
Create Token Create Token
</button> </button>
</div> </div>
</dees-tile>
<dees-tile>
<div slot="header" class="section-header">
<div class="section-heading">
<span class="section-title">CI Tokens (Service-specific)</span>
<span class="section-subtitle">Tokens tied to individual services for CI/CD pipelines</span>
</div>
</div>
${this.ciTokens.length > 0 ? html` ${this.ciTokens.length > 0 ? html`
<div class="token-list"> <div class="token-list">
${this.ciTokens.map(token => this.renderToken(token))} ${this.ciTokens.map(token => this.renderToken(token))}
@@ -265,10 +277,18 @@ export class SzTokensView extends DeesElement {
` : html` ` : html`
<div class="empty-state"> <div class="empty-state">
<div class="empty-text">No CI tokens created</div> <div class="empty-text">No CI tokens created</div>
<button class="empty-button" @click=${() => this.handleCreate('ci')}>Create CI Token</button>
</div> </div>
`} `}
<div slot="footer" class="section-footer">
<button class="tile-button primary" @click=${() => this.handleCreate('ci')}>
<svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<line x1="12" y1="5" x2="12" y2="19"></line>
<line x1="5" y1="12" x2="19" y2="12"></line>
</svg>
Create Token
</button>
</div> </div>
</dees-tile>
`; `;
} }
+7 -1
View File
@@ -138,6 +138,12 @@ export class SzDemoAppShell extends DeesElement {
iconName: 'lucide:Mail', iconName: 'lucide:Mail',
content: 'sz-demo-view-mta', content: 'sz-demo-view-mta',
}, },
{
id: 'routes',
name: 'Routes',
iconName: 'lucide:Route',
content: 'sz-demo-view-routes',
},
{ {
id: 'settings', id: 'settings',
name: 'Settings', name: 'Settings',
@@ -153,7 +159,7 @@ export class SzDemoAppShell extends DeesElement {
}, },
{ {
name: 'Infrastructure', name: 'Infrastructure',
views: ['services', 'network', 'registries', 'mta'], views: ['services', 'network', 'registries', 'mta', 'routes'],
}, },
{ {
name: 'Administration', name: 'Administration',
+1 -1
View File
@@ -5,7 +5,7 @@
"moduleResolution": "NodeNext", "moduleResolution": "NodeNext",
"esModuleInterop": true, "esModuleInterop": true,
"verbatimModuleSyntax": true, "verbatimModuleSyntax": true,
"skipLibCheck": true "types": ["node"]
}, },
"exclude": [ "exclude": [
"dist_*/**/*.d.ts" "dist_*/**/*.d.ts"