import { t } from "@lingui/macro"; import { LitElement, html, customElement, property, TemplateResult, CSSResult, css } from "lit-element"; import PFLogin from "@patternfly/patternfly/components/Login/login.css"; import PFBase from "@patternfly/patternfly/patternfly-base.css"; import PFTitle from "@patternfly/patternfly/components/Title/title.css"; import PFBackgroundImage from "@patternfly/patternfly/components/BackgroundImage/background-image.css"; import PFList from "@patternfly/patternfly/components/List/list.css"; import PFButton from "@patternfly/patternfly/components/Button/button.css"; import AKGlobal from "../authentik.css"; import { unsafeHTML } from "lit-html/directives/unsafe-html"; import "./access_denied/FlowAccessDenied"; import "./stages/authenticator_static/AuthenticatorStaticStage"; import "./stages/authenticator_totp/AuthenticatorTOTPStage"; import "./stages/authenticator_validate/AuthenticatorValidateStage"; import "./stages/authenticator_webauthn/WebAuthnAuthenticatorRegisterStage"; import "./stages/autosubmit/AutosubmitStage"; import "./stages/captcha/CaptchaStage"; import "./stages/consent/ConsentStage"; import "./stages/dummy/DummyStage"; import "./stages/email/EmailStage"; import "./stages/identification/IdentificationStage"; import "./stages/password/PasswordStage"; import "./stages/prompt/PromptStage"; import "./sources/plex/PlexLoginInit"; import { ShellChallenge, RedirectChallenge } from "../api/Flows"; import { IdentificationChallenge } from "./stages/identification/IdentificationStage"; import { PasswordChallenge } from "./stages/password/PasswordStage"; import { ConsentChallenge } from "./stages/consent/ConsentStage"; import { EmailChallenge } from "./stages/email/EmailStage"; import { AutosubmitChallenge } from "./stages/autosubmit/AutosubmitStage"; import { PromptChallenge } from "./stages/prompt/PromptStage"; import { AuthenticatorTOTPChallenge } from "./stages/authenticator_totp/AuthenticatorTOTPStage"; import { AuthenticatorStaticChallenge } from "./stages/authenticator_static/AuthenticatorStaticStage"; import { AuthenticatorValidateStageChallenge } from "./stages/authenticator_validate/AuthenticatorValidateStage"; import { WebAuthnAuthenticatorRegisterChallenge } from "./stages/authenticator_webauthn/WebAuthnAuthenticatorRegisterStage"; import { CaptchaChallenge } from "./stages/captcha/CaptchaStage"; import { StageHost } from "./stages/base"; import { Challenge, ChallengeTypeEnum, Config, FlowsApi } from "authentik-api"; import { config, DEFAULT_CONFIG } from "../api/Config"; import { ifDefined } from "lit-html/directives/if-defined"; import { until } from "lit-html/directives/until"; import { AccessDeniedChallenge } from "./access_denied/FlowAccessDenied"; import { PFSize } from "../elements/Spinner"; import { TITLE_DEFAULT } from "../constants"; import { configureSentry } from "../api/Sentry"; import { PlexAuthenticationChallenge } from "./sources/plex/PlexLoginInit"; @customElement("ak-flow-executor") export class FlowExecutor extends LitElement implements StageHost { flowSlug: string; @property({attribute: false}) challenge?: Challenge; @property({type: Boolean}) loading = false; @property({ attribute: false }) config?: Config; static get styles(): CSSResult[] { return [PFBase, PFLogin, PFButton, PFTitle, PFList, PFBackgroundImage, AKGlobal].concat(css` .ak-loading { display: flex; height: 100%; width: 100%; justify-content: center; align-items: center; position: absolute; background-color: var(--pf-global--BackgroundColor--dark-transparent-100); z-index: 1; } .ak-hidden { display: none; } :host { position: relative; } .ak-exception { font-family: monospace; overflow-x: scroll; } `); } constructor() { super(); this.addEventListener("ak-flow-submit", () => { this.submit(); }); this.flowSlug = window.location.pathname.split("/")[3]; } setBackground(url: string): void { this.shadowRoot?.querySelectorAll(".pf-c-background-image").forEach((bg) => { bg.style.setProperty("--ak-flow-background", `url('${url}')`); }); } private postUpdate(): void { config().then(config => { if (this.challenge?.title) { document.title = `${this.challenge.title} - ${config.brandingTitle}`; } else { document.title = config.brandingTitle || TITLE_DEFAULT; } }); } submit(formData?: T): Promise { this.loading = true; return new FlowsApi(DEFAULT_CONFIG).flowsExecutorSolveRaw({ flowSlug: this.flowSlug, data: formData || {}, query: window.location.search.substring(1), }).then((challengeRaw) => { return challengeRaw.raw.json(); }).then((data) => { this.challenge = data; this.postUpdate(); }).catch((e: Response) => { this.errorMessage(e.statusText); }).finally(() => { this.loading = false; }); } firstUpdated(): void { configureSentry().then((config) => { this.config = config; }); this.loading = true; new FlowsApi(DEFAULT_CONFIG).flowsExecutorGetRaw({ flowSlug: this.flowSlug, query: window.location.search.substring(1), }).then((challengeRaw) => { return challengeRaw.raw.json(); }).then((challenge) => { this.challenge = challenge as Challenge; // Only set background on first update, flow won't change throughout execution if (this.challenge?.background) { this.setBackground(this.challenge.background); } this.postUpdate(); }).catch((e: Response) => { // Catch JSON or Update errors this.errorMessage(e.statusText); }).finally(() => { this.loading = false; }); } errorMessage(error: string): void { this.challenge = { type: ChallengeTypeEnum.Shell, body: ` ` }; } renderLoading(): TemplateResult { return html`
`; } renderChallenge(): TemplateResult { if (!this.challenge) { return html``; } switch (this.challenge.type) { case ChallengeTypeEnum.Redirect: console.debug("authentik/flows: redirecting to url from server", (this.challenge as RedirectChallenge).to); window.location.assign((this.challenge as RedirectChallenge).to); return html` `; case ChallengeTypeEnum.Shell: return html`${unsafeHTML((this.challenge as ShellChallenge).body)}`; case ChallengeTypeEnum.Native: switch (this.challenge.component) { case "ak-stage-access-denied": return html``; case "ak-stage-identification": return html``; case "ak-stage-password": return html``; case "ak-stage-captcha": return html``; case "ak-stage-consent": return html``; case "ak-stage-dummy": return html``; case "ak-stage-email": return html``; case "ak-stage-autosubmit": return html``; case "ak-stage-prompt": return html``; case "ak-stage-authenticator-totp": return html``; case "ak-stage-authenticator-static": return html``; case "ak-stage-authenticator-webauthn": return html``; case "ak-stage-authenticator-validate": return html``; case "ak-flow-sources-plex": return html``; default: break; } break; default: console.debug(`authentik/flows: unexpected data type ${this.challenge.type}`); break; } return html``; } renderChallengeWrapper(): TemplateResult { if (!this.challenge) { return html` `; } return html` ${this.loading ? this.renderLoading() : html``} ${this.renderChallenge()} `; } render(): TemplateResult { return html`
`; } }