Files
catalog/ts_web/elements/sz-platform-service-detail-view.ts

717 lines
24 KiB
TypeScript

import {
DeesElement,
customElement,
html,
css,
cssManager,
property,
type TemplateResult,
} from '@design.estate/dees-element';
declare global {
interface HTMLElementTagNameMap {
'sz-platform-service-detail-view': SzPlatformServiceDetailView;
}
}
export interface IPlatformServiceDetail {
id: string;
name: string;
type: 'mongodb' | 'minio' | 'clickhouse' | 'redis';
status: 'running' | 'stopped' | 'error';
version: string;
host: string;
port: number;
credentials?: {
username?: string;
password?: string;
accessKey?: string;
secretKey?: string;
};
config: Record<string, any>;
metrics?: {
cpu: number;
memory: number;
storage: number;
connections?: number;
};
}
export interface IPlatformLogEntry {
timestamp: string;
level: 'info' | 'warn' | 'error' | 'debug';
message: string;
}
@customElement('sz-platform-service-detail-view')
export class SzPlatformServiceDetailView extends DeesElement {
public static demo = () => html`
<div style="padding: 24px; max-width: 1000px;">
<sz-platform-service-detail-view
.service=${{
id: '1',
name: 'MongoDB',
type: 'mongodb',
status: 'running',
version: '7.0.4',
host: 'localhost',
port: 27017,
credentials: { username: 'admin', password: '••••••••' },
config: { replicaSet: 'rs0', authEnabled: true },
metrics: { cpu: 12, memory: 45, storage: 23, connections: 8 },
}}
.logs=${[
{ timestamp: '2024-01-20 14:30:22', level: 'info', message: 'Connection accepted from 127.0.0.1:54321' },
{ timestamp: '2024-01-20 14:30:20', level: 'info', message: 'Index build completed on collection users' },
{ timestamp: '2024-01-20 14:30:15', level: 'warn', message: 'Slow query detected: 1.2s on collection orders' },
{ timestamp: '2024-01-20 14:30:10', level: 'info', message: 'Checkpoint complete' },
]}
></sz-platform-service-detail-view>
</div>
`;
public static demoGroups = ['Platform'];
@property({ type: Object })
public accessor service: IPlatformServiceDetail | null = null;
@property({ type: Array })
public accessor logs: IPlatformLogEntry[] = [];
@property({ type: Boolean })
public accessor actionLoading: boolean = false;
public static styles = [
cssManager.defaultStyles,
css`
:host {
display: block;
}
.header {
display: flex;
justify-content: space-between;
align-items: flex-start;
margin-bottom: 24px;
}
.header-info {
display: flex;
align-items: center;
gap: 16px;
}
.service-icon {
width: 56px;
height: 56px;
background: ${cssManager.bdTheme('#f4f4f5', '#27272a')};
border-radius: 12px;
display: flex;
align-items: center;
justify-content: center;
}
.service-icon svg {
width: 28px;
height: 28px;
color: ${cssManager.bdTheme('#18181b', '#fafafa')};
}
.service-details {
display: flex;
flex-direction: column;
gap: 4px;
}
.service-name {
font-size: 22px;
font-weight: 600;
color: ${cssManager.bdTheme('#18181b', '#fafafa')};
}
.service-meta {
display: flex;
align-items: center;
gap: 12px;
font-size: 14px;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')};
}
.status-badge {
display: inline-flex;
align-items: center;
gap: 6px;
padding: 4px 10px;
border-radius: 9999px;
font-size: 13px;
font-weight: 500;
}
.status-badge.running {
background: ${cssManager.bdTheme('#dcfce7', 'rgba(34, 197, 94, 0.2)')};
color: ${cssManager.bdTheme('#16a34a', '#22c55e')};
}
.status-badge.stopped {
background: ${cssManager.bdTheme('#f4f4f5', '#27272a')};
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')};
}
.status-badge.error {
background: ${cssManager.bdTheme('#fee2e2', 'rgba(239, 68, 68, 0.2)')};
color: ${cssManager.bdTheme('#dc2626', '#ef4444')};
}
.status-dot {
width: 8px;
height: 8px;
border-radius: 50%;
background: currentColor;
}
.header-actions {
display: flex;
gap: 8px;
}
.action-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;
}
.action-button:hover:not(:disabled) {
background: ${cssManager.bdTheme('#f4f4f5', '#18181b')};
}
.action-button:disabled {
opacity: 0.6;
cursor: not-allowed;
}
.action-button svg {
width: 14px;
height: 14px;
}
.action-button.danger {
color: ${cssManager.bdTheme('#dc2626', '#ef4444')};
border-color: ${cssManager.bdTheme('#fee2e2', 'rgba(239, 68, 68, 0.3)')};
}
.action-button.danger:hover:not(:disabled) {
background: ${cssManager.bdTheme('#fee2e2', 'rgba(239, 68, 68, 0.2)')};
}
.grid {
display: grid;
grid-template-columns: 1fr 1fr;
gap: 16px;
margin-bottom: 16px;
}
@media (max-width: 768px) {
.grid {
grid-template-columns: 1fr;
}
}
.section {
background: ${cssManager.bdTheme('#ffffff', '#09090b')};
border: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')};
border-radius: 8px;
overflow: hidden;
}
.section.full-width {
grid-column: 1 / -1;
}
.section-header {
display: flex;
justify-content: space-between;
align-items: center;
padding: 14px 16px;
border-bottom: 1px solid ${cssManager.bdTheme('#e4e4e7', '#27272a')};
background: ${cssManager.bdTheme('#f4f4f5', '#18181b')};
}
.section-title {
font-size: 14px;
font-weight: 600;
color: ${cssManager.bdTheme('#18181b', '#fafafa')};
display: flex;
align-items: center;
gap: 8px;
}
.section-title svg {
width: 16px;
height: 16px;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')};
}
.section-content {
padding: 16px;
}
.info-row {
display: flex;
justify-content: space-between;
align-items: center;
padding: 10px 0;
border-bottom: 1px solid ${cssManager.bdTheme('#f4f4f5', '#27272a')};
}
.info-row:last-child {
border-bottom: none;
}
.info-label {
font-size: 13px;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')};
}
.info-value {
font-size: 13px;
font-weight: 500;
color: ${cssManager.bdTheme('#18181b', '#fafafa')};
font-family: monospace;
display: flex;
align-items: center;
gap: 8px;
}
.copy-button {
padding: 4px;
background: transparent;
border: none;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')};
cursor: pointer;
border-radius: 4px;
transition: all 200ms ease;
}
.copy-button:hover {
background: ${cssManager.bdTheme('#f4f4f5', '#27272a')};
color: ${cssManager.bdTheme('#18181b', '#fafafa')};
}
.metrics-grid {
display: grid;
grid-template-columns: repeat(4, 1fr);
gap: 12px;
}
@media (max-width: 600px) {
.metrics-grid {
grid-template-columns: repeat(2, 1fr);
}
}
.metric-card {
text-align: center;
padding: 12px;
background: ${cssManager.bdTheme('#f4f4f5', '#18181b')};
border-radius: 6px;
}
.metric-value {
font-size: 20px;
font-weight: 600;
color: ${cssManager.bdTheme('#18181b', '#fafafa')};
}
.metric-label {
font-size: 12px;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')};
margin-top: 2px;
}
.progress-bar {
height: 4px;
background: ${cssManager.bdTheme('#e4e4e7', '#27272a')};
border-radius: 2px;
margin-top: 8px;
overflow: hidden;
}
.progress-fill {
height: 100%;
border-radius: 2px;
transition: width 300ms ease;
}
.progress-fill.low {
background: ${cssManager.bdTheme('#22c55e', '#22c55e')};
}
.progress-fill.medium {
background: ${cssManager.bdTheme('#eab308', '#eab308')};
}
.progress-fill.high {
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 {
display: flex;
justify-content: space-between;
padding: 8px 0;
border-bottom: 1px solid ${cssManager.bdTheme('#f4f4f5', '#27272a')};
}
.config-item:last-child {
border-bottom: none;
}
.config-key {
font-size: 13px;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')};
}
.config-value {
font-size: 13px;
font-weight: 500;
color: ${cssManager.bdTheme('#18181b', '#fafafa')};
}
.config-value.true {
color: ${cssManager.bdTheme('#16a34a', '#22c55e')};
}
.config-value.false {
color: ${cssManager.bdTheme('#dc2626', '#ef4444')};
}
.empty-state {
text-align: center;
padding: 40px 20px;
color: ${cssManager.bdTheme('#71717a', '#a1a1aa')};
}
`,
];
public render(): TemplateResult {
if (!this.service) {
return html`<div class="empty-state">No service selected</div>`;
}
return html`
<div class="header">
<div class="header-info">
<div class="service-icon">
${this.renderServiceIcon()}
</div>
<div class="service-details">
<div class="service-name">${this.service.name}</div>
<div class="service-meta">
<span class="status-badge ${this.service.status}">
<span class="status-dot"></span>
${this.service.status.charAt(0).toUpperCase() + this.service.status.slice(1)}
</span>
<span>Version ${this.service.version}</span>
</div>
</div>
</div>
<div class="header-actions">
${this.service.status === 'running' ? html`
<button class="action-button" ?disabled=${this.actionLoading} @click=${() => this.handleRestart()}>
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<polyline points="23 4 23 10 17 10"></polyline>
<polyline points="1 20 1 14 7 14"></polyline>
<path d="M3.51 9a9 9 0 0 1 14.85-3.36L23 10M1 14l4.64 4.36A9 9 0 0 0 20.49 15"></path>
</svg>
Restart
</button>
<button class="action-button danger" ?disabled=${this.actionLoading} @click=${() => this.handleStop()}>
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<rect x="6" y="6" width="12" height="12" rx="1"></rect>
</svg>
Stop
</button>
` : html`
<button class="action-button" ?disabled=${this.actionLoading} @click=${() => this.handleStart()}>
<svg viewBox="0 0 24 24" fill="currentColor">
<polygon points="5,3 19,12 5,21"></polygon>
</svg>
Start
</button>
`}
</div>
</div>
<div class="grid">
<!-- Connection Info -->
<div class="section">
<div class="section-header">
<div class="section-title">
<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="M14 11a5 5 0 0 0-7.54-.54l-3 3a5 5 0 0 0 7.07 7.07l1.71-1.71"></path>
</svg>
Connection
</div>
</div>
<div class="section-content">
<div class="info-row">
<span class="info-label">Host</span>
<span class="info-value">
${this.service.host}
<button class="copy-button" @click=${() => this.copyToClipboard(this.service!.host)}>
<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<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>
</svg>
</button>
</span>
</div>
<div class="info-row">
<span class="info-label">Port</span>
<span class="info-value">${this.service.port}</span>
</div>
${this.service.credentials?.username ? html`
<div class="info-row">
<span class="info-label">Username</span>
<span class="info-value">
${this.service.credentials.username}
<button class="copy-button" @click=${() => this.copyToClipboard(this.service!.credentials!.username!)}>
<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<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>
</svg>
</button>
</span>
</div>
<div class="info-row">
<span class="info-label">Password</span>
<span class="info-value">••••••••</span>
</div>
` : ''}
${this.service.credentials?.accessKey ? html`
<div class="info-row">
<span class="info-label">Access Key</span>
<span class="info-value">
${this.service.credentials.accessKey}
<button class="copy-button" @click=${() => this.copyToClipboard(this.service!.credentials!.accessKey!)}>
<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<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>
</svg>
</button>
</span>
</div>
<div class="info-row">
<span class="info-label">Secret Key</span>
<span class="info-value">••••••••</span>
</div>
` : ''}
</div>
</div>
<!-- Configuration -->
<div class="section">
<div class="section-header">
<div class="section-title">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<circle cx="12" cy="12" r="3"></circle>
<path d="M19.4 15a1.65 1.65 0 0 0 .33 1.82l.06.06a2 2 0 0 1 0 2.83 2 2 0 0 1-2.83 0l-.06-.06a1.65 1.65 0 0 0-1.82-.33 1.65 1.65 0 0 0-1 1.51V21a2 2 0 0 1-2 2 2 2 0 0 1-2-2v-.09A1.65 1.65 0 0 0 9 19.4a1.65 1.65 0 0 0-1.82.33l-.06.06a2 2 0 0 1-2.83 0 2 2 0 0 1 0-2.83l.06-.06a1.65 1.65 0 0 0 .33-1.82 1.65 1.65 0 0 0-1.51-1H3a2 2 0 0 1-2-2 2 2 0 0 1 2-2h.09A1.65 1.65 0 0 0 4.6 9a1.65 1.65 0 0 0-.33-1.82l-.06-.06a2 2 0 0 1 0-2.83 2 2 0 0 1 2.83 0l.06.06a1.65 1.65 0 0 0 1.82.33H9a1.65 1.65 0 0 0 1-1.51V3a2 2 0 0 1 2-2 2 2 0 0 1 2 2v.09a1.65 1.65 0 0 0 1 1.51 1.65 1.65 0 0 0 1.82-.33l.06-.06a2 2 0 0 1 2.83 0 2 2 0 0 1 0 2.83l-.06.06a1.65 1.65 0 0 0-.33 1.82V9a1.65 1.65 0 0 0 1.51 1H21a2 2 0 0 1 2 2 2 2 0 0 1-2 2h-.09a1.65 1.65 0 0 0-1.51 1z"></path>
</svg>
Configuration
</div>
</div>
<div class="section-content">
${Object.entries(this.service.config).map(([key, value]) => html`
<div class="config-item">
<span class="config-key">${this.formatConfigKey(key)}</span>
<span class="config-value ${typeof value === 'boolean' ? (value ? 'true' : 'false') : ''}">${this.formatConfigValue(value)}</span>
</div>
`)}
</div>
</div>
<!-- Metrics -->
${this.service.metrics ? html`
<div class="section full-width">
<div class="section-header">
<div class="section-title">
<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="12" y1="20" x2="12" y2="4"></line>
<line x1="6" y1="20" x2="6" y2="14"></line>
</svg>
Resource Usage
</div>
</div>
<div class="section-content">
<div class="metrics-grid">
<div class="metric-card">
<div class="metric-value">${this.service.metrics.cpu}%</div>
<div class="metric-label">CPU</div>
<div class="progress-bar">
<div class="progress-fill ${this.getProgressClass(this.service.metrics.cpu)}" style="width: ${this.service.metrics.cpu}%"></div>
</div>
</div>
<div class="metric-card">
<div class="metric-value">${this.service.metrics.memory}%</div>
<div class="metric-label">Memory</div>
<div class="progress-bar">
<div class="progress-fill ${this.getProgressClass(this.service.metrics.memory)}" style="width: ${this.service.metrics.memory}%"></div>
</div>
</div>
<div class="metric-card">
<div class="metric-value">${this.service.metrics.storage}%</div>
<div class="metric-label">Storage</div>
<div class="progress-bar">
<div class="progress-fill ${this.getProgressClass(this.service.metrics.storage)}" style="width: ${this.service.metrics.storage}%"></div>
</div>
</div>
${this.service.metrics.connections !== undefined ? html`
<div class="metric-card">
<div class="metric-value">${this.service.metrics.connections}</div>
<div class="metric-label">Connections</div>
</div>
` : ''}
</div>
</div>
</div>
` : ''}
<!-- Logs -->
<div class="section full-width">
<div class="section-header">
<div class="section-title">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<polyline points="4 17 10 11 4 5"></polyline>
<line x1="12" y1="19" x2="20" y2="19"></line>
</svg>
Logs
</div>
</div>
<div class="section-content">
<div class="log-container">
${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>
`;
}
private renderServiceIcon(): TemplateResult {
const type = this.service?.type;
switch (type) {
case 'mongodb':
return html`<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>`;
case 'minio':
return html`<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>`;
case 'clickhouse':
return html`<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>`;
case 'redis':
return html`<svg viewBox="0 0 24 24" fill="currentColor"><path d="M12 2L2 7l10 5 10-5-10-5zM2 17l10 5 10-5M2 12l10 5 10-5"/></svg>`;
default:
return 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>`;
}
}
private getProgressClass(value: number): string {
if (value < 50) return 'low';
if (value < 80) return 'medium';
return 'high';
}
private formatConfigKey(key: string): string {
return key.replace(/([A-Z])/g, ' $1').replace(/^./, str => str.toUpperCase());
}
private formatConfigValue(value: any): string {
if (typeof value === 'boolean') return value ? 'Enabled' : 'Disabled';
return String(value);
}
private copyToClipboard(text: string) {
navigator.clipboard.writeText(text);
this.dispatchEvent(new CustomEvent('copy', { detail: text, bubbles: true, composed: true }));
}
private handleStart() {
this.dispatchEvent(new CustomEvent('start', { detail: this.service, bubbles: true, composed: true }));
}
private handleStop() {
this.dispatchEvent(new CustomEvent('stop', { detail: this.service, bubbles: true, composed: true }));
}
private handleRestart() {
this.dispatchEvent(new CustomEvent('restart', { detail: this.service, bubbles: true, composed: true }));
}
}