import { EVENT_REFRESH } from "@goauthentik/common/constants"; import { MessageLevel } from "@goauthentik/common/messages"; import { PFSize } from "@goauthentik/elements/Spinner"; import { ModalButton } from "@goauthentik/elements/buttons/ModalButton"; import "@goauthentik/elements/buttons/SpinnerButton"; import { showMessage } from "@goauthentik/elements/messages/MessageContainer"; import { PaginatedResponse } from "@goauthentik/elements/table/Table"; import { Table, TableColumn } from "@goauthentik/elements/table/Table"; import { t } from "@lingui/macro"; import { CSSResult, TemplateResult, html } from "lit"; import { customElement, property, state } from "lit/decorators.js"; import { until } from "lit/directives/until.js"; import PFList from "@patternfly/patternfly/components/List/list.css"; import { UsedBy, UsedByActionEnum } from "@goauthentik/api"; type BulkDeleteMetadata = { key: string; value: string }[]; @customElement("ak-delete-objects-table") export class DeleteObjectsTable extends Table { expandable = true; paginated = false; @property({ attribute: false }) objects: T[] = []; @property({ attribute: false }) metadata!: (item: T) => BulkDeleteMetadata; @property({ attribute: false }) usedBy?: (item: T) => Promise; @state() usedByData: Map = new Map(); static get styles(): CSSResult[] { return super.styles.concat(PFList); } // eslint-disable-next-line @typescript-eslint/no-unused-vars async apiEndpoint(page: number): Promise> { return Promise.resolve({ pagination: { count: this.objects.length, current: 1, totalPages: 1, startIndex: 1, endIndex: this.objects.length, }, results: this.objects, }); } columns(): TableColumn[] { return this.metadata(this.objects[0]).map((element) => { return new TableColumn(element.key); }); } row(item: T): TemplateResult[] { return this.metadata(item).map((element) => { return html`${element.value}`; }); } renderToolbarContainer(): TemplateResult { return html``; } renderExpanded(item: T): TemplateResult { const handler = async () => { if (!this.usedByData.has(item) && this.usedBy) { this.usedByData.set(item, await this.usedBy(item)); } return this.renderUsedBy(this.usedByData.get(item) || []); }; return html`
${this.usedBy ? until(handler(), html``) : html``}
`; } renderUsedBy(usedBy: UsedBy[]): TemplateResult { if (usedBy.length < 1) { return html`${t`Not used by any other object.`}`; } return html`
    ${usedBy.map((ub) => { let consequence = ""; switch (ub.action) { case UsedByActionEnum.Cascade: consequence = t`object will be DELETED`; break; case UsedByActionEnum.CascadeMany: consequence = t`connection will be deleted`; break; case UsedByActionEnum.SetDefault: consequence = t`reference will be reset to default value`; break; case UsedByActionEnum.SetNull: consequence = t`reference will be set to an empty value`; break; } return html`
  • ${t`${ub.name} (${consequence})`}
  • `; })}
`; } } @customElement("ak-forms-delete-bulk") export class DeleteBulkForm extends ModalButton { @property({ attribute: false }) // eslint-disable-next-line @typescript-eslint/no-explicit-any objects: any[] = []; @property() objectLabel?: string; @property() actionLabel?: string; @property() actionSubtext?: string; @property({ attribute: false }) // eslint-disable-next-line @typescript-eslint/no-explicit-any metadata: (item: any) => BulkDeleteMetadata = (item: any) => { const rec = item as Record; const meta = []; if (Object.prototype.hasOwnProperty.call(rec, "name")) { meta.push({ key: t`Name`, value: rec.name as string }); } if (Object.prototype.hasOwnProperty.call(rec, "pk")) { meta.push({ key: t`ID`, value: rec.pk as string }); } return meta; }; @property({ attribute: false }) // eslint-disable-next-line @typescript-eslint/no-explicit-any usedBy?: (item: any) => Promise; @property({ attribute: false }) // eslint-disable-next-line @typescript-eslint/no-explicit-any delete!: (item: any) => Promise; confirm(): Promise { return Promise.all( this.objects.map((item) => { return this.delete(item); }), ) .then(() => { this.onSuccess(); this.open = false; this.dispatchEvent( new CustomEvent(EVENT_REFRESH, { bubbles: true, composed: true, }), ); }) .catch((e) => { this.onError(e); throw e; }); } onSuccess(): void { showMessage({ message: t`Successfully deleted ${this.objects.length} ${this.objectLabel}`, level: MessageLevel.success, }); } onError(e: Error): void { showMessage({ message: t`Failed to delete ${this.objectLabel}: ${e.toString()}`, level: MessageLevel.error, }); } renderModalInner(): TemplateResult { return html`

${this.actionLabel ? this.actionLabel : t`Delete ${this.objectLabel}`}

${this.actionSubtext ? this.actionSubtext : t`Are you sure you want to delete ${this.objects.length} ${this.objectLabel}?`}

`; } }