import * as plugins from '../../../plugins.js'; import * as appstate from '../../../appstate.js'; import { viewHostCss } from '../../shared/index.js'; import { DeesElement, customElement, html, state, css, cssManager, type TemplateResult, } from '@design.estate/dees-element'; @customElement('gitops-view-managedsecrets') export class GitopsViewManagedSecrets extends DeesElement { @state() accessor managedSecretsState: appstate.IManagedSecretsState = { managedSecrets: [] }; @state() accessor connectionsState: appstate.IConnectionsState = { connections: [], activeConnectionId: null, }; @state() accessor dataState: appstate.IDataState = { projects: [], groups: [], secrets: [], pipelines: [], pipelineJobs: [], currentJobLog: '', }; private _autoRefreshHandler: () => void; constructor() { super(); const msSub = appstate.managedSecretsStatePart .select((s) => s) .subscribe((s) => { this.managedSecretsState = s; }); this.rxSubscriptions.push(msSub); const connSub = appstate.connectionsStatePart .select((s) => s) .subscribe((s) => { this.connectionsState = s; }); this.rxSubscriptions.push(connSub); const dataSub = appstate.dataStatePart .select((s) => s) .subscribe((s) => { this.dataState = s; }); this.rxSubscriptions.push(dataSub); this._autoRefreshHandler = () => this.refresh(); document.addEventListener('gitops-auto-refresh', this._autoRefreshHandler); } public override disconnectedCallback() { super.disconnectedCallback(); document.removeEventListener('gitops-auto-refresh', this._autoRefreshHandler); } public static styles = [ cssManager.defaultStyles, viewHostCss, css` .target-list { margin: 8px 0; } .target-item { display: flex; align-items: center; justify-content: space-between; padding: 6px 10px; margin: 4px 0; background: rgba(255, 255, 255, 0.05); border-radius: 4px; color: #ccc; font-size: 13px; } .target-item .remove-btn { cursor: pointer; color: #e74c3c; font-size: 16px; padding: 0 4px; } .status-ok { color: #2ecc71; } .status-error { color: #e74c3c; } .status-pending { color: #f39c12; } `, ]; public render(): TemplateResult { return html`
Managed Secrets
Centrally managed secrets pushed as GITOPS_{key} to configured targets
this.addManagedSecret()}>Add Managed Secret this.pushAll()}>Push All this.refresh()}>Refresh
({ Key: item.key, 'On Target': 'GITOPS_' + item.key, Description: item.description || '-', Targets: String(item.targets.length), Status: this.summarizeStatus(item.targetStatuses), 'Last Pushed': item.lastPushedAt ? new Date(item.lastPushedAt).toLocaleString() : 'Never', })} .dataActions=${[ { name: 'Edit', iconName: 'lucide:edit', type: ['inRow', 'contextmenu'], actionFunc: async ({ item }: any) => { await this.editManagedSecret(item); }, }, { name: 'Push', iconName: 'lucide:upload', type: ['inRow', 'contextmenu'], actionFunc: async ({ item }: any) => { await this.pushOne(item); }, }, { name: 'View Targets', iconName: 'lucide:list', type: ['inRow', 'contextmenu'], actionFunc: async ({ item }: any) => { await this.viewTargets(item); }, }, { name: 'Delete', iconName: 'lucide:trash2', type: ['inRow', 'contextmenu'], actionFunc: async ({ item }: any) => { await this.deleteManagedSecret(item); }, }, ]} > `; } async firstUpdated() { await appstate.connectionsStatePart.dispatchAction(appstate.fetchConnectionsAction, null); await appstate.managedSecretsStatePart.dispatchAction(appstate.fetchManagedSecretsAction, null); } private summarizeStatus(statuses: any[]): string { if (!statuses || statuses.length === 0) return 'Not pushed'; const ok = statuses.filter((s: any) => s.status === 'success').length; const err = statuses.filter((s: any) => s.status === 'error').length; if (err === 0) return `All OK (${ok})`; return `${ok} OK / ${err} Failed`; } private async refresh() { await appstate.managedSecretsStatePart.dispatchAction(appstate.fetchManagedSecretsAction, null); } private async pushAll() { await appstate.managedSecretsStatePart.dispatchAction(appstate.pushAllManagedSecretsAction, null); } private async pushOne(item: any) { await appstate.managedSecretsStatePart.dispatchAction(appstate.pushManagedSecretAction, { managedSecretId: item.id, }); } private async deleteManagedSecret(item: any) { await plugins.deesCatalog.DeesModal.createAndShow({ heading: 'Delete Managed Secret', content: html`

Are you sure you want to delete managed secret "${item.key}"?
This will also remove GITOPS_${item.key} from all ${item.targets.length} target(s).

`, menuOptions: [ { name: 'Cancel', action: async (modal: any) => { modal.destroy(); } }, { name: 'Delete', action: async (modal: any) => { await appstate.managedSecretsStatePart.dispatchAction(appstate.deleteManagedSecretAction, { managedSecretId: item.id, }); modal.destroy(); }, }, ], }); } private async viewTargets(item: any) { const targetRows = (item.targetStatuses && item.targetStatuses.length > 0) ? item.targetStatuses : item.targets.map((t: any) => ({ ...t, status: 'pending' })); const statusIcon = (status: string) => { if (status === 'success') return html`OK`; if (status === 'error') return html`Error`; return html`Pending`; }; await plugins.deesCatalog.DeesModal.createAndShow({ heading: `Targets for ${item.key}`, content: html`
${targetRows.map((t: any) => html`
${t.scopeName || t.scopeId}
${t.scope} on ${this.getConnectionName(t.connectionId)}
${t.error ? html`
${t.error}
` : ''}
${statusIcon(t.status)}
`)} ${targetRows.length === 0 ? html`

No targets configured.

` : ''}
`, menuOptions: [ { name: 'Close', action: async (modal: any) => { modal.destroy(); } }, ], }); } private getConnectionName(connectionId: string): string { const conn = this.connectionsState.connections.find((c) => c.id === connectionId); return conn ? conn.name : connectionId; } private async addManagedSecret() { // Load entities for all connections const connections = this.connectionsState.connections; let targets: any[] = []; let selectedConnId = connections.length > 0 ? connections[0].id : ''; let selectedScope: 'project' | 'group' = 'project'; // Pre-load entities for first connection if (selectedConnId) { await appstate.dataStatePart.dispatchAction(appstate.fetchProjectsAction, { connectionId: selectedConnId, }); } const buildTargetListHtml = () => { if (targets.length === 0) return html`

No targets added yet.

`; return html`${targets.map((t, i) => html`
${t.scopeName} (${t.scope}) on ${this.getConnectionName(t.connectionId)} { targets.splice(i, 1); this.requestUpdate(); }}>x
`)}`; }; await plugins.deesCatalog.DeesModal.createAndShow({ heading: 'Add Managed Secret', content: html`
Targets
({ option: c.name, key: c.id }))} .selectedOption=${connections.length > 0 ? { option: connections[0].name, key: connections[0].id } : undefined} @selectedOption=${async (e: CustomEvent) => { selectedConnId = e.detail.key; if (selectedScope === 'project') { await appstate.dataStatePart.dispatchAction(appstate.fetchProjectsAction, { connectionId: selectedConnId }); } else { await appstate.dataStatePart.dispatchAction(appstate.fetchGroupsAction, { connectionId: selectedConnId }); } }} > { selectedScope = e.detail.key as 'project' | 'group'; if (selectedScope === 'project') { await appstate.dataStatePart.dispatchAction(appstate.fetchProjectsAction, { connectionId: selectedConnId }); } else { await appstate.dataStatePart.dispatchAction(appstate.fetchGroupsAction, { connectionId: selectedConnId }); } }} > { const data = appstate.dataStatePart.getState(); const items = selectedScope === 'project' ? data.projects : data.groups; return items.map((item: any) => ({ option: item.fullPath || item.name, key: item.id })); })()} > { const data = appstate.dataStatePart.getState(); const items = selectedScope === 'project' ? data.projects : data.groups; // Find entity dropdown value const modal = document.querySelector('dees-modal'); if (!modal) return; const entityDropdowns = modal.shadowRoot?.querySelectorAll('dees-input-dropdown'); let entityKey = ''; let entityName = ''; if (entityDropdowns) { for (const dd of entityDropdowns) { if ((dd as any).key === 'targetEntity') { const sel = (dd as any).selectedOption; if (sel) { entityKey = sel.key; entityName = sel.option; } } } } if (!entityKey) return; // Avoid duplicates const exists = targets.some( (t) => t.connectionId === selectedConnId && t.scope === selectedScope && t.scopeId === entityKey, ); if (exists) return; targets.push({ connectionId: selectedConnId, scope: selectedScope, scopeId: entityKey, scopeName: entityName, }); }}>Add Target
${buildTargetListHtml()}
`, menuOptions: [ { name: 'Cancel', action: async (modal: any) => { modal.destroy(); } }, { name: 'Create', action: async (modal: any) => { const inputs = modal.shadowRoot.querySelectorAll('dees-input-text'); const data: any = {}; for (const input of inputs) { data[input.key] = input.value || ''; } if (!data.key) return; await appstate.managedSecretsStatePart.dispatchAction(appstate.createManagedSecretAction, { key: data.key, value: data.value, description: data.description || undefined, targets, }); modal.destroy(); }, }, ], }); } private async editManagedSecret(item: any) { const connections = this.connectionsState.connections; let targets = [...item.targets]; let selectedConnId = connections.length > 0 ? connections[0].id : ''; let selectedScope: 'project' | 'group' = 'project'; if (selectedConnId) { await appstate.dataStatePart.dispatchAction(appstate.fetchProjectsAction, { connectionId: selectedConnId, }); } await plugins.deesCatalog.DeesModal.createAndShow({ heading: `Edit: ${item.key}`, content: html`
Targets
({ option: c.name, key: c.id }))} .selectedOption=${connections.length > 0 ? { option: connections[0].name, key: connections[0].id } : undefined} @selectedOption=${async (e: CustomEvent) => { selectedConnId = e.detail.key; if (selectedScope === 'project') { await appstate.dataStatePart.dispatchAction(appstate.fetchProjectsAction, { connectionId: selectedConnId }); } else { await appstate.dataStatePart.dispatchAction(appstate.fetchGroupsAction, { connectionId: selectedConnId }); } }} > { selectedScope = e.detail.key as 'project' | 'group'; if (selectedScope === 'project') { await appstate.dataStatePart.dispatchAction(appstate.fetchProjectsAction, { connectionId: selectedConnId }); } else { await appstate.dataStatePart.dispatchAction(appstate.fetchGroupsAction, { connectionId: selectedConnId }); } }} > { const data = appstate.dataStatePart.getState(); const items = selectedScope === 'project' ? data.projects : data.groups; return items.map((item: any) => ({ option: item.fullPath || item.name, key: item.id })); })()} > { const modal = document.querySelector('dees-modal'); if (!modal) return; const entityDropdowns = modal.shadowRoot?.querySelectorAll('dees-input-dropdown'); let entityKey = ''; let entityName = ''; if (entityDropdowns) { for (const dd of entityDropdowns) { if ((dd as any).key === 'targetEntity') { const sel = (dd as any).selectedOption; if (sel) { entityKey = sel.key; entityName = sel.option; } } } } if (!entityKey) return; const exists = targets.some( (t: any) => t.connectionId === selectedConnId && t.scope === selectedScope && t.scopeId === entityKey, ); if (exists) return; targets.push({ connectionId: selectedConnId, scope: selectedScope, scopeId: entityKey, scopeName: entityName, }); }}>Add Target
${targets.length === 0 ? html`

No targets.

` : targets.map((t: any, i: number) => html`
${t.scopeName} (${t.scope}) on ${this.getConnectionName(t.connectionId)} { targets.splice(i, 1); }}>x
`)}
`, menuOptions: [ { name: 'Cancel', action: async (modal: any) => { modal.destroy(); } }, { name: 'Update', action: async (modal: any) => { const inputs = modal.shadowRoot.querySelectorAll('dees-input-text'); const data: any = {}; for (const input of inputs) { data[input.key] = input.value || ''; } const updatePayload: any = { managedSecretId: item.id, targets, description: data.description || undefined, }; if (data.value) { updatePayload.value = data.value; } await appstate.managedSecretsStatePart.dispatchAction(appstate.updateManagedSecretAction, updatePayload); modal.destroy(); }, }, ], }); } }