import "@goauthentik/admin/providers/RelatedApplicationButton"; import "@goauthentik/admin/providers/saml/SAMLProviderForm"; import "@goauthentik/admin/rbac/ObjectPermissionsPage"; import { DEFAULT_CONFIG } from "@goauthentik/common/api/config"; import { EVENT_REFRESH } from "@goauthentik/common/constants"; import { MessageLevel } from "@goauthentik/common/messages"; import renderDescriptionList from "@goauthentik/components/DescriptionList"; import "@goauthentik/components/events/ObjectChangelog"; import { AKElement } from "@goauthentik/elements/Base"; import "@goauthentik/elements/CodeMirror"; import { CodeMirrorMode } from "@goauthentik/elements/CodeMirror"; import "@goauthentik/elements/EmptyState"; import "@goauthentik/elements/Tabs"; import "@goauthentik/elements/buttons/ActionButton"; import "@goauthentik/elements/buttons/ModalButton"; import "@goauthentik/elements/buttons/SpinnerButton"; import { showMessage } from "@goauthentik/elements/messages/MessageContainer"; import { msg } from "@lit/localize"; import { CSSResult, PropertyValues, TemplateResult, html } from "lit"; import { customElement, property, state } from "lit/decorators.js"; import { ifDefined } from "lit/directives/if-defined.js"; import PFBanner from "@patternfly/patternfly/components/Banner/banner.css"; import PFButton from "@patternfly/patternfly/components/Button/button.css"; import PFCard from "@patternfly/patternfly/components/Card/card.css"; import PFContent from "@patternfly/patternfly/components/Content/content.css"; import PFDescriptionList from "@patternfly/patternfly/components/DescriptionList/description-list.css"; import PFForm from "@patternfly/patternfly/components/Form/form.css"; import PFFormControl from "@patternfly/patternfly/components/FormControl/form-control.css"; import PFList from "@patternfly/patternfly/components/List/list.css"; import PFPage from "@patternfly/patternfly/components/Page/page.css"; import PFGrid from "@patternfly/patternfly/layouts/Grid/grid.css"; import PFBase from "@patternfly/patternfly/patternfly-base.css"; import { CertificateKeyPair, CoreApi, CoreUsersListRequest, CryptoApi, ProvidersApi, RbacPermissionsAssignedByUsersListModelEnum, SAMLMetadata, SAMLProvider, User, } from "@goauthentik/api"; interface SAMLPreviewAttribute { attributes: { Name: string; Value: string[]; }[]; nameID: string; } @customElement("ak-provider-saml-view") export class SAMLProviderViewPage extends AKElement { @property({ type: Number }) providerID?: number; @state() provider?: SAMLProvider; @state() preview?: SAMLPreviewAttribute; @state() metadata?: SAMLMetadata; @state() signer?: CertificateKeyPair; @state() verifier?: CertificateKeyPair; @state() previewUser?: User; static get styles(): CSSResult[] { return [ PFBase, PFButton, PFPage, PFGrid, PFContent, PFCard, PFList, PFDescriptionList, PFForm, PFFormControl, PFBanner, ]; } constructor() { super(); this.addEventListener(EVENT_REFRESH, () => { if (!this.provider?.pk) return; this.providerID = this.provider?.pk; }); } fetchPreview(): void { new ProvidersApi(DEFAULT_CONFIG) .providersSamlPreviewUserRetrieve({ id: this.provider?.pk || 0, forUser: this.previewUser?.pk, }) .then((preview) => { this.preview = preview.preview as SAMLPreviewAttribute; }); } fetchCertificate(kpUuid: string) { return new CryptoApi(DEFAULT_CONFIG).cryptoCertificatekeypairsRetrieve({ kpUuid }); } fetchSigningCertificate(kpUuid: string) { this.fetchCertificate(kpUuid).then((kp) => (this.signer = kp)); } fetchVerificationCertificate(kpUuid: string) { this.fetchCertificate(kpUuid).then((kp) => (this.verifier = kp)); } fetchProvider(id: number) { new ProvidersApi(DEFAULT_CONFIG).providersSamlRetrieve({ id }).then((prov) => { this.provider = prov; if (this.provider.signingKp) { this.fetchSigningCertificate(this.provider.signingKp); } if (this.provider.verificationKp) { this.fetchVerificationCertificate(this.provider.verificationKp); } }); } willUpdate(changedProperties: PropertyValues) { if (changedProperties.has("providerID") && this.providerID) { this.fetchProvider(this.providerID); } } renderRelatedObjects(): TemplateResult { const relatedObjects = []; if (this.provider?.assignedApplicationName) { relatedObjects.push( html`
${msg("Metadata")}
${msg("Download")} { if (!navigator.clipboard) { return Promise.resolve( showMessage({ level: MessageLevel.info, message: this.provider?.urlDownloadMetadata || "", }), ); } return navigator.clipboard.writeText( this.provider?.urlDownloadMetadata || "", ); }} > ${msg("Copy download URL")}
`, ); } if (this.signer) { relatedObjects.push( html`
${msg("Download signing certificate")}
`, ); } return html`
${msg("Related objects")}
${relatedObjects.length > 0 ? relatedObjects : html`-`}
`; } render(): TemplateResult { if (!this.provider) { return html``; } return html`
${this.renderTabOverview()}
${this.renderTabMetadata()}
{ this.fetchPreview(); }} > ${this.renderTabPreview()}
`; } renderTabOverview(): TemplateResult { if (!this.provider) { return html``; } return html`${ this.provider?.assignedApplicationName ? html`` : html`
${msg("Warning: Provider is not used by an Application.")}
` }
${msg("Name")}
${this.provider.name}
${msg("Assigned to application")}
${msg( "ACS URL", )}
${this.provider.acsUrl}
${msg( "Audience", )}
${this.provider.audience || "-"}
${msg( "Issuer", )}
${this.provider.issuer}
${this.renderRelatedObjects()} ${ this.provider.assignedApplicationName ? html`
${msg("SAML Configuration")}
` : html`` }
`; } renderTabMetadata(): TemplateResult { if (!this.provider) { return html``; } return html` ${this.provider.assignedApplicationName ? html`
{ new ProvidersApi(DEFAULT_CONFIG) .providersSamlMetadataRetrieve({ id: this.provider?.pk || 0, }) .then((metadata) => (this.metadata = metadata)); }} >
${msg("SAML Metadata")}
${msg("Download")} { if (!navigator.clipboard) { return Promise.resolve( showMessage({ level: MessageLevel.info, message: this.provider?.urlDownloadMetadata || "", }), ); } return navigator.clipboard.writeText( this.provider?.urlDownloadMetadata || "", ); }} > ${msg("Copy download URL")}
` : html``} `; } renderTabPreview(): TemplateResult { if (!this.preview) { return html``; } return html`
${msg("Example SAML attributes")}
${renderDescriptionList([ [ msg("Preview for user"), html` => { const args: CoreUsersListRequest = { ordering: "username", }; if (query !== undefined) { args.search = query; } const users = await new CoreApi( DEFAULT_CONFIG, ).coreUsersList(args); return users.results; }} .renderElement=${(user: User): string => { return user.username; }} .renderDescription=${(user: User): TemplateResult => { return html`${user.name}`; }} .value=${(user: User | undefined): number | undefined => { return user?.pk; }} .selected=${(user: User): boolean => { return user.pk === this.previewUser?.pk; }} ?blankable=${true} @ak-change=${(ev: CustomEvent) => { this.previewUser = ev.detail.value; this.fetchPreview(); }} > `, ], ])}
${msg("NameID attribute")}
${this.preview?.nameID}
${this.preview?.attributes.map((attr) => { return html`
${attr.Name}
    ${attr.Value.map((value) => { return html`
  • ${value}
  • `; })}
`; })}
`; } } declare global { interface HTMLElementTagNameMap { "ak-provider-saml-view": SAMLProviderViewPage; } }